From c0035e5baa2383b8c19add9059e30a96c71de2db Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Fri, 29 Apr 2022 13:43:05 -0500 Subject: [PATCH 01/30] initial zone agg --- .gitignore | 4 + .pre-commit-config.yaml | 30 +++ README.md | 23 ++ notebooks/ZoneAggDemo.ipynb | 361 +++++++++++++++++++++++++++++++ sandag_rsm/__init__.py | 3 + sandag_rsm/data_load/__init__.py | 0 sandag_rsm/data_load/triplist.py | 56 +++++ sandag_rsm/data_load/zones.py | 90 ++++++++ sandag_rsm/logging.py | 28 +++ sandag_rsm/poi.py | 49 +++++ sandag_rsm/zone_agg.py | 160 ++++++++++++++ setup.cfg | 28 +++ 12 files changed, 832 insertions(+) create mode 100644 .gitignore create mode 100644 .pre-commit-config.yaml create mode 100644 notebooks/ZoneAggDemo.ipynb create mode 100644 sandag_rsm/__init__.py create mode 100644 sandag_rsm/data_load/__init__.py create mode 100644 sandag_rsm/data_load/triplist.py create mode 100644 sandag_rsm/data_load/zones.py create mode 100644 sandag_rsm/logging.py create mode 100644 sandag_rsm/poi.py create mode 100644 sandag_rsm/zone_agg.py create mode 100644 setup.cfg diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..a28b28c --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +sandag_rsm/__pycache__ +**/__pycache__ +.idea +.ipynb_checkpoints diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 0000000..35cc926 --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,30 @@ +repos: + +- repo: https://github.com/pre-commit/pre-commit-hooks + rev: v4.1.0 + hooks: + - id: check-yaml + - id: end-of-file-fixer + exclude: .*\.ipynb + - id: trailing-whitespace + +- repo: https://github.com/kynan/nbstripout + rev: 0.5.0 + hooks: + - id: nbstripout + +- repo: https://github.com/pycqa/isort + rev: 5.10.1 + hooks: + - id: isort + args: ["--profile", "black", "--filter-files"] + +- repo: https://github.com/psf/black + rev: 21.12b0 + hooks: + - id: black + +- repo: https://github.com/PyCQA/flake8 + rev: 4.0.1 + hooks: + - id: flake8 diff --git a/README.md b/README.md index 48f8545..aa8752a 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,25 @@ # RSM Rapid Strategic Model for the San Diego Association of Governments + + + +## Code Formatting + +This repo use several tools to ensure a consistent code format throughout the project: + +- [Black](https://black.readthedocs.io/en/stable/) for standardized code formatting, +- [Flake8](http://flake8.pycqa.org/en/latest/) for general code quality, +- [isort](https://github.com/timothycrosley/isort) for standardized order in imports, and +- [nbstripout](https://github.com/kynan/nbstripout) to ensure notebooks are committed + to the GitHub repository without bulky outputs included. + +We highly recommend that you setup [pre-commit hooks](https://pre-commit.com/) +to automatically run all the above tools every time you make a git commit. This +can be done by running: + +```shell +pre-commit install +``` + +from the root of the repository. You can skip the pre-commit checks +with `git commit --no-verify`. diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb new file mode 100644 index 0000000..8714e15 --- /dev/null +++ b/notebooks/ZoneAggDemo.ipynb @@ -0,0 +1,361 @@ +{ + "cells": [ + { + "cell_type": "code", + "execution_count": null, + "id": "4fc00298", + "metadata": {}, + "outputs": [], + "source": [ + "from sandag_rsm.data_load.zones import load_mgra_data\n", + "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra\n", + "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", + "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b67524d3", + "metadata": {}, + "outputs": [], + "source": [ + "data_dir = \"~/Library/CloudStorage/OneDrive-SharedLibraries-CambridgeSystematics/PROJ SANDAG RSM - General/Data\"" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "c88b96b5", + "metadata": {}, + "outputs": [], + "source": [ + "mgra = load_mgra_data(data_dir=data_dir, simplify_tolerance=10, topo=True)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ae5d7f1c", + "metadata": {}, + "outputs": [], + "source": [ + "mgra['taz20'] = mgra.taz % 20" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "f3593b1c", + "metadata": {}, + "outputs": [], + "source": [ + "trips = load_trip_list(\"FromSANDAG-Files/indivTripData_3.csv.gz\", data_dir=data_dir)\n", + "trip_mode_shares = trip_mode_shares_by_mgra(trips, mgras=mgra.MGRA)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "e64f7a0b", + "metadata": {}, + "outputs": [], + "source": [ + "mgra = mgra.join(trip_mode_shares.add_prefix(\"modeshare_\"), on='MGRA')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "65c404fa", + "metadata": {}, + "outputs": [], + "source": [ + "poi = poi_taz_mgra(mgra)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "569cfadc", + "metadata": {}, + "outputs": [], + "source": [ + "poi" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b3665a66", + "metadata": {}, + "outputs": [], + "source": [ + "import openmatrix" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "9b0e8910", + "metadata": {}, + "outputs": [], + "source": [ + "s = openmatrix.open_file(\n", + " os.path.expanduser(\n", + " os.path.join(data_dir, \"FromSANDAG-Files/traffic_skims_AM.omx\"),\n", + " ),\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "6cd4cee2", + "metadata": {}, + "outputs": [], + "source": [ + "mgra = attach_poi_taz_skims(\n", + " mgra,\n", + " s,\n", + " names='AM_SOV_TR_M_TIME',\n", + " poi=poi,\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "9d7ce6dc", + "metadata": {}, + "outputs": [], + "source": [ + "mgra.info(1)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "04f9b597", + "metadata": {}, + "outputs": [], + "source": [ + "d1 = mgra.query(\"district27 == 1\")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a93bd34a", + "metadata": {}, + "outputs": [], + "source": [ + "viewer(d1, color='popden', marker_line_width=0)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "f66915c5", + "metadata": {}, + "outputs": [], + "source": [ + "viewer(d1, color='modeshare_WT', marker_line_width=0)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1792481d", + "metadata": {}, + "outputs": [], + "source": [ + "mgra.info(1)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "61b2c6c7", + "metadata": {}, + "outputs": [], + "source": [ + "import itertools\n", + "cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", + "cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", + "cluster_factors" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "3234d883", + "metadata": {}, + "outputs": [], + "source": [ + "kmeans = aggregate_zones(\n", + " d1, \n", + " cluster_factors=cluster_factors, \n", + " n_zones=300,\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7e7b56ea", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=kmeans, colors=d1, color_col='popden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4f0f73c4", + "metadata": {}, + "outputs": [], + "source": [ + "kmeans" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "14e60056", + "metadata": {}, + "outputs": [], + "source": [ + "agglom1 = aggregate_zones(d1, cluster_factors=cluster_factors, n_zones=300, method='agglom')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "abe08363", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom1, colors=d1, color_col='popden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "65d89c03", + "metadata": {}, + "outputs": [], + "source": [ + "agglom2 = aggregate_zones(\n", + " d1, cluster_factors=cluster_factors, \n", + " n_zones=300, method='agglom_adj', use_xy=False,\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b3d23e2d", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom2, colors=d1, color_col='popden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "cd43c71f", + "metadata": {}, + "outputs": [], + "source": [ + "agglom3 = aggregate_zones(d1, cluster_factors=cluster_factors, \n", + " cluster_factors_onehot={'taz':5},\n", + " n_zones=100, method='agglom_adj', use_xy=False)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ef76f562", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom3, colors=d1, color_col='popden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "346536be", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom3, colors=d1, color_col='modeshare_NM')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "c7ca8388", + "metadata": {}, + "outputs": [], + "source": [ + "d1['taz20'] = d1.taz % 20" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "00cfa00b", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom3, colors=d1, color_col='taz20')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "aa31c88d", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.9.10" + }, + "toc": { + "base_numbering": 1, + "nav_menu": {}, + "number_sections": false, + "sideBar": true, + "skip_h1_title": false, + "title_cell": "Table of Contents", + "title_sidebar": "Contents", + "toc_cell": false, + "toc_position": {}, + "toc_section_display": true, + "toc_window_display": false + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/sandag_rsm/__init__.py b/sandag_rsm/__init__.py new file mode 100644 index 0000000..96299dd --- /dev/null +++ b/sandag_rsm/__init__.py @@ -0,0 +1,3 @@ +from .logging import logging_start + +logging_start(20) diff --git a/sandag_rsm/data_load/__init__.py b/sandag_rsm/data_load/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/sandag_rsm/data_load/triplist.py b/sandag_rsm/data_load/triplist.py new file mode 100644 index 0000000..b029546 --- /dev/null +++ b/sandag_rsm/data_load/triplist.py @@ -0,0 +1,56 @@ +import os + +import pandas as pd + + +def load_trip_list( + trips_filename="indivTripData_3.csv.gz", + data_dir=None, +): + if data_dir is not None: + data_dir = os.path.expanduser(data_dir) + cwd = os.getcwd() + os.chdir(data_dir) + else: + cwd = None + + try: + trips = pd.read_csv(trips_filename) + return trips + + finally: + # change back to original cwd + os.chdir(cwd) + + +def trip_mode_shares_by_mgra( + trips, + background_per_mgra=50, + mgras=None, +): + trip_modes = { + 1: "Au", # Drive Alone + 2: "Au", # Shared Ride 2 + 3: "Au", # Shared Ride 3 + 4: "NM", # Walk + 5: "NM", # Bike + 6: "WT", # Walk to Transit + 7: "DT", # Park and Ride to Transit + 8: "DT", # Kiss and Ride to Transit + 9: "Au", # TNC to Transit + 10: "Au", # Taxi + 11: "Au", # TNC Single + 12: "Au", # TNC Shared + 13: "Au", # School Bus + } + trip_mode_cat = trips["trip_mode"].apply(trip_modes.get) + tmo = trips.groupby([trips.orig_mgra, trip_mode_cat]).size().unstack().fillna(0) + tmd = trips.groupby([trips.dest_mgra, trip_mode_cat]).size().unstack().fillna(0) + tm = tmo + tmd + tm_total = tm.sum() + background = background_per_mgra * tm_total / tm_total.sum() + if mgras is not None: + tm = tm.reindex(mgras).fillna(0) + tm = tm + background + tripmodeshare = tm.div(tm.sum(axis=1), axis=0) + return tripmodeshare diff --git a/sandag_rsm/data_load/zones.py b/sandag_rsm/data_load/zones.py new file mode 100644 index 0000000..1f68aef --- /dev/null +++ b/sandag_rsm/data_load/zones.py @@ -0,0 +1,90 @@ +import logging +import os +import warnings + +import geopandas as gpd +import pandas as pd +import pyproj +from shapely.ops import orient # version >=1.7a2 + +logger = logging.getLogger(__name__) + + +def geometry_cleanup(gdf): + gdf.geometry = gdf.geometry.apply(orient, args=(-1,)) + gdf.geometry = gdf.geometry.buffer(0) + return gdf + + +def simplify_shapefile( + shapefilename="MGRASHAPE.zip", + simplify_tolerance=1, + prequantize=False, + layername="MGRA", + topo=True, +): + gpkg_filename = ( + os.path.splitext(shapefilename)[0] + f"_simplified_{simplify_tolerance}.gpkg" + ) + if os.path.exists(gpkg_filename): + gdf = gpd.read_file(gpkg_filename) + return geometry_cleanup(gdf) + gdf = gpd.read_file(shapefilename) + if topo: + try: + import topojson as tp + except ImportError: + warnings.warn("topojson is not installed") + gdf.geometry = gdf.geometry.simplify(simplify_tolerance) + return geometry_cleanup(gdf) + else: + logger.info("converting to epsg:3857") + gdf = gdf.to_crs(pyproj.CRS.from_epsg(3857)) + logger.info("creating topology") + topo = tp.Topology(gdf, prequantize=prequantize) + logger.info("simplifying topology") + topo = topo.toposimplify(simplify_tolerance) + logger.info("converting to gdf") + gdf = topo.to_gdf() + gdf.crs = pyproj.CRS.from_epsg(3857) + logger.info("checking orientation") + gdf.geometry = gdf.geometry.apply(orient, args=(-1,)) + logger.info("completed") + gdf.to_file(gpkg_filename, layer=layername, driver="GPKG") + return geometry_cleanup(gdf) + else: + if simplify_tolerance is not None: + gdf.geometry = gdf.geometry.simplify(simplify_tolerance) + return geometry_cleanup(gdf) + + +def load_mgra_data( + shapefilename="MGRASHAPE.zip", + supplemental_features="mgra13_based_input2016.csv", + data_dir=None, + simplify_tolerance=1, + prequantize=False, + topo=True, +): + if data_dir is not None: + data_dir = os.path.expanduser(data_dir) + cwd = os.getcwd() + os.chdir(data_dir) + else: + cwd = None + + try: + gdf = simplify_shapefile( + shapefilename=shapefilename, + simplify_tolerance=simplify_tolerance, + layername="MGRA", + prequantize=prequantize, + topo=topo, + ) + sdf = pd.read_csv(supplemental_features) + mgra = gdf.merge(sdf, left_on="MGRA", right_on="mgra") + return mgra + + finally: + # change back to original cwd + os.chdir(cwd) diff --git a/sandag_rsm/logging.py b/sandag_rsm/logging.py new file mode 100644 index 0000000..5af270d --- /dev/null +++ b/sandag_rsm/logging.py @@ -0,0 +1,28 @@ +import logging +import sys + + +class ElapsedTimeFormatter(logging.Formatter): + def format(self, record): + duration_milliseconds = record.relativeCreated + hours, rem = divmod(duration_milliseconds / 1000, 3600) + minutes, seconds = divmod(rem, 60) + if hours: + record.elapsedTime = "{:0>2}:{:0>2}:{:05.2f}".format( + int(hours), int(minutes), seconds + ) + else: + record.elapsedTime = "{:0>2}:{:05.2f}".format(int(minutes), seconds) + return super(ElapsedTimeFormatter, self).format(record) + + +def logging_start(level=None): + formatter = ElapsedTimeFormatter( + fmt="[{elapsedTime}] {levelname:s}: {message:s}", + style="{", + ) + handler = logging.StreamHandler(sys.stdout) + handler.setFormatter(formatter) + logging.getLogger().addHandler(handler) + if level is not None: + logging.getLogger().setLevel(level) diff --git a/sandag_rsm/poi.py b/sandag_rsm/poi.py new file mode 100644 index 0000000..ff567fa --- /dev/null +++ b/sandag_rsm/poi.py @@ -0,0 +1,49 @@ +import pandas as pd +import pyproj +import shapely.geometry.point + +# lat-lon of certain points +points_of_interest = dict( + san_diego_city_hall=(32.71691, -117.16282), + outside_pendleton_gate=(33.20722, -117.38973), + escondido_city_hall=(33.122711, -117.08309), + viejas_casino=(32.842097, -116.705582), + san_ysidro_trolley=(32.544536, -117.02963), +) + + +def poi_taz_mgra(gdf): + zones = {} + mgra4326 = gdf.to_crs(pyproj.CRS.from_epsg(4326)) + for name, latlon in points_of_interest.items(): + pt = shapely.geometry.point.Point(*reversed(latlon)) + y = mgra4326.contains(pt) + if y.sum() == 1: + target = mgra4326[y].iloc[0] + zones[name] = {"taz": target.taz, "mgra": target.mgra} + return zones + + +def attach_poi_taz_skims( + gdf, + skims_omx, + names, + poi=None, +): + if poi is None: + poi = poi_taz_mgra(gdf) + if isinstance(names, str): + names = [names] + zone_nums = skims_omx.root.lookup.zone_number + cols = {} + for k in poi: + ktaz = poi[k]["taz"] + for name in names: + cols[f"{k}_{name}"] = pd.Series( + skims_omx.root.data[name][ktaz - 1], + index=zone_nums[:], + ) + add_to_gdf = {} + for c in cols: + add_to_gdf[c] = gdf["taz"].map(cols[c]) + return gdf.assign(**add_to_gdf) diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py new file mode 100644 index 0000000..80c0ace --- /dev/null +++ b/sandag_rsm/zone_agg.py @@ -0,0 +1,160 @@ +import logging + +import networkx as nx +import numpy as np +import pyproj +from sklearn.cluster import AgglomerativeClustering, KMeans +from sklearn.preprocessing import OneHotEncoder + +logger = logging.getLogger(__name__) + + +def aggregate_zones( + mgra_gdf, + method="kmeans", + n_zones=2000, + random_state=0, + cluster_factors=None, + cluster_factors_onehot=None, + use_xy=True, + explicit_agg=(), # TODO + agg_instruction=None, +): + """ + Aggregate zones. + + Parameters + ---------- + mgra_gdf : GeoDataFrame + Geometry and attibutes of MGRAs + method : {'kmeans', 'agglom', 'agglom_adj'} + n_zones : int + random_state : RandomState or int + cluster_factors : dict + cluster_factors_onehot : dict + use_xy : bool + Use X and Y coordinates as a cluster factor + explicit_agg : list[int or list] + A list containing integers (individual MGRAs that should not be aggregated) + or lists of integers (groups of MGRAs that should be aggregated exactly as + given, with no less and no more) + agg_instruction : dict + Dictionary passed to pandas `agg` that says how to aggregate data columns. + + Returns + ------- + GeoDataFrame + """ + if cluster_factors is None: + cluster_factors = {} + mgra_gdf = mgra_gdf.copy() + if use_xy: + geometry = mgra_gdf.centroid + X = list(geometry.apply(lambda p: p.x)) + Y = list(geometry.apply(lambda p: p.y)) + factors = [X, Y] + else: + factors = [] + for cf, cf_wgt in cluster_factors.items(): + factors.append(cf_wgt * mgra_gdf[cf].values.astype(np.float32)) + if cluster_factors_onehot: + for cf, cf_wgt in cluster_factors_onehot.items(): + factors.append(cf_wgt * OneHotEncoder().fit_transform(mgra_gdf[[cf]])) + from scipy.sparse import hstack + + factors2d = [] + for j in factors: + if j.ndim < 2: + factors2d.append(np.expand_dims(j, -1)) + else: + factors2d.append(j) + data = hstack(factors2d).toarray() + else: + data = np.array(factors).T + + if method == "kmeans": + kmeans = KMeans(n_clusters=n_zones, random_state=random_state) + kmeans.fit(data) + cluster_id = kmeans.labels_ + elif method == "agglom": + agglom = AgglomerativeClustering( + n_clusters=n_zones, affinity="euclidean", linkage="ward" + ) + agglom.fit_predict(data) + cluster_id = agglom.labels_ + elif method == "agglom_adj": + from libpysal.weights import Rook + + w_rook = Rook.from_dataframe(mgra_gdf) + adj_mat = nx.adjacency_matrix(w_rook.to_networkx()) + agglom = AgglomerativeClustering( + n_clusters=n_zones, + affinity="euclidean", + linkage="ward", + connectivity=adj_mat, + ) + agglom.fit_predict(data) + cluster_id = agglom.labels_ + else: + raise NotImplementedError(method) + mgra_gdf["cluster_id"] = cluster_id + if agg_instruction is None: + # TODO: fill out the aggregation data system + # Define a lambda function to compute the weighted mean: + wgt_avg = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].area) + agg_instruction = { + "pop": "sum", + "popden": wgt_avg, + } + dissolved = mgra_gdf[["cluster_id", "geometry"]].dissolve(by="cluster_id") + dissolved = dissolved.join(mgra_gdf.groupby("cluster_id").agg(agg_instruction)) + return dissolved + + +def viewer( + gdf, + *, + simplify_tolerance=None, + color=None, + transparent=False, + **kwargs, +): + import plotly.express as px + + gdf = gdf.copy() + if simplify_tolerance is not None: + gdf.geometry = gdf.geometry.simplify(tolerance=simplify_tolerance) + gdf = gdf.to_crs(pyproj.CRS.from_epsg(4326)) + kwargs1 = {} + if color is not None: + kwargs1["color"] = color + fig = px.choropleth( + gdf, + geojson=gdf.geometry, + locations=gdf.index, + **kwargs1, + ) + fig.update_shapes() + fig.update_geos( + visible=False, + fitbounds="locations", + ) + fig.update_layout(height=300, margin={"r": 0, "t": 0, "l": 0, "b": 0}) + if kwargs: + fig.update_traces(**kwargs) + if transparent: + fig.update_traces( + colorscale=((0.0, "rgba(0, 0, 0, 0.0)"), (1.0, "rgba(0, 0, 0, 0.0)")) + ) + return fig + + +def viewer2( + edges, + colors, + color_col, +): + coloring_map = viewer(colors, color=color_col, marker_line_width=0) + edge_map = viewer(edges, transparent=True, marker_line_color="white") + coloring_map.add_trace(edge_map.data[0]) + return coloring_map diff --git a/setup.cfg b/setup.cfg new file mode 100644 index 0000000..dcad608 --- /dev/null +++ b/setup.cfg @@ -0,0 +1,28 @@ +[metadata] +name = sandag_rsm +url = https://github.com/SANDAG/RSM + +[options] +packages = find: +zip_safe = False +include_package_data = True +python_requires = >=3.7 +install_requires = + numpy >= 1.19 + pandas + scikit-learn + geopandas + libpysal + pyproj + plotly + networkx + + +[flake8] +exclude = + .git, + __pycache__, + docs/_build, + sandag_rsm/__init__.py +max-line-length = 160 +extend-ignore = E203, E731 From e0c3e9bdaf91e60c5af40426b7c831f6ee64a1f2 Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Mon, 2 May 2022 14:43:12 -0500 Subject: [PATCH 02/30] Adding aggregation fields for zone aggregation --- sandag_rsm/zone_agg.py | 132 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 130 insertions(+), 2 deletions(-) diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index 80c0ace..8cde903 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -3,6 +3,7 @@ import networkx as nx import numpy as np import pyproj +from statistics import mode from sklearn.cluster import AgglomerativeClustering, KMeans from sklearn.preprocessing import OneHotEncoder @@ -101,13 +102,140 @@ def aggregate_zones( if agg_instruction is None: # TODO: fill out the aggregation data system # Define a lambda function to compute the weighted mean: - wgt_avg = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].area) + + wgt_avg_hh = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].hh) if mgra_gdf.loc[x.index].hh.sum()> 0 else 0 + wgt_avg_hpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].hstallssam) if mgra_gdf.loc[x.index].hstallssam.sum()>0 else 0 + wgt_avg_dpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].dstallssam) if mgra_gdf.loc[x.index].dstallssam.sum()>0 else 0 + wgt_avg_mpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].mstallssam) if mgra_gdf.loc[x.index].mstallssam.sum()>0 else 0 + wgt_avg_mtc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 + wgt_avg_mat = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 + wgt_avg_dud = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hh']) if mgra_gdf.loc[x.index, 'hh'].sum()>0 else 0 + wgt_avg_empden = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'emp_total']) if mgra_gdf.loc[x.index, 'emp_total'].sum()>0 else 0 + wgt_avg_rtempden = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'emp_total']) if mgra_gdf.loc[x.index, 'emp_retail'].sum()>0 else 0 + wgt_avg_peden = lambda x: np.average(x, weights=(mgra_gdf.loc[x.index, 'emp_total'] + mgra_gdf.loc[x.index, 'pop'])) if (mgra_gdf.loc[x.index, 'emp_total'].sum()+mgra_gdf.loc[x.index, 'pop'].sum())>0 else 0 + get_mode = lambda x: mode(x) agg_instruction = { + "hs": "sum", + "hs_sf": "sum", + "hs_mf": "sum", + "hs_mh": "sum", + "hh": "sum", + "hh_sf": "sum", + "hh_mf": "sum", + "hh_mh": "sum", + "gq_civ": "sum", + "gq_mil": "sum", + "i1": "sum", + "i2": "sum", + "i3": "sum", + "i4": "sum", + "i5": "sum", + "i6": "sum", + "i7": "sum", + "i8": "sum", + "i9": "sum", + "i10": "sum", + "hhs": wgt_avg_hh, "pop": "sum", - "popden": wgt_avg, + "hhp": "sum", + "emp_ag": "sum", + "emp_const_non_bldg_prod": "sum", + "emp_const_non_bldg_office": "sum", + "emp_utilities_prod": "sum", + "emp_utilities_office": "sum", + "emp_const_bldg_prod": "sum", + "emp_const_bldg_office": "sum", + "emp_mfg_prod": "sum", + "emp_mfg_office": "sum", + "emp_whsle_whs": "sum", + "emp_trans": "sum", + "emp_retail": "sum", + "emp_prof_bus_svcs": "sum", + "emp_prof_bus_svcs_bldg_maint": "sum", + "emp_pvt_ed_k12": "sum", + "emp_pvt_ed_post_k12_oth": "sum", + "emp_health": "sum", + "emp_personal_svcs_office": "sum", + "emp_amusement": "sum", + "emp_hotel": "sum", + "emp_restaurant_bar": "sum", + "emp_personal_svcs_retail": "sum", + "emp_religious": "sum", + "emp_pvt_hh": "sum", + "emp_state_local_gov_ent": "sum", + "emp_fed_non_mil": "sum", + "emp_fed_mil": "sum", + "emp_state_local_gov_blue": "sum", + "emp_state_local_gov_white": "sum", + "emp_public_ed": "sum", + "emp_own_occ_dwell_mgmt": "sum", + "emp_fed_gov_accts": "sum", + "emp_st_lcl_gov_accts": "sum", + "emp_cap_accts": "sum", + "emp_total": "sum", + "enrollgradekto8": "sum", + "enrollgrade9to12": "sum", + "collegeenroll": "sum", + "othercollegeenroll": "sum", + "adultschenrl": "sum", + "ech_dist": get_mode, + "hch_dist": get_mode, + "parkarea": "max", + "hstallsoth": "sum", + "hstallssam": "sum", + "hparkcost": wgt_avg_hpc, + "numfreehrs": wgt_avg_hpc, + "dstallsoth": "sum", + "dstallssam": "sum", + "dparkcost": wgt_avg_dpc, + "mstallsoth": "sum", + "mstallssam": "sum", + "mparkcost": wgt_avg_mpc, + "parkactive": "sum", + "openspaceparkpreserve": "sum", + "beachactive": "sum", + "budgetroom": "sum", + "economyroom": "sum", + "luxuryroom": "sum", + "midpriceroom": "sum", + "upscaleroom": "sum", + "hotelroomtotal": "sum", + #"luz_id": "sum", + "truckregiontype": "sum", + #"district27": "sum", + "milestocoast": wgt_avg_mtc, + #"acres": "sum", + #"effective_acres": "sum", + #"land_acres": "sum", + "MicroAccessTime": wgt_avg_mat, + "remoteAVParking": "max", + "refueling_stations": "sum", + "totint": "sum", + "duden": wgt_avg_dud, + "empden": wgt_avg_empden, + #"popden": "sum", + "retempden": wgt_avg_rtempden, + #"totintbin": "sum", #bins in original data 0, 80, 130 + #"empdenbin": "sum", #bins in original data 0, 10, 30 + #"dudenbin": "sum", #bins in original data 0, 5, 10 + "PopEmpDenPerMi": wgt_avg_peden } dissolved = mgra_gdf[["cluster_id", "geometry"]].dissolve(by="cluster_id") dissolved = dissolved.join(mgra_gdf.groupby("cluster_id").agg(agg_instruction)) + + #adding bins + dissolved['totintbin'] = 1 + dissolved.loc[(dissolved['totintbin'] >= 80) & (dissolved['totintbin'] < 130), 'totintbin'] = 2 + dissolved.loc[(dissolved['totintbin'] >= 130), 'totintbin'] = 3 + + dissolved['empdenbin'] = 1 + dissolved.loc[(dissolved['empdenbin'] >= 10) & (dissolved['empdenbin'] < 30), 'empdenbin'] = 2 + dissolved.loc[(dissolved['empdenbin'] >= 30), 'empdenbin'] = 3 + + dissolved['dudenbin'] = 1 + dissolved.loc[(dissolved['dudenbin'] >=5 ) & (dissolved['dudenbin'] < 10), 'dudenbin'] = 2 + dissolved.loc[(dissolved['dudenbin'] >= 10), 'dudebin'] = 3 + return dissolved From 6681e1ecaaab74a3cfaa467c4740bfbe6c489e8a Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Mon, 2 May 2022 14:47:36 -0500 Subject: [PATCH 03/30] Updating column references --- sandag_rsm/zone_agg.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index 8cde903..6bf6910 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -103,11 +103,11 @@ def aggregate_zones( # TODO: fill out the aggregation data system # Define a lambda function to compute the weighted mean: - wgt_avg_hh = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].hh) if mgra_gdf.loc[x.index].hh.sum()> 0 else 0 - wgt_avg_hpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].hstallssam) if mgra_gdf.loc[x.index].hstallssam.sum()>0 else 0 - wgt_avg_dpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].dstallssam) if mgra_gdf.loc[x.index].dstallssam.sum()>0 else 0 - wgt_avg_mpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index].mstallssam) if mgra_gdf.loc[x.index].mstallssam.sum()>0 else 0 - wgt_avg_mtc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 + wgt_avg_hh = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hh']) if mgra_gdf.loc[x.index, 'hh'].sum()> 0 else 0 + wgt_avg_hpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hstallssam']) if mgra_gdf.loc[x.index,'hstallssam'].sum()>0 else 0 + wgt_avg_dpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index,'dstallssam']) if mgra_gdf.loc[x.index,'dstallssam'].sum()>0 else 0 + wgt_avg_mpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'mstallssam']) if mgra_gdf.loc[x.index,'mstallssam'].sum()>0 else 0 + wgt_avg_mtc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 wgt_avg_mat = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 wgt_avg_dud = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hh']) if mgra_gdf.loc[x.index, 'hh'].sum()>0 else 0 wgt_avg_empden = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'emp_total']) if mgra_gdf.loc[x.index, 'emp_total'].sum()>0 else 0 From 44095f4e9781f6c5d3894f0f03bbc7229419b533 Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Mon, 2 May 2022 15:19:41 -0500 Subject: [PATCH 04/30] implementing explicit aggregation --- sandag_rsm/zone_agg.py | 28 ++++++++++++++++++++++++---- 1 file changed, 24 insertions(+), 4 deletions(-) diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index 6bf6910..b52abd4 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -2,6 +2,7 @@ import networkx as nx import numpy as np +import pandas as pd import pyproj from statistics import mode from sklearn.cluster import AgglomerativeClustering, KMeans @@ -46,9 +47,23 @@ def aggregate_zones( ------- GeoDataFrame """ + + + if cluster_factors is None: cluster_factors = {} - mgra_gdf = mgra_gdf.copy() + + + if not explicit_agg: + mgra_gdf = mgra_gdf.copy() + n_zones_upd = n_zones + + else: + mgra_gdf = mgra_gdf.loc[~mgra_gdf['mgra'].isin(explicit_agg)] + mgra_gdf_explicit = mgra_gdf.loc[mgra_gdf['mgra'].isin(explicit_agg)] + n_zones_upd = n_zones - len(mgra_gdf_explicit) + + if use_xy: geometry = mgra_gdf.centroid X = list(geometry.apply(lambda p: p.x)) @@ -74,12 +89,12 @@ def aggregate_zones( data = np.array(factors).T if method == "kmeans": - kmeans = KMeans(n_clusters=n_zones, random_state=random_state) + kmeans = KMeans(n_clusters=n_zones_upd, random_state=random_state) kmeans.fit(data) cluster_id = kmeans.labels_ elif method == "agglom": agglom = AgglomerativeClustering( - n_clusters=n_zones, affinity="euclidean", linkage="ward" + n_clusters=n_zones_upd, affinity="euclidean", linkage="ward" ) agglom.fit_predict(data) cluster_id = agglom.labels_ @@ -89,7 +104,7 @@ def aggregate_zones( w_rook = Rook.from_dataframe(mgra_gdf) adj_mat = nx.adjacency_matrix(w_rook.to_networkx()) agglom = AgglomerativeClustering( - n_clusters=n_zones, + n_clusters=n_zones_upd, affinity="euclidean", linkage="ward", connectivity=adj_mat, @@ -236,6 +251,11 @@ def aggregate_zones( dissolved.loc[(dissolved['dudenbin'] >=5 ) & (dissolved['dudenbin'] < 10), 'dudenbin'] = 2 dissolved.loc[(dissolved['dudenbin'] >= 10), 'dudebin'] = 3 + + dissolved["cluster_id"] = list(range(n_zones_upd, n_zones, 1)) + dissolved = pd.concat([dissolved, mgra_gdf_explicit], ignore_index = False) + dissolved = dissolved.reset_index(drop=True) + return dissolved From ab6f4973098d575747bcc1bb93b80d0dc066570d Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Tue, 3 May 2022 16:41:51 -0500 Subject: [PATCH 05/30] zone agg updates --- .gitignore | 2 + README.md | 7 + notebooks/ZoneAggDemo.ipynb | 162 ++++++++++++++------- pyproject.toml | 27 ++++ sandag_rsm/data_load/skims.py | 27 ++++ sandag_rsm/poi.py | 54 +++++-- sandag_rsm/zone_agg.py | 255 +++++++++++++++++++++++++--------- 7 files changed, 409 insertions(+), 125 deletions(-) create mode 100644 pyproject.toml create mode 100644 sandag_rsm/data_load/skims.py diff --git a/.gitignore b/.gitignore index a28b28c..15e9211 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,5 @@ sandag_rsm/__pycache__ **/__pycache__ .idea .ipynb_checkpoints +sandag_rsm.egg-info +_version.py diff --git a/README.md b/README.md index aa8752a..6fc1aa7 100644 --- a/README.md +++ b/README.md @@ -1,7 +1,14 @@ # RSM Rapid Strategic Model for the San Diego Association of Governments +## Installing +To install, activate the python or conda environment you want to use, +the cd into the repository directory and run: + +```shell +python -m pip install -e . +``` ## Code Formatting diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index 8714e15..d7a73ca 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -10,7 +10,7 @@ "from sandag_rsm.data_load.zones import load_mgra_data\n", "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra\n", "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", - "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2" + "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts" ] }, { @@ -54,6 +54,16 @@ "trip_mode_shares = trip_mode_shares_by_mgra(trips, mgras=mgra.MGRA)" ] }, + { + "cell_type": "code", + "execution_count": null, + "id": "37ff7b19", + "metadata": {}, + "outputs": [], + "source": [ + "trip_mode_shares" + ] + }, { "cell_type": "code", "execution_count": null, @@ -87,117 +97,131 @@ { "cell_type": "code", "execution_count": null, - "id": "b3665a66", + "id": "1499fc7f", "metadata": {}, "outputs": [], "source": [ - "import openmatrix" + "cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}" ] }, { "cell_type": "code", "execution_count": null, - "id": "9b0e8910", + "id": "6cd4cee2", "metadata": {}, "outputs": [], "source": [ - "s = openmatrix.open_file(\n", - " os.path.expanduser(\n", - " os.path.join(data_dir, \"FromSANDAG-Files/traffic_skims_AM.omx\"),\n", - " ),\n", + "mgra, cluster_factors = attach_poi_taz_skims(\n", + " mgra,\n", + " \"FromSANDAG-Files/traffic_skims_AM.omx\",\n", + " names='AM_SOV_TR_M_TIME',\n", + " poi=poi,\n", + " data_dir=data_dir,\n", + " cluster_factors=cluster_factors,\n", ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "6cd4cee2", + "id": "04f9b597", "metadata": {}, "outputs": [], "source": [ - "mgra = attach_poi_taz_skims(\n", - " mgra,\n", - " s,\n", - " names='AM_SOV_TR_M_TIME',\n", - " poi=poi,\n", - ")" + "d1 = mgra.query(\"district27 == 1\")" ] }, { "cell_type": "code", "execution_count": null, - "id": "9d7ce6dc", + "id": "a93bd34a", "metadata": {}, "outputs": [], "source": [ - "mgra.info(1)" + "viewer(d1, color='popden', marker_line_width=1)" ] }, { "cell_type": "code", "execution_count": null, - "id": "04f9b597", + "id": "f66915c5", "metadata": {}, "outputs": [], "source": [ - "d1 = mgra.query(\"district27 == 1\")" + "# viewer(d1, color='modeshare_WT', marker_line_width=0)" ] }, { "cell_type": "code", "execution_count": null, - "id": "a93bd34a", + "id": "61b2c6c7", "metadata": {}, "outputs": [], "source": [ - "viewer(d1, color='popden', marker_line_width=0)" + "# import itertools\n", + "# cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", + "# cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", + "cluster_factors" ] }, { "cell_type": "code", "execution_count": null, - "id": "f66915c5", + "id": "3234d883", "metadata": {}, "outputs": [], "source": [ - "viewer(d1, color='modeshare_WT', marker_line_width=0)" + "kmeans = aggregate_zones(\n", + " d1, \n", + " cluster_factors=cluster_factors, \n", + " n_zones=300,\n", + " explicit_agg=[\n", + " 14571, 14574, 14578, \n", + " [15300, 15303, 15304, 15305],\n", + " ],\n", + " use_xy=1e-6,\n", + ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "1792481d", + "id": "1b1fbdee", "metadata": {}, "outputs": [], "source": [ - "mgra.info(1)" + "from sandag_rsm.zone_agg import aggregate_zones_within_districts\n", + "\n", + "kmeans = aggregate_zones_within_districts(\n", + " mgra, \n", + " cluster_factors=cluster_factors, \n", + " n_zones=2000,\n", + " use_xy=1e-6,\n", + " explicit_agg=[\n", + " 14571, 14574, 14578, \n", + " [15300, 15303, 15304, 15305],\n", + " ],\n", + ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "61b2c6c7", + "id": "02023d6a", "metadata": {}, "outputs": [], "source": [ - "import itertools\n", - "cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", - "cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", - "cluster_factors" + "(kmeans)" ] }, { "cell_type": "code", "execution_count": null, - "id": "3234d883", + "id": "20027832", "metadata": {}, "outputs": [], "source": [ - "kmeans = aggregate_zones(\n", - " d1, \n", - " cluster_factors=cluster_factors, \n", - " n_zones=300,\n", - ")" + "kmeans = kmeans.reset_index(drop=True)" ] }, { @@ -207,18 +231,16 @@ "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=kmeans, colors=d1, color_col='popden')" + "viewer2(edges=kmeans, colors=kmeans, color_col='empden')" ] }, { "cell_type": "code", "execution_count": null, - "id": "4f0f73c4", + "id": "2f68ec2b", "metadata": {}, "outputs": [], - "source": [ - "kmeans" - ] + "source": [] }, { "cell_type": "code", @@ -237,7 +259,7 @@ "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=agglom1, colors=d1, color_col='popden')" + "# viewer2(edges=agglom1, colors=d1, color_col='popden')" ] }, { @@ -250,6 +272,10 @@ "agglom2 = aggregate_zones(\n", " d1, cluster_factors=cluster_factors, \n", " n_zones=300, method='agglom_adj', use_xy=False,\n", + " explicit_agg=[\n", + " 14377, 14380, 14384, \n", + " [15107, 15108, 15106, 15101],\n", + " ],\n", ")" ] }, @@ -260,7 +286,7 @@ "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=agglom2, colors=d1, color_col='popden')" + "# viewer2(edges=agglom2, colors=d1, color_col='popden')" ] }, { @@ -270,9 +296,18 @@ "metadata": {}, "outputs": [], "source": [ - "agglom3 = aggregate_zones(d1, cluster_factors=cluster_factors, \n", - " cluster_factors_onehot={'taz':5},\n", - " n_zones=100, method='agglom_adj', use_xy=False)" + "agglom3 = aggregate_zones(\n", + " d1, \n", + " cluster_factors=cluster_factors, \n", + " cluster_factors_onehot={'taz':15},\n", + " n_zones=300, \n", + " method='agglom_adj', \n", + " use_xy=False,\n", + " explicit_agg=[\n", + " 14377, 14380, 14384, \n", + " [15107, 15108, 15106, 15101],\n", + " ],\n", + ")" ] }, { @@ -282,7 +317,7 @@ "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=agglom3, colors=d1, color_col='popden')" + "# viewer2(edges=agglom3, colors=d1, color_col='popden')" ] }, { @@ -292,33 +327,54 @@ "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=agglom3, colors=d1, color_col='modeshare_NM')" + "# viewer2(edges=agglom3, colors=d1, color_col='modeshare_NM')" ] }, { "cell_type": "code", "execution_count": null, - "id": "c7ca8388", + "id": "00cfa00b", "metadata": {}, "outputs": [], "source": [ - "d1['taz20'] = d1.taz % 20" + "# viewer2(edges=agglom3, colors=d1, color_col='taz20')" ] }, { "cell_type": "code", "execution_count": null, - "id": "00cfa00b", + "id": "aa31c88d", "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=agglom3, colors=d1, color_col='taz20')" + "agglom3full = aggregate_zones_within_districts(\n", + " mgra, \n", + " cluster_factors=cluster_factors, \n", + " cluster_factors_onehot={'taz':15},\n", + " n_zones=2000,\n", + " method='agglom_adj', \n", + " use_xy=1e-4,\n", + " explicit_agg=[\n", + " 14571, 14574, 14578, \n", + " [15300, 15303, 15304, 15305],\n", + " ],\n", + ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "aa31c88d", + "id": "45efcfe4", + "metadata": {}, + "outputs": [], + "source": [ + "viewer2(edges=agglom3full, colors=agglom3full, color_col='empden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "fbd425c6", "metadata": {}, "outputs": [], "source": [] diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..77ae764 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,27 @@ +[build-system] +requires = [ + "setuptools>=42", + "wheel", + "setuptools_scm[toml]>=3.4", + "setuptools_scm_git_archive", +] +build-backend = "setuptools.build_meta" + +[tool.setuptools_scm] +fallback_version = "0.0.1" +write_to = "sandag_rsm/_version.py" + +[tool.isort] +profile = "black" +skip_gitignore = true +float_to_top = true +default_section = "THIRDPARTY" +known_first_party = "sandag_rsm" + +[tool.pytest.ini_options] +minversion = "6.0" +addopts = "-v --nbmake --disable-warnings" +testpaths = [ + "sandag_rsm/tests", + "docs", +] diff --git a/sandag_rsm/data_load/skims.py b/sandag_rsm/data_load/skims.py new file mode 100644 index 0000000..c28e491 --- /dev/null +++ b/sandag_rsm/data_load/skims.py @@ -0,0 +1,27 @@ +import os + +import openmatrix + + +def open_skims( + skims_filename="FromSANDAG-Files/traffic_skims_AM.omx", + data_dir=None, +): + + if data_dir is not None: + data_dir = os.path.expanduser(data_dir) + cwd = os.getcwd() + os.chdir(data_dir) + else: + cwd = None + + try: + s = openmatrix.open_file( + os.path.join(data_dir, skims_filename), + mode="r", + ) + return s + + finally: + # change back to original cwd + os.chdir(cwd) diff --git a/sandag_rsm/poi.py b/sandag_rsm/poi.py index ff567fa..57f1a79 100644 --- a/sandag_rsm/poi.py +++ b/sandag_rsm/poi.py @@ -1,7 +1,12 @@ +import itertools +from pathlib import Path + import pandas as pd import pyproj import shapely.geometry.point +from .data_load.skims import open_skims + # lat-lon of certain points points_of_interest = dict( san_diego_city_hall=(32.71691, -117.16282), @@ -25,19 +30,47 @@ def poi_taz_mgra(gdf): def attach_poi_taz_skims( - gdf, - skims_omx, - names, - poi=None, + gdf, skims_omx, names, poi=None, data_dir=None, taz_col="taz", cluster_factors=None ): + """ + Attach TAZ-based skim values to rows of a geodataframe. + + Parameters + ---------- + gdf : GeoDataFrame + The skimmed values will be added as columns to this [geo]dataframe. + If the POI's are given explicitly, this could be a regular pandas + DataFrame, otherwise the geometry is used to find the TAZ's of the + points of interest. + skims_omx : path-like or openmatrix.File + names : str or Mapping + Keys give the names of matrix tables to load out of the skims file. + Values give the relative weight for each table (used later in + clustering). + poi : Mapping + Maps named points of interest to the 'taz' id of each. If not given, + these will be computed based on the `gdf`. + data_dir : path-like, optional + Directory where the `skims_omx` file can be found, if not the current + working directory. + cluster_factors : Mapping, optional + Existing cluster_factors, to which the new factors are added. + + Returns + ------- + gdf : GeoDataFrame + cluster_factors : Mapping + """ if poi is None: poi = poi_taz_mgra(gdf) if isinstance(names, str): - names = [names] + names = {names: 1.0} + if isinstance(skims_omx, (str, Path)): + skims_omx = open_skims(skims_omx, data_dir=data_dir) zone_nums = skims_omx.root.lookup.zone_number cols = {} for k in poi: - ktaz = poi[k]["taz"] + ktaz = poi[k][taz_col] for name in names: cols[f"{k}_{name}"] = pd.Series( skims_omx.root.data[name][ktaz - 1], @@ -45,5 +78,10 @@ def attach_poi_taz_skims( ) add_to_gdf = {} for c in cols: - add_to_gdf[c] = gdf["taz"].map(cols[c]) - return gdf.assign(**add_to_gdf) + add_to_gdf[c] = gdf[taz_col].map(cols[c]) + if cluster_factors is None: + cluster_factors = {} + new_cluster_factors = { + f"{i}_{j}": names[j] for i, j in itertools.product(poi.keys(), names.keys()) + } + return gdf.assign(**add_to_gdf), cluster_factors | new_cluster_factors diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index b52abd4..b8c6488 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -1,10 +1,13 @@ import logging +from functools import partial +from numbers import Number +from statistics import mode import networkx as nx import numpy as np import pandas as pd import pyproj -from statistics import mode +from scipy.optimize import minimize_scalar from sklearn.cluster import AgglomerativeClustering, KMeans from sklearn.preprocessing import OneHotEncoder @@ -19,7 +22,7 @@ def aggregate_zones( cluster_factors=None, cluster_factors_onehot=None, use_xy=True, - explicit_agg=(), # TODO + explicit_agg=(), agg_instruction=None, ): """ @@ -48,34 +51,45 @@ def aggregate_zones( GeoDataFrame """ - - if cluster_factors is None: cluster_factors = {} - - if not explicit_agg: - mgra_gdf = mgra_gdf.copy() - n_zones_upd = n_zones - + n = 1 + if explicit_agg: + explicit_agg_ids = {} + for i in explicit_agg: + if isinstance(i, Number): + explicit_agg_ids[i] = n + else: + for j in i: + explicit_agg_ids[j] = n + n += 1 + in_explicit = mgra_gdf["mgra"].isin(explicit_agg_ids) + mgra_gdf_algo = mgra_gdf.loc[~in_explicit].copy() + mgra_gdf_explicit = mgra_gdf.loc[in_explicit].copy() + mgra_gdf_explicit["cluster_id"] = mgra_gdf_explicit["mgra"].map( + explicit_agg_ids + ) + n_zones_algorithm = n_zones - len( + mgra_gdf_explicit["cluster_id"].value_counts() + ) else: - mgra_gdf = mgra_gdf.loc[~mgra_gdf['mgra'].isin(explicit_agg)] - mgra_gdf_explicit = mgra_gdf.loc[mgra_gdf['mgra'].isin(explicit_agg)] - n_zones_upd = n_zones - len(mgra_gdf_explicit) - + mgra_gdf_algo = mgra_gdf.copy() + mgra_gdf_explicit = None + n_zones_algorithm = n_zones if use_xy: - geometry = mgra_gdf.centroid + geometry = mgra_gdf_algo.centroid X = list(geometry.apply(lambda p: p.x)) Y = list(geometry.apply(lambda p: p.y)) - factors = [X, Y] + factors = [np.asarray(X) * use_xy, np.asarray(Y) * use_xy] else: factors = [] for cf, cf_wgt in cluster_factors.items(): - factors.append(cf_wgt * mgra_gdf[cf].values.astype(np.float32)) + factors.append(cf_wgt * mgra_gdf_algo[cf].values.astype(np.float32)) if cluster_factors_onehot: for cf, cf_wgt in cluster_factors_onehot.items(): - factors.append(cf_wgt * OneHotEncoder().fit_transform(mgra_gdf[[cf]])) + factors.append(cf_wgt * OneHotEncoder().fit_transform(mgra_gdf_algo[[cf]])) from scipy.sparse import hstack factors2d = [] @@ -89,22 +103,22 @@ def aggregate_zones( data = np.array(factors).T if method == "kmeans": - kmeans = KMeans(n_clusters=n_zones_upd, random_state=random_state) + kmeans = KMeans(n_clusters=n_zones_algorithm, random_state=random_state) kmeans.fit(data) cluster_id = kmeans.labels_ elif method == "agglom": agglom = AgglomerativeClustering( - n_clusters=n_zones_upd, affinity="euclidean", linkage="ward" + n_clusters=n_zones_algorithm, affinity="euclidean", linkage="ward" ) agglom.fit_predict(data) cluster_id = agglom.labels_ elif method == "agglom_adj": from libpysal.weights import Rook - w_rook = Rook.from_dataframe(mgra_gdf) + w_rook = Rook.from_dataframe(mgra_gdf_algo) adj_mat = nx.adjacency_matrix(w_rook.to_networkx()) agglom = AgglomerativeClustering( - n_clusters=n_zones_upd, + n_clusters=n_zones_algorithm, affinity="euclidean", linkage="ward", connectivity=adj_mat, @@ -113,21 +127,59 @@ def aggregate_zones( cluster_id = agglom.labels_ else: raise NotImplementedError(method) - mgra_gdf["cluster_id"] = cluster_id + mgra_gdf_algo["cluster_id"] = cluster_id if agg_instruction is None: # TODO: fill out the aggregation data system # Define a lambda function to compute the weighted mean: - - wgt_avg_hh = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hh']) if mgra_gdf.loc[x.index, 'hh'].sum()> 0 else 0 - wgt_avg_hpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hstallssam']) if mgra_gdf.loc[x.index,'hstallssam'].sum()>0 else 0 - wgt_avg_dpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index,'dstallssam']) if mgra_gdf.loc[x.index,'dstallssam'].sum()>0 else 0 - wgt_avg_mpc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'mstallssam']) if mgra_gdf.loc[x.index,'mstallssam'].sum()>0 else 0 - wgt_avg_mtc = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 - wgt_avg_mat = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'pop']) if mgra_gdf.loc[x.index, 'pop'].sum()>0 else 0 - wgt_avg_dud = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'hh']) if mgra_gdf.loc[x.index, 'hh'].sum()>0 else 0 - wgt_avg_empden = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'emp_total']) if mgra_gdf.loc[x.index, 'emp_total'].sum()>0 else 0 - wgt_avg_rtempden = lambda x: np.average(x, weights=mgra_gdf.loc[x.index, 'emp_total']) if mgra_gdf.loc[x.index, 'emp_retail'].sum()>0 else 0 - wgt_avg_peden = lambda x: np.average(x, weights=(mgra_gdf.loc[x.index, 'emp_total'] + mgra_gdf.loc[x.index, 'pop'])) if (mgra_gdf.loc[x.index, 'emp_total'].sum()+mgra_gdf.loc[x.index, 'pop'].sum())>0 else 0 + wgt_avg_by_hh = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "hh"]) + if mgra_gdf.loc[x.index, "hh"].sum() > 0 + else 0 + ) + wgt_avg_hpc = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "hstallssam"]) + if mgra_gdf.loc[x.index, "hstallssam"].sum() > 0 + else 0 + ) + wgt_avg_dpc = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "dstallssam"]) + if mgra_gdf.loc[x.index, "dstallssam"].sum() > 0 + else 0 + ) + wgt_avg_mpc = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "mstallssam"]) + if mgra_gdf.loc[x.index, "mstallssam"].sum() > 0 + else 0 + ) + wgt_avg_by_pop = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "pop"]) + if mgra_gdf.loc[x.index, "pop"].sum() > 0 + else 0 + ) + wgt_avg_empden = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "emp_total"]) + if mgra_gdf.loc[x.index, "emp_total"].sum() > 0 + else 0 + ) + wgt_avg_rtempden = ( + lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "emp_retail"]) + if mgra_gdf.loc[x.index, "emp_retail"].sum() > 0 + else 0 + ) + wgt_avg_peden = ( + lambda x: np.average( + x, + weights=( + mgra_gdf.loc[x.index, "emp_total"] + mgra_gdf.loc[x.index, "pop"] + ), + ) + if ( + mgra_gdf.loc[x.index, "emp_total"].sum() + + mgra_gdf.loc[x.index, "pop"].sum() + ) + > 0 + else 0 + ) get_mode = lambda x: mode(x) agg_instruction = { "hs": "sum", @@ -150,7 +202,7 @@ def aggregate_zones( "i8": "sum", "i9": "sum", "i10": "sum", - "hhs": wgt_avg_hh, + "hhs": wgt_avg_by_hh, "pop": "sum", "hhp": "sum", "emp_ag": "sum", @@ -215,48 +267,123 @@ def aggregate_zones( "midpriceroom": "sum", "upscaleroom": "sum", "hotelroomtotal": "sum", - #"luz_id": "sum", + # "luz_id": "sum", "truckregiontype": "sum", - #"district27": "sum", - "milestocoast": wgt_avg_mtc, - #"acres": "sum", - #"effective_acres": "sum", - #"land_acres": "sum", - "MicroAccessTime": wgt_avg_mat, + # "district27": "sum", + "milestocoast": wgt_avg_by_pop, + # "acres": "sum", + # "effective_acres": "sum", + # "land_acres": "sum", + "MicroAccessTime": wgt_avg_by_pop, "remoteAVParking": "max", "refueling_stations": "sum", "totint": "sum", - "duden": wgt_avg_dud, + "duden": wgt_avg_by_hh, "empden": wgt_avg_empden, - #"popden": "sum", + # "popden": "sum", "retempden": wgt_avg_rtempden, - #"totintbin": "sum", #bins in original data 0, 80, 130 - #"empdenbin": "sum", #bins in original data 0, 10, 30 - #"dudenbin": "sum", #bins in original data 0, 5, 10 - "PopEmpDenPerMi": wgt_avg_peden + # "totintbin": "sum", #bins in original data 0, 80, 130 + # "empdenbin": "sum", #bins in original data 0, 10, 30 + # "dudenbin": "sum", #bins in original data 0, 5, 10 + "PopEmpDenPerMi": wgt_avg_peden, } - dissolved = mgra_gdf[["cluster_id", "geometry"]].dissolve(by="cluster_id") - dissolved = dissolved.join(mgra_gdf.groupby("cluster_id").agg(agg_instruction)) - #adding bins - dissolved['totintbin'] = 1 - dissolved.loc[(dissolved['totintbin'] >= 80) & (dissolved['totintbin'] < 130), 'totintbin'] = 2 - dissolved.loc[(dissolved['totintbin'] >= 130), 'totintbin'] = 3 + pending = [] + for df in [mgra_gdf_algo, mgra_gdf_explicit]: + + dissolved = df[["cluster_id", "geometry"]].dissolve(by="cluster_id") + dissolved = dissolved.join( + mgra_gdf_algo.groupby("cluster_id").agg(agg_instruction) + ) + + # adding bins + dissolved["totintbin"] = 1 + dissolved.loc[ + (dissolved["totintbin"] >= 80) & (dissolved["totintbin"] < 130), "totintbin" + ] = 2 + dissolved.loc[(dissolved["totintbin"] >= 130), "totintbin"] = 3 - dissolved['empdenbin'] = 1 - dissolved.loc[(dissolved['empdenbin'] >= 10) & (dissolved['empdenbin'] < 30), 'empdenbin'] = 2 - dissolved.loc[(dissolved['empdenbin'] >= 30), 'empdenbin'] = 3 + dissolved["empdenbin"] = 1 + dissolved.loc[ + (dissolved["empdenbin"] >= 10) & (dissolved["empdenbin"] < 30), "empdenbin" + ] = 2 + dissolved.loc[(dissolved["empdenbin"] >= 30), "empdenbin"] = 3 - dissolved['dudenbin'] = 1 - dissolved.loc[(dissolved['dudenbin'] >=5 ) & (dissolved['dudenbin'] < 10), 'dudenbin'] = 2 - dissolved.loc[(dissolved['dudenbin'] >= 10), 'dudebin'] = 3 + dissolved["dudenbin"] = 1 + dissolved.loc[ + (dissolved["dudenbin"] >= 5) & (dissolved["dudenbin"] < 10), "dudenbin" + ] = 2 + dissolved.loc[(dissolved["dudenbin"] >= 10), "dudenbin"] = 3 - - dissolved["cluster_id"] = list(range(n_zones_upd, n_zones, 1)) - dissolved = pd.concat([dissolved, mgra_gdf_explicit], ignore_index = False) - dissolved = dissolved.reset_index(drop=True) + pending.append(dissolved) - return dissolved + pending[0]["cluster_id"] = list(range(n, n + n_zones_algorithm)) + pending[0] = pending[0][[c for c in pending[1].columns if c in pending[0].columns]] + pending[1] = pending[1][[c for c in pending[0].columns if c in pending[1].columns]] + combined = pd.concat(pending, ignore_index=False) + combined = combined.reset_index(drop=True) + + return combined + + +def _scale_zones(x, zones_by_district, district_focus): + x = np.abs(x) + agg_by_district = pd.Series(zones_by_district) * x + for i, j in district_focus.items(): + agg_by_district[i] *= j + agg_by_district = np.minimum(agg_by_district, zones_by_district) + agg_by_district = np.maximum(agg_by_district, 1) + return agg_by_district + + +def _rescale_zones(x, n_z, zones_by_district, district_focus): + return (_scale_zones(x, zones_by_district, district_focus).sum() - n_z) ** 2 + + +def aggregate_zones_within_districts( + mgra_gdf, + method="kmeans", + n_zones=2000, + random_state=0, + cluster_factors=None, + cluster_factors_onehot=None, + use_xy=True, + explicit_agg=(), + agg_instruction=None, + district_col="district27", + district_focus=None, +): + logger.info("aggregate_zones_within_districts...") + if district_focus is None: + district_focus = {} + zones_by_district = mgra_gdf[district_col].value_counts() + rescale_zones = partial( + _rescale_zones, + n_z=n_zones, + zones_by_district=zones_by_district, + district_focus=district_focus, + ) + zone_factor = minimize_scalar(rescale_zones).x + agg_by_district = ( + _scale_zones(zone_factor, zones_by_district, district_focus).round().astype(int) + ) + out = [] + for district_n, district_z in agg_by_district.items(): + logger.info(f"combining district {district_n} into {district_z} zones") + out.append( + aggregate_zones( + mgra_gdf[mgra_gdf[district_col] == district_n], + method=method, + n_zones=district_z, + random_state=random_state + district_n, + cluster_factors=cluster_factors, + cluster_factors_onehot=cluster_factors_onehot, + use_xy=use_xy, + explicit_agg=explicit_agg, + agg_instruction=agg_instruction, + ) + ) + return pd.concat(out).reset_index(drop=True) def viewer( From 660dbadf7d7b474f2c8133aa4d126673ac3e9379 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Thu, 5 May 2022 09:06:18 -0500 Subject: [PATCH 06/30] notebook --- notebooks/ZoneAggDemo.ipynb | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index d7a73ca..0381249 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -97,7 +97,7 @@ { "cell_type": "code", "execution_count": null, - "id": "1499fc7f", + "id": "2c974741", "metadata": {}, "outputs": [], "source": [ @@ -186,7 +186,7 @@ { "cell_type": "code", "execution_count": null, - "id": "1b1fbdee", + "id": "872ac31b", "metadata": {}, "outputs": [], "source": [ @@ -207,7 +207,7 @@ { "cell_type": "code", "execution_count": null, - "id": "02023d6a", + "id": "35211c71", "metadata": {}, "outputs": [], "source": [ @@ -217,7 +217,7 @@ { "cell_type": "code", "execution_count": null, - "id": "20027832", + "id": "f5d6cd56", "metadata": {}, "outputs": [], "source": [ @@ -237,7 +237,7 @@ { "cell_type": "code", "execution_count": null, - "id": "2f68ec2b", + "id": "9cedfdb9", "metadata": {}, "outputs": [], "source": [] @@ -350,7 +350,7 @@ "agglom3full = aggregate_zones_within_districts(\n", " mgra, \n", " cluster_factors=cluster_factors, \n", - " cluster_factors_onehot={'taz':15},\n", + " cluster_factors_onehot={'taz':100},\n", " n_zones=2000,\n", " method='agglom_adj', \n", " use_xy=1e-4,\n", @@ -364,7 +364,7 @@ { "cell_type": "code", "execution_count": null, - "id": "45efcfe4", + "id": "18d6902a", "metadata": {}, "outputs": [], "source": [ @@ -374,7 +374,7 @@ { "cell_type": "code", "execution_count": null, - "id": "fbd425c6", + "id": "391a94af", "metadata": {}, "outputs": [], "source": [] From bc38c459d4ee7c7d3070d2a99248adcbc6c03bd1 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 9 May 2022 13:32:24 -0500 Subject: [PATCH 07/30] zone agg on TAZs --- README.md | 2 +- environment.yaml | 1 + notebooks/ZoneAggDemo.ipynb | 217 ++++++---------- sandag_rsm/data_load/skims.py | 2 +- sandag_rsm/data_load/triplist.py | 54 +++- sandag_rsm/data_load/zones.py | 2 +- sandag_rsm/poi.py | 8 +- sandag_rsm/zone_agg.py | 409 ++++++++++++++++--------------- setup.cfg | 10 +- 9 files changed, 361 insertions(+), 344 deletions(-) diff --git a/README.md b/README.md index 673e61b..1182060 100644 --- a/README.md +++ b/README.md @@ -46,7 +46,7 @@ On the host machine, run: ```shell docker run -v $(pwd):/home/mambauser/sandag_rsm -p 8899:8899 \ - -it sandag_rsm jupyter notebook --ip 0.0.0.0 --no-browser --allow-root \ + -it --rm sandag_rsm jupyter notebook --ip 0.0.0.0 --no-browser --allow-root \ --port 8899 --notebook-dir=/home/mambauser ``` diff --git a/environment.yaml b/environment.yaml index 2a01ecd..e150f0d 100644 --- a/environment.yaml +++ b/environment.yaml @@ -14,6 +14,7 @@ dependencies: - libpysal - networkx - notebook + - openmatrix - pandas - plotly - pyproj diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index 0381249..07fd8de 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -8,9 +8,9 @@ "outputs": [], "source": [ "from sandag_rsm.data_load.zones import load_mgra_data\n", - "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra\n", + "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, trip_mode_shares_by_taz\n", "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", - "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts" + "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts, merge_zone_data" ] }, { @@ -20,7 +20,7 @@ "metadata": {}, "outputs": [], "source": [ - "data_dir = \"~/Library/CloudStorage/OneDrive-SharedLibraries-CambridgeSystematics/PROJ SANDAG RSM - General/Data\"" + "data_dir = \"../test/data\"" ] }, { @@ -46,22 +46,31 @@ { "cell_type": "code", "execution_count": null, - "id": "f3593b1c", + "id": "e6d1c5ef", "metadata": {}, "outputs": [], "source": [ - "trips = load_trip_list(\"FromSANDAG-Files/indivTripData_3.csv.gz\", data_dir=data_dir)\n", - "trip_mode_shares = trip_mode_shares_by_mgra(trips, mgras=mgra.MGRA)" + "trips = load_trip_list(\"trips_sample.pq\", data_dir=data_dir)" ] }, { "cell_type": "code", "execution_count": null, - "id": "37ff7b19", + "id": "4bb2cb35", "metadata": {}, "outputs": [], "source": [ - "trip_mode_shares" + "tazs = merge_zone_data(mgra, cluster_id=\"taz\")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "f3593b1c", + "metadata": {}, + "outputs": [], + "source": [ + "trip_mode_shares = trip_mode_shares_by_taz(trips, tazs=tazs.index, mgra_gdf=mgra)" ] }, { @@ -71,7 +80,7 @@ "metadata": {}, "outputs": [], "source": [ - "mgra = mgra.join(trip_mode_shares.add_prefix(\"modeshare_\"), on='MGRA')" + "tazs = tazs.join(trip_mode_shares.add_prefix(\"modeshare_\"), on='taz')" ] }, { @@ -111,9 +120,9 @@ "metadata": {}, "outputs": [], "source": [ - "mgra, cluster_factors = attach_poi_taz_skims(\n", - " mgra,\n", - " \"FromSANDAG-Files/traffic_skims_AM.omx\",\n", + "tazs, cluster_factors = attach_poi_taz_skims(\n", + " tazs,\n", + " \"traffic_skims_AM_mini.omx\",\n", " names='AM_SOV_TR_M_TIME',\n", " poi=poi,\n", " data_dir=data_dir,\n", @@ -124,120 +133,100 @@ { "cell_type": "code", "execution_count": null, - "id": "04f9b597", + "id": "772c319e", "metadata": {}, "outputs": [], "source": [ - "d1 = mgra.query(\"district27 == 1\")" + "explicit_agg=[\n", + " 571, 588, 606, \n", + " [143, 270, 15],\n", + "]\n" ] }, { "cell_type": "code", "execution_count": null, - "id": "a93bd34a", + "id": "2f7c2125", "metadata": {}, "outputs": [], "source": [ - "viewer(d1, color='popden', marker_line_width=1)" + "d1 = tazs.query(\"district27 == 1\")" ] }, { "cell_type": "code", "execution_count": null, - "id": "f66915c5", + "id": "a93bd34a", "metadata": {}, "outputs": [], "source": [ - "# viewer(d1, color='modeshare_WT', marker_line_width=0)" + "viewer(d1, color='popden', marker_line_width=0)" ] }, { "cell_type": "code", "execution_count": null, - "id": "61b2c6c7", + "id": "073d0c19", "metadata": {}, "outputs": [], "source": [ - "# import itertools\n", - "# cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", - "# cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", - "cluster_factors" + "viewer(d1, color='outside_pendleton_gate_AM_SOV_TR_M_TIME', marker_line_width=0)" ] }, { "cell_type": "code", "execution_count": null, - "id": "3234d883", - "metadata": {}, - "outputs": [], - "source": [ - "kmeans = aggregate_zones(\n", - " d1, \n", - " cluster_factors=cluster_factors, \n", - " n_zones=300,\n", - " explicit_agg=[\n", - " 14571, 14574, 14578, \n", - " [15300, 15303, 15304, 15305],\n", - " ],\n", - " use_xy=1e-6,\n", - ")" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "872ac31b", + "id": "f66915c5", "metadata": {}, "outputs": [], "source": [ - "from sandag_rsm.zone_agg import aggregate_zones_within_districts\n", - "\n", - "kmeans = aggregate_zones_within_districts(\n", - " mgra, \n", - " cluster_factors=cluster_factors, \n", - " n_zones=2000,\n", - " use_xy=1e-6,\n", - " explicit_agg=[\n", - " 14571, 14574, 14578, \n", - " [15300, 15303, 15304, 15305],\n", - " ],\n", - ")" + "viewer(d1, color='modeshare_WT', marker_line_width=0)" ] }, { "cell_type": "code", "execution_count": null, - "id": "35211c71", + "id": "61b2c6c7", "metadata": {}, "outputs": [], "source": [ - "(kmeans)" + "# import itertools\n", + "# cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", + "# cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", + "cluster_factors" ] }, { "cell_type": "code", "execution_count": null, - "id": "f5d6cd56", + "id": "3234d883", "metadata": {}, "outputs": [], "source": [ - "kmeans = kmeans.reset_index(drop=True)" + "kmeans1 = aggregate_zones(\n", + " d1, \n", + " cluster_factors=cluster_factors, \n", + " n_zones=100,\n", + " explicit_agg=explicit_agg,\n", + " explicit_col='taz',\n", + " use_xy=1e-6,\n", + ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "7e7b56ea", + "id": "b50a4bc3", "metadata": {}, "outputs": [], "source": [ - "viewer2(edges=kmeans, colors=kmeans, color_col='empden')" + "viewer2(edges=kmeans1, colors=d1, color_col='empden')" ] }, { "cell_type": "code", "execution_count": null, - "id": "9cedfdb9", + "id": "1a760ac4", "metadata": {}, "outputs": [], "source": [] @@ -245,99 +234,40 @@ { "cell_type": "code", "execution_count": null, - "id": "14e60056", - "metadata": {}, - "outputs": [], - "source": [ - "agglom1 = aggregate_zones(d1, cluster_factors=cluster_factors, n_zones=300, method='agglom')" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "abe08363", - "metadata": {}, - "outputs": [], - "source": [ - "# viewer2(edges=agglom1, colors=d1, color_col='popden')" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "65d89c03", - "metadata": {}, - "outputs": [], - "source": [ - "agglom2 = aggregate_zones(\n", - " d1, cluster_factors=cluster_factors, \n", - " n_zones=300, method='agglom_adj', use_xy=False,\n", - " explicit_agg=[\n", - " 14377, 14380, 14384, \n", - " [15107, 15108, 15106, 15101],\n", - " ],\n", - ")" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "b3d23e2d", - "metadata": {}, - "outputs": [], - "source": [ - "# viewer2(edges=agglom2, colors=d1, color_col='popden')" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "cd43c71f", + "id": "872ac31b", "metadata": {}, "outputs": [], "source": [ - "agglom3 = aggregate_zones(\n", - " d1, \n", + "from sandag_rsm.zone_agg import aggregate_zones_within_districts\n", + "\n", + "kmeans = aggregate_zones_within_districts(\n", + " tazs, \n", " cluster_factors=cluster_factors, \n", - " cluster_factors_onehot={'taz':15},\n", - " n_zones=300, \n", - " method='agglom_adj', \n", - " use_xy=False,\n", - " explicit_agg=[\n", - " 14377, 14380, 14384, \n", - " [15107, 15108, 15106, 15101],\n", - " ],\n", + " n_zones=1000,\n", + " use_xy=1e-6,\n", + " explicit_agg=explicit_agg,\n", + " explicit_col='taz',\n", ")" ] }, { "cell_type": "code", "execution_count": null, - "id": "ef76f562", - "metadata": {}, - "outputs": [], - "source": [ - "# viewer2(edges=agglom3, colors=d1, color_col='popden')" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "346536be", + "id": "f5d6cd56", "metadata": {}, "outputs": [], "source": [ - "# viewer2(edges=agglom3, colors=d1, color_col='modeshare_NM')" + "kmeans = kmeans.reset_index(drop=True)" ] }, { "cell_type": "code", "execution_count": null, - "id": "00cfa00b", + "id": "7e7b56ea", "metadata": {}, "outputs": [], "source": [ - "# viewer2(edges=agglom3, colors=d1, color_col='taz20')" + "viewer2(edges=kmeans, colors=kmeans, color_col='empden')" ] }, { @@ -348,16 +278,13 @@ "outputs": [], "source": [ "agglom3full = aggregate_zones_within_districts(\n", - " mgra, \n", + " tazs, \n", " cluster_factors=cluster_factors, \n", - " cluster_factors_onehot={'taz':100},\n", " n_zones=2000,\n", " method='agglom_adj', \n", " use_xy=1e-4,\n", - " explicit_agg=[\n", - " 14571, 14574, 14578, \n", - " [15300, 15303, 15304, 15305],\n", - " ],\n", + " explicit_agg=explicit_agg,\n", + " explicit_col='taz',\n", ")" ] }, @@ -377,6 +304,16 @@ "id": "391a94af", "metadata": {}, "outputs": [], + "source": [ + "viewer2(edges=agglom3full, colors=agglom3full, color_col='popden')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ba9b4203", + "metadata": {}, + "outputs": [], "source": [] } ], diff --git a/sandag_rsm/data_load/skims.py b/sandag_rsm/data_load/skims.py index c28e491..824de00 100644 --- a/sandag_rsm/data_load/skims.py +++ b/sandag_rsm/data_load/skims.py @@ -17,7 +17,7 @@ def open_skims( try: s = openmatrix.open_file( - os.path.join(data_dir, skims_filename), + skims_filename, mode="r", ) return s diff --git a/sandag_rsm/data_load/triplist.py b/sandag_rsm/data_load/triplist.py index b029546..a1963ef 100644 --- a/sandag_rsm/data_load/triplist.py +++ b/sandag_rsm/data_load/triplist.py @@ -4,7 +4,7 @@ def load_trip_list( - trips_filename="indivTripData_3.csv.gz", + trips_filename="indivTripData_3.parquet", data_dir=None, ): if data_dir is not None: @@ -15,7 +15,10 @@ def load_trip_list( cwd = None try: - trips = pd.read_csv(trips_filename) + if trips_filename.endswith(".pq") or trips_filename.endswith(".parquet"): + trips = pd.read_parquet(trips_filename) + else: + trips = pd.read_csv(trips_filename) return trips finally: @@ -54,3 +57,50 @@ def trip_mode_shares_by_mgra( tm = tm + background tripmodeshare = tm.div(tm.sum(axis=1), axis=0) return tripmodeshare + + +def trip_mode_shares_by_taz( + trips, + mgra_to_taz=None, + background_per_taz=50, + tazs=None, + mgra_gdf=None, +): + if mgra_gdf is not None and mgra_to_taz is None: + mgra_to_taz = pd.Series(mgra_gdf.taz.values, index=mgra_gdf.MGRA) + trip_modes = { + 1: "Au", # Drive Alone + 2: "Au", # Shared Ride 2 + 3: "Au", # Shared Ride 3 + 4: "NM", # Walk + 5: "NM", # Bike + 6: "WT", # Walk to Transit + 7: "DT", # Park and Ride to Transit + 8: "DT", # Kiss and Ride to Transit + 9: "Au", # TNC to Transit + 10: "Au", # Taxi + 11: "Au", # TNC Single + 12: "Au", # TNC Shared + 13: "Au", # School Bus + } + trip_mode_cat = trips["trip_mode"].apply(trip_modes.get) + tmo = ( + trips.groupby([trips.orig_mgra.map(mgra_to_taz), trip_mode_cat]) + .size() + .unstack() + .fillna(0) + ) + tmd = ( + trips.groupby([trips.dest_mgra.map(mgra_to_taz), trip_mode_cat]) + .size() + .unstack() + .fillna(0) + ) + tm = tmo + tmd + tm_total = tm.sum() + background = background_per_taz * tm_total / tm_total.sum() + if tazs is not None: + tm = tm.reindex(tazs).fillna(0) + tm = tm + background + tripmodeshare = tm.div(tm.sum(axis=1), axis=0) + return tripmodeshare diff --git a/sandag_rsm/data_load/zones.py b/sandag_rsm/data_load/zones.py index 1f68aef..54d50c1 100644 --- a/sandag_rsm/data_load/zones.py +++ b/sandag_rsm/data_load/zones.py @@ -60,7 +60,7 @@ def simplify_shapefile( def load_mgra_data( shapefilename="MGRASHAPE.zip", - supplemental_features="mgra13_based_input2016.csv", + supplemental_features="mgra13_based_input2016.csv.gz", data_dir=None, simplify_tolerance=1, prequantize=False, diff --git a/sandag_rsm/poi.py b/sandag_rsm/poi.py index 57f1a79..d5c0a03 100644 --- a/sandag_rsm/poi.py +++ b/sandag_rsm/poi.py @@ -77,8 +77,14 @@ def attach_poi_taz_skims( index=zone_nums[:], ) add_to_gdf = {} + if taz_col in gdf: + gdf_taz_col = gdf[taz_col] + elif gdf.index.name == taz_col: + gdf_taz_col = pd.Series(data=gdf.index, index=gdf.index) + else: + raise KeyError(taz_col) for c in cols: - add_to_gdf[c] = gdf[taz_col].map(cols[c]) + add_to_gdf[c] = gdf_taz_col.map(cols[c]) if cluster_factors is None: cluster_factors = {} new_cluster_factors = { diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index b8c6488..8f9356a 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -14,6 +14,181 @@ logger = logging.getLogger(__name__) +def merge_zone_data( + gdf, + agg_instruction=None, + cluster_id="cluster_id", +): + if agg_instruction is None: + # TODO: fill out the aggregation data system + # make a sliver of area series as a backstop for weighted avg on other things + # we add this small value to each weighting, so that if the desired weighting values + # are all zero for any weighted average, the geographic area becomes the backup + # weight ... and if any are non-zero, then these areas round off to effectively nil. + gdf_area_small = gdf.area / gdf.area.mean() / 1000 + + def wgt_avg_by(x, c): + try: + return np.average( + x, weights=gdf.loc[x.index, c] + gdf_area_small.loc[x.index] + ) + except ZeroDivisionError: + return np.average(x) + + wgt_avg_by_hh = lambda x: wgt_avg_by(x, "hh") + wgt_avg_hpc = lambda x: wgt_avg_by(x, "hstallssam") + wgt_avg_dpc = lambda x: wgt_avg_by(x, "dstallssam") + wgt_avg_mpc = lambda x: wgt_avg_by(x, "mstallssam") + wgt_avg_by_pop = lambda x: wgt_avg_by(x, "pop") + wgt_avg_empden = lambda x: wgt_avg_by(x, "emp_total") + wgt_avg_popden = lambda x: wgt_avg_by(x, "pop") + wgt_avg_rtempden = lambda x: wgt_avg_by(x, "emp_retail") + + def wgt_avg_peden(x): + try: + return np.average( + x, + weights=gdf.loc[x.index, "emp_total"] + + gdf.loc[x.index, "pop"] + + gdf_area_small.loc[x.index], + ) + except ZeroDivisionError: + return np.average(x) + + get_mode = lambda x: mode(x) + agg_instruction = { + "hs": "sum", + "hs_sf": "sum", + "hs_mf": "sum", + "hs_mh": "sum", + "hh": "sum", + "hh_sf": "sum", + "hh_mf": "sum", + "hh_mh": "sum", + "gq_civ": "sum", + "gq_mil": "sum", + "i1": "sum", + "i2": "sum", + "i3": "sum", + "i4": "sum", + "i5": "sum", + "i6": "sum", + "i7": "sum", + "i8": "sum", + "i9": "sum", + "i10": "sum", + "hhs": wgt_avg_by_hh, + "pop": "sum", + "hhp": "sum", + "emp_ag": "sum", + "emp_const_non_bldg_prod": "sum", + "emp_const_non_bldg_office": "sum", + "emp_utilities_prod": "sum", + "emp_utilities_office": "sum", + "emp_const_bldg_prod": "sum", + "emp_const_bldg_office": "sum", + "emp_mfg_prod": "sum", + "emp_mfg_office": "sum", + "emp_whsle_whs": "sum", + "emp_trans": "sum", + "emp_retail": "sum", + "emp_prof_bus_svcs": "sum", + "emp_prof_bus_svcs_bldg_maint": "sum", + "emp_pvt_ed_k12": "sum", + "emp_pvt_ed_post_k12_oth": "sum", + "emp_health": "sum", + "emp_personal_svcs_office": "sum", + "emp_amusement": "sum", + "emp_hotel": "sum", + "emp_restaurant_bar": "sum", + "emp_personal_svcs_retail": "sum", + "emp_religious": "sum", + "emp_pvt_hh": "sum", + "emp_state_local_gov_ent": "sum", + "emp_fed_non_mil": "sum", + "emp_fed_mil": "sum", + "emp_state_local_gov_blue": "sum", + "emp_state_local_gov_white": "sum", + "emp_public_ed": "sum", + "emp_own_occ_dwell_mgmt": "sum", + "emp_fed_gov_accts": "sum", + "emp_st_lcl_gov_accts": "sum", + "emp_cap_accts": "sum", + "emp_total": "sum", + "enrollgradekto8": "sum", + "enrollgrade9to12": "sum", + "collegeenroll": "sum", + "othercollegeenroll": "sum", + "adultschenrl": "sum", + "ech_dist": get_mode, + "hch_dist": get_mode, + "parkarea": "max", + "hstallsoth": "sum", + "hstallssam": "sum", + "hparkcost": wgt_avg_hpc, + "numfreehrs": wgt_avg_hpc, + "dstallsoth": "sum", + "dstallssam": "sum", + "dparkcost": wgt_avg_dpc, + "mstallsoth": "sum", + "mstallssam": "sum", + "mparkcost": wgt_avg_mpc, + "parkactive": "sum", + "openspaceparkpreserve": "sum", + "beachactive": "sum", + "budgetroom": "sum", + "economyroom": "sum", + "luxuryroom": "sum", + "midpriceroom": "sum", + "upscaleroom": "sum", + "hotelroomtotal": "sum", + # "luz_id": "sum", + "truckregiontype": "sum", + "district27": get_mode, + "milestocoast": wgt_avg_by_pop, + # "acres": "sum", + # "effective_acres": "sum", + # "land_acres": "sum", + "MicroAccessTime": wgt_avg_by_pop, + "remoteAVParking": "max", + "refueling_stations": "sum", + "totint": "sum", + "duden": wgt_avg_by_hh, + "empden": wgt_avg_empden, + "popden": wgt_avg_popden, + "retempden": wgt_avg_rtempden, + # "totintbin": "sum", #bins in original data 0, 80, 130 + # "empdenbin": "sum", #bins in original data 0, 10, 30 + # "dudenbin": "sum", #bins in original data 0, 5, 10 + "PopEmpDenPerMi": wgt_avg_peden, + } + + dissolved = gdf[[cluster_id, "geometry"]].dissolve(by=cluster_id) + other_data = gdf.groupby(cluster_id).agg(agg_instruction) + dissolved = dissolved.join(other_data) + + # adding bins + dissolved["totintbin"] = 1 + dissolved.loc[ + (dissolved["totintbin"] >= 80) & (dissolved["totintbin"] < 130), "totintbin" + ] = 2 + dissolved.loc[(dissolved["totintbin"] >= 130), "totintbin"] = 3 + + dissolved["empdenbin"] = 1 + dissolved.loc[ + (dissolved["empdenbin"] >= 10) & (dissolved["empdenbin"] < 30), "empdenbin" + ] = 2 + dissolved.loc[(dissolved["empdenbin"] >= 30), "empdenbin"] = 3 + + dissolved["dudenbin"] = 1 + dissolved.loc[ + (dissolved["dudenbin"] >= 5) & (dissolved["dudenbin"] < 10), "dudenbin" + ] = 2 + dissolved.loc[(dissolved["dudenbin"] >= 10), "dudenbin"] = 3 + + return dissolved + + def aggregate_zones( mgra_gdf, method="kmeans", @@ -23,6 +198,7 @@ def aggregate_zones( cluster_factors_onehot=None, use_xy=True, explicit_agg=(), + explicit_col="mgra", agg_instruction=None, ): """ @@ -43,6 +219,9 @@ def aggregate_zones( A list containing integers (individual MGRAs that should not be aggregated) or lists of integers (groups of MGRAs that should be aggregated exactly as given, with no less and no more) + explicit_col : str + The name of the column containing the ID's from `explicit_agg`, usually + 'mgra' or 'taz' agg_instruction : dict Dictionary passed to pandas `agg` that says how to aggregate data columns. @@ -64,10 +243,13 @@ def aggregate_zones( for j in i: explicit_agg_ids[j] = n n += 1 - in_explicit = mgra_gdf["mgra"].isin(explicit_agg_ids) + if explicit_col == mgra_gdf.index.name: + mgra_gdf = mgra_gdf.reset_index() + mgra_gdf.index = mgra_gdf[explicit_col] + in_explicit = mgra_gdf[explicit_col].isin(explicit_agg_ids) mgra_gdf_algo = mgra_gdf.loc[~in_explicit].copy() mgra_gdf_explicit = mgra_gdf.loc[in_explicit].copy() - mgra_gdf_explicit["cluster_id"] = mgra_gdf_explicit["mgra"].map( + mgra_gdf_explicit["cluster_id"] = mgra_gdf_explicit[explicit_col].map( explicit_agg_ids ) n_zones_algorithm = n_zones - len( @@ -128,199 +310,35 @@ def aggregate_zones( else: raise NotImplementedError(method) mgra_gdf_algo["cluster_id"] = cluster_id - if agg_instruction is None: - # TODO: fill out the aggregation data system - # Define a lambda function to compute the weighted mean: - wgt_avg_by_hh = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "hh"]) - if mgra_gdf.loc[x.index, "hh"].sum() > 0 - else 0 - ) - wgt_avg_hpc = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "hstallssam"]) - if mgra_gdf.loc[x.index, "hstallssam"].sum() > 0 - else 0 - ) - wgt_avg_dpc = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "dstallssam"]) - if mgra_gdf.loc[x.index, "dstallssam"].sum() > 0 - else 0 - ) - wgt_avg_mpc = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "mstallssam"]) - if mgra_gdf.loc[x.index, "mstallssam"].sum() > 0 - else 0 - ) - wgt_avg_by_pop = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "pop"]) - if mgra_gdf.loc[x.index, "pop"].sum() > 0 - else 0 - ) - wgt_avg_empden = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "emp_total"]) - if mgra_gdf.loc[x.index, "emp_total"].sum() > 0 - else 0 - ) - wgt_avg_rtempden = ( - lambda x: np.average(x, weights=mgra_gdf.loc[x.index, "emp_retail"]) - if mgra_gdf.loc[x.index, "emp_retail"].sum() > 0 - else 0 + + if mgra_gdf_explicit is None or len(mgra_gdf_explicit) == 0: + combined = merge_zone_data( + mgra_gdf_algo, + agg_instruction, + cluster_id="cluster_id", ) - wgt_avg_peden = ( - lambda x: np.average( - x, - weights=( - mgra_gdf.loc[x.index, "emp_total"] + mgra_gdf.loc[x.index, "pop"] - ), - ) - if ( - mgra_gdf.loc[x.index, "emp_total"].sum() - + mgra_gdf.loc[x.index, "pop"].sum() + combined["cluster_id"] = list(range(n, n + n_zones_algorithm)) + else: + pending = [] + for df in [mgra_gdf_algo, mgra_gdf_explicit]: + logger.info(f"... merging {len(df)}") + pending.append( + merge_zone_data( + df, + agg_instruction, + cluster_id="cluster_id", + ).reset_index() ) - > 0 - else 0 - ) - get_mode = lambda x: mode(x) - agg_instruction = { - "hs": "sum", - "hs_sf": "sum", - "hs_mf": "sum", - "hs_mh": "sum", - "hh": "sum", - "hh_sf": "sum", - "hh_mf": "sum", - "hh_mh": "sum", - "gq_civ": "sum", - "gq_mil": "sum", - "i1": "sum", - "i2": "sum", - "i3": "sum", - "i4": "sum", - "i5": "sum", - "i6": "sum", - "i7": "sum", - "i8": "sum", - "i9": "sum", - "i10": "sum", - "hhs": wgt_avg_by_hh, - "pop": "sum", - "hhp": "sum", - "emp_ag": "sum", - "emp_const_non_bldg_prod": "sum", - "emp_const_non_bldg_office": "sum", - "emp_utilities_prod": "sum", - "emp_utilities_office": "sum", - "emp_const_bldg_prod": "sum", - "emp_const_bldg_office": "sum", - "emp_mfg_prod": "sum", - "emp_mfg_office": "sum", - "emp_whsle_whs": "sum", - "emp_trans": "sum", - "emp_retail": "sum", - "emp_prof_bus_svcs": "sum", - "emp_prof_bus_svcs_bldg_maint": "sum", - "emp_pvt_ed_k12": "sum", - "emp_pvt_ed_post_k12_oth": "sum", - "emp_health": "sum", - "emp_personal_svcs_office": "sum", - "emp_amusement": "sum", - "emp_hotel": "sum", - "emp_restaurant_bar": "sum", - "emp_personal_svcs_retail": "sum", - "emp_religious": "sum", - "emp_pvt_hh": "sum", - "emp_state_local_gov_ent": "sum", - "emp_fed_non_mil": "sum", - "emp_fed_mil": "sum", - "emp_state_local_gov_blue": "sum", - "emp_state_local_gov_white": "sum", - "emp_public_ed": "sum", - "emp_own_occ_dwell_mgmt": "sum", - "emp_fed_gov_accts": "sum", - "emp_st_lcl_gov_accts": "sum", - "emp_cap_accts": "sum", - "emp_total": "sum", - "enrollgradekto8": "sum", - "enrollgrade9to12": "sum", - "collegeenroll": "sum", - "othercollegeenroll": "sum", - "adultschenrl": "sum", - "ech_dist": get_mode, - "hch_dist": get_mode, - "parkarea": "max", - "hstallsoth": "sum", - "hstallssam": "sum", - "hparkcost": wgt_avg_hpc, - "numfreehrs": wgt_avg_hpc, - "dstallsoth": "sum", - "dstallssam": "sum", - "dparkcost": wgt_avg_dpc, - "mstallsoth": "sum", - "mstallssam": "sum", - "mparkcost": wgt_avg_mpc, - "parkactive": "sum", - "openspaceparkpreserve": "sum", - "beachactive": "sum", - "budgetroom": "sum", - "economyroom": "sum", - "luxuryroom": "sum", - "midpriceroom": "sum", - "upscaleroom": "sum", - "hotelroomtotal": "sum", - # "luz_id": "sum", - "truckregiontype": "sum", - # "district27": "sum", - "milestocoast": wgt_avg_by_pop, - # "acres": "sum", - # "effective_acres": "sum", - # "land_acres": "sum", - "MicroAccessTime": wgt_avg_by_pop, - "remoteAVParking": "max", - "refueling_stations": "sum", - "totint": "sum", - "duden": wgt_avg_by_hh, - "empden": wgt_avg_empden, - # "popden": "sum", - "retempden": wgt_avg_rtempden, - # "totintbin": "sum", #bins in original data 0, 80, 130 - # "empdenbin": "sum", #bins in original data 0, 10, 30 - # "dudenbin": "sum", #bins in original data 0, 5, 10 - "PopEmpDenPerMi": wgt_avg_peden, - } - pending = [] - for df in [mgra_gdf_algo, mgra_gdf_explicit]: + pending[0]["cluster_id"] = list(range(n, n + n_zones_algorithm)) - dissolved = df[["cluster_id", "geometry"]].dissolve(by="cluster_id") - dissolved = dissolved.join( - mgra_gdf_algo.groupby("cluster_id").agg(agg_instruction) - ) - - # adding bins - dissolved["totintbin"] = 1 - dissolved.loc[ - (dissolved["totintbin"] >= 80) & (dissolved["totintbin"] < 130), "totintbin" - ] = 2 - dissolved.loc[(dissolved["totintbin"] >= 130), "totintbin"] = 3 - - dissolved["empdenbin"] = 1 - dissolved.loc[ - (dissolved["empdenbin"] >= 10) & (dissolved["empdenbin"] < 30), "empdenbin" - ] = 2 - dissolved.loc[(dissolved["empdenbin"] >= 30), "empdenbin"] = 3 - - dissolved["dudenbin"] = 1 - dissolved.loc[ - (dissolved["dudenbin"] >= 5) & (dissolved["dudenbin"] < 10), "dudenbin" - ] = 2 - dissolved.loc[(dissolved["dudenbin"] >= 10), "dudenbin"] = 3 - - pending.append(dissolved) - - pending[0]["cluster_id"] = list(range(n, n + n_zones_algorithm)) - pending[0] = pending[0][[c for c in pending[1].columns if c in pending[0].columns]] - pending[1] = pending[1][[c for c in pending[0].columns if c in pending[1].columns]] - combined = pd.concat(pending, ignore_index=False) + pending[0] = pending[0][ + [c for c in pending[1].columns if c in pending[0].columns] + ] + pending[1] = pending[1][ + [c for c in pending[0].columns if c in pending[1].columns] + ] + combined = pd.concat(pending, ignore_index=False) combined = combined.reset_index(drop=True) return combined @@ -349,6 +367,7 @@ def aggregate_zones_within_districts( cluster_factors_onehot=None, use_xy=True, explicit_agg=(), + explicit_col="mgra", agg_instruction=None, district_col="district27", district_focus=None, @@ -369,10 +388,13 @@ def aggregate_zones_within_districts( ) out = [] for district_n, district_z in agg_by_district.items(): - logger.info(f"combining district {district_n} into {district_z} zones") + district_gdf = mgra_gdf[mgra_gdf[district_col] == district_n] + logger.info( + f"combining district {district_n} from {len(district_gdf)} zones into {district_z} zones" + ) out.append( aggregate_zones( - mgra_gdf[mgra_gdf[district_col] == district_n], + district_gdf, method=method, n_zones=district_z, random_state=random_state + district_n, @@ -380,6 +402,7 @@ def aggregate_zones_within_districts( cluster_factors_onehot=cluster_factors_onehot, use_xy=use_xy, explicit_agg=explicit_agg, + explicit_col=explicit_col, agg_instruction=agg_instruction, ) ) diff --git a/setup.cfg b/setup.cfg index dcad608..107771c 100644 --- a/setup.cfg +++ b/setup.cfg @@ -9,14 +9,14 @@ include_package_data = True python_requires = >=3.7 install_requires = numpy >= 1.19 - pandas - scikit-learn geopandas libpysal - pyproj - plotly networkx - + openmatrix + pandas + plotly + pyproj + scikit-learn [flake8] exclude = From 136c7b9e76265428e7847c79320864327d779e58 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 9 May 2022 13:32:53 -0500 Subject: [PATCH 08/30] test data --- test/data/MGRASHAPE_simplified_10.gpkg | 3 +++ test/data/mgra13_based_input2016.csv.gz | 3 +++ test/data/traffic_skims_AM_mini.omx | 3 +++ test/data/trips_sample.pq | 3 +++ 4 files changed, 12 insertions(+) create mode 100644 test/data/MGRASHAPE_simplified_10.gpkg create mode 100644 test/data/mgra13_based_input2016.csv.gz create mode 100644 test/data/traffic_skims_AM_mini.omx create mode 100644 test/data/trips_sample.pq diff --git a/test/data/MGRASHAPE_simplified_10.gpkg b/test/data/MGRASHAPE_simplified_10.gpkg new file mode 100644 index 0000000..6a07426 --- /dev/null +++ b/test/data/MGRASHAPE_simplified_10.gpkg @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:d6f94e5a8affd64a7d59fe383af9148028dc1878ce0f8c1eafbb5d6f305448a9 +size 10338304 diff --git a/test/data/mgra13_based_input2016.csv.gz b/test/data/mgra13_based_input2016.csv.gz new file mode 100644 index 0000000..13ef17f --- /dev/null +++ b/test/data/mgra13_based_input2016.csv.gz @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:248f5e5bbc9ae3ea2dd39531162dfaf159d617b217f8597f14eecf4343e070f1 +size 1469613 diff --git a/test/data/traffic_skims_AM_mini.omx b/test/data/traffic_skims_AM_mini.omx new file mode 100644 index 0000000..6235363 --- /dev/null +++ b/test/data/traffic_skims_AM_mini.omx @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:2e927e8d96a24679a74f385d8e7c44f5249b37c0793744c53250272a7952e343 +size 77484563 diff --git a/test/data/trips_sample.pq b/test/data/trips_sample.pq new file mode 100644 index 0000000..044fe45 --- /dev/null +++ b/test/data/trips_sample.pq @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:8a8d14dbf4cb54d19aac9d7d018dd9737471b681d80e72b2bd7e727859bd1301 +size 14155356 From 10f1de6b5a192390e29e7b7b585daa59c308f24e Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 9 May 2022 13:33:11 -0500 Subject: [PATCH 09/30] git lfs --- .gitattributes | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 .gitattributes diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..800ab1d --- /dev/null +++ b/.gitattributes @@ -0,0 +1,4 @@ +*.omx filter=lfs diff=lfs merge=lfs -text +*.csv.gz filter=lfs diff=lfs merge=lfs -text +*.gpkg filter=lfs diff=lfs merge=lfs -text +*.pq filter=lfs diff=lfs merge=lfs -text From 08433d07931a70cc25193732d2843c6d0d46c038 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 9 May 2022 13:45:13 -0500 Subject: [PATCH 10/30] git-lfs in env --- environment.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/environment.yaml b/environment.yaml index e150f0d..45f7d82 100644 --- a/environment.yaml +++ b/environment.yaml @@ -10,6 +10,7 @@ dependencies: - numpy>=1.19 - geopandas - git + - git-lfs - jupyter - libpysal - networkx From dbb91f165221151327bbd01fab0ebdadbb7ab109 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 9 May 2022 14:55:11 -0500 Subject: [PATCH 11/30] add pyarrow dependency --- README.md | 11 ++++++++++- environment.yaml | 1 + setup.cfg | 1 + 3 files changed, 12 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 1182060..e3f4e14 100644 --- a/README.md +++ b/README.md @@ -42,7 +42,7 @@ docker build --tag sandag_rsm . ### Jupyter Notebook for Development -On the host machine, run: +On the host machine, on linux or macOS run: ```shell docker run -v $(pwd):/home/mambauser/sandag_rsm -p 8899:8899 \ @@ -50,4 +50,13 @@ docker run -v $(pwd):/home/mambauser/sandag_rsm -p 8899:8899 \ --port 8899 --notebook-dir=/home/mambauser ``` +or in `cwd` on Windows, run: + +```shell +docker run -v %cd%:/home/mambauser/sandag_rsm -p 8899:8899 ^ + -it --rm sandag_rsm jupyter notebook --ip 0.0.0.0 --no-browser --allow-root ^ + --port 8899 --notebook-dir=/home/mambauser +``` + + Then visit `http://127.0.0.1:8899/tree` in your browser. diff --git a/environment.yaml b/environment.yaml index 45f7d82..5799404 100644 --- a/environment.yaml +++ b/environment.yaml @@ -18,6 +18,7 @@ dependencies: - openmatrix - pandas - plotly + - pyarrow - pyproj - requests=2.25.1 - scikit-learn diff --git a/setup.cfg b/setup.cfg index 107771c..ef2a05f 100644 --- a/setup.cfg +++ b/setup.cfg @@ -15,6 +15,7 @@ install_requires = openmatrix pandas plotly + pyarrow pyproj scikit-learn From 89b6d4c342e6a9be894ab56ef4f99190e6c80e57 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 11 May 2022 09:53:58 -0500 Subject: [PATCH 12/30] notes --- README.md | 21 +++++++++++++++++++-- 1 file changed, 19 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index e3f4e14..c4d9257 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,16 @@ # RSM Rapid Strategic Model for the San Diego Association of Governments +## Source Code Access + +The source code for the RSM is stored in this repository. You can access it +via GitHub, or check out the repository using Git. Some larger files (especially +for testing) are stored using [git-lfs](https://git-lfs.github.com/) (large file +storage). This is mostly transparent, but for best results you do need to make +sure that the LFS extension is installed before you clone the repository. Visit +[git-lfs](https://git-lfs.github.com/) for platform-specific instructions. + + ## Installing To install, activate the python or conda environment you want to use, @@ -10,6 +20,13 @@ the cd into the repository directory and run: python -m pip install -e . ``` +This will make the `sandag_rsm` package available, so you can `import sandag_rsm` +to access the functions in this tool, without regard for the current working +directory or pointing the python path to the right place(s). Using the `-e` flag +installs in `editable` mode, so if you make changes or pull updates from GitHub, +those updates will be available to Python without re-installing. + + ## Code Formatting This repo use several tools to ensure a consistent code format throughout the project: @@ -34,7 +51,8 @@ with `git commit --no-verify`. ## Developing with Docker -To build the docker container, change into the repository root and run: +This project uses [Docker](https://www.docker.com/). For development, to build +the docker container, change into the repository root and run: ```shell docker build --tag sandag_rsm . @@ -58,5 +76,4 @@ docker run -v %cd%:/home/mambauser/sandag_rsm -p 8899:8899 ^ --port 8899 --notebook-dir=/home/mambauser ``` - Then visit `http://127.0.0.1:8899/tree` in your browser. From 943d49557284eaa60e55b94b8887963590022422 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Thu, 12 May 2022 18:36:40 -0500 Subject: [PATCH 13/30] script to create test data --- sandag_rsm/data_load/zones.py | 11 ++++-- test/data/prepare_test_data.py | 61 ++++++++++++++++++++++++++++++++++ 2 files changed, 69 insertions(+), 3 deletions(-) create mode 100644 test/data/prepare_test_data.py diff --git a/sandag_rsm/data_load/zones.py b/sandag_rsm/data_load/zones.py index 54d50c1..c6d0a70 100644 --- a/sandag_rsm/data_load/zones.py +++ b/sandag_rsm/data_load/zones.py @@ -22,10 +22,15 @@ def simplify_shapefile( prequantize=False, layername="MGRA", topo=True, + output_filename=None, ): - gpkg_filename = ( - os.path.splitext(shapefilename)[0] + f"_simplified_{simplify_tolerance}.gpkg" - ) + if output_filename is not None: + gpkg_filename = output_filename + else: + gpkg_filename = ( + os.path.splitext(shapefilename)[0] + + f"_simplified_{simplify_tolerance}.gpkg" + ) if os.path.exists(gpkg_filename): gdf = gpd.read_file(gpkg_filename) return geometry_cleanup(gdf) diff --git a/test/data/prepare_test_data.py b/test/data/prepare_test_data.py new file mode 100644 index 0000000..8b27f01 --- /dev/null +++ b/test/data/prepare_test_data.py @@ -0,0 +1,61 @@ +# SANDAG Test Data Generator +# This script create the minimal test data for the SANDAG RSM, from +# an original full-scale data source. + +import gzip +import os +import shutil +from pathlib import Path + +import numpy as np +import openmatrix as omx +import pandas as pd + +from sandag_rsm.data_load.zones import simplify_shapefile + +this_dir = Path(os.path.dirname(__file__)) + + +def mgra_geopackage(shapefilename="MGRASHAPE.zip"): + simplify_shapefile( + shapefilename=shapefilename, + simplify_tolerance=10, + prequantize=False, + layername="MGRA", + topo=True, + output_filename=this_dir / "MGRASHAPE_simplified_10.gpkg", + ) + + +def mini_skim(full_skim="Data/traffic_skims_AM.omx", tablename="AM_SOV_TR_M_TIME"): + output_filename = this_dir / "traffic_skims_AM_mini.omx" + f = omx.open_file(full_skim, mode="r") + out = omx.open_file(str(output_filename), mode="w") + out[tablename] = f[tablename][:].astype(np.float32) + out.create_array( + "/lookup", + "zone_number", + f.root["lookup"]["zone_number"][:], + ) + out.flush() + out.close() + f.close() + + +def gz_copy(in_file, out_file="mgra13_based_input2016.csv.gz"): + with open(in_file, "rb") as f_in: + with gzip.open(out_file, "wb") as f_out: + shutil.copyfileobj(f_in, f_out) + + +def sample_trips(in_file="indivTripData_3.csv.gz", out_file="trips_sample.pq"): + trips = pd.read_csv(in_file) + trips = trips.iloc[::25] + trips.to_parquet(out_file) + + +if __name__ == "__main__": + mgra_geopackage(shapefilename="Data/MGRASHAPE.zip") + mini_skim("Data/traffic_skims_AM.omx", "AM_SOV_TR_M_TIME") + gz_copy("Data/mgra13_based_input2016.csv") + sample_trips("Data/indivTripData_3.csv.gz") From 279bee799ffd5118250348006c436a267b3892f2 Mon Sep 17 00:00:00 2001 From: David Ory Date: Thu, 26 May 2022 13:48:48 -0400 Subject: [PATCH 14/30] Remove GIT LFS, point notebook to `resources` dir --- .gitattributes | 4 --- .gitignore | 2 ++ notebooks/ZoneAggDemo.ipynb | 66 ++++++++++++++++++++++++++----------- 3 files changed, 49 insertions(+), 23 deletions(-) diff --git a/.gitattributes b/.gitattributes index 800ab1d..e69de29 100644 --- a/.gitattributes +++ b/.gitattributes @@ -1,4 +0,0 @@ -*.omx filter=lfs diff=lfs merge=lfs -text -*.csv.gz filter=lfs diff=lfs merge=lfs -text -*.gpkg filter=lfs diff=lfs merge=lfs -text -*.pq filter=lfs diff=lfs merge=lfs -text diff --git a/.gitignore b/.gitignore index 15e9211..f031a21 100644 --- a/.gitignore +++ b/.gitignore @@ -4,3 +4,5 @@ sandag_rsm/__pycache__ .ipynb_checkpoints sandag_rsm.egg-info _version.py +.DS_Store +test/data/* diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index 07fd8de..bd54d79 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -7,20 +7,64 @@ "metadata": {}, "outputs": [], "source": [ + "import requests\n", + "\n", "from sandag_rsm.data_load.zones import load_mgra_data\n", "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, trip_mode_shares_by_taz\n", "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts, merge_zone_data" ] }, + { + "cell_type": "markdown", + "id": "de5861aa", + "metadata": {}, + "source": [ + "## Remote I/O" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1d4896bb", + "metadata": {}, + "outputs": [], + "source": [ + "data_dir = \"../test/data/\"\n", + "resource_url = \"https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/\"" + ] + }, { "cell_type": "code", "execution_count": null, - "id": "b67524d3", + "id": "4f6ca008", "metadata": {}, "outputs": [], "source": [ - "data_dir = \"../test/data\"" + "mgra_filename = \"mgra13_based_input2016.csv.gz\"\n", + "skim_filename = \"traffic_skims_AM_mini.omx\"\n", + "trips_filename = \"trips_sample.pq\"\n", + "download_files_vector = [mgra_filename, skim_filename, trips_filename]" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "691bbb9b", + "metadata": {}, + "outputs": [], + "source": [ + "for download_file in download_files_vector:\n", + " r = requests.get((resource_url+download_file), allow_redirects=True)\n", + " open((data_dir+download_file), 'wb').write(r.content)" + ] + }, + { + "cell_type": "markdown", + "id": "fe41ea9a", + "metadata": {}, + "source": [ + "## Demo" ] }, { @@ -223,14 +267,6 @@ "viewer2(edges=kmeans1, colors=d1, color_col='empden')" ] }, - { - "cell_type": "code", - "execution_count": null, - "id": "1a760ac4", - "metadata": {}, - "outputs": [], - "source": [] - }, { "cell_type": "code", "execution_count": null, @@ -307,14 +343,6 @@ "source": [ "viewer2(edges=agglom3full, colors=agglom3full, color_col='popden')" ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "ba9b4203", - "metadata": {}, - "outputs": [], - "source": [] } ], "metadata": { @@ -333,7 +361,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.9.10" + "version": "3.9.12" }, "toc": { "base_numbering": 1, From 8652779938681f6bf28b12aaed25c795a35bc3fc Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 8 Jun 2022 10:47:41 -0500 Subject: [PATCH 15/30] remove big files these are now in `resources` --- test/data/mgra13_based_input2016.csv.gz | 3 --- test/data/traffic_skims_AM_mini.omx | 3 --- test/data/trips_sample.pq | 3 --- 3 files changed, 9 deletions(-) delete mode 100644 test/data/mgra13_based_input2016.csv.gz delete mode 100644 test/data/traffic_skims_AM_mini.omx delete mode 100644 test/data/trips_sample.pq diff --git a/test/data/mgra13_based_input2016.csv.gz b/test/data/mgra13_based_input2016.csv.gz deleted file mode 100644 index 13ef17f..0000000 --- a/test/data/mgra13_based_input2016.csv.gz +++ /dev/null @@ -1,3 +0,0 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:248f5e5bbc9ae3ea2dd39531162dfaf159d617b217f8597f14eecf4343e070f1 -size 1469613 diff --git a/test/data/traffic_skims_AM_mini.omx b/test/data/traffic_skims_AM_mini.omx deleted file mode 100644 index 6235363..0000000 --- a/test/data/traffic_skims_AM_mini.omx +++ /dev/null @@ -1,3 +0,0 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:2e927e8d96a24679a74f385d8e7c44f5249b37c0793744c53250272a7952e343 -size 77484563 diff --git a/test/data/trips_sample.pq b/test/data/trips_sample.pq deleted file mode 100644 index 044fe45..0000000 --- a/test/data/trips_sample.pq +++ /dev/null @@ -1,3 +0,0 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:8a8d14dbf4cb54d19aac9d7d018dd9737471b681d80e72b2bd7e727859bd1301 -size 14155356 From 2d1df97fd349680b244b9f6da787369e409a556d Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 8 Jun 2022 10:49:33 -0500 Subject: [PATCH 16/30] remove big file --- test/data/MGRASHAPE_simplified_10.gpkg | 3 --- 1 file changed, 3 deletions(-) delete mode 100644 test/data/MGRASHAPE_simplified_10.gpkg diff --git a/test/data/MGRASHAPE_simplified_10.gpkg b/test/data/MGRASHAPE_simplified_10.gpkg deleted file mode 100644 index 6a07426..0000000 --- a/test/data/MGRASHAPE_simplified_10.gpkg +++ /dev/null @@ -1,3 +0,0 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:d6f94e5a8affd64a7d59fe383af9148028dc1878ce0f8c1eafbb5d6f305448a9 -size 10338304 From cb305fd5256bc467d84cba3362057e0df746eaae Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Tue, 21 Jun 2022 16:59:08 -0500 Subject: [PATCH 17/30] crosswalks and centroids --- sandag_rsm/zone_agg.py | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index 8f9356a..44e0afe 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -3,6 +3,7 @@ from numbers import Number from statistics import mode +import geopandas as gpd import networkx as nx import numpy as np import pandas as pd @@ -409,6 +410,36 @@ def aggregate_zones_within_districts( return pd.concat(out).reset_index(drop=True) +def make_crosswalk(new_zones, old_zones, new_index="cluster_id", old_index=None): + if new_index is not None and ( + new_index in new_zones.columns or not isinstance(new_index, str) + ): + new_zones = new_zones.set_index(new_index) + if old_index is not None and ( + old_index in old_zones.columns or not isinstance(old_index, str) + ): + old_zones = old_zones.set_index(old_index) + crosswalk = new_zones[["geometry"]].sjoin( + gpd.GeoDataFrame( + geometry=old_zones.representative_point(), + index=old_zones.index, + ).to_crs(new_zones.crs), + how="right", + predicate="contains", + ) + crosswalk = crosswalk["index_left"].astype(int).rename(new_index).to_frame() + if old_zones.index.name: + crosswalk = crosswalk.rename_axis(index=old_zones.index.name) + return crosswalk.reset_index() + + +def mark_centroids(gdf, crs="NAD 1983 StatePlane California VI FIPS 0406 Feet"): + c = gdf.to_crs("NAD 1983 StatePlane California VI FIPS 0406 Feet").centroid + gdf["centroid_x"] = c.x.astype(int) + gdf["centroid_y"] = c.y.astype(int) + return gdf + + def viewer( gdf, *, From 46edf957be87bd303c8897be5b8262c18bca7d55 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Tue, 21 Jun 2022 17:03:42 -0500 Subject: [PATCH 18/30] crosswalks and centroid mapping in notebook --- notebooks/ZoneAggDemo.ipynb | 58 +++++++++++++++++++++++++++++++++++-- 1 file changed, 56 insertions(+), 2 deletions(-) diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index bd54d79..4d78d67 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -8,11 +8,15 @@ "outputs": [], "source": [ "import requests\n", + "import os\n", "\n", "from sandag_rsm.data_load.zones import load_mgra_data\n", - "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, trip_mode_shares_by_taz\n", + "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, \\\n", + " trip_mode_shares_by_taz\n", "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", - "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts, merge_zone_data" + "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, \\\n", + " aggregate_zones_within_districts, merge_zone_data, make_crosswalk, \\\n", + " mark_centroids" ] }, { @@ -324,6 +328,56 @@ ")" ] }, + { + "cell_type": "code", + "execution_count": null, + "id": "b1ac6e48", + "metadata": {}, + "outputs": [], + "source": [ + "taz_crosswalk = make_crosswalk(agglom3full, tazs, old_index='taz').sort_values('taz')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "08c914b6", + "metadata": {}, + "outputs": [], + "source": [ + "mgra_crosswalk = make_crosswalk(agglom3full, mgra, old_index='MGRA').sort_values('MGRA')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "219babb2", + "metadata": {}, + "outputs": [], + "source": [ + "agglom3full = mark_centroids(agglom3full)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4a3b908f", + "metadata": {}, + "outputs": [], + "source": [ + "mgra_crosswalk.to_csv(\"mgra_crosswalk.csv\", index=False)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a5effc15", + "metadata": {}, + "outputs": [], + "source": [ + "taz_crosswalk.to_csv(\"taz_crosswalk.csv\", index=False)" + ] + }, { "cell_type": "code", "execution_count": null, From c4e4c090712e2370a89daf80e57b15785fa2ea2f Mon Sep 17 00:00:00 2001 From: elias-sanz Date: Wed, 29 Jun 2022 16:07:28 -0700 Subject: [PATCH 19/30] Translate Demand, initial commit --- notebooks/TranslateDemand.ipynb | 248 ++++++++++++++++++++++++++++++++ sandag_rsm/emme.py | 37 +++++ sandag_rsm/translate.py | 46 ++++++ 3 files changed, 331 insertions(+) create mode 100644 notebooks/TranslateDemand.ipynb create mode 100644 sandag_rsm/emme.py create mode 100644 sandag_rsm/translate.py diff --git a/notebooks/TranslateDemand.ipynb b/notebooks/TranslateDemand.ipynb new file mode 100644 index 0000000..1165ef3 --- /dev/null +++ b/notebooks/TranslateDemand.ipynb @@ -0,0 +1,248 @@ +{ + "cells": [ + { + "cell_type": "code", + "execution_count": null, + "id": "4fc00298", + "metadata": {}, + "outputs": [], + "source": [ + "import requests\n", + "\n", + "from sandag_rsm.data_load.zones import load_mgra_data\n", + "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, trip_mode_shares_by_taz\n", + "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", + "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts, merge_zone_data\n", + "\n", + "from sandag_rsm.translate import translate_demand\n", + "import os\n", + "import numpy as np\n", + "import pandas as pd\n", + "import geopandas as gpd" + ] + }, + { + "cell_type": "markdown", + "id": "de5861aa", + "metadata": {}, + "source": [ + "## Remote I/O" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1d4896bb", + "metadata": {}, + "outputs": [], + "source": [ + "data_dir = \"../test/data/\"\n", + "export_dir = \"../test/export/\"\n", + "resource_url = \"https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/\"" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4f6ca008", + "metadata": {}, + "outputs": [], + "source": [ + "skim_filename = \"traffic_skims_AM_mini.omx\"\n", + "matrix_name = 'AM_SOV_TR_M_TIME'" + ] + }, + { + "cell_type": "raw", + "id": "9323795c-fa59-4669-bc08-56b5b4732066", + "metadata": {}, + "source": [ + "for download_file in download_files_vector:\n", + " r = requests.get((resource_url+download_file), allow_redirects=True)\n", + " open((data_dir+download_file), 'wb').write(r.content)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "2c158043-e9cf-40c5-a8c3-c74ef9eccc1d", + "metadata": {}, + "outputs": [], + "source": [ + "## Create Dummy 'zone_to_cluster' Dictionary\n", + "zone_to_cluster = {i:1+i%100 for i in range(4996)}" + ] + }, + { + "cell_type": "raw", + "id": "ad52614e-4d7d-42f9-8c4a-6239457dd0ff", + "metadata": {}, + "source": [ + "## Use Prototype Crosswalks for Testing\n", + "zone_to_cluster = pd.read_csv(os.path.join(data_dir, 'taz_crosswalk.csv')).sort_values('taz')\n", + "zone_to_cluster['taz'] = zone_to_cluster['taz'] - 1\n", + "zone_to_cluster = dict(zip(zone_to_cluster['taz'], zone_to_cluster['cluster_id']))" + ] + }, + { + "cell_type": "markdown", + "id": "fe41ea9a", + "metadata": {}, + "source": [ + "## Demo" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "458ad160-33cf-4149-94ca-b436c2a5aafb", + "metadata": {}, + "outputs": [], + "source": [ + "matrix_load, matrix = translate_demand(skim_filename, matrix_name, zone_to_cluster, data_dir, export_dir)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "eb4bcad1-fe01-405c-8da7-96ac040a10d6", + "metadata": {}, + "outputs": [], + "source": [ + "print('Loaded Matrix')\n", + "print(f'total sum: {matrix_load.sum().sum():,.0f}\\n')\n", + "print(matrix_load.sum().describe().apply(\"{0:,.2f}\".format))" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "692ecce3-946b-4ca1-ac01-b8610991b8bc", + "metadata": {}, + "outputs": [], + "source": [ + "print('Aggregated Matrix')\n", + "print(f'total sum: {matrix.sum().sum():,.0f}\\n')\n", + "print(matrix.sum().describe().apply(\"{0:,.2f}\".format))" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "c3030557-03f4-402c-8da2-306c0b4f268c", + "metadata": {}, + "outputs": [], + "source": [ + "matrix_load" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "650baf9b-569f-424b-a703-5cb7c75110d4", + "metadata": {}, + "outputs": [], + "source": [ + "matrix" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a9552989-8251-445f-a3b1-4070d66c06fa", + "metadata": {}, + "outputs": [], + "source": [] + }, + { + "cell_type": "markdown", + "id": "a639d2b1-c5fd-461d-8a58-6fc26b6987a7", + "metadata": {}, + "source": [ + "## Visualize" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b9f93146-e68b-4d62-a38f-4b655335fcdb", + "metadata": {}, + "outputs": [], + "source": [ + "use_cluster = 731\n", + "use_col = f'{matrix_name}_{use_cluster}'\n", + "\n", + "plot_col = matrix[use_cluster].reset_index()\n", + "plot_col.columns = ['cluster_id', use_col]" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "5ac6939e-2dac-4ffa-9007-1243ec608e32", + "metadata": {}, + "outputs": [], + "source": [ + "mgra = gpd.read_file(os.path.join(data_dir, 'MGRASHAPE.zip'))\n", + "mgra_xref = pd.read_csv(os.path.join(data_dir, 'mgra_crosswalk.csv'))\n", + "\n", + "clusters = pd.merge(mgra, mgra_xref, on='MGRA', how='left')\n", + "clusters = clusters[['cluster_id', 'geometry']].dissolve(by='cluster_id').reset_index()\n", + "\n", + "clusters = pd.merge(clusters, plot_col, on='cluster_id', how='left')" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a93bd34a", + "metadata": {}, + "outputs": [], + "source": [ + "viewer(clusters, color=use_col, marker_line_width=0)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "da9761de-2fec-4ba5-a5c1-f0f276398195", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.9.10" + }, + "toc": { + "base_numbering": 1, + "nav_menu": {}, + "number_sections": false, + "sideBar": true, + "skip_h1_title": false, + "title_cell": "Table of Contents", + "title_sidebar": "Contents", + "toc_cell": false, + "toc_position": {}, + "toc_section_display": true, + "toc_window_display": false + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/sandag_rsm/emme.py b/sandag_rsm/emme.py new file mode 100644 index 0000000..d9124d0 --- /dev/null +++ b/sandag_rsm/emme.py @@ -0,0 +1,37 @@ +import os +import inro.emme.database.emmebank as _eb +import inro.modeller as _m + + +def load_matrix_emme( + skims_file, emme_path +): +""" +SAMPLE CODE +_m = inro.modeller +NAMESPACE = "inro.emme.data.matrix.export_to_omx" +export_to_omx = _m.Modeller().tool(NAMESPACE) +emmebank_dir = os.path.dirname(_m.Modeller().emmebank.path) +omx_file = os.path.join(emmebank_dir, "exported_demand_matrices.omx") +export_to_omx(matrices=["mf1", "mf2"], + export_file=omx_file, + append_to_file=False) +""" + + + +def save_matrix_emme( + emme_path, matrix_name +): +""" +SAMPLE CODE +_m = inro.modeller +NAMESPACE = "inro.emme.data.matrix.create_matrix" +create_matrix = _m.Modeller().tool(NAMESPACE) +new_mat = create_matrix(matrix_id="mf13", + matrix_name="diftrt", + matrix_description="transit time difference", + default_value=0) + + + diff --git a/sandag_rsm/translate.py b/sandag_rsm/translate.py new file mode 100644 index 0000000..3a5f298 --- /dev/null +++ b/sandag_rsm/translate.py @@ -0,0 +1,46 @@ +import os +import logging +import pandas as pd +from pathlib import Path + +from .data_load.skims import open_skims + +logger = logging.getLogger(__name__) + + +def translate_demand( + skims_file, matrix_name, agg_map, data_dir=None, export_dir=None +): + matrix_load = load_matrix(skims_file, matrix_name, data_dir) + matrix = aggregate_matrix(matrix_load, agg_map) + export_matrix(matrix, matrix_name, export_dir) + + return matrix_load, matrix + + +def load_matrix( + skims_file, matrix_name, data_dir +): + logger.info('Loading Matrix: matrix_name') + # if isinstance(skims_file, emme_path): + # matrix_load = load_matrix_emme(skims_file, emme_path=emme_path) + if isinstance(skims_file, (str, Path)): + matrix_load = open_skims(skims_file, data_dir=data_dir) + + return pd.DataFrame(matrix_load[matrix_name]) + +def aggregate_matrix( + matrix, agg_map +): + logger.info('\tAggregating Matrix') + matrix = matrix.rename(columns=(agg_map)) + matrix = matrix.rename(index=(agg_map)) + + return matrix.stack().groupby(level=[0,1]).sum().unstack() + +def export_matrix( + matrix, matrix_name, export_dir +): + logger.info('\tExporting Matrix') + matrix.to_csv(os.path.join(export_dir, f'{matrix_name}.csv')) + # save_matrix_emme(emme_path, matrix_name) From cb98e9c65e41b72457a04c95d9e61c4cbac42943 Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Thu, 30 Jun 2022 09:44:31 -0400 Subject: [PATCH 20/30] sandag model - 14.2.2 --- sandag_abm/src/main/emme/init_emme_project.py | 97 + .../src/main/emme/python_virtualenv.pth | 3 + sandag_abm/src/main/emme/solutions.mtbx | Bin 0 -> 118784 bytes .../main/emme/solutions_unconsolidated.mtbx | Bin 0 -> 24576 bytes .../assignment/build_transit_scenario.py | 679 +++ .../toolbox/assignment/traffic_assignment.py | 1087 ++++ .../toolbox/assignment/transit_assignment.py | 785 +++ .../assignment/transit_select_analysis.py | 217 + .../src/main/emme/toolbox/build_toolbox.py | 411 ++ .../diagnostic/mode_choice_diagnostic.py | 669 +++ .../export/export_data_loader_matrices.py | 302 + .../export/export_data_loader_network.py | 1203 ++++ .../export/export_for_commercial_vehicle.py | 158 + .../toolbox/export/export_for_transponder.py | 374 ++ .../export/export_tap_adjacent_lines.py | 122 + .../toolbox/export/export_traffic_skims.py | 95 + .../toolbox/export/export_transit_skims.py | 108 + .../emme/toolbox/import/import_auto_demand.py | 516 ++ .../emme/toolbox/import/import_network.py | 2250 ++++++++ .../emme/toolbox/import/import_seed_demand.py | 193 + .../toolbox/import/import_transit_demand.py | 230 + .../main/emme/toolbox/import/input_checker.py | 767 +++ .../src/main/emme/toolbox/import/run4Ds.py | 412 ++ .../toolbox/initialize/initialize_matrices.py | 426 ++ .../initialize/initialize_transit_database.py | 168 + .../src/main/emme/toolbox/master_run.py | 1202 ++++ .../model/commercial_vehicle/distribution.py | 197 + .../model/commercial_vehicle/generation.py | 187 + .../run_commercial_vehicle_model.py | 122 + .../model/commercial_vehicle/time_of_day.py | 99 + .../commercial_vehicle/toll_diversion.py | 119 + .../emme/toolbox/model/external_external.py | 190 + .../emme/toolbox/model/external_internal.py | 331 ++ .../emme/toolbox/model/truck/distribution.py | 288 + .../emme/toolbox/model/truck/generation.py | 443 ++ .../toolbox/model/truck/run_truck_model.py | 130 + .../src/main/emme/toolbox/utilities/demand.py | 297 + .../emme/toolbox/utilities/file_manager.py | 370 ++ .../main/emme/toolbox/utilities/general.py | 386 ++ .../main/emme/toolbox/utilities/omxwrapper.py | 91 + .../main/emme/toolbox/utilities/properties.py | 599 ++ .../emme/toolbox/utilities/run_summary.py | 326 ++ .../emme/toolbox/validation/validation.py | 290 + sandag_abm/src/main/gisdk/SandagCommon.rsc | 405 ++ sandag_abm/src/main/gisdk/TC2OMX.rsc | 69 + sandag_abm/src/main/gisdk/TruckModel.rsc | 1504 +++++ sandag_abm/src/main/gisdk/Utilities.rsc | 568 ++ sandag_abm/src/main/gisdk/commVehDist.rsc | 106 + .../src/main/gisdk/commVehDiversion.rsc | 71 + sandag_abm/src/main/gisdk/commVehGen.rsc | 184 + sandag_abm/src/main/gisdk/commVehTOD.rsc | 122 + .../src/main/gisdk/create_LUZ_Skims.rsc | 254 + sandag_abm/src/main/gisdk/createhwynet.rsc | 2509 ++++++++ sandag_abm/src/main/gisdk/createtodtables.rsc | 916 +++ sandag_abm/src/main/gisdk/createtrnroutes.rsc | 655 +++ sandag_abm/src/main/gisdk/dbox.rsc | 256 + sandag_abm/src/main/gisdk/exportTCData.rsc | 40 + .../src/main/gisdk/externalInternal.rsc | 356 ++ sandag_abm/src/main/gisdk/gui_generic.rsc | 45 + sandag_abm/src/main/gisdk/hwyassign.rsc | 896 +++ sandag_abm/src/main/gisdk/hwyskim.rsc | 1257 ++++ .../main/gisdk/matrixPrecisionReduction.rsc | 51 + sandag_abm/src/main/gisdk/parameter.rsc | 8 + sandag_abm/src/main/gisdk/sandag_abm.lst | 22 + .../src/main/gisdk/sandag_abm_generic.lst | 23 + .../src/main/gisdk/sandag_abm_master.rsc | 411 ++ .../src/main/gisdk/sandag_abm_outputs.rsc | 293 + sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc | 223 + sandag_abm/src/main/gisdk/sellink_volume.rsc | 85 + sandag_abm/src/main/gisdk/trnassign.rsc | 177 + sandag_abm/src/main/gisdk/trnskim.rsc | 1038 ++++ .../accessibilities/AccessibilitiesDMU.java | 236 + .../accessibilities/AccessibilitiesTable.java | 407 ++ .../AutoAndNonMotorizedSkimsCalculator.java | 1024 ++++ .../abm/accessibilities/AutoSkimsDMU.java | 82 + .../AutoTazSkimsCalculator.java | 206 + .../BestTransitPathCalculator.java | 1548 +++++ .../accessibilities/BuildAccessibilities.java | 1534 +++++ .../accessibilities/DcUtilitiesTaskJppf.java | 880 +++ .../DriveTransitWalkSkimsCalculator.java | 313 + .../MandatoryAccessibilitiesCalculator.java | 593 ++ .../MandatoryAccessibilitiesDMU.java | 206 + .../accessibilities/McLogsumsAppender.java | 980 ++++ ...andatoryDcEstimationMcLogsumsAppender.java | 368 ++ ...ndatoryTodEstimationMcLogsumsAppender.java | 277 + .../accessibilities/NonTransitUtilities.java | 641 +++ .../ParkLocationEstimationAppender.java | 297 + .../abm/accessibilities/SkimsAppender.java | 788 +++ ...opLocationEstimationMcLogsumsAppender.java | 331 ++ .../StopLocationSampleCalculator.java | 502 ++ .../StoredTransitSkimData.java | 51 + .../accessibilities/StoredUtilityData.java | 133 + ...SubtourTodEstimationMcLogsumsAppender.java | 266 + .../abm/accessibilities/TransitPath.java | 45 + .../accessibilities/TripSkimsAppender.java | 569 ++ .../VisitorTourLocationChoiceAppender.java | 367 ++ .../WalkTransitDriveSkimsCalculator.java | 295 + .../WalkTransitWalkSkimsCalculator.java | 299 + .../accessibilities/XBorderSkimsAppender.java | 518 ++ .../abm/active/AbstractNetworkFactory.java | 51 + ...ctPathChoiceEdgeAssignmentApplication.java | 217 + ...ractPathChoiceLogsumMatrixApplication.java | 422 ++ .../active/AbstractShortestPathResultSet.java | 20 + .../active/BasicShortestPathResultSet.java | 59 + .../org/sandag/abm/active/BinarySearch.java | 35 + .../CompositeShortestPathResultSet.java | 57 + .../active/DestinationNotFoundException.java | 10 + .../main/java/org/sandag/abm/active/Edge.java | 9 + .../org/sandag/abm/active/EdgeEvaluator.java | 6 + .../abm/active/IntrazonalCalculation.java | 35 + .../abm/active/IntrazonalCalculations.java | 365 ++ .../ModifiableShortestPathResultSet.java | 11 + .../java/org/sandag/abm/active/Network.java | 163 + .../org/sandag/abm/active/NetworkFactory.java | 35 + .../main/java/org/sandag/abm/active/Node.java | 7 + .../java/org/sandag/abm/active/NodePair.java | 50 + .../active/ParallelSingleSourceDijkstra.java | 230 + .../main/java/org/sandag/abm/active/Path.java | 82 + .../abm/active/PathAlternativeList.java | 177 + ...lternativeListGenerationConfiguration.java | 52 + .../abm/active/PathAlternativeListWriter.java | 59 + .../active/RepeatedSingleSourceDijkstra.java | 156 + .../sandag/abm/active/ShortestPathResult.java | 31 + .../abm/active/ShortestPathResultSet.java | 13 + .../abm/active/ShortestPathStrategy.java | 11 + .../org/sandag/abm/active/SimpleEdge.java | 50 + .../org/sandag/abm/active/SimpleNetwork.java | 141 + .../org/sandag/abm/active/SimpleNode.java | 45 + .../sandag/abm/active/SimpleTraversal.java | 46 + .../java/org/sandag/abm/active/Traversal.java | Bin 0 -> 131 bytes .../sandag/abm/active/TraversalEvaluator.java | 6 + .../sandag/BikeAssignmentTripReader.java | 216 + .../abm/active/sandag/PropertyParser.java | 99 + .../abm/active/sandag/SandagBikeEdge.java | 19 + ...lternativeListGenerationConfiguration.java | 36 + .../sandag/SandagBikeNetworkFactory.java | 581 ++ .../abm/active/sandag/SandagBikeNode.java | 16 + ...lternativeListGenerationConfiguration.java | 430 ++ .../sandag/SandagBikePathAlternatives.java | 207 + .../sandag/SandagBikePathChoiceDmu.java | 300 + ...kePathChoiceEdgeAssignmentApplication.java | 281 + ...BikePathChoiceLogsumMatrixApplication.java | 317 + .../sandag/SandagBikePathChoiceModel.java | 134 + ...lternativeListGenerationConfiguration.java | 35 + .../active/sandag/SandagBikeTraversal.java | 22 + ...lternativeListGenerationConfiguration.java | 62 + ...lternativeListGenerationConfiguration.java | 62 + ...lternativeListGenerationConfiguration.java | 310 + ...WalkPathChoiceLogsumMatrixApplication.java | 232 + ...lternativeListGenerationConfiguration.java | 62 + .../sandag/abm/active/sandag/TurnType.java | 35 + .../abm/airport/AirportDestChoiceModel.java | 581 ++ .../sandag/abm/airport/AirportDmuFactory.java | 35 + .../abm/airport/AirportDmuFactoryIf.java | 11 + .../abm/airport/AirportModeChoiceModel.java | 378 ++ .../org/sandag/abm/airport/AirportModel.java | 233 + .../sandag/abm/airport/AirportModelDMU.java | 478 ++ .../abm/airport/AirportModelStructure.java | 239 + .../org/sandag/abm/airport/AirportParty.java | 321 ++ .../abm/airport/AirportPartyManager.java | 410 ++ .../sandag/abm/airport/AirportTripTables.java | 707 +++ .../application/SandagAppendMcLogsumDMU.java | 615 ++ .../SandagAtWorkSubtourFrequencyDMU.java | 64 + .../SandagAutoOwnershipChoiceDMU.java | 104 + ...dagCoordinatedDailyActivityPatternDMU.java | 176 + .../SandagCreateTripGenerationFiles.java | 1055 ++++ .../application/SandagCtrampApplication.java | 23 + .../application/SandagCtrampDmuFactory.java | 170 + .../abm/application/SandagDcSoaDMU.java | 89 + .../abm/application/SandagDestChoiceDMU.java | 161 + .../SandagDestChoiceSoaTwoStageModelDMU.java | 158 + ...estChoiceSoaTwoStageTazDistUtilityDMU.java | 66 + .../SandagHouseholdDataManager.java | 624 ++ .../SandagHouseholdDataManager2.java | 788 +++ ...agIndividualMandatoryTourFrequencyDMU.java | 101 + ...ndividualNonMandatoryTourFrequencyDMU.java | 272 + .../SandagInternalExternalTripChoiceDMU.java | 50 + .../application/SandagJointTourModelsDMU.java | 137 + .../abm/application/SandagMGRAtoPNR.java | 287 + .../SandagMicromobilityChoiceDMU.java | 52 + .../abm/application/SandagModelStructure.java | 1299 +++++ .../application/SandagParkingChoiceDMU.java | 94 + .../SandagParkingProvisionChoiceDMU.java | 83 + .../SandagSamplePopulationGenerator.java | 101 + .../application/SandagStopFrequencyDMU.java | 255 + .../application/SandagStopLocationDMU.java | 132 + .../abm/application/SandagSummitFile.java | 820 +++ .../abm/application/SandagTelecommuteDMU.java | 67 + .../sandag/abm/application/SandagTestSOA.java | 216 + .../abm/application/SandagTourBasedModel.java | 358 ++ ...SandagTourDepartureTimeAndDurationDMU.java | 345 ++ .../application/SandagTourModeChoiceDMU.java | 502 ++ .../SandagTransponderChoiceDMU.java | 56 + .../application/SandagTripModeChoiceDMU.java | 572 ++ .../abm/application/SandagTripTables.java | 904 +++ .../crossborder/CrossBorderDmuFactory.java | 51 + .../crossborder/CrossBorderDmuFactoryIf.java | 17 + .../abm/crossborder/CrossBorderModel.java | 525 ++ .../CrossBorderModelStructure.java | 106 + .../CrossBorderStationDestChoiceDMU.java | 405 ++ .../CrossBorderStationDestChoiceModel.java | 818 +++ .../abm/crossborder/CrossBorderStop.java | 187 + .../CrossBorderStopFrequencyModel.java | 316 + .../CrossBorderStopLocationChoiceDMU.java | 406 ++ .../CrossBorderStopLocationChoiceModel.java | 536 ++ .../CrossBorderStopPurposeModel.java | 233 + .../CrossBorderStopTimeOfDayChoiceModel.java | 365 ++ .../abm/crossborder/CrossBorderTour.java | 367 ++ .../crossborder/CrossBorderTourManager.java | 358 ++ .../CrossBorderTourModeChoiceDMU.java | 406 ++ .../CrossBorderTourModeChoiceModel.java | 590 ++ .../CrossBorderTourTimeOfDayChoiceModel.java | 188 + .../abm/crossborder/CrossBorderTrip.java | 502 ++ .../CrossBorderTripModeChoiceDMU.java | 762 +++ .../CrossBorderTripModeChoiceModel.java | 369 ++ .../crossborder/CrossBorderTripTables.java | 704 +++ .../abm/ctramp/AtWorkSubtourFrequencyDMU.java | 204 + .../abm/ctramp/AutoOwnershipChoiceDMU.java | 267 + .../org/sandag/abm/ctramp/BikeLogsum.java | 259 + .../sandag/abm/ctramp/BikeLogsumSegment.java | 117 + .../sandag/abm/ctramp/ConnectionHelper.java | 55 + .../CoordinatedDailyActivityPatternDMU.java | 442 ++ .../sandag/abm/ctramp/CtrampApplication.java | 2481 ++++++++ .../sandag/abm/ctramp/CtrampDmuFactoryIf.java | 52 + .../org/sandag/abm/ctramp/DAOException.java | 23 + .../java/org/sandag/abm/ctramp/DcSoaDMU.java | 215 + .../org/sandag/abm/ctramp/Definitions.java | 22 + .../org/sandag/abm/ctramp/DestChoiceDMU.java | 394 ++ .../abm/ctramp/DestChoiceModelManager.java | 1082 ++++ .../org/sandag/abm/ctramp/DestChoiceSize.java | 965 ++++ .../abm/ctramp/DestChoiceTwoStageModel.java | 625 ++ .../ctramp/DestChoiceTwoStageModelDMU.java | 408 ++ ...iceTwoStageSoaProbabilitiesCalculator.java | 159 + ...hoiceTwoStageSoaTazDistanceUtilityDMU.java | 156 + .../DestinationSampleOfAlternativesModel.java | 612 ++ .../java/org/sandag/abm/ctramp/Household.java | 1847 ++++++ .../HouseholdAtWorkSubtourFrequencyModel.java | 334 ++ .../ctramp/HouseholdAutoOwnershipModel.java | 343 ++ .../ctramp/HouseholdChoiceModelRunner.java | 254 + .../abm/ctramp/HouseholdChoiceModels.java | 980 ++++ .../ctramp/HouseholdChoiceModelsManager.java | 682 +++ .../ctramp/HouseholdChoiceModelsTaskJppf.java | 207 + ...dCoordinatedDailyActivityPatternModel.java | 1423 +++++ .../abm/ctramp/HouseholdDataManager.java | 2039 +++++++ .../abm/ctramp/HouseholdDataManagerIf.java | 137 + .../abm/ctramp/HouseholdDataManagerRmi.java | 373 ++ .../abm/ctramp/HouseholdDataWriter.java | 1871 ++++++ ...MandatoryTourDepartureAndDurationTime.java | 1672 ++++++ ...IndividualMandatoryTourFrequencyModel.java | 414 ++ ...ividualNonMandatoryTourFrequencyModel.java | 823 +++ .../sandag/abm/ctramp/HouseholdValidator.java | 82 + .../IndividualMandatoryTourFrequencyDMU.java | 301 + ...ndividualNonMandatoryTourFrequencyDMU.java | 773 +++ .../ctramp/IntermediateStopChoiceModels.java | 5075 +++++++++++++++++ .../ctramp/InternalExternalTripChoiceDMU.java | 113 + .../InternalExternalTripChoiceModel.java | 120 + .../sandag/abm/ctramp/JointTourModels.java | 916 +++ .../sandag/abm/ctramp/JointTourModelsDMU.java | 310 + .../abm/ctramp/MandatoryDestChoiceModel.java | 885 +++ .../sandag/abm/ctramp/MatrixDataServer.java | 209 + .../abm/ctramp/MatrixDataServerRmi.java | 98 + .../abm/ctramp/McLogsumsCalculator.java | 754 +++ .../abm/ctramp/MicromobilityChoiceDMU.java | 119 + .../abm/ctramp/MicromobilityChoiceModel.java | 449 ++ .../org/sandag/abm/ctramp/ModelStructure.java | 600 ++ .../java/org/sandag/abm/ctramp/MyLogit.java | 115 + .../ctramp/NonMandatoryDestChoiceModel.java | 1527 +++++ ...MandatoryTourDepartureAndDurationTime.java | 1448 +++++ .../sandag/abm/ctramp/ParkingChoiceDMU.java | 332 ++ .../abm/ctramp/ParkingProvisionChoiceDMU.java | 256 + .../abm/ctramp/ParkingProvisionModel.java | 216 + .../java/org/sandag/abm/ctramp/Person.java | 2032 +++++++ .../ctramp/SchoolEscortChauffeurResult.java | 67 + .../abm/ctramp/SchoolEscortChildResult.java | 68 + .../abm/ctramp/SchoolEscortingBundle.java | 499 ++ .../sandag/abm/ctramp/SchoolEscortingDmu.java | 1716 ++++++ .../abm/ctramp/SchoolEscortingModel.java | 2154 +++++++ .../abm/ctramp/SchoolLocationChoiceModel.java | 603 ++ .../ctramp/SchoolLocationChoiceTaskJppf.java | 238 + .../SchoolLocationChoiceTaskJppfNew.java | 252 + .../abm/ctramp/SegmentedSparseMatrix.java | 26 + .../java/org/sandag/abm/ctramp/SoaDMU.java | 10 + .../org/sandag/abm/ctramp/SqliteService.java | 118 + .../main/java/org/sandag/abm/ctramp/Stop.java | 304 + .../org/sandag/abm/ctramp/StopDCSoaDMU.java | 167 + .../ctramp/StopDepartArrivePeriodModel.java | 144 + .../sandag/abm/ctramp/StopDestChoiceSize.java | 214 + .../sandag/abm/ctramp/StopFrequencyDMU.java | 428 ++ .../sandag/abm/ctramp/StopFrequencyModel.java | 811 +++ .../sandag/abm/ctramp/StopLocationDMU.java | 411 ++ .../SubtourDepartureAndDurationTime.java | 956 ++++ .../abm/ctramp/SubtourDestChoiceModel.java | 1185 ++++ .../ctramp/TNCAndTaxiWaitTimeCalculator.java | 264 + .../org/sandag/abm/ctramp/TazDataHandler.java | 870 +++ .../sandag/abm/ctramp/TazDataHandlerRmi.java | 297 + .../java/org/sandag/abm/ctramp/TazDataIf.java | 152 + .../org/sandag/abm/ctramp/TelecommuteDMU.java | 172 + .../sandag/abm/ctramp/TelecommuteModel.java | 144 + .../ctramp/TimeCoefficientDistributions.java | 284 + .../java/org/sandag/abm/ctramp/TimeDMU.java | 105 + .../main/java/org/sandag/abm/ctramp/Tour.java | 875 +++ .../TourDepartureTimeAndDurationDMU.java | 864 +++ .../sandag/abm/ctramp/TourModeChoiceDMU.java | 523 ++ .../abm/ctramp/TourModeChoiceModel.java | 762 +++ .../ctramp/TourVehicleTypeChoiceModel.java | 189 + .../abm/ctramp/TransponderChoiceDMU.java | 158 + .../abm/ctramp/TransponderChoiceModel.java | 131 + .../sandag/abm/ctramp/TripModeChoiceDMU.java | 846 +++ .../UsualWorkSchoolLocationChoiceModel.java | 1003 ++++ .../main/java/org/sandag/abm/ctramp/Util.java | 379 ++ .../java/org/sandag/abm/ctramp/UtilRmi.java | 137 + .../abm/ctramp/WorkLocationChoiceModel.java | 692 +++ .../ctramp/WorkLocationChoiceTaskJppf.java | 246 + .../ctramp/WorkLocationChoiceTaskJppfNew.java | 253 + .../dta/postprocessing/PostprocessModel.java | 247 + .../postprocessing/broadTODProcessing.java | 224 + .../postprocessing/detailedTODProcessing.java | 563 ++ .../abm/dta/postprocessing/dtaTrip.java | 386 ++ .../spatialDisaggregationModel.java | 144 + .../todDisaggregationModel.java | 118 + .../InternalExternalDmuFactory.java | 40 + .../InternalExternalDmuFactoryIf.java | 13 + .../InternalExternalModel.java | 309 + .../InternalExternalModelStructure.java | 95 + .../InternalExternalTour.java | 307 + .../InternalExternalTourDestChoiceDMU.java | 114 + .../InternalExternalTourDestChoiceModel.java | 172 + .../InternalExternalTourManager.java | 377 ++ ...ernalExternalTourTimeOfDayChoiceModel.java | 182 + .../InternalExternalTrip.java | 313 + .../InternalExternalTripModeChoiceDMU.java | 551 ++ .../InternalExternalTripModeChoiceModel.java | 277 + .../InternalExternalTripTables.java | 698 +++ .../maas/HouseholdAVAllocationManager.java | 1496 +++++ .../abm/maas/HouseholdAVAllocationModel.java | 671 +++ ...holdAVAllocationModelParkingChoiceDMU.java | 239 + .../HouseholdAVAllocationModelRunner.java | 260 + ...seholdAVAllocationModelTripUtilityDMU.java | 101 + ...holdAVAllocationModelVehicleChoiceDMU.java | 212 + .../java/org/sandag/abm/maas/PersonTrip.java | 277 + .../sandag/abm/maas/PersonTripManager.java | 1087 ++++ .../org/sandag/abm/maas/TNCFleetModel.java | 448 ++ .../java/org/sandag/abm/maas/TNCVehicle.java | 156 + .../sandag/abm/maas/TNCVehicleManager.java | 878 +++ .../org/sandag/abm/maas/TNCVehicleTrip.java | 428 ++ .../sandag/abm/maas/TransportCostManager.java | 438 ++ .../org/sandag/abm/modechoice/AutoDMU.java | 136 + .../org/sandag/abm/modechoice/AutoUEC.java | 138 + .../org/sandag/abm/modechoice/Constants.java | 56 + .../org/sandag/abm/modechoice/MaasDMU.java | 128 + .../org/sandag/abm/modechoice/MaasUEC.java | 141 + .../abm/modechoice/MgraDataManager.java | 1551 +++++ .../java/org/sandag/abm/modechoice/Modes.java | 115 + .../sandag/abm/modechoice/NonMotorDMU.java | 146 + .../sandag/abm/modechoice/NonMotorUEC.java | 189 + .../sandag/abm/modechoice/TapDataManager.java | 321 ++ .../sandag/abm/modechoice/TazDataManager.java | 924 +++ .../abm/modechoice/TransitDriveAccessDMU.java | 465 ++ .../abm/modechoice/TransitWalkAccessDMU.java | 332 ++ .../abm/reporting/AbstractCsvExporter.java | 32 + .../org/sandag/abm/reporting/CVMExporter.java | 304 + .../org/sandag/abm/reporting/CVMScaler.java | 179 + .../java/org/sandag/abm/reporting/CsvRow.java | 22 + .../sandag/abm/reporting/CsvWriterThread.java | 86 + .../sandag/abm/reporting/DataExporter.java | 2688 +++++++++ .../abm/reporting/DoubleFormatUtil.java | 484 ++ .../org/sandag/abm/reporting/IExporter.java | 11 + .../org/sandag/abm/reporting/IMatrixDao.java | 8 + .../sandag/abm/reporting/OMXMatrixDao.java | 36 + .../org/sandag/abm/reporting/SkimBuilder.java | 721 +++ .../abm/reporting/TranscadMatrixDao.java | 28 + .../abm/reporting/TransitTimeReporter.java | 449 ++ .../abm/reporting/TruckCsvExporter.java | 58 + .../reporting/TruckCsvPublisherThread.java | 67 + .../abm/reporting/TruckOmxExporter.java | 56 + .../emfac2011/AquavisDataBuilder.java | 47 + .../emfac2011/Emfac2011AquavisIntrazonal.java | 45 + .../reporting/emfac2011/Emfac2011Data.java | 302 + .../emfac2011/Emfac2011Definitions.java | 86 + .../emfac2011/Emfac2011InputFileCreator.java | 577 ++ .../emfac2011/Emfac2011Properties.java | 101 + .../reporting/emfac2011/Emfac2011Runner.java | 216 + .../emfac2011/Emfac2011SpeedCategory.java | 73 + .../reporting/emfac2011/Emfac2011SqlUtil.java | 145 + .../emfac2011/Emfac2011VehicleType.java | 146 + .../reporting/emfac2011/RunEmfacDialog.java | 174 + .../reporting/emfac2011/SandagAutoModes.java | 11 + .../specialevent/SpecialEventDmuFactory.java | 42 + .../SpecialEventDmuFactoryIf.java | 13 + .../abm/specialevent/SpecialEventModel.java | 347 ++ .../SpecialEventOriginChoiceDMU.java | 205 + .../SpecialEventOriginChoiceModel.java | 384 ++ .../abm/specialevent/SpecialEventTour.java | 291 + .../specialevent/SpecialEventTourManager.java | 342 ++ .../abm/specialevent/SpecialEventTrip.java | 292 + .../SpecialEventTripModeChoiceDMU.java | 451 ++ .../SpecialEventTripModeChoiceModel.java | 284 + .../specialevent/SpecialEventTripTables.java | 595 ++ .../org/sandag/abm/survey/OutputTapPairs.java | 363 ++ .../sandag/abm/utilities/CreateLogsums.java | 410 ++ .../sandag/abm/utilities/ErrorLogging.java | 41 + .../abm/utilities/ModelOutputReader.java | 836 +++ .../sandag/abm/utilities/RunModeChoice.java | 402 ++ .../abm/utilities/TapAtConsistencyCheck.java | 183 + .../abm/validation/MainApplication.java | 437 ++ .../sandag/abm/visitor/VisitorDmuFactory.java | 55 + .../abm/visitor/VisitorDmuFactoryIf.java | 18 + .../VisitorMicromobilityChoiceDMU.java | 139 + .../VisitorMicromobilityChoiceModel.java | 325 ++ .../org/sandag/abm/visitor/VisitorModel.java | 499 ++ .../abm/visitor/VisitorModelStructure.java | 95 + .../org/sandag/abm/visitor/VisitorStop.java | 176 + .../visitor/VisitorStopFrequencyModel.java | 325 ++ .../visitor/VisitorStopLocationChoiceDMU.java | 406 ++ .../VisitorStopLocationChoiceModel.java | 538 ++ .../abm/visitor/VisitorStopPurposeModel.java | 233 + .../VisitorStopTimeOfDayChoiceModel.java | 365 ++ .../org/sandag/abm/visitor/VisitorTour.java | 350 ++ .../abm/visitor/VisitorTourDestChoiceDMU.java | 317 + .../visitor/VisitorTourDestChoiceModel.java | 586 ++ .../visitor/VisitorTourEstimationFile.java | 382 ++ .../abm/visitor/VisitorTourManager.java | 531 ++ .../abm/visitor/VisitorTourModeChoiceDMU.java | 570 ++ .../visitor/VisitorTourModeChoiceModel.java | 399 ++ .../VisitorTourTimeOfDayChoiceModel.java | 192 + .../org/sandag/abm/visitor/VisitorTrip.java | 518 ++ .../abm/visitor/VisitorTripModeChoiceDMU.java | 883 +++ .../visitor/VisitorTripModeChoiceModel.java | 327 ++ .../sandag/abm/visitor/VisitorTripTables.java | 682 +++ .../AlternativeUsesMatrices.java | 42 + .../ChangingTravelAttributeGetter.java | 52 + .../cvm/activityTravel/CodedAlternative.java | 35 + .../CoefficientFormatError.java | 61 + .../cvm/activityTravel/DurationModel.java | 91 + .../activityTravel/HouseholdInterface.java | 46 + .../LoggingStopAlternative.java | 79 + .../cvm/activityTravel/ModelUsesMatrices.java | 39 + .../activityTravel/ModelWithCoefficients.java | 54 + .../cvm/activityTravel/PersonInterface.java | 54 + .../RealNumberDistribution.java | 37 + .../org/sandag/cvm/activityTravel/Stop.java | 42 + .../cvm/activityTravel/StopAlternative.java | 106 + .../sandag/cvm/activityTravel/StopChoice.java | 230 + .../org/sandag/cvm/activityTravel/Tour.java | 290 + .../cvm/activityTravel/TourInterface.java | 61 + .../TourNextStopPurposeChoice.java | 45 + .../sandag/cvm/activityTravel/TourType.java | 115 + .../cvm/activityTravel/TravelTimeTracker.java | 117 + .../sandag/cvm/activityTravel/TripMode.java | 74 + .../cvm/activityTravel/TripModeChoice.java | 96 + .../activityTravel/VehicleTourTypeChoice.java | 51 + .../activityTravel/ZonePairDisutility.java | 7 + .../cvm/AlogitLogitModelNest.java | 21 + .../cvm/CommercialNextStopChoice.java | 64 + .../cvm/CommercialNextStopPurposeChoice.java | 249 + .../activityTravel/cvm/CommercialTour.java | 328 ++ .../cvm/CommercialTravelTimeTracker.java | 77 + .../cvm/CommercialTripMode.java | 256 + .../cvm/CommercialVehicleTourType.java | 112 + .../cvm/CommercialVehicleTourTypeChoice.java | 238 + .../cvm/CommercialVehicleTripModeChoice.java | 216 + .../activityTravel/cvm/DurationModel2.java | 49 + .../cvm/GenerateCommercialTours.java | 855 +++ .../cvm/TourStartTimeModel.java | 115 + .../cvm/VehicleTourTypeNest.java | 16 + .../calgary/weekend/GenerateWeekendTours.java | 292 + .../weekend/NextWeekendTourStartTime.java | 101 + .../cvm/calgary/weekend/TourInTimeBand.java | 185 + .../cvm/calgary/weekend/WeekendHousehold.java | 366 ++ .../cvm/calgary/weekend/WeekendPerson.java | 105 + .../calgary/weekend/WeekendStopChoice.java | 54 + .../weekend/WeekendStopPurposeChoice.java | 217 + .../cvm/calgary/weekend/WeekendTour.java | 345 ++ .../cvm/calgary/weekend/WeekendTourType.java | 73 + .../weekend/WeekendTourTypeChoice.java | 130 + .../weekend/WeekendTravelTimeTracker.java | 72 + .../cvm/common/datafile/BaseDataFile.java | 446 ++ .../cvm/common/datafile/BinaryFileReader.java | 114 + .../cvm/common/datafile/BinaryFileWriter.java | 106 + .../cvm/common/datafile/CSVFileReader.java | 810 +++ .../cvm/common/datafile/CSVFileWriter.java | 248 + .../cvm/common/datafile/D211FileReader.java | 304 + .../cvm/common/datafile/D231FileReader.java | 129 + .../cvm/common/datafile/DBFFileReader.java | 225 + .../cvm/common/datafile/DBFFileWriter.java | 200 + .../sandag/cvm/common/datafile/DataFile.java | 232 + .../cvm/common/datafile/DataHeader.java | 114 + .../cvm/common/datafile/DataReader.java | 94 + .../sandag/cvm/common/datafile/DataTypes.java | 38 + .../cvm/common/datafile/DataWriter.java | 105 + .../datafile/DbByteArrayOutputStream.java | 47 + .../cvm/common/datafile/DiskObjectArray.java | 227 + .../cvm/common/datafile/ExcelFileReader.java | 692 +++ .../sandag/cvm/common/datafile/FileType.java | 55 + .../datafile/FixedFormatTextFileReader.java | 315 + .../common/datafile/GeneralDecimalFormat.java | 53 + .../cvm/common/datafile/JDBCTableReader.java | 221 + .../cvm/common/datafile/JDBCTableWriter.java | 136 + .../datafile/MissingValueException.java | 37 + .../datafile/MultipleValueException.java | 39 + .../common/datafile/NEW_CSVFileReader.java | 647 +++ .../common/datafile/OLD_CSVFileReader.java | 702 +++ .../cvm/common/datafile/ReportGenerator.java | 28 + .../common/datafile/TableDataFileReader.java | 106 + .../common/datafile/TableDataFileWriter.java | 105 + .../cvm/common/datafile/TableDataReader.java | 67 + .../cvm/common/datafile/TableDataSet.java | 1927 +++++++ .../datafile/TableDataSetCacheCollection.java | 218 + .../datafile/TableDataSetCollection.java | 200 + .../datafile/TableDataSetCrosstabber.java | 407 ++ .../common/datafile/TableDataSetIndex.java | 225 + .../datafile/TableDataSetIndexedValue.java | 601 ++ .../common/datafile/TableDataSetLoader.java | 59 + .../datafile/TableDataSetSoftReference.java | 40 + .../cvm/common/datafile/TableDataWriter.java | 67 + .../sandag/cvm/common/datafile/TextFile.java | 240 + .../common/datafile/tests/BinaryFileTest.java | 105 + .../datafile/tests/CSVFileReaderTest.java | 83 + .../datafile/tests/CSVFileWriterTest.java | 64 + .../common/datafile/tests/DataFileTest.java | 85 + .../datafile/tests/DiskObjectArrayTest.java | 193 + .../datafile/tests/JDBCTableReaderTest.java | 87 + .../datafile/tests/TableDataSetTest.java | 322 ++ .../common/discreteEvent/EventDispatcher.java | 33 + .../cvm/common/discreteEvent/EventQueue.java | 34 + .../discreteEvent/TestEventDispatcher.java | 65 + .../cvm/common/discreteEvent/TimedEvent.java | 57 + .../emme2/Emme2MatrixHashtableReader.java | 65 + .../common/emme2/IndexConditionFunction.java | 124 + .../cvm/common/emme2/IndexLinearFunction.java | 147 + .../common/emme2/MatrixAndTAZTableCache.java | 83 + .../cvm/common/emme2/MatrixCacheReader.java | 45 + .../emme2/TestEmme2311MatrixReader.java | 29 + .../common/model/AggregateAlternative.java | 26 + .../sandag/cvm/common/model/Alternative.java | 26 + .../model/ChoiceModelOverflowException.java | 20 + .../cvm/common/model/DiscreteChoiceModel.java | 63 + .../model/DiscreteChoiceModelInterface.java | 47 + .../common/model/FixedUtilityAlternative.java | 37 + .../cvm/common/model/GumbelErrorTerm.java | 77 + .../model/LinearInParametersFunction.java | 72 + .../sandag/cvm/common/model/LogitModel.java | 261 + .../common/model/NoAlternativeAvailable.java | 28 + ...icalDerivativeSingleParameterFunction.java | 34 + .../cvm/common/model/RandomVariable.java | 35 + .../common/model/SingleParameterFunction.java | 28 + .../cvm/common/model/TestLogitModel.java | 63 + .../model/UtilityMaximizingChoiceModel.java | 64 + ...imizingChoiceModelWithErrorTermVector.java | 84 + .../cvm/common/skims/HDF5MatrixReader.java | 395 ++ .../cvm/common/skims/HDF5MatrixWriter.java | 281 + .../skims/LinearFunctionOfSomeSkims.java | 109 + .../org/sandag/cvm/common/skims/Location.java | 8 + .../skims/OMXMatrixCollectionReader.java | 67 + .../sandag/cvm/common/skims/SomeSkims.java | 437 ++ .../skims/TranscadMatrixCollectionReader.java | 67 + .../skims/TravelAttributesInterface.java | 23 + .../TravelUtilityCalculatorInterface.java | 28 + .../patternDetail/DestinationRandomTerms.java | 121 + .../TestDestinationRandomTerms.java | 273 + .../htm/applications/SandagCountyModel.java | 352 ++ .../org/sandag/htm/applications/ohio.java | 118 + .../htm/applications/sandagZonalModel.java | 338 ++ .../sandag/htm/applications/sandag_tm.java | 45 + .../sandag/htm/applications/utilities.java | 217 + .../org/sandag/htm/applications/yumaMPO.java | 180 + .../org/sandag/htm/processFAF/ModesFAF.java | 19 + .../org/sandag/htm/processFAF/ReadFAF4.java | 522 ++ .../htm/processFAF/commodityClassType.java | 13 + .../htm/processFAF/convertTonsToTrucks.java | 113 + .../htm/processFAF/countyTruckModel.java | 597 ++ .../disaggregateAndAggregateFlows.java | 67 + .../htm/processFAF/disaggregateFlows.java | 1357 +++++ .../org/sandag/htm/processFAF/fafUtils.java | 331 ++ .../org/sandag/htm/processFAF/readFAF2.java | 547 ++ .../org/sandag/htm/processFAF/readFAF3.java | 520 ++ .../sandag/htm/processFAF/reportFormat.java | 23 + .../src/main/python/assignScenarioID.py | 27 + .../main/python/calculate_micromobility.py | 273 + sandag_abm/src/main/python/checkFreeSpace.py | 33 + sandag_abm/src/main/python/check_output.py | 139 + sandag_abm/src/main/python/cvm_analysis.zip | Bin 0 -> 29456 bytes .../src/main/python/cvm_input_create.py | 462 ++ .../main/python/dataExporter/abmScenario.py | 3914 +++++++++++++ .../main/python/dataExporter/environment.yml | 92 + .../python/dataExporter/hwyShapeExport.py | 523 ++ .../src/main/python/dataExporter/serialRun.py | 168 + .../main/python/dataExporter/skimAppender.py | 1895 ++++++ .../src/main/python/database_summary.py | 190 + sandag_abm/src/main/python/excel_update.py | 85 + sandag_abm/src/main/python/parameterUpdate.py | 60 + .../pythonGUI/createStudyAndScenario.py | 237 + .../main/python/pythonGUI/parameterEditor.py | 253 + .../src/main/python/pythonGUI/popupMsg.py | 17 + sandag_abm/src/main/python/pythonGUI/setup.py | 9 + .../src/main/python/pythonGUI/stringFinder.py | 21 + .../src/main/python/pythonGUI/validatorGUI.py | 122 + .../src/main/python/remote_run_traffic.py | 123 + sandag_abm/src/main/python/sdcvm.py | 840 +++ sandag_abm/src/main/python/sdcvm_settings.py | 43 + sandag_abm/src/main/python/sdcvm_settings.pyc | Bin 0 -> 1332 bytes sandag_abm/src/main/python/sdcvm_summarize.py | 99 + sandag_abm/src/main/python/serverswap.py | 92 + .../src/main/r/INRIX_OutlierAnalysis_Final.R | 532 ++ .../src/main/r/RegressionAnalysis_Final.R | 953 ++++ sandag_abm/src/main/r/summarize_SR125data.R | 141 + sandag_abm/src/main/r/utilfunc.R | 256 + sandag_abm/src/main/r/visualizer/Master.R | 94 + .../src/main/r/visualizer/SummarizeABM2016.R | 2448 ++++++++ .../src/main/r/visualizer/_SYSTEM_VARIABLES.R | 100 + .../src/main/r/visualizer/workersByMAZ.R | 114 + .../src/main/resources/BatchSubstitute.bat | 20 + sandag_abm/src/main/resources/CTRampEnv.bat | 61 + sandag_abm/src/main/resources/CheckOutput.bat | 8 + .../main/resources/CreateD2TAccessFile.bat | 9 + .../src/main/resources/DataExporter.bat | 33 + .../src/main/resources/DataLoadRequest.bat | 7 + sandag_abm/src/main/resources/DataSummary.bat | 14 + sandag_abm/src/main/resources/ExcelUpdate.bat | 16 + .../src/main/resources/FHWADataExporter.bat | 142 + .../main/resources/GnuWin32/bin/libiconv2.dll | Bin 0 -> 898048 bytes .../main/resources/GnuWin32/bin/libintl3.dll | Bin 0 -> 101888 bytes .../src/main/resources/GnuWin32/bin/tee.exe | Bin 0 -> 24576 bytes sandag_abm/src/main/resources/HPPowerOff.bat | 1 + sandag_abm/src/main/resources/HPPowerOn.bat | 1 + .../src/main/resources/RunEMFAC2011.cmd | 12 + .../src/main/resources/RunEMFAC2014.cmd | 16 + sandag_abm/src/main/resources/RunViz.bat | 126 + .../src/main/resources/StartHHAndNodes.cmd | 44 + .../src/main/resources/assignScenarioID.cmd | 6 + .../checkAtTransitNetworkConsistency.cmd | 44 + .../src/main/resources/checkFreeSpaceOnC.bat | 2 + .../resources/copy_networkfiles_to_study.cmd | 18 + .../src/main/resources/copy_networks.cmd | 17 + .../src/main/resources/create_scenario.cmd | 107 + sandag_abm/src/main/resources/cvm.bat | 30 + sandag_abm/src/main/resources/cvm.properties | 39 + sandag_abm/src/main/resources/emme_python.bat | 31 + sandag_abm/src/main/resources/init_emme.cmd | 37 + sandag_abm/src/main/resources/jhdf.dll | Bin 0 -> 843264 bytes sandag_abm/src/main/resources/jhdf5.dll | Bin 0 -> 2284544 bytes .../src/main/resources/jppf-client.properties | 141 + .../jppf-clientDistributed.properties | 37 + .../src/main/resources/jppf-driver.properties | 297 + .../main/resources/jppf-sandag01.properties | 298 + .../main/resources/jppf-sandag02.properties | 298 + .../main/resources/jppf-sandag03.properties | 298 + .../main/resources/jppf-sandag04.properties | 298 + .../main/resources/log4j-client.properties | 39 + .../main/resources/log4j-driver.properties | 42 + .../main/resources/log4j-sandag01.properties | 46 + .../src/main/resources/log4j-sandag01.xml | 439 ++ .../main/resources/log4j-sandag02.properties | 46 + .../src/main/resources/log4j-sandag02.xml | 439 ++ .../main/resources/log4j-sandag03.properties | 46 + .../src/main/resources/log4j-sandag03.xml | 439 ++ .../main/resources/log4j-sandag04.properties | 46 + .../src/main/resources/log4j-sandag04.xml | 439 ++ sandag_abm/src/main/resources/log4j.xml | 443 ++ .../main/resources/log4j_AtTransitCheck.xml | 36 + sandag_abm/src/main/resources/log4j_d2t.xml | 36 + sandag_abm/src/main/resources/log4j_hh.xml | 36 + sandag_abm/src/main/resources/log4j_mtx.xml | 36 + sandag_abm/src/main/resources/log4j_test.xml | 455 ++ sandag_abm/src/main/resources/mapAndRun.bat | 33 + sandag_abm/src/main/resources/pskill.exe | Bin 0 -> 621944 bytes sandag_abm/src/main/resources/runDriver.cmd | 17 + sandag_abm/src/main/resources/runHhMgr.cmd | 54 + .../src/main/resources/runHhMgr_log.bat | 18 + sandag_abm/src/main/resources/runMtxMgr.cmd | 43 + .../src/main/resources/runMtxMgr_log.bat | 17 + sandag_abm/src/main/resources/runSandag01.cmd | 16 + .../src/main/resources/runSandag01_log.bat | 13 + sandag_abm/src/main/resources/runSandag02.cmd | 15 + sandag_abm/src/main/resources/runSandag03.cmd | 15 + sandag_abm/src/main/resources/runSandag04.cmd | 15 + .../src/main/resources/runSandagAbm_MAAS.cmd | 59 + .../resources/runSandagAbm_MCDiagnostic.cmd | 55 + .../src/main/resources/runSandagAbm_SDRM.cmd | 68 + .../src/main/resources/runSandagAbm_SEM.cmd | 57 + .../src/main/resources/runSandagAbm_SMM.cmd | 86 + .../main/resources/runSandagBikeLogsums.cmd | 45 + .../resources/runSandagBikeRouteChoice.cmd | 38 + .../main/resources/runSandagWalkLogsums.cmd | 45 + .../src/main/resources/runTransitReporter.cmd | 48 + .../src/main/resources/sandag_abm.properties | 1357 +++++ sandag_abm/src/main/resources/serverswap.bat | 9 + .../src/main/resources/serverswap_files.csv | 25 + sandag_abm/src/main/resources/setup.bat | 1 + sandag_abm/src/main/resources/stopABM.cmd | 5 + sandag_abm/src/main/resources/taskkill.bat | 2 + .../resources/updateYearSpecificProps.bat | 6 + sandag_abm/src/main/resources/w9xpopen.exe | Bin 0 -> 49664 bytes 693 files changed, 217402 insertions(+) create mode 100644 sandag_abm/src/main/emme/init_emme_project.py create mode 100644 sandag_abm/src/main/emme/python_virtualenv.pth create mode 100644 sandag_abm/src/main/emme/solutions.mtbx create mode 100644 sandag_abm/src/main/emme/solutions_unconsolidated.mtbx create mode 100644 sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py create mode 100644 sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py create mode 100644 sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py create mode 100644 sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py create mode 100644 sandag_abm/src/main/emme/toolbox/build_toolbox.py create mode 100644 sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py create mode 100644 sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/import_network.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/input_checker.py create mode 100644 sandag_abm/src/main/emme/toolbox/import/run4Ds.py create mode 100644 sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py create mode 100644 sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py create mode 100644 sandag_abm/src/main/emme/toolbox/master_run.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/external_external.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/external_internal.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/distribution.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/generation.py create mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/demand.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/file_manager.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/general.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/properties.py create mode 100644 sandag_abm/src/main/emme/toolbox/utilities/run_summary.py create mode 100644 sandag_abm/src/main/emme/toolbox/validation/validation.py create mode 100644 sandag_abm/src/main/gisdk/SandagCommon.rsc create mode 100644 sandag_abm/src/main/gisdk/TC2OMX.rsc create mode 100644 sandag_abm/src/main/gisdk/TruckModel.rsc create mode 100644 sandag_abm/src/main/gisdk/Utilities.rsc create mode 100644 sandag_abm/src/main/gisdk/commVehDist.rsc create mode 100644 sandag_abm/src/main/gisdk/commVehDiversion.rsc create mode 100644 sandag_abm/src/main/gisdk/commVehGen.rsc create mode 100644 sandag_abm/src/main/gisdk/commVehTOD.rsc create mode 100644 sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc create mode 100644 sandag_abm/src/main/gisdk/createhwynet.rsc create mode 100644 sandag_abm/src/main/gisdk/createtodtables.rsc create mode 100644 sandag_abm/src/main/gisdk/createtrnroutes.rsc create mode 100644 sandag_abm/src/main/gisdk/dbox.rsc create mode 100644 sandag_abm/src/main/gisdk/exportTCData.rsc create mode 100644 sandag_abm/src/main/gisdk/externalInternal.rsc create mode 100644 sandag_abm/src/main/gisdk/gui_generic.rsc create mode 100644 sandag_abm/src/main/gisdk/hwyassign.rsc create mode 100644 sandag_abm/src/main/gisdk/hwyskim.rsc create mode 100644 sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc create mode 100644 sandag_abm/src/main/gisdk/parameter.rsc create mode 100644 sandag_abm/src/main/gisdk/sandag_abm.lst create mode 100644 sandag_abm/src/main/gisdk/sandag_abm_generic.lst create mode 100644 sandag_abm/src/main/gisdk/sandag_abm_master.rsc create mode 100644 sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc create mode 100644 sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc create mode 100644 sandag_abm/src/main/gisdk/sellink_volume.rsc create mode 100644 sandag_abm/src/main/gisdk/trnassign.rsc create mode 100644 sandag_abm/src/main/gisdk/trnskim.rsc create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/SubtourTodEstimationMcLogsumsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Edge.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Network.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Node.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Path.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceLogsumMatrixApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppfNew.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SegmentedSparseMatrix.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandler.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Util.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppfNew.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/PostprocessModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/detailedTODProcessing.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/todDisaggregationModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ModelUsesMatrices.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ModelWithCoefficients.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/PersonInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/RealNumberDistribution.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Stop.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTripModeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTourType.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTourTypeChoice.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTravelTimeTracker.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BaseDataFile.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataFile.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/FileType.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/FixedFormatTextFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/MissingValueException.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/MultipleValueException.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/NEW_CSVFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/OLD_CSVFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ReportGenerator.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataFileReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataFileWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSet.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetIndex.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetIndexedValue.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetLoader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetSoftReference.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TextFile.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TimedEvent.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexLinearFunction.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/MatrixAndTAZTableCache.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/MatrixCacheReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/TestEmme2311MatrixReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/LogitModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixWriter.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/LinearFunctionOfSomeSkims.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/Location.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/OMXMatrixCollectionReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java create mode 100644 sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/TestDestinationRandomTerms.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/SandagCountyModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java create mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java create mode 100644 sandag_abm/src/main/python/assignScenarioID.py create mode 100644 sandag_abm/src/main/python/calculate_micromobility.py create mode 100644 sandag_abm/src/main/python/checkFreeSpace.py create mode 100644 sandag_abm/src/main/python/check_output.py create mode 100644 sandag_abm/src/main/python/cvm_analysis.zip create mode 100644 sandag_abm/src/main/python/cvm_input_create.py create mode 100644 sandag_abm/src/main/python/dataExporter/abmScenario.py create mode 100644 sandag_abm/src/main/python/dataExporter/environment.yml create mode 100644 sandag_abm/src/main/python/dataExporter/hwyShapeExport.py create mode 100644 sandag_abm/src/main/python/dataExporter/serialRun.py create mode 100644 sandag_abm/src/main/python/dataExporter/skimAppender.py create mode 100644 sandag_abm/src/main/python/database_summary.py create mode 100644 sandag_abm/src/main/python/excel_update.py create mode 100644 sandag_abm/src/main/python/parameterUpdate.py create mode 100644 sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py create mode 100644 sandag_abm/src/main/python/pythonGUI/parameterEditor.py create mode 100644 sandag_abm/src/main/python/pythonGUI/popupMsg.py create mode 100644 sandag_abm/src/main/python/pythonGUI/setup.py create mode 100644 sandag_abm/src/main/python/pythonGUI/stringFinder.py create mode 100644 sandag_abm/src/main/python/pythonGUI/validatorGUI.py create mode 100644 sandag_abm/src/main/python/remote_run_traffic.py create mode 100644 sandag_abm/src/main/python/sdcvm.py create mode 100644 sandag_abm/src/main/python/sdcvm_settings.py create mode 100644 sandag_abm/src/main/python/sdcvm_settings.pyc create mode 100644 sandag_abm/src/main/python/sdcvm_summarize.py create mode 100644 sandag_abm/src/main/python/serverswap.py create mode 100644 sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R create mode 100644 sandag_abm/src/main/r/RegressionAnalysis_Final.R create mode 100644 sandag_abm/src/main/r/summarize_SR125data.R create mode 100644 sandag_abm/src/main/r/utilfunc.R create mode 100644 sandag_abm/src/main/r/visualizer/Master.R create mode 100644 sandag_abm/src/main/r/visualizer/SummarizeABM2016.R create mode 100644 sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R create mode 100644 sandag_abm/src/main/r/visualizer/workersByMAZ.R create mode 100644 sandag_abm/src/main/resources/BatchSubstitute.bat create mode 100644 sandag_abm/src/main/resources/CTRampEnv.bat create mode 100644 sandag_abm/src/main/resources/CheckOutput.bat create mode 100644 sandag_abm/src/main/resources/CreateD2TAccessFile.bat create mode 100644 sandag_abm/src/main/resources/DataExporter.bat create mode 100644 sandag_abm/src/main/resources/DataLoadRequest.bat create mode 100644 sandag_abm/src/main/resources/DataSummary.bat create mode 100644 sandag_abm/src/main/resources/ExcelUpdate.bat create mode 100644 sandag_abm/src/main/resources/FHWADataExporter.bat create mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll create mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll create mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/tee.exe create mode 100644 sandag_abm/src/main/resources/HPPowerOff.bat create mode 100644 sandag_abm/src/main/resources/HPPowerOn.bat create mode 100644 sandag_abm/src/main/resources/RunEMFAC2011.cmd create mode 100644 sandag_abm/src/main/resources/RunEMFAC2014.cmd create mode 100644 sandag_abm/src/main/resources/RunViz.bat create mode 100644 sandag_abm/src/main/resources/StartHHAndNodes.cmd create mode 100644 sandag_abm/src/main/resources/assignScenarioID.cmd create mode 100644 sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd create mode 100644 sandag_abm/src/main/resources/checkFreeSpaceOnC.bat create mode 100644 sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd create mode 100644 sandag_abm/src/main/resources/copy_networks.cmd create mode 100644 sandag_abm/src/main/resources/create_scenario.cmd create mode 100644 sandag_abm/src/main/resources/cvm.bat create mode 100644 sandag_abm/src/main/resources/cvm.properties create mode 100644 sandag_abm/src/main/resources/emme_python.bat create mode 100644 sandag_abm/src/main/resources/init_emme.cmd create mode 100644 sandag_abm/src/main/resources/jhdf.dll create mode 100644 sandag_abm/src/main/resources/jhdf5.dll create mode 100644 sandag_abm/src/main/resources/jppf-client.properties create mode 100644 sandag_abm/src/main/resources/jppf-clientDistributed.properties create mode 100644 sandag_abm/src/main/resources/jppf-driver.properties create mode 100644 sandag_abm/src/main/resources/jppf-sandag01.properties create mode 100644 sandag_abm/src/main/resources/jppf-sandag02.properties create mode 100644 sandag_abm/src/main/resources/jppf-sandag03.properties create mode 100644 sandag_abm/src/main/resources/jppf-sandag04.properties create mode 100644 sandag_abm/src/main/resources/log4j-client.properties create mode 100644 sandag_abm/src/main/resources/log4j-driver.properties create mode 100644 sandag_abm/src/main/resources/log4j-sandag01.properties create mode 100644 sandag_abm/src/main/resources/log4j-sandag01.xml create mode 100644 sandag_abm/src/main/resources/log4j-sandag02.properties create mode 100644 sandag_abm/src/main/resources/log4j-sandag02.xml create mode 100644 sandag_abm/src/main/resources/log4j-sandag03.properties create mode 100644 sandag_abm/src/main/resources/log4j-sandag03.xml create mode 100644 sandag_abm/src/main/resources/log4j-sandag04.properties create mode 100644 sandag_abm/src/main/resources/log4j-sandag04.xml create mode 100644 sandag_abm/src/main/resources/log4j.xml create mode 100644 sandag_abm/src/main/resources/log4j_AtTransitCheck.xml create mode 100644 sandag_abm/src/main/resources/log4j_d2t.xml create mode 100644 sandag_abm/src/main/resources/log4j_hh.xml create mode 100644 sandag_abm/src/main/resources/log4j_mtx.xml create mode 100644 sandag_abm/src/main/resources/log4j_test.xml create mode 100644 sandag_abm/src/main/resources/mapAndRun.bat create mode 100644 sandag_abm/src/main/resources/pskill.exe create mode 100644 sandag_abm/src/main/resources/runDriver.cmd create mode 100644 sandag_abm/src/main/resources/runHhMgr.cmd create mode 100644 sandag_abm/src/main/resources/runHhMgr_log.bat create mode 100644 sandag_abm/src/main/resources/runMtxMgr.cmd create mode 100644 sandag_abm/src/main/resources/runMtxMgr_log.bat create mode 100644 sandag_abm/src/main/resources/runSandag01.cmd create mode 100644 sandag_abm/src/main/resources/runSandag01_log.bat create mode 100644 sandag_abm/src/main/resources/runSandag02.cmd create mode 100644 sandag_abm/src/main/resources/runSandag03.cmd create mode 100644 sandag_abm/src/main/resources/runSandag04.cmd create mode 100644 sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd create mode 100644 sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd create mode 100644 sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd create mode 100644 sandag_abm/src/main/resources/runSandagAbm_SEM.cmd create mode 100644 sandag_abm/src/main/resources/runSandagAbm_SMM.cmd create mode 100644 sandag_abm/src/main/resources/runSandagBikeLogsums.cmd create mode 100644 sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd create mode 100644 sandag_abm/src/main/resources/runSandagWalkLogsums.cmd create mode 100644 sandag_abm/src/main/resources/runTransitReporter.cmd create mode 100644 sandag_abm/src/main/resources/sandag_abm.properties create mode 100644 sandag_abm/src/main/resources/serverswap.bat create mode 100644 sandag_abm/src/main/resources/serverswap_files.csv create mode 100644 sandag_abm/src/main/resources/setup.bat create mode 100644 sandag_abm/src/main/resources/stopABM.cmd create mode 100644 sandag_abm/src/main/resources/taskkill.bat create mode 100644 sandag_abm/src/main/resources/updateYearSpecificProps.bat create mode 100644 sandag_abm/src/main/resources/w9xpopen.exe diff --git a/sandag_abm/src/main/emme/init_emme_project.py b/sandag_abm/src/main/emme/init_emme_project.py new file mode 100644 index 0000000..2809405 --- /dev/null +++ b/sandag_abm/src/main/emme/init_emme_project.py @@ -0,0 +1,97 @@ +#/////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2019. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// init_emme_project.py /// +#//// /// +#//// Usage: init_emme_project.py [-r root] [-t title] /// +#//// /// +#//// [-r root]: Specifies the root directory in which to create /// +#//// the Emme project. /// +#//// If omitted, defaults to the current working directory /// +#//// [-t title]: The title of the Emme project and Emme database. /// +#//// If omitted, defaults to SANDAG empty database. /// +#//// [-v emmeversion]: Emme version to use to create the project. /// +#//// If omitted, defaults to 4.3.7. /// +#//// /// +#//// /// +#//// /// +#//// /// +#/////////////////////////////////////////////////////////////////////////////// + +import inro.emme.desktop.app as _app +import inro.emme.desktop.types as _ws_types +import inro.emme.database.emmebank as _eb +import argparse +import os + +WKT_PROJECTION = 'PROJCS["NAD_1983_NSRS2007_StatePlane_California_VI_FIPS_0406_Ft_US",GEOGCS["GCS_NAD_1983_NSRS2007",DATUM["D_NAD_1983_NSRS2007",SPHEROID["GRS_1980",6378137.0,298.257222101]],PRIMEM["Greenwich",0.0],UNIT["Degree",0.0174532925199433]],PROJECTION["Lambert_Conformal_Conic"],PARAMETER["False_Easting",6561666.666666666],PARAMETER["False_Northing",1640416.666666667],PARAMETER["Central_Meridian",-116.25],PARAMETER["Standard_Parallel_1",32.78333333333333],PARAMETER["Standard_Parallel_2",33.88333333333333],PARAMETER["Latitude_Of_Origin",32.16666666666666],UNIT["Foot_US",0.3048006096012192]];-118608900 -91259500 3048.00609601219;-100000 10000;-100000 10000;3.28083333333333E-03;0.001;0.001;IsHighPrecision' + +def init_emme_project(root, title, emmeversion): + project_path = _app.create_project(root, "emme_project") + desktop = _app.start_dedicated( + project=project_path, user_initials="WS", visible=False) + project = desktop.project + project.name = "SANDAG Emme project" + prj_file_path = os.path.join(os.path.dirname(project_path), 'NAD 1983 NSRS2007 StatePlane California VI FIPS 0406 (US Feet).prj') + with open(prj_file_path, 'w') as f: + f.write(WKT_PROJECTION) + project.spatial_reference_file = prj_file_path + project.initial_view = _ws_types.Box(6.18187e+06, 1.75917e+06, 6.42519e+06, 1.89371e+06) + project_root = os.path.dirname(project_path) + dimensions = { + 'scalar_matrices': 9999, + 'destination_matrices': 999, + 'origin_matrices': 999, + 'full_matrices': 1600, + + 'scenarios': 10, + 'centroids': 5000, + 'regular_nodes': 29999, + 'links': 90000, + 'turn_entries': 13000, + 'transit_vehicles': 200, + 'transit_lines': 450, + 'transit_segments': 40000, + 'extra_attribute_values': 28000000, + + 'functions': 99, + 'operators': 5000 + } + + # for Emme version > 4.3.7, add the sola_analyses dimension + if emmeversion != '4.3.7': + dimensions['sola_analyses'] = 240 + + os.mkdir(os.path.join(project_root, "Database")) + emmebank = _eb.create(os.path.join(project_root, "Database", "emmebank"), dimensions) + emmebank.title = title + emmebank.coord_unit_length = 0.000189394 # feet to miles + emmebank.unit_of_length = "mi" + emmebank.unit_of_cost = "$" + emmebank.unit_of_energy = "MJ" + emmebank.node_number_digits = 6 + emmebank.use_engineering_notation = True + scenario = emmebank.create_scenario(100) + scenario.title = "Empty scenario" + emmebank.dispose() + + desktop.data_explorer().add_database(emmebank.path) + desktop.add_modeller_toolbox("%<$ProjectPath>%/scripts/sandag_toolbox.mtbx") + desktop.add_modeller_toolbox("%<$ProjectPath>%/scripts/solutions.mtbx") + project.save() + + +if __name__ == "__main__": + parser = argparse.ArgumentParser(description="Create a new empty Emme project and database with Sandag defaults.") + parser.add_argument('-r', '--root', help="path to the root ABM folder, default is the working folder", + default=os.path.abspath(os.getcwd())) + parser.add_argument('-t', '--title', help="the Emmebank title", + default="SANDAG empty database") + parser.add_argument('-v', '--emmeversion', help='the Emme version', default='4.3.7') + args = parser.parse_args() + + init_emme_project(args.root, args.title, args.emmeversion) diff --git a/sandag_abm/src/main/emme/python_virtualenv.pth b/sandag_abm/src/main/emme/python_virtualenv.pth new file mode 100644 index 0000000..c169252 --- /dev/null +++ b/sandag_abm/src/main/emme/python_virtualenv.pth @@ -0,0 +1,3 @@ +# Inserts defined python_virtualenv site-packages into the python module search path if defined +# +import sys, os; r=os.environ.get("PYTHON_VIRTUALENV"); t = 1 if r is None else sys.path.insert(0, os.path.join(r, "Lib\site-packages")); \ No newline at end of file diff --git a/sandag_abm/src/main/emme/solutions.mtbx b/sandag_abm/src/main/emme/solutions.mtbx new file mode 100644 index 0000000000000000000000000000000000000000..10d5345c4324f915c320462cba2c8740f5fe522e GIT binary patch literal 118784 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 literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/emme/solutions_unconsolidated.mtbx b/sandag_abm/src/main/emme/solutions_unconsolidated.mtbx new file mode 100644 index 0000000000000000000000000000000000000000..0674c2b5978a6c4e5ea11a14d6fb308a1417fd8d GIT binary patch literal 24576 zcmeI4&u`mg7{~p(PE%)z->g%Js;P=g5o^^hiSxs4h9=>pPFAf=Mw&4OLY|v=t+yoZ zVrO&`mzIBn3;zP*#08{XCM39XLPA4A95_sZ%MS3^iIcQqV>?ZQ=T#EF@8kE!^L$=k zo^NdLytidKhIrfRb#zC}GEs&Q=3P-_7$yMUY54lZZF7DDj%1?#tZINsMxRE6IVO1K zYla@vw6G9aiaZWm(8PiSkN^@u0!RP}AOR$R1dzZB6FABQgXdS~XO1s8J#%l*=oxLJ zV{{!`YeCI1{AQ`HDh*X^)Hf>?wJ!SRVXsh8E!XO*xS?*T(97zMI)uP3s!HjKSg+j> zi)v-F8WN!ldzS09ZBch%?cKg(*y4>VDoit+T{BzaWwExkksKS|38~eMBY8S_{$gV0 zcxC8x%WC!~y&knsxjnK!)9+D_7jfLDFm&4=JNA3RGH|@W1>*|~x?53K%fPH!Qg?iA_R_f9@~Stkt9K0XeWhNyqSObL`+B>7 z%5<&1xvkW1idWT}6Em!e?%?I+Y+yQGSs;wrZ5fC5y*4C#n%;LTudR&((4=wQP5fru zYwy6rG6~=F?)y5#S1HA`Vu)7N{#b{!) z{%N3p+FtT&u)+|pqwk%h>lla5DT{8learGSb~?VYLT-96hAeI4Uf<|8jZ@936KGDI zTynQ&6-F+9N`mo3f*e)6DSZ#^QH4D*&B#1J>^CQo{R?BqblRiKXE_2}Oz0B^K3I?d z5fCP{L z57&;+Ky(}lAOR$R z1dsp{Kmter2_OL^fCP}h|AfF>Y>?skN`%>QtmeLMJ7!aKdiw3#aMW$v-0Qke*+fZ@ zVfX~}n*9HNHu5b)|DccQSM+oG3B5&YbdCzbQ{gw^C*gbHmarQ6EAlIJ!h!^l01`j~ zNB{{S0VIF~kN^@u0rRap0F^55J!U9Jv>4l{)wVQADW&CNUe;tKpVM+#Nzn>Q9)@jNErXun`1y0q zeY4$ymkfNA`S@&z<6{YroSzNzJjchGj@h+(xfzZ(Vs5O(+6P)mUQZXY*^E}&kR>gb zFJ`quT25;@N!nP?{VutbwJZtVxswry`S7h!>`Q$LfXP%AFM+GSs5q&@K84 z{gHk_|D-MYA^nbiDLkPHO$iT$pM@Xjo3HF5$Aysq5aM^L zj)1^Ge1YK%NC`3GstX~|8REnf6?h3m1@bbY!b}u$cQPR;1j8UGkfES3?M>~;2mt}a e0&hJpOb*nJIWo`)ro1^O@_*KqIrxs7|Njm5dXxJA literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py b/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py new file mode 100644 index 0000000..ec8d4a9 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py @@ -0,0 +1,679 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// build_transit_scenario.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# The build transit scenario tool generates a new scenario in the Transit +# database (under the Database_transit directory) as a copy of a scenario in +# the base (traffic assignment) database. The base traffic scenario should have +# valid results from a traffic assignment for the travel times on links to be +# available for transit lines in mixed traffic operation. +# +# +# Inputs: +# period: the corresponding period for the scenario +# base_scenario_id: the base traffic assignment scenario in the main Emme database +# scenario_id: the ID to use for the new scenario in the Transit Emme database +# scenario_title: the title for the new scenario +# data_table_name: the root name for the source data table for the timed transfer +# line pairs and the day and regional pass costs. +# Usually the ScenarioYear +# overwrite: overwrite the scenario if it already exists. +# +# +# Script example: +""" +import inro.modeller as _m +import os +modeller = _m.Modeller() +desktop = modeller.desktop + +build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") +transit_assign = modeller.tool("sandag.assignment.transit_assignment") +load_properties = modeller.tool('sandag.utilities.properties') + +project_dir = os.path.dirname(desktop.project_path()) +main_directory = os.path.dirname(project_dir) +props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) +main_emmebank = os.path.join(project_dir, "Database", "emmebank") +scenario_id = 100 +base_scenario = main_emmebank.scenario(scenario_id) + +transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") + +periods = ["EA", "AM", "MD", "PM", "EV"] +period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) +num_processors = "MAX-1" +scenarioYear = str(props["scenarioYear"]) + +for number, period in period_ids: + src_period_scenario = main_emmebank.scenario(number) + transit_assign_scen = build_transit_scen( + period=period, base_scenario=src_period_scenario, + transit_emmebank=transit_emmebank, + scenario_id=src_period_scenario.id, + scenario_title="%s %s transit assign" % (base_scenario.title, period), + data_table_name=scenarioYear, overwrite=True) + transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, + skims_only=True, num_processors=num_processors) +""" + + + +TOOLBOX_ORDER = 21 + + +import inro.modeller as _m +import inro.emme.core.exception as _except +import inro.emme.database.emmebank as _eb +import traceback as _traceback +from copy import deepcopy as _copy +from collections import defaultdict as _defaultdict +import contextlib as _context + +import os +import sys +import math + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class BuildTransitNetwork(_m.Tool(), gen_utils.Snapshot): + + period = _m.Attribute(unicode) + scenario_id = _m.Attribute(int) + base_scenario_id = _m.Attribute(str) + + data_table_name = _m.Attribute(unicode) + scenario_title = _m.Attribute(unicode) + overwrite = _m.Attribute(bool) + + tool_run_msg = "" + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.data_table_name = None + self.base_scenario = _m.Modeller().scenario + self.scenario_id = 100 + self.scenario_title = "" + self.overwrite = False + self.attributes = [ + "period", "scenario_id", "base_scenario_id", + "data_table_name", "scenario_title", "overwrite"] + + def page(self): + if not self.data_table_name: + load_properties = _m.Modeller().tool('sandag.utilities.properties') + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_directory = os.path.dirname(project_dir) + props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + self.data_table_name = props["scenarioYear"] + + pb = _m.ToolPageBuilder(self) + pb.title = "Build transit network" + pb.description = """ + Builds the transit network for the specified period based + on existing base (traffic + transit) scenario.""" + pb.branding_text = "- SANDAG - " + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + options = [("EA", "Early AM"), + ("AM", "AM peak"), + ("MD", "Mid-day"), + ("PM", "PM peak"), + ("EV", "Evening")] + pb.add_select("period", options, title="Period:") + + root_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database", "emmebank")) + options = [(scen.id, "%s - %s" % (scen.id, scen.title)) for scen in main_emmebank.scenarios()] + pb.add_select("base_scenario_id", options, + title="Base scenario (with traffic and transit data):", + note="With period traffic results from main (traffic assignment) database at:
%s" % main_emmebank.path) + + pb.add_text_box("scenario_id", title="ID for transit assignment scenario:") + pb.add_text_box("scenario_title", title="Scenario title:", size=80) + pb.add_text_box("data_table_name", title="Data table prefix name:", note="Default is the ScenarioYear") + pb.add_checkbox("overwrite", title=" ", label="Overwrite existing scenario") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + root_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database", "emmebank")) + base_scenario = main_emmebank.scenario(self.base_scenario_id) + transit_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database_transit", "emmebank")) + results = self( + self.period, base_scenario, transit_emmebank, + self.scenario_id, self.scenario_title, + self.data_table_name, self.overwrite) + run_msg = "Transit scenario created" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, period, base_scenario, transit_emmebank, scenario_id, scenario_title, + data_table_name, overwrite=False): + modeller = _m.Modeller() + attrs = { + "period": period, + "base_scenario_id": base_scenario.id, + "transit_emmebank": transit_emmebank.path, + "scenario_id": scenario_id, + "scenario_title": scenario_title, + "data_table_name": data_table_name, + "overwrite": overwrite, + "self": str(self) + } + with _m.logbook_trace("Build transit network for period %s" % period, attributes=attrs): + gen_utils.log_snapshot("Build transit network", str(self), attrs) + copy_scenario = modeller.tool( + "inro.emme.data.scenario.copy_scenario") + periods = ["EA", "AM", "MD", "PM", "EV"] + if not period in periods: + raise Exception( + 'period: unknown value - specify one of %s' % periods) + + transit_assignment = modeller.tool( + "sandag.assignment.transit_assignment") + if transit_emmebank.scenario(scenario_id): + if overwrite: + transit_emmebank.delete_scenario(scenario_id) + else: + raise Exception("scenario_id: scenario %s already exists" % scenario_id) + + scenario = transit_emmebank.create_scenario(scenario_id) + scenario.title = scenario_title[:80] + scenario.has_traffic_results = base_scenario.has_traffic_results + scenario.has_transit_results = base_scenario.has_transit_results + for attr in sorted(base_scenario.extra_attributes(), key=lambda x: x._id): + dst_attr = scenario.create_extra_attribute(attr.type, attr.name, attr.default_value) + dst_attr.description = attr.description + for field in base_scenario.network_fields(): + scenario.create_network_field(field.type, field.name, field.atype, field.description) + network = base_scenario.get_network() + new_attrs = [ + ("TRANSIT_LINE", "@xfer_from_day", "Fare for xfer from daypass/trolley"), + ("TRANSIT_LINE", "@xfer_from_premium", "Fare for first xfer from premium"), + ("TRANSIT_LINE", "@xfer_from_coaster", "Fare for first xfer from coaster"), + ("TRANSIT_LINE", "@xfer_regional_pass", "0-fare for regional pass"), + ("TRANSIT_SEGMENT", "@xfer_from_bus", "Fare for first xfer from bus"), + ("TRANSIT_SEGMENT", "@headway_seg", "Headway adj for special xfers"), + ("TRANSIT_SEGMENT", "@transfer_penalty_s", "Xfer pen adj for special xfers"), + ("TRANSIT_SEGMENT", "@layover_board", "Boarding cost adj for special xfers"), + ("NODE", "@network_adj", "Model: 1=TAP adj, 2=circle, 3=timedxfer"), + ("NODE", "@network_adj_src", "Orig src node for timedxfer splits"), + ] + for elem, name, desc in new_attrs: + attr = scenario.create_extra_attribute(elem, name) + attr.description = desc + network.create_attribute(elem, name) + network.create_attribute("TRANSIT_LINE", "xfer_from_bus") + self._init_node_id(network) + + transit_passes = gen_utils.DataTableProc("%s_transit_passes" % data_table_name) + transit_passes = {row["pass_type"]: row["cost"] for row in transit_passes} + day_pass = float(transit_passes["day_pass"]) / 2.0 + regional_pass = float(transit_passes["regional_pass"]) / 2.0 + params = transit_assignment.get_perception_parameters(period) + mode_groups = transit_assignment.group_modes_by_fare(network, day_pass) + + bus_fares = {} + for mode_id, fares in mode_groups["bus"]: + for fare, count in fares.items(): + bus_fares[fare] = bus_fares.get(fare, 0) + count + # set nominal bus fare as unweighted average of two most frequent fares + bus_fares = sorted(bus_fares.items(), key=lambda x: x[1], reverse=True) + + if len(bus_fares) >= 2: + bus_fare = (bus_fares[0][0] + bus_fares[1][0]) / 2 + elif len(bus_fares) == 1: # unless there is only one fare value, in which case use that one + bus_fare = bus_fares[0][0] + else: + bus_fare = 0 + # find max premium mode fare + premium_fare = 0 + for mode_id, fares in mode_groups["premium"]: + for fare in fares.keys(): + premium_fare = max(premium_fare, fare) + # find max coaster_fare by checking the cumulative fare along each line + coaster_fare = 0 + for line in network.transit_lines(): + if line.mode.id != "c": + continue + segments = line.segments() + first = segments.next() + fare = first["@coaster_fare_board"] + for seg in segments: + fare += seg["@coaster_fare_inveh"] + coaster_fare = max(coaster_fare, fare) + + bus_fare_modes = [x[0] for x in mode_groups["bus"]] # have a bus fare, less than the day pass + day_pass_modes = [x[0] for x in mode_groups["day_pass"]] # boarding fare is the same as the day pass + premium_fare_modes = ["c"] + [x[0] for x in mode_groups["premium"]] # special premium services not covered by day pass + + for line in list(network.transit_lines()): + # remove the "unavailable" lines in this period + if line[params["xfer_headway"]] == 0: + network.delete_transit_line(line) + continue + # Adjust fare perception by VOT + line[params["fare"]] = line[params["fare"]] / params["vot"] + # set the fare increments for transfer combinations with day pass / regional pass + if line.mode.id in bus_fare_modes: + line["xfer_from_bus"] = max(min(day_pass - line["@fare"], line["@fare"]), 0) + line["@xfer_from_day"] = 0.0 + line["@xfer_from_premium"] = max(min(regional_pass - premium_fare, line["@fare"]), 0) + line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) + elif line.mode.id in day_pass_modes: + line["xfer_from_bus"] = max(day_pass - bus_fare, 0.0) + line["@xfer_from_day"] = 0.0 + line["@xfer_from_premium"] = max(min(regional_pass - premium_fare, line["@fare"]), 0) + line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) + elif line.mode.id in premium_fare_modes: + if line["@fare"] > day_pass or line.mode.id == "c": + # increment from bus to regional + line["xfer_from_bus"] = max(regional_pass - bus_fare, 0) + line["@xfer_from_day"] = max(regional_pass - day_pass, 0) + else: + # some "premium" modes lines are really regular fare + # increment from bus to day pass + line["xfer_from_bus"] = max(day_pass - bus_fare, 0) + line["@xfer_from_day"] = 0.0 + line["@xfer_from_premium"] = max(regional_pass - premium_fare, 0) + line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) + + for segment in network.transit_segments(): + line = segment.line + segment["@headway_seg"] = line[params["xfer_headway"]] + segment["@transfer_penalty_s"] = line["@transfer_penalty"] + segment["@xfer_from_bus"] = line["xfer_from_bus"] + network.delete_attribute("TRANSIT_LINE", "xfer_from_bus") + + self.taps_to_centroids(network) + # changed to allow timed xfers for different periods + timed_transfers_with_walk = list(gen_utils.DataTableProc("%s_timed_xfer_%s" % (data_table_name,period))) + self.timed_transfers(network, timed_transfers_with_walk, period) + #self.connect_circle_lines(network) + self.duplicate_tap_adajcent_stops(network) + # The fixed guideway travel times are stored in "@trtime_link_xx" + # and copied to data2 (ul2) for the ttf + # The congested auto times for mixed traffic are in "@auto_time" + # (output from traffic assignment) which needs to be copied to auto_time (a.k.a. timau) + # (The auto_time attribute is generated from the VDF values which include reliability factor) + src_attrs = [params["fixed_link_time"]] + dst_attrs = ["data2"] + if scenario.has_traffic_results and "@auto_time" in scenario.attributes("LINK"): + src_attrs.append("@auto_time") + dst_attrs.append("auto_time") + values = network.get_attribute_values("LINK", src_attrs) + network.set_attribute_values("LINK", dst_attrs, values) + scenario.publish_network(network) + + return scenario + + @_m.logbook_trace("Convert TAP nodes to centroids") + def taps_to_centroids(self, network): + # delete existing traffic centroids + for centroid in list(network.centroids()): + network.delete_node(centroid, cascade=True) + + node_attrs = network.attributes("NODE") + link_attrs = network.attributes("LINK") + for node in list(network.nodes()): + if node["@tap_id"] > 0: + centroid = network.create_node(node["@tap_id"], is_centroid=True) + for attr in node_attrs: + centroid[attr] = node[attr] + for link in node.outgoing_links(): + connector = network.create_link(centroid, link.j_node, link.modes) + connector.vertices = link.vertices + for attr in link_attrs: + connector[attr] = link[attr] + for link in node.incoming_links(): + connector = network.create_link(link.i_node, centroid, link.modes) + connector.vertices = link.vertices + for attr in link_attrs: + connector[attr] = link[attr] + network.delete_node(node, cascade=True) + + @_m.logbook_trace("Duplicate TAP access and transfer access stops") + def duplicate_tap_adajcent_stops(self, network): + # Expand network by duplicating TAP adjacent stops + network.create_attribute("NODE", "tap_stop", False) + all_transit_modes = set([mode for mode in network.modes() if mode.type == "TRANSIT"]) + access_mode = set([network.mode("a")]) + transfer_mode = network.mode("x") + walk_mode = network.mode("w") + + # Mark TAP adjacent stops and split TAP connectors + for centroid in network.centroids(): + out_links = list(centroid.outgoing_links()) + in_links = list(centroid.incoming_links()) + for link in out_links + in_links: + link.length = 0.0005 # setting length so that connector access time = 0.01 + for link in out_links: + real_stop = link.j_node + has_adjacent_transfer_links = False + has_adjacent_walk_links = False + for stop_link in real_stop.outgoing_links(): + if stop_link == link.reverse_link: + continue + if transfer_mode in stop_link.modes : + has_adjacent_transfer_links = True + if walk_mode in stop_link.modes : + has_adjacent_walk_links = True + + if has_adjacent_transfer_links or has_adjacent_walk_links: + length = link.length + tap_stop = network.split_link(centroid, real_stop, self._get_node_id(), include_reverse=True) + tap_stop["@network_adj"] = 1 + real_stop.tap_stop = tap_stop + transit_access_link = network.link(real_stop, tap_stop) + for link in transit_access_link, transit_access_link.reverse_link: + link.modes = all_transit_modes + link.length = 0 + for p in ["ea", "am", "md", "pm", "ev"]: + link["@time_link_" + p] = 0 + access_link = network.link(tap_stop, centroid) + access_link.modes = access_mode + access_link.reverse_link.modes = access_mode + access_link.length = length + access_link.reverse_link.length = length + + line_attributes = network.attributes("TRANSIT_LINE") + seg_attributes = network.attributes("TRANSIT_SEGMENT") + + # re-route the transit lines through the new TAP-stops + for line in network.transit_lines(): + # store line and segment data for re-routing + line_data = dict((k, line[k]) for k in line_attributes) + line_data["id"] = line.id + line_data["vehicle"] = line.vehicle + + seg_data = {} + itinerary = [] + tap_adjacent_stops = [] + + for seg in line.segments(include_hidden=True): + seg_data[(seg.i_node, seg.j_node, seg.loop_index)] = \ + dict((k, seg[k]) for k in seg_attributes) + itinerary.append(seg.i_node.number) + if seg.i_node.tap_stop and seg.allow_boardings: + # insert tap_stop, real_stop loop after tap_stop + real_stop = seg.i_node + tap_stop = real_stop.tap_stop + itinerary.extend([tap_stop.number, real_stop.number]) + tap_adjacent_stops.append(len(itinerary) - 1) # index of "real" stop in itinerary + + if tap_adjacent_stops: + network.delete_transit_line(line) + new_line = network.create_transit_line( + line_data.pop("id"), + line_data.pop("vehicle"), + itinerary) + for k, v in line_data.iteritems(): + new_line[k] = v + + for seg in new_line.segments(include_hidden=True): + data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) + for k, v in data.iteritems(): + seg[k] = v + for index in tap_adjacent_stops: + access_seg = new_line.segment(index - 2) + egress_seg = new_line.segment(index - 1) + real_seg = new_line.segment(index) + for k in seg_attributes: + access_seg[k] = egress_seg[k] = real_seg[k] + access_seg.allow_boardings = False + access_seg.allow_alightings = True + access_seg.transit_time_func = 3 + access_seg.dwell_time = real_seg.dwell_time + egress_seg.allow_boardings = True + egress_seg.allow_alightings = True + egress_seg.transit_time_func = 3 + egress_seg.dwell_time = 0 + real_seg.allow_boardings = True + real_seg.allow_alightings = False + real_seg.dwell_time = 0 + + network.delete_attribute("NODE", "tap_stop") + + @_m.logbook_trace("Add timed-transfer links", save_arguments=True) + def timed_transfers(self, network, timed_transfers_with_walk, period): + no_walk_link_error = "no walk link from line %s to %s" + node_not_found_error = "node %s not found in itinerary for line %s; "\ + "the to_line may end at the transfer stop" + + def find_walk_link(from_line, to_line): + to_nodes = set([s.i_node for s in to_line.segments(True) + if s.allow_boardings]) + link_candidates = [] + for seg in from_line.segments(True): + if not s.allow_alightings: + continue + for link in seg.i_node.outgoing_links(): + if link.j_node in to_nodes: + link_candidates.append(link) + if not link_candidates: + raise Exception(no_walk_link_error % (from_line, to_line)) + # if there are multiple connecting links take the shortest one + return sorted(link_candidates, key=lambda x: x.length)[0] + + def link_on_line(line, node, near_side_stop): + node = network.node(node) + if near_side_stop: + for seg in line.segments(): + if seg.j_node == node: + return seg.link + else: + for seg in line.segments(): + if seg.i_node == node: + return seg.link + raise Exception(node_not_found_error % (node, line)) + + # Group parallel transfers together (same pair of alighting-boarding nodes from the same line) + walk_transfers = _defaultdict(lambda: []) + for i, transfer in enumerate(timed_transfers_with_walk, start=1): + try: + from_line = network.transit_line(transfer["from_line"]) + if not from_line: + raise Exception("from_line %s does not exist" % transfer["from_line"]) + to_line = network.transit_line(transfer["to_line"]) + if not to_line: + raise Exception("to_line %s does not exist" % transfer["to_line"]) + walk_link = find_walk_link(from_line, to_line) + from_link = link_on_line(from_line, walk_link.i_node, near_side_stop=True) + to_link = link_on_line(to_line, walk_link.j_node, near_side_stop=False) + walk_transfers[(from_link, to_link)].append({ + "to_line": to_line, + "from_line": from_line, + "walk_link": walk_link, + "wait": transfer["wait_time"], + }) + except Exception as error: + new_message = "Timed transfer[%s]: %s" % (i, error.message) + raise type(error), type(error)(new_message), sys.exc_info()[2] + + # If there is only one transfer at the location (redundant case) + # OR all transfers are from the same line (can have different waits) + # OR all transfers are to the same line and have the same wait + # Merge all transfers onto the same transfer node + network_transfers = [] + for (from_link, to_link), transfers in walk_transfers.iteritems(): + walk_links = set([t["walk_link"] for t in transfers]) + from_lines = set([t["from_line"] for t in transfers]) + to_lines = set([t["to_line"] for t in transfers]) + waits = set(t["wait"] for t in transfers) + if len(transfers) == 1 or len(from_lines) == 1 or (len(to_lines) == 1 and len(waits) == 1): + network_transfers.append({ + "from_link": from_link, + "to_link": to_link, + "to_lines": list(to_lines), + "from_lines": list(from_lines), + "walk_link": walk_links.pop(), + "wait": dict((t["to_line"], t["wait"]) for t in transfers)}) + else: + for transfer in transfers: + network_transfers.append({ + "from_link": from_link, + "to_link": to_link, + "to_lines": [transfer["to_line"]], + "from_lines": [transfer["from_line"]], + "walk_link": transfer["walk_link"], + "wait": {transfer["to_line"]: transfer["wait"]}}) + + def split_link(link, node_id, lines, split_links, stop_attr, waits=None): + near_side_stop = (stop_attr == "allow_alightings") + orig_link = link + if link in split_links: + link = split_links[link] + i_node, j_node = link.i_node, link.j_node + length = link.length + proportion = min(0.006 / length, 0.2) + if near_side_stop: + proportion = 1 - proportion + new_node = network.split_link(i_node, j_node, node_id, False, proportion) + new_node["@network_adj"] = 3 + new_node["@network_adj_src"] = orig_link.j_node.number if near_side_stop else orig_link.i_node.number + in_link = network.link(i_node, new_node) + out_link = network.link(new_node, j_node) + split_links[orig_link] = in_link if near_side_stop else out_link + if near_side_stop: + in_link.length = length + out_link.length = 0 + for p in ["ea", "am", "md", "pm", "ev"]: + out_link["@trtime_link_" + p] = 0 + else: + out_link.length = length + in_link.length = 0 + for p in ["ea", "am", "md", "pm", "ev"]: + in_link["@trtime_link_" + p] = 0 + + for seg in in_link.segments(): + if not near_side_stop: + seg.transit_time_func = 3 + seg["@coaster_fare_inveh"] = 0 + for seg in out_link.segments(): + if near_side_stop: + seg.transit_time_func = 3 + seg.allow_alightings = seg.allow_boardings = False + seg.dwell_time = 0 + if seg.line in lines: + seg[stop_attr] = True + if stop_attr == "allow_boardings": + seg["@headway_seg"] = float(waits[seg.line]) * 2 + return new_node + + # process the transfer points, split links and set attributes + split_links = {} + for transfer in network_transfers: + new_alight_node = split_link( + transfer["from_link"], self._get_node_id(), transfer["from_lines"], + split_links, "allow_alightings") + new_board_node = split_link( + transfer["to_link"], self._get_node_id(), transfer["to_lines"], + split_links, "allow_boardings", waits=transfer["wait"]) + walk_link = transfer["walk_link"] + transfer_link = network.create_link( + new_alight_node, new_board_node, [network.mode("x")]) + for attr in network.attributes("LINK"): + transfer_link[attr] = walk_link[attr] + + @_m.logbook_trace("Add circle line free layover transfers") + def connect_circle_lines(self, network): + network.create_attribute("NODE", "circle_lines") + line_attributes = network.attributes("TRANSIT_LINE") + seg_attributes = network.attributes("TRANSIT_SEGMENT") + + def offset_coords(node): + rho = math.sqrt(5000) + phi = 3 * math.pi / 4 + node.circle_lines * math.pi / 12 + x = node.x + rho * math.cos(phi) + y = node.y + rho * math.sin(phi) + node.circle_lines += 1 + return(x, y) + + transit_lines = list(network.transit_lines()) + for line in transit_lines: + first_seg = line.segment(0) + last_seg = line.segment(-1) + if first_seg.i_node == last_seg.i_node: + # Add new node, offset from existing node + start_node = line.segment(0).i_node + xfer_node = network.create_node(self._get_node_id(), False) + xfer_node["@network_adj"] = 2 + xfer_node.x, xfer_node.y = offset_coords(start_node) + network.create_link(start_node, xfer_node, [line.vehicle.mode]) + network.create_link(xfer_node, start_node, [line.vehicle.mode]) + + # copy transit line data, re-route itinerary to and from new node + line_data = dict((k, line[k]) for k in line_attributes) + line_data["id"] = line.id + line_data["vehicle"] = line.vehicle + first_seg.allow_boardings = True + first_seg.allow_alightings = False + first_seg_data = dict((k, first_seg[k]) for k in seg_attributes) + first_seg_data.update({ + "@headway_seg": 0.01, "dwell_time": 0, "transit_time_func": 3, + "@transfer_penalty_s": 0, "@xfer_from_bus": 0, "@layover_board": 1 + }) + last_seg.allow_boardings = False + last_seg.allow_alightings = True + last_seg_data = dict((k, last_seg[k]) for k in seg_attributes) + last_seg_data.update({ + "@headway_seg": 0.01, "dwell_time": 5.0, "transit_time_func": 3 + # incremental dwell time for layover of 5 min + # Note: some lines seem to have a layover of 0, most of 5 mins + }) + seg_data = { + (xfer_node, start_node, 1): first_seg_data, + (xfer_node, None, 1): last_seg_data} + itinerary = [xfer_node.number] + for seg in line.segments(): + seg_data[(seg.i_node, seg.j_node, seg.loop_index)] = dict((k, seg[k]) for k in seg_attributes) + itinerary.append(seg.i_node.number) + last_seg = line.segment(-1) + seg_data[(last_seg.i_node, xfer_node, 1)] = dict((k, last_seg[k]) for k in seg_attributes) + seg_data[(last_seg.i_node, xfer_node, 1)]["transit_time_func"] = 3 + itinerary.extend([last_seg.i_node.number, xfer_node.number]) + + network.delete_transit_line(line) + new_line = network.create_transit_line( + line_data.pop("id"), line_data.pop("vehicle"), itinerary) + for k, v in line_data.iteritems(): + new_line[k] = v + for seg in new_line.segments(include_hidden=True): + data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) + for k, v in data.iteritems(): + seg[k] = v + + network.delete_attribute("NODE", "circle_lines") + + def _init_node_id(self, network): + new_node_id = max(n.number for n in network.nodes()) + self._new_node_id = math.ceil(new_node_id / 10000.0) * 10000 + + def _get_node_id(self): + self._new_node_id += 1 + return self._new_node_id diff --git a/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py b/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py new file mode 100644 index 0000000..cdf652e --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py @@ -0,0 +1,1087 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// traffic_assignment.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# The Traffic assignment tool runs the traffic assignment and skims per +# period on the current primary scenario. +# +# The traffic assignment is a 15-class assignment with generalized cost on +# links and BPR-type volume-delay functions which include capacities on links +# and at intersection approaches. The assignment is run using the +# fast-converging Second-Order Linear Approximation (SOLA) method in Emme to +# a relative gap of 5x10-4. The per-link fixed costs include toll values and +# operating costs which vary by class of demand. +# Assignment matrices and resulting network flows are always in PCE. +# +# Inputs: +# period: the time-of-day period, one of EA, AM, MD, PM, EV. +# msa_iteration: global iteration number. If greater than 1, existing flow +# values must be present and the resulting flows on links and turns will +# be the weighted average of this assignment and the existing values. +# relative_gap: minimum relative stopping criteria. +# max_iterations: maximum iterations stopping criteria. +# num_processors: number of processors to use for the traffic assignments. +# select_link: specify one or more select link analysis setups as a list of +# specifications with three keys: +# "expression": selection expression to identify the link(s) of interest. +# "suffix": the suffix to use in the naming of per-class result +# attributes and matrices, up to 6 characters. +# "threshold": the minimum number of links which must be encountered +# for the path selection. +# Example: +# select_link = [ +# {"expression": "@tov_id=4578 or @tcov_id=9203", "suffix": "fwy", "threshold": "1"} +# ] +# raise_zero_dist: if checked, the distance skim for the SOVGP is checked for +# any zero values, which would indicate a disconnected zone, in which case +# an error is raised and the model run is halted. +# +# Matrices: +# All traffic demand and skim matrices. +# See list of classes under __call__ method, or matrix list under report method. +# +# Script example: +""" +import inro.modeller as _m +import os +import inro.emme.database.emmebank as _eb + +modeller = _m.Modeller() +desktop = modeller.desktop +traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") +export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") +load_properties = modeller.tool('sandag.utilities.properties') +project_dir = os.path.dirname(_m.Modeller().desktop.project.path) +main_directory = os.path.dirname(project_dir) +output_dir = os.path.join(main_directory, "output") +props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + + +main_emmebank = os.path.join(project_dir, "Database", "emmebank") +scenario_id = 100 +base_scenario = main_emmebank.scenario(scenario_id) + +periods = ["EA", "AM", "MD", "PM", "EV"] +period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) + +msa_iteration = 1 +relative_gap = 0.0005 +max_assign_iterations = 100 +num_processors = "MAX-1" +select_link = None # Optional select link specification + +for number, period in period_ids: + period_scenario = main_emmebank.scenario(number) + traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, + num_processors, period_scenario, select_link) + omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) + if msa_iteration < 4: + export_traffic_skims(period, omx_file, base_scenario) +""" + + +TOOLBOX_ORDER = 20 + + +import inro.modeller as _m +import inro.emme.core.exception as _except +import traceback as _traceback +from contextlib import contextmanager as _context +import numpy +import array +import os +import json as _json + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class TrafficAssignment(_m.Tool(), gen_utils.Snapshot): + + period = _m.Attribute(unicode) + msa_iteration = _m.Attribute(int) + relative_gap = _m.Attribute(float) + max_iterations = _m.Attribute(int) + num_processors = _m.Attribute(str) + select_link = _m.Attribute(unicode) + raise_zero_dist = _m.Attribute(bool) + stochastic = _m.Attribute(bool) + input_directory = _m.Attribute(str) + + tool_run_msg = "" + + def __init__(self): + self.msa_iteration = 1 + self.relative_gap = 0.0005 + self.max_iterations = 100 + self.num_processors = "MAX-1" + self.raise_zero_dist = True + self.select_link = '[]' + self.stochastic = False + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.attributes = ["period", "msa_iteration", "relative_gap", "max_iterations", + "num_processors", "select_link", "raise_zero_dist", "stochastic", "input_directory"] + version = os.environ.get("EMMEPATH", "") + self._version = version[-5:] if version else "" + self._skim_classes_separately = True # Used for debugging only + self._stats = {} + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Traffic assignment" + pb.description = """ +The Traffic assignment tool runs the traffic assignment and skims per +period on the current primary scenario. +
+The traffic assignment is a 15-class assignment with generalized cost on +links and BPR-type volume-delay functions which include capacities on links +and at intersection approaches. The assignment is run using the +fast-converging Second-Order Linear Approximation (SOLA) method in Emme to +a relative gap of 5x10-4. The per-link fixed costs include toll values and +operating costs which vary by class of demand. +Assignment matrices and resulting network flows are always in PCE. +""" + pb.branding_text = "- SANDAG - Assignment" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + options = [("EA","Early AM"), + ("AM","AM peak"), + ("MD","Mid-day"), + ("PM","PM peak"), + ("EV","Evening")] + pb.add_select("period", options, title="Period:") + pb.add_text_box("msa_iteration", title="MSA iteration:", note="If >1 will apply MSA to flows.") + pb.add_text_box("relative_gap", title="Relative gap:") + pb.add_text_box("max_iterations", title="Max iterations:") + dem_utils.add_select_processors("num_processors", pb, self) + pb.add_checkbox("raise_zero_dist", title=" ", label="Raise on zero distance value", + note="Check for and raise an exception if a zero value is found in the SOVGP_DIST matrix.") + pb.add_checkbox( + 'stochastic', + title=" ", + label="Run as a stochastic assignment", + note="If the current MSA iteration is the last (4th) one, the SOLA traffic assignment is replaced with a stochastic traffic assignment." + ) + pb.add_select_file('input_directory', 'directory', title='Select input directory') + self._add_select_link_interface(pb) + return pb.render() + + + def _add_select_link_interface(self, pb): + pb.add_html(""" +""") + pb.add_text_box("select_link", multi_line=True) + pb.wrap_html(title="Select link(s):", + body=""" + + + +

+ +
+
+
+ Click for available attributes +
+
+ +
+
""") + pb.add_html(""" +""" % {"tool_proxy_tag": pb.tool_proxy_tag, }) + + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + results = self(self.period, self.msa_iteration, self.relative_gap, self.max_iterations, + self.num_processors, scenario, self.select_link, self.raise_zero_dist, + self.stochastic, self.input_directory) + run_msg = "Traffic assignment completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, period, msa_iteration, relative_gap, max_iterations, num_processors, scenario, + select_link=[], raise_zero_dist=True, stochastic=False, input_directory=None): + select_link = _json.loads(select_link) if isinstance(select_link, basestring) else select_link + attrs = { + "period": period, + "msa_iteration": msa_iteration, + "relative_gap": relative_gap, + "max_iterations": max_iterations, + "num_processors": num_processors, + "scenario": scenario.id, + "select_link": _json.dumps(select_link), + "raise_zero_dist": raise_zero_dist, + "stochastic": stochastic, + "input_directory": input_directory, + "self": str(self) + } + self._stats = {} + with _m.logbook_trace("Traffic assignment for period %s" % period, attributes=attrs): + gen_utils.log_snapshot("Traffic assignment", str(self), attrs) + periods = ["EA", "AM", "MD", "PM", "EV"] + if not period in periods: + raise _except.ArgumentError( + 'period: unknown value - specify one of %s' % periods) + num_processors = dem_utils.parse_num_processors(num_processors) + # Main list of assignment classes + classes = [ + { # 0 + "name": 'SOV_NT_L', "mode": 's', "PCE": 1, "VOT": 8.81, "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 1 + "name": 'SOV_TR_L', "mode": 'S', "PCE": 1, "VOT": 8.81, "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 2 + "name": 'HOV2_L', "mode": 'H', "PCE": 1, "VOT": 8.81, "cost": '@cost_hov2', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] + }, + { # 3 + "name": 'HOV3_L', "mode": 'I', "PCE": 1, "VOT": 8.81, "cost": '@cost_hov3', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] + }, + { # 4 + "name": 'SOV_NT_M', "mode": 's', "PCE": 1, "VOT": 18.0, "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 5 + "name": 'SOV_TR_M', "mode": 'S', "PCE": 1, "VOT": 18.0, "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 6 + "name": 'HOV2_M', "mode": 'H', "PCE": 1, "VOT": 18.0, "cost": '@cost_hov2', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] + }, + { # 7 + "name": 'HOV3_M', "mode": 'I', "PCE": 1, "VOT": 18.0, "cost": '@cost_hov3', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] + }, + { # 8 + "name": 'SOV_NT_H', "mode": 's', "PCE": 1, "VOT": 85., "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 9 + "name": 'SOV_TR_H', "mode": 'S', "PCE": 1, "VOT": 85., "cost": '@cost_auto', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] + }, + { # 10 + "name": 'HOV2_H', "mode": 'H', "PCE": 1, "VOT": 85., "cost": '@cost_hov2', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] + }, + { # 11 + "name": 'HOV3_H', "mode": 'I', "PCE": 1, "VOT": 85., "cost": '@cost_hov3', + "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] + }, + { # 12 + "name": 'TRK_L', "mode": 'T', "PCE": 1.3, "VOT": 67., "cost": '@cost_lgt_truck', + "skims": ["TIME", "DIST", "TOLLCOST.TRK_L"] + }, + { # 13 + "name": 'TRK_M', "mode": 'M', "PCE": 1.5, "VOT": 68., "cost": '@cost_med_truck', + "skims": ["TIME", "DIST", "TOLLCOST.TRK_M"] + }, + { # 14 + "name": 'TRK_H', "mode": 'V', "PCE": 2.5, "VOT": 89., "cost": '@cost_hvy_truck', + "skims": ["TIME", "DIST", "TOLLCOST.TRK_H"] + }, + ] + + # change mode to allow sovntp on SR125 + # TODO: incorporate this into import_network instead + # also, consider updating mode definitions + self.change_mode_sovntp(scenario) + + if period == "MD" and (msa_iteration == 1 or not scenario.mode('D')): + self.prepare_midday_generic_truck(scenario) + + if 1 < msa_iteration < 4: + # Link and turn flows + link_attrs = ["auto_volume"] + turn_attrs = ["auto_volume"] + for traffic_class in classes: + link_attrs.append("@%s" % (traffic_class["name"].lower())) + turn_attrs.append("@p%s" % (traffic_class["name"].lower())) + msa_link_flows = scenario.get_attribute_values("LINK", link_attrs)[1:] + msa_turn_flows = scenario.get_attribute_values("TURN", turn_attrs)[1:] + + if stochastic: + self.run_stochastic_assignment( + period, + relative_gap, + max_iterations, + num_processors, + scenario, + classes, + input_directory + ) + else: + self.run_assignment(period, relative_gap, max_iterations, num_processors, scenario, classes, select_link) + + + if 1 < msa_iteration < 4: + link_flows = scenario.get_attribute_values("LINK", link_attrs) + values = [link_flows.pop(0)] + for msa_array, flow_array in zip(msa_link_flows, link_flows): + msa_vals = numpy.frombuffer(msa_array, dtype='float32') + flow_vals = numpy.frombuffer(flow_array, dtype='float32') + result = msa_vals + (1.0 / msa_iteration) * (flow_vals - msa_vals) + result_array = array.array('f') + result_array.fromstring(result.tostring()) + values.append(result_array) + scenario.set_attribute_values("LINK", link_attrs, values) + + turn_flows = scenario.get_attribute_values("TURN", turn_attrs) + values = [turn_flows.pop(0)] + for msa_array, flow_array in zip(msa_turn_flows, turn_flows): + msa_vals = numpy.frombuffer(msa_array, dtype='float32') + flow_vals = numpy.frombuffer(flow_array, dtype='float32') + result = msa_vals + (1.0 / msa_iteration) * (flow_vals - msa_vals) + result_array = array.array('f') + result_array.fromstring(result.tostring()) + values.append(result_array) + scenario.set_attribute_values("TURN", turn_attrs, values) + + self.calc_network_results(period, num_processors, scenario) + + if msa_iteration <= 4: + self.run_skims(period, num_processors, scenario, classes) + self.report(period, scenario, classes) + # Check that the distance matrix is valid (no disconnected zones) + # Using SOVGPL class as representative + if raise_zero_dist: + name = "%s_SOV_TR_H_DIST" % period + dist_stats = self._stats[name] + if dist_stats[1] == 0: + zones = scenario.zone_numbers + matrix = scenario.emmebank.matrix(name) + data = matrix.get_numpy_data(scenario) + row, col = numpy.unravel_index(data.argmin(), data.shape) + row, col = zones[row], zones[col] + raise Exception("Disconnected zone error: 0 value found in matrix %s from zone %s to %s" % (name, row, col)) + + def run_assignment(self, period, relative_gap, max_iterations, num_processors, scenario, classes, select_link): + emmebank = scenario.emmebank + + modeller = _m.Modeller() + set_extra_function_para = modeller.tool( + "inro.emme.traffic_assignment.set_extra_function_parameters") + create_attribute = modeller.tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + traffic_assign = modeller.tool( + "inro.emme.traffic_assignment.sola_traffic_assignment") + net_calc = gen_utils.NetworkCalculator(scenario) + + if period in ["AM", "PM"]: + # For freeway links in AM and PM periods, convert VDF to type 25 + net_calc("vdf", "25", "vdf=10") + + p = period.lower() + assign_spec = self.base_assignment_spec( + relative_gap, max_iterations, num_processors) + with _m.logbook_trace("Prepare traffic data for period %s" % period): + with _m.logbook_trace("Input link attributes"): + # set extra attributes for the period for VDF + # ul1 = @time_link (period) + # ul2 = transit flow -> volad (for assignment only) + # ul3 = @capacity_link (period) + el1 = "@green_to_cycle" + el2 = "@sta_reliability" + el3 = "@capacity_inter" + set_extra_function_para(el1, el2, el3, emmebank=emmebank) + + # set green to cycle to el1=@green_to_cycle for VDF + att_name = "@green_to_cycle_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@green_to_cycle" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set static reliability to el2=@sta_reliability for VDF + att_name = "@sta_reliability_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@sta_reliability" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set capacity_inter to el3=@capacity_inter for VDF + att_name = "@capacity_inter_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@capacity_inter" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set link time + net_calc("ul1", "@time_link_%s" % p, "modes=d") + net_calc("ul3", "@capacity_link_%s" % p, "modes=d") + # set number of lanes (not used in VDF, just for reference) + net_calc("lanes", "@lane_%s" % p, "modes=d") + if period in ["EA", "MD", "EV"]: + # For links with signals inactive in the off-peak periods, convert VDF to type 11 + net_calc("vdf", "11", "modes=d and @green_to_cycle=0 and @traffic_control=4,5 and vdf=24") + # # Set HOV2 cost attribute + # create_attribute("LINK", "@cost_hov2_%s" % p, "toll (non-mngd) + cost for HOV2", + # 0, overwrite=True, scenario=scenario) + # net_calc("@cost_hov2_%s" % p, "@cost_hov_%s" % p, "modes=d") + # net_calc("@cost_hov2_%s" % p, "@cost_auto_%s" % p, "@lane_restriction=3") + + with _m.logbook_trace("Transit line headway and background traffic"): + # set headway for the period + hdw = {"ea": "@headway_op", + "am": "@headway_am", + "md": "@headway_op", + "pm": "@headway_pm", + "ev": "@headway_op"} + net_calc("hdw", hdw[p], {"transit_line": "all"}) + + # transit vehicle as background flow with periods + period_hours = {'ea': 3, 'am': 3, 'md': 6.5, 'pm': 3.5, 'ev': 5} + expression = "(60 / hdw) * vauteq * %s" % (period_hours[p]) + net_calc("ul2", "0", "modes=d") + net_calc("ul2", expression, + selections={"link": "modes=d", "transit_line": "hdw=0.02,9999"}, + aggregation="+") + + with _m.logbook_trace("Per-class flow attributes"): + for traffic_class in classes: + demand = 'mf"%s_%s"' % (period, traffic_class["name"]) + link_cost = "%s_%s" % (traffic_class["cost"], p) if traffic_class["cost"] else "@cost_operating" + + att_name = "@%s" % (traffic_class["name"].lower()) + att_des = "%s %s link volume" % (period, traffic_class["name"]) + link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) + att_name = "@p%s" % (traffic_class["name"].lower()) + att_des = "%s %s turn volume" % (period, traffic_class["name"]) + turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) + + class_spec = { + "mode": traffic_class["mode"], + "demand": demand, + "generalized_cost": { + "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] + }, + "results": { + "link_volumes": link_flow.id, "turn_volumes": turn_flow.id, + "od_travel_times": None + } + } + assign_spec["classes"].append(class_spec) + if select_link: + for class_spec in assign_spec["classes"]: + class_spec["path_analyses"] = [] + for sub_spec in select_link: + expr = sub_spec["expression"] + suffix = sub_spec["suffix"] + threshold = sub_spec["threshold"] + if not expr and not suffix: + continue + with _m.logbook_trace("Prepare for select link analysis '%s' - %s" % (expr, suffix)): + slink = create_attribute("LINK", "@slink_%s" % suffix, "selected link for %s" % suffix, 0, + overwrite=True, scenario=scenario) + net_calc(slink.id, "1", expr) + with _m.logbook_trace("Initialize result matrices and extra attributes"): + for traffic_class, class_spec in zip(classes, assign_spec["classes"]): + att_name = "@sl_%s_%s" % (traffic_class["name"].lower(), suffix) + att_des = "%s %s '%s' sel link flow"% (period, traffic_class["name"], suffix) + link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) + att_name = "@psl_%s_%s" % (traffic_class["name"].lower(), suffix) + att_des = "%s %s '%s' sel turn flow" % (period, traffic_class["name"], suffix) + turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) + + name = "SELDEM_%s_%s_%s" % (period, traffic_class["name"], suffix) + desc = "Selected demand for %s %s %s" % (period, traffic_class["name"], suffix) + seldem = dem_utils.create_full_matrix(name, desc, scenario=scenario) + + # add select link analysis + class_spec["path_analyses"].append({ + "link_component": slink.id, + "turn_component": None, + "operator": "+", + "selection_threshold": { "lower": threshold, "upper": 999999}, + "path_to_od_composition": { + "considered_paths": "SELECTED", + "multiply_path_proportions_by": {"analyzed_demand": True, "path_value": False} + }, + "analyzed_demand": None, + "results": { + "selected_link_volumes": link_flow.id, + "selected_turn_volumes": turn_flow.id, + "od_values": seldem.named_id + } + }) + # Run assignment + traffic_assign(assign_spec, scenario, chart_log_interval=2) + return + + def run_stochastic_assignment( + self, period, relative_gap, max_iterations, num_processors, scenario, + classes, input_directory): + load_properties = _m.Modeller().tool('sandag.utilities.properties') + main_directory = os.path.dirname(input_directory) + props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + distribution_type = props['stochasticHighwayAssignment.distributionType'] + replications = props['stochasticHighwayAssignment.replications'] + a_parameter = props['stochasticHighwayAssignment.aParameter'] + b_parameter = props['stochasticHighwayAssignment.bParameter'] + seed = props['stochasticHighwayAssignment.seed'] + + emmebank = scenario.emmebank + + modeller = _m.Modeller() + set_extra_function_para = modeller.tool( + "inro.emme.traffic_assignment.set_extra_function_parameters") + create_attribute = modeller.tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + traffic_assign = modeller.tool( + "solutions.stochastic_traffic_assignment") + net_calc = gen_utils.NetworkCalculator(scenario) + + if period in ["AM", "PM"]: + # For freeway links in AM and PM periods, convert VDF to type 25 + net_calc("vdf", "25", "vdf=10") + + p = period.lower() + assign_spec = self.base_assignment_spec( + relative_gap, max_iterations, num_processors) + assign_spec['background_traffic'] = { + "link_component": None, + "turn_component": None, + "add_transit_vehicles": True + } + with _m.logbook_trace("Prepare traffic data for period %s" % period): + with _m.logbook_trace("Input link attributes"): + # set extra attributes for the period for VDF + # ul1 = @time_link (period) + # ul2 = transit flow -> volad (for assignment only) + # ul3 = @capacity_link (period) + el1 = "@green_to_cycle" + el2 = "@sta_reliability" + el3 = "@capacity_inter" + set_extra_function_para(el1, el2, el3, emmebank=emmebank) + + # set green to cycle to el1=@green_to_cycle for VDF + att_name = "@green_to_cycle_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@green_to_cycle" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set static reliability to el2=@sta_reliability for VDF + att_name = "@sta_reliability_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@sta_reliability" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set capacity_inter to el3=@capacity_inter for VDF + att_name = "@capacity_inter_%s" % p + att = scenario.extra_attribute(att_name) + new_att_name = "@capacity_inter" + create_attribute("LINK", new_att_name, att.description, + 0, overwrite=True, scenario=scenario) + net_calc(new_att_name, att_name, "modes=d") + # set link time + net_calc("ul1", "@time_link_%s" % p, "modes=d") + net_calc("ul3", "@capacity_link_%s" % p, "modes=d") + # set number of lanes (not used in VDF, just for reference) + net_calc("lanes", "@lane_%s" % p, "modes=d") + if period in ["EA", "MD", "EV"]: + # For links with signals inactive in the off-peak periods, convert VDF to type 11 + net_calc("vdf", "11", "modes=d and @green_to_cycle=0 and @traffic_control=4,5 and vdf=24") + # # Set HOV2 cost attribute + # create_attribute("LINK", "@cost_hov2_%s" % p, "toll (non-mngd) + cost for HOV2", + # 0, overwrite=True, scenario=scenario) + # net_calc("@cost_hov2_%s" % p, "@cost_hov_%s" % p, "modes=d") + # net_calc("@cost_hov2_%s" % p, "@cost_auto_%s" % p, "@lane_restriction=3") + + with _m.logbook_trace("Transit line headway and background traffic"): + # set headway for the period: format is (attribute_name, period duration in hours) + hdw = {"ea": ("@headway_op", 3), + "am": ("@headway_am", 3), + "md": ("@headway_op", 6.5), + "pm": ("@headway_pm", 3.5), + "ev": ("@headway_op", 5)} + net_calc('ul2', '0', {'link': 'all'}) + net_calc('hdw', '9999.99', {'transit_line': 'all'}) + net_calc( + 'hdw', "{hdw} / {p} ".format(hdw=hdw[p][0], p=hdw[p][1]), + {"transit_line": "%s=0.02,9999" % hdw[p][0]} + ) + + with _m.logbook_trace("Per-class flow attributes"): + for traffic_class in classes: + demand = 'mf"%s_%s"' % (period, traffic_class["name"]) + link_cost = "%s_%s" % (traffic_class["cost"], p) if traffic_class["cost"] else "@cost_operating" + + att_name = "@%s" % (traffic_class["name"].lower()) + att_des = "%s %s link volume" % (period, traffic_class["name"]) + link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) + att_name = "@p%s" % (traffic_class["name"].lower()) + att_des = "%s %s turn volume" % (period, traffic_class["name"]) + turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) + + class_spec = { + "mode": traffic_class["mode"], + "demand": demand, + "generalized_cost": { + "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] + }, + "results": { + "link_volumes": link_flow.id, "turn_volumes": turn_flow.id, + "od_travel_times": None + } + } + assign_spec["classes"].append(class_spec) + + # Run assignment + traffic_assign( + assign_spec, + dist_par={'type': distribution_type, 'A': a_parameter, 'B': b_parameter}, + replications=replications, + seed=seed, + orig_func=False, + random_term='ul2', + compute_travel_times=False, + scenario=scenario + ) + + with _m.logbook_trace("Reset transit line headways"): + # set headway for the period + hdw = {"ea": "@headway_op", + "am": "@headway_am", + "md": "@headway_op", + "pm": "@headway_pm", + "ev": "@headway_op"} + net_calc("hdw", hdw[p], {"transit_line": "all"}) + return + + def calc_network_results(self, period, num_processors, scenario): + modeller = _m.Modeller() + create_attribute = modeller.tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + net_calc = gen_utils.NetworkCalculator(scenario) + emmebank = scenario.emmebank + p = period.lower() + # ul2 is the total flow (volau + volad) in the skim assignment + with _m.logbook_trace("Calculation of attributes for skims"): + link_attributes = [ + ("@hovdist", "distance for HOV"), + ("@tollcost", "Toll cost for SOV autos"), + ("@h2tollcost", "Toll cost for hov2"), + ("@h3tollcost", "Toll cost for hov3"), + ("@trk_ltollcost", "Toll cost for light trucks"), + ("@trk_mtollcost", "Toll cost for medium trucks"), + ("@trk_htollcost", "Toll cost for heavy trucks"), + ("@mlcost", "Manage lane cost in cents"), + ("@tolldist", "Toll distance"), + ("@h2tolldist", "Toll distance for hov2"), + ("@h3tolldist", "Toll distance for hov3"), + ("@reliability", "Reliability factor"), + ("@reliability_sq", "Reliability factor squared"), + ("@auto_volume", "traffic link volume (volau)"), + ("@auto_time", "traffic link time (timau)"), + ] + for name, description in link_attributes: + create_attribute("LINK", name, description, + 0, overwrite=True, scenario=scenario) + create_attribute("TURN", "@auto_time_turn", "traffic turn time (ptimau)", + overwrite=True, scenario=scenario) + + net_calc("@hovdist", "length", {"link": "@lane_restriction=2,3"}) + net_calc("@tollcost", "@cost_auto_%s - @cost_operating" % p) + net_calc("@h2tollcost", "@cost_hov2_%s - @cost_operating" % p, {"link": "@lane_restriction=3,4"}) + net_calc("@h3tollcost", "@cost_hov3_%s - @cost_operating" % p, {"link": "@lane_restriction=4"}) + net_calc("@trk_ltollcost", "@cost_lgt_truck_%s - @cost_operating" % p) + net_calc("@trk_mtollcost", "@cost_med_truck_%s - @cost_operating" % p) + net_calc("@trk_htollcost", "@cost_hvy_truck_%s - @cost_operating" % p) + net_calc("@mlcost", "@toll_%s" % p, {"link": "not @lane_restriction=4"}) + net_calc("@tolldist", "length", {"link": "@lane_restriction=2,4"}) + net_calc("@h2tolldist", "length", {"link": "@lane_restriction=3,4"}) + net_calc("@h3tolldist", "length", {"link": "@lane_restriction=4"}) + net_calc("@auto_volume", "volau", {"link": "modes=d"}) + net_calc("ul2", "volau+volad", {"link": "modes=d"}) + vdfs = [f for f in emmebank.functions() if f.type == "VOLUME_DELAY"] + exf_pars = emmebank.extra_function_parameters + for function in vdfs: + expression = function.expression + for exf_par in ["el1", "el2", "el3"]: + expression = expression.replace(exf_par, getattr(exf_pars, exf_par)) + # split function into time component and reliability component + time_expr, reliability_expr = expression.split("*(1+@sta_reliability+") + net_calc("@auto_time", time_expr, {"link": "vdf=%s" % function.id[2:]}) + net_calc("@reliability", "(@sta_reliability+" + reliability_expr, + {"link": "vdf=%s" % function.id[2:]}) + + net_calc("@reliability_sq", "@reliability**2", {"link": "modes=d"}) + net_calc("@auto_time_turn", "ptimau*(ptimau.gt.0)", + {"incoming_link": "all", "outgoing_link": "all"}) + + def run_skims(self, period, num_processors, scenario, classes): + modeller = _m.Modeller() + traffic_assign = modeller.tool( + "inro.emme.traffic_assignment.sola_traffic_assignment") + emmebank = scenario.emmebank + p = period.lower() + analysis_link = { + "TIME": "@auto_time", + "DIST": "length", + "HOVDIST": "@hovdist", + "TOLLCOST.SOV": "@tollcost", + "TOLLCOST.HOV2": "@h2tollcost", + "TOLLCOST.HOV3": "@h3tollcost", + "TOLLCOST.TRK_L": "@trk_ltollcost", + "TOLLCOST.TRK_M": "@trk_mtollcost", + "TOLLCOST.TRK_H": "@trk_htollcost", + "MLCOST": "@mlcost", + "TOLLDIST": "@tolldist", + "TOLLDIST.HOV2": "@h2tolldist", + "TOLLDIST.HOV3": "@h3tolldist", + "REL": "@reliability_sq" + } + analysis_turn = {"TIME": "@auto_time_turn"} + with self.setup_skims(period, scenario): + if period == "MD": + gen_truck_mode = 'D' + classes.append({ + "name": 'TRK', "mode": gen_truck_mode, "PCE": 1, "VOT": 67., "cost": '', + "skims": ["TIME"] + }) + skim_spec = self.base_assignment_spec(0, 0, num_processors, background_traffic=False) + for traffic_class in classes: + if not traffic_class["skims"]: + continue + class_analysis = [] + if "GENCOST" in traffic_class["skims"]: + od_travel_times = 'mf"%s_%s_%s"' % (period, traffic_class["name"], "GENCOST") + traffic_class["skims"].remove("GENCOST") + else: + od_travel_times = None + for skim_type in traffic_class["skims"]: + skim_name = skim_type.split(".")[0] + class_analysis.append({ + "link_component": analysis_link.get(skim_type), + "turn_component": analysis_turn.get(skim_type), + "operator": "+", + "selection_threshold": {"lower": None, "upper": None}, + "path_to_od_composition": { + "considered_paths": "ALL", + "multiply_path_proportions_by": + {"analyzed_demand": False, "path_value": True} + }, + "results": { + "od_values": 'mf"%s_%s_%s"' % (period, traffic_class["name"], skim_name), + "selected_link_volumes": None, + "selected_turn_volumes": None + } + }) + if traffic_class["cost"]: + link_cost = "%s_%s" % (traffic_class["cost"], p) + else: + link_cost = "@cost_operating" + skim_spec["classes"].append({ + "mode": traffic_class["mode"], + "demand": 'ms"zero"', # 0 demand for skim with 0 iteration and fix flow in ul2 in vdf + "generalized_cost": { + "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] + }, + "results": { + "link_volumes": None, "turn_volumes": None, + "od_travel_times": {"shortest_paths": od_travel_times} + }, + "path_analyses": class_analysis, + }) + + # skim assignment + if self._skim_classes_separately: + # Debugging check + skim_classes = skim_spec["classes"][:] + for kls in skim_classes: + skim_spec["classes"] = [kls] + traffic_assign(skim_spec, scenario) + else: + traffic_assign(skim_spec, scenario) + + # compute diagonal value for TIME and DIST + with _m.logbook_trace("Compute diagonal values for period %s" % period): + num_cells = len(scenario.zone_numbers) ** 2 + for traffic_class in classes: + class_name = traffic_class["name"] + skims = traffic_class["skims"] + with _m.logbook_trace("Class %s" % class_name): + for skim_type in skims: + skim_name = skim_type.split(".")[0] + name = '%s_%s_%s' % (period, class_name, skim_name) + matrix = emmebank.matrix(name) + data = matrix.get_numpy_data(scenario) + if skim_name == "TIME" or skim_name == "DIST": + numpy.fill_diagonal(data, 999999999.0) + data[numpy.diag_indices_from(data)] = 0.5 * numpy.nanmin(data[::,12::], 1) + internal_data = data[12::, 12::] # Exclude the first 12 zones, external zones + self._stats[name] = (name, internal_data.min(), internal_data.max(), internal_data.mean(), internal_data.sum(), 0) + elif skim_name == "REL": + data = numpy.sqrt(data) + else: + self._stats[name] = (name, data.min(), data.max(), data.mean(), data.sum(), 0) + numpy.fill_diagonal(data, 0.0) + matrix.set_numpy_data(data, scenario) + return + + def base_assignment_spec(self, relative_gap, max_iterations, num_processors, background_traffic=True): + base_spec = { + "type": "SOLA_TRAFFIC_ASSIGNMENT", + "background_traffic": None, + "classes": [], + "stopping_criteria": { + "max_iterations": int(max_iterations), "best_relative_gap": 0.0, + "relative_gap": float(relative_gap), "normalized_gap": 0.0 + }, + "performance_settings": {"number_of_processors": num_processors}, + } + if background_traffic: + base_spec["background_traffic"] = { + "link_component": "ul2", # ul2 = transit flow of the period + "turn_component": None, + "add_transit_vehicles": False + } + return base_spec + + @_context + def setup_skims(self, period, scenario): + emmebank = scenario.emmebank + with _m.logbook_trace("Extract skims for period %s" % period): + # temp_functions converts to skim-type VDFs + with temp_functions(emmebank): + backup_attributes = {"LINK": ["data2", "auto_volume", "auto_time", "additional_volume"]} + with gen_utils.backup_and_restore(scenario, backup_attributes): + yield + + def prepare_midday_generic_truck(self, scenario): + modeller = _m.Modeller() + create_mode = modeller.tool( + "inro.emme.data.network.mode.create_mode") + delete_mode = modeller.tool( + "inro.emme.data.network.mode.delete_mode") + change_link_modes = modeller.tool( + "inro.emme.data.network.base.change_link_modes") + with _m.logbook_trace("Preparation for generic truck skim"): + gen_truck_mode = 'D' + truck_mode = scenario.mode(gen_truck_mode) + if not truck_mode: + truck_mode = create_mode( + mode_type="AUX_AUTO", mode_id=gen_truck_mode, + mode_description="all trucks", scenario=scenario) + change_link_modes(modes=[truck_mode], action="ADD", + selection="modes=vVmMtT", scenario=scenario) + + #added by RSG (nagendra.dhakar@rsginc.com) for collapsed assignment classes testing + #this adds non-transponder SOV mode to SR-125 links + # TODO: move this to the network_import step for consistency and foward-compatibility + def change_mode_sovntp(self, scenario): + modeller = _m.Modeller() + change_link_modes = modeller.tool( + "inro.emme.data.network.base.change_link_modes") + with _m.logbook_trace("Preparation for sov ntp assignment"): + gen_sov_mode = 's' + sov_mode = scenario.mode(gen_sov_mode) + change_link_modes(modes=[sov_mode], action="ADD", + selection="@lane_restriction=4", scenario=scenario) + + def report(self, period, scenario, classes): + emmebank = scenario.emmebank + text = ['
'] + matrices = [] + for traffic_class in classes: + matrices.extend(["%s_%s" % (traffic_class["name"], s.split(".")[0]) for s in traffic_class["skims"]]) + num_zones = len(scenario.zone_numbers) + num_cells = num_zones ** 2 + text.append(""" + Number of zones: %s. Number of O-D pairs: %s. + Values outside -9999999, 9999999 are masked in summaries.
""" % (num_zones, num_cells)) + text.append("%-25s %9s %9s %9s %13s %9s" % ("name", "min", "max", "mean", "sum", "mask num")) + for name in matrices: + name = period + "_" + name + matrix = emmebank.matrix(name) + stats = self._stats.get(name) + if stats is None: + data = matrix.get_numpy_data(scenario) + data = numpy.ma.masked_outside(data, -9999999, 9999999, copy=False) + stats = (name, data.min(), data.max(), data.mean(), data.sum(), num_cells-data.count()) + text.append("%-25s %9.4g %9.4g %9.4g %13.7g %9d" % stats) + text.append("
") + title = 'Traffic impedance summary for period %s' % period + report = _m.PageBuilder(title) + report.wrap_html('Matrix details', "
".join(text)) + _m.logbook_write(title, report.render()) + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg + + @_m.method(return_type=unicode) + def get_link_attributes(self): + export_utils = _m.Modeller().module("inro.emme.utility.export_utilities") + return export_utils.get_link_attributes(_m.Modeller().scenario) + + +@_context +def temp_functions(emmebank): + change_function = _m.Modeller().tool( + "inro.emme.data.function.change_function") + orig_expression = {} + with _m.logbook_trace("Set functions to skim parameter"): + for func in emmebank.functions(): + if func.prefix=="fd": + exp = func.expression + orig_expression[func] = exp + if "volau+volad" in exp: + exp = exp.replace("volau+volad", "ul2") + change_function(func, exp, emmebank) + try: + yield + finally: + with _m.logbook_trace("Reset functions to assignment parameters"): + for func, expression in orig_expression.iteritems(): + change_function(func, expression, emmebank) + diff --git a/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py b/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py new file mode 100644 index 0000000..cd23ba6 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py @@ -0,0 +1,785 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// transit_assignment.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# The Transit assignment tool runs the transit assignment and skims for each +# period on the current primary scenario. +# +# The Build transit network tool must be run first to prepare the scenario for +# assignment. Note that this tool must be run with the Transit database +# (under the Database_transit directory) open (as the active database in the +# Emme desktop). +# +# +# Inputs: +# period: the time-of-day period, one of EA, AM, MD, PM, EV. +# scenario: Transit assignment scenario +# skims_only: Only run assignment for skim matrices, if True only two assignments +# are run to generate the skim matrices for the BUS and ALL skim classes. +# Otherwise, all 15 assignments are run to generate the total network flows. +# num_processors: number of processors to use for the traffic assignments. +# +# Matrices: +# All transit demand and skim matrices. +# See list of matrices under report method. +# +# Script example: +""" +import inro.modeller as _m +import os +modeller = _m.Modeller() +desktop = modeller.desktop + +build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") +transit_assign = modeller.tool("sandag.assignment.transit_assignment") +load_properties = modeller.tool('sandag.utilities.properties') + +project_dir = os.path.dirname(desktop.project_path()) +main_directory = os.path.dirname(project_dir) +props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) +main_emmebank = os.path.join(project_dir, "Database", "emmebank") +scenario_id = 100 +base_scenario = main_emmebank.scenario(scenario_id) + +transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") + +periods = ["EA", "AM", "MD", "PM", "EV"] +period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) +num_processors = "MAX-1" +scenarioYear = str(props["scenarioYear"]) + +for number, period in period_ids: + src_period_scenario = main_emmebank.scenario(number) + transit_assign_scen = build_transit_scen( + period=period, base_scenario=src_period_scenario, + transit_emmebank=transit_emmebank, + scenario_id=src_period_scenario.id, + scenario_title="%s %s transit assign" % (base_scenario.title, period), + data_table_name=scenarioYear, overwrite=True) + transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, + skims_only=True, num_processors=num_processors) + +omx_file = os.path.join(output_dir, "transit_skims.omx") +export_transit_skims(omx_file, periods, transit_scenario) +""" + + +TOOLBOX_ORDER = 21 + + +import inro.modeller as _m +import inro.emme.core.exception as _except +import traceback as _traceback +from copy import deepcopy as _copy +from collections import defaultdict as _defaultdict, OrderedDict +import contextlib as _context +import numpy + +import os +import sys +import math + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class TransitAssignment(_m.Tool(), gen_utils.Snapshot): + + period = _m.Attribute(unicode) + scenario = _m.Attribute(_m.InstanceType) + data_table_name = _m.Attribute(unicode) + assignment_only = _m.Attribute(bool) + skims_only = _m.Attribute(bool) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.assignment_only = False + self.skims_only = False + self.scenario = _m.Modeller().scenario + self.num_processors = "MAX-1" + self.attributes = [ + "period", "scenario", "data_table_name", "assignment_only", "skims_only", "num_processors"] + self._dt_db = _m.Modeller().desktop.project.data_tables() + self._matrix_cache = {} # used to hold data for reporting and post-processing of skims + + def from_snapshot(self, snapshot): + super(TransitAssignment, self).from_snapshot(snapshot) + # custom from_snapshot to load scenario and database objects + self.scenario = _m.Modeller().emmebank.scenario(self.scenario) + return self + + def page(self): + if not self.data_table_name: + load_properties = _m.Modeller().tool('sandag.utilities.properties') + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_directory = os.path.dirname(project_dir) + props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + self.data_table_name = props["scenarioYear"] + + pb = _m.ToolPageBuilder(self) + pb.title = "Transit assignment" + pb.description = """Assign transit demand for the selected time period.""" + pb.branding_text = "- SANDAG - " + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + options = [("EA", "Early AM"), + ("AM", "AM peak"), + ("MD", "Mid-day"), + ("PM", "PM peak"), + ("EV", "Evening")] + pb.add_select("period", options, title="Period:") + pb.add_select_scenario("scenario", + title="Transit assignment scenario:") + pb.add_text_box("data_table_name", title="Data table prefix name:", note="Default is the ScenarioYear") + pb.add_checkbox("assignment_only", title=" ", label="Only assign trips (no skims)") + pb.add_checkbox("skims_only", title=" ", label="Only run assignments relevant for skims") + dem_utils.add_select_processors("num_processors", pb, self) + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + results = self( + self.period, self.scenario, self.data_table_name, + self.assignment_only, self.skims_only, self.num_processors) + run_msg = "Transit assignment completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, period, scenario, data_table_name, assignment_only=False, skims_only=False, + num_processors="MAX-1"): + attrs = { + "period": period, + "scenario": scenario.id, + "data_table_name": data_table_name, + "assignment_only": assignment_only, + "skims_only": skims_only, + "num_processors": num_processors, + "self": str(self) + } + self.scenario = scenario + if not scenario.has_traffic_results: + raise Exception("missing traffic assignment results for period %s scenario %s" % (period, scenario)) + emmebank = scenario.emmebank + with self.setup(attrs): + gen_utils.log_snapshot("Transit assignment", str(self), attrs) + periods = ["EA", "AM", "MD", "PM", "EV"] + if not period in periods: + raise Exception('period: unknown value - specify one of %s' % periods) + num_processors = dem_utils.parse_num_processors(num_processors) + params = self.get_perception_parameters(period) + network = scenario.get_partial_network( + element_types=["TRANSIT_LINE"], include_attributes=True) + coaster_mode = network.mode("c") + params["coaster_fare_percep"] = 0 + for line in list(network.transit_lines()): + # get the coaster fare perception for use in journey levels + if line.mode == coaster_mode: + params["coaster_fare_percep"] = line[params["fare"]] + break + + transit_passes = gen_utils.DataTableProc("%s_transit_passes" % data_table_name) + transit_passes = {row["pass_type"]: row["cost"] for row in transit_passes} + day_pass = float(transit_passes["day_pass"]) / 2.0 + regional_pass = float(transit_passes["regional_pass"]) / 2.0 + + self.run_assignment(period, params, network, day_pass, skims_only, num_processors) + + if not assignment_only: + # max_fare = day_pass for local bus and regional_pass for premium modes + self.run_skims("BUS", period, params, day_pass, num_processors, network) + self.run_skims("PREM", period, params, regional_pass, num_processors, network) + self.run_skims("ALLPEN", period, params, regional_pass, num_processors, network) + self.mask_allpen(period) + self.report(period) + + @_context.contextmanager + def setup(self, attrs): + self._matrix_cache = {} # initialize cache at beginning of run + emmebank = self.scenario.emmebank + period = attrs["period"] + with _m.logbook_trace("Transit assignment for period %s" % period, attributes=attrs): + with gen_utils.temp_matrices(emmebank, "FULL", 3) as matrices: + matrices[0].name = "TEMP_IN_VEHICLE_COST" + matrices[1].name = "TEMP_LAYOVER_BOARD" + matrices[2].name = "TEMP_PERCEIVED_FARE" + try: + yield + finally: + self._matrix_cache = {} # clear cache at end of run + + def get_perception_parameters(self, period): + perception_parameters = { + "EA": { + "vot": 0.27, + "init_wait": 1.5, + "xfer_wait": 3.0, + "walk": 2.0, + "init_headway": "@headway_rev_op", + "xfer_headway": "@headway_op", + "fare": "@fare_per_op", + "in_vehicle": "@vehicle_per_op", + "fixed_link_time": "@trtime_link_ea" + }, + "AM": { + "vot": 0.27, + "init_wait": 1.5, + "xfer_wait": 3.0, + "walk": 2.0, + "init_headway": "@headway_rev_am", + "xfer_headway": "@headway_am", + "fare": "@fare_per_pk", + "in_vehicle": "@vehicle_per_pk", + "fixed_link_time": "@trtime_link_am" + }, + "MD": { + "vot": 0.27, + "init_wait": 1.5, + "xfer_wait": 3.0, + "walk": 2.0, + "init_headway": "@headway_rev_op", + "xfer_headway": "@headway_op", + "fare": "@fare_per_op", + "in_vehicle": "@vehicle_per_op", + "fixed_link_time": "@trtime_link_md" + }, + "PM": { + "vot": 0.27, + "init_wait": 1.5, + "xfer_wait": 3.0, + "walk": 2.0, + "init_headway": "@headway_rev_pm", + "xfer_headway": "@headway_pm", + "fare": "@fare_per_pk", + "in_vehicle": "@vehicle_per_pk", + "fixed_link_time": "@trtime_link_pm" + }, + "EV": { + "vot": 0.27, + "init_wait": 1.5, + "xfer_wait": 3.0, + "walk": 2.0, + "init_headway": "@headway_rev_op", + "xfer_headway": "@headway_op", + "fare": "@fare_per_op", + "in_vehicle": "@vehicle_per_op", + "fixed_link_time": "@trtime_link_ev" + } + } + return perception_parameters[period] + + def group_modes_by_fare(self, network, day_pass_cost): + # Identify all the unique boarding fare values + fare_set = {mode.id: _defaultdict(lambda:0) + for mode in network.modes() + if mode.type == "TRANSIT"} + for line in network.transit_lines(): + fare_set[line.mode.id][line["@fare"]] += 1 + del fare_set['c'] # remove coaster mode, this fare is handled separately + # group the modes relative to day_pass + mode_groups = { + "bus": [], # have a bus fare, less than 1/2 day pass + "day_pass": [], # boarding fare is the same as 1/2 day pass + "premium": [] # special premium services not covered by day pass + } + for mode_id, fares in fare_set.items(): + try: + max_fare = max(fares.keys()) + except ValueError: + continue # an empty set means this mode is unused in this period + if numpy.isclose(max_fare, day_pass_cost, rtol=0.0001): + mode_groups["day_pass"].append((mode_id, fares)) + elif max_fare < day_pass_cost: + mode_groups["bus"].append((mode_id, fares)) + else: + mode_groups["premium"].append((mode_id, fares)) + return mode_groups + + def all_modes_journey_levels(self, params, network, day_pass_cost): + transfer_penalty = {"on_segments": {"penalty": "@transfer_penalty_s", "perception_factor": 5.0}} + transfer_wait = { + "effective_headways": "@headway_seg", + "headway_fraction": 0.5, + "perception_factor": params["xfer_wait"], + "spread_factor": 1.0 + } + mode_groups = self.group_modes_by_fare(network, day_pass_cost) + + def get_transition_rules(next_level): + rules = [] + for name, group in mode_groups.items(): + for mode_id, fares in group: + rules.append({"mode": mode_id, "next_journey_level": next_level[name]}) + rules.append({"mode": "c", "next_journey_level": next_level["coaster"]}) + return rules + + journey_levels = [ + { + "description": "base", + "destinations_reachable": False, + "transition_rules": get_transition_rules({"bus": 1, "day_pass": 2, "premium": 3, "coaster": 4}), + "boarding_time": {"global": {"penalty": 0, "perception_factor": 1}}, + "waiting_time": { + "effective_headways": params["init_headway"], "headway_fraction": 0.5, + "perception_factor": params["init_wait"], "spread_factor": 1.0 + }, + "boarding_cost": { + "on_lines": {"penalty": "@fare", "perception_factor": params["fare"]}, + "on_segments": {"penalty": "@coaster_fare_board", "perception_factor": params["coaster_fare_percep"]}, + }, + }, + { + "description": "boarded_bus", + "destinations_reachable": True, + "transition_rules": get_transition_rules({"bus": 2, "day_pass": 2, "premium": 5, "coaster": 5}), + "boarding_time": transfer_penalty, + "waiting_time": transfer_wait, + "boarding_cost": { + # xfer from bus fare is on segments so circle lines get free transfer + "on_segments": {"penalty": "@xfer_from_bus", "perception_factor": params["fare"]}, + }, + }, + { + "description": "day_pass", + "destinations_reachable": True, + "transition_rules": get_transition_rules({"bus": 2, "day_pass": 2, "premium": 5, "coaster": 5}), + "boarding_time": transfer_penalty, + "waiting_time": transfer_wait, + "boarding_cost": { + "on_lines": {"penalty": "@xfer_from_day", "perception_factor": params["fare"]}, + }, + }, + { + "description": "boarded_premium", + "destinations_reachable": True, + "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), + "boarding_time": transfer_penalty, + "waiting_time": transfer_wait, + "boarding_cost": { + "on_lines": {"penalty": "@xfer_from_premium", "perception_factor": params["fare"]}, + }, + }, + { + "description": "boarded_coaster", + "destinations_reachable": True, + "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), + "boarding_time": transfer_penalty, + "waiting_time": transfer_wait, + "boarding_cost": { + "on_lines": {"penalty": "@xfer_from_coaster", "perception_factor": params["fare"]}, + }, + }, + { + "description": "regional_pass", + "destinations_reachable": True, + "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), + "boarding_time": transfer_penalty, + "waiting_time": transfer_wait, + "boarding_cost": { + "on_lines": {"penalty": "@xfer_regional_pass", "perception_factor": params["fare"]}, + }, + } + ] + return journey_levels + + def filter_journey_levels_by_mode(self, modes, journey_levels): + # remove rules for unused modes from provided journey_levels + # (restrict to provided modes) + journey_levels = _copy(journey_levels) + for level in journey_levels: + rules = level["transition_rules"] + rules = [r for r in rules if r["mode"] in modes] + level["transition_rules"] = rules + # count level transition rules references to find unused levels + num_levels = len(journey_levels) + level_count = [0] * len(journey_levels) + + def follow_rule(next_level): + level_count[next_level] += 1 + if level_count[next_level] > 1: + return + for rule in journey_levels[next_level]["transition_rules"]: + follow_rule(rule["next_journey_level"]) + + follow_rule(0) + # remove unreachable levels + # and find new index for transition rules for remaining levels + level_map = {i:i for i in range(num_levels)} + for level_id, count in reversed(list(enumerate(level_count))): + if count == 0: + for index in range(level_id, num_levels): + level_map[index] -= 1 + del journey_levels[level_id] + # re-index remaining journey_levels + for level in journey_levels: + for rule in level["transition_rules"]: + next_level = rule["next_journey_level"] + rule["next_journey_level"] = level_map[next_level] + return journey_levels + + @_m.logbook_trace("Transit assignment by demand set", save_arguments=True) + def run_assignment(self, period, params, network, day_pass_cost, skims_only, num_processors): + modeller = _m.Modeller() + scenario = self.scenario + emmebank = scenario.emmebank + assign_transit = modeller.tool( + "inro.emme.transit_assignment.extended_transit_assignment") + + walk_modes = ["a", "w", "x"] + local_bus_mode = ["b"] + premium_modes = ["c", "l", "e", "p", "r", "y", "o"] + + # get the generic all-modes journey levels table + journey_levels = self.all_modes_journey_levels(params, network, day_pass_cost) + local_bus_journey_levels = self.filter_journey_levels_by_mode(local_bus_mode, journey_levels) + premium_modes_journey_levels = self.filter_journey_levels_by_mode(premium_modes, journey_levels) + # All modes transfer penalty assignment uses penalty of 15 minutes + for level in journey_levels[1:]: + level["boarding_time"] = {"global": {"penalty": 15, "perception_factor": 1}} + + base_spec = { + "type": "EXTENDED_TRANSIT_ASSIGNMENT", + "modes": [], + "demand": "", + "waiting_time": { + "effective_headways": params["init_headway"], "headway_fraction": 0.5, + "perception_factor": params["init_wait"], "spread_factor": 1.0 + }, + # Fare attributes + "boarding_cost": {"global": {"penalty": 0, "perception_factor": 1}}, + "boarding_time": {"global": {"penalty": 0, "perception_factor": 1}}, + "in_vehicle_cost": {"penalty": "@coaster_fare_inveh", + "perception_factor": params["coaster_fare_percep"]}, + "in_vehicle_time": {"perception_factor": params["in_vehicle"]}, + "aux_transit_time": {"perception_factor": params["walk"]}, + "aux_transit_cost": None, + "journey_levels": [], + "flow_distribution_between_lines": {"consider_total_impedance": False}, + "flow_distribution_at_origins": { + "fixed_proportions_on_connectors": None, + "choices_at_origins": "OPTIMAL_STRATEGY" + }, + "flow_distribution_at_regular_nodes_with_aux_transit_choices": { + "choices_at_regular_nodes": "OPTIMAL_STRATEGY" + }, + #"circular_lines": { + # "stay": True + #}, + "connector_to_connector_path_prohibition": None, + "od_results": {"total_impedance": None}, + "performance_settings": {"number_of_processors": num_processors} + } + + skim_parameters = OrderedDict([ + ("BUS", { + "modes": walk_modes + local_bus_mode, + "journey_levels": local_bus_journey_levels + }), + ("PREM", { + "modes": walk_modes + premium_modes, + "journey_levels": premium_modes_journey_levels + }), + ("ALLPEN", { + "modes": walk_modes + local_bus_mode + premium_modes, + "journey_levels": journey_levels + }), + ]) + + if skims_only: + access_modes = ["WLK"] + else: + access_modes = ["WLK", "PNR", "KNR"] + add_volumes = False + for a_name in access_modes: + for mode_name, parameters in skim_parameters.iteritems(): + spec = _copy(base_spec) + name = "%s_%s%s" % (period, a_name, mode_name) + spec["modes"] = parameters["modes"] + spec["demand"] = 'mf"%s"' % name + spec["journey_levels"] = parameters["journey_levels"] + assign_transit(spec, class_name=name, add_volumes=add_volumes, scenario=self.scenario) + add_volumes = True + + @_m.logbook_trace("Extract skims", save_arguments=True) + def run_skims(self, name, period, params, max_fare, num_processors, network): + modeller = _m.Modeller() + scenario = self.scenario + emmebank = scenario.emmebank + matrix_calc = modeller.tool( + "inro.emme.matrix_calculation.matrix_calculator") + network_calc = modeller.tool( + "inro.emme.network_calculation.network_calculator") + matrix_results = modeller.tool( + "inro.emme.transit_assignment.extended.matrix_results") + path_analysis = modeller.tool( + "inro.emme.transit_assignment.extended.path_based_analysis") + strategy_analysis = modeller.tool( + "inro.emme.transit_assignment.extended.strategy_based_analysis") + + class_name = "%s_WLK%s" % (period, name) + skim_name = "%s_%s" % (period, name) + self.run_skims.logbook_cursor.write(name="Extract skims for %s, using assignment class %s" % (name, class_name)) + + with _m.logbook_trace("First and total wait time, number of boardings, fares, total walk time, in-vehicle time"): + # First and total wait time, number of boardings, fares, total walk time, in-vehicle time + spec = { + "type": "EXTENDED_TRANSIT_MATRIX_RESULTS", + "actual_first_waiting_times": 'mf"%s_FIRSTWAIT"' % skim_name, + "actual_total_waiting_times": 'mf"%s_TOTALWAIT"' % skim_name, + "total_impedance": 'mf"%s_GENCOST"' % skim_name, + "by_mode_subset": { + "modes": [mode.id for mode in network.modes() if mode.type == "TRANSIT" or mode.type == "AUX_TRANSIT"], + "avg_boardings": 'mf"%s_XFERS"' % skim_name, + "actual_in_vehicle_times": 'mf"%s_TOTALIVTT"' % skim_name, + "actual_in_vehicle_costs": 'mf"TEMP_IN_VEHICLE_COST"', + "actual_total_boarding_costs": 'mf"%s_FARE"' % skim_name, + "perceived_total_boarding_costs": 'mf"TEMP_PERCEIVED_FARE"', + "actual_aux_transit_times": 'mf"%s_TOTALWALK"' % skim_name, + }, + } + matrix_results(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + with _m.logbook_trace("Distance and in-vehicle time by mode"): + mode_combinations = [ + ("BUS", ["b"], ["IVTT", "DIST"]), + ("LRT", ["l"], ["IVTT", "DIST"]), + ("CMR", ["c"], ["IVTT", "DIST"]), + ("EXP", ["e", "p"], ["IVTT", "DIST"]), + ("BRT", ["r", "y"], ["DIST"]), + ("BRTRED", ["r"], ["IVTT"]), + ("BRTYEL", ["y"], ["IVTT"]), + ("TIER1", ["o"], ["IVTT", "DIST"]), + ] + for mode_name, modes, skim_types in mode_combinations: + dist = 'mf"%s_%sDIST"' % (skim_name, mode_name) if "DIST" in skim_types else None + ivtt = 'mf"%s_%sIVTT"' % (skim_name, mode_name) if "IVTT" in skim_types else None + spec = { + "type": "EXTENDED_TRANSIT_MATRIX_RESULTS", + "by_mode_subset": { + "modes": modes, + "distance": dist, + "actual_in_vehicle_times": ivtt, + }, + } + matrix_results(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + # Sum total distance + spec = { + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_TOTDIST"' % skim_name, + "expression": ('mf"{0}_BUSDIST" + mf"{0}_LRTDIST" + mf"{0}_CMRDIST"' + ' + mf"{0}_EXPDIST" + mf"{0}_BRTDIST" + mf"{0}_TIER1DIST"').format(skim_name), + } + matrix_calc(spec, scenario=scenario, num_processors=num_processors) + + # convert number of boardings to number of transfers + # and subtract transfers to the same line at layover points + with _m.logbook_trace("Number of transfers and total fare"): + spec = { + "trip_components": {"boarding": "@layover_board"}, + "sub_path_combination_operator": "+", + "sub_strategy_combination_operator": "average", + "selected_demand_and_transit_volumes": { + "sub_strategies_to_retain": "ALL", + "selection_threshold": {"lower": -999999, "upper": 999999} + }, + "results": { + "strategy_values": 'TEMP_LAYOVER_BOARD', + }, + "type": "EXTENDED_TRANSIT_STRATEGY_ANALYSIS" + } + strategy_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + spec = { + "type": "MATRIX_CALCULATION", + "constraint":{ + "by_value": { + "od_values": 'mf"%s_XFERS"' % skim_name, + "interval_min": 1, "interval_max": 9999999, + "condition": "INCLUDE"}, + }, + "result": 'mf"%s_XFERS"' % skim_name, + "expression": '(%s_XFERS - 1 - TEMP_LAYOVER_BOARD).max.0' % skim_name, + } + matrix_calc(spec, scenario=scenario, num_processors=num_processors) + + # sum in-vehicle cost and boarding cost to get the fare paid + spec = { + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_FARE"' % skim_name, + "expression": '(%s_FARE + TEMP_IN_VEHICLE_COST).min.%s' % (skim_name, max_fare), + } + matrix_calc(spec, scenario=scenario, num_processors=num_processors) + + # walk access time - get distance and convert to time with 3 miles / hr + with _m.logbook_trace("Walk time access, egress and xfer"): + path_spec = { + "portion_of_path": "ORIGIN_TO_INITIAL_BOARDING", + "trip_components": {"aux_transit": "length",}, + "path_operator": "+", + "path_selection_threshold": {"lower": 0, "upper": 999999 }, + "path_to_od_aggregation": { + "operator": "average", + "aggregated_path_values": 'mf"%s_ACCWALK"' % skim_name, + }, + "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" + } + path_analysis(path_spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + + # walk egress time - get distance and convert to time with 3 miles/ hr + path_spec = { + "portion_of_path": "FINAL_ALIGHTING_TO_DESTINATION", + "trip_components": {"aux_transit": "length",}, + "path_operator": "+", + "path_selection_threshold": {"lower": 0, "upper": 999999 }, + "path_to_od_aggregation": { + "operator": "average", + "aggregated_path_values": 'mf"%s_EGRWALK"' % skim_name + }, + "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" + } + path_analysis(path_spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + + spec_list = [ + { # walk access time - convert to time with 3 miles/ hr + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_ACCWALK"' % skim_name, + "expression": '60.0 * %s_ACCWALK / 3.0' % skim_name, + }, + { # walk egress time - convert to time with 3 miles/ hr + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_EGRWALK"' % skim_name, + "expression": '60.0 * %s_EGRWALK / 3.0' % skim_name, + }, + { # transfer walk time = total - access - egress + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_XFERWALK"' % skim_name, + "expression": '({name}_TOTALWALK - {name}_ACCWALK - {name}_EGRWALK).max.0'.format(name=skim_name), + }] + matrix_calc(spec_list, scenario=scenario, num_processors=num_processors) + + # transfer wait time + with _m.logbook_trace("Wait time - xfer"): + spec = { + "type": "MATRIX_CALCULATION", + "constraint":{ + "by_value": { + "od_values": 'mf"%s_TOTALWAIT"' % skim_name, + "interval_min": 0, "interval_max": 9999999, + "condition": "INCLUDE"}, + }, + "result": 'mf"%s_XFERWAIT"' % skim_name, + "expression": '({name}_TOTALWAIT - {name}_FIRSTWAIT).max.0'.format(name=skim_name), + } + matrix_calc(spec, scenario=scenario, num_processors=num_processors) + + with _m.logbook_trace("Calculate dwell time"): + with gen_utils.temp_attrs(scenario, "TRANSIT_SEGMENT", ["@dwt_for_analysis"]): + values = scenario.get_attribute_values("TRANSIT_SEGMENT", ["dwell_time"]) + scenario.set_attribute_values("TRANSIT_SEGMENT", ["@dwt_for_analysis"], values) + + spec = { + "trip_components": {"in_vehicle": "@dwt_for_analysis"}, + "sub_path_combination_operator": "+", + "sub_strategy_combination_operator": "average", + "selected_demand_and_transit_volumes": { + "sub_strategies_to_retain": "ALL", + "selection_threshold": {"lower": -999999, "upper": 999999} + }, + "results": { + "strategy_values": 'mf"%s_DWELLTIME"' % skim_name, + }, + "type": "EXTENDED_TRANSIT_STRATEGY_ANALYSIS" + } + strategy_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + + expr_params = _copy(params) + expr_params["xfers"] = 15.0 + expr_params["name"] = skim_name + spec = { + "type": "MATRIX_CALCULATION", + "constraint": None, + "result": 'mf"%s_GENCOST"' % skim_name, + "expression": ("{xfer_wait} * {name}_TOTALWAIT " + "- ({xfer_wait} - {init_wait}) * {name}_FIRSTWAIT " + "+ 1.0 * {name}_TOTALIVTT + 0.5 * {name}_BUSIVTT" + "+ (1 / {vot}) * (TEMP_PERCEIVED_FARE + {coaster_fare_percep} * TEMP_IN_VEHICLE_COST)" + "+ {xfers} *({name}_XFERS.max.0) " + "+ {walk} * {name}_TOTALWALK").format(**expr_params) + } + matrix_calc(spec, scenario=scenario, num_processors=num_processors) + return + + def mask_allpen(self, period): + # Reset skims to 0 if not both local and premium + skims = [ + "FIRSTWAIT", "TOTALWAIT", "DWELLTIME", "BUSIVTT", "XFERS", "TOTALWALK", + "LRTIVTT", "CMRIVTT", "EXPIVTT", "LTDEXPIVTT", "BRTREDIVTT", "BRTYELIVTT", "TIER1IVTT", + "GENCOST", "XFERWAIT", "FARE", + "ACCWALK", "XFERWALK", "EGRWALK", "TOTALIVTT", + "BUSDIST", "LRTDIST", "CMRDIST", "EXPDIST", "BRTDIST" , "TIER1DIST"] + localivt_skim = self.get_matrix_data(period + "_ALLPEN_BUSIVTT") + totalivt_skim = self.get_matrix_data(period + "_ALLPEN_TOTALIVTT") + has_premium = numpy.greater((totalivt_skim - localivt_skim), 0) + has_both = numpy.greater(localivt_skim, 0) * has_premium + for skim in skims: + mat_name = period + "_ALLPEN_" + skim + data = self.get_matrix_data(mat_name) + self.set_matrix_data(mat_name, data * has_both) + + def get_matrix_data(self, name): + data = self._matrix_cache.get(name) + if data is None: + matrix = self.scenario.emmebank.matrix(name) + data = matrix.get_numpy_data(self.scenario) + self._matrix_cache[name] = data + return data + + def set_matrix_data(self, name, data): + matrix = self.scenario.emmebank.matrix(name) + self._matrix_cache[name] = data + matrix.set_numpy_data(data, self.scenario) + + def report(self, period): + text = ['
'] + init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") + matrices = init_matrices.get_matrix_names("transit_skims", [period], self.scenario) + num_zones = len(self.scenario.zone_numbers) + num_cells = num_zones ** 2 + text.append( + "Number of zones: %s. Number of O-D pairs: %s. " + "Values outside -9999999, 9999999 are masked in summaries.
" % (num_zones, num_cells)) + text.append("%-25s %9s %9s %9s %13s %9s" % ("name", "min", "max", "mean", "sum", "mask num")) + for name in matrices: + data = self.get_matrix_data(name) + data = numpy.ma.masked_outside(data, -9999999, 9999999, copy=False) + stats = (name, data.min(), data.max(), data.mean(), data.sum(), num_cells-data.count()) + text.append("%-25s %9.4g %9.4g %9.4g %13.7g %9d" % stats) + text.append("
") + title = 'Transit impedance summary for period %s' % period + report = _m.PageBuilder(title) + report.wrap_html('Matrix details', "
".join(text)) + _m.logbook_write(title, report.render()) diff --git a/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py b/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py new file mode 100644 index 0000000..8f4b387 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py @@ -0,0 +1,217 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// transit_select_analysis.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# This tool runs select type network analysis on the results of one or more +# transit assignments. It is run as a post-process tool after the assignment +# tools are complete, using the saved transit strategies. Any number of +# analyses can be run without needing to rerun the assignments. +# +# +# Inputs: +# Trip components for selection: pick one or more extra attributes which +# identify the network elements of interest by trip component: +# in_vehicle +# aux_transit +# initial_boarding +# transfer_boarding +# transfer_alighting +# final_alighting +# Result suffix: the suffix to use in the naming of per-class result +# attributes and matrices, up to 6 characters. +# Threshold: the minimum number of elements which must be encountered +# for the path selection. +# Scenario: the scenario to analyse. +# +# +# Script example: +""" +import inro.modeller as _m +import os +modeller = _m.Modeller() +desktop = modeller.desktop + +select_link = modeller.tool("sandag.assignment.transit_select_link") + +project_dir = os.path.dirname(desktop.project_path()) +main_directory = os.path.dirname(project_dir) + +transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") + +periods = ["EA", "AM", "MD", "PM", "EV"] +period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) + +suffix = "LRT" +threshold = 1 +num_processors = "MAX-1" +selection = { + "in_vehicle": None, + "aux_transit": None, + "initial_boarding": "@selected_line", + "transfer_boarding": None, + "transfer_alighting": None, + "final_alighting": None, +} + +for number, period in period_ids: + scenario = transit_emmebank.scenario(number) + select_link(selection, suffix, threshold, scenario, num_processors) +""" + +TOOLBOX_ORDER = 25 + + +import inro.modeller as _m +import inro.emme.core.exception as _except +import traceback as _traceback + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class TransitSelectAnalysis(_m.Tool(), gen_utils.Snapshot): + + in_vehicle = _m.Attribute(_m.InstanceType) + aux_transit = _m.Attribute(_m.InstanceType) + initial_boarding = _m.Attribute(_m.InstanceType) + transfer_boarding = _m.Attribute(_m.InstanceType) + transfer_alighting = _m.Attribute(_m.InstanceType) + final_alighting = _m.Attribute(_m.InstanceType) + + suffix = _m.Attribute(str) + threshold = _m.Attribute(int) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + def __init__(self): + self.threshold = 1 + self.num_processors = "MAX-1" + self.attributes = [ + "in_vehicle", "aux_transit", "initial_boarding", "transfer_boarding", + "transfer_alighting", "final_alighting", "suffix", "threshold", + "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Transit select analysis" + pb.description = """ + Run select type of analysis (select link, select node, select line ...) on + the results of the transit assignment(s) using a path-based analysis. + Can be used after a transit assignment has been completed.""" + pb.branding_text = "- SANDAG - Assignment" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + with pb.section("Trip components for selection:"): + domains = ["LINK", "NODE", "TRANSIT_SEGMENT", "TRANSIT_LINE"] + pb.add_select_extra_attribute("in_vehicle", title="In-vehicle", filter=domains, allow_none=True) + pb.add_select_extra_attribute("aux_transit", title="Auxilary transit", filter=domains, allow_none=True) + pb.add_select_extra_attribute("initial_boarding", title="Initial boarding", filter=domains, allow_none=True) + pb.add_select_extra_attribute("transfer_boarding", title="Transfer boarding", filter=domains, allow_none=True) + pb.add_select_extra_attribute("transfer_alighting", title="Transfer alighting", filter=domains, allow_none=True) + pb.add_select_extra_attribute("final_alighting", title="Final alighting", filter=domains, allow_none=True) + + pb.add_text_box("suffix", title="Suffix for results (matrices and attributes):", size=6, + note="The suffix to use in the naming of per-class result attributes and matrices, up to 6 characters. " + "Should be unique (existing attributes / matrices will be overwritten).") + pb.add_text_box("threshold", title="Threshold for selection:", + note="The minimum number of links which must be encountered for the path selection. " + "The default value of 1 indicates an 'any' link selection.") + dem_utils.add_select_processors("num_processors", pb, self) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + selection = { + "in_vehicle": self.in_vehicle, + "aux_transit": self.aux_transit, + "initial_boarding": self.initial_boarding, + "transfer_boarding": self.transfer_boarding, + "transfer_alighting": self.transfer_alighting, + "final_alighting": self.final_alighting, + } + scenario = _m.Modeller().scenario + results = self(selection, self.suffix, self.threshold, scenario, self.num_processors) + run_msg = "Traffic assignment completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, selection, suffix, threshold, scenario, num_processors): + attrs = { + "selection": selection, + "suffix": suffix, + "threshold": threshold, + "scenario": scenario.id, + "num_processors": num_processors + } + with _m.logbook_trace("Transit select analysis %s" % suffix, attributes=attrs): + attrs.update(dict((k,v) for k,v in attrs["selection"].iteritems())) + gen_utils.log_snapshot("Transit select analysis", str(self), attrs) + + path_analysis = _m.Modeller().tool( + "inro.emme.transit_assignment.extended.path_based_analysis") + create_attribute = _m.Modeller().tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + + spec = { + "portion_of_path": "COMPLETE", + "trip_components": selection, + "path_operator": "+", + "path_selection_threshold": {"lower": threshold, "upper": 999999}, + "path_to_od_aggregation": None, + "constraint": None, + "analyzed_demand": None, + "results_from_retained_paths": None, + "path_to_od_statistics": None, + "path_details": None, + "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" + } + strategies = scenario.transit_strategies + classes = [x.name for x in strategies.strat_files()] + if not classes: + raise Exception("Results for multi-class transit assignment not available") + + for class_name in classes: + with _m.logbook_trace("Analysis for class %s" % class_name): + seldem_name = "SELDEM_%s_%s" % (class_name, suffix) + desc = "Selected demand for %s %s" % (class_name, suffix) + seldem = dem_utils.create_full_matrix(seldem_name, desc, scenario=scenario) + results_from_retained_paths = { + "paths_to_retain": "SELECTED", + "demand": seldem.named_id, + } + attributes = [ + ("transit_volumes", "TRANSIT_SEGMENT", "@seltr_%s_%s", "%s '%s' sel segment flow"), + ("aux_transit_volumes", "LINK", "@selax_%s_%s", "%s '%s' sel aux transit flow"), + ("total_boardings", "TRANSIT_SEGMENT", "@selbr_%s_%s", "%s '%s' sel boardings"), + ("total_alightings", "TRANSIT_SEGMENT", "@selal_%s_%s", "%s '%s' sel alightings"), + ] + mode_name = class_name.lower()[3:] + for key, domain, name, desc in attributes: + attr = create_attribute(domain, name % (mode_name, suffix), desc % (class_name, suffix), + 0, overwrite=True, scenario=scenario) + results_from_retained_paths[key] = attr.id + spec["results_from_retained_paths"] = results_from_retained_paths + path_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/build_toolbox.py b/sandag_abm/src/main/emme/toolbox/build_toolbox.py new file mode 100644 index 0000000..1a9d1b7 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/build_toolbox.py @@ -0,0 +1,411 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// build_toolbox.py /// +#//// /// +#//// Generates an mtbx (Emme Modeller Toolbox), based on the structure /// +#//// of the Python source tree. /// +#//// /// +#//// Usage: build_toolbox.py [-s source_folder] [-p toolbox_path] /// +#//// /// +#//// [-p toolbox_path]: Specifies the name of the MTBX file. /// +#//// If omitted,defaults to "sandag_toolbox.mtbx" /// +#//// [-s source_folder]: The location of the source code folder. /// +#//// If omitted, defaults to the working directory. /// +#//// [-l] [--link] Build the toolbox with references to the files /// +#//// Use with developing or debugging scripts, changes to the /// +#//// scripts can be used with a "Refresh" of the toolbox /// +#//// [-c] [--consolidate] Build the toolbox with copies of the /// +#//// scripts included inside the toolbox. /// +#//// Use to have a "frozen" version of the scripts with node /// +#//// changes available. /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Example: +# python "T:\projects\sr13\develop\emme_conversion\git\sandag_abm\ABM_EMME\src\main\emme\toolbox\build_toolbox.py" --link +# -p "T:\projects\sr14\abm2_test\abm_runs\14_2_0\2035D_Hyperloop\emme_project\Scripts\sandag_toolbox.mtbx" +# -s T:\projects\sr13\develop\emme_conversion\git\sandag_abm\ABM_EMME\src\main\emme\toolbox + + +import os +import re +from datetime import datetime +import subprocess +import sqlite3.dbapi2 as sqllib +import base64 +import pickle + + +def check_namespace(ns): + if not re.match("^[a-zA-Z][a-zA-Z0-9_]*$", ns): + raise Exception("Namespace '%s' is invalid" % ns) + + +def get_emme_version(): + emme_process = subprocess.Popen(['Emme', '-V'], stdout=subprocess.PIPE, stderr=subprocess.PIPE) + output = emme_process.communicate()[0] + return output.split(',')[0] + + +def usc_transform(value): + try: + return unicode(value) + except Exception: + return unicode(str(value), encoding="raw-unicode-escape") + + +class BaseNode(object): + def __init__(self, namespace, title): + check_namespace(namespace) + self.namespace = namespace + self.title = title + self.element_id = None + self.parent = None + self.root = None + self.children = [] + + def add_folder(self, namespace): + node = FolderNode(namespace, parent=self) + self.children.append(node) + return node + + def add_tool(self, script_path, namespace): + try: + node = ToolNode(namespace, script_path, parent=self) + self.children.append(node) + with open(script_path, 'r') as f: + for line in f: + if line.startswith("TOOLBOX_ORDER"): + node.order = int(line.split("=")[1]) + if line.startswith("TOOLBOX_TITLE"): + title = line.split("=")[1].strip() + node.title = title[1:-1] # exclude first and last quotes + except Exception, e: + print script_path, namespace + print type(e), str(e) + return None + return node + + def consolidate(self): + for child in self.children: + child.consolidate() + + def set_toolbox_order(self): + self.element_id = self.root.next_id() + self.children.sort(key=lambda x: x.order) + for child in self.children: + child.set_toolbox_order() + + +class ElementTree(BaseNode): + + def __init__(self, namespace, title): + super(ElementTree, self).__init__(namespace, title) + self.next_element_id = 0 + self.begin = str(datetime.now()) + self.version = "Emme %s" % get_emme_version() + self.root = self + + def next_id(self): + self.next_element_id += 1 + return self.next_element_id + + +class FolderNode(BaseNode): + + def __init__(self, namespace, parent): + title = namespace.replace("_", " ").capitalize() + super(FolderNode, self).__init__(namespace, title) + self.parent = parent + self.root = parent.root + self.element_id = None + + @property + def order(self): + child_order = [child.order for child in self.children if child.order is not None] + if child_order: + return min(child_order) + return None + + +class ToolNode(): + + def __init__(self, namespace, script_path, parent): + check_namespace(namespace) + self.namespace = namespace + self.title = namespace.replace("_", " ").capitalize() + + self.root = parent.root + self.parent = parent + self.element_id = None + self.order = None + + self.script = script_path + self.extension = '.py' + self.code = '' + + def consolidate(self): + with open(self.script, 'r') as f: + code = f.read() + self.code = usc_transform(base64.b64encode(pickle.dumps(code))) + self.script = '' + + def set_toolbox_order(self): + self.element_id = self.root.next_id() + +class MTBXDatabase(): + FORMAT_MAGIC_NUMBER = 'B8C224F6_7C94_4E6F_8C2C_5CC06F145271' + TOOLBOX_MAGIC_NUMBER = 'TOOLBOX_C6809332_CD61_45B3_9060_411D825669F8' + CATEGORY_MAGIC_NUMBER = 'CATEGORY_984876A0_3350_4374_B47C_6D9C5A47BBC8' + TOOL_MAGIC_NUMBER = 'TOOL_1AC06B56_6A54_431A_9515_0BF77013646F' + + def __init__(self, filepath, title): + if os.path.exists(filepath): + os.remove(filepath) + + self.db = sqllib.connect(filepath) + + self._create_attribute_table() + self._create_element_table() + self._create_document_table() + self._create_triggers() + + self._initialize_documents_table(title) + + def _create_attribute_table(self): + sql = """CREATE TABLE attributes( + element_id INTEGER REFERENCES elements(element_id), + name VARCHAR, + value VARCHAR, + PRIMARY KEY(element_id, name));""" + + self.db.execute(sql) + + def _create_element_table(self): + sql = """CREATE TABLE elements( + element_id INTEGER PRIMARY KEY AUTOINCREMENT, + parent_id INTEGER REFERENCES elements(element_id), + document_id INTEGER REFERENCES documents(document_id), + tag VARCHAR, + text VARCHAR, + tail VARCHAR);""" + + self.db.execute(sql) + + def _create_document_table(self): + sql = """CREATE TABLE documents( + document_id INTEGER PRIMARY KEY AUTOINCREMENT, + title VARCHAR);""" + + self.db.execute(sql) + + def _create_triggers(self): + sql = """CREATE TRIGGER documents_delete + BEFORE DELETE on documents + FOR EACH ROW BEGIN + DELETE FROM elements WHERE document_id = OLD.document_id; + END""" + + self.db.execute(sql) + + sql = """CREATE TRIGGER elements_delete + BEFORE DELETE on elements + FOR EACH ROW BEGIN + DELETE FROM attributes WHERE element_id = OLD.element_id; + END""" + + self.db.execute(sql) + + def _initialize_documents_table(self, title): + sql = """INSERT INTO documents (document_id, title) + VALUES (1, '%s');""" % title + + self.db.execute(sql) + self.db.commit() + + def populate_tables_from_tree(self, tree): + + #Insert into the elements table + column_string = "element_id, document_id, tag, text, tail" + value_string = "{id}, 1, '{title}', '', ''".format( + id=tree.element_id, title=tree.title) + sql = """INSERT INTO elements (%s) + VALUES (%s);""" % (column_string, value_string) + self.db.execute(sql) + + #Insert into the attributes table + column_string = "element_id, name, value" + atts = {'major': '', + 'format': MTBXDatabase.FORMAT_MAGIC_NUMBER, + 'begin': tree.begin, + 'version': tree.version, + 'maintenance': '', + 'minor': '', + 'name': tree.title, + 'description': '', + 'namespace': tree.namespace, + MTBXDatabase.TOOLBOX_MAGIC_NUMBER: 'True'} + for key, val in atts.iteritems(): + value_string = "{id}, '{name}', '{value}'".format( + id=tree.element_id, name=key, value=val) + sql = """INSERT INTO attributes (%s) + VALUES (%s);""" % (column_string, value_string) + self.db.execute(sql) + + self.db.commit() + + #Handle children nodes + for child in tree.children: + if isinstance(child, ToolNode): + self._insert_tool(child) + else: + self._insert_folder(child) + + def _insert_folder(self, node): + #Insert into the elements table + column_string = "element_id, parent_id, document_id, tag, text, tail" + value_string = "{id}, {parent}, 1, '{title}', '', ''".format( + id=node.element_id, parent=node.parent.element_id, title=node.title) + sql = """INSERT INTO elements (%s) + VALUES (%s);""" % (column_string, value_string) + self.db.execute(sql) + + #Insert into the attributes table + column_string = "element_id, name, value" + atts = {'namespace': node.namespace, + 'description': '', + 'name': node.title, + 'children': [c.element_id for c in node.children], + MTBXDatabase.CATEGORY_MAGIC_NUMBER: 'True'} + for key, val in atts.iteritems(): + value_string = "{id}, '{name}', '{value}'".format( + id=node.element_id, name=key, value=val) + sql = """INSERT INTO attributes (%s) + VALUES (%s);""" % (column_string, value_string) + self.db.execute(sql) + + self.db.commit() + + #Handle children nodes + for child in node.children: + if isinstance(child, ToolNode): + self._insert_tool(child) + else: + self._insert_folder(child) + + def _insert_tool(self, node): + #Insert into the elements table + column_string = "element_id, parent_id, document_id, tag, text, tail" + value_string = "{id}, {parent}, 1, '{title}', '', ''".format( + id=node.element_id, parent=node.parent.element_id, title=node.title) + + sql = """INSERT INTO elements (%s) + VALUES (%s);""" % (column_string, value_string) + self.db.execute(sql) + + #Insert into the attributes table + column_string = "element_id, name, value" + atts = {'code': node.code, + 'description': '', + 'script': node.script, + 'namespace': node.namespace, + 'python_suffix': node.extension, + 'name': node.title, + MTBXDatabase.TOOL_MAGIC_NUMBER: 'True'} + for key, val in atts.iteritems(): + value_string = "{id}, '{name}', '{value!s}'".format( + id=node.element_id, name=key, value=val) + sql = """INSERT INTO attributes (%s) + VALUES (?, ?, ?);""" % column_string + self.db.execute(sql, (node.element_id, key, val)) + + self.db.commit() + + +def build_toolbox(toolbox_file, source_folder, title, namespace, consolidate): + print "------------------------" + print " Build Toolbox Utility" + print "------------------------" + print "" + print "toolbox: %s" % toolbox_file + print "source folder: %s" % source_folder + print "title: %s" % title + print "namespace: %s" % namespace + print "" + + print "Loading toolbox structure" + tree = ElementTree(namespace, title) + explore_source_folder(source_folder, tree) + tree.set_toolbox_order() + print "Done. Found %s elements." % (tree.next_element_id) + if consolidate: + print "Consolidating code..." + tree.consolidate() + print "Consolidate done" + + print "" + print "Building MTBX file..." + mtbx = MTBXDatabase(toolbox_file, title) + mtbx.populate_tables_from_tree(tree) + print "Build MTBX file done." + + +def explore_source_folder(root_folder_path, parent_node): + folders = [] + files = [] + for item in os.listdir(root_folder_path): + itempath = os.path.join(root_folder_path, item) + if os.path.isfile(itempath): + name, extension = os.path.splitext(item) + if extension != '.py': + continue # skip non-Python files + if os.path.normpath(itempath) == os.path.normpath(os.path.abspath(__file__)): + continue # skip this file + files.append((name, extension)) + else: + folders.append(item) + + for foldername in folders: + folderpath = os.path.join(root_folder_path, foldername) + folder_node = parent_node.add_folder(namespace=foldername) + explore_source_folder(folderpath, folder_node) + + for filename, ext in files: + script_path = os.path.join(root_folder_path, filename + ext) + parent_node.add_tool(script_path, namespace=filename) + + +if __name__ == "__main__": + ''' + Usage: build_toolbox.py [-p toolbox_path] [-s source_folder] [-l] [-c] + ''' + + import argparse + parser = argparse.ArgumentParser() + parser.add_argument('-s', '--src', help= "Path to the source code folder. Default is the working folder.") + parser.add_argument('-p', '--path', help= "Output file path. Default is 'sandag_toolbox.mtbx' in the source code folder.") + parser.add_argument('-l', '--link', help= "Link the python source files from their current location (instead of consolidate (compile) the toolbox).", action= 'store_true') + parser.add_argument('-c', '--consolidate', help= "Consolidate (compile) the toolbox (default option).", action= 'store_true') + + args = parser.parse_args() + + source_folder = args.src or os.path.dirname(os.path.abspath(__file__)) + folder_name = os.path.split(source_folder)[1] + toolbox_file = args.path or "sandag_toolbox.mtbx" + title = "SANDAG toolbox" + namespace = "sandag" + consolidate = args.consolidate + link = args.link + if consolidate and link: + raise Exception("-l and -c (--link and --consolidate) are mutually exclusive options") + if not consolidate and not link: + consolidate = True # default if neither is specified + + build_toolbox(toolbox_file, source_folder, title, namespace, consolidate) diff --git a/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py b/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py new file mode 100644 index 0000000..f9f1ab1 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py @@ -0,0 +1,669 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright RSG, 2019-2020. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/mode_choice_diagnostic.py /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Diagnostic tool for the SANDAG activity-based travel model mode choice results. +# This script first generates synthetic population files for target markets. +# Users may input target market parameters via the "syn_pop_attributes.yaml" file. +# Users must additionally input origin and destination MAZs (i.e. MGRAs) via the +# "origin_mgra.csv" and "destination_mgra.csv" files. +# +# Once all synthetic population files have been created, the script creates a copy of +# the "sandag_abm.properties" file and modifies specific property parameters so that +# it is compatible with a the mode choice diagnostic tool. The modified properties +# file is renamed as "sandag_abm_mcd.properties" +# +# Finally, the mode choice diagnostic tool is run via "runSandagAbm_MCDiagnostic.cmd" +# The mode choice diagnostic tool uses the synthetic population files as inputs and +# outputs a tour file with utilities and probabilities for each tour mode. +# +# Files referenced: +# input\mcd\destination_mgra.csv +# input\mcd\origin_mgra.csv +# input\mcd\syn_pop_attributes.yaml +# output\mcd\mcd_households.csv +# output\mcd\mcd_persons.csv +# output\mcd\mcd_output_households.csv +# output\mcd\mcd_output_persons.csv +# output\mcd\mcd_work_location.csv +# output\mcd\mcd_tour_file.csv +# conf\sandag_abm.properties +# bin\runSandagAbm_MCDiagnostic.cmd + +import inro.modeller as _m + +import pandas as pd +import collections, os +import shutil as _shutil +import yaml +import warnings +import traceback as _traceback +import tempfile as _tempfile +import subprocess as _subprocess + +warnings.filterwarnings("ignore") + +_join = os.path.join +_dir = os.path.dirname + +class mode_choice_diagnostic(_m.Tool()): + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = _dir(_m.Modeller().desktop.project.path) + self.main_directory = _dir(project_dir) + self.properties_path = _join(_dir(project_dir), "conf") + self.mcd_out_path = _join(_dir(project_dir), "output", "mcd") + self.syn_pop_attributes_path = _join(_dir(project_dir), "input", "mcd", "syn_pop_attributes.yaml") + self.origin_mgra_path = _join(_dir(project_dir), "input", "mcd", "origin_mgra.csv") + self.destination_mgra_path = _join(_dir(project_dir), "input", "mcd", "destination_mgra.csv") + self.household_df = pd.DataFrame() + self.household_out_df = pd.DataFrame() + self.person_df = pd.DataFrame() + self.person_out_df = pd.DataFrame() + self.work_location_df = pd.DataFrame() + self.tour_df = pd.DataFrame() + self.household_attributes = {} + self.person_attributes = {} + self.tour_attributes = {} + self._log_level = "ENABLED" + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Mode Choice Diagnostic Tool" + pb.description = """ + Diagnostic tool for the activity-based travel model mode choice results.
+
+
+ This tool first generates synthetic population files for specified target markets. + Users may edit target market attributes via a configuration file. + Users may additionally select origin and destination MAZs (i.e. MGRAs) of interest via + input CSV files.

+ The configuration file and MAZ selection CSV files are read from the following locations:
+
    +
  • input\mcd\syn_pop_attributes.yaml
  • +
  • input\mcd\origin_mgra.csv
  • +
  • input\mcd\destination_mgra.csv
  • +
+ The synthetic population generator outputs the following files:
+
    +
  • output\mcd\mcd_households.csv
  • +
  • output\mcd\mcd_persons.csv
  • +
  • output\mcd\mcd_output_households.csv
  • +
  • output\mcd\mcd_output_persons.csv
  • +
  • output\mcd\mcd_work_location.csv
  • +
  • output\mcd\mcd_tour_file.csv
  • +
+ Once all synthetic population files have been created, the script creates a copy of + the "sandag_abm.properties" file and modifies specific property parameters so that + it is compatible with the mode choice diagnostic tool. The modified properties + file is renamed and output as "conf\sandag_abm_mcd.properties"
+
+ Finally, the mode choice diagnostic tool is run via runSandagAbm_MCDiagnostic.cmd + The mode choice diagnostic tool uses the synthetic population files as inputs and + outputs a tour file with utilities and probabilities for each tour mode. The tour file + is output as "output\mcd\indivTourData_5.csv" +
+ """ + pb.branding_text = "SANDAG - Mode Choice Diagnostic Tool" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self() + run_msg = "Mode Choice Diagnostic Complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self): + _m.logbook_write("Started running mode choice diagnostic...") + + # check if transit shapefiles are present in mcd input directory + # if present, will move to mcd output directory + _m.logbook_write("Checking for transit shapefiles...") + self.check_shp() + + # run synthetic population generator + _m.logbook_write("Creating synthetic population...") + self.syn_pop() + + # copy and edit properties file + _m.logbook_write("Copying and editing properties file...") + mcd_props = self.copy_edit_props() + + self.set_global_logbook_level(mcd_props) + + drive, path_no_drive = os.path.splitdrive(self.main_directory) + + # run matrix manager + _m.logbook_write("Running matrix manager...") + self.run_proc("runMtxMgr.cmd", [drive, drive + path_no_drive], "Start matrix manager") + + # run driver + _m.logbook_write("Running JPPF driver...") + self.run_proc("runDriver.cmd", [drive, drive + path_no_drive], "Start JPPF driver") + + # run household manager + _m.logbook_write("Running household manager, JPPF driver, and nodes...") + self.run_proc("StartHHAndNodes.cmd", [drive, path_no_drive], "Start household manager, JPPF driver, and nodes") + + # run diagnostic tool + _m.logbook_write("Running mode choice diagnostic tool...") + path_forward_slash = path_no_drive.replace("\\", "/") + self.run_proc( + "runSandagAbm_MCDiagnostic.cmd", + [drive, drive + path_forward_slash, 1.0, 5], + "Java-Run Mode Choice Diagnostic Tool", capture_output=True) + + # move final output mcd files to the mcd output directory + self.move_mcd_files() + + def syn_pop(self): + # Creates sample synthetic population files for desired target market. Files will in turn + # be used as inputs to the mode choice diagnostic tool + + load_properties = _m.Modeller().tool("sandag.utilities.properties") + props = load_properties(self.properties_path) + + mgra_data_path = _join(self.main_directory, props["mgra.socec.file"]) + + file_paths = [self.syn_pop_attributes_path, self.origin_mgra_path, self.destination_mgra_path, mgra_data_path] + + for path in file_paths: + if not os.path.exists(path): + raise Exception("missing file '%s'" % (path)) + + # create output directory if it donesn't already exist + if not os.path.exists(self.mcd_out_path): + os.makedirs(self.mcd_out_path) + + # read inputs + mgra_data = pd.read_csv(mgra_data_path)[['mgra', 'taz']] + origin_mgra_data = list(set(pd.read_csv(self.origin_mgra_path)['MGRA'])) + destination_mgra_data = list(set(pd.read_csv(self.destination_mgra_path)['MGRA'])) + + # read synthetic population attributes + with open (self.syn_pop_attributes_path) as file: + syn_pop_attributes = yaml.load(file, Loader = yaml.Loader) + + self.household_attributes = syn_pop_attributes["household"] + self.person_attributes = syn_pop_attributes["person"] + self.tour_attributes = syn_pop_attributes["tour"] + + # create households input file + self.household_in(origin_mgra_data, mgra_data) + + # create households output file + self.household_out() + + # create persons input file + self.person_in() + + # create persons output file + self.person_out() + + # create output work location file + self.work_location(destination_mgra_data) + + # create individual tour file + self.tour() + + def household_in(self, origin_mgra_data, mgra_data): + # Creates the input household file + + # fixed household attributes + household = collections.OrderedDict([ + ('hworkers', [1]), # number of hh workers: one worker per household + ('persons', [2]), # number of hh persons: two persons per household + ('version', [0]), # synthetic population version + ]) + + household_df = pd.DataFrame.from_dict(household) + household_df = self.replicate_df_for_variable(household_df, 'mgra', origin_mgra_data) + for key, values in self.household_attributes.items(): + household_df = self.replicate_df_for_variable(household_df, key, self.maybe_list(values)) + household_df['hinccat1'] = household_df.apply(lambda hh_row: self.hinccat(hh_row), axis = 1) + household_df = self.replicate_df_for_variable(household_df, 'poverty', [1]) + household_df = pd.merge(left = household_df, right = mgra_data, on = 'mgra') + household_df = household_df.reset_index(drop = True) + household_df['hhid'] = household_df.index + 1 + household_df['household_serial_no'] = 0 + + # reorder columns + household_df = household_df[['hhid', 'household_serial_no', 'taz', 'mgra', 'hinccat1', 'hinc', 'hworkers', + 'veh','persons', 'hht', 'bldgsz', 'unittype', 'version', 'poverty']] + + self.household_df = household_df + + # print + household_df.to_csv(_join(self.mcd_out_path, 'mcd_households.csv'), index = False) + + def household_out(self): + # Creates the output household file + + household_out_df = self.household_df.copy() + household_out_df = household_out_df[['hhid', 'mgra', 'hinc', 'veh']] + household_out_df['transponder'] = 1 + household_out_df['cdap_pattern'] = 'MNj' + household_out_df['out_escort_choice'] = 0 + household_out_df['inb_escort_choice'] = 0 + household_out_df['jtf_choice'] = 0 + if self.tour_attributes['av_avail']: + household_out_df['AVs'] = household_out_df['veh'] + household_out_df['veh'] = 0 + else: + household_out_df['AVs'] = 0 + + # rename columns + household_out_df.rename(columns = {'hhid':'hh_id', 'mgra':'home_mgra', 'hinc':'income', 'veh':'HVs'}, + inplace = True) + + self.household_out_df = household_out_df + + # print + household_out_df.to_csv(_join(self.mcd_out_path, 'mcd_output_households.csv'), index = False) + + def person_in(self): + # Creates the input person file + + # fixed person attributes + persons = collections.OrderedDict([ + ('pnum', [1, 2]), # person number: two per household + ('pemploy', [1, 3]), # employment status: full-time employee and unemployed + ('ptype', [1, 4]), # person type: full-time worker and non-working adult + ('occen5', [0, 0]), # occupation + ('occsoc5', ['11-1021', '00-0000']), # occupation code# + ('indcen', [0, 0]), # industry code + ('weeks', [1, 0]), # weeks worked + ('hours', [35, 0]), # hours worked + ('race1p', [9, 9]), # race + ('hisp', [1, 1]), # hispanic flag + ('version', [9, 9]), # synthetic population run version: 9 is new for disaggregate population + ('timeFactorWork', [1, 1]), # work travel time factor: 1 is the mean + ('timeFactorNonWork', [1, 1]), # non work travel time factor: 2 is the mean + ('DAP', ['M', 'N']) # daily activity pattern: M (Mandatory), N (Non-Mandatory) + ]) + + persons.update(self.person_attributes) + person_df = pd.DataFrame.from_dict(persons) + person_df['join_key'] = 1 + self.household_df['join_key'] = 1 + person_df = pd.merge(left = person_df, right = self.household_df[['hhid','household_serial_no', 'join_key']]).\ + drop(columns = ['join_key']) + person_df['pstudent'] = person_df.apply(lambda person_row: self.pstudent(person_row), axis = 1) + person_df = person_df.sort_values(by = 'hhid') + person_df = person_df.reset_index(drop = True) + person_df['perid'] = person_df.index + 1 + + # reorder columns + person_df = person_df[['hhid', 'perid', 'household_serial_no', 'pnum', 'age', 'sex', 'miltary', 'pemploy', + 'pstudent', 'ptype', 'educ', 'grade', 'occen5', 'occsoc5', 'indcen', 'weeks', 'hours', + 'race1p', 'hisp', 'version', 'timeFactorWork', 'timeFactorNonWork', 'DAP']] + + self.person_df = person_df + + # print + person_df.to_csv(_join(self.mcd_out_path, 'mcd_persons.csv'), index = False) + + def person_out(self): + # Creates the output person file + + person_out_df = self.person_df.copy() + person_out_df = person_out_df[['hhid', 'perid', 'pnum', 'age', 'sex', 'ptype', 'DAP', + 'timeFactorWork', 'timeFactorNonWork']] + person_out_df['gender'] = person_out_df['sex'].apply(lambda x: 'male' if x == 1 else 'female') + person_out_df['type'] = person_out_df.apply(lambda person_row: self.p_type(person_row), axis = 1) + person_out_df['value_of_time'] = 0 + person_out_df['imf_choice'] = person_out_df['pnum'].apply(lambda x: 1 if x == 1 else 0) + person_out_df['inmf_choice'] = person_out_df['pnum'].apply(lambda x: 0 if x == 1 else 36) + person_out_df['fp_choice'] = person_out_df['pnum'].apply(lambda x: 2 if x == 1 else -1) + person_out_df['reimb_pct'] = 0 + person_out_df['tele_choice'] = person_out_df['pnum'].apply(lambda x: 1 if x == 1 else -1) + person_out_df['ie_choice'] = 1 + + # drop columns not required + person_out_df.drop(columns = ['sex', 'ptype'], inplace = True) + + # rename columns + person_out_df.rename(columns = {'hhid':'hh_id', 'perid':'person_id', + 'pnum':'person_num', 'DAP':'activity_pattern'}, + inplace = True) + + # reorder columns + person_out_df = person_out_df[['hh_id', 'person_id', 'person_num', 'age', 'gender', 'type', 'value_of_time', + 'activity_pattern', 'imf_choice', 'inmf_choice', 'fp_choice', 'reimb_pct', + 'tele_choice', 'ie_choice', 'timeFactorWork', 'timeFactorNonWork']] + + self.person_out_df = person_out_df + + # print + person_out_df.to_csv(_join(self.mcd_out_path, 'mcd_output_persons.csv'), index = False) + + def work_location(self, destination_mgra_data): + # Creates the output work location file + + # create copies and subset household and person dataframes + household_subset_df = self.household_df.copy() + person_subset_df = self.person_df.copy() + household_subset_df = household_subset_df[['hhid', 'mgra', 'hinc']] + person_subset_df = person_subset_df[['hhid', 'perid', 'pnum', 'ptype', 'age', 'pemploy', 'pstudent']] + + # merge to create work location dataframe + work_location_df = pd.merge(left = household_subset_df, right = person_subset_df, on = 'hhid') + work_location_df['WorkSegment'] = work_location_df['pnum'].apply(lambda x: 0 if x == 1 else -1) + work_location_df['SchoolSegment'] = -1 + work_location_df = self.replicate_df_for_variable(work_location_df, 'WorkLocation', self.maybe_list(destination_mgra_data)) + work_location_df['WorkLocationDistance'] = 0 + work_location_df['WorkLocationLogsum'] = 0 + work_location_df['SchoolLocation'] = -1 + work_location_df['SchoolLocationDistance'] = 0 + work_location_df['SchoolLocationLogsum'] = 0 + + # rename columns + work_location_df.rename(columns = {'hhid':'HHID', 'mgra':'homeMGRA', 'hinc':'income', 'perid':'personID', + 'pnum':'personNum', 'ptype':'personType', 'age':'personAge', + 'pemploy':'Employment Category', 'pstudent':'StudentCategory'}, + inplace = True) + + # reorder columns + work_location_df = work_location_df[['HHID', 'homeMGRA', 'income', 'personID', 'personNum', 'personType', + 'personAge', 'Employment Category', 'StudentCategory', 'WorkSegment', + 'SchoolSegment', 'WorkLocation', 'WorkLocationDistance', 'WorkLocationLogsum', + 'SchoolLocation', 'SchoolLocationDistance', 'SchoolLocationLogsum']] + + self.work_location_df = work_location_df + + # print + work_location_df.to_csv(_join(self.mcd_out_path, 'mcd_work_location.csv'), index = False) + + def tour(self): + # Creates the individual tour file + + tour_df = self.work_location_df.copy() + tour_df = tour_df[['HHID', 'personID', 'personNum', 'personType', 'homeMGRA', 'WorkLocation']] + tour_df = tour_df.sort_values(by = list(tour_df.columns), ascending = True) + tour_df['tour_id'] = tour_df.groupby(['HHID', 'personID']).cumcount() + tour_df['tour_category'] = tour_df['personNum'].\ + apply(lambda x: 'INDIVIDUAL_MANDATORY' if x == 1 else 'INDIVIDUAL_NON_MANDATORY') + tour_df['tour_purpose'] = tour_df['personNum'].apply(lambda x: 'Work' if x == 1 else 'Shop') + tour_df['start_period'] = tour_df['personNum'].apply(lambda x: self.tour_attributes['start_period'][0] if x == 1 else self.tour_attributes['start_period'][1]) + tour_df['end_period'] = tour_df['personNum'].apply(lambda x: self.tour_attributes['end_period'][0] if x == 1 else self.tour_attributes['end_period'][1]) + tour_df['tour_mode'] = 0 + if self.tour_attributes['av_avail']: + tour_df['av_avail'] = 1 + else: + tour_df['av_avail'] = 0 + tour_df['tour_distance'] = 0 + tour_df['atwork_freq'] = tour_df['personNum'].apply(lambda x: 1 if x == 1 else 0) + tour_df['num_ob_stops'] = 0 + tour_df['num_ib_stops'] = 0 + tour_df['valueOfTime'] = 0 + tour_df['escort_type_out'] = 0 + tour_df['escort_type_in'] = 0 + tour_df['driver_num_out'] = 0 + tour_df['driver_num_in'] = 0 + + # utilities 1 through 13 + util_cols = [] + for x in range(1, 14, 1): + col_name = 'util_{}'.format(x) + tour_df[col_name] = 0 + util_cols.append(col_name) + + # probabilities 1 through 13 + prob_cols = [] + for x in range(1, 14, 1): + col_name = 'prob_{}'.format(x) + tour_df[col_name] = 0 + prob_cols.append(col_name) + + # rename columns + tour_df.rename(columns = {'HHID':'hh_id', 'personID':'person_id', 'personNum':'person_num', + 'personType':'person_type', 'homeMGRA':'orig_mgra', 'WorkLocation':'dest_mgra'}, + inplace = True) + + # reorder columns + tour_df = tour_df[['hh_id', 'person_id', 'person_num', 'person_type', 'tour_id', 'tour_category', + 'tour_purpose', 'orig_mgra', 'dest_mgra', 'start_period', 'end_period', + 'tour_mode', 'av_avail', 'tour_distance', 'atwork_freq', 'num_ob_stops', + 'num_ib_stops', 'valueOfTime', 'escort_type_out', 'escort_type_in', + 'driver_num_out', 'driver_num_in'] + util_cols + prob_cols] + + self.tour_df = tour_df + + # print + tour_df.to_csv(_join(self.mcd_out_path, 'mcd_tour_file.csv'), index = False) + + def replicate_df_for_variable(self, df, var_name, var_values): + new_var_df = pd.DataFrame({var_name: var_values}) + new_var_df['join_key'] = 1 + df['join_key'] = 1 + + ret_df = pd.merge(left = df, right = new_var_df, how = 'outer').drop(columns=['join_key']) + return ret_df + + def maybe_list(self, values): + if (type(values) is not list) and (type(values) is not int): + raise Exception('Attribute values may only be of type list or int.') + if type(values) is not list: + return [values] + else: + return values + + def hinccat(self, hh_row): + if hh_row['hinc'] < 30000: + return 1 + if hh_row['hinc'] >= 30000 and hh_row['hinc'] < 60000: + return 2 + if hh_row['hinc'] >= 60000 and hh_row['hinc'] < 100000: + return 3 + if hh_row['hinc'] >= 100000 and hh_row['hinc'] < 150000: + return 4 + if hh_row['hinc'] >= 150000: + return 5 + + def pstudent(self, person_row): + if person_row['grade'] == 0: + return 3 + if person_row['grade'] == 1: + return 1 + if person_row['grade'] == 2: + return 1 + if person_row['grade'] == 3: + return 1 + if person_row['grade'] == 4: + return 1 + if person_row['grade'] == 5: + return 1 + if person_row['grade'] == 6: + return 2 + if person_row['grade'] == 7: + return 2 + + def p_type(self, person_row): + if person_row['ptype'] == 1: + return 'Full-time worker' + if person_row['ptype'] == 2: + return 'Part-time worker' + if person_row['ptype'] == 3: + return 'University student' + if person_row['ptype'] == 4: + return 'Non-worker' + if person_row['ptype'] == 5: + return 'Retired' + if person_row['ptype'] == 6: + return 'Student of driving age' + if person_row['ptype'] == 7: + return 'Student of non-driving age' + if person_row['ptype'] == 8: + return 'Child too young for school' + + def copy_edit_props(self): + # Copy and edit properties file tokens to be compatible with the mode choice diagnostic tool + + load_properties = _m.Modeller().tool("sandag.utilities.properties") + mcd_props = load_properties(_join(self.properties_path, "sandag_abm.properties")) + + # update properties + + # PopSyn inputs + mcd_props["RunModel.MandatoryTourModeChoice"] = "true" + mcd_props["RunModel.IndividualNonMandatoryTourModeChoice"] = "true" + + # data file paths + mcd_props["PopulationSynthesizer.InputToCTRAMP.HouseholdFile"] = "output/mcd/mcd_households.csv" + mcd_props["PopulationSynthesizer.InputToCTRAMP.PersonFile"] = "output/mcd/mcd_persons.csv" + mcd_props["Accessibilities.HouseholdDataFile"] = "output/mcd/mcd_output_households.csv" + mcd_props["Accessibilities.PersonDataFile"] = "output/mcd/mcd_output_persons.csv" + mcd_props["Accessibilities.IndivTourDataFile"] = "output/mcd/mcd_tour_file.csv" + mcd_props["Accessibilities.JointTourDataFile"] = "" + mcd_props["Accessibilities.IndivTripDataFile"] = "" + mcd_props["Accessibilities.JointTripDataFile"] = "" + + # model component run flags + mcd_props["RunModel.PreAutoOwnership"] = "false" + mcd_props["RunModel.UsualWorkAndSchoolLocationChoice"] = "false" + mcd_props["RunModel.AutoOwnership"] = "false" + mcd_props["RunModel.TransponderChoice"] = "false" + mcd_props["RunModel.FreeParking"] = "false" + mcd_props["RunModel.CoordinatedDailyActivityPattern"] = "false" + mcd_props["RunModel.IndividualMandatoryTourFrequency"] = "false" + mcd_props["RunModel.MandatoryTourModeChoice"] = "true" + mcd_props["RunModel.MandatoryTourDepartureTimeAndDuration"] = "false" + mcd_props["RunModel.SchoolEscortModel"] = "false" + mcd_props["RunModel.JointTourFrequency"] = "false" + mcd_props["RunModel.JointTourLocationChoice"] = "false" + mcd_props["RunModel.JointTourDepartureTimeAndDuration"] = "false" + mcd_props["RunModel.JointTourModeChoice"] = "true" + mcd_props["RunModel.IndividualNonMandatoryTourFrequency"] = "false" + mcd_props["RunModel.IndividualNonMandatoryTourLocationChoice"] = "false" + mcd_props["RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration"] = "false" + mcd_props["RunModel.IndividualNonMandatoryTourModeChoice"] = "true" + mcd_props["RunModel.AtWorkSubTourFrequency"] = "false" + mcd_props["RunModel.AtWorkSubTourLocationChoice"] = "false" + mcd_props["RunModel.AtWorkSubTourDepartureTimeAndDuration"] = "false" + mcd_props["RunModel.AtWorkSubTourModeChoice"] = "true" + mcd_props["RunModel.StopFrequency"] = "false" + mcd_props["RunModel.StopLocation"] = "false" + + mcd_props.save(_join(self.properties_path, "sandag_abm_mcd.properties")) + + return(mcd_props) + + def run_proc(self, name, arguments, log_message, capture_output=False): + path = _join(self.main_directory, "bin", name) + if not os.path.exists(path): + raise Exception("No command / batch file '%s'" % path) + command = path + " " + " ".join([str(x) for x in arguments]) + attrs = {"command": command, "name": name, "arguments": arguments} + with _m.logbook_trace(log_message, attributes=attrs): + if capture_output and self._log_level != "NO_EXTERNAL_REPORTS": + report = _m.PageBuilder(title="Process run %s" % name) + report.add_html('Command:

%s

' % command) + # temporary file to capture output error messages generated by Java + err_file_ref, err_file_path = _tempfile.mkstemp(suffix='.log') + err_file = os.fdopen(err_file_ref, "w") + try: + output = _subprocess.check_output(command, stderr=err_file, cwd=self.main_directory, shell=True) + report.add_html('Output:

%s
' % output) + except _subprocess.CalledProcessError as error: + report.add_html('Output:

%s
' % error.output) + raise + finally: + err_file.close() + with open(err_file_path, 'r') as f: + error_msg = f.read() + os.remove(err_file_path) + if error_msg: + report.add_html('Error message(s):

%s
' % error_msg) + try: + # No raise on writing report error + # due to observed issue with runs generating reports which cause + # errors when logged + _m.logbook_write("Process run %s report" % name, report.render()) + except Exception as error: + print _time.strftime("%Y-%M-%d %H:%m:%S") + print "Error writing report '%s' to logbook" % name + print error + print _traceback.format_exc(error) + if self._log_level == "DISABLE_ON_ERROR": + _m.logbook_level(_m.LogbookLevel.NONE) + else: + _subprocess.check_call(command, cwd=self.main_directory, shell=True) + + def set_global_logbook_level(self, props): + self._log_level = props.get("RunModel.LogbookLevel", "ENABLED") + log_all = _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.VALUE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG + log_states = { + "ENABLED": log_all, + "DISABLE_ON_ERROR": log_all, + "NO_EXTERNAL_REPORTS": log_all, + "NO_REPORTS": _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, + "TITLES_ONLY": _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, + "DISABLED": _m.LogbookLevel.NONE, + } + _m.logbook_write("Setting logbook level to %s" % self._log_level) + try: + _m.logbook_level(log_states[self._log_level]) + except KeyError: + raise Exception("properties.RunModel.LogLevel: value must be one of %s" % ",".join(log_states.keys())) + + def move_mcd_files(self): + + out_directory = _join(self.main_directory, "output") + + hh_data = "householdData_5.csv" + ind_tour = "indivTourData_5.csv" + ind_trip = "indivTripData_5.csv" + joint_tour = "jointTourData_5.csv" + joint_trip = "jointTripData_5.csv" + per_data = "personData_5.csv" + mgra_park = "mgraParkingCost.csv" + + files = [hh_data, ind_tour, ind_trip, joint_tour, joint_trip, per_data, mgra_park] + + for file in files: + src = _join(out_directory, file) + if not os.path.exists(src): + raise Exception("missing output file '%s'" % (src)) + dst = _join(self.mcd_out_path, file) + _shutil.move(src, dst) + + def check_shp(self): + + in_directory = _join(self.main_directory, "input", "mcd") + out_directory = self.mcd_out_path + + shp_names = ["tapcov", "rtcov"] + + for shp in shp_names: + + files_to_move = [f for f in os.listdir(in_directory) if shp in f] + + for file in files_to_move: + + src = _join(in_directory, file) + dst = _join(out_directory, file) + if not os.path.exists(src): + raise Exception("missing shapefile '%s'" % (src)) + _shutil.move(src, dst) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py new file mode 100644 index 0000000..c0df738 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py @@ -0,0 +1,302 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export/export_data_loader_matrices.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports the matrix results to OMX and csv files for use by the Java Data +# export process and the Data loader to the reporting database. +# +# +# Inputs: +# output_dir: the output directory for the created files +# base_scenario_id: scenario ID for the base scenario (same used in the Import network tool) +# transit_scenario_id: scenario ID for the base transit scenario +# +# Files created: +# CSV format files +# ../report/trucktrip.csv +# ../report/eetrip.csv +# ../report/eitrip.csv +# OMX format files +# trip_pp.omx +# +# +# Script example: +""" + import os + import inro.emme.database.emmebank as _eb + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + output_dir = os.path.join(main_directory, "output") + num_processors = "MAX-1" + export_data_loader_matrices = modeller.tool( + "sandag.export.export_data_loader_matrices") + export_data_loader_matrices(output_dir, 100, main_emmebank, transit_emmebank, num_processors) +""" +TOOLBOX_ORDER = 74 + + +import inro.modeller as _m +import inro.emme.database.emmebank as _eb +import traceback as _traceback +from collections import OrderedDict +import os +import numpy +import warnings +import tables + + +warnings.filterwarnings('ignore', category=tables.NaturalNameWarning) +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + +_join = os.path.join +_dir = os.path.dirname + +class ExportDataLoaderMatrices(_m.Tool(), gen_utils.Snapshot): + + output_dir = _m.Attribute(str) + base_scenario_id = _m.Attribute(int) + transit_scenario_id = _m.Attribute(int) + + tool_run_msg = "" + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.output_dir = os.path.join(os.path.dirname(project_dir), "output") + self.base_scenario_id = 100 + self.transit_scenario_id = 100 + self.periods = ["EA", "AM", "MD", "PM", "EV"] + self.attributes = ["main_directory", "base_scenario_id", "transit_scenario_id"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export matrices for Data Loader" + pb.description = """ + Export model results to OMX files for export by Data Exporter + to CSV format for load in SQL Data loader.""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('output_dir', 'directory', + title='Select output directory') + + pb.add_text_box('base_scenario_id', title="Base scenario ID:", size=10) + pb.add_text_box('transit_scenario_id', title="Transit scenario ID:", size=10) + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + base_emmebank = _eb.Emmebank(os.path.join(project_dir, "Database", "emmebank")) + transit_emmebank = _eb.Emmebank(os.path.join(project_dir, "Database_transit", "emmebank")) + base_scenario = base_emmebank.scenario(self.base_scenario_id) + transit_scenario = transit_emmebank.scenario(self.transit_scenario_id) + + results = self(self.output_dir, base_scenario, transit_scenario) + run_msg = "Export completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export matrices for Data Loader", save_arguments=True) + def __call__(self, output_dir, base_scenario, transit_scenario): + attrs = { + "output_dir": output_dir, + "base_scenario_id": base_scenario.id, + "transit_scenario_id": transit_scenario.id, + "self": str(self) + } + gen_utils.log_snapshot("Export Matrices for Data Loader", str(self), attrs) + self.output_dir = output_dir + self.base_scenario = base_scenario + self.transit_scenario = transit_scenario + + self.truck_demand() + self.external_demand() + self.total_demand() + + @_m.logbook_trace("Export truck demand") + def truck_demand(self): + name_mapping = [ + # ("lhdn", "TRKLGP", 1.3), + # ("mhdn", "TRKMGP", 1.5), + # ("hhdn", "TRKHGP", 2.5), + ("lhdt", "TRK_L", 1.3), + ("mhdt", "TRK_M", 1.5), + ("hhdt", "TRK_H", 2.5), + ] + scenario = self.base_scenario + emmebank = scenario.emmebank + zones = scenario.zone_numbers + formater = lambda x: ("%.5f" % x).rstrip('0').rstrip(".") + truck_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "trucktrip.csv") + + # get auto operating cost + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.output_dir), "conf", "sandag_abm.properties")) + try: + aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) + except ValueError: + raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") + + with open(truck_trip_path, 'w') as f: + f.write("OTAZ,DTAZ,TOD,MODE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") + for period in self.periods: + for key, name, pce in name_mapping: + matrix_data = emmebank.matrix(period + "_" + name + "_VEH").get_data(scenario) + matrix_data_time = emmebank.matrix(period + "_" + name + "_TIME").get_data(scenario) + matrix_data_dist = emmebank.matrix(period + "_" + name + "_DIST").get_data(scenario) + matrix_data_tollcost = emmebank.matrix(period + "_" + name + "_TOLLCOST").get_data(scenario) + rounded_demand = 0 + for orig in zones: + for dest in zones: + value = matrix_data.get(orig, dest) + # skip trips less than 0.00001 to avoid 0 trips records in database + if value < 0.00001: + rounded_demand += value + continue + time = matrix_data_time.get(orig, dest) + distance = matrix_data_dist.get(orig, dest) + tollcost = matrix_data_tollcost.get(orig, dest) + od_aoc = distance * aoc + f.write(",".join([str(orig), str(dest), period, key, formater(value), formater(time), formater(distance), formater(od_aoc), formater(tollcost)])) + f.write("\n") + if rounded_demand > 0: + print period + "_" + name + "_VEH", "rounded_demand", rounded_demand + + def external_demand(self): + #get auto operating cost + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.output_dir), "conf", "sandag_abm.properties")) + try: + aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) + except ValueError: + raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") + + # EXTERNAL-EXTERNAL TRIP TABLE (toll-eligible) + name_mapping = [ + ("DA", "SOV"), + ("S2", "HOV2"), + ("S3", "HOV3"), + ] + scenario = self.base_scenario + emmebank = scenario.emmebank + zones = scenario.zone_numbers + formater = lambda x: ("%.5f" % x).rstrip('0').rstrip(".") + ee_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "eetrip.csv") + with _m.logbook_trace("Export external-external demand"): + with open(ee_trip_path, 'w') as f: + f.write("OTAZ,DTAZ,TOD,MODE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") + for period in self.periods: + matrix_data_time = emmebank.matrix(period + "_SOV_NT_M_TIME").get_data(scenario) + matrix_data_dist = emmebank.matrix(period + "_SOV_NT_M_DIST").get_data(scenario) + matrix_data_tollcost = emmebank.matrix(period + "_SOV_NT_M_TOLLCOST").get_data(scenario) + for key, name in name_mapping: + matrix_data = emmebank.matrix(period + "_" + name + "_EETRIPS").get_data(scenario) + rounded_demand = 0 + for orig in zones: + for dest in zones: + value = matrix_data.get(orig, dest) + # skip trips less than 0.00001 to avoid 0 trips records in database + if value < 0.00001: + rounded_demand += value + continue + time = matrix_data_time.get(orig, dest) + distance = matrix_data_dist.get(orig, dest) + tollcost = 0 + tollcost = matrix_data_tollcost.get(orig, dest) + od_aoc = distance * aoc + f.write(",".join( + [str(orig), str(dest), period, key, formater(value), formater(time), + formater(distance), formater(od_aoc), formater(tollcost)])) + f.write("\n") + if rounded_demand > 0: + print period + "_" + name + "_EETRIPS", "rounded_demand", rounded_demand + + # EXTERNAL-INTERNAL TRIP TABLE + name_mapping = [ + ("DAN", "SOVGP"), + ("DAT", "SOVTOLL"), + ("S2N", "HOV2HOV"), + ("S2T", "HOV2TOLL"), + ("S3N", "HOV3HOV"), + ("S3T", "HOV3TOLL"), + ] + ei_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "eitrip.csv") + + with _m.logbook_trace("Export external-internal demand"): + with open(ei_trip_path, 'w') as f: + f.write("OTAZ,DTAZ,TOD,MODE,PURPOSE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") + for period in self.periods: + matrix_data_time = emmebank.matrix(period + "_SOV_TR_M_TIME").get_data(scenario) + matrix_data_dist = emmebank.matrix(period + "_SOV_TR_M_DIST").get_data(scenario) + if "TOLL" in name: + matrix_data_tollcost = emmebank.matrix(period + "_SOV_NT_M_TOLLCOST").get_data(scenario) + for purpose in ["WORK", "NONWORK"]: + for key, name in name_mapping: + matrix_data = emmebank.matrix(period + "_" + name + "_EI" + purpose).get_data(scenario) + rounded_demand = 0 + for orig in zones: + for dest in zones: + value = matrix_data.get(orig, dest) + # skip trips less than 0.00001 to avoid 0 trips records in database + if value < 0.00001: + rounded_demand += value + continue + time = matrix_data_time.get(orig, dest) + distance = matrix_data_dist.get(orig, dest) + tollcost = 0 + if "TOLL" in name: + tollcost = matrix_data_tollcost.get(orig, dest) + od_aoc = distance * aoc + f.write(",".join( + [str(orig), str(dest), period, key, purpose, formater(value), formater(time), + formater(distance), formater(od_aoc), formater(tollcost)])) + f.write("\n") + if rounded_demand > 0: + print period + "_" + name + "_EI" + purpose, "rounded_demand", rounded_demand + + @_m.logbook_trace("Export total auto and truck demand to OMX") + def total_demand(self): + for period in self.periods: + matrices = { + "%s_SOV_NT_L": 'mf"%s_SOV_NT_L"', + "%s_SOV_TR_L": 'mf"%s_SOV_TR_L"', + "%s_HOV2_L": 'mf"%s_HOV2_L"', + "%s_HOV3_L": 'mf"%s_HOV3_L"', + "%s_SOV_NT_M": 'mf"%s_SOV_NT_M"', + "%s_SOV_TR_M": 'mf"%s_SOV_TR_M"', + "%s_HOV2_M": 'mf"%s_HOV2_M"', + "%s_HOV3_M": 'mf"%s_HOV3_M"', + "%s_SOV_NT_H": 'mf"%s_SOV_NT_H"', + "%s_SOV_TR_H": 'mf"%s_SOV_TR_H"', + "%s_HOV2_H": 'mf"%s_HOV2_H"', + "%s_HOV3_H": 'mf"%s_HOV3_H"', + "%s_TRK_H": 'mf"%s_TRK_H"', + "%s_TRK_L": 'mf"%s_TRK_L"', + "%s_TRK_M": 'mf"%s_TRK_M"', + } + matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) + omx_file = os.path.join(self.output_dir, "trip_%s.omx" % period) + with gen_utils.ExportOMX(omx_file, self.base_scenario) as exporter: + exporter.write_matrices(matrices) + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py new file mode 100644 index 0000000..ec38ed3 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py @@ -0,0 +1,1203 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export_data_loader_network.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports the network results to csv file for use by the Java Data export process +# and the Data loader to the reporting database. +# +# +# Inputs: +# main_directory: main ABM directory +# base_scenario_id: scenario ID for the base scenario (same used in the Import network tool) +# traffic_emmebank: the base, traffic, Emme database +# transit_emmebank: the transit database +# num_processors: number of processors to use in the transit analysis calculations +# +# Files created: +# report/hwyload_pp.csv +# report/hwy_tcad.csv rename to hwyTcad.csv +# report/transit_aggflow.csv +# report/transit_flow.csv +# report/transit_onoff.csv +# report/trrt.csv rename to transitRoute.csv +# report/trstop.csv renmae to transitStop.csv +# report/transitTap.csv +# report/transitLink.csv +# +# Script example: +""" + import os + import inro.emme.database.emmebank as _eb + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database_transit", "emmebank")) + num_processors = "MAX-1" + export_data_loader_network = modeller.tool( + "sandag.export.export_data_loader_network") + export_data_loader_network(main_directory, 100, main_emmebank, transit_emmebank, num_processors) +""" + +TOOLBOX_ORDER = 73 + + +import inro.modeller as _m +import traceback as _traceback +import inro.emme.database.emmebank as _eb +import inro.emme.desktop.worksheet as _ws +import inro.emme.datatable as _dt +import inro.emme.core.exception as _except +from contextlib import contextmanager as _context +from collections import OrderedDict +from itertools import chain as _chain +import math +import os +import pandas as pd +import numpy as _np + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + +format = lambda x: ("%.6f" % x).rstrip('0').rstrip(".") +id_format = lambda x: str(int(x)) + +class ExportDataLoaderNetwork(_m.Tool(), gen_utils.Snapshot): + + main_directory = _m.Attribute(str) + base_scenario_id = _m.Attribute(int) + traffic_emmebank = _m.Attribute(str) + transit_emmebank = _m.Attribute(str) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.main_directory = os.path.dirname(project_dir) + self.base_scenario_id = 100 + self.traffic_emmebank = os.path.join(project_dir, "Database", "emmebank") + self.transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") + self.num_processors = "MAX-1" + self.attributes = ["main_directory", "base_scenario_id", "traffic_emmebank", "transit_emmebank", "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export network for Data Loader" + pb.description = """ +Export network results to csv files for SQL data loader.""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('main_directory', 'directory', + title='Select main directory') + + pb.add_text_box('base_scenario_id', title="Base scenario ID:", size=10) + pb.add_select_file('traffic_emmebank', 'file', + title='Select traffic emmebank') + pb.add_select_file('transit_emmebank', 'file', + title='Select transit emmebank') + + dem_utils.add_select_processors("num_processors", pb, self) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + results = self(self.main_directory, self.base_scenario_id, + self.traffic_emmebank, self.transit_emmebank, + self.num_processors) + run_msg = "Export completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export network results for Data Loader", save_arguments=True) + def __call__(self, main_directory, base_scenario_id, traffic_emmebank, transit_emmebank, num_processors): + attrs = { + "traffic_emmebank": str(traffic_emmebank), + "transit_emmebank": str(transit_emmebank), + "main_directory": main_directory, + "base_scenario_id": base_scenario_id, + "self": str(self) + } + gen_utils.log_snapshot("Export network results", str(self), attrs) + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + + traffic_emmebank = _eb.Emmebank(traffic_emmebank) + transit_emmebank = _eb.Emmebank(transit_emmebank) + export_path = os.path.join(main_directory, "report") + input_path = os.path.join(main_directory,"input") + num_processors = dem_utils.parse_num_processors(num_processors) + + periods = ["EA", "AM", "MD", "PM", "EV"] + period_scenario_ids = OrderedDict((v, i) for i, v in enumerate(periods, start=base_scenario_id + 1)) + + base_scenario = traffic_emmebank.scenario(base_scenario_id) + + self.export_traffic_attribute(base_scenario, export_path, traffic_emmebank, period_scenario_ids, props) + self.export_traffic_load_by_period(export_path, traffic_emmebank, period_scenario_ids) + self.export_transit_results(export_path, input_path, transit_emmebank, period_scenario_ids, num_processors) + self.export_geometry(export_path, traffic_emmebank) + + @_m.logbook_trace("Export traffic attribute data") + def export_traffic_attribute(self, base_scenario, export_path, traffic_emmebank, period_scenario_ids, props): + # Several column names are legacy from the original network files + # and data loader process, and are populated with zeros. + # items are ("column name", "attribute name") or ("column name", ("attribute name", default)) + hwylink_attrs = [ + ("ID", "@tcov_id"), + ("Length", "length"), + ("Dir", "is_one_way"), + ("hwycov-id:1", "@tcov_id"), + ("ID:1", "@tcov_id"), + ("Length:1", "length_feet"), + ("QID", "zero"), + ("CCSTYLE", "zero"), + ("UVOL", "zero"), + ("AVOL", "zero"), + ("TMP1", "zero"), + ("TMP2", "zero"), + ("PLOT", "zero"), + ("SPHERE", "@sphere"), + ("RTNO", "zero"), + ("LKNO", "zero"), + ("NM", "#name"), + ("FXNM", "#name_from"), + ("TXNM", "#name_to"), + ("AN", "i"), + ("BN", "j"), + ("COJUR", "zero"), + ("COSTAT", "zero"), + ("COLOC", "zero"), + ("RLOOP", "zero"), + ("ADTLK", "zero"), + ("ADTVL", "zero"), + ("PKPCT", "zero"), + ("TRPCT", "zero"), + ("SECNO", "zero"), + ("DIR:1", "zero"), + ("FFC", "type"), + ("CLASS", "zero"), + ("ASPD", "@speed_adjusted"), + ("IYR", "@year_open_traffic"), + ("IPROJ", "@project_code"), + ("IJUR", "@jurisdiction_type"), + ("IFC", "type"), + ("IHOV", "@lane_restriction"), + ("ITRUCK", "@truck_restriction"), + ("ISPD", "@speed_posted"), + ("ITSPD", "zero"), + ("IWAY", "iway"), + ("IMED", "@median"), + ("COST", "@cost_operating"), + ("ITOLLO", "@toll_md"), + ("ITOLLA", "@toll_am"), + ("ITOLLP", "@toll_pm"), + ] + directional_attrs = [ + ("ABLNO", "@lane_md", "0"), + ("ABLNA", "@lane_am", "0"), + ("ABLNP", "@lane_pm", "0"), + ("ABAU", "@lane_auxiliary", "0"), + ("ABPCT", "zero", "0"), + ("ABPHF", "zero", "0"), + ("ABCNT", "@traffic_control", "0"), + ("ABTL", "@turn_thru", "0"), + ("ABRL", "@turn_right", "0"), + ("ABLL", "@turn_left", "0"), + ("ABTLB", "zero", "0"), + ("ABRLB", "zero", "0"), + ("ABLLB", "zero", "0"), + ("ABGC", "@green_to_cycle_init", "0"), + ("ABPLC", "per_lane_capacity", "1900"), + ("ABCPO", "@capacity_link_md", "999999"), + ("ABCPA", "@capacity_link_am", "999999"), + ("ABCPP", "@capacity_link_pm", "999999"), + ("ABCXO", "@capacity_inter_md", "999999"), + ("ABCXA", "@capacity_inter_am", "999999"), + ("ABCXP", "@capacity_inter_pm", "999999"), + ("ABCHO", "@capacity_hourly_op", "0"), + ("ABCHA", "@capacity_hourly_am", "0"), + ("ABCHP", "@capacity_hourly_pm", "0"), + ("ABTMO", "@time_link_md", "999"), + ("ABTMA", "@time_link_am", "999"), + ("ABTMP", "@time_link_pm", "999"), + ("ABTXO", "@time_inter_md", "0"), + ("ABTXA", "@time_inter_am", "0"), + ("ABTXP", "@time_inter_pm", "0"), + ("ABCST", "zero", "999.999"), + ("ABVLA", "zero", "0"), + ("ABVLP", "zero", "0"), + ("ABLOS", "zero", "0"), + ] + for key, name, default in directional_attrs: + hwylink_attrs.append((key, name)) + for key, name, default in directional_attrs: + hwylink_attrs.append(("BA" + key[2:], (name, default))) + hwylink_attrs.append(("relifac", "relifac")) + + time_period_atts = [ + ("ITOLL2", "@toll"), + ("ITOLL3", "@cost_auto"), + ("ITOLL4", "@cost_med_truck"), + ("ITOLL5", "@cost_hvy_truck"), + ("ITOLL", "toll_hov"), + ("ABCP", "@capacity_link", "999999"), + ("ABCX", "@capacity_inter", "999999"), + ("ABTM", "@time_link", "999"), + ("ABTX", "@time_inter", "0"), + ("ABLN", "@lane", "0"), + ("ABSCST", "sov_total_gencost", ""), + ("ABH2CST", "hov2_total_gencost", ""), + ("ABH3CST", "hov3_total_gencost", ""), + ("ABSTM", "auto_time", ""), + ("ABHTM", "auto_time", ""), + ] + periods = ["_ea", "_am", "_md", "_pm", "_ev"] + for column in time_period_atts: + for period in periods: + key = column[0] + period.upper() + name = column[1] + period + hwylink_attrs.append((key, name)) + if key.startswith("AB"): + for period in periods: + key = column[0] + period.upper() + name = column[1] + period + default = column[2] + hwylink_attrs.append(("BA" + key[2:], (name, default))) + for period in periods: + key = "ABPRELOAD" + period.upper() + name = "additional_volume" + period + default = "0" + hwylink_attrs.append((key, name)) + hwylink_attrs.append(("BA" + key[2:], (name, default))) + + vdf_attrs = [ + ("AB_GCRatio", "@green_to_cycle", ""), + ("AB_Cycle", "@cycle", ""), + ("AB_PF", "progression_factor", ""), + ("ALPHA1", "alpha1", "0.8"), + ("BETA1", "beta1", "4"), + ("ALPHA2", "alpha2", "4.5"), + ("BETA2", "beta2", "2"), + ] + for key, name, default in vdf_attrs: + name = name + "_am" if name.startswith("@") else name + hwylink_attrs.append((key, name)) + if key.startswith("AB"): + hwylink_attrs.append(("BA" + key[2:], (name, default))) + for period in periods: + for key, name, default in vdf_attrs: + name = name + period if name.startswith("@") else name + default = default or "0" + hwylink_attrs.append((key + period.upper(), name)) + if key.startswith("AB"): + hwylink_attrs.append(("BA" + key[2:] + period.upper(), (name, default))) + + network = base_scenario.get_partial_network(["LINK"], include_attributes=True) + + #copy assignment from period scenarios + for period, scenario_id in period_scenario_ids.iteritems(): + from_scenario = traffic_emmebank.scenario(scenario_id) + src_attrs = ["@auto_time", "additional_volume"] + dst_attrs = ["auto_time_" + period.lower(), + "additional_volume_" + period.lower()] + for dst_attr in dst_attrs: + network.create_attribute("LINK", dst_attr) + values = from_scenario.get_attribute_values("LINK", src_attrs) + network.set_attribute_values("LINK", dst_attrs, values) + # add in and calculate additional columns + new_attrs = [ + ("zero", 0), ("is_one_way", 0), ("iway", 2), ("length_feet", 0), + ("toll_hov", 0), ("per_lane_capacity", 1900), + ("progression_factor", 1.0), ("alpha1", 0.8), ("beta1", 4.0), + ("alpha2", 4.5), ("beta2", 2.0), ("relifac", 1.0), + ] + for name, default in new_attrs: + network.create_attribute("LINK", name, default) + for period in periods: + network.create_attribute("LINK", "toll_hov" + period, 0) + network.create_attribute("LINK", "sov_total_gencost" + period, 0) + network.create_attribute("LINK", "hov2_total_gencost" + period, 0) + network.create_attribute("LINK", "hov3_total_gencost" + period, 0) + for link in network.links(): + link.is_one_way = 1 if link.reverse_link else 0 + link.iway = 2 if link.reverse_link else 1 + link.length_feet = link.length * 5280 + for period in periods: + link["toll_hov" + period] = link["@cost_hov2" + period] - link["@cost_operating"] + link["sov_total_gencost" + period] = link["auto_time" + period] + link["@cost_auto" + period] + link["hov2_total_gencost" + period] = link["auto_time" + period] + link["@cost_hov2" + period] + link["hov3_total_gencost" + period] = link["auto_time" + period] + link["@cost_hov3" + period] + if link.volume_delay_func == 24: + link.alpha2 = 6.0 + link.per_lane_capacity = max([(link["@capacity_link" + p] / link["@lane" + p]) + for p in periods if link["@lane" + p] > 0] + [0]) + + hwylink_atts_file = os.path.join(export_path, "hwy_tcad.csv") + busPCE = props["transit.bus.pceveh"] + self.export_traffic_to_csv(hwylink_atts_file, hwylink_attrs, network, busPCE) + + @_m.logbook_trace("Export traffic load data by period") + def export_traffic_load_by_period(self, export_path, traffic_emmebank, period_scenario_ids): + create_attribute = _m.Modeller().tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + net_calculator = _m.Modeller().tool( + "inro.emme.network_calculation.network_calculator") + hwyload_attrs = [("ID1", "@tcov_id")] + + dir_atts = [ + ("AB_Flow_PCE", "@pce_flow"), # sum of pce flow + ("AB_Time", "@auto_time"), # computed vdf based on pce flow + ("AB_VOC", "@voc"), + ("AB_V_Dist_T", "length"), + ("AB_VHT", "@vht"), + ("AB_Speed", "@speed"), + ("AB_VDF", "@msa_time"), + ("AB_MSA_Flow", "@msa_flow"), + ("AB_MSA_Time", "@msa_time"), + ("AB_Flow_SOV_NTPL", "@sov_nt_l"), + ("AB_Flow_SOV_TPL", "@sov_tr_l"), + ("AB_Flow_SR2L", "@hov2_l"), + ("AB_Flow_SR3L", "@hov3_l"), + ("AB_Flow_SOV_NTPM", "@sov_nt_m"), + ("AB_Flow_SOV_TPM", "@sov_tr_m"), + ("AB_Flow_SR2M", "@hov2_m"), + ("AB_Flow_SR3M", "@hov3_m"), + ("AB_Flow_SOV_NTPH", "@sov_nt_h"), + ("AB_Flow_SOV_TPH", "@sov_tr_h"), + ("AB_Flow_SR2H", "@hov2_h"), + ("AB_Flow_SR3H", "@hov3_h"), + ("AB_Flow_lhd", "@trk_l_non_pce"), + ("AB_Flow_mhd", "@trk_m_non_pce"), + ("AB_Flow_hhd", "@trk_h_non_pce"), + ("AB_Flow", "@non_pce_flow"), + ] + + for key, attr in dir_atts: + hwyload_attrs.append((key, attr)) + hwyload_attrs.append((key.replace("AB_", "BA_"), (attr, ""))) # default for BA on one-way links is blank + for p, scen_id in period_scenario_ids.iteritems(): + scenario = traffic_emmebank.scenario(scen_id) + new_atts = [ + ("@speed", "link travel speed", "length*60/@auto_time"), + ("@sov_nt_all", "total number of SOV GP vehicles", + "@sov_nt_l+@sov_nt_m+@sov_nt_h" ), + ("@sov_tr_all", "total number of SOV TOLL vehicles", + "@sov_tr_l+@sov_tr_m+@sov_tr_h" ), + ("@hov2_all", "total number of HOV2 HOV vehicles", + "@hov2_l+@hov2_m+@hov2_h" ), + ("@hov3_all", "total number of HOV3 HOV vehicles", + "@hov3_l+@hov3_m+@hov3_h" ), + ("@trk_l_non_pce", "total number of light trucks in non-Pce", + "(@trk_l)/1.3" ), + ("@trk_m_non_pce", "total medium trucks in non-Pce", + "(@trk_m)/1.5" ), + ("@trk_h_non_pce", "total heavy trucks in non-Pce", + "(@trk_h)/2.5" ), + ("@pce_flow", "total number of vehicles in Pce", + "@sov_nt_all+@sov_tr_all+ \ + @hov2_all+ \ + @hov3_all+ \ + (@trk_l) + (@trk_m) + \ + (@trk_h) + volad" ), + ("@non_pce_flow", "total number of vehicles in non-Pce", + "@sov_nt_all+@sov_tr_all+ \ + @hov2_all+ \ + @hov3_all+ \ + (@trk_l)/1.3 + (@trk_m)/1.5 + \ + (@trk_h)/2.5 + volad/3" ), #volad includes bus flow - pce factor is 3 + ("@msa_flow", "MSA flow", "@non_pce_flow"), #flow from final assignment + ("@msa_time", "MSA time", "timau"), #skim assignment time on msa flow + ("@voc", "volume over capacity", "@pce_flow/ul3"), #pce flow over road capacity + ("@vht", "vehicle hours travelled", "@non_pce_flow*@auto_time/60") #vehicle flow (non-pce)*time + ] + + for name, des, formula in new_atts: + att = scenario.extra_attribute(name) + if not att: + att = create_attribute("LINK", name, des, 0, overwrite=True, scenario=scenario) + cal_spec = {"result": att.id, + "expression": formula, + "aggregation": None, + "selections": {"link": "mode=d"}, + "type": "NETWORK_CALCULATION" + } + net_calculator(cal_spec, scenario=scenario) + file_path = os.path.join(export_path, "hwyload_%s.csv" % p) + network = self.get_partial_network(scenario, {"LINK": ["@tcov_id"] + [a[1] for a in dir_atts]}) + self.export_traffic_to_csv(file_path, hwyload_attrs, network) + + def export_traffic_to_csv(self, filename, att_list, network, busPCE = None): + auto_mode = network.mode("d") + # only the original forward direction links and auto links only + links = [l for l in network.links() + if l["@tcov_id"] > 0 and + (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes)) + ] + links.sort(key=lambda l: l["@tcov_id"]) + with open(filename, 'w') as fout: + fout.write(",".join(['"%s"' % x[0] for x in att_list])) + fout.write("\n") + for link in links: + key, att = att_list[0] # expected to be the link id + values = [id_format(link[att])] + reverse_link = link.reverse_link + for key, att in att_list[1:]: + if key == "AN": + values.append(link.i_node.id) + elif key == "BN": + values.append(link.j_node.id) + elif key.startswith("BA"): + name, default = att + if reverse_link and (abs(link["@tcov_id"]) == abs(reverse_link["@tcov_id"])): + if "additional_volume" in name: + values.append(format(float(reverse_link[name]) / busPCE)) + else: + values.append(format(reverse_link[name])) + else: + values.append(default) + + #values.append(format(reverse_link[name]) if reverse_link else default) + elif att.startswith("#"): + values.append('"%s"' % link[att]) + else: + if "additional_volume" in att: + values.append(format(float(link[att]) / busPCE)) + else: + values.append(format(link[att])) + fout.write(",".join(values)) + fout.write("\n") + + @_m.logbook_trace("Export transit results") + def export_transit_results(self, export_path, input_path, transit_emmebank, period_scenario_ids, num_processors): + # Note: Node analysis for transfers is VERY time consuming + # this implementation will be replaced when new Emme version is available + + trrt_atts = ["Route_ID","Route_Name","Mode","AM_Headway","PM_Headway","OP_Headway","Night_Headway","Night_Hours","Config","Fare"] + trstop_atts = ["Stop_ID","Route_ID","Link_ID","Pass_Count","Milepost","Longitude","Latitude","NearNode","FareZone","StopName"] + + #transit route file + trrt_infile = os.path.join(input_path, "trrt.csv") + trrt = pd.read_csv(trrt_infile) + trrt = trrt.rename(columns=lambda x:x.strip()) + trrt_out = trrt[trrt_atts] + trrt_outfile = os.path.join(export_path, "trrt.csv") + trrt_out.to_csv(trrt_outfile, index=False) + + #transit stop file + trstop_infile = os.path.join(input_path, "trstop.csv") + trstop = pd.read_csv(trstop_infile) + trstop = trstop.rename(columns={"HwyNode":"NearNode"}) + trstop = trstop.rename(columns=lambda x:x.strip()) + trstop_out = trstop[trstop_atts] + trstop_outfile = os.path.join(export_path, "trstop.csv") + trstop_out.to_csv(trstop_outfile, index=False) + + use_node_analysis_to_get_transit_transfers = False + + copy_scenario = _m.Modeller().tool( + "inro.emme.data.scenario.copy_scenario") + create_attribute = _m.Modeller().tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + net_calculator = _m.Modeller().tool( + "inro.emme.network_calculation.network_calculator") + copy_attribute= _m.Modeller().tool( + "inro.emme.data.network.copy_attribute") + delete_scenario = _m.Modeller().tool( + "inro.emme.data.scenario.delete_scenario") + transit_flow_atts = [ + "MODE", + "ACCESSMODE", + "TOD", + "ROUTE", + "FROM_STOP", + "TO_STOP", + "CENTROID", + "FROMMP", + "TOMP", + "TRANSITFLOW", + "BASEIVTT", + "COST", + "VOC", + ] + transit_aggregate_flow_atts = [ + "MODE", + "ACCESSMODE", + "TOD", + "LINK_ID", + "AB_TransitFlow", + "BA_TransitFlow", + "AB_NonTransit", + "BA_NonTransit", + "AB_TotalFlow", + "BA_TotalFlow", + "AB_Access_Walk_Flow", + "BA_Access_Walk_Flow", + "AB_Xfer_Walk_Flow", + "BA_Xfer_Walk_Flow", + "AB_Egress_Walk_Flow", + "BA_Egress_Walk_Flow" + ] + transit_onoff_atts = [ + "MODE", + "ACCESSMODE", + "TOD", + "ROUTE", + "STOP", + "BOARDINGS", + "ALIGHTINGS", + "WALKACCESSON", + "DIRECTTRANSFERON", + "WALKTRANSFERON", + "DIRECTTRANSFEROFF", + "WALKTRANSFEROFF", + "EGRESSOFF" + ] + + transit_flow_file = os.path.join(export_path, "transit_flow.csv") + fout_seg = open(transit_flow_file, 'w') + fout_seg.write(",".join(['"%s"' % x for x in transit_flow_atts])) + fout_seg.write("\n") + + transit_aggregate_flow_file = os.path.join(export_path, "transit_aggflow.csv") + fout_link = open(transit_aggregate_flow_file, 'w') + fout_link.write(",".join(['"%s"' % x for x in transit_aggregate_flow_atts])) + fout_link.write("\n") + + transit_onoff_file = os.path.join(export_path, "transit_onoff.csv") + fout_stop = open(transit_onoff_file, 'w') + fout_stop.write(",".join(['"%s"' % x for x in transit_onoff_atts])) + fout_stop.write("\n") + try: + access_modes = ["WLK", "PNR", "KNR"] + main_modes = ["BUS", "PREM","ALLPEN"] + all_modes = ["b", "c", "e", "l", "r", "p", "y", "o", "a", "w", "x"] + local_bus_modes = ["b", "a", "w", "x"] + premium_modes = ["c", "l", "e", "p", "r", "y", "o", "a", "w", "x"] + for tod, scen_id in period_scenario_ids.iteritems(): + with _m.logbook_trace("Processing period %s" % tod): + scenario = transit_emmebank.scenario(scen_id) + # attributes + total_walk_flow = create_attribute("LINK", "@volax", "total walk flow on links", + 0, overwrite=True, scenario=scenario) + segment_flow = create_attribute("TRANSIT_SEGMENT", "@voltr", "transit segment flow", + 0, overwrite=True, scenario=scenario) + link_transit_flow = create_attribute("LINK", "@link_voltr", "total transit flow on link", + 0, overwrite=True, scenario=scenario) + initial_boardings = create_attribute("TRANSIT_SEGMENT", + "@init_boardings", "transit initial boardings", + 0, overwrite=True, scenario=scenario) + xfer_boardings = create_attribute("TRANSIT_SEGMENT", + "@xfer_boardings", "transit transfer boardings", + 0, overwrite=True, scenario=scenario) + total_boardings = create_attribute("TRANSIT_SEGMENT", + "@total_boardings", "transit total boardings", + 0, overwrite=True, scenario=scenario) + final_alightings = create_attribute("TRANSIT_SEGMENT", + "@final_alightings", "transit final alightings", + 0, overwrite=True, scenario=scenario) + xfer_alightings = create_attribute("TRANSIT_SEGMENT", + "@xfer_alightings", "transit transfer alightings", + 0, overwrite=True, scenario=scenario) + total_alightings = create_attribute("TRANSIT_SEGMENT", + "@total_alightings", "transit total alightings", + 0, overwrite=True, scenario=scenario) + + access_walk_flow = create_attribute("LINK", + "@access_walk_flow", "access walks (orig to init board)", + 0, overwrite=True, scenario=scenario) + xfer_walk_flow = create_attribute("LINK", + "@xfer_walk_flow", "xfer walks (init board to final alight)", + 0, overwrite=True, scenario=scenario) + egress_walk_flow = create_attribute("LINK", + "@egress_walk_flow", "egress walks (final alight to dest)", + 0, overwrite=True, scenario=scenario) + + for main_mode in main_modes: + mode = main_mode + if main_mode == "BUS": + mode_list = local_bus_modes + elif main_mode == "PREM": + mode_list = premium_modes + else: + mode_list = all_modes + + for access_type in access_modes: + with _m.logbook_trace("Main mode %s access mode %s" % (main_mode, access_type)): + class_name = "%s_%s%s" % (tod, access_type, main_mode) + segment_results = { + "transit_volumes": segment_flow.id, + "initial_boardings": initial_boardings.id, + "total_boardings": total_boardings.id, + "final_alightings": final_alightings.id, + "total_alightings": total_alightings.id, + "transfer_boardings": xfer_boardings.id, + "transfer_alightings": xfer_alightings.id + } + link_results = { + "total_walk_flow": total_walk_flow.id, + "link_transit_flow": link_transit_flow.id, + "access_walk_flow": access_walk_flow.id, + "xfer_walk_flow": xfer_walk_flow.id, + "egress_walk_flow": egress_walk_flow.id + } + + self.calc_additional_results( + scenario, class_name, num_processors, + total_walk_flow, segment_results, link_transit_flow, + access_walk_flow, xfer_walk_flow, egress_walk_flow) + attributes = { + "NODE": ["@network_adj", "@network_adj_src"],#, "initial_boardings", "final_alightings"], + "LINK": link_results.values() + ["@tcov_id", "length"], + "TRANSIT_LINE": ["@route_id"], + "TRANSIT_SEGMENT": segment_results.values() + [ + "transit_time", "dwell_time", "@stop_id", "allow_boardings", "allow_alightings"], + } + network = self.get_partial_network(scenario, attributes) + self.collapse_network_adjustments(network, segment_results, link_results) + # =============================================== + # analysis for nodes with/without walk option + if use_node_analysis_to_get_transit_transfers: + stop_on, stop_off = self.transfer_analysis(scenario, class_name, num_processors) + else: + stop_on, stop_off = {}, {} + # =============================================== + transit_modes = [m for m in network.modes() if m.type in ("TRANSIT", "AUX_TRANSIT")] + links = [link for link in network.links() + if link["@tcov_id"] > 0 and (link.modes.union(transit_modes))] + links.sort(key=lambda l: l["@tcov_id"]) + lines = [line for line in network.transit_lines() if line.mode.id in mode_list] + lines.sort(key=lambda l: l["@route_id"]) + + label = ",".join([mode, access_type, tod]) + self.output_transit_flow(label, lines, segment_flow.id, fout_seg) + self.output_transit_aggregate_flow( + label, links, link_transit_flow.id, total_walk_flow.id, access_walk_flow.id, + xfer_walk_flow.id, egress_walk_flow.id, fout_link) + self.output_transit_onoff( + label, lines, total_boardings.id, total_alightings.id, initial_boardings.id, + xfer_boardings.id, xfer_alightings.id, final_alightings.id, + stop_on, stop_off, fout_stop) + finally: + fout_stop.close() + fout_link.close() + fout_seg.close() + return + + @_m.logbook_trace("Export geometries") + def export_geometry(self, export_path, traffic_emmebank): + # --------------------------Export Transit Nework Geometory----------------------------- + # domain: NODE, LINK, TURN, TRANSIT_LINE, TRANSIT_VEHICLE, TRANSIT_SEGMENT + def export_as_csv(domain, attributes, scenario = None): + if scenario is None: + scenario = _m.Modeller().scenario + initial_scenario = _m.Modeller().scenario + #if initial_scenario.number != scenario.number: + #data_explorer.replace_primary_scenario(scenario) + # Create the network table + network_table = project.new_network_table(domain) + for k, a in enumerate(attributes): + column = _ws.Column() + column.name = column.expression = a + network_table.add_column(k, column) + # Extract data + data = network_table.get_data() + f = _np.vectorize(lambda x: x.text) # required to get the WKT representation of the geometry column + data_dict = {} + for a in data.attributes(): + if isinstance(a, _dt.GeometryAttribute): + data_dict[a.name] = f(a.values) + else: + data_dict[a.name] = a.values + df = pd.DataFrame(data_dict) + + network_table.close() + #if initial_scenario.number != scenario.number: + # data_explorer.replace_primary_scenario(initial_scenario) + return df + + desktop = _m.Modeller().desktop + desktop.refresh_data() + data_explorer = desktop.data_explorer() + previous_active_database = data_explorer.active_database() + try: + desktop_traffic_database = data_explorer.add_database(traffic_emmebank.path) + desktop_traffic_database.open() + except Exception as error: + import traceback + print (traceback.format_exc()) + project = desktop.project + scenario = _m.Modeller().emmebank.scenario(101) + data_explorer.replace_primary_scenario(scenario) + node_attributes = ['i','@tap_id'] + link_attributes = ['i', 'j', '@tcov_id', 'modes'] + transit_line_attributes = ['line', 'routeID'] + transit_segment_attributes = ['line', 'i', 'j', 'loop_index','@tcov_id','@stop_id'] + mode_talbe = ['mode', 'type'] + network_table = project.new_network_table('MODE') + for k, a in enumerate(mode_talbe): + column = _ws.Column() + column.name = column.expression = a + network_table.add_column(k, column) + data = network_table.get_data() + data_dict = {} + for a in data.attributes(): + data_dict[a.name] = a.values + df = pd.DataFrame(data_dict) + mode_list = df[df['type'].isin([2.0, 3.0])]['mode'].tolist() + + df = export_as_csv('NODE', node_attributes, scenario) + df = df[['@tap_id', 'geometry']] + is_tap = df['@tap_id'] > 0 + df = df[is_tap] + df.columns = ['tapID', 'geometry'] + df.to_csv(os.path.join(export_path, 'transitTap.csv'), index=False) + + df = export_as_csv('TRANSIT_LINE', transit_line_attributes) + df = df[['line', 'geometry']] + df.columns = ['Route_Name', 'geometry'] + df['Route_Name'] = df['Route_Name'].astype(int) + df_routeFull = pd.read_csv(os.path.join(export_path, 'trrt.csv')) + result = pd.merge(df_routeFull, df, how='left', on=['Route_Name']) + result.to_csv(os.path.join(export_path, 'transitRoute.csv'), index=False) + os.remove(os.path.join(export_path, 'trrt.csv')) + + df = export_as_csv('TRANSIT_SEGMENT', transit_segment_attributes, None) + df_seg = df[['@tcov_id', 'geometry']] + df_seg.columns = ['trcovID', 'geometry'] + df_seg = df_seg.drop_duplicates() + #df_seg.to_csv(os.path.join(export_path, 'transitLink.csv'), index=False) + #df_stop = df[(df['@stop_id'] > 0) & (df['@tcov_id'] > 0)] + df_stop = df[(df['@stop_id'] > 0)] + df_stop = df_stop[['@stop_id', 'geometry']] + df_stop = df_stop.drop_duplicates() + df_stop.columns = ['Stop_ID', 'geometry'] + temp=[] + for value in df_stop['geometry']: + value=value.split(',') + value[0]=value[0]+')' + value[0]=value[0].replace("LINESTRING", "POINT") + temp.append(value[0]) + df_stop['geometry'] = temp + df_stopFull = pd.read_csv(os.path.join(export_path, 'trstop.csv')) + result = pd.merge(df_stopFull, df_stop, how='left', on=['Stop_ID']) + result.to_csv(os.path.join(export_path, 'transitStop.csv'), index=False) + os.remove(os.path.join(export_path, 'trstop.csv')) + + df = export_as_csv('LINK', link_attributes, None) + df_link = df[['@tcov_id', 'geometry']] + df_link.columns = ['hwycov-id:1', 'geometry'] + df_linkFull = pd.read_csv(os.path.join(export_path, 'hwy_tcad.csv')) + result = pd.merge(df_linkFull, df_link, how='left', on=['hwycov-id:1']) + result.to_csv(os.path.join(export_path, 'hwyTcad.csv'), index=False) + os.remove(os.path.join(export_path, 'hwy_tcad.csv')) + ##mode_list = ['Y','b','c','e','l','p','r','y','a','x','w']## + df_transit_link = df[df.modes.str.contains('|'.join(mode_list))] + df_transit_link = df_transit_link[['@tcov_id', 'geometry']] + df_transit_link.columns = ['trcovID', 'geometry'] + df_transit_link = df_transit_link[df_transit_link['trcovID'] != 0] + df_transit_link['AB'] = df_transit_link['trcovID'].apply(lambda x: 1 if x > 0 else 0) + df_transit_link['trcovID'] = abs(df_transit_link['trcovID']) + df_transit_link = df_transit_link[['trcovID', 'AB', 'geometry']] + df_transit_link.to_csv(os.path.join(export_path, 'transitLink.csv'), index=False) + network_table.close() + try: + previous_active_database.open() + data_explorer.remove_database(desktop_traffic_database) + except: + pass + + def get_partial_network(self, scenario, attributes): + domains = attributes.keys() + network = scenario.get_partial_network(domains, include_attributes=False) + for domain, attrs in attributes.iteritems(): + if attrs: + values = scenario.get_attribute_values(domain, attrs) + network.set_attribute_values(domain, attrs, values) + return network + + def output_transit_flow(self, label, lines, segment_flow, fout_seg): + # output segment data (transit_flow) + centroid = "0" # always 0 + voc = "" # volume/capacity, not actually used, + for line in lines: + line_id = id_format(line["@route_id"]) + ivtt = from_mp = to_mp = 0 + segments = iter(line.segments(include_hidden=True)) + seg = segments.next() + from_stop = id_format(seg["@stop_id"]) + for next_seg in segments: + to_mp += seg.link.length + ivtt += seg.transit_time - next_seg.dwell_time + transit_flow = seg[segment_flow] + seg = next_seg + if not next_seg.allow_alightings: + continue + to_stop = id_format(next_seg["@stop_id"]) + formatted_ivtt = format(ivtt) + fout_seg.write(",".join([ + label, line_id, from_stop, to_stop, centroid, format(from_mp), format(to_mp), + format(transit_flow), formatted_ivtt, formatted_ivtt, voc])) + fout_seg.write("\n") + from_stop = to_stop + from_mp = to_mp + ivtt = 0 + + def output_transit_aggregate_flow(self, label, links, + link_transit_flow, total_walk_flow, access_walk_flow, + xfer_walk_flow, egress_walk_flow, fout_link): + # output link data (transit_aggregate_flow) + for link in links: + link_id = id_format(link["@tcov_id"]) + ab_transit_flow = link[link_transit_flow] + ab_non_transit_flow = link[total_walk_flow] + ab_total_flow = ab_transit_flow + ab_non_transit_flow + ab_access_walk_flow = link[access_walk_flow] + ab_xfer_walk_flow = link[xfer_walk_flow] + ab_egress_walk_flow = link[egress_walk_flow] + if link.reverse_link: + ba_transit_flow = link.reverse_link[link_transit_flow] + ba_non_transit_flow = link.reverse_link[total_walk_flow] + ba_total_flow = ba_transit_flow + ba_non_transit_flow + ba_access_walk_flow = link.reverse_link[access_walk_flow] + ba_xfer_walk_flow = link.reverse_link[xfer_walk_flow] + ba_egress_walk_flow = link.reverse_link[egress_walk_flow] + else: + ba_transit_flow = 0.0 + ba_non_transit_flow = 0.0 + ba_total_flow = 0.0 + ba_access_walk_flow = 0.0 + ba_xfer_walk_flow = 0.0 + ba_egress_walk_flow = 0.0 + + fout_link.write(",".join( + [label, link_id, + format(ab_transit_flow), format(ba_transit_flow), + format(ab_non_transit_flow), format(ba_non_transit_flow), + format(ab_total_flow), format(ba_total_flow), + format(ab_access_walk_flow), format(ba_access_walk_flow), + format(ab_xfer_walk_flow), format(ba_xfer_walk_flow), + format(ab_egress_walk_flow), format(ba_egress_walk_flow)])) + fout_link.write("\n") + + def output_transit_onoff(self, label, lines, + total_boardings, total_alightings, initial_boardings, + xfer_boardings, xfer_alightings, final_alightings, + stop_on, stop_off, fout_stop): + # output stop data (transit_onoff) + for line in lines: + line_id = id_format(line["@route_id"]) + for seg in line.segments(True): + if not (seg.allow_alightings or seg.allow_boardings): + continue + i_node = seg.i_node.id + boardings = seg[total_boardings] + alightings = seg[total_alightings] + walk_access_on = seg[initial_boardings] + direct_xfer_on = seg[xfer_boardings] + walk_xfer_on = 0.0 + direct_xfer_off = seg[xfer_alightings] + walk_xfer_off = 0.0 + if stop_on.has_key(i_node): + if stop_on[i_node].has_key(line.id): + if direct_xfer_on > 0: + walk_xfer_on = direct_xfer_on - stop_on[i_node][line.id] + direct_xfer_on = stop_on[i_node][line.id] + if stop_off.has_key(i_node): + if stop_off[i_node].has_key(line.id): + if direct_xfer_off > 0: + walk_xfer_off = direct_xfer_off - stop_off[i_node][line.id] + direct_xfer_off = stop_off[i_node][line.id] + + egress_off = seg[final_alightings] + fout_stop.write(",".join([ + label, line_id, id_format(seg["@stop_id"]), + format(boardings), format(alightings), format(walk_access_on), + format(direct_xfer_on), format(walk_xfer_on), format(direct_xfer_off), + format(walk_xfer_off), format(egress_off)])) + fout_stop.write("\n") + + def collapse_network_adjustments(self, network, segment_results, link_results): + segment_alights = [v for k, v in segment_results.items() if "alightings" in k] + segment_boards = [v for k, v in segment_results.items() if "boardings" in k] + ["transit_boardings"] + segment_result_attrs = segment_alights + segment_boards + link_result_attrs = link_results.values() + ["aux_transit_volume"] + link_attrs = network.attributes("LINK") + link_modified_attrs = [ + "length", "@trtime_link_ea", "@trtime_link_am", "@trtime_link_md", + "@trtime_link_pm", "@trtime_link_ev", link_results["link_transit_flow"]] + seg_attrs = network.attributes("TRANSIT_SEGMENT") + line_attrs = network.attributes("TRANSIT_LINE") + + transit_modes = set([network.mode(m) for m in "blryepc"]) + aux_modes = set([network.mode(m) for m in "wxa"]) + xfer_mode = network.mode('x') + + def copy_seg_attrs(src_seg, dst_seg): + for attr in segment_result_attrs: + dst_seg[attr] += src_seg[attr] + dst_seg["allow_alightings"] |= src_seg["allow_alightings"] + dst_seg["allow_boardings"] |= src_seg["allow_boardings"] + + def get_xfer_link(node, timed_xfer_link, is_outgoing=True): + links = node.outgoing_links() if is_outgoing else node.incoming_links() + for link in links: + if xfer_mode in link.modes and link.length == timed_xfer_link.length: + return link + return None + + lines_to_update = set([]) + nodes_to_merge = [] + nodes_to_delete = [] + + for node in network.regular_nodes(): + if node["@network_adj"] == 1: + nodes_to_merge.append(node) + # copy boarding / alighting attributes for the segments to the original segment / stop + for seg in node.incoming_segments(): + lines_to_update.add(seg.line) + copy_seg_attrs(seg, seg.line.segment(seg.number+2)) + for seg in node.outgoing_segments(): + lines_to_update.add(seg.line) + copy_seg_attrs(seg, seg.line.segment(seg.number+1)) + elif node["@network_adj"] == 2: + nodes_to_delete.append(node) + # copy boarding / alighting attributes for the segments to the original segment / stop + for seg in node.outgoing_segments(True): + lines_to_update.add(seg.line) + if seg.j_node: + copy_seg_attrs(seg, seg.line.segment(seg.number+1)) + else: + copy_seg_attrs(seg, seg.line.segment(seg.number-1)) + elif node["@network_adj"] == 3: + orig_node = network.node(node["@network_adj_src"]) + # Remove transfer walk links and copy data to source walk link + for link in node.outgoing_links(): + if xfer_mode in link.modes and link.j_node["@network_adj"] == 3: + orig_xfer_link = get_xfer_link(orig_node, link) + for attr in link_result_attrs: + orig_xfer_link[attr] += link[attr] + network.delete_link(link.i_node, link.j_node) + # Sum link and segment results and merge links + mapping = network.merge_links_mapping(node) + for (link1, link2), attr_map in mapping['links'].iteritems(): + for attr in link_modified_attrs: + attr_map[attr] = max(link1[attr], link2[attr]) + + for (seg1, seg2), attr_map in mapping['transit_segments'].iteritems(): + if seg2.allow_alightings: + for attr in seg_attrs: + attr_map[attr] = seg1[attr] + else: # if it is a boarding stop or non-stop + for attr in seg_attrs: + attr_map[attr] = max(seg1[attr], seg2[attr]) + attr_map["transit_time_func"] = min(seg1["transit_time_func"], seg2["transit_time_func"]) + for attr in segment_boards: + attr_map[attr] = seg1[attr] + seg2[attr] + next_seg = seg2.line.segment(seg2.number+1) + for attr in segment_alights: + next_seg[attr] += seg2[attr] + network.merge_links(node, mapping) + + # Backup transit lines with altered routes and remove from network + lines = [] + for line in lines_to_update: + seg_data = {} + itinerary = [] + for seg in line.segments(include_hidden=True): + if seg.i_node["@network_adj"] in [1,2] or (seg.j_node and seg.j_node["@network_adj"] == 1): + continue + # for circle line transfers, j_node is now None for new "hidden" segment + j_node = seg.j_node + if (seg.j_node and seg.j_node["@network_adj"] == 2): + j_node = None + seg_data[(seg.i_node, j_node, seg.loop_index)] = dict((k, seg[k]) for k in seg_attrs) + itinerary.append(seg.i_node.number) + + lines.append({ + "id": line.id, + "vehicle": line.vehicle, + "itinerary": itinerary, + "attributes": dict((k, line[k]) for k in line_attrs), + "seg_attributes": seg_data}) + network.delete_transit_line(line) + # Remove duplicate network elements (undo network adjustments) + for node in nodes_to_delete: + for link in _chain(node.incoming_links(), node.outgoing_links()): + network.delete_link(link.i_node, link.j_node) + network.delete_node(node) + for node in nodes_to_merge: + mapping = network.merge_links_mapping(node) + for (link1, link2), attr_map in mapping["links"].iteritems(): + if link2.j_node.is_centroid: + link1, link2 = link2, link1 + for attr in link_attrs: + attr_map[attr] = link1[attr] + network.merge_links(node, mapping) + # Re-create transit lines on new itineraries + for line_data in lines: + new_line = network.create_transit_line( + line_data["id"], line_data["vehicle"], line_data["itinerary"]) + for k, v in line_data["attributes"].iteritems(): + new_line[k] = v + seg_data = line_data["seg_attributes"] + for seg in new_line.segments(include_hidden=True): + data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) + for k, v in data.iteritems(): + seg[k] = v + + def calc_additional_results(self, scenario, class_name, num_processors, + total_walk_flow, segment_results, link_transit_flow, + access_walk_flow, xfer_walk_flow, egress_walk_flow): + network_results = _m.Modeller().tool( + "inro.emme.transit_assignment.extended.network_results") + path_based_analysis = _m.Modeller().tool( + "inro.emme.transit_assignment.extended.path_based_analysis") + net_calculator = _m.Modeller().tool( + "inro.emme.network_calculation.network_calculator") + + spec = { + "on_links": { + "aux_transit_volumes": total_walk_flow.id + }, + "on_segments": segment_results, + "aggregated_from_segments": None, + "analyzed_demand": None, + "constraint": None, + "type": "EXTENDED_TRANSIT_NETWORK_RESULTS" + } + network_results(specification=spec, scenario=scenario, + class_name=class_name, num_processors=num_processors) + cal_spec = { + "result": "%s" % link_transit_flow.id, + "expression": "%s" % segment_results["transit_volumes"], + "aggregation": "+", + "selections": { + "link": "all", + "transit_line": "all" + }, + "type": "NETWORK_CALCULATION" + } + net_calculator(cal_spec, scenario=scenario) + + walk_flows = [("INITIAL_BOARDING_TO_FINAL_ALIGHTING", access_walk_flow.id), + ("INITIAL_BOARDING_TO_FINAL_ALIGHTING", xfer_walk_flow.id), + ("FINAL_ALIGHTING_TO_DESTINATION", egress_walk_flow.id)] + for portion_of_path, aux_transit_volumes in walk_flows: + spec = { + "portion_of_path": portion_of_path, + "trip_components": { + "in_vehicle": None, + "aux_transit": "length", + "initial_boarding": None, + "transfer_boarding": None, + "transfer_alighting": None, + "final_alighting": None + }, + "path_operator": ".max.", + "path_selection_threshold": { + "lower": -1.0, + "upper": 999999.0 + }, + "path_to_od_aggregation": None, + "constraint": None, + "analyzed_demand": None, + "results_from_retained_paths": { + "paths_to_retain": "SELECTED", + "aux_transit_volumes": aux_transit_volumes + }, + "path_to_od_statistics": None, + "path_details": None, + "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" + } + path_based_analysis( + specification=spec, scenario=scenario, + class_name=class_name, num_processors=num_processors) + + def transfer_analysis(self, scenario, net, class_name, num_processors): + create_attribute = _m.Modeller().tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + transfers_at_stops = _m.Modeller().tool( + "inro.emme.transit_assignment.extended.apps.transfers_at_stops") + + # find stop with/without walk transfer option + stop_walk_list = [] # stop (id) with walk option + stop_flag = "@stop_flag" + create_attribute("NODE", att, "1=stop without walk option, 2=otherwise", + 0, overwrite=True, scenario=scenario) + stop_nline = "@stop_nline" + create_attribute("NODE", stop_nline, "number of lines on the stop", + 0, overwrite=True, scenario=scenario) + + for line in net.transit_lines(): + for seg in line.segments(True): + node = seg.i_node + if seg.allow_alightings or seg.allow_boardings: + node[stop_nline] += 1 + if node[stop_flag] > 0 : #node checked + continue + if seg.allow_alightings or seg.allow_boardings: + node[stop_flag] = 1 + for ilink in node.incoming_links(): + # skip connector + if ilink.i_node.is_centroid: + continue + for m in ilink.modes: + if m.type=="AUX_TRANSIT": + node[stop_flag] = 2 + stop_walk_list.append(node.id) + break + if node[stop_flag]>=2: + break + if node[stop_flag]>=2: + continue + for olink in node.outgoing_links(): + # skip connector + if olink.j_node.is_centroid: + continue + for m in olink.modes: + if m.type=="AUX_TRANSIT": + node[stop_flag] = 2 + stop_walk_list.append(node.id) + break + if node[stop_flag]>=2: + break + #scenario.publish_network(net) + stop_off = {} + stop_on = {} + for stop in stop_walk_list: + stop_off[stop] = {} + stop_on[stop] = {} + selection = "i=%s" % stop + results = transfers_at_stops( + selection, scenario=scenario, + class_name=class_name, num_processors=num_processors) + for off_line in results: + stop_off[stop][off_line] = 0.0 + for on_line in results[off_line]: + trip = float(results[off_line][on_line]) + stop_off[stop][off_line] += trip + if not stop_on[stop].has_key(on_line): + stop_on[stop][on_line] = 0.0 + stop_on[stop][on_line] += trip + return stop_off, stop_on + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py b/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py new file mode 100644 index 0000000..fee626d --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py @@ -0,0 +1,158 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export/export_for_commercial_vehicle.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports the required skims in CSV format for the commercial vehicle model. +# +# +# Inputs: +# source: +# +# Files referenced: +# +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + source_dir = os.path.join(main_directory, "input") + title = "Base 2012 scenario" + tool = modeller.tool("sandag.export.export_for_commercial_vehicle") +""" + + +TOOLBOX_ORDER = 51 + + +import inro.modeller as _m +import numpy as _np +import subprocess as _subprocess +import tempfile as _tempfile +import traceback as _traceback +import os + +_join = os.path.join +_dir = os.path.dirname + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ExportForCommercialVehicleModel(_m.Tool(), gen_utils.Snapshot): + + output_directory = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = _dir(_m.Modeller().desktop.project.path) + self.output_directory = _join(_dir(project_dir), "output") + self.attributes = ["output_directory"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export for commercial vehicle model" + pb.description = """ + Exports the required skims in CSV format for the commercial vehicle model. + """ + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('output_directory', 'directory', + title='Select output directory') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.output_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Export skims for commercial vehicle model', save_arguments=True) + def __call__(self, output_directory, scenario): + emmebank = scenario.emmebank + modes = ['ldn', 'ldt', 'lhdn', 'lhdt', 'mhdn', 'mhdt', 'hhdn', 'hhdt'] + classes = ['SOV_NT_H', 'SOV_TR_H', 'TRK_L', 'TRK_L', 'TRK_M', 'TRK_M', 'TRK_H', 'TRK_H'] + # Mappings between COMMVEH modes and Emme classes + mode_class = dict(zip(modes, classes)) + class_mode = dict(zip(classes, modes)) + + is_toll_mode = lambda m: m.endswith('t') + #periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + period = "MD" + skims = ['TIME', 'DIST', 'TOLLCOST'] + DUCoef = [ + [-0.313, -0.138, -0.01], + [-0.313, -0.492, -0.01], + [-0.302, -0.580, -0.02] + ] + # Mappings for DUCoef utility index + modes_util = { + 'ldn': 0, + 'ldt': 0, + 'lhdn': 1, + 'lhdt': 1, + 'mhdn': 1, + 'mhdt': 1, + 'hhdn': 2, + 'hhdt': 2 + } + + # Lookup relevant skims as numpy arrays + skim_mat = {} + for cls in classes: + for skim in skims: + name = '%s_%s_%s' % (period, cls, skim) + if name not in skim_mat: + skim_mat[name] = emmebank.matrix(name).get_numpy_data(scenario) + + output_matrices = { + 'impldt_MD_Time.txt': skim_mat['MD_SOV_TR_H_TIME'], + 'impldt_MD_Dist.txt': skim_mat['MD_SOV_TR_H_DIST'], + } + + # Calculate DU matrices in numpy + for mode in modes: + time = skim_mat['%s_%s_TIME' % (period, mode_class[mode])] + distance = skim_mat['%s_%s_DIST' % (period, mode_class[mode])] + # All classes now have a tollcost skim available + toll_cost = skim_mat['%s_%s_TOLLCOST' % (period, mode_class[mode])] + _np.fill_diagonal(toll_cost, 0) + + coeffs = DUCoef[modes_util[mode]] + disutil_mat = coeffs[0] * time + coeffs[1] * distance + coeffs[2] * toll_cost + output_matrices['imp%s_%s_DU.txt' % (mode, period)] = disutil_mat + + # Insert row number into first column of the array + # Note: assumes zone IDs are continuous + for key, array in output_matrices.iteritems(): + output_matrices[key] = _np.insert(array, 0, range(1, array.shape[0]+1), axis=1) + + # Output DU matrices to CSV + # Print first column as integer, subsequent columns as floats rounded to 6 decimals + fmt_spec = ['%i'] + ['%.6f'] * (disutil_mat.shape[0]) + # Save to separate files + for name, array in output_matrices.iteritems(): + _np.savetxt(_join(output_directory, name), array, fmt=fmt_spec, delimiter=',') diff --git a/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py b/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py new file mode 100644 index 0000000..f5736a3 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py @@ -0,0 +1,374 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2019-2020. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export_for_transponder.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# + + +TOOLBOX_ORDER = 57 + + +import inro.modeller as _m + +import numpy as _np +import pandas as _pd +import string as _string +import traceback as _traceback +import math +import os +_dir, _join = os.path.dirname, os.path.join + +from shapely.geometry import MultiLineString, Point, LineString +from contextlib import contextmanager as _context +from itertools import izip as _izip + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module('sandag.utilities.demand') + + +class ExportForTransponder(_m.Tool(), gen_utils.Snapshot): + + scenario = _m.Attribute(_m.InstanceType) + output_directory = _m.Attribute(unicode) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + def __init__(self): + project_dir = _dir(_m.Modeller().desktop.project.path) + modeller = _m.Modeller() + if modeller.emmebank.path == _join(project_dir, "Database", "emmebank"): + self.scenario = modeller.emmebank.scenario(102) + self.num_processors = "max-1" + self.output_directory = _join(_dir(project_dir), "output") + self.attributes = ["scenario", "output_directory", "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export for transponder ownership model" + pb.description = """ +

Calculates and exports the following results for each origin zone:

+
    +
  • "MLDIST" - Managed lane distance - straight-line distance to the + nearest managed lane facility +
  • "AVGTTS" - Average travel time savings - average travel time savings + across all possible destinations. +
  • "PCTDETOUR" - Percent detour - The percent difference between the AM + transponder travel time and the AM non-transponder travel time + to sample zones when the general purpose lanes parallel to all toll + lanes using transponders are not available. +
+ .""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_scenario("scenario", + title="Representative scenario") + pb.add_select_file('output_directory', 'directory', + title='Select output directory') + + dem_utils.add_select_processors("num_processors", pb, self) + return pb.render() + + + def run(self): + self.tool_run_msg = "" + try: + self(self.output_directory, self.num_processors, self.scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export results for transponder ownership model", save_arguments=True) + def __call__(self, output_directory, num_processors, scenario): + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties( + _join(_dir(output_directory), "conf", "sandag_abm.properties")) + input_directory = _join(_dir(output_directory), "input") + num_processors = dem_utils.parse_num_processors(num_processors) + network = scenario.get_network() + distances = self.ml_facility_dist(network) + savings = self.avg_travel_time_savings(scenario, input_directory, props, num_processors) + detour = self.percent_detour(scenario, network, props, num_processors) + self.export_results(output_directory, scenario, distances, savings, detour) + + @_m.logbook_trace("Calculate distance to nearest managed lane facility") + def ml_facility_dist(self, network): + # DIST: Straight line distance to the nearest ML facility (nearest link with a ML Cost) + # managed lane is : + # HOV2+ only (carpool lane): "IFC" = 1 and "IHOV" = 2 and "ITOLLO" = 0 and "ITOLLA" = 0 and "ITOLLP" = 0 + # HOV3+ only (carpool lane): "IFC" = 1 and "IHOV" = 3 and "ITOLLO" = 0 and "ITOLLA" = 0 and "ITOLLP" = 0 + # HOV2+ & HOT (managed lane. HOV 2+ free. SOV pay toll): ): "IFC" = 1 and "IHOV" = 2 and "ITOLLO" > 0 and "ITOLLA" > 0 and "ITOLLP" > 0 + # HOV2+ & HOT (managed lane. HOV 3+ free. HOV2 & SOV pay toll): ): "IFC" = 1 and "IHOV" = 3 and "ITOLLO" > 0 and "ITOLLA" > 0 and "ITOLLP" > 0 + # Tollway (all vehicles tolled): "IFC" = 1 and "IHOV" = 4 + #$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ + # NOTE: NOT ALL MANAGED LANE LINKS HAVE A TOLL COST, + # SOME COSTS ARE JUST SPECIFIED ON THE ENTRANCE / EXIT LINKS + #$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ + + ml_link_coords = [] + ml_links = [] + for link in network.links(): + if link["type"] == 1 and link["@lane_restriction"] in (2,3) and ( + link["@toll_am"] + link["@toll_md"] + link["@toll_pm"]) > 0: + ml_link_coords.append(LineString(link.shape)) + ml_links.append(link) + ml_link_collection = MultiLineString(ml_link_coords) + distances = [] + for zone in network.centroids(): + zone_point = Point((zone.x, zone.y)) + distances.append(zone_point.distance(ml_link_collection) / 5280) + + # distances is a Python list of the distance from each zone to nearest ML link + # in same order as centroids + return distances + + @_m.logbook_trace("Calculate average travel time savings") + def avg_travel_time_savings(self, scenario, input_directory, props, num_processors): + # AVGTTS: The average travel savings of all households in each zone over all possible + # work destinations d. + # This average was calculated using an expected value with probabilities taken + # from a simplified destination + # choice model. The expected travel time savings of households in a zone z is + # SUM[d](NTTime[zd] - TRTime[zd]) * Employment[d] * exp(-0.01*NTTime[zd]) / + # SUM[d]Employmentd * exp(-0.01*NTTime[zd]) + # + # NTTime[zd] = AM_NTTime[zd] + PM_NTTime[dz] + # TRTime[zd] = AM_TRTime[zd] + PM_TRTime[dz] + + emmebank = scenario.emmebank + year = int(props['scenarioYear']) + mgra = _pd.read_csv( + _join(input_directory, 'mgra13_based_input%s.csv' % year)) + taz = mgra[['taz', 'emp_total']].groupby('taz').sum() + taz.reset_index(inplace=True) + taz = dem_utils.add_missing_zones(taz, scenario) + taz.reset_index(inplace=True) + + with setup_for_tt_savings_calc(emmebank): + employment_matrix = emmebank.matrix("mdemployment") + employment_matrix.set_numpy_data(taz["emp_total"].values, scenario.id) + matrix_calc = dem_utils.MatrixCalculator(scenario, num_processors) + matrix_calc.add("NTTime", "AM_SOV_NT_M_TIME + PM_SOV_NT_M_TIME'") + matrix_calc.add("TRTime", "AM_SOV_TR_M_TIME + PM_SOV_TR_M_TIME'") + matrix_calc.add("numerator", "((NTTime - TRTime).max.0) * employment * exp(-0.01 * NTTime)", + aggregation={"destinations": "+"}) + matrix_calc.add("denominator", "employment * exp(-0.01 * NTTime)", + aggregation={"destinations": "+"}) + matrix_calc.add("AVGTTS", "numerator / denominator") + matrix_calc.run() + avg_tts = emmebank.matrix("AVGTTS").get_numpy_data(scenario.id) + return avg_tts + + @_m.logbook_trace("Calculate percent detour without managed lane facilities") + def percent_detour(self, scenario, network, props, num_processors): + # PCTDETOUR: The percent difference between the AM non-toll travel time + # to a sample downtown zone and the AM non-toll travel time to downtown + # when the general purpose lanes parallel to all toll lanes requiring + # transponders are not available. This variable + # is calculated as + # 100*(TimeWithoutFacility - NonTransponderTime) / NonTransponderTime + + destinations = props["transponder.destinations"] + + network.create_attribute("NODE", "@root") + network.create_attribute("NODE", "@leaf") + + mode_id = get_available_mode_id(network) + new_mode = network.create_mode("AUX_AUTO", mode_id) + sov_non_toll_mode = network.mode("s") + + # Find special managed links and potential parallel GP facilities + ml_link_coords = [] + freeway_links = [] + for link in network.links(): + if link["@lane_restriction"] in [2, 3] and link["type"] == 1 and ( + link["@toll_am"] + link["@toll_md"] + link["@toll_pm"]) > 0: + ml_link_coords.append(LineString(link.shape)) + if sov_non_toll_mode in link.modes: + link.modes |= set([new_mode]) + if link["type"] == 1: + freeway_links.append(link) + + # remove mode from nearby GP links to special managed lanes + ml_link_collection = MultiLineString(ml_link_coords) + for link in freeway_links: + link_shape = LineString(link.shape) + distance = link_shape.distance(ml_link_collection) + if distance < 100: + for ml_shape in ml_link_collection: + if ml_shape.distance(link_shape) and close_bearing(link_shape, ml_shape): + link.modes -= set([new_mode]) + break + + for node in network.centroids(): + node["@root"] = 1 + for dst in destinations: + network.node(dst)["@leaf"] = 1 + + reverse_auto_network(network, "@auto_time") + detour_impedances = shortest_paths_impedances( + network, new_mode, "@auto_time", destinations) + direct_impedances = shortest_paths_impedances( + network, sov_non_toll_mode, "@auto_time", destinations) + + percent_detour = (detour_impedances - direct_impedances) / direct_impedances + avg_percent_detour = _np.sum(percent_detour, axis=1) / len(destinations) + avg_percent_detour = _np.nan_to_num(avg_percent_detour) + return avg_percent_detour + + @_m.logbook_trace("Export results to transponderModelAccessibilities.csv file") + def export_results(self, output_directory, scenario, distances, savings, detour): + zones = scenario.zone_numbers + output_file = _join(output_directory, "transponderModelAccessibilities.csv") + with open(output_file, 'w') as f: + f.write("TAZ,DIST,AVGTTS,PCTDETOUR\n") + for row in _izip(zones, distances, savings, detour): + f.write("%d, %.4f, %.5f, %.5f\n" % row) + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg + + +def reverse_auto_network(network, link_cost): + # swap directionality of modes and specified link costs, as well as turn prohibitions + # delete all transit lines + for line in network.transit_lines(): + network.delete_transit_line(line) + + # backup modes so that turns can be swapped (auto mode remains avialable) + network.create_attribute("LINK", "backup_modes") + for link in network.links(): + link.backup_modes = link.modes + # add new reverse links (where needed) and get the one-way links to be deleted + auto_mode = network.mode("d") + links_to_delete = [] + for link in network.links(): + reverse_link = network.link(link.j_node.id, link.i_node.id) + if reverse_link is None: + reverse_link = network.create_link(link.j_node.id, link.i_node.id, link.modes) + reverse_link.backup_modes = reverse_link.modes + links_to_delete.append(link) + reverse_link.modes |= link.modes + + # reverse the turn data + visited = set([]) + for turn in network.turns(): + if turn in visited: + continue + reverse_turn = network.turn(turn.k_node, turn.j_node, turn.i_node) + time, reverse_time = turn["data1"], turn["data1"] + turn["data1"], turn["data1"] = time, reverse_time + tpf, reverse_tpf = turn.penalty_func, reverse_turn.penalty_func + reverse_turn.penalty_func, turn.penalty_func = tpf, reverse_tpf + visited.add(turn) + visited.add(reverse_turn) + + # reverse the link data + visited = set([]) + for link in network.links(): + if link in visited: + continue + reverse_link = network.link(link.j_node.id, link.i_node.id) + time, reverse_time = link[link_cost], reverse_link[link_cost] + reverse_link[link_cost], link[link_cost] = time, reverse_time + reverse_link.modes, link.modes = link.backup_modes, reverse_link.backup_modes + visited.add(link) + visited.add(reverse_link) + + # delete the one-way links + for link in links_to_delete: + network.delete_link(link.i_node, link.j_node) + + +def shortest_paths_impedances(network, mode, link_cost, destinations): + excluded_links = [] + for link in network.links(): + if mode not in link.modes: + excluded_links.append(link) + + impedances = [] + for dest_id in destinations: + tree = network.shortest_path_tree( + dest_id, link_cost, excluded_links=excluded_links, consider_turns=True) + costs = [] + for node in network.centroids(): + if node.number == dest_id: + costs.append(0) + else: + try: + path_cost = tree.cost_to_node(node.id) + except KeyError: + path_cost = 600 + costs.append(path_cost) + impedances.append(costs) + return _np.array(impedances).T + + +@_context +def setup_for_tt_savings_calc(emmebank): + with gen_utils.temp_matrices(emmebank, "FULL", 2) as mf: + mf[0].name = "NTTime" + mf[0].description = "Temp AM + PM' Auto non-transponder time" + mf[1].name = "TRTime" + mf[1].description = "Temp AM + PM' Auto transponder time" + with gen_utils.temp_matrices(emmebank, "ORIGIN", 3) as mo: + mo[0].name = "numerator" + mo[1].name = "denominator" + mo[2].name = "AVGTTS" + mo[2].description = "Temp average travel time savings" + with gen_utils.temp_matrices(emmebank, "DESTINATION", 1) as md: + md[0].name = "employment" + md[0].description = "Temp employment per zone" + yield + +@_context +def get_temp_scenario(src_scenario): + delete_scenario = _m.Modeller().tool( + "inro.emme.data.scenario.delete_scenario") + emmebank = src_scenario.emmebank + scenario_id = get_available_scenario_id(emmebank) + temp_scenario = emmebank.copy_scenario(src_scenario, scenario_id) + try: + yield temp_scenario + finally: + delete_scenario(temp_scenario) + +def get_available_mode_id(network): + for mode_id in _string.letters: + if network.mode(mode_id) is None: + return mode_id + +def get_available_scenario_id(emmebank): + for i in range(1,10000): + if not emmebank.scenario(i): + return i + +def bearing(shape): + pt1 = shape.coords[0] + pt2 = shape.coords[-1] + x_diff = pt2[0] - pt1[0] + y_diff = pt2[1] - pt1[1] + return math.degrees(math.atan2(y_diff, x_diff)) + +def close_bearing(shape1, shape2, tol=25): + b1 = bearing(shape1) + b2 = bearing(shape2) + diff = (b1 - b2) % 360 + if diff >= 180: + diff -= 360 + return abs(diff) < tol diff --git a/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py b/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py new file mode 100644 index 0000000..068481a --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py @@ -0,0 +1,122 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export/export_tap_adjacent_lines.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports a list of transit lines adjacent to each TAP. +# +# +# Inputs: +# file_path: export path for the tap adjacency file +# scenario: scenario ID for the base scenario (same used in the Import network tool) +# +# Files created: +# output/tapLines.csv (or as specified) +# +# +# Script example: +""" +import inro.modeller as _m +import os +modeller = _m.Modeller() +desktop = modeller.desktop + +export_tap_adjacent_lines = modeller.tool("sandag.export.export_tap_adjacent_lines") + +project_dir = os.path.dirname(desktop.project_path()) +main_directory = os.path.dirname(project_dir) +output_dir = os.path.join(main_directory, "output") + +main_emmebank = os.path.join(project_dir, "Database", "emmebank") +scenario_id = 100 +base_scenario = main_emmebank.scenario(scenario_id) + +export_tap_adjacent_lines(os.path.join(output_dir, "tapLines.csv"), base_scenario) + +""" + + +TOOLBOX_ORDER = 75 + + +import inro.modeller as _m +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ExportLines(_m.Tool(), gen_utils.Snapshot): + + file_path = _m.Attribute(unicode) + + tool_run_msg = "" + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_dir = os.path.dirname(project_dir) + self.file_path = os.path.join(main_dir, "output", "tapLines.csv") + self.attributes = ["file_path"] + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export TAP adjacent lines" + pb.description = """Exports a list of the transit lines adjacent to each tap.""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('file_path', 'save_file',title='Select file path') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.file_path, scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export list of TAP adjacent lines", save_arguments=True) + def __call__(self, file_path, scenario): + attributes = {"file_path": file_path} + gen_utils.log_snapshot("Export list of TAP adjacent lines", str(self), attributes) + + network = scenario.get_partial_network( + ["NODE", "TRANSIT_LINE"], include_attributes=False) + values = scenario.get_attribute_values("NODE", ["@tap_id"]) + network.set_attribute_values("NODE", ["@tap_id"], values) + with open(file_path, 'w') as f: + f.write("TAP,LINES\n") + for node in network.nodes(): + if node["@tap_id"] == 0: + continue + lines = set([]) + for link in node.outgoing_links(): + for seg in link.j_node.outgoing_segments(include_hidden=True): + if seg.allow_alightings: + lines.add(seg.line) + if not lines: + continue + f.write("%d," % node["@tap_id"]) + f.write(" ".join([l.id for l in lines])) + f.write("\n") diff --git a/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py b/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py new file mode 100644 index 0000000..247eb60 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py @@ -0,0 +1,95 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export/export_traffic_skims.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports the traffic skims for use in the disaggregate demand models (CT-RAMP) +# and the data loader process. +# +# Note the matrix name mapping from the OMX file names to the Emme database names. +# +# Inputs: +# omx_file: output directory to read the OMX files from +# period: the period for which to export the skim matrices, "EA", "AM", "MD", "PM", "EV" +# scenario: base traffic scenario to use for reference zone system +# +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + output_dir = os.path.join(main_directory, "output") + scenario = modeller.scenario + periods = ["EA", "AM", "MD", "PM", "EV"] + export_traffic_skims = modeller.tool("sandag.import.export_traffic_skims") + for period in periods: + omx_file_path = os.path.join(output_dir, "traffic_skims_%s.omx" % period + export_traffic_skims(output_dir, period, scenario) +""" + +TOOLBOX_ORDER = 71 + + +import inro.modeller as _m +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ExportSkims(_m.Tool(), gen_utils.Snapshot): + + omx_file = _m.Attribute(unicode) + period = _m.Attribute(str) + tool_run_msg = "" + + def __init__(self): + self.attributes = ["omx_file", "period"] + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export traffic skims" + pb.description = """Export the skim matrices to OMX format for the selected period.""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_select_file('omx_file', 'save_file', title='Select OMX file') + options = [(x, x) for x in ["EA", "AM", "MD", "PM", "EV"]] + pb.add_select("period", keyvalues=options, title="Select corresponding period") + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.period, self.omx_file, scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export traffic skims to OMX", save_arguments=True) + def __call__(self, period, omx_file, scenario): + attributes = {"omx_file": omx_file, "period": period} + gen_utils.log_snapshot("Export traffic skims to OMX", str(self), attributes) + init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") + matrices = init_matrices.get_matrix_names("traffic_skims", [period], scenario) + with gen_utils.ExportOMX(omx_file, scenario, omx_key="NAME") as exporter: + exporter.write_matrices(matrices) diff --git a/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py b/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py new file mode 100644 index 0000000..7c22691 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py @@ -0,0 +1,108 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// export/export_transit_skims.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Exports the transit skims for use in the disaggregate demand models (CT-RAMP) +# and the data loader process. +# +# Note the matrix name mapping from the OMX file names to the Emme database names. +# +# Inputs: +# omx_file: output directory to read the OMX files from +# periods: list of periods, using the standard two-character abbreviation +# big_to_zero: replace big values (>10E6) with zero +# This is used in the final iteration skim (after the demand models are +# complete) to filter large values from the OMX files which are not +# compatible with the data loader process +# scenario: transit scenario to use for reference zone system +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + output_dir = os.path.join(main_directory, "output") + scenario = modeller.scenario + export_transit_skims = modeller.tool("sandag.import.export_transit_skims") + omx_file_path = os.path.join(output_dir, "transit_skims.omx" + export_transit_skims(output_dir, period, scenario) +""" + + +TOOLBOX_ORDER = 72 + + +import inro.modeller as _m +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ExportSkims(_m.Tool(), gen_utils.Snapshot): + omx_file = _m.Attribute(unicode) + periods = _m.Attribute(unicode) + big_to_zero = _m.Attribute(bool) + + tool_run_msg = "" + + def __init__(self): + self.attributes = ["omx_file", "periods", "big_to_zero"] + self.periods = "EA, AM, MD, PM, EV" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Export transit skim matrices" + pb.description = """Export the skim matrices to OMX format for all periods.""" + pb.branding_text = "- SANDAG - Export" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_select_file('omx_file', 'save_file', title='Select OMX file') + pb.add_text_box('periods', title="Selected periods:") + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + periods = [x.strip() for x in self.periods.split(",")] + self(self.omx_file, periods, scenario, self.big_to_zero) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Export transit skims to OMX", save_arguments=True) + def __call__(self, omx_file, periods, scenario, big_to_zero=False): + attributes = {"omx_file": omx_file, "periods": periods, "big_to_zero": big_to_zero} + gen_utils.log_snapshot("Export transit skims to OMX", str(self), attributes) + init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") + matrices = init_matrices.get_matrix_names( + "transit_skims", periods, scenario) + with gen_utils.ExportOMX(omx_file, scenario, omx_key="NAME") as exporter: + if big_to_zero: + emmebank = scenario.emmebank + for name in matrices: + matrix = emmebank.matrix(name) + array = matrix.get_numpy_data(scenario) + array[array>10E6] = 0 + exporter.write_array(array, exporter.generate_key(matrix)) + else: + exporter.write_matrices(matrices) diff --git a/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py new file mode 100644 index 0000000..375a27c --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py @@ -0,0 +1,516 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/import_auto_demand.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Imports the auto demand matrices generated from an iteration of the disaggregate +# demand models (CT-RAMP) and adds the saved disaggregated demand matrices to +# generate the total auto demand in preparation for the auto assignment. +# +# Note the matrix name mapping from the OMX file names to the Emme database names. +# +# Inputs: +# external_zones: set of external zone IDs as a range "1-12" +# output_dir: output directory to read the OMX files from +# num_processors: number of processors to use in the matrix calculations +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: pp is time period, one of EA, AM, MD, PM, EV, vot is one of low, med, high +# output/autoInternalExternalTrips_pp_vot.omx +# output/autoVisitorTrips_pp_vot.omx +# output/autoCrossBorderTrips_pp_vot.omx +# output/autoAirportTrips.SAN_pp_vot.omx +# output/autoAirportTrips.CDX_pp_vot.omx (if they exist) +# output/autoTrips_pp_vot.omx +# output/othrTrips_pp.omx (added to high vot) +# output/TripMatrices.csv +# output/EmptyAVTrips.omx (added to high vot) +# output/TNCVehicleTrips_pp.omx (added to high vot) +# +# Matrix inputs: +# pp_SOVGP_EIWORK, pp_SOVGP_EINONWORK, pp_SOVTOLL_EIWORK, pp_SOVTOLL_EINONWORK, +# pp_HOV2HOV_EIWORK, pp_HOV2HOV_EINONWORK, pp_HOV2TOLL_EIWORK, pp_HOV2TOLL_EINONWORK, +# pp_HOV3HOV_EIWORK, pp_HOV3HOV_EINONWORK, pp_HOV3TOLL_EIWORK, pp_HOV3TOLL_EINONWORK +# pp_SOV_EETRIPS, pp_HOV2_EETRIPS, pp_HOV3_EETRIPS +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV, v is one of L, M, H +# pp_SOV_TR_v, pp_SOV_NT_v, pp_HOV2_v, pp_HOV3_v, pp_HOV3_v +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + output_dir = os.path.join(main_directory, "output") + external_zones = "1-12" + num_processors = "MAX-1" + base_scenario = modeller.scenario + import_auto_demand = modeller.tool("sandag.import.import_auto_demand") + import_auto_demand(external_zones, output_dir, num_processors, base_scenario) +""" + +TOOLBOX_ORDER = 13 + + +import inro.modeller as _m +import traceback as _traceback +import pandas as _pandas +import os +import numpy +from contextlib import contextmanager as _context + +_join = os.path.join + +dem_utils = _m.Modeller().module('sandag.utilities.demand') +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ImportMatrices(_m.Tool(), gen_utils.Snapshot): + + external_zones = _m.Attribute(str) + output_dir = _m.Attribute(unicode) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.external_zones = "1-12" + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_dir = os.path.dirname(project_dir) + self.main_dir = main_dir + self.output_dir = os.path.join(main_dir, "output") + self.num_processors = "MAX-1" + self.attributes = ["external_zones", "output_dir", "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Import auto demand and sum matrices" + pb.description = """ +
+ Imports the trip matrices generated by CT-RAMP in OMX format, + the commercial vehicle demand in CSV format, + and adds the demand from the aggregate models for the final + trip assignments.
+ A total of 90 OMX files are expected, for 5 time periods + EA, AM, MD, PM and EV, and value-of-time level low, med or high, + with internal matrices by SOV, HOV2, HOV3+ and toll access type: +
    +
  • autoInternalExternalTrips_pp_vot.omx
  • +
  • autoVisitorTrips_pp_vot.omx
  • +
  • autoCrossBorderTrips_pp_vot.omx
  • +
  • autoAirportTrips.SAN_pp_vot.omx
  • +
  • autoAirportTrips.CDX_pp_vot.omx (optional)
  • +
  • autoTrips_pp_vot.omx
  • +
  • othrTrips_pp.omx (added to high vot)
  • +
  • EmptyAVTrips.omx (added to high vot)
  • +
  • TNCVehicleTrips_pp.omx (added to high vot)
  • +
+ As well as one CSV file "TripMatrices.csv" for the commercial vehicle trips. + Adds the aggregate demand from the + external-external and external-internal demand matrices: +
    +
  • pp_SOVGP_EETRIPS, pp_HOV2HOV_EETRIPS, pp_HOV3HOV_EETRIPS
  • +
  • pp_SOVGP_EIWORK, pp_SOVGP_EINONWORK, pp_SOVTOLL_EIWORK, pp_SOVTOLL_EINONWORK
  • +
  • pp_HOV2HOV_EIWORK, pp_HOV2HOV_EINONWORK, pp_HOV2TOLL_EIWORK, pp_HOV2TOLL_EINONWORK
  • +
  • pp_HOV3HOV_EIWORK, pp_HOV3HOV_EINONWORK, pp_HOV3TOLL_EIWORK, pp_HOV3TOLL_EINONWORK
  • +
+ to the time-of-day total demand matrices. +
+
+ """ + pb.branding_text = "- SANDAG - Model" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_select_file('output_dir', 'directory', + title='Select output directory') + pb.add_text_box("external_zones", title="External zones:") + dem_utils.add_select_processors("num_processors", pb, self) + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.output_dir, self.external_zones, self.num_processors, scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Create TOD auto trip tables", save_arguments=True) + def __call__(self, output_dir, external_zones, num_processors, scenario): + attributes = { + "output_dir": output_dir, + "external_zones": external_zones, + "num_processors": num_processors} + gen_utils.log_snapshot("Create TOD auto trip tables", str(self), attributes) + + #get parameters from sandag_abm.properties + modeller = _m.Modeller() + load_properties = modeller.tool('sandag.utilities.properties') + props = load_properties(_join(self.main_dir, "conf", "sandag_abm.properties")) + + self.scenario = scenario + self.output_dir = output_dir + self.external_zones = external_zones + self.num_processors = num_processors + self.import_traffic_trips(props) + self.import_commercial_vehicle_demand(props) + #self.convert_light_trucks_to_pce() + self.add_aggregate_demand() + + @_context + def setup(self): + emmebank = self.scenario.emmebank + self._matrix_cache = {} + with gen_utils.OMXManager(self.output_dir, "%sTrips%s%s.omx") as omx_manager: + try: + yield omx_manager + finally: + for name, value in self._matrix_cache.iteritems(): + matrix = emmebank.matrix(name) + matrix.set_numpy_data(value, self.scenario.id) + + def set_data(self, name, value): + if name in self._matrix_cache: + value = value + self._matrix_cache[name] + self._matrix_cache[name] = value + + @_m.logbook_trace("Import CT-RAMP traffic trips from OMX") + def import_traffic_trips(self, props): + title = "Import CT-RAMP traffic trips from OMX report" + report = _m.PageBuilder(title) + + taxi_da_share = props["Taxi.da.share"] + taxi_s2_share = props["Taxi.s2.share"] + taxi_s3_share = props["Taxi.s3.share"] + taxi_pce = props["Taxi.passengersPerVehicle"] + tnc_single_da_share = props["TNC.single.da.share"] + tnc_single_s2_share = props["TNC.single.s2.share"] + tnc_single_s3_share = props["TNC.single.s3.share"] + tnc_single_pce = props["TNC.single.passengersPerVehicle"] + tnc_shared_da_share = props["TNC.shared.da.share"] + tnc_shared_s2_share = props["TNC.shared.s2.share"] + tnc_shared_s3_share = props["TNC.shared.s3.share"] + tnc_shared_pce = props["TNC.shared.passengersPerVehicle"] + av_share = props["Mobility.AV.Share"] + + periods = ["_EA", "_AM", "_MD", "_PM", "_EV"] + vot_bins = ["_low", "_med", "_high"] + mode_shares = [ + ("mf%s_SOV_TR_H", { + "TAXI": taxi_da_share / taxi_pce + }), + ("mf%s_HOV2_H", { + "TAXI": taxi_s2_share / taxi_pce + }), + ("mf%s_HOV3_H", { + "TAXI": taxi_s3_share / taxi_pce + }), + ] + + with self.setup() as omx_manager: + # SOV transponder "TRPDR" = "TR" and non-transponder "NOTRPDR" = "NT" + for period in periods: + for vot in vot_bins: + # SOV non-transponder demand + matrix_name = "mf%s_SOV_NT_%s" % (period[1:], vot[1].upper()) + logbook_label = "Import auto from OMX SOVNOTRPDR to matrix %s" % (matrix_name) + resident_demand = omx_manager.lookup(("auto", period, vot), "SOVNOTRPDR%s" % period) + visitor_demand = omx_manager.lookup(("autoVisitor", period, vot), "SOV%s" % period) + cross_border_demand = omx_manager.lookup(("autoCrossBorder", period, vot), "SOV%s" % period) + # NOTE: No non-transponder airport or internal-external demand + total_ct_ramp_trips = ( + resident_demand + visitor_demand + cross_border_demand) + dem_utils.demand_report([ + ("resident", resident_demand), + ("cross_border", cross_border_demand), + ("visitor", visitor_demand), + ("total", total_ct_ramp_trips) + ], + logbook_label, self.scenario, report) + self.set_data(matrix_name, total_ct_ramp_trips) + + # SOV transponder demand + matrix_name = "mf%s_SOV_TR_%s" % (period[1:], vot[1].upper()) + logbook_label = "Import auto from OMX SOVTRPDR to matrix %s" % (matrix_name) + resident_demand = omx_manager.lookup(("auto", period, vot), "SOVTRPDR%s" % period) + # NOTE: No transponder visitor or cross-border demand + airport_demand = omx_manager.lookup(("autoAirport", ".SAN" + period, vot), "SOV%s" % period) + if omx_manager.file_exists(("autoAirport", ".CBX" + period, vot)): + airport_demand += omx_manager.lookup(("autoAirport", ".CBX" + period, vot), "SOV%s" % period) + internal_external_demand = omx_manager.lookup(("autoInternalExternal", period, vot), "SOV%s" % period) + + total_ct_ramp_trips = ( + resident_demand + airport_demand + internal_external_demand) + dem_utils.demand_report([ + ("resident", resident_demand), + ("airport", airport_demand), + ("internal_external", internal_external_demand), + ("total", total_ct_ramp_trips) + ], + logbook_label, self.scenario, report) + self.set_data(matrix_name, total_ct_ramp_trips) + + # HOV2 and HOV3 demand + matrix_name_map = [ + ("mf%s_HOV2_%s", "SR2%s"), + ("mf%s_HOV3_%s", "SR3%s") + ] + for matrix_name_tmplt, omx_name in matrix_name_map: + matrix_name = matrix_name_tmplt % (period[1:], vot[1].upper()) + logbook_label = "Import auto from OMX %s to matrix %s" % (omx_name[:3], matrix_name) + resident_demand = ( + omx_manager.lookup(("auto", period, vot), omx_name % ("TRPDR" + period)) + + omx_manager.lookup(("auto", period, vot), omx_name % ("NOTRPDR" + period))) + visitor_demand = omx_manager.lookup(("autoVisitor", period, vot), omx_name % period) + cross_border_demand = omx_manager.lookup(("autoCrossBorder", period, vot), omx_name % period) + airport_demand = omx_manager.lookup(("autoAirport", ".SAN" + period, vot), omx_name % period) + if omx_manager.file_exists(("autoAirport", ".CBX" + period, vot)): + airport_demand += omx_manager.lookup(("autoAirport", ".CBX" + period, vot), omx_name % period) + internal_external_demand = omx_manager.lookup(("autoInternalExternal", period, vot), omx_name % period) + + total_ct_ramp_trips = ( + resident_demand + visitor_demand + cross_border_demand + airport_demand + internal_external_demand) + dem_utils.demand_report([ + ("resident", resident_demand), + ("cross_border", cross_border_demand), + ("visitor", visitor_demand), + ("airport", airport_demand), + ("internal_external", internal_external_demand), + ("total", total_ct_ramp_trips) + ], + logbook_label, self.scenario, report) + self.set_data(matrix_name, total_ct_ramp_trips) + + # add TNC and TAXI demand to vot="high" + for matrix_name_tmplt, share in mode_shares: + matrix_name = matrix_name_tmplt % period[1:] + logbook_label = "Import othr from TAXI, empty AV, and TNC to matrix %s" % (matrix_name) + resident_taxi_demand = ( + omx_manager.lookup(("othr", period, ""), "TAXI" + period) * share["TAXI"]) + visitor_taxi_demand = ( + omx_manager.lookup(("othrVisitor", period, ""), "TAXI" + period) * share["TAXI"]) + cross_border_taxi_demand = ( + omx_manager.lookup(("othrCrossBorder", period, ""), "TAXI" + period) * share["TAXI"]) + # airport SAN + airport_taxi_demand = ( + omx_manager.lookup(("othrAirport", ".SAN", period), "TAXI" + period) * share["TAXI"]) + # airport CBX (optional) + if omx_manager.file_exists(("othrAirport", ".CBX", period)): + airport_taxi_demand += ( + omx_manager.lookup(("othrAirport",".CBX", period), "TAXI" + period) * share["TAXI"]) + internal_external_taxi_demand = ( + omx_manager.lookup(("othrInternalExternal", period, ""), "TAXI" + period) * share["TAXI"]) + + #AV routing models and TNC fleet model demand + empty_av_demand = omx_manager.lookup(("EmptyAV","",""), "EmptyAV%s" % period) + tnc_demand_0 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_0" % period) + tnc_demand_1 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_1" % period) + tnc_demand_2 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_2" % period) + tnc_demand_3 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_3" % period) + + #AVs: no driver. No AVs: driver + #AVs: 0 and 1 passenger would be SOV. there will be empty vehicles as well. No AVs: 0 passanger would be SOV + #AVs: 2 passenger would be HOV2. No AVs: 1 passenger would be HOV2 + #AVs: 3 passenger would be HOV3. No AVs: 2 and 3 passengers would be HOV3 + if (av_share>0): + if (matrix_name_tmplt[5:-2] == "SOV_TR"): + av_demand = empty_av_demand + tnc_demand_0 + tnc_demand_1 + elif (matrix_name_tmplt[5:-2] == "HOV2"): + av_demand = tnc_demand_2 + else: + av_demand = tnc_demand_3 + else: + if (matrix_name_tmplt[5:-2] == "SOV_TR"): + av_demand = tnc_demand_0 + elif (matrix_name_tmplt[5:-2] == "HOV2"): + av_demand = tnc_demand_1 + else: + av_demand = tnc_demand_2 + tnc_demand_3 + + total_ct_ramp_trips = ( + resident_taxi_demand + visitor_taxi_demand + cross_border_taxi_demand + + airport_taxi_demand + internal_external_taxi_demand + av_demand) + dem_utils.demand_report([ + ("resident_taxi", resident_taxi_demand), + ("visitor_taxi", visitor_taxi_demand), + ("cross_border_taxi", cross_border_taxi_demand), + ("airport_taxi", airport_taxi_demand), + ("internal_external_taxi", internal_external_taxi_demand), + ("av_fleet", av_demand), + ("total", total_ct_ramp_trips) + ], + logbook_label, self.scenario, report) + self.set_data(matrix_name, total_ct_ramp_trips) + _m.logbook_write(title, report.render()) + + @_m.logbook_trace('Import commercial vehicle demand') + def import_commercial_vehicle_demand(self, props): + scale_factor = props["cvm.scale_factor"] + scale_light = props["cvm.scale_light"] + scale_medium = props["cvm.scale_medium"] + scale_heavy = props["cvm.scale_heavy"] + share_light = props["cvm.share.light"] + share_medium = props["cvm.share.medium"] + share_heavy = props["cvm.share.heavy"] + + scenario = self.scenario + emmebank = scenario.emmebank + + mapping = {} + periods = ["EA", "AM", "MD", "PM", "EV"] + # The SOV demand is modified in-place, which was imported + # prior from the CT-RAMP demand + # The truck demand in vehicles is copied from separate matrices + for index, period in enumerate(periods): + mapping["CVM_%s:LNT" % period] = { + "orig": "%s_SOV_TR_H" % period, + "dest": "%s_SOV_TR_H" % period, + "pce": 1.0, + "scale": scale_light[index], + "share": share_light, + "period": period + } + mapping["CVM_%s:INT" % period] = { + "orig": "%s_TRK_L_VEH" % period, + "dest": "%s_TRK_L" % period, + "pce": 1.3, + "scale": scale_medium[index], + "share": share_medium, + "period": period + } + mapping["CVM_%s:MNT" % period] = { + "orig": "%s_TRK_M_VEH" % period, + "dest": "%s_TRK_M" % period, + "pce": 1.5, + "scale": scale_medium[index], + "share": share_medium, + "period": period + } + mapping["CVM_%s:HNT" % period] = { + "orig": "%s_TRK_H_VEH" % period, + "dest": "%s_TRK_H" % period, + "pce": 2.5, + "scale": scale_heavy[index], + "share": share_heavy, + "period": period + } + with _m.logbook_trace('Load starting SOV and truck matrices'): + for key, value in mapping.iteritems(): + value["array"] = emmebank.matrix(value["orig"]).get_numpy_data(scenario) + + with _m.logbook_trace('Processing CVM from TripMatrices.csv'): + path = os.path.join(self.output_dir, "TripMatrices.csv") + table = _pandas.read_csv(path) + for key, value in mapping.iteritems(): + cvm_array = table[key].values.reshape((4996, 4996)) # reshape method deprecated since v 0.19.0, yma, 2/12/2019 + #factor in cvm demand by the scale factor used in trip generation + cvm_array = cvm_array/scale_factor + #scale trips to take care of underestimation + cvm_array = cvm_array * value["scale"] + + #add remaining share to the correspnding truck matrix + value["array"] = value["array"] + (cvm_array * (1-value["share"])) + + #add cvm truck vehicles to light-heavy trucks + for key, value in mapping.iteritems(): + period = value["period"] + cvm_vehs = ['L','M','H'] + if key == "CVM_%s:INT" % period: + for veh in cvm_vehs: + key_new = "CVM_%s:%sNT" % (period, veh) + value_new = mapping[key_new] + if value_new["share"] != 0.0: + cvm_array = table[key_new].values.reshape((4996, 4996)) + cvm_array = cvm_array/scale_factor + cvm_array = cvm_array * value_new["scale"] + value["array"] = value["array"] + (cvm_array * value_new["share"]) + matrix_unique = {} + with _m.logbook_trace('Save SOV matrix and convert CV and truck vehicle demand to PCEs for assignment'): + for key, value in mapping.iteritems(): + matrix = emmebank.matrix(value["dest"]) + array = value["array"] * value["pce"] + if (matrix in matrix_unique.keys()): + array = array + emmebank.matrix(value["dest"]).get_numpy_data(scenario) + matrix.set_numpy_data(array, scenario) + matrix_unique[matrix] = 1 + + @_m.logbook_trace('Convert light truck vehicle demand to PCEs for assignment') + def convert_light_trucks_to_pce(self): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + # Calculate PCEs for trucks + periods = ["EA", "AM", "MD", "PM", "EV"] + mat_trucks = ['TRK_L'] + pce_values = [1.3] + for period in periods: + with matrix_calc.trace_run("Period %s" % period): + for name, pce in zip(mat_trucks, pce_values): + demand_name = 'mf%s_%s' % (period, name) + matrix_calc.add(demand_name, '(%s_VEH * %s).max.0' % (demand_name, pce)) + + @_m.logbook_trace('Add aggregate demand') + def add_aggregate_demand(self): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + periods = ["EA", "AM", "MD", "PM", "EV"] + vots = ["L", "M", "H"] + # External-internal trips DO have transponder + # all SOV trips go to SOVTP + with matrix_calc.trace_run("Add external-internal trips to auto demand"): + modes = ["SOVGP", "SOVTOLL", "HOV2HOV", "HOV2TOLL", "HOV3HOV", "HOV3TOLL"] + modes_assign = {"SOVGP": "SOV_TR", + "SOVTOLL": "SOV_TR", + "HOV2HOV": "HOV2", + "HOV2TOLL": "HOV2", + "HOV3HOV": "HOV3", + "HOV3TOLL": "HOV3"} + for period in periods: + for mode in modes: + for vot in vots: + # Segment imported demand into 3 equal parts for VOT Low/Med/High + assign_mode = modes_assign[mode] + params = {'p': period, 'm': mode, 'v': vot, 'am': assign_mode} + matrix_calc.add("mf%s_%s_%s" % (period, assign_mode, vot), + "mf%(p)s_%(am)s_%(v)s " + "+ (1.0/3.0)*mf%(p)s_%(m)s_EIWORK " + "+ (1.0/3.0)*mf%(p)s_%(m)s_EINONWORK" % params) + + # External - external faster with single-processor as number of O-D pairs is so small (12 X 12) + # External-external trips do not have transpnder + # all SOV trips go to SOVNTP + matrix_calc.num_processors = 0 + with matrix_calc.trace_run("Add external-external trips to auto demand"): + modes = ["SOV", "HOV2", "HOV3"] + for period in periods: + for mode in modes: + for vot in vots: + # Segment imported demand into 3 equal parts for VOT Low/Med/High + params = {'p': period, 'm': mode, 'v': vot} + if (mode == "SOV"): + matrix_calc.add( + "mf%(p)s_%(m)s_NT_%(v)s" % params, + "mf%(p)s_%(m)s_NT_%(v)s + (1.0/3.0)*mf%(p)s_%(m)s_EETRIPS" % params, + {"origins": self.external_zones, "destinations": self.external_zones}) + else: + matrix_calc.add( + "mf%(p)s_%(m)s_%(v)s" % params, + "mf%(p)s_%(m)s_%(v)s + (1.0/3.0)*mf%(p)s_%(m)s_EETRIPS" % params, + {"origins": self.external_zones, "destinations": self.external_zones}) diff --git a/sandag_abm/src/main/emme/toolbox/import/import_network.py b/sandag_abm/src/main/emme/toolbox/import/import_network.py new file mode 100644 index 0000000..4da01bc --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/import_network.py @@ -0,0 +1,2250 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/import_network.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Imports the network from the input network files. +# +# +# Inputs: +# source: path to the location of the input network files +# traffic_scenario_id: optional scenario to store the imported network from the traffic files only +# transit_scenario_id: optional scenario to store the imported network from the transit files only +# merged_scenario_id: scenario to store the combined traffic and transit data from all network files +# title: the title to use for the imported scenario +# save_data_tables: if checked, create a data table for each reference file for viewing in the Emme Desktop +# data_table_name: prefix to use to identify all data tables +# overwrite: check to overwrite any existing data tables or scenarios with the same ID or name +# emmebank: the Emme database in which to create the scenario. Default is the current open database +# +# Files referenced: +# hwycov.e00: base nodes and links for traffic network with traffic attributes in ESRI input exchange format +# linktypeturns.dbf: fixed turn travel times by to/from link type (field IFC) pairs +# turns.csv: turn bans and fixed costs by link from/to ID (field HWYCOV-ID) +# trcov.e00: base nodes and links for transit network in ESRI input exchange format +# trrt.csv: transit routes and their attributes +# trlink.csv: itineraries for each route as sequence of link IDs (TRCOV-ID field) +# trstop.csv: transit stop attributes +# timexfer_period.csv: table of timed transfer pairs of lines, by period +# mode5tod.csv: global (per-mode) transit cost and perception attributes +# special_fares.txt: table listing special fares in terms of boarding and incremental in-vehicle costs. +# off_peak_toll_factors.csv (optional): factors to calculate the toll for EA, MD, and EV periods from the OP toll input for specified facilities +# vehicle_class_toll_factors.csv (optional): factors to adjust the toll cost by facility name and class (DA, S2, S3, TRK_L, TRK_M, TRK_H) +# +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + source_dir = os.path.join(main_directory, "input") + title = "Base 2012 scenario" + import_network = modeller.tool("sandag.import.import_network") + import_network(output_dir, merged_scenario_id=100, title=title, + data_table_name="2012_base", overwrite=True) +""" + + +TOOLBOX_ORDER = 11 + + +import inro.modeller as _m +import inro.emme.datatable as _dt +import inro.emme.network as _network +from inro.emme.core.exception import Error as _NetworkError + +from itertools import izip as _izip +from collections import defaultdict as _defaultdict, OrderedDict +from contextlib import contextmanager as _context +import fiona as _fiona + +from math import ceil as _ceiling +from copy import deepcopy as _copy +import numpy as _np +import heapq as _heapq + +import traceback as _traceback +import os + +import pandas as pd +from scipy.spatial import distance + +_join = os.path.join +_dir = os.path.dirname + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + +FILE_NAMES = { + "FARES": "special_fares.txt", + "OFF_PEAK": "off_peak_toll_factors.csv", + "VEHICLE_CLASS": "vehicle_class_toll_factors.csv" +} + + +class ImportNetwork(_m.Tool(), gen_utils.Snapshot): + + source = _m.Attribute(unicode) + traffic_scenario_id = _m.Attribute(int) + transit_scenario_id = _m.Attribute(int) + merged_scenario_id = _m.Attribute(int) + overwrite = _m.Attribute(bool) + title = _m.Attribute(unicode) + save_data_tables = _m.Attribute(bool) + data_table_name = _m.Attribute(unicode) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self._log = [] + self._error = [] + project_dir = _dir(_m.Modeller().desktop.project.path) + self.source = _join(_dir(project_dir), "input") + self.overwrite = False + self.title = "" + self.data_table_name = "" + self.attributes = [ + "source", "traffic_scenario_id", "transit_scenario_id", "merged_scenario_id", + "overwrite", "title", "save_data_tables", "data_table_name"] + + def page(self): + if not self.data_table_name: + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + self.data_table_name = props["scenarioYear"] + + pb = _m.ToolPageBuilder(self) + pb.title = "Import network" + pb.description = """ + Create an Emme network from the E00 and associated files + generated from TCOVED. + The timed transfer is stored in data tables with the suffix "_timed_xfers_period". +
+
+
+ The following files are used: +
    +
  • hwycov.e00
  • +
  • LINKTYPETURNS.DBF
  • +
  • turns.csv
  • +
  • trcov.e00
  • +
  • trrt.csv
  • +
  • trlink.csv
  • +
  • trstop.csv
  • +
  • timexfer_period.csv, where period = EA,AM,MD,PM,EV
  • +
  • MODE5TOD.csv
  • +
  • special_fares.txt
  • +
  • off_peak_toll_factors.csv (optional)
  • +
  • vehicle_class_toll_factors.csv (optional)
  • +
+
+ """ + pb.branding_text = "- SANDAG - Import" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file("source", window_type="directory", file_filter="", + title="Source directory:",) + + pb.add_text_box("traffic_scenario_id", size=6, title="Scenario ID for traffic (optional):") + pb.add_text_box("transit_scenario_id", size=6, title="Scenario ID for transit (optional):") + pb.add_text_box("merged_scenario_id", size=6, title="Scenario ID for merged network:") + pb.add_text_box("title", size=80, title="Scenario title:") + pb.add_checkbox("save_data_tables", title=" ", label="Save reference data tables of file data") + pb.add_text_box("data_table_name", size=80, title="Name for data tables:", + note="Prefix name to use for all saved data tables") + pb.add_checkbox("overwrite", title=" ", label="Overwrite existing scenarios and data tables") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self.emmebank = _m.Modeller().emmebank + with self.setup(): + self.execute() + run_msg = "Network import complete" + if self._error: + run_msg += " with %s non-fatal errors. See logbook for details" % len(self._error) + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc()) + raise + + def __call__(self, source, + traffic_scenario_id=None, transit_scenario_id=None, merged_scenario_id=None, + title="", save_data_tables=False, data_table_name="", overwrite=False, + emmebank=None): + + self.source = source + self.traffic_scenario_id = traffic_scenario_id + self.transit_scenario_id = transit_scenario_id + self.merged_scenario_id = merged_scenario_id + self.title = title + self.save_data_tables = save_data_tables + self.data_table_name = data_table_name + self.overwrite = overwrite + if not emmebank: + self.emmebank = _m.Modeller().emmebank + else: + self.emmebank = emmebank + + with self.setup(): + self.execute() + + return self.emmebank.scenario(merged_scenario_id) + + @_context + def setup(self): + self._log = [] + self._error = [] + fatal_error = False + attributes = OrderedDict([ + ("self", str(self)), + ("source", self.source), + ("traffic_scenario_id", self.traffic_scenario_id), + ("transit_scenario_id", self.transit_scenario_id), + ("merged_scenario_id", self.merged_scenario_id), + ("title", self.title), + ("save_data_tables", self.save_data_tables), + ("data_table_name", self.data_table_name), + ("overwrite", self.overwrite), + ]) + self._log = [{ + "content": attributes.items(), + "type": "table", "header": ["name", "value"], + "title": "Tool input values" + }] + with _m.logbook_trace("Import network", attributes=attributes) as trace: + gen_utils.log_snapshot("Import network", str(self), attributes) + try: + yield + except Exception as error: + self._log.append({"type": "text", "content": error}) + trace_text = _traceback.format_exc().replace("\n", "
") + self._log.append({"type": "text", "content": trace_text}) + self._error.append(error) + fatal_error = True + raise + finally: + self.log_report() + if self._error: + if fatal_error: + trace.write("Import network failed (%s errors)" % len(self._error), attributes=attributes) + else: + trace.write("Import network completed (%s non-fatal errors)" % len(self._error), attributes=attributes) + + def execute(self): + traffic_attr_map = { + "NODE": { + "interchange": ("@interchange", "DERIVED", "EXTRA", "is interchange node") + }, + "LINK": OrderedDict([ + ("HWYCOV-ID", ("@tcov_id", "TWO_WAY", "EXTRA", "SANDAG-assigned link ID")), + ("SPHERE", ("@sphere", "TWO_WAY", "EXTRA", "Jurisdiction sphere of influence")), + ("NM", ("#name", "TWO_WAY", "STRING", "Street name")), + ("FXNM", ("#name_from", "TWO_WAY", "STRING", "Cross street at the FROM end")), + ("TXNM", ("#name_to", "TWO_WAY", "STRING", "Cross street name at the TO end")), + ("DIR", ("@direction_cardinal", "TWO_WAY", "EXTRA", "Link direction")), + ("ASPD", ("@speed_adjusted", "TWO_WAY", "EXTRA", "Adjusted link speed (miles/hr)")), + ("IYR", ("@year_open_traffic", "TWO_WAY", "EXTRA", "The year the link opened to traffic")), + ("IPROJ", ("@project_code", "TWO_WAY", "EXTRA", "Project number for use with hwyproj.xls")), + ("IJUR", ("@jurisdiction_type", "TWO_WAY", "EXTRA", "Link jurisdiction type")), + ("IFC", ("type", "TWO_WAY", "STANDARD", "")), + ("IHOV", ("@lane_restriction", "TWO_WAY", "EXTRA", "Link operation type")), + ("ITRUCK", ("@truck_restriction", "TWO_WAY", "EXTRA", "Truck restriction code (ITRUCK)")), + ("ISPD", ("@speed_posted", "TWO_WAY", "EXTRA", "Posted speed limit (mph)")), + ("IMED", ("@median", "TWO_WAY", "EXTRA", "Median type")), + ("AU", ("@lane_auxiliary", "ONE_WAY", "EXTRA", "Number of auxiliary lanes")), + ("CNT", ("@traffic_control", "ONE_WAY", "EXTRA", "Intersection control type")), + ("TL", ("@turn_thru", "ONE_WAY", "EXTRA", "Intersection approach through lanes")), + ("RL", ("@turn_right", "ONE_WAY", "EXTRA", "Intersection approach right-turn lanes")), + ("LL", ("@turn_left", "ONE_WAY", "EXTRA", "Intersection approach left-turn lanes")), + ("GC", ("@green_to_cycle_init", "ONE_WAY", "EXTRA", "Initial green-to-cycle ratio")), + ("CHO", ("@capacity_hourly_op", "ONE_WAY", "EXTRA", "Off-Peak hourly mid-link capacity")), + ("CHA", ("@capacity_hourly_am", "ONE_WAY", "EXTRA", "AM Peak hourly mid-link capacity")), + ("CHP", ("@capacity_hourly_pm", "ONE_WAY", "EXTRA", "PM Peak hourly mid-link capacity")), + # These attributes are expanded from 3 time periods to 5 + ("ITOLLO", ("toll_op", "TWO_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("ITOLLA", ("toll_am", "TWO_WAY", "INTERNAL", "")), + ("ITOLLP", ("toll_pm", "TWO_WAY", "INTERNAL", "")), + ("LNO", ("lane_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("LNA", ("lane_am", "ONE_WAY", "INTERNAL", "")), + ("LNP", ("lane_pm", "ONE_WAY", "INTERNAL", "")), + ("CPO", ("capacity_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("CPA", ("capacity_link_am", "ONE_WAY", "INTERNAL", "")), + ("CPP", ("capacity_link_pm", "ONE_WAY", "INTERNAL", "")), + ("CXO", ("capacity_inter_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("CXA", ("capacity_inter_am", "ONE_WAY", "INTERNAL", "")), + ("CXP", ("capacity_inter_pm", "ONE_WAY", "INTERNAL", "")), + ("TMO", ("time_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("TMA", ("time_link_am", "ONE_WAY", "INTERNAL", "")), + ("TMP", ("time_link_pm", "ONE_WAY", "INTERNAL", "")), + ("TXO", ("time_inter_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("TXA", ("time_inter_am", "ONE_WAY", "INTERNAL", "")), + ("TXP", ("time_inter_pm", "ONE_WAY", "INTERNAL", "")), + # These three attributes are used to cross-reference the turn directions + ("TLB", ("through_link", "ONE_WAY", "INTERNAL", "")), + ("RLB", ("right_link", "ONE_WAY", "INTERNAL", "")), + ("LLB", ("left_link", "ONE_WAY", "INTERNAL", "")), + ("@cost_operating", ("@cost_operating","DERIVED", "EXTRA", "Fuel and maintenance cost")), + ("INTDIST_UP", ("@intdist_up", "DERIVED", "EXTRA", "Upstream major intersection distance")), + ("INTDIST_DOWN", ("@intdist_down", "DERIVED", "EXTRA", "Downstream major intersection distance")), + ]) + } + time_period_attrs = OrderedDict([ + ("@cost_auto", "toll + cost autos"), + ("@cost_hov2", "toll (non-mngd) + cost HOV2"), + ("@cost_hov3", "toll (non-mngd) + cost HOV3+"), + ("@cost_lgt_truck", "toll + cost light trucks"), + ("@cost_med_truck", "toll + cost medium trucks"), + ("@cost_hvy_truck", "toll + cost heavy trucks"), + ("@cycle", "cycle length (minutes)"), + ("@green_to_cycle", "green to cycle ratio"), + ("@capacity_link", "mid-link capacity"), + ("@capacity_inter", "approach capacity"), + ("@toll", "toll cost (cent)"), + ("@lane", "number of lanes"), + ("@time_link", "link time in minutes"), + ("@time_inter", "intersection delay time"), + ("@sta_reliability", "static reliability") + ]) + time_name = { + "_ea": "Early AM ", "_am": "AM Peak ", "_md": "Mid-day ", "_pm": "PM Peak ", "_ev": "Evening " + } + time_periods = ["_ea", "_am", "_md", "_pm", "_ev"] + for attr, desc_tmplt in time_period_attrs.iteritems(): + for time in time_periods: + traffic_attr_map["LINK"][attr + time] = \ + (attr + time, "DERIVED", "EXTRA", time_name[time] + desc_tmplt) + + transit_attr_map = { + "NODE": OrderedDict([ + ("@tap_id", ("@tap_id", "DERIVED", "EXTRA", "Transit-access point ID")), + ]), + "LINK": OrderedDict([ + ("TRCOV-ID", ("@tcov_id", "TWO_WAY", "EXTRA", "SANDAG-assigned link ID")), + ("NM", ("#name", "TWO_WAY", "STRING", "Street name")), + ("FXNM", ("#name_from", "TWO_WAY", "STRING", "Cross street at the FROM end")), + ("TXNM", ("#name_to", "TWO_WAY", "STRING", "Cross street name at the TO end")), + ("DIR", ("@direction_cardinal", "TWO_WAY", "EXTRA", "Link direction")), + ("OSPD", ("@speed_observed", "TWO_WAY", "EXTRA", "Observed speed")), + ("IYR", ("@year_open_traffic", "TWO_WAY", "EXTRA", "The year the link opened to traffic ")), + ("IFC", ("type", "TWO_WAY", "STANDARD", "")), + ("IHOV", ("@lane_restriction_tr", "TWO_WAY", "EXTRA", "Link operation type")), + ("ISPD", ("@speed_posted_tr_l", "TWO_WAY", "EXTRA", "Posted speed limit (mph)")), + ("IMED", ("@median", "TWO_WAY", "EXTRA", "Median type")), + ("TMO", ("trtime_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), + ("TMEA", ("@trtime_link_ea", "DERIVED", "EXTRA", "Early AM transit link time in minutes")), + ("TMA", ("@trtime_link_am", "ONE_WAY", "EXTRA", "AM Peak transit link time in minutes")), + ("TMMD", ("@trtime_link_md", "DERIVED", "EXTRA", "Mid-day transit link time in minutes")), + ("TMP", ("@trtime_link_pm", "ONE_WAY", "EXTRA", "PM Peak transit link time in minutes")), + ("TMEV", ("@trtime_link_ev", "DERIVED", "EXTRA", "Evening transit link time in minutes")), + ("MINMODE", ("@mode_hierarchy", "TWO_WAY", "EXTRA", "Transit mode type")), + ]), + "TRANSIT_LINE": OrderedDict([ + ("AM_Headway", ("@headway_am", "TRRT", "EXTRA", "AM Peak actual headway")), + ("PM_Headway", ("@headway_pm", "TRRT", "EXTRA", "PM Peak actual headway")), + ("OP_Headway", ("@headway_op", "TRRT", "EXTRA", "Off-Peak actual headway")), + ("Night_Headway", ("@headway_night", "TRRT", "EXTRA", "Night actual headway")), + ("AM_Headway_rev", ("@headway_rev_am", "DERIVED", "EXTRA", "AM Peak revised headway")), + ("PM_Headway_rev", ("@headway_rev_pm", "DERIVED", "EXTRA", "PM Peak revised headway")), + ("OP_Headway_rev", ("@headway_rev_op", "DERIVED", "EXTRA", "Off-Peak revised headway")), + ("WT_IVTPK", ("@vehicle_per_pk", "MODE5TOD", "EXTRA", "Peak in-vehicle perception factor")), + ("WT_IVTOP", ("@vehicle_per_op", "MODE5TOD", "EXTRA", "Off-Peak in-vehicle perception factor")), + ("WT_FAREPK", ("@fare_per_pk", "MODE5TOD", "EXTRA", "Peak fare perception factor")), + ("WT_FAREOP", ("@fare_per_op", "MODE5TOD", "EXTRA", "Off-Peak fare perception factor")), + ("DWELLTIME", ("default_dwell_time", "MODE5TOD", "INTERNAL", "")), + ("Fare", ("@fare", "TRRT", "EXTRA", "Boarding fare ($)")), + ("@transfer_penalty",("@transfer_penalty","DERIVED", "EXTRA", "Transfer penalty (min)")), + ("Route_ID", ("@route_id", "TRRT", "EXTRA", "Transit line internal ID")), + ("Night_Hours", ("@night_hours", "TRRT", "EXTRA", "Night hours")), + ("Config", ("@config", "TRRT", "EXTRA", "Config ID (same as route name)")), + ]), + "TRANSIT_SEGMENT": OrderedDict([ + ("Stop_ID", ("@stop_id", "TRSTOP", "EXTRA", "Stop ID from trcov")), + ("Pass_Count", ("@pass_count", "TRSTOP", "EXTRA", "Number of times this stop is passed")), + ("Milepost", ("@milepost", "TRSTOP", "EXTRA", "Distance from start of line")), + ("FareZone", ("@fare_zone", "TRSTOP", "EXTRA", "Fare zone ID")), + ("StopName", ("#stop_name", "TRSTOP", "STRING", "Name of stop")), + ("@coaster_fare_board", ("@coaster_fare_board", "DERIVED", "EXTRA", "Boarding fare for coaster")), + ("@coaster_fare_inveh", ("@coaster_fare_inveh", "DERIVED", "EXTRA", "Incremental fare for Coaster")), + ]) + } + + create_scenario = _m.Modeller().tool( + "inro.emme.data.scenario.create_scenario") + + file_names = [ + "hwycov.e00", "LINKTYPETURNS.DBF", "turns.csv", + "trcov.e00", "trrt.csv", "trlink.csv", "trstop.csv", + "timexfer_EA.csv", "timexfer_AM.csv","timexfer_MD.csv", + "timexfer_PM.csv","timexfer_EV.csv","MODE5TOD.csv", + ] + for name in file_names: + file_path = _join(self.source, name) + if not os.path.exists(file_path): + raise Exception("missing file '%s' in directory %s" % (name, self.source)) + + title = self.title + if not title: + existing_scenario = self.emmebank.scenario(self.merged_scenario_id) + if existing_scenario: + title = existing_scenario.title + + def create_attributes(scenario, attr_map): + for elem_type, mapping in attr_map.iteritems(): + for name, _tcoved_type, emme_type, desc in mapping.values(): + if emme_type == "EXTRA": + if not scenario.extra_attribute(name): + xatt = scenario.create_extra_attribute(elem_type, name) + xatt.description = desc + elif emme_type == "STRING": + if not scenario.network_field(elem_type, name): + scenario.create_network_field(elem_type, name, 'STRING', description=desc) + + if self.traffic_scenario_id: + traffic_scenario = create_scenario( + self.traffic_scenario_id, title + " Traffic", + overwrite=self.overwrite, emmebank=self.emmebank) + create_attributes(traffic_scenario, traffic_attr_map) + else: + traffic_scenario = None + if self.transit_scenario_id: + transit_scenario = create_scenario( + self.transit_scenario_id, title + " Transit", + overwrite=self.overwrite, emmebank=self.emmebank) + create_attributes(transit_scenario, transit_attr_map) + else: + transit_scenario = None + if self.merged_scenario_id: + scenario = create_scenario( + self.merged_scenario_id, title, + overwrite=self.overwrite, emmebank=self.emmebank) + create_attributes(scenario, traffic_attr_map) + create_attributes(scenario, transit_attr_map) + else: + scenario = traffic_scenario or transit_scenario + + traffic_network = _network.Network() + transit_network = _network.Network() + try: + if self.traffic_scenario_id or self.merged_scenario_id: + for elem_type, attrs in traffic_attr_map.iteritems(): + log_content = [] + for k, v in attrs.iteritems(): + if v[3] == "DERIVED": + k = "--" + log_content.append([k] + list(v)) + self._log.append({ + "content": log_content, + "type": "table", + "header": ["TCOVED", "Emme", "Source", "Type", "Description"], + "title": "Traffic %s attributes" % elem_type.lower().replace("_", " "), + "disclosure": True + }) + try: + self.create_traffic_base(traffic_network, traffic_attr_map) + self.create_turns(traffic_network) + self.calc_traffic_attributes(traffic_network) + self.check_zone_access(traffic_network, traffic_network.mode("d")) + finally: + if traffic_scenario: + traffic_scenario.publish_network(traffic_network, resolve_attributes=True) + + if self.transit_scenario_id or self.merged_scenario_id: + for elem_type, attrs in transit_attr_map.iteritems(): + log_content = [] + for k, v in attrs.iteritems(): + if v[3] == "DERIVED": + k = "--" + log_content.append([k] + list(v)) + self._log.append({ + "content": log_content, + "type": "table", + "header": ["TCOVED", "Emme", "Source", "Type", "Description"], + "title": "Transit %s attributes" % elem_type.lower().replace("_", " "), + "disclosure": True + }) + try: + self.create_transit_base(transit_network, transit_attr_map) + self.create_transit_lines(transit_network, transit_attr_map) + self.calc_transit_attributes(transit_network) + finally: + if transit_scenario: + for link in transit_network.links(): + if link.type <= 0: + link.type = 99 + transit_scenario.publish_network(transit_network, resolve_attributes=True) + if self.merged_scenario_id: + self.add_transit_to_traffic(traffic_network, transit_network) + self.adjust_network(traffic_network) + finally: + if self.merged_scenario_id: + scenario.publish_network(traffic_network, resolve_attributes=True) + + self.set_functions(scenario) + self.check_connectivity(scenario) + + def create_traffic_base(self, network, attr_map): + self._log.append({"type": "header", "content": "Import traffic base network from hwycov.e00"}) + hwy_data = gen_utils.DataTableProc("ARC", _join(self.source, "hwycov.e00")) + + if self.save_data_tables: + hwy_data.save("%s_hwycov" % self.data_table_name, self.overwrite) + + for elem_type in "NODE", "TURN": + mapping = attr_map.get(elem_type) + if not mapping: + continue + for field, (attr, tcoved_type, emme_type, desc) in mapping.iteritems(): + default = "" if emme_type == "STRING" else 0 + network.create_attribute(elem_type, attr, default) + + # Create Modes + dummy_auto = network.create_mode("AUTO", "d") + hov2 = network.create_mode("AUX_AUTO", "h") + hov2_toll = network.create_mode("AUX_AUTO", "H") + hov3 = network.create_mode("AUX_AUTO", "i") + hov3_toll = network.create_mode("AUX_AUTO", "I") + sov = network.create_mode("AUX_AUTO", "s") + sov_toll = network.create_mode("AUX_AUTO", "S") + heavy_trk = network.create_mode("AUX_AUTO", "v") + heavy_trk_toll = network.create_mode("AUX_AUTO", "V") + medium_trk = network.create_mode("AUX_AUTO", "m") + medium_trk_toll = network.create_mode("AUX_AUTO", "M") + light_trk = network.create_mode("AUX_AUTO", "t") + light_trk_toll = network.create_mode("AUX_AUTO", "T") + + dummy_auto.description = "dummy auto" + sov.description = "SOV" + hov2.description = "HOV2" + hov3.description = "HOV3+" + light_trk.description = "TRKL" + medium_trk.description = "TRKM" + heavy_trk.description = "TRKH" + + sov_toll.description = "SOV TOLL" + hov2_toll.description = "HOV2 TOLL" + hov3_toll.description = "HOV3+ TOLL" + light_trk_toll.description = "TRKL TOLL" + medium_trk_toll.description = "TRKM TOLL" + heavy_trk_toll.description = "TRKH TOLL" + + is_centroid = lambda arc, node : (arc["IFC"] == 10) and (node == "AN") + + # Note: only truck types 1, 3, 4, and 7 found in 2012 base network + modes_gp_lanes= { + 1: set([dummy_auto, sov, hov2, hov3, light_trk, medium_trk, heavy_trk, + sov_toll, hov2_toll, hov3_toll, light_trk_toll, medium_trk_toll, + heavy_trk_toll]), + 2: set([dummy_auto, sov, hov2, hov3, light_trk, medium_trk, + sov_toll, hov2_toll, hov3_toll, light_trk_toll, medium_trk_toll]), + 3: set([dummy_auto, sov, hov2, hov3, light_trk, sov_toll, hov2_toll, + hov3_toll, light_trk_toll]), + 4: set([dummy_auto, sov, hov2, hov3, sov_toll, hov2_toll, hov3_toll]), + 5: set([dummy_auto, heavy_trk, heavy_trk_toll]), + 6: set([dummy_auto, medium_trk, heavy_trk, medium_trk_toll, heavy_trk_toll]), + 7: set([dummy_auto, light_trk, medium_trk, heavy_trk, light_trk_toll, + medium_trk_toll, heavy_trk_toll]), + } + modes_toll_lanes = { + 1: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll, + medium_trk_toll, heavy_trk_toll]), + 2: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll, + medium_trk_toll]), + 3: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll]), + 4: set([dummy_auto, sov_toll, hov2_toll, hov3_toll]), + 5: set([dummy_auto, heavy_trk_toll]), + 6: set([dummy_auto, medium_trk_toll, heavy_trk_toll]), + 7: set([dummy_auto, light_trk_toll, medium_trk_toll, heavy_trk_toll]), + } + modes_HOV2 = set([dummy_auto, hov2, hov3, hov2_toll, hov3_toll]) + modes_HOV3 = set([dummy_auto, hov3, hov3_toll]) + + + def define_modes(arc): + if arc["IFC"] == 10: # connector + return modes_gp_lanes[1] + elif arc["IHOV"] == 1: + return modes_gp_lanes[arc["ITRUCK"]] + elif arc["IHOV"] == 2: + # managed lanes, free for HOV2 and HOV3+, tolls for SOV + if arc["ITOLLO"] + arc["ITOLLA"] + arc["ITOLLP"] > 0: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 + # special case of I-15 managed lanes base year and 2020, no build + elif arc["IFC"] == 1 and arc["IPROJ"] in [41, 42, 486, 373, 711]: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 + elif arc["IFC"] == 8 or arc["IFC"] == 9: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 + else: + return modes_HOV2 + elif arc["IHOV"] == 3: + # managed lanes, free for HOV3+, tolls for SOV and HOV2 + if arc["ITOLLO"] + arc["ITOLLA"] + arc["ITOLLP"] > 0: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 + # special case of I-15 managed lanes for base year and 2020, no build + elif arc["IFC"] == 1 and arc["IPROJ"] in [41, 42, 486, 373, 711]: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 + elif arc["IFC"] == 8 or arc["IFC"] == 9: + return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 + else: + return modes_HOV3 + elif arc["IHOV"] == 4: + return modes_toll_lanes[arc["ITRUCK"]] + else: + return modes_gp_lanes[arc["ITRUCK"]] + + self._create_base_net( + hwy_data, network, mode_callback=define_modes, centroid_callback=is_centroid, + arc_id_name="HWYCOV-ID", link_attr_map=attr_map["LINK"]) + self._log.append({"type": "text", "content": "Import traffic base network complete"}) + + def create_transit_base(self, network, attr_map): + self._log.append({"type": "header", "content": "Import transit base network from trcov.e00"}) + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + transit_data = gen_utils.DataTableProc("ARC", _join(self.source, "trcov.e00")) + + if self.save_data_tables: + transit_data.save("%s_trcov" % self.data_table_name, self.overwrite) + + # aux mode speed is always 3 (miles/hr) + access = network.create_mode("AUX_TRANSIT", "a") + transfer = network.create_mode("AUX_TRANSIT", "x") + walk = network.create_mode("AUX_TRANSIT", "w") + + bus = network.create_mode("TRANSIT", "b") + express_bus = network.create_mode("TRANSIT", "e") + ltdexp_bus = network.create_mode("TRANSIT", "p") + brt_red = network.create_mode("TRANSIT", "r") + brt_yellow = network.create_mode("TRANSIT", "y") + lrt = network.create_mode("TRANSIT", "l") + coaster_rail = network.create_mode("TRANSIT", "c") + tier1 = network.create_mode("TRANSIT", "o") + + access.description = "ACCESS" + transfer.description = "TRANSFER" + walk.description = "WALK" + bus.description = "BUS" # (vehicle type 100, PCE=3.0) + express_bus.description = "EXP BUS" # (vehicle type 90 , PCE=3.0) + ltdexp_bus.description = "LTDEXP BUS" # (vehicle type 80 , PCE=3.0) + lrt.description = "LRT" # (vehicle type 50) + brt_yellow.description = "BRT YEL" # (vehicle type 60 , PCE=3.0) + brt_red.description = "BRT RED" # (vehicle type 70 , PCE=3.0) + coaster_rail.description = "CMR" # (vehicle type 40) + tier1.description = "TIER1" # (vehicle type 45) + + access.speed = 3 + transfer.speed = 3 + walk.speed = 3 + + # define TAP connectors as centroids + is_centroid = lambda arc, node: (int(arc["MINMODE"]) == 3) and (node == "BN") + + mode_setting = { + 1: set([transfer]), # 1 = special transfer walk links between certain nearby stops + 2: set([walk]), # 2 = walk links in the downtown area + 3: set([access]), # 3 = the special TAP connectors + 4: set([coaster_rail]), # 4 = Coaster Rail Line + 5: set([lrt]), # 5 = Light Rail Transit (LRT) Line + 6: set([brt_yellow, ltdexp_bus, express_bus, bus]), # 6 = Yellow Car Bus Rapid Transit (BRT) + 7: set([brt_red, ltdexp_bus, express_bus, bus]), # 7 = Red Car Bus Rapid Transit (BRT) + 8: set([ltdexp_bus, express_bus, bus]), # 8 = Limited Express Bus + 9: set([ltdexp_bus, express_bus, bus]), # 9 = Express Bus + 10: set([ltdexp_bus, express_bus, bus]), # 10 = Local Bus + } + tier1_rail_link_name = props["transit.newMode"] + + def define_modes(arc): + if arc["NM"] == tier1_rail_link_name: + return set([tier1]) + return mode_setting[arc["MINMODE"]] + + arc_filter = lambda arc: (arc["MINMODE"] > 2) + + # first pass to create the main base network for vehicles, xfer links and TAPs + self._create_base_net( + transit_data, network, mode_callback=define_modes, centroid_callback=is_centroid, + arc_id_name="TRCOV-ID", link_attr_map=attr_map["LINK"], arc_filter=arc_filter) + + # second pass to add special walk links / modify modes on existing links + reverse_dir_map = {1:3, 3:1, 2:4, 4:2, 0:0} + + def set_reverse_link(link, modes): + reverse_link = link.reverse_link + if reverse_link: + reverse_link.modes |= modes + else: + reverse_link = network.create_link(link.j_node, link.i_node, modes) + for attr in network.attributes("LINK"): + reverse_link[attr] = link[attr] + reverse_link["@direction_cardinal"] = reverse_dir_map[link["@direction_cardinal"]] + reverse_link["@tcov_id"] = -1*link["@tcov_id"] + reverse_link.vertices = list(reversed(link.vertices)) + + def epsilon_compare(a, b, epsilon): + return abs((a - b) / (a if abs(a) > 1 else 1)) <= epsilon + + for arc in transit_data: + # possible improvement: snap walk nodes to nearby node if not matched and within distance + if arc_filter(arc): + continue + if float(arc["AN"]) == 0 or float(arc["BN"]) == 0: + self._log.append({"type": "text", + "content": "Node ID 0 in AN (%s) or BN (%s) for link ID %s." % + (arc["AN"], arc["BN"], arc["TRCOV-ID"])}) + continue + coordinates = arc["geo_coordinates"] + arc_length = arc["LENGTH"] / 5280.0 # convert feet to miles + i_node = get_node(network, arc['AN'], coordinates[0], False) + j_node = get_node(network, arc['BN'], coordinates[-1], False) + modes = define_modes(arc) + link = network.link(i_node, j_node) + split_link_case = False + if link: + link.modes |= modes + else: + # Note: additional cases of "tunnel" walk links could be + # considered to optimize network matching + # check if this a special "split" link case where + # we do not need to add a "tunnel" walk link + for link1 in i_node.outgoing_links(): + if split_link_case: + break + for link2 in link1.j_node.outgoing_links(): + if link2.j_node == j_node: + if epsilon_compare(link1.length + link2.length, arc_length, 10**-5): + self._log.append({"type": "text", + "content": "Walk link AN %s BN %s matched to two links TCOV-ID %s, %s" % + (arc['AN'], arc['BN'], link1["@tcov_id"], link2["@tcov_id"])}) + link1.modes |= modes + link2.modes |= modes + set_reverse_link(link1, modes) + set_reverse_link(link2, modes) + split_link_case = True + break + if not split_link_case: + link = network.create_link(i_node, j_node, modes) + link.length = arc_length + if len(coordinates) > 2: + link.vertices = coordinates[1:-1] + if not split_link_case: + set_reverse_link(link, modes) + self._log.append({"type": "text", "content": "Import transit base network complete"}) + + def _create_base_net(self, data, network, mode_callback, centroid_callback, arc_id_name, link_attr_map, arc_filter=None): + forward_attr_map = {} + reverse_attr_map = {} + for field, (name, tcoved_type, emme_type, desc) in link_attr_map.iteritems(): + if emme_type != "STANDARD": + default = "" if emme_type == "STRING" else 0 + network.create_attribute("LINK", name, default) + + if field in [arc_id_name, "DIR"]: + # these attributes are special cases for reverse link + forward_attr_map[field] = name + elif tcoved_type == "TWO_WAY": + forward_attr_map[field] = name + reverse_attr_map[field] = name + elif tcoved_type == "ONE_WAY": + forward_attr_map["AB" + field] = name + reverse_attr_map["BA" + field] = name + + emme_id_name = forward_attr_map[arc_id_name] + dir_name = forward_attr_map["DIR"] + reverse_dir_map = {1:3, 3:1, 2:4, 4:2, 0:0} + new_node_id = max(data.values("AN").max(), data.values("BN").max()) + 1 + if arc_filter is None: + arc_filter = lambda arc : True + + # Create nodes and links + for arc in data: + if not arc_filter(arc): + continue + if float(arc["AN"]) == 0 or float(arc["BN"]) == 0: + self._log.append({"type": "text", + "content": "Node ID 0 in AN (%s) or BN (%s) for link ID %s." % + (arc["AN"], arc["BN"], arc[arc_id_name])}) + continue + coordinates = arc["geo_coordinates"] + i_node = get_node(network, arc['AN'], coordinates[0], centroid_callback(arc, "AN")) + j_node = get_node(network, arc['BN'], coordinates[-1], centroid_callback(arc, "BN")) + existing_link = network.link(i_node, j_node) + if existing_link: + msg = "Duplicate link between AN %s and BN %s. Link IDs %s and %s." % \ + (arc["AN"], arc["BN"], existing_link[emme_id_name], arc[arc_id_name]) + self._log.append({"type": "text", "content": msg}) + self._error.append(msg) + self._split_link(network, i_node, j_node, new_node_id) + new_node_id += 1 + + modes = mode_callback(arc) + link = network.create_link(i_node, j_node, modes) + link.length = arc["LENGTH"] / 5280.0 # convert feet to miles + if len(coordinates) > 2: + link.vertices = coordinates[1:-1] + for field, attr in forward_attr_map.iteritems(): + link[attr] = arc[field] + if arc["IWAY"] == 2 or arc["IWAY"] == 0: + reverse_link = network.create_link(j_node, i_node, modes) + reverse_link.length = link.length + reverse_link.vertices = list(reversed(link.vertices)) + for field, attr in reverse_attr_map.iteritems(): + reverse_link[attr] = arc[field] + reverse_link[emme_id_name] = -1*arc[arc_id_name] + reverse_link[dir_name] = reverse_dir_map[arc["DIR"]] + + def create_transit_lines(self, network, attr_map): + self._log.append({"type": "header", "content": "Import transit lines"}) + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + fatal_errors = 0 + # Route_ID,Route_Name,Mode,AM_Headway,PM_Headway,OP_Headway,Night_Headway,Night_Hours,Config,Fare + transit_line_data = gen_utils.DataTableProc("trrt", _join(self.source, "trrt.csv")) + # Route_ID,Link_ID,Direction + transit_link_data = gen_utils.DataTableProc("trlink", _join(self.source, "trlink.csv")) + # Stop_ID,Route_ID,Link_ID,Pass_Count,Milepost,Longitude, Latitude,HwyNode,TrnNode,FareZone,StopName + transit_stop_data = gen_utils.DataTableProc("trstop", _join(self.source, "trstop.csv")) + # From_line,To_line,Board_stop,Wait_time + # Note: Board_stop is not used + # Timed xfer data + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + timed_xfer_data = {} + for period in periods: + timed_xfer_data[period] = gen_utils.DataTableProc("timexfer_"+period, _join(self.source, "timexfer_"+period+".csv")) + + mode_properties = gen_utils.DataTableProc("MODE5TOD", _join(self.source, "MODE5TOD.csv"), convert_numeric=True) + mode_details = {} + for record in mode_properties: + mode_details[int(record["MODE_ID"])] = record + + if self.save_data_tables: + transit_link_data.save("%s_trlink" % self.data_table_name, self.overwrite) + transit_line_data.save("%s_trrt" % self.data_table_name, self.overwrite) + transit_stop_data.save("%s_trstop" % self.data_table_name, self.overwrite) + mode_properties.save("%s_MODE5TOD" % self.data_table_name, self.overwrite) + + coaster = network.create_transit_vehicle(40, 'c') # 4 coaster + trolley = network.create_transit_vehicle(50, 'l') # 5 sprinter/trolley + brt_yellow = network.create_transit_vehicle(60, 'y') # 6 BRT yellow line (future line) + brt_red = network.create_transit_vehicle(70, 'r') # 7 BRT red line (future line) + premium_bus = network.create_transit_vehicle(80, 'p') # 8 prem express + express_bus = network.create_transit_vehicle(90, 'e') # 9 regular express + local_bus = network.create_transit_vehicle(100, 'b') # 10 local bus + tier1 = network.create_transit_vehicle(45, 'o') # 11 Tier 1 + + brt_yellow.auto_equivalent = 3.0 + brt_red.auto_equivalent = 3.0 + premium_bus.auto_equivalent = 3.0 + express_bus.auto_equivalent = 3.0 + local_bus.auto_equivalent = 3.0 + + # Capacities - for reference / post-assignment analysis + tier1.seated_capacity, tier1.total_capacity = 7 * 142, 7 * 276 + trolley.seated_capacity, trolley.total_capacity = 4 * 64, 4 * 200 + brt_yellow.seated_capacity, brt_yellow.total_capacity = 32, 70 + brt_red.seated_capacity, brt_red.total_capacity = 32, 70 + premium_bus.seated_capacity, premium_bus.total_capacity = 32, 70 + express_bus.seated_capacity, express_bus.total_capacity = 32, 70 + local_bus.seated_capacity, local_bus.total_capacity = 32, 70 + + trrt_attrs = [] + mode5tod_attrs = [] + for elem_type in "TRANSIT_LINE", "TRANSIT_SEGMENT", "NODE": + mapping = attr_map[elem_type] + for field, (attr, tcoved_type, emme_type, desc) in mapping.iteritems(): + default = "" if emme_type == "STRING" else 0 + network.create_attribute(elem_type, attr, default) + if tcoved_type == "TRRT": + trrt_attrs.append((field, attr)) + elif tcoved_type == "MODE5TOD": + mode5tod_attrs.append((field, attr)) + + # Pre-process transit line (trrt.csv) to know the route names for errors / warnings + transit_line_records = list(transit_line_data) + line_names = {} + for record in transit_line_records: + line_names[int(record["Route_ID"])] = record["Route_Name"].strip() + + links = dict((link["@tcov_id"], link) for link in network.links()) + transit_routes = _defaultdict(lambda: []) + for record in transit_link_data: + line_ref = line_names.get(int(record["Route_ID"]), record["Route_ID"]) + link_id = int(record["Link_ID"]) + if "+" in record["Direction"]: + link = links.get(link_id) + else: + link = links.get(-1*link_id) + if not link: + link = links.get(link_id) + if link and not link.reverse_link: + reverse_link = network.create_link(link.j_node, link.i_node, link.modes) + reverse_link.vertices = list(reversed(link.vertices)) + for attr in network.attributes("LINK"): + if attr not in set(["vertices"]): + reverse_link[attr] = link[attr] + reverse_link["@tcov_id"] = -1 * link["@tcov_id"] + msg = "Transit line %s : Missing reverse link with ID %s (%s) (reverse link created)" % ( + line_ref, record["Link_ID"], link) + self._log.append({"type": "text", "content": msg}) + self._error.append("Transit route import: " + msg) + link = reverse_link + if not link: + msg = "Transit line %s : No link with ID %s, line not created" % ( + line_ref, record["Link_ID"]) + self._log.append({"type": "text", "content": msg}) + self._error.append("Transit route import: " + msg) + fatal_errors += 1 + continue + transit_routes[int(record["Route_ID"])].append(link) + + # lookup list of special tier 1 mode route names + tier1_rail_route_names = [str(n) for n in props["transit.newMode.route"]] + dummy_links = set([]) + transit_lines = {} + for record in transit_line_records: + try: + route = transit_routes[int(record["Route_ID"])] + # Find if name matches one of the names listed in transit.newMode.route and convert to tier 1 rail + is_tier1_rail = False + for name in tier1_rail_route_names: + if str(record["Route_Name"]).startswith(name): + print 'record["Route_Name"]2', record["Route_Name"] + is_tier1_rail = True + break + if is_tier1_rail: + vehicle_type = 45 + mode = network.transit_vehicle(vehicle_type).mode + else: + vehicle_type = int(record["Mode"]) * 10 + mode = network.transit_vehicle(vehicle_type).mode + prev_link = route[0] + itinerary = [prev_link] + for link in route[1:]: + if prev_link.j_node != link.i_node: # filling in the missing gap + msg = "line %s : Links not adjacent, shortest path interpolation used (%s and %s)" % ( + record["Route_Name"], prev_link["@tcov_id"], link["@tcov_id"]) + log_record = {"type": "text", "content": msg} + self._log.append(log_record) + sub_path = find_path(prev_link, link, mode) + itinerary.extend(sub_path) + log_record["content"] = log_record["content"] + " through %s links" % (len(sub_path)) + itinerary.append(link) + prev_link = link + + node_itinerary = [itinerary[0].i_node] + [l.j_node for l in itinerary] + try: + tline = network.create_transit_line( + record["Route_Name"].strip(), vehicle_type, node_itinerary) + except: + msg = "Transit line %s : missing mode added to at least one link" % ( + record["Route_Name"]) + self._log.append({"type": "text", "content": msg}) + for link in itinerary: + link.modes |= set([mode]) + tline = network.create_transit_line( + record["Route_Name"].strip(), vehicle_type, node_itinerary) + + for field, attr in trrt_attrs: + tline[attr] = float(record[field]) + if is_tier1_rail: + line_details = mode_details[11] + else: + line_details = mode_details[int(record["Mode"])] + for field, attr in mode5tod_attrs: + tline[attr] = float(line_details[field]) + #"XFERPENTM": "Transfer penalty time: " + #"WTXFERTM": "Transfer perception:" + # NOTE: an additional transfer penalty perception factor of 5.0 is included + # in assignment + tline["@transfer_penalty"] = float(line_details["XFERPENTM"]) * float(line_details["WTXFERTM"]) + tline.headway = tline["@headway_am"] if tline["@headway_am"] > 0 else 999 + tline.layover_time = 5 + + transit_lines[int(record["Route_ID"])] = tline + for segment in tline.segments(): + segment.allow_boardings = False + segment.allow_alightings = False + segment.transit_time_func = 2 + # ft2 = ul2 -> copied @trtime_link_XX + # segments on links matched to auto network (with auto mode) are changed to ft1 = timau + except Exception as error: + msg = "Transit line %s: %s %s" % (record["Route_Name"], type(error), error) + self._log.append({"type": "text", "content": msg}) + trace_text = _traceback.format_exc().replace("\n", "
") + self._log.append({"type": "text", "content": trace_text}) + self._error.append("Transit route import: line %s not created" % record["Route_Name"]) + fatal_errors += 1 + for link in dummy_links: + network.delete_link(link.i_node, link.j_node) + + line_stops = _defaultdict(lambda: []) + for record in transit_stop_data: + try: + line_name = line_names[int(record["Route_ID"])] + line_stops[line_name].append(record) + except KeyError: + self._log.append( + {"type": "text", + "content": "Stop %s: could not find transit line by ID %s (link ID %s)" % ( + record["Stop_ID"], record["Route_ID"], record["Link_ID"])}) + + seg_float_attr_map = [] + seg_string_attr_map = [] + for field, (attr, t_type, e_type, desc) in attr_map["TRANSIT_SEGMENT"].iteritems(): + if t_type == "TRSTOP": + if e_type == "STRING": + seg_string_attr_map.append([field, attr]) + else: + seg_float_attr_map.append([field, attr]) + + for line_name, stops in line_stops.iteritems(): + tline = network.transit_line(line_name) + if not tline: + continue + itinerary = tline.segments(include_hidden=True) + segment = itinerary.next() + tcov_id = abs(segment.link["@tcov_id"]) + for stop in stops: + if "DUMMY" in stop["StopName"]: + continue + link_id = int(stop['Link_ID']) + node_id = int(stop['TrnNode']) + while tcov_id != link_id: + segment = itinerary.next() + if segment.link is None: + break + tcov_id = abs(segment.link["@tcov_id"]) + + if node_id == segment.i_node.number: + pass + elif node_id == segment.j_node.number: + segment = itinerary.next() # its the next segment + else: + msg = "Transit line %s: could not find stop on link ID %s at node ID %s" % (line_name, link_id, node_id) + self._log.append({"type": "text", "content": msg}) + self._error.append(msg) + fatal_errors += 1 + continue + segment.allow_boardings = True + segment.allow_alightings = True + segment.dwell_time = tline.default_dwell_time + for field, attr in seg_string_attr_map: + segment[attr] = stop[field] + for field, attr in seg_float_attr_map: + segment[attr] = float(stop[field]) + + def lookup_line(ident): + line = network.transit_line(ident) + if line: + return line.id + line = transit_lines.get(int(ident)) + if line: + return line.id + raise Exception("'%s' is not a route name or route ID" % ident) + + # Normalizing the case of the headers as different examples have been seen + for period in periods: + norm_data = [] + for record in timed_xfer_data[period]: + norm_record = {} + for key, val in record.iteritems(): + norm_record[key.lower()] = val + norm_data.append(norm_record) + + from_line, to_line, wait_time = [], [], [] + for i, record in enumerate(norm_data, start=2): + try: + from_line.append(lookup_line(record["from_line"])) + to_line.append(lookup_line(record["to_line"])) + wait_time.append(float(record["wait_time"])) + except Exception as error: + msg = "Error processing timexfer_%s.csv on file line %s: %s" % (period, i, error) + self._log.append({"type": "text", "content": msg}) + self._error.append(msg) + fatal_errors += 1 + + timed_xfer = _dt.Data() + timed_xfer.add_attribute(_dt.Attribute("from_line", _np.array(from_line).astype("O"))) + timed_xfer.add_attribute(_dt.Attribute("to_line", _np.array(to_line).astype("O"))) + timed_xfer.add_attribute(_dt.Attribute("wait_time", _np.array(wait_time))) + # Creates and saves the new table + gen_utils.DataTableProc("%s_timed_xfer_%s" % (self.data_table_name, period), data=timed_xfer) + + if fatal_errors > 0: + raise Exception("Cannot create transit network, %s fatal errors found" % fatal_errors) + self._log.append({"type": "text", "content": "Import transit lines complete"}) + + def calc_transit_attributes(self, network): + self._log.append({"type": "header", "content": "Calculate derived transit attributes"}) + # - TM by 5 TOD periods copied from TM for 3 time periods + # NOTE: the values of @trtime_link_## are only used for + # separate guideway. + # Links shared with the traffic network use the + # assignment results in timau + for link in network.links(): + for time in ["_ea", "_md", "_ev"]: + link["@trtime_link" + time] = link["trtime_link_op"] + if link.type == 0: # walk only links have IFC ==0 + link.type = 99 + + # ON TRANSIT LINES + # Set 3-period headway based on revised headway calculation + for line in network.transit_lines(): + for period in ["am", "pm", "op"]: + line["@headway_rev_" + period] = revised_headway(line["@headway_" + period]) + + for c in network.centroids(): + c["@tap_id"] = c.number + + # Special incremental boarding and in-vehicle fares + # to recreate the coaster zone fares + fares_file_name = FILE_NAMES["FARES"] + special_fare_path = _join(self.source, fares_file_name) + if os.path.isfile(special_fare_path): + with open(special_fare_path) as fare_file: + self._log.append({"type": "text", "content": "Using fare details (for coaster) from %s" % fares_file_name}) + special_fares = None + yaml_installed = True + try: + import yaml + special_fares = yaml.load(fare_file) + self._log.append({"type": "text", "content": yaml.dump(special_fares).replace("\n", "
")}) + except ImportError: + yaml_installed = False + except: + pass + if special_fares is None: + try: + import json + special_fares = json.load(fare_file) + self._log.append({"type": "text", "content": json.dumps(special_fares, indent=4).replace("\n", "
")}) + except: + pass + if special_fares is None: + msg = "YAML or JSON" if yaml_installed else "JSON (YAML parser not installed)" + raise Exception(fares_file_name + ": file could not be parsed as " + msg) + else: + # Default coaster fare for 2012 base year + special_fares = { + "boarding_cost": { + "base": [ + {"line": "398104", "cost" : 4.0}, + {"line": "398204", "cost" : 4.0} + ], + "stop_increment": [ + {"line": "398104", "stop": "SORRENTO VALLEY", "cost": 0.5}, + {"line": "398204", "stop": "SORRENTO VALLEY", "cost": 0.5} + ] + }, + "in_vehicle_cost": [ + {"line": "398104", "from": "SOLANA BEACH", "cost": 1.0}, + {"line": "398104", "from": "SORRENTO VALLEY", "cost": 0.5}, + {"line": "398204", "from": "OLD TOWN", "cost": 1.0}, + {"line": "398204", "from": "SORRENTO VALLEY", "cost": 0.5} + ], + "day_pass": 5.0, + "regional_pass": 12.0 + } + self._log.append({"type": "text", "content": "Using default coaster fare based on 2012 base year setup."}) + + def get_line(line_id): + line = network.transit_line(line_id) + if line is None: + raise Exception("%s: line does not exist: %s" % (fares_file_name, line_id)) + return line + + for record in special_fares["boarding_cost"]["base"]: + line = get_line(record["line"]) + line["@fare"] = 0 + for seg in line.segments(): + seg["@coaster_fare_board"] = record["cost"] + for record in special_fares["boarding_cost"].get("stop_increment", []): + line = get_line(record["line"]) + for seg in line.segments(True): + if record["stop"] in seg["#stop_name"]: + seg["@coaster_fare_board"] += record["cost"] + break + for record in special_fares["in_vehicle_cost"]: + line = get_line(record["line"]) + for seg in line.segments(True): + if record["from"] in seg["#stop_name"]: + seg["@coaster_fare_inveh"] = record["cost"] + break + pass_cost_keys = ['day_pass', 'regional_pass'] + pass_costs = [] + for key in pass_cost_keys: + cost = special_fares.get(key) + if cost is None: + raise Exception("key '%s' missing from %s" % (key, fares_file_name)) + pass_costs.append(cost) + pass_values = _dt.Data() + pass_values.add_attribute(_dt.Attribute("pass_type", _np.array(pass_cost_keys).astype("O"))) + pass_values.add_attribute(_dt.Attribute("cost", _np.array(pass_costs).astype("f8"))) + gen_utils.DataTableProc("%s_transit_passes" % self.data_table_name, data=pass_values) + self._log.append({"type": "text", "content": "Calculate derived transit attributes complete"}) + return + + def create_turns(self, network): + self._log.append({"type": "header", "content": "Import turns and turn restrictions"}) + self._log.append({"type": "text", "content": "Process LINKTYPETURNS.DBF for turn prohibited by type"}) + # Process LINKTYPETURNS.DBF for turn prohibited by type + with _fiona.open(_join(self.source, "LINKTYPETURNS.DBF"), 'r') as f: + link_type_turns = _defaultdict(lambda: {}) + for record in f: + record = record['properties'] + link_type_turns[record["FROM"]][record["TO"]] = { + "LEFT": record["LEFT"], + "RIGHT": record["RIGHT"], + "STRAIGHT": record["STRAIGHT"], + "UTURN": record["UTURN"] + } + for from_link in network.links(): + if from_link.type in link_type_turns: + to_link_turns = link_type_turns[from_link.type] + for to_link in from_link.j_node.outgoing_links(): + if to_link.type in to_link_turns: + record = to_link_turns[to_link.type] + if not from_link.j_node.is_intersection: + network.create_intersection(from_link.j_node) + turn = network.turn(from_link.i_node, from_link.j_node, to_link.j_node) + turn.penalty_func = 1 + if to_link["@tcov_id"] == from_link["left_link"]: + turn.data1 = record["LEFT"] + elif to_link["@tcov_id"] == from_link["through_link"]: + turn.data1 = record["STRAIGHT"] + elif to_link["@tcov_id"] == from_link["right_link"]: + turn.data1 = record["RIGHT"] + else: + turn.data1 = record["UTURN"] + + self._log.append({"type": "text", "content": "Process turns.csv for turn prohibited by ID"}) + turn_data = gen_utils.DataTableProc("turns", _join(self.source, "turns.csv")) + if self.save_data_tables: + turn_data.save("%s_turns" % self.data_table_name, self.overwrite) + links = dict((link["@tcov_id"], link) for link in network.links()) + + # Process turns.csv for prohibited turns from_id, to_id, penalty + for i, record in enumerate(turn_data): + from_link_id, to_link_id = int(record["from_id"]), int(record["to_id"]) + from_link, to_link = links[from_link_id], links[to_link_id] + if from_link.j_node == to_link.i_node: + pass + elif from_link.j_node == to_link.j_node: + to_link = to_link.reverse_link + elif from_link.i_node == to_link.i_node: + from_link = from_link.reverse_link + elif from_link.i_node == to_link.j_node: + from_link = from_link.reverse_link + to_link = to_link.reverse_link + else: + msg = "Record %s: links are not adjacent %s - %s." % (i, from_link_id, to_link_id) + self._log.append({"type": "text", "content": msg}) + self._error.append("Turn import: " + msg) + continue + if not from_link or not to_link: + msg = "Record %s: links adjacent but in reverse direction %s - %s." % (i, from_link_id, to_link_id) + self._log.append({"type": "text", "content": msg}) + self._error.append("Turn import: " + msg) + continue + + node = from_link.j_node + if not node.is_intersection: + network.create_intersection(node) + turn = network.turn(from_link.i_node, node, to_link.j_node) + if not record["penalty"]: + turn.penalty_func = 0 # prohibit turn + else: + turn.penalty_func = 1 + turn.data1 = float(record["penalty"]) + self._log.append({"type": "text", "content": "Import turns and turn restrictions complete"}) + + def calc_traffic_attributes(self, network): + self._log.append({"type": "header", "content": "Calculate derived traffic attributes"}) + # "COST": "@cost_operating" + # "ITOLL": "@toll_flag" # ITOLL - Toll + 100 *[0,1] if managed lane (I-15 tolls) + # Note: toll_flag is no longer used + # "ITOLL2": "@toll" # ITOLL2 - Toll + # "ITOLL3": "@cost_auto" # ITOLL3 - Toll + AOC + # "@cost_hov" + # "ITOLL4": "@cost_med_truck" # ITOLL4 - Toll * 1.03 + AOC + # "ITOLL5": "@cost_hvy_truck" # ITOLL5 - Toll * 2.33 + AOC + fatal_errors = 0 + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + try: + aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) + except ValueError: + raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") + scenario_year = int(props["scenarioYear"]) + periods = ["EA", "AM", "MD", "PM", "EV"] + time_periods = ["_ea", "_am", "_md", "_pm", "_ev"] + src_time_periods = ["_op", "_am", "_op", "_pm", "_op"] + mode_d = network.mode('d') + + # Calculate upstream and downstream interchange distance + # First, label the intersection nodes as nodes with type 1 links (freeway) and + # type 8 links (freeway-to-freeway ramp) + network.create_attribute("NODE", "is_interchange") + interchange_points = [] + for node in network.nodes(): + adj_links = list(node.incoming_links()) + list(node.outgoing_links()) + has_freeway_links = bool( + [l for l in adj_links + if l.type == 1 and mode_d in l.modes]) + has_ramp_links = bool( + [l for l in adj_links + if l.type == 8 and mode_d in l.modes and not "HOV" in l["#name"]]) + if has_freeway_links and has_ramp_links: + node.is_interchange = True + interchange_points.append(node) + else: + node.is_interchange = False + for node in network.nodes(): + node["@interchange"] = node.is_interchange + + for link in network.links(): + if link.type == 1 and mode_d in link.modes: + link["@intdist_down"] = interchange_distance(link, "DOWNSTREAM") + link["@intdist_up"] = interchange_distance(link, "UPSTREAM") + self._log.append({"type": "text", "content": "Calculate of nearest interchange distance complete"}) + + # Static reliability parameters + # freeway coefficients + freeway_rel = { + "intercept": 0.1078, + "speed>70": 0.01393, + "upstream": 0.011, + "downstream": 0.0005445, + } + # arterial/ramp/other coefficients + road_rel = { + "intercept": 0.0546552, + "lanes": { + 1: 0.0, + 2: 0.0103589, + 3: 0.0361211, + 4: 0.0446958, + 5: 0.0 + }, + "speed": { + "<35": 0, + 35: 0.0075674, + 40: 0.0091012, + 45: 0.0080996, + 50: -0.0022938, + ">50": -0.0046211 + }, + "control": { + 0: 0, # Uncontrolled + 1: 0.0030973, # Signal + 2: -0.0063281, # Stop + 3: -0.0063281, # Stop + 4: 0.0127692, # Other, Railway, etc. + } + } + for link in network.links(): + # Change SR125 toll speed to 70MPH + if link["@lane_restriction"] == 4 and link.type == 1: + link["@speed_posted"] = 70 + + link["@cost_operating"] = link.length * aoc + + # Expand off-peak TOD attributes, copy peak period attributes + for time, src_time in zip(time_periods, src_time_periods): + link["@lane" + time] = link["lane" + src_time] + link["@time_link" + time] = link["time_link" + src_time] + + # add link delay (30 sec=0.5mins) to HOV connectors to discourage travel + if link.type == 8 and (link["@lane_restriction"] == 2 or link["@lane_restriction"] == 3): + link["@time_link" + time] = link["@time_link" + time] + 0.375 + + # make speed on HOV lanes (70mph) the same as parallel GP lanes (65mph) + # - set speed back to posted speed - increase travel time by (speed_adj/speed_posted) + if link.type == 1 and (link["@lane_restriction"] == 2 or link["@lane_restriction"] == 3): + speed_adj = link["@speed_adjusted"] + speed_posted = link["@speed_posted"] + if speed_adj>0: + link["@time_link" + time] = (speed_adj/(speed_posted*1.0)) * link["@time_link" + time] + + link["@time_inter" + time] = link["time_inter" + src_time] + link["@toll" + time] = link["toll" + src_time] + + off_peak_factor_file = FILE_NAMES["OFF_PEAK"] + if os.path.exists(_join(self.source, off_peak_factor_file)): + msg = "Adjusting off-peak tolls based on factors from %s" % off_peak_factor_file + self._log.append({"type": "text", "content": msg}) + tolled_links = list(link for link in network.links() if link["toll_op"] > 0) + # NOTE: CSV Reader sets the field names to UPPERCASE for consistency + with gen_utils.CSVReader(_join(self.source, off_peak_factor_file)) as r: + for row in r: + name = row["FACILITY_NAME"] + ea_factor = float(row["OP_EA_FACTOR"]) + md_factor = float(row["OP_MD_FACTOR"]) + ev_factor = float(row["OP_EV_FACTOR"]) + count = 0 + for link in tolled_links: + if name in link["#name"]: + count += 1 + link["@toll_ea"] = link["@toll_ea"] * ea_factor + link["@toll_md"] = link["@toll_md"] * md_factor + link["@toll_ev"] = link["@toll_ev"] * ev_factor + + msg = "Facility name '%s' matched to %s links." % (name, count) + msg += " Adjusted off-peak period tolls EA: %s, MD: %s, EV: %s" % (ea_factor, md_factor, ev_factor) + self._log.append({"type": "text2", "content": msg}) + + for link in network.links(): + factors = [(3.0/12.0), 1.0, (6.5/12.0), (3.5/3.0), (8.0/12.0)] + for f, time, src_time in zip(factors, time_periods, src_time_periods): + if link["capacity_link" + src_time] != 999999: + link["@capacity_link" + time] = f * link["capacity_link" + src_time] + else: + link["@capacity_link" + time] = 999999 + if link["capacity_inter" + src_time] != 999999: + link["@capacity_inter" + time] = f * link["capacity_inter" + src_time] + else: + link["@capacity_inter" + time] = 999999 + + # Required file + vehicle_class_factor_file = FILE_NAMES["VEHICLE_CLASS"] + facility_factors = _defaultdict(lambda: {}) + facility_factors["DEFAULT_FACTORS"] = { + "ALL": { + "auto": 1.0, + "hov2": 1.0, + "hov3": 1.0, + "lgt_truck": 1.0, + "med_truck": 1.03, + "hvy_truck": 2.03 + }, + "count": 0 + } + if os.path.exists(_join(self.source, vehicle_class_factor_file)): + msg = "Adjusting tolls based on factors from %s" % vehicle_class_factor_file + self._log.append({"type": "text", "content": msg}) + # NOTE: CSV Reader sets the field names to UPPERCASE for consistency + with gen_utils.CSVReader(_join(self.source, vehicle_class_factor_file)) as r: + for row in r: + if "YEAR" in r.fields and int(row["YEAR"]) != scenario_year: # optional year column + continue + name = row["FACILITY_NAME"] + # optional time-of-day entry, default to ALL if no column or blank + fac_time = row.get("TIME_OF_DAY") + if fac_time is None: + fac_time = "ALL" + facility_factors[name][fac_time] = { + "auto": float(row["DA_FACTOR"]), + "hov2": float(row["S2_FACTOR"]), + "hov3": float(row["S3_FACTOR"]), + "lgt_truck": float(row["TRK_L_FACTOR"]), + "med_truck": float(row["TRK_M_FACTOR"]), + "hvy_truck": float(row["TRK_H_FACTOR"]) + } + facility_factors[name]["count"] = 0 + + # validate ToD entry, either list EA, AM, MD, PM and EV, or ALL, but not both + for name, factors in facility_factors.iteritems(): + # default keys should be "ALL" and "count" + if "ALL" in factors: + if len(factors) > 2: + fatal_errors += 1 + msg = ("Individual time periods and 'ALL' (or blank) listed under " + "TIME_OF_DAY column in {} for facility {}").format(vehicle_class_factor_file, name) + self._log.append({"type": "text", "content": msg}) + self._error.append(msg) + elif set(periods + ["count"]) != set(factors.keys()): + fatal_errors += 1 + msg = ("Missing time periods {} under TIME_OF_DAY column in {} for facility {}").format( + (set(periods) - set(factors.keys())), vehicle_class_factor_file, name) + self._log.append({"type": "text", "content": msg}) + self._error.append(msg) + + def lookup_link_name(link): + for attr_name in ["#name", "#name_from", "#name_to"]: + for name, _factors in facility_factors.iteritems(): + if name in link[attr_name]: + return _factors + return facility_factors["DEFAULT_FACTORS"] + + def match_facility_factors(link): + factors = lookup_link_name(link) + factors["count"] += 1 + factors = _copy(factors) + del factors["count"] + # @lane_restriction = 2 or 3 overrides hov2 and hov3 costs + if link["@lane_restriction"] == 2: + for _, time_factors in factors.iteritems(): + time_factors["hov2"] = 0.0 + time_factors["hov3"] = 0.0 + elif link["@lane_restriction"] == 3: + for _, time_factors in factors.iteritems(): + time_factors["hov3"] = 0.0 + return factors + + vehicle_classes = ["auto", "hov2", "hov3", "lgt_truck", "med_truck", "hvy_truck"] + for link in network.links(): + if sum(link["@toll" + time] for time in time_periods) > 0: + factors = match_facility_factors(link) + for time, period in zip(time_periods, periods): + time_factors = factors.get(period, factors.get("ALL")) + for name in vehicle_classes: + link["@cost_" + name + time] = time_factors[name] * link["@toll" + time] + link["@cost_operating"] + else: + for time in time_periods: + for name in vehicle_classes: + link["@cost_" + name + time] = link["@cost_operating"] + for name, class_factors in facility_factors.iteritems(): + msg = "Facility name '%s' matched to %s links." % (name, class_factors["count"]) + self._log.append({"type": "text2", "content": msg}) + + self._log.append({"type": "text", "content": "Calculation and time period expansion of costs, tolls, capacities and times complete"}) + + # calculate static reliability + for link in network.links(): + for time in time_periods: + sta_reliability = "@sta_reliability" + time + # if freeway apply freeway parameters to this link + if link["type"] == 1 and link["@lane" + time] > 0: + high_speed_factor = freeway_rel["speed>70"] if link["@speed_posted"] >= 70 else 0.0 + upstream_factor = freeway_rel["upstream"] * 1 / link["@intdist_up"] + downstream_factor = freeway_rel["downstream"] * 1 / link["@intdist_down"] + link[sta_reliability] = ( + freeway_rel["intercept"] + high_speed_factor + upstream_factor + downstream_factor) + # arterial/ramp/other apply road parameters + elif link["type"] <= 9 and link["@lane" + time] > 0: + lane_factor = road_rel["lanes"].get(link["@lane" + time], 0.0) + speed_bin = link["@speed_posted"] + if speed_bin < 35: + speed_bin = "<35" + elif speed_bin > 50: + speed_bin = ">50" + speed_factor = road_rel["speed"][speed_bin] + control_bin = min(max(link["@traffic_control"], 0), 4) + control_factor = road_rel["control"][control_bin] + link[sta_reliability] = road_rel["intercept"] + lane_factor + speed_factor + control_factor + else: + link[sta_reliability] = 0.0 + self._log.append({"type": "text", "content": "Calculate of link static reliability factors complete"}) + + # Cycle length matrix + # Intersecting Link + # Approach Link 2 3 4 5 6 7 8 9 + # IFC Description + # 2 Prime Arterial 2.5 2 2 2 2 2 2 2 + # 3 Major Arterial 2 2 2 2 2 2 2 2 + # 4 Collector 2 2 1.5 1.5 1.5 1.5 1.5 1.5 + # 5 Local Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + # 6 Rural Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + # 7 Local Road 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + # 8 Freeway connector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + # 9 Local Ramp 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + + # Volume-delay functions + # fd10: freeway node approach + # fd11: non-intersection node approach + # fd20: cycle length 1.25 + # fd21: cycle length 1.5 + # fd22: cycle length 2.0 + # fd23: cycle length 2.5 + # fd24: cycle length 2.5 and metered ramp + # fd25: freeway node approach AM and PM only + network.create_attribute("LINK", "green_to_cycle") + network.create_attribute("LINK", "cycle") + vdf_cycle_map = {1.25: 20, 1.5: 21, 2.0: 22, 2.5: 23} + for node in network.nodes(): + incoming = list(node.incoming_links()) + outgoing = list(node.outgoing_links()) + is_signal = False + for link in incoming: + if link["@green_to_cycle_init"] > 0: + is_signal = True + break + if is_signal: + lcs = [link.type for link in incoming + outgoing] + min_lc = max(lcs) # Note: minimum class is actually the HIGHEST value, + max_lc = min(lcs) # and maximum is the LOWEST + + for link in incoming: + # Metered ramps + if link["@traffic_control"] in [4, 5]: + link["cycle"] = 2.5 + link["green_to_cycle"] = 0.42 + link.volume_delay_func = 24 + # Stops + elif link["@traffic_control"] in [2, 3]: + link["cycle"] = 1.25 + link["green_to_cycle"] = 0.42 + link.volume_delay_func = 20 + elif link["@green_to_cycle_init"] > 0 and is_signal: + if link.type == 2: + c_len = 2.5 if min_lc == 2 else 2.0 + elif link.type == 3: + c_len = 2.0 # Major arterial & anything + elif link.type == 4: + c_len = 1.5 if max_lc > 2 else 2.0 + elif link.type > 4: + if max_lc > 4: + c_len = 1.25 + elif max_lc == 4: + c_len = 1.5 + else: + c_len = 2.0 + if link["@green_to_cycle_init"] > 10: + link["green_to_cycle"] = link["@green_to_cycle_init"] / 100.0 + if link["green_to_cycle"] > 1.0: + link["green_to_cycle"] = 1.0 + link["cycle"] = c_len + link.volume_delay_func = vdf_cycle_map[c_len] + elif link.type == 1: + link.volume_delay_func = 10 # freeway + else: + link.volume_delay_func = 11 # non-controlled approach + self._log.append({"type": "text", "content": "Derive cycle, green_to_cycle, and VDF by approach node complete"}) + + for link in network.links(): + if link.volume_delay_func in [10, 11]: + continue + if link["@traffic_control"] in [4, 5]: + # Ramp meter controlled links are only enabled during the peak periods + for time in ["_am", "_pm"]: + link["@cycle" + time] = link["cycle"] + link["@green_to_cycle" + time] = link["green_to_cycle"] + else: + for time in time_periods: + link["@cycle" + time] = link["cycle"] + link["@green_to_cycle" + time] = link["green_to_cycle"] + self._log.append({"type": "text", "content": "Setting of time period @cycle and @green_to_cycle complete"}) + + network.delete_attribute("LINK", "green_to_cycle") + network.delete_attribute("LINK", "cycle") + network.delete_attribute("NODE", "is_interchange") + self._log.append({"type": "text", "content": "Calculate derived traffic attributes complete"}) + if fatal_errors > 0: + raise Exception("%s fatal errors during calculation of traffic attributes" % fatal_errors) + return + + def check_zone_access(self, network, mode): + # Verify that every centroid has at least one available + # access and egress connector + for centroid in network.centroids(): + access = egress = False + for link in centroid.outgoing_links(): + if mode in link.modes: + if link.j_node.is_intersection: + for turn in link.outgoing_turns(): + if turn.i_node != turn.k_node and turn.penalty_func != 0: + egress = True + else: + egress = True + if not egress: + raise Exception("No egress permitted from zone %s" % centroid.id) + for link in centroid.incoming_links(): + if mode in link.modes: + if link.j_node.is_intersection: + for turn in link.incoming_turns(): + if turn.i_node != turn.k_node and turn.penalty_func != 0: + access = True + else: + access = True + if not access: + raise Exception("No access permitted to zone %s" % centroid.id) + + def add_transit_to_traffic(self, hwy_network, tr_network): + if not self.merged_scenario_id or not hwy_network or not tr_network: + return + self._log.append({"type": "header", "content": "Merge transit network to traffic network"}) + fatal_errors = 0 + for tr_mode in tr_network.modes(): + hwy_mode = hwy_network.create_mode(tr_mode.type, tr_mode.id) + hwy_mode.description = tr_mode.description + hwy_mode.speed = tr_mode.speed + for tr_veh in tr_network.transit_vehicles(): + hwy_veh = hwy_network.create_transit_vehicle(tr_veh.id, tr_veh.mode.id) + hwy_veh.description = tr_veh.description + hwy_veh.auto_equivalent = tr_veh.auto_equivalent + hwy_veh.seated_capacity = tr_veh.seated_capacity + hwy_veh.total_capacity = tr_veh.total_capacity + + for elem_type in ["NODE", "LINK", "TRANSIT_LINE", "TRANSIT_SEGMENT"]: + for attr in tr_network.attributes(elem_type): + if not attr in hwy_network.attributes(elem_type): + default = "" if attr.startswith("#") else 0 + new_attr = hwy_network.create_attribute(elem_type, attr, default) + + hwy_link_index = dict((l["@tcov_id"], l) for l in hwy_network.links()) + hwy_node_position_index = dict(((n.x, n.y), n) for n in hwy_network.nodes()) + hwy_node_index = dict() + not_matched_links = [] + for tr_link in tr_network.links(): + tcov_id = tr_link["@tcov_id"] + if tcov_id == 0: + i_node = hwy_node_position_index.get((tr_link.i_node.x, tr_link.i_node.y)) + j_node = hwy_node_position_index.get((tr_link.j_node.x, tr_link.j_node.y)) + if i_node and j_node: + hwy_link = hwy_network.link(i_node, j_node) + else: + hwy_link = None + else: + hwy_link = hwy_link_index.get(tcov_id) + if not hwy_link: + not_matched_links.append(tr_link) + else: + hwy_node_index[tr_link.i_node] = hwy_link.i_node + hwy_node_index[tr_link.j_node] = hwy_link.j_node + hwy_link.modes |= tr_link.modes + + new_node_id = max(n.number for n in hwy_network.nodes()) + new_node_id = int(_ceiling(new_node_id / 10000.0) * 10000) + bus_mode = tr_network.mode("b") + + def lookup_node(src_node, new_node_id): + node = hwy_node_index.get(src_node) + if not node: + node = hwy_node_position_index.get((src_node.x, src_node.y)) + if not node: + node = hwy_network.create_regular_node(new_node_id) + new_node_id += 1 + for attr in tr_network.attributes("NODE"): + node[attr] = src_node[attr] + hwy_node_index[src_node] = node + return node, new_node_id + + for tr_link in not_matched_links: + i_node, new_node_id = lookup_node(tr_link.i_node, new_node_id) + j_node, new_node_id = lookup_node(tr_link.j_node, new_node_id) + # check for duplicate but different links + # All cases to be logged and then an error raised at end + ex_link = hwy_network.link(i_node, j_node) + if ex_link: + self._log.append({ + "type": "text", + "content": "Duplicate links between the same nodes with different IDs in traffic/transit merge. " + "Traffic link ID %s, transit link ID %s." % (ex_link["@tcov_id"], tr_link["@tcov_id"]) + }) + self._error.append("Duplicate links with different IDs between traffic (%s) and transit (%s) networks" % + (ex_link["@tcov_id"], tr_link["@tcov_id"])) + self._split_link(hwy_network, i_node, j_node, new_node_id) + new_node_id += 1 + fatal_errors += 1 + try: + link = hwy_network.create_link(i_node, j_node, tr_link.modes) + except Exception as error: + self._log.append({ + "type": "text", + "content": "Error creating link '%s', I-node '%s', J-node '%s'. Error message %s" % + (tr_link["@tcov_id"], i_node, j_node, error) + }) + self._error.append("Cannot create transit link '%s' in traffic network" % tr_link["@tcov_id"]) + fatal_errors += 1 + continue + hwy_link_index[tr_link["@tcov_id"]] = link + for attr in tr_network.attributes("LINK"): + link[attr] = tr_link[attr] + link.vertices = tr_link.vertices + + # Create transit lines and copy segment data + for tr_line in tr_network.transit_lines(): + itinerary = [] + for seg in tr_line.segments(True): + itinerary.append(hwy_node_index[seg.i_node]) + try: + hwy_line = hwy_network.create_transit_line(tr_line.id, tr_line.vehicle.id, itinerary) + except Exception as error: + msg = "Transit line %s, error message %s" % (tr_line.id, error) + self._log.append({"type": "text", "content": msg}) + self._error.append("Cannot create transit line '%s' in traffic network" % tr_line.id) + fatal_errors += 1 + continue + for attr in hwy_network.attributes("TRANSIT_LINE"): + hwy_line[attr] = tr_line[attr] + for tr_seg, hwy_seg in _izip(tr_line.segments(True), hwy_line.segments(True)): + for attr in hwy_network.attributes("TRANSIT_SEGMENT"): + hwy_seg[attr] = tr_seg[attr] + + # Change ttf from ft2 (fixed speed) to ft1 (congested auto time) + auto_mode = hwy_network.mode("d") + for hwy_link in hwy_network.links(): + if auto_mode in hwy_link.modes: + for seg in hwy_link.segments(): + seg.transit_time_func = 1 + if fatal_errors > 0: + raise Exception("Cannot merge traffic and transit network, %s fatal errors found" % fatal_errors) + + self._log.append({"type": "text", "content": "Merge transit network to traffic network complete"}) + + def _split_link(self, network, i_node, j_node, new_node_id): + # Attribute types to maintain consistency for correspondence with incoming / outgoing link data + periods = ["ea", "am", "md", "pm", "ev"] + approach_attrs = ["@traffic_control", "@turn_thru", "@turn_right", "@turn_left", + "@lane_auxiliary", "@green_to_cycle_init"] + for p_attr in ["@green_to_cycle_", "@time_inter_", "@cycle_"]: + approach_attrs.extend([p_attr + p for p in periods]) + capacity_inter = ["@capacity_inter_" + p for p in periods] + cost_attrs = ["@cost_operating"] + for p_attr in ["@cost_lgt_truck_", "@cost_med_truck_", "@cost_hvy_truck_", "@cost_hov2_", + "@cost_hov3_", "@cost_auto_", "@time_link_", "@trtime_link_", "@toll_"]: + cost_attrs.extend([p_attr + p for p in periods]) + approach_attrs = [a for a in approach_attrs if a in network.attributes("LINK")] + capacity_inter = [a for a in capacity_inter if a in network.attributes("LINK")] + cost_attrs = [a for a in cost_attrs if a in network.attributes("LINK")] + + new_node = network.split_link(i_node, j_node, new_node_id) + + # Correct attributes on the split links + for link in new_node.incoming_links(): + link["#name_to"] = "" + for attr in approach_attrs: + link[attr] = 0 + for attr in capacity_inter: + link[attr] = 999999 + for attr in cost_attrs: + link[attr] = 0.5 * link[attr] + link.volume_delay_func = 10 + for link in new_node.outgoing_links(): + link["#name_from"] = "" + for attr in cost_attrs: + link[attr] = 0.5 * link[attr] + + def adjust_network(self, hwy_network): + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + taz_cwk = pd.read_csv(_join(self.source, props["taz.to.cluster.crosswalk.file"]), index_col = 0) + taz_cwk = taz_cwk['cluster_id'].to_dict() + + centroid_nodes = [] + exclude_nodes = [] + + for node in range(1,13,1): + exclude_nodes.append(hwy_network.node(node)) + + for node in hwy_network.centroids(): + if not node in exclude_nodes: + centroid_nodes.append(node) + + i_nodes = [] + j_nodes = [] + data1 = [] + length = [] + links = [] + + for link in hwy_network.links(): + if link.i_node in centroid_nodes: + links.append(link) + i_nodes.append(int(link.i_node)) + j_nodes.append(int(link.j_node)) + data1.append(link.data1) + length.append(link.length) + + df = pd.DataFrame({'links' : links, 'i_nodes' : i_nodes, 'j_nodes': j_nodes, 'ul1_org': data1, 'length_org':length}) + df['i_nodes_new'] = df['i_nodes'].map(taz_cwk) + + j_nodes_list = df['j_nodes'].unique() + j_nodes_list = [hwy_network.node(x) for x in j_nodes_list] + + j_nodes = [] + j_x = [] + j_y = [] + for nodes in hwy_network.nodes(): + if nodes in j_nodes_list: + j_nodes.append(nodes) + j_x.append(nodes.x) + j_y.append(nodes.y) + + j_nodes_XY = pd.DataFrame({'j_nodes' : j_nodes, 'j_x' : j_x, 'j_y': j_y}) + j_nodes_XY['j_nodes'] = [int(x) for x in j_nodes_XY['j_nodes']] + df = pd.merge(df, j_nodes_XY, on = 'j_nodes', how = 'left') + + agg_node_coords = pd.read_csv(_join(self.source, props["cluster.zone.centroid.file"])) + df = pd.merge(df, agg_node_coords, left_on = 'i_nodes_new', right_on = 'cluster_id', how = 'left') + df = df.drop(columns = 'cluster_id') + df = df.rename(columns = {'centroid_x' : 'i_new_x', 'centroid_y' : 'i_new_y'}) + + i_coords = zip(df['j_x'], df['j_y']) + j_coords = zip(df['i_new_x'], df['i_new_y']) + + df['length'] = [distance.euclidean(i, j)/5280.0 for i,j in zip(i_coords, j_coords)] + + #delete all the existing centroid nodes + for index,row in df.iterrows(): + if hwy_network.node(row['i_nodes']): + hwy_network.delete_node(row['i_nodes'], True) + + # create new nodes (centroids of clusters) + for index,row in agg_node_coords.iterrows(): + new_node = hwy_network.create_node(row['cluster_id'], is_centroid = True) + new_node.x = int(row['centroid_x']) + new_node.y = int(row['centroid_y']) + + df['type'] = 10 + df['num_lanes'] = 1.0 + df['vdf'] = 11 + df['ul3'] = 999999 + + # create new links + for index,row in df.iterrows(): + try: + link_ij = hwy_network.create_link(row['i_nodes_new'], row['j_nodes'], + modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) + link_ij.length = row['length'] + link_ij.type = row['type'] + link_ij.num_lanes = row['num_lanes'] + link_ij.volume_delay_func = row['vdf'] + link_ij.data3 = row['ul3'] + + link_ji = hwy_network.create_link(row['j_nodes'], row['i_nodes_new'], + modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) + link_ji.length = row['length'] + link_ji.type = row['type'] + link_ji.num_lanes = row['num_lanes'] + link_ji.volume_delay_func = row['vdf'] + link_ji.data3 = row['ul3'] + + except: + continue + + + @_m.logbook_trace("Set database functions (VDF, TPF and TTF)") + def set_functions(self, scenario): + create_function = _m.Modeller().tool( + "inro.emme.data.function.create_function") + set_extra_function_params = _m.Modeller().tool( + "inro.emme.traffic_assignment.set_extra_function_parameters") + emmebank = self.emmebank + for f_id in ["fd10", "fd11", "fd20", "fd21", "fd22", "fd23", "fd24", "fp1", "ft1", "ft2", "ft3", "ft4"]: + function = emmebank.function(f_id) + if function: + emmebank.delete_function(function) + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) + smartSignalf_CL = props["smartSignal.factor.LC"] + smartSignalf_MA = props["smartSignal.factor.MA"] + smartSignalf_PA = props["smartSignal.factor.PA"] + atdmf = props["atdm.factor"] + + reliability_tmplt = ( + "* (1 + el2 + {0}*(".format(atdmf)+ + "( {factor[LOS_C]} * ( put(get(1).min.1.5) - {threshold[LOS_C]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_C]})" + "+ ( {factor[LOS_D]} * ( get(2) - {threshold[LOS_D]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_D]})" + "+ ( {factor[LOS_E]} * ( get(2) - {threshold[LOS_E]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_E]})" + "+ ( {factor[LOS_FL]} * ( get(2) - {threshold[LOS_FL]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_FL]})" + "+ ( {factor[LOS_FH]} * ( get(2) - {threshold[LOS_FH]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_FH]})" + "))") + parameters = { + "freeway": { + "factor": { + "LOS_C": 0.2429, "LOS_D": 0.1705, "LOS_E": -0.2278, "LOS_FL": -0.1983, "LOS_FH": 1.022 + }, + "threshold": { + "LOS_C": 0.7, "LOS_D": 0.8, "LOS_E": 0.9, "LOS_FL": 1.0, "LOS_FH": 1.2 + }, + }, + "road": { # for arterials, ramps, collectors, local roads, etc. + "factor": { + "LOS_C": 0.1561, "LOS_D": 0.0, "LOS_E": 0.0, "LOS_FL": -0.449, "LOS_FH": 0.0 + }, + "threshold": { + "LOS_C": 0.7, "LOS_D": 0.8, "LOS_E": 0.9, "LOS_FL": 1.0, "LOS_FH": 1.2 + }, + } + } + # freeway fd10 + create_function( + "fd10", + "(ul1 * (1.0 + 0.24 * put((volau + volad) / ul3) ** 5.5))" + + reliability_tmplt.format(**parameters["freeway"]), + emmebank=emmebank) + # non-freeway link which is not an intersection approach fd11 + create_function( + "fd11", + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0))" + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + create_function( + "fd20", # Local collector and lower intersection and stop controlled approaches + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" + "1.25 / 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))" + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + create_function( + "fd21", # Collector intersection approaches + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" + "{0} * 1.5/ 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_CL) + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + create_function( + "fd22", # Major arterial and major or prime arterial intersection approaches + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" + "{0} * 2.0 / 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_MA) + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + create_function( + "fd23", # Primary arterial intersection approaches + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" + "{0} * 2.5/ 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_PA) + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + create_function( + "fd24", # Metered ramps + "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" + "2.5/ 2 * (1-el1) ** 2 * (1.0 + 6.0 * ( (volau + volad) / el3 ) ** 2.0))" + + reliability_tmplt.format(**parameters["road"]), + emmebank=emmebank) + # freeway fd25 (AM and PM only) + create_function( + "fd25", + "(ul1 * (1.0 + 0.6 * put((volau + volad) / ul3) ** 4))" + + reliability_tmplt.format(**parameters["freeway"]), + emmebank=emmebank) + + set_extra_function_params( + el1="@green_to_cycle", el2="@sta_reliability", el3="@capacity_inter_am", + emmebank=emmebank) + + create_function("fp1", "up1", emmebank=emmebank) # fixed cost turns stored in turn data 1 (up1) + + # buses in mixed traffic, use auto time + create_function("ft1", "timau", emmebank=emmebank) + # fixed speed for separate guideway operations + create_function("ft2", "ul2", emmebank=emmebank) + # special 0-cost segments for prohibition of walk to different stop from centroid + create_function("ft3", "0", emmebank=emmebank) + # fixed guideway systems according to vehicle speed (not used at the moment) + create_function("ft4", "60 * length / speed", emmebank=emmebank) + + @_m.logbook_trace("Traffic zone connectivity check") + def check_connectivity(self, scenario): + modeller = _m.Modeller() + sola_assign = modeller.tool( + "inro.emme.traffic_assignment.sola_traffic_assignment") + set_extra_function_para = modeller.tool( + "inro.emme.traffic_assignment.set_extra_function_parameters") + create_matrix = _m.Modeller().tool( + "inro.emme.data.matrix.create_matrix") + net_calc = gen_utils.NetworkCalculator(scenario) + + emmebank = scenario.emmebank + zone_index = dict(enumerate(scenario.zone_numbers)) + num_processors = dem_utils.parse_num_processors("MAX-1") + + # Note matrix is also created in initialize_matrices + create_matrix("ms1", "zero", "zero", scenario=scenario, overwrite=True) + with gen_utils.temp_matrices(emmebank, "FULL", 1) as (result_matrix,): + result_matrix.name = "TEMP_SOV_TRAVEL_TIME" + set_extra_function_para( + el1="@green_to_cycle_am", + el2="@sta_reliability_am", + el3="@capacity_inter_am", emmebank=emmebank) + net_calc("ul1", "@time_link_am", "modes=d") + net_calc("ul3", "@capacity_link_am", "modes=d") + net_calc("lanes", "@lane_am", "modes=d") + spec = { + "type": "SOLA_TRAFFIC_ASSIGNMENT", + "background_traffic": None, + "classes": [ + { + "mode": "S", # SOV toll mode + "demand": 'ms"zero"', + "generalized_cost": None, + "results": { + "od_travel_times": {"shortest_paths": result_matrix.named_id} + } + } + ], + "stopping_criteria": { + "max_iterations": 0, "best_relative_gap": 0.0, + "relative_gap": 0.0, "normalized_gap": 0.0 + }, + "performance_settings": {"number_of_processors": num_processors}, + } + sola_assign(spec, scenario=scenario) + travel_time = result_matrix.get_numpy_data(scenario) + + is_disconnected = (travel_time == 1e20) + disconnected_pairs = is_disconnected.sum() + if disconnected_pairs > 0: + error_msg = "Connectivity error(s) between %s O-D pairs" % disconnected_pairs + self._log.append({"type": "header", "content": error_msg}) + count_disconnects = [] + for axis, term in [(0, "from"), (1, "to")]: + axis_totals = is_disconnected.sum(axis=axis) + for i, v in enumerate(axis_totals): + if v > 0: + count_disconnects.append((zone_index[i], term, v)) + count_disconnects.sort(key=lambda x: x[2], reverse=True) + for z, direction, count in count_disconnects[:50]: + msg ="Zone %s disconnected %s %d other zones" % (z, direction, count) + self._log.append({"type": "text", "content": msg}) + if disconnected_pairs > 50: + self._log.append({"type": "text", "content": "[List truncated]"}) + raise Exception(error_msg) + self._log.append({"type": "header", "content": + "Zone connectivity verified for AM period on SOV toll ('S') mode"}) + scenario.has_traffic_results = False + + def log_report(self): + report = _m.PageBuilder(title="Import network from TCOVED files report") + try: + if self._error: + report.add_html("
Errors detected during import: %s
" % len(self._error)) + error_msg = ["
    "] + for error in self._error: + error_msg.append("
  • %s
  • " % error) + error_msg.append("
") + report.add_html("".join(error_msg)) + else: + report.add_html("No errors detected during import") + + for item in self._log: + if item["type"] == "text": + report.add_html("
%s
" % item["content"]) + if item["type"] == "text2": + report.add_html("
%s
" % item["content"]) + elif item["type"] == "header": + report.add_html("

%s

" % item["content"]) + elif item["type"] == "table": + table_msg = ["
", "

%s

" % item["title"]] + if "header" in item: + table_msg.append("") + for label in item["header"]: + table_msg.append("" % label) + table_msg.append("") + for row in item["content"]: + table_msg.append("") + for cell in row: + table_msg.append("" % cell) + table_msg.append("") + table_msg.append("
%s
%s
") + report.add_html("".join(table_msg)) + + except Exception as error: + # no raise during report to avoid masking real error + report.add_html("Error generating report") + report.add_html(unicode(error)) + report.add_html(_traceback.format_exc()) + + _m.logbook_write("Import network report", report.render()) + + +def get_node(network, number, coordinates, is_centroid): + node = network.node(number) + if not node: + node = network.create_node(number, is_centroid) + node.x, node.y = coordinates + return node + + +# shortest path interpolation +def find_path(orig_link, dest_link, mode): + visited = set([]) + visited_add = visited.add + back_links = {} + heap = [] + + for link in orig_link.j_node.outgoing_links(): + if mode in link.modes: + back_links[link] = None + _heapq.heappush(heap, (link["length"], link)) + + link_found = False + try: + while not link_found: + link_cost, link = _heapq.heappop(heap) + if link in visited: + continue + visited_add(link) + for outgoing in link.j_node.outgoing_links(): + if mode not in outgoing.modes: + continue + if outgoing in visited: + continue + back_links[outgoing] = link + if outgoing == dest_link: + link_found = True + break + outgoing_cost = link_cost + link["length"] + _heapq.heappush(heap, (outgoing_cost, outgoing)) + except IndexError: + pass # IndexError if heap is empty + if not link_found: + raise NoPathException( + "no path found between links with trcov_id %s and %s (Emme IDs %s and %s)" % ( + orig_link["@tcov_id"], dest_link["@tcov_id"], orig_link, dest_link)) + + prev_link = back_links[dest_link] + route = [] + while prev_link: + route.append(prev_link) + prev_link = back_links[prev_link] + return list(reversed(route)) + + +class NoPathException(Exception): + pass + + +def revised_headway(headway): + # CALCULATE REVISED HEADWAY + # new headway calculation is less aggressive; also only being used for initial wait + # It uses a negative exponential formula to calculate headway + # + if headway <= 10: + rev_headway = headway + else: + rev_headway = headway * (0.275 + 0.788 * _np.exp(-0.011*headway)) + return rev_headway + + +def interchange_distance(orig_link, direction): + visited = set([]) + visited_add = visited.add + back_links = {} + heap = [] + if direction == "DOWNSTREAM": + get_links = lambda l: l.j_node.outgoing_links() + check_far_node = lambda l: l.j_node.is_interchange + elif direction == "UPSTREAM": + get_links = lambda l: l.i_node.incoming_links() + check_far_node = lambda l: l.i_node.is_interchange + # Shortest path search for nearest interchange node along freeway + for link in get_links(orig_link): + _heapq.heappush(heap, (link["length"], link)) + interchange_found = False + try: + while not interchange_found: + link_cost, link = _heapq.heappop(heap) + if link in visited: + continue + visited_add(link) + if check_far_node(link): + interchange_found = True + break + for next_link in get_links(link): + if next_link in visited: + continue + next_cost = link_cost + link["length"] + _heapq.heappush(heap, (next_cost, next_link)) + except IndexError: + # IndexError if heap is empty + # case where start / end of highway, dist = 99 + return 99 + return orig_link["length"] / 2.0 + link_cost diff --git a/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py new file mode 100644 index 0000000..2b87f35 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py @@ -0,0 +1,193 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/import_seed_demand.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Imports the warm start demand matrices from specified OMX files for auto, truck and transit. +# +# Note the matrix name mapping from the OMX file names to the Emme database names. +# +# Inputs: +# omx_file: source +# demand_type: The type of demand in the provided OMX file, one of "AUTO", "TRUCK", "TRANSIT". +# Used to determine the matrix mapping for the import. +# period: The period for which to import the matrices, one of "EA", "AM", "MD", "PM", "EV" +# scenario: traffic scenario to use for reference zone system +# convert_truck_to_pce: boolean, if True the result matrices are adjusted to PCEs instead of +# vehicles (default, and required for traffic assignment). Only used if the demand_type is TRUCK. +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# For AUTO: +# pp_SOVNTP, pp_SOVTB, pp_HOV2, pp_HOV3 +# For TRUCK: +# pp_TRKH, pp_TRKL, pp_TRKM +# For TRANSIT: +# pp_WLKBUS, pp_WLKLRT, pp_WLKCMR, pp_WLKEXP, pp_WLKBRT, +# pp_PNRBUS, pp_PNRLRT, pp_PNRCMR, pp_PNREXP, pp_PNRBRT, +# pp_KNRBUS, pp_KNRLRT, pp_KNRCMR, pp_KNREXP, pp_KNRBRT +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + period = "AM" + input_omx_file = os.path.join(main_directory, "input", "trip_%s.omx" % period) + demand_type = "TRUCK" + demand_as_pce = True + base_scenario = modeller.scenario + import_seed_demand = modeller.tool("sandag.import.import_seed_demand") + import_seed_demand(input_omx_file, demand_type, period, demand_as_pce, base_scenario) +""" + + +TOOLBOX_ORDER = 12 + + +import inro.modeller as _m +import inro.emme.matrix as _matrix +import traceback as _traceback + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +_omx = _m.Modeller().module("sandag.utilities.omxwrapper") + + +class ImportMatrices(_m.Tool(), gen_utils.Snapshot): + + omx_file = _m.Attribute(unicode) + demand_type = _m.Attribute(str) + period = _m.Attribute(str) + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.attributes = ["omx_file", "demand_type", "period"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Import demand matrices" + pb.description = """Imports the seed demand matrices.""" + pb.branding_text = "- SANDAG - Import" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('omx_file', 'file', + title='Select input OMX file') + options = [(x, x) for x in ["AUTO", "TRUCK", "TRANSIT"]] + pb.add_select("demand_type", keyvalues=options, title="Select corresponding demand type") + options = [(x, x) for x in ["EA", "AM", "MD", "PM", "EV"]] + pb.add_select("period", keyvalues=options, title="Select corresponding period") + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.omx_file, self.demand_type, self.period, scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, omx_file, demand_type, period, scenario): + attributes = { + "omx_file": omx_file, + "demand_type": demand_type, + "period": period, + "scenario": scenario.id, + "self": str(self) + } + with _m.logbook_trace("Import %s matrices for period %s" % (demand_type, period), attributes=attributes): + gen_utils.log_snapshot("Import matrices", str(self), attributes) + demand_types = ["AUTO", "TRUCK", "TRANSIT"] + if demand_type not in demand_types: + raise Exception("Invalid demand_type, must be one of %s" % demand_types) + periods = ["EA", "AM", "MD", "PM", "EV"] + if period not in periods: + raise Exception("Invalid period, must be one of %s" % periods) + + if demand_type == "AUTO": + # TODO: update for new seed matrices + matrices = { + '%s_SOV_NT_L': 'mf"%s_SOV_NT_L"', + '%s_SOV_TR_L': 'mf"%s_SOV_TR_L"', + '%s_HOV2_L': 'mf"%s_HOV2_L"', + '%s_HOV3_L': 'mf"%s_HOV3_L"', + '%s_SOV_NT_M': 'mf"%s_SOV_NT_M"', + '%s_SOV_TR_M': 'mf"%s_SOV_TR_M"', + '%s_HOV2_M': 'mf"%s_HOV2_M"', + '%s_HOV3_M': 'mf"%s_HOV3_M"', + '%s_SOV_NT_H': 'mf"%s_SOV_NT_H"', + '%s_SOV_TR_H': 'mf"%s_SOV_TR_H"', + '%s_HOV2_H': 'mf"%s_HOV2_H"', + '%s_HOV3_H': 'mf"%s_HOV3_H"'} + matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) + self._import_from_omx(omx_file, matrices, scenario) + + if demand_type == "TRUCK": + # TODO: update for new seed matrices + matrices = { + '%s_TRK_H': 'mf"%s_TRK_H"', + '%s_TRK_L': 'mf"%s_TRK_L"', + '%s_TRK_M': 'mf"%s_TRK_M"'} + matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) + self._import_from_omx(omx_file, matrices, scenario) + + if demand_type == "TRANSIT": + matrices = { + 'SET1': 'mf"%s_WLKBUS"', + 'SET2': 'mf"%s_WLKPREM"', + 'SET3': 'mf"%s_WLKALLPEN"',} + matrices = dict((k, v % period) for k, v in matrices.iteritems()) + # special custom mapping from subset of TAPs to all TAPs + self._import_from_omx(omx_file, matrices, scenario) + + def _import_from_omx(self, file_path, matrices, scenario): + matrices_to_write = {} + emme_zones = scenario.zone_numbers + emmebank = scenario.emmebank + omx_file_obj = _omx.open_file(file_path, 'r') + try: + zone_mapping = omx_file_obj.mapping(omx_file_obj.list_mappings()[0]).items() + zone_mapping.sort(key=lambda x: x[1]) + omx_zones = [x[0] for x in zone_mapping] + for omx_name, emme_name in matrices.iteritems(): + omx_data = omx_file_obj[omx_name].read() + if emme_name not in matrices_to_write: + matrices_to_write[emme_name] = omx_data + else: + # Allow multiple src matrices from OMX to sum to same matrix in Emme + matrices_to_write[emme_name] = omx_data + matrices_to_write[emme_name] + except Exception as error: + import traceback + print (traceback.format_exc()) + omx_file_obj.close() + + if omx_zones != emme_zones: + # special custom mapping from subset of TAPs to all TAPs + for emme_name, omx_data in matrices_to_write.iteritems(): + matrix_data = _matrix.MatrixData(type='f', indices=[omx_zones, omx_zones]) + matrix_data.from_numpy(omx_data) + expanded_matrix_data = matrix_data.expand([emme_zones, emme_zones]) + matrix = emmebank.matrix(emme_name) + matrix.set_data(expanded_matrix_data, scenario) + else: + for emme_name, omx_data in matrices_to_write.iteritems(): + matrix = emmebank.matrix(emme_name) + matrix.set_numpy_data(omx_data, scenario) diff --git a/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py new file mode 100644 index 0000000..827baff --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py @@ -0,0 +1,230 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/import_transit_demand.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Imports the transit demand generated from an iteration of the disaggregate +# demand models (CT-RAMP) in preparation for the transit assignment +# +# Note the matrix name mapping from the OMX file names to the Emme database names. +# +# Inputs: +# output_dir: output directory to read the OMX files from +# scenario: transit scenario to use for reference zone system +# +# Files referenced: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# output/tranTrips_pp.omx +# output/tranCrossBorderTrips_pp.omx +# output/tranAirportTrips.SAN_pp.omx +# output/tranAirportTrips.CBX_pp.omx (optional) +# output/tranVisitorTrips_pp.omx +# output/tranInternalExternalTrips_pp.omx +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# pp_WLKBUS, pp_WLKLRT, pp_WLKCMR, pp_WLKEXP, pp_WLKBRT, +# pp_PNRBUS, pp_PNRLRT, pp_PNRCMR, pp_PNREXP, pp_PNRBRT, +# pp_KNRBUS, pp_KNRLRT, pp_KNRCMR, pp_KNREXP, pp_KNRBRT +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + output_dir = os.path.join(main_directory, "output") + scenario = modeller.scenario + import_transit_demand = modeller.tool("sandag.import.import_transit_demand") + import_transit_demand(output_dir, scenario) +""" + + +TOOLBOX_ORDER = 14 + + +import inro.modeller as _m +import inro.emme.matrix as _matrix +import traceback as _traceback +import os + + +dem_utils = _m.Modeller().module('sandag.utilities.demand') +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ImportMatrices(_m.Tool(), gen_utils.Snapshot): + + output_dir = _m.Attribute(unicode) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + main_dir = os.path.dirname(project_dir) + self.output_dir = os.path.join(main_dir, "output") + self.attributes = ["output_dir"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Import transit demand" + pb.description = """ +
+ Imports the trip matrices generated by CT-RAMP in OMX format.
+ A total of 30 OMX files are expected, for 5 time periods + EA, AM, MD, PM and EV, with internal matrices by 3 model segments + (assignment access sets) and 3 access modes (walk, PNR, KNR): +
    +
  • tranTrips_pp.omx
  • +
  • tranCrossBorderTrips_pp.omx
  • +
  • tranAirportTrips.SAN_pp.omx
  • +
  • tranAirportTrips.CBX_pp.omx (optional)
  • +
  • tranVisitorTrips_pp.omx
  • +
  • tranInternalExternalTrips_pp.omx
  • +
+
+ """ + pb.branding_text = "- SANDAG - Model" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_select_file('output_dir', 'directory', + title='Select output directory') + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.output_dir, scenario) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Create TOD transit trip tables", save_arguments=True) + def __call__(self, output_dir, scenario): + attributes = {"output_dir": output_dir} + gen_utils.log_snapshot("Sum demand", str(self), attributes) + + self.scenario = scenario + self.output_dir = output_dir + self.import_transit_trips() + + @_m.logbook_trace("Import CT-RAMP transit trips from OMX") + def import_transit_trips(self): + emmebank = self.scenario.emmebank + emme_zones = self.scenario.zone_numbers + matrix_name_tmplts = [ + ("mf%s_%sBUS", "%s_%s_set1_%s"), + ("mf%s_%sPREM", "%s_%s_set2_%s"), + ("mf%s_%sALLPEN", "%s_%s_set3_%s") + ] + periods = ["EA", "AM", "MD", "PM", "EV"] + access_modes = ["WLK", "PNR", "KNR"] + matrix_names = [] + for period in periods: + for acc_mode in access_modes: + #for trip_set in trip_sets: + for emme_name, omx_name in matrix_name_tmplts: + matrix_names.append( + ("_" + period, emme_name % (period, acc_mode), omx_name % (acc_mode, "%s", period))) + + with gen_utils.OMXManager(self.output_dir, "tran%sTrips%s.omx") as omx_manager: + for period, matrix_name, omx_key in matrix_names: + logbook_label = "Report on import from OMX key %s to matrix %s" % (omx_key % "SET", matrix_name) + + #add both KNR_SET and TNC_SET into KNR + if ("KNR" in matrix_name): + #resident + person_knr_demand = omx_manager.lookup(("", period), omx_key % "SET") + person_tnc_Demand = omx_manager.lookup(("", period), omx_key.replace("KNR","TNC") % "SET") + person_demand = person_knr_demand + person_tnc_Demand + #visitor + visitor_knr_demand = omx_manager.lookup(("Visitor", period), omx_key % "SET") + visitor_tnc_Demand = omx_manager.lookup(("Visitor", period), omx_key.replace("KNR","TNC") % "SET") + visitor_demand = visitor_knr_demand + visitor_tnc_Demand + #cross border + cross_border_knr_demand = omx_manager.lookup(("CrossBorder", period), omx_key % "SET") + cross_border_tnc_Demand = omx_manager.lookup(("CrossBorder", period), omx_key.replace("KNR","TNC") % "SET") + cross_border_demand = cross_border_knr_demand + cross_border_tnc_Demand + #airport SAN + airport_knr_demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key % "SET") + airport_tnc_Demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key.replace("KNR","TNC") % "SET") + #airport CBX + if omx_manager.file_exists(("Airport", ".CBX" + period)): + airport_knr_demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key % "SET") + airport_tnc_Demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key.replace("KNR","TNC") % "SET") + airport_demand = airport_knr_demand + airport_tnc_Demand + #internal external + internal_external_knr_demand = omx_manager.lookup(("InternalExternal", period), omx_key % "SET") + internal_external_tnc_Demand = omx_manager.lookup(("InternalExternal", period), omx_key.replace("KNR","TNC") % "SET") + internal_external_demand = internal_external_knr_demand + internal_external_tnc_Demand + else: + person_demand = omx_manager.lookup(("", period), omx_key % "SET") + visitor_demand = omx_manager.lookup(("Visitor", period), omx_key % "SET") + cross_border_demand = omx_manager.lookup(("CrossBorder", period), omx_key % "SET" ) + airport_demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key % "SET") + if omx_manager.file_exists(("Airport", ".CBX" + period)): + airport_demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key % "SET") + + internal_external_demand = omx_manager.lookup(("InternalExternal", period), omx_key % "SET") + + total_ct_ramp_trips = person_demand #for testing only + total_ct_ramp_trips = ( + visitor_demand + cross_border_demand + airport_demand + + person_demand + internal_external_demand) + + # Check the OMX zones are the same Emme database, assume all files have the same zones + omx_zones = omx_manager.zone_list("tranTrips%s.omx" % period) + matrix = emmebank.matrix(matrix_name) + if omx_zones != emme_zones: + matrix_data = _matrix.MatrixData(type='f', indices=[omx_zones, omx_zones]) + matrix_data.from_numpy(total_ct_ramp_trips) + expanded_matrix_data = matrix_data.expand([emme_zones, emme_zones]) + matrix.set_data(expanded_matrix_data, self.scenario) + else: + matrix.set_numpy_data(total_ct_ramp_trips, self.scenario) + + if ("KNR" in matrix_name): + dem_utils.demand_report([ + ("person_demand", person_demand), + (" person_knr_demand", person_knr_demand), + (" person_tnc_Demand", person_tnc_Demand), + ("internal_external_demand", internal_external_demand), + (" internal_external_knr_demand", internal_external_knr_demand), + (" internal_external_tnc_Demand", internal_external_tnc_Demand), + ("cross_border_demand", cross_border_demand), + (" cross_border_knr_demand", cross_border_knr_demand), + (" cross_border_tnc_Demand", cross_border_tnc_Demand), + ("airport_demand", airport_demand), + (" airport_knr_demand", airport_knr_demand), + (" airport_tnc_Demand", airport_tnc_Demand), + ("visitor_demand", visitor_demand), + (" visitor_knr_demand", visitor_knr_demand), + (" visitor_tnc_Demand", visitor_tnc_Demand), + ("total_ct_ramp_trips", total_ct_ramp_trips) + ], logbook_label, self.scenario) + else: + dem_utils.demand_report([ + ("person_demand", person_demand), + ("internal_external_demand", internal_external_demand), + ("cross_border_demand", cross_border_demand), + ("airport_demand", airport_demand), + ("visitor_demand", visitor_demand), + ("total_ct_ramp_trips", total_ct_ramp_trips) + ], logbook_label, self.scenario) diff --git a/sandag_abm/src/main/emme/toolbox/import/input_checker.py b/sandag_abm/src/main/emme/toolbox/import/input_checker.py new file mode 100644 index 0000000..eedffcb --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/input_checker.py @@ -0,0 +1,767 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright RSG, 2019-2020. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/input_checker.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Reviews all inputs to SANDAG ABM for possible issues that will result in model errors +# +# Files referenced: +# input_checker\config\inputs_checks.csv +# input_checker\config\inputs_list.csv + +import os, shutil, sys, time, csv, logging +import win32com.client as com +import numpy as np +import pandas as pd +import traceback as _traceback +import datetime +import warnings +from simpledbf import Dbf5 +import inro.modeller as _m +import inro.emme.database.emmebank as _eb +import inro.director.util.qtdialog as dialog +import textwrap + +warnings.filterwarnings("ignore") + +_join = os.path.join +_dir = os.path.dirname + +class input_checker(_m.Tool()): + + path = _m.Attribute(unicode) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = _dir(_m.Modeller().desktop.project.path) + self.path = _dir(project_dir) + self.input_checker_path = '' + self.inputs_list_path = '' + self.inputs_checks_path = '' + self.log_path = '' + self.logical_log_path = '' + self.prop_input_paths = {} + self.inputs_list = pd.DataFrame() + self.inputs_checks = pd.DataFrame() + self.inputs = {} + self.results = {} + self.result_list = {} + self.problem_ids = {} + self.report_stat = {} + self.num_fatal = int() + self.num_warning = int() + self.num_logical = int() + self.logical_fails = pd.DataFrame() + self.scenario_df = pd.DataFrame() + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Input Checker" + pb.description = """ + Reviews all inputs to SANDAG ABM for possible issues that could result + in model errors. List of inputs and checks are read from two CSV files: +
+
+
    +
  • input_checker\config\inputs_checks.csv
  • +
  • input_checker\config\inputs_list.csv
  • +
+
+ The input checker goes through the list of checks and evaluates each + one as True or False. A summary file is produced at the end with results + for each check. The input checker additionally outputs a report for + failed checks of severity type Logical with more than 25 failed records. + The additional summary report lists every failed record. + The following reports are output: +
+
+
    +
  • input_checker\inputCheckerSummary_[YEAR-MM-DD].txt
  • +
  • completeLogicalFails_[YEAR-MM-DD].txt
  • +
+
+ """ + pb.branding_text = "SANDAG - Input Checker" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self(path = self.path) + run_msg = "Input Checker Complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, path = ""): + _m.logbook_write("Started running input checker...") + + self.path = path + + self.input_checker_path = _join(self.path, 'input_checker') + self.inputs_list_path = _join(self.input_checker_path, 'config', 'inputs_list.csv') + self.inputs_checks_path = _join(self.input_checker_path, 'config', 'inputs_checks.csv') + + file_paths = [self.inputs_list_path, self.inputs_checks_path] + for path in file_paths: + if not os.path.exists(path): + raise Exception("missing file '%s'" % (path)) + + _m.logbook_write("Reading inputs...") + self.read_inputs() + + _m.logbook_write("Conducting checks...") + self.checks() + + _m.logbook_write("Writing logical fail logs...") + self.write_logical_log() + + _m.logbook_write("Writing logs...") + self.write_log() + + _m.logbook_write("Checking for logical errors...") + self.check_logical() + + _m.logbook_write("Checking for fatal errors...") + self.check_num_fatal() + + _m.logbook_write("Finisehd running input checker") + + def read_inputs(self): + # read list of inputs from CSV file + self.inputs_list = pd.read_csv(self.inputs_list_path) + + # remove all commented inputs from the inputs list + self.inputs_list = self.inputs_list.loc[[not i for i in (self.inputs_list['Input_Table'].str.startswith('#'))]] + + # obtain file paths from the sandag_abm.properties + self.prop_file_paths() + + # load emmebank + eb_path = _join(self.path, "emme_project", "Database", "emmebank") + eb = _eb.Emmebank(eb_path) + + # load emme network + network = eb.scenario(100).get_network() + + # create extra network attributes (maybe temporary) + + # link modes_str attribute + network.create_attribute("LINK", "mode_str") + for link in network.links(): + link.mode_str = "".join([m.id for m in link.modes]) + + # link isTransit flag attribute + network.create_attribute("LINK", "isTransit") + transit_modes = set([m for m in network.modes() if m.type == "TRANSIT"]) + for link in network.links(): + link.isTransit = bool(link.modes.intersection(transit_modes)) + + # transit segment isFirst and isLast flags attributes + network.create_attribute("TRANSIT_SEGMENT", "isFirst", False) + network.create_attribute("TRANSIT_SEGMENT", "isLast", False) + for line in network.transit_lines(): + first_seg = line.segment(0) + last_seg = line.segment(-2) + first_seg.isFirst = True + last_seg.isLast = True + + # node isCentroid flag attribute + network.create_attribute("NODE", "isCentroid", False) + centroids = [c for c in network.nodes() if c.is_centroid] + for node in network.nodes(): + node.isCentroid = bool(node in centroids) + + # node numInLinks and numOutLinks attributes + network.create_attribute("NODE", "numInLinks") + network.create_attribute("NODE", "numOutLinks") + for node in network.nodes(): + node.numInLinks = len(list(node.incoming_links())) + node.numOutLinks = len(list(node.outgoing_links())) + + # node hasLocalConnection flag attribute + class BreakLoop (Exception): + pass + + network.create_attribute("NODE", "hasLocalConnection", False) + for node in network.centroids(): + try: + for zone_connector in node.incoming_links(): + for local_link in zone_connector.i_node.incoming_links(): + if local_link["@lane_restriction"] == 1.0: + node.hasLocalConnection = True + raise BreakLoop("") + except: + pass + + # transit line hasTAP flag attribute + network.create_attribute("TRANSIT_LINE", "hasTAP", False) + for line in network.transit_lines(): + has_first_tap = False + has_last_tap = False + for link in line.segment(0).i_node.outgoing_links(): + if link.j_node["@tap_id"] > 0: + has_first_tap = True + break + for link in line.segment(-2).j_node.outgoing_links(): + if link.j_node["@tap_id"] > 0: + has_last_tap = True + break + line.hasTAP = has_first_tap and has_last_tap + + # link names attribute + network.create_attribute("LINK", "linkNames") + for link in network.links(): + link.linkNames = str(link['#name'] + "," + link['#name_from'] + "," + link['#name_to']) + + def get_emme_object(emme_network, emme_network_object, fields_to_export): + # Emme network attribute and object names + net_attr = { + 'NODE':'nodes', + 'LINK':'links', + 'TRANSIT_SEGMENT':'transit_segments', + 'TRANSIT_LINE':'transit_lines', + 'CENTROID':'centroids' + } + + # read-in entire emme network object as a list + get_objs = 'list(emme_network.' + net_attr[emme_network_object] + '())' + uda = eval(get_objs) + + # get list of network object attributes + obj_attr = [] + if fields_to_export[0] in ['all','All','ALL']: + if emme_network_object == 'CENTROID': + obj_attr = emme_network.attributes('NODE') + else: + obj_attr = emme_network.attributes(emme_network_object) + else: + obj_attr = fields_to_export + + # instantiate list of network objects + net_objs = [] + for i in range(len(uda)): + obj_fields = [] + get_id = 'uda[i].id' + obj_fields.append(eval(get_id)) + for attr in obj_attr: + get_field = 'uda[i]["' + attr + '"]' + obj_fields.append(eval(get_field)) + net_objs.append(obj_fields) + net_obj_df = pd.DataFrame(net_objs, columns = ['id'] + obj_attr) + + return(net_obj_df) + + for item, row in self.inputs_list.iterrows(): + + table_name = row['Input_Table'] + emme_network_object = row['Emme_Object'] + column_map = row['Column_Map'] + fields_to_export = row['Fields'].split(',') + + # obtain emme network object, csv or dbf input + if not (pd.isnull(emme_network_object)): + df = get_emme_object(network, emme_network_object, fields_to_export) + self.inputs[table_name] = df + else: + input_path = self.prop_input_paths[table_name] + input_ext = os.path.splitext(input_path)[1] + if input_ext == '.csv': + df = pd.read_csv(_join(self.path, input_path)) + self.inputs[table_name] = df + else: + dbf_path = input_path + if '%project.folder%' in dbf_path: + dbf_path = dbf_path.replace('%project.folder%/', '') + dbf = Dbf5(_join(self.path, dbf_path)) + df = dbf.to_dataframe() + self.inputs[table_name] = df + + # add scenario year + self.inputs['scenario'] = self.scenario_df + + def checks(self): + # read all input DFs into memory + for key, df in self.inputs.items(): + expr = key + ' = df' + exec(expr) + + # copy of locals(), a dictionary of all local variables + calc_dict = locals() + + # read list of checks from CSV file + self.inputs_checks = pd.read_csv(self.inputs_checks_path) + + # remove all commented checks from the checks list + self.inputs_checks = self.inputs_checks.loc[[not i for i in (self.inputs_checks['Test'].str.startswith('#'))]] + + # perform calculations and add user-defined data frame subsets + for item, row in self.inputs_checks.iterrows(): + + test = row['Test'] + table = row['Input_Table'] + id_col = row['Input_ID_Column'] + expr = row['Expression'] + test_vals = row['Test_Vals'] + if not (pd.isnull(row['Test_Vals'])): + test_vals = test_vals.split(',') + test_vals = [txt.strip() for txt in test_vals] + test_type = row['Type'] + Severity = row['Severity'] + stat_expr = row['Report_Statistic'] + + if test_type == 'Calculation': + + try: + calc_expr = test + ' = ' + expr + exec(calc_expr, {}, calc_dict) + calc_out = eval(expr, calc_dict) + except Exception as error: + print('An error occurred with the calculation: {}'.format(test)) + raise + + if str(type(calc_out)) == "": + print('added '+ row['Test'] + ' as new DataFrame input') + self.inputs[row['Test']] = calc_out + self.inputs_list = self.inputs_list.append({'Input_Table': row['Test'],'Property_Token':'NA','Emme_Object':'NA', \ + 'Fields':'NA','Column_Map':'NA','Input_Description':'NA'}, ignore_index = True) + self.inputs_checks = self.inputs_checks.append({'Test':test, 'Input_Table': table, 'Input_ID_Column':id_col, 'Severity':Severity, \ + 'Type':test_type, 'Expression': expr, 'Test_Vals':test_vals, 'Report_Statistic':stat_expr, 'Test_Description': row['Test_Description']}, \ + ignore_index = True) + + # loop through list of checks and conduct all checks + # checks must evaluate to True if inputs are correct + for item, row in self.inputs_checks.iterrows(): + + test = row['Test'] + table = row['Input_Table'] + id_col = row['Input_ID_Column'] + expr = row['Expression'] + test_vals = row['Test_Vals'] + if not (pd.isnull(row['Test_Vals'])): + test_vals = test_vals.split(',') + test_vals = [txt.strip() for txt in test_vals] + test_type = row['Type'] + Severity = row['Severity'] + stat_expr = row['Report_Statistic'] + + if test_type == 'Test': + + if (pd.isnull(row['Test_Vals'])): + + # perform test + try: + out = eval(expr, calc_dict) + except Exception as error: + print('An error occurred with the check: {}'.format(test)) + raise + + # check if test result is a series + if str(type(out)) == "": + # for series, the test must be evaluated across all items + # result is False if a single False is found + self.results[test] = not (False in out.values) + + # reverse results list since we need all False IDs + reverse_results = [not i for i in out.values] + error_expr = table + "['" + id_col + "']" + "[reverse_results]" + error_id_list = eval(error_expr) + + # report first 25 problem IDs in the log + self.problem_ids[test] = error_id_list if error_id_list.size > 0 else [] + + # compute report statistics + if (pd.isnull(stat_expr)): + self.report_stat[test] = '' + else: + stat_list = eval(stat_expr) + self.report_stat[test] = stat_list[reverse_results] + else: + self.results[test] = out + self.problem_ids[test] = [] + if (pd.isnull(stat_expr)): + self.report_stat[test] = '' + else: + self.report_stat[test] = eval(stat_expr) + else: + # loop through test_vals and perform test for each item + self.result_list[test] = [] + for test_val in test_vals: + # perform test (test result must not be of type Series) + try: + out = eval(expr) + except Exception as error: + print('An error occurred with the check: {}'.format(test)) + raise + + # compute report statistic + if (pd.isnull(stat_expr)): + self.report_stat[test] = '' + else: + self.report_stat[test] = eval(stat_expr) + + # append to list + self.result_list[test].append(out) + self.results[test] = not (False in self.result_list[test]) + self.problem_ids[test] = [] + else: + # perform calculation + try: + calc_expr = test + ' = ' + expr + exec(calc_expr, {}, calc_dict) + except Exception as error: + print('An error occurred with the calculation: {}'.format(test)) + raise + + def prop_file_paths(self): + prop_files = self.inputs_list[['Input_Table','Property_Token']].dropna() + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(self.path, 'conf', 'sandag_abm.properties')) + + for item, row in prop_files.iterrows(): + input_table = row['Input_Table'] + input_path = props[row['Property_Token']] + self.prop_input_paths[input_table] = input_path + + # obtain scenario year + self.scenario_df['Year'] = [props['scenarioYear']] + + def write_log(self): + # function to write out the input checker log file + # there are four blocks + # - Introduction + # - Summary of checks + # - Action Required: FATAL, LOGICAL, WARNINGS + # - List of passed checks + + # create log file + now = datetime.datetime.now() + + self.log_path = _join(self.input_checker_path, ('inputCheckerSummary_' + now.strftime("[%Y-%m-%d]") + '.txt')) + f = open(self.log_path, 'wb') + + # define re-usable elements + seperator1 = '###########################################################' + seperator2 = '***********************************************************' + + # write out Header + f.write(seperator1 + seperator1 + "\r\n") + f.write(seperator1 + seperator1 + "\r\n\r\n") + f.write("\t SANDAG ABM Input Checker Summary File \r\n") + f.write("\t _____________________________________ \r\n\r\n\r\n") + f.write("\t Created on: " + now.strftime("%Y-%m-%d %H:%M") + "\r\n\r\n") + f.write("\t Notes:-\r\n") + f.write("\t The SANDAG ABM Input Checker performs various QA/QC checks on SANDAG ABM inputs as specified by the user.\r\n") + f.write("\t The Input Checker allows the user to specify three severity levels for each QA/QC check:\r\n\r\n") + f.write("\t 1) FATAL 2) LOGICAL 3) WARNING\r\n\r\n") + f.write("\t FATAL Checks: The failure of these checks would result in a FATAL errors in the SANDAG ABM run.\r\n") + f.write("\t In case of FATAL failure, the Input Checker returns a return code of 1 to the\r\n") + f.write("\t main SANDAG ABM model, cauing the model run to halt.\r\n") + f.write("\t LOGICAL Checks: The failure of these checks indicate logical inconsistencies in the inputs.\r\n") + f.write("\t With logical errors in inputs, the SANDAG ABM outputs may not be meaningful.\r\n") + f.write("\t WARNING Checks: The failure of Warning checks would indicate problems in data that would not.\r\n") + f.write("\t halt the run or affect model outputs but might indicate an issue with inputs.\r\n\r\n\r\n") + f.write("\t The contents of this summary file are organized as follows: \r\n\r\n") + f.write("\t TALLY OF FAILED CHECKS:\r\n") + f.write("\t -----------------------\r\n") + f.write("\t A tally of all failed checks per severity level\r\n\r\n") + f.write("\t IMMEDIATE ACTION REQUIRED:\r\n") + f.write("\t -------------------------\r\n") + f.write("\t A log under this heading will be generated in case of failure of a FATAL check\r\n\r\n") + f.write("\t ACTION REQUIRED:\r\n") + f.write("\t ---------------\r\n") + f.write("\t A log under this heading will be generated in case of failure of a LOGICAL check\r\n\r\n") + f.write("\t WARNINGS:\r\n") + f.write("\t ---------\r\n") + f.write("\t A log under this heading will be generated in case of failure of a WARNING check\r\n\r\n") + f.write("\t SUMMARY OF ALL PASSED CHECKS:\r\n") + f.write("\t ----------------------------\r\n") + f.write("\t A complete listing of results of all passed checks\r\n\r\n") + f.write(seperator1 + seperator1 + "\r\n") + f.write(seperator1 + seperator1 + "\r\n\r\n\r\n\r\n") + + # combine results, inputs_checks and inputs_list + self.inputs_checks['result'] = self.inputs_checks['Test'].map(self.results) + checks_df = pd.merge(self.inputs_checks, self.inputs_list, on='Input_Table') + checks_df = checks_df[checks_df.Type=='Test'] + checks_df['reverse_result'] = [not i for i in checks_df.result] + + # get count of all FATAL failures + self.num_fatal = checks_df.result[(checks_df.Severity=='Fatal') & (checks_df.reverse_result)].count() + + # get count of all LOGICAL failures + self.num_logical = checks_df.result[(checks_df.Severity=='Logical') & (checks_df.reverse_result)].count() + self.logical_fails = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] + + # get count of all WARNING failures + self.num_warning = checks_df.result[(checks_df.Severity=='Warning') & (checks_df.reverse_result)].count() + + # write summary of failed checks + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "TALLY OF FAILED CHECKS \r\n") + f.write('\t' + "---------------------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n\t") + f.write(' Number of Fatal Errors: ' + str(self.num_fatal)) + f.write('\r\n\t Number of Logical Errors: ' + str(self.num_logical)) + f.write('\r\n\t Number of Warnings: ' + str(self.num_warning)) + + def write_check_log(self, fh, row): + # define constants + seperator2 = '-----------------------------------------------------------' + + # integerize problem ID list + problem_ids = self.problem_ids[row['Test']] + #problem_ids = [int(x) for x in problem_ids] + + # write check summary + fh.write('\r\n\r\n' + seperator2 + seperator2) + fh.write("\r\n\t Input File Name: " + ('NA' if not pd.isnull(row['Emme_Object']) else + (self.prop_input_paths[row['Input_Table']].rsplit('/', 1)[-1]))) + fh.write("\r\n\t Input File Location: " + ('NA' if not pd.isnull(row['Emme_Object']) else + (_join(self.input_checker_path, self.prop_input_paths[row['Input_Table']].replace('/','\\'))))) + fh.write("\r\n\t Emme Object: " + (row['Emme_Object'] if not pd.isnull(row['Emme_Object']) else 'NA')) + fh.write("\r\n\t Input Description: " + (row['Input_Description'] if not pd.isnull(row['Input_Description']) else "")) + fh.write("\r\n\t Test Name: " + row['Test']) + fh.write("\r\n\t Test_Description: " + (row['Test_Description'] if not pd.isnull(row['Test_Description']) else "")) + fh.write("\r\n\t Test Severity: " + row['Severity']) + fh.write("\r\n\r\n\t TEST RESULT: " + ('PASSED' if row['result'] else 'FAILED')) + + # display problem IDs for failed column checks + wrapper = textwrap.TextWrapper(width = 70) + if (not row['result']) & (len(problem_ids)>0) : + fh.write("\r\n\t TEST failed for following values of ID Column: " + row['Input_ID_Column'] + " (only up to 25 IDs displayed)") + fh.write("\r\n\t " + row['Input_ID_Column'] + ": " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, problem_ids[0:25]))))) + if not (pd.isnull(row['Report_Statistic'])): + this_report_stat = self.report_stat[row['Test']] + fh.write("\r\n\t Test Statistics: " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, this_report_stat[0:25]))))) + fh.write("\r\n\t Total number of failures: " + str(len(self.problem_ids[row['Test']]))) + if ((len(self.problem_ids[row['Test']])) > 25) and (row['Severity'] == 'Logical'): + fh.write("\r\n\t Open {} for complete list of failed Logical failures.".format(self.logical_log_path)) + else: + if not (pd.isnull(row['Report_Statistic'])): + fh.write("\r\n\t Test Statistic: " + str(self.report_stat[row['Test']])) + + # display result for each test val if it was specified + if not (pd.isnull(row['Test_Vals'])): + fh.write("\r\n\t TEST results for each test val") + result_tuples = zip(row['Test_Vals'].split(","), self.result_list[row['Test']]) + fh.write("\r\n\t ") + fh.write(','.join('[{} - {}]'.format(x[0],x[1]) for x in result_tuples)) + + fh.write("\r\n" + seperator2 + seperator2 + "\r\n\r\n") + + # write out IMMEDIATE ACTION REQUIRED section if needed + if self.num_fatal > 0: + fatal_checks = checks_df[(checks_df.Severity=='Fatal') & (checks_df.reverse_result)] + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "IMMEDIATE ACTION REQUIRED \r\n") + f.write('\t' + "------------------------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n") + + # write out log for each check + for item, row in fatal_checks.iterrows(): + #self.write_check_log(f, row, self.problem_ids[row['Test']]) + #write_check_log(self, f, row, self.problem_ids[row['Test']]) + write_check_log(self, f, row) + + # write out ACTION REQUIRED section if needed + if self.num_logical > 0: + logical_checks = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "ACTION REQUIRED \r\n") + f.write('\t' + "--------------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n") + + #write out log for each check + for item, row in logical_checks.iterrows(): + write_check_log(self, f, row) + + # write out WARNINGS section if needed + if self.num_warning > 0: + warning_checks = checks_df[(checks_df.Severity=='Warning') & (checks_df.reverse_result)] + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "WARNINGS \r\n") + f.write('\t' + "-------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n") + + # write out log for each check + for item, row in warning_checks.iterrows(): + write_check_log(self, f, row) + + # write out the complete listing of all checks that passed + passed_checks = checks_df[(checks_df.result)] + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "LOG OF ALL PASSED CHECKS \r\n") + f.write('\t' + "------------------------ \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n") + + # write out log for each check + for item, row in passed_checks.iterrows(): + write_check_log(self, f, row) + + f.close() + + def write_logical_log(self): + # function to write out the complete list of Logical failures + + # combine results, inputs_checks and inputs_list + self.inputs_checks['result'] = self.inputs_checks['Test'].map(self.results) + checks_df = pd.merge(self.inputs_checks, self.inputs_list, on='Input_Table') + checks_df = checks_df[checks_df.Type=='Test'] + checks_df['reverse_result'] = [not i for i in checks_df.result] + + # get count of all LOGICAL failures + self.num_logical = checks_df.result[(checks_df.Severity=='Logical') & (checks_df.reverse_result)].count() + self.logical_fails = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] + + log_fail_id_tally = 0 + if self.num_logical > 0: + for item, row in self.logical_fails.iterrows(): + problem_ids = self.problem_ids[row['Test']] + if len(problem_ids) > 0: + log_fail_id_tally += 1 + + if log_fail_id_tally > 0: + + # create log file + now = datetime.datetime.now() + + self.logical_log_path = _join(self.input_checker_path, ('completeLogicalFails_' + now.strftime("[%Y-%m-%d]") + '.txt')) + f = open(self.logical_log_path, 'wb') + + # define re-usable elements + seperator1 = '###########################################################' + seperator2 = '***********************************************************' + + # write out Header + f.write(seperator1 + seperator1 + "\r\n") + f.write(seperator1 + seperator1 + "\r\n\r\n") + f.write("\t SANDAG ABM Input Checker Logical Failures Complete List \r\n") + f.write("\t _______________________________________________________ \r\n\r\n\r\n") + f.write("\t Created on: " + now.strftime("%Y-%m-%d %H:%M") + "\r\n\r\n") + f.write("\t Notes:-\r\n") + f.write("\t The SANDAG ABM Input Checker performs various QA/QC checks on SANDAG ABM inputs as specified by the user.\r\n") + f.write("\t The Input Checker allows the user to specify three severity levels for each QA/QC check:\r\n\r\n") + f.write("\t 1) FATAL 2) LOGICAL 3) WARNING\r\n\r\n") + f.write("\t This file provides the complete list of failed checks for checks of severity type Logical. \r\n") + f.write(seperator1 + seperator1 + "\r\n") + f.write(seperator1 + seperator1 + "\r\n\r\n\r\n\r\n") + + # write total number of failed logical checks + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "TALLY OF FAILED CHECKS \r\n") + f.write('\t' + "---------------------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n\t") + f.write('\r\n\t Number of Logical Errors: ' + str(self.num_logical)) + + def write_logical_check_log(self, fh, row): + # define constants + seperator2 = '-----------------------------------------------------------' + + # integerize problem ID list + problem_ids = self.problem_ids[row['Test']] + #problem_ids = [int(x) for x in problem_ids] + + # write check summary + fh.write('\r\n\r\n' + seperator2 + seperator2) + fh.write("\r\n\t Input File Name: " + ('NA' if not pd.isnull(row['Emme_Object']) else + (self.prop_input_paths[row['Input_Table']].rsplit('/', 1)[-1]))) + fh.write("\r\n\t Input File Location: " + ('NA' if not pd.isnull(row['Emme_Object']) else + (_join(self.input_checker_path, self.prop_input_paths[row['Input_Table']].replace('/','\\'))))) + fh.write("\r\n\t Emme Object: " + (row['Emme_Object'] if not pd.isnull(row['Emme_Object']) else 'NA')) + fh.write("\r\n\t Input Description: " + (row['Input_Description'] if not pd.isnull(row['Input_Description']) else "")) + fh.write("\r\n\t Test Name: " + row['Test']) + fh.write("\r\n\t Test_Description: " + (row['Test_Description'] if not pd.isnull(row['Test_Description']) else "")) + fh.write("\r\n\t Test Severity: " + row['Severity']) + fh.write("\r\n\r\n\t TEST RESULT: " + ('PASSED' if row['result'] else 'FAILED')) + + # display problem IDs for failed column checks + wrapper = textwrap.TextWrapper(width = 70) + if (not row['result']) & (len(problem_ids)>0) : + fh.write("\r\n\t TEST failed for following values of ID Column: " + row['Input_ID_Column']) + fh.write("\r\n\t " + row['Input_ID_Column'] + ": " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, problem_ids))))) + if not (pd.isnull(row['Report_Statistic'])): + this_report_stat = self.report_stat[row['Test']] + fh.write("\r\n\t Test Statistics: " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, this_report_stat))))) + fh.write("\r\n\t Total number of failures: " + str(len(self.problem_ids[row['Test']]))) + else: + if not (pd.isnull(row['Report_Statistic'])): + fh.write("\r\n\t Test Statistic: " + str(self.report_stat[row['Test']])) + + # display result for each test val if it was specified + if not (pd.isnull(row['Test_Vals'])): + fh.write("\r\n\t TEST results for each test val") + result_tuples = zip(row['Test_Vals'].split(","), self.result_list[row['Test']]) + fh.write("\r\n\t ") + fh.write(','.join('[{} - {}]'.format(x[0],x[1]) for x in result_tuples)) + + fh.write("\r\n" + seperator2 + seperator2 + "\r\n\r\n") + + # write out ACTION REQUIRED section if needed + if self.num_logical > 0: + logical_checks = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] + f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n\r\n") + f.write('\t' + "LOG OF ALL FAILED LOGICAL CHECKS \r\n") + f.write('\t' + "-------------------------------- \r\n\r\n") + f.write(seperator2 + seperator2 + "\r\n") + f.write(seperator2 + seperator2 + "\r\n") + + #write out log for each check + for item, row in logical_checks.iterrows(): + if len(self.problem_ids[row['Test']]) > 25: + write_logical_check_log(self, f, row) + + f.close() + + def check_logical(self): + if self.num_logical > 0: + # raise exception for each logical check fail + for item, row in self.logical_fails.iterrows(): + answer = dialog.alert_question( + message = "The following Logical check resulted in at least 1 error: {} \n Open {} for details. \ + \n\n Click OK to continue or Cancel to stop run.".format(row['Test'], self.log_path), + title = "Logical Check Error", + answers = [("OK", dialog.YES_ROLE), ("Cancel", dialog.REJECT_ROLE)] + ) + + if answer == 1: + raise Exception("Input checker was cancelled") + + def check_num_fatal(self): + # return code to the main model based on input checks and results + if self.num_fatal > 0: + raise Exception("Input checker failed, {} fatal errors found. Open {} for details.".format(self.num_fatal, self.log_path)) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/import/run4Ds.py b/sandag_abm/src/main/emme/toolbox/import/run4Ds.py new file mode 100644 index 0000000..89ff0d0 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/run4Ds.py @@ -0,0 +1,412 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright RSG, 2019-2020. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/run4Ds.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Generates density variables and adds in mgra socio economic variables +# +# +# Inputs: +# path: path to the current scenario +# ref_path: path to the comparison model scenario +# int_radius: buffer radius for intersection counts +# maps: default unchecked - means not generating spatial heat maps for +# intersection counts. This functionality requires +# following packages; geopandas, folium, and branca +# +# File referenced: +# input\mgra13_based_input2016.csv +# input\SANDAG_Bike_Net.dbf +# input\SANDAG_Bike_Node.dbf +# output\walkMgraEquivMinutes.csv +# +# Script example +# python C:\ABM_runs\maint_2019_RSG\Tasks\4ds\emme_toolbox\emme\toolbox\import\run4Ds.py +# 0.65 r'C:\ABM_runs\maint_2019_RSG\Model\ABM2_14_2_0' r'C:\ABM_runs\maint_2019_RSG\Model\abm_test_fortran_4d' + + +TOOLBOX_ORDER = 10 + +#import modules +import inro.modeller as _m +from simpledbf import Dbf5 +import os +import pandas as pd, numpy as np +#import datetime +import matplotlib.pyplot as plt +import seaborn as sns +import warnings +import traceback as _traceback + +warnings.filterwarnings("ignore") + +_join = os.path.join +_dir = os.path.dirname + +gen_utils = _m.Modeller().module("sandag.utilities.general") + +class FourDs(_m.Tool()): + + path = _m.Attribute(unicode) + ref_path = _m.Attribute(unicode) + int_radius = _m.Attribute(float) + maps = _m.Attribute(bool) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self._log = [] + self._error = [] + project_dir = _dir(_m.Modeller().desktop.project.path) + self.path = _dir(project_dir) + self.mgradata_file = '' + self.equivmins_file = '' + self.inNet = '' + self.inNode = '' + self.ref_path = '' + self.maps = False + self.int_radius = 0.65 #mile + self.oth_radius = self.int_radius #same as intersection radius + self.new_cols = ['totint','duden','empden','popden','retempden','totintbin','empdenbin','dudenbin','PopEmpDenPerMi'] + self.continuous_fields = ['totint', 'popden', 'empden', 'retempden'] + self.discrete_fields = ['totintbin', 'empdenbin', 'dudenbin'] + self.mgra_shape_file = '' + self.base = pd.DataFrame() + self.build = pd.DataFrame() + self.mgra_data = pd.DataFrame() + self.base_cols = [] + self.attributes = ["path", "int_radius", "ref_path"] + + def page(self): + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(self.path, "conf", "sandag_abm.properties")) + self.ref_path = props["visualizer.reference.path"] + + pb = _m.ToolPageBuilder(self) + pb.title = "Run 4Ds" + pb.description = """ + Generate Density Variables. + Generated from MGRA socio economic file and active transportation (AT) network. +
+
+ The following files are used: +
+
    +
  • input\mgra13_based_input2016.csv
  • +
  • input\SANDAG_Bike_Net.dbf
  • +
  • input\SANDAG_Bike_Node.dbf
  • +
  • output\walkMgraEquivMinutes.csv
  • +
+
+ """ + pb.branding_text = "SANDAG - Run 4Ds" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file("path", window_type="directory", file_filter="", + title="Source directory:",) + + pb.add_text_box("int_radius", size=6, title="Buffer size (miles):") + #pb.add_checkbox("maps", title=" ", label="Generate 4D maps") + pb.add_select_file("ref_path", window_type="directory", file_filter="", title="Reference directory for comparison") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self(path=self.path, int_radius=self.int_radius, ref_path=self.ref_path) + run_msg = "Run 4Ds complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, path= "", + int_radius = 0.65, + ref_path = ""): + _m.logbook_write("Started running 4Ds ...") + + self.path = path + self.ref_path = ref_path + self.int_radius = int_radius + #self.maps = maps + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(self.path, "conf", "sandag_abm.properties")) + + + self.mgradata_file = props["mgra.socec.file"] #input/filename + self.syn_households_file = props["PopulationSynthesizer.InputToCTRAMP.HouseholdFile"] #input/filename + self.equivmins_file = props["active.logsum.matrix.file.walk.mgra"] #filename + self.inNet = os.path.basename(props["active.edge.file"]) #filename + self.inNode = os.path.basename(props["active.node.file"]) #filename + + attributes = { + "path": self.path, + "ref_path": self.ref_path, + "int_radius": self.int_radius, + "maps": self.maps, + } + gen_utils.log_snapshot("Run 4Ds", str(self), attributes) + + file_paths = [_join(self.path, self.mgradata_file),_join(self.path, self.syn_households_file),_join(self.path, "output", self.equivmins_file), _join(self.path, "input", self.inNet), _join(self.path, "input", self.inNode)] + for path in file_paths: + if not os.path.exists(path): + raise Exception("missing file '%s'" % (path)) + + self.mgra_data = pd.read_csv(os.path.join(self.path,self.mgradata_file)) + self.base_cols = self.mgra_data.columns.tolist() + + _m.logbook_write("Tagging intersections to mgra") + self.get_intersection_count() + + _m.logbook_write("Generating density variables") + self.get_density() + + _m.logbook_write("Creating comparison plots") + self.make_plots() + + _m.logbook_write("Finished running 4Ds") + + def get_intersection_count(self): + links = Dbf5(_join(self.path, "input", self.inNet)) + links = links.to_dataframe() + + nodes = Dbf5(_join(self.path, "input", self.inNode)) + nodes = nodes.to_dataframe() + + nodes_int = nodes.loc[(nodes.NodeLev_ID < 100000000)] + + #links + #remove taz, mgra, and tap connectors + links = links.loc[(links.A <100000000) & (links.B <100000000)] + + #remove freeways (Func_Class=1), ramps (Func_Class=2), and others (Func_Class =0 or -1) + links = links.loc[(links.Func_Class > 2)] + links['link_count'] = 1 + + #aggregate by Node A and Node B + links_nodeA = links[['A', 'link_count']].groupby('A').sum().reset_index() + links_nodeB = links[['B', 'link_count']].groupby('B').sum().reset_index() + + #merge the two and keep all records from both dataframes (how='outer') + nodes_linkcount = pd.merge(links_nodeA, links_nodeB, left_on='A', right_on='B', how = 'outer') + nodes_linkcount = nodes_linkcount.fillna(0) + nodes_linkcount['link_count'] = nodes_linkcount['link_count_x'] + nodes_linkcount['link_count_y'] + + #get node id from both dataframes + nodes_linkcount['N']=0 + nodes_linkcount['N'][nodes_linkcount.A>0] = nodes_linkcount['A'] + nodes_linkcount['N'][nodes_linkcount.B>0] = nodes_linkcount['B'] + nodes_linkcount['N']=nodes_linkcount['N'].astype(float) + nodes_linkcount = nodes_linkcount[['N','link_count']] + + #keep nodes with 3+ link count + intersections_temp = nodes_linkcount.loc[nodes_linkcount.link_count>=3] + + #get node X and Y + intersections = pd.merge(intersections_temp,nodes_int[['NodeLev_ID','XCOORD','YCOORD']], left_on = 'N', right_on = 'NodeLev_ID', how = 'left') + intersections = intersections[['N','XCOORD','YCOORD']] + intersections = intersections.rename(columns = {'XCOORD': 'X', 'YCOORD': 'Y'}) + + mgra_nodes = nodes[nodes.MGRA > 0][['MGRA','XCOORD','YCOORD']] + mgra_nodes.columns = ['mgra','x','y'] + int_dict = {} + for int in intersections.iterrows(): + mgra_nodes['dist'] = np.sqrt((int[1][1] - mgra_nodes['x'])**2+(int[1][2] - mgra_nodes['y'])**2) + int_dict[int[1][0]] = mgra_nodes.loc[mgra_nodes['dist'] == mgra_nodes['dist'].min()]['mgra'].values[0] + + intersections['near_mgra'] = intersections['N'].map(int_dict) + intersections = intersections.groupby('near_mgra', as_index = False).count()[['near_mgra','N']].rename(columns = {'near_mgra':'mgra','N':'icnt'}) + try: + self.mgra_data = self.mgra_data.drop('icnt',axis = 1).merge(intersections, how = 'outer', on = "mgra") + except: + self.mgra_data = self.mgra_data.merge(intersections, how = 'outer', on = "mgra") + + def get_density(self): + if len(self.mgra_data) == 0: + mgra_landuse = pd.read_csv(os.path.join(self.path, self.mgradata_file)) + else: + mgra_landuse = self.mgra_data + + # get population from synthetic population instead of mgra data file + syn_pop = pd.read_csv(os.path.join(self.path, self.syn_households_file)) + syn_pop = syn_pop.rename(columns = {'MGRA':'mgra'})[['persons','mgra']].groupby('mgra',as_index = False).sum() + #remove if 4D columns exist + for col in self.new_cols: + if col in self.base_cols: + self.base_cols.remove(col) + mgra_landuse = mgra_landuse.drop(col,axis=1) + + #merge syntetic population to landuse + mgra_landuse = mgra_landuse.merge(syn_pop, how = 'left', on = 'mgra') + #all street distance + equiv_min = pd.read_csv(_join(self.path, "output", self.equivmins_file)) + equiv_min['dist'] = equiv_min['actual']/60*3 + print("MGRA input landuse: " + self.mgradata_file) + + def density_function(mgra_in): + eqmn = equiv_min[equiv_min['i'] == mgra_in] + mgra_circa_int = eqmn[eqmn['dist'] < self.int_radius]['j'].unique() + mgra_circa_oth = eqmn[eqmn['dist'] < self.oth_radius]['j'].unique() + totEmp = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_total'].sum() + totRet = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_retail'].sum() + mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_personal_svcs_retail'].sum() + mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_restaurant_bar'].sum() + totHH = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['hh'].sum() + totPop = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['persons'].sum() + totAcres = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['land_acres'].sum() + totInt = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_int)]['icnt'].sum() + if(totAcres>0): + empDen = totEmp/totAcres + retDen = totRet/totAcres + duDen = totHH/totAcres + popDen = totPop/totAcres + popEmpDenPerMi = (totEmp+totPop)/(totAcres/640) #Acres to miles + tot_icnt = totInt + else: + empDen = 0 + retDen = 0 + duDen = 0 + popDen = 0 + popEmpDenPerMi = 0 + tot_icnt = 0 + return tot_icnt,duDen,empDen,popDen,retDen,popEmpDenPerMi + + #new_cols = [0-'totint',1-'duden',2-'empden',3-'popden',4-'retempden',5-'totintbin',6-'empdenbin',7-'dudenbin',8-'PopEmpDenPerMi'] + mgra_landuse[self.new_cols[0]],mgra_landuse[self.new_cols[1]],mgra_landuse[self.new_cols[2]],mgra_landuse[self.new_cols[3]],mgra_landuse[self.new_cols[4]],mgra_landuse[self.new_cols[8]] = zip(*mgra_landuse['mgra'].map(density_function)) + + mgra_landuse = mgra_landuse.fillna(0) + mgra_landuse[self.new_cols[5]] = np.where(mgra_landuse[self.new_cols[0]] < 80, 1, np.where(mgra_landuse[self.new_cols[0]] < 130, 2, 3)) + mgra_landuse[self.new_cols[6]] = np.where(mgra_landuse[self.new_cols[2]] < 10, 1, np.where(mgra_landuse[self.new_cols[2]] < 30, 2,3)) + mgra_landuse[self.new_cols[7]] = np.where(mgra_landuse[self.new_cols[1]] < 5, 1, np.where(mgra_landuse[self.new_cols[1]] < 10, 2,3)) + + mgra_landuse[self.base_cols+self.new_cols].to_csv(os.path.join(self.path, self.mgradata_file), index = False, float_format='%.4f' ) + + self.mgra_data = mgra_landuse + print( "*** Finished ***") + + #plot comparisons of build and old density values and create heat maps + def make_plots(self): + if len(self.mgra_data) == 0: + self.build = pd.read_csv(os.path.join(self.path, self.mgradata_file)) + else: + self.build = self.mgra_data + + def plot_continuous(field): + #colors + rsg_orange = '#f68b1f' + rsg_marine = '#006fa1' + #rsg_leaf = '#63af5e' + #rsg_grey = '#48484a' + #rsg_mist = '#dcddde' + + max = self.base[field].max() + self.base[field].max()%5 + div = max/5 if max/5 >= 10 else max/2 + bins = np.linspace(0,max,div) + plt.hist(self.base[field], bins, normed = True, alpha = 0.5, label = 'Base', color = rsg_marine) + plt.hist(self.build[field], bins, normed = True, alpha = 0.5, label = 'Build', color = rsg_orange) + mean_base = self.base[field].mean() + mean = self.build[field].mean() + median_base = self.base[field].median() + median = self.build[field].median() + plt.axvline(mean_base, color = 'b', linestyle = '-', label = 'Base Mean') + plt.axvline(median_base, color = 'b', linestyle = '--', label = 'Base Median') + plt.axvline(mean, color = 'r', linestyle = '-', label = 'Build Mean') + plt.axvline(median, color = 'r', linestyle = '--',label = 'Build Median') + plt.legend(loc = 'upper right') + ylims = plt.ylim()[1] + plt.text(mean_base + div/4, ylims-ylims/32, "mean: {:0.2f}".format(mean_base), color = 'b') + plt.text(mean_base + div/4, ylims - 5*ylims/32, "median: {:0.0f}".format(median_base), color = 'b') + plt.text(mean_base + div/4, ylims-2*ylims/32, "mean: {:0.2f}".format(mean), size = 'medium',color = 'r') + plt.text(mean_base + div/4, ylims-6*ylims/32, "median: {:0.0f}".format(median), color = 'r') + plt.text(self.base[field].min() , ylims/32, "min: {:0.0f}".format(self.base[field].min()), color = 'b') + plt.text(self.base[field].max()-div , ylims/32, "max: {:0.0f}".format(self.base[field].max()), color = 'b') + plt.text(self.build[field].min() , 2*ylims/32, "min: {:0.0f}".format(self.build[field].min()), color = 'r') + plt.text(self.base[field].max()-div , 2*ylims/32, "max: {:0.0f}".format(self.build[field].max()), color = 'r') + + plt.xlabel(field) + plt.ylabel("MGRA's") + plt.title(field.replace('den','') + ' Density') + outfile = _join(self.path, "output", '4Ds_{}_plot.png'.format(field)) + if os.path.isfile(outfile): + os.remove(outfile) + plt.savefig(outfile) + plt.clf() + + def plot_discrete(field): + fig, ax = plt.subplots() + df1 = discretedf_base.groupby(field, as_index = False).agg({'mgra':'count','type':'first'}) + df2 = discretedf_build.groupby(field, as_index = False).agg({'mgra':'count','type':'first'}) + df = df1.append(df2) + ax = sns.barplot(x=field, y = 'mgra', hue = 'type', data = df) + ax.set_title(field) + outfile = _join(self.path, "output", '4Ds_{}_plot.png'.format(field)) + if os.path.isfile(outfile): + os.remove(outfile) + ax.get_figure().savefig(outfile) + + self.base = pd.read_csv(self.ref_path) + self.base['type'] = 'base' + self.build['type'] = 'build' + + discretedf_base = self.base[['mgra','type']+self.discrete_fields] + discretedf_build = self.build[['mgra','type']+self.discrete_fields] + + for f in self.continuous_fields: + plot_continuous(f) + for f in self.discrete_fields: + plot_discrete(f) + + if self.maps: + import geopandas as gpd + import folium + from branca.colormap import linear + compare_int = self.base.merge(self.build, how = 'outer', on = 'mgra', suffixes = ['_base','_build']) + compare_int['diff'] = compare_int['TotInt'] - compare_int['totint'] + + compare_int = gpd.read_file(self.mgra_shape_file).rename(columns = {'MGRA':'mgra'}).merge(compare_int, how = 'left', on = 'mgra') + compare_int = compare_int.to_crs({'init': 'epsg:4326'}) + + colormap = linear.OrRd_09.scale( + compare_int.TotInt.min(), + compare_int.TotInt.max()) + colormapA = linear.RdBu_04.scale( + compare_int['diff'].min(), + compare_int['diff'].min()*-1) + + compare_int['colordiff'] = compare_int['diff'].map(lambda n: colormapA(n)) + compare_int['colororig'] = compare_int['TotInt'].map(lambda n: colormap(n)) + compare_int['colornew'] = compare_int['totint'].map(lambda n: colormap(n)) + + def makeheatmap(self,df, colormp,color_field,caption): + mapname = folium.Map(location=[32.76, -117.15], zoom_start = 13.459) + folium.GeoJson(compare_int, + style_function=lambda feature: { + 'fillColor': feature['properties'][color_field], + 'color' : rsg_marine, + 'weight' : 0, + 'fillOpacity' : 0.75, + }).add_to(mapname) + + colormp.caption = caption + colormp.add_to(mapname) + return mapname + + makeheatmap(compare_int,colormapA,'colordiff','Intersection Diff (base - build)').save('diff_intersections.html') + makeheatmap(compare_int,colormap,'colororig','Intersections').save('base_intersections.html') + makeheatmap(compare_int,colormap,'colororig','Intersections').save('build_intersections.html') diff --git a/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py b/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py new file mode 100644 index 0000000..f4e880a --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py @@ -0,0 +1,426 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// initialize_matrices.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Coordinates the initialization of all matrices. +# The matrix names are listed for each of the model components / steps, +# and the matrix IDs are assigned consistently from the set of matrices. +# In each of the model steps the matrices are only referenced by name, +# never by ID. +# +# +# Inputs: +# components: A list of the model components / steps for which to initialize matrices +# One or more of "traffic_demand", "transit_demand", +# "traffic_skims", "transit_skims", "external_internal_model", +# "external_external_model", "truck_model", "commercial_vehicle_model" +# periods: A list of periods for which to initialize matrices, "EA", "AM", "MD", "PM", "EV" +# scenario: scenario to use for reference zone system and the emmebank in which +# the matrices will be created +# +# Script example: +""" + import os + import inro.emme.database.emmebank as _eb + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + periods = ["EA", "AM", "MD", "PM", "EV"] + traffic_components = [ + "traffic_demand", "traffic_skims", "external_internal_model", + "external_external_model", "truck_model", "commercial_vehicle_model"] + transit_components = ["transit_demand", "transit_skims"] + base_scenario = main_emmebank.scenario(100) + transit_scenario = transit_emmebank.scenario(100) + initialize_matrices = modeller.tool("sandag.initialize.initialize_matrices") + # Create / initialize matrices in the base, traffic emmebank + initialize_matrices(traffic_components, periods, base_scenario) + # Create / initialize matrices in the transit emmebank + initialize_matrices(transit_components, periods, transit_scenario) +""" + + +TOOLBOX_ORDER = 9 + + +import inro.modeller as _m +import traceback as _traceback + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class Initialize(_m.Tool(), gen_utils.Snapshot): + + components = _m.Attribute(_m.ListType) + periods = _m.Attribute(_m.ListType) + delete_all_existing = _m.Attribute(bool) + + tool_run_msg = "" + + def __init__(self): + self._all_components = [ + "traffic_demand", + "transit_demand", + "traffic_skims", + "transit_skims", + "external_internal_model", + "external_external_model", + "truck_model", + "commercial_vehicle_model", + ] + self._all_periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + self.components = self._all_components[:] + self.periods = self._all_periods[:] + self.attributes = ["components", "periods", "delete_all_existing"] + self._matrices = {} + self._count = {} + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Initialize matrices" + pb.description = """Creates and initializes the required matrices + for the selected components / sub-models. + Includes all components by default.""" + pb.branding_text = "- SANDAG" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select("components", keyvalues=[(k,k) for k in self._all_components], + title="Select components:") + pb.add_select("periods", keyvalues=[(k,k) for k in self._all_periods], + title="Select periods:") + pb.add_checkbox("delete_all_existing", label="Delete all existing matrices") + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.components, self.periods, scenario, self.delete_all_existing) + run_msg = "Tool completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace("Create and initialize matrices", save_arguments=True) + def __call__(self, components, periods, scenario, delete_all_existing=False): + attributes = { + "components": components, + "periods": periods, + "delete_all_existing": delete_all_existing + } + gen_utils.log_snapshot("Initialize matrices", str(self), attributes) + + self.scenario = scenario + emmebank = scenario.emmebank + self._create_matrix_tool = _m.Modeller().tool( + "inro.emme.data.matrix.create_matrix") + if components == "all": + components = self._all_components[:] + if periods == "all": + periods = self._all_periods[:] + if delete_all_existing: + with _m.logbook_trace("Delete all existing matrices"): + for matrix in emmebank.matrices(): + emmebank.delete_matrix(matrix) + self.generate_matrix_list(self.scenario) + matrices = [] + for component in components: + matrices.extend(self.create_matrices(component, periods)) + # Note: matrix is also created in import_network + self._create_matrix_tool("ms1", "zero", "zero", scenario=self.scenario, overwrite=True) + return matrices + + def generate_matrix_list(self, scenario): + self._matrices = dict( + (name, dict((k, []) for k in self._all_periods + ["ALL"])) + for name in self._all_components) + self._count = {"ms": 2, "md": 100, "mo": 100, "mf": 100} + + for component in self._all_components: + fcn = getattr(self, component) + fcn() + # check dimensions can fit full set of matrices + type_names = [ + ('mf', 'full_matrices'), + ('mo', 'origin_matrices'), + ('md', 'destination_matrices'), + ('ms', 'scalar_matrices')] + dims = scenario.emmebank.dimensions + for prefix, name in type_names: + if self._count[prefix] > dims[name]: + raise Exception("emmebank capacity error, increase %s to at least %s" % (name, self._count[prefix])) + + def traffic_demand(self): + tmplt_matrices = [ + ("SOV_NT_L", "SOV non-transponder demand low VOT"), + ("SOV_TR_L", "SOV transponder demand low VOT"), + ("HOV2_L", "HOV2 demand low VOT"), + ("HOV3_L", "HOV3+ demand low VOT"), + ("SOV_NT_M", "SOV non-transponder demand medium VOT"), + ("SOV_TR_M", "SOV transponder demand medium VOT"), + ("HOV2_M", "HOV2 demand medium VOT"), + ("HOV3_M", "HOV3+ demand medium VOT"), + ("SOV_NT_H", "SOV non-transponder demand high VOT"), + ("SOV_TR_H", "SOV transponder demand high VOT"), + ("HOV2_H", "HOV2 demand high VOT"), + ("HOV3_H", "HOV3+ demand high VOT"), + ("TRK_H", "Truck Heavy PCE demand"), + ("TRK_L", "Truck Light PCE demand"), + ("TRK_M", "Truck Medium PCE demand"), + ] + for period in self._all_periods: + self.add_matrices("traffic_demand", period, + [("mf", period + "_" + name, period + " " + desc) + for name, desc in tmplt_matrices]) + + def transit_demand(self): + tmplt_matrices = [ + ("BUS", "local bus demand"), + ("PREM", "Premium modes demand"), + ("ALLPEN", "all modes xfer pen demand"), + ] + for period in self._all_periods: + for a_name in ["WLK", "PNR", "KNR"]: + self.add_matrices("transit_demand", period, + [("mf", "%s_%s%s" % (period, a_name, name), "%s %s access %s" % (period, a_name, desc)) + for name, desc in tmplt_matrices]) + + def traffic_skims(self): + tp_desc = {"TR": "transponder", "NT": "non-transponder"} + vot_desc = {"L": "low", "M": "medium", "H": "high"} + truck_desc = {"L": "light", "M": "medium", "H": "heavy"} + + sov_tmplt_matrices = [ + ("TIME", "SOV %s travel time"), + ("DIST", "SOV %s distance"), + ("REL", "SOV %s reliability skim"), + ("TOLLCOST", "SOV %s toll cost $0.01"), + ("TOLLDIST", "SOV %s distance on toll facility"), + ] + hov_tmplt_matrices = [ + ("TIME", "HOV%s travel time"), + ("DIST", "HOV%s distance"), + ("REL", "HOV%s reliability skim"), + ("TOLLCOST", "HOV%s toll cost $0.01"), + ("TOLLDIST", "HOV%s distance on toll facility"), + ("HOVDIST", "HOV%s HOV distance on HOV facility") + ] + truck_tmplt_matrices = [ + ("TIME", "Truck %s travel time"), + ("DIST", "Truck %s distance"), + ("TOLLCOST", "Truck %s toll cost $0.01") + ] + for period in self._all_periods: + for vot_type in "L", "M", "H": + for tp_type in "NT", "TR": + cls_name = "SOV_" + tp_type + "_" + vot_type + cls_desc = tp_desc[tp_type] + " " + vot_desc[vot_type] + " VOT" + self.add_matrices("traffic_skims", period, + [("mf", period + "_" + cls_name + "_" + name, period + " " + desc % cls_desc) for name, desc in sov_tmplt_matrices]) + for hov_type in "2", "3": + cls_name = "HOV" + hov_type + "_" + vot_type + cls_desc = hov_type + " " + vot_desc[vot_type] + " VOT" + self.add_matrices("traffic_skims", period, + [("mf", period + "_" + cls_name + "_" + name, + period + " " + desc % cls_desc) + for name, desc in hov_tmplt_matrices]) + for truck_type in "L", "M", "H": + cls_name = "TRK" + "_" + truck_type + cls_desc = truck_desc[truck_type] + self.add_matrices("traffic_skims", period, + [("mf", period + "_" + cls_name + "_" + name, + period + " " + desc % cls_desc) + for name, desc in truck_tmplt_matrices]) + + self.add_matrices("traffic_skims", "MD", + [("mf", "MD_TRK_TIME", "MD Truck generic travel time")]) + + def transit_skims(self): + tmplt_matrices = [ + ("GENCOST", "total impedance"), + ("FIRSTWAIT", "first wait time"), + ("XFERWAIT", "transfer wait time"), + ("TOTALWAIT", "total wait time"), + ("FARE", "fare"), + ("XFERS", "num transfers"), + ("ACCWALK", "access walk time"), + ("XFERWALK", "transfer walk time"), + ("EGRWALK", "egress walk time"), + ("TOTALWALK", "total walk time"), + ("TOTALIVTT", "in-vehicle time"), + ("DWELLTIME", "dwell time"), + ("BUSIVTT", "local bus in-vehicle time"), + ("LRTIVTT", "LRT in-vehicle time"), + ("CMRIVTT", "Rail in-vehicle time"), + ("EXPIVTT", "Express in-vehicle time"), + ("LTDEXPIVTT", "Ltd exp bus in-vehicle time"), + ("BRTREDIVTT", "BRT red in-vehicle time"), + ("BRTYELIVTT", "BRT yellow in-vehicle time"), + ("TIER1IVTT", "Tier1 in-vehicle time"), + ("BUSDIST", "Bus IV distance"), + ("LRTDIST", "LRT IV distance"), + ("CMRDIST", "Rail IV distance"), + ("EXPDIST", "Express and Ltd IV distance"), + ("BRTDIST", "BRT red and yel IV distance"), + ("TIER1DIST", "Tier1 distance"), + ("TOTDIST", "Total transit distance") + ] + skim_sets = [ + ("BUS", "Local bus only"), + ("PREM", "Premium modes only"), + ("ALLPEN", "All w/ xfer pen") + ] + for period in self._all_periods: + for set_name, set_desc in skim_sets: + self.add_matrices("transit_skims", period, + [("mf", period + "_" + set_name + "_" + name, + period + " " + set_desc + ": " + desc) + for name, desc in tmplt_matrices]) + + def truck_model(self): + tmplt_matrices = [ + ("TRKL", "Truck Light"), + ("TRKM", "Truck Medium"), + ("TRKH", "Truck Heavy"), + ("TRKEI", "Truck external-internal"), + ("TRKIE", "Truck internal-external"), + ] + self.add_matrices("truck_model", "ALL", + [("mo", name + '_PROD', desc + ' production') + for name, desc in tmplt_matrices]) + self.add_matrices("truck_model", "ALL", + [("md", name + '_ATTR', desc + ' attraction') + for name, desc in tmplt_matrices]) + + tmplt_matrices = [ + ("TRKEE_DEMAND", "Truck total external-external demand"), + ("TRKL_FRICTION", "Truck Light friction factors"), + ("TRKM_FRICTION", "Truck Medium friction factors"), + ("TRKH_FRICTION", "Truck Heavy friction factors"), + ("TRKIE_FRICTION", "Truck internal-external friction factors"), + ("TRKEI_FRICTION", "Truck external-internal friction factors"), + ("TRKL_DEMAND", "Truck Light total demand"), + ("TRKM_DEMAND", "Truck Medium total demand"), + ("TRKH_DEMAND", "Truck Heavy total demand"), + ("TRKIE_DEMAND", "Truck internal-external total demand"), + ("TRKEI_DEMAND", "Truck external-internal total demand"), + ] + self.add_matrices("truck_model", "ALL", + [("mf", name, desc) for name, desc in tmplt_matrices]) + + # TODO: remove GP and TOLL matrices, no longer used + tmplt_matrices = [ + ("TRK_L_VEH", "Truck Light demand"), + ("TRKLGP_VEH", "Truck Light GP-only vehicle demand"), + ("TRKLTOLL_VEH", "Truck Light toll vehicle demand"), + ("TRK_M_VEH", "Truck Medium demand"), + ("TRKMGP_VEH", "Truck Medium GP-only vehicle demand"), + ("TRKMTOLL_VEH", "Truck Medium toll vehicle demand"), + ("TRK_H_VEH", "Truck Heavy demand"), + ("TRKHGP_VEH", "Truck Heavy GP-only vehicle demand"), + ("TRKHTOLL_VEH", "Truck Heavy toll vehicle demand"), + ] + for period in self._all_periods: + self.add_matrices("truck_model", period, + [("mf", period + "_" + name, period + " " + desc) + for name, desc in tmplt_matrices]) + + def commercial_vehicle_model(self): + # TODO : remove commercial vehicle matrices, no longer used + tmplt_matrices = [ + ('mo', 'COMVEH_PROD', 'Commercial vehicle production'), + ('md', 'COMVEH_ATTR', 'Commercial vehicle attraction'), + ('mf', 'COMVEH_BLENDED_SKIM', 'Commercial vehicle blended skim'), + ('mf', 'COMVEH_FRICTION', 'Commercial vehicle friction factors'), + ('mf', 'COMVEH_TOTAL_DEMAND', 'Commercial vehicle total demand all periods'), + ] + self.add_matrices("commercial_vehicle_model", "ALL", + [(ident, name, desc) for ident, name, desc in tmplt_matrices]) + + tmplt_matrices = [ + ('COMVEH', 'Commerical vehicle total demand'), + ('COMVEHGP', 'Commerical vehicle GP demand'), + ('COMVEHTOLL', 'Commerical vehicle Toll demand'), + ] + for period in self._all_periods: + self.add_matrices("commercial_vehicle_model", period, + [("mf", period + "_" + name, period + " " + desc) + for name, desc in tmplt_matrices]) + + def external_internal_model(self): + tmplt_matrices = [ + ('SOVTOLL_EIWORK', 'US to SD SOV Work TOLL demand'), + ('HOV2TOLL_EIWORK', 'US to SD HOV2 Work TOLL demand'), + ('HOV3TOLL_EIWORK', 'US to SD HOV3 Work TOLL demand'), + ('SOVGP_EIWORK', 'US to SD SOV Work GP demand'), + ('HOV2HOV_EIWORK', 'US to SD HOV2 Work HOV demand'), + ('HOV3HOV_EIWORK', 'US to SD HOV3 Work HOV demand'), + ('SOVTOLL_EINONWORK', 'US to SD SOV Non-Work TOLL demand'), + ('HOV2TOLL_EINONWORK', 'US to SD HOV2 Non-Work TOLL demand'), + ('HOV3TOLL_EINONWORK', 'US to SD HOV3 Non-Work TOLL demand'), + ('SOVGP_EINONWORK', 'US to SD SOV Non-Work GP demand'), + ('HOV2HOV_EINONWORK', 'US to SD HOV2 Non-Work HOV demand'), + ('HOV3HOV_EINONWORK', 'US to SD HOV3 Non-Work HOV demand'), + ] + for period in self._all_periods: + self.add_matrices("external_internal_model", period, + [("mf", period + "_" + name, period + " " + desc) + for name, desc in tmplt_matrices]) + + def external_external_model(self): + self.add_matrices("external_external_model", "ALL", + [("mf", "ALL_TOTAL_EETRIPS", "All periods Total for all modes external-external trips")]) + tmplt_matrices = [ + ('SOV_EETRIPS', 'SOV external-external demand'), + ('HOV2_EETRIPS', 'HOV2 external-external demand'), + ('HOV3_EETRIPS', 'HOV3 external-external demand'), + ] + for period in self._all_periods: + self.add_matrices("external_external_model", period, + [("mf", period + "_" + name, period + " " + desc) + for name, desc in tmplt_matrices]) + + def add_matrices(self, component, period, matrices): + for ident, name, desc in matrices: + self._matrices[component][period].append([ident+str(self._count[ident]), name, desc]) + self._count[ident] += 1 + + def create_matrices(self, component, periods): + with _m.logbook_trace("Create matrices for component %s" % (component.replace("_", " "))): + emmebank = self.scenario.emmebank + matrices = [] + for period in periods + ["ALL"]: + with _m.logbook_trace("For period %s" % (period)): + for ident, name, desc in self._matrices[component][period]: + existing_matrix = emmebank.matrix(name) + if existing_matrix and (existing_matrix.id != ident): + raise Exception("Matrix name conflict '%s', with id %s instead of %s. Delete all matrices first." + % (name, existing_matrix.id, ident)) + matrices.append(self._create_matrix_tool(ident, name, desc, scenario=self.scenario, overwrite=True)) + return matrices + + def get_matrix_names(self, component, periods, scenario): + self.generate_matrix_list(scenario) + matrices = [] + for period in periods: + matrices.extend([m[1] for m in self._matrices[component][period]]) + return matrices + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py b/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py new file mode 100644 index 0000000..513dba3 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py @@ -0,0 +1,168 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// initialize_transit_databse.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Coordinates the initialization of all matrices. +# The matrix names are listed for each of the model components / steps, +# and the matrix IDs are assigned consistently from the set of matrices. +# In each of the model steps the matrices are only referenced by name, +# never by ID. +# +# +# Inputs: +# components: A list of the model components / steps for which to initialize matrices +# One or more of "traffic_demand", "transit_demand", +# "traffic_skims", "transit_skims", "external_internal_model", +# "external_external_model", "truck_model", "commercial_vehicle_model" +# periods: A list of periods for which to initialize matrices, "EA", "AM", "MD", "PM", "EV" +# scenario: scenario to use for reference zone system and the emmebank in which +# the matrices will be created. Defaults to the current primary scenario. +# +# Script example: +""" + import os + import inro.emme.database.emmebank as _eb + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) + base_scenario = main_emmebank.scenario(100) + initialize_transit_db = modeller.tool("sandag.initialize.initialize_transit_database") + initialize_transit_db(base_scenario) +""" +TOOLBOX_ORDER = 8 + + +import inro.modeller as _m +import inro.emme.network as _network +import inro.emme.database.emmebank as _eb +from inro.emme.desktop.exception import AddDatabaseError +import traceback as _traceback +import shutil as _shutil +import time +import os + +join = os.path.join + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class InitializeTransitDatabase(_m.Tool(), gen_utils.Snapshot): + + base_scenario = _m.Attribute(_m.InstanceType) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.base_scenario = _m.Modeller().scenario + self.attributes = ["base_scenario"] + + def from_snapshot(self, snapshot): + super(InitializeTransitDatabase, self).from_snapshot(snapshot) + # custom from_snapshot to load scenario object + self.base_scenario = _m.Modeller().emmebank.scenario(self.base_scenario) + return self + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Initialize transit database" + pb.description = """Create and setup database for transit assignments under 'Database_transit' directory. + Will overwrite an existing database. The TAZs will be removed and TAP nodes converted to zones.""" + pb.branding_text = "- SANDAG" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_scenario("base_scenario", + title="Base scenario:", note="Base traffic and transit scenario with TAZs.") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self(self.base_scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Initialize transit database', save_arguments=True) + def __call__(self, base_scenario, add_database=True): + attributes = {"base_scenario": base_scenario.id} + gen_utils.log_snapshot("Initialize transit database", str(self), attributes) + create_function = _m.Modeller().tool("inro.emme.data.function.create_function") + build_transit_scen = _m.Modeller().tool("sandag.assignment.build_transit_scenario") + load_properties = _m.Modeller().tool('sandag.utilities.properties') + + base_eb = base_scenario.emmebank + project_dir = os.path.dirname(os.path.dirname(base_eb.path)) + main_directory = os.path.dirname(project_dir) + props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) + scenarioYear = props["scenarioYear"] + + transit_db_dir = join(project_dir, "Database_transit") + transit_db_path = join(transit_db_dir, "emmebank") + network = base_scenario.get_partial_network(["NODE"], include_attributes=True) + num_zones = sum([1 for n in network.nodes() if n["@tap_id"] > 0]) + dimensions = base_eb.dimensions + dimensions["centroids"] = num_zones + dimensions["scenarios"] = 10 + if not os.path.exists(transit_db_dir): + os.mkdir(transit_db_dir) + if os.path.exists(transit_db_path): + transit_eb = _eb.Emmebank(transit_db_path) + for scenario in transit_eb.scenarios(): + transit_eb.delete_scenario(scenario.id) + for function in transit_eb.functions(): + transit_eb.delete_function(function.id) + if transit_eb.dimensions != dimensions: + _eb.change_dimensions(transit_db_path, dimensions, keep_backup=False) + else: + transit_eb = _eb.create(transit_db_path, dimensions) + + transit_eb.title = base_eb.title[:65] + "-transit" + transit_eb.coord_unit_length = base_eb.coord_unit_length + transit_eb.unit_of_length = base_eb.unit_of_length + transit_eb.unit_of_cost = base_eb.unit_of_cost + transit_eb.unit_of_energy = base_eb.unit_of_energy + transit_eb.use_engineering_notation = base_eb.use_engineering_notation + transit_eb.node_number_digits = base_eb.node_number_digits + + zone_scenario = build_transit_scen( + period="AM", base_scenario=base_scenario, transit_emmebank=transit_eb, + scenario_id=base_scenario.id, scenario_title="%s transit zones" % (base_scenario.title), + data_table_name=scenarioYear, overwrite=True) + for function in base_scenario.emmebank.functions(): + create_function(function.id, function.expression, transit_eb) + if add_database: + self.add_database(transit_eb) + return zone_scenario + + def add_database(self, emmebank): + modeller = _m.Modeller() + desktop = modeller.desktop + data_explorer = desktop.data_explorer() + for db in data_explorer.databases(): + if os.path.normpath(db.path) == os.path.normpath(emmebank.path): + return + try: + data_explorer.add_database(emmebank.path) + except AddDatabaseError: + pass # database has already been added to the project diff --git a/sandag_abm/src/main/emme/toolbox/master_run.py b/sandag_abm/src/main/emme/toolbox/master_run.py new file mode 100644 index 0000000..b24daf9 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/master_run.py @@ -0,0 +1,1202 @@ +# ////////////////////////////////////////////////////////////////////////////// +# //// /// +# //// Copyright INRO, 2016-2017. /// +# //// Rights to use and modify are granted to the /// +# //// San Diego Association of Governments and partner agencies. /// +# //// This copyright notice must be preserved. /// +# //// /// +# //// model/master_run.py /// +# //// /// +# //// /// +# //// /// +# //// /// +# ////////////////////////////////////////////////////////////////////////////// +# +# The Master run tool is the primary method to operate the SANDAG +# travel demand model. It operates all the model components. +# +# main_directory: Main ABM directory: directory which contains all of the +# ABM scenario data, including this project. The default is the parent +# directory of the current Emme project. +# scenario_id: Scenario ID for the base imported network data. The result +# scenarios are indexed in the next five scenarios by time period. +# scenario_title: title to use for the scenario. +# emmebank_title: title to use for the Emmebank (Emme database) +# num_processors: the number of processors to use for traffic and transit +# assignments and skims, aggregate demand models (where required) and +# other parallelized procedures in Emme. Default is Max available - 1. +# Properties loaded from conf/sandag_abm.properties: +# When using the tool UI, the sandag_abm.properties file is read +# and the values cached and the inputs below are pre-set. When the tool +# is started button is clicked this file is written out with the +# values specified. +# Sample rate by iteration: three values for the sample rates for each iteration +# Start from iteration: iteration from which to start the model run +# Skip steps: optional checkboxes to skip model steps. +# Note that most steps are dependent upon the results of the previous steps. +# Select link: add select link analyses for traffic. +# See the Select link analysis section under the Traffic assignment tool. +# +# Also reads and processes the per-scenario +# vehicle_class_availability.csv (optional): 0 or 1 indicators by vehicle class and specified facilities to indicate availability +# +# Script example: +""" +import inro.modeller as _m +import os +modeller = _m.Modeller() +desktop = modeller.desktop + +master_run = modeller.tool("sandag.master_run") +main_directory = os.path.dirname(os.path.dirname(desktop.project_path())) +scenario_id = 100 +scenario_title = "Base 2015 scenario" +emmebank_title = "Base 2015 with updated landuse" +num_processors = "MAX-1" +master_run(main_directory, scenario_id, scenario_title, emmebank_title, num_processors) +""" + +TOOLBOX_ORDER = 1 +VIRUTALENV_PATH = "C:\\python_virtualenv\\abm14_2_0" + +import inro.modeller as _m +import inro.emme.database.emmebank as _eb + +import traceback as _traceback +import glob as _glob +import subprocess as _subprocess +import ctypes as _ctypes +import json as _json +import shutil as _shutil +import tempfile as _tempfile +from copy import deepcopy as _copy +from collections import defaultdict as _defaultdict +import time as _time +import socket as _socket +import sys +import os + +import pandas as pd +import numpy as np +import csv +import datetime +import pyodbc +import win32com.client as win32 + +_join = os.path.join +_dir = os.path.dirname +_norm = os.path.normpath + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") +props_utils = _m.Modeller().module("sandag.utilities.properties") + + +class MasterRun(props_utils.PropertiesSetter, _m.Tool(), gen_utils.Snapshot): + main_directory = _m.Attribute(unicode) + scenario_id = _m.Attribute(int) + scenario_title = _m.Attribute(unicode) + emmebank_title = _m.Attribute(unicode) + num_processors = _m.Attribute(str) + select_link = _m.Attribute(unicode) + username = _m.Attribute(unicode) + password = _m.Attribute(unicode) + + properties_path = _m.Attribute(unicode) + + tool_run_msg = "" + + def __init__(self): + super(MasterRun, self).__init__() + project_dir = _dir(_m.Modeller().desktop.project.path) + self.main_directory = _dir(project_dir) + self.properties_path = _join(_dir(project_dir), "conf", "sandag_abm.properties") + self.scenario_id = 100 + self.scenario_title = "" + self.emmebank_title = "" + self.num_processors = "MAX-1" + self.select_link = '[]' + self.username = os.environ.get("USERNAME") + self.attributes = [ + "main_directory", "scenario_id", "scenario_title", "emmebank_title", + "num_processors", "select_link" + ] + self._log_level = "ENABLED" + self.LOCAL_ROOT = "C:\\abm_runs" + + def page(self): + self.load_properties() + pb = _m.ToolPageBuilder(self) + pb.title = "Master run ABM" + pb.description = """Runs the SANDAG ABM, assignments, and other demand model tools.""" + pb.branding_text = "- SANDAG - Model" + tool_proxy_tag = pb.tool_proxy_tag + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('main_directory', 'directory', + title='Select main ABM directory', note='') + pb.add_text_box('scenario_id', title="Scenario ID:") + pb.add_text_box('scenario_title', title="Scenario title:", size=80) + pb.add_text_box('emmebank_title', title="Emmebank title:", size=60) + dem_utils.add_select_processors("num_processors", pb, self) + + # username and password input for distributed assignment + # username also used in the folder name for the local drive operation + pb.add_html(''' +
+
Credentials for remote run
+
+ Username: + + Password: + +
+
+ Note: required for running distributed traffic assignments using PsExec. +
+ Distributed / single node modes are configured in "config/server-config.csv". +
The username is also used for the folder name when running on the local drive. +
+
''' % {"tool_proxy_tag": tool_proxy_tag}) + + # defined in properties utilities + self.add_properties_interface(pb, disclosure=True) + # redirect properties file after browse of main_directory + pb.add_html(""" +""" % {"tool_proxy_tag": tool_proxy_tag}) + + traffic_assign = _m.Modeller().tool("sandag.assignment.traffic_assignment") + traffic_assign._add_select_link_interface(pb) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self.save_properties() + self(self.main_directory, self.scenario_id, self.scenario_title, self.emmebank_title, + self.num_processors, self.select_link, username=self.username, password=self.password) + run_msg = "Model run complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception(error, _traceback.format_exc()) + + raise + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + @_m.logbook_trace("Master run model", save_arguments=True) + def __call__(self, main_directory, scenario_id, scenario_title, emmebank_title, num_processors, + select_link=None, periods=["EA", "AM", "MD", "PM", "EV"], username=None, password=None): + attributes = { + "main_directory": main_directory, + "scenario_id": scenario_id, + "scenario_title": scenario_title, + "emmebank_title": emmebank_title, + "num_processors": num_processors, + "select_link": select_link, + "periods": periods, + "username": username, + } + gen_utils.log_snapshot("Master run model", str(self), attributes) + + modeller = _m.Modeller() + # Checking that the virtualenv path is set and the folder is installed + if not os.path.exists(VIRUTALENV_PATH): + raise Exception("Python virtual environment not installed at expected location %s" % VIRUTALENV_PATH) + venv_path = os.environ.get("PYTHON_VIRTUALENV") + if not venv_path: + raise Exception("Environment variable PYTHON_VIRTUALENV not set, start Emme from 'start_emme_with_virtualenv.bat'") + if not venv_path == VIRUTALENV_PATH: + raise Exception("PYTHON_VIRTUALENV is not the expected value (%s instead of %s)" % (venv_path, VIRUTALENV_PATH)) + venv_path_found = False + for path in sys.path: + if VIRUTALENV_PATH in path: + venv_path_found = True + break + if not venv_path_found: + raise Exception("Python virtual environment not found in system path %s" % VIRUTALENV_PATH) + copy_scenario = modeller.tool("inro.emme.data.scenario.copy_scenario") + run4Ds = modeller.tool("sandag.import.run4Ds") + import_network = modeller.tool("sandag.import.import_network") + input_checker = modeller.tool("sandag.import.input_checker") + init_transit_db = modeller.tool("sandag.initialize.initialize_transit_database") + init_matrices = modeller.tool("sandag.initialize.initialize_matrices") + import_demand = modeller.tool("sandag.import.import_seed_demand") + build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") + transit_assign = modeller.tool("sandag.assignment.transit_assignment") + run_truck = modeller.tool("sandag.model.truck.run_truck_model") + external_internal = modeller.tool("sandag.model.external_internal") + external_external = modeller.tool("sandag.model.external_external") + import_auto_demand = modeller.tool("sandag.import.import_auto_demand") + import_transit_demand = modeller.tool("sandag.import.import_transit_demand") + export_transit_skims = modeller.tool("sandag.export.export_transit_skims") + export_for_transponder = modeller.tool("sandag.export.export_for_transponder") + export_network_data = modeller.tool("sandag.export.export_data_loader_network") + export_matrix_data = modeller.tool("sandag.export.export_data_loader_matrices") + export_tap_adjacent_lines = modeller.tool("sandag.export.export_tap_adjacent_lines") + export_for_commercial_vehicle = modeller.tool("sandag.export.export_for_commercial_vehicle") + validation = modeller.tool("sandag.validation.validation") + file_manager = modeller.tool("sandag.utilities.file_manager") + utils = modeller.module('sandag.utilities.demand') + load_properties = modeller.tool('sandag.utilities.properties') + run_summary = modeller.tool("sandag.utilities.run_summary") + + self.username = username + self.password = password + + props = load_properties(_join(main_directory, "conf", "sandag_abm.properties")) + props.set_year_specific_properties(_join(main_directory, "input", "parametersByYears.csv")) + props.set_year_specific_properties(_join(main_directory, "input", "filesByYears.csv")) + props.save() + # Log current state of props file for debugging of UI / file sync issues + attributes = dict((name, props["RunModel." + name]) for name in self._run_model_names) + _m.logbook_write("SANDAG properties file", attributes=attributes) + if self._properties: # Tool has been called via the UI + # Compare UI values and file values to make sure they are the same + error_text = ("Different value found in sandag_abm.properties than specified in UI for '%s'. " + "Close sandag_abm.properties if open in any text editor, check UI and re-run.") + for name in self._run_model_names: + if getattr(self, name) != props["RunModel." + name]: + raise Exception(error_text % name) + + scenarioYear = str(props["scenarioYear"]) + startFromIteration = props["RunModel.startFromIteration"] + precision = props["RunModel.MatrixPrecision"] + minSpaceOnC = props["RunModel.minSpaceOnC"] + sample_rate = props["sample_rates"] + end_iteration = len(sample_rate) + scale_factor = props["cvm.scale_factor"] + visualizer_reference_path = props["visualizer.reference.path"] + visualizer_output_file = props["visualizer.output"] + visualizer_reference_label = props["visualizer.reference.label"] + visualizer_build_label = props["visualizer.build.label"] + mgraInputFile = props["mgra.socec.file"] + + period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) + + useLocalDrive = props["RunModel.useLocalDrive"] + + skip4Ds = props["RunModel.skip4Ds"] + skipInputChecker = props["RunModel.skipInputChecker"] + skipInitialization = props["RunModel.skipInitialization"] + deleteAllMatrices = props["RunModel.deleteAllMatrices"] + skipCopyWarmupTripTables = props["RunModel.skipCopyWarmupTripTables"] + skipCopyBikeLogsum = props["RunModel.skipCopyBikeLogsum"] + skipCopyWalkImpedance = props["RunModel.skipCopyWalkImpedance"] + skipWalkLogsums = props["RunModel.skipWalkLogsums"] + skipBikeLogsums = props["RunModel.skipBikeLogsums"] + skipBuildNetwork = props["RunModel.skipBuildNetwork"] + skipHighwayAssignment = props["RunModel.skipHighwayAssignment"] + skipTransitSkimming = props["RunModel.skipTransitSkimming"] + skipTransponderExport = props["RunModel.skipTransponderExport"] + skipCoreABM = props["RunModel.skipCoreABM"] + skipOtherSimulateModel = props["RunModel.skipOtherSimulateModel"] + skipMAASModel = props["RunModel.skipMAASModel"] + skipCTM = props["RunModel.skipCTM"] + skipEI = props["RunModel.skipEI"] + skipExternal = props["RunModel.skipExternalExternal"] + skipTruck = props["RunModel.skipTruck"] + skipTripTableCreation = props["RunModel.skipTripTableCreation"] + skipFinalHighwayAssignment = props["RunModel.skipFinalHighwayAssignment"] + skipFinalHighwayAssignmentStochastic = props["RunModel.skipFinalHighwayAssignmentStochastic"] + if skipFinalHighwayAssignmentStochastic == True: + makeFinalHighwayAssignmentStochastic = False + else: + makeFinalHighwayAssignmentStochastic = True + skipFinalTransitAssignment = props["RunModel.skipFinalTransitAssignment"] + skipVisualizer = props["RunModel.skipVisualizer"] + skipDataExport = props["RunModel.skipDataExport"] + skipDataLoadRequest = props["RunModel.skipDataLoadRequest"] + skipDeleteIntermediateFiles = props["RunModel.skipDeleteIntermediateFiles"] + skipTransitShed = props["RunModel.skipTransitShed"] + transitShedThreshold = props["transitShed.threshold"] + transitShedTOD = props["transitShed.TOD"] + + #check if visualizer.reference.path is valid in filesbyyears.csv + if not os.path.exists(visualizer_reference_path): + raise Exception("Visualizer reference %s does not exist. Check filesbyyears.csv." %(visualizer_reference_path)) + + if useLocalDrive: + folder_name = os.path.basename(main_directory) + if not os.path.exists(_join(self.LOCAL_ROOT, username, folder_name, "report")): # check free space only if it is a new run + self.check_free_space(minSpaceOnC) + # if initialization copy ALL files from remote + # else check file meta data and copy those that have changed + initialize = (skipInitialization == False and startFromIteration == 1) + local_directory = file_manager( + "DOWNLOAD", main_directory, username, scenario_id, initialize=initialize) + self._path = local_directory + else: + self._path = main_directory + + drive, path_no_drive = os.path.splitdrive(self._path) + path_forward_slash = path_no_drive.replace("\\", "/") + input_dir = _join(self._path, "input") + input_truck_dir = _join(self._path, "input_truck") + output_dir = _join(self._path, "output") + validation_dir = _join(self._path, "analysis/validation") + main_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database", "emmebank")) + if emmebank_title: + main_emmebank.title = emmebank_title + external_zones = "1-12" + + travel_modes = ["auto", "tran", "nmot", "othr"] + core_abm_files = ["Trips*.omx", "InternalExternalTrips*.omx"] + core_abm_files = [mode + name for name in core_abm_files for mode in travel_modes] + smm_abm_files = ["AirportTrips*.omx", "CrossBorderTrips*.omx", "VisitorTrips*.omx"] + smm_abm_files = [mode + name for name in smm_abm_files for mode in travel_modes] + maas_abm_files = ["EmptyAVTrips.omx", "TNCVehicleTrips*.omx"] + + relative_gap = props["convergence"] + max_assign_iterations = 1000 + mgra_lu_input_file = props["mgra.socec.file"] + + with _m.logbook_trace("Setup and initialization"): + self.set_global_logbook_level(props) + + # Swap Server Configurations + self.run_proc("serverswap.bat", [drive, path_no_drive, path_forward_slash], "Run ServerSwap") + self.check_for_fatal(_join(self._path, "logFiles", "serverswap.log"), + "ServerSwap failed! Open logFiles/serverswap.log for details.") + self.run_proc("checkAtTransitNetworkConsistency.cmd", [drive, path_forward_slash], + "Checking if AT and Transit Networks are consistent") + self.check_for_fatal(_join(self._path, "logFiles", "AtTransitCheck_event.log"), + "AT and Transit network consistency checking failed! Open AtTransitCheck_event.log for details.") + + if startFromIteration == 1: # only run the setup / init steps if starting from iteration 1 + if not skipWalkLogsums: + self.run_proc("runSandagWalkLogsums.cmd", [drive, path_forward_slash], + "Walk - create AT logsums and impedances") + if not skipCopyWalkImpedance: + self.copy_files(["walkMgraEquivMinutes.csv", "walkMgraTapEquivMinutes.csv", "microMgraEquivMinutes.csv", "microMgraTapEquivMinutes.csv"], + input_dir, output_dir) + + if not skip4Ds: + run4Ds(path=self._path, int_radius=0.65, ref_path=visualizer_reference_path) + + + mgraFile = 'mgra13_based_input' + str(scenarioYear) + '.csv' + self.complete_work(scenarioYear, input_dir, output_dir, mgraFile, "walkMgraEquivMinutes.csv") + + if not skipBuildNetwork: + base_scenario = import_network( + source=input_dir, + merged_scenario_id=scenario_id, + title=scenario_title, + data_table_name=scenarioYear, + overwrite=True, + emmebank=main_emmebank) + + if "modify_network.py" in os.listdir(os.getcwd()): + try: + with _m.logbook_trace("Modify network script"): + import modify_network + reload(modify_network) + modify_network.run(base_scenario) + except ImportError as e: + pass + + if not skipInputChecker: + input_checker(path=self._path) + + export_tap_adjacent_lines(_join(output_dir, "tapLines.csv"), base_scenario) + # parse vehicle availablility file by time-of-day + availability_file = "vehicle_class_availability.csv" + availabilities = self.parse_availability_file(_join(input_dir, availability_file), periods) + # initialize per time-period scenarios + for number, period in period_ids: + title = "%s - %s assign" % (base_scenario.title, period) + # copy_scenario(base_scenario, number, title, overwrite=True) + _m.logbook_write( + name="Copy scenario %s to %s" % (base_scenario.number, number), + attributes={ + 'from_scenario': base_scenario.number, + 'scenario_id': number, + 'overwrite': True, + 'scenario_title': title + } + ) + if main_emmebank.scenario(number): + main_emmebank.delete_scenario(number) + scenario = main_emmebank.copy_scenario(base_scenario.number, number) + scenario.title = title + # Apply availabilities by facility and vehicle class to this time period + self.apply_availabilities(period, scenario, availabilities) + else: + base_scenario = main_emmebank.scenario(scenario_id) + + if not skipInitialization: + # initialize traffic demand, skims, truck, CV, EI, EE matrices + traffic_components = [ + "traffic_skims", + "truck_model", + "external_internal_model", "external_external_model"] + if not skipCopyWarmupTripTables: + traffic_components.append("traffic_demand") + init_matrices(traffic_components, periods, base_scenario, deleteAllMatrices) + + transit_scenario = init_transit_db(base_scenario, add_database=not useLocalDrive) + transit_emmebank = transit_scenario.emmebank + transit_components = ["transit_skims"] + if not skipCopyWarmupTripTables: + transit_components.append("transit_demand") + init_matrices(transit_components, periods, transit_scenario, deleteAllMatrices) + else: + transit_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database_transit", "emmebank")) + transit_scenario = transit_emmebank.scenario(base_scenario.number) + + if not skipCopyWarmupTripTables: + # import seed auto demand and seed truck demand + for period in periods: + omx_file = _join(input_dir, "trip_%s.omx" % period) + import_demand(omx_file, "AUTO", period, base_scenario) + import_demand(omx_file, "TRUCK", period, base_scenario) + + if not skipBikeLogsums: + self.run_proc("runSandagBikeLogsums.cmd", [drive, path_forward_slash], + "Bike - create AT logsums and impedances") + if not skipCopyBikeLogsum: + self.copy_files(["bikeMgraLogsum.csv", "bikeTazLogsum.csv"], input_dir, output_dir) + + else: + base_scenario = main_emmebank.scenario(scenario_id) + transit_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database_transit", "emmebank")) + transit_scenario = transit_emmebank.scenario(base_scenario.number) + + # Check that setup files were generated + self.run_proc("CheckOutput.bat", [drive + path_no_drive, 'Setup'], "Check for outputs") + + # Note: iteration indexes from 0, msa_iteration indexes from 1 + for iteration in range(startFromIteration - 1, end_iteration): + msa_iteration = iteration + 1 + with _m.logbook_trace("Iteration %s" % msa_iteration): + if not skipCoreABM[iteration] or not skipOtherSimulateModel[iteration] or not skipMAASModel[iteration]: + self.run_proc("runMtxMgr.cmd", [drive, drive + path_no_drive], "Start matrix manager") + self.run_proc("runHhMgr.cmd", [drive, drive + path_no_drive], "Start Hh manager") + + if not skipHighwayAssignment[iteration]: + # run traffic assignment + # export traffic skims + with _m.logbook_trace("Traffic assignment and skims"): + self.run_traffic_assignments( + base_scenario, period_ids, msa_iteration, relative_gap, + max_assign_iterations, num_processors) + self.run_proc("CreateD2TAccessFile.bat", [drive, path_forward_slash], + "Create drive to transit access file", capture_output=True) + + if not skipTransitSkimming[iteration]: + # run transit assignment + # export transit skims + with _m.logbook_trace("Transit assignments and skims"): + for number, period in period_ids: + src_period_scenario = main_emmebank.scenario(number) + transit_assign_scen = build_transit_scen( + period=period, base_scenario=src_period_scenario, + transit_emmebank=transit_emmebank, + scenario_id=src_period_scenario.id, + scenario_title="%s %s transit assign" % (base_scenario.title, period), + data_table_name=scenarioYear, overwrite=True) + transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, + skims_only=True, num_processors=num_processors) + + omx_file = _join(output_dir, "transit_skims.omx") + export_transit_skims(omx_file, periods, transit_scenario) + + if not skipTransponderExport[iteration]: + am_scenario = main_emmebank.scenario(base_scenario.number + 2) + export_for_transponder(output_dir, num_processors, am_scenario) + + # For each step move trip matrices so run will stop if ctramp model + # doesn't produced csv/omx files for assignment + # also needed as CT-RAMP does not overwrite existing files + if not skipCoreABM[iteration]: + self.remove_prev_iter_files(core_abm_files, output_dir, iteration) + self.run_proc( + "runSandagAbm_SDRM.cmd", + [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], + "Java-Run CT-RAMP", capture_output=True) + if not skipOtherSimulateModel[iteration]: + self.remove_prev_iter_files(smm_abm_files, output_dir, iteration) + self.run_proc( + "runSandagAbm_SMM.cmd", + [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], + "Java-Run airport model, visitor model, cross-border model", capture_output=True) + + if not skipMAASModel[iteration]: + self.remove_prev_iter_files(maas_abm_files, output_dir, iteration) + self.run_proc( + "runSandagAbm_MAAS.cmd", + [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], + "Java-Run AV allocation model and TNC routing model", capture_output=True) + + if not skipCTM[iteration]: + export_for_commercial_vehicle(output_dir, base_scenario) + self.run_proc( + "cvm.bat", + [drive, path_no_drive, path_forward_slash, scale_factor, mgra_lu_input_file, + "tazcentroids_cvm.csv"], + "Commercial vehicle model", capture_output=True) + if msa_iteration == startFromIteration: + external_zones = "1-12" + if not skipTruck[iteration]: + # run truck model (generate truck trips) + run_truck(True, input_dir, input_truck_dir, num_processors, base_scenario) + # run EI model "US to SD External Trip Model" + if not skipEI[iteration]: + external_internal(input_dir, base_scenario) + # run EE model + if not skipExternal[iteration]: + external_external(input_dir, external_zones, base_scenario) + + # import demand from all sub-market models from CT-RAMP and + # add CV trips to auto demand + # add EE and EI trips to auto demand + if not skipTripTableCreation[iteration]: + import_auto_demand(output_dir, external_zones, num_processors, base_scenario) + + if not skipFinalHighwayAssignment: + with _m.logbook_trace("Final traffic assignments"): + # Final iteration is assignment only, no skims + final_iteration = 4 + self.run_traffic_assignments( + base_scenario, period_ids, final_iteration, relative_gap, max_assign_iterations, + num_processors, select_link, makeFinalHighwayAssignmentStochastic, input_dir) + + if not skipFinalTransitAssignment: + import_transit_demand(output_dir, transit_scenario) + with _m.logbook_trace("Final transit assignments"): + # Final iteration includes the transit skims per ABM-1072 + for number, period in period_ids: + src_period_scenario = main_emmebank.scenario(number) + transit_assign_scen = build_transit_scen( + period=period, base_scenario=src_period_scenario, + transit_emmebank=transit_emmebank, scenario_id=src_period_scenario.id, + scenario_title="%s - %s transit assign" % (base_scenario.title, period), + data_table_name=scenarioYear, overwrite=True) + transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, + num_processors=num_processors) + omx_file = _join(output_dir, "transit_skims.omx") + export_transit_skims(omx_file, periods, transit_scenario, big_to_zero=True) + + if not skipTransitShed: + # write walk and drive transit sheds + self.run_proc("runtransitreporter.cmd", [drive, path_forward_slash, transitShedThreshold, transitShedTOD], + "Create walk and drive transit sheds", + capture_output=True) + + if not skipVisualizer: + self.run_proc("RunViz.bat", + [drive, path_no_drive, visualizer_reference_path, visualizer_output_file, "NO", visualizer_reference_label, visualizer_build_label, mgraInputFile], + "HTML Visualizer", capture_output=True) + + if not skipDataExport: + # export network and matrix results from Emme directly to T if using local drive + output_directory = _join(self._path, "output") + export_network_data(self._path, scenario_id, main_emmebank, transit_emmebank, num_processors) + export_matrix_data(output_directory, base_scenario, transit_scenario) + # export core ABM data + # Note: uses relative project structure, so cannot redirect to T drive + self.run_proc("DataExporter.bat", [drive, path_no_drive], "Export core ABM data",capture_output=True) + + #Validation for 2016 scenario + if scenarioYear == "2016": + validation(self._path, main_emmebank, base_scenario) # to create source_EMME.xlsx + + # #Create Worksheet for ABM Validation using PowerBI Visualization #JY: can be uncommented if deciding to incorporate PowerBI vis in ABM workflow + # self.run_proc("VisPowerBI.bat", # forced to update excel links + # [drive, path_no_drive, scenarioYear, 0], + # "VisPowerBI", + # capture_output=True) + + ### CL: Below step is temporarily used to update validation output files. When Gregor complete Upload procedure, below step should be removed. 05/31/20 + # self.run_proc("ExcelUpdate.bat", # forced to update excel links + # [drive, path_no_drive, scenarioYear, 0], + # "ExcelUpdate", + # capture_output=True) + + ### ES: Commented out until this segment is updated to reference new database. 9/10/20 ### + # add segments below for auto-reporting, YMA, 1/23/2019 + # add this loop to find the sceanro_id in the [dimension].[scenario] table + + #database_scenario_id = 0 + #int_hour = 0 + #while int_hour <= 96: + + # database_scenario_id = self.sql_select_scenario(scenarioYear, end_iteration, + # sample_rate[end_iteration - 1], path_no_drive, + # start_db_time) + # if database_scenario_id > 0: + # break + + # int_hour = int_hour + 1 + # _time.sleep(900) # wait for 15 mins + + # if load failed, then send notification email + #if database_scenario_id == 0 and int_hour > 96: + # str_request_check_result = self.sql_check_load_request(scenarioYear, path_no_drive, username, + # start_db_time) + # print(str_request_check_result) + # sys.exit(0) + # self.send_notification(str_request_check_result,username) #not working in server + #else: + # print(database_scenario_id) + # self.run_proc("DataSummary.bat", # get summary from database, added for auto-reporting + # [drive, path_no_drive, scenarioYear, database_scenario_id], + # "Data Summary") + + # self.run_proc("ExcelUpdate.bat", # forced to update excel links + # [drive, path_no_drive, scenarioYear, database_scenario_id], + # "Excel Update", + # capture_output=True) + + # terminate all java processes + _subprocess.call("taskkill /F /IM java.exe") + + # close all DOS windows + _subprocess.call("taskkill /F /IM cmd.exe") + + # UPLOAD DATA AND SWITCH PATHS + if useLocalDrive: + file_manager("UPLOAD", main_directory, username, scenario_id, + delete_local_files=not skipDeleteIntermediateFiles) + self._path = main_directory + drive, path_no_drive = os.path.splitdrive(self._path) + # self._path = main_directory + # drive, path_no_drive = os.path.splitdrive(self._path) + init_transit_db.add_database( + _eb.Emmebank(_join(main_directory, "emme_project", "Database_transit", "emmebank"))) + + if not skipDataLoadRequest: + start_db_time = datetime.datetime.now() # record the time to search for request id in the load request table, YMA, 1/23/2019 + # start_db_time = start_db_time + datetime.timedelta(minutes=0) + self.run_proc("DataLoadRequest.bat", + [drive + path_no_drive, end_iteration, scenarioYear, sample_rate[end_iteration - 1]], + "Data load request") + + # delete trip table files in iteration sub folder if model finishes without errors + if not useLocalDrive and not skipDeleteIntermediateFiles: + for msa_iteration in range(startFromIteration, end_iteration + 1): + self.delete_files( + ["auto*Trips*.omx", "tran*Trips*.omx", "nmot*.omx", "othr*.omx", "trip*.omx"], + _join(output_dir, "iter%s" % (msa_iteration))) + + # record run time + run_summary(path=self._path) + + def set_global_logbook_level(self, props): + self._log_level = props.get("RunModel.LogbookLevel", "ENABLED") + log_all = _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.VALUE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG + log_states = { + "ENABLED": log_all, + "DISABLE_ON_ERROR": log_all, + "NO_EXTERNAL_REPORTS": log_all, + "NO_REPORTS": _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, + "TITLES_ONLY": _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, + "DISABLED": _m.LogbookLevel.NONE, + } + _m.logbook_write("Setting logbook level to %s" % self._log_level) + try: + _m.logbook_level(log_states[self._log_level]) + except KeyError: + raise Exception("properties.RunModel.LogLevel: value must be one of %s" % ",".join(log_states.keys())) + + def run_traffic_assignments(self, base_scenario, period_ids, msa_iteration, relative_gap, + max_assign_iterations, num_processors, select_link=None, + makeFinalHighwayAssignmentStochastic=False, input_dir=None): + modeller = _m.Modeller() + traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") + export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") + output_dir = _join(self._path, "output") + main_emmebank = base_scenario.emmebank + + machine_name = _socket.gethostname().lower() + with open(_join(self._path, "conf", "server-config.csv")) as f: + columns = f.next().split(",") + for line in f: + values = dict(zip(columns, line.split(","))) + name = values["ServerName"].lower() + if name == machine_name: + server_config = values + break + else: + _m.logbook_write("Warning: current machine name not found in " + "conf\\server-config.csv ServerName column") + server_config = {"SNODE": "yes"} + distributed = server_config["SNODE"] == "no" + if distributed and not makeFinalHighwayAssignmentStochastic: + scen_map = dict((p, main_emmebank.scenario(n)) for n, p in period_ids) + input_args = { + "msa_iteration": msa_iteration, + "relative_gap": relative_gap, + "max_assign_iterations": max_assign_iterations, + "select_link": select_link + } + + periods_node1 = ["PM", "MD"] + input_args["num_processors"] = server_config["THREADN1"], + database_path1, skim_names1 = self.setup_remote_database( + [scen_map[p] for p in periods_node1], periods_node1, 1, msa_iteration) + self.start_assignments( + server_config["NODE1"], database_path1, periods_node1, scen_map, input_args) + + periods_node2 = ["AM"] + input_args["num_processors"] = server_config["THREADN2"] + database_path2, skim_names2 = self.setup_remote_database( + [scen_map[p] for p in periods_node2], periods_node2, 2, msa_iteration) + self.start_assignments( + server_config["NODE2"], database_path2, periods_node2, scen_map, input_args) + + try: + # run assignments locally + periods_local = ["EA", "EV"] + for period in periods_local: + local_scenario = scen_map[period] + traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, + num_processors, local_scenario, select_link) + omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) + if msa_iteration <= 4: + export_traffic_skims(period, omx_file, base_scenario) + scenarios = { + database_path1: [scen_map[p] for p in periods_node1], + database_path2: [scen_map[p] for p in periods_node2] + } + skim_names = { + database_path1: skim_names1, database_path2: skim_names2 + } + self.wait_and_copy([database_path1, database_path2], scenarios, skim_names) + except: + # Note: this will kill ALL python processes - not suitable if servers are being + # used for other tasks + _subprocess.call("taskkill /F /T /S \\\\%s /IM python.exe" % server_config["NODE1"]) + _subprocess.call("taskkill /F /T /S \\\\%s /IM python.exe" % server_config["NODE2"]) + raise + else: + for number, period in period_ids: + period_scenario = main_emmebank.scenario(number) + traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, + num_processors, period_scenario, select_link, stochastic=makeFinalHighwayAssignmentStochastic, input_directory=input_dir) + omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) + if msa_iteration <= 4: + export_traffic_skims(period, omx_file, base_scenario) + + def run_proc(self, name, arguments, log_message, capture_output=False): + path = _join(self._path, "bin", name) + if not os.path.exists(path): + raise Exception("No command / batch file '%s'" % path) + command = path + " " + " ".join([str(x) for x in arguments]) + attrs = {"command": command, "name": name, "arguments": arguments} + with _m.logbook_trace(log_message, attributes=attrs): + if capture_output and self._log_level != "NO_EXTERNAL_REPORTS": + report = _m.PageBuilder(title="Process run %s" % name) + report.add_html('Command:

%s

' % command) + # temporary file to capture output error messages generated by Java + err_file_ref, err_file_path = _tempfile.mkstemp(suffix='.log') + err_file = os.fdopen(err_file_ref, "w") + try: + output = _subprocess.check_output(command, stderr=err_file, cwd=self._path, shell=True) + report.add_html('Output:

%s
' % output) + except _subprocess.CalledProcessError as error: + report.add_html('Output:

%s
' % error.output) + raise + finally: + err_file.close() + with open(err_file_path, 'r') as f: + error_msg = f.read() + os.remove(err_file_path) + if error_msg: + report.add_html('Error message(s):

%s
' % error_msg) + try: + # No raise on writing report error + # due to observed issue with runs generating reports which cause + # errors when logged + _m.logbook_write("Process run %s report" % name, report.render()) + except Exception as error: + print _time.strftime("%Y-%M-%d %H:%m:%S") + print "Error writing report '%s' to logbook" % name + print error + print _traceback.format_exc(error) + if self._log_level == "DISABLE_ON_ERROR": + _m.logbook_level(_m.LogbookLevel.NONE) + else: + _subprocess.check_call(command, cwd=self._path, shell=True) + + @_m.logbook_trace("Check free space on C") + def check_free_space(self, min_space): + path = "c:\\" + temp, total, free = _ctypes.c_ulonglong(), _ctypes.c_ulonglong(), _ctypes.c_ulonglong() + if sys.version_info >= (3,) or isinstance(path, unicode): + fun = _ctypes.windll.kernel32.GetDiskFreeSpaceExW + else: + fun = _ctypes.windll.kernel32.GetDiskFreeSpaceExA + ret = fun(path, _ctypes.byref(temp), _ctypes.byref(total), _ctypes.byref(free)) + if ret == 0: + raise _ctypes.WinError() + total = total.value / (1024.0 ** 3) + free = free.value / (1024.0 ** 3) + if free < min_space: + raise Exception("Free space on C drive %s is less than %s" % (free, min_space)) + + def remove_prev_iter_files(self, file_names, output_dir, iteration): + if iteration == 0: + self.delete_files(file_names, output_dir) + else: + self.move_files(file_names, output_dir, _join(output_dir, "iter%s" % (iteration))) + + def copy_files(self, file_names, from_dir, to_dir): + with _m.logbook_trace("Copy files %s" % ", ".join(file_names)): + for file_name in file_names: + from_file = _join(from_dir, file_name) + _shutil.copy(from_file, to_dir) + + def complete_work(self, scenarioYear, input_dir, output_dir, input_file, output_file): + + fullList = np.array(pd.read_csv(_join(input_dir, input_file))['mgra']) + workList = np.array(pd.read_csv(_join(output_dir, output_file))['i']) + + list_set = set(workList) + unique_list = (list(list_set)) + notMatch = [x for x in fullList if x not in unique_list] + + if notMatch: + out_file = _join(output_dir, output_file) + with open(out_file, 'ab') as csvfile: + spamwriter = csv.writer(csvfile) + # spamwriter.writerow([]) + for item in notMatch: + # pdb.set_trace() + spamwriter.writerow([item, item, '30', '30', '30']) + + def move_files(self, file_names, from_dir, to_dir): + with _m.logbook_trace("Move files %s" % ", ".join(file_names)): + if not os.path.exists(to_dir): + os.mkdir(to_dir) + for file_name in file_names: + all_files = _glob.glob(_join(from_dir, file_name)) + for path in all_files: + try: + dst_file = _join(to_dir, os.path.basename(path)) + if os.path.exists(dst_file): + os.remove(dst_file) + _shutil.move(path, to_dir) + except Exception as error: + _m.logbook_write( + "Error moving file %s" % path, {"error": _traceback.format_exc(error)}) + + def delete_files(self, file_names, directory): + with _m.logbook_trace("Delete files %s" % ", ".join(file_names)): + for file_name in file_names: + all_files = _glob.glob(_join(directory, file_name)) + for path in all_files: + os.remove(path) + + def check_for_fatal(self, file_name, error_msg): + with open(file_name, 'a+') as f: + for line in f: + if "FATAL" in line: + raise Exception(error_msg) + + def set_active(self, emmebank): + modeller = _m.Modeller() + desktop = modeller.desktop + data_explorer = desktop.data_explorer() + for db in data_explorer.databases(): + if _norm(db.path) == _norm(unicode(emmebank)): + db.open() + return db + return None + + def parse_availability_file(self, file_path, periods): + if os.path.exists(file_path): + availabilities = _defaultdict(lambda: _defaultdict(lambda: dict())) + # NOTE: CSV Reader sets the field names to UPPERCASE for consistency + with gen_utils.CSVReader(file_path) as r: + for row in r: + name = row.pop("FACILITY_NAME") + class_name = row.pop("VEHICLE_CLASS") + for period in periods: + is_avail = int(row[period + "_AVAIL"]) + if is_avail not in [1, 0]: + msg = "Error processing file '%s': value for period %s class %s facility %s is not 1 or 0" + raise Exception(msg % (file_path, period, class_name, name)) + availabilities[period][name][class_name] = is_avail + else: + availabilities = None + return availabilities + + def apply_availabilities(self, period, scenario, availabilities): + if availabilities is None: + return + + network = scenario.get_network() + hov2 = network.mode("h") + hov2_trnpdr = network.mode("H") + hov3 = network.mode("i") + hov3_trnpdr = network.mode("I") + sov = network.mode("s") + sov_trnpdr = network.mode("S") + heavy_trk = network.mode("v") + heavy_trk_trnpdr = network.mode("V") + medium_trk = network.mode("m") + medium_trk_trnpdr = network.mode("M") + light_trk = network.mode("t") + light_trk_trnpdr = network.mode("T") + + class_mode_map = { + "DA": set([sov_trnpdr, sov]), + "S2": set([hov2_trnpdr, hov2]), + "S3": set([hov3_trnpdr, hov3]), + "TRK_L": set([light_trk_trnpdr, light_trk]), + "TRK_M": set([medium_trk_trnpdr, medium_trk]), + "TRK_H": set([heavy_trk_trnpdr, heavy_trk]), + } + report = ["
Link mode changes
"] + for name, class_availabilities in availabilities[period].iteritems(): + report.append("
%s
" % name) + changes = _defaultdict(lambda: 0) + for link in network.links(): + if name in link["#name"]: + for class_name, is_avail in class_availabilities.iteritems(): + modes = class_mode_map[class_name] + if is_avail == 1 and not modes.issubset(link.modes): + link.modes |= modes + changes["added %s to" % class_name] += 1 + elif is_avail == 0 and modes.issubset(link.modes): + link.modes -= modes + changes["removed %s from" % class_name] += 1 + report.append("
    ") + for x in changes.iteritems(): + report.append("
  • %s %s links
  • " % x) + report.append("
") + scenario.publish_network(network) + + title = "Apply global class availabilities by faclity name for period %s" % period + log_report = _m.PageBuilder(title=title) + for item in report: + log_report.add_html(item) + _m.logbook_write(title, log_report.render()) + + def setup_remote_database(self, src_scenarios, periods, remote_num, msa_iteration): + with _m.logbook_trace("Set up remote database #%s for %s" % (remote_num, ", ".join(periods))): + init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") + create_function = _m.Modeller().tool("inro.emme.data.function.create_function") + src_emmebank = src_scenarios[0].emmebank + remote_db_dir = _join(self._path, "emme_project", "Database_remote" + str(remote_num)) + if msa_iteration == 1: + # Create and initialize database at first iteration, overwrite existing + if os.path.exists(remote_db_dir): + _shutil.rmtree(remote_db_dir) + _time.sleep(1) + os.mkdir(remote_db_dir) + dimensions = src_emmebank.dimensions + dimensions["scenarios"] = len(src_scenarios) + remote_emmebank = _eb.create(_join(remote_db_dir, "emmebank"), dimensions) + try: + remote_emmebank.title = src_emmebank.title + remote_emmebank.coord_unit_length = src_emmebank.coord_unit_length + remote_emmebank.unit_of_length = src_emmebank.unit_of_length + remote_emmebank.unit_of_cost = src_emmebank.unit_of_cost + remote_emmebank.unit_of_energy = src_emmebank.unit_of_energy + remote_emmebank.use_engineering_notation = src_emmebank.use_engineering_notation + remote_emmebank.node_number_digits = src_emmebank.node_number_digits + + for src_scen in src_scenarios: + remote_scen = remote_emmebank.create_scenario(src_scen.id) + remote_scen.title = src_scen.title + for attr in sorted(src_scen.extra_attributes(), key=lambda x: x._id): + dst_attr = remote_scen.create_extra_attribute( + attr.type, attr.name, attr.default_value) + dst_attr.description = attr.description + for field in src_scen.network_fields(): + remote_scen.create_network_field( + field.type, field.name, field.atype, field.description) + remote_scen.has_traffic_results = src_scen.has_traffic_results + remote_scen.has_transit_results = src_scen.has_transit_results + remote_scen.publish_network(src_scen.get_network()) + for function in src_emmebank.functions(): + create_function(function.id, function.expression, remote_emmebank) + init_matrices(["traffic_skims", "traffic_demand"], periods, remote_scen) + finally: + remote_emmebank.dispose() + + src_scen = src_scenarios[0] + with _m.logbook_trace("Copy demand matrices to remote database"): + with _eb.Emmebank(_join(remote_db_dir, "emmebank")) as remote_emmebank: + demand_matrices = init_matrices.get_matrix_names("traffic_demand", periods, src_scen) + for matrix_name in demand_matrices: + matrix = remote_emmebank.matrix(matrix_name) + src_matrix = src_emmebank.matrix(matrix_name) + if matrix.type == "SCALAR": + matrix.data = src_matrix.data + else: + matrix.set_data(src_matrix.get_data(src_scen.id), src_scen.id) + skim_matrices = init_matrices.get_matrix_names("traffic_skims", periods, src_scen) + return remote_db_dir, skim_matrices + + def start_assignments(self, machine, database_path, periods, scenarios, assign_args): + with _m.logbook_trace("Start remote process for traffic assignments %s" % (", ".join(periods))): + assign_args["database_path"] = database_path + end_path = _join(database_path, "finish") + if os.path.exists(end_path): + os.remove(end_path) + for period in periods: + assign_args["period_scenario"] = scenarios[period].id + assign_args["period"] = period + with open(_join(database_path, "start_%s.args" % period), 'w') as f: + _json.dump(assign_args, f, indent=4) + script_dir = _join(self._path, "python") + bin_dir = _join(self._path, "bin") + args = [ + 'start %s\\PsExec.exe' % bin_dir, + '-c', + '-f', + '\\\\%s' % machine, + '-u \%s' % self.username, + '-p %s' % self.password, + "-d", + '%s\\emme_python.bat' % bin_dir, + "T:", + self._path, + '%s\\remote_run_traffic.py' % script_dir, + database_path, + ] + command = " ".join(args) + p = _subprocess.Popen(command, shell=True) + + @_m.logbook_trace("Wait for remote assignments to complete and copy results") + def wait_and_copy(self, database_dirs, scenarios, matrices): + database_dirs = database_dirs[:] + wait = True + while wait: + _time.sleep(5) + for path in database_dirs: + end_path = _join(path, "finish") + if os.path.exists(end_path): + database_dirs.remove(path) + _time.sleep(2) + self.check_for_fatal( + end_path, "error during remote run of traffic assignment. " + "Check logFiles/traffic_assign_database_remote*.log") + self.copy_results(path, scenarios[path], matrices[path]) + if not database_dirs: + wait = False + + @_m.logbook_trace("Copy skim results from remote database", save_arguments=True) + def copy_results(self, database_path, scenarios, matrices): + with _eb.Emmebank(_join(database_path, "emmebank")) as remote_emmebank: + for dst_scen in scenarios: + remote_scen = remote_emmebank.scenario(dst_scen.id) + # Create extra attributes and network fields which do not exist + for attr in sorted(remote_scen.extra_attributes(), key=lambda x: x._id): + if not dst_scen.extra_attribute(attr.name): + dst_attr = dst_scen.create_extra_attribute( + attr.type, attr.name, attr.default_value) + dst_attr.description = attr.description + for field in remote_scen.network_fields(): + if not dst_scen.network_field(field.type, field.name): + dst_scen.create_network_field( + field.type, field.name, field.atype, field.description) + dst_scen.has_traffic_results = remote_scen.has_traffic_results + dst_scen.has_transit_results = remote_scen.has_transit_results + + dst_scen.publish_network(remote_scen.get_network()) + + dst_emmebank = dst_scen.emmebank + scen_id = dst_scen.id + for matrix_id in matrices: + src_matrix = remote_emmebank.matrix(matrix_id) + dst_matrix = dst_emmebank.matrix(matrix_id) + dst_matrix.set_data(src_matrix.get_data(scen_id), scen_id) + + @_m.method(return_type=unicode) + def get_link_attributes(self): + export_utils = _m.Modeller().module("inro.emme.utility.export_utilities") + return export_utils.get_link_attributes(_m.Modeller().scenario) + + def sql_select_scenario(self, year, iteration, sample, path, dbtime): # YMA, 1/24/2019 + """Return scenario_id from [dimension].[scenario] given path""" + + import datetime + + sql_con = pyodbc.connect(driver='{SQL Server}', + server='sql2014a8', + database='abm_2', + trusted_connection='yes') + + # dbtime = dbtime + datetime.timedelta(days=0) + + df = pd.read_sql_query( + sql=("SELECT [scenario_id] " + "FROM [dimension].[scenario]" + "WHERE [year] = ? AND [iteration] = ? AND [sample_rate]= ? AND [path] Like ('%' + ? + '%') AND [date_loaded] > ? "), + con=sql_con, + params=[year, iteration, sample, path, dbtime] + ) + + if len(df) > 0: + return (df.iloc[len(df) - 1]['scenario_id']) + else: + return 0 + + def sql_check_load_request(self, year, path, user, ldtime): # YMA, 1/24/2019 + """Return information from [data_load].[load_request] given path,username,and requested time""" + + import datetime + + t0 = ldtime + datetime.timedelta(minutes=-1) + t1 = t0 + datetime.timedelta(minutes=30) + + sql_con = pyodbc.connect(driver='{SQL Server}', + server='sql2014a8', + database='abm_2', + trusted_connection='yes') + + df = pd.read_sql_query( + sql=( + "SELECT [load_request_id],[year],[name],[path],[iteration],[sample_rate],[abm_version],[user_name],[date_requested],[loading],[loading_failed],[scenario_id] " + "FROM [data_load].[load_request] " + "WHERE [year] = ? AND [path] LIKE ('%' + ? + '%') AND [user_name] LIKE ('%' + ? + '%') AND [date_requested] >= ? AND [date_requested] <= ? "), + con=sql_con, + params=[year, path, user, t0, t1] + ) + + if len(df) > 0: + return "You have successfully made the loading request, but the loading to the database failed. \r\nThe information is below. \r\n\r\n" + df.to_string() + else: + return "The data load request was not successfully made, please double check the [data_load].[load_request] table to confirm." + + +''' + def send_notification(self,str_message,user): # YMA, 1/24/2019, not working on server + """automate to send email notification if load request or loading failed""" + + import win32com.client as win32 + + outlook = win32.Dispatch('outlook.application') + Msg = outlook.CreateItem(0) + Msg.To = user + '@sandag.org' + Msg.CC = 'yma@sandag.org' + Msg.Subject = 'Loading Scenario to Database Failed' + Msg.body = str_message + '\r\n' + '\r\n' + 'This email alert is auto generated.\r\n' + 'Please do not respond.\r\n' + Msg.send''' diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py new file mode 100644 index 0000000..c271e56 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py @@ -0,0 +1,197 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/commercial_vehicle/distribution.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Distributes the total daily trips from production and attraction vectors. +# Friction factors are calculated based on a blended travel time skim of +# 1/3 AM_SOV_NT_M_TIME and 2/3 MD_SOV_NT_M_TIME, and a table of friction factor +# lookup values from commVehFF.csv. +# +# Inputs: +# input_directory: source directory for input files +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# input/commVehFF.csv +# +# Matrix inputs: +# moCOMVEH_PROD, mdCOMVEH_ATTR +# mfAM_SOV_NT_M_TIME, mfMD_SOV_NT_M_TIME +# +# Matrix intermediates (only used internally): +# mfCOMVEH_BLENDED_SKIM, mfCOMVEH_FRICTION +# +# Matrix results: +# mfCOMVEH_TOTAL_DEMAND +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(project_dir, "input") + base_scenario = modeller.scenario + distribution = modeller.tool("sandag.model.commercial_vehicle.distribution") + distribution(input_dir, base_scenario) +""" + + +TOOLBOX_ORDER = 53 + + +import inro.modeller as _m +import traceback as _traceback + +import pandas as pd +import os +import numpy as np + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class CommercialVehicleDistribution(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.attributes = ["input_directory"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Commercial Vehicle Distribution" + pb.description = """ +
+ Calculates total daily trips. + The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. +
+ A simple gravity model is used to distribute the truck trips. + A blended travel time of + 1/3 AM_SOV_NT_M_TIME and 2/3 MD_SOV_NT_M_TIME is used, along with + friction factor lookup table stored in commVehFF.csv. +
+ Input: +
  • + (1) Level-of-service matrices for the AM peak period (6 am to 10 am) 'mfAM_SOVGPM_TIME' + and midday period (10 am to 3 pm) 'mfMD_SOVGPM_TIME' + which contain congested travel time (in minutes). +
  • + (2) Trip generation results 'moCOMVEH_PROD' and 'mdCOMVEH_ATTR' +
  • + (4) A table of friction factors in commVehFF.csv with: +
    • + (a) impedance measure (blended travel time) index; +
    • + (b) friction factors +
    +
+ Output: +
  • + (1) A trip table matrix 'mfCOMVEH_TOTAL_DEMAND' + of daily class-specific truck trips for very small trucks. +
  • + (2) A blended travel time matrix 'mfCOMVEH_BLENDED_SKIM' +
+
""" + pb.branding_text = "- SANDAG - Model - Commercial vehicle" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Commercial vehicle distribution') + def __call__(self, input_directory, scenario): + attributes = {"input_directory": input_directory} + gen_utils.log_snapshot("Commercial vehicle distribution", str(self), attributes) + self.calc_blended_skims(scenario) + self.calc_friction_matrix(input_directory, scenario) + self.balance_matrix(scenario) + + @_m.logbook_trace('Calculate blended skims') + def calc_blended_skims(self, scenario): + matrix_calc = _m.Modeller().tool( + 'inro.emme.matrix_calculation.matrix_calculator') + spec = { + "expression": "0.3333 * mfAM_SOV_NT_M_TIME + 0.6666 * mfMD_SOV_NT_M_TIME", + "result": "mfCOMVEH_BLENDED_SKIM", + "type": "MATRIX_CALCULATION" + } + matrix_calc(spec, scenario=scenario) + + # Prevent intrazonal trips + spec = { + "expression": "99999 * (p.eq.q) + mfCOMVEH_BLENDED_SKIM * (p.ne.q)", + "result": "mfCOMVEH_BLENDED_SKIM", + "type": "MATRIX_CALCULATION" + } + matrix_calc(spec, scenario=scenario) + + @_m.logbook_trace('Calculate friction factor matrix') + def calc_friction_matrix(self, input_directory, scenario): + emmebank = scenario.emmebank + + imp_matrix = emmebank.matrix('mfCOMVEH_BLENDED_SKIM') + friction_matrix = emmebank.matrix('mfCOMVEH_FRICTION') + imp_array = imp_matrix.get_numpy_data(scenario_id=scenario.id) + + # create the vector function to bin the impedances and get friction values + friction_table = pd.read_csv( + os.path.join(input_directory, 'commVehFF.csv'), + header=None, names=['index', 'factors']) + + factors_array = friction_table.factors.values + max_index = len(factors_array) - 1 + + # interpolation: floor + values_array = np.clip(np.floor(imp_array).astype(int) - 1, 0, max_index) + friction_array = np.take(factors_array, values_array) + friction_matrix.set_numpy_data(friction_array, scenario_id=scenario.id) + + def balance_matrix(self, scenario): + balance = _m.Modeller().tool( + 'inro.emme.matrix_calculation.matrix_balancing') + spec = { + "type": "MATRIX_BALANCING", + "od_values_to_balance": "mfCOMVEH_FRICTION", + "origin_totals": "moCOMVEH_PROD", + "destination_totals": "mdCOMVEH_ATTR", + "results": { + "od_balanced_values": "mfCOMVEH_TOTAL_DEMAND", + }, + "max_iterations": 100, + "max_relative_error": 0.001 + } + balance(spec, scenario=scenario) diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py new file mode 100644 index 0000000..9842de1 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py @@ -0,0 +1,187 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/commercial_vehicle/generation.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the commercial vehicle generation step, calculate commercial vehicle +# productions and attractions. Linear regression models generate trip ends, +# balancing attractions to productions. +# +# Inputs: +# input_directory: source directory for most input files, including demographics and trip rates +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file +# input/mgra13_based_inputYEAR.csv +# +# Matrix results: +# moCOMVEH_PROD, mdCOMVEH_ATTR +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(project_dir, "input") + base_scenario = modeller.scenario + generation = modeller.tool("sandag.model.commercial_vehicle.generation") + generation(input_dir, base_scenario) +""" + + +TOOLBOX_ORDER = 52 + + +import inro.modeller as _m +import traceback as _traceback + +import pandas as pd +import os + + +dem_utils = _m.Modeller().module('sandag.utilities.demand') +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class CommercialVehicleDistribution(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.attributes = ["input_directory"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Commercial Vehicle Generation" + pb.description = """ +
+ Calculate commerical vehicle productions and attractions + based on mgra13_based_inputYYYY.csv. + The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. + + Linear regression models generate trip ends, balancing attractions to productions. +
+ Input: MGRA file in CSV format with the following fields: +
    +
  • + (a) TOTEMP, total employment (same regardless of classification system); +
  • + (b) RETEMPN, retail trade employment per the NAICS classification system; +
  • + (c) FPSEMPN, financial and professional services employment per the NAICS classification system; +
  • + (d) HEREMPN, health, educational, and recreational employment per the NAICS classification system; +
  • + (e) OTHEMPN, other employment per the NAICS classification system; +
  • + (f) AGREMPN, agricultural employmentper the NAICS classificatin system; +
  • + (g) MWTEMPN, manufacturing, warehousing, and transportation emp;loyment per the NAICS classification system; and, +
  • + (h) TOTHH, total households. +
+
+ Output: Trip productions and attractions in matrices 'moCOMMVEH_PROD' and 'mdCOMMVEH_ATTR' respectively. +
+ """ + + pb.branding_text = "- SANDAG - Model - Commercial vehicle" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Commercial vehicle generation') + def __call__(self, input_directory, scenario): + attributes = {"input_directory": input_directory} + gen_utils.log_snapshot("Commercial vehicle generation", str(self), attributes) + emmebank = scenario.emmebank + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + main_directory = os.path.dirname(input_directory) + props = load_properties( + os.path.join(main_directory, "conf", "sandag_abm.properties")) + year = props['scenarioYear'] + mgra = pd.read_csv( + os.path.join(input_directory, 'mgra13_based_input%s.csv' % year)) + + calibration = 1.4 + + mgra['RETEMPN'] = mgra.emp_retail + mgra.emp_personal_svcs_retail + mgra['FPSEMPN'] = mgra.emp_prof_bus_svcs + mgra['HEREMPN'] = mgra.emp_health + mgra.emp_pvt_ed_k12 \ + + mgra.emp_pvt_ed_post_k12_oth + mgra.emp_amusement + mgra['AGREMPN'] = mgra.emp_ag + mgra['MWTEMPN'] = mgra.emp_const_non_bldg_prod \ + + mgra.emp_const_bldg_prod + mgra.emp_mfg_prod \ + + mgra.emp_trans + mgra['MILITARY'] = mgra.emp_fed_mil + mgra['TOTEMP'] = mgra.emp_total + mgra['OTHEMPN'] = mgra.TOTEMP - (mgra.RETEMPN + mgra.FPSEMPN + + mgra.HEREMPN + mgra.AGREMPN + + mgra.MWTEMPN + mgra.MILITARY) + mgra['TOTHH'] = mgra.hh + + mgra['PROD'] = calibration * ( + 0.95409 * mgra.RETEMPN + 0.54333 * mgra.FPSEMPN + + 0.50769 * mgra.HEREMPN + 0.63558 * mgra.OTHEMPN + + 1.10181 * mgra.AGREMPN + 0.81576 * mgra.MWTEMPN + + 0.15000 * mgra.MILITARY + 0.1 * mgra.TOTHH) + mgra['ATTR'] = mgra.PROD + + # Adjustment to match military CTM trips to match gate counts + military_ctm_adjustment = props["RunModel.militaryCtmAdjustment"] + if military_ctm_adjustment: + mgra_m = pd.read_csv(os.path.join( + input_directory, 'cvm_military_adjustment.csv')) + mgra = pd.merge(mgra, mgra_m, on='mgra', how='outer') + mgra.fillna(1, inplace=True) + mgra['PROD'] = mgra['PROD'] * mgra['scale'] + mgra['ATTR'] = mgra['ATTR'] * mgra['scale'] + + taz = mgra.groupby('taz').sum() + taz.reset_index(inplace=True) + taz = dem_utils.add_missing_zones(taz, scenario) + taz.sort('taz', inplace=True) + + prod = emmebank.matrix('moCOMVEH_PROD') + attr = emmebank.matrix('mdCOMVEH_ATTR') + prod.set_numpy_data(taz.PROD.values, scenario) + attr.set_numpy_data(taz.ATTR.values, scenario) + + return taz[['taz', 'PROD', 'ATTR']] diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py new file mode 100644 index 0000000..3899456 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py @@ -0,0 +1,122 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/commercial_vehicle/run_commercial_vehicle_model.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the commercial vehicle model, the generation, distribution, time-of-day, +# and toll diversion tools, in sequence. +# 1) Generates +# 2) Generates +# 3) Distributes trips based on congested skims +# 4) Time-of-day split +# 5) Applies toll diversion model with toll and non-toll SOV skims +# +# The very small truck generation model is based on the Phoenix +# four-tire truck model documented in the TMIP Quick Response Freight Manual. +# +# Inputs: +# run_generation: boolean, if True run generation tool. +# input_directory: source directory for most input files, including demographics and trip rates +# (see generation and distribtuion tools) +# scenario: traffic scenario to use for reference zone system +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(project_dir, "input") + base_scenario = modeller.scenario + run_truck = modeller.tool("sandag.model.commercial_vehicle.run_commercial_vehicle") + run_truck(True, input_dir, base_scenario) +""" + +TOOLBOX_ORDER = 51 + + +import inro.modeller as _m +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class CommercialVehicleModel(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + run_generation = _m.Attribute(bool) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.run_generation = True + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.attributes = ["input_directory", "run_generation"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Commercial vehicle model" + pb.description = """ +
+ Run the 4 steps of the commercial vehicle model: generation, distribution, + time of day, toll diversion. + + The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. +
+""" + pb.branding_text = "- SANDAG - Model - Commercial vehicle" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_checkbox("run_generation", title=" ", label="Run generation (first iteration)") + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.run_generation, self.input_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Commercial vehicle model', save_arguments=True) + def __call__(self, run_generation, input_directory, scenario): + attributes = {"run_generation": run_generation, "input_directory": input_directory} + gen_utils.log_snapshot("Commercial vehicle model", str(self), attributes) + generation = _m.Modeller().tool( + 'sandag.model.commercial_vehicle.generation') + distribution = _m.Modeller().tool( + 'sandag.model.commercial_vehicle.distribution') + time_of_day = _m.Modeller().tool( + 'sandag.model.commercial_vehicle.time_of_day') + diversion = _m.Modeller().tool( + 'sandag.model.commercial_vehicle.toll_diversion') + if run_generation: + generation(input_directory, scenario) + distribution(input_directory, scenario) + time_of_day(scenario) + diversion(scenario) diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py new file mode 100644 index 0000000..163bba7 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py @@ -0,0 +1,99 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/commercial_vehicle/time_of_day.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Applies time-of-day factoring to the Commercial vehicle total daily demand. +# The diurnal factors are taken from the BAYCAST-90 model with adjustments +# made during calibration to the very small truck values to better match counts. +# +# Inputs: +# scenario: traffic scenario to use for reference zone system +# +# Matrix inputs: +# mfCOMVEH_TOTAL_DEMAND +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# mfpp_COMVEH +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + base_scenario = modeller.scenario + time_of_day = modeller.tool("sandag.model.commercial_vehicle.time_of_day") + time_of_day(base_scenario) +""" + +TOOLBOX_ORDER = 54 + + +import inro.modeller as _m +import traceback as _traceback + + +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class TimeOfDay(_m.Tool()): + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Commercial Vehicle Time of Day split" + pb.description = """ +
+ Commercial vehicle time-of-day factoring. + The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. +
+ The diurnal factors are taken from the BAYCAST-90 model with adjustments + made during calibration to the very small truck values to better match counts. +

Input: A production/attraction format trip table matrix of daily very small truck trips.

+

Output: Five, time-of-day-specific trip table matrices for very small trucks, + of the form 'mfpp_COMVEH'. +

+
""" + pb.branding_text = "- SANDAG - Model - Commercial vehicle" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Commercial vehicle Time of Day split') + def __call__(self, scenario): + matrix_calc = dem_utils.MatrixCalculator(scenario, 0) + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + period_factors = [0.0235, 0.1, 0.5080, 0.1980, 0.1705] + for p, f in zip(periods, period_factors): + matrix_calc.add( + "mf%s_COMVEH" % p, + "%s * 0.5 * (mfCOMVEH_TOTAL_DEMAND + mfCOMVEH_TOTAL_DEMAND')" % f) + matrix_calc.run() diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py new file mode 100644 index 0000000..07bc4fe --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py @@ -0,0 +1,119 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/commercial_vehicle/toll_diversion.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# Applies toll and non-toll split to Commercial vehicle time period demand. +# Uses the travel TIME for GP and TOLL modes as well as the TOLLCOST +# by time period. +# +# Inputs: +# scenario: traffic scenario to use for reference zone system +# +# Matrix inputs: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# mfpp_COMVEH +# mfpp_SOVGPM_TIME, mfpp_SOVTOLLM_TIME, mfpp_SOVTOLLM_TOLLCOST +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# mfpp_COMVEHGP, mfpp_COMVEHTOLL +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + base_scenario = modeller.scenario + toll_diversion = modeller.tool("sandag.model.commercial_vehicle.toll_diversion") + toll_diversion(base_scenario) +""" + +TOOLBOX_ORDER = 55 + + +import inro.modeller as _m +import traceback as _traceback + + +gen_utils = _m.Modeller().module('sandag.utilities.general') +dem_utils = _m.Modeller().module('sandag.utilities.demand') + + +class TollDiversion(_m.Tool()): + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Commercial vehicle toll diversion" + pb.description = """ +
+ Commercial vehicle toll and non-toll (GP) split. + The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. +
+

Input: Time-of-day-specific trip table matrices 'mfpp_COMVEH', + and travel time for GP and TOLL modes 'mfpp_SOVGPM_TIME', 'mfpp_SOVTOLLM_TIME', + and toll cost 'mfpp_SOVTOLLM_TOLLCOST' (medium VOT bin). +

+

Output: Corresponding time-of-day 'mfpp_COMVEHGP' and 'mfpp_COMVEHTOLL' + trip demand matrices.

+
+""" + pb.branding_text = "- SANDAG - Model - Commercial vehicle" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Commercial vehicle toll diversion') + def __call__(self, scenario): + emmebank = scenario.emmebank + matrix_calc = dem_utils.MatrixCalculator(scenario, "MAX-1") + init_matrix = _m.Modeller().tool( + "inro.emme.data.matrix.init_matrix") + + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + for p in periods: + init_matrix("mf%s_COMVEHTOLL" % p, scenario=scenario) + + nest = 10 + vot = 0.02 + toll_factor = 1 + for p in periods: + with matrix_calc.trace_run("Diversion for %s" % p): + init_matrix("mf%s_COMVEHTOLL" % p, scenario=scenario) + params = {'p': p, 'v': vot, 'tf': toll_factor, 'n': nest} + utility = ('(mf%(p)s_SOVGPM_TIME - mf%(p)s_SOVTOLLM_TIME' + '- %(v)s * mf%(p)s_SOVTOLLM_TOLLCOST * %(tf)s) / %(n)s') % params + matrix_calc.add( + "mf%s_COMVEHTOLL" % p, + "mf%s_COMVEH / (1 + exp(- %s))" % (p, utility), + ["mf%s_SOVTOLLM_TOLLCOST" % p, 0, 0, "EXCLUDE"]) + matrix_calc.add( + "mf%s_COMVEHGP" % p, "mf%(p)s_COMVEH - mf%(p)s_COMVEHTOLL" % {'p': p}) diff --git a/sandag_abm/src/main/emme/toolbox/model/external_external.py b/sandag_abm/src/main/emme/toolbox/model/external_external.py new file mode 100644 index 0000000..c928d36 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/external_external.py @@ -0,0 +1,190 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/external_external.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the external-external, cross-regional demand model. Imports the total +# daily demand from file and splits by time-of-day and SOVGP, HOV2HOV, and +# HOV3HOV classes using fixed factors. +# +# +# Inputs: +# input_directory: source directory for input file +# external_zones: the set of external zones specified as a range, default is "1-12" +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: YEAR is replaced by scenarioYear in the conf/sandag_abm.properties file +# input/mgra13_based_inputYEAR.csv +# input/externalInternalControlTotalsByYear.csv +# input/externalInternalControlTotals.csv +# (if externalInternalControlTotalsByYear.csv is unavailable) +# +# Matrix results: +# pp_SOV_EETRIPS, pp_HOV2_EETRIPS, pp_HOV3_EETRIPS +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(main_directory, "input") + external_zones = "1-12" + base_scenario = modeller.scenario + external_external = modeller.tool("sandag.model.external_external") + external_external(input_dir, external_zones, base_scenario) +""" + + +TOOLBOX_ORDER = 62 + + +import inro.modeller as _m + +import multiprocessing as _multiprocessing +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class ExternalExternal(_m.Tool(), gen_utils.Snapshot): + input_directory = _m.Attribute(unicode) + external_zones = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.external_zones = "1-12" + self.attributes = ["external_zones", "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "External external model" + pb.description = """ + Total trips are read from externalExternalTripsByYear.csv for + the year in sandag_abm.properties. If this file does not exist + externalExternalTrips.csv will be used instead. + The total trips are split by time-of-day and traffic class + SOVGP, HOV2HOV, and HOV3HOV using fixed factors. + """ + pb.branding_text = "- SANDAG - Model" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + pb.add_text_box("external_zones", title="External zones:") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, self.external_zones, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('External-external model', save_arguments=True) + def __call__(self, input_directory, external_zones, scenario): + attributes = { + "external_zones": external_zones, + "input_directory": input_directory, + } + gen_utils.log_snapshot("External-external model", str(self), attributes) + emmebank = scenario.emmebank + matrix_calc = _m.Modeller().tool( + "inro.emme.matrix_calculation.matrix_calculator") + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties( + os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) + year = int(props['scenarioYear']) + + periods = ["EA", "AM", "MD", "PM", "EV"] + time_of_day_factors = [0.074, 0.137, 0.472, 0.183, 0.133] + modes = ["SOV", "HOV2", "HOV3"] + mode_factors = [0.43, 0.42, 0.15] + + ee_matrix = emmebank.matrix("ALL_TOTAL_EETRIPS") + matrix_data = ee_matrix.get_data(scenario) + file_path = os.path.join( + input_directory, "externalExternalTripsByYear.csv") + if os.path.isfile(file_path): + with open(file_path, 'r') as f: + header = f.readline() + for line in f: + tyear, orig, dest, trips = line.split(",") + if int(tyear) == year: + matrix_data.set(int(orig), int(dest), float(trips)) + else: + file_path = os.path.join( + input_directory, "externalExternalTrips.csv") + if not os.path.isfile(file_path): + raise Exception("External-external model: no file 'externalExternalTrips.csv' or 'externalExternalTripsByYear.csv'") + with open(file_path, 'r') as f: + header = f.readline() + for line in f: + orig, dest, trips = line.split(",") + matrix_data.set(int(orig), int(dest), float(trips)) + _m.logbook_write("Control totals read from %s" % file_path) + ee_matrix.set_data(matrix_data, scenario) + + # factor for final demand matrix by time and mode type + # all external-external trips are non-toll + # SOV_GP, SR2_HOV SR3_HOV = "SOV", "HOV2", "HOV3" + for period, tod_fac in zip(periods, time_of_day_factors): + for mode, mode_fac in zip(modes, mode_factors): + spec = { + "expression": "ALL_TOTAL_EETRIPS * %s * %s" % (tod_fac, mode_fac), + "result": "mf%s_%s_EETRIPS" % (period, mode), + "constraint": { + "by_zone": { + "origins": external_zones, + "destinations": external_zones + } + }, + "type": "MATRIX_CALCULATION" + } + matrix_calc(spec, scenario=scenario) + + precision = float(props['RunModel.MatrixPrecision']) + self.matrix_rounding(scenario, precision) + + @_m.logbook_trace('Controlled rounding of demand') + def matrix_rounding(self, scenario, precision): + round_matrix = _m.Modeller().tool( + "inro.emme.matrix_calculation.matrix_controlled_rounding") + emmebank = scenario.emmebank + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + modes = ["SOV", "HOV2", "HOV3"] + for period in periods: + for mode in modes: + matrix = emmebank.matrix("mf%s_%s_EETRIPS" % (period, mode)) + report = round_matrix(demand_to_round=matrix, + rounded_demand=matrix, + min_demand=precision, + values_to_round="SMALLER_THAN_MIN", + scenario=scenario) + diff --git a/sandag_abm/src/main/emme/toolbox/model/external_internal.py b/sandag_abm/src/main/emme/toolbox/model/external_internal.py new file mode 100644 index 0000000..19203e6 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/external_internal.py @@ -0,0 +1,331 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/external_internal.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the external USA to internal demand model. +# 1) Work and non-work trip gateway total trips are read from control totals +# 2) Generates internal trip ends based on relative attractiveness from employment (by category) and households +# 3) Applies time-of-day and occupancy factors +# 4) Applies toll diversion model with toll and non-toll skims +# Control totals are read from externalInternalControlTotalsByYear.csv for +# the specified year in sandag_abm.properties. If this file does not exist +# externalInternalControlTotals.csv will be used instead. +# +# Inputs: +# input_directory: source directory for most input files, including demographics and trip rates +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: YEAR is replaced by scenarioYear in the conf/sandag_abm.properties file +# input/mgra13_based_inputYEAR.csv +# input/externalInternalControlTotalsByYear.csv +# input/externalInternalControlTotals.csv +# (if externalInternalControlTotalsByYear.csv is unavailable) +# +# Matrix results: +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(main_directory, "input") + base_scenario = modeller.scenario + external_internal = modeller.tool("sandag.model.external_internal") + external_internal(input_dir, input_truck_dir, base_scenario) +""" + +TOOLBOX_ORDER = 61 + + +import inro.modeller as _m +import numpy as np +import pandas as pd +import traceback as _traceback +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +class ExternalInternal(_m.Tool(), gen_utils.Snapshot): + input_directory = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.attributes = ["input_directory"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "External internal model" + pb.description = """ + Runs the external USA to internal demand model. + Control totals are read from externalInternalControlTotalsByYear.csv for + the specified year in sandag_abm.properties. If this file does not exist + externalInternalControlTotals.csv will be used instead.""" + pb.branding_text = "- SANDAG - Model" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('External-internal model', save_arguments=True) + def __call__(self, input_directory, scenario): + attributes = {"input_directory": input_directory} + gen_utils.log_snapshot("External-internal model", str(self), attributes) + np.seterr(divide='ignore', invalid='ignore') + + emmebank = scenario.emmebank + zones = scenario.zone_numbers + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties( + os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) + + year = int(props['scenarioYear']) + mgra = pd.read_csv( + os.path.join(input_directory, 'mgra13_based_input%s.csv' % year)) + + # Load data + file_path = os.path.join( + input_directory, "externalInternalControlTotalsByYear.csv") + if os.path.isfile(file_path): + control_totals = pd.read_csv(file_path) + control_totals = control_totals[control_totals.year==year] + control_totals = control_totals.drop("year", axis=1) + else: + file_path = os.path.join( + input_directory, 'externalInternalControlTotals.csv') + if not os.path.isfile(file_path): + raise Exception( + "External-internal model: no file 'externalInternalControlTotals.csv' " + "or 'externalInternalControlTotalsByYear.csv'") + control_totals = pd.read_csv(file_path) + _m.logbook_write("Control totals read from %s" % file_path) + + # Aggregate purposes + mgra['emp_blu'] = (mgra.emp_const_non_bldg_prod + + mgra.emp_const_non_bldg_office + + mgra.emp_utilities_prod + + mgra.emp_utilities_office + + mgra.emp_const_bldg_prod + + mgra.emp_const_bldg_office + + mgra.emp_mfg_prod + + mgra.emp_mfg_office + + mgra.emp_whsle_whs + + mgra.emp_trans) + + mgra['emp_svc'] = (mgra.emp_prof_bus_svcs + + mgra.emp_prof_bus_svcs_bldg_maint + + mgra.emp_personal_svcs_office + + mgra.emp_personal_svcs_retail) + + mgra['emp_edu'] = (mgra.emp_pvt_ed_k12 + + mgra.emp_pvt_ed_post_k12_oth + + mgra.emp_public_ed) + + mgra['emp_gov'] = (mgra.emp_state_local_gov_ent + + mgra.emp_fed_non_mil + + mgra.emp_fed_non_mil + + mgra.emp_state_local_gov_blue + + mgra.emp_state_local_gov_white) + + mgra['emp_ent'] = (mgra.emp_amusement + + mgra.emp_hotel + + mgra.emp_restaurant_bar) + + mgra['emp_oth'] = (mgra.emp_religious + + mgra.emp_pvt_hh + + mgra.emp_fed_mil) + + mgra['work_size'] = (mgra.emp_blu + + 1.364 * mgra.emp_retail + + 4.264 * mgra.emp_ent + + 0.781 * mgra.emp_svc + + 1.403 * mgra.emp_edu + + 1.779 * mgra.emp_health + + 0.819 * mgra.emp_gov + + 0.708 * mgra.emp_oth) + + mgra['non_work_size'] = (mgra.hh + + 1.069 * mgra.emp_blu + + 4.001 * mgra.emp_retail + + 6.274 * mgra.emp_ent + + 0.901 * mgra.emp_svc + + 1.129 * mgra.emp_edu + + 2.754 * mgra.emp_health + + 1.407 * mgra.emp_gov + + 0.304 * mgra.emp_oth) + + # aggregate to TAZ + taz = mgra[['taz', 'work_size', 'non_work_size']].groupby('taz').sum() + taz.reset_index(inplace=True) + taz = dem_utils.add_missing_zones(taz, scenario) + taz.sort_values('taz', ascending=True, inplace=True) # method sort was deprecated since pandas version 0.20.0, yma, 2/12/2019 + taz.reset_index(inplace=True, drop=True) + control_totals = pd.merge(control_totals, taz[['taz']], how='outer') + control_totals.sort_values('taz', inplace=True) # method sort was deprecated since pandas version 0.20.0, yma, 2/12/2019 + + length_skim = emmebank.matrix('mf"MD_SOV_TR_M_DIST"').get_numpy_data(scenario) + + # Compute probabilities for work purpose + wrk_dist_coef = -0.029 + wrk_prob = taz.work_size.values * np.exp(wrk_dist_coef * length_skim) + wrk_sum = np.sum(wrk_prob, 1) + wrk_prob = wrk_prob / wrk_sum[:, np.newaxis] + wrk_prob = np.nan_to_num(wrk_prob) + # Apply probabilities to control totals + wrk_pa_mtx = wrk_prob * control_totals.work.values[:, np.newaxis] + wrk_pa_mtx = np.nan_to_num(wrk_pa_mtx) + wrk_pa_mtx = wrk_pa_mtx.astype("float32") + + # compute probabilities for non work purpose + non_wrk_dist_coef = -0.006 + nwrk_prob = taz.non_work_size.values * np.exp(non_wrk_dist_coef * length_skim) + non_wrk_sum = np.sum(nwrk_prob, 1) + nwrk_prob = nwrk_prob / non_wrk_sum[:, np.newaxis] + nwrk_prob = np.nan_to_num(nwrk_prob) + # Apply probabilities to control totals + nwrk_pa_mtx = nwrk_prob * control_totals.nonwork.values[:, np.newaxis] + nwrk_pa_mtx = np.nan_to_num(nwrk_pa_mtx) + nwrk_pa_mtx = nwrk_pa_mtx.astype("float32") + + # Convert PA to OD and apply Diurnal Facotrs + wrk_ap_mtx = 0.5 * np.transpose(wrk_pa_mtx) + wrk_pa_mtx = 0.5 * wrk_pa_mtx + nwrk_ap_mtx = 0.5 * np.transpose(nwrk_pa_mtx) + nwrk_pa_mtx = 0.5 * nwrk_pa_mtx + + # Apply occupancy and diurnal factors + work_time_PA_factors = [0.26, 0.26, 0.41, 0.06, 0.02] + work_time_AP_factors = [0.08, 0.07, 0.41, 0.42, 0.02] + + nonwork_time_PA_factors = [0.25, 0.39, 0.30, 0.04, 0.02] + nonwork_time_AP_factors = [0.12, 0.11, 0.37, 0.38, 0.02] + + work_occupancy_factors = [0.58, 0.31, 0.11] + nonwork_occupancy_factors = [0.55, 0.29, 0.15] + + # value of time is in cents per minute (toll cost is in cents) + vot_work = 15.00 # $9.00/hr + vot_non_work = 22.86 # $13.70/hr + ivt_coef = -0.03 + + gp_modes = ["SOVGP", "HOV2HOV", "HOV3HOV"] + toll_modes = ["SOVTOLL", "HOV2TOLL", "HOV3TOLL"] + # TODO: the GP vs. TOLL distinction should be collapsed + # (all demand added to transponder demand in import_auto_demand) + skim_lookup = { + "SOVGP": "SOV_NT_M", + "HOV2HOV": "HOV2_M", + "HOV3HOV": "HOV3_M", + "SOVTOLL": "SOV_TR_M", + "HOV2TOLL": "HOV2_M", + "HOV3TOLL": "HOV3_M" + } + periods = ["EA", "AM", "MD", "PM", "EV"] + for p, w_d_pa, w_d_ap, nw_d_pa, nw_d_ap in zip( + periods, work_time_PA_factors, work_time_AP_factors, + nonwork_time_PA_factors, nonwork_time_AP_factors): + for gp_mode, toll_mode, w_o, nw_o in zip( + gp_modes, toll_modes, work_occupancy_factors, nonwork_occupancy_factors): + wrk_mtx = w_o * (w_d_pa * wrk_pa_mtx + w_d_ap * wrk_ap_mtx) + nwrk_mtx = nw_o * (nw_d_pa * nwrk_pa_mtx + nw_d_ap * nwrk_ap_mtx) + + # Toll choice split + f_tm_imp = emmebank.matrix('mf%s_%s_TIME' % (p, skim_lookup[gp_mode])).get_numpy_data(scenario) + t_tm_imp = emmebank.matrix('mf%s_%s_TIME' % (p, skim_lookup[toll_mode])).get_numpy_data(scenario) + t_cst_imp = emmebank.matrix('mf%s_%s_TOLLCOST' % (p, skim_lookup[toll_mode])).get_numpy_data(scenario) + + # Toll diversion for work purpose + # TODO: .mod no longer needed, to confirm + wrk_toll_prb = np.exp( + ivt_coef * (t_tm_imp - f_tm_imp + np.mod(t_cst_imp, 10000) / vot_work) - 3.39 + ) + wrk_toll_prb[t_cst_imp <= 0] = 0 + wrk_toll_prb = wrk_toll_prb / (1 + wrk_toll_prb) + work_matrix_toll = wrk_mtx * wrk_toll_prb + work_matrix_non_toll = wrk_mtx * (1 - wrk_toll_prb) + + toll_eiwork = emmebank.matrix('%s_%s_EIWORK' % (p, toll_mode)) + gp_ei_work = emmebank.matrix('%s_%s_EIWORK' % (p, gp_mode)) + toll_eiwork.set_numpy_data(work_matrix_toll, scenario) + gp_ei_work.set_numpy_data(work_matrix_non_toll, scenario) + + # Toll diversion for non work purpose + nwrk_toll_prb = np.exp( + ivt_coef * (t_tm_imp - f_tm_imp + np.mod(t_cst_imp, 10000) / vot_non_work) - 3.39 + ) + + nwrk_toll_prb[t_cst_imp <= 0] = 0 + nwrk_toll_prb = nwrk_toll_prb / (1 + nwrk_toll_prb) + + non_work_toll_matrix = nwrk_mtx * nwrk_toll_prb + non_work_gp_matrix = nwrk_mtx * (1 - nwrk_toll_prb) + + toll_einonwork = emmebank.matrix('%s_%s_EINONWORK' % (p, toll_mode)) + gp_einonwork = emmebank.matrix('%s_%s_EINONWORK' % (p, gp_mode)) + toll_einonwork.set_numpy_data(non_work_toll_matrix, scenario) + gp_einonwork.set_numpy_data(non_work_gp_matrix, scenario) + + precision = float(props['RunModel.MatrixPrecision']) + self.matrix_rounding(scenario, precision) + + @_m.logbook_trace('Controlled rounding of demand') + def matrix_rounding(self, scenario, precision): + round_matrix = _m.Modeller().tool( + "inro.emme.matrix_calculation.matrix_controlled_rounding") + emmebank = scenario.emmebank + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + modes = ["SOVGP", "HOV2HOV", "HOV3HOV", "SOVTOLL", "HOV2TOLL", "HOV3TOLL"] + purpose_types = ["EIWORK", "EINONWORK"] + for period in periods: + for mode in modes: + for purpose in purpose_types: + matrix = emmebank.matrix("mf%s_%s_%s" % (period, mode, purpose)) + try: + report = round_matrix(demand_to_round=matrix, + rounded_demand=matrix, + min_demand=precision, + values_to_round="SMALLER_THAN_MIN", + scenario=scenario) + except: + max_val = matrix.get_numpy_data(scenario.id).max() + if max_val == 0: + # if max_val is 0 the error is that the matrix is 0, log a warning + _m.logbook_write('Warning: matrix %s is all 0s' % matrix.named_id) + else: + raise diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py b/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py new file mode 100644 index 0000000..7a9207a --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py @@ -0,0 +1,288 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/truck/distribution.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the truck distribution step. Distributes truck trips with congested +# skims and splits by time of day. +# The distribution is based on the mid-day travel time for the "generic" +# truck skim "mfMD_TRK_TIME". Applies truck toll diversion model with +# toll and non-toll skims. +# +# Inputs: +# input_directory: source directory for input files +# num_processors: Number of processors to use, either as a number or "MAX-#" +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file +# input/TruckTripRates.csv +# input/mgra13_based_inputYEAR.csv +# input/specialGenerators.csv +# +# Matrix inputs: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# moTRKL_PROD, moTRKM_PROD, moTRKH_PROD, moTRKEI_PROD, moTRKIE_PROD +# mdTRKL_ATTR, mdTRKM_ATTR, mdTRKH_ATTR, mdTRKEI_ATTR, mdTRKIE_ATTR +# mfTRKEE_DEMAND +# mfMD_TRK_TIME +# mfpp_TRKLGP_TIME, mfpp_TRKLTOLL_TIME, mfpp_TRKLTOLL_TOLLCOST +# mfpp_TRKMGP_TIME, mfpp_TRKMTOLL_TIME, mfpp_TRKMTOLL_TOLLCOST +# mfpp_TRKHGP_TIME, mfpp_TRKHTOLL_TIME, mfpp_TRKHTOLL_TOLLCOST +# +# Matrix intermediates (only used internally): +# mfTRKEI_FRICTION, mfTRKIE_FRICTION, mfTRKL_FRICTION, mfTRKM_FRICTION, mfTRKH_FRICTION +# +# Matrix results: +# Note: pp is time period, one of EA, AM, MD, PM, EV +# mfpp_TRKLGP_VEH, mfpp_TRKMGP_VEH, mfpp_TRKHGP_VEH +# mfpp_TRKLTOLL_VEH, mfpp_TRKMTOLL_VEH, mfpp_TRKHTOLL_VEH +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(main_directory, "input") + num_processors = "MAX-1" + base_scenario = modeller.scenario + distribution = modeller.tool("sandag.model.truck.distribution") + distribution(input_dir, num_processors, base_scenario) +""" + + +TOOLBOX_ORDER = 43 + +import traceback as _traceback +import pandas as pd +import numpy as np +import os + +import inro.modeller as _m + + +gen_utils = _m.Modeller().module('sandag.utilities.general') +dem_utils = _m.Modeller().module('sandag.utilities.demand') + + +class TruckModel(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.num_processors = "MAX-1" + self.attributes = ["input_directory", "num_processors"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Truck distribution" + pb.description = """ +
+ Distributes truck trips with congested skims and splits by time of day. + The distribution is based on the mid-day travel time for the "generic" truck + skim "mfMD_TRK_TIME". +
+ Applies truck toll diversion model with toll and non-toll skims, + and generates truck vehicle trips. +
+ Note that the truck vehicle trips must be converted to PCE values by the Import auto + demand tool and stored in matrices without the _VEH ending for the auto assignment. +
+ """ + pb.branding_text = "- SANDAG - Model - Truck" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + dem_utils.add_select_processors("num_processors", pb, self) + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, self.num_processors, scenario) + run_msg = "Truck trip distribution complete." + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Truck distribution') + def __call__(self, input_directory, num_processors, scenario): + attributes = { + "input_directory": input_directory, + "num_processors": num_processors + } + gen_utils.log_snapshot("Truck distribution", str(self), attributes) + self.scenario = scenario + self.num_processors = num_processors + + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties( + os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) + + with _m.logbook_trace('Daily demand matrices'): + coefficents = [0.045, 0.03, 0.03, 0.03, 0.03] + truck_list = ['L', 'M', 'H', 'IE', 'EI'] + # distribution based on the "generic" truck MD time only + time_skim = scenario.emmebank.matrix('mf"MD_TRK_TIME"') + for truck_type, coeff in zip(truck_list, coefficents): + with _m.logbook_trace('Create %s daily demand matrix' % truck_type): + self.calc_friction_factors(truck_type, time_skim, coeff) + self.matrix_balancing(truck_type) + + self.split_external_demand() + self.split_into_time_of_day() + # NOTE: TOLL diversion skipped with new class definitions + #self.toll_diversion() + + with _m.logbook_trace('Reduce matrix precision'): + precision = props['RunModel.MatrixPrecision'] + matrices = [] + for t, pce in [('L', 1.3), ('M', 1.5), ('H', 2.5)]: + for p in ['EA', 'AM', 'MD', 'PM', 'EV']: + matrices.append('mf%s_TRK_%s_VEH' % (p, t)) + dem_utils.reduce_matrix_precision(matrices, precision, num_processors, scenario) + + @_m.logbook_trace('Create friction factors matrix') + def calc_friction_factors(self, truck_type, impedance, coeff): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + matrix_calc.run_single('mfTRK%s_FRICTION' % truck_type, + 'exp(-%s*%s)' % (coeff, impedance.named_id)) + return + + def matrix_balancing(self, truck_type): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + emmebank = self.scenario.emmebank + with _m.logbook_trace('Matrix balancing for %s' % truck_type): + if truck_type == 'IE': + with gen_utils.temp_matrices(emmebank, "DESTINATION") as (temp_md,): + temp_md.name = 'TRKIE_ROWTOTAL' + matrix_calc.add('md"TRKIE_ROWTOTAL"', 'mf"TRKIE_FRICTION"', aggregation={"origins": "+", "destinations": None}) + matrix_calc.add('mf"TRKIE_DEMAND"', 'mf"TRKIE_FRICTION" * md"TRKIE_ATTR" / md"TRKIE_ROWTOTAL"', + constraint=['md"TRKIE_ROWTOTAL"', 0, 0, "EXCLUDE"]) + matrix_calc.run() + + elif truck_type == 'EI': + with gen_utils.temp_matrices(emmebank, "ORIGIN") as (temp_mo,): + temp_mo.name = 'TRKEI_COLTOTAL' + matrix_calc.add('mo"TRKEI_COLTOTAL"', 'mf"TRKEI_FRICTION"', aggregation={"origins": None, "destinations": "+"}) + matrix_calc.add('mf"TRKEI_DEMAND"', 'mf"TRKEI_FRICTION" * mo"TRKEI_PROD" / mo"TRKEI_COLTOTAL"', + constraint=['mo"TRKEI_COLTOTAL"', 0, 0, "EXCLUDE"]) + matrix_calc.run() + else: + matrix_balancing = _m.Modeller().tool( + 'inro.emme.matrix_calculation.matrix_balancing') + spec = { + "type": "MATRIX_BALANCING", + "od_values_to_balance": 'mf"TRK%s_FRICTION"' % truck_type, + "origin_totals": 'mo"TRK%s_PROD"' % truck_type, + "destination_totals": 'md"TRK%s_ATTR"' % truck_type, + "results": { + "od_balanced_values": 'mf"TRK%s_DEMAND"' % truck_type, + }, + "max_iterations": 100, + "max_relative_error": 0.01 + } + matrix_balancing(spec, self.scenario) + + @_m.logbook_trace('Split cross-regional demand by truck type') + def split_external_demand(self): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + + truck_types = ['L', 'M', 'H'] + truck_share = [0.307, 0.155, 0.538] + for t_type, share in zip(truck_types, truck_share): + matrix_calc.add('mf"TRK%s_DEMAND"' % (t_type), + '%s * (mf"TRKEI_DEMAND" + mf"TRKIE_DEMAND" + mf"TRKEE_DEMAND")' % (share)) + # Set intrazonal truck trips to 0 + matrix_calc.add('mf"TRK%s_DEMAND"' % (t_type), 'mf"TRK%s_DEMAND" * (p.ne.q)' % (t_type)) + matrix_calc.run() + + @_m.logbook_trace('Distribute daily demand into time of day') + def split_into_time_of_day(self): + matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + time_share = [0.1018, 0.1698, 0.4284, 0.1543, 0.1457] + border_time_share = [0.0188, 0.1812, 0.4629, 0.2310, 0.1061] + border_correction = [bs/s for bs, s in zip(border_time_share, time_share)] + + truck_types = ['L', 'M', 'H'] + truck_names = {"L": "light trucks", "M": "medium trucks", "H": "heavy trucks"} + + for period, share, border_corr in zip(periods, time_share, border_correction): + for t in truck_types: + with matrix_calc.trace_run('Calculate %s demand matrix for %s' % (period, truck_names[t])): + tod_demand = 'mf"%s_TRK_%s_VEH"' % (period, t) + matrix_calc.add(tod_demand, 'mf"TRK%s_DEMAND"' % (t)) + matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, share)) + matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, border_corr), + {"origins": "1-5", "destinations": "1-9999"}) + matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, border_corr), + {"origins": "1-9999", "destinations": "1-5"}) + + @_m.logbook_trace('Toll diversion') + def toll_diversion(self): + # NOTE: toll diversion skipped + pass + # matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) + # nest_factor = 10 + # vot = 0.02 # cent/min + # periods = ['EA', 'AM', 'MD', 'PM', 'EV'] + # truck_types = ['L', 'M', 'H'] + # truck_toll_factors = [1, 1.03, 2.33] + + # for period in periods: + # for truck, toll_factor in zip(truck_types, truck_toll_factors): + # with matrix_calc.trace_run('Toll diversion for period %s, truck type %s' % (period, truck) ): + # # Define the utility expression + # utility = """ + # ( + # (mf"%(p)s_TRK%(t)sGP_TIME" - mf"%(p)s_TRK%(t)sTOLL_TIME") + # - %(vot)s * mf"%(p)s_TRK%(t)sTOLL_TOLLCOST" * %(t_fact)s + # ) + # / %(n_fact)s + # """ % { + # 'p': period, + # 't': truck, + # 'vot': vot, + # 't_fact': toll_factor, + # 'n_fact': nest_factor + # } + # # If there is no toll probability of using toll is 0 + # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), '0') + # # If there is a non-zero toll value compute the share of + # # toll-available passengers using the utility expression defined earlier + # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), + # 'mf"%(p)s_TRK%(t)s" * (1/(1 + exp(- %(u)s)))' % {'p': period, 't': truck, 'u': utility}, + # ['mf"%s_TRK%sTOLL_TOLLCOST"' % (period, truck), 0, 0 , "EXCLUDE"]) + # # if non-toll path is not available (GP time=0), set all demand to toll + # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), + # 'mf"%(p)s_TRK%(t)s"' % {'p': period, 't': truck}, + # ['mf"%(p)s_TRK%(t)sGP_TIME"' % {'p': period, 't': truck}, 0, 0 , "INCLUDE"]) + # # Compute the truck demand for non toll + # matrix_calc.add('mf"%s_TRK%sGP_VEH"' % (period, truck), + # '(mf"%(p)s_TRK%(t)s" - mf"%(p)s_TRK%(t)sTOLL_VEH").max.0' % {'p': period, 't': truck}) diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/generation.py b/sandag_abm/src/main/emme/toolbox/model/truck/generation.py new file mode 100644 index 0000000..c0c10a8 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/truck/generation.py @@ -0,0 +1,443 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// model/truck/generation.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the truck generation step. Generates standard truck trip and special (military) truck trips, +# and generates regional truck trips, IE trips, EI trips and EE trips and balances truck trips. +# +# Inputs: +# input_directory: source directory for most input files, including demographics and trip rates +# input_truck_directory: source for special truck files +# scenario: traffic scenario to use for reference zone system +# +# Files referenced: +# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file +# input/TruckTripRates.csv +# file referenced by key mgra.socec.file, e.g. input/mgra13_based_inputYEAR.csv +# input/specialGenerators.csv +# input_truck/regionalIEtripsYEAR.csv +# input_truck/regionalEItripsYEAR.csv +# input_truck/regionalEEtripsYEAR.csv +# +# Matrix results: +# moTRKL_PROD, moTRKM_PROD, moTRKH_PROD, moTRKEI_PROD, moTRKIE_PROD +# mdTRKL_ATTR, mdTRKM_ATTR, mdTRKH_ATTR, mdTRKEI_ATTR, mdTRKIE_ATTR +# mfTRKEE_DEMAND +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(project_dir, "input") + input_truck_dir = os.path.join(project_dir, "input_truck") + base_scenario = modeller.scenario + generation = modeller.tool("sandag.model.truck.generation") + generation(input_dir, input_truck_dir, base_scenario) +""" + + + + +TOOLBOX_ORDER = 42 + + +import inro.modeller as _m +import traceback as _traceback +import numpy as np +import pandas as pd +import os + + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class TruckGeneration(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + input_truck_directory = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.input_truck_directory = os.path.join(os.path.dirname(project_dir), "input_truck") + self.attributes = ["input_directory", "input_truck_directory"] + self._properties = None + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Truck generation" + pb.description = """ +
+ Generates standard truck trip and special (military) truck trips as well as + regional truck trips, IE trips, EI trips and EE trips and balances truck trips + productions / attractions. +
""" + pb.branding_text = "- SANDAG - Model - Truck" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + pb.add_select_file('input_truck_directory', 'directory', + title='Select truck input directory') + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.input_directory, self.input_truck_directory, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Truck generation') + def __call__(self, input_directory, input_truck_directory, scenario): + attributes = {"input_directory": input_directory, "input_truck_directory": input_truck_directory} + gen_utils.log_snapshot("Truck generation", str(self), attributes) + self.input_directory = input_directory + self.input_truck_directory = input_truck_directory + self.scenario = scenario + load_properties = _m.Modeller().tool('sandag.utilities.properties') + + self._properties = load_properties( + os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) + base_trucks_PA = self.truck_standard_generation() + special_trucks_PA = self.special_truck_generation(base_trucks_PA) + trucks_PA = self.balance_truck_PA(special_trucks_PA) + self.store_PA_to_matrices(trucks_PA) + self.read_external_external_demand() + return trucks_PA + + def truck_standard_generation(self): + year = self._properties['truck.FFyear'] + is_interim_year, prev_year, next_year = self.interim_year_check(year) + head, mgra_input_file = os.path.split(self._properties['mgra.socec.file']) + if is_interim_year: + taz_prev_year = self.create_demographics_by_taz( + mgra_input_file.replace(str(year), str(prev_year))) + taz_next_year = self.create_demographics_by_taz( + mgra_input_file.replace(str(year), str(next_year))) + taz = self.interpolate_df(prev_year, year, next_year, taz_prev_year, taz_next_year) + else: + taz = self.create_demographics_by_taz(mgra_input_file) + + trip_rates = pd.read_csv( + os.path.join(self.input_directory, 'TruckTripRates.csv')) + taz = pd.merge(taz, trip_rates, + left_on='truckregiontype', right_on='RegionType', how='left') + taz.fillna(0, inplace=True) + + # Compute lhd truck productions "AGREMPN", "CONEMPN", "RETEMPN", "GOVEMPN", + # "MANEMPN", "UTLEMPN", "WHSEMPN", "OTHEMPN" + taz['LHD_Productions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_L_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TG_L_Retail'] \ + + taz['emp_gov'] * taz['TG_L_Government'] \ + + taz['emp_mfg'] * taz['TG_L_Manufacturing'] \ + + taz['emp_twu'] * taz['TG_L_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TG_L_Wholesale'] \ + + taz['emp_other'] * taz['TG_L_Other'] \ + + taz['hh'] * taz['TG_L_Households'] + + taz['LHD_Attractions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_L_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TA_L_Retail'] \ + + taz['emp_gov'] * taz['TA_L_Government'] \ + + taz['emp_mfg'] * taz['TA_L_Manufacturing'] \ + + taz['emp_twu'] * taz['TA_L_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TA_L_Wholesale'] \ + + taz['emp_other'] * taz['TA_L_Other'] \ + + taz['hh'] * taz['TA_L_Households'] + + taz['MHD_Productions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_M_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TG_M_Retail'] \ + + taz['emp_gov'] * taz['TG_M_Government'] \ + + taz['emp_mfg'] * taz['TG_M_Manufacturing'] \ + + taz['emp_twu'] * taz['TG_M_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TG_M_Wholesale'] \ + + taz['emp_other'] * taz['TG_M_Other'] \ + + taz['hh'] * taz['TG_M_Households'] + + taz['MHD_Attractions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_M_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TA_M_Retail'] \ + + taz['emp_gov'] * taz['TA_M_Government'] \ + + taz['emp_mfg'] * taz['TA_M_Manufacturing'] \ + + taz['emp_twu'] * taz['TA_M_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TA_M_Wholesale'] \ + + taz['emp_other'] * taz['TA_M_Other'] \ + + taz['hh'] * taz['TA_M_Households'] + + taz['HHD_Productions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_H_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TG_H_Retail'] \ + + taz['emp_gov'] * taz['TG_H_Government'] \ + + taz['emp_mfg'] * taz['TG_H_Manufacturing'] \ + + taz['emp_twu'] * taz['TG_H_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TG_H_Wholesale'] \ + + taz['emp_other'] * taz['TG_H_Other'] \ + + taz['hh'] * taz['TG_H_Households'] + + taz['HHD_Attractions'] = \ + (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_H_Ag/Min/Constr'] \ + + taz['emp_retrade'] * taz['TA_H_Retail'] \ + + taz['emp_gov'] * taz['TA_H_Government'] \ + + taz['emp_mfg'] * taz['TA_H_Manufacturing'] \ + + taz['emp_twu'] * taz['TA_H_Transp/Utilities'] \ + + taz['emp_whtrade'] * taz['TA_H_Wholesale'] \ + + taz['emp_other'] * taz['TA_H_Other'] \ + + taz['hh'] * taz['TA_H_Households'] + + taz.reset_index(inplace=True) + taz = taz[['taz', + 'LHD_Productions', 'LHD_Attractions', + 'MHD_Productions', 'MHD_Attractions', + 'HHD_Productions', 'HHD_Attractions']] + return taz + + # Creates households and employments by TAZ. + # Specific to the truck trip generation model. + # Inputs: + # - sandag.properties + # - input/mgra13_based_input20XX.csv (referenced by mgra.socec.file in properties file) + def create_demographics_by_taz(self, mgra_input_file): + utils = _m.Modeller().module('sandag.utilities.demand') + dt = _m.Modeller().desktop.project.data_tables() + file_path = os.path.join(self.input_directory, mgra_input_file) + if not os.path.exists(file_path): + raise Exception("MGRA input file '%s' does not exist" % file_path) + mgra = pd.read_csv(file_path) + # Combine employment fields that match to the truck trip rate classification + mgra['TOTEMP'] = mgra.emp_total + mgra['emp_agmin'] = mgra.emp_ag + mgra['emp_cons'] = mgra.emp_const_bldg_prod + mgra.emp_const_bldg_office + mgra['emp_retrade'] = mgra.emp_retail + mgra.emp_personal_svcs_retail + mgra['emp_gov']= mgra.emp_state_local_gov_ent \ + + mgra.emp_state_local_gov_blue \ + + mgra.emp_state_local_gov_white \ + + mgra.emp_fed_non_mil \ + + mgra.emp_fed_mil + mgra['emp_mfg'] = mgra.emp_mfg_prod \ + + mgra.emp_mfg_office + mgra['emp_twu'] = mgra.emp_trans \ + + mgra.emp_utilities_office \ + + mgra.emp_utilities_prod + mgra['emp_whtrade'] = mgra.emp_whsle_whs + mgra['emp_other'] = mgra.TOTEMP \ + - mgra.emp_agmin \ + - mgra.emp_cons \ + - mgra.emp_retrade \ + - mgra.emp_gov \ + - mgra.emp_mfg \ + - mgra.emp_twu \ + - mgra.emp_whtrade + + f = { + 'truckregiontype':['mean'], + 'emp_agmin':['sum'], + 'emp_cons': ['sum'], + 'emp_retrade': ['sum'], + 'emp_gov': ['sum'], + 'emp_mfg': ['sum'], + 'emp_twu': ['sum'], + 'emp_whtrade': ['sum'], + 'emp_other': ['sum'], + 'hh': ['sum'] + } + + mgra = mgra[['truckregiontype', 'emp_agmin', 'emp_cons', + 'emp_retrade', 'emp_gov', 'emp_mfg', 'emp_twu', + 'emp_whtrade', 'emp_other', 'taz', 'hh']] + taz = mgra.groupby('taz').agg(f) + taz.reset_index(inplace=True) + taz.columns = taz.columns.droplevel(-1) + # Add external zones + taz = utils.add_missing_zones(taz, self.scenario) + return taz + + # Add trucks generated by special generators, such as military sites, + # mail to//from airport, cruise ships etc + # Inputs: + # - input/specialGenerators.csv + # - dataframe: base_trucks + def special_truck_generation(self, base_trucks): + year = self._properties['truck.FFyear'] + is_interim_year, prev_year, next_year = self.interim_year_check(year) + spec_gen = pd.read_csv(os.path.join(self.input_directory, 'specialGenerators.csv')) + spec_gen = pd.merge(spec_gen, base_trucks, + left_on=['TAZ'], right_on=['taz'], how='outer') + spec_gen.fillna(0, inplace=True) + if is_interim_year: + year_ratio = float(year - prev_year) / (next_year - prev_year) + prev_year, next_year = 'Y%s' % prev_year, 'Y%s' % next_year + spec_gen['Y%s' % year] = spec_gen[prev_year] + year_ratio * (spec_gen[next_year] - spec_gen[prev_year]) + + for t in ['L', 'M', 'H']: + spec_gen['%sHD_Attr' % t] = spec_gen['%sHD_Attractions' % t] + \ + (spec_gen['Y%s' % year] * + spec_gen['trkAttraction'] * + spec_gen['%shdShare' % t.lower()]) + spec_gen['%sHD_Prod' % t] = spec_gen['%sHD_Productions' % t] + \ + (spec_gen['Y%s' % year] * + spec_gen['trkProduction'] * + spec_gen['%shdShare' % t.lower()]) + + special_trucks = spec_gen[ + ['taz', 'LHD_Prod', 'LHD_Attr', 'MHD_Prod', 'MHD_Attr', 'HHD_Prod', 'HHD_Attr']] + return special_trucks + + # Balance truck Productions and Attractions + def balance_truck_PA(self, truck_pa): + truck_pa = self.balance_internal_truck_PA(truck_pa) + regional_truck_pa = self.get_regional_truck_PA() + truck_pa = self.add_balanced_regional_PA(regional_truck_pa, truck_pa) + truck_pa.fillna(0, inplace=True) + + truck_pa['TRKL_Prod'] = truck_pa['LHD_Prod'] + truck_pa['TRKM_Prod'] = truck_pa['MHD_Prod'] + truck_pa['TRKH_Prod'] = truck_pa['HHD_Prod'] + truck_pa['TRKIE_Prod'] = truck_pa['IE_Prod'] + truck_pa['TRKEI_Prod'] = truck_pa['EI_Prod'] + + truck_pa['TRKL_Attr'] = truck_pa['LHD_Attr'] + truck_pa['TRKM_Attr'] = truck_pa['MHD_Attr'] + truck_pa['TRKH_Attr'] = truck_pa['HHD_Attr'] + truck_pa['TRKIE_Attr'] = truck_pa['IE_Attr'] + truck_pa['TRKEI_Attr'] = truck_pa['EI_Attr'] + return truck_pa + + def get_regional_truck_PA(self): + year = self._properties['truck.FFyear'] + trips = {} + regional_trip_types = ['IE', 'EI', 'EE'] + is_interim_year, prev_year, next_year = self.interim_year_check(year) + if is_interim_year: + for t in regional_trip_types: + prev_trips = pd.read_csv(os.path.join( + self.input_truck_directory, + 'regional%strips%s.csv' % (t, prev_year))) + next_trips = pd.read_csv(os.path.join( + self.input_truck_directory, + 'regional%strips%s.csv' % (t, next_year))) + trips_df = self.interpolate_df(prev_year, year, next_year, prev_trips, next_trips) + trips[t] = trips_df + + for t in regional_trip_types: + trips[t] = pd.read_csv(os.path.join( + self.input_truck_directory, + 'regional%strips%s.csv' % (t, year))) + return trips + + def balance_internal_truck_PA(self, truck_pa): + truck_types = ['LHD', 'MHD', 'HHD'] + for t in truck_types: + s1 = truck_pa['%s_Prod' % t].sum() + s2 = truck_pa['%s_Attr' % t].sum() + avg = (s1 + s2)/2.0 + w1 = avg / s1 + w2 = avg / s2 + truck_pa['%s_Prod_unbalanced' % t] = truck_pa['%s_Prod' % t] + truck_pa['%s_Attr_unbalanced' % t] = truck_pa['%s_Attr' % t] + truck_pa['%s_Prod' % t] = truck_pa['%s_Prod' % t] * w1 + truck_pa['%s_Attr' % t] = truck_pa['%s_Attr' % t] * w2 + return truck_pa + + # Balance only EI and IE. EE truck trips are already balanced and can be + # directly imported as a matrix + def add_balanced_regional_PA(self, regional_trips, truck_pa): + ei_trips = regional_trips['EI'] + ei_trips = ei_trips.groupby('fromZone').sum() + ei_trips.reset_index(inplace=True) + + truck_pa = pd.merge(truck_pa, + ei_trips[['fromZone', 'EITrucks']], + left_on='taz', right_on='fromZone', + how='outer') + + sum_ei = ei_trips['EITrucks'].sum() + sum_hhd_attr = truck_pa['HHD_Attr'].sum() + truck_pa['EI_Attr'] = truck_pa['HHD_Attr'] * sum_ei / sum_hhd_attr + truck_pa['EI_Prod'] = truck_pa['EITrucks'] + + ie_trips = regional_trips['IE'] + ie_trips = ie_trips.groupby('toZone').sum() + ie_trips.reset_index(inplace=True) + truck_pa = pd.merge(truck_pa, + ie_trips[['toZone', 'IETrucks']], + left_on='taz', right_on='toZone', + how='outer') + + sum_ie = ie_trips['IETrucks'].sum() + sum_hhd_prod = truck_pa['HHD_Prod'].sum() + truck_pa['IE_Prod'] = truck_pa['HHD_Prod'] * sum_ie / sum_hhd_prod + truck_pa['IE_Attr'] = truck_pa['IETrucks'] + truck_pa.fillna(0, inplace=True) + + return truck_pa + + def store_PA_to_matrices(self, truck_pa): + emmebank = self.scenario.emmebank + truck_pa.sort_values('taz', inplace=True) #sort method was deprecated since version 0.20.0, yma, 2/12/2019 + control_to_store = ['L', 'M', 'H', 'EI', 'IE'] + for t in control_to_store: + prod = emmebank.matrix('moTRK%s_PROD' % t) + prod.set_numpy_data(truck_pa['TRK%s_Prod' % t].values, self.scenario) + attr = emmebank.matrix('mdTRK%s_ATTR' % t) + attr.set_numpy_data(truck_pa['TRK%s_Attr' % t].values, self.scenario) + + @_m.logbook_trace('External - external truck matrix') + def read_external_external_demand(self): + utils = _m.Modeller().module('sandag.utilities.demand') + emmebank = self.scenario.emmebank + regional_trips = self.get_regional_truck_PA() + ee = regional_trips['EE'] + m_ee = emmebank.matrix('mfTRKEE_DEMAND') + m_ee_data = m_ee.get_data(self.scenario) + for i, row in ee.iterrows(): + m_ee_data.set(row['fromZone'], row['toZone'], row['EETrucks']) + m_ee.set_data(m_ee_data, self.scenario) + + def interpolate_df(self, prev_year, new_year, next_year, prev_year_df, next_year_df): + current_year_df = pd.DataFrame() + year_ratio = float(new_year - prev_year) / (next_year - prev_year) + for key in prev_year_df.columns: + current_year_df[key] = ( + prev_year_df[key] + year_ratio * (next_year_df[key] - prev_year_df[key])) + return current_year_df + + def interim_year_check(self, year): + years_with_data = self._properties['truck.DFyear'] + if year in years_with_data: + return (False, year, year) + else: + next_year_idx = np.searchsorted(years_with_data, year) + if next_year_idx == 0 or next_year_idx > len(years_with_data): + raise Exception('Cannot interpolate data for year %s' % year) + prev_year = years_with_data[next_year_idx - 1] + next_year = years_with_data[next_year_idx] + return (True, prev_year, next_year) diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py b/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py new file mode 100644 index 0000000..8185ca0 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py @@ -0,0 +1,130 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// truck/run_truck_model.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# Runs the truck model, the generation and distribution tools, in sequence. +# 1) Generates standard truck trip and special (military) truck trips +# 2) Generates regional truck trips, IE trips, EI trips and EE trips and +# balances truck trips +# 3) Distributes trips based on congested skims and splits by time of day +# 4) Applies truck toll diversion model with toll and non-toll skims +# +# Inputs: +# run_generation: boolean, if True run generation tool. +# input_directory: source directory for most input files, including demographics and trip rates +# (see generation and distribtuion tools) +# input_truck_directory: source for special truck files (see generation tool) +# num_processors: Number of processors to use, either as a number or "MAX-#" +# scenario: traffic scenario to use for reference zone system +# +# Script example: +""" + import os + modeller = inro.modeller.Modeller() + project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) + input_dir = os.path.join(project_dir, "input") + input_truck_dir = os.path.join(project_dir, "input_truck") + base_scenario = modeller.scenario + num_processors = "MAX-1" + run_truck = modeller.tool("sandag.model.truck.run_truck_model") + run_truck(True, input_dir, input_truck_dir, num_processors, base_scenario) +""" + + + +TOOLBOX_ORDER = 41 + + +import inro.modeller as _m +import traceback as _traceback +import os + + +dem_utils = _m.Modeller().module("sandag.utilities.demand") +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class TruckModel(_m.Tool(), gen_utils.Snapshot): + + input_directory = _m.Attribute(str) + input_truck_directory = _m.Attribute(str) + run_generation = _m.Attribute(bool) + num_processors = _m.Attribute(str) + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def __init__(self): + self.run_generation = True + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.input_directory = os.path.join(os.path.dirname(project_dir), "input") + self.input_truck_directory = os.path.join(os.path.dirname(project_dir), "input_truck") + self.num_processors = "MAX-1" + self.attributes = ["input_directory", "input_truck_directory", "run_generation", "num_processors"] + + def page(self): + # Equivalent to TruckModel.rsc + pb = _m.ToolPageBuilder(self) + pb.title = "Truck model" + pb.description = """ +
+ 1) Generates standard truck trip and special (military) truck trips
+ 2) Gets regional truck trips, IE trips, EI trips and EE trips and balances truck trips
+ 3) Distributes truck trips with congested skims and splits by time of day
+ 4) Applies truck toll diversion model with free-flow toll and non-toll skims
+
+""" + pb.branding_text = "- SANDAG - Model - Truck" + + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + pb.add_checkbox("run_generation", title=" ", label="Run generation (first iteration)") + + pb.add_select_file('input_directory', 'directory', + title='Select input directory') + pb.add_select_file('input_truck_directory', 'directory', + title='Select truck input directory') + dem_utils.add_select_processors("num_processors", pb, self) + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + scenario = _m.Modeller().scenario + self(self.run_generation, self.input_directory, self.input_truck_directory, self.num_processors, scenario) + run_msg = "Tool complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + @_m.logbook_trace('Truck model', save_arguments=True) + def __call__(self, run_generation, input_directory, input_truck_directory, num_processors, scenario): + attributes = { + "input_directory": input_directory, "input_truck_directory": input_truck_directory, + "run_generation": run_generation, "num_processors": num_processors + } + gen_utils.log_snapshot("Truck model", str(self), attributes) + + generation = _m.Modeller().tool('sandag.model.truck.generation') + distribution = _m.Modeller().tool('sandag.model.truck.distribution') + + if run_generation: + generation(input_directory, input_truck_directory, scenario) + distribution(input_directory, num_processors, scenario) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/demand.py b/sandag_abm/src/main/emme/toolbox/utilities/demand.py new file mode 100644 index 0000000..bcd0f5f --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/demand.py @@ -0,0 +1,297 @@ +##////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// utilities/demand.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// + +TOOLBOX_ORDER = 101 + + +import inro.emme.datatable as _dt +import inro.modeller as _m +from collections import OrderedDict +from contextlib import contextmanager as _context +from copy import deepcopy as _copy +import multiprocessing as _multiprocessing +import re as _re +import pandas as _pandas +import numpy as _numpy +import os + + +class Utils(_m.Tool()): + + def page(self): + pb = _m.ToolPageBuilder(self, runnable=False) + pb.title = 'Demand utility' + pb.description = """Utility tool / module for common code. Not runnable.""" + pb.branding_text = ' - SANDAG - Utilities' + return pb.render() + + +# Read a CSV file, store it as a DataTable and return a representative DataFrame +def csv_to_data_table(path, overwrite=False): + layer_name = os.path.splitext(os.path.basename(path))[0] + data_source = _dt.DataSource(path) + data = data_source.layer(layer_name).get_data() + desktop = _m.Modeller().desktop + dt_db = desktop.project.data_tables() + table = dt_db.create_table(layer_name, data, overwrite=overwrite) + return table_to_dataframe(table) + + +# Convert a DataTable into a DataFrame +def table_to_dataframe(table): + if type(table) == str: + desktop = _m.Modeller().desktop + dt_db = desktop.project.data_tables() + table_name = table + table = dt_db.table(table) + if not table: + raise Exception('%s is not a valid table name.' %table_name) + + df = _pandas.DataFrame() + for attribute in table.get_data().attributes(): + try: + df[attribute.name] = attribute.values.astype(float) + except Exception, e: + df[attribute.name] = attribute.values + + return df + + +# Convert a dataframe to a datatable +def dataframe_to_table(df, name): + desktop = _m.Modeller().desktop + dt_db = desktop.project.data_tables() + data = _dt.Data() + for key in df.columns: + found_dtype = False + dtypes = [ + (bool, True, 'BOOLEAN'), + (int, 0, 'INTEGER32'), + (int, 0, 'INTEGER'), + (float, 0, 'REAL') + ] + for caster, default, name in dtypes: + try: + df[[key]] = df[[key]].fillna(default) + values = df[key].astype(caster) + attribute = _dt.Attribute(key, values, name) + found_dtype = True + break + except ValueError: + pass + + if not found_dtype: + df[[key]] = df[[key]].fillna(0) + values = df[key].astype(str) + attribute = _dt.Attribute(key, values, 'STRING') + + data.add_attribute(attribute) + + table = dt_db.create_table(name, data, overwrite=True) + return table + +# Add missing (usually external zones 1 to 12) zones to the DataFrame +# and populate with zeros +def add_missing_zones(df, scenario): + all_zones = scenario.zone_numbers + existing_zones = df['taz'].values + missing_zones = set(all_zones) - set(existing_zones) + num_missing = len(missing_zones) + if num_missing == 0: + return df + + ext_df = _pandas.DataFrame() + for c in df.columns: + ext_df[c] = _numpy.zeros(num_missing) + ext_df['taz'] = _numpy.array(list(missing_zones)) + df = _pandas.concat([df, ext_df]) + df = df.sort_values('taz', ascending=True) # sort method was deprecated in version 0.20.0,yma,2/12/2019 + return df + + +def add_select_processors(tool_attr_name, pb, tool): + max_processors = _multiprocessing.cpu_count() + tool._max_processors = max_processors + options = [("MAX-1", "Maximum available - 1"), ("MAX", "Maximum available")] + options.extend([(n, "%s processors" % n) for n in range(1, max_processors + 1) ]) + pb.add_select(tool_attr_name, options, title="Number of processors:") + + +def parse_num_processors(value): + max_processors = _multiprocessing.cpu_count() + if isinstance(value, int): + return value + if isinstance(value, basestring): + if value == "MAX": + return max_processors + if _re.match("^[0-9]+$", value): + return int(value) + result = _re.split("^MAX[\s]*-[\s]*", value) + if len(result) == 2: + return max(max_processors - int(result[1]), 1) + if value: + return int(value) + return value + +class MatrixCalculator(object): + def __init__(self, scenario, num_processors=0): + self._scenario = scenario + self._matrix_calc = _m.Modeller().tool( + "inro.emme.matrix_calculation.matrix_calculator") + self._specs = [] + self._last_report = None + self.num_processors = num_processors + + @property + def num_processors(self): + return self._num_processors + + @num_processors.setter + def num_processors(self, value): + self._num_processors = parse_num_processors(value) + + @property + def last_report(self): + return _copy(self._last_report) + + @_context + def trace_run(self, name): + with _m.logbook_trace(name): + yield + self.run() + + def add(self, result, expression, constraint=None, aggregation=None): + spec = self._format_spec(result, expression, constraint, aggregation) + self._specs.append(spec) + + def _format_spec(self, result, expression, constraint, aggregation): + spec = { + "result": result, + "expression": expression, + "type": "MATRIX_CALCULATION" + } + if constraint is not None: + if isinstance(constraint, (list, tuple)): + # specified as list of by_value inputs + constraint = { + "by_value": { + "od_values": constraint[0], + "interval_min": constraint[1], + "interval_max": constraint[2], + "condition": constraint[3] + } + } + elif "od_values" in constraint: + # specified as the by_value sub-dictionary only + constraint = {"by_value": constraint} + # By zone constraints + elif ("destinations" in constraint or "origins" in constraint): + # specified as the by_zone sub-dictionary only + constraint = {"by_zone": constraint} + # otherwise, specified as a regular full constraint dictionary + if "by_value" in constraint: + # cast the inputs to the correct values + constraint["by_value"]["od_values"] = \ + str(constraint["by_value"]["od_values"]) + constraint["by_value"]["condition"] = \ + constraint["by_value"]["condition"].upper() + spec["constraint"] = constraint + + #Add None for missing key values if needed + if "by_value" not in constraint: + constraint["by_value"] = None + if "by_zone" not in constraint: + constraint["by_zone"] = None + + else: + spec["constraint"] = None + + if aggregation is not None: + if isinstance(aggregation, basestring): + aggregation = {"origins": aggregation} + spec["aggregation"] = aggregation + else: + spec["aggregation"] = None + return spec + + def add_spec(self, spec): + self._specs.append(spec) + + def run(self): + specs, self._specs = self._specs, [] + report = self._matrix_calc(specs, scenario=self._scenario, + num_processors=self._num_processors) + self._last_report = report + return report + + def run_single(self, result, expression, constraint=None, aggregation=None): + spec = self._format_spec(result, expression, constraint, aggregation) + return self._matrix_calc(spec, scenario=self._scenario, + num_processors=self._num_processors) + + +def reduce_matrix_precision(matrices, precision, num_processors, scenario): + emmebank = scenario.emmebank + calc = MatrixCalculator(scenario, num_processors) + gen_utils = _m.Modeller().module('sandag.utilities.general') + with gen_utils.temp_matrices(emmebank, "SCALAR", 2) as (sum1, sum2): + sum1.name = "ORIGINAL_SUM" + sum2.name = "ROUNDED_SUM" + for mat in matrices: + mat = emmebank.matrix(mat).named_id + with calc.trace_run('Reduce precision for matrix %s' % mat): + calc.add(sum1.named_id, mat, aggregation={"destinations": "+", "origins": "+"}) + calc.add(mat, "{mat} * ({mat} >= {precision})".format( + mat=mat, precision=precision)) + calc.add(sum2.named_id, mat, aggregation={"destinations": "+", "origins": "+"}) + calc.add(sum2.named_id, "({sum2} + ({sum2} == 0))".format(sum2=sum2.named_id)) + calc.add(mat, "{mat} * ({sum1} / {sum2})".format( + mat=mat, sum2=sum2.named_id, sum1=sum1.named_id)) + + +def create_full_matrix(name, desc, scenario): + create_matrix = _m.Modeller().tool( + "inro.emme.data.matrix.create_matrix") + emmebank = scenario.emmebank + matrix = emmebank.matrix(name) + if matrix: + ident = matrix.id + else: + used_ids = set([]) + for m in emmebank.matrices(): + if m.prefix == "mf": + used_ids.add(int(m.id[2:])) + for i in range(900, emmebank.dimensions["full_matrices"]): + if i not in used_ids: + ident = "mf" + str(i) + break + else: + raise Exception("Not enough available matrix IDs for selected demand. Change database dimensions to increase full matrices.") + return create_matrix(ident, name, desc, scenario=scenario, overwrite=True) + + +def demand_report(matrices, label, scenario, report=None): + text = ['
'] + text.append("%-28s %13s" % ("name", "sum")) + for name, data in matrices: + stats = (name, data.sum()) + text.append("%-28s %13.7g" % stats) + text.append("
") + title = "Demand summary" + if report is None: + report = _m.PageBuilder(title) + report.wrap_html('Matrix details', "
".join(text)) + _m.logbook_write(label, report.render()) + else: + report.wrap_html(label, "
".join(text)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py b/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py new file mode 100644 index 0000000..6e56df5 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py @@ -0,0 +1,370 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2018. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// utilities/file_manager.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +TOOLBOX_ORDER = 104 + + +import inro.modeller as _m +import inro.emme.database.emmebank as _eb +import inro.director.logging as _log + +import traceback as _traceback +import shutil as _shutil +import time as _time +import os +from fnmatch import fnmatch as _fnmatch +from math import log10 + +_join = os.path.join +_dir = os.path.dirname +_norm = os.path.normpath + +gen_utils = _m.Modeller().module("sandag.utilities.general") + + +class FileManagerTool(_m.Tool(), gen_utils.Snapshot): + + operation = _m.Attribute(unicode) + remote_dir = _m.Attribute(unicode) + local_dir = _m.Attribute(unicode) + user_folder = _m.Attribute(unicode) + scenario_id = _m.Attribute(unicode) + initialize = _m.Attribute(_m.BooleanType) + delete_local_files = _m.Attribute(_m.BooleanType) + + tool_run_msg = "" + LOCAL_ROOT = "C:\\abm_runs" + + def __init__(self): + self.operation = "UPLOAD" + project_dir = _dir(_m.Modeller().desktop.project.path) + self.remote_dir = _dir(project_dir) + folder_name = os.path.basename(self.remote_dir) + self.user_folder = os.environ.get("USERNAME") + self.scenario_id = 100 + self.initialize = True + self.delete_local_files = True + self.attributes = [ + "operation", "remote_dir", "local_dir", "user_folder", + "scenario_id", "initialize", "delete_local_files" + ] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "File run manager utility" + pb.description = """ +

+ Utility tool to manually manage the use of the local drive for subsequent model run. + The remote data can be downloaded (copied) to the local drive; + or the local data can be uploaded to the remote drive. + In normal operation this tool does not need to run manually, but in case of an + error it may be necessary to upload the project data in order to run on + a different machine, or operate directly on the server. +

+

+ Note that file masks are used from config/sandag_abm.properties to identify which + files to copy. See RunModel.FileMask.Upload and RunModel.FileMask.Download for + upload and download respectively. +

""" + pb.branding_text = "- SANDAG" + if self.tool_run_msg: + pb.add_html(self.tool_run_msg) + + pb.add_radio_group('operation', title="File copy operation", + keyvalues=[("UPLOAD", "Upload from local directory to remote directory"), + ("DOWNLOAD", "Download from remote directory to local directory")], ) + pb.add_select_file('remote_dir','directory', + title='Select remote ABM directory (e.g. on T drive)', note='') + pb.add_text_box('user_folder', title="User folder (for local drive):") + pb.add_text_box('scenario_id', title="Base scenario ID:") + pb.add_checkbox_group( + [{"attribute": "delete_local_files", "label": "Delete all local files on completion (upload only)"}, + {"attribute": "initialize", "label": "Initialize all local files; if false only download files which are different (download only)"}]) + pb.add_html(""" +""") + + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self(self.operation, self.remote_dir, self.user_folder, self.scenario_id, + self.initialize, self.delete_local_files) + run_msg = "File copying complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + def __call__(self, operation, remote_dir, user_folder, scenario_id, initialize=True, delete_local_files=True): + load_properties = _m.Modeller().tool('sandag.utilities.properties') + props = load_properties(_join(remote_dir, "conf", "sandag_abm.properties")) + if operation == "DOWNLOAD": + file_masks = props.get("RunModel.FileMask.Download") + return self.download(remote_dir, user_folder, scenario_id, initialize, file_masks) + elif operation == "UPLOAD": + file_masks = props.get("RunModel.FileMask.Upload") + self.upload(remote_dir, user_folder, scenario_id, delete_local_files, file_masks) + else: + raise Exception("operation must be one of UPLOAD or DOWNLOAD") + + @_m.logbook_trace("Copy project data to local drive", save_arguments=True) + def download(self, remote_dir, user_folder, scenario_id, initialize, file_masks): + folder_name = os.path.basename(remote_dir) + user_folder = user_folder or os.environ.get("USERNAME") + if not user_folder: + raise Exception("Username must be specified for local drive operation " + "(or define USERNAME environment variable)") + if not os.path.exists(self.LOCAL_ROOT): + os.mkdir(self.LOCAL_ROOT) + user_directory = _join(self.LOCAL_ROOT, user_folder) + if not os.path.exists(user_directory): + os.mkdir(user_directory) + local_dir = _join(user_directory, folder_name) + if not os.path.exists(local_dir): + os.mkdir(local_dir) + + self._report = ["Copy"] + self._stats = {"size": 0, "count": 0} + if not file_masks: + # suggested default: "output", "report", "sql", "logFiles" + file_masks = [] + file_masks = [_join(remote_dir, p) for p in file_masks] + file_masks.append(_join(remote_dir, "emme_project")) + if initialize: + # make sure that all of the root directories are created + root_dirs = [ + "application", "bin", "conf", "emme_project", "input", "input_truck", + "logFiles", "output", "python", "report", "sql", "uec" + ] + for name in root_dirs: + if not os.path.exists(_join(local_dir, name)): + os.mkdir(_join(local_dir, name)) + # create new Emmebanks with scenario and matrix data + title_fcn = lambda t: "(local) " + t[:50] + emmebank_paths = self._copy_emme_data( + src=remote_dir, dst=local_dir, initialize=True, + title_fcn=title_fcn, scenario_id=scenario_id) + # add new emmebanks to the open project + # db_paths = set([db.core_emmebank.path for db in data_explorer.databases()]) + # for path in emmebank_paths: + # if path not in db_paths: + # _m.Modeller().desktop.data_explorer().add_database(path) + + # copy all files (except Emme project, and other file_masks) + self._copy_dir(src=remote_dir, dst=local_dir, + file_masks=file_masks, check_metadata=not initialize) + self.log_report() + return local_dir + + @_m.logbook_trace("Copy project data to remote drive", save_arguments=True) + def upload(self, remote_dir, user_folder, scenario_id, delete_local_files, file_masks): + folder_name = os.path.basename(remote_dir) + user_folder = user_folder or os.environ.get("USERNAME") + user_directory = _join(self.LOCAL_ROOT, user_folder) + local_dir = _join(user_directory, folder_name) + + self._report = [] + self._stats = {"size": 0, "count": 0} + if not file_masks: + # suggested defaults: "application", "bin", "input", "input_truck", "uec", + # "output\\iter*", "output\\*_1.csv", "output\\*_2.csv" + file_masks = [] + # prepend the src dir to the project masks + file_masks = [_join(local_dir, p) for p in file_masks] + # add to mask the emme_project folder + file_masks.append(_join(local_dir, "emme_project")) + + title_fcn = lambda t: t[8:] if t.startswith("(local)") else t + emmebank_paths = self._copy_emme_data( + src=local_dir, dst=remote_dir, title_fcn=title_fcn, scenario_id=scenario_id) + # copy all files (except Emme project, and other file_masks) + self._copy_dir(src=local_dir, dst=remote_dir, file_masks=file_masks) + self.log_report() + + # data_explorer = _m.Modeller().desktop.data_explorer() + # for path in emmebank_paths: + # for db in data_explorer.databases(): + # if db.core_emmebank.path == path: + # db.close() + # data_explorer.remove_database(db) + # data_explorer.databases()[0].open() + + if delete_local_files: + # small pause for file handles to close + _time.sleep(2) + for name in os.listdir(local_dir): + path = os.path.join(local_dir, name) + if os.path.isfile(path): + try: # no raise, local files can be left behind + os.remove(path) + except: + pass + elif os.path.isdir(path): + try: + _shutil.rmtree(path, ignore_errors=False) + except: + pass + + _shutil.rmtree(local_dir, ignore_errors=False) + + def _copy_emme_data(self, src, dst, title_fcn, scenario_id, initialize=False): + # copy data from Database and Database_transit using API and import tool + # create new emmebanks and copy emmebank data to local drive + import_from_db = _m.Modeller().tool("inro.emme.data.database.import_from_database") + emmebank_paths = [] + for db_dir in ["Database", "Database_transit"]: + src_db_path = _join(src, "emme_project", db_dir, "emmebank") + if not os.path.exists(src_db_path): + # skip if the database does not exist (will be created later) + continue + src_db = _eb.Emmebank(src_db_path) + dst_db_dir = _join(dst, "emme_project", db_dir) + dst_db_path = _join(dst_db_dir, "emmebank") + emmebank_paths.append(dst_db_path) + self._report.append("Copying Emme data
from %s
to %s" % (src_db_path, dst_db_path)) + self._report.append("Start: %s" % _time.strftime("%c")) + if initialize: + # remove any existing database (overwrite) + if os.path.exists(dst_db_path): + self._report.append("Warning: overwritting existing Emme database %s" % dst_db_path) + dst_db = _eb.Emmebank(dst_db_path) + dst_db.dispose() + if os.path.exists(dst_db_dir): + gen_utils.retry(lambda: _shutil.rmtree(dst_db_dir)) + gen_utils.retry(lambda: os.mkdir(dst_db_dir)) + dst_db = _eb.create(dst_db_path, src_db.dimensions) + else: + if not os.path.exists(dst_db_dir): + os.mkdir(dst_db_dir) + if os.path.exists(dst_db_path): + dst_db = _eb.Emmebank(dst_db_path) + else: + dst_db = _eb.create(dst_db_path, src_db.dimensions) + + dst_db.title = title_fcn(src_db.title) + for prop in ["coord_unit_length", "unit_of_length", "unit_of_cost", + "unit_of_energy", "use_engineering_notation", "node_number_digits"]: + setattr(dst_db, prop, getattr(src_db, prop)) + + if initialize: + src_db.dispose() + continue + + exfpars = [p for p in dir(src_db.extra_function_parameters) if p.startswith("e")] + for exfpar in exfpars: + value = getattr(src_db.extra_function_parameters, exfpar) + setattr(dst_db.extra_function_parameters, exfpar, value) + + for s in src_db.scenarios(): + if dst_db.scenario(s.id): + dst_db.delete_scenario(s) + for f in src_db.functions(): + if dst_db.function(f.id): + dst_db.delete_function(f) + for m in src_db.matrices(): + if dst_db.matrix(m.id): + dst_db.delete_matrix(m) + for p in dst_db.partitions(): + p.description = "" + p.initialize(0) + ref_scen = dst_db.scenario(999) + if not ref_scen: + ref_scen = dst_db.create_scenario(999) + import_from_db( + src_database=src_db, + src_scenario_ids=[s.id for s in src_db.scenarios()], + src_function_ids=[f.id for f in src_db.functions()], + copy_path_strat_files=True, + dst_database=dst_db, + dst_zone_system_scenario=ref_scen) + dst_db.delete_scenario(999) + src_matrices = [m.id for m in src_db.matrices()] + src_partitions = [p.id for p in src_db.partitions() + if not(p.description == '' and not (sum(p.raw_data)))] + if src_matrices or src_partitions: + import_from_db( + src_database=src_db, + src_zone_system_scenario=src_db.scenario(scenario_id), + src_matrix_ids=src_matrices, + src_partition_ids=src_partitions, + dst_database=dst_db, + dst_zone_system_scenario=dst_db.scenario(scenario_id)) + src_db.dispose() + self._report.append("End: %s" % _time.strftime("%c")) + return emmebank_paths + + def _copy_dir(self, src, dst, file_masks, check_metadata=False): + for name in os.listdir(src): + src_path = _join(src, name) + skip_file = bool([1 for mask in file_masks if _fnmatch(src_path, mask)]) + if skip_file: + continue + dst_path = _join(dst, name) + if os.path.isfile(src_path): + size = os.path.getsize(src_path) + if check_metadata and os.path.exists(dst_path): + same_size = os.path.getsize(dst_path) == size + same_time = os.path.getmtime(dst_path) == os.path.getmtime(src_path) + if same_size and same_time: + continue + self._report.append(_time.strftime("%c")) + self._report.append(dst_path + file_size(size)) + self._stats["size"] += size + self._stats["count"] += 1 + # shutil.copy2 performs 5-10 times faster on download, and ~20% faster on upload + # than os.system copy calls + src_time = os.path.getmtime(src_path) + if name == 'persons.csv' or "mgra13_based" in name: + src_time = os.path.getmtime(src_path) + if os.path.exists(dst_path): + dest_time = os.path.getmtime(dst_path) + if dest_time <= src_time: + _shutil.copy2(src_path, dst_path) + else: + pass + else: + _shutil.copy2(src_path, dst_path) + else: + _shutil.copy2(src_path, dst_path) + self._report.append(_time.strftime("%c")) + elif os.path.isdir(src_path): + if not os.path.exists(dst_path): + os.mkdir(dst_path) + self._report.append(dst_path) + self._copy_dir(src_path, dst_path, file_masks, check_metadata) + + def log_report(self): + size, count = file_size(self._stats["size"]), self._stats["count"] + name = "File copy report: copied {count} files {size}".format(count=count, size=size) + report = _m.PageBuilder(title=name) + report.add_html("
".join(self._report)) + _m.logbook_write(name, report.render()) + + +_suffixes = ['bytes', 'KiB', 'MiB', 'GiB', 'TiB'] + +def file_size(size): + order = int(log10(size) / 3) if size else 0 + return ' {} {}'.format(round(float(size) / (10**(order*3)), 1), _suffixes[order]) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/general.py b/sandag_abm/src/main/emme/toolbox/utilities/general.py new file mode 100644 index 0000000..6879412 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/general.py @@ -0,0 +1,386 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// transit_assignment.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// + +TOOLBOX_ORDER = 102 + + +import inro.modeller as _m +import inro.emme.datatable as _dt +import inro.emme.core.exception as _except +from osgeo import ogr as _ogr +from contextlib import contextmanager as _context +from itertools import izip as _izip +import traceback as _traceback +import re as _re +import json as _json +import time as _time +import os +import numpy as _numpy + +_omx = _m.Modeller().module("sandag.utilities.omxwrapper") + + +class UtilityTool(_m.Tool()): + + tool_run_msg = "" + + def page(self): + pb = _m.ToolPageBuilder(self, runnable=False) + pb.title = "General utility" + pb.description = """Utility tool / module for common code. Not runnable.""" + pb.branding_text = "- SANDAG" + if self.tool_run_msg: + pb.add_html(self.tool_run_msg) + + return pb.render() + + def run(self): + pass + + +class NetworkCalculator(object): + def __init__(self, scenario): + self._scenario = scenario + self._network_calc = _m.Modeller().tool( + "inro.emme.network_calculation.network_calculator") + + def __call__(self, result, expression, selections=None, aggregation=None): + spec = { + "result": result, + "expression": expression, + "aggregation": aggregation, + "type": "NETWORK_CALCULATION" + } + if selections is not None: + if isinstance(selections, basestring): + selections = {"link": selections} + spec["selections"] = selections + else: + spec["selections"] = {"link": "all"} + return self._network_calc(spec, self._scenario) + + +@_context +def temp_matrices(emmebank, mat_type, total=1, default_value=0.0): + matrices = [] + try: + while len(matrices) != int(total): + try: + ident = emmebank.available_matrix_identifier(mat_type) + except _except.CapacityError: + raise _except.CapacityError( + "Insufficient room for %s required temp matrices." % total) + matrices.append(emmebank.create_matrix(ident, default_value)) + yield matrices[:] + finally: + for matrix in matrices: + # In case of transient file conflicts and lag in windows file handles over the network + # attempt to delete file 10 times with increasing delays 0.05, 0.2, 0.45, 0.8 ... 5 + remove_matrix = lambda: emmebank.delete_matrix(matrix) + retry(remove_matrix) + + +def retry(fcn, attempts=10, init_wait=0.05, error_types=(RuntimeError, WindowsError)): + for attempt in range(1, attempts + 1): + try: + fcn() + return + except error_types: + if attempt > attempts: + raise + _time.sleep(init_wait * (attempt**2)) + + +@_context +def temp_attrs(scenario, attr_type, idents, default_value=0.0): + attrs = [] + try: + for ident in idents: + attrs.append(scenario.create_extra_attribute(attr_type, ident, default_value)) + yield attrs[:] + finally: + for attr in attrs: + scenario.delete_extra_attribute(attr) + + +@_context +def backup_and_restore(scenario, backup_attributes): + backup = {} + for elem_type, attributes in backup_attributes.iteritems(): + backup[elem_type] = scenario.get_attribute_values(elem_type, attributes) + try: + yield + finally: + for elem_type, attributes in backup_attributes.iteritems(): + scenario.set_attribute_values(elem_type, attributes, backup[elem_type]) + + +class DataTableProc(object): + + def __init__(self, table_name, path=None, data=None, convert_numeric=False): + modeller = _m.Modeller() + desktop = modeller.desktop + project = desktop.project + self._dt_db = dt_db = project.data_tables() + self._convert_numeric = convert_numeric + if path: + #try: + source = _dt.DataSource(path) + #except: + # raise Exception("Cannot open file at %s" % path) + layer = source.layer(table_name) + self._data = layer.get_data() + elif data: + table = dt_db.create_table(table_name, data, overwrite=True) + self._data = data + else: + table = dt_db.table(table_name) + self._data = table.get_data() + self._load_data() + + def _load_data(self): + data = self._data + if self._convert_numeric: + values = [] + for a in data.attributes(): + attr_values = _numpy.copy(a.values) + attr_values[attr_values == ''] = 0 + try: + values.append(attr_values.astype("int")) + except ValueError: + try: + values.append(attr_values.astype("float")) + except ValueError: + values.append(a.values) + self._values = values + else: + self._values = [a.values for a in data.attributes()] + self._attr_names = [a.name for a in data.attributes()] + self._index = dict((k, i) for i,k in enumerate(self._attr_names)) + if "geometry" in self._attr_names: + geo_coords = [] + attr = data.attribute("geometry") + for record in attr.values: + geo_obj = _ogr.CreateGeometryFromWkt(record.text) + geo_coords.append(geo_obj.GetPoints()) + self._values.append(geo_coords) + self._attr_names.append("geo_coordinates") + + def __iter__(self): + values, attr_names = self._values, self._attr_names + return (dict(_izip(attr_names, record)) + for record in _izip(*values)) + + def save(self, name, overwrite=False): + self._dt_db.create_table(name, self._data, overwrite=overwrite) + + def values(self, name): + index = self._index[name] + return self._values[index] + + +class Snapshot(object): + def __getitem__(self, key): + return getattr(self, key) + + def __setitem__(self, key, value): + setattr(self, key, value) + + def to_snapshot(self): + try: + attributes = getattr(self, "attributes", []) + snapshot = {} + for name in attributes: + snapshot[name] = unicode(self[name]) + return _json.dumps(snapshot) + except Exception: + return "{}" + + def from_snapshot(self, snapshot): + try: + snapshot = _json.loads(snapshot) + attributes = getattr(self, "attributes", []) + for name in attributes: + self[name] = snapshot[name] + except Exception, error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error), False) + return self + + def get_state(self): + attributes = getattr(self, "attributes", []) + state = {} + for name in attributes: + try: + state[name] = self[name] + except _m.AttributeError, error: + state[name] = unicode(error) + return state + + +def log_snapshot(name, namespace, snapshot): + try: + _m.logbook_snapshot(name=name, comment="", namespace=namespace, + value=_json.dumps(snapshot)) + except Exception as error: + print error + + +class ExportOMX(object): + def __init__(self, file_path, scenario, omx_key="NAME"): + self.file_path = file_path + self.scenario = scenario + self.emmebank = scenario.emmebank + self.omx_key = omx_key + + @property + def omx_key(self): + return self._omx_key + + @omx_key.setter + def omx_key(self, omx_key): + self._omx_key = omx_key + text_encoding = self.emmebank.text_encoding + if omx_key == "ID_NAME": + self.generate_key = lambda m: "%s_%s" % ( + m.id.encode(text_encoding), m.name.encode(text_encoding)) + elif omx_key == "NAME": + self.generate_key = lambda m: m.name.encode(text_encoding) + elif omx_key == "ID": + self.generate_key = lambda m: m.id.encode(text_encoding) + + def __enter__(self): + self.trace = _m.logbook_trace(name="Export matrices to OMX", + attributes={ + "file_path": self.file_path, "omx_key": self.omx_key, + "scenario": self.scenario, "emmebank": self.emmebank.path}) + self.trace.__enter__() + self.omx_file = _omx.open_file(self.file_path, 'w') + try: + self.omx_file.create_mapping('zone_number', self.scenario.zone_numbers) + except LookupError: + pass + return self + + def __exit__(self, exc_type, exc_val, exc_tb): + self.omx_file.close() + self.trace.__exit__(exc_type, exc_val, exc_tb) + + def write_matrices(self, matrices): + if isinstance(matrices, dict): + for key, matrix in matrices.iteritems(): + self.write_matrix(matrix, key) + else: + for matrix in matrices: + self.write_matrix(matrix) + + def write_matrix(self, matrix, key=None): + text_encoding = self.emmebank.text_encoding + matrix = self.emmebank.matrix(matrix) + if key is None: + key = self.generate_key(matrix) + numpy_array = matrix.get_numpy_data(self.scenario.id) + if matrix.type == "DESTINATION": + n_zones = len(numpy_array) + numpy_array = _numpy.resize(numpy_array, (1, n_zones)) + elif matrix.type == "ORIGIN": + n_zones = len(numpy_array) + numpy_array = _numpy.resize(numpy_array, (n_zones, 1)) + attrs = {"description": matrix.description.encode(text_encoding)} + self.write_array(numpy_array, key, attrs) + + def write_clipped_array(self, numpy_array, key, a_min, a_max=None, attrs={}): + if a_max is not None: + numpy_array = numpy_array.clip(a_min, a_max) + else: + numpy_array = numpy_array.clip(a_min) + self.write_array(numpy_array, key, attrs) + + def write_array(self, numpy_array, key, attrs={}): + shape = numpy_array.shape + if len(shape) == 2: + chunkshape = (1, shape[0]) + else: + chunkshape = None + attrs["source"] = "Emme" + numpy_array = numpy_array.astype(dtype="float64", copy=False) + omx_matrix = self.omx_file.create_matrix( + key, obj=numpy_array, chunkshape=chunkshape, attrs=attrs) + + +class OMXManager(object): + def __init__(self, directory, name_tmplt): + self._directory = directory + self._name_tmplt = name_tmplt + self._omx_files = {} + + def lookup(self, name_args, key): + file_name = self._name_tmplt % name_args + omx_file = self._omx_files.get(file_name) + if omx_file is None: + file_path = os.path.join(self._directory, file_name) + omx_file = _omx.open_file(file_path, 'r') + self._omx_files[file_name] = omx_file + return omx_file[key].read() + + def file_exists(self, name_args): + file_name = self._name_tmplt % name_args + file_path = os.path.join(self._directory, file_name) + return os.path.isfile(file_path) + + def zone_list(self, file_name): + omx_file = self._omx_files[file_name] + mapping_name = omx_file.list_mappings()[0] + zone_mapping = omx_file.mapping(mapping_name).items() + zone_mapping.sort(key=lambda x: x[1]) + omx_zones = [x[0] for x in zone_mapping] + return omx_zones + + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_val, exc_tb): + for omx_file in self._omx_files.values(): + omx_file.close() + self._omx_files = {} + + +class CSVReader(object): + def __init__(self, path): + self._path = path + self._f = None + self._fields = None + + def __enter__(self): + self._f = open(self._path) + header = self._f.next() + self._fields = [h.strip().upper() for h in header.split(",")] + return self + + def __exit__(self, exception_type, exception_value, traceback): + self._f.close() + self._f = None + self._fields = None + + def __iter__(self): + return self + + @property + def fields(self): + return list(self._fields) + + def next(self): + line = self._f.next() + tokens = [t.strip() for t in line.split(",")] + return dict(zip(self._fields, tokens)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py b/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py new file mode 100644 index 0000000..67564f8 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py @@ -0,0 +1,91 @@ +##////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2019. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// utilities/omxwrapper.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#/////////////////////////////////////////////////////////////////////////////// +import inro.modeller as _m + + +try: + import openmatrix as _omx + + + def open_file(file_path, mode): + return OmxMatrix(_omx.open_file(file_path, mode)) +except Exception, e: + import omx as _omx + + + def open_file(file_path, mode): + return OmxMatrix(_omx.openFile(file_path, mode)) + +class OmxMatrix(object): + + def __init__(self, matrix): + self.matrix = matrix + + def mapping(self, name): + return self.matrix.mapping(name) + + def list_mappings(self): + return self.matrix.listMappings() + + def __getitem__(self, key): + return self.matrix[key] + + def __setitem__(self, key, value): + self.matrix[key] = value + + def create_mapping(self, name, ids): + exception_raised = False + try: + self.matrix.create_mapping(name, ids) # Emme 44 and above + except Exception, e: + exception_raised = True + + if exception_raised: + self.matrix.createMapping(name, ids) # Emme 437 + + + def create_matrix(self, key, obj, chunkshape, attrs): + exception_raised = False + try: # Emme 44 and above + self.matrix.create_matrix( + key, + obj=obj, + chunkshape=chunkshape, + attrs=attrs + ) + except Exception, e: + exception_raised = True + + if exception_raised: # Emme 437 + self.matrix.createMatrix( + key, + obj=obj, + chunkshape=chunkshape, + attrs=attrs + ) + + def close(self): + self.matrix.close() + + + +class OmxWrapper(_m.Tool()): + def page(self): + pb = _m.ToolPageBuilder( + self, + runnable=False, + title="OMX wrapper", + description="OMX utility for handling of OMX related libraries" + ) + return pb.render() \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/utilities/properties.py b/sandag_abm/src/main/emme/toolbox/utilities/properties.py new file mode 100644 index 0000000..228bc07 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/properties.py @@ -0,0 +1,599 @@ +##////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// utilities/properties.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// + +TOOLBOX_ORDER = 103 + + +import inro.modeller as _m +import traceback as _traceback +from collections import OrderedDict +import csv +import os +import time + + +class PropertiesSetter(object): + + startFromIteration = _m.Attribute(int) + sample_rates = _m.Attribute(str) + + useLocalDrive = _m.Attribute(bool) + skip4Ds = _m.Attribute(bool) + skipBuildNetwork = _m.Attribute(bool) + skipInputChecker = _m.Attribute(bool) + skipInitialization = _m.Attribute(bool) + deleteAllMatrices = _m.Attribute(bool) + skipCopyWarmupTripTables = _m.Attribute(bool) + skipWalkLogsums = _m.Attribute(bool) + skipCopyWalkImpedance = _m.Attribute(bool) + skipBikeLogsums = _m.Attribute(bool) + skipCopyBikeLogsum = _m.Attribute(bool) + + skipHighwayAssignment_1 = _m.Attribute(bool) + skipHighwayAssignment_2 = _m.Attribute(bool) + skipHighwayAssignment_3 = _m.Attribute(bool) + skipTransitSkimming_1 = _m.Attribute(bool) + skipTransitSkimming_2 = _m.Attribute(bool) + skipTransitSkimming_3 = _m.Attribute(bool) + skipTransponderExport_1 = _m.Attribute(bool) + skipTransponderExport_2 = _m.Attribute(bool) + skipTransponderExport_3 = _m.Attribute(bool) + skipCoreABM_1 = _m.Attribute(bool) + skipCoreABM_2 = _m.Attribute(bool) + skipCoreABM_3 = _m.Attribute(bool) + skipOtherSimulateModel_1 = _m.Attribute(bool) + skipOtherSimulateModel_2 = _m.Attribute(bool) + skipOtherSimulateModel_3 = _m.Attribute(bool) + skipMAASModel_1 = _m.Attribute(bool) + skipMAASModel_2 = _m.Attribute(bool) + skipMAASModel_3 = _m.Attribute(bool) + skipCTM_1 = _m.Attribute(bool) + skipCTM_2 = _m.Attribute(bool) + skipCTM_3 = _m.Attribute(bool) + skipEI_1 = _m.Attribute(bool) + skipEI_2 = _m.Attribute(bool) + skipEI_3 = _m.Attribute(bool) + skipExternalExternal_1 = _m.Attribute(bool) + skipExternalExternal_2 = _m.Attribute(bool) + skipExternalExternal_3 = _m.Attribute(bool) + skipTruck_1 = _m.Attribute(bool) + skipTruck_2 = _m.Attribute(bool) + skipTruck_3 = _m.Attribute(bool) + skipTripTableCreation_1 = _m.Attribute(bool) + skipTripTableCreation_2 = _m.Attribute(bool) + skipTripTableCreation_3 = _m.Attribute(bool) + + skipFinalHighwayAssignment = _m.Attribute(bool) + skipFinalHighwayAssignmentStochastic = _m.Attribute(bool) + skipFinalTransitAssignment = _m.Attribute(bool) + skipVisualizer = _m.Attribute(bool) + skipDataExport = _m.Attribute(bool) + skipDataLoadRequest = _m.Attribute(bool) + skipDeleteIntermediateFiles = _m.Attribute(bool) + + def _get_list_prop(self, name): + return [getattr(self, name + suffix) for suffix in ["_1", "_2", "_3"]] + + def _set_list_prop(self, name, value): + try: + for v_sub, suffix in zip(value, ["_1", "_2", "_3"]): + setattr(self, name + suffix, v_sub) + except: + for suffix in ["_1", "_2", "_3"]: + setattr(self, name + suffix, False) + + skipHighwayAssignment = property( + fget=lambda self: self._get_list_prop("skipHighwayAssignment"), + fset=lambda self, value: self._set_list_prop("skipHighwayAssignment", value)) + skipTransitSkimming = property( + fget=lambda self: self._get_list_prop("skipTransitSkimming"), + fset=lambda self, value: self._set_list_prop("skipTransitSkimming", value)) + skipTransponderExport = property( + fget=lambda self: self._get_list_prop("skipTransponderExport"), + fset=lambda self, value: self._set_list_prop("skipTransponderExport", value)) + skipCoreABM = property( + fget=lambda self: self._get_list_prop("skipCoreABM"), + fset=lambda self, value: self._set_list_prop("skipCoreABM", value)) + skipOtherSimulateModel = property( + fget=lambda self: self._get_list_prop("skipOtherSimulateModel"), + fset=lambda self, value: self._set_list_prop("skipOtherSimulateModel", value)) + skipMAASModel = property( + fget=lambda self: self._get_list_prop("skipMAASModel"), + fset=lambda self, value: self._set_list_prop("skipMAASModel", value)) + skipCTM = property( + fget=lambda self: self._get_list_prop("skipCTM"), + fset=lambda self, value: self._set_list_prop("skipCTM", value)) + skipEI = property( + fget=lambda self: self._get_list_prop("skipEI"), + fset=lambda self, value: self._set_list_prop("skipEI", value)) + skipExternalExternal = property( + fget=lambda self: self._get_list_prop("skipExternalExternal"), + fset=lambda self, value: self._set_list_prop("skipExternalExternal", value)) + skipTruck = property( + fget=lambda self: self._get_list_prop("skipTruck"), + fset=lambda self, value: self._set_list_prop("skipTruck", value)) + skipTripTableCreation = property( + fget=lambda self: self._get_list_prop("skipTripTableCreation"), + fset=lambda self, value: self._set_list_prop("skipTripTableCreation", value)) + + def __init__(self): + self._run_model_names = ( + "useLocalDrive", "skip4Ds", "skipInputChecker", + "startFromIteration", "skipInitialization", "deleteAllMatrices", "skipCopyWarmupTripTables", + "skipCopyBikeLogsum", "skipCopyWalkImpedance", "skipWalkLogsums", "skipBikeLogsums", "skipBuildNetwork", + "skipHighwayAssignment", "skipTransitSkimming", "skipTransponderExport", "skipCoreABM", "skipOtherSimulateModel", "skipMAASModel","skipCTM", + "skipEI", "skipExternalExternal", "skipTruck", "skipTripTableCreation", "skipFinalHighwayAssignment", 'skipFinalHighwayAssignmentStochastic', + "skipFinalTransitAssignment", "skipVisualizer", "skipDataExport", "skipDataLoadRequest", + "skipDeleteIntermediateFiles") + self._properties = None + + def add_properties_interface(self, pb, disclosure=False): + tool_proxy_tag = pb.tool_proxy_tag + title = "Run model - skip steps" + + pb.add_text_box('sample_rates', title="Sample rate by iteration:", size=20) + + contents = [""" +
+
+ +            + +
+ + + + + + + + """ % {"tool_proxy_tag": tool_proxy_tag}] + + skip_startup_items = [ + ("useLocalDrive", "Use the local drive during the model run"), + ("skip4Ds", "Skip running 4Ds"), + ("skipBuildNetwork", "Skip build of highway and transit network"), + ("skipInputChecker", "Skip running input checker"), + ("skipInitialization", "Skip matrix and transit database initialization"), + ("deleteAllMatrices", "    Delete all matrices"), + ("skipCopyWarmupTripTables","Skip import of warmup trip tables"), + ("skipWalkLogsums", "Skip walk logsums"), + ("skipCopyWalkImpedance", "Skip copy of walk impedance"), + ("skipBikeLogsums", "Skip bike logsums"), + ("skipCopyBikeLogsum", "Skip copy of bike logsum"), + ] + skip_per_iteration_items = [ + ("skipHighwayAssignment", "Skip highway assignments and skims"), + ("skipTransitSkimming", "Skip transit skims"), + ("skipTransponderExport", "Skip transponder accessibilities"), + ("skipCoreABM", "Skip core ABM"), + ("skipOtherSimulateModel", "Skip other simulation model"), + ("skipMAASModel", "Skip MAAS model"), + ("skipCTM", "Skip commercial vehicle sub-model"), + ("skipTruck", "Skip truck sub-model"), + ("skipEI", "Skip external-internal sub-model"), + ("skipExternalExternal", "Skip external-external sub-model"), + ("skipTripTableCreation", "Skip trip table creation"), + ] + skip_final_items = [ + ("skipFinalHighwayAssignment", "Skip final highway assignments"), + ("skipFinalHighwayAssignmentStochastic", "    Skip stochastic assignment"), + ("skipFinalTransitAssignment", "Skip final transit assignments"), + ("skipVisualizer", "Skip running visualizer"), + ("skipDataExport", "Skip data export"), + ("skipDataLoadRequest", "Skip data load request"), + ("skipDeleteIntermediateFiles", "Skip delete intermediate files"), + ] + + if disclosure: + contents.insert(0, """ +
+
%s
""" % title) + title = "" + + checkbox = '
' + checkbox_no_data = '' + + for name, label in skip_startup_items: + contents.append("" % label) + contents.append(checkbox % {"name": name, "tag": tool_proxy_tag}) + contents.append("") + contents.append("") + for i in range(1,4): + contents.append(checkbox_no_data % {"name": "all" + "_" + str(i)}) + for name, label in skip_per_iteration_items: + contents.append("" % label) + for i in range(1,4): + contents.append(checkbox % {"name": name + "_" + str(i), "tag": tool_proxy_tag}) + for name, label in skip_final_items: + contents.append("" % label) + contents.append("") + contents.append(checkbox % {"name": name, "tag": tool_proxy_tag}) + + contents.append("
Iteration 1Iteration 2Iteration 3
%s
Set / reset all
    %s
%s
") + if disclosure: + contents.append("") + + pb.wrap_html(title, "".join(contents)) + + pb.add_html(""" +""" % {"tool_proxy_tag": tool_proxy_tag, + "iter_items": str([x[0] for x in skip_per_iteration_items]), + "startup_items": str([x[0] for x in skip_startup_items]), + }) + return + + @_m.method(return_type=bool, argument_types=(str,)) + def get_value(self, name): + return bool(getattr(self, name)) + + @_m.method() + def load_properties(self): + if not os.path.exists(self.properties_path): + return + self._properties = props = Properties(self.properties_path) + _m.logbook_write("SANDAG properties interface load") + + self.startFromIteration = props.get("RunModel.startFromIteration", 1) + self.sample_rates = ",".join(str(x) for x in props.get("sample_rates")) + + self.useLocalDrive = props.get("RunModel.useLocalDrive", True) + self.skip4Ds = props.get("RunModel.skip4Ds", False) + self.skipBuildNetwork = props.get("RunModel.skipBuildNetwork", False) + self.skipInputChecker = props.get("RunModel.skipInputChecker", False) + self.skipInitialization = props.get("RunModel.skipInitialization", False) + self.deleteAllMatrices = props.get("RunModel.deleteAllMatrices", False) + self.skipCopyWarmupTripTables = props.get("RunModel.skipCopyWarmupTripTables", False) + self.skipWalkLogsums = props.get("RunModel.skipWalkLogsums", False) + self.skipCopyWalkImpedance = props.get("RunModel.skipCopyWalkImpedance", False) + self.skipBikeLogsums = props.get("RunModel.skipBikeLogsums", False) + self.skipCopyBikeLogsum = props.get("RunModel.skipCopyBikeLogsum", False) + + self.skipHighwayAssignment = props.get("RunModel.skipHighwayAssignment", [False, False, False]) + self.skipTransitSkimming = props.get("RunModel.skipTransitSkimming", [False, False, False]) + self.skipTransponderExport = props.get("RunModel.skipTransponderExport", [False, False, False]) + self.skipCoreABM = props.get("RunModel.skipCoreABM", [False, False, False]) + self.skipOtherSimulateModel = props.get("RunModel.skipOtherSimulateModel", [False, False, False]) + self.skipMAASModel = props.get("RunModel.skipMAASModel", [False, False, False]) + self.skipCTM = props.get("RunModel.skipCTM", [False, False, False]) + self.skipEI = props.get("RunModel.skipEI", [False, False, False]) + self.skipExternalExternal = props.get("RunModel.skipExternalExternal", [False, False, False]) + self.skipTruck = props.get("RunModel.skipTruck", [False, False, False]) + self.skipTripTableCreation = props.get("RunModel.skipTripTableCreation", [False, False, False]) + + self.skipFinalHighwayAssignment = props.get("RunModel.skipFinalHighwayAssignment", False) + self.skipFinalHighwayAssignmentStochastic = props.get("RunModel.skipFinalHighwayAssignmentStochastic", True) + self.skipFinalTransitAssignment = props.get("RunModel.skipFinalTransitAssignment", False) + self.skipVisualizer = props.get("RunModel.skipVisualizer", False) + self.skipDataExport = props.get("RunModel.skipDataExport", False) + self.skipDataLoadRequest = props.get("RunModel.skipDataLoadRequest", False) + self.skipDeleteIntermediateFiles = props.get("RunModel.skipDeleteIntermediateFiles", False) + + def save_properties(self): + props = self._properties + props["RunModel.startFromIteration"] = self.startFromIteration + props["sample_rates"] = [float(x) for x in self.sample_rates.split(",")] + + props["RunModel.useLocalDrive"] = self.useLocalDrive + props["RunModel.skip4Ds"] = self.skip4Ds + props["RunModel.skipBuildNetwork"] = self.skipBuildNetwork + props["RunModel.skipInputChecker"] = self.skipInputChecker + props["RunModel.skipInitialization"] = self.skipInitialization + props["RunModel.deleteAllMatrices"] = self.deleteAllMatrices + props["RunModel.skipCopyWarmupTripTables"] = self.skipCopyWarmupTripTables + props["RunModel.skipWalkLogsums"] = self.skipWalkLogsums + props["RunModel.skipCopyWalkImpedance"] = self.skipCopyWalkImpedance + props["RunModel.skipBikeLogsums"] = self.skipBikeLogsums + props["RunModel.skipCopyBikeLogsum"] = self.skipCopyBikeLogsum + + props["RunModel.skipHighwayAssignment"] = self.skipHighwayAssignment + props["RunModel.skipTransitSkimming"] = self.skipTransitSkimming + props["RunModel.skipTransponderExport"] = self.skipTransponderExport + props["RunModel.skipCoreABM"] = self.skipCoreABM + props["RunModel.skipOtherSimulateModel"] = self.skipOtherSimulateModel + props["RunModel.skipMAASModel"] = self.skipMAASModel + props["RunModel.skipCTM"] = self.skipCTM + props["RunModel.skipEI"] = self.skipEI + props["RunModel.skipExternalExternal"] = self.skipExternalExternal + props["RunModel.skipTruck"] = self.skipTruck + props["RunModel.skipTripTableCreation"] = self.skipTripTableCreation + + props["RunModel.skipFinalHighwayAssignment"] = self.skipFinalHighwayAssignment + props["RunModel.skipFinalHighwayAssignmentStochastic"] = self.skipFinalHighwayAssignmentStochastic + props["RunModel.skipFinalTransitAssignment"] = self.skipFinalTransitAssignment + props["RunModel.skipVisualizer"] = self.skipVisualizer + props["RunModel.skipDataExport"] = self.skipDataExport + props["RunModel.skipDataLoadRequest"] = self.skipDataLoadRequest + props["RunModel.skipDeleteIntermediateFiles"] = self.skipDeleteIntermediateFiles + + props.save() + + # Log current state of props interface for debugging of UI / file sync issues + tool_attributes = dict((name, getattr(self, name)) for name in self._run_model_names) + _m.logbook_write("SANDAG properties interface save", attributes=tool_attributes) + + +class PropertiesTool(PropertiesSetter, _m.Tool()): + + properties_path = _m.Attribute(unicode) + + def __init__(self): + super(PropertiesTool, self).__init__() + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.properties_path = os.path.join( + os.path.dirname(project_dir), "conf", "sandag_abm.properties") + + tool_run_msg = "" + + @_m.method(return_type=_m.UnicodeType) + def tool_run_msg_status(self): + return self.tool_run_msg + + def page(self): + if os.path.exists(self.properties_path): + self.load_properties() + pb = _m.ToolPageBuilder(self) + pb.title = 'Set properties' + pb.description = """Properties setting tool.""" + pb.branding_text = ' - SANDAG - Utilities' + tool_proxy_tag = pb.tool_proxy_tag + + pb.add_select_file('properties_path', 'file', title='Path to properties file:') + + pb.wrap_html("", """ +
""") + + pb.add_html(""" +""" % {"tool_proxy_tag": tool_proxy_tag}) + self.add_properties_interface(pb) + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + self.save_properties() + message = "Properties file saved" + self.tool_run_msg = _m.PageBuilder.format_info(message, escape=False) + except Exception, e: + self.tool_run_msg = _m.PageBuilder.format_exception( + e, _traceback.format_exc(e)) + raise + + def __call__(self, file_path): + return Properties(file_path) + + +class Properties(object): + + def __init__(self, path): + if os.path.isdir(path): + path = os.path.join(path, "sandag_abm.properties") + if not os.path.isfile(path): + raise Exception("properties files does not exist '%s'" % path) + self._path = os.path.normpath(os.path.abspath(path)) + self.load_properties() + + def load_properties(self): + self._prop = prop = OrderedDict() + self._comments = comments = {} + with open(self._path, 'r') as properties: + comment = [] + for line in properties: + line = line.strip() + if not line or line.startswith('#'): + comment.append(line) + continue + key, value = line.split('=') + key = key.strip() + tokens = value.split(',') + if len(tokens) > 1: + value = self._parse_list(tokens) + else: + value = self._parse(value) + prop[key] = value + comments[key], comment = comment, [] + self._timestamp = os.path.getmtime(self._path) + + def _parse_list(self, values): + converted_values = [] + for v in values: + converted_values.append(self._parse(v)) + return converted_values + + def _parse(self, value): + value = str(value).strip() + if value == 'true': + return True + elif value == 'false': + return False + for caster in int, float: + try: + return caster(value) + except ValueError: + pass + return value + + def _format(self, value): + if isinstance(value, bool): + return "true" if value else "false" + return str(value) + + def save(self, path=None): + if not path: + path = self._path + # check for possible interference if user edits the + # properties files directly while it is already open in Modeller + timestamp = os.path.getmtime(path) + if timestamp != self._timestamp: + raise Exception("%s file conflict - edited externally after loading" % path) + self["SavedFrom"] = "Emme Modeller properties writer Process ID %s" % os.getpid() + self["SavedLast"] = time.strftime("%b-%d-%Y %H:%M:%S") + with open(path, 'w') as f: + for key, value in self.iteritems(): + if isinstance(value, list): + value = ",".join([self._format(v) for v in value]) + else: + value = self._format(value) + comment = self._comments.get(key) + if comment: + for line in comment: + f.write(line) + f.write("\n") + f.write("%s = %s\n" % (key, value)) + self._timestamp = os.path.getmtime(path) + + def set_year_specific_properties(self, file_path): + with open(file_path, 'r') as f: + reader = csv.DictReader(f) + properties_by_year = {} + for row in reader: + year = str(row.pop("year")) + properties_by_year[year] = row + year_properties = properties_by_year.get(str(self["scenarioBuild"])) + if year_properties is None: + raise Exception("Row with year %s not found in %s" % (self["scenarioBuild"], file_path)) + self.update(year_properties) + + def __setitem__(self, key, item): + self._prop[key] = item + + def __getitem__(self, key): + return self._prop[key] + + def __repr__(self): + return "Properties(%s)" % self._path + + def __len__(self): + return len(self._prop) + + def __delitem__(self, key): + del self._prop[key] + + def clear(self): + return self._prop.clear() + + def has_key(self, k): + return self._prop.has_key(k) + + def pop(self, k, d=None): + return self._prop.pop(k, d) + + def update(self, *args, **kwargs): + return self._prop.update(*args, **kwargs) + + def keys(self): + return self._prop.keys() + + def values(self): + return self._prop.values() + + def items(self): + return self._prop.items() + + def iteritems(self): + return self._prop.iteritems() + + def pop(self, *args): + return self._prop.pop(*args) + + def get(self, k, default=None): + try: + return self[k] + except KeyError: + return default + + def __cmp__(self, dict): + return cmp(self._prop, dict) + + def __contains__(self, item): + return item in self._prop + + def __iter__(self): + return iter(self._prop) + + def __unicode__(self): + return unicode(repr(self._prop)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py b/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py new file mode 100644 index 0000000..ebaf2e1 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py @@ -0,0 +1,326 @@ +""" ABM Run Time Summary Tool + +Generates a CSV file containing a run time summary for +a completed ABM run. Utilizes the Emme Modeler API to +query the Emme logbook for the run time information. + +""" + +# Importing libraries +import os +import pandas as pd +import traceback as _traceback +import inro.emme.desktop.app as _app +import inro.modeller as _m +from functools import reduce + +_dir = os.path.dirname +_join = os.path.join + +ATTR_SUFFIX = "_304A7365_C276_493A_AB3B_9B2D195E203F" + +# Define unneeded entries +exclude = ('Copy project data to local drive', + 'Export results for transponder ownership model', + 'Check free space on C', + 'Data load request', + 'Delete', + 'Move', + 'Create drive', + 'Start matrix', + 'Start JPPF', + 'Start Hh', + 'Start HH') + + +class RunTime(_m.Tool()): + + def __init__(self): + project_dir = _dir(_m.Modeller().desktop.project.path) + self.path = _dir(project_dir) + self.output_path = '' + self.output_summary_path = '' + self.begin = '' + self.end = '' + + def run(self): + """ + Executes Run Time Summary tool + """ + self.tool_run_msg = "" + try: + self(path=self.path) + run_msg = "Run Time Tool Complete" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg, + escape=False) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + + return + + def __call__(self, path=""): + """ + Calculates ABM run times and saves to CSV file + + :param path: Scenario file path + """ + + # Get element IDs for model runs + run_ids = self.get_runs() + + # Define needed attributes (begin and end times) + self.begin = "begin" + ATTR_SUFFIX + self.end = "end" + ATTR_SUFFIX + attrs = [self.begin, self.end] + + runtime_dfs = [] + for run_id in run_ids: + name = 'Run ID: {}'.format(run_id) + + # Creating dummy total time + total_entry = (run_id, 'Total Run Time', '0:0') + + # Get second level child entry run times if they exist + child_runtimes = self.get_child_runtimes(run_id, attrs) + + # Get third (final) level child entry run times + final_runtimes = [total_entry] + for index, info in enumerate(child_runtimes): + if info[1] == 'Final traffic assignments': + final_runtimes.append([0, 'Iteration 4', 0]) + final_runtimes.append(info) + if 'Iteration' in info[1]: + iter_str = '_{}'.format(info[1]) + + # Manually inserting matrix, hh, node, and jppf runtimes + start_proc = "Start Matrix manager, JPPF Driver, " + \ + "HH manager, and Nodes manager" + iter_str + final_runtimes += [[0, start_proc, '0:01']] + + # Add iteration to children + iteration_children = self.get_child_runtimes( + info[0], attrs) + for index, child in enumerate(iteration_children): + step = child[1] + iteration_children[index][1] = step + iter_str + + final_runtimes += iteration_children + + # Create run time summary table + index = [x[1] for x in final_runtimes] + values = [x[2] for x in final_runtimes] + runtime_series = pd.Series(index=index, data=values) + runtime_series.name = name + runtime_df = runtime_series.to_frame() + + # Create intial time + zero_time = pd.to_datetime('0:0', format='%H:%M') + + # Calculate iteration 4 run time if it exists + iter_str = 'Iteration 4' + if iter_str in runtime_df.index: + iter_4_index = runtime_df.index.get_loc('Iteration 4') + iter_4_df = runtime_df.iloc[iter_4_index+1:, :].copy() + iter_4_df[name] = (pd.to_datetime( + iter_4_df[name], format='%H:%M') - + zero_time) + iter_4_time = iter_4_df[name].sum() + runtime_df.loc[iter_str, :] = self.format_runtime(iter_4_time) + + # Calculate total runtime + is_iter_row = pd.Series(runtime_df.index).str.startswith('Iter') + total_df = runtime_df[~is_iter_row.values].copy() + total_df[name] = (pd.to_datetime(total_df[name], format='%H:%M') - + zero_time) + total_time = total_df[name].sum() + run_str = 'Total Run Time' + runtime_df.loc[run_str, :] = self.format_runtime(total_time) + + # Remove unneeded entries + is_excluded = pd.Series(runtime_df.index).str.startswith(exclude) + runtime_df = runtime_df[~(is_excluded.values)] + runtime_dfs.append(runtime_df) + + # Merge all run time data frames if more than one exists and save + file_name = 'runtime_summary.csv' + self.output_path = _join(path, 'output', file_name) + result = self.combine_dfs(runtime_dfs) + if result[1]: + result[0].to_csv(self.output_path, header=True, index=False) + else: + result[0].to_csv(self.output_path, header=False) + + return + + def get_runs(self): + """ + Queries the Emme logbook to retrieve the IDs of all + model runs. + + :returns: List of IDs of model runs + """ + + # Emme logbook query + query = """ + SELECT elements.element_id, elements.tag + FROM elements + JOIN attributes KEYVAL1 ON (elements.element_id=KEYVAL1.element_id) + WHERE (KEYVAL1.name=="self" + AND KEYVAL1.value LIKE "sandag.master_run") + ORDER BY elements.element_id ASC + """ + all_entries = _m.logbook_query(query) + + # Retrieves model run IDs + run_ids = [] + for entry in all_entries: + parent_id = entry[0] + run_ids.append(parent_id) + + if len(run_ids) == 0: + raise ValueError('A model run does not exist.') + + return run_ids + + def get_attributes(self, element_id): + """ + Queries all the attributes of an Emme logbook element + + :param element_id: Integer ID of element + :returns: List of tuples containing information for + different attributes of an element. + """ + + # Emme logbook query + query = """ + SELECT name, value FROM attributes + WHERE attributes.element_id == %i + """ % element_id + + return _m.logbook_query(query) + + def format_runtime(self, time): + """ + Transforms a datetime object to a reformatted + date string. Formatted as '{hours}:{'minutes'}' + + :param time Datetime object + """ + + hours = str(int(time.total_seconds() // 3600)) + minutes = str(int((time.total_seconds() % 3600) // 60)).zfill(2) + formatted_runtime = hours + ":" + minutes + + return formatted_runtime + + def calc_runtime(self, begin, end): + """ + Helper function for get_child_runtimes + + Converts beginning and end datetime strings into + a formatted time delta. Formatted as '{hours}:{minutes}' + + :param begin: String representing beginning date + :param end: String representating ending date + :returns: String representing element runtime + """ + + # Calculate total run time + total_runtime = pd.to_datetime(end) - pd.to_datetime(begin) + + # Format run time: '{hours}:{minutes}' + formatted_runtime = self.format_runtime(total_runtime) + + # Defaulting zero second times to 1 second + if formatted_runtime == '0:00': + formatted_runtime = '0:01' + + return formatted_runtime + + def get_children(self, parent_id): + """ + Retrieves all child elements for a parent element + + :param parent_id: Integer ID of parent element + :returns: List of tuples containing child IDs and names + """ + + # Emme logbook query + query = """ + SELECT elements.element_id, elements.tag + FROM elements WHERE parent_id==%i + ORDER BY elements.element_id ASC + """ % parent_id + child_entries = _m.logbook_query(query) + + return child_entries + + def get_child_runtimes(self, parent_id, attrs): + """ + Calculates the run times for the child elements of + a parent element + + :param parent_id: Integer ID of parent element + :param attrs: List of strings representing attributes to query + :returns: List of tuples containing information for child elements + """ + + # Get child elements + all_child_entries = self.get_children(parent_id) + + # Calculates run times for each child element + runtime_child_entries = [] + for element_id, name in all_child_entries: + attributes = dict(self.get_attributes(element_id)) + + # Gets element information if desired attribute is + # available and it is not included in the excluded list + if attrs[0] in attributes: + begin = attributes[attrs[0]] + + # Handles cases where model fails mid iteration + try: + end = attributes[attrs[1]] + runtime = self.calc_runtime(begin, end) + except KeyError: + end = None + runtime = None + runtime_child_entries.append([element_id, name, runtime]) + + return runtime_child_entries + + def combine_dfs(self, df_list): + """ + Combines a list of Pandas DataFrames into a single + summary DataFrame + + :param df_list: List of Pandas DataFrames + :returns: Tuple contianing single run time summary DataFrame + and boolean whether it contains multiple runs + """ + if len(df_list) > 1: + # Drop tables with less than 2 entries + final_dfs = [] + for df in df_list: + if len(df.dropna()) > 1: + final_dfs.append(df.reset_index(drop=False)) + + # Merge all data frames + final_df = reduce(lambda left, right: + pd.merge(left, right, on=['index'], how='outer'), + final_dfs) + + # Remove appended iteration markers + final_df['index'] = (final_df['index'].apply( + lambda x: x.split('_')[0])) + + final_df = final_df.rename(columns={'index': 'Step'}) + result = (final_df, True) + + else: + final_df = df_list[0] + result = (final_df, False) + + return result diff --git a/sandag_abm/src/main/emme/toolbox/validation/validation.py b/sandag_abm/src/main/emme/toolbox/validation/validation.py new file mode 100644 index 0000000..13ae759 --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/validation/validation.py @@ -0,0 +1,290 @@ +""" +Created on March 2020 + +@author: cliu +""" + +TOOLBOX_ORDER = 105 + +import inro.modeller as _m +import traceback as _traceback +import inro.emme.database.emmebank as _eb +import inro.emme.desktop.app as _app +import inro.emme.core.exception as _except +from collections import OrderedDict +import os +import pandas as pd +import openpyxl +from functools import reduce + + +gen_utils = _m.Modeller().module("sandag.utilities.general") +dem_utils = _m.Modeller().module("sandag.utilities.demand") + + +format = lambda x: ("%.6f" % x).rstrip('0').rstrip(".") +id_format = lambda x: str(int(x)) + +class validation(_m.Tool(), gen_utils.Snapshot): + + main_directory = _m.Attribute(str) + base_scenario_id = _m.Attribute(int) + traffic_emmebank = _m.Attribute(str) + #transit_emmebank = _m.Attribute(str) + attributes = _m.Attribute(str) + + tool_run_msg = "" + + def __init__(self): + project_dir = os.path.dirname(_m.Modeller().desktop.project.path) + self.main_directory = os.path.dirname(project_dir) + self.base_scenario_id = 100 + self.traffic_emmebank = os.path.join(project_dir, "Database", "emmebank") + #self.transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") + self.attributes = ["main_directory", "traffic_emmebank", "transit_emmebank","base_scenario_id"] + + def page(self): + pb = _m.ToolPageBuilder(self) + pb.title = "Validation Procedure" + pb.description = """ +Export traffic flow to Excel files for base year validation.""" + pb.branding_text = "- SANDAG - Validation" + if self.tool_run_msg != "": + pb.tool_run_status(self.tool_run_msg_status) + + pb.add_select_file('main_directory', 'directory', + title='Select main directory') + + pb.add_select_file('traffic_emmebank', 'file', + title='Select traffic emmebank') + #pb.add_select_file('transit_emmebank', 'file', + # title='Select transit emmebank') + return pb.render() + + def run(self): + self.tool_run_msg = "" + try: + results = self(self.main_directory, self.traffic_emmebank, self.base_scenario_id) + #results = self(self.main_directory, self.traffic_emmebank, self.transit_emmebank, self.base_scenario_id) + run_msg = "Export completed" + self.tool_run_msg = _m.PageBuilder.format_info(run_msg) + except Exception as error: + self.tool_run_msg = _m.PageBuilder.format_exception( + error, _traceback.format_exc(error)) + raise + @_m.logbook_trace("Export network data for base year Validation", save_arguments=True) + + def __call__(self, main_directory, traffic_emmebank, base_scenario_id): + #def __call__(self, main_directory, traffic_emmebank, transit_emmebank, base_scenario_id): + print "in validation module" + attrs = { + "main_directory": main_directory, + "traffic_emmebank": str(traffic_emmebank), + #"transit_emmebank": str(transit_emmebank), + "base_scenario_id": base_scenario_id, + "self": str(self) + } + + gen_utils.log_snapshot("Validation procedure", str(self), attrs) + + traffic_emmebank = _eb.Emmebank(traffic_emmebank) + #transit_emmebank = _eb.Emmebank(transit_emmebank) + export_path = os.path.join(main_directory, "analysis/validation") + transitbank_path = os.path.join(main_directory, "emme_project/Database_transit/emmebank") + source_file = os.path.join(export_path, "source_EMME.xlsx") + df = pd.read_excel(source_file, header=None, sheet_name='raw') + writer = pd.ExcelWriter(source_file, engine='openpyxl') + book = openpyxl.load_workbook(source_file) + writer.book = book + writer.sheets = dict((ws.title, ws) for ws in book.worksheets) + + #periods = ["EA"] + periods = ["EA", "AM", "MD", "PM", "EV"] + + period_scenario_ids = OrderedDict((v, i) for i, v in enumerate(periods, start=int(base_scenario_id) + 1)) + + #-------export tranffic data-------- + dfHwycov = pd.read_excel(source_file, sheetname='raw', usecols ="A") + for p, scen_id in period_scenario_ids.iteritems(): + base_scenario = traffic_emmebank.scenario(scen_id) + + #create and calculate @trk_non_pce + create_attribute = _m.Modeller().tool( + "inro.emme.data.extra_attribute.create_extra_attribute") + net_calculator = _m.Modeller().tool( + "inro.emme.network_calculation.network_calculator") + try: + att = create_attribute("LINK", "@trk_non_pce", "total trucs in non-Pce", 0, overwrite=True, scenario = base_scenario) + except: #if "@trk_non_pce" has been created + pass + cal_spec = {"result": att.id, + "expression": "@trk_l_non_pce+@trk_m_non_pce+@trk_h_non_pce", + "aggregation": None, + "selections": {"link": "mode=d"}, + "type": "NETWORK_CALCULATION" + } + net_calculator(cal_spec, scenario = base_scenario) + + network = base_scenario.get_partial_network(["LINK"], include_attributes=True) + + df, dftr, dfsp = self.export_traffic(export_path, traffic_emmebank, scen_id, network, source_file, p, dfHwycov) + dfsp[p + "_Speed"] = dfsp[p + "_Speed"].astype(int) + if p == "EA": + df_total = df + dftr_total = dftr + dfsp_total = dfsp + else: + df_total = df_total.join(df[p + "_Flow"]) + dftr_total = dftr_total.join(dftr[p + "_TruckFlow"]) + dfsp_total = dfsp_total.join(dfsp[p + "_Speed"]) + + df_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=0, startrow=0) + dftr_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=7, startrow=0) + dfsp_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=14, startrow=0) + writer.save() + + #----------------------------------export transit data---------------------------------- + + desktop = _m.Modeller().desktop + data_explorer = desktop.data_explorer() + + try: + data_explorer.add_database(transitbank_path) + except: + pass #if transit database already included in the project + all_databases = data_explorer.databases() + for database in all_databases: + if "transit" in database.name(): + database.open() + break + for p, scen_id in period_scenario_ids.iteritems(): + scen = database.scenario_by_number(scen_id) + data_explorer.replace_primary_scenario(scen) + self.export_transit(export_path, desktop, p) + + # -----------------close or remove transit databack from the project----------------- + database.close() + if "T:" not in main_directory: + data_explorer.remove_database(database) + all_databases = data_explorer.databases() + for database in all_databases: + if "transit" not in database.name(): + database.open() + break + + #------Combine into one datafram and write out + routeDict = {'c':23, 'l':24, 'y':25, 'r':26, 'p':27, 'e':28, 'b':29} + filenames = [] + for p in periods: + file = os.path.join(export_path, "transit_" + p + ".csv") + filenames.append(file) + + df_detail = [] + df_board = [] + df_passMile = [] + + for f in filenames: + df_detail.append(pd.read_csv(f)) + + pnum = 0 + for datafm in df_detail: + datafm['route'] = datafm.apply(lambda row: str(int(row.Line/1000))+row.Mode, axis = 1) + df_board.append(datafm.groupby(['route'])['Pass.'].agg('sum').reset_index()) + df_board[pnum].rename(columns = {'Pass.':periods[pnum]+'_Board'}, inplace=True) + + df_passMile.append(datafm.groupby(['route'])['Pass. dist.'].agg('sum').reset_index()) + df_passMile[pnum].rename(columns = {'Pass. dist.':periods[pnum]+'_PsgrMile'}, inplace=True) + + pnum += 1 + + frame1 = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), df_board ) + frame2 = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), df_passMile ) + frame = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), [frame1,frame2]) + + idx = 1 + frame.insert(loc=idx, column='mode_transit_route_id', value="") + frame['mode_transit_route_id'] = frame['route'].apply(lambda x: routeDict[x[-1]]) + frame['route'] = frame['route'].apply(lambda x: int(x[:-1])) + + writer = pd.ExcelWriter(source_file, engine='openpyxl') + book = openpyxl.load_workbook(source_file) + writer.book = book + writer.sheets = dict((ws.title, ws) for ws in book.worksheets) + frame.to_excel(writer, sheet_name='transit_general', header=True, index=False, startcol=1, startrow=0) + writer.save() + + for p in periods: + file = os.path.join(export_path, "transit_" + p + ".csv") + os.remove(file) + + @_m.logbook_trace("Export traffic load data by period - validaton") + def export_traffic(self, export_path, traffic_emmebank, scen_id, network, filename, period, dfHwycov): + def get_network_value(attrs, emmeAttrName, headerStr, df): + reverse_link = link.reverse_link + key, att = attrs[0] # expected to be the link id + values = [id_format(link[att])] + reverse_link = link.reverse_link + for key, att in attrs[1:]: + if key == "AN": + values.append(link.i_node.id) + elif key == "BN": + values.append(link.j_node.id) + elif key.startswith("BA"): + print "line 148 key, att", key, att + name, default = att + if reverse_link and (abs(link["@tcov_id"]) == abs(reverse_link["@tcov_id"])): + values.append(format(reverse_link[name])) + else: + values.append(default) + elif att.startswith("#"): + values.append('"%s"' % link[att]) + else: + values.append(format(link[emmeAttrName])) + df = df.append({'TCOVID': values[0], period + headerStr: values[1]}, ignore_index=True) + df['TCOVID'] = df['TCOVID'].astype(float) + df[period + headerStr] = df[period + headerStr].astype(float) + return df + + # only the original forward direction links and auto links only + hwyload_attrs = [("TCOVID", "@tcov_id"), (period + "_Flow", "@non_pce_flow")] + trkload_attrs = [("TCOVID", "@tcov_id"), (period + "_TruckFlow", "@trk_non_pce")] + speedload_attrs = [("TCOVID", "@tcov_id"), (period + "_Speed", "@speed")] + df = pd.DataFrame(columns=['TCOVID', period + "_Flow"]) + dftr = pd.DataFrame(columns=['TCOVID', period + "_TruckFlow"]) + dfsp = pd.DataFrame(columns=['TCOVID', period + "_Speed"]) + print "scen_id", scen_id + auto_mode = network.mode("d") + scenario = traffic_emmebank.scenario(scen_id) + links = [l for l in network.links() if (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes))] + #links = [l for l in network.links() if l["@tcov_id"] > 0 and (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes))] + links.sort(key=lambda l: l["@tcov_id"]) + + for link in links: + if link["@tcov_id"] in dfHwycov['TCOVID'].tolist(): + reverse_link = link.reverse_link + df = get_network_value(hwyload_attrs, '@non_pce_flow', "_Flow", df) + dftr = get_network_value(trkload_attrs, '@trk_non_pce', "_TruckFlow", dftr) + dfsp = get_network_value(speedload_attrs, '@speed', "_Speed", dfsp) + return df, dftr, dfsp + + @_m.logbook_trace("Export transit load data by period - validaton") + def export_transit(self, export_path, desktop, p): + project_table_db = desktop.project.data_tables() + ws_path = ["General", "Results Analysis", "Transit", "Summaries", "Summary by line"] + root_ws_f = desktop.root_worksheet_folder() + table_item = root_ws_f.find_item(ws_path) + transit_table = table_item.open() + + #for i in delete_table_list: + # transit_table.delete_column(i) + transit_dt = transit_table.save_as_data_table("Transit_Summary", overwrite=True) + transit_table.close() + + transit_data = transit_dt.get_data() + project_path = r'T:\ABM\ABM_FY19\model_runs\ABM2Plus\SenTests4TAC' + transit_filepath = os.path.join(export_path, "transit_"+ p +".csv") + transit_data.export_to_csv(transit_filepath, separator=",") + + @_m.method(return_type=unicode) + def tool_run_msg_status(self): + return self.tool_run_msg \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/SandagCommon.rsc b/sandag_abm/src/main/gisdk/SandagCommon.rsc new file mode 100644 index 0000000..ac345f1 --- /dev/null +++ b/sandag_abm/src/main/gisdk/SandagCommon.rsc @@ -0,0 +1,405 @@ +//**************************************************************** +//**************************************************************** +//Common Macros +// +//Find a string in a text file +//read properties +//Export Matrix to CSV +//Export Matrix +//Matrix Size +//Create Matrix +//Aggregate Matrices +//Go GetMatrixCoreNames +//Get SL Query # +//close all +//CloseViews +//date and time +//SDdeletefile +//SDcopyfile +//SDrenamefile +//HwycadLog +//ForecastYearStr +//ForecastYearInt +//DeleteInterimFiles +//FileCheckDelete +//**************************************************************** +//**************************************************************** +Macro "find String"(file,key) +//file a string in a text file +//file--text file name; key--string to be found in text file +//return 0 is string found, otherwise return 1 +//wsu 6-20-2014 + shared path, path_study + result=1 + +// Get file name + dif2=GetDirectoryInfo(path+"\\"+file, "file") + if dif2.length>0 then do //use scenario file + fptr=openfile(path+"\\"+file,"r") + end + else do //use study file from data directory + fptr=openfile(path_study+"\\data\\"+file,"r") + end + + while not FileAtEOF(fptr) do + pos = PositionFrom(,ReadLine(fptr),key) + if pos>0 then do + result=0 + end + end + CloseFile(fptr) + Return(result) +EndMacro + +Macro "read properties"(file,key,ctype) + //ctype as string - Character Type - Valid "I" or anything else + //macro only reads integers and strings + //reads property as string and returns either an integer or a string + + shared path, path_study + +// Get file name + dif2=GetDirectoryInfo(path+"\\"+file, "file") + if dif2.length>0 then do //use scenario file + fptr=openfile(path+"\\"+file,"r") + end + else do //use study file from data directory + fptr=openfile(path_study+"\\data\\"+file,"r") + end + +// Search key in properties file + a = ReadArray(fptr) + for k=1 to a.length do + // search for the key (line number is stored as k value) + pos1 = position(a[k],key) + if pos1 =1 then do + // gets the integer on the rightside of "=" + keyword=ParseString(a[k], "=") + keyvaltrim=trim(keyword[2]) + if ctype = "I" then do // integer + keyval=S2I(keyvaltrim) + end + else do // if not I then it's a string + keyval = keyvaltrim // gets the string on the rightside of "=" + end + end + end + CloseFile(fptr) + Return(keyval) +EndMacro + +Macro "read properties array"(file,key,ctype) + //this Macro is to read an array property,index is 1-based. + shared path, path_study + + pStr=RunMacro("read properties",file,key,ctype) + pStr=trim(pStr) + pArray=ParseString(pStr, ",") + Return(pArray) +EndMacro + +Macro "Export Matrix to CSV" (path,filename,corename,filenameout) + m = OpenMatrix(path+"\\"+filename, "True") + mc = CreateMatrixCurrency(m,corename,,,) + rows = GetMatrixRowLabels(mc) + ExportMatrix(mc, rows, "Rows", "CSV", path+"\\"+filenameout, ) + return(1) +EndMacro + +Macro "Export Matrix" (path,filename,corename,filenameout,outputtype) + //path as string - path="T:\\transnet2\\devel\\sr12\\sr12_byear\\byear" + //filename as string - must be a matrix - filename="SLAgg.mtx" + //corename as string - corename="DAN" + //filenameout as string - filenameout="SLAgg.csv" + //outputtype as string - ("dBASE", "FFA", "FFB" or "CSV") + + m = OpenMatrix(path+"\\"+filename, "True") + mc = CreateMatrixCurrency(m,corename,,,) + rows = GetMatrixRowLabels(mc) + ExportMatrix(mc, rows, "Rows", outputtype, path+"\\"+filenameout, ) + return(1) +EndMacro + +Macro "Matrix Size" (path, filename, corename) + //gets the size (number of zones) in the matrix - useful for sr11 vs sr12 and for split zones + m = OpenMatrix(path+"\\"+filename, "True") + base_indicies = GetMatrixBaseIndex(m) + mc = CreateMatrixCurrency(m, corename, base_indicies[1], base_indicies[2], ) + v = GetMatrixVector(mc, {{"Marginal", "Row Count"}}) + vcount = VectorStatistic(v, "Count", ) + return(vcount) +EndMacro + +Macro "Create Matrix" (path, filename, label, corenames, zone) + Opts = null + Opts.[File Name] = (path+"\\"+filename) + Opts.Label = label + Opts.Type = "Float" + Opts.Tables = corenames + Opts.[Column Major] = "No" + Opts.[File Based] = "Yes" + Opts.Compression = 0 + m = CreateMatrixFromScratch(label, zone, zone, Opts) + return(1) +EndMacro + +Macro "Aggregate Matrices" (path, xref, xrefcol1, xrefcol2, mtx, corenm, aggmtx) + // Aggregate Matrix Options + m = OpenMatrix(path+"\\"+mtx, "True") + base_indicies = GetMatrixBaseIndex(m) + Opts = null + Opts.Input.[Matrix Currency] = {path+"\\"+mtx, corenm, base_indicies[1], base_indicies[2]} + Opts.Input.[Aggregation View] = {xref, "xref"} + Opts.Global.[Row Names] = {"xref."+xrefcol1, "xref."+xrefcol2} + Opts.Global.[Column Names] = {"xref."+xrefcol1, "xref."+xrefcol2} + Opts.Output.[Aggregated Matrix].Label = "AggMtx"+"_"+corenm + Opts.Output.[Aggregated Matrix].Compression = 1 + Opts.Output.[Aggregated Matrix].[File Name] = path+"\\"+aggmtx + + ok = RunMacro("TCB Run Operation", 1, "Aggregate Matrix", Opts, ) + return(ok) +EndMacro + +Macro "Go GetMatrixCoreNames" (path, matrix) + m = OpenMatrix(path+"\\"+matrix, ) + core_names=GetMatrixCoreNames(m) + return(core_names) +EndMacro + +Macro "Get SL Query # from QRY" (path) + + selinkqry_file="\\selectlink_query.qry" + fptr_from = OpenFile(path + selinkqry_file, "r") + qry_array = readarray(fptr_from) + // query = 0 + query_list = null + + for j = 1 to qry_array.length do + line = qry_array[j] + p1 = Position(line, "\") + p2 = Position(line, "\<\/name\>") + if p1 > 0 & p2 > 0 then do + start = p1 + 6 + name_len = p2 - start + name = Substring(line, start, name_len) + query_list = query_list + {name} + end + // if Position(qry_array[j], "\") >0 then query = query + 1 + end + return(query_list) +EndMacro + +Macro "close all" + maps = GetMaps() + if maps <> null then do + for k = 1 to maps[1].length do + SetMapSaveFlag(maps[1][k],"False") + end + end + RunMacro("G30 File Close All") + mtxs = GetMatrices() + if mtxs <> null then do + handles = mtxs[1] + for k = 1 to handles.length do + handles[k] = null + end + end + views = GetViews() + if views <> null then do + handles = views[1] + for k = 1 to handles.length do + handles[k] = null + end + end +EndMacro + +Macro "CloseViews" + vws = GetViewNames() + for i = 1 to vws.length do + CloseView(vws[i]) + end +EndMacro + +// returns a nicely formatted day and time +Macro "date and time" + date_arr = ParseString(GetDateAndTime(), " ") + day = date_arr[1] + mth = date_arr[2] + num = date_arr[3] + time = Left(date_arr[4], StringLength(date_arr[4])-3) + year = SubString(date_arr[5], 1, 4) + today = mth + "/" + num + "/" + year + " " + time + //showmessage(today) + Return(today) +EndMacro + +Macro "SDdeletefile"(arr) + file=arr[1] + dif2=GetDirectoryInfo(file,"file") + if dif2.length>0 then deletefile(file) + ok=1 + quit: + return(ok) +EndMacro + +Macro "SDcopyfile"(arr) + file1=arr[1] + file2=arr[2] + dif2=GetDirectoryInfo(file2,"file") + if dif2.length>0 then deletefile(file2) + dif2=GetDirectoryInfo(file1,"file") + if dif2.length>0 then copyfile(file1,file2) + ok=1 + quit: + return(ok) +EndMacro + +Macro "SDrenamefile"(arr) + file1=arr[1] + file2=arr[2] + dif1=GetDirectoryInfo(file2,"file") + if dif1.length>0 then deletefile(file2) + dif2=GetDirectoryInfo(file1,"file") + if dif2.length>0 then RenameFile(file1, file2) + ok=1 + quit: + return(ok) +EndMacro + +Macro "HwycadLog"(arr) + shared path + fprlog=null + log1=arr[1] + log2=arr[2] + dif2=GetDirectoryInfo(path+"\\hwycadx.log","file") + if dif2.length>0 then fprlog=OpenFile(path+"\\hwycadx.log","a") + else fprlog=OpenFile(path+"\\hwycadx.log","w") + mytime=GetDateAndTime() + writeline(fprlog,mytime+", "+log1+", "+log2) + CloseFile(fprlog) + fprlog = null + return() +EndMacro + +Macro "ForecastYearStr" + shared path_study,path + fptr = OpenFile(path+"\\year", "r") + strYear = ReadLine(fptr) + closefile(fptr) + return(strYear) +EndMacro + +Macro "ForecastYearInt" + //usage: myyear=RunMacro("ForecastYearInt") + shared path_study,path + fptr = OpenFile(path+"\\year", "r") + strFyear = ReadLine(fptr) + closefile(fptr) + intFyear=S2I(strFyear) + return(intFyear) +EndMacro + +Macro "DeleteInterimFiles" (path, FileNameArray,RscName,MacroName,FileDescription) + RunMacro("HwycadLog",{RscName+": "+MacroName,"SDdeletefile, "+FileDescription}) + for i = 1 to FileNameArray.length do //delete existing files + ok=RunMacro("SDdeletefile",{path+"\\"+FileNameArray[i]}) if !ok then goto quit + end + quit: + return(ok) +EndMacro + +Macro "FileCheckDelete" (path,filename) + //usage: RunMacro("FileCheckDelete",path,filename) where path and filename are strings + di = GetDirectoryInfo(path+"\\"+filename, "File") + if di.length > 0 then do + ok=RunMacro("SDdeletefile",{path+"\\"+filename}) + return(ok) + end +EndMacro + +// Macro "getpathdirectory" doesn't allow the selected path with different path_study. +Macro "GetPathDirectory" + shared path,path_study,scr + opts={{"Initial Directory", path_study}} + tmp_path=choosedirectory("Choose an alternative directory in the same study area", opts) + strlen=len(tmp_path) + for i = 1 to strlen do + tmp=right(tmp_path,i) + tmpx=left(tmp,1) + if tmpx="\\" then goto endfor + end + endfor: + strlenx=strlen-i + tmppath_study=left(tmp_path,strlenx) + if path_study=tmppath_study then do + path=tmp_path + tmp_flag=0 + for i=1 to scr.length do + if scr[i]=path then do + tmp_flag=1 + i=scr.length+1 + end + else i=i+1 + end + if tmp_flag=0 then do + tmp = CopyArray(scr) + tmp = tmp + {tmp_path} + scr = CopyArray(tmp) + end + //showmessage("write description of the alternative in the head file") + //x=RunProgram("notepad "+path+"\\head",) + mytime=GetDateAndTime() + fptr=openfile(path+"\\tplog","a") + WriteLine(fptr, mytime) + closefile(fptr) + //showmessage("type in the reason why you are doing the model run in tplog") + //x=RunProgram("notepad "+path+"\\tplog",) + end + else do + path=null + msg1="The alternative directory selected is invalid because it has different study area! " + msg2="Please select again within the same study area " + msg3=" or use the Browse button to select a different study area." + showMessage(msg1+msg2+path_study+msg3) + end +EndMacro +/*********************************************************************************************************************************** +* +* Run Program +* Runs the program for a set of control files +* +***********************************************************************************************************************************/ + +Macro "Run Program" (scenarioDirectory, executableString, controlString) + + + //drive letter + path = SplitPath(scenarioDirectory) + + //open the batch file to run + fileString = scenarioDirectory+"\\programs\\source.bat" + ptr = OpenFile(fileString, "w") + WriteLine(ptr,path[1]) + WriteLine(ptr,"cd "+scenarioDirectory ) + + runString = "call "+executableString + " " + controlString + WriteLine(ptr,runString) + + //write the return code check + failString = "IF NOT ERRORLEVEL = 0 ECHO "+controlString+" > failed.txt" + WriteLine(ptr,failString) + + CloseFile(ptr) + status = RunProgram(fileString, {{"Minimize", "True"}}) + + info = GetFileInfo(scenarioDirectory+"\\failed.txt") + if(info != null) then do + ret_value=0 + goto quit + end + + Return(1) + quit: + Return( RunMacro("TCB Closing", ret_value, True ) ) +EndMacro + diff --git a/sandag_abm/src/main/gisdk/TC2OMX.rsc b/sandag_abm/src/main/gisdk/TC2OMX.rsc new file mode 100644 index 0000000..09fa7cb --- /dev/null +++ b/sandag_abm/src/main/gisdk/TC2OMX.rsc @@ -0,0 +1,69 @@ +Macro "TC to OMX" + + p="T:\\ABM\\ActivitySim\\SANDAG_ActivitySim\\data\\skims" + + files = GetDirectoryInfo(RunMacro("FormPath",{p,"*"}),"All") + for i = 1 to files.length do + f = RunMacro("FormPath",{p,files[i][1]}) + subs = ParseString(f,".", {{"Include Empty",True}}) + if files[i][2] = "file" then do + if subs[2]="mtx" then do + RunMacro("ExportMatrix",subs[1]+".mtx") + end + end + end +EndMacro + +Macro "ExportMatrix" (matrix) + subs = ParseString(matrix,".", {{"Include Empty",True}}) + m = OpenMatrix(matrix, "True") + mc = CreateMatrixCurrency(m,,,,) + CopyMatrix(mc, { + {"File Name", subs[1]+".omx"}, + {"OMX", "True"} + } + ) +EndMacro + +Macro "FormPath" (path_elements) + if TypeOf(path_elements) <> "array" then do + ShowMessage("Must form a path out of a list of elements, not: " + TypeOf(path_elements)) + ShowMessage(2) + end + //path_elements is an array of elements + path = "" + for i = 1 to path_elements.length do + //change / to \ + p = RunMacro("NormalizePath",path_elements[i]) + if Right(p,1) = "\\" then do + if Len(p) > 1 then do + p = Substring(p,1,Len(p)-1) + end + else do + p = "" + end + end + if Left(p,1) = "\\" then do + if Len(p) > 1 then do + p = Substring(p,2,Len(p)) + end + else do + p = "" + end + end + if path = "" then do + path = p + end + else do + path = path + "\\" + p + end + end + return(path) +EndMacro + +Macro "NormalizePath" (path) + if Len(path) > 1 and path[2] = ":" then do + path = Lower(path[1]) + Right(path,Len(path)-1) + end + return(Substitute(path,"/","\\",)) +EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/TruckModel.rsc b/sandag_abm/src/main/gisdk/TruckModel.rsc new file mode 100644 index 0000000..05cda11 --- /dev/null +++ b/sandag_abm/src/main/gisdk/TruckModel.rsc @@ -0,0 +1,1504 @@ +/********************************************************************************************************** +Runs truck model +About: + Script to run the SANDAG Truck Model in TransCAD + Study Area: County of San Diego, California + TransCAD version 4.8 Build 545 + Author: Parsons Brinckerhoff (R. Moeckel, S. Gupta, C. Frazier) + Developed: June to December 2008 + +Modifications: + 1) Added truck toll diversion model to go from three truck types + to six truck types ({lhd/mhd/hhd} * {toll/non-toll}) + Ben Stabler, stabler@pbworld.com, 12/02/10 + + 2) Modified to integrate with SANDAG ABM + Amar Sarvepalli, sarvepalli@pbworld.com, 09/06/12 + + 3) Modified to remove truck skimming; added truck skims to hwyskim_vot.rsc + JEF joel.freedman@rsginc.com, 5/10/2015 + +Steps: + 1) Generates standard truck trip and special (military) truck trips + 2) Gets regional truck trips, IE trips, EI trips and EE trips and balances truck trips + 3) Distributes truck trips with congested skims and splits by time of day + 4) Applies truck toll diversion model with free-flow toll and non-toll skims + + Note: truck trip generation and free-flow skims are run only for the first iteration + + + +**********************************************************************************************************/ +Macro "truck model"(properties, iteration) + shared path, inputDir, outputDir, inputTruckDir + + // read properties from sandag_abm.properties in /conf folder + properties = "\\conf\\sandag_abm.properties" + startFromIteration = s2i(RunMacro("read properties",properties,"RunModel.startFromIteration", "S")) + + // Generate trips and free-flow truck skims for the first iteration + if (iteration = startFromIteration) then do + // Generate daily truck trips + RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","truck-tripgen"}) + ok = RunMacro("truck-tripgen",properties) + if !ok then goto quit + end + + // Distribute daily truck trips and split them by time period + RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","trkDistribution,(properties)"}) + ok = RunMacro("trkDistribution",properties) + if !ok then goto quit + + // Apply toll-diversion model + RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","trk toll diversion model"}) + ok = RunMacro("trk toll diversion model") + if !ok then goto quit + + run_ok = 1 + RunMacro("close all") + Return(run_ok) + + quit: + RunMacro("close all") + Return(ok) +EndMacro + + + +/********************************************************************************************************** +Generates daily truck trips (standard and special generations) + +Inputs: + sandag.properties + +Outputs: + output\gmTruckDataBalanced.bin + output\regionalEEtrips.csv + +**********************************************************************************************************/ +Macro "truck-tripgen"(properties) + shared path, inputDir, outputDir, mxzone + dim arrInterimYear[3] + + strFyear = RunMacro("read properties",properties,"truck.FFyear","S") + intFyear = StringToInt(strFyear) + arrInterimYear=RunMacro("InterimYearCheck",properties,intFyear) + + RunMacro("trkStdTripGen",strFyear,intFyear,arrInterimYear,properties) + RunMacro("trkSpecialGen",strFyear,intFyear,arrInterimYear) + RunMacro("trkBalance",strFyear,intFyear,arrInterimYear) + run_ok=1 + + exit: + RunMacro("close all") + Return(run_ok) +EndMacro + + + + +/********************************************************************************************************** +Produces standard truck trips + +Inputs: + input\mgra13_based_input2010.csv + input\TruckTripRates.csv + output\hhByTaz.csv.csv + output\empByTaz.csv.csv + + (optional, used for interpolation) + input\hhByTaz.csv + input\hhByTaz.csv + input\empByTaz.csv + input\empByTaz.csv + + (optional, used only for landuse override) + input\lu.csv + output\EmpDistLUovrAgg.csv + output\hhluagg.csv + +Outputs: + output\gmTruckDataII.csv + +**********************************************************************************************************/ +Macro "trkStdTripGen" (strFyear,intFyear,arrInterimYear,properties) + shared path, inputDir, outputDir, mxzone + booInterimYear = arrInterimYear[1] + + // Creates household and employment data by taz from mgra data + RunMacro("Create hh and emp by taz") + + //-------------------------------------------------- + //This section checks available data and interpolates if doesn't exist + //-------------------------------------------------- + // Do interpolate = True + if booInterimYear = 2 then do + prevYear = arrInterimYear[2] + nextYear = arrInterimYear[3] + end + + // Override landuse data if option is "True" + check_luOverride = RunMacro("read properties",properties,"truck.luOverRide", "S") + if check_luOverride = "True" then do + RunMacro("EmploymentLUOverride") + RunMacro("EmploymentDistLUOverride") + RunMacro("HouseholdLUOverride") + end + + + // If data is not available then interpolate from closest available years + if booInterimYear = 2 then do + // Copy prev and next year data files to output directory + ok=RunMacro("SDcopyfile",{inputDir+"\\hhByTaz"+I2S(prevYear)+".csv",outputDir+"\\hhByTaz_prev.csv"}) + ok=RunMacro("SDcopyfile",{inputDir+"\\hhByTaz"+I2S(nextYear)+".csv",outputDir+"\\hhByTaz_next.csv"}) + ok=RunMacro("SDcopyfile",{inputDir+"\\empByTaz"+I2S(prevYear)+".csv",outputDir+"\\empByTaz_prev.csv"}) + ok=RunMacro("SDcopyfile",{inputDir+"\\empByTaz"+I2S(nextYear)+".csv",outputDir+"\\empByTaz_next.csv"}) + + // Interpolate data from prev and next years + ok=RunMacro("Interpolate",{"hhByTaz_prev.csv","hhByTaz_next.csv","hhByTaz.csv",intFyear,prevYear,nextYear}) + ok=RunMacro("Interpolate",{"empByTaz_prev.csv","empByTaz_next.csv","empByTaz.csv",intFyear,prevYear,nextYear}) + + // Delete prev and next year data files + ok=RunMacro("SDdeletefile",{outputDir+"\\hhByTaz_prev.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\hhByTaz_next.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\empByTaz_prev.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\empByTaz_next.csv"}) + end + + // join data and parameter tables + empView = Opentable("Employment", "CSV", {outputDir+"\\empByTaz.csv"}) + hhView = Opentable("HouseHolds", "CSV", {outputDir+"\\hhByTaz.csv"}) + + + //-------------------------------------------------- + // This section overrides LU for employment and households + //-------------------------------------------------- + if check_luOverride = "True" then do + + //LU override file for Employment + empAggDistLUovr_filecsv = outputDir+"\\EmpDistLUovrAgg.csv" + di = GetDirectoryInfo(empAggDistLUovr_filecsv, "File") + + // Export employment by TAZ to binary file and get zones + ExportView("Employment|", "FFB", outputDir+"\\empByTaz.BIN", null, ) + vwEmpBin = Opentable("EmploymentByZoneBin", "FFB", {outputDir+"\\empByTaz.BIN"}) + vwEmpDistLUovrAgg = Opentable("EmploymentDistributionOverride", "CSV", {empAggDistLUovr_filecsv}) + vectZonerecords = GetRecordCount(vwEmpDistLUovrAgg, null) + vectZone = GetDataVectors(vwEmpDistLUovrAgg+"|",{"zone"},) + + // Convert real to string zones + for i = 1 to vectZonerecords do + strZone = RealToString(vectZone[1][i]) + rh = LocateRecord("EmploymentByZoneBin|","TAZ",{strZone},{{"Exact", "True"}}) + x = GetView() + DeleteRecord("EmploymentByZoneBin", rh) + end + + // Get data from all fields in EmpDistLUovrAgg file + fldsEmpDistLUovrAgg = GetFields(vwEmpDistLUovrAgg, "All") + vectEMPovr = GetDataVectors(vwEmpDistLUovrAgg+"|",fldsEmpDistLUovrAgg[1],) + + // Write fields to employment by TAZ + for i = 1 to vectZonerecords do + rh = AddRecord(vwEmpBin, { + {"TAZ", R2I(vectEMPovr[1][i])}, + {"First TAZ", R2I(vectEMPovr[2][i])}, + {fldsEmpDistLUovrAgg[1][3], vectEMPovr[3][i]}, + {fldsEmpDistLUovrAgg[1][4], vectEMPovr[4][i]}, + {fldsEmpDistLUovrAgg[1][5], vectEMPovr[5][i]}, + {fldsEmpDistLUovrAgg[1][6], vectEMPovr[6][i]}, + {fldsEmpDistLUovrAgg[1][7], vectEMPovr[7][i]}, + {fldsEmpDistLUovrAgg[1][8], vectEMPovr[8][i]}, + {fldsEmpDistLUovrAgg[1][9], vectEMPovr[9][i]}, + {fldsEmpDistLUovrAgg[1][10], vectEMPovr[10][i]}, + {fldsEmpDistLUovrAgg[1][11], vectEMPovr[11][i]} + }) + end + + // LU override file for Households + HHLUagg_filecsv = outputDir+"\\hhluagg.csv" + di = GetDirectoryInfo(HHLUagg_filecsv, "File") + + // Export households by TAZ to binary file and get zones + ExportView(hhView+"|", "FFB", outputDir+"\\hhByTaz.BIN", null, ) + vwHHBin = Opentable("HouseholdsByZoneBin", "FFB", {outputDir+"\\HHByTaz.BIN"}) + vwHHLUagg = Opentable("HouseholdsOverride", "CSV", {HHLUagg_filecsv}) + vectHHZoneRecords = GetRecordCount(vwHHLUagg, null) + vectHHZone = GetDataVectors(vwHHLUagg+"|",{"TAZ"},) + + // Convert real to string zones + for i = 1 to vectHHZoneRecords do + strZone = RealToString(vectHHZone[1][i]) + rh = LocateRecord("HouseholdsByZoneBin|","TAZ",{strZone},{{"Exact", "True"}}) + DeleteRecord("HouseholdsByZoneBin", rh) + end + + // Get data from all fields in hhluagg file + fldsHHLUAgg = GetFields(vwHHLUagg, "All") + vectHHovr = GetDataVectors(vwHHLUagg+"|",fldsHHLUAgg[1],) + + // Write fields to households by TAZ + for i = 1 to vectHHZoneRecords do + rh = AddRecord(vwHHBin, { + {"TAZ", R2I(vectHHovr[1][i])}, + {"First TAZ", R2I(vectHHovr[2][i])}, + {fldsHHLUAgg[1][3], vectHHovr[3][i]} + }) + end + end + + + //-------------------------------------------------- + // This section applies truck trip production and attraction rates by truck type + //-------------------------------------------------- + // Open truck data file + viewtripRates = Opentable("TruckTripRates", "CSV", {inputDir+"\\TruckTripRates.csv"}) + + // Join all data + jv1 = Joinviews("JV1", hhView+".ZONE", empView+".ZONE",) + jv9 = Joinviews("JV9", jv1+"."+hhView+".TruckRegionType", viewtripRates+".RegionType",) + Setview(jv9) + + // Compute lhd truck productions "AGREMPN","CONEMPN","RETEMPN","GOVEMPN","MANEMPN","UTLEMPN","WHSEMPN","OTHEMPN"} + lhd_p = CreateExpression(jv9, "LHD_ProductionsTemp", "Nz(emp_agmin + emp_cons) * [TG_L_Ag/Min/Constr] + Nz(emp_retrade) * TG_L_Retail + Nz(emp_gov) * TG_L_Government + Nz(emp_mfg) * TG_L_Manufacturing + Nz(emp_twu) * [TG_L_Transp/Utilities]", ) + lhd_p = CreateExpression(jv9, "LHD_Productions", "Nz(LHD_ProductionsTemp) + Nz(emp_whtrade) * TG_L_Wholesale + Nz(emp_other) * TG_L_Other + Nz(HH) * TG_L_Households", ) + + // Compute lhd truck attractions + lhd_a = CreateExpression(jv9, "LHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_L_Ag/Min/Constr] + emp_retrade * TA_L_Retail + emp_gov * TA_L_Government + emp_mfg * TA_L_Manufacturing + emp_twu * [TA_L_Transp/Utilities] + emp_whtrade * TA_L_Wholesale", ) + lhd_a = CreateExpression(jv9, "LHD_Attractions", "Nz(LHD_AttractionsTemp) + Nz(emp_other) * TA_L_Other + Nz(HH) * TA_L_Households", ) + + // Compute mhd truck productions + mhd_p = CreateExpression(jv9, "MHD_ProductionsTemp", "(emp_agmin + emp_cons) * [TG_M_Ag/Min/Constr] + emp_retrade * TG_M_Retail + emp_gov * TG_M_Government + emp_mfg * TG_M_Manufacturing + emp_twu * [TG_M_Transp/Utilities] + emp_whtrade * TG_M_Wholesale ", ) + mhd_p = CreateExpression(jv9, "MHD_Productions", "Nz(MHD_ProductionsTemp) + Nz(emp_other) * TG_M_Other + Nz(HH) * TG_M_Households", ) + + // Compute mhd truck attractions + mhd_a = CreateExpression(jv9, "MHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_M_Ag/Min/Constr] + emp_retrade * TA_M_Retail + emp_gov * TA_M_Government + emp_mfg * TA_M_Manufacturing + emp_twu * [TA_M_Transp/Utilities] + emp_whtrade * TA_M_Wholesale", ) + mhd_a = CreateExpression(jv9, "MHD_Attractions", "Nz(MHD_AttractionsTemp) + Nz(emp_other) * TA_M_Other + Nz(HH) * TA_M_Households", ) + + // Compute hhd truck productions + hhd_p = CreateExpression(jv9, "HHD_ProductionsTemp", "(emp_agmin + emp_cons) * [TG_H_Ag/Min/Constr] + emp_retrade * TG_H_Retail + emp_gov * TG_H_Government + emp_mfg * TG_H_Manufacturing + emp_twu * [TG_H_Transp/Utilities] + emp_whtrade * TG_H_Wholesale ", ) + hhd_p = CreateExpression(jv9, "HHD_Productions", "Nz(HHD_ProductionsTemp) + Nz(emp_other) * TG_H_Other + Nz(HH) * TG_H_Households", ) + + // Compute hhd truck attractions + hhd_a = CreateExpression(jv9, "HHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_H_Ag/Min/Constr] + emp_retrade * TA_H_Retail + emp_gov * TA_H_Government + emp_mfg * TA_H_Manufacturing + emp_twu * [TA_H_Transp/Utilities] + emp_whtrade * TA_H_Wholesale", ) + hhd_a = CreateExpression(jv9, "HHD_Attractions", "Nz(HHD_AttractionsTemp) + Nz(emp_other) * TA_H_Other + Nz(HH) * TA_H_Households", ) + + // Export the productions and attractions to csv file + RunMacro("HwycadLog",{"TruckModel.rsc: trkStdGen","ExportView P&A"}) + ExportView(jv9+"|", "CSV", outputDir+"\\gmTruckDataII.csv", {hhView+".ZONE", "LHD_Productions", "LHD_Attractions", "MHD_Productions", "MHD_Attractions", "HHD_Productions", "HHD_Attractions"}, {{"CSV Header"}}) + + + // Close all and delete temp files + RunMacro("close all") + ok=RunMacro("SDdeletefile",{outputDir+"\\hhdata.csv"}) + if!ok then goto exit + ok=RunMacro("SDdeletefile",{outputDir+"\\emp.csv"}) + if!ok then goto exit + + run_ok=1 + exit: + RunMacro("close all") + Return(run_ok) +EndMacro + + + +/********************************************************************************************************** +Creates households by taz and employment by taz files to use in the truck trip generation model + +Inputs: + sandag.properties + input\mgra13_based_input2012.csv + +Outputs: + output\empByTaz.csv + output\hhByTaz.csv + +**********************************************************************************************************/ +Macro "Create hh and emp by taz" + shared path, inputDir, outputDir, mxzone , scenarioYear + mgraDataFile = "mgra13_based_input"+scenarioYear+".csv" + empbytaz = "empByTaz.csv" + hhbytaz = "hhByTaz.csv" + + RunMacro("SDcopyfile",{inputDir+"\\"+mgraDataFile,outputDir+"\\"+mgraDataFile}) + mgraView = OpenTable("MGRA View", "CSV", {outputDir+"\\"+mgraDataFile}, {{"Shared", "True"}}) + + // Get data fields into vectors + mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_total = GetDataVector(mgraView+"|", "emp_total", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + + + // Combine employment fields that match to the truck trip rate classification + TOTEMP = emp_total // Total employment + emp_agmin = emp_ag // Agriculture + Mining + emp_cons = emp_const_bldg_prod + emp_const_bldg_office // Construction + emp_retrade = emp_retail + emp_personal_svcs_retail // Retail + emp_gov = emp_state_local_gov_ent + emp_state_local_gov_blue + emp_state_local_gov_white + emp_fed_non_mil + emp_fed_mil // Government + emp_mfg = emp_mfg_prod + emp_mfg_office // Manufacturing + emp_twu = emp_trans + emp_utilities_office + emp_utilities_prod // Transportation + Utilities + emp_whtrade = emp_whsle_whs // Wholesale + emp_other = TOTEMP - (emp_agmin +emp_cons+ emp_retrade + emp_gov + emp_mfg + emp_twu + emp_whtrade) // Other + + + // Add fields employment by taz + strct = GetTableStructure(mgraView) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"emp_agmin" , "Float", 8, 4, "True", , , , , , , null}} + strct = strct + {{"emp_cons" , "Float", 8, 4, "True", , , , , , , null}} + strct = strct + {{"emp_retrade" , "Float", 8, 0, "True", , , , , , , null}} + strct = strct + {{"emp_gov" , "Float", 8, 0, "True", , , , , , , null}} + strct = strct + {{"emp_mfg" , "Float", 8, 0, "True", , , , , , , null}} + strct = strct + {{"emp_twu" , "Float", 8, 0, "True", , , , , , , null}} + strct = strct + {{"emp_whtrade" , "Float", 8, 0, "True", , , , , , , null}} + strct = strct + {{"emp_other" , "Float", 8, 4, "True", , , , , , , null}} + ModifyTable(mgraView, strct) + + ExportView(mgraView+"|","FFB",outputDir+"\\temp.bin",{"MGRA","TAZ","HH","TruckRegionType","emp_agmin","emp_cons","emp_retrade","emp_gov","emp_mfg","emp_twu","emp_whtrade","emp_other"},) + MgraTazView = OpenTable("MGRA View", "FFB", {outputDir+"\\temp.bin"},) + + // Set data into new fields + SetDataVectors(MgraTazView+"|",{{"emp_agmin" ,emp_agmin }, + {"emp_cons" ,emp_cons }, + {"emp_retrade",emp_retrade}, + {"emp_gov" ,emp_gov }, + {"emp_mfg" ,emp_mfg }, + {"emp_twu" ,emp_twu }, + {"emp_whtrade",emp_whtrade}, + {"emp_other" ,emp_other }},) + + // Aggregate employment fields by taz + emp = AggregateTable("employment", MgraTazView+"|","FFB", outputDir+"\\emp_temp.bin","TAZ", { + {"TruckRegionType","avg","TruckRegionType"}, + {"emp_agmin" ,"sum","emp_agmin" }, + {"emp_cons" ,"sum","emp_cons" }, + {"emp_retrade" ,"sum","emp_retrade" }, + {"emp_gov" ,"sum","emp_gov" }, + {"emp_mfg" ,"sum","emp_mfg" }, + {"emp_twu" ,"sum","emp_twu" }, + {"emp_whtrade" ,"sum","emp_whtrade" }, + {"emp_other" ,"sum","emp_other" } + },null) + + emp_view = OpenTable("emp_view", "FFB", {outputDir+"\\emp_temp.bin"},) + RenameField(emp_view+".Avg TruckRegionType", "TruckRegionType") + + // Create a temp file with all zones (internal + external zones) from the highway network file + db_file = outputDir+"\\hwy.dbd" + {node_lyr,} = RunMacro("TCB Add DB Layers", db_file,,) + SetLayer(node_lyr) + n= SelectByQuery("Zones", "Several", "Select * where ID <= "+ String(mxzone),) + zones = GetDataVector(node_lyr+"|Zones", "ID",{{"Sort Order", {{"ID", "Ascending"}}}}) + + // Create a temp file with all zones + all_vw = CreateTable("allzones", outputDir+"\\temp_zones.bin", "FFB",{ + {"ZONE", "Integer", 8, null, "Yes"}}) + SetView(all_vw) + for i = 1 to zones.length do + rh = AddRecord(all_vw, {{"ZONE", zones[i]}}) + end + + join_vw = JoinViews("joined_view1", all_vw+".ZONE",emp_view+".TAZ",) + ExportView(join_vw+"|","CSV",outputDir+"\\"+empbytaz,,) + CloseView(join_vw) + + + // Aggregate households by taz + hh = AggregateTable("households", MgraTazView+"|","FFB", outputDir+"\\hh_temp.bin","TAZ", { + {"TruckRegionType","avg","TruckRegionType"}, + {"HH" ,"sum","HH"} + },null) + + hh_view = OpenTable("hh_temp", "FFB", {outputDir+"\\hh_temp.bin"},) + RenameField(hh_view+".Avg TruckRegionType", "TruckRegionType") + join_vw = JoinViews("joined_view1", all_vw+".ZONE",hh_view+".TAZ",) + ExportView(join_vw+"|","CSV",outputDir+"\\"+hhbytaz,,) + + RunMacro("close all") + + DeleteFile(outputDir+"\\temp_zones.bin") + DeleteFile(outputDir+"\\emp_temp.bin") + DeleteFile(outputDir+"\\hh_temp.bin") +EndMacro + + + +/********************************************************************************************************** +Creates household override file from a land use file LU.CSV + +Inputs: + input\lu.csv + +Outputs: + output\hhlu.csv + output\hhluagg.csv + +**********************************************************************************************************/ +Macro "HouseholdLUOverride" + // Creates Household Override file from a Land Use based LU.csv + shared path, inputDir, outputDir + RunMacro("TCB Init") + + // Copy LU.csv, Open Copy, and Rename Fields - no header line in lu.csv + ok=RunMacro("SDcopyfile",{inputDir+"\\lu.csv",outputDir+"\\hhlu.csv"}) if!ok then goto quit + vwHHLUovr = Opentable("HHLUOverride", "CSV",{outputDir+"\\hhlu.csv"}) + RenameField(vwHHLUovr+".FIELD_1", "TAZ") + RenameField(vwHHLUovr+".FIELD_2", "RateType") + RenameField(vwHHLUovr+".FIELD_3", "LUCode") + RenameField(vwHHLUovr+".FIELD_4", "HH") + Setview(vwHHLUovr) + + // Select HH's From LU.csv + // Ratetype=DU=1 + qry1 = "Select * where RateType = 1" + DUqry = SelectByQuery("HHSelection", "Several", qry1,) + + // Aggregate HH's to TAZ Level (might have SF & MF as separate codes) + RunMacro("HwycadLog",{"TruckModel.rsc: HouseholdLUOverride","AggregateTable"}) + aggtable = AggregateTable("AggHHLUOvr","HHLUOverride|HHSelection", "CSV", outputDir+"\\hhluagg.csv", "TAZ", { + {"TAZ","dominant"}, + {"HH","sum", } + }, {"Missing as zero"}) + + done: + RunMacro("close all") + Return( RunMacro("TCB Closing", 1, "FALSE" ) ) + + quit: + RunMacro("close all") + Return( RunMacro("TCB Closing", 0, "FALSE" ) ) +EndMacro + + +/********************************************************************************************************** +Creates Employment Override File from a land use file LU.CSV + +Inputs: + input\lu.csv + input\emplbylu.csv + input\Zone_sphere.csv + input\emp_lu_ksf.csv + input\emp_lu_rm.csv + input\emp_lu_site.csv + +Outputs: + output\EmpLUovr.bin" + output\EmpLUovr.csv" +**********************************************************************************************************/ +Macro "EmploymentLUOverride" + shared path,inputDir, outputDir + + // Remove dcc & dcb files if already exist + ok=RunMacro("SDdeletefile",{inputDir+"\\emplbylu.dcc"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{inputDir+"\\Zone_sphere.dcc"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{inputDir+"\\emp_lu_ksf.dcc"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{inputDir+"\\emp_lu_site.dcc"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{inputDir+"\\lu.dcc"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{outputDir+"\\EmpLUovr.dcb"}) + if!ok then goto quit + ok=RunMacro("SDdeletefile",{outputDir+"\\EmpLUovr.dcc"}) + if!ok then goto quit + + // Open data files and LU.CSV + vwEmpbyLU = Opentable("EmploymentByLU", "CSV", {inputDir+"\\emplbylu.csv"}) + vwZoneSphere = Opentable("XREF_ZoneSphere","CSV", {inputDir+"\\Zone_sphere.csv"}) + vwAcreToKSF = Opentable("Conv_Acre_KSF", "CSV", {inputDir+"\\emp_lu_ksf.csv"}) + vwAcreToRM = Opentable("Conv_Acre_RM", "CSV", {inputDir+"\\emp_lu_rm.csv"}) + vwEmpSite = Opentable("SiteEmploymentbySphere", "CSV", {inputDir+"\\emp_lu_site.csv"}) + vwLUovr = Opentable("LUOverride", "CSV", {inputDir+"\\lu.csv"}) + + // Rename fields for LU.CSV - no header line in file + RenameField(vwLUovr+".FIELD_1", "TAZ") + RenameField(vwLUovr+".FIELD_2", "RateType") + RenameField(vwLUovr+".FIELD_3", "LUCode") + RenameField(vwLUovr+".FIELD_4", "Amt") + + // Join data files to LU.CSV + Setview(vwLUovr) + jvwLUovrSphere = Joinviews("jvwLUovrSphere", vwLUovr+".TAZ", vwZoneSphere+".Zone",) + jvwLUovrSphereEmp = Joinviews("jvwLUovrSphereEmp", jvwLUovrSphere+".LUCode", vwEmpbyLU+".LU",) + jvwLUovrSphereEmpKSF = Joinviews("jvwLUovrSphereEmpKSF", jvwLUovrSphereEmp+".LUCode", vwAcreToKSF+".LU",) + jvwLUovrSphereEmpKSFSite = Joinviews("jvwLUovrSphereEmpKSFSite", jvwLUovrSphereEmpKSF+".LUCode", vwEmpSite+".LU",) + jvwLUovrSphereEmpKSFSiteRM = Joinviews("jvwLUovrSphereEmpKSFSiteRM", jvwLUovrSphereEmpKSFSite+".LUCode", vwAcreToRM+".LU",) + + // Set Output Files + empLUovr_file = outputDir+"\\EmpLUovr.BIN" + empLUovr_filecsv = outputDir+"\\EmpLUovr.csv" + + // Export Joined View and Add Employment Fields + Setview(jvwLUovrSphereEmpKSFSiteRM) + ExportView(jvwLUovrSphereEmpKSFSiteRM+"|", "FFB", empLUovr_file, null, { + {"Additional Fields",{ {"empbyacres", "Real", 16, 4, },{"empdirect", "Real", 16, 4, },{"empbysite", "Real", 16, 4, },{"empbyksf", "Real", 16, 4, },{"empbyrm", "Real", 16, 4, },{"emp", "Real", 16, 4, }} }, + } ) + + // Apply Acre Based Employment Rates + // RateType=2=Acre + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbyacres"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere=1404 and RateType=2) then nz(sph1404acre*Amt) else if (Sphere=1441 and RateType=2) then nz(sph1441acre*Amt) else if (Sphere>=1900 and RateType=2) then nz(sph1900acre*Amt) else if RateType=2 then nz(sphOtheracre*Amt)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Add Directly Entered Employment + // RateType=3=Employee + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empdirect"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if RateType=3 then nz(Amt)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Apply Site Based Employment Rates + // RateType=4=Site + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbysite"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere=1404 and RateType=4) then nz(sph1404site*Amt) else if (Sphere=1441 and RateType=4) then nz(sph1441site*Amt) else if (Sphere>=1900 and RateType=4) then nz(sph1900site*Amt) else if RateType=4 then nz(sphOthersite*Amt)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Convert KSF to Acres and Apply Acre Based Employment Rates + // RateType=6=KSF + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbyksf"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere=1404 and RateType=6) then nz(sph1404acre*Amt*ksf2acre) else if (Sphere=1441 and RateType=6) then nz(sph1441acre*Amt*ksf2acre)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbyksf"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere>=1900 and RateType=6) then nz(sph1900acre*Amt*ksf2acre) else if RateType=6 then nz(sphOtheracre*Amt*ksf2acre)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Convert KSF to Acres and Apply Acre Based Employment Rates + // RateType=7=Hotel Room + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbyRM"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere=1404 and RateType=7) then nz(sph1404acre*Amt*rm2acre) else if (Sphere=1441 and RateType=7) then nz(sph1441acre*Amt*rm2acre)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"empbyRM"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if (Sphere>=1900 and RateType=7) then nz(sph1900acre*Amt*rm2acre) else if RateType=7 then nz(sphOtheracre*Amt*rm2acre)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Combine Acre, KSF, and Site Employment into EMP Field + Opts = null + Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} + Opts.Global.Fields = {"emp"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"nz(empbyacres)+nz(empdirect)+nz(empbysite)+nz(empbyksf)+nz(empbyrm)"} + ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ret_value then goto quit + + // Export Employment Override File + RunMacro("HwycadLog",{"TruckModel.rsc: EmploymentLUOverride","ExportView"}) + CloseView(jvwLUovrSphereEmpKSFSiteRM) + jvwLUovrSphereEmpKSFSiteRM=opentable("jvwLUovrSphereEmpKSFSiteRM", "FFB", {empLUovr_file,}) + Setview(jvwLUovrSphereEmpKSFSiteRM) + ExportView(jvwLUovrSphereEmpKSFSiteRM+"|", "CSV", empLUovr_filecsv, + {"TAZ","RateType","LUCode","Amt","emp"}, + { + {"CSV Header", "False"}, + {"CSV Drop Quotes", "True"} + } ) + + done: + RunMacro("close all") + Return( RunMacro("TCB Closing", 1, "FALSE" ) ) + + quit: + RunMacro("close all") + Return( RunMacro("TCB Closing", 0, "FALSE" ) ) +EndMacro + + +/********************************************************************************************************** +Takes employment override file and creates employment distribution to employment categories + +Inputs: + output\EmpLUovr.csv" + input\empldistbylu.csv + +Outputs: + output\EmpDistLUovrAgg.csv + output\EmpDistLUovr.csv" + +**********************************************************************************************************/ +Macro "EmploymentDistLUOverride" + shared path,inputDir, outputDir + RunMacro("TCB Init") + + // Open Employment Override and Employment Distribution Files + empLUovr_filecsv = outputDir+"\\EmpLUovr.csv" + vwEmpLUovr=opentable("EmploymentLUoverride", "CSV", {empLUovr_filecsv,}) + vwEmpDistbyLU = Opentable("EmploymentDistribution", "CSV", {inputDir+"\\empldistbylu.csv"}) + + // Join Files + Setview(vwEmpLUovr) + jvwEmpLUovrDist = Joinviews("EmploymentOverrideWithDistribution", vwEmpLUovr+".LUCode", vwEmpDistbyLU+".plu",) + + // Set Output Files + empAggDistLUovr_filecsv = outputDir+"\\EmpDistLUovrAgg.csv" + empDistLUovr_filecsv = outputDir+"\\EmpDistLUovr.csv" + + // Calculate Employment Distribution + expZone = CreateExpression(jvwEmpLUovrDist, "zone", "TAZ", ) + expEmp_mil = CreateExpression(jvwEmpLUovrDist, "emp_mil", "Nz(emp*mil)", ) + expEmp_agmin = CreateExpression(jvwEmpLUovrDist, "emp_agmin", "Nz(emp*ag)", ) + expEmp_cons = CreateExpression(jvwEmpLUovrDist, "emp_cons", "Nz(emp*con)", ) + expEmp_mfg = CreateExpression(jvwEmpLUovrDist, "emp_mfg", "Nz(emp*mfg)", ) + expEmp_whtrade = CreateExpression(jvwEmpLUovrDist, "emp_whtrade", "Nz(emp*whtrade)", ) + expEmp_retrade = CreateExpression(jvwEmpLUovrDist, "emp_retrade", "Nz(emp*retrade)", ) + expEmp_twu = CreateExpression(jvwEmpLUovrDist, "emp_twu", "Nz(emp*twu)", ) + expEmp_fre = CreateExpression(jvwEmpLUovrDist, "emp_fre", "Nz(emp*fre)", ) + expEmp_info = CreateExpression(jvwEmpLUovrDist, "emp_info", "Nz(emp*info)", ) + expEmp_pbs = CreateExpression(jvwEmpLUovrDist, "emp_pbs", "Nz(emp*pbs)", ) + expEmp_lh = CreateExpression(jvwEmpLUovrDist, "emp_lh", "Nz(emp*lh)", ) + expEmp_os = CreateExpression(jvwEmpLUovrDist, "emp_os", "Nz(emp*os)", ) + expEmp_edhs = CreateExpression(jvwEmpLUovrDist, "emp_edhs", "Nz(emp*edhs)", ) + expEmp_gov = CreateExpression(jvwEmpLUovrDist, "emp_gov", "Nz(emp*gov)", ) + expEmp_sedw = CreateExpression(jvwEmpLUovrDist, "emp_sedw", "Nz(emp*sedw)", ) + expEmp_civ = CreateExpression(jvwEmpLUovrDist, "emp_civ", "emp_agmin + emp_cons + emp_mfg + emp_whtrade + emp_retrade + emp_twu + emp_fre + emp_info + emp_pbs + emp_lh + emp_os + emp_edhs + emp_gov + emp_sedw", ) + + // Export employment distribution before aggregation + ExportView(jvwEmpLUovrDist+"|", "CSV", empDistLUovr_filecsv, {"zone", "emp", "emp_civ", "emp_mil", "emp_agmin", "emp_cons", "emp_mfg", "emp_whtrade", "emp_retrade", "emp_twu", "emp_fre", "emp_info", "emp_pbs", "emp_lh", "emp_os", "emp_edhs", "emp_gov", "emp_sedw"}, {{"CSV Header", "True"},{"CSV Drop Quotes", "True"}}) + + // Aggregate employment distribution by LU Code by TAZ to employment distribution by TAZ + RunMacro("HwycadLog",{"TruckModel.rsc: EmploymentDistLUOverride","AggregateTable"}) + vwEmpDistLUovr=opentable("DistributedEmploymentOverride", "CSV", {empDistLUovr_filecsv,}) + Setview(vwEmpDistLUovr) + AggAll = CreateSet("AggregatedZones") + SelectAll(AggAll) + sets_list = GetSets(vwEmpDistLUovr) + aggtable = AggregateTable("ZoneAggEmpDistLUOvr","DistributedEmploymentOverride|AggregatedZones", "CSV", empAggDistLUovr_filecsv, "zone", { + {"zone","dominant"}, + {"emp_agmin","sum", }, + {"emp_cons","sum", }, + {"emp_retrade","sum", }, + {"emp_gov","sum", }, + {"emp_mfg","sum", }, + {"emp_twu","sum", }, + {"emp_whtrade","sum", }, + {"emp_os","sum", }, + {"emp_sedw","sum", } + }, {"Missing as zero"}) + + done: + RunMacro("close all") + Return( RunMacro("TCB Closing", 1, "FALSE" ) ) + + quit: + RunMacro("close all") + Return( RunMacro("TCB Closing", 0, "FALSE" ) ) +EndMacro + + +/********************************************************************************************************** +Checks whether data is available for forecast year and if not then interpolates from closet available years + +Inputs: + forecast year + +Outputs: + returns an array {availability of data, previous data year, next data year} +**********************************************************************************************************/ +Macro "InterimYearCheck" (properties,intFyear) + dim arrInterimYear[3] + + // Reads all years for which data is available + DFyear =RunMacro("read properties",properties,"truck.DFyear","S") + + // Lists data year with delimiter "," + arrAllDFyears = ParseString(DFyear, ",") + for i = 1 to arrAllDFyears.length do + intTargetYear = s2i(trim(arrAllDFyears[i])) + if i < arrAllDFyears.length then do + intNextTargetYear = s2i(trim(arrAllDFyears[i+1])) + end + + // Forecast year has data available + if intFyear = intTargetYear then do + // Do interpolate = False + arrInterimYear[1]=1 + end + + // Forecast year has no data + else do + // Check for previous and next closest years + if (intFyear > intTargetYear & intFyear < intNextTargetYear) then do + // Do interpolate = True + arrInterimYear[1] = 2 + // Gets previous closest year + arrInterimYear[2] = intTargetYear + // Gets next closest year + arrInterimYear[3] = intNextTargetYear + end + end + end + + Return(arrInterimYear) +EndMacro + + + +/********************************************************************************************************** +Adds trucks generated by special generators, such as military sites, mail to/from airport, cruise ships, etc + +Inputs: + input\specialGenerators.csv + output\gmTruckDataII.csv + +Outputs: + output\gmTruckDataIISP.csv + +**********************************************************************************************************/ +Macro "trkSpecialGen" (strFyear,intFyear,arrInterimYear) + shared path, inputDir, outputDir + + booInterimYear = arrInterimYear[1] + // 1 = Forecast year, 2 = Interim year and needs interpolation + if booInterimYear = 2 then do + prevYear = arrInterimYear[2] + nextYear = arrInterimYear[3] + end + + // Open truck trips and truck special generators + baseTrucks = Opentable("TruckGeneration", "CSV", {outputDir+"\\gmTruckDataII.csv"}) + specGenerators = Opentable("SpecGenMilit", "CSV",{inputDir+"\\specialGenerators.csv"}) + jv1 = Joinviews("JV1", baseTrucks+".ZONE", specGenerators+".TAZ", ) + Setview(jv1) + + // Forecast year has data available + if booInterimYear = 1 then do + col_name="Y"+strFyear + lhd_a = CreateExpression(jv1, "LHD_Attr", "LHD_Attractions + Nz("+col_name+" * trkAttraction * lhdShare)", ) + lhd_p = CreateExpression(jv1, "LHD_Prod", "LHD_Productions + Nz("+col_name+" * trkProduction * lhdShare)", ) + mhd_a = CreateExpression(jv1, "MHD_Attr", "MHD_Attractions + Nz("+col_name+" * trkAttraction * mhdShare)", ) + mhd_p = CreateExpression(jv1, "MHD_Prod", "MHD_Productions + Nz("+col_name+" * trkProduction * mhdShare)", ) + hhd_a = CreateExpression(jv1, "HHD_Attr", "HHD_Attractions + Nz("+col_name+" * trkAttraction * hhdShare)", ) + hhd_p = CreateExpression(jv1, "HHD_Prod", "HHD_Productions + Nz("+col_name+" * trkProduction * hhdShare)", ) + RunMacro("HwycadLog",{"TruckModel.rsc: trkSpecialGen","ExportView P&A + Special Generators"}) + ExportView(jv1+"|", "CSV", outputDir+"\\gmTruckDataIISP.csv", {baseTrucks+".ZONE", "LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr"}, {{"CSV Header"}}) + end + + // If data is not available then interpolate from closest available years + else do + // Get previous and next closest year + dim IY_Year[2] + IY_Year[1] = prevYear + IY_Year[2] = nextYear + + for j = 1 to 2 do + view="jv"+I2S(j) + Setview(view) + col_name="Y"+I2S(IY_Year[j]) + lhd_a = CreateExpression(view, "LHD_Attr", "LHD_Attractions + Nz("+col_name+" * trkAttraction * lhdShare)", ) + lhd_p = CreateExpression(view, "LHD_Prod", "LHD_Productions + Nz("+col_name+" * trkProduction * lhdShare)", ) + mhd_a = CreateExpression(view, "MHD_Attr", "MHD_Attractions + Nz("+col_name+" * trkAttraction * mhdShare)", ) + mhd_p = CreateExpression(view, "MHD_Prod", "MHD_Productions + Nz("+col_name+" * trkProduction * mhdShare)", ) + hhd_a = CreateExpression(view, "HHD_Attr", "HHD_Attractions + Nz("+col_name+" * trkAttraction * hhdShare)", ) + hhd_p = CreateExpression(view, "HHD_Prod", "HHD_Productions + Nz("+col_name+" * trkProduction * hhdShare)", ) + RunMacro("HwycadLog",{"TruckModel.rsc: trkSpecialGen","ExportView P&A + Special Generators"}) + ExportView(view+"|", "CSV", outputDir+"\\gmTruckDataIISP"+I2S(IY_Year[j])+".csv", {"ZONE", "LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr"}, {{"CSV Header"}}) + Closeview(view) + jv2 = Joinviews("JV2", baseTrucks+".ZONE", specGenerators+".TAZ", ) + end + + ok=RunMacro("Interpolate",{"gmTruckDataIISP"+I2S(prevYear)+".csv","gmTruckDataIISP"+I2S(nextYear)+".csv","gmTruckDataIISP.csv",intFyear,prevYear,nextYear}) + end + +EndMacro + + +/********************************************************************************************************** +Balances total production and attraction for lhd, mhd, hhd, ei and ie trips + +Inputs: + input\specialGenerators.csv + output\gmTruckDataII.csv + + inputTruckDir\regionalEItrips.csv + inputTruckDir\regionalEItrips.csv + inputTruckDir\regionalIEtrips.csv + inputTruckDir\regionalIEtrips.csv + inputTruckDir\regionalEEtrips.csv + inputTruckDir\regionalEEtrips.csv + +Outputs: + output\gmTruckDataBalanced.bin + +**********************************************************************************************************/ +Macro "trkBalance" (strFyear,intFyear,arrInterimYear) + shared path, inputDir, outputDir, inputTruckDir + + // Check if interim year and interpolate trip files if it is + booInterimYear = arrInterimYear[1] + + // 1 = Forecast year, 2 = Interim year and needs interpolation + if booInterimYear = 2 then do + prevYear = arrInterimYear[2] + nextYear = arrInterimYear[3] + + // copy files to scenario directory + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEItrips"+I2S(prevYear)+".csv",outputDir+"\\regionalEItrips_prev.csv"}) + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEItrips"+I2S(nextYear)+".csv",outputDir+"\\regionalEItrips_next.csv"}) + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalIEtrips"+I2S(prevYear)+".csv",outputDir+"\\regionalIEtrips_prev.csv"}) + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalIEtrips"+I2S(nextYear)+".csv",outputDir+"\\regionalIEtrips_next.csv"}) + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+I2S(prevYear)+".csv",outputDir+"\\regionalEEtrips_prev.csv"}) + ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+I2S(nextYear)+".csv",outputDir+"\\regionalEEtrips_next.csv"}) + + // Call Macro Interpolate //arr = {"previous year data file","next year data file","new year data file name(macro will create this)} + ok=RunMacro("Interpolate",{"regionalEItrips_prev.csv","regionalEItrips_next.csv","regionalEItrips.csv",intFyear,prevYear,nextYear}) + ok=RunMacro("Interpolate",{"regionalIEtrips_prev.csv","regionalIEtrips_next.csv","regionalIEtrips.csv",intFyear,prevYear,nextYear}) + ok=RunMacro("Interpolate",{"regionalEEtrips_prev.csv","regionalEEtrips_next.csv","regionalEEtrips.csv",intFyear,prevYear,nextYear}) + + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_prev.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_next.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_prev.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_next.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_prev.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_next.csv"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_prev.dcc"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_next.dcc"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_prev.dcc"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_next.dcc"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_prev.dcc"}) + ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_next.dcc"}) + end + + // If not interim year then copy EE to output folder + if booInterimYear = 1 then do + RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+strFyear+".csv",outputDir+"\\regionalEEtrips.csv"}) + end + // Balance each truck type (except EE [external-external], which is already balanced) + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance LHD"}) + RunMacro("trkBalanceOneType", "LHD") + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance MHD"}) + RunMacro("trkBalanceOneType", "MHD") + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance HHD"}) + RunMacro("trkBalanceOneType", "HHD") + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance EI"}) + RunMacro("trkBalanceRegionalTrucks", "EI", strFyear,booInterimYear) + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance IE"}) + RunMacro("trkBalanceRegionalTrucks", "IE", strFyear,booInterimYear) + + // Combine single balanced files + vw1 = Opentable("BALANCE_LHD", "FFB", {outputDir+"\\gmTruckDataBal_LHD.bin",}) + vw2 = Opentable("BALANCE_MHD", "FFB", {outputDir+"\\gmTruckDataBal_MHD.bin",}) + vw3 = Opentable("BALANCE_HHD", "FFB", {outputDir+"\\gmTruckDataBal_HHD.bin",}) + vw4 = Opentable("BALANCE_EI", "FFB", {outputDir+"\\regionalEItripsBalanced.bin",}) + RenameField("BALANCE_EI.HHD_Attr" ,"EI_Attr") + RenameField("BALANCE_EI.EItrucks" ,"EI_Prod") + + Opts = null + Opts.Input.[Dataview Set] = {outputDir+"\\regionalEItripsBalanced.bin" , vw4} + Opts.Global.Fields = {"EI_Prod"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if EI_Prod=null then 0.00 else EI_Prod"} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + + vw5 = Opentable("BALANCE_IE", "FFB", {outputDir+ "\\regionalIEtripsBalanced.bin",}) + RenameField("BALANCE_IE.IEtrucks" ,"IE_Attr") + RenameField("BALANCE_IE.HHD_Prod" ,"IE_Prod") + Opts = null + Opts.Input.[Dataview Set] = {outputDir+ "\\regionalIEtripsBalanced.bin" , vw5} + Opts.Global.Fields = {"IE_Attr"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"if IE_Attr=null then 0.00 else IE_Attr"} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + + jv1 = Joinviews("JV1", vw1 + ".ID1", vw2 + ".ID1",) + jv2 = Joinviews("JV2", jv1 + ".BALANCE_LHD.ID1", vw3 + ".ID1",) + jv3 = Joinviews("JV3", jv2 + ".BALANCE_LHD.ID1", vw4 + ".ID1",) + jv4 = Joinviews("JV4", jv3 + ".BALANCE_LHD.ID1", vw5 + ".ID1",) + RenameField("BALANCE_LHD.ID1", "ID") + + Setview(jv4) + n = SelectByQuery("notzeros", "several", "Select * where ID <> 0",) + + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","ExportView - Combined Balanced Flow"}) + ExportView("JV4|notzeros", "FFB", outputDir+"\\gmTruckDataBalanced.bin", {"ID","LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr", "IE_Prod", "IE_Attr", "EI_Prod","EI_Attr"},) + + Closeview(jv1) + Closeview(jv2) + Closeview(jv3) + Closeview(jv4) + Closeview(vw1) + Closeview(vw2) + Closeview(vw3) + Closeview(vw4) + Closeview(vw5) +EndMacro + + +/********************************************************************************************************** +Balances total production and attraction for truck type (type = lhd, mhd or hhd) + +Inputs: + output\gmTruckDataIISP.csv + +Outputs: + output\gmTruckDataBal_lhd.bin + output\gmTruckDataBal_mhd.bin + output\gmTruckDataBal_hhd.bin +**********************************************************************************************************/ +Macro "trkBalanceOneType" (type) + shared path, inputDir, outputDir + + RunMacro("TCB Init") + + Opts = null + Opts.Input.[Data View Set] = {outputDir+"\\gmTruckDataIISP.csv", "gmTruckDataIISP"} + Opts.Field.[Vector 1] = {"[gmTruckDataIISP]." + type + "_Prod"} + Opts.Field.[Vector 2] = {"[gmTruckDataIISP]." + type + "_Attr"} + Opts.Global.[Holding Method] = {"Weighted Sum"} + Opts.Global.[Percent Weight] = {50} + Opts.Global.[Sum Weight] = {100} + Opts.Global.[Store Type] = 1 + Opts.Output.[Output Table] = outputDir+"\\gmTruckDataBal_" + type + ".bin" + + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalanceOneType", "Balance "+type}) + ok=RunMacro("TCB Run Procedure", "Balance", Opts, &Ret) +EndMacro + + +/********************************************************************************************************** +Balances EI trips where sum of E is fixed and I of truck attraction is adjusted or +IE trips where sum of E is fixed and I of truck production is adjusted + +Inputs: + output\regionalIEtrips.csv + output\regionalEItrips.csv + inputTruckDir\regionalIEtrips.csv + inputTruckDir\regionalIEtrips.csv + inputTruckDir\regionalEItrips.csv + inputTruckDir\regionalEItrips.csv + output\gmTruckDataBal_HHD.bin + +Outputs: + output\regionaltripsBalanced.bin + +**********************************************************************************************************/ +Macro "trkBalanceRegionalTrucks" (direction,strFyear,booInterimYear) + shared path, inputDir, outputDir, inputTruckDir + + // Not an interim year, read regional trips from inputTruckDir + if booInterimYear=1 then do + strPathRegTrips=inputTruckDir+"\\regional"+direction+"trips"+strFyear+".csv" + end + + //Interim year, read regional trips from output directory + else do + strPathRegTrips=outputDir+"\\regional"+direction+"trips.csv" + end + + Opts = null + if direction = "EI" then do + Opts.Input.[Data Set] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "fromZone"}, "BALANCE_HHD+regionalEItrips"+strFyear} + Opts.Input.[Data View] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "fromZone"}, "BALANCE_HHD+regionalEItrips"+strFyear} + end + else do + Opts.Input.[Data Set] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "toZone"}, "BALANCE_HHD+regionalIEtrips"+strFyear} + Opts.Input.[Data View] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "toZone"}, "BALANCE_HHD+regionalIEtrips"+strFyear} + end + + Opts.Input.[V1 Holding Sets] = {} + Opts.Input.[V2 Holding Sets] = {} + if direction = "EI" then do + Opts.Field.[Vector 1] = {"[BALANCE_HHD+regionalEItrips"+strFyear+"].HHD_Attr"} + Opts.Field.[Vector 2] = {"[BALANCE_HHD+regionalEItrips"+strFyear+"].EITrucks"} + end + else do + Opts.Field.[Vector 1] = {"[BALANCE_HHD+regionalIEtrips"+strFyear+"].HHD_Prod"} + Opts.Field.[Vector 2] = {"[BALANCE_HHD+regionalIEtrips"+strFyear+"].IETrucks"} + end + + Opts.Global.Pairs = 1 + Opts.Global.[Holding Method] = {2} + Opts.Global.[Percent Weight] = {50} + Opts.Global.[Sum Weight] = {100} + Opts.Global.[V1 Options] = {1} + Opts.Global.[V2 Options] = {1} + Opts.Global.[Store Type] = 1 + Opts.Output.[Output Table] = outputDir+"\\regional"+direction+"tripsBalanced.bin" + + RunMacro("HwycadLog",{"TruckModel.rsc: trkBalanceRegionalTrucks","Balance "+direction}) + RunMacro("TCB Run Procedure", 1, "Balance", Opts) +EndMacro + + + +/********************************************************************************************************** +Distributes truck lhd, mhd, hhd, ie and ei truck trips + +Inputs: + output\gmTruckDataBalanced.bin + output\regionalEEtrips.csv + output\imptrk_EA.mtx + output\imptrk_AM.mtx + output\imptrk_MD.mtx + output\imptrk_PM.mtx + output\imptrk_EV.mtx + +Outputs: + output\distributionMatricesTruck.mtx + outputDir\regionalEEtrips.mtx + outputDir\dailyDistributionMatricesTruckAll.mtx + outputDir\dailyDistributionMatricesTruckEA.mtx + outputDir\dailyDistributionMatricesTruckAM.mtx + outputDir\dailyDistributionMatricesTruckMD.mtx + outputDir\dailyDistributionMatricesTruckPM.mtx + outputDir\dailyDistributionMatricesTruckEV.mtx + +**********************************************************************************************************/ + +Macro "trkDistribution" (properties) + shared path, inputDir, outputDir + + // Get forecast year + strFyear=RunMacro("read properties",properties,"truck.FFyear","S") + + //-------------------------------------------------- + // This section does the truck trip distribution + //-------------------------------------------------- + // Use mid-day (md) truck skim for distribution + periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} + p = 3 + md_truck_skim = outputDir +"\\imptrk"+periods[p]+".mtx" + md_truck_FF = "*SCST"+periods[p] + md_truck_IM = "*STM"+periods[p]+" (Skim)" + md_truck_KF = "*STM"+periods[p]+" (Skim)" + + // Open md truck skims + Opts.Input.[FF Matrix Currencies] = {{md_truck_skim, md_truck_FF,,},{md_truck_skim, md_truck_FF,,}, {md_truck_skim, md_truck_FF,,}, {md_truck_skim, md_truck_FF,,},{md_truck_skim, md_truck_FF,,}} + Opts.Input.[Imp Matrix Currencies] = {{md_truck_skim, md_truck_IM,,},{md_truck_skim, md_truck_IM,,}, {md_truck_skim, md_truck_IM,,}, {md_truck_skim, md_truck_IM,,},{md_truck_skim, md_truck_IM,,}} + Opts.Input.[KF Matrix Currencies] = {{md_truck_skim, md_truck_KF,,},{md_truck_skim, md_truck_KF,,}, {md_truck_skim, md_truck_KF,,}, {md_truck_skim, md_truck_KF,,},{md_truck_skim, md_truck_KF,,}} + + // Open truck trips + Opts.Input.[PA View Set] = {outputDir+"\\gmTruckDataBalanced.bin", "gmTruckDataBalanced"} + + // Set gravity model settings + Opts.Input.[FF Tables] = {{outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}} + Opts.Field.[Prod Fields] = {"gmTruckDataBalanced.LHD_Prod", "gmTruckDataBalanced.MHD_Prod", "gmTruckDataBalanced.HHD_Prod", "gmTruckDataBalanced.IE_Prod", "gmTruckDataBalanced.EI_Prod"} + Opts.Field.[Attr Fields] = {"gmTruckDataBalanced.LHD_Attr", "gmTruckDataBalanced.MHD_Attr", "gmTruckDataBalanced.HHD_Attr", "gmTruckDataBalanced.IE_Attr", "gmTruckDataBalanced.EI_Attr"} + Opts.Field.[FF Table Fields] = {"gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID"} + Opts.Field.[FF Table Times] = {"gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID"} + Opts.Global.[Purpose Names] = {"lhd", "mhd", "hhd", "IE", "EI"} + Opts.Global.Iterations = {100, 100, 100, 100, 100} + Opts.Global.Convergence = {0.01, 0.01, 0.01, 0.01, 0.01} + Opts.Global.[Constraint Type] = {"Double", "Double", "Double", "Columns", "Rows"} + Opts.Global.[Fric Factor Type] = {"Exponential", "Exponential", "Exponential", "Exponential", "Exponential"} + Opts.Global.[A List] = {1, 1, 1, 1, 1} + Opts.Global.[B List] = {0.3, 0.3, 0.3, 0.3, 0.3} + Opts.Global.[C List] = {0.045, 0.03, 0.03, 0.03, 0.03} + Opts.Flag.[Use K Factors] = {0, 0, 0, 0, 0} + Opts.Output.[Output Matrix].Label = "Distribution Matrix" + Opts.Output.[Output Matrix].Compression = 1 + Opts.Output.[Output Matrix].[File Name] = outputDir + "\\distributionMatricesTruck.mtx" + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Gravity"}) + RunMacro("TCB Run Procedure", 1, "Gravity", Opts) + + + //-------------------------------------------------- + // Adds IE, EI and EE trips to the truck trips by type + //-------------------------------------------------- + // Create EE trip matrix + viewEE = Opentable("EEtrips", "CSV", {outputDir + "\\regionalEEtrips.csv"}) + matrixEE = CreateMatrixFromView("EEmatrix", "EEtrips|", "fromZone", "toZone", {"EEtrucks"}, {{"File Name", outputDir + "\\regionalEEtrips.mtx"}, {"Type", "Float"}, {"Sparse", "No"}, {"Column Major", "No"}, {"File Based", "Yes"}}) + Closeview(viewEE) + + // Split truck trips by type + trkShare = {0.307, 0.155, 0.538} + trkTypes = {"lhd", "mhd", "hhd"} + + Opts = null + Opts.Input.[Matrix Currencies] = {{outputDir + "\\distributionMatricesTruck.mtx", "lhd", "Row ID's", "Col ID's"}, + {outputDir + "\\distributionMatricesTruck.mtx", "mhd", "Row ID's", "Col ID's"}, + {outputDir + "\\distributionMatricesTruck.mtx", "hhd", "Row ID's", "Col ID's"}, + {outputDir + "\\distributionMatricesTruck.mtx", "IE", "Row ID's", "Col ID's"}, + {outputDir + "\\distributionMatricesTruck.mtx", "EI", "Row ID's", "Col ID's"}, + {outputDir + "\\regionalEEtrips.mtx", "EEtrucks", "fromZone", "toZone"}} + Opts.Global.Operation = "Union" + Opts.Output.[Combined Matrix].Label = "Union Combine" + Opts.Output.[Combined Matrix].Compression = 1 + Opts.Output.[Combined Matrix].[File Name] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Combine Matrix Files"}) + RunMacro("TCB Run Operation", 1, "Combine Matrix Files", Opts) + + + // Aportion and add IE, EI and EE trips to LHD, MHD and HHD + for i = 1 to trkTypes.length do + Opts = null + Opts.Input.[Matrix Currency] = {outputDir + "\\dailyDistributionMatricesTruckAll.mtx", trkTypes[i], "Rows", "Columns"} + Opts.Input.[Core Currencies] = {{outputDir + "\\dailyDistributionMatricesTruckAll.mtx",trkTypes[i], "Rows", "Columns"}, + {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","EI", "Rows", "Columns"}, + {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","IE", "Rows", "Columns"}, + {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","EEtrucks", "Rows", "Columns"}} + Opts.Global.Method = 7 + Opts.Global.[Cell Range] = 2 + Opts.Global.[Matrix K] = {1, trkShare[i], trkShare[i], trkShare[i]} + Opts.Global.[Force Missing] = "No" + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Fill Matrices"}) + ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) + end + + // Drop IE matrix core + Opts = null + Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" + Opts.global.[Drop Core] = {"IE"} + RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) + + // Drop EI matrix core + Opts = null + Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" + Opts.global.[Drop Core] = {"EI"} + RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) + + // Drop EE matrix core + Opts = null + Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" + Opts.global.[Drop Core] = {"EEtrucks"} + RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) + + + //-------------------------------------------------- + // Set intrazonal truck trips to 0. Note: intrazonal truck trips are not necessarily 0 in reality, but they are + // not simulated in this model. Having an undefined value for intrazonal trips provides problems in the emission + // estimation of the EMFAC2007 model. Therefore, intrazonals are artificially set to 0. + //-------------------------------------------------- + Opts = null + Opts.Input.[Matrix Currency] = {outputDir + "\\dailyDistributionMatricesTruckAll.mtx", "lhd", "Rows", "Columns"} + Opts.Global.Method = 1 + Opts.Global.Value = 0 + Opts.Global.[Cell Range] = 3 + Opts.Global.[Matrix Range] = 3 + Opts.Global.[Matrix List] = {"lhd", "mhd", "hhd"} + ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) + if !ok then goto quit + + // Split into time of day + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, EA"}) + RunMacro("splitIntoTimeOfDay", 1) // EA Off-peak + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, AM"}) + RunMacro("splitIntoTimeOfDay", 2) // AM Peak + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, MD"}) + RunMacro("splitIntoTimeOfDay", 3) // MD Off-peak + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, PM"}) + RunMacro("splitIntoTimeOfDay", 4) // PM Peak + RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, EV"}) + RunMacro("splitIntoTimeOfDay", 5) // EV Off-peak + + run_ok=1 + Return(run_ok) + + quit: + Return(ok) +EndMacro + + +/********************************************************************************************************** +Splits daily truck trips to five periods EA, AM, MD, PM and EV + +Inputs: + outputDir\dailyDistributionMatricesTruckAll.mtx + timeShare = {0.1018, 0.1698, 0.4284, 0.1543, 0.1457} + borderTimeShare = {0.0188, 0.1812, 0.4629, 0.2310, 0.1061} + +Outputs: + outputDir\dailyDistributionMatricesTruckEA.mtx + outputDir\dailyDistributionMatricesTruckAM.mtx + outputDir\dailyDistributionMatricesTruckMD.mtx + outputDir\dailyDistributionMatricesTruckPM.mtx + outputDir\dailyDistributionMatricesTruckEV.mtx + +**********************************************************************************************************/ +Macro "splitIntoTimeOfDay" (period) + shared path, inputDir, outputDir + + // Split lhd, mhd and hhd truck trips into time-of-day periods + periodName = {"EA","AM","MD","PM","EV"} + mode = {"lhd", "mhd", "hhd"} + timeShare = {0.1018, 0.1698, 0.4284, 0.1543, 0.1457} // share of truck trips per time period for all zones except border crossings + borderTimeShare = {0.0188, 0.1812, 0.4629, 0.2310, 0.1061} // share of truck trips per time period for border crossings + dim borderCorrection[timeShare.length] // correct values at border after multiplication with timeShare + for i = 1 to borderCorrection.length do + borderCorrection[i] = borderTimeShare[i] / timeShare[i] + end + + // Copy original matrix into time-of-day matrix + mat = OpenMatrix(outputDir + "\\dailyDistributionMatricesTruckAll.mtx", ) + mc = CreateMatrixCurrency(mat, "lhd", "Rows", "Columns", ) + newFile = outputDir + "\\dailyDistributionMatricesTruck" + periodName[period] + ".mtx" + label = "Truck Table " + periodName[period] + " Peak" + new_mat = CopyMatrix(mc, {{"File Name", newFile}, {"Label", label}, {"File Based", "Yes"}}) + + // Multiply entire matrix with time-of-day share + Opts = null + Opts.Input.[Matrix Currency] = {newFile, "lhd", "Rows", "Columns"} + Opts.Global.Method = 5 + Opts.Global.Value = timeShare[period] + Opts.Global.[Cell Range] = 2 + Opts.Global.[Matrix Range] = 3 + Opts.Global.[Matrix List] = {"lhd", "mhd", "hhd"} + RunMacro("HwycadLog",{"TruckModel.rsc: splitIntoTimeOfDay","Fill Matrices"}) + ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) + if !ok then goto quit + + // Correct time-of-day share for destination zones 1 through 5 (border zones) + for i = 1 to mode.length do + mat = OpenMatrix(newFile,) + mc = CreateMatrixCurrency(mat, mode[i], "Rows", "Columns", ) + cols = {"1", "2", "3", "4", "5"} + operation = {"Multiply", borderCorrection[period]} + FillMatrix(mc, null, cols, operation, ) + + mat = OpenMatrix(newFile,) + mc = CreateMatrixCurrency(mat, mode[i], "Rows", "Columns", ) + rows = {"1", "2", "3", "4", "5"} + operation = {"Multiply", borderCorrection[period]} + FillMatrix(mc, rows, null, operation, ) + end + + run_ok=1 + Return(run_ok) + + quit: + Return(ok) +EndMacro + + + + +/********************************************************************************************************** + Truck Toll Diversion Model + Splits Truck Demand to Non-Toll and Toll + called after truck model but before combine trktrips + Ben Stabler, stabler@pbworld.com, 12/02/10 + +Inputs: + Each skim matrix is suffixed with xx, yy where: + xx is mode indicating following truck types: + lhdn, mhdn, hhdn, lhdt, mhdt, and hhdt + + yy is period, as follows: + EA: Early AM + AM: AM peak + MD: Midday + PM: PM peak + EV: Evening + + output\impXXYY.mtx + outputDir\dailyDistributionMatricesTruckYY.mtx + +Outputs: + Adds toll and non-toll cores to + outputDir\dailyDistributionMatricesTruckYY.mtx + +**********************************************************************************************************/ +Macro "trk toll diversion model" + shared path, inputDir, outputDir + RunMacro("TCB Init") + + // Toll diversion curve settings + nest_param = 10 + vot = 0.02 //(minutes/cent) + + periodName = {"EA","AM","MD","PM","EV"} // must be consistent with filename arrays below + trkTypes = {"lhd", "mhd", "hhd"} // truck types + trkTollFactor = {1, 1.03, 2.33} // truck toll factor + + // Loop by time period + for period = 1 to periodName.length do + + // Open truck trips + fileNameTruck = outputDir + "\\dailyDistributionMatricesTruck" + periodName[period] + ".mtx" + m = OpenMatrix(fileNameTruck,) + + // Loop by truck class + for trkType = 1 to trkTypes.length do + nontollmtx=outputDir+"\\imp"+trkTypes[trkType]+"n_"+periodName[period]+".mtx" // non-toll skims + tollmtx=outputDir+"\\imp"+trkTypes[trkType]+"t_"+periodName[period]+".mtx" // toll skims + + // Check and if exist, drop toll and non-toll matrix cores by truck type + coreNames = GetMatrixCoreNames(m) + for c = 1 to coreNames.length do + if (coreNames[c] = trkTypes[trkType] + "t") then DropMatrixCore(m, trkTypes[trkType] + "t") + if (coreNames[c] = trkTypes[trkType] + "n") then DropMatrixCore(m, trkTypes[trkType] + "n") + end + + // Add toll and non-toll matrix + AddMatrixCore(m, trkTypes[trkType] + "t") + AddMatrixCore(m, trkTypes[trkType] + "n") + + // Diversion curve (time is in minutes, cost is in cents) + utility = "(([impedance truck].[*STM_"+periodName[period]+" (Skim)] - [impedance truck toll].[*STM_"+periodName[period]+" (Skim)]) - " + + String(vot) + " * " + "[impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"] * " + String(trkTollFactor[trkType]) + " ) / " + String(nest_param) + + expression = "if([impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"]!=0) then ( 1 / ( 1 + exp(-" + utility + ") ) ) else [impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"]" + + // Calculate toll matrix + Opts = null + Opts.Input.[Matrix Currency] = {fileNameTruck, trkTypes[trkType] + "t", "Rows", "Columns"} + Opts.Input.[Formula Currencies] = {{nontollmtx, "*STM_"+periodName[period]+" (Skim)", "Origin", "Destination"}, {tollmtx, "*STM_"+periodName[period]+" (Skim)", "Origin", "Destination"}} + Opts.Global.Method = 11 + Opts.Global.[Cell Range] = 2 + Opts.Global.[Expression Text] = "[" + trkTypes[trkType] + "] * " + expression + Opts.Global.[Formula Labels] = {"impedance truck", "impedance truck toll"} + Opts.Global.[Force Missing] = "Yes" + ok = RunMacro("TCB Run Operation", "Fill Matrices", Opts) + if !ok then goto quit + + // Calculate non-toll matrix + mc_n = CreateMatrixCurrency(m, trkTypes[trkType] + "n", "Rows", "Columns",) + mc_t = CreateMatrixCurrency(m, trkTypes[trkType] + "t", "Rows", "Columns",) + mc = CreateMatrixCurrency(m, trkTypes[trkType], "Rows", "Columns",) + mc_n := mc - mc_t + end + end + + //return 1 if macro completed + run_ok = 1 + Return(run_ok) + + quit: + Return(ok) + +EndMacro + + +/********************************************************************************************************** +Used to gerenarte data for the forecast years from the years where data is avaialble +Interpolates data from the closest previous year and next years +All files are from read and written to output directory + +Inputs: + output\file.csv + output\file.csv + +Outputs: + output\file.csv + +**********************************************************************************************************/ +Macro "Interpolate" (arr) + shared outputDir + prevfile= arr[1] + nextfile= arr[2] + newfile = arr[3] + Fyear = arr[4] + prevYear= arr[5] + nextYear= arr[6] + + newview = ParseString(newfile, ".") // Get file name and use it as view name + pview = ParseString(prevfile, ".") // Get file name + nview = ParseString(nextfile, ".") // Get file name + + // Delete dictionary files if exist + di = GetDirectoryInfo(outputDir+"\\"+pview[1]+".dcc", "File") + if di.length > 0 then do + ok=RunMacro("SDdeletefile",{outputDir+"\\"+pview[1]+".dcc"}) + end + di = GetDirectoryInfo(outputDir+"\\"+nview[1]+".dcc", "File") + if di.length > 0 then do + ok=RunMacro("SDdeletefile",{outputDir+"\\"+nview[1]+".dcc"}) + end + di = GetDirectoryInfo(outputDir+"\\"+newview[1]+".dcc", "File") + if di.length > 0 then do + ok=RunMacro("SDdeletefile",{outputDir+"\\"+newview[1]+".dcc"}) + end + + // Open previous year data + prevview = Opentable("prevview", "CSV", {outputDir+"\\"+prevfile,}) + + // Open next year data + nextview = Opentable("nextview", "CSV", {outputDir+"\\"+nextfile,}) + + // Number of zones (assuming both prev and next year contain same number of zones) + zones = GetRecordCount(prevview, null) + + // Get data table structure + str = GetTableStructure(prevview) + dim pfieldName[str.length],nfieldName[str.length] + for i =1 to str.length do + pfieldName[i] = prevview+"."+str[i][1] // gets field names from previous year data + nfieldName[i] = nextview+"."+str[i][1] // gets field names from previous year data + end + + // Get fields from previous and next years + prevf = GetDataVectors(prevview+"|",pfieldName,) + nextf = GetDataVectors(nextview+"|",nfieldName,) + + // Create a Fyear table and do interpolation + view = CreateTable(newview[1], outputDir+"\\"+ newfile, "CSV", str) + SetView(view) + + // Interpolate and set values for Fyear + for i = 1 to zones do // row loop + dim v[str.length] // array to hold new field computation + v[1] = {str[1][1], prevf[1][i]} // fill new fields, first field is zone and no interpolation + + for j = 1 to str.length do // field loop + // interpolate + v[j] = {str[j][1] , (prevf[j][i]+((Fyear-prevYear)*((nextf[j][i]-prevf[j][i])/(nextYear-prevYear))))} + end + rh = AddRecord(view,v) + end + + // Close all opened views + vws = GetViewNames() + for p = 1 to vws.length do + CloseView(vws[p]) + end +EndMacro + diff --git a/sandag_abm/src/main/gisdk/Utilities.rsc b/sandag_abm/src/main/gisdk/Utilities.rsc new file mode 100644 index 0000000..7fcefcc --- /dev/null +++ b/sandag_abm/src/main/gisdk/Utilities.rsc @@ -0,0 +1,568 @@ +Macro "ModifyOptionsOption" (options_array,option_key,key,value) + spec = options_array.(option_key) + spec.(key) = value +EndMacro + +Macro "CloseAll" + // close all files in workspace + map_arr=GetMaps() + if ArrayLength(map_arr)>0 then do + open_maps=ArrayLength(map_arr[1]) + for mm=1 to open_maps do + CloseMap(map_arr[1][mm]) + end + end + + On NotFound goto no_more_eds + still_more_eds: + CloseEditor() + goto still_more_eds + + no_more_eds: + On NotFound default + + view_arr=GetViews() + if ArrayLength(view_arr)>0 then do + On NotFound goto cont_views + open_views=ArrayLength(view_arr[1]) + for vv=1 to open_views do + CloseView(view_arr[1][vv]) + cont_views: + end + end +endMacro + +Macro "IsMapOpen" (map) + maps = GetMapNames() + for i = 1 to maps.length do + if maps[i] = map then do + return(True) + end + end + return(False) +EndMacro + +Macro "IsViewOpen" (view) + views = GetViewNames() + for i = 1 to views.length do + if views[i] = view then do + return(True) + end + end + return(False) +EndMacro + +Macro "SafeDeleteFile" (file) + //just ignores any errors + if GetFileInfo(file) <> null then do + On Error goto safe_delete_error + DeleteFile(file) + safe_delete_error: + On Error default + end +EndMacro + +Macro "DeleteFiles" (path) + files = GetDirectoryInfo(path,"All") + for i = 1 to files.length do + DeleteFile(RunMacro("FormPath",{path,files[i][1]})) + end +EndMacro + +Macro "SafeDeleteFiles" (path) + files = GetDirectoryInfo(path,"All") + for i = 1 to files.length do + RunMacro("SafeDeleteFile",RunMacro("FormPath",{path,files[i][1]})) + end +EndMacro + +Macro "SafeDeleteDatabase" (database_file) + //just ignores any errors + On Error goto safe_delete_database_error + On NotFound goto safe_delete_database_error + DeleteDatabase(file) + safe_delete_database_error: + On Error default + On NotFound default +EndMacro + +Macro "NormalizePath" (path) + if Len(path) > 1 and path[2] = ":" then do + path = Lower(path[1]) + Right(path,Len(path)-1) + end + return(Substitute(path,"/","\\",)) +EndMacro + +Macro "FormPath" (path_elements) + if TypeOf(path_elements) <> "array" then do + ShowMessage("Must form a path out of a list of elements, not: " + TypeOf(path_elements)) + ShowMessage(2) + end + //path_elements is an array of elements + path = "" + for i = 1 to path_elements.length do + //change / to \ + p = RunMacro("NormalizePath",path_elements[i]) + if Right(p,1) = "\\" then do + if Len(p) > 1 then do + p = Substring(p,1,Len(p)-1) + end + else do + p = "" + end + end + if Left(p,1) = "\\" then do + if Len(p) > 1 then do + p = Substring(p,2,Len(p)) + end + else do + p = "" + end + end + if path = "" then do + path = p + end + else do + path = path + "\\" + p + end + end + return(path) +EndMacro + +Macro "CreateMapForDatabase" (database_file,map_name) + linfo=GetDBInfo(database_file) + scope=linfo[1] + maps = GetMapNames() + map_name_not_ok = true + while map_name_not_ok do + map_name_not_ok = false + for i = 1 to maps.length do + if maps[i] = map_name then do + map_name = map_name + " " + map_name_not_ok = true + end + end + end + map=createMap(map_name,{{"Scope", scope},{"Auto Project","True"}}) + SetMapUnits("Miles") +EndMacro + +Macro "OpenDatabaseInMap" (database_file,map) + info=GetDBInfo(database_file) + map_made = False + if map <> null then do + map_made = RunMacro("IsMapOpen",map) + end + else do + map = info[2] + end + if not map_made then do + RunMacro("CreateMapForDatabase",database_file,map) + end + NewLayer=GetDBLayers(database_file) + layer = AddLayer(map,NewLayer[1],database_file,NewLayer[1]) + if NewLayer.length = 2 then do + //assumes it is a network file, and hides the nodes and adds the lines + SetLayerVisibility(map + "|" + layer,"False") + AddLayer(map,NewLayer[2],database_file,NewLayer[2]) + end + return(map) +EndMacro + +Macro "OpenDatabase" (database_file) + return(RunMacro("OpenDatabaseInMap",database_file,)) +EndMacro + +Macro "OpenRouteSystemInMap" (route_system_file,map) + info = GetRouteSystemInfo(route_system_file) + map_made = False + if map <> null then do + map_made = RunMacro("IsMapOpen",map) + end + else do + map = info[3].Label + end + if not map_made then do + RunMacro("CreateMapForDatabase",info[1],map) + end + RunMacro("Set Default RS Style",AddRouteSystemLayer(map,info[3].Label,route_system_file,),"TRUE","FALSE") + return(map) +EndMacro + +Macro "OpenRouteSystem" (route_system_file) + return(RunMacro("OpenRouteSystemInMap",route_system_file,)) +EndMacro + +Macro "CleanRecordValuesOptionsArray" (options_array,view_name) + for i = 1 to options_array.length do + options_array[i][1] = Substitute(options_array[i][1],view_name + ".","",) + end +EndMacro + +Macro "FormFieldSpec" (view,field) + //don't think the following is necessary + //issue_chars = {":"} + //fix = False + //for i = 1 to issue_chars.length do + // if Position(field,issue_chars[i]) > 0 then do + // fix = True + // end + //end + //if fix then do + // field = "[" + field + "]" + //end + return(view + "." + field) +EndMacro + +Macro "ToString" (value) + type = TypeOf(value) + if type = "string" then do + return(value) + end + else if type = "int" then do + return(i2s(value)) + end + else if type = "double" then do + return(r2s(value)) + end + else if type = "null" then do + return("") + end + ShowMessage("Type " + type + " not supported by ToString method") +EndMacro + +Macro "GetArrayIndex" (array,value) + //returns the index of value in array, or 0 if it is not found + type = TypeOf(value) + for i = 1 to array.length do + if TypeOf(array[i]) = type and array[i] = value then do + return(i) + end + end + return(0) +EndMacro + +Macro "ArraysEqual" (array1,array2) + if array1.length <> array2.length then do + return(False) + end + for i = 1 to array1.length do + if TypeOf(array1[i]) = "array" then do + if TypeOf(array2[i]) = "array" then do + if not RunMacro("ArraysEqual",array1[i],array2[i]) then do + return(False) + end + end + else do + return(False) + end + end + else if TypeOf(array2[i]) = "array" then do + return(False) + end + else do + if array1[i] <> array2[i] then do + return(False) + end + end + end + return(True) +EndMacro + +Macro "GetDatabaseColumns" (database_file,layer_name) + columns = null + if database_file <> null and GetFileInfo(database_file) <> null then do + current_layer = GetLayer() + current_view = GetView() + lyr = AddLayerToWorkspace("__temp__",database_file,layer_name,{{"Shared","True"}}) + layer_in_use = lyr <> "__temp__" + SetLayer(lyr) + v = GetView() + info = GetTableStructure(v) + for i = 1 to info.length do + columns = columns + {info[i][1]} + end + if not layer_in_use then do + DropLayerFromWorkspace(lyr) + end + if current_layer <> null then do + SetLayer(current_layer) + end + if current_view <> null then do + SetView(current_view) + end + end + return(columns) +EndMacro + +//same as built in TC function, but with error checking for escape and for if a file is in use +Macro "ChooseFileName" (file_types,title,options) + on escape do + fname = null + goto cfn_done + end + openfile: + fname = ChooseFileName(file_types,title,options) + if FileCheckUsage({fname},) then do + ShowMessage("File already in use. Please choose again.") + goto openfile + end + cfn_done: + on escape default + return(fname) +EndMacro + +Macro "RunProgram" (program_with_arguments,working_directory) //can't get output file to work right now..boo hoo + wd = "" + if working_directory <> null then do + wd = " /D" + working_directory + end + RunProgram("cmd /s /c \"start \"cmd\" " + wd + " /WAIT " + program_with_arguments + "\"",) +EndMacro + +Macro "AddElementToSortedArraySet" (array,element) + index = array.length + 1 + not_done = True + for i = 1 to array.length do + if not_done then do + if element = array[i] then do + index = -1 + not_done = False + end + else if element < array[i] then do + index = i + not_done = False + end + end + end + if index > 0 then do + array = InsertArrayElements(array,index,{element}) + end + return(array) +EndMacro + +Macro "ClearAndDeleteDirectory" (path) + //this doesn't do any error handling + info = GetDirectoryInfo(RunMacro("FormPath",{path,"*"}),"All") + for i = 1 to info.length do + f = RunMacro("FormPath",{path,info[i][1]}) + if info[i][2] = "file" then do + DeleteFile(f) + end + else if info[i][2] = "directory" then do + RunMacro("ClearAndDeleteDirectory",f) + end + end + RemoveDirectory(path) +EndMacro + +Macro "ReadPropertiesFile" (properties_file) + props = null + f = OpenFile(properties_file,"r") + while not FileAtEOF(f) do + line = Trim(ReadLine(f)) + if Len(line) > 0 then do + subs = ParseString(line,"=", {{"Include Empty",True}}) + key = subs[1] + value = JoinStrings(Subarray(subs,2,subs.length-1),"=") + props.(Trim(key)) = Trim(value) + end + end + CloseFile(f) + return(props) +EndMacro + +Macro "DetokenizePropertyValues" (properties,token_map) + for i = 1 to properties.length do + value = token_map[i][2] + for j = 1 to token_map.length do + value = Substitute(value,token_map[i][1],token_map[i][2],) + end + token_map[i][2] = value + end +EndMacro + +Macro "ComputeAreaBufferOverlayPercentages" (area_layer_file,centroid_layer_file,centroid_query,area_taz_field,node_taz_field,buffer_size) + //assumes node layer holds centroids from area layer, and bases its buffer around this + //returns array of percentage arrays, each holding {centroid_taz,overlay_taz,percentage} + omap_name = GetMap() + olayer_name = GetLayer() + oview_name = GetView() + + map = RunMacro("OpenDatabase",area_layer_file) + RunMacro("OpenDatabaseInMap",centroid_layer_file,map) + node_layer = GetMapLayers(map,"Point") + node_layer = node_layer[1][1] + area_layer = GetMapLayers(map,"Area") + area_layer = area_layer[1][1] + + SetLayer(node_layer) + centroid_selection = "centroids" + SelectByQuery(centroid_selection,"Several",centroid_query) + node_ids = GetSetIDs(node_layer + "|" + centroid_selection) + node_to_taz = null + for i = 1 to node_ids.length do + node_id = node_ids[i] + value = GetRecordValues(node_layer,IDToRecordHandle(node_id),{node_taz_field}) + node_to_taz.(i2s(node_id)) = value[1][2] + end + + percentages = null + temp_dir = GetFileInfo(area_layer_file) + temp_dir = Substring(area_layer_file,1,Len(area_layer_file) - Len(temp_dir[1])) + intersection_file = "temp_buffers.dbd" + percentages_file = "tempintersect" + temp_intersection_file = RunMacro("FormPath",{temp_dir,intersection_file}) + temp_percentages_file = RunMacro("FormPath",{temp_dir,percentages_file}) + EnableProgressBar("Calculating Area Buffer Percentages (buffer = " + r2s(buffer_size) + ")", 1) // Allow only a single progress bar + CreateProgressBar("", "True") + + nlen = node_ids.length + //for i = 1 to nlen do + for i = 1 to 20 do + node_id = node_ids[i] + node_taz = node_to_taz.(i2s(node_id)) + stat = UpdateProgressBar("Zone: " + i2s(node_taz), r2i(i/nlen*100)) + if stat = "True" then do + percentages = null + goto quit_loop + end + SetLayer(node_layer) + SelectByQuery("centroid","Several","SELECT * WHERE id=" + i2s(node_id)) + CreateBuffers(temp_intersection_file,"buffers",{"centroid"},"Value",{buffer_size},{{"Interior","Separate"},{"Exterior","Separate"}}) + + NewLayer = GetDBLayers(temp_intersection_file) + intersection_layer = AddLayer(map,"inter",temp_intersection_file,NewLayer[1]) + SetLayer(area_layer) + n = SelectByVicinity("subtaz","several",node_layer+"|centroid",buffer_size,{{"Inclusion","Intersecting"}}) + if n > 0 then do + ComputeIntersectionPercentages({intersection_layer, area_layer + "|subtaz"}, temp_percentages_file + ".bin",) + t = OpenTable("int_table", "FFB", {temp_percentages_file + ".bin"},) + tbar = t + "|" + rh = GetFirstRecord(tbar,) + while rh <> null do + vals = GetRecordValues(t,rh,{"Area_1", "Area_2","Percent_2"}) + if vals[1][2] = 1 and vals[2][2] <> 0 then do + value = GetRecordValues(area_layer,IDToRecordHandle(vals[2][2]),{area_taz_field}) + area_taz = node_to_taz.(i2s(value[1][2])) + percentages = percentages + {{node_taz,area_taz,vals[3][2]}} + end + rh = GetNextRecord(t+"|",,) + end + CloseView(t) + end + DropLayer(map,intersection_layer) + end + + quit_loop: + DestroyProgressBar() + CloseMap(map) + if omap_name <> null then do + SetMap(omap_name) + if olayer_name <> null then do + SetLayer(olayer_name) + end + end + if oview_name <> null then do + SetView(oview_name) + end + DeleteDatabase(temp_intersection_file) + DeleteFile(temp_percentages_file + ".bin") + DeleteFile(temp_percentages_file + ".BX") + DeleteFile(temp_percentages_file + ".dcb") + + return(percentages) +EndMacro + +Macro "ExportBintoCSV"(input_file_base, output_file_base) + + view = OpenTable("Binary Table","FFB",{input_file_base+".bin",}, {{"Shared", "True"}}) + SetView(view) + ExportView(view+"|", "CSV", output_file_base+".csv",,{{"CSV Header", "True"}, {"Force Numeric Type", "double"}}) + CloseView(view) + ok=1 + quit: + return(ok) +EndMacro + + +Macro "ComputeAreaOverlayPercentages" (area_layer_file,overlay_layer_file,area_id_field,overlay_id_field) + //returns percentage array, each element holding {area_id,overlay_id,% of overlay in area} + omap_name = GetMap() + olayer_name = GetLayer() + oview_name = GetView() + + map = RunMacro("OpenDatabase",area_layer_file) + area_layer = GetMapLayers(map,"Area") + area_layer = area_layer[1][1] + RunMacro("OpenDatabaseInMap",overlay_layer_file,map) + overlay_layer = GetMapLayers(map,"Area") + if overlay_layer[1][1] = area_layer then do + overlay_layer = overlay_layer[1][2] + end + else do + overlay_layer = overlay_layer[1][1] + end + + area_ids = GetSetIDs(area_layer + "|") + + percentages = null + temp_dir = GetFileInfo(area_layer_file) + temp_dir = Substring(area_layer_file,1,Len(area_layer_file) - Len(temp_dir[1])) + percentages_file = "tempintersect" + temp_percentages_file = RunMacro("FormPath",{temp_dir,percentages_file}) + EnableProgressBar("Calculating Area Intersections", 1) // Allow only a single progress bar + CreateProgressBar("", "True") + + nlen = area_ids.length + for i = 1 to nlen do + area_id = area_ids[i] + stat = UpdateProgressBar("Area id: " + i2s(area_id), r2i(i/nlen*100)) + if stat = "True" then do + percentages = null + goto quit_loop + end + SetLayer(area_layer) + SelectByQuery("select","Several","SELECT * WHERE id=" + i2s(area_id)) + area_sid = GetRecordValues(area_layer,IDToRecordHandle(area_id),{area_id_field}) + area_sid = area_sid[1][2] + SetLayer(overlay_layer) + n = SelectByVicinity("subtaz","several",area_layer+"|select",0,{{"Inclusion","Intersecting"}}) + if n > 0 then do + ComputeIntersectionPercentages({area_layer+"|select",overlay_layer + "|subtaz"}, temp_percentages_file + ".bin",) + t = OpenTable("int_table", "FFB", {temp_percentages_file + ".bin"},) + tbar = t + "|" + rh = GetFirstRecord(tbar,) + while rh <> null do + vals = GetRecordValues(t,rh,{"Area_1", "Area_2","Percent_2"}) + if vals[1][2] > 0 and vals[2][2] <> 0 and vals[3][2] > 0.0 then do + value = GetRecordValues(overlay_layer,IDToRecordHandle(vals[2][2]),{overlay_id_field}) + percentages = percentages + {{area_sid,value[1][2],vals[3][2]}} + end + rh = GetNextRecord(t+"|",,) + end + CloseView(t) + end + end + + quit_loop: + DestroyProgressBar() + CloseMap(map) + if omap_name <> null then do + SetMap(omap_name) + if olayer_name <> null then do + SetLayer(olayer_name) + end + end + if oview_name <> null then do + SetView(oview_name) + end + DeleteFile(temp_percentages_file + ".bin") + DeleteFile(temp_percentages_file + ".BX") + DeleteFile(temp_percentages_file + ".dcb") + + return(percentages) +EndMacro + + + diff --git a/sandag_abm/src/main/gisdk/commVehDist.rsc b/sandag_abm/src/main/gisdk/commVehDist.rsc new file mode 100644 index 0000000..5afba39 --- /dev/null +++ b/sandag_abm/src/main/gisdk/commVehDist.rsc @@ -0,0 +1,106 @@ +/************************************************************** + CommVehTOD.rsc + + TransCAD Macro used to run truck commercial vehicle distribution model. The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. + + A simple gravity model is used to distribute the truck trips. A blended travel time is used as the impedance measure, + specifically the weighted average of the AM travel time (one-third weight) and the midday travel time (two-thirds weight). + + Input: (1) Level-of-service matrices for the AM peak period (6 am to 10 am) and midday period (10 am to 3 pm) + which contain truck-class specific estimates of congested travel time (in minutes) + (2) Trip generation results in ASCII format with the following fields + (a) TAZ: zone number; + (b) PROD: very small truck trip productions; + (c) ATTR: very small truck trip attractions; + (4) A table of friction factors in ASCII format with the following fields (each 12 columns wide): (a) + impedance measure (blended travel time); (b) friction factors for very small trucks; + + Output: (1) A production/attraction trip table matrix of daily class-specific truck trips for very small trucks. + (2) A blended travel time matrix + + See also: (1) CommVehGen.rsc, which applies the generation model. + (2) CommVehTOD.rsc, which applies diurnal factors to the daily trips generated here. + + authors: jef (2012 03 11) dto (2011 09 08); dto (2010 08 31); cp (date unknown) + + +**************************************************************/ +Macro "Commercial Vehicle Distribution" + + shared path, inputDir, outputDir + + /* testing + RunMacro("TCB Init") + scenarioDirectory = "d:\\projects\\SANDAG\\AB_Model\\commercial_vehicles" + */ + + tazCommTripFile = "tazCommVeh.csv" + amMatrixName = "impcvn_AM.mtx" + amTableName = "*STM_AM (Skim)" + mdMatrixName = "impcvn_MD.mtx" + mdTableName = "*STM_MD (Skim)" + + frictionTable = "commVehFF.csv" + pa_tb = outputDir+"\\"+tazCommTripFile + ff_tb = inputDir+"\\"+frictionTable + + //outputs + blendMatrixName = "blendMatrix.mtx" + commVehTripTable = "commVehTrips.mtx" + + //create blended skim + amMatrix = OpenMatrix(outputDir + "\\"+amMatrixName, ) + amMC = CreateMatrixCurrency(amMatrix, amTableName, "Origin", "Destination", ) + mdMatrix = OpenMatrix(outputDir + "\\"+mdMatrixName, ) + mdMC = CreateMatrixCurrency(mdMatrix, mdTableName, "Origin", "Destination", ) + + blendMatrix = CopyMatrix(amMC, {{"File Name", outputDir+"\\"+blendMatrixName}, + {"Label", "AMMDBlend"}, + {"Table", amTableName}, + {"File Based", "Yes"}}) + + blendMC = CreateMatrixCurrency(blendMatrix, amTableName, "Origin", "Destination", ) + + blendMC := 0.3333*amMC + 0.6666*mdMC + + //prevent intrazonal + EvaluateMatrixExpression(blendMC, "99999", null, null,{{"Diagonal","true"}} ) + + + ff_vw = RunMacro("TCB OpenTable",,, {ff_tb}) + ok = (ff_vw <> null) if !ok then goto quit + + pa_vw = RunMacro("TCB OpenTable",,, {pa_tb}) + ok = (pa_vw <> null) if !ok then goto quit + + //Gravity Application + Opts = null + Opts.Input.[PA View Set] = {pa_tb} + Opts.Field.[Prod Fields] = {pa_vw + ".PROD"} + Opts.Field.[Attr Fields] = {pa_vw + ".ATTR"} + Opts.Input.[FF Matrix Currencies] = {{outputDir + "\\" + blendMatrixName, amTableName, "Origin", "Destination"}} + Opts.Input.[Imp Matrix Currencies] = {{outputDir + "\\" + blendMatrixName, amTableName, "Origin", "Destination"}} + Opts.Input.[FF Tables] = {{ff_tb}} + Opts.Input.[KF Matrix Currencies] = {{outputDir + "\\" +blendMatrixName, amTableName, "Origin", "Destination"}} + Opts.Field.[FF Table Fields] = {ff_vw +".FF"} + Opts.Field.[FF Table Times] = {ff_vw +".TIME"} + Opts.Global.[Purpose Names] = {"CommVeh"} + Opts.Global.Iterations = {50} + Opts.Global.Convergence = {0.1} + Opts.Global.[Constraint Type] = {"Double"} + Opts.Global.[Fric Factor Type] = {"Table"} + Opts.Global.[A List] = {1} + Opts.Global.[B List] = {0.3} + Opts.Global.[C List] = {0.005} + Opts.Flag.[Use K Factors] = {0} + Opts.Output.[Output Matrix].Label = "Output Matrix" + Opts.Output.[Output Matrix].[File Name] = outputDir + "\\" +commVehTripTable + ok = RunMacro("TCB Run Procedure", "Gravity", Opts) + + + RunMacro("close all") + quit: + Return(ok) + +EndMacro diff --git a/sandag_abm/src/main/gisdk/commVehDiversion.rsc b/sandag_abm/src/main/gisdk/commVehDiversion.rsc new file mode 100644 index 0000000..ba1c928 --- /dev/null +++ b/sandag_abm/src/main/gisdk/commVehDiversion.rsc @@ -0,0 +1,71 @@ +Macro "cv toll diversion model" + shared path, inputDir, outputDir +/* RunMacro("TCB Init") + //inputs + path = "D:\\projects\\sandag\\series13\\2012_test" + inputDir = path+"\\input" + outputDir = path+"\\output" + scenarioDirectory = "D:\\projects\\SANDAG\\series13\\2012_test" +*/ + // Toll diversion curve settings + nest_param = 10 + vot = 0.02 //(minutes/cent), currently $0.50 a minute + + periodName = {"EA","AM","MD","PM","EV"} // must be consistent with filename arrays below + //periodName = {"AM"} // must be consistent with filename arrays below + cvTollFactor = 1 // cv toll factor + + // Loop by time period + for period = 1 to periodName.length do + + // Open cv trips + fileNameCV = outputDir + "\\commVehTODTrips.mtx" + m = OpenMatrix(fileNameCV,) + + nontollmtx=outputDir+"\\impcv"+"n_"+periodName[period]+".mtx" // non-toll commercial vehicle skims + tollmtx=outputDir+"\\impcv"+"t_"+periodName[period]+".mtx" // toll commercial vehicle skims + OpenMatrix(nontollmtx,) + OpenMatrix(tollmtx,) + + // Add toll and non-toll matrix + AddMatrixCore(m, periodName[period]+ " Toll") + AddMatrixCore(m, periodName[period]+ " NonToll") + + // Diversion curve (time is in minutes, cost is in cents) + // First scale the toll cost since the cost is scaled for SR125 + tollCost = "[Output Matrix:1].[cvt - ITOLL2_"+ + periodName[period]+"]" + utility = "(([Output Matrix].[*STM_"+periodName[period]+" (Skim)] - [Output Matrix:1].[*STM_"+periodName[period]+" (Skim)]) - " + + String(vot) + " * " + tollCost + " * " + String(cvTollFactor) + " ) / " + String(nest_param) + + expression = "if(" + tollCost + "!=0) then ( 1 / ( 1 + exp(-" + utility + ") ) ) else " + tollCost + + // Calculate toll matrix + Opts = null + Opts.Input.[Matrix Currency] = {fileNameCV, periodName[period]+ " Toll", "Row ID's", "Col ID's"} + Opts.Input.[Formula Currencies] = {{nontollmtx, "*CVCST_"+periodName[period], "Origin", "Destination"}, {tollmtx, "*CVCST_"+periodName[period], "Origin", "Destination"}} + Opts.Global.Method = 11 + Opts.Global.[Cell Range] = 2 + Opts.Global.[Expression Text] = "[" + periodName[period] + " Trips] * " + expression + Opts.Global.[Formula Labels] = {"Output Matrix", "Output Matrix:1"} + Opts.Global.[Force Missing] = "Yes" + ok = RunMacro("TCB Run Operation", "Fill Matrices", Opts) + if !ok then goto quit + + // Calculate non-toll matrix + mc_n = CreateMatrixCurrency(m, periodName[period]+ " NonToll", "Row ID's", "Col ID's",) + mc_t = CreateMatrixCurrency(m, periodName[period]+ " Toll", "Row ID's", "Col ID's",) + mc = CreateMatrixCurrency(m, periodName[period]+ " Trips" , "Row ID's", "Col ID's",) + mc_n := mc - mc_t + + end + + //return 1 if macro completed + run_ok = 1 + Return(run_ok) + + quit: + Return(ok) + +EndMacro + \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/commVehGen.rsc b/sandag_abm/src/main/gisdk/commVehGen.rsc new file mode 100644 index 0000000..ccb8239 --- /dev/null +++ b/sandag_abm/src/main/gisdk/commVehGen.rsc @@ -0,0 +1,184 @@ +/************************************************************** + CommVehGen.rsc + + TransCAD Macro used to run truck trip generation model. The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. + + Linear regression models generate trip ends, balancing attractions to productions. + + Input: (1) MGRA file in CSV format with the following fields: (a) TOTEMP, total employment (same regardless + of classification system); (b) RETEMPN, retail trade employment per the NAICS classification system; + (c) FPSEMPN, financial and professional services employment per the NAICS classification system; (d) + HEREMPN, health, educational, and recreational employment per the NAICS classification system; (e) + OTHEMPN, other employment per the NAICS classification system; (f) AGREMPN, agricultural employment + per the NAICS classificatin system; (g) MWTEMPN, manufacturing, warehousing, and transportation + emp;loyment per the NAICS classification system; and, (h) TOTHH, total households. + + Output: (1) An ASCII file containing the following fields: (a) zone number; (b) very small truck trip productions; + (c) very small truck trip attractions + + See also: (1) TruckTripDistribution.job, which applies the distribution model. + (2) TruckTimeOfDay.job, which applies diurnal factors to the daily trips generated here. + (3) TruckTollChoice.job, which applies a toll/no toll choice model for trucks. + + version: 0.1 + authors: dto (2010 08 31); jef (2012 03 07) + + 5-2013 wsu fixed indexing bug + 7-2013 jef reduced truck rate for military employment to 0.3 + 8-2014 wsu reduced truck rate for military employment to 0.15 + +**************************************************************/ +Macro "Commercial Vehicle Generation" + + shared path, inputDir, outputDir ,scenarioYear + + mgraCommTripFile = "mgraCommVeh.csv" + tazCommTripFile = "tazCommVeh.csv" + + writeMgraData = true + calibrationFactor = 1.4 + + // read in the mgra data in CSV format + mgraView = OpenTable("MGRA View", "CSV", {inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) + + mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_total = GetDataVector(mgraView+"|", "emp_total", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + + RETEMPN = emp_retail + emp_personal_svcs_retail + FPSEMPN = emp_prof_bus_svcs + HEREMPN = emp_health + emp_pvt_ed_k12 + emp_pvt_ed_post_k12_oth + emp_amusement + AGREMPN = emp_ag + MWTEMPN = emp_const_non_bldg_prod + emp_const_bldg_prod + emp_mfg_prod + emp_trans + MILITARY = emp_fed_mil + TOTEMP = emp_total + OTHEMPN = TOTEMP - (RETEMPN + FPSEMPN + HEREMPN + AGREMPN + MWTEMPN + MILITARY) + TOTHH = hh + + verySmallP = calibrationFactor * (0.95409 * RETEMPN + 0.54333 * FPSEMPN + 0.50769 * HEREMPN + + 0.63558 * OTHEMPN + 1.10181 * AGREMPN + 0.81576 * MWTEMPN + + 0.15000 * MILITARY + + 0.1 * TOTHH) + + + // Wu added this section for military CTM trips adjustment to match military gate counts + properties = "\\conf\\sandag_abm.properties" + militaryCtmAdjustment = RunMacro("read properties",properties,"RunModel.militaryCtmAdjustment", "S") + if militaryCtmAdjustment = "true" then do + mgraView_m = OpenTable("Military MGRA View", "CSV", {inputDir+"\\cvm_military_adjustment.csv"}, {{"Shared", "True"}}) + //base_id = GetDataVector(mgraView_m+"|", "ID", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + //base_name = GetDataVector(mgraView_m+"|", "base", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + //taz_m = GetDataVector(mgraView_m+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + //mgra_m = GetDataVector(mgraView_m+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + base_id = GetDataVector(mgraView_m+"|", "ID",) + base_name = GetDataVector(mgraView_m+"|", "base",) + taz_m = GetDataVector(mgraView_m+"|", "TAZ",) + mgra_m = GetDataVector(mgraView_m+"|", "mgra",) + scale = GetDataVector(mgraView_m+"|", "scale", {{"Type", {{"scale", "Float"}}}} ) + + //scale verySmallP and verySmallA: why is his called verySmallP and verySmallA??? + for i = 1 to mgra.length do + for j = 1 to mgra_m.length do + if mgra[i] = mgra_m[j] then do + verySmallP[i]=verySmallP[i]*scale[j] + end + end + end + end + + verySmallA = verySmallP + + if writeMgraData = true then do + //create a table with the mgra trips + truckTripsMgra = CreateTable("truckTripsMgra",outputDir+"\\"+mgraCommTripFile, "CSV", { + {"MGRA", "Integer", 8, null, }, + {"PROD", "Real", 12, 4, }, + {"ATTR", "Real", 12, 4, } + }) + end + + //create a table with the taz trips + truckTripsTaz= CreateTable("truckTripsTaz",outputDir+"\\"+tazCommTripFile, "CSV", { + {"TAZ", "Integer", 8, null, }, + {"PROD", "Real", 12, 4, }, + {"ATTR", "Real", 12, 4, } + }) + + //now aggregate by TAZ + maxTaz = 0 + for i=1 to taz.length do + if taz[i] > maxTaz then maxTaz = taz[i] + end + + //arrays for holding productions and attractions by TAZ + dim tazProd[maxTaz] + dim tazAttr[maxTaz] + + //initialize arrays to 0 + for i = 1 to tazProd.length do + tazProd[i]=0 + tazAttr[i]=0 + end + + //aggregate mgra to taz arrays + for i = 1 to mgra.length do + + tazNumber = taz[i] + tazProd[tazNumber] = tazProd[tazNumber] + verySmallP[i] + tazAttr[tazNumber] = tazAttr[tazNumber] + verySmallA[i] + + if writeMgraData = true then do + AddRecord("truckTripsMgra", { + {"MGRA", mgra[i]}, + {"PROD", verySmallP[i]}, + {"ATTR", verySmallA[i]} + }) + end + end + + if writeMgraData = true then CloseView(truckTripsMgra) + + //add taz data to table + for i = 1 to maxTaz do + + AddRecord("truckTripsTaz", { + {"TAZ", i}, + {"PROD", tazProd[i]}, + {"ATTR", tazAttr[i]} + }) + end + + RunMacro("close all") + Return(1) +EndMacro diff --git a/sandag_abm/src/main/gisdk/commVehTOD.rsc b/sandag_abm/src/main/gisdk/commVehTOD.rsc new file mode 100644 index 0000000..9b1a459 --- /dev/null +++ b/sandag_abm/src/main/gisdk/commVehTOD.rsc @@ -0,0 +1,122 @@ +/************************************************************** + CommVehTOD.rsc + + TransCAD Macro used to run truck commercial vehicle time-of-day factoring. The very small truck generation model is based on the Phoenix + four-tire truck model documented in the TMIP Quick Response Freight Manual. + + The diurnal factors are taken from the BAYCAST-90 model with adjustments made during calibration to the very + small truck values to better match counts. + + Input: A production/attraction format trip table matrix of daily very small truck trips. + + Output: Five, time-of-day-specific trip table matrices containing very small trucks. + + See also: (1) CommVehGen.rsc, which applies the generation model. + (2) CommVehDist.rsc, which applies the distribution model. + + authors: jef (2012 03 11) dto (2011 09 08); dto (2010 08 31); cp (date unknown) + + +**************************************************************/ +Macro "Commercial Vehicle Time Of Day" (scenarioDirectory) + shared path, inputDir, outputDir + +/* + RunMacro("TCB Init") + + //inputs + scenarioDirectory = "d:\\projects\\SANDAG\\AB_Model\\commercial_vehicles" +*/ + + commVehTripTable = "commVehTrips.mtx" + + //outputs + commVehTODTable = "commVehTODTrips.mtx" + + //open input table + dailyMatrix = OpenMatrix(outputDir + "\\"+commVehTripTable, ) + dailyMC = CreateMatrixCurrency(dailyMatrix, "CommVeh", ,, ) + + //create transposed daily trip table + tmat = TransposeMatrix(dailyMatrix, {{"File Name", outputDir + "\\"+commVehTripTable+"t"}, + {"Label", "commVehT"}, + {"Type", "Double"}, + {"Sparse", "No"}, + {"Column Major", "No"}, + {"File Based", "No"}}) + + transMC = CreateMatrixCurrency(tmat, "commVeh", ,, ) + + //create output matrix + todMatrix = CopyMatrix(dailyMC, {{"File Name", outputDir+"\\"+commVehTODTable}, + {"Label", "CommVehTOD"}, + {"File Based", "Yes"}}) + + Opts = null + Opts.Input.[Input Matrix] =outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "OD Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "EA Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "AM Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "MD Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "PM Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable + Opts.Input.[New Core] = "EV Trips" + ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) + if !ok then goto quit + + odMC = CreateMatrixCurrency(todMatrix, "OD Trips", ,, ) + eaMC = CreateMatrixCurrency(todMatrix, "EA Trips", ,, ) + amMC = CreateMatrixCurrency(todMatrix, "AM Trips", ,, ) + mdMC = CreateMatrixCurrency(todMatrix, "MD Trips", ,, ) + pmMC = CreateMatrixCurrency(todMatrix, "PM Trips", ,, ) + evMC = CreateMatrixCurrency(todMatrix, "EV Trips", ,, ) + + odMC := (0.5 * dailyMC) + (0.5 * transMC) + + // - early AM + eaMC := 0.0235 * odMC + + + // - AM peak + amMC := 0.1000 * odMC + + + // - midday + mdMC := 0.5080 * odMC + + // - PM peak + pmMC := 0.1980 * odMC + + + // - evening + evMC := 0.1705 * odMC + + RunMacro("close all") + quit: + Return(ok) + +EndMacro diff --git a/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc b/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc new file mode 100644 index 0000000..9c74141 --- /dev/null +++ b/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc @@ -0,0 +1,254 @@ +/* + Create landuse skims(LUZ skims): + +About: + Creates LUZ skims from the following TAZ skims including length, time, and cost: + impdat_AM.mtx (Length (Skim), *STM_AM (Skim), dat_AM – itoll_AM) + impdat_MD.mtx (Length (Skim), *STM_MD (Skim), dat_MD – itoll_MD) + impmhdt_AM.mtx (Length (Skim), *STM_AM (Skim), mhdt – ITOLL2_AM) + impmhdt_MD.mtx (Length (Skim), *STM_MD (Skim), mhdt – ITOLL2_MD) + +Inputs: + 1) luzToTazSeries13.xls (Luz to TAZ reference) + 2) ExternalZones.xls (Luz internal to external reference) + 3) impdat_AM.mtx + 4) impdat_MD.mtx + 5) impmhdt_AM.mtx + 6) impmhdt_MD.mtx + +Outputs: + 1) impdat_AM.mtx (csv) + 2) impdat_MD.mtx (csv) + 3) impmhdt_AM.mtx (csv) + 4) impmhdt_MD.mtx (csv) + + +*/ + +Macro "Create LUZ Skims" + + shared path, inputDir, outputDir + + // Input Files + ext_luz_excel = inputDir+"\\ExternalZones.xls" + luz_taz_excel = inputDir+"\\luzToTazSeries13.xls" + + // Temp files + ext_luz_file = outputDir+"\\ExternalZones.bin" + luz_taz_file = outputDir+"\\luzToTazSeries13.bin" + tempfile = outputDir+"\\temp_luz.bin" + luzskims_bin = outputDir+"\\temp_luz_export.bin" + + // Convert excel to bin file + ExportExcel(ext_luz_excel, "FFB", ext_luz_file, ) + ExportExcel(luz_taz_excel, "FFB", luz_taz_file, ) + + // Open tables + luztaz_view = Opentable("luztaz", "FFB", {luz_taz_file}) + ext_luz_view = Opentable("luzIE", "FFB", {ext_luz_file}) + + /* ------------------------------------------------------------------------------------------------ + // Step 0: Prepares input and output files + --------------------------------------------------------------------------------------------------*/ + // Create list of LUZ I+E zones + LUZ_I = V2A(GetDataVector(luztaz_view+"|","luz_id",)) + LUZ_E = V2A(GetDataVector(ext_luz_view+"|","External LUZ",{{"Sort Order", {{"External LUZ", "Ascending"}}}})) + LUZ_IE = LUZ_I + LUZ_E + + // Get the maximum internal zone + max_LUZ_I = ArrayMax(LUZ_I) + min_LUZ_E = ArrayMin(LUZ_E) + max_LUZ_E = ArrayMax(LUZ_E) + + // Write the list to a file (temp luz zonal file) + luz_vw = CreateTable("luz", tempfile, "FFB",{ + {"luz_id", "Integer", 8, null, "Yes"}}) + SetView(view) + for r = 1 to LUZ_IE.length do + rh = AddRecord(luz_vw, { + {"luz_id", LUZ_IE[r]} + }) + end + + + period = {"AM","MD"} + vehicle = {"dat","mhdt"} + + for v =1 to vehicle.length do + for p= 1 to period.length do + + /* ------------------------------------------------------------------------------------------------ + //Step 1: Create matrix with LUZ internal and external Zones + --------------------------------------------------------------------------------------------------*/ + // Input taz skim + tazskims = outputDir+"\\imp"+vehicle[v]+"_"+period[p]+".mtx" + + // Output luz skim (mtx and csv files) + luzskims = outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".mtx" + luzskims_csv = outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".csv" + + // List of cores (same core names for inputs and outputs) + m = OpenMatrix(tazskims, ) + coreNames = GetMatrixCoreNames(m) + coreNames = Subarray(coreNames,2,3) // Only the distance, time, & toll skims + + // Open the temp luz zonal file + luz_view = Opentable("luz", "FFB", {tempfile}) + + // Create output matrix file with both LUZ internal & external zones + luz_mat =CreateMatrix({luz_view+"|", "luz_id", "All"}, + {luz_view+"|", "luz_id", "All"}, + {{"File Name", luzskims}, {"Type", "Float"}, {"Tables",coreNames}}) + + // Create LUZ II, IE/EI and EE indices + SetView(luz_view) + set_i = SelectByQuery("Internal", "Several", "Select * where luz_id <= "+ String(max_LUZ_I),) + Internal = CreateMatrixIndex("Internal", luz_mat, "Both", luz_view +"|Internal", "luz_id", "luz_id" ) + + set_e = SelectByQuery("External", "Several", "Select * where (luz_id >= "+ String(min_LUZ_E) +" & luz_id <= "+ String(max_LUZ_E)+")",) + External = CreateMatrixIndex("External", luz_mat, "Both", luz_view +"|External", "luz_id", "luz_id" ) + CloseView(luz_view) + + // Create luz currencies for II, EI and IE (each array has 3 currencies; one of each core) + mc_luz_II = CreateMatrixCurrencies(luz_mat,"Internal","Internal",) + mc_luz_EI = CreateMatrixCurrencies(luz_mat,"External","Internal",) + mc_luz_IE = CreateMatrixCurrencies(luz_mat,"Internal","External",) + mc_luz_EE = CreateMatrixCurrencies(luz_mat,"External","External",) + + /* ------------------------------------------------------------------------------------------------ + // Step 2: Create LUZ internal skims from TAZ skims + --------------------------------------------------------------------------------------------------*/ + // Create aggregate tables for each selected cores + for c = 1 to coreNames.length do + + // Create currency in the input file + mc = CreateMatrixCurrency(m, coreNames[c], , , ) + row_names = {"luztaz.taz", "luztaz.luz_id"} + col_names = {"luztaz.taz", "luztaz.luz_id"} + + // Create LUZ internal skims + tempLuzMtx = outputDir+"\\temp"+vehicle[v]+period[p]+"_"+String(c)+".mtx" + AggregateMatrix(mc, row_names, col_names, + {{"File Name", tempLuzMtx}, + {"Label", "LUZ"+coreNames[c]}, + {"File Based", "Yes"}}) + + // Add the aggregate table to the internal zones in the new core + mat = OpenMatrix(tempLuzMtx, ) + mc_temp = CreateMatrixCurrency(mat, coreNames[c], , , ) + mc_luz_II.(coreNames[c]):= mc_temp + + // Get the internal zones to the corresponding + intZones = GetDataVector(ext_luz_view+"|","Internal Cordon LUZ",) + extZones = GetDataVector(ext_luz_view+"|","External LUZ",) + distance = GetDataVector(ext_luz_view+"|","Miles to be Added to Cordon Point",) + time = GetDataVector(ext_luz_view+"|","Minutes to be Added to Cordon Point",) + + // Add the EI and IE mat values based on cordon data + for e = 1 to extZones.length do + // Get mat values for the corresponding internal zones + vec_EI = GetMatrixVector(mc_temp, {{"Row",intZones[e]}}) + vec_IE = GetMatrixVector(mc_temp, {{"Column",intZones[e]}}) + + // Add cordon time, distance (depending on the core) + if (c = 1) then do // length + vec_EI = vec_EI + distance[e] + vec_IE = vec_IE + distance[e] + end + if (c = 2) then do // time + vec_EI = vec_EI + time[e] + vec_IE = vec_IE + time[e] + end + + // Set vectors to the matrix + SetMatrixVector(mc_luz_EI.(coreNames[c]), vec_EI, {{"Row",extZones[e]}}) + SetMatrixVector(mc_luz_IE.(coreNames[c]), vec_IE, {{"Column",extZones[e]}}) + end // ext zones + + // EE values are filled based on II, and IE values + for i = 1 to extZones.length do + for j = 1 to extZones.length do + // Get II value if Intrazonal + if i=j then do + II_val = GetMatrixValue(mc_luz_II.(coreNames[c]), String(intZones[i]),String(intZones[j])) + SetMatrixValue(mc_luz_EE.(coreNames[c]),String(extZones[i]),String(extZones[j]),II_val) + end + else do + IE_val = GetMatrixValue(mc_luz_IE.(coreNames[c]), String(intZones[i]),String(extZones[j])) + if (c=1) then IE_val = IE_val + distance[i] + if (c=2) then IE_val = IE_val + time[i] + SetMatrixValue(mc_luz_EE.(coreNames[c]),String(extZones[i]),String(extZones[j]),IE_val) + end // if + end // j loop + end // i loop + end // cores + + /* ------------------------------------------------------------------------------------------------ + // Step 3: Export to CSV file + --------------------------------------------------------------------------------------------------*/ + // Export matrix values to a temp bin file + SetMatrixIndex(luz_mat, "All", "All") + CreateTableFromMatrix(luz_mat, luzskims_bin, "FFB", {{"Complete", "Yes"}}) + + // Add header and then export to csv file + export_vw = OpenTable("luztaz", "FFB", {luzskims_bin}) + strct = GetTableStructure(export_vw) + for s = 1 to strct.length do + strct[s] = strct[s] + {strct[s][1]} + end + + // Rename to first and second columns to Origin and Destination + strct[1][1] = "origin LUZ" + strct[2][1] = "destination LUZ" + ModifyTable(export_vw, strct) + + // Export to CSV file + ExportView(export_vw+"|", "CSV", luzskims_csv, null,{{"CSV Header", "True"}}) + CloseView(export_vw) + + end // time period + end // vehicle + + + /* ------------------------------------------------------------------------------------------------ + // Step 4: Close all views and delete temp files + --------------------------------------------------------------------------------------------------*/ + vws = GetViewNames() + if vws<> null then do + for w = 1 to vws.length do + CloseView(vws[w]) + end + end + + // Close matrices + mtxs = GetMatrices() + if mtxs <> null then do + handles = mtxs[1] + for m = 1 to handles.length do + handles[m] = null + end + end + + // Close rest + RunMacro("G30 File Close All") + + // Delete temp matrices + for v =1 to vehicle.length do + for p= 1 to period.length do + for c = 1 to coreNames.length do + DeleteFile(outputDir+"\\temp"+vehicle[v]+period[p]+"_"+String(c)+".mtx") + end + // Also delete tcad headers for the CSV file (as the csv files have headers) + DeleteFile(outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".DCC") + end + end + + // Delete temp luz files + delFiles = {"temp_luz.bin","temp_luz.BX","temp_luz.DCB","luzToTazSeries13.bin","luzToTazSeries13.DCB", + "ExternalZones.bin","ExternalZones.DCB","temp_luz_export.bin","temp_luz_export.DCB"} + for d =1 to delFiles.length do + info = GetFileInfo(outputDir+"\\"+delFiles[d]) + if info[1] <> null then DeleteFile(outputDir+"\\"+delFiles[d]) + end + +EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/createhwynet.rsc b/sandag_abm/src/main/gisdk/createhwynet.rsc new file mode 100644 index 0000000..b6c5cf5 --- /dev/null +++ b/sandag_abm/src/main/gisdk/createhwynet.rsc @@ -0,0 +1,2509 @@ +//******************************************************************** +//procedure to import e00 file to geo dbd file +//create highway network +//written on 4/19/01 +//macro "import highway layer", macro"fill oneway streets", +//macro "createhwynet1" +// +//input files: hwycov.e00 - hwy line layer ESRI exchange file +//output files: hwy.dbd - hwy line geographic file +// hwycad.log- a log file +// hwycad.err - error file with error info +//Oct 08, 2010: Added Lines 164-186, Create a copy of Toll fields +//Oct 08, 2010: Added Lines 284-287 Build Highway Network with ITOLL fields +//April 22, 2014: Wu checked all SR125 related changes are included +//Feb 02, 2016: Added reliability fields +//******************************************************************** + +macro "run create hwy" + shared path,inputDir,outputDir,mxzone + +/* exported highway layer is copied manually to the output folder (I15 SB toll entry/exit links are modified by RSG) + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","import highway layer"}) + ok=RunMacro("import highway layer") + if !ok then goto quit +*/ + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","copy highway database"}) + ok=RunMacro("copy database") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","fill oneway streets"}) + ok=RunMacro("fill oneway streets") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add TOD attributes"}) + ok=RunMacro("add TOD attributes") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","calculate distance to/from major interchange"}) + ok=RunMacro("DistanceToInterchange") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add reliability fields"}) + ok=RunMacro("add reliability fields") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add preload attributes"}) + ok=RunMacro("add preload attributes") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","Code VDF fields"}) + ok=RunMacro("Code VDF fields") + if !ok then goto quit + + RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","create hwynet"}) + ok=RunMacro("create hwynet") + if !ok then goto quit + + quit: + return(ok) +EndMacro + +/********************************************************************************************************** + import e00 file + if numofzone=tdz then e00file=hwycovtdz.e00, hwytdz.dbd + else e00file=hwy.e00, hwy.dbd + + Inputs: + input\turns.csv + input\turns.DCC + input\hwycov.e00 + + Outputs: + output\turns.dbf + output\hwytmp.dbd + output\hwy.dbd + +**********************************************************************************************************/ +macro "import highway layer" + shared path, inputDir,outputDir + ok=0 + + RunMacro("close all") + + di=GetDirectoryInfo(path + "\\tchc1.err","file") + if di.length>0 then do + ok=0 + RunMacro("TCB Error","chech tchc1.err file!") + goto exit + end + + di=GetDirectoryInfo(path + "\\tchc.err","file") + if di.length>0 then do + ok=0 + RunMacro("TCB Error","chech tchc.err file!") + goto exit + end + + di=GetDirectoryInfo(inputDir + "\\turns.csv","file") + if di.length=0 then do + ok=0 + RunMacro("TCB Error","turns.csv does not exist!") + goto exit + end + + // assume that the dictionary file exists in the input directory for the model run + //ok=RunMacro("SDcopyfile",{path_study+"\\data\\turns.DCC",path+"\\turns.DCC"}) + //if !ok then goto exit + + //export turns.csv to turns.dbf + vw = OpenTable("turns", "CSV", {inputDir+"\\turns.csv",}) + ExportView("turns|", "dbase", outputDir+"\\turns.dbf",,) + + // writeline(fpr,mytime+", exporting turns.csv to turns.dbf") + + // import e00 file + e00file="hwycov.e00" + + //check e00 file exists + di = GetDirectoryInfo(inputDir +"\\"+e00file, "File") + if di.length = 0 then do + ok=0 + RunMacro("TCB Error",e00file+" does not exist!") + goto exit + end + + ImportE00(inputDir +"\\"+e00file, outputDir + "\\hwytmp.dbd","line",outputDir + "\\hwytmp.bin",{ + {"Label","street line file"}, + {"Layer Name","hwyline"}, + {"optimize","True"}, + {"Median Split", "True"}, + {"Node Layer Name", "hwynode"}, + {"Node Table", outputDir + "\\hwytmp_.bin"}, + {"Projection","NAD83:406",{"_cdist=1000","_limit=1000","units=us-ft"}}, + }) + + //writeline(fpr,mytime+", importing e00 file") + + //export geo file by specify the line id field and the node id field + db_file=outputDir + "\\hwytmp.dbd" + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto exit + + allflds=Getfields(link_lyr,"All") + fullflds=allflds[2] + allnodeflds = GetFields(node_lyr, "All") + + // need to specify full field specifications + lineidfield = link_lyr+".hwycov-id" + nodeidfield = "hwynode.hnode"//for centroids purposes + + opts = {{"Layer Name", "hwyline"}, + {"File Type", "FFB"}, + {"ID Field", lineidfield}, + {"Field Spec", fullflds}, + {"Indexed Fields", {fullflds[1]}}, + {"Label", "street line file"}, + {"Node layer name","hwynode"}, + {"Node ID Field", nodeidfield}, + {"Node Field Spec", allnodeflds[2]}} + + if node_idx > 1 then + opts = opts + {{"Node ID Field", node_aflds[2][node_idx - 1]}} + hwy_db=outputDir + "\\hwy.dbd" + + exportgeography(link_lyr,hwy_db,opts) + + // writeline(fpr,mytime+", exporting e00 file") + + RunMacro("close all") //before delete db_file, close it + deleteDatabase(db_file) + + ok=1 + exit: + //if fpr<>null then closefile(fpr) + return(ok) +endMacro + +/* +copies database (hwy.dbd) and turns file (TURNS.DBF) + +this is required after edits made to the transcad highway database for I15 SB managed lane links + +*/ + +macro "copy database" + shared path, inputDir, outputDir + + hwy_db_in = inputDir + "\\hwy.dbd" + hwy_db_out = outputDir + "\\hwy.dbd" + + /// copye highway database + CopyDatabase(hwy_db_in, hwy_db_out) + + // copy turns file + CopyFile(inputDir+"\\TURNS.DBF", outputDir+"\\TURNS.DBF") + + ok=1 + return(ok) + +endMacro + +/********************************************************************************************************** + fill oneway street with dir field, and calculate toll fields and change AOC and add reliability factor + + Inputs + output\hwy.dbd + + Outputs: + output\hwy.dbd (modified) + + Adds fields to link layer (by period: _EA, _AM, _MD, _PM, _EV) + ITOLL - Toll + 10000 *[0,1] if SR125 toll lane + ITOLL2 - Toll + ITOLL3 - Toll + AOC + ITOLL4 - Toll * 1.03 + AOC + ITOLL5 - Toll * 2.33 + AOC + + Note: Link operation type (IHOV) where: + 1 = General purpose + 2 = 2+ HOV (Managed lanes if lanes > 1) + 3 = 3+ HOV (Managed lanes if lanes > 1) + 4 = Toll lanes + + + +**********************************************************************************************************/ +macro "fill oneway streets" + shared path, inputDir, outputDir + ok=0 + + RunMacro("close all") + + properties = "\\conf\\sandag_abm.properties" + aoc_f = RunMacro("read properties",properties,"aoc.fuel", "S") + aoc_m = RunMacro("read properties",properties,"aoc.maintenance", "S") + aoc=S2R(aoc_f)+S2R(aoc_m) + + db_file=outputDir+"\\hwy.dbd" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr = db_file + "|" + link_lyr + +// writeline(fpr,mytime+", fill one way streets") +// closefile(fpr) + +/* + //oneway streets, dir = 1 + Opts = null + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select * where iway = 1"} + Opts.Global.Fields = {"Dir"} + Opts.Global.Method = "Value" + Opts.Global.Parameter = {1} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit +*/ + //CHANGE SR125 TOLL SPEED TO 70MPH (ISPD=70) DELETE THIS SECTION AFTER TESTING + Opts = null + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ISPD"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = "if ihov=4 and IFC=1 then 70 else ISPD" + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + + // Create RELIABILITY OF FACILITY (TOLL) field + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"relifac", "Real", 10, 2, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + //change reliability field for SR125 to 0.65, and all other facilities are 0 + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"relifac"} + Opts.Global.Method = "Formula" +// Opts.Global.Parameter = "if ihov=4 & ifc=1 then 0.65 else 1" + //since we now have reliability fields, setting all reliability factors to 1 + Opts.Global.Parameter = "1" + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + //change AOC to appropriate value for year and cents per mile in COST field and add reliability factor to COST calc. + Opts = null + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"COST"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = "Length * "+R2S(aoc)+" * relifac" + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // Create copy of ITOLL fields to preserved original settings + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // changed field types to real (for I15 tolls) - by nagendra.dhakar@rsginc.com + strct = strct + {{"ITOLL2_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL2_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL2_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL2_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL2_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + tollfld={{"ITOLL2_EA"},{"ITOLL2_AM"},{"ITOLL2_MD"},{"ITOLL2_PM"},{"ITOLL2_EV"}} + tollfld_flg={{"ITOLLO"},{"ITOLLA"},{"ITOLLO"},{"ITOLLP"},{"ITOLLO"}} //note - change this once e00 file contains fields for each of 5 periods + + // set SR125 tolls + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = tollfld_flg[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // clear I15 tolls from previous step + // set other link tolls to 0 - creates a problem in skimming if left to null + // added by nagendra.dhakar@rsginc.com + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select *where ihov=2"} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Value" + Opts.Global.Parameter = {0} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + + // set I15 tolls - added by nagendra.dhakar@rsginc.com + RunMacro("set I15 tolls", link_lyr, tollfld) + + // Create ITOLL3 fields with ITOLL2A and COST + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"ITOLL3_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL3_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL3_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL3_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL3_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + tollfld={{"ITOLL3_EA"},{"ITOLL3_AM"},{"ITOLL3_MD"},{"ITOLL3_PM"},{"ITOLL3_EV"}} + tollfld_flg={{"ITOLL2_EA+COST"},{"ITOLL2_AM+COST"},{"ITOLL2_MD+COST"},{"ITOLL2_PM+COST"},{"ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = tollfld_flg[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // Create ITOLL4 fields with 1.03*(ITOLL2) and COST + // ITOLL4 = is applied to LHD and MHD only + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"ITOLL4_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL4_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL4_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL4_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL4_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + tollfld={{"ITOLL4_EA"},{"ITOLL4_AM"},{"ITOLL4_MD"},{"ITOLL4_PM"},{"ITOLL4_EV"}} + tollfld_flg={{"1.03*ITOLL2_EA+COST"},{"1.03*ITOLL2_AM+COST"},{"1.03*ITOLL2_MD+COST"},{"1.03*ITOLL2_PM+COST"},{"1.03*ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = tollfld_flg[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // Create ITOLL5 fields with 2.33*(ITOLL2) and COST + // ITOLL5 = is applied to HHD only + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"ITOLL5_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL5_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL5_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL5_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL5_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + tollfld={{"ITOLL5_EA"},{"ITOLL5_AM"},{"ITOLL5_MD"},{"ITOLL5_PM"},{"ITOLL5_EV"}} + tollfld_flg={{"2.33*ITOLL2_EA+COST"},{"2.33*ITOLL2_AM+COST"},{"2.33*ITOLL2_MD+COST"},{"2.33*ITOLL2_PM+COST"},{"2.33*ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = tollfld_flg[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // Create ITOLL fields + vw = GetView() + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + strct = strct + {{"ITOLL_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ITOLL_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + //adding $100 to toll fields to flag toll values from manage lane toll values in skim matrix + tollfld={{"ITOLL_EA"},{"ITOLL_AM"},{"ITOLL_MD"},{"ITOLL_PM"},{"ITOLL_EV"}} + tollfld_flg={{"if ihov=4 then ITOLL2_EA+10000 else ITOLL2_EA"}, + {"if ihov=4 then ITOLL2_AM+10000 else ITOLL2_AM"}, + {"if ihov=4 then ITOLL2_MD+10000 else ITOLL2_MD"}, + {"if ihov=4 then ITOLL2_PM+10000 else ITOLL2_PM"}, + {"if ihov=4 then ITOLL2_EV+10000 else ITOLL2_EV"}} //note - change this once e00 file contains fields for each of 5 periods + + // modified by nagendra.dhakar@rsginc.com to calculate every toll field from itoll2, which are set to the tolls fields in tcoved + + for i=1 to tollfld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tollfld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = tollfld_flg[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + RunMacro("close all") + + ok=1 + quit: + return(ok) +EndMacro + +/********************************************************************************************************* +add i-15 tolls by direction and period + + link ids and corresponding tolls are inputs + toll values are coded by link ids + tolls are determined by gate-to-gate toll optimization, solved using excel solver + tolls from two methods are used + -traversed links (NB PM and SB AM) + -entry and exit links (remaining) + +by: nagendra.dhakar@rsginc.com +**********************************************************************************************************/ + +Macro "set I15 tolls" (lyr, toll_fields) + shared path, inputDir, outputDir + + direction = {"NB","SB"} + periods={"EA","AM","MD","PM","EV"} + + toll_links = {} + tolls = {} + + // NB toll links and corresponding tolls + toll_links.NB = {} + + toll_links.NB.traverse = {29716,460,526,23044,459,463,512,464,469,470,510,29368,9808} + toll_links.NB.entryexit = {31143,29472,52505,52507,52508,475,34231,52511,52512,34229,34228,38793,29765,29766,52513,29764,26766} + + tolls.NB = {} + + tolls.NB.traverse = {} + tolls.NB.entryexit = {} + + // tolls are in cents + tolls.NB.entryexit.EA = {35.00,35.00,35.00,35.00,35.00,15.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00} + tolls.NB.entryexit.AM = {45.05,42.43,31.54,30.00,30.00,20.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,17.41,32.59,32.59,32.59} + tolls.NB.entryexit.MD = {69.91,73.91,70.42,66.12,51.61,0.00,26.88,25.00,25.00,25.00,12.07,37.93,47.51,3.35,46.65,60.66,65.74} + tolls.NB.traverse.PM = {21.83,31.11,50.00,55.34,113.23,50.00,50.00,50.00,50.00,50.00,50.00,50.00,0.00} + tolls.NB.entryexit.EV = {41.73,36.26,32.01,30.00,30.00,20.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,17.77,32.23,32.23,32.23} + + // SB toll links and corresponding tolls + toll_links.SB = {} +/* + // old network + toll_links.SB.traverse = {12193,25749,29442,23128,515,31204,520,22275,524,525,553,528,29415} + toll_links.SB.entryexit = {52514,38796,29768,38794,29763,52510,52509,52506,52510,34227,34233,29407,26398,29767,34226,34232,29471,52515} +*/ + // new network + toll_links.SB.traverse = {12193,25749,52567,23128,515,31204,52569,52550,524,525,52555,52559,52561,52565} + toll_links.SB.entryexit = {52568,52570,29768,38794,29763,52560,52562,52566,52556,34227,34233,29407,26398,52571,52572,29767,52575,52576,52574,34226,34232,29471,52573} + + tolls.SB = {} + + tolls.SB.traverse = {} + tolls.SB.entryexit = {} +/* + // old network + tolls.SB.entryexit.EA = {26.69,25.54,26.96,25.54,39.23,25.54,24.46,25.54,25.54,25.54,24.46,24.46,36.30,23.31,24.46,24.46,28.04,0.00} + tolls.SB.traverse.AM = {0.00,50.00,50.00,89.82,50.00,50.00,63.74,0.00,0.11,76.40,38.80,63.58,83.28} + tolls.SB.entryexit.MD = {26.39,25.00,25.00,25.00,35.00,25.47,22.74,27.26,25.47,25.00,24.53,25.00,35.61,23.61,25.00,25.00,32.65,0.00} + tolls.SB.entryexit.PM = {25.00,25.00,25.00,25.00,35.00,25.00,24.34,25.66,25.00,25.00,25.00,25.00,35.00,25.00,25.00,25.00,26.69,0.00} + tolls.SB.entryexit.EV = {25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,35.00,25.00,25.00,25.00,25.00,0.00} +*/ + // new network + tolls.SB.traverse.AM = {0.00,59.74,50.00,80.42,50.00,50.00,69.24,0.00,3.18,50.00,50.17,19.06,50.00,84.26} + tolls.SB.entryexit.EA = {25.00,25.00,25.00,25.00,35.00,7.29,7.29,7.29,17.30,25.00,17.30,17.30,35.00,15.00,25.00,25.00,32.70,42.71,32.70,25.00,25.00,42.71,25.00} + tolls.SB.entryexit.MD = {32.80,25.00,25.00,25.00,32.80,11.43,11.43,10.65,18.85,25.00,18.85,18.85,32.80,17.20,25.00,17.20,31.15,38.57,31.15,25.00,25.00,39.35,25.00} + tolls.SB.entryexit.PM = {27.78,25.00,25.00,25.00,35.00,12.67,12.67,12.67,19.36,25.00,19.36,19.36,35.00,15.00,25.00,22.22,30.64,37.33,30.64,25.00,25.00,37.33,25.00} + tolls.SB.entryexit.EV = {29.12,25.00,25.00,25.00,35.00,13.14,13.14,13.14,21.56,25.00,21.56,21.56,35.00,15.00,25.00,20.88,28.44,36.86,28.44,25.00,25.00,36.86,25.00} + + for dir=1 to 2 do + for per=1 to periods.length do + // locate record + + if (direction[dir]="NB" and periods[per] = "PM") or (direction[dir]="SB" and periods[per] = "AM") then method = "traverse" + else method = "entryexit" + + links_array = toll_links.(direction[dir]).(method) + tolls_array = tolls.(direction[dir]).(method).(periods[per]) + + // set toll values + for i=1 to links_array.length do + record_handle = LocateRecord (lyr+"|", "ID", {links_array[i]},{{"Exact", "True"}}) + SetRecordValues(lyr, record_handle, {{toll_fields[per][1],tolls_array[i]}}) + end + end + end + +EndMacro + +/********************************************************************************************************** + add link attributes for tod periods + + Inputs + output\hwy.dbd + + Outputs: + output\hwy.dbd (modified) + +**********************************************************************************************************/ +Macro "add TOD attributes" + + shared path, inputDir, outputDir + ok=0 + + db_file=outputDir+"\\hwy.dbd" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr = db_file + "|" + link_lyr + + vw = SetView(link_lyr) + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // AB Link capacity + strct = strct + {{"ABCP_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCP_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCP_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCP_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCP_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Link capacity + strct = strct + {{"BACP_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACP_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACP_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACP_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACP_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Intersection capacity + strct = strct + {{"ABCX_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCX_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCX_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCX_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCX_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Intersection capacity + strct = strct + {{"BACX_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACX_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACX_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACX_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACX_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Link time + strct = strct + {{"ABTM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTM_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Link time + strct = strct + {{"BATM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATM_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Intersection time + strct = strct + {{"ABTX_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTX_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTX_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTX_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABTX_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Intersection time + strct = strct + {{"BATX_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATX_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATX_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATX_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BATX_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Lanes + strct = strct + {{"ABLN_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLN_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLN_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLN_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLN_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Lanes + strct = strct + {{"BALN_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALN_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALN_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALN_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALN_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Drive-alone cost + strct = strct + {{"ABSCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Drive-alone cost + strct = strct + {{"BASCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Shared 2 cost + strct = strct + {{"ABH2CST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH2CST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH2CST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH2CST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH2CST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Shared 2 cost + strct = strct + {{"BAH2CST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH2CST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH2CST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH2CST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH2CST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Shared-3 cost + strct = strct + {{"ABH3CST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH3CST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH3CST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH3CST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABH3CST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Shared-3 cost + strct = strct + {{"BAH3CST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH3CST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH3CST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH3CST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAH3CST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Light-Heavy truck cost + strct = strct + {{"ABLHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Light-Heavy truck cost + strct = strct + {{"BALHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Medium-Heavy truck cost + strct = strct + {{"ABMHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABMHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABMHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABMHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABMHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Medium-Heavy truck cost + strct = strct + {{"BAMHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAMHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAMHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAMHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAMHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Heavy-Heavy truck cost + strct = strct + {{"ABHHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Heavy-Heavy truck cost + strct = strct + {{"BAHHCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHHCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHHCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHHCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHHCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB Commercial vehicle cost + strct = strct + {{"ABCVCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCVCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCVCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCVCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABCVCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Commercial vehicle cost + strct = strct + {{"BACVCST_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACVCST_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACVCST_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACVCST_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BACVCST_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB SOV Time + strct = strct + {{"ABSTM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTM_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA SOV Time + strct = strct + {{"BASTM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTM_EV", "Real", 14, 6, "True", , , , , , , null}} + + // AB HOV Time + strct = strct + {{"ABHTM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHTM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHTM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHTM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABHTM_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA HOV Time + strct = strct + {{"BAHTM_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHTM_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHTM_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHTM_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BAHTM_EV", "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + // initialize time and cost fields to 999999 + tod_fld = {{"ABSCST_EA"},{"ABSCST_AM"},{"ABSCST_MD"},{"ABSCST_PM"},{"ABSCST_EV"}, + {"BASCST_EA"},{"BASCST_AM"},{"BASCST_MD"},{"BASCST_PM"},{"BASCST_EV"}, + {"ABH2CST_EA"},{"ABH2CST_AM"},{"ABH2CST_MD"},{"ABH2CST_PM"},{"ABH2CST_EV"}, + {"BAH2CST_EA"},{"BAH2CST_AM"},{"BAH2CST_MD"},{"BAH2CST_PM"},{"BAH2CST_EV"}, + {"ABH3CST_EA"},{"ABH3CST_AM"},{"ABH3CST_MD"},{"ABH3CST_PM"},{"ABH3CST_EV"}, + {"BAH3CST_EA"},{"BAH3CST_AM"},{"BAH3CST_MD"},{"BAH3CST_PM"},{"BAH3CST_EV"}, + {"ABSTM_EA"},{"ABSTM_AM"},{"ABSTM_MD"},{"ABSTM_PM"},{"ABSTM_EV"}, + {"BASTM_EA"},{"BASTM_AM"},{"BASTM_MD"},{"BASTM_PM"},{"BASTM_EV"}, + {"ABHTM_EA"},{"ABHTM_AM"},{"ABHTM_MD"},{"ABHTM_PM"},{"ABHTM_EV"}, + {"BAHTM_EA"},{"BAHTM_AM"},{"BAHTM_MD"},{"BAHTM_PM"},{"BAHTM_EV"}, + {"ABLHCST_EA"},{"ABLHCST_AM"},{"ABLHCST_MD"},{"ABLHCST_PM"},{"ABLHCST_EV"}, + {"BALHCST_EA"},{"BALHCST_AM"},{"BALHCST_MD"},{"BALHCST_PM"},{"BALHCST_EV"}, + {"ABMHCST_EA"},{"ABMHCST_AM"},{"ABMHCST_MD"},{"ABMHCST_PM"},{"ABMHCST_EV"}, + {"BAMHCST_EA"},{"BAMHCST_AM"},{"BAMHCST_MD"},{"BAMHCST_PM"},{"BAMHCST_EV"}, + {"ABHHCST_EA"},{"ABHHCST_AM"},{"ABHHCST_MD"},{"ABHHCST_PM"},{"ABHHCST_EV"}, + {"BAHHCST_EA"},{"BAHHCST_AM"},{"BAHHCST_MD"},{"BAHHCST_PM"},{"BAHHCST_EV"}, + {"ABCVCST_EA"},{"ABCVCST_AM"},{"ABCVCST_MD"},{"ABCVCST_PM"},{"ABCVCST_EV"}, + {"BACVCST_EA"},{"BACVCST_AM"},{"BACVCST_MD"},{"BACVCST_PM"},{"BACVCST_EV"} + } + + // now calculate fields + for i=1 to tod_fld.length do + + calcString = {"999999"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tod_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + + // set capacity fields + tod_fld ={{"ABCP_EA"},{"ABCP_AM"},{"ABCP_MD"},{"ABCP_PM"},{"ABCP_EV"}, //BA link capacity + {"BACP_EA"},{"BACP_AM"},{"BACP_MD"},{"BACP_PM"},{"BACP_EV"}, //AB link capacity + {"ABCX_EA"},{"ABCX_AM"},{"ABCX_MD"},{"ABCX_PM"},{"ABCX_EV"}, //BA intersection capacity + {"BACX_EA"},{"BACX_AM"},{"BACX_MD"},{"BACX_PM"},{"BACX_EV"}} //AB intersection capacity + + org_fld ={"ABCPO","ABCPA","ABCPO","ABCPP","ABCPO", + "BACPO","BACPA","BACPO","BACPP","BACPO", + "ABCXO","ABCXA","ABCXO","ABCXP","ABCXO", + "BACXO","BACXA","BACXO","BACXP","BACXO"} + + factor ={"3/12","1","6.5/12","3.5/3","8/12", + "3/12","1","6.5/12","3.5/3","8/12", + "3/12","1","6.5/12","3.5/3","8/12", + "3/12","1","6.5/12","3.5/3","8/12"} + + // now calculate capacity + for i=1 to tod_fld.length do + + calcString = {"if "+org_fld[i]+ " != 999999 then " + factor[i] + " * " + org_fld[i] + " else 999999"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tod_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + + // set time fields + tod_fld ={{"ABTM_EA"},{"ABTM_AM"},{"ABTM_MD"},{"ABTM_PM"},{"ABTM_EV"}, //BA link time + {"BATM_EA"},{"BATM_AM"},{"BATM_MD"},{"BATM_PM"},{"BATM_EV"}, //AB link time + {"ABTX_EA"},{"ABTX_AM"},{"ABTX_MD"},{"ABTX_PM"},{"ABTX_EV"}, //BA intersection time + {"BATX_EA"},{"BATX_AM"},{"BATX_MD"},{"BATX_PM"},{"BATX_EV"}} //AB intersection time + + org_fld ={"ABTMO","ABTMA","ABTMO","ABTMP","ABTMO", + "BATMO","BATMA","BATMO","BATMP","BATMO", + "ABTXO","ABTXA","ABTXO","ABTXP","ABTXO", + "BATXO","BATXA","BATXO","BATXP","BATXO"} + + factor ={"1","1","1","1","1", + "1","1","1","1","1", + "1","1","1","1","1", + "1","1","1","1","1"} + + // now calculate time + for i=1 to tod_fld.length do + + calcString = { factor[i] + " * " + org_fld[i]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tod_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // set lane fields + tod_fld ={{"ABLN_EA"},{"ABLN_AM"},{"ABLN_MD"},{"ABLN_PM"},{"ABLN_EV"}, //AB lanes + {"BALN_EA"},{"BALN_AM"},{"BALN_MD"},{"BALN_PM"},{"BALN_EV"}} //BA lanes + + org_fld ={"ABLNO","ABLNA","ABLNO","ABLNP","ABLNO", + "BALNO","BALNA","BALNO","BALNP","BALNO"} + + factor ={"1","1","1","1","1", + "1","1","1","1","1"} + + // now calculate time + for i=1 to tod_fld.length do + + calcString = { factor[i] + " * " + org_fld[i]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = tod_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + RunMacro("close all") + + ok=1 + quit: + return(ok) +EndMacro + +/********************************************************************************************************** + Add link fields for reliability + + v/c factor fields: + {"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, + {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}, + + static reliability fields: + {"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, + {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"} + + interchange fields - used in static reliability calculations + {"INTDIST_UP"},{"INTDIST_DOWN"} + + Regression equations for static reliability: + + static reliability(freeway) = intercept + coeff1*ISPD70 + coeff2*1/MajorUpstream + coeff3*1/MajorDownstream + static reliability(arterial) = intercept + coeff1*NumLanesOneLane + coeff2*NumLanesCatTwoLane + coeff3*NumLanesCatThreeLane + coeff4*NumLanesCatFourLanes + coeff5*NumLanesFiveMoreLane + + coeff6*ISPD.CatISPD35Less + coeff7*ISPD.CatISPD35 + coeff8*ISPD.CatISPD40 + coeff9*ISPD.CatISPD45 + coeff1*ISPD.CatISPD50 + coeff10*ISPD.CatISPD50More + + coeff11*ICNT.EstSignal + coeff12*ICNT.EstStop + coeff13*ICNT.EstRailRoad + + Where; + ISPD70: 1 if ISPD=70 else 0 (ISPD is posted speed) + MajorUpstream: distance to major interchange upstream (miles) + MajorDownstream: distance to major interchange downstream (miles) + NumLanesOneLane: 1 if lane=1 else 0 + NumLanesCatTwoLane: 1 if lane=2 else 0 + NumLanesCatThreeLane: 1 if lane=3 else 0 + NumLanesCatFourLanes: 1 if lane=4 else 0 + NumLanesFiveMoreLane: 1 if lane>=5 else 0 + ISPD.CatISPD35Less: 1 if ISPD <35 else 0 + ISPD.CatISPD35: 1 if ISPD =35 else 0 + ISPD.CatISPD40: 1 if ISPD =40 else 0 + ISPD.CatISPD45: 1 if ISPD =45 else 0 + ISPD.CatISPD50: 1 if ISPD =50 else 0 + ISPD.CatISPD50More: 1 if ISPD >=50 else 0 + ICNT.EstSignal: 1 if ICNT=1 else 0 (ICNT is intersection control type); signal-controlled + ICNT.EstStop: 1 if ICNT=2 or ICNT=3 else 0; stop-controlled + ICNT.EstRailRoad: 1 if ICNT>3 else 0; other - railroad etc. + + Steps: + 1. add new fields + 2. populate with default values + 3. calculate v/c factor fields by setting them to estimated coefficients by facility type - freeway and arterial. Ramp and other use arterial coefficients. + 4. pupulate interchange fields by joining highway database with major interchange distance file (output from distance to interchange macro). + 5. calculate static reliability fields for freeway + 6. calculate static reliability fields for arterial, ramp, and other + + Inputs + output\hwy.dbd + output\MajorInterchangeDistance.csv + + Outputs: + output\hwy.dbd (modified) + +by: nagendra.dhakar@rsginc.com +**********************************************************************************************************/ +Macro "add reliability fields" + + shared path, inputDir, outputDir + ok=0 + + db_file=outputDir+"\\hwy.dbd" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr = db_file + "|" + link_lyr + + vw = SetView(link_lyr) + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // **** step 1. add new fields + + // AB v/c factors + strct = strct + {{"ABLOSC_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLOSD_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLOSE_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLOSFL_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABLOSFH_FACT", "Real", 14, 6, "True", , , , , , , null}} + + // BA v/c factors + strct = strct + {{"BALOSC_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALOSD_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALOSE_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALOSFL_FACT", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BALOSFH_FACT", "Real", 14, 6, "True", , , , , , , null}} + + // AB Static Reliability + strct = strct + {{"ABSTATREL_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTATREL_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTATREL_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTATREL_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ABSTATREL_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA Static Reliability + strct = strct + {{"BASTATREL_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTATREL_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTATREL_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTATREL_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BASTATREL_EV", "Real", 14, 6, "True", , , , , , , null}} + + // interchange distance + strct = strct + {{"INTDIST_UP", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"INTDIST_DOWN", "Real", 14, 6, "True", , , , , , , null}} + + // AB total reliability + strct = strct + {{"AB_TOTREL_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_TOTREL_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_TOTREL_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_TOTREL_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_TOTREL_EV", "Real", 14, 6, "True", , , , , , , null}} + + // BA total reliability + strct = strct + {{"BA_TOTREL_EA", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_TOTREL_AM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_TOTREL_MD", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_TOTREL_PM", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_TOTREL_EV", "Real", 14, 6, "True", , , , , , , null}} + + // reliability fields + reliability_fld = {{"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, + {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}, + {"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, + {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"}, + {"AB_TOTREL_EA"},{"AB_TOTREL_AM"},{"AB_TOTREL_MD"},{"AB_TOTREL_PM"},{"AB_TOTREL_EV"}, + {"BA_TOTREL_EA"},{"BA_TOTREL_AM"},{"BA_TOTREL_MD"},{"BA_TOTREL_PM"},{"BA_TOTREL_EV"}} + + ModifyTable(view1, strct) + + // for debug + //RunMacro("TCB Init") + + // **** step 2. populate with default value of 0 + for i=1 to reliability_fld.length do + + calcString = {"0"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = reliability_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // **** step 3. calculate v/c factor fields + + los_fld ={{"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, //BA link time + {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}} //AB intersection time + + factor_freeway ={"0.2429","0.1705","-0.2278","-0.1983","1.022", + "0.2429","0.1705","-0.2278","-0.1983","1.022"} + + factor_arterial ={"0.1561","0.0","0.0","-0.1449","0", + "0.1561","0.0","0.0","-0.1449","0"} + + facility_type = {"freeway","arterial","ramp","other"} // freeway (IFC=1), arterial (IFC=2,3), ramp (IFC=8,9), other (IFC=4,5,6,7) + + // lower and upper bounds of IFC for respective facility type = {freeway, arterial, ramp, other} + lwr_bound = {"1","2","8","4"} + upr_bound = {"1","3","9","7"} + + // now calculate v/c factor fields + for fac_type=1 to facility_type.length do + + // set factors (coefficients) for facility type + if fac_type=1 then factor=factor_freeway + else factor=factor_arterial + + for i=1 to los_fld.length do + + query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = los_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = factor[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + end + + // **** step 4. populate interchange fields (upstream/downstream distance to major interchange) + + distance_file = outputDir+"\\MajorInterchangeDistance.csv" + distance_fld = {"updistance","downdistance"} + + // interchange distance fields + interchange_fld = {{"INTDIST_UP"},{"INTDIST_DOWN"}} + + // set initial value to 9999 + for i=1 to interchange_fld.length do + + calcString = {"9999"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = interchange_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // now set to distances - in miles + for i=1 to interchange_fld.length do + Opts = null + Opts.Input.[Dataview Set] = {{db_link_lyr, distance_file,{"ID"},{"LinkID"}},"JoinedView"} + Opts.Global.Fields = interchange_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = distance_fld[i] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + // **** step 5. calculate static reliability fields for freeway + + // static reliability(freeway) = intercept + coeff1*ISPD70 + coeff2*1/MajorUpstream + coeff3*1/MajorDownstream + + static_fld = {{"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, + {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"}} + + // Freeway coefficients + intercept = {"0.1078"} + speed_factor = {"0.01393"} // ISPD70 + interchange_factor = {"0.011","0.0005445"} //MajorUpstream.Inverse, MajorDownstream.Inverse + + fac_type=1 + factor=factor_freeway + + for i=1 to static_fld.length do + query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] + + // intercept + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = intercept[1] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // ISPD - add to intercept + calcString = {"if ISPD=70 then " + static_fld[i][1] + "+" +speed_factor[1] + " else " + static_fld[i][1]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // Upstream interchange distance - apply inverse and add to intercept and ISPD + for j=1 to interchange_factor.length do + + calcString = {"if " + interchange_fld[j][1] + "<>null then " + static_fld[i][1] + "+" +interchange_factor[j]+"*1/"+interchange_fld[j][1] + " else " + static_fld[i][1]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + + end + + // **** step 6. calculate static reliability fields for arterial, ramp, and other + + // Factors (coefficients) + intercept = {"0.0546552"} + lane_factor = {"0.0","0.0103589","0.0361211","0.0446958","0.0"} //{NumLanesOneLane, NumLanesCatTwoLane, NumLanesCatThreeLane, NumLanesCatFourLanes, NumLanesFiveMoreLane} + speed_factor = {"0.0","0.0075674","0.0091012","0.0080996","-0.0022938","-0.0046211"} //{ISPD.CatISPD35Less (base), ISPD.CatISPD35, ISPD.CatISPD40, ISPD.CatISPD45, ISPD.CatISPD50, ISPD.CatISPD50More} + intersection_factor = {"0.0030973","-0.0063281","0.0127692"} //{ICNT.EstSignal, ICNT.EstStop, ICNT.EstRailRoad} + + // lane fields in network + lane_fld ={{"ABLN_EA"},{"ABLN_AM"},{"ABLN_MD"},{"ABLN_PM"},{"ABLN_EV"}, //AB lanes + {"BALN_EA"},{"BALN_AM"},{"BALN_MD"},{"BALN_PM"},{"BALN_EV"}} //BA lanes + + // intersection fields in network + intersection_fld = {{"ABCNT"},{"ABCNT"},{"ABCNT"},{"ABCNT"},{"ABCNT"}, + {"BACNT"},{"BACNT"},{"BACNT"},{"BACNT"},{"BACNT"}} + + // static reliability(arterial) = intercept + coeff1*NumLanesOneLane + coeff2*NumLanesCatTwoLane + coeff3*umLanesCatThreeLane + coeff4*NumLanesCatFourLanes + coeff5*NumLanesFiveMoreLane+ + // coeff6*ISPD.CatISPD35Less + coeff7*ISPD.CatISPD35 + coeff8*ISPD.CatISPD40 + coeff9*ISPD.CatISPD45 + coeff1*ISPD.CatISPD50 + coeff10*ISPD.CatISPD50More+ + // coeff11*ICNT.EstSignal + coeff12*ICNT.EstStop + coeff13*ICNT.EstRailRoad + + for fac_type=2 to facility_type.length do + + // selection query - to identify links with a facility type + query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] + + for i=1 to static_fld.length do + + // intercept + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = intercept[1] + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // NumLanes Factors + calcString = {"if " + lane_fld[i][1] + "=1 then "+static_fld[i][1] + "+" +lane_factor[1]+ + " else if " + lane_fld[i][1] + "=2 then "+static_fld[i][1] + "+" +lane_factor[2]+ + " else if " + lane_fld[i][1] + "=3 then "+static_fld[i][1] + "+" +lane_factor[3]+ + " else if " + lane_fld[i][1] + "=4 then "+static_fld[i][1] + "+" +lane_factor[4]+ + " else "+static_fld[i][1] + "+" +lane_factor[5]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // Speed Factors - + calcString = {"if ISPD<35 then "+static_fld[i][1] + "+" +speed_factor[1]+ + " else if ISPD=35 then "+static_fld[i][1] + "+" +speed_factor[2]+ + " else if ISPD=40 then "+static_fld[i][1] + "+" +speed_factor[3]+ + " else if ISPD=45 then "+static_fld[i][1] + "+" +speed_factor[4]+ + " else if ISPD=50 then "+static_fld[i][1] + "+" +speed_factor[5]+ + " else "+static_fld[i][1] + "+" +speed_factor[6]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // Intersection Factors + calcString = {"if " + intersection_fld[i][1]+ "=1 then " + static_fld[i][1] + "+" +intersection_factor[1]+ + " else if " + intersection_fld[i][1]+ "=2 or " + intersection_fld[i][1]+ "=3 then " + static_fld[i][1] + "+" + intersection_factor[2]+ + " else if " + intersection_fld[i][1]+ ">3 then " + static_fld[i][1] + "+" + intersection_factor[3] + + " else " + static_fld[i][1]} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} + Opts.Global.Fields = static_fld[i] + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + end + end + + RunMacro("close all") + + ok=1 + quit: + return(ok) +EndMacro +/************************************************************************************************ +add preload attributes + +Adds fields to highway line layer for storing preload volumes (currently bus volumes) + +************************************************************************************************/ +Macro "add preload attributes" + + shared path, inputDir, outputDir + + db_file=outputDir+"\\hwy.dbd" + + periods={"_EA","_AM","_MD","_PM","_EV"} + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr = db_file + "|" + link_lyr + + vw = SetView(link_lyr) + strct = GetTableStructure(vw) + + // Copy the current name to the end of strct + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // Add fields to the output table + new_struct = strct + { + {"ABPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}} + + // Modify table structure + ModifyTable(vw, new_struct) + + // initialize to 0 + for i = 1 to periods.length do + + ABField = "ABPRELOAD"+periods[i] + BAField = "BAPRELOAD"+periods[i] + + //initialize to 0 + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {ABField} + Opts.Global.Method = "Value" + Opts.Global.Parameter = {0} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {BAField} + Opts.Global.Method = "Value" + Opts.Global.Parameter = {0} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + end + + + RunMacro("close all") + + ok=1 + quit: + return(ok) + +EndMacro +/************************************************************************************************** + +Macro Code VDF fields + +This macro codes fields for the Tucson volume-delay function on the input highway line layer. It +should be run prior to constructing highway networks for skim-building & assignment. Eventually +the logic in this macro will be replaced by GIS code. + +Functional class (IFC) + + 1 = Freeway + 2 = Prime arterial + 3 = Major arterial + 4 = Collector + 5 = Local collector + 6 = Rural collector + 7 = Local (non Circulation Element) road + 8 = Freeway connector ramps + 9 = Local ramps + 10 = Zone connectors + +Cycle length matrix + + Intersecting Link +Approach Link 2 3 4 5 6 7 8 9 +IFC Description Prime Arterial Major Arterial Collector Local Collector Rural Collector Local Road Freeway connector Local Ramp +2 Prime Arterial 2.5 2 2 2 2 2 2 2 +3 Major Arterial 2 2 2 2 2 2 2 2 +4 Collector 2 2 1.5 1.5 1.5 1.5 1.5 1.5 +5 Local Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 +6 Rural Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 +7 Local Road 2 2 1.5 1.25 1.25 1.25 1.25 1.25 +8 Freeway connector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 +9 Local Ramp 2 2 1.5 1.25 1.25 1.25 1.25 1.25 + +Ramp with meter (abcnt = 4 or 5) + Cycle length = 2.5 + GC ratio 0.42 (adjusted down from 0.5 for yellow) + +Stop controlled intersection + Cycle length =1.25 + GC ratio 0.42 (adjusted down from 0.5 for yellow) + +*************************************************************************************************/ +Macro "Code VDF fields" + shared path, inputDir, outputDir + + db_file=outputDir+"\\hwy.dbd" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr = db_file + "|" + link_lyr + + + // Add AB_Cycle, AB_PF, BA_Cycle, and BA_PF + vw = SetView(link_lyr) + strct = GetTableStructure(vw) + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + strct = strct + {{"AB_GCRatio", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_GCRatio", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_Cycle", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_Cycle", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_PF", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_PF", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ALPHA1", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BETA1", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ALPHA2", "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BETA2", "Real", 14, 6, "True", , , , , , , null}} + ModifyTable(view1, strct) + + // Now set the view to the node layer + SetLayer(node_lyr) + + nodes = GetDataVector(node_lyr+"|", "ID",) + + //create from/end node field in the line layer + start_fld = CreateNodeField(link_lyr, "start_node", node_lyr+".ID", "From", ) + end_fld = CreateNodeField(link_lyr, "end_node", node_lyr+".ID", "To", ) + + //get the count of records for both line and node layer + tot_line_count = GetRecordCount(link_lyr, ) //get number of records in linelayer + tot_node_count = GetRecordCount(node_lyr, ) //get number of records in nodelayer + + //initilize several vectors +// linkclass = {Vector(tot_line_count, "Float", {{"Constant", null}}),Vector(tot_line_count, "Float", {{"Constant", null}})} //line link-classes + + //pass the attibutes to the vectors +// linkclass = GetDataVector(link_lyr+"|", "IFC",) + +// lineId = GetDataVector(link_lyr+"|", "ID",) + + + //go node by node + for i = 1 to tot_node_count do + + //find the IDs of the links that connect at the node, put the effective links to vector 'link_list' + link_list = null + rec_handles = null + + all_link_list = GetNodeLinks(nodes[i]) //get all links connecting at this node + link_list = Vector(all_link_list.length, "Short", {{"Constant", 0}}) //to contain non-connector/ramp links coming to the node + + all_rec_handles = Vector(all_link_list.length, "String", {{"Constant", null}}) //to contain the record handle of all links coming to the node + rec_handles = Vector(all_link_list.length, "String", {{"Constant", null}}) //to contain the record handle of the non-connector/ramp links coming to the node + + //count how many links entering the node + link_count = 0 + two_oneway = 0 + signal = 0 + + for j = 1 to all_link_list.length do + record_handle = LocateRecord(link_lyr+"|","ID",{all_link_list[j]}, {{"Exact", "True"}}) + all_rec_handles[j] = record_handle + + ends_at_node_AB_direction = 0 + ends_at_node_BA_direction = 0 + + signal_at_end = 0 + + if(link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_AB_direction = 1 + if(link_lyr.start_node = nodes[i] and link_lyr.Dir = -1) then ends_at_node_AB_direction = 1 + if(link_lyr.start_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_BA_direction = 1 + + if ( ends_at_node_AB_direction = 1 and (link_lyr.[ABGC] <> null and link_lyr.[ABGC] > 0)) then signal_at_end = 1 + if ( ends_at_node_BA_direction = 1 and (link_lyr.[BAGC] <> null and link_lyr.[BAGC] > 0)) then signal_at_end = 1 + + + //only count the links that have approach toward the node + if( ends_at_node_AB_direction = 1 and signal_at_end = 1) then do + signal = signal + 1 + end + else if( ends_at_node_BA_direction and signal_at_end = 1) then do + signal = signal + 1 + end + + link_count = link_count + 1 + link_list[link_count] = all_link_list[j] + rec_handles[link_count] = record_handle + if link_lyr.Dir <> 0 then two_oneway = two_oneway+1 + + + end + + // if at least one incoming link has a gc ratio + if (signal>0) then do + min_lc = 999 + max_lc = 0 + //process the links and find the lowest and highest linkclasses + for j = 1 to link_count do + //find the line record that owns the line ID + SetRecord(link_lyr, rec_handles[j]) //set the current record with the record handle stored in vector 'rec_handles' + + if ((link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) or + (link_lyr.start_node = nodes[i] and link_lyr.Dir = -1))then do + if link_lyr.[IFC] <> null and link_lyr.[IFC] > 1 and link_lyr.[IFC] < 10 then do //don't count freeways or centroid connectors + if link_lyr.[IFC] > max_lc then max_lc = link_lyr.[IFC] + if link_lyr.[IFC] < min_lc then min_lc = link_lyr.[IFC] + end + end + + end + end + + //iterate through all links at this node and set cycle length + for j = 1 to all_link_list.length do + + SetRecord(link_lyr, all_rec_handles[j]) //set the current record with the record handle stored in vector 'all_rec_handles' + + if(link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_AB_direction = 1 + if(link_lyr.start_node = nodes[i] and link_lyr.Dir = -1) then ends_at_node_AB_direction = 1 + if(link_lyr.start_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_BA_direction = 1 + + // Set AB fields for links whose end node is this node and are coded in the A->B direction + if (ends_at_node_AB_direction = 1) then do + + //defaults are 1.25 minute cycle length and 1.0 progression factor + c_len = 1.25 + p_factor = 1.0 + + //set up the cycle length for AB direction if there is a gc ratio and more than 2 links + if (link_lyr.[ABGC]<>0 and signal > 0) then do + + if (link_lyr.[IFC] = 2) then do + if (max_lc = 2) then c_len = 2.5 //Prime arterial & Prime arterial + else c_len = 2.0 //Prime arterial & anything lower + end + else if (link_lyr.[IFC] = 3) then do + if (max_lc > 3) then c_len = 2.0 //Major arterial & anything lower than a Major arterial + else c_len = 2.0 //Major arterial & Prime arterial or Major arterial + end + else if (link_lyr.[IFC] = 4) then do + if (min_lc < 4) then c_len = 2.0 //Anything lower than a Major arterial & Prime arterial + else c_len = 1.5 //Anything lower than a Major arterial & anything lower than a Prime arterial + end + else if (link_lyr.[IFC] > 4) then do + if (min_lc < 4) then c_len = 2.0 + if (min_lc = 4) then c_len = 1.5 + if (min_lc > 4) then c_len = 1.25 + end + + //update attributes + if( link_lyr.[ABGC] > 10) then link_lyr.[AB_GCRatio] = link_lyr.[ABGC]/100 + if( link_lyr.[AB_GCRatio] > 1.0) then link_lyr.[AB_GCRatio] = 1.0 + + link_lyr.[AB_Cycle] = c_len + link_lyr.[AB_PF] = p_factor + + end + + end // end for AB links + + // Set BA fields for links whose start node is this node and are coded in the A->B direction + if (ends_at_node_BA_direction = 1 ) then do + + // Only code links with an existing GC ratio (indicating a signalized intersection) + if (link_lyr.[BAGC]<>0 and signal > 0) then do + + //defaults are 0.4 gc ratio, 1.25 minute cycle length and 1.0 progression factor + gc_ratio = 0.4 + c_len = 1.25 + p_factor = 1.0 + + if (link_lyr.[IFC] = 2) then do + if (max_lc = 2) then c_len = 2.5 //Prime arterial & Prime arterial + else c_len = 2.0 //Prime arterial & anything lower + end + else if (link_lyr.[IFC] = 3) then do + if (max_lc > 3) then c_len = 2.0 //Major arterial & anything lower than a Major arterial + else c_len = 2.0 //Major arterial & Prime arterial or Major arterial + end + else if (link_lyr.[IFC] = 4) then do + if (min_lc < 4) then c_len = 2.0 //Anything lower than a Major arterial & Prime arterial + else c_len = 1.5 //Anything lower than a Major arterial & anything lower than a Prime arterial + end + else if (link_lyr.[IFC] > 4) then do + if (min_lc < 4) then c_len = 2.0 + if (min_lc = 4) then c_len = 1.5 + if (min_lc > 4) then c_len = 1.25 + end + + //update attributes + if( link_lyr.[BAGC] > 10) then link_lyr.[BA_GCRatio] = link_lyr.[BAGC]/100 + if( link_lyr.[BA_GCRatio] > 1.0) then link_lyr.[BA_GCRatio] = 1.0 + link_lyr.[BA_Cycle] = c_len + link_lyr.[BA_PF] = p_factor + + end + + end // end for BA links + + //code metered ramps AB Direction + if(ends_at_node_AB_direction = 1 and (link_lyr.[ABCNT]= 4 or link_lyr.[ABCNT] = 5)) then do + link_lyr.[AB_Cycle] = 2.5 + link_lyr.[AB_GCRatio] = 0.42 + link_lyr.[AB_PF] = 1.0 + end + + //code metered ramps BA Direction + if(ends_at_node_BA_direction = 1 and (link_lyr.[BACNT]= 4 or link_lyr.[BACNT] = 5)) then do + link_lyr.[BA_Cycle] = 2.5 + link_lyr.[BA_GCRatio] = 0.42 + link_lyr.[BA_PF] = 1.0 + end + + //code stops AB Direction + if(ends_at_node_AB_direction = 1 and (link_lyr.[ABCNT]= 2 or link_lyr.[ABCNT] = 3)) then do + link_lyr.[AB_Cycle] = 1.25 + link_lyr.[AB_GCRatio] = 0.42 + link_lyr.[AB_PF] = 1.0 + end + + //code stops BA Direction + if(ends_at_node_BA_direction = 1 and (link_lyr.[BACNT]= 2 or link_lyr.[BACNT] = 3)) then do + link_lyr.[BA_Cycle] = 1.25 + link_lyr.[BA_GCRatio] = 0.42 + link_lyr.[BA_PF] = 1.0 + end + + end // end for links + + end // end for nodes + + + // set alpha1 and beta1 fields, which are based upon free-flow speed to match POSTLOAD loaded time factors + lwr_bound = { " 0", "25", "30", "35", "40", "45", "50", "55", "60", "65", "70", "75"} + upr_bound = { "24", "29", "34", "39", "44", "49", "54", "59", "64", "69", "74", "99"} + alpha1 = {"0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8"} + beta1 = { "4", "4", "4", "4", "4", "4", "4", "4", "4", "4" , "4", "4"} + + for j = 1 to lwr_bound.length do + + //alpha1 + calcString = { "if ISPD >= " + lwr_bound[j] + " and ISPD <= "+upr_bound[j] + " then "+alpha1[j]+" else ALPHA1"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ALPHA1"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + //beta1 + calcString = { "if ISPD >= " + lwr_bound[j] + " and ISPD <= "+upr_bound[j] + " then "+beta1[j]+" else BETA1"} + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"BETA1"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = calcString + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + end + + //set alpha2 and beta2 fields (note that signalized intersections and stop-controlled intersections have same parameters, only meters vary) + alpha2_default = "4.5" + beta2_default = "2.0" + alpha2_meter = "6.0" + beta2_meter = "2.0" + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ALPHA2","BETA2"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {alpha2_default, beta2_default} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr,"Selection", "Select * where (abcnt=4 or abcnt=5)"} + Opts.Global.Fields = {"ALPHA2","BETA2"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {alpha2_meter,beta2_meter} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + // replicate across all time periods and add fields for vdf parameters + periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} + meters = { 0, 1, 0, 1, 0} + + for i = 1 to periods.length do + + // Add AB_Cycle, AB_PF, BA_Cycle, and BA_PF + vw = SetView(link_lyr) + strct = GetTableStructure(vw) + for j = 1 to strct.length do + strct[j] = strct[j] + {strct[j][1]} + end + + strct = strct + {{"AB_GCRatio"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_GCRatio"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_Cycle"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_Cycle"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"AB_PF"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BA_PF"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ALPHA1"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BETA1"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"ALPHA2"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + strct = strct + {{"BETA2"+periods[i], "Real", 14, 6, "True", , , , , , , null}} + + ModifyTable(view1, strct) + + in_fld ={"AB_GCRatio", "BA_GCRatio", "AB_Cycle","BA_Cycle","AB_PF","BA_PF"} + + out_fld ={"AB_GCRatio"+periods[i],"BA_GCRatio"+periods[i],"AB_Cycle"+periods[i],"BA_Cycle"+periods[i],"AB_PF"+periods[i],"BA_PF"+periods[i]} + + + values = {0.0,0.0,0.0,0.0,1.0,1.0} + // set GCRatio, Cycle length, PF + for j=1 to out_fld.length do + + //initialize to 0 + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {out_fld[j]} + Opts.Global.Method = "Value" + Opts.Global.Parameter = {values[j]} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select * where "+in_fld[j]+"> 0 and "+in_fld[j]+"<>null"} + Opts.Global.Fields = {out_fld[j]} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {in_fld[j]} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + end + + // reset GCRatio, cycle length and PF fields to 0 if metered ramp and off-peak period + if(meters[i] = 0) then do + + for j=1 to out_fld.length do + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr,"Selection", "Select * where (abcnt=4 or abcnt=5)"} + Opts.Global.Fields = {out_fld[j]} + Opts.Global.Method = "Value" + Opts.Global.Parameter = {values[j]} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + end + end + + + //set alpha1 and beta1 fields, which currently do not vary by time period + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ALPHA1"+periods[i], "BETA1"+periods[i]} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"ALPHA1","BETA1"} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + //set alpha2 and beta2 fields, which currently do not vary by time period + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ALPHA2"+periods[i],"BETA2"+periods[i]} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"ALPHA2","BETA2"} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + end // end for periods + + quit: + return(ok) +EndMacro +//***************************************************************************************************************************************************************** + +/************************************************************************8 +build highway network in TransCAD format +input file: + hwy.dbd - hwy line geographic file + turns.csv - turn prohibitor csv file, fields: from (link), to (link), penalty (null) + inktypeturns.dbf - dbf file between freeways and ramp added 0.5 min penalty + linktypelog.dbf - link type look up binary file, format for MMA assignment + +output file: hwy.net - hwy network file + link field included in network file: + Length (in miles) + IFC: functional classification, also used as link type look up field + *TM_EA (ABTM_EA/BATM_EA): Early AM period link travel time + *TM_AM (ABTM_AM/BATM_AM): AM Peak period link travel time + *TM_MD (ABTM_MD/BATM_MD): Midday period link travel time + *TM_PM (ABTM_PM/BATM_PM): PM Peak period link travel time + *TM_EV (ABTM_EV/BATM_EV): Evening period link travel time + *CP_EA (ABCP_EA/BACP_EA): Early AM period link capacity + *CP_AM (ABCP_AM/BACP_AM): AM Peak period link capacity + *CP_MD (ABCP_MD/BACP_MD): Midday period link capacity + *CP_PM (ABCP_PM/BACP_PM): PM Peak period link capacity + *CP_EV (ABCP_EV/BACP_EV): Evening period link capacity + *TX_EA (ABTX_EA/BATX_EA): Early AM period intersection travel time + *TX_AM (ABTX_AM/BATX_AM): AM Peak period intersection travel time + *TX_MD (ABTX_MD/BATX_MD): Midday period intersection travel time + *TX_PM (ABTX_PM/BATX_PM): PM Peak period intersection travel time + *TX_EV (ABTX_EV/BATX_EV): Evening period intersection travel time + *CX_EA (ABCX_EA/BACX_EA): Early AM period intersection capacity + *CX_AM (ABCX_AM/BACX_AM): AM Peak period intersection capacity + *CX_MD (ABCX_MD/BACX_MD): Midday period intersection capacity + *CX_PM (ABCX_PM/BACX_PM): PM Peak period intersection capacity + *CX_EV (ABCX_EV/BACX_EV): Evening period intersection capacity + COST: cost of distance (in cents) 19cents/mile + *CST (ABCST/BACST): generalized cost (in cents) of 19cents/mile + 35cents/minute + *SCST + *H2CST + *H3CST + ID: link ID of hwycad-id + + specify zone centroids, and create network + change network settings with linktype lookup table + turn prohibitors and link type turn penalty file + +************************************************************************************/ + +macro "create hwynet" + shared path, mxzone, inputDir, outputDir + ok = 0 + + RunMacro("close all") + + + // fpr=openfile(path+"\\hwycad.log","a") + // mytime=GetDateAndTime() + + //input file + db_file = outputDir + "\\hwy.dbd" + d_tp_tb = inputDir + "\\linktypeturns.dbf" //turn penalty in cents + s_tp_tb = outputDir + "\\turns.dbf" + + //output files + net_file = outputDir+"\\hwy.net" + + di2= GetDirectoryInfo(d_tp_tb, "file") + di3= GetDirectoryInfo(s_tp_tb, "file") + //check for files + if di2.length=0 and di3.length=0 then do + RunMacro("TCB Error",d_tp_tb+" "+s_tp_tb+" does not exist!") + goto quit + end + + // RunMacro("TCB Init") + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + db_link_lyr=db_file+"|"+link_lyr + db_node_lyr=db_file+"|"+node_lyr + + // STEP 1: Build Highway Network + Opts = null + Opts.Input.[Link Set] = {db_link_lyr,link_lyr} + Opts.Global.[Network Options].[Node ID] = node_lyr+ ".ID" + Opts.Global.[Network Options].[Link ID] = link_lyr + ".ID" + Opts.Global.[Network Options].[Turn Penalties] = "Yes" + Opts.Global.[Network Options].[Keep Duplicate Links] = "FALSE" + Opts.Global.[Network Options].[Ignore Link Direction] = "FALSE" + Opts.Global.[Link Options] = {{"Length", link_lyr+".Length", link_lyr+".Length"}, + {"ID", link_lyr+".ID", link_lyr+".ID"}, + {"IFC", link_lyr+".IFC", link_lyr+".IFC"}, + {"IHOV", link_lyr+".IHOV", link_lyr+".IHOV"}, + {"COST", link_lyr+".COST", link_lyr+".COST"}, + {"*LN_EA", link_lyr + ".ABLN_EA", link_lyr + ".BALN_EA"}, + {"*LN_AM", link_lyr + ".ABLN_AM", link_lyr + ".BALN_AM"}, + {"*LN_MD", link_lyr + ".ABLN_MD", link_lyr + ".BALN_MD"}, + {"*LN_PM", link_lyr + ".ABLN_PM", link_lyr + ".BALN_PM"}, + {"*LN_EV", link_lyr + ".ABLN_EV", link_lyr + ".BALN_EV"}, + {"ITOLL_EA", link_lyr + ".ITOLL_EA", link_lyr + ".ITOLL_EA"}, // Oct-08-2010, added to include toll+cost + {"ITOLL_AM", link_lyr + ".ITOLL_AM", link_lyr + ".ITOLL_AM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL_MD", link_lyr + ".ITOLL_MD", link_lyr + ".ITOLL_MD"}, // Oct-08-2010, added to include toll+cost + {"ITOLL_PM", link_lyr + ".ITOLL_PM", link_lyr + ".ITOLL_PM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL_EV", link_lyr + ".ITOLL_EV", link_lyr + ".ITOLL_EV"}, // Oct-08-2010, added to include toll+cost + {"ITOLL2_EA", link_lyr + ".ITOLL2_EA", link_lyr + ".ITOLL2_EA"}, // Oct-08-2010, added to include toll+cost + {"ITOLL2_AM", link_lyr + ".ITOLL2_AM", link_lyr + ".ITOLL2_AM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL2_MD", link_lyr + ".ITOLL2_MD", link_lyr + ".ITOLL2_MD"}, // Oct-08-2010, added to include toll+cost + {"ITOLL2_PM", link_lyr + ".ITOLL2_PM", link_lyr + ".ITOLL2_PM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL2_EV", link_lyr + ".ITOLL2_EV", link_lyr + ".ITOLL2_EV"}, // Oct-08-2010, added to include toll+cost + {"ITOLL3_EA", link_lyr + ".ITOLL3_EA", link_lyr + ".ITOLL3_EA"}, // Oct-08-2010, added to include toll+cost + {"ITOLL3_AM", link_lyr + ".ITOLL3_AM", link_lyr + ".ITOLL3_AM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL3_MD", link_lyr + ".ITOLL3_MD", link_lyr + ".ITOLL3_MD"}, // Oct-08-2010, added to include toll+cost + {"ITOLL3_PM", link_lyr + ".ITOLL3_PM", link_lyr + ".ITOLL3_PM"}, // Oct-08-2010, added to include toll+cost + {"ITOLL3_EV", link_lyr + ".ITOLL3_EV", link_lyr + ".ITOLL3_EV"}, // Oct-08-2010, added to include toll+cost + {"ITOLL4_EA", link_lyr + ".ITOLL4_EA", link_lyr + ".ITOLL4_EA"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost + {"ITOLL4_AM", link_lyr + ".ITOLL4_AM", link_lyr + ".ITOLL4_AM"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost + {"ITOLL4_MD", link_lyr + ".ITOLL4_MD", link_lyr + ".ITOLL4_MD"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost + {"ITOLL4_PM", link_lyr + ".ITOLL4_PM", link_lyr + ".ITOLL4_PM"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost + {"ITOLL4_EV", link_lyr + ".ITOLL4_EV", link_lyr + ".ITOLL4_EV"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost + {"ITOLL5_EA", link_lyr + ".ITOLL5_EA", link_lyr + ".ITOLL5_EA"}, // Nov-3-2010, added hhd toll = 3*toll+cost + {"ITOLL5_AM", link_lyr + ".ITOLL5_AM", link_lyr + ".ITOLL5_AM"}, // Nov-3-2010, added hhd toll = 3*toll+cost + {"ITOLL5_MD", link_lyr + ".ITOLL5_MD", link_lyr + ".ITOLL5_MD"}, // Nov-3-2010, added hhd toll = 3*toll+cost + {"ITOLL5_PM", link_lyr + ".ITOLL5_PM", link_lyr + ".ITOLL5_PM"}, // Nov-3-2010, added hhd toll = 3*toll+cost + {"ITOLL5_EV", link_lyr + ".ITOLL5_EV", link_lyr + ".ITOLL5_EV"}, // Nov-3-2010, added hhd toll = 3*toll+cost + {"ITRUCK", link_lyr + ".ITRUCK", link_lyr + ".ITRUCK"}, // Sep-30-2011, added ITRUCK to the network + {"*CP_EA", link_lyr+".ABCP_EA", link_lyr+".BACP_EA"}, + {"*CP_AM", link_lyr+".ABCP_AM", link_lyr+".BACP_AM"}, + {"*CP_MD", link_lyr+".ABCP_MD", link_lyr+".BACP_MD"}, + {"*CP_PM", link_lyr+".ABCP_PM", link_lyr+".BACP_PM"}, + {"*CP_EV", link_lyr+".ABCP_EV", link_lyr+".BACP_EV"}, + {"*CX_EA", link_lyr+".ABCX_EA", link_lyr+".BACX_EA"}, + {"*CX_AM", link_lyr+".ABCX_AM", link_lyr+".BACX_AM"}, + {"*CX_MD", link_lyr+".ABCX_MD", link_lyr+".BACX_MD"}, + {"*CX_PM", link_lyr+".ABCX_PM", link_lyr+".BACX_PM"}, + {"*CX_EV", link_lyr+".ABCX_EV", link_lyr+".BACX_EV"}, + {"*TM_EA", link_lyr+".ABTM_EA", link_lyr+".BATM_EA"}, + {"*TM_AM", link_lyr+".ABTM_AM", link_lyr+".BATM_AM"}, + {"*TM_MD", link_lyr+".ABTM_MD", link_lyr+".BATM_MD"}, + {"*TM_PM", link_lyr+".ABTM_PM", link_lyr+".BATM_PM"}, + {"*TM_EV", link_lyr+".ABTM_EV", link_lyr+".BATM_EV"}, + {"*TX_EA", link_lyr+".ABTX_EA", link_lyr+".BATX_EA"}, + {"*TX_AM", link_lyr+".ABTX_AM", link_lyr+".BATX_AM"}, + {"*TX_MD", link_lyr+".ABTX_MD", link_lyr+".BATX_MD"}, + {"*TX_PM", link_lyr+".ABTX_PM", link_lyr+".BATX_PM"}, + {"*TX_EV", link_lyr+".ABTX_EV", link_lyr+".BATX_EV"}, + {"*CST", link_lyr+".ABCST", link_lyr+".BACST"}, + {"*SCST_EA", link_lyr+".ABSCST_EA", link_lyr+".BASCST_EA"}, + {"*SCST_AM", link_lyr+".ABSCST_AM", link_lyr+".BASCST_AM"}, + {"*SCST_MD", link_lyr+".ABSCST_MD", link_lyr+".BASCST_MD"}, + {"*SCST_PM", link_lyr+".ABSCST_PM", link_lyr+".BASCST_PM"}, + {"*SCST_EV", link_lyr+".ABSCST_EV", link_lyr+".BASCST_EV"}, + {"*H2CST_EA", link_lyr+".ABH2CST_EA", link_lyr+".BAH2CST_EA"}, + {"*H2CST_AM", link_lyr+".ABH2CST_AM", link_lyr+".BAH2CST_AM"}, + {"*H2CST_MD", link_lyr+".ABH2CST_MD", link_lyr+".BAH2CST_MD"}, + {"*H2CST_PM", link_lyr+".ABH2CST_PM", link_lyr+".BAH2CST_PM"}, + {"*H2CST_EV", link_lyr+".ABH2CST_EV", link_lyr+".BAH2CST_EV"}, + {"*H3CST_EA", link_lyr+".ABH3CST_EA", link_lyr+".BAH3CST_EA"}, + {"*H3CST_AM", link_lyr+".ABH3CST_AM", link_lyr+".BAH3CST_AM"}, + {"*H3CST_MD", link_lyr+".ABH3CST_MD", link_lyr+".BAH3CST_MD"}, + {"*H3CST_PM", link_lyr+".ABH3CST_PM", link_lyr+".BAH3CST_PM"}, + {"*H3CST_EV", link_lyr+".ABH3CST_EV", link_lyr+".BAH3CST_EV"}, + {"*LHCST_EA", link_lyr+".ABLHCST_EA", link_lyr+".BALHCST_EA"}, + {"*LHCST_AM", link_lyr+".ABLHCST_AM", link_lyr+".BALHCST_AM"}, + {"*LHCST_MD", link_lyr+".ABLHCST_MD", link_lyr+".BALHCST_MD"}, + {"*LHCST_PM", link_lyr+".ABLHCST_PM", link_lyr+".BALHCST_PM"}, + {"*LHCST_EV", link_lyr+".ABLHCST_EV", link_lyr+".BALHCST_EV"}, + {"*MHCST_EA", link_lyr+".ABMHCST_EA", link_lyr+".BAMHCST_EA"}, + {"*MHCST_AM", link_lyr+".ABMHCST_AM", link_lyr+".BAMHCST_AM"}, + {"*MHCST_MD", link_lyr+".ABMHCST_MD", link_lyr+".BAMHCST_MD"}, + {"*MHCST_PM", link_lyr+".ABMHCST_PM", link_lyr+".BAMHCST_PM"}, + {"*MHCST_EV", link_lyr+".ABMHCST_EV", link_lyr+".BAMHCST_EV"}, + {"*HHCST_EA", link_lyr+".ABHHCST_EA", link_lyr+".BAHHCST_EA"}, + {"*HHCST_AM", link_lyr+".ABHHCST_AM", link_lyr+".BAHHCST_AM"}, + {"*HHCST_MD", link_lyr+".ABHHCST_MD", link_lyr+".BAHHCST_MD"}, + {"*HHCST_PM", link_lyr+".ABHHCST_PM", link_lyr+".BAHHCST_PM"}, + {"*HHCST_EV", link_lyr+".ABHHCST_EV", link_lyr+".BAHHCST_EV"}, + {"*CVCST_EA", link_lyr+".ABCVCST_EA", link_lyr+".BACVCST_EA"}, + {"*CVCST_AM", link_lyr+".ABCVCST_AM", link_lyr+".BACVCST_AM"}, + {"*CVCST_MD", link_lyr+".ABCVCST_MD", link_lyr+".BACVCST_MD"}, + {"*CVCST_PM", link_lyr+".ABCVCST_PM", link_lyr+".BACVCST_PM"}, + {"*CVCST_EV", link_lyr+".ABCVCST_EV", link_lyr+".BACVCST_EV"}, + {"*STM_EA", link_lyr+".ABSTM_EA", link_lyr+".BASTM_EA"}, + {"*STM_AM", link_lyr+".ABSTM_AM", link_lyr+".BASTM_AM"}, + {"*STM_MD", link_lyr+".ABSTM_MD", link_lyr+".BASTM_MD"}, + {"*STM_PM", link_lyr+".ABSTM_PM", link_lyr+".BASTM_PM"}, + {"*STM_EV", link_lyr+".ABSTM_EV", link_lyr+".BASTM_EV"}, + {"*HTM_EA", link_lyr+".ABHTM_EA", link_lyr+".BAHTM_EA"}, + {"*HTM_AM", link_lyr+".ABHTM_AM", link_lyr+".BAHTM_AM"}, + {"*HTM_MD", link_lyr+".ABHTM_MD", link_lyr+".BAHTM_MD"}, + {"*HTM_PM", link_lyr+".ABHTM_PM", link_lyr+".BAHTM_PM"}, + {"*HTM_EV", link_lyr+".ABHTM_EV", link_lyr+".BAHTM_EV"}, + {"*GCRATIO_EA", link_lyr+".AB_GCRatio_EA", link_lyr+".BA_GCRatio_EA"}, + {"*GCRATIO_AM", link_lyr+".AB_GCRatio_AM", link_lyr+".BA_GCRatio_AM"}, + {"*GCRATIO_MD", link_lyr+".AB_GCRatio_MD", link_lyr+".BA_GCRatio_MD"}, + {"*GCRATIO_PM", link_lyr+".AB_GCRatio_PM", link_lyr+".BA_GCRatio_PM"}, + {"*GCRATIO_EV", link_lyr+".AB_GCRatio_EV", link_lyr+".BA_GCRatio_EV"}, + {"*CYCLE_EA", link_lyr+".AB_Cycle_EA", link_lyr+".BA_Cycle_EA"}, + {"*CYCLE_AM", link_lyr+".AB_Cycle_AM", link_lyr+".BA_Cycle_AM"}, + {"*CYCLE_MD", link_lyr+".AB_Cycle_MD", link_lyr+".BA_Cycle_MD"}, + {"*CYCLE_PM", link_lyr+".AB_Cycle_PM", link_lyr+".BA_Cycle_PM"}, + {"*CYCLE_EV", link_lyr+".AB_Cycle_EV", link_lyr+".BA_Cycle_EV"}, + {"*PF_EA", link_lyr+".AB_PF_EA", link_lyr+".BA_PF_EA"}, + {"*PF_AM", link_lyr+".AB_PF_AM", link_lyr+".BA_PF_AM"}, + {"*PF_MD", link_lyr+".AB_PF_MD", link_lyr+".BA_PF_MD"}, + {"*PF_PM", link_lyr+".AB_PF_PM", link_lyr+".BA_PF_PM"}, + {"*PF_EV", link_lyr+".AB_PF_EV", link_lyr+".BA_PF_EV"}, + {"*ALPHA1_EA", link_lyr+".ALPHA1_EA", link_lyr+".ALPHA1_EA"}, + {"*ALPHA1_AM", link_lyr+".ALPHA1_AM", link_lyr+".ALPHA1_AM"}, + {"*ALPHA1_MD", link_lyr+".ALPHA1_MD", link_lyr+".ALPHA1_MD"}, + {"*ALPHA1_PM", link_lyr+".ALPHA1_PM", link_lyr+".ALPHA1_PM"}, + {"*ALPHA1_EV", link_lyr+".ALPHA1_EV", link_lyr+".ALPHA1_EV"}, + {"*BETA1_EA", link_lyr+".BETA1_EA", link_lyr+".BETA1_EA"}, + {"*BETA1_AM", link_lyr+".BETA1_AM", link_lyr+".BETA1_AM"}, + {"*BETA1_MD", link_lyr+".BETA1_MD", link_lyr+".BETA1_MD"}, + {"*BETA1_PM", link_lyr+".BETA1_PM", link_lyr+".BETA1_PM"}, + {"*BETA1_EV", link_lyr+".BETA1_EV", link_lyr+".BETA1_EV"}, + {"*ALPHA2_EA", link_lyr+".ALPHA2_EA", link_lyr+".ALPHA2_EA"}, + {"*ALPHA2_AM", link_lyr+".ALPHA2_AM", link_lyr+".ALPHA2_AM"}, + {"*ALPHA2_MD", link_lyr+".ALPHA2_MD", link_lyr+".ALPHA2_MD"}, + {"*ALPHA2_PM", link_lyr+".ALPHA2_PM", link_lyr+".ALPHA2_PM"}, + {"*ALPHA2_EV", link_lyr+".ALPHA2_EV", link_lyr+".ALPHA2_EV"}, + {"*BETA2_EA", link_lyr+".BETA2_EA", link_lyr+".BETA2_EA"}, + {"*BETA2_AM", link_lyr+".BETA2_AM", link_lyr+".BETA2_AM"}, + {"*BETA2_MD", link_lyr+".BETA2_MD", link_lyr+".BETA2_MD"}, + {"*BETA2_PM", link_lyr+".BETA2_PM", link_lyr+".BETA2_PM"}, + {"*BETA2_EV", link_lyr+".BETA2_EV", link_lyr+".BETA2_EV"}, + {"*PRELOAD_EA", link_lyr+".ABPRELOAD_EA", link_lyr+".BAPRELOAD_EA"}, + {"*PRELOAD_AM", link_lyr+".ABPRELOAD_AM", link_lyr+".BAPRELOAD_AM"}, + {"*PRELOAD_MD", link_lyr+".ABPRELOAD_MD", link_lyr+".BAPRELOAD_MD"}, + {"*PRELOAD_PM", link_lyr+".ABPRELOAD_PM", link_lyr+".BAPRELOAD_PM"}, + {"*PRELOAD_EV", link_lyr+".ABPRELOAD_EV", link_lyr+".BAPRELOAD_EV"}, + {"*LOSC_FACT", link_lyr+".ABLOSC_FACT", link_lyr+".BALOSC_FACT"}, // added for reliability - 02/02/2016 + {"*LOSD_FACT", link_lyr+".ABLOSD_FACT", link_lyr+".BALOSD_FACT"}, // added for reliability - 02/02/2016 + {"*LOSE_FACT", link_lyr+".ABLOSE_FACT", link_lyr+".BALOSE_FACT"}, // added for reliability - 02/02/2016 + {"*LOSFL_FACT", link_lyr+".ABLOSFL_FACT", link_lyr+".BALOSFL_FACT"}, // added for reliability - 02/02/2016 + {"*LOSFH_FACT", link_lyr+".ABLOSFH_FACT", link_lyr+".BALOSFH_FACT"}, // added for reliability - 02/02/2016 + {"*STATREL_EA", link_lyr+".ABSTATREL_EA", link_lyr+".BASTATREL_EA"}, // added for reliability - 02/02/2016 + {"*STATREL_AM", link_lyr+".ABSTATREL_AM", link_lyr+".BASTATREL_AM"}, // added for reliability - 02/02/2016 + {"*STATREL_MD", link_lyr+".ABSTATREL_MD", link_lyr+".BASTATREL_MD"}, // added for reliability - 02/02/2016 + {"*STATREL_PM", link_lyr+".ABSTATREL_PM", link_lyr+".BASTATREL_PM"}, // added for reliability - 02/02/2016 + {"*STATREL_EV", link_lyr+".ABSTATREL_EV", link_lyr+".BASTATREL_EV"}, + {"*_TOTREL_EA", link_lyr+".AB_TOTREL_EA", link_lyr+".BA_TOTREL_EA"}, + {"*_TOTREL_AM", link_lyr+".AB_TOTREL_AM", link_lyr+".BA_TOTREL_AM"}, + {"*_TOTREL_MD", link_lyr+".AB_TOTREL_MD", link_lyr+".BA_TOTREL_MD"}, + {"*_TOTREL_PM", link_lyr+".AB_TOTREL_PM", link_lyr+".BA_TOTREL_PM"}, + {"*_TOTREL_EV", link_lyr+".AB_TOTREL_EV", link_lyr+".BA_TOTREL_EV"}} // added for reliability - 02/02/2016 + + // add two node fields into the network for turning movement purposes, by JXu + Opts.Global.[Node Options].ID = node_lyr + ".ID" + Opts.Global.[Node Options].DATA = node_lyr + ".temp" + Opts.Output.[Network File] = net_file + RunMacro("HwycadLog",{"createhwynet_turn.rsc: create hwynet1","Build Highway Network"}) + ok = RunMacro("TCB Run Operation", 1, "Build Highway Network", Opts) + if !ok then goto quit + + // STEP 2: Highway Network Setting + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Centroids Set] = {db_node_lyr, node_lyr, "Selection", "select * where ID <="+i2s(mxzone)} + Opts.Global.[Spc Turn Pen Method] = 3 + Opts.Input.[Def Turn Pen Table] = {d_tp_tb} + Opts.Input.[Spc Turn Pen Table] = {s_tp_tb} + Opts.Field.[Link type] = "IFC" + Opts.Global.[Global Turn Penalties] = {0, 0, 0, 0} + Opts.Flag.[Use Link Types] = "True" + RunMacro("HwycadLog",{"createhwynet_turn.rsc: create hwynet1","Highway Network Setting"}) + ok = RunMacro("TCB Run Operation", 2, "Highway Network Setting", Opts) + if !ok then goto quit + + mytime=GetDateAndTime() + + RunMacro("close all") //before delete db_file, close it + + + //writeline(fpr,mytime+", network setting") + ok=1 + quit: + //if fpr<>null then closefile(fpr) + return(ok) +endMacro +/************************************************************************************* +Calculate upstream and downsstream distances to major interchanges for freeway segments + +steps: +1. identify major interchange nodes +2. for each freeway segment, calculate upstream and downstream segment + +input: + output\\hwy.dbd + +output: + output\\MajorInterchangeDistance.csv + +by: nagendra.dhakar@rsginc.com +*************************************************************************************/ +Macro "DistanceToInterchange" + shared path, inputDir, outputDir + shared interchanges, freeways, linklayer, nodelayer + + ok=0 + + // input highway database + db_file=outputDir+"\\hwy.dbd" + hwy_dbd=db_file + + // output settings + out_file = outputDir+"\\MajorInterchangeDistance.csv" + + // add layers + layers = GetDBLayers(hwy_dbd) + linklayer=layers[2] + nodelayer=layers[1] + + db_linklayer=hwy_dbd+"|"+linklayer + db_nodelayer=hwy_dbd+"|"+nodelayer + + info = GetDBInfo(hwy_dbd) + temp_map = CreateMap("temp",{{"scope",info[1]}}) + + temp_layer = AddLayer(temp_map,linklayer,hwy_dbd,linklayer) + temp_layer = AddLayer(temp_map,nodelayer,hwy_dbd,nodelayer) + + // Identify Interchanges + SetLayer(linklayer) + + // Major interchange + query_ramps = 'Select * where (IFC=8) AND position(NM,"HOV")=0' // Major interchange - HOV access connectors are removed + MaxLinks = 50 + + on Error do ShowMessage("The SQL query: (" + query_ramps + ") is not correct.") end + VerifyQuery(query_ramps) + + nramps = SelectByQuery("Ramp Set", "Several", query_ramps,) + + query_freeways = "Select * where IFC=1" + on Error do ShowMessage("The SQL query: (" + query_freeways + ") is not correct.") end + VerifyQuery(query_freeways) + + nfreeways = SelectByQuery("Freeway Set", "Several",query_freeways,) + + // get freeway segments ids, IHOV, ANode BNode + freeways = GetDataVectors("Freeway Set", {"ID","IHOV"},) + + // intersect with nodes to select connected nodes + SetLayer(nodelayer) + nnodes = SelectByLinks("Interchange Set", "Several", "Ramp Set") + ninterchanges = SelectByLinks("Interchange Set", "Subset", "Freeway Set") + + // get interchange ids + interchanges = GetDataVector("Interchange Set","ID",) + + outfile = OpenFile(out_file, "w") + WriteLine(outfile, JoinStrings({"LinkID","Length","updistance","downdistance","ihov","UpLinks","DownLinks"},",")) + + on Error goto quit + + // loop through the freeway segments + dim upstreamlinkset[freeways[1].Length,MaxLinks-1], downstreamlinkset[freeways[1].Length,MaxLinks-1] + + CreateProgressBar("Calculating Interchange Distances", "True") + + for i=1 to freeways[1].Length do + perc=RealToInt(100*i/freeways[1].Length) + + UpdateProgressBar("Calculating Interchange Distances for LinkID: " +string(freeways[1][i]), perc) + + SetLayer(linklayer) + linkid = freeways[1][i] + ihov = freeways[2][i] + + query = "Select * where ID="+String(linkid) + count = SelectByQuery("Selection", "Several", query,) + linklength = GetDataVector("Selection", "Length",) + + nodes = GetEndPoints(linkid) + FromNode=nodes[1] + ToNode=nodes[2] + + // upstream - from node + isInterchange = RunMacro("NodeIsInterchange", FromNode) + + BaseLink = linkid + j=1 + + query1="n/a" + query2="n/a" + upstreamdistance=linklength[1]*0.5 + downstreamdistance=linklength[1]*0.5 + numupstreamlinks=0 + numdownstreamlinks=0 + + if (ihov<>2) then do + while (isInterchange=0) do + SetLayer(nodelayer) + links = GetNodeLinks(FromNode) // links set that meet at the node + + coordinates_base=GetCoordsFromLinks(linklayer, , {{BaseLink,1}}) + + // find the freeway link that is not the current link + upstreamlink = RunMacro("FindNextLink",links,BaseLink) + + if (upstreamlink <> null) then do + + coordinates_upstream=GetCoordsFromLinks(linklayer, , {{upstreamlink,1}}) + opposite = RunMacro("IsOppositeDirection",coordinates_base[1],coordinates_upstream[1]) // 1- true, 0- false + + if (opposite=0 and j1 and upstreamlink= linkid) then isInterchange=1 + else do + upstreamlinkset[i][j] = upstreamlink + j=j+1 + end + end + else do + isInterchange=1 + if (j=MaxLinks) then upstreamdistance=99 + end + end + else do + isInterchange=1 + if (j=MaxLinks) then upstreamdistance=99 + end + + end + + numupstreamlinks =j-1 + + // calculate upstream distance + SetLayer(linklayer) + + if j>=2 then do + query1 = "Select * where ID="+String(upstreamlinkset[i][1]) + + if j>2 then do + for iter=2 to numupstreamlinks do + query1 = JoinStrings({query1," or ID=",r2s(upstreamlinkset[i][iter])},"") + end + end + + count = SelectByQuery("Selection", "Several", query1,) + lengths = GetDataVector("Selection", "Length",) + upstreamdistance=VectorStatistic(lengths,"Sum",) + + // add half of the current link length to make the distance from midpoint + upstreamdistance = upstreamdistance + (linklength[1]*0.5) + end + + + // downstream - to node + isInterchange = RunMacro("NodeIsInterchange", ToNode) + + BaseLink = linkid + j=1 + while (isInterchange=0) do + SetLayer(nodelayer) + links = GetNodeLinks(ToNode) + + coordinates_base=GetCoordsFromLinks(linklayer, , {{BaseLink,-1}}) + + // find the freeway link that is not the current link + downstreamlink = RunMacro("FindNextLink",links,BaseLink) + + if (downstreamlink <> null) then do + coordinates_downstream=GetCoordsFromLinks(linklayer, , {{downstreamlink,-1}}) + opposite = RunMacro("IsOppositeDirection",coordinates_base[1],coordinates_downstream[1]) // 1- true, 0- false + + if (opposite=0 and j1 and downstreamlink=linkid) then isInterchange=1 + else do + downstreamlinkset[i][j] = downstreamlink + j=j+1 + end + + end + else do + isInterchange=1 + if (j=MaxLinks) then downstreamdistance=99 + end + end + else do + isInterchange=1 + if (j=MaxLinks) then downstreamdistance=99 + end + + end + numdownstreamlinks =j-1 + + // calculate downstream distance + SetLayer(linklayer) + + if j>=2 then do + query2 = "Select * where ID="+String(downstreamlinkset[i][1]) + + if j>2 then do + for iter=2 to numdownstreamlinks do + query2 = JoinStrings({query2," or ID=",r2s(downstreamlinkset[i][iter])},"") + end + end + + count = SelectByQuery("Selection", "Several", query2,) + lengths = GetDataVector("Selection", "Length",) + downstreamdistance=VectorStatistic(lengths,"Sum",) + + // add half of the current link length to make the distance from midpoint + downstreamdistance = downstreamdistance + (linklength[1]*0.5) + + end + + end + + else do + // HOV segments - set default value of 9999 miles. Distances are not calculated as HOV segments are pretty reliable. + upstreamdistance = 9999 + downstreamdistance = 9999 + numupstreamlinks = 9999 + numdownstreamlinks = 9999 + + end + + WriteLine(outfile, JoinStrings({i2s(linkid),r2s(linklength[1]),r2s(upstreamdistance),r2s(downstreamdistance),i2s(ihov),i2s(numupstreamlinks),i2s(numdownstreamlinks)},",")) + + end + + CloseFile(outfile) + + DestroyProgressBar() + ok=1 + return(ok) + + quit: + showmessage("Error, i: " + string(i) + ", j: " + string(j) + ", linkid: " + string(linkid)) + +EndMacro +/************************************************************************************* +Check if the nodes is an interchange: 0-No, 1- Yes +**************************************************************************************/ +Macro "NodeIsInterchange" (nodeid) + shared interchanges + + isInter = 0 + + for i=1 to interchanges.Length do + if nodeid=interchanges[i] then isInter = 1 + end + + return (isInter) + +EndMacro +/************************************************************************************* +Identify forward links: + Identify links that are not the previous link (linkid) + Selects the link that is also a freeway and assign that as the next link + Assumption: there is only one next freeway link +**************************************************************************************/ +Macro "FindNextLink" (linkset,linkid) + shared linklayer + + nextlink = null + + for i=1 to linkset.Length do + if linkid <> linkset[i] then do + IsFreeway = RunMacro("LinkIsFreeway",linkset[i]) + if (IsFreeway=1) then do + IsWrongDirection = RunMacro("WrongDirection",linkset[i],linkid) + IsHov = RunMacro("LinkIsHov",linkset[i]) + if (IsHov=0 & IsWrongDirection=0) then nextlink = linkset[i] + end + end + end + + return (nextlink) + +EndMacro +/************************************************************************************* +Check if the link is in the wrong direction +**************************************************************************************/ +Macro "WrongDirection" (link,linkid) + shared linklayer + + SetLayer(linklayer) + nodes1 = GetEndPoints(linkid) + nodes2 = GetEndPoints(link) + + tonode1 = nodes1[2] // base link + tonode2 = nodes2[2] // next link + + // compare ToNode - if they are same then wrong direction + if tonode1 = tonode2 then return(1) + else return(0) + +EndMacro + +/************************************************************************************* +Check if the links is a freeway link: 0-No, 1- Yes +**************************************************************************************/ +Macro "LinkIsFreeway" (link) + shared freeways + + freeway=0 + for j=1 to freeways[1].Length do + if link = freeways[1][j] then freeway=1 + end + + return (freeway) + +EndMacro +/************************************************************************************* +Check if link is a HOV segment +**************************************************************************************/ +Macro "LinkIsHov" (link) + shared freeways + + hov=0 + for j=1 to freeways[1].Length do + if link = freeways[1][j] then do + if freeways[2][j]=2 then hov=1 + end + end + + return(hov) + +EndMacro +/************************************************************************************* +Find link direction (coordinates as input) +**************************************************************************************/ +Macro "GetLinkDirection" (coordinates) + + maxnum = coordinates.length + + nodeA = coordinates[1] + nodeB = coordinates[maxnum] + + deltaX = nodeB.lon-nodeA.lon + deltaY = nodeB.lat-nodeA.lat + + if deltaY>0 then slope1 = deltaX/deltaY + else slope1 = deltaX + + if deltaX>0 then slope2 = deltaY/deltaX + else slope2 = deltaY + + direction = "" + if abs(slope1) > abs(slope2) then do + //pre_dir = "EW" + if deltaX<0 then direction = "WB" + else direction = "EB" + end + else do + //pre_dir = "NS" + if deltaY<0 then direction = "SB" + else direction = "NB" + end + + return(direction) + +EndMacro +/************************************************************************************* +Check if the two segments (coordinates as input) have opposite direction +**************************************************************************************/ +Macro "IsOppositeDirection" (coordinates1, coordinates2) + + direction1 = RunMacro("GetLinkDirection",coordinates1) + direction2 = RunMacro("GetLinkDirection",coordinates2) + + if direction1="NB" and direction2="SB" then return(1) + else if direction2="NB" and direction1="SB" then return(1) + else if direction1="EB" and direction2="WB" then return(1) + else if direction2="EB" and direction1="WB" then return(1) + else return(0) + +EndMacro diff --git a/sandag_abm/src/main/gisdk/createtodtables.rsc b/sandag_abm/src/main/gisdk/createtodtables.rsc new file mode 100644 index 0000000..677d87d --- /dev/null +++ b/sandag_abm/src/main/gisdk/createtodtables.rsc @@ -0,0 +1,916 @@ +/************************************************************** + CreateAutoTables //modified "externalExternalTripsByYear.csv" input file and code, on 09/16/16, YMA + + Inputs + input\airportAutoTrips_XX.mtx + input\autoTrips_XX.mtx + input\extTrip_XX.mtx + + where XX is period = {_EA,_AM,_MD,_PM,_EV} + + input\commVehTODTrips.mtx + input\dailyDistributionMatricesTruckam.mtx + input\dailyDistributionMatricesTruckpm.mtx + input\dailyDistributionMatricesTruckop.mtx + + Outputs + output\Trip_XX.mtx + +*******************************************************************/ +Macro "Create Auto Tables" + + shared path, inputDir, outputDir + + periods={"_EA","_AM","_MD","_PM","_EV"} + + // read properties from sandag_abm.properties in /conf folder + properties = "\\conf\\sandag_abm.properties" + skipSpecialEventModel = RunMacro("read properties",properties,"RunModel.skipSpecialEventModel", "S") + //VOT bins for non resident models, 1->3 + votBinEE = 3 + votBinExternalInternal = 3 + votBinCommercialVehicles=3 + + /* + truckTables = { + inputDir+"\\dailyDistributionMatricesTruckam.mtx", + inputDir+"\\dailyDistributionMatricesTruckop.mtx", + inputDir+"\\dailyDistributionMatricesTruckpm.mtx" } + + truckPeriods = {2, 1, 2, 3, 2} + truckFactors = {0.1, 1.0, 0.65, 1.0, 0.25} + truckMatrices ={"lhdn","lhdt","mhdn","mhdt","hhdn","hhdt"} + */ + + truckTables = { + outputDir+"\\dailyDistributionMatricesTruckEA.mtx", + outputDir+"\\dailyDistributionMatricesTruckAM.mtx", + outputDir+"\\dailyDistributionMatricesTruckMD.mtx", + outputDir+"\\dailyDistributionMatricesTruckPM.mtx", + outputDir+"\\dailyDistributionMatricesTruckEV.mtx" } + + dim truckMatrices[truckTables.length] + dim truckCurrencies[truckTables.length] + + for i = 1 to truckTables.length do + // create truck matrix currencies + truckMatrices[i] = OpenMatrix(truckTables[i], ) + truckCurrencies[i] = CreateMatrixCurrencies(truckMatrices[i], , , ) + end + + externalInternalTables = { + outputDir+"\\usSdWrk", + outputDir+"\\usSdNon" + } + + //create external-external matrix from csv input file + ok = RunMacro("TCB Run Macro", 1,"Create External-External Trip Matrix",{}) + if !ok then goto quit + + //create external-external currencies + externalExternalMatrixName = outputDir + "\\externalExternalTrips.mtx" + externalExternalMatrix = OpenMatrix(externalExternalMatrixName, ) + + externalExternalCurrency = CreateMatrixCurrency(externalExternalMatrix,'Trips',,,) + externalExternalDiurnalFactors = { 0.074, 0.137, 0.472, 0.183, 0.133} + externalExternalOccupancyFactors = {0.43, 0.42, 0.15 } + + + internalExternalTables = { + { + outputDir+"\\autoInternalExternalTrips_EA_low.mtx", + outputDir+"\\autoInternalExternalTrips_AM_low.mtx", + outputDir+"\\autoInternalExternalTrips_MD_low.mtx", + outputDir+"\\autoInternalExternalTrips_PM_low.mtx", + outputDir+"\\autoInternalExternalTrips_EV_low.mtx"}, + { + outputDir+"\\autoInternalExternalTrips_EA_med.mtx", + outputDir+"\\autoInternalExternalTrips_AM_med.mtx", + outputDir+"\\autoInternalExternalTrips_MD_med.mtx", + outputDir+"\\autoInternalExternalTrips_PM_med.mtx", + outputDir+"\\autoInternalExternalTrips_EV_med.mtx" + }, + { + outputDir+"\\autoInternalExternalTrips_EA_high.mtx", + outputDir+"\\autoInternalExternalTrips_AM_high.mtx", + outputDir+"\\autoInternalExternalTrips_MD_high.mtx", + outputDir+"\\autoInternalExternalTrips_PM_high.mtx", + outputDir+"\\autoInternalExternalTrips_EV_high.mtx" + } + } + visitorTables = { + { + outputDir+"\\autoVisitorTrips_EA_low.mtx", + outputDir+"\\autoVisitorTrips_AM_low.mtx", + outputDir+"\\autoVisitorTrips_MD_low.mtx", + outputDir+"\\autoVisitorTrips_PM_low.mtx", + outputDir+"\\autoVisitorTrips_EV_low.mtx"}, + { + outputDir+"\\autoVisitorTrips_EA_med.mtx", + outputDir+"\\autoVisitorTrips_AM_med.mtx", + outputDir+"\\autoVisitorTrips_MD_med.mtx", + outputDir+"\\autoVisitorTrips_PM_med.mtx", + outputDir+"\\autoVisitorTrips_EV_med.mtx"}, + { + outputDir+"\\autoVisitorTrips_EA_high.mtx", + outputDir+"\\autoVisitorTrips_AM_high.mtx", + outputDir+"\\autoVisitorTrips_MD_high.mtx", + outputDir+"\\autoVisitorTrips_PM_high.mtx", + outputDir+"\\autoVisitorTrips_EV_high.mtx"} + } + crossBorderTables = { + { + outputDir+"\\autoCrossBorderTrips_EA_low.mtx", + outputDir+"\\autoCrossBorderTrips_AM_low.mtx", + outputDir+"\\autoCrossBorderTrips_MD_low.mtx", + outputDir+"\\autoCrossBorderTrips_PM_low.mtx", + outputDir+"\\autoCrossBorderTrips_EV_low.mtx"}, + { + outputDir+"\\autoCrossBorderTrips_EA_med.mtx", + outputDir+"\\autoCrossBorderTrips_AM_med.mtx", + outputDir+"\\autoCrossBorderTrips_MD_med.mtx", + outputDir+"\\autoCrossBorderTrips_PM_med.mtx", + outputDir+"\\autoCrossBorderTrips_EV_med.mtx"}, + { + outputDir+"\\autoCrossBorderTrips_EA_high.mtx", + outputDir+"\\autoCrossBorderTrips_AM_high.mtx", + outputDir+"\\autoCrossBorderTrips_MD_high.mtx", + outputDir+"\\autoCrossBorderTrips_PM_high.mtx", + outputDir+"\\autoCrossBorderTrips_EV_high.mtx"} + } + + airportTables = { + { + outputDir+"\\autoAirportTrips_EA_low.mtx", + outputDir+"\\autoAirportTrips_AM_low.mtx", + outputDir+"\\autoAirportTrips_MD_low.mtx", + outputDir+"\\autoAirportTrips_PM_low.mtx", + outputDir+"\\autoAirportTrips_EV_low.mtx"}, + { + outputDir+"\\autoAirportTrips_EA_med.mtx", + outputDir+"\\autoAirportTrips_AM_med.mtx", + outputDir+"\\autoAirportTrips_MD_med.mtx", + outputDir+"\\autoAirportTrips_PM_med.mtx", + outputDir+"\\autoAirportTrips_EV_med.mtx"}, + { + outputDir+"\\autoAirportTrips_EA_high.mtx", + outputDir+"\\autoAirportTrips_AM_high.mtx", + outputDir+"\\autoAirportTrips_MD_high.mtx", + outputDir+"\\autoAirportTrips_PM_high.mtx", + outputDir+"\\autoAirportTrips_EV_high.mtx"} + } + personTables = { + { + outputDir+"\\autoTrips_EA_low.mtx", + outputDir+"\\autoTrips_AM_low.mtx", + outputDir+"\\autoTrips_MD_low.mtx", + outputDir+"\\autoTrips_PM_low.mtx", + outputDir+"\\autoTrips_EV_low.mtx"}, + { + outputDir+"\\autoTrips_EA_med.mtx", + outputDir+"\\autoTrips_AM_med.mtx", + outputDir+"\\autoTrips_MD_med.mtx", + outputDir+"\\autoTrips_PM_med.mtx", + outputDir+"\\autoTrips_EV_med.mtx"}, + { + outputDir+"\\autoTrips_EA_high.mtx", + outputDir+"\\autoTrips_AM_high.mtx", + outputDir+"\\autoTrips_MD_high.mtx", + outputDir+"\\autoTrips_PM_high.mtx", + outputDir+"\\autoTrips_EV_high.mtx"} + } + //the following table names have the period appended + CTRampMatrices = {"SOV_GP","SOV_PAY","SR2_GP","SR2_HOV","SR2_PAY","SR3_GP","SR3_HOV","SR3_PAY"} + + //output files + outMatrixNames = {"Trip"+periods[1]+"_VOT.mtx", "Trip"+periods[2]+"_VOT.mtx", "Trip"+periods[3]+"_VOT.mtx", "Trip"+periods[4]+"_VOT.mtx", "Trip"+periods[5]+"_VOT.mtx"} + outTableNames = {"SOV_GP_LOW", "SOV_PAY_LOW", "SR2_GP_LOW","SR2_HOV_LOW", "SR2_PAY_LOW", "SR3_GP_LOW","SR3_HOV_LOW","SR3_PAY_LOW", + "SOV_GP_MED", "SOV_PAY_MED", "SR2_GP_MED","SR2_HOV_MED", "SR2_PAY_MED", "SR3_GP_MED","SR3_HOV_MED","SR3_PAY_MED", + "SOV_GP_HIGH", "SOV_PAY_HIGH", "SR2_GP_HIGH","SR2_HOV_HIGH", "SR2_PAY_HIGH", "SR3_GP_HIGH","SR3_HOV_HIGH","SR3_PAY_HIGH", + "lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} + + commVehTable = outputDir+"\\commVehTODTrips.mtx" + commVehMatrices = {"EA NonToll","AM NonToll","MD NonToll","PM NonToll","EV NonToll","EA Toll","AM Toll","MD Toll","PM Toll","EV Toll"} + + // create comm vehicle matrix currencies + commVehMatrix = OpenMatrix(commVehTable, ) + commVehCurrencies = CreateMatrixCurrencies(commVehMatrix, , , ) + + + for i = 1 to periods.length do + + //open person trip matrix currencies + personMatrixLow = OpenMatrix(personTables[1][i], ) + personCurrenciesLow = CreateMatrixCurrencies(personMatrixLow, , , ) + + personMatrixMed = OpenMatrix(personTables[2][i], ) + personCurrenciesMed = CreateMatrixCurrencies(personMatrixMed, , , ) + + personMatrixHigh = OpenMatrix(personTables[3][i], ) + personCurrenciesHigh = CreateMatrixCurrencies(personMatrixHigh, , , ) + + totalOutMatrices = CTRampMatrices.length * 3 + + counter = 0 + dim curr_array[totalOutMatrices] + for j = 1 to totalOutMatrices do + counter = counter + 1 + curr_array[j] = CreateMatrixCurrency(personMatrixLow, CTRampMatrices[counter]+periods[i], ,, ) + if counter = CTRampMatrices.length then counter=0 + + end + + //open internal-external matrix currencies + internalExternalMatrixLow = OpenMatrix(internalExternalTables[1][i], ) + internalExternalCurrenciesLow = CreateMatrixCurrencies(internalExternalMatrixLow, , , ) + + internalExternalMatrixMed = OpenMatrix(internalExternalTables[2][i], ) + internalExternalCurrenciesMed = CreateMatrixCurrencies(internalExternalMatrixMed, , , ) + + internalExternalMatrixHigh = OpenMatrix(internalExternalTables[3][i], ) + internalExternalCurrenciesHigh = CreateMatrixCurrencies(internalExternalMatrixHigh, , , ) + + //open airport matrix currencies + airportMatrixLow = OpenMatrix(airportTables[1][i], ) + airportCurrenciesLow = CreateMatrixCurrencies(airportMatrixLow, , , ) + + airportMatrixMed = OpenMatrix(airportTables[2][i], ) + airportCurrenciesMed = CreateMatrixCurrencies(airportMatrixMed, , , ) + + airportMatrixHigh = OpenMatrix(airportTables[3][i], ) + airportCurrenciesHigh = CreateMatrixCurrencies(airportMatrixHigh, , , ) + + //open visitor matrix currencies + visitorMatrixLow = OpenMatrix(visitorTables[1][i], ) + visitorCurrenciesLow = CreateMatrixCurrencies(visitorMatrixLow, , , ) + + visitorMatrixMed = OpenMatrix(visitorTables[2][i], ) + visitorCurrenciesMed = CreateMatrixCurrencies(visitorMatrixMed, , , ) + + visitorMatrixHigh = OpenMatrix(visitorTables[3][i], ) + visitorCurrenciesHigh = CreateMatrixCurrencies(visitorMatrixHigh, , , ) + + //open cross border matrix currencies + crossBorderMatrixLow = OpenMatrix(crossBorderTables[1][i], ) + crossBorderCurrenciesLow = CreateMatrixCurrencies(crossBorderMatrixLow, , , ) + + crossBorderMatrixMed = OpenMatrix(crossBorderTables[2][i], ) + crossBorderCurrenciesMed = CreateMatrixCurrencies(crossBorderMatrixMed, , , ) + + crossBorderMatrixHigh = OpenMatrix(crossBorderTables[3][i], ) + crossBorderCurrenciesHigh = CreateMatrixCurrencies(crossBorderMatrixHigh, , , ) + + + //open external-internal work matrix currencies + externalInternalWrkMatrix = OpenMatrix(externalInternalTables[1]+periods[i]+".mtx", ) + externalInternalWrkCurrencies = CreateMatrixCurrencies(externalInternalWrkMatrix, , , ) + + //open external-internal non-work matrix currencies + externalInternalNonMatrix = OpenMatrix(externalInternalTables[2]+periods[i]+".mtx", ) + externalInternalNonCurrencies = CreateMatrixCurrencies(externalInternalNonMatrix, , , ) + + //open special event matrix currencies; Wu added 5/16/2017 + specialEventMatrix = OpenMatrix(specialEventables[i], ) + specialEventCurrencies = CreateMatrixCurrencies(specialEventMatrix, , , ) + + + //create output trip table and matrix currencies for this time period + outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outputDir+"\\"+outMatrixNames[i]}, + {"Label", outMatrixNames[i]}, + {"Tables",outTableNames}, + {"File Based", "Yes"}}) + SetMatrixCoreNames(outMatrix, outTableNames) + + outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) + +//LOW VOT + // calculate output matrices + outCurrencies.SOV_GP_LOW :=Nz(outCurrencies.SOV_GP_LOW) + outCurrencies.SOV_GP_LOW := Nz(personCurrenciesLow.("SOV_GP"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SOV_GP"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SOV_GP"+periods[i])) + + Nz(airportCurrenciesLow.("SOV_GP"+periods[i])) + + Nz(visitorCurrenciesLow.("SOV_GP"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) + if votBinExternalExternal=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) + if votBinCommercialVehicles=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + Nz(commVehCurrencies.(commVehMatrices[i])) + + outCurrencies.SOV_PAY_LOW :=Nz(outCurrencies.SOV_PAY_LOW) + outCurrencies.SOV_PAY_LOW := Nz(personCurrenciesLow.("SOV_PAY"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SOV_PAY"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SOV_PAY"+periods[i])) + + Nz(airportCurrenciesLow.("SOV_PAY"+periods[i])) + + Nz(visitorCurrenciesLow.("SOV_PAY"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SOV_PAY_LOW := outCurrencies.SOV_PAY_LOW + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) + if votBinCommercialVehicles=1 then outCurrencies.SOV_PAY_LOW := outCurrencies.SOV_PAY_LOW + Nz(commVehCurrencies.(commVehMatrices[i+5])) + + outCurrencies.SR2_GP_LOW :=Nz(outCurrencies.SR2_GP_LOW) + outCurrencies.SR2_GP_LOW := Nz(personCurrenciesLow.("SR2_GP"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR2_GP"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR2_GP"+periods[i])) + + Nz(airportCurrenciesLow.("SR2_GP"+periods[i])) + + Nz(visitorCurrenciesLow.("SR2_GP"+periods[i])) + + outCurrencies.SR2_HOV_LOW :=Nz(outCurrencies.SR2_HOV_LOW) + outCurrencies.SR2_HOV_LOW := Nz(personCurrenciesLow.("SR2_HOV"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR2_HOV"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR2_HOV"+periods[i])) + + Nz(airportCurrenciesLow.("SR2_HOV"+periods[i])) + + Nz(visitorCurrenciesLow.("SR2_HOV"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SR2_HOV_LOW := outCurrencies.SR2_HOV_LOW + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) + if votBinExternalExternal=1 then outCurrencies.SR2_HOV_LOW := outCurrencies.SR2_HOV_LOW + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] + + outCurrencies.SR2_PAY_LOW :=Nz(outCurrencies.SR2_PAY_LOW) + outCurrencies.SR2_PAY_LOW := Nz(personCurrenciesLow.("SR2_PAY"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR2_PAY"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR2_PAY"+periods[i])) + + Nz(airportCurrenciesLow.("SR2_PAY"+periods[i])) + + Nz(visitorCurrenciesLow.("SR2_PAY"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SR2_PAY_LOW := outCurrencies.SR2_PAY_LOW + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) + + outCurrencies.SR3_GP_LOW :=Nz(outCurrencies.SR3_GP_LOW) + outCurrencies.SR3_GP_LOW := Nz(personCurrenciesLow.("SR3_GP"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR3_GP"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR3_GP"+periods[i])) + + Nz(airportCurrenciesLow.("SR3_GP"+periods[i])) + + Nz(visitorCurrenciesLow.("SR3_GP"+periods[i])) + + outCurrencies.SR3_HOV_LOW :=Nz(outCurrencies.SR3_HOV_LOW) + outCurrencies.SR3_HOV_LOW := Nz(personCurrenciesLow.("SR3_HOV"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR3_HOV"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR3_HOV"+periods[i])) + + Nz(airportCurrenciesLow.("SR3_HOV"+periods[i])) + + Nz(visitorCurrenciesLow.("SR3_HOV"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SR3_HOV_LOW := outCurrencies.SR3_HOV_LOW + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) + if votBinExternalExternal=1 then outCurrencies.SR3_HOV_LOW := outCurrencies.SR3_HOV_LOW + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] + + outCurrencies.SR3_PAY_LOW :=Nz(outCurrencies.SR3_PAY_LOW) + outCurrencies.SR3_PAY_LOW := Nz(personCurrenciesLow.("SR3_PAY"+periods[i])) + + Nz(internalExternalCurrenciesLow.("SR3_PAY"+periods[i])) + + Nz(crossBorderCurrenciesLow.("SR3_PAY"+periods[i])) + + Nz(airportCurrenciesLow.("SR3_PAY"+periods[i])) + + Nz(visitorCurrenciesLow.("SR3_PAY"+periods[i])) + + if votBinExternalInternal=1 then outCurrencies.SR3_PAY_LOW := outCurrencies.SR3_PAY_LOW + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) + +//MED VOT + // calculate output matrices + outCurrencies.SOV_GP_MED :=Nz(outCurrencies.SOV_GP_MED) + outCurrencies.SOV_GP_MED := Nz(personCurrenciesMed.("SOV_GP"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SOV_GP"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SOV_GP"+periods[i])) + + Nz(airportCurrenciesMed.("SOV_GP"+periods[i])) + + Nz(visitorCurrenciesMed.("SOV_GP"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) + if votBinExternalExternal=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) + if votBinCommercialVehicles=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + Nz(commVehCurrencies.(commVehMatrices[i])) + + outCurrencies.SOV_PAY_MED :=Nz(outCurrencies.SOV_PAY_MED) + outCurrencies.SOV_PAY_MED := Nz(personCurrenciesMed.("SOV_PAY"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SOV_PAY"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SOV_PAY"+periods[i])) + + Nz(airportCurrenciesMed.("SOV_PAY"+periods[i])) + + Nz(visitorCurrenciesMed.("SOV_PAY"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SOV_PAY_MED := outCurrencies.SOV_PAY_MED + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) + if votBinCommercialVehicles=2 then outCurrencies.SOV_PAY_MED := outCurrencies.SOV_PAY_MED + Nz(commVehCurrencies.(commVehMatrices[i+5])) + + outCurrencies.SR2_GP_MED :=Nz(outCurrencies.SR2_GP_MED) + outCurrencies.SR2_GP_MED := Nz(personCurrenciesMed.("SR2_GP"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR2_GP"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR2_GP"+periods[i])) + + Nz(airportCurrenciesMed.("SR2_GP"+periods[i])) + + Nz(visitorCurrenciesMed.("SR2_GP"+periods[i])) + + outCurrencies.SR2_HOV_MED :=Nz(outCurrencies.SR2_HOV_MED) + outCurrencies.SR2_HOV_MED := Nz(personCurrenciesMed.("SR2_HOV"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR2_HOV"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR2_HOV"+periods[i])) + + Nz(airportCurrenciesMed.("SR2_HOV"+periods[i])) + + Nz(visitorCurrenciesMed.("SR2_HOV"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SR2_HOV_MED := outCurrencies.SR2_HOV_MED + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) + if votBinExternalExternal=2 then outCurrencies.SR2_HOV_MED := outCurrencies.SR2_HOV_MED + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] + + outCurrencies.SR2_PAY_MED :=Nz(outCurrencies.SR2_PAY_MED) + outCurrencies.SR2_PAY_MED := Nz(personCurrenciesMed.("SR2_PAY"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR2_PAY"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR2_PAY"+periods[i])) + + Nz(airportCurrenciesMed.("SR2_PAY"+periods[i])) + + Nz(visitorCurrenciesMed.("SR2_PAY"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SR2_PAY_MED := outCurrencies.SR2_PAY_MED + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) + + outCurrencies.SR3_GP_MED :=Nz(outCurrencies.SR3_GP_MED) + outCurrencies.SR3_GP_MED := Nz(personCurrenciesMed.("SR3_GP"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR3_GP"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR3_GP"+periods[i])) + + Nz(airportCurrenciesMed.("SR3_GP"+periods[i])) + + Nz(visitorCurrenciesMed.("SR3_GP"+periods[i])) + + outCurrencies.SR3_HOV_MED :=Nz(outCurrencies.SR3_HOV_MED) + outCurrencies.SR3_HOV_MED := Nz(personCurrenciesMed.("SR3_HOV"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR3_HOV"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR3_HOV"+periods[i])) + + Nz(airportCurrenciesMed.("SR3_HOV"+periods[i])) + + Nz(visitorCurrenciesMed.("SR3_HOV"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SR3_HOV_MED := outCurrencies.SR3_HOV_MED + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) + if votBinExternalExternal=2 then outCurrencies.SR3_HOV_MED := outCurrencies.SR3_HOV_MED + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] + + outCurrencies.SR3_PAY_MED :=Nz(outCurrencies.SR3_PAY_MED) + outCurrencies.SR3_PAY_MED := Nz(personCurrenciesMed.("SR3_PAY"+periods[i])) + + Nz(internalExternalCurrenciesMed.("SR3_PAY"+periods[i])) + + Nz(crossBorderCurrenciesMed.("SR3_PAY"+periods[i])) + + Nz(airportCurrenciesMed.("SR3_PAY"+periods[i])) + + Nz(visitorCurrenciesMed.("SR3_PAY"+periods[i])) + + if votBinExternalInternal=2 then outCurrencies.SR3_PAY_MED := outCurrencies.SR3_PAY_MED + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) + +//HIGH VOT + // calculate output matrices + outCurrencies.SOV_GP_HIGH :=Nz(outCurrencies.SOV_GP_HIGH) + outCurrencies.SOV_GP_HIGH := Nz(personCurrenciesHigh.("SOV_GP"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SOV_GP"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SOV_GP"+periods[i])) + + Nz(airportCurrenciesHigh.("SOV_GP"+periods[i])) + + Nz(visitorCurrenciesHigh.("SOV_GP"+periods[i])) + + if votBinExternalInternal=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) + if votBinExternalExternal=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) + if votBinCommercialVehicles=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + Nz(commVehCurrencies.(commVehMatrices[i])) + + outCurrencies.SOV_PAY_HIGH :=Nz(outCurrencies.SOV_PAY_HIGH) + outCurrencies.SOV_PAY_HIGH := Nz(personCurrenciesHigh.("SOV_PAY"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SOV_PAY"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SOV_PAY"+periods[i])) + + Nz(airportCurrenciesHigh.("SOV_PAY"+periods[i])) + + Nz(visitorCurrenciesHigh.("SOV_PAY"+periods[i])) + + if votBinExternalInternal=3 then outCurrencies.SOV_PAY_HIGH := outCurrencies.SOV_PAY_HIGH + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) + if votBinCommercialVehicles=3 then outCurrencies.SOV_PAY_HIGH := outCurrencies.SOV_PAY_HIGH + Nz(commVehCurrencies.(commVehMatrices[i+5])) + + outCurrencies.SR2_GP_HIGH :=Nz(outCurrencies.SR2_GP_HIGH) + outCurrencies.SR2_GP_HIGH := Nz(personCurrenciesHigh.("SR2_GP"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR2_GP"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR2_GP"+periods[i])) + + Nz(airportCurrenciesHigh.("SR2_GP"+periods[i])) + + Nz(visitorCurrenciesHigh.("SR2_GP"+periods[i])) + + outCurrencies.SR2_HOV_HIGH :=Nz(outCurrencies.SR2_HOV_HIGH) + outCurrencies.SR2_HOV_HIGH := Nz(personCurrenciesHigh.("SR2_HOV"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR2_HOV"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR2_HOV"+periods[i])) + + Nz(airportCurrenciesHigh.("SR2_HOV"+periods[i])) + + Nz(visitorCurrenciesHigh.("SR2_HOV"+periods[i])) + + if votBinExternalInternal=3 then outCurrencies.SR2_HOV_HIGH := outCurrencies.SR2_HOV_HIGH + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) + if votBinExternalExternal=3 then outCurrencies.SR2_HOV_HIGH := outCurrencies.SR2_HOV_HIGH + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] + + outCurrencies.SR2_PAY_HIGH :=Nz(outCurrencies.SR2_PAY_HIGH) + outCurrencies.SR2_PAY_HIGH := Nz(personCurrenciesHigh.("SR2_PAY"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR2_PAY"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR2_PAY"+periods[i])) + + Nz(airportCurrenciesHigh.("SR2_PAY"+periods[i])) + + Nz(visitorCurrenciesHigh.("SR2_PAY"+periods[i])) + + if votBinExternalInternal=3 then outCurrencies.SR2_PAY_HIGH := outCurrencies.SR2_PAY_HIGH + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) + + outCurrencies.SR3_GP_HIGH :=Nz(outCurrencies.SR3_GP_HIGH) + outCurrencies.SR3_GP_HIGH := Nz(personCurrenciesHigh.("SR3_GP"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR3_GP"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR3_GP"+periods[i])) + + Nz(airportCurrencies.("SR3_GP"+periods[i])) + + Nz(visitorCurrencies.("SR3_GP"+periods[i])) + + outCurrencies.SR3_HOV_HIGH :=Nz(outCurrencies.SR3_HOV_HIGH) + outCurrencies.SR3_HOV_HIGH := Nz(personCurrenciesHigh.("SR3_HOV"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR3_HOV"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR3_HOV"+periods[i])) + + Nz(airportCurrenciesHigh.("SR3_HOV"+periods[i])) + + Nz(visitorCurrenciesHigh.("SR3_HOV"+periods[i])) + + + if votBinExternalInternal=3 then outCurrencies.SR3_HOV_HIGH := outCurrencies.SR3_HOV_HIGH + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) + if votBinExternalExternal=3 then outCurrencies.SR3_HOV_HIGH := outCurrencies.SR3_HOV_HIGH + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] + + outCurrencies.SR3_PAY_HIGH :=Nz(outCurrencies.SR3_PAY_HIGH) + outCurrencies.SR3_PAY_HIGH := Nz(personCurrenciesHigh.("SR3_PAY"+periods[i])) + + Nz(internalExternalCurrenciesHigh.("SR3_PAY"+periods[i])) + + Nz(crossBorderCurrenciesHigh.("SR3_PAY"+periods[i])) + + Nz(airportCurrenciesHigh.("SR3_PAY"+periods[i])) + + Nz(visitorCurrenciesHigh.("SR3_PAY"+periods[i])) + + if votBinExternalInternal=3 then outCurrencies.SR3_PAY_HIGH := outCurrencies.SR3_PAY_HIGH + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) + + outCurrencies.lhdn := truckCurrencies[i].lhdn + outCurrencies.mhdn := truckCurrencies[i].mhdn + outCurrencies.hhdn := truckCurrencies[i].hhdn + outCurrencies.lhdt := truckCurrencies[i].lhdt + outCurrencies.mhdt := truckCurrencies[i].mhdt + outCurrencies.hhdt := truckCurrencies[i].hhdt + end + RunMacro("close all" ) + quit: + Return(1 ) + +EndMacro + + +/************************************************************** + CreateTransitTables + + Inputs + input\tranTrips_XX.mtx + input\tranAirportTrips_XX.mtx + + where XX is period = {_EA,_AM,_MD,_PM,_EV} + + + Outputs + output\tranTotalTrips_XX.mtx + + TODO: Add Mexican resident and visitor trips + +*******************************************************************/ +Macro "Create Transit Tables" + + shared path, inputDir, outputDir + + periods={"_EA","_AM","_MD","_PM","_EV"} + + dim personFiles[periods.length] + dim airportFiles[periods.length] + dim crossBorderFiles[periods.length] + dim visitorFiles[periods.length] + dim internalExternalFiles[periods.length] + dim outFiles[periods.length] + + + for i = 1 to periods.length do + personFiles[i] = outputDir+"\\tranTrips"+periods[i]+".mtx" + airportFiles[i] = outputDir+"\\tranAirportTrips"+periods[i]+".mtx" + crossBorderFiles[i] = outputDir+"\\tranCrossBorderTrips"+periods[i]+".mtx" + visitorFiles[i] = outputDir+"\\tranVisitorTrips"+periods[i]+".mtx" + internalExternalFiles[i] = outputDir+"\\tranInternalExternalTrips"+periods[i]+".mtx" + outFiles[i] = outputDir+"\\tranTotalTrips"+periods[i]+".mtx" + end + + + //the following table names have the period appended + tableNames = {"WLK_LOC","WLK_EXP","WLK_BRT","WLK_LRT","WLK_CMR","PNR_LOC","PNR_EXP","PNR_BRT","PNR_LRT","PNR_CMR","KNR_LOC","KNR_EXP","KNR_BRT","KNR_LRT","KNR_CMR"} + + + segments = {"CTRAMP","Airport","Visitor","CrossBorder","Int-Ext","Total"} + numberSegments = segments.length + dim statistics[numberSegments] + dim totalTrips[numberSegments,tableNames.length] + + for i = 1 to periods.length do + + //open person trip matrix currencies + personMatrix = OpenMatrix(personFiles[i], ) + personCurrencies = CreateMatrixCurrencies(personMatrix, , , ) + + //open airport matrix currencies + airportMatrix = OpenMatrix(airportFiles[i], ) + airportCurrencies = CreateMatrixCurrencies(airportMatrix, , , ) + + //open visitor matrix currencies + visitorMatrix = OpenMatrix(visitorFiles[i], ) + visitorCurrencies = CreateMatrixCurrencies(visitorMatrix, , , ) + + //open crossBorder matrix currencies + crossBorderMatrix = OpenMatrix(crossBorderFiles[i], ) + crossBorderCurrencies = CreateMatrixCurrencies(crossBorderMatrix, , , ) + + //open internalExternal matrix currencies + internalExternalMatrix = OpenMatrix(internalExternalFiles[i], ) + internalExternalCurrencies = CreateMatrixCurrencies(internalExternalMatrix, , , ) + + + dim curr_array[tableNames.length] + for j = 1 to curr_array.length do + curr_array[j] = CreateMatrixCurrency(personMatrix, tableNames[j]+periods[i], ,, ) + end + + //create output trip table and matrix currencies for this time period + outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outFiles[i]}, + {"Label", outFiles[i]}, + {"Tables",tableNames}, + {"Compression",0}, + {"File Based", "Yes"}}) + SetMatrixCoreNames(outMatrix, tableNames) + + outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) + + + // calculate output matrices + for j = 1 to tableNames.length do + + outCurrencies.(tableNames[j]) := personCurrencies.(tableNames[j]+periods[i]) + + airportCurrencies.(tableNames[j]+periods[i]) + + visitorCurrencies.(tableNames[j]+periods[i]) + + crossBorderCurrencies.(tableNames[j]+periods[i]) + + internalExternalCurrencies.(tableNames[j]+periods[i]) + + end + + //for reporting + statistics[1] = MatrixStatistics(personMatrix,) + statistics[2] = MatrixStatistics(airportMatrix,) + statistics[3] = MatrixStatistics(visitorMatrix,) + statistics[4] = MatrixStatistics(crossBorderMatrix,) + statistics[5] = MatrixStatistics(internalExternalMatrix,) + statistics[6] = MatrixStatistics(outMatrix,) + + // calculate totals and save in arrays + for j = 1 to tableNames.length do + + for k = 1 to numberSegments do + + // Sum the tables in the person trip file + + if k<6 then totalTrips[k][j] = statistics[k].(tableNames[j]+periods[i]).Sum else totalTrips[k][j] = statistics[k].(tableNames[j]).Sum + end + end + + //write the table for inputs to the report file + AppendToReportFile(0, "Transit Factoring for Period "+periods[i], {{"Section", "True"}}) + fileColumn = { {"Name", "File"}, {"Percentage Width", 20}, {"Alignment", "Left"}} + modeColumns = null + for j = 1 to tableNames.length do + modeColumns = modeColumns + { { {"Name", tableNames[j]}, {"Percentage Width", (100-20)/tableNames.length}, {"Alignment", "Left"}, {"Decimals", 0} } } + end + columns = {fileColumn} + modeColumns + AppendTableToReportFile( columns, {{"Title", "Transit Factor Input File Totals"}}) + + for j = 1 to numberSegments do + outRow = null + for k = 1 to tableNames.length do + outRow = outRow + {totalTrips[j][k] } + end + outRow = { segments[j] } + outRow + AppendRowToReportFile(outRow,) + end + + CloseReportFileSection() + + + + end + RunMacro("close all" ) + quit: + Return(1 ) + +EndMacro +/************************************************************** +Create TOD Tables From 4Step Model + + TransCAD Macro used to create trip tables from 4-step model + for assignment to 5 tod networks for first iteration of AB + model. Only needs to be run once for any given scenario year. + + Inputs + input\trptollam2.mtx + input\trptollop2.mtx + input\trptollpm2.mtx + + Outputs + output\Trip_EA.mtx + output\Trip_AM.mtx + output\Trip_MD.mtx + output\Trip_PM.mtx + output\Trip_EV.mtx + +Each matrix has the following cores: + SOV_GP SOV General purpose lanes + SOV_PAY SOV Toll eligible + SR2_GP SR2 General purpose lanes + SR2_HOV SR2 HOV lanes + SR2_PAY SR2 Toll eligible + SR3_GP SR3 General purpose lanes + SR3_HOV SR3 HOV lanes + SR3_PAY SR3 Toll eligible + lhdn Light heavy duty general purpose lanes + mhdn Medium heavy duty general purpose lanes + hhdn Heavy heavy duty general purpose lanes + lhdt Light heavy duty toll eligibl + mhdt Medium heavy duty general purpose lanes + hhdt Heavy heavy duty general purpose lanes + +**************************************************************/ +Macro "Create TOD Tables From 4Step Model" + + shared path, inputDir, outputDir + + /* + inputDir = "c:\\projects\\sandag\\series12\\base2008\\input" + outputDir = "c:\\projects\\sandag\\series12\\base2008\\output" + */ + + //inputs + amMatrixName = "trptollam2.mtx" + opMatrixName = "trptollop2.mtx" + pmMatrixName = "trptollpm2.mtx" + + inTableNames = {"dan", "dat", "s2nn", "s2nh", "s2th", "M1", "M2", "M3", "lhdn","mhdn","hhdn", "lhdt","mhdt","hhdt"} + + + //output files + outMatrixNames = {"Trip_EA.mtx", "Trip_AM.mtx", "Trip_MD.mtx", "Trip_PM.mtx", "Trip_EV.mtx"} + outTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} + + // open input matrices + amMatrix = OpenMatrix(inputDir+"\\"+amMatrixName, ) + opMatrix = OpenMatrix(inputDir+"\\"+opMatrixName, ) + pmMatrix = OpenMatrix(inputDir+"\\"+pmMatrixName, ) + + // create input matrix currencies + dim inCurrencies[outMatrixNames.length,inTableNames.length] + for i = 1 to inTableNames.length do + inCurrencies[1][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + inCurrencies[2][i] = CreateMatrixCurrency(amMatrix, inTableNames[i], ,, ) + inCurrencies[3][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + inCurrencies[4][i] = CreateMatrixCurrency(pmMatrix, inTableNames[i], ,, ) + inCurrencies[5][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + end + + + // create the output matrices, copying the matrix structure of the input matrices. Then rename the matrices. + dim outMatrices[outMatrixNames.length] + dim outCurrencies[outMatrixNames.length,outTableNames.length] + for i = 1 to outMatrixNames.length do + + outMatrices[i] = CopyMatrixStructure(inCurrencies[i], {{"File Name", outputDir+"\\"+outMatrixNames[i]}, + {"Label", outMatrixNames[i]}, + {"Tables",outTableNames}, + {"File Based", "Yes"}}) + SetMatrixCoreNames(outMatrices[i], outTableNames) + + for j = 1 to outTableNames.length do + outCurrencies[i][j] = CreateMatrixCurrency(outMatrices[i], outTableNames[j], ,, ) + end + end + + // factor the off-peak input table to 3 time periods (1=EA,3=MD,5=PM, and the factors are generic across input tables) + for i = 1 to outTableNames.length do + outCurrencies[1][i] := inCurrencies[1][i] * 0.10 + outCurrencies[2][i] := inCurrencies[2][i] + outCurrencies[3][i] := inCurrencies[3][i] * 0.65 + outCurrencies[4][i] := inCurrencies[4][i] + outCurrencies[5][i] := inCurrencies[5][i] * 0.25 + end + + Return(1) + + + +EndMacro + + +/*************************************************************************************************************************** + + + +****************************************************************************************************************************/ +Macro "Create EE & EI Trips" + + + shared path, inputDir, outputDir, inputTruckDir, mxzone, mxtap, mxext,mxlink,mxrte + + //inputs + amMatrixName = "trptollam2.mtx" + opMatrixName = "trptollop2.mtx" + pmMatrixName = "trptollpm2.mtx" + + inTableNames = {"dan", "dat", "s2nn", "s2nh", "s2th", "M1", "M2", "M3", "lhdn","mhdn","hhdn", "lhdt","mhdt","hhdt"} + + //output files + outMatrixNames = {"ExtTrip_EA.mtx", "ExtTrip_AM.mtx", "ExtTrip_MD.mtx", "ExtTrip_PM.mtx", "ExtTrip_EV.mtx"} + outTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} + + // open input matrices + amMatrix = OpenMatrix(inputDir+"\\"+amMatrixName, ) + opMatrix = OpenMatrix(inputDir+"\\"+opMatrixName, ) + pmMatrix = OpenMatrix(inputDir+"\\"+pmMatrixName, ) + + // create input matrix currencies + dim inCurrencies[outMatrixNames.length,inTableNames.length] + for i = 1 to inTableNames.length do + inCurrencies[1][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + inCurrencies[2][i] = CreateMatrixCurrency(amMatrix, inTableNames[i], ,, ) + inCurrencies[3][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + inCurrencies[4][i] = CreateMatrixCurrency(pmMatrix, inTableNames[i], ,, ) + inCurrencies[5][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) + end + + //create an array of internal zone ids + dim zones[mxzone] + for i = 1 to zones.length do + zones[i]=i2s(i+mxext) + end + // create the output matrices, copying the matrix structure of the input matrices. Then rename the matrices. + dim outMatrices[outMatrixNames.length] + dim outCurrencies[outMatrixNames.length,outTableNames.length] + for i = 1 to outMatrixNames.length do + + outMatrices[i] = CopyMatrixStructure(inCurrencies[i], {{"File Name", outputDir+"\\"+outMatrixNames[i]}, + {"Label", outMatrixNames[i]}, + {"Tables",outTableNames}, + {"File Based", "Yes"}}) + SetMatrixCoreNames(outMatrices[i], outTableNames) + + for j = 1 to outTableNames.length do + outCurrencies[i][j] = CreateMatrixCurrency(outMatrices[i], outTableNames[j], ,, ) + + // set the output matrix to the input matrix + outCurrencies[i][j] := inCurrencies[i][j] + + //set the internal-internal values to 0 + FillMatrix(outCurrencies[i][j], zones, zones, {"Copy", 0.0},) + + end + end + + // factor the off-peak input table to 3 time periods (1=EA,3=MD,5=PM, and the factors are generic across input tables) + for i = 1 to outTableNames.length do + outCurrencies[1][i] := outCurrencies[1][i] * 0.10 + outCurrencies[2][i] := outCurrencies[2][i] + outCurrencies[3][i] := outCurrencies[3][i] * 0.65 + outCurrencies[4][i] := outCurrencies[4][i] + outCurrencies[5][i] := outCurrencies[5][i] * 0.25 + end + RunMacro("close all" ) + quit: + Return(1 ) + + +EndMacro +/*************************************************************************************************************************** + +Create external-external trip table -- modified 09/16/16 YMA + +****************************************************************************************************************************/ + +Macro "Create External-External Trip Matrix" // modified "externalExternalTripsByYear.csv" input and code, 09/14/16 YMA + + shared path, inputDir, outputDir, mxzone, mxext, scenarioYear + + //TODO: open external-external matrix here + //externalExternalFileName = inputDir+"\\externalExternalTrips.csv" // old input file + externalExternalFileName = inputDir+"\\externalExternalTripsByYear.csv" + externalExternalMatrixName = outputDir + "\\externalExternalTrips.mtx" + + extExtView = OpenTable("extExt", "CSV", {externalExternalFileName}, {{"Shared", "True"}}) + + opts = {} + opts.Label = "Trips" + opts.Type = "Float" + opts.Tables = {"2012","2014","2016","2017","2020","2025","2030","2035","2040","2045","2050"} + opts.[File Name] = externalExternalMatrixName + + extMatrix = CreateMatrixFromScratch(externalExternalMatrixName,mxzone,mxzone,opts) + coreNames = GetMatrixCoreNames(extMatrix ) + + for c = 1 to coreNames.length do + + if s2i(coreNames[c]) = s2i(scenarioYear) then do + + extCurren = CreateMatrixCurrency(extMatrix, coreNames[c], , , ) + extCurren := 0 + + rec = LocateRecord(extExtView+"|","year", {scenarioYear},{{"Exact", "True"}}) + + while rec <> null do + rec_vals = GetRecordValues(extExtView, rec, {"year","originTaz","destinationTaz","Trips"}) + year = rec_vals[1][2] + if year = s2i(scenarioYear) then do + originTaz = rec_vals[2][2] + destinationTaz = rec_vals[3][2] + Trips = rec_vals[4][2] + SetMatrixValue(extCurren, i2s(originTaz), i2s(destinationTaz), Trips) + rec= GetNextrecord(extExtView+"|",rec ,{{"year", "Ascending"},{"originTaz", "Ascending"}}) + end + else do + rec=null + end + end + SetMatrixCoreName(extMatrix, coreNames[c], "Trips") + end + else DropMatrixCore(extMatrix,coreNames[c]) + end + + + RunMacro("close all" ) + quit: + Return(1 ) +EndMacro + + diff --git a/sandag_abm/src/main/gisdk/createtrnroutes.rsc b/sandag_abm/src/main/gisdk/createtrnroutes.rsc new file mode 100644 index 0000000..ae1f5e9 --- /dev/null +++ b/sandag_abm/src/main/gisdk/createtrnroutes.rsc @@ -0,0 +1,655 @@ +/****************************************************************************** + +Create all transit + +update travel time with congested time from adjuststr.bin - Macro "update travel time" +then create transit network from selection set of routes - Macro"create transit networks" +(input: mode.dbf include fields: mode_id, mode_name, fare, fare_type, fare_field ) + +c********************************************************************************/ + +Macro "Create all transit" + shared path, inputDir, outputDir, mxtap + + ok=RunMacro("Import transit layer",{}) + if !ok then goto quit + + ok=RunMacro("Add transit time fields") + if !ok then goto quit + + ok=RunMacro("Update stop xy") + if !ok then goto quit + + ok=RunMacro("Create transit routes") + if !ok then goto quit + + ok=RunMacro("calc preload") + if !ok then goto quit + + ok=RunMacro("update preload fields") + if !ok then goto quit + + ok = RunMacro("TCB Run Macro", 1, "update headways",{}) + if !ok then goto quit + + RunMacro("close all") + + ok=1 + quit: + return(ok) + +EndMacro + +/************************************************************************************ + +Import transit layer + +import e00 and export to geo file + +Inputs + input\trcov.e00 + +Outputs + output\transit.dbd + + +************************************************************************************/ + +Macro "Import transit layer" + shared path, inputDir, outputDir, mxtap + + //check e00 file exists + di = GetDirectoryInfo(inputDir+"\\trcov.e00", "File") + if di.length = 0 then do + RunMacro("TCB Error", "trcov.e00 doesn't exist") + return(0) + end + + + ImportE00(inputDir + "\\trcov.e00", outputDir + "\\trtmp.dbd","line",outputDir+ "\\trtmp.bin",{ + {"Label","transit line file"}, + {"Layer Name","transit"}, + {"optimize","True"}, + {"Median Split", "True"}, + {"Node Layer Name", "Endpoints"}, + {"Node Table", outputDir + "\\trtmp_.bin"}, + {"Projection","NAD83:406",{"_cdist=1000","_limit=1000","units=us-ft"}}, + }) + + //open the intermediate transit line layer geo file + map = RunMacro("G30 new map", outputDir + "\\trtmp.dbd","False") + SetLayer("transit") + allflds=GetFields("transit","All") + fullflds=allflds[2] + allnodeflds = GetFields("endpoints", "All") + + // need to specify full field specifications + lineidfield = "transit.trcov-id"//arcinfo id field + nodeidfield = "endpoints.tnode"//for centroids purposes + + opts = {{"Layer Name", "transit"}, + {"File Type", "FFB"}, + {"ID Field", lineidfield}, + {"Field Spec", fullflds}, + {"Indexed Fields", {fullflds[1]}}, + {"Label", "transit line file"}, + {"Node layer name","trnode"}, + {"Node ID Field", nodeidfield}, + {"Node Field Spec", allnodeflds[2]}} + + if node_idx > 1 then + opts = opts + {{"Node ID Field", node_aflds[2][node_idx - 1]}} + + ExportGeography("transit",outputDir + "\\transit.dbd",opts) + + RunMacro("close all") + DeleteDatabase(outputDir+"\\trtmp.dbd") + + return(1) + + quit: + return(0) +EndMacro + +/******************************************************************************************* + +Add transit time fields + +Adds fields to the transit line layer. Local and premium time fields are added for +each period and direction. The field names are + +xxField_yy where + +Field is LOCTIME (local transit time) or PRETIME (premium transit time) +Field is TM + +xx is AB or BA +yy is period + EA: Early AM + AM: AM peak + MD: Midday + PM: PM peak + EV: Evening + +Note: this should be replaced by a revised transit network + +*******************************************************************************************/ +Macro "Add transit time fields" + + shared path, inputDir, outputDir + db_file = outputDir+"\\transit.dbd" + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + db_link_lyr = db_file + "|" + link_lyr + SetLayer(link_lyr) + vw = GetView() + strct = GetTableStructure(vw) + + // Copy the current name to the end of strct + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // Add fields to the output table + new_struct = strct + { + {"ABTM_EA", "real", 14, 6, "False",,,,,,, null}, + {"BATM_EA", "real", 14, 6, "False",,,,,,, null}, + {"ABTM_AM", "real", 14, 6, "False",,,,,,, null}, + {"BATM_AM", "real", 14, 6, "False",,,,,,, null}, + {"ABTM_MD", "real", 14, 6, "False",,,,,,, null}, + {"BATM_MD", "real", 14, 6, "False",,,,,,, null}, + {"ABTM_PM", "real", 14, 6, "False",,,,,,, null}, + {"BATM_PM", "real", 14, 6, "False",,,,,,, null}, + {"ABTM_EV", "real", 14, 6, "False",,,,,,, null}, + {"BATM_EV", "real", 14, 6, "False",,,,,,, null}} + + // Modify table structure + ModifyTable(vw, new_struct) + vw = GetView() + + // Set time fields to their respective input fields (especially needed for transit-only links) + periods = {"_EA","_AM","_MD","_PM","_EV"} + orig_periods = {"O", "A", "O", "P", "O"} + for i = 1 to periods.length do + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"ABTM"+periods[i]} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"ABTM"+orig_periods[i]} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + Opts = null + Opts.Input.[View Set] = {db_link_lyr, link_lyr} + Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} + Opts.Global.Fields = {"BATM"+periods[i]} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = {"BATM"+orig_periods[i]} + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + if !ok then goto quit + + + end + + + RunMacro("close all") + + ok=1 + quit: + return(ok) + +EndMacro + +/*************************************************************************************************************************** +Update stop xy + +Update transit node file with longitude and latitude of nearest node(?) + +Input files: + input\trstop.bin + +Output files: + output\transit.dbd + +***************************************************************************************************************************/ +Macro "Update stop xy" + shared path, inputDir, outputDir + + Opts = null + Opts.Input.[Dataview Set] = {{inputDir+"\\trstop.bin", outputDir+"\\transit.dbd|trnode", "NearNode", "ID"}, "trstop+trnode"} + Opts.Global.Fields = {"trstop.Longitude"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = "trnode.Longitude" + + ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) + + if !ok then goto quit + + // STEP 2: Fill Dataview + Opts = null + Opts.Input.[Dataview Set] = {{inputDir+"\\trstop.bin", outputDir+"\\transit.dbd|trnode", "NearNode", "ID"}, "trstop+trnode"} + Opts.Global.Fields = {"trstop.Latitude"} + Opts.Global.Method = "Formula" + Opts.Global.Parameter = "trnode.Latitude" + + ok = RunMacro("TCB Run Operation", 2, "Fill Dataview", Opts) + if !ok then goto quit + ok=1 + quit: + Return(ok) + +EndMacro + +/************************************************************************** +Create transit routes + +Create transit routes from table, and add the following fields to route table + +mode +headway +real route +fare +configdir + + +input files: + input\trlink.bin binary file for routes with link-id numbers + rte_number: sequential route number + link_id: node id + direction: +/- + input\trstop.bin stop table + input\trrt.bin route table + output\transit.dbd transit line layer with the updated congested travel time + +output files: + output\transitrt.rts transit route file + +***************************************************************************/ +Macro "Create transit routes" + shared path, inputDir, outputDir + + //check input bin files exist + lk_tb=inputDir+"\\trlink.bin" + stp_tb=inputDir+"\\trstop.bin" + rte_tb=inputDir+"\\trrt.bin" + + fnm=lk_tb + di = GetDirectoryInfo(fnm, "File") + if di.length = 0 then do + ok=0 + RunMacro("TCB Error",fnm +"does not exist!") + goto quit + end + + fnm=stp_tb + di = GetDirectoryInfo(fnm, "File") + if di.length = 0 then do + ok=0 + RunMacro("TCB Error",fnm +"does not exist!") + goto quit + end + + fnm=rte_tb + di = GetDirectoryInfo(fnm, "File") + if di.length = 0 then do + ok=0 + RunMacro("TCB Error",fnm +"does not exist!") + goto quit + end + + // delete any old index file left from last time + fnm=outputDir+"\\trlink.bx" + ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit + + fnm=outputDir+"\\trstop.bx" + ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit + + fnm=outputDir+"\\trrt.bx" + ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit + + Opts = null + Opts = {{"Routes Table" , rte_tb}, + {"Stops Table", stp_tb}, + {"Stops", "Route Stops",}} + Opts.Label = "Transit Routes" + Opts.Name = "Transit Routes" + geo_path = outputDir+"\\transit.dbd" + geo_layer = "transit" + rte_file=outputDir+"\\transitrt.rts" + info = CreateRouteSystemFromTables(lk_tb, geo_path, geo_layer,rte_file , Opts) + map = RunMacro("G30 new rt map", rte_file, "False", "False",) + + RunMacro("close all") + + ok=1 + quit: + return(ok) +EndMacro +/******************************************************************** + +Create pre-load volumes on highway network from bus routes in +transit line layer. + + + +input files: hwycov.e00 - hwy line layer ESRI exchange file +output files: hwy.dbd - hwy line geographic file + hwycad.log- a log file + hwycad.err - error file with error info + +v0.1 5/28/2012 jef + +********************************************************************/ + +macro "calc preload" + shared path,inputDir,outputDir,mxzone + + db_file=outputDir+"\\transit.dbd" + rte_file=outputDir+"\\transitrt.rts" + + + periods = {"_EA","_AM","_MD","_PM","_EV"} + hours = { 3, 3, 6.5, 3.5, 5} + headway_flds = {"OP_Headway","AM_Headway","OP_Headway","PM_Headway","OP_Headway"} + + bus_pce = 3.0 + + //load transit line layer and route file + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) // line database + LinkIDField = link_lyr+".ID" + {rs_lyr, stop_lyr, ph_lyr} = RunMacro("TCB Add RS Layers", rte_file, "ALL",) // route system + if rs_lyr = null then goto quit + + //add fields for transit pce + SetLayer(link_lyr) + vw = GetView() + strct = GetTableStructure(vw) + + // Copy the current name to the end of strct + for i = 1 to strct.length do + strct[i] = strct[i] + {strct[i][1]} + end + + // Add fields to the output table + new_struct = strct + { + {"ABPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, + {"ABPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}, + {"BAPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}} + + // Modify table structure + ModifyTable(vw, new_struct) + + // query to determine valid routes + SetLayer(rs_lyr) + qry = "Select * where Mode > 0" + sel="All" + + n = SelectByQuery(sel, "Several", qry,) + CreateProgressBar("Loading...", "True") + RT_ViewSet = rs_lyr+"|" + + rh = GetFirstRecord(RT_ViewSet, null) + nRecords = GetRecordCount(rs_lyr, sel ) + count = 1 + + // loop through each route + while rh <> null do + + + // loop through periods + for i = 1 to periods.length do // for time period + + + hdwyvals = GetRecordValues(rs_lyr, rh, {"Route_ID", headway_flds[i]}) // get route headway + rtnm = GetRouteNam(rs_lyr, hdwyvals[1][2]) + + // get the links for each route + rt_links = GetRouteLinks(rs_lyr, rtnm) + msg = "Loading Route " + rtnm + " ..." + if UpdateProgressBar(msg, r2i(count/nRecords*100)) = "True" then do + ShowMessage("Execution stopped by user.") + DestroyProgressBar() + Return() + end + + // calculate bus frequency based on headway + veh_per_hour=0.0 + if (hdwyvals[2][2] <> null and hdwyvals[2][2]>0) then veh_per_hour = 60.0 / hdwyvals[2][2] // 60 / HDWY + + if veh_per_hour > 0 then do + View_Set = link_lyr + "|" + + // loop for every link along the route + for link = 1 to rt_links.length do + + // set record for the link + rh2 = LocateRecord(View_Set, LinkIDField, {rt_links[link][1], rt_links[link][2]},) + if rh2 <> null then do + + + ABFillField = "ABPRELOAD"+periods[i] + BAFillField = "BAPRELOAD"+periods[i] + + // get bus flow + fldvals = GetRecordValues(link_lyr, rh2, {ABFillField, BAFillField}) + ab_val = fldvals[1][2] + ba_val = fldvals[2][2] + + transit_pce = veh_per_hour * hours[i] * bus_pce + + if rt_links[link][2] = 1 then do // FORWARD + + if fldvals[1][2] = null then do + ab_val = transit_pce + end + else do + ab_val = fldvals[1][2] + transit_pce + end + end + else do // REVERSE + if fldvals[2][2] = null then do + ba_val = transit_pce + end + else do + ba_val = fldvals[2][2] + transit_pce + end + end + + // set the proper link id with the preload value + SetRecordValues(link_lyr, rh2, {{ABFillField, ab_val}, {BAFillField, ba_val}}) + + end + + end + end + end + + count = count + 1 + next_rcd: + rh = GetNextRecord(RT_ViewSet, null, null) + end + DestroyProgressBar() + + RunMacro("close all") + + ok=1 + quit: + return(ok) +EndMacro +/********************************************************************************************************** +Update preload fields + +Updates preload fields on the highway line layer from the transit line layer +**********************************************************************************************************/ +Macro "update preload fields" + + shared path, inputDir, outputDir + + periods = {"_EA","_AM","_MD","_PM","_EV"} + db_file=outputDir+"\\hwy.dbd" + net_file = outputDir + "\\hwy.net" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + vw = SetView(link_lyr) + + for i = 1 to periods.length do + + transitTable = outputDir+"\\transit.bin" + ABField = "ABPRELOAD"+periods[i] + BAField = "BAPRELOAD"+periods[i] + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, transitTable, {"ID"}, {"[TRCOV-ID]"}}, ABField } + Opts.Global.Fields = {"hwyline."+ABField} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if transit."+ABField+" <>null then transit."+ABField+" else 0.0" } + ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ok then goto quit + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, transitTable, {"ID"}, {"[TRCOV-ID]"}}, BAField} + Opts.Global.Fields = {"hwyline."+BAField} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if transit."+BAField+" <>null then transit."+BAField+" else 0.0" } + ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ok then goto quit + + end + + + //update the highway network with the new fields + for i = 1 to periods.length do + + field = "*PRELOAD"+periods[i] + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*PRELOAD"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABPRELOAD"+periods[i],link_lyr+".BAPRELOAD"+periods[i] } } + Opts.Global.Options.Constants = {1} + ok = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ok then goto quit + end + + RunMacro("close all") + + ok=1 + quit: + return(ok) + +EndMacro + +/********************************************************************************************************** +Update headway fields based upon Vovsha headway function + +**********************************************************************************************************/ +Macro "update headways" + + shared path, inputDir, outputDir + + db_file=outputDir+"\\transit.dbd" + rte_file=outputDir+"\\transitrt.rts" + + headway_flds = {"AM_Headway","OP_Headway","PM_Headway"} + + dim rev_headway[headway_flds.length] + + //load transit line layer and route file + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) // line database + LinkIDField = link_lyr+".ID" + {rs_lyr, stop_lyr, ph_lyr} = RunMacro("TCB Add RS Layers", rte_file, "ALL",) // route system + if rs_lyr = null then goto quit + + SetLayer(rs_lyr) + vw = GetView() + + RT_ViewSet = rs_lyr+"|" + + rh = GetFirstRecord(RT_ViewSet, null) + + // loop through each route + while rh <> null do + + hdwyvals = GetRecordValues(rs_lyr, rh, headway_flds) // get route headway + + //for each headway + for i = 1 to hdwyvals.length do + + //calculate revised headway + rev_headway[i] = RunMacro("calculate revised headway",hdwyvals[i][2]) + + // set the headways back in the route record + SetRecordValues(rs_lyr, rh, {{headway_flds[i], rev_headway[i]}}) + end + + rh = GetNextRecord(RT_ViewSet, null, null) + end + + + RunMacro("close all") + + ok=1 + quit: + Return(ok) +EndMacro + + +/***************************************************************************** + + Revised headways + +*****************************************************************************/ +Macro "calculate revised headway" (headway) + + // CALCULATE REVISED HEADWAY + + slope_1=1.0 //slope for 1st segment (high frequency) transit + slope_2=0.8 //slope for 2nd segemnt (med frequency) transit + slope_3=0.7 //slope for 3rd segment (low frequency) transit + slope_4=0.2 //slope for 4th segment (very low freq) transit + + break_1=10 //breakpoint of 1st segment, min + break_2=20 //breakpoint of 2nd segment, min + break_3=30 //breakpoint of 3rd segment, min + + + if headway < break_1 then do + rev_headway = headway * slope_1 + end + else if headway < break_2 then do + part_1_headway = break_1 * slope_1 + part_2_headway = (headway - break_1) * slope_2 + rev_headway = part_1_headway + part_2_headway + end + else if headway < break_3 then do + part_1_headway = break_1 * slope_1 + part_2_headway = (break_2 - break_1) * slope_2 + part_3_headway = (headway - break_2) * slope_3 + rev_headway = part_1_headway + part_2_headway + part_3_headway + end + else do + part_1_headway = break_1 * slope_1 + part_2_headway = (break_2 - break_1) * slope_2 + part_3_headway = (break_3 - break_2) * slope_3 + part_4_headway = (headway - break_3) * slope_4 + rev_headway = part_1_headway + part_2_headway + part_3_headway + part_4_headway + end + + Return(rev_headway) + +EndMacro + + \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/dbox.rsc b/sandag_abm/src/main/gisdk/dbox.rsc new file mode 100644 index 0000000..c111630 --- /dev/null +++ b/sandag_abm/src/main/gisdk/dbox.rsc @@ -0,0 +1,256 @@ + +dBox "Run ABM" title: "SANDAG ABM" + init do + shared path, path_study + shared scr + + path_study="${workpath}" + scen_namefile = "\\scen_name.txt" + scen_info = GetFileInfo(path_study + scen_namefile) + if scen_info<>null then do + fptr_from = OpenFile(path_study + scen_namefile, "r") + scr=readarray(fptr_from) + end + else do + scr=null + while scr=Null do + RunMacro("getpathdirectory") + end + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + end +//end of editing + scen_num={1} + + path=scr[scen_num[1]] + +enditem + +text "Study Area:" 1,0,12 +text 12,0,34,1 Framed variable: path_study + +Button "Browse..." 48.5, 3, 16 do + opts={{"Initial Directory", path_study}} + path_study = ChooseDirectory("Choose a Study Area Directory",opts) + path_fortran=path_study+"\\fortran" + path_vb=path_study+"\\vb" + scen_info = GetFileInfo(path_study + scen_namefile) + if scen_info<>null then do + fptr_from = OpenFile(path_study + scen_namefile, "r") + scr=readarray(fptr_from) + end + else do + scr=null + while scr=Null do + RunMacro("getpathdirectory") + end + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + end + scen_num={1} + path=scr[scen_num[1]] + +enditem +//end of editing + +text "Add Scenario pathes in sequence order" 1,1.5, 38 + scroll list "scens" 1, 2.5, 46, 9 list:scr variable: scen_num multiple + help: "Select one or more scenarios to run" do + if scr = null then do // if any scenario chosen + while scr=null do + RunMacro("getpathdirectory") + end + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + scen_num ={1} // first scenario chosen + if tab_indx=1 then RunMacro("enable all1") else RunMacro("enable allfdlp") //enable all the "run" buttons + end + path=scr[scen_num[1]] + +enditem + +button "Quit" 48.5, 0, 16 do + batch_run_mode= false +//added by JXu on Nov 1, 2006 + maps = GetMaps() + if maps <> null then do + for i = 1 to maps[1].length do + SetMapSaveFlag(maps[1][i],"False") + end + end + RunMacro("G30 File Close All") + mtxs = GetMatrices() + if mtxs <> null then do + handles = mtxs[1] + for i = 1 to handles.length do + handles[i] = null + end + end +//end of editing + Return() +enditem + +button "Add" same , 1.5, 7 do + RunMacro("getpathdirectory") +//added by JXu + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) +//end of editing + scen_num={1} + path=scr[scen_num[1]] + enableitem("Delete") +enditem + +// delete a scenario +button "Delete" 57.5,1.5,7 do + scr = ExcludeArrayElements(scr,scen_num[1],scen_num.length) + while scr= null do + RunMacro("getpathdirectory") + end + + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + scen_num={1} + path=scr[scen_num[1]] + +enditem + + +button "Change Study" 48.5, 4.5, 16 do + scr = null + RunMacro("getstudydirectory") + scen_info = GetFileInfo(path_study + scen_namefile) + if scen_info<>null then do + fptr_from = OpenFile(path_study + scen_namefile, "r") + scr=readarray(fptr_from) + tmp_flag=0 + for i=1 to scr.length do + if scr[i]=path then do + i=scr.length+1 + tmp_flag=1 + end + else i=i+1 + end + if tmp_flag=0 then do + scr=scr+{path} + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + end + end + else do + scr={path} + fptr_from = OpenFile(path_study + scen_namefile, "w") + WriteArray(fptr_from,scr) + closefile(fptr_from) + end + + scen_num={1} + + path=scr[scen_num[1]] +enditem + + + button "Run ABM" same, 6, 16 do + hideDbox() + RunMacro("TCB Init") + for sc = 1 to scen_num.length do + path = scr[ scen_num[sc] ] + ok = RunMacro("Run SANDAG ABM") + if !ok then goto exit + end + exit: + showdbox() + RunMacro("TCB Closing", run_ok, "False") + enditem + + + button "Export Data" same, 7.5, 16 do + hideDbox() + RunMacro("TCB Init") + for sc = 1 to scen_num.length do + path = scr[ scen_num[sc] ] + ok = RunMacro("ExportSandagData") + if !ok then goto exit + end + exit: + showdbox() + RunMacro("TCB Closing", run_ok, "False") + enditem + + button "Sum Transit Sellink" same, 9, 16 do + hideDbox() + RunMacro("TCB Init") + for sc = 1 to scen_num.length do + path = scr[ scen_num[sc] ] + ok = RunMacro("Sum Up Select Link Transit Trips") + if !ok then goto exit + end + exit: + showdbox() + RunMacro("TCB Closing", run_ok, "False") + enditem + + button "Sum Hwy Sellink" same, 10.5, 16 do + hideDbox() + RunMacro("TCB Init") + for sc = 1 to scen_num.length do + path = scr[ scen_num[sc] ] + ok = RunMacro("Sum Up Select Link Highway Trips") + if !ok then goto exit + end + exit: + showdbox() + RunMacro("TCB Closing", run_ok, "False") + enditem + + +EndDbox + +// Macro "getpathdirectory" doesn't allow the selected path with different path_study. +Macro "getpathdirectory" + shared path,path_study,scr + opts={{"Initial Directory", path_study}} + tmp_path=choosedirectory("Choose an alternative directory in the same study area", opts) + strlen=len(tmp_path) + for i = 1 to strlen do + tmp=right(tmp_path,i) + tmpx=left(tmp,1) + if tmpx="\\" then goto endfor + end + endfor: + strlenx=strlen-i + tmppath_study=left(tmp_path,strlenx) + if path_study=tmppath_study then do + path=tmp_path + tmp_flag=0 + for i=1 to scr.length do + if scr[i]=path then do + tmp_flag=1 + i=scr.length+1 + end + else i=i+1 + end + if tmp_flag=0 then do + tmp = CopyArray(scr) + tmp = tmp + {tmp_path} + scr = CopyArray(tmp) + end + //showmessage("write description of the alternative in the head file") + //x=RunProgram("notepad "+path+"\\head",) + mytime=GetDateAndTime() + + end + else do + path=null + msg1="The alternative directory selected is invalid because it has different study area! " + msg2="Please select again within the same study area " + msg3=" or use the Browse button to select a different study area." + showMessage(msg1+msg2+path_study+msg3) + end +EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/exportTCData.rsc b/sandag_abm/src/main/gisdk/exportTCData.rsc new file mode 100644 index 0000000..2b1ce4b --- /dev/null +++ b/sandag_abm/src/main/gisdk/exportTCData.rsc @@ -0,0 +1,40 @@ +Macro "ExportSandagData" + + shared path_study, path, inputDir, outputDir + + RunMacro("close all") + RunMacro("TCB Init") + + + inputDir = path+"\\input" + outputDir = path+"\\output" + reportDir = path+"\\report" + + network_file = outputDir+"\\hwy.dbd" + + output_network_file = reportDir + "\\hwy_tcad" + output_transit_onoff_file = reportDir + "\\transit_onoff" + output_transit_flow_file = reportDir + "\\transit_flow" + output_transit_aggflow_file = reportDir + "\\transit_aggflow" + input_route_file = inputDir + "\\trrt" + output_route_file = reportDir + "\\trrt" + input_stop_file = inputDir + "\\trstop" + output_stop_file = reportDir + "\\trstop" + + input_hwyload_file = RunMacro("FormPath",{outputDir,"hwyload_"}) + output_hwyload_file = RunMacro("FormPath",{reportDir,"hwyload_"}) + + external_zones = {"1","2","3","4","5","6","7","8","9","10","11","12"} + + RunMacro("ExportNetworkToCsv",network_file,output_network_file) + RunMacro("ExportHwyloadtoCSV",input_hwyload_file,output_hwyload_file) + RunMacro("ExportBintoCSV",input_route_file, output_route_file) + RunMacro("ExportBintoCSV",input_stop_file, output_stop_file) + RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildOnOffOptions"),output_transit_onoff_file) + RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildTransitFlowOptions"),output_transit_flow_file) + RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildAggFlowOptions"),output_transit_aggflow_file) + + RunMacro("G30 File Close All") + return(1) + +EndMacro diff --git a/sandag_abm/src/main/gisdk/externalInternal.rsc b/sandag_abm/src/main/gisdk/externalInternal.rsc new file mode 100644 index 0000000..d59e81d --- /dev/null +++ b/sandag_abm/src/main/gisdk/externalInternal.rsc @@ -0,0 +1,356 @@ +Macro "US to SD External Trip Model" + + shared path, inputDir, outputDir, mxext, scenarioYear + + controlTotals = "externalInternalControlTotals.csv" + + controlTotalsView = OpenTable("Control Totals", "CSV", {inputDir+"\\"+controlTotals}, {{"Shared", "True"}}) + mgraView = OpenTable("MGRA View", "CSV",{inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) + + eaDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_EA_high.mtx", ) + eaDanMC = CreateMatrixCurrencies(eaDanMatrix,,,) + eaDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_EA_high.mtx", ) + eaDatMC = CreateMatrixCurrencies(eaDatMatrix,,,) + eaS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_EA_high.mtx", ) + eaS2nhMC = CreateMatrixCurrencies(eaS2nhMatrix,,,) + eaS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_EA_high.mtx", ) + eaS2thMC = CreateMatrixCurrencies(eaS2thMatrix,,,) + eaS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_EA_high.mtx", ) + eaS3nhMC = CreateMatrixCurrencies(eaS3nhMatrix,,,) + eaS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_EA_high.mtx", ) + eaS3thMC = CreateMatrixCurrencies(eaS3thMatrix,,,) + + amDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_AM_high.mtx", ) + amDanMC = CreateMatrixCurrencies(amDanMatrix,,,) + amDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_AM_high.mtx", ) + amDatMC = CreateMatrixCurrencies(amDatMatrix,,,) + amS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_AM_high.mtx", ) + amS2nhMC = CreateMatrixCurrencies(amS2nhMatrix,,,) + amS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_AM_high.mtx", ) + amS2thMC = CreateMatrixCurrencies(amS2thMatrix,,,) + amS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_AM_high.mtx", ) + amS3nhMC = CreateMatrixCurrencies(amS3nhMatrix,,,) + amS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_AM_high.mtx", ) + amS3thMC = CreateMatrixCurrencies(amS3thMatrix,,,) + + mdDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_MD_high.mtx", ) + mdDanMC = CreateMatrixCurrencies(mdDanMatrix,,,) + mdDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_MD_high.mtx", ) + mdDatMC = CreateMatrixCurrencies(mdDatMatrix,,,) + mdS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_MD_high.mtx", ) + mdS2nhMC = CreateMatrixCurrencies(mdS2nhMatrix,,,) + mdS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_MD_high.mtx", ) + mdS2thMC = CreateMatrixCurrencies(mdS2thMatrix,,,) + mdS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_MD_high.mtx", ) + mdS3nhMC = CreateMatrixCurrencies(mdS3nhMatrix,,,) + mdS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_MD_high.mtx", ) + mdS3thMC = CreateMatrixCurrencies(mdS3thMatrix,,,) + + pmDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_PM_high.mtx", ) + pmDanMC = CreateMatrixCurrencies(pmDanMatrix,,,) + pmDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_PM_high.mtx", ) + pmDatMC = CreateMatrixCurrencies(pmDatMatrix,,,) + pmS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_PM_high.mtx", ) + pmS2nhMC = CreateMatrixCurrencies(pmS2nhMatrix,,,) + pmS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_PM_high.mtx", ) + pmS2thMC = CreateMatrixCurrencies(pmS2thMatrix,,,) + pmS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_PM_high.mtx", ) + pmS3nhMC = CreateMatrixCurrencies(pmS3nhMatrix,,,) + pmS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_PM_high.mtx", ) + pmS3thMC = CreateMatrixCurrencies(pmS3thMatrix,,,) + + evDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_EV_high.mtx", ) + evDanMC = CreateMatrixCurrencies(evDanMatrix,,,) + evDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_EV_high.mtx", ) + evDatMC = CreateMatrixCurrencies(evDatMatrix,,,) + evS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_EV_high.mtx", ) + evS2nhMC = CreateMatrixCurrencies(evS2nhMatrix,,,) + evS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_EV_high.mtx", ) + evS2thMC = CreateMatrixCurrencies(evS2thMatrix,,,) + evS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_EV_high.mtx", ) + evS3nhMC = CreateMatrixCurrencies(evS3nhMatrix,,,) + evS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_EV_high.mtx", ) + evS3thMC = CreateMatrixCurrencies(evS3thMatrix,,,) + + freeTimeCurrencies = {eaDanMC.("*STM_EA (Skim)"),eaS2nhMC.("*HTM_EA (Skim)"),eaS3nhMC.("*HTM_EA (Skim)"), + amDanMC.("*STM_AM (Skim)"),amS2nhMC.("*HTM_AM (Skim)"),amS3nhMC.("*HTM_AM (Skim)"), + mdDanMC.("*STM_MD (Skim)"),mdS2nhMC.("*HTM_MD (Skim)"),mdS3nhMC.("*HTM_MD (Skim)"), + pmDanMC.("*STM_PM (Skim)"),pmS2nhMC.("*HTM_PM (Skim)"),pmS3nhMC.("*HTM_PM (Skim)"), + evDanMC.("*STM_EV (Skim)"),evS2nhMC.("*HTM_EV (Skim)"),evS3nhMC.("*HTM_EV (Skim)")} + + tollTimeCurrencies = {eaDatMC.("*STM_EA (Skim)"),eaS2thMC.("*HTM_EA (Skim)"),eaS3thMC.("*HTM_EA (Skim)"), + amDatMC.("*STM_AM (Skim)"),amS2thMC.("*HTM_AM (Skim)"),amS3thMC.("*HTM_AM (Skim)"), + mdDatMC.("*STM_MD (Skim)"),mdS2thMC.("*HTM_MD (Skim)"),mdS3thMC.("*HTM_MD (Skim)"), + pmDatMC.("*STM_PM (Skim)"),pmS2thMC.("*HTM_PM (Skim)"),pmS3thMC.("*HTM_PM (Skim)"), + evDatMC.("*STM_EV (Skim)"),evS2thMC.("*HTM_EV (Skim)"),evS3thMC.("*HTM_EV (Skim)")} + + //mod(eaS3thMC.("s3th_EA - itoll_EA"),10000), + + tollCostCurrencies = {eaDatMC.("dat_EA - itoll_EA"),eaS2thMC.("s2t_EA - itoll_EA"),eaS3thMC.("s3t_EA - itoll_EA"), + amDatMC.("dat_AM - itoll_AM"),amS2thMC.("s2t_AM - itoll_AM"),amS3thMC.("s3t_AM - itoll_AM"), + mdDatMC.("dat_MD - itoll_MD"),mdS2thMC.("s2t_MD - itoll_MD"),mdS3thMC.("s3t_MD - itoll_MD"), + pmDatMC.("dat_PM - itoll_PM"),pmS2thMC.("s2t_PM - itoll_PM"),pmS3thMC.("s3t_PM - itoll_PM"), + evDatMC.("dat_EV - itoll_EV"),evS2thMC.("s2t_EV - itoll_EV"),evS3thMC.("s3t_EV - itoll_EV")} + + controlTaz = GetDataVector(controlTotalsView+"|", "taz", {{"Sort Order", {{"taz", "Ascending"}}}} ) + controlWrk = GetDataVector(controlTotalsView+"|", "work", {{"Sort Order", {{"taz", "Ascending"}}}} ) + controlNon = GetDataVector(controlTotalsView+"|", "nonwork", {{"Sort Order", {{"taz", "Ascending"}}}} ) + + // create mgra size vectrors + + mgraView = OpenTable("MGRA View", "CSV", {inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) + + mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + emp_public_ed = GetDataVector(mgraView+"|", "emp_public_ed", {{"Sort Order", {{"mgra", "Ascending"}}}} ) + + emp_blu = emp_const_non_bldg_prod + emp_const_non_bldg_office + emp_utilities_prod + emp_utilities_office + emp_const_bldg_prod + emp_const_bldg_office + emp_mfg_prod + emp_mfg_office + emp_whsle_whs + emp_trans + emp_svc = emp_prof_bus_svcs + emp_prof_bus_svcs_bldg_maint + emp_personal_svcs_office + emp_personal_svcs_retail + emp_edu = emp_pvt_ed_k12 + emp_pvt_ed_post_k12_oth + emp_public_ed + emp_gov = emp_state_local_gov_ent + emp_fed_non_mil + emp_fed_non_mil + emp_state_local_gov_blue + emp_state_local_gov_white + emp_ent = emp_amusement + emp_hotel + emp_restaurant_bar + emp_oth = emp_religious + emp_pvt_hh + emp_fed_mil + + wrk_size_mgra = ( emp_blu + + 1.364 * emp_retail + + 4.264 * emp_ent + + 0.781 * emp_svc + + 1.403 * emp_edu + + 1.779 * emp_health + + 0.819 * emp_gov + + 0.708 * emp_oth ) + + non_size_mgra = ( hh + + 1.069 * emp_blu + + 4.001 * emp_retail + + 6.274 * emp_ent + + 0.901 * emp_svc + + 1.129 * emp_edu + + 2.754 * emp_health + + 1.407 * emp_gov + + 0.304 * emp_oth ) + + // aggregate to TAZ + + rowLabels = GetMatrixRowLabels(mdDatMC.("Length (Skim)")) + numZones = rowLabels.length + dim wrkSizeTaz[numZones] + dim nonSizeTaz[numZones] + + for i=1 to numZones do + wrkSizeTaz[i] = 0 + nonSizeTaz[i] = 0 + end + + for i=1 to mgra.length do + tazNumber = taz[i] + wrkSizeTaz[tazNumber] = wrkSizeTaz[tazNumber] + wrk_size_mgra[i] + nonSizeTaz[tazNumber] = nonSizeTaz[tazNumber] + non_size_mgra[i] + end + + wrkSizeVector = ArrayToVector(wrkSizeTaz,{{"Row Based","False"},{"Type","Float"}} ) + nonSizeVector = ArrayToVector(nonSizeTaz,{{"Row Based","False"},{"Type","Float"}}) + + + // Initialize matrices + + opts = {} + opts.Label = "Trips" + opts.Type = "Double" + opts.Tables = {'Trips'} + + opts.[File Name] = outputDir+"\\"+"usSdWrkPA.mtx" + wrkMatrixPA = CreateMatrixFromScratch("wrkMatrixPA",numZones,numZones,opts) + wrkCurrenPA = CreateMatrixCurrency(wrkMatrixPA,'Trips',,,) + + opts.[File Name] = outputDir+"\\"+"usSdNonPA.mtx" + nonMatrixPA = CreateMatrixFromScratch("nonMatrixPA",numZones,numZones,opts) + nonCurrenPA = CreateMatrixCurrency(nonMatrixPA,'Trips',,,) + + //create exponentiated distance impedance matrices + opts.Label = "Prob" + opts.Type = "Double" + opts.Tables = {'Prob'} + + opts.[File Name] = outputDir+"\\"+"wrkProb.mtx" + wrkProbMatrix = CreateMatrixFromScratch("wrkProb",numZones,numZones,opts) + wrkProb = CreateMatrixCurrency(wrkProbMatrix,'Prob',,,) + + opts.[File Name] = outputDir+"\\"+"nonProb.mtx" + nonProbMatrix = CreateMatrixFromScratch("nonProb",numZones,numZones,opts) + nonProb = CreateMatrixCurrency(nonProbMatrix,'Prob',,,) + + wrkDistCoef = -0.029 + nonDistCoef = -0.006 + + wrkProb := wrkSizeVector * exp( wrkDistCoef * mdDatMC.("Length (Skim)")) + nonProb := nonSizeVector * exp( nonDistCoef * mdDatMC.("Length (Skim)")) + + wrkSumVector = GetMatrixVector(wrkProb, {{"Marginal", "Row Sum"}}) + wrkProb := wrkProb/wrkSumVector + + nonSumVector = GetMatrixVector(nonProb, {{"Marginal", "Row Sum"}}) + nonProb := nonProb/nonSumVector + + wrkCurrenPA := 0 + nonCurrenPA := 0 + + // Loop over external zones and set values in output matrix + for i=1 to controlTaz.length do + + wrkTotal = controlWrk[i] + nonTotal = controlNon[i] + + wrkTripVector = wrkTotal * GetMatrixVector(wrkProb,{{"Row", controlTaz[i]}}) + nonTripVector = nonTotal * GetMatrixVector(wrkProb,{{"Row", controlTaz[i]}}) + + SetMatrixVector(wrkCurrenPA, wrkTripVector, {{"Row",controlTaz[i]}}) + SetMatrixVector(nonCurrenPA, nonTripVector, {{"Row",controlTaz[i]}}) + + end + + // Convert PA to OD and Apply Diurnal Factors + opts.Label = "Trips" + opts.Type = "Float" + opts.Tables = {'Trips'} + + opts.[File Name] = outputDir+"\\"+"usSdWrkDaily.mtx" + wrkMatrixAP = TransposeMatrix(wrkMatrixPA,opts) + wrkCurrenAP = CreateMatrixCurrency(wrkMatrixAP,'Trips',,,) + + opts.[File Name] = outputDir+"\\"+"usSdNonDaily.mtx" + nonMatrixAP = TransposeMatrix(nonMatrixPA,opts) + nonCurrenAP = CreateMatrixCurrency(nonMatrixAP,'Trips',,,) + + wrkCurrenPA := 0.5 * wrkCurrenPA + nonCurrenPA := 0.5 * nonCurrenPA + wrkCurrenAP := 0.5 * wrkCurrenAP + nonCurrenAP := 0.5 * nonCurrenAP + + // Apply Occupancy and Diurnal Factors + + opts.Tables = {"DAN","S2N","S3N","DAT","S2T","S3T"} + + opts.[File Name] = outputDir+"\\"+"usSdWrk_EA.mtx" + wrkMatrixEA = CreateMatrixFromScratch("wrkMatrixEA",numZones,numZones,opts) + wrkCurrenEA = CreateMatrixCurrencies(wrkMatrixEA,,,) + opts.[File Name] = outputDir+"\\"+"usSdWrk_AM.mtx" + wrkMatrixAM = CreateMatrixFromScratch("wrkMatrixAM",numZones,numZones,opts) + wrkCurrenAM = CreateMatrixCurrencies(wrkMatrixAM,,,) + opts.[File Name] = outputDir+"\\"+"usSdWrk_MD.mtx" + wrkMatrixMD = CreateMatrixFromScratch("wrkMatrixMD",numZones,numZones,opts) + wrkCurrenMD = CreateMatrixCurrencies(wrkMatrixMD,,,) + opts.[File Name] = outputDir+"\\"+"usSdWrk_PM.mtx" + wrkMatrixPM = CreateMatrixFromScratch("wrkMatrixPM",numZones,numZones,opts) + wrkCurrenPM = CreateMatrixCurrencies(wrkMatrixPM,,,) + opts.[File Name] = outputDir+"\\"+"usSdWrk_EV.mtx" + wrkMatrixEV = CreateMatrixFromScratch("wrkMatrixEV",numZones,numZones,opts) + wrkCurrenEV = CreateMatrixCurrencies(wrkMatrixEV,,,) + + opts.[File Name] = outputDir+"\\"+"usSdNon_EA.mtx" + nonMatrixEA = CreateMatrixFromScratch("nonMatrixEA",numZones,numZones,opts) + nonCurrenEA = CreateMatrixCurrencies(nonMatrixEA,,,) + opts.[File Name] = outputDir+"\\"+"usSdNon_AM.mtx" + nonMatrixAM = CreateMatrixFromScratch("nonMatrixAM",numZones,numZones,opts) + nonCurrenAM = CreateMatrixCurrencies(nonMatrixAM,,,) + opts.[File Name] = outputDir+"\\"+"usSdNon_MD.mtx" + nonMatrixMD = CreateMatrixFromScratch("nonMatrixMD",numZones,numZones,opts) + nonCurrenMD = CreateMatrixCurrencies(nonMatrixMD,,,) + opts.[File Name] = outputDir+"\\"+"usSdNon_PM.mtx" + nonMatrixPM = CreateMatrixFromScratch("nonMatrixPM",numZones,numZones,opts) + nonCurrenPM = CreateMatrixCurrencies(nonMatrixPM,,,) + opts.[File Name] = outputDir+"\\"+"usSdNon_EV.mtx" + nonMatrixEV = CreateMatrixFromScratch("nonMatrixEV",numZones,numZones,opts) + nonCurrenEV = CreateMatrixCurrencies(nonMatrixEV,,,) + + wrkCurrenAll = {wrkCurrenEA,wrkCurrenAM,wrkCurrenMD,wrkCurrenPM,wrkCurrenEV} + nonCurrenAll = {nonCurrenEA,nonCurrenAM,nonCurrenMD,nonCurrenPM,nonCurrenEV} + + wrkDiurnalPA = {0.26,0.26,0.41,0.06,0.02} + wrkDiurnalAP = {0.08,0.07,0.41,0.42,0.02} + + nonDiurnalPA = {0.25,0.39,0.30,0.04,0.02} + nonDiurnalAP = {0.12,0.11,0.37,0.38,0.02} + + wrkOccupancy = {0.58,0.31,0.11} + nonOccupancy = {0.55,0.29,0.15} + + matrixNames = {"DAN","S2N","S3N","DAT","S2T","S3T"} + + for periodIdx=1 to 5 do + for occupIdx = 1 to 3 do + + wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) := wrkOccupancy[occupIdx] * ( wrkDiurnalPA[periodIdx] * wrkCurrenPA + wrkDiurnalAP[periodIdx] * wrkCurrenAP ) + nonCurrenAll[periodIdx].(matrixNames[occupIdx]) := nonOccupancy[occupIdx] * ( nonDiurnalPA[periodIdx] * nonCurrenPA + nonDiurnalAP[periodIdx] * nonCurrenAP ) + + end + end + + // Toll choice split + + // values of time is cents per minute (toll cost is in cents) + votWork = 15.00 // $9.00/hr + votNonwork = 22.86 // $13.70/hr + ivtCoef = -0.03 + + for periodIdx=1 to 5 do + for occupIdx = 1 to 3 do + + currIndex = (periodIdx - 1) * 3 + occupIdx + + //wrkProb is work toll probability + wrkProb := if tollCostCurrencies[currIndex]>1000 then exp(ivtCoef * ( tollTimeCurrencies[ currIndex] - freeTimeCurrencies[ currIndex ] + mod(tollCostCurrencies[ currIndex ],10000) / votWork ) - 3.39) else + exp(ivtCoef * ( tollTimeCurrencies[ currIndex] - freeTimeCurrencies[ currIndex ] + tollCostCurrencies[ currIndex ] / votWork ) - 3.39) + + wrkProb := if tollCostCurrencies[ currIndex ] > 0 then wrkProb else 0 + wrkProb := wrkProb / ( 1 + wrkProb ) + + + wrkCurrenAll[periodIdx].(matrixNames[occupIdx+3]) := wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) * wrkProb + wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) := wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) * (1.0 - wrkProb) + + //nonProb is non-work toll probability + nonProb := exp(ivtCoef * ( tollTimeCurrencies[ currIndex ] - freeTimeCurrencies[ currIndex] + mod(tollCostCurrencies[ currIndex],10000) / votNonwork )- 3.39) + nonProb := if tollCostCurrencies[ currIndex ] > 0 then nonProb else 0 + nonProb := nonProb / ( 1 + nonProb ) + nonCurrenAll[periodIdx].(matrixNames[occupIdx+3]) := nonCurrenAll[periodIdx].(matrixNames[occupIdx]) * nonProb + nonCurrenAll[periodIdx].(matrixNames[occupIdx]) := nonCurrenAll[periodIdx].(matrixNames[occupIdx]) * (1.0 - nonProb) + + end + end + + RunMacro("close all") + Return(1) + quit: + Return(0) +EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/gui_generic.rsc b/sandag_abm/src/main/gisdk/gui_generic.rsc new file mode 100644 index 0000000..5692403 --- /dev/null +++ b/sandag_abm/src/main/gisdk/gui_generic.rsc @@ -0,0 +1,45 @@ + +dBox "Setup Scenario" title: "SANDAG ABM" + init do + shared path, path_study + path = "${workpath}" +enditem + +// set model run parameters +button "Set Model Parameters" 0,0,30, 2 do + RunMacro("TCB Init") + runString = "T:\\ABM\\release\\ABM\\${version}\\dist\\parameterEditor.exe "+path + RunMacro("HwycadLog",{"gui.rsc:","Create a scenario"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Create a scenario", runString) +enditem + +// run model +button "Run Model" 0, 3, 30, 2 do + //hideDbox() + RunMacro("TCB Init") + RunMacro("getpathdirectory") + pFile_info = GetFileInfo(path+'\\conf\\sandag_abm.properties') + if pFile_info=null then do + CopyFile(path+"\\conf\\sandag_abm_standard.properties", path+"\\conf\\sandag_abm.properties") + end + ok = RunMacro("Run SANDAG ABM") + if !ok then goto exit + exit: + showdbox() + RunMacro("TCB Closing", run_ok, "False") +enditem + +//exit +button "Quit" 0, 6, 30, 2 do + RunMacro("G30 File Close All") + Return() +enditem + +EndDbox + +// Macro "getpathdirectory" doesn't allow the selected path with different path_study. +Macro "getpathdirectory" + shared path,path_study,scr + opts={{"Initial Directory", path}} + path=choosedirectory("Choose a scenario folder", opts) +EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/hwyassign.rsc b/sandag_abm/src/main/gisdk/hwyassign.rsc new file mode 100644 index 0000000..d6809c5 --- /dev/null +++ b/sandag_abm/src/main/gisdk/hwyassign.rsc @@ -0,0 +1,896 @@ +/********************************************************************************* +Multi-model Multi-class Assignement +macro "hwy assignment" + +input files: hwy.dbd + hwy.net + Trip_EA.mtx: Early AM auto trip matrix file + Trip_AM.mtx: AM Peak auto trip matrix file + Trip_MD.mtx: Midday auto trip matrix file + Trip_PM.mtx: PM Peak auto trip matrix file + Trip_EV.mtx: Evening auto trip matrix file + +each file has 14 cores: + + Name Description + ------- --------------------------------------- + SOV_GP Drive Alone Non-Toll + SOV_PAY Drive Alone Toll + SR2_GP Shared-ride 2 Person Non-HOV Non-Toll + SR2_HOV Shared-ride 2 Person HOV Non-Toll + SR2_PAY Shared-ride 2 Person HOV Toll Eligible + SR3_GP Shared-ride 3+ Person Non-HOV Non-Toll + SR2_HOV Shared-ride 3+ Person HOV Non-Toll + SR2_PAY Shared-ride 3+ Person HOV Toll Eligible + lhdn Light heavy-duty Truck Non-Toll + mhdn Medium heavy-duty Truck Non-Toll + hhdnv Heavy heavy-duty Truck Non-Toll + lhdt Light heavy-duty Truck Toll + mhdt Medium heavy-duty Truck Toll + hhdt Heavy heavy-duty Truck Toll + + Functions are added by J Xu between Dec 2006 and March 2007 + (1) Select Link Analysis and split the resulting flow table by each select link inquiries; + (2) Enhanced highway assignment for four different cases, involving toll. + (3) Turning movement + +output files: + + hwyload_EA.bin: Early AM loaded network binary file + hwyload_AM.bin: Am Peak loaded network binary file + hwyload_MD.bin: Midday loaded network binary file + hwyload_PM.bin: PM Peak loaded network binary file + hwyload_EV.bin: Evening loaded network binary file + +Optionally (for select link and turning movements): + + turns_EA.bin: Early AM turning movement file + turns_AM.bin: Am Peak turning movement file + turns_MD.bin: Midday turning movement file + turns_PM.bin: PM Peak turning movement file + turns_EV.bin: Evening turning movement file + + select_EA.mtx: Early AM select link trip matrix file + select_AM.mtx: Am Peak select link trip matrix file + select_MD.mtx: Midday select link trip matrix file + select_PM.mtx: PM Peak select link trip matrix file + select_EV.mtx: Evening select link trip matrix file + + +SANDAG ABM Version 1.0 + JEF 2012-03-20 + changed linktypeturnscst.dbf to linktypeturns.dbf as Joel suggested. +*************************************************************************************/ + +Macro "hwy assignment" (args) + + Shared path, inputDir, outputDir, mxzone + + properties = "\\conf\\sandag_abm.properties" + convergence = RunMacro("read properties",properties,"convergence", "S") + assign_reliability="true" + + turn_file="\\nodes.txt" + turn_flag=0 + NumofCPU = 8 + iteration = args[1] + assignByVOT= args[2] + + // for debug + + periods = {"_EA","_AM","_MD","_PM","_EV"} + + RunMacro("close all") + dim excl_qry[periods.length],excl_toll[periods.length],excl_dat[periods.length],excl_s2nh[periods.length],excl_s2th[periods.length],excl_s3nh[periods.length] + dim excl_s3th[periods.length],excl_lhdn[periods.length],excl_mhdn[periods.length],excl_hhdn[periods.length],excl_lhdt[periods.length],excl_mhdt[periods.length] + dim excl_hhdt[periods.length],toll_fld[periods.length],toll_fld2[periods.length] + + linkt= {"*TM_EA","*TM_AM","*TM_MD","*TM_PM","*TM_EV"} + linkcap={"*CP_EA","*CP_AM","*CP_MD","*CP_PM","*CP_EV"} +// xt= {"*TX_EA","*TX_AM","*TX_MD","*TX_PM","*TX_EV"} + xcap= {"*CX_EA","*CX_AM","*CX_MD","*CX_PM","*CX_EV"} + + cycle={"*CYCLE_EA","*CYCLE_AM","*CYCLE_MD","*CYCLE_PM","*CYCLE_EV"} + pfact={"*PF_EA","*PF_AM","*PF_MD","*PF_PM","*PF_EV"} + gcrat={"*GCRATIO_EA","*GCRATIO_AM","*GCRATIO_MD","*GCRATIO_PM","*GCRATIO_EV"} + alpha1={"*ALPHA1_EA","*ALPHA1_AM","*ALPHA1_MD","*ALPHA1_PM","*ALPHA1_EV"} + beta1={"*BETA1_EA","*BETA1_AM","*BETA1_MD","*BETA1_PM","*BETA1_EV"} + alpha2={"*ALPHA2_EA","*ALPHA2_AM","*ALPHA2_MD","*ALPHA2_PM","*ALPHA2_EV"} + beta2={"*BETA2_EA","*BETA2_AM","*BETA2_MD","*BETA2_PM","*BETA2_EV"} + preload={"*PRELOAD_EA","*PRELOAD_AM","*PRELOAD_MD","*PRELOAD_PM","*PRELOAD_EV"} + statrel={"*STATREL_EA","*STATREL_AM","*STATREL_MD","*STATREL_PM","*STATREL_EV"} + + db_file = outputDir + "\\hwy.dbd" + net_file= outputDir+"\\hwy.net" + + turn={"turns_EA.bin","turns_AM.bin","turns_MD.bin","turns_PM.bin","turns_EV.bin"} + selectlink_mtx={"select_EA.mtx","select_AM.mtx","select_MD.mtx","select_PM.mtx","select_EV.mtx"} //added for select link analysis by JXu + selinkqry_file="selectlink_query.txt" + if GetFileInfo(inputDir + "\\"+ selinkqry_file) <> null then do //select link analysis is only available in stage II + selink_flag =1 + fptr_from = OpenFile(inputDir + "\\"+selinkqry_file, "r") + tmp_qry=readarray(fptr_from) + index =1 + selinkqry_name=null + selink_qry=null + subs=null + while index <=ArrayLength(tmp_qry) do + if left(trim(tmp_qry[index]),1)!="*" then do + subs=ParseString(trim(tmp_qry[index]),",") + if subs!=null then do + query=subs[3] + if ArrayLength(subs)>3 then do + for i=4 to ArrayLength(subs) do + query=query+" "+subs[2]+" "+subs[i] + end + end + selinkqry_name=selinkqry_name+{subs[1]} + selink_qry=selink_qry+{query} + end + end + index = index + 1 + end + end + + asign = {"hwyload_EA.bin","hwyload_AM.bin","hwyload_MD.bin","hwyload_PM.bin","hwyload_EV.bin"} + oue_path = {"oue_path_EA.obt", "oue_path_AM.obt","oue_path_MD.obt","oue_path_PM.obt","oue_path_EV.obt"} + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + db_link_lyr=db_file+"|"+link_lyr + db_node_lyr=db_file+"|"+node_lyr + + // drive-alone non-toll exclusion set + excl_dan={db_link_lyr, link_lyr, "dan", "Select * where !((ihov=1|ifc>7)&ITRUCK<5)"} + + // shared-2 non-toll non-HOV exclusion set + excl_s2nn=excl_dan + + // shared 3+ non-toll non-HOV exclusion set + excl_s3nn=excl_dan + + + for i = 1 to periods.length do + // drive-alone toll exclusion set + //excl_dat[i]={db_link_lyr, link_lyr, "dat", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)"} + excl_dat[i]={db_link_lyr, link_lyr, "dat", "Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)"} + + // shared-2 non-toll HOV exclusion set + excl_s2nh[i]={db_link_lyr, link_lyr, "s2nh", "Select * where !((ihov=1|(ihov=2&abln"+periods[i]+" <9)|ifc>7)&ITRUCK<5)"} + + // shared-2 toll HOV exclusion set + excl_s2th[i]={db_link_lyr, link_lyr, "s2th", "Select * where !(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)"} + + // shared=3+ non-toll non-HOV exclusion set + excl_s3nh[i]={db_link_lyr, link_lyr, "s3nh", "Select * where !((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)"} + + // shared=3+ toll HOV exclusion set + excl_s3th[i]={db_link_lyr, link_lyr, "s3th", "Select * where abln"+periods[i]+"=9|ITRUCK>4"} + + // light-heavy truck non-toll exclusion set + excl_lhdn[i]={db_link_lyr, link_lyr, "lhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK<4|ITRUCK=7))"} + + // medium-heavy truck non-toll exclusion set + excl_mhdn[i]={db_link_lyr, link_lyr, "mhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK<3|ITRUCK>5))"} + + // heavy-heavy truck non-toll exclusion set + excl_hhdn[i]={db_link_lyr, link_lyr, "hhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))"} + + // light-heavy truck toll exclusion set + //excl_lhdt[i]={db_link_lyr, link_lyr, "lhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))"} + excl_lhdt[i]={db_link_lyr, link_lyr, "lhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))"} + + // medium-heavy truck toll exclusion set + //excl_mhdt[i]={db_link_lyr, link_lyr, "mhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))"} + excl_mhdt[i]={db_link_lyr, link_lyr, "mhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))"} + + // heavy-heavy truck toll exclusion set + //excl_hhdt[i]={db_link_lyr, link_lyr, "hhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))"} + excl_hhdt[i]={db_link_lyr, link_lyr, "hhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))"} + + end + + //reset exclusion array value based on the selection set results + + set = "dat" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + for i = 1 to periods.length do + //n = SelectByQuery(set, "Several","Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)",) + n = SelectByQuery(set, "Several","Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)",) + if n = 0 then excl_dat[i]=null + end + + set = "s2nh" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + for i = 1 to periods.length do + n = SelectByQuery(set, "Several","Select * where !((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)",) + if n = 0 then excl_s2nh[i]=null + end + + set = "s2th" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + for i = 1 to periods.length do + n = SelectByQuery(set, "Several", "Select * where !(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)",) + if n = 0 then excl_s2th[i]=null + end + + set = "s3nh" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + for i = 1 to periods.length do + n = SelectByQuery(set, "Several","Select * where !((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)",) + if n = 0 then excl_s3nh[i]=null + end + + set = "s3th" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + for i = 1 to periods.length do + n = SelectByQuery(set, "Several", "Select * where abln"+periods[i]+"=9|ITRUCK>4",) + if n = 0 then excl_s3th[i]=null + end + + if assignByVOT = "false" then do + trip={"Trip_EA.mtx","Trip_AM.mtx","Trip_MD.mtx","Trip_PM.mtx","Trip_EV.mtx"} + num_class=14 + vehclass={1,2,3,4,5,6,7,8,9,10,11,12,13,14} + class_PCE={1,1,1,1,1,1,1,1,1.3,1.5,2.5,1.3,1.5,2.5} + VOT={67,67,67,67,67,67,67,67,67,68,89,67,68,89} // vot is in cents per minute: 67 cents/min = $40.20/hour + + for i = 1 to periods.length do + excl_qry[i]={excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i],excl_lhdn[i],excl_mhdn[i],excl_hhdn[i],excl_lhdt[i],excl_mhdt[i],excl_hhdt[i]} + toll_fld2[i]= {"COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","COST","ITOLL3"+periods[i],"ITOLL4"+periods[i],"ITOLL5"+periods[i]} + end + end + else do + + // for VOT assignment, explode the passenger modes 1 through 8 to 1 through 24. First 8 low vot, next 8 med vot, final 8 high vot. Then 6 truck classes for total 30 classes. + trip={"Trip"+periods[1]+"_VOT.mtx", "Trip"+periods[2]+"_VOT.mtx", "Trip"+periods[3]+"_VOT.mtx", "Trip"+periods[4]+"_VOT.mtx", "Trip"+periods[5]+"_VOT.mtx"} + num_class=30 + vehclass={1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30} + class_PCE={1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1.3,1.5,2.5,1.3,1.5,2.5} + VOT={16.6,16.6,16.6,16.6,16.6,16.6,16.6,16.6,33.3,33.3,33.3,33.3,33.3,33.3,33.3,33.3,100,100,100,100,100,100,100,100,67,68,89,67,68,89} //assuming $10/hr, $20/hr, and $60/hr as max of each range + + for i = 1 to periods.length do + excl_qry[i]={ + excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], + excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], + excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], + excl_lhdn[i],excl_mhdn[i],excl_hhdn[i],excl_lhdt[i],excl_mhdt[i],excl_hhdt[i]} + toll_fld2[i]= { + "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], + "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], + "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], + "COST","COST","COST","ITOLL3"+periods[i],"ITOLL4"+periods[i],"ITOLL5"+periods[i]} + end + end + + + + //Prepare selection set for turning movement report, by JXu + if (turn_flag=1 & iteration=4) then do + if GetFileInfo(inputDir+turn_file)!=null then do + fptr_turn = OpenFile(inputDir + turn_file,"r") + tmp_qry=readarray(fptr_turn) + turn_qry=null + index=1 + while index <=ArrayLength(tmp_qry) do + if index=1 then + turn_qry = "select * where " + "ID=" + tmp_qry[1] + else + turn_qry = turn_qry + " OR " + "ID=" + tmp_qry[index] + if tmp_qry[index]="all" or tmp_qry[index]="All" or tmp_qry[index]="ALL" then do + turn_qry = "select * where ID>"+i2s(mxzone) //select all nodes except centroids + index=ArrayLength(tmp_qry) + end + index=index+1 + end + closefile(fptr_turn) + end + else turn_qry = "select * where temp=1" + if GetFileInfo(path+"\\turn.err")!=null then do + ok=RunMacro("SDdeletefile",{path+"\\turn.err"}) + if !ok then goto quit + end + + tmpset = "turn" + vw_set = node_lyr + "|" + tmpset + SetLayer(node_lyr) + n = SelectByQuery(tmpset , "Several", turn_qry,) + if n = 0 then do + showmessage("Warning!!! No intersections selected for turning movement.") + fp_tmp = OpenFile(path + "\\turn.err","w") + WriteArray(fp_tmp,{"No intersections have been selected for turning movement."}) + closefile(fp_tmp) + return(1) + end + end + + + //set hwy.net with turn penalty of time in minutes + d_tp_tb = inputDir + "\\linktypeturns.dbf" + s_tp_tb = outputDir + "\\turns.dbf" + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Toll Set] = {db_link_lyr, link_lyr} + Opts.Input.[Centroids Set] = {db_node_lyr, node_lyr, "Selection", "select * where ID <="+i2s(mxzone)} + Opts.Global.[Spc Turn Pen Method] = 3 + Opts.Input.[Def Turn Pen Table] = {d_tp_tb} + Opts.Input.[Spc Turn Pen Table] = {s_tp_tb} + Opts.Field.[Link type] = "IFC" + Opts.Global.[Global Turn Penalties] = {0, 0, 0, 0} + Opts.Flag.[Use Link Types] = "True" + RunMacro("HwycadLog",{"hwyassign.rsc: hwy assignment","Highway Network Setting"}) + ok = RunMacro("TCB Run Operation", 1, "Highway Network Setting", Opts) + if !ok then goto quit + + // STEP 1: MMA + for i = 1 to periods.length do + + //set the vdf DLL and vdf field names based on whether reliability is being used or not +// if assign_reliability="true" then do + vdf_fields = {linkt[i], linkcap[i], xcap[i], cycle[i],pfact[i], gcrat[i], alpha1[i], beta1[i], alpha2[i], beta2[i], + "*LOSC_FACT","*LOSD_FACT","*LOSE_FACT", "*LOSFL_FACT", "*LOSFH_FACT", statrel[i], "Length", preload[i]} + vdf_file = "shrp.vdf" + vdf_defaults={ , , ,1.5 ,1 , 0.4 , 0.15, 4, 0.15, 4, 0, 0, 0, 0, 0, 0, 0, 0 } +// end +/* else do + vdf_fields = {linkt[i], linkcap[i], xcap[i], cycle[i],pfact[i], gcrat[i], alpha1[i], beta1[i], alpha2[i], beta2[i], preload[i]} + vdf_file = "tucson_vdf_rev.vdf" + vdf_defaults={ , , ,1.5 ,1 , 0.4 , 0.15, 4, 0.15, 4, 0 } + end + */ + net = ReadNetwork(net_file) + NetOpts = null + NetOpts.[Link ID] = link_lyr+".ID" + NetOpts.[Type] = "Enable" + NetOpts.[Write to file] = "Yes" + ChangeLinkStatus(net,, NetOpts) + + // Open the trip table to assign, and get the first table name + ODMatrix = outputDir + "\\"+trip[i] + m = OpenMatrix(ODMatrix,) + matrixCores = GetMatrixCoreNames(GetMatrix()) + coreName = matrixCores[1] + + + //settings for highway assignment + Opts = null + Opts.Global.[Force Threads] = 2 + Opts.Input.Database = db_file + Opts.Input.Network = net_file + + Opts.Input.[OD Matrix Currency] = {ODMatrix, coreName, , } + Opts.Input.[Exclusion Link Sets] = excl_qry[i] + Opts.Field.[Vehicle Classes] = vehclass + Opts.Field.[Fixed Toll Fields] = toll_fld2[i] + Opts.Field.[VDF Fld Names] = vdf_fields + Opts.Global.[Number of Classes] = num_class + Opts.Global.[Class PCEs] = class_PCE + Opts.Global.[Class VOIs] = VOT + Opts.Global.[Load Method] = "NCFW" + Opts.Global.[N Conjugate] = 2 + Opts.Global.[Loading Multiplier] = 1 + Opts.Global.Convergence = 0.0005 + Opts.Global.[Cost Function File] = vdf_file + Opts.Global.[VDF Defaults] = vdf_defaults + Opts.Global.[Iterations]=1000 + Opts.Flag.[Do Share Report] = 1 + Opts.Output.[Flow Table] = outputDir+"\\"+asign[i] + if (turn_flag=1 & iteration=4) then Opts.Input.[Turning Movement Node Set] = {db_node_lyr, node_lyr, "Selection", turn_qry} + if (turn_flag=1 & iteration=4) then Opts.Flag.[Do Turn Movement] = 1 + if (turn_flag=1 & iteration=4) then Opts.Output.[Movement Table] = outputDir+"\\"+turn[i] + Opts.Field.[MSA Flow] = "_MSAFlow" + periods[i] + Opts.Field.[MSA Cost] = "_MSACost" + periods[i] + Opts.Field.[MSA Time] = "_MSATime" + periods[i] + Opts.Global.[MSA Iteration] = iteration + if (selink_flag = 1 & iteration = 4) then do + Opts.Global.[Critical Queries] = selink_qry + Opts.Global.[Critical Set names] = selinkqry_name + Opts.Output.[Critical Matrix].Label = "Select Link Matrix" + Opts.Output.[Critical Matrix].Compression = 1 + Opts.Output.[Critical Matrix].[File Name] = outputDir +"\\"+selectlink_mtx[i] + end + RunMacro("HwycadLog",{"hwyassign.rsc: hwy assignment","MMA: "+asign[i]}) + ok = RunMacro("TCB Run Procedure", i, "MMA", Opts) + if !ok then goto quit + end + if!ok then goto quit + + ok=1 + quit: + RunMacro("close all") + return(ok) +EndMacro + +//added by JXu to split the flow table by queries. + +Macro "Selink Flow Split" (arr_selink) + shared path, inputDir, outputDir + asign=arr_selink[1] + selinkqry_name=arr_selink[2] + m=ArrayLength(selinkqry_name)+1 + dim new_flowtb[ArrayLength(asign),ArrayLength(selinkqry_name)+1] //All new flow table names after splitting for OP, AM and PM period assignments (3x5=15 names) + for i=1 to ArrayLength(asign) do + for j=1 to ArrayLength(selinkqry_name) do + new_flowtb[i][j] = outputDir+"\\"+left(asign[i],len(asign[i])-4)+"sl"+ i2s(j) +".bin" + end + new_flowtb[i][ArrayLength(selinkqry_name)+1]=outputDir+"\\"+asign[i] + end +//rename the original flow table file name to avoid the file name conflit with the splitted flow table. + for i=1 to ArrayLength(asign) do + new_file=left(asign[i],len(asign[i])-4)+"_orig.bin" + dict_nm = left(asign[i],len(asign[i])-4)+".dcb" + newdict_nm = left(asign[i],len(asign[i])-4)+"_orig.dcb" + ok=RunMacro("SDrenamefile",{outputDir+"\\"+asign[i],outputDir+"\\"+new_file}) if!ok then goto quit + ok=RunMacro("SDrenamefile",{outputDir+"\\"+dict_nm,outputDir+"\\"+newdict_nm}) if!ok then goto quit + asign[i]=new_file + end + +// This loop closes all views: + tmp = GetViews() + if tmp<>null then + for i = 1 to ArrayLength(tmp[1]) do + CloseView(tmp[1][i]) + end + + flowtb_vw = OpenTable("Flow Table", "FFB", {outputDir+"\\"+asign[1],}) + flowtb_fldinfo = GetViewStructure(flowtb_vw) //Get all the fields info from the flow table + dim flds_flag[ArrayLength(flowtb_fldinfo),ArrayLength(selinkqry_name)+1] +//flds_flag[i][j]=1 if flowtb_fldinfo[i][1] will be exported to .bin file for query j; +//if j is more than number of queries, then flds_flag[i][j] decides if flowtb_fldinfo[i][1] will be exported to .bin flow table without query info. + for i=1 to ArrayLength(flowtb_fldinfo) do + flag_tot = 0 + for j=1 to ArrayLength(selinkqry_name) do + if Position(flowtb_fldinfo[i][1],selinkqry_name[j])=0 then do + flds_flag[i][j]=0 + flag_tot=flag_tot+1 + end + else flds_flag[i][j]=1 + end + if flag_tot = ArrayLength(selinkqry_name) then do + for j=1 to ArrayLength(selinkqry_name) do + flds_flag[i][j]=1 + end + flds_flag[i][ArrayLength(selinkqry_name)+1]=1 //This will be exported to the flow table of original format without any query info + end + else flds_flag[i][ArrayLength(selinkqry_name)+1]=0 + end + + dim newflds[ArrayLength(selinkqry_name)+1] + for j=1 to ArrayLength(selinkqry_name)+1 do + for i=1 to ArrayLength(flowtb_fldinfo) do + if flds_flag[i][j]=1 then + newflds[j]=newflds[j]+{flowtb_fldinfo[i][1]} + end + end + + for i=1 to arraylength(asign) do + flow_vw = OpenTable("All Flow", "FFB", {outputDir+"\\"+asign[i],}) + for j=1 to arraylength(selinkqry_name) do + ExportView(flow_vw+"|", "FFB", new_flowtb[i][j],newflds[j], + {{"Additional Fields",{{"AB_Flow_"+selinkqry_name[j],"Real",15,4,"No"}, + {"BA_Flow_"+selinkqry_name[j],"Real",15,4,"No"}, + {"Tot_Flow_"+selinkqry_name[j],"Real",15,4,"No"}}} + }) + end + ExportView(flow_vw+"|", "FFB", new_flowtb[i][m],newflds[m],) + end + + //Fill in the newly added fields in the splitted flow table for each query + for i=1 to arraylength(asign) do + newflow_vw=null + newflow_fldinfo=null + For j=1 to arraylength(selinkqry_name) do + newflow_vw = OpenTable("Splitted Flow", "FFB", {new_flowtb[i][j],}) + newflow_fldinfo = GetViewStructure(newflow_vw) + AB_qry_flds=null + BA_qry_flds=null + for k=1 to ArrayLength(newflow_fldinfo) do + if Position(newflow_fldinfo[k][1],selinkqry_name[j])<>0 then do + if Position(newflow_fldinfo[k][1],"AB")<>0 then + AB_qry_flds=AB_qry_flds+{newflow_fldinfo[k][1]} + if Position(newflow_fldinfo[k][1],"BA")<>0 then + BA_qry_flds=BA_qry_flds+{newflow_fldinfo[k][1]} + end + end + order = {{"ID1", "Ascending"}} + rh = GetFirstRecord(newflow_vw+ "|", order) + while rh <> null do + AB_vals=0 + BA_vals=0 + for k=1 to ArrayLength(AB_qry_flds) do + vals=GetRecordValues(newflow_vw, rh, {AB_qry_flds[k],BA_qry_flds[k]}) + AB_vals=AB_vals+NZ(vals[1][2]) + BA_vals=BA_vals+NZ(vals[2][2]) + Tot_vals=AB_vals+BA_vals + end + SetRecordValues(newflow_vw, rh, {{"AB_Flow_"+selinkqry_name[j], AB_vals}, + {"BA_Flow_"+selinkqry_name[j], BA_vals}, + {"Tot_Flow_"+selinkqry_name[j],Tot_vals}}) + setRecord(newflow_vw, rh) + rh = GetNextRecord(newflow_vw + "|", rh, order) + end + end + end + + ok=1 + quit: + RunMacro("close all") + return(ok) + +endMacro + + +/********************************************************************************************************** + + combine truck tt_nt assign + + +**********************************************************************************************************/ +Macro "combine truck tt_nt assign"(arr) + shared path, inputDir, outputDir + stage = arr[1] + + asignbin={"lodtollop2.bin","lodtollam2.bin","lodtollpm2.bin"} + asigndcb={"lodtollop2.DCB","lodtollam2.DCB","lodtollpm2.DCB"} + copybin={"lodtollclassop2.bin","lodtollclassam2.bin","lodtollclasspm2.bin"} + copydcb={"lodtollclassop2.DCB","lodtollclassam2.DCB","lodtollclasspm2.DCB"} + viewNames ={"lodtollop2","lodtollam2","lodtollpm2"} + + // Copy files + for k=1 to 3 do + // check if copy files already exist, if exist delete + file=outputDir+"\\"+copybin[k] + dif2=GetDirectoryInfo(file,"file") + if dif2.length>0 then deletefile(file) + ok=1 + + CopyTableFiles(null,"FFB", outputDir+"\\"+asignbin[k], outputDir+"\\"+asigndcb[k],outputDir+"\\"+copybin[k], outputDir+"\\"+copydcb[k]) + + // delete the original highway files once copied + // check if copy files already exist, if exist delete + file=outputDir+"\\"+asignbin[k] + dif2=GetDirectoryInfo(file,"file") + if dif2.length>0 then deletefile(file) + ok=1 + + // Get copied files + view = OpenTable("assignment", "FFB", {outputDir+"\\"+copybin[k],} ) + ok1 = (view1 != null) + + // number of records + records = GetRecordCount(view, null) + + hov3_info = GetFileInfo(inputDir+"\\hov3") + hov3out_info = GetFileInfo(inputDir+"\\hov3out") + if (hov3_info=null & hov3out_info=null) then do + //get fields + fvector = GetDataVectors(view+"|",{"ID1", "AB_Flow_PCE", "BA_Flow_PCE", "Tot_Flow_PCE", "AB_Time", + "BA_Time", "Max_Time","AB_VOC","BA_VOC","Max_VOC","AB_V_Dist_T", + "BA_V_Dist_T","Tot_V_Dist_T","AB_VHT","BA_VHT","Tot_VHT", + "AB_Speed","BA_Speed","AB_VDF","BA_VDF","Max_VDF", + "AB_Flow_dan","BA_Flow_dan","AB_Flow_dat","BA_Flow_dat", + "AB_Flow_s2nn","BA_Flow_s2nn","AB_Flow_s2nh","BA_Flow_s2nh","AB_Flow_s2th","BA_Flow_s2th", + "AB_Flow_M1","BA_Flow_M1","AB_Flow_M2","BA_Flow_M2","AB_Flow_M3","BA_Flow_M3", + "AB_Flow_lhdn","BA_Flow_lhdn","AB_Flow_lhdt","BA_Flow_lhdt", + "AB_Flow_mhdn","BA_Flow_mhdn","AB_Flow_mhdt","BA_Flow_mhdt", + "AB_Flow_hhdn","BA_Flow_hhdn","AB_Flow_hhdt","BA_Flow_hhdt", + "AB_Flow","BA_Flow","Tot_Flow"},) + //create output file + view = CreateTable(viewNames[k], outputDir+"\\"+asignbin[k], "FFB", { + {"ID1" , "Integer (4 bytes)" , 10, null,"No", }, + {"AB_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Flow "}, + {"BA_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Flow "}, + {"Tot_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Flow "}, + {"AB_Time" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Travel Time "}, + {"BA_Time" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Travel Time "}, + {"Max_Time" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Loaded Time "}, + {"AB_VOC" , "Real (8 bytes)" , 15, 4 ,"No","AB Volume to Capacity Ratio "}, + {"BA_VOC" , "Real (8 bytes)" , 15, 4 ,"No","BA Volume to Capacity Ratio "}, + {"Max_VOC" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Volume to Capacity Ratio "}, + {"AB_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle miles or km of travel "}, + {"BA_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle miles or km of travel "}, + {"Tot_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle miles or km of travel "}, + {"AB_VHT" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle hours of travel "}, + {"BA_VHT" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle hours of travel "}, + {"Tot_VHT" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle hours of travel "}, + {"AB_Speed" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Speed "}, + {"BA_Speed" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Speed "}, + {"AB_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Volume Delay Function "}, + {"BA_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Volume Delay Function "}, + {"Max_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Link Volume Delay Function Value "}, + {"AB_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dan "}, + {"BA_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dan "}, + {"AB_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dat "}, + {"BA_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dat "}, + {"AB_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nn "}, + {"BA_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nn "}, + {"AB_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nh "}, + {"BA_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nh "}, + {"AB_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2th "}, + {"BA_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2th "}, + {"AB_Flow_M1" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M1 "}, + {"BA_Flow_M1" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M1 "}, + {"AB_Flow_M2" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M2 "}, + {"BA_Flow_M2" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M2 "}, + {"AB_Flow_M3" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M3 "}, + {"BA_Flow_M3" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M3 "}, + {"AB_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for lhd "}, + {"BA_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for lhd "}, + {"AB_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for mhd "}, + {"BA_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for mhd "}, + {"AB_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for hhd "}, + {"BA_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for hhd "}, + {"AB_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Veh Flow "}, + {"BA_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Veh Flow "}, + {"Tot_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Veh Flow "} + }) + SetView(view) + + //calculate and set values + for i = 1 to records do + rh = AddRecord(view, { + {"ID1" , fvector[1 ][i]}, + {"AB_Flow_PCE" , fvector[2 ][i]}, + {"BA_Flow_PCE" , fvector[3 ][i]}, + {"Tot_Flow_PCE" , fvector[4 ][i]}, + {"AB_Time" , fvector[5 ][i]}, + {"BA_Time" , fvector[6 ][i]}, + {"Max_Time" , fvector[7 ][i]}, + {"AB_VOC" , fvector[8 ][i]}, + {"BA_VOC" , fvector[9 ][i]}, + {"Max_VOC" , fvector[10][i]}, + {"AB_V_Dist_T" , fvector[11][i]}, + {"BA_V_Dist_T" , fvector[12][i]}, + {"Tot_V_Dist_T" , fvector[13][i]}, + {"AB_VHT" , fvector[14][i]}, + {"BA_VHT" , fvector[15][i]}, + {"Tot_VHT" , fvector[16][i]}, + {"AB_Speed" , fvector[17][i]}, + {"BA_Speed" , fvector[18][i]}, + {"AB_VDF" , fvector[19][i]}, + {"BA_VDF" , fvector[20][i]}, + {"Max_VDF" , fvector[21][i]}, + {"AB_Flow_dan" , fvector[22][i]}, + {"BA_Flow_dan" , fvector[23][i]}, + {"AB_Flow_dat" , fvector[24][i]}, + {"BA_Flow_dat" , fvector[25][i]}, + {"AB_Flow_s2nn" , fvector[26][i]}, + {"BA_Flow_s2nn" , fvector[27][i]}, + {"AB_Flow_s2nh" , fvector[28][i]}, + {"BA_Flow_s2nh" , fvector[29][i]}, + {"AB_Flow_s2th" , fvector[30][i]}, + {"BA_Flow_s2th" , fvector[31][i]}, + {"AB_Flow_M1" , fvector[32][i]}, + {"BA_Flow_M1" , fvector[33][i]}, + {"AB_Flow_M2" , fvector[34][i]}, + {"BA_Flow_M2" , fvector[35][i]}, + {"AB_Flow_M3" , fvector[36][i]}, + {"BA_Flow_M3" , fvector[37][i]}, + {"AB_Flow_lhd" , fvector[38][i] + fvector[40][i]}, + {"BA_Flow_lhd" , fvector[39][i] + fvector[41][i]}, + {"AB_Flow_mhd" , fvector[42][i] + fvector[44][i]}, + {"BA_Flow_mhd" , fvector[43][i] + fvector[45][i]}, + {"AB_Flow_hhd" , fvector[46][i] + fvector[48][i]}, + {"BA_Flow_hhd" , fvector[47][i] + fvector[49][i]}, + {"AB_Flow" , fvector[50][i]}, + {"BA_Flow" , fvector[51][i]}, + {"Tot_Flow" , fvector[52][i]} + }) + end + end + else do + fvector = GetDataVectors(view+"|",{"ID1", "AB_Flow_PCE", "BA_Flow_PCE", "Tot_Flow_PCE", "AB_Time", + "BA_Time", "Max_Time","AB_VOC","BA_VOC","Max_VOC","AB_V_Dist_T", + "BA_V_Dist_T","Tot_V_Dist_T","AB_VHT","BA_VHT","Tot_VHT", + "AB_Speed","BA_Speed","AB_VDF","BA_VDF","Max_VDF", + "AB_Flow_dan","BA_Flow_dan","AB_Flow_dat","BA_Flow_dat", + "AB_Flow_s2nn","BA_Flow_s2nn","AB_Flow_s2nh","BA_Flow_s2nh","AB_Flow_s2th","BA_Flow_s2th", + "AB_Flow_s3nn","BA_Flow_s3nn","AB_Flow_s3nh","BA_Flow_s3nh","AB_Flow_s3th","BA_Flow_s3th", + "AB_Flow_lhdn","BA_Flow_lhdn","AB_Flow_lhdt","BA_Flow_lhdt", + "AB_Flow_mhdn","BA_Flow_mhdn","AB_Flow_mhdt","BA_Flow_mhdt", + "AB_Flow_hhdn","BA_Flow_hhdn","AB_Flow_hhdt","BA_Flow_hhdt", + "AB_Flow","BA_Flow","Tot_Flow"},) + //create output file + view = CreateTable(viewNames[k], path+"\\"+asignbin[k], "FFB", { + {"ID1" , "Integer (4 bytes)" , 10, null,"No", }, + {"AB_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Flow "}, + {"BA_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Flow "}, + {"Tot_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Flow "}, + {"AB_Time" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Travel Time "}, + {"BA_Time" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Travel Time "}, + {"Max_Time" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Loaded Time "}, + {"AB_VOC" , "Real (8 bytes)" , 15, 4 ,"No","AB Volume to Capacity Ratio "}, + {"BA_VOC" , "Real (8 bytes)" , 15, 4 ,"No","BA Volume to Capacity Ratio "}, + {"Max_VOC" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Volume to Capacity Ratio "}, + {"AB_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle miles or km of travel "}, + {"BA_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle miles or km of travel "}, + {"Tot_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle miles or km of travel "}, + {"AB_VHT" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle hours of travel "}, + {"BA_VHT" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle hours of travel "}, + {"Tot_VHT" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle hours of travel "}, + {"AB_Speed" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Speed "}, + {"BA_Speed" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Speed "}, + {"AB_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Volume Delay Function "}, + {"BA_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Volume Delay Function "}, + {"Max_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Link Volume Delay Function Value "}, + {"AB_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dan "}, + {"BA_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dan "}, + {"AB_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dat "}, + {"BA_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dat "}, + {"AB_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nn "}, + {"BA_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nn "}, + {"AB_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nh "}, + {"BA_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nh "}, + {"AB_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2th "}, + {"BA_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2th "}, + {"AB_Flow_s3nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3nn "}, + {"BA_Flow_s3nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3nn "}, + {"AB_Flow_s3nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3nh "}, + {"BA_Flow_s3nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3nh "}, + {"AB_Flow_s3th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3th "}, + {"BA_Flow_s3th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3th "}, + {"AB_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for lhd "}, + {"BA_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for lhd "}, + {"AB_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for mhd "}, + {"BA_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for mhd "}, + {"AB_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for hhd "}, + {"BA_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for hhd "}, + {"AB_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Veh Flow "}, + {"BA_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Veh Flow "}, + {"Tot_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Veh Flow "} + }) + SetView(view) + + //calculate and set values + for i = 1 to records do + rh = AddRecord(view, { + {"ID1" , fvector[1 ][i]}, + {"AB_Flow_PCE" , fvector[2 ][i]}, + {"BA_Flow_PCE" , fvector[3 ][i]}, + {"Tot_Flow_PCE" , fvector[4 ][i]}, + {"AB_Time" , fvector[5 ][i]}, + {"BA_Time" , fvector[6 ][i]}, + {"Max_Time" , fvector[7 ][i]}, + {"AB_VOC" , fvector[8 ][i]}, + {"BA_VOC" , fvector[9 ][i]}, + {"Max_VOC" , fvector[10][i]}, + {"AB_V_Dist_T" , fvector[11][i]}, + {"BA_V_Dist_T" , fvector[12][i]}, + {"Tot_V_Dist_T" , fvector[13][i]}, + {"AB_VHT" , fvector[14][i]}, + {"BA_VHT" , fvector[15][i]}, + {"Tot_VHT" , fvector[16][i]}, + {"AB_Speed" , fvector[17][i]}, + {"BA_Speed" , fvector[18][i]}, + {"AB_VDF" , fvector[19][i]}, + {"BA_VDF" , fvector[20][i]}, + {"Max_VDF" , fvector[21][i]}, + {"AB_Flow_dan" , fvector[22][i]}, + {"BA_Flow_dan" , fvector[23][i]}, + {"AB_Flow_dat" , fvector[24][i]}, + {"BA_Flow_dat" , fvector[25][i]}, + {"AB_Flow_s2nn" , fvector[26][i]}, + {"BA_Flow_s2nn" , fvector[27][i]}, + {"AB_Flow_s2nh" , fvector[28][i]}, + {"BA_Flow_s2nh" , fvector[29][i]}, + {"AB_Flow_s2th" , fvector[30][i]}, + {"BA_Flow_s2th" , fvector[31][i]}, + {"AB_Flow_s3nn" , fvector[32][i]}, + {"BA_Flow_s3nn" , fvector[33][i]}, + {"AB_Flow_s3nh" , fvector[34][i]}, + {"BA_Flow_s3nh" , fvector[35][i]}, + {"AB_Flow_s3th" , fvector[36][i]}, + {"BA_Flow_s3th" , fvector[37][i]}, + {"AB_Flow_lhd" , fvector[38][i] + fvector[40][i]}, + {"BA_Flow_lhd" , fvector[39][i] + fvector[41][i]}, + {"AB_Flow_mhd" , fvector[42][i] + fvector[44][i]}, + {"BA_Flow_mhd" , fvector[43][i] + fvector[45][i]}, + {"AB_Flow_hhd" , fvector[46][i] + fvector[48][i]}, + {"BA_Flow_hhd" , fvector[47][i] + fvector[49][i]}, + {"AB_Flow" , fvector[50][i]}, + {"BA_Flow" , fvector[51][i]}, + {"Tot_Flow" , fvector[52][i]} + }) + end + end + end // end for loop + + vws = GetViewNames() + for p = 1 to vws.length do + CloseView(vws[p]) + end + return(1) + + quit: + return(0) +EndMacro + + + + +Macro "create trip tables by VOT"(args) + + shared path, inputDir, outputDir + + inFiles={"Trip_EA.mtx","Trip_AM.mtx","Trip_MD.mtx","Trip_PM.mtx","Trip_EV.mtx"} + outFiles={"Trip_EA_VOT.mtx","Trip_AM_VOT.mtx","Trip_MD_VOT.mtx","Trip_PM_VOT.mtx","Trip_EV_VOT.mtx"} + inTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} + outTableNames = { + "SOV_GP_LOW", "SOV_PAY_LOW", "SR2_GP_LOW","SR2_HOV_LOW", "SR2_PAY_LOW", "SR3_GP_LOW","SR3_HOV_LOW","SR3_PAY_LOW", + "SOV_GP_MED", "SOV_PAY_MED", "SR2_GP_MED","SR2_HOV_MED", "SR2_PAY_MED", "SR3_GP_MED","SR3_HOV_MED","SR3_PAY_MED", + "SOV_GP_HI", "SOV_PAY_HI", "SR2_GP_HI","SR2_HOV_HI", "SR2_PAY_HI", "SR3_GP_HI","SR3_HOV_HI","SR3_PAY_HI", + "lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} + + for i = 1 to inFiles.length do + + //open person trip matrix currencies + inMatrix = OpenMatrix(outputDir+"\\"+inFiles[i], ) + inCurrencies = CreateMatrixCurrencies(inMatrix, , , ) + + dim curr_array[inTableNames.length] + for j = 1 to inTableNames.length do + curr_array[j] = CreateMatrixCurrency(inMatrix, inTableNames[j], ,, ) + end + + //create output trip table and matrix currencies for this time period + outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outputDir+"\\"+outFiles[i]}, + {"Label", outFiles[i]}, + {"Tables",outTableNames}, + {"File Based", "Yes"}}) + SetMatrixCoreNames(outMatrix, outTableNames) + + outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) + + // calculate output matrices + outCurrencies.SOV_GP_LOW := inCurrencies.SOV_GP * 0.3333333 + outCurrencies.SOV_GP_MED := inCurrencies.SOV_GP * 0.3333333 + outCurrencies.SOV_GP_HI := inCurrencies.SOV_GP * 0.3333333 + + outCurrencies.SOV_PAY_LOW := inCurrencies.PAY_GP * 0.3333333 + outCurrencies.SOV_PAY_MED := inCurrencies.PAY_GP * 0.3333333 + outCurrencies.SOV_PAY_HI := inCurrencies.PAY_GP * 0.3333333 + + outCurrencies.SR2_GP_LOW := inCurrencies.SR2_GP * 0.3333333 + outCurrencies.SR2_GP_MED := inCurrencies.SR2_GP * 0.3333333 + outCurrencies.SR2_GP_HI := inCurrencies.SR2_GP * 0.3333333 + + outCurrencies.SR2_HOV_LOW := inCurrencies.SR2_HOV * 0.3333333 + outCurrencies.SR2_HOV_MED := inCurrencies.SR2_HOV * 0.3333333 + outCurrencies.SR2_HOV_HI := inCurrencies.SR2_HOV * 0.3333333 + + outCurrencies.SR2_PAY_LOW := inCurrencies.SR2_PAY * 0.3333333 + outCurrencies.SR2_PAY_MED := inCurrencies.SR2_PAY * 0.3333333 + outCurrencies.SR2_PAY_HI := inCurrencies.SR2_PAY * 0.3333333 + + outCurrencies.SR3_GP_LOW := inCurrencies.SR3_GP * 0.3333333 + outCurrencies.SR3_GP_MED := inCurrencies.SR3_GP * 0.3333333 + outCurrencies.SR3_GP_HI := inCurrencies.SR3_GP * 0.3333333 + + outCurrencies.SR3_HOV_LOW := inCurrencies.SR3_HOV * 0.3333333 + outCurrencies.SR3_HOV_MED := inCurrencies.SR3_HOV * 0.3333333 + outCurrencies.SR3_HOV_HI := inCurrencies.SR3_HOV * 0.3333333 + + outCurrencies.SR3_PAY_LOW := inCurrencies.SR3_PAY * 0.3333333 + outCurrencies.SR3_PAY_MED := inCurrencies.SR3_PAY * 0.3333333 + outCurrencies.SR3_PAY_HI := inCurrencies.SR3_PAY * 0.3333333 + + outCurrencies.lhdn := inCurrencies.lhdn + outCurrencies.mhdn := inCurrencies.mhdn + outCurrencies.hhdn := inCurrencies.hhdn + outCurrencies.lhdt := inCurrencies.lhdt + outCurrencies.mhdt := inCurrencies.mhdt + outCurrencies.hhdt := inCurrencies.hhdt + + end + RunMacro("close all" ) + + Return(1) + quit: + Return(0) +EndMacro diff --git a/sandag_abm/src/main/gisdk/hwyskim.rsc b/sandag_abm/src/main/gisdk/hwyskim.rsc new file mode 100644 index 0000000..01576e0 --- /dev/null +++ b/sandag_abm/src/main/gisdk/hwyskim.rsc @@ -0,0 +1,1257 @@ +/*********************************************** +Hwy skim all + +This macro calls macro "Update highway network", which updates highway network with times +from last highway assignment, and then skims highway network by calling macro "hwy skim" for +the following modes: + +dant Drive-alone non-toll +dat Drive-alone toll +s2nh Shared-2 non-toll HOV +s2th Shared-2 toll HOV +s3nh Shared-3 non-toll HOV +s3th Shared-3 toll HOV +truck Truck + +***********************************************/ +Macro "Hwy skim all" (args) + + skimByVOT= args[1] + + if skimByVOT="false" then do + + da_vot=67.00 // $0.67 cents per minute VOT ($40.2 per hour) + s2_vot=67.00 + s3_vot=67.00 + lh_vot=67.00 + mh_vot=68.00 + hh_vot=89.00 + cv_vot=67.00 + + vot_array = {da_vot, s2_vot, s3_vot, lh_vot, mh_vot, hh_vot, cv_vot} + + ok=RunMacro("Update highway network", vot_array) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"dant",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"dat",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s2nh",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s2th",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s3nh",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s3th",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"cvn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"cvt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"lhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"lhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"mhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"mhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"hhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"hhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"truck",}) + if !ok then goto quit + + end + else do + + vot_bins = {"low", "med", "high"} + // da, s2, s3, lh, mh, hh, cv} + vot_by_bin = {{16.6, 16.6, 16.6, 67.0, 68.0, 89.0, 67.0}, + {33.3, 33.3, 33.3, 67.0, 68.0, 89.0, 67.0}, + { 100, 100, 100, 67.0, 68.0, 89.0, 67.0} + } + + for i = 1 to vot_bins.length do + + ok=RunMacro("Update highway network", vot_by_bin[i]) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"dant",vot_bins[i]}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"dat",vot_bins[i]}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s2nh",vot_bins[i]}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s2th",vot_bins[i]}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s3nh",vot_bins[i]}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"s3th",vot_bins[i]}) + if !ok then goto quit + +/* + // reliability for time periods (15-mins) + for tod=1 to 96 do + + if (tod=25 or tod=26 or tod=35 or tod=36) then do //AM Period - 30 min shoulders + ok=RunMacro("hwy skim time bins",{"dant",vot_bins[i],tod}) + if !ok then goto quit + + ok=RunMacro("hwy skim time bins",{"dat",vot_bins[i],tod}) + if !ok then goto quit + + ok=RunMacro("hwy skim time bins",{"s2nh",vot_bins[i],tod}) + if !ok then goto quit + + ok=RunMacro("hwy skim time bins",{"s2th",vot_bins[i],tod}) + if !ok then goto quit + + ok=RunMacro("hwy skim time bins",{"s3nh",vot_bins[i],tod}) + if !ok then goto quit + + ok=RunMacro("hwy skim time bins",{"s3th",vot_bins[i],tod}) + if !ok then goto quit + end + + end + */ + end + + // don't skim commercial vehicles or trucks by vot + ok=RunMacro("hwy skim",{"cvn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"cvt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"lhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"lhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"mhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"mhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"hhdn",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"hhdt",}) + if !ok then goto quit + + ok=RunMacro("hwy skim",{"truck",}) + if !ok then goto quit + + end + + return(1) + + quit: + return(0) + +EndMacro + +/******************************************************************************** + +Update highway network + +Updates highway line layer and network with fields from latest assignment flow tables. + + Arguments + 1 drive-alone value-of-time (cents/min) + 2 shared 2 value-of-time (cents/min) + 3 shared 3+ value-of-time (cents/min) + 4 light-heavy truck value-of-time (cents/min) + 5 medium-heavy truck value-of-time (cents/min) + 6 heavy-heavy truck value-of-time (cents/min) + 7 commercial vehicle value-of-time (cents/min) + + +The following fields are updated on the line layer: + +Field Description +------- --------------- +STM SOV time +HTM HOV time +SCST SOV generalized cost +H2CST Shared-2 generalized cost +H3CST Shared-3 generalized cost +LHCST Light-heavy truck generalized cost +MHCST Medium-heavy truck generalized cost +HHCST Heavy-heavy truck generalized cost +CVCST Heavy-heavy truck generalized cost + + +Each field is xxField_yy where: + + xx is AB or BA indicating direction + yy is period, as follows: + EA: Early AM + AM: AM peak + MD: Midday + PM: PM peak + EV: Evening + +Inputs: + input\hwy.dbd Highway line layer + input\hwy.net Highway network + output\hwyload_yy.bin Loaded flow table from assignment, one per period (yy) + +Outputs: + output\hwy.dbd Updated highway line layer + output\hwy.net Updated highway network + +********************************************************************************/ +Macro "Update highway network" (args) + + shared path, inputDir, outputDir + + da_vot= args[1] + s2_vot= args[2] + s3_vot= args[3] + lh_vot= args[4] + mh_vot= args[5] + hh_vot= args[6] + cv_vot= args[7] + + // input files + db_file = outputDir + "\\hwy.dbd" + net_file = outputDir + "\\hwy.net" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + db_node_lyr = db_file + "|" + node_lyr + + periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} + + da_vot=da_vot*60/100 //Convert to dollars per hour VOT so don't have to change gen cost function below + s2_vot=s2_vot*60/100 + s3_vot=s3_vot*60/100 + lh_vot=lh_vot*60/100 + mh_vot=mh_vot*60/100 + hh_vot=hh_vot*60/100 + cv_vot=cv_vot*60/100 + + //Recompute generalized cost using MSA cost in flow table, + for i = 1 to periods.length do + + flowTable = outputDir+"\\hwyload"+periods[i]+".bin" + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"sovtime"+periods[i] } + Opts.Global.Fields = {"ABSTM"+periods[i],"BASTM"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost" , + "BA_MSA_Cost" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"hovtime"+periods[i] } + Opts.Global.Fields = {"ABHTM"+periods[i],"BAHTM"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost" , + "BA_MSA_Cost" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } + Opts.Global.Fields = {"ABSCST"+periods[i],"BASCST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(da_vot)+"*60)" , + "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(da_vot)+"*60)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // Light-Heavy truck cost + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } + Opts.Global.Fields = {"ABLHCST"+periods[i],"BALHCST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(lh_vot)+"*60)" , + "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(lh_vot)+"*60)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // Medium-Heavy truck cost + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } + Opts.Global.Fields = {"ABMHCST"+periods[i],"BAMHCST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(mh_vot)+"*60)" , + "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(mh_vot)+"*60)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // Heavy-Heavy truck cost + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } + Opts.Global.Fields = {"ABHHCST"+periods[i],"BAHHCST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(hh_vot)+"*60)" , + "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(hh_vot)+"*60)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // Commercial vehicle cost + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } + Opts.Global.Fields = {"ABCVCST"+periods[i],"BACVCST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(cv_vot)+"*60)" , + "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(cv_vot)+"*60)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"s2join"+periods[i] } + Opts.Global.Fields = {"ABH2CST"+periods[i],"BAH2CST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if (IHOV=3 or IHOV=4) then (AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s2_vot)+"*60)) else AB_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" , + "if (IHOV=3 or IHOV=4) then (BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s2_vot)+"*60)) else BA_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"s3join"+periods[i] } + Opts.Global.Fields = {"ABH3CST"+periods[i],"BAH3CST"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if IHOV=4 then (AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s3_vot)+"*60)) else AB_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" , + "if IHOV=4 then (BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s3_vot)+"*60)) else BA_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + // Total Reliability - The Dataview Set is a joined view of the link layer and the flow table, based on link ID + // calculate as square of link reliability - after skimming take square root of the total reliability + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"reliabilityjoin"+periods[i] } + Opts.Global.Fields = {"AB_TOTREL"+periods[i],"BA_TOTREL"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"pow(AB_MSA_Cost - AB_MSA_Time,2)", + "pow(BA_MSA_Cost - BA_MSA_Time,2)" } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + + //Now update the network with the calculated cost fields + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*STM"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABSTM"+periods[i],link_lyr+".BASTM"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*HTM"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABHTM"+periods[i],link_lyr+".BAHTM"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*SCST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABSCST"+periods[i],link_lyr+".BASCST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*H2CST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABH2CST"+periods[i],link_lyr+".BAH2CST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*H3CST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABH3CST"+periods[i],link_lyr+".BAH3CST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*LHCST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABLHCST"+periods[i],link_lyr+".BALHCST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*MHCST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABMHCST"+periods[i],link_lyr+".BAMHCST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*HHCST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABHHCST"+periods[i],link_lyr+".BAHHCST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*CVCST"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".ABCVCST"+periods[i],link_lyr+".BACVCST"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + // update total reliability fields + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*_TOTREL"+periods[i] + Opts.Global.Options.[Link Fields] = { {link_lyr+".AB_TOTREL"+periods[i],link_lyr+".BA_TOTREL"+periods[i] } } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + end + + + ok=1 + runmacro("close all") + quit: + + return(ok) + +EndMacro + +/******************************************************************************************** + +hwy skim + +Skim highway network for the following modes (passed as argument) + + +Mode Description Cost attribute +---- --------------------- -------------- +truck Truck SCST +lhdn Light-heavy-duty non-toll LHCST +mhdn Medium-heavy-duty non-toll MHCST +hhdn Heavy-heavy-duty non-toll HHCST +lhdt Light-heavy-duty toll LHCST +mhdt Medium-heavy-duty toll MHCST +hhdt Heavy-heavy-duty toll HHCST +cvn Commercial vehicle non-toll CVCST +cvt Commercial vehicle toll CVCST +dant Drive-alone non-toll SCST +dat Drive-alone toll SCST +s2nh Shared-2 non-toll HOV H2CST +s2th Shared-2 toll non-HOV H2CST +s3nh Shared-3 non-toll HOV H3CST +s3th Shared-3 toll HOV H3CST + +Note: dant skims also apply to shared-2 non-toll, non-HOV and shared 3+ non-toll, non-HOV + +v1.0 jef 3/30/2012 +v2.0 jef 5/10/2015 added value-of-time bins and commercial vehicle modes + +*/ +Macro "hwy skim" (args) + + shared path, inputDir, outputDir, mxzone + + mode=args[1] + + //vot_bin is the value-of-time bin that will be appended to each skim name; prepend "_" + if args[2]=null then vot_bin="" else vot_bin="_"+args[2] + + dim skimbyset1[3],skimbyset2[3] + + + // input files + db_file = outputDir + "\\hwy.dbd" + net_file = outputDir + "\\hwy.net" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + db_node_lyr = db_file + "|" + node_lyr + net = ReadNetwork(net_file) + + periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} + + for i = 1 to periods.length do + + skimbyset1 = null // second skim varaible (in addition to LENGTH) + skimbyset2 = null // third skim variable + skimbyset3 = null // fourth skim variable + + // The truck skim is used for heavy trip distribution + if mode = "truck" then do + + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!((ihov=1|ihov=4|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for exclusion link set + + set = "TrkToll"+periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (ihov=4 and (ITRUCK=1|ITRUCK>4))",) + if n > 0 then skimbyset1={vw_set, {"itoll"+periods[i]}} + + // skimbyset2 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imptrk"+periods[i]+vot_bin+".mtx" // output skim matrices + + end + + // The skims by weight class are used for truck toll diversion + else if mode = "lhdn" then do // light duty truck non-toll + + CostFld = "*LHCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!((ihov=1|ifc>7)&(ITRUCK<4|ITRUCK=7))" // query for lhd non-toll exclusion link set + + // skimbyset1 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + else if mode = "mhdn" then do // medium duty truck non-toll + + CostFld = "*MHCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!((ihov=1|ifc>7)&(ITRUCK<3|ITRUCK>5))" // query for mhd non-toll exclusion link set + + // skimbyset1 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + else if mode = "hhdn" then do // heavy duty truck non-toll + + CostFld = "*HHCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd non-toll exclusion link set + + // skimbyset1 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + + else if mode = "lhdt" then do + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))" // query for lhd toll exclusion link set + + tollfield = "ITOLL2" // toll value + + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) + if n = 0 then excl_qry=null // reset value if no selection records + + // skimbyset1 = toll + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} + + // skimbyset2 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + else if mode = "mhdt" then do + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))" // query for mhd toll exclusion link set + + tollfield = "ITOLL2" // toll value + + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) + if n = 0 then excl_qry=null // reset value if no selection records + + // skimbyset1 = toll + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} + + // skimbyset2 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} + + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + else if mode = "hhdt" then do + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd toll exclusion link set + + tollfield = "ITOLL2" // toll value + + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) + if n = 0 then excl_qry=null // reset value if no selection records + + // skimbyset1 = toll + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} + + // skimbyset2 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices + end + + else if mode = "cvn" then do // commercial vehicles non-toll + CostFld = "*CVCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd non-toll exclusion link set + + // skimbyset1 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "impcvn"+periods[i]+vot_bin+".mtx" // output skim matrices + end + + else if mode = "cvt" then do // commercial vehicle toll skims; uses same selection set as drive-alone toll + CostFld = "*CVCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several","Select * where "+excl_qry,) + if n = 0 then excl_qry=null //reset value if no selection records + + tollfield = "ITOLL2" // toll value + + // skimbyset1 = toll + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} + + // skimbyset2 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "impcvt"+periods[i]+vot_bin+".mtx" + end + else if mode = "dant" then do + + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) + + excl_qry = "!(ihov=1&ITRUCK<5)" // query for exclusion link set + + // skimbyset1 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "impdan"+periods[i]+vot_bin+".mtx" // output skim matrices + end + else if mode = "dat" then do + + CostFld = "*SCST"+periods[i] // minimizing cost field + SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) + + //excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several","Select * where "+excl_qry,) + if n = 0 then excl_qry=null //reset value if no selection records + + // skimbyset1 = length on toll lanes + set = "datdst"+periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + //n = SelectByQuery(set, "Several", "Select * where ((ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))&ITRUCK<5)",) + n = SelectByQuery(set, "Several", "Select * where ((ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))&ITRUCK<5)",) + if n > 0 then skimbyset1={vw_set, {"Length"}} + + // skimbyset2 = cost + set = mode + periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} + + // skimbyset3 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "impdat"+periods[i]+vot_bin+".mtx" + end + else if mode = "s2nh" then do + + CostFld = "*H2CST"+periods[i] // minimizing cost field + SkimVar1 = "*HTM"+periods[i] + + excl_qry ="!((ihov=1|(ihov=2&abln"+periods[i]+" <9)|ifc>7)&ITRUCK<5)"//initialize the value + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several","Select * where "+excl_qry,) + if n = 0 then excl_qry=null //reset value if no selection records + + set = "s2hdst" + periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (abln"+periods[i]+"<9 and ihov=2 and ITRUCK<5)",) + if n > 0 then skimbyset2={vw_set, {"Length"}} + + // skimbyset3 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imps2nh"+periods[i]+vot_bin+".mtx" + end + else if mode = "s2th" then do + + CostFld = "*H2CST"+periods[i] // minimizing cost field + SkimVar1 ="*HTM"+periods[i] + + excl_qry = "!(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)" + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several","Select * where " + excl_qry,) + if n = 0 then excl_qry=null //reset value if no selection records + + set = "s2tdst"+periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (((abln"+periods[i]+"<9 & ihov=2) | (ihov=4 | (ihov=3 & itoll"+periods[i]+" >0 & abln"+periods[i]+"< 9)))& ITRUCK<5)",) + if n > 0 then skimbyset1={vw_set, {"Length"}} + + set = "s2t"+periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where ((ihov=4|(ihov=3 & itoll"+periods[i]+" >0 & abln"+periods[i]+" < 9)) & ITRUCK<5)",) + if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} + + // skimbyset3 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imps2th"+periods[i]+vot_bin+".mtx" + + end + else if mode = "s3nh" then do + + CostFld = "*H3CST"+periods[i] // minimizing cost field + SkimVar1 = "*HTM" +periods[i] + + excl_qry = "!((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)" + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) + if n = 0 then excl_qry=null + + set = "s3hdst"+periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (abln"+periods[i]+"<9 & (ihov=2 | ihov=3) & ITRUCK<5)",) + if n > 0 then skimbyset2={vw_set, {"Length"}} + + // skimbyset3 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imps3nh"+periods[i]+vot_bin+".mtx" + + end + else if mode = "s3th" then do + + CostFld = "*H3CST" + periods[i] // minimizing cost field + SkimVar1 = "*HTM" + periods[i] + + excl_qry = "(abln"+periods[i]+"=9 | ITRUCK >4)" + set = mode + periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) + if n = 0 then excl_qry=null + + set = "s3tdst" + periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (((abln"+periods[i]+"<9 and (ihov=2 or ihov=3)) or ihov=4)and ITRUCK <5)",) + if n > 0 then skimbyset1={vw_set, {"Length"}} + + set = "s3t" + periods[i] + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where (ihov=4 and ITRUCK <5)",) + if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} + + // skimbyset3 = reliability + set = mode + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links + if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} + + skimmat = "imps3th"+periods[i]+vot_bin+".mtx" + end + + + //delete existing skim file + ok=RunMacro("SDdeletefile",{outputDir+"\\"+skimmat}) + if !ok then goto quit + + //skim network + set = "AllLinks" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectAll(set) + NetOpts = null + NetOpts.[Link ID] = link_lyr+".ID" + NetOpts.[Type] = "Enable" + ChangeLinkStatus(net,vw_set, NetOpts) // first enable all links + + if excl_qry<>null then do + set = "toll" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where "+excl_qry,) + NetOpts = null + NetOpts.[Link ID] = link_lyr+".ID" + NetOpts.[Type] = "Disable" + ChangeLinkStatus(net,vw_set, NetOpts) // disable exclusion query + end + + Opts = null + Opts.Input.Network = net_file + Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Centroids", "select * where ID <= " + i2s(mxzone)} + Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Centroids"} + Opts.Input.[Via Set] = {db_node_lyr, node_lyr} + Opts.Field.Minimize = CostFld + Opts.Field.Nodes = node_lyr + ".ID" + Opts.Field.[Skim Fields]={{"Length","All"},{SkimVar1,"All"}} + + // provide skimset + if skimbyset1 <> null then do + if skimbyset2 <> null then do + if skimbyset3 <> null then + Opts.Field.[Skim by Set]={skimbyset1,skimbyset2,skimbyset3} + else + Opts.Field.[Skim by Set]={skimbyset1,skimbyset2} + end + else if skimbyset3 <> null then + Opts.Field.[Skim by Set]={skimbyset1,skimbyset3} + else + Opts.Field.[Skim by Set]={skimbyset1} + end + else if skimbyset2 <> null then do + if skimbyset3 <> null then + Opts.Field.[Skim by Set]={skimbyset2,skimbyset3} + else + Opts.Field.[Skim by Set]={skimbyset2} + end + else if skimbyset3 <> null then + Opts.Field.[Skim by Set]={skimbyset3} + + //end of previous if string + if (mode = "lhdn" | mode = "mhdn" | mode = "hhdn" | mode = "lhdt" | mode = "mhdt" | mode = "hhdt") then + if (mode = "lhdn" | mode = "mhdn" | mode = "hhdn") then + Opts.Output.[Output Matrix].Label = "impedance truck" + else if (mode = "lhdt" | mode = "mhdt" | mode = "hhdt") then + Opts.Output.[Output Matrix].Label = "impedance truck toll" + else + Opts.Output.[Output Matrix].Label = "congested " + mode + " impedance" + Opts.Output.[Output Matrix].Compression = 0 //uncompressed, for version 4.8 plus + Opts.Output.[Output Matrix].[File Name] = outputDir + "\\"+skimmat + + RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim","TCSPMAT: "+skimmat+"; "+CostFld}) + ok = RunMacro("TCB Run Procedure", 1, "TCSPMAT", Opts) + if !ok then goto quit + + // STEP 2: Intrazonal added by Ziying Ouyang, June 3, 2009 + // mtxcore={"Length (Skim)"}+{SkimVar1[i]+" (Skim)"}+{SkimVar2[i]+" (Skim)"}+{SkimVar3[i]+" (Skim)"} + mtxcore={"Length (Skim)"}+{SkimVar1+" (Skim)"} + for j = 1 to mtxcore.length do + Opts = null + Opts.Global.Factor = 0.5 + Opts.Global.Neighbors = 3 + Opts.Global.Operation = 1 + Opts.Global.[Treat Missing] = 2 + Opts.Input.[Matrix Currency] = {outputDir + "\\"+skimmat,mtxcore[j], , } + RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim","Intrazonal: "+skimmat+"; "+mtxcore[j]}) + ok = RunMacro("TCB Run Procedure", j, "Intrazonal", Opts) + if !ok then goto quit + end + + // take square root of the reliability which is sum of square of link reliability - write code after generating skims - todo + mtxcore = mode+" - *_TOTREL"+periods[i] + m=OpenMatrix(outputDir + "\\"+skimmat,) + mc=CreateMatrixCurrency(m,mtxcore,,,) + mc:=Nz(mc) // zero out intra-zonal values + mc:=sqrt(mc) // take square root + + end + + ok=1 + runmacro("close all") + quit: + + return(ok) + +EndMacro + +/* + +hwy skim 15 mins time slices + +Skim reliability with following shift variables: + +Variable Time Bin Estimate-Freeway Estimate-Arterial +-------- -------- ---------------- ----------------- +BeforeAM.Step1 32 -0.0183 -0.0054 +BeforeAM.Step2 29 0.0092 -0.0032 +BeforeAM.Step3 26 0.0107 0.0030 +BeforeAM.Step4 20 -0.0019 0.0055 +AfterAM.Step1 32 -0.0082 -0.0009 +AfterAM.Step2 36 0.0000 0.0000 +AfterAM.Step3 39 0.0000 0.0000 +BeforePM.Step1 70 -0.0067 0.0011 +BeforePM.Step2 66 -0.0028 0.0000 +BeforePM.Step3 62 0.0094 -0.0018 +BeforePM.Step4 58 0.0000 0.0000 +AfterPM.Step1 70 -0.0077 -0.0079 +AfterPM.Step2 71 0.0000 0.0025 +AfterPM.Step3 79 0.0075 0.0037 + +{"EA","AM","MD","PM","EV1","EV2"} = {{15,24},{25,36},{37,62},{63,76},{77,96},{0,14}} + +*/ + +macro "hwy skim time bins" (args) + + shared path, inputDir, outputDir, mxzone + + mode = args[1] + + //vot_bin is the value-of-time bin that will be appended to each skim name; prepend "_" + if args[2]=null then vot_bin="" else vot_bin="_"+args[2] + timebin=args[3] + + // period thresholds + Peak_AM = 32 + Low_MD = 41 + Peak_PM = 70 + + // shift variable settings + // {beforeAM_step1, beforeAM_step2,beforeAM_step3,beforeAM_step4,afterAM_step1,afterAM_step2,afterAM_step3,beforePM_step1,beforePM_step2,beforePM_step3,beforePM_step4,afterPM_step1,afterPM_step2,afterPM_step3} + time_bins = {Peak_AM,29,26,20,Peak_AM,36,39,Peak_PM,66,62,58,Peak_PM,71,79} + factor_freeway = {-0.0183,0.0092,0.0107,-0.0019,-0.0082,0.0000,0.0000,-0.0067,-0.0028,0.0094,0.0000,-0.0077,0.0000,0.0075} + factor_arterial = {-0.0054,-0.0032,0.0030,0.0055,-0.0009,0.0000,0.0000,0.0011,0.0000,-0.0018,0.0000,-0.0079,0.0025,0.0037} + + facility_type = {"freeway","arterial","ramp","other"} // freeway (IFC=1), arterial (IFC=2,3), ramp (IFC=8,9), other (IFC=4,5,6,7) + periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} + + // lower and upper bounds of IFC for respective facility type = {freeway, arterial, ramp, other} + lwr_bound = {"1","2","8","4"} + upr_bound = {"1","3","9","7"} + + // input files + db_file = outputDir + "\\hwy.dbd" + net_file = outputDir + "\\hwy.net" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + db_node_lyr = db_file + "|" + node_lyr + net = ReadNetwork(net_file) + + //Recompute total reliability for 15 mins time slices + for fac_type=1 to facility_type.length do + + if fac_type=1 then factor=factor_freeway + else factor=factor_arterial + + // corresponding model time period + if timebin>=15 and timebin<=24 then period = periods[1] // EA + if timebin>=25 and timebin<=36 then period = periods[2] // AM + if timebin>=37 and timebin<=62 then period = periods[3] // MD + if timebin>=63 and timebin<=76 then period = periods[4] // PM + if (timebin>=1 and timebin<=14) or (timebin>=77 and timebin<=96) then period = periods[5] // EV + + // initialize shift variables + beforeAM_step1=0 + beforeAM_step2=0 + beforeAM_step3=0 + beforeAM_step4=0 + afterAM_step1=0 + afterAM_step2=0 + afterAM_step3=0 + beforePM_step1=0 + beforePM_step2=0 + beforePM_step3=0 + beforePM_step4=0 + afterPM_step1=0 + afterPM_step2=0 + afterPM_step3=0 + + // calculate shift reliability + + // before AM + if timebintime_bins[5] and timebintime_bins[6] and timebintime_bins[7] and timebin=Low_MD and timebin=Low_MD and timebin=Low_MD and timebin=Low_MD and timebintime_bins[12] then afterPM_step1=factor[12]*(timebin-time_bins[12]) + if timebin>time_bins[13] then afterPM_step2=factor[13]*(timebin-time_bins[13]) + if timebin>time_bins[14] then afterPM_step3=factor[14]*(timebin-time_bins[14]) + + // calculate total shift reliability factors + shift_factor = beforeAM_step1 + beforeAM_step2 + beforeAM_step3 + beforeAM_step4 + + afterAM_step1 + afterAM_step2 + afterAM_step3 + + beforePM_step1 + beforePM_step2 + beforePM_step3 + beforePM_step4 + + afterPM_step1 + afterPM_step2 + afterPM_step3 + + // expressions to calculate AB/BA shift reliability - squareof link reliability + expression_AB = "AB_TOTREL" + period + "+ pow(" + String(shift_factor) + "*Length*AB_Time,2)" + expression_BA = "BA_TOTREL" + period + "+ pow(" + String(shift_factor) + "*Length*BA_Time,2)" + + flowTable = outputDir+"\\hwyload"+period+".bin" + + query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] + + // Total Reliability - The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"reliabilityjoin"+period, "selection", query} + Opts.Global.Fields = {"AB_TOTREL"+period,"BA_TOTREL"+period} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {expression_AB, + expression_BA } + ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ret_value then goto quit + end + + // update total reliability fields + Opts = null + Opts.Input.Database = db_file + Opts.Input.Network = net_file + Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} + Opts.Global.[Fields Indices] = "*_TOTREL"+period + Opts.Global.Options.[Link Fields] = { {link_lyr+".AB_TOTREL"+period,link_lyr+".BA_TOTREL"+period} } + Opts.Global.Options.Constants = {1} + ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) + if !ret_value then goto quit + + // settings for skim + if mode = "dant" then do + CostFld = "*SCST"+period // minimizing cost field + excl_qry = "!(ihov=1&ITRUCK<5)" // query for exclusion link set + skimmat = "impdan_"+String(timebin)+vot_bin+".mtx" // output skim matrices + end + else if mode = "dat" then do + CostFld = "*SCST"+period // minimizing cost field + excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+period+">0&abln"+period+"<9)))|ifc>7)&ITRUCK<5)" + skimmat = "impdat_"+String(timebin)+vot_bin+".mtx" + end + else if mode = "s2nh" then do + CostFld = "*H2CST"+period // minimizing cost field + excl_qry ="!((ihov=1|(ihov=2&abln"+period+" <9)|ifc>7)&ITRUCK<5)"//initialize the value + skimmat = "imps2nh"+String(timebin)+vot_bin+".mtx" + end + else if mode = "s2th" then do + CostFld = "*H2CST"+period // minimizing cost field + excl_qry = "!(((ihov=1|(ihov=2&abln"+period+"<9)|ihov=4|(ihov=3&itoll"+period+">0&abln"+period+"<9))|ifc>7)&ITRUCK<5)" + skimmat = "imps2th"+String(timebin)+vot_bin+".mtx" + end + else if mode = "s3nh" then do + CostFld = "*H3CST"+period // minimizing cost field + excl_qry = "!((ihov=1|((ihov=2|ihov=3)&abln"+period+"<9)|ifc>7)& ITRUCK<5)" + skimmat = "imps3nh"+String(timebin)+vot_bin+".mtx" + end + else if mode = "s3th" then do + CostFld = "*H3CST"+period // minimizing cost field + excl_qry = "(abln"+period+"=9 | ITRUCK >4)" + skimmat = "imps3th"+String(timebin)+vot_bin+".mtx" + end + + //skim network + set = "AllLinks" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectAll(set) + NetOpts = null + NetOpts.[Link ID] = link_lyr+".ID" + NetOpts.[Type] = "Enable" + ChangeLinkStatus(net,vw_set, NetOpts) // first enable all links + + if excl_qry<>null then do + set = "toll" + vw_set = link_lyr + "|" + set + SetLayer(link_lyr) + n = SelectByQuery(set, "Several", "Select * where "+excl_qry,) + NetOpts = null + NetOpts.[Link ID] = link_lyr+".ID" + NetOpts.[Type] = "Disable" + ChangeLinkStatus(net,vw_set, NetOpts) // disable exclusion query + end + + Opts = null + Opts.Input.Network = net_file + Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Centroids", "select * where ID <= " + i2s(mxzone)} + Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Centroids"} + Opts.Input.[Via Set] = {db_node_lyr, node_lyr} + Opts.Field.Minimize = CostFld + Opts.Field.Nodes = node_lyr + ".ID" + //Opts.Field.[Skim Fields].["*_TOTREL"+period]="All" + Opts.Field.[Skim Fields]={{"*_TOTREL"+period,"All"}} + Opts.Output.[Output Matrix].Label = "reliability " + mode + + Opts.Output.[Output Matrix].Compression = 0 //uncompressed, for version 4.8 plus + Opts.Output.[Output Matrix].[File Name] = outputDir + "\\"+skimmat + + RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim time bins","TCSPMAT: "+skimmat+"; "+CostFld}) + ok = RunMacro("TCB Run Procedure", 1, "TCSPMAT", Opts, &Ret) + if !ok then goto quit + + // update skims by taking square root of reliability + mtxcore = "*_TOTREL"+period + " (Skim)" + m=OpenMatrix(outputDir + "\\"+skimmat,) + mc=CreateMatrixCurrency(m,mtxcore,,,) + mc:=Nz(mc) // zero out intra-zonal values + mc:=sqrt(mc) // take square root + + ok=1 + runmacro("close all") + quit: + + return(ok) + +EndMacro + diff --git a/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc b/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc new file mode 100644 index 0000000..e605f73 --- /dev/null +++ b/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc @@ -0,0 +1,51 @@ +/********************************************************************************************************** +Reduce matrix precision +About: + Script to reduce matrix precision. Precision defined in property file. + Author: Wu Sun, SANDAG + Developed: May 2015 + +Note: +Aggregate models such as truck, commercial vehicle, and EI models tend to have fake precisions. +The fake precisions make these matrices very large. +When matrixes are loaded into ABM database, it takes a large amount of DB space. +This scripts is to reduce fake precisions to make space managable. +**********************************************************************************************************/ +Macro "reduce matrix precision"(dir, mat, precision) + + //get matrix info + m = OpenMatrix(dir+"//"+mat, ) + coreNames = GetMatrixCoreNames(m) + numCores=coreNames.length + + //initialize arrays + dim sum[numCores] + dim rsum[numCores] + + //get matrix currency + currency=RunMacro("set input matrix currencies",dir, mat) + + for i = 1 to numCores do + //zero out null cells + currency[i]:=Nz(currency[i]) + + //sums of matrix by core + marginal_sums = GetMatrixMarginals(currency[i], "Sum", "row" ) + sum[i]=Sum(marginal_sums) + RunMacro("HwycadLog",{"matrixPrecisionReduction.rsc:","sum of "+coreNames[i]+" before reduction:"+r2s(sum[i])}) + + //reduce matrix precision using matrix expression. + expr="if(["+coreNames[i]+"]<"+precision+") then 0.0 else ["+coreNames[i]+"]" + EvaluateMatrixExpression(currency[i], expr,,, ) + rmarginal_sums = GetMatrixMarginals(currency[i], "Sum", "row" ) + rsum[i]=Sum(rmarginal_sums) + RunMacro("HwycadLog",{"matrixPrecisionReduction.rsc:","sum of "+coreNames[i]+" after reduction:"+r2s(rsum[i])}) + end + + //scale up reduced matrix to orirginal sum + for i = 1 to numCores do + currency[i]:=currency[i]*sum[i]/rsum[i] + end + +EndMacro + diff --git a/sandag_abm/src/main/gisdk/parameter.rsc b/sandag_abm/src/main/gisdk/parameter.rsc new file mode 100644 index 0000000..d569002 --- /dev/null +++ b/sandag_abm/src/main/gisdk/parameter.rsc @@ -0,0 +1,8 @@ +macro "parameters" +shared mxzone,mxtap,mxext,mxlink,mxrte +mxzone=4996 +mxtap=2500 +mxext=12 +mxrte=380 +mxlink=41000 +endmacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/sandag_abm.lst b/sandag_abm/src/main/gisdk/sandag_abm.lst new file mode 100644 index 0000000..07cd1e8 --- /dev/null +++ b/sandag_abm/src/main/gisdk/sandag_abm.lst @@ -0,0 +1,22 @@ +${workpath}\\${year}\\gisdk\\dbox.rsc +${workpath}\\${year}\\gisdk\\sandag_abm_master.rsc +${workpath}\\${year}\\gisdk\\parameter.rsc +${workpath}\\${year}\\gisdk\\SandagCommon.rsc +${workpath}\\${year}\\gisdk\\createhwynet.rsc +${workpath}\\${year}\\gisdk\\hwyassign.rsc +${workpath}\\${year}\\gisdk\\hwyskim.rsc +${workpath}\\${year}\\gisdk\\createtrnroutes.rsc +${workpath}\\${year}\\gisdk\\trnskim.rsc +${workpath}\\${year}\\gisdk\\trnassign.rsc +${workpath}\\${year}\\gisdk\\commVehGen.rsc +${workpath}\\${year}\\gisdk\\commVehDist.rsc +${workpath}\\${year}\\gisdk\\commVehTOD.rsc +${workpath}\\${year}\\gisdk\\commVehDiversion.rsc +${workpath}\\${year}\\gisdk\\createtodtables.rsc +${workpath}\\${year}\\gisdk\\externalInternal.rsc +${workpath}\\${year}\\gisdk\\TruckModel.rsc +${workpath}\\${year}\\gisdk\\create_LUZ_Skims.rsc +${workpath}\\${year}\\gisdk\\sandag_abm_outputs.rsc +${workpath}\\${year}\\gisdk\\exportTCData.rsc +${workpath}\\${year}\\gisdk\\Utilities.rsc +${workpath}\\${year}\\gisdk\\matrixPrecisionReduction.rsc diff --git a/sandag_abm/src/main/gisdk/sandag_abm_generic.lst b/sandag_abm/src/main/gisdk/sandag_abm_generic.lst new file mode 100644 index 0000000..2f8b0f4 --- /dev/null +++ b/sandag_abm/src/main/gisdk/sandag_abm_generic.lst @@ -0,0 +1,23 @@ +${workpath}\\gisdk\\dbox.rsc +${workpath}\\gisdk\\gui.rsc +${workpath}\\gisdk\\sandag_abm_master.rsc +${workpath}\\gisdk\\parameter.rsc +${workpath}\\gisdk\\SandagCommon.rsc +${workpath}\\gisdk\\createhwynet.rsc +${workpath}\\gisdk\\hwyassign.rsc +${workpath}\\gisdk\\hwyskim.rsc +${workpath}\\gisdk\\createtrnroutes.rsc +${workpath}\\gisdk\\trnskim.rsc +${workpath}\\gisdk\\trnassign.rsc +${workpath}\\gisdk\\commVehGen.rsc +${workpath}\\gisdk\\commVehDist.rsc +${workpath}\\gisdk\\commVehTOD.rsc +${workpath}\\gisdk\\commVehDiversion.rsc +${workpath}\\gisdk\\createtodtables.rsc +${workpath}\\gisdk\\externalInternal.rsc +${workpath}\\gisdk\\TruckModel.rsc +${workpath}\\gisdk\\create_LUZ_Skims.rsc +${workpath}\\gisdk\\sandag_abm_outputs.rsc +${workpath}\\gisdk\\exportTCData.rsc +${workpath}\\gisdk\\Utilities.rsc +${workpath}\\gisdk\\matrixPrecisionReduction.rsc diff --git a/sandag_abm/src/main/gisdk/sandag_abm_master.rsc b/sandag_abm/src/main/gisdk/sandag_abm_master.rsc new file mode 100644 index 0000000..ccd497b --- /dev/null +++ b/sandag_abm/src/main/gisdk/sandag_abm_master.rsc @@ -0,0 +1,411 @@ +Macro "Run SANDAG ABM" + + RunMacro("TCB Init") + + shared path, inputDir, outputDir, inputTruckDir, mxzone, mxtap, mxext,mxlink,mxrte,scenarioYear,version + + + sample_rate = { 0.2, 0.5, 1.0 } + max_iterations=sample_rate.length //number of feedback loops + skimByVOT = "true" + assignByVOT = "true" + + path = "${workpath}" + + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","*********Model Run Starting************"}) + + path_parts = SplitPath(path) + path_no_drive = path_parts[2]+path_parts[3] + drive=path_parts[1] + path_forward_slash = Substitute(path_no_drive, "\\", "/", ) + + inputDir = path+"\\input" + outputDir = path+"\\output" + inputTruckDir = path+"\\input_truck" + + SetLogFileName(path+"\\logFiles\\tclog.xml") + SetReportFileName(path+"\\logFiles\\tcreport.xml") + + RunMacro("parameters") + + // read properties from sandag_abm.properties in /conf folder + properties = "\\conf\\sandag_abm.properties" + sample_rate = RunMacro("read properties array",properties,"sample_rates", "S") + max_iterations=sample_rate.length //number of feedback loops + scenarioYear = RunMacro("read properties",properties,"scenarioYear", "S") + skipCopyWarmupTripTables = RunMacro("read properties",properties,"RunModel.skipCopyWarmupTripTables", "S") + skipCopyBikeLogsum = RunMacro("read properties",properties,"RunModel.skipCopyBikeLogsum", "S") + skipCopyWalkImpedance= RunMacro("read properties",properties,"RunModel.skipCopyWalkImpedance", "S") + skipWalkLogsums= RunMacro("read properties",properties,"RunModel.skipWalkLogsums", "S") + skipBikeLogsums= RunMacro("read properties",properties,"RunModel.skipBikeLogsums", "S") + skipBuildHwyNetwork = RunMacro("read properties",properties,"RunModel.skipBuildHwyNetwork", "S") + skipBuildTransitNetwork= RunMacro("read properties",properties,"RunModel.skipBuildTransitNetwork", "S") + startFromIteration = s2i(RunMacro("read properties",properties,"RunModel.startFromIteration", "S")) + skipHighwayAssignment = RunMacro("read properties array",properties,"RunModel.skipHighwayAssignment", "S") + skipHighwaySkimming = RunMacro("read properties array",properties,"RunModel.skipHighwaySkimming", "S") + skipTransitSkimming = RunMacro("read properties array",properties,"RunModel.skipTransitSkimming", "S") + skipCoreABM = RunMacro("read properties array",properties,"RunModel.skipCoreABM", "S") + skipOtherSimulateModel = RunMacro("read properties array",properties,"RunModel.skipOtherSimulateModel", "S") + skipSpecialEventModel = RunMacro("read properties array",properties,"RunModel.skipSpecialEventModel", "S") + skipCTM = RunMacro("read properties array",properties,"RunModel.skipCTM", "S") + skipEI = RunMacro("read properties array",properties,"RunModel.skipEI", "S") + skipTruck = RunMacro("read properties array",properties,"RunModel.skipTruck", "S") + skipTripTableCreation = RunMacro("read properties array",properties,"RunModel.skipTripTableCreation", "S") + skipFinalHighwayAssignment = RunMacro("read properties",properties,"RunModel.skipFinalHighwayAssignment", "S") + skipFinalTransitAssignment = RunMacro("read properties",properties,"RunModel.skipFinalTransitAssignment", "S") + skipFinalHighwaySkimming = RunMacro("read properties",properties,"RunModel.skipFinalHighwaySkimming", "S") + skipFinalTransitSkimming = RunMacro("read properties",properties,"RunModel.skipFinalTransitSkimming", "S") + skipLUZSkimCreation = RunMacro("read properties",properties,"RunModel.skipLUZSkimCreation", "S") + skipDataExport = RunMacro("read properties",properties,"RunModel.skipDataExport", "S") + skipDataLoadRequest = RunMacro("read properties",properties,"RunModel.skipDataLoadRequest", "S") + skipDeleteIntermediateFiles = RunMacro("read properties",properties,"RunModel.skipDeleteIntermediateFiles", "S") + precision = RunMacro("read properties",properties,"RunModel.MatrixPrecision", "S") + minSpaceOnC=RunMacro("read properties",properties,"RunModel.minSpaceOnC", "S") + + // Swap Server Configurations + runString = path+"\\bin\\serverswap.bat "+drive+" "+path_no_drive+" "+path_forward_slash + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Server Config Swap: "+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run ServerSwap", runString) + if !ok then goto quit + ok=RunMacro("find String","\\logFiles\\serverswap.log","FATAL") + if !ok then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","ServerSwap failed! Open logFiles/serverswap.log for details."}) + goto quit + end + + //Update year specific properties + runString = path+"\\bin\\updateYearSpecificProps.bat "+drive+" "+path_no_drive+" "+path_forward_slash + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Update year specific properties: "+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Update Year Specific Properties", runString) + if !ok then goto quit + + + //check free space on C drive + runString = path+"\\bin\\checkFreeSpaceOnC.bat "+minSpaceOnC + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Checking if there is enough space on C drive: "+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Check space on C drive", runString) + + //check AT and Transit networks consistency + runString = path+"\\bin\\checkAtTransitNetworkConsistency.cmd "+drive+" "+path_forward_slash + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Checking if AT and Transit Networks are consistent: "+" "+runString}) + RunProgram(runString, ) + ok=RunMacro("find String","\\logFiles\\AtTransitCheck_event.log","FATAL") + if !ok then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","AT and Transit network consistency chekcing failed! Open AtTransitCheck_event.log for details."}) + goto quit + end + + // copy bike logsums from input to output folder + if skipCopyBikeLogsum = "false" then do + CopyFile(inputDir+"\\bikeMgraLogsum.csv", outputDir+"\\bikeMgraLogsum.csv") + CopyFile(inputDir+"\\bikeTazLogsum.csv", outputDir+"\\bikeTazLogsum.csv") + end + if skipBikeLogsums = "false" then do + runString = path+"\\bin\\runSandagBikeLogsums.cmd "+drive+" "+path_forward_slash + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run create AT logsums and walk impedances"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run AT-Logsums", runString) + if !ok then goto quit + end + + // copy walk impedance from input to output folder + if skipCopyWalkImpedance = "false" then do + CopyFile(inputDir+"\\walkMgraEquivMinutes.csv", outputDir+"\\walkMgraEquivMinutes.csv") + CopyFile(inputDir+"\\walkMgraTapEquivMinutes.csv", outputDir+"\\walkMgraTapEquivMinutes.csv") + end + if skipWalkLogsums = "false" then do + runString = path+"\\bin\\runSandagWalkLogsums.cmd "+drive+" "+path_forward_slash + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run create AT logsums and walk impedances"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run AT-Logsums", runString) + if !ok then goto quit + end + + // copy initial trip tables from input to output folder + if skipCopyWarmupTripTables = "false" then do + CopyFile(inputDir+"\\Trip_EA_VOT.mtx", outputDir+"\\Trip_EA_VOT.mtx") + CopyFile(inputDir+"\\Trip_AM_VOT.mtx", outputDir+"\\Trip_AM_VOT.mtx") + CopyFile(inputDir+"\\Trip_MD_VOT.mtx", outputDir+"\\Trip_MD_VOT.mtx") + CopyFile(inputDir+"\\Trip_PM_VOT.mtx", outputDir+"\\Trip_PM_VOT.mtx") + CopyFile(inputDir+"\\Trip_EV_VOT.mtx", outputDir+"\\Trip_EV_VOT.mtx") + end + + // Build highway network + if skipBuildHwyNetwork = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run create hwy"}) + ok = RunMacro("TCB Run Macro", 1, "run create hwy",{}) + if !ok then goto quit + end + + // Create transit routes + if skipBuildTransitNetwork = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run create all transit"}) + ok = RunMacro("TCB Run Macro", 1, "Create all transit",{}) + if !ok then goto quit + end + + //Looping + for iteration = startFromIteration to max_iterations do + + if skipCoreABM[iteration] = "false" or skipOtherSimulateModel[iteration] = "false" or skipSpecialEventModel[iteration] = "false" then do //Wu modified to add special event model 5/19/2017 + // Start matrix manager locally + runString = path+"\\bin\\runMtxMgr.cmd "+drive+" "+path_no_drive + ok = RunMacro("TCB Run Command", 1, "Start matrix manager", runString) + if !ok then goto quit + + // Start JPPF driver + runString = path+"\\bin\\runDriver.cmd "+drive+" "+path_no_drive + ok = RunMacro("TCB Run Command", 1, "Start JPPF Driver", runString) + if !ok then goto quit + + // Start HH Manager, and worker nodes + runString = path+"\\bin\\StartHHAndNodes.cmd "+drive+" "+path_no_drive + ok = RunMacro("TCB Run Command", 1, "Start HH Manager, JPPF Driver, and nodes", runString) + if !ok then goto quit + end + + // Run highway assignment + if skipHighwayAssignment[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - hwy assignment"}) + ok = RunMacro("TCB Run Macro", 1, "hwy assignment",{iteration, assignByVOT}) + if !ok then goto quit + end + + // Skim highway network + if skipHighwaySkimming[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Hwy skim all"}) + ok = RunMacro("TCB Run Macro", 1, "Hwy skim all",{skimByVOT}) + if !ok then goto quit + + // Create drive to transit access file + runString = path+"\\bin\\CreateD2TAccessFile.bat "+drive+" "+path_forward_slash + + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java - Creating drive to transit access file"}) + ok = RunMacro("TCB Run Command", 1, "Creating drive to transit access file", runString) + if !ok then goto quit + end + + // Skim transit network + if skipTransitSkimming[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Build transit skims"}) + ok = RunMacro("TCB Run Macro", 1, "Build transit skims",{}) + if !ok then goto quit + end + + // First move some trip matrices so model will crash if ctramp model doesn't produced csv/mtx files for assignment + if (iteration > startFromIteration) then do + fromDir = outputDir + toDir = outputDir+"\\iter"+String(iteration-1) + //check for directory of output + if GetDirectoryInfo(toDir, "Directory")=null then do + CreateDirectory( toDir) + end + status = RunProgram("cmd.exe /c copy "+fromDir+"\\auto*.mtx "+toDir+"\\auto*.mtx",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\tran*.mtx "+toDir+"\\tran*.mtx",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\nmot*.mtx "+toDir+"\\nmot*.mtx",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\othr*.mtx "+toDir+"\\othr*.mtx",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\trip*.mtx "+toDir+"\\trip*.mtx",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\internalExternalTrips.csv "+toDir+"\\internalExternalTrips.csv",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\airport_out.csv "+toDir+"\\airport_out.csv",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\crossBorderTrips.csv "+toDir+"\\crossBorderTrips.csv",) + status = RunProgram("cmd.exe /c copy "+fromDir+"\\visitorTrips.csv "+toDir+"\\visitorTrips.csv",) + +// status = RunProgram("cmd.exe /c copy "+fromDir+"\\daily*.mtx "+toDir+"\\daily*.mtx",) +// status = RunProgram("cmd.exe /c copy "+fromDir+"\\comm*.mtx "+toDir+"\\comm*.mtx",) + + status = RunProgram("cmd.exe /c del "+fromDir+"\\auto*.mtx",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\tran*.mtx",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\nmot*.mtx",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\othr*.mtx",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\trip*.mtx",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\internalExternalTrips.csv",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\airport_out.csv",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\crossBorderTrips.csv",) + status = RunProgram("cmd.exe /c del "+fromDir+"\\visitorTrips.csv",) + +// status = RunProgram("cmd.exe /c del "+fromDir+"\\daily*.mtx",) +// status = RunProgram("cmd.exe /c del "+fromDir+"\\comm*.mtx",) + + end + + + // Run CT-RAMP model + if skipCoreABM[iteration] = "false" then do + runString = path+"\\bin\\runSandagAbm_SDRM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run CT-RAMP"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) + if !ok then goto quit + end + + + // Run airport model, visitor model, cross-border model, internal-external model + if skipOtherSimulateModel[iteration] = "false" then do + runString = path+"\\bin\\runSandagAbm_SMM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run airport model, visitor model, cross-border model"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) + if !ok then goto quit + end + + // Run special event model + if skipSpecialEventModel[iteration] = "false" then do + runString = path+"\\bin\\runSandagAbm_SEM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run special event model"+" "+runString}) + ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) + if !ok then goto quit + end + + + if skipCTM[iteration] = "false" then do + //Commercial vehicle trip generation + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle generation"}) + ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Generation",{}) + if !ok then goto quit + + //Commercial vehicle trip distribution + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle distribution"}) + ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Distribution",{}) + if !ok then goto quit + + //Commercial vehicle time-of-day + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle Time Of Day"}) + ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Time Of Day",{}) + if !ok then goto quit + + //Commercial vehicle toll diversion model + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle Toll Diversion"}) + ok = RunMacro("TCB Run Macro", 1, "cv toll diversion model",{}) + if !ok then goto quit + + // reduce commerical travel matrix precisions + RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce matrix precision for commVehTODTrips.mtx"}) + RunMacro("reduce matrix precision",outputDir,"commVehTODTrips.mtx", precision) + end + + /* + @WSU 2-23-2017 + Run EI and truck models only in the starting iteration (not necessarily the 1st iteration) + Purpose: + Reduce number of iterations to cut model run time by 2-2.5 hrs + Notes: + 1) Combined EI and truck trips are less than 2.5% of total trips; + 2) Trip generations are not sensitive to skims, total EI and truck demands are not affected; + 3) Skims only affect EI and truck destination choices + */ + if iteration = startFromIteration then do + //Run External(U.S.)-Internal Model + if skipEI[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - US to SD External Trip Model"}) + ok = RunMacro("TCB Run Macro", 1, "US to SD External Trip Model",{}) + if !ok then goto quit + + // reduce EI matrix precisions + + m={"usSdWrk_EA.mtx","usSdWrk_AM.mtx","usSdWrk_MD.mtx","usSdWrk_PM.mtx","usSdWrk_EV.mtx","usSdNon_EA.mtx","usSdNon_AM.mtx","usSdNon_MD.mtx","usSdNon_PM.mtx","usSdNon_EV.mtx"} + for i = 1 to m.length do + RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce precision for:"+m[i]}) + RunMacro("reduce matrix precision",outputDir,m[i], precision) + end + end + + //Run Truck Model + if skipTruck[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - truck model"}) + ok = RunMacro("truck model",properties, iteration) + if !ok then goto quit + + // reduce truck matrix precisions + m={"dailyDistributionMatricesTruckEA.mtx","dailyDistributionMatricesTruckAM.mtx","dailyDistributionMatricesTruckMD.mtx","dailyDistributionMatricesTruckPM.mtx","dailyDistributionMatricesTruckEV.mtx"} + for i = 1 to m.length do + RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce precision for:"+m[i]}) + RunMacro("reduce matrix precision",outputDir,m[i], precision) + end + end + end + + //Construct trip tables + if skipTripTableCreation[iteration] = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create Auto Tables"}) + ok = RunMacro("TCB Run Macro", 1, "Create Auto Tables",{}) + if !ok then goto quit + + // reduce EE matrix precisions + RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce matrix precision for externalExternalTrips.mtx"}) + RunMacro("reduce matrix precision",outputDir,"externalExternalTrips.mtx", precision) + end + + end + + // Run final highway assignment + if skipFinalHighwayAssignment = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - hwy assignment"}) + ok = RunMacro("TCB Run Macro", 1, "hwy assignment",{4,assignByVOT}) + if !ok then goto quit + end + + if skipFinalTransitAssignment = "false" then do + //Construct transit trip tables + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create Transit Tables"}) + ok = RunMacro("TCB Run Macro", 1, "Create Transit Tables",{}) + if !ok then goto quit + + //Run final and only transit assignment + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Assign Transit"}) + ok = RunMacro("TCB Run Macro", 1, "Assign Transit",{4}) + if !ok then goto quit + end + + // Skim highway network based on final highway assignment + if skipFinalHighwaySkimming = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Hwy skim all"}) + ok = RunMacro("TCB Run Macro", 1, "Hwy skim all",{}) + if !ok then goto quit + end + + // Skim transit network based on final transit assignemnt + if skipFinalTransitSkimming = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Build transit skims"}) + ok = RunMacro("TCB Run Macro", 1, "Build transit skims",{}) + if !ok then goto quit + end + + //Create LUZ skims + if skipLUZSkimCreation = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create LUZ Skims"}) + ok = RunMacro("TCB Run Macro", 1, "Create LUZ Skims",{}) + if !ok then goto quit + end + + //export TransCAD data (networks and trip tables) + if skipDataExport = "false" then do + RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Export TransCAD Data"}) + ok = RunMacro("TCB Run Macro", 1, "ExportSandagData",{}) + if !ok then goto quit + + // export core ABM data + runString = path+"\\bin\\DataExporter.bat "+drive+" "+path_no_drive + ok = RunMacro("TCB Run Command", 1, "Export core ABM data", runString) + if !ok then goto quit + end + + //request data load after model run finish successfully + if skipDataLoadRequest = "false" then do + runString = path+"\\bin\\DataLoadRequest.bat "+drive+path_no_drive+" "+String(max_iterations)+" "+scenarioYear+" "+sample_rate[max_iterations] + ok = RunMacro("TCB Run Command", 1, "Data load request", runString) + if !ok then goto quit + end + + // delete trip table files in iteration sub folder if model finishes without crashing + if skipDeleteIntermediateFiles = "false" then do + for iteration = startFromIteration to max_iterations-1 do + toDir = outputDir+"\\iter"+String(iteration-1) + status = RunProgram("cmd.exe /c del "+toDir+"\\auto*.mtx",) + status = RunProgram("cmd.exe /c del "+toDir+"\\tran*.mtx",) + status = RunProgram("cmd.exe /c del "+toDir+"\\nmot*.mtx",) + status = RunProgram("cmd.exe /c del "+toDir+"\\othr*.mtx",) + status = RunProgram("cmd.exe /c del "+toDir+"\\trip*.mtx",) + end + end + + RunMacro("TCB Closing", ok, "False") + return(1) + quit: + return(0) +EndMacro diff --git a/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc b/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc new file mode 100644 index 0000000..5213abf --- /dev/null +++ b/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc @@ -0,0 +1,293 @@ +Macro "_OpenTable" (file,view_name) + ext = Lower(Right(file,3)) + if ext = "csv" then do + type = "CSV" + end + else if ext = "bin" then do + type = "FFB" + end + else do + ShowMessage("Cannot open table of type " + ext) + ShowMessage(2) + end + return(OpenTable(view_name,type,{file,})) +EndMacro + +Macro "GetTodToken" + return("%%TOD%%") +EndMacro + +Macro "GetSkimToken" + return("%%SKIM%%") +EndMacro + +Macro "GetTodPeriods" + return({"EA","AM","MD","PM","EV"}) +EndMacro + +Macro "GetSkimPeriods" + return({"AM","PM","OP"}) +EndMacro + +Macro "GetTodSkimMapping" + mapping = null + mapping.EA = "OP" + mapping.AM = "AM" + mapping.MD = "OP" + mapping.PM = "PM" + mapping.EV = "OP" + return(mapping) +EndMacro + +Macro "GetHighwayModes" + return({"SOV_GP","SOV_PAY","SR2_GP","SR2_HOV","SR2_PAY","SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"}) +EndMacro + +Macro "GetTransitModes" + return({"LOC","LRT","EXP","CMR","BRT"}) +EndMacro + +Macro "GetTransitAccessModes" + return({"WLK","PNR","KNR"}) +EndMacro + +Macro "ExportNetworkToCsv" (network_file,output_file_base) + {node_layer,line_layer} = GetDBLayers(network_file) + network_layer = AddLayerToWorkspace("network_line_layer",network_file,line_layer) + SetLayer(network_layer) + view = GetView() + ExportView(view+"|","CSV",output_file_base+".csv",,{{"CSV Header", "True"},{"Force Numeric Type", "double"}}) + CloseView(view) + +EndMacro + + +Macro "ExportHwyloadtoCSV"(input_file_base, output_file_base) + + tod_periods = RunMacro("GetTodPeriods") + + for t = 1 to tod_periods.length do + tod = tod_periods[t] + input_file= input_file_base+tod+".bin" + view = OpenTable("Binary Table","FFB",{input_file,}, {{"Shared", "True"}}) + SetView(view) + ExportView(view+"|", "CSV", output_file_base+tod+".csv", + {"ID1", + "AB_Flow_PCE", + "BA_Flow_PCE", + "AB_Time", + "BA_Time", + "AB_VOC", + "BA_VOC", + "AB_V_Dist_T", + "BA_V_Dist_T", + "AB_VHT", + "BA_VHT", + "AB_Speed", + "BA_Speed", + "AB_VDF", + "BA_VDF", + "AB_MSA_Flow", + "BA_MSA_Flow", + "AB_MSA_Time", + "BA_MSA_Time", + "AB_Flow_SOV_GP_LOW", + "BA_Flow_SOV_GP_LOW", + "AB_Flow_SOV_PAY_LOW", + "BA_Flow_SOV_PAY_LOW", + "AB_Flow_SR2_GP_LOW", + "BA_Flow_SR2_GP_LOW", + "AB_Flow_SR2_HOV_LOW", + "BA_Flow_SR2_HOV_LOW", + "AB_Flow_SR2_PAY_LOW", + "BA_Flow_SR2_PAY_LOW", + "AB_Flow_SR3_GP_LOW", + "BA_Flow_SR3_GP_LOW", + "AB_Flow_SR3_HOV_LOW", + "BA_Flow_SR3_HOV_LOW", + "AB_Flow_SR3_PAY_LOW", + "BA_Flow_SR3_PAY_LOW", + "AB_Flow_SOV_GP_MED", + "BA_Flow_SOV_GP_MED", + "AB_Flow_SOV_PAY_MED", + "BA_Flow_SOV_PAY_MED", + "AB_Flow_SR2_GP_MED", + "BA_Flow_SR2_GP_MED", + "AB_Flow_SR2_HOV_MED", + "BA_Flow_SR2_HOV_MED", + "AB_Flow_SR2_PAY_MED", + "BA_Flow_SR2_PAY_MED", + "AB_Flow_SR3_GP_MED", + "BA_Flow_SR3_GP_MED", + "AB_Flow_SR3_HOV_MED", + "BA_Flow_SR3_HOV_MED", + "AB_Flow_SR3_PAY_MED", + "BA_Flow_SR3_PAY_MED", + "AB_Flow_SOV_GP_HIGH", + "BA_Flow_SOV_GP_HIGH", + "AB_Flow_SOV_PAY_HIGH", + "BA_Flow_SOV_PAY_HIGH", + "AB_Flow_SR2_GP_HIGH", + "BA_Flow_SR2_GP_HIGH", + "AB_Flow_SR2_HOV_HIGH", + "BA_Flow_SR2_HOV_HIGH", + "AB_Flow_SR2_PAY_HIGH", + "BA_Flow_SR2_PAY_HIGH", + "AB_Flow_SR3_GP_HIGH", + "BA_Flow_SR3_GP_HIGH", + "AB_Flow_SR3_HOV_HIGH", + "BA_Flow_SR3_HOV_HIGH", + "AB_Flow_SR3_PAY_HIGH", + "BA_Flow_SR3_PAY_HIGH", + "AB_Flow_lhdn", + "BA_Flow_lhdn", + "AB_Flow_mhdn", + "BA_Flow_mhdn", + "AB_Flow_hhdn", + "BA_Flow_hhdn", + "AB_Flow_lhdt", + "BA_Flow_lhdt", + "AB_Flow_mhdt", + "BA_Flow_mhdt", + "AB_Flow_hhdt", + "BA_Flow_hhdt", + "AB_Flow", + "BA_Flow"}, + {{"CSV Header", "True"},{"Force Numeric Type", "double"}}) + CloseView(view) + end + ok=1 + quit: + return(ok) +EndMacro + + + + + +Macro "BuildTransitFlowOptions" + //name,source_name,primary key column + skim_token = RunMacro("GetSkimToken") + topts = {{"ROUTE", "Route" ,True }, + {"FROM_STOP", "From_Stop" ,True }, + {"TO_STOP", "To_Stop" ,True }, + {"CENTROID", "Centroid" ,False }, + {"FROMMP", "From_MP" ,False }, + {"TOMP", "To_MP" ,False }, + {"TRANSITFLOW", "TransitFlow" ,False }, + {"BASEIVTT", "BaseIVTT" ,False }, + {"COST", "Cost" ,False }, + {"VOC", "VOC" ,False }} + fopts = {"flow",""} + return({topts,fopts}) +EndMacro + +Macro "BuildOnOffOptions" + //name,source_name,primary key column + skim_token = RunMacro("GetSkimToken") + topts = {{"ROUTE", "ROUTE" ,True }, + {"STOP", "STOP" ,True }, + {"BOARDINGS", "On" ,False}, + {"ALIGHTINGS", "Off" ,False}, + {"WALKACCESSON", "WalkAccessOn" ,False}, + {"DIRECTTRANSFERON", "DirectTransferOn" ,False}, + {"WALKTRANSFERON", "WalkTransferOn" ,False}, + {"DIRECTTRANSFEROFF", "DirectTransferOff" ,False}, + {"WALKTRANSFEROFF", "WalkTransferOff" ,False}, + {"EGRESSOFF", "EgressOff" ,False}} + fopts = {"ono",""} + return({topts,fopts}) +EndMacro + + +Macro "BuildAggFlowOptions" + //name,source_name,primary key column + skim_token = RunMacro("GetSkimToken") + topts = {{"LINK_ID", "ID1" ,True }, + {"AB_TransitFlow", "AB_TransitFlow" ,false}, + {"BA_TransitFlow", "BA_TransitFlow" ,False}, + {"AB_NonTransit", "AB_NonTransit" ,False}, + {"BA_NonTransit", "BA_NonTransit" ,False}, + {"AB_TotalFlow", "AB_TotalFlow" ,False}, + {"BA_TotalFlow", "BA_TotalFlow" ,False}, + {"AB_Access_Walk_Flow", "AB_Access_Walk_Flow" ,False}, + {"BA_Access_Walk_Flow", "BA_Access_Walk_Flow" ,False}, + {"AB_Xfer_Walk_Flow", "AB_Xfer_Walk_Flow" ,False}, + {"BA_Xfer_Walk_Flow", "BA_Xfer_Walk_Flow" ,False}, + {"AB_Egress_Walk_Flow", "AB_Egress_Walk_Flow" ,False}, + {"BA_Egress_Walk_Flow", "BA_Egress_Walk_Flow" ,False}} + fopts = {"agg",""} + return({topts,fopts}) +EndMacro + + + +Macro "ExportTransitTablesToCsv" (results_dir,transit_options,output_file_base) + fopts = transit_options[2] + transit_options = transit_options[1] + header = "MODE,ACCESSMODE,TOD" + for i = 1 to transit_options.length do + header = header + "," + transit_options[i][1] + end + + f = OpenFile(output_file_base + ".csv","w") + WriteLine(f,header) + + table_name = Upper(Right(output_file_base,Len(output_file_base) - PositionTo(,output_file_base,"\\"))) + + transit_modes = RunMacro("GetTransitModes") + transit_access_modes = RunMacro("GetTransitAccessModes") + tod_periods = RunMacro("GetTodPeriods") + tod_skim_mapping = RunMacro("GetTodSkimMapping") + skim_token = RunMacro("GetSkimToken") + + for i = 1 to tod_periods.length do + tod = tod_periods[i] + for t = 1 to transit_modes.length do + transit_mode = transit_modes[t] + for ta = 1 to transit_access_modes.length do + transit_access_mode = transit_access_modes[ta] + tf = RunMacro("FormPath",{results_dir,fopts[1] + transit_access_modes[ta] + "_" + transit_mode + "_" + tod + fopts[2] + ".bin"}) + if GetFileInfo(tf) <> null then do + view = RunMacro("_OpenTable",tf,"tview") + + rh = GetFirstRecord(view + "|",) + while rh <> null do + line = transit_mode + "," + transit_access_mode + "," + tod + for to = 1 to transit_options.length do + line = line + "," + RunMacro("ToString",view.(Substitute(transit_options[to][2],skim_token,tod_skim_mapping.(tod),))) + end + WriteLine(f,line) + structure = GetViewStructure(view) + for to = 1 to transit_options.length do + field_id = -1 + field = Substitute(transit_options[to][2],skim_token,tod_skim_mapping.(tod),) + for s = 1 to structure.length do + if structure[s][1] = field then do + field_id = s + end + end + if field_id < 1 then do + ShowMessage("couldn't find field: " + field) + ShowMessage(2) + end + + end + rh = GetNextRecord(view + "|",rh,) + end + CloseView(view) + end + end + end + end + + CloseFile(f) + +EndMacro + +Macro "SaveMatrix" (matrix_in,file_out) + m = OpenMatrix(matrix_in,) + CreateTableFromMatrix(m,file_out,"CSV",{{"Complete","Yes"}}) +EndMacro + + diff --git a/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc b/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc new file mode 100644 index 0000000..d0be578 --- /dev/null +++ b/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc @@ -0,0 +1,223 @@ +//Aggregate transit select link trip matrix +//Input: trn_sellinkxxxxx_xx.mtx select link trip table by access mode, line haul mode, and time of day +// a text file with a list of TAPs to sum up trips +//Ouput: trn_sellink_total.csv +//Author: Ziying Ouyang +// @ziying.ouyang@sandag.org +//Date: Feb 5, 2014 + + +Macro "Sum Up Select Link Transit Trips" + shared path_study, path, outputDir + RunMacro("TCB Init") + + sum_all = 0 + sum_sg = 0 + pct_sg = 0 + + outputDir = path+"\\output" + sellink_file = outputDir+"\\trn_sellinkWLK_LRT_EA.mtx" + if GetFileInfo(sellink_file) <> null then do + periodName = {"_EA", "_AM", "_MD", "_PM", "_EV"} + + matrixCore = { + "WLK_LOC", + "WLK_EXP", + "WLK_BRT", + "WLK_LRT", + "WLK_CMR", + "PNR_LOC", + "PNR_EXP", + "PNR_BRT", + "PNR_LRT", + "PNR_CMR", + "KNR_LOC", + "KNR_EXP", + "KNR_BRT", + "KNR_LRT", + "KNR_CMR" } + + //open smart growth tap CSV file + in_file = path_study + "\\sg_tap.csv" + if GetFileInfo(in_file) <> null then do + + view = OpenTable("SmartGrowthTap","CSV",{in_file,}, {{"Shared", "True"}}) + SetView(view) + vw_TAP = GetView() + view_set = vw_TAP +"|" + + for per = 1 to periodName.length do + for mat = 1 to matrixCore.length do + + + sellkmtx = outputDir+"\\trn_sellink"+matrixCore[mat]+periodName[per]+".mtx" + m = OpenMatrix(sellkmtx,) + + sg_orig_index = CreateMatrixIndex("Smart Growth Orig", m, "Row",view_set, "sg_tap", "sg_tap" ) + sg_dest_index = CreateMatrixIndex("Smart Growth Dest", m, "Column",view_set, "sg_tap", "sg_tap" ) + + mc = CreateMatrixCurrency(m, , , , ) + mc_sg_orig = CreateMatrixCurrency(m, ,"Smart Growth Orig","Columns", ) + mc_sg_dest = CreateMatrixCurrency(m, ,"Rows","Smart Growth Dest", ) + mc_sg_to_sg = CreateMatrixCurrency(m, ,"Smart Growth Orig","Smart Growth Dest", ) + + sum_row = GetMatrixMarginals(mc, "Sum", "row" ) + + sum_row_sg_orig = GetMatrixMarginals(mc_sg_orig, "Sum", "row" ) + sum_row_sg_dest = GetMatrixMarginals(mc_sg_dest, "Sum", "row" ) + sum_row_sg_to_sg = GetMatrixMarginals(mc_sg_to_sg, "Sum", "row" ) + + sum_all = sum_all + Sum(sum_row) + sum_sg = sum_sg + Sum(sum_row_sg_orig) + Sum(sum_row_sg_dest) - Sum(sum_row_sg_to_sg) + + DeleteMatrixIndex(m, "Smart Growth Orig") + DeleteMatrixIndex(m, "Smart Growth Dest") + + //close matrix + mc = null + m = null + + end + end + + if sum_all > 0 then pct_sg = sum_sg /sum_all + + out_file = path_study + "\\trn_sellink_sg.csv" + + dif2 = GetDirectoryInfo(out_file,"file") + if dif2.length <= 0 then do + fpr = OpenFile(out_file,"w") + WriteLine(fpr, "Scenario, Total select link trips, smart growth select link trips, % of sg trips") + end + else do + fpr = OpenFile(out_file,"a") + end + + WriteLine(fpr,path + "," + r2s(sum_all) + "," + r2s(sum_sg) + "," + r2s(pct_sg)) + CloseFile(fpr) + end + else ShowMessage("Missing smart growth tap file in " + path_study) + End + RunMacro("close all") + RunMacro("TCB Closing", ok, "False") + return(1) +EndMacro + +Macro "Sum Up Select Link Highway Trips" + + shared path_study, path, outputDir + RunMacro("TCB Init") + + outputDir = path+"\\output" + sellink_file = outputDir+"\\select_EA.mtx" + if GetFileInfo(sellink_file) <> null then do + + m = OpenMatrix(sellink_file,) + coreNames = GetMatrixCoreNames(m) + m=null + + //hard coded the occupancy rate, could be improved based on coreNames + occupancy = {1, 1, 2, 2, 2, 3.34, 3.34, 3.34, 1, 1, 1, 1, 1, 1} + + sum_all = 0 + sum_sg = 0 + pct_sg = 0 + sum_ind = 0 + pct_ind = 0 + + periodName = {"_EA","_AM","_MD","_PM","_EV"} + + //open smart growth TAZ CSV file + in_file = path_study + "\\sg_taz.csv" + in_file_indian = path_study + "\\indian_reservation_taz.csv" + + if GetFileInfo(in_file) <> null then do + view = OpenTable("SmartGrowthTAZ","CSV",{in_file,}, {{"Shared", "True"}}) + SetView(view) + vw_TAZ = GetView() + view_set = vw_TAZ +"|" + + view_ind = OpenTable("IndianReservationTAZ","CSV",{in_file_indian,}, {{"Shared", "True"}}) + SetView(view_ind) + vw_TAZ_ind = GetView() + view_set_ind = vw_TAZ_ind +"|" + + for per = 1 to periodName.length do + + sellkmtx = outputDir + "\\select" + periodName[per] +".mtx" + m = OpenMatrix(sellkmtx,) + + sg_index_orig = CreateMatrixIndex("SG Index Orig", m, "Row",view_set,"SG_TAZ" , "SG_TAZ" ) + sg_index_dest = CreateMatrixIndex("SG Index Dest", m, "Column",view_set,"SG_TAZ" , "SG_TAZ" ) + + ind_index_orig = CreateMatrixIndex("Indian Index Orig", m, "Row",view_set_ind,"Indian_TAZ" , "Indian_TAZ" ) + ind_index_dest = CreateMatrixIndex("Indian Index Dest", m, "Column",view_set_ind,"Indian_TAZ" , "Indian_TAZ" ) + + mc = CreateMatrixCurrencies(m, "Rows", "Columns", ) + mc_sg_orig = CreateMatrixCurrencies(m, "SG Index Orig","Columns",) + mc_sg_dest = CreateMatrixCurrencies(m, "Rows","SG Index Dest",) + mc_sg_to_sg = CreateMatrixCurrencies(m, "SG Index Orig","SG Index Dest",) + + mc_ind_orig = CreateMatrixCurrencies(m, "Indian Index Orig","Columns",) + mc_ind_dest = CreateMatrixCurrencies(m, "Rows","Indian Index Dest",) + mc_ind_to_ind = CreateMatrixCurrencies(m, "Indian Index Orig","Indian Index Dest",) + + // select link assignment trip matrix includes the total trips as the last core + for core = 1 to coreNames.length - 1 do + sum_row = GetMatrixMarginals(mc.(coreNames[core]), "Sum", "row" ) + + sum_row_sg_orig = GetMatrixMarginals(mc_sg_orig.(coreNames[core]), "Sum", "row" ) + sum_row_sg_dest = GetMatrixMarginals(mc_sg_dest.(coreNames[core]), "Sum", "row" ) + sum_row_sg_to_sg = GetMatrixMarginals(mc_sg_to_sg.(coreNames[core]), "Sum", "row" ) + + + sum_row_ind_orig = GetMatrixMarginals(mc_ind_orig.(coreNames[core]), "Sum", "row" ) + sum_row_ind_dest = GetMatrixMarginals(mc_ind_dest.(coreNames[core]), "Sum", "row" ) + sum_row_ind_to_ind = GetMatrixMarginals(mc_ind_to_ind.(coreNames[core]), "Sum", "row" ) + + sum_all = sum_all + Sum(sum_row) * occupancy[core] + sum_sg = sum_sg + (Sum(sum_row_sg_orig) + Sum(sum_row_sg_dest)) * occupancy[core] - Sum(sum_row_sg_to_sg) * occupancy[core] + sum_ind = sum_ind + (Sum(sum_row_ind_orig) + Sum(sum_row_ind_dest)) * occupancy[core] - Sum(sum_row_ind_to_ind) * occupancy[core] + + end + + //close matrix + DeleteMatrixIndex(m, "SG Index Orig") + DeleteMatrixIndex(m, "SG Index Dest") + DeleteMatrixIndex(m, "Indian Index Orig") + DeleteMatrixIndex(m, "Indian Index Dest") + mc = null + mc_sg = null + mc_ind = null + m = null + end + + if sum_all > 0 then do + pct_sg = sum_sg /sum_all + pct_ind = sum_ind/sum_all + end + + out_file = path_study + "\\hwy_sellink.csv" + + dif2 = GetDirectoryInfo(out_file,"file") + if dif2.length <= 0 then do + fpr = OpenFile(out_file,"w") + WriteLine(fpr, "Scenario, Total select link trips, smart growth select link trips, % of sg trips, indian reservation trips, % of indian reservation trips") + end + else do + fpr = OpenFile(out_file,"a") + end + + WriteLine(fpr,path + "," + r2s(sum_all) + "," + r2s(sum_sg) + "," + r2s(pct_sg) + "," + r2s(sum_ind) + "," + r2s(pct_ind)) + CloseFile(fpr) + + end + else ShowMessage("Missing smart growth TAZ file in " + path_study) + + end + RunMacro("close all") + RunMacro("TCB Closing", ok, "False") + return(1) + +EndMacro + diff --git a/sandag_abm/src/main/gisdk/sellink_volume.rsc b/sandag_abm/src/main/gisdk/sellink_volume.rsc new file mode 100644 index 0000000..dd462c3 --- /dev/null +++ b/sandag_abm/src/main/gisdk/sellink_volume.rsc @@ -0,0 +1,85 @@ +/* + Export daily select link volumes by direction + Need to figure out vary by # of query + author: Ziying Ouyang zou@sandag.org + date: 12/16/2015 +*/ +Macro "ExportHwyloadtoCSV Select Link" + + shared path, input_path, output_path + path="T:\\projects\\sr13\\version13_3_0\\abm_runs\\2012" + input_path = path+"\\input" + output_path = path+"\\output" + + query_list = RunMacro("Get SL Query # from QRY",input_path) + if query_list.length > 0 then do + + Dim v_ab_flow_slk[query_list.length], v_ba_flow_slk[query_list.length] + Dim v_ab_flow_slk_pk[2,query_list.length], v_ba_flow_slk_pk[2,query_list.length] + input_file = {"hwyload_EA.bin","hwyload_AM.bin","hwyload_MD.bin","hwyload_PM.bin","hwyload_EV.bin"} + + fields = {"ID1"} + for j = 1 to query_list.length do + fields = fields + {"AB_Flow_" + query_list[j]} + fields = fields + {"BA_Flow_" + query_list[j]} + end + + for i = 1 to input_file.length do + view = OpenTable("Binary Table","FFB",{output_path+"\\"+input_file[i],}, {{"Shared", "True"}}) + SetView(view) + v_lodselk = GetDataVectors(view+"|", fields, ) + for j = 1 to query_list.length do + v_ab_flow_slk[j] = Nz(v_ab_flow_slk[j]) + v_lodselk[2*(j-1)+2] + v_ba_flow_slk[j] = Nz(v_ba_flow_slk[j]) + v_lodselk[2*(j-1)+3] + end + if i = 2 or i = 4 then do //save AM/PM select link ab/ba volumes (AM 1, PM 2) + for j = 1 to query_list.length do + v_ab_flow_slk_pk[i/2][j] = v_lodselk[2*(j-1)+2] + v_ba_flow_slk_pk[i/2][j] = v_lodselk[2*(j-1)+3] + end + end + + end + + header = "ID1" + for j = 1 to query_list.length do + header = header + "," + "AB_Flow_" + query_list[j] + "," + "BA_Flow_" + query_list[j] + "," + "Tot_Flow_" + query_list[j] + end + + f = OpenFile(output_path+ "\\"+"loadselk.csv","w") + WriteLine(f,header) + + for i = 1 to v_lodselk[1].length do + line = i2s(v_lodselk[1][i]) + for j = 1 to query_list.length do + line = line + "," + r2s(v_ab_flow_slk[j][i]) + "," + r2s(Nz(v_ba_flow_slk[j][i])) + "," + r2s(v_ab_flow_slk[j][i]+Nz(v_ba_flow_slk[j][i])) + end + WriteLine(f,line) + end + + closeFile(f) + //Peak Period Select Link Volumes + header = "ID1" + for j = 1 to query_list.length do + header = header + "," + "AB_Flow_AM_" + query_list[j] + "," + "BA_Flow_AM_" + query_list[j] + "," + "Tot_Flow_AM_" + query_list[j] + "," + "AB_Flow_PM_" + query_list[j] + "," + "BA_Flow_PM_" + query_list[j] + "," + "Tot_Flow_PM_" + query_list[j] + end + + f = OpenFile(output_path+ "\\"+"loadselkpk.csv","w") + WriteLine(f,header) + + for i = 1 to v_lodselk[1].length do + line = i2s(v_lodselk[1][i]) + for j = 1 to query_list.length do + line = line + "," + r2s(v_ab_flow_slk_pk[1][j][i]) + "," + r2s(Nz(v_ba_flow_slk_pk[1][j][i])) + "," + r2s(v_ab_flow_slk_pk[1][j][i]+Nz(v_ba_flow_slk_pk[1][j][i])) + line = line + "," + r2s(v_ab_flow_slk_pk[2][j][i]) + "," + r2s(Nz(v_ba_flow_slk_pk[2][j][i])) + "," + r2s(v_ab_flow_slk_pk[2][j][i]+Nz(v_ba_flow_slk_pk[2][j][i])) + end + WriteLine(f,line) + end + + CloseFile(f) + end + + else ShowMessage("Number of select links is 0") + RunMacro("close all") + +EndMacro diff --git a/sandag_abm/src/main/gisdk/trnassign.rsc b/sandag_abm/src/main/gisdk/trnassign.rsc new file mode 100644 index 0000000..e61c1e7 --- /dev/null +++ b/sandag_abm/src/main/gisdk/trnassign.rsc @@ -0,0 +1,177 @@ +/********************************************************************************* +Transit Assignment + + input files: transitrt.rts + CT-RAMP trip tables ("tranTrips_period.mtx", where period = EA, AM, MD, PM, and NT) + each file has 15 cores, named as follows: ACC_LHM_PER + where: + ACC = access mode - WLK,PNR,KNR + LHM = line-haul mode - LOC,EXP,BRT,LRT,CMR + PER = period - EA, AM, MD, PM, NT + Transit networks (localpk.tnw,localop,tnw,prempk.tnw,premop.tnw) + Transit route file (transitrt.rts) + output files: + 75 flow bin file (3 access modes * 5 line-haul modes * 5 time periods) + 75 walk bin file + 75 onoff bin file + 75 collapsed onoff files in both binary and csv format + +*********************************************************************************/ + +Macro "Assign Transit" (args) + + + shared path,inputDir, outputDir, mxtap + +iteration = args[1] +rt_file=outputDir + "\\transitrt.rts" + +periodName = {"_EA", "_AM", "_MD", "_PM", "_EV"} + +matrixCore = { + "WLK_LOC", + "WLK_EXP", + "WLK_BRT", + "WLK_LRT", + "WLK_CMR", + "PNR_LOC", + "PNR_EXP", + "PNR_BRT", + "PNR_LRT", + "PNR_CMR", + "KNR_LOC", + "KNR_EXP", + "KNR_BRT", + "KNR_LRT", + "KNR_CMR" } + + network={"locl","prem","prem","prem","prem","locl","prem","prem","prem","prem","locl","prem","prem","prem","prem"} + dim onOffTables[matrixCore.length * periodName.length] + + selinkqry = inputDir + "\\"+"sellink_transit.qry" + if GetFileInfo(selinkqry) <> null then sellink_flag = 1 //select link analysis is only for last iteration (4) + + i = 0 + k = 1 + for per = 1 to periodName.length do + for mat = 1 to matrixCore.length do + + i = i + 1 + + networkFile = outputDir+"\\"+network[mat]+periodName[per]+".tnw" + matrixFile = outputDir+"\\tranTotalTrips"+periodName[per]+".mtx" + matrixName = matrixCore[mat] + flowFile = outputDir+"\\flow"+matrixCore[mat]+periodName[per]+".bin" + walkFile = outputDir+"\\ntl"+matrixCore[mat]+periodName[per]+".bin" + onOffFile = outputDir+"\\ono"+matrixCore[mat]+periodName[per]+".bin" + aggFile = outputDir+"\\agg"+matrixCore[mat]+periodName[per]+".bin" + + onOffTables[k] = onOffFile + k = k +1 + + if sellink_flag = 1 & iteration = 4 then sellkmtx = outputDir+"\\trn_sellink"+matrixCore[mat]+periodName[per]+".mtx" + + + // STEP 1: Transit Assignment + Opts = null + Opts.Input.[Transit RS] = rt_file + Opts.Input.Network = networkFile + Opts.Input.[OD Matrix Currency] = {matrixFile,matrixName,,} + + Opts.Output.[Flow Table] = flowFile + Opts.Output.[Walk Flow Table] = walkFile + Opts.Output.[OnOff Table] = onOffFile + Opts.Output.[Aggre Table] = aggFile + Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 + if sellink_flag = 1 & iteration = 4 then do + Opts.Flag.critFlag = 1 + Opts.Global.[Critical Query File] = selinkqry + Opts.Output.[Critical Matrix].Label = "Critical Matrix" + Opts.Output.[Critical Matrix].[File Name] = sellkmtx + end + RunMacro("HwycadLog",{"trassigns.rsc: transit assigns","Transit Assignment PF: "+matrixCore[mat]+periodName[per]}) + ok = RunMacro("TCB Run Procedure", (per*100+mat), "Transit Assignment PF", Opts) + + if !ok then goto quit + + end + end + + properties = "\\conf\\sandag_abm.properties" + collapseOnOffByRoute = RunMacro("read properties",properties,"RunModel.collapseOnOffByRoute", "S") + if collapseOnOffByRoute = "true" then do + ok = RunMacro("Collapse OnOffs By Route", onOffTables, rt_file) + if !ok then goto quit + end + + quit: + RunMacro("close all") + Return( ok ) + +EndMacro +/************************************************************* +* +* A macro that will collapse transit on-offs by route and append +* route name. +* +* Arguments +* onOffTables An array of on-off tables +* rtsfile A transit route file +* +*************************************************************/ +Macro "Collapse OnOffs By Route" (onOffTables,rtsfile) + + {rte_lyr,stp_lyr,} = RunMacro("TCB Add RS Layers", rtsfile, "ALL", ) + + fields = { + {"On","Sum",}, + {"Off","Sum",}, + {"DriveAccessOn","Sum",}, + {"WalkAccessOn","Sum",}, + {"DirectTransferOn","Sum",}, + {"WalkTransferOn","Sum",}, + {"DirectTransferOff","Sum",}, + {"WalkTransferOff","Sum",}, + {"EgressOff","Sum",} + } + + // for all on off tables + for i = 1 to onOffTables.length do + + onOffView = OpenTable("OnOffTable", "FFB", {onOffTables[i], null}) + path = SplitPath(onOffTables[i]) + outFile = path[1]+path[2]+path[3]+"_COLL.bin" + + fields = GetFields(onOffView, "All") + + //include all fields in each table except for STOP and ROUTE + collFields = null + for j = 1 to fields[1].length do + + if(fields[1][j] !="STOP" and fields[1][j]!= "ROUTE") then do + + collFields = collFields + {{fields[1][j],"Sum",}} + + end + end + + // Collapse stops out of the table by collapsing on ROUTE + rslt = AggregateTable("CollapsedView", onOffView+"|", "FFB", outFile, "ROUTE", collFields, ) + + CloseView(onOffView) + + // Join the route layer for route name and other potentially useful data + onOffCollView = OpenTable("OnOffTableColl", "FFB", {outFile}) + joinedView = JoinViews("OnOffJoin", onOffCollView+".Route", rte_lyr+".Route_ID",) + + // Write the joined data to a binary file + outJoinFile = path[1]+path[2]+path[3]+"_COLL_JOIN.bin" + ExportView(joinedView+"|","FFB", outJoinFile , , ) + outJoinFile = path[1]+path[2]+path[3]+"_COLL_JOIN.csv" + ExportView(joinedView+"|","CSV", outJoinFile , , ) + end + + Return(1) + quit: + Return( RunMacro("TCB Closing", ret_value, True ) ) +EndMacro diff --git a/sandag_abm/src/main/gisdk/trnskim.rsc b/sandag_abm/src/main/gisdk/trnskim.rsc new file mode 100644 index 0000000..4313570 --- /dev/null +++ b/sandag_abm/src/main/gisdk/trnskim.rsc @@ -0,0 +1,1038 @@ +/******************************************************************************** +transit skim matrix to create skim matrix files + skim values includes fare, in vehicle time, initial wait time, + transfer wait time, tranfer walk time, access time, egress time + dwell time, transfer penalty, # of transfers + *TMO, or *TMP (depends on which time period) +for premium service skim *TMO, or *TMP by mode, additional cores were created +input files: transit.dbd - transit line layer + transitrt.rts - transit route layer + localop.tnw - local bus off peak transit network + localpk.tnw - local bus peak transit network + premop.tnw - premium bus service off peak transit network + prempk.tnw - premium bus service peak transit network + modenu.dbf - dbf file for mode table + fare.mtx - fare matrix for zonal fares for lightrail and coaster + +output files: + implbopx2.mtx - skim matrix file for local bus off peak service + implbpkx2.mtx - skim matrix file for local bus peak service + imppropx2.mtx - skim matrix file for premium bus off peak service + impprpkx2.mtx - skim matrix file for premium bus peak service + implr.mtx - skim ltrzone + impcr.mtx - skim crzone +history + 10/8/03 zou to change the fare system to flat fare and skim + rail link fields separately. + 9/3/04 retain the commuter rail from the light rail category + 1/25/05 rewrite the script using moduels + 4/14/09 ZOU changed it to 3tod +********************************************************************************/ +Macro "Create transit network" + + shared path,inputDir, outputDir, mxtap + ok=RunMacro("Create transit networks") + if !ok then goto quit + Return(1) + + quit: + Return(0) +EndMacro + +/*********************************************************************************** +*******************************************************************/ + +Macro "Build transit skims" + + shared path,inputDir, outputDir, mxtap + + ok = RunMacro("Update transit time fields") + if !ok then goto quit + + /* + ok=RunMacro("Create rail net") + if !ok then goto quit + */ + ok=RunMacro("Create transit networks") + if !ok then goto quit + + ok = RunMacro("Skim transit networks") + if !ok then goto quit + + /* + ok= RunMacro("zero null ivt time") + if !ok then goto quit +*/ + ok= RunMacro("Process transit skims") + if !ok then goto quit + + Return(1) + + quit: + Return(0) +EndMacro +/*********************************************************************************** + + +Inputs: + input\mode5tod.dbf Mode table + output\transit.dbd Transit line layer + output\xxxx_yy.tnw Transit networks + + where xxxxx is transit mode (locl or prem) and yy is period (EA, AM, MD, PM, EV) + +Outputs: + impxxxx_yy.mtx Transit skims + + where xxxxx is transit mode (locl or prem) and yy is period (EA, AM, MD, PM, EV) + +***********************************************************************************/ +Macro "Skim transit networks" + + shared path,inputDir, outputDir, mxtap + NumofCPU=2 + + mode_tb = inputDir+"\\mode5tod.dbf" + // xfer_tb = path+"\\modexfer.dbf" + db_file = outputDir + "\\transit.dbd" + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + db_node_lyr = db_file + "|" + node_lyr + + periods = {"_EA","_AM","_MD","_PM","_EV"} + modes = {"prem","locl"} + + //varies by modes + skmodes={{4,5,6,7,8,9,10},} + skvar={{"Fare", "Initial Wait Time", "Transfer Wait Time", "Transfer Walk Time", "Access Walk Time","Egress Walk Time","Dwelling Time", "Number of Transfers"}, + {"Fare", "In-Vehicle Time", "Initial Wait Time", "Transfer Wait Time", "Transfer Walk Time","Access Walk Time","Egress Walk Time", "Dwelling Time", "Number of Transfers"}} + + // not sure what the PRE and LOC time are yet + skvars={skvar[1]+{"Length","*TM"},skvar[2]} + + + for i = 1 to periods.length do + for j = 1 to modes.length do + Opts = null + // Opts.Global.[Force Threads] = NumofCPU + Opts.Input.Database = db_file + Opts.Input.Network = outputDir+"\\"+modes[j]+periods[i]+".tnw" + Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Selection", "Select * where id <"+i2s(mxtap)} + Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Selection"} + Opts.Input.[Mode Table] = {mode_tb} + // Opts.Input.[Mode Xfer Table] = {xfer_tb} + Opts.Global.[Skim Var] = skvars[j] + Opts.Global.[OD Layer Type] = 2 + if skmodes<> null then Opts.Global.[Skim Modes] = skmodes[j] + Opts.Flag.[Do Skimming] = 1 + Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 + Opts.Output.[Skim Matrix].Label = "Skim Matrix ("+modes[j]+periods[i]+")" + Opts.Output.[Skim Matrix].Compression = 0 //uncompressed + Opts.Output.[Skim Matrix].[File Name] = outputDir+"\\imp"+modes[j]+periods[i]+".mtx" + // Opts.Output.[TPS Table] = outputDir+"\\"+modes[j]+periods[i]+".tps" + ok = RunMacro("TCB Run Procedure", i, "Transit Skim PF", Opts) + + if !ok then goto quit + end + end + + ok=1 + quit: + RunMacro("close all") + Return(ok) +EndMacro +/*********************************************************************************** + + +Inputs: + input\mode5tod.dbf + output\transit.dbd + +***********************************************************************************/ +Macro "Special transit skims" (arr) + + shared path,inputDir, outputDir, mxtap + + skimvar=arr[1] + + mode_tb = inputDir+"\\mode5tod.dbf" + // xfer_tb = path+"\\modexfer.dbf" + db_file = outputDir + "\\transit.dbd" + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + db_node_lyr = db_file + "|" + node_lyr + + fpr=openfile(path+"\\hwycad.log","a") + mytime=GetDateAndTime() + writeline(fpr,mytime+", skim trnets") + + if skimvar="rail" then do //skim lrtzone and crzone to get the trolley and coaster fares + skvars={{"lrtzone"},{"Crzone"}} + trskimmtxs={"implr.mtx","impcr.mtx"} + trnets={"lr.tnw","cr.tnw"} + mtxdes={"Skim Matrix (light Rail)","Skim Matrix (coaster)"} + end + else if skimvar="fwylen" then do //skim length and fwylen + skvars={{"length","fwylen"},{"length","fwylen"}} + trskimmtxs={"imppropdst2.mtx","impprpkdst2.mtx"} + trnets={"prem_MD.tnw","prem_AM.tnw"} + mtxdes={"Length (Shortest OP Premium Path)","Length (Shortest PK Premium Path)"} + end + + for i = 1 to trnets.length do + Opts = null + Opts.Global.[Force Threads] = NumofCPU + Opts.Input.Database = db_file + Opts.Input.Network = path+"\\"+trnets[i] + Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Selection", "Select * where id <"+i2s(mxtap)} + Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Selection"} + Opts.Input.[Mode Table] = {mode_tb} + // Opts.Input.[Mode Xfer Table] = {xfer_tb} + Opts.Global.[Skim Var] = skvars[i] + Opts.Global.[OD Layer Type] = 2 + if skmodes<> null then Opts.Global.[Skim Modes] = skmodes[i] + Opts.Flag.[Do Skimming] = 1 + Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 + Opts.Output.[Skim Matrix].Label = mtxdes[i] + Opts.Output.[Skim Matrix].Compression = 0 //uncompressed + Opts.Output.[Skim Matrix].[File Name] = path+"\\"+trskimmtxs[i] + ok = RunMacro("TCB Run Procedure", i, "Transit Skim PF", Opts) + // ok = RunMacro("TCB Run Procedure", i, "Transit Skim Max Fare", Opts)//using maximized fare + + if !ok then goto quit + end + + ok=1 + quit: + RunMacro("close all") + Return(ok) +EndMacro + +/*************************************************************************************************************************** +This macro puts zeros in the null cells for unprocessed transit skims: premium and local modes +3/19/2015 Wu modified to zero out all ivts +****************************************************************************************************************************/ +Macro "zero null ivt time" + + shared path, outputDir + periods = {"_EA","_AM","_MD","_PM","_EV"} + + for i=1 to periods.length do + + //open matrix + fileNameSkimP = outputDir + "\\impprem"+periods[i]+".mtx" + mp = OpenMatrix(fileNameSkimP,) + currsp= CreateMatrixCurrencies(mp, , , ) + currsp.("*TM (Local)"):=Nz(currsp.("*TM (Local)")) + currsp.("*TM (Commuter Rail)"):=Nz(currsp.("*TM (Commuter Rail)")) + currsp.("*TM (Light Rail)"):=Nz(currsp.("*TM (Light Rail)")) + currsp.("*TM (Regional BRT (Yello)"):=Nz(currsp.("*TM (Regional BRT (Yello)")) + currsp.("*TM (Regional BRT (Red))"):=Nz(currsp.("*TM (Regional BRT (Red))")) + currsp.("*TM (Limited Express)"):=Nz(currsp.("*TM (Limited Express)")) + currsp.("*TM (Express)"):=Nz(currsp.("*TM (Express)")) + + fileNameSkimL = outputDir + "\\implocl"+periods[i]+".mtx" + ml = OpenMatrix(fileNameSkimL,) + currsl= CreateMatrixCurrencies(ml, , , ) + currsl.("In-Vehicle Time"):=Nz(currsl.("In-Vehicle Time")) + end + quit: + Return(1) +EndMacro + +/*********************************************************************************************************************************** + Extract Main Mode from Transit Skims + + This macro: +------------ + 1- extracts the main mode from transit skims + 2- writes main mode as a core to output transit matrix + + Inputs: +---------- + + output\impxxxx_yy.mtx + + where: + xxxx is mode (locl or prem) + yy is time period (EA, AM, MD, PM, NT) + + + Outputs: + + + output\impxxxx_yyo.mtx + + where: + xxxx is mode (locl or prem) + yy is time period (EA, AM, MD, PM, NT) + + + Input transit modes: +---------------------- + CR, LR, BRT Yellow, BRT Red, Limited EXP, EXP, LB + + Output transit modes: +----------------------- + 1-CR Mode choice code 4 + 2-LR Mode choice code 5 + 3-BRT (BRT Yellow+BRT Red) Mode choice code 6 + 4-EXP (EXP+Limited EXP) Mode choice code 7 + 5-LB Mode choice code 8 + + Input-output Matrix cores correspondence table: +------------------------------------------------- + 1) Premium input-output + Input Output + Fare(1) Fare(1) + Initial Wait Time(2) Initial Wait Time(2) + Transfer Wait Time(3) Transfer Wait Time(3) + Transfer Walk Time(4)+Access Walk Time(5)+Egress Walk Time(6) Walk Time(4) + Dwelling time(7) ------ + Number of Transfers(8) Number of Transfers(5) + Length:CR(9) Length:CR(6) + Length:LR(10) Length:LR(7) + Length:BRT Yellow(11)+BRT Red(12) Length:BRT(8) + Length:Limited EXP(13)+EXP(14) Length:EXP(9) + Length:LB(15) Length:LB(10) + IVT:CR(16) IVT:CR(11) + IVT:LR(17) IVT:LR(12) + IVT:BRT Yellow(18)+BRT Red(19) IVT:BRT(13) + IVT:Limited EXP(20)+EXP(21) IVT:EXP(14) + IVT:LB(22) IVT:LB(15) + --- IVT:Sum(16) + --- IVT:Main Mode(17) + + 2) Local input-output + Input Output + Fare(1) Fare(1) + In-Vehicle Time(2)+Dwelling Time(8) Total IV Time(2) + Initial Wait Time(3) Initial Wait Time(3) + Transfer Wait Time(4) Transfer Wait Time(4) + Transfer Walk Time(5)+Access Walk Time(6)+Egress Walk Time(7) Walk Time(5) + Number of Transfers(9) Number of Transfers(6) + + Author: Wu Sun + wsu@sandag.org, SANDAG + 05/25/09 ver2, + modified 4/16/2012 jef - for AB model + + +***********************************************************************************************************************************/ +Macro "Process transit skims" + shared path,inputDir, outputDir + + periods = {"_EA","_AM","_MD","_PM","_EV"} + modes = {"prem","locl"} + + //output core names, by mode + outMatrixCores={{"Fare","Initial Wait Time","Transfer Wait Time","Walk Time", "Number of Transfers", "Length:CR", + "Length:LR","Length:BRT","Length:EXP","Length:LB","IVT:CR","IVT:LR","IVT:BRT","IVT:EXP","IVT:LB","IVT:Sum","Main Mode"}, + {"Fare","Total IV Time","Initial Wait Time","Transfer Wait Time","Walk Time", "Number of Transfers"}} + + //input output matrix core lookup table, by mode + //Note: if index is set to -1, it represents an aggregation + inOutCoreLookUp={{1,2,3,-1,8,9,10,-1,-1,15,16,17,-1,-1,22}, + {1,-1,3,4,-1,9}} + + //skim aggretation lookup table, by mode + //Note: items match up with those in inOutCoreLookUp array where index is set to -1 + aggLookUp={ {{4,5,6},{11,12},{13,14},{18,19},{20,21}}, + {{2,8},{5,6,7}}} + + //dwelling time core index, by mode + // Note: set to -1 if no dwelling time allocation + dwlTimeIndex={7, -1} + + //dwelling time allocation line-haul modes (line-haul modes that ivt times need to be adjusted using dwelling time), by mode + // Note 1: rail modes are not included because they already include station dwell time. + // Note 2: Set to -1 if no dwelling time allocation + dwlAlloModes={{13,14,15}, -1} + + //ivt core start index, by mode + ivtStartIndex={11,2} + + //number of line-haul modes in output matrices, by mode + numOutModes={5,1} + + //calculate indices, including ivt sum index, main mode index, and ivt end idnex + dim ivtSumIndex[modes.length] + dim ivtEndIndex[modes.length] + for i=1 to modes.length do + ivtSumIndex[i]=outMatrixCores[i].length-1 + ivtEndIndex[i]=ivtStartIndex[i]+numOutModes[i]-1 + end + + //evaluation expression for coding main mode, by mode + //Note: if no main mode coding is necessary, leave null + expr={{"if(([IVT:LB]/ [IVT:Sum]) >0.5) then 8 else if [IVT:Sum]=null then 0 else if ([IVT:EXP]> [IVT:CR] & [IVT:EXP]> [IVT:LR] & [IVT:EXP]> [IVT:BRT] & [IVT:EXP]> [IVT:LB]) then 7 else if ([IVT:BRT]> [IVT:CR] & [IVT:BRT]> [IVT:LR] ) then 6", + "if ([Main Mode]=null & [IVT:LR]> [IVT:CR]) then 5 else if([Main Mode]=null) then 4 else [Main Mode]"},} + + + //-------------------------------------------------- + //This section aggregates matrices, allocates dwell time, and extracts main mode + //-------------------------------------------------- + + for i=1 to modes.length do + for j=1 to periods.length do + + //set input matrix currencies + inputFile = "imp"+modes[i]+periods[j]+".mtx" + inMatrixCurrency=RunMacro("set input matrix currencies",outputDir,inputFile) + + //set up output matrix currencies, empty at this point + outputFile = "imp"+modes[i]+periods[j]+"o.mtx" + matrixLabel = modes[i]+" "+periods[j] + outMatrixCurrency=RunMacro("set output matrix currencies",outputDir,inMatrixCurrency[1],outputFile,outMatrixCores[i],matrixLabel) + + //populate output matrix currencies except 'ivt sum' and 'main mode' cores + outMatrixCurrency=RunMacro("transit aggregate skims",inMatrixCurrency,outMatrixCurrency,inOutCoreLookUp[i],aggLookUp[i]) + + //populate 'main mode' core in output matrix + if expr[i]<>null then do + outMatrixCurrency=RunMacro("set main mode",inMatrixCurrency,outMatrixCurrency,dwlTimeIndex[i],ivtStartIndex[i],ivtEndIndex[i],ivtSumIndex[i],dwlAlloModes[i],expr[i]) + end + + end + end + + Return(1) +EndMacro + +/******************************************************************************************************************************** +set input matrix currencies + +Create and return an array of matrix currencies for the specified file in the specified directory + +*********************************************************************************************************************************/ +Macro "set input matrix currencies" (dir, trnInSkim) + //inputs, keyed to scenarioDirectory + inskim=dir+"\\"+trnInSkim + + //open input transit matrices + inMatrix = OpenMatrix(inskim, "True") + inMatrixCores = GetMatrixCoreNames(inMatrix) + numCoresIn=inMatrixCores.length + matrixInfo=GetMatrixInfo(inMatrix) + + //create inMatrix currencies + dim inMatrixCurrency[numCoresIn] + for i = 1 to numCoresIn do + inMatrixCurrency[i] = CreateMatrixCurrency(inMatrix, inMatrixCores[i], null, null, ) + end + + Return(inMatrixCurrency) +EndMacro + +/******************************************************************************************************************************** +set output matrix currencies + +Create a matrix file and return an array of matrix currencies for the file + +*********************************************************************************************************************************/ + +Macro "set output matrix currencies" (dir, inMatrixCurrency, trnOutSkim, outMatrixCores, label) + //outputs, keyed to scenarioDirectory + outskim=dir+"\\"+trnOutSkim + + //outMatrix core length + numCoresOut=outMatrixCores.length + + //outMatirx structure + dim outMatrixStructure[numCoresOut] + for i=1 to numCoresOut do + outMatrixStructure[i]=inMatrixCurrency + end + + //Create the output transit matrix (with main mode core) + Opts = null + Opts.[Compression] = 1 + Opts.[Tables] = outMatrixCores + Opts.[Type] = "Float" + Opts.[File Name] =outskim + Opts.[Label] = label + outMatrix = CopyMatrixStructure(outMatrixStructure,Opts) + + //create outMatrix currencies + dim outMatrixCurrency[numCoresOut] + for i=1 to numCoresOut do + outMatrixCurrency[i] = CreateMatrixCurrency(outMatrix, outMatrixCores[i], null, null, ) + outMatrixCurrency[i]:=0.0 + end + + //return outMatrixCurrency + return(outMatrixCurrency) +EndMacro + +/******************************************************************************************************************************** +transit aggregate skims + +Aggregate matrix currrencies in the input file and store the results in the output file + +*********************************************************************************************************************************/ + +Macro "transit aggregate skims"(inMatrixCurrency,outMatrixCurrency,inOutCoreLookUp,aggLookUp) + //populate all outMatrix cores except the last 2, using input-output core lookup table and aggregation lookup table + aggCounter=0 + for i = 1 to inOutCoreLookUp.length do + outMatrixCurrency[i]:=0.0 + lookupIndex=inOutCoreLookUp[i] + if lookupIndex=-1 then do + aggCounter=aggCounter+1 + aggIndices=aggLookUp[aggCounter] + for j=1 to aggIndices.length do + //name = inMatrixCurrency[aggIndices[j]].Core + //expr = "if ["+ name + "] = null then 0.0 else [" + name +"]" + //EvaluateMatrixExpression(inMatrixCurrency[aggIndices[j]], expr, , , ) + outMatrixCurrency[i]:=outMatrixCurrency[i]+Nz(inMatrixCurrency[aggIndices[j]]) + end + end + else do + outMatrixCurrency[i]:=Nz(inMatrixCurrency[lookupIndex]) + end + end + + return(outMatrixCurrency) +EndMacro + +/******************************************************************************************************************************** +set main mode + +Set the main mode in the file based upon the expression + +*********************************************************************************************************************************/ + +Macro "set main mode"(inMatrixCurrency,outMatrixCurrency,dwlTimeIndex,ivtStartIndex,ivtEndIndex,ivtSumIndex,dam,expr) + + //initialize outMatrixCurrency[ivtSumIndex], use as a temporary currency for storing sum values + outMatrixCurrency[ivtSumIndex]:=0.0 + + //sum ivt of dwelling allocation modes + for i=1 to dam.length do + outMatrixCurrency[ivtSumIndex]:=outMatrixCurrency[ivtSumIndex]+outMatrixCurrency[dam[i]] + end + + //allocate dwelling time + for i=1 to dam.length do + outMatrixCurrency[dam[i]]:=outMatrixCurrency[dam[i]]*(1.0+inMatrixCurrency[dwlTimeIndex]/outMatrixCurrency[ivtSumIndex]) + end + + //zero out outMatrixCurrency[ivtSumIndex] + outMatrixCurrency[ivtSumIndex]:=0.0 + + //set total ivt time to outMatrix + for i=ivtStartIndex to ivtEndIndex do + outMatrixCurrency[ivtSumIndex]:=outMatrixCurrency[ivtSumIndex]+Nz(outMatrixCurrency[i]) + end + + //set main mode to outMatrix using an expression + for i=1 to expr.length do + EvaluateMatrixExpression(outMatrixCurrency[outMatrixCurrency.length], expr[i],,, ) + end + + return(outMatrixCurrency) +EndMacro +/********************************************************************************* + +Update transit time fields + +This macro updates transit time fields with MSA times from flow tables + +The following fields are updated: + +xxField_yy where + +Field is TM + +xx is AB or BA +yy is period + EA: Early AM + AM: AM peak + MD: Midday + PM: PM peak + EV: Evening + + TODO: Update times for transit based on postload code pasted below + +*********************************************************************************/ +Macro "Update transit time fields" + + shared path, inputDir, outputDir + + periods = {"_EA","_AM","_MD","_PM","_EV"} + + // input files + db_file = outputDir + "\\transit.dbd" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) + SetLayer(link_lyr) + vw = GetView() + + /* + if(shoulder) then !adjust times for freeway shoulder operation + xxspd=xlen*60.0/xtime + if(xxspd.lt.35.0) xtime=xlen*(60.0/35.0) + endif + tranlt(ipk,idir)=xtime + brtlt(ipk,idir)=xtime + + adjust time for priority treatment + if(aatfc(1).gt.1.and.aatfc(1).lt.8.and.artxlkid(aatid).and. + * myear.eq.2030) then !adjust times for priority treatment + xtime=xtime*0.90 + + if(aatcnt(idir,1).eq.5) then + hovlt(ipk,idir)=hovlt(ipk,idir)*hovfac(ipk) !hovfac = 0.33 for peak, 1 for off-peak + tranlt(ipk,idir)=tranlt(ipk,idir)*hovfac(ipk) + brtlt(ipk,idir)=brtlt(ipk,idir)*hovfac(ipk) + + */ + + //Recompute generalized cost using MSA cost in flow table, for links with MSA cost (so that transit only links aren't overwritten with null) + for i = 1 to periods.length do + flowTable = outputDir+"\\hwyload"+periods[i]+".bin" + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"AB time"+periods[i] } + Opts.Global.Fields = {"ABTM"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if AB_MSA_Cost<>null then AB_MSA_Cost else ABTM"+periods[i] } + ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ok then goto quit + + // The Dataview Set is a joined view of the link layer and the flow table, based on link ID + Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"BA time"+periods[i]} + Opts.Global.Fields = {"BATM"+periods[i]} // the field to fill + Opts.Global.Method = "Formula" // the fill method + Opts.Global.Parameter = {"if BA_MSA_Cost<>null then BA_MSA_Cost else BATM"+periods[i] } + ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) + if !ok then goto quit + + end + + ok=1 + quit: + RunMacro("close all") + return(ok) + +EndMacro + +/*************************************************************************************** +Create rail net + +This script creates two rail networks; one for commuter rail and one for light-rail. The +networks are used to create light rail zonal fare matrix by skimming light rail stops for LR network. + +to create lrzone skim matrix replaced the value of lrzone by light rail fares + +Inputs: + input\modenu061.dbf Mode table + output\transit.dbd Transit line layer + output\transitrt.rts Transit route system + +Outputs: + output\cr.tnw Commuter rail transit network + output\lr.tnw Light rail transit network + +3/16/05 zou c +***************************************************************************************/ +Macro "Create rail net" + + shared path,mxtap, inputDir, outputDir + + db_file=outputDir+"\\transit.dbd" + rte_file=outputDir+"\\transitrt.rts" + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + db_link_lyr=db_file+"|"+link_lyr + db_node_lyr=db_file+"|"+node_lyr + rte_lyr = RunMacro("TCB Add RS Layers", rte_file, , ) + + stp_lyr = GetStopsLayerFromRS(rte_lyr) + db_rte_lyr = rte_file + "|" + rte_lyr + db_stp_lyr = rte_file + "|" + stp_lyr + + //route selection set names + sets = {"crpk","lrpk"} + + //selection query strings + //rcu - changed query to drop routes 399 and 599 12/27/06 + // todo: fix this + query_strs={"select * where am_headway >0 and route_name <> '399103' and route_name <> '399203' and route_name <> '599101' and route_name <> '599201'", + "select * where am_headway >0 and route_name <> '399103' and route_name <> '399203' and route_name <> '599101' and route_name <> '599201'"} + trnets = {"cr.tnw","lr.tnw"} + + // Build 4(2?) Transit Networks + for i = 1 to trnets.length do + Opts = null + Opts.Input.[Transit RS] = rte_file + Opts.Input.[RS Set] = {db_rte_lyr, rte_lyr,sets[i],query_strs[i]} + Opts.Input.[Walk Link Set] = {db_link_lyr, link_lyr, "walklink", "Select * where MINMODE<4"} + Opts.Input.[Stop Set] = {db_stp_lyr, stp_lyr} + Opts.Global.[Network Options].[Route Attributes].mode = {rte_lyr+".mode"} + Opts.Global.[Network Options].[Route Attributes].AM_HEADWAY = {rte_lyr+".AM_HEADWAY"} + Opts.Global.[Network Options].[Route Attributes].OP_HEADWAY = {rte_lyr+".OP_HEADWAY"} + Opts.Global.[Network Options].[Route Attributes].FARE = {rte_lyr+".FARE"} + Opts.Global.[Network Options].[Street Attributes].Length = {link_lyr+".Length",link_lyr+".Length"} + Opts.Global.[Network Options].[Street Attributes].[*TM_MD] = {link_lyr+".ABTM_MD", link_lyr+".BATM_MD"} + Opts.Global.[Network Options].[Street Attributes].[*TM_AM] = {link_lyr+".ABTM_AM", link_lyr+".BATM_AM"} + Opts.Global.[Network Options].[Street Attributes].LRTZONE = {link_lyr+".LRTZONE", link_lyr+".LRTZONE"} + Opts.Global.[Network Options].[Street Attributes].CRZONE = {link_lyr+".CRZONE", link_lyr+".CRZONE"} + Opts.Global.[Network Options].Walk = "Yes" + Opts.Global.[Network Options].Overide = {stp_lyr+".ID", stp_lyr+".nearnode"} + Opts.Global.[Network Options].[Link Attributes] = + {{"Length", {link_lyr+".Length",link_lyr+".Length"}, "SUMFRAC"}, + {"*TM_MD", {link_lyr+".ABTM_MD", link_lyr+".BATM_MD"}, "SUMFRAC"}, + {"*TM_AM", {link_lyr+".ABTM_AM", link_lyr+".BATM_AM"}, "SUMFRAC"}, + {"lrtzone",{link_lyr+".lrtzone", link_lyr+".lrtzone"}, "SUMFRAC"}, + {"crzone", {link_lyr+".crzone", link_lyr+".crzone"}, "SUMFRAC"}} + Opts.Global.[Network Options].[Mode Field] = rte_lyr+".Mode" + Opts.Global.[Network Options].[Walk Mode] = link_lyr+".minmode" + Opts.Output.[Network File] = outputDir+"\\"+trnets[i] + RunMacro("HwycadLog",{"createtrnnet.rsc: Create rail net","Build Transit Network: "+trnets[i]}) + ok = RunMacro("TCB Run Operation", i, "Build Transit Network", Opts) + + if !ok then goto quit + end + + mode_tb = inputDir+"\\modenu061.dbf" + mode_vw = RunMacro("TCB OpenTable",,, {mode_tb}) + + // xfer_tb = path+"\\xferfares.dbf" + // xferf_vw = RunMacro("TCB OpenTable",,, {xfer_tb}) + + //transit network settings + impds = {"*tma", "*tma"} + impwts = {mode_vw+".wt_ivtpk", mode_vw+".wt_ivtpk"} + iwtwts = {mode_vw+".wt_fwtpk", mode_vw+".wt_fwtpk"} + xwaitwts = {mode_vw+".wt_xwtpk", mode_vw+".wt_xwtpk"} + modeused={mode_vw+".crmode",mode_vw+".lrmode"} + + for i = 1 to trnets.length do + Opts = null + Opts.Input.[Transit RS] = rte_file + Opts.Input.[Transit Network] = outputDir+"\\"+trnets[i] + Opts.Input.[Mode Table] = {mode_tb} + // Opts.Input.[Mode Cost Table] = {xfer_tb} + //Opts.Input.[Fare Currency] = {inputDir+"\\fare.mtx", "coaster fare", , } + Opts.Input.[Centroid Set] = {db_node_lyr,node_lyr, "Selection", "Select * where ID<"+i2s(mxtap)} + Opts.Field.[Link Impedance] = "*TM" + Opts.Field.[Route Headway] = headways[i] + Opts.Field.[Route Fare] = "Fare" + Opts.Field.[Stop Zone ID] = "farezone" + Opts.Field.[Mode Fare Type] = mode_vw+".faretype" + Opts.Field.[Mode Fare Core] = mode_vw+".farefield" + Opts.Field.[Mode Fare Weight] = farewts[i] + Opts.Field.[Mode Xfer Time] = mode_vw+".xferpentm" + Opts.Field.[Mode Xfer Weight] = mode_vw+".wtxfertm" + Opts.Field.[Mode Impedance] = trntime[i] //impedance by transit mode + Opts.Field.[Mode Imp Weight] = impwts[i] + Opts.Field.[Mode IWait Weight] = iwtwts[i] + Opts.Field.[Mode XWait Weight] = xwaitwts[i] + Opts.Field.[Mode Dwell Weight] = impwts[i] + Opts.Field.[Mode Dwell On Time] = mode_vw+".dwelltime" + Opts.Field.[Mode Used] = modeused[j] + Opts.Field.[Mode Access] = mode_vw+".mode_acces" + Opts.Field.[Mode Egress] = mode_vw+".mode_egres" + Opts.Field.[Inter-Mode Xfer From] =xferf_vw+".from" + Opts.Field.[Inter-Mode Xfer To] = xferf_vw+".to" + Opts.Field.[Inter-Mode Xfer Stop] = xferf_vw+".stop" + Opts.Field.[Inter-Mode Xfer Proh] = xferf_vw+".prohibitio" + Opts.Field.[Inter-Mode Xfer Time] = xferf_vw+".xfer_penal" + Opts.Field.[Inter-Mode Xfer Fare] = xferf_vw+".fare" + Opts.Field.[Inter-Mode Xfer Wait] = xferf_vw+".wait_time" + Opts.Global.[Class Names] = {"Class 1"} + Opts.Global.[Class Description] = {"Class 1"} + Opts.Global.[current class] = "Class 1" + Opts.Global.[Global Fare Type] = "Flat" + Opts.Global.[Global Fare Value] = 2.25 + Opts.Global.[Global Xfer Fare] = 0 + Opts.Global.[Global Fare Core] = "coaster fare" + Opts.Global.[Global Fare Weight] = 1 + Opts.Global.[Global Imp Weight] = 1 + Opts.Global.[Global Init Weight] = 1 + Opts.Global.[Global Xfer Weight] = 1 + Opts.Global.[Global IWait Weight] = 2 + Opts.Global.[Global XWait Weight] = 2 + Opts.Global.[Global Dwell Weight] = 1 + Opts.Global.[Global Dwell On Time] = 0 + Opts.Global.[Global Dwell Off Time] = 0 + Opts.Global.[Global Headway] = 30 + Opts.Global.[Global Init Time] = 0 + Opts.Global.[Global Xfer Time] = 10 + Opts.Global.[Global Max IWait] = 60 + Opts.Global.[Global Min IWait] = 2 + Opts.Global.[Global Max XWait] = 60 + Opts.Global.[Global Min XWait] = 2 + Opts.Global.[Global Layover Time] = 5 + Opts.Global.[Global Max WACC Path] = 20 + Opts.Global.[Global Max Access] = 30 + Opts.Global.[Global Max Egress] = 30 + Opts.Global.[Global Max Transfer] = 20 + Opts.Global.[Global Max Imp] = 180 + Opts.Global.[Value of Time] = vot[i] + Opts.Global.[Max Xfer Number] = 3 + Opts.Global.[Max Trip Time] = 999 + Opts.Global.[Walk Weight] = 1.8 + Opts.Global.[Zonal Fare Method] = 1 + Opts.Global.[Interarrival Para] = 0.5 + Opts.Global.[Path Threshold] = 0 + Opts.Flag.[Use All Walk Path] = "No" + Opts.Flag.[Use Mode] = "Yes" + Opts.Flag.[Use Mode Cost] = "Yes" + Opts.Flag.[Combine By Mode] = "Yes" + Opts.Flag.[Fare By Mode] = "No" + Opts.Flag.[M2M Fare Method] = 2 + Opts.Flag.[Fare System] = 3 + Opts.Flag.[Use Park and Ride] = "No" + Opts.Flag.[Use Egress Park and Ride] = "No" + Opts.Flag.[Use P&R Walk Access] = "No" + Opts.Flag.[Use P&R Walk Egress] = "No" + Opts.Flag.[Use Parking Capacity] = "No" + RunMacro("HwycadLog",{"createtrnnet.rsc: create rail net","Transit Network Setting PF: "+trnets[i]}) + ok = RunMacro("TCB Run Operation", i, "Transit Network Setting PF", Opts) + if !ok then goto quit + end + + ok=1 + quit: +// if fpr<> null then closefile(fpr) + Return(ok) +EndMacro + + +/******************************************************************************* +Create transit networks + +create 6 transit network from the route system: +local bus op, local bus pk, premium op, premium pk + +Input files: + output\transit.dbd + output\transitrt.rts + input\timexfer.dbf + input\mode3tod.dbf - dbf file for mode table + input\modexfer.dbf + output\fare.mtx + + +output files: + localop.tnw - off peak local bus transit network + localpk.tnw - peak local bus transit network + prepop.tnw - off peak premium service transit network + preppk.tnw - peak premium service transit network + +*****************************************************************************/ +Macro "Create transit networks" + shared path, inputDir, outputDir, mxtap + + db_file=outputDir+"\\transit.dbd" + rte_file=outputDir+"\\transitrt.rts" + timexfer_tb = inputDir+"\\timexfer.bin" + mode_tb = inputDir+"\\mode5tod.dbf" + modexfer_tb = inputDir+"\\modexfer.dbf" + + periods = {"_EA","_AM","_MD","_PM","_EV"} + modes = {"prem","locl"} + + // timexfer_per = {"NO", "YES", "NO", "YES", "NO"} + timexfer_per = {"NO", "YES", "NO", "NO", "NO"} + timexfer_mod = {"YES", "NO"} + + + + ftype = RunMacro("G30 table type", timexfer_tb) + view = OpenTable("test", ftype, {timexfer_tb}) + if view = null then do + RunMacro("TCB Error", "Can't open table " + timexfer_tb) + Return(0) + end + + {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) + ok = (node_lyr <> null && link_lyr <> null) + if !ok then goto quit + + db_link_lyr=db_file+"|"+link_lyr + db_node_lyr=db_file+"|"+node_lyr + rte_lyr = RunMacro("TCB Add RS Layers", rte_file, , ) + stp_lyr = GetStopsLayerFromRS(rte_lyr) + db_rte_lyr = rte_file + "|" + rte_lyr + db_stp_lyr = rte_file + "|" + stp_lyr + + /* + fpr=OpenFile(path+"\\hwycad.log","a") + mytime=GetDateAndTime() + writeline(fpr,mytime+", create trnets") + */ + + //selection query strings: vary by period + query_strs={"select * where op_headway >0", + "select * where am_headway >0", + "select * where op_headway >0", + "select * where pm_headway >0", + "select * where op_headway >0"} + + // Build Transit Networks + for i = 1 to periods.length do + for j = 1 to modes.length do + Opts = null + Opts.Input.[Transit RS] = rte_file + Opts.Input.[RS Set] = {db_rte_lyr, rte_lyr,modes[j]+periods[i],query_strs[i]} + Opts.Input.[Walk Link Set] = {db_link_lyr, link_lyr, "walklink", "Select * where MINMODE<4"} + Opts.Input.[Stop Set] = {db_stp_lyr, stp_lyr} + Opts.Global.[Network Options].[Route Attributes].mode = {rte_lyr+".mode"} + Opts.Global.[Network Options].[Route Attributes].OP_HEADWAY = {rte_lyr+".OP_HEADWAY"} + Opts.Global.[Network Options].[Route Attributes].AM_HEADWAY = {rte_lyr+".AM_HEADWAY"} + Opts.Global.[Network Options].[Route Attributes].PM_HEADWAY = {rte_lyr+".PM_HEADWAY"} + Opts.Global.[Network Options].[Route Attributes].FARE = {rte_lyr+".FARE"} + Opts.Global.[Network Options].[Stop Attributes].farezone = {stp_lyr+".farezone"} + Opts.Global.[Network Options].[Street Attributes].Length = {link_lyr+".Length",link_lyr+".Length"} + Opts.Global.[Network Options].[Street Attributes].FWYLEN = {link_lyr+".FWYLEN", link_lyr+".FWYLEN"} + Opts.Global.[Network Options].[Street Attributes].[*TM] = {link_lyr+".ABTM"+periods[i], link_lyr+".BATM"+periods[i]} + // Opts.Global.[Network Options].[Street Attributes].[*PRETIME] = {link_lyr+".ABPRETIME"+periods[i], link_lyr+".BAPRETIME"+periods[i]} + // Opts.Global.[Network Options].[Street Attributes].[*LOCTIME] = {link_lyr+".ABLOCTIME"+periods[i], link_lyr+".BALOCTIME"+periods[i]} + Opts.Global.[Network Options].[Street Attributes].LRTZONE = {link_lyr+".LRTZONE", link_lyr+".LRTZONE"} + Opts.Global.[Network Options].[Street Attributes].CRZONE = {link_lyr+".CRZONE", link_lyr+".CRZONE"} + Opts.Global.[Network Options].Walk = "Yes" + Opts.Global.[Network Options].Overide = {stp_lyr+".ID", stp_lyr+".nearnode"} + Opts.Global.[Network Options].[Link Attributes] = + {{"Length", {link_lyr+".Length",link_lyr+".Length"}, "SUMFRAC"}, + {"fwylen",{link_lyr+".fwylen", link_lyr+".fwylen"}, "SUMFRAC"}, + {"*TM", {link_lyr+".ABTM"+periods[i], link_lyr+".BATM"+periods[i]}, "SUMFRAC"}, + // {"*PRETIME", {link_lyr+".ABPRETIME"+periods[i], link_lyr+".BAPRETIME"+periods[i]}, "SUMFRAC"}, + // {"*LOCTIME", {link_lyr+".ABLOCTIME"+periods[i], link_lyr+".BALOCTIME"+periods[i]}, "SUMFRAC"}, + {"lrtzone",{link_lyr+".lrtzone", link_lyr+".lrtzone"}, "SUMFRAC"}, + {"crzone", {link_lyr+".crzone", link_lyr+".crzone"}, "SUMFRAC"}} + Opts.Global.[Network Options].[Mode Field] = rte_lyr+".Mode" + Opts.Global.[Network Options].[Walk Mode] = link_lyr+".minmode" + Opts.Output.[Network File] = outputDir+"\\"+modes[j]+periods[i]+".tnw" + + ok = RunMacro("TCB Run Operation", i, "Build Transit Network", Opts) + if !ok then goto quit + end + end + + dif2=GetDirectoryInfo(timexfer_tb,"file") + if dif2.length>0 then blnxfer=1 else blnxfer=0 + + mode_vw = RunMacro("TCB OpenTable",,, {mode_tb}) + xferf_vw = RunMacro("TCB OpenTable",,, {modexfer_tb}) + + // following vary by period + headways = {"op_headway", "am_headway", "op_headway", "pm_headway", "op_headway"} + trntime= {mode_vw+".TRNTIME_EA",mode_vw+".TRNTIME_AM",mode_vw+".TRNTIME_MD",mode_vw+".TRNTIME_PM",mode_vw+".TRNTIME_EV"}//transit travel time by mode + impwts = {mode_vw+".wt_ivtop", mode_vw+".wt_ivtpk", mode_vw+".wt_ivtop", mode_vw+".wt_ivtpk", mode_vw+".wt_ivtop"} + iwtwts = {mode_vw+".wt_fwtop", mode_vw+".wt_fwtpk", mode_vw+".wt_fwtop", mode_vw+".wt_fwtpk", mode_vw+".wt_fwtop"} + xwaitwts = {mode_vw+".wt_xwtop", mode_vw+".wt_xwtpk", mode_vw+".wt_xwtop", mode_vw+".wt_xwtpk", mode_vw+".wt_xwtop"} + farewts= {mode_vw+".wt_fareop", mode_vw+".wt_farepk", mode_vw+".wt_fareop", mode_vw+".wt_farepk", mode_vw+".wt_fareop"} + vot = {0.05, 0.1, 0.05, 0.1, 0.05} //PB recommended 0.1 + wt_walk = { 1.6, 1.8, 1.6, 1.8, 1.6} + + // following varies by mode + modeused={mode_vw+".premode",mode_vw+".locmode"} + faresys={3, 1} //3, mixed fare, 1, flat fare + + + //transit network settings in TransCAD 6.0 R2 + for i = 1 to periods.length do + for j = 1 to modes.length do + Opts = null + Opts.Input.[Transit RS] = rte_file + Opts.Input.[Transit Network] = outputDir+"\\"+modes[j]+periods[i]+".tnw" + Opts.Input.[Mode Table] = {mode_tb} + Opts.Input.[Mode Cost Table] = {modexfer_tb} + Opts.Input.[Fare Currency] = {inputDir+"\\fare.mtx", "coaster fare", , } + // add timed transfers to premium peak networks (?) + if blnxfer=1 and timexfer_per[i] ="YES" and timexfer_mod[j]="YES" then Opts.Input.[Xfer Wait Table] = {timexfer_tb} + Opts.Input.[Centroid Set] = {db_node_lyr,node_lyr, "Selection", "Select * where ID<"+i2s(mxtap)} + Opts.Field.[Link Impedance] = "*TM" + Opts.Field.[Route Headway] = headways[i] + Opts.Field.[Route Fare] = "Fare" + Opts.Field.[Stop Zone ID] = "farezone" + Opts.Field.[Mode Fare Type] = mode_vw+".faretype" + Opts.Field.[Mode Fare Core] = mode_vw+".farefield" + Opts.Field.[Mode Fare Weight] = farewts[i] + Opts.Field.[Mode Xfer Time] = mode_vw+".xferpentm" + Opts.Field.[Mode Xfer Weight] = mode_vw+".wtxfertm" + Opts.Field.[Mode Impedance] = trntime[i] //impedance by transit mode + Opts.Field.[Mode Imp Weight] = impwts[i] + Opts.Field.[Mode IWait Weight] = iwtwts[i] + Opts.Field.[Mode XWait Weight] = xwaitwts[i] + Opts.Field.[Mode Dwell Weight] = impwts[i] + Opts.Field.[Mode Dwell On Time] = mode_vw+".dwelltime" + Opts.Field.[Mode Used] = modeused[j] + Opts.Field.[Mode Access] = mode_vw+".mode_acces" + Opts.Field.[Mode Egress] = mode_vw+".mode_egres" + Opts.Field.[Inter-Mode Xfer From] =xferf_vw+".from" + Opts.Field.[Inter-Mode Xfer To] = xferf_vw+".to" + Opts.Field.[Inter-Mode Xfer Stop] = xferf_vw+".stop" + Opts.Field.[Inter-Mode Xfer Proh] = xferf_vw+".prohibitio" + Opts.Field.[Inter-Mode Xfer Time] = xferf_vw+".xfer_penal" + Opts.Field.[Inter-Mode Xfer Fare] = xferf_vw+".fare" + Opts.Field.[Inter-Mode Xfer Wait] = xferf_vw+".wait_time" + Opts.Global.[Class Names] = {"Class 1"} + Opts.Global.[Class Description] = {"Class 1"} + Opts.Global.[current class] = "Class 1" + Opts.Global.[Global Fare Type] = "Flat" + Opts.Global.[Global Fare Value] = 2.25 + Opts.Global.[Global Xfer Fare] = 0 + Opts.Global.[Global Fare Core] = "coaster fare" + Opts.Global.[Global Fare Weight] = 1 + Opts.Global.[Global Imp Weight] = 1 + Opts.Global.[Global Init Weight] = 1 + Opts.Global.[Global Xfer Weight] = 1 + Opts.Global.[Global IWait Weight] = 2 + Opts.Global.[Global XWait Weight] = 2 + Opts.Global.[Global Dwell Weight] = 1 + Opts.Global.[Global Dwell On Time] = 0 + Opts.Global.[Global Dwell Off Time] = 0 + Opts.Global.[Global Headway] = 30 + Opts.Global.[Global Init Time] = 0 + Opts.Global.[Global Xfer Time] = 10 + Opts.Global.[Global Max IWait] = 60 + Opts.Global.[Global Min IWait] = 2 + Opts.Global.[Global Max XWait] = 60 + Opts.Global.[Global Min XWait] = 2 + Opts.Global.[Global Layover Time] = 5 + Opts.Global.[Global Max WACC Path] = 20 + Opts.Global.[Global Max Access] = 30 + Opts.Global.[Global Max Egress] = 30 + Opts.Global.[Global Max Transfer] = 20 + Opts.Global.[Global Max Imp] = 180 + Opts.Global.[Value of Time] = vot[i] + Opts.Global.[Max Xfer Number] = 3 + Opts.Global.[Max Trip Time] = 999 + Opts.Global.[Walk Weight] = 1.8 + Opts.Global.[Zonal Fare Method] = 1 + Opts.Global.[Interarrival Para] = 0.5 + Opts.Global.[Path Threshold] = 0 + Opts.Flag.[Use All Walk Path] = "No" + Opts.Flag.[Use Mode] = "Yes" + Opts.Flag.[Use Mode Cost] = "Yes" + Opts.Flag.[Combine By Mode] = "Yes" + Opts.Flag.[Fare By Mode] = "No" + Opts.Flag.[M2M Fare Method] = 2 + Opts.Flag.[Fare System] = 3 + Opts.Flag.[Use Park and Ride] = "No" + Opts.Flag.[Use Egress Park and Ride] = "No" + Opts.Flag.[Use P&R Walk Access] = "No" + Opts.Flag.[Use P&R Walk Egress] = "No" + Opts.Flag.[Use Parking Capacity] = "No" + ok = RunMacro("TCB Run Operation", i, "Transit Network Setting PF", Opts) + if !ok then goto quit + end + end + + ok=1 + quit: + if fpr <> null then closefile(fpr) + RunMacro("close all") + Return(ok) +EndMacro diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java new file mode 100644 index 0000000..8c13f8d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java @@ -0,0 +1,236 @@ +package org.sandag.abm.accessibilities; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Household; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; + +public class AccessibilitiesDMU + implements Serializable, VariableTable +{ + + + protected transient Logger logger = Logger.getLogger(AccessibilitiesDMU.class); + + protected HashMap methodIndexMap; + + private double[] workSizeTerms; + private double[] schoolSizeTerms; + private double[] sizeTerms; + // size + // terms + // purpose + // (as + // defined + // in + // uec), + // for + // a + // given + // mgra + + private double[] logsums; // logsums/accessibilities, + // for + // a + // given + // mgra-pair + + private Household hhObject; + + // the alternativeData tabledataset has the following fields + // sizeTermIndex: Used to index into the sizeTerms array + // logsumIndex: Used to index into the logsums array + private TableDataSet alternativeData; + private int logsumIndex, sizeIndex; + + private int autoSufficiency; + + public AccessibilitiesDMU() + { + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getAutoSufficiency", 0); + methodIndexMap.put("getSizeTerm", 1); + methodIndexMap.put("getLogsum", 2); + methodIndexMap.put("getNumPreschool", 3); + methodIndexMap.put("getNumGradeSchoolStudents", 4); + methodIndexMap.put("getNumHighSchoolStudents", 5); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getAutoSufficiency(); + case 1: + return getSizeTerm(arrayIndex); + case 2: + return getLogsum(arrayIndex); + case 3: + return getNumPreschool(); + case 4: + return getNumGradeSchoolStudents(); + case 5: + return getNumHighSchoolStudents(); + case 6: + return getWorkSizeTerm(arrayIndex); + case 7: + return getSchoolSizeTerm(arrayIndex); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public void setAlternativeData(TableDataSet alternativeData) + { + this.alternativeData = alternativeData; + logsumIndex = alternativeData.getColumnPosition("logsumIndex"); + sizeIndex = alternativeData.getColumnPosition("sizeTermIndex"); + + } + + public int getNumPreschool() + { + return hhObject.getNumPreschool(); + } + + public int getNumGradeSchoolStudents() + { + return hhObject.getNumGradeSchoolStudents(); + } + + public int getNumHighSchoolStudents() + { + return hhObject.getNumHighSchoolStudents(); + } + + public int getAutoSufficiency() + { + return autoSufficiency; + } + + public void setHouseholdObject(Household hh) + { + hhObject = hh; + } + + public void setAutoSufficiency(int autoSufficiency) + { + this.autoSufficiency = autoSufficiency; + } + + public void setSizeTerms(double[] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + public void setWorkSizeTerms(double[] sizeTerms) + { + workSizeTerms = sizeTerms; + } + + public void setSchoolSizeTerms(double[] sizeTerms) + { + schoolSizeTerms = sizeTerms; + } + + public void setLogsums(double[] logsums) + { + this.logsums = logsums; + } + + /** + * For the given alternative, look up the work size term and return it. + * + * @param alt + * @return + */ + public double getWorkSizeTerm(int alt) + { + + int index = (int) alternativeData.getValueAt(alt, sizeIndex); + + return workSizeTerms[index]; + } + + /** + * For the given alternative, look up the school size term and return it. + * + * @param alt + * @return + */ + public double getSchoolSizeTerm(int alt) + { + + int index = (int) alternativeData.getValueAt(alt, sizeIndex); + + return schoolSizeTerms[index]; + } + + /** + * For the given alternative, look up the size term and return it. + * + * @param alt + * @return + */ + public double getSizeTerm(int alt) + { + + int index = (int) alternativeData.getValueAt(alt, sizeIndex); + + return sizeTerms[index]; + } + + /** + * For the given alternative, look up the size term and return it. + * + * @param alt + * @return + */ + public double getLogsum(int alt) + { + + int index = (int) alternativeData.getValueAt(alt, logsumIndex); + + return logsums[index]; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java new file mode 100644 index 0000000..d44f2fe --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java @@ -0,0 +1,407 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import org.apache.log4j.Logger; +import com.pb.common.datafile.CSVFileWriter; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class holds the accessibility table that is built, or reads it from a + * previously written file. + * + * @author Jim Hicks + * @version May, 2011 + */ +public final class AccessibilitiesTable + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(AccessibilitiesTable.class); + + private static final int NONMANDATORY_AUTO_ACCESSIBILITY_FIELD_NUMBER = 1; + private static final int NONMANDATORY_TRANSIT_ACCESSIBILITY_FIELD_NUMBER = 2; + private static final int NONMANDATORY_NONMOTOR_ACCESSIBILITY_FIELD_NUMBER = 3; + private static final int NONMANDATORY_SOV_0_ACCESSIBILITY_FIELD_NUMBER = 4; + private static final int NONMANDATORY_SOV_1_ACCESSIBILITY_FIELD_NUMBER = 5; + private static final int NONMANDATORY_SOV_2_ACCESSIBILITY_FIELD_NUMBER = 6; + private static final int NONMANDATORY_HOV_0_ACCESSIBILITY_FIELD_NUMBER = 7; + private static final int NONMANDATORY_HOV_1_ACCESSIBILITY_FIELD_NUMBER = 8; + private static final int NONMANDATORY_HOV_2_ACCESSIBILITY_FIELD_NUMBER = 9; + private static final int SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 10; + private static final int SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 11; + private static final int SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 12; + private static final int MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 13; + private static final int MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 14; + private static final int MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 15; + private static final int EAT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 16; + private static final int EAT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 17; + private static final int EAT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 18; + private static final int VISIT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 19; + private static final int VISIT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 20; + private static final int VISIT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 21; + private static final int DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 22; + private static final int DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 23; + private static final int DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 24; + private static final int ESCORT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 25; + private static final int ESCORT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 26; + private static final int ESCORT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 27; + private static final int SHOP_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 28; + private static final int SHOP_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 29; + private static final int SHOP_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 30; + private static final int MAINT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 31; + private static final int MAINT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 32; + private static final int MAINT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 33; + private static final int EAT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 34; + private static final int EAT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 35; + private static final int EAT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 36; + private static final int VISIT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 37; + private static final int VISIT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 38; + private static final int VISIT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 39; + private static final int DISCR_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 40; + private static final int DISCR_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 41; + private static final int DISCR_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 42; + private static final int ATWORK_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 43; + private static final int ATWORK_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 44; + private static final int TOTAL_EMPLOYMENT_ACCESSIBILITY_INDEX = 45; + private static final int ATWORK_ACCESSIBILITY_NMOT_INDEX = 46; + private static final int ALLHH_ACCESSIBILITY_TRANSIT_INDEX = 47; + private static final int NONMANDATORY_MAAS_ACCESSIBILITY_FIELD_NUMBER = 48; + + // accessibilities by mgra, accessibility alternative + private float[][] accessibilities; + + /** + * array of previously computed accessibilities + * + * @param computedAccessibilities + * array of accessibilities + * + * use this constructor if the accessibilities were calculated as + * opposed to read from a file. + */ + public AccessibilitiesTable(float[][] computedAccessibilities) + { + accessibilities = computedAccessibilities; + } + + /** + * file name for store accessibilities + * + * @param accessibilitiesInputFileName + * path and filename of file to read + * + * use this constructor if the accessibilities are to be read + * from a file. + */ + public AccessibilitiesTable(String accessibilitiesInputFileName) + { + readAccessibilityTableFromFile(accessibilitiesInputFileName); + } + + private void readAccessibilityTableFromFile(String fileName) + { + + File accFile = new File(fileName); + + // read in the csv table + TableDataSet accTable; + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + accTable = reader.readFile(accFile); + + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading accessibility data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(); + } + + // create accessibilities array as a 1-based array + float[][] temp = accTable.getValues(); + accessibilities = new float[temp.length + 1][]; + for (int i = 0; i < temp.length; i++) + { + accessibilities[i + 1] = new float[temp[i].length]; + for (int j = 0; j < temp[i].length; j++) + { + accessibilities[i + 1][j] = temp[i][j]; + } + } + + } + + public void writeAccessibilityTableToFile(String accFileName) + { + + File accFile = new File(accFileName); + + // the accessibilities array is indexed by mgra values which might no be + // consecutive. + // create an arraylist of data table rows, with the last field being the + // mgra value, + // convert to a tabledataset, then write to a csv file. + + ArrayList dataColumnHeadings = new ArrayList(); + dataColumnHeadings.add("NONMAN_AUTO"); + dataColumnHeadings.add("NONMAN_TRANSIT"); + dataColumnHeadings.add("NONMAN_NONMOTOR"); + dataColumnHeadings.add("NONMAN_SOV_0"); + dataColumnHeadings.add("NONMAN_SOV_1"); + dataColumnHeadings.add("NONMAN_SOV_2"); + dataColumnHeadings.add("NONMAN_HOV_0"); + dataColumnHeadings.add("NONMAN_HOV_1"); + dataColumnHeadings.add("NONMAN_HOV_2"); + dataColumnHeadings.add("SHOP_HOV_0"); + dataColumnHeadings.add("SHOP_HOV_1"); + dataColumnHeadings.add("SHOP_HOV_2"); + dataColumnHeadings.add("MAINT_HOV_0"); + dataColumnHeadings.add("MAINT_HOV_1"); + dataColumnHeadings.add("MAINT_HOV_2"); + dataColumnHeadings.add("EAT_HOV_0"); + dataColumnHeadings.add("EAT_HOV_1"); + dataColumnHeadings.add("EAT_HOV_2"); + dataColumnHeadings.add("VISIT_HOV_0"); + dataColumnHeadings.add("VISIT_HOV_1"); + dataColumnHeadings.add("VISIT_HOV_2"); + dataColumnHeadings.add("DISCR_HOV_0"); + dataColumnHeadings.add("DISCR_HOV_1"); + dataColumnHeadings.add("DISCR_HOV_2"); + dataColumnHeadings.add("ESCORT_HOV_0"); + dataColumnHeadings.add("ESCORT_HOV_1"); + dataColumnHeadings.add("ESCORT_HOV_2"); + dataColumnHeadings.add("SHOP_SOV_0"); + dataColumnHeadings.add("SHOP_SOV_1"); + dataColumnHeadings.add("SHOP_SOV_2"); + dataColumnHeadings.add("MAINT_SOV_0"); + dataColumnHeadings.add("MAINT_SOV_1"); + dataColumnHeadings.add("MAINT_SOV_2"); + dataColumnHeadings.add("EAT_SOV_0"); + dataColumnHeadings.add("EAT_SOV_1"); + dataColumnHeadings.add("EAT_SOV_2"); + dataColumnHeadings.add("VISIT_SOV_0"); + dataColumnHeadings.add("VISIT_SOV_1"); + dataColumnHeadings.add("VISIT_SOV_2"); + dataColumnHeadings.add("DISCR_SOV_0"); + dataColumnHeadings.add("DISCR_SOV_1"); + dataColumnHeadings.add("DISCR_SOV_2"); + dataColumnHeadings.add("ATWORK_SOV_0"); + dataColumnHeadings.add("ATWORK_SOV_2"); + dataColumnHeadings.add("TOTAL_EMP"); + dataColumnHeadings.add("ATWORK_NM"); + dataColumnHeadings.add("ALL_HHS_TRANSIT"); + dataColumnHeadings.add("NONMAN_MAAS"); + dataColumnHeadings.add("MGRA"); + + // copy accessibilities array into a 0-based array + float[][] dataTableValues = new float[accessibilities.length - 1][]; + for (int r = 1; r < accessibilities.length; r++) + { + dataTableValues[r - 1] = new float[accessibilities[r].length]; + for (int c = 0; c < accessibilities[r].length; c++) + { + dataTableValues[r - 1][c] = accessibilities[r][c]; + } + } + + TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); + CSVFileWriter csv = new CSVFileWriter(); + try + { + csv.writeFile(accData, accFile); + } catch (IOException e) + { + logger.error("Error trying to write accessiblities data file " + accFileName); + throw new RuntimeException(e); + } + + } + + public void writeLandUseAccessibilityTableToFile(String luAccFileName, float[][] luAccessibility) + { + + File accFile = new File(luAccFileName); + + // the accessibilities array is indexed by mgra values which might no be + // consecutive. + // create an arraylist of data table rows, with the last field being the + // mgra value, + // convert to a tabledataset, then write to a csv file. + + ArrayList dataTableRows = new ArrayList(); + ArrayList dataColumnHeadings = new ArrayList(); + dataColumnHeadings.add("AM_WORK_1"); + dataColumnHeadings.add("AM_WORK_2"); + dataColumnHeadings.add("AM_WORK_3"); + dataColumnHeadings.add("AM_WORK_4"); + dataColumnHeadings.add("AM_WORK_5"); + dataColumnHeadings.add("AM_WORK_6"); + dataColumnHeadings.add("AM_SCHOOL_1"); + dataColumnHeadings.add("AM_SCHOOL_2"); + dataColumnHeadings.add("AM_SCHOOL_3"); + dataColumnHeadings.add("AM_SCHOOL_4"); + dataColumnHeadings.add("AM_SCHOOL_5"); + dataColumnHeadings.add("MD_NONMAN_LS0"); + dataColumnHeadings.add("MD_NONMAN_LS1"); + dataColumnHeadings.add("MD_NONMAN_LS2"); + dataColumnHeadings.add("LUZ"); + + for (int r = 0; r < luAccessibility.length; r++) + { + + if (luAccessibility[r] != null) + { + + float[] values = new float[luAccessibility[r].length]; + for (int c = 0; c < luAccessibility[r].length; c++) + values[c] = luAccessibility[r][c]; + + dataTableRows.add(values); + + } + + } + + float[][] dataTableValues = new float[dataTableRows.size()][]; + for (int r = 0; r < dataTableValues.length; r++) + dataTableValues[r] = dataTableRows.get(r); + + TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); + CSVFileWriter csv = new CSVFileWriter(); + try + { + csv.writeFile(accData, accFile); + } catch (IOException e) + { + logger.error("Error trying to write land use accessiblities data file " + luAccFileName); + throw new RuntimeException(e); + } + + } + + public void writeLandUseLogsumTablesToFile(String luLogsumFileName, double[][][][] luLogsums) + { + + File accFile = new File(luLogsumFileName); + + // the accessibilities array is indexed by mgra values which might no be + // consecutive. + // create an arraylist of data table rows, with the last field being the + // mgra value, + // convert to a tabledataset, then write to a csv file. + + ArrayList dataTableRows = new ArrayList(); + ArrayList dataColumnHeadings = new ArrayList(); + dataColumnHeadings.add("OrigLuz"); + dataColumnHeadings.add("DestLuz"); + dataColumnHeadings.add("AM_LS0"); + dataColumnHeadings.add("AM_LS1"); + dataColumnHeadings.add("AM_LS2"); + dataColumnHeadings.add("MD_LS0"); + dataColumnHeadings.add("MD_LS1"); + dataColumnHeadings.add("MD_LS2"); + + for (int l = 1; l <= BuildAccessibilities.MAX_LUZ; l++) + { + for (int m = 1; m <= BuildAccessibilities.MAX_LUZ; m++) + { + float[] values = new float[8]; + values[0] = l; + values[1] = m; + values[2] = (float) luLogsums[0][0][l][m]; + values[3] = (float) luLogsums[0][1][l][m]; + values[4] = (float) luLogsums[0][2][l][m]; + values[5] = (float) luLogsums[1][0][l][m]; + values[6] = (float) luLogsums[1][1][l][m]; + values[7] = (float) luLogsums[1][2][l][m]; + dataTableRows.add(values); + } + } + + float[][] dataTableValues = new float[dataTableRows.size()][]; + for (int r = 0; r < dataTableValues.length; r++) + dataTableValues[r] = dataTableRows.get(r); + + TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); + CSVFileWriter csv = new CSVFileWriter(); + try + { + csv.writeFile(accData, accFile); + } catch (IOException e) + { + logger.error("Error trying to write land use logsums data file " + luLogsumFileName); + throw new RuntimeException(e); + } + + } + + public float getAggregateAccessibility(String type, int homeMgra) + { + float returnValue = 0; + + if (type.equalsIgnoreCase("auto")) returnValue = accessibilities[homeMgra][NONMANDATORY_AUTO_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("transit")) returnValue = accessibilities[homeMgra][NONMANDATORY_TRANSIT_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("maas")) returnValue = accessibilities[homeMgra][NONMANDATORY_MAAS_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("nonmotor")) returnValue = accessibilities[homeMgra][NONMANDATORY_NONMOTOR_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("sov0")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_0_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("sov1")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_1_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("sov2")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_2_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("hov0")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_0_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("hov1")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_1_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("hov2")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_2_ACCESSIBILITY_FIELD_NUMBER - 1]; + else if (type.equalsIgnoreCase("shop0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shop1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shop2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maint0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maint1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maint2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("eatOut0")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("eatOut1")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("eatOut2")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("visit0")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("visit1")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("visit2")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discr0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discr1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discr2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("escort0")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("escort1")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("escort2")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("totEmp")) returnValue = accessibilities[homeMgra][TOTAL_EMPLOYMENT_ACCESSIBILITY_INDEX - 1]; + else if (type.equalsIgnoreCase("shopSov0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shopSov1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shopSov2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintSov0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintSov1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintSov2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrSov0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrSov1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrSov2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shopHov0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shopHov1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("shopHov2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintHov0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintHov1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("maintHov2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrHov0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrHov1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; + else if (type.equalsIgnoreCase("discrHov2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; + else + { + logger.error("argument type = " + + type + + " is not valid"); + throw new RuntimeException(); + } + + return returnValue; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java new file mode 100644 index 0000000..78e29ee --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java @@ -0,0 +1,1024 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * This class is used to return auto skim values and non-motorized skim values + * for MGRA pairs associated with estimation data file records. + * + * @author Jim Hicks + * @version March, 2010 + */ +public class AutoAndNonMotorizedSkimsCalculator + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(AutoAndNonMotorizedSkimsCalculator.class); + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; + + // set the indices used for the non-motorized names array and the return + // skims + // array + private static final int WALK_INDEX = 0; + private static final int BIKE_INDEX = 1; + + private static final double WALK_SPEED = 3.0; // mph + private static final double BIKE_SPEED = 12.0; // mph + + // declare an array of UEC objects, 1 for each time period + private UtilityExpressionCalculator[] autoSkimUECs; + private IndexValues iv; + + // A simple DMU with no variables + private VariableTable dmu = null; + private AutoSkimsDMU autoSkimsDMU = null; + + private MgraDataManager mgraManager; + + private static final String[] AUTO_SKIM_NAMES = { + "da_nt_time", + "da_nt_fftime", + "da_nt_dist", + "da_nt_toll", + "da_nt_tdist", + "da_nt_std", + "da_tr_time", + "da_tr_fftime", + "da_tr_dist", + "da_tr_toll", + "da_tr_tdist", + "da_tr_std", + "s2_time", + "s2_fftime", + "s2_dist", + "s2_hdist", + "s2_toll", + "s2_tdist", + "s2_std", + "s3_time", + "s3_fftime", + "s3_dist", + "s3_hdist", + "s3_toll", + "s3_tdist", + "s3_std"}; + private static final int NUM_AUTO_SKIMS = AUTO_SKIM_NAMES.length; + + private static final String[] AUTO_SKIM_DESCRIPTIONS = { + "SOV Non-transponder Time", //0 + "SOV Non-transponder Free-flow Time", //1 + "SOV Non-transponder Distance", //2 + "SOV Non-transponder Toll", //3 + "SOV Non-transponder Toll Distance", //4 + "SOV Non-transponder Std. Deviation", //5 + "SOV Transponder Time", //6 + "SOV Transponder Free-flow Time", //7 + "SOV Transponder Distance", //8 + "SOV Transponder Toll", //9 + "SOV Transponder Toll Distance", //10 + "SOV Transponder Std. Deviation", //11 + "Shared 2 Time", //12 + "Shared 2 Free-flow Time", //13 + "Shared 2 Distance", //14 + "Shared 2 HOV Distance", //15 + "Shared 2 Toll", //16 + "Shared 2 Toll Distance", //17 + "Shared 2 Std. Deviation", //18 + "Shared 3+ Time", //19 + "Shared 3+ Free-flow Time", //20 + "Shared 3+ Distance", //21 + "Shared 3+ HOV Distance", //22 + "Shared 3+ Toll", //23 + "Shared 3+ Toll Distance", //24 + "Shared 3+ Std. Deviation" //25 + }; + + private static final String[] NM_SKIM_NAMES = {"walkTime", "bikeTime"}; + private static final int NUM_NM_SKIMS = NM_SKIM_NAMES.length; + + private static final String[] NM_SKIM_DESCRIPTIONS = {"walk time", "bike time"}; + + private double[][][] storedFromTazDistanceSkims; + private double[][][] storedToTazDistanceSkims; + + /** + * Get distance from taz to all zones. + * + * @param taz + * @param period + * @return An array of distances to all other zones. + */ + public double[] getTazDistanceFromTaz(int taz, int period) + { + + return storedFromTazDistanceSkims[period][taz]; + } + + /** + * Get distance from taz to all zones. + * + * @param taz + * @param period + * @return An array of distances to all other zones. + */ + public double[] getTazDistanceToTaz(int taz, int period) + { + + return storedToTazDistanceSkims[period][taz]; + } + + public AutoAndNonMotorizedSkimsCalculator(HashMap rbMap) + { + + // Create the UECs + String uecPath = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = uecPath + + Util.getStringValueFromPropertyMap(rbMap, "skims.auto.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.data.page"); + int autoSkimEaPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.ea.page"); + int autoSkimAmPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.am.page"); + int autoSkimMdPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.md.page"); + int autoSkimPmPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.pm.page"); + int autoSkimEvPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.ev.page"); + + File uecFile = new File(uecFileName); + autoSkimsDMU = new AutoSkimsDMU(); + autoSkimUECs = new UtilityExpressionCalculator[NUM_PERIODS]; + autoSkimUECs[EA] = new UtilityExpressionCalculator(uecFile, autoSkimEaPage, dataPage, + rbMap, autoSkimsDMU); + autoSkimUECs[AM] = new UtilityExpressionCalculator(uecFile, autoSkimAmPage, dataPage, + rbMap, autoSkimsDMU); + autoSkimUECs[MD] = new UtilityExpressionCalculator(uecFile, autoSkimMdPage, dataPage, + rbMap, autoSkimsDMU); + autoSkimUECs[PM] = new UtilityExpressionCalculator(uecFile, autoSkimPmPage, dataPage, + rbMap, autoSkimsDMU); + autoSkimUECs[EV] = new UtilityExpressionCalculator(uecFile, autoSkimEvPage, dataPage, + rbMap, autoSkimsDMU); + iv = new IndexValues(); + + mgraManager = MgraDataManager.getInstance(); + + // distances = new double[mgraManager.getMaxMgra()+1]; + } + + public void setTazDistanceSkimArrays(double[][][] storedFromTazDistanceSkims, + double[][][] storedToTazDistanceSkims) + { + this.storedFromTazDistanceSkims = storedFromTazDistanceSkims; + this.storedToTazDistanceSkims = storedToTazDistanceSkims; + } + + /** + * Return the array of auto skims for the origin MGRA, destination MGRA, and + * departure time period. Used for appending skims data to estimation files, + * not part of main ABM. + * + * @param origMgra + * Origin MGRA + * @param workMgra + * Destination MGRA + * @param departPeriod + * Departure skim period index (currently 1-5) + * @param vot Value-of-time ($/hr) + * @return Array of 9 skim values for the MGRA pair and departure period + */ + public double[] getAutoSkims(int origMgra, int destMgra, int departPeriod, float vot, boolean debug, + Logger logger) + { + + String separator = ""; + String header = ""; + if (debug) + { + logger.info(""); + logger.info(""); + header = "get auto skims debug info for origMgra=" + origMgra + ", destMgra=" + + destMgra + ", period index=" + departPeriod + ", period label=" + + PERIODS[departPeriod]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + } + + // assign a helper UEC object to the array element for the desired + // departure + // time period + UtilityExpressionCalculator autoSkimUEC = autoSkimUECs[departPeriod]; + + // declare the array to hold the skim values, which will be returned + double[] autoSkims = null; + + if (origMgra > 0 && destMgra > 0) + { + + int oTaz = mgraManager.getTaz(origMgra); + int dTaz = mgraManager.getTaz(destMgra); + + iv.setOriginZone(oTaz); + iv.setDestZone(dTaz); + + autoSkimsDMU.setVOT(vot); + + // use the UEC to return skim values for the orign/destination TAZs + // associated with the MGRAs + autoSkims = autoSkimUEC.solve(iv, autoSkimsDMU, null); + if (debug) + autoSkimUEC.logAnswersArray(logger, String.format( + "autoSkimUEC: oMgra=%d, dMgra=%d, period=%d", origMgra, destMgra, + departPeriod)); + + } + + if (debug) + { + + logger.info(separator); + logger.info(header); + logger.info(separator); + + logger.info("auto skims array values"); + logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); + for (int i = 0; i < autoSkims.length; i++) + { + logger.info(String.format("%5d %40s %15.2f", i, AUTO_SKIM_DESCRIPTIONS[i], + autoSkims[i])); + } + + } + + return autoSkims; + + } + + //Wu modified for xborder trips; 9/27/2019 + public double[] getAutoSkimsByTAZ(int origTAZ, int destTAZ, int departPeriod, float vot, boolean debug, + Logger logger) + { + + String separator = ""; + String header = ""; + if (debug) + { + logger.info(""); + logger.info(""); + header = "get auto skims debug info for origTaz=" + origTAZ + ", destTaz=" + + destTAZ + ", period index=" + departPeriod + ", period label=" + + PERIODS[departPeriod]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + } + + // assign a helper UEC object to the array element for the desired + // departure + // time period + UtilityExpressionCalculator autoSkimUEC = autoSkimUECs[departPeriod]; + + // declare the array to hold the skim values, which will be returned + double[] autoSkims = null; + + if (origTAZ > 0 && destTAZ > 0) + { + + int oTaz = origTAZ; + int dTaz = destTAZ; + + iv.setOriginZone(oTaz); + iv.setDestZone(dTaz); + + autoSkimsDMU.setVOT(vot); + + // use the UEC to return skim values for the orign/destination TAZs + // associated with the MGRAs + autoSkims = autoSkimUEC.solve(iv, autoSkimsDMU, null); + if (debug) + autoSkimUEC.logAnswersArray(logger, String.format( + "autoSkimUEC: oTaz=%d, dTaz=%d, period=%d", origTAZ, destTAZ, + departPeriod)); + + } + + if (debug) + { + + logger.info(separator); + logger.info(header); + logger.info(separator); + + logger.info("auto skims array values"); + logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); + for (int i = 0; i < autoSkims.length; i++) + { + logger.info(String.format("%5d %40s %15.2f", i, AUTO_SKIM_DESCRIPTIONS[i], + autoSkims[i])); + } + + } + + return autoSkims; + + } + + /** + * Get the non-motorized skims. + * + * Get all the mgras within walking distance of the origin mgra. If the set + * of mgras is not null, and the destination mgra is in the set, get the + * walk and bike times from the mgraManager; + * + * If the destination mgra is not within walking distance of the origin + * MGRA, get the drive-alone non-toll off-peak distance skim value for the + * mgra pair and calculate the walk time and bike time. + * + * @param origMgra + * The origin mgra + * @param destMgra + * The destination mgra + * @return An array of distances + */ + public double[] getNonMotorizedSkims(int origMgra, int destMgra, int departPeriod, + boolean debug, Logger logger) + { + + String separator = ""; + String header = ""; + if (debug) + { + logger.info(""); + logger.info(""); + header = "get non-motorized skims debug info for origMgra=" + origMgra + ", destMgra=" + + destMgra; + for (int i = 0; i < header.length(); i++) + separator += "^"; + } + + double[] nmSkims = new double[NUM_NM_SKIMS]; + + // get the array of mgras within walking distance of the origin + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); + + // if one of the walk mgras is the destination, set the skim values and + // return + if (walkMgras != null) + { + + for (int wMgra : walkMgras) + { + + if (wMgra == destMgra) + { + nmSkims[WALK_INDEX] = mgraManager.getMgraToMgraWalkTime(origMgra, destMgra); + nmSkims[BIKE_INDEX] = mgraManager.getMgraToMgraBikeTime(origMgra, destMgra); + + if (debug) + { + + logger.info(separator); + logger.info(header); + logger.info(separator); + + logger.info("non-motorized skims array values"); + logger.info("determined from the mgraManager for an mgra pair within walking distance of each other."); + logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + logger.info(String.format("%5s %40s %15s", "-----", "----------", + "----------")); + for (int i = 0; i < nmSkims.length; i++) + { + logger.info(String.format("%5d %40s %15.2f", i, + NM_SKIM_DESCRIPTIONS[i], nmSkims[i])); + } + + } + + return nmSkims; + } + + } + + } + + // the destination was not within walk distance, so calculate walk and + // bike + // times from the TAZ-TAZ skim distance + int oTaz = mgraManager.getTaz(origMgra); + int dTaz = mgraManager.getTaz(destMgra); + + if (debug) + { + + logger.info(separator); + logger.info(header); + logger.info(separator); + + logger.info("non-motorized skims array values"); + logger.info("calculated for an mgra pair not within walking distance of each other."); + logger.info("origTaz = " + oTaz + ", destTaz = " + dTaz + ", period = " + departPeriod + + ", od distance = " + + (float) storedFromTazDistanceSkims[departPeriod][oTaz][dTaz]); + logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); + for (int i = 0; i < nmSkims.length; i++) + { + logger.info(String + .format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], nmSkims[i])); + } + + } + + nmSkims[WALK_INDEX] = (storedFromTazDistanceSkims[departPeriod][oTaz][dTaz] / WALK_SPEED) * 60.0; + nmSkims[BIKE_INDEX] = (storedFromTazDistanceSkims[departPeriod][oTaz][dTaz] / BIKE_SPEED) * 60.0; + + return nmSkims; + + } + + /* + * public double[] getNonMotorizedSkims(int origMgra, int destMgra, int + * departPeriod, boolean debug, Logger logger) { + * + * String separator = ""; String header = ""; if (debug) { logger.info(""); + * logger.info(""); header = + * "get non-motorized skims debug info for origMgra=" + origMgra + + * ", destMgra=" + destMgra; for (int i = 0; i < header.length(); i++) + * separator += "^"; } + * + * double[] nmSkims = new double[NUM_NM_SKIMS]; + * + * // get the array of mgras within walking distance of the origin int[] + * walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); + * + * // if one of the walk mgras is the destination, set the skim values and + * // return if (walkMgras != null) { + * + * for (int wMgra : walkMgras) { + * + * if (wMgra == destMgra) { nmSkims[WALK_INDEX] = + * mgraManager.getMgraToMgraWalkTime(origMgra, destMgra); + * nmSkims[BIKE_INDEX] = mgraManager.getMgraToMgraBikeTime(origMgra, + * destMgra); + * + * if (debug) { + * + * logger.info(separator); logger.info(header); logger.info(separator); + * + * logger.info("non-motorized skims array values"); logger .info( + * "determined from the mgraManager for an mgra pair within walking distance of each other." + * ); logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + * logger.info(String.format("%5s %40s %15s", "-----", "----------", + * "----------")); for (int i = 0; i < nmSkims.length; i++) { + * logger.info(String.format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], + * nmSkims[i])); } + * + * } + * + * return nmSkims; } + * + * } + * + * } + * + * // the destination was not within walk distance, so calculate walk and + * bike // times from the TAZ-TAZ skim distance int oTaz = + * mgraManager.getTaz(origMgra); int dTaz = mgraManager.getTaz(destMgra); + * + * iv.setOriginZone(oTaz); iv.setDestZone(dTaz); + * + * // get the DA NT OP distance value for the mgra pair double[] autoSkims = + * autoSkimUECs[OP].solve(iv, dmu, null); double distance = autoSkims[2]; + * + * nmSkims[WALK_INDEX] = (distance / WALK_SPEED) * 60.0; nmSkims[BIKE_INDEX] + * = (distance / BIKE_SPEED) * 60.0; + * + * if (debug) { + * + * logger.info(separator); logger.info(header); logger.info(separator); + * + * logger.info("non-motorized skims array values"); logger.info( + * "calculated for an mgra pair not within walking distance of each other." + * ); logger.info("origTaz = " + oTaz + ", destTaz = " + dTaz + + * ", od distance = " + (float) distance); + * logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); + * logger.info(String.format("%5s %40s %15s", "-----", "----------", + * "----------")); for (int i = 0; i < nmSkims.length; i++) { + * logger.info(String .format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], + * nmSkims[i])); } + * + * } + * + * return nmSkims; + * + * } + */ + /** + * Get all the mgras within walking distance of the origin mgra and set the + * distances to those mgras. + * + * Then loop through all mgras without a distance and get the drive-alone + * non-toll off-peak distance skim value for the taz pair associated with + * each mgra pair. + * + * @param origMgra + * The origin mgra + * @param An + * array in which to put the distances + * @param tourModeIsAuto + * is a boolean set to true if tour mode is not non-motorized, + * transit, or school bus. if auto tour mode, then no need to + * determine walk distance, and drive skims can be used directly. + * public void getDistancesFromMgra( int origMgra, double[] + * distances, boolean tourModeIsAuto ) { + * + * Arrays.fill(distances, 0); + * + * if ( ! tourModeIsAuto ){ + * + * // get the array of mgras within walking distance of the + * destination int[] walkMgras = + * mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); + * + * // set the distance values for the mgras walkable to the + * destination if (walkMgras != null) { + * + * // get distances, in feet, and convert to miles for (int wMgra + * : walkMgras) distances[wMgra] = + * mgraManager.getMgraToMgraWalkDistFrom(origMgra, wMgra) / + * 5280.0; + * + * } + * + * } + * + * + * int oTaz = mgraManager.getTaz(origMgra); + * iv.setOriginZone(oTaz); for (int wMgra=1; wMgra <= + * mgraManager.getMaxMgra(); wMgra++) { + * + * // skip mgras where distance has already been set if ( + * distances[wMgra] > 0 ) continue; + * + * // calculate distances from the TAZ-TAZ skim distance int dTaz + * = mgraManager.getTaz(wMgra); iv.setDestZone(dTaz); double[] + * autoSkims = autoSkimUECs[OP].solve(iv, dmu, null); + * + * distances[wMgra] = autoSkims[2]; } + * + * } + */ + + /** + * Get all the mgras within walking distance of the origin mgra and set the + * distances to those mgras. + * + * Then loop through all mgras without a distance and get the drive-alone + * non-toll off-peak distance skim value for the taz pair associated with + * each mgra pair. + * + * @param origMgra + * The origin mgra + * @param An + * array in which to put the distances + * @param tourModeIsAuto + * is a boolean set to true if tour mode is not non-motorized, + * transit, or school bus. if auto tour mode, then no need to + * determine walk distance, and drive skims can be used directly. + * + */ + + public void getDistancesFromMgra(int origMgra, double[] distances, boolean tourModeIsAuto) + { + + Arrays.fill(distances, 0); + + if (!tourModeIsAuto) + { + + // get the array of mgras within walking distance of the destination + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); + + // set the distance values for the mgras walkable to the destination + if (walkMgras != null) + { + + // get distances, in feet, and convert to miles + for (int wMgra : walkMgras) + distances[wMgra] = mgraManager.getMgraToMgraWalkDistFrom(origMgra, wMgra) / 5280.0; + + } + + } + + int oTaz = mgraManager.getTaz(origMgra); + + for (int wMgra = 1; wMgra <= mgraManager.getMaxMgra(); wMgra++) + { + + // skip mgras where distance has already been set + if (distances[wMgra] > 0) continue; + + int dTaz = mgraManager.getTaz(wMgra); + distances[wMgra] = storedFromTazDistanceSkims[MD][oTaz][dTaz]; + } + + } + + /** + * Get all the mgras within walking distance of the destination mgra and set + * the distances from those mgras. + * + * Then loop through all mgras without a distance and get the drive-alone + * non-toll off-peak distance skim value for the taz pair associated with + * each mgra pair. + * + * @param destMgra + * The destination mgra + * @param An + * array in which to put the distances + * @param tourModeIsAuto + * is a boolean set to true if tour mode is not non-motorized, + * transit, or school bus. if auto tour mode, then no need to + * determine walk distance, and drive skims can be used directly. + */ + + public void getDistancesToMgra(int destMgra, double[] distances, boolean tourModeIsAuto) + { + + Arrays.fill(distances, 0); + + if (!tourModeIsAuto) + { + + // get the array of mgras within walking distance of the destination + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); + + // set the distance values for the mgras walkable to the destination + if (walkMgras != null) + { + + // get distances, in feet, and convert to miles + // get distances from destMgra since this is the direction of + // distances read from the data file + for (int wMgra : walkMgras) + distances[wMgra] = mgraManager.getMgraToMgraWalkDistTo(wMgra, destMgra) / 5280.0; + + } + + } + + // if the TAZ distances have not been computed yet for this destination + // TAZ, compute them from the UEC. + int dTaz = mgraManager.getTaz(destMgra); + + for (int wMgra = 1; wMgra <= mgraManager.getMaxMgra(); wMgra++) + { + + // skip mgras where distance has already been set + if (distances[wMgra] > 0) continue; + + int oTaz = mgraManager.getTaz(wMgra); + distances[wMgra] = storedToTazDistanceSkims[MD][dTaz][oTaz]; + + } + + } + + /* + * public void getDistancesToMgra( int destMgra, double[] distances, boolean + * tourModeIsAuto ) { + * + * Arrays.fill(distances, 0); + * + * if ( ! tourModeIsAuto ){ + * + * // get the array of mgras within walking distance of the destination + * int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); + * + * // set the distance values for the mgras walkable to the destination if + * (walkMgras != null) { + * + * // get distances, in feet, and convert to miles // get distances from + * destMgra since this is the direction of distances read from the data file + * for (int wMgra : walkMgras) distances[wMgra] = + * mgraManager.getMgraToMgraWalkDistTo(destMgra, wMgra) / 5280.0; + * + * } + * + * } + * + * + * int dTaz = mgraManager.getTaz(destMgra); iv.setDestZone(dTaz); for (int + * wMgra=1; wMgra <= mgraManager.getMaxMgra(); wMgra++) { + * + * // skip mgras where distance has already been set if ( distances[wMgra] > + * 0 ) continue; + * + * // calculate distances from the TAZ-TAZ skim distance int oTaz = + * mgraManager.getTaz(wMgra); iv.setOriginZone(oTaz); double[] autoSkims = + * autoSkimUECs[OP].solve(iv, dmu, null); + * + * distances[wMgra] = autoSkims[2]; } + * + * } + */ + + /* + * public double[] getDistancesToMgra( int destMgra, boolean tourModeIsAuto + * ) { + * + * + * Arrays.fill(distances, 0); + * + * if ( ! tourModeIsAuto ){ + * + * // get the array of mgras within walking distance of the destination + * int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); + * + * // set the distance values for the mgras walkable to the destination if + * (walkMgras != null) { + * + * // get distances, in feet, and convert to miles // get distances from + * destMgra since this is the direction of distances read from the data file + * for (int wMgra : walkMgras) distances[wMgra] = + * mgraManager.getMgraToMgraWalkDistTo(destMgra, wMgra) / 5280.0; + * + * } + * + * } + * + * + * + * // if the TAZ distances have not been computed yet for this destination + * TAZ, compute them from the UEC. int dTaz = mgraManager.getTaz(destMgra); + * + * for (int wMgra=1; wMgra <= mgraManager.getMaxMgra(); wMgra++) { + * + * // skip mgras where distance has already been set if ( distances[wMgra] > + * 0 ) continue; + * + * int oTaz = mgraManager.getTaz(wMgra); distances[wMgra] = + * storedToTazDistanceSkims[OP][dTaz][oTaz]; + * + * } + * + * return Arrays.copyOf( distances, distances.length ); + * + * } + * + * + * /** Calculate utility expressions for auto skims for a given origin to + * get distances to all destination mgras, and return off-peak sov distance. + * + * @param oMgra The origin mgra + * + * @return An array of off-peak sov distances + */ + /* + * public void getOpSkimDistancesFromMgra( int oMgra, double[] distances ) { + * + * int oTaz = mgraManager.getTaz(oMgra); + * + * for (int i=1; i <= mgraManager.getMaxMgra(); i++) { + * + * // calculate distances from the TAZ-TAZ skim distance int dTaz = + * mgraManager.getTaz(i); distances[i] = + * storedFromTazDistanceSkims[OP][oTaz][dTaz]; + * + * } + * + * } + */ + + /* + * public double[] getAmPkSkimDistancesFromMgra( int oMgra ) { + * + * int oTaz = mgraManager.getTaz(oMgra); + * + * for (int i=1; i <= mgraManager.getMaxMgra(); i++) { + * + * // calculate distances from the TAZ-TAZ skim distance int dTaz = + * mgraManager.getTaz(i); distances[i] = + * storedFromTazDistanceSkims[AM][oTaz][dTaz]; + * + * } + * + * return Arrays.copyOf( distances, distances.length ); + * + * } + */ + + /* + * public double[] getOpSkimDistancesFromMgra(int oMgra) { + * + * double[] distances = new double[mgraManager.getMaxMgra() + 1]; + * + * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); + * + * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { + * + * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); + * + * // sov distance double[] autoResults = autoSkimUECs[OP].solve(iv, dmu, + * null); distances[i] = autoResults[2]; + * + * } + * + * return distances; } + */ + + /* + * public void getOpSkimDistancesFromMgra(int oMgra, double[] distances) { + * + * Arrays.fill( distances, 0 ); + * + * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); + * + * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { + * + * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); + * + * // sov distance double[] autoResults = autoSkimUECs[OP].solve(iv, dmu, + * null); distances[i] = autoResults[2]; + * + * } + * + * } + */ + + public void getOpSkimDistancesFromMgra(int oMgra, double[] distances) + { + + int oTaz = mgraManager.getTaz(oMgra); + + for (int i = 1; i <= mgraManager.getMaxMgra(); i++) + { + + // calculate distances from the TAZ-TAZ skim distance + int dTaz = mgraManager.getTaz(i); + distances[i] = storedFromTazDistanceSkims[MD][oTaz][dTaz]; + + } + + } + + /** + * Calculate utility expressions for auto skims for a given origin to get + * distances to all destination mgras, and return am peak sov distance. + * + * @param oMgra + * The origin mgra + * @return An array of am peak sov distances public double[] + * getAmPkSkimDistancesFromMgra(int oMgra) { + * + * double[] distances = new double[mgraManager.getMaxMgra() + 1]; + * + * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); + * + * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { + * + * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); + * + * // sov distance double[] autoResults = autoSkimUECs[AM].solve(iv, + * dmu, null); distances[i] = autoResults[2]; + * + * } + * + * return distances; } + */ + + public void getAmPkSkimDistancesFromMgra(int oMgra, double[] distances) + { + + int oTaz = mgraManager.getTaz(oMgra); + + for (int i = 1; i <= mgraManager.getMaxMgra(); i++) + { + + // calculate distances from the TAZ-TAZ skim distance + int dTaz = mgraManager.getTaz(i); + distances[i] = storedFromTazDistanceSkims[AM][oTaz][dTaz]; + + } + + } + + /* + * public void getAmPkSkimDistancesFromMgra(int oMgra, double[] distances) { + * + * Arrays.fill( distances, 0 ); + * + * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); + * + * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { + * + * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); + * + * // sov distance double[] autoResults = autoSkimUECs[AM].solve(iv, dmu, + * null); distances[i] = autoResults[2]; + * + * } + * + * } + */ + + /** + * log a report of the final skim values for the MGRA odt + * + * @param orig + * is the origin mgra for the segment + * @param dest + * is the destination mgra for the segment + * @param depart + * is the departure period for the segment + * @param bestTapPairs + * is an int[][] of TAP values with the first dimesion the ride + * mode and second dimension a 2 element array with best orig and + * dest TAP + * @param returnedSkims + * is a double[][] of skim values with the first dimesion the + * ride mode indices and second dimention the skim categories + */ + public void logReturnedSkims(int orig, int dest, int depart, double[] skims, String skimLabel, + Logger logger) + { + + String separator = ""; + String header = ""; + + logger.info(""); + logger.info(""); + header = skimLabel + " skim value tables for origMgra=" + orig + ", destMgra=" + dest + + ", departperiod=" + depart; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + logger.info(separator); + logger.info(header); + logger.info(""); + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + tableRecord = String.format("%-5d %12.5f ", i + 1, skims[i]); + logger.info(tableRecord); + } + + logger.info(""); + logger.info(separator); + } + + public int getNumSkimPeriods() + { + return NUM_PERIODS; + } + + public int getNumAutoSkims() + { + return NUM_AUTO_SKIMS; + } + + public String[] getAutoSkimNames() + { + return AUTO_SKIM_NAMES; + } + + public int getNumNmSkims() + { + return NUM_NM_SKIMS; + } + + public String[] getNmSkimNames() + { + return NM_SKIM_NAMES; + } + + public int getNmWalkTimeSkimIndex() + { + return WALK_INDEX; + } + + public int getNmBikeTimeSkimIndex() + { + return BIKE_INDEX; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java new file mode 100644 index 0000000..8467036 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java @@ -0,0 +1,82 @@ +package org.sandag.abm.accessibilities; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +public class AutoSkimsDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(AutoSkimsDMU.class); + + protected HashMap methodIndexMap; + + protected float vot; + + public AutoSkimsDMU() + { + setupMethodIndexMap(); + } + + public float getVOT() + { + return vot; + } + + public void setVOT(float vot) + { + this.vot = vot; + } + + + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getVOT", 0); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getVOT(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java new file mode 100644 index 0000000..9e0811d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java @@ -0,0 +1,206 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.Serializable; +import java.util.HashMap; + +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * This class is used to return auto skim values and non-motorized skim values + * for MGRA pairs associated with estimation data file records. + * + * @author Jim Hicks + * @version March, 2010 + */ +public class AutoTazSkimsCalculator + implements Serializable +{ + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + + // declare an array of UEC objects, 1 for each time period + private UtilityExpressionCalculator[] autoDistOD_UECs; + + // The simple auto skims UEC does not use any DMU variables + private VariableTable dmu = null; + + private TazDataManager tazManager; + + private double[][][] storedFromTazDistanceSkims; + private double[][][] storedToTazDistanceSkims; + private int maxTaz; + + public AutoTazSkimsCalculator(HashMap rbMap) + { + + // Create the UECs + String uecPath = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = uecPath + + Util.getStringValueFromPropertyMap(rbMap, "taz.distance.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "taz.distance.data.page"); + int autoSkimEaOdPage = Util + .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.ea.page"); + int autoSkimAmOdPage = Util + .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.am.page"); + int autoSkimMdOdPage = Util + .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.md.page"); + int autoSkimPmOdPage = Util + .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.pm.page"); + int autoSkimEvOdPage = Util + .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.ev.page"); + + File uecFile = new File(uecFileName); + autoDistOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; + autoDistOD_UECs[EA] = new UtilityExpressionCalculator(uecFile, autoSkimEaOdPage, dataPage, + rbMap, dmu); + autoDistOD_UECs[AM] = new UtilityExpressionCalculator(uecFile, autoSkimAmOdPage, dataPage, + rbMap, dmu); + autoDistOD_UECs[MD] = new UtilityExpressionCalculator(uecFile, autoSkimMdOdPage, dataPage, + rbMap, dmu); + autoDistOD_UECs[PM] = new UtilityExpressionCalculator(uecFile, autoSkimPmOdPage, dataPage, + rbMap, dmu); + autoDistOD_UECs[EV] = new UtilityExpressionCalculator(uecFile, autoSkimEvOdPage, dataPage, + rbMap, dmu); + + tazManager = TazDataManager.getInstance(); + maxTaz = tazManager.getMaxTaz(); + + storedFromTazDistanceSkims = new double[NUM_PERIODS + 1][maxTaz + 1][]; + storedToTazDistanceSkims = new double[NUM_PERIODS + 1][maxTaz + 1][]; + + } + + /** + * Get all the mgras within walking distance of the origin mgra and set the + * distances to those mgras. + * + * Then loop through all mgras without a distance and get the drive-alone + * non-toll off-peak distance skim value for the taz pair associated with + * each mgra pair. + * + * @param origMgra + * The origin mgra + * @param An + * array in which to put the distances + * @param tourModeIsAuto + * is a boolean set to true if tour mode is not non-motorized, + * transit, or school bus. if auto tour mode, then no need to + * determine walk distance, and drive skims can be used directly. + */ + public void computeTazDistanceArrays() + { + + IndexValues iv = new IndexValues(); + + for (int oTaz = 1; oTaz <= maxTaz; oTaz++) + { + + storedFromTazDistanceSkims[EA][oTaz] = new double[maxTaz + 1]; + storedToTazDistanceSkims[EA][oTaz] = new double[maxTaz + 1]; + storedFromTazDistanceSkims[AM][oTaz] = new double[maxTaz + 1]; + storedToTazDistanceSkims[AM][oTaz] = new double[maxTaz + 1]; + storedFromTazDistanceSkims[MD][oTaz] = new double[maxTaz + 1]; + storedToTazDistanceSkims[MD][oTaz] = new double[maxTaz + 1]; + storedFromTazDistanceSkims[PM][oTaz] = new double[maxTaz + 1]; + storedToTazDistanceSkims[PM][oTaz] = new double[maxTaz + 1]; + storedFromTazDistanceSkims[EV][oTaz] = new double[maxTaz + 1]; + storedToTazDistanceSkims[EV][oTaz] = new double[maxTaz + 1]; + + } + + for (int oTaz = 1; oTaz <= maxTaz; oTaz++) + { + + iv.setOriginZone(oTaz); + + double[] eaAutoDist = autoDistOD_UECs[EA].solve(iv, dmu, null); + double[] amAutoDist = autoDistOD_UECs[AM].solve(iv, dmu, null); + double[] mdAutoDist = autoDistOD_UECs[MD].solve(iv, dmu, null); + double[] pmAutoDist = autoDistOD_UECs[PM].solve(iv, dmu, null); + double[] evAutoDist = autoDistOD_UECs[EV].solve(iv, dmu, null); + + for (int d = 0; d < maxTaz; d++) + { + + storedFromTazDistanceSkims[EA][oTaz][d + 1] = eaAutoDist[d]; + storedFromTazDistanceSkims[AM][oTaz][d + 1] = amAutoDist[d]; + storedFromTazDistanceSkims[MD][oTaz][d + 1] = mdAutoDist[d]; + storedFromTazDistanceSkims[PM][oTaz][d + 1] = pmAutoDist[d]; + storedFromTazDistanceSkims[EV][oTaz][d + 1] = evAutoDist[d]; + + storedToTazDistanceSkims[EA][d + 1][oTaz] = eaAutoDist[d]; + storedToTazDistanceSkims[AM][d + 1][oTaz] = amAutoDist[d]; + storedToTazDistanceSkims[MD][d + 1][oTaz] = mdAutoDist[d]; + storedToTazDistanceSkims[PM][d + 1][oTaz] = pmAutoDist[d]; + storedToTazDistanceSkims[EV][d + 1][oTaz] = evAutoDist[d]; + + } + + // iv.setDestZone( oTaz ); + // + // amAutoDist = autoDistDO_UECs[AM].solve(iv, dmu, null); + // opAutoDist = autoDistDO_UECs[OP].solve(iv, dmu, null); + // + // for (int d=0; d < maxTaz; d++) + // { + // + // storedToTazDistanceSkims[AM][d+1][oTaz] = amAutoDist[d]; + // storedToTazDistanceSkims[OP][d+1][oTaz] = opAutoDist[d]; + // + // } + + } + + } + + public double getTazToTazDistance(int period, int oTaz, int dTaz){ + + return storedFromTazDistanceSkims[period][oTaz][dTaz]; + } + + public double[][][] getStoredFromTazToAllTazsDistanceSkims() + { + return storedFromTazDistanceSkims; + } + + public double[][][] getStoredToTazFromAllTazsDistanceSkims() + { + return storedToTazDistanceSkims; + } + + public void clearStoredTazsDistanceSkims() + { + + for (int i = 0; i < storedFromTazDistanceSkims.length; i++) + { + for (int j = 0; j < storedFromTazDistanceSkims[i].length; j++) + storedFromTazDistanceSkims[i][j] = null; + storedFromTazDistanceSkims[i] = null; + } + storedFromTazDistanceSkims = null; + + for (int i = 0; i < storedToTazDistanceSkims.length; i++) + { + for (int j = 0; j < storedToTazDistanceSkims[i].length; j++) + storedToTazDistanceSkims[i][j] = null; + storedToTazDistanceSkims[i] = null; + } + storedToTazDistanceSkims = null; + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java new file mode 100644 index 0000000..a31110a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java @@ -0,0 +1,1548 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.PrintWriter; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashMap; +import java.util.concurrent.ConcurrentHashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.util.Tracer; + +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.Modes.AccessMode; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; +import org.sandag.abm.reporting.OMXMatrixDao; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.matrix.Matrix; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.newmodel.Alternative; +import com.pb.common.newmodel.ConcreteAlternative; +import com.pb.common.newmodel.LogitModel; +/** + * WalkPathUEC calculates the best walk-transit utilities for a given MGRA pair. + * + * @author Joel Freedman + * @version 1.0, May 2009 + */ +public class BestTransitPathCalculator implements Serializable +{ + + private transient Logger logger = Logger.getLogger(BestTransitPathCalculator.class); + + //TODO: combine APP_TYPE_xxx constants into a enum structure + public static final int APP_TYPE_GENERIC = 0; + public static final int APP_TYPE_TOURMC = 1; + public static final int APP_TYPE_TRIPMC = 2; + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + + public static final float NA = -999; + public static final int WTW = 0; + public static final int WTD = 1; + public static final int DTW = 2; + public static final int[] ACC_EGR = {WTW,WTD,DTW}; + public static final int NUM_ACC_EGR = ACC_EGR.length; + public static final String[] ACC_EGR_STRING = {"WTW","WTD","DTW"}; + + // seek and trace + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + protected Tracer tracer; + + private TazDataManager tazManager; + private TapDataManager tapManager; + private MgraDataManager mgraManager; + + private int maxMgra; + private int maxTap; + private int maxTaz; + + // piece-wise utilities are being computed + private UtilityExpressionCalculator walkAccessUEC; + private UtilityExpressionCalculator walkEgressUEC; + private UtilityExpressionCalculator driveAccessUEC; + private UtilityExpressionCalculator driveEgressUEC; + private UtilityExpressionCalculator tapToTapUEC; + private UtilityExpressionCalculator driveAccDisutilityUEC; + private UtilityExpressionCalculator driveEgrDisutilityUEC; + + private static final String TAPS_SKIM = "taps.skim"; + private static final String TAPS_SKIM_DIST = "taps.skim.dist"; + + + // utility data cache for each transit path segment + private StoredUtilityData storedDataObject; //Encapsulates data shared by the BestTransitPathCalculator objects created for each hh choice model object + // note that access/egress utilities are independent of transit skim set + private float[][] storedWalkAccessUtils; // references StoredUtilityData.storedWalkAccessUtils + private float[][] storedDriveAccessUtils;// references StoredUtilityData.storedDriveAccessUtils + private float[][] storedWalkEgressUtils; // references StoredUtilityData.storedWalkEgressUtils + private float[][] storedDriveEgressUtils;// references StoredUtilityData.storedDriveEgressUtils + private HashMap>> storedDepartPeriodTapTapUtils; //references StoredUtilityData.storedDepartPeriodTapTapUtils + + private IndexValues index = new IndexValues(); + + // arrays storing information about the n (array length) best paths + private double[] bestUtilities; + private double[] bestAccessUtilities; + private double[] bestEgressUtilities; + private int[] bestPTap; + private int[] bestATap; + private int[] bestSet; //since two of the best paths can be in the same set, need to store set as well + + private int numSkimSets; + private int numTransitAlts; + private int[] maxLogsumUtilitiesBySkimSet; //maximum number of utilities for each skims set in logsum calcs + private int[] utilityCount; //counter for utilities + private double[] expUtilities; //exponentiated utility array for path choice + + private double nestingCoefficient; + private static double WORST_UTILITY = -500; + + /** + * Constructor. + * + * @param rbMap HashMap + * @param UECFileName The path/name of the UEC containing the walk-transit model. + * @param modelSheet The sheet (0-indexed) containing the model specification. + * @param dataSheet The sheet (0-indexed) containing the data specification. + */ + public BestTransitPathCalculator(HashMap rbMap) + { + + // read in resource bundle properties + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if ( trace ) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + + + String uecPath = Util.getStringValueFromPropertyMap(rbMap,CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = Paths.get(uecPath,rbMap.get("utility.bestTransitPath.uec.file")).toString(); + + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, + "utility.bestTransitPath.data.page"); + + int walkAccessPage = Util.getIntegerValueFromPropertyMap(rbMap, + "utility.bestTransitPath.walkAccess.page"); + int driveAccessPage = Util.getIntegerValueFromPropertyMap(rbMap, + "utility.bestTransitPath.driveAccess.page"); + int walkEgressPage = Util.getIntegerValueFromPropertyMap(rbMap, + "utility.bestTransitPath.walkEgress.page"); + int driveEgressPage = Util.getIntegerValueFromPropertyMap(rbMap, + "utility.bestTransitPath.driveEgress.page"); + int tapToTapPage = Util.getIntegerValueFromPropertyMap( rbMap, + "utility.bestTransitPath.tapToTap.page" ); + int driveAccDisutilityPage = Util.getIntegerValueFromPropertyMap( rbMap, + "utility.bestTransitPath.driveAccDisutility.page" ); + int driveEgrDisutilityPage = Util.getIntegerValueFromPropertyMap( rbMap, + "utility.bestTransitPath.driveEgrDisutility.page" ); + + + File uecFile = new File(uecFileName); + walkAccessUEC = createUEC(uecFile, walkAccessPage, dataPage, rbMap, new TransitWalkAccessDMU()); + driveAccessUEC = createUEC(uecFile, driveAccessPage, dataPage, rbMap, new TransitDriveAccessDMU()); + walkEgressUEC = createUEC(uecFile, walkEgressPage, dataPage, rbMap, new TransitWalkAccessDMU()); + driveEgressUEC = createUEC(uecFile, driveEgressPage, dataPage, rbMap, new TransitDriveAccessDMU()); + tapToTapUEC = createUEC(uecFile, tapToTapPage, dataPage, rbMap, new TransitWalkAccessDMU()); + driveAccDisutilityUEC = createUEC(uecFile, driveAccDisutilityPage, dataPage, rbMap, new TransitDriveAccessDMU()); + driveEgrDisutilityUEC = createUEC(uecFile, driveEgrDisutilityPage, dataPage, rbMap, new TransitDriveAccessDMU()); + + mgraManager = MgraDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + maxMgra = mgraManager.getMaxMgra(); + maxTap = mgraManager.getMaxTap(); + maxTaz = tazManager.getMaxTaz(); + + // these arrays are shared by the BestTransitPathCalculator objects created for each hh choice model object + storedDataObject = StoredUtilityData.getInstance( maxMgra, maxTap, maxTaz, ACC_EGR, ModelStructure.PERIODCODES); + storedWalkAccessUtils = storedDataObject.getStoredWalkAccessUtils(); + storedDriveAccessUtils = storedDataObject.getStoredDriveAccessUtils(); + storedWalkEgressUtils = storedDataObject.getStoredWalkEgressUtils(); + storedDriveEgressUtils = storedDataObject.getStoredDriveEgressUtils(); + storedDepartPeriodTapTapUtils = storedDataObject.getStoredDepartPeriodTapTapUtils(); + + //setup arrays + numSkimSets = Util.getIntegerValueFromPropertyMap( rbMap, "utility.bestTransitPath.skim.sets" ); + numTransitAlts = Util.getIntegerValueFromPropertyMap( rbMap, "utility.bestTransitPath.alts" ); + + bestUtilities = new double[numTransitAlts]; + bestPTap = new int[numTransitAlts]; + bestATap = new int[numTransitAlts]; + bestSet = new int[numTransitAlts]; + maxLogsumUtilitiesBySkimSet = Util.getIntegerArrayFromPropertyMap(rbMap, "utility.bestTransitPath.maxPathsPerSkimSetForLogsum"); + utilityCount = new int[numSkimSets]; + expUtilities = new double[numTransitAlts]; + + nestingCoefficient = new Double(Util.getStringValueFromPropertyMap(rbMap, "utility.bestTransitPath.nesting.coeff")).floatValue(); + + } + + + + /** + * This is the main method that finds the best N TAP-pairs. It + * cycles through walk TAPs at the origin end (associated with the origin MGRA) + * and alighting TAPs at the destination end (associated with the destination + * MGRA) and calculates a utility for every available alt for each TAP + * pair. It stores the N origin and destination TAP that had the best utility. + * + * @param pMgra The origin/production MGRA. + * @param aMgra The destination/attraction MGRA. + * + */ + public void findBestWalkTransitWalkTaps(TransitWalkAccessDMU walkDmu, int period, int pMgra, int aMgra, boolean debug, Logger myLogger) + { + + clearBestArrays(Double.NEGATIVE_INFINITY); + + int[] pMgraSet = mgraManager.getMgraWlkTapsDistArray()[pMgra][0]; + int[] aMgraSet = mgraManager.getMgraWlkTapsDistArray()[aMgra][0]; + + if (pMgraSet == null || aMgraSet == null) + { + return; + } + + int pTaz = mgraManager.getTaz(pMgra); + int aTaz = mgraManager.getTaz(aMgra); + + boolean writeCalculations = false; + if ((tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz))|| debug) + { + writeCalculations = true; + } + + //create transit path collection + ArrayList paths = new ArrayList(); + + for (int pTap : pMgraSet) + { + + // Calculate the pMgra to pTap walk access utility values + float accUtil; + if (storedWalkAccessUtils[pMgra][pTap] == StoredUtilityData.default_utility) { + accUtil = calcWalkAccessUtility(walkDmu, pMgra, pTap, writeCalculations, myLogger); + storedWalkAccessUtils[pMgra][pTap] = accUtil; + } else { + accUtil = storedWalkAccessUtils[pMgra][pTap]; + if(writeCalculations){ + myLogger.info("Stored walk access utility from Mgra "+pMgra+" to Tap "+pTap+" is "+accUtil); + } + } + + for (int aTap : aMgraSet) + { + + // Calculate the aTap to aMgra walk egress utility values + float egrUtil; + if (storedWalkEgressUtils[aTap][aMgra] == StoredUtilityData.default_utility) { + egrUtil = calcWalkEgressUtility(walkDmu, aTap, aMgra, writeCalculations, myLogger); + storedWalkEgressUtils[aTap][aMgra] = egrUtil; + } else { + egrUtil = storedWalkEgressUtils[aTap][aMgra]; + if(writeCalculations){ + myLogger.info("Stored walk egress utility from Tap "+aTap+" to Mgra "+aMgra+" is "+egrUtil); + } + } + + // Calculate the pTap to aTap utility values + float tapTapUtil[] = new float[numSkimSets]; + if(!storedDepartPeriodTapTapUtils.get(WTW).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { + + //loop across number of skim sets the pTap to aTap utility values + for (int set=0; set paths = new ArrayList(); + + float[][][] tapParkingInfo = tapManager.getTapParkingInfo(); + + int[] pTapArray = tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode); + for ( int pTap : pTapArray ) + { + // Calculate the pTaz to pTap drive access utility values + float accUtil; + float accDisutil; + if (storedDriveAccessUtils[pTaz][pTap] == StoredUtilityData.default_utility) { + accUtil = calcDriveAccessUtility(driveDmu, pMgra, pTaz, pTap, accMode, writeCalculations, myLogger); + storedDriveAccessUtils[pTaz][pTap] = accUtil; + } else { + accUtil = storedDriveAccessUtils[pTaz][pTap]; + if(writeCalculations) + myLogger.info("Stored drive access utility from TAZ "+pTaz+" to Tap "+pTap+" is "+accUtil); + } + + + int lotID = (int)tapParkingInfo[pTap][0][0]; // lot ID + float lotCapacity = tapParkingInfo[pTap][2][0]; // lot capacity + + if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) + || (accMode == AccessMode.KISS_N_RIDE)) + { + + //always calculate the access disutility since it changes based on od + accDisutil = calcDriveAccessRatioDisutility(driveDmu, pMgra, pTaz, pTap, odDistance, accMode, writeCalculations, myLogger); + + for (int aTap : mgraManager.getMgraWlkTapsDistArray()[aMgra][0]) + { + + // Calculate the aTap to aMgra walk egress utility values + float egrUtil; + if (storedWalkEgressUtils[aTap][aMgra] == StoredUtilityData.default_utility) { + egrUtil = calcWalkEgressUtility(walkDmu, aTap, aMgra, writeCalculations, myLogger); + storedWalkEgressUtils[aTap][aMgra] = egrUtil; + } else { + egrUtil = storedWalkEgressUtils[aTap][aMgra]; + if(writeCalculations){ + myLogger.info("Stored walk egress utility from Tap "+aTap+" to Mgra "+aMgra+" is "+egrUtil); + } + } + // Calculate the pTap to aTap utility values + float tapTapUtil[] = new float[numSkimSets]; + if(!storedDepartPeriodTapTapUtils.get(DTW).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { + + //loop across number of skim sets the pTap to aTap utility values + for (int set=0; set paths = new ArrayList(); + + for (int pTap : mgraManager.getMgraWlkTapsDistArray()[pMgra][0]) + { + // Calculate the pMgra to pTap walk access utility values + float accUtil; + if (storedWalkAccessUtils[pMgra][pTap] == StoredUtilityData.default_utility) { + accUtil = calcWalkAccessUtility(walkDmu, pMgra, pTap, writeCalculations, myLogger); + storedWalkAccessUtils[pMgra][pTap] = accUtil; + } else { + accUtil = storedWalkAccessUtils[pMgra][pTap]; + if(writeCalculations){ + myLogger.info("Stored walk access utility from Mgra "+pMgra+" to Tap "+pTap+" is "+accUtil); + } + } + for (int aTap : tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode)) + { + + int lotID = (int) tapManager.getTapParkingInfo()[aTap][0][0]; // lot + // ID + float lotCapacity = tapManager.getTapParkingInfo()[aTap][2][0]; // lot + // capacity + if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) + || (accMode == AccessMode.KISS_N_RIDE)) + { + + // Calculate the aTap to aMgra drive egress utility values + float egrUtil; + float egrDisutil; + if (storedDriveEgressUtils[aTap][aTaz] == StoredUtilityData.default_utility) { + egrUtil = calcDriveEgressUtility(driveDmu, aTap, aTaz, aMgra, accMode, writeCalculations, myLogger); + storedDriveEgressUtils[aTap][aTaz] = egrUtil; + } else { + egrUtil = storedDriveEgressUtils[aTap][aTaz]; + if(writeCalculations){ + myLogger.info("Stored drive egress utility from Tap "+aTap+" to TAZ "+aTaz+" is "+egrUtil); + } + } + + //always calculate the access disutility since it changes based on od + egrDisutil = calcDriveEgressRatioDisutility(driveDmu, aTap, aMgra, aTaz, odDistance, accMode, writeCalculations, myLogger); + + // Calculate the pTap to aTap utility values + float tapTapUtil[] = new float[numSkimSets]; + if(!storedDepartPeriodTapTapUtils.get(WTD).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { + + //loop across number of skim sets the pTap to aTap utility values + for (int set=0; set paths Collection of paths + */ + public void trimPaths(ArrayList paths) + { + + //sort paths by total utility in reverse order to get highest utility first + Collections.sort(paths, Collections.reverseOrder()); + + //get best N paths + int count = 0; + for(TransitPath path : paths) { + + if (path.getTotalUtility() > NA) { + + //get data + bestUtilities[count] = path.getTotalUtility(); + bestPTap[count] = path.pTap; + bestATap[count] = path.aTap; + bestSet[count] = path.set; + + count = count + 1; + if(count == numTransitAlts) { + break; + } + } + } + } + + public float calcPathUtility(TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accEgr, int period, int origMgra, int pTap, int aTap, int destMgra, int set, boolean myTrace, Logger myLogger, float odDistance) { + + float accUtil =NA; + float egrUtil =NA; + float tapTapUtil =NA; + float accDisutil =0f; + float egrDisutil =0f; + + + if(accEgr==WTW) { + accUtil = calcWalkAccessUtility(walkDmu, origMgra, pTap, myTrace, myLogger); + egrUtil = calcWalkEgressUtility(walkDmu, aTap, destMgra, myTrace, myLogger); + tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, WTW, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); + } else if(accEgr==WTD) { + int aTaz = mgraManager.getTaz(destMgra); + AccessMode accMode = AccessMode.PARK_N_RIDE; + accUtil = calcWalkAccessUtility(walkDmu, origMgra, pTap, myTrace, myLogger); + egrUtil = calcDriveEgressUtility(driveDmu, aTap, aTaz, destMgra, accMode, myTrace, myLogger); + egrDisutil = calcDriveEgressRatioDisutility(driveDmu, aTap, destMgra,aTaz, odDistance, accMode, myTrace, myLogger); + tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, WTD, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); + } else if(accEgr==DTW) { + int pTaz = mgraManager.getTaz(origMgra); + AccessMode accMode = AccessMode.PARK_N_RIDE; + accUtil = calcDriveAccessUtility(driveDmu, origMgra, pTaz, pTap, accMode, myTrace, myLogger); + accDisutil = calcDriveAccessRatioDisutility(driveDmu, origMgra, pTaz, pTap, odDistance, accMode, myTrace, myLogger); + egrUtil = calcWalkEgressUtility(walkDmu, aTap, destMgra, myTrace, myLogger); + tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, DTW, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); + } + return(accUtil + tapTapUtil + egrUtil + accDisutil + egrDisutil); + } + + /** + * Return the array of transit best tap pairs for the given access/egress mode, origin MGRA, + * destination MGRA, and departure time period. + * + * @param TransitWalkAccessDMU walkDmu + * @param TransitDriveAccessDMU driveDmu + * @param Modes.AccessMode accMode + * @param origMgra Origin MGRA + * @param workMgra Destination MGRA + * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 =OffPeak period + * @param debug boolean flag to indicate if debugging reports should be logged + * @param logger Logger to which debugging reports should be logged if debug is true + * @return double[][] Array of best tap pair values - rows are N-path, columns are orig tap, dest tap, skim set, utility + */ + public double[][] getBestTapPairs(TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accMode, int origMgra, int destMgra, int departPeriod, boolean debug, Logger myLogger, float odDistance) + { + + String separator = ""; + String header = ""; + if (debug) + { + myLogger.info(""); + myLogger.info(""); + header = ACC_EGR[accMode] + " best tap pairs debug info for origMgra=" + origMgra + + ", destMgra=" + destMgra + ", period index=" + departPeriod + + ", period label=" + ModelStructure.SKIM_PERIOD_STRINGS[departPeriod]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + myLogger.info(""); + myLogger.info(separator); + myLogger.info("Calculating " + header); + } + + double[][] bestTaps = null; + + if(accMode==WTW) { + findBestWalkTransitWalkTaps(walkDmu, departPeriod, origMgra, destMgra, debug, myLogger); + } else if(accMode==DTW) { + findBestDriveTransitWalkTaps(walkDmu, driveDmu, departPeriod, origMgra, destMgra, debug, myLogger, odDistance); + } else if(accMode==WTD) { + findBestWalkTransitDriveTaps(walkDmu, driveDmu, departPeriod, origMgra, destMgra, debug, myLogger, odDistance); + } + + // get and log the best tap-tap utilities by alt + double[] bestUtilities = getBestUtilities(); + bestTaps = new double[bestUtilities.length][]; + + for (int i = 0; i < bestUtilities.length; i++) + { + //only initialize tap data if valid; otherwise null array + if (bestUtilities[i] > NA) bestTaps[i] = getBestTaps(i); + } + + // log the best utilities and tap pairs for each alt + if (debug) + { + myLogger.info(""); + myLogger.info(separator); + myLogger.info(header); + myLogger.info("Final Best Utilities:"); + myLogger.info("Alt, Alt, Utility, bestITap, bestJTap, bestSet"); + for (int i = 0; i < bestUtilities.length; i++) + { + myLogger.info(i + "," + i + "," + bestUtilities[i] + "," + + (bestTaps[i] == null ? "NA" : bestTaps[i][0]) + "," + + (bestTaps[i] == null ? "NA" : bestTaps[i][1]) + "," + + (bestTaps[i] == null ? "NA" : bestTaps[i][2])); + } + + myLogger.info(separator); + } + return bestTaps; + } + + /** + * Calculate utilities for the best tap pairs using person specific attributes. + * + * @param double[][] bestTapPairs + * @param TransitWalkAccessDMU walkDmu + * @param TransitDriveAccessDMU driveDmu + * @param Modes.AccessMode accMode + * @param origMgra Origin MGRA + * @param workMgra Destination MGRA + * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 =OffPeak period + * @param debug boolean flag to indicate if debugging reports should be logged + * @param logger Logger to which debugging reports should be logged if debug is true + * @return double[][] Array of best tap pair values - rows are N-path, columns are orig tap, dest tap, skim set, utility + */ + public double[][] calcPersonSpecificUtilities(double[][] bestTapPairs, TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accMode, int origMgra, int destMgra, int departPeriod, boolean debug, Logger myLogger, float odDistance) + { + + String separator = ""; + String header = ""; + if (debug) + { + myLogger.info(""); + myLogger.info(""); + header = accMode + " best tap pairs person specific utility info for origMgra=" + origMgra + + ", destMgra=" + destMgra + ", period index=" + departPeriod + + ", period label=" + ModelStructure.SKIM_PERIOD_STRINGS[departPeriod]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + myLogger.info(""); + myLogger.info(separator); + myLogger.info("Calculating " + header); + } + + //re-calculate utilities + for (int i = 0; i < bestTapPairs.length; i++) { + if (bestTapPairs[i] != null) { + int pTap = (int)bestTapPairs[i][0]; + int aTap = (int)bestTapPairs[i][1]; + int set = (int)bestTapPairs[i][2]; + double utility = calcPathUtility(walkDmu, driveDmu, accMode, departPeriod, origMgra, pTap, aTap, destMgra, set, debug, myLogger, odDistance); + bestTapPairs[i][3] = utility; + } + } + + // log the best utilities and tap pairs for each alt + if (debug) + { + myLogger.info(""); + myLogger.info(separator); + myLogger.info(header); + myLogger.info("Final Person Specific Best Utilities:"); + myLogger.info("Alt, Alt, Utility, bestITap, bestJTap, bestSet"); + int availableModeCount = 0; + for (int i = 0; i < bestUtilities.length; i++) + { + if (bestTapPairs[i] != null) availableModeCount++; + + myLogger.info(i + "," + i + "," + + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][3]) + "," + + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][0]) + "," + + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][1]) + "," + + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][2])); + } + + myLogger.info(separator); + } + return bestTapPairs; + } + + /* + private LogitModel setupTripLogSum(double[][] bestTapPairs, boolean myTrace, Logger myLogger) { + + //must size logit model ahead of time + int alts = 0; + for (int i=0; i0){ + double cumProb=0; + //re-iterate through paths and calculate probability, choose alternative based on rnum + for(int i = 0; i0) + logsum = Math.log(sumExpUtility); + return logsum; + } + + /** + * Get the best path logsum, subject to constraints. The constraints + * are that the logsum only include a certain number of paths for each + * skim set, as defined in the property utility.bestTransitPath.maxPathsPerSkimSetForLogsum. + * This allows the logsum + * to reduce or eliminate path overlap should any exist in the path set, without having + * access to actual route data in the utility calculation. Transit trips + * are still subject to choice across all paths in the best utility set. + * + * @return The constrained transit logsum. + */ + public double getTransitBestPathLogsum(double[][] bestTapPairs, boolean myTrace, Logger myLogger){ + + double logsum = NA; + double sumExpUtility = getSumExpUtilities(bestTapPairs, myTrace, myLogger); + if(sumExpUtility>0.0) + logsum = Math.log(sumExpUtility); + + if(myTrace) + myLogger.info("Best Transit Path Logsum "+logsum); + return logsum; + } + + + /** + * Get the sum of exponentiated utilities, subject to constraints. The constraints + * are that the sum only include a certain number of paths for each + * skim set, as defined in the property utility.bestTransitPath.maxPathsPerSkimSetForLogsum. + * to reduce or eliminate path overlap should any exist in the path set, without having + * access to actual route data in the utility calculation. Transit trips + * are still subject to choice across all paths in the best utility set. + * + * @param bestTapPairs The tap pairs to calculate the sum exponentiated utility over + * @param myTrace Trace calculations + * @param myLogger The logger to write tracing to + * + * @return The constrained sum of exponentiated utilities. + */ + public double getSumExpUtilities(double[][] bestTapPairs, boolean myTrace, Logger myLogger){ + double sumExpUtility=0; + + if(myTrace){ + myLogger.info("Calculating sum of exponentiated utilities for transit best TAP pairs"); + myLogger.info("Best_Path Utility Skim_Set Included? ExpUtility Sum"); + } + + //utilityCount tracks how many utilities included in logsum calc by skimset + Arrays.fill(utilityCount,0); + for(int i = 0; i WORST_UTILITY){ + int skimSet = bestSet[i]; + + //only include the utility in the logsum if the count + //by skimset hasn't been met yet. + if(utilityCount[skimSet] rbMap, VariableTable dmu) + { + return new UtilityExpressionCalculator(uecSpreadsheet, modelSheet, dataSheet, rbMap, dmu); + } + + /** + * Clears the arrays. This method gets called for two different purposes. One is + * to compare alternatives based on utilities and the other based on + * exponentiated utilities. For this reason, the bestUtilities will be + * initialized by the value passed in as an argument set by the calling method. + * + * @param initialization value + */ + public void clearBestArrays(double initialValue) + { + Arrays.fill(bestUtilities, initialValue); + Arrays.fill(bestPTap, 0); + Arrays.fill(bestATap, 0); + Arrays.fill(bestSet, 0); + } + + /** + * Get the best ptap, atap, and skim set in an array. Only to be called after trimPaths() has been called. + * + * @param alt. + * @return element 0 = best ptap, element 1 = best atap, element 2 = set, element 3= utility + */ + public double[] getBestTaps(int alt) + { + + double[] bestTaps = new double[4]; + + bestTaps[0] = bestPTap[alt]; + bestTaps[1] = bestATap[alt]; + bestTaps[2] = bestSet[alt]; + bestTaps[3] = bestUtilities[alt]; + + return bestTaps; + } + + /** + * Get the best transit alt. Returns null if no transit alt has a valid utility. + * Call only after calling findBestWalkTransitWalkTaps(). + * + * @return The best transit alt (highest utility), or null if no alt have a valid utility. + */ + public int getBestTransitAlt() + { + + int best = -1; + double bestUtility = Double.NEGATIVE_INFINITY; + for (int i = 0; i < bestUtilities.length; ++i) + { + if (bestUtilities[i] > bestUtility) { + best = i; + bestUtility = bestUtilities[i]; + } + } + + int returnSet = best; + if (best > -1) { + returnSet = best; + } + return returnSet; + } + + + /** + * This method writes the utilities for all TAP-pairs for each ride mode. + * It cycles through walk TAPs at the origin end (associated with the origin + * MGRA) and alighting TAPs at the destination end (associated with the + * destination MGRA) and calculates a utility for every available ride mode + * for each TAP pair and writes the result to the outwriter. + * + * The results written will be as follows: + * label,WTW,period,pTap,aTap,mode,combinedUtilities[mode] + * + * @param period The time period (AM, PM, Off) + * @param pMgra + * The origin/production MGRA. + * @param aMgra + * The destination/attraction MGRA. + * @param myLogger A logger for logging problems + * @param outwriter A printwriter for writing results + * @param label A label for the record. + */ + public void writeAllWalkTransitWalkTaps(int period, int pMgra, int aMgra, Logger myLogger, PrintWriter outwriter, String label) + { + + //TODO: Fix this + + /* + + clearBestArrays(Double.NEGATIVE_INFINITY); + + int[] pMgraSet = mgraManager.getMgraWlkTapsDistArray()[pMgra][0]; + int[] aMgraSet = mgraManager.getMgraWlkTapsDistArray()[aMgra][0]; + + if (pMgraSet == null || aMgraSet == null) + { + return; + } + + int pPos = -1; + for (int pTap : pMgraSet) + { + // used to know where we are in time/dist arrays for taps + pPos++; + + // Set the pMgra to pTap walk access utility values, if they haven't + // already been computed. + setWalkAccessUtility(pMgra, pPos, pTap, false, myLogger); + + int aPos = -1; + for (int aTap : aMgraSet) + { + // used to know where we are in time/dist arrays for taps + aPos++; + + // set the pTap to aTap utility values, if they haven't already + // been computed. + setUtilitiesForTapPair(WTW, period, pTap, aTap, false, myLogger); + + // Set the aTap to aMgra walk egress utility values, if they + // haven't already been computed. + setWalkEgressUtility(aTap, aMgra, aPos, false, myLogger); + + // write the utilities for each ride mode + try + { + for (int i = 0; i < combinedUtilities.length; i++){ + combinedUtilities[i] = storedWalkAccessUtils[pMgra][pTap][i] + + storedTapToTapUtils[WTW][period][pTap][aTap][i] + + storedWalkEgressUtils[aTap][aMgra][i]; + + if(combinedUtilities[i]>-500){ + outwriter.print(label); + outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",WTW,period,pTap,aTap,i,combinedUtilities[i]); + } + } + } catch (Exception e) + { + logger.error("exception computing combinedUtilities for WTW"); + logger.error("aTap=" + aTap + "pTap=" + pTap + "aMgra=" + aMgra + "pMgra=" + + pMgra + "period=" + period, e); + throw new RuntimeException(); + } + + + + } + } + */ + } + + /** + * This method writes all TAP-pairs for each ride mode. It cycles + * through drive access TAPs at the origin end (associated with the origin + * MGRA) and alighting TAPs at the destination end (associated with the + * destination MGRA) and calculates a utility for every available ride mode + * for each TAP pair. + * The results written will be as follows: + * + * label,DTW,period,pTap,aTap,mode,combinedUtilities[mode] + * + * @param period The time period (AM, PM, Off) + * @param pMgra + * The origin/production MGRA. + * @param aMgra + * The destination/attraction MGRA. + * @param myLogger A logger for logging problems + * @param outwriter A printwriter for writing results + * @param label A label for the record. + * + */ + public void writeAllDriveTransitWalkTaps(int period, int pMgra, int aMgra, + Logger myLogger, PrintWriter outwriter, String label) + { + + // TODO: Fix this + + /* + + clearBestArrays(Double.NEGATIVE_INFINITY); + + Modes.AccessMode accMode = AccessMode.PARK_N_RIDE; + + int pTaz = mgraManager.getTaz(pMgra); + + if (tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode) == null + || mgraManager.getMgraWlkTapsDistArray()[aMgra][0] == null) + { + return; + } + + float[][][] tapParkingInfo = tapManager.getTapParkingInfo(); + + int pPos = -1; + int[] pTapArray = tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode); + for (int pTap : pTapArray) + { + pPos++; // used to know where we are in time/dist arrays for taps + + // Set the pTaz to pTap drive access utility values, if they haven't + // already been computed. + setDriveAccessUtility(pTaz, pPos, pTap, accMode, false, myLogger); + + int lotID = (int) tapParkingInfo[pTap][0][0]; // lot ID + float lotCapacity = tapParkingInfo[pTap][2][0]; // lot capacity + + if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) + || (accMode == AccessMode.KISS_N_RIDE)) + { + + int aPos = -1; + for (int aTap : mgraManager.getMgraWlkTapsDistArray()[aMgra][0]) + { + aPos++; + + // Set the aTap to aMgra walk egress utility values, if they + // haven't already been computed. + setWalkEgressUtility(aTap, aMgra, aPos, false, myLogger); + + + // set the pTap to aTap utility values, if they haven't + // already been computed. + setUtilitiesForTapPair(DTW, period, pTap, aTap, false, myLogger); + + // compare the utilities for this TAP pair to previously + // calculated utilities for each ride mode and store the TAP numbers if + // this TAP pair is the best. + try + { + for (int i = 0; i < combinedUtilities.length; i++){ + combinedUtilities[i] = storedDriveAccessUtils[pTaz][pTap][i] + + storedTapToTapUtils[DTW][period][pTap][aTap][i] + + storedWalkEgressUtils[aTap][aMgra][i]; + if(combinedUtilities[i]>-500){ + outwriter.print(label); + outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",DTW,period,pTap,aTap,i,combinedUtilities[i]); + } + } + } catch (Exception e) + { + logger.error("exception computing combinedUtilities for DTW"); + logger.error("aTap=" + aTap + ",pTap=" + pTap + ",aMgra=" + aMgra + + ",pMgra=" + pMgra + ",period=" + period, e); + throw new RuntimeException(); + } + + } + } + + } + */ + } + + /** + * This method finds the best TAP-pairs for each ride mode. It cycles + * through drive access TAPs at the origin end (associated with the origin + * MGRA) and alighting TAPs at the destination end (associated with the + * destination MGRA) and calculates a utility for every available ride mode + * for each TAP pair. + * The results written will be as follows: + * + * label,WTD,period,pTap,aTap,mode,combinedUtilities[mode] + * + * @param period The time period (AM, PM, Off) + * @param pMgra + * The origin/production MGRA. + * @param aMgra + * The destination/attraction MGRA. + * @param myLogger A logger for logging problems + * @param outwriter A printwriter for writing results + * @param label A label for the record. + + * + */ + public void writeAllWalkTransitDriveTaps(int period, int pMgra, int aMgra, + Logger myLogger, PrintWriter outwriter, String label) + { + + // TODO: Fix this + + /* + clearBestArrays(Double.NEGATIVE_INFINITY); + + Modes.AccessMode accMode = AccessMode.PARK_N_RIDE; + + int aTaz = mgraManager.getTaz(aMgra); + + if (mgraManager.getMgraWlkTapsDistArray()[pMgra][0] == null + || tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode) == null) + { + return; + } + + int pPos = -1; + for (int pTap : mgraManager.getMgraWlkTapsDistArray()[pMgra][0]) + { + pPos++; // used to know where we are in time/dist arrays for taps + + // Set the pMgra to pTap walk access utility values, if they haven't + // already been computed. + setWalkAccessUtility(pMgra, pPos, pTap, false, myLogger); + + int aPos = -1; + for (int aTap : tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode)) + { + aPos++; + + int lotID = (int) tapManager.getTapParkingInfo()[aTap][0][0]; // lot + // ID + float lotCapacity = tapManager.getTapParkingInfo()[aTap][2][0]; // lot + // capacity + if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) + || (accMode == AccessMode.KISS_N_RIDE)) + { + + // Set the pTaz to pTap drive access utility values, if they + // haven't already been computed. + setDriveEgressUtility(aTap, aTaz, aPos, accMode, false, myLogger); + + // set the pTap to aTap utility values, if they haven't + // already + // been computed. + setUtilitiesForTapPair(WTD, period, pTap, aTap, false, myLogger); + + // compare the utilities for this TAP pair to previously + // calculated utilities for each ride mode and store the TAP numbers if + // this TAP pair is the best. + try + { + for (int i = 0; i < combinedUtilities.length; i++){ + combinedUtilities[i] = storedWalkAccessUtils[pMgra][pTap][i] + + storedTapToTapUtils[WTD][period][pTap][aTap][i] + + storedDriveEgressUtils[aTap][aTaz][i]; + if(combinedUtilities[i]>-500){ + outwriter.print(label); + outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",WTD,period,pTap,aTap,i,combinedUtilities[i]); + } + } + } catch (Exception e) + { + logger.error("exception computing combinedUtilities for WTD"); + logger.error("aTap=" + aTap + ",pTap=" + pTap + ",aMgra=" + aMgra + + ",pMgra=" + pMgra + ",period=" + period, e); + throw new RuntimeException(); + } + + } + } + + } + */ + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java new file mode 100644 index 0000000..ae7c5da --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java @@ -0,0 +1,1534 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Set; +import java.util.TreeSet; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +/** + * This class builds accessibility components for all modes. + * + * @author Joel Freedman + * @version May, 2009 + */ +public final class BuildAccessibilities + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(BuildAccessibilities.class); + + /* + * OLD names private static final String[] WORK_OCCUP_SEGMENT_NAME_LIST = { + * "White Collar", "Services", "Health", "Retail and Food", "Blue Collar", + * "Military" }; private static final int[] WORK_OCCUP_SEGMENT_VALUE_LIST = + * { 71, 72,74, 75, 76, 77 }; + */ + private static final String[] WORK_OCCUP_SEGMENT_NAME_LIST = { + "Management Business Science and Arts", "Services", "Sales and Office", + "Natural Resources Construction and Maintenance", + "Production Transportation and Material Moving", "Military" }; + private static final int[] WORK_OCCUP_SEGMENT_VALUE_LIST = {50, 51, 53, + 54, 55, 56 }; + + // these segment group labels and indices are used for creating all the + // school location choice segments + public static final String[] SCHOOL_DC_SIZE_SEGMENT_NAME_LIST = {"preschool", + "k-8", "unified k-8", "9-12", "unified 9-12", "univ typical", "univ non-typical"}; + public static final int PRESCHOOL_SEGMENT_GROUP_INDEX = 0; + public static final int GRADE_SCHOOL_SEGMENT_GROUP_INDEX = 1; + public static final int UNIFIED_GRADE_SCHOOL_SEGMENT_GROUP_INDEX = 2; + public static final int HIGH_SCHOOL_SEGMENT_GROUP_INDEX = 3; + public static final int UNIFIED_HIGH_SCHOOL_SEGMENT_GROUP_INDEX = 4; + public static final int UNIV_TYPICAL_SEGMENT_GROUP_INDEX = 5; + public static final int UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX = 6; + + private static final int UNIFIED_DISTRICT_OFFSET = 1000000; + + // these indices define the alternative numbers and size term calculation + // indices + public static final int PRESCHOOL_ALT_INDEX = 0; + public static final int GRADE_SCHOOL_ALT_INDEX = 1; + public static final int HIGH_SCHOOL_ALT_INDEX = 2; + public static final int UNIV_TYPICAL_ALT_INDEX = 3; + public static final int UNIV_NONTYPICAL_ALT_INDEX = 4; + + // school segments: preschool, grade school, high school, university + // typical, university non-typical + private static final int[] SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX = {3, 4, 5, 6, 6}; + private static final int[] SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX = {5, 4, 3, 2, 2}; + private static final int[] SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX = {3, 3, 3, 2, 2}; + + // MAX_LUZ is the largest LUZ value, including external LUZs. + public static final int MAX_LUZ = 236; + + // using trip mode choice IVT coefficient value + public static final double TIME_COEFFICIENT = -0.032; + + public static final int[] EXTERNAL_LUZS = {230, 231, 232, + 233, 234, 235, 236 }; + + public static final int[] EXTERNAL_LUZ_CORDON_LUZS = {156, 156, 213, + 157, 122, 156, 156 }; + + public static final int[] MINUTES_TO_ADD_TO_CORDON_FOR_EXTERNAL_LUZ = {105, 480, + 1800, 1740, 155, 8160, 8760 }; + + public static final int[] CORDON_LUZS = {122, 156, 157, + 213 }; + + public static final int[][] EXTERNAL_LUZS_FOR_CORDON_LUZ = { {234}, + {230, 231, 235, 236}, {233}, {232} }; + + // in the LU logsums array, 1st dimension is averaging type, 2nd is pk or + // op, 3rd is auto sufficiency segment, 4th is orig LUZ, 5th is dest LUZ. + public static final int SIMPLE = 0; + public static final int LOGIT = 1; + public static final int PK = 0; + public static final int OP = 1; + public static final int LS0 = 0; + public static final int LS1 = 1; + public static final int LS2 = 2; + + private int[][] externalLuzsForCordonLuz; + private int[] cordonLuzForExternalLuz; + private int[] cordonLuzMinutesForExternalLuz; + + public static final int NUM_AVG_METHODS = 2; + public static final int NUM_PERIODS = 2; + public static final int NUM_SUFFICIENCY_SEGMENTS = 3; + + private static BuildAccessibilities objInstance = null; + + private int[] nonUniversitySegments; + private int[] universitySegments; + + private int univTypicalSegment; + private int univNonTypicalSegment; + + private int[] mgraGsDistrict; + private int[] mgraHsDistrict; + private HashMap gsDistrictIndexMap = new HashMap(); + private HashMap hsDistrictIndexMap = new HashMap(); + + private String[] schoolSegmentSizeNames; + private int[] schoolDcSoaUecSheets; + private int[] schoolDcUecSheets; + private int[] schoolStfUecSheets; + + // a set of school segment indices for which shadow pricing is not done - + // currently includes pre-school segment only. + private HashSet noShadowPriceSchoolSegmentIndices; + + private HashMap psSegmentIndexNameMap; + private HashMap psSegmentNameIndexMap; + private HashMap gsSegmentIndexNameMap; + private HashMap gsSegmentNameIndexMap; + private HashMap hsSegmentIndexNameMap; + private HashMap hsSegmentNameIndexMap; + private HashMap univTypSegmentIndexNameMap; + private HashMap univTypSegmentNameIndexMap; + private HashMap univNonTypSegmentIndexNameMap; + private HashMap univNonTypSegmentNameIndexMap; + + private HashMap schoolSegmentIndexNameMap; + private HashMap schoolSegmentNameIndexMap; + + private HashMap workSegmentIndexNameMap; + private HashMap workSegmentNameIndexMap; + + public static final int TOTAL_LOGSUM_FIELD_NUMBER = 13; + + private static int numThreads = 10; + private static final int DISTRIBUTED_PACKET_SIZE = 1000; + + private HashMap workerOccupValueSegmentIndexMap; + + private UtilityExpressionCalculator constantsUEC; + private UtilityExpressionCalculator sizeTermUEC; + private UtilityExpressionCalculator workerSizeTermUEC; + private UtilityExpressionCalculator schoolSizeTermUEC; + + private AccessibilitiesDMU aDmu; + + private IndexValues iv; + + private double[][][] sovExpUtilities; + private double[][][] hovExpUtilities; + private double[][][] nMotorExpUtilities; + + private MgraDataManager mgraManager; + + // purpose (defined in UEC) + private double[][] sizeTerms; // mgra, + + // indicates whether this mgra has a size term + private boolean[] hasSizeTerm; // mgra, + + // purpose (defined in UEC) + private double[][] workerSizeTerms; // mgra, + + // purpose (defined in UEC) + private double[][] schoolSizeTerms; // mgra, + + // Land Use size terms for school are not segmented by school district + private double[][] luSchoolSizeTerms; + + // auto sufficiency (0 autos, autos=adults), + // and mode (SOV,HOV,Walk-Transit,Non-Motorized) + private double[][] expConstants; + + // accessibilities by mgra, accessibility alternative + private AccessibilitiesTable accessibilitiesTableObject; + + // array for storing land use accessibility mode choice logsum values + private double[][][][][] landUseLogsums; + private int[][] landUseLogsumsCount; + private float[][] landUseAccessibilities; + + private boolean calculateLuAccessibilities; + + private static final int MARKET_SEGMENTS = 3; + + public static final int ESCORT_INDEX = 0; + public static final int SHOP_INDEX = 1; + public static final int OTH_MAINT_INDEX = 2; + public static final int EATOUT_INDEX = 3; + public static final int VISIT_INDEX = 4; + public static final int OTH_DISCR_INDEX = 5; + + private HashMap nonMandatorySizeSegmentNameIndexMap; + + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + private Tracer tracer; + private boolean seek; + + private int maxMgra; + + private boolean accessibilitiesBuilt = false; + private boolean logResults=false; + + private BuildAccessibilities() + { + } + + public static synchronized BuildAccessibilities getInstance() + { + if (objInstance == null) + { + objInstance = new BuildAccessibilities(); + objInstance.accessibilitiesBuilt = false; + return objInstance; + } else + { + objInstance.accessibilitiesBuilt = true; + return objInstance; + } + } + + public void setupBuildAccessibilities(HashMap rbMap, + boolean calculateLuAccessibilities) + { + + this.calculateLuAccessibilities = calculateLuAccessibilities; + + logResults = Util.getStringValueFromPropertyMap(rbMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + + Runtime runtime = Runtime.getRuntime(); + if (numThreads < 0) + { + int nrOfProcessors = runtime.availableProcessors(); + numThreads = nrOfProcessors; + } + + gsDistrictIndexMap = new HashMap(); + hsDistrictIndexMap = new HashMap(); + workerOccupValueSegmentIndexMap = new HashMap(); + + for (int i = 0; i < WORK_OCCUP_SEGMENT_VALUE_LIST.length; ++i) + { + workerOccupValueSegmentIndexMap.put(WORK_OCCUP_SEGMENT_VALUE_LIST[i], i); + } + aDmu = new AccessibilitiesDMU(); + + // Create the UECs + String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.data.page"); + int constantsPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.constants.page"); + int sizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sizeTerm.page"); + int workerSizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, + "acc.workerSizeTerm.page"); + int schoolSizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, + "acc.schoolSizeTerm.page"); + + File uecFile = new File(uecFileName); + constantsUEC = new UtilityExpressionCalculator(uecFile, constantsPage, dataPage, rbMap, + aDmu); + sizeTermUEC = new UtilityExpressionCalculator(uecFile, sizeTermPage, dataPage, rbMap, aDmu); + workerSizeTermUEC = new UtilityExpressionCalculator(uecFile, workerSizeTermPage, dataPage, + rbMap, aDmu); + schoolSizeTermUEC = new UtilityExpressionCalculator(uecFile, schoolSizeTermPage, dataPage, + rbMap, aDmu); + + mgraManager = MgraDataManager.getInstance(rbMap); + maxMgra = mgraManager.getMaxMgra(); + + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + iv = new IndexValues(); + + workSegmentIndexNameMap = new HashMap(); + workSegmentNameIndexMap = new HashMap(); + + noShadowPriceSchoolSegmentIndices = new HashSet(); + + schoolSegmentIndexNameMap = new HashMap(); + schoolSegmentNameIndexMap = new HashMap(); + + psSegmentIndexNameMap = new HashMap(); + psSegmentNameIndexMap = new HashMap(); + gsSegmentIndexNameMap = new HashMap(); + gsSegmentNameIndexMap = new HashMap(); + hsSegmentIndexNameMap = new HashMap(); + hsSegmentNameIndexMap = new HashMap(); + univTypSegmentIndexNameMap = new HashMap(); + univTypSegmentNameIndexMap = new HashMap(); + univNonTypSegmentIndexNameMap = new HashMap(); + univNonTypSegmentNameIndexMap = new HashMap(); + + nonMandatorySizeSegmentNameIndexMap = new HashMap(); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, + ESCORT_INDEX); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, + SHOP_INDEX); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, + OTH_MAINT_INDEX); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, + EATOUT_INDEX); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, + VISIT_INDEX); + nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME, + OTH_DISCR_INDEX); + + } + + + // This method is only called if the main ABM command line argument for + // calculating land use accessibilities is true + // otherwise, the array lanUseLogsums is null and serves as an indicator + // that LU accessibilities are not needed + public void setCalculatedLandUseAccessibilities() + { + + landUseLogsums = new double[NUM_AVG_METHODS][NUM_PERIODS][NUM_SUFFICIENCY_SEGMENTS][MAX_LUZ + 1][MAX_LUZ + 1]; + landUseLogsumsCount = new int[MAX_LUZ + 1][MAX_LUZ + 1]; + + int maxMgra = mgraManager.getMaxMgra(); + landUseAccessibilities = new float[maxMgra + 1][]; + + externalLuzsForCordonLuz = new int[MAX_LUZ + 1][]; + cordonLuzForExternalLuz = new int[MAX_LUZ + 1]; + cordonLuzMinutesForExternalLuz = new int[MAX_LUZ + 1]; + + // associate the array of external LUZs that belong to each cordon LUZ. + for (int i = 0; i < CORDON_LUZS.length; i++) + { + int cordonLuz = CORDON_LUZS[i]; + externalLuzsForCordonLuz[cordonLuz] = EXTERNAL_LUZS_FOR_CORDON_LUZ[i]; + } + + // associate the cordon LUZ that belongs to each external LUZ. + for (int i = 0; i < EXTERNAL_LUZS.length; i++) + { + int externalLuz = EXTERNAL_LUZS[i]; + cordonLuzForExternalLuz[externalLuz] = EXTERNAL_LUZ_CORDON_LUZS[i]; + } + + // associate the minutes to add to cordon LUZ to represent time to each + // external LUZ. + for (int i = 0; i < EXTERNAL_LUZS.length; i++) + { + int externalLuz = EXTERNAL_LUZS[i]; + cordonLuzMinutesForExternalLuz[externalLuz] = MINUTES_TO_ADD_TO_CORDON_FOR_EXTERNAL_LUZ[i]; + } + + } + + /** + * Calculate size terms and store in sizeTerms array. This method + * initializes the sizeTerms array and loops through mgras in the + * mgraManager, calculates the size term for all size term purposes as + * defined in the size term uec, and stores the results in the sizeTerms + * array. + * + */ + public void calculateSizeTerms() + { + + logger.info("Calculating Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int alternatives = sizeTermUEC.getNumberOfAlternatives(); + sizeTerms = new double[maxMgra + 1][alternatives]; + hasSizeTerm = new boolean[maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = sizeTermUEC.solve(iv, aDmu, null); + + // if ( mgra < 100 ) + // sizeTermUEC.logAnswersArray(logger, + // "NonMandatory Size Terms, MGRA = " + mgra ); + + // store the size terms + for (int purp = 0; purp < alternatives; ++purp) + { + sizeTerms[mgra][purp] = utilities[purp]; + if (sizeTerms[mgra][purp] > 0) hasSizeTerm[mgra] = true; + } + + // log + if (tracer.isTraceOn() && tracer.isTraceZone(taz)) + { + + logger.info("Size Term calculations for mgra " + mgra); + sizeTermUEC.logResultsArray(logger, 0, mgra); + + } + } + } + + /** + * Calculate size terms used for worker DC and store in workerSizeTerms + * array. This method initializes the workerSizeTerms array and loops + * through mgras in the mgraManager, calculates the size term for all work + * size term occupation categories as defined in the worker size term uec, + * and stores the results in the workerSizeTerms array. + * + */ + public void calculateWorkerSizeTerms() + { + + logger.info("Calculating Worker DC Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int alternatives = workerSizeTermUEC.getNumberOfAlternatives(); + workerSizeTerms = new double[alternatives][maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = workerSizeTermUEC.solve(iv, aDmu, null); + + // store the size terms + for (int segment = 0; segment < alternatives; segment++) + { + workerSizeTerms[segment][mgra] = utilities[segment]; + } + + // log + if (tracer.isTraceOn() && tracer.isTraceZone(taz)) + { + logger.info("Worker Size Term calculations for mgra " + mgra); + workerSizeTermUEC.logResultsArray(logger, 0, mgra); + } + } + } + + /** + * Calculate size terms used for school DC and store in schoolSizeTerms + * array. This method initializes the schoolSizeTerms array and loops + * through mgras in the mgraManager, calculates the size term for all school + * size term categories, preschool is defined in the preschool size term + * uec, K-8 and 9-12 use their respective enrollments as size terms, and + * university uses a size term uec, segmented by "typical student". Size + * terms for preschool, k-8, 9-12, university typical amd university + * non-typical are stored in the studentSizeTerms array. + * + */ + public void calculateSchoolSizeTerms() + { + + logger.info("Calculating Student DC Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + + String[] schoolSizeNames = getSchoolSegmentNameList(); + schoolSizeTerms = new double[schoolSizeNames.length][maxMgra + 1]; + luSchoolSizeTerms = new double[UNIV_NONTYPICAL_ALT_INDEX + 1][maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + // int dummy=0; + // if ( mgra == 1801 ){ + // dummy = 1; + // } + + int gsDistrict = getMgraGradeSchoolDistrict(mgra); + int hsDistrict = getMgraHighSchoolDistrict(mgra); + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = schoolSizeTermUEC.solve(iv, aDmu, null); + + // store the preschool size terms + schoolSizeTerms[PRESCHOOL_ALT_INDEX][mgra] = utilities[PRESCHOOL_ALT_INDEX]; + luSchoolSizeTerms[PRESCHOOL_ALT_INDEX][mgra] = utilities[PRESCHOOL_ALT_INDEX]; + + // store the grade school size term for the district this mgra is in + int seg = getGsDistrictIndex(gsDistrict); + schoolSizeTerms[seg][mgra] = utilities[GRADE_SCHOOL_ALT_INDEX]; + luSchoolSizeTerms[GRADE_SCHOOL_ALT_INDEX][mgra] = utilities[GRADE_SCHOOL_ALT_INDEX]; + + // store the high school size term for the district this mgra is in + seg = getHsDistrictIndex(hsDistrict); + schoolSizeTerms[seg][mgra] = utilities[HIGH_SCHOOL_ALT_INDEX]; + luSchoolSizeTerms[HIGH_SCHOOL_ALT_INDEX][mgra] = utilities[HIGH_SCHOOL_ALT_INDEX]; + + // store the university typical size terms + schoolSizeTerms[univTypicalSegment][mgra] = utilities[UNIV_TYPICAL_ALT_INDEX]; + luSchoolSizeTerms[UNIV_TYPICAL_ALT_INDEX][mgra] = utilities[UNIV_TYPICAL_ALT_INDEX]; + + // store the university non-typical size terms + schoolSizeTerms[univNonTypicalSegment][mgra] = utilities[UNIV_NONTYPICAL_ALT_INDEX]; + luSchoolSizeTerms[UNIV_NONTYPICAL_ALT_INDEX][mgra] = utilities[UNIV_NONTYPICAL_ALT_INDEX]; + + // log + if (tracer.isTraceOn() && tracer.isTraceZone(taz)) + { + logger.info("School Size Term calculations for mgra " + mgra); + schoolSizeTermUEC.logResultsArray(logger, 0, mgra); + } + } + } + + public double[][] calculateSchoolSegmentFactors() + { + + ArrayList mgras = mgraManager.getMgras(); + int maxMgra = mgraManager.getMaxMgra(); + String[] schoolSizeNames = getSchoolSegmentNameList(); + double[][] schoolFactors = new double[schoolSizeNames.length][maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + // store the size terms + double univEnrollment = getTotalMgraUniversityEnrollment(mgra); + + for (int seg : nonUniversitySegments) + schoolFactors[seg][mgra] = 1.0; + + double totalSize = 0.0; + for (int seg : universitySegments) + totalSize += schoolSizeTerms[seg][mgra]; + + for (int seg : universitySegments) + if (totalSize == 0) schoolFactors[seg][mgra] = 0; + else schoolFactors[seg][mgra] = univEnrollment / totalSize; + + } + + return schoolFactors; + + } + + /** + * Calculate constant terms, exponentiate, and store in constants array. + */ + public void calculateConstants() + { + + logger.info("Calculating constants"); + + int modes = constantsUEC.getNumberOfAlternatives(); + expConstants = new double[MARKET_SEGMENTS + 1][modes]; // last element + // in + // market segments is + // for total + + for (int i = 0; i < MARKET_SEGMENTS + 1; ++i) + { + + aDmu.setAutoSufficiency(i); + + double[] utilities = constantsUEC.solve(iv, aDmu, null); + + // exponentiate the constants + for (int j = 0; j < modes; ++j) + { + expConstants[i][j] = Math.exp(utilities[j]); + logger.info("Exp. Constant, market " + i + " mode " + j + " = " + + expConstants[i][j]); + } + } + } + + public void calculateDCUtilitiesDistributed(HashMap rbMap) + { + + int packetSize = DISTRIBUTED_PACKET_SIZE; + int numPackets = mgraManager.getMgras().size(); + + int startIndex = 0; + int endIndex = 0; + + ArrayList startEndIndexList = new ArrayList(); + + // assign start, end MGRA ranges to be used to assign to tasks + while (endIndex < numPackets - 1) + { + endIndex = startIndex + packetSize - 1; + + if (endIndex + packetSize > numPackets) endIndex = numPackets - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += packetSize; + } + + float[][] accessibilities = submitTasks(startEndIndexList, rbMap); + accessibilitiesTableObject = new AccessibilitiesTable(accessibilities); + + // output data + String projectDirectory = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(rbMap, "acc.output.file"); + accessibilitiesTableObject.writeAccessibilityTableToFile(accFileName); + accessibilitiesBuilt = true; + + if (landUseLogsums != null) + { + + boolean useSimpleMethod = false; + useSimpleMethod = Util.getBooleanValueFromPropertyMap(rbMap, + "lu.acc.simple.averaging.method"); + + boolean useLogitMethod = false; + useLogitMethod = Util.getBooleanValueFromPropertyMap(rbMap, + "lu.acc.logit.averaging.method"); + + // output land use data + String luAccFileName = projectDirectory + + Util.getStringValueFromPropertyMap(rbMap, "lu.acc.output.file"); + float[][][] luzAccessibilities = aggregateMgraAccessibilitiesToLuz(landUseAccessibilities); + if (useSimpleMethod) + accessibilitiesTableObject + .writeLandUseAccessibilityTableToFile( + luAccFileName.substring(0, luAccFileName.indexOf('.')) + "_simple" + + ".csv", luzAccessibilities[SIMPLE]); + if (useLogitMethod) + accessibilitiesTableObject.writeLandUseAccessibilityTableToFile( + luAccFileName.substring(0, luAccFileName.indexOf('.')) + "_logit" + ".csv", + luzAccessibilities[LOGIT]); + + // output land use mode choice logsums + String luMcLogsumsFileName = projectDirectory + + Util.getStringValueFromPropertyMap(rbMap, "lu.acc.mc.logsums.output.file"); + if (useSimpleMethod) + accessibilitiesTableObject.writeLandUseLogsumTablesToFile( + luMcLogsumsFileName.substring(0, luMcLogsumsFileName.indexOf('.')) + + "_simple" + ".csv", landUseLogsums[SIMPLE]); + if (useLogitMethod) + accessibilitiesTableObject.writeLandUseLogsumTablesToFile( + luMcLogsumsFileName.substring(0, luMcLogsumsFileName.indexOf('.')) + + "_logit" + ".csv", landUseLogsums[LOGIT]); + + } + + } + + private float[][][] aggregateMgraAccessibilitiesToLuz(float[][] landUseAccessibilities) + { + + // simple averaging uses accumulated logsum values, logit averaging uses + // accumulated utility values + float[][][] returnValues = new float[NUM_AVG_METHODS][MAX_LUZ + 1][]; + int[] numberOfMgraValues = new int[MAX_LUZ + 1]; + + for (int i = 1; i < landUseAccessibilities.length; i++) + { + + if (landUseAccessibilities[i] == null) continue; + + // the last column of accessibilities holds the mgra value + int numColumns = landUseAccessibilities[i].length; + int mgra = (int) landUseAccessibilities[i][numColumns - 1]; + int luz = mgraManager.getMgraLuz(mgra); + + // allocate the columns for aggregate arrays + for (int k = 0; k < NUM_AVG_METHODS; k++) + { + if (returnValues[k][luz] == null) + { + returnValues[k][luz] = new float[numColumns]; + returnValues[k][luz][numColumns - 1] = luz; + } + } + + numberOfMgraValues[luz]++; + + // aggregate the mgra column values in the corresponding luz row of + // the return array - excluding the last column + for (int j = 0; j < numColumns - 1; j++) + { + returnValues[SIMPLE][luz][j] += landUseAccessibilities[i][j]; + returnValues[LOGIT][luz][j] += Math.exp(landUseAccessibilities[i][j]); + } + + // calculate logsums from external LUZs to all destination LUZs if + // the origin LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[luz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[luz]) + { + + // allocate the columns for aggregate arrays + for (int k = 0; k < NUM_AVG_METHODS; k++) + { + if (returnValues[k][exLuz] == null) + { + returnValues[k][exLuz] = new float[numColumns]; + returnValues[k][exLuz][numColumns - 1] = exLuz; + } + } + + numberOfMgraValues[exLuz]++; + + for (int j = 0; j < numColumns - 1; j++) + { + returnValues[SIMPLE][exLuz][j] += landUseAccessibilities[i][j]; + returnValues[LOGIT][exLuz][j] += Math.exp(landUseAccessibilities[i][j]); + } + + } + + } + + } + + for (int i = 0; i < returnValues[SIMPLE].length; i++) + { + + if (returnValues[SIMPLE][i] == null) continue; + + // the last column of accessibilities holds the mgra value + int numColumns = returnValues[SIMPLE][i].length; + int luz = (int) returnValues[SIMPLE][i][numColumns - 1]; + + // average the the luz values of the return array - excluding the + // last column + for (int j = 0; j < numColumns - 1; j++) + { + returnValues[SIMPLE][luz][j] /= numberOfMgraValues[luz]; + returnValues[LOGIT][luz][j] = (float) Math.log(returnValues[LOGIT][luz][j]); + } + + // calculate logsums from external LUZs to all destination LUZs if + // the origin LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[luz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[luz]) + { + + for (int j = 0; j < numColumns - 1; j++) + { + returnValues[SIMPLE][exLuz][j] /= numberOfMgraValues[exLuz]; + returnValues[LOGIT][exLuz][j] = (float) Math + .log(returnValues[LOGIT][exLuz][j]); + } + + } + + } + + } + + return returnValues; + + } + + public void readAccessibilityTableFromFile(String accFileName) + { + accessibilitiesTableObject = new AccessibilitiesTable(accFileName); + logger.info("accessibilities table read from file."); + } + + public boolean getAccessibilitiesAreBuilt() + { + return objInstance.accessibilitiesBuilt; + } + + public AccessibilitiesTable getAccessibilitiesTableObject() + { + return accessibilitiesTableObject; + } + + /** + * @param client + * is a JPPFClient object which is used to establish a connection + * to a computing node, submit tasks, and receive results. + */ + private float[][] submitTasks(ArrayList startEndIndexList, HashMap rbMap) + { + + // Create a setup task object and submit it to the computing node. + // This setup task creates the HouseholdChoiceModelManager and causes it + // to + // create the necessary numuber + // of HouseholdChoiceModels objects which will operate in parallel on + // the + // computing node. + + ExecutorService exec = Executors.newFixedThreadPool(numThreads); + ArrayList>> results = new ArrayList>>(); + + float[][][][][] accumulatedLandUseLogsums; + int[][] accumulatedLandUseLogsumsCount; + + int startIndex = 0; + int endIndex = 0; + int taskIndex = 1; + for (int[] startEndIndices : startEndIndexList) + { + startIndex = startEndIndices[0]; + endIndex = startEndIndices[1]; + + logger.info(String.format("creating TASK: %d range: %d to %d.", taskIndex, startIndex, + endIndex)); + + DcUtilitiesTaskJppf task = new DcUtilitiesTaskJppf(taskIndex, startIndex, endIndex, + sovExpUtilities, hovExpUtilities, nMotorExpUtilities, hasSizeTerm, + expConstants, sizeTerms, workerSizeTerms, luSchoolSizeTerms, + externalLuzsForCordonLuz, cordonLuzForExternalLuz, + cordonLuzMinutesForExternalLuz, rbMap, calculateLuAccessibilities); + + results.add(exec.submit(task)); + taskIndex++; + } + + float[][] accessibilities = new float[maxMgra + 1][]; + + for (Future> fs : results) + { + + try + { + List resultBundle = fs.get(); + int task = (Integer) resultBundle.get(0); + int start = (Integer) resultBundle.get(1); + int end = (Integer) resultBundle.get(2); + if (logResults) logger.info(String.format("returned TASK: %d, start=%d, end=%d.", task, start, end)); + float[][] taskAccessibilities = (float[][]) resultBundle.get(3); + for (int i = 0; i < taskAccessibilities.length; i++) + { + if (taskAccessibilities[i] == null) continue; + int numColumns = taskAccessibilities[i].length; + int mgra = (int) taskAccessibilities[i][numColumns - 1]; + accessibilities[mgra] = taskAccessibilities[i]; + } + + if (landUseLogsums != null) + { + + // get land use accessibilities result if they were + // calculated + taskAccessibilities = (float[][]) resultBundle.get(4); + if (taskAccessibilities != null) + { + for (int i = 0; i < taskAccessibilities.length; i++) + { + if (taskAccessibilities[i] == null) continue; + int numColumns = taskAccessibilities[i].length; + int mgra = (int) taskAccessibilities[i][numColumns - 1]; + landUseAccessibilities[mgra] = taskAccessibilities[i]; + } + } + + // store the land use logsums array + accumulatedLandUseLogsums = (float[][][][][]) resultBundle.get(5); + accumulatedLandUseLogsumsCount = (int[][]) resultBundle.get(6); + accumulateLandUseModeChoiceLogsums(accumulatedLandUseLogsums, + accumulatedLandUseLogsumsCount); + + } + + } catch (InterruptedException e) + { + e.printStackTrace(); + throw new RuntimeException(); + } catch (ExecutionException e) + { + logger.error("Exception returned in place of result object.", e); + throw new RuntimeException(); + } finally + { + exec.shutdown(); + } + + } // future + + // calculate the averages for the land use mode choice logsums + if (landUseLogsums != null) averageLandUseModeChoiceLogsums(); + + return accessibilities; + + } + + private void averageLandUseModeChoiceLogsums() + { + + for (int i = 0; i < NUM_PERIODS; i++) + { + for (int j = 0; j < NUM_SUFFICIENCY_SEGMENTS; j++) + { + for (int k = 1; k <= MAX_LUZ; k++) + { + for (int m = 1; m <= MAX_LUZ; m++) + { + landUseLogsums[SIMPLE][i][j][k][m] = landUseLogsums[SIMPLE][i][j][k][m] + / landUseLogsumsCount[k][m]; + landUseLogsums[LOGIT][i][j][k][m] = Math + .log(landUseLogsums[LOGIT][i][j][k][m]); + } + } + } + } + } + + private void accumulateLandUseModeChoiceLogsums(float[][][][][] accumulatedLandUseLogsums, + int[][] accumulatedLandUseLogsumsCount) + { + + for (int k = 1; k <= MAX_LUZ; k++) + { + for (int m = 1; m <= MAX_LUZ; m++) + { + landUseLogsumsCount[k][m] += accumulatedLandUseLogsumsCount[k][m]; + for (int i = 0; i < NUM_PERIODS; i++) + { + for (int j = 0; j < NUM_SUFFICIENCY_SEGMENTS; j++) + { + landUseLogsums[SIMPLE][i][j][k][m] += accumulatedLandUseLogsums[SIMPLE][i][j][k][m]; + landUseLogsums[LOGIT][i][j][k][m] += accumulatedLandUseLogsums[LOGIT][i][j][k][m]; + } + } + } + } + + } + + public double[][] getExpConstants() + { + return expConstants; + } + + /** + * @return the array of alternative labels from the UEC used to calculate + * work tour destination choice size terms. + */ + public String[] getWorkSegmentNameList() + { + return WORK_OCCUP_SEGMENT_NAME_LIST; + } + + /** + * @return the table, MGRAs by occupations, of size terms calcuated for + * worker DC. + */ + public double[][] getWorkerSizeTerms() + { + return workerSizeTerms; + } + + /** + * @return the array of alternative labels from the UEC used to calculate + * work tour destination choice size terms. + */ + public String[] getSchoolSegmentNameList() + { + return schoolSegmentSizeNames; + } + + /** + * Specify the mapping between PECAS occupation codes, occupation segment + * labels, and work location choice segment indices. + */ + public void createWorkSegmentNameIndices() + { + + // get the list of segment names for worker tour destination choice size + String[] occupNameList = WORK_OCCUP_SEGMENT_NAME_LIST; + + // get the list of segment values for worker tour destination choice + // size + int[] occupValueList = WORK_OCCUP_SEGMENT_VALUE_LIST; + + for (int value : occupValueList) + { + int index = workerOccupValueSegmentIndexMap.get(value); + String name = occupNameList[index]; + workSegmentIndexNameMap.put(index, name); + workSegmentNameIndexMap.put(name, index); + } + + } + + /** + * Specify/create the mapping between school segment labels, and school + * location choice segment indices. + */ + public void createSchoolSegmentNameIndices() + { + + ArrayList segmentNames = new ArrayList(); + Set gsDistrictSet = getGradeSchoolDistrictIndices(); + Set hsDistrictSet = getHighSchoolDistrictIndices(); + + String segmentName = ""; + + // add preschool segment to list of segments which will not be shadow + // price adjusted + int sizeSegmentIndex = 0; + segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[PRESCHOOL_SEGMENT_GROUP_INDEX]; + noShadowPriceSchoolSegmentIndices.add(sizeSegmentIndex); + + // add preschool segment as the first segment, with index=0. + segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[PRESCHOOL_SEGMENT_GROUP_INDEX]; + segmentNames.add(segmentName); + schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + psSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + psSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + + // increment the segmentIndex so it's new value is the first grade + // school index + // add grade school segments to list + sizeSegmentIndex++; + gsDistrictIndexMap = new HashMap(); + for (int gsDist : gsDistrictSet) + { + if (gsDist > UNIFIED_DISTRICT_OFFSET) segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIFIED_GRADE_SCHOOL_SEGMENT_GROUP_INDEX] + + "_" + (gsDist - UNIFIED_DISTRICT_OFFSET); + else segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[GRADE_SCHOOL_SEGMENT_GROUP_INDEX] + + "_" + gsDist; + segmentNames.add(segmentName); + schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + gsSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + gsSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + gsDistrictIndexMap.put(gsDist, sizeSegmentIndex); + sizeSegmentIndex++; + } + + // add high school segments to list + hsDistrictIndexMap = new HashMap(); + for (int hsDist : hsDistrictSet) + { + if (hsDist > UNIFIED_DISTRICT_OFFSET) segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIFIED_HIGH_SCHOOL_SEGMENT_GROUP_INDEX] + + "_" + (hsDist - UNIFIED_DISTRICT_OFFSET); + else segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[HIGH_SCHOOL_SEGMENT_GROUP_INDEX] + + "_" + hsDist; + segmentNames.add(segmentName); + schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + hsSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + hsSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + hsDistrictIndexMap.put(hsDist, sizeSegmentIndex); + sizeSegmentIndex++; + } + + // add typical university/colleger segments to list + segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIV_TYPICAL_SEGMENT_GROUP_INDEX]; + segmentNames.add(segmentName); + schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + univTypSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + univTypSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + univTypicalSegment = sizeSegmentIndex; + sizeSegmentIndex++; + + // add non-typical university/colleger segments to list + segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]; + segmentNames.add(segmentName); + schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + univNonTypSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); + univNonTypSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); + univNonTypicalSegment = sizeSegmentIndex; + + // create arrays dimensioned as the number of school segments created + // and assign their values + + // 2 university segments + universitySegments = new int[2]; + + // num GS Dists + num HS dists + preschool non-university segments + nonUniversitySegments = new int[segmentNames.size() - universitySegments.length]; + + int u = 0; + int nu = 0; + + schoolSegmentSizeNames = new String[segmentNames.size()]; + for (int i = 0; i < schoolSegmentSizeNames.length; i++) + schoolSegmentSizeNames[i] = segmentNames.get(i); + + schoolDcSoaUecSheets = new int[schoolSegmentSizeNames.length]; + schoolDcUecSheets = new int[schoolSegmentSizeNames.length]; + schoolStfUecSheets = new int[schoolSegmentSizeNames.length]; + + sizeSegmentIndex = 0; + schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; + schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; + schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; + nonUniversitySegments[nu++] = sizeSegmentIndex; + sizeSegmentIndex++; + + for (int segmentIndex : gsDistrictIndexMap.values()) + { + schoolDcSoaUecSheets[segmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; + schoolDcUecSheets[segmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; + schoolStfUecSheets[segmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; + nonUniversitySegments[nu++] = segmentIndex; + sizeSegmentIndex++; + } + + for (int segmentIndex : hsDistrictIndexMap.values()) + { + schoolDcSoaUecSheets[segmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; + schoolDcUecSheets[segmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; + schoolStfUecSheets[segmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; + nonUniversitySegments[nu++] = segmentIndex; + sizeSegmentIndex++; + } + + schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; + schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; + schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; + universitySegments[u++] = sizeSegmentIndex; + sizeSegmentIndex++; + + schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; + schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; + schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; + universitySegments[u++] = sizeSegmentIndex; + + } + + public HashSet getNoShadowPriceSchoolSegmentIndexSet() + { + return noShadowPriceSchoolSegmentIndices; + } + + public HashMap getSchoolSegmentIndexNameMap() + { + return schoolSegmentIndexNameMap; + } + + public HashMap getSchoolSegmentNameIndexMap() + { + return schoolSegmentNameIndexMap; + } + + /** + * @return pre-school segment index to segment name hashmap + */ + public HashMap getPsSegmentIndexNameMap() + { + return psSegmentIndexNameMap; + } + + /** + * @return pre-school segment name to segment index hashmap + */ + public HashMap getPsSegmentNameIndexMap() + { + return psSegmentNameIndexMap; + } + + /** + * @return grade school segment index to segment name hashmap + */ + public HashMap getGsSegmentIndexNameMap() + { + return gsSegmentIndexNameMap; + } + + /** + * @return grade school segment name to segment index hashmap + */ + public HashMap getGsSegmentNameIndexMap() + { + return gsSegmentNameIndexMap; + } + + /** + * @return high school segment index to segment name hashmap + */ + public HashMap getHsSegmentIndexNameMap() + { + return hsSegmentIndexNameMap; + } + + /** + * @return high school segment name to segment index hashmap + */ + public HashMap getHsSegmentNameIndexMap() + { + return hsSegmentNameIndexMap; + } + + /** + * @return university typical segment index to segment name hashmap + */ + public HashMap getUnivTypicalSegmentIndexNameMap() + { + return univTypSegmentIndexNameMap; + } + + /** + * @return university typical segment name to segment index hashmap + */ + public HashMap getUnivTypicalSegmentNameIndexMap() + { + return univTypSegmentNameIndexMap; + } + + /** + * @return university non typical segment index to segment name hashmap + */ + public HashMap getUnivNonTypicalSegmentIndexNameMap() + { + return univNonTypSegmentIndexNameMap; + } + + /** + * @return university non typical segment name to segment index hashmap + */ + public HashMap getUnivNonTypicalSegmentNameIndexMap() + { + return univNonTypSegmentNameIndexMap; + } + + public HashMap getWorkSegmentIndexNameMap() + { + return workSegmentIndexNameMap; + } + + public HashMap getWorkSegmentNameIndexMap() + { + return workSegmentNameIndexMap; + } + + public HashMap getWorkOccupValueIndexMap() + { + return workerOccupValueSegmentIndexMap; + } + + public int getGsDistrictIndex(int district) + { + return gsDistrictIndexMap.get(district); + } + + public int getHsDistrictIndex(int district) + { + return hsDistrictIndexMap.get(district); + } + + public int[] getSchoolDcSoaUecSheets() + { + return schoolDcSoaUecSheets; + } + + public int[] getSchoolDcUecSheets() + { + return schoolDcUecSheets; + } + + /** + * @return the array of stop frequency uec model sheet indices, indexed by + * school segment + */ + public int[] getSchoolStfUecSheets() + { + return schoolStfUecSheets; + } + + /** + * @return the table, MGRAs by school types, of size terms calcuated for + * school DC. + */ + public double[][] getSchoolSizeTerms() + { + return schoolSizeTerms; + } + + /** + * @return the table, MGRAs by non-mandatory types, of size terms calcuated. + */ + public double[][] getSizeTerms() + { + return sizeTerms; + } + + /** + * @param mgra + * for which table data is desired + * @return population for the specified mgra. + */ + public double getMgraPopulation(int mgra) + { + return mgraManager.getMgraPopulation(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return households for the specified mgra. + */ + public double getMgraHouseholds(int mgra) + { + return mgraManager.getMgraHouseholds(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return grade school enrollment for the specified mgra. + */ + public double getMgraGradeSchoolEnrollment(int mgra) + { + return mgraManager.getMgraGradeSchoolEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return high school enrollment for the specified mgra. + */ + public double getMgraHighSchoolEnrollment(int mgra) + { + return mgraManager.getMgraHighSchoolEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return university enrollment for the specified mgra. + */ + public double getTotalMgraUniversityEnrollment(int mgra) + { + return mgraManager.getMgraUniversityEnrollment(mgra) + + mgraManager.getMgraOtherCollegeEnrollment(mgra) + + mgraManager.getMgraAdultSchoolEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return university enrollment for the specified mgra. + */ + public double getMgraUniversityEnrollment(int mgra) + { + return mgraManager.getMgraUniversityEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return other college enrollment for the specified mgra. + */ + public double getMgraOtherCollegeEnrollment(int mgra) + { + return mgraManager.getMgraOtherCollegeEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return adult school enrollment for the specified mgra. + */ + public double getMgraAdultSchoolEnrollment(int mgra) + { + return mgraManager.getMgraAdultSchoolEnrollment(mgra); + } + + /** + * @param mgra + * for which table data is desired + * @return grade school district for the specified mgra. + */ + public int getMgraGradeSchoolDistrict(int mgra) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); + int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); + if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + return gsDist; + } + + /** + * @param mgra + * for which table data is desired + * @return high school district for the specified mgra. + */ + public int getMgraHighSchoolDistrict(int mgra) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); + int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); + if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + return hsDist; + } + + /** + * @param mgra + * for which table data is desired + * @return school location choice model segment for the specified mgra. + */ + public int getMgraGradeSchoolSegmentIndex(int mgra) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); + int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); + if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + + return gsDistrictIndexMap.get(gsDist); + } + + /** + * @param mgra + * for which table data is desired + * @return school location choice model segment for the specified mgra. + */ + public int getMgraHighSchoolSegmentIndex(int mgra) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); + int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); + if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + + return hsDistrictIndexMap.get(hsDist); + } + + /** + * @return set of unique grade school district indices + */ + private TreeSet getGradeSchoolDistrictIndices() + { + int maxMgra = mgraManager.getMaxMgra(); + mgraGsDistrict = new int[maxMgra + 1]; + TreeSet set = new TreeSet(); + for (int r = 1; r <= maxMgra; r++) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(r); + int hsDist = mgraManager.getMgraHighSchoolDistrict(r); + if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + set.add(gsDist); + mgraGsDistrict[r] = gsDist; + } + return set; + } + + /** + * @return set of unique high school district indices + */ + private TreeSet getHighSchoolDistrictIndices() + { + int maxMgra = mgraManager.getMaxMgra(); + mgraHsDistrict = new int[maxMgra + 1]; + TreeSet set = new TreeSet(); + for (int r = 1; r <= maxMgra; r++) + { + int gsDist = mgraManager.getMgraGradeSchoolDistrict(r); + int hsDist = mgraManager.getMgraHighSchoolDistrict(r); + if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; + set.add(hsDist); + mgraHsDistrict[r] = hsDist; + } + return set; + } + + public int[] getMgraGsDistrict() + { + return mgraGsDistrict; + } + + public int[] getMgraHsDistrict() + { + return mgraHsDistrict; + } + + public HashMap getGsDistrictIndexMap() + { + return gsDistrictIndexMap; + } + + public HashMap getHsDistrictIndexMap() + { + return hsDistrictIndexMap; + } + + public HashMap getNonMandatoryPurposeNameIndexMap() + { + return nonMandatorySizeSegmentNameIndexMap; + } + + public int getEscortSizeArraySegmentIndex() + { + return nonMandatorySizeSegmentNameIndexMap.get(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java new file mode 100644 index 0000000..9d05d1a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java @@ -0,0 +1,880 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.PrintWriter; +import java.net.UnknownHostException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.concurrent.Callable; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + + +public class DcUtilitiesTaskJppf + implements Callable> +{ + + private static final int MIN_EXP_FUNCTION_ARGUMENT = -500; + + private static final String[] LOGSUM_SEGMENTS = {"SOV ", + "HOV ", "Transit ", "NMotorized", "SOVLS_0 ", "SOVLS_1 ", "SOVLS_2 ", + "HOVLS_0_OP", "HOVLS_1_OP", "HOVLS_2_OP", "HOVLS_0_PK", "HOVLS_1_PK", "HOVLS_2_PK", + "TOTAL", "MAAS" }; + + public static final String[] LU_LOGSUM_SEGMENTS = {"LS_0_PK", "LS_1_PK", + "LS_2_PK", "LS_0_OP", "LS_1_OP", "LS_2_OP", "All_PK " }; + + // setting to -1 will prevent debug files from being written + private static final int DEBUG_ILUZ = -1; + private static final int DEBUG_JLUZ = -1; + // private static final int DEBUG_ILUZ = 66; + // private static final int DEBUG_JLUZ = 82; + + private static final int MAX_LU_SIZE_TERM_INDEX = 23; + private static final int MAX_LU_NONMAN_SIZE_TERM_INDEX = 12; + private static final int MAX_LU_WORK_SIZE_TERM_INDEX = 18; + private static final int MAX_LU_SCHOOL_SIZE_TERM_INDEX = 23; + + private MgraDataManager mgraManager; + private boolean[] hasSizeTerm; + private double[][] expConstants; + private double[][] sizeTerms; + private double[][] luSizeTerms; + + // store taz-taz exponentiated utilities (period, from taz, to taz) + private double[][][] sovExpUtilities; + private double[][][] hovExpUtilities; + private double[][][] nMotorExpUtilities; + private double[][][] maasExpUtilities; + + + private float[][] accessibilities; + private float[][] luAccessibilities; + + private int startRange; + private int endRange; + private int taskIndex; + + private boolean seek; + private Tracer tracer; + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + + private BestTransitPathCalculator bestPathCalculator; + + private UtilityExpressionCalculator dcUEC; + private AccessibilitiesDMU aDmu; + + private UtilityExpressionCalculator luUEC; + private AccessibilitiesDMU luDmu; + + private HashMap rbMap; + + private int[][] externalLuzsForCordonLuz; + private int[] cordonLuzForExternalLuz; + private int[] cordonLuzMinutesForExternalLuz; + + private boolean calculateLuAccessibilities; + private PrintWriter outStream; + + public DcUtilitiesTaskJppf(int taskIndex, int startRange, int endRange, + double[][][] mySovExpUtilities, double[][][] myHovExpUtilities, + double[][][] myNMotorExpUtilities, boolean[] hasSizeTerm, double[][] expConstants, + double[][] sizeTerms, double[][] workSizeTerms, double[][] schoolSizeTerms, + int[][] myExternalLuzsForCordonLuz, int[] myCordonLuzForExternalLuz, + int[] myCordonLuzMinutesForExternalLuz, HashMap myRbMap, + boolean myCalculateLuAccessibilities) + { + + rbMap = myRbMap; + sovExpUtilities = mySovExpUtilities; + hovExpUtilities = myHovExpUtilities; + nMotorExpUtilities = myNMotorExpUtilities; + externalLuzsForCordonLuz = myExternalLuzsForCordonLuz; + cordonLuzForExternalLuz = myCordonLuzForExternalLuz; + cordonLuzMinutesForExternalLuz = myCordonLuzMinutesForExternalLuz; + calculateLuAccessibilities = myCalculateLuAccessibilities; + + mgraManager = MgraDataManager.getInstance(rbMap); + + aDmu = new AccessibilitiesDMU(); + + String dcUecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.dcUtility.uec.file"); + int dcDataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.dcUtility.data.page"); + int dcUtilityPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.dcUtility.page"); + + File dcUecFile = new File(dcUecFileName); + dcUEC = new UtilityExpressionCalculator(dcUecFile, dcUtilityPage, dcDataPage, rbMap, aDmu); + + accessibilities = new float[mgraManager.getMaxMgra() + 1][]; + + if (calculateLuAccessibilities) + { + + luDmu = new AccessibilitiesDMU(); + + dcUecFileName = Util.getStringValueFromPropertyMap(rbMap, "lu.acc.dcUtility.uec.file"); + dcDataPage = Util.getIntegerValueFromPropertyMap(rbMap, "lu.acc.dcUtility.data.page"); + dcUtilityPage = Util.getIntegerValueFromPropertyMap(rbMap, "lu.acc.dcUtility.page"); + + File luUecFile = new File(dcUecFileName); + luUEC = new UtilityExpressionCalculator(luUecFile, dcUtilityPage, dcDataPage, rbMap, + luDmu); + + TableDataSet luAltData = luUEC.getAlternativeData(); + luDmu.setAlternativeData(luAltData); + int luAlts = luUEC.getNumberOfAlternatives(); + + luAccessibilities = new float[mgraManager.getMaxMgra() + 1][luAlts + 1]; + + // combine non-mandatory, work, and school size terms into one array + // to be indexed into as follows + // 0-12 non-mandatory, 13-18 work, 19-23 school + luSizeTerms = new double[sizeTerms.length][MAX_LU_SIZE_TERM_INDEX + 1]; + for (int c = 0; c <= MAX_LU_NONMAN_SIZE_TERM_INDEX; c++) + for (int r = 0; r < sizeTerms.length; r++) + luSizeTerms[r][c] = sizeTerms[r][c]; + + for (int c = 0; c < workSizeTerms.length; c++) + for (int r = 0; r < workSizeTerms[c].length; r++) + luSizeTerms[r][c + MAX_LU_NONMAN_SIZE_TERM_INDEX + 1] = workSizeTerms[c][r]; + + for (int c = 0; c < schoolSizeTerms.length; c++) + for (int r = 0; r < schoolSizeTerms[c].length; r++) + luSizeTerms[r][c + MAX_LU_WORK_SIZE_TERM_INDEX + 1] = schoolSizeTerms[c][r]; + + // for ( int c=MAX_LU_NONMAN_SIZE_TERM_INDEX+1; c <= + // MAX_LU_WORK_SIZE_TERM_INDEX; c++ ) + // for ( int r=0; r < workSizeTerms.length; r++ ) + // luSizeTerms[r][c] = + // workSizeTerms[r][c-MAX_LU_NONMAN_SIZE_TERM_INDEX-1]; + // + // for ( int c=MAX_LU_WORK_SIZE_TERM_INDEX+1; c <= + // MAX_LU_SCHOOL_SIZE_TERM_INDEX; c++ ) + // for ( int r=0; r < schoolSizeTerms.length; r++ ) + // luSizeTerms[r][c] = + // schoolSizeTerms[r][c-MAX_LU_WORK_SIZE_TERM_INDEX-1]; + + // try { + // outStream = new PrintWriter( new BufferedWriter( new FileWriter( + // "landUseModeChoiceLogsumCheck" + "_" + taskIndex + ".csv" ) ) ); + // } + // catch (IOException e) { + // System.out.println("IO Exception writing file for checking integerizing procedure: " + // ); + // e.printStackTrace(); + // System.exit(-1); + // } + + } + + this.taskIndex = taskIndex; + this.startRange = startRange; + this.endRange = endRange; + this.hasSizeTerm = hasSizeTerm; + this.expConstants = expConstants; + this.sizeTerms = sizeTerms; + + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + } + + public String getId() + { + return Integer.toString(taskIndex); + } + + public List call() + { + + Logger logger = Logger.getLogger(this.getClass()); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + ", task:" + + taskIndex + "] " + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + logger.info(threadName + " - Calculating Accessibilities"); + + NonTransitUtilities ntUtilities = new NonTransitUtilities(rbMap, sovExpUtilities, + hovExpUtilities, nMotorExpUtilities, maasExpUtilities); + // ntUtilities.setAllUtilities(ntUtilitiesArrays); + // ntUtilities.setNonMotorUtilsMap(ntUtilitiesMap); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(rbMap); + bestPathCalculator = logsumHelper.getBestTransitPathCalculator(); + AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + + // get the accessibilities alternatives + int alts = dcUEC.getNumberOfAlternatives(); + TableDataSet altData = dcUEC.getAlternativeData(); + aDmu.setAlternativeData(altData); + + int luAlts = -1; + double[] luUtilities = null; + ArrayList luUtilityList = null; + float[][][][][] accumulatedLandUseLogsums = null; + int[][] accumulatedLandUseLogsumsCount = null; + + if (calculateLuAccessibilities) + { + // get the land use accessibilities alternatives + luAlts = luUEC.getNumberOfAlternatives(); + TableDataSet luAltData = luUEC.getAlternativeData(); + luDmu.setAlternativeData(luAltData); + + // declare logsums array for LU accessibility + luUtilities = new double[LU_LOGSUM_SEGMENTS.length]; + + luUtilityList = new ArrayList(); + + accumulatedLandUseLogsums = new float[BuildAccessibilities.NUM_AVG_METHODS][BuildAccessibilities.NUM_PERIODS][BuildAccessibilities.NUM_SUFFICIENCY_SEGMENTS][BuildAccessibilities.MAX_LUZ + 1][BuildAccessibilities.MAX_LUZ + 1]; + + accumulatedLandUseLogsumsCount = new int[BuildAccessibilities.MAX_LUZ + 1][BuildAccessibilities.MAX_LUZ + 1]; + + } + + // declare logsums array for ABM + double[] logsums = new double[LOGSUM_SEGMENTS.length]; + + float[] luUtilityResult = new float[LU_LOGSUM_SEGMENTS.length + 2]; + + // DMUs for this UEC + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // LOOP OVER RANGE OF ORIGIN MGRA + ArrayList mgraValues = mgraManager.getMgras(); + for (int i = startRange; i <= endRange; i++) + { // Origin MGRA + + int iMgra = mgraValues.get(i); + + accessibilities[iMgra] = new float[alts + 1]; + + // pre-calculate the hov, sov, and non-motorized exponentiated + // utilities for the origin MGRA. + // the method called returns cached values if they were already + // calculated. + ntUtilities.buildUtilitiesForOrigMgraAndPeriod(iMgra, + NonTransitUtilities.PEAK_PERIOD_INDEX); + ntUtilities.buildUtilitiesForOrigMgraAndPeriod(iMgra, + NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + + // if(originMgras<=10 || (originMgras % 500) ==0 ) + // logger.info("...Origin MGRA "+iMgra); + + int iTaz = mgraManager.getTaz(iMgra); + boolean trace = false; + + if (tracer.isTraceOn() && tracer.isTraceZone(iTaz)) + { + + logger.info("origMGRA, destMGRA, OPSOV, OPHOV, WTRAN, NMOT, SOV0OP, SOV1OP, SOV2OP, HOV0OP, HOV1OP, HOV2OP, HOV0PK, HOV1PK, HOV2PK, ALL"); + trace = true; + } + // for tracing accessibility and logsum calculations + String accString = null; + + // LOOP OVER DESTINATION MGRA + for (Integer jMgra : mgraManager.getMgras()) + { // Destination MGRA + + if (!hasSizeTerm[jMgra]) continue; + + int jTaz = mgraManager.getTaz(jMgra); + + if (seek && !trace) continue; + + double opSovExpUtility = 0; + double opHovExpUtility = 0; + double opMaasExpUtility = 0; + try + { + opSovExpUtility = ntUtilities.getSovExpUtility(iTaz, jTaz, + NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + opHovExpUtility = ntUtilities.getHovExpUtility(iTaz, jTaz, + NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + opMaasExpUtility = ntUtilities.getMaasExpUtility(iTaz, jTaz, + NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + // opSovExpUtility = + // ntUtilities.getAllUtilities()[0][0][iTaz][jTaz]; + // opHovExpUtility = + // ntUtilities.getAllUtilities()[1][0][iTaz][jTaz]; + } catch (Exception e) + { + logger.error("exception for op sov/hov utilitiy taskIndex=" + taskIndex + + ", i=" + i + ", startRange=" + startRange + ", endRange=" + endRange, + e); + System.exit(-1); + } + + // calculate walk-transit exponentiated utility + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.MD_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger); + + // sum the exponentiated utilities over modes + double opWTExpUtility = bestPathCalculator.getSumExpUtilities(); + + double pkSovExpUtility = 0; + double pkHovExpUtility = 0; + double pkMaasExpUtility = 0; + try + { + pkSovExpUtility = ntUtilities.getSovExpUtility(iTaz, jTaz, + NonTransitUtilities.PEAK_PERIOD_INDEX); + pkHovExpUtility = ntUtilities.getHovExpUtility(iTaz, jTaz, + NonTransitUtilities.PEAK_PERIOD_INDEX); + + pkMaasExpUtility = ntUtilities.getMaasExpUtility(iTaz, jTaz, + NonTransitUtilities.PEAK_PERIOD_INDEX); + + // pkSovExpUtility = + // ntUtilities.getAllUtilities()[0][1][iTaz][jTaz]; + // pkHovExpUtility = + // ntUtilities.getAllUtilities()[1][1][iTaz][jTaz]; + } catch (Exception e) + { + logger.error("exception for pk sov/hov utility taskIndex=" + taskIndex + ", i=" + + i + ", startRange=" + startRange + ", endRange=" + endRange, e); + System.exit(-1); + } + + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger); + + // sum the exponentiated utilities over modes + double pkWTExpUtility = bestPathCalculator.getSumExpUtilities(); + + double pkDTExpUtility = 0; + double opDTExpUtility = 0; + + if (calculateLuAccessibilities) + { + + float odDistance = (float) anm.getTazDistanceFromTaz(iTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[jTaz]; + + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger, odDistance); + + // sum the exponentiated utilities over modes + double driveTransitWalkUtilities[] = bestPathCalculator.getBestUtilities(); + if(trace){ + logger.info("PK Drive Transit Utilities (TAP pair number,utility, sum)"); + } + for (int k=0; k < driveTransitWalkUtilities.length; k++){ + if ( driveTransitWalkUtilities[k] > MIN_EXP_FUNCTION_ARGUMENT ) + pkDTExpUtility += Math.exp(driveTransitWalkUtilities[k]); + if(trace) + logger.info(k+","+driveTransitWalkUtilities[k]+","+pkDTExpUtility); + } + + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.MD_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger, odDistance); + + // sum the exponentiated utilities over modes + driveTransitWalkUtilities = bestPathCalculator.getBestUtilities(); + if(trace){ + logger.info("OP Drive Transit Utilities (TAP pair number,utility, sum)"); + } + for (int k=0; k < driveTransitWalkUtilities.length; k++){ + if ( driveTransitWalkUtilities[k] > MIN_EXP_FUNCTION_ARGUMENT ) + opDTExpUtility += Math.exp(driveTransitWalkUtilities[k]); + if(trace) + logger.info(k+","+driveTransitWalkUtilities[k]+","+opDTExpUtility); + } + + } + + double nmExpUtility = 0; + try + { + nmExpUtility = ntUtilities.getNMotorExpUtility(iMgra, jMgra, + NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + } catch (Exception e) + { + logger.error("exception for non-motorized utilitiy taskIndex=" + taskIndex + + ", i=" + i + ", startRange=" + startRange + ", endRange=" + endRange, + e); + System.exit(-1); + } + + Arrays.fill(logsums, -999f); + + // 0: OP SOV + logsums[0] = Math.log(opSovExpUtility); + + // 1: OP HOV + logsums[1] = Math.log(opHovExpUtility); + + // 2: Walk-Transit + if (opWTExpUtility > 0) logsums[2] = Math.log(opWTExpUtility); + + // 3: Non-Motorized + if (nmExpUtility > 0) logsums[3] = Math.log(nmExpUtility); + + // 4: SOVLS_0 + logsums[4] = Math.log(opSovExpUtility * expConstants[0][0] + opWTExpUtility + * expConstants[0][2] + nmExpUtility * expConstants[0][3]); + // 5: SOVLS_1 + logsums[5] = Math.log(opSovExpUtility * expConstants[1][0] + opWTExpUtility + * expConstants[1][2] + nmExpUtility * expConstants[1][3]); + + // 6: SOVLS_2 + logsums[6] = Math.log(opSovExpUtility * expConstants[2][0] + opWTExpUtility + * expConstants[2][2] + nmExpUtility * expConstants[2][3]); + + // 7: HOVLS_0_OP + logsums[7] = Math.log(opHovExpUtility * expConstants[0][1] + opWTExpUtility + * expConstants[0][2] + nmExpUtility * expConstants[0][3]); + + // 8: HOVLS_1_OP + logsums[8] = Math.log(opHovExpUtility * expConstants[1][1] + opWTExpUtility + * expConstants[1][2] + nmExpUtility * expConstants[1][3]); + + // 9: HOVLS_2_OP + logsums[9] = Math.log(opHovExpUtility * expConstants[2][1] + opWTExpUtility + * expConstants[2][2] + nmExpUtility * expConstants[2][3]); + + // 10: HOVLS_0_PK + logsums[10] = Math.log(pkHovExpUtility * expConstants[0][1] + pkWTExpUtility + * expConstants[0][2] + nmExpUtility * expConstants[0][3]); + + // 11: HOVLS_1_PK + logsums[11] = Math.log(pkHovExpUtility * expConstants[1][1] + pkWTExpUtility + * expConstants[1][2] + nmExpUtility * expConstants[1][3]); + + // 12: HOVLS_2_PK + logsums[12] = Math.log(pkHovExpUtility * expConstants[2][1] + pkWTExpUtility + * expConstants[2][2] + nmExpUtility * expConstants[2][3]); + + // 13: ALL + logsums[13] = Math.log(pkSovExpUtility * expConstants[3][0] + pkHovExpUtility + * expConstants[3][1] + pkWTExpUtility * expConstants[3][2] + nmExpUtility + * expConstants[3][3]); + + // 14: MAAS + logsums[14] = Math.log(opMaasExpUtility); + + aDmu.setLogsums(logsums); + aDmu.setSizeTerms(sizeTerms[jMgra]); + // double[] utilities = dcUEC.solve(iv, aDmu, null); + + if (trace) + { + String printString = new String(); + printString += (iMgra + "," + jMgra); + for (int j = 0; j < 14; ++j) + { + printString += "," + String.format("%9.2f", logsums[j]); + } + logger.info(printString); + + accString = new String(); + accString = "iMgra, jMgra, Alternative, Logsum, SizeTerm, Accessibility\n"; + + } + // add accessibilities for origin mgra + for (int alt = 0; alt < alts; ++alt) + { + + double logsum = aDmu.getLogsum(alt + 1); + double sizeTerm = aDmu.getSizeTerm(alt + 1); + + accessibilities[iMgra][alt] += (Math.exp(logsum) * sizeTerm); + + if (trace) + { + accString += iMgra + "," + alt + "," + logsum + "," + sizeTerm + "," + + accessibilities[iMgra][alt] + "\n"; + } + + } + + // if luModeChoiceLogsums is null, is has not been initialized, + // meaning that LU accessibility calculations are not needed + if (calculateLuAccessibilities) + { + + // 0: AM Mode Choice utility for 0-autos auto sufficiency + luUtilities[0] = pkSovExpUtility * expConstants[0][0] + pkHovExpUtility + * expConstants[0][1] + pkWTExpUtility * expConstants[0][2] + + pkDTExpUtility * expConstants[0][4] + nmExpUtility + * expConstants[0][3]; + + // 1: AM Mode Choice utility for autos=adults auto + // sufficiency + luUtilities[2] = pkSovExpUtility * expConstants[2][0] + pkHovExpUtility + * expConstants[2][1] + pkWTExpUtility * expConstants[2][2] + + pkDTExpUtility * expConstants[2][4] + nmExpUtility + * expConstants[2][3]; + + // 3: MD Mode Choice utility for 0-autos auto sufficiency + luUtilities[3] = opSovExpUtility * expConstants[0][0] + opHovExpUtility + * expConstants[0][1] + opWTExpUtility * expConstants[0][2] + + opDTExpUtility * expConstants[0][4] + nmExpUtility + * expConstants[0][3]; + + // 4: MD Mode Choice utility for autos=adults auto + // sufficiency + luUtilities[5] = opSovExpUtility * expConstants[2][0] + opHovExpUtility + * expConstants[2][1] + opWTExpUtility * expConstants[2][2] + + opDTExpUtility * expConstants[2][4] + nmExpUtility + * expConstants[2][3]; + + // 6: AM Mode Choice utility for all households + luUtilities[6] = pkSovExpUtility * expConstants[3][0] + pkHovExpUtility + * expConstants[3][1] + pkWTExpUtility * expConstants[3][2] + + pkDTExpUtility * expConstants[3][4] + nmExpUtility + * expConstants[3][3]; + + // calculate non-mandatory destination choice logsums + + luDmu.setLogsums(luUtilities); + luDmu.setSizeTerms(luSizeTerms[jMgra]); + + if (trace) + { + String printString = new String(); + printString += (iMgra + "," + jMgra); + for (int j = 0; j < luUtilities.length; ++j) + { + printString += "," + String.format("%9.2f", luUtilities[j]); + } + logger.info(printString); + + accString = new String(); + accString = "Non-mandatory: iMgra, jMgra, Alternative, LU_Logsum, SizeTerm, LU_Accessibility\n"; + + } + // add accessibilities for origin mgra + for (int alt = 0; alt < luAlts; ++alt) + { + + double logsum = luDmu.getLogsum(alt + 1); + double sizeTerm = luDmu.getSizeTerm(alt + 1); + + luAccessibilities[iMgra][alt] += (logsum * sizeTerm); + + if (trace) + { + accString += iMgra + "," + alt + "," + logsum + "," + sizeTerm + "," + + luAccessibilities[iMgra][alt] + "\n"; + } + } + + // save calculated utilities in a table to return tot the + // calling method for accumulating + luUtilityResult[0] = iMgra; + luUtilityResult[1] = jMgra; + for (int k = 0; k < luUtilities.length; k++) + luUtilityResult[2 + k] = (float) luUtilities[k]; + + accumulateLandUseModeChoiceLogsums(luUtilityResult, + accumulatedLandUseLogsumsCount, accumulatedLandUseLogsums); + + } + + } // end for destinations + + if (trace) + { + logger.info(accString); + } + + // calculate the logsum + for (int alt = 0; alt < alts; ++alt) + { + if (accessibilities[iMgra][alt] > 0) + accessibilities[iMgra][alt] = (float) Math.log(accessibilities[iMgra][alt]); + } + accessibilities[iMgra][alts] = iMgra; + + if (calculateLuAccessibilities) + { + // calculate the land use accessibility logsums + for (int alt = 0; alt < luAlts; ++alt) + { + if (luAccessibilities[iMgra][alt] > 0) + luAccessibilities[iMgra][alt] = (float) Math + .log(luAccessibilities[iMgra][alt]); + + } + luAccessibilities[iMgra][luAlts] = iMgra; + } + + } + + List resultBundle = new ArrayList(7); + resultBundle.add(taskIndex); + resultBundle.add(startRange); + resultBundle.add(endRange); + resultBundle.add(accessibilities); + + if (calculateLuAccessibilities) + { + resultBundle.add(luAccessibilities); + resultBundle.add(accumulatedLandUseLogsums); + resultBundle.add(accumulatedLandUseLogsumsCount); + } else + { + resultBundle.add(null); + resultBundle.add(null); + resultBundle.add(null); + } + + // outStream.close(); + + return resultBundle; + + } + + // private void debugLandUseModeChoiceLogsums( int iMgra, int jMgra, int + // iLuz, int jLuz, float[] luUtilities ) { + // + // String record = ( iLuz + "," + jLuz + "," + iMgra + "," + jMgra + "," + + // luUtilities[0] ); + // // don't need to report the last logsum (not used for mode choice + // logsums) + // for( int j=1; j < luUtilities.length - 1; j++ ) + // record += ( "," + luUtilities[j] ); + // outStream.println ( record ); + // + // } + + private void accumulateLandUseModeChoiceLogsums(float[] luUtilitiesValues, + int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) + { + + float[] luUtilities = new float[DcUtilitiesTaskJppf.LU_LOGSUM_SEGMENTS.length]; + + int iMgra = (int) luUtilitiesValues[0]; + int jMgra = (int) luUtilitiesValues[1]; + + int iLuz = mgraManager.getMgraLuz(iMgra); + int jLuz = mgraManager.getMgraLuz(jMgra); + + for (int i = 0; i < luUtilities.length; i++) + luUtilities[i] = luUtilitiesValues[i + 2]; + + accumulatedLandUseLogsumsCount[iLuz][jLuz]++; + + accumulateSimple(iLuz, jLuz, luUtilities, accumulatedLandUseLogsumsCount, + accumulatedLandUseLogsums); + accumulateLogit(iLuz, jLuz, luUtilities, accumulatedLandUseLogsumsCount, + accumulatedLandUseLogsums); + + // if ( iLuz == DEBUG_ILUZ && jLuz == DEBUG_JLUZ ) + // debugLandUseModeChoiceLogsums( iMgra, jMgra, iLuz, jLuz, luUtilities + // ); + + } + + private void accumulateSimple(int iLuz, int jLuz, float[] luUtilities, + int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) + { + + // simple averaging uses accumulated logsum values + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][jLuz] += Math + .log(luUtilities[0]); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][jLuz] += Math + .log(luUtilities[1]); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][jLuz] += Math + .log(luUtilities[2]); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][jLuz] += Math + .log(luUtilities[3]); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][jLuz] += Math + .log(luUtilities[4]); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][jLuz] += Math + .log(luUtilities[5]); + + // calculate logsums from external LUZs to all destination LUZs if the + // origin LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[iLuz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[iLuz]) + { + + double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] + * BuildAccessibilities.TIME_COEFFICIENT); + + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][exLuz][jLuz] += Math + .log(luUtilities[0] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][exLuz][jLuz] += Math + .log(luUtilities[1] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][exLuz][jLuz] += Math + .log(luUtilities[2] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][exLuz][jLuz] += Math + .log(luUtilities[3] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][exLuz][jLuz] += Math + .log(luUtilities[4] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][exLuz][jLuz] += Math + .log(luUtilities[5] + additionalUtility); + + accumulatedLandUseLogsumsCount[exLuz][jLuz]++; + + } + + } + + // calculate logsums to external LUZs from all origin LUZs if the + // destination LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[jLuz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[jLuz]) + { + + double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] + * BuildAccessibilities.TIME_COEFFICIENT); + + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][exLuz] += Math + .log(luUtilities[0] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][exLuz] += Math + .log(luUtilities[1] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][exLuz] += Math + .log(luUtilities[2] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][exLuz] += Math + .log(luUtilities[3] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][exLuz] += Math + .log(luUtilities[4] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][exLuz] += Math + .log(luUtilities[5] + additionalUtility); + + accumulatedLandUseLogsumsCount[iLuz][exLuz]++; + + } + + } + + } + + private void accumulateLogit(int iLuz, int jLuz, float[] luUtilities, + int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) + { + + // logit averaging uses accumulated utility values + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][jLuz] += luUtilities[0]; + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][jLuz] += luUtilities[1]; + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][jLuz] += luUtilities[2]; + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][jLuz] += luUtilities[3]; + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][jLuz] += luUtilities[4]; + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][jLuz] += luUtilities[5]; + + // calculate logsums from external LUZs to all destination LUZs if the + // origin LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[iLuz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[iLuz]) + { + + double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] + * BuildAccessibilities.TIME_COEFFICIENT); + + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][exLuz][jLuz] += (luUtilities[0] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][exLuz][jLuz] += (luUtilities[1] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][exLuz][jLuz] += (luUtilities[2] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][exLuz][jLuz] += (luUtilities[3] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][exLuz][jLuz] += (luUtilities[4] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][exLuz][jLuz] += (luUtilities[5] + additionalUtility); + + accumulatedLandUseLogsumsCount[exLuz][jLuz]++; + + } + + } + + // calculate logsums to external LUZs from all origin LUZs if the + // destination LUZ is a cordon LUZ + if (externalLuzsForCordonLuz[jLuz] != null) + { + + for (int exLuz : externalLuzsForCordonLuz[jLuz]) + { + + double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] + * BuildAccessibilities.TIME_COEFFICIENT); + + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][exLuz] += (luUtilities[0] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][exLuz] += (luUtilities[1] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][exLuz] += (luUtilities[2] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][exLuz] += (luUtilities[3] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][exLuz] += (luUtilities[4] + additionalUtility); + accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][exLuz] += (luUtilities[5] + additionalUtility); + + accumulatedLandUseLogsumsCount[iLuz][exLuz]++; + + } + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java new file mode 100644 index 0000000..d28d05c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java @@ -0,0 +1,313 @@ +package org.sandag.abm.accessibilities; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import java.io.File; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; + +/** + * This class is used to return drive-transit-walk skim values for MGRA pairs + * associated with estimation data file records. + * + * @author Jim Hicks + * @version March, 2010 + */ +public class DriveTransitWalkSkimsCalculator + implements Serializable +{ + + private transient Logger primaryLogger; + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; + + private static final int ACCESS_TIME_INDEX = 0; + private static final int EGRESS_TIME_INDEX = 1; + private static final int NA = -999; + + private int maxDTWSkimSets = 5; + private int[] NUM_SKIMS; + private double[] defaultSkims; + + // declare UEC object + private UtilityExpressionCalculator driveWalkSkimUEC; + private IndexValues iv; + + // The simple auto skims UEC does not use any DMU variables + private TransitDriveAccessDMU dmu = new TransitDriveAccessDMU(); // DMU + // for + // this + // UEC + + private MgraDataManager mgraManager; + private int maxTap; + private String[] skimNames; + + + // skim values array for transit service type(local, premium), + // depart skim period(am, pm, op), + // and Tap-Tap pair. + private double[][][][][] storedDepartPeriodTapTapSkims; + + private BestTransitPathCalculator bestPathUEC; + + private MatrixDataServerIf ms; + + public DriveTransitWalkSkimsCalculator(HashMap rbMap) + { + mgraManager = MgraDataManager.getInstance(); + maxTap = mgraManager.getMaxTap(); + } + + public void setup(HashMap rbMap, Logger logger, + BestTransitPathCalculator myBestPathUEC) + { + + primaryLogger = logger; + + // set the best transit path utility UECs + bestPathUEC = myBestPathUEC; + + // Create the skim UECs + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.drive.transit.walk.data.page"); + int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.drive.transit.walk.skim.page"); + int dtwNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.drive.transit.walk.skims"); + String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.drive.transit.walk.uec.file")).toString(); + File uecFile = new File(uecFileName); + driveWalkSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); + + skimNames = driveWalkSkimUEC.getAlternativeNames(); + + //setup index values + iv = new IndexValues(); + + //setup default skim values + defaultSkims = new double[dtwNumSkims]; + for (int j = 0; j < dtwNumSkims; j++) { + defaultSkims[j] = NA; + } + + // point the stored Array of skims: by Prem or Local, DepartPeriod, O tap, D tap, skim values[] to a shared data store + StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxDTWSkimSets, NUM_PERIODS, maxTap ); + storedDepartPeriodTapTapSkims = storedDataObject.getStoredDtwDepartPeriodTapTapSkims(); + + } + + + + /** + * Return the array of drive-transit-walk skims for the ride mode, origin TAP, + * destination TAP, and departure time period. + * + * @param set for set source skims + * @param origTap best Origin TAP for the MGRA pair + * @param workTap best Destination TAP for the MGRA pair + * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 = + * OffPeak period + * @return Array of 55 skim values for the MGRA pair and departure period + */ + public double[] getDriveTransitWalkSkims(int set, double pDriveTime, double aWalkTime, int origTap, int destTap, + int departPeriod, boolean debug) + { + + dmu.setDriveTimeToTap(pDriveTime); + dmu.setMgraTapWalkTime(aWalkTime); + + iv.setOriginZone(origTap); + iv.setDestZone(destTap); + + // allocate space for the origin tap if it hasn't been allocated already + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) + { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; + } + + // if the destTap skims are not already stored, calculate them and store + // them + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) + { + dmu.setTOD(departPeriod); + dmu.setSet(set); + double[] results = driveWalkSkimUEC.solve(iv, dmu, null); + if (debug) + driveWalkSkimUEC.logAnswersArray(primaryLogger, "Drive-Walk Skims"); + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pDriveTime; + } + catch ( Exception e ) { + primaryLogger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pDriveTime=" + pDriveTime); + primaryLogger.error ("exception setting drive-transit-walk drive access time in stored array.", e); + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aWalkTime; + } + catch ( Exception e ) { + primaryLogger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aWalkTime=" + aWalkTime); + primaryLogger.error ("exception setting drive-transit-walk walk egress time in stored array.", e); + } + return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; + + + } + + + public double[] getNullTransitSkims() + { + return defaultSkims; + } + + /** + * Start the matrix server + * + * @param rb is a ResourceBundle for the properties file for this application + */ + private void startMatrixServer(ResourceBundle rb) + { + + primaryLogger.info(""); + primaryLogger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + primaryLogger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + primaryLogger.error(String + .format("exception caught running ctramp model components -- exiting."), e); + throw new RuntimeException(); + + } + + } + + /** + * log a report of the final skim values for the MGRA odt + * + * @param odt is an int[] with the first element the origin mgra and the second + * element the dest mgra and third element the departure period index + * @param bestTapPairs is an int[][] of TAP values with the first dimesion the + * ride mode and second dimension a 2 element array with best orig and + * dest TAP + * @param returnedSkims is a double[][] of skim values with the first dimesion + * the ride mode indices and second dimention the skim categories + */ + public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) + { + + Modes.TransitMode[] mode = Modes.TransitMode.values(); + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String separator = ""; + String header = ""; + + primaryLogger.info(""); + primaryLogger.info(""); + header = "Returned drive-transit-walk skim value tables for origMgra=" + odt[0] + + ", destMgra=" + odt[1] + ", period index=" + odt[2] + ", period label=" + + PERIODS[odt[2]]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + primaryLogger.info(separator); + primaryLogger.info(header); + primaryLogger.info(""); + + String modeHeading = String.format("%-12s %3s ", "RideMode:", mode[0]); + for (int i = 1; i < bestTapPairs.length; i++) + modeHeading += String.format(" %3s ", mode[i]); + primaryLogger.info(modeHeading); + + String[] logValues = {"NA", "NA"}; + if (bestTapPairs[0] != null) + { + logValues[0] = String.valueOf(bestTapPairs[0][0]); + logValues[1] = String.valueOf(bestTapPairs[0][1]); + } + String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", logValues[0], + logValues[1]); + + for (int i = 1; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] != null) + { + logValues[0] = String.valueOf(bestTapPairs[i][0]); + logValues[1] = String.valueOf(bestTapPairs[i][1]); + } else + { + logValues[0] = "NA"; + logValues[1] = "NA"; + } + tapHeading += String.format(" %4s-%4s ", logValues[0], logValues[1]); + } + primaryLogger.info(tapHeading); + + String underLine = String.format("%-12s %9s ", "---------", "---------"); + for (int i = 1; i < bestTapPairs.length; i++) + underLine += String.format(" %9s ", "---------"); + primaryLogger.info(underLine); + + for (int j = 0; j < ncols; j++) + { + String tableRecord = ""; + if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, + skims[0][j]); + else tableRecord = String.format("%-12d %12s ", j + 1, ""); + for (int i = 1; i < bestTapPairs.length; i++) + { + if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); + else tableRecord += String.format(" %12s ", ""); + } + primaryLogger.info(tableRecord); + } + + primaryLogger.info(""); + primaryLogger.info(separator); + } + + public String[] getSkimNames() { + return skimNames; + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java new file mode 100644 index 0000000..c158ad1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java @@ -0,0 +1,593 @@ +package org.sandag.abm.accessibilities; + +import com.pb.common.util.Tracer; +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import java.io.File; +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; +import org.sandag.abm.modechoice.Modes; + +/** + * This class builds accessibility components for all modes. + * + * @author Joel Freedman + * @version May, 2009 + */ +public class MandatoryAccessibilitiesCalculator + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(MandatoryAccessibilitiesCalculator.class); + + private static final int MIN_EXP_FUNCTION_ARGUMENT = -500; + + private static final int PEAK_NONTOLL_SOV_TIME_INDEX = 0; + private static final int PEAK_NONTOLL_SOV_DIST_INDEX = 1; + private static final int OFFPEAK_NONTOLL_SOV_TIME_INDEX = 2; + private static final int OFFPEAK_NONTOLL_SOV_DIST_INDEX = 3; + + private UtilityExpressionCalculator autoSkimUEC; + private UtilityExpressionCalculator bestWalkTransitUEC; + private UtilityExpressionCalculator bestDriveTransitUEC; + private UtilityExpressionCalculator autoLogsumUEC; + private UtilityExpressionCalculator transitLogsumUEC; + + private MandatoryAccessibilitiesDMU dmu; + private IndexValues iv; + + private NonTransitUtilities ntUtilities; + + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + + // auto sufficiency (0 autos, autos=adults), + // and mode (SOV,HOV,Walk-Transit,Non-Motorized) + private double[][] expConstants; + + private String[] accNames = { + "SovTime", // 0 + "SovDist", // 1 + "WTTime", // 2 + "DTTime", // 3 + "SovUtility", // 4 + "WTUtility", // 5 + "AutoLogsum", // 6 + "WTLogsum", // 7 + "TransitLogsum", // 8 + "WTRailShare", // 9 + "DTRailShare", // 10 + "DTLogsum", // 11 + "HovUtility" // 12 + }; + + private BestTransitPathCalculator bestPathCalculator; + + + public MandatoryAccessibilitiesCalculator(HashMap rbMap, + NonTransitUtilities aNtUtilities, double[][] aExpConstants, BestTransitPathCalculator myBestPathCalculator) + { + + ntUtilities = aNtUtilities; + expConstants = aExpConstants; + + // Create the UECs + String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.mandatory.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.data.page"); + int autoSkimPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.auto.page"); + int bestWalkTransitPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.bestWalkTransit.page"); + int bestDriveTransitPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.bestDriveTransit.page"); + int autoLogsumPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.autoLogsum.page"); + int transitLogsumPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.transitLogsum.page"); + + + dmu = new MandatoryAccessibilitiesDMU(); + + File uecFile = new File(uecFileName); + autoSkimUEC = new UtilityExpressionCalculator(uecFile, autoSkimPage, dataPage, rbMap, dmu); + bestWalkTransitUEC = new UtilityExpressionCalculator(uecFile, bestWalkTransitPage, dataPage, rbMap, dmu); + bestDriveTransitUEC = new UtilityExpressionCalculator(uecFile, bestDriveTransitPage, dataPage, rbMap, dmu); + autoLogsumUEC = new UtilityExpressionCalculator(uecFile, autoLogsumPage, dataPage, rbMap, dmu); + transitLogsumUEC = new UtilityExpressionCalculator(uecFile, transitLogsumPage, dataPage, rbMap, dmu); + + iv = new IndexValues(); + + tazManager = TazDataManager.getInstance(); + tapManager = TapDataManager.getInstance(); + mgraManager = MgraDataManager.getInstance(); + + bestPathCalculator = myBestPathCalculator; + } + + + public double[] calculateWorkerMandatoryAccessibilities(int hhMgra, int workMgra) + { + return calculateAccessibilitiesForMgraPair(hhMgra, workMgra, false, null); + } + + public double[] calculateStudentMandatoryAccessibilities(int hhMgra, int schoolMgra) + { + return calculateAccessibilitiesForMgraPair(hhMgra, schoolMgra, false, null); + } + + /** + * Calculate the work logsum for the household MGRA and sampled work location + * MGRA. + * + * @param hhMgra Household MGRA + * @param workMgra Sampled work MGRA + * @param autoSufficiency Auto sufficiency category + * @return Work mode choice logsum + */ + public double calculateWorkLogsum(int hhMgra, int workMgra, int autoSufficiency, boolean debug, Logger aLogger) + { + + String separator = ""; + String header = ""; + if (debug) + { + aLogger.info(""); + aLogger.info(""); + header = "calculateWorkLogsum() debug info for homeMgra=" + hhMgra + ", workMgra=" + workMgra; + for (int i = 0; i < header.length(); i++) + separator += "^"; + } + + double[] accessibilities = calculateAccessibilitiesForMgraPair(hhMgra, workMgra, debug, aLogger); + + double sovUtility = accessibilities[4]; + double hovUtility = accessibilities[12]; + double transitLogsum = accessibilities[8]; // includes both walk and drive access + double nmExpUtility = ntUtilities.getNMotorExpUtility(hhMgra, workMgra, NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + + // constrain auto sufficiency to 0,1,2 + autoSufficiency = Math.min(autoSufficiency, 2); + + double utilSum = Math.exp(sovUtility) * expConstants[autoSufficiency][0] + + Math.exp(hovUtility) * expConstants[autoSufficiency][1] + + Math.exp(transitLogsum) * expConstants[autoSufficiency][2] + + nmExpUtility * expConstants[autoSufficiency][3]; + + double logsum = Math.log(utilSum); + + if (debug) + { + + aLogger.info(separator); + aLogger.info(header); + aLogger.info(separator); + + aLogger.info("accessibilities array values"); + aLogger.info(String.format("%5s %15s %15s", "i", "accName", "value")); + aLogger.info(String.format("%5s %15s %15s", "-----", "----------", "----------")); + for (int i = 0; i < accessibilities.length; i++) + { + aLogger.info(String.format("%5d %15s %15.5e", i, accNames[i], accessibilities[i])); + } + + aLogger.info(""); + aLogger.info(""); + aLogger.info("logsum component values"); + aLogger.info(String.format("autoSufficiency = %d", autoSufficiency)); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "sovUtility", sovUtility, "exp(sovUtility)", Math.exp(sovUtility), String + .format("expConst suff=%d 0", autoSufficiency), + expConstants[autoSufficiency][0])); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "hovUtility", hovUtility, "exp(hovUtility)", Math.exp(hovUtility), String + .format("expConst suff=%d 1", autoSufficiency), + expConstants[autoSufficiency][1])); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "transitLogsum", transitLogsum, "exp(transitLogsum)", Math.exp(transitLogsum), + String.format("expConst suff=%d 2", autoSufficiency), + expConstants[autoSufficiency][2])); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e", "nmExpUtility", + nmExpUtility, String.format("expConst suff=%d 3", autoSufficiency), + expConstants[autoSufficiency][3])); + aLogger.info(String.format("%-15s = %15.5e", "utilSum", utilSum)); + aLogger.info(String.format("%-15s = %15.5e", "logsum", logsum)); + aLogger.info(separator); + } + + return logsum; + } + + /** + * Calculate the school logsum for the household MGRA and sampled school location + * MGRA. + * + * @param hhMgra Household MGRA + * @param schoolMgra Sampled work MGRA + * @param autoSufficiency Auto sufficiency category + * @param studentType Student type 0=Pre-school (SOV not available) 1=K-8 (SOV + * not available) 2=9-12 (Normal car-sufficiency-based logsum) + * 3=College/university(typical) (Normal car-sufficiency-based logsum) + * 4=College/university(non-typical) (Normal car-sufficiency-based + * logsum) + * @return School mode choice logsum + */ + public double calculateSchoolLogsum(int hhMgra, int schoolMgra, int autoSufficiency, int studentType, boolean debug, Logger aLogger) + { + + String separator = ""; + String header = ""; + if (debug) + { + aLogger.info(""); + aLogger.info(""); + header = "calculateSchoolLogsum() debug info for homeMgra=" + hhMgra + ", schoolMgra=" + + schoolMgra; + for (int i = 0; i < header.length(); i++) + separator += "^"; + } + + double[] accessibilities = calculateAccessibilitiesForMgraPair(hhMgra, schoolMgra, debug, aLogger); + + double sovUtility = accessibilities[4]; + double hovUtility = accessibilities[12]; + double transitLogsum = accessibilities[8]; // includes both walk and drive access + double nmExpUtility = ntUtilities.getNMotorExpUtility(hhMgra, schoolMgra, NonTransitUtilities.OFFPEAK_PERIOD_INDEX); + + // constrain auto sufficiency to 0,1,2 + autoSufficiency = Math.min(autoSufficiency, 2); + + double logsum = Math.exp(hovUtility) * expConstants[autoSufficiency][1] + + Math.exp(transitLogsum) * expConstants[autoSufficiency][2] + + nmExpUtility * expConstants[autoSufficiency][3]; + + // used for debugging + double logsum1 = logsum; + + if (studentType >= 2) + { + logsum = logsum + Math.exp(sovUtility) * expConstants[autoSufficiency][0]; + } + + // used for debugging + double logsum2 = logsum; + + logsum = Math.log(logsum); + + if (debug) + { + + aLogger.info(separator); + aLogger.info(header); + aLogger.info(separator); + + aLogger.info("accessibilities array values"); + aLogger.info(String.format("%5s %15s %15s", "i", "accName", "value")); + aLogger.info(String.format("%5s %15s %15s", "-----", "----------", "----------")); + for (int i = 0; i < accessibilities.length; i++) + { + aLogger.info(String.format("%5d %15s %15.5e", i, accNames[i], accessibilities[i])); + } + + aLogger.info(""); + aLogger.info(""); + aLogger.info("logsum component values"); + aLogger.info(String.format("autoSufficiency = %d", autoSufficiency)); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "hovUtility", hovUtility, "exp(hovUtility)", Math.exp(hovUtility), + String.format("expConst suff=%d 1", autoSufficiency), + expConstants[autoSufficiency][1])); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "transitLogsum", transitLogsum, "exp(transitLogsum)", Math.exp(transitLogsum), + String.format("expConst suff=%d 2", autoSufficiency), + expConstants[autoSufficiency][2])); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e", "nmExpUtility", + nmExpUtility, String.format("expConst suff=%d 3", autoSufficiency), + expConstants[autoSufficiency][3])); + aLogger.info(String.format("%s = %15.5e", "utility sum (before adding sovUtility)", + logsum1)); + if (studentType >= 2) + { + aLogger.info(String.format("studentType = %d", studentType)); + aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", + "sovUtility", sovUtility, "exp(sovUtility)", Math.exp(sovUtility), String.format("expConst suff=%d 0", autoSufficiency), + expConstants[autoSufficiency][0])); + aLogger.info(String.format("%s = %15.5e", "utility sum (after adding sovUtility)", logsum2)); + } else + { + aLogger.info(String.format( + "studentType = %d, no additional contribution to utility sum", + studentType)); + } + aLogger.info(String.format("%s = %15.5e, %s = %15.5e", "final utility sum", logsum2, + "final logsum", logsum)); + aLogger.info(separator); + } + + return logsum; + } + + /** + * Calculate the accessibilities for a given origin and destination mgra + * + * @param oMgra The origin mgra + * @param dMgra The destination mgra + * @return An array of accessibilities + */ + public double[] calculateAccessibilitiesForMgraPair(int oMgra, int dMgra, boolean debug, Logger aLogger) + { + + double[] accessibilities = new double[accNames.length]; + + // DMUs for this UEC + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + if (oMgra > 0 && dMgra > 0) + { + + int oTaz = mgraManager.getTaz(oMgra); + int dTaz = mgraManager.getTaz(dMgra); + + iv.setOriginZone(oTaz); + iv.setDestZone(dTaz); + + // sov time and distance + double[] autoResults = autoSkimUEC.solve(iv, dmu, null); + if (debug) + autoSkimUEC.logAnswersArray( aLogger, String.format( "autoSkimUEC: oMgra=%d, dMgra=%d", oMgra, dMgra ) ); + + // autoResults[0] is peak non-toll sov time, autoResults[1] is peak non-toll sov dist + // autoResults[2] is off-peak non-toll sov time, autoResults[3] is off-peak non-toll sov dist + accessibilities[0] = autoResults[PEAK_NONTOLL_SOV_TIME_INDEX]; + accessibilities[1] = autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]; + + // pre-calculate the hov, sov, and non-motorized exponentiated utilities for the origin MGRA. + // the method called returns cached values if they were already calculated. + ntUtilities.buildUtilitiesForOrigMgraAndPeriod( oMgra, NonTransitUtilities.PEAK_PERIOD_INDEX ); + + // auto logsum + double pkSovExpUtility = ntUtilities.getSovExpUtility(oTaz, dTaz, NonTransitUtilities.PEAK_PERIOD_INDEX); + double pkHovExpUtility = ntUtilities.getHovExpUtility(oTaz, dTaz, NonTransitUtilities.PEAK_PERIOD_INDEX); + + dmu.setSovNestLogsum(-999); + if (pkSovExpUtility > 0) + { + dmu.setSovNestLogsum(Math.log(pkSovExpUtility)); + accessibilities[4] = dmu.getSovNestLogsum(); + } + dmu.setHovNestLogsum(-999); + if (pkHovExpUtility > 0) + { + dmu.setHovNestLogsum(Math.log(pkHovExpUtility)); + accessibilities[12] = dmu.getHovNestLogsum(); + } + + double[] autoLogsum = autoLogsumUEC.solve(iv, dmu, null); + if (debug) + autoLogsumUEC.logAnswersArray(aLogger, String.format( + "autoLogsumUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); + accessibilities[6] = autoLogsum[0]; + + + ////////////////////////////////////////////////////////////////////////// + // walk transit + ////////////////////////////////////////////////////////////////////////// + + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, oMgra, dMgra, debug, aLogger); + + // sum the exponentiated utilities over modes + double sumWlkExpUtilities = bestPathCalculator.getSumExpUtilities(); + double[] walkTransitWalkUtilities = bestPathCalculator.getBestUtilities(); + + // calculate ln( sum of exponentiated utilities ) and set in accessibilities array and the dmu object + if (sumWlkExpUtilities > 0) + accessibilities[7] = Math.log(sumWlkExpUtilities); + else + accessibilities[7] = -999; + + dmu.setWlkNestLogsum(accessibilities[7]); + + + int bestAlt = bestPathCalculator.getBestTransitAlt(); + + if (bestAlt >= 0) + { + double[] bestTaps = bestPathCalculator.getBestTaps(bestAlt); + int oTapPosition = mgraManager.getTapPosition(oMgra, (int)bestTaps[0]); + int dTapPosition = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); + int set = (int)bestTaps[2]; + + if (oTapPosition == -1 || dTapPosition == -1) + { + logger.fatal("Error: Best walk transit alt " + bestAlt + " found for origin mgra " + + oMgra + " to destination mgra " + dMgra + " but oTap pos " + + oTapPosition + " and dTap pos " + dTapPosition); + throw new RuntimeException(); + } + + if (walkTransitWalkUtilities[bestAlt] <= MIN_EXP_FUNCTION_ARGUMENT) + { + logger.fatal("Error: Best walk transit alt " + bestAlt + " found for origin mgra " + + oMgra + " to destination mgra " + dMgra + " but Utility = " + + walkTransitWalkUtilities[bestAlt]); + throw new RuntimeException(); + } + accessibilities[5] = Math.log(walkTransitWalkUtilities[bestAlt]); + + //set access and egress times + int oPos = mgraManager.getTapPosition(oMgra, (int)bestTaps[0]); + float mgraTapWalkTime = mgraManager.getMgraToTapWalkTime(oMgra, oPos); + dmu.setMgraTapWalkTime(mgraTapWalkTime); + + int dPos = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); + float tapMgraWalkTime = mgraManager.getMgraToTapWalkTime(dMgra, dPos); + dmu.setTapMgraWalkTime(tapMgraWalkTime); + + dmu.setBestSet(set); + iv.setOriginZone((int)bestTaps[0]); + iv.setDestZone((int)bestTaps[1]); + double[] wlkTransitTimes = bestWalkTransitUEC.solve(iv, dmu, null); + + if (debug){ + bestWalkTransitUEC.logAnswersArray(aLogger, String.format("bestWalkTransitUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); + } + + accessibilities[2] = wlkTransitTimes[0]; + accessibilities[9] = wlkTransitTimes[1]; + + } + + ////////////////////////////////////////////////////////////////////////// + // drive transit + ////////////////////////////////////////////////////////////////////////// + + // determine the best transit path, which also stores the best utilities array and the best mode + bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, oMgra, dMgra, debug, aLogger, (float) autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]); + + // sum the exponentiated utilities over modes + double sumDrvExpUtilities = 0; + double[] driveTransitWalkUtilities = bestPathCalculator.getBestUtilities(); + for (int i=0; i < driveTransitWalkUtilities.length; i++){ + if ( driveTransitWalkUtilities[i] > MIN_EXP_FUNCTION_ARGUMENT ) + sumDrvExpUtilities += Math.exp(driveTransitWalkUtilities[i]); + } + + + // calculate ln( sum of exponentiated utilities ) and set in accessibilities array and the dmu object + if (sumDrvExpUtilities > 0) + accessibilities[11] = Math.log(sumDrvExpUtilities); + else + accessibilities[11] = -999; + + dmu.setDrvNestLogsum(accessibilities[11]); + + + bestAlt = bestPathCalculator.getBestTransitAlt(); + + if (bestAlt >= 0) + { + double[] bestTaps = bestPathCalculator.getBestTaps(bestAlt); + int oTapPosition = tazManager.getTapPosition(oTaz, (int)bestTaps[0], Modes.AccessMode.PARK_N_RIDE); + int dTapPosition = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); + int set = (int)bestTaps[2]; + + if (oTapPosition == -1 || dTapPosition == -1) + { + logger.fatal("Error: Best drive transit alt " + bestAlt + " found for origin mgra " + + oMgra + " to destination mgra " + dMgra + " but oTap pos " + + oTapPosition + " and dTap pos " + dTapPosition); + throw new RuntimeException(); + } + + if (driveTransitWalkUtilities[bestAlt] <= MIN_EXP_FUNCTION_ARGUMENT) + { + logger.fatal("Error: Best drive transit alt " + bestAlt + " found for origin mgra " + + oMgra + " to destination mgra " + dMgra + " but Utility = " + + driveTransitWalkUtilities[bestAlt]); + throw new RuntimeException(); + } + + //set access and egress times + dmu.setDriveTimeToTap(tazManager.getTapTime(oTaz, oTapPosition, Modes.AccessMode.PARK_N_RIDE)); + dmu.setDriveDistToTap(tazManager.getTapDist(oTaz, oTapPosition, Modes.AccessMode.PARK_N_RIDE)); + + int dPos = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); + float tapMgraWalkTime = mgraManager.getMgraToTapWalkTime(dMgra, dPos); + dmu.setTapMgraWalkTime(tapMgraWalkTime); + + dmu.setBestSet(set); + iv.setOriginZone((int)bestTaps[0]); + iv.setDestZone((int)bestTaps[1]); + double[] drvTransitTimes = bestDriveTransitUEC.solve(iv, dmu, null); + + if (debug){ + bestDriveTransitUEC.logAnswersArray(aLogger, String.format("bestDriveTransitUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); + } + + accessibilities[3] = drvTransitTimes[0]; + accessibilities[10] = drvTransitTimes[1]; + + } + + + double[] transitLogsumResults = transitLogsumUEC.solve(iv, dmu, null); + if (debug){ + transitLogsumUEC.logAnswersArray(aLogger, String.format("transitLogsumUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); + } + + // transit logsum results array has only 1 alternative, so result is in 0 element. + accessibilities[8] = transitLogsumResults[0]; + + } // end if oMgra and dMgra > 0 + + return accessibilities; + } + + /** + * Calculate auto skims for a given origin to all destination mgras, and return + * auto distance. + * + * @param oMgra The origin mgra + * @return An array of distances + */ + public double[] calculateDistancesForAllMgras(int oMgra) + { + + double[] distances = new double[mgraManager.getMaxMgra() + 1]; + + int oTaz = mgraManager.getTaz(oMgra); + iv.setOriginZone(oTaz); + + for (int i = 0; i < mgraManager.getMgras().size(); i++) + { + + int dTaz = mgraManager.getTaz(mgraManager.getMgras().get(i)); + iv.setDestZone(dTaz); + + // sov distance + double[] autoResults = autoSkimUEC.solve(iv, dmu, null); + distances[dTaz] = autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]; + + } + + return distances; + } + + /** + * Calculate auto skims for a given origin to all destination mgras, and return + * auto distance. + * + * @param oMgra The origin mgra + * @return An array of distances + */ + public double[] calculateOffPeakDistancesForAllMgras(int oMgra) + { + + double[] distances = new double[mgraManager.getMaxMgra() + 1]; + + int oTaz = mgraManager.getTaz(oMgra); + iv.setOriginZone(oTaz); + + for (int i = 0; i < mgraManager.getMgras().size(); i++) + { + + int dTaz = mgraManager.getTaz(mgraManager.getMgras().get(i)); + iv.setDestZone(dTaz); + + // sov distance + double[] autoResults = autoSkimUEC.solve(iv, dmu, null); + distances[dTaz] = autoResults[OFFPEAK_NONTOLL_SOV_DIST_INDEX]; + + } + + return distances; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java new file mode 100644 index 0000000..cc68cf4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java @@ -0,0 +1,206 @@ +package org.sandag.abm.accessibilities; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +public class MandatoryAccessibilitiesDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(MandatoryAccessibilitiesDMU.class); + + protected HashMap methodIndexMap; + + protected double sovNestLogsum; + protected double hovNestLogsum; + protected double wlkNestLogsum; + protected double drvNestLogsum; + protected int bestSet; + protected float mgraTapWalkTime; + protected float tapMgraWalkTime; + protected float driveDistToTap; + protected float driveTimeToTap; + protected int autoSufficiency; + + public MandatoryAccessibilitiesDMU() + { + setupMethodIndexMap(); + } + + public int getAutoSufficiency() + { + return autoSufficiency; + } + + public void setAutoSufficiency(int autoSufficiency) + { + this.autoSufficiency = autoSufficiency; + } + + public double getSovNestLogsum() + { + return sovNestLogsum; + } + + public void setSovNestLogsum(double sovNestLogsum) + { + this.sovNestLogsum = sovNestLogsum; + } + + public double getHovNestLogsum() + { + return hovNestLogsum; + } + + public void setHovNestLogsum(double hovNestLogsum) + { + this.hovNestLogsum = hovNestLogsum; + } + + public double getWlkNestLogsum() + { + return wlkNestLogsum; + } + + public void setWlkNestLogsum(double wlkNestLogsum) + { + this.wlkNestLogsum = wlkNestLogsum; + } + + public double getDrvNestLogsum() + { + return drvNestLogsum; + } + + public void setDrvNestLogsum(double drvNestLogsum) + { + this.drvNestLogsum = drvNestLogsum; + } + + public int getBestSet() + { + return bestSet; + } + + public void setBestSet(int bestSet) + { + this.bestSet = bestSet; + } + + public float getMgraTapWalkTime() + { + return mgraTapWalkTime; + } + + public void setMgraTapWalkTime(float mgraTapWalkTime) + { + this.mgraTapWalkTime = mgraTapWalkTime; + } + + public float getTapMgraWalkTime() + { + return tapMgraWalkTime; + } + + public void setTapMgraWalkTime(float tapMgraWalkTime) + { + this.tapMgraWalkTime = tapMgraWalkTime; + } + + public float getDriveDistToTap() + { + return driveDistToTap; + } + + public void setDriveDistToTap(float drvDistToTap) + { + this.driveDistToTap = drvDistToTap; + } + + public float getDriveTimeToTap() + { + return driveTimeToTap; + } + + public void setDriveTimeToTap(float drvTimeToTap) + { + this.driveTimeToTap = drvTimeToTap; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getBestSet", 0); + methodIndexMap.put("getDriveDistToTap", 1); + methodIndexMap.put("getDriveTimeToTap", 2); + methodIndexMap.put("getWlkNestLogsum", 3); + methodIndexMap.put("getDrvNestLogsum", 4); + methodIndexMap.put("getSovNestLogsum", 5); + methodIndexMap.put("getHovNestLogsum", 6); + methodIndexMap.put("getMgraTapWalkTime", 7); + methodIndexMap.put("getTapMgraWalkTime", 8); + methodIndexMap.put("getAutoSufficiency", 9); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getBestSet(); + case 1: + return getDriveDistToTap(); + case 2: + return getDriveTimeToTap(); + case 3: + return getWlkNestLogsum(); + case 4: + return getDrvNestLogsum(); + case 5: + return getSovNestLogsum(); + case 6: + return getHovNestLogsum(); + case 7: + return getMgraTapWalkTime(); + case 8: + return getTapMgraWalkTime(); + case 9: + return getAutoSufficiency(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java new file mode 100644 index 0000000..f2af9c9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java @@ -0,0 +1,980 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagAppendMcLogsumDMU; +import org.sandag.abm.application.SandagTripModeChoiceDMU; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class McLogsumsAppender + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(McLogsumsAppender.class); + protected transient Logger slcSoaLogger = Logger.getLogger("slcSoa"); + protected transient Logger nonManLogsumsLogger = Logger.getLogger("nonManLogsums"); + + protected int debugEstimationFileRecord1; + protected int debugEstimationFileRecord2; + + protected static final String ESTIMATION_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; + public static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; + public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; + + public static final int WORK_SHEET = 1; + public static final int UNIVERSITY_SHEET = 2; + public static final int SCHOOL_SHEET = 3; + public static final int MAINTENANCE_SHEET = 4; + public static final int DISCRETIONARY_SHEET = 5; + public static final int SUBTOUR_SHEET = 6; + public static final int[] MC_PURPOSE_SHEET_INDICES = {-1, WORK_SHEET, + UNIVERSITY_SHEET, SCHOOL_SHEET, MAINTENANCE_SHEET, MAINTENANCE_SHEET, + MAINTENANCE_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, + SUBTOUR_SHEET }; + + public static final int WORK_CATEGORY = 0; + public static final int UNIVERSITY_CATEGORY = 1; + public static final int SCHOOL_CATEGORY = 2; + public static final int MAINTENANCE_CATEGORY = 3; + public static final int DISCRETIONARY_CATEGORY = 4; + public static final int SUBTOUR_CATEGORY = 5; + public static final String[] PURPOSE_CATEGORY_LABELS = {"work", "university", + "school", "maintenance", "discretionary", "subtour" }; + public static final int[] PURPOSE_CATEGORIES = {-1, WORK_CATEGORY, + UNIVERSITY_CATEGORY, SCHOOL_CATEGORY, MAINTENANCE_CATEGORY, MAINTENANCE_CATEGORY, + MAINTENANCE_CATEGORY, DISCRETIONARY_CATEGORY, DISCRETIONARY_CATEGORY, + DISCRETIONARY_CATEGORY, SUBTOUR_CATEGORY }; + + protected static final int ORIG_MGRA = 1; + protected static final int DEST_MGRA = 2; + protected static final int ADULTS = 3; + protected static final int AUTOS = 4; + protected static final int HHSIZE = 5; + protected static final int FEMALE = 6; + protected static final int AGE = 7; + protected static final int JOINT = 8; + protected static final int PARTYSIZE = 9; + protected static final int TOUR_PURPOSE = 10; + protected static final int INCOME = 11; + protected static final int ESCORT = 12; + protected static final int DEPART_PERIOD = 13; + protected static final int ARRIVE_PERIOD = 14; + protected static final int SAMPNO = 15; + protected static final int WORK_TOUR_MODE = 16; + protected static final int OUT_STOPS = 17; + protected static final int IN_STOPS = 18; + protected static final int FIRST_TRIP = 19; + protected static final int LAST_TRIP = 20; + protected static final int TOUR_MODE = 21; + protected static final int TRIP_PERIOD = 22; + protected static final int CHOSEN_MGRA = 23; + protected static final int DIRECTION = 24; + protected static final int PORTION = 25; + protected static final int PERNO = 26; + protected static final int TOUR_ID = 27; + protected static final int TRIPNO = 28; + protected static final int STOPID = 29; + protected static final int STOPNO = 30; + protected static final int STOP_PURPOSE = 31; + protected static final int NUM_FIELDS = 32; + + private static final int INBOUND_DIRCETION_CODE = 2; + + // estimation file defines time periods as: + // 1 | Early AM: 3:00 AM - 5:59 AM | + // 2 | AM Peak: 6:00 AM - 8:59 AM | + // 3 | Early MD: 9:00 AM - 11:59 PM | + // 4 | Late MD: 12:00 PM - 3:29 PM | + // 5 | PM Peak: 3:30 PM - 6:59 PM | + // 6 | Evening: 7:00 PM - 2:59 AM | + + protected static final int LAST_EA_INDEX = 3; + protected static final int LAST_AM_INDEX = 9; + protected static final int LAST_MD_INDEX = 22; + protected static final int LAST_PM_INDEX = 29; + + protected static final int EA = 1; + protected static final int AM = 2; + protected static final int MD = 3; + protected static final int PM = 4; + protected static final int EV = 5; + + protected static final int EA_D = 1; // 5am + protected static final int AM_D = 5; // 7am + protected static final int MD_D = 15; // 12pm + protected static final int PM_D = 27; // 6pm + protected static final int EV_D = 35; // 10pm + protected static final int[] DEFAULT_DEPART_INDICES = {-1, EA_D, AM_D, MD_D, + PM_D, EV_D }; + + protected static final int EA_A = 2; // 5:30am + protected static final int AM_A = 6; // 7:30am + protected static final int MD_A = 16; // 12:30pm + protected static final int PM_A = 28; // 6:30pm + protected static final int EV_A = 36; // 10:30pm + protected static final int[] DEFAULT_ARRIVE_INDICES = {-1, EA_A, AM_A, MD_A, + PM_A, EV_A }; + + protected String[][] departArriveCombinationLabels = { {"EA", "EA"}, + {"EA", "AM"}, {"EA", "MD"}, {"EA", "PM"}, {"EA", "EV"}, {"AM", "AM"}, {"AM", "MD"}, + {"AM", "PM"}, {"AM", "EV"}, {"MD", "MD"}, {"MD", "PM"}, {"MD", "EV"}, {"PM", "PM"}, + {"PM", "EV"}, {"EV", "EV"} }; + + protected int[][] departArriveCombinations = { {EA, EA}, {EA, AM}, + {EA, MD}, {EA, PM}, {EA, EV}, {AM, AM}, {AM, MD}, {AM, PM}, {AM, EV}, {MD, MD}, + {MD, PM}, {MD, EV}, {PM, PM}, {PM, EV}, {EV, EV} }; + + private BestTransitPathCalculator bestPathUEC; + + // modeChoiceLogsums is an array of logsums for each unique depart/arrive + // skim + // period combination, for each sample destination + protected double[][] modeChoiceLogsums; + protected double[][] tripModeChoiceLogsums; + + protected double[] tripModeChoiceSegmentLogsums = new double[2]; + + protected double[] tripModeChoiceSegmentStoredProbabilities; + + // departArriveLogsums is the array of values for all 15 depart/arrive + // combinations + protected double[][] departArriveLogsums; + + protected int chosenLogsumTodIndex = 0; + + protected int[] chosenDepartArriveCombination = new int[2]; + + protected int numMgraFields; + protected int[] mgraSetForLogsums; + + protected int[][] mgras; + + protected TazDataManager tazs; + protected MgraDataManager mgraManager; + protected TapDataManager tapManager; + protected MatrixDataServerIf ms; + + protected ModelStructure modelStructure; + + protected int[][] bestTapPairs; + protected double[] nmSkimsOut; + protected double[] nmSkimsIn; + protected double[] lbSkimsOut; + protected double[] lbSkimsIn; + protected double[] ebSkimsOut; + protected double[] ebSkimsIn; + protected double[] brSkimsOut; + protected double[] brSkimsIn; + protected double[] lrSkimsOut; + protected double[] lrSkimsIn; + protected double[] crSkimsOut; + protected double[] crSkimsIn; + + protected double[] lsWgtAvgCostM; + protected double[] lsWgtAvgCostD; + protected double[] lsWgtAvgCostH; + + protected int totalTime1 = 0; + protected int totalTime2 = 0; + + private McLogsumsCalculator logsumHelper; + + public McLogsumsAppender(HashMap propertyMap) + { + logsumHelper = new McLogsumsCalculator(); + + logsumHelper.setupSkimCalculators(propertyMap); + + } + + public BestTransitPathCalculator getBestTransitPathCalculator() + { + return bestPathUEC; + } + + protected TableDataSet getEstimationDataTableDataSet(HashMap rbMap) + { + + String estFileName = Util.getStringValueFromPropertyMap(rbMap, + ESTIMATION_DATA_RECORDS_FILE_KEY); + if (estFileName == null) + { + logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); + logger.error("Properties file target: " + ESTIMATION_DATA_RECORDS_FILE_KEY + + " not found."); + logger.error("Please specify a filename value for the " + + ESTIMATION_DATA_RECORDS_FILE_KEY + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFile(new File(estFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", + estFileName)); + throw new RuntimeException(e); + } + + } + + protected void calculateModeChoiceLogsums(HashMap rbMap, + ChoiceModelApplication mcModel, SandagAppendMcLogsumDMU mcDmuObject, int[] odt, + int[] mgraSample, int[] departAvailable, int[] arriveAvailable, boolean chosenTodOnly) + { + + boolean isChosenDepartArriveCombo = false; + + int origMgra = odt[ORIG_MGRA]; + int chosenMgra = odt[DEST_MGRA]; + + int inc = 0; + if (odt[INCOME] == 1) inc = 7500; + else if (odt[INCOME] == 2) inc = 22500; + else if (odt[INCOME] == 3) inc = 37500; + else if (odt[INCOME] == 4) inc = 52500; + else if (odt[INCOME] == 5) inc = 67500; + else if (odt[INCOME] == 6) inc = 87500; + else if (odt[INCOME] == 7) inc = 112500; + else if (odt[INCOME] == 8) inc = 137500; + else if (odt[INCOME] == 9) inc = 175000; + else if (odt[INCOME] == 10) inc = 250000; + else if (odt[INCOME] == 99) inc = 37500; + + mcDmuObject.setIncomeInDollars(inc); + mcDmuObject.setAdults(odt[ADULTS]); + mcDmuObject.setAutos(odt[AUTOS]); + mcDmuObject.setHhSize(odt[HHSIZE]); + mcDmuObject.setPersonIsFemale(odt[FEMALE]); + mcDmuObject.setAge(odt[AGE]); + mcDmuObject.setTourCategoryJoint(odt[JOINT]); + mcDmuObject.setTourCategoryEscort(odt[ESCORT]); + mcDmuObject.setNumberOfParticipantsInJointTour(odt[PARTYSIZE]); + + mcDmuObject.setWorkTourModeIsSOV(odt[WORK_TOUR_MODE] == 1 || odt[WORK_TOUR_MODE] == 2 ? 1 + : 0); + mcDmuObject.setWorkTourModeIsBike(odt[WORK_TOUR_MODE] == 10 ? 1 : 0); + mcDmuObject.setWorkTourModeIsHOV(odt[WORK_TOUR_MODE] >= 3 || odt[WORK_TOUR_MODE] <= 8 ? 1 + : 0); + + mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); + mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); + mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); + + mcDmuObject + .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); + + chosenDepartArriveCombination[0] = odt[DEPART_PERIOD]; + chosenDepartArriveCombination[1] = odt[ARRIVE_PERIOD]; + + // create an array with the chosen dest and the sample dests, for which + // to + // compute the logsums + mgraSetForLogsums[0] = chosenMgra; + for (int m = 0; m < mgraSample.length; m++) + mgraSetForLogsums[m + 1] = mgraSample[m]; + + int m = 0; + for (int destMgra : mgraSetForLogsums) + { + + if (mcModel != null && destMgra > 0) + { + // set the mode choice attributes needed by @variables in the + // UEC + // spreadsheets + mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, destMgra, false); + + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(destMgra)); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(destMgra)); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(destMgra)); + + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(destMgra))); + + modeChoiceLogsums[m] = new double[modelStructure.getSkimPeriodCombinationIndices().length]; + Arrays.fill(modeChoiceLogsums[m], -999); + } + + // compute the logsum for each depart/arrival time combination for + // the + // selected destination mgra + int i = 0; + for (int[] combo : departArriveCombinations) + { + + // mcModel might be null in the case where an estimation file + // record + // contains multiple purposes and the logsums are not desired + // for + // some purposes. + // destMgra might be null in the case of destination choice + // where + // some sample destinations are repeated, so the set of 30 or 40 + // contain 0s to reflect that. + if (mcModel == null || destMgra == 0) + { + continue; + } + + if (combo[0] == chosenDepartArriveCombination[0] + && combo[1] == chosenDepartArriveCombination[1]) + { + isChosenDepartArriveCombo = true; + chosenLogsumTodIndex = i; + } + + if (!chosenTodOnly || isChosenDepartArriveCombo) + { + + int departPeriod = DEFAULT_DEPART_INDICES[combo[0]]; + int arrivePeriod = DEFAULT_ARRIVE_INDICES[combo[1]]; + + // if the depart/arrive combination was flagged as + // unavailable, + // can skip the logsum calculation + if (unavailableCombination(departPeriod, arrivePeriod, departAvailable, + arriveAvailable)) + { + departArriveLogsums[m][i++] = -999; + continue; + } + + int logsumIndex = modelStructure.getSkimPeriodCombinationIndex(departPeriod, + arrivePeriod); + + // if a depart/arrive period combination results in a logsum + // index that's already had logsums computed, skip to next + // combination. + if (modeChoiceLogsums[m][logsumIndex] > -999) + { + departArriveLogsums[m][i++] = modeChoiceLogsums[m][logsumIndex]; + continue; + } + + mcDmuObject.setDepartPeriod(departPeriod); + mcDmuObject.setArrivePeriod(arrivePeriod); + + double logsum = logsumHelper.calculateTourMcLogsum(origMgra, destMgra, + departPeriod, arrivePeriod, mcModel, mcDmuObject); + + modeChoiceLogsums[m][logsumIndex] = logsum; + departArriveLogsums[m][i] = logsum; + + // write UEC calculation results to logsum specific log file + // if + // its the chosen dest and its the chosen time combo + if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2) + && (m == 0) /* && isChosenDepartArriveCombo */) + { + + nonManLogsumsLogger.info("Logsum[" + i + + "] calculation for estimation file record number " + odt[0]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); + nonManLogsumsLogger.info("mc purpose sheet = " + + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); + nonManLogsumsLogger.info("purpose category = " + + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " + + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); + nonManLogsumsLogger.info("origin mgra = " + odt[ORIG_MGRA]); + nonManLogsumsLogger.info("destination mgra = " + odt[DEST_MGRA]); + nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); + nonManLogsumsLogger.info("destination taz = " + + mgraManager.getTaz(destMgra)); + nonManLogsumsLogger.info("depart interval = " + + departArriveCombinationLabels[i][0] + ", @timeOutbound = " + + mcDmuObject.getTimeOutbound() + ", chosen depart = " + + departPeriod); + nonManLogsumsLogger.info("arrive interval = " + + departArriveCombinationLabels[i][1] + ", @timeInbound = " + + mcDmuObject.getTimeInbound() + ", chosen arrive = " + + arrivePeriod); + nonManLogsumsLogger.info("income category = " + odt[INCOME] + + ", @income = " + mcDmuObject.getIncome()); + nonManLogsumsLogger.info("adults = " + odt[ADULTS]); + nonManLogsumsLogger.info("autos = " + odt[AUTOS]); + nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); + nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " + + mcDmuObject.getFemale()); + nonManLogsumsLogger.info("jointTourCategory = " + odt[JOINT] + + ", @tourcategoryJoint = " + mcDmuObject.getTourCategoryJoint()); + nonManLogsumsLogger.info("partySize = " + odt[PARTYSIZE]); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info(""); + + mcModel.logUECResults(nonManLogsumsLogger, "Est Record: " + odt[0]); + nonManLogsumsLogger.info("Logsum Calculation for index: " + logsumIndex + + " , Logsum value: " + modeChoiceLogsums[m][logsumIndex]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger.info(""); + + isChosenDepartArriveCombo = false; + } + + } + i++; + } + + m++; + } + + } + + protected void calculateTripModeChoiceLogsums(HashMap rbMap, + ChoiceModelApplication mcModel, SandagTripModeChoiceDMU mcDmuObject, int[] odt, + int[] mgraSample) + { + + int origMgra = odt[ORIG_MGRA]; + int destMgra = odt[DEST_MGRA]; + int chosenMgra = odt[CHOSEN_MGRA]; + + for (int m = 0; m < tripModeChoiceLogsums.length; m++) + { + tripModeChoiceLogsums[m][0] = -999; + tripModeChoiceLogsums[m][1] = -999; + } + + if (origMgra == 0 || destMgra == 0 || odt[TOUR_MODE] == 0) return; + + int inc = 0; + if (odt[INCOME] == 1) inc = 7500; + else if (odt[INCOME] == 2) inc = 22500; + else if (odt[INCOME] == 3) inc = 37500; + else if (odt[INCOME] == 4) inc = 52500; + else if (odt[INCOME] == 5) inc = 67500; + else if (odt[INCOME] == 6) inc = 87500; + else if (odt[INCOME] == 7) inc = 112500; + else if (odt[INCOME] == 8) inc = 137500; + else if (odt[INCOME] == 9) inc = 175000; + else if (odt[INCOME] == 10) inc = 250000; + else if (odt[INCOME] == 99) inc = 37500; + + mcDmuObject.setOutboundHalfTourDirection(odt[DIRECTION]); + + mcDmuObject.setJointTour(odt[JOINT]); + mcDmuObject + .setEscortTour(odt[TOUR_PURPOSE] == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 + : 0); + + mcDmuObject.setIncomeInDollars(inc); + mcDmuObject.setAdults(odt[ADULTS]); + mcDmuObject.setAutos(odt[AUTOS]); + mcDmuObject.setAge(odt[AGE]); + mcDmuObject.setHhSize(odt[HHSIZE]); + mcDmuObject.setPersonIsFemale(odt[FEMALE] == 2 ? 1 : 0); + + mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(odt[TOUR_MODE]) ? 1 + : 0); + + mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); + mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); + mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); + + mcDmuObject + .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); + + mcDmuObject.setDepartPeriod(odt[DEPART_PERIOD]); + mcDmuObject.setTripPeriod(odt[TRIP_PERIOD]); + + int departPeriod = odt[TRIP_PERIOD]; + + // create an array with the chosen dest and the sample dests, for which + // to + // compute the logsums + mgraSetForLogsums[0] = chosenMgra; + for (int m = 0; m < mgraSample.length; m++) + { + mgraSetForLogsums[m + 1] = mgraSample[m]; + } + + if (mcModel == null || origMgra == 0) return; + + int m = 0; + for (int sampleMgra : mgraSetForLogsums) + { + + tripModeChoiceLogsums[m][0] = -999; + tripModeChoiceLogsums[m][1] = -999; + + if (mcModel != null && sampleMgra > 0) + { + // set the mode choice attributes needed by @variables in the + // UEC + // spreadsheets + mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, sampleMgra, false); + + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(sampleMgra)); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(sampleMgra)); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(sampleMgra)); + + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(sampleMgra))); + + if(odt[DIRECTION]==INBOUND_DIRCETION_CODE) + mcDmuObject.setInbound(true); + else + mcDmuObject.setInbound(false); + } + + // mcModel might be null in the case where an estimation file record + // contains multiple purposes and the logsums are not desired for + // some purposes. + // destMgra might be null in the case of destination choice where + // some sample destinations are repeated, so the set of 30 or 40 + // contain 0s to reflect that. + if (mcModel == null || sampleMgra == 0) + { + continue; + } + + // write UEC calculation results to logsum specific log file if + // its the chosen dest and its the chosen time combo + if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) + { + + nonManLogsumsLogger.info("IK Logsum calculation for estimation file record number " + + odt[0]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); + nonManLogsumsLogger.info("mc purpose sheet = " + + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); + nonManLogsumsLogger.info("purpose category = " + + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " + + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); + nonManLogsumsLogger.info("tour mode = " + odt[TOUR_MODE]); + nonManLogsumsLogger.info("origin mgra = " + origMgra); + nonManLogsumsLogger.info("sample destination mgra = " + sampleMgra); + nonManLogsumsLogger.info("final destination mgra = " + destMgra); + nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); + nonManLogsumsLogger.info("sample destination taz = " + + mgraManager.getTaz(sampleMgra)); + nonManLogsumsLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); + nonManLogsumsLogger.info("depart interval = " + departPeriod); + nonManLogsumsLogger.info("income category = " + odt[INCOME] + ", @income = " + + mcDmuObject.getIncome()); + nonManLogsumsLogger.info("adults = " + odt[ADULTS]); + nonManLogsumsLogger.info("autos = " + odt[AUTOS]); + nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); + nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " + + mcDmuObject.getFemale()); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info(""); + + mcDmuObject.getDmuIndexValues().setDebug(true); + } + + if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) + { + logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + + double logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, + mcModel, mcDmuObject, nonManLogsumsLogger); + tripModeChoiceLogsums[m][0] = logsum; + + if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) + { + nonManLogsumsLogger.info("IK Logsum value: " + tripModeChoiceLogsums[m][0]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger.info(""); + } + + // write UEC calculation results to logsum specific log file if + // its the chosen dest and its the chosen time combo + if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) + { + + nonManLogsumsLogger.info("KJ Logsum calculation for estimation file record number " + + odt[0]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); + nonManLogsumsLogger.info("mc purpose sheet = " + + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); + nonManLogsumsLogger.info("purpose category = " + + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " + + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); + nonManLogsumsLogger.info("origin mgra = " + sampleMgra); + nonManLogsumsLogger.info("sample destination mgra = " + destMgra); + nonManLogsumsLogger.info("final destination mgra = " + destMgra); + nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(sampleMgra)); + nonManLogsumsLogger + .info("sample destination taz = " + mgraManager.getTaz(destMgra)); + nonManLogsumsLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); + nonManLogsumsLogger.info("depart interval = " + departPeriod); + nonManLogsumsLogger.info("income category = " + odt[INCOME] + ", @income = " + + mcDmuObject.getIncome()); + nonManLogsumsLogger.info("adults = " + odt[ADULTS]); + nonManLogsumsLogger.info("autos = " + odt[AUTOS]); + nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); + nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " + + mcDmuObject.getFemale()); + nonManLogsumsLogger + .info("--------------------------------------------------------------------------------------------------------"); + nonManLogsumsLogger.info(""); + + mcDmuObject.getDmuIndexValues().setDebug(true); + } + + if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) + { + logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + + logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, + mcModel, mcDmuObject, nonManLogsumsLogger); + tripModeChoiceLogsums[m][1] = logsum; + + if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) + { + nonManLogsumsLogger.info("KJ Logsum value: " + tripModeChoiceLogsums[m][1]); + nonManLogsumsLogger.info(""); + nonManLogsumsLogger.info(""); + } + + m++; + } + + } + + protected double[] calculateTripModeChoiceLogsumForEstimationRecord( + HashMap rbMap, ChoiceModelApplication mcModel, + SandagTripModeChoiceDMU mcDmuObject, int[] odt, int sampleMgra) + { + + int origMgra = odt[ORIG_MGRA]; + int destMgra = odt[DEST_MGRA]; + + double[] tripModeChoiceLogsums = new double[2]; + tripModeChoiceLogsums[0] = -999; + tripModeChoiceLogsums[1] = -999; + + // mcModel would be null if the estimation file record has a stop + // purpose for which no ChoiceModelApplication has been defined. + if (origMgra == 0 || destMgra == 0 || sampleMgra == 0 || odt[TOUR_MODE] == 0 + || mcModel == null) return tripModeChoiceLogsums; + + mcDmuObject.setOutboundHalfTourDirection(odt[DIRECTION]); + + mcDmuObject.setJointTour(odt[JOINT]); + mcDmuObject + .setEscortTour(odt[TOUR_PURPOSE] == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 + : 0); + + int inc = 0; + if (odt[INCOME] == 1) inc = 7500; + else if (odt[INCOME] == 2) inc = 22500; + else if (odt[INCOME] == 3) inc = 37500; + else if (odt[INCOME] == 4) inc = 52500; + else if (odt[INCOME] == 5) inc = 67500; + else if (odt[INCOME] == 6) inc = 87500; + else if (odt[INCOME] == 7) inc = 112500; + else if (odt[INCOME] == 8) inc = 137500; + else if (odt[INCOME] == 9) inc = 175000; + else if (odt[INCOME] == 10) inc = 250000; + else if (odt[INCOME] == 99) inc = 37500; + + mcDmuObject.setIncomeInDollars(inc); + mcDmuObject.setAdults(odt[ADULTS]); + mcDmuObject.setAutos(odt[AUTOS]); + mcDmuObject.setAge(odt[AGE]); + mcDmuObject.setHhSize(odt[HHSIZE]); + mcDmuObject.setPersonIsFemale(odt[FEMALE] == 2 ? 1 : 0); + + mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(odt[TOUR_MODE]) ? 1 : 0); + mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(odt[TOUR_MODE]) ? 1 + : 0); + + mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); + mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); + mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); + + mcDmuObject + .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); + + mcDmuObject.setDepartPeriod(odt[DEPART_PERIOD]); + mcDmuObject.setTripPeriod(odt[TRIP_PERIOD]); + + int departPeriod = odt[TRIP_PERIOD]; + + // set the mode choice attributes needed by @variables in the UEC + // spreadsheets + mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, sampleMgra, false); + + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(sampleMgra)); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(sampleMgra)); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(sampleMgra)); + + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(sampleMgra))); + + if (mcDmuObject.getDmuIndexValues().getDebug()) + { + + // write UEC calculation results to logsum specific log file if + // its the chosen dest and its the chosen time combo + slcSoaLogger.info("IK Logsum calculation for estimation file record number " + odt[0]); + slcSoaLogger.info(""); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); + slcSoaLogger.info("mc purpose sheet = " + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); + slcSoaLogger.info("purpose category = " + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " + + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); + slcSoaLogger.info("tour mode = " + odt[TOUR_MODE]); + slcSoaLogger.info("origin mgra = " + origMgra); + slcSoaLogger.info("sample destination mgra = " + sampleMgra); + slcSoaLogger.info("final destination mgra = " + destMgra); + slcSoaLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); + slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(sampleMgra)); + slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); + slcSoaLogger.info("depart interval = " + departPeriod); + slcSoaLogger.info("income category = " + odt[INCOME] + ", @income = " + + mcDmuObject.getIncome()); + slcSoaLogger.info("adults = " + odt[ADULTS]); + slcSoaLogger.info("autos = " + odt[AUTOS]); + slcSoaLogger.info("hhsize = " + odt[HHSIZE]); + slcSoaLogger.info("gender = " + odt[FEMALE] + ", @female = " + mcDmuObject.getFemale()); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info(""); + + } + + if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) + { + logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + + double logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, + mcModel, mcDmuObject, nonManLogsumsLogger); + tripModeChoiceLogsums[0] = logsum; + + if (mcDmuObject.getDmuIndexValues().getDebug()) + { + + slcSoaLogger.info("IK Mode Choice Logsum value: " + tripModeChoiceLogsums[0]); + slcSoaLogger.info(""); + slcSoaLogger.info(""); + + // write UEC calculation results to logsum specific log file if + // its the chosen dest and its the chosen time combo + slcSoaLogger.info("KJ Logsum calculation for estimation file record number " + odt[0]); + slcSoaLogger.info(""); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); + slcSoaLogger.info("mc purpose sheet = " + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); + slcSoaLogger.info("purpose category = " + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " + + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); + slcSoaLogger.info("origin mgra = " + sampleMgra); + slcSoaLogger.info("sample destination mgra = " + destMgra); + slcSoaLogger.info("final destination mgra = " + destMgra); + slcSoaLogger.info("origin taz = " + mgraManager.getTaz(sampleMgra)); + slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(destMgra)); + slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); + slcSoaLogger.info("depart interval = " + departPeriod); + slcSoaLogger.info("income category = " + odt[INCOME] + ", @income = " + + mcDmuObject.getIncome()); + slcSoaLogger.info("adults = " + odt[ADULTS]); + slcSoaLogger.info("autos = " + odt[AUTOS]); + slcSoaLogger.info("hhsize = " + odt[HHSIZE]); + slcSoaLogger.info("gender = " + odt[FEMALE] + ", @female = " + mcDmuObject.getFemale()); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info(""); + + } + + if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) + { + logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, + departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); + + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, departPeriod, + mcDmuObject.getDmuIndexValues().getDebug()); + + logsum = logsumHelper.calculateTripMcLogsum(sampleMgra, destMgra, departPeriod, mcModel, + mcDmuObject, nonManLogsumsLogger); + tripModeChoiceLogsums[1] = logsum; + + if (mcDmuObject.getDmuIndexValues().getDebug()) + { + + slcSoaLogger.info("KJ Mode Choice Logsum value: " + tripModeChoiceLogsums[1]); + slcSoaLogger.info(""); + slcSoaLogger.info(""); + + } + + return tripModeChoiceLogsums; + + } + + protected int getModelPeriodFromTodIndex(int index) + { + int returnValue = -1; + if (index <= LAST_EA_INDEX) returnValue = EA; + else if (index <= LAST_AM_INDEX) returnValue = AM; + else if (index <= LAST_MD_INDEX) returnValue = MD; + else if (index <= LAST_PM_INDEX) returnValue = PM; + else returnValue = EV; + + return returnValue; + } + + /** + * + * @param departPeriod + * is the departure interval + * @param arrivePeriod + * is the arrival interval + * @param departAvailable + * is the model time period the departure interval belongs to + * (EA, AM, MD, PM, EV) + * @param arriveAvailable + * is the model time period the arrival interval belongs to (EA, + * AM, MD, PM, EV) + * @return true if the depart and/or arrival periods are unavailable, false + * if both are available. + */ + protected boolean unavailableCombination(int departPeriod, int arrivePeriod, + int[] departAvailable, int[] arriveAvailable) + { + + int departModelPeriod = getModelPeriodFromTodIndex(departPeriod); + int arriveModelPeriod = getModelPeriodFromTodIndex(arrivePeriod); + + boolean returnValue = true; + if (departAvailable[departModelPeriod] == 1 && arriveAvailable[arriveModelPeriod] == 1) + returnValue = false; + + return returnValue; + + } + + /** + * return the array of mode choice model cumulative probabilities determined + * while computing the mode choice logsum for the trip segmen during stop + * location choice. These probabilities arrays are stored for each sampled + * stop location so that when the selected sample stop location is known, + * the mode choice can be drawn from the already computed probabilities. + * + * @return mode choice cumulative probabilities array + */ + public double[] getStoredSegmentCumulativeProbabilities() + { + return tripModeChoiceSegmentStoredProbabilities; + } + + /** + * Start the matrix server + * + * @param rb + * is a ResourceBundle for the properties file for this + * application + */ + protected void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error("exception caught running ctramp model components -- exiting.", e); + throw new RuntimeException(); + + } + + } + + public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() + { + return logsumHelper.getAnmSkimCalculator(); + } + + public void setTazDistanceSkimArrays(double[][][] storedFromTazDistanceSkims, + double[][][] storedToTazDistanceSkims) + { + AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); + anm.setTazDistanceSkimArrays(storedFromTazDistanceSkims, storedToTazDistanceSkims); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java new file mode 100644 index 0000000..8a9f4dd --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java @@ -0,0 +1,368 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.application.SandagAppendMcLogsumDMU; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +public final class NonMandatoryDcEstimationMcLogsumsAppender + extends McLogsumsAppender +{ + + private static final int DEBUG_EST_RECORD1 = 1; + private static final int DEBUG_EST_RECORD2 = -1; + + /* + * for DC estimation file + */ + private static final int SEQ_FIELD = 2; + + // for Atwork subtour DC + // private static final int ORIG_MGRA_FIELD = 79; + // private static final int DEST_MGRA_FIELD = 220; + // private static final int MGRA1_FIELD = 221; + // private static final int PURPOSE_INDEX_OFFSET = 4; + + // for Escort DC + private static final int ORIG_MGRA_FIELD = 76; + private static final int DEST_MGRA_FIELD = 79; + private static final int MGRA1_FIELD = 217; + private static final int PURPOSE_INDEX_OFFSET = 0; + + // for NonMandatory DC + // private static final int ORIG_MGRA_FIELD = 76; + // private static final int DEST_MGRA_FIELD = 79; + // private static final int MGRA1_FIELD = 221; + // private static final int PURPOSE_INDEX_OFFSET = 0; + + private static final int DEPART_PERIOD_FIELD = 189; + private static final int ARRIVE_PERIOD_FIELD = 190; + private static final int INCOME_FIELD = 20; + private static final int ADULTS_FIELD = 32; + private static final int AUTOS_FIELD = 6; + private static final int HHSIZE_FIELD = 5; + private static final int GENDER_FIELD = 38; + private static final int AGE_FIELD = 39; + private static final int PURPOSE_FIELD = 80; + private static final int JOINT_ID_FIELD = 125; + private static final int JOINT_PURPOSE_FIELD = 126; + private static final int JOINT_P1_FIELD = 151; + private static final int JOINT_P2_FIELD = 152; + private static final int JOINT_P3_FIELD = 153; + private static final int JOINT_P4_FIELD = 154; + private static final int JOINT_P5_FIELD = 155; + private static final int NUM_MGRA_FIELDS = 30; + + private static final String OUTPUT_SAMPLE_DEST_LOGSUMS = "output.sample.dest.logsums"; + + public NonMandatoryDcEstimationMcLogsumsAppender(HashMap rbMap) + { + super(rbMap); + + debugEstimationFileRecord1 = DEBUG_EST_RECORD1; + debugEstimationFileRecord2 = DEBUG_EST_RECORD2; + + numMgraFields = NUM_MGRA_FIELDS; + } + + private void runLogsumAppender(ResourceBundle rb) + { + + totalTime1 = 0; + totalTime2 = 0; + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + mgraSetForLogsums = new int[numMgraFields + 1]; + + // allocate the logsums array for the chosen destination alternative + modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; + + departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; + + String outputAllKey = Util.getStringValueFromPropertyMap(rbMap, OUTPUT_SAMPLE_DEST_LOGSUMS); + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + "dc.est.skims.output.file"); + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + int dotIndex = outputFileName.indexOf("."); + String baseName = outputFileName.substring(0, dotIndex); + String extension = outputFileName.substring(dotIndex); + + // output1 is only written if "all" was set in propoerties file + String outputName1 = ""; + if (outputAllKey.equalsIgnoreCase("all")) outputName1 = baseName + "_" + "all" + extension; + + // output1 is written in any case + String outputName2 = baseName + "_" + "chosen" + extension; + + PrintWriter outStream1 = null; + PrintWriter outStream2 = null; + + try + { + if (outputAllKey.equalsIgnoreCase("all")) + outStream1 = new PrintWriter(new BufferedWriter(new FileWriter( + new File(outputName1)))); + outStream2 = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName2)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + writeDcFile(rbMap, outStream1, outStream2); + + logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); + logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); + + } + + private void writeDcFile(HashMap rbMap, PrintWriter outStream1, + PrintWriter outStream2) + { + + // print the chosen destMgra and the depart/arrive logsum field names to + // both + // files + if (outStream1 != null) outStream1.print("seq,sampno,chosenMgra"); + + // attach the OB and IB period labels to the logsum field names for each + // period + if (outStream1 != null) + { + for (String[] labels : departArriveCombinationLabels) + outStream1.print(",logsum" + labels[0] + labels[1]); + } + + outStream2.print("seq,sampno,chosenMgra,chosenTodLogsum"); + + // print each set of sample destMgra and the depart/arrive logsum + // fieldnames + // to file 1. + // print each set of sample destMgra and the chosen depart/arrive logsum + // fieldname to file 2. + for (int m = 1; m < departArriveLogsums.length; m++) + { + if (outStream1 != null) + { + outStream1.print(",sampleMgra" + m); + for (String[] labels : departArriveCombinationLabels) + outStream1.print(",logsum" + m + labels[0] + labels[1]); + } + + outStream2.print(",sampleMgra" + m); + outStream2.print(",sampleLogsum" + m); + } + if (outStream1 != null) outStream1.print("\n"); + outStream2.print("\n"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + int[] mgraSet = mgras[i]; + + odtSet[0] = seq; + + if (outStream1 != null) + { + outStream1.print(seq + "," + odtSet[SAMPNO]); + } + outStream2.print(seq + "," + odtSet[SAMPNO]); + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + + int[] departAvailable = {-1, 1, 1, 1, 1, 1}; + int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; + calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], + mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); + + // write chosen dest and logsums to both files + if (outStream1 != null) + { + outStream1.print("," + odtSet[DEST_MGRA]); + for (double logsum : departArriveLogsums[0]) + outStream1.printf(",%.8f", logsum); + } + + outStream2.print("," + odtSet[DEST_MGRA]); + outStream2.printf(",%.8f", departArriveLogsums[0][chosenLogsumTodIndex]); + + // write logsum sets for each dest in the sample to file 1 + for (int m = 1; m < departArriveLogsums.length; m++) + { + if (outStream1 != null) + { + outStream1.print("," + mgraSet[m - 1]); + for (double logsum : departArriveLogsums[m]) + outStream1.printf(",%.8f", logsum); + } + + outStream2.print("," + mgraSet[m - 1]); + outStream2.printf(",%.8f", departArriveLogsums[m][chosenLogsumTodIndex]); + } + if (outStream1 != null) outStream1.print("\n"); + outStream2.print("\n"); + + if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); + + seq++; + } + + if (outStream1 != null) outStream1.close(); + outStream2.close(); + + } + + private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); + int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); + + int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); + int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); + int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); + int[] income = hisTds.getColumnAsInt(INCOME_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); + int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); + int[] jointPerson1Participates = hisTds.getColumnAsInt(JOINT_P1_FIELD); + int[] jointPerson2Participates = hisTds.getColumnAsInt(JOINT_P2_FIELD); + int[] jointPerson3Participates = hisTds.getColumnAsInt(JOINT_P3_FIELD); + int[] jointPerson4Participates = hisTds.getColumnAsInt(JOINT_P4_FIELD); + int[] jointPerson5Participates = hisTds.getColumnAsInt(JOINT_P5_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][SAMPNO] = hisseq[r - 1]; + + odts[r - 1][DEPART_PERIOD] = departs[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][INCOME] = income[r - 1]; + odts[r - 1][ADULTS] = adults[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][HHSIZE] = hhsize[r - 1]; + odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; + + // the offest constant is used because at-work subtours in + // estimation file are coded as work purpose index (=1), + // but the model index to use is 5. Nonmandatory and escort files + // have correct purpose codes, so offset is 0. + int purposeIndex = purpose[r - 1] + PURPOSE_INDEX_OFFSET; + + odts[r - 1][ESCORT] = purposeIndex == 4 ? 1 : 0; + + odts[r - 1][PARTYSIZE] = jointPerson1Participates[r - 1] + + jointPerson2Participates[r - 1] + jointPerson3Participates[r - 1] + + jointPerson4Participates[r - 1] + jointPerson5Participates[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purposeIndex > 4 ? jtPurpose[r - 1] + : purposeIndex; + + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + NonMandatoryDcEstimationMcLogsumsAppender appender = new NonMandatoryDcEstimationMcLogsumsAppender( + rbMap); + + appender.startMatrixServer(rb); + appender.runLogsumAppender(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java new file mode 100644 index 0000000..f66cd14 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java @@ -0,0 +1,277 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.application.SandagAppendMcLogsumDMU; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +public final class NonMandatoryTodEstimationMcLogsumsAppender + extends McLogsumsAppender +{ + + private static final int DEBUG_EST_RECORD1 = 1; + private static final int DEBUG_EST_RECORD2 = -1; + + private static final int SEQ_FIELD = 7; + private static final int ORIG_MGRA_FIELD = 42; + private static final int DEST_MGRA_FIELD = 44; + private static final int DEPART_PERIOD_FIELD = 4; + private static final int ARRIVE_PERIOD_FIELD = 5; + private static final int INCOME_FIELD = 149; + private static final int ADULTS_FT_FIELD = 150; + private static final int ADULTS_PT_FIELD = 151; + private static final int ADULTS_UN_FIELD = 152; + private static final int ADULTS_RT_FIELD = 153; + private static final int ADULTS_NW_FIELD = 154; + private static final int AUTOS_FIELD = 146; + private static final int HHSIZE_FIELD = 147; + private static final int GENDER_FIELD = 24; + private static final int AGE_FIELD = 25; + private static final int PURPOSE_FIELD = 45; + private static final int JOINT_PURPOSE_FIELD = 58; + private static final int JOINT_ID_FIELD = 57; + private static final int JOINT_PARTICIPANTS_FIELD = 104; + private static final int MGRA1_FIELD = 1; + private static final int NUM_MGRA_FIELDS = 0; + + private NonMandatoryTodEstimationMcLogsumsAppender(HashMap rbMap) + { + super(rbMap); + + debugEstimationFileRecord1 = DEBUG_EST_RECORD1; + debugEstimationFileRecord2 = DEBUG_EST_RECORD2; + + numMgraFields = NUM_MGRA_FIELDS; + + } + + private void runLogsumAppender(ResourceBundle rb) + { + + totalTime1 = 0; + totalTime2 = 0; + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + mgraSetForLogsums = new int[numMgraFields + 1]; + + // allocate the logsums array for the chosen destination alternative + modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; + + departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + "tod.est.skims.output.file"); + + PrintWriter outStream = null; + + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + try + { + outStream = new PrintWriter( + new BufferedWriter(new FileWriter(new File(outputFileName)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + writeTodFile(rbMap, outStream); + + logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); + logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); + + } + + private void writeTodFile(HashMap rbMap, PrintWriter outStream2) + { + + // print the chosen destMgra and the depart/arrive logsum field names to + // the + // file + outStream2.print("seq,hisseq,chosenMgra"); + for (String[] labels : departArriveCombinationLabels) + { + outStream2.print(",logsum" + labels[0] + labels[1]); + } + outStream2.print("\n"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getTodEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + int[] mgraSet = mgras[i]; + + odtSet[0] = seq; + + outStream2.print(seq + "," + odtSet[SAMPNO]); + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + + int[] departAvailable = {-1, 1, 1, 1, 1, 1}; + int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; + calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], + mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); + + // write chosen dest and logsums to both files + outStream2.print("," + odtSet[DEST_MGRA]); + for (double logsum : departArriveLogsums[0]) + { + outStream2.printf(",%.8f", logsum); + } + outStream2.print("\n"); + + if (seq % 1000 == 0) logger.info("wrote TOD Estimation file record: " + seq); + + seq++; + } + + outStream2.close(); + + } + + private int[][] getTodEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); + int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); + + int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); + int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); + int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); + int[] income = hisTds.getColumnAsInt(INCOME_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] adultsFt = hisTds.getColumnAsInt(ADULTS_FT_FIELD); + int[] adultsPt = hisTds.getColumnAsInt(ADULTS_PT_FIELD); + int[] adultsUn = hisTds.getColumnAsInt(ADULTS_UN_FIELD); + int[] adultsRt = hisTds.getColumnAsInt(ADULTS_RT_FIELD); + int[] adultsNw = hisTds.getColumnAsInt(ADULTS_NW_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); + int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); + int[] jointParticipants = hisTds.getColumnAsInt(JOINT_PARTICIPANTS_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][SAMPNO] = hisseq[r - 1]; + + odts[r - 1][DEPART_PERIOD] = departs[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][INCOME] = income[r - 1]; + odts[r - 1][ADULTS] = adultsFt[r - 1] + adultsPt[r - 1] + adultsUn[r - 1] + + adultsRt[r - 1] + adultsNw[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][HHSIZE] = hhsize[r - 1]; + odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; + odts[r - 1][ESCORT] = purpose[r - 1] == 4 ? 1 : 0; + odts[r - 1][PARTYSIZE] = jointParticipants[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purpose[r - 1] > 4 ? jtPurpose[r - 1] + : purpose[r - 1]; + + } + + return odts; + } + + public static void main(String[] args) + { + + long startTime = System.currentTimeMillis(); + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + NonMandatoryTodEstimationMcLogsumsAppender appender = new NonMandatoryTodEstimationMcLogsumsAppender( + rbMap); + + appender.startMatrixServer(rb); + appender.runLogsumAppender(rb); + + System.out.println("total runtime = " + ((System.currentTimeMillis() - startTime) / 1000) + + " seconds."); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java new file mode 100644 index 0000000..d7f07be --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java @@ -0,0 +1,641 @@ +package org.sandag.abm.accessibilities; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; +import java.io.File; +import java.io.Serializable; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.AutoUEC; +import org.sandag.abm.modechoice.MaasUEC; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.NonMotorUEC; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; +import com.pb.common.util.Tracer; + +/** + * This class builds utility components for auto modes (SOV and HOV). + * + * @author Joel Freedman + * @version May, 2009 + */ +public class NonTransitUtilities + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(NonTransitUtilities.class); + + public static final int OFFPEAK_PERIOD_INDEX = 0; + public static final int PEAK_PERIOD_INDEX = 1; + + private static final String[] SOVPERIODS = {"OP", "PK"}; + private static final String[] HOVPERIODS = {"OP", "PK"}; + private static final String[] NMTPERIODS = {"OP"}; + private static final String[] MAASPERIODS = {"OP", "PK"}; + + + // store taz-taz exponentiated utilities (period, from taz, to taz) + private double[][][] sovExpUtilities; + private double[][][] hovExpUtilities; + private double[][][] nMotorExpUtilities; + private double[][][] maasExpUtilities; + + private double[] avgTazHourlyParkingCost; + private float[] avgTazTaxiWaitTime; + private float[] avgTazSingleTNCWaitTime; + private float[] avgTazSharedTNCWaitTime; + + // A HashMap of non-motorized utilities, period,oMgra,dMgra (where dMgra is + // ragged) + private HashMap[][] mgraNMotorExpUtilities; + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private AutoUEC[] sovUEC; + private AutoUEC[] hovUEC; + private NonMotorUEC[] nMotorUEC; + private MaasUEC[] maasUEC; + + private int maxTaz; + + private boolean trace, seek; + private Tracer tracer; + + //added for TNC and Taxi modes + TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; + + + /** + * Constructor. + * + * @param rb + * Resourcebundle with path to acc.uec.file, acc.data.page, + * acc.sov.offpeak.page, and acc.hov.offpeak.page + */ + + public NonTransitUtilities(HashMap rbMap, double[][][] mySovExpUtilities, + double[][][] myHovExpUtilities, double[][][] myNMotorExpUtilities, double[][][] myMaasExpUtilities) + { + + sovExpUtilities = mySovExpUtilities; + hovExpUtilities = myHovExpUtilities; + nMotorExpUtilities = myNMotorExpUtilities; + maasExpUtilities = myMaasExpUtilities; + + mgraManager = MgraDataManager.getInstance(); + tazManager = TazDataManager.getInstance(rbMap); + + maxTaz = tazManager.maxTaz; + + logger.info("max Taz " + maxTaz); + + // Create the peak and off-peak UECs + String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.data.page"); + int offpeakSOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sov.offpeak.page"); + int offpeakHOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.hov.offpeak.page"); + int peakSOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sov.peak.page"); + int peakHOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.hov.peak.page"); + int nonMotorPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.nonmotorized.page"); + int peakMaasPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.maas.peak.page"); + int offpeakMaasPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.maas.offpeak.page"); + + sovUEC = new AutoUEC[SOVPERIODS.length]; + hovUEC = new AutoUEC[HOVPERIODS.length]; + nMotorUEC = new NonMotorUEC[NMTPERIODS.length]; + maasUEC = new MaasUEC[MAASPERIODS.length]; + + sovUEC[OFFPEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, offpeakSOVPage, dataPage); + sovUEC[PEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, peakSOVPage, dataPage); + hovUEC[OFFPEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, offpeakHOVPage, dataPage); + hovUEC[PEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, peakHOVPage, dataPage); + nMotorUEC[OFFPEAK_PERIOD_INDEX] = new NonMotorUEC(rbMap, uecFileName, nonMotorPage, + dataPage); + maasUEC[OFFPEAK_PERIOD_INDEX] = new MaasUEC(rbMap, uecFileName, offpeakMaasPage, dataPage); + maasUEC[PEAK_PERIOD_INDEX] = new MaasUEC(rbMap, uecFileName, peakMaasPage, dataPage); + + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + int[] traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + int[] traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + logger.info("Setting trace zone pair in NonTransitUtilities Object for i: "+ traceOtaz[i] + " j: " + traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + int maxMgra = mgraManager.getMaxMgra(); +// logger.info("max Mgra " + maxMgra); + + mgraNMotorExpUtilities = new HashMap[NMTPERIODS.length][maxMgra + 1]; + + sovExpUtilities = new double[SOVPERIODS.length][maxTaz + 1][]; + hovExpUtilities = new double[HOVPERIODS.length][maxTaz + 1][]; + nMotorExpUtilities = new double[NMTPERIODS.length][maxTaz + 1][]; + + maasExpUtilities = new double[MAASPERIODS.length][maxTaz + 1][]; + + calculateAverageTazParkingCosts(); + + tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); + tncTaxiWaitTimeCalculator.createWaitTimeDistributions(rbMap); + + calculateAverageMaasWaitTimes(); + + + + } + + /** + * set the utilities values created by another object by calling + * buildUtilities() + */ + public void setAllUtilities(double[][][][] ntUtilities) + { + this.sovExpUtilities = ntUtilities[0]; + this.hovExpUtilities = ntUtilities[1]; + this.nMotorExpUtilities = ntUtilities[2]; + this.maasExpUtilities = ntUtilities[3]; + } + + /** + * get the set of utilities arrays built by calling buildUtilities(). + * + * @return array of 4 utilities arrays: sovExpUtilities, hovExpUtilities, + * nMotorExpUtilities, maasExpUtilities + */ + public double[][][][] getAllUtilities() + { + double[][][][] allUtilities = new double[3][][][]; + allUtilities[0] = sovExpUtilities; + allUtilities[1] = hovExpUtilities; + allUtilities[2] = nMotorExpUtilities; + allUtilities[3] = maasExpUtilities; + return allUtilities; + } + + /** + * set the HashMap of non-motorized utilities, period,oMgra,dMgra (where + * dMgra is ragged) + * + * @param mgraNMotorExpUtilities + */ + public void setNonMotorUtilsMap(HashMap[][] aMgraNMotorExpUtilities) + { + this.mgraNMotorExpUtilities = aMgraNMotorExpUtilities; + } + + /** + * get the HashMap of non-motorized utilities, period,oMgra,dMgra (where + * dMgra is ragged) that was built by calling buildUtilities(). + * + * @return mgraNMotorExpUtilities + */ + public HashMap[][] getNonMotorUtilsMap() + { + return mgraNMotorExpUtilities; + } + + /** + * Build SOV, HOV, and non-motorized exponentiated utilities for all + * TAZ-pairs. Also builds non-motorized exponentiated utilities for close-in + * mgra pairs. + * + */ + public void buildUtilities() + { + } + + /* + * public void buildUtilities() { + * + * logger.info("Calculating Non-Transit Zonal Utilities"); + * + * // first calculate the Tap-Tap utilities, exponentiate, and store + * logger.info("Calculating Taz-Taz utilities"); + * + * int maxMgra = mgraManager.getMaxMgra(); logger.info("max Mgra " + + * maxMgra); + * + * mgraNMotorExpUtilities = new HashMap[NMTPERIODS.length][maxMgra + 1]; + * + * sovExpUtilities = new double[SOVPERIODS.length][maxTaz + 1][maxTaz + 1]; + * hovExpUtilities = new double[HOVPERIODS.length][maxTaz + 1][maxTaz + 1]; + * nMotorExpUtilities = new double[NMTPERIODS.length][maxTaz + 1][maxTaz + + * 1]; + * + * for (int iTaz = 1; iTaz <= maxTaz; ++iTaz) { + * + * if (iTaz <= 10 || (iTaz % 500) == 0) logger.info("...Origin TAZ " + + * iTaz); + * + * // calculate the utilities for close-in mgras int[] oMgras = + * tazManager.getMgraArray(iTaz); if ( oMgras == null ) continue; + * + * for (int oMgra : oMgras) { + * + * // if there are mgras within walking distance if + * (mgraManager.getMgrasWithinWalkDistanceFrom(oMgra) != null) { int[] + * dMgras = mgraManager.getMgrasWithinWalkDistanceFrom(oMgra); + * + * int mgraNumber = 0; + * + * // cycle through periods, and calculate utilities for (int period = 0; + * period < NMTPERIODS.length; ++period) { + * + * mgraNMotorExpUtilities[period][oMgra] = new HashMap(); + * + * // cycle through the destination mgras for (int dMgra : dMgras) { + * + * double nmtUtility = nMotorUEC[period].calculateUtilitiesForMgraPair( + * oMgra, dMgra ); + * + * // exponentiate the utility if (nmtUtility > -500) + * mgraNMotorExpUtilities[period][oMgra].put( dMgra, Math.exp(nmtUtility) ); + * ++mgraNumber; + * + * } + * + * } + * + * } } + * + * for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) { + * + * if (seek && !tracer.isTraceZonePair(iTaz, jTaz)) continue; + * + * for (int period = 0; period < SOVPERIODS.length; ++period) { + * + * double sovUtility = sovUEC[period].calculateUtilitiesForTazPair(iTaz, + * jTaz); // exponentiate the SOV utility if (sovUtility > -500) + * sovExpUtilities[period][iTaz][jTaz] = Math.exp(sovUtility); } + * + * for (int period = 0; period < HOVPERIODS.length; ++period) { + * + * double hovUtility = hovUEC[period].calculateUtilitiesForTazPair(iTaz, + * jTaz); // exponentiate the SOV utility if (hovUtility > -500) + * hovExpUtilities[period][iTaz][jTaz] = Math.exp(hovUtility); } + * + * for (int period = 0; period < NMTPERIODS.length; ++period) { + * + * double nmtUtility = nMotorUEC[period].calculateUtilitiesForTazPair(iTaz, + * jTaz); // exponentiate the SOV utility if (nmtUtility > -500) + * nMotorExpUtilities[period][iTaz][jTaz] = Math.exp(nmtUtility); } + * + * } } + * + * } + */ + + /** + * calculate an average TAZ parking cost to use in accessibilities + * calculation which are done at TAZ level. + */ + private void calculateAverageTazParkingCosts() + { + + avgTazHourlyParkingCost = new double[maxTaz + 1]; + + for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) + { + + int[] mgras = tazManager.getMgraArray(jTaz); + if (mgras == null || mgras.length == 0) continue; + + double cost = 0; + int count = 0; + for (int mgra : mgras) + { + float mgraCost = mgraManager.getMgraHourlyParkingCost(mgra); + if (mgraCost > 0) + { + cost += mgraCost; + count++; + } + if (count > 0) cost /= count; + } + + avgTazHourlyParkingCost[jTaz] = cost; + + } + + } + + /** + * calculate an average TAZ parking cost to use in accessibilities + * calculation which are done at TAZ level. + */ + private void calculateAverageMaasWaitTimes() + { + + avgTazTaxiWaitTime = new float[maxTaz + 1]; + avgTazSingleTNCWaitTime = new float[maxTaz + 1]; + avgTazSharedTNCWaitTime = new float[maxTaz + 1]; + + for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) + { + + int[] mgras = tazManager.getMgraArray(jTaz); + if (mgras == null || mgras.length == 0) continue; + + float taxiWait = 0; + float singleTNCWait = 0; + float sharedTNCWait = 0; + int singleTNCCount = 0; + int sharedTNCCount = 0; + int taxiCount = 0; + + + for (int mgra : mgras) + { + float popEmpDen = (float) mgraManager.getPopEmpPerSqMi(mgra); + float singleTNCWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDen); + float sharedTNCWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDen); + float taxiWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDen); + + if (taxiWaitTime > 0) + { + taxiWait += taxiWaitTime; + taxiCount++; + } + if (singleTNCWaitTime > 0) + { + singleTNCWait += singleTNCWaitTime; + singleTNCCount++; + } + if (sharedTNCWaitTime > 0) + { + sharedTNCWait += sharedTNCWaitTime; + sharedTNCCount++; + } + if (taxiCount > 0) taxiWait /= taxiCount; + if (singleTNCCount > 0) singleTNCWait /= singleTNCCount; + if (sharedTNCCount > 0) sharedTNCWait /= sharedTNCCount; + } + + avgTazTaxiWaitTime[jTaz] = taxiWait; + avgTazSingleTNCWaitTime[jTaz] = singleTNCWait; + avgTazSharedTNCWaitTime[jTaz] = sharedTNCWait; + } + + } + + public void buildUtilitiesForOrigMgraAndPeriod(int iMgra, int period) + { + + int iTaz = mgraManager.getTaz(iMgra); + if (sovExpUtilities[period][iTaz] != null) return; + + sovExpUtilities[period][iTaz] = new double[maxTaz + 1]; + hovExpUtilities[period][iTaz] = new double[maxTaz + 1]; + maasExpUtilities[period][iTaz] = new double[maxTaz + 1]; + + for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) + { + + double sovUtility = sovUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, + avgTazHourlyParkingCost[jTaz]); + // exponentiate the SOV utility + if (sovUtility > -500) sovExpUtilities[period][iTaz][jTaz] = Math.exp(sovUtility); + + double hovUtility = hovUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, + avgTazHourlyParkingCost[jTaz]); + // exponentiate the HOV utility + if (hovUtility > -500) hovExpUtilities[period][iTaz][jTaz] = Math.exp(hovUtility); + + double maasUtility = maasUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, + avgTazTaxiWaitTime[jTaz], avgTazSingleTNCWaitTime[jTaz], avgTazSharedTNCWaitTime[jTaz]); + // exponentiate the SOV utility + if (maasUtility > -500) maasExpUtilities[period][iTaz][jTaz] = Math.exp(maasUtility); + + + } + + // non-motorized utilities are only needed for off-peak period, so if + // period index == 1 (peak) no nead to calculate off-peak + if (nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz] == null) + { + + nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz] = new double[maxTaz + 1]; + + for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) + { + + double nmtUtility = nMotorUEC[OFFPEAK_PERIOD_INDEX].calculateUtilitiesForTazPair( + iTaz, jTaz); + // exponentiate the SOV utility + if (nmtUtility > -500) + nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz][jTaz] = Math.exp(nmtUtility); + + } + + } + + } + + /** + * Get the non-motorized exponentiated utility for the mgra-pair and period. + * This method will return the taz-taz exponentiated non-motorized utility + * if the mgra-mgra exp utility doesn't exist. Otherwise the mgra-mgra exp. + * utility will be returned. + * + * @param iMgra + * Origin/production mgra. + * @param jMgra + * Destination/attraction mgra. + * @param period + * Period. + * @return The non-motorized exponentiated utility. + */ + /* + * public double getNMotorExpUtility(int iMgra, int jMgra, int period) { // + * no mgra-mgra utilities for this origin if + * (mgraNMotorExpUtilities[period][iMgra] == null) { int iTaz = + * mgraManager.getTaz(iMgra); int jTaz = mgraManager.getTaz(jMgra); return + * nMotorExpUtilities[period][iTaz][jTaz]; } + * + * // mgra-mgra utilities exist if + * (mgraNMotorExpUtilities[period][iMgra].containsKey(jMgra)) { return + * mgraNMotorExpUtilities[period][iMgra].get(jMgra); } + * + * // no mgra-mgra utilities for this destination int iTaz = + * mgraManager.getTaz(iMgra); int jTaz = mgraManager.getTaz(jMgra); return + * nMotorExpUtilities[period][iTaz][jTaz]; } + */ + public double getNMotorExpUtility(int iMgra, int jMgra, int period) + { + + // if no utilities exist for period and origin mgra, try to compute them + if (mgraNMotorExpUtilities[period][iMgra] == null) + { + + // get the mgras within walking distance of the iMgra + int[] dMgras = mgraManager.getMgrasWithinWalkDistanceFrom(iMgra); + + if (dMgras == null) + { + mgraNMotorExpUtilities[period][iMgra] = new HashMap(0); + } else + { + mgraNMotorExpUtilities[period][iMgra] = new HashMap(dMgras.length); + + // cycle through the destination mgras + for (int dMgra : dMgras) + { + // calculate utility for the specified mgra and period + double nmtUtility = nMotorUEC[period].calculateUtilitiesForMgraPair(iMgra, + dMgra); + + // exponentiate the utility + if (nmtUtility > -500) + mgraNMotorExpUtilities[period][iMgra].put(dMgra, Math.exp(nmtUtility)); + } + + } + + } + + // if jMgra is in the HashMap, return its utility value + if (mgraNMotorExpUtilities[period][iMgra].containsKey(jMgra)) + return mgraNMotorExpUtilities[period][iMgra].get(jMgra); + + // otherwise, get exponentiated utilities based on highway skim values + // for the taz pair associated with iMgra and jMgra. + int iTaz = mgraManager.getTaz(iMgra); + int jTaz = mgraManager.getTaz(jMgra); + return nMotorExpUtilities[period][iTaz][jTaz]; + + } + + /** + * Get the SOV Exponentiated Utility for a given ptaz, ataz, and period + * + * @param pTaz + * Production/Origin TAZ + * @param aTaz + * Attraction/Destination TAZ + * @param period + * Period + * @return SOV Exponentiated Utility. + */ + public double getSovExpUtility(int pTaz, int aTaz, int period) + { + return sovExpUtilities[period][pTaz][aTaz]; + } + + /** + * Get the HOV Exponentiated Utility for a given ptaz, ataz, and period + * + * @param pTaz + * Production/Origin TAZ + * @param aTaz + * Attraction/Destination TAZ + * @param period + * Period + * @return SOV Exponentiated Utility. + */ + public double getHovExpUtility(int pTaz, int aTaz, int period) + { + return hovExpUtilities[period][pTaz][aTaz]; + } + + /** + * Get the Maas Exponentiated Utility for a given ptaz, ataz, and period + * + * @param pTaz + * Production/Origin TAZ + * @param aTaz + * Attraction/Destination TAZ + * @param period + * Period + * @return Maas Exponentiated Utility. + */ + public double getMaasExpUtility(int pTaz, int aTaz, int period) + { + return maasExpUtilities[period][pTaz][aTaz]; + } + /** + * The main method runs this class, for testing purposes. + * + * @param args + * args[0] is the property file for this test run. + */ + public static void main(String[] args) + { + + ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + boolean os64bit = false; + MatrixDataServer matrixServer = null; + + os64bit = Boolean.parseBoolean(Util.getStringValueFromPropertyMap(rbMap, + "operatingsystem.64bit")); + if (os64bit) + { + + String serverAddress = Util.getStringValueFromPropertyMap(rbMap, "server.address"); + + int serverPort = Util.getIntegerValueFromPropertyMap(rbMap, "server.port"); + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + + matrixServer = new MatrixDataServer(); + + // bind this concrete object with the cajo library objects for + // managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + System.out.println(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort)); + e.printStackTrace(); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + System.out.println(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort)); + e.printStackTrace(); + throw new RuntimeException(); + } + } + + double[][][] sovExpUtilities = null; + double[][][] hovExpUtilities = null; + double[][][] nMotorExpUtilities = null; + double[][][] maasExpUtilities = null; + + NonTransitUtilities au = new NonTransitUtilities(rbMap, sovExpUtilities, hovExpUtilities, + nMotorExpUtilities, maasExpUtilities); + au.buildUtilities(); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java new file mode 100644 index 0000000..3023ff4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java @@ -0,0 +1,297 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +public final class ParkLocationEstimationAppender +{ + + private transient Logger logger = Logger.getLogger(ParkLocationEstimationAppender.class); + + private static final int DEBUG_EST_RECORD = 1; + + private static final String ESTIMATION_DATA_RECORDS_FILE_KEY = "plc.estimation.data.file"; + private static final String OUTPUT_DATA_RECORDS_FILE_KEY = "plc.est.skims.output.file"; + + // define input table field indices + private static final int ID_FIELD = 1; + private static final int DEPART_PERIOD_FIELD = 2; + private static final int TYPE_FIELD = 4; + private static final int ORIG_FIELD = 7; + private static final int DEST_FIELD = 8; + + // define indices for storing input data in an internal table. + // start field indices at 1; reserve 0 for the input file record sequence + // number. + private static final int ID = 1; + private static final int DEPART = 2; + private static final int TYPE = 3; + private static final int ORIG = 4; + private static final int DEST = 5; + private static final int NUM_FIELDS = 5; + + private static final int OD_TYPE_INDEX = 0; + private static final int OP_TYPE_INDEX = 1; + private static final int PD_TYPE_INDEX = 2; + private static final int[] TYPE_INDICES = {OD_TYPE_INDEX, OP_TYPE_INDEX, + PD_TYPE_INDEX }; + private static final String OD_TYPE = "OD"; + private static final String OP_TYPE = "OP"; + private static final String PD_TYPE = "PD"; + private static final String[] TYPE_LABELS = {OD_TYPE, OP_TYPE, PD_TYPE}; + + private static final int AUTO_TIME_SKIM_INDEX = 0; + private static final int AUTO_DIST_SKIM_INDEX = 2; + + private static final double WALK_SPEED = 3.0; // mph; + + private static final float defaultVOT = 15.0f; + private MatrixDataServerIf ms; + + public ParkLocationEstimationAppender() + { + } + + private void runAppender(ResourceBundle rb) + { + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + MgraDataManager mgraManager = MgraDataManager.getInstance(rbMap); + SandagModelStructure modelStructure = new SandagModelStructure(); + + double[] distances = new double[mgraManager.getMaxMgra() + 1]; + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + OUTPUT_DATA_RECORDS_FILE_KEY); + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + int dotIndex = outputFileName.indexOf("."); + String baseName = outputFileName.substring(0, dotIndex); + String extension = outputFileName.substring(dotIndex); + String outputName = baseName + extension; + + PrintWriter outStream = null; + + try + { + outStream = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + outStream + .println("seq,id,origMgra,destMgra,departPeriod,index,autoDist,autoTime,walkDist,walkTime"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getPlcEstimationData(estTds); + + AutoAndNonMotorizedSkimsCalculator anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + odtSet[0] = seq; + + double aDist = 999; + double aTime = 999; + double wDist = 999; + double wTime = 999; + if (odtSet[ORIG] > 0 && odtSet[DEST] > 0) + { + + int skimPeriodIndex = modelStructure.getSkimPeriodIndex(odtSet[DEPART]) + 1; // depart + // skim + // period + double[] autoSkims = anm.getAutoSkims(odtSet[ORIG], odtSet[DEST], skimPeriodIndex, defaultVOT, + (seq == DEBUG_EST_RECORD), logger); + aDist = autoSkims[AUTO_DIST_SKIM_INDEX]; + aTime = autoSkims[AUTO_TIME_SKIM_INDEX]; + + // get the array of mgras within walking distance of the + // destination + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(odtSet[ORIG]); + + // set the distance values for the mgras walkable to the + // destination + if (walkMgras != null) + { + + // get distances, in feet, and convert to miles + for (int wMgra : walkMgras) + { + if (wMgra == odtSet[DEST]) + { + wDist = mgraManager.getMgraToMgraWalkDistFrom(odtSet[ORIG], wMgra) / 5280.0; + wTime = (wDist / WALK_SPEED) * 60.0; + break; + } + } + } + + } + + outStream.println(seq + "," + odtSet[ID] + "," + odtSet[ORIG] + "," + odtSet[DEST] + + "," + odtSet[DEPART] + "," + TYPE_LABELS[odtSet[TYPE]] + "," + aDist + "," + + aTime + "," + wDist + "," + wTime); + + if (seq % 1000 == 0) logger.info("wrote PLC Estimation file record: " + seq); + + seq++; + } + + outStream.close(); + + } + + private int[][] getPlcEstimationData(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS + 1]; + + int[] ids = hisTds.getColumnAsInt(ID_FIELD); + int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); + String[] types = hisTds.getColumnAsString(TYPE_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_FIELD); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + odts[r - 1][ID] = ids[r - 1]; + odts[r - 1][DEPART] = departs[r - 1]; + + for (int i = 0; i < TYPE_INDICES.length; i++) + { + if (types[r - 1].equalsIgnoreCase(TYPE_LABELS[i])) + { + odts[r - 1][TYPE] = TYPE_INDICES[i]; + break; + } + } + + odts[r - 1][ORIG] = origs[r - 1]; + odts[r - 1][DEST] = dests[r - 1]; + + } + + return odts; + } + + protected TableDataSet getEstimationDataTableDataSet(HashMap rbMap) + { + + String estFileName = Util.getStringValueFromPropertyMap(rbMap, + ESTIMATION_DATA_RECORDS_FILE_KEY); + if (estFileName == null) + { + logger.error("Error getting the filename from the properties file for the Sandag estimation data records file."); + logger.error("Properties file target: " + ESTIMATION_DATA_RECORDS_FILE_KEY + + " not found."); + logger.error("Please specify a filename value for the " + + ESTIMATION_DATA_RECORDS_FILE_KEY + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFile(new File(estFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag estimation data records file: %s into TableDataSet object.", + estFileName)); + throw new RuntimeException(e); + } + + } + + /** + * Start the matrix server + * + * @param rb + * is a ResourceBundle for the properties file for this + * application + */ + protected void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error("exception caught running ctramp model components -- exiting.", e); + throw new RuntimeException(); + + } + + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + ParkLocationEstimationAppender appender = new ParkLocationEstimationAppender(); + + appender.startMatrixServer(rb); + appender.runAppender(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java new file mode 100644 index 0000000..95726b6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java @@ -0,0 +1,788 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +public final class SkimsAppender +{ + + protected transient Logger logger = Logger.getLogger(SkimsAppender.class); + + private static final String OBS_DATA_RECORDS_FILE_KEY = "onBoard.survey.file"; + private static final String HIS_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; + + private static final int OBS_UNIQUE_ID = 1; + private static final int OBS_ORIG_MGRA = 78; + private static final int OBS_DEST_MGRA = 79; + + // used for trip file: + private static final int OBS_OUT_TOUR_PERIOD = 133; + private static final int OBS_IN_TOUR_PERIOD = 134; + + // used for tour file: + // private static final int OBS_DEPART_PERIOD = 132; + // private static final int OBS_ARRIVE_PERIOD = 133; + + /* + * for home based tour mode choice estimation files private static final int + * HIS_ORIG_MGRA = 72; private static final int HIS_DEST_MGRA = 75; private + * static final int HIS_DEPART_PERIOD = 185; private static final int + * HIS_ARRIVE_PERIOD = 186; + */ + + /* + * for work based tour mode choice estimation files + */ + private static final int HIS_ORIG_MGRA = 76; + private static final int HIS_DEST_MGRA = 84; + private static final int HIS_DEPART_PERIOD = 159; + private static final int HIS_ARRIVE_PERIOD = 160; + + // survey periods are: 0=not used, 1=03:00-05:59, 2=06:00-08:59, + // 3=09:00-11:59, + // 4=12:00-15:29, 5=15:30-18:59, 6=19:00-02:59 + // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), + // 6=3(OP) + + // define a conversion array to convert period values in the survey file to + // skim + // period indices used in this propgram: 1=am peak, 2=pm peak, + // 3=off-peak. + private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; + private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; + + private MatrixDataServerIf ms; + private BestTransitPathCalculator bestPathUEC; + + TransitWalkAccessDMU walkDmu; + + private SkimsAppender() + { + } + + private void runSkimsAppender(ResourceBundle rb) + { + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + // instantiate these objects right away + TazDataManager tazs = TazDataManager.getInstance(rbMap); + MgraDataManager mgraManager = MgraDataManager.getInstance(rbMap); + TapDataManager tapManager = TapDataManager.getInstance(rbMap); + + AutoAndNonMotorizedSkimsCalculator anm = null; + WalkTransitWalkSkimsCalculator wtw = null; + WalkTransitDriveSkimsCalculator wtd = null; + DriveTransitWalkSkimsCalculator dtw = null; + + Logger autoLogger = Logger.getLogger("auto"); + Logger wtwLogger = Logger.getLogger("wtw"); + Logger wtdLogger = Logger.getLogger("wtd"); + Logger dtwLogger = Logger.getLogger("dtw"); + + String outputFileNameObs = Util.getStringValueFromPropertyMap(rbMap, + "obs.skims.output.file"); + String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, + "his.skims.output.file"); + + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + + FileWriter writer; + PrintWriter outStreamObs = null; + PrintWriter outStreamHis = null; + + PrintWriter[] outStreamObsTod = new PrintWriter[SKIM_PERIOD_LABELS.length]; + PrintWriter[] outStreamHisTod = new PrintWriter[SKIM_PERIOD_LABELS.length]; + + if (!outputFileNameObs.isEmpty() || !outputFileNameHis.isEmpty()) + { + + anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + bestPathUEC = logsumHelper.getBestTransitPathCalculator(); + + wtw = new WalkTransitWalkSkimsCalculator(rbMap); + wtw.setup(rbMap, wtwLogger, bestPathUEC); + + wtd = new WalkTransitDriveSkimsCalculator(rbMap); + wtd.setup(rbMap, wtdLogger, bestPathUEC); + + dtw = new DriveTransitWalkSkimsCalculator(rbMap); + dtw.setup(rbMap, dtwLogger, bestPathUEC); + + String heading = "Seq,Id"; + + heading += ",obOrigMgra,obDestMgra,obPeriod"; + heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); + heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); + heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); + heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); + heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); + + heading += ",ObsSeq,Id,ibOrigMgra,ibDestMgra,ibPeriod"; + heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); + heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); + heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); + heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); + heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); + + if (!outputFileNameObs.isEmpty()) + { + try + { + // create an output stream for the mode choice estimation + // file + // with + // observed TOD + writer = new FileWriter(new File(outputFileNameObs)); + outStreamObs = new PrintWriter(new BufferedWriter(writer)); + + // create an array of similar files, 1 for each TOD period, + // for + // TOD + // choice estimation + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + int dotIndex = outputFileNameObs.lastIndexOf('.'); + String newName = outputFileNameObs.substring(0, dotIndex) + "_" + + SKIM_PERIOD_LABELS[i] + outputFileNameObs.substring(dotIndex); + writer = new FileWriter(new File(newName)); + outStreamObsTod[i] = new PrintWriter(new BufferedWriter(writer)); + } + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileNameObs)); + throw new RuntimeException(e); + } + + outStreamObs.println("obs" + heading); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamObsTod[i].println("obs" + heading); + } + + if (!outputFileNameHis.isEmpty()) + { + try + { + writer = new FileWriter(new File(outputFileNameHis)); + outStreamHis = new PrintWriter(new BufferedWriter(writer)); + + // create an array of similar files, 1 for each TOD period, + // for + // TOD + // choice estimation + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + int dotIndex = outputFileNameHis.lastIndexOf('.'); + String newName = outputFileNameHis.substring(0, dotIndex) + "_" + + SKIM_PERIOD_LABELS[i] + outputFileNameHis.substring(dotIndex); + writer = new FileWriter(new File(newName)); + outStreamHisTod[i] = new PrintWriter(new BufferedWriter(writer)); + } + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileNameHis)); + throw new RuntimeException(e); + } + + outStreamHis.println("his" + heading); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamHisTod[i].println("his" + heading); + } + + } + + Logger[] loggers = new Logger[4]; + loggers[0] = autoLogger; + loggers[1] = autoLogger; + loggers[2] = wtdLogger; + loggers[3] = dtwLogger; + + float defaultVOT= 15f; //$15 default VOT + + int[] odt = new int[5]; + + if (!outputFileNameObs.isEmpty()) + { + TableDataSet obsTds = getOnBoardSurveyTableDataSet(rbMap); + int[][] obsOdts = getOnBoardSurveyOrigDestTimes(obsTds); + + // write skims data for on-board survey records + int seq = 1; + for (int[] obsOdt : obsOdts) + { + // write outbound direction + odt[0] = obsOdt[0]; // orig + odt[1] = obsOdt[1]; // dest + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[obsOdt[2]]; // depart skim + // period + odt[3] = obsOdt[3]; + odt[4] = obsOdt[4]; + + if (odt[0] == 0 || odt[1] == 0) + { + outStreamObs.println(String.format("%d,%d,%d,%d,%d", seq, odt[4], odt[0], + odt[1], odt[2])); + seq++; + continue; + } + + // index + + // for debugging a specific mgra pair + // odt[0] = 25646; + // odt[1] = 4319; + // odt[2] = 1; + + boolean debugFlag = false; + if (odt[0] == 25646 && odt[1] == 4319) debugFlag = true; + + writeSkimsToFile(seq, outStreamObs, debugFlag, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); + + // set odt[2] to be each skim priod index (1,2,3) and write a + // separate + // output file + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + odt[2] = i + 1; + writeSkimsToFile(seq, outStreamObsTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, + loggers); + } + + // write inbound direction + odt[0] = obsOdt[1]; // dest + odt[1] = obsOdt[0]; // orig + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[obsOdt[3]]; // arrival + // skim + // period + // index + + outStreamObs.print(","); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamObsTod[i].print(","); + + writeSkimsToFile(seq, outStreamObs, debugFlag, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); + + // set odt[2] to be each skim priod index (1,2,3) and write a + // separate + // output file + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + odt[2] = i + 1; + writeSkimsToFile(seq, outStreamObsTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, + loggers); + } + + if (outStreamObs != null) + { + outStreamObs.println(""); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamObsTod[i].println(""); + } + + if (seq % 1000 == 0) logger.info("wrote OBS record: " + seq); + + seq++; + } + if (outStreamObs != null) + { + outStreamObs.close(); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamObsTod[i].close(); + } + } + + if (!outputFileNameHis.isEmpty()) + { + TableDataSet hisTds = getHomeInterviewSurveyTableDataSet(rbMap); + int[][] hisOdts = getHomeInterviewSurveyOrigDestTimes(hisTds); + + // write skims data for home interview survey records + int seq = 1; + for (int[] hisOdt : hisOdts) + { + // write outbound direction + odt[0] = hisOdt[0]; // orig + odt[1] = hisOdt[1]; // dest + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim + // period + // index + writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); + + // set odt[2] to be each skim priod index (1,2,3) and write a + // separate + // output file + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + odt[2] = i + 1; + writeSkimsToFile(seq, outStreamHisTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, + loggers); + } + + // write inbound direction + odt[0] = hisOdt[1]; // dest + odt[1] = hisOdt[0]; // orig + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[3]]; // arrival + // skim + // period + // index + + outStreamHis.print(","); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamHisTod[i].print(","); + + writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); + + // set odt[2] to be each skim priod index (1,2,3) and write a + // separate + // output file + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + { + odt[2] = i + 1; + writeSkimsToFile(seq, outStreamHisTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, + loggers); + } + + if (outStreamHis != null) + { + outStreamHis.println(""); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamHisTod[i].println(""); + } + + if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); + + seq++; + } + if (outStreamHis != null) + { + outStreamHis.close(); + for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) + outStreamHisTod[i].close(); + } + } + + } + + private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, + int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, + WalkTransitDriveSkimsCalculator wtd, DriveTransitWalkSkimsCalculator dtw, float vot, + Logger[] loggers) + { + + Logger autoLogger = loggers[0]; + Logger wtwLogger = loggers[1]; + Logger wtdLogger = loggers[2]; + Logger dtwLogger = loggers[3]; + + int[][] bestTapPairs = null; + double[][] returnedSkims = null; + + if (outStream != null) + outStream.print(String.format("%d,%d,%d,%d,%d", sequence, odt[4], odt[0], odt[1], + odt[2])); + + double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], vot,loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); + + if (outStream != null) + { + String autoRecord = getAutoSkimsRecord(skims); + outStream.print(autoRecord); + } + + skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); + + if (outStream != null) + { + String nmRecord = getAutoSkimsRecord(skims); + outStream.print(nmRecord); + } + + /* TODO: + * Fix the following code + + + BestTransitPathCalculator bestTransitPathCalculator = wtw.getBestPathUEC(); + + bestTransitPathCalculator.findBestWalkTransitWalkTaps(walkDmu, odt[2], odt[0], odt[1], loggingEnabled, wtwLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); + else + { + returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, bestPathUEC.getBestAccessTime(i), + bestPathUEC.getBestEgressTime(i), bestTapPairs[i][0], bestTapPairs[i][1], + odt[2], loggingEnabled); + } + } + if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + if (outStream != null) + { + String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.print(wtwRecord); + } + + bestTapPairs = wtd.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtdLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = wtd.getNullTransitSkims(i); + else + { + returnedSkims[i] = wtd.getWalkTransitDriveSkims(i, + bestPathUEC.getBestAccessTime(i), bestPathUEC.getBestEgressTime(i), + bestTapPairs[i][0], bestTapPairs[i][1], odt[2], loggingEnabled); + } + } + if (loggingEnabled) wtd.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + if (outStream != null) + { + String wtdRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.print(wtdRecord); + } + + bestTapPairs = dtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, dtwLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = dtw.getNullTransitSkims(i); + else + { + returnedSkims[i] = dtw.getDriveTransitWalkSkims(i, + bestPathUEC.getBestAccessTime(i), bestPathUEC.getBestEgressTime(i), + bestTapPairs[i][0], bestTapPairs[i][1], odt[2], loggingEnabled); + } + } + if (loggingEnabled) dtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + if (outStream != null) + { + String dtwRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.print(dtwRecord); + } + */ + } + + /** + * Start the matrix server + * + * @param rb + * is a ResourceBundle for the properties file for this + * application + */ + private void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[][] of skim values with the first dimesion the + * ride mode indices and second dimention the skim categories + */ + private String getTransitSkimsRecord(int[] odt, double[][] skims) + { + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + for (int j = 0; j < skims[i].length; j++) + tableRecord += String.format(",%.5f", skims[i][j]); + } + + return tableRecord; + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[] of skim values + */ + private String getAutoSkimsRecord(double[] skims) + { + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + tableRecord += String.format(",%.5f", skims[i]); + } + + return tableRecord; + + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getTransitSkimsHeaderRecord(String transitServiceLabel, String[] skimNames) + { + + Modes.TransitMode[] mode = Modes.TransitMode.values(); + + String heading = ""; + + for (int i = 0; i < mode.length; i++) + { + for (int j = 0; j < skimNames.length; j++) + heading += String.format(",%s_%s_%s", transitServiceLabel, mode[i], + skimNames[j]); + } + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getAutoSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + private TableDataSet getOnBoardSurveyTableDataSet(HashMap rbMap) + { + + String obsFileName = Util.getStringValueFromPropertyMap(rbMap, OBS_DATA_RECORDS_FILE_KEY); + if (obsFileName == null) + { + logger.error("Error getting the filename from the properties file for the Sandag on-board survey data records file."); + logger.error("Properties file target: " + OBS_DATA_RECORDS_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + OBS_DATA_RECORDS_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFile(new File(obsFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag on-board survey data records file: %s into TableDataSet object.", + obsFileName)); + throw new RuntimeException(e); + } + + } + + private TableDataSet getHomeInterviewSurveyTableDataSet(HashMap rbMap) + { + + String hisFileName = Util.getStringValueFromPropertyMap(rbMap, HIS_DATA_RECORDS_FILE_KEY); + if (hisFileName == null) + { + logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); + logger.error("Properties file target: " + HIS_DATA_RECORDS_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + HIS_DATA_RECORDS_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFile(new File(hisFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", + hisFileName)); + throw new RuntimeException(e); + } + + } + + private int[][] getOnBoardSurveyOrigDestTimes(TableDataSet obsTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[obsTds.getRowCount()][5]; + + int[] origs = obsTds.getColumnAsInt(OBS_ORIG_MGRA); + int[] dests = obsTds.getColumnAsInt(OBS_DEST_MGRA); + int[] departs = obsTds.getColumnAsInt(OBS_OUT_TOUR_PERIOD); + int[] arrives = obsTds.getColumnAsInt(OBS_IN_TOUR_PERIOD); + int[] ids = obsTds.getColumnAsInt(OBS_UNIQUE_ID); + + for (int r = 1; r <= obsTds.getRowCount(); r++) + { + odts[r - 1][0] = origs[r - 1]; + odts[r - 1][1] = dests[r - 1]; + odts[r - 1][2] = departs[r - 1]; + odts[r - 1][3] = arrives[r - 1]; + odts[r - 1][4] = ids[r - 1]; + } + + return odts; + } + + private int[][] getHomeInterviewSurveyOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][4]; + + int[] origs = hisTds.getColumnAsInt(HIS_ORIG_MGRA); + int[] dests = hisTds.getColumnAsInt(HIS_DEST_MGRA); + int[] departs = hisTds.getColumnAsInt(HIS_DEPART_PERIOD); + int[] arrives = hisTds.getColumnAsInt(HIS_ARRIVE_PERIOD); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + odts[r - 1][0] = origs[r - 1]; + odts[r - 1][1] = dests[r - 1]; + odts[r - 1][2] = departs[r - 1]; + odts[r - 1][3] = arrives[r - 1]; + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + SkimsAppender appender = new SkimsAppender(); + + appender.startMatrixServer(rb); + appender.runSkimsAppender(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java new file mode 100644 index 0000000..deeb3b8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java @@ -0,0 +1,331 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTripModeChoiceDMU; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +public final class StopLocationEstimationMcLogsumsAppender + extends McLogsumsAppender +{ + + private static final int DEBUG_EST_RECORD1 = 266; + private static final int DEBUG_EST_RECORD2 = -1; + + /* + * for stop location choice estimation file + */ + private static final int PORTION_FIELD = 1; + private static final int SAMPNO_FIELD = 2; + private static final int PERNO_FIELD = 3; + private static final int TOUR_ID_FIELD = 4; + private static final int TRIPNO_FIELD = 5; + private static final int STOPID_FIELD = 6; + private static final int STOPNO_FIELD = 7; + + private static final int ORIG_MGRA_FIELD = 16; + private static final int DEST_MGRA_FIELD = 23; + private static final int CHOSEN_MGRA_FIELD = 17; + private static final int MGRA1_FIELD = 49; + + private static final int TOUR_DEPART_PERIOD_FIELD = 40; + private static final int TOUR_ARRIVE_PERIOD_FIELD = 41; + private static final int TRIP_START_PERIOD_FIELD = 29; + private static final int TOUR_MODE_FIELD = 30; + private static final int INCOME_FIELD = 24; + private static final int ADULTS_FIELD = 47; + private static final int AUTOS_FIELD = 28; + private static final int HHSIZE_FIELD = 27; + private static final int GENDER_FIELD = 26; + private static final int OUT_STOPS_FIELD = 45; + private static final int IN_STOPS_FIELD = 46; + private static final int FIRST_TRIP_FIELD = 43; + private static final int LAST_TRIP_FIELD = 44; + private static final int PURPOSE_FIELD = 12; + private static final int AGE_FIELD = 25; + private static final int DIR_FIELD = 42; + private static final int J_TOUR_ID_FIELD = 10; + private static final int J_TOUR_PARTICIPANTS_FIELD = 11; + private static final int NUM_MGRA_FIELDS = 30; + + public StopLocationEstimationMcLogsumsAppender(HashMap rbMap) + { + super(rbMap); + + debugEstimationFileRecord1 = DEBUG_EST_RECORD1; + debugEstimationFileRecord2 = DEBUG_EST_RECORD2; + + numMgraFields = NUM_MGRA_FIELDS; + } + + private void runLogsumAppender(ResourceBundle rb) + { + + totalTime1 = 0; + totalTime2 = 0; + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + mgraSetForLogsums = new int[numMgraFields + 1]; + + // allocate the logsums array for the chosen destination alternative + tripModeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][2]; + + departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + "slc.est.skims.output.file"); + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + int dotIndex = outputFileName.indexOf("."); + String baseName = outputFileName.substring(0, dotIndex); + String extension = outputFileName.substring(dotIndex); + + String outputName = baseName + extension; + + PrintWriter outStream = null; + + try + { + outStream = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + writeDcFile(rbMap, outStream); + + logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); + logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); + + } + + private void writeDcFile(HashMap rbMap, PrintWriter outStream) + { + + outStream + .print("seq,portion,sampn,perno,tour_id,tripno,stopid,stopno,chosenMgra,chosenMgraLogsumIK,chosenMgraLogsumKJ"); + + // print each set of sample destMgra and the depart/arrive logsum + // fieldnames + // to file 1. + // print each set of sample destMgra and the chosen depart/arrive logsum + // fieldname to file 2. + for (int m = 1; m < tripModeChoiceLogsums.length; m++) + { + outStream.print(",sampleMgra_" + m); + outStream.print(",sampleLogsumIK_" + m); + outStream.print(",sampleLogsumKJ_" + m); + } + outStream.print("\n"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + SandagTripModeChoiceDMU mcDmuObject = new SandagTripModeChoiceDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + int[] mgraSet = mgras[i]; + + odtSet[0] = seq; + + outStream.print(seq + "," + odtSet[PORTION] + "," + odtSet[SAMPNO] + "," + + odtSet[PERNO] + "," + odtSet[TOUR_ID] + "," + odtSet[TRIPNO] + "," + + odtSet[STOPID] + "," + odtSet[STOPNO]); + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + + try + { + calculateTripModeChoiceLogsums(rbMap, mcModel[category], mcDmuObject, odtSet, + mgraSet); + } catch (Exception e) + { + logger.error("exception caught processing survey record for i = " + i); + throw new RuntimeException(); + } + + outStream.print("," + odtSet[CHOSEN_MGRA]); + outStream + .printf(",%.8f,%.8f", tripModeChoiceLogsums[0][0], tripModeChoiceLogsums[0][1]); + + // write logsum sets for each dest in the sample to file 1 + for (int m = 1; m < tripModeChoiceLogsums.length; m++) + { + outStream.print("," + mgraSet[m - 1]); + outStream.printf(",%.8f,%.8f", tripModeChoiceLogsums[m][0], + tripModeChoiceLogsums[m][1]); + } + outStream.print("\n"); + + if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); + + seq++; + } + + outStream.close(); + + } + + private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] tourDeparts = hisTds.getColumnAsInt(TOUR_DEPART_PERIOD_FIELD); + int[] tourArrives = hisTds.getColumnAsInt(TOUR_ARRIVE_PERIOD_FIELD); + int[] tripStarts = hisTds.getColumnAsInt(TRIP_START_PERIOD_FIELD); + + int[] direction = hisTds.getColumnAsInt(DIR_FIELD); + + int[] portion = hisTds.getColumnAsInt(PORTION_FIELD); + int[] sampno = hisTds.getColumnAsInt(SAMPNO_FIELD); + int[] perno = hisTds.getColumnAsInt(PERNO_FIELD); + int[] tour_id = hisTds.getColumnAsInt(TOUR_ID_FIELD); + int[] tripno = hisTds.getColumnAsInt(TRIPNO_FIELD); + int[] stopid = hisTds.getColumnAsInt(STOPID_FIELD); + int[] stopno = hisTds.getColumnAsInt(STOPNO_FIELD); + int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); + int[] jTourId = hisTds.getColumnAsInt(J_TOUR_ID_FIELD); + int[] jTourParticipants = hisTds.getColumnAsInt(J_TOUR_PARTICIPANTS_FIELD); + int[] income = hisTds.getColumnAsInt(INCOME_FIELD); + int[] mode = hisTds.getColumnAsInt(TOUR_MODE_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] chosen = hisTds.getColumnAsInt(CHOSEN_MGRA_FIELD); + int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); + int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); + int[] outStops = hisTds.getColumnAsInt(OUT_STOPS_FIELD); + int[] inStops = hisTds.getColumnAsInt(IN_STOPS_FIELD); + int[] firstTrip = hisTds.getColumnAsInt(FIRST_TRIP_FIELD); + int[] lastTrip = hisTds.getColumnAsInt(LAST_TRIP_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][PORTION] = portion[r - 1]; + odts[r - 1][SAMPNO] = sampno[r - 1]; + odts[r - 1][PERNO] = perno[r - 1]; + odts[r - 1][TOUR_ID] = tour_id[r - 1]; + odts[r - 1][TRIPNO] = tripno[r - 1]; + odts[r - 1][STOPID] = stopid[r - 1]; + odts[r - 1][STOPNO] = stopno[r - 1]; + + odts[r - 1][DEPART_PERIOD] = tourDeparts[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = tourArrives[r - 1]; + odts[r - 1][TRIP_PERIOD] = tripStarts[r - 1]; + + odts[r - 1][DIRECTION] = direction[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][CHOSEN_MGRA] = chosen[r - 1]; + odts[r - 1][TOUR_MODE] = mode[r - 1]; + odts[r - 1][INCOME] = income[r - 1]; + odts[r - 1][ADULTS] = adults[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][HHSIZE] = hhsize[r - 1]; + odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; + odts[r - 1][FIRST_TRIP] = firstTrip[r - 1]; + odts[r - 1][LAST_TRIP] = lastTrip[r - 1]; + odts[r - 1][OUT_STOPS] = outStops[r - 1]; + odts[r - 1][IN_STOPS] = inStops[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = purpose[r - 1]; + odts[r - 1][JOINT] = jTourId[r - 1] > 0 ? 1 : 0; + odts[r - 1][PARTYSIZE] = jTourParticipants[r - 1]; + + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + StopLocationEstimationMcLogsumsAppender appender = new StopLocationEstimationMcLogsumsAppender( + rbMap); + + appender.startMatrixServer(rb); + appender.runLogsumAppender(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java new file mode 100644 index 0000000..b6ffadb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java @@ -0,0 +1,502 @@ +package org.sandag.abm.accessibilities; + +import java.io.File; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTripModeChoiceDMU; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.ResourceUtil; + +public final class StopLocationSampleCalculator + extends McLogsumsAppender +{ + + private static final int DEBUG_EST_RECORD = 5; + + /* + * for stop location choice estimation file + */ + private static final int PORTION_FIELD = 1; + private static final int SAMPNO_FIELD = 2; + private static final int PERNO_FIELD = 3; + private static final int TOUR_ID_FIELD = 4; + private static final int TRIPNO_FIELD = 5; + private static final int STOPID_FIELD = 6; + private static final int STOPNO_FIELD = 7; + + private static final int ORIG_MGRA_FIELD = 16; + private static final int DEST_MGRA_FIELD = 23; + private static final int CHOSEN_MGRA_FIELD = 17; + private static final int MGRA1_FIELD = 49; + + private static final int TOUR_DEPART_PERIOD_FIELD = 40; + private static final int TOUR_ARRIVE_PERIOD_FIELD = 41; + private static final int TRIP_START_PERIOD_FIELD = 29; + private static final int TOUR_MODE_FIELD = 30; + private static final int INCOME_FIELD = 24; + private static final int ADULTS_FIELD = 47; + private static final int AUTOS_FIELD = 28; + private static final int HHSIZE_FIELD = 27; + private static final int GENDER_FIELD = 26; + private static final int OUT_STOPS_FIELD = 45; + private static final int IN_STOPS_FIELD = 46; + private static final int FIRST_TRIP_FIELD = 43; + private static final int LAST_TRIP_FIELD = 44; + private static final int TOUR_PURPOSE_FIELD = 12; + private static final int STOP_PURPOSE_FIELD = 12; + private static final int AGE_FIELD = 25; + private static final int DIR_FIELD = 42; + private static final int J_TOUR_ID_FIELD = 10; + private static final int J_TOUR_PARTICIPANTS_FIELD = 11; + private static final int NUM_MGRA_FIELDS = 30; + + private static final String PROPERTIES_UEC_SLC_SOA_CHOICE = "slc.soa.uec.file"; + private static final String PROPERTIES_UEC_STOP_SOA_SIZE = "slc.soa.size.uec.file"; + private static final String PROPERTIES_UEC_STOP_SOA_SIZE_DATA = "slc.soa.size.uec.data.page"; + private static final String PROPERTIES_UEC_STOP_SOA_SIZE_MODEL = "slc.soa.size.uec.model.page"; + + private static final int WORK_STOP_PURPOSE_INDEX = 1; + private static final int UNIV_STOP_PURPOSE_INDEX = 2; + private static final int ESCORT_STOP_PURPOSE_INDEX = 4; + private static final int SHOP_STOP_PURPOSE_INDEX = 5; + private static final int MAINT_STOP_PURPOSE_INDEX = 6; + private static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; + private static final int VISIT_STOP_PURPOSE_INDEX = 8; + private static final int DISCR_STOP_PURPOSE_INDEX = 9; + private static final int MAX_STOP_PURPOSE_INDEX = 9; + + private static final int WORK_STOP_PURPOSE_SOA_SIZE_INDEX = 0; + private static final int UNIV_STOP_PURPOSE_SOA_SIZE_INDEX = 1; + private static final int ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX = 2; + private static final int ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX = 3; + private static final int ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 4; + private static final int ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 5; + private static final int SHOP_STOP_PURPOSE_SOA_SIZE_INDEX = 6; + private static final int MAINT_STOP_PURPOSE_SOA_SIZE_INDEX = 7; + private static final int EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX = 8; + private static final int VISIT_STOP_PURPOSE_SOA_SIZE_INDEX = 9; + private static final int DISCR_STOP_PURPOSE_SOA_SIZE_INDEX = 10; + + private static final int AUTO_DIST_SKIM_INDEX = 2; + + private McLogsumsCalculator logsumHelper; + + private static final float defaultVOT = 15.0f; + + public StopLocationSampleCalculator(HashMap rbMap) + { + super(rbMap); + debugEstimationFileRecord1 = DEBUG_EST_RECORD; + + } + + private void runSampleProbabilitiesCalculator(ResourceBundle rb) + { + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(rbMap); + + SandagTripModeChoiceDMU mcDmuObject = new SandagTripModeChoiceDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + + /* + * SandagStopDCSoaDMU slcSoaDmuObject = new + * SandagStopDCSoaDMU(modelStructure); + * + * String slcSoaUecFile = rbMap.get(PROPERTIES_UEC_SLC_SOA_CHOICE); + * slcSoaUecFile = uecPath + slcSoaUecFile; + * + * ChoiceModelApplication[] slcSoaModel = new + * ChoiceModelApplication[MAX_STOP_PURPOSE_INDEX+1]; + * slcSoaModel[WORK_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, WORK_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[UNIV_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, UNIV_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[ESCORT_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, ESCORT_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[SHOP_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, SHOP_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[EAT_OUT_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, EAT_OUT_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[MAINT_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, MAINT_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[VISIT_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, VISIT_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + * slcSoaModel[DISCR_STOP_PURPOSE_INDEX] = new + * ChoiceModelApplication(slcSoaUecFile, DISCR_STOP_PURPOSE_INDEX, 0, + * rbMap, (VariableTable) slcSoaDmuObject); + */ + + double absoulteDistanceDeviationCoefficient = -0.05; + + int i = DEBUG_EST_RECORD - 1; + + int[] odtSet = estDataOdts[i]; + + int mgra = mgras[i][24]; + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + // int stopPurpose = odtSet[STOP_PURPOSE]; + int stopPurpose = 6; + + double[] logsums = null; + try + { + logsums = calculateTripModeChoiceLogsumForEstimationRecord(rbMap, mcModel[category], + mcDmuObject, odtSet, mgra); + } catch (Exception e) + { + logger.error("exception caught calculating trip mode choice logsum for survey record i = " + + (i + 1)); + throw new RuntimeException(); + } + + double[] slcSoaSizeTerms = null; + try + { + boolean preSchoolInHh = false; + boolean gradeSchoolInHh = false; + boolean highSchoolInHh = false; + double[][] sizeTerms = calculateSlcSoaSizeTerms(rbMap, mgra); + slcSoaSizeTerms = getSlcSoaSizeTermsForStopPurpose(stopPurpose, preSchoolInHh, + gradeSchoolInHh, highSchoolInHh, sizeTerms); + } catch (Exception e) + { + logger.error("exception caught calculating stop location choice size terms for survey record i = " + + (i + 1)); + throw new RuntimeException(); + } + + try + { + int origMgra = odtSet[ORIG_MGRA]; + int destMgra = odtSet[DEST_MGRA]; + int departPeriod = odtSet[TRIP_PERIOD]; // depart period + int skimPeriodIndex = modelStructure.getSkimPeriodIndex(departPeriod) + 1; // depart + // skim + + double odDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(origMgra, destMgra, + skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; + double osDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(origMgra, mgra, + skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; + double sdDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(mgra, destMgra, + skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; + double distance = osDist + sdDist - odDist; + int availability = 1; + + boolean walkTransitIsAvailable = false; + if (mgraManager.getMgraWlkTapsDistArray()[mgra][0] != null) + walkTransitIsAvailable = true; + + double util = -999; + if (availability == 1) + util = Math.log(slcSoaSizeTerms[mgra]) + absoulteDistanceDeviationCoefficient + * distance; + + logUtilityCalculation(origMgra, mgra, destMgra, departPeriod, skimPeriodIndex, osDist, + sdDist, odDist, distance, slcSoaSizeTerms[mgra], util, logsums); + } catch (Exception e) + { + logger.error("exception caught calculating and logging utility calculation for survey record i = " + + (i + 1)); + throw new RuntimeException(); + } + + } + + private void logUtilityCalculation(int origMgra, int sampleMgra, int destMgra, + int departPeriodIndex, int skimPeriodIndex, double osDist, double sdDist, + double odDist, double distance, double size, double util, double[] logsums) + { + + // write UEC calculation results to logsum specific log file if + // its the chosen dest and its the chosen time combo + slcSoaLogger.info("Stop Location Sample Probabilities Calculation:"); + slcSoaLogger.info(""); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info("origin mgra = " + origMgra); + slcSoaLogger.info("sample destination mgra = " + sampleMgra); + slcSoaLogger.info("final destination mgra = " + destMgra); + slcSoaLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); + slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(sampleMgra)); + slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); + slcSoaLogger.info("depart period interval = " + departPeriodIndex); + slcSoaLogger.info("skim period index = " + skimPeriodIndex); + slcSoaLogger.info("orig to stop distance = " + osDist); + slcSoaLogger.info("stop to dest distance = " + sdDist); + slcSoaLogger.info("orig to dest distance = " + odDist); + slcSoaLogger.info("distance = " + distance); + slcSoaLogger.info("size = " + size); + slcSoaLogger.info(""); + slcSoaLogger.info("util = size * exp ( -0.05*distance )"); + slcSoaLogger.info("util = " + util); + slcSoaLogger.info("os logsum = " + logsums[0]); + slcSoaLogger.info("sd logsum = " + logsums[1]); + slcSoaLogger + .info("--------------------------------------------------------------------------------------------------------"); + slcSoaLogger.info(""); + + } + + private double[] getSlcSoaSizeTermsForStopPurpose(int stopPurpose, boolean preSchoolInHh, + boolean gradeSchoolInHh, boolean highSchoolInHh, double[][] sizeTerms) + { + + double[] slcSoaSizeTerms = null; + switch (stopPurpose) + { + + case WORK_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case UNIV_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case ESCORT_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; + + // add preschool size term if the hh has a preschool child + if (preSchoolInHh) + { + for (int j = 0; j < sizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) + slcSoaSizeTerms[j] += sizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; + } + + // add grade school size term if the hh has a grade school child + if (gradeSchoolInHh) + { + for (int j = 0; j < sizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) + slcSoaSizeTerms[j] += sizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; + } + + // add high school size term if the hh has a high school child + if (highSchoolInHh) + { + for (int j = 0; j < sizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) + slcSoaSizeTerms[j] += sizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; + } + break; + case SHOP_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case MAINT_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case EAT_OUT_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case VISIT_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case DISCR_STOP_PURPOSE_INDEX: + slcSoaSizeTerms = sizeTerms[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + } + + return slcSoaSizeTerms; + + } + + private double[][] calculateSlcSoaSizeTerms(HashMap rbMap, int sampleMgra) + { + + logger.info(""); + logger.info(""); + logger.info("Calculating Stop Location SOA Size Terms"); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String slcSoaSizeUecFile = rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE); + slcSoaSizeUecFile = uecPath + slcSoaSizeUecFile; + int slcSoaSizeUecData = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE_DATA)); + int slcSoaSizeUecModel = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE_MODEL)); + + IndexValues iv = new IndexValues(); + UtilityExpressionCalculator slcSoaSizeUec = new UtilityExpressionCalculator(new File( + slcSoaSizeUecFile), slcSoaSizeUecModel, slcSoaSizeUecData, rbMap, null); + + ArrayList mgras = mgraManager.getMgras(); + int maxMgra = mgraManager.getMaxMgra(); + int alternatives = slcSoaSizeUec.getNumberOfAlternatives(); + double[][] slcSoaSize = new double[alternatives][maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + iv.setZoneIndex(mgra); + double[] utilities = slcSoaSizeUec.solve(iv, null, null); + + if (mgra == sampleMgra) + slcSoaSizeUec.logAnswersArray(slcSoaLogger, "Stop Location SOA Size Terms, MGRA = " + + mgra); + + // store the size terms + for (int i = 0; i < alternatives; i++) + slcSoaSize[i][mgra] = utilities[i]; + + } + + return slcSoaSize; + + } + + private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] tourDeparts = hisTds.getColumnAsInt(TOUR_DEPART_PERIOD_FIELD); + int[] tourArrives = hisTds.getColumnAsInt(TOUR_ARRIVE_PERIOD_FIELD); + int[] tripStarts = hisTds.getColumnAsInt(TRIP_START_PERIOD_FIELD); + + int[] direction = hisTds.getColumnAsInt(DIR_FIELD); + + int[] portion = hisTds.getColumnAsInt(PORTION_FIELD); + int[] sampno = hisTds.getColumnAsInt(SAMPNO_FIELD); + int[] perno = hisTds.getColumnAsInt(PERNO_FIELD); + int[] tour_id = hisTds.getColumnAsInt(TOUR_ID_FIELD); + int[] tripno = hisTds.getColumnAsInt(TRIPNO_FIELD); + int[] stopid = hisTds.getColumnAsInt(STOPID_FIELD); + int[] stopno = hisTds.getColumnAsInt(STOPNO_FIELD); + int[] tourPurpose = hisTds.getColumnAsInt(TOUR_PURPOSE_FIELD); + int[] stopPurpose = hisTds.getColumnAsInt(STOP_PURPOSE_FIELD); + int[] jTourId = hisTds.getColumnAsInt(J_TOUR_ID_FIELD); + int[] jTourParticipants = hisTds.getColumnAsInt(J_TOUR_PARTICIPANTS_FIELD); + int[] income = hisTds.getColumnAsInt(INCOME_FIELD); + int[] mode = hisTds.getColumnAsInt(TOUR_MODE_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] chosen = hisTds.getColumnAsInt(CHOSEN_MGRA_FIELD); + int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); + int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); + int[] outStops = hisTds.getColumnAsInt(OUT_STOPS_FIELD); + int[] inStops = hisTds.getColumnAsInt(IN_STOPS_FIELD); + int[] firstTrip = hisTds.getColumnAsInt(FIRST_TRIP_FIELD); + int[] lastTrip = hisTds.getColumnAsInt(LAST_TRIP_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][PORTION] = portion[r - 1]; + odts[r - 1][SAMPNO] = sampno[r - 1]; + odts[r - 1][PERNO] = perno[r - 1]; + odts[r - 1][TOUR_ID] = tour_id[r - 1]; + odts[r - 1][TRIPNO] = tripno[r - 1]; + odts[r - 1][STOPID] = stopid[r - 1]; + odts[r - 1][STOPNO] = stopno[r - 1]; + + odts[r - 1][DEPART_PERIOD] = tourDeparts[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = tourArrives[r - 1]; + odts[r - 1][TRIP_PERIOD] = tripStarts[r - 1]; + + odts[r - 1][DIRECTION] = direction[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][CHOSEN_MGRA] = chosen[r - 1]; + odts[r - 1][TOUR_MODE] = mode[r - 1]; + odts[r - 1][INCOME] = income[r - 1]; + odts[r - 1][ADULTS] = adults[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][HHSIZE] = hhsize[r - 1]; + odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; + odts[r - 1][FIRST_TRIP] = firstTrip[r - 1]; + odts[r - 1][LAST_TRIP] = lastTrip[r - 1]; + odts[r - 1][OUT_STOPS] = outStops[r - 1]; + odts[r - 1][IN_STOPS] = inStops[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = tourPurpose[r - 1]; + odts[r - 1][STOP_PURPOSE] = stopPurpose[r - 1]; + odts[r - 1][JOINT] = jTourId[r - 1] > 0 ? 1 : 0; + odts[r - 1][PARTYSIZE] = jTourParticipants[r - 1]; + + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + StopLocationSampleCalculator appender = new StopLocationSampleCalculator(rbMap); + + appender.startMatrixServer(rb); + appender.runSampleProbabilitiesCalculator(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java new file mode 100644 index 0000000..0f37752 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java @@ -0,0 +1,51 @@ +package org.sandag.abm.accessibilities; + + +public class StoredTransitSkimData +{ + + private static StoredTransitSkimData objInstance = null; + private int numSets; + + // these arrays are shared by McLogsumsAppender objects and are used by wtw, wtd, and dtw calculators. + private double[][][][][] storedWtwDepartPeriodTapTapSkims; + private double[][][][][] storedWtdDepartPeriodTapTapSkims; + private double[][][][][] storedDtwDepartPeriodTapTapSkims; + + + private StoredTransitSkimData(){ + this.numSets = numSets; + } + + public static synchronized StoredTransitSkimData getInstance( int numSets, int numPeriods, int maxTap ) + { + if (objInstance == null) { + objInstance = new StoredTransitSkimData(); + objInstance.setupStoredDataArrays(numSets, numPeriods, maxTap ); + return objInstance; + } + else { + return objInstance; + } + } + + private void setupStoredDataArrays(int numSets, int numPeriods, int maxTap ){ + storedWtwDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; + storedWtdDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; + storedDtwDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; + } + + public double[][][][][] getStoredWtwDepartPeriodTapTapSkims() { + return storedWtwDepartPeriodTapTapSkims; + } + + public double[][][][][] getStoredWtdDepartPeriodTapTapSkims() { + return storedWtdDepartPeriodTapTapSkims; + } + + public double[][][][][] getStoredDtwDepartPeriodTapTapSkims() { + return storedDtwDepartPeriodTapTapSkims; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java new file mode 100644 index 0000000..51644a5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java @@ -0,0 +1,133 @@ +package org.sandag.abm.accessibilities; + +import java.util.concurrent.ConcurrentHashMap; +import java.util.HashMap; + + +public class StoredUtilityData +{ + + private static StoredUtilityData objInstance = null; + public static final float default_utility = -999; + + // these arrays are shared by multiple BestTransitPathCalculator objects in a distributed computing environment + private float[][] storedWalkAccessUtils; // dim#1: MGRA id, dim#2: TAP id + private float[][] storedDriveAccessUtils; // dim#1: TAZ id, dim#2: TAP id + private float[][] storedWalkEgressUtils; // dim#1: TAP id, dim#2: MGRA id + private float[][] storedDriveEgressUtils; // dim#1: TAP id, dim#2: TAZ id + + // {0:WTW, 1:WTD, 2:DTW} -> TOD period number -> pTAP*100000+aTAP -> utility + private HashMap>> storedDepartPeriodTapTapUtils; + + + private StoredUtilityData(){ + } + + public static synchronized StoredUtilityData getInstance( int maxMgra, int maxTap, int maxTaz, int[] accEgrSegments, int[] periods) + { + if (objInstance == null) { + objInstance = new StoredUtilityData(); + objInstance.setupStoredDataArrays( maxMgra, maxTap, maxTaz, accEgrSegments, periods); + return objInstance; + } + else { + return objInstance; + } + } + + private void setupStoredDataArrays( int maxMgra, int maxTap, int maxTaz, int[] accEgrSegments, int[] periods){ + // dimension the arrays + storedWalkAccessUtils = new float[maxMgra + 1][maxTap + 1]; + storedDriveAccessUtils = new float[maxTaz + 1][maxTap + 1]; + storedWalkEgressUtils = new float[maxTap + 1][maxMgra + 1]; + storedDriveEgressUtils = new float[maxTap + 1][maxTaz + 1]; + // assign default values to array elements + for (int i=0; i<=maxMgra; i++) + for (int j=0; j<=maxTap; j++) { + storedWalkAccessUtils[i][j] = default_utility; + storedWalkEgressUtils[j][i] = default_utility; + } + // assign default values to array elements + for (int i=0; i<=maxTaz; i++) + for (int j=0; j<=maxTap; j++) { + storedDriveAccessUtils[i][j] = default_utility; + storedDriveEgressUtils[j][i] = default_utility; + } + + //put into concurrent hashmap + storedDepartPeriodTapTapUtils = new HashMap>>(); + for(int i=0; i>()); + for(int j=0; j> hm = storedDepartPeriodTapTapUtils.get(accEgrSegments[i]); + hm.put(periods[j], new ConcurrentHashMap()); //key method paTapKey below + } + } + } + + public float[][] getStoredWalkAccessUtils() { + return storedWalkAccessUtils; + } + + public float[][] getStoredDriveAccessUtils() { + return storedDriveAccessUtils; + } + + public float[][] getStoredWalkEgressUtils() { + return storedWalkEgressUtils; + } + + public float[][]getStoredDriveEgressUtils() { + return storedDriveEgressUtils; + } + + public HashMap>> getStoredDepartPeriodTapTapUtils() { + return storedDepartPeriodTapTapUtils; + } + + //create p to a hash key - up to 99,999 + public long paTapKey(int p, int a) { + return(p * 100000 + a); + } + + //convert double array to float array + public float[] d2f(double[] d) { + float[] f = new float[d.length]; + for(int i=0; i rbMap) + { + super(rbMap); + + debugEstimationFileRecord1 = DEBUG_EST_RECORD1; + debugEstimationFileRecord2 = DEBUG_EST_RECORD2; + + } + + private void runLogsumAppender(ResourceBundle rb) + { + + totalTime1 = 0; + totalTime2 = 0; + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + mgraSetForLogsums = new int[numMgraFields + 1]; + + // allocate the logsums array for the chosen destination alternative + modeChoiceLogsums = new double[1][]; + + departArriveLogsums = new double[1][departArriveCombinations.length]; + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + "tod.est.skims.output.file"); + + PrintWriter outStream = null; + + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + try + { + outStream = new PrintWriter( + new BufferedWriter(new FileWriter(new File(outputFileName)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + writeTodFile(rbMap, outStream); + + logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); + logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); + + } + + private void writeTodFile(HashMap rbMap, PrintWriter outStream2) + { + + // print the chosen destMgra and the depart/arrive logsum field names to + // the + // file + outStream2.print("seq,hisseq,chosenMgra"); + for (String[] labels : departArriveCombinationLabels) + { + outStream2.print(",logsum" + labels[0] + labels[1]); + } + outStream2.print("\n"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getTodEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + int[] mgraSet = mgras[i]; + + odtSet[0] = seq; + + outStream2.print(seq + "," + odtSet[SAMPNO]); + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + + int[] departAvailable = {-1, 1, 1, 1, 1, 1}; + int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; + calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], + mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); + + // write chosen dest and logsums to both files + outStream2.print("," + odtSet[DEST_MGRA]); + for (double logsum : departArriveLogsums[0]) + { + outStream2.printf(",%.8f", logsum); + } + outStream2.print("\n"); + + if (seq % 1000 == 0) logger.info("wrote TOD Estimation file record: " + seq); + + seq++; + } + + outStream2.close(); + + } + + private int[][] getTodEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); + int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); + + int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] workTourMode = hisTds.getColumnAsInt(WORK_TOUR_MODE_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][SAMPNO] = hisseq[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = SUBTOUR_PURPOSE; + + odts[r - 1][DEPART_PERIOD] = departs[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][WORK_TOUR_MODE] = workTourMode[r - 1]; + + } + + return odts; + } + + public static void main(String[] args) + { + + long startTime = System.currentTimeMillis(); + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + SubtourTodEstimationMcLogsumsAppender appender = new SubtourTodEstimationMcLogsumsAppender( + rbMap); + + appender.startMatrixServer(rb); + appender.runLogsumAppender(rb); + + /* + * used this to read/parse the UEC expressions - debugging the UEC + * sheet. + * + * HashMap rbMap = + * ResourceUtil.changeResourceBundleIntoHashMap(rb); + * + * String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + * String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + * mcUecFile = uecPath + mcUecFile; + * + * ModelStructure modelStructure = new SandagModelStructure(); + * SandagAppendMcLogsumDMU mcDmuObject = new + * SandagAppendMcLogsumDMU(modelStructure); ChoiceModelApplication + * mcModel = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, + * rbMap, (VariableTable) mcDmuObject); + */ + + System.out.println("total runtime = " + ((System.currentTimeMillis() - startTime) / 1000) + + " seconds."); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java new file mode 100644 index 0000000..c3985af --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java @@ -0,0 +1,45 @@ +package org.sandag.abm.accessibilities; + +public class TransitPath implements Comparable{ + + public int oMaz; + public int dMaz; + public int pTap; + public int aTap; + public int set; + public int accEgr; + public float accUtil; + public float tapTapUtil; + public float egrUtil; + public static final int NA = -999; + + public TransitPath(int oMaz, int dMaz, int pTap, int aTap, int set, int accEgr, float accUtil, float tapTapUtil, float egrUtil) { + this.oMaz = oMaz; + this.dMaz = dMaz; + this.pTap = pTap; + this.aTap = aTap; + this.set = set; + this.accEgr = accEgr; + this.accUtil = accUtil; + this.tapTapUtil = tapTapUtil; + this.egrUtil = egrUtil; + } + + public float getTotalUtility() { + return(accUtil + tapTapUtil + egrUtil); + } + + @Override + public int compareTo(TransitPath o) { + + //return compareTo value + if ( getTotalUtility() < o.getTotalUtility() ) { + return -1; + } else if (getTotalUtility() == o.getTotalUtility()) { + return 0; + } else { + return 1; + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java new file mode 100644 index 0000000..dab1318 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java @@ -0,0 +1,569 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +public final class TripSkimsAppender +{ + + protected transient Logger logger = Logger.getLogger(TripSkimsAppender.class); + + /* + * for trip mode choice estimation files + */ + private static final String HIS_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; + + // private static final int HIS_SAMPNO = 2; + // private static final int HIS_ORIG_MGRA = 188; + // private static final int HIS_DEST_MGRA = 189; + // private static final int HIS_DEPART_PERIOD = 192; + + private static final int HIS_SAMPNO = 1; + private static final int HIS_ORIG_MGRA = 2; + private static final int HIS_DEST_MGRA = 3; + private static final int HIS_DEPART_PERIOD = 5; + + // private static final int HIS_SAMPNO = 2; + // private static final int HIS_ORIG_MGRA = 247; + // private static final int HIS_DEST_MGRA = 248; + // private static final int HIS_DEPART_PERIOD = 325; + + // survey periods are: + // 0=not used, + // 1=03:00-05:59, + // 2=06:00-08:59, + // 3=09:00-11:59, + // 4=12:00-15:29, + // 5=15:30-18:59, + // 6=19:00-02:59 + // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), + // 6=3(OP) + + // define a conversion array to convert period values in the survey file to + // skim + // period indices used in this propgram: 1=am peak, 2=pm peak, 3=off-peak. + private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; + private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; + + private static int debugOrigMgra = 0; + private static int debugDestMgra = 0; + private static int departModelPeriod = 0; + + private MatrixDataServerIf ms; + private BestTransitPathCalculator bestPathUEC; + + private static final float defaultVOT = 15.0f; + + private TripSkimsAppender() + { + } + + private void runSkimsAppender(ResourceBundle rb) + { + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + Logger autoLogger = Logger.getLogger("auto"); + Logger wtwLogger = Logger.getLogger("wtw"); + Logger wtdLogger = Logger.getLogger("wtd"); + Logger dtwLogger = Logger.getLogger("dtw"); + + String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, + "his.trip.skims.output.file"); + + FileWriter writer; + PrintWriter outStreamHis = null; + + McLogsumsAppender logsumHelper = new McLogsumsAppender(rbMap); + bestPathUEC = logsumHelper.getBestTransitPathCalculator(); + + AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); + WalkTransitWalkSkimsCalculator wtw = new WalkTransitWalkSkimsCalculator(rbMap); + WalkTransitDriveSkimsCalculator wtd = new WalkTransitDriveSkimsCalculator(rbMap); + DriveTransitWalkSkimsCalculator dtw = new DriveTransitWalkSkimsCalculator(rbMap); + + String heading = "seq,sampno"; + + heading += ",origMgra,destMgra,departPeriod"; + heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); + heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); + heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); + heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); + heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); + + try + { + writer = new FileWriter(new File(outputFileNameHis)); + outStreamHis = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileNameHis)); + throw new RuntimeException(e); + } + outStreamHis.println("his" + heading); + + Logger[] loggers = new Logger[4]; + loggers[0] = autoLogger; + loggers[1] = wtwLogger; + loggers[2] = wtdLogger; + loggers[3] = dtwLogger; + + int[] odt = new int[4]; + + TableDataSet hisTds = getHomeInterviewSurveyTableDataSet(rbMap); + int[][] hisOdts = getHomeInterviewSurveyOrigDestTimes(hisTds); + // 11100, pnrCoaster, mgra 4357 = taz 3641, mgra 26931 = taz 1140 + // int[][] hisOdts = { { 4357, 26931, 5, 0 } }; + // 25040, wlkCoaster, mgra 4989 = taz 3270, mgra 7796 = taz 1986 + // int[][] hisOdts = { { 7796, 4989, 2, 0 } }; + + // if ( debugOrigMgra <= 0 || debugDestMgra <= 0 || departModelPeriod <= + // 0 || departModelPeriod > 6 ) + // { + // logger.error("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); + // System.exit(-1); + // } + // int[][] hisOdts = { { debugOrigMgra, debugDestMgra, + // departModelPeriod, 0 } }; + + // write skims data for home interview survey records + int seq = 1; + for (int[] hisOdt : hisOdts) + { + // write outbound direction + odt[0] = hisOdt[0]; // orig + odt[1] = hisOdt[1]; // dest + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim + // period + odt[3] = hisOdt[3]; + + try + { + + int dummy = 0; + if (seq == 3424) + { + dummy = 1; + } + + writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, loggers); + } catch (Exception e) + { + logger.error("Exception caught processing record: " + seq + " of " + hisOdts.length + + "."); + break; + } + + if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); + + seq++; + } + + outStreamHis.close(); + + } + + private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, + int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, + WalkTransitDriveSkimsCalculator wtd, DriveTransitWalkSkimsCalculator dtw, + Logger[] loggers) + { + + Logger autoLogger = loggers[0]; + Logger wtwLogger = loggers[1]; + Logger wtdLogger = loggers[2]; + Logger dtwLogger = loggers[3]; + + int[][] bestTapPairs = null; + double[][] returnedSkims = null; + + outStream.print(String.format("%d,%d,%d,%d,%d", sequence, odt[3], odt[0], odt[1], odt[2])); + + double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], defaultVOT, loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); + + String autoRecord = getAutoSkimsRecord(skims); + outStream.print(autoRecord); + + skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); + + String nmRecord = getAutoSkimsRecord(skims); + outStream.print(nmRecord); + + /* + * TODO: Fix this code + + bestTapPairs = wtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtwLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); + else + { + returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, BestTransitPathCalculator + .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), + BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], + bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], + odt[2], loggingEnabled); + } + } + if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.print(wtwRecord); + + bestTapPairs = wtd.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtdLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = wtd.getNullTransitSkims(i); + else + { + returnedSkims[i] = wtd.getWalkTransitDriveSkims(i, BestTransitPathCalculator + .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), + BestTransitPathCalculator.findDriveTransitEgressTime(odt[1], + bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], + odt[2], loggingEnabled); + } + } + if (loggingEnabled) wtd.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + String wtdRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.print(wtdRecord); + + bestTapPairs = dtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, dtwLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = dtw.getNullTransitSkims(i); + else + { + returnedSkims[i] = dtw.getDriveTransitWalkSkims(i, BestTransitPathCalculator + .findDriveTransitAccessTime(odt[0], bestTapPairs[i][0]), + BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], + bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], + odt[2], loggingEnabled); + } + } + if (loggingEnabled) dtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + String dtwRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.println(dtwRecord); + */ + } + + /** + * Start the matrix server + * + * @param rb + * is a ResourceBundle for the properties file for this + * application + */ + private void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[][] of skim values with the first dimesion the + * ride mode indices and second dimention the skim categories + */ + private String getTransitSkimsRecord(int[] odt, double[][] skims) + { + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + for (int j = 0; j < skims[i].length; j++) + tableRecord += String.format(",%.5f", skims[i][j]); + } + + return tableRecord; + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[] of skim values + */ + private String getAutoSkimsRecord(double[] skims) + { + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + tableRecord += String.format(",%.5f", skims[i]); + } + + return tableRecord; + + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getTransitSkimsHeaderRecord(String transitServiveLabel, String[] skimNames) + { + + Modes.TransitMode[] mode = Modes.TransitMode.values(); + + String heading = ""; + + for (int i = 0; i < mode.length; i++) + { + for (int j = 0; j < skimNames.length; j++) + heading += String.format(",%s_%s_%s", transitServiveLabel, mode[i], + skimNames[j]); + + } + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getAutoSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + private TableDataSet getHomeInterviewSurveyTableDataSet(HashMap rbMap) + { + + String hisFileName = Util.getStringValueFromPropertyMap(rbMap, HIS_DATA_RECORDS_FILE_KEY); + if (hisFileName == null) + { + logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); + logger.error("Properties file target: " + HIS_DATA_RECORDS_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + HIS_DATA_RECORDS_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFile(new File(hisFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", + hisFileName)); + throw new RuntimeException(e); + } + + } + + private int[][] getHomeInterviewSurveyOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure period(1-6), and his sampno. + int[][] odts = new int[hisTds.getRowCount()][4]; + + int[] sampnos = hisTds.getColumnAsInt(HIS_SAMPNO); + int[] origs = hisTds.getColumnAsInt(HIS_ORIG_MGRA); + int[] dests = hisTds.getColumnAsInt(HIS_DEST_MGRA); + int[] departs = hisTds.getColumnAsInt(HIS_DEPART_PERIOD); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + odts[r - 1][0] = origs[r - 1]; + odts[r - 1][1] = dests[r - 1]; + odts[r - 1][2] = departs[r - 1]; + odts[r - 1][3] = sampnos[r - 1]; + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb = null; + if (args.length == 0) + { + System.out + .println(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else if (args.length == 4) + { + rb = ResourceBundle.getBundle(args[0]); + + debugOrigMgra = Integer.parseInt(args[1]); + debugDestMgra = Integer.parseInt(args[2]); + departModelPeriod = Integer.parseInt(args[3]); + } else + { + System.out + .println("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); + System.exit(-1); + } + + try + { + + MatrixDataServerIf ms = null; + String serverAddress = null; + int serverPort = -1; + + HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + System.out.println(""); + System.out.println(""); + serverAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + + String serverPortString = (String) propertyMap.get("RunModel.MatrixServerPort"); + if (serverPortString != null) serverPort = Integer.parseInt(serverPortString); + + if (serverAddress != null && serverPort > 0) + { + try + { + System.out.println("attempting connection to matrix server " + serverAddress + + ":" + serverPort); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + System.out.println("connected to matrix server " + serverAddress + ":" + + serverPort); + + } catch (Exception e) + { + System.out + .println("exception caught running ctramp model components -- exiting."); + e.printStackTrace(); + throw new RuntimeException(); + } + } + + TazDataManager tazs = TazDataManager.getInstance(propertyMap); + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TapDataManager tapManager = TapDataManager.getInstance(propertyMap); + + // create an appender object and run it + TripSkimsAppender appender = new TripSkimsAppender(); + appender.runSkimsAppender(rb); + + } catch (RuntimeException e) + { + e.printStackTrace(); + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java new file mode 100644 index 0000000..dafb12e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java @@ -0,0 +1,367 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.application.SandagAppendMcLogsumDMU; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +public final class VisitorTourLocationChoiceAppender + extends McLogsumsAppender +{ + + private static final int DEBUG_EST_RECORD1 = 1; + private static final int DEBUG_EST_RECORD2 = -1; + + /* + * for DC estimation file + */ + private static final int SEQ_FIELD = 2; + + // for Atwork subtour DC + // private static final int ORIG_MGRA_FIELD = 79; + // private static final int DEST_MGRA_FIELD = 220; + // private static final int MGRA1_FIELD = 221; + // private static final int PURPOSE_INDEX_OFFSET = 4; + + // for Escort DC + private static final int ORIG_MGRA_FIELD = 76; + private static final int DEST_MGRA_FIELD = 79; + private static final int MGRA1_FIELD = 217; + private static final int PURPOSE_INDEX_OFFSET = 0; + + // for NonMandatory DC + // private static final int ORIG_MGRA_FIELD = 76; + // private static final int DEST_MGRA_FIELD = 79; + // private static final int MGRA1_FIELD = 221; + // private static final int PURPOSE_INDEX_OFFSET = 0; + + private static final int DEPART_PERIOD_FIELD = 189; + private static final int ARRIVE_PERIOD_FIELD = 190; + private static final int INCOME_FIELD = 20; + private static final int ADULTS_FIELD = 32; + private static final int AUTOS_FIELD = 6; + private static final int HHSIZE_FIELD = 5; + private static final int GENDER_FIELD = 38; + private static final int AGE_FIELD = 39; + private static final int PURPOSE_FIELD = 80; + private static final int JOINT_ID_FIELD = 125; + private static final int JOINT_PURPOSE_FIELD = 126; + private static final int JOINT_P1_FIELD = 151; + private static final int JOINT_P2_FIELD = 152; + private static final int JOINT_P3_FIELD = 153; + private static final int JOINT_P4_FIELD = 154; + private static final int JOINT_P5_FIELD = 155; + private static final int NUM_MGRA_FIELDS = 30; + + private static final String OUTPUT_SAMPLE_DEST_LOGSUMS = "output.sample.dest.logsums"; + + public VisitorTourLocationChoiceAppender(HashMap rbMap) + { + super(rbMap); + + debugEstimationFileRecord1 = DEBUG_EST_RECORD1; + debugEstimationFileRecord2 = DEBUG_EST_RECORD2; + + numMgraFields = NUM_MGRA_FIELDS; + } + + private void runLogsumAppender(ResourceBundle rb) + { + + totalTime1 = 0; + totalTime2 = 0; + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + tazs = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + mgraSetForLogsums = new int[numMgraFields + 1]; + + // allocate the logsums array for the chosen destination alternative + modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; + + departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; + + String outputAllKey = Util.getStringValueFromPropertyMap(rbMap, OUTPUT_SAMPLE_DEST_LOGSUMS); + + String outputFileName = Util.getStringValueFromPropertyMap(rbMap, + "dc.est.skims.output.file"); + if (outputFileName == null) + { + logger.info("no output file name was specified in the properties file. Nothing to do."); + return; + } + + int dotIndex = outputFileName.indexOf("."); + String baseName = outputFileName.substring(0, dotIndex); + String extension = outputFileName.substring(dotIndex); + + // output1 is only written if "all" was set in propoerties file + String outputName1 = ""; + if (outputAllKey.equalsIgnoreCase("all")) outputName1 = baseName + "_" + "all" + extension; + + // output1 is written in any case + String outputName2 = baseName + "_" + "chosen" + extension; + + PrintWriter outStream1 = null; + PrintWriter outStream2 = null; + + try + { + if (outputAllKey.equalsIgnoreCase("all")) + outStream1 = new PrintWriter(new BufferedWriter(new FileWriter( + new File(outputName1)))); + outStream2 = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName2)))); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileName)); + throw new RuntimeException(e); + } + + writeDcFile(rbMap, outStream1, outStream2); + + logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); + logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); + + } + + private void writeDcFile(HashMap rbMap, PrintWriter outStream1, + PrintWriter outStream2) + { + + // print the chosen destMgra and the depart/arrive logsum field names to + // both + // files + if (outStream1 != null) outStream1.print("seq,sampno,chosenMgra"); + + // attach the OB and IB period labels to the logsum field names for each + // period + if (outStream1 != null) + { + for (String[] labels : departArriveCombinationLabels) + outStream1.print(",logsum" + labels[0] + labels[1]); + } + + outStream2.print("seq,sampno,chosenMgra,chosenTodLogsum"); + + // print each set of sample destMgra and the depart/arrive logsum + // fieldnames + // to file 1. + // print each set of sample destMgra and the chosen depart/arrive logsum + // fieldname to file 2. + for (int m = 1; m < departArriveLogsums.length; m++) + { + if (outStream1 != null) + { + outStream1.print(",sampleMgra" + m); + for (String[] labels : departArriveCombinationLabels) + outStream1.print(",logsum" + m + labels[0] + labels[1]); + } + + outStream2.print(",sampleMgra" + m); + outStream2.print(",sampleLogsum" + m); + } + if (outStream1 != null) outStream1.print("\n"); + outStream2.print("\n"); + + TableDataSet estTds = getEstimationDataTableDataSet(rbMap); + int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); + + ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; + mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, + rbMap, (VariableTable) mcDmuObject); + mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); + mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, + (VariableTable) mcDmuObject); + + // write skims data for estimation data file records + int seq = 1; + for (int i = 0; i < estDataOdts.length; i++) + { + + int[] odtSet = estDataOdts[i]; + int[] mgraSet = mgras[i]; + + odtSet[0] = seq; + + if (outStream1 != null) + { + outStream1.print(seq + "," + odtSet[SAMPNO]); + } + outStream2.print(seq + "," + odtSet[SAMPNO]); + + int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; + + int[] departAvailable = {-1, 1, 1, 1, 1, 1}; + int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; + calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], + mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); + + // write chosen dest and logsums to both files + if (outStream1 != null) + { + outStream1.print("," + odtSet[DEST_MGRA]); + for (double logsum : departArriveLogsums[0]) + outStream1.printf(",%.8f", logsum); + } + + outStream2.print("," + odtSet[DEST_MGRA]); + outStream2.printf(",%.8f", departArriveLogsums[0][chosenLogsumTodIndex]); + + // write logsum sets for each dest in the sample to file 1 + for (int m = 1; m < departArriveLogsums.length; m++) + { + if (outStream1 != null) + { + outStream1.print("," + mgraSet[m - 1]); + for (double logsum : departArriveLogsums[m]) + outStream1.printf(",%.8f", logsum); + } + + outStream2.print("," + mgraSet[m - 1]); + outStream2.printf(",%.8f", departArriveLogsums[m][chosenLogsumTodIndex]); + } + if (outStream1 != null) outStream1.print("\n"); + outStream2.print("\n"); + + if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); + + seq++; + } + + if (outStream1 != null) outStream1.close(); + outStream2.close(); + + } + + private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure + // period(1-6), and arrival period(1-6). + int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; + mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; + int[][] mgraData = new int[NUM_MGRA_FIELDS][]; + + int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); + int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); + + int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); + int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); + int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); + int[] income = hisTds.getColumnAsInt(INCOME_FIELD); + int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); + int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); + int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); + int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); + int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); + int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); + int[] age = hisTds.getColumnAsInt(AGE_FIELD); + int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); + int[] jointPerson1Participates = hisTds.getColumnAsInt(JOINT_P1_FIELD); + int[] jointPerson2Participates = hisTds.getColumnAsInt(JOINT_P2_FIELD); + int[] jointPerson3Participates = hisTds.getColumnAsInt(JOINT_P3_FIELD); + int[] jointPerson4Participates = hisTds.getColumnAsInt(JOINT_P4_FIELD); + int[] jointPerson5Participates = hisTds.getColumnAsInt(JOINT_P5_FIELD); + + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + for (int i = 0; i < NUM_MGRA_FIELDS; i++) + mgras[r - 1][i] = mgraData[i][r - 1]; + + odts[r - 1][SAMPNO] = hisseq[r - 1]; + + odts[r - 1][DEPART_PERIOD] = departs[r - 1]; + odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; + + odts[r - 1][ORIG_MGRA] = origs[r - 1]; + odts[r - 1][DEST_MGRA] = dests[r - 1]; + odts[r - 1][INCOME] = income[r - 1]; + odts[r - 1][ADULTS] = adults[r - 1]; + odts[r - 1][AUTOS] = autos[r - 1]; + odts[r - 1][HHSIZE] = hhsize[r - 1]; + odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; + odts[r - 1][AGE] = age[r - 1]; + odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; + + // the offest constant is used because at-work subtours in + // estimation file are coded as work purpose index (=1), + // but the model index to use is 5. Nonmandatory and escort files + // have correct purpose codes, so offset is 0. + int purposeIndex = purpose[r - 1] + PURPOSE_INDEX_OFFSET; + + odts[r - 1][ESCORT] = purposeIndex == 4 ? 1 : 0; + + odts[r - 1][PARTYSIZE] = jointPerson1Participates[r - 1] + + jointPerson2Participates[r - 1] + jointPerson3Participates[r - 1] + + jointPerson4Participates[r - 1] + jointPerson5Participates[r - 1]; + + odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purposeIndex > 4 ? jtPurpose[r - 1] + : purposeIndex; + + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb; + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + VisitorTourLocationChoiceAppender appender = new VisitorTourLocationChoiceAppender(rbMap); + + appender.startMatrixServer(rb); + appender.runLogsumAppender(rb); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java new file mode 100644 index 0000000..d97d976 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java @@ -0,0 +1,295 @@ +package org.sandag.abm.accessibilities; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import java.io.File; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; + +/** + * This class is used to return walk-transit-drive skim values for MGRA pairs + * associated with estimation data file records. + * + * @author Jim Hicks + * @version March, 2010 + */ +public class WalkTransitDriveSkimsCalculator + implements Serializable +{ + + private transient Logger logger; + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; + + private static final int ACCESS_TIME_INDEX = 0; + private static final int EGRESS_TIME_INDEX = 1; + private static final int NA = -999; + + private int maxWTDSkimSets = 5; + private int[] NUM_SKIMS; + private double[] defaultSkims; + + // declare UEC object + private UtilityExpressionCalculator walkDriveSkimUEC; + private IndexValues iv; + + // The simple auto skims UEC does not use any DMU variables + private TransitDriveAccessDMU dmu = new TransitDriveAccessDMU(); // DMU + // for + // this + // UEC + + private BestTransitPathCalculator bestPathUEC; + + private MgraDataManager mgraManager; + private int maxTap; + + private String[] skimNames; + + // skim values for transit service type(local, premium), + // transit ride mode(lbs, ebs, brt, lrt, crl), + // depart skim period(am, pm, op), and Tap-Tap pair. + private double[][][][][] storedDepartPeriodTapTapSkims; + + private MatrixDataServerIf ms; + + public WalkTransitDriveSkimsCalculator(HashMap rbMap) + { + mgraManager = MgraDataManager.getInstance(); + maxTap = mgraManager.getMaxTap(); + } + + public void setup(HashMap rbMap, Logger aLogger, + BestTransitPathCalculator myBestPathUEC) + { + + logger = aLogger; + + // set the best transit path utility UECs + bestPathUEC = myBestPathUEC; + + // Create the skim UECs + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.drive.data.page"); + int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.drive.skim.page"); + int wtdNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.walk.transit.drive.skims"); + String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.walk.transit.drive.uec.file")).toString(); + File uecFile = new File(uecFileName); + walkDriveSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); + + skimNames = walkDriveSkimUEC.getAlternativeNames(); + + //setup index values + iv = new IndexValues(); + + //setup default skim values + defaultSkims = new double[wtdNumSkims]; + for (int j = 0; j < wtdNumSkims; j++) { + defaultSkims[j] = NA; + } + + // point the stored Array of skims: by Prem or Local, DepartPeriod, O tap, D tap, skim values[] to a shared data store + StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxWTDSkimSets, NUM_PERIODS, maxTap ); + storedDepartPeriodTapTapSkims = storedDataObject.getStoredWtdDepartPeriodTapTapSkims(); + } + + + + /** + * Return the array of walk-transit-drive skims for the ride mode, origin TAP, + * destination TAP, and departure time period. + * + * @param set for set source skims + * @param origTap best Origin TAP for the MGRA pair + * @param workTap best Destination TAP for the MGRA pair + * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 = + * OffPeak period + * @return Array of 55 skim values for the MGRA pair and departure period + */ + public double[] getWalkTransitDriveSkims(int set, double pWalkTime, double aDriveTime, int origTap, int destTap, int departPeriod, boolean debug) + { + + dmu.setMgraTapWalkTime(pWalkTime); + dmu.setDriveTimeFromTap(aDriveTime); + + iv.setOriginZone(origTap); + iv.setDestZone(destTap); + + // allocate space for the origin tap if it hasn't been allocated already + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) + { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; + } + + // if the destTap skims are not already stored, calculate them and store + // them + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) + { + dmu.setTOD(departPeriod); + dmu.setSet(set); + double[] results = walkDriveSkimUEC.solve(iv, dmu, null); + if (debug) + walkDriveSkimUEC.logAnswersArray(logger, "Walk-Drive Skims"); + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pWalkTime; + } + catch ( Exception e ) { + logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pWalkTime=" + pWalkTime); + logger.error ("exception setting walk-transit-drive walk access time in stored array.", e); + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aDriveTime; + } + catch ( Exception e ) { + logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aDriveTime=" + aDriveTime); + logger.error ("exception setting walk-transit-drive drive egress time in stored array.", e); + } + return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; + + + } + + public double[] getNullTransitSkims() + { + return defaultSkims; + } + + /** + * Start the matrix server + * + * @param rb is a ResourceBundle for the properties file for this application + */ + private void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error(String + .format("exception caught running ctramp model components -- exiting."), e); + throw new RuntimeException(); + + } + + } + + /** + * log a report of the final skim values for the MGRA odt + * + * @param odt is an int[] with the first element the origin mgra and the second + * element the dest mgra and third element the departure period index + * @param bestTapPairs is an int[][] of TAP values with the first dimesion the + * ride mode and second dimension a 2 element array with best orig and + * dest TAP + * @param returnedSkims is a double[][] of skim values with the first dimesion + * the ride mode indices and second dimention the skim categories + */ + public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) + { + + Modes.TransitMode[] mode = Modes.TransitMode.values(); + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String separator = ""; + String header = ""; + + logger.info(""); + logger.info(""); + header = "Returned walk-transit-drive skim value tables for origMgra=" + odt[0] + + ", destMgra=" + odt[1] + ", period index=" + odt[2] + ", period label=" + + PERIODS[odt[2]]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + logger.info(separator); + logger.info(header); + logger.info(""); + + String modeHeading = String.format("%-12s %3s ", "RideMode:", mode[0]); + for (int i = 1; i < bestTapPairs.length; i++) + modeHeading += String.format(" %3s ", mode[i]); + logger.info(modeHeading); + + String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", + bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][0]) : "NA", + bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][1]) : "NA"); + for (int i = 1; i < bestTapPairs.length; i++) + tapHeading += String.format(" %4s-%4s ", bestTapPairs[i] != null ? String + .valueOf(bestTapPairs[i][0]) : "NA", bestTapPairs[i] != null ? String + .valueOf(bestTapPairs[i][1]) : "NA"); + logger.info(tapHeading); + + String underLine = String.format("%-12s %9s ", "---------", "---------"); + for (int i = 1; i < bestTapPairs.length; i++) + underLine += String.format(" %9s ", "---------"); + logger.info(underLine); + + for (int j = 0; j < ncols; j++) + { + String tableRecord = ""; + if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, + skims[0][j]); + else tableRecord = String.format("%-12d %12s ", j + 1, ""); + for (int i = 1; i < bestTapPairs.length; i++) + { + if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); + else tableRecord += String.format(" %12s ", ""); + } + logger.info(tableRecord); + } + + logger.info(""); + logger.info(separator); + } + + public String[] getSkimNames() { + return skimNames; + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java new file mode 100644 index 0000000..3f90731 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java @@ -0,0 +1,299 @@ +package org.sandag.abm.accessibilities; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import java.io.File; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; + +/** + * This class is used to return walk-transit-walk skim values for MGRA pairs + * associated with estimation data file records. + * + * @author Jim Hicks + * @version March, 2010 + */ +public class WalkTransitWalkSkimsCalculator + implements Serializable +{ + + private transient Logger logger; + + private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; + + private static final int ACCESS_TIME_INDEX = 0; + private static final int EGRESS_TIME_INDEX = 1; + private static final int NA = -999; + + private int maxWTWSkimSets = 5; + private int[] NUM_SKIMS; + private double[] defaultSkims; + + // declare UEC object + private UtilityExpressionCalculator walkWalkSkimUEC; + private IndexValues iv; + + private String[] skimNames; + + // The simple auto skims UEC does not use any DMU variables + private TransitWalkAccessDMU dmu = new TransitWalkAccessDMU(); + // DMU + // for + // this + // UEC + + private MgraDataManager mgraManager; + private int maxTap; + + // skim values for transit skim set + // depart skim period(am, pm, op) + // and Tap-Tap pair. + private double[][][][][] storedDepartPeriodTapTapSkims; + + private BestTransitPathCalculator bestPathUEC; + + private MatrixDataServerIf ms; + + public WalkTransitWalkSkimsCalculator(HashMap rbMap) + { + mgraManager = MgraDataManager.getInstance(); + maxTap = mgraManager.getMaxTap(); + } + + public void setup(HashMap rbMap, Logger aLogger, BestTransitPathCalculator myBestPathUEC) + { + + logger = aLogger; + + // Create the utility UECs + bestPathUEC = myBestPathUEC; + + // Create the skim UECs + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.walk.data.page"); + int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.walk.skim.page"); + int wtwNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.walk.transit.walk.skims"); + String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.walk.transit.walk.uec.file")).toString(); + File uecFile = new File(uecFileName); + walkWalkSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); + + //setup index values + iv = new IndexValues(); + + //setup default skim values + defaultSkims = new double[wtwNumSkims]; + for (int j = 0; j < wtwNumSkims; j++) { + defaultSkims[j] = NA; + } + + skimNames = walkWalkSkimUEC.getAlternativeNames(); + + // point the stored Array of skims: skim set, period, O tap, D tap, skim values[] to a shared data store + StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxWTWSkimSets, NUM_PERIODS, maxTap ); + storedDepartPeriodTapTapSkims = storedDataObject.getStoredWtwDepartPeriodTapTapSkims(); + + } + + + + /** + * Return the array of walk-transit skims for the ride mode, origin TAP, + * destination TAP, and departure time period. + * + * @param set for set source skims + * @param origTap best Origin TAP for the MGRA pair + * @param destTap best Destination TAP for the MGRA pair + * @param departPeriod skim period index for the departure period - 0 = AM + * period, 1 = PM period, 2 = OffPeak period + * @return Array of skim values for the MGRA pair and departure period for the + * skim set + */ + public double[] getWalkTransitWalkSkims(int set, double pWalkTime, double aWalkTime, int origTap, int destTap, + int departPeriod, boolean debug) + { + + dmu.setMgraTapWalkTime(pWalkTime); + dmu.setTapMgraWalkTime(aWalkTime); + + iv.setOriginZone(origTap); + iv.setDestZone(destTap); + + // allocate space for the origin tap if it hasn't been allocated already + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) + { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; + } + + // if the destTap skims are not already stored, calculate them and store + // them + if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) + { + dmu.setTOD(departPeriod); + dmu.setSet(set); + double[] results = walkWalkSkimUEC.solve(iv, dmu, null); + if (debug) + walkWalkSkimUEC.logAnswersArray(logger, "Walk-Walk Tap-Tap Skims"); + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pWalkTime; + } + catch ( Exception e ) { + logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pWalkTime=" + pWalkTime); + logger.error ("exception setting walk-transit-walk walk access time in stored array.", e); + } + + try { + storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aWalkTime; + } + catch ( Exception e ) { + logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aWalkTime=" + aWalkTime); + logger.error ("exception setting walk-transit-walk walk egress time in stored array.", e); + } + return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; + + + } + + public double[] getNullTransitSkims() + { + return defaultSkims; + } + + /** + * Start the matrix server + * + * @param rb is a ResourceBundle for the properties file for this application + */ + private void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error(String + .format("exception caught running ctramp model components -- exiting."), e); + throw new RuntimeException(); + + } + + } + + /** + * log a report of the final skim values for the MGRA odt + * + * @param odt is an int[] with the first element the origin mgra and the second + * element the dest mgra and third element the departure period index + * @param bestTapPairs is an int[][] of TAP values with the first dimesion the + * ride mode and second dimension a 2 element array with best orig and + * dest TAP + * @param returnedSkims is a double[][] of skim values with the first dimesion + * the ride mode indices and second dimention the skim categories + */ + public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) + { + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String separator = ""; + String header = ""; + + logger.info(""); + logger.info(""); + header = "Returned walktransit skim value tables for origMgra=" + odt[0] + ", destMgra=" + + odt[1] + ", period index=" + odt[2] + ", period label=" + PERIODS[odt[2]]; + for (int i = 0; i < header.length(); i++) + separator += "^"; + + logger.info(separator); + logger.info(header); + logger.info(""); + + String modeHeading = String.format("%-12s %3s ", "Alt:"); + for (int i = 1; i < bestTapPairs.length; i++) + modeHeading += String.format(" %3s ", i); + logger.info(modeHeading); + + String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", + bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][0]) : "NA", + bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][1]) : "NA"); + for (int i = 1; i < bestTapPairs.length; i++) + tapHeading += String.format(" %4s-%4s ", bestTapPairs[i] != null ? String + .valueOf(bestTapPairs[i][0]) : "NA", bestTapPairs[i] != null ? String + .valueOf(bestTapPairs[i][1]) : "NA"); + logger.info(tapHeading); + + String underLine = String.format("%-12s %9s ", "---------", "---------"); + for (int i = 1; i < bestTapPairs.length; i++) + underLine += String.format(" %9s ", "---------"); + logger.info(underLine); + + for (int j = 0; j < ncols; j++) + { + String tableRecord = ""; + if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, + skims[0][j]); + else tableRecord = String.format("%-12d %12s ", j + 1, ""); + for (int i = 1; i < bestTapPairs.length; i++) + { + if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); + else tableRecord += String.format(" %12s ", ""); + } + logger.info(tableRecord); + } + + logger.info(""); + logger.info(separator); + } + + public String[] getSkimNames() { + return skimNames; + } + + public BestTransitPathCalculator getBestPathUEC() { + return bestPathUEC; + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java new file mode 100644 index 0000000..1b29ea5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java @@ -0,0 +1,518 @@ +package org.sandag.abm.accessibilities; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +public final class XBorderSkimsAppender +{ + + protected transient Logger logger = Logger.getLogger(XBorderSkimsAppender.class); + + /* + * for trip mode choice estimation files + */ + private static final String MGRA_O_D_RECORDS_FILE_KEY = "xborder.mgra.list.file"; + private static final String APPENDED_SKIMS_FILE_KEY = "xborder.appended.file"; + + private String[] inputFormats = {"NUMBER", "NUMBER", "NUMBER", + "STRING", "NUMBER", "NUMBER", "STRING" }; + private static final int INPUT_ORIG_MGRA = 5; + private static final int INPUT_DEST_MGRA = 6; + private static final int INPUT_DEPART_PERIOD = 7; + + // survey periods are: + // 0=not used, + // 1=03:00-05:59, + // 2=06:00-08:59, + // 3=09:00-11:59, + // 4=12:00-15:29, + // 5=15:30-18:59, + // 6=19:00-02:59 + // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), + // 6=3(OP) + + // define a conversion array to convert period values in the survey file to + // skim + // period indices used in this propgram: 1=am peak, 2=pm peak, 3=off-peak. + private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; + private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; + + private static int debugOrigMgra = 0; + private static int debugDestMgra = 0; + private static int departModelPeriod = 0; + + private MatrixDataServerIf ms; + private BestTransitPathCalculator bestPathUEC; + private static final float defaultVOT = 15.0f; + + private XBorderSkimsAppender() + { + } + + private void runSkimsAppender(ResourceBundle rb) + { + + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + Logger autoLogger = Logger.getLogger("auto"); + Logger wtwLogger = Logger.getLogger("wtw"); + + String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, + APPENDED_SKIMS_FILE_KEY); + + FileWriter writer; + PrintWriter outStreamHis = null; + + AutoTazSkimsCalculator tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + McLogsumsAppender logsumHelper = new McLogsumsAppender(rbMap); + bestPathUEC = logsumHelper.getBestTransitPathCalculator(); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); + WalkTransitWalkSkimsCalculator wtw = new WalkTransitWalkSkimsCalculator(rbMap); + + String heading = "seq"; + + heading += ",origMgra,destMgra,departPeriod"; + heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); + heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); + heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); + + try + { + writer = new FileWriter(new File(outputFileNameHis)); + outStreamHis = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening output skims file: %s.", + outputFileNameHis)); + throw new RuntimeException(e); + } + outStreamHis.println(heading); + + Logger[] loggers = new Logger[4]; + loggers[0] = autoLogger; + loggers[1] = wtwLogger; + + int[] odt = new int[3]; + + TableDataSet hisTds = getInputTableDataSet(rbMap); + int[][] hisOdts = getInputOrigDestTimes(hisTds); + // 11100, pnrCoaster, mgra 4357 = taz 3641, mgra 26931 = taz 1140 + // int[][] hisOdts = { { 4357, 26931, 5, 0 } }; + // 25040, wlkCoaster, mgra 4989 = taz 3270, mgra 7796 = taz 1986 + // int[][] hisOdts = { { 7796, 4989, 2, 0 } }; + + // if ( debugOrigMgra <= 0 || debugDestMgra <= 0 || departModelPeriod <= + // 0 || departModelPeriod > 6 ) + // { + // logger.error("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); + // System.exit(-1); + // } + // int[][] hisOdts = { { debugOrigMgra, debugDestMgra, + // departModelPeriod, 0 } }; + + // write skims data for home interview survey records + int seq = 1; + for (int[] hisOdt : hisOdts) + { + // write outbound direction + odt[0] = hisOdt[0]; // orig + odt[1] = hisOdt[1]; // dest + odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim + // period + + try + { + + writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, loggers); + } catch (Exception e) + { + logger.error("Exception caught processing record: " + seq + " of " + hisOdts.length + + "."); + break; + } + + if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); + + seq++; + } + + outStreamHis.close(); + + } + + private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, + int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, + Logger[] loggers) + { + + Logger autoLogger = loggers[0]; + Logger wtwLogger = loggers[1]; + + int[][] bestTapPairs = null; + double[][] returnedSkims = null; + + outStream.print(String.format("%d,%d,%d,%s", sequence, odt[0], odt[1], + SKIM_PERIOD_LABELS[odt[2] - 1])); + + double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], defaultVOT, loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); + + String autoRecord = getAutoSkimsRecord(skims); + outStream.print(autoRecord); + + skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); + if (loggingEnabled) + anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); + + String nmRecord = getAutoSkimsRecord(skims); + outStream.print(nmRecord); + + /* + * TODO: Fix this code + + bestTapPairs = wtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtwLogger); + returnedSkims = new double[bestTapPairs.length][]; + for (int i = 0; i < bestTapPairs.length; i++) + { + if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); + else + { + returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, BestTransitPathCalculator + .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), + BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], + bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], + odt[2], loggingEnabled); + } + } + if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); + + String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); + outStream.println(wtwRecord); + */ + } + + /** + * Start the matrix server + * + * @param rb + * is a ResourceBundle for the properties file for this + * application + */ + private void startMatrixServer(ResourceBundle rb) + { + + logger.info(""); + logger.info(""); + String serverAddress = rb.getString("RunModel.MatrixServerAddress"); + int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try + { + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[][] of skim values with the first dimesion the + * ride mode indices and second dimention the skim categories + */ + private String getTransitSkimsRecord(int[] odt, double[][] skims) + { + + int nrows = skims.length; + int ncols = 0; + for (int i = 0; i < nrows; i++) + if (skims[i].length > ncols) ncols = skims[i].length; + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + for (int j = 0; j < skims[i].length; j++) + tableRecord += String.format(",%.5f", skims[i][j]); + } + + return tableRecord; + + } + + /** + * create a String which can be written to an output file with all the skim + * values for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + * @param skims + * is a double[] of skim values + */ + private String getAutoSkimsRecord(double[] skims) + { + + String tableRecord = ""; + for (int i = 0; i < skims.length; i++) + { + tableRecord += String.format(",%.5f", skims[i]); + } + + return tableRecord; + + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getTransitSkimsHeaderRecord(String transitServiveLabel, String[] skimNames) + { + + Modes.TransitMode[] mode = Modes.TransitMode.values(); + + String heading = ""; + + for (int i = 0; i < mode.length; i++) + { + for (int j = 0; j < skimNames.length; j++) + heading += String.format(",%s_%s_%s", transitServiveLabel, mode[i], + skimNames[j]); + + } + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getAutoSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + /** + * create a String for the output file header record which can be written to + * an output file with all the skim value namess for the orig/dest/period. + * + * @param odt + * is an int[] with the first element the origin mgra and the + * second element the dest mgra and third element the departure + * period index + */ + private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) + { + + String heading = ""; + + for (int i = 0; i < names.length; i++) + heading += String.format(",%s_%s", label, names[i]); + + return heading; + } + + private TableDataSet getInputTableDataSet(HashMap rbMap) + { + + String hisFileName = Util.getStringValueFromPropertyMap(rbMap, MGRA_O_D_RECORDS_FILE_KEY); + if (hisFileName == null) + { + logger.error("Error getting the filename from the properties file for the XBorder MGRA List data records file."); + logger.error("Properties file target: " + MGRA_O_D_RECORDS_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + MGRA_O_D_RECORDS_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + try + { + TableDataSet inTds = null; + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTds = reader.readFileWithFormats(new File(hisFileName), inputFormats); + // inTds = reader.readFile(new File(hisFileName)); + return inTds; + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading Sandag XBorder MGRA List data records file: %s into TableDataSet object.", + hisFileName)); + throw new RuntimeException(e); + } + + } + + private int[][] getInputOrigDestTimes(TableDataSet hisTds) + { + + // odts are an array with elements: origin mgra, destination mgra, + // departure period(1-6), and his sampno. + int[][] odts = new int[hisTds.getRowCount()][4]; + + int[] origs = hisTds.getColumnAsInt(INPUT_ORIG_MGRA); + int[] dests = hisTds.getColumnAsInt(INPUT_DEST_MGRA); + String[] departStrings = hisTds.getColumnAsString(INPUT_DEPART_PERIOD); + int[] departs = new int[departStrings.length]; + + for (int r = 1; r <= hisTds.getRowCount(); r++) + { + if (departStrings[r - 1].equalsIgnoreCase("am")) departs[r - 1] = 2; + else if (departStrings[r - 1].equalsIgnoreCase("pm")) departs[r - 1] = 5; + else if (departStrings[r - 1].equalsIgnoreCase("op")) departs[r - 1] = 1; + else departs[r - 1] = -1; + + odts[r - 1][0] = origs[r - 1]; + odts[r - 1][1] = dests[r - 1]; + odts[r - 1][2] = departs[r - 1]; + } + + return odts; + } + + public static void main(String[] args) + { + + ResourceBundle rb = null; + if (args.length == 0) + { + System.out + .println(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else if (args.length == 4) + { + rb = ResourceBundle.getBundle(args[0]); + + debugOrigMgra = Integer.parseInt(args[1]); + debugDestMgra = Integer.parseInt(args[2]); + departModelPeriod = Integer.parseInt(args[3]); + } else + { + System.out + .println("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); + System.exit(-1); + } + + try + { + + MatrixDataServerIf ms = null; + String serverAddress = null; + int serverPort = -1; + + HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + System.out.println(""); + System.out.println(""); + serverAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + + String serverPortString = (String) propertyMap.get("RunModel.MatrixServerPort"); + if (serverPortString != null) serverPort = Integer.parseInt(serverPortString); + + if (serverAddress != null && serverPort > 0) + { + try + { + System.out.println("attempting connection to matrix server " + serverAddress + + ":" + serverPort); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + System.out.println("connected to matrix server " + serverAddress + ":" + + serverPort); + + } catch (Exception e) + { + System.out + .println("exception caught running ctramp model components -- exiting."); + e.printStackTrace(); + throw new RuntimeException(); + } + } + + TazDataManager tazs = TazDataManager.getInstance(propertyMap); + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TapDataManager tapManager = TapDataManager.getInstance(propertyMap); + + // create an appender object and run it + XBorderSkimsAppender appender = new XBorderSkimsAppender(); + appender.runSkimsAppender(rb); + + } catch (RuntimeException e) + { + + e.printStackTrace(); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java new file mode 100644 index 0000000..b3fd57d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java @@ -0,0 +1,51 @@ +package org.sandag.abm.active; + +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.LinkedHashSet; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; +import java.util.Set; + +public abstract class AbstractNetworkFactory, T extends Traversal> + extends NetworkFactory +{ + + @Override + public Network createNetwork() + { + Network network = new SimpleNetwork<>(getNodes(), getEdges(), getTraversals()); + calculateDerivedNodeAttributes(network); + calculateDerivedEdgeAttributes(network); + calculateDerivedTraversalAttributes(network); + return network; + } + + @Override + protected Collection getTraversals() + { + Collection edges = getEdges(); + Map> predecessors = new HashMap<>(); + for (N node : getNodes()) + predecessors.put(node, new LinkedList()); + for (E edge : edges) + predecessors.get(edge.getToNode()).add(edge); + Set traversals = new LinkedHashSet<>(); + for (E toEdge : getEdges()) + { + for (E fromEdge : predecessors.get(toEdge.getFromNode())) + if (!isReversal(fromEdge, toEdge)) traversals.add(getTraversal(fromEdge, toEdge)); + } + return Collections.unmodifiableCollection(traversals); + } + + private boolean isReversal(E edge1, E edge2) + { + return (edge1.getToNode().equals(edge2.getFromNode())) + && (edge1.getFromNode().equals(edge2.getToNode())); + } + + abstract protected T getTraversal(E fromEdge, E toEdge); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java new file mode 100644 index 0000000..c94f312 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java @@ -0,0 +1,217 @@ +package org.sandag.abm.active; + +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.Queue; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentLinkedQueue; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.atomic.AtomicInteger; +import org.apache.log4j.Logger; + +public abstract class AbstractPathChoiceEdgeAssignmentApplication, T extends Traversal> +{ + private static final Logger logger = Logger.getLogger(AbstractPathChoiceEdgeAssignmentApplication.class); + + protected PathAlternativeListGenerationConfiguration configuration; + private boolean randomCostSeeded; + private double maxCost; + private TraversalEvaluator traversalCostEvaluator; + private EdgeEvaluator edgeLengthEvaluator; + long startTime; + protected Network network; + String outputDir; + + double[] sampleDistanceBreaks; + double[] samplePathSizes; + double[] sampleMinCounts; + double[] sampleMaxCounts; + + private static final int TRIP_PROGRESS_REPORT_COUNT = 1000; + + public AbstractPathChoiceEdgeAssignmentApplication( + PathAlternativeListGenerationConfiguration configuration) + { + this.configuration = configuration; + configuration.getOriginZonalCentroidIdMap(); + this.randomCostSeeded = configuration.isRandomCostSeeded(); + this.maxCost = configuration.getMaxCost(); + this.traversalCostEvaluator = configuration.getTraversalCostEvaluator(); + this.edgeLengthEvaluator = configuration.getEdgeLengthEvaluator(); + this.sampleDistanceBreaks = configuration.getSampleDistanceBreaks(); + this.samplePathSizes = configuration.getSamplePathSizes(); + this.sampleMinCounts = configuration.getSampleMinCounts(); + this.sampleMaxCounts = configuration.getSampleMaxCounts(); + this.network = configuration.getNetwork(); + this.outputDir = configuration.getOutputDirectory(); + } + + protected abstract Map assignTrip(int tripNum, + PathAlternativeList alternativeList); + + public Map assignTrips(List tripNums) + { + logger.info("Assigning trips..."); + logger.info("Writing to " + outputDir); + ConcurrentHashMap volumes = new ConcurrentHashMap<>(); + int threadCount = Runtime.getRuntime().availableProcessors() -1; + ExecutorService executor = Executors.newFixedThreadPool(threadCount); + final Queue tripQueue = new ConcurrentLinkedQueue<>(tripNums); + final CountDownLatch latch = new CountDownLatch(threadCount); + final AtomicInteger counter = new AtomicInteger(); + startTime = System.currentTimeMillis(); + for (int i = 0; i < threadCount; i++) + executor.execute(new CalculationTask(tripQueue, counter, latch, volumes)); + try + { + latch.await(); + } catch (InterruptedException e) + { + throw new RuntimeException(e); + } + executor.shutdown(); + + return volumes; + } + + private class CalculationTask + implements Runnable + { + private final Queue tripQueue; + private final AtomicInteger counter; + private final CountDownLatch latch; + private final ConcurrentHashMap volumes; + + private CalculationTask(Queue tripQueue, AtomicInteger counter, + CountDownLatch latch, ConcurrentHashMap volumes) + { + this.tripQueue = tripQueue; + this.counter = counter; + this.latch = latch; + this.volumes = volumes; + } + + private PathAlternativeList generateAlternatives(int tripId) + { + Set singleOriginNode = new HashSet<>(); + Set singleDestinationNode = new HashSet<>(); + + EdgeEvaluator randomizedEdgeCost; + ShortestPathStrategy shortestPathStrategy; + ShortestPathResultSet result; + + singleOriginNode.add(getOriginNode(tripId)); + singleDestinationNode.add(getDestinationNode(tripId)); + + NodePair odPair = new NodePair<>(getOriginNode(tripId), getDestinationNode(tripId)); + PathAlternativeList alternativeList = new PathAlternativeList<>(odPair, network, + edgeLengthEvaluator); + + TraversalEvaluator zeroTraversalEvaluator = new ZeroTraversalEvaluator(); + + shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, + edgeLengthEvaluator, zeroTraversalEvaluator); + result = shortestPathStrategy.getShortestPaths(singleOriginNode, singleDestinationNode, + Double.MAX_VALUE); + if (result.getShortestPathResult(odPair) == null) + { + logger.error("no path found for trip with origin " + getOriginNode(tripId) + + " and destination " + getDestinationNode(tripId)); + return alternativeList; + } + double distance = result.getShortestPathResult(odPair).getCost(); + int distanceIndex = findFirstIndexGreaterThan(distance, sampleDistanceBreaks); + + for (int iterCount = 1; iterCount <= sampleMinCounts[distanceIndex]; iterCount++) + { + if (randomCostSeeded) + { + randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, + Objects.hash(tripId, iterCount)); + } else + { + randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, 0); + } + + shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, + randomizedEdgeCost, traversalCostEvaluator); + result = shortestPathStrategy.getShortestPaths(singleOriginNode, + singleDestinationNode, Double.MAX_VALUE); + + alternativeList.add(result.getShortestPathResult(odPair).getPath()); + } + return alternativeList; + } + + public void run() + { + while (tripQueue.size() > 0) + { + int tripId = tripQueue.poll(); + PathAlternativeList alternativeList = generateAlternatives(tripId); + + if (alternativeList.getCount() > 0) + { + Map tripVolumes = assignTrip(tripId, alternativeList); + + for (E edge : tripVolumes.keySet()) + { + if (volumes.containsKey(edge)) + { + double[] values = volumes.get(edge); + for (int i = 0; i < values.length; i++) + values[i] += tripVolumes.get(edge)[i]; + volumes.put(edge, values); + } else + { + volumes.put(edge, tripVolumes.get(edge)); + } + } + } + + int c = counter.addAndGet(1); + if ((c % TRIP_PROGRESS_REPORT_COUNT) == 0) + { + System.out.println(" done with " + c + " trips, run time: " + + (System.currentTimeMillis() - startTime) / 1000 + " sec."); + } + } + + latch.countDown(); + } + } + + protected abstract N getOriginNode(int tripId); + + protected abstract N getDestinationNode(int tripId); + + private class ZeroTraversalEvaluator + implements TraversalEvaluator + { + private ZeroTraversalEvaluator() + { + } + + public double evaluate(T traversal) + { + return 0.0; + } + } + + protected int findFirstIndexGreaterThan(double value, double[] array) + { + for (int i = 0; i < array.length; i++) + { + if (array[i] >= value) + { + return i; + } + } + return array.length; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java new file mode 100644 index 0000000..04c4abc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java @@ -0,0 +1,422 @@ +package org.sandag.abm.active; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.Queue; +import java.util.Random; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentLinkedQueue; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.atomic.AtomicInteger; +import org.apache.log4j.Logger; + +public abstract class AbstractPathChoiceLogsumMatrixApplication, T extends Traversal> +{ + private static final Logger logger = Logger.getLogger(AbstractPathChoiceLogsumMatrixApplication.class); + + protected PathAlternativeListGenerationConfiguration configuration; + Network network; + Map> nearbyZonalDistanceMap; + Map originZonalCentroidIdMap; + Map destinationZonalCentroidIdMap; + double[] sampleDistanceBreaks; + double[] samplePathSizes; + double[] sampleMinCounts; + double[] sampleMaxCounts; + EdgeEvaluator edgeLengthEvaluator; + EdgeEvaluator edgeCostEvaluator; + TraversalEvaluator traversalCostEvaluator; + double maxCost; + long startTime; + String outputDir; + Set traceOrigins; + protected Map propertyMap; + boolean randomCostSeeded; + boolean intrazonalsNeeded; + + private static final int ORIGIN_PROGRESS_REPORT_COUNT = 50; + private static final double DOUBLE_PRECISION_TOLERANCE = 0.001; + + protected abstract double[] calculateMarketSegmentLogsums( + PathAlternativeList alternativeList); + + protected abstract List> getMarketSegmentIntrazonalCalculations(); + + public AbstractPathChoiceLogsumMatrixApplication( + PathAlternativeListGenerationConfiguration configuration) + { + this.configuration = configuration; + this.network = configuration.getNetwork(); + this.nearbyZonalDistanceMap = Collections.unmodifiableMap(configuration + .getNearbyZonalDistanceMap()); + this.originZonalCentroidIdMap = Collections.unmodifiableMap(configuration + .getOriginZonalCentroidIdMap()); + this.destinationZonalCentroidIdMap = Collections.unmodifiableMap(configuration + .getDestinationZonalCentroidIdMap()); + this.sampleDistanceBreaks = configuration.getSampleDistanceBreaks(); + this.samplePathSizes = configuration.getSamplePathSizes(); + this.sampleMinCounts = configuration.getSampleMinCounts(); + this.sampleMaxCounts = configuration.getSampleMaxCounts(); + this.edgeLengthEvaluator = configuration.getEdgeLengthEvaluator(); + this.edgeCostEvaluator = configuration.getEdgeCostEvaluator(); + this.traversalCostEvaluator = configuration.getTraversalCostEvaluator(); + this.maxCost = configuration.getMaxCost(); + this.outputDir = configuration.getOutputDirectory(); + this.traceOrigins = configuration.getTraceOrigins(); + this.propertyMap = configuration.getPropertyMap(); + this.randomCostSeeded = configuration.isRandomCostSeeded(); + this.intrazonalsNeeded = configuration.isIntrazonalsNeeded(); + } + + public Map, double[]> calculateMarketSegmentLogsums() + { + logger.info("Generating path alternative lists..."); + logger.info("Writing to " + outputDir); + Map> logsums = new ConcurrentHashMap<>(); + startTime = System.currentTimeMillis(); + int threadCount = Runtime.getRuntime().availableProcessors() -1; + ExecutorService executor = Executors.newFixedThreadPool(threadCount); + final Queue originQueue = new ConcurrentLinkedQueue<>( + originZonalCentroidIdMap.keySet()); + + final ConcurrentHashMap> insufficientSamplePairs = new ConcurrentHashMap<>(); + final CountDownLatch latch = new CountDownLatch(threadCount); + final AtomicInteger counter = new AtomicInteger(); + for (int i = 0; i < threadCount; i++) + executor.execute(new CalculationTask(originQueue, counter, latch, logsums, + insufficientSamplePairs)); + try + { + latch.await(); + } catch (InterruptedException e) + { + throw new RuntimeException(e); + } + executor.shutdown(); + + /* + * for (int origin : insufficientSamplePairs.keySet() ) { String message + * = "Sample insufficient for origin zone " + origin + + * " and destination zones "; for (int destination : + * insufficientSamplePairs.get(origin) ) { message = message + + * destination + " "; } System.out.println(message); } + */ + + int totalPairs = 0; + for (int o : nearbyZonalDistanceMap.keySet()) + { + totalPairs += nearbyZonalDistanceMap.get(o).size(); + } + logger.info("Total OD pairs: " + totalPairs); + + int totalInsuffPairs = 0; + for (int o : insufficientSamplePairs.keySet()) + { + totalInsuffPairs += insufficientSamplePairs.get(o).size(); + } + + logger.info("Total insufficient sample pairs: " + totalInsuffPairs); + + if (intrazonalsNeeded) + { + logger.info("Calculating intrazonals"); + List> intrazonalCalculations = getMarketSegmentIntrazonalCalculations(); + int segments = intrazonalCalculations.size(); + for (int segment = 0; segment < segments; segment++) + { + for (N origin : logsums.keySet()) + { + Map originLogsums = logsums.get(origin); + if (segment == 0) originLogsums.put(origin, new double[segments]); + originLogsums.get(origin)[segment] = intrazonalCalculations.get(segment) + .getIntrazonalValue(origin, originLogsums, segment); + } + } + } + + Map, double[]> pairLogsums = new HashMap<>(); + for (N oNode : logsums.keySet()) + { + for (N dNode : logsums.get(oNode).keySet()) + { + pairLogsums.put(new NodePair(oNode, dNode), logsums.get(oNode).get(dNode)); + } + } + + return pairLogsums; + } + + private int findFirstIndexGreaterThan(double value, double[] array) + { + for (int i = 0; i < array.length; i++) + { + if (array[i] >= value) + { + return i; + } + } + return array.length; + } + + private class CalculationTask + implements Runnable + { + private final Queue originQueue; + private final AtomicInteger counter; + private final CountDownLatch latch; + + private final ConcurrentHashMap> insufficientSamplePairs; + private final Map> logsums; + + private CalculationTask(Queue originQueue, AtomicInteger counter, + CountDownLatch latch, Map> logsums, + ConcurrentHashMap> insufficientSamplePairs) + { + this.originQueue = originQueue; + this.counter = counter; + this.latch = latch; + this.insufficientSamplePairs = insufficientSamplePairs; + this.logsums = logsums; + } + + private Map, PathAlternativeList> generateAlternatives(int origin) + { + Set singleOriginNode = new HashSet<>(); + Set destinationNodes = new HashSet<>(); + Map destinationZoneMap = new HashMap<>(); + Map destinationDistanceMap = new HashMap<>(); + Map destinationPathSizeMap = new HashMap<>(); + Map destinationMinCountMap = new HashMap<>(); + Map destinationMaxCountMap = new HashMap<>(); + HashMap, PathAlternativeList> alternativeLists = new HashMap<>(); + EdgeEvaluator randomizedEdgeCost; + ShortestPathStrategy shortestPathStrategy; + ShortestPathResultSet result; + int distanceIndex; + + singleOriginNode.add(network.getNode(originZonalCentroidIdMap.get(origin))); + N destinationNode = null; + PathAlternativeList alternativeList; + + if (nearbyZonalDistanceMap.containsKey(origin)) + { + for (int destination : nearbyZonalDistanceMap.get(origin).keySet()) + { + try + { + destinationNode = network.getNode(destinationZonalCentroidIdMap + .get(destination)); + } catch (NullPointerException e) + { + logger.warn(destinationZonalCentroidIdMap.get(destination)); + } + destinationNodes.add(destinationNode); + destinationDistanceMap.put(destinationNode, nearbyZonalDistanceMap.get(origin) + .get(destination)); + destinationZoneMap.put(destinationNode, destination); + distanceIndex = findFirstIndexGreaterThan( + destinationDistanceMap.get(destinationNode), sampleDistanceBreaks); + destinationPathSizeMap.put(destinationNode, samplePathSizes[distanceIndex]); + destinationMinCountMap.put(destinationNode, sampleMinCounts[distanceIndex]); + destinationMaxCountMap.put(destinationNode, sampleMaxCounts[distanceIndex]); + } + } + + int iterCount = 1; + while (destinationNodes.size() > 0) + { + + if (randomCostSeeded) + { + randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, + Objects.hash(origin, iterCount)); + } else + { + randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, 0); + } + + shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, + randomizedEdgeCost, traversalCostEvaluator); + result = shortestPathStrategy.getShortestPaths(singleOriginNode, destinationNodes, + maxCost); + + for (NodePair odPair : result) + { + if (!alternativeLists.containsKey(odPair)) + { + alternativeLists.put(odPair, new PathAlternativeList(odPair, network, + edgeLengthEvaluator)); + } + alternativeList = alternativeLists.get(odPair); + alternativeList.add(result.getShortestPathResult(odPair).getPath()); + destinationNode = odPair.getToNode(); + + if (alternativeList.getSizeMeasureTotal() >= destinationPathSizeMap + .get(destinationNode) - DOUBLE_PRECISION_TOLERANCE + && iterCount >= destinationMinCountMap.get(destinationNode)) + { + destinationNodes.remove(odPair.getToNode()); + alternativeList.clearPathSizeCalculator(); + } else if (iterCount >= destinationMaxCountMap.get(destinationNode)) + { + destinationNodes.remove(odPair.getToNode()); + alternativeList.clearPathSizeCalculator(); + if (!insufficientSamplePairs.containsKey(origin)) + insufficientSamplePairs.put(origin, new ArrayList()); + insufficientSamplePairs.get(origin).add( + destinationZoneMap.get(destinationNode)); + } + } + + iterCount++; + } + + if (traceOrigins.contains(origin)) + { + try + { + PathAlternativeListWriter writer = new PathAlternativeListWriter( + outputDir + "origpaths_" + origin + ".csv", outputDir + "origlinks_" + + origin + ".csv"); + writer.writeHeaders(); + for (PathAlternativeList list : alternativeLists.values()) + { + writer.write(list); + } + writer.close(); + } catch (IOException e) + { + throw new RuntimeException(e.getMessage()); + } + + } + + for (NodePair odPair : alternativeLists.keySet()) + alternativeLists.put( + odPair, + resampleAlternatives(alternativeLists.get(odPair), + destinationPathSizeMap.get(odPair.getToNode()))); + + return alternativeLists; + } + + private PathAlternativeList resampleAlternatives(PathAlternativeList alts, + double targetSize) + { + if (targetSize >= alts.getSizeMeasureTotal()) + { + return alts; + } + Random r; + if (randomCostSeeded) + { + r = new Random(alts.getODPair().hashCode()); + } else + { + r = new Random(); + } + PathAlternativeList newAlts = new PathAlternativeList<>(alts.getODPair(), + network, alts.getLengthEvaluator()); + double[] prob = new double[alts.getCount()]; + double[] cum = new double[alts.getCount()]; + double tot = 0.0; + for (int i = 0; i < prob.length; i++) + { + prob[i] = alts.getSizeMeasures().get(i) / alts.getSizeMeasureTotal(); + tot = tot + prob[i]; + cum[i] = tot; + } + cum[alts.getCount() - 1] = 1.0; + + while (newAlts.getSizeMeasureTotal() < targetSize + && newAlts.getCount() < alts.getCount()) + { + double p = r.nextDouble(); + int idx = BinarySearch.binarySearch(cum, p); + newAlts.add(alts.get(idx)); + double curProb = cum[idx]; + if (idx > 0) + { + curProb = curProb - cum[idx - 1]; + } + for (int i = 0; i < cum.length; i++) + { + if (i < idx) + { + cum[i] = cum[i] / (1 - curProb); + } else + { + cum[i] = (cum[i] - curProb) / (1 - curProb); + } + } + } + return newAlts; + } + + @Override + public void run() + { + while (originQueue.size() > 0) + { + int origin = originQueue.poll(); + + Map, PathAlternativeList> alternativeLists = generateAlternatives(origin); + + if (traceOrigins.contains(origin)) + { + try + { + PathAlternativeListWriter writer = new PathAlternativeListWriter( + outputDir + "resamplepaths_" + origin + ".csv", outputDir + + "resamplelinks_" + origin + ".csv"); + writer.writeHeaders(); + for (PathAlternativeList list : alternativeLists.values()) + { + writer.write(list); + } + writer.close(); + } catch (IOException e) + { + throw new RuntimeException(e.getMessage()); + } + } + + double[] logsumValues; + for (NodePair odPair : alternativeLists.keySet()) + { + + if (!odPair.getFromNode().equals(odPair.getToNode())) + { + + logsumValues = calculateMarketSegmentLogsums(alternativeLists.get(odPair)); + if (!logsums.containsKey(odPair.getFromNode())) + { + logsums.put(odPair.getFromNode(), new ConcurrentHashMap()); + } + logsums.get(odPair.getFromNode()).put(odPair.getToNode(), logsumValues); + } + + } + + int c = counter.addAndGet(1); + if ((c % ORIGIN_PROGRESS_REPORT_COUNT) == 0) + { + logger.info(" done with " + c + " origins, run time: " + + (System.currentTimeMillis() - startTime) / 1000 + " sec."); + } + } + + latch.countDown(); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java new file mode 100644 index 0000000..8921833 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java @@ -0,0 +1,20 @@ +package org.sandag.abm.active; + +public abstract class AbstractShortestPathResultSet + implements ModifiableShortestPathResultSet +{ + + @Override + public void addResult(NodePair od, Path path, double cost) + { + addResult(new ShortestPathResult(od, path, cost)); + } + + @Override + public void addAll(ShortestPathResultSet results) + { + for (ShortestPathResult result : results.getResults()) + addResult(result); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java new file mode 100644 index 0000000..417a9e1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java @@ -0,0 +1,59 @@ +package org.sandag.abm.active; + +import java.util.Collection; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; + +public class BasicShortestPathResultSet + extends AbstractShortestPathResultSet +{ + private final Map, ShortestPathResult> results; + + public BasicShortestPathResultSet() + { + results = new HashMap<>(); // iteration order may not matter, but just + // in case, this is cheap + } + + @Override + public void addResult(ShortestPathResult spResult) + { + ShortestPathResult spr = results.put(spResult.getOriginDestination(), spResult); + if (spr != null) + throw new IllegalArgumentException("Repeated shortest path results for node pair: (" + + spResult.getOriginDestination().getFromNode().getId() + "," + + spResult.getOriginDestination().getToNode().getId() + ")"); + } + + @Override + public void addResult(NodePair od, Path path, double cost) + { + addResult(new ShortestPathResult(od, path, cost)); + } + + @Override + public Iterator> iterator() + { + return results.keySet().iterator(); + } + + @Override + public ShortestPathResult getShortestPathResult(NodePair od) + { + return results.get(od); + } + + @Override + public int size() + { + return results.size(); + } + + @Override + public Collection> getResults() + { + return results.values(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java b/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java new file mode 100644 index 0000000..c76b467 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java @@ -0,0 +1,35 @@ +package org.sandag.abm.active; + +public class BinarySearch +{ + public static int binarySearch(double[] values, double target) + { + return binarySearch(values, target, 0, values.length - 1); + } + + public static int binarySearch(double[] values, double target, int lower, int upper) + { + if (lower <= upper) + { + int mid = (lower + upper) / 2; + + switch (Double.compare(values[mid], target)) + { + case 0: + return mid; + case 1: + return binarySearch(values, target, lower, upper - 1); + case -1: + return binarySearch(values, target, mid + 1, upper); + } + } + + if (values[lower] >= target) + { + return lower; + } else + { + return lower + 1; + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java new file mode 100644 index 0000000..eddfe63 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java @@ -0,0 +1,57 @@ +package org.sandag.abm.active; + +import java.util.Collection; +import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; + +public class CompositeShortestPathResultSet + implements ShortestPathResultSet +{ + private final Map, ShortestPathResultSet> spResultsLookup; + + public CompositeShortestPathResultSet() + { + spResultsLookup = new HashMap<>(); + } + + public void addShortestPathResults(ShortestPathResultSet spResults) + { + for (NodePair nodePair : spResults) + if (spResultsLookup.put(nodePair, spResults) != null) + throw new IllegalArgumentException( + "Repeated shortest path results for node pair: (" + + nodePair.getFromNode().getId() + "," + + nodePair.getToNode().getId() + ")"); + } + + @Override + public Iterator> iterator() + { + return spResultsLookup.keySet().iterator(); + } + + @Override + public ShortestPathResult getShortestPathResult(NodePair od) + { + return spResultsLookup.get(od).getShortestPathResult(od); + } + + @Override + public int size() + { + return spResultsLookup.size(); + } + + @Override + public Collection> getResults() + { + List> results = new LinkedList<>(); + for (ShortestPathResultSet spr : spResultsLookup.values()) + results.addAll(spr.getResults()); + return results; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java b/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java new file mode 100644 index 0000000..cee8240 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java @@ -0,0 +1,10 @@ +package org.sandag.abm.active; + +public class DestinationNotFoundException + extends Exception +{ + public DestinationNotFoundException(String message) + { + super(message); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java b/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java new file mode 100644 index 0000000..4d25097 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java @@ -0,0 +1,9 @@ +package org.sandag.abm.active; + +public interface Edge + extends Comparable> +{ + N getFromNode(); + + N getToNode(); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java b/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java new file mode 100644 index 0000000..1ee55f4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java @@ -0,0 +1,6 @@ +package org.sandag.abm.active; + +public interface EdgeEvaluator> +{ + double evaluate(E edge); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java new file mode 100644 index 0000000..16e757b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java @@ -0,0 +1,35 @@ +package org.sandag.abm.active; + +import java.util.Map; + +/** + * The {@code IntrazonalCalculation} class provides a framework for calculation + * intrazonals. + * + * @param + * The type of the zone nodes. + */ +public interface IntrazonalCalculation +{ + /** + * Get the intrazonal value given the origin node and the logsum values with + * that origin node. The logsum values may be stratified across markets, so + * the index of the market of interest is also provided. + * + * @param originNode + * The origin node. + * + * @param logsums + * The logsums with {@code originNode} as their origin. The + * logsums are stored as a map with the destination node as the + * key and an array of logsums as the value. The logsum array has + * a different logsum for each market. + * + * @param logsumIndex + * The index for the logsum of interest in the logsum arrays + * provided in {@code logsums}. + * + * @return the intrazonal logsum for {@code originNode}. + */ + double getIntrazonalValue(N originNode, Map logsums, int logsumIndex); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java new file mode 100644 index 0000000..be82cc1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java @@ -0,0 +1,365 @@ +package org.sandag.abm.active; + +import java.util.Map; +import org.apache.log4j.Logger; + +/** + * The {@code IntrazonalCalculations} class provides convenient default + * implementations of the {IntrazonalCalculation} interface. + * + */ +public class IntrazonalCalculations +{ + + private static final Logger logger = Logger.getLogger(IntrazonalCalculations.class); + + // this class is more of a static factory provider, so constructor is hidden + private IntrazonalCalculations() + { + } + + /** + * The {@code Factorizer} interface provides a framework for transforming an + * input value. It is essentially a function of one variable. + */ + public static interface Factorizer + { + /** + * Factor, or transform, an input value. + * + * @param inputValue + * The input value. + * + * @return the transformation of {@code inputValue}. + */ + double factor(double inputValue); + } + + /** + * Get a simple {@code Factorizer} implementation which applies a linear + * scale and offset. That is, for an input x, the function will + * return the following value: + *

+ * factor*x + offset + * + * @param factor + * The multiplicative factor. + * + * @param offset + * The addititive offset. + * + * @return the factorizer which will linearly scale and offset an input. + */ + public static Factorizer simpleFactorizer(final double factor, final double offset) + { + return new Factorizer() + { + @Override + public double factor(double inputValue) + { + return inputValue * factor + offset; + } + }; + } + + /** + * Get a {@code Factorizer} which applies a linear transformation, with + * different scale and offset for positive and negative input values. That + * is, for an input x, the factorizer function will return the + * following value + *

+ * factor*x + offset + *

+ * where factor and offset may differ according to + * whether x is positive or negative (if x is 0, + * it is considered positive). + * + * @param negativeFactor + * The multiplicative factor for negative input values. + * + * @param negativeOffset + * The additivie offset for negative input values. + * + * @param positiveFactor + * The multiplicative factor for positive input values. + * + * @param positiveOffset + * The additivie offset for positive input values. + * + * @return the factorizer which will linearly scale and offset an input, + * using different transoformations based on the input's sign. + */ + public static Factorizer positiveNegativeFactorizer(final double negativeFactor, + final double negativeOffset, final double positiveFactor, final double positiveOffset) + { + return new Factorizer() + { + @Override + public double factor(double inputValue) + { + if (inputValue < 0) return negativeFactor * inputValue + negativeOffset; + return positiveFactor * inputValue + positiveOffset; + } + }; + } + + /** + * Get an {@code IntrazonalCalculation} which will apply a function to the + * sum of the largest origin-based logsum values. That is, an intrazonal + * value is calculated by a function (defined by a {@code Factorizer}) which + * acts on the sum of the largest maxCount logsum values whose + * origin is the intrazonal's zone, where maxCount is set by + * the call to this function. + * + * @param + * The type of the zone nodes. + * + * @param factorizer + * The factorizer used to calculate the intrazonal value. + * + * @param maxCount + * The number of logsum values to be used in the intrazonal + * calculation. + * + * @return an intrazonal calculation which will apply {@code factorizer} to + * the sum of the largest {@code maxCount} logsum values. + */ + public static IntrazonalCalculation maxFactorIntrazonalCalculation( + final Factorizer factorizer, final int maxCount) + { + return new IntrazonalCalculation() + { + + @Override + public double getIntrazonalValue(N originNode, Map logsums, int logsumIndex) + { + MinHeap maxValues = new MinHeap(maxCount); + int initialCount = maxCount; + double minValue = 0; // will be filled in when needed + for (N node : logsums.keySet()) + { + if (!node.equals(originNode)) + { + double value = logsums.get(node)[logsumIndex]; + if (initialCount > 0) + { + maxValues.insert(value); + if (--initialCount == 0) minValue = maxValues.getMin(); + } else if (value > minValue) + { + maxValues.removeMin(); + maxValues.insert(value); + minValue = maxValues.getMin(); + } + } + } + return factorizer.factor(maxValues.getSum()); + } + }; + } + + /** + * Get an {@code IntrazonalCalculation} which will apply a function to the + * sum of the smallest origin-based logsum values. That is, an intrazonal + * value is calculated by a function (defined by a {@code Factorizer}) which + * acts on the sum of the smallest minCount logsum values whose + * origin is the intrazonal's zone, where minCount is set by + * the call to this function. + * + * @param + * The type of the zone nodes. + * + * @param factorizer + * The factorizer used to calculate the intrazonal value. + * + * @param minCount + * The number of logsum values to be used in the intrazonal + * calculation. + * + * @return an intrazonal calculation which will apply {@code factorizer} to + * the sum of the smallest {@code minCount} logsum values. + */ + public static IntrazonalCalculation minFactorIntrazonalCalculation( + final Factorizer factorizer, final int minCount) + { + return new IntrazonalCalculation() + { + + @Override + public double getIntrazonalValue(N originNode, Map logsums, int logsumIndex) + { + MaxHeap minValues = new MaxHeap(minCount); + int initialCount = minCount; + double maxValue = 0; // will be filled in when needed + for (N node : logsums.keySet()) + { + if (!node.equals(originNode)) + { + double value = logsums.get(node)[logsumIndex]; + if (initialCount > 0) + { + minValues.insert(value); + if (--initialCount == 0) maxValue = minValues.getMax(); + } else if (value < maxValue) + { + minValues.removeMax(); + minValues.insert(value); + maxValue = minValues.getMax(); + } + } + } + return factorizer.factor(minValues.getSum()); + } + }; + } + + private static class Heap + { + protected final double[] heap; + protected int end; + + private Heap(int size) + { + heap = new double[size]; + end = 0; + } + + public double getSum() + { + double sum = 0; + for (int i = 0; i < end; i++) + sum += heap[i]; + return sum; + } + } + + private static class MaxHeap + extends Heap + { + + private MaxHeap(int size) + { + super(size); + } + + public void insert(double value) + { + int point = end++; + if (point == 0) + { + heap[0] = value; + return; + } + while (point > 0) + { + int newPoint = (point - 1) / 2; + if (heap[newPoint] < value) + { + heap[point] = heap[newPoint]; + point = newPoint; + } else + { + heap[point] = value; + break; + } + if (point == 0) heap[0] = value; + } + } + + public double getMax() + { + return heap[0]; + } + + public double removeMax() + { + double max = heap[0]; + heap[0] = heap[--end]; + double value = heap[0]; + int point = 0; + while (true) + { + int left = 2 * point + 1; + int right = left + 1; + int largest = point; + if ((left < end) && (heap[left] > heap[largest])) largest = left; + if ((right < end) && (heap[right] > heap[largest])) largest = right; + if (largest != point) + { + heap[point] = heap[largest]; + point = largest; + } else + { + heap[point] = value; + break; + } + } + return max; + } + } + + private static class MinHeap + extends Heap + { + + private MinHeap(int size) + { + super(size); + } + + public void insert(double value) + { + int point = end++; + if (point == 0) + { + heap[0] = value; + return; + } + while (point > 0) + { + int newPoint = (point - 1) / 2; + if (heap[newPoint] > value) + { + heap[point] = heap[newPoint]; + point = newPoint; + } else + { + heap[point] = value; + break; + } + if (point == 0) heap[0] = value; + } + } + + public double getMin() + { + return heap[0]; + } + + public double removeMin() + { + double min = heap[0]; + heap[0] = heap[--end]; + double value = heap[0]; + int point = 0; + while (true) + { + int left = 2 * point + 1; + int right = left + 1; + int largest = point; + if ((left < end) && (heap[left] < heap[largest])) largest = left; + if ((right < end) && (heap[right] < heap[largest])) largest = right; + if (largest != point) + { + heap[point] = heap[largest]; + point = largest; + } else + { + heap[point] = value; + break; + } + } + return min; + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java new file mode 100644 index 0000000..5a52993 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java @@ -0,0 +1,11 @@ +package org.sandag.abm.active; + +public interface ModifiableShortestPathResultSet + extends ShortestPathResultSet +{ + void addResult(ShortestPathResult spResult); + + void addResult(NodePair od, Path path, double cost); + + void addAll(ShortestPathResultSet results); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Network.java b/sandag_abm/src/main/java/org/sandag/abm/active/Network.java new file mode 100644 index 0000000..025035e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/Network.java @@ -0,0 +1,163 @@ +package org.sandag.abm.active; + +import java.util.Collection; +import java.util.Iterator; + +public interface Network, T extends Traversal> +{ + N getNode(int nodeId); + + E getEdge(N fromNode, N toNode); + + E getEdge(NodePair nodes); + + T getTraversal(E fromEdge, E toEdge); + + Collection getSuccessors(N node); + + Collection getPredecessors(N node); + + Iterator nodeIterator(); + + Iterator edgeIterator(); + + Iterator traversalIterator(); + + boolean containsNodeId(int id); + + boolean containsNode(N node); + + boolean containsEdge(N fromNode, N toNode); + + boolean containsTraversal(E fromEdge, E toEdge); + + // public void addNode(T node) + // { + // if ( nodeIndex.containsKey(node.getId()) ) { + // throw new RuntimeException("Network already contains Node with id " + + // node.getId()); + // } + // nodeIndex.put(node.getId(), nodes.size()); + // nodes.add(node); + // if (! successorIndex.containsKey(node.getId()) ) { + // successorIndex.put(node.getId(), new ArrayList()); } + // if (! predecessorIndex.containsKey(node.getId()) ) { + // predecessorIndex.put(node.getId(), new ArrayList()); } + // } + // + // public void addEdge(U edge) + // { + // int fromId = edge.getFromNode(); + // int toId = edge.getToNode(); + // EdgeKey edgeIndexKey = new EdgeKey(fromId, toId); + // + // if ( edgeIndex.containsKey(edgeIndexKey) ) { + // throw new RuntimeException("Network already contains Edge with fromId " + + // edge.getFromNode() + " and toId " + edge.getToNode()); + // } + // + // edgeIndex.put(edgeIndexKey, edges.size()); + // edges.add(edge); + // + // if ( ! successorIndex.containsKey(fromId) ) { successorIndex.put(fromId, + // new ArrayList()); } + // if ( ! predecessorIndex.containsKey(toId) ) { predecessorIndex.put(toId, + // new ArrayList()); } + // + // if ( ! successorIndex.get(fromId).contains(toId) ) { + // successorIndex.get(fromId).add(toId); } + // if ( ! predecessorIndex.get(toId).contains(fromId) ) { + // predecessorIndex.get(toId).add(fromId); } + // } + // + // public void addTraversal(V traversal) + // { + // int startId = traversal.getStartId(); + // int thruId = traversal.getThruId(); + // int endId = traversal.getEndId(); + // TraversalKey traversalIndexKey = new TraversalKey(startId, thruId, + // endId); + // + // traversalIndex.put(traversalIndexKey, traversals.size()); + // traversals.add(traversal); + // } + // + // public boolean containsNodeId(int id) { + // return nodeIndex.containsKey(id); + // } + // + // public boolean containsEdgeIds(int[] ids) { + // return edgeIndex.containsKey(new EdgeKey(ids[0],ids[1])); + // } + // + // public boolean containsTraversalIds(int[] ids) { + // return traversalIndex.containsKey(new + // TraversalKey(ids[0],ids[1],ids[2])); + // } + // + // private class EdgeKey { + // private int fromId, toId; + // + // EdgeKey(int fromId, int toId) { + // this.fromId = fromId; + // this.toId = toId; + // } + // + // @Override + // public int hashCode() + // { + // final int prime = 31; + // int result = 1; + // result = prime * result + fromId; + // result = prime * result + toId; + // return result; + // } + // + // @Override + // public boolean equals(Object obj) + // { + // if (this == obj) return true; + // if (obj == null) return false; + // if (getClass() != obj.getClass()) return false; + // EdgeKey other = (EdgeKey) obj; + // if (fromId != other.fromId) return false; + // if (toId != other.toId) return false; + // return true; + // } + // } + // + // private class TraversalKey { + // private int startId, thruId, endId; + // + // public TraversalKey(int startId, int thruId, int endId) + // { + // this.startId = startId; + // this.thruId = thruId; + // this.endId = endId; + // } + // + // @Override + // public int hashCode() + // { + // final int prime = 31; + // int result = 1; + // result = prime * result + endId; + // result = prime * result + startId; + // result = prime * result + thruId; + // return result; + // } + // + // @Override + // public boolean equals(Object obj) + // { + // if (this == obj) return true; + // if (obj == null) return false; + // if (getClass() != obj.getClass()) return false; + // TraversalKey other = (TraversalKey) obj; + // if (endId != other.endId) return false; + // if (startId != other.startId) return false; + // if (thruId != other.thruId) return false; + // return true; + // } + // } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java new file mode 100644 index 0000000..02d471b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java @@ -0,0 +1,35 @@ +package org.sandag.abm.active; + +import java.util.Collection; + +public abstract class NetworkFactory, T extends Traversal> +{ + + public Network createNetwork() + { + Network network = new SimpleNetwork<>(getNodes(), getEdges(), getTraversals()); + calculateDerivedNodeAttributes(network); + calculateDerivedEdgeAttributes(network); + calculateDerivedTraversalAttributes(network); + return network; + } + + protected abstract Collection getNodes(); + + protected abstract Collection getEdges(); + + protected abstract Collection getTraversals(); + + protected void calculateDerivedNodeAttributes(Network network) + { + } + + protected void calculateDerivedEdgeAttributes(Network network) + { + } + + protected void calculateDerivedTraversalAttributes(Network network) + { + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Node.java b/sandag_abm/src/main/java/org/sandag/abm/active/Node.java new file mode 100644 index 0000000..d0fcd45 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/Node.java @@ -0,0 +1,7 @@ +package org.sandag.abm.active; + +public interface Node + extends Comparable +{ + int getId(); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java b/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java new file mode 100644 index 0000000..c375a3d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java @@ -0,0 +1,50 @@ +package org.sandag.abm.active; + +import java.util.Objects; + +public class NodePair + implements Comparable> +{ + private final N fromNode; + private final N toNode; + + public NodePair(N fromNode, N toNode) + { + this.fromNode = fromNode; + this.toNode = toNode; + } + + public int compareTo(NodePair other) + { + int c = fromNode.compareTo(other.fromNode); + if (c == 0) c = toNode.compareTo(other.toNode); + return c; + } + + public N getFromNode() + { + return fromNode; + } + + public N getToNode() + { + return toNode; + } + + public boolean equals(Object other) + { + if ((other == null) || (!(other instanceof NodePair))) return false; + NodePair np = (NodePair) other; + return (fromNode.equals(np.fromNode)) && (toNode.equals(np.toNode)); + } + + public int hashCode() + { + return Objects.hash(fromNode, toNode); + } + + public String toString() + { + return ""; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java b/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java new file mode 100644 index 0000000..ca82302 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java @@ -0,0 +1,230 @@ +package org.sandag.abm.active; + +import java.util.HashSet; +import java.util.Queue; +import java.util.Set; +import java.util.concurrent.ConcurrentLinkedQueue; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.ForkJoinPool; +import java.util.concurrent.atomic.AtomicInteger; +import com.pb.sawdust.util.concurrent.DnCRecursiveTask; + +public class ParallelSingleSourceDijkstra + implements ShortestPathStrategy +{ + private final ShortestPathStrategy sp; + private final ParallelMethod method; + private final int SEGMENT_SIZE = 50; + + public ParallelSingleSourceDijkstra(ShortestPathStrategy sp, ParallelMethod method) + { + this.sp = sp; + this.method = method; + } + + public static enum ParallelMethod + { + FORK_JOIN, QUEUE + } + + @Override + public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, + double maxCost) + { + switch (method) + { + case FORK_JOIN: + { + ShortestPathRecursiveTask task = new ShortestPathRecursiveTask(sp, originNodes, + destinationNodes, maxCost); + new ForkJoinPool().execute(task); + ModifiableShortestPathResultSet sprc = task.getResult(); + return sprc; + } + case QUEUE: + { + int threadCount = Runtime.getRuntime().availableProcessors(); + ExecutorService executor = Executors.newFixedThreadPool(threadCount); + final Queue> sprcQueue = new ConcurrentLinkedQueue<>(); + final Queue originNodeQueue = new ConcurrentLinkedQueue<>(originNodes); + ThreadLocal> sprcThreadLocal = new ThreadLocal>() + { + @Override + public ModifiableShortestPathResultSet initialValue() + { + ModifiableShortestPathResultSet sprc = new BasicShortestPathResultSet<>(); + sprcQueue.add(sprc); + return sprc; + } + }; + final CountDownLatch latch = new CountDownLatch(threadCount); + final AtomicInteger counter = new AtomicInteger(); + for (int i = 0; i < threadCount; i++) + executor.execute(new QueueMethodTask(sp, originNodeQueue, destinationNodes, + maxCost, counter, sprcThreadLocal, latch)); + try + { + latch.await(); + } catch (InterruptedException e) + { + throw new RuntimeException(e); + } + executor.shutdown(); + + ModifiableShortestPathResultSet finalContainer = null; + for (ModifiableShortestPathResultSet sprc : sprcQueue) + if (finalContainer == null) finalContainer = sprc; + else finalContainer.addAll(sprc); + + return finalContainer; + } + default: + throw new IllegalStateException("Should not be here."); + } + } + + @Override + public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes) + { + return getShortestPaths(originNodes, destinationNodes, Double.POSITIVE_INFINITY); + } + + private class QueueMethodTask + implements Runnable + { + private final ShortestPathStrategy sp; + private final Queue originNodes; + private final Set destinationNodes; + private final double maxCost; + private final AtomicInteger counter; + private final ThreadLocal> spr; + private final CountDownLatch latch; + + private QueueMethodTask(ShortestPathStrategy sp, Queue originNodes, + Set destinationNodes, double maxCost, AtomicInteger counter, + ThreadLocal> spr, CountDownLatch latch) + { + this.sp = sp; + this.destinationNodes = destinationNodes; + this.originNodes = originNodes; + this.maxCost = maxCost; + this.counter = counter; + this.spr = spr; + this.latch = latch; + } + + @Override + public void run() + { + int segmentSize = SEGMENT_SIZE; + final Set origins = new HashSet<>(); + while (originNodes.size() > 0) + { + while ((originNodes.size() > 0) && (origins.size() < segmentSize)) + { + N origin = originNodes.poll(); + if (origin != null) origins.add(origin); + } + if (origins.size() == 0) break; + ShortestPathResultSet result = sp.getShortestPaths(origins, destinationNodes, + maxCost); + ModifiableShortestPathResultSet sprc = spr.get(); + for (ShortestPathResult spResult : result.getResults()) + sprc.addResult(spResult); + int c = counter.addAndGet(origins.size()); + if (c % segmentSize < origins.size()) + System.out.println(" done with " + ((c / segmentSize) * segmentSize) + + " origins"); + origins.clear(); + } + latch.countDown(); + } + } + + private class ShortestPathRecursiveTask + extends DnCRecursiveTask> + { + AtomicInteger counter; + private final ShortestPathStrategy sp; + private final Set destinations; + private final Node[] origins; + private final double maxCost; + + protected ShortestPathRecursiveTask(ShortestPathStrategy sp, Set origins, + Set destinations, double maxCost) + { + super(0, origins.size()); + this.sp = sp; + this.origins = origins.toArray(new Node[origins.size()]); + this.destinations = destinations; + this.maxCost = maxCost; + counter = new AtomicInteger(0); + } + + protected ShortestPathRecursiveTask(long start, long length, + DnCRecursiveTask> next, + ShortestPathStrategy sp, Node[] origins, Set destinations, double maxCost, + AtomicInteger counter) + { + super(start, length, next); + this.sp = sp; + this.origins = origins; + this.destinations = destinations; + this.maxCost = maxCost; + this.counter = counter; + } + + @Override + @SuppressWarnings("unchecked") + // origins only hold N, we just can't declare as such because of + // generics + protected ModifiableShortestPathResultSet computeTask(long start, long length) + { + Set originNodes = new HashSet<>(); + int end = (int) (start + length); + for (int n = (int) start; n < end; n++) + originNodes.add((N) origins[n]); + ShortestPathResultSet result = sp.getShortestPaths(originNodes, destinations); + ModifiableShortestPathResultSet spr = new BasicShortestPathResultSet<>(); + for (ShortestPathResult spResult : result.getResults()) + spr.addResult(spResult); + + int c = counter.addAndGet((int) length); + if (c % 10 < length) + System.out.println(" done with " + ((c / 10) * 10) + " origins"); + return spr; + } + + @Override + protected boolean continueDividing(long newLength) + { + return (newLength > 5) && (getSurplusQueuedTaskCount() < 3); + } + + @Override + protected DnCRecursiveTask> getNextTask(long start, + long length, DnCRecursiveTask> next) + { + return new ShortestPathRecursiveTask(start, length, next, sp, origins, destinations, + maxCost, counter); + } + + @Override + protected ModifiableShortestPathResultSet joinResults( + ModifiableShortestPathResultSet spr1, ModifiableShortestPathResultSet spr2) + { + if (spr1.size() > spr2.size()) + { + spr1.addAll(spr2); + return spr1; + } else + { + spr2.addAll(spr1); + return spr2; + } + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Path.java b/sandag_abm/src/main/java/org/sandag/abm/active/Path.java new file mode 100644 index 0000000..337db3b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/Path.java @@ -0,0 +1,82 @@ +package org.sandag.abm.active; + +import java.util.Iterator; +import java.util.NoSuchElementException; + +public class Path + implements Iterable +{ + private final Path predecessorPath; + private final N next; + private final int length; + + public Path(Path predecessorPath, N next) + { + this.predecessorPath = predecessorPath; + this.next = next; + this.length = predecessorPath == null ? 1 : predecessorPath.length + 1; + } + + public Path(N first) + { + this(null, first); + } + + public int getLength() + { + return length; + } + + public N getNode(int index) + { + if (index < 0 || index >= length) + throw new IllegalArgumentException("Invalid index " + index + " for path of length " + + length); + return getNodeNoChecks(index + 1); + } + + private N getNodeNoChecks(int index) + { // index here is 1-based, not zero based! + return index == length ? next : predecessorPath.getNodeNoChecks(index); + } + + public Path extendPath(N next) + { + return new Path(this, next); + } + + public String getPathString() + { + StringBuilder sb = new StringBuilder(); + for (N n : this) + sb.append(n.getId()).append(n == next ? "" : " "); + return sb.toString(); + } + + public Iterator iterator() + { + return new Iterator() + { + private int point = 0; + + @Override + public boolean hasNext() + { + return point < length; + } + + @Override + public N next() + { + if (point < length) return getNodeNoChecks(++point); + else throw new NoSuchElementException(); + } + + @Override + public void remove() + { + throw new UnsupportedOperationException(); + } + }; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java new file mode 100644 index 0000000..d8c4e25 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java @@ -0,0 +1,177 @@ +package org.sandag.abm.active; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +public class PathAlternativeList> +{ + private List> paths; + private NodePair odPair; + private List sizeMeasures; + private PathSizeCalculator sizeCalculator; + private double sizeMeasureTotal; + private boolean sizeMeasuresUpdated; + Network network; + EdgeEvaluator lengthEvaluator; + + public PathAlternativeList(NodePair odPair, Network network, + EdgeEvaluator lengthEvaluator) + { + paths = new ArrayList>(); + sizeMeasures = new ArrayList(); + this.odPair = odPair; + sizeMeasuresUpdated = true; + this.network = network; + this.lengthEvaluator = lengthEvaluator; + this.sizeCalculator = new PathSizeCalculator(this); + this.sizeMeasureTotal = 0.0; + } + + public Network getNetwork() + { + return network; + } + + public void add(Path path) + { + if (!path.getNode(0).equals(odPair.getFromNode()) + || !path.getNode(path.getLength() - 1).equals(odPair.getToNode())) + { + throw new IllegalStateException( + "OD pair of path does not match that of path alternative list"); + } + for (Path otherPath : paths) + { + if (path.equals(otherPath)) + { + return; + } + } + paths.add(path); + sizeMeasures.add(0.0); + if (sizeCalculator == null) + { + sizeMeasuresUpdated = false; + } else + { + sizeCalculator.update(); + } + } + + public List getSizeMeasures() + { + return sizeMeasures; + } + + private void setSizeMeasure(int index, double value) + { + sizeMeasureTotal += value - sizeMeasures.get(index); + sizeMeasures.set(index, value); + } + + public double getSizeMeasureTotal() + { + return sizeMeasureTotal; + } + + public int getCount() + { + return paths.size(); + } + + public Path get(int index) + { + return paths.get(index); + } + + public boolean areSizeMeasuresUpdated() + { + return sizeMeasuresUpdated; + } + + public void clearPathSizeCalculator() + { + sizeCalculator = null; + } + + public void restartPathSizeCalculator() + { + if (sizeCalculator == null) + { + sizeCalculator = new PathSizeCalculator(this); + sizeMeasuresUpdated = true; + } + } + + private class PathSizeCalculator + { + Map> incidenceMap; + List lengths; + PathAlternativeList alternatives; + int nUsingEdge; + double edgeLength; + + private PathSizeCalculator(PathAlternativeList alternatives) + { + incidenceMap = new HashMap>(); + lengths = new ArrayList(); + this.alternatives = alternatives; + if (alternatives.getCount() > 0) + { + for (int i = 0; i < alternatives.getCount(); i++) + { + alternatives.setSizeMeasure(i, 0.0); + update(); + } + } + } + + private void update() + { + lengths.add(0.0); + N previous = null; + E edge; + int index = lengths.size() - 1; + double decrement; + for (N node : alternatives.get(index)) + { + if (previous != null) + { + edge = network.getEdge(previous, node); + if (!incidenceMap.containsKey(edge)) + { + incidenceMap.put(edge, new ArrayList()); + } + incidenceMap.get(edge).add(index); + edgeLength = lengthEvaluator.evaluate(edge); + lengths.set(index, lengths.get(index) + edgeLength); + nUsingEdge = incidenceMap.get(edge).size(); + alternatives.setSizeMeasure(index, alternatives.getSizeMeasures().get(index) + + edgeLength / nUsingEdge); + for (Integer i : incidenceMap.get(edge).subList(0, nUsingEdge - 1)) + { + decrement = edgeLength / lengths.get(i) / (nUsingEdge) / (nUsingEdge - 1); + alternatives.setSizeMeasure(i, alternatives.getSizeMeasures().get(i) + - decrement); + } + } + previous = node; + } + alternatives.setSizeMeasure(index, + alternatives.getSizeMeasures().get(index) / lengths.get(index)); + } + + } + + public NodePair getODPair() + { + return odPair; + } + + public EdgeEvaluator getLengthEvaluator() + { + return lengthEvaluator; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..756f2ad --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java @@ -0,0 +1,52 @@ +package org.sandag.abm.active; + +import java.util.Map; +import java.util.Set; + +public interface PathAlternativeListGenerationConfiguration, T extends Traversal> +{ + public Network getNetwork(); + + public EdgeEvaluator getEdgeLengthEvaluator(); + + public EdgeEvaluator getEdgeCostEvaluator(); + + public TraversalEvaluator getTraversalCostEvaluator(); + + public double getMaxCost(); + + public double[] getSampleDistanceBreaks(); + + public double[] getSamplePathSizes(); + + public double[] getSampleMinCounts(); + + public double[] getSampleMaxCounts(); + + public boolean isRandomCostSeeded(); + + public Map> getNearbyZonalDistanceMap(); + + public Map getOriginZonalCentroidIdMap(); + + public Map getDestinationZonalCentroidIdMap(); + + public String getOutputDirectory(); + + public Set getTraceOrigins(); + + public Map getPropertyMap(); + + public Map getInverseOriginZonalCentroidIdMap(); + + public Map getInverseDestinationZonalCentroidIdMap(); + + public boolean isTraceExclusive(); + + public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed); + + public boolean isIntrazonalsNeeded(); + + public double getDefaultMinutesPerMile(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java new file mode 100644 index 0000000..a446045 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java @@ -0,0 +1,59 @@ +package org.sandag.abm.active; + +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; + +public class PathAlternativeListWriter> + implements AutoCloseable +{ + private FileWriter pathWriter; + private FileWriter linkWriter; + + public PathAlternativeListWriter(String pathFileName, String linkFileName) throws IOException + { + pathWriter = new FileWriter(new File(pathFileName)); + linkWriter = new FileWriter(new File(linkFileName)); + } + + public void writeHeaders() throws IOException + { + pathWriter.write("alt,origNode,destNode,length,size\n"); + linkWriter.write("alt,origNode,destNode,link,fromNode,toNode\n"); + } + + public void write(PathAlternativeList alternativeList) throws IOException + { + Path path; + int index = 1; + for (int i = 0; i < alternativeList.getCount(); i++) + { + path = alternativeList.get(i); + pathWriter.write(index + "," + path.getNode(0).getId() + "," + + path.getNode(path.getLength() - 1).getId() + "," + path.getLength() + "," + + alternativeList.getSizeMeasures().get(i) + "\n"); + N previous = null; + int j = 0; + for (N node : path) + { + if (previous != null) + { + linkWriter.write(index + "," + path.getNode(0).getId() + "," + + path.getNode(path.getLength() - 1).getId() + "," + j + "," + + previous.getId() + "," + node.getId() + "\n"); + } + previous = node; + j++; + } + index++; + } + } + + public void close() throws IOException + { + pathWriter.flush(); + pathWriter.close(); + linkWriter.flush(); + linkWriter.close(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java b/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java new file mode 100644 index 0000000..bf2ba01 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java @@ -0,0 +1,156 @@ +package org.sandag.abm.active; + +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.PriorityQueue; +import java.util.Set; + +public class RepeatedSingleSourceDijkstra, T extends Traversal> + implements ShortestPathStrategy +{ + private final Network network; + private final EdgeEvaluator edgeEvaluator; + private final TraversalEvaluator traversalEvaluator; + + public RepeatedSingleSourceDijkstra(Network network, EdgeEvaluator edgeEvaluator, + TraversalEvaluator traversalEvaluator) + { + this.network = network; + this.edgeEvaluator = edgeEvaluator; + this.traversalEvaluator = traversalEvaluator; + } + + private class TraversedEdge + implements Comparable + { + private final E edge; + private final double cost; + private final Path path; + + private TraversedEdge(E edge, double cost, Path path) + { + this.edge = edge; + this.cost = cost; + this.path = path; + } + + public int compareTo(TraversedEdge other) + { + return Double.compare(cost, other.cost); + } + } + + @Override + public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes) + { + return getShortestPaths(originNodes, destinationNodes, Double.POSITIVE_INFINITY); + } + + @Override + public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, + double maxCost) + { + ModifiableShortestPathResultSet spResults = new BasicShortestPathResultSet<>(); + for (N originNode : originNodes) + spResults.addAll(getShortestPaths(originNode, destinationNodes, maxCost)); + return spResults; + } + + protected ShortestPathResultSet getShortestPaths(N originNode, Set destinationNodes, + double maxCost) + { + + BasicShortestPathResultSet spResults = new BasicShortestPathResultSet<>(); + Map finalCosts = new HashMap<>(); // cost to (and including) + // edge + + PriorityQueue traversalQueue = new PriorityQueue<>(); + + Set targets = new HashSet<>(destinationNodes); + Path basePath = new Path<>(originNode); + + // Don't remove origin node, and then we can force a circle for + // intrazonal trips + // if (targets.contains(originNode)) { + // targets.remove(originNode); + // costs.put(originNode,0.0); + // paths.put(originNode,basePath); + // } + + // initialize traversalQueue and costs + for (N successor : network.getSuccessors(originNode)) + { + E edge = network.getEdge(originNode, successor); + double edgeCost = edgeEvaluator.evaluate(edge); + if (edgeCost < 0) + { + throw new RuntimeException("Negative weight found for edge with fromNode " + + edge.getFromNode().getId() + " and toNode " + edge.getToNode().getId()); + } + + if (edgeCost < maxCost) + { + TraversedEdge traversedEdge = new TraversedEdge(edge, edgeCost, + basePath.extendPath(successor)); + traversalQueue.add(traversedEdge); + } + } + + double traversalCost; + + // dijkstra + while (!traversalQueue.isEmpty() && !targets.isEmpty()) + { + TraversedEdge traversedEdge = traversalQueue.poll(); + E edge = traversedEdge.edge; + + if (finalCosts.containsKey(edge)) // already considered + continue; + Path path = traversedEdge.path; + double cost = traversedEdge.cost; + + finalCosts.put(edge, cost); + N fromNode = edge.getFromNode(); + N toNode = edge.getToNode(); + if (targets.remove(toNode)) + { + spResults.addResult(new NodePair(originNode, toNode), path, cost); + } + + for (N successor : network.getSuccessors(toNode)) + { + if (successor.equals(fromNode)) continue; // no u-turns will be + // allowed, so don't + // pollute heap + T traversal = network.getTraversal(traversedEdge.edge, + network.getEdge(toNode, successor)); + traversalCost = evaluateTraversalCost(traversal); + if (traversalCost < 0) + { + throw new RuntimeException( + "Negative weight found for traversal with start node " + + traversal.getFromEdge().getFromNode().getId() + + ", thru node " + traversal.getFromEdge().getToNode().getId() + + ", and end node " + traversal.getToEdge().getToNode().getId()); + } + traversalCost += cost; + if (traversalCost < maxCost) + traversalQueue.add(new TraversedEdge(traversal.getToEdge(), traversalCost, path + .extendPath(successor))); + } + } + + // Not returning null path references and infinite costs for nodes not + // found for possibility of insufficient memory + + return spResults; + } + + protected double evaluateTraversalCost(T traversal) + { + return edgeEvaluator.evaluate(traversal.getToEdge()) + + traversalEvaluator.evaluate(traversal); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java new file mode 100644 index 0000000..3804815 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java @@ -0,0 +1,31 @@ +package org.sandag.abm.active; + +public class ShortestPathResult +{ + private final NodePair od; + private final Path path; + private final double cost; + + public ShortestPathResult(NodePair od, Path path, double cost) + { + this.od = od; + this.path = path; + this.cost = cost; + } + + public NodePair getOriginDestination() + { + return od; + } + + public Path getPath() + { + return path; + } + + public double getCost() + { + return cost; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java new file mode 100644 index 0000000..cf2334d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java @@ -0,0 +1,13 @@ +package org.sandag.abm.active; + +import java.util.Collection; + +public interface ShortestPathResultSet + extends Iterable> +{ + int size(); + + ShortestPathResult getShortestPathResult(NodePair od); + + Collection> getResults(); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java new file mode 100644 index 0000000..9d47c64 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java @@ -0,0 +1,11 @@ +package org.sandag.abm.active; + +import java.util.Set; + +public interface ShortestPathStrategy +{ + ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, + double maxCost); + + ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java new file mode 100644 index 0000000..0bdbe5b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java @@ -0,0 +1,50 @@ +package org.sandag.abm.active; + +import java.util.Objects; + +public class SimpleEdge + implements Edge +{ + private final N fromNode; + private final N toNode; + + public SimpleEdge(N fromNode, N toNode) + { + this.fromNode = fromNode; + this.toNode = toNode; + } + + @Override + public N getFromNode() + { + return fromNode; + } + + @Override + public N getToNode() + { + return toNode; + } + + @Override + public int compareTo(Edge o) + { + int fromResult = this.fromNode.compareTo(o.getFromNode()); + int toResult = this.toNode.compareTo(o.getToNode()); + return fromResult + ((fromResult == 0) ? 1 : 0) * toResult; + } + + @Override + public int hashCode() + { + return Objects.hash(fromNode, toNode); + } + + @Override + public boolean equals(Object o) + { + if ((o == null) || !(o instanceof Edge)) return false; + Edge other = (Edge) o; + return fromNode.equals(other.getFromNode()) && toNode.equals(other.getToNode()); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java new file mode 100644 index 0000000..d88553b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java @@ -0,0 +1,141 @@ +package org.sandag.abm.active; + +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedHashMap; +import java.util.LinkedList; +import java.util.Map; + +public class SimpleNetwork, T extends Traversal> + implements Network +{ + private final Map nodes; + private final Map, E> edges; + private final Map traversals; + + private final Map> successors; + private final Map> predecessors; + + public SimpleNetwork(Collection nodes, Collection edges, Collection traversals) + { + this.nodes = new LinkedHashMap<>(); // use LinkedHashMap for fast + // iteration over keys + this.edges = new LinkedHashMap<>(); + this.traversals = new LinkedHashMap<>(); + + for (N node : nodes) + this.nodes.put(node.getId(), node); + for (E edge : edges) + this.edges.put(new NodePair(edge.getFromNode(), edge.getToNode()), edge); + for (T traversal : traversals) + this.traversals.put(new EdgePair(traversal.getFromEdge(), traversal.getToEdge()), + traversal); + + successors = new HashMap<>(); // save memory and insertion time over + // LinkedHashMap + predecessors = new HashMap<>(); + + for (N node : this.nodes.values()) + { + successors.put(node, new LinkedList()); + predecessors.put(node, new LinkedList()); + } + for (NodePair nodePair : this.edges.keySet()) + { + N from = nodePair.getFromNode(); + N to = nodePair.getToNode(); + successors.get(from).add(to); + predecessors.get(to).add(from); + } + } + + @Override + public N getNode(int nodeId) + { + return nodes.get(nodeId); + } + + @Override + public E getEdge(N fromNode, N toNode) + { + return getEdge(new NodePair(fromNode, toNode)); + } + + @Override + public E getEdge(NodePair nodes) + { + return edges.get(nodes); + } + + @Override + public T getTraversal(E fromEdge, E toEdge) + { + return traversals.get(new EdgePair(fromEdge, toEdge)); + } + + @Override + public Collection getSuccessors(Node node) + { + return Collections.unmodifiableCollection(successors.get(node)); + } + + @Override + public Collection getPredecessors(Node node) + { + return Collections.unmodifiableCollection(predecessors.get(node)); + } + + @Override + public Iterator nodeIterator() + { + return nodes.values().iterator(); + } + + @Override + public Iterator edgeIterator() + { + return edges.values().iterator(); + } + + @Override + public Iterator traversalIterator() + { + return traversals.values().iterator(); + } + + @Override + public boolean containsNodeId(int id) + { + return nodes.containsKey(id); + } + + @Override + public boolean containsNode(Node node) + { + return nodes.containsValue(node); + } + + @Override + public boolean containsEdge(N fromNode, N toNode) + { + return edges.containsKey(new NodePair(fromNode, toNode)); + } + + @Override + public boolean containsTraversal(E fromEdge, E toEdge) + { + return traversals.containsKey(new EdgePair(fromEdge, toEdge)); + } + + private class EdgePair + extends SimpleTraversal + { + public EdgePair(E fromEdge, E toEdge) + { + super(fromEdge, toEdge); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java new file mode 100644 index 0000000..f3b6bc9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java @@ -0,0 +1,45 @@ +package org.sandag.abm.active; + +import java.util.Objects; + +public class SimpleNode + implements Node +{ + private final int id; + + public SimpleNode(int id) + { + this.id = id; + } + + @Override + public int getId() + { + return id; + } + + @Override + public int compareTo(Node node) + { + return Integer.compare(id, node.getId()); + } + + @Override + public int hashCode() + { + return Objects.hash(id); + } + + @Override + public boolean equals(Object o) + { + if ((o == null) || !(o instanceof Node)) return false; + return id == ((Node) o).getId(); + } + + public String toString() + { + return ""; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java new file mode 100644 index 0000000..02b7f49 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java @@ -0,0 +1,46 @@ +package org.sandag.abm.active; + +import java.util.Objects; + +public class SimpleTraversal> + implements Traversal +{ + private final E fromEdge; + private final E toEdge; + + public SimpleTraversal(E fromEdge, E toEdge) + { + this.fromEdge = fromEdge; + this.toEdge = toEdge; + } + + @Override + public E getFromEdge() + { + return fromEdge; + } + + @Override + public E getToEdge() + { + return toEdge; + } + + @Override + public int hashCode() + { + return Objects.hash(fromEdge, toEdge); + } + + @Override + public boolean equals(Object obj) + { + if ((obj == null) || (!(obj instanceof Traversal))) return false; + Traversal traversal = (Traversal) obj; + if (fromEdge == null) return (fromEdge == traversal.getFromEdge()) + && (toEdge.equals(traversal.getToEdge())); + else return (fromEdge.equals(traversal.getFromEdge())) + && (toEdge.equals(traversal.getToEdge())); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java new file mode 100644 index 0000000000000000000000000000000000000000..6e508e144dc957939a3ef1e872486f5df66c528d GIT binary patch literal 131 LcmZQz7;pdp0D}Ml literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java b/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java new file mode 100644 index 0000000..0e4c197 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java @@ -0,0 +1,6 @@ +package org.sandag.abm.active; + +public interface TraversalEvaluator> +{ + double evaluate(T traversal); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java new file mode 100644 index 0000000..2729f27 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java @@ -0,0 +1,216 @@ +package org.sandag.abm.active.sandag; + +import java.io.BufferedReader; +import java.io.FileReader; +import java.io.IOException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Stop; +import org.sandag.abm.ctramp.Tour; + +public class BikeAssignmentTripReader +{ + private String indivTripFileName, jointTripFileName, indivTourFileName, + jointTourFileName, personFileName, hhFileName; + private static final int DEFAULT_MANDATORY_PURPOSE_INDEX = 1; + private static final int DEFAULT_NONMANDATORY_PURPOSE_INDEX = 4; + private static final int BIKE_MODE_INDEX = 10; + + private static String PROPERTIES_HOUSEHOLD_FILENAME = "PopulationSynthesizer.InputToCTRAMP.HouseholdFile"; + private static String PROPERTIES_PERSON_FILENAME = "PopulationSynthesizer.InputToCTRAMP.PersonFile"; + private static String PROPERTIES_INDIV_TOUR_FILENAME = "Results.IndivTourDataFile"; + private static String PROPERTIES_JOINT_TOUR_FILENAME = "Results.JointTourDataFile"; + private static String PROPERTIES_INDIV_TRIP_FILENAME = "Results.IndivTripDataFile"; + private static String PROPERTIES_JOINT_TRIP_FILENAME = "Results.JointTripDataFile"; + private static String PROPERTIES_PROJECT_DIR = "Project.Directory"; + + public BikeAssignmentTripReader(Map propertyMap) + { + String dir = propertyMap.get(PROPERTIES_PROJECT_DIR); + this.indivTripFileName = dir + "/" + propertyMap.get(PROPERTIES_INDIV_TRIP_FILENAME); + this.jointTripFileName = dir + "/" + propertyMap.get(PROPERTIES_JOINT_TRIP_FILENAME); + this.indivTourFileName = dir + "/" + propertyMap.get(PROPERTIES_INDIV_TOUR_FILENAME); + this.jointTourFileName = dir + "/" + propertyMap.get(PROPERTIES_JOINT_TOUR_FILENAME); + this.personFileName = dir + "/" + propertyMap.get(PROPERTIES_PERSON_FILENAME); + this.hhFileName = dir + "/" + propertyMap.get(PROPERTIES_HOUSEHOLD_FILENAME); + } + + public BikeAssignmentTripReader(Map propertyMap, int iter) + { + String dir = propertyMap.get(PROPERTIES_PROJECT_DIR); + this.indivTripFileName = dir + + "/" + + propertyMap.get(PROPERTIES_INDIV_TRIP_FILENAME).substring(0, + PROPERTIES_INDIV_TRIP_FILENAME.length() - 5) + "_" + iter + ".csv"; + this.jointTripFileName = dir + + "/" + + propertyMap.get(PROPERTIES_JOINT_TRIP_FILENAME).substring(0, + PROPERTIES_JOINT_TRIP_FILENAME.length() - 5) + "_" + iter + ".csv"; + this.indivTourFileName = dir + + "/" + + propertyMap.get(PROPERTIES_INDIV_TOUR_FILENAME).substring(0, + PROPERTIES_INDIV_TOUR_FILENAME.length() - 5) + "_" + iter + ".csv"; + this.jointTourFileName = dir + + "/" + + propertyMap.get(PROPERTIES_JOINT_TOUR_FILENAME).substring(0, + PROPERTIES_JOINT_TOUR_FILENAME.length() - 5) + "_" + iter + ".csv"; + this.personFileName = dir + "/" + propertyMap.get(PROPERTIES_PERSON_FILENAME); + this.hhFileName = dir + "/" + propertyMap.get(PROPERTIES_HOUSEHOLD_FILENAME); + } + + public List createTripList() + { + + SandagModelStructure modelStructure = new SandagModelStructure(); + + Map indivTourMap = new HashMap<>(); + Map> jointTourMap = new HashMap<>(); + Map hhMap = new HashMap<>(); + List stops = new ArrayList<>(); + + try + { + + String line; + BufferedReader reader = new BufferedReader(new FileReader(hhFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + Household h = new Household(modelStructure); + int hhSize = Integer.parseInt(row[8]); + h.setHhSize(hhSize); + int hhId = Integer.parseInt(row[0]); + hhMap.put(hhId, h); + } + reader.close(); + + reader = new BufferedReader(new FileReader(personFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + int hhId = Integer.parseInt(row[0]); + int perId = Integer.parseInt(row[1]); + int perNo = Integer.parseInt(row[3]); + Household h = hhMap.get(hhId); + Person p = h.getPerson(perNo); + int gender = Integer.parseInt(row[5]); + p.setPersGender(gender); + p.setPersId(perId); + } + reader.close(); + + reader = new BufferedReader(new FileReader(indivTourFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + int hhId = Integer.parseInt(row[0]); + int perId = Integer.parseInt(row[1]); + int perNo = Integer.parseInt(row[2]); + int tourId = Integer.parseInt(row[4]); + String tourCategory = row[5]; + int tourPurpose = DEFAULT_MANDATORY_PURPOSE_INDEX; + if (tourCategory != "MANDATORY") + { + tourPurpose = DEFAULT_NONMANDATORY_PURPOSE_INDEX; + } + Tour t = new Tour(hhMap.get(hhId).getPerson(perNo), tourId, tourPurpose); + if (!indivTourMap.containsKey(perId)) + { + indivTourMap.put(perId, new Tour[20]); + } + indivTourMap.get(perId)[tourId] = t; + } + reader.close(); + + reader = new BufferedReader(new FileReader(jointTourFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + int hhId = Integer.parseInt(row[0]); + int tourId = Integer.parseInt(row[1]); + String[] tourParty = row[5].split(" "); + Household h = hhMap.get(hhId); + int[] participantArray = new int[tourParty.length]; + for (int i = 0; i < participantArray.length; i++) + { + participantArray[i] = Integer.parseInt(tourParty[i]); + } + Tour t = new Tour(h, "", modelStructure.JOINT_NON_MANDATORY_CATEGORY, + DEFAULT_NONMANDATORY_PURPOSE_INDEX); + t.setPersonObject(h.getPerson(participantArray[0])); + t.setPersonNumArray(participantArray); + t.setTourId(tourId); + if (!jointTourMap.containsKey(hhId)) + { + jointTourMap.put(hhId, new ArrayList()); + } + jointTourMap.get(hhId).add(t); + } + reader.close(); + + reader = new BufferedReader(new FileReader(indivTripFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + int tripMode = Integer.parseInt(row[13]); + if (tripMode == BIKE_MODE_INDEX) + { + int perId = Integer.parseInt(row[1]); + int tourId = Integer.parseInt(row[3]); + boolean inbound = Boolean.parseBoolean(row[5]); + Tour t = indivTourMap.get(perId)[tourId]; + Stop s = new Stop(t, "", "", 0, inbound, 0); + int stopPeriod = Integer.parseInt(row[12]); + s.setStopPeriod(stopPeriod); + int oMgra = Integer.parseInt(row[9]); + int dMgra = Integer.parseInt(row[10]); + s.setOrig(oMgra); + s.setDest(dMgra); + stops.add(s); + } + } + reader.close(); + + reader = new BufferedReader(new FileReader(jointTripFileName)); + reader.readLine(); + while ((line = reader.readLine()) != null) + { + String[] row = line.split(","); + int tripMode = Integer.parseInt(row[11]); + if (tripMode == BIKE_MODE_INDEX) + { + int hhId = Integer.parseInt(row[0]); + int tourId = Integer.parseInt(row[1]); + boolean inbound = Boolean.parseBoolean(row[3]); + Tour t = jointTourMap.get(hhId).get(tourId); + Stop s = new Stop(t, "", "", 0, inbound, 0); + int stopPeriod = Integer.parseInt(row[10]); + s.setStopPeriod(stopPeriod); + int oMgra = Integer.parseInt(row[7]); + int dMgra = Integer.parseInt(row[8]); + s.setOrig(oMgra); + s.setDest(dMgra); + stops.add(s); + } + } + reader.close(); + + } catch (IOException e) + { + throw new RuntimeException(e); + } + + return stops; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java new file mode 100644 index 0000000..1ffdbbb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java @@ -0,0 +1,99 @@ +package org.sandag.abm.active.sandag; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +public class PropertyParser +{ + public Map propertyMap; + + public PropertyParser(Map propertyMap) + { + this.propertyMap = propertyMap; + } + + public List parseStringPropertyList(String property) + { + return Arrays.asList(propertyMap.get(property.trim()).split("\\s*,\\s*")); + } + + public List parseFloatPropertyList(String property) + { + List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); + List floatList = new ArrayList(); + for (String str : stringList) + { + floatList.add(Float.parseFloat(str)); + } + return floatList; + } + + public double[] parseDoublePropertyArray(String property) + { + List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); + double[] array = new double[stringList.size()]; + for (int i = 0; i < stringList.size(); i++) + { + array[i] = Double.parseDouble(stringList.get(i)); + } + return array; + } + + public List parseIntPropertyList(String property) + { + List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); + List intList = new ArrayList(); + for (String str : stringList) + { + intList.add(Integer.parseInt(str)); + } + return intList; + } + + public Map mapStringPropertyListToStrings(String keyProperty, + String stringValueProperty) + { + Map map = new HashMap(); + List keys = parseStringPropertyList(keyProperty); + List values = parseStringPropertyList(stringValueProperty); + for (int i = 0; i < keys.size(); i += 1) + { + map.put(keys.get(i), values.get(i)); + } + return map; + } + + public Map mapStringPropertyListToFloats(String keyProperty, + String stringValueProperty) + { + Map map = new HashMap(); + List keys = parseStringPropertyList(keyProperty); + List values = parseFloatPropertyList(stringValueProperty); + for (int i = 0; i < keys.size(); i += 1) + { + map.put(keys.get(i), values.get(i)); + } + return map; + } + + private boolean isIntValueIn(int value, List referenceValues) + { + for (int v : referenceValues) + { + if (v == value) + { + return true; + } + } + return false; + } + + public boolean isIntValueInPropertyList(int value, String property) + { + List referenceValues = parseIntPropertyList(property); + return isIntValueIn(value, referenceValues); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java new file mode 100644 index 0000000..7b5a52f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java @@ -0,0 +1,19 @@ +package org.sandag.abm.active.sandag; + +import org.sandag.abm.active.SimpleEdge; + +public class SandagBikeEdge + extends SimpleEdge +{ + public volatile byte bikeClass, lanes, functionalClass; + public volatile boolean centroidConnector, autosPermitted, cycleTrack, bikeBlvd; + public volatile float distance, scenicIndex; + public volatile short gain; + public volatile double bikeCost, walkCost; + public long roadsegid; + + public SandagBikeEdge(SandagBikeNode fromNode, SandagBikeNode toNode) + { + super(fromNode, toNode); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..8280f42 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,36 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import org.sandag.abm.active.Network; + +public class SandagBikeMgraPathAlternativeListGenerationConfiguration + extends SandagBikePathAlternativeListGenerationConfiguration +{ + + public SandagBikeMgraPathAlternativeListGenerationConfiguration( + Map propertyMap, + Network network) + { + super(propertyMap, network); + this.PROPERTIES_MAXDIST_ZONE = Double.parseDouble(propertyMap.get("active.maxdist.bike.mgra")); + this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; + } + + protected void createZonalCentroidIdMap() + { + System.out.println("Creating MGRA Zonal Centroid Id Map..."); + zonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.mgra > 0) + { + zonalCentroidIdMap.put((int) n.mgra, n.getId()); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java new file mode 100644 index 0000000..e5aebca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java @@ -0,0 +1,581 @@ +package org.sandag.abm.active.sandag; + +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.lang.reflect.Field; +import java.util.Collection; +import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedHashSet; +import java.util.Map; +import java.util.Set; +import org.apache.log4j.Logger; +import org.sandag.abm.active.AbstractNetworkFactory; +import org.sandag.abm.active.Network; +import com.linuxense.javadbf.DBFReader; + +public class SandagBikeNetworkFactory + extends AbstractNetworkFactory +{ + protected Logger logger = Logger.getLogger(SandagBikeNetworkFactory.class); + private Map propertyMap; + private PropertyParser propertyParser; + private Collection nodes = null; + private Collection edges = null; + private Collection traversals = null; + + private static final String PROPERTIES_NODE_FILE = "active.node.file"; + private static final String PROPERTIES_NODE_ID = "active.node.id"; + private static final String PROPERTIES_NODE_FIELDNAMES = "active.node.fieldnames"; + private static final String PROPERTIES_NODE_COLUMNS = "active.node.columns"; + private static final String PROPERTIES_EDGE_FILE = "active.edge.file"; + private static final String PROPERTIES_EDGE_ANODE = "active.edge.anode"; + private static final String PROPERTIES_EDGE_BNODE = "active.edge.bnode"; + private static final String PROPERTIES_EDGE_DIRECTIONAL = "active.edge.directional"; + private static final String PROPERTIES_EDGE_FIELDNAMES = "active.edge.fieldnames"; + private static final String PROPERTIES_EDGE_COLUMNS_AB = "active.edge.columns.ab"; + private static final String PROPERTIES_EDGE_COLUMNS_BA = "active.edge.columns.ba"; + private static final String PROPERTIES_EDGE_CENTROID_FIELD = "active.edge.centroid.field"; + private static final String PROPERTIES_EDGE_CENTROID_VALUE = "active.edge.centroid.value"; + private static final String PROPERTIES_EDGE_AUTOSPERMITTED_FIELD = "active.edge.autospermitted.field"; + private static final String PROPERTIES_EDGE_AUTOSPERMITTED_VALUES = "active.edge.autospermitted.values"; + + private static final double TURN_ANGLE_TOLERANCE = Math.PI / 6; + private static final double DISTANCE_CONVERSION_FACTOR = 0.000189; + private static final double INACCESSIBLE_COST_COEF = 999.0; + + private static final String PROPERTIES_COEF_DISTCLA0 = "active.coef.distcla0"; + private static final String PROPERTIES_COEF_DISTCLA1 = "active.coef.distcla1"; + private static final String PROPERTIES_COEF_DISTCLA2 = "active.coef.distcla2"; + private static final String PROPERTIES_COEF_DISTCLA3 = "active.coef.distcla3"; + private static final String PROPERTIES_COEF_DARTNE2 = "active.coef.dartne2"; + private static final String PROPERTIES_COEF_DWRONGWY = "active.coef.dwrongwy"; + private static final String PROPERTIES_COEF_GAIN = "active.coef.gain"; + private static final String PROPERTIES_COEF_TURN = "active.coef.turn"; + private static final String PROPERTIES_COEF_GAIN_WALK = "active.coef.gain.walk"; + private static final String PROPERTIES_COEF_DCYCTRAC = "active.coef.dcyctrac"; + private static final String PROPERTIES_COEF_DBIKBLVD = "active.coef.dbikblvd"; + private static final String PROPERTIES_COEF_SIGNALS = "active.coef.signals"; + private static final String PROPERTIES_COEF_UNLFRMA = "active.coef.unlfrma"; + private static final String PROPERTIES_COEF_UNLFRMI = "active.coef.unlfrmi"; + private static final String PROPERTIES_COEF_UNTOMA = "active.coef.untoma"; + private static final String PROPERTIES_COEF_UNTOMI = "active.coef.untomi"; + private static final String PROPERTIES_COEF_DISTANCE_WALK = "active.walk.minutes.per.mile"; + + + public SandagBikeNetworkFactory(Map propertyMap) + { + this.propertyMap = propertyMap; + propertyParser = new PropertyParser(propertyMap); + } + + protected Collection readNodes() + { + Set nodes = new LinkedHashSet<>(); + try + { + InputStream stream = new FileInputStream(propertyMap.get(PROPERTIES_NODE_FILE)); + DBFReader reader = new DBFReader(stream); + Map fieldMap = propertyParser.mapStringPropertyListToStrings( + PROPERTIES_NODE_FIELDNAMES, PROPERTIES_NODE_COLUMNS); + Field f; + int fieldCount = reader.getFieldCount(); + Map labels = new HashMap(); + for (int i = 0; i < fieldCount; i++) + { + labels.put(reader.getField(i).getName(), i); + } + Object[] rowObjects; + while ((rowObjects = reader.nextRecord()) != null) + { + int id = ((Number) rowObjects[labels.get(propertyMap.get(PROPERTIES_NODE_ID))]) + .intValue(); + SandagBikeNode node = new SandagBikeNode(id); + for (String fieldName : fieldMap.keySet()) + { + try + { + f = node.getClass().getField(fieldName); + setNumericFieldWithCast(node, f, + (Number) rowObjects[labels.get(fieldMap.get(fieldName))]); + } catch (NoSuchFieldException | SecurityException e) + { + logger.error("Exception caught getting class field " + fieldName + + " for object of class " + node.getClass().getName(), e); + throw new RuntimeException(); + } + } + nodes.add(node); + } + } catch (IOException e) + { + logger.error("Exception caught reading nodes from disk.", e); + throw new RuntimeException(); + } + return nodes; + } + + protected Collection readEdges(Collection nodes) + { + Set edges = new LinkedHashSet<>(); + Map idNodeMap = new HashMap<>(); + for (SandagBikeNode node : nodes) + idNodeMap.put(node.getId(), node); + + try + { + InputStream stream = new FileInputStream(propertyMap.get(PROPERTIES_EDGE_FILE)); + DBFReader reader = new DBFReader(stream); + Map abFieldMap = propertyParser.mapStringPropertyListToStrings( + PROPERTIES_EDGE_FIELDNAMES, PROPERTIES_EDGE_COLUMNS_AB); + Map baFieldMap = new HashMap(); + boolean directional = Boolean + .parseBoolean(propertyMap.get(PROPERTIES_EDGE_DIRECTIONAL)); + if (!directional) + { + baFieldMap = propertyParser.mapStringPropertyListToStrings( + PROPERTIES_EDGE_FIELDNAMES, PROPERTIES_EDGE_COLUMNS_BA); + } + int columnCount = reader.getFieldCount(); + Map labels = new HashMap(); + for (int i = 0; i < columnCount; i++) + { + labels.put(reader.getField(i).getName(), i); + } + Object[] rowObjects; + while ((rowObjects = reader.nextRecord()) != null) + { + SandagBikeNode a = idNodeMap.get(((Number) rowObjects[labels.get(propertyMap + .get(PROPERTIES_EDGE_ANODE))]).intValue()); + SandagBikeNode b = idNodeMap.get(((Number) rowObjects[labels.get(propertyMap + .get(PROPERTIES_EDGE_BNODE))]).intValue()); + + SandagBikeEdge edge = new SandagBikeEdge(a, b); + for (String fieldName : abFieldMap.keySet()) + { + try + { + Field f = edge.getClass().getField(fieldName); + setNumericFieldWithCast(edge, f, + (Number) rowObjects[labels.get(abFieldMap.get(fieldName))]); + } catch (NoSuchFieldException | SecurityException e) + { + logger.error("Exception caught getting class field " + fieldName + + " for object of class " + edge.getClass().getName(), e); + throw new RuntimeException(); + } + } + edges.add(edge); + + if (!directional) + { + edge = new SandagBikeEdge(b, a); + for (String fieldName : baFieldMap.keySet()) + { + try + { + Field f = edge.getClass().getField(fieldName); + setNumericFieldWithCast(edge, f, + (Number) rowObjects[labels.get(baFieldMap.get(fieldName))]); + } catch (NoSuchFieldException | SecurityException e) + { + logger.error("Exception caught getting class field " + fieldName + + " for object of class " + edge.getClass().getName(), e); + throw new RuntimeException(); + } + } + edges.add(edge); + } + } + } catch (IOException e) + { + logger.error("Exception caught reading edges from disk.", e); + throw new RuntimeException(); + } + return edges; + } + + @Override + protected void calculateDerivedNodeAttributes( + Network network) + { + Iterator nodeIterator = network.nodeIterator(); + while (nodeIterator.hasNext()) + { + SandagBikeNode n = nodeIterator.next(); + n.centroid = (n.mgra > 0) || (n.taz > 0); + if (n.mgra > 0) + { + n.taz = 0; + } + } + } + + @Override + protected void calculateDerivedEdgeAttributes( + Network network) + { + try + { + Iterator edgeIterator = network.edgeIterator(); + Field apf = SandagBikeEdge.class.getField(propertyMap + .get(PROPERTIES_EDGE_AUTOSPERMITTED_FIELD)); + Field cf = SandagBikeEdge.class.getField(propertyMap + .get(PROPERTIES_EDGE_CENTROID_FIELD)); + while (edgeIterator.hasNext()) + { + SandagBikeEdge edge = edgeIterator.next(); + edge.autosPermitted = propertyParser.isIntValueInPropertyList(apf.getInt(edge), + PROPERTIES_EDGE_AUTOSPERMITTED_VALUES); + edge.centroidConnector = propertyParser.isIntValueInPropertyList(cf.getInt(edge), + PROPERTIES_EDGE_CENTROID_VALUE); + edge.distance = edge.distance * (float) DISTANCE_CONVERSION_FACTOR; + edge.bikeCost = (double) edge.distance + * (Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA0)) + * ((edge.bikeClass < 1 ? 1 : 0) + (edge.bikeClass > 3 ? 1 : 0)) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA1)) + * (edge.bikeClass == 1 ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA2)) + * (edge.bikeClass == 2 ? 1 : 0) * (edge.cycleTrack ? 0 : 1) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA3)) + * (edge.bikeClass == 3 ? 1 : 0) * (edge.bikeBlvd ? 0 : 1) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DARTNE2)) + * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) + * ((edge.functionalClass < 4 && edge.functionalClass > 0) ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DWRONGWY)) + * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DCYCTRAC)) + * (edge.cycleTrack ? 1 : 0) + Double.parseDouble(propertyMap + .get(PROPERTIES_COEF_DBIKBLVD)) * (edge.bikeBlvd ? 1 : 0)) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN)) * edge.gain + + INACCESSIBLE_COST_COEF + * ((edge.functionalClass < 3 && edge.functionalClass > 0) ? 1 : 0); + edge.walkCost = (double) edge.distance + * Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTANCE_WALK)) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN_WALK)) + * edge.gain; + } + + } catch (NoSuchFieldException | IllegalAccessException e) + { + logger.error("Exception caught calculating derived edge attributes.", e); + throw new RuntimeException(); + } + } + + @Override + protected void calculateDerivedTraversalAttributes( + Network network) + { + Iterator traversalIterator = network.traversalIterator(); + while (traversalIterator.hasNext()) + { + SandagBikeTraversal t = traversalIterator.next(); + t.turnType = calculateTurnType(t, network); + t.thruCentroid = t.getFromEdge().centroidConnector && t.getToEdge().centroidConnector; + boolean signalized = t.getFromEdge().getToNode().signalized; + boolean fromMajorArt = t.getFromEdge().functionalClass <= 3 + && t.getFromEdge().functionalClass > 0 && t.getFromEdge().bikeClass != 1; + boolean fromMinorArt = t.getFromEdge().functionalClass == 4 + && t.getFromEdge().bikeClass != 1; + t.signalExclRightAndThruJunction = signalized && t.turnType != TurnType.RIGHT + && !isThruJunction(t, network); + t.unsigLeftFromMajorArt = !signalized && fromMajorArt && t.turnType == TurnType.LEFT; + t.unsigLeftFromMinorArt = !signalized && fromMinorArt && t.turnType == TurnType.LEFT; + t.unsigCrossMajorArt = !signalized && isCrossingOfMajorArterial(t, network); + t.unsigCrossMinorArt = !signalized && isCrossingOfMinorArterial(t, network); + t.cost = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_TURN)) + * ((t.turnType != TurnType.NONE) ? 1 : 0) + INACCESSIBLE_COST_COEF + * (t.thruCentroid ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_SIGNALS)) + * (t.signalExclRightAndThruJunction ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNLFRMA)) + * (t.unsigLeftFromMajorArt ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNLFRMI)) + * (t.unsigLeftFromMinorArt ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNTOMA)) + * (t.unsigCrossMajorArt ? 1 : 0) + + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNTOMI)) + * (t.unsigCrossMinorArt ? 1 : 0); + } + } + + private boolean isCrossingOfMajorArterial(SandagBikeTraversal t, + Network network) + { + SandagBikeNode startNode = t.getFromEdge().getFromNode(); + SandagBikeNode thruNode = t.getFromEdge().getToNode(); + SandagBikeNode endNode = t.getToEdge().getToNode(); + + if (startNode.centroid || thruNode.centroid || endNode.centroid) + { + return false; + } + + SandagBikeEdge edge = t.getToEdge(); + if (t.turnType == TurnType.LEFT && edge.functionalClass <= 3 && edge.functionalClass > 0 + && edge.bikeClass != 1) + { + return true; + } + + if (t.turnType != TurnType.NONE) + { + return false; + } + + int majorArtCount = 0; + for (SandagBikeNode successor : network.getSuccessors(thruNode)) + { + edge = network.getEdge(thruNode, successor); + boolean majorArt = edge.functionalClass <= 3 && edge.functionalClass > 0 + && edge.bikeClass != 1; + if (majorArt && (!(successor.equals(startNode))) && (!(successor.equals(endNode)))) + { + majorArtCount += 1; + } + } + + return majorArtCount >= 2; + } + + private boolean isCrossingOfMinorArterial(SandagBikeTraversal t, + Network network) + { + if (isCrossingOfMajorArterial(t, network)) + { + return false; + } + + SandagBikeNode startNode = t.getFromEdge().getFromNode(); + SandagBikeNode thruNode = t.getFromEdge().getToNode(); + SandagBikeNode endNode = t.getToEdge().getToNode(); + + if (startNode.centroid || thruNode.centroid || endNode.centroid) + { + return false; + } + + SandagBikeEdge edge = t.getToEdge(); + if (t.turnType == TurnType.LEFT && edge.functionalClass == 4 && edge.bikeClass != 1) + { + return true; + } + + if (t.turnType != TurnType.NONE) + { + return false; + } + + int artCount = 0; + for (SandagBikeNode successor : network.getSuccessors(thruNode)) + { + edge = network.getEdge(thruNode, successor); + boolean art = edge.functionalClass <= 4 && edge.functionalClass > 0 + && edge.bikeClass != 1; + if (art && (!(successor.equals(startNode))) && (!(successor.equals(endNode)))) + { + artCount += 1; + } + } + + return artCount >= 2; + } + + private boolean isThruJunction(SandagBikeTraversal t, + Network network) + { + SandagBikeNode startNode = t.getFromEdge().getFromNode(); + SandagBikeNode thruNode = t.getFromEdge().getToNode(); + SandagBikeNode endNode = t.getToEdge().getToNode(); + + if (startNode.centroid || thruNode.centroid || endNode.centroid) + { + return false; + } + + if (t.turnType != TurnType.NONE) + { + return false; + } + + boolean rightTurnExists = false; + for (SandagBikeNode successor : network.getSuccessors(thruNode)) + { + SandagBikeTraversal traversal = network.getTraversal(t.getFromEdge(), + network.getEdge(thruNode, successor)); + if ((!(successor.equals(startNode))) && (!(successor.equals(endNode)))) + { + rightTurnExists = traversal.turnType == TurnType.NONE; + } + } + + return !rightTurnExists; + } + + private double calculateTraversalAngle(SandagBikeTraversal t) + { + float xDiff1 = t.getFromEdge().getToNode().x - t.getFromEdge().getFromNode().x; + float xDiff2 = t.getToEdge().getToNode().x - t.getToEdge().getFromNode().x; + float yDiff1 = t.getFromEdge().getToNode().y - t.getFromEdge().getFromNode().y; + float yDiff2 = t.getToEdge().getToNode().y - t.getToEdge().getFromNode().y; + + double angle = Math.atan2(yDiff2, xDiff2) - Math.atan2(yDiff1, xDiff1); + + if (angle > Math.PI) + { + angle = angle - 2 * Math.PI; + } + if (angle < -Math.PI) + { + angle = angle + 2 * Math.PI; + } + + return angle; + } + + private TurnType calculateTurnType(SandagBikeTraversal t, + Network network) + { + SandagBikeNode startNode = t.getFromEdge().getFromNode(); + SandagBikeNode thruNode = t.getFromEdge().getToNode(); + SandagBikeNode endNode = t.getToEdge().getToNode(); + + if (startNode.centroid || thruNode.centroid || endNode.centroid) + { + return TurnType.NONE; + } + + if (startNode.equals(endNode)) + { + return TurnType.REVERSAL; + } + + double thisAngle = calculateTraversalAngle(t); + + if (thisAngle < -Math.PI + TURN_ANGLE_TOLERANCE + || thisAngle > Math.PI - TURN_ANGLE_TOLERANCE) + { + return TurnType.REVERSAL; + } + + double minAngle = Math.PI; + double maxAngle = -Math.PI; + double minAbsAngle = Math.PI; + double currentAngle; + int legCount = 1; + + SandagBikeEdge startEdge = network.getEdge(startNode, thruNode); + for (SandagBikeNode successor : network.getSuccessors(thruNode)) + { + SandagBikeEdge edge = network.getEdge(thruNode, successor); + if (edge.autosPermitted && (!(successor.equals(startNode)))) + { + currentAngle = calculateTraversalAngle(network.getTraversal(startEdge, edge)); + minAngle = Math.min(minAngle, currentAngle); + maxAngle = Math.max(maxAngle, currentAngle); + minAbsAngle = Math.min(minAbsAngle, Math.abs(currentAngle)); + legCount += 1; + } + } + + if (legCount <= 2) + { + return TurnType.NONE; + } else if (legCount == 3) + { + if (thisAngle <= minAngle && Math.abs(thisAngle) > TURN_ANGLE_TOLERANCE) + { + return TurnType.RIGHT; + } else if (thisAngle >= maxAngle && Math.abs(thisAngle) > TURN_ANGLE_TOLERANCE) + { + return TurnType.LEFT; + } else + { + return TurnType.NONE; + } + } else + { + if (Math.abs(thisAngle) <= minAbsAngle + || (Math.abs(thisAngle) < TURN_ANGLE_TOLERANCE && thisAngle > minAngle && thisAngle < maxAngle)) + { + return TurnType.NONE; + } else if (thisAngle < 0) + { + return TurnType.RIGHT; + } else + { + return TurnType.LEFT; + } + } + } + + private void setNumericFieldWithCast(Object o, Field f, Number n) + { + Class c = f.getType(); + try + { + if (c.equals(Integer.class) || c.equals(Integer.TYPE)) + { + f.set(o, n.intValue()); + } else if (c.equals(Float.class) || c.equals(Float.TYPE)) + { + f.set(o, n.floatValue()); + } else if (c.equals(Double.class) || c.equals(Double.TYPE)) + { + f.set(o, n.doubleValue()); + } else if (c.equals(Boolean.class) || c.equals(Boolean.TYPE)) + { + f.set(o, n.intValue() == 1); + } else if (c.equals(Byte.class) || c.equals(Byte.TYPE)) + { + f.set(o, n.byteValue()); + } else if (c.equals(Short.class) || c.equals(Short.TYPE)) + { + f.set(o, n.shortValue()); + } else if (c.equals(Long.class) || c.equals(Long.TYPE)) + { + f.set(o, n.longValue()); + } else + { + throw new RuntimeException("Field " + f.getName() + " in class " + + o.getClass().getName() + " is not numeric"); + } + } catch (IllegalArgumentException | IllegalAccessException e) + { + logger.error("Exception caught setting class field " + f.getName() + + " for object of class " + o.getClass().getName(), e); + throw new RuntimeException(); + } + } + + private void loadNetworkData() + { + if (nodes == null) + { + nodes = readNodes(); + edges = readEdges(nodes); + } + } + + @Override + protected Collection getNodes() + { + loadNetworkData(); + return nodes; + } + + @Override + protected Collection getEdges() + { + loadNetworkData(); + return edges; + } + + @Override + protected SandagBikeTraversal getTraversal(SandagBikeEdge fromEdge, SandagBikeEdge toEdge) + { + return new SandagBikeTraversal(fromEdge, toEdge); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java new file mode 100644 index 0000000..a25d42d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java @@ -0,0 +1,16 @@ +package org.sandag.abm.active.sandag; + +import org.sandag.abm.active.SimpleNode; + +public class SandagBikeNode + extends SimpleNode +{ + public volatile float x, y; + public volatile short mgra, taz, tap; + public volatile boolean signalized, centroid; + + public SandagBikeNode(int id) + { + super(id); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..d685c6b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java @@ -0,0 +1,430 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Objects; +import java.util.Random; +import java.util.Set; +import org.sandag.abm.active.EdgeEvaluator; +import org.sandag.abm.active.Network; +import org.sandag.abm.active.NodePair; +import org.sandag.abm.active.ParallelSingleSourceDijkstra; +import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; +import org.sandag.abm.active.RepeatedSingleSourceDijkstra; +import org.sandag.abm.active.ShortestPathResultSet; +import org.sandag.abm.active.ShortestPathStrategy; +import org.sandag.abm.active.TraversalEvaluator; + +public abstract class SandagBikePathAlternativeListGenerationConfiguration + implements + PathAlternativeListGenerationConfiguration +{ + + public static final String PROPERTIES_COEF_DISTCLA0 = "active.coef.distcla0"; + public static final String PROPERTIES_COEF_DISTCLA1 = "active.coef.distcla1"; + public static final String PROPERTIES_COEF_DISTCLA2 = "active.coef.distcla2"; + public static final String PROPERTIES_COEF_DISTCLA3 = "active.coef.distcla3"; + public static final String PROPERTIES_COEF_DARTNE2 = "active.coef.dartne2"; + public static final String PROPERTIES_COEF_DWRONGWY = "active.coef.dwrongwy"; + public static final String PROPERTIES_COEF_GAIN = "active.coef.gain"; + public static final String PROPERTIES_COEF_TURN = "active.coef.turn"; + public static final String PROPERTIES_COEF_GAIN_WALK = "active.coef.gain.walk"; + public static final String PROPERTIES_COEF_DCYCTRAC = "active.coef.dcyctrac"; + public static final String PROPERTIES_COEF_DBIKBLVD = "active.coef.dbikblvd"; + public static final String PROPERTIES_COEF_SIGNALS = "active.coef.signals"; + public static final String PROPERTIES_COEF_UNLFRMA = "active.coef.unlfrma"; + public static final String PROPERTIES_COEF_UNLFRMI = "active.coef.unlfrmi"; + public static final String PROPERTIES_COEF_UNTOMA = "active.coef.untoma"; + public static final String PROPERTIES_COEF_UNTOMI = "active.coef.untomi"; + public static final String PROPERTIES_COEF_NONSCENIC = "active.coef.nonscenic"; + public static final String PROPERTIES_BIKE_MINUTES_PER_MILE = "active.bike.minutes.per.mile"; + public static final String PROPERTIES_OUTPUT = "active.output.bike"; + private static final double INACCESSIBLE_COST_COEF = 999.0; + + protected Map propertyMap; + protected PropertyParser propertyParser; + protected final String PROPERTIES_SAMPLE_MAXCOST = "active.sample.maxcost"; + protected final String PROPERTIES_SAMPLE_RANDOM_SEEDED = "active.sample.random.seeded"; + protected final String PROPERTIES_SAMPLE_DISTANCE_BREAKS = "active.sample.distance.breaks"; + protected final String PROPERTIES_SAMPLE_PATHSIZES = "active.sample.pathsizes"; + protected final String PROPERTIES_SAMPLE_COUNT_MIN = "active.sample.count.min"; + protected final String PROPERTIES_SAMPLE_COUNT_MAX = "active.sample.count.max"; + protected final String PROPERTIES_TRACE_EXCLUSIVE = "active.trace.exclusive"; + protected final String PROPERTIES_RANDOM_SCALE_COEF = "active.sample.random.scale.coef"; + protected final String PROPERTIES_RANDOM_SCALE_LINK = "active.sample.random.scale.link"; + + protected final String PROPERTIES_TRACE_OUTPUTASSIGNMENTPATHS = "active.trace.outputassignmentpaths"; + + protected double PROPERTIES_MAXDIST_ZONE; + protected String PROPERTIES_TRACE_ORIGINS; + + protected Map> nearbyZonalDistanceMap; + protected Map zonalCentroidIdMap; + protected Network network; + private final double bikeMinutesPerMile; + + public SandagBikePathAlternativeListGenerationConfiguration(Map propertyMap, + Network network) + { + this.propertyMap = propertyMap; + this.propertyParser = new PropertyParser(propertyMap); + this.nearbyZonalDistanceMap = null; + this.zonalCentroidIdMap = null; + this.network = network; + bikeMinutesPerMile = Double.parseDouble(propertyMap.get(PROPERTIES_BIKE_MINUTES_PER_MILE)); + } + + public Set getTraceOrigins() + { + return propertyMap.containsKey(PROPERTIES_TRACE_ORIGINS) ? new HashSet<>( + propertyParser.parseIntPropertyList(PROPERTIES_TRACE_ORIGINS)) + : new HashSet(); + } + + @Override + public Network getNetwork() + { + return network; + } + + public String getOutputDirectory() + { + return propertyMap.get(PROPERTIES_OUTPUT); + } + + static class SandagBikeDistanceEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.distance; + } + } + + static class SandagBikeAccessibleDistanceEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.distance + (edge.bikeCost > 998 ? 999 : 0); + } + } + + static class ZeroTraversalEvaluator + implements TraversalEvaluator + { + public double evaluate(SandagBikeTraversal traversal) + { + return 999 * (traversal.thruCentroid ? 1 : 0); + } + } + + @Override + public EdgeEvaluator getEdgeLengthEvaluator() + { + return new SandagBikeDistanceEvaluator(); + } + + @Override + public EdgeEvaluator getEdgeCostEvaluator() + { + final class SandagBikeEdgeCostEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.bikeCost; + } + } + + return new SandagBikeEdgeCostEvaluator(); + } + + @Override + public TraversalEvaluator getTraversalCostEvaluator() + { + final class SandagBikeTraversalCostEvaluator + implements TraversalEvaluator + { + public double evaluate(SandagBikeTraversal traversal) + { + return traversal.cost; + } + } + + return new SandagBikeTraversalCostEvaluator(); + } + + @Override + public double getMaxCost() + { + return Double.parseDouble(propertyMap.get(PROPERTIES_SAMPLE_MAXCOST)); + } + + @Override + public double getDefaultMinutesPerMile() + { + return bikeMinutesPerMile; + } + + @Override + public double[] getSampleDistanceBreaks() + { + return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_DISTANCE_BREAKS); + } + + @Override + public double[] getSamplePathSizes() + { + return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_PATHSIZES); + } + + @Override + public double[] getSampleMinCounts() + { + return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_COUNT_MIN); + } + + @Override + public double[] getSampleMaxCounts() + { + return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_COUNT_MAX); + } + + @Override + public boolean isRandomCostSeeded() + { + return Boolean.parseBoolean(propertyMap.get(PROPERTIES_SAMPLE_RANDOM_SEEDED)); + } + + @Override + public Map> getNearbyZonalDistanceMap() + { + if (nearbyZonalDistanceMap == null) + { + nearbyZonalDistanceMap = new HashMap<>(); + ShortestPathStrategy sps = new ParallelSingleSourceDijkstra( + new RepeatedSingleSourceDijkstra( + network, new SandagBikeAccessibleDistanceEvaluator(), + new ZeroTraversalEvaluator()), + ParallelSingleSourceDijkstra.ParallelMethod.QUEUE); + if (zonalCentroidIdMap == null) + { + createZonalCentroidIdMap(); + } + Set originNodes = new HashSet<>(); + Set destinationNodes = new HashSet<>(); + Map inverseOriginZonalCentroidMap = new HashMap<>(); + Map inverseDestinationZonalCentroidMap = new HashMap<>(); + SandagBikeNode n; + Map relevantOriginZonalCentroidIdMap = getOriginZonalCentroidIdMap(); + Map destinationZonalCentroidIdMap = getDestinationZonalCentroidIdMap(); + for (int zone : relevantOriginZonalCentroidIdMap.keySet()) + { + n = network.getNode(zonalCentroidIdMap.get(zone)); + originNodes.add(n); + inverseOriginZonalCentroidMap.put(n, zone); + } + for (int zone : destinationZonalCentroidIdMap.keySet()) + { + n = network.getNode(zonalCentroidIdMap.get(zone)); + destinationNodes.add(n); + inverseDestinationZonalCentroidMap.put(n, zone); + } + System.out.println("Calculating nearby Zonal Distance Map"); + ShortestPathResultSet resultSet = sps.getShortestPaths(originNodes, + destinationNodes, PROPERTIES_MAXDIST_ZONE); + int originZone, destinationZone; + for (NodePair odPair : resultSet) + { + originZone = inverseOriginZonalCentroidMap.get(odPair.getFromNode()); + destinationZone = inverseDestinationZonalCentroidMap.get(odPair.getToNode()); + if (!nearbyZonalDistanceMap.containsKey(originZone)) + { + nearbyZonalDistanceMap.put(originZone, new HashMap()); + } + nearbyZonalDistanceMap.get(originZone).put(destinationZone, + resultSet.getShortestPathResult(odPair).getCost()); + } + } + return nearbyZonalDistanceMap; + } + + @Override + public Map getOriginZonalCentroidIdMap() + { + if (zonalCentroidIdMap == null) + { + createZonalCentroidIdMap(); + } + + if (isTraceExclusive()) + { + Map m = new HashMap<>(); + for (int o : getTraceOrigins()) + { + m.put(o, zonalCentroidIdMap.get(o)); + } + return m; + } else return zonalCentroidIdMap; + } + + public Map getOriginZonalCentroidIdMapNonExclusiveOfTrace() + { + if (zonalCentroidIdMap == null) + { + createZonalCentroidIdMap(); + } + + return zonalCentroidIdMap; + } + + @Override + public Map getDestinationZonalCentroidIdMap() + { + return getOriginZonalCentroidIdMapNonExclusiveOfTrace(); + } + + @Override + public Map getPropertyMap() + { + return propertyMap; + } + + protected abstract void createZonalCentroidIdMap(); + + public Map getInverseOriginZonalCentroidIdMap() + { + HashMap newMap = new HashMap<>(); + Map origMap = getOriginZonalCentroidIdMap(); + for (Integer o : origMap.keySet()) + { + newMap.put(origMap.get(o), o); + } + return newMap; + } + + public Map getInverseDestinationZonalCentroidIdMap() + { + HashMap newMap = new HashMap<>(); + Map origMap = getDestinationZonalCentroidIdMap(); + for (Integer d : origMap.keySet()) + { + newMap.put(origMap.get(d), d); + } + return newMap; + } + + @Override + public boolean isTraceExclusive() + { + return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_EXCLUSIVE)); + } + + private class RandomizedEdgeCostEvaluator + implements EdgeEvaluator + { + long seed; + Random random; + double cDistCla0, cDistCla1, cDistCla2, cDistCla3, cArtNe2, cWrongWay, cCycTrac, cBikeBlvd, + cGain, cNonScenic; + + public RandomizedEdgeCostEvaluator(long seed) + { + this.seed = seed; + + if (isRandomCostSeeded()) + { + random = new Random(seed); + } else + { + random = new Random(); + } + + cDistCla0 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA0)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cDistCla1 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA1)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cDistCla2 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA2)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cDistCla3 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA3)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cArtNe2 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DARTNE2)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cWrongWay = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DWRONGWY)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cCycTrac = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DCYCTRAC)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cBikeBlvd = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DBIKBLVD)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cGain = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + cNonScenic = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_NONSCENIC)) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) + * (2 * random.nextDouble() - 1)); + + } + + public double evaluate(SandagBikeEdge edge) + { + + if (isRandomCostSeeded()) + { + random = new Random(Objects.hash(seed, edge)); + } else + { + random = new Random(); + } + + return (edge.distance + * ((cDistCla0 * ((edge.bikeClass < 1 ? 1 : 0) + (edge.bikeClass > 3 ? 1 : 0)) + + cDistCla1 * (edge.bikeClass == 1 ? 1 : 0) + cDistCla2 + * (edge.bikeClass == 2 ? 1 : 0) * (edge.cycleTrack ? 0 : 1) + cDistCla3 + * (edge.bikeClass == 3 ? 1 : 0) * (edge.bikeBlvd ? 0 : 1) + cArtNe2 + * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) + * ((edge.functionalClass < 5 && edge.functionalClass > 0) ? 1 : 0) + + cWrongWay * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0) + + cCycTrac * (edge.cycleTrack ? 1 : 0) + cBikeBlvd + * (edge.bikeBlvd ? 1 : 0)) + cNonScenic * (1-edge.scenicIndex) + ) + + cGain * edge.gain + ) + * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_LINK)) + * (random.nextBoolean() ? 1 : -1)) + INACCESSIBLE_COST_COEF + * ((edge.functionalClass < 3 && edge.functionalClass > 0) ? 1 : 0); + } + } + + public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed) + { + + if (iter == 1) + { + return getEdgeCostEvaluator(); + } else + { + return new RandomizedEdgeCostEvaluator(seed); + } + + } + + public boolean isIntrazonalsNeeded() + { + return true; + } + + public boolean isAssignmentPathOutputNeeded() + { + return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_OUTPUTASSIGNMENTPATHS)); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java new file mode 100644 index 0000000..d2d0923 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java @@ -0,0 +1,207 @@ +package org.sandag.abm.active.sandag; + +import java.util.ArrayList; +import java.util.List; +import org.sandag.abm.active.Network; +import org.sandag.abm.active.PathAlternativeList; + +public class SandagBikePathAlternatives +{ + private final PathAlternativeList pathAlternativeList; + private List distance, distClass1, + distClass2, distClass3, distArtNoLane, distWrongWay, distCycTrack, distBikeBlvd, distScenic, gain, + turns, signals, unlfrma, unlfrmi, untoma, untomi, netCost; + private Network network; + + @SuppressWarnings("unchecked") + public SandagBikePathAlternatives( + PathAlternativeList pathAlternativeList) + { + this.pathAlternativeList = pathAlternativeList; + this.network = (Network) pathAlternativeList + .getNetwork(); + calcAndStoreAttributes(); + } + + private void calcAndStoreAttributes() + { + distance = new ArrayList<>(); + distClass1 = new ArrayList<>(); + distClass2 = new ArrayList<>(); + distClass3 = new ArrayList<>(); + distArtNoLane = new ArrayList<>(); + distWrongWay = new ArrayList<>(); + distCycTrack = new ArrayList<>(); + distBikeBlvd = new ArrayList<>(); + distScenic = new ArrayList<>(); + gain = new ArrayList<>(); + turns = new ArrayList<>(); + signals = new ArrayList<>(); + unlfrma = new ArrayList<>(); + unlfrmi = new ArrayList<>(); + untoma = new ArrayList<>(); + untomi = new ArrayList<>(); + netCost = new ArrayList<>(); + + for (int i = 0; i < getPathCount(); i++) + { + distance.add(0.0); + distClass1.add(0.0); + distClass2.add(0.0); + distClass3.add(0.0); + distArtNoLane.add(0.0); + distWrongWay.add(0.0); + distCycTrack.add(0.0); + distBikeBlvd.add(0.0); + distScenic.add(0.0); + gain.add(0.0); + turns.add(0.0); + signals.add(0.0); + unlfrma.add(0.0); + unlfrmi.add(0.0); + untoma.add(0.0); + untomi.add(0.0); + netCost.add(0.0); + SandagBikeNode parent = null, grandparent = null; + for (SandagBikeNode current : pathAlternativeList.get(i)) + { + if (parent != null) + { + SandagBikeEdge edge = network.getEdge(parent, current); + distance.set(i, distance.get(i) + edge.distance); + distClass1.set(i, distClass1.get(i) + edge.distance + * (edge.bikeClass == 1 ? 1 : 0)); + distClass2.set(i, distClass2.get(i) + edge.distance + * (edge.bikeClass == 2 ? 1 : 0)); + distClass3.set(i, distClass3.get(i) + edge.distance + * (edge.bikeClass == 3 ? 1 : 0)); + distArtNoLane.set(i, distArtNoLane.get(i) + edge.distance + * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) + * ((edge.functionalClass < 4 && edge.functionalClass > 0) ? 1 : 0)); + distWrongWay.set(i, distWrongWay.get(i) + edge.distance + * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0)); + distCycTrack.set(i, distCycTrack.get(i) + edge.distance + * (edge.cycleTrack ? 1 : 0)); + distBikeBlvd.set(i, distBikeBlvd.get(i) + edge.distance + * (edge.bikeBlvd ? 1 : 0)); + distScenic.set(i, distScenic.get(i) + edge.distance + * edge.scenicIndex); + gain.set(i, gain.get(i) + edge.gain); + netCost.set(i, netCost.get(i) + edge.bikeCost); + if (grandparent != null) + { + SandagBikeEdge fromEdge = network.getEdge(grandparent, parent); + SandagBikeTraversal traversal = network.getTraversal(fromEdge, edge); + turns.set(i, turns.get(i) + (traversal.turnType != TurnType.NONE ? 1 : 0)); + signals.set(i, signals.get(i) + + (traversal.signalExclRightAndThruJunction ? 1 : 0)); + unlfrma.set(i, unlfrma.get(i) + (traversal.unsigLeftFromMajorArt ? 1 : 0)); + unlfrmi.set(i, unlfrmi.get(i) + (traversal.unsigLeftFromMinorArt ? 1 : 0)); + untoma.set(i, untoma.get(i) + (traversal.unsigCrossMajorArt ? 1 : 0)); + untomi.set(i, untomi.get(i) + (traversal.unsigCrossMinorArt ? 1 : 0)); + netCost.set(i, netCost.get(i) + traversal.cost); + } + } + grandparent = parent; + parent = current; + } + } + } + + public double getSizeAlt(int path) + { + return pathAlternativeList.getSizeMeasures().get(path) + / pathAlternativeList.getSizeMeasureTotal(); + } + + public double getDistanceAlt(int path) + { + return distance.get(path); + } + + public double getDistanceClass1Alt(int path) + { + return distClass1.get(path); + } + + public double getDistanceClass2Alt(int path) + { + return distClass2.get(path); + } + + public double getDistanceClass3Alt(int path) + { + return distClass3.get(path); + } + + public double getDistanceArtNoLaneAlt(int path) + { + return distArtNoLane.get(path); + } + + public double getDistanceCycleTrackAlt(int path) + { + return distCycTrack.get(path); + } + + public double getDistanceBikeBlvdAlt(int path) + { + return distBikeBlvd.get(path); + } + + public double getDistanceWrongWayAlt(int path) + { + return distWrongWay.get(path); + } + + public double getDistanceScenicAlt(int path) + { + return distScenic.get(path); + } + + public double getGainAlt(int path) + { + return gain.get(path); + } + + public double getTurnsAlt(int path) + { + return turns.get(path); + } + + public double getSignalsAlt(int path) + { + return signals.get(path); + } + + public double getUnsigLeftFromMajorArtAlt(int path) + { + return unlfrma.get(path); + } + + public double getUnsigLeftFromMinorArtAlt(int path) + { + return unlfrmi.get(path); + } + + public double getUnsigCrossMajorArtAlt(int path) + { + return untoma.get(path); + } + + public double getUnsigCrossMinorArtAlt(int path) + { + return untomi.get(path); + } + + public int getPathCount() + { + return pathAlternativeList.getCount(); + } + + public double getNetworkCostAlt(int path) + { + return netCost.get(path); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java new file mode 100644 index 0000000..ce01a7d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java @@ -0,0 +1,300 @@ +package org.sandag.abm.active.sandag; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Map; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class SandagBikePathChoiceDmu + implements VariableTable, Serializable +{ + + private final transient Logger logger = Logger.getLogger(SandagBikePathChoiceDmu.class); + + protected final Map methodIndexMap; + private final IndexValues dmuIndex; + + private int personIsFemale; + private int isInboundTrip; + private int tourPurpose; + private SandagBikePathAlternatives paths; + private int pathCount = 0; + + public SandagBikePathChoiceDmu() + { + methodIndexMap = new HashMap<>(); + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + } + + // not needed right now + // public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, + // int destIndex, boolean debug) { + // dmuIndex.setHHIndex(hhIndex); + // dmuIndex.setZoneIndex(zoneIndex); + // dmuIndex.setOriginZone(origIndex); + // dmuIndex.setDestZone(destIndex); + // + // dmuIndex.setDebug(false); + // dmuIndex.setDebugLabel(""); + // if (debug) { + // dmuIndex.setDebug(true); + // dmuIndex.setDebugLabel("Debug Path Choice UEC"); + // } + // + // } + + public void setPersonIsFemale(boolean isFemale) + { + personIsFemale = isFemale ? 1 : 0; + } + + public void setIsInboundTrip(boolean isInboundTrip) + { + this.isInboundTrip = isInboundTrip ? 1 : 0; + } + + public void setTourPurpose(int tourPurpose) + { + this.tourPurpose = tourPurpose; + } + + public void setPathAlternatives(SandagBikePathAlternatives paths) + { + this.paths = paths; + pathCount = paths.getPathCount(); + } + + public SandagBikePathAlternatives getPathAlternatives() + { + return paths; + } + + public int getFemale() + { + return personIsFemale; + } + + public int getInbound() + { + return isInboundTrip; + } + + public int getTourPurpose() + { + return tourPurpose; + } + + public double getSizeAlt(int path) + { + return paths.getSizeAlt(path - 1); + } + + public double getDistanceAlt(int path) + { + return paths.getDistanceAlt(path - 1); + } + + public double getDistanceClass1Alt(int path) + { + return paths.getDistanceClass1Alt(path - 1); + } + + public double getDistanceClass2Alt(int path) + { + return paths.getDistanceClass2Alt(path - 1); + } + + public double getDistanceClass3Alt(int path) + { + return paths.getDistanceClass3Alt(path - 1); + } + + public double getDistanceArtNoLaneAlt(int path) + { + return paths.getDistanceArtNoLaneAlt(path - 1); + } + + public double getDistanceCycleTrackAlt(int path) + { + return paths.getDistanceCycleTrackAlt(path - 1); + } + + public double getDistanceBikeBlvdAlt(int path) + { + return paths.getDistanceBikeBlvdAlt(path - 1); + } + + public double getDistanceWrongWayAlt(int path) + { + return paths.getDistanceWrongWayAlt(path - 1); + } + + public double getDistanceScenicAlt(int path) + { + return paths.getDistanceScenicAlt(path - 1); + } + + public double getGainAlt(int path) + { + return paths.getGainAlt(path - 1); + } + + public double getTurnsAlt(int path) + { + return paths.getTurnsAlt(path - 1); + } + + public double getSignalsAlt(int path) + { + return paths.getSignalsAlt(path - 1); + } + + public double getUnsigLeftFromMajorArtAlt(int path) + { + return paths.getUnsigLeftFromMajorArtAlt(path - 1); + } + + public double getUnsigLeftFromMinorArtAlt(int path) + { + return paths.getUnsigLeftFromMinorArtAlt(path - 1); + } + + public double getUnsigCrossMajorArtAlt(int path) + { + return paths.getUnsigCrossMajorArtAlt(path - 1); + } + + public double getUnsigCrossMinorArtAlt(int path) + { + return paths.getUnsigCrossMinorArtAlt(path - 1); + } + + public double getNetworkCostAlt(int path) + { + return paths.getNetworkCostAlt(path - 1); + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + @Override + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + @Override + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + @Override + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + @Override + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + @Override + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + private void setupMethodIndexMap() + { + methodIndexMap.clear(); + + methodIndexMap.put("getSizeAlt", 0); + methodIndexMap.put("getDistanceAlt", 1); + methodIndexMap.put("getDistanceClass1Alt", 2); + methodIndexMap.put("getDistanceClass2Alt", 3); + methodIndexMap.put("getDistanceClass3Alt", 4); + methodIndexMap.put("getDistanceArtNoLaneAlt", 5); + methodIndexMap.put("getDistanceCycleTrackAlt", 6); + methodIndexMap.put("getDistanceBikeBlvdAlt", 7); + methodIndexMap.put("getDistanceWrongWayAlt", 8); + methodIndexMap.put("getGainAlt", 9); + methodIndexMap.put("getTurnsAlt", 10); + + methodIndexMap.put("getFemale", 11); + methodIndexMap.put("getInbound", 12); + methodIndexMap.put("getTourPurpose", 13); + + methodIndexMap.put("getSignalsAlt", 14); + methodIndexMap.put("getUnsigLeftFromMajorArtAlt", 15); + methodIndexMap.put("getUnsigLeftFromMinorArtAlt", 16); + methodIndexMap.put("getUnsigCrossMajorArtAlt", 17); + methodIndexMap.put("getUnsigCrossMinorArtAlt", 18); + methodIndexMap.put("getNetworkCostAlt", 19); + methodIndexMap.put("getDistanceScenicAlt", 20); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + switch (variableIndex) + { + case 0: + return getSizeAlt(arrayIndex); + case 1: + return getDistanceAlt(arrayIndex); + case 2: + return getDistanceClass1Alt(arrayIndex); + case 3: + return getDistanceClass2Alt(arrayIndex); + case 4: + return getDistanceClass3Alt(arrayIndex); + case 5: + return getDistanceArtNoLaneAlt(arrayIndex); + case 6: + return getDistanceCycleTrackAlt(arrayIndex); + case 7: + return getDistanceBikeBlvdAlt(arrayIndex); + case 8: + return getDistanceWrongWayAlt(arrayIndex); + case 9: + return getGainAlt(arrayIndex); + case 10: + return getTurnsAlt(arrayIndex); + case 11: + return getFemale(); + case 12: + return getInbound(); + case 13: + return getTourPurpose(); + case 14: + return getSignalsAlt(arrayIndex); + case 15: + return getUnsigLeftFromMajorArtAlt(arrayIndex); + case 16: + return getUnsigLeftFromMinorArtAlt(arrayIndex); + case 17: + return getUnsigCrossMajorArtAlt(arrayIndex); + case 18: + return getUnsigCrossMinorArtAlt(arrayIndex); + case 19: + return getNetworkCostAlt(arrayIndex); + case 20: + return getDistanceScenicAlt(arrayIndex); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java new file mode 100644 index 0000000..0e41082 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java @@ -0,0 +1,281 @@ +package org.sandag.abm.active.sandag; + +import java.io.IOException; +import java.io.PrintWriter; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.text.DecimalFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Iterator; +import java.util.List; +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.active.AbstractPathChoiceEdgeAssignmentApplication; +import org.sandag.abm.active.Network; +import org.sandag.abm.active.PathAlternativeList; +import org.sandag.abm.ctramp.Stop; +import org.sandag.abm.ctramp.Tour; +import com.pb.common.util.ResourceUtil; + +public class SandagBikePathChoiceEdgeAssignmentApplication + extends + AbstractPathChoiceEdgeAssignmentApplication +{ + private static final Logger logger = Logger.getLogger(SandagBikePathChoiceEdgeAssignmentApplication.class); + private static final String BIKE_ASSIGNMENT_FILE_PROPERTY = "active.assignment.file.bike"; + + List stops; + final static String[] TIME_PERIOD_LABELS = {"EA", "AM", + "MD", "PM", "EV" }; + final static double[] TIME_PERIOD_BREAKS = {3, 9, 22, 29, + 99 }; + + private ThreadLocal model; + private Map storedPathData; + + private class PathData { + PathAlternativeList pal; + int tripNum; + int householdId; + int tourId; + int stopId; + int inbound; + double[] probs; + + public PathData(PathAlternativeList pal, int tripNum, + int householdId, int tourId, int stopId, int inbound, double[] probs) + { + this.pal = pal; + this.tripNum = tripNum; + this.householdId = householdId; + this.tourId = tourId; + this.stopId = stopId; + this.inbound = inbound; + this.probs = probs; + } + } + + SandagBikePathAlternativeListGenerationConfiguration configuration; + + public SandagBikePathChoiceEdgeAssignmentApplication( + SandagBikePathAlternativeListGenerationConfiguration configuration, + List stops, final Map propertyMap) + { + super(configuration); + this.stops = stops; + model = new ThreadLocal() + { + @Override + protected SandagBikePathChoiceModel initialValue() + { + return new SandagBikePathChoiceModel((HashMap) propertyMap); + } + }; + storedPathData = new ConcurrentHashMap<>(); + this.configuration = configuration; + } + + @Override + protected Map assignTrip(int tripNum, + PathAlternativeList alternativeList) + { + Stop stop = stops.get(tripNum); + Tour tour = stop.getTour(); + SandagBikePathAlternatives paths = new SandagBikePathAlternatives(alternativeList); + double[] probs = model.get().getPathProbabilities(tour.getPersonObject(), paths, + stop.isInboundStop(), tour, false); + double numPersons = 1; + if (tour.getPersonNumArray() != null && tour.getPersonNumArray().length > 1) + { + numPersons = tour.getPersonNumArray().length; + } + int periodIdx = findFirstIndexGreaterThan((double) stop.getStopPeriod(), TIME_PERIOD_BREAKS); + + Map volumes = new HashMap<>(); + for (int pathIdx = 0; pathIdx < probs.length; pathIdx++) + { + SandagBikeNode parent = null; + for (SandagBikeNode node : alternativeList.get(pathIdx)) + { + if (parent != null) + { + SandagBikeEdge edge = network.getEdge(parent, node); + double[] values; + if (volumes.containsKey(edge)) + { + values = volumes.get(edge); + } else + { + values = new double[TIME_PERIOD_BREAKS.length]; + Arrays.fill(values, 0.0); + } + values[periodIdx] += probs[pathIdx] * numPersons; + volumes.put(edge, values); + } + parent = node; + } + } + + if ( this.configuration.isAssignmentPathOutputNeeded() ) { + PathData pd = new PathData(alternativeList,tripNum,tour.getHhId(),tour.getTourId(),stop.isInboundStop() ? 1 : 0,stop.getStopId(),probs); + storedPathData.put(tripNum, pd); + } + + return volumes; + } + + @Override + protected SandagBikeNode getOriginNode(int tripId) + { + return network.getNode(configuration.getOriginZonalCentroidIdMap().get( + stops.get(tripId).getOrig())); + } + + @Override + protected SandagBikeNode getDestinationNode(int tripId) + { + return network.getNode(configuration.getDestinationZonalCentroidIdMap().get( + stops.get(tripId).getDest())); + } + + public static void main(String... args) + { + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } + logger.info("loading property file: " + + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() + .toString()); + + if (args.length < 2) + { + logger.error(String.format("no sample rate was specified as an argument.")); + return; + } + double sampleRate = Double.parseDouble(args[1]); + + // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; + @SuppressWarnings("unchecked") + // this is ok - the map will be String->String + Map propertyMap = (Map) ResourceUtil + .getResourceBundleAsHashMap(args[0]); + DecimalFormat formatter = new DecimalFormat("#.###"); + + SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); + Network network = factory + .createNetwork(); + + SandagBikePathAlternativeListGenerationConfiguration configuration = new SandagBikeMgraPathAlternativeListGenerationConfiguration( + propertyMap, network); + + BikeAssignmentTripReader reader; + if (args.length >= 3) + { + reader = new BikeAssignmentTripReader(propertyMap, Integer.parseInt(args[2])); + } else + { + reader = new BikeAssignmentTripReader(propertyMap); + } + + List stops = reader.createTripList(); + + Path outputDirectory = Paths.get(configuration.getOutputDirectory()); + Path outputFile = outputDirectory.resolve(propertyMap.get(BIKE_ASSIGNMENT_FILE_PROPERTY)); + SandagBikePathChoiceEdgeAssignmentApplication application = new SandagBikePathChoiceEdgeAssignmentApplication( + configuration, stops, propertyMap); + + List relevantTripNums = new ArrayList<>(); + for (int i = 0; i < stops.size(); i++) + relevantTripNums.add(i); + + Map volumes = application.assignTrips(relevantTripNums); + + try + { + Files.createDirectories(outputDirectory); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + StringBuilder sb = new StringBuilder("roadsegid,from,to"); + for (String segment : TIME_PERIOD_LABELS) + sb.append(",").append(segment); + writer.println(sb.toString()); + Iterator edgeIterator = network.edgeIterator(); + while (edgeIterator.hasNext()) + { + SandagBikeEdge edge = edgeIterator.next(); + sb = new StringBuilder(); + sb.append(edge.roadsegid).append(",").append(edge.getFromNode().getId()).append(",").append(edge.getToNode().getId()); + if (volumes.containsKey(edge)) + { + for (double value : volumes.get(edge)) + sb.append(",").append(formatter.format(value / sampleRate)); + } else + { + for (int i = 0; i < TIME_PERIOD_BREAKS.length; i++) + sb.append(",").append(formatter.format(0.0)); + } + writer.println(sb.toString()); + } + } catch (IOException e) + { + logger.fatal(e); + throw new RuntimeException(e); + } + + if ( configuration.isAssignmentPathOutputNeeded() ) { + + Path probFile = outputDirectory.resolve("bikeAssignmentDisaggregatePathProbabilities.csv"); + + try (PrintWriter writer = new PrintWriter(probFile.toFile())) + { + writer.println("tripid,hhid,tourid,stopid,inbound,pathid,prob"); + for (PathData pd : application.storedPathData.values()) { + for (int i=0;i +{ + private static final Logger logger = Logger.getLogger(SandagBikePathChoiceLogsumMatrixApplication.class); + + public static final String PROPERTIES_DEBUG_ORIGIN = "active.debug.origin"; + public static final String PROPERTIES_DEBUG_DESTINATION = "active.debug.destination"; + public static final String PROPERTIES_WRITE_DERIVED_BIKE_NETWORK = "active.bike.write.derived.network"; + public static final String PROPERTIES_DERIVED_NETWORK_EDGES_FILE = "active.bike.derived.network.edges"; + public static final String PROPERTIES_DERIVED_NETWORK_NODES_FILE = "active.bike.derived.network.nodes"; + public static final String PROPERTIES_DERIVED_NETWORK_TRAVERSALS_FILE = "active.bike.derived.network.traversals"; + + private static final String[] MARKET_SEGMENT_NAMES = {"logsum"}; + private static final int[] MARKET_SEGMENT_GENDER_VALUES = {1}; + private static final int[] MARKET_SEGMENT_TOUR_PURPOSE_INDICES = {1}; + private static final boolean[] MARKET_SEGMENT_INBOUND_TRIP_VALUES = {false}; + + private ThreadLocal model; + private Person[] persons; + private Tour[] tours; + private final boolean runDebug; + private final int debugOrigin; + private final int debugDestination; + + public SandagBikePathChoiceLogsumMatrixApplication( + PathAlternativeListGenerationConfiguration configuration, + final Map propertyMap) + { + super(configuration); + model = new ThreadLocal() + { + @Override + protected SandagBikePathChoiceModel initialValue() + { + return new SandagBikePathChoiceModel((HashMap) propertyMap); + } + }; + persons = new Person[MARKET_SEGMENT_NAMES.length]; + tours = new Tour[MARKET_SEGMENT_NAMES.length]; + + // for dummy person + SandagModelStructure modelStructure = new SandagModelStructure(); + for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) + { + persons[i] = new Person(null, 1, modelStructure); + persons[i].setPersGender(MARKET_SEGMENT_GENDER_VALUES[i]); + tours[i] = new Tour(persons[i], 1, MARKET_SEGMENT_TOUR_PURPOSE_INDICES[i]); + } + + debugOrigin = propertyMap.containsKey(PROPERTIES_DEBUG_ORIGIN) ? Integer + .parseInt(this.propertyMap.get(PROPERTIES_DEBUG_ORIGIN)) : -1; + debugDestination = propertyMap.containsKey(PROPERTIES_DEBUG_DESTINATION) ? Integer + .parseInt(this.propertyMap.get(PROPERTIES_DEBUG_DESTINATION)) : -1; + runDebug = (debugOrigin > 0) && (debugDestination > 0); + } + + @Override + protected double[] calculateMarketSegmentLogsums( + PathAlternativeList alternativeList) + { + SandagBikePathAlternatives alts = new SandagBikePathAlternatives(alternativeList); + double[] logsums = new double[MARKET_SEGMENT_NAMES.length + 1]; + + boolean debug = runDebug + && (alternativeList.getODPair().getFromNode().getId() == debugOrigin) + && (alternativeList.getODPair().getToNode().getId() == debugDestination); + + for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) + logsums[i] = model.get().getPathLogsums(persons[i], alts, + MARKET_SEGMENT_INBOUND_TRIP_VALUES[i], tours[i], debug); + + double[] probs = model.get().getPathProbabilities(persons[0], alts, false, tours[0], debug); + double avgDist = 0; + for (int i = 0; i < alts.getPathCount(); i++) + avgDist += probs[i] * alts.getDistanceAlt(i); + logsums[logsums.length - 1] = avgDist * configuration.getDefaultMinutesPerMile(); + + return logsums; + } + + @Override + protected List> getMarketSegmentIntrazonalCalculations() + { + List> intrazonalCalculations = new ArrayList<>(); + IntrazonalCalculation logsumIntrazonalCalculation = IntrazonalCalculations + .maxFactorIntrazonalCalculation( + IntrazonalCalculations.positiveNegativeFactorizer(0.5, 0, 2, 0), 1); + // IntrazonalCalculations.maxFactorIntrazonalCalculation(IntrazonalCalculations.simpleFactorizer(1,Math.log(2)),1); + for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) + intrazonalCalculations.add(logsumIntrazonalCalculation); + // do distance + intrazonalCalculations.add(IntrazonalCalculations + .minFactorIntrazonalCalculation( + IntrazonalCalculations.simpleFactorizer(0.5, 0), 1)); + return intrazonalCalculations; + } + + // these methods might be moved to the network classes... + public static void writeDerivedNetworkEdges( + Network network, Path outputFile) + { + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + logger.info("Writing edges with derived attributes to " + outputFile.toString()); + StringBuilder sb = new StringBuilder(); + sb.append("fromNode").append(",").append("toNode").append(",").append("bikeClass") + .append(",").append("lanes").append(",").append("functionalClass").append(",") + .append("centroidConnector").append(",").append("autosPermitted").append(",") + .append("cycleTrack").append(",").append("bikeBlvd").append(",") + .append("distance").append(",").append("gain").append(",").append("bikeCost") + .append(",").append("walkCost"); + writer.println(sb.toString()); + Iterator it = network.edgeIterator(); + while (it.hasNext()) + { + SandagBikeEdge edge = it.next(); + sb = new StringBuilder(); + sb.append(edge.getFromNode().getId()).append(",").append(edge.getToNode().getId()) + .append(",").append(edge.bikeClass).append(",").append(edge.lanes) + .append(",").append(edge.functionalClass).append(",") + .append(edge.centroidConnector).append(",").append(edge.autosPermitted) + .append(",").append(edge.cycleTrack).append(",").append(edge.bikeBlvd) + .append(",").append(edge.distance).append(",").append(edge.gain) + .append(",").append(edge.bikeCost).append(",").append(edge.walkCost); + writer.println(sb.toString()); + } + } catch (IOException e) + { + logger.fatal(e); + throw new RuntimeException(e); + } + } + + public static void writeDerivedNetworkNodes( + Network network, Path outputFile) + { + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + logger.info("Writing nodes with derived attributes to " + outputFile.toString()); + StringBuilder sb = new StringBuilder(); + sb.append("id").append(",").append("x").append(",").append("y").append(",") + .append("mgra").append(",").append("taz").append(",").append("tap").append(",") + .append("signalized").append(",").append("centroid"); + writer.println(sb.toString()); + Iterator it = network.nodeIterator(); + while (it.hasNext()) + { + SandagBikeNode node = it.next(); + sb = new StringBuilder(); + sb.append(node.getId()).append(",").append(node.x).append(",").append(node.y) + .append(",").append(node.mgra).append(",").append(node.taz).append(",") + .append(node.tap).append(",").append(node.signalized).append(",") + .append(node.centroid); + writer.println(sb.toString()); + } + } catch (IOException e) + { + logger.fatal(e); + throw new RuntimeException(e); + } + } + + public static void writeDerivedNetworkTraversals( + Network network, Path outputFile) + { + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + logger.info("Writing traversals with derived attributes to " + outputFile.toString()); + StringBuilder sb = new StringBuilder(); + sb.append("start").append(",").append("thru").append(",").append("end").append(",") + .append("turnType").append(",").append("bikecost").append(",") + .append("thruCentroid").append(",").append("signalExclRight").append(",") + .append("unlfrma").append(",").append("unlfrmi").append(",").append("unxma") + .append(",").append("unxmi"); + writer.println(sb.toString()); + Iterator it = network.traversalIterator(); + while (it.hasNext()) + { + SandagBikeTraversal traversal = it.next(); + sb = new StringBuilder(); + sb.append(traversal.getFromEdge().getFromNode().getId()).append(",") + .append(traversal.getFromEdge().getToNode().getId()).append(",") + .append(traversal.getToEdge().getToNode().getId()).append(",") + .append(traversal.turnType.getKey()).append(",").append(traversal.cost) + .append(",").append(traversal.thruCentroid).append(",") + .append(traversal.signalExclRightAndThruJunction).append(",") + .append(traversal.unsigLeftFromMajorArt).append(",") + .append(traversal.unsigLeftFromMinorArt).append(",") + .append(traversal.unsigCrossMajorArt).append(",") + .append(traversal.unsigCrossMinorArt); + writer.println(sb.toString()); + } + } catch (IOException e) + { + logger.fatal(e); + throw new RuntimeException(e); + } + } + + public static void main(String... args) + { + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } + logger.info("loading property file: " + + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() + .toString()); + + // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; + @SuppressWarnings("unchecked") + // this is ok - the map will be String->String + Map propertyMap = (Map) ResourceUtil + .getResourceBundleAsHashMap(args[0]); + DecimalFormat formatter = new DecimalFormat("#.###"); + + SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); + Network network = factory + .createNetwork(); + + // order matters, taz first, then mgra, so use linked hash map + Map, String> configurationOutputMap = new LinkedHashMap<>(); + configurationOutputMap.put(new SandagBikeTazPathAlternativeListGenerationConfiguration( + propertyMap, network), propertyMap.get(BikeLogsum.BIKE_LOGSUM_TAZ_FILE_PROPERTY)); + configurationOutputMap.put(new SandagBikeMgraPathAlternativeListGenerationConfiguration( + propertyMap, network), propertyMap.get(BikeLogsum.BIKE_LOGSUM_MGRA_FILE_PROPERTY)); + + for (PathAlternativeListGenerationConfiguration configuration : configurationOutputMap + .keySet()) + { + Path outputDirectory = Paths.get(configuration.getOutputDirectory()); + Path outputFile = outputDirectory.resolve(configurationOutputMap.get(configuration)); + SandagBikePathChoiceLogsumMatrixApplication application = new SandagBikePathChoiceLogsumMatrixApplication( + configuration, propertyMap); + + Map origins = configuration.getInverseOriginZonalCentroidIdMap(); + Map dests = configuration.getInverseDestinationZonalCentroidIdMap(); + + try + { + Files.createDirectories(outputDirectory); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + Map, double[]> logsums = application + .calculateMarketSegmentLogsums(); + + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + StringBuilder sb = new StringBuilder("i,j"); + for (String segment : MARKET_SEGMENT_NAMES) + sb.append(",").append(segment); + sb.append(",time"); + writer.println(sb.toString()); + for (NodePair od : logsums.keySet()) + { + sb = new StringBuilder(); + sb.append(origins.get(od.getFromNode().getId())).append(",") + .append(dests.get(od.getToNode().getId())); + for (double value : logsums.get(od)) + sb.append(",").append(formatter.format(value)); + writer.println(sb.toString()); + } + } catch (IOException e) + { + logger.fatal(e); + throw new RuntimeException(e); + } + } + + if (Boolean.parseBoolean(propertyMap.get(PROPERTIES_WRITE_DERIVED_BIKE_NETWORK))) + { + Path outputDirectory = Paths.get(propertyMap + .get(SandagBikePathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT)); + writeDerivedNetworkEdges(network, + outputDirectory.resolve(propertyMap.get(PROPERTIES_DERIVED_NETWORK_EDGES_FILE))); + writeDerivedNetworkNodes(network, + outputDirectory.resolve(propertyMap.get(PROPERTIES_DERIVED_NETWORK_NODES_FILE))); + writeDerivedNetworkTraversals(network, outputDirectory.resolve(propertyMap + .get(PROPERTIES_DERIVED_NETWORK_TRAVERSALS_FILE))); + } + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java new file mode 100644 index 0000000..77932da --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java @@ -0,0 +1,134 @@ +package org.sandag.abm.active.sandag; + +import java.util.Arrays; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Tour; +import com.pb.common.newmodel.ChoiceModelApplication; + +// note this class is not thread safe! - use a ThreadLocal or something to +// isolate it amongst threads +public class SandagBikePathChoiceModel +{ + public static final String PATH_CHOICE_MODEL_UEC_SPREADSHEET_PROPERTY = "path.choice.uec.spreadsheet"; + public static final String PATH_CHOICE_MODEL_UEC_MODEL_SHEET_PROPERTY = "path.choice.uec.model.sheet"; + public static final String PATH_CHOICE_MODEL_UEC_DATA_SHEET_PROPERTY = "path.choice.uec.data.sheet"; + public static final String PATH_CHOICE_MODEL_MAX_PATH_COUNT_PROPERTY = "path.choice.max.path.count"; + + public static final String PATH_CHOICE_ALTS_ID_TOKEN_PROPERTY = "path.alts.id.token"; + public static final String PATH_CHOICE_ALTS_FILE_PROPERTY = "path.alts.file"; + public static final String PATH_CHOICE_ALTS_LINK_FILE_PROPERTY = "path.alts.link.file"; + + private static final Logger logger = Logger.getLogger(SandagBikePathChoiceModel.class); + + private final ThreadLocal model; + private final ThreadLocal dmu; + private final int maxPathCount; + private final ThreadLocal pathAltsAvailable; + private final ThreadLocal pathAltsSample; + + public SandagBikePathChoiceModel(final HashMap propertyMap) + { + final String uecSpreadsheet = propertyMap.get(PATH_CHOICE_MODEL_UEC_SPREADSHEET_PROPERTY); + final int modelSheet = Integer.parseInt(propertyMap + .get(PATH_CHOICE_MODEL_UEC_MODEL_SHEET_PROPERTY)); + final int dataSheet = Integer.parseInt(propertyMap + .get(PATH_CHOICE_MODEL_UEC_DATA_SHEET_PROPERTY)); + + dmu = new ThreadLocal() + { + @Override + protected SandagBikePathChoiceDmu initialValue() + { + return new SandagBikePathChoiceDmu(); + } + }; + model = new ThreadLocal() + { + @Override + protected ChoiceModelApplication initialValue() + { + return new ChoiceModelApplication(uecSpreadsheet, modelSheet, dataSheet, + propertyMap, dmu.get()); + } + }; + + maxPathCount = Integer.parseInt(propertyMap.get(PATH_CHOICE_MODEL_MAX_PATH_COUNT_PROPERTY)); + pathAltsAvailable = new ThreadLocal() + { + @Override + protected boolean[] initialValue() + { + return new boolean[maxPathCount + 1]; + } + }; + pathAltsSample = new ThreadLocal() + { + @Override + protected int[] initialValue() + { + return new int[maxPathCount + 1]; + } + }; + } + + public double[] getPathProbabilities(Person person, SandagBikePathAlternatives paths, + boolean inboundTrip, Tour tour, boolean debug) + { + applyPathChoiceModel(person, paths, inboundTrip, tour); + double[] probabilities = Arrays + .copyOf(model.get().getProbabilities(), paths.getPathCount()); + if (debug) debugPathChoiceModel(person, paths, inboundTrip, tour, probabilities); + return probabilities; + + } + + public double getPathLogsums(Person person, SandagBikePathAlternatives paths, + boolean inboundTrip, Tour tour, boolean debug) + { + applyPathChoiceModel(person, paths, inboundTrip, tour); + if (debug) + { + model.get().logUECResults(logger); + logger.info("logsum: " + model.get().getLogsum()); + } + return model.get().getLogsum(); + } + + private void applyPathChoiceModel(Person person, SandagBikePathAlternatives paths, + boolean inboundTrip, Tour tour) + { + SandagBikePathChoiceDmu dmu = this.dmu.get(); + dmu.setPersonIsFemale(person.getPersonIsFemale() == 1); + dmu.setTourPurpose(tour.getTourPrimaryPurposeIndex()); + dmu.setIsInboundTrip(inboundTrip); + dmu.setPathAlternatives(paths); + + boolean[] pathAltsAvailable = this.pathAltsAvailable.get(); + int[] pathAltsSample = this.pathAltsSample.get(); + Arrays.fill(pathAltsAvailable, false); + Arrays.fill(pathAltsSample, 0); + + for (int i = 1; i <= paths.getPathCount(); i++) + { + pathAltsAvailable[i] = true; + pathAltsSample[i] = 1; + } + + model.get().computeUtilities(dmu, dmu.getDmuIndexValues(), pathAltsAvailable, + pathAltsSample); + } + + private void debugPathChoiceModel(Person person, SandagBikePathAlternatives paths, + boolean inboundTrip, Tour tour, double[] probabilities) + { + // want: person id, tour id, inbound trip, path count, path + // probabilities + logger.info(String + .format("debug of path choice model for person id %s, tour id %s, inbound trip: %s, path alternative count: %s", + person.getPersonId(), tour.getTourId(), inboundTrip, paths.getPathCount())); + logger.info(" path probabilities: " + Arrays.toString(probabilities)); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..0eaef70 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,35 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import org.sandag.abm.active.Network; + +public class SandagBikeTazPathAlternativeListGenerationConfiguration + extends SandagBikePathAlternativeListGenerationConfiguration +{ + + public SandagBikeTazPathAlternativeListGenerationConfiguration(Map propertyMap, + Network network) + { + super(propertyMap, network); + this.PROPERTIES_MAXDIST_ZONE = Double.parseDouble(propertyMap.get("active.maxdist.bike.taz")); + this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.taz"; + } + + protected void createZonalCentroidIdMap() + { + zonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.taz > 0) + { + zonalCentroidIdMap.put((int) n.taz, n.getId()); + } + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java new file mode 100644 index 0000000..d29c38b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java @@ -0,0 +1,22 @@ +package org.sandag.abm.active.sandag; + +import org.sandag.abm.active.SimpleTraversal; + +public class SandagBikeTraversal + extends SimpleTraversal +{ + public volatile TurnType turnType; + public volatile double cost; + public volatile boolean thruCentroid, signalExclRightAndThruJunction, unsigLeftFromMajorArt, + unsigLeftFromMinorArt, unsigCrossMajorArt, unsigCrossMinorArt; + + public SandagBikeTraversal(SandagBikeEdge fromEdge, SandagBikeEdge toEdge) + { + super(fromEdge, toEdge); + } + + public SandagBikeTraversal(SandagBikeEdge edge) + { + super(null, edge); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..a9c7992 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,62 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import org.sandag.abm.active.Network; + +public class SandagWalkMgraMgraPathAlternativeListGenerationConfiguration + extends SandagWalkPathAlternativeListGenerationConfiguration +{ + + public SandagWalkMgraMgraPathAlternativeListGenerationConfiguration( + Map propertyMap, + Network network) + { + super(propertyMap, network); + this.PROPERTIES_MAXDIST_ZONE = Math.max( + Math.max( + Double.parseDouble(propertyMap.get("active.maxdist.walk.mgra")), + Double.parseDouble(propertyMap.get("active.maxdist.micromobility.mgra"))), + Double.parseDouble(propertyMap.get("active.maxdist.microtransit.mgra"))); + this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; + } + + protected void createOriginZonalCentroidIdMap() + { + System.out.println("Creating MGRA Origin Zonal Centroid Id Map..."); + originZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.mgra > 0) + { + originZonalCentroidIdMap.put((int) n.mgra, n.getId()); + } + } + } + + protected void createDestinationZonalCentroidIdMap() + { + System.out.println("Creating MGRA Destination Zonal Centroid Id Map..."); + destinationZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.mgra > 0) + { + destinationZonalCentroidIdMap.put((int) n.mgra, n.getId()); + } + } + } + + public boolean isIntrazonalsNeeded() + { + return true; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..ee27177 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,62 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import org.sandag.abm.active.Network; + +public class SandagWalkMgraTapPathAlternativeListGenerationConfiguration + extends SandagWalkPathAlternativeListGenerationConfiguration +{ + + public SandagWalkMgraTapPathAlternativeListGenerationConfiguration( + Map propertyMap, + Network network) + { + super(propertyMap, network); + this.PROPERTIES_MAXDIST_ZONE = Math.max( + Math.max( + Double.parseDouble(propertyMap.get("active.maxdist.walk.tap")), + Double.parseDouble(propertyMap.get("active.maxdist.micromobility.tap"))), + Double.parseDouble(propertyMap.get("active.maxdist.microtransit.tap"))); + this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; + } + + protected void createOriginZonalCentroidIdMap() + { + System.out.println("Creating MGRA Origin Zonal Centroid Id Map..."); + originZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.mgra > 0) + { + originZonalCentroidIdMap.put((int) n.mgra, n.getId()); + } + } + } + + protected void createDestinationZonalCentroidIdMap() + { + System.out.println("Creating TAP Destination Zonal Centroid Id Map..."); + destinationZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.tap > 0) + { + destinationZonalCentroidIdMap.put((int) n.tap, n.getId()); + } + } + } + + public boolean isIntrazonalsNeeded() + { + return false; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..14fa304 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,310 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import org.sandag.abm.active.EdgeEvaluator; +import org.sandag.abm.active.Network; +import org.sandag.abm.active.NodePair; +import org.sandag.abm.active.ParallelSingleSourceDijkstra; +import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; +import org.sandag.abm.active.RepeatedSingleSourceDijkstra; +import org.sandag.abm.active.ShortestPathResultSet; +import org.sandag.abm.active.ShortestPathStrategy; +import org.sandag.abm.active.TraversalEvaluator; + +public abstract class SandagWalkPathAlternativeListGenerationConfiguration + implements + PathAlternativeListGenerationConfiguration +{ + public static final String PROPERTIES_SAMPLE_MAXCOST = "active.sample.maxcost"; + public static final String PROPERTIES_OUTPUT = "active.output.walk"; + public static final String PROPERTIES_TRACE_EXCLUSIVE = "active.trace.exclusive"; + public static final String PROPERTIES_WALK_MINUTES_PER_MILE = "active.walk.minutes.per.mile"; + + protected Map propertyMap; + protected PropertyParser propertyParser; + + protected double PROPERTIES_MAXDIST_ZONE; + protected String PROPERTIES_TRACE_ORIGINS; + + protected Map> nearbyZonalDistanceMap; + protected Map originZonalCentroidIdMap; + protected Map destinationZonalCentroidIdMap; + protected Network network; + private final double walkMinutesPerMile; + + public SandagWalkPathAlternativeListGenerationConfiguration(Map propertyMap, + Network network) + { + this.propertyMap = propertyMap; + this.propertyParser = new PropertyParser(propertyMap); + this.nearbyZonalDistanceMap = null; + this.originZonalCentroidIdMap = null; + this.destinationZonalCentroidIdMap = null; + this.network = network; + walkMinutesPerMile = Double.parseDouble(propertyMap.get(PROPERTIES_WALK_MINUTES_PER_MILE)); + } + + public Set getTraceOrigins() + { + return propertyMap.containsKey(PROPERTIES_TRACE_ORIGINS) ? new HashSet<>( + propertyParser.parseIntPropertyList(PROPERTIES_TRACE_ORIGINS)) + : new HashSet(); + } + + @Override + public Network getNetwork() + { + return network; + } + + public String getOutputDirectory() + { + return propertyMap.get(PROPERTIES_OUTPUT); + } + + static class SandagBikeDistanceEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.distance; + } + } + + static class SandagWalkAccessibleDistanceEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.distance + (edge.walkCost > 998 ? 999 : 0); + } + } + + static class ZeroTraversalEvaluator + implements TraversalEvaluator + { + public double evaluate(SandagBikeTraversal traversal) + { + return 999 * (traversal.thruCentroid ? 1 : 0); + } + } + + @Override + public EdgeEvaluator getEdgeLengthEvaluator() + { + return new SandagBikeDistanceEvaluator(); + } + + @Override + public EdgeEvaluator getEdgeCostEvaluator() + { + final class SandagWalkEdgeCostEvaluator + implements EdgeEvaluator + { + public double evaluate(SandagBikeEdge edge) + { + return edge.walkCost; + } + } + + return new SandagWalkEdgeCostEvaluator(); + } + + @Override + public TraversalEvaluator getTraversalCostEvaluator() + { + return new ZeroTraversalEvaluator(); + } + + @Override + public double getMaxCost() + { + return Double.parseDouble(propertyMap.get(PROPERTIES_SAMPLE_MAXCOST)); + } + + @Override + public double getDefaultMinutesPerMile() + { + return walkMinutesPerMile; + } + + @Override + public double[] getSampleDistanceBreaks() + { + return new double[] {99.0}; + } + + @Override + public double[] getSamplePathSizes() + { + return new double[] {1.0}; + } + + @Override + public double[] getSampleMinCounts() + { + return new double[] {1.0}; + } + + @Override + public double[] getSampleMaxCounts() + { + return new double[] {1.0}; + } + + @Override + public boolean isRandomCostSeeded() + { + return false; + } + + @Override + public Map> getNearbyZonalDistanceMap() + { + if (nearbyZonalDistanceMap == null) + { + nearbyZonalDistanceMap = new HashMap<>(); + ShortestPathStrategy sps = new ParallelSingleSourceDijkstra( + new RepeatedSingleSourceDijkstra( + network, new SandagWalkAccessibleDistanceEvaluator(), + new ZeroTraversalEvaluator()), + ParallelSingleSourceDijkstra.ParallelMethod.QUEUE); + if (originZonalCentroidIdMap == null) + { + createOriginZonalCentroidIdMap(); + } + if (destinationZonalCentroidIdMap == null) + { + createDestinationZonalCentroidIdMap(); + } + Set originNodes = new HashSet<>(); + Set destinationNodes = new HashSet<>(); + Map inverseOriginZonalCentroidIdMap = new HashMap<>(); + Map inverseDestinationZonalCentroidIdMap = new HashMap<>(); + SandagBikeNode n; + Map relevantOriginZonalCentroidIdMap = getOriginZonalCentroidIdMap(); + for (int zone : relevantOriginZonalCentroidIdMap.keySet()) + { + n = network.getNode(originZonalCentroidIdMap.get(zone)); + originNodes.add(n); + inverseOriginZonalCentroidIdMap.put(n, zone); + } + for (int zone : destinationZonalCentroidIdMap.keySet()) + { + n = network.getNode(destinationZonalCentroidIdMap.get(zone)); + destinationNodes.add(n); + inverseDestinationZonalCentroidIdMap.put(n, zone); + } + System.out.println("Calculating nearby Zonal Distance Map"); + ShortestPathResultSet resultSet = sps.getShortestPaths(originNodes, + destinationNodes, PROPERTIES_MAXDIST_ZONE); + int originZone, destinationZone; + for (NodePair odPair : resultSet) + { + originZone = inverseOriginZonalCentroidIdMap.get(odPair.getFromNode()); + destinationZone = inverseDestinationZonalCentroidIdMap.get(odPair.getToNode()); + if (!nearbyZonalDistanceMap.containsKey(originZone)) + { + nearbyZonalDistanceMap.put(originZone, new HashMap()); + } + nearbyZonalDistanceMap.get(originZone).put(destinationZone, + resultSet.getShortestPathResult(odPair).getCost()); + } + } + return nearbyZonalDistanceMap; + } + + @Override + public Map getOriginZonalCentroidIdMap() + { + if (originZonalCentroidIdMap == null) + { + createOriginZonalCentroidIdMap(); + } + + if (isTraceExclusive()) + { + Map m = new HashMap<>(); + for (int o : getTraceOrigins()) + { + m.put(o, originZonalCentroidIdMap.get(o)); + } + return m; + } else return originZonalCentroidIdMap; + } + + public Map getOriginZonalCentroidIdMapNonExclusiveOfTrace() + { + if (originZonalCentroidIdMap == null) + { + createOriginZonalCentroidIdMap(); + } + + return originZonalCentroidIdMap; + } + + @Override + public Map getDestinationZonalCentroidIdMap() + { + if (destinationZonalCentroidIdMap == null) + { + createDestinationZonalCentroidIdMap(); + } + + return destinationZonalCentroidIdMap; + } + + @Override + public Map getPropertyMap() + { + return propertyMap; + } + + protected abstract void createOriginZonalCentroidIdMap(); + + protected abstract void createDestinationZonalCentroidIdMap(); + + public Map getInverseOriginZonalCentroidIdMap() + { + HashMap newMap = new HashMap<>(); + Map origMap = getOriginZonalCentroidIdMap(); + for (Integer o : origMap.keySet()) + { + newMap.put(origMap.get(o), o); + } + return newMap; + } + + public Map getInverseDestinationZonalCentroidIdMap() + { + HashMap newMap = new HashMap<>(); + Map origMap = getDestinationZonalCentroidIdMap(); + for (Integer o : origMap.keySet()) + { + newMap.put(origMap.get(o), o); + } + return newMap; + } + + @Override + public boolean isTraceExclusive() + { + return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_EXCLUSIVE)); + } + + public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed) + { + return getEdgeCostEvaluator(); + } + + @Override + public abstract boolean isIntrazonalsNeeded(); + + public boolean isAssignmentPathOutputNeeded() + { + return false; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java new file mode 100644 index 0000000..442c736 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java @@ -0,0 +1,232 @@ +package org.sandag.abm.active.sandag; + +import java.io.IOException; +import java.io.PrintWriter; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.text.DecimalFormat; +import java.util.Arrays; +import java.util.List; +import java.util.Map; +import java.util.TreeSet; +import org.apache.log4j.Logger; +import org.sandag.abm.active.AbstractPathChoiceLogsumMatrixApplication; +import org.sandag.abm.active.IntrazonalCalculation; +import org.sandag.abm.active.IntrazonalCalculations; +import org.sandag.abm.active.Network; +import org.sandag.abm.active.NodePair; +import org.sandag.abm.active.PathAlternativeList; +import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; +import com.pb.common.util.ResourceUtil; + +public class SandagWalkPathChoiceLogsumMatrixApplication + extends + AbstractPathChoiceLogsumMatrixApplication +{ + private static final Logger logger = Logger.getLogger(SandagWalkPathChoiceLogsumMatrixApplication.class); + + public static final String WALK_LOGSUM_SKIM_MGRA_MGRA_FILE_PROPERTY = "active.logsum.matrix.file.walk.mgra"; + public static final String WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY = "active.logsum.matrix.file.walk.mgratap"; + + private final PathAlternativeListGenerationConfiguration configuration; + + public SandagWalkPathChoiceLogsumMatrixApplication( + PathAlternativeListGenerationConfiguration configuration) + { + super(configuration); + this.configuration = configuration; + } + + @Override + protected double[] calculateMarketSegmentLogsums( + PathAlternativeList alternativeList) + { + if (alternativeList.getCount() > 1) + { + throw new UnsupportedOperationException( + "Walk logsums cannot be calculated for alternative lists containing multiple paths"); + } + + double utility = 0; + double distance = 0; + double gain = 0; + SandagBikeNode parent = null; + for (SandagBikeNode n : alternativeList.get(0)) + { + if (parent != null) + { + utility += configuration.getNetwork().getEdge(parent, n).walkCost; + distance += configuration.getNetwork().getEdge(parent, n).distance; + gain += configuration.getNetwork().getEdge(parent,n).gain; + } + parent = n; + } + + return new double[] {utility, distance * configuration.getDefaultMinutesPerMile(),gain}; + } + + @Override + protected List> getMarketSegmentIntrazonalCalculations() + { + IntrazonalCalculation intrazonalCalculation = IntrazonalCalculations + .minFactorIntrazonalCalculation( + IntrazonalCalculations.simpleFactorizer(0.5, 0), 1); + // do time then distance then gain + return Arrays.asList(intrazonalCalculation, intrazonalCalculation, intrazonalCalculation); + } + + public static void main(String... args) + { + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } + logger.info("loading property file: " + + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() + .toString()); + + logger.info("Building walk skims"); + // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; + @SuppressWarnings("unchecked") + // this is ok - the map will be String->String + Map propertyMap = (Map) ResourceUtil + .getResourceBundleAsHashMap(args[0]); + + SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); + Network network = factory + .createNetwork(); + + DecimalFormat formatter = new DecimalFormat("#.###"); + + logger.info("Generating mgra->mgra walk skims"); + // mgra->mgra + PathAlternativeListGenerationConfiguration configuration = new SandagWalkMgraMgraPathAlternativeListGenerationConfiguration( + propertyMap, network); + SandagWalkPathChoiceLogsumMatrixApplication application = new SandagWalkPathChoiceLogsumMatrixApplication( + configuration); + Map, double[]> logsums = application + .calculateMarketSegmentLogsums(); + + Path outputDirectory = Paths.get(configuration.getOutputDirectory()); + Path outputFile = outputDirectory.resolve(propertyMap + .get(WALK_LOGSUM_SKIM_MGRA_MGRA_FILE_PROPERTY)); + + try + { + Files.createDirectories(outputDirectory); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + Map originCentroids = configuration.getInverseOriginZonalCentroidIdMap(); + Map destinationCentroids = configuration + .getInverseDestinationZonalCentroidIdMap(); + + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + writer.println("i,j,percieved,actual,gain"); + StringBuilder sb; + for (NodePair od : new TreeSet<>(logsums.keySet())) + { // sort them so the output "looks nice" + sb = new StringBuilder(); + sb.append(originCentroids.get(od.getFromNode().getId())).append(","); + sb.append(destinationCentroids.get(od.getToNode().getId())).append(","); + double[] values = logsums.get(od); + sb.append(formatter.format(values[0])).append(","); // percieved + // time + sb.append(formatter.format(values[1])).append(","); // actual time + sb.append(formatter.format(values[2])); //gain + writer.println(sb.toString()); + } + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("Generating mgra->tap walk skims"); + // mgra->tap + configuration = new SandagWalkMgraTapPathAlternativeListGenerationConfiguration( + propertyMap, network); + application = new SandagWalkPathChoiceLogsumMatrixApplication(configuration); + Map, double[]> mgraTapLogsums = application + .calculateMarketSegmentLogsums(); + + // for later - get from the first configuration + outputDirectory = Paths.get(configuration.getOutputDirectory()); + outputFile = outputDirectory.resolve(propertyMap + .get(WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY)); + originCentroids = configuration.getInverseOriginZonalCentroidIdMap(); + destinationCentroids = configuration.getInverseDestinationZonalCentroidIdMap(); + + // tap->mgra + configuration = new SandagWalkTapMgraPathAlternativeListGenerationConfiguration( + propertyMap, network); + application = new SandagWalkPathChoiceLogsumMatrixApplication(configuration); + Map, double[]> tapMgraLogsums = application + .calculateMarketSegmentLogsums(); + + // resolve if not a pair + int initialSize = mgraTapLogsums.size() + tapMgraLogsums.size(); + + for (NodePair mgraTapPair : mgraTapLogsums.keySet()) + { + NodePair tapMgraPair = new NodePair( + mgraTapPair.getToNode(), mgraTapPair.getFromNode()); + if (!tapMgraLogsums.containsKey(tapMgraPair)) + tapMgraLogsums.put(tapMgraPair, mgraTapLogsums.get(mgraTapPair)); + } + + for (NodePair tapMgraPair : tapMgraLogsums.keySet()) + { + NodePair mgraTapPair = new NodePair( + tapMgraPair.getToNode(), tapMgraPair.getFromNode()); + if (!mgraTapLogsums.containsKey(mgraTapPair)) + mgraTapLogsums.put(mgraTapPair, tapMgraLogsums.get(tapMgraPair)); + } + int asymmPairCount = initialSize - (mgraTapLogsums.size() + tapMgraLogsums.size()); + if (asymmPairCount > 0) + logger.info("Boarding or alighting times defaulted to transpose for " + asymmPairCount + + " mgra tap pairs with missing asymmetrical information"); + + try + { + Files.createDirectories(outputDirectory); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + try (PrintWriter writer = new PrintWriter(outputFile.toFile())) + { + writer.println("mgra,tap,boardingPerceived,boardingActual,alightingPerceived,alightingActual,boardingGain,alightingGain"); + StringBuilder sb; + for (NodePair od : new TreeSet<>(mgraTapLogsums.keySet())) + { // sort them so the output "looks nice" + sb = new StringBuilder(); + sb.append(originCentroids.get(od.getFromNode().getId())).append(","); + sb.append(destinationCentroids.get(od.getToNode().getId())).append(","); + double[] boardingValues = mgraTapLogsums.get(od); + sb.append(formatter.format(boardingValues[0])).append(","); // boarding + // percieved + sb.append(formatter.format(boardingValues[1])).append(","); // boarding + // actual + double[] alightingValues = tapMgraLogsums.get(new NodePair<>(od.getToNode(), od + .getFromNode())); + sb.append(formatter.format(alightingValues[0])).append(","); // alighting + // percieved + sb.append(formatter.format(alightingValues[1])).append(","); // alighting + // actual + sb.append(formatter.format(boardingValues[2])).append(","); // boarding gain + sb.append(formatter.format(alightingValues[2])); // alighting gain + writer.println(sb.toString()); + } + } catch (IOException e) + { + throw new RuntimeException(e); + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java new file mode 100644 index 0000000..ead9201 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java @@ -0,0 +1,62 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import org.sandag.abm.active.Network; + +public class SandagWalkTapMgraPathAlternativeListGenerationConfiguration + extends SandagWalkPathAlternativeListGenerationConfiguration +{ + + public SandagWalkTapMgraPathAlternativeListGenerationConfiguration( + Map propertyMap, + Network network) + { + super(propertyMap, network); + this.PROPERTIES_MAXDIST_ZONE = Math.max( + Math.max( + Double.parseDouble(propertyMap.get("active.maxdist.walk.tap")), + Double.parseDouble(propertyMap.get("active.maxdist.micromobility.tap"))), + Double.parseDouble(propertyMap.get("active.maxdist.microtransit.tap"))); + this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.tap"; + } + + protected void createOriginZonalCentroidIdMap() + { + System.out.println("Creating TAP Zonal Centroid Id Map..."); + originZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.tap > 0) + { + originZonalCentroidIdMap.put((int) n.tap, n.getId()); + } + } + } + + protected void createDestinationZonalCentroidIdMap() + { + System.out.println("Creating MGRA Zonal Centroid Id Map..."); + destinationZonalCentroidIdMap = new HashMap(); + Iterator nodeIterator = network.nodeIterator(); + SandagBikeNode n; + while (nodeIterator.hasNext()) + { + n = nodeIterator.next(); + if (n.mgra > 0) + { + destinationZonalCentroidIdMap.put((int) n.mgra, n.getId()); + } + } + } + + public boolean isIntrazonalsNeeded() + { + return false; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java new file mode 100644 index 0000000..4a63681 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java @@ -0,0 +1,35 @@ +package org.sandag.abm.active.sandag; + +import java.util.HashMap; +import java.util.Map; + +public enum TurnType +{ + NONE(0), LEFT(1), RIGHT(2), REVERSAL(3); + + private int key; + private static Map map = new HashMap(); + + static + { + for (TurnType t : TurnType.values()) + { + map.put(t.key, t); + } + } + + private TurnType(final int key) + { + this.key = key; + } + + public static TurnType valueOf(int key) + { + return map.get(key); + } + + public int getKey() + { + return key; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java new file mode 100644 index 0000000..ed1b4f1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java @@ -0,0 +1,581 @@ +package org.sandag.abm.airport; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +public class AirportDestChoiceModel +{ + + private double[][] sizeTerms; // by + // segment, + // tazNumber + private double[][][] mgraProbabilities; // by + // segment, + // tazNumber, + // mgra + // index + // (sequential, + // 0-based) + private double[][] tazProbabilities; // by + // segment, + // origin + // taz + // 0-based + // (note + // since + // airport + // model, + // only + // 1 + // taz + // dimension + // is + // needed) + private int[] zipCodes; // by + // taz + + private TableDataSet alternativeData; // the + // alternatives, + // with + // a + // dest + // field + // indicating + // tazNumber + + private transient Logger logger = Logger.getLogger("airportModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication[] destModel; + private UtilityExpressionCalculator sizeTermUEC; + private Tracer tracer; + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + private boolean seek; + private HashMap rbMap; + private TableDataSet externalDataSet; + private int externalAirportColumn; + private int tazColumn; + private int mazOutColumn; + private int mazInbColumn; + + + private int airportMgra; + private int airportTaz; + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of airport model DMUs + */ + public AirportDestChoiceModel(HashMap rbMap, AirportDmuFactoryIf dmuFactory, String airportCode) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String airportDistUecFileName = Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".dc.uec.file"); + airportDistUecFileName = uecFileDirectory + airportDistUecFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".dc.data.page")); + int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".dc.size.page")); + + // read the model pages from the property file, create one choice model + // for each + destModel = new ChoiceModelApplication[AirportModelStructure.INTERNAL_PURPOSES]; + for (int i = 0; i < AirportModelStructure.INTERNAL_PURPOSES; ++i) + { + + // get page from property file + String purposeName = "airport."+airportCode+".dc.segment" + (i + 1) + ".page"; + String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); + purposeString.replaceAll(" ", ""); + int destModelPage = Integer.parseInt(purposeString); + + // initiate a DMU for each segment + AirportModelDMU dcDmu = dmuFactory.getAirportModelDMU(); + + // create a ChoiceModelApplication object for the filename, model + // page and data page. + destModel[i] = new ChoiceModelApplication(airportDistUecFileName, destModelPage, + dataPage, rbMap, (VariableTable) dcDmu); + } + + // get the alternative data from the first segment + UtilityExpressionCalculator uec = destModel[0].getUEC(); + alternativeData = uec.getAlternativeData(); + + // create a UEC to solve size terms for each MGRA + sizeTermUEC = new UtilityExpressionCalculator(new File(airportDistUecFileName), sizePage, + dataPage, rbMap, dmuFactory.getAirportModelDMU()); + + // set up the tracer object + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + airportMgra = Util.getIntegerValueFromPropertyMap(rbMap, "airport."+airportCode+".airportMgra"); + airportTaz = mgraManager.getTaz(airportMgra); + + // calculate the zip code array + calculateZipCodes(); + + //read the external station dataset into memory + String externalStationFile = Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".externalStationFile"); + readExternalPercentages(externalStationFile,airportCode); + } + + /** + * Iterate through the segments, calculating the mgra probabilities for each + */ + public void calculateMgraProbabilities(AirportDmuFactoryIf dmuFactory) + { + + logger.info("Calculating Airport Model Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int alternatives = sizeTermUEC.getNumberOfAlternatives(); + + double[][] mgraSizeTerms = new double[alternatives][maxMgra + 1]; + IndexValues iv = new IndexValues(); + AirportModelDMU aDmu = dmuFactory.getAirportModelDMU(); + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = sizeTermUEC.solve(iv, aDmu, null); + + // store the size terms + for (int segment = 0; segment < alternatives; ++segment) + { + mgraSizeTerms[segment][mgra] = utilities[segment]; + } + + // log + if (tracer.isTraceOn() && tracer.isTraceZone(taz)) + { + + logger.info("Size Term calculations for mgra " + mgra); + sizeTermUEC.logResultsArray(logger, 0, mgra); + + } + } + + // now iterate through tazs, calculate probabilities + int[] tazs = tazManager.getTazs(); + int maxTaz = tazManager.getMaxTaz(); + + // initialize arrays + mgraProbabilities = new double[alternatives][maxTaz + 1][]; + sizeTerms = new double[alternatives][maxTaz + 1]; + + // calculate arrays + for (int segment = 0; segment < alternatives; ++segment) + { + for (int taz = 0; taz < tazs.length; ++taz) + { + int tazNumber = tazs[taz]; + int[] mgraArray = tazManager.getMgraArray(tazNumber); + + // initialize the vector of mgras for this purpose-taz + mgraProbabilities[segment][tazNumber] = new double[mgraArray.length]; + + // first calculate the sum of size for all the mgras in the taz + double sum = 0; + for (int mgra = 0; mgra < mgraArray.length; ++mgra) + { + + int mgraNumber = mgraArray[mgra]; + + sum += mgraSizeTerms[segment][mgraNumber]; + } + // store the logsum in the size term array by taz + if (sum > 0.0) sizeTerms[segment][tazNumber] = Math.log(sum + 1.0); + + // now calculate the cumulative probability distribution + double lastProb = 0.0; + for (int mgra = 0; mgra < mgraArray.length; ++mgra) + { + + int mgraNumber = mgraArray[mgra]; + if (sum > 0.0) + mgraProbabilities[segment][tazNumber][mgra] = lastProb + + mgraSizeTerms[segment][mgraNumber] / sum; + lastProb = mgraProbabilities[segment][tazNumber][mgra]; + } + if (sum > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) + logger.info("Error: segment " + segment + " taz " + tazNumber + + " cum prob adds up to " + lastProb); + } + + } + logger.info("Finished Calculating Airport Model Size Terms"); + } + + /** + * Calculate the zip codes at a taz level from the mgra data file. This + * requires the mgra data to be specified as mgra.socec.file in the + * properties file. The mgra file must have four fields: zone, taz, pop, and + * zip The taz zip is coded based upon the highest population mgra within + * the taz. + * + * @return a zip code array dimensioned by taz numbers + */ + public void calculateZipCodes() + { + + logger.info("Calculating Airport Model TAZ Zip Code Array"); + + zipCodes = new int[tazManager.maxTaz + 1]; + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String fileName = directory + Util.getStringValueFromPropertyMap(rbMap, "mgra.socec.file"); + + logger.info("Begin reading the data in file " + fileName); + TableDataSet mgraTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + mgraTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + // iterate through TAZs and store zip codes in zipCodes array + int[] tazs = tazManager.getTazs(); + for (int i = 0; i < tazs.length; ++i) + { + + int tazNumber = tazs[i]; + int maxPop = 0; + int zip = 0; + for (int row = 1; row <= mgraTable.getRowCount(); ++row) + { + + if (mgraTable.getValueAt(row, "taz") == tazNumber) + { + int pop = (int) mgraTable.getValueAt(row, "pop"); + if (pop > maxPop) + { + maxPop = pop; + zip = (int) mgraTable.getValueAt(row, "ZIP09"); + } + } + } + // if iterate through mgra data, and no mgras with pop found, then + // choose zip of first mgra in taz + if (zip == 0) + { + for (int row = 1; row <= mgraTable.getRowCount(); ++row) + { + + if (mgraTable.getValueAt(row, "taz") == tazNumber) + { + zip = (int) mgraTable.getValueAt(row, "ZIP09"); + break; + } + } + } + // store zip in array + zipCodes[tazNumber] = zip; + } + logger.info("Finished Calculating Airport Model TAZ Zip Code Array"); + } + + /** + * Calculate taz probabilities. This method initializes and calculates the + * tazProbabilities array. + */ + public void calculateTazProbabilities(AirportDmuFactoryIf dmuFactory) + { + + if (sizeTerms == null) + { + logger.error("Error: attemping to execute airportmodel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); + throw new RuntimeException(); + } + + logger.info("Calculating Airport Model TAZ Probabilities Arrays"); + + // initialize taz probabilities array + int segments = sizeTerms.length; + int maxTaz = tazManager.getMaxTaz(); + tazProbabilities = new double[segments][maxTaz + 1]; + + // Note: this is an aggregate model to calculate utilities, but we need + // income so we need a party object + AirportParty airportParty = new AirportParty(1001); + + AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); + dmu.setZips(zipCodes); + dmu.setSizeTerms(sizeTerms); + dmu.setAirportParty(airportParty); + + int airportTaz = mgraManager.getTaz(airportMgra); + + // segments are combinations of 4 purposes and 8 income groups, which + // apply only to resident purposes + for (int segment = 0; segment < segments; ++segment) + { + + int purpose = AirportModelStructure.getPurposeFromDCSizeSegment(segment); + int income = AirportModelStructure.getIncomeFromDCSizeSegment(segment); + + airportParty.setPurpose((byte) purpose); + airportParty.setIncome((byte) income); + + // set airport taz as origin. Destination tazs controlled by + // alternative file. + IndexValues dmuIndex = dmu.getDmuIndex(); + dmuIndex.setOriginZone(airportTaz); + + // Calculate utilities & probabilities + destModel[purpose].computeUtilities(dmu, dmuIndex); + + // Store probabilities (by segment) + tazProbabilities[segment] = Arrays.copyOf( + destModel[purpose].getCumulativeProbabilities(), + destModel[purpose].getCumulativeProbabilities().length); + } + logger.info("Finished Calculating Airport Model TAZ Probabilities Arrays"); + } + + /** + * Choose an MGRA + * + * @param purpose + * Purpose + * @param income + * Income + * @param randomNumber + * Random number + * @return The chosen MGRA number + */ + public int chooseMGRA(int purpose, int income, double randomNumber) + { + + // first find a TAZ + int segment = AirportModelStructure.getDCSizeSegment(purpose, income); + int alt = 0; + double[] tazCumProb = tazProbabilities[segment]; + double altProb = 0; + double cumProb = 0; + for (int i = 0; i < tazCumProb.length; ++i) + { + if (tazCumProb[i] > randomNumber) + { + alt = i; + if (i != 0) + { + cumProb = tazCumProb[i - 1]; + altProb = tazCumProb[i] - tazCumProb[i - 1]; + } else + { + altProb = tazCumProb[i]; + } + break; + } + } + + // get the taz number of the alternative, and an array of mgras in that + // taz + int tazNumber = (int) alternativeData.getValueAt(alt + 1, "dest"); + int[] mgraArray = tazManager.getMgraArray(tazNumber); + + // now find an MGRA in the taz corresponding to the random number drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives probability + int mgraNumber = 0; + double[] mgraCumProb = mgraProbabilities[segment][tazNumber]; + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > randomNumber) + { + mgraNumber = mgraArray[i]; + } + } + // return the chosen MGRA number + return mgraNumber; + } + + /** + * Iterate through an array of AirportParty objects, choosing origin MGRAs + * for each and setting the result back in the objects. + * + * @param airportParties + * An array of AirportParty objects + */ + public void chooseOrigins(AirportParty[] airportParties) + { + + // iterate through the array, choosing mgras and setting them + for (AirportParty party : airportParties) + { + + int income = party.getIncome(); + int purpose = party.getPurpose(); + double random = party.getRandom(); + int mgra = -99; + if (purpose < AirportModelStructure.INTERNAL_PURPOSES){ + mgra = chooseMGRA(purpose, income, random); + + // if this is a departing travel party, the origin is the chosen + // mgra, and the destination is the airport terminal + if (party.getDirection() == AirportModelStructure.DEPARTURE) + { + party.setOriginMGRA(mgra); + party.setOriginTAZ(mgraManager.getTaz(mgra)); + party.setDestinationMGRA(airportMgra); + party.setDestinationTAZ(airportTaz); + } else + { + party.setOriginMGRA(airportMgra); + party.setOriginTAZ(airportTaz); + party.setDestinationMGRA(mgra); + party.setDestinationTAZ(mgraManager.getTaz(mgra)); + } + + }else{ + chooseExternalStation(party, random); + } + } + } + /** + * Read a csv file with probabilities by external station. Required fields in the file: + * + * taz The TAZ number of the external station + * mgraOut The MGRA number for trips leaving the region (closest MGRA to outbound external TAZ) + * mgraRet The MGRA number for trips returning to the region (closest MGRA to inbound external TAZ) + * AIRPORTNAME.pct The share of trips entering\exiting at the external station for the airport + * + * + * @param fileName The name of the external station file + */ + private void readExternalPercentages(String fileName, String airportCode){ + + logger.info("Begin reading the data in file " + fileName); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + externalDataSet = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + externalAirportColumn = externalDataSet.getColumnPosition(airportCode+".pct"); + tazColumn = externalDataSet.getColumnPosition("taz"); + mazOutColumn = externalDataSet.getColumnPosition("mgraOut"); + mazInbColumn = externalDataSet.getColumnPosition("mgraRet"); + + if(externalAirportColumn<=0|| tazColumn<=0 || mazOutColumn<=0 || mazInbColumn<=0){ + logger.error("Check fields in external station file "+fileName); + logger.error("File should have fields taz, mgraOut, mgraReg and "+airportCode+".pct"); + throw new RuntimeException(); + } + + logger.info("End reading the data in file " + fileName); + + } + + /** + * Choose an external TAZ and associated MAZ for the tour if it is external. The + * probabilities are in the external station file. + * + * @param airportParty The airport party + * @param randomNumber A uniform random number + */ + private void chooseExternalStation(AirportParty airportParty, double randomNumber){ + + double cumProb=0; + int taz = -1; + int maz = -1; + for(int row = 1; row <= externalDataSet.getRowCount();++row){ + + cumProb += externalDataSet.getValueAt(row,externalAirportColumn); + if(cumProb > randomNumber){ + taz = (int) externalDataSet.getValueAt(row, tazColumn); + + if(airportParty.getDirection() == AirportModelStructure.DEPARTURE){ + maz = (int) externalDataSet.getValueAt(row,mazInbColumn); + airportParty.setOriginTAZ(taz); + airportParty.setOriginMGRA(maz); + airportParty.setDestinationMGRA(airportMgra); + airportParty.setDestinationTAZ(airportTaz); + }else{ + maz = (int) externalDataSet.getValueAt(row,mazOutColumn); + airportParty.setDestinationTAZ(taz); + airportParty.setDestinationMGRA(maz); + airportParty.setOriginMGRA(airportMgra); + airportParty.setOriginTAZ(airportTaz); + + } + + } + + } + if(taz==-1||maz==-1){ + logger.fatal("Error: could not find external destination for airport tour"); + logger.fatal("Make sure probabilities add up to 1.0 in external station file"); + throw new RuntimeException(); + } + + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java new file mode 100644 index 0000000..9a79e69 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java @@ -0,0 +1,35 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.airport; + +import java.io.Serializable; + +/** + * ArcCtrampDmuFactory is a class that creates Airport Model DMU objects + * + * @author Joel Freedman + */ +public class AirportDmuFactory + implements AirportDmuFactoryIf, Serializable +{ + + //private AirportModelStructure airportModelStructure; + + public AirportDmuFactory()//AirportModelStructure modelStructure) + { + //this.airportModelStructure = modelStructure; + } + + public AirportModelDMU getAirportModelDMU() + { + return new AirportModelDMU(null); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java new file mode 100644 index 0000000..8f233cb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java @@ -0,0 +1,11 @@ +package org.sandag.abm.airport; + +/** + * A DMU factory interface + */ +public interface AirportDmuFactoryIf +{ + + AirportModelDMU getAirportModelDMU(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java new file mode 100644 index 0000000..b988d14 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java @@ -0,0 +1,378 @@ +package org.sandag.abm.airport; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; +import org.sandag.abm.accessibilities.McLogsumsAppender; +import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +public class AirportModeChoiceModel +{ + private transient Logger logger = Logger.getLogger("airportModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication driveAloneModel; + private ChoiceModelApplication shared2Model; + private ChoiceModelApplication shared3Model; + private ChoiceModelApplication transitModel; + private ChoiceModelApplication accessModel; + + private Tracer tracer; + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + private boolean seek; + private HashMap rbMap; + + private McLogsumsCalculator logsumHelper; + private TripModeChoiceDMU mcDmuObject; + private AutoTazSkimsCalculator tazDistanceCalculator; + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of airport model DMUs + */ + public AirportModeChoiceModel(HashMap rbMap, AirportDmuFactoryIf dmuFactory, String airportCode) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String airportModeUecFileName = Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.uec.file"); + airportModeUecFileName = uecFileDirectory + airportModeUecFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.data.page")); + int daPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.da.page")); + int s2Page = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.s2.page")); + int s3Page = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.s3.page")); + int transitPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.transit.page")); + int accessPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".mc.accessMode.page")); + + logger.info("Creating Airport Model Mode Choice Application UECs"); + + // create a DMU + AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); + + // create a ChoiceModelApplication object for drive-alone mode choice + driveAloneModel = new ChoiceModelApplication(airportModeUecFileName, daPage, dataPage, + rbMap, (VariableTable) dmu); + + // create a ChoiceModelApplication object for shared 2 mode choice + shared2Model = new ChoiceModelApplication(airportModeUecFileName, s2Page, dataPage, rbMap, + (VariableTable) dmu); + + // create a ChoiceModelApplication object for shared 3+ mode choice + shared3Model = new ChoiceModelApplication(airportModeUecFileName, s3Page, dataPage, rbMap, + (VariableTable) dmu); + + // create a ChoiceModelApplication object for transit mode choice + transitModel = new ChoiceModelApplication(airportModeUecFileName, transitPage, dataPage, + rbMap, (VariableTable) dmu); + + // create a ChoiceModelApplication object for access mode choice + accessModel = new ChoiceModelApplication(airportModeUecFileName, accessPage, dataPage, + rbMap, (VariableTable) dmu); + + logger.info("Finished Creating Airport Model Mode Choice Application UECs"); + + // set up the tracer object + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(rbMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + SandagModelStructure modelStructure = new SandagModelStructure(); + mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); + + } + + + /** + * Choose airport arrival mode and trip mode for this party. Store results + * in the party object passed as argument. + * + * @param party + * The travel party + * @param dmu + * An airport model DMU + */ + public void chooseMode(AirportParty party, AirportModelDMU dmu) + { + + int origMgra = party.getOriginMGRA(); + int destMgra = party.getDestinationMGRA(); + int origTaz = mgraManager.getTaz(origMgra); + int destTaz = mgraManager.getTaz(destMgra); + int period = party.getDepartTime(); + + boolean inbound = false; + if (party.getDirection() == AirportModelStructure.ARRIVAL) inbound = true; + + dmu.setAirportParty(party); + dmu.setDmuIndexValues(party.getID(), origTaz, destTaz); + + // set trip mc dmu values for transit logsum (gets replaced below by uec values) + double c_ivt = -0.03; + double c_cost = - 0.0003; + + // Solve trip mode level utilities + mcDmuObject.setIvtCoeff(c_ivt); + mcDmuObject.setCostCoeff(c_cost); + double walkTransitLogsum = -999.0; + double driveTransitLogsum = -999.0; + + // if 1-person party, solve for the drive-alone and 2-person logsums + if (party.getSize() == 1) + { + driveAloneModel.computeUtilities(dmu, dmu.getDmuIndex()); + double driveAloneLogsum = driveAloneModel.getLogsum(); + dmu.setDriveAloneLogsum(driveAloneLogsum); + + c_ivt = driveAloneModel.getUEC().lookupVariableIndex("c_ivt"); + c_cost = driveAloneModel.getUEC().lookupVariableIndex("c_cost"); + + shared2Model.computeUtilities(dmu, dmu.getDmuIndex()); + double shared2Logsum = shared2Model.getLogsum(); + dmu.setShared2Logsum(shared2Logsum); + + } else if (party.getSize() == 2) + { // if 2-person party solve for the + // shared 2 and shared 3+ logsums + shared2Model.computeUtilities(dmu, dmu.getDmuIndex()); + double shared2Logsum = shared2Model.getLogsum(); + dmu.setShared2Logsum(shared2Logsum); + + shared3Model.computeUtilities(dmu, dmu.getDmuIndex()); + double shared3Logsum = shared3Model.getLogsum(); + dmu.setShared3Logsum(shared3Logsum); + + c_ivt = shared2Model.getUEC().lookupVariableIndex("c_ivt"); + c_cost = shared2Model.getUEC().lookupVariableIndex("c_cost"); + + } else + { // if 3+ person party, solve the shared 3+ logsums + shared3Model.computeUtilities(dmu, dmu.getDmuIndex()); + double shared3Logsum = shared3Model.getLogsum(); + dmu.setShared3Logsum(shared3Logsum); + + c_ivt = shared3Model.getUEC().lookupVariableIndex("c_ivt"); + c_cost = shared3Model.getUEC().lookupVariableIndex("c_cost"); + } + + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); + walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + if (party.getDirection() == AirportModelStructure.DEPARTURE) + { + logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); + } else + { + logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); + } + + dmu.setWalkTransitLogsum(walkTransitLogsum); + dmu.setDriveTransitLogsum(driveTransitLogsum); + + transitModel.computeUtilities(dmu, dmu.getDmuIndex()); + double transitLogsum = transitModel.getLogsum(); + dmu.setTransitLogsum(transitLogsum); + + // calculate access mode utility and choose access mode + accessModel.computeUtilities(dmu, dmu.getDmuIndex()); + int accessMode = accessModel.getChoiceResult(party.getRandom()); + party.setArrivalMode((byte) accessMode); + + // choose trip mode + int tripMode = 0; + int occupancy = AirportModelStructure.getOccupancy(accessMode, party.getSize()); + double randomNumber = party.getRandom(); + + float valueOfTime = 0; + + if ((accessMode != AirportModelStructure.TRANSIT) && (! AirportModelStructure.taxiTncMode(accessMode))) + { + if (occupancy == 1) + { + tripMode = occupancy; + + //following gets vot from UEC + UtilityExpressionCalculator uec = driveAloneModel.getUEC(); + int votIndex = uec.lookupVariableIndex("vot"); + valueOfTime = (float) uec.getValueForIndex(votIndex); + + } else if (occupancy == 2) + { + tripMode = occupancy; + + //following gets vot from UEC + UtilityExpressionCalculator uec = shared2Model.getUEC(); + int votIndex = uec.lookupVariableIndex("vot"); + valueOfTime = (float) uec.getValueForIndex(votIndex); + + } else if (occupancy > 2) + { + tripMode = 3; + + //following gets vot from UEC + UtilityExpressionCalculator uec = shared3Model.getUEC(); + int votIndex = uec.lookupVariableIndex("vot"); + valueOfTime = (float) uec.getValueForIndex(votIndex); + + } + } else if (accessMode == AirportModelStructure.TRANSIT) + { + int choice = transitModel.getChoiceResult(randomNumber); + double[][] bestTapPairs; + if (choice == 1){ + tripMode = AirportModelStructure.REALLOCATE_WLKTRN; //walk-transit + bestTapPairs = logsumHelper.getBestWtwTripTaps(); + } + else if (choice == 2){ + tripMode = AirportModelStructure.REALLOCATE_KNRPERTRN; //knr-personal tNCVehicle + if (party.getDirection() == AirportModelStructure.DEPARTURE) + bestTapPairs = logsumHelper.getBestDtwTripTaps(); + else + bestTapPairs = logsumHelper.getBestWtdTripTaps(); + } + else { + tripMode = AirportModelStructure.REALLOCATE_KNRTNCTRN; //knr-TNC + if (party.getDirection() == AirportModelStructure.DEPARTURE) + bestTapPairs = logsumHelper.getBestDtwTripTaps(); + else + bestTapPairs = logsumHelper.getBestWtdTripTaps(); + } + + //pick transit path from N-paths + double rn = party.getRandom(); + int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, party.getDebugChoiceModels(), logger); + int boardTap = (int) bestTapPairs[pathIndex][0]; + int alightTap = (int) bestTapPairs[pathIndex][1]; + int set = (int) bestTapPairs[pathIndex][2]; + party.setBoardTap(boardTap); + party.setAlightTap(alightTap); + party.setSet(set); + + //following gets vot from UEC + UtilityExpressionCalculator uec = transitModel.getUEC(); + int votIndex = uec.lookupVariableIndex("vot"); + valueOfTime = (float) uec.getValueForIndex(votIndex); + + }else if(accessMode == AirportModelStructure.TAXI){ + + tripMode=AirportModelStructure.REALLOCATE_TAXI; + } + else if(accessMode == AirportModelStructure.TNC_SINGLE){ + + tripMode=AirportModelStructure.REALLOCATE_TNCSINGLE; + + } + else if(accessMode == AirportModelStructure.TNC_SHARED){ + + tripMode=AirportModelStructure.REALLOCATE_TNCSHARED; + + } + + //set the VOT + if(AirportModelStructure.taxiTncMode(accessMode) ) { + UtilityExpressionCalculator uec = null; + + //following gets vot from UEC + if(occupancy==1) + uec = driveAloneModel.getUEC(); + else if (occupancy==2) + uec = shared2Model.getUEC(); + else + uec = shared3Model.getUEC(); + + int votIndex = uec.lookupVariableIndex("vot"); + valueOfTime = (float) uec.getValueForIndex(votIndex); + + } + party.setMode((byte) tripMode); + party.setValueOfTime(valueOfTime); + } + + /** + * Choose modes for internal trips. + * + * @param airportParties + * An array of travel parties, with destinations already chosen. + * @param dmuFactory + * A DMU Factory. + */ + public void chooseModes(AirportParty[] airportParties, AirportDmuFactoryIf dmuFactory) + { + + AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); + // iterate through the array, choosing mgras and setting them + for (AirportParty party : airportParties) + { + + int ID = party.getID(); + + if ((ID <= 5) || (ID % 100) == 0) + logger.info("Choosing mode for party " + party.getID()); + + chooseMode(party, dmu); + + } + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java new file mode 100644 index 0000000..f9c3159 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java @@ -0,0 +1,233 @@ +package org.sandag.abm.airport; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +public class AirportModel +{ + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + + private MatrixDataServerRmi ms; + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + private HashMap rbMap; + private static float sampleRate; + private static int iteration; + private static String airportCode; + + /** + * Constructor + * + * @param rbMap + */ + public AirportModel(HashMap rbMap, float aSampleRate) + { + this.rbMap = rbMap; + this.sampleRate=aSampleRate; + } + + /** + * Run airport model. + */ + public void runModel() + { + Runtime gfg = Runtime.getRuntime(); + long memory1; + // checking the total memeory + System.out.println("Total memory is: "+ gfg.totalMemory()); + // checking free memory + memory1 = gfg.freeMemory(); + System.out.println("Initial free memory at Airport model: "+ memory1); + // calling the garbage collector on demand + gfg.gc(); + memory1 = gfg.freeMemory(); + System.out.println("Free memory after garbage "+ "collection: " + memory1); + + AirportDmuFactory dmuFactory = new AirportDmuFactory(); + + AirportPartyManager apm = new AirportPartyManager(rbMap, sampleRate, airportCode); + + apm.generateAirportParties(); + AirportParty[] parties = apm.getParties(); + + AirportDestChoiceModel destChoiceModel = new AirportDestChoiceModel(rbMap, dmuFactory,airportCode); + destChoiceModel.calculateMgraProbabilities(dmuFactory); + destChoiceModel.calculateTazProbabilities(dmuFactory); + destChoiceModel.chooseOrigins(parties); + + AirportModeChoiceModel modeChoiceModel = new AirportModeChoiceModel(rbMap, dmuFactory,airportCode); + modeChoiceModel.chooseModes(parties, dmuFactory); + + apm.writeOutputFile(rbMap); + + logger.info("Airport Model successfully completed!"); + + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @param args + */ + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Airport Model")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else { + propertiesFile = args[0]; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + if(args[i].equalsIgnoreCase("-airport")){ + airportCode = args[i+1]; + } + } + } + + logger.info("Airport Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + AirportModel airportModel = new AirportModel(pMap, sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = airportModel.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + airportModel.ms = matrixServer; + } else + { + airportModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + airportModel.ms.testRemote("AirportModel"); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(airportModel.ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + airportModel.runModel(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java new file mode 100644 index 0000000..b564cd4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java @@ -0,0 +1,478 @@ +package org.sandag.abm.airport; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class AirportModelDMU + implements Serializable, VariableTable +{ + protected IndexValues dmuIndex; + + private AirportParty airportParty; + private double[][] sizeTerms; // dimensioned + // by + // segment, + // taz + private int[] zips; // dimensioned + // by + // taz + public static final int OUT = 0; + public static final int IN = 1; + protected static final int NUM_DIR = 2; + + // estimation file defines time periods as: + // 1 | Early AM: 3:00 AM - 5:59 AM | + // 2 | AM Peak: 6:00 AM - 8:59 AM | + // 3 | Early MD: 9:00 AM - 11:59 PM | + // 4 | Late MD: 12:00 PM - 3:29 PM | + // 5 | PM Peak: 3:30 PM - 6:59 PM | + // 6 | Evening: 7:00 PM - 2:59 AM | + + protected static final int LAST_EA_INDEX = 3; + protected static final int LAST_AM_INDEX = 9; + protected static final int LAST_MD_INDEX = 22; + protected static final int LAST_PM_INDEX = 29; + + protected static final int EA = 1; + protected static final int AM = 2; + protected static final int MD = 3; + protected static final int PM = 4; + protected static final int EV = 5; + + protected static final int EA_D = 1; // 5am + protected static final int AM_D = 5; // 7am + protected static final int MD_D = 15; // 12pm + protected static final int PM_D = 27; // 6pm + protected static final int EV_D = 35; // 10pm + protected static final int[] DEFAULT_DEPART_INDICES = {-1, EA_D, AM_D, + MD_D, PM_D, EV_D }; + + protected static final int EA_A = 2; // 5:30am + protected static final int AM_A = 6; // 7:30am + protected static final int MD_A = 16; // 12:30pm + protected static final int PM_A = 28; // 6:30pm + protected static final int EV_A = 36; // 10:30pm + protected static final int[] DEFAULT_ARRIVE_INDICES = {-1, EA_A, AM_A, + MD_A, PM_A, EV_A }; + + protected String[][] departArriveCombinationLabels = { {"EA", "EA"}, + {"EA", "AM"}, {"EA", "MD"}, {"EA", "PM"}, {"EA", "EV"}, {"AM", "AM"}, {"AM", "MD"}, + {"AM", "PM"}, {"AM", "EV"}, {"MD", "MD"}, {"MD", "PM"}, {"MD", "EV"}, {"PM", "PM"}, + {"PM", "EV"}, {"EV", "EV"} }; + + protected int[][] departArriveCombinations = { {EA, EA}, {EA, AM}, + {EA, MD}, {EA, PM}, {EA, EV}, {AM, AM}, {AM, MD}, {AM, PM}, {AM, EV}, {MD, MD}, + {MD, PM}, {MD, EV}, {PM, PM}, {PM, EV}, {EV, EV} }; + + private double driveAloneLogsum; + private double shared2Logsum; + private double shared3Logsum; + private double transitLogsum; + + protected double walkTransitLogsum; + protected double driveTransitLogsum; + + protected Logger _logger = null; + + protected HashMap methodIndexMap; + + + public AirportModelDMU(Logger logger) + { + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + if (logger == null) + { + _logger = Logger.getLogger(AirportModelDMU.class); + } else _logger = logger; + } + + /** + * Set up the method index hashmap, where the key is the getter method for a + * data item and the value is the index. + */ + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + methodIndexMap.put("getDirection", 0); + methodIndexMap.put("getPurpose", 1); + methodIndexMap.put("getSize", 2); + methodIndexMap.put("getIncome", 3); + methodIndexMap.put("getDepartTime", 4); + methodIndexMap.put("getNights", 5); + methodIndexMap.put("getOriginMGRA", 6); + methodIndexMap.put("getLnDestChoiceSizeTazAlt", 7); + methodIndexMap.put("getDestZipAlt", 8); + + methodIndexMap.put("getWalkTransitLogsum", 10); + methodIndexMap.put("getDriveTransitLogsum", 11); + + methodIndexMap.put("getAvAvailable", 70); + + methodIndexMap.put("getDriveAloneLogsum", 90); + methodIndexMap.put("getShared2Logsum", 91); + methodIndexMap.put("getShared3Logsum", 92); + methodIndexMap.put("getTransitLogsum", 93); + + } + + /** + * Look up and return the value for the variable according to the index. + * + */ + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getDirection(); + break; + case 1: + returnValue = getPurpose(); + break; + case 2: + returnValue = getSize(); + break; + case 3: + returnValue = getIncome(); + break; + case 4: + returnValue = getDepartTime(); + break; + case 5: + returnValue = getNights(); + break; + case 6: + returnValue = getOriginMGRA(); + break; + case 7: + returnValue = getLnDestChoiceSizeTazAlt(arrayIndex); + break; + case 8: + returnValue = getDestZipAlt(arrayIndex); + break; + case 10: + returnValue = getWalkTransitLogsum(); + break; + case 11: + returnValue = getDriveTransitLogsum(); + break; + case 70: + returnValue = getAvAvailable(); + break; + case 90: + returnValue = getDriveAloneLogsum(); + break; + case 91: + returnValue = getShared2Logsum(); + break; + case 92: + returnValue = getShared3Logsum(); + break; + case 93: + returnValue = getTransitLogsum(); + break; + default: + _logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + + } + + + /** + * Get travel party direction. + * + * @return Travel party direction. + */ + public int getDirection() + { + return airportParty.getDirection(); + } + + /** + * Get travel party purpose. + * + * @return Travel party direction. + */ + public int getPurpose() + { + return airportParty.getPurpose(); + } + + /** + * Get travel party size. + * + * @return Travel party size. + */ + public int getSize() + { + return airportParty.getSize(); + } + + /** + * Get travel party income. + * + * @return Travel party income. + */ + public int getIncome() + { + return airportParty.getIncome(); + } + + /** + * Get the departure time for the trip + * + * @return Trip departure time. + */ + public int getDepartTime() + { + return airportParty.getDepartTime(); + } + + /** + * Get the number of nights + * + * @return Travel party number of nights. + */ + public int getNights() + { + return airportParty.getNights(); + } + + /** + * Get the origin(non-airport) MGRA + * + * @return Travel party origin MGRA + */ + public int getOriginMGRA() + { + return airportParty.getOriginMGRA(); + } + + public int getAvAvailable() { + + if(airportParty.getAvAvailable()) + return 1; + + return 0; + } + + + /** + * Set the index values for this DMU. + * + * @param id + * @param origTaz + * @param destTaz + */ + public void setDmuIndexValues(int id, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(id); + dmuIndex.setZoneIndex(origTaz); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (airportParty.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug Airport Model"); + } + + } + + /** + * Get the appropriate size term for this purpose and income market. + * + * @param tazNumber + * the number of the taz + * @return the size term + */ + public double getLnDestChoiceSizeTazAlt(int alt) + { + + int purpose = getPurpose(); + int income = getIncome(); + + int segment = AirportModelStructure.getDCSizeSegment(purpose, income); + + return sizeTerms[segment][alt]; + } + + /** + * Get the destination district for this alternative. + * + * @param tazNumber + * @return number of destination district. + */ + public int getDestZipAlt(int alt) + { + + return zips[alt]; + } + + /** + * Set size terms + * + * @param sizeTerms + * A double[][] array dimensioned by segments (purp\income + * groups) and taz numbers + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + /** + * set the zip codes + * + * @param zips + * int[] dimensioned by taz number + */ + public void setZips(int[] zips) + { + this.zips = zips; + } + + /** + * Set the airport party object. + * + * @param party + * The airport party. + */ + public void setAirportParty(AirportParty party) + { + + airportParty = party; + } + + /** + * @return the dmuIndex + */ + public IndexValues getDmuIndex() + { + return dmuIndex; + } + + /** + * @return the driveAloneLogsum + */ + public double getDriveAloneLogsum() + { + return driveAloneLogsum; + } + + /** + * @param driveAloneLogsum + * the driveAloneLogsum to set + */ + public void setDriveAloneLogsum(double driveAloneLogsum) + { + this.driveAloneLogsum = driveAloneLogsum; + } + + /** + * @return the shared2Logsum + */ + public double getShared2Logsum() + { + return shared2Logsum; + } + + /** + * @param shared2Logsum + * the shared2Logsum to set + */ + public void setShared2Logsum(double shared2Logsum) + { + this.shared2Logsum = shared2Logsum; + } + + /** + * @return the shared3Logsum + */ + public double getShared3Logsum() + { + return shared3Logsum; + } + + /** + * @param shared3Logsum + * the shared3Logsum to set + */ + public void setShared3Logsum(double shared3Logsum) + { + this.shared3Logsum = shared3Logsum; + } + + /** + * @return the transitLogsum + */ + public double getTransitLogsum() + { + return transitLogsum; + } + + /** + * @param transitLogsum + * the transitLogsum to set + */ + public void setTransitLogsum(double transitLogsum) + { + this.transitLogsum = transitLogsum; + } + + public double getWalkTransitLogsum() { + return walkTransitLogsum; + } + + public void setWalkTransitLogsum(double walkTransitLogsum) { + this.walkTransitLogsum = walkTransitLogsum; + } + + public double getDriveTransitLogsum() { + return driveTransitLogsum; + } + + public void setDriveTransitLogsum(double driveTransitLogsum) { + this.driveTransitLogsum = driveTransitLogsum; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java new file mode 100644 index 0000000..4b664ca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java @@ -0,0 +1,239 @@ +package org.sandag.abm.airport; + +public final class AirportModelStructure +{ + + public static final byte PURPOSES = 5; + public static final byte RESIDENT_BUSINESS = 0; + public static final byte RESIDENT_PERSONAL = 1; + public static final byte VISITOR_BUSINESS = 2; + public static final byte VISITOR_PERSONAL = 3; + public static final byte EXTERNAL = 4; + + public static final byte INTERNAL_PURPOSES = 4; + + public static final byte DEPARTURE = 0; + public static final byte ARRIVAL = 1; + + public static final byte INCOME_SEGMENTS = 8; + public static final byte DC_SIZE_SEGMENTS = INCOME_SEGMENTS * 2 + 2; + + public static final int AM = 0; + public static final int PM = 1; + public static final int OP = 2; + public static final int[] SKIM_PERIODS = {AM, PM, OP}; + public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; + public static final int UPPER_EA = 3; + public static final int UPPER_AM = 9; + public static final int UPPER_MD = 22; + public static final int UPPER_PM = 29; + public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; + + public static final byte ACCESS_MODES = 11; + + public static final byte PARK_TMNL = 1; + public static final byte PARK_SANOFF = 2; + public static final byte PARK_PVTOFF = 3; + public static final byte PUDO_ESC = 4; + public static final byte PUDO_CURB = 5; + public static final byte RENTAL = 6; + public static final byte TAXI = 7; + public static final byte TNC_SINGLE = 8; + public static final byte TNC_SHARED = 9; + public static final byte SHUTTLE_VAN = 10; + public static final byte TRANSIT = 11; + + //reallocate the trip modes from the access choice model to ones that the trip table and other code can read, consistent with + //resident models. + public static final byte REALLOCATE_WLKTRN = 6; //walk access + public static final byte REALLOCATE_KNRPERTRN = 8; //knr-personal tNCVehicle + public static final byte REALLOCATE_KNRTNCTRN = 9; //knr-TNC + public static final byte REALLOCATE_TAXI = 10; + public static final byte REALLOCATE_TNCSINGLE = 11; + public static final byte REALLOCATE_TNCSHARED = 12; + + private AirportModelStructure() + { + } + + /** + * Calculate and return the destination choice size term segment + * + * @param purpose + * @param income + * @return The dc size term segment, currently 0-17, where: 0-7 are 8 income + * groups for RES_BUS 8-15 are 8 income groups for RES_PER 16 is + * VIS_BUS 17 is VIS_PER + */ + public static int getDCSizeSegment(int purpose, int income) + { + + int segment = -1; + + // size terms for resident trips are dimensioned by income + if (purpose < 2) + { + segment = purpose * INCOME_SEGMENTS + income; + } else + { + segment = 2 * INCOME_SEGMENTS + purpose - 2; + } + return segment; + + } + + /** + * Calculate the purpose from the dc size segment. + * + * @param segment + * The dc size segment (0-17) + * @return The purpose + */ + public static int getPurposeFromDCSizeSegment(int segment) + { + + int purpose = -1; + + if (segment < INCOME_SEGMENTS) + { + purpose = 0; + } else if (segment < (AirportModelStructure.INCOME_SEGMENTS * 2)) + { + purpose = 1; + } else if (segment == (AirportModelStructure.INCOME_SEGMENTS * 2)) purpose = 2; + else purpose = 3; + + return purpose; + } + + /** + * Calculate the income from the dc size segment. + * + * @param segment + * The dc size segment (0-17) + * @return The income (defaults to 3 if not a resident purpose) + */ + public static int getIncomeFromDCSizeSegment(int segment) + { + + int income = 3; + + if (segment < AirportModelStructure.INCOME_SEGMENTS) + { + income = (byte) segment; + } else if (segment < (AirportModelStructure.INCOME_SEGMENTS * 2)) + income = ((byte) (segment - AirportModelStructure.INCOME_SEGMENTS)); + + return income; + } + + /** + * return the Skim period index 0=am, 1=pm, 2=off-peak + */ + public static int getSkimPeriodIndex(int departPeriod) + { + + int skimPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; + else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; + else skimPeriodIndex = OP; + + return skimPeriodIndex; + + } + + /** + * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV + */ + public static int getModelPeriodIndex(int departPeriod) + { + + int modelPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; + else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; + else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; + else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; + else modelPeriodIndex = 4; + + return modelPeriodIndex; + + } + + public static String getModelPeriodLabel(int period) + { + return MODEL_PERIOD_LABELS[period]; + } + + public static int getNumberModelPeriods() + { + return MODEL_PERIOD_LABELS.length; + } + + public static String getSkimMatrixPeriodString(int period) + { + int index = getSkimPeriodIndex(period); + return SKIM_PERIOD_STRINGS[index]; + } + + /** + * Get the tNCVehicle occupancy based upon the access mode and the party size. + * + * @param accessMode + * Access mode, 1-based, consistent with definitions above. + * @param partySize + * Number of passengers in travel party + * @return The (minimum) occupancy of the tNCVehicle trip to/from the airport. + */ + public static int getOccupancy(int accessMode, int partySize) + { + + switch (accessMode) + { + case PARK_TMNL: + return partySize; + case PARK_SANOFF: + return partySize; + case PARK_PVTOFF: + return partySize; + case PUDO_ESC: + return partySize + 1; + case PUDO_CURB: + return partySize + 1; + case RENTAL: + return partySize; + case TAXI: + return partySize + 1; + case TNC_SINGLE: + return partySize + 1; + case TNC_SHARED: + return partySize + 1; + case SHUTTLE_VAN: + return partySize + 1; + case TRANSIT: + return partySize; + + default: + throw new RuntimeException( + "Error: AccessMode not found in AirportModel.AirportModelStructure"); + + } + } + + public static boolean taxiTncMode(int accessMode) { + + switch (accessMode) { + case TAXI: + return true; + case TNC_SINGLE: + return true; + case TNC_SHARED: + return true; + } + + return false; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java new file mode 100644 index 0000000..4f96be2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java @@ -0,0 +1,321 @@ +package org.sandag.abm.airport; + +import java.io.Serializable; +import com.pb.common.math.MersenneTwister; + +public class AirportParty + implements Serializable +{ + + private MersenneTwister random; + private int ID; + + // following variables determined via simulation + private byte direction; + private byte purpose; + private byte size; + private byte income; + private int departTime; + private byte nights; + + private boolean debugChoiceModels; + + // following variables chosen via choice models + private int originMGRA; + private int destinationMGRA; + private int originTAZ; + private int destinationTAZ; + private byte mode; + private byte arrivalMode; + + private float valueOfTime; + + private int boardTap; + private int alightTap; + private int set = -1; + + private boolean avAvailable; + /** + * Public constructor. + * + * @param seed + * A seed for the random number generator. + */ + public AirportParty(long seed) + { + + random = new MersenneTwister(seed); + } + + /** + * @return the iD + */ + public int getID() + { + return ID; + } + + /** + * @param iD + * the iD to set + */ + public void setID(int iD) + { + ID = iD; + } + + /** + * @return the purpose + */ + public byte getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(byte purpose) + { + this.purpose = purpose; + } + + /** + * @return the size + */ + public byte getSize() + { + return size; + } + + /** + * @param size + * the size to set + */ + public void setSize(byte size) + { + this.size = size; + } + + /** + * @return the income + */ + public byte getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(byte income) + { + this.income = income; + } + + /** + * @return the departTime + */ + public int getDepartTime() + { + return departTime; + } + + /** + * @param departTime + * the departTime to set + */ + public void setDepartTime(int departTime) + { + this.departTime = departTime; + } + + /** + * @return the direction + */ + public byte getDirection() + { + return direction; + } + + /** + * @param direction + * the direction to set + */ + public void setDirection(byte direction) + { + this.direction = direction; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() + { + return originMGRA; + } + + /** + * @param originMGRA + * the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) + { + this.originMGRA = originMGRA; + } + + /** + * @return the trip mode + */ + public byte getMode() + { + return mode; + } + + /** + * @param mode + * the trip mode to set + */ + public void setMode(byte mode) + { + this.mode = mode; + } + + /** + * @return the arrivalMode + */ + public byte getArrivalMode() + { + return arrivalMode; + } + + /** + * @param arrivalMode + * the arrivalMode to set + */ + public void setArrivalMode(byte arrivalMode) + { + this.arrivalMode = arrivalMode; + } + + /** + * @return the nights + */ + public byte getNights() + { + return nights; + } + + /** + * @param nights + * the nights to set + */ + public void setNights(byte nights) + { + this.nights = nights; + } + + /** + * Get a random number from the parties random class. + * + * @return A random number. + */ + public double getRandom() + { + return random.nextDouble(); + } + + /** + * @return the debugChoiceModels + */ + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + /** + * @param debugChoiceModels + * the debugChoiceModels to set + */ + public void setDebugChoiceModels(boolean debugChoiceModels) + { + this.debugChoiceModels = debugChoiceModels; + } + + + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() + { + return destinationMGRA; + } + + /** + * @param destinationMGRA + * the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) + { + this.destinationMGRA = destinationMGRA; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public int getBoardTap() { + return boardTap; + } + + public void setBoardTap(int boardTap) { + this.boardTap = boardTap; + } + + public int getAlightTap() { + return alightTap; + } + + public void setAlightTap(int alightTap) { + this.alightTap = alightTap; + } + + public int getSet() { + return set; + } + + public void setSet(int set) { + this.set = set; + } + + public int getOriginTAZ() { + return originTAZ; + } + + public void setOriginTAZ(int originTAZ) { + this.originTAZ = originTAZ; + } + + public int getDestinationTAZ() { + return destinationTAZ; + } + + public void setDestinationTAZ(int destinationTAZ) { + this.destinationTAZ = destinationTAZ; + } + + public boolean getAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(boolean avAvailable) { + this.avAvailable = avAvailable; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java new file mode 100644 index 0000000..75d7794 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java @@ -0,0 +1,410 @@ +package org.sandag.abm.airport; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +public class AirportPartyManager +{ + + private static Logger logger = Logger.getLogger("SandagTourBasedModel.class"); + + private AirportParty[] parties; + + private double[] purposeDistribution; + private double[][] sizeDistribution; + private double[][] durationDistribution; + private double[][] incomeDistribution; + private double[][] departureDistribution; + private double[][] arrivalDistribution; + + + SandagModelStructure sandagStructure; + private String airportCode; + + private float avShare; + + + /** + * Constructor. Reads properties file and opens/stores all probability + * distributions for sampling. Estimates number of airport travel parties + * and initializes parties[]. + * + * @param resourceFile + * Property file. + */ + public AirportPartyManager(HashMap rbMap, float sampleRate, String airportCode) + { + sandagStructure = new SandagModelStructure(); + this.airportCode = airportCode; + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String purposeFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".purpose.file"); + String sizeFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".size.file"); + String durationFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".duration.file"); + String incomeFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".income.file"); + String departFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".departureTime.file"); + String arriveFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".arrivalTime.file"); + + // Read the distributions + setPurposeDistribution(purposeFile); + sizeDistribution = setDistribution(sizeDistribution, sizeFile); + durationDistribution = setDistribution(durationDistribution, durationFile); + incomeDistribution = setDistribution(incomeDistribution, incomeFile); + departureDistribution = setDistribution(departureDistribution, departFile); + arrivalDistribution = setDistribution(arrivalDistribution, arriveFile); + + // calculate total number of parties + float enplanements = new Float(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".enplanements").replace(",", "")); + float connectingPassengers = new Float(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".connecting").replace(",", "")); + float annualFactor = new Float(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".annualizationFactor")); + float averageSize = new Float(Util.getStringValueFromPropertyMap(rbMap, + "airport."+airportCode+".averageSize")); + + + avShare = Util.getFloatValueFromPropertyMap(rbMap, "Mobility.AV.Share"); + + float directPassengers = (enplanements - connectingPassengers) / annualFactor; + int totalParties = (int) (directPassengers / averageSize) * 2; + parties = new AirportParty[(int)(totalParties*sampleRate)]; + + logger.info("Total airport parties: " + totalParties); + } + + /** + * Create parties based upon total parties (calculated in constructor). Fill + * parties[] with travel parties, assuming one-half are arriving and + * one-half are departing. Simulate party characteristics (income, size, + * duration, time departing from origin or arriving at airport) from + * distributions, also read in during constructor. + * + */ + public void generateAirportParties() + { + + int departures = parties.length / 2; + int arrivals = parties.length - departures; + int totalParties = 0; + int totalPassengers = 0; + for (int i = 0; i < departures; ++i) + { + + AirportParty party = new AirportParty(i * 101 + 1000); + + // simulate from distributions + party.setDirection(AirportModelStructure.DEPARTURE); + byte purpose = (byte) choosePurpose(party.getRandom()); + byte size = (byte) chooseFromDistribution(purpose, sizeDistribution, party.getRandom()); + byte nights = (byte) chooseFromDistribution(purpose, durationDistribution, + party.getRandom()); + byte income = (byte) chooseFromDistribution(purpose, incomeDistribution, + party.getRandom()); + byte period = (byte) chooseFromDistribution(purpose, departureDistribution, + party.getRandom()); + + if(party.getRandom() random) return alt; + } + return -99; + } + + /** + * Choose a purpose. + * + * @param random + * A uniform random number. + * @return the purpose. + */ + protected int choosePurpose(double random) + { + // iterate through the probability array and choose + for (int alt = 0; alt < purposeDistribution.length; ++alt) + { + if (purposeDistribution[alt] > random) return alt; + } + return -99; + } + + /** + * Create a text file and write all records to the file. + * + */ + public void writeOutputFile(HashMap rbMap) + { + + // Open file and print header + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String fileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".output.file"); + + PrintWriter writer = null; + try + { + writer = new PrintWriter(new BufferedWriter(new FileWriter(fileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + fileName + " for writing\n"); + throw new RuntimeException(); + } + String headerString = new String( + "id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ," + + "destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); + writer.print(headerString); + + // Iterate through the array, printing records to the file + for (int i = 0; i < parties.length; ++i) + { + + String record = new String(parties[i].getID() + "," + parties[i].getDirection() + "," + + parties[i].getPurpose() + "," + parties[i].getSize() + "," + + parties[i].getIncome() + "," + parties[i].getNights() + "," + + parties[i].getDepartTime() + "," + parties[i].getOriginMGRA() + "," + + parties[i].getDestinationMGRA() + "," + + parties[i].getOriginTAZ() + "," + parties[i].getDestinationTAZ() + "," + + parties[i].getMode() + "," + + (parties[i].getAvAvailable() ? 1 : 0) + "," + + parties[i].getArrivalMode() + "," + parties[i].getBoardTap() + "," + + + parties[i].getAlightTap() + "," + parties[i].getSet() + "," + + String.format("%9.2f", parties[i].getValueOfTime()) + "\n"); + writer.print(record); + } + writer.close(); + + } + + /** + * @return the parties + */ + public AirportParty[] getParties() + { + return parties; + } + + + /* + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Airport Model Party Manager")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + AirportPartyManager apm = new AirportPartyManager(pMap); + + apm.generateAirportParties(); + + apm.writeOutputFile(pMap); + + logger.info("Airport Model successfully completed!"); + + } +*/ +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java new file mode 100644 index 0000000..b6fbab1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java @@ -0,0 +1,707 @@ +package org.sandag.abm.airport; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.crossborder.CrossBorderTripTables; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class AirportTripTables +{ + + private static Logger logger = Logger.getLogger("tripTables"); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TableDataSet tripData; + + // Some parameters + private int[] modeIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // total + // modes, + // returns + // 0=auto + // modes, + // 1=non-motor, + // 2=transit, + // 3= + // other + private int[] matrixIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // modes, + // returns + // the + // element + // of + // the + // matrix + // array + // to + // store + // value + + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private String[] purposeName = {"RES_BUS", "RES_PER", "VIS_BUS", + "VIS_PER" }; + private HashMap rbMap; + + // matrices are indexed by modes, vot bins, submodes + private Matrix[][][] matrix; + + private ResourceBundle rb; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + private String airportCode; + + private MatrixDataServerRmi ms; + private float sampleRate = 1; + private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; + private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; + private float valueOfTimeThresholdLow = 0; + private float valueOfTimeThresholdMed = 0; + //value of time bins by mode group + int[] votBins = {3,1,1,1}; + + public int numSkimSets; + + + public float getSampleRate() { + return sampleRate; + } + + public void setSampleRate(float sampleRate) { + this.sampleRate = sampleRate; + } + + public AirportTripTables(HashMap rbMap, String airportCode) + { + + this.rbMap = rbMap; + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTourModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + } else if (modelStructure.getTourModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + } else if (modelStructure.getTourModeIsWalkTransit(i) + || modelStructure.getTourModeIsDriveTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + } + } + //value of time thresholds + valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); + valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); + this.airportCode = airportCode; + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][][]; + for (int i = 0; i < numberOfModes; ++i) + { + + String modeName; + matrix[i] = new Matrix[votBins[i]][]; + + for(int j = 0; j< votBins[i];++j){ + if (i == 0) + { + matrix[i][j] = new Matrix[autoModes]; + for (int k = 0; k < autoModes; ++k) + { + modeName = modelStructure.getModeName(k + 1); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 1) + { + matrix[i][j] = new Matrix[nmotModes]; + for (int k = 0; k < nmotModes; ++k) + { + modeName = modelStructure.getModeName(k + 1 + autoModes); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 2) + { + matrix[i][j] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l1) + votBin = getValueOfTimeBin(valueOfTime); + + if (mode == 0) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); + } else if (mode == 1) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } else if (mode == 2) + { + + if (boardTap == 0 || alightTap == 0) continue; + + //store transit trips in matrices + mat = (matrixIndex[tripMode]*numSkimSets)+set; + float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); + matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); + + // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) + if (modelStructure.getTourModeIsDriveTransit(tripMode)) + { + + // add the tNCVehicle trip portion to the trip table + if (inbound == 0) + { // from origin to lot (boarding tap) + int PNRTAZ = tapManager.getTazForTap(boardTap); + value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); + matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); + + } else + { // from lot (alighting tap) to destination + int PNRTAZ = tapManager.getTazForTap(alightTap); + value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); + matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); + } + + } + } else + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } + + // generate another drive-alone trip in the opposite direction for + // pickup/dropoff + if (accMode == AirportModelStructure.PUDO_CURB + || accMode == AirportModelStructure.PUDO_CURB) + { + mode = 0; // auto mode + if (SandagModelStructure.getTripModeIsPay(tripMode)) // if the + // passenger + // chose + // pay, + // assume + // the + // driver + // will + // also + // pay + mat = 1; + else mat = 0; + float value = matrix[mode][votBin][mat].getValueAt(destinationTAZ, originTAZ); + matrix[mode][votBin][mat].setValueAt(destinationTAZ, originTAZ, (value + vehicleTrips)); + } + //logger.info("End creating trip tables for period " + timePeriod); + } + } + + /** + * Get the output trip table file names from the properties file, and write + * trip tables for all modes for the given time period. + * + * @param period + * Time period, which will be used to find the period time string + * to append to each trip table matrix file + */ + public void writeTrips(int period, MatrixType mt) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "scenario.path"); + String per = modelStructure.getModelPeriodLabel(period); + String[][] end = new String[4][]; + String[] fileName = new String[4]; + + fileName[0] = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.autoTripMatrix"); + fileName[1] = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.nMotTripMatrix"); + fileName[2] = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.tranTripMatrix"); + fileName[3] = directory + + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.othrTripMatrix"); + + //the end of the name depends on whether there are multiple vot bins or not + String[] votBinName = {"low","med","high"}; + + for(int i = 0; i<4;++i){ + end[i] = new String[votBins[i]]; + for(int j = 0; j < votBins[i];++j){ + if(votBins[i]>1) + end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; + else + end[i][j] = "_" + per + ".omx"; + } + } + for (int i = 0; i < 4; ++i){ + for(int j = 0; j < votBins[i];++j){ + try + { + //Delete the file if it exists + File f = new File(fileName[i]+end[i][j]); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); + f.delete(); + } + + if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); + else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); + } catch (Exception e) + { + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName[i] +end[i][j] + ", for mode index = " + i, e); + throw new RuntimeException(); + } + } + } + + + } + + /** + * Return the value of time bin 0 through 2 based on the thresholds provided in the property map + * @param valueOfTime + * @return value of time bin 0 through 2 + */ + public int getValueOfTimeBin(float valueOfTime){ + + if(valueOfTime pMap; + String propertiesFile = null; + String airportCode = null; + + logger.info(String.format( + "SANDAG Airport Model Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + + float sampleRate = 1.0f; + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-airport")) + { + airportCode = args[i + 1]; + } + } + + AirportTripTables tripTables = new AirportTripTables(pMap, airportCode); + logger.info("Airport Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration+" -airport "+airportCode); + + tripTables.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + tripTables.ms = matrixServer; + } else + { + tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + tripTables.ms.testRemote("AirportTripTables"); + + // mdm = MatrixDataManager.getInstance(); + // mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + tripTables.createTripTables(mt); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java new file mode 100644 index 0000000..74dff9b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java @@ -0,0 +1,615 @@ +package org.sandag.abm.application; + +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; +import java.util.HashMap; +import java.util.Iterator; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.TourModeChoiceDMU; + +import com.pb.common.calculator.IndexValues; + +public class SandagAppendMcLogsumDMU + extends TourModeChoiceDMU +{ + + private int departPeriod; + private int arrivePeriod; + + private int incomeInDollars; + private int adults; + private int autos; + private int hhSize; + private int personIsFemale; + private int age; + private int tourCategoryJoint; + private int tourCategoryEscort; + private int numberOfParticipantsInJointTour; + private int workTourModeIsHOV; + private int workTourModeIsSOV; + private int workTourModeIsBike; + private int tourCategorySubtour; + + public SandagAppendMcLogsumDMU(ModelStructure modelStructure, Logger aLogger) + { + super(modelStructure, aLogger); + setupMethodIndexMap(); + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + + public void setNonMotorizedWalkTimeOut(double walkTime) + { + nmWalkTimeOut = walkTime; + } + + public void setNonMotorizedWalkTimeIn(double walkTime) + { + nmWalkTimeIn = walkTime; + } + + public void setNonMotorizedBikeTimeOut(double bikeTime) + { + nmBikeTimeOut = bikeTime; + } + + public void setNonMotorizedBikeTimeIn(double bikeTime) + { + nmBikeTimeIn = bikeTime; + } + + public float getTimeOutbound() + { + return departPeriod; + } + + public float getTimeInbound() + { + return arrivePeriod; + } + + public void setDepartPeriod(int period) + { + departPeriod = period; + } + + public void setArrivePeriod(int period) + { + arrivePeriod = period; + } + + public void setHhSize(int arg) + { + hhSize = arg; + } + + public void setAge(int arg) + { + age = arg; + } + + public void setTourCategoryJoint(int arg) + { + tourCategoryJoint = arg; + } + + public void setTourCategoryEscort(int arg) + { + tourCategoryEscort = arg; + } + + public void setNumberOfParticipantsInJointTour(int arg) + { + numberOfParticipantsInJointTour = arg; + } + + public void setWorkTourModeIsSOV(int arg) + { + workTourModeIsSOV = arg; + } + + public void setWorkTourModeIsHOV(int arg) + { + workTourModeIsHOV = arg; + } + + public void setWorkTourModeIsBike(int arg) + { + workTourModeIsBike = arg; + } + + public void setPTazTerminalTime(float arg) + { + pTazTerminalTime = arg; + } + + public void setATazTerminalTime(float arg) + { + aTazTerminalTime = arg; + } + + public void setIncomeInDollars(int arg) + { + incomeInDollars = arg; + } + + public int getIncome() + { + return incomeInDollars; + } + + public void setAdults(int arg) + { + adults = arg; + } + + public int getAdults() + { + return adults; + } + + public void setAutos(int arg) + { + autos = arg; + } + + public int getAutos() + { + return autos; + } + + public int getAge() + { + return age; + } + + public int getHhSize() + { + return hhSize; + } + + public int getTourCategoryJoint() + { + return tourCategoryJoint; + } + + public int getTourCategoryEscort() + { + return tourCategoryEscort; + } + + public int getNumberOfParticipantsInJointTour() + { + return numberOfParticipantsInJointTour; + } + + public int getWorkTourModeIsSov() + { + return workTourModeIsSOV; + } + + public int getWorkTourModeIsHov() + { + return workTourModeIsHOV; + } + + public int getWorkTourModeIsBike() + { + return workTourModeIsBike; + } + + public void setPersonIsFemale(int arg) + { + personIsFemale = arg; + } + + public int getFemale() + { + return personIsFemale; + } + + public void setOrigDuDen(double arg) + { + origDuDen = arg; + } + + public void setOrigEmpDen(double arg) + { + origEmpDen = arg; + } + + public void setOrigTotInt(double arg) + { + origTotInt = arg; + } + + public void setDestDuDen(double arg) + { + destDuDen = arg; + } + + public void setDestEmpDen(double arg) + { + destEmpDen = arg; + } + + public void setDestTotInt(double arg) + { + destTotInt = arg; + } + + public double getODUDen() + { + return origDuDen; + } + + public double getOEmpDen() + { + return origEmpDen; + } + + public double getOTotInt() + { + return origTotInt; + } + + public double getDDUDen() + { + return destDuDen; + } + + public double getDEmpDen() + { + return destEmpDen; + } + + public double getDTotInt() + { + return destTotInt; + } + + public double getNm_walkTime_out() + { + return nmWalkTimeOut; + } + + public double getNm_walkTime_in() + { + return nmWalkTimeIn; + } + + public double getNm_bikeTime_out() + { + return nmBikeTimeOut; + } + + public double getNm_bikeTime_in() + { + return nmBikeTimeIn; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getIncome", 2); + methodIndexMap.put("getAdults", 3); + methodIndexMap.put("getFemale", 4); + methodIndexMap.put("getHhSize", 5); + methodIndexMap.put("getAutos", 6); + methodIndexMap.put("getAge", 7); + methodIndexMap.put("getTourCategoryJoint", 8); + methodIndexMap.put("getNumberOfParticipantsInJointTour", 9); + methodIndexMap.put("getWorkTourModeIsSov", 10); + methodIndexMap.put("getWorkTourModeIsBike", 11); + methodIndexMap.put("getWorkTourModeIsHov", 12); + methodIndexMap.put("getPTazTerminalTime", 14); + methodIndexMap.put("getATazTerminalTime", 15); + methodIndexMap.put("getODUDen", 16); + methodIndexMap.put("getOEmpDen", 17); + methodIndexMap.put("getOTotInt", 18); + methodIndexMap.put("getDDUDen", 19); + methodIndexMap.put("getDEmpDen", 20); + methodIndexMap.put("getDTotInt", 21); + methodIndexMap.put("getTourCategoryEscort", 22); + + methodIndexMap.put("getNm_walkTime_out", 90); + methodIndexMap.put("getNm_walkTime_in", 91); + methodIndexMap.put("getNm_bikeTime_out", 92); + methodIndexMap.put("getNm_bikeTime_in", 93); + methodIndexMap.put("getWtw_LB_ivt_out", 176); + methodIndexMap.put("getWtw_LB_ivt_in", 177); + methodIndexMap.put("getWtw_EB_ivt_out", 178); + methodIndexMap.put("getWtw_EB_ivt_in", 179); + methodIndexMap.put("getWtw_BRT_ivt_out", 180); + methodIndexMap.put("getWtw_BRT_ivt_in", 181); + methodIndexMap.put("getWtw_LRT_ivt_out", 182); + methodIndexMap.put("getWtw_LRT_ivt_in", 183); + methodIndexMap.put("getWtw_CR_ivt_out", 184); + methodIndexMap.put("getWtw_CR_ivt_in", 185); + methodIndexMap.put("getWtw_fwait_out", 186); + methodIndexMap.put("getWtw_fwait_in", 187); + methodIndexMap.put("getWtw_xwait_out", 188); + methodIndexMap.put("getWtw_xwait_in", 189); + methodIndexMap.put("getWtw_AccTime_out", 190); + methodIndexMap.put("getWtw_AccTime_in", 191); + methodIndexMap.put("getWtw_EgrTime_out", 192); + methodIndexMap.put("getWtw_EgrTime_in", 193); + methodIndexMap.put("getWtw_WalkAuxTime_out", 194); + methodIndexMap.put("getWtw_WalkAuxTime_in", 195); + methodIndexMap.put("getWtw_fare_out", 196); + methodIndexMap.put("getWtw_fare_in", 197); + methodIndexMap.put("getWtw_xfers_out", 198); + methodIndexMap.put("getWtw_xfers_in", 199); + + methodIndexMap.put("getWtd_LB_ivt_out", 276); + methodIndexMap.put("getWtd_LB_ivt_in", 277); + methodIndexMap.put("getWtd_EB_ivt_out", 278); + methodIndexMap.put("getWtd_EB_ivt_in", 279); + methodIndexMap.put("getWtd_BRT_ivt_out", 280); + methodIndexMap.put("getWtd_BRT_ivt_in", 281); + methodIndexMap.put("getWtd_LRT_ivt_out", 282); + methodIndexMap.put("getWtd_LRT_ivt_in", 283); + methodIndexMap.put("getWtd_CR_ivt_out", 284); + methodIndexMap.put("getWtd_CR_ivt_in", 285); + methodIndexMap.put("getWtd_fwait_out", 286); + methodIndexMap.put("getWtd_fwait_in", 287); + methodIndexMap.put("getWtd_xwait_out", 288); + methodIndexMap.put("getWtd_xwait_in", 289); + methodIndexMap.put("getWtd_AccTime_out", 290); + methodIndexMap.put("getWtd_AccTime_in", 291); + methodIndexMap.put("getWtd_EgrTime_out", 292); + methodIndexMap.put("getWtd_EgrTime_in", 293); + methodIndexMap.put("getWtd_WalkAuxTime_out", 294); + methodIndexMap.put("getWtd_WalkAuxTime_in", 295); + methodIndexMap.put("getWtd_fare_out", 296); + methodIndexMap.put("getWtd_fare_in", 297); + methodIndexMap.put("getWtd_xfers_out", 298); + methodIndexMap.put("getWtd_xfers_in", 299); + methodIndexMap.put("getDtw_LB_ivt_out", 376); + methodIndexMap.put("getDtw_LB_ivt_in", 377); + methodIndexMap.put("getDtw_EB_ivt_out", 378); + methodIndexMap.put("getDtw_EB_ivt_in", 379); + methodIndexMap.put("getDtw_BRT_ivt_out", 380); + methodIndexMap.put("getDtw_BRT_ivt_in", 381); + methodIndexMap.put("getDtw_LRT_ivt_out", 382); + methodIndexMap.put("getDtw_LRT_ivt_in", 383); + methodIndexMap.put("getDtw_CR_ivt_out", 384); + methodIndexMap.put("getDtw_CR_ivt_in", 385); + methodIndexMap.put("getDtw_fwait_out", 386); + methodIndexMap.put("getDtw_fwait_in", 387); + methodIndexMap.put("getDtw_xwait_out", 388); + methodIndexMap.put("getDtw_xwait_in", 389); + methodIndexMap.put("getDtw_AccTime_out", 390); + methodIndexMap.put("getDtw_AccTime_in", 391); + methodIndexMap.put("getDtw_EgrTime_out", 392); + methodIndexMap.put("getDtw_EgrTime_in", 393); + methodIndexMap.put("getDtw_WalkAuxTime_out", 394); + methodIndexMap.put("getDtw_WalkAuxTime_in", 395); + methodIndexMap.put("getDtw_fare_out", 396); + methodIndexMap.put("getDtw_fare_in", 397); + methodIndexMap.put("getDtw_xfers_out", 398); + methodIndexMap.put("getDtw_xfers_in", 399); + + + } + + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 2: + returnValue = getIncome(); + break; + case 3: + returnValue = getAdults(); + break; + case 4: + returnValue = getFemale(); + break; + case 5: + returnValue = getHhSize(); + break; + case 6: + returnValue = getAutos(); + break; + case 7: + returnValue = getAge(); + break; + case 8: + returnValue = getTourCategoryJoint(); + break; + case 9: + returnValue = getNumberOfParticipantsInJointTour(); + break; + case 10: + returnValue = getWorkTourModeIsSov(); + break; + case 11: + returnValue = getWorkTourModeIsBike(); + break; + case 12: + returnValue = getWorkTourModeIsHov(); + break; + case 14: + returnValue = getPTazTerminalTime(); + break; + case 15: + returnValue = getATazTerminalTime(); + break; + case 16: + returnValue = getODUDen(); + break; + case 17: + returnValue = getOEmpDen(); + break; + case 18: + returnValue = getOTotInt(); + break; + case 19: + returnValue = getDDUDen(); + break; + case 20: + returnValue = getDEmpDen(); + break; + case 21: + returnValue = getDTotInt(); + break; + case 22: + returnValue = getTourCategoryEscort(); + break; + case 90: + returnValue = getNm_walkTime_out(); + break; + case 91: + returnValue = getNm_walkTime_in(); + break; + case 92: + returnValue = getNm_bikeTime_out(); + break; + case 93: + returnValue = getNm_bikeTime_in(); + break; + /* TODO + case 176: + methodIndexMap.put("getWtw_LB_ivt_out", 176); + case 177: + methodIndexMap.put("getWtw_LB_ivt_in", 177); + case 178: + methodIndexMap.put("getWtw_EB_ivt_out", 178); + case 179: + methodIndexMap.put("getWtw_EB_ivt_in", 179); + case 180: + methodIndexMap.put("getWtw_BRT_ivt_out", 180); + case 181: + methodIndexMap.put("getWtw_BRT_ivt_in", 181); + case 182: + methodIndexMap.put("getWtw_LRT_ivt_out", 182); + case 183: + methodIndexMap.put("getWtw_LRT_ivt_in", 183); + methodIndexMap.put("getWtw_CR_ivt_out", 184); + methodIndexMap.put("getWtw_CR_ivt_in", 185); + methodIndexMap.put("getWtw_fwait_out", 186); + methodIndexMap.put("getWtw_fwait_in", 187); + methodIndexMap.put("getWtw_xwait_out", 188); + methodIndexMap.put("getWtw_xwait_in", 189); + methodIndexMap.put("getWtw_AccTime_out", 190); + methodIndexMap.put("getWtw_AccTime_in", 191); + methodIndexMap.put("getWtw_EgrTime_out", 192); + methodIndexMap.put("getWtw_EgrTime_in", 193); + methodIndexMap.put("getWtw_WalkAuxTime_out", 194); + methodIndexMap.put("getWtw_WalkAuxTime_in", 195); + methodIndexMap.put("getWtw_fare_out", 196); + methodIndexMap.put("getWtw_fare_in", 197); + methodIndexMap.put("getWtw_xfers_out", 198); + methodIndexMap.put("getWtw_xfers_in", 199); + + methodIndexMap.put("getWtd_LB_ivt_out", 276); + methodIndexMap.put("getWtd_LB_ivt_in", 277); + methodIndexMap.put("getWtd_EB_ivt_out", 278); + methodIndexMap.put("getWtd_EB_ivt_in", 279); + methodIndexMap.put("getWtd_BRT_ivt_out", 280); + methodIndexMap.put("getWtd_BRT_ivt_in", 281); + methodIndexMap.put("getWtd_LRT_ivt_out", 282); + methodIndexMap.put("getWtd_LRT_ivt_in", 283); + methodIndexMap.put("getWtd_CR_ivt_out", 284); + methodIndexMap.put("getWtd_CR_ivt_in", 285); + methodIndexMap.put("getWtd_fwait_out", 286); + methodIndexMap.put("getWtd_fwait_in", 287); + methodIndexMap.put("getWtd_xwait_out", 288); + methodIndexMap.put("getWtd_xwait_in", 289); + methodIndexMap.put("getWtd_AccTime_out", 290); + methodIndexMap.put("getWtd_AccTime_in", 291); + methodIndexMap.put("getWtd_EgrTime_out", 292); + methodIndexMap.put("getWtd_EgrTime_in", 293); + methodIndexMap.put("getWtd_WalkAuxTime_out", 294); + methodIndexMap.put("getWtd_WalkAuxTime_in", 295); + methodIndexMap.put("getWtd_fare_out", 296); + methodIndexMap.put("getWtd_fare_in", 297); + methodIndexMap.put("getWtd_xfers_out", 298); + methodIndexMap.put("getWtd_xfers_in", 299); + methodIndexMap.put("getDtw_LB_ivt_out", 376); + methodIndexMap.put("getDtw_LB_ivt_in", 377); + methodIndexMap.put("getDtw_EB_ivt_out", 378); + methodIndexMap.put("getDtw_EB_ivt_in", 379); + methodIndexMap.put("getDtw_BRT_ivt_out", 380); + methodIndexMap.put("getDtw_BRT_ivt_in", 381); + methodIndexMap.put("getDtw_LRT_ivt_out", 382); + methodIndexMap.put("getDtw_LRT_ivt_in", 383); + methodIndexMap.put("getDtw_CR_ivt_out", 384); + methodIndexMap.put("getDtw_CR_ivt_in", 385); + methodIndexMap.put("getDtw_fwait_out", 386); + methodIndexMap.put("getDtw_fwait_in", 387); + methodIndexMap.put("getDtw_xwait_out", 388); + methodIndexMap.put("getDtw_xwait_in", 389); + methodIndexMap.put("getDtw_AccTime_out", 390); + methodIndexMap.put("getDtw_AccTime_in", 391); + methodIndexMap.put("getDtw_EgrTime_out", 392); + methodIndexMap.put("getDtw_EgrTime_in", 393); + methodIndexMap.put("getDtw_WalkAuxTime_out", 394); + methodIndexMap.put("getDtw_WalkAuxTime_in", 395); + methodIndexMap.put("getDtw_fare_out", 396); + methodIndexMap.put("getDtw_fare_in", 397); + methodIndexMap.put("getDtw_xfers_out", 398); + methodIndexMap.put("getDtw_xfers_in", 399); + */ + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + return returnValue; + + } + + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java new file mode 100644 index 0000000..82ae86d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java @@ -0,0 +1,64 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.AtWorkSubtourFrequencyDMU; +import org.sandag.abm.ctramp.ModelStructure; +import com.pb.common.calculator.VariableTable; + +public class SandagAtWorkSubtourFrequencyDMU + extends AtWorkSubtourFrequencyDMU + implements VariableTable +{ + + public SandagAtWorkSubtourFrequencyDMU(ModelStructure modelStructure) + { + super(modelStructure); + this.modelStructure = modelStructure; + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getIncomeInDollars", 0); + methodIndexMap.put("getPersonType", 1); + methodIndexMap.put("getFemale", 2); + methodIndexMap.put("getDrivers", 3); + methodIndexMap.put("getNumPreschoolChildren", 4); + methodIndexMap.put("getNumIndivEatOutTours", 5); + methodIndexMap.put("getNumTotalTours", 6); + methodIndexMap.put("getNmEatOutAccessibilityWorkplace", 7); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getIncomeInDollars(); + case 1: + return getPersonType(); + case 2: + return getFemale(); + case 3: + return getDrivers(); + case 4: + return getNumPreschoolChildren(); + case 5: + return getNumIndivEatOutTours(); + case 6: + return getNumTotalTours(); + case 7: + return getNmEatOutAccessibilityWorkplace(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java new file mode 100644 index 0000000..3c25f8d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java @@ -0,0 +1,104 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.AutoOwnershipChoiceDMU; + +public class SandagAutoOwnershipChoiceDMU + extends AutoOwnershipChoiceDMU +{ + + public SandagAutoOwnershipChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getDrivers", 1); + methodIndexMap.put("getNumFtWorkers", 2); + methodIndexMap.put("getNumPtWorkers", 3); + methodIndexMap.put("getNumPersons18to24", 4); + methodIndexMap.put("getNumPersons6to15", 5); + methodIndexMap.put("getNumPersons80plus", 6); + methodIndexMap.put("getNumPersons65to79", 7); + methodIndexMap.put("getHhIncomeInDollars", 8); + methodIndexMap.put("getNumHighSchoolGraduates", 9); + methodIndexMap.put("getDetachedDwellingType", 10); + methodIndexMap.put("getUseAccessibilities", 11); + methodIndexMap.put("getHomeTazNonMotorizedAccessibility", 12); + methodIndexMap.put("getHomeTazAutoAccessibility", 13); + methodIndexMap.put("getHomeTazTransitAccessibility", 14); + methodIndexMap.put("getWorkAutoDependency", 15); + methodIndexMap.put("getSchoolAutoDependency", 16); + methodIndexMap.put("getWorkersRailProportion", 17); + methodIndexMap.put("getStudentsRailProportion", 18); + methodIndexMap.put("getGq", 19); + methodIndexMap.put("getNumPersons18to35", 25); + methodIndexMap.put("getNumPersons65plus", 26); + methodIndexMap.put("getWorkAutoTime", 27); + methodIndexMap.put("getHomeTazMaasAccessibility", 28); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getDrivers(); + case 2: + return getNumFtWorkers(); + case 3: + return getNumPtWorkers(); + case 4: + return getNumPersons18to24(); + case 5: + return getNumPersons6to15(); + case 6: + return getNumPersons80plus(); + case 7: + return getNumPersons65to79(); + case 8: + return getHhIncomeInDollars(); + case 9: + return getNumHighSchoolGraduates(); + case 10: + return getDetachedDwellingType(); + case 11: + return getUseAccessibilities(); + case 12: + return getHomeTazNonMotorizedAccessibility(); + case 13: + return getHomeTazAutoAccessibility(); + case 14: + return getHomeTazTransitAccessibility(); + case 15: + return getWorkAutoDependency(); + case 16: + return getSchoolAutoDependency(); + case 17: + return getWorkersRailProportion(); + case 18: + return getStudentsRailProportion(); + case 19: + return getGq(); + case 25: + return getNumPersons18to35(); + case 26: + return getNumPersons65Plus(); + case 27: + return getWorkAutoTime(); + case 28: + return getHomeTazMaasAccessibility(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java new file mode 100644 index 0000000..f69cfdc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java @@ -0,0 +1,176 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.CoordinatedDailyActivityPatternDMU; + +public class SandagCoordinatedDailyActivityPatternDMU + extends CoordinatedDailyActivityPatternDMU +{ + + public SandagCoordinatedDailyActivityPatternDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getFullTimeWorkerA", 0); + methodIndexMap.put("getFullTimeWorkerB", 1); + methodIndexMap.put("getFullTimeWorkerC", 2); + methodIndexMap.put("getPartTimeWorkerA", 3); + methodIndexMap.put("getPartTimeWorkerB", 4); + methodIndexMap.put("getPartTimeWorkerC", 5); + methodIndexMap.put("getUniversityStudentA", 6); + methodIndexMap.put("getUniversityStudentB", 7); + methodIndexMap.put("getUniversityStudentC", 8); + methodIndexMap.put("getNonWorkingAdultA", 9); + methodIndexMap.put("getNonWorkingAdultB", 10); + methodIndexMap.put("getNonWorkingAdultC", 11); + methodIndexMap.put("getRetiredA", 12); + methodIndexMap.put("getRetiredB", 13); + methodIndexMap.put("getRetiredC", 14); + methodIndexMap.put("getDrivingAgeSchoolChildA", 15); + methodIndexMap.put("getDrivingAgeSchoolChildB", 16); + methodIndexMap.put("getDrivingAgeSchoolChildC", 17); + methodIndexMap.put("getPreDrivingAgeSchoolChildA", 18); + methodIndexMap.put("getPreDrivingAgeSchoolChildB", 19); + methodIndexMap.put("getPreDrivingAgeSchoolChildC", 20); + methodIndexMap.put("getPreSchoolChildA", 21); + methodIndexMap.put("getPreSchoolChildB", 22); + methodIndexMap.put("getPreSchoolChildC", 23); + methodIndexMap.put("getAgeA", 24); + methodIndexMap.put("getFemaleA", 25); + methodIndexMap.put("getMoreCarsThanWorkers", 26); + methodIndexMap.put("getFewerCarsThanWorkers", 27); + methodIndexMap.put("getZeroCars", 28); + methodIndexMap.put("getHHIncomeInDollars", 29); + methodIndexMap.put("getHhDetach", 30); + methodIndexMap.put("getUsualWorkLocationIsHomeA", 31); + methodIndexMap.put("getNoUsualWorkLocationA", 32); + methodIndexMap.put("getNoUsualSchoolLocationA", 33); + methodIndexMap.put("getHhSize", 34); + methodIndexMap.put("getWorkLocationModeChoiceLogsumA", 35); + methodIndexMap.put("getSchoolLocationModeChoiceLogsumA", 36); + methodIndexMap.put("getRetailAccessibility", 37); + methodIndexMap.put("getNumAdultsWithNonMandatoryDap", 38); + methodIndexMap.put("getNumAdultsWithMandatoryDap", 39); + methodIndexMap.put("getNumKidsWithNonMandatoryDap", 40); + methodIndexMap.put("getNumKidsWithMandatoryDap", 41); + methodIndexMap.put("getAllAdultsAtHome", 42); + methodIndexMap.put("getWorkAccessForMandatoryDap", 43); + methodIndexMap.put("getTelecommuteFrequencyA", 44); + methodIndexMap.put("getTelecommuteFrequencyB", 45); + methodIndexMap.put("getTelecommuteFrequencyC", 46); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getFullTimeWorkerA(); + case 1: + return getFullTimeWorkerB(); + case 2: + return getFullTimeWorkerC(); + case 3: + return getPartTimeWorkerA(); + case 4: + return getPartTimeWorkerB(); + case 5: + return getPartTimeWorkerC(); + case 6: + return getUniversityStudentA(); + case 7: + return getUniversityStudentB(); + case 8: + return getUniversityStudentC(); + case 9: + return getNonWorkingAdultA(); + case 10: + return getNonWorkingAdultB(); + case 11: + return getNonWorkingAdultC(); + case 12: + return getRetiredA(); + case 13: + return getRetiredB(); + case 14: + return getRetiredC(); + case 15: + return getDrivingAgeSchoolChildA(); + case 16: + return getDrivingAgeSchoolChildB(); + case 17: + return getDrivingAgeSchoolChildC(); + case 18: + return getPreDrivingAgeSchoolChildA(); + case 19: + return getPreDrivingAgeSchoolChildB(); + case 20: + return getPreDrivingAgeSchoolChildC(); + case 21: + return getPreSchoolChildA(); + case 22: + return getPreSchoolChildB(); + case 23: + return getPreSchoolChildC(); + case 24: + return getAgeA(); + case 25: + return getFemaleA(); + case 26: + return getMoreCarsThanWorkers(); + case 27: + return getFewerCarsThanWorkers(); + case 28: + return getZeroCars(); + case 29: + return getHHIncomeInDollars(); + case 30: + return getHhDetach(); + case 31: + return getUsualWorkLocationIsHomeA(); + case 32: + return getNoUsualWorkLocationA(); + case 33: + return getNoUsualSchoolLocationA(); + case 34: + return getHhSize(); + case 35: + return getWorkLocationModeChoiceLogsumA(); + case 36: + return getSchoolLocationModeChoiceLogsumA(); + case 37: + return getRetailAccessibility(); + case 38: + return getNumAdultsWithNonMandatoryDap(); + case 39: + return getNumAdultsWithMandatoryDap(); + case 40: + return getNumKidsWithNonMandatoryDap(); + case 41: + return getNumKidsWithMandatoryDap(); + case 42: + return getAllAdultsAtHome(); + case 43: + return getWorkAccessForMandatoryDap(); + case 44: + return getTelecommuteFrequencyA(); + case 45: + return getTelecommuteFrequencyB(); + case 46: + return getTelecommuteFrequencyC(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java new file mode 100644 index 0000000..f329b78 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java @@ -0,0 +1,1055 @@ +package org.sandag.abm.application; + +import java.io.BufferedReader; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.IOException; +import java.util.HashMap; +import java.util.ResourceBundle; +import java.util.StringTokenizer; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.datafile.CSVFileWriter; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +public class SandagCreateTripGenerationFiles +{ + + private static Logger logger = Logger.getLogger(SandagCreateTripGenerationFiles.class); + + private static final String SANDAG_TRIP_GEN_FILE_KEY = "trip.model.trips.file"; + private static final String ABM_TRIP_GEN_FILE_KEY = "output.trips.file"; + private static final String ABM_INDIV_TRIP_FILE_KEY = "abm.individual.trip.file"; + private static final String TAZ_TDZ_CORRESP_KEY = "taz.tdz.corresp.file"; + private static final String SCALE_NHB_KEY = "scale.nhb"; + + private static final String TAZ_COLUMN_HEADING = "taz"; + private static final String TDZ_COLUMN_HEADING = "tdz"; + + private static final String TRIP_ORIG_PURPOSE_FIELD_NAME = "orig_purpose"; + private static final String TRIP_DEST_PURPOSE_FIELD_NAME = "dest_purpose"; + private static final String TRIP_ORIG_MGRA_FIELD_NAME = "orig_maz"; + private static final String TRIP_DEST_MGRA_FIELD_NAME = "dest_maz"; + private static final String[] HH_HEADINGS = { + TRIP_ORIG_PURPOSE_FIELD_NAME, TRIP_DEST_PURPOSE_FIELD_NAME, TRIP_ORIG_MGRA_FIELD_NAME, + TRIP_DEST_MGRA_FIELD_NAME }; + + private static final int MAX_PURPOSE_INDEX = 10; + + private static final int MIN_EXTERNAL_TDZ = 1; + private static final int MAX_EXTERNAL_TDZ = 12; + + private static final String TAZ_FIELD_HEADING = "zone"; + + private static final String[] TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS = {"a1", "a2", + "a3", "a4", "a5", "a8" }; + private static final String[] TRIP_MODEL_HOME_BASED_PRODUCTION_HEADINGS = {"p1", "p2", + "p3", "p4", "p5", "p8" }; + private static final String[] TRIP_MODEL_NON_HOME_BASED_ATTRACTION_HEADINGS = {"a6", "a7"}; + private static final String[] TRIP_MODEL_NON_HOME_BASED_PRODUCTION_HEADINGS = {"p6", "p7"}; + private static final String[] TRIP_MODEL_OTHER_BASED_PRODUCTION_HEADINGS = {"p9", "p10"}; + private static final String[] TRIP_MODEL_OTHER_BASED_ATTRACTION_HEADINGS = {"a9", "a10"}; + + private static final int[] AB_MODEL_HOME_BASED_PRODUCTION_INDICES = {1, 2, 3, 4, 5, 8}; + private static final int[] AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES = {6, 7}; + private static final int[] AB_MODEL_OTHER_PRODUCTION_INDICES = {9, 10}; + + private float[][] ieTrips; + + private static final String[] TABLE_HEADINGS = {"zone", "p1", + "p2", "p3", "p4", "p5", "p6", "p7", "p8", "p9", "p10", "a1", "a2", "a3", "a4", "a5", + "a6", "a7", "a8", "a9", "a10" }; + private static final String[] TABLE_HEADING_DESCRIPTIONS = {"zone", + "home based work", "home based university", "home based school", "home based shop", + "home based other", "non home based work related", "non home based other", + "home based escort", "home based visitor", "home based airport", "home based work", + "home based university", "home based school", "home based shop", "home based other", + "non home based work related", "non home based other", "home based escort", + "home based visitor", "home based airport" }; + + private MgraDataManager mgraManager; + private int maxTdz; + + public SandagCreateTripGenerationFiles(HashMap rbMap) + { + + mgraManager = MgraDataManager.getInstance(rbMap); + + } + + public void createTripGenFile(HashMap rbMap) + { + + String tgInputFile = rbMap.get(SANDAG_TRIP_GEN_FILE_KEY); + if (tgInputFile == null) + { + logger.error("Error getting the filename from the properties file for the input Sandag Trip Prods/Attrs by MDZ file."); + logger.error("Properties file target: " + SANDAG_TRIP_GEN_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + SANDAG_TRIP_GEN_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + String tgOutputFile = rbMap.get(ABM_TRIP_GEN_FILE_KEY); + if (tgOutputFile == null) + { + logger.error("Error getting the filename from the properties file to use for the new Trip Prods/Attrs by MDZ file created."); + logger.error("Properties file target: " + ABM_TRIP_GEN_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + ABM_TRIP_GEN_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + String abmTripFile = rbMap.get(ABM_INDIV_TRIP_FILE_KEY); + if (abmTripFile == null) + { + logger.error("Error getting the filename from the properties file to use for the ABM Model individual trips file."); + logger.error("Properties file target: " + ABM_INDIV_TRIP_FILE_KEY + " not found."); + logger.error("Please specify a filename value for the " + ABM_INDIV_TRIP_FILE_KEY + + " property."); + throw new RuntimeException(); + } + + String correspFile = rbMap.get(TAZ_TDZ_CORRESP_KEY); + if (correspFile == null) + { + logger.error("Error getting the filename from the properties file to use for the TAZ / TDZ correspondence file."); + logger.error("Properties file target: " + TAZ_TDZ_CORRESP_KEY + " not found."); + logger.error("Please specify a filename value for the " + TAZ_TDZ_CORRESP_KEY + + " property."); + throw new RuntimeException(); + } + + // default is false + boolean scaleNhbToAbm = false; + String scaleNhbToAbmString = rbMap.get(SCALE_NHB_KEY); + if (scaleNhbToAbmString != null && scaleNhbToAbmString.equalsIgnoreCase("true")) + scaleNhbToAbm = true; + logger.info("parameter to enable scaling NHB prods/attrs has a value of: " + scaleNhbToAbm); + + HashMap tazTdzMap = createTazTdzMap(correspFile); + + TableDataSet inTgTds = readInputTripGenFile(tgInputFile); + + int[][] tdzTrips = readInputAbmIndivTripFile(abmTripFile, tazTdzMap); + + TableDataSet outAbmTds = produceAbmTripTableDataSet(inTgTds, tdzTrips, scaleNhbToAbm); + + writeAbmTripGenFile(tgOutputFile, outAbmTds); + + logger.info(""); + logger.info(""); + logger.info("finished producing new trip generation files from the ABM trip data."); + } + + private TableDataSet readInputTripGenFile(String fileName) + { + + TableDataSet inTgTds = null; + + try + { + logger.info(""); + logger.info(""); + logger.info("reading input trip generation file."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + inTgTds = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading input trip generation data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + // create TDZ by purpose arrays for IE Prods and IE attrs from the trip + // model + // these will be added into the home-based AB model prods and attrs + // use the first dimension 0 element to accumulate totals by purpose for + // logging + ieTrips = new float[maxTdz + 1][2 * MAX_PURPOSE_INDEX + 1]; + for (int i = 0; i < inTgTds.getRowCount(); i++) + { + int tdz = (int) inTgTds.getValueAt(i + 1, TAZ_FIELD_HEADING); + for (int j = 1; j < inTgTds.getColumnCount(); j++) + { + if (tdz >= MIN_EXTERNAL_TDZ && tdz <= MAX_EXTERNAL_TDZ) + { + ieTrips[i + 1][j] = inTgTds.getValueAt(i + 1, j + 1); + ieTrips[0][j] += inTgTds.getValueAt(i + 1, j + 1); + } + } + } + + // log column totals + logger.info(""); + logger.info(""); + logger.info("\t" + inTgTds.getRowCount() + " rows in input file."); + logger.info("\t" + inTgTds.getColumnCount() + " columns in input file."); + logger.info(""); + logger.info(String.format("\t%-15s %-30s %15s %15s", "Column Name", "Column Purpose", + "Column Total", "Int-Ext")); + + String[] headings = inTgTds.getColumnLabels(); + logger.info(String.format("\t%-15s %-30s %15s %15s", headings[0], "N/A", "N/A", "N/A")); + float totProd = 0; + float totAttr = 0; + float totalIeProds = 0; + float totalIeAttrs = 0; + float columnSum = 0; + for (int i = 1; i < inTgTds.getColumnCount(); i++) + { + + columnSum = inTgTds.getColumnTotal(i + 1); + + // 1st 10 fields after zone are production fields, next 10 are + // attraction + // fields + if (i <= 10) + { + totProd += columnSum; + totalIeProds += ieTrips[0][i]; + } else + { + totAttr += columnSum; + totalIeAttrs += ieTrips[0][i]; + } + + logger.info(String.format("\t%-15s %-30s %15.1f %15.1f", headings[i], + TABLE_HEADING_DESCRIPTIONS[i], columnSum, ieTrips[0][i])); + + } + + logger.info(""); + logger.info(""); + logger.info(String.format("\ttotal productions = %15.1f", totProd)); + logger.info(String.format("\ttotal attractions = %15.1f", totAttr)); + logger.info(String.format("\ttotal IE productions = %12.1f", totalIeProds)); + logger.info(String.format("\ttotal IE attractions = %12.1f", totalIeAttrs)); + logger.info(""); + + return inTgTds; + } + + private HashMap createTazTdzMap(String correspFile) + { + + TableDataSet tazTdzTds = null; + + try + { + logger.info(""); + logger.info(""); + logger.info("reading input taz-tdz correspondence file."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + tazTdzTds = reader.readFile(new File(correspFile)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading input taz-tdz correspondence file: %s into TableDataSet object.", + correspFile)); + throw new RuntimeException(e); + } + + maxTdz = 0; + HashMap tazTdzMap = new HashMap(); + for (int r = 1; r <= tazTdzTds.getRowCount(); r++) + { + int taz = (int) tazTdzTds.getValueAt(r, TAZ_COLUMN_HEADING); + int tdz = (int) tazTdzTds.getValueAt(r, TDZ_COLUMN_HEADING); + tazTdzMap.put(taz, tdz); + + if (tdz > maxTdz) maxTdz = tdz; + } + + // a trip record with origin or destination mgra=0 or mgra=-1 (location + // not + // determined) should map to tdz=0 + // if a trip record mgra is not greater than 0, its taz will be 0 - then + // the + // following entry will map it to tdz=0. + tazTdzMap.put(0, 0); + + return tazTdzMap; + + } + + private int[][] readInputAbmIndivTripFile(String fileName, HashMap tazTdzMap) + { + + String origPurpose = ""; + String destPurpose = ""; + int origMgra = 0; + int destMgra = 0; + int homeTdz = 0; + int tripPurposeIndex = 0; + + // open the file for reading + String delimSet = ",\t\n\r\f\""; + BufferedReader inputStream = null; + try + { + inputStream = new BufferedReader(new FileReader(new File(fileName))); + } catch (FileNotFoundException e) + { + logger.fatal(String.format("Exception occurred reading input abm indiv trip file: %s.", + fileName)); + throw new RuntimeException(e); + } + + // first parse the trip file field names from the first record and + // associate + // column position with fields specified to be read + HashMap columnIndexHeadingMap = new HashMap(); + String line = ""; + try + { + line = inputStream.readLine(); + } catch (IOException e) + { + logger.fatal(String.format( + "Exception occurred reading header record of input abm indiv trip file: %s.", + fileName)); + logger.fatal(String.format("line = %s.", line)); + throw new RuntimeException(e); + } + StringTokenizer st = new StringTokenizer(line, delimSet); + int col = 0; + while (st.hasMoreTokens()) + { + String label = st.nextToken(); + for (String heading : HH_HEADINGS) + { + if (heading.equalsIgnoreCase(label)) + { + columnIndexHeadingMap.put(col, heading); + break; + } + } + col++; + } + + // dimension the array to hold trips summarized by tdz and trip model + // purpose + int[][] abmTdzTrips = new int[maxTdz + 1][MAX_PURPOSE_INDEX + 1]; + int[] abmTdzTotalTrips = new int[MAX_PURPOSE_INDEX + 1]; + + // read the trip records from the file + int lineCount = 0; + try + { + + while ((line = inputStream.readLine()) != null) + { + + lineCount++; + + // get the values for the fields specified. + col = 0; + st = new StringTokenizer(line, delimSet); + while (st.hasMoreTokens()) + { + String fieldValue = st.nextToken(); + if (columnIndexHeadingMap.containsKey(col)) + { + String fieldName = columnIndexHeadingMap.get(col++); + + if (fieldName.equalsIgnoreCase(TRIP_ORIG_PURPOSE_FIELD_NAME)) + { + origPurpose = fieldValue; + } else if (fieldName.equalsIgnoreCase(TRIP_DEST_PURPOSE_FIELD_NAME)) + { + destPurpose = fieldValue; + } else if (fieldName.equalsIgnoreCase(TRIP_ORIG_MGRA_FIELD_NAME)) + { + origMgra = Integer.parseInt(fieldValue); + } else if (fieldName.equalsIgnoreCase(TRIP_DEST_MGRA_FIELD_NAME)) + { + destMgra = Integer.parseInt(fieldValue); + + // don't need to process any more fields + break; + + } + + } else + { + col++; + } + + } + + int homeTaz = 0; + try + { + if (origPurpose.equalsIgnoreCase("Home") && origMgra > 0) homeTaz = mgraManager + .getTaz(origMgra); + else if (destPurpose.equalsIgnoreCase("Home") && destMgra > 0) + homeTaz = mgraManager.getTaz(destMgra); + } catch (Exception e) + { + logger.error("error getting home taz from mgraManager for origPurpose = " + + origPurpose + ", origMgra = " + origMgra + ", destPurpose = " + + destPurpose + ", destMgra = " + destMgra + ", lineCount = " + + lineCount); + throw new RuntimeException(e); + } + + try + { + homeTdz = tazTdzMap.get(homeTaz); + } catch (Exception e) + { + logger.error("error getting home tdz from tazTdzMap for homeTaz = " + homeTaz + + ", lineCount = " + lineCount); + throw new RuntimeException(e); + } + + try + { + // get the trip based model purpose index for this abm model + // trip + tripPurposeIndex = getTripModelPurposeForAbmTrip(origPurpose, destPurpose); + } catch (Exception e) + { + logger.error("error getting tripPurposeIndex for origPurpose = " + origPurpose + + ", destPurpose = " + destPurpose + ", lineCount = " + lineCount); + throw new RuntimeException(e); + } + + // accumulate trips in table + if (tripPurposeIndex >= 1 && tripPurposeIndex <= 5 || tripPurposeIndex == 8) + { + if (homeTdz > 0) abmTdzTrips[homeTdz][tripPurposeIndex]++; + else + { + logger.error("home tdz is le 0 for home-based trip."); + throw new RuntimeException(); + } + } else + { + if (homeTdz > 0) + { + logger.error("home tdz is gt 0 for non-home-based trip."); + throw new RuntimeException(); + } + abmTdzTrips[homeTdz][tripPurposeIndex]++; + } + abmTdzTotalTrips[tripPurposeIndex]++; + + } + + } catch (NumberFormatException e) + { + logger.fatal(String + .format("NumberFormatException occurred reading record of input abm indiv trip file: %s.", + fileName)); + logger.fatal(String.format("last record number read = %d.", lineCount)); + } catch (IOException e) + { + logger.fatal(String.format( + "IOException occurred reading record of input abm indiv trip file: %s.", + fileName)); + logger.fatal(String.format("last record number read = %d.", lineCount)); + } + + logger.info(lineCount + " trip records read from " + fileName); + + // log a summary report of trips by trip model purpose + logger.info(""); + logger.info(""); + logger.info("ABM Trip file trips by TM purpose"); + logger.info(String.format("\t%-15s %-30s %15s", "Column Name", "Column Purpose", + "Column Total")); + String[] headings = {"", "hbw", "hbu", "hbc", "hbs", "hbo", "nhw", "nho", "hbp"}; + int total = 0; + for (int i = 1; i < headings.length; i++) + { + logger.info(String.format("\t%-15s %-30s %15d", headings[i], + TABLE_HEADING_DESCRIPTIONS[i], abmTdzTotalTrips[i])); + total += abmTdzTotalTrips[i]; + } + logger.info(String.format("\t%-15s %-30s %15d", "Total", "", total)); + + return abmTdzTrips; + + } + + private int getTripModelPurposeForAbmTrip(String origPurpose, String destPurpose) + { + + /* + * assignment rules: replace tpurp4s = 1 if orig_purpose=="Home" & + * dest_purpose=="Work"; replace tpurp4s = 1 if orig_purpose=="Work" & + * dest_purpose=="Home"; replace tpurp4s = 2 if orig_purpose=="Home" & + * dest_purpose=="University"; replace tpurp4s = 2 if + * orig_purpose=="University" & dest_purpose=="Home"; replace tpurp4s = + * 3 if orig_purpose=="Home" & dest_purpose=="School"; replace tpurp4s = + * 3 if orig_purpose=="School" & dest_purpose=="Home"; replace tpurp4s = + * 4 if orig_purpose=="Home" & dest_purpose=="Shop"; replace tpurp4s = 4 + * if orig_purpose=="Shop" & dest_purpose=="Home"; replace tpurp4s = 5 + * if orig_purpose=="Home" & (dest_purpose=="Maintenance" | + * dest_purpose=="Eating Out" | dest_purpose=="Visiting" | + * dest_purpose=="Discretionary"); replace tpurp4s = 5 if + * dest_purpose=="Home" & (orig_purpose=="Maintenance" | + * orig_purpose=="Eating Out" | orig_purpose=="Visiting" | + * orig_purpose=="Discretionary"); replace tpurp4s = 8 if + * orig_purpose=="Home" & dest_purpose=="Escort"; replace tpurp4s = 8 if + * orig_purpose=="Escort" & dest_purpose=="Home"; replace tpurp4s = 6 if + * orig_purpose=="Work" & dest_purpose!="Home"; replace tpurp4s = 6 if + * orig_purpose!="Home" & dest_purpose=="Work"; replace tpurp4s = 6 if + * orig_purpose=="Work-Based" | dest_purpose=="Work-Based"; replace + * tpurp4s = 7 if tpurp4s==0; + */ + + int tripPurposeIndex = 0; + if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Work")) tripPurposeIndex = 1; + else if (origPurpose.equalsIgnoreCase("Work") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 1; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("University")) tripPurposeIndex = 2; + else if (origPurpose.equalsIgnoreCase("University") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 2; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("School")) tripPurposeIndex = 3; + else if (origPurpose.equalsIgnoreCase("School") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 3; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Shop")) tripPurposeIndex = 4; + else if (origPurpose.equalsIgnoreCase("Shop") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 4; + else if (origPurpose.equalsIgnoreCase("Home") + && destPurpose.equalsIgnoreCase("Maintenance")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Maintenance") + && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Eating Out")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Eating Out") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Visiting")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Visiting") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Home") + && destPurpose.equalsIgnoreCase("Discretionary")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Discretionary") + && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Home") + && destPurpose.equalsIgnoreCase("Work Related")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Work Related") + && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; + else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Escort")) tripPurposeIndex = 8; + else if (origPurpose.equalsIgnoreCase("Escort") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 8; + else if (origPurpose.equalsIgnoreCase("Work") && (!destPurpose.equalsIgnoreCase("Home"))) tripPurposeIndex = 6; + else if ((!destPurpose.equalsIgnoreCase("Home")) && destPurpose.equalsIgnoreCase("Work")) tripPurposeIndex = 6; + else if (origPurpose.equalsIgnoreCase("Work-Based") + || destPurpose.equalsIgnoreCase("Work-Based")) tripPurposeIndex = 6; + else tripPurposeIndex = 7; + + return tripPurposeIndex; + + } + + private TableDataSet produceAbmTripTableDataSet(TableDataSet inTgTds, int[][] tdzTrips, + boolean scaleNhbToAbm) + { + + float[][] newTrips = new float[maxTdz][2 * MAX_PURPOSE_INDEX + 1]; + + saveAbmHbProdsAndScaleTmAttrs(inTgTds, tdzTrips, newTrips); + + saveTmNhbProdsAsAbmProds(inTgTds, tdzTrips, scaleNhbToAbm, newTrips); + + saveTmNhbAttrsAsAbmProds(inTgTds, tdzTrips, scaleNhbToAbm, newTrips); + + addTmIeAttrsToAbmHbProds(inTgTds, tdzTrips, newTrips); + + addTmIeProdsToAbmHbAttrs(inTgTds, tdzTrips, newTrips); + + addTmAirportAndVisitorProdsToAbmProds(inTgTds, tdzTrips, newTrips); + + addTmAirportAndVisitorAttrsToAbmAttrs(inTgTds, tdzTrips, newTrips); + + saveZoneField(inTgTds, newTrips); + + TableDataSet abmTds = createFinalTableDataset(newTrips); + + return abmTds; + } + + private void saveAbmHbProdsAndScaleTmAttrs(TableDataSet inTgTds, int[][] tdzTrips, + float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("transferring ABM home-based productions and scaling TM attractions:"); + logger.info(String.format("%10s %-30s %15s %15s %15s %15s %15s", "TM Heading", + "TM Purpose", "TM Attrs", "ABM Attrs", "Scale Factor", "New ABM Prods", + "New ABM Attrs")); + int index = 0; + for (String heading : TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS) + { + + // get the trip model attractions, the TableDataSet returns a 0s + // based + // array + float[] values = inTgTds.getColumnAsFloat(heading); + + // add up the total TM attractions for this purpose + float total = 0; + for (int i = 0; i < values.length; i++) + total += values[i]; + + // add up the total ABM productions for this purpose + int abTotal = 0; + int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; + for (int i = 1; i < tdzTrips.length; i++) + abTotal += tdzTrips[i][abProdIndex]; + + // get the scale factor to scale trip model atractions to ABM + // productions + // by purpose + double scaleFactor = 0.0; + if (total > 0) + { + scaleFactor = abTotal / total; + } else + { + logger.error("attempting to scale an array which sums to 0.0."); + throw new RuntimeException(); + } + + // get the scaled attractions for the purpose + double[] scaledAttrs = getScaledValues(values, scaleFactor); + + // determine the final array column index into which to store the + // scaled + // attractions + int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; + + // save the scaled attractions in the final array + float abmAttrs = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][abAttrIndex] = (float) scaledAttrs[i]; + abmAttrs += newTrips[i][abAttrIndex]; + } + + // save the ABM productions in the final array + float abmProds = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][abProdIndex] = tdzTrips[i + 1][abProdIndex]; + abmProds += newTrips[i][abProdIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f %15.1f", + heading, TABLE_HEADING_DESCRIPTIONS[abAttrIndex], total, abTotal, scaleFactor, + abmProds, abmAttrs)); + + index++; + + } + } + + private void saveTmNhbProdsAsAbmProds(TableDataSet inTgTds, int[][] tdzTrips, + boolean scaleNhbToAbm, float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("non-home-based TM productions to total ABM productions:"); + logger.info(String.format("%10s %-30s %15s %15s %15s %15s", "TM Heading", + "TM Purpose", "TM Prods", "ABM Prods", "Scale Factor", "New ABM Prods")); + int index = 0; + for (String heading : TRIP_MODEL_NON_HOME_BASED_PRODUCTION_HEADINGS) + { + + // get the trip model nhb productions, the TableDataSet returns a 0s + // based array + float[] values = inTgTds.getColumnAsFloat(heading); + + // add up total TM productions by purpose + float total = 0; + for (int i = 0; i < values.length; i++) + total += values[i]; + + // get the total ab model productions for this purpose + int abProdIndex = AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES[index]; + int abTotal = tdzTrips[0][abProdIndex]; + + // get the scale factor to scale trip model productions to ABM + // productions by purpose + double scaleFactor = 0.0; + if (scaleNhbToAbm) + { + if (total > 0) + { + scaleFactor = abTotal / total; + } else + { + logger.error("attempting to scale an array which sums to 0.0."); + throw new RuntimeException(); + } + } else + { + scaleFactor = 1.0; + } + + // get the scaled productions for the purpose + double[] scaledProds = getScaledValues(values, scaleFactor); + + // save the scaled attractions in the final array + float abmProds = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][abProdIndex] = (float) scaledProds[i]; + abmProds += newTrips[i][abProdIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[abProdIndex], total, abTotal, scaleFactor, abmProds)); + + index++; + + } + } + + private void saveTmNhbAttrsAsAbmProds(TableDataSet inTgTds, int[][] tdzTrips, + boolean scaleNhbToAbm, float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("non-home-based TM attractions to total ABM productions:"); + logger.info(String.format("%10s %-30s %15s %15s %15s %15s", "TM Heading", + "TM Purpose", "TM Attrs", "ABM Prods", "Scale Factor", "New ABM Attrs")); + int index = 0; + for (String heading : TRIP_MODEL_NON_HOME_BASED_ATTRACTION_HEADINGS) + { + + // get the trip model nhb attractions, the TableDataSet returns a 0s + // based array + float[] values = inTgTds.getColumnAsFloat(heading); + + // add up total TM attracctions by purpose + float total = 0; + for (int i = 0; i < values.length; i++) + total += values[i]; + + // get the total ab model productions for this purpose + int abProdIndex = AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES[index]; + int abTotal = tdzTrips[0][abProdIndex]; + + // get the scale factor to scale trip model atractions to ABM + // productions + // by purpose + double scaleFactor = 0.0; + if (scaleNhbToAbm) + { + if (total > 0) + { + scaleFactor = abTotal / total; + } else + { + logger.error("attempting to scale an array which sums to 0.0."); + throw new RuntimeException(); + } + } else + { + scaleFactor = 1.0; + } + + // get the scaled attractions for the purpose + double[] scaledAttrs = getScaledValues(values, scaleFactor); + + // determine the final array column index into which to store the + // scaled + // attractions + int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; + + // save the scaled attractions in the final array + float abmAttrs = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][abAttrIndex] = (float) scaledAttrs[i]; + abmAttrs += newTrips[i][abAttrIndex]; + } + + index++; + + logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[abAttrIndex], total, abTotal, scaleFactor, abmAttrs)); + + } + } + + private void addTmIeAttrsToAbmHbProds(TableDataSet inTgTds, int[][] tdzTrips, float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("adding IE attrs from trip model to home-based ABM prods:"); + logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", + "ABM Prods", "IE attrs", "new ABM Prods")); + int index = 0; + for (String heading : TRIP_MODEL_HOME_BASED_PRODUCTION_HEADINGS) + { + + int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; + int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; + + // save the new ABM productions in the final array + float abTotal = 0; + float newTotal = 0; + for (int i = 0; i < newTrips.length; i++) + { + abTotal += newTrips[i][abProdIndex]; + newTrips[i][abProdIndex] += ieTrips[i + 1][abAttrIndex]; + newTotal += newTrips[i][abProdIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15.1f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[abProdIndex], abTotal, ieTrips[0][abAttrIndex], + newTotal)); + + index++; + + } + } + + private void addTmIeProdsToAbmHbAttrs(TableDataSet inTgTds, int[][] tdzTrips, float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("adding IE prods from trip model to home-based ABM attrs:"); + logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", + "ABM Attrs", "IE prods", "new ABM Attrs")); + int index = 0; + for (String heading : TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS) + { + + int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; + int abAttrIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index] + MAX_PURPOSE_INDEX; + + // save the new ABM attractions in the final array + float abTotal = 0; + float newTotal = 0; + for (int i = 0; i < newTrips.length; i++) + { + abTotal += newTrips[i][abAttrIndex]; + newTrips[i][abAttrIndex] += ieTrips[i + 1][abProdIndex]; + newTotal += newTrips[i][abAttrIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15.1f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[abAttrIndex], abTotal, ieTrips[0][abProdIndex], + newTotal)); + + index++; + + } + } + + private void addTmAirportAndVisitorProdsToAbmProds(TableDataSet inTgTds, int[][] tdzTrips, + float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("airport and visitor TM to ABM productions:"); + logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", + "TM Prods", "Scale Factor", "New ABM Prods")); + int index = 0; + for (String heading : TRIP_MODEL_OTHER_BASED_PRODUCTION_HEADINGS) + { + + // get the trip model productions + float[] pValues = inTgTds.getColumnAsFloat(heading); + + // determine the final array column index into which to store the + // scaled + // attractions + int prodIndex = AB_MODEL_OTHER_PRODUCTION_INDICES[index]; + + // save the productions in the final array + // add up the original values + float pTotal = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][prodIndex] = pValues[i]; + pTotal += newTrips[i][prodIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15.6f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[prodIndex], pTotal, 1.0, pTotal)); + + index++; + + } + } + + private void addTmAirportAndVisitorAttrsToAbmAttrs(TableDataSet inTgTds, int[][] tdzTrips, + float[][] newTrips) + { + logger.info(""); + logger.info(""); + logger.info("airport and visitor TM to ABM attractions:"); + logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", + "TM Attrs", "Scale Factor", "New ABM Attrs")); + int index = 0; + for (String heading : TRIP_MODEL_OTHER_BASED_ATTRACTION_HEADINGS) + { + + // get the trip model attractions + float[] aValues = inTgTds.getColumnAsFloat(heading); + + // determine the final array column index into which to store the + // scaled + // attractions + int attrIndex = AB_MODEL_OTHER_PRODUCTION_INDICES[index] + MAX_PURPOSE_INDEX; + + // save the attractions in the final array + float aTotal = 0; + for (int i = 0; i < newTrips.length; i++) + { + newTrips[i][attrIndex] = aValues[i]; + aTotal += newTrips[i][attrIndex]; + } + + logger.info(String.format("%10s %-30s %15.1f %15.6f %15.1f", heading, + TABLE_HEADING_DESCRIPTIONS[attrIndex], aTotal, 1.0, aTotal)); + + index++; + + } + } + + private void saveZoneField(TableDataSet inTgTds, float[][] newTrips) + { + // save the zone field in final table + float[] values = inTgTds.getColumnAsFloat(TAZ_FIELD_HEADING); + int tazFieldIndex = inTgTds.getColumnPosition(TAZ_FIELD_HEADING) - 1; + for (int i = 0; i < newTrips.length; i++) + newTrips[i][tazFieldIndex] = values[i]; + } + + private TableDataSet createFinalTableDataset(float[][] newTrips) + { + TableDataSet abmTds = TableDataSet.create(newTrips, TABLE_HEADINGS); + + float[] newIeTrips = new float[2 * MAX_PURPOSE_INDEX + 1]; + for (int i = 0; i < abmTds.getRowCount(); i++) + { + int tdz = (int) abmTds.getValueAt(i + 1, TAZ_FIELD_HEADING); + for (int j = 1; j < abmTds.getColumnCount(); j++) + { + if (tdz >= MIN_EXTERNAL_TDZ && tdz <= MAX_EXTERNAL_TDZ) + { + newIeTrips[j] += abmTds.getValueAt(i + 1, j + 1); + } + } + } + + logger.info(""); + logger.info(""); + logger.info("summary of newly created trip generation file."); + logger.info("\t" + abmTds.getRowCount() + " rows in output file."); + logger.info("\t" + abmTds.getColumnCount() + " columns in output file."); + logger.info(""); + logger.info(String.format("\t%-15s %-30s %15s %15s", "Column Name", "Column Purpose", + "Column Total", "Int-Ext")); + + String[] headings = abmTds.getColumnLabels(); + logger.info(String.format("\t%-15s %-30s %15s %15s", headings[0], "N/A", "N/A", "N/A")); + float totProd = 0; + float totAttr = 0; + float totalIeProds = 0; + float totalIeAttrs = 0; + float columnSum = 0; + for (int i = 1; i < abmTds.getColumnCount(); i++) + { + + columnSum = abmTds.getColumnTotal(i + 1); + + // 1st 10 fields after zone are production fields, next 10 are + // attraction + // fields + if (i <= 10) + { + totProd += columnSum; + totalIeProds += newIeTrips[i]; + } else + { + totAttr += columnSum; + totalIeAttrs += newIeTrips[i]; + } + + logger.info(String.format("\t%-15s %-30s %15.1f %15.1f", headings[i], + TABLE_HEADING_DESCRIPTIONS[i], columnSum, newIeTrips[i])); + + } + + logger.info(""); + logger.info(""); + logger.info(String.format("\ttotal productions = %15.1f", totProd)); + logger.info(String.format("\ttotal attractions = %15.1f", totAttr)); + logger.info(String.format("\ttotal IE productions = %12.1f", totalIeProds)); + logger.info(String.format("\ttotal IE attractions = %12.1f", totalIeAttrs)); + logger.info(""); + + return abmTds; + } + + private void writeAbmTripGenFile(String tgOutputFile, TableDataSet outAbmTds) + { + + CSVFileWriter writer = new CSVFileWriter(); + try + { + writer.writeFile(outAbmTds, new File(tgOutputFile)); + } catch (IOException e) + { + logger.fatal(String + .format("Exception occurred writing new trip generation data file = %s from TableDataSet object.", + tgOutputFile)); + throw new RuntimeException(e); + } + } + + private double[] getScaledValues(float[] values, double scaleFactor) + { + + double[] scaledValues = new double[values.length]; + for (int i = 0; i < values.length; i++) + scaledValues[i] = values[i] * scaleFactor; + + return scaledValues; + } + + public static void main(String[] args) throws Exception + { + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else + { + + String baseName; + if (args[0].endsWith(".properties")) + { + int index = args[0].indexOf(".properties"); + baseName = args[0].substring(0, index); + } else + { + baseName = args[0]; + } + + ResourceBundle rb = ResourceBundle.getBundle(baseName); + HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + SandagCreateTripGenerationFiles mainObject = new SandagCreateTripGenerationFiles(rbMap); + + // pass true as an argument if NHB trips from the trip model are to + // be + // scaled to the number from the activity-based model + mainObject.createTripGenFile(rbMap); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java new file mode 100644 index 0000000..8dad984 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java @@ -0,0 +1,23 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import java.util.ResourceBundle; +import org.sandag.abm.ctramp.CtrampApplication; + +public class SandagCtrampApplication + extends CtrampApplication +{ + + public static final String PROGRAM_VERSION = "09June2008"; + public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; + + public SandagCtrampApplication(ResourceBundle rb, HashMap rbMap, + boolean calculateLandUseAccessibilities) + { + super(rb, rbMap, calculateLandUseAccessibilities); + + projectDirectory = rbMap.get(PROPERTIES_PROJECT_DIRECTORY); + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java new file mode 100644 index 0000000..c473901 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java @@ -0,0 +1,170 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.application; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Map; + +import org.sandag.abm.ctramp.AtWorkSubtourFrequencyDMU; +import org.sandag.abm.ctramp.AutoOwnershipChoiceDMU; +import org.sandag.abm.ctramp.BikeLogsum; +import org.sandag.abm.ctramp.CoordinatedDailyActivityPatternDMU; +import org.sandag.abm.ctramp.CtrampDmuFactoryIf; +import org.sandag.abm.ctramp.DcSoaDMU; +import org.sandag.abm.ctramp.DestChoiceDMU; +import org.sandag.abm.ctramp.DestChoiceTwoStageModelDMU; +import org.sandag.abm.ctramp.DestChoiceTwoStageSoaTazDistanceUtilityDMU; +import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyDMU; +import org.sandag.abm.ctramp.IndividualNonMandatoryTourFrequencyDMU; +import org.sandag.abm.ctramp.InternalExternalTripChoiceDMU; +import org.sandag.abm.ctramp.JointTourModelsDMU; +import org.sandag.abm.ctramp.MicromobilityChoiceDMU; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.ParkingChoiceDMU; +import org.sandag.abm.ctramp.ParkingProvisionChoiceDMU; +import org.sandag.abm.ctramp.StopFrequencyDMU; +import org.sandag.abm.ctramp.StopLocationDMU; +import org.sandag.abm.ctramp.TelecommuteDMU; +import org.sandag.abm.ctramp.TourDepartureTimeAndDurationDMU; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import org.sandag.abm.ctramp.TransponderChoiceDMU; +import org.sandag.abm.ctramp.TripModeChoiceDMU; + +/** + * ArcCtrampDmuFactory is a class that ... + * + * @author Kimberly Grommes + * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. + */ +public class SandagCtrampDmuFactory + implements CtrampDmuFactoryIf, Serializable +{ + + private ModelStructure modelStructure; + private Map propertyMap; + + public SandagCtrampDmuFactory(ModelStructure modelStructure, Map propertyMap) + { + this.modelStructure = modelStructure; + this.propertyMap = propertyMap; + } + + public AutoOwnershipChoiceDMU getAutoOwnershipDMU() + { + return new SandagAutoOwnershipChoiceDMU(); + } + + public TransponderChoiceDMU getTransponderChoiceDMU() + { + return new SandagTransponderChoiceDMU(); + } + + public TelecommuteDMU getTelecommuteDMU() + { + return new SandagTelecommuteDMU(); + } + + public InternalExternalTripChoiceDMU getInternalExternalTripChoiceDMU() + { + return new SandagInternalExternalTripChoiceDMU(); + } + + public ParkingProvisionChoiceDMU getFreeParkingChoiceDMU() + { + return new SandagParkingProvisionChoiceDMU(); + } + + public CoordinatedDailyActivityPatternDMU getCoordinatedDailyActivityPatternDMU() + { + return new SandagCoordinatedDailyActivityPatternDMU(); + } + + public DcSoaDMU getDcSoaDMU() + { + return new SandagDcSoaDMU(); + } + + public DestChoiceDMU getDestChoiceDMU() + { + return new SandagDestChoiceDMU(modelStructure); + } + + public DestChoiceTwoStageModelDMU getDestChoiceSoaTwoStageDMU() + { + return new SandagDestChoiceSoaTwoStageModelDMU(modelStructure); + } + + public DestChoiceTwoStageSoaTazDistanceUtilityDMU getDestChoiceSoaTwoStageTazDistUtilityDMU() + { + return new SandagDestChoiceSoaTwoStageTazDistUtilityDMU(); + } + + public TourModeChoiceDMU getModeChoiceDMU() + { + SandagTourModeChoiceDMU dmu = new SandagTourModeChoiceDMU(modelStructure,null); + dmu.setBikeLogsum(BikeLogsum.getBikeLogsum(propertyMap)); + return dmu; + } + + public IndividualMandatoryTourFrequencyDMU getIndividualMandatoryTourFrequencyDMU() + { + return new SandagIndividualMandatoryTourFrequencyDMU(); + } + + public TourDepartureTimeAndDurationDMU getTourDepartureTimeAndDurationDMU() + { + return new SandagTourDepartureTimeAndDurationDMU(modelStructure); + } + + public AtWorkSubtourFrequencyDMU getAtWorkSubtourFrequencyDMU() + { + return new SandagAtWorkSubtourFrequencyDMU(modelStructure); + } + + public JointTourModelsDMU getJointTourModelsDMU() + { + return new SandagJointTourModelsDMU(modelStructure); + } + + public IndividualNonMandatoryTourFrequencyDMU getIndividualNonMandatoryTourFrequencyDMU() + { + return new SandagIndividualNonMandatoryTourFrequencyDMU(); + } + + public StopFrequencyDMU getStopFrequencyDMU() + { + return new SandagStopFrequencyDMU(modelStructure); + } + + public StopLocationDMU getStopLocationDMU() + { + return new SandagStopLocationDMU(modelStructure,propertyMap); + } + + public TripModeChoiceDMU getTripModeChoiceDMU() + { + SandagTripModeChoiceDMU dmu = new SandagTripModeChoiceDMU(modelStructure,null); + dmu.setBikeLogsum(BikeLogsum.getBikeLogsum(propertyMap)); + return dmu; + } + + public ParkingChoiceDMU getParkingChoiceDMU() + { + return new SandagParkingChoiceDMU(); + } + + public MicromobilityChoiceDMU getMicromobilityChoiceDMU() + { + return new SandagMicromobilityChoiceDMU(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java new file mode 100644 index 0000000..8e17074 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java @@ -0,0 +1,89 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.DcSoaDMU; + +public class SandagDcSoaDMU + extends DcSoaDMU +{ + + public SandagDcSoaDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getLnDcSizeAlt", 0); + methodIndexMap.put("getOriginToMgraDistanceAlt", 1); + methodIndexMap.put("getTourPurposeIsEscort", 2); + methodIndexMap.put("getNumPreschool", 3); + methodIndexMap.put("getNumGradeSchoolStudents", 4); + methodIndexMap.put("getNumHighSchoolStudents", 5); + methodIndexMap.put("getDcSizeAlt", 6); + methodIndexMap.put("getHouseholdsDestAlt", 8); + methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 9); + methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 10); + methodIndexMap.put("getGradeSchoolDistrictDestAlt", 11); + methodIndexMap.put("getHomeMgraGradeSchoolDistrict", 12); + methodIndexMap.put("getHighSchoolDistrictDestAlt", 14); + methodIndexMap.put("getHomeMgraHighSchoolDistrict", 15); + methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); + //methodIndexMap.put("getHomeMgra", 17); + + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + public double getLnDcSizeAlt(int alt) + { + return getLnDcSize(alt); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getLnDcSizeAlt(arrayIndex); + case 1: + return getOriginToMgraDistanceAlt(arrayIndex); + case 2: + return getTourPurposeIsEscort(); + case 3: + return getNumPreschool(); + case 4: + return getNumGradeSchoolStudents(); + case 5: + return getNumHighSchoolStudents(); + case 6: + return getDcSizeAlt(arrayIndex); + case 8: + return getHouseholdsDestAlt(arrayIndex); + case 9: + return getGradeSchoolEnrollmentDestAlt(arrayIndex); + case 10: + return getHighSchoolEnrollmentDestAlt(arrayIndex); + case 11: + return getGradeSchoolDistrictDestAlt(arrayIndex); + case 12: + return getHomeMgraGradeSchoolDistrict(); + case 14: + return getHighSchoolDistrictDestAlt(arrayIndex); + case 15: + return getHomeMgraHighSchoolDistrict(); + case 16: + return getUniversityEnrollmentDestAlt(arrayIndex); + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java new file mode 100644 index 0000000..163c6b6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java @@ -0,0 +1,161 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.DestChoiceDMU; +import org.sandag.abm.ctramp.ModelStructure; + +public class SandagDestChoiceDMU + extends DestChoiceDMU +{ + + public SandagDestChoiceDMU(ModelStructure modelStructure) + { + super(modelStructure); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getMcLogsumDestAlt", 3); + methodIndexMap.put("getNumGradeSchoolStudents", 4); + methodIndexMap.put("getNumHighSchoolStudents", 5); + methodIndexMap.put("getHouseholdsDestAlt", 8); + methodIndexMap.put("getPopulationDestAlt", 9); + methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 10); + methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 11); + methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); + methodIndexMap.put("getPersonIsWorker", 20); + methodIndexMap.put("getPersonHasBachelors", 21); + methodIndexMap.put("getPersonType", 22); + methodIndexMap.put("getSubtourType", 23); + methodIndexMap.put("getDcSoaCorrectionsAlt", 24); + methodIndexMap.put("getNumberOfNonWorkingAdults", 25); + methodIndexMap.put("getNumPreschool", 26); + methodIndexMap.put("getFemale", 27); + methodIndexMap.put("getIncome", 28); + methodIndexMap.put("getFemaleWorker", 29); + methodIndexMap.put("getIncomeInDollars", 30); + methodIndexMap.put("getAutos", 31); + methodIndexMap.put("getWorkers", 32); + methodIndexMap.put("getNumChildrenUnder16", 33); + methodIndexMap.put("getNumChildrenUnder19", 34); + methodIndexMap.put("getAge", 35); + methodIndexMap.put("getFullTimeWorker", 36); + methodIndexMap.put("getWorkTaz", 37); + methodIndexMap.put("getWorkTourModeIsSOV", 38); + methodIndexMap.put("getTourIsJoint", 39); + methodIndexMap.put("getOpSovDistanceAlt", 42); + methodIndexMap.put("getLnDcSizeAlt", 43); + methodIndexMap.put("getWorkAccessibility", 44); + methodIndexMap.put("getNonMandatoryAccessibilityAlt", 45); + methodIndexMap.put("getToursLeft", 46); + methodIndexMap.put("getMaxWindow", 47); + methodIndexMap.put("getDcSizeAlt", 48); + } + + public void setMcLogsum(int mgra, double logsum) + { + modeChoiceLogsums[mgra] = logsum; + } + + public double getLogsumDestAlt(int alt) + { + return getMcLogsumDestAlt(alt); + } + + public int getPersonIsFullTimeWorker() + { + return person.getPersonIsFullTimeWorker(); + } + + /* + * public int getSubtourType() { if ( + * tour.getTourCategory().equalsIgnoreCase( ModelStructure.AT_WORK_CATEGORY + * ) ) return tour.getTourPurposeIndex(); else return 0; } + */ + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 3: + return getMcLogsumDestAlt(arrayIndex); + case 4: + return getNumGradeSchoolStudents(); + case 5: + return getNumHighSchoolStudents(); + case 8: + return getHouseholdsDestAlt(arrayIndex); + case 9: + return getPopulationDestAlt(arrayIndex); + case 10: + return getGradeSchoolEnrollmentDestAlt(arrayIndex); + case 11: + return getHighSchoolEnrollmentDestAlt(arrayIndex); + case 16: + return getUniversityEnrollmentDestAlt(arrayIndex); + case 20: + return getPersonIsWorker(); + case 21: + return getPersonHasBachelors(); + case 22: + return getPersonType(); + case 24: + return getDcSoaCorrectionsAlt(arrayIndex); + case 25: + return getNumberOfNonWorkingAdults(); + case 26: + return getNumPreschool(); + case 27: + return getFemale(); + case 28: + return getIncome(); + case 29: + return getFemaleWorker(); + case 30: + return getIncomeInDollars(); + case 31: + return getAutos(); + case 32: + return getWorkers(); + case 33: + return getNumChildrenUnder16(); + case 34: + return getNumChildrenUnder19(); + case 35: + return getAge(); + case 36: + return getFullTimeWorker(); + case 37: + return getWorkTaz(); + case 38: + return getWorkTourModeIsSOV(); + case 39: + return getTourIsJoint(); + case 42: + return getOpSovDistanceAlt(arrayIndex); + case 43: + return getLnDcSizeAlt(arrayIndex); + case 44: + return getWorkAccessibility(); + case 45: + return getNonMandatoryAccessibilityAlt(arrayIndex); + case 46: + return getToursLeftCount(); + case 47: + return getMaxContinuousAvailableWindow(); + case 48: + return getDcSizeAlt(arrayIndex); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java new file mode 100644 index 0000000..aae0533 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java @@ -0,0 +1,158 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.DestChoiceTwoStageModelDMU; +import org.sandag.abm.ctramp.ModelStructure; + +public class SandagDestChoiceSoaTwoStageModelDMU + extends DestChoiceTwoStageModelDMU +{ + + public SandagDestChoiceSoaTwoStageModelDMU(ModelStructure modelStructure) + { + super(modelStructure); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getMcLogsumDestAlt", 3); + methodIndexMap.put("getNumGradeSchoolStudents", 4); + methodIndexMap.put("getNumHighSchoolStudents", 5); + methodIndexMap.put("getHouseholdsDestAlt", 8); + methodIndexMap.put("getPopulationDestAlt", 9); + methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 10); + methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 11); + methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); + methodIndexMap.put("getPersonIsWorker", 20); + methodIndexMap.put("getPersonHasBachelors", 21); + methodIndexMap.put("getPersonType", 22); + methodIndexMap.put("getSubtourType", 23); + methodIndexMap.put("getDcSoaCorrectionsAlt", 24); + methodIndexMap.put("getNumberOfNonWorkingAdults", 25); + methodIndexMap.put("getNumPreschool", 26); + methodIndexMap.put("getFemale", 27); + methodIndexMap.put("getIncome", 28); + methodIndexMap.put("getFemaleWorker", 29); + methodIndexMap.put("getIncomeInDollars", 30); + methodIndexMap.put("getAutos", 31); + methodIndexMap.put("getWorkers", 32); + methodIndexMap.put("getNumChildrenUnder16", 33); + methodIndexMap.put("getNumChildrenUnder19", 34); + methodIndexMap.put("getAge", 35); + methodIndexMap.put("getFullTimeWorker", 36); + methodIndexMap.put("getWorkTaz", 37); + methodIndexMap.put("getWorkTourModeIsSOV", 38); + methodIndexMap.put("getTourIsJoint", 39); + methodIndexMap.put("getOpSovDistanceAlt", 42); + methodIndexMap.put("getLnDcSizeAlt", 43); + methodIndexMap.put("getWorkAccessibility", 44); + methodIndexMap.put("getNonMandatoryAccessibilityAlt", 45); + methodIndexMap.put("getToursLeft", 46); + methodIndexMap.put("getMaxWindow", 47); + } + + public void setMcLogsum(int sampleIndex, double logsum) + { + modeChoiceLogsums[sampleIndex] = logsum; + } + + public double getLogsumDestAlt(int alt) + { + return getMcLogsumDestAlt(alt); + } + + public int getPersonIsFullTimeWorker() + { + return person.getPersonIsFullTimeWorker(); + } + + /* + * public int getSubtourType() { if ( + * tour.getTourCategory().equalsIgnoreCase( ModelStructure.AT_WORK_CATEGORY + * ) ) return tour.getTourPurposeIndex(); else return 0; } + */ + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 3: + return getMcLogsumDestAlt(arrayIndex); + case 4: + return getNumGradeSchoolStudents(); + case 5: + return getNumHighSchoolStudents(); + case 8: + return getHouseholdsDestAlt(arrayIndex); + case 9: + return getPopulationDestAlt(arrayIndex); + case 10: + return getGradeSchoolEnrollmentDestAlt(arrayIndex); + case 11: + return getHighSchoolEnrollmentDestAlt(arrayIndex); + case 16: + return getUniversityEnrollmentDestAlt(arrayIndex); + case 20: + return getPersonIsWorker(); + case 21: + return getPersonHasBachelors(); + case 22: + return getPersonType(); + case 24: + return getDcSoaCorrectionsAlt(arrayIndex); + case 25: + return getNumberOfNonWorkingAdults(); + case 26: + return getNumPreschool(); + case 27: + return getFemale(); + case 28: + return getIncome(); + case 29: + return getFemaleWorker(); + case 30: + return getIncomeInDollars(); + case 31: + return getAutos(); + case 32: + return getWorkers(); + case 33: + return getNumChildrenUnder16(); + case 34: + return getNumChildrenUnder19(); + case 35: + return getAge(); + case 36: + return getFullTimeWorker(); + case 37: + return getWorkTaz(); + case 38: + return getWorkTourModeIsSOV(); + case 39: + return getTourIsJoint(); + case 42: + return getOpSovDistanceAlt(arrayIndex); + case 43: + return getLnDcSizeAlt(arrayIndex); + case 44: + return getWorkAccessibility(); + case 45: + return getNonMandatoryAccessibilityAlt(arrayIndex); + case 46: + return getToursLeftCount(); + case 47: + return getMaxContinuousAvailableWindow(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java new file mode 100644 index 0000000..62e1688 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java @@ -0,0 +1,66 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.DestChoiceTwoStageSoaTazDistanceUtilityDMU; + +public class SandagDestChoiceSoaTwoStageTazDistUtilityDMU + extends DestChoiceTwoStageSoaTazDistanceUtilityDMU +{ + + public SandagDestChoiceSoaTwoStageTazDistUtilityDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getLnDestChoiceSizeTazAlt", 0); + methodIndexMap.put("getSizeTazAlt", 1); + methodIndexMap.put("getUniversityEnrollmentTazAlt", 2); + methodIndexMap.put("getGradeSchoolDistrictTazAlt", 3); + methodIndexMap.put("getHighSchoolDistrictTazAlt", 4); + methodIndexMap.put("getHomeTazGradeSchoolDistrict", 5); + methodIndexMap.put("getHomeTazHighSchoolDistrict", 6); + methodIndexMap.put("getGradeSchoolEnrollmentTazAlt", 7); + methodIndexMap.put("getHighSchoolEnrollmentTazAlt", 8); + methodIndexMap.put("getHouseholdsTazAlt", 9); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getLnDestChoiceSizeTazAlt(arrayIndex); + case 1: + return getSizeTazAlt(arrayIndex); + case 2: + return getUniversityEnrollmentTazAlt(arrayIndex); + case 3: + return getGradeSchoolDistrictTazAlt(arrayIndex); + case 4: + return getHighSchoolDistrictTazAlt(arrayIndex); + case 5: + return getHomeTazGradeSchoolDistrict(); + case 6: + return getHomeTazHighSchoolDistrict(); + case 7: + return getGradeSchoolEnrollmentTazAlt(arrayIndex); + case 8: + return getHighSchoolEnrollmentTazAlt(arrayIndex); + case 9: + return getHouseholdsTazAlt(arrayIndex); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java new file mode 100644 index 0000000..805df60 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java @@ -0,0 +1,624 @@ +package org.sandag.abm.application; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileReader; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.HouseholdDataManager; +import org.sandag.abm.ctramp.Person; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * @author Jim Hicks + * + * Class for managing household and person object data read from + * synthetic population files. + */ +public class SandagHouseholdDataManager + extends HouseholdDataManager +{ + + public static final String HH_DATA_SERVER_NAME = SandagHouseholdDataManager.class + .getCanonicalName(); + public static final String HH_DATA_SERVER_ADDRESS = "127.0.0.1"; + public static final int HH_DATA_SERVER_PORT = 1139; + + public static final String PROPERTIES_OCCUP_CODES = "PopulationSynthesizer.OccupCodes"; + public static final String PROPERTIES_INDUSTRY_CODES = "PopulationSynthesizer.IndustryCodes"; + + public SandagHouseholdDataManager() + { + super(); + } + + /** + * Associate data in hh and person TableDataSets read from synthetic + * population files with Household objects and Person objects with + * Households. + * + */ + public void mapTablesToHouseholdObjects() + { + + logger.info("mapping popsyn household and person data records to objects."); + + int id = -1; + Household[] hhArray = new Household[hhTable.getRowCount()]; + + int invalidPersonTypeCount1 = 0; + int invalidPersonTypeCount2 = 0; + int invalidPersonTypeCount3 = 0; + + // read the corrrespondence files for mapping persons to occupation and + int[] occCodes = readOccupCorrespondenceData(); + int[] indCodes = readIndustryCorrespondenceData(); + + // get the maximum HH id value to use to dimension the hhIndex + // correspondence + // array. + // the hhIndex array will store the hhArray index number for the given + // hh + // index. + int maxHhId = 0; + for (int r = 1; r <= hhTable.getRowCount(); r++) + { + id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); + if (id > maxHhId) maxHhId = id; + } + hhIndexArray = new int[maxHhId + 1]; + int[] sortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); + + // for each household table record + for (int r = 1; r <= hhTable.getRowCount(); r++) + { + + try + { + + // create a Household object + Household hh = new Household(modelStructure); + + // get required values from table record and store in Household + // object + id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); + hh.setHhId(id, inputRandomSeed); + + // set the household in the hhIndexArray in random order + int index = sortedIndices[r - 1]; + hhIndexArray[hh.getHhId()] = index; + + int htaz = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); + hh.setHhTaz(htaz); + + int hmgra = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); + hh.setHhMgra(hmgra); + + double rn = hh.getHhRandom().nextDouble(); + int origWalkSubzone = getInitialOriginWalkSegment(htaz, rn); + hh.setHhWalkSubzone(origWalkSubzone); + + // autos could be modeled or from PUMA + int numAutos = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_AUTOS_FIELD_NAME)); + hh.setHhAutos(numAutos); + + // set the hhSize variable and create Person objects for each + // person + int numPersons = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_SIZE_FIELD_NAME)); + hh.setHhSize(numPersons); + + int numWorkers = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_WORKERS_FIELD_NAME)); + hh.setHhWorkers(numWorkers); + + int incomeCat = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_INCOME_CATEGORY_FIELD_NAME)); + hh.setHhIncomeCategory(incomeCat); + + int incomeInDollars = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_INCOME_DOLLARS_FIELD_NAME)); + hh.setHhIncomeInDollars(incomeInDollars); + + // 0=Housing unit, 1=Institutional group quarters, + // 2=Noninstitutional + // group quarters + int unitType = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_UNITTYPE_FIELD_NAME)); + hh.setUnitType(unitType); + + // 1=Family household:married-couple, 2=Family household:male + // householder,no wife present, 3=Family household:female + // householder,no + // husband present + // 4=Nonfamily household:male householder, living alone, + // 5=Nonfamily + // household:male householder, not living alone, + // 6=Nonfamily household:female householder, living alone, + // 7=Nonfamily household:female householder, not living alone + int type = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_TYPE_FIELD_NAME)); + hh.setHhType(type); + + // 1=mobile home, 2=one-family house detached from any other + // house, + // 3=one-family house attached to one or more houses, + // 4=building with 2 apartments, 5=building with 3 or 4 + // apartments, + // 6=building with 5 to 9 apartments, + // 7=building with 10 to 19 apartments, 8=building with 20 to 49 + // apartments, + // 9=building with 50 or more apartments, 10=Boat,RV,van,etc. + int bldgsz = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_BLDGSZ_FIELD_NAME)); + hh.setHhBldgsz(bldgsz); + + hh.initializeWindows(); + hhArray[index] = hh; + + } catch (Exception e) + { + + logger.fatal(String + .format("exception caught mapping household data record to a Household object, r=%d, id=%d.", + r, id)); + throw new RuntimeException(e); + + } + + } + + int hhid = -1; + int oldHhid = -1; + int i = -1; + int persNum = -1; + int persId = -1; + int fieldCount = 0; + + // for each person table record + for (int r = 1; r <= personTable.getRowCount(); r++) + { + + try + { + + // get the Household object for this person data to be stored in + hhid = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); + int index = hhIndexArray[hhid]; + Household hh = hhArray[index]; + fieldCount = 1; + + if (oldHhid < hhid) + { + oldHhid = hhid; + persNum = 1; + } + + // get the Person object for this person data to be stored in + persId = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_PERSON_ID_FIELD_NAME)); + Person person = hh.getPerson(persNum++); + person.setPersId(persId); + fieldCount++; + + // get required values from table record and store in Person + // object + int age = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_AGE_FIELD_NAME)); + person.setPersAge(age); + fieldCount++; + + int gender = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_GENDER_FIELD_NAME)); + person.setPersGender(gender); + fieldCount++; + + int occcen1 = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_OCCCEN1_FIELD_NAME)); + int pecasOccup = occCodes[occcen1]; + + if (pecasOccup == 0) logger.warn("pecasOccup==0 for occcen1=" + occcen1); + + int indcen = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_INDCEN_FIELD_NAME)); + int activityCode = indCodes[indcen]; + + if ((pecasOccup == 71) + && (activityCode == 2 || activityCode == 4 || activityCode == 6 + || activityCode == 8 || activityCode == 29)) activityCode++; + + if ((pecasOccup == 76) + && (activityCode == 3 || activityCode == 5 || activityCode == 7 + || activityCode == 9 || activityCode == 30)) activityCode--; + + if ((pecasOccup == 76) && (activityCode == 13)) activityCode = 14; + + if ((pecasOccup == 71) && (activityCode == 14)) activityCode = 13; + + if ((pecasOccup == 75) && (activityCode == 18)) activityCode = 22; + + if ((pecasOccup == 71) && (activityCode == 22)) activityCode = 18; + + if (activityCode == 28) pecasOccup = 77; + + person.setPersActivityCode(activityCode); + fieldCount++; + + person.setPersPecasOccup(pecasOccup); + fieldCount++; + + // Employment status (1-employed FT, 2-employed PT, 3-not + // employed, + // 4-under age 16) + int empCat = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME)); + person.setPersEmploymentCategory(empCat); + fieldCount++; + + // Student status (1 - student in grade or high school; 2 - + // student + // in college or higher; 3 - not a student) + int studentCat = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_STUDENT_CATEGORY_FIELD_NAME)); + person.setPersStudentCategory(studentCat); + fieldCount++; + + // Person type (1-FT worker age 16+, 2-PT worker nonstudent age + // 16+, + // 3-university student, 4-nonworker nonstudent age 16-64, + // 5-nonworker nonstudent age 65+, + // 6-"age 16-19 student, not FT wrkr or univ stud", 7-age 6-15 + // schpred, 8 under age 6 presch + int personType = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_TYPE_CATEGORY_FIELD_NAME)); + person.setPersonTypeCategory(personType); + fieldCount++; + + // Person educational attainment level to determine high school + // graduate status ( < 9 - not a graduate, 10+ - high school + // graduate + // and + // beyond) + int educ = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_EDUCATION_ATTAINMENT_FIELD_NAME)); + if (educ >= 9) person.setPersonIsHighSchoolGraduate(true); + else person.setPersonIsHighSchoolGraduate(false); + fieldCount++; + + // Person educational attainment level to determine higher + // education + // status ( > 12 - at least a bachelor's degree ) + if (educ >= 13) person.setPersonHasBachelors(true); + else person.setPersonHasBachelors(false); + fieldCount++; + + // Person grade enrolled in ( 0-"not enrolled", 1-"preschool", + // 2-"Kindergarten", 3-"Grade 1 to grade 4", + // 4-"Grade 5 to grade 8", 5-"Grade 9 to grade 12", + // 6-"College undergraduate", + // 7-"Graduate or professional school" ) + int grade = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_GRADE_ENROLLED_FIELD_NAME)); + person.setPersonIsGradeSchool(false); + person.setPersonIsHighSchool(false); + if (grade >= 2 && grade <= 4) person.setPersonIsGradeSchool(true); + else if (grade == 5) person.setPersonIsHighSchool(true); + fieldCount++; + + // if person is a university student but has school age student + // category value, reset student category value + if (personType == Person.PersonType.University_student.ordinal() + && studentCat != Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal()) + { + studentCat = Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount1++; + // if person is a student of any kind but has full-time + // employment + // status, reset student category value to non-student + } else if (studentCat != Person.StudentStatus.NON_STUDENT.ordinal() + && empCat == Person.EmployStatus.FULL_TIME.ordinal()) + { + studentCat = Person.StudentStatus.NON_STUDENT.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount2++; + } + fieldCount++; + + // check consistency of student category and person type + if (studentCat == Person.StudentStatus.NON_STUDENT.ordinal()) + { + + if (person.getPersonIsStudentNonDriving() == 1 + || person.getPersonIsStudentDriving() == 1) + { + studentCat = Person.StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount3++; + } + + } + fieldCount++; + + } catch (Exception e) + { + + logger.fatal("exception caught mapping person data record to a Person object, " + + String.format( + "r=%d, i=%d, hhid=%d, persid=%d, persnum=%d, fieldCount=%d.", r, i, + hhid, persId, persNum, fieldCount)); + throw new RuntimeException(e); + + } + + } // person loop + + hhs = hhArray; + + logger.warn(invalidPersonTypeCount1 + + " person type = university and student category = non-student person records" + + " had their student category changed to university or higher."); + logger.warn(invalidPersonTypeCount2 + + " Student category = student and employment category = full-time worker person records" + + " had their student category changed to non-student."); + logger.warn(invalidPersonTypeCount3 + + " Student category = non-student and person type = student person records" + + " had their student category changed to student high school or less."); + + } + + /** + * if called, must be called after readData so that the size of the full + * population is known. + * + * @param hhFileName + * @param persFileName + * @param numHhs + */ + public void createSamplePopulationFiles(String hhFileName, String persFileName, + String newHhFileName, String newPersFileName, int numHhs) + { + + int maximumHhId = 0; + for (int i = 0; i < hhs.length; i++) + { + int id = hhs[i].getHhId(); + if (id > maximumHhId) maximumHhId = id; + } + + int[] testHhs = new int[maximumHhId + 1]; + + int[] sortedIndices = getRandomOrderHhIndexArray(hhs.length); + + for (int i = 0; i < numHhs; i++) + { + int k = sortedIndices[i]; + int hhId = hhs[k].getHhId(); + testHhs[hhId] = 1; + } + + String hString = ""; + int hCount = 0; + try + { + + logger.info(String.format("writing sample household file for %d households", numHhs)); + + PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newHhFileName))); + BufferedReader in = new BufferedReader(new FileReader(hhFileName)); + + // read headers and write to output files + hString = in.readLine(); + out.write(hString + "\n"); + hCount++; + int count = 0; + + while ((hString = in.readLine()) != null) + { + hCount++; + int endOfField = hString.indexOf(','); + int hhId = Integer.parseInt(hString.substring(0, endOfField)); + + // if it's a sample hh, write the hh and the person records + if (testHhs[hhId] == 1) + { + out.write(hString + "\n"); + count++; + if (count == numHhs) break; + } + } + + out.close(); + + } catch (IOException e) + { + logger.fatal("IO Exception caught creating sample synpop household file."); + logger.fatal(String.format("reading hh file = %s, writing sample hh file = %s.", + hhFileName, newHhFileName)); + logger.fatal(String.format("hString = %s, hCount = %d.", hString, hCount)); + } + + String pString = ""; + int pCount = 0; + try + { + + logger.info(String.format("writing sample person file for selected households")); + + PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newPersFileName))); + BufferedReader in = new BufferedReader(new FileReader(persFileName)); + + // read headers and write to output files + pString = in.readLine(); + out.write(pString + "\n"); + pCount++; + int count = 0; + int oldId = 0; + while ((pString = in.readLine()) != null) + { + pCount++; + int endOfField = pString.indexOf(','); + int hhId = Integer.parseInt(pString.substring(0, endOfField)); + + // if it's a sample hh, write the hh and the person records + if (testHhs[hhId] == 1) + { + out.write(pString + "\n"); + if (hhId > oldId) count++; + } else + { + if (count == numHhs) break; + } + + oldId = hhId; + + } + + out.close(); + + } catch (IOException e) + { + logger.fatal("IO Exception caught creating sample synpop person file."); + logger.fatal(String.format( + "reading person file = %s, writing sample person file = %s.", persFileName, + newPersFileName)); + logger.fatal(String.format("pString = %s, pCount = %d.", pString, pCount)); + } + + } + + public static void main(String[] args) throws Exception + { + + String serverAddress = HH_DATA_SERVER_ADDRESS; + int serverPort = HH_DATA_SERVER_PORT; + + // optional arguments + for (int i = 0; i < args.length; i++) + { + if (args[i].equalsIgnoreCase("-hostname")) + { + serverAddress = args[i + 1]; + } + + if (args[i].equalsIgnoreCase("-port")) + { + serverPort = Integer.parseInt(args[i + 1]); + } + } + + Remote.config(serverAddress, HH_DATA_SERVER_PORT, null, 0); + + SandagHouseholdDataManager hhDataManager = new SandagHouseholdDataManager(); + + ItemServer.bind(hhDataManager, HH_DATA_SERVER_NAME); + + System.out.println(String.format( + "SandagHouseholdDataManager server class started on: %s:%d", serverAddress, + serverPort)); + + } + + public int[] getJointToursByHomeMgra(String purposeString) + { + // TODO Auto-generated method stub + return null; + } + + private int[] readOccupCorrespondenceData() + { + + TableDataSet occTable = null; + + // construct input household file name from properties file values + String occupFileName = propertyMap.get(PROPERTIES_OCCUP_CODES); + String fileName = projectDirectory + "/" + occupFileName; + + try + { + logger.info("reading occupation codes data file for creating occupation segments."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + occTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String.format( + "Exception occurred occupation codes data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + // get the array of indices from the TableDataSet + int[] occcen1Col = occTable.getColumnAsInt("occcen1"); + int[] occupCol = occTable.getColumnAsInt("pecas_occ"); + + // get the max index value, to use for array dimensions + int maxOcc = 0; + for (int occ : occcen1Col) + if (occ > maxOcc) maxOcc = occ; + + int[] occcen1Occup = new int[maxOcc + 1]; + for (int i = 0; i < occcen1Col.length; i++) + { + int index = occcen1Col[i]; + int value = occupCol[i]; + occcen1Occup[index] = value; + } + + return occcen1Occup; + } + + private int[] readIndustryCorrespondenceData() + { + + TableDataSet indTable = null; + + // construct input household file name from properties file values + String indFileName = propertyMap.get(PROPERTIES_INDUSTRY_CODES); + String fileName = projectDirectory + "/" + indFileName; + + try + { + logger.info("reading industry codes data file for creating industry segments."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + indTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading indistry codes data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + // get the array of indices from the TableDataSet + int[] indcenCol = indTable.getColumnAsInt("indcen"); + int[] activityCol = indTable.getColumnAsInt("activity_code"); + + // get the max index value, to use for array dimensions + int maxInd = 0; + for (int ind : indcenCol) + if (ind > maxInd) maxInd = ind; + + int[] indcenIndustry = new int[maxInd + 1]; + for (int i = 0; i < indcenCol.length; i++) + { + int index = indcenCol[i]; + int value = activityCol[i]; + indcenIndustry[index] = value; + } + + return indcenIndustry; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java new file mode 100644 index 0000000..d6364c4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java @@ -0,0 +1,788 @@ +package org.sandag.abm.application; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileReader; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; + +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.HouseholdDataManager; +import org.sandag.abm.ctramp.Person; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.IndexSort; +import com.pb.common.util.PropertyMap; + +/** + * @author Jim Hicks + * + * Class for managing household and person object data read from + * synthetic population files. + */ +public class SandagHouseholdDataManager2 + extends HouseholdDataManager +{ + + public static final String HH_DATA_SERVER_NAME = SandagHouseholdDataManager.class + .getCanonicalName(); + public static final String HH_DATA_SERVER_ADDRESS = "127.0.0.1"; + public static final int HH_DATA_SERVER_PORT = 1139; + + public static final String PROPERTIES_OCCUP_CODES = "PopulationSynthesizer.OccupCodes"; + public static final String PROPERTIES_INDUSTRY_CODES = "PopulationSynthesizer.IndustryCodes"; + public static final String PROPERTIES_MILITARY_INDUSTRY_RANGE = "PopulationSynthesizer.MilitaryIndustryRange"; + + + private int militaryIndustryLow; + private int militaryIndustryHigh; + + public SandagHouseholdDataManager2() + { + super(); + } + + /** + * Associate data in hh and person TableDataSets read from synthetic + * population files with Household objects and Person objects with + * Households. + * + */ + public void mapTablesToHouseholdObjects() + { + + logger.info("mapping popsyn household and person data records to objects."); + + int id = -1; + + int invalidPersonTypeCount1 = 0; + int invalidPersonTypeCount2 = 0; + int invalidPersonTypeCount3 = 0; + + // read the correspondence files for mapping persons to occupation and + HashMap occCodes = readOccupCorrespondenceData(); + int[] indCodes = readIndustryCorrespondenceData(); + + // get the maximum HH id value to use to dimension the hhIndex + // correspondence + // array. The hhIndex array will store the hhArray index number for the + // given + // hh index. + int maxHhId = 0; + int hhIDColumn = hhTable.getColumnPosition(HH_ID_FIELD_NAME); + + for (int r = 1; r <= hhTable.getRowCount(); r++) + { + id = (int) hhTable.getValueAt(r, hhIDColumn); + if (id > maxHhId) maxHhId = id; + } + hhIndexArray = new int[maxHhId + 1]; + + // get an index array for households sorted in random order - to remove + // the original order + int[] firstSortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); + + // get a second index array for households sorted in random order - to + // select a sample from the randomly ordered hhs + int[] randomSortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); + + hhs = null; + + int numHouseholdsInSample = (int) (hhTable.getRowCount() * sampleRate); + Household[] hhArray = new Household[numHouseholdsInSample]; + + // String outputFileName = "sample_hh_mgra_taz_seed_" + sampleSeed + + // ".csv"; + // PrintWriter outStream = null; + // try { + // outStream = new PrintWriter(new BufferedWriter(new FileWriter(new + // File(outputFileName)))); + // outStream.println("i,mgra,taz"); + // } + // catch (IOException e) { + // logger.fatal(String.format("Exception occurred opening output skims file: %s.", + // outputFileName)); + // throw new RuntimeException(e); + // } + + int[] tempFreqArray = new int[40000]; + int[] hhOriginSortArray = new int[numHouseholdsInSample]; + for (int i = 0; i < numHouseholdsInSample; i++) + { + int r = firstSortedIndices[randomSortedIndices[i]] + 1; + // int hhId = (int) hhTable.getValueAt(r, + // hhTable.getColumnPosition(HH_ID_FIELD_NAME)); + int hhMgra = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); + int hhTaz = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); + hhOriginSortArray[i] = hhMgra; + tempFreqArray[hhMgra]++; + + // outStream.println(i + "," + hhMgra + "," + hhTaz); + } + + // outStream.close(); + // System.exit(1); + + int mgrasInSample = 0; + for (int i = 0; i < tempFreqArray.length; i++) + { + if (tempFreqArray[i] > 0) mgrasInSample++; + } + logger.info(mgrasInSample + " unique MGRA values in the " + (sampleRate * 100) + + "% sample."); + + // get an index array for households sorted in order of home mgra + int[] newOrder = new int[numHouseholdsInSample]; + int[] sortedIndices = IndexSort.indexSort(hhOriginSortArray); + for (int i = 0; i < sortedIndices.length; i++) + { + int k = sortedIndices[i]; + newOrder[k] = i; + } + + // for each household in the sample + for (int i = 0; i < numHouseholdsInSample; i++) + { + int r = firstSortedIndices[randomSortedIndices[i]] + 1; + try + { + // create a Household object + Household hh = new Household(modelStructure); + + // get required values from table record and store in Household + // object + id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); + hh.setHhId(id, inputRandomSeed); + + // set the household in the hhIndexArray in random order + int newIndex = newOrder[i]; + hhIndexArray[hh.getHhId()] = newIndex; + + int htaz = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); + hh.setHhTaz(htaz); + + int hmgra = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); + hh.setHhMgra(hmgra); + + // autos could be modeled or from PUMA + int numAutos = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_AUTOS_FIELD_NAME)); + hh.setHhAutos(numAutos); + + // set the hhSize variable and create Person objects for each + // person + int numPersons = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_SIZE_FIELD_NAME)); + hh.setHhSize(numPersons); + + int numWorkers = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_WORKERS_FIELD_NAME)); + hh.setHhWorkers(numWorkers); + + int incomeCat = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_INCOME_CATEGORY_FIELD_NAME)); + hh.setHhIncomeCategory(incomeCat); + + int incomeInDollars = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_INCOME_DOLLARS_FIELD_NAME)); + hh.setHhIncomeInDollars(incomeInDollars); + + // 0=Housing unit, 1=Institutional group quarters, + // 2=Noninstitutional + // group quarters + int unitType = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_UNITTYPE_FIELD_NAME)); + hh.setUnitType(unitType); + + // 1=Family household:married-couple, 2=Family household:male + // householder,no wife present, 3=Family household:female + // householder,no + // husband present + // 4=Nonfamily household:male householder, living alone, + // 5=Nonfamily + // household:male householder, not living alone, + // 6=Nonfamily household:female householder, living alone, + // 7=Nonfamily household:female householder, not living alone + int type = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_TYPE_FIELD_NAME)); + hh.setHhType(type); + + // 1=mobile home, 2=one-family house detached from any other + // house, + // 3=one-family house attached to one or more houses, + // 4=building with 2 apartments, 5=building with 3 or 4 + // apartments, + // 6=building with 5 to 9 apartments, + // 7=building with 10 to 19 apartments, 8=building with 20 to 49 + // apartments, + // 9=building with 50 or more apartments, 10=Boat,RV,van,etc. + int bldgsz = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(HH_BLDGSZ_FIELD_NAME)); + hh.setHhBldgsz(bldgsz); + + hh.initializeWindows(); + hhArray[newIndex] = hh; + + } catch (Exception e) + { + + logger.fatal(String + .format("exception caught mapping household data record to a Household object, r=%d, id=%d.", + r, id)); + throw new RuntimeException(e); + + } + + } + + int[] personHhStart = new int[maxHhId + 1]; + int[] personHhEnd = new int[maxHhId + 1]; + + // get hhid for person record 1 + int hhid = (int) personTable.getValueAt(1, + personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); + personHhStart[hhid] = 1; + int oldHhid = hhid; + + for (int r = 1; r <= personTable.getRowCount(); r++) + { + + // get the Household object for this person data to be stored in + hhid = (int) personTable.getValueAt(r, + personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); + + if (hhid != oldHhid) + { + personHhEnd[oldHhid] = r - 1; + oldHhid = hhid; + personHhStart[hhid] = r; + } + + } + personHhEnd[hhid] = personTable.getRowCount(); + + int r = 0; + int p = 0; + int persId = 0; + int persNum = 0; + int fieldCount = 0; + + for (int i = 0; i < numHouseholdsInSample; i++) + { + + try + { + + r = firstSortedIndices[randomSortedIndices[i]] + 1; + + hhid = (int) hhTable.getValueAt(r, + hhTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); + + int index = hhIndexArray[hhid]; + Household hh = hhArray[index]; + + persNum = 1; + + for (p = personHhStart[hhid]; p <= personHhEnd[hhid]; p++) + { + + fieldCount = 0; + + // get the Person object for this person data to be stored + // in + persId = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_PERSON_ID_FIELD_NAME)); + Person person = hh.getPerson(persNum++); + person.setPersId(persId); + fieldCount++; + + // get required values from table record and store in Person + // object + int age = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_AGE_FIELD_NAME)); + person.setPersAge(age); + fieldCount++; + + int gender = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_GENDER_FIELD_NAME)); + person.setPersGender(gender); + fieldCount++; + /* + * int occcen1 = (int) personTable.getValueAt(p, + * personTable.getColumnPosition(PERSON_SOC_FIELD_NAME)); + * int pecasOccup = occCodes[occcen1]; + */ + int military = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_MILITARY_FIELD_NAME)); + int pecasOccup = 0; + + String occsoc = personTable.getStringValueAt(p, + personTable.getColumnPosition(PERSON_SOC_FIELD_NAME)); + + int indcen = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_INDCEN_FIELD_NAME)); + int activityCode = indCodes[indcen]; + + if (military == 1) // in active military + pecasOccup = 56; + else if (military != 1 && indcen >= militaryIndustryLow + && indcen <= militaryIndustryHigh) // not active + // military but + // military + // contractor + pecasOccup = 56; + else + { + if (!occCodes.containsKey(occsoc)) + { + logger.fatal("Error: Occupation code " + occsoc + " for hhid " + hhid + + " person " + p + " not found in occupation file"); + throw new RuntimeException(); + } + pecasOccup = occCodes.get(occsoc); // everyone else + if (pecasOccup == 0) logger.warn("pecasOccup==0 for occsoc==" + occsoc); + } + + person.setPersActivityCode(activityCode); + fieldCount++; + + person.setPersPecasOccup(pecasOccup); + fieldCount++; + + /* + * These are the old codes, based upon census occupation + * definitions if ((pecasOccup == 71) && (activityCode == 2 + * || activityCode == 4 || activityCode == 6 || activityCode + * == 8 || activityCode == 29)) activityCode++; + * + * if ((pecasOccup == 76) && (activityCode == 3 || + * activityCode == 5 || activityCode == 7 || activityCode == + * 9 || activityCode == 30)) activityCode--; + * + * if ((pecasOccup == 76) && (activityCode == 13)) + * activityCode = 14; + * + * if ((pecasOccup == 71) && (activityCode == 14)) + * activityCode = 13; + * + * if ((pecasOccup == 75) && (activityCode == 18)) + * activityCode = 22; + * + * if ((pecasOccup == 71) && (activityCode == 22)) + * activityCode = 18; + * + * if (activityCode == 28) pecasOccup = 77; + */ + + // Employment status (1-employed FT, 2-employed PT, 3-not + // employed, 4-under age 16) + int empCat = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME)); + + // recode PEMPLOY to 3 for persons whose age is 16 or + // greater and who have PEMPLOY set to 4 + if (empCat == 4 && age >= 16) empCat = 3; + person.setPersEmploymentCategory(empCat); + + fieldCount++; + + // Student status (1 - student in grade or high school; 2 - + // student in college or higher; 3 - not a student) + int studentCat = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_STUDENT_CATEGORY_FIELD_NAME)); + person.setPersStudentCategory(studentCat); + fieldCount++; + + // Person type (1-FT worker age 16+, 2-PT worker nonstudent + // age + // 16+, 3-university student, 4-nonworker nonstudent age + // 16-64, + // 5-nonworker nonstudent age 65+, + // 6-"age 16-19 student, not FT wrkr or univ stud", 7-age + // 6-15 + // schpred, 8 under age 6 presch + int personType = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_TYPE_CATEGORY_FIELD_NAME)); + person.setPersonTypeCategory(personType); + fieldCount++; + + // Person educational attainment level to determine high + // school + // graduate status ( < 9 - not a graduate, 10+ - high school + // graduate + // and beyond) + int educ = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_EDUCATION_ATTAINMENT_FIELD_NAME)); + if (educ >= 9) person.setPersonIsHighSchoolGraduate(true); + else person.setPersonIsHighSchoolGraduate(false); + fieldCount++; + + // Person educational attainment level to determine higher + // education status ( > 12 - at least a bachelor's degree ) + if (educ >= 13) person.setPersonHasBachelors(true); + else person.setPersonHasBachelors(false); + fieldCount++; + + // Person grade enrolled in ( 0-"not enrolled", + // 1-"preschool", + // 2-"Kindergarten", 3-"Grade 1 to grade 4", + // 4-"Grade 5 to grade 8", 5-"Grade 9 to grade 12", + // 6-"College undergraduate", + // 7-"Graduate or professional school" + // ) + int grade = (int) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_GRADE_ENROLLED_FIELD_NAME)); + person.setPersonIsGradeSchool(false); + person.setPersonIsHighSchool(false); + if (grade >= 2 && grade <= 4) + { + // change person type if person was 5 or under but + // enrolled in K-8. + if (person.getPersonIsPreschoolChild() == 1) + person.setPersonTypeCategory(Person.PersonType.Student_age_6_15_schpred + .ordinal()); + + person.setPersonIsGradeSchool(true); + } else if (grade == 5) + { + person.setPersonIsHighSchool(true); + } + fieldCount++; + + // if person is a university student but has school age + // student + // category value, reset student category value + if (personType == Person.PersonType.University_student.ordinal() + && studentCat != Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER + .ordinal()) + { + studentCat = Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount1++; + } else if (studentCat != Person.StudentStatus.NON_STUDENT.ordinal() + && empCat == Person.EmployStatus.FULL_TIME.ordinal()) + { + // if person is a student of any kind but has full-time + // employment status, reset student category value to + // non-student + studentCat = Person.StudentStatus.NON_STUDENT.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount2++; + } + fieldCount++; + + // check consistency of student category and person type + if (studentCat == Person.StudentStatus.NON_STUDENT.ordinal()) + { + + if (person.getPersonIsStudentNonDriving() == 1 + || person.getPersonIsStudentDriving() == 1) + { + studentCat = Person.StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal(); + person.setPersStudentCategory(studentCat); + invalidPersonTypeCount3++; + } + + } + fieldCount++; + + double timeFactorWork = 1.0; + double timeFactorNonWork = 1.0; + if(readTimeFactors){ + timeFactorWork = (double) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_TIMEFACTOR_WORK_FIELD_NAME)); + timeFactorNonWork = (double) personTable.getValueAt(p, + personTable.getColumnPosition(PERSON_TIMEFACTOR_NONWORK_FIELD_NAME)); + } + person.setTimeFactorWork(timeFactorWork); + person.setTimeFactorNonWork(timeFactorNonWork); + + } + + } catch (Exception e) + { + + logger.fatal(String + .format("exception caught mapping person data record to a Person object, i=%d, r=%d, p=%d, hhid=%d, persid=%d, persnum=%d, fieldCount=%d.", + i, r, p, hhid, persId, persNum, fieldCount)); + throw new RuntimeException(e); + + } + + } // person loop + + hhs = hhArray; + + logger.warn(invalidPersonTypeCount1 + + " person type = university and student category = non-student person records had their student category changed to university or higher."); + logger.warn(invalidPersonTypeCount2 + + " Student category = student and employment category = full-time worker person records had their student category changed to non-student."); + logger.warn(invalidPersonTypeCount3 + + " Student category = non-student and person type = student person records had their student category changed to student high school or less."); + + // logger.info("Setting distributed values of time. "); + // setDistributedValuesOfTime(); + + } + + /** + * if called, must be called after readData so that the size of the full + * population is known. + * + * @param hhFileName + * @param persFileName + * @param numHhs + */ + public void createSamplePopulationFiles(String hhFileName, String persFileName, + String newHhFileName, String newPersFileName, int numHhs) + { + + int maximumHhId = 0; + for (int i = 0; i < hhs.length; i++) + { + int id = hhs[i].getHhId(); + if (id > maximumHhId) maximumHhId = id; + } + + int[] testHhs = new int[maximumHhId + 1]; + + int[] sortedIndices = getRandomOrderHhIndexArray(hhs.length); + + for (int i = 0; i < numHhs; i++) + { + int k = sortedIndices[i]; + int hhId = hhs[k].getHhId(); + testHhs[hhId] = 1; + } + + String hString = ""; + int hCount = 0; + try + { + + logger.info(String.format("writing sample household file for %d households", numHhs)); + + PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newHhFileName))); + BufferedReader in = new BufferedReader(new FileReader(hhFileName)); + + // read headers and write to output files + hString = in.readLine(); + out.write(hString + "\n"); + hCount++; + int count = 0; + + while ((hString = in.readLine()) != null) + { + hCount++; + int endOfField = hString.indexOf(','); + int hhId = Integer.parseInt(hString.substring(0, endOfField)); + + // if it's a sample hh, write the hh and the person records + if (testHhs[hhId] == 1) + { + out.write(hString + "\n"); + count++; + if (count == numHhs) break; + } + } + + out.close(); + + } catch (IOException e) + { + logger.fatal("IO Exception caught creating sample synpop household file."); + logger.fatal(String.format("reading hh file = %s, writing sample hh file = %s.", + hhFileName, newHhFileName)); + logger.fatal(String.format("hString = %s, hCount = %d.", hString, hCount)); + } + + String pString = ""; + int pCount = 0; + try + { + + logger.info(String.format("writing sample person file for selected households")); + + PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newPersFileName))); + BufferedReader in = new BufferedReader(new FileReader(persFileName)); + + // read headers and write to output files + pString = in.readLine(); + out.write(pString + "\n"); + pCount++; + int count = 0; + int oldId = 0; + while ((pString = in.readLine()) != null) + { + pCount++; + int endOfField = pString.indexOf(','); + int hhId = Integer.parseInt(pString.substring(0, endOfField)); + + // if it's a sample hh, write the hh and the person records + if (testHhs[hhId] == 1) + { + out.write(pString + "\n"); + if (hhId > oldId) count++; + } else + { + if (count == numHhs) break; + } + + oldId = hhId; + + } + + out.close(); + + } catch (IOException e) + { + logger.fatal("IO Exception caught creating sample synpop person file."); + logger.fatal(String.format( + "reading person file = %s, writing sample person file = %s.", persFileName, + newPersFileName)); + logger.fatal(String.format("pString = %s, pCount = %d.", pString, pCount)); + } + + } + + public static void main(String[] args) throws Exception + { + + String serverAddress = HH_DATA_SERVER_ADDRESS; + int serverPort = HH_DATA_SERVER_PORT; + + // optional arguments + for (int i = 0; i < args.length; i++) + { + if (args[i].equalsIgnoreCase("-hostname")) + { + serverAddress = args[i + 1]; + } + + if (args[i].equalsIgnoreCase("-port")) + { + serverPort = Integer.parseInt(args[i + 1]); + } + } + + Remote.config(serverAddress, serverPort, null, 0); + + SandagHouseholdDataManager2 hhDataManager = new SandagHouseholdDataManager2(); + + ItemServer.bind(hhDataManager, HH_DATA_SERVER_NAME); + + System.out.println(String.format( + "SandagHouseholdDataManager2 server class started on: %s:%d", serverAddress, + serverPort)); + + } + + public int[] getJointToursByHomeMgra(String purposeString) + { + // TODO Auto-generated method stub + return null; + } + + /** + * This method reads a cross-walk file between the occsoc code in Census and + * the PECAS occupation categories. It stores the result in a HashMap and + * returns it. + * + * @return + */ + private HashMap readOccupCorrespondenceData() + { + + int[] militaryRange = PropertyMap.getIntegerArrayFromPropertyMap(propertyMap, + PROPERTIES_MILITARY_INDUSTRY_RANGE); + militaryIndustryLow = militaryRange[0]; + militaryIndustryHigh = militaryRange[1]; + + TableDataSet occTable = null; + + // construct input household file name from properties file values + String occupFileName = propertyMap.get(PROPERTIES_OCCUP_CODES); + String fileName = projectDirectory + "/" + occupFileName; + + try + { + logger.info("reading occupation codes data file for creating occupation segments."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + occTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String.format( + "Exception occurred occupation codes data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + HashMap occMap = new HashMap(); + + for (int i = 1; i <= occTable.getRowCount(); ++i) + { + + String soc = occTable.getStringValueAt(i, "occsoc5"); + int occ = (int) occTable.getValueAt(i, "commodity_id"); + occMap.put(soc, occ); + } + + return occMap; + } + + private int[] readIndustryCorrespondenceData() + { + + TableDataSet indTable = null; + + // construct input household file name from properties file values + String indFileName = propertyMap.get(PROPERTIES_INDUSTRY_CODES); + String fileName = projectDirectory + "/" + indFileName; + + try + { + logger.info("reading industry codes data file for creating industry segments."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + indTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading indistry codes data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + // get the array of indices from the TableDataSet + int[] indcenCol = indTable.getColumnAsInt("indcen"); + int[] activityCol = indTable.getColumnAsInt("activity_code"); + + // get the max index value, to use for array dimensions + int maxInd = 0; + for (int ind : indcenCol) + if (ind > maxInd) maxInd = ind; + + int[] indcenIndustry = new int[maxInd + 1]; + for (int i = 0; i < indcenCol.length; i++) + { + int index = indcenCol[i]; + int value = activityCol[i]; + indcenIndustry[index] = value; + } + + return indcenIndustry; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java new file mode 100644 index 0000000..0a40886 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java @@ -0,0 +1,101 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyDMU; + +/** + * ArcIndividualMandatoryTourFrequencyDMU is a class that ... + * + * @author Kimberly Grommes + * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. + */ +public class SandagIndividualMandatoryTourFrequencyDMU + extends IndividualMandatoryTourFrequencyDMU +{ + + public SandagIndividualMandatoryTourFrequencyDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getDistanceToWorkLocation", 1); + methodIndexMap.put("getDistanceToSchoolLocation", 2); + methodIndexMap.put("getEscortAccessibility", 3); + methodIndexMap.put("getDrivers", 4); + methodIndexMap.put("getPreschoolChildren", 5); + methodIndexMap.put("getNumberOfChildren6To18WithoutMandatoryActivity", 6); + methodIndexMap.put("getNonFamilyHousehold", 7); + methodIndexMap.put("getIncomeInDollars", 8); + methodIndexMap.put("getPersonType", 9); + methodIndexMap.put("getFemale", 10); + methodIndexMap.put("getAutos", 11); + methodIndexMap.put("getAge", 12); + methodIndexMap.put("getBestTimeToWorkLocation", 13); + methodIndexMap.put("getNotEmployed", 14); + methodIndexMap.put("getWorkAtHome", 15); + methodIndexMap.put("getSchoolAtHome", 16); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getDistanceToWorkLocation(); + case 2: + return getDistanceToSchoolLocation(); + case 3: + return getEscortAccessibility(); + case 4: + return getDrivers(); + case 5: + return getPreschoolChildren(); + case 6: + return getNumberOfChildren6To18WithoutMandatoryActivity(); + case 7: + return getNonFamilyHousehold(); + case 8: + return getIncomeInDollars(); + case 9: + return getPersonType(); + case 10: + return getFemale(); + case 11: + return getAutos(); + case 12: + return getAge(); + case 13: + return getBestTimeToWorkLocation(); + case 14: + return getNotEmployed(); + case 15: + return getWorkAtHome(); + case 16: + return getSchoolAtHome(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java new file mode 100644 index 0000000..5e675e8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java @@ -0,0 +1,272 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.IndividualNonMandatoryTourFrequencyDMU; + +/** + * ArcIndividualNonMandatoryTourFrequencyDMU is a class that ... + * + */ +public class SandagIndividualNonMandatoryTourFrequencyDMU + extends IndividualNonMandatoryTourFrequencyDMU +{ + + public SandagIndividualNonMandatoryTourFrequencyDMU() + { + super(); + setupMethodIndexMap(); + + // set names used in SANDAG stop purpose file + TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME = "escort"; + TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME = "shopping"; + TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME = "othmaint"; + TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME = "eatout"; + TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME = "visit"; + TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME = "othdiscr"; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getIncomeInDollars", 0); + methodIndexMap.put("getHouseholdSize", 1); + methodIndexMap.put("getNumAutos", 2); + methodIndexMap.put("getCarsEqualsWorkers", 3); + methodIndexMap.put("getMoreCarsThanWorkers", 4); + methodIndexMap.put("getNumAdults", 5); + methodIndexMap.put("getNumChildren", 6); + methodIndexMap.put("getPersonIsAdult", 7); + methodIndexMap.put("getPersonIsChild", 8); + methodIndexMap.put("getPersonIsFullTimeWorker", 9); + methodIndexMap.put("getPersonIsPartTimeWorker", 10); + methodIndexMap.put("getPersonIsUniversity", 11); + methodIndexMap.put("getPersonIsNonworker", 12); + methodIndexMap.put("getPersonIsPreschool", 13); + methodIndexMap.put("getPersonIsStudentNonDriving", 14); + methodIndexMap.put("getPersonIsStudentDriving", 15); + methodIndexMap.put("getPersonStaysHome", 16); + methodIndexMap.put("getFemale", 17); + methodIndexMap.put("getFullTimeWorkers", 18); + methodIndexMap.put("getPartTimeWorkers", 19); + methodIndexMap.put("getUniversityStudents", 20); + methodIndexMap.put("getNonWorkers", 21); + methodIndexMap.put("getDrivingAgeStudents", 22); + methodIndexMap.put("getNonDrivingAgeStudents", 23); + methodIndexMap.put("getPreSchoolers", 24); + // methodIndexMap.put("getMaxAdultOverlaps", 26); + // methodIndexMap.put("getMaxChildOverlaps", 27); + // methodIndexMap.put("getMaxMixedOverlaps", 28); + // methodIndexMap.put("getMaxPairwiseOverlapAdult", 29); + // methodIndexMap.put("getMaxPairwiseOverlapChild", 30); + // methodIndexMap.put("getWindowBeforeFirstMandJointTour", 31); + // methodIndexMap.put("getWindowBetweenFirstLastMandJointTour", 32); + // methodIndexMap.put("getWindowAfterLastMandJointTour", 33); + methodIndexMap.put("getNumHhFtWorkers", 34); + methodIndexMap.put("getNumHhPtWorkers", 35); + methodIndexMap.put("getNumHhUnivStudents", 36); + methodIndexMap.put("getNumHhNonWorkAdults", 37); + methodIndexMap.put("getNumHhRetired", 38); + methodIndexMap.put("getNumHhDrivingStudents", 39); + methodIndexMap.put("getNumHhNonDrivingStudents", 40); + methodIndexMap.put("getNumHhPreschool", 41); + methodIndexMap.put("getTravelActiveAdults ", 42); + methodIndexMap.put("getTravelActiveChildren ", 43); + methodIndexMap.put("getNumMandatoryTours", 44); + methodIndexMap.put("getNumJointShoppingTours", 45); + methodIndexMap.put("getNumJointOthMaintTours", 46); + methodIndexMap.put("getNumJointEatOutTours", 47); + methodIndexMap.put("getNumJointSocialTours", 48); + methodIndexMap.put("getNumJointOthDiscrTours", 49); + methodIndexMap.put("getJTours", 50); + methodIndexMap.put("getPreDrivingAtHome", 51); + methodIndexMap.put("getPreschoolAtHome", 52); + methodIndexMap.put("getDistanceToWorkLocation", 53); + methodIndexMap.put("getDistanceToSchoolLocation", 54); + methodIndexMap.put("getEscortAccessibility", 55); + methodIndexMap.put("getShopAccessibility", 56); + methodIndexMap.put("getMaintAccessibility", 57); + methodIndexMap.put("getEatOutAccessibility", 58); + methodIndexMap.put("getVisitAccessibility", 59); + methodIndexMap.put("getDiscrAccessibility", 60); + methodIndexMap.put("getCdapIndex", 61); + methodIndexMap.put("getNonMotorizedDcLogsum", 62); + methodIndexMap.put("getNumPredrivingKidsGoOut", 63); + methodIndexMap.put("getNumPreschoolKidsGoOut", 64); + methodIndexMap.put("getCollegeEducation", 65); + methodIndexMap.put("getLowEducation", 66); + methodIndexMap.put("getDetachedHh", 67); + methodIndexMap.put("getWorksAtHome", 68); + methodIndexMap.put("getWorkAccessibility", 69); + methodIndexMap.put("getSchoolAccessibility", 70); + methodIndexMap.put("getTelecommuteFrequency", 71); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + + case 0: + return getIncomeInDollars(); + case 1: + return getHouseholdSize(); + case 2: + return getNumAutos(); + case 3: + return getCarsEqualsWorkers(); + case 4: + return getMoreCarsThanWorkers(); + case 5: + return getNumAdults(); + case 6: + return getNumChildren(); + case 7: + return getPersonIsAdult(); + case 8: + return getPersonIsChild(); + case 9: + return getPersonIsFullTimeWorker(); + case 10: + return getPersonIsPartTimeWorker(); + case 11: + return getPersonIsUniversity(); + case 12: + return getPersonIsNonworker(); + case 13: + return getPersonIsPreschool(); + case 14: + return getPersonIsStudentNonDriving(); + case 15: + return getPersonIsStudentDriving(); + case 16: + return getPersonStaysHome(); + case 17: + return getFemale(); + case 18: + return getFullTimeWorkers(); + case 19: + return getPartTimeWorkers(); + case 20: + return getUniversityStudents(); + case 21: + return getNonWorkers(); + case 22: + return getDrivingAgeStudents(); + case 23: + return getNonDrivingAgeStudents(); + case 24: + return getPreSchoolers(); + // case 26: + // return getMaxAdultOverlaps(); + // case 27: + // return getMaxChildOverlaps(); + // case 28: + // return getMaxMixedOverlaps(); + // case 29: + // return getMaxPairwiseOverlapAdult(); + // case 30: + // return getMaxPairwiseOverlapChild(); + // case 31: + // return getWindowBeforeFirstMandJointTour(); + // case 32: + // return getWindowBetweenFirstLastMandJointTour(); + // case 33: + // return getWindowAfterLastMandJointTour(); + case 34: + return getNumHhFtWorkers(); + case 35: + return getNumHhPtWorkers(); + case 36: + return getNumHhUnivStudents(); + case 37: + return getNumHhNonWorkAdults(); + case 38: + return getNumHhRetired(); + case 39: + return getNumHhDrivingStudents(); + case 40: + return getNumHhNonDrivingStudents(); + case 41: + return getNumHhPreschool(); + case 42: + return getTravelActiveAdults(); + case 43: + return getTravelActiveChildren(); + case 44: + return getNumMandatoryTours(); + case 45: + return getNumJointShoppingTours(); + case 46: + return getNumJointOthMaintTours(); + case 47: + return getNumJointEatOutTours(); + case 48: + return getNumJointSocialTours(); + case 49: + return getNumJointOthDiscrTours(); + case 50: + return getJTours(); + case 51: + return getPreDrivingAtHome(); + case 52: + return getPreschoolAtHome(); + case 53: + return getDistanceToWorkLocation(); + case 54: + return getDistanceToSchoolLocation(); + case 55: + return getEscortAccessibility(); + case 56: + return getShopAccessibility(); + case 57: + return getMaintAccessibility(); + case 58: + return getEatOutAccessibility(); + case 59: + return getVisitAccessibility(); + case 60: + return getDiscrAccessibility(); + case 61: + return getCdapIndex(); + case 62: + return getNonMotorizedDcLogsum(); + case 63: + return getNumPredrivingKidsGoOut(); + case 64: + return getNumPreschoolKidsGoOut(); + case 65: + return getCollegeEducation(); + case 66: + return getLowEducation(); + case 67: + return getDetachedHh(); + case 68: + return getWorksAtHome(); + case 69: + return getWorkAccessibility(); + case 70: + return getSchoolAccessibility(); + case 71: + return getTelecommuteFrequency(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java new file mode 100644 index 0000000..fbab4ac --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java @@ -0,0 +1,50 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.InternalExternalTripChoiceDMU; + +public class SandagInternalExternalTripChoiceDMU + extends InternalExternalTripChoiceDMU +{ + + private transient Logger logger = Logger.getLogger(SandagInternalExternalTripChoiceDMU.class); + + public SandagInternalExternalTripChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getDistanceToCordonsLogsum", 0); + methodIndexMap.put("getVehiclesPerHouseholdMember", 1); + methodIndexMap.put("getHhIncomeInDollars", 2); + methodIndexMap.put("getAge", 3); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getDistanceToCordonsLogsum(); + case 1: + return getVehiclesPerHouseholdMember(); + case 2: + return getHhIncomeInDollars(); + case 3: + return getAge(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java new file mode 100644 index 0000000..f57b1d3 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java @@ -0,0 +1,137 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.JointTourModelsDMU; +import org.sandag.abm.ctramp.ModelStructure; + +public class SandagJointTourModelsDMU + extends JointTourModelsDMU +{ + + public SandagJointTourModelsDMU(ModelStructure modelStructure) + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getActiveCountFullTimeWorkers", 1); + methodIndexMap.put("getActiveCountPartTimeWorkers", 2); + methodIndexMap.put("getActiveCountUnivStudents", 3); + methodIndexMap.put("getActiveCountNonWorkers", 4); + methodIndexMap.put("getActiveCountRetirees", 5); + methodIndexMap.put("getActiveCountDrivingAgeSchoolChildren", 6); + methodIndexMap.put("getActiveCountPreDrivingAgeSchoolChildren", 7); + methodIndexMap.put("getActiveCountPreSchoolChildren", 8); + methodIndexMap.put("getMaxPairwiseAdultOverlapsHh", 9); + methodIndexMap.put("getMaxPairwiseChildOverlapsHh", 10); + methodIndexMap.put("getMaxPairwiseMixedOverlapsHh", 11); + methodIndexMap.put("getMaxPairwiseOverlapOtherAdults", 12); + methodIndexMap.put("getMaxPairwiseOverlapOtherChildren", 13); + methodIndexMap.put("getTravelActiveAdults", 14); + methodIndexMap.put("getTravelActiveChildren", 15); + methodIndexMap.put("getPersonStaysHome", 16); + methodIndexMap.put("getIncomeLessThan30K", 17); + methodIndexMap.put("getIncome30Kto60K", 18); + methodIndexMap.put("getIncomeMoreThan100K", 19); + methodIndexMap.put("getNumAdults", 20); + methodIndexMap.put("getNumChildren", 21); + methodIndexMap.put("getHhWorkers", 22); + methodIndexMap.put("getAutoOwnership", 23); + methodIndexMap.put("getTourPurposeIsMaint", 24); + methodIndexMap.put("getTourPurposeIsEat", 25); + methodIndexMap.put("getTourPurposeIsVisit", 26); + methodIndexMap.put("getTourPurposeIsDiscr", 27); + methodIndexMap.put("getPersonType", 28); + methodIndexMap.put("getJointTourComposition", 29); + methodIndexMap.put("getJTours", 30); + methodIndexMap.put("getShopHOVAccessibility", 31); + methodIndexMap.put("getMaintHOVAccessibility", 32); + methodIndexMap.put("getDiscrHOVAccessibility", 33); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getActiveCountFullTimeWorkers(); + case 2: + return getActiveCountPartTimeWorkers(); + case 3: + return getActiveCountUnivStudents(); + case 4: + return getActiveCountNonWorkers(); + case 5: + return getActiveCountRetirees(); + case 6: + return getActiveCountDrivingAgeSchoolChildren(); + case 7: + return getActiveCountPreDrivingAgeSchoolChildren(); + case 8: + return getActiveCountPreSchoolChildren(); + case 9: + return getMaxPairwiseAdultOverlapsHh(); + case 10: + return getMaxPairwiseChildOverlapsHh(); + case 11: + return getMaxPairwiseMixedOverlapsHh(); + case 12: + return getMaxPairwiseOverlapOtherAdults(); + case 13: + return getMaxPairwiseOverlapOtherChildren(); + case 14: + return getTravelActiveAdults(); + case 15: + return getTravelActiveChildren(); + case 16: + return getPersonStaysHome(); + case 17: + return getIncomeLessThan30K(); + case 18: + return getIncome30Kto60K(); + case 19: + return getIncomeMoreThan100K(); + case 20: + return getNumAdults(); + case 21: + return getNumChildren(); + case 22: + return getHhWorkers(); + case 23: + return getAutoOwnership(); + case 24: + return getTourPurposeIsMaint(); + case 25: + return getTourPurposeIsEat(); + case 26: + return getTourPurposeIsVisit(); + case 27: + return getTourPurposeIsDiscr(); + case 28: + return getPersonType(); + case 29: + return getJointTourComposition(); + case 30: + return getJTours(); + case 31: + return getShopHOVAccessibility(); + case 32: + return getMaintHOVAccessibility(); + case 33: + return getDiscrHOVAccessibility(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java new file mode 100644 index 0000000..2c74056 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java @@ -0,0 +1,287 @@ +package org.sandag.abm.application; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileInputStream; +import java.io.FileWriter; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.MissingResourceException; +import java.util.Properties; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.active.sandag.PropertyParser; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.reporting.CsvRow; +import org.sandag.abm.reporting.DataExporter; +import org.sandag.abm.reporting.OMXMatrixDao; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +/** + * @author malinovskiyy + * Produces p&r and k&r access connections + * input files: + * impdan_xx.mtx: SOV non toll xx peak period skim matrix (3 cores: *SCST_XX (generalized cost), length(Skim) (distance in mile), and *STM_XX (Skim) (single driver xx time in minutes) + * tap.ptype: tap, lot id, parking type, taz, capacity, distance from lot to tap + * output files + * access.prp + */ +public class SandagMGRAtoPNR { + + private static Logger logger = Logger.getLogger("createPNRAccessFile"); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TapDataManager tapManager; + private int[] taps; + private int[] tazs; + private float[][][] tapParkingInfo; + private HashMap tapMap; + private HashMap> bestTapsMap; + private Matrix distanceMtx; + private Matrix timeMtx; + + private final Properties properties; + private final OMXMatrixDao mtxDao; + + private static final String FORMAL_PREMIUM = "taps.formal.premium.maxDist"; //8.0f; + private static final String FORMAL_EXPRESS = "taps.formal.express.maxDist"; //8.0f; + private static final String FORMAL_LOCAL = "taps.formal.local.maxDist"; //4.0f; + private static final String INFORMAL_PREMIUM = "taps.informal.premium.maxDist"; //4.0f; + private static final String INFORMAL_EXPRESS = "taps.informal.express.maxDist"; //4.0f; + private static final String INFORMAL_LOCAL = "taps.informal.local.maxDist"; //2.0f; + private static final String PREMIUM_MODES = "taps.premium.modes"; //new ArrayList(){{add(4); add(5); add(6); add(7);}}; + private static final String EXPRESS_MODES = "taps.express.modes"; //new ArrayList(){{add(8); add(9);}}; + private static final String LOCAL_MODES = "taps.local.modes"; //new ArrayList(){{add(10);}}; + + private static final String TAPS_SKIM = "taps.skim"; + private static final String TAPS_SKIM_DIST = "taps.skim.dist"; + private static final String TAPS_SKIM_TIME = "taps.skim.time"; + + private static final String EXTERNAL_TAZs = "external.tazs"; + + private double formalPremiumMaxD; + private double formalExpressMaxD; + private double formalLocalMaxD; + private double informalPremiumMaxD; + private double informalExpressMaxD; + private double informalLocalMaxD; + + private ArrayList premiumModes; + private ArrayList expressModes; + private ArrayList localModes; + private ArrayList externalTAZs; + + private static final String PROJECT_PATH_PROPERTY_TOKEN = "%project.folder%"; + + public SandagMGRAtoPNR(Properties theProperties, OMXMatrixDao aMtxDao, String projectPath, HashMap rbMap) + { + this.properties = theProperties; + this.mtxDao = aMtxDao; + + tapManager = TapDataManager.getInstance(rbMap); + //tazManager = TazDataManager.getInstance(rbMap); + this.taps = tapManager.getTaps(); + this.tapParkingInfo = tapManager.getTapParkingInfo(); + this.tapMap = getTAPMap(); + this.bestTapsMap = new HashMap>(); + this.distanceMtx = aMtxDao.getMatrix((String)properties.get(TAPS_SKIM),(String)properties.get(TAPS_SKIM_DIST)); + this.timeMtx = aMtxDao.getMatrix((String)properties.get(TAPS_SKIM),(String)properties.get(TAPS_SKIM_TIME)); + this.tazs = this.distanceMtx.getExternalRowNumbers(); + + + formalPremiumMaxD = Double.parseDouble((String) properties.get(FORMAL_PREMIUM)); + formalExpressMaxD = Double.parseDouble((String) properties.get(FORMAL_EXPRESS)); + formalLocalMaxD = Double.parseDouble((String) properties.get(FORMAL_LOCAL)); + informalPremiumMaxD = Double.parseDouble((String) properties.get(INFORMAL_PREMIUM)); + informalExpressMaxD = Double.parseDouble((String) properties.get(INFORMAL_EXPRESS)); + informalLocalMaxD = Double.parseDouble((String) properties.get(INFORMAL_LOCAL)); + + List stringList = Arrays.asList(((String) properties.get(PREMIUM_MODES)).split("\\s*,\\s*")); + premiumModes = new ArrayList(); + for (int i = 0; i < stringList.size(); i++){ + premiumModes.add(Integer.parseInt(stringList.get(i))); + } + + stringList = Arrays.asList(((String) properties.get(EXPRESS_MODES)).split("\\s*,\\s*")); + expressModes = new ArrayList(); + for (int i = 0; i < stringList.size(); i++){ + expressModes.add(Integer.parseInt(stringList.get(i))); + } + + stringList = Arrays.asList(((String) properties.get(LOCAL_MODES)).split("\\s*,\\s*")); + localModes = new ArrayList(); + for (int i = 0; i < stringList.size(); i++){ + localModes.add(Integer.parseInt(stringList.get(i))); + } + + stringList = Arrays.asList(((String) properties.get(EXTERNAL_TAZs)).split("\\s*,\\s*")); + externalTAZs = new ArrayList(); + for (int i = 0; i < stringList.size(); i++){ + externalTAZs.add(Integer.parseInt(stringList.get(i))); + } + } + + + public HashMap getTAPMap(){ + HashMap tm = new HashMap(); + for(int i = 0; i < taps.length; i++){ + int tap = taps[i]; + if(tap < tapParkingInfo.length && tapParkingInfo[tap] != null && tapParkingInfo[tap][0] != null){ + int taz = (int) tapParkingInfo[tap][1][0]; + tm.put(tap, taz); + } + } + return tm; + } + + public void nearestTAPs(){ + for(int i = 1; i < tazs.length; i++) { + int currTAZ = tazs[i]; + + //Skip externals + if(externalTAZs.contains(currTAZ)) + continue; + + ArrayList reachableTAPs = new ArrayList(); + bestTapsMap.put(currTAZ, reachableTAPs); + + HashMap modeMap = new HashMap(); + ArrayList addedTaps = new ArrayList(); + + for(Integer j : tapMap.keySet()){ + int tapTAZ = tapMap.get(j); + //distance to taz with the current tap + float dist = distanceMtx.getValueAt(currTAZ, tapTAZ); + float time = timeMtx.getValueAt(currTAZ, tapTAZ); + int lotType = (int) tapParkingInfo[j][3][0]; + int mode = (int) tapParkingInfo[j][5][0]; + //dist = (float) (dist + (tapParkingInfo[j][4][0] / 5280.0)); + if(!modeMap.containsKey(mode) || modeMap.get(mode)[2] > dist){ + float[] vals = new float[4]; + vals[0] = j; + vals[1] = time; + vals[2] = dist; + vals[3] = mode; + modeMap.put(mode, vals); + } + + //formal, premium, less than 8 miles + if( (lotType == 1 && premiumModes.contains(mode) && dist < formalPremiumMaxD) || + //formal, express, less than 8 miles + (lotType == 1 && expressModes.contains(mode) && dist < formalExpressMaxD) || + //formal, local, less than 4 miles + (lotType == 1 && localModes.contains(mode) && dist < formalLocalMaxD) || + //informal, premium, less than 4 miles + (lotType > 1 && premiumModes.contains(mode) && dist < informalPremiumMaxD) || + //informal, express, less than 4 miles + (lotType > 1 && expressModes.contains(mode) && dist < informalExpressMaxD) || + //informal, local, less than 2 miles + (lotType > 1 && localModes.contains(mode) && dist < informalLocalMaxD) ){ + float[] vals = new float[4]; + vals[0] = j; + vals[1] = time; + vals[2] = dist; + vals[3] = mode; + bestTapsMap.get(currTAZ).add(vals); + addedTaps.add(j); + } + } + for(Integer m : modeMap.keySet()){ + float[] closestTAPvals = modeMap.get(m); + int tap = (int) closestTAPvals[0]; + if(!addedTaps.contains(tap)){ //Put best taps by mode into bestTapsMap if they are not already there + bestTapsMap.get(currTAZ).add(closestTAPvals); + addedTaps.add(tap); + } + } + } + } + + /*The file has five columns: TAZ, TAP, travel time (min) *100, distance (mile) *100 and mode. + */ + public void writeResults(String filename) throws IOException{ + BufferedWriter writer = null; + try{ + writer = new BufferedWriter(new FileWriter(new File(filename))); + + for(int i = 1; i < tazs.length; i++) { + int currTAZ = tazs[i]; + + //Skip externals + if(externalTAZs.contains(currTAZ)) + continue; + + if(bestTapsMap.get(currTAZ).size() > 0){ + //NEW CSV FORMAT (tabular: TAZ, TAP, TIME, DIST, MODE) + for(int k = 0; k < bestTapsMap.get(currTAZ).size(); k++){ + writer.write( (int)(currTAZ) + "," + + (int)(bestTapsMap.get(currTAZ).get(k)[0]) + "," + + (double)(bestTapsMap.get(currTAZ).get(k)[1]) + "," + + (double)(bestTapsMap.get(currTAZ).get(k)[2]) + "," + + (int)(bestTapsMap.get(currTAZ).get(k)[3]) + "\n"); + } + } + } + }finally{ + if (writer != null) writer.close(); + } + } + + + /** + * @param args + */ + public static void main(String... args) throws Exception + { + HashMap pMap; + String propertiesFile = null; + + logger.info("Generating access**.prp files"); + if (args.length == 0) + { + logger.error(String.format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + + Properties properties = new Properties(); + properties.load(new FileInputStream("conf/sandag_abm.properties")); + + List definedTables = new ArrayList(); + for (String table : properties.getProperty("Report.tables").trim().split(",")) + definedTables.add(table.trim().toLowerCase()); + + String path = ClassLoader.getSystemResource("").getPath(); + path = path.substring(1, path.length() - 2); + String appPath = path.substring(0, path.lastIndexOf("/")); + + for (Object key : properties.keySet()) + { + String value = (String) properties.get(key); + properties.setProperty((String) key, value.replace(PROJECT_PATH_PROPERTY_TOKEN, appPath)); + } + + OMXMatrixDao mtxDao = new OMXMatrixDao(properties); + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + SandagMGRAtoPNR accessWriter = new SandagMGRAtoPNR(properties, mtxDao, appPath, pMap); + accessWriter.nearestTAPs(); + accessWriter.writeResults(properties.getProperty("taz.driveaccess.taps.file")); + } +} + diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java new file mode 100644 index 0000000..ffe60cf --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java @@ -0,0 +1,52 @@ +package org.sandag.abm.application; + +import java.util.HashMap; + +import org.sandag.abm.ctramp.MicromobilityChoiceDMU; + +public class SandagMicromobilityChoiceDMU + extends MicromobilityChoiceDMU +{ + + public SandagMicromobilityChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getIvtCoeff", 0); + methodIndexMap.put("getCostCoeff", 1); + methodIndexMap.put("getWalkTime", 2); + methodIndexMap.put("getIsTransit", 3); + methodIndexMap.put("getMicroTransitAvailable", 4); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getIvtCoeff(); + case 1: + return getCostCoeff(); + case 2: + return getWalkTime(); + case 3: + return isTransit()? 1 : 0; + case 4: + return isMicroTransitAvailable() ? 1 : 0; + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java new file mode 100644 index 0000000..774844c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java @@ -0,0 +1,1299 @@ +package org.sandag.abm.application; + +import java.util.ArrayList; +import java.util.HashMap; + +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.Modes; + +public class SandagModelStructure + extends ModelStructure +{ + + public final String[] MANDATORY_DC_PURPOSE_NAMES = { + WORK_PURPOSE_NAME, UNIVERSITY_PURPOSE_NAME, SCHOOL_PURPOSE_NAME }; + public final String[] WORK_PURPOSE_SEGMENT_NAMES = { + "low", "med", "high", "very high", "part time" }; + public final String[] UNIVERSITY_PURPOSE_SEGMENT_NAMES = {}; + public final String[] SCHOOL_PURPOSE_SEGMENT_NAMES = { + "predrive", "drive" }; + + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_LO = 1; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_MD = 2; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_HI = 3; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_VHI = 4; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_PT = 5; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_UNIVERSITY_UNIVERSITY = 6; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_UNDER_SIXTEEN = 7; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_SIXTEEN_PLUS = 8; + + public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK = 1; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_UNIVERSITY = 2; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL = 3; + + public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK = 1; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_UNIVERSITY = 2; + public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL = 3; + + public final int MANDATORY_STOP_FREQ_UEC_INDEX_WORK = 1; + public final int MANDATORY_STOP_FREQ_UEC_INDEX_UNIVERSITY = 2; + public final int MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL = 3; + + public final int MANDATORY_STOP_LOC_UEC_INDEX_WORK = 1; + public final int MANDATORY_STOP_LOC_UEC_INDEX_UNIVERSITY = 1; + public final int MANDATORY_STOP_LOC_UEC_INDEX_SCHOOL = 1; + + public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_WORK = 1; + public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_UNIVERSITY = 2; + public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_SCHOOL = 3; + + public final String[] NON_MANDATORY_DC_PURPOSE_NAMES = { + "escort", "shopping", "eatOut", "othMaint", "visit", "othDiscr" }; + public final String[] ESCORT_PURPOSE_SEGMENT_NAMES = { + "kids", "no kids" }; + public final String[] SHOPPING_PURPOSE_SEGMENT_NAMES = {}; + public final String[] EAT_OUT_PURPOSE_SEGMENT_NAMES = {}; + public final String[] OTH_MAINT_PURPOSE_SEGMENT_NAMES = {}; + public final String[] SOCIAL_PURPOSE_SEGMENT_NAMES = {}; + public final String[] OTH_DISCR_PURPOSE_SEGMENT_NAMES = {}; + + /* + * public final int NON_MANDATORY_SOA_UEC_INDEX_ESCORT_KIDS = 9; public + * final int NON_MANDATORY_SOA_UEC_INDEX_ESCORT_NO_KIDS = 10; public final + * int NON_MANDATORY_SOA_UEC_INDEX_SHOPPING = 11; public final int + * NON_MANDATORY_SOA_UEC_INDEX_EAT_OUT = 12; public final int + * NON_MANDATORY_SOA_UEC_INDEX_OTHER_MAINT = 13; public final int + * NON_MANDATORY_SOA_UEC_INDEX_SOCIAL = 14; public final int + * NON_MANDATORY_SOA_UEC_INDEX_OTHER_DISCR = 15; + * + * public final int NON_MANDATORY_DC_UEC_INDEX_ESCORT_KIDS = 4; public final + * int NON_MANDATORY_DC_UEC_INDEX_ESCORT_NO_KIDS = 4; public final int + * NON_MANDATORY_DC_UEC_INDEX_SHOPPING = 5; public final int + * NON_MANDATORY_DC_UEC_INDEX_EAT_OUT = 6; public final int + * NON_MANDATORY_DC_UEC_INDEX_OTHER_MAINT = 7; public final int + * NON_MANDATORY_DC_UEC_INDEX_SOCIAL = 8; public final int + * NON_MANDATORY_DC_UEC_INDEX_OTHER_DISCR = 9; + * + * public final int NON_MANDATORY_MC_UEC_INDEX_ESCORT_KIDS = 4; public final + * int NON_MANDATORY_MC_UEC_INDEX_ESCORT_NO_KIDS = 4; public final int + * NON_MANDATORY_MC_UEC_INDEX_SHOPPING = 4; public final int + * NON_MANDATORY_MC_UEC_INDEX_EAT_OUT = 4; public final int + * NON_MANDATORY_MC_UEC_INDEX_OTHER_MAINT = 4; public final int + * NON_MANDATORY_MC_UEC_INDEX_SOCIAL = 4; public final int + * NON_MANDATORY_MC_UEC_INDEX_OTHER_DISCR = 4; + * + * public final int NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT = 4; public + * final int NON_MANDATORY_STOP_FREQ_UEC_INDEX_SHOPPING = 5; public final + * int NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_MAINT = 6; public final int + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_EAT_OUT = 7; public final int + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SOCIAL = 8; public final int + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_DISCR = 9; + * + * public final int NON_MANDATORY_STOP_LOC_UEC_INDEX_ESCORT = 2; public + * final int NON_MANDATORY_STOP_LOC_UEC_INDEX_SHOPPING = 3; public final int + * NON_MANDATORY_STOP_LOC_UEC_INDEX_EAT_OUT = 4; public final int + * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_MAINT = 5; public final int + * NON_MANDATORY_STOP_LOC_UEC_INDEX_SOCIAL = 6; public final int + * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_DISCR = 7; + * + * public final int NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX = 4; + */ + public final String[] AT_WORK_DC_PURPOSE_NAMES = {"atwork"}; + public final String[] AT_WORK_DC_SIZE_SEGMENT_NAMES = { + "cbd", "urban", "suburban", "rural" }; + + public final int AT_WORK_SOA_UEC_INDEX_EAT = 16; + public final int AT_WORK_SOA_UEC_INDEX_BUSINESS = 17; + public final int AT_WORK_SOA_UEC_INDEX_MAINT = 18; + + public final int AT_WORK_DC_UEC_INDEX_EAT = 10; + public final int AT_WORK_DC_UEC_INDEX_BUSINESS = 10; + public final int AT_WORK_DC_UEC_INDEX_MAINT = 10; + + public final int AT_WORK_MC_UEC_INDEX_EAT = 5; + public final int AT_WORK_MC_UEC_INDEX_BUSINESS = 5; + public final int AT_WORK_MC_UEC_INDEX_MAINT = 5; + + public final int SD_AT_WORK_PURPOSE_INDEX_EAT = 1; + public final int SD_AT_WORK_PURPOSE_INDEX_BUSINESS = 2; + public final int SD_AT_WORK_PURPOSE_INDEX_MAINT = 3; + + public final int AT_WORK_STOP_FREQ_UEC_INDEX_EAT = 9; + public final int AT_WORK_STOP_FREQ_UEC_INDEX_BUSINESS = 9; + public final int AT_WORK_STOP_FREQ_UEC_INDEX_MAINT = 9; + + // TODO: set these values from project specific code. + public static final int[] SOV_ALTS = { + 1 }; + public static final int[] HOV_ALTS = { + 2, 3 }; + public static final int[] HOV2_ALTS = { + 2 }; + public static final int[] HOV3_ALTS = { + 3 }; + public static final int[] WALK_ALTS = {4}; + public static final int[] BIKE_ALTS = {5}; + public static final int[] NON_MOTORIZED_ALTS = { + 4, 5 }; + public static final int[] TRANSIT_ALTS = { + 6, 7, 8, 9 }; + public static final int[] WALK_TRANSIT_ALTS = { + 6 }; + public static final int[] DRIVE_TRANSIT_ALTS = { + 7, 8, 9 }; + public static final int[] PNR_ALTS = { + 7 }; + public static final int[] KNR_ALTS = { + 8, 9 }; + + public static final int TNC_TRANSIT_ALT = 9; + + public static final int[] SCHOOL_BUS_ALTS = {13}; + public static final int[] TRIP_SOV_ALTS = { + 1 }; + public static final int[] TRIP_HOV_ALTS = { + 2,3 }; + + // public static final int[] PAY_ALTS = { + // 2, 4, 6 }; + + public static final int[] OTHER_ALTS = {10,11,12,13}; + + private static final int WALK = 4; + private static final int BIKE = 5; + + public static final int SCHOOL_BUS = 13; + public static final int TAXI = 10; + public static final int[] TNC_ALTS = {11,12}; + public static final int[] MAAS_ALTS = {10,11,12}; + + public static final String[] modeName = {"SOV","SR2","SR3", + "WALK","BIKE","WLK_SET","PNR_SET","KNR_SET","TNC_SET","TAXI","TNC_SINGLE","TNC_SHARED","SCHLBUS"}; + + public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 13; + + public final double[][] CDAP_6_PLUS_PROPORTIONS = { + {0.0, 0.0, 0.0}, {0.79647, 0.09368, 0.10985}, {0.61678, 0.25757, 0.12565}, + {0.69229, 0.15641, 0.15130}, {0.00000, 0.67169, 0.32831}, {0.00000, 0.54295, 0.45705}, + {0.77609, 0.06004, 0.16387}, {0.68514, 0.09144, 0.22342}, {0.14056, 0.06512, 0.79432} }; + + public static final String[] JTF_ALTERNATIVE_LABELS = { + "0_tours", "1_Shop", "1_Main", "1_Eat", "1_Visit", "1_Disc", "2_SS", "2_SM", "2_SE", + "2_SV", "2_SD", "2_MM", "2_ME", "2_MV", "2_MD", "2_EE", "2_EV", "2_ED", "2_VV", "2_VD", + "2_DD" }; + public static final String[] AWF_ALTERNATIVE_LABELS = { + "0_subTours", "1_eat", "1_business", "1_other", "2_business", "2 other", + "2_eat_business" }; + + public static final int MIN_DRIVING_AGE = 16; + + public static final int MAX_STOPS_PER_DIRECTION = 4; + + public SandagModelStructure() + { + super(); + + jtfAltLabels = JTF_ALTERNATIVE_LABELS; + awfAltLabels = AWF_ALTERNATIVE_LABELS; + + dcSizePurposeSegmentMap = new HashMap>(); + + dcSizeIndexSegmentMap = new HashMap(); + dcSizeSegmentIndexMap = new HashMap(); + dcSizeArrayIndexPurposeMap = new HashMap(); + dcSizeArrayPurposeIndexMap = new HashMap(); + + setMandatoryPurposeNameValues(); + + setUsualWorkAndSchoolLocationSoaUecSheetIndexValues(); + setUsualWorkAndSchoolLocationUecSheetIndexValues(); + setUsualWorkAndSchoolLocationModeChoiceUecSheetIndexValues(); + + setMandatoryStopFreqUecSheetIndexValues(); + setMandatoryStopLocUecSheetIndexValues(); + setMandatoryTripModeChoiceUecSheetIndexValues(); + + setNonMandatoryPurposeNameValues(); + + /* + * setNonMandatoryDcSoaUecSheetIndexValues(); + * setNonMandatoryDcUecSheetIndexValues(); + * setNonMandatoryModeChoiceUecSheetIndexValues(); + * + * setNonMandatoryStopFreqUecSheetIndexValues(); + * setNonMandatoryStopLocUecSheetIndexValues(); + * setNonMandatoryTripModeChoiceUecSheetIndexValues(); + */ + setAtWorkPurposeNameValues(); + + setAtWorkDcSoaUecSheetIndexValues(); + setAtWorkDcUecSheetIndexValues(); + setAtWorkModeChoiceUecSheetIndexValues(); + + setAtWorkStopFreqUecSheetIndexValues(); + + createDcSizePurposeSegmentMap(); + + // mapModelSegmentsToDcSizeArraySegments(); + + } + + /* + * private void mapModelSegmentsToDcSizeArraySegments() { + * + * Logger logger = Logger.getLogger(this.getClass()); + * + * dcSizeDcModelPurposeMap = new HashMap(); + * dcModelDcSizePurposeMap = new HashMap(); + * + * // loop over soa model names and map top dc size array indices for (int i + * = 0; i < dcModelPurposeIndexMap.size(); i++) { String modelSegment = + * dcModelIndexPurposeMap.get(i); + * + * // look for this modelSegment name in the dc size array names map, with + * // and without "_segment". if + * (dcSizeArrayPurposeIndexMap.containsKey(modelSegment)) { + * dcSizeDcModelPurposeMap.put(modelSegment, modelSegment); + * dcModelDcSizePurposeMap.put(modelSegment, modelSegment); } else { int + * underscoreIndex = modelSegment.indexOf('_'); if (underscoreIndex < 0) { + * if (dcSizeArrayPurposeIndexMap.containsKey(modelSegment + "_" + + * modelSegment)) { dcSizeDcModelPurposeMap .put(modelSegment + "_" + + * modelSegment, modelSegment); dcModelDcSizePurposeMap .put(modelSegment, + * modelSegment + "_" + modelSegment); } else { logger .error(String + * .format( + * "could not establish correspondence between DC SOA model purpose string = %s" + * , modelSegment)); + * logger.error(String.format("and a DC array purpose string:")); int j = 0; + * for (String key : dcSizeArrayPurposeIndexMap.keySet()) + * logger.error(String.format("%-2d: %s", ++j, key)); throw new + * RuntimeException(); } } else { // all at-work size segments should map to + * one model segment if (modelSegment.substring(0, + * underscoreIndex).equalsIgnoreCase( AT_WORK_PURPOSE_NAME)) { + * dcSizeDcModelPurposeMap.put(AT_WORK_PURPOSE_NAME + "_" + + * AT_WORK_PURPOSE_NAME, modelSegment); + * dcModelDcSizePurposeMap.put(modelSegment, AT_WORK_PURPOSE_NAME + "_" + + * AT_WORK_PURPOSE_NAME); } else { logger .error(String .format( + * "could not establish correspondence between DC SOA model purpose string = %s" + * , modelSegment)); + * logger.error(String.format("and a DC array purpose string:")); int j = 0; + * for (String key : dcSizeArrayPurposeIndexMap.keySet()) + * logger.error(String.format("%-2d: %s", ++j, key)); throw new + * RuntimeException(); } } } + * + * } + * + * } + */ + + public String getSchoolPurpose(int age) + { + if (age < MIN_DRIVING_AGE) return (schoolPurposeName + "_" + SCHOOL_PURPOSE_SEGMENT_NAMES[0]) + .toLowerCase(); + else return (schoolPurposeName + "_" + SCHOOL_PURPOSE_SEGMENT_NAMES[1]).toLowerCase(); + } + + public String getSchoolPurpose() + { + return schoolPurposeName.toLowerCase(); + } + + public String getUniversityPurpose() + { + return universityPurposeName.toLowerCase(); + } + + public String getWorkPurpose(int incomeCategory) + { + return getWorkPurpose(false, incomeCategory); + } + + public String getWorkPurpose(boolean isPtWorker, int incomeCategory) + { + if (isPtWorker) return (workPurposeName + "_" + WORK_PURPOSE_SEGMENT_NAMES[WORK_PURPOSE_SEGMENT_NAMES.length - 1]) + .toLowerCase(); + else return (workPurposeName + "_" + WORK_PURPOSE_SEGMENT_NAMES[incomeCategory - 1]) + .toLowerCase(); + } + + public boolean getTripModeIsSovOrHov(int tripMode) + { + + for (int i = 0; i < TRIP_SOV_ALTS.length; i++) + { + if (TRIP_SOV_ALTS[i] == tripMode) return true; + } + + for (int i = 0; i < TRIP_HOV_ALTS.length; i++) + { + if (TRIP_HOV_ALTS[i] == tripMode) return true; + } + + return false; + } + + public boolean getTourModeIsSov(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < SOV_ALTS.length; i++) + { + if (SOV_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsHov(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < HOV_ALTS.length; i++) + { + if (HOV_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsS2(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < HOV2_ALTS.length; i++) + { + if (HOV2_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsS3(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < HOV3_ALTS.length; i++) + { + if (HOV3_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsSovOrHov(int tourMode) + { + for (int i = 0; i < SOV_ALTS.length; i++) + { + if (SOV_ALTS[i] == tourMode) return true; + } + + for (int i = 0; i < HOV_ALTS.length; i++) + { + if (HOV_ALTS[i] == tourMode) return true; + } + + // if (tourMode == TAXI) return true; + + return false; + } + + public boolean getTourModeIsNonMotorized(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < NON_MOTORIZED_ALTS.length; i++) + { + if (NON_MOTORIZED_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsBike(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < BIKE_ALTS.length; i++) + { + if (BIKE_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsWalk(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < WALK_ALTS.length; i++) + { + if (WALK_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + + public boolean getTourModeIsTransit(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < TRANSIT_ALTS.length; i++) + { + if (TRANSIT_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsWalkTransit(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < WALK_TRANSIT_ALTS.length; ++i) + { + if (WALK_TRANSIT_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsDriveTransit(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < DRIVE_TRANSIT_ALTS.length; i++) + { + if (DRIVE_TRANSIT_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsPnr(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < PNR_ALTS.length; i++) + { + if (PNR_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsKnr(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < KNR_ALTS.length; i++) + { + if (KNR_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTourModeIsTncTransit(int tourMode) + { + if (TNC_TRANSIT_ALT == tourMode) + return true; + else + return false; + } + + public boolean getTourModeIsMaas(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < MAAS_ALTS.length; i++) + { + if (MAAS_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + public boolean getTourModeIsSchoolBus(int tourMode) + { + boolean returnValue = false; + for (int i = 0; i < SCHOOL_BUS_ALTS.length; i++) + { + if (SCHOOL_BUS_ALTS[i] == tourMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public static boolean getTripModeIsPay(int tripMode) + { + boolean returnValue = false; + /* for (int i = 0; i < PAY_ALTS.length; i++) + { + if (PAY_ALTS[i] == tripMode) + { + returnValue = true; + break; + } + } +*/ + return returnValue; + } + + public boolean getTripModeIsTransit(int tripMode){ + + boolean returnValue = false; + for( int i = 0; i < TRANSIT_ALTS.length;++i ) + if(tripMode==TRANSIT_ALTS[i]){ + returnValue = true; + break; + } + + return returnValue; + } + + /** + * Get the name of the mode + * + * @param mode + * The mode index + * @return The name of the mode + */ + public String getModeName(int mode) + { + + return modeName[mode - 1]; + + } + + private int createPurposeIndexMaps(String purposeName, String[] segmentNames, int index, + String categoryString) + { + + HashMap segmentMap = new HashMap(); + String key = ""; + if (segmentNames.length > 0) + { + for (int i = 0; i < segmentNames.length; i++) + { + segmentMap.put(segmentNames[i].toLowerCase(), i); + key = purposeName.toLowerCase() + "_" + segmentNames[i].toLowerCase(); + dcSizeIndexSegmentMap.put(index, key); + dcSizeSegmentIndexMap.put(key, index++); + } + } else + { + segmentMap.put(purposeName.toLowerCase(), 0); + key = purposeName.toLowerCase() + "_" + purposeName.toLowerCase(); + dcSizeIndexSegmentMap.put(index, key); + dcSizeSegmentIndexMap.put(key, index++); + } + dcSizePurposeSegmentMap.put(purposeName.toLowerCase(), segmentMap); + + return index; + + } + + /** + * This method defines the segmentation for which destination choice size + * variables are calculated. + */ + private void createDcSizePurposeSegmentMap() + { + + int index = 0; + + // put work purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(WORK_PURPOSE_NAME, WORK_PURPOSE_SEGMENT_NAMES, index, + MANDATORY_CATEGORY); + + // put university purpose segments, by which DC Size calculations are + // segmented, into a map to be stored by purpose name. + index = createPurposeIndexMaps(UNIVERSITY_PURPOSE_NAME, UNIVERSITY_PURPOSE_SEGMENT_NAMES, + index, MANDATORY_CATEGORY); + + // put school purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(SCHOOL_PURPOSE_NAME, SCHOOL_PURPOSE_SEGMENT_NAMES, index, + MANDATORY_CATEGORY); + + // put escort purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(ESCORT_PURPOSE_NAME, ESCORT_PURPOSE_SEGMENT_NAMES, index, + INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put shopping purpose segments, by which DC Size calculations are + // segmented, into a map to be stored by purpose name. + index = createPurposeIndexMaps(SHOPPING_PURPOSE_NAME, SHOPPING_PURPOSE_SEGMENT_NAMES, + index, INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put eat out purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(EAT_OUT_PURPOSE_NAME, EAT_OUT_PURPOSE_SEGMENT_NAMES, index, + INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put oth main purpose segments, by which DC Size calculations are + // segmented, into a map to be stored by purpose name. + index = createPurposeIndexMaps(OTH_MAINT_PURPOSE_NAME, OTH_MAINT_PURPOSE_SEGMENT_NAMES, + index, INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put social purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(SOCIAL_PURPOSE_NAME, SOCIAL_PURPOSE_SEGMENT_NAMES, index, + INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put oth discr purpose segments, by which DC Size calculations are + // segmented, into a map to be stored by purpose name. + index = createPurposeIndexMaps(OTH_DISCR_PURPOSE_NAME, OTH_DISCR_PURPOSE_SEGMENT_NAMES, + index, INDIVIDUAL_NON_MANDATORY_CATEGORY); + + // put at work purpose segments, by which DC Size calculations are + // segmented, + // into a map to be stored by purpose name. + index = createPurposeIndexMaps(AT_WORK_PURPOSE_NAME, AT_WORK_DC_SIZE_SEGMENT_NAMES, index, + AT_WORK_CATEGORY); + + } + + public HashMap> getDcSizePurposeSegmentMap() + { + return dcSizePurposeSegmentMap; + } + + private void setMandatoryPurposeNameValues() + { + + int index = 0; + + WORK_PURPOSE_NAME = "work"; + UNIVERSITY_PURPOSE_NAME = "university"; + SCHOOL_PURPOSE_NAME = "school"; + + int numDcSizePurposeSegments = 0; + if (WORK_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += WORK_PURPOSE_SEGMENT_NAMES.length; + else numDcSizePurposeSegments += 1; + if (UNIVERSITY_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += UNIVERSITY_PURPOSE_SEGMENT_NAMES.length; + else numDcSizePurposeSegments += 1; + if (SCHOOL_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += SCHOOL_PURPOSE_SEGMENT_NAMES.length; + else numDcSizePurposeSegments += 1; + + mandatoryDcModelPurposeNames = new String[numDcSizePurposeSegments]; + + workPurposeName = WORK_PURPOSE_NAME.toLowerCase(); + workPurposeSegmentNames = new String[WORK_PURPOSE_SEGMENT_NAMES.length]; + if (workPurposeSegmentNames.length > 0) + { + for (int i = 0; i < WORK_PURPOSE_SEGMENT_NAMES.length; i++) + { + workPurposeSegmentNames[i] = WORK_PURPOSE_SEGMENT_NAMES[i].toLowerCase(); + mandatoryDcModelPurposeNames[index] = workPurposeName + "_" + + workPurposeSegmentNames[i]; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each work + // purpose_segment + dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + index++; + } + } else + { + mandatoryDcModelPurposeNames[index] = workPurposeName; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each work purpose_segment + String name = mandatoryDcModelPurposeNames[index] + "_" + + mandatoryDcModelPurposeNames[index]; + dcSizeArrayIndexPurposeMap.put(index, name); + dcSizeArrayPurposeIndexMap.put(name, index); + index++; + } + + universityPurposeName = UNIVERSITY_PURPOSE_NAME.toLowerCase(); + universityPurposeSegmentNames = new String[UNIVERSITY_PURPOSE_SEGMENT_NAMES.length]; + if (universityPurposeSegmentNames.length > 0) + { + for (int i = 0; i < universityPurposeSegmentNames.length; i++) + { + universityPurposeSegmentNames[i] = UNIVERSITY_PURPOSE_SEGMENT_NAMES[i] + .toLowerCase(); + mandatoryDcModelPurposeNames[index] = universityPurposeName + "_" + + universityPurposeSegmentNames[i]; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each university + // purpose_segment + dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + index++; + } + } else + { + mandatoryDcModelPurposeNames[index] = universityPurposeName; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each university + // purpose_segment + String name = mandatoryDcModelPurposeNames[index] + "_" + + mandatoryDcModelPurposeNames[index]; + dcSizeArrayIndexPurposeMap.put(index, name); + dcSizeArrayPurposeIndexMap.put(name, index); + index++; + } + + schoolPurposeName = SCHOOL_PURPOSE_NAME.toLowerCase(); + schoolPurposeSegmentNames = new String[SCHOOL_PURPOSE_SEGMENT_NAMES.length]; + if (schoolPurposeSegmentNames.length > 0) + { + for (int i = 0; i < schoolPurposeSegmentNames.length; i++) + { + schoolPurposeSegmentNames[i] = SCHOOL_PURPOSE_SEGMENT_NAMES[i].toLowerCase(); + mandatoryDcModelPurposeNames[index] = schoolPurposeName + "_" + + schoolPurposeSegmentNames[i]; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each school + // purpose_segment + dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + index++; + } + } else + { + mandatoryDcModelPurposeNames[index] = schoolPurposeName; + dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); + dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); + + // a separate size term is calculated for each school + // purpose_segment + String name = mandatoryDcModelPurposeNames[index] + "_" + + mandatoryDcModelPurposeNames[index]; + dcSizeArrayIndexPurposeMap.put(index, name); + dcSizeArrayPurposeIndexMap.put(name, index); + } + + } + + private void setUsualWorkAndSchoolLocationSoaUecSheetIndexValues() + { + dcSoaUecIndexMap.put("work_low", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_LO); + dcSoaUecIndexMap.put("work_med", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_MD); + dcSoaUecIndexMap.put("work_high", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_HI); + dcSoaUecIndexMap.put("work_very high", + USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_VHI); + dcSoaUecIndexMap + .put("work_part time", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_PT); + dcSoaUecIndexMap.put("university", + USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_UNIVERSITY_UNIVERSITY); + dcSoaUecIndexMap.put("school_predrive", + USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_UNDER_SIXTEEN); + dcSoaUecIndexMap.put("school_drive", + USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_SIXTEEN_PLUS); + } + + private void setUsualWorkAndSchoolLocationUecSheetIndexValues() + { + dcUecIndexMap.put("work_low", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); + dcUecIndexMap.put("work_med", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); + dcUecIndexMap.put("work_high", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); + dcUecIndexMap.put("work_very high", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); + dcUecIndexMap.put("work_part time", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); + dcUecIndexMap.put("university", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_UNIVERSITY); + dcUecIndexMap.put("school_predrive", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL); + dcUecIndexMap.put("school_drive", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL); + } + + private void setUsualWorkAndSchoolLocationModeChoiceUecSheetIndexValues() + { + tourModeChoiceUecIndexMap.put("work_low", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); + tourModeChoiceUecIndexMap.put("work_med", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); + tourModeChoiceUecIndexMap.put("work_high", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); + tourModeChoiceUecIndexMap.put("work_very high", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); + tourModeChoiceUecIndexMap.put("work_part time", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); + tourModeChoiceUecIndexMap.put("university", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_UNIVERSITY); + tourModeChoiceUecIndexMap.put("school_predrive", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL); + tourModeChoiceUecIndexMap.put("school_drive", + USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL); + } + + private void setMandatoryStopFreqUecSheetIndexValues() + { + stopFreqUecIndexMap.put("work_low", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); + stopFreqUecIndexMap.put("work_med", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); + stopFreqUecIndexMap.put("work_high", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); + stopFreqUecIndexMap.put("work_very high", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); + stopFreqUecIndexMap.put("work_part time", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); + stopFreqUecIndexMap.put("university", MANDATORY_STOP_FREQ_UEC_INDEX_UNIVERSITY); + stopFreqUecIndexMap.put("school_predrive", MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL); + stopFreqUecIndexMap.put("school_drive", MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL); + } + + private void setMandatoryStopLocUecSheetIndexValues() + { + stopLocUecIndexMap.put(WORK_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); + stopLocUecIndexMap.put(UNIVERSITY_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); + stopLocUecIndexMap.put(SCHOOL_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); + } + + private void setMandatoryTripModeChoiceUecSheetIndexValues() + { + tripModeChoiceUecIndexMap.put(WORK_PURPOSE_NAME, MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_WORK); + tripModeChoiceUecIndexMap.put(UNIVERSITY_PURPOSE_NAME, + MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_UNIVERSITY); + tripModeChoiceUecIndexMap.put(SCHOOL_PURPOSE_NAME, + MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_SCHOOL); + } + + private void setNonMandatoryPurposeNameValues() + { + + ESCORT_PURPOSE_NAME = "escort"; + SHOPPING_PURPOSE_NAME = "shopping"; + EAT_OUT_PURPOSE_NAME = "eatout"; + OTH_MAINT_PURPOSE_NAME = "othmaint"; + SOCIAL_PURPOSE_NAME = "visit"; + OTH_DISCR_PURPOSE_NAME = "othdiscr"; + + // initialize index to the length of the mandatory names list already + // developed. + int index = dcSizeArrayPurposeIndexMap.size(); + + ESCORT_SEGMENT_NAMES = ESCORT_PURPOSE_SEGMENT_NAMES; + + // ESCORT is the only non-mandatory purpose with segments + ArrayList purposeNamesList = new ArrayList(); + for (int i = 0; i < NON_MANDATORY_DC_PURPOSE_NAMES.length; i++) + { + if (NON_MANDATORY_DC_PURPOSE_NAMES[i].equalsIgnoreCase(ESCORT_PURPOSE_NAME)) + { + for (int j = 0; j < ESCORT_SEGMENT_NAMES.length; j++) + { + String name = (ESCORT_PURPOSE_NAME + "_" + ESCORT_SEGMENT_NAMES[j]) + .toLowerCase(); + purposeNamesList.add(name); + dcModelPurposeIndexMap.put(name, index); + dcModelIndexPurposeMap.put(index, name); + + // a separate size term is calculated for each non-mandatory + // purpose_segment + dcSizeArrayIndexPurposeMap.put(index, name); + dcSizeArrayPurposeIndexMap.put(name, index); + index++; + } + } else + { + String name = NON_MANDATORY_DC_PURPOSE_NAMES[i].toLowerCase(); + purposeNamesList.add(name); + dcModelPurposeIndexMap.put(name, index); + dcModelIndexPurposeMap.put(index, name); + + // a separate size term is calculated for each non-mandatory + // purpose_segment + dcSizeArrayIndexPurposeMap.put(index, name + "_" + name); + dcSizeArrayPurposeIndexMap.put(name + "_" + name, index); + index++; + } + } + + int escortOffset = ESCORT_SEGMENT_NAMES.length; + + jointDcModelPurposeNames = new String[purposeNamesList.size() - escortOffset]; + nonMandatoryDcModelPurposeNames = new String[purposeNamesList.size()]; + for (int i = 0; i < purposeNamesList.size(); i++) + { + nonMandatoryDcModelPurposeNames[i] = purposeNamesList.get(i); + if (i > escortOffset - 1) + jointDcModelPurposeNames[i - escortOffset] = purposeNamesList.get(i); + } + + } + + /* + * private void setNonMandatoryDcSoaUecSheetIndexValues() { + * dcSoaUecIndexMap.put("escort_kids", + * NON_MANDATORY_SOA_UEC_INDEX_ESCORT_KIDS); + * dcSoaUecIndexMap.put("escort_no kids", + * NON_MANDATORY_SOA_UEC_INDEX_ESCORT_NO_KIDS); + * dcSoaUecIndexMap.put("shopping", NON_MANDATORY_SOA_UEC_INDEX_SHOPPING); + * dcSoaUecIndexMap.put("eatout", NON_MANDATORY_SOA_UEC_INDEX_EAT_OUT); + * dcSoaUecIndexMap.put("othmaint", + * NON_MANDATORY_SOA_UEC_INDEX_OTHER_MAINT); dcSoaUecIndexMap.put("social", + * NON_MANDATORY_SOA_UEC_INDEX_SOCIAL); dcSoaUecIndexMap.put("othdiscr", + * NON_MANDATORY_SOA_UEC_INDEX_OTHER_DISCR); } + * + * private void setNonMandatoryDcUecSheetIndexValues() { + * dcUecIndexMap.put("escort_kids", NON_MANDATORY_DC_UEC_INDEX_ESCORT_KIDS); + * dcUecIndexMap.put("escort_no kids", + * NON_MANDATORY_DC_UEC_INDEX_ESCORT_NO_KIDS); dcUecIndexMap.put("shopping", + * NON_MANDATORY_DC_UEC_INDEX_SHOPPING); dcUecIndexMap.put("eatout", + * NON_MANDATORY_DC_UEC_INDEX_EAT_OUT); dcUecIndexMap.put("othmaint", + * NON_MANDATORY_DC_UEC_INDEX_OTHER_MAINT); dcUecIndexMap.put("social", + * NON_MANDATORY_DC_UEC_INDEX_SOCIAL); dcUecIndexMap.put("othdiscr", + * NON_MANDATORY_DC_UEC_INDEX_OTHER_DISCR); } + * + * private void setNonMandatoryModeChoiceUecSheetIndexValues() { + * tourModeChoiceUecIndexMap.put("escort_kids", + * NON_MANDATORY_MC_UEC_INDEX_ESCORT_KIDS); + * tourModeChoiceUecIndexMap.put("escort_no kids", + * NON_MANDATORY_MC_UEC_INDEX_ESCORT_NO_KIDS); + * tourModeChoiceUecIndexMap.put("shopping", + * NON_MANDATORY_MC_UEC_INDEX_SHOPPING); + * tourModeChoiceUecIndexMap.put("eatout", + * NON_MANDATORY_MC_UEC_INDEX_EAT_OUT); + * tourModeChoiceUecIndexMap.put("othmaint", + * NON_MANDATORY_MC_UEC_INDEX_OTHER_MAINT); + * tourModeChoiceUecIndexMap.put("social", + * NON_MANDATORY_MC_UEC_INDEX_SOCIAL); + * tourModeChoiceUecIndexMap.put("othdiscr", + * NON_MANDATORY_MC_UEC_INDEX_OTHER_DISCR); } + * + * private void setNonMandatoryStopFreqUecSheetIndexValues() { + * stopFreqUecIndexMap.put("escort_kids", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT); + * stopFreqUecIndexMap.put("escort_no kids", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT); + * stopFreqUecIndexMap.put("shopping", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SHOPPING); + * stopFreqUecIndexMap.put("eatout", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_EAT_OUT); + * stopFreqUecIndexMap.put("othmaint", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_MAINT); + * stopFreqUecIndexMap.put("social", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SOCIAL); + * stopFreqUecIndexMap.put("othdiscr", + * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_DISCR); } + * + * private void setNonMandatoryStopLocUecSheetIndexValues() { + * stopLocUecIndexMap.put(ESCORT_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_ESCORT); + * stopLocUecIndexMap.put(SHOPPING_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_SHOPPING); + * stopLocUecIndexMap.put(EAT_OUT_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_EAT_OUT); stopLocUecIndexMap + * .put(OTH_MAINT_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_MAINT); + * stopLocUecIndexMap.put(SOCIAL_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_SOCIAL); stopLocUecIndexMap + * .put(OTH_DISCR_PURPOSE_NAME, + * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_DISCR); } + * + * private void setNonMandatoryTripModeChoiceUecSheetIndexValues() { + * tripModeChoiceUecIndexMap .put(ESCORT_PURPOSE_NAME, + * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); + * tripModeChoiceUecIndexMap.put(SHOPPING_PURPOSE_NAME, + * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); + * tripModeChoiceUecIndexMap.put(EAT_OUT_PURPOSE_NAME, + * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); + * tripModeChoiceUecIndexMap.put(OTH_MAINT_PURPOSE_NAME, + * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); tripModeChoiceUecIndexMap + * .put(SOCIAL_PURPOSE_NAME, NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); + * tripModeChoiceUecIndexMap.put(OTH_DISCR_PURPOSE_NAME, + * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); } + */ + private void setAtWorkPurposeNameValues() + { + + AT_WORK_PURPOSE_NAME = "atwork"; + + AT_WORK_EAT_PURPOSE_NAME = "eat"; + AT_WORK_BUSINESS_PURPOSE_NAME = "business"; + AT_WORK_MAINT_PURPOSE_NAME = "other"; + + AT_WORK_PURPOSE_INDEX_EAT = SD_AT_WORK_PURPOSE_INDEX_EAT; + AT_WORK_PURPOSE_INDEX_BUSINESS = SD_AT_WORK_PURPOSE_INDEX_BUSINESS; + AT_WORK_PURPOSE_INDEX_MAINT = SD_AT_WORK_PURPOSE_INDEX_MAINT; + + AT_WORK_SEGMENT_NAMES = new String[3]; + AT_WORK_SEGMENT_NAMES[0] = AT_WORK_EAT_PURPOSE_NAME; + AT_WORK_SEGMENT_NAMES[1] = AT_WORK_BUSINESS_PURPOSE_NAME; + AT_WORK_SEGMENT_NAMES[2] = AT_WORK_MAINT_PURPOSE_NAME; + + // initialize index to the length of the home-based tour names list + // already + // developed. + int index = dcSizeArrayPurposeIndexMap.size(); + + // the same size term is used by each at-work soa model + dcSizeArrayIndexPurposeMap.put(index, AT_WORK_PURPOSE_NAME + "_" + AT_WORK_PURPOSE_NAME); + dcSizeArrayPurposeIndexMap.put(AT_WORK_PURPOSE_NAME + "_" + AT_WORK_PURPOSE_NAME, index); + + ArrayList purposeNamesList = new ArrayList(); + for (int j = 0; j < AT_WORK_SEGMENT_NAMES.length; j++) + { + String name = (AT_WORK_PURPOSE_NAME + "_" + AT_WORK_SEGMENT_NAMES[j]).toLowerCase(); + purposeNamesList.add(name); + dcModelPurposeIndexMap.put(name, index); + dcModelIndexPurposeMap.put(index, name); + index++; + } + + atWorkDcModelPurposeNames = new String[purposeNamesList.size()]; + for (int i = 0; i < purposeNamesList.size(); i++) + { + atWorkDcModelPurposeNames[i] = purposeNamesList.get(i); + } + + } + + private void setAtWorkDcSoaUecSheetIndexValues() + { + dcSoaUecIndexMap.put("atwork_eat", AT_WORK_SOA_UEC_INDEX_EAT); + dcSoaUecIndexMap.put("atwork_business", AT_WORK_SOA_UEC_INDEX_BUSINESS); + dcSoaUecIndexMap.put("atwork_other", AT_WORK_SOA_UEC_INDEX_MAINT); + } + + private void setAtWorkDcUecSheetIndexValues() + { + dcUecIndexMap.put("atwork_eat", AT_WORK_DC_UEC_INDEX_EAT); + dcUecIndexMap.put("atwork_business", AT_WORK_DC_UEC_INDEX_BUSINESS); + dcUecIndexMap.put("atwork_other", AT_WORK_DC_UEC_INDEX_MAINT); + } + + private void setAtWorkModeChoiceUecSheetIndexValues() + { + tourModeChoiceUecIndexMap.put("atwork_eat", AT_WORK_MC_UEC_INDEX_EAT); + tourModeChoiceUecIndexMap.put("atwork_business", AT_WORK_MC_UEC_INDEX_BUSINESS); + tourModeChoiceUecIndexMap.put("atwork_other", AT_WORK_MC_UEC_INDEX_MAINT); + } + + private void setAtWorkStopFreqUecSheetIndexValues() + { + stopFreqUecIndexMap.put("atwork_eat", AT_WORK_STOP_FREQ_UEC_INDEX_EAT); + stopFreqUecIndexMap.put("atwork_business", AT_WORK_STOP_FREQ_UEC_INDEX_BUSINESS); + stopFreqUecIndexMap.put("atwork_other", AT_WORK_STOP_FREQ_UEC_INDEX_MAINT); + } + + public double[][] getCdap6PlusProps() + { + return CDAP_6_PLUS_PROPORTIONS; + } + + public String getModelPeriodLabel(int period) + { + return MODEL_PERIOD_LABELS[period]; + } + + public int getNumberModelPeriods() + { + return MODEL_PERIOD_LABELS.length; + } + + public String getSkimMatrixPeriodString(int period) + { + int index = getSkimPeriodIndex(period); + return SKIM_PERIOD_STRINGS[index]; + } + + public int getDefaultAmPeriod() + { + return getTimePeriodIndexForTime(800); + } + + public int getDefaultPmPeriod() + { + return getTimePeriodIndexForTime(1700); + } + + public int getDefaultMdPeriod() + { + return getTimePeriodIndexForTime(1400); + } + + public int[] getSkimPeriodCombinationIndices() + { + return SKIM_PERIOD_COMBINATION_INDICES; + } + + public int getSkimPeriodCombinationIndex(int startPeriod, int endPeriod) + { + + int startPeriodIndex = getSkimPeriodIndex(startPeriod); + int endPeriodIndex = getSkimPeriodIndex(endPeriod); + + if (SKIM_PERIOD_COMBINATIONS[startPeriodIndex][endPeriodIndex] < 0) + { + String errorString = String + .format("startPeriod=%d, startPeriod=%d, endPeriod=%d, endPeriod=%d is invalid combination.", + startPeriod, startPeriodIndex, endPeriod, endPeriodIndex); + throw new RuntimeException(errorString); + } else + { + return SKIM_PERIOD_COMBINATIONS[startPeriodIndex][endPeriodIndex]; + } + + } + + public int getMaxTourModeIndex() + { + return MAXIMUM_TOUR_MODE_ALT_INDEX; + } + + public HashMap getWorkSegmentNameIndexMap() + { + return workSegmentNameIndexMap; + } + + public void setWorkSegmentNameIndexMap(HashMap argMap) + { + workSegmentNameIndexMap = argMap; + } + + public HashMap getSchoolSegmentNameIndexMap() + { + return schoolSegmentNameIndexMap; + } + + public void setSchoolSegmentNameIndexMap(HashMap argMap) + { + schoolSegmentNameIndexMap = argMap; + } + + public HashMap getWorkSegmentIndexNameMap() + { + return workSegmentIndexNameMap; + } + + public void setWorkSegmentIndexNameMap(HashMap argMap) + { + workSegmentIndexNameMap = argMap; + } + + public HashMap getSchoolSegmentIndexNameMap() + { + return schoolSegmentIndexNameMap; + } + + public void setSchoolSegmentIndexNameMap(HashMap argMap) + { + schoolSegmentIndexNameMap = argMap; + } + + public void setJtfAltLabels(String[] labels) + { + jtfAltLabels = labels; + } + + public String[] getJtfAltLabels() + { + return jtfAltLabels; + } + + public boolean getTripModeIsWalkTransit(int tripMode) + { + + for (int i = 0; i < WALK_TRANSIT_ALTS.length; i++) + { + if (WALK_TRANSIT_ALTS[i] == tripMode) return true; + } + + return false; + } + + public boolean getTripModeIsPnrTransit(int tripMode) + { + + for (int i = 0; i < PNR_ALTS.length; i++) + { + if (PNR_ALTS[i] == tripMode) return true; + } + + return false; + } + + public boolean getTripModeIsKnrTransit(int tripMode) + { + + for (int i = 0; i < KNR_ALTS.length; i++) + { + if (KNR_ALTS[i] == tripMode) return true; + } + + return false; + } + + public boolean getTripModeIsNonMotorized(int i) + { + + if (i == WALK || i == BIKE) return true; + else return false; + } + + public boolean getTripModeIsS2(int tripMode) + { + boolean returnValue = false; + for (int i = 0; i < HOV2_ALTS.length; i++) + { + if (HOV2_ALTS[i] == tripMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public boolean getTripModeIsS3(int tripMode) + { + boolean returnValue = false; + for (int i = 0; i < HOV3_ALTS.length; i++) + { + if (HOV3_ALTS[i] == tripMode) + { + returnValue = true; + break; + } + } + return returnValue; + } + + public int getMaxStopsPerDirection(){ + + return MAX_STOPS_PER_DIRECTION; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java new file mode 100644 index 0000000..99fb216 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java @@ -0,0 +1,94 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.ParkingChoiceDMU; + +public class SandagParkingChoiceDMU + extends ParkingChoiceDMU +{ + + public SandagParkingChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getParkMgraAlt", 1); + methodIndexMap.put("getDistanceTripOrigToParkAlt", 2); + methodIndexMap.put("getDistanceTripDestFromParkAlt", 3); + methodIndexMap.put("getDestSameAsParkAlt", 4); + methodIndexMap.put("getPersonType", 5); + methodIndexMap.put("getActivityIntervals", 6); + methodIndexMap.put("getTripDestPurpose", 7); + methodIndexMap.put("getLsWgtAvgCostM", 8); + methodIndexMap.put("getMstallsoth", 9); + methodIndexMap.put("getMstallssam", 10); + methodIndexMap.put("getMparkcost", 11); + methodIndexMap.put("getDstallsoth", 12); + methodIndexMap.put("getDstallssam", 13); + methodIndexMap.put("getDparkcost", 14); + methodIndexMap.put("getHstallsoth", 15); + methodIndexMap.put("getHstallssam", 16); + methodIndexMap.put("getHparkcost", 17); + methodIndexMap.put("getNumfreehrs", 18); + methodIndexMap.put("getReimbPct", 19); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + + case 1: + return getParkMgraAlt(arrayIndex); + case 2: + return getDistanceTripOrigToParkAlt(arrayIndex); + case 3: + return getDistanceTripDestFromParkAlt(arrayIndex); + case 4: + return getDestSameAsParkAlt(arrayIndex); + case 5: + return getPersonType(); + case 6: + return getActivityIntervals(); + case 7: + return getTripDestPurpose(); + case 8: + return getLsWgtAvgCostM(arrayIndex); + case 9: + return getMstallsoth(arrayIndex); + case 10: + return getMstallssam(arrayIndex); + case 11: + return getMparkcost(arrayIndex); + case 12: + return getDstallsoth(arrayIndex); + case 13: + return getDstallssam(arrayIndex); + case 14: + return getDparkcost(arrayIndex); + case 15: + return getHstallsoth(arrayIndex); + case 16: + return getHstallssam(arrayIndex); + case 17: + return getHparkcost(arrayIndex); + case 18: + return getNumfreehrs(arrayIndex); + case 19: + return getReimbPct(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java new file mode 100644 index 0000000..a3c242e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java @@ -0,0 +1,83 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.ParkingProvisionChoiceDMU; + +public class SandagParkingProvisionChoiceDMU + extends ParkingProvisionChoiceDMU +{ + + public SandagParkingProvisionChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getHhIncomeInDollars", 0); + methodIndexMap.put("getWorkLocMgra", 1); + methodIndexMap.put("getLsWgtAvgCostM", 2); + methodIndexMap.put("getLsWgtAvgCostD", 3); + methodIndexMap.put("getLsWgtAvgCostH", 4); + methodIndexMap.put("getMgraParkArea", 5); + methodIndexMap.put("getNumFreeHours", 6); + methodIndexMap.put("getMStallsOth", 7); + methodIndexMap.put("getMStallsSam", 8); + methodIndexMap.put("getMParkCost", 9); + methodIndexMap.put("getDStallsOth", 10); + methodIndexMap.put("getDStallsSam", 11); + methodIndexMap.put("getDParkCost", 12); + methodIndexMap.put("getHStallsOth", 13); + methodIndexMap.put("getHStallsSam", 14); + methodIndexMap.put("getHParkCost", 15); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getIncomeInDollars(); + case 1: + return getWorkLocMgra(); + case 2: + return getLsWgtAvgCostM(); + case 3: + return getLsWgtAvgCostD(); + case 4: + return getLsWgtAvgCostH(); + case 5: + return getMgraParkArea(); + case 6: + return getNumFreeHours(); + case 7: + return getMStallsOth(); + case 8: + return getMStallsSam(); + case 9: + return getMParkCost(); + case 10: + return getDStallsOth(); + case 11: + return getDStallsSam(); + case 12: + return getDParkCost(); + case 13: + return getHStallsOth(); + case 14: + return getHStallsSam(); + case 15: + return getHParkCost(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java new file mode 100644 index 0000000..1d61c60 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java @@ -0,0 +1,101 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import com.pb.common.util.ResourceUtil; + +public final class SandagSamplePopulationGenerator +{ + + private static Logger logger = Logger.getLogger(SandagSamplePopulationGenerator.class); + + public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; + private ResourceBundle rb; + + /** + * + * @param rb + * , java.util.ResourceBundle containing environment settings + * from a properties file specified on the command line + * @param baseName + * , String containing basename (without .properites) from which + * ResourceBundle was created. + * @param globalIterationNumber + * , int iteration number for which the model is run, set by + * another process controlling a model stream with feedback. + * @param iterationSampleRate + * , float percentage [0.0, 1.0] inicating the portion of all + * households to be modeled. + * + * This object defines the implementation of the ARC tour based, + * activity based travel demand model. + */ + private SandagSamplePopulationGenerator(ResourceBundle rb) + { + this.rb = rb; + } + + private void generateSampleFiles() + { + + // new a ctramp application object + HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + + // create a new local instance of the household array manager + SandagHouseholdDataManager householdDataManager = new SandagHouseholdDataManager(); + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population files + // and + // apply its tables to objects mapping method. + String inputHouseholdFileName = "data/inputs/hhfile.csv"; + String inputPersonFileName = "data/inputs/personfile.csv"; + householdDataManager.setHouseholdSampleRate(1.0f, 0); + + SandagModelStructure modelStructure = new SandagModelStructure(); + + householdDataManager.setModelStructure(modelStructure); + householdDataManager.readPopulationFiles(inputHouseholdFileName, inputPersonFileName); + householdDataManager.mapTablesToHouseholdObjects(); + + householdDataManager.createSamplePopulationFiles( + "/jim/projects/sandag/data/inputs/hhfile.csv", + "/jim/projects/sandag/data/inputs/personfile.csv", + "/jim/projects/sandag/data/inputs/hhfile_1000.csv", + "/jim/projects/sandag/data/inputs/personfile_1000.csv", 1000); + } + + public static void main(String[] args) + { + + ResourceBundle rb = null; + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + } + + // create an instance of this class for main() to use. + SandagSamplePopulationGenerator mainObject = new SandagSamplePopulationGenerator(rb); + + // run tour based models + try + { + mainObject.generateSampleFiles(); + } catch (RuntimeException e) + { + logger.error( + "RuntimeException caught in SandagSamplePopulationGenerator.main() -- exiting.", + e); + } + + System.exit(0); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java new file mode 100644 index 0000000..e0c8a7c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java @@ -0,0 +1,255 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.StopFrequencyDMU; + +public class SandagStopFrequencyDMU + extends StopFrequencyDMU +{ + + // the SANDAG UEC worksheet numbers defined below are used to associate + // worksheet + // pages to CTRAMP purpose indices + private static final int WORK_MODEL_SHEET = 1; + private static final int UNIVERSITY_MODEL_SHEET = 2; + private static final int SCHOOL_MODEL_SHEET = 3; + private static final int ESCORT_MODEL_SHEET = 4; + private static final int SHOPPING_MODEL_SHEET = 5; + private static final int MAINT_MODEL_SHEET = 6; + private static final int EAT_OUT_MODEL_SHEET = 7; + private static final int VISITING_MODEL_SHEET = 8; + private static final int DISCR_MODEL_SHEET = 9; + private static final int WORK_BASED_MODEL_SHEET = 10; + + private HashMap tourPurposeModelSheetMap; + private HashMap tourPurposeChoiceModelIndexMap; + private int[] modelSheetValues; + + public SandagStopFrequencyDMU(ModelStructure modelStructure) + { + super(modelStructure); + setupModelIndexMappings(); + setupMethodIndexMap(); + + // set names used in SANDAG stop purpose file + STOP_PURPOSE_FILE_WORK_NAME = "Work"; + STOP_PURPOSE_FILE_UNIVERSITY_NAME = "University"; + STOP_PURPOSE_FILE_SCHOOL_NAME = "School"; + STOP_PURPOSE_FILE_ESCORT_NAME = "Escort"; + STOP_PURPOSE_FILE_SHOPPING_NAME = "Shop"; + STOP_PURPOSE_FILE_MAINT_NAME = "Maintenance"; + STOP_PURPOSE_FILE_EAT_OUT_NAME = "Eating Out"; + STOP_PURPOSE_FILE_VISIT_NAME = "Visiting"; + STOP_PURPOSE_FILE_DISCR_NAME = "Discretionary"; + STOP_PURPOSE_FILE_WORK_BASED_NAME = "Work-Based"; + } + + private void setupModelIndexMappings() + { + + // setup the mapping from tour primary purpose indices to the worksheet + // page + // indices + tourPurposeModelSheetMap = new HashMap(); + tourPurposeModelSheetMap.put(ModelStructure.WORK_PRIMARY_PURPOSE_INDEX, WORK_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX, + UNIVERSITY_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX, + SCHOOL_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX, + ESCORT_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX, + SHOPPING_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX, + MAINT_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_INDEX, + EAT_OUT_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_INDEX, + VISITING_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_INDEX, + DISCR_MODEL_SHEET); + tourPurposeModelSheetMap.put(ModelStructure.WORK_BASED_PRIMARY_PURPOSE_INDEX, + WORK_BASED_MODEL_SHEET); + + // setup a mapping between primary tour purpose indices and + // ChoiceModelApplication array indices + // so that only as many ChoiceModelApplication objects are created in + // the + // Stop Frequency model implementation + // as there are worksheet model pages. + tourPurposeChoiceModelIndexMap = new HashMap(); + + int modelIndex = 0; + HashMap modelSheetIndexMap = new HashMap(); + for (int modelPurposeKey : tourPurposeModelSheetMap.keySet()) + { + + // get the sheet number associated with the tour purpose + int modelSheetKey = tourPurposeModelSheetMap.get(modelPurposeKey); + + // if the sheet number already exists in the sheet index to choice + // model + // index mapping, get that index + // and use it for the purpose to model index mapping + if (modelSheetIndexMap.containsKey(modelSheetKey)) + { + int index = modelSheetIndexMap.get(WORK_MODEL_SHEET); + tourPurposeChoiceModelIndexMap.put(modelPurposeKey, index); + } else + { + // otherwise add this sheet number to the model index mapping + // and use + // it + // for the purpose to model index mapping. + modelSheetIndexMap.put(modelSheetKey, modelIndex); + tourPurposeChoiceModelIndexMap.put(modelPurposeKey, modelIndex); + modelIndex++; + } + } + + modelSheetValues = new int[modelIndex]; + int i = 0; + for (int sheet : modelSheetIndexMap.keySet()) + modelSheetValues[i++] = sheet; + + } + + /** + * @return the array of unique worksheet model sheet values for whic a + * ChoiceModelApplication object will be created. The size of this + * array determines the number of ChoiceModelApplication objects. + */ + public int[] getModelSheetValuesArray() + { + return modelSheetValues; + } + + /** + * @return the HashMap that relates primary tour purpose + * indices to ChoiceModelApplication array indices. + */ + public HashMap getTourPurposeChoiceModelIndexMap() + { + return tourPurposeChoiceModelIndexMap; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getNumFtWorkers", 0); + methodIndexMap.put("getNumPtWorkers", 1); + methodIndexMap.put("getIncomeInDollars", 2); + methodIndexMap.put("getPersonType", 3); + methodIndexMap.put("getHhSize", 4); + methodIndexMap.put("getNumHhDrivingStudents", 5); + methodIndexMap.put("getNumHhNonDrivingStudents", 6); + methodIndexMap.put("getNumHhPreschool", 7); + methodIndexMap.put("getWorkTours", 8); + methodIndexMap.put("getTotalTours", 9); + methodIndexMap.put("getTotalHouseholdTours", 10); + methodIndexMap.put("getWorkLocationDistance", 11); + methodIndexMap.put("getSchoolLocationDistance", 12); + methodIndexMap.put("getAge", 13); + methodIndexMap.put("getSchoolTours", 14); + methodIndexMap.put("getEscortTours", 15); + methodIndexMap.put("getShoppingTours", 16); + methodIndexMap.put("getMaintenanceTours", 17); + methodIndexMap.put("getEatTours", 18); + methodIndexMap.put("getVisitTours", 19); + methodIndexMap.put("getDiscretionaryTours", 20); + methodIndexMap.put("getShoppingAccessibility", 21); + methodIndexMap.put("getMaintenanceAccessibility", 22); + methodIndexMap.put("getDiscretionaryAccessibility", 23); + methodIndexMap.put("getIsJoint", 24); + methodIndexMap.put("getTourDurationHours", 25); + methodIndexMap.put("getTourModeIsAuto", 26); + methodIndexMap.put("getTourModeIsTransit", 27); + methodIndexMap.put("getTourModeIsNonMotorized", 28); + methodIndexMap.put("getTourModeIsSchoolBus", 29); + methodIndexMap.put("getTourDepartPeriod", 30); + methodIndexMap.put("getTourArrivePeriod", 31); + methodIndexMap.put("getTelecommuteFrequency", 32); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getNumFtWorkers(); + case 1: + return getNumPtWorkers(); + case 2: + return getIncomeInDollars(); + case 3: + return getPersonType(); + case 4: + return getHhSize(); + case 5: + return getNumHhDrivingStudents(); + case 6: + return getNumHhNonDrivingStudents(); + case 7: + return getNumHhPreschool(); + case 8: + return getWorkTours(); + case 9: + return getTotalTours(); + case 10: + return getTotalHouseholdTours(); + case 11: + return getWorkLocationDistance(); + case 12: + return getSchoolLocationDistance(); + case 13: + return getAge(); + case 14: + return getSchoolTours(); + case 15: + return getEscortTours(); + case 16: + return getShoppingTours(); + case 17: + return getMaintenanceTours(); + case 18: + return getEatTours(); + case 19: + return getVisitTours(); + case 20: + return getDiscretionaryTours(); + case 21: + return getShoppingAccessibility(); + case 22: + return getMaintenanceAccessibility(); + case 23: + return getDiscretionaryAccessibility(); + case 24: + return getTourIsJoint(); + case 25: + return getTourDurationInHours(); + case 26: + return getTourModeIsAuto(); + case 27: + return getTourModeIsTransit(); + case 28: + return getTourModeIsNonMotorized(); + case 29: + return getTourModeIsSchoolBus(); + case 30: + return getTourDepartPeriod(); + case 31: + return getTourArrivePeriod(); + case 32: + return getTelecommuteFrequency(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java new file mode 100644 index 0000000..40f1885 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java @@ -0,0 +1,132 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import java.util.Map; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.BikeLogsum; +import org.sandag.abm.ctramp.BikeLogsumSegment; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Stop; +import org.sandag.abm.ctramp.StopLocationDMU; +import org.sandag.abm.ctramp.Tour; + +public class SandagStopLocationDMU + extends StopLocationDMU +{ + public SandagStopLocationDMU(ModelStructure modelStructure, Map rbMap) + { + super(modelStructure); + setupMethodIndexMap(); + } + + public void setStopObject(Stop myStop) + { + super.setStopObject(myStop); + } + + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getSlcSoaCorrectionsAlt", 0); + methodIndexMap.put("getOrigToMgraDistanceAlt", 1); + methodIndexMap.put("getMgraToDestDistanceAlt", 2); + methodIndexMap.put("getOdDistance", 3); + methodIndexMap.put("getTourModeIsWalk", 4); + methodIndexMap.put("getTourModeIsBike", 5); + methodIndexMap.put("getTourModeIsWalkTransit", 6); + methodIndexMap.put("getWalkTransitAvailableAlt", 7); + methodIndexMap.put("getLnSlcSizeAlt", 8); + methodIndexMap.put("getStopPurpose", 9); + methodIndexMap.put("getTourPurpose", 10); + methodIndexMap.put("getTourMode", 11); + methodIndexMap.put("getStopNumber", 12); + methodIndexMap.put("getStopsOnHalfTour", 13); + methodIndexMap.put("getInboundStop", 14); + methodIndexMap.put("getTourIsJoint", 15); + methodIndexMap.put("getFemale", 16); + methodIndexMap.put("getAge", 17); + methodIndexMap.put("getTourOrigToMgraDistanceAlt", 18); + methodIndexMap.put("getMgraToTourDestDistanceAlt", 19); + methodIndexMap.put("getMcLogsumAlt", 20); + methodIndexMap.put("getSampleMgraAlt", 21); + methodIndexMap.put("getLnSlcSizeSampleAlt", 22); + methodIndexMap.put("getIncome", 23); + methodIndexMap.put("getOrigToMgraBikeLogsumAlt", 24); + methodIndexMap.put("getMgraToDestBikeLogsumAlt", 25); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getSlcSoaCorrectionsAlt(arrayIndex); + case 1: + return getOrigToMgraDistanceAlt(arrayIndex); + case 2: + return getMgraToDestDistanceAlt(arrayIndex); + case 3: + return getOdDistance(); + case 4: + return getTourModeIsWalk(); + case 5: + return getTourModeIsBike(); + case 6: + return getTourModeIsWalkTransit(); + case 7: + return getWalkTransitAvailableAlt(arrayIndex); + case 8: + return getLnSlcSizeAlt(arrayIndex); + case 9: + return getStopPurpose(); + case 10: + return getTourPurpose(); + case 11: + return getTourMode(); + case 12: + return getStopNumber(); + case 13: + return getStopsOnHalfTour(); + case 14: + return getInboundStop(); + case 15: + return getTourIsJoint(); + case 16: + return getFemale(); + case 17: + return getAge(); + case 18: + return getTourOrigToMgraDistanceAlt(arrayIndex); + case 19: + return getMgraToTourDestDistanceAlt(arrayIndex); + case 20: + return getMcLogsumAlt(arrayIndex); + case 21: + return getSampleMgraAlt(arrayIndex); + case 22: + return getLnSlcSizeSampleAlt(arrayIndex); + case 23: + return getIncomeInDollars(); + case 24: + return getOrigToMgraBikeLogsumAlt(arrayIndex); + case 25: + return getMgraToDestBikeLogsumAlt(arrayIndex); + + default: + Logger logger = Logger.getLogger(StopLocationDMU.class); + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java new file mode 100644 index 0000000..dfa63ff --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java @@ -0,0 +1,820 @@ +package org.sandag.abm.application; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import java.util.ResourceBundle; +import java.util.StringTokenizer; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.summit.ConcreteSummitRecord; +import com.pb.common.summit.SummitHeader; +import com.pb.common.summit.SummitRecordTable; +import com.pb.common.util.ResourceUtil; + +public class SandagSummitFile +{ + + private static Logger logger = Logger.getLogger(SandagSummitFile.class); + + private HashMap personsOver18; + private HashMap autosOwned; + + // Summit record table (one per file) + private SummitRecordTable summitRecordTable; + + private TableDataSet tourData; + + // Some parameters + private int modes; + private int upperEA; // Upper + // limit + // on + // time + // of + // day + // for + // the + // Early + // morning + // time + // period + private int upperAM; // Upper + // limit + // on + // time + // of + // day + // for + // the + // AM + // peak + // time + // period + private int upperMD; // Upper + // limit + // on + // time + // of + // day + // for + // the + // Midday + // time + // period + private int upperPM; // Upper + // limit + // on + // time + // of + // day + // for + // the + // PM + // time + // peak + // period + private int[] walkTransitModes; + private int[] driveTransitModes; + + // an array of file numbers, one per purpose + private int[] fileNumber; + private int numberOfFiles; + + private static final String[] PURPOSE_NAME = {"Work", "University", "School", "Escort", + "Shop", "Maintenance", "EatingOut", "Visiting", "Discretionary", "WorkBased"}; + private float[] ivtCoeff; + private String[] fileName; + + ResourceBundle rb; + MgraDataManager mdm; + TazDataManager tdm; + + public SandagSummitFile(String resourceFile) + { + + rb = ResourceUtil.getPropertyBundle(new File(resourceFile)); + mdm = MgraDataManager.getInstance(ResourceUtil.changeResourceBundleIntoHashMap(rb)); + tdm = TazDataManager.getInstance(ResourceUtil.changeResourceBundleIntoHashMap(rb)); + + // Time period limits + upperEA = Integer.valueOf(rb.getString("summit.upperEA")); + upperAM = Integer.valueOf(rb.getString("summit.upperAM")); + upperMD = Integer.valueOf(rb.getString("summit.upperMD")); + upperPM = Integer.valueOf(rb.getString("summit.upperPM")); + + // Find what file to store each purpose in + numberOfFiles = 0; + fileNumber = new int[PURPOSE_NAME.length]; + for (int i = 0; i < PURPOSE_NAME.length; ++i) + { + String fileString = "summit.purpose." + PURPOSE_NAME[i]; + fileNumber[i] = Integer.valueOf(rb.getString(fileString)) - 1; + numberOfFiles = Math.max(fileNumber[i] + 1, numberOfFiles); + } + + // Get the name of each file + fileName = new String[numberOfFiles]; + for (int i = 0; i < numberOfFiles; ++i) + { + String nameString = "summit.filename." + (i + 1); + fileName[i] = rb.getString(nameString); + } + + // Get the ivt coefficients for each file + ivtCoeff = new float[numberOfFiles]; + for (int i = 0; i < numberOfFiles; ++i) + { + String ivtString = "summit.ivt.file." + (i + 1); + ivtCoeff[i] = Float.valueOf(rb.getString(ivtString)); + } + + // set the arrays + modes = Integer.valueOf(rb.getString("summit.modes")); + walkTransitModes = new int[modes]; + driveTransitModes = new int[modes]; + + String modeArray = rb.getString("summit.mode.array").replace(" ", ""); + StringTokenizer inToken = new StringTokenizer(modeArray, ","); + int mode = 0; + while (inToken.hasMoreElements()) + { + int modeValue = Integer.valueOf(inToken.nextToken()); + logger.info("Mode " + mode + " value " + modeValue); + if (modeValue == 1) walkTransitModes[mode] = 1; + else if (modeValue == 2) driveTransitModes[mode] = 1; + + ++mode; + } + + } + + /** + * Create Summit files for all purposes and both individual and joint tour + * files. + * + */ + public void createSummitFiles() + { + + // Read the household file + String directory = rb.getString("Project.Directory"); + String hhFile = rb.getString("Results.HouseholdDataFile"); + readHouseholdFile(directory + hhFile); + + // Read the person file + String perFile = rb.getString("Results.PersonDataFile"); + readPersonFile(directory + perFile); + + // Open the individual tour file and start processing + String tourFile = rb.getString("Results.IndivTourDataFile"); + openTourFile(directory + tourFile); + + String outputDirectory = rb.getString("summit.output.directory"); + + for (int i = 0; i < getNumberOfFiles(); ++i) + { + + // Create the summit table + createSummitFile(i); + + // Write the summit output file + String purpose = getPurpose(i); + writeFile(outputDirectory + purpose + ".bin", i); + + } + + // Open the joint tour file and start processing + tourFile = rb.getString("Results.JointTourDataFile"); + openTourFile(directory + tourFile); + + for (int i = 0; i < getNumberOfFiles(); ++i) + { + + // Create the summit table + createSummitFile(i); + + // Write the summit output file + String purpose = getPurpose(i); + writeFile(outputDirectory + "jnt_" + purpose + ".bin", i); + + } + + } + + /** + * Get the number of SUMMIT Files as set in the properties file. + * + * @return The number of SUMMIT files. + */ + public int getNumberOfFiles() + { + return numberOfFiles; + } + + /** + * Read household records and store autos owned. + * + * @param fileName + * household file path/name. + */ + public void readHouseholdFile(String fileName) + { + + autosOwned = new HashMap(); + + logger.info("Begin reading the data in file " + fileName); + + TableDataSet hhData; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + hhData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + // iterate through the table and save number of autos + for (int i = 1; i <= hhData.getRowCount(); ++i) + { + long hhID = (long) hhData.getValueAt(i, "hh_ID"); + int autos = (int) hhData.getValueAt(i, "autos"); + autosOwned.put(hhID, autos); + } + logger.info("End reading the data in file " + fileName); + } + + /** + * Read person file and store persons >= 18. + * + * @param fileName + * Person file path/name. + */ + public void readPersonFile(String fileName) + { + personsOver18 = new HashMap(); + + logger.info("Begin reading the data in file " + fileName); + + TableDataSet personData; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + personData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + // iterate through the table and save number of persons>=18 + int personCount = 0; + long hhID_last = -99; + for (int i = 1; i <= personData.getRowCount(); ++i) + { + long hhID = (long) personData.getValueAt(i, "hh_ID"); + int age = (int) personData.getValueAt(i, "age"); + + // this record is a new household + if (hhID != hhID_last && i > 1) + { + personsOver18.put(hhID_last, personCount); + personCount = 0; + } + + if (age >= 18) ++personCount; + + hhID_last = hhID; + } + // save the last household + personsOver18.put(hhID_last, personCount); + + logger.info("End reading the data in file " + fileName); + + } + + /** + * Open a tour file for subsequent reading. + */ + public void openTourFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + tourData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + } + + /** + * This is the main workhorse method in this class. It creates a + * SummitRecordTable, and then iterates over records in the tour file. If + * the tour purpose for the record is mapped to the fileNumber argument, a + * ConcreteSummitRecord is created and the attributes are set, and the + * record is added to the SummitRecordTable. After all tour records have + * been read, the table is finalized. + * + * After this method is run, the file header information can be set and the + * SummitRecordTable can be written to a SUMMIT file. + * + * @param fileNumber + */ + public void createSummitFile(int fileNumber) + { + + String purpose = getPurpose(fileNumber); + logger.info("Begin creating SUMMIT record table for purpose " + purpose); + + // get column for start of utilities, probabilities + int start_util = tourData.getColumnPosition("util_1"); + int start_prob = tourData.getColumnPosition("prob_1"); + + boolean jointTour = tourData.containsColumn("tour_participants"); + int participantsCol = 0; + if (jointTour) + { + participantsCol = tourData.getColumnPosition("tour_participants"); + } + + // arrays of utilities, probabilities + float[] util = new float[modes]; + float[] prob = new float[modes]; + + // Instantiate a new SummitRecordTable + summitRecordTable = new SummitRecordTable(); + + // iterate through the tour data and save summit record tables + for (int i = 1; i <= tourData.getRowCount(); ++i) + { + + //if (i <= 5 || i % 1000 == 0) logger.info("Reading record " + i); + + String tourPurpose = tourData.getStringValueAt(i, "tour_purpose"); + int tableNumber = calculateSummitTable(tourPurpose); + + if (tableNumber != fileNumber) continue; + + long hhID = (long) tourData.getValueAt(i, "hh_id"); + + int originMGRA = (int) tourData.getValueAt(i, "orig_maz"); + int destinationMGRA = (int) tourData.getValueAt(i, "dest_maz"); + + int departPeriod = (int) tourData.getValueAt(i, "start_period"); + int arrivePeriod = (int) tourData.getValueAt(i, "end_period"); + + // get utilities, probabilities + for (int j = 0; j < modes; ++j) + { + util[j] = tourData.getValueAt(i, start_util + j); + prob[j] = tourData.getValueAt(i, start_prob + j); + } + + // calculate some necessary information for the SUMMIT record + int autoSufficiency = calculateAutoSufficiency(hhID); + short originTAZ = (short) mdm.getTaz(originMGRA); + short destinationTAZ = (short) mdm.getTaz(destinationMGRA); + int periodMarket = calculatePeriodMarket(departPeriod, arrivePeriod); + short marketSegment = (short) calculateMarketSegment(autoSufficiency, periodMarket); + float expUtility = calculateNonTransitExpUtility(util); // not used + float wtAvailShare = calculateWalkTransitAvailableShare(prob); + float dtAvailShare = calculateDriveTransitOnlyShare(prob, wtAvailShare); + float wtProb = calculateTransitShareOfWalkTransit(prob, wtAvailShare); + float dtProb = calculateTransitShareOfDriveTransitOnly(prob, dtAvailShare); + float aggExpUtility = calculateAggregateExpUtility(util); + + float participants = 1.0f; + if (jointTour) + { + String participantString = tourData.getStringValueAt(i, participantsCol); + for (int j = 0; j < participantString.length(); ++j) + if (participantString.charAt(j) == ' ') participants += 1; + } + // Create a new summit record, and set all attributes + ConcreteSummitRecord summitRecord = new ConcreteSummitRecord(); + + summitRecord.setPtaz(originTAZ); + summitRecord.setAtaz(destinationTAZ); + summitRecord.setMarket(marketSegment); + summitRecord.setTrips(participants); + summitRecord.setMotorizedTrips(participants); + summitRecord.setExpAuto(aggExpUtility); + summitRecord.setWalkTransitAvailableShare(wtAvailShare); + summitRecord.setDriveTransitOnlyShare(dtAvailShare); + summitRecord.setTransitShareOfWalkTransit(wtProb); + summitRecord.setTransitShareOfDriveTransitOnly(dtProb); + + // Insert the record into the record table + summitRecordTable.insertRecord(summitRecord); + + } + + logger.info("End creating SUMMIT record table for purpose " + purpose); + + logger.info("Begin finalizing table"); + summitRecordTable.finalizeTable(); + logger.info("End finalizing table"); + + } + + /** + * Calculate and return the market for the departure and arrival time + * periods. + * + * Market 0 = Departure & arrival in peak Market 1 = Departure & arrival in + * mixed periods (peak and off-peak) Market 2 = Departure & arrival in + * off-peak + * + * @param departPeriod + * 1-39 representing time in 30 min. increments, starting at 5 AM + * @param arrivePeriod + * 1-39 representing time in 30 min. increments, starting at 5 AM + * @return Market, as defined above. + */ + public int calculatePeriodMarket(int departPeriod, int arrivePeriod) + { + + int departPeak = 0; + int arrivePeak = 0; + + // check if departure is in peak period + if (departPeriod > upperEA && departPeriod <= upperAM) departPeak = 1; + else if (departPeriod > upperMD && departPeriod <= upperPM) departPeak = 1; + + // check if arrival is in peak period + if (arrivePeriod > upperEA && arrivePeriod <= upperAM) arrivePeak = 1; + else if (arrivePeriod > upperMD && arrivePeriod <= upperPM) arrivePeak = 1; + + /* + * Arrival & departure = peak, period = 0 Mixed peak & off-peak, period + * = 1 Arrival & departure = off-peal, period = 2 + */ + if (departPeak == 1 && arrivePeak == 1) return 0; + else if (departPeak == 0 && arrivePeak == 0) return 2; + + return 1; + } + + /** + * Calculate and return market segment based on auto sufficiency and time + * period combination. + * + * Market AutoSuff Period 1 0 0 Peak 2 0 1 Mixed 2 0 2 Off-Peak 3 1 0 Peak 4 + * 1 1 Mixed 4 1 2 Off-Peak 5 2 0 Peak 6 2 1 Mixed 6 2 2 Off-Peak + * + * + * @param autoSufficiency + * 0 = 0 autos, 1=autos < adults, 2 = autos >= adults + * @param periodMarket + * 0 = Peak, 1 = Mixed, 2 = Off-Peak + * @return Market for Summit record, as per above table. + */ + public int calculateMarketSegment(int autoSufficiency, int periodMarket) + { + + int market = 0; + + switch (autoSufficiency) + { + case 0: + market = (periodMarket == 0) ? 1 : 2; + break; + case 1: + market = (periodMarket == 0) ? 3 : 4; + break; + case 2: + market = (periodMarket == 0) ? 5 : 6; + break; + default: + logger.fatal("Error: Could not calculate market segment auto sufficiency " + + autoSufficiency); + throw new RuntimeException(); + } + + return market; + } + + /** + * Determine what table to use for tour purpose, based on fileNumber array + * set from properties file. + * + * @param tourPurpose + * @return Table for tour purpose + */ + public int calculateSummitTable(String tourPurpose) + { + + if (tourPurpose.contentEquals("Work")) return fileNumber[0]; + else if (tourPurpose.contentEquals("University")) return fileNumber[1]; + else if (tourPurpose.contentEquals("School")) return fileNumber[2]; + else if (tourPurpose.contentEquals("Escort")) return fileNumber[3]; + else if (tourPurpose.contentEquals("Shop")) return fileNumber[4]; + else if (tourPurpose.contentEquals("Maintenance")) return fileNumber[5]; + else if (tourPurpose.contentEquals("Eating Out")) return fileNumber[6]; + else if (tourPurpose.contentEquals("Visiting")) return fileNumber[7]; + else if (tourPurpose.contentEquals("Discretionary")) return fileNumber[8]; + else if (tourPurpose.contentEquals("Work-Based")) return fileNumber[9]; + else + { + logger.error("Error: Tour purpose " + tourPurpose + " not recognized"); + } + + return 99; + } + + /** + * Look up the purpose string based on the file number, for use in SUMMIT + * file header. + * + * @param fileNumber + * @return A string for the purpose (see above). + */ + public String getPurpose(int fileNumber) + { + + return fileName[fileNumber]; + } + + /** + * Calculate market segment (auto sufficiency) + * + * 0 = 0 autos owned 1 = autos > 0 & autos < adults (persons 18+) 2 = autos + * > adults + * + * @param hhID + * Household ID + * @return marketSegment + */ + public int calculateAutoSufficiency(long hhID) + { + + int drivers = personsOver18.get(hhID); + int autos = autosOwned.get(hhID); + + if (autos > 0) if (autos < drivers) return 1; + else return 2; + + return 0; + } + + /** + * Calculate the total non-transit exponentiated utility. The method uses + * the walkTransitModes array and the driveTransitModes array to determine + * which modes are non-transit, and the sum of their exponentiated utilities + * is calculated and returned. + * + * @param util + * An array of utilities, by mode. -999 indicates mode not + * available. + * @return Sum of exponentiated utilities of non-transit modes. + */ + public float calculateNonTransitExpUtility(float[] util) + { + + float expUtility = 0.0f; + + for (int i = 0; i < modes; ++i) + if (walkTransitModes[i] != 1 && driveTransitModes[i] != 1) + expUtility += (float) Math.exp(util[i]); + return expUtility; + } + + /** + * Calculate the share of walk-transit available: 1 if any walk-transit mode + * is available, as indicated by a non-zero probability, else 0. The method + * iterates through the probability array and returns a 1 if the probability + * is non-zero for any walk-transit mode, as indicated by the + * walkTransitModes array. + * + * @param prob + * An array of probabilities, dimensioned by modes. + * @return 1 if walk-transit is available for the record, else 0. + */ + public float calculateWalkTransitAvailableShare(float[] prob) + { + + // iterate through the probability array + for (int i = 0; i < modes; ++i) + if (walkTransitModes[i] == 1 && prob[i] > 0) return 1.0f; + + // no walk-transit modes with non-zero probability + return 0.0f; + } + + /** + * Calculate the share of drive-transit only available: 1 if any + * drive-transit mode is available, as indicated by a non-zero probability, + * and all walk-transit modes are not available. The method iterates through + * the probability array and returns a 1 if the probability is non-zero for + * any drive-transit mode, as indicated by the driveTransitModes array. + * + * @param prob + * An array of probabilities, dimensioned by modes. + * @param walkTransitAvailableShare + * 1 if walk-transit available, else 0. + * @return 1 if drive-transit only is available for the record, else 0. + */ + + public float calculateDriveTransitOnlyShare(float[] prob, float walkTransitAvailableShare) + { + + // if walk-transit is available, then drive-transit only share is 0. + if (walkTransitAvailableShare > 0) return 0.0f; + + // iterate through the probability array + for (int i = 0; i < modes; ++i) + if (driveTransitModes[i] == 1 && prob[i] > 0) return 1.0f; + + // no drive-transit modes with non-zero probability + return 0.0f; + + } + + /** + * Calculate the total transit probability for records with walk-transit + * available. The method returns 0 if walk-transit is not available. If + * walk-transit is available, the method iterates through the probability + * array, adding all transit mode probabilities. The sum is returned. + * + * @param prob + * An array of probabilities, one per mode. + * @param walkTransitAvailableShare + * 1 if walk-transit is available, else 0. + * @return The total transit probability if walk-transit is available, else + * 0. + */ + public float calculateTransitShareOfWalkTransit(float[] prob, float walkTransitAvailableShare) + { + + float transitShare = 0.0f; + + // if walk-transit is unavailable, then walk-transit share is 0. + if (walkTransitAvailableShare == 0) return transitShare; + + // iterate through the probability array + for (int i = 0; i < modes; ++i) + if (walkTransitModes[i] == 1 || driveTransitModes[i] == 1) transitShare += prob[i]; + + return transitShare; + + } + + /** + * Calculate the total transit probability for records where only + * drive-transit available. The method returns 0 if only drive-transit is + * not available. If only drive-transit is available, the method iterates + * through the probability array, adding all drive-transit mode + * probabilities. The sum is returned. + * + * @param prob + * An array of probabilities, one per mode. + * @param driveTransitOnlyAvailableShare + * 1 if only drive-transit is available, else 0. + * @return The total transit probability if only drive-transit is available, + * else 0. + */ + public float calculateTransitShareOfDriveTransitOnly(float[] prob, + float driveTransitOnlyAvailableShare) + { + + float transitShare = 0.0f; + + // if drive-transit is unavailable, then walk-transit share is 0. + if (driveTransitOnlyAvailableShare == 0) return transitShare; + + // iterate through the probability array + for (int i = 0; i < modes; ++i) + if (driveTransitModes[i] == 1) transitShare += prob[i]; + + return transitShare; + + } + + /** + * Calculate the logsum by taking ln[Sum (exp(utility)). + * + * @param util + * Array of utilities + * @return Logsum + */ + public float calculateAggregateExpUtility(float[] util) + { + + float aggExpUtility = 0.0f; + + for (int i = 0; i < util.length; ++i) + aggExpUtility += Math.exp(util[i]); + + return aggExpUtility; + } + + /** + * Get the in-tNCVehicle time coefficient for the file, based on the values + * read in the properties file. + * + * @param fileNumber + * @return The in-tNCVehicle time coefficient for the file. + */ + public float getIVTCoefficient(int fileNumber) + { + return ivtCoeff[fileNumber]; + } + + /** + * Create a Summit file header. + * + * @param fileNumber + * @return Header record for Summit file. + */ + public SummitHeader createSummitHeader(int fileNumber) + { + + SummitHeader header = new SummitHeader(); + + int zones = tdm.getMaxTaz(); + String purpose = getPurpose(fileNumber); + header.setZones(zones); + header.setMarketSegments(6); + + float ivt = getIVTCoefficient(fileNumber); + header.setTransitInVehicleTime(ivt); + header.setAutoInVehicleTime(ivt); + + header.setPurpose(purpose); + header.setTimeOfDay("ALL"); + header.setTitle("SANDAG CT-RAMP MODEL SUMMIT FILE"); + return header; + } + + public void writeFile(String fileName, int fileNumber) + { + + SummitHeader header = createSummitHeader(fileNumber); + summitRecordTable.writeTable(fileName, header); + + } + + /** + * @param args + */ + public static void main(String[] args) + { + + // Create a new SandagSummitFile + String propertiesFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\sandag_abm.properties"; + SandagSummitFile summitFile = new SandagSummitFile(propertiesFile); + + // Read the household file + String hhFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\householdData_1.csv"; + summitFile.readHouseholdFile(hhFile); + + // Read the person file + String perFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\personData_1.csv"; + summitFile.readPersonFile(perFile); + + // Open the individual tour file and start processing + String tourFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\indivTourData_1.csv"; + summitFile.openTourFile(tourFile); + + for (int i = 0; i < summitFile.getNumberOfFiles(); ++i) + { + + // Create the summit table + summitFile.createSummitFile(i); + + // Write the summit output file + String outputFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\"; + String purpose = summitFile.getPurpose(i); + summitFile.writeFile(outputFile + purpose + ".bin", i); + + } + + // Open the joint tour file and start processing + tourFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\jointTourData_1.csv"; + summitFile.openTourFile(tourFile); + + for (int i = 0; i < summitFile.getNumberOfFiles(); ++i) + { + + // Create the summit table + summitFile.createSummitFile(i); + + // Write the summit output file + String outputFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\jnt_"; + String purpose = summitFile.getPurpose(i); + summitFile.writeFile(outputFile + purpose + ".bin", i); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java new file mode 100644 index 0000000..e0f4051 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java @@ -0,0 +1,67 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.ParkingChoiceDMU; +import org.sandag.abm.ctramp.TelecommuteDMU; + +public class SandagTelecommuteDMU + extends TelecommuteDMU +{ + + public SandagTelecommuteDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getIncomeInDollars", 1); + methodIndexMap.put("getNumberOfAdults", 2); + methodIndexMap.put("getHasKids_0_5", 3); + methodIndexMap.put("getHasKids_6_12", 4); + methodIndexMap.put("getFemale", 5); + methodIndexMap.put("getPersonType", 6); + methodIndexMap.put("getNumberOfAutos", 7); + methodIndexMap.put("getOccupation", 8); + methodIndexMap.put("getPaysToPark", 9); + methodIndexMap.put("getWorkDistance", 10); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + + case 1: + return getIncomeInDollars(); + case 2: + return getNumberOfAdults(); + case 3: + return getHasKids_0_5(); + case 4: + return getHasKids_6_12(); + case 5: + return getFemale(); + case 6: + return getPersonType(); + case 7: + return getNumberOfAutos(); + case 8: + return getOccupation(); + case 9: + return getPaysToPark(); + case 10: + return getWorkDistance(); + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java new file mode 100644 index 0000000..92ba6ff --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java @@ -0,0 +1,216 @@ +package org.sandag.abm.application; + +import com.pb.common.matrix.Matrix; + +public class SandagTestSOA +{ + + private Matrix distanceMatrix; + private Matrix expDistanceMatrix; + private float[] size; + private float[] lnSize; + + public SandagTestSOA() + { + + } + + /** + * Create the distance matrix + * + * @param zones + * The number of zones + */ + public void createDistanceMatrix(int zones) + { + + distanceMatrix = new Matrix(zones, zones); + + for (int i = 1; i <= zones; ++i) + for (int j = 1; j <= zones; ++j) + { + float distance = (float) Math.random() * 100; + distanceMatrix.setValueAt(i, j, distance); + } + } + + /** + * Create the exponentiated distance matrix + * + * @param zones + * The number of zones + */ + public void createExpDistanceMatrix(float distParam, int zones) + { + + long createTime = -System.currentTimeMillis(); + expDistanceMatrix = new Matrix(zones, zones); + + for (int i = 1; i <= zones; ++i) + for (int j = 1; j <= zones; ++j) + { + float expDist = (float) distParam * distanceMatrix.getValueAt(i, j); + expDist = (float) Math.exp(expDist); + expDistanceMatrix.setValueAt(i, j, expDist); + } + createTime += System.currentTimeMillis(); + System.out.println("Time to exponentiate distance matrix " + createTime); + } + + /** + * Create the size terms + */ + public void createSizeTerms(int zones) + { + + size = new float[zones + 1]; + lnSize = new float[zones + 1]; + + for (int i = 1; i <= zones; ++i) + { + size[i] = (float) Math.random() * 1000; + lnSize[i] = (float) Math.log(size[i]); + } + } + + public void calculateProbabilitiesOldWay(int observations, int zones, float distParam) + { + + long oldWayTime = -System.currentTimeMillis(); + + float[] prob = new float[zones + 1]; + float[] expUtil = new float[zones + 1]; + float sumExp = 0; + + for (int obs = 0; obs < observations; ++obs) + { + + int origin = (int) (Math.random() * (zones - 1)) + 1; + int destination = (int) (Math.random() * (zones - 1)) + 1; + + float odDist = distanceMatrix.getValueAt(origin, destination); + + // calculate utilities + for (int stop = 1; stop <= zones; ++stop) + { + float osDist = distanceMatrix.getValueAt(origin, stop); + float sdDist = distanceMatrix.getValueAt(stop, destination); + + float util = distParam * (osDist + sdDist - odDist) + lnSize[stop]; + expUtil[stop] = (float) Math.exp(util); + sumExp += expUtil[stop]; + } + + // calculate probabilities + for (int stop = 1; stop <= zones; ++stop) + prob[stop] = expUtil[stop] / sumExp; + } + oldWayTime += System.currentTimeMillis(); + System.out.println("Time to calculate probabilities old way tazs " + oldWayTime); + + } + + public void calculateProbabilitiesOldWayMGRAs(int observations, int zones, float distParam, + int mgras) + { + + long oldWayMGRATime = -System.currentTimeMillis(); + + float[] prob = new float[mgras + 1]; + float[] expUtil = new float[mgras + 1]; + float sumExp = 0; + + for (int obs = 0; obs < observations; ++obs) + { + + int origin = (int) (Math.random() * (zones - 1)) + 1; + int destination = (int) (Math.random() * (zones - 1)) + 1; + + float odDist = distanceMatrix.getValueAt(origin, destination); + + int stopIndex = 1; + // calculate utilities + for (int stop = 1; stop <= mgras; ++stop) + { + + float osDist = distanceMatrix.getValueAt(origin, stopIndex); + float sdDist = distanceMatrix.getValueAt(stopIndex, destination); + + float util = distParam * (osDist + sdDist - odDist) + lnSize[stopIndex]; + expUtil[stopIndex] = (float) Math.exp(util); + sumExp += expUtil[stopIndex]; + + ++stopIndex; + if (stopIndex > zones) stopIndex = 1; + } + + // calculate probabilities + for (int stop = 1; stop <= mgras; ++stop) + prob[stop] = expUtil[stop] / sumExp; + + } + oldWayMGRATime += System.currentTimeMillis(); + System.out.println("Time to calculate probabilities old way mgras " + oldWayMGRATime); + + } + + public void calculateProbabilitiesNewWay(int observations, int zones) + { + + long newWayTime = -System.currentTimeMillis(); + + float[] prob = new float[zones + 1]; + float[] expUtil = new float[zones + 1]; + float sumExp = 0; + + for (int obs = 0; obs < observations; ++obs) + { + + int origin = (int) (Math.random() * (zones - 1)) + 1; + int destination = (int) (Math.random() * (zones - 1)) + 1; + + float odExpDist = expDistanceMatrix.getValueAt(origin, destination); + + // calculate utilities + for (int stop = 1; stop <= zones; ++stop) + { + float osExpDist = expDistanceMatrix.getValueAt(origin, stop); + float sdExpDist = expDistanceMatrix.getValueAt(stop, destination); + + expUtil[stop] = osExpDist * sdExpDist / odExpDist * size[stop]; + sumExp += expUtil[stop]; + } + + // calculate probabilities + for (int stop = 1; stop <= zones; ++stop) + prob[stop] = expUtil[stop] / sumExp; + } + + newWayTime += System.currentTimeMillis(); + System.out.println("Time to calculate probabilities new way tazs " + newWayTime); + } + + /** + * @param args + */ + public static void main(String[] args) + { + + int zones = 4600; + int observations = 500000; + float distParam = (float) -0.05; + int mgras = 32000; + + SandagTestSOA soa = new SandagTestSOA(); + + soa.createDistanceMatrix(zones); + soa.createExpDistanceMatrix(distParam, zones); + soa.createSizeTerms(zones); + + soa.calculateProbabilitiesOldWayMGRAs(observations, zones, distParam, mgras); + soa.calculateProbabilitiesOldWay(observations, zones, distParam); + soa.calculateProbabilitiesNewWay(observations, zones); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java new file mode 100644 index 0000000..86aaf7e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java @@ -0,0 +1,358 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; + +import org.sandag.abm.ctramp.HouseholdDataManager; +import org.sandag.abm.ctramp.HouseholdDataManagerIf; +import org.sandag.abm.ctramp.HouseholdDataManagerRmi; +import com.pb.common.util.ResourceUtil; + +public final class SandagTourBasedModel +{ + + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + + public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; + + private static final int DEFAULT_ITERATION_NUMBER = 1; + private static final float DEFAULT_SAMPLE_RATE = 1.0f; + private static final int DEFAULT_SAMPLE_SEED = 0; + + public static final int DEBUG_CHOICE_MODEL_HHID = 740151; + + private ResourceBundle rb; + + // values for these variables are set as command line arguments, or default + // vaues + // are used if no command line arguments are specified. + private int globalIterationNumber = 0; + private float iterationSampleRate = 0f; + private int sampleSeed = 0; + private boolean calculateLandUseAccessibilities = false; + + /** + * + * @param rb + * , java.util.ResourceBundle containing environment settings + * from a properties file specified on the command line + * @param globalIterationNumber + * , int iteration number for which the model is run, set by + * another process controlling a model stream with feedback. + * @param iterationSampleRate + * , float percentage [0.0, 1.0] inicating the portion of all + * households to be modeled. + * + * This object defines the implementation of the ARC tour based, + * activity based travel demand model. + */ + private SandagTourBasedModel(ResourceBundle aRb, HashMap aPropertyMap, + int aGlobalIterationNumber, float aIterationSampleRate, boolean aCalculateLandUseAccessibilities) + { + rb = aRb; + globalIterationNumber = aGlobalIterationNumber; + iterationSampleRate = aIterationSampleRate; + sampleSeed = Integer.parseInt(rb.getString("Model.Random.Seed")); + calculateLandUseAccessibilities = aCalculateLandUseAccessibilities; + } + + private void runTourBasedModel(HashMap propertyMap) + { + + // new a ctramp application object + SandagCtrampApplication ctrampApplication = new SandagCtrampApplication(rb, propertyMap, + calculateLandUseAccessibilities); + + // create modelStructure object + SandagModelStructure modelStructure = new SandagModelStructure(); + + boolean localHandlers = false; + + String hhHandlerAddress = ""; + int hhServerPort = 0; + try + { + // get household server address. if none is specified a local server + // in + // the current process will be started. + hhHandlerAddress = rb.getString("RunModel.HouseholdServerAddress"); + try + { + // get household server port. + hhServerPort = Integer.parseInt(rb.getString("RunModel.HouseholdServerPort")); + localHandlers = false; + } catch (MissingResourceException e) + { + // if no household data server address entry is found, the + // object + // will be created in the local process + localHandlers = true; + } + } catch (MissingResourceException e) + { + localHandlers = true; + } + + String testString; + // if ( localHandlers ) { + // tazDataHandler = new SandagTazDataHandler(rb, projectDirectory); + // } + // else { + // tazDataHandler = new TazDataHandlerRmi( + // ArcTazDataHandler.ZONAL_DATA_SERVER_ADDRESS, + // ArcTazDataHandler.ZONAL_DATA_SERVER_PORT, + // ArcTazDataHandler.ZONAL_DATA_SERVER_NAME ); + // testString = tazDataHandler.testRemote(); + // logger.info ( "TazDataHandler test: " + testString ); + // } + + // setup the ctramp application + ctrampApplication.setupModels(modelStructure); + + // generate the synthetic population + // ARCPopulationSynthesizer populationSynthesizer = new + // ARCPopulationSynthesizer( propertiesFileBaseName ); + // ctrampApplication.runPopulationSynthesizer( populationSynthesizer ); + + HouseholdDataManagerIf householdDataManager; + + try + { + + if (localHandlers) + { + + // create a new local instance of the household array manager + householdDataManager = new SandagHouseholdDataManager2(); + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + String inputHouseholdFileName = rb + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = rb + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, + inputHouseholdFileName, inputPersonFileName); + + } else + { + + householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, + SandagHouseholdDataManager2.HH_DATA_SERVER_NAME); + testString = householdDataManager.testRemote(); + logger.info("HouseholdDataManager test: " + testString); + + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + boolean restartHhServer = false; + try + { + // possible values for the following can be none, ao, cdap, + // imtf, + // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, + // inmtod, + // stf, stl + String restartModel = rb.getString("RunModel.RestartWithHhServer"); + if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; + else if (restartModel.equalsIgnoreCase("uwsl") + || restartModel.equalsIgnoreCase("ao") + || restartModel.equalsIgnoreCase("fp") + || restartModel.equalsIgnoreCase("cdap") + || restartModel.equalsIgnoreCase("imtf") + || restartModel.equalsIgnoreCase("imtod") + || restartModel.equalsIgnoreCase("awf") + || restartModel.equalsIgnoreCase("awl") + || restartModel.equalsIgnoreCase("awtod") + || restartModel.equalsIgnoreCase("jtf") + || restartModel.equalsIgnoreCase("jtl") + || restartModel.equalsIgnoreCase("jtod") + || restartModel.equalsIgnoreCase("inmtf") + || restartModel.equalsIgnoreCase("inmtl") + || restartModel.equalsIgnoreCase("inmtod") + || restartModel.equalsIgnoreCase("stf") + || restartModel.equalsIgnoreCase("stl")) restartHhServer = false; + } catch (MissingResourceException e) + { + restartHhServer = true; + } + + if (restartHhServer) + { + + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = rb + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = rb + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, + inputHouseholdFileName, inputPersonFileName); + + } else + { + + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setDebugHhIdsFromHashmap(); + householdDataManager.setTraceHouseholdSet(); + + // set the random number sequence for household objects + // accordingly based on which model components are + // assumed to have already run and are stored in the remote + // HouseholdDataManager object. + ctrampApplication.restartModels(householdDataManager); + + } + + } + + // create a factory object to pass to various model components from + // which + // they can create DMU objects + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + // run the models + ctrampApplication.runModels(householdDataManager, dmuFactory, globalIterationNumber, + iterationSampleRate); + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + } + + public static void main(String[] args) + { + Runtime gfg = Runtime.getRuntime(); + long memory1; + // checking the total memeory + System.out.println("Total memory is: "+ gfg.totalMemory()); + // checking free memory + memory1 = gfg.freeMemory(); + System.out.println("Initial free memory at Resident model: "+ memory1); + // calling the garbage collector on demand + gfg.gc(); + memory1 = gfg.freeMemory(); + System.out.println("Free memory after garbage "+ "collection: " + memory1); + + long startTime = System.currentTimeMillis(); + int globalIterationNumber = -1; + float iterationSampleRate = -1.0f; + //int sampleSeed = -1; + boolean calculateLandUseAccessibilities = false; + + ResourceBundle rb = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + pMap = ResourceUtil.getResourceBundleAsHashMap(args[0]); + + // optional arguments + for (int i = 1; i < args.length; i++) + { + + if (args[i].equalsIgnoreCase("-iteration")) + { + globalIterationNumber = Integer.parseInt(args[i + 1]); + logger.info(String.format("-iteration %d.", globalIterationNumber)); + } + + if (args[i].equalsIgnoreCase("-sampleRate")) + { + iterationSampleRate = Float.parseFloat(args[i + 1]); + logger.info(String.format("-sampleRate %.4f.", iterationSampleRate)); + } + + /* + if (args[i].equalsIgnoreCase("-sampleSeed")) + { + sampleSeed = Integer.parseInt(args[i + 1]); + logger.info(String.format("-sampleSeed %d.", sampleSeed)); + } + */ + + if (args[i].equalsIgnoreCase("-luAcc")) + { + calculateLandUseAccessibilities = Boolean.parseBoolean(args[i + 1]); + logger.info(String.format("-luAcc %s.", calculateLandUseAccessibilities)); + } + + } + + if (globalIterationNumber < 0) + { + globalIterationNumber = DEFAULT_ITERATION_NUMBER; + logger.info(String.format("no -iteration flag, default value %d used.", + globalIterationNumber)); + } + + if (iterationSampleRate < 0) + { + iterationSampleRate = DEFAULT_SAMPLE_RATE; + logger.info(String.format("no -sampleRate flag, default value %.4f used.", + iterationSampleRate)); + } + + /* + if (sampleSeed < 0) + { + sampleSeed = DEFAULT_SAMPLE_SEED; + logger.info(String + .format("no -sampleSeed flag, default value %d used.", sampleSeed)); + } + */ + + } + + // create an instance of this class for main() to use. + SandagTourBasedModel mainObject = new SandagTourBasedModel(rb, pMap, globalIterationNumber, + iterationSampleRate, calculateLandUseAccessibilities); + + // run tour based models + try + { + + logger.info(""); + logger.info("starting tour based model."); + mainObject.runTourBasedModel(pMap); + + } catch (RuntimeException e) + { + logger.error( + "RuntimeException caught in org.sandag.abm.application.SandagTourBasedModel.main() -- exiting.", + e); + } + + logger.info(""); + logger.info(""); + logger.info("SANDAG Activity Based Model finished in " + + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); + + System.exit(0); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java new file mode 100644 index 0000000..6971a98 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java @@ -0,0 +1,345 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.Definitions; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.TourDepartureTimeAndDurationDMU; + +/** + * ArcTourDepartureTimeAndDurationDMU is a class that ... + * + * @author Kimberly Grommes + * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. + */ +public class SandagTourDepartureTimeAndDurationDMU + extends TourDepartureTimeAndDurationDMU +{ + + public SandagTourDepartureTimeAndDurationDMU(ModelStructure modelStructure) + { + super(modelStructure); + setupMethodIndexMap(); + } + + public double getDestinationEmploymentDensity() + { + return destEmpDen; + } + + public int getIncomeLessThan30k() + { + float incomeInDollars = (float) household.getIncomeInDollars(); + return (incomeInDollars < 30000) ? 1 : 0; + } + + public int getIncome30kTo60k() + { + float incomeInDollars = (float) household.getIncomeInDollars(); + return (incomeInDollars >= 30000 && incomeInDollars < 60000) ? 1 : 0; + } + + public int getIncomeHigherThan100k() + { + float incomeInDollars = (float) household.getIncomeInDollars(); + return (incomeInDollars >= 100000) ? 1 : 0; + } + + public int getAge() + { + return getPersonAge(); + } + + public int getFemale() + { + return getPersonIsFemale(); + } + + public int getFemaleWithPreschooler() + { + return ((getPersonIsFemale() == 1) && (getNumPreschoolChildrenInHh() > 1)) ? 1 : 0; + } + + public int getDrivingAgeStudent() + { + return (getStudentDrivingAge() == 1) ? 1 : 0; + } + + public int getSchoolChildWithMandatoryTour() + { + return (getStudentNonDrivingAge() == 1 && getPersonMandatoryTotal() > 0) ? 1 : 0; + } + + public int getUniversityWithMandatoryPattern() + { + return (getUniversityStudent() == 1 && person.getCdapActivity().equalsIgnoreCase( + Definitions.MANDATORY_PATTERN)) ? 1 : 0; + } + + public int getWorkerWithMandatoryPattern() + { + return ((getFullTimeWorker() == 1 || getPartTimeWorker() == 1) && person.getCdapActivity() + .equalsIgnoreCase(Definitions.MANDATORY_PATTERN)) ? 1 : 0; + } + + public int getPreschoolChildWithMandatoryTour() + { + return (getPreschool() == 1 && getPersonMandatoryTotal() > 0) ? 1 : 0; + } + + public int getNonWorkerInHH() + { + return (getNumNonWorkingAdultsInHh() > 0) ? 1 : 0; + } + + public int getJointTour() + { + return (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) ? 1 : 0; + } + + public int getIndividualTour() + { + return (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) ? 1 : 0; + } + + public int getJointTourInHH() + { + return (getHhJointTotal() > 0) ? 1 : 0; + } + + public int getSubsequentTourIsWorkTour() + { + return subsequentTourIsWork; + } + + public int getSubsequentTourIsSchoolTour() + { + return subsequentTourIsSchool; + } + + public int getNumberOfNonEscortingIndividualTours() + { + return getPersonNonMandatoryTotalNoEscort(); + } + + public int getNumberOfDiscretionaryTours() + { + return getPersonJointAndIndivDiscrToursTotal(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getDestinationEmploymentDensity", 1); + methodIndexMap.put("getIncomeLessThan30k", 2); + methodIndexMap.put("getIncome30kTo60k", 3); + methodIndexMap.put("getIncomeHigherThan100k", 4); + methodIndexMap.put("getAge", 5); + methodIndexMap.put("getFemale", 6); + methodIndexMap.put("getFemaleWithPreschooler", 7); + methodIndexMap.put("getFullTimeWorker", 8); + methodIndexMap.put("getPartTimeWorker", 9); + methodIndexMap.put("getUniversityStudent", 10); + methodIndexMap.put("getDrivingAgeStudent", 11); + methodIndexMap.put("getNonWorkerInHH", 12); + methodIndexMap.put("getJointTourInHH", 13); + methodIndexMap.put("getFirstTour", 14); + methodIndexMap.put("getSubsequentTour", 15); + methodIndexMap.put("getModeChoiceLogsumAlt", 16); + methodIndexMap.put("getSubsequentTourIsWorkTour", 17); + methodIndexMap.put("getSubsequentTourIsSchoolTour", 18); + methodIndexMap.put("getEndOfPreviousTour", 19); + methodIndexMap.put("getAllAdultsFullTimeWorkers", 20); + methodIndexMap.put("getNonWorker", 21); + methodIndexMap.put("getRetired", 22); + methodIndexMap.put("getSchoolChildWithMandatoryTour", 23); + methodIndexMap.put("getPreschoolChildWithMandatoryTour", 24); + methodIndexMap.put("getNumberOfNonEscortingIndividualTours", 25); + methodIndexMap.put("getNumberOfDiscretionaryTours", 26); + methodIndexMap.put("getIndividualTour", 27); + methodIndexMap.put("getJointTour", 28); + methodIndexMap.put("getHouseholdSize", 29); + methodIndexMap.put("getKidsOnJointTour", 30); + methodIndexMap.put("getAdditionalShoppingTours", 31); + methodIndexMap.put("getAdditionalMaintenanceTours", 32); + methodIndexMap.put("getAdditionalVisitingTours", 33); + methodIndexMap.put("getAdditionalDiscretionaryTours", 34); + methodIndexMap.put("getMaximumAvailableTimeWindow", 35); + methodIndexMap.put("getWorkerWithMandatoryPattern", 36); + methodIndexMap.put("getUnivStudentWithMandatoryPattern", 37); + methodIndexMap.put("getHhChildUnder16", 38); + methodIndexMap.put("getToursLeftToSchedule", 39); + methodIndexMap.put("getPreDrivingAgeChild", 40); + methodIndexMap.put("getJointTourPartySize", 41); + methodIndexMap.put("getSubtourPurposeIsEatOut", 42); + methodIndexMap.put("getSubtourPurposeIsBusiness", 43); + methodIndexMap.put("getSubtourPurposeIsOther", 44); + methodIndexMap.put("getMaxJointTimeWindow", 45); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + case 1: + returnValue = getDestinationEmploymentDensity(); + break; + case 2: + returnValue = getIncomeLessThan30k(); + break; + case 3: + returnValue = getIncome30kTo60k(); + break; + case 4: + returnValue = getIncomeHigherThan100k(); + break; + case 5: + returnValue = getAge(); + break; + case 6: + returnValue = getFemale(); + break; + case 7: + returnValue = getFemaleWithPreschooler(); + break; + case 8: + returnValue = getFullTimeWorker(); + break; + case 9: + returnValue = getPartTimeWorker(); + break; + case 10: + returnValue = getUniversityStudent(); + break; + case 11: + returnValue = getDrivingAgeStudent(); + break; + case 12: + returnValue = getNonWorkerInHH(); + break; + case 13: + returnValue = getJointTourInHH(); + break; + case 14: + returnValue = getFirstTour(); + break; + case 15: + returnValue = getSubsequentTour(); + break; + case 16: + returnValue = getModeChoiceLogsumAlt(arrayIndex); + break; + case 17: + returnValue = getSubsequentTourIsWorkTour(); + break; + case 18: + returnValue = getSubsequentTourIsSchoolTour(); + break; + case 19: + returnValue = getEndOfPreviousTour(); + break; + case 20: + returnValue = getAllAdultsFullTimeWorkers(); + break; + case 21: + returnValue = getNonWorker(); + break; + case 22: + returnValue = getRetired(); + break; + case 23: + returnValue = getSchoolChildWithMandatoryTour(); + break; + case 24: + returnValue = getPreschoolChildWithMandatoryTour(); + break; + case 25: + returnValue = getNumberOfNonEscortingIndividualTours(); + break; + case 26: + returnValue = getNumberOfDiscretionaryTours(); + break; + case 27: + returnValue = getIndividualTour(); + break; + case 28: + returnValue = getJointTour(); + break; + case 29: + returnValue = getHouseholdSize(); + break; + case 30: + returnValue = getKidsOnJointTour(); + break; + case 31: + returnValue = getNumIndivShopTours() - 1; + break; + case 32: + returnValue = getNumIndivMaintTours() - 1; + break; + case 33: + returnValue = getNumIndivVisitTours() - 1; + break; + case 34: + returnValue = getNumIndivDiscrTours() - 1; + break; + case 35: + returnValue = getMaximumAvailableTimeWindow(); + break; + case 36: + returnValue = getWorkerWithMandatoryPattern(); + break; + case 37: + returnValue = getUniversityWithMandatoryPattern(); + break; + case 38: + returnValue = getNumChildrenUnder16InHh() > 0 ? 1 : 0; + break; + case 39: + returnValue = getToursLeftToSchedule(); + break; + case 40: + returnValue = getPreDrivingAgeChild(); + break; + case 41: + returnValue = getJointTourPartySize(); + break; + case 42: + returnValue = getSubtourPurposeIsEatOut(); + break; + case 43: + returnValue = getSubtourPurposeIsBusiness(); + break; + case 44: + returnValue = getSubtourPurposeIsOther(); + break; + case 45: + returnValue = getMaxJointTimeWindow(); + break; + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java new file mode 100644 index 0000000..79e76ed --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java @@ -0,0 +1,502 @@ +package org.sandag.abm.application; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import org.sandag.abm.ctramp.BikeLogsum; +import org.sandag.abm.ctramp.BikeLogsumSegment; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Tour; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import org.sandag.abm.ctramp.ModelStructure; + + +public class SandagTourModeChoiceDMU + extends TourModeChoiceDMU +{ + + private int setPersonHhTourCounter = 0; + private BikeLogsum bls; + protected double inboundFemaleBikeLogsum; + protected double outboundFemaleBikeLogsum; + protected double inboundMaleBikeLogsum; + protected double outboundMaleBikeLogsum; + protected double femaleInParty; + protected double maleInParty; + + public SandagTourModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) + { + super(modelStructure, aLogger); + setupMethodIndexMap(); + } + + public float getTimeOutbound() + { + return tour.getTourDepartPeriod(); + } + + public float getTimeInbound() + { + return tour.getTourArrivePeriod(); + } + + public int getIncome() + { + return hh.getIncomeInDollars(); + } + + public int getAdults() + { + return hh.getNumPersons18plus(); + } + + public int getFemale() + { + return person.getPersonIsFemale(); + } + + public void setOrigDuDen(double arg) + { + origDuDen = arg; + } + + public void setOrigEmpDen(double arg) + { + origEmpDen = arg; + } + + public void setOrigTotInt(double arg) + { + origTotInt = arg; + } + + public void setDestDuDen(double arg) + { + destDuDen = arg; + } + + public void setDestEmpDen(double arg) + { + destEmpDen = arg; + } + + public void setDestTotInt(double arg) + { + destTotInt = arg; + } + + public double getODUDen() + { + return origDuDen; + } + + public double getOEmpDen() + { + return origEmpDen; + } + + public double getOTotInt() + { + return origTotInt; + } + + public double getDDUDen() + { + return destDuDen; + } + + public double getDEmpDen() + { + return destEmpDen; + } + + public double getDTotInt() + { + return destTotInt; + } + + public double getNm_walkTime_out() + { + return getNmWalkTimeOut(); + } + + public double getNm_walkTime_in() + { + return getNmWalkTimeIn(); + } + + public double getNm_bikeTime_out() + { + return getNmBikeTimeOut(); + } + + public double getNm_bikeTime_in() + { + return getNmBikeTimeIn(); + } + + + public void setBikeLogsum(BikeLogsum bls) + { + this.bls = bls; + } + + public void setPersonObject(Person person) + { + super.setPersonObject(person); + checkSetPersonHhTour(); + } + + public void setHouseholdObject(Household hh) + { + super.setHouseholdObject(hh); + checkSetPersonHhTour(); + } + + public void setTourObject(Tour tour) + { + super.setTourObject(tour); + checkSetPersonHhTour(); + } + + private void checkSetPersonHhTour() + { + setPersonHhTourCounter = (setPersonHhTourCounter+1) % 3; + if (setPersonHhTourCounter == 0) { + setParty(person,tour,hh); + setBikeLogsum(); + } + } + + public double getFemaleInParty() + { + return femaleInParty; + } + + public double getMaleInParty() + { + return maleInParty; + } + + public void setParty(Person person, Tour tour, Household hh) + { + if (person != null) { + femaleInParty = person.getPersonIsFemale(); + maleInParty = femaleInParty == 0 ? 1 : 0; + } else { + femaleInParty = 0; + maleInParty = 0; + for (int participant : tour.getPersonNumArray()) { + if (hh.getPerson(participant).getPersonIsFemale() == 1) + femaleInParty = 1; + else + maleInParty = 1; + } + } + } + + public double getInboundFemaleBikeLogsum() + { + return inboundFemaleBikeLogsum; + } + + public double getOutboundFemaleBikeLogsum() + { + return outboundFemaleBikeLogsum; + } + + public double getInboundMaleBikeLogsum() + { + return inboundMaleBikeLogsum; + } + + public double getOutboundMaleBikeLogsum() + { + return outboundMaleBikeLogsum; + } + + + private void setBikeLogsum(double inboundFemaleBikeLogsum, double outboundFemaleBikeLogsum, + double inboundMaleBikeLogsum , double outboundMaleBikeLogsum) + { + this.inboundFemaleBikeLogsum = inboundFemaleBikeLogsum; + this.outboundFemaleBikeLogsum = outboundFemaleBikeLogsum; + this.inboundMaleBikeLogsum = inboundMaleBikeLogsum; + this.outboundMaleBikeLogsum = outboundMaleBikeLogsum; + } + + private void setBikeLogsum() + { + int origin = tour.getTourOrigMgra(); + int dest = tour.getTourDestMgra(); + boolean mandatory = tour.getTourPrimaryPurposeIndex() <= 3; + setBikeLogsum(bls.getLogsum(new BikeLogsumSegment(true,mandatory,true),dest,origin), + bls.getLogsum(new BikeLogsumSegment(true,mandatory,false),origin,dest), + bls.getLogsum(new BikeLogsumSegment(false,mandatory,true),dest,origin), + bls.getLogsum(new BikeLogsumSegment(false,mandatory,false),origin,dest)); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getIncomeCategory", 2); + methodIndexMap.put("getAdults", 3); + methodIndexMap.put("getFemale", 4); + methodIndexMap.put("getHhSize", 5); + methodIndexMap.put("getAutos", 6); + methodIndexMap.put("getAge", 7); + methodIndexMap.put("getTourCategoryJoint", 8); + methodIndexMap.put("getNumberOfParticipantsInJointTour", 9); + methodIndexMap.put("getWorkTourModeIsSov", 10); + methodIndexMap.put("getWorkTourModeIsBike", 11); + methodIndexMap.put("getWorkTourModeIsHov", 12); + methodIndexMap.put("getPTazTerminalTime", 14); + methodIndexMap.put("getATazTerminalTime", 15); + methodIndexMap.put("getODUDen", 16); + methodIndexMap.put("getOEmpDen", 17); + methodIndexMap.put("getOTotInt", 18); + methodIndexMap.put("getDDUDen", 19); + methodIndexMap.put("getDEmpDen", 20); + methodIndexMap.put("getDTotInt", 21); + methodIndexMap.put("getTourCategoryEscort", 22); + methodIndexMap.put("getMonthlyParkingCost", 23); + methodIndexMap.put("getDailyParkingCost", 24); + methodIndexMap.put("getHourlyParkingCost", 25); + methodIndexMap.put("getReimburseProportion", 26); + methodIndexMap.put("getPersonType", 27); + methodIndexMap.put("getFreeParkingEligibility", 28); + methodIndexMap.put("getParkingArea", 29); + + methodIndexMap.put("getWorkTimeFactor", 30); + methodIndexMap.put("getNonWorkTimeFactor", 31); + methodIndexMap.put("getJointTourTimeFactor", 32); + methodIndexMap.put("getTransponderOwnership", 33); + + methodIndexMap.put("getFemaleInParty", 50); + methodIndexMap.put("getMaleInParty", 51); + methodIndexMap.put("getInboundFemaleBikeLogsum", 52); + methodIndexMap.put("getOutboundFemaleBikeLogsum", 53); + methodIndexMap.put("getInboundMaleBikeLogsum", 54); + methodIndexMap.put("getOutboundMaleBikeLogsum", 55); + + methodIndexMap.put("getIvtCoeff", 56); + methodIndexMap.put("getCostCoeff", 57); + methodIndexMap.put("getIncomeInDollars", 58); + methodIndexMap.put("getWalkSetLogSum", 59); + methodIndexMap.put("getPnrSetLogSum", 60); + methodIndexMap.put("getKnrSetLogSum", 61); + + methodIndexMap.put( "getOrigTaxiWaitTime", 70 ); + methodIndexMap.put( "getDestTaxiWaitTime", 71 ); + methodIndexMap.put( "getOrigSingleTNCWaitTime", 72 ); + methodIndexMap.put( "getDestSingleTNCWaitTime", 73 ); + methodIndexMap.put( "getOrigSharedTNCWaitTime", 74 ); + methodIndexMap.put( "getDestSharedTNCWaitTime", 75 ); + methodIndexMap.put( "getUseOwnedAV", 76); + + methodIndexMap.put("getNm_walkTime_out", 90); + methodIndexMap.put("getNm_walkTime_in", 91); + methodIndexMap.put("getNm_bikeTime_out", 92); + methodIndexMap.put("getNm_bikeTime_in", 93); + + methodIndexMap.put("getOriginMgra", 96); + methodIndexMap.put("getDestMgra", 97); + + + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 2: + returnValue = getIncomeCategory(); + break; + case 3: + returnValue = getAdults(); + break; + case 4: + returnValue = getFemale(); + break; + case 5: + returnValue = getHhSize(); + break; + case 6: + returnValue = getAutos(); + break; + case 7: + returnValue = getAge(); + break; + case 8: + returnValue = getTourCategoryJoint(); + break; + case 9: + returnValue = getNumberOfParticipantsInJointTour(); + break; + case 10: + returnValue = getWorkTourModeIsSov(); + break; + case 11: + returnValue = getWorkTourModeIsBike(); + break; + case 12: + returnValue = getWorkTourModeIsHov(); + break; + case 14: + returnValue = getPTazTerminalTime(); + break; + case 15: + returnValue = getATazTerminalTime(); + break; + case 16: + returnValue = getODUDen(); + break; + case 17: + returnValue = getOEmpDen(); + break; + case 18: + returnValue = getOTotInt(); + break; + case 19: + returnValue = getDDUDen(); + break; + case 20: + returnValue = getDEmpDen(); + break; + case 21: + returnValue = getDTotInt(); + break; + case 22: + returnValue = getTourCategoryEscort(); + break; + case 23: + returnValue = getMonthlyParkingCost(); + break; + case 24: + returnValue = getDailyParkingCost(); + break; + case 25: + returnValue = getHourlyParkingCost(); + break; + case 26: + returnValue = getReimburseProportion(); + break; + case 27: + returnValue = getPersonType(); + break; + case 28: + returnValue = getFreeParkingEligibility(); + break; + case 29: + returnValue = getParkingArea(); + break; + case 30: + returnValue = getWorkTimeFactor(); + break; + case 31: + returnValue = getNonWorkTimeFactor(); + break; + case 32: + returnValue = getJointTourTimeFactor(); + break; + case 33: + returnValue = getTransponderOwnership(); + break; + case 50: + returnValue = getFemaleInParty(); + break; + case 51: + returnValue = getMaleInParty(); + break; + case 52: + returnValue = getInboundFemaleBikeLogsum(); + break; + case 53: + returnValue = getOutboundFemaleBikeLogsum(); + break; + case 54: + returnValue = getInboundMaleBikeLogsum(); + break; + case 55: + returnValue = getOutboundMaleBikeLogsum(); + break; + case 56: + returnValue = getIvtCoeff(); + break; + case 57: + returnValue = getCostCoeff(); + break; + case 58: + returnValue = getIncomeInDollars(); + break; + case 59: + returnValue = getTransitLogSum(WTW, true) + getTransitLogSum(WTW, false); + break; + case 60: + returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); + break; + case 61: + returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); + break; + + case 70: return getOrigTaxiWaitTime(); + case 71: return getDestTaxiWaitTime(); + case 72: return getOrigSingleTNCWaitTime(); + case 73: return getDestSingleTNCWaitTime(); + case 74: return getOrigSharedTNCWaitTime(); + case 75: return getDestSharedTNCWaitTime(); + case 76: return getUseOwnedAV(); + + + + + + + + + + case 90: + returnValue = getNmWalkTimeOut(); + break; + case 91: + returnValue = getNmWalkTimeIn(); + break; + case 92: + returnValue = getNmBikeTimeOut(); + break; + case 93: + returnValue = getNmBikeTimeIn(); + break; + case 96: + returnValue = getOriginMgra(); + break; + case 97: + returnValue = getDestMgra(); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + } + + return returnValue; + + } +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java new file mode 100644 index 0000000..0ac87da --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java @@ -0,0 +1,56 @@ +package org.sandag.abm.application; + +import java.util.HashMap; +import org.sandag.abm.ctramp.TransponderChoiceDMU; + +public class SandagTransponderChoiceDMU + extends TransponderChoiceDMU +{ + + public SandagTransponderChoiceDMU() + { + super(); + setupMethodIndexMap(); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getAutoOwnership", 0); + methodIndexMap.put("getPctHighIncome", 1); + methodIndexMap.put("getPctMultipleAutos", 2); + methodIndexMap.put("getAvgtts", 3); + methodIndexMap.put("getDistanceFromFacility", 4); + methodIndexMap.put("getPctAltTimeCBD", 5); + methodIndexMap.put("getAvgTransitAccess", 6); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getAutoOwnership(); + case 1: + return getPctIncome100Kplus(); + case 2: + return getPctTazMultpleAutos(); + case 3: + return getExpectedTravelTimeSavings(); + case 4: + return getTransponderDistance(); + case 5: + return getPctDetour(); + case 6: + return getAccessibility(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java new file mode 100644 index 0000000..a2a3b2f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java @@ -0,0 +1,572 @@ +package org.sandag.abm.application; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.BikeLogsum; +import org.sandag.abm.ctramp.BikeLogsumSegment; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Tour; +import org.sandag.abm.ctramp.TripModeChoiceDMU; + +import com.pb.common.calculator.IndexValues; + +public class SandagTripModeChoiceDMU + extends TripModeChoiceDMU +{ + private int setPersonHhTourCounter = 0; + private BikeLogsum bls; + protected double femaleBikeLogsum; + protected double maleBikeLogsum; + protected double femaleInParty; + protected double maleInParty; + + public SandagTripModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) + { + super(modelStructure, aLogger); + setupMethodIndexMap(); + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU destination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getEscortTour() + { + return escortTour; + } + + @Override + public int getTourCategoryJoint() + { + return jointTour; + } + + @Override + public int getNumberOfParticipantsInJointTour() + { + return partySize; + } + + public int getAutos() + { + return autos; + } + + public int getAge() + { + return age; + } + + public int getAdults() + { + return adults; + } + + public int getHhSize() + { + return hhSize; + } + + public int getFemale() + { + return personIsFemale; + } + + public int getIncome() + { + return incomeInDollars; + } + + @Override + public float getTimeOutbound() + { + return departPeriod; + } + + @Override + public float getTimeInbound() + { + return arrivePeriod; + } + + public int getTimeTrip() + { + return tripPeriod; + } + + public int getOutboundStops() + { + return outboundStops; + } + + public int getReturnStops() + { + return inboundStops; + } + + public int getTourModeIsDA() + { + return tourModeIsDA; + } + + public int getTourModeIsS2() + { + return tourModeIsS2; + } + + public int getTourModeIsS3() + { + return tourModeIsS3; + } + + public int getTourModeIsWalk() + { + return tourModeIsWalk; + } + + public int getTourModeIsBike() + { + return tourModeIsBike; + } + + public int getTourModeIsWTran() + { + return tourModeIsWTran; + } + + public int getTourModeIsPNR() + { + return tourModeIsPnr; + } + + public int getTourModeIsKNR() + { + return tourModeIsKnr; + } + + public int getTourModeIsSchBus() + { + return tourModeIsSchBus; + } + + public void setBikeLogsum(BikeLogsum bls) + { + this.bls = bls; + } + + public void setPersonObject(Person person) + { + super.setPersonObject(person); + checkSetPersonHhTour(); + } + + public void setHouseholdObject(Household hh) + { + super.setHouseholdObject(hh); + checkSetPersonHhTour(); + } + + public void setTourObject(Tour tour) + { + super.setTourObject(tour); + checkSetPersonHhTour(); + } + + private void checkSetPersonHhTour() + { + setPersonHhTourCounter = (setPersonHhTourCounter+1) % 3; + if (setPersonHhTourCounter == 0) { + setParty(person,tour,hh); + } + } + + public double getFemaleInParty() + { + return femaleInParty; + } + + public double getMaleInParty() + { + return maleInParty; + } + + public void setParty(Person person, Tour tour, Household hh) + { + if (person != null) { + femaleInParty = person.getPersonIsFemale(); + maleInParty = femaleInParty == 0 ? 1 : 0; + } else { + femaleInParty = 0; + maleInParty = 0; + for (int participant : tour.getPersonNumArray()) { + if (hh.getPerson(participant).getPersonIsFemale() == 1) + femaleInParty = 1; + else + maleInParty = 1; + } + } + } + + public void setBikeLogsum(int origin, int dest, boolean inbound) { + boolean mandatory = tour.getTourPrimaryPurposeIndex() <= 3; + femaleBikeLogsum = bls.getLogsum(new BikeLogsumSegment(true,mandatory,inbound),origin,dest); + maleBikeLogsum = bls.getLogsum(new BikeLogsumSegment(false,mandatory,inbound),origin,dest); + } + + public double getFemaleBikeLogsum() { + return femaleBikeLogsum; + } + + public double getMaleBikeLogsum() { + return maleBikeLogsum; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getAutos", 1); + methodIndexMap.put("getAdults", 2); + methodIndexMap.put("getHhSize", 3); + methodIndexMap.put("getFemale", 4); + methodIndexMap.put("getIncomeCategory", 5); + methodIndexMap.put("getTimeOutbound", 6); + methodIndexMap.put("getTimeInbound", 7); + methodIndexMap.put("getTimeTrip", 8); + methodIndexMap.put("getTourCategoryJoint", 9); + methodIndexMap.put("getNumberOfParticipantsInJointTour", 10); + methodIndexMap.put("getOutboundStops", 11); + methodIndexMap.put("getReturnStops", 12); + methodIndexMap.put("getFirstTrip", 13); + methodIndexMap.put("getLastTrip", 14); + methodIndexMap.put("getTourModeIsDA", 15); + methodIndexMap.put("getTourModeIsS2", 16); + methodIndexMap.put("getTourModeIsS3", 17); + methodIndexMap.put("getTourModeIsWalk", 18); + methodIndexMap.put("getTourModeIsBike", 19); + methodIndexMap.put("getTourModeIsWTran", 20); + methodIndexMap.put("getTourModeIsPNR", 21); + methodIndexMap.put("getTourModeIsKNR", 22); + methodIndexMap.put("getODUDen", 23); + methodIndexMap.put("getOEmpDen", 24); + methodIndexMap.put("getOTotInt", 25); + methodIndexMap.put("getDDUDen", 26); + methodIndexMap.put("getDEmpDen", 27); + methodIndexMap.put("getDTotInt", 28); + methodIndexMap.put("getPTazTerminalTime", 30); + methodIndexMap.put("getATazTerminalTime", 31); + methodIndexMap.put("getAge", 32); + methodIndexMap.put("getTourModeIsSchBus", 33); + methodIndexMap.put("getEscortTour", 34); + methodIndexMap.put("getAutoModeAllowedForTripSegment", 35); + methodIndexMap.put("getWalkModeAllowedForTripSegment", 36); + methodIndexMap.put("getSegmentIsIk", 37); + methodIndexMap.put("getReimburseAmount", 38); + methodIndexMap.put("getMonthlyParkingCostTourDest", 39); + methodIndexMap.put("getDailyParkingCostTourDest", 40); + methodIndexMap.put("getHourlyParkingCostTourDest", 41); + methodIndexMap.put("getHourlyParkingCostTripOrig", 42); + methodIndexMap.put("getHourlyParkingCostTripDest", 43); + methodIndexMap.put("getTripOrigIsTourDest", 44); + methodIndexMap.put("getTripDestIsTourDest", 45); + methodIndexMap.put("getFreeOnsite", 46); + methodIndexMap.put("getPersonType", 47); + + methodIndexMap.put("getFemaleInParty", 50); + methodIndexMap.put("getMaleInParty", 51); + methodIndexMap.put("getFemaleBikeLogsum", 52); + methodIndexMap.put("getMaleBikeLogsum", 53); + + methodIndexMap.put("getTransponderOwnership", 54); + methodIndexMap.put("getWorkTimeFactor", 55); + methodIndexMap.put("getNonWorkTimeFactor", 56); + methodIndexMap.put("getJointTourTimeFactor", 57); + + methodIndexMap.put("getInbound",58); + + methodIndexMap.put("getIncomeInDollars",59); + methodIndexMap.put("getIvtCoeff", 60); + methodIndexMap.put("getCostCoeff", 61); + + methodIndexMap.put("getWalkSetLogSum", 62); + methodIndexMap.put("getPnrSetLogSum", 63); + methodIndexMap.put("getKnrSetLogSum", 64); + + methodIndexMap.put("getWaitTimeTaxi", 70); + methodIndexMap.put("getWaitTimeSingleTNC", 71); + methodIndexMap.put("getWaitTimeSharedTNC", 72); + methodIndexMap.put("getUseOwnedAV", 73); + + methodIndexMap.put("getNm_walkTime", 90); + methodIndexMap.put("getNm_bikeTime", 91); + + methodIndexMap.put("getOriginMgra", 93); + methodIndexMap.put("getDestMgra", 94); + + + methodIndexMap.put("getTourModeIsTncTransit", 95); + methodIndexMap.put("getTourModeIsMaas", 96); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 1: + returnValue = getAutos(); + break; + case 2: + returnValue = getAdults(); + break; + case 3: + returnValue = getHhSize(); + break; + case 4: + returnValue = getFemale(); + break; + case 5: + returnValue = getIncome(); + break; + case 6: + returnValue = getTimeOutbound(); + break; + case 7: + returnValue = getTimeInbound(); + break; + case 8: + returnValue = getTimeTrip(); + break; + case 9: + returnValue = getTourCategoryJoint(); + break; + case 10: + returnValue = getNumberOfParticipantsInJointTour(); + break; + case 11: + returnValue = getOutboundStops(); + break; + case 12: + returnValue = getReturnStops(); + break; + case 13: + returnValue = getFirstTrip(); + break; + case 14: + returnValue = getLastTrip(); + break; + case 15: + returnValue = getTourModeIsDA(); + break; + case 16: + returnValue = getTourModeIsS2(); + break; + case 17: + returnValue = getTourModeIsS3(); + break; + case 18: + returnValue = getTourModeIsWalk(); + break; + case 19: + returnValue = getTourModeIsBike(); + break; + case 20: + returnValue = getTourModeIsWTran(); + break; + case 21: + returnValue = getTourModeIsPnr(); + break; + case 22: + returnValue = getTourModeIsKnr(); + break; + case 23: + returnValue = getODUDen(); + break; + case 24: + returnValue = getOEmpDen(); + break; + case 25: + returnValue = getOTotInt(); + break; + case 26: + returnValue = getDDUDen(); + break; + case 27: + returnValue = getDEmpDen(); + break; + case 28: + returnValue = getDTotInt(); + break; + case 30: + returnValue = getPTazTerminalTime(); + break; + case 31: + returnValue = getATazTerminalTime(); + break; + case 32: + returnValue = getAge(); + break; + case 33: + returnValue = getTourModeIsSchBus(); + break; + case 34: + returnValue = getEscortTour(); + break; + case 35: + returnValue = getAutoModeAllowedForTripSegment(); + break; + case 36: + returnValue = getWalkModeAllowedForTripSegment(); + break; + case 37: + returnValue = getSegmentIsIk(); + break; + case 38: + returnValue = getReimburseAmount(); + break; + case 39: + returnValue = getMonthlyParkingCostTourDest(); + break; + case 40: + returnValue = getDailyParkingCostTourDest(); + break; + case 41: + returnValue = getHourlyParkingCostTourDest(); + break; + case 42: + returnValue = getHourlyParkingCostTripOrig(); + break; + case 43: + returnValue = getHourlyParkingCostTripDest(); + break; + case 44: + returnValue = getTripOrigIsTourDest(); + break; + case 45: + returnValue = getTripDestIsTourDest(); + break; + case 46: + returnValue = getFreeOnsite(); + break; + case 47: + returnValue = getPersonType(); + break; + case 50: + returnValue = getFemaleInParty(); + break; + case 51: + returnValue = getMaleInParty(); + break; + case 52: + returnValue = getFemaleBikeLogsum(); + break; + case 53: + returnValue = getMaleBikeLogsum(); + break; + case 54: + returnValue = getTransponderOwnership(); + break; + case 55: + returnValue = getWorkTimeFactor(); + break; + case 56: + returnValue = getNonWorkTimeFactor(); + break; + case 57: + returnValue = getJointTourTimeFactor(); + break; + case 58: + returnValue = getInbound(); + break; + case 59: + returnValue = getIncomeInDollars(); + break; + case 60: + returnValue = getIvtCoeff(); + break; + case 61: + returnValue = getCostCoeff(); + break; + case 62: + returnValue = getTransitLogSum(WTW); + break; + case 63: + if ( outboundHalfTourDirection == 1 ) + returnValue = getTransitLogSum(DTW); + else + returnValue = getTransitLogSum(WTD); + break; + case 64: + if ( outboundHalfTourDirection == 1 ) + returnValue = getTransitLogSum(DTW); + else + returnValue = getTransitLogSum(WTD); + break; + case 70: return getWaitTimeTaxi(); + case 71: return getWaitTimeSingleTNC(); + case 72: return getWaitTimeSharedTNC(); + case 73: return getUseOwnedAV(); + case 90: + returnValue = getNm_walkTime(); + break; + case 91: + returnValue = getNm_bikeTime(); + break; + case 93: + returnValue = getOriginMgra(); + break; + case 94: + returnValue = getDestMgra(); + break; + case 95: + returnValue = getTourModeIsTncTransit(); + break; + case 96: + returnValue = getTourModeIsMaas(); + break; + + + default: + logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + } +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java new file mode 100644 index 0000000..a8c81f0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java @@ -0,0 +1,904 @@ +package org.sandag.abm.application; + +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixIO32BitJvm; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class SandagTripTables +{ + + private static Logger logger = Logger.getLogger("tripTables"); + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; + private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; + + + private TableDataSet indivTripData; + private TableDataSet jointTripData; + + // Some parameters + private int[] modeIndex; // an + private int[] matrixIndex; // an + + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private String[] purposeName = {"Work", "University", "School", + "Escort", "Shop", "Maintenance", "EatingOut", "Visiting", "Discretionary", "WorkBased"}; + + // matrices are indexed by modes (auto, non-mot,tran,other), valueoftime bins, and sub-modes(shared2gp, etc). + private Matrix[][][] matrix; + + private HashMap rbMap; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + + + private float[][] CBDVehicles; // an // period + private float[][] PNRVehicles; // an + private float sampleRate; + private int iteration; + private MatrixType mt; + private MatrixDataServerRmi ms; + + private String[] indivColumns = {"stop_period", "orig_mgra", + "dest_mgra", "trip_mode", "inbound", "trip_board_tap", "trip_alight_tap", "set", + "parking_mgra", "tour_purpose", "valueOfTime", "transponder_avail" }; + + private String[] jointColumns = {"stop_period", "orig_mgra", + "dest_mgra", "trip_mode", "inbound", "trip_board_tap", "trip_alight_tap", "set", + "parking_mgra", "tour_purpose", "num_participants", "valueOfTime", "transponder_avail"}; + + private HashMap averageOcc3Plus; // a + + private float valueOfTimeThresholdLow = 0; + private float valueOfTimeThresholdMed = 0; + //value of time bins by mode group + int[] votBins = {3,1,1,1}; + + boolean segmentByTransponderOwnership; + + public int numSkimSets; + + + /** + * Constructor. + * + * @param rbMap + * HashMap formed from a property map, which includes environment + * variables and arguments passed in as -d to VM + * @param sampleRate + * Sample rate 0->1.0 + * @param iteration + * Iteration number, program will look for trip file names with + * _iteration appended + */ + public SandagTripTables(HashMap rbMap, float sampleRate, int iteration) + { + + this.rbMap = rbMap; + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + segmentByTransponderOwnership = Util.getBooleanValueFromPropertyMap(rbMap,"Results.segmentByTransponderOwnership"); + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTourModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + logger.info("Tour mode "+i+" is auto"); + } else if (modelStructure.getTourModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + logger.info("Tour mode "+i+" is non-motorized"); + } else if (modelStructure.getTourModeIsWalkTransit(i) + || modelStructure.getTourModeIsDriveTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + logger.info("Tour mode "+i+" is transit"); + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + logger.info("Tour mode "+i+" is other"); + } + } + logger.info("Total auto modes = "+autoModes); + logger.info("Total non-motorized modes = "+nmotModes); + logger.info("Total transit modes = "+tranModes); + logger.info("Total other modes = "+othrModes); + + readOccupancies(); + // Initialize arrays (need for all periods, so initialize here) + CBDVehicles = new float[mgraManager.getMaxMgra() + 1][numberOfPeriods]; + PNRVehicles = new float[tapManager.getMaxTap() + 1][numberOfPeriods]; + + setSampleRate(sampleRate); + setIteration(iteration); + + //value of time thresholds + valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); + valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); + + } + + /** + * Read occupancies from the properties file and store in the + * averageOcc3Plus HashMap + */ + public void readOccupancies() + { + + averageOcc3Plus = new HashMap(); + + for (int i = 0; i < purposeName.length; ++i) + { + String searchString = "occ3plus.purpose." + purposeName[i]; + float occupancy = new Float(Util.getStringValueFromPropertyMap(rbMap, searchString)); + averageOcc3Plus.put(purposeName[i], occupancy); + } + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) and value of time group + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][][]; + for (int i = 0; i < numberOfModes; ++i) + { + matrix[i] = new Matrix[votBins[i]][]; + + for(int j = 0; j< votBins[i];++j){ + + String modeName; + + + if (i == 0) + { + + int autoModeSegments = autoModes; + String transponderLabel = ""; + + if(segmentByTransponderOwnership) { + autoModeSegments *=2; //twice as many since segmentation would be by number of auto modes and by 0,1 for ownership + transponderLabel = "NOTRPDR"; + } + matrix[i][j] = new Matrix[autoModeSegments]; + + for (int k = 0; k < autoModes; ++k) + { + modeName = modelStructure.getModeName(k + 1); + matrix[i][j][k] = new Matrix(modeName + transponderLabel + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + + for (int k = autoModes; k < autoModeSegments; ++k) + { + modeName = modelStructure.getModeName((k + 1)-autoModes); + matrix[i][j][k] = new Matrix(modeName + "TRPDR"+ "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + + + + } else if (i == 1){ + + matrix[i][j] = new Matrix[nmotModes]; + for (int k = 0; k < nmotModes; ++k) + { + modeName = modelStructure.getModeName(k + 1 + autoModes); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + + } else if (i == 2){ + + matrix[i][j] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l1) + votBin = getValueOfTimeBin(valueOfTime); + + if (mode == 0) + { + if(segmentByTransponderOwnership) { + int ownsTransponder = (int) tripData.getValueAt(i, "transponder_avail"); + if(ownsTransponder==1) + mat = mat + SandagModelStructure.TRIP_SOV_ALTS.length + SandagModelStructure.TRIP_HOV_ALTS.length; + } + // look up what taz the parking mgra is in, and re-assign the + // trip destination to the parking taz + if (parkingMGRA > 0) + { + parkingTaz = mgraManager.getTaz(parkingMGRA); + destinationTAZ = parkingTaz; + CBDVehicles[parkingMGRA][period] = CBDVehicles[parkingMGRA][period] + + vehicleTrips; + } + + + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); + } else if (mode == 1) + { + + + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } else if (mode == 2) + { + + if (boardTap == 0 || alightTap == 0) continue; + + //store transit trips in matrices + mat = (matrixIndex[tripMode]*numSkimSets)+set; + + float value=0; + try{ + value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); + }catch(Exception e){ + + logger.fatal("Error trying to get transit trips from matrix"); + logger.fatal("boardTap,alightTap,set: "+boardTap+","+alightTap+","+set); + logger.fatal("tripMode,mode,votBin,mat: "+tripMode+","+mode+","+votBin+","+mat); + logger.fatal("number of skimsets: "+numSkimSets); + logger.fatal("total board taps in matrix:" + matrix[mode][votBin][mat].getRowCount()); + logger.fatal("total alight taps in matrix:" + matrix[mode][votBin][mat].getColumnCount()); + throw new RuntimeException(e); + } + matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); + + // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) + if (modelStructure.getTourModeIsDriveTransit(tripMode)) + { + + // add the tNCVehicle trip portion to the trip table + if (inbound == 0) + { // from origin to lot (boarding tap) + int PNRTAZ = tapManager.getTazForTap(boardTap); + + + value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); + matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); + + // and increment up the array of parked vehicles at the + // lot + ++PNRVehicles[boardTap][period]; + + } else + { // from lot (alighting tap) to destination + int PNRTAZ = tapManager.getTazForTap(alightTap); + + + value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); + matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); + } + + } + } else + { + + + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } + } + + //logger.info("End creating trip tables for period " + timePeriod); + } + + /** + * Return the value of time bin 0 through 2 based on the thresholds provided in the property map + * @param valueOfTime + * @return value of time bin 0 through 2 + */ + public int getValueOfTimeBin(float valueOfTime){ + + if(valueOfTime1) + end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; + else + end[i][j] = "_" + per + ".omx"; + } + } + + for (int i = 0; i < 4; ++i) + { + for(int j = 0; j < votBins[i];++j){ + try + { + //Delete the file if it exists + File f = new File(fileName[i]+end[i][j]); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); + f.delete(); + } + + if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); + else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); + } catch (Exception e) + { + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName[i] +end[i][j] + ", for mode index = " + i, e); + throw new RuntimeException(); + } + } + } + + } + + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + /** + * Connect to matrix server + */ + private void connectToMatrixServer() + { + + + // get matrix server address and port + String matrixServerAddress = Util.getStringValueFromPropertyMap(rbMap, "RunModel.MatrixServerAddress"); + int serverPort = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, "RunModel.MatrixServerPort")); + + ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + logger.info("connected to matrix data server"); + } + + /** + * Write a file of vehicles parking in parking-constrained areas by MGRA. + * + * @param fileName + * The name of the csv file to write to. + */ + public void writeCBDFile(String fileName) + { + + try + { + FileWriter writer = new FileWriter(fileName); + + // write header + writer.append("MGRA,"); + + for (int j = 0; j < numberOfPeriods; ++j) + writer.append(modelStructure.getModelPeriodLabel(j) + ","); + + writer.append("Total\n"); + + // iterate through mgras + for (int i = 0; i < CBDVehicles.length; ++i) + { + + float totalVehicles = 0; + for (int j = 0; j < numberOfPeriods; ++j) + { + totalVehicles += CBDVehicles[i][j]; + } + + // only write the mgra if there are vehicles parked there + if (totalVehicles > 0) + { + + writer.append(Integer.toString(i)); + + // iterate through periods + for (int j = 0; j < numberOfPeriods; ++j) + writer.append("," + Float.toString(CBDVehicles[i][j])); + + writer.append("," + Float.toString(totalVehicles) + "\n"); + writer.flush(); + } + } + writer.flush(); + writer.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + + } + + /** + * Write a file of vehicles parking in PNR lots by TAP. + * + * @param fileName + * The name of the csv file to write to. + */ + public void writePNRFile(String fileName) + { + + try + { + FileWriter writer = new FileWriter(fileName); + + // write header + writer.append("TAP,"); + + for (int j = 0; j < numberOfPeriods; ++j) + writer.append(modelStructure.getModelPeriodLabel(j) + ","); + + writer.append("Total\n"); + + // iterate through taps + for (int i = 0; i < PNRVehicles.length; ++i) + { + + float totalVehicles = 0; + for (int j = 0; j < numberOfPeriods; ++j) + { + totalVehicles += PNRVehicles[i][j]; + } + + // only write the tap if there are vehicles parked there + if (totalVehicles > 0) + { + + writer.append(Integer.toString(i)); + + // iterate through periods + for (int j = 0; j < numberOfPeriods; ++j) + writer.append("," + Float.toString(PNRVehicles[i][j])); + + writer.append("," + Float.toString(totalVehicles) + "\n"); + writer.flush(); + } + } + writer.flush(); + writer.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + + } + + /** + * Set the sample rate + * + * @param sampleRate + * The sample rate, used for expanding trips + */ + public void setSampleRate(float sampleRate) + { + this.sampleRate = sampleRate; + } + + /** + * Set the iteration number + * + * @param sampleRate + * The iteration number, should be appended to trip files as + * _iteration + */ + public void setIteration(int iteration) + { + this.iteration = iteration; + } + + + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + + float sampleRate = 1.0f; + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + + logger.info(String.format("-sampleRate %.4f.", sampleRate)); + logger.info("-iteration " + iteration); + + SandagTripTables tripTables = new SandagTripTables(pMap, sampleRate, iteration); + + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + tripTables.mt = MatrixType.lookUpMatrixType(matrixTypeName); + tripTables.createTripTables(tripTables.mt); + + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java new file mode 100644 index 0000000..00b2800 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java @@ -0,0 +1,51 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.crossborder; + +import java.io.Serializable; + +/** + * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects + * + * @author Joel Freedman + */ +public class CrossBorderDmuFactory + implements CrossBorderDmuFactoryIf, Serializable +{ + + private CrossBorderModelStructure crossBorderModelStructure; + + public CrossBorderDmuFactory(CrossBorderModelStructure modelStructure) + { + this.crossBorderModelStructure = modelStructure; + } + + public CrossBorderTourModeChoiceDMU getCrossBorderTourModeChoiceDMU() + { + return new CrossBorderTourModeChoiceDMU(crossBorderModelStructure); + } + + public CrossBorderTripModeChoiceDMU getCrossBorderTripModeChoiceDMU() + { + return new CrossBorderTripModeChoiceDMU(crossBorderModelStructure, null); + } + + public CrossBorderStationDestChoiceDMU getCrossBorderStationChoiceDMU() + { + return new CrossBorderStationDestChoiceDMU(crossBorderModelStructure); + } + + public CrossBorderStopLocationChoiceDMU getCrossBorderStopLocationChoiceDMU() + { + return new CrossBorderStopLocationChoiceDMU(crossBorderModelStructure); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java new file mode 100644 index 0000000..70b68da --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java @@ -0,0 +1,17 @@ +package org.sandag.abm.crossborder; + +/** + * A DMU factory interface + */ +public interface CrossBorderDmuFactoryIf +{ + + CrossBorderTourModeChoiceDMU getCrossBorderTourModeChoiceDMU(); + + CrossBorderStationDestChoiceDMU getCrossBorderStationChoiceDMU(); + + CrossBorderTripModeChoiceDMU getCrossBorderTripModeChoiceDMU(); + + CrossBorderStopLocationChoiceDMU getCrossBorderStopLocationChoiceDMU(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java new file mode 100644 index 0000000..48f2c2b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java @@ -0,0 +1,525 @@ +package org.sandag.abm.crossborder; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.concurrent.ForkJoinPool; +import java.util.concurrent.atomic.AtomicBoolean; +import java.util.concurrent.atomic.AtomicInteger; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.crossborder.CrossBorderTourManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; +import com.pb.sawdust.util.concurrent.DnCRecursiveAction; + +public class CrossBorderModel +{ + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + public static final String RUN_MODEL_CONCURRENT_PROPERTY_KEY = "crossBorder.run.concurrent"; + public static final String CONCURRENT_PARALLELISM_PROPERTY_KEY = "crossBorder.concurrent.parallelism"; + + private static final Logger LOGGER = Logger.getLogger(SandagTourBasedModel.class); + private static final Object INITIALIZATION_LOCK = new Object(); + + private MatrixDataServerRmi ms; + private HashMap rbMap; + private AutoTazSkimsCalculator tazDistanceCalculator; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private CrossBorderTourManager tourManager; + + private boolean seek; + private int traceId; + private double sampleRate = 1; + + /** + * Constructor + * + * @param rbMap + */ + public CrossBorderModel(HashMap rbMap) + { + this.rbMap = rbMap; + + synchronized (INITIALIZATION_LOCK) + { // lock to make sure only one of + // these actually initializes + // things so we don't cross + // threads + mgraManager = MgraDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + tourManager = new CrossBorderTourManager(rbMap); + } + + seek = Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.seek")); + traceId = Integer.valueOf(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trace")); + + } + + // global variable used for reporting + private static final AtomicInteger TOUR_COUNTER = new AtomicInteger(0); + private final AtomicBoolean calculatorsInitialized = new AtomicBoolean(false); + + /** + * Run the model for a subset of tours in an array of tours. + * + * @param tours + * The array of tours. + * @param start + * The starting index of the tours to process. + * @param end + * The (exclusive) ending index of the tours to process. + */ + private void runModel(CrossBorderTour[] tours, int start, int end) + { + CrossBorderModelStructure modelStructure = new CrossBorderModelStructure(); + CrossBorderDmuFactoryIf dmuFactory = new CrossBorderDmuFactory(modelStructure); + + if (!calculatorsInitialized.get()) + { + // only let one thread in to initialize + synchronized (calculatorsInitialized) + { + // if still not initialized, then this is the first in so do the + // initialization (otherwise skip) + if (!calculatorsInitialized.get()) + { + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + calculatorsInitialized.set(true); + } + } + } + + CrossBorderTourTimeOfDayChoiceModel todChoiceModel = new CrossBorderTourTimeOfDayChoiceModel( + rbMap); + CrossBorderStationDestChoiceModel destChoiceModel = new CrossBorderStationDestChoiceModel( + rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + CrossBorderTourModeChoiceModel tourModeChoiceModel = new CrossBorderTourModeChoiceModel(rbMap, modelStructure, dmuFactory, + tazDistanceCalculator); + + CrossBorderTripModeChoiceModel tripModeChoiceModel = new CrossBorderTripModeChoiceModel(rbMap, modelStructure, + dmuFactory, tazDistanceCalculator); + destChoiceModel.calculateSizeTerms(dmuFactory); + destChoiceModel.calculateTazProbabilities(dmuFactory); + + CrossBorderStopFrequencyModel stopFrequencyModel = new CrossBorderStopFrequencyModel(rbMap); + CrossBorderStopPurposeModel stopPurposeModel = new CrossBorderStopPurposeModel(rbMap); + + CrossBorderStopTimeOfDayChoiceModel stopTodChoiceModel = new CrossBorderStopTimeOfDayChoiceModel( + rbMap); + CrossBorderStopLocationChoiceModel stopLocationChoiceModel = new CrossBorderStopLocationChoiceModel( + rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + + double[][] mgraSizeTerms = destChoiceModel.getMgraSizeTerms(); + double[][] tazSizeTerms = destChoiceModel.getTazSizeTerms(); + double[][][] mgraProbabilities = destChoiceModel.getMgraProbabilities(); + stopLocationChoiceModel.setMgraSizeTerms(mgraSizeTerms); + stopLocationChoiceModel.setTazSizeTerms(tazSizeTerms); + stopLocationChoiceModel.setMgraProbabilities(mgraProbabilities); + String purposeControlFileName = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory")+"input/crossBorder_tourPurpose_control.csv"; + + for (int i = start; i < end; i++) + { + CrossBorderTour tour = tours[i]; + + // sample tours + double rand = tour.getRandom(); + if (rand > sampleRate) continue; + + int tourCount = TOUR_COUNTER.incrementAndGet(); + if (tourCount % 1000 == 0) LOGGER.info("Processing tour " + tourCount); + + if (seek && tour.getID() != traceId) continue; + + if (tour.getID() == traceId) tour.setDebugChoiceModels(true); + + todChoiceModel.calculateTourTOD(tour); + destChoiceModel.chooseStationAndDestination(tour); + resetCrossingPurpose(purposeControlFileName,tour); + tourModeChoiceModel.chooseTourMode(tour); + stopFrequencyModel.calculateStopFrequency(tour); + stopPurposeModel.calculateStopPurposes(tour); + + int outboundStops = tour.getNumberOutboundStops(); + int inboundStops = tour.getNumberInboundStops(); + + // choose TOD for stops and location of each + if (outboundStops > 0) + { + CrossBorderStop[] stops = tour.getOutboundStops(); + for (CrossBorderStop stop : stops) + { + stopTodChoiceModel.chooseTOD(tour, stop); + stopLocationChoiceModel.chooseStopLocation(tour, stop); + } + } + if (inboundStops > 0) + { + CrossBorderStop[] stops = tour.getInboundStops(); + for (CrossBorderStop stop : stops) + { + stopTodChoiceModel.chooseTOD(tour, stop); + stopLocationChoiceModel.chooseStopLocation(tour, stop); + } + } + + // generate trips and choose mode for them + CrossBorderTrip[] trips = new CrossBorderTrip[outboundStops + inboundStops + 2]; + int tripNumber = 0; + + // outbound stops + if (outboundStops > 0) + { + CrossBorderStop[] stops = tour.getOutboundStops(); + for (CrossBorderStop stop : stops) + { + // generate a trip to the stop and choose a mode for it + trips[tripNumber] = new CrossBorderTrip(tour, stop, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + // generate a trip from the last stop to the tour destination + trips[tripNumber] = new CrossBorderTrip(tour, stops[stops.length - 1], false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + } else + { + // generate an outbound trip from the tour origin to the + // destination and choose a mode + trips[tripNumber] = new CrossBorderTrip(tour, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + + // inbound stops + if (inboundStops > 0) + { + CrossBorderStop[] stops = tour.getInboundStops(); + for (CrossBorderStop stop : stops) + { + // generate a trip to the stop and choose a mode for it + trips[tripNumber] = new CrossBorderTrip(tour, stop, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + // generate a trip from the last stop to the tour origin + trips[tripNumber] = new CrossBorderTrip(tour, stops[stops.length - 1], false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } else + { + + // generate an inbound trip from the tour destination to the + // origin and choose a mode + trips[tripNumber] = new CrossBorderTrip(tour, false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + + // set the trips in the tour object + tour.setTrips(trips); + + } + + } + + /** + * This class is the divide-and-conquer action (void return task) for + * running the cross-border model using the fork-join framework. The + * divisible problem is an array of tours, and the actual work is the + * {@link CrossBorderModel#runModel(CrossBorderTour[],int,int)} method, + * applied to a section of the array. + */ + private class CrossBorderModelAction + extends DnCRecursiveAction + { + private final HashMap rbMap; + private final CrossBorderTour[] tours; + + private CrossBorderModelAction(HashMap rbMap, CrossBorderTour[] tours) + { + super(0, tours.length); + this.rbMap = rbMap; + this.tours = tours; + } + + private CrossBorderModelAction(HashMap rbMap, CrossBorderTour[] tours, + long start, long length, DnCRecursiveAction next) + { + super(start, length, next); + this.rbMap = rbMap; + this.tours = tours; + } + + @Override + protected void computeAction(long start, long length) + { + runModel(tours, (int) start, (int) (start + length)); + } + + @Override + protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) + { + return new CrossBorderModelAction(rbMap, tours, start, length, next); + } + + @Override + protected boolean continueDividing(long length) + { + // if there are 3 extra tasks queued up, then start executing + // if there are 1000 or less tours to process, then start executing + // otherwise, keep dividing to build up tasks for the threads to + // process + return getSurplusQueuedTaskCount() < 3 && length > 1000; + } + } + + /** + * Run visitor model. + */ + public void runModel() + { + tourManager.generateCrossBorderTours(); + CrossBorderTour[] tours = tourManager.getTours(); + + // get new keys to see if we want to run in concurrent mode, and the + // parallelism + // (defaults to single threaded and parallelism = # of processors) + // note that concurrent can use up memory very quickly, so setting the + // parallelism might be prudent + boolean concurrent = rbMap.containsKey(RUN_MODEL_CONCURRENT_PROPERTY_KEY) + && Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, + RUN_MODEL_CONCURRENT_PROPERTY_KEY)); + int parallelism = rbMap.containsKey(CONCURRENT_PARALLELISM_PROPERTY_KEY) ? Integer + .valueOf(Util.getStringValueFromPropertyMap(rbMap, + CONCURRENT_PARALLELISM_PROPERTY_KEY)) : Runtime.getRuntime() + .availableProcessors(); + + if (concurrent) + { // use fork-join + CrossBorderModelAction action = new CrossBorderModelAction(rbMap, tours); + new ForkJoinPool(parallelism).execute(action); + action.getResult(); // wait for finish + } else + { // single-threaded: call the model runner in this thread + runModel(tours, 0, tours.length); + } + + tourManager.writeOutputFile(rbMap); + LOGGER.info("Cross Border Model successfully completed!"); + } + + /** + * @return the sampleRate + */ + public double getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(double sampleRate) + { + this.sampleRate = sampleRate; + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + LOGGER.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + LOGGER.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * Reset tour NB crossing purpose using purpose distribution by POE from 2011 survey + */ + private void resetCrossingPurpose(String purposeControlFile, CrossBorderTour tour) { + int poe=tour.getPoe(); + double [][] purpDistributionByPoe=new double[5][CrossBorderModelStructure.NUMBER_CROSSBORDER_PURPOSES]; + + // Read the distributions by poe + for (int i=0; i<5; i++) { + purpDistributionByPoe[i] = tourManager.setPurposeDistribution(purposeControlFile,purpDistributionByPoe[i],i+3); + } + int purpose = tourManager.choosePurpose(tour.getRandom(), purpDistributionByPoe[poe]); + tour.setPurpose((byte) purpose); + } + + /** + * @param args + */ + public static void main(String[] args) + { + Runtime gfg = Runtime.getRuntime(); + long memory1; + // checking the total memeory + System.out.println("Total memory is: "+ gfg.totalMemory()); + // checking free memory + memory1 = gfg.freeMemory(); + System.out.println("Initial free memory at Xborder model: "+ memory1); + // calling the garbage collector on demand + gfg.gc(); + memory1 = gfg.freeMemory(); + System.out.println("Free memory after garbage "+ "collection: " + memory1); + + String propertiesFile = null; + HashMap pMap; + + LOGGER.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + LOGGER.info(String.format("Running Cross-Border Model")); + + if (args.length == 0) + { + LOGGER.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + float sampleRate = 1.0f; + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + + LOGGER.info("Crossborder Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + CrossBorderModel crossBorderModel = new CrossBorderModel(pMap); + crossBorderModel.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = crossBorderModel.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + crossBorderModel.ms = matrixServer; + } else + { + crossBorderModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + crossBorderModel.ms.testRemote("CrossBorderModel"); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(crossBorderModel.ms); + } + + } + + } catch (Exception e) + { + + LOGGER.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + crossBorderModel.runModel(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java new file mode 100644 index 0000000..d8baad1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java @@ -0,0 +1,106 @@ +package org.sandag.abm.crossborder; + +import org.sandag.abm.application.SandagModelStructure; + +public class CrossBorderModelStructure + extends SandagModelStructure +{ + + public static final byte NUMBER_CROSSBORDER_PURPOSES = 6; + public static final byte WORK = 0; + public static final byte SCHOOL = 1; + public static final byte CARGO = 2; + public static final byte SHOP = 3; + public static final byte VISIT = 4; + public static final byte OTHER = 5; + + public static final String[] CROSSBORDER_PURPOSES = {"WORK", "SCHOOL", "CARGO", "SHOP", + "VISIT", "OTHER" }; + + public static final byte DEPARTURE = 0; + public static final byte ARRIVAL = 1; + + public static final int AM = 0; + public static final int PM = 1; + public static final int OP = 2; + public static final int[] SKIM_PERIODS = {AM, PM, OP}; + public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; + public static final int UPPER_EA = 3; + public static final int UPPER_AM = 9; + public static final int UPPER_MD = 22; + public static final int UPPER_PM = 29; + public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; + + public static final byte TOUR_MODES = 4; + + public static final byte DRIVEALONE = 1; + public static final byte SHARED2 = 2; + public static final byte SHARED3 = 3; + public static final byte WALK = 4; + + // note that time periods start at 1 and go to 40 + public static final byte TIME_PERIODS = 40; + + /** + * Calculate and return the destination choice size term segment + * + * @param purpose + * @return Right now, just the purpose is returned. + */ + public static int getDCSizeSegment(int purpose) + { + + return purpose; + + } + + /** + * Calculate the purpose from the dc size segment. + * + * @param segment + * The dc size segment (0-17) + * @return The purpose + */ + public static int getPurposeFromDCSizeSegment(int segment) + { + + return segment; + } + + /** + * return the Skim period index 0=am, 1=pm, 2=off-peak + */ + public static int getSkimPeriodIndex(int departPeriod) + { + + int skimPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; + else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; + else skimPeriodIndex = OP; + + return skimPeriodIndex; + + } + + /** + * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV + */ + public static int getModelPeriodIndex(int departPeriod) + { + + int modelPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; + else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; + else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; + else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; + else modelPeriodIndex = 4; + + return modelPeriodIndex; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java new file mode 100644 index 0000000..b72dfc7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java @@ -0,0 +1,405 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class CrossBorderStationDestChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("crossBorderModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected float tourDepartPeriod; + protected float tourArrivePeriod; + protected int purpose; + protected double[][] sizeTerms; // by + // purpose, + // alternative + // (station-taz + // or + // sampled + // station-mgra) + protected double[] stationSizeTerms; // by + // alternative + // (station-taz + // or + // sampled + // station-mgra) + protected double[] correctionFactors; // by + // alternative + // (sampled + // station-mgra + // pair, + // for + // full + // model + // only) + protected double[] tourModeLogsums; // by + // alternative + // (sampled + // station-mgra + // pair, + // for + // full + // model + // only) + protected int[] poeNumbers; // by + // alternative + // (station-taz + // or + // sampled + // station-mgra) + protected int[] originTazs; // by + // alternative + // (station-taz + // or + // sampled + // station-mgra) + protected int[] destinationTazs; // by + // alternative + // (station-taz + // or + // sampled + // station-mgra) + + protected double nmWalkTimeOut; + protected double nmWalkTimeIn; + protected double nmBikeTimeOut; + protected double nmBikeTimeIn; + protected double lsWgtAvgCostM; + protected double lsWgtAvgCostD; + protected double lsWgtAvgCostH; + + public CrossBorderStationDestChoiceDMU(CrossBorderModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Get the POE number for the alternative. + * + * @param alt + * Either station-taz or sampled station-mgra + * @return + */ + public int getPoe(int alt) + { + return poeNumbers[alt]; + } + + /** + * Set the poe number array + * + * @param poeNumbers + * An array of POE numbers, one for each alternative (either + * station-taz or sampled station-mgra) + */ + public void setPoeNumbers(int[] poeNumbers) + { + this.poeNumbers = poeNumbers; + } + + /** + * Get the tour mode choice logsum for the sampled station-mgra pair. + * + * @param alt + * Sampled station-mgra + * @return + */ + public double getTourModeLogsum(int alt) + { + return tourModeLogsums[alt]; + } + + /** + * Set the tour mode choice logsums + * + * @param poeNumbers + * An array of tour mode choice logsums, one for each alternative + * (sampled station-mgra) + */ + public void setTourModeLogsums(double[] logsums) + { + this.tourModeLogsums = logsums; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + /** + * @return the sizeTerms. The size term is the size of the alternative north + * of the border. It is indexed by alternative, where alternative is + * either taz-station pair or mgra-station pair, depending on + * whether the DMU is being used for the SOA model or the actual + * model. + */ + public double getSizeTerm(int alt) + { + return sizeTerms[purpose][alt]; + } + + /** + * @param sizeTerms + * the sizeTerms to set. The size term is the size of the + * alternative north of the border. It is indexed by alternative, + * where alternative is either taz-station pair or mgra-station + * pair, depending on whether the DMU is being used for the SOA + * model or the actual model. + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + /** + * @return the accessibility of the station to population south of the + * border. The size term is indexed by alternative, where + * alternative is either taz-station pair or mgra-station pair, + * depending on whether the DMU is being used for the SOA model or + * the actual model. + */ + public double getStationPopulationAccessibility(int alt) + { + return stationSizeTerms[alt]; + } + + /** + * @param accessibilities + * is the accessibility of the station to population south of the + * border. The size term is indexed by alternative, where + * alternative is either taz-station pair or mgra-station pair, + * depending on whether the DMU is being used for the SOA model + * or the actual model. + */ + public void setStationPopulationAccessibilities(double[] accessibilities) + { + this.stationSizeTerms = accessibilities; + } + + /** + * @return the correctionFactors + */ + public double getCorrectionFactor(int alt) + { + return correctionFactors[alt]; + } + + /** + * @param correctionFactors + * the correctionFactors to set + */ + public void setCorrectionFactors(double[] correctionFactors) + { + this.correctionFactors = correctionFactors; + } + + /** + * @return the origin taz + */ + public int getOriginTaz(int alt) + { + return originTazs[alt]; + } + + /** + * @param originTazs + * The origin tazs to set + */ + public void setOriginTazs(int[] originTazs) + { + this.originTazs = originTazs; + } + + /** + * @return the destination taz + */ + public int getDestinationTaz(int alt) + { + return destinationTazs[alt]; + } + + /** + * @param stopTazs + * The destination tazs to set + */ + public void setDestinationTazs(int[] destinationTazs) + { + this.destinationTazs = destinationTazs; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the purpose + */ + public int getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(int purpose) + { + this.purpose = purpose; + } + + public float getTimeOutbound() + { + return tourDepartPeriod; + } + + public float getTimeInbound() + { + return tourArrivePeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(float tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(float tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getStationPopulationAccessibility", 2); + methodIndexMap.put("getSizeTerm", 3); + methodIndexMap.put("getCorrectionFactor", 4); + methodIndexMap.put("getPoe", 5); + methodIndexMap.put("getPurpose", 6); + methodIndexMap.put("getTourModeLogsum", 7); + methodIndexMap.put("getOriginTaz", 8); + methodIndexMap.put("getDestinationTaz", 9); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 2: + returnValue = getStationPopulationAccessibility(arrayIndex); + break; + case 3: + returnValue = getSizeTerm(arrayIndex); + break; + case 4: + returnValue = getCorrectionFactor(arrayIndex); + break; + case 5: + returnValue = getPoe(arrayIndex); + break; + case 6: + returnValue = getPurpose(); + break; + case 7: + returnValue = getTourModeLogsum(arrayIndex); + break; + case 8: + returnValue = getOriginTaz(arrayIndex); + break; + case 9: + returnValue = getDestinationTaz(arrayIndex); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java new file mode 100644 index 0000000..14a6600 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java @@ -0,0 +1,818 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +/** + * This class is used for both the sample of alternatives and the full + * destination choice model for border crossing tours. + * + * The model first calculates a set of station-level logsums which represent the + * attractiveness of each station based upon the accessibility to Mexico + * populations (taking into account population size of Colonias and distance + * between each station and each Colonia. The model then creates a sample of + * alternatives where each alternative is a pair of border crossing station + * (entry MGRA) and destination MGRA in San Diego County, by tour purpose. This + * is sampled from for each tour, and a mode choice logsum is calculated for + * each station-MGRA pair. The full destination choice model is run on the + * sample with the mode choice logsums influencing station - destination choice, + * and a station-MGRA pair is chosen for each tour. + * + * @author Freedman + * + */ +public class CrossBorderStationDestChoiceModel +{ + + private double[][] mgraSizeTerms; // by purpose, MGRA + private double[][] tazSizeTerms; // by purpose, TAZ + private double[][] tazStationProbabilities; // by purpose, station-TAZ + // alternative + private double[][][] mgraProbabilities; // by purpose, TAZ, MGRA + private double[] stationLogsums; // by entry station, logsum + // from colonia to + // station + private double[] soaStationLogsums; // by station-TAZ + // alternative, station + // logsums + private double[][] soaSizeTerms; // by purpose, station-TAZ + // alternative, + // size terms for tazs + private int[] soaOriginTazs; // by station-TAZ + // alternative, origin Taz + private int[] soaDestinationTazs; // by station-TAZ + // alternative, destination + // Taz + + private int[] sampledDestinationMgra; // destination mgra for each + // of n + // samples + private int[] sampledEntryMgra; // entry mgra for each of n + // samples + private double[][] sampledSizeTerms; // size term for each of n + // samples (1st + // dimension is purpose) + private double[] sampledStationLogsums; // station logsum for each of + // n + // samples + private int[] sampledStations; // POE for each of n samples + private int[] sampledOriginTazs; // Origin Taz for each of n + // samples + private int[] sampledDestinationTazs; // Destination Taz for each + // of n + // samples + + private double[] sampledCorrectionFactors; // correction factor for each + // of + // n samples + private double[] tourModeChoiceLogsums; // mode choice logsum for + // each of n + // samples + + private class KeyClass + { + int station; + int mgra; + + @Override + public boolean equals(Object obj) + { + if (obj instanceof KeyClass) + { + return station == (((KeyClass) obj).station) && mgra == (((KeyClass) obj).mgra); + } + return false; + } + + @Override + public int hashCode() + { + return station * 10000 + mgra; + } + + } + + private KeyClass key; + private HashMap frequencyChosen; // by + // alternative, + // number + // of + // times + // chosen + + private TableDataSet alternativeData; // the + // alternatives, + // with + // the + // following + // fields: + // "EntryMGRA" + // - + // indicating + // border + // crossing + // entry + // MGRA + // "dest" + // - + // indicating + // the + // destination + // TAZ + // in + // San + // Diego + // County + + private int stations; // number + // of + // stations + private int sampleRate; + + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication soaModel; + private ChoiceModelApplication destModel; + private CrossBorderTourModeChoiceModel tourModeChoiceModel; + CrossBorderStationDestChoiceDMU dmu; + McLogsumsCalculator logsumHelper; + + private UtilityExpressionCalculator sizeTermUEC; + private Tracer tracer; + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + private boolean seek; + private HashMap rbMap; + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of cross border model DMUs + */ + public CrossBorderStationDestChoiceModel(HashMap rbMap, + CrossBorderModelStructure myStructure, CrossBorderDmuFactoryIf dmuFactory, + AutoTazSkimsCalculator tazDistanceCalculator) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String crossBorderDCSoaFileName = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.soa.uec.file"); + crossBorderDCSoaFileName = uecFileDirectory + crossBorderDCSoaFileName; + + int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.soa.data.page")); + int soaSizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.soa.size.page")); + int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.soa.model.page")); + + String crossBorderDCFileName = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.uec.file"); + crossBorderDCFileName = uecFileDirectory + crossBorderDCFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.data.page")); + int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.model.page")); + + // read the model pages from the property file, create one choice model + // for each + CrossBorderStationDestChoiceDMU dcDmu = dmuFactory.getCrossBorderStationChoiceDMU(); + + // create a ChoiceModelApplication object for the SOA model. + soaModel = new ChoiceModelApplication(crossBorderDCSoaFileName, soaModelPage, soaDataPage, + rbMap, (VariableTable) dcDmu); + + // create a ChoiceModelApplication object for the full model. + destModel = new ChoiceModelApplication(crossBorderDCFileName, modelPage, dataPage, rbMap, + (VariableTable) dcDmu); + sampleRate = destModel.getAlternativeNames().length; + + // get the alternative data from the model + UtilityExpressionCalculator uec = soaModel.getUEC(); + alternativeData = uec.getAlternativeData(); + + // create a UEC to solve size terms for each MGRA + sizeTermUEC = new UtilityExpressionCalculator(new File(crossBorderDCSoaFileName), + soaSizePage, soaDataPage, rbMap, dmuFactory.getCrossBorderStationChoiceDMU()); + // set up the tracer object + trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String coloniaDistanceFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.colonia.file"); + coloniaDistanceFile = directory + coloniaDistanceFile; + + // calculate logsums for each station (based on Super-Colonia population + // and distance to station) + float distanceParam = new Float(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.dc.colonia.distance.parameter")); + calculateStationLogsum(coloniaDistanceFile, distanceParam); + + // arrays of sampled station-mgra pairs + sampledDestinationMgra = new int[sampleRate + 1]; + sampledEntryMgra = new int[sampleRate + 1]; + sampledCorrectionFactors = new double[sampleRate + 1]; + frequencyChosen = new HashMap(); + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(rbMap); + + // this sets by thread, so do it outside of initialization + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + // set up a tour mode choice model for calculation of tour mode + // probabilities + tourModeChoiceModel = new CrossBorderTourModeChoiceModel(rbMap, myStructure, dmuFactory, + tazDistanceCalculator); + + tourModeChoiceLogsums = new double[sampleRate + 1]; + sampledSizeTerms = new double[myStructure.CROSSBORDER_PURPOSES.length][sampleRate + 1]; + sampledStationLogsums = new double[sampleRate + 1]; + sampledStations = new int[sampleRate + 1]; + + sampledOriginTazs = new int[sampleRate + 1]; + sampledDestinationTazs = new int[sampleRate + 1]; + + + } + + /** + * Calculate the station logsum. Station logsums are based on distance from + * supercolonia to station and population of supercolonia, as follows: + * + * stationLogsum_i = LN [ Sum( exp(distanceParam * distance) * population) ] + * + * supercolonia population and distances are stored in @param fileName. + * Fields in file include: + * + * Population Population of supercolonia Distance_MGRANumber where + * MGRANumber is the number of the MGRA corresponding to the entry station, + * with one field for each possible entry station. + * + * @param fileName + * Name of file containing supercolonia population and distance + * @param distanceParameter + * Parameter for distance. + */ + private void calculateStationLogsum(String fileName, float distanceParameter) + { + + logger.info("Calculating Station Logsum"); + + logger.info("Begin reading the data in file " + fileName); + TableDataSet coloniaTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + coloniaTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + stations = 0; + // iterate through columns in table, calculate number of station + // alternatives (entry stations) + String[] columnLabels = coloniaTable.getColumnLabels(); + for (int i = 0; i < columnLabels.length; ++i) + { + String label = columnLabels[i]; + if (label.contains("Distance_")) + { + ++stations; + } + } + + // iterate through stations and calculate logsum + stationLogsums = new double[stations]; + int colonias = coloniaTable.getRowCount(); + for (int i = 0; i < colonias; ++i) + { + + float population = coloniaTable.getValueAt(i + 1, "Population"); + + for (int j = 0; j < stations; ++j) + { + + float distance = coloniaTable.getValueAt(i + 1, "Distance_poe" + j); + if (population > 0) + stationLogsums[j] += Math.exp(distanceParameter * distance) * population; + } + } + + // take natural log + for (int i = 0; i < stations; ++i) + stationLogsums[i] = Math.log(stationLogsums[i]); + + logger.info("Finished Calculating Border Crossing Station Logsum"); + + } + + /** + * Calculate size terms + */ + public void calculateSizeTerms(CrossBorderDmuFactoryIf dmuFactory) + { + + logger.info("Calculating Cross Border Model MGRA Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int maxTaz = tazManager.getMaxTaz(); + int purposes = sizeTermUEC.getNumberOfAlternatives(); + + mgraSizeTerms = new double[purposes][maxMgra + 1]; + tazSizeTerms = new double[purposes][maxTaz + 1]; + IndexValues iv = new IndexValues(); + CrossBorderStationDestChoiceDMU aDmu = dmuFactory.getCrossBorderStationChoiceDMU(); + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = sizeTermUEC.solve(iv, aDmu, null); + + // store the size terms + for (int purpose = 0; purpose < purposes; ++purpose) + { + + mgraSizeTerms[purpose][mgra] = utilities[purpose]; + tazSizeTerms[purpose][taz] += utilities[purpose]; + } + + // log + if (tracer.isTraceOn() && tracer.isTraceZone(taz)) + { + + logger.info("Size Term calculations for mgra " + mgra); + sizeTermUEC.logResultsArray(logger, 0, mgra); + + } + } + + // now calculate probability of selecting each MGRA within each TAZ for + // SOA + mgraProbabilities = new double[purposes][maxTaz + 1][]; + int[] tazs = tazManager.getTazs(); + + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazs.length; ++taz) + { + int tazNumber = tazs[taz]; + int[] mgraArray = tazManager.getMgraArray(tazNumber); + + // initialize the vector of mgras for this purpose-taz + mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; + + // now calculate the cumulative probability distribution + double lastProb = 0.0; + for (int mgra = 0; mgra < mgraArray.length; ++mgra) + { + + int mgraNumber = mgraArray[mgra]; + if (tazSizeTerms[purpose][tazNumber] > 0.0) + mgraProbabilities[purpose][tazNumber][mgra] = lastProb + + mgraSizeTerms[purpose][mgraNumber] + / tazSizeTerms[purpose][tazNumber]; + lastProb = mgraProbabilities[purpose][tazNumber][mgra]; + } + if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) + logger.info("Error: purpose " + purpose + " taz " + tazNumber + + " cum prob adds up to " + lastProb); + } + + } + + // calculate logged size terms for mgra and taz vectors to be used in + // dmu + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) + if (tazSizeTerms[purpose][taz] > 0.0) + tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); + + for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) + if (mgraSizeTerms[purpose][mgra] > 0.0) + mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); + + } + logger.info("Finished Calculating Cross Border Model MGRA Size Terms"); + } + + /** + * Calculate taz probabilities. This method initializes and calculates the + * tazProbabilities array. + */ + public void calculateTazProbabilities(CrossBorderDmuFactoryIf dmuFactory) + { + + if (tazSizeTerms == null) + { + logger.error("Error: attemping to execute CrossBorderStationDestChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); + throw new RuntimeException(); + } + + logger.info("Calculating Cross Border Model TAZ-Station Probabilities Arrays"); + + // initialize taz probabilities array + int purposes = tazSizeTerms.length; + + // initialize the index for station population accessibility and taz + // size term + int alternatives = soaModel.getNumberOfAlternatives(); + soaStationLogsums = new double[alternatives + 1]; // by station-TAZ + // alternative - + // station logsums + soaSizeTerms = new double[purposes][alternatives + 1]; // by purpose, + // station-TAZ + // alternative - + // size terms + // for tazs + soaOriginTazs = new int[alternatives + 1]; + soaDestinationTazs = new int[alternatives + 1]; + + // iterate through the alternatives in the alternatives file and set the + // size term and station logsum for each alternative + UtilityExpressionCalculator soaModelUEC = soaModel.getUEC(); + TableDataSet altData = soaModelUEC.getAlternativeData(); + + int rowCount = altData.getRowCount(); + for (int row = 1; row <= rowCount; ++row) + { + + int entryMgra = (int) altData.getValueAt(row, "mgra_entry"); + int poe = (int) altData.getValueAt(row, "poe"); + int destinationTaz = (int) altData.getValueAt(row, "dest"); + + soaStationLogsums[row] = stationLogsums[poe]; + + for (int purpose = 0; purpose < purposes; ++purpose) + soaSizeTerms[purpose][row] = tazSizeTerms[purpose][destinationTaz]; + + // set the origin taz + soaOriginTazs[row] = mgraManager.getTaz(entryMgra); + + // set the destination taz + soaDestinationTazs[row] = destinationTaz; + + } + + dmu = dmuFactory.getCrossBorderStationChoiceDMU(); + + // set size terms for each taz + dmu.setSizeTerms(soaSizeTerms); + + // set population accessibility for each station + dmu.setStationPopulationAccessibilities(soaStationLogsums); + + // set the stations for each alternative + int poeField = altData.getColumnPosition("poe"); + int[] poeNumbers = altData.getColumnAsInt(poeField, 1); // return field + // as 1-based + + dmu.setPoeNumbers(poeNumbers); + + // set origin and destination tazs + dmu.setOriginTazs(soaOriginTazs); + dmu.setDestinationTazs(soaDestinationTazs); + + // initialize array to hold taz-station probabilities + tazStationProbabilities = new double[purposes][alternatives + 1]; + + // iterate through purposes, calculate probabilities for each and store + // in array + for (int purpose = 0; purpose < purposes; ++purpose) + { + + dmu.setPurpose(purpose); + + // Calculate utilities & probabilities + soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + // Store probabilities (by purpose) + tazStationProbabilities[purpose] = Arrays.copyOf(soaModel.getCumulativeProbabilities(), + soaModel.getCumulativeProbabilities().length); + } + logger.info("Finished Calculating Cross Border Model TAZ-Station Probabilities Arrays"); + } + + /** + * Choose a Station-MGRA alternative for sampling + * + * @param tour + * CrossBorderTour with purpose and Random + * @return An array of station-mgra pairs + */ + private void chooseStationMgraSample(CrossBorderTour tour) + { + + frequencyChosen.clear(); + + // choose sample, set station logsums and mgra size terms + int purpose = tour.getPurpose(); + for (int sample = 1; sample <= sampleRate; ++sample) + { + + // first find a TAZ and station + int alt = 0; + double[] tazCumProb = tazStationProbabilities[purpose]; + double altProb = 0; + double cumProb = 0; + double random = tour.getRandom(); + for (int i = 0; i < tazCumProb.length; ++i) + { + if (tazCumProb[i] > random) + { + alt = i; + if (i != 0) + { + cumProb = tazCumProb[i - 1]; + altProb = tazCumProb[i] - tazCumProb[i - 1]; + } else + { + altProb = tazCumProb[i]; + } + break; + } + } + + // get the taz number of the alternative, and an array of mgras in + // that taz + int destinationTaz = (int) alternativeData.getValueAt(alt + 1, "dest"); + int poe = (int) alternativeData.getValueAt(alt + 1, "poe"); + int entryMgra = (int) alternativeData.getValueAt(alt + 1, "mgra_entry"); + sampledEntryMgra[sample] = entryMgra; + int[] mgraArray = tazManager.getMgraArray(destinationTaz); + + // set the origin taz + sampledOriginTazs[sample] = (int) alternativeData.getValueAt(alt + 1, "poe_taz"); + + // set the destination taz + sampledDestinationTazs[sample] = destinationTaz; + + // now find an MGRA in the taz corresponding to the random number + // drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives + // probability + int mgraNumber = 0; + double[] mgraCumProb = mgraProbabilities[purpose][destinationTaz]; + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > random && mgraCumProb[i] > 0) + { + mgraNumber = mgraArray[i]; + sampledDestinationMgra[sample] = mgraNumber; + + // for now, store the probability in the correction factors + // array + sampledCorrectionFactors[sample] = mgraCumProb[i] * altProb; + + break; + } + } + + // store frequency chosen + key = new KeyClass(); + key.mgra = mgraNumber; + key.station = entryMgra; + if (!frequencyChosen.containsKey(key)) + { + frequencyChosen.put(key, 1); + } else + { + int freq = frequencyChosen.get(key); + frequencyChosen.put(key, freq + 1); + } + + // set station logsums + sampledStationLogsums[sample] = stationLogsums[poe]; + + // set the size terms for the sample + sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; + + // set the sampled station number + sampledStations[sample] = poe; + + } + // calculate correction factors + for (int sample = 1; sample <= sampleRate; ++sample) + { + key = new KeyClass(); + key.mgra = sampledDestinationMgra[sample]; + key.station = sampledEntryMgra[sample]; + int freq = frequencyChosen.get(key); + sampledCorrectionFactors[sample] = (float) Math.log((double) freq + / sampledCorrectionFactors[sample]); + + } + + } + + /** + * Use the tour mode choice model to calculate the logsum for each sampled + * station-mgra pair and store in the array. + * + * @param tour + * The tour attributes used are tour purpose, depart and arrive + * periods, and sentri availability. + */ + private void calculateLogsumsForSample(CrossBorderTour tour) + { + + for (int sample = 1; sample <= sampleRate; ++sample) + { + + if (sampledEntryMgra[sample] > 0) + { + + int originMgra = sampledEntryMgra[sample]; + int destinationMgra = sampledDestinationMgra[sample]; + + tour.setOriginMGRA(originMgra); + tour.setOriginTAZ(sampledOriginTazs[sample]); + tour.setDestinationMGRA(destinationMgra); + tour.setDestinationTAZ(mgraManager.getTaz(destinationMgra)); + tour.setPoe(sampledStations[sample]); + + double logsum = tourModeChoiceModel.getLogsum(tour, logger, "Sample logsum " + + sample, "tour " + tour.getID()); + tourModeChoiceLogsums[sample] = logsum; + } else tourModeChoiceLogsums[sample] = 0; + + } + + } + + /** + * Choose a station and internal destination MGRA for the tour. + * + * @param tour + * A cross border tour with a tour purpose and departure\arrival + * time and SENTRI availability members. + */ + public void chooseStationAndDestination(CrossBorderTour tour) + { + + chooseStationMgraSample(tour); + calculateLogsumsForSample(tour); + + double random = tour.getRandom(); + dmu.setPurpose(tour.getPurpose()); + + // set size terms for each sampled station-mgra pair corresponding to + // mgra + dmu.setSizeTerms(sampledSizeTerms); + + // set population accessibility for each station-mgra pair corresponding + // to station + dmu.setStationPopulationAccessibilities(sampledStationLogsums); + + // set the sampled stations + dmu.setPoeNumbers(sampledStations); + + // set the correction factors + dmu.setCorrectionFactors(sampledCorrectionFactors); + + // set the tour mode choice logsums + dmu.setTourModeLogsums(tourModeChoiceLogsums); + + // set the origin and destination tazs + dmu.setOriginTazs(sampledOriginTazs); + dmu.setDestinationTazs(sampledDestinationTazs); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Choosing station-destination alternative from sample"); + tour.logTourObject(logger, 1000); + + // log the sample + logSample(); + destModel.choiceModelUtilityTraceLoggerHeading("Station-destination model", "tour " + + tour.getID()); + } + + destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + destModel.logUECResults(logger, "Station-destination model"); + } + int alt = destModel.getChoiceResult(random); + + int entryMgra = sampledEntryMgra[alt]; + int primaryDestination = sampledDestinationMgra[alt]; + int poe = sampledStations[alt]; + int taz = sampledOriginTazs[alt]; + + tour.setOriginMGRA(entryMgra); + tour.setOriginTAZ(taz); + tour.setDestinationMGRA(primaryDestination); + tour.setDestinationTAZ(mgraManager.getTaz(primaryDestination)); + tour.setPoe(poe); + + } + + public CrossBorderTourModeChoiceModel getTourModeChoiceModel() + { + return tourModeChoiceModel; + } + + public void logSample() + { + + logger.info("Sampled station-destination alternatives"); + + logger.info("\nAlt POE EntryMgra DestMgra"); + for (int i = 1; i <= sampleRate; ++i) + { + logger.info(i + " " + sampledStations[i] + " " + sampledEntryMgra[i] + " " + + sampledDestinationMgra[i]); + } + logger.info(""); + } + + /** + * @return the mgraSizeTerms + */ + public double[][] getMgraSizeTerms() + { + return mgraSizeTerms; + } + + /** + * @return the tazSizeTerms + */ + public double[][] getTazSizeTerms() + { + return tazSizeTerms; + } + + /** + * @return the mgraProbabilities + */ + public double[][][] getMgraProbabilities() + { + return mgraProbabilities; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java new file mode 100644 index 0000000..6382a30 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java @@ -0,0 +1,187 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import org.apache.log4j.Logger; + +public class CrossBorderStop + implements Serializable +{ + + private int id; + private int mode; + private int period; + private boolean inbound; + private int mgra; + private int TAZ; + private byte purpose; + + CrossBorderTour parentTour; + + public CrossBorderStop(CrossBorderTour parentTour, int id, boolean inbound) + { + this.parentTour = parentTour; + this.id = id; + this.inbound = inbound; + } + + /** + * @return the mgra + */ + public int getMgra() + { + return mgra; + } + + /** + * @param mgra + * the mgra to set + */ + public void setMgra(int mgra) + { + this.mgra = mgra; + } + + public int getTAZ() + { + return TAZ; + } + + public void setTAZ(int tAZ) + { + TAZ = tAZ; + } + + public void setMode(int mode) + { + this.mode = mode; + } + + public void setPeriod(int period) + { + this.period = period; + } + + /** + * @return the id + */ + public int getId() + { + return id; + } + + /** + * @param id + * the id to set + */ + public void setId(int id) + { + this.id = id; + } + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the parentTour + */ + public CrossBorderTour getParentTour() + { + return parentTour; + } + + /** + * @param parentTour + * the parentTour to set + */ + public void setParentTour(CrossBorderTour parentTour) + { + this.parentTour = parentTour; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(byte stopPurposeIndex) + { + this.purpose = stopPurposeIndex; + } + + public byte getPurpose() + { + return purpose; + } + + public int getMode() + { + return mode; + } + + public int getStopPeriod() + { + return period; + } + + public CrossBorderTour getTour() + { + return parentTour; + } + + public int getStopId() + { + return id; + } + + public void logStopObject(Logger logger, int totalChars) + { + + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + String purposeString = CrossBorderModelStructure.CROSSBORDER_PURPOSES[purpose]; + logHelper(logger, "stopId: ", id, totalChars); + logHelper(logger, "mgra: ", mgra, totalChars); + logHelper(logger, "mode: ", mode, totalChars); + logHelper(logger, "purpose: ", purposeString, totalChars); + logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); + logHelper(logger, inbound ? "outbound departPeriod: " : "inbound arrivePeriod: ", period, + totalChars); + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public static void logHelper(Logger logger, String label, int value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } + + public static void logHelper(Logger logger, String label, String value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java new file mode 100644 index 0000000..b546780 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java @@ -0,0 +1,316 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the stop frequency model for cross border tours. It is + * currently based on a static probability distribution stored in an input file, + * and indexed into by tour purpose and duration. + * + * @author Freedman + * + */ +public class CrossBorderStopFrequencyModel +{ + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private double[][] cumProbability; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + private int[][] lowerBoundDurationHours; // by + // purpose, + // alternative: + // lower + // bound + // in + // hours + private int[][] upperBoundDurationHours; // by + // purpose, + // alternative: + // upper + // bound + // in + // hours + private int[][] outboundStops; // by + // purpose, + // alternative: + // number + // of + // outbound + // stops + private int[][] inboundStops; // by + // purpose, + // alternative: + // number + // of + // inbound + // stops + CrossBorderModelStructure modelStructure; + + /** + * Constructor. + */ + public CrossBorderStopFrequencyModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.stop.frequency.file"); + stopFrequencyFile = directory + stopFrequencyFile; + + modelStructure = new CrossBorderModelStructure(); + + readStopFrequencyFile(stopFrequencyFile); + + } + + /** + * Read the stop frequency distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readStopFrequencyFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating stop frequency probability distribution"); + + int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 + + int[] alts = new int[purposes]; + + // take a pass through the data and see how many alternatives there are + // for each purpose + int rowCount = probabilityTable.getRowCount(); + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + ++alts[purpose]; + } + + // initialize all the arrays + cumProbability = new double[purposes][]; + lowerBoundDurationHours = new int[purposes][]; + upperBoundDurationHours = new int[purposes][]; + outboundStops = new int[purposes][]; + inboundStops = new int[purposes][]; + + for (int i = 0; i < purposes; ++i) + { + cumProbability[i] = new double[alts[i]]; + lowerBoundDurationHours[i] = new int[alts[i]]; + upperBoundDurationHours[i] = new int[alts[i]]; + outboundStops[i] = new int[alts[i]]; + inboundStops[i] = new int[alts[i]]; + } + + // fill up arrays + int lastPurpose = 0; + int lastLowerBound = 0; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + int lowerBound = (int) probabilityTable.getValueAt(row, "DurationLo"); + int upperBound = (int) probabilityTable.getValueAt(row, "DurationHi"); + int outStops = (int) probabilityTable.getValueAt(row, "Outbound"); + int inbStops = (int) probabilityTable.getValueAt(row, "Inbound"); + + // reset cumulative probability if new purpose or lower-bound + if (purpose != lastPurpose || lowerBound != lastLowerBound) + { + + // log cumulative probability just in case + /* + logger.info("Cumulative probability for purpose " + purpose + " lower bound " + + lowerBound + " is " + cumProb); + */ + cumProb = 0; + } + + if (purpose != lastPurpose) alt = 0; + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + lowerBoundDurationHours[purpose][alt] = lowerBound; + upperBoundDurationHours[purpose][alt] = upperBound; + outboundStops[purpose][alt] = outStops; + inboundStops[purpose][alt] = inbStops; + + ++alt; + + lastPurpose = purpose; + lastLowerBound = lowerBound; + } + + logger.info("End calculating stop frequency probability distribution"); +/* + for (int purp = 0; purp < purposes; ++purp) + { + for (int a = 0; a < cumProbability[purp].length; ++a) + { + logger.info("Purpose " + purp + " lower " + lowerBoundDurationHours[purp][a] + + " upper " + upperBoundDurationHours[purp][a] + " cumProb " + + cumProbability[purp][a]); + } + } +*/ + } + + /** + * Calculate tour time of day for the tour. + * + * @param tour + * A cross border tour (with purpose) + */ + public void calculateStopFrequency(CrossBorderTour tour) + { + + int purpose = tour.getPurpose(); + double random = tour.getRandom(); + + if (tour.getDebugChoiceModels()) + { + logger.info("Choosing stop frequency for purpose " + + modelStructure.CROSSBORDER_PURPOSES[purpose] + " using random number " + + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + if (!tourIsInRange(tour, lowerBoundDurationHours[purpose][i], + upperBoundDurationHours[purpose][i])) continue; + + if (random < cumProbability[purpose][i]) + { + int outStops = outboundStops[purpose][i]; + int inbStops = inboundStops[purpose][i]; + + if (outStops > 0) + { + CrossBorderStop[] stops = generateOutboundStops(tour, outStops); + tour.setOutboundStops(stops); + } + + if (inbStops > 0) + { + CrossBorderStop[] stops = generateInboundStops(tour, inbStops); + tour.setInboundStops(stops); + } + if (tour.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose " + outStops + " outbound stops and " + inbStops + + " inbound stops"); + logger.info(""); + } + break; + } + } + + } + + /** + * Check if the tour duration is in range + * + * @param tour + * @param lowerBound + * @param upperBound + * @return True if tour duration is greater than or equal to lower and + */ + private boolean tourIsInRange(CrossBorderTour tour, int lowerBound, int upperBound) + { + + float depart = (float) tour.getDepartTime(); + float arrive = (float) tour.getArriveTime(); + + float halfHours = arrive + 1 - depart; // at least 30 minutes + float tourDurationInHours = halfHours * (float) 0.5; + + if (tourDurationInHours >= lowerBound && tourDurationInHours <= upperBound) return true; + + return false; + } + + /** + * Generate an array of outbound stops, from tour origin to primary + * destination, in order. + * + * @param tour + * The parent tour. + * @param numberOfStops + * Number of stops from stop frequency model. + * @return The array of outbound stops. + */ + private CrossBorderStop[] generateOutboundStops(CrossBorderTour tour, int numberOfStops) + { + + CrossBorderStop[] stops = new CrossBorderStop[numberOfStops]; + + for (int i = 0; i < stops.length; ++i) + { + CrossBorderStop stop = new CrossBorderStop(tour, i, false); + stops[i] = stop; + stop.setInbound(false); + stop.setParentTour(tour); + } + + return stops; + } + + /** + * Generate an array of inbound stops, from primary dest back to tour + * origin, in order. + * + * @param tour + * Parent tour. + * @param numberOfStops + * Number of stops from stop frequency model. + * @return The array of inbound stops. + */ + private CrossBorderStop[] generateInboundStops(CrossBorderTour tour, int numberOfStops) + { + + CrossBorderStop[] stops = new CrossBorderStop[numberOfStops]; + + for (int i = 0; i < stops.length; ++i) + { + CrossBorderStop stop = new CrossBorderStop(tour, i, true); + stops[i] = stop; + stop.setInbound(true); + stop.setParentTour(tour); + + } + + return stops; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java new file mode 100644 index 0000000..1307d94 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java @@ -0,0 +1,406 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class CrossBorderStopLocationChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("crossBorderModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected int purpose; + protected int stopsOnHalfTour; + protected int stopNumber; + protected int inboundStop; + protected int tourDuration; + + protected double[][] sizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgra) + protected double[] correctionFactors; // by + // alternative + // (sampled + // mgra, + // for + // full + // model + // only) + + protected int[] sampleNumber; // by + // alternative + // (taz + // or + // sampled + // mgra) + + protected double[] osMcLogsumAlt; + protected double[] sdMcLogsumAlt; + + protected double[] tourOrigToStopDistanceAlt; + protected double[] stopToTourDestDistanceAlt; + + public CrossBorderStopLocationChoiceDMU(CrossBorderModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + /** + * @return the stopsOnHalfTour + */ + public int getStopsOnHalfTour() + { + return stopsOnHalfTour; + } + + /** + * @param stopsOnHalfTour + * the stopsOnHalfTour to set + */ + public void setStopsOnHalfTour(int stopsOnHalfTour) + { + this.stopsOnHalfTour = stopsOnHalfTour; + } + + /** + * @return the stopNumber + */ + public int getStopNumber() + { + return stopNumber; + } + + /** + * @param stopNumber + * the stopNumber to set + */ + public void setStopNumber(int stopNumber) + { + this.stopNumber = stopNumber; + } + + /** + * @return the inboundStop + */ + public int getInboundStop() + { + return inboundStop; + } + + /** + * @param inboundStop + * the inboundStop to set + */ + public void setInboundStop(int inboundStop) + { + this.inboundStop = inboundStop; + } + + /** + * @return the tourDuration + */ + public int getTourDuration() + { + return tourDuration; + } + + /** + * @param tourDuration + * the tourDuration to set + */ + public void setTourDuration(int tourDuration) + { + this.tourDuration = tourDuration; + } + + /** + * @return the sampleNumber + */ + public int getSampleNumber(int alt) + { + return sampleNumber[alt]; + } + + /** + * @param sampleNumber + * the sampleNumber to set + */ + public void setSampleNumber(int[] sampleNumber) + { + this.sampleNumber = sampleNumber; + } + + /** + * @return the osMcLogsumAlt + */ + public double getOsMcLogsumAlt(int alt) + { + return osMcLogsumAlt[alt]; + } + + /** + * @param osMcLogsumAlt + * the osMcLogsumAlt to set + */ + public void setOsMcLogsumAlt(double[] osMcLogsumAlt) + { + this.osMcLogsumAlt = osMcLogsumAlt; + } + + /** + * @return the sdMcLogsumAlt + */ + public double getSdMcLogsumAlt(int alt) + { + return sdMcLogsumAlt[alt]; + } + + /** + * @param sdMcLogsumAlt + * the sdMcLogsumAlt to set + */ + public void setSdMcLogsumAlt(double[] sdMcLogsumAlt) + { + this.sdMcLogsumAlt = sdMcLogsumAlt; + } + + /** + * @return the tourOrigToStopDistanceAlt + */ + public double getTourOrigToStopDistanceAlt(int alt) + { + return tourOrigToStopDistanceAlt[alt]; + } + + /** + * @param tourOrigToStopDistanceAlt + * the tourOrigToStopDistanceAlt to set + */ + public void setTourOrigToStopDistanceAlt(double[] tourOrigToStopDistanceAlt) + { + this.tourOrigToStopDistanceAlt = tourOrigToStopDistanceAlt; + } + + /** + * @return the stopToTourDestDistanceAlt + */ + public double getStopToTourDestDistanceAlt(int alt) + { + return stopToTourDestDistanceAlt[alt]; + } + + /** + * @param stopToTourDestDistanceAlt + * the stopToTourDestDistanceAlt to set + */ + public void setStopToTourDestDistanceAlt(double[] stopToTourDestDistanceAlt) + { + this.stopToTourDestDistanceAlt = stopToTourDestDistanceAlt; + } + + /** + * @return the sizeTerms. The size term is the size of the alternative north + * of the border. It is indexed by alternative, where alternative is + * either taz-station pair or mgra-station pair, depending on + * whether the DMU is being used for the SOA model or the actual + * model. + */ + public double getSizeTerm(int alt) + { + return sizeTerms[purpose][alt]; + } + + /** + * @param sizeTerms + * the sizeTerms to set. The size term is the size of the + * alternative north of the border. It is indexed by alternative, + * where alternative is either taz-station pair or mgra-station + * pair, depending on whether the DMU is being used for the SOA + * model or the actual model. + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + /** + * @return the correctionFactors + */ + public double getCorrectionFactor(int alt) + { + return correctionFactors[alt]; + } + + /** + * @param correctionFactors + * the correctionFactors to set + */ + public void setCorrectionFactors(double[] correctionFactors) + { + this.correctionFactors = correctionFactors; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the purpose + */ + public int getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(int purpose) + { + this.purpose = purpose; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + methodIndexMap.put("getPurpose", 0); + methodIndexMap.put("getStopsOnHalfTour", 1); + methodIndexMap.put("getStopNumber", 2); + methodIndexMap.put("getInboundStop", 3); + methodIndexMap.put("getTourDuration", 4); + + methodIndexMap.put("getSizeTerm", 5); + methodIndexMap.put("getCorrectionFactor", 6); + methodIndexMap.put("getSampleNumber", 7); + methodIndexMap.put("getOsMcLogsumAlt", 8); + methodIndexMap.put("getSdMcLogsumAlt", 9); + methodIndexMap.put("getTourOrigToStopDistanceAlt", 10); + methodIndexMap.put("getStopToTourDestDistanceAlt", 11); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getPurpose(); + break; + case 1: + returnValue = getStopsOnHalfTour(); + break; + case 2: + returnValue = getStopNumber(); + break; + case 3: + returnValue = getInboundStop(); + break; + case 4: + returnValue = getTourDuration(); + break; + case 5: + returnValue = getSizeTerm(arrayIndex); + break; + case 6: + returnValue = getCorrectionFactor(arrayIndex); + break; + case 7: + returnValue = getSampleNumber(arrayIndex); + break; + case 8: + returnValue = getOsMcLogsumAlt(arrayIndex); + break; + case 9: + returnValue = getSdMcLogsumAlt(arrayIndex); + break; + case 10: + returnValue = getTourOrigToStopDistanceAlt(arrayIndex); + break; + case 11: + returnValue = getStopToTourDestDistanceAlt(arrayIndex); + break; + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java new file mode 100644 index 0000000..4394466 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java @@ -0,0 +1,536 @@ +package org.sandag.abm.crossborder; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.ConcreteAlternative; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class CrossBorderStopLocationChoiceModel +{ + + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + // private McLogsumsCalculator logsumHelper; + private CrossBorderModelStructure modelStructure; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private CrossBorderStopLocationChoiceDMU dmu; + private CrossBorderTripModeChoiceModel tripModeChoiceModel; + double logsum = 0; + private ChoiceModelApplication soaModel; + private ChoiceModelApplication destModel; + private McLogsumsCalculator logsumHelper; + + + // the following arrays are calculated in the station-destination choice + // model and passed in the constructor. + private double[][] mgraSizeTerms; // by + // purpose, + // MGRA + private double[][][] mgraProbabilities; // by + // purpose, + // TAZ, + // MGRA + + private TableDataSet alternativeData; // the + // alternatives, + // with + // a + // "dest" + // - + // indicating + // the + // destination + // TAZ + // in + // San + // Diego + // County + + // following are used for each taz alternative + private double[] soaTourOrigToStopDistanceAlt; // by + // TAZ + private double[] soaStopToTourDestDistanceAlt; // by + // TAZ + private double[][] tazSizeTerms; // by + // purpose, + // TAZ + // - + // set + // by + // constructor + + // following are used for sampled mgras + private int sampleRate; + private double[][] sampledSizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgra) + private double[] correctionFactors; // by + // alternative + // (sampled + // mgra, + // for + // full + // model + // only) + private int[] sampledTazs; // by + // alternative + // (sampled + // taz) + private int[] sampledMgras; // by + // alternative(sampled + // mgra) + private double[] tourOrigToStopDistanceAlt; + private double[] stopToTourDestDistanceAlt; + private double[] osMcLogsumAlt; + private double[] sdMcLogsumAlt; + + HashMap frequencyChosen; + + private CrossBorderTrip trip; + + private int originMgra; // the + // origin + // MGRA + // of + // the + // stop + // (originMgra + // -> + // stopMgra + // -> + // destinationMgra) + private int destinationMgra; // the + // destination + // MGRA + // of + // the + // stop + // (originMgra + // -> + // stopMgra + // -> + // destinationMgra) + private int originTAZ; + private int destinationTAZ; + private AutoTazSkimsCalculator tazDistanceCalculator; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public CrossBorderStopLocationChoiceModel(HashMap propertyMap, + CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + modelStructure = myModelStructure; + + this.tazDistanceCalculator = tazDistanceCalculator; + setupStopLocationChoiceModel(propertyMap, dmuFactory); + + frequencyChosen = new HashMap(); + + trip = new CrossBorderTrip(); + + } + + /** + * Read the UEC file and set up the stop destination choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupStopLocationChoiceModel(HashMap rbMap, + CrossBorderDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up cross border stop location choice model.")); + + dmu = dmuFactory.getCrossBorderStopLocationChoiceDMU(); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String crossBorderStopLocationSoaFileName = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.soa.uec.file"); + crossBorderStopLocationSoaFileName = uecFileDirectory + crossBorderStopLocationSoaFileName; + + int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.soa.data.page")); + int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.soa.model.page")); + + String crossBorderStopLocationFileName = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.uec.file"); + crossBorderStopLocationFileName = uecFileDirectory + crossBorderStopLocationFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.data.page")); + int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.slc.model.page")); + + // create a ChoiceModelApplication object for the SOA model. + soaModel = new ChoiceModelApplication(crossBorderStopLocationSoaFileName, soaModelPage, + soaDataPage, rbMap, (VariableTable) dmu); + + // create a ChoiceModelApplication object for the full model. + destModel = new ChoiceModelApplication(crossBorderStopLocationFileName, modelPage, + dataPage, rbMap, (VariableTable) dmu); + sampleRate = destModel.getAlternativeNames().length; + + // get the alternative data + UtilityExpressionCalculator uec = soaModel.getUEC(); + alternativeData = uec.getAlternativeData(); + int purposes = modelStructure.CROSSBORDER_PURPOSES.length; + + sampledSizeTerms = new double[purposes][sampleRate + 1]; // by purpose, + // alternative + // (taz or + // sampled + // mgra) + correctionFactors = new double[sampleRate + 1]; // by alternative + // (sampled mgra, for + // full model only) + sampledTazs = new int[sampleRate + 1]; // by alternative (sampled taz) + sampledMgras = new int[sampleRate + 1]; // by alternative (sampled mgra) + tourOrigToStopDistanceAlt = new double[sampleRate + 1]; + stopToTourDestDistanceAlt = new double[sampleRate + 1]; + osMcLogsumAlt = new double[sampleRate + 1]; + sdMcLogsumAlt = new double[sampleRate + 1]; + + tripModeChoiceModel = new CrossBorderTripModeChoiceModel(rbMap, modelStructure, + dmuFactory, tazDistanceCalculator); + logsumHelper = tripModeChoiceModel.getMcLogsumsCalculator(); + + + } + + /** + * Create a sample for the tour and stop. + * + * @param tour + * @param stop + */ + private void createSample(CrossBorderTour tour, CrossBorderStop stop) + { + + int purpose = tour.getPurpose(); + int period = modelStructure.AM; + + dmu.setPurpose(purpose); + boolean inbound = stop.isInbound(); + if (inbound) + { + dmu.setInboundStop(1); + dmu.setStopsOnHalfTour(tour.getNumberInboundStops()); + + // destination for inbound stops is always tour origin + destinationMgra = tour.getOriginMGRA(); + destinationTAZ = mgraManager.getTaz(destinationMgra); + + // origin for inbound stops is tour destination if first stop, or + // last chosen stop location + if (stop.getId() == 0) + { + originMgra = tour.getDestinationMGRA(); + originTAZ = mgraManager.getTaz(originMgra); + } else + { + CrossBorderStop[] stops = tour.getInboundStops(); + originMgra = stops[stop.getId() - 1].getMgra(); + originTAZ = mgraManager.getTaz(originMgra); + } + + } else + { + dmu.setInboundStop(0); + dmu.setStopsOnHalfTour(tour.getNumberOutboundStops()); + + // destination for outbound stops is always tour destination + destinationMgra = tour.getDestinationMGRA(); + destinationTAZ = mgraManager.getTaz(destinationMgra); + + // origin for outbound stops is tour origin if first stop, or last + // chosen stop location + if (stop.getId() == 0) + { + originMgra = tour.getOriginMGRA(); + originTAZ = mgraManager.getTaz(originMgra); + } else + { + CrossBorderStop[] stops = tour.getOutboundStops(); + originMgra = stops[stop.getId() - 1].getMgra(); + originTAZ = mgraManager.getTaz(originMgra); + } + } + dmu.setStopNumber(stop.getId() + 1); + dmu.setDmuIndexValues(originTAZ, originTAZ, originTAZ, 0, false); + + // distances + soaTourOrigToStopDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceFromTaz( + originTAZ, period); + soaStopToTourDestDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceToTaz( + destinationTAZ, period); + dmu.setTourOrigToStopDistanceAlt(soaTourOrigToStopDistanceAlt); + dmu.setStopToTourDestDistanceAlt(soaStopToTourDestDistanceAlt); + + dmu.setSizeTerms(tazSizeTerms); + + // solve for each sample + frequencyChosen.clear(); + for (int sample = 1; sample <= sampleRate; ++sample) + { + + // solve the UEC + soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + // choose a TAZ + double random = tour.getRandom(); + ConcreteAlternative[] alts = soaModel.getAlternatives(); + double cumProb = 0; + double altProb = 0; + int sampledTaz = -1; + for (int i = 0; i < alts.length; ++i) + { + cumProb += alts[i].getProbability(); + if (random < cumProb) + { + sampledTaz = (int) alternativeData.getValueAt(i + 1, "dest"); + altProb = alts[i].getProbability(); + break; + } + } + + // set the sampled taz in the array + sampledTazs[sample] = sampledTaz; + + // now find an MGRA in the taz corresponding to the random number + // drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives + // probability + int[] mgraArray = tazManager.getMgraArray(sampledTaz); + int mgraNumber = 0; + double[] mgraCumProb = mgraProbabilities[purpose][sampledTaz]; + + if (mgraCumProb == null) + { + logger.error("Error: mgraCumProb array is null for purpose " + purpose + + " sampledTaz " + sampledTaz + " hhID " + tour.getID()); + throw new RuntimeException(); + } + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > random && mgraCumProb[i] > 0) + { + mgraNumber = mgraArray[i]; + sampledMgras[sample] = mgraNumber; + + // for now, store the probability in the correction factors + // array + correctionFactors[sample] = mgraCumProb[i] * altProb; + + break; + } + } + + // store frequency chosen + if (!frequencyChosen.containsKey(mgraNumber)) + { + frequencyChosen.put(mgraNumber, 1); + } else + { + int freq = frequencyChosen.get(mgraNumber); + frequencyChosen.put(mgraNumber, freq + 1); + } + + // set the size terms for the sample + sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; + + // set the distances for the sample + tourOrigToStopDistanceAlt[sample] = soaTourOrigToStopDistanceAlt[sampledTaz]; + stopToTourDestDistanceAlt[sample] = soaStopToTourDestDistanceAlt[sampledTaz]; + + } + // calculate correction factors + for (int sample = 1; sample <= sampleRate; ++sample) + { + int mgra = sampledMgras[sample]; + int freq = frequencyChosen.get(mgra); + correctionFactors[sample] = (float) Math.log((double) freq / correctionFactors[sample]); + + } + + } + + /** + * Choose a stop location from the sample. + * + * @param tour + * The cross border tour. + * @param stop + * The cross border stop. + */ + public void chooseStopLocation(CrossBorderTour tour, CrossBorderStop stop) + { + + // create a sample of mgras and set all of the dmu properties + createSample(tour, stop); + dmu.setCorrectionFactors(correctionFactors); + dmu.setSizeTerms(sampledSizeTerms); + dmu.setTourOrigToStopDistanceAlt(stopToTourDestDistanceAlt); + dmu.setStopToTourDestDistanceAlt(stopToTourDestDistanceAlt); + dmu.setSampleNumber(sampledMgras); + + // calculate trip mode choice logsums to and from stop + for (int i = 1; i <= sampleRate; ++i) + { + + // to stop (originMgra -> stopMgra ) + trip.initializeFromStop(tour, stop, true); + trip.setOriginMgra(originMgra); + trip.setOriginTAZ(originTAZ); + trip.setDestinationMgra(sampledMgras[i]); + trip.setDestinationTAZ(mgraManager.getTaz(sampledMgras[i])); + double logsum = tripModeChoiceModel.computeUtilities(tour, trip); + osMcLogsumAlt[i] = logsum; + + // from stop (stopMgra -> destinationMgra) + trip.initializeFromStop(tour, stop, true); + trip.setOriginMgra(sampledMgras[i]); + trip.setOriginTAZ(mgraManager.getTaz(sampledMgras[i])); + trip.setDestinationMgra(destinationMgra); + trip.setDestinationTAZ(destinationTAZ); + logsum = tripModeChoiceModel.computeUtilities(tour, trip); + sdMcLogsumAlt[i] = logsum; + + } + dmu.setOsMcLogsumAlt(osMcLogsumAlt); + dmu.setSdMcLogsumAlt(sdMcLogsumAlt); + + // log headers to traceLogger + if (tour.getDebugChoiceModels()) + { + String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() + + " purpose " + modelStructure.CROSSBORDER_PURPOSES[stop.getPurpose()]; + destModel.choiceModelUtilityTraceLoggerHeading( + "Intermediate stop location choice model", decisionMakerLabel); + } + + destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + double random = tour.getRandom(); + int alt = destModel.getChoiceResult(random); + int destMgra = sampledMgras[alt]; + stop.setMgra(destMgra); + stop.setTAZ(mgraManager.getTaz(destMgra)); + + // write UEC calculation results and choice + if (tour.getDebugChoiceModels()) + { + String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() + + " purpose " + modelStructure.CROSSBORDER_PURPOSES[stop.getPurpose()]; + String loggingHeader = String.format("%s %s", + "Intermediate stop location choice model", decisionMakerLabel); + destModel.logUECResults(logger, loggingHeader); + logger.info("Chose alternative " + alt + " mgra " + destMgra + " with random number " + + random); + logger.info(""); + logger.info(""); + } + + } + + /** + * @return the mgraSizeTerms + */ + public double[][] getMgraSizeTerms() + { + return mgraSizeTerms; + } + + /** + * @return the mgraProbabilities + */ + public double[][][] getMgraProbabilities() + { + return mgraProbabilities; + } + + /** + * @return the tazSizeTerms + */ + public double[][] getTazSizeTerms() + { + return tazSizeTerms; + } + + /** + * Set mgra size terms: must call before choosing location. + * + * @param mgraSizeTerms + */ + public void setMgraSizeTerms(double[][] mgraSizeTerms) + { + + if (mgraSizeTerms == null) + { + logger.error("Error attempting to set MGRASizeTerms in CrossBorderStopLocationChoiceModel: MGRASizeTerms are null"); + throw new RuntimeException(); + } + this.mgraSizeTerms = mgraSizeTerms; + } + + /** + * Set taz size terms: must call before choosing location. + * + * @param tazSizeTerms + */ + public void setTazSizeTerms(double[][] tazSizeTerms) + { + if (tazSizeTerms == null) + { + logger.error("Error attempting to set TazSizeTerms in CrossBorderStopLocationChoiceModel: TazSizeTerms are null"); + throw new RuntimeException(); + } + this.tazSizeTerms = tazSizeTerms; + } + + /** + * Set the mgra probabilities. Must call before choosing location. + * + * @param mgraProbabilities + */ + public void setMgraProbabilities(double[][][] mgraProbabilities) + { + if (mgraProbabilities == null) + { + logger.error("Error attempting to set mgraProbabilities in CrossBorderStopLocationChoiceModel: mgraProbabilities are null"); + throw new RuntimeException(); + } + this.mgraProbabilities = mgraProbabilities; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java new file mode 100644 index 0000000..a2cdfb5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java @@ -0,0 +1,233 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the stop purpose choice model for cross border tours. It is + * currently based on a static probability distribution stored in an input file, + * and indexed into by purpose, tour leg direction (inbound or outbound), the + * stop number, and whether there is just one or multiple stops on the tour leg. + * + * @author Freedman + * + */ +public class CrossBorderStopPurposeModel +{ + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private double[][] cumProbability; // by + // alternative, + // stop + // purpose: + // cumulative + // probability + // distribution + CrossBorderModelStructure modelStructure; + + HashMap arrayElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + // tour + // purpose, + // tour + // leg + // direction, + // stop + // number, + // and + // stop + // complexity. + + /** + * Constructor. + */ + public CrossBorderStopPurposeModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.stop.purpose.file"); + stopFrequencyFile = directory + stopFrequencyFile; + + modelStructure = new CrossBorderModelStructure(); + + arrayElementMap = new HashMap(); + readStopPurposeFile(stopFrequencyFile); + + } + + /** + * Read the stop frequency distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readStopPurposeFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating stop purpose probability distribution"); + + // take a pass through the data and see how many alternatives there are + // for each purpose + int rowCount = probabilityTable.getRowCount(); + int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 + + cumProbability = new double[rowCount][purposes]; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "TourPurp"); + + int inbound = (int) probabilityTable.getValueAt(row, "Inbound"); + int stopNumber = (int) probabilityTable.getValueAt(row, "StopNum"); + int multiple = (int) probabilityTable.getValueAt(row, "Multiple"); + + // store cumulative probabilities + float cumProb = 0; + for (int p = 0; p < purposes; ++p) + { + String label = "StopPurp" + p; + cumProb += probabilityTable.getValueAt(row, label); + cumProbability[row - 1][p] += cumProb; + } + + if (Math.abs(cumProb - 1.0) > 0.00001) + logger.info("Cumulative probability for tour purpose " + purpose + " inbound " + + inbound + " stopNumber " + stopNumber + " multiple " + multiple + " is " + + cumProb); + + int key = getKey(purpose, inbound, stopNumber, multiple); + arrayElementMap.put(key, row - 1); + + } + + logger.info("End calculating stop purpose probability distribution"); + + } + + /** + * Get the key for the arrayElementMap. + * + * @param tourPurp + * Tour purpose + * @param isInbound + * 1 if the stop is on the inbound direction, else 0. + * @param stopNumber + * The number of the stop. + * @param multipleStopsOnLeg + * 1 if multiple stops on leg, else 0. + * @return arrayElementMap key. + */ + private int getKey(int tourPurp, int isInbound, int stopNumber, int multipleStopsOnLeg) + { + + return tourPurp * 1000 + isInbound * 100 + stopNumber * 10 + multipleStopsOnLeg; + } + + /** + * Calculate purposes all stops on the tour + * + * @param tour + * A cross border tour (with tour purpose) + */ + public void calculateStopPurposes(CrossBorderTour tour) + { + + // outbound stops first + if (tour.getNumberOutboundStops() != 0) + { + + int tourPurp = tour.getPurpose(); + CrossBorderStop[] stops = tour.getOutboundStops(); + int multiple = 0; + if (stops.length > 1) multiple = 1; + + // iterate through stop list and calculate purpose for each + for (int i = 0; i < stops.length; ++i) + { + int key = getKey(tourPurp, 0, i + 1, multiple); + int element = arrayElementMap.get(key); + double[] cumProb = cumProbability[element]; + double rand = tour.getRandom(); + int purpose = chooseFromDistribution(rand, cumProb); + stops[i].setPurpose((byte) purpose); + } + } + // inbound stops last + if (tour.getNumberInboundStops() != 0) + { + + int tourPurp = tour.getPurpose(); + CrossBorderStop[] stops = tour.getInboundStops(); + int multiple = 0; + if (stops.length > 1) multiple = 1; + + // iterate through stop list and calculate purpose for each + for (int i = 0; i < stops.length; ++i) + { + int key = getKey(tourPurp, 1, i + 1, multiple); + int element = arrayElementMap.get(key); + double[] cumProb = cumProbability[element]; + double rand = tour.getRandom(); + int purpose = chooseFromDistribution(rand, cumProb); + stops[i].setPurpose((byte) purpose); + } + } + } + + /** + * Choose purpose from the cumulative probability distribution + * + * @param random + * Uniformly distributed random number + * @param cumProb + * Cumulative probability distribution + * @return Stop purpose (0 init). + */ + private int chooseFromDistribution(double random, double[] cumProb) + { + + int choice = -1; + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + choice = i; + break; + } + + } + return choice; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java new file mode 100644 index 0000000..5e3d4a1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java @@ -0,0 +1,365 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the TOD choice model for cross border tours. It is currently + * based on a static probability distribution stored in an input file, and + * indexed into by purpose. + * + * @author Freedman + * + */ +public class CrossBorderStopTimeOfDayChoiceModel +{ + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private double[][] outboundCumProbability; // by + // alternative: + // outbound + // cumulative + // probability + // distribution + private int[] outboundOffsets; // by + // alternative: + // offsets + // for + // outbound + // stop + // duration + // choice + + private double[][] inboundCumProbability; // by + // alternative: + // inbound + // cumulative + // probability + // distribution + private int[] inboundOffsets; // by + // alternative: + // offsets + // for + // inbound + // stop + // duration + // choice + private CrossBorderModelStructure modelStructure; + + private HashMap outboundElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + // tour duration and stop number. + + private HashMap inboundElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + + // tour duration and stop number. + + /** + * Constructor. + */ + public CrossBorderStopTimeOfDayChoiceModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String outboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.stop.outbound.duration.file"); + String inboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.stop.inbound.duration.file"); + + outboundDurationFile = directory + outboundDurationFile; + inboundDurationFile = directory + inboundDurationFile; + + modelStructure = new CrossBorderModelStructure(); + + outboundElementMap = new HashMap(); + readOutboundFile(outboundDurationFile); + + inboundElementMap = new HashMap(); + readInboundFile(inboundDurationFile); + } + + /** + * Read the outbound stop duration file and store the cumulative probability + * distribution as well as the offsets and set the key map to index into the + * probability array. + * + * @param fileName + */ + public void readOutboundFile(String fileName) + { + TableDataSet outboundTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + outboundTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + int columns = outboundTable.getColumnCount(); + int rows = outboundTable.getRowCount(); + outboundCumProbability = new double[rows][columns - 3]; + + // first three columns are index fields, rest are offsets + outboundOffsets = new int[columns - 3]; + for (int i = 4; i <= columns; ++i) + { + String offset = outboundTable.getColumnLabel(i); + outboundOffsets[i - 4] = new Integer(offset); + } + + // now fill in cumulative probability array + for (int row = 1; row <= rows; ++row) + { + + int lowerBound = (int) outboundTable.getValueAt(row, "RemainingLow"); + int upperBound = (int) outboundTable.getValueAt(row, "RemainingHigh"); + int stopNumber = (int) outboundTable.getValueAt(row, "Stop"); + + for (int duration = lowerBound; duration <= upperBound; ++duration) + { + int key = getKey(stopNumber, duration); + outboundElementMap.put(key, row - 1); + } + + // cumulative probability distribution + double cumProb = 0; + for (int col = 4; col <= columns; ++col) + { + cumProb += outboundTable.getValueAt(row, col); + outboundCumProbability[row - 1][col - 4] = cumProb; + } + + } + + } + + /** + * Read the inbound stop duration file and store the cumulative probability + * distribution as well as the offsets and set the key map to index into the + * probability array. + * + * @param fileName + */ + public void readInboundFile(String fileName) + { + TableDataSet inboundTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + inboundTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + int columns = inboundTable.getColumnCount(); + int rows = inboundTable.getRowCount(); + inboundCumProbability = new double[rows][columns - 3]; + + // first three columns are index fields, rest are offsets + inboundOffsets = new int[columns - 3]; + for (int i = 4; i <= columns; ++i) + { + String offset = inboundTable.getColumnLabel(i); + inboundOffsets[i - 4] = new Integer(offset); + } + + // now fill in cumulative probability array + for (int row = 1; row <= rows; ++row) + { + + int lowerBound = (int) inboundTable.getValueAt(row, "RemainingLow"); + int upperBound = (int) inboundTable.getValueAt(row, "RemainingHigh"); + int stopNumber = (int) inboundTable.getValueAt(row, "Stop"); + + for (int duration = lowerBound; duration <= upperBound; ++duration) + { + int key = getKey(stopNumber, duration); + inboundElementMap.put(key, row - 1); + } + // cumulative probability distribution + double cumProb = 0; + for (int col = 4; col <= columns; ++col) + { + cumProb += inboundTable.getValueAt(row, col); + inboundCumProbability[row - 1][col - 4] = cumProb; + } + + } + + } + + /** + * Get the key for the arrayElementMap. + * + * @param stopNumber + * stop number + * @param periodsRemaining + * Remaining time periods + * @return arrayElementMap key. + */ + private int getKey(int stopNumber, int periodsRemaining) + { + + return periodsRemaining * 10 + stopNumber; + } + + /** + * Choose the stop time of day period. + * + * @param tour + * @param stop + */ + public void chooseTOD(CrossBorderTour tour, CrossBorderStop stop) + { + + boolean inbound = stop.isInbound(); + int stopNumber = stop.getId() + 1; + int arrivalPeriod = tour.getArriveTime(); + + if (!inbound) + { + + // find the departure time + int departPeriod = 0; + if (stop.getId() == 0) departPeriod = tour.getDepartTime(); + else + { + CrossBorderStop[] stops = tour.getOutboundStops(); + departPeriod = stops[stop.getId() - 1].getStopPeriod(); + } + + int periodsRemaining = arrivalPeriod - departPeriod; + + int key = getKey(stopNumber, periodsRemaining); + int element = outboundElementMap.get(key); + double[] cumProb = outboundCumProbability[element]; + double random = tour.getRandom(); + + // iterate through the offset distribution, choose an offset, and + // set in the stop + if (tour.getDebugChoiceModels()) + { + logger.info("Stop TOD Choice Model for tour " + tour.getID() + " outbound stop " + + stop.getId() + " periods remaining " + periodsRemaining); + logger.info(" random number " + random); + } + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + int offset = outboundOffsets[i]; + int period = departPeriod + offset; + stop.setPeriod(period); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Chose alt " + i + " offset " + offset + " from depart period " + + departPeriod); + logger.info("Stop period is " + stop.getStopPeriod()); + + } + break; + + } + } + } else + { + // inbound stop + + // find the departure time + int departPeriod = 0; + + // first inbound stop + if (stop.getId() == 0) + { + + // there were outbound stops + if (tour.getOutboundStops() != null) + { + CrossBorderStop[] outboundStops = tour.getOutboundStops(); + departPeriod = outboundStops[outboundStops.length - 1].getStopPeriod(); + } else + { + // no outbound stops + departPeriod = tour.getDepartTime(); + } + } else + { + // not first inbound stop + CrossBorderStop[] stops = tour.getInboundStops(); + departPeriod = stops[stop.getId() - 1].getStopPeriod(); + } + + int periodsRemaining = arrivalPeriod - departPeriod; + + int key = getKey(stopNumber, periodsRemaining); + int element = inboundElementMap.get(key); + double[] cumProb = inboundCumProbability[element]; + double random = tour.getRandom(); + if (tour.getDebugChoiceModels()) + { + logger.info("Stop TOD Choice Model for tour " + tour.getID() + " inbound stop " + + stop.getId() + " periods remaining " + periodsRemaining); + logger.info("Random number " + random); + } + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + int offset = inboundOffsets[i]; + int arrivePeriod = tour.getArriveTime(); + int period = arrivePeriod + offset; + stop.setPeriod(period); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Chose alt " + i + " offset " + offset + " from arrive period " + + arrivePeriod); + logger.info("Stop period is " + stop.getStopPeriod()); + + } + break; + } + } + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java new file mode 100644 index 0000000..59d9c2b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java @@ -0,0 +1,367 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Household; +import com.pb.common.math.MersenneTwister; + +public class CrossBorderTour + implements Serializable +{ + + private MersenneTwister random; + private int ID; + + // following variables determined via simulation + private byte purpose; + private boolean sentriAvailable; + + private CrossBorderStop[] outboundStops; + private CrossBorderStop[] inboundStops; + + private CrossBorderTrip[] trips; + + private int departTime; + private int arriveTime; + + private boolean debugChoiceModels; + + // following variables chosen via choice models + private int poe; + private int originMGRA; + private int destinationMGRA; + private int originTAZ; + private int destinationTAZ; + private byte tourMode; + private float workTimeFactor; + private float nonWorkTimeFactor; + private float valueOfTime; + + private boolean avAvailable; + + /** + * Public constructor. + * + * @param seed + * A seed for the random number generator. + */ + public CrossBorderTour(long seed) + { + + random = new MersenneTwister(seed); + } + + /** + * @return the iD + */ + public int getID() + { + return ID; + } + + /** + * @param iD + * the iD to set + */ + public void setID(int iD) + { + ID = iD; + } + + /** + * @return the sentriAvailable + */ + public boolean isSentriAvailable() + { + return sentriAvailable; + } + + /** + * @return the poe + */ + public int getPoe() + { + return poe; + } + + /** + * @param poe + * the poe to set + */ + public void setPoe(int poe) + { + this.poe = poe; + } + + /** + * @param sentriAvailable + * the sentriAvailable to set + */ + public void setSentriAvailable(boolean sentriAvailable) + { + this.sentriAvailable = sentriAvailable; + } + + /** + * @return the purpose + */ + public byte getPurpose() + { + return purpose; + } + + /** + * @return the outboundStops + */ + public CrossBorderStop[] getOutboundStops() + { + return outboundStops; + } + + /** + * @param outboundStops + * the outboundStops to set + */ + public void setOutboundStops(CrossBorderStop[] outboundStops) + { + this.outboundStops = outboundStops; + } + + /** + * @return the inboundStops + */ + public CrossBorderStop[] getInboundStops() + { + return inboundStops; + } + + /** + * @param inboundStops + * the inboundStops to set + */ + public void setInboundStops(CrossBorderStop[] inboundStops) + { + this.inboundStops = inboundStops; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(byte purpose) + { + this.purpose = purpose; + } + + /** + * @return the departTime + */ + public int getDepartTime() + { + return departTime; + } + + /** + * @param departTime + * the departTime to set + */ + public void setDepartTime(int departTime) + { + this.departTime = departTime; + } + + public CrossBorderTrip[] getTrips() + { + return trips; + } + + public void setTrips(CrossBorderTrip[] trips) + { + this.trips = trips; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() + { + return originMGRA; + } + + /** + * @param originMGRA + * the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) + { + this.originMGRA = originMGRA; + } + + public int getOriginTAZ() + { + return originTAZ; + } + + public void setOriginTAZ(int originTAZ) + { + this.originTAZ = originTAZ; + } + + public int getDestinationTAZ() + { + return destinationTAZ; + } + + public void setDestinationTAZ(int destinationTAZ) + { + this.destinationTAZ = destinationTAZ; + } + + /** + * @return the tour mode + */ + public byte getTourMode() + { + return tourMode; + } + + /** + * @param mode + * the tour mode to set + */ + public void setTourMode(byte mode) + { + this.tourMode = mode; + } + + /** + * Get a random number from the parties random class. + * + * @return A random number. + */ + public double getRandom() + { + return random.nextDouble(); + } + + /** + * @return the debugChoiceModels + */ + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + /** + * @param debugChoiceModels + * the debugChoiceModels to set + */ + public void setDebugChoiceModels(boolean debugChoiceModels) + { + this.debugChoiceModels = debugChoiceModels; + } + + + /** + * Get the number of outbound stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberOutboundStops() + { + if (outboundStops == null) return 0; + else return outboundStops.length; + + } + + /** + * Get the number of return stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberInboundStops() + { + if (inboundStops == null) return 0; + else return inboundStops.length; + + } + + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() + { + return destinationMGRA; + } + + /** + * @param destinationMGRA + * the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) + { + this.destinationMGRA = destinationMGRA; + } + + public void setArriveTime(int arriveTime) + { + this.arriveTime = arriveTime; + } + + public int getArriveTime() + { + return arriveTime; + } + + public double getWorkTimeFactor() { + return workTimeFactor; + } + + public void setWorkTimeFactor(float workTimeFactor) { + this.workTimeFactor = workTimeFactor; + } + + public double getNonWorkTimeFactor() { + return nonWorkTimeFactor; + } + + public void setNonWorkTimeFactor(float nonWorkTimeFactor) { + this.nonWorkTimeFactor = nonWorkTimeFactor; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public boolean isAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(boolean avAvailable) { + this.avAvailable = avAvailable; + } + + public void logTourObject(Logger logger, int totalChars) + { + + Household.logHelper(logger, "tourId: ", ID, totalChars); + Household.logHelper(logger, "tourPurpose: ", purpose, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); + Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); + Household.logHelper(logger, "tourMode: ", tourMode, totalChars); + Household.logHelper(logger, "avAvailable:", (avAvailable ? 0 : 1), totalChars); + + String tempString = null; + + logger.info(tempString); + + logger.info(tempString); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java new file mode 100644 index 0000000..97bfdb7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java @@ -0,0 +1,358 @@ +package org.sandag.abm.crossborder; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.TimeCoefficientDistributions; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +public class CrossBorderTourManager +{ + + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + + private CrossBorderTour[] tours; + private int totalTours; + + private double sentriShare; + + private double[] sentriPurposeDistribution; + private double[] nonSentriPurposeDistribution; + + CrossBorderModelStructure modelStructure; + SandagModelStructure sandagStructure; + private boolean seek; + private int traceId; + + private float avShare; + + TimeCoefficientDistributions timeDistributions; + + boolean distributedTimeCoefficients = false; + + /** + * Constructor. Reads properties file and opens/stores all probability + * distributions for sampling. Estimates number of airport travel parties + * and initializes parties[]. + * + * @param resourceFile + * Property file. + * + * Creates the array of cross-border tours. + */ + public CrossBorderTourManager(HashMap rbMap) + { + + modelStructure = new CrossBorderModelStructure(); + sandagStructure = new SandagModelStructure(); + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String nonSentriPurposeFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.purpose.nonsentri.file"); + String sentriPurposeFile = directory + + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.purpose.sentri.file"); + + // the share of cross-border tours that are sentri is an input + sentriShare = new Double(Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.sentriShare")); + + // Read the distributions + sentriPurposeDistribution = setPurposeDistribution(sentriPurposeFile, + sentriPurposeDistribution,2); + nonSentriPurposeDistribution = setPurposeDistribution(nonSentriPurposeFile, + nonSentriPurposeDistribution,2); + totalTours = new Integer(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.tours") + .replace(",", "")); + + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trace")); + + distributedTimeCoefficients = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "distributedTimeCoefficients")); + + if(distributedTimeCoefficients) { + timeDistributions = new TimeCoefficientDistributions(); + timeDistributions.createTimeDistributions(rbMap); + } + + avShare = Util.getFloatValueFromPropertyMap(rbMap, "crossBorder.avShare"); + + } + + /** + * Generate and attribute cross border tours + */ + public void generateCrossBorderTours() + { + + // calculate total number of cross border tours + tours = new CrossBorderTour[totalTours]; + + logger.info("Total cross border tours: " + totalTours); + + for (int i = 0; i < tours.length; ++i) + { + + long seed = i * 10 + 1001; + CrossBorderTour tour = new CrossBorderTour(seed); + + tours[i] = tour; + + tour.setID(i + 1); + + // determine if tour is sentri, and calculate tour purpose + if (tour.getRandom() < sentriShare) + { + tour.setSentriAvailable(true); + int purpose = choosePurpose(tour.getRandom(), sentriPurposeDistribution); + tour.setPurpose((byte) purpose); + } else + { + tour.setSentriAvailable(false); + int purpose = choosePurpose(tour.getRandom(), nonSentriPurposeDistribution); + tour.setPurpose((byte) purpose); + } + + //set time factors + double workTimeFactor = 1.0; + double nonWorkTimeFactor = 1.0; + + if(distributedTimeCoefficients){ + double rnum = tour.getRandom(); + workTimeFactor = timeDistributions.sampleFromWorkDistribution(rnum); + nonWorkTimeFactor = timeDistributions.sampleFromNonWorkDistribution(rnum); + + } + tour.setWorkTimeFactor((float)workTimeFactor); + tour.setNonWorkTimeFactor((float)nonWorkTimeFactor); + + if(tour.getRandom() < avShare) + tour.setAvAvailable(true); + + } + } + + /** + * Read file containing probabilities by purpose. Store cumulative + * distribution in purposeDistribution. + * + * @param fileName + * Name of file containing two columns, one row for each purpose. + * First column has purpose number, second column has + * probability. + */ + protected double[] setPurposeDistribution(String fileName, double[] purposeDistribution, int position) + { + //logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; + purposeDistribution = new double[purposes]; + + double total_prob = 0.0; + // calculate and store cumulative probability distribution + for (int purp = 0; purp < purposes; ++purp) + { + + double probability = probabilityTable.getValueAt(purp + 1, position); + + total_prob += probability; + purposeDistribution[purp] = total_prob; + + } + //logger.info("End storing cumulative probabilies from file " + fileName); + + return purposeDistribution; + } + + /** + * Choose a purpose. + * + * @param random + * A uniform random number. + * @return the purpose. + */ + protected int choosePurpose(double random, double[] purposeDistribution) + { + // iterate through the probability array and choose + for (int alt = 0; alt < purposeDistribution.length; ++alt) + { + if (purposeDistribution[alt] > random) return alt; + } + return -99; + } + + /** + * Create a text file and write all records to the file. + * + */ + public void writeOutputFile(HashMap rbMap) + { + + // Open file and print header + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String tourFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.tour.output.file"); + String tripFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trip.output.file"); + + logger.info("Writing cross border tours to file " + tourFileName); + logger.info("Writing cross border trips to file " + tripFileName); + + PrintWriter tourWriter = null; + try + { + tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tourFileName + " for writing\n"); + throw new RuntimeException(); + } + String tourHeaderString = new String( + "id,purpose,sentri,poe,departTime,arriveTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tourMode,avAvailable,workTimeFactor,nonWorkTimeFactor,valueOfTime\n"); + tourWriter.print(tourHeaderString); + + PrintWriter tripWriter = null; + try + { + tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tripFileName + " for writing\n"); + throw new RuntimeException(); + } + String tripHeaderString = new String( + "tourID,tripID,originPurp,destPurp,originMGRA,destinationMGRA,originTAZ,destinationTAZ,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,avAvailable,boardingTap,alightingTap,set,workTimeFactor,nonWorkTimeFactor,valueOfTime,parkingCost\n"); + tripWriter.print(tripHeaderString); + + // Iterate through the array, printing records to the file + for (int i = 0; i < tours.length; ++i) + { + + CrossBorderTour tour = tours[i]; + + if (seek && tour.getID() != traceId) continue; + + CrossBorderTrip[] trips = tours[i].getTrips(); + + if (trips == null) continue; + + writeTour(tour, tourWriter); + + for (int j = 0; j < trips.length; ++j) + { + writeTrip(tour, trips[j], j + 1, tripWriter); + } + } + + tourWriter.close(); + tripWriter.close(); + + } + + /** + * Write the tour to the PrintWriter + * + * @param tour + * @param writer + */ + private void writeTour(CrossBorderTour tour, PrintWriter writer) + { + String record = new String(tour.getID() + "," + tour.getPurpose() + "," + + tour.isSentriAvailable() + "," + tour.getPoe() + "," + tour.getDepartTime() + "," + + tour.getArriveTime() + "," + tour.getOriginMGRA() + "," + + tour.getDestinationMGRA() + "," + tour.getOriginTAZ() + "," + + tour.getDestinationTAZ() + "," + tour.getTourMode() + "," + + (tour.isAvAvailable() ? 1 : 0) + "," + + String.format("%9.2f",tour.getWorkTimeFactor()) + "," + + String.format("%9.2f",tour.getNonWorkTimeFactor()) + "," + + String.format("%9.2f", tour.getValueOfTime()) +"\n"); + writer.print(record); + + } + + /** + * Write the trip to the PrintWriter + * + * @param tour + * @param trip + * @param tripNumber + * @param writer + */ + private void writeTrip(CrossBorderTour tour, CrossBorderTrip trip, int tripNumber, + PrintWriter writer) + { + + String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginPurpose() + + "," + trip.getDestinationPurpose() + "," + trip.getOriginMgra() + "," + + trip.getDestinationMgra() + "," + trip.getOriginTAZ() + "," + + trip.getDestinationTAZ() + "," + trip.isInbound() + "," + + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() + + "," + trip.getPeriod() + "," + trip.getTripMode() + "," + + (tour.isAvAvailable() ? 1 : 0) + "," + + trip.getBoardTap() + "," + trip.getAlightTap() + "," + + trip.getSet() + "," + + String.format("%9.2f",tour.getWorkTimeFactor()) + "," + + String.format("%9.2f",tour.getNonWorkTimeFactor()) + "," + + String.format("%9.2f", trip.getValueOfTime()) + "," + + String.format("%9.2f", trip.getParkingCost())+ "\n"); + writer.print(record); + } + /** + * @return the parties + */ + public CrossBorderTour[] getTours() + { + return tours; + } + + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Cross Border Model Tour Manager")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + CrossBorderTourManager apm = new CrossBorderTourManager(pMap); + apm.generateCrossBorderTours(); + apm.writeOutputFile(pMap); + + logger.info("Cross-Border Tour Manager successfully completed!"); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java new file mode 100644 index 0000000..9fbf960 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java @@ -0,0 +1,406 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class CrossBorderTourModeChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected double tourPurpose; + protected double tourModelsSentri; + protected double borderWaitStd; + protected double borderWaitPed; + protected double borderWaitSentri; + + protected double outboundTripMcLogsumDA; + protected double outboundTripMcLogsumSR2; + protected double outboundTripMcLogsumSR3; + protected double outboundTripMcLogsumWalk; + protected double inboundTripMcLogsumDA; + protected double inboundTripMcLogsumSR2; + protected double inboundTripMcLogsumSR3; + protected double inboundTripMcLogsumWalk; + + /** + * Constructor. + * + * @param modelStructure + */ + public CrossBorderTourModeChoiceDMU(CrossBorderModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * @return the tourPurpose + */ + public double getTourPurpose() + { + return tourPurpose; + } + + /** + * @param tourPurpose + * the tourPurpose to set + */ + public void setTourPurpose(double tourPurpose) + { + this.tourPurpose = tourPurpose; + } + + /** + * @return the tourModelsSentri + */ + public double getTourModelsSentri() + { + return tourModelsSentri; + } + + /** + * @param tourModelsSentri + * the tourModelsSentri to set + */ + public void setTourModelsSentri(double tourModelsSentri) + { + this.tourModelsSentri = tourModelsSentri; + } + + /** + * @return the borderWaitStd + */ + public double getBorderWaitStd() + { + return borderWaitStd; + } + + /** + * @param borderWaitStd + * the borderWaitStd to set + */ + public void setBorderWaitStd(double borderWaitStd) + { + this.borderWaitStd = borderWaitStd; + } + + /** + * @return the borderWaitPed + */ + public double getBorderWaitPed() + { + return borderWaitPed; + } + + /** + * @param borderWaitPed + * the borderWaitPed to set + */ + public void setBorderWaitPed(double borderWaitPed) + { + this.borderWaitPed = borderWaitPed; + } + + /** + * @return the borderWaitSentri + */ + public double getBorderWaitSentri() + { + return borderWaitSentri; + } + + /** + * @param borderWaitSentri + * the borderWaitSentri to set + */ + public void setBorderWaitSentri(double borderWaitSentri) + { + this.borderWaitSentri = borderWaitSentri; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the outboundTripMcLogsumDA + */ + public double getOutboundTripMcLogsumDA() + { + return outboundTripMcLogsumDA; + } + + /** + * @param outboundTripMcLogsumDA + * the outboundTripMcLogsumDA to set + */ + public void setOutboundTripMcLogsumDA(double outboundTripMcLogsumDA) + { + this.outboundTripMcLogsumDA = outboundTripMcLogsumDA; + } + + /** + * @return the outboundTripMcLogsumSR2 + */ + public double getOutboundTripMcLogsumSR2() + { + return outboundTripMcLogsumSR2; + } + + /** + * @param outboundTripMcLogsumSR2 + * the outboundTripMcLogsumSR2 to set + */ + public void setOutboundTripMcLogsumSR2(double outboundTripMcLogsumSR2) + { + this.outboundTripMcLogsumSR2 = outboundTripMcLogsumSR2; + } + + /** + * @return the outboundTripMcLogsumSR3 + */ + public double getOutboundTripMcLogsumSR3() + { + return outboundTripMcLogsumSR3; + } + + /** + * @param outboundTripMcLogsumSR3 + * the outboundTripMcLogsumSR3 to set + */ + public void setOutboundTripMcLogsumSR3(double outboundTripMcLogsumSR3) + { + this.outboundTripMcLogsumSR3 = outboundTripMcLogsumSR3; + } + + /** + * @return the outboundTripMcLogsumWalk + */ + public double getOutboundTripMcLogsumWalk() + { + return outboundTripMcLogsumWalk; + } + + /** + * @param outboundTripMcLogsumWalk + * the outboundTripMcLogsumWalk to set + */ + public void setOutboundTripMcLogsumWalk(double outboundTripMcLogsumWalk) + { + this.outboundTripMcLogsumWalk = outboundTripMcLogsumWalk; + } + + /** + * @return the inboundTripMcLogsumDA + */ + public double getInboundTripMcLogsumDA() + { + return inboundTripMcLogsumDA; + } + + /** + * @param inboundTripMcLogsumDA + * the inboundTripMcLogsumDA to set + */ + public void setInboundTripMcLogsumDA(double inboundTripMcLogsumDA) + { + this.inboundTripMcLogsumDA = inboundTripMcLogsumDA; + } + + /** + * @return the inboundTripMcLogsumSR2 + */ + public double getInboundTripMcLogsumSR2() + { + return inboundTripMcLogsumSR2; + } + + /** + * @param inboundTripMcLogsumSR2 + * the inboundTripMcLogsumSR2 to set + */ + public void setInboundTripMcLogsumSR2(double inboundTripMcLogsumSR2) + { + this.inboundTripMcLogsumSR2 = inboundTripMcLogsumSR2; + } + + /** + * @return the inboundTripMcLogsumSR3 + */ + public double getInboundTripMcLogsumSR3() + { + return inboundTripMcLogsumSR3; + } + + /** + * @param inboundTripMcLogsumSR3 + * the inboundTripMcLogsumSR3 to set + */ + public void setInboundTripMcLogsumSR3(double inboundTripMcLogsumSR3) + { + this.inboundTripMcLogsumSR3 = inboundTripMcLogsumSR3; + } + + /** + * @return the inboundTripMcLogsumWalk + */ + public double getInboundTripMcLogsumWalk() + { + return inboundTripMcLogsumWalk; + } + + /** + * @param inboundTripMcLogsumWalk + * the inboundTripMcLogsumWalk to set + */ + public void setInboundTripMcLogsumWalk(double inboundTripMcLogsumWalk) + { + this.inboundTripMcLogsumWalk = inboundTripMcLogsumWalk; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTourPurpose", 0); + methodIndexMap.put("getTourModelsSentri", 1); + methodIndexMap.put("getBorderWaitStd", 2); + methodIndexMap.put("getBorderWaitPed", 3); + methodIndexMap.put("getBorderWaitSentri", 4); + + methodIndexMap.put("getOutboundTripMcLogsumDA", 30); + methodIndexMap.put("getOutboundTripMcLogsumSR2", 31); + methodIndexMap.put("getOutboundTripMcLogsumSR3", 32); + methodIndexMap.put("getOutboundTripMcLogsumWalk", 33); + methodIndexMap.put("getInboundTripMcLogsumDA", 34); + methodIndexMap.put("getInboundTripMcLogsumSR2", 35); + methodIndexMap.put("getInboundTripMcLogsumSR3", 36); + methodIndexMap.put("getInboundTripMcLogsumWalk", 37); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTourPurpose(); + break; + case 1: + returnValue = getTourModelsSentri(); + break; + case 2: + returnValue = getBorderWaitStd(); + break; + case 3: + returnValue = getBorderWaitPed(); + break; + case 4: + returnValue = getBorderWaitSentri(); + break; + case 30: + returnValue = getOutboundTripMcLogsumDA(); + break; + case 31: + returnValue = getOutboundTripMcLogsumSR2(); + break; + case 32: + returnValue = getOutboundTripMcLogsumSR3(); + break; + case 33: + returnValue = getOutboundTripMcLogsumWalk(); + break; + case 34: + returnValue = getInboundTripMcLogsumDA(); + break; + case 35: + returnValue = getInboundTripMcLogsumSR2(); + break; + case 36: + returnValue = getInboundTripMcLogsumSR3(); + break; + case 37: + returnValue = getInboundTripMcLogsumWalk(); + break; + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java new file mode 100644 index 0000000..aa15f6f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java @@ -0,0 +1,590 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class CrossBorderTourModeChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + public static final boolean DEBUG_BEST_PATHS = false; + + private MgraDataManager mgraManager; + + /** + * A private class used as a key to store the wait times for a particular + * station. + * + * @author Freedman + * + */ + private class WaitTimeClass + { + int[] beginPeriod; // by time periods + int[] endPeriod; // by time periods + float[] StandardWait; // by time periods + float[] SENTRIWait; // by time periods + float[] PedestrianWait; // by time periods + } + + HashMap waitTimeMap; + + private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "crossBorder.tour.mc.uec.file"; + private static final String PROPERTIES_UEC_TOUR_DATA_SHEET = "crossBorder.tour.mc.data.page"; + private static final String PROPERTIES_UEC_MANDATORY_MODEL_SHEET = "crossBorder.tour.mc.mandatory.model.page"; + private static final String PROPERTIES_UEC_NONMANDATORY_MODEL_SHEET = "crossBorder.tour.mc.nonmandatory.model.page"; + private static final String PROPERTIES_POE_WAITTIMES = "crossBorder.poe.waittime.file"; + + private ChoiceModelApplication[] mcModel; // by + // segment + // - + // mandatory + // vs + // non-mandatory + // (each + // has + // different + // nesting + // coefficients) + private CrossBorderTripModeChoiceModel tripModeChoiceModel; + private CrossBorderTourModeChoiceDMU mcDmuObject; + private McLogsumsCalculator logsumHelper; + + private CrossBorderModelStructure modelStructure; + + private String tourCategory; + + private String[] modeAltNames; + + private boolean saveUtilsProbsFlag = false; + + double logsum = 0; + + // placeholders for calculation of logsums + private CrossBorderTour tour; + private CrossBorderTrip trip; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public CrossBorderTourModeChoiceModel(HashMap propertyMap, + CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, + AutoTazSkimsCalculator tazDistanceCalculator) + { + + mgraManager = MgraDataManager.getInstance(propertyMap); + modelStructure = myModelStructure; + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + mcDmuObject = dmuFactory.getCrossBorderTourModeChoiceDMU(); + setupModeChoiceModelApplicationArray(propertyMap); + + // Create a trip mode choice model object for calculation of logsums + tripModeChoiceModel = new CrossBorderTripModeChoiceModel(propertyMap, myModelStructure, + dmuFactory, tazDistanceCalculator); + + tour = new CrossBorderTour(0); + trip = new CrossBorderTrip(); + + String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); + String waitTimeFile = Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_POE_WAITTIMES); + waitTimeFile = directory + waitTimeFile; + + readWaitTimeFile(waitTimeFile); + } + + /** + * Read UECs and create model application objects. + * + * @param propertyMap + */ + private void setupModeChoiceModelApplicationArray(HashMap propertyMap) + { + + logger.info(String + .format("Setting up cross border tour (border crossing) mode choice model.")); + + // locate the mandatory tour mode choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + logger.info("Will read mcUECFile " + mcUecFile); + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_TOUR_DATA_SHEET)); + int mandatoryModelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_MANDATORY_MODEL_SHEET)); + int nonmandatoryModelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_NONMANDATORY_MODEL_SHEET)); + + // default is to not save the tour mode choice utils and probs for each + // tour + String saveUtilsProbsString = propertyMap + .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); + if (saveUtilsProbsString != null) + { + if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; + } + + mcModel = new ChoiceModelApplication[2]; + + mcModel[0] = new ChoiceModelApplication(mcUecFile, mandatoryModelPage, dataPage, + propertyMap, (VariableTable) mcDmuObject); + mcModel[1] = new ChoiceModelApplication(mcUecFile, nonmandatoryModelPage, dataPage, + propertyMap, (VariableTable) mcDmuObject); + + modeAltNames = mcModel[0].getAlternativeNames(); + + } + + /** + * Get the Logsum. + * + * @param tour + * @param modelLogger + * @param choiceModelDescription + * @param decisionMakerLabel + * @return + */ + public double getLogsum(CrossBorderTour tour, Logger modelLogger, + String choiceModelDescription, String decisionMakerLabel) + { + + // set all tour mode DMU attributes including calculation of trip mode + // choice logsums for inbound & outbound directions. + setDmuAttributes(tour); + + return getModeChoiceLogsum(tour, modelLogger, choiceModelDescription, decisionMakerLabel); + + } + + /** + * Set the tour mode choice attributes. + * + * @param tour + */ + public void setDmuAttributes(CrossBorderTour tour) + { + + codeWaitTime(tour); + setTripLogsums(tour); + } + + /** + * Code wait times in the mc dmu object. + * + * @param tour + * The tour with an origin MGRA and departure time period. + */ + public void codeWaitTime(CrossBorderTour tour) + { + + // get the wait time class from the waitTimeMap HashMap + int station = tour.getPoe(); + int period = tour.getDepartTime(); + WaitTimeClass wait = waitTimeMap.get(station); + int[] beginTime = wait.beginPeriod; + int[] endTime = wait.endPeriod; + + // iterate through time arrays, find corresponding row, and set wait + // times + for (int i = 0; i < beginTime.length; ++i) + { + if (period >= beginTime[i] && period <= endTime[i]) + { + mcDmuObject.borderWaitStd = wait.StandardWait[i]; + mcDmuObject.borderWaitSentri = wait.SENTRIWait[i]; + mcDmuObject.borderWaitPed = wait.PedestrianWait[i]; + break; + } + } + } + + /** + * Set trip mode choice logsums (outbound and inbound) for calculation of + * tour mode choice model. + * + * @param tour + * The tour with other attributes such as origin, destination, + * purpose coded. + */ + public void setTripLogsums(CrossBorderTour tour) + { + + // outbound + trip.initializeFromTour(tour, true); + + // DA logsum + tour.setTourMode(modelStructure.DRIVEALONE); + double logsumDAOut = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setOutboundTripMcLogsumDA(logsumDAOut); + + // S2 logsum + tour.setTourMode(modelStructure.SHARED2); + double logsumS2Out = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setOutboundTripMcLogsumSR2(logsumS2Out); + + // S2 logsum + tour.setTourMode(modelStructure.SHARED3); + double logsumS3Out = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setOutboundTripMcLogsumSR3(logsumS3Out); + + // walk logsum + tour.setTourMode(modelStructure.WALK); + double logsumWalkOut = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setOutboundTripMcLogsumWalk(logsumWalkOut); + + // inbound + trip.initializeFromTour(tour, false); + + // DA logsum + tour.setTourMode(modelStructure.DRIVEALONE); + double logsumDAIn = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setInboundTripMcLogsumDA(logsumDAIn); + + // S2 logsum + tour.setTourMode(modelStructure.SHARED2); + double logsumS2In = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setInboundTripMcLogsumSR2(logsumS2In); + + // S2 logsum + tour.setTourMode(modelStructure.SHARED3); + double logsumS3In = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setInboundTripMcLogsumSR3(logsumS3In); + + // walk logsum + tour.setTourMode(modelStructure.WALK); + double logsumWalkIn = tripModeChoiceModel.computeUtilities(tour, trip); + mcDmuObject.setInboundTripMcLogsumWalk(logsumWalkIn); + + } + + /** + * Get an index into the mcModel array for the tour purpose. + * + * @param tour + * @return The index. + */ + public int getModelIndex(CrossBorderTour tour) + { + int modelIndex = 1; + if (tour.getPurpose() == modelStructure.WORK || tour.getPurpose() == modelStructure.SCHOOL) + modelIndex = 0; + return modelIndex; + } + + /** + * Get the tour mode choice logsum. + * + * @param tour + * @param modelLogger + * @param choiceModelDescription + * @param decisionMakerLabel + * @return Tour mode choice logsum + */ + public double getModeChoiceLogsum(CrossBorderTour tour, Logger modelLogger, + String choiceModelDescription, String decisionMakerLabel) + { + setDmuAttributes(tour); + + int modelIndex = getModelIndex(tour); + + // log headers to traceLogger + if (tour.getDebugChoiceModels()) + { + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + } + + mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); + + double logsum = mcModel[modelIndex].getLogsum(); + + // write UEC calculation results to separate model specific log file + if (tour.getDebugChoiceModels()) + { + String loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); + modelLogger.info(""); + modelLogger.info(""); + } + + return logsum; + + } + + /** + * Use to choose tour mode and set result in tour object. Also set value of time in tour object. + * + * @param tour + * The crossborder tour + */ + public void chooseTourMode(CrossBorderTour tour) + { + + byte tourMode = (byte) getModeChoice(tour); + tour.setTourMode(tourMode); + + float valueOfTime = tripModeChoiceModel.getTourValueOfTime(tourMode); + tour.setValueOfTime(valueOfTime); + } + + /** + * Use to return the tour mode without setting in the tour object. + * + * @param tour + * The cross border tour whose mode to choose. + * @return An integer corresponding to the tour mode. + */ + public int getModeChoice(CrossBorderTour tour) + { + int modelIndex = getModelIndex(tour); + + Logger modelLogger = null; + modelLogger = logger; + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + String separator = ""; + + if (tour.getDebugChoiceModels()) + { + String purposeName = modelStructure.CROSSBORDER_PURPOSES[tour.getPurpose()]; + choiceModelDescription = String.format( + "%s Tour Mode Choice Model for: Purpose=%s, Origin=%d, Dest=%d", tourCategory, + purposeName, tour.getOriginMGRA(), tour.getDestinationMGRA()); + decisionMakerLabel = String.format(" tour ID =%d", tour.getID()); + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + tour.logTourObject(modelLogger, loggingHeader.length()); + } + + setDmuAttributes(tour); + + mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); + + double rn = tour.getRandom(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen; + if (mcModel[modelIndex].getAvailabilityCount() > 0) + { + + chosen = mcModel[modelIndex].getChoiceResult(rn); + + } else + { + + String purposeName = modelStructure.CROSSBORDER_PURPOSES[tour.getPurpose()]; + choiceModelDescription = String + .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", + tourCategory, purposeName, tour.getOriginMGRA(), + tour.getDestinationMGRA()); + decisionMakerLabel = String.format("TourId=%d", tour.getID()); + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + tour.logTourObject(modelLogger, loggingHeader.length()); + + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + modelLogger.info(""); + modelLogger.info(""); + + logger.error(String + .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", + tour.getID(), tourCategory)); + throw new RuntimeException(); + } + + // debug output + if (tour.getDebugChoiceModels()) + { + + double[] utilities = mcModel[modelIndex].getUtilities(); // 0s-indexing + double[] probabilities = mcModel[modelIndex].getProbabilities(); // 0s-indexing + boolean[] availabilities = mcModel[modelIndex].getAvailabilities(); // 1s-indexing + String[] altNames = mcModel[modelIndex].getAlternativeNames(); // 0s-indexing + + modelLogger.info("Tour Id: " + tour.getID()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("-------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < mcModel[modelIndex].getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %s", k + 1, altNames[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f", altString, rn)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to log file + mcModel[modelIndex].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + mcModel[modelIndex].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + mcModel[modelIndex].logLogitCalculations(choiceModelDescription, decisionMakerLabel); + + // write UEC calculation results to separate model specific log file + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + } + + if (saveUtilsProbsFlag) + { + + // get the utilities and probabilities arrays for the tour mode + // choice + // model for this tour and save them to the tour object + double[] dUtils = mcModel[modelIndex].getUtilities(); + double[] dProbs = mcModel[modelIndex].getProbabilities(); + + float[] utils = new float[dUtils.length]; + float[] probs = new float[dUtils.length]; + for (int k = 0; k < dUtils.length; k++) + { + utils[k] = (float) dUtils[k]; + probs[k] = (float) dProbs[k]; + } + + // tour.setTourModalUtilities(utils); + // tour.setTourModalProbabilities(probs); + + } + + return chosen; + + } + + /** + * Read wait time file and store wait times in waitTimeMap HashMap. + * + * @param fileName + * Name of file containing station, beginPeriod, endPeriod and + * wait time for standard, SENTRI, and pedestrians. + */ + protected void readWaitTimeFile(String fileName) + { + logger.info("Begin reading the data in file " + fileName); + TableDataSet waitTimeTable; + waitTimeMap = new HashMap(); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + waitTimeTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + int rowCount = waitTimeTable.getRowCount(); + + // iterate through file and fill up waitTimeMap + int lastStation = -999; + int index = 0; + WaitTimeClass wait = null; + for (int row = 1; row <= rowCount; ++row) + { + + int station = (int) waitTimeTable.getValueAt(row, "poe"); + if (station != lastStation) + { + wait = new WaitTimeClass(); + wait.beginPeriod = new int[modelStructure.TIME_PERIODS]; + wait.endPeriod = new int[modelStructure.TIME_PERIODS]; + wait.StandardWait = new float[modelStructure.TIME_PERIODS]; + wait.SENTRIWait = new float[modelStructure.TIME_PERIODS]; + wait.PedestrianWait = new float[modelStructure.TIME_PERIODS]; + index = 0; + lastStation = station; + } else + { + ++index; + } + + wait.beginPeriod[index] = (int) waitTimeTable.getValueAt(row, "StartPeriod"); + wait.endPeriod[index] = (int) waitTimeTable.getValueAt(row, "EndPeriod"); + wait.StandardWait[index] = waitTimeTable.getValueAt(row, "StandardWait"); + wait.SENTRIWait[index] = waitTimeTable.getValueAt(row, "SENTRIWait"); + wait.PedestrianWait[index] = waitTimeTable.getValueAt(row, "PedestrianWait"); + + waitTimeMap.put(station, wait); + + } + logger.info("End reading the data in file " + fileName); + + } + + public String[] getModeAltNames(int purposeIndex) + { + return modeAltNames; + } + + /** + * @return the tripModeChoiceModel + */ + public CrossBorderTripModeChoiceModel getTripModeChoiceModel() + { + return tripModeChoiceModel; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java new file mode 100644 index 0000000..31e4bb6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java @@ -0,0 +1,188 @@ +package org.sandag.abm.crossborder; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the TOD choice model for cross border tours. It is currently + * based on a static probability distribution stored in an input file, and + * indexed into by purpose. + * + * @author Freedman + * + */ +public class CrossBorderTourTimeOfDayChoiceModel +{ + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private double[][] cumProbability; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + private int[][] outboundPeriod; // by + // purpose, + // alternative: + // outbound + // period + private int[][] returnPeriod; // by + // purpose, + // alternative: + // return + // period + CrossBorderModelStructure modelStructure; + + /** + * Constructor. + */ + public CrossBorderTourTimeOfDayChoiceModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, + "crossBorder.tour.tod.file"); + stationDiurnalFile = directory + stationDiurnalFile; + + modelStructure = new CrossBorderModelStructure(); + + readTODFile(stationDiurnalFile); + + } + + /** + * Read the TOD distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readTODFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating tour TOD probability distribution"); + + int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 + int periods = modelStructure.TIME_PERIODS; // start at 1 + int periodCombinations = periods * (periods + 1) / 2; + + cumProbability = new double[purposes][periodCombinations]; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + outboundPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // outbound + // period + returnPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // return period + + // fill up arrays + int rowCount = probabilityTable.getRowCount(); + int lastPurpose = -99; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); + int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); + + // continue if return period before outbound period + if (retPer < outPer) continue; + + // reset if new purpose + if (purpose != lastPurpose) + { + + // log cumulative probability just in case + /* + if (lastPurpose != -99) + logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); + */ + cumProb = 0; + alt = 0; + } + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + outboundPeriod[purpose][alt] = outPer; + returnPeriod[purpose][alt] = retPer; + + //temporary + //logger.info("row="+row+" alt="+alt+" purpose="+purpose+" outPer="+outPer+" retPer="+retPer+" cumProb="+cumProb); + + ++alt; + + lastPurpose = purpose; + } + + logger.info("End calculating tour TOD probability distribution"); + + } + + /** + * Calculate tour time of day for the tour. + * + * @param tour + * A cross border tour (with purpose) + */ + public void calculateTourTOD(CrossBorderTour tour) + { + + int purpose = tour.getPurpose(); + double random = tour.getRandom(); + + if (tour.getDebugChoiceModels()) + { + logger.info("Choosing tour time of day for purpose " + + modelStructure.CROSSBORDER_PURPOSES[purpose] + " using random number " + + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + if (random < cumProbability[purpose][i]) + { + int depart = outboundPeriod[purpose][i]; + int arrive = returnPeriod[purpose][i]; + tour.setDepartTime(depart); + tour.setArriveTime(arrive); + break; + } + } + + if (tour.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " + + tour.getArriveTime()); + logger.info(""); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java new file mode 100644 index 0000000..02d205c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java @@ -0,0 +1,502 @@ +package org.sandag.abm.crossborder; + +public class CrossBorderTrip +{ + + private int originMgra; + private int destinationMgra; + private int originTAZ; + private int destinationTAZ; + private int tripMode; + private byte originPurpose; + private byte destinationPurpose; + private byte period; + private boolean inbound; + private boolean firstTrip; + private boolean lastTrip; + private boolean originIsTourDestination; + private boolean destinationIsTourDestination; + + private float parkingCost; + + private int boardTap; + private int alightTap; + private int set = -1; + float valueOfTime; + + /** + * Default constructor; nothing initialized. + */ + public CrossBorderTrip() + { + + } + + /** + * Create a cross border trip from a tour leg (no stops). + * + * @param tour + * The tour. + * @param outbound + * Outbound direction + */ + public CrossBorderTrip(CrossBorderTour tour, boolean outbound) + { + + initializeFromTour(tour, outbound); + + } + + /** + * Initilize from the tour. + * + * @param tour + * The tour. + * @param outbound + * Outbound direction. + */ + public void initializeFromTour(CrossBorderTour tour, boolean outbound) + { + // Note: mode is unknown + if (outbound) + { + this.originMgra = tour.getOriginMGRA(); + this.destinationMgra = tour.getDestinationMGRA(); + this.originTAZ = tour.getOriginTAZ(); + this.destinationTAZ = tour.getDestinationTAZ(); + this.originPurpose = -1; + this.destinationPurpose = tour.getPurpose(); + this.period = (byte) tour.getDepartTime(); + this.inbound = false; + this.firstTrip = true; + this.lastTrip = false; + this.originIsTourDestination = false; + this.destinationIsTourDestination = true; + } else + { + this.originMgra = tour.getDestinationMGRA(); + this.destinationMgra = tour.getOriginMGRA(); + this.originTAZ = tour.getDestinationTAZ(); + this.destinationTAZ = tour.getOriginTAZ(); + this.originPurpose = tour.getPurpose(); + this.destinationPurpose = -1; + this.period = (byte) tour.getArriveTime(); + this.inbound = true; + this.firstTrip = false; + this.lastTrip = true; + this.originIsTourDestination = true; + this.destinationIsTourDestination = false; + } + + } + + /** + * Create a cross border trip from a tour\stop. Note: trip mode is unknown. + * Stop period is only known for first, last stop on tour. + * + * @param tour + * The tour. + * @param stop + * The stop + */ + public CrossBorderTrip(CrossBorderTour tour, CrossBorderStop stop, boolean toStop) + { + + initializeFromStop(tour, stop, toStop); + } + + /** + * Initialize from stop attributes. A trip will be created to the stop if + * toStop is true, else a trip will be created from the stop. Use after all + * stop locations are known, or else reset the stop origin and destination + * mgras accordingly after using. + * + * @param tour + * @param stop + * @param toStop + */ + public void initializeFromStop(CrossBorderTour tour, CrossBorderStop stop, boolean toStop) + { + + this.inbound = stop.isInbound(); + this.destinationIsTourDestination = false; + this.originIsTourDestination = false; + + // if trip to stop, destination is stop mgra; else origin is stop mgra + if (toStop) + { + this.destinationMgra = stop.getMgra(); + this.destinationTAZ = stop.getTAZ(); + this.destinationPurpose = stop.getPurpose(); + } else + { + this.originMgra = stop.getMgra(); + this.originTAZ = stop.getTAZ(); + this.originPurpose = stop.getPurpose(); + } + CrossBorderStop[] stops; + + if (!inbound) stops = tour.getOutboundStops(); + else stops = tour.getInboundStops(); + + // if outbound, and trip is to stop + if (!inbound && toStop) + { + + // first trip on outbound journey, origin is tour origin + if (stop.getId() == 0) + { + this.originMgra = tour.getOriginMGRA(); + this.originTAZ = tour.getOriginTAZ(); + this.originPurpose = -1; + this.period = (byte) tour.getDepartTime(); + } else + { + // not first trip on outbound journey, origin is last stop + this.originMgra = stops[stop.getId() - 1].getMgra(); // last + // stop + // location + this.originTAZ = stops[stop.getId() - 1].getTAZ(); // last stop + // location + this.originPurpose = stops[stop.getId() - 1].getPurpose(); // last + // stop + // location + this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); + } + } else if (!inbound && !toStop) + { + // outbound and trip is from stop to either next stop or tour + // destination. + + // last trip on outbound journey, destination is tour destination + if (stop.getId() == (stops.length - 1)) + { + this.destinationMgra = tour.getDestinationMGRA(); + this.destinationTAZ = tour.getDestinationTAZ(); + this.destinationPurpose = tour.getPurpose(); + this.destinationIsTourDestination = true; + } else + { + // not last trip on outbound journey, destination is next stop + this.destinationMgra = stops[stop.getId() + 1].getMgra(); + this.destinationTAZ = stops[stop.getId() + 1].getTAZ(); + this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); + } + + // the period for the trip is the origin for the trip + if (stop.getId() == 0) this.period = (byte) tour.getDepartTime(); + else this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); + + } else if (inbound && toStop) + { + // inbound, trip is to stop from either tour destination or last + // stop. + + // first inbound trip; origin is tour destination + if (stop.getId() == 0) + { + this.originMgra = tour.getDestinationMGRA(); + this.originTAZ = tour.getDestinationTAZ(); + this.originPurpose = tour.getPurpose(); + this.originIsTourDestination = true; + } else + { + // not first inbound trip; origin is last stop + this.originMgra = stops[stop.getId() - 1].getMgra(); // last + // stop + // location + this.originTAZ = stops[stop.getId() - 1].getTAZ(); // last stop + // location + this.originPurpose = stops[stop.getId() - 1].getPurpose(); + } + + // the period for the trip is the destination for the trip + if (stop.getId() == stops.length - 1) this.period = (byte) tour.getArriveTime(); + else this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); + } else + { + // inbound, trip is from stop to either next stop or tour origin. + + // last trip, destination is back to tour origin + if (stop.getId() == (stops.length - 1)) + { + this.destinationMgra = tour.getOriginMGRA(); + this.destinationTAZ = tour.getOriginTAZ(); + this.destinationPurpose = -1; + this.period = (byte) tour.getArriveTime(); + } else + { + // not last trip, destination is next stop + this.destinationMgra = stops[stop.getId() + 1].getMgra(); + this.destinationTAZ = stops[stop.getId() + 1].getTAZ(); + this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); + this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); + } + } + + // code period for first trip on tour + if (toStop && !inbound && stop.getId() == 0) + { + this.firstTrip = true; + this.lastTrip = false; + this.period = (byte) tour.getDepartTime(); + } + // code period for last trip on tour + if (!toStop && inbound && stop.getId() == (stops.length - 1)) + { + this.firstTrip = false; + this.lastTrip = true; + this.period = (byte) tour.getArriveTime(); + } + + } + + /** + * @return the period + */ + public byte getPeriod() + { + return period; + } + + /** + * @param period + * the period to set + */ + public void setPeriod(byte period) + { + this.period = period; + } + + /** + * @return the origin purpose + */ + public byte getOriginPurpose() + { + return originPurpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setOriginPurpose(byte purpose) + { + this.originPurpose = purpose; + } + + /** + * @return the destination purpose + */ + public byte getDestinationPurpose() + { + return destinationPurpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setDestinationPurpose(byte purpose) + { + this.destinationPurpose = purpose; + } + + /** + * @return the originMgra + */ + public int getOriginMgra() + { + return originMgra; + } + + /** + * @param originMgra + * the originMgra to set + */ + public void setOriginMgra(int originMgra) + { + this.originMgra = originMgra; + } + + /** + * @return the destinationMgra + */ + public int getDestinationMgra() + { + return destinationMgra; + } + + /** + * @param destinationMgra + * the destinationMgra to set + */ + public void setDestinationMgra(int destinationMgra) + { + this.destinationMgra = destinationMgra; + } + + public int getOriginTAZ() + { + return originTAZ; + } + + public void setOriginTAZ(int originTAZ) + { + this.originTAZ = originTAZ; + } + + public int getDestinationTAZ() + { + return destinationTAZ; + } + + public void setDestinationTAZ(int destinationTAZ) + { + this.destinationTAZ = destinationTAZ; + } + + /** + * @return the tripMode + */ + public int getTripMode() + { + return tripMode; + } + + /** + * @param tripMode + * the tripMode to set + */ + public void setTripMode(int tripMode) + { + this.tripMode = tripMode; + } + + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the firstTrip + */ + public boolean isFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(boolean firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public boolean isLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(boolean lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the originIsTourDestination + */ + public boolean isOriginIsTourDestination() + { + return originIsTourDestination; + } + + /** + * @param originIsTourDestination + * the originIsTourDestination to set + */ + public void setOriginIsTourDestination(boolean originIsTourDestination) + { + this.originIsTourDestination = originIsTourDestination; + } + + /** + * @return the destinationIsTourDestination + */ + public boolean isDestinationIsTourDestination() + { + return destinationIsTourDestination; + } + + /** + * @param destinationIsTourDestination + * the destinationIsTourDestination to set + */ + public void setDestinationIsTourDestination(boolean destinationIsTourDestination) + { + this.destinationIsTourDestination = destinationIsTourDestination; + } + + public int getBoardTap() { + return boardTap; + } + + public void setBoardTap(int boardTap) { + this.boardTap = boardTap; + } + + public int getAlightTap() { + return alightTap; + } + + public void setAlightTap(int alightTap) { + this.alightTap = alightTap; + } + + public int getSet() { + return set; + } + + public void setSet(int set) { + this.set = set; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public float getParkingCost() { + return parkingCost; + } + + public void setParkingCost(float parkingCost) { + this.parkingCost = parkingCost; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java new file mode 100644 index 0000000..da3854f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java @@ -0,0 +1,762 @@ +package org.sandag.abm.crossborder; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.McLogsumsCalculator; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class CrossBorderTripModeChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(CrossBorderTripModeChoiceDMU.class); + + protected static final int WTW = McLogsumsCalculator.WTW; + protected static final int WTD = McLogsumsCalculator.WTD; + protected static final int DTW = McLogsumsCalculator.DTW; + protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + protected int tourDepartPeriod; + protected int tourArrivePeriod; + protected int tripPeriod; + protected int workTour; + protected int outboundStops; + protected int returnStops; + protected int firstTrip; + protected int lastTrip; + protected int tourModeIsDA; + protected int tourModeIsS2; + protected int tourModeIsS3; + protected int tourModeIsWalk; + protected int tourCrossingIsSentri; + protected float hourlyParkingCostTourDest; + protected float dailyParkingCostTourDest; + protected float monthlyParkingCostTourDest; + protected int tripOrigIsTourDest; + protected int tripDestIsTourDest; + protected float hourlyParkingCostTripOrig; + protected float hourlyParkingCostTripDest; + protected float workTimeFactor; + protected float nonWorkTimeFactor; + protected int avAvailable; + + protected double nmWalkTime; + protected double nmBikeTime; + protected HashMap methodIndexMap; + protected double ivtCoeff; + protected double costCoeff; + protected double walkTransitLogsum; + protected double pnrTransitLogsum; + protected double knrTransitLogsum; + + protected IndexValues dmuIndex; + protected int outboundHalfTourDirection; + + protected float waitTimeTaxi; + protected float waitTimeSingleTNC; + protected float waitTimeSharedTNC; + + public CrossBorderTripModeChoiceDMU(CrossBorderModelStructure modelStructure, Logger aLogger) + { + if (aLogger == null) + { + aLogger = Logger.getLogger("crossBorderModel"); + } + logger = aLogger; + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the tripPeriod + */ + public int getTripPeriod() + { + return tripPeriod; + } + + /** + * @param tripPeriod + * the tripPeriod to set + */ + public void setTripPeriod(int tripPeriod) + { + this.tripPeriod = tripPeriod; + } + + /** + * @return the workTour + */ + public int getWorkTour() + { + return workTour; + } + + /** + * @param workTour + * the workTour to set + */ + public void setWorkTour(int workTour) + { + this.workTour = workTour; + } + + /** + * @return the outboundStops + */ + public int getOutboundStops() + { + return outboundStops; + } + + /** + * @param outboundStops + * the outboundStops to set + */ + public void setOutboundStops(int outboundStops) + { + this.outboundStops = outboundStops; + } + + /** + * @return the returnStops + */ + public int getReturnStops() + { + return returnStops; + } + + /** + * @param returnStops + * the returnStops to set + */ + public void setReturnStops(int returnStops) + { + this.returnStops = returnStops; + } + + /** + * @return the firstTrip + */ + public int getFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(int firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public int getLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(int lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the tourModeIsDA + */ + public int getTourModeIsDA() + { + return tourModeIsDA; + } + + /** + * @param tourModeIsDA + * the tourModeIsDA to set + */ + public void setTourModeIsDA(int tourModeIsDA) + { + this.tourModeIsDA = tourModeIsDA; + } + + /** + * @return the tourModeIsS2 + */ + public int getTourModeIsS2() + { + return tourModeIsS2; + } + + /** + * @param tourModeIsS2 + * the tourModeIsS2 to set + */ + public void setTourModeIsS2(int tourModeIsS2) + { + this.tourModeIsS2 = tourModeIsS2; + } + + /** + * @return the tourModeIsS3 + */ + public int getTourModeIsS3() + { + return tourModeIsS3; + } + + /** + * @param tourModeIsS3 + * the tourModeIsS3 to set + */ + public void setTourModeIsS3(int tourModeIsS3) + { + this.tourModeIsS3 = tourModeIsS3; + } + + /** + * @return the tourModeIsWalk + */ + public int getTourModeIsWalk() + { + return tourModeIsWalk; + } + + /** + * @param tourModeIsWalk + * the tourModeIsWalk to set + */ + public void setTourModeIsWalk(int tourModeIsWalk) + { + this.tourModeIsWalk = tourModeIsWalk; + } + + /** + * @return the tourModeIsSentri + */ + public int getTourCrossingIsSentri() + { + return tourCrossingIsSentri; + } + + /** + * @param tourModeIsSentri + * the tourModeIsSentri to set + */ + public void setTourCrossingIsSentri(int tourCrossingIsSentri) + { + this.tourCrossingIsSentri = tourCrossingIsSentri; + } + + /** + * @return the hourlyParkingCostTourDest + */ + public float getHourlyParkingCostTourDest() + { + return hourlyParkingCostTourDest; + } + + /** + * @param hourlyParkingCostTourDest + * the hourlyParkingCostTourDest to set + */ + public void setHourlyParkingCostTourDest(float hourlyParkingCostTourDest) + { + this.hourlyParkingCostTourDest = hourlyParkingCostTourDest; + } + + /** + * @return the dailyParkingCostTourDest + */ + public float getDailyParkingCostTourDest() + { + return dailyParkingCostTourDest; + } + + /** + * @param dailyParkingCostTourDest + * the dailyParkingCostTourDest to set + */ + public void setDailyParkingCostTourDest(float dailyParkingCostTourDest) + { + this.dailyParkingCostTourDest = dailyParkingCostTourDest; + } + + /** + * @return the monthlyParkingCostTourDest + */ + public float getMonthlyParkingCostTourDest() + { + return monthlyParkingCostTourDest; + } + + /** + * @param monthlyParkingCostTourDest + * the monthlyParkingCostTourDest to set + */ + public void setMonthlyParkingCostTourDest(float monthlyParkingCostTourDest) + { + this.monthlyParkingCostTourDest = monthlyParkingCostTourDest; + } + + /** + * @return the tripOrigIsTourDest + */ + public int getTripOrigIsTourDest() + { + return tripOrigIsTourDest; + } + + /** + * @param tripOrigIsTourDest + * the tripOrigIsTourDest to set + */ + public void setTripOrigIsTourDest(int tripOrigIsTourDest) + { + this.tripOrigIsTourDest = tripOrigIsTourDest; + } + + /** + * @return the tripDestIsTourDest + */ + public int getTripDestIsTourDest() + { + return tripDestIsTourDest; + } + + /** + * @param tripDestIsTourDest + * the tripDestIsTourDest to set + */ + public void setTripDestIsTourDest(int tripDestIsTourDest) + { + this.tripDestIsTourDest = tripDestIsTourDest; + } + + /** + * @return the hourlyParkingCostTripOrig + */ + public float getHourlyParkingCostTripOrig() + { + return hourlyParkingCostTripOrig; + } + + /** + * @param hourlyParkingCostTripOrig + * the hourlyParkingCostTripOrig to set + */ + public void setHourlyParkingCostTripOrig(float hourlyParkingCostTripOrig) + { + this.hourlyParkingCostTripOrig = hourlyParkingCostTripOrig; + } + + /** + * @return the hourlyParkingCostTripDest + */ + public float getHourlyParkingCostTripDest() + { + return hourlyParkingCostTripDest; + } + + /** + * @param hourlyParkingCostTripDest + * the hourlyParkingCostTripDest to set + */ + public void setHourlyParkingCostTripDest(float hourlyParkingCostTripDest) + { + this.hourlyParkingCostTripDest = hourlyParkingCostTripDest; + } + + /** + * @return the outboundHalfTourDirection + */ + public int getOutboundHalfTourDirection() + { + return outboundHalfTourDirection; + } + + /** + * @param outboundHalfTourDirection + * the outboundHalfTourDirection to set + */ + public void setOutboundHalfTourDirection(int outboundHalfTourDirection) + { + this.outboundHalfTourDirection = outboundHalfTourDirection; + } + + /** + * @return the tourDepartPeriod + */ + public int getTourDepartPeriod() + { + return tourDepartPeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(int tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(int tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + /** + * @return the tourArrivePeriod + */ + public int getTourArrivePeriod() + { + return tourArrivePeriod; + } + + public double getNm_walkTime() + { + return nmWalkTime; + } + + public void setNonMotorizedWalkTime(double nmWalkTime) + { + this.nmWalkTime = nmWalkTime; + } + + public void setNonMotorizedBikeTime(double nmBikeTime) + { + this.nmBikeTime = nmBikeTime; + } + + public double getNm_bikeTime() + { + return nmBikeTime; + } + + + public float getWorkTimeFactor() { + return workTimeFactor; + } + + public void setWorkTimeFactor(float workTimeFactor) { + this.workTimeFactor = workTimeFactor; + } + + public float getNonWorkTimeFactor() { + return nonWorkTimeFactor; + } + + public void setNonWorkTimeFactor(float nonWorkTimeFactor) { + this.nonWorkTimeFactor = nonWorkTimeFactor; + } + + public double getIvtCoeff() { + return ivtCoeff; + } + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public double getCostCoeff() { + return costCoeff; + } + + public double getWalkTransitLogsum() { + return walkTransitLogsum; + } + + public void setWalkTransitLogsum(double walkTransitLogsum) { + this.walkTransitLogsum = walkTransitLogsum; + } + + public double getPnrTransitLogsum() { + return pnrTransitLogsum; + } + + public void setPnrTransitLogsum(double pnrTransitLogsum) { + this.pnrTransitLogsum = pnrTransitLogsum; + } + + public double getKnrTransitLogsum() { + return knrTransitLogsum; + } + + public void setKnrTransitLogsum(double knrTransitLogsum) { + this.knrTransitLogsum = knrTransitLogsum; + } + + + + + public int getAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(int avAvailable) { + this.avAvailable = avAvailable; + } + + public float getWaitTimeTaxi() { + return waitTimeTaxi; + } + + public void setWaitTimeTaxi(float waitTimeTaxi) { + this.waitTimeTaxi = waitTimeTaxi; + } + + public float getWaitTimeSingleTNC() { + return waitTimeSingleTNC; + } + + public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { + this.waitTimeSingleTNC = waitTimeSingleTNC; + } + + public float getWaitTimeSharedTNC() { + return waitTimeSharedTNC; + } + + public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { + this.waitTimeSharedTNC = waitTimeSharedTNC; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTourDepartPeriod", 0); + methodIndexMap.put("getTourArrivePeriod", 1); + methodIndexMap.put("getTripPeriod", 2); + + methodIndexMap.put("getWorkTour", 4); + methodIndexMap.put("getOutboundStops", 5); + methodIndexMap.put("getReturnStops", 6); + methodIndexMap.put("getFirstTrip", 7); + methodIndexMap.put("getLastTrip", 8); + methodIndexMap.put("getTourModeIsDA", 9); + methodIndexMap.put("getTourModeIsS2", 10); + methodIndexMap.put("getTourModeIsS3", 11); + methodIndexMap.put("getTourModeIsWalk", 12); + methodIndexMap.put("getTourCrossingIsSentri", 13); + methodIndexMap.put("getHourlyParkingCostTourDest", 14); + methodIndexMap.put("getDailyParkingCostTourDest", 15); + methodIndexMap.put("getMonthlyParkingCostTourDest", 16); + methodIndexMap.put("getTripOrigIsTourDest", 17); + methodIndexMap.put("getTripDestIsTourDest", 18); + methodIndexMap.put("getHourlyParkingCostTripOrig", 19); + methodIndexMap.put("getHourlyParkingCostTripDest", 20); + + methodIndexMap.put("getWorkTimeFactor", 50); + methodIndexMap.put("getNonWorkTimeFactor", 51); + + methodIndexMap.put("getIvtCoeff", 60); + methodIndexMap.put("getCostCoeff", 61); + + methodIndexMap.put("getWalkSetLogSum", 62); + methodIndexMap.put("getPnrSetLogSum", 63); + methodIndexMap.put("getKnrSetLogSum", 64); + + methodIndexMap.put("getWaitTimeTaxi", 70); + methodIndexMap.put("getWaitTimeSingleTNC", 71); + methodIndexMap.put("getWaitTimeSharedTNC", 72); + + methodIndexMap.put("getNm_walkTime", 90); + methodIndexMap.put("getNm_bikeTime", 91); + + methodIndexMap.put("getAvAvailable", 95); + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + case 0: + returnValue = getTourDepartPeriod(); + break; + case 1: + returnValue = getTourArrivePeriod(); + break; + case 2: + returnValue = getTripPeriod(); + break; + case 4: + returnValue = getWorkTour(); + break; + case 5: + returnValue = getOutboundStops(); + break; + case 6: + returnValue = getReturnStops(); + break; + case 7: + returnValue = getFirstTrip(); + break; + case 8: + returnValue = getLastTrip(); + break; + case 9: + returnValue = getTourModeIsDA(); + break; + case 10: + returnValue = getTourModeIsS2(); + break; + case 11: + returnValue = getTourModeIsS3(); + break; + case 12: + returnValue = getTourModeIsWalk(); + break; + case 13: + returnValue = getTourCrossingIsSentri(); + break; + case 14: + returnValue = getHourlyParkingCostTourDest(); + break; + case 15: + returnValue = getDailyParkingCostTourDest(); + break; + case 16: + returnValue = getMonthlyParkingCostTourDest(); + break; + case 17: + returnValue = getTripOrigIsTourDest(); + break; + case 18: + returnValue = getTripDestIsTourDest(); + break; + case 19: + returnValue = getHourlyParkingCostTripOrig(); + break; + case 20: + returnValue = getHourlyParkingCostTripDest(); + break; + case 50: + returnValue = getWorkTimeFactor(); + break; + case 51: + returnValue = getNonWorkTimeFactor(); + break; + case 60: + returnValue = getIvtCoeff(); + break; + case 61: + returnValue = getCostCoeff(); + break; + case 62: + returnValue = getWalkTransitLogsum(); + break; + case 63: + returnValue = getPnrTransitLogsum(); + break; + case 64: + returnValue = getKnrTransitLogsum(); + break; + case 70: return getWaitTimeTaxi(); + case 71: return getWaitTimeSingleTNC(); + case 72: return getWaitTimeSharedTNC(); + + case 90: + returnValue = getNm_walkTime(); + break; + case 91: + returnValue = getNm_bikeTime(); + break; + case 95: + returnValue = getAvAvailable(); + break; + + default: + logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java new file mode 100644 index 0000000..1d40c89 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java @@ -0,0 +1,369 @@ +package org.sandag.abm.crossborder; + +import java.util.HashMap; +import java.util.Random; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class CrossBorderTripModeChoiceModel +{ + + private transient Logger logger = Logger.getLogger("crossBorderModel"); + + private AutoAndNonMotorizedSkimsCalculator anm; + private McLogsumsCalculator logsumHelper; + private CrossBorderModelStructure modelStructure; + private SandagModelStructure sandagModelStructure; + private TazDataManager tazs; + private MgraDataManager mgraManager; + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + private CrossBorderTripModeChoiceDMU dmu; + private ChoiceModelApplication tripModeChoiceModel; + double logsum = 0; + + private TripModeChoiceDMU mcDmuObject; + private AutoTazSkimsCalculator tazDistanceCalculator; + + + private static final String PROPERTIES_UEC_DATA_SHEET = "crossBorder.trip.mc.data.page"; + private static final String PROPERTIES_UEC_MODEL_SHEET = "crossBorder.trip.mc.model.page"; + private static final String PROPERTIES_UEC_FILE = "crossBorder.trip.mc.uec.file"; + + private TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public CrossBorderTripModeChoiceModel(HashMap propertyMap, + CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + tazs = TazDataManager.getInstance(propertyMap); + mgraManager = MgraDataManager.getInstance(propertyMap); + + lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); + lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); + lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); + + modelStructure = myModelStructure; + sandagModelStructure = new SandagModelStructure(); + + this.tazDistanceCalculator = tazDistanceCalculator; + + setupTripModeChoiceModel(propertyMap, dmuFactory); + + } + + /** + * Read the UEC file and set up the trip mode choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupTripModeChoiceModel(HashMap propertyMap, + CrossBorderDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up cross border trip mode choice model.")); + + dmu = dmuFactory.getCrossBorderTripModeChoiceDMU(); + + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_DATA_SHEET)); + int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_MODEL_SHEET)); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); + tripModeUecFile = uecPath + tripModeUecFile; + + tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, + propertyMap, (VariableTable) dmu); + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + SandagModelStructure modelStructure = new SandagModelStructure(); + mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); + + tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); + tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); + + } + + /** + * Calculate utilities and return logsum for the tour and stop. + * + * @param tour + * @param trip + */ + public double computeUtilities(CrossBorderTour tour, CrossBorderTrip trip) + { + + setDmuAttributes(tour, trip); + + tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + tour.logTourObject(logger, 100); + tripModeChoiceModel.logUECResults(logger, "Cross border trip mode choice model"); + + } + + logsum = tripModeChoiceModel.getLogsum(); + + if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); + + return logsum; + + } + + /** + * Choose a mode and store in the trip object. + * + * @param tour + * CrossBorderTour + * @param trip + * CrossBorderTrip + * + */ + public void chooseMode(CrossBorderTour tour, CrossBorderTrip trip) + { + computeUtilities(tour, trip); + + double rand = tour.getRandom(); + int mode=0; + try{ + mode = tripModeChoiceModel.getChoiceResult(rand); + trip.setTripMode(mode); + float vot = getTripValueOfTime(mode); + trip.setValueOfTime(vot); + float parkingCost = getTripParkingCost(mode); + trip.setParkingCost(parkingCost); + + if(sandagModelStructure.getTripModeIsTransit(mode)){ + double[][] bestTapPairs = logsumHelper.getBestWtwTripTaps(); + //pick transit path from N-paths + double rn = tour.getRandom(); + int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); + int boardTap = (int) bestTapPairs[pathIndex][0]; + int alightTap = (int) bestTapPairs[pathIndex][1]; + int set = (int) bestTapPairs[pathIndex][2]; + trip.setBoardTap(boardTap); + trip.setAlightTap(alightTap); + trip.setSet(set); + } + }catch(Exception e){ + logger.info("rand="+rand); + tour.logTourObject(logger, 100); + logger.error(e.getMessage()); + } + + } + + /** + * Return parking cost from UEC if auto trip, else return 0. + * + * @param tripMode + * @return Parking cost if auto mode, else 0 + */ + public float getTripParkingCost(int tripMode) { + + float parkingCost=0; + + if(sandagModelStructure.getTripModeIsSovOrHov(tripMode)) { + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + int parkingCostIndex = uec.lookupVariableIndex("parkingCost"); + parkingCost = (float) uec.getValueForIndex(parkingCostIndex); + return parkingCost; + } + return parkingCost; + } + + /** + * This method looks up the value of time from the last call to the UEC and returns + * it based on the occupancy of the mode passed in as an argument. this method ensures + * that the value of time at a tour level is the same for all trips on the tour (even + * though the actual trip level VOT might vary based on the trip occupancy). + * + * @param tourMode + * @return The value of time + */ + public float getTourValueOfTime(int tourMode){ + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + + double vot = 0.0; + + if(tourMode== modelStructure.SHARED2){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = uec.getValueForIndex(votIndex); + }else if (tourMode== modelStructure.SHARED3){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = uec.getValueForIndex(votIndex); + } + return (float) (vot * 0.5); //take half the VOT (assumed for tours) + + + } + + /** + * This method looks up the value of time from the last call to the UEC and returns + * it based on the occupancy of the mode passed in as an argument. this method ensures + * that the value of time at a tour level is the same for all trips on the tour (even + * though the actual trip level VOT might vary based on the trip occupancy). + * + * @param tripMode + * @return The value of time + */ + public float getTripValueOfTime(int tripMode){ + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + + double vot = 0.0; + + if(modelStructure.getTripModeIsS2(tripMode)){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = uec.getValueForIndex(votIndex); + }else if (modelStructure.getTripModeIsS3(tripMode)){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = uec.getValueForIndex(votIndex); + } + return (float) vot; + + + } + + /** + * Set DMU attributes. + * + * @param tour + * @param trip + */ + public void setDmuAttributes(CrossBorderTour tour, CrossBorderTrip trip) + { + + int tourDestinationMgra = tour.getDestinationMGRA(); + int tripOriginMgra = trip.getOriginMgra(); + int tripDestinationMgra = trip.getDestinationMgra(); + + int tripOriginTaz = trip.getOriginTAZ(); + int tripDestinationTaz = trip.getDestinationTAZ(); + + dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, + tour.getDebugChoiceModels()); + + dmu.setTourDepartPeriod(tour.getDepartTime()); + dmu.setTourArrivePeriod(tour.getArriveTime()); + dmu.setTripPeriod(trip.getPeriod()); + + dmu.setWorkTimeFactor((float)tour.getWorkTimeFactor()); + dmu.setNonWorkTimeFactor((float)tour.getNonWorkTimeFactor()); + + // set trip mc dmu values for transit logsum (gets replaced below by uec values) + double c_ivt = -0.03; + double c_cost = - 0.0003; + + // Solve trip mode level utilities + mcDmuObject.setIvtCoeff(c_ivt); + mcDmuObject.setCostCoeff(c_cost); + double walkTransitLogsum = -999.0; + + logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); + dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); + + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, tripOriginMgra, tripDestinationMgra, trip.getPeriod(), tour.getDebugChoiceModels()); + walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + dmu.setWalkTransitLogsum(walkTransitLogsum); + + if (tour.getPurpose() == modelStructure.WORK) dmu.setWorkTour(1); + else dmu.setWorkTour(0); + + dmu.setOutboundStops(tour.getNumberInboundStops()); + dmu.setReturnStops(tour.getNumberInboundStops()); + + if (trip.isFirstTrip()) dmu.setFirstTrip(1); + else dmu.setFirstTrip(0); + + if (trip.isLastTrip()) dmu.setLastTrip(1); + else dmu.setLastTrip(0); + + if (tour.getTourMode() == modelStructure.DRIVEALONE) dmu.setTourModeIsDA(1); + else dmu.setTourModeIsDA(0); + + if (tour.getTourMode() == modelStructure.SHARED2) dmu.setTourModeIsS2(1); + else dmu.setTourModeIsS2(0); + + if (tour.getTourMode() == modelStructure.SHARED3) dmu.setTourModeIsS3(1); + else dmu.setTourModeIsS3(0); + + if (tour.getTourMode() == modelStructure.WALK) dmu.setTourModeIsWalk(1); + else dmu.setTourModeIsWalk(0); + + if (tour.isSentriAvailable()) dmu.setTourCrossingIsSentri(1); + else dmu.setTourCrossingIsSentri(0); + + if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); + else dmu.setTripOrigIsTourDest(0); + + if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); + else dmu.setTripDestIsTourDest(0); + + dmu.setHourlyParkingCostTourDest((float) lsWgtAvgCostH[tourDestinationMgra]); + dmu.setDailyParkingCostTourDest((float) lsWgtAvgCostD[tourDestinationMgra]); + dmu.setMonthlyParkingCostTourDest((float) lsWgtAvgCostM[tourDestinationMgra]); + dmu.setHourlyParkingCostTripOrig((float) lsWgtAvgCostH[tripOriginMgra]); + dmu.setHourlyParkingCostTripDest((float) lsWgtAvgCostH[tripDestinationMgra]); + + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(trip.getOriginMgra()); + + double rnum = tour.getRandom(); + float waitTimeSingleTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + float waitTimeSharedTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + float waitTimeTaxi = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + dmu.setWaitTimeSingleTNC(waitTimeSingleTNC); + dmu.setWaitTimeSharedTNC(waitTimeSharedTNC); + dmu.setWaitTimeTaxi(waitTimeTaxi); + + + } + + public McLogsumsCalculator getMcLogsumsCalculator(){ + return logsumHelper; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java new file mode 100644 index 0000000..3aca0c5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java @@ -0,0 +1,704 @@ +package org.sandag.abm.crossborder; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class CrossBorderTripTables +{ + + private static Logger logger = Logger.getLogger("tripTables"); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TableDataSet tripData; + + // Some parameters + private int[] modeIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // total + // modes, + // returns + // 0=auto + // modes, + // 1=non-motor, + // 2=transit, + // 3= + // other + private int[] matrixIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // modes, + // returns + // the + // element + // of + // the + // matrix + // array + // to + // store + // value + + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private HashMap rbMap; + + // matrices are indexed by modes, vot bins, submodes + private Matrix[][][] matrix; + + private ResourceBundle rb; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + + private float averageOcc3Plus = 3.5f; + private float sampleRate = 1; + private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; + private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; + private float valueOfTimeThresholdLow = 0; + private float valueOfTimeThresholdMed = 0; + //value of time bins by mode group + int[] votBins = {3,1,1,1}; + + public int numSkimSets; + + + /** + * @return the sampleRate + */ + public float getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(float sampleRate) + { + this.sampleRate = sampleRate; + } + + private MatrixDataServerRmi ms; + + public CrossBorderTripTables(HashMap rbMap) + { + + this.rbMap = rbMap; + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTripModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + } else if (modelStructure.getTripModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + } else if (modelStructure.getTripModeIsWalkTransit(i) + || modelStructure.getTripModeIsPnrTransit(i) + || modelStructure.getTripModeIsKnrTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + } + } + //value of time thresholds + valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); + valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); + + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][][]; + for (int i = 0; i < numberOfModes; ++i) + { + + String modeName; + + matrix[i] = new Matrix[votBins[i]][]; + + for(int j = 0; j< votBins[i];++j){ + if (i == 0) + { + matrix[i][j] = new Matrix[autoModes]; + for (int k = 0; k < autoModes; ++k) + { + modeName = modelStructure.getModeName(k + 1); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 1) + { + matrix[i][j] = new Matrix[nmotModes]; + for (int k = 0; k < nmotModes; ++k) + { + modeName = modelStructure.getModeName(k + 1 + autoModes); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 2) + { + matrix[i][j] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l1) + votBin = getValueOfTimeBin(valueOfTime); + + if (mode == 0) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); + } else if (mode == 1) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } else if (mode == 2) + { + + if (boardTap == 0 || alightTap == 0) continue; + + //store transit trips in matrices + mat = (matrixIndex[tripMode]*numSkimSets)+set; + float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); + matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); + + // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) + if (modelStructure.getTourModeIsDriveTransit(tripMode)) + { + + boolean inbound = tripData.getBooleanValueAt(i, "inbound"); + + // add the tNCVehicle trip portion to the trip table + if (!inbound) + { // from origin to lot (boarding tap) + int PNRTAZ = tapManager.getTazForTap(boardTap); + value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); + matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); + + } else + { // from lot (alighting tap) to destination + int PNRTAZ = tapManager.getTazForTap(alightTap); + value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); + matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); + } + + } + } else + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } + + //logger.info("End creating trip tables for period " + timePeriod); + } + } + + /** + * Return the value of time bin 0 through 2 based on the thresholds provided in the property map + * @param valueOfTime + * @return value of time bin 0 through 2 + */ + public int getValueOfTimeBin(float valueOfTime){ + + if(valueOfTime1) + end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; + else + end[i][j] = "_" + per + ".omx"; + } + } + for (int i = 0; i < 4; ++i){ + for(int j = 0; j < votBins[i];++j){ + try + { + //Delete the file if it exists + File f = new File(fileName[i]+end[i][j]); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); + f.delete(); + } + + if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); + else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); + } catch (Exception e) + { + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName[i] +end[i][j] + ", for mode index = " + i, e); + throw new RuntimeException(); + } + } + } + + } + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + /** + * Start matrix server + * + * @param serverAddress + * @param serverPort + * @param mt + * @return + */ + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @param args + */ + public static void main(String[] args) + { + + HashMap pMap; + String propertiesFile = null; + + logger.info(String.format( + "SANDAG Cross-Border Model Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + CrossBorderTripTables tripTables = new CrossBorderTripTables(pMap); + float sampleRate = 1.0f; + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + + logger.info("Crossborder Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + tripTables.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + tripTables.ms = matrixServer; + } else + { + tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + tripTables.ms.testRemote("CrossBorderTripTables"); + + // mdm = MatrixDataManager.getInstance(); + // mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + tripTables.createTripTables(mt); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java new file mode 100644 index 0000000..fa50023 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java @@ -0,0 +1,204 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class AtWorkSubtourFrequencyDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(AtWorkSubtourFrequencyDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Person person; + protected Tour tour; + protected IndexValues dmuIndex; + + protected double nmEatOutAccessibillity; + + protected ModelStructure modelStructure; + + public AtWorkSubtourFrequencyDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + dmuIndex = new IndexValues(); + } + + public Household getHouseholdObject() + { + return hh; + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setPersonObject(Person persObject) + { + person = persObject; + } + + public void setTourObject(Tour tourObject) + { + tour = tourObject; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug INMTF UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return household income category + */ + public int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + /** + * @return person type category index + */ + public int getPersonType() + { + return person.getPersonTypeNumber(); + } + + /** + * @return person type category index + */ + public int getFemale() + { + if (person.getPersonIsFemale() == 1) return 1; + else return 0; + } + + /** + * @return number of driving age people in household + */ + public int getDrivers() + { + return hh.getDrivers(); + } + + /** + * @return number of people of preschool person type in household + */ + public int getNumPreschoolChildren() + { + return hh.getNumPreschool(); + } + + /** + * @return number of individual non-mandatory eat-out tours for the person. + */ + public int getNumIndivEatOutTours() + { + int numTours = 0; + ArrayList tourList = person.getListOfIndividualNonMandatoryTours(); + if (tourList != null) + { + for (Tour t : tourList) + { + String tourPurpose = t.getTourPurpose(); + if (tourPurpose.equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) + { + numTours++; + } + } + } + return numTours; + } + + /** + * @return total mandatory and non-mandatory tours for the person. + */ + public int getNumTotalTours() + { + int numTours = 0; + + ArrayList wTourList = person.getListOfWorkTours(); + if (wTourList != null) numTours += wTourList.size(); + + ArrayList sTourList = person.getListOfSchoolTours(); + if (sTourList != null) numTours += sTourList.size(); + + ArrayList nmTourList = person.getListOfIndividualNonMandatoryTours(); + if (nmTourList != null) numTours += nmTourList.size(); + + return numTours; + } + + public double getNmEatOutAccessibilityWorkplace() + { + return nmEatOutAccessibillity; + } + + /** + * set the value of the non-mandatory eat out accessibility for this + * decision maker + */ + public void setNmEatOutAccessibilityWorkplace(double nmEatOutAccessibillity) + { + this.nmEatOutAccessibillity = nmEatOutAccessibillity; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java new file mode 100644 index 0000000..1708b43 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java @@ -0,0 +1,267 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class AutoOwnershipChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(AutoOwnershipChoiceDMU.class); + + protected HashMap methodIndexMap; + + private Household hh; + private IndexValues dmuIndex; + + private boolean useAccessibility = false; + + private double workAutoDependency = 0.0; + private double schoolAutoDependency = 0.0; + private double workAutoTime = 0.0; + + private double workersRailProportion = 0.0; + private double studentsRailProportion = 0.0; + + private double homeTazAutoAccessibility = 0.0; + private double homeTazTransitAccessibility = 0.0; + private double homeTazNonMotorizedAccessibility = 0.0; + private double homeTazMaasAccessibility = 0.0; + + public AutoOwnershipChoiceDMU() + { + dmuIndex = new IndexValues(); + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug AO UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public Household getHouseholdObject() + { + return hh; + } + + public int getGq() + { + return hh.getIsGroupQuarters(); + } + + public int getDrivers() + { + return hh.getDrivers(); + } + + public int getNumFtWorkers() + { + return hh.getNumFtWorkers(); + } + + public int getNumPtWorkers() + { + return hh.getNumPtWorkers(); + } + + public int getNumPersons18to24() + { + return hh.getNumPersons18to24(); + } + + public int getNumPersons18to35(){ + return hh.getNumPersons18to35(); + } + + public int getNumPersons6to15() + { + return hh.getNumPersons6to15(); + } + + public int getNumPersons65Plus() + { + return (hh.getNumPersons65to79()+hh.getNumPersons80plus()); + } + + public int getNumPersons80plus() + { + return hh.getNumPersons80plus(); + } + + public int getNumPersons65to79() + { + return hh.getNumPersons65to79(); + } + + public int getHhIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + public int getNumHighSchoolGraduates() + { + return hh.getNumHighSchoolGraduates(); + } + + public int getDetachedDwellingType() + { + return hh.getHhBldgsz(); + } + + public double getUseAccessibilities() + { + return useAccessibility ? 1 : 0; + } + + public double getHomeTazAutoAccessibility() + { + return homeTazAutoAccessibility; + } + + public double getHomeTazTransitAccessibility() + { + return homeTazTransitAccessibility; + } + + public double getHomeTazMaasAccessibility() + { + return homeTazMaasAccessibility; + } + + public double getHomeTazNonMotorizedAccessibility() + { + return homeTazNonMotorizedAccessibility; + } + + public double getWorkAutoDependency() + { + return workAutoDependency; + } + + public double getSchoolAutoDependency() + { + return schoolAutoDependency; + } + + public double getWorkAutoTime() { + return workAutoTime; + } + + public void setWorkAutoTime(double workAutoTime) { + this.workAutoTime = workAutoTime; + } + + public double getWorkersRailProportion() + { + return workersRailProportion; + } + + public double getStudentsRailProportion() + { + return studentsRailProportion; + } + + public void setUseAccessibilities(boolean flag) + { + useAccessibility = flag; + } + + public void setHomeTazAutoAccessibility(double acc) + { + homeTazAutoAccessibility = acc; + } + + public void setHomeTazTransitAccessibility(double acc) + { + homeTazTransitAccessibility = acc; + } + + public void setHomeTazMaasAccessibility(double acc) + { + homeTazMaasAccessibility = acc; + } + + public void setHomeTazNonMotorizedAccessibility(double acc) + { + homeTazNonMotorizedAccessibility = acc; + } + + public void setWorkAutoDependency(double value) + { + workAutoDependency = value; + } + + public void setSchoolAutoDependency(double value) + { + schoolAutoDependency = value; + } + + public void setWorkersRailProportion(double proportion) + { + workersRailProportion = proportion; + } + + public void setStudentsRailProportion(double proportion) + { + studentsRailProportion = proportion; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java new file mode 100644 index 0000000..7105499 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java @@ -0,0 +1,259 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedReader; +import java.io.FileReader; +import java.io.IOException; +import java.io.ObjectStreamException; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; + +import org.apache.log4j.Level; +import org.apache.log4j.LogManager; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.util.ResourceUtil; + +/** + * The {@code BikeLogsum} class holds bike logsums for use in the SANDAG model. This class is intended to be used as a singleton, and so the only way to + * access it is via the {@code getBikeLogsum} static method. It is constructed on demand and safe for concurrent access. + *

+ * Internally, the logsums are held in a mapping using node-pairs as keys. The taz and mgra pairs are held in the same mapping, with the tazs multiplied + * by -1 to avoid conflicts. To ensure good performance when building the object, a good guess as to the number of node pairs (maz pairs plus taz pairs) + * can be provided (a default value of 26 million will be used otherwise) via the {@code BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY} property. + */ +public class BikeLogsum implements SegmentedSparseMatrix,Serializable { + private static final long serialVersionUID = 660793106399818667L; + private static Logger logger = Logger.getLogger(BikeLogsum.class); + + public static final String BIKE_LOGSUM_OUTPUT_PROPERTY = "active.output.bike"; + public static final String BIKE_LOGSUM_MGRA_FILE_PROPERTY = "active.logsum.matrix.file.bike.mgra"; + public static final String BIKE_LOGSUM_TAZ_FILE_PROPERTY = "active.logsum.matrix.file.bike.taz"; + public static final String BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY = "active.logsum.matrix.node.pair.count"; + /** + * The default logsum node pair count. + */ + public static final int DEFAULT_BIKE_LOGSUM_NODE_PAIR_COUNT = 26_000_000; //testing found 18_880_631, so this should be good enough to start + + + private Map logsum; + private int[] mgraIndex; + + private static volatile BikeLogsum instance = null; + + /** + * Get the {@code BikeLogsum} instance. + * + * @param rbMap + * The model property mapping. + * + * @return the {@code BikeLogsum} instance. + */ + public static BikeLogsum getBikeLogsum(Map rbMap) { + if (instance == null) { + synchronized (BikeLogsum.class) { + if (instance == null) { //check again to see if we waited for another thread to do the initialization already + int nodePairCount = rbMap.containsKey(BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY) ? + Integer.parseInt(rbMap.get(BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY)) : DEFAULT_BIKE_LOGSUM_NODE_PAIR_COUNT; + String mgraFile = Paths.get(rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY),rbMap.get(MgraDataManager.PROPERTIES_MGRA_DATA_FILE)).toString(); + String tazLogsumFile = Paths.get(rbMap.get(BIKE_LOGSUM_OUTPUT_PROPERTY),rbMap.get(BIKE_LOGSUM_TAZ_FILE_PROPERTY)).toString(); + String mgraLogsumFile = Paths.get(rbMap.get(BIKE_LOGSUM_OUTPUT_PROPERTY),rbMap.get(BIKE_LOGSUM_MGRA_FILE_PROPERTY)).toString(); + instance = new BikeLogsum(tazLogsumFile,mgraLogsumFile,nodePairCount,mgraFile); + } + } + } + return instance; + } + + private BikeLogsum(String tazLogsumFile, String mgraLogsumFile, int nodePairCount, String mgraFile) { + logsum = new HashMap(nodePairCount,1.01f); //capacity of nodepairs, plus a little buffer just in case + Map> tazMgraMapping = loadTazMgraMapping(mgraFile); + mgraIndex = buildMgraIndex(tazMgraMapping); + loadLogsum(tazLogsumFile,true); + loadLogsum(mgraLogsumFile,false); + } + + private int[] buildMgraIndex(Map> tazMgraMapping) { + int maxMgra = 0; + for (Set mgras : tazMgraMapping.values()) + for (int mgra : mgras) + if (maxMgra < mgra) + maxMgra = mgra; + + int[] mgraIndex = new int[maxMgra+1]; + for (int taz : tazMgraMapping.keySet()) + for (int mgra : tazMgraMapping.get(taz)) + mgraIndex[mgra] = -1*taz; + + return mgraIndex; + } + + private Map> loadTazMgraMapping(String mgraFile) { + Map> tazMgraMapping = new HashMap<>(); + boolean first = true; + String mgraColumnName = MgraDataManager.MGRA_FIELD_NAME.toLowerCase(); + String tazColumnName = MgraDataManager.MGRA_TAZ_FIELD_NAME.toLowerCase(); + int mgraColumn = -1; + int tazColumn = -1; + try (BufferedReader reader = new BufferedReader(new FileReader(mgraFile))) { + String line; + while ((line = reader.readLine()) != null) { + String[] lineData = line.trim().split(","); + if (first) { + for (int i = 0; i < lineData.length; i++) { + String column = lineData[i].toLowerCase(); + if (column.equals(mgraColumnName)) + mgraColumn = i; + if (column.equals(tazColumnName)) + tazColumn = i; + } + first = false; + continue; + } + if (lineData.length < 2) + continue; + int mgra = Integer.parseInt(lineData[mgraColumn]); + int taz = Integer.parseInt(lineData[tazColumn]); + if (!tazMgraMapping.containsKey(taz)) + tazMgraMapping.put(taz,new HashSet()); + tazMgraMapping.get(taz).add(mgra); + } + + } catch (IOException e) { + throw new RuntimeException(e); + } + return tazMgraMapping; + } + + private void loadLogsum(String logsumFile, boolean taz) { + logger.info("Processing bike logsum from " + logsumFile); + int counter = 0; + long startTime = System.currentTimeMillis(); + + int segmentWidth = BikeLogsumSegment.segmentWidth(); + try (BufferedReader reader = new BufferedReader(new FileReader(logsumFile))) { + int logsumIndex = -1; + int timeIndex = -1; + boolean first = true; + + String line; + while ((line = reader.readLine()) != null) { + String[] lineData = line.trim().split(","); + for (int i = 0; i < lineData.length; i++) + lineData[i] = lineData[i].trim(); + if (first) { + for (int i = 2; i < lineData.length; i++) { //first two are for row and column + String columnName = lineData[i].toLowerCase(); + if (columnName.contains("logsum")) + logsumIndex = i; + if (columnName.contains("time")) + timeIndex = i; + } + first = false; + continue; + } + if (++counter % 100_000 == 0) + logger.debug("Finished processing " + counter + " node pairs (logsum lookup size: " + logsum.size() + ")"); + //if we ever bring back segmented logsums, then this will be a bit more complicated + // the basic idea is all logsums first, then times (in same order) so lookups are straightforward + // without having to replicate the hashmap, which is a big data structure + double[] data = new double[] {Double.parseDouble(lineData[logsumIndex]),Double.parseDouble(lineData[timeIndex])}; + + int fromZone = Integer.parseInt(lineData[0]); + int toZone = Integer.parseInt(lineData[1]); + int indexFactor = taz ? -1 : 1; + MatrixLookup ml = new MatrixLookup(indexFactor*fromZone,indexFactor*toZone); + logsum.put(ml,data); + + } + } catch (IOException e) { + throw new RuntimeException(e); + } + logger.info("Finished processing " + counter + " node pairs (logsum lookup size: " + logsum.size() + ") in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes"); + } + + private double[] getLogsums(int rowId, int columnId) { + double[] logsums = logsum.get(new MatrixLookup(rowId,columnId)); + if (logsums == null) + logsums = logsum.get(new MatrixLookup(mgraIndex[rowId],mgraIndex[columnId])); + return logsums; + } + + @Override + public double getValue(BikeLogsumSegment segment, int rowId, int columnId) { + double[] logsums = getLogsums(rowId,columnId); + return logsums == null ? -999 : logsums[segment.getSegmentId()]; + } + + public double getLogsum(BikeLogsumSegment segment, int rowId, int columnId) { + return getValue(segment,rowId,columnId); + } + + public double getTime(BikeLogsumSegment segment, int rowId, int columnId) { + double[] logsums = getLogsums(rowId,columnId); + return logsums == null ? Double.POSITIVE_INFINITY : logsums[segment.getSegmentId()+BikeLogsumSegment.segmentWidth()]; + } + + private static class MatrixLookup implements Serializable { + private static final long serialVersionUID = -5048040835197200584L; + private final int row; + private final int column; + + private MatrixLookup(int row, int column) { + this.row = row; + this.column = column; + } + + public boolean equals(Object o) { + if ((o == null) || (!(o instanceof MatrixLookup))) + return false; + MatrixLookup ml = (MatrixLookup) o; + return (row == ml.row) && (column == ml.column); + } + + public int hashCode() { + return row + 37*column; + } + } + + private void writeObject(java.io.ObjectOutputStream out) throws IOException { + out.writeObject(logsum); + out.writeObject(mgraIndex); + } + + @SuppressWarnings("unchecked") + private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { + logsum = (Map) in.readObject(); + mgraIndex = (int[]) in.readObject(); + synchronized (BikeLogsum.class) { + //ensures singleton - readResolve will ensure all get this single value + //we need to allow the above reading of fields, though, so that deserialization is aligned correctly + if (instance == null) + instance = this; + } + } + + private Object readResolve() throws ObjectStreamException { + return instance; //ensures singelton + } + + public static void main(String ... args) { + org.apache.log4j.BasicConfigurator.configure(); + LogManager.getRootLogger().setLevel(Level.INFO); + logger.info("usage: org.sandag.abm.ctramp.BikeLogsum properties origin_mgra dest_mgra"); + Map properties = ResourceUtil.getResourceBundleAsHashMap(args[0]); + int originMgra = Integer.parseInt(args[1]); + int destMgra = Integer.parseInt(args[2]); + BikeLogsum bls = BikeLogsum.getBikeLogsum(properties); + + BikeLogsumSegment defaultSegment = new BikeLogsumSegment(true,true,true); + double logsum = bls.getLogsum(new BikeLogsumSegment(true,true,true),originMgra,destMgra); + double time = bls.getTime(new BikeLogsumSegment(true,true,true),originMgra,destMgra); + logger.info(String.format("omgra: %s, dmgra: %s, logsum: %s, time: %s",originMgra,destMgra,logsum,time)); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java new file mode 100644 index 0000000..9c2efc0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java @@ -0,0 +1,117 @@ +package org.sandag.abm.ctramp; + +import java.io.IOException; +import java.io.Serializable; + +/** + * The {@code BikeLogsumSegment} class provides a segmentation for bicycle logsums used in the SANDAG model. The segmentation is currently + * based on three variables: + *

+ *

    + *
  • sex (male/female)
  • + *
  • tour type (mandatory/non-mandatory)
  • + *
  • trip direction (outbound/inbound) - only used if tour type is mandatory
  • + *
+ *

+ * This segmentation maps the 6 possible unique combinations into the integer indices 0 through 5, which can then be used as a lookup to a + * zero-based array or list data structure. + */ +public class BikeLogsumSegment implements Serializable { + private static final long serialVersionUID = -8429882786837391491L; + + private int segmentId; + + /** + * Constructor specifying the segment parameters. + * + * @param isFemale + * {@code true} if the sex is female. + * + * @param mandatory + * {@code true} if the tour type is mandatory. + * + * @param inbound + * {@code true} if the trip direction is inbound. + */ + public BikeLogsumSegment(boolean isFemale, boolean mandatory, boolean inbound) { + segmentId = formSegmentId(isFemale,mandatory,inbound); + } + + private int formSegmentId(boolean isFemale, boolean mandatory, boolean inbound) { + return 0; //only one segment now, but leaving in this structure in case it is needed in the future + } + + /** + * Get the segment index (0-5) for this segment. + * + * @return return this segment's index id. + */ + public int getSegmentId() { + return segmentId; + } + + /** + * Get the total number of segments available from this class. All segment index ids will be less than this value. + * + * @return the number of unique segments provided by this class. + */ + public static int segmentWidth() { + return 1; + } + + /** + * Get the segments corresponding to the tour-specific segment values. This will give all permutations which can then be combined to form a + * tour-level (as opposed to trip-level) composite logsum. + * + * @param isFemale + * {@code true} if the sex is female. + * + * @param mandatory + * {@code true} if the tour type is mandatory. + * + * @return an array of all possible segments with {@code isFemale} and {@code mandatory}. + */ + public static BikeLogsumSegment[] getTourSegments(boolean isFemale, boolean mandatory) { + return new BikeLogsumSegment[] {new BikeLogsumSegment(isFemale,mandatory,true), + new BikeLogsumSegment(isFemale,mandatory,false)}; + } + + /** + * Get the segments corresponding to the tour-specific, person-inspecific segment values. This will give all permutations which can then be + * combined to form a tour- and household-level (as opposed to trip-level) composite logsum. + * + * @param mandatory + * {@code true} if the tour type is mandatory. + * + * @return an array of all possible segments with {@code mandatory}. + */ + public static BikeLogsumSegment[] getTourSegments(boolean mandatory) { + return new BikeLogsumSegment[] {new BikeLogsumSegment(true,mandatory,true), + new BikeLogsumSegment(true,mandatory,false), + new BikeLogsumSegment(false,mandatory,true), + new BikeLogsumSegment(false,mandatory,false)}; + } + + @Override + public String toString() { + StringBuilder sb = new StringBuilder(" 0; + boolean isMandatory = (segmentId & 2) > 0; + boolean inbound = (segmentId & 4) > 0; + sb.append(isFemale ? "female" : "male"); + sb.append(",").append(isMandatory ? "mandatory" : "non-mandatory"); + if (isMandatory) + sb.append(",").append(inbound ? "inbound" : "outbound"); + sb.append(">"); + return sb.toString(); + } + + private void writeObject(java.io.ObjectOutputStream out) throws IOException { + out.writeInt(segmentId); + } + + private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { + segmentId = in.readInt(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java new file mode 100644 index 0000000..7084678 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java @@ -0,0 +1,55 @@ +package org.sandag.abm.ctramp; + +import java.sql.Connection; +import java.sql.DriverManager; +import java.sql.SQLException; + +public final class ConnectionHelper +{ + + private String url; + private static ConnectionHelper instance; + + private ConnectionHelper(String fileName) + { + try + { + Class.forName("org.sqlite.JDBC"); + // url = "jdbc:sqlite:/c:/jim/projects/baylanta/data/status.db"; + url = "jdbc:sqlite:/" + fileName; + } catch (Exception e) + { + e.printStackTrace(); + } + } + + public static Connection getConnection(String fileName) throws SQLException + { + if (instance == null) + { + instance = new ConnectionHelper(fileName); + } + try + { + return DriverManager.getConnection(instance.url); + } catch (SQLException e) + { + throw e; + } + } + + public static void close(Connection connection) + { + try + { + if (connection != null) + { + connection.close(); + } + } catch (SQLException e) + { + e.printStackTrace(); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java new file mode 100644 index 0000000..93d988c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java @@ -0,0 +1,442 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * Decision making unit object for the Coordinated Daily Activity Pattern Model. + * This DMU contains all the getters specified in the UEC, i.e. all the "@" + * variables. + * + * @author D. Ory + * + */ +public class CoordinatedDailyActivityPatternDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(CoordinatedDailyActivityPatternDMU.class); + + protected HashMap methodIndexMap; + + protected IndexValues dmuIndex; + + protected Household householdObject; + protected Person personA, personB, personC; + protected double workModeChoiceLogsumA; + protected double schoolModeChoiceLogsumA; + protected double retailAccessibility; + + protected double workAccessForMandatoryDap; + + protected int numAdultsWithNonMandatoryDap; + protected int numAdultsWithMandatoryDap; + protected int numKidsWithNonMandatoryDap; + protected int numKidsWithMandatoryDap; + protected int allAdultsAtHome; + + public CoordinatedDailyActivityPatternDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int zoneId) + { + dmuIndex.setZoneIndex(zoneId); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (householdObject.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug CDAP UEC"); + } + + } + + public IndexValues getIndexValues() + { + return dmuIndex; + } + + public void setHousehold(Household passedInHouseholdObject) + { + + householdObject = passedInHouseholdObject; + + // set the household index + dmuIndex.setHHIndex(passedInHouseholdObject.getHhId()); + + // set the home zone as the zone index + dmuIndex.setZoneIndex(passedInHouseholdObject.getHhMgra()); + } + + public void setPersonA(Person passedInPersonA) + { + this.personA = passedInPersonA; + } + + public void setPersonB(Person passedInPersonB) + { + this.personB = passedInPersonB; + } + + public void setPersonC(Person passedInPersonC) + { + this.personC = passedInPersonC; + } + + // full-time worker + public int getFullTimeWorkerA() + { + return (personA.getPersonTypeIsFullTimeWorker()); + } + + public int getFullTimeWorkerB() + { + return (personB.getPersonTypeIsFullTimeWorker()); + } + + public int getFullTimeWorkerC() + { + return (personC.getPersonTypeIsFullTimeWorker()); + } + + // part-time worker + public int getPartTimeWorkerA() + { + return (personA.getPersonTypeIsPartTimeWorker()); + } + + public int getPartTimeWorkerB() + { + return (personB.getPersonTypeIsPartTimeWorker()); + } + + public int getPartTimeWorkerC() + { + return (personC.getPersonTypeIsPartTimeWorker()); + } + + // university student + public int getUniversityStudentA() + { + return (personA.getPersonIsUniversityStudent()); + } + + public int getUniversityStudentB() + { + return (personB.getPersonIsUniversityStudent()); + } + + public int getUniversityStudentC() + { + return (personC.getPersonIsUniversityStudent()); + } + + // non-working adult + public int getNonWorkingAdultA() + { + return (personA.getPersonIsNonWorkingAdultUnder65()); + } + + public int getNonWorkingAdultB() + { + return (personB.getPersonIsNonWorkingAdultUnder65()); + } + + public int getNonWorkingAdultC() + { + return (personC.getPersonIsNonWorkingAdultUnder65()); + } + + // retired + public int getRetiredA() + { + return (personA.getPersonIsNonWorkingAdultOver65()); + } + + public int getRetiredB() + { + return (personB.getPersonIsNonWorkingAdultOver65()); + } + + public int getRetiredC() + { + return (personC.getPersonIsNonWorkingAdultOver65()); + } + + // driving age school child + public int getDrivingAgeSchoolChildA() + { + return (personA.getPersonIsStudentDriving()); + } + + public int getDrivingAgeSchoolChildB() + { + return (personB.getPersonIsStudentDriving()); + } + + public int getDrivingAgeSchoolChildC() + { + return (personC.getPersonIsStudentDriving()); + } + + // non-driving school-age child + public int getPreDrivingAgeSchoolChildA() + { + return (personA.getPersonIsStudentNonDriving()); + } + + public int getPreDrivingAgeSchoolChildB() + { + return (personB.getPersonIsStudentNonDriving()); + } + + public int getPreDrivingAgeSchoolChildC() + { + return (personC.getPersonIsStudentNonDriving()); + } + + // pre-school child + public int getPreSchoolChildA() + { + return (personA.getPersonIsPreschoolChild()); + } + + public int getPreSchoolChildB() + { + return (personB.getPersonIsPreschoolChild()); + } + + public int getPreSchoolChildC() + { + return (personC.getPersonIsPreschoolChild()); + } + + // age + public int getAgeA() + { + return (personA.getAge()); + } + + // female + public int getFemaleA() + { + return (personA.getPersonIsFemale()); + } + + // household more cars than workers + public int getMoreCarsThanWorkers() + { + + int workers = householdObject.getWorkers(); + int autos = householdObject.getAutosOwned(); + + if (autos > workers) return 1; + return 0; + + } + + // household fewer cars than workers + public int getFewerCarsThanWorkers() + { + + int workers = householdObject.getWorkers(); + int autos = householdObject.getAutosOwned(); + + if (autos < workers) return 1; + return 0; + + } + + // household with zero cars + public int getZeroCars() + { + int autos = householdObject.getAutosOwned(); + if (autos == 0) return 1; + return 0; + + } + + // household income + public int getHHIncomeInDollars() + { + return householdObject.getIncomeInDollars(); + } + + public int getUsualWorkLocationIsHomeA() + { + if (personA.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 1; + else return 0; + } + + public int getNoUsualWorkLocationA() + { + if (personA.getWorkLocation() > 0 + && personA.getWorkLocation() != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 0; + else return 1; + } + + // no usual school location is 1 if person is a student, location is home + // mgra, + // and distance to school is 0. + public int getNoUsualSchoolLocationA() + { + if (personA.getPersonIsStudent() == 1 + && personA.getUsualSchoolLocation() == personA.getHouseholdObject().getHhMgra() + && personA.getSchoolLocationDistance() == 0) return 0; + else return 1; + } + + public int getHhSize() + { + + int hhSize = Math.min(HouseholdCoordinatedDailyActivityPatternModel.MAX_MODEL_HH_SIZE, + householdObject.getSize()); + + return (hhSize); + } + + public int getHhDetach() + { + return householdObject.getHhBldgsz(); + } + + public int getNumAdultsWithNonMandatoryDap() + { + return numAdultsWithNonMandatoryDap; + } + + public int getNumAdultsWithMandatoryDap() + { + return numAdultsWithMandatoryDap; + } + + public int getNumKidsWithNonMandatoryDap() + { + return numKidsWithNonMandatoryDap; + } + + public int getNumKidsWithMandatoryDap() + { + return numKidsWithMandatoryDap; + } + + public void setNumAdultsWithNonMandatoryDap(int value) + { + numAdultsWithNonMandatoryDap = value; + } + + public void setNumAdultsWithMandatoryDap(int value) + { + numAdultsWithMandatoryDap = value; + } + + public void setNumKidsWithNonMandatoryDap(int value) + { + numKidsWithNonMandatoryDap = value; + } + + public void setNumKidsWithMandatoryDap(int value) + { + numKidsWithMandatoryDap = value; + } + + public int getAllAdultsAtHome() + { + return allAdultsAtHome; + } + + public void setAllAdultsAtHome(int value) + { + allAdultsAtHome = value; + } + + public void setWorkAccessForMandatoryDap(double logsum) + { + workAccessForMandatoryDap = logsum; + } + + public double getWorkAccessForMandatoryDap() + { + return workAccessForMandatoryDap; + } + + public void setWorkLocationModeChoiceLogsumA(double logsum) + { + workModeChoiceLogsumA = logsum; + } + + public double getWorkLocationModeChoiceLogsumA() + { + return workModeChoiceLogsumA; + } + + public void setSchoolLocationModeChoiceLogsumA(double logsum) + { + schoolModeChoiceLogsumA = logsum; + } + + public double getSchoolLocationModeChoiceLogsumA() + { + return schoolModeChoiceLogsumA; + } + + public void setRetailAccessibility(double logsum) + { + retailAccessibility = logsum; + } + + public double getRetailAccessibility() + { + return retailAccessibility; + } + + public int getTelecommuteFrequencyA() { + return personA.getTelecommuteChoice(); + } + + public int getTelecommuteFrequencyB() { + return personB.getTelecommuteChoice(); + } + + public int getTelecommuteFrequencyC() { + return personC.getTelecommuteChoice(); + } + + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java new file mode 100644 index 0000000..bfb4032 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java @@ -0,0 +1,2481 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.NotSerializableException; +import java.io.PrintWriter; +import java.io.Serializable; +import java.rmi.RemoteException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.MissingResourceException; +import java.util.ResourceBundle; +import java.util.Set; +import java.util.TreeMap; +import org.apache.log4j.Logger; +import org.jppf.client.JPPFClient; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.StoredUtilityData; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.DataFile; +import com.pb.common.datafile.DataReader; +import com.pb.common.datafile.DataWriter; +import com.pb.common.matrix.MatrixIO32BitJvm; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; +// import org.sandag.abm.accessibilities.BuildAccessibilities; +// import +// org.sandag.abm.ctramp.HouseholdDataWriter; +// import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyModel; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +// 1.0.1 - 09/21/09 - starting point for SANDAG AB model implementation - AO +// model + +public class CtrampApplication + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(CtrampApplication.class); + + public static final String VERSION = "2.0.0"; + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + + public static final String PROPERTIES_BASE_NAME = "ctramp"; + public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; + + public static final String PROPERTIES_UEC_PATH = "uec.path"; + public static final String SQLITE_DATABASE_FILENAME = "Sqlite.DatabaseFileName"; + + public static final String PROPERTIES_RUN_POPSYN = "RunModel.PopulationSynthesizer"; + public static final String PROPERTIES_RUN_PRE_AUTO_OWNERSHIP = "RunModel.PreAutoOwnership"; + public static final String PROPERTIES_RUN_WORKSCHOOL_CHOICE = "RunModel.UsualWorkAndSchoolLocationChoice"; + public static final String PROPERTIES_RUN_AUTO_OWNERSHIP = "RunModel.AutoOwnership"; + public static final String PROPERTIES_RUN_TRANSPONDER_CHOICE = "RunModel.TransponderChoice"; + public static final String PROPERTIES_RUN_FREE_PARKING_AVAILABLE = "RunModel.FreeParking"; + public static final String PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP = "RunModel.InternalExternal"; + public static final String PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN = "RunModel.CoordinatedDailyActivityPattern"; + public static final String PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ = "RunModel.IndividualMandatoryTourFrequency"; + public static final String PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR = "RunModel.MandatoryTourDepartureTimeAndDuration"; + public static final String PROPERTIES_RUN_SCHOOL_ESCORT_MODEL = "RunModel.SchoolEscortModel"; + public static final String PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE = "RunModel.MandatoryTourModeChoice"; + public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ = "RunModel.AtWorkSubTourFrequency"; + public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE = "RunModel.AtWorkSubTourLocationChoice"; + public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE = "RunModel.AtWorkSubTourModeChoice"; + public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR = "RunModel.AtWorkSubTourDepartureTimeAndDuration"; + public static final String PROPERTIES_RUN_JOINT_TOUR_FREQ = "RunModel.JointTourFrequency"; + public static final String PROPERTIES_RUN_JOINT_LOCATION_CHOICE = "RunModel.JointTourLocationChoice"; + public static final String PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE = "RunModel.JointTourModeChoice"; + public static final String PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR = "RunModel.JointTourDepartureTimeAndDuration"; + public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ = "RunModel.IndividualNonMandatoryTourFrequency"; + public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE = "RunModel.IndividualNonMandatoryTourLocationChoice"; + public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE = "RunModel.IndividualNonMandatoryTourModeChoice"; + public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR = "RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration"; + public static final String PROPERTIES_RUN_STOP_FREQUENCY = "RunModel.StopFrequency"; + public static final String PROPERTIES_RUN_STOP_LOCATION = "RunModel.StopLocation"; + + public static final String PROPERTIES_RUN_WORK_LOC_CHOICE_KEY = "uwsl.run.workLocChoice"; + public static final String PROPERTIES_RUN_SCHOOL_LOC_CHOICE_KEY = "uwsl.run.schoolLocChoice"; + public static final String PROPERTIES_WRITE_WORK_SCHOOL_LOC_RESULTS_KEY = "uwsl.write.results"; + + public static final String PROPERTIES_UEC_AUTO_OWNERSHIP = "UecFile.AutoOwnership"; + public static final String PROPERTIES_UEC_DAILY_ACTIVITY_PATTERN = "UecFile.CoordinatedDailyActivityPattern"; + public static final String PROPERTIES_UEC_INDIV_MANDATORY_TOUR_FREQ = "UecFile.IndividualMandatoryTourFrequency"; + public static final String PROPERTIES_UEC_MAND_TOUR_DEP_TIME_AND_DUR = "UecFile.TourDepartureTimeAndDuration"; + public static final String PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ = "UecFile.IndividualNonMandatoryTourFrequency"; + + public static final String PROPERTIES_CLEAR_MATRIX_MANAGER_ON_START = "RunModel.Clear.MatrixMgr.At.Start"; + + public static final String READ_ACCESSIBILITIES = "acc.read.input.file"; + + // TODO eventually move to model-specific structure object + public static final int TOUR_MODE_CHOICE_WORK_MODEL_UEC_PAGE = 1; + public static final int TOUR_MODE_CHOICE_UNIVERSITY_MODEL_UEC_PAGE = 2; + public static final int TOUR_MODE_CHOICE_HIGH_SCHOOL_MODEL_UEC_PAGE = 3; + public static final int TOUR_MODE_CHOICE_GRADE_SCHOOL_MODEL_UEC_PAGE = 4; + + // TODO eventually move to model-specific model structure object + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_MODEL_UEC_PAGE = 1; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_DEPARTURE_UEC_PAGE = 2; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_DURATION_UEC_PAGE = 3; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_ARRIVAL_UEC_PAGE = 4; + + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_MODEL_UEC_PAGE = 5; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_DEPARTURE_UEC_PAGE = 6; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_DURATION_UEC_PAGE = 7; + public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_ARRIVAL_UEC_PAGE = 8; + + public static final String PROPERTIES_SCHEDULING_NUMBER_OF_TIME_PERIODS = "Scheduling.NumberOfTimePeriods"; + public static final String PROPERTIES_SCHEDULING_FIRST_TIME_PERIOD = "Scheduling.FirstTimePeriod"; + + static final String PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER = "RunModel.RestartWithHhServer"; + + static final String PROPERTIES_HOUSEHOLD_DISK_OBJECT_FILE_NAME = "Households.disk.object.base.name"; + static final String PROPERTIES_HOUSEHOLD_DISK_OBJECT_KEY = "Read.HouseholdDiskObjectFile"; + + public static final String PROPERTIES_RESULTS_AUTO_OWNERSHIP = "Results.AutoOwnership"; + public static final String PROPERTIES_RESULTS_CDAP = "Results.CoordinatedDailyActivityPattern"; + + public static final String PROPERTIES_OUTPUT_WRITE_SWITCH = "CTRAMP.Output.WriteToDiskSwitch"; + public static final String PROPERTIES_OUTPUT_HOUSEHOLD_FILE = "CTRAMP.Output.HouseholdFile"; + public static final String PROPERTIES_OUTPUT_PERSON_FILE = "CTRAMP.Output.PersonFile"; + + public static final String PROPERTIES_WRITE_DATA_TO_FILE = "Results.WriteDataToFiles"; + public static final String PROPERTIES_WRITE_DATA_TO_DATABASE = "Results.WriteDataToDatabase"; + + public static final String PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS = "TourModeChoice.Save.UtilsAndProbs"; + + public static final String PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE = "UsualWorkLocationChoice.ShadowPrice.Input.File"; + public static final String PROPERTIES_SCHOOL_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE = "UsualSchoolLocationChoice.ShadowPrice.Input.File"; + + public static final String PROPERTIES_NUMBER_OF_GLOBAL_ITERATIONS = "Global.iterations"; + + public static final String ALT_FIELD_NAME = "a"; + public static final String START_FIELD_NAME = "depart"; + public static final String END_FIELD_NAME = "arrive"; + + private static final int NUM_WRITE_PACKETS = 1000; + + private ResourceBundle resourceBundle; + private HashMap propertyMap; + + private MatrixDataServerIf ms; + private MatrixDataManager mdm; + + private ModelStructure modelStructure; + protected String projectDirectory; + + private HashMap> cdapByHhSizeAndPattern; + private HashMap> cdapByPersonTypeAndActivity; + + private BuildAccessibilities aggAcc; + private boolean calculateLandUseAccessibilities; + + public CtrampApplication(ResourceBundle rb, HashMap rbMap, + boolean calculateLandUseAccessibilities) + { + resourceBundle = rb; + propertyMap = rbMap; + this.calculateLandUseAccessibilities = calculateLandUseAccessibilities; + } + + public void setupModels(ModelStructure modelStructure) + { + + this.modelStructure = modelStructure; + + } + + // public void runPopulationSynthesizer( SANDAGPopSyn populationSynthesizer + // ){ + // + // // run population synthesizer + // boolean runModelPopulationSynthesizer = + // ResourceUtil.getBooleanProperty(resourceBundle, PROPERTIES_RUN_POPSYN); + // if(runModelPopulationSynthesizer){ + // populationSynthesizer.run(); + // } + // + // } + + public void runModels(HouseholdDataManagerIf householdDataManager, + CtrampDmuFactoryIf dmuFactory, int globalIterationNumber, float iterationSampleRate) + { + + logger.info("Running JPPF CtrampApplication.runModels() for " + + householdDataManager.getNumHouseholds() + " households."); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(propertyMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(propertyMap, + "RunModel.MatrixServerPort"); + } catch (RuntimeException e) + { + // if no matrix server address entry is found, leave undefined -- + // it's eithe not needed or show could create an error. + } + } catch (RuntimeException e) + { + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServer matrixServer = null; + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = startMatrixServerProcess(matrixServerAddress, serverPort); + ms = matrixServer; + } else + { + ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error(String + .format("exception caught running ctramp model components -- exiting."), e); + throw new RuntimeException(e); + + } + + // get the property that indicates if the MatrixDataServer should be + // cleared. + // default is to clear the manager + boolean clearMatrixMgr = true; + try + { + clearMatrixMgr = Util.getBooleanValueFromPropertyMap(propertyMap, + PROPERTIES_CLEAR_MATRIX_MANAGER_ON_START); + } catch (RuntimeException e) + { + // catch the RuntimeExcption that's thrown if the property key + // is not found in the properties file. + // no need to anything - the boolean clearMatrixMgr was + // initialized to the default action. + } + + // if the property to clear matrices is true and a remote + // MatrixDataServer is being used, clear the matrices. + // if matrices are being read directly into the current process, no + // need to clear. + if (clearMatrixMgr && !matrixServerAddress.equalsIgnoreCase("localhost")) ms.clear(); + + // run core activity based model for the specified iteration + runModelSequence(globalIterationNumber, householdDataManager, dmuFactory); + + } + + /** + * This method maintains the sequencing of the various AB Model choice model + * components + * + * @param iteration + * is the global iteration number in the sequence of AB Model + * runs during feedback + * @param householdDataManager + * is the handle to the household object data manager + * @param dmuFactory + * is the factory object for creating DMU objects used in choice + * models + */ + private void runModelSequence(int iteration, HouseholdDataManagerIf householdDataManager, + CtrampDmuFactoryIf dmuFactory) + { + + String restartModel = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER); + boolean logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + if (restartModel == null) restartModel = "none"; + if (!restartModel.equalsIgnoreCase("none")) restartModels(householdDataManager); + + JPPFClient jppfClient = null; + + boolean runPreAutoOwnershipChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_PRE_AUTO_OWNERSHIP); + if (runPreAutoOwnershipChoiceModel) + { + + logger.info("creating Accessibilities Object for Pre-AO."); + buildNonMandatoryAccessibilities(calculateLandUseAccessibilities); + + logger.info("starting Pre-Auto Ownership Model."); + HashMap propertyMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + + householdDataManager.resetPreAoRandom(); + + HouseholdAutoOwnershipModel aoModel = new HouseholdAutoOwnershipModel(propertyMap, + dmuFactory, aggAcc.getAccessibilitiesTableObject(), null); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + for (int i = 0; i < householdArray.length; ++i) + { + + try + { + aoModel.applyModel(householdArray[i], true); + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught running pre-AO for i=%d, startIndex=%d, endIndex=%d, hhId=%d.", + i, startIndex, endIndex, householdArray[i].getHhId())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + householdDataManager.setHhArray(householdArray, startIndex); + + } + + saveAoResults(householdDataManager, projectDirectory, true); + + // clear the zonal data used in the AO UEC so a different zonal data + // file + // (MGRA data) can be used later by other UECs. + // TableDataSetManager tableDataManager = + // TableDataSetManager.getInstance(); + // tableDataManager.clearData(); + } + logger.info("flag to run pre-AO was set to: " + runPreAutoOwnershipChoiceModel); + logAoResults(householdDataManager, true); + + boolean runUsualWorkSchoolChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_WORKSCHOOL_CHOICE); + if (runUsualWorkSchoolChoiceModel) + { + + boolean runWorkLocationChoice = false; + try + { + String stringValue = resourceBundle.getString(PROPERTIES_RUN_WORK_LOC_CHOICE_KEY); + runWorkLocationChoice = stringValue.equalsIgnoreCase("true"); + } catch (MissingResourceException e) + { + // default value is true if property was not defined + runWorkLocationChoice = true; + } + logger.info("flag to run work location choice was set to: " + runWorkLocationChoice); + + boolean runSchoolLocationChoice = false; + try + { + String stringValue = resourceBundle.getString(PROPERTIES_RUN_SCHOOL_LOC_CHOICE_KEY); + runSchoolLocationChoice = stringValue.equalsIgnoreCase("true"); + } catch (MissingResourceException e) + { + // default value is true if property was not defined + runSchoolLocationChoice = true; + } + logger.info("flag to run school location choice was set to: " + runSchoolLocationChoice); + + boolean writeLocationChoiceResultsFile = false; + try + { + String stringValue = resourceBundle + .getString(PROPERTIES_WRITE_WORK_SCHOOL_LOC_RESULTS_KEY); + writeLocationChoiceResultsFile = stringValue.equalsIgnoreCase("true"); + } catch (MissingResourceException e) + { + // default value is true if property was not defined + writeLocationChoiceResultsFile = true; + } + logger.info("flag to write uwsl result was set to: " + writeLocationChoiceResultsFile); + + if (aggAcc == null) + { + logger.info("creating Accessibilities Object for UWSL."); + buildNonMandatoryAccessibilities(calculateLandUseAccessibilities); + } + + // new the usual school and location choice model object + jppfClient = new JPPFClient(); + UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( + resourceBundle, restartModel, jppfClient, modelStructure, ms, dmuFactory, + aggAcc); + + if (runWorkLocationChoice) + { + // calculate and get the array of worker size terms table - + // MGRAs by + // occupations + aggAcc.createWorkSegmentNameIndices(); + aggAcc.calculateWorkerSizeTerms(); + double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); + + // run the model + logger.info("starting usual work location choice."); + usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, + workerSizeTerms); + logger.info("finished with usual work location choice."); + } + + if (runSchoolLocationChoice) + { + aggAcc.createSchoolSegmentNameIndices(); + aggAcc.calculateSchoolSizeTerms(); + double[][] schoolFactors = aggAcc.calculateSchoolSegmentFactors(); + double[][] schoolSizeTerms = aggAcc.getSchoolSizeTerms(); + + logger.info("starting usual school location choice."); + usualWorkSchoolLocationChoiceModel.runSchoolLocationChoiceModel( + householdDataManager, schoolSizeTerms, schoolFactors); + logger.info("finished with usual school location choice."); + } + + if (writeLocationChoiceResultsFile) + { + logger.info("writing work/school location choice results file; may take a few minutes ..."); + usualWorkSchoolLocationChoiceModel.saveResults(householdDataManager, + projectDirectory, iteration); + logger.info(String + .format("finished writing work/school location choice results file.")); + } + + usualWorkSchoolLocationChoiceModel = null; + + } + logger.info("flag to run UWSL was set to: " + runUsualWorkSchoolChoiceModel); + + boolean runAutoOwnershipChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_AUTO_OWNERSHIP); + boolean runTransponderChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_TRANSPONDER_CHOICE); + boolean runFreeParkingChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_FREE_PARKING_AVAILABLE); + boolean runInternalExternalTripChoiceModel = ResourceUtil.getBooleanProperty( + resourceBundle, PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP); + boolean runCoordinatedDailyActivityPatternChoiceModel = ResourceUtil.getBooleanProperty( + resourceBundle, PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN); + boolean runMandatoryTourFreqChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ); + boolean runJointTourFreqChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_JOINT_TOUR_FREQ); + boolean runIndivNonManTourFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ); + boolean runAtWorkSubTourFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ); + boolean runStopFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_STOP_FREQUENCY); + + if (runAutoOwnershipChoiceModel || runTransponderChoiceModel || runFreeParkingChoiceModel + || runInternalExternalTripChoiceModel + || runCoordinatedDailyActivityPatternChoiceModel || runMandatoryTourFreqChoiceModel + || runIndivNonManTourFrequencyModel || runAtWorkSubTourFrequencyModel + || runStopFrequencyModel) + { + + // We're resetting the random number sequence used by pre-AO for the + // primary AO + if (runAutoOwnershipChoiceModel) householdDataManager.resetPreAoRandom(); + + if (runTransponderChoiceModel) + householdDataManager.computeTransponderChoiceTazPercentArrays(); + + logger.info("starting HouseholdChoiceModelRunner."); + HashMap propertyMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner(propertyMap, + jppfClient, restartModel, householdDataManager, ms, modelStructure, dmuFactory); + runner.runHouseholdChoiceModels(); + + if (runAutoOwnershipChoiceModel) + { + saveAoResults(householdDataManager, projectDirectory, false); + if (logResults) logAoResults(householdDataManager, false); + } + + if (runTransponderChoiceModel) + { + if (logResults) logTpResults(householdDataManager); + } + + if (runFreeParkingChoiceModel) + { + if (logResults) logFpResults(householdDataManager); + } + + if (runInternalExternalTripChoiceModel) + { + if (logResults) logIeResults(householdDataManager); + } + + if (runCoordinatedDailyActivityPatternChoiceModel) + { + saveCdapResults(householdDataManager, projectDirectory); + if (logResults) logCdapResults(householdDataManager); + } + + if (runMandatoryTourFreqChoiceModel) + { + if (logResults) logImtfResults(householdDataManager); + } + + if (runJointTourFreqChoiceModel) + { + if (logResults) logJointModelResults(householdDataManager, dmuFactory); + } + + if (runAtWorkSubTourFrequencyModel) + { + if (logResults) logAtWorkSubtourFreqResults(householdDataManager); + } + + if (runStopFrequencyModel) + { + if (logResults) logIndivStfResults(householdDataManager); + } + + } + + jppfClient.close(); + + boolean writeTextFileFlag = false; + boolean writeSqliteFlag = false; + try + { + writeTextFileFlag = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_WRITE_DATA_TO_FILE); + } catch (MissingResourceException e) + { + // if exception is caught while getting property file value, then + // boolean + // flag remains false + } + try + { + writeSqliteFlag = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_WRITE_DATA_TO_DATABASE); + } catch (MissingResourceException e) + { + // if exception is caught while getting property file value, then + // boolean + // flag remains false + } + + HouseholdDataWriter dataWriter = null; + if (writeTextFileFlag || writeSqliteFlag) + { + dataWriter = new HouseholdDataWriter(propertyMap, modelStructure, iteration); + + if (writeTextFileFlag) dataWriter.writeDataToFiles(householdDataManager); + + if (writeSqliteFlag) + { + String dbFilename = ""; + try + { + String baseDir = resourceBundle.getString(PROPERTIES_PROJECT_DIRECTORY); + dbFilename = baseDir + resourceBundle.getString(SQLITE_DATABASE_FILENAME) + "_" + + iteration; + dataWriter.writeDataToDatabase(householdDataManager, dbFilename); + } catch (MissingResourceException e) + { + // if exception is caught while getting property file value, + // then + // boolean flag remains false + } + } + } + + } + + /** + * Build the mandatory accessibilities object used by the usual work and + * school location choice models + * + * @return BuildAccessibilities object containing mandatory size term and + * logsum information private void buildMandatoryAccessibilities() { + * + * HashMap propertyMap = + * ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle); + * + * if ( aggAcc == null ) aggAcc = new BuildAccessibilities( + * propertyMap ); + * + * MatrixDataManager mdm = MatrixDataManager.getInstance(); + * mdm.setMatrixDataServerObject( ms ); + * + * aggAcc.setupBuildAccessibilities( propertyMap ); + * aggAcc.calculateConstants(); + * + * // do this in dest choice model + * //aggAcc.buildAccessibilityComponents( propertyMap ); + * + * } + */ + + /** + * Build the non-mandatory accessibilities object used by the auto ownership + * model + * + * @return BuildAccessibilities object containing non-mandatory size term + * and logsum information + */ + private void buildNonMandatoryAccessibilities(boolean calculateLandUseAccessibilities) + { + + HashMap propertyMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + + aggAcc = BuildAccessibilities.getInstance(); + aggAcc.setupBuildAccessibilities(propertyMap, calculateLandUseAccessibilities); + + + if(calculateLandUseAccessibilities) + aggAcc.setCalculatedLandUseAccessibilities(); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateWorkerSizeTerms(); + aggAcc.createSchoolSegmentNameIndices(); + aggAcc.calculateSchoolSizeTerms(); + aggAcc.calculateConstants(); + // aggAcc.buildAccessibilityComponents(propertyMap); + + boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, + READ_ACCESSIBILITIES); + if (readAccessibilities) + { + + // output data + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + if (isJppfRunningDistributed()) { + // release the memory used to store the access-tap, tap-egress, and + // tap-tap utilities while calculating accessibilities for the + // client program + // don't do this if we are running jppf in local mode + HashMap rbMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + StoredUtilityData.getInstance(MgraDataManager.getInstance(rbMap).getMaxMgra(), + MgraDataManager.getInstance(rbMap).getMaxTap(), + TazDataManager.getInstance(rbMap).getMaxTaz(), + BestTransitPathCalculator.ACC_EGR, ModelStructure.PERIODCODES) + .deallocateArrays(); + + MatrixDataManager.getInstance().clearData(); + } + } + + } + + private boolean isJppfRunningDistributed() { + //note: this assumes that the jppf config file is being entered in through a system property, and that it is a property file + String jppfConfigFile = System.getProperty("jppf.config"); + ResourceBundle jppfConfig = ResourceUtil.getResourceBundle(jppfConfigFile.replace(".properties","")); + try { + return !jppfConfig.getString("jppf.local.execution.enabled").equalsIgnoreCase("true"); + } catch (MissingResourceException e) { + return false; + } + } + + /* + * method used in original ARC implementation private void runIteration( int + * iteration, HouseholdDataManagerIf householdDataManager, + * CtrampDmuFactoryIf dmuFactory ) { String restartModel = ""; if ( + * hhDiskObjectKey != null && ! hhDiskObjectKey.equalsIgnoreCase("none") ) { + * String doFileName = hhDiskObjectFile + "_" + hhDiskObjectKey; + * householdDataManager.createHhArrayFromSerializedObjectInFile( doFileName, + * hhDiskObjectKey ); restartModel = hhDiskObjectKey; restartModels ( + * householdDataManager ); } else { restartModel = ResourceUtil.getProperty( + * resourceBundle, PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); if ( + * restartModel == null ) restartModel = "none"; if ( ! + * restartModel.equalsIgnoreCase("none") ) restartModels ( + * householdDataManager ); } JPPFClient jppfClient = new JPPFClient(); + * boolean runUsualWorkSchoolChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_WORKSCHOOL_CHOICE); if(runUsualWorkSchoolChoiceModel){ // + * create an object for calculating destination choice attraction size terms + * and managing shadow price calculations. DestChoiceSize dcSizeObj = new + * DestChoiceSize( modelStructure, tazDataManager ); // new the usual school + * and location choice model object UsualWorkSchoolLocationChoiceModel + * usualWorkSchoolLocationChoiceModel = new + * UsualWorkSchoolLocationChoiceModel(resourceBundle, restartModel, + * jppfClient, modelStructure, ms, tazDataManager, dcSizeObj, dmuFactory ); + * // run the model logger.info ( + * "starting usual work and school location choice."); + * usualWorkSchoolLocationChoiceModel + * .runSchoolAndLocationChoiceModel(householdDataManager); logger.info ( + * "finished with usual work and school location choice."); logger.info ( + * "writing work/school location choice results file; may take a few minutes ..." + * ); usualWorkSchoolLocationChoiceModel.saveResults( householdDataManager, + * projectDirectory, iteration ); logger.info ( + * String.format("finished writing results file.") ); + * usualWorkSchoolLocationChoiceModel = null; dcSizeObj = null; System.gc(); + * // write a disk object fle for the householdDataManager, in case we want + * to restart from the next step. if ( hhDiskObjectFile != null ) { + * logger.info ( + * "writing household disk object file after work/school location choice; may take a long time ..." + * ); String hhFileName = String.format( "%s_%d_ao", hhDiskObjectFile, + * iteration ); + * householdDataManager.createSerializedHhArrayInFileFromObject( hhFileName, + * "ao" ); logger.info (String.format( + * "finished writing household disk object file = %s after uwsl; continuing to household choice models ..." + * , hhFileName) ); } } boolean runAutoOwnershipChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_AUTO_OWNERSHIP ); boolean runFreeParkingChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_FREE_PARKING_AVAILABLE ); boolean + * runCoordinatedDailyActivityPatternChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN ); boolean + * runMandatoryTourFreqChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ ); boolean + * runMandatoryTourTimeOfDayChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR ); boolean + * runMandatoryTourModeChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE ); boolean + * runJointTourFrequencyModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_JOINT_TOUR_FREQ ); boolean runJointTourLocationChoiceModel + * = ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_JOINT_LOCATION_CHOICE ); boolean + * runJointTourModeChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE ); boolean + * runJointTourDepartureTimeAndDurationModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR ); boolean + * runIndivNonManTourFrequencyModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ ); boolean + * runIndivNonManTourLocationChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE ); boolean + * runIndivNonManTourModeChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE ); boolean + * runIndivNonManTourDepartureTimeAndDurationModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR ); boolean + * runAtWorkSubTourFrequencyModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ ); boolean + * runAtWorkSubtourLocationChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE ); boolean + * runAtWorkSubtourModeChoiceModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE ); boolean + * runAtWorkSubtourDepartureTimeAndDurationModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR ); boolean + * runStopFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_STOP_FREQUENCY ); boolean runStopLocationModel = + * ResourceUtil.getBooleanProperty(resourceBundle, + * PROPERTIES_RUN_STOP_LOCATION ); boolean runHouseholdModels = false; if ( + * runAutoOwnershipChoiceModel || runFreeParkingChoiceModel || + * runCoordinatedDailyActivityPatternChoiceModel || + * runMandatoryTourFreqChoiceModel || runMandatoryTourModeChoiceModel || + * runMandatoryTourTimeOfDayChoiceModel || runJointTourFrequencyModel || + * runJointTourLocationChoiceModel || runJointTourModeChoiceModel || + * runJointTourDepartureTimeAndDurationModel || + * runIndivNonManTourFrequencyModel || runIndivNonManTourLocationChoiceModel + * || runIndivNonManTourModeChoiceModel || + * runIndivNonManTourDepartureTimeAndDurationModel || + * runAtWorkSubTourFrequencyModel || runAtWorkSubtourLocationChoiceModel || + * runAtWorkSubtourModeChoiceModel || + * runAtWorkSubtourDepartureTimeAndDurationModel || runStopFrequencyModel || + * runStopLocationModel ) runHouseholdModels = true; // disk object file is + * labeled with the next component eligible to be run if model restarted + * String lastComponent = "uwsl"; String nextComponent = "ao"; if( + * runHouseholdModels ) { logger.info ( + * "starting HouseholdChoiceModelRunner." ); HashMap + * propertyMap = + * ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle); + * HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( + * propertyMap, jppfClient, restartModel, householdDataManager, ms, + * modelStructure, tazDataManager, dmuFactory ); + * runner.runHouseholdChoiceModels(); if( runAutoOwnershipChoiceModel ){ + * saveAoResults( householdDataManager, projectDirectory ); logAoResults( + * householdDataManager ); lastComponent = "ao"; nextComponent = "fp"; } if( + * runFreeParkingChoiceModel ){ logFpResults( householdDataManager ); + * lastComponent = "fp"; nextComponent = "cdap"; } if( + * runCoordinatedDailyActivityPatternChoiceModel ){ saveCdapResults( + * householdDataManager, projectDirectory ); logCdapResults( + * householdDataManager ); lastComponent = "cdap"; nextComponent = "imtf"; } + * if( runMandatoryTourFreqChoiceModel ){ logImtfResults( + * householdDataManager ); lastComponent = "imtf"; nextComponent = "imtod"; + * } if( runMandatoryTourTimeOfDayChoiceModel || + * runMandatoryTourModeChoiceModel ){ lastComponent = "imtod"; nextComponent + * = "jtf"; } if( runJointTourFrequencyModel ){ logJointModelResults( + * householdDataManager ); lastComponent = "jtf"; nextComponent = "jtl"; } + * if( runJointTourLocationChoiceModel ){ lastComponent = "jtl"; + * nextComponent = "jtod"; } if( runJointTourDepartureTimeAndDurationModel + * || runJointTourModeChoiceModel ){ lastComponent = "jtod"; nextComponent = + * "inmtf"; } if( runIndivNonManTourFrequencyModel ){ lastComponent = + * "inmtf"; nextComponent = "inmtl"; } if( + * runIndivNonManTourLocationChoiceModel ){ lastComponent = "inmtl"; + * nextComponent = "inmtod"; } if( + * runIndivNonManTourDepartureTimeAndDurationModel || + * runIndivNonManTourModeChoiceModel ){ lastComponent = "inmtod"; + * nextComponent = "awf"; } if( runAtWorkSubTourFrequencyModel ){ + * logAtWorkSubtourFreqResults( householdDataManager ); lastComponent = + * "awf"; nextComponent = "awl"; } if( runAtWorkSubtourLocationChoiceModel + * ){ lastComponent = "awl"; nextComponent = "awtod"; } if( + * runAtWorkSubtourDepartureTimeAndDurationModel || + * runAtWorkSubtourModeChoiceModel ){ lastComponent = "awtod"; nextComponent + * = "stf"; } if( runStopFrequencyModel ){ lastComponent = "stf"; + * nextComponent = "stl"; } if( runStopLocationModel ){ lastComponent = + * "stl"; nextComponent = "done"; } // write a disk object fle for the + * householdDataManager, in case we want to restart from the next step. if ( + * hhDiskObjectFile != null && ! lastComponent.equalsIgnoreCase("uwsl") ) { + * logger.info (String.format( + * "writing household disk object file after %s choice model; may take a long time ..." + * , lastComponent) ); String hhFileName = hhDiskObjectFile + "_" + + * nextComponent; + * householdDataManager.createSerializedHhArrayInFileFromObject( hhFileName, + * nextComponent ); logger.info ( + * String.format("finished writing household disk object file = %s.", + * hhFileName) ); } logger.info ( + * "finished with HouseholdChoiceModelRunner." ); } + */ + + public String getProjectDirectoryName() + { + return projectDirectory; + } + + private MatrixDataServer startMatrixServerProcess(String serverAddress, int serverPort) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServer matrixServer = new MatrixDataServer(); + + // bind this concrete object with the cajo library objects for managing RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + public void restartModels(HouseholdDataManagerIf householdDataManager) + { + + // if no filename was specified for the previous shadow price info, + // restartIter == -1, and random counts will be reset to 0. + int restartIter = -1; + String fileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); + if (fileName != null) + { + fileName = projectDirectory + fileName; + int underScoreIndex = fileName.lastIndexOf('_'); + int dotIndex = fileName.lastIndexOf('.'); + restartIter = Integer.parseInt(fileName.substring(underScoreIndex + 1, dotIndex)); + } + + boolean runPreAutoOwnershipModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_PRE_AUTO_OWNERSHIP); + if (runPreAutoOwnershipModel) + { + householdDataManager.resetPreAoRandom(); + } else + { + boolean runUsualWorkSchoolChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_WORKSCHOOL_CHOICE); + if (runUsualWorkSchoolChoiceModel) + { + householdDataManager.resetUwslRandom(restartIter + 1); + } else + { + boolean runAutoOwnershipModel = ResourceUtil.getBooleanProperty(resourceBundle, + PROPERTIES_RUN_AUTO_OWNERSHIP); + if (runAutoOwnershipModel) + { + // We're resetting the random number sequence used by pre-AO + // for + // the primary AO + householdDataManager.resetPreAoRandom(); + // householdDataManager.resetAoRandom( restartIter+1 ); + } else + { + // boolean runFreeParkingAvailableModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_FREE_PARKING_AVAILABLE); + // if ( runFreeParkingAvailableModel ) { + // householdDataManager.resetFpRandom(); + // } + // else { + boolean runCoordinatedDailyActivityPatternModel = ResourceUtil + .getBooleanProperty(resourceBundle, + PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN); + if (runCoordinatedDailyActivityPatternModel) + { + householdDataManager.resetCdapRandom(); + } else + { + boolean runIndividualMandatoryTourFrequencyModel = ResourceUtil + .getBooleanProperty(resourceBundle, + PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ); + if (runIndividualMandatoryTourFrequencyModel) + { + householdDataManager.resetImtfRandom(); + } else + { + // boolean + // runIndividualMandatoryTourDepartureAndDurationModel + // = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR); + // if ( + // runIndividualMandatoryTourDepartureAndDurationModel + // ) + // { + // householdDataManager.resetImtodRandom(); + // } + // else { + // boolean runJointTourFrequencyModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_JOINT_TOUR_FREQ); + // if ( runJointTourFrequencyModel ) { + // householdDataManager.resetJtfRandom(); + // } + // else { + // boolean runJointTourLocationModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_JOINT_LOCATION_CHOICE); + // if ( runJointTourLocationModel ) { + // householdDataManager.resetJtlRandom(); + // } + // else { + // boolean runJointTourDepartureAndDurationModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR); + // if ( runJointTourDepartureAndDurationModel ) { + // householdDataManager.resetJtodRandom(); + // } + // else { + boolean runIndividualNonMandatoryTourFrequencyModel = ResourceUtil + .getBooleanProperty(resourceBundle, + PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ); + if (runIndividualNonMandatoryTourFrequencyModel) + { + householdDataManager.resetInmtfRandom(); + } + // else { + // boolean + // runIndividualNonMandatoryTourLocationModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE); + // if ( runIndividualNonMandatoryTourLocationModel ) + // { + // householdDataManager.resetInmtlRandom(); + // } + // else { + // boolean + // runIndividualNonMandatoryTourDepartureAndDurationModel + // = ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR); + // if ( + // runIndividualNonMandatoryTourDepartureAndDurationModel + // ) { + // householdDataManager.resetInmtodRandom(); + // } + // else { + boolean runAtWorkSubTourFrequencyModel = ResourceUtil + .getBooleanProperty(resourceBundle, + PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ); + if (runAtWorkSubTourFrequencyModel) + { + householdDataManager.resetAwfRandom(); + } + // else { + // boolean runAtWorkSubtourLocationChoiceModel = + // ResourceUtil.getBooleanProperty( resourceBundle, + // PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE ); + // if ( runAtWorkSubtourLocationChoiceModel ) { + // householdDataManager.resetAwlRandom(); + // } + // else { + // boolean + // runAtWorkSubtourDepartureTimeAndDurationModel + // = ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR); + // if ( + // runAtWorkSubtourDepartureTimeAndDurationModel ) { + // householdDataManager.resetAwtodRandom(); + // } + // else { + boolean runStopFrequencyModel = ResourceUtil.getBooleanProperty( + resourceBundle, PROPERTIES_RUN_STOP_FREQUENCY); + if (runStopFrequencyModel) + { + householdDataManager.resetStfRandom(); + } + // else { + // boolean runStopLocationModel = + // ResourceUtil.getBooleanProperty(resourceBundle, + // PROPERTIES_RUN_STOP_LOCATION); + // if ( runStopLocationModel ) { + // householdDataManager.resetStlRandom(); + // } + // } + // } + // } + // } + // } + // } + // } + // } + // } + // } + // } + // } + } + } + } + } + } + } + + /** + * private void createSerializedObjectInFileFromObject( Object + * objectToSerialize, String serializedObjectFileName, String + * serializedObjectKey ){ try{ DataFile dataFile = new DataFile( + * serializedObjectFileName, 1 ); DataWriter dw = new DataWriter( + * serializedObjectKey ); dw.writeObject( objectToSerialize ); + * dataFile.insertRecord( dw ); dataFile.close(); } + * catch(NotSerializableException e) { logger.error( String.format( + * "NotSerializableException for %s. Trying to create serialized object with key=%s, in filename=%s." + * , objectToSerialize.getClass().getName(), serializedObjectKey, + * serializedObjectFileName ), e ); throw new RuntimeException(); } + * catch(IOException e) { logger.error( String.format( + * "IOException trying to write disk object file=%s, with key=%s for writing." + * , serializedObjectFileName, serializedObjectKey ), e ); throw new + * RuntimeException(); } } + * + * + * private Object createObjectFromSerializedObjectInFile( Object newObject, + * String serializedObjectFileName, String serializedObjectKey ){ try{ + * DataFile dataFile = new DataFile( serializedObjectFileName, "r" ); + * DataReader dr = dataFile.readRecord( serializedObjectKey ); newObject = + * dr.readObject(); dataFile.close(); return newObject; } catch(IOException + * e) { logger.error( String.format( + * "IOException trying to read disk object file=%s, with key=%s.", + * serializedObjectFileName, serializedObjectKey ), e ); throw new + * RuntimeException(); } catch(ClassNotFoundException e) { logger.error( + * String.format + * ("could not instantiate %s object, with key=%s from filename=%s.", + * newObject.getClass().getName(), serializedObjectFileName, + * serializedObjectKey ), e ); throw new RuntimeException(); } } + **/ + /** + * Loops through the households in the HouseholdDataManager, gets the auto + * ownership result for each household, and writes a text file with hhid and + * auto ownership. + * + * @param householdDataManager + * is the object from which the array of household objects can be + * retrieved. + * @param projectDirectory + * is the root directory for the output file named + */ + private void saveAoResults(HouseholdDataManagerIf householdDataManager, + String projectDirectory, boolean preModel) + { + + String aoResultsFileName; + try + { + + aoResultsFileName = resourceBundle.getString(PROPERTIES_RESULTS_AUTO_OWNERSHIP); + + // change the filename property value to include "_pre" at the end + // of the + // name before the extension, if this is a pre-auto ownership run + if (preModel) + { + int dotIndex = aoResultsFileName.indexOf('.'); + if (dotIndex > 0) + { + String beforeDot = aoResultsFileName.substring(0, dotIndex); + String afterDot = aoResultsFileName.substring(dotIndex); + aoResultsFileName = beforeDot + "_pre" + afterDot; + } else + { + aoResultsFileName += "_pre"; + } + } + + } catch (MissingResourceException e) + { + // if filename not specified in properties file, don't need to write + // it. + return; + } + + FileWriter writer; + PrintWriter outStream = null; + if (aoResultsFileName != null) + { + + aoResultsFileName = projectDirectory + aoResultsFileName; + + try + { + writer = new FileWriter(new File(aoResultsFileName)); + outStream = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening AO results file: %s.", + aoResultsFileName)); + throw new RuntimeException(e); + } + + outStream.println("HHID,AO"); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + int hhid = household.getHhId(); + int ao = household.getAutosOwned(); + + outStream.println(String.format("%d,%d", hhid, ao)); + + } + + } + + outStream.close(); + + } + + } + + private void logAoResults(HouseholdDataManagerIf householdDataManager, boolean preModel) + { + + String[] aoRowCategoryLabel = {"0 autos", "1 auto", "2 autos", "3 autos", "4 or more autos"}; + String[] aoColCategoryLabel = {"Non-GQ HHs", "GQ HHs",}; + + logger.info(""); + logger.info(""); + logger.info((preModel ? "Pre-" : "") + "Auto Ownership Model Results"); + String header = String.format("%-16s", "Category"); + for (String label : aoColCategoryLabel) + header += String.format("%15s", label); + header += String.format("%15s", "Total HHs"); + logger.info(header); + + // track the results + int[][] hhsByAutoOwnership = new int[aoRowCategoryLabel.length][aoColCategoryLabel.length]; + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + int ao = household.getAutosOwned(); + if (ao > hhsByAutoOwnership.length - 1) ao = hhsByAutoOwnership.length - 1; + + int gq = household.getIsGroupQuarters(); + hhsByAutoOwnership[ao][gq]++; + + } + + } + + int[] colTotals = new int[aoColCategoryLabel.length]; + for (int i = 0; i < hhsByAutoOwnership.length; i++) + { + + int rowTotal = 0; + String logString = String.format("%-16s", aoRowCategoryLabel[i]); + for (int j = 0; j < hhsByAutoOwnership[i].length; j++) + { + int value = hhsByAutoOwnership[i][j]; + logString += String.format("%15d", value); + rowTotal += value; + colTotals[j] += value; + } + logString += String.format("%15d", rowTotal); + logger.info(logString); + + } + + int total = 0; + String colTotalsString = String.format("%-16s", "Total"); + for (int j = 0; j < colTotals.length; j++) + { + colTotalsString += String.format("%15d", colTotals[j]); + total += colTotals[j]; + } + colTotalsString += String.format("%15d", total); + logger.info(colTotalsString); + + } + + private void logTpResults(HouseholdDataManagerIf householdDataManager) + { + + logger.info(""); + logger.info(""); + logger.info("Transponder Choice Model Results"); + logger.info(String.format("%-16s %20s", "Category", "Num Households")); + logger.info(String.format("%-16s %20s", "----------", "------------------")); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + int numYes = 0; + int numNo = 0; + int numOther = 0; + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + if (household.getTpChoice() + 1 == TransponderChoiceModel.TP_MODEL_NO_ALT) numNo++; + else if (household.getTpChoice() + 1 == TransponderChoiceModel.TP_MODEL_YES_ALT) numYes++; + else numOther++; + + } + + } + + logger.info(String.format("%-16s %20d", "No", numNo)); + logger.info(String.format("%-16s %20d", "Yes", numYes)); + logger.info(String.format("%-16s %20d", "Other", numOther)); + + logger.info(String.format("%-16s %20s", "----------", "------------------")); + logger.info(String.format("%-16s %20d", "Total", (numNo + numYes + numOther))); + + } + + private void logFpResults(HouseholdDataManagerIf householdDataManager) + { + + String[] fpCategoryLabel = {"No Choice Made", "Free Available", "Must Pay", "Reimbursed"}; + + logger.info(""); + logger.info(""); + logger.info("Free Parking Choice Model Results"); + logger.info(String.format("%-16s %20s %20s %20s %20s", "Category", "Workers in area 1", + "Workers in area 2", "Workers in area 3", "Workers in area 4")); + logger.info(String.format("%-16s %20s %20s %20s %20s", "----------", + "------------------", "------------------", "------------------", + "------------------")); + + // track the results by 4 work areas - only workers in area 1 should + // have made choices + int numParkAreas = 4; + int[][] workLocationsByFreeParking; + workLocationsByFreeParking = new int[fpCategoryLabel.length][numParkAreas]; + + // get the correspndence between mgra and park area to associate work + // locations with areas + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + int[] parkAreas = mgraManager.getMgraParkAreas(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + Person[] persons = household.getPersons(); + for (int p = 1; p < persons.length; p++) + { + int workLocation = persons[p].getWorkLocation(); + if (workLocation > 0 + && workLocation != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + int area = parkAreas[workLocation]; + int areaIndex = area - 1; + + int fp = persons[p].getFreeParkingAvailableResult(); + int freqIndex = 0; + if (fp > 0) freqIndex = fp; + + workLocationsByFreeParking[freqIndex][areaIndex]++; + } + } + + } + + } + + int[] total = new int[numParkAreas]; + for (int i = 0; i < workLocationsByFreeParking.length; i++) + { + logger.info(String.format("%-16s %20d %20d %20d %20d", fpCategoryLabel[i], + workLocationsByFreeParking[i][0], workLocationsByFreeParking[i][1], + workLocationsByFreeParking[i][2], workLocationsByFreeParking[i][3])); + for (int j = 0; j < numParkAreas; j++) + total[j] += workLocationsByFreeParking[i][j]; + } + logger.info(String.format("%-16s %20s %20s %20s %20s", "----------", + "------------------", "------------------", "------------------", + "------------------")); + logger.info(String.format("%-16s %20d %20d %20d %20d", "Totals", total[0], total[1], + total[2], total[3])); + + } + + private void logIeResults(HouseholdDataManagerIf householdDataManager) + { + + String[] ieCategoryLabel = {"No IE Trip", "Yes IE Trip"}; + + logger.info(""); + logger.info(""); + logger.info("Internal-External Trip Choice Model Results"); + logger.info(String.format("%-30s %20s %20s %20s", "Person Type", ieCategoryLabel[0], + ieCategoryLabel[1], "Total")); + logger.info(String.format("%-30s %20s %20s %20s", "-------------", "-------------", + "-------------", "---------")); + + // summarize yes/no choice by person type + int[][] personTypeByIeChoice; + personTypeByIeChoice = new int[Person.PERSON_TYPE_NAME_ARRAY.length][2]; + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + Person[] persons = household.getPersons(); + for (int p = 1; p < persons.length; p++) + { + + int ie = persons[p].getInternalExternalTripChoiceResult(); + + // ie = 1 means no, 2 means yes. Get index by subtracting 1. + // person typ indices are 1 based, so subtract 1 for array + // index + try + { + personTypeByIeChoice[persons[p].getPersonTypeNumber() - 1][ie - 1]++; + } catch (ArrayIndexOutOfBoundsException e) + { + logger.error("array index error"); + logger.error("hhid=" + household.getHhId() + ", p=" + p + ", ie=" + ie + + ", personType=" + persons[p].getPersonTypeNumber(), e); + } + } + + } + + } + + int[] totals = new int[2]; + for (int i = 0; i < personTypeByIeChoice.length; i++) + { + int total = personTypeByIeChoice[i][0] + personTypeByIeChoice[i][1]; + logger.info(String.format("%-30s %20d %20d %20d", Person.PERSON_TYPE_NAME_ARRAY[i], + personTypeByIeChoice[i][0], personTypeByIeChoice[i][1], total)); + totals[0] += personTypeByIeChoice[i][0]; + totals[1] += personTypeByIeChoice[i][1]; + } + logger.info(String.format("%-30s %20s %20s %20s", "-------------", "-------------", + "-------------", "---------")); + logger.info(String.format("%-30s %20d %20d %20d", "Totals", totals[0], totals[1], + (totals[0] + totals[1]))); + + } + + /** + * Records the coordinated daily activity pattern model results to the + * logger. A household-level summary simply records each pattern type and a + * person-level summary summarizes the activity choice by person type + * (full-time worker, university student, etc). + * + */ + public void logCdapResults(HouseholdDataManagerIf householdDataManager) + { + + String[] activityNameArray = {Definitions.MANDATORY_PATTERN, + Definitions.NONMANDATORY_PATTERN, Definitions.HOME_PATTERN}; + + getLogReportSummaries(householdDataManager); + + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info("Coordinated Daily Activity Pattern Model Results"); + + // count of activities by person type + logger.info(" "); + logger.info("CDAP Results: Count of activities by person type"); + String firstHeader = "Person type "; + String secondHeader = "----------------------------- "; + for (int i = 0; i < activityNameArray.length; ++i) + { + firstHeader += " " + activityNameArray[i] + " "; + secondHeader += "--------- "; + } + + firstHeader += " Total"; + secondHeader += "---------"; + + logger.info(firstHeader); + logger.info(secondHeader); + + int[] columnTotals = new int[activityNameArray.length]; + + for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) + { + String personType = Person.PERSON_TYPE_NAME_ARRAY[i]; + String stringToLog = String.format("%-30s", personType); + int lineTotal = 0; + + if (cdapByPersonTypeAndActivity.containsKey(personType)) + { + + for (int j = 0; j < activityNameArray.length; ++j) + { + int count = 0; + if (cdapByPersonTypeAndActivity.get(personType).containsKey( + activityNameArray[j])) + { + count = cdapByPersonTypeAndActivity.get(personType).get( + activityNameArray[j]); + } + stringToLog += String.format("%10d", count); + + lineTotal += count; + columnTotals[j] += count; + } // j + + } // if key + + stringToLog += String.format("%10d", lineTotal); + logger.info(stringToLog); + + } // i + + logger.info(secondHeader); + + String stringToLog = String.format("%-30s", "Total"); + int lineTotal = 0; + for (int j = 0; j < activityNameArray.length; ++j) + { + stringToLog += String.format("%10d", columnTotals[j]); + lineTotal += columnTotals[j]; + } // j + + stringToLog += String.format("%10d", lineTotal); + logger.info(stringToLog); + + // count of patterns + logger.info(" "); + logger.info(" "); + logger.info("CDAP Results: Count of patterns"); + logger.info("Pattern Count"); + logger.info("------------------ ---------"); + + // sort the map by hh size first + Set hhSizeKeySet = cdapByHhSizeAndPattern.keySet(); + Integer[] hhSizeKeyArray = new Integer[hhSizeKeySet.size()]; + hhSizeKeySet.toArray(hhSizeKeyArray); + Arrays.sort(hhSizeKeyArray); + + int total = 0; + for (int i = 0; i < hhSizeKeyArray.length; ++i) + { + + // sort the patterns alphabetically + HashMap patternMap = cdapByHhSizeAndPattern.get(hhSizeKeyArray[i]); + Set patternKeySet = patternMap.keySet(); + String[] patternKeyArray = new String[patternKeySet.size()]; + patternKeySet.toArray(patternKeyArray); + Arrays.sort(patternKeyArray); + for (int j = 0; j < patternKeyArray.length; ++j) + { + int count = patternMap.get(patternKeyArray[j]); + total += count; + logger.info(String.format("%-18s%10d", patternKeyArray[j], count)); + } + + } + + logger.info("------------------ ---------"); + logger.info(String.format("%-18s%10d", "Total", total)); + logger.info(" "); + + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info(" "); + logger.info(" "); + + } + + /** + * Logs the results of the individual mandatory tour frequency model. + * + */ + public void logImtfResults(HouseholdDataManagerIf householdDataManager) + { + + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info("Individual Mandatory Tour Frequency Model Results"); + + // count of model results + logger.info(" "); + String firstHeader = "Person type "; + String secondHeader = "----------------------------- "; + + String[] choiceResults = HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_RESULTS; + + // summarize results + HashMap countByPersonType = new HashMap(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Person[] personArray = householdArray[i].getPersons(); + for (int j = 1; j < personArray.length; j++) + { + + // only summarize persons with mandatory pattern + String personActivity = personArray[j].getCdapActivity(); + if (personActivity != null + && personArray[j].getCdapActivity().equalsIgnoreCase("M")) + { + + String personTypeString = personArray[j].getPersonType(); + int choice = personArray[j].getImtfChoice(); + + if (choice == 0) + { + + // there are 5 IMTF alts, so it's the offset for the + // extra at home categories + if (personArray[j].getPersonEmploymentCategoryIndex() < Person.EmployStatus.NOT_EMPLOYED + .ordinal() + && personArray[j].getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) choice = 5 + 1; + else if (personArray[j].getPersonEmploymentCategoryIndex() < Person.EmployStatus.NOT_EMPLOYED + .ordinal() + && personArray[j].getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) choice = 5 + 2; + else if (personArray[j].getPersonIsStudent() < Person.EmployStatus.NOT_EMPLOYED + .ordinal() + && personArray[j].getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) choice = 5 + 3; + else if (personArray[j].getPersonIsStudent() < Person.EmployStatus.NOT_EMPLOYED + .ordinal() + && personArray[j].getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) + choice = 5 + 4; + + } + + // count the results + if (countByPersonType.containsKey(personTypeString)) + { + + int[] counterArray = countByPersonType.get(personTypeString); + counterArray[choice - 1]++; + countByPersonType.put(personTypeString, counterArray); + + } else + { + + int[] counterArray = new int[choiceResults.length]; + counterArray[choice - 1]++; + countByPersonType.put(personTypeString, counterArray); + + } + } + + } + + } + + } + + for (int i = 0; i < choiceResults.length; ++i) + { + firstHeader += String.format("%12s", choiceResults[i]); + secondHeader += "----------- "; + } + + firstHeader += String.format("%12s", "Total"); + secondHeader += "-----------"; + + logger.info(firstHeader); + logger.info(secondHeader); + + int[] columnTotals = new int[choiceResults.length]; + + int lineTotal = 0; + for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) + { + String personTypeString = Person.PERSON_TYPE_NAME_ARRAY[i]; + String stringToLog = String.format("%-30s", personTypeString); + + if (countByPersonType.containsKey(personTypeString)) + { + + lineTotal = 0; + int[] countArray = countByPersonType.get(personTypeString); + for (int j = 0; j < choiceResults.length; ++j) + { + stringToLog += String.format("%12d", countArray[j]); + columnTotals[j] += countArray[j]; + lineTotal += countArray[j]; + } // j + } else + { + // if key + // log zeros + lineTotal = 0; + for (int j = 0; j < choiceResults.length; ++j) + { + stringToLog += String.format("%12d", 0); + } + } + + stringToLog += String.format("%12d", lineTotal); + + logger.info(stringToLog); + + } // i + + String stringToLog = String.format("%-30s", "Total"); + lineTotal = 0; + for (int j = 0; j < choiceResults.length; ++j) + { + stringToLog += String.format("%12d", columnTotals[j]); + lineTotal += columnTotals[j]; + } // j + + logger.info(secondHeader); + stringToLog += String.format("%12d", lineTotal); + logger.info(stringToLog); + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info(" "); + logger.info(" "); + + } + + private void logJointModelResults(HouseholdDataManagerIf householdDataManager, + CtrampDmuFactoryIf dmuFactory) + { + + String uecFileDirectory = ResourceUtil.getProperty(resourceBundle, PROPERTIES_UEC_PATH); + String uecFileName = ResourceUtil.getProperty(resourceBundle, + JointTourModels.UEC_FILE_PROPERTIES_TARGET); + uecFileName = uecFileDirectory + uecFileName; + + int dataSheet = ResourceUtil.getIntegerProperty(resourceBundle, + JointTourModels.UEC_DATA_PAGE_TARGET); + int freqCompSheet = ResourceUtil.getIntegerProperty(resourceBundle, + JointTourModels.UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE); + + // get the alternative names + JointTourModelsDMU dmuObject = dmuFactory.getJointTourModelsDMU(); + ChoiceModelApplication jointTourFrequencyModel = new ChoiceModelApplication(uecFileName, + freqCompSheet, dataSheet, + ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle), + (VariableTable) dmuObject); + String[] altLabels = jointTourFrequencyModel.getAlternativeNames(); + + // this is the first index in the summary array for choices made by + // eligible households + int[] jointTourChoiceFreq = new int[altLabels.length + 1]; + + TreeMap partySizeFreq = new TreeMap(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Tour[] jt = householdArray[i].getJointTourArray(); + int jtfAlt = householdArray[i].getJointTourFreqChosenAlt(); + + if (jt == null) + { + + if (jtfAlt > 0) + { + logger.error(String + .format("HHID=%d, joint tour array is null, but a valid alternative=%d is recorded for the household.", + householdArray[i].getHhId(), jtfAlt)); + throw new RuntimeException(); + } + + jointTourChoiceFreq[0]++; + + } else + { + + if (jtfAlt < 1) + { + logger.error(String + .format("HHID=%d, joint tour array is not null, but an invalid alternative=%d is recorded for the household.", + householdArray[i].getHhId(), jtfAlt)); + throw new RuntimeException(); + } + + jointTourChoiceFreq[jtfAlt]++; + + // determine party size frequency for joint tours generated + Person[] persons = householdArray[i].getPersons(); + for (int j = 0; j < jt.length; j++) + { + + int compAlt = jt[j].getJointTourComposition(); + + // determine number of children and adults in tour + int adults = 0; + int children = 0; + int[] participants = jt[j].getPersonNumArray(); + for (int k = 0; k < participants.length; k++) + { + int index = participants[k]; + Person person = persons[index]; + if (person.getPersonIsAdult() == 1) adults++; + else children++; + } + + // create a key to use for a frequency map for + // "JointTourPurpose_Composition_NumAdults_NumChildren" + String key = String.format("%s_%d_%d_%d", jt[j].getTourPurpose(), compAlt, + adults, children); + + int value = 0; + if (partySizeFreq.containsKey(key)) value = partySizeFreq.get(key); + partySizeFreq.put(key, ++value); + + } + + } + + } + + } + + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info("Joint Tour Frequency and Joint Tour Composition Model Results"); + + logger.info(" "); + logger.info("Frequency Table of Households by Joint Tour Frequency Choice"); + logger.info(String.format("%-5s %-26s %12s", "Alt", "Alt Name", "Households")); + + int rowTotal = jointTourChoiceFreq[0]; + logger.info(String.format("%-5d %-26s %12d", 0, "None", jointTourChoiceFreq[0])); + for (int i = 1; i <= altLabels.length; i++) + { + logger.info(String.format("%-5d %-26s %12d", i, altLabels[i - 1], + jointTourChoiceFreq[i])); + rowTotal += jointTourChoiceFreq[i]; + } + logger.info(String.format("%-34s %12d", "Total Households", rowTotal)); + + logger.info(" "); + logger.info(" "); + logger.info(" "); + + logger.info("Frequency Table of Joint Tours by All Parties Generated"); + logger.info(String.format("%-5s %-20s %-15s %10s %10s %10s", "N", "Purpose", + "Type", "Adults", "Children", "Freq")); + + int count = 1; + for (String key : partySizeFreq.keySet()) + { + + int start = 0; + int end = 0; + int compIndex = 0; + int adults = 0; + int children = 0; + String indexString = ""; + String purpose = ""; + + start = 0; + end = key.indexOf('_', start); + purpose = key.substring(start, end); + + start = end + 1; + end = key.indexOf('_', start); + indexString = key.substring(start, end); + compIndex = Integer.parseInt(indexString); + + start = end + 1; + end = key.indexOf('_', start); + indexString = key.substring(start, end); + adults = Integer.parseInt(indexString); + + start = end + 1; + indexString = key.substring(start); + children = Integer.parseInt(indexString); + + logger.info(String.format("%-5d %-20s %-15s %10d %10d %10d", count++, + purpose, JointTourModels.JOINT_TOUR_COMPOSITION_NAMES[compIndex], adults, + children, partySizeFreq.get(key))); + } + + logger.info(" "); + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info(" "); + + } + + private void getLogReportSummaries(HouseholdDataManagerIf householdDataManager) + { + + // summary collections + cdapByHhSizeAndPattern = new HashMap>(); + cdapByPersonTypeAndActivity = new HashMap>(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] partialHhArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (Household hhObject : partialHhArray) + { + + // get the household's activity pattern choice + String pattern = hhObject.getCoordinatedDailyActivityPattern(); + if (pattern == null) continue; + + Person[] personArray = hhObject.getPersons(); + for (int j = 1; j < personArray.length; j++) + { + + // get person's activity string + String activityString = personArray[j].getCdapActivity(); + + // get the person type to simmarize results by + String personTypeString = personArray[j].getPersonType(); + + // check if the person type is in the map + if (cdapByPersonTypeAndActivity.containsKey(personTypeString)) + { + + HashMap activityCountMap = cdapByPersonTypeAndActivity + .get(personTypeString); + + // check if the activity is in the activity map + int currentCount = 1; + if (activityCountMap.containsKey(activityString)) + currentCount = activityCountMap.get(activityString) + 1; + + activityCountMap.put(activityString, currentCount); + cdapByPersonTypeAndActivity.put(personTypeString, activityCountMap); + + } else + { + + HashMap activityCountMap = new HashMap(); + activityCountMap.put(activityString, 1); + cdapByPersonTypeAndActivity.put(personTypeString, activityCountMap); + + } // is personType in map if + + } // j (person loop) + + // count each type of pattern string by hhSize + if ((!cdapByHhSizeAndPattern.isEmpty()) + && cdapByHhSizeAndPattern.containsKey(pattern.length())) + { + + HashMap patternCountMap = cdapByHhSizeAndPattern.get(pattern + .length()); + + int currentCount = 1; + if (patternCountMap.containsKey(pattern)) + currentCount = patternCountMap.get(pattern) + 1; + patternCountMap.put(pattern, currentCount); + cdapByHhSizeAndPattern.put(pattern.length(), patternCountMap); + + } else + { + + HashMap patternCountMap = new HashMap(); + patternCountMap.put(pattern, 1); + cdapByHhSizeAndPattern.put(pattern.length(), patternCountMap); + + } // is personType in map if + + } + + } + + } + + /** + * Loops through the households in the HouseholdDataManager, gets the + * coordinated daily activity pattern for each person in the household, and + * writes a text file with hhid, personid, persnum, and activity pattern. + * + * @param householdDataManager + */ + public void saveCdapResults(HouseholdDataManagerIf householdDataManager, String projectDirectory) + { + + String cdapResultsFileName; + try + { + cdapResultsFileName = resourceBundle.getString(PROPERTIES_RESULTS_CDAP); + } catch (MissingResourceException e) + { + // if filename not specified in properties file, don't need to write + // it. + return; + } + + FileWriter writer; + PrintWriter outStream = null; + if (cdapResultsFileName != null) + { + + cdapResultsFileName = projectDirectory + cdapResultsFileName; + + try + { + writer = new FileWriter(new File(cdapResultsFileName)); + outStream = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening CDAP results file: %s.", + cdapResultsFileName)); + throw new RuntimeException(e); + } + + outStream.println("HHID,PersonID,PersonNum,PersonType,ActivityString"); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + int hhid = household.getHhId(); + + // get the pattern for each person + Person[] personArray = household.getPersons(); + for (int j = 1; j < personArray.length; j++) + { + + Person person = personArray[j]; + + int persId = person.getPersonId(); + int persNum = person.getPersonNum(); + int persType = person.getPersonTypeNumber(); + String activityString = person.getCdapActivity(); + + outStream.println(String.format("%d,%d,%d,%d,%s", hhid, persId, persNum, + persType, activityString)); + + } // j (person loop) + + } + + } + + outStream.close(); + + } + + } + + /** + * Logs the results of the model. + * + */ + public void logAtWorkSubtourFreqResults(HouseholdDataManagerIf householdDataManager) + { + + String[] alternativeNames = modelStructure.getAwfAltLabels(); + HashMap awfByPersonType = new HashMap(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + for (int i = 0; i < householdArray.length; ++i) + { + + // get this household's person array + Person[] personArray = householdArray[i].getPersons(); + + // loop through the person array (1-based) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + // loop through the work tours for this person + ArrayList tourList = person.getListOfWorkTours(); + if (tourList == null || tourList.size() == 0) continue; + + // count the results by person type + String personTypeString = person.getPersonType(); + + for (Tour workTour : tourList) + { + + int choice = 0; + if (person.getListOfAtWorkSubtours().size() == 0) choice = 1; + else + { + choice = workTour.getSubtourFreqChoice(); + if (choice == 0) choice++; + } + + int dummy = 0; + if (person.getPersonTypeNumber() == 7) + { + dummy = 1; + } + + // count the results by person type + if (awfByPersonType.containsKey(personTypeString)) + { + int[] counterArray = awfByPersonType.get(personTypeString); + counterArray[choice - 1]++; + awfByPersonType.put(personTypeString, counterArray); + + } else + { + int[] counterArray = new int[alternativeNames.length]; + counterArray[choice - 1]++; + awfByPersonType.put(personTypeString, counterArray); + } + + } + + } + + } + + } + + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info("At-Work Subtour Frequency Model Results"); + + // count of model results + logger.info(" "); + String firstHeader = "Person type "; + String secondHeader = "--------------------------- "; + + for (int i = 0; i < alternativeNames.length; ++i) + { + firstHeader += String.format("%16s", alternativeNames[i]); + secondHeader += "------------ "; + } + + firstHeader += String.format("%16s", "Total"); + secondHeader += "------------"; + + logger.info(firstHeader); + logger.info(secondHeader); + + int[] columnTotals = new int[alternativeNames.length]; + + int lineTotal = 0; + for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) + { + String personTypeString = Person.PERSON_TYPE_NAME_ARRAY[i]; + String stringToLog = String.format("%-28s", personTypeString); + + if (awfByPersonType.containsKey(personTypeString)) + { + + lineTotal = 0; + int[] countArray = awfByPersonType.get(personTypeString); + for (int j = 0; j < alternativeNames.length; ++j) + { + stringToLog += String.format("%16d", countArray[j]); + columnTotals[j] += countArray[j]; + lineTotal += countArray[j]; + } // j + + } else + { + // if key + // log zeros + lineTotal = 0; + for (int j = 0; j < alternativeNames.length; ++j) + { + stringToLog += String.format("%16d", 0); + } + } + + stringToLog += String.format("%16d", lineTotal); + + logger.info(stringToLog); + + } // i + + String stringToLog = String.format("%-28s", "Total"); + lineTotal = 0; + for (int j = 0; j < alternativeNames.length; ++j) + { + stringToLog += String.format("%16d", columnTotals[j]); + lineTotal += columnTotals[j]; + } // j + + logger.info(secondHeader); + stringToLog += String.format("%16d", lineTotal); + logger.info(stringToLog); + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info(" "); + logger.info(" "); + + } + + /** + * Logs the results of the individual tour stop frequency model. + * + */ + public void logIndivStfResults(HouseholdDataManagerIf householdDataManager) + { + + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info("Individual Tour Stop Frequency Model Results"); + + // count of model results + logger.info(" "); + String firstHeader = "Tour Purpose "; + String secondHeader = "--------------- "; + + int[] obStopsAlt = StopFrequencyDMU.NUM_OB_STOPS_FOR_ALT; + int[] ibStopsAlt = StopFrequencyDMU.NUM_IB_STOPS_FOR_ALT; + + // 10 purposes + int[][] chosen = new int[obStopsAlt.length][11]; + HashMap indexPurposeMap = modelStructure.getIndexPrimaryPurposeNameMap(); + HashMap purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Person[] personArray = householdArray[i].getPersons(); + for (int j = 1; j < personArray.length; j++) + { + + List tourList = new ArrayList(); + + // apply stop frequency for all person tours + tourList.addAll(personArray[j].getListOfWorkTours()); + tourList.addAll(personArray[j].getListOfSchoolTours()); + tourList.addAll(personArray[j].getListOfIndividualNonMandatoryTours()); + tourList.addAll(personArray[j].getListOfAtWorkSubtours()); + + for (Tour t : tourList) + { + + int index = t.getTourPrimaryPurposeIndex(); + int choice = t.getStopFreqChoice(); + chosen[choice][index]++; + + } + + } + + } + + } + + for (int i = 1; i < chosen[1].length; ++i) + { + firstHeader += String.format("%18s", indexPurposeMap.get(i)); + secondHeader += " --------------- "; + } + + firstHeader += String.format("%18s", "Total"); + secondHeader += " --------------- "; + + logger.info(firstHeader); + logger.info(secondHeader); + + int[] columnTotals = new int[chosen[1].length]; + + int lineTotal = 0; + for (int i = 1; i < chosen.length; ++i) + { + String stringToLog = String.format("%d out, %d in ", obStopsAlt[i], ibStopsAlt[i]); + + lineTotal = 0; + int[] countArray = chosen[i]; + for (int j = 1; j < countArray.length; ++j) + { + stringToLog += String.format("%18d", countArray[j]); + columnTotals[j] += countArray[j]; + lineTotal += countArray[j]; + } // j + + stringToLog += String.format("%18d", lineTotal); + + logger.info(stringToLog); + + } // i + + String stringToLog = String.format("%-17s", "Total"); + lineTotal = 0; + for (int j = 1; j < chosen[1].length; ++j) + { + stringToLog += String.format("%18d", columnTotals[j]); + lineTotal += columnTotals[j]; + } // j + + logger.info(secondHeader); + stringToLog += String.format("%18d", lineTotal); + logger.info(stringToLog); + logger.info(" "); + logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); + logger.info(" "); + logger.info(" "); + + } + + private void createSerializedObjectInFileFromObject(Object objectToSerialize, + String serializedObjectFileName, String serializedObjectKey) + { + try + { + DataFile dataFile = new DataFile(serializedObjectFileName, 1); + DataWriter dw = new DataWriter(serializedObjectKey); + dw.writeObject(objectToSerialize); + dataFile.insertRecord(dw); + dataFile.close(); + } catch (NotSerializableException e) + { + logger.error( + String.format( + "NotSerializableException for %s. Trying to create serialized object with key=%s, in filename=%s.", + objectToSerialize.getClass().getName(), serializedObjectKey, + serializedObjectFileName), e); + throw new RuntimeException(); + } catch (IOException e) + { + logger.error(String.format( + "IOException trying to write disk object file=%s, with key=%s for writing.", + serializedObjectFileName, serializedObjectKey), e); + throw new RuntimeException(); + } + } + + private Object createObjectFromSerializedObjectInFile(Object newObject, + String serializedObjectFileName, String serializedObjectKey) + { + try + { + DataFile dataFile = new DataFile(serializedObjectFileName, "r"); + DataReader dr = dataFile.readRecord(serializedObjectKey); + newObject = dr.readObject(); + dataFile.close(); + return newObject; + } catch (IOException e) + { + logger.error(String.format( + "IOException trying to read disk object file=%s, with key=%s.", + serializedObjectFileName, serializedObjectKey), e); + throw new RuntimeException(); + } catch (ClassNotFoundException e) + { + logger.error(String.format( + "could not instantiate %s object, with key=%s from filename=%s.", newObject + .getClass().getName(), serializedObjectFileName, serializedObjectKey), + e); + throw new RuntimeException(); + } + } + + private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) + { + + ArrayList startEndIndexList = new ArrayList(); + + int startIndex = 0; + int endIndex = 0; + + while (endIndex < numberOfHouseholds - 1) + { + endIndex = startIndex + NUM_WRITE_PACKETS - 1; + if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) + endIndex = numberOfHouseholds - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += NUM_WRITE_PACKETS; + } + + return startEndIndexList; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java new file mode 100644 index 0000000..95f7801 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java @@ -0,0 +1,52 @@ +package org.sandag.abm.ctramp; + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 9, 2008 Time: 3:13:17 PM To + * change this template use File | Settings | File Templates. + */ +public interface CtrampDmuFactoryIf +{ + + AutoOwnershipChoiceDMU getAutoOwnershipDMU(); + + ParkingProvisionChoiceDMU getFreeParkingChoiceDMU(); + + TelecommuteDMU getTelecommuteDMU(); + + TransponderChoiceDMU getTransponderChoiceDMU(); + + InternalExternalTripChoiceDMU getInternalExternalTripChoiceDMU(); + + CoordinatedDailyActivityPatternDMU getCoordinatedDailyActivityPatternDMU(); + + DcSoaDMU getDcSoaDMU(); + + DestChoiceDMU getDestChoiceDMU(); + + DestChoiceTwoStageModelDMU getDestChoiceSoaTwoStageDMU(); + + DestChoiceTwoStageSoaTazDistanceUtilityDMU getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + TourModeChoiceDMU getModeChoiceDMU(); + + IndividualMandatoryTourFrequencyDMU getIndividualMandatoryTourFrequencyDMU(); + + TourDepartureTimeAndDurationDMU getTourDepartureTimeAndDurationDMU(); + + AtWorkSubtourFrequencyDMU getAtWorkSubtourFrequencyDMU(); + + JointTourModelsDMU getJointTourModelsDMU(); + + IndividualNonMandatoryTourFrequencyDMU getIndividualNonMandatoryTourFrequencyDMU(); + + StopFrequencyDMU getStopFrequencyDMU(); + + StopLocationDMU getStopLocationDMU(); + + TripModeChoiceDMU getTripModeChoiceDMU(); + + ParkingChoiceDMU getParkingChoiceDMU(); + + MicromobilityChoiceDMU getMicromobilityChoiceDMU(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java new file mode 100644 index 0000000..d2fc746 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java @@ -0,0 +1,23 @@ +package org.sandag.abm.ctramp; + +public class DAOException + extends RuntimeException +{ + static final long serialVersionUID = -1881205326938716446L; + + public DAOException(String message) + { + super(message); + } + + public DAOException(Throwable cause) + { + super(cause); + } + + public DAOException(String message, Throwable cause) + { + super(message, cause); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java new file mode 100644 index 0000000..769f85d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java @@ -0,0 +1,215 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class DcSoaDMU + implements SoaDMU, Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(DcSoaDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Person person; + protected Tour tour; + + protected IndexValues dmuIndex = null; + protected String dmuLabel = "Origin Location"; + + protected double[] dcSize; + protected double[] distance; + + protected BuildAccessibilities aggAcc; + + public DcSoaDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug DC SOA UEC"); + } + + } + + public void setAggAcc(BuildAccessibilities myAggAcc) + { + aggAcc = myAggAcc; + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setPersonObject(Person personObject) + { + person = personObject; + } + + public void setTourObject(Tour tourObject) + { + tour = tourObject; + } + + public void setDestChoiceSize(double[] dcSize) + { + this.dcSize = dcSize; + } + + public void setDestDistance(double[] distance) + { + this.distance = distance; + } + + public double[] getDestDistance() + { + return distance; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public Household getHouseholdObject() + { + return hh; + } + + public int getTourPurposeIsEscort() + { + return tour.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME) ? 1 : 0; + } + + public int getNumPreschool() + { + return hh.getNumPreschool(); + } + + public int getNumGradeSchoolStudents() + { + return hh.getNumGradeSchoolStudents(); + } + + public int getNumHighSchoolStudents() + { + return hh.getNumHighSchoolStudents(); + } + + protected double getLnDcSize(int alt) + { + + double size = dcSize[alt]; + + double logSize = 0.0; + logSize = Math.log(size + 1); + + return logSize; + + } + + protected double getDcSizeAlt(int alt) + { + return dcSize[alt]; + } + + protected double getHouseholdsDestAlt(int mgra) + { + return aggAcc.getMgraHouseholds(mgra); + } + + protected double getGradeSchoolEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraGradeSchoolEnrollment(mgra); + } + + protected double getHighSchoolEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraHighSchoolEnrollment(mgra); + } + + public int getGradeSchoolDistrictDestAlt(int mgra) + { + return aggAcc.getMgraGradeSchoolDistrict(mgra); + } + + public int getHomeMgraGradeSchoolDistrict() + { + return aggAcc.getMgraGradeSchoolDistrict(hh.getHhMgra()); + } + + public double getHighSchoolDistrictDestAlt(int mgra) + { + return aggAcc.getMgraHighSchoolDistrict(mgra); + } + + public double getHomeMgraHighSchoolDistrict() + { + return aggAcc.getMgraHighSchoolDistrict(hh.getHhMgra()); + } + + public double getOriginToMgraDistanceAlt(int alt) + { + return distance[alt]; + } + + public double getUniversityEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraUniversityEnrollment(mgra); + } + + public String getDmuLabel() + { + return dmuLabel; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java new file mode 100644 index 0000000..496426e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java @@ -0,0 +1,22 @@ +package org.sandag.abm.ctramp; + +/** + * This class holds definitions that are inherant in CT-RAMP based models + * including: person types tour category types purposes activity types + * + * @author Jim + * + */ +public final class Definitions +{ + + // Coordinated daily activity pattern type definitions + public static final String MANDATORY_PATTERN = "M"; + public static final String NONMANDATORY_PATTERN = "N"; + public static final String HOME_PATTERN = "H"; + + private Definitions() + { + // Not implemented in utility classes + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java new file mode 100644 index 0000000..be69e84 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java @@ -0,0 +1,394 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public abstract class DestChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(DestChoiceDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Person person; + protected Tour tour; + protected IndexValues dmuIndex = null; + + protected double workAccessibility; + protected double nonMandatoryAccessibility; + + protected double[] homeMgraNonMandatoryAccessibilityArray; + protected double[] homeMgraTotalEmploymentAccessibilityArray; + protected double[] homeMgraSizeArray; + protected double[] homeMgraDistanceArray; + protected double[] modeChoiceLogsums; + protected double[] dcSoaCorrections; + + protected int toursLeftCount; + + protected ModelStructure modelStructure; + protected MgraDataManager mgraManager; + protected BuildAccessibilities aggAcc; + protected AccessibilitiesTable accTable; + + public DestChoiceDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + initDmuObject(); + } + + public abstract void setMcLogsum(int mgra, double logsum); + + private void initDmuObject() + { + + dmuIndex = new IndexValues(); + + // create default objects - some choice models use these as place + // holders for values + person = new Person(null, -1, modelStructure); + hh = new Household(modelStructure); + + mgraManager = MgraDataManager.getInstance(); + + int maxMgra = mgraManager.getMaxMgra(); + + modeChoiceLogsums = new double[maxMgra + 1]; + dcSoaCorrections = new double[maxMgra + 1]; + + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setPersonObject(Person personObject) + { + person = personObject; + } + + public void setTourObject(Tour tour) + { + this.tour = tour; + } + + public void setAggAcc(BuildAccessibilities aggAcc) + { + this.aggAcc = aggAcc; + } + + public void setAccTable(AccessibilitiesTable myAccTable) + { + accTable = myAccTable; + } + + public void setDestChoiceSize(double[] homeMgraSizeArray) + { + this.homeMgraSizeArray = homeMgraSizeArray; + } + + public void setDestChoiceDistance(double[] homeMgraDistanceArray) + { + this.homeMgraDistanceArray = homeMgraDistanceArray; + } + + public void setDcSoaCorrections(int mgra, double correction) + { + dcSoaCorrections[mgra] = correction; + } + + public void setNonMandatoryAccessibility(double nonMandatoryAccessibility) + { + this.nonMandatoryAccessibility = nonMandatoryAccessibility; + } + + public void setToursLeftCount(int count) + { + toursLeftCount = count; + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug DC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public Household getHouseholdObject() + { + return hh; + } + + public Person getPersonObject() + { + return person; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + protected int getToursLeftCount() + { + return toursLeftCount; + } + + protected int getMaxContinuousAvailableWindow() + { + + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return hh + .getMaxJointTimeWindow(tour); + else return person.getMaximumContinuousAvailableWindow(); + } + + protected double getDcSoaCorrectionsAlt(int alt) + { + return dcSoaCorrections[alt]; + } + + protected double getMcLogsumDestAlt(int mgra) + { + return modeChoiceLogsums[mgra]; + } + + protected double getPopulationDestAlt(int mgra) + { + return aggAcc.getMgraPopulation(mgra); + } + + protected double getHouseholdsDestAlt(int mgra) + { + return aggAcc.getMgraHouseholds(mgra); + } + + protected double getGradeSchoolEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraGradeSchoolEnrollment(mgra); + } + + protected double getHighSchoolEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraHighSchoolEnrollment(mgra); + } + + protected double getUniversityEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraUniversityEnrollment(mgra); + } + + protected double getOtherCollegeEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraOtherCollegeEnrollment(mgra); + } + + protected double getAdultSchoolEnrollmentDestAlt(int mgra) + { + return aggAcc.getMgraAdultSchoolEnrollment(mgra); + } + + protected int getIncome() + { + return hh.getIncomeCategory(); + } + + protected int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + protected int getAutos() + { + return hh.getAutosOwned(); + } + + protected int getWorkers() + { + return hh.getWorkers(); + } + + protected int getNumberOfNonWorkingAdults() + { + return hh.getNumberOfNonWorkingAdults(); + } + + protected int getNumPreschool() + { + return hh.getNumPreschool(); + } + + public int getNumGradeSchoolStudents() + { + return hh.getNumGradeSchoolStudents(); + } + + public int getNumHighSchoolStudents() + { + return hh.getNumHighSchoolStudents(); + } + + protected int getNumChildrenUnder16() + { + return hh.getNumChildrenUnder16(); + } + + protected int getNumChildrenUnder19() + { + return hh.getNumChildrenUnder19(); + } + + protected int getAge() + { + return person.getAge(); + } + + protected int getFemaleWorker() + { + if (person.getPersonIsFemale() == 1) return 1; + else return 0; + } + + protected int getFemale() + { + if (person.getPersonIsFemale() == 1) return 1; + else return 0; + } + + protected int getFullTimeWorker() + { + if (person.getPersonIsFullTimeWorker() == 1) return 1; + else return 0; + } + + protected int getTypicalUniversityStudent() + { + return person.getPersonIsTypicalUniversityStudent(); + } + + protected int getPersonType() + { + return person.getPersonTypeNumber(); + } + + protected int getPersonHasBachelors() + { + return person.getHasBachelors(); + } + + protected int getPersonIsWorker() + { + return person.getPersonIsWorker(); + } + + protected int getWorkTaz() + { + return person.getWorkLocation(); + } + + protected int getWorkTourModeIsSOV() + { + boolean tourModeIsSov = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); + if (tourModeIsSov) return 1; + else return 0; + } + + protected int getTourIsJoint() + { + return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 + : 0; + } + + protected double getTotEmpAccessibilityAlt(int alt) + { + return homeMgraTotalEmploymentAccessibilityArray[alt]; + } + + protected double getNonMandatoryAccessibilityAlt(int alt) + { + return accTable.getAggregateAccessibility("nonmotor", alt); + } + + protected double getOpSovDistanceAlt(int alt) + { + return homeMgraDistanceArray[alt]; + } + + protected double getLnDcSizeAlt(int alt) + { + return Math.log(homeMgraSizeArray[alt] + 1); + } + + protected double getDcSizeAlt(int alt) + { + return homeMgraSizeArray[alt]; + } + + protected void setWorkAccessibility(double accessibility) + { + workAccessibility = accessibility; + } + + protected double getWorkAccessibility() + { + return workAccessibility; + } + + protected double getNonMandatoryAccessibility() + { + return nonMandatoryAccessibility; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java new file mode 100644 index 0000000..bd4c33f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java @@ -0,0 +1,1082 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.Serializable; +import java.util.Date; +import java.util.HashMap; +import java.util.LinkedList; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; + +public class DestChoiceModelManager + implements Serializable +{ + + private static String PROPERTIES_WORK_DC_SOA_UEC_FILE = "work.soa.uec.file"; + private static String PROPERTIES_WORK_DC_SOA_UEC_MODEL_PAGE = "work.soa.uec.model"; + private static String PROPERTIES_WORK_DC_SOA_UEC_DATA_PAGE = "work.soa.uec.data"; + + private static String PROPERTIES_UNIV_DC_SOA_UEC_FILE = "univ.soa.uec.file"; + private static String PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE = "univ.soa.uec.model"; + private static String PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE = "univ.soa.uec.data"; + + private static String PROPERTIES_HS_DC_SOA_UEC_FILE = "hs.soa.uec.file"; + private static String PROPERTIES_HS_DC_SOA_UEC_MODEL_PAGE = "hs.soa.uec.model"; + private static String PROPERTIES_HS_DC_SOA_UEC_DATA_PAGE = "hs.soa.uec.data"; + + private static String PROPERTIES_GS_DC_SOA_UEC_FILE = "gs.soa.uec.file"; + private static String PROPERTIES_GS_DC_SOA_UEC_MODEL_PAGE = "gs.soa.uec.model"; + private static String PROPERTIES_GS_DC_SOA_UEC_DATA_PAGE = "gs.soa.uec.data"; + + private static String PROPERTIES_PS_DC_SOA_UEC_FILE = "ps.soa.uec.file"; + private static String PROPERTIES_PS_DC_SOA_UEC_MODEL_PAGE = "ps.soa.uec.model"; + private static String PROPERTIES_PS_DC_SOA_UEC_DATA_PAGE = "ps.soa.uec.data"; + + private static final int PRESCHOOL_ALT_INDEX = BuildAccessibilities.PRESCHOOL_ALT_INDEX; + private static final int GRADE_SCHOOL_ALT_INDEX = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; + private static final int HIGH_SCHOOL_ALT_INDEX = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; + private static final int UNIV_TYPICAL_ALT_INDEX = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; + private static final int UNIV_NONTYPICAL_ALT_INDEX = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; + private static final int NUMBER_OF_SCHOOL_SEGMENT_TYPES = 5; + + private static transient Logger logger = Logger.getLogger(DestChoiceModelManager.class); + + private static DestChoiceModelManager objInstance = null; + + private LinkedList modelQueueWorkLoc = null; + private LinkedList modelQueueSchoolLoc = null; + private LinkedList modelQueueWork = null; + private LinkedList modelQueueSchool = null; + + private MgraDataManager mgraManager; + private TazDataManager tdm; + + private int maxTaz; + + private BuildAccessibilities aggAcc; + + private HashMap propertyMap; + private String dcUecFileName; + private String soaUecFileName; + private int soaSampleSize; + private String modeChoiceUecFileName; + private CtrampDmuFactoryIf dmuFactory; + + private int modelIndexWork; + private int modelIndexSchool; + private int currentIteration; + + private DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu; + private DestChoiceTwoStageSoaProbabilitiesCalculator workLocSoaDistProbsObject; + private DestChoiceTwoStageSoaProbabilitiesCalculator psLocSoaDistProbsObject; + private DestChoiceTwoStageSoaProbabilitiesCalculator gsLocSoaDistProbsObject; + private DestChoiceTwoStageSoaProbabilitiesCalculator hsLocSoaDistProbsObject; + private DestChoiceTwoStageSoaProbabilitiesCalculator univLocSoaDistProbsObject; + + // the first dimension on these arrays is work location segments (worker + // occupations) + private double[][][] workSizeProbs; + private double[][][] workTazDistProbs; + + // the first dimension on these arrays is school location segment type (ps, + // gs, hs, univTypical, univNonTypical) + private double[][][] schoolSizeProbs; + private double[][][] schoolTazDistProbs; + + private AutoTazSkimsCalculator tazDistanceCalculator; + + private boolean managerIsSetup = false; + + private int completedHouseholdsWork; + private int completedHouseholdsSchool; + private boolean logResults=false; + + private DestChoiceModelManager() + { + } + + public static synchronized DestChoiceModelManager getInstance() + { + // logger.info( + // "beginning of DestChoiceModelManager.getInstance() - objInstance address = " + // + objInstance ); + if (objInstance == null) + { + objInstance = new DestChoiceModelManager(); + // logger.info( + // "after new DestChoiceModelManager() - objInstance address = " + + // objInstance ); + return objInstance; + } else + { + // logger.info( + // "returning current DestChoiceModelManager() - objInstance address = " + // + objInstance ); + return objInstance; + } + } + + // the task instances should call needToInitialize() first, then this method + // if necessary. + public synchronized void managerSetup(HashMap propertyMap, + ModelStructure modelStructure, MatrixDataServerIf ms, String dcUecFileName, + String soaUecFileName, int soaSampleSize, CtrampDmuFactoryIf dmuFactory, + String restartModelString) + { + + if (managerIsSetup) return; + + // get the HouseholdChoiceModelsManager instance and clear the objects + // that hold large memory references + HouseholdChoiceModelsManager.getInstance().clearHhModels(); + + modelIndexWork = 0; + modelIndexSchool = 0; + completedHouseholdsWork = 0; + completedHouseholdsSchool = 0; + + System.out.println(String.format("initializing DC ModelManager: thread=%s.", Thread + .currentThread().getName())); + + this.propertyMap = propertyMap; + this.dcUecFileName = dcUecFileName; + this.soaUecFileName = soaUecFileName; + this.soaSampleSize = soaSampleSize; + this.dmuFactory = dmuFactory; + + logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + + mgraManager = MgraDataManager.getInstance(propertyMap); + tdm = TazDataManager.getInstance(propertyMap); + maxTaz = tdm.getMaxTaz(); + + modelQueueWorkLoc = new LinkedList(); + modelQueueSchoolLoc = new LinkedList(); + modelQueueWork = new LinkedList(); + modelQueueSchool = new LinkedList(); + + // Initialize the MatrixDataManager to use the MatrixDataServer instance + // passed in, unless ms is null. + if (ms == null) + { + + logger.info(Thread.currentThread().getName() + + ": No remote MatrixServer being used, MatrixDataManager will get created when needed by DestChoiceModelManager."); + } else + { + + String testString = ms.testRemote(Thread.currentThread().getName()); + logger.info(String.format(Thread.currentThread().getName() + + ": DestChoiceModelManager connecting to remote MatrixDataServer.")); + logger.info(String.format("MatrixDataServer connection test: %s", testString)); + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + } + + aggAcc = BuildAccessibilities.getInstance(); + + // assume that if the filename exists, at was created previously, either + // in another model run, or by the main client + // if the filename doesn't exist, then calculate the accessibilities + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + boolean accFileReadFlag = Util.getBooleanValueFromPropertyMap(propertyMap, + CtrampApplication.READ_ACCESSIBILITIES); + + if ((new File(accFileName)).canRead()) + { + + logger.info("filling Accessibilities Object in DestChoiceModelManager by reading file: " + + accFileName + "."); + aggAcc.readAccessibilityTableFromFile(accFileName); + + aggAcc.setupBuildAccessibilities(propertyMap, false); + aggAcc.createSchoolSegmentNameIndices(); + + aggAcc.calculateSizeTerms(); + + } else + { + + aggAcc.setupBuildAccessibilities(propertyMap, false); + aggAcc.createSchoolSegmentNameIndices(); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + + logger.info("filling Accessibilities Object in DestChoiceModelManager by calculating them."); + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + + // compute the array of cumulative taz distance based SOA probabilities + // for each origin taz. + locChoiceDistSoaDmu = dmuFactory.getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + managerIsSetup = true; + + } + + public synchronized void returnWorkLocModelObject(WorkLocationChoiceModel dcModel, + int taskIndex, int startIndex, int endIndex) + { + modelQueueWorkLoc.add(dcModel); + completedHouseholdsWork += (endIndex - startIndex + 1); + if(logResults){ + logger.info(String + .format("returned workLocationChoice[%d,%d] to workQueueLoc, task=%d, thread=%s, completedHouseholds=%d.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread + .currentThread().getName(), completedHouseholdsWork)); + } + } + + public synchronized void returnDcWorkModelObject(MandatoryDestChoiceModel dcModel, + int taskIndex, int startIndex, int endIndex) + { + modelQueueWork.add(dcModel); + completedHouseholdsWork += (endIndex - startIndex + 1); + if(logResults){ + logger.info(String + .format("returned dcModelWork[%d,%d] to workQueue, task=%d, thread=%s, completedHouseholds=%d.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread + .currentThread().getName(), completedHouseholdsWork)); + } + } + + public synchronized void returnSchoolLocModelObject(SchoolLocationChoiceModel dcModel, + int taskIndex, int startIndex, int endIndex) + { + modelQueueSchoolLoc.add(dcModel); + completedHouseholdsSchool += (endIndex - startIndex + 1); + if(logResults){ + logger.info(String + .format("returned schoolLocationChoice[%d,%d] to schoolQueueLoc, task=%d, thread=%s, completedHouseholds=%d.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread + .currentThread().getName(), completedHouseholdsSchool)); + } + } + + public synchronized void returnDcSchoolModelObject(MandatoryDestChoiceModel dcModel, + int taskIndex, int startIndex, int endIndex) + { + modelQueueSchool.add(dcModel); + completedHouseholdsSchool += (endIndex - startIndex + 1); + if(logResults){ + logger.info(String + .format("returned dcModelSchool[%d,%d] to schoolQueue, task=%d, thread=%s, completedHouseholds=%d.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread + .currentThread().getName(), completedHouseholdsSchool)); + } + } + + public synchronized WorkLocationChoiceModel getWorkLocModelObject(int taskIndex, int iteration, + DestChoiceSize dcSizeObj, int[] uecIndices, int[] soaUecIndices) + { + + // can release memory for the school location choice probabilities + // before running school location choice + clearSchoolProbabilitiesArrys(); + + WorkLocationChoiceModel dcModel = null; + + if (!modelQueueWorkLoc.isEmpty()) + { + + // the first task processed with an iteration parameter greater than + // the manager's + // current iteration updates the manager's SOA size and dist + // probabilities arrays and + // updates the iteration count. + if (iteration > currentIteration) + { + + // update the arrays of cumulative probabilities based on mgra + // size for mgras within each origin taz. + double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); + updateWorkSoaProbabilities(workLocSoaDistProbsObject, dcSizeObj, workSizeProbs, + workTazDistProbs, dcSizeArray); + + currentIteration = iteration; + completedHouseholdsWork = 0; + } + + dcModel = modelQueueWorkLoc.remove(); + dcModel.setDcSizeObject(dcSizeObj); + + if(logResults){ + logger.info(String.format( + "removed workLocationChoice[%d,%d] from workQueueLoc, task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + } + + } else + { + + if (modelIndexWork == 0 && iteration == 0) + { + + // compute the arrays of cumulative probabilities based on mgra + // size for mgras within each origin taz. + logger.info("pre-computing work SOA Distance and Size probabilities."); + workLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_WORK_DC_SOA_UEC_FILE, + PROPERTIES_WORK_DC_SOA_UEC_MODEL_PAGE, PROPERTIES_WORK_DC_SOA_UEC_DATA_PAGE); + double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); + workSizeProbs = new double[dcSizeArray.length][maxTaz][]; + workTazDistProbs = new double[dcSizeArray.length][][]; + updateWorkSoaProbabilities(workLocSoaDistProbsObject, dcSizeObj, workSizeProbs, + workTazDistProbs, dcSizeArray); + + currentIteration = 0; + completedHouseholdsWork = 0; + } + + modelIndexWork++; + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + // pass null in instead if modelStructure, since it's not available + // and won't be needed for logsum calculation. + TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, + TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + + dcModel = new WorkLocationChoiceModel(modelIndexWork, propertyMap, dcSizeObj, aggAcc, + dcUecFileName, soaUecFileName, soaSampleSize, modeChoiceUecFileName, + dmuFactory, immcModel, workSizeProbs, workTazDistProbs); + + dcModel.setupWorkSegments(uecIndices, soaUecIndices); + dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, + soaSampleSize); + + logger.info(String.format("created workLocationChoice[%d,%d], task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + + } + + return dcModel; + + } + + public synchronized MandatoryDestChoiceModel getDcWorkModelObject(int taskIndex, int iteration, + DestChoiceSize dcSizeObj, int[] uecIndices, int[] soaUecIndices) + { + + MandatoryDestChoiceModel dcModel = null; + + if (!modelQueueWork.isEmpty()) + { + + // the first task processed with an iteration parameter greater than + // the manager's + // current iteration updates the manager's SOA size and dist + // probabilities arrays and + // updates the iteration count. + if (iteration > currentIteration) + { + + currentIteration = iteration; + completedHouseholdsWork = 0; + } + + dcModel = modelQueueWork.remove(); + dcModel.setDcSizeObject(dcSizeObj); + + if(logResults){ + logger.info(String.format( + "removed dcModelWork[%d,%d] from workQueue, task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + } + + } else + { + + if (modelIndexWork == 0 && iteration == 0) + { + + currentIteration = 0; + completedHouseholdsWork = 0; + } + + modelIndexWork++; + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + // pass null in instead if modelStructure, since it's not available + // and won't be needed for logsum calculation. + TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, + TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + + dcModel = new MandatoryDestChoiceModel(modelIndexWork, propertyMap, dcSizeObj, aggAcc, + mgraManager, dcUecFileName, soaUecFileName, soaSampleSize, + modeChoiceUecFileName, dmuFactory, immcModel); + + dcModel.setupWorkSegments(uecIndices, soaUecIndices); + dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, + soaSampleSize); + + logger.info(String.format("created dcModelWork[%d,%d], task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + + } + + return dcModel; + + } + + public synchronized SchoolLocationChoiceModel getSchoolLocModelObject(int taskIndex, + int iteration, DestChoiceSize dcSizeObj) + { + // can release memory for the work location choice probabilities before + // running school location choice + clearWorkProbabilitiesArrys(); + clearWorkLocModels(); + + SchoolLocationChoiceModel dcModel = null; + + int[] gsDistrict = new int[maxTaz + 1]; + int[] hsDistrict = new int[maxTaz + 1]; + double[] univEnrollment = new double[maxTaz + 1]; + + if (!modelQueueSchoolLoc.isEmpty()) + { + + // the first task processed with an iteration parameter greater than + // the + // manager's current iteration count clears the dcModel cache and + // updates the iteration count. + if (iteration > currentIteration) + { + + // compute the exponentiated distance utilities that all + // segments of this tour purpose will share + double[][] tazDistExpUtils = null; + + logger.info("updating pre-school SOA Distance and Size probabilities."); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(psLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getPsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[PRESCHOOL_ALT_INDEX], + schoolTazDistProbs[PRESCHOOL_ALT_INDEX]); + + logger.info("updating grade school SOA Distance and Size probabilities."); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(gsLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getGsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[GRADE_SCHOOL_ALT_INDEX], + schoolTazDistProbs[GRADE_SCHOOL_ALT_INDEX]); + + logger.info("updating high school SOA Distance and Size probabilities."); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(hsLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getHsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[HIGH_SCHOOL_ALT_INDEX], + schoolTazDistProbs[HIGH_SCHOOL_ALT_INDEX]); + + logger.info("updating university-typical school SOA Distance and Size probabilities."); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(univLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getUnivTypicalSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[UNIV_TYPICAL_ALT_INDEX], + schoolTazDistProbs[UNIV_TYPICAL_ALT_INDEX]); + + logger.info("updating university-non-typical school SOA Distance and Size probabilities."); + updateSchoolSoaProbabilities(aggAcc.getUnivNonTypicalSegmentNameIndexMap(), + dcSizeObj, tazDistExpUtils, schoolSizeProbs[UNIV_NONTYPICAL_ALT_INDEX], + schoolTazDistProbs[UNIV_NONTYPICAL_ALT_INDEX]); + + currentIteration = iteration; + completedHouseholdsSchool = 0; + + } + + dcModel = modelQueueSchoolLoc.remove(); + dcModel.setDcSizeObject(dcSizeObj); + if(logResults){ + logger.info(String.format( + "removed schoolLocationChoice[%d,%d] from schoolQueueLoc, task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + } + + } else + { + + if (modelIndexSchool == 0 && iteration == 0) + { + + // if the schoolSizeProbs array is null, no task has yet + // initialized the probabilities arrays, so enter the block. + // if not null, the arrays have been computed, so it's ok to + // skip. + if (schoolSizeProbs == null) + { + + // compute the exponentiated distance utilities that all + // segments of this tour purpose will share + double[][] tazDistExpUtils = null; + + int[] gsDistrictByMgra = aggAcc.getMgraGsDistrict(); + int[] hsDistrictByMgra = aggAcc.getMgraHsDistrict(); + + // determine university enrollment by TAZs + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray != null) + { + for (int mgra : mgraArray) + { + univEnrollment[taz] = aggAcc.getMgraUniversityEnrollment(mgra); + } + } + } + locChoiceDistSoaDmu.setTazUnivEnrollment(univEnrollment); + + // determine grade school and high school districts by TAZs + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray != null) + { + for (int mgra : mgraArray) + { + gsDistrict[taz] = gsDistrictByMgra[mgra]; + hsDistrict[taz] = hsDistrictByMgra[mgra]; + break; + } + } + } + locChoiceDistSoaDmu.setTazGsDistricts(gsDistrict); + locChoiceDistSoaDmu.setTazHsDistricts(hsDistrict); + + schoolSizeProbs = new double[NUMBER_OF_SCHOOL_SEGMENT_TYPES][maxTaz][]; + schoolTazDistProbs = new double[NUMBER_OF_SCHOOL_SEGMENT_TYPES][maxTaz][maxTaz]; + + // compute the arrays of cumulative probabilities based on + // mgra size for mgras within each origin taz. + try + { + logger.info("pre-computing pre-school SOA Distance and Size probabilities."); + psLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_PS_DC_SOA_UEC_FILE, + PROPERTIES_PS_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_PS_DC_SOA_UEC_DATA_PAGE); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(psLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getPsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[PRESCHOOL_ALT_INDEX], + schoolTazDistProbs[PRESCHOOL_ALT_INDEX]); + } catch (Exception e) + { + logger.error("exception caught updating pre-school SOA probabilities", e); + System.exit(-1); + } + + try + { + logger.info("pre-computing grade school SOA Distance and Size probabilities."); + gsLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_GS_DC_SOA_UEC_FILE, + PROPERTIES_GS_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_GS_DC_SOA_UEC_DATA_PAGE); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(gsLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getGsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[GRADE_SCHOOL_ALT_INDEX], + schoolTazDistProbs[GRADE_SCHOOL_ALT_INDEX]); + } catch (Exception e) + { + logger.error("exception caught updating grade school SOA probabilities", e); + System.exit(-1); + } + + try + { + logger.info("pre-computing high school SOA Distance and Size probabilities."); + hsLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_HS_DC_SOA_UEC_FILE, + PROPERTIES_HS_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_HS_DC_SOA_UEC_DATA_PAGE); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(hsLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getHsSegmentNameIndexMap(), dcSizeObj, + tazDistExpUtils, schoolSizeProbs[HIGH_SCHOOL_ALT_INDEX], + schoolTazDistProbs[HIGH_SCHOOL_ALT_INDEX]); + } catch (Exception e) + { + logger.error("exception caught updating high school SOA probabilities", e); + System.exit(-1); + } + + try + { + logger.info("pre-computing university-typical SOA Distance and Size probabilities."); + univLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_UNIV_DC_SOA_UEC_FILE, + PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE); + tazDistExpUtils = computeTazDistanceExponentiatedUtilities(univLocSoaDistProbsObject); + updateSchoolSoaProbabilities(aggAcc.getUnivTypicalSegmentNameIndexMap(), + dcSizeObj, tazDistExpUtils, + schoolSizeProbs[UNIV_TYPICAL_ALT_INDEX], + schoolTazDistProbs[UNIV_TYPICAL_ALT_INDEX]); + } catch (Exception e) + { + logger.error("exception caught updating university SOA probabilities", e); + System.exit(-1); + } + + try + { + logger.info("pre-computing university-non-typical SOA Distance and Size probabilities."); + univLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_UNIV_DC_SOA_UEC_FILE, + PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE); + updateSchoolSoaProbabilities(aggAcc.getUnivNonTypicalSegmentNameIndexMap(), + dcSizeObj, tazDistExpUtils, + schoolSizeProbs[UNIV_NONTYPICAL_ALT_INDEX], + schoolTazDistProbs[UNIV_NONTYPICAL_ALT_INDEX]); + } catch (Exception e) + { + logger.error("exception caught updating university SOA probabilities", e); + System.exit(-1); + } + + currentIteration = 0; + completedHouseholdsSchool = 0; + + } + + } + + modelIndexSchool++; + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + // pass null in instead if modelStructure, since it's not available + // and won't be needed for logsum calculation. + TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, + TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + + dcModel = new SchoolLocationChoiceModel(modelIndexSchool, propertyMap, dcSizeObj, + aggAcc, dcUecFileName, soaUecFileName, soaSampleSize, modeChoiceUecFileName, + dmuFactory, immcModel, schoolSizeProbs, schoolTazDistProbs); + + dcModel.setupSchoolSegments(); + dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, + soaSampleSize); + + logger.info(String.format("created schoolLocationChoice[%d,%d], task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + + } + + return dcModel; + + } + + public synchronized MandatoryDestChoiceModel getDcSchoolModelObject(int taskIndex, + int iteration, DestChoiceSize dcSizeObj) + { + + MandatoryDestChoiceModel dcModel = null; + if (!modelQueueSchool.isEmpty()) + { + + // the first task processed with an iteration parameter greater than + // the + // manager's current iteration count clears the dcModel cache and + // updates the iteration count. + if (iteration > currentIteration) + { + + currentIteration = iteration; + completedHouseholdsSchool = 0; + + } + + dcModel = modelQueueSchool.remove(); + dcModel.setDcSizeObject(dcSizeObj); + if(logResults){ + logger.info(String.format( + "removed dcModelSchool[%d,%d] from schoolQueue, task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + } + + } else + { + + if (modelIndexSchool == 0 && iteration == 0) + { + currentIteration = 0; + completedHouseholdsSchool = 0; + } + + modelIndexSchool++; + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + // pass null in instead if modelStructure, since it's not available + // and won't be needed for logsum calculation. + TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, + TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + + dcModel = new MandatoryDestChoiceModel(modelIndexSchool, propertyMap, dcSizeObj, + aggAcc, mgraManager, dcUecFileName, soaUecFileName, soaSampleSize, + modeChoiceUecFileName, dmuFactory, immcModel); + + dcModel.setupSchoolSegments(); + dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, + soaSampleSize); + + logger.info(String.format("created dcModelSchool[%d,%d], task=%d, thread=%s.", + currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() + .getName())); + + } + + return dcModel; + + } + + public synchronized void clearDcModels() + { + + clearWorkLocModels(); + clearSchoolLocModels(); + clearWorkProbabilitiesArrys(); + clearSchoolProbabilitiesArrys(); + + if (tazDistanceCalculator != null) + { + tazDistanceCalculator.clearStoredTazsDistanceSkims(); + tazDistanceCalculator = null; + } + + logger.info("DestChoiceModelManager elements cleared."); + + } + + private void clearWorkLocModels() + { + + if (modelQueueWorkLoc != null && !modelQueueWorkLoc.isEmpty()) + { + + logger.info(String.format( + "%s: clearing dc choice models modelQueueWorkLoc, thread=%s.", new Date(), + Thread.currentThread().getName())); + + while (!modelQueueWorkLoc.isEmpty()) + modelQueueWorkLoc.remove(); + modelIndexWork = 0; + completedHouseholdsWork = 0; + + } + + if (modelQueueWork != null && !modelQueueWork.isEmpty()) + { + + logger.info(String.format("%s: clearing dc choice models modelQueueWork, thread=%s.", + new Date(), Thread.currentThread().getName())); + while (!modelQueueWork.isEmpty()) + modelQueueWork.remove(); + + modelIndexWork = 0; + completedHouseholdsWork = 0; + + } + + } + + private void clearSchoolLocModels() + { + + if (modelQueueSchoolLoc != null && !modelQueueSchoolLoc.isEmpty()) + { + + logger.info(String.format( + "%s: clearing dc choice models modelQueueSchoolLoc, thread=%s.", new Date(), + Thread.currentThread().getName())); + while (!modelQueueSchoolLoc.isEmpty()) + modelQueueSchoolLoc.remove(); + + modelIndexSchool = 0; + completedHouseholdsSchool = 0; + + } + + if (modelQueueSchool != null && !modelQueueSchool.isEmpty()) + { + + logger.info(String.format( + "%s: clearing dc choice models modelQueueSchool, thread=%s.", new Date(), + Thread.currentThread().getName())); + while (!modelQueueSchool.isEmpty()) + modelQueueSchool.remove(); + + modelIndexSchool = 0; + completedHouseholdsSchool = 0; + + } + + } + + private void clearWorkProbabilitiesArrys() + { + + // null out the cache of probabilities arrays for work location choice + if (workSizeProbs != null) + { + for (int i = 0; i < workSizeProbs.length; i++) + { + if (workSizeProbs[i] != null) + { + for (int j = 0; j < workSizeProbs[i].length; j++) + workSizeProbs[i][j] = null; + } + workSizeProbs[i] = null; + } + workSizeProbs = null; + } + + if (workTazDistProbs != null) + { + for (int i = 0; i < workTazDistProbs.length; i++) + { + if (workTazDistProbs[i] != null) + { + for (int j = 0; j < workTazDistProbs[i].length; j++) + workTazDistProbs[i][j] = null; + } + workTazDistProbs[i] = null; + } + workTazDistProbs = null; + } + + } + + private void clearSchoolProbabilitiesArrys() + { + + // null out the cache of probabilities arrays for work location choice + if (schoolSizeProbs != null) + { + for (int i = 0; i < schoolSizeProbs.length; i++) + { + if (schoolSizeProbs[i] != null) + { + for (int j = 0; j < schoolSizeProbs[i].length; j++) + schoolSizeProbs[i][j] = null; + } + schoolSizeProbs[i] = null; + } + schoolSizeProbs = null; + } + + if (schoolTazDistProbs != null) + { + for (int i = 0; i < schoolTazDistProbs.length; i++) + { + if (schoolTazDistProbs[i] != null) + { + for (int j = 0; j < schoolTazDistProbs[i].length; j++) + schoolTazDistProbs[i][j] = null; + } + schoolTazDistProbs[i] = null; + } + schoolTazDistProbs = null; + } + + } + + private void updateWorkSoaProbabilities( + DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, + DestChoiceSize dcSizeObj, double[][][] sizeProbs, double[][][] tazDistProbs, + double[][] dcSizeArray) + { + + HashMap segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); + + for (String segmentName : segmentNameIndexMap.keySet()) + { + + int segmentIndex = segmentNameIndexMap.get(segmentName); + + // compute the TAZ size values from the mgra values and the + // correspondence between mgras and tazs. + double[] tazSize = computeTazSize(dcSizeArray[segmentIndex]); + locChoiceDistSoaDmu.setDestChoiceTazSize(tazSize); + + // tazDistProbs[segmentIndex] = + // locChoiceSoaDistProbsObject.computeDistanceProbabilities( 3737, + // locChoiceDistSoaDmu ); + tazDistProbs[segmentIndex] = locChoiceSoaDistProbsObject + .computeDistanceProbabilities(locChoiceDistSoaDmu); + + computeSizeSegmentProbabilities(sizeProbs[segmentIndex], dcSizeArray[segmentIndex]); + + } + + } + + private void updateSchoolSoaProbabilities(HashMap segmentNameIndexMap, + DestChoiceSize dcSizeObj, double[][] tazDistExpUtils, double[][] sizeProbs, + double[][] tazDistProbs) + { + + double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); + + double[] tempExpUtils = new double[tazDistExpUtils.length]; + + // compute an array of SOA probabilities for each segment + for (String segmentName : segmentNameIndexMap.keySet()) + { + + // compute the TAZ size values from the mgra values and the + // correspondence between mgras and tazs. + int segmentIndex = segmentNameIndexMap.get(segmentName); + double[] tazSize = computeTazSize(dcSizeArray[segmentIndex]); + + // compute the taz dist probabilities from the exponentiated + // utilities for this segmnet and the taz size terms + for (int i = 0; i < tazDistExpUtils.length; i++) + { + + // compute the final exponentiated utilities by multiplying with + // taz size, and accumulate total exponentiated utility. + double totalExpUtil = 0; + for (int j = 0; j < tempExpUtils.length; j++) + { + tempExpUtils[j] = tazDistExpUtils[i][j] * tazSize[j + 1]; + totalExpUtil += tempExpUtils[j]; + } + + if (totalExpUtil > 0) + { + + // compute the SOA cumulative probabilities + tazDistProbs[i][0] = tempExpUtils[0] / totalExpUtil; + for (int j = 1; j < tempExpUtils.length - 1; j++) + { + double prob = tempExpUtils[j] / totalExpUtil; + tazDistProbs[i][j] = tazDistProbs[i][j - 1] + prob; + } + tazDistProbs[i][tempExpUtils.length - 1] = 1.0; + + } + + } + + computeSizeSegmentProbabilities(sizeProbs, dcSizeArray[segmentIndex]); + + } + + } + + private double[][] computeTazDistanceExponentiatedUtilities( + DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject) + { + + double[][] tazDistExpUtils = locChoiceSoaDistProbsObject + .computeDistanceUtilities(locChoiceDistSoaDmu); + for (int i = 0; i < tazDistExpUtils.length; i++) + for (int j = 0; j < tazDistExpUtils[i].length; j++) + { + if (tazDistExpUtils[i][j] < -500) tazDistExpUtils[i][j] = 0; + else tazDistExpUtils[i][j] = Math.exp(tazDistExpUtils[i][j]); + } + + return tazDistExpUtils; + + } + + private double[] computeTazSize(double[] size) + { + + double[] tazSize = new double[maxTaz + 1]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray != null) + { + for (int mgra : mgraArray) + { + tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); + } + } + + } + + return tazSize; + + } + + private void computeSizeSegmentProbabilities(double[][] sizeProbs, double[] size) + { + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + + if (mgraArray == null) + { + sizeProbs[taz - 1] = new double[0]; + } else + { + double totalSize = 0; + for (int mgra : mgraArray) + totalSize += size[mgra] + (size[mgra] > 0 ? 1 : 0); + + if (totalSize > 0) + { + sizeProbs[taz - 1] = new double[mgraArray.length]; + for (int i = 0; i < mgraArray.length; i++) + { + double mgraSize = size[mgraArray[i]]; + if (mgraSize > 0) mgraSize += 1; + sizeProbs[taz - 1][i] = mgraSize / totalSize; + } + } else if (sizeProbs[taz - 1] == null) + { + sizeProbs[taz - 1] = new double[0]; + } + } + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java new file mode 100644 index 0000000..695b2fc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java @@ -0,0 +1,965 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.text.DecimalFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.HashSet; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.datafile.CSVFileWriter; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * Handles building and storing destination choice size variables + * + */ + +public class DestChoiceSize + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(DestChoiceSize.class); + private transient Logger convergeLogger = Logger.getLogger("converge"); + + public static final String PROPERTIES_DC_SHADOW_OUTPUT = "uwsl.ShadowPricing.OutputFile"; + public static final String PROPERTIES_WORK_DC_SHADOW_NITER = "uwsl.ShadowPricing.Work.MaximumIterations"; + public static final String PROPERTIES_SCHOOL_DC_SHADOW_NITER = "uwsl.ShadowPricing.School.MaximumIterations"; + + private int numSegments; + private double[][] segmentSizeTerms; + private HashMap segmentIndexNameMap; + private HashMap segmentNameIndexMap; + private HashSet noShadowPriceSchoolSegmentIndices; + private MgraDataManager mgraManager; + + // 1st dimension is an index for the set of DC Size variables used in Sample + // of + // Alternative choice and destination choice, + // 2nd dimension is zone number (1,...,numZones), 3rd dimension walk subzone + // index is 0: no walk %, 1: shrt %, 2: long %. + protected double[][] dcSize; + protected double[][] originalSize; + protected double[][] originalAdjSize; + protected double[][] scaledSize; + protected double[][] balanceSize; + protected double[][] previousSize; + protected double[][] shadowPrice; + + protected double[][] externalFactors; + + protected int maxShadowPriceIterations; + + protected String dcShadowOutputFileName; + + protected boolean dcSizeCalculated = false; + + /** + * + * @param propertyMap + * is the model properties file key:value pairs + * @param segmentNameIndexMap + * is a map from segment name to size term array index. + * @param segmentIndexNameMap + * is a map from size term array index to segment name. + * @param segmentSizeTerms + * is an array by segment index and MGRA index + */ + public DestChoiceSize(HashMap propertyMap, + HashMap segmentIndexNameMap, + HashMap segmentNameIndexMap, double[][] segmentSizeTerms, + int maxIterations) + { + + this.segmentIndexNameMap = segmentIndexNameMap; + this.segmentNameIndexMap = segmentNameIndexMap; + this.segmentSizeTerms = segmentSizeTerms; + + // get the number of segments from the segmentIndexNameMap + numSegments = segmentIndexNameMap.size(); + + maxShadowPriceIterations = maxIterations; + + String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap, + CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + dcShadowOutputFileName = projectDirectory + propertyMap.get(PROPERTIES_DC_SHADOW_OUTPUT); + + mgraManager = MgraDataManager.getInstance(); + + // set default external factors array (all 1.0s) + // an object creating a destChoice object can override these valuse by + // calling setExternalFactors(). + externalFactors = new double[numSegments][mgraManager.getMaxMgra() + 1]; + for (int i = 0; i < numSegments; i++) + Arrays.fill(externalFactors[i], 1.0); + + } + + /** + * @return a boolean for whether or not the size terms in this object have + * been calculated + */ + public boolean getDcSizeCalculated() + { + return dcSizeCalculated; + } + + public HashMap getSegmentIndexNameMap() + { + return segmentIndexNameMap; + } + + public HashMap getSegmentNameIndexMap() + { + return segmentNameIndexMap; + } + + /** + * @return the maximum number of shadow price iterations set in the + * properties file + */ + public int getMaxShadowPriceIterations() + { + return maxShadowPriceIterations; + } + + public void setExternalFactors(double[][] factors) + { + externalFactors = factors; + } + + public void setNoShadowPriceSchoolSegmentIndices(HashSet indexSet) + { + noShadowPriceSchoolSegmentIndices = indexSet; + } + + /** + * Scale the destination choice size values so that the total modeled + * destinations by segment match the total origins. total Origin/Destination + * constraining usuallu done for home oriented mandatory tours, e.g. work + * university, school. This method also has the capability to read a file of + * destination size adjustments and apply them during the balancing + * procedure. This capability was used in the Morpc model and was + * transferred to the Baylanta project, but may or may not be used. + * + * @param originsByHomeZone + * - total long term choice origin locations (i.e. number of + * workers, university students, or school age students) in + * residence zone, subzone, by segment. + * + */ + public void balanceSizeVariables(int[][] originsByHomeMgra) + { + + // store the original size variable values. + // set the initial sizeBalance values to the original size variable + // values. + originalSize = duplicateDouble2DArray(segmentSizeTerms); + balanceSize = duplicateDouble2DArray(segmentSizeTerms); + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + // create the shadow price array - num + shadowPrice = new double[numSegments][maxMgra + 1]; + + // create the total origin locations array to store total tours by + // segment + double[] totalOriginLocations = new double[numSegments]; + + // create the total destination choice size array to store total tours + // by + // segment + double[] totalDestSize = new double[numSegments]; + + // initialize shadow prices with 1.0 + // accumulate total tours and size by segment. + for (int i = 0; i < numSegments; i++) + { + for (int j = 1; j <= maxMgra; j++) + { + shadowPrice[i][j] = 1.0; + totalOriginLocations[i] += originsByHomeMgra[i][j]; + totalDestSize[i] += (segmentSizeTerms[i][j] * externalFactors[i][j]); + } // j (mgra) + } // i (segment) + + // log a report of total origin locations by segment + logger.info(""); + logger.info("total origin locations by segment before any balancing, destination size adjustments, or shadow price scaling:"); + double segmentSum = 0.0; + for (int i = 0; i < numSegments; i++) + { + String segmentString = segmentIndexNameMap.get(i); + segmentSum += totalOriginLocations[i]; + logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, + totalOriginLocations[i])); + } // i + logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); + logger.info(""); + + // log a report of total destination choice size calculated by segment + logger.info(""); + logger.info("total destination choice size by segment before any balancing, destination choice size adjustments, or shadow price scaling:"); + segmentSum = 0.0; + for (int i = 0; i < numSegments; i++) + { + String segmentString = segmentIndexNameMap.get(i); + segmentSum += totalDestSize[i]; + logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, + totalDestSize[i])); + } + logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); + logger.info(""); + + // save original adjusted size variable arrays prior to balancing - used + // in + // reporting size variable calculations to output files. + originalAdjSize = duplicateDouble2DArray(balanceSize); + + // Balance destination choice size variables to equal total origin + // locations by segment. + // The scaledSize calculated is what is adjusted by shadow pricing + // adjustments, and dcSize, the array referenced + // by UEC DMUs is a duplicate copy of this array after the shadow + // pricing + // calculations are made. + scaledSize = new double[balanceSize.length][maxMgra + 1]; + double tot = 0.0; + for (int i = 0; i < numSegments; i++) + { + + tot = 0.0; + for (int j = 1; j <= maxMgra; j++) + { + + if (totalDestSize[i] > 0.0) scaledSize[i][j] = (balanceSize[i][j] + * externalFactors[i][j] * totalOriginLocations[i]) + / totalDestSize[i]; + else scaledSize[i][j] = 0.0f; + + tot += scaledSize[i][j]; + + } + + } + + // set destination choice size variables for the first iteration of + // shadow + // pricing to calculated scaled values + dcSize = duplicateDouble2DArray(scaledSize); + + // sum scaled destination size values by segment for reporting + double[] sumScaled = new double[numSegments]; + for (int i = 0; i < numSegments; i++) + { + for (int j = 1; j <= maxMgra; j++) + sumScaled[i] += scaledSize[i][j]; + } + + // log a report of total destination locations by segment + logger.info(""); + logger.info("total destination choice size by segment after destination choice size adjustments, after shadow price scaling:"); + segmentSum = 0.0; + for (int i = 0; i < numSegments; i++) + { + String segmentString = segmentIndexNameMap.get(i); + segmentSum += sumScaled[i]; + logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, sumScaled[i])); + } + logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); + logger.info(""); + + // save scaled size variables used in shadow price adjustmnents for + // reporting + // to output file + previousSize = new double[numSegments][]; + for (int i = 0; i < numSegments; i++) + previousSize[i] = duplicateDouble1DArray(scaledSize[i]); + + } + + public double getDcSize(int segmentIndex, int mgra) + { + return dcSize[segmentIndex][mgra]; + } + + public double getDcSize(String segmentName, int mgra) + { + int segmentIndex = segmentNameIndexMap.get(segmentName); + return dcSize[segmentIndex][mgra]; + } + + public double[][] getDcSizeArray() + { + return dcSize; + } + + public int getNumberOfSegments() + { + return dcSize.length; + } + + public void updateSizeVariables() + { + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + for (int i = 0; i < numSegments; i++) + { + for (int j = 1; j <= maxMgra; j++) + { + dcSize[i][j] = scaledSize[i][j] * shadowPrice[i][j]; + if (dcSize[i][j] < 0.0f) dcSize[i][j] = 0.0f; + } + } + + } + + public void updateShadowPrices(int[][] modeledDestinationLocationsByDestMgra) + { + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + for (int i = 0; i < numSegments; i++) + { + if (noShadowPriceSchoolSegmentIndices != null + && noShadowPriceSchoolSegmentIndices.contains(i)) continue; + + for (int j = 1; j <= maxMgra; j++) + { + if (modeledDestinationLocationsByDestMgra[i][j] > 0) + shadowPrice[i][j] *= (scaledSize[i][j] / modeledDestinationLocationsByDestMgra[i][j]); + // else + // shadowPrice[i][j] *= scaledSize[i][j]; + } + } + + } + + public void reportMaxDiff(int iteration, int[][] modeledDestinationLocationsByDestMgra) + { + + double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; + double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, Double.MAX_VALUE}; + + int[] nObs = new int[maxSize.length]; + double[] sse = new double[maxSize.length]; + double[] sumObs = new double[maxSize.length]; + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + logger.info("Shadow Price Iteration " + iteration); + + double minRange = 0.0; + for (int r = 0; r < maxSize.length; r++) + { + + logger.info(String + .format("Frequency of chosen mgra locations with non-zero DC Size < %s by range of relative error", + (maxSize[r] < 1000000 ? String.format("%.1f", maxSize[r]) : "+Inf"))); + logger.info(String.format("%-6s %-55s %15s %15s %15s %15s %15s %15s %15s %8s", + "index", "segment", "0 DCs", "< 5%", "< 10%", "< 25%", "< 50%", "< 100%", + "100% +", "Total")); + + int tot = 0; + int[] tots = new int[maxDeltas.length + 1]; + String logRecord = ""; + for (int i = 0; i < numSegments; i++) + { + + tot = 0; + int[] freqs = new int[maxDeltas.length + 1]; + int nonZeroSizeLocs = 0; + for (int j = 1; j <= maxMgra; j++) + { + + if (scaledSize[i][j] > minRange && scaledSize[i][j] <= maxSize[r]) + { + + nonZeroSizeLocs++; + + if (modeledDestinationLocationsByDestMgra[i][j] == 0.0) + { + // store the number of DC alternatives where DC Size + // > 0, + // but alternative was not chosen. + // relative error measure is not meaningful for this + // case, so report number of cases separately. + freqs[0]++; + + // calculations for %RMSE + sse[r] += scaledSize[i][j] * scaledSize[i][j]; + } else + { + + double relDiff = Math.abs(scaledSize[i][j] + - modeledDestinationLocationsByDestMgra[i][j]) + / scaledSize[i][j]; + for (int k = 0; k < maxDeltas.length; k++) + { + if (relDiff < maxDeltas[k]) + { + // store number of DC alternatives chosen + // where + // DC Size > 0, by relative error range. + freqs[k + 1]++; + break; + } + } + + // calculations for %RMSE + sse[r] += relDiff * relDiff; + } + + // calculations for %RMSE + sumObs[r] += scaledSize[i][j]; + nObs[r]++; + + } + + } + + for (int k = 0; k < freqs.length; k++) + { + tots[k] += freqs[k]; + tot += freqs[k]; + } + + String segmentString = segmentIndexNameMap.get(i); + logRecord = String.format("%-6d %-55s", i, segmentString); + + for (int k = 0; k < freqs.length; k++) + { + float pct = 0.0f; + if (tot > 0) pct = (float) (100.0 * freqs[k] / tot); + logRecord += String.format(" %6d (%5.1f%%)", freqs[k], pct); + } + + logRecord += String.format(" %8d", tot); + logger.info(logRecord); + + } + + tot = 0; + for (int k = 0; k < tots.length; k++) + { + tot += tots[k]; + } + + logRecord = String.format("%-6s %-55s", " ", "Total"); + String underline = String.format("------------------------"); + + for (int k = 0; k < tots.length; k++) + { + float pct = 0.0f; + if (tot > 0) pct = (float) (100.0 * tots[k] / tot); + logRecord += String.format(" %6d (%5.1f%%)", tots[k], pct); + underline += String.format("----------------"); + } + + logRecord += String.format(" %8d", tot); + underline += String.format("---------"); + + logger.info(underline); + logger.info(logRecord); + + double rmse = -1.0; + if (nObs[r] > 1) + rmse = 100.0 * (Math.sqrt(sse[r] / (nObs[r] - 1)) / (sumObs[r] / nObs[r])); + + logger.info("%RMSE = " + + (rmse < 0 ? "N/A, no observations" : String.format( + "%.1f, with mean %.1f, for %d observations.", rmse, + (sumObs[r] / nObs[r]), nObs[r]))); + + logger.info(""); + + minRange = maxSize[r]; + + } + + logger.info(""); + logger.info(""); + + } + + public void saveSchoolMaxDiffValues(int iteration, int[][] modeledDestinationLocationsByDestMgra) + { + + // define labels for the schoolsegment categories + String[] segmentRangelabels = {"Pre-School", "K-8", "9-12", "Univ"}; + + // define the highest index value for the range of segments for the + // school segment category + int[] segmentRange = {0, 36, 54, 56}; + + double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; + double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, 999.9}; + + int[][][] freqs = new int[segmentRangelabels.length][maxSize.length][maxDeltas.length]; + + int[][] nObs = new int[segmentRangelabels.length][maxSize.length]; + double[][] sse = new double[segmentRangelabels.length][maxSize.length]; + double[][] sumObs = new double[segmentRangelabels.length][maxSize.length]; + + double[][] rmse = new double[segmentRangelabels.length][maxSize.length]; + double[][] meanSize = new double[segmentRangelabels.length][maxSize.length]; + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + convergeLogger.info("School Shadow Price Iteration " + iteration); + + double[] minRange = new double[segmentRangelabels.length]; + + int minS = 0; + for (int s = 0; s < segmentRangelabels.length; s++) + { + + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(segmentRangelabels[s] + " convergence statistics"); + + if (s > 0) minS = segmentRange[s - 1] + 1; + + for (int r = 0; r < maxSize.length; r++) + { + + for (int i = minS; i <= segmentRange[s]; i++) + { + + for (int j = 1; j <= maxMgra; j++) + { + + if (scaledSize[i][j] > minRange[s] && scaledSize[i][j] <= maxSize[r]) + { + + if (modeledDestinationLocationsByDestMgra[i][j] > 0.0) + { + + int delta = maxDeltas.length - 1; + double diff = Math.abs(scaledSize[i][j] + - modeledDestinationLocationsByDestMgra[i][j]); + double relDiff = diff / scaledSize[i][j]; + for (int k = 0; k < maxDeltas.length; k++) + { + if (relDiff < maxDeltas[k]) + { + delta = k; + break; + } + } + + freqs[s][r][delta]++; + + // calculations for %RMSE + sse[s][r] += (diff * diff); + + } + + // calculations for %RMSE + sumObs[s][r] += scaledSize[i][j]; + nObs[s][r]++; + + } + + } + + } + + rmse[s][r] = -1.0; + if (nObs[s][r] > 1) + { + meanSize[s][r] = sumObs[s][r] / nObs[s][r]; + rmse[s][r] = 100.0 * (Math.sqrt((sse[s][r] / (nObs[s][r] - 1))) / meanSize[s][r]); + } + + minRange[s] = maxSize[r]; + + } + + convergeLogger.info("%RMSE by DC Size Range Category"); + for (int i = 0; i < maxSize.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], rmse[s][i])); + convergeLogger.info(String + .format("%-8s %14.2f", " 1000+", rmse[s][maxSize.length - 1])); + + convergeLogger.info(""); + + convergeLogger.info("%Mean DC Size by DC Size Range Category"); + for (int i = 0; i < maxSize.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], meanSize[s][i])); + convergeLogger.info(String.format("%-8s %14.2f", " 1000+", + meanSize[s][maxSize.length - 1])); + + convergeLogger.info(""); + + convergeLogger.info("Freq of MGRAs by DC Size Range Category and Relative Error"); + for (int r = 0; r < maxSize.length - 1; r++) + { + + convergeLogger.info(String.format("Size < %-8.0f", maxSize[r])); + for (int i = 0; i < maxDeltas.length - 1; i++) + convergeLogger.info(String + .format("< %-8.2f %12d", maxDeltas[i], freqs[s][r][i])); + convergeLogger.info(String.format("%-8s %14d", " 1.0+", + freqs[s][r][maxDeltas.length - 1])); + + convergeLogger.info(""); + } + + convergeLogger.info(String.format("Size >= 1000")); + for (int i = 0; i < maxDeltas.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12d", maxDeltas[i], + freqs[s][maxSize.length - 1][i])); + convergeLogger.info(String.format("%-8s %14d", " 1.0+", + freqs[s][maxSize.length - 1][maxDeltas.length - 1])); + + convergeLogger.info(""); + + } + + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(""); + + } + + public void saveWorkMaxDiffValues(int iteration, int[][] modeledDestinationLocationsByDestMgra) + { + + // define labels for the schoolsegment categories + String[] segmentRangelabels = {"White Collar", "Services", "Health", "Retail and Food", + "Blue Collar", "Military"}; + + // define the highest index value for the range of segments for the + // school segment category + int[] segmentRange = {0, 1, 2, 3, 4, 5}; + + double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; + double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, 999.9}; + + int[][][] freqs = new int[segmentRangelabels.length][maxSize.length][maxDeltas.length]; + + int[][] nObs = new int[segmentRangelabels.length][maxSize.length]; + double[][] sse = new double[segmentRangelabels.length][maxSize.length]; + double[][] sumObs = new double[segmentRangelabels.length][maxSize.length]; + + double[][] rmse = new double[segmentRangelabels.length][maxSize.length]; + double[][] meanSize = new double[segmentRangelabels.length][maxSize.length]; + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + convergeLogger.info("Work Shadow Price Iteration " + iteration); + + double[] minRange = new double[segmentRangelabels.length]; + + int minS = 0; + for (int s = 0; s < segmentRangelabels.length; s++) + { + + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(segmentRangelabels[s] + " convergence statistics"); + + if (s > 0) minS = segmentRange[s - 1] + 1; + + for (int r = 0; r < maxSize.length; r++) + { + + for (int i = minS; i <= segmentRange[s]; i++) + { + + for (int j = 1; j <= maxMgra; j++) + { + + if (scaledSize[i][j] > minRange[s] && scaledSize[i][j] <= maxSize[r]) + { + + if (modeledDestinationLocationsByDestMgra[i][j] > 0.0) + { + + int delta = maxDeltas.length - 1; + double diff = Math.abs(scaledSize[i][j] + - modeledDestinationLocationsByDestMgra[i][j]); + double relDiff = diff / scaledSize[i][j]; + for (int k = 0; k < maxDeltas.length; k++) + { + if (relDiff < maxDeltas[k]) + { + delta = k; + break; + } + } + + freqs[s][r][delta]++; + + // calculations for %RMSE + sse[s][r] += (diff * diff); + + } + + // calculations for %RMSE + sumObs[s][r] += scaledSize[i][j]; + nObs[s][r]++; + + } + + } + + } + + rmse[s][r] = -1.0; + if (nObs[s][r] > 1) + { + meanSize[s][r] = sumObs[s][r] / nObs[s][r]; + rmse[s][r] = 100.0 * (Math.sqrt((sse[s][r] / (nObs[s][r] - 1))) / meanSize[s][r]); + } + + minRange[s] = maxSize[r]; + + } + + convergeLogger.info("%RMSE by DC Size Range Category"); + for (int i = 0; i < maxSize.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], rmse[s][i])); + convergeLogger.info(String + .format("%-8s %14.2f", " 1000+", rmse[s][maxSize.length - 1])); + + convergeLogger.info(""); + + convergeLogger.info("%Mean DC Size by DC Size Range Category"); + for (int i = 0; i < maxSize.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], meanSize[s][i])); + convergeLogger.info(String.format("%-8s %14.2f", " 1000+", + meanSize[s][maxSize.length - 1])); + + convergeLogger.info(""); + + convergeLogger.info("Freq of MGRAs by DC Size Range Category and Relative Error"); + for (int r = 0; r < maxSize.length - 1; r++) + { + + convergeLogger.info(String.format("Size < %-8.0f", maxSize[r])); + for (int i = 0; i < maxDeltas.length - 1; i++) + convergeLogger.info(String + .format("< %-8.2f %12d", maxDeltas[i], freqs[s][r][i])); + convergeLogger.info(String.format("%-8s %14d", " 1.0+", + freqs[s][r][maxDeltas.length - 1])); + + convergeLogger.info(""); + } + + convergeLogger.info(String.format("Size >= 1000")); + for (int i = 0; i < maxDeltas.length - 1; i++) + convergeLogger.info(String.format("< %-8.2f %12d", maxDeltas[i], + freqs[s][maxSize.length - 1][i])); + convergeLogger.info(String.format("%-8s %14d", " 1.0+", + freqs[s][maxSize.length - 1][maxDeltas.length - 1])); + + convergeLogger.info(""); + } + + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(""); + convergeLogger.info(""); + + } + + public boolean getSegmentIsInSkipSegmentSet(int segment) + { + return noShadowPriceSchoolSegmentIndices.contains(segment); + } + + public void updateShadowPricingInfo(int iteration, int[][] originsByHomeMgra, + int[][] modeledDestinationLocationsByDestMgra, String mandatoryType) + { + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + ArrayList tableHeadings = new ArrayList(); + tableHeadings.add("alt"); + tableHeadings.add("mgra"); + + for (int i = 0; i < numSegments; i++) + { + + String segmentString = segmentIndexNameMap.get(i); + + tableHeadings.add(String.format("%s_origins", segmentString)); + tableHeadings.add(String.format("%s_sizeOriginal", segmentString)); + tableHeadings.add(String.format("%s_sizeAdjOriginal", segmentString)); + tableHeadings.add(String.format("%s_sizeScaled", segmentString)); + tableHeadings.add(String.format("%s_sizePrevious", segmentString)); + tableHeadings.add(String.format("%s_modeledDests", segmentString)); + tableHeadings.add(String.format("%s_sizeFinal", segmentString)); + tableHeadings.add(String.format("%s_shadowPrices", segmentString)); + + } + + // define a TableDataSet for use in writing output file + float[][] tableData = new float[maxMgra + 1][tableHeadings.size()]; + + int alt = 0; + for (int i = 1; i <= maxMgra; i++) + { + + tableData[alt][0] = alt + 1; + tableData[alt][1] = i; + + int index = 2; + + for (int p = 0; p < numSegments; p++) + { + tableData[alt][index++] = (float) originsByHomeMgra[p][i]; + tableData[alt][index++] = (float) originalSize[p][i]; + tableData[alt][index++] = (float) originalAdjSize[p][i]; + tableData[alt][index++] = (float) scaledSize[p][i]; + tableData[alt][index++] = (float) previousSize[p][i]; + tableData[alt][index++] = (float) modeledDestinationLocationsByDestMgra[p][i]; + tableData[alt][index++] = (float) dcSize[p][i]; + tableData[alt][index++] = (float) shadowPrice[p][i]; + } + alt++; + + } + + TableDataSet outputTable = TableDataSet.create(tableData, tableHeadings); + + // write outputTable to new output file + try + { + String newFilename = this.dcShadowOutputFileName.replaceFirst(".csv", "_" + + mandatoryType + "_" + iteration + ".csv"); + CSVFileWriter writer = new CSVFileWriter(); + writer.writeFile(outputTable, new File(newFilename), + new DecimalFormat("#.000000000000")); + // writer.writeFile( outputTable, new File(newFilename) ); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + // save scaled size variables used in shadow price adjustmnents for + // reporting + // to output file + for (int i = 0; i < numSegments; i++) + previousSize[i] = duplicateDouble1DArray(dcSize[i]); + + } + + public void restoreShadowPricingInfo(String fileName) + { + + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + + TableDataSet tds = null; + try + { + tds = reader.readFileAsDouble(new File(fileName)); + } catch (IOException e) + { + logger.error("exception reading saved shadow price file: " + fileName + + " from previous model run.", e); + } + + // the following are based on format used to write the shadow pricing + // file + // first three columns are indices: ALT, ZONE, SUBZONE. + int columnIndex = 2; + int numberOfColumnsPerPurpose = 8; + int scaledSizeColumnOffset = 3; + int previousSizeColumnOffset = 4; + int finalSizeColumnOffset = 6; + int finalShadowPriceOffset = 7; + + // get the number of MGRAs + int maxMgra = mgraManager.getMaxMgra(); + + for (int i = 0; i < numSegments; i++) + { + + // first restore the scaled size values; getColumnAsFloat(column) + // takes a + // 1s based column value, returns a 0s based array of values + int column = columnIndex + i * numberOfColumnsPerPurpose + scaledSizeColumnOffset + 1; + double[] columnData = tds.getColumnAsDoubleFromDouble(column); + for (int z = 1; z <= maxMgra; z++) + scaledSize[i][z] = columnData[z - 1]; + + // next restore the final size values + column = columnIndex + i * numberOfColumnsPerPurpose + finalSizeColumnOffset + 1; + columnData = tds.getColumnAsDoubleFromDouble(column); + for (int z = 1; z <= maxMgra; z++) + dcSize[i][z] = columnData[z - 1]; + + // next restore the previous size values from the final size of the + // previous iteration + column = columnIndex + i * numberOfColumnsPerPurpose + finalSizeColumnOffset + 1; + columnData = tds.getColumnAsDoubleFromDouble(column); + for (int z = 1; z <= maxMgra; z++) + previousSize[i][z] = columnData[z - 1]; + + // finally restore the final shadow price values + column = columnIndex + i * numberOfColumnsPerPurpose + finalShadowPriceOffset + 1; + columnData = tds.getColumnAsDoubleFromDouble(column); + for (int z = 1; z <= maxMgra; z++) + shadowPrice[i][z] = columnData[z - 1]; + + } + + } + + /** + * Create a new double[], dimension it exactly as the argument array, and + * copy the element values from the argument array to the new one. + * + * @param in + * a 1-dimension double array to be duplicated + * @return an exact duplicate of the argument array + */ + private double[] duplicateDouble1DArray(double[] in) + { + double[] out = new double[in.length]; + for (int i = 0; i < in.length; i++) + { + out[i] = in[i]; + } + return out; + } + + /** + * Create a new double[][], dimension it exactly as the argument array, and + * copy the element values from the argument array to the new one. + * + * @param in + * a 2-dimensional double array to be duplicated + * @return an exact duplicate of the argument array + */ + private double[][] duplicateDouble2DArray(double[][] in) + { + double[][] out = new double[in.length][]; + for (int i = 0; i < in.length; i++) + { + out[i] = new double[in[i].length]; + for (int j = 0; j < in[i].length; j++) + { + out[i][j] = in[i][j]; + } + } + return out; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java new file mode 100644 index 0000000..afb76d0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java @@ -0,0 +1,625 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +public class DestChoiceTwoStageModel +{ + + private transient Logger logger = Logger.getLogger(DestChoiceTwoStageModel.class); + + // dimensioned for maxMgra, holds the number of times a specific mgra was + // selected to be in the the sample + private int[] mgraSampleFreq; + + // these arrays are dimesnioned to the maxSampleSize and hold values mgra + // selected for the sample. + private int[] sampleMgras; + private double[] sampleProbabilities; + + // these arrays are dimesnioned to the maxSampleSize, but hold values up to + // the number of unique mgras selected for the sample. + // array values after the number of unique selected mgras are default array + // values. + private int[] uniqueMgraSample; + private double[] uniqueCorrectionFactors; + + // this array holds the sampleIndex associated with a unique mgra, and is + // used to lookup the sample probability for the unique mgra. + private int[] uniqueSampleIndices; + + // use this variable to keep track of the number of unique sampled mgras + // while choosing the sample + private int uniqueIndex; + + private TazDataManager tdm; + private MgraDataManager mgraManager; + + // for each purpose index, the 2D array is 0-based on origTaz and is 0-based + // on destTaz, giving cumulative taz distance probabilities. + private double[][][] tazDistCumProbs; + + // for each purpose index, the 2D array is 0-based on TAZ and is 0-based on + // MGRAs in the taz, giving size probabilities for MGRAs in the TAZ. + private double[][][] mgraSizeProbs; + + private double[][][] slcSizeProbs; + private double[][] slcTazSize; + private double[][] slcTazDistExpUtils; + + // create an array to re-use to hold cumulative probabilities for selecting + // an MGRA from a TAZ. + private double[] tempMgraCumProbs = new double[200]; + + private double[] slcTazProbs; + private double[] slcTazCumProbs; + private int maxTaz; + + private long soaRunTime; + + public DestChoiceTwoStageModel(HashMap propertyMap, int soaMaxSampleSize) + { + + mgraManager = MgraDataManager.getInstance(propertyMap); + int maxMgra = mgraManager.getMaxMgra(); + + tdm = TazDataManager.getInstance(propertyMap); + maxTaz = tdm.getMaxTaz(); + + slcTazProbs = new double[maxTaz]; + slcTazCumProbs = new double[maxTaz]; + + mgraSampleFreq = new int[maxMgra + 1]; + + sampleMgras = new int[soaMaxSampleSize]; + sampleProbabilities = new double[soaMaxSampleSize]; + + uniqueMgraSample = new int[soaMaxSampleSize]; + uniqueCorrectionFactors = new double[soaMaxSampleSize]; + + uniqueSampleIndices = new int[soaMaxSampleSize]; + } + + private void resetSampleArrays() + { + Arrays.fill(mgraSampleFreq, 0); + Arrays.fill(sampleMgras, 0); + Arrays.fill(sampleProbabilities, 0); + Arrays.fill(uniqueMgraSample, 0); + Arrays.fill(uniqueSampleIndices, -1); + Arrays.fill(uniqueCorrectionFactors, 0); + uniqueIndex = 0; + } + + /** + * get the array of unique mgras selected in the sample. The number of + * unique mgras may be fewer than the number selected for the sample - the + * overall sample size. If so, values in this array from + * 0,...,numUniqueMgras-1 will be the selected unique mgras, and values from + * numUniqueMgras,...,maxSampleSize-1 will be 0. + * + * @return uniqueMgraSample array. + */ + public int[] getUniqueSampleMgras() + { + return uniqueMgraSample; + } + + /** + * get the number of unique mgra values in the sample. It gives the + * upperbound of unique values in uniqueMgraSample[0,...,numUniqueMgras-1]. + * + * @return number of unique mgra values in the sample + */ + public int getNumberofUniqueMgrasInSample() + { + return uniqueIndex; + } + + public double[] getUniqueSampleMgraCorrectionFactors() + { + + for (int i = 0; i < uniqueIndex; i++) + { + int chosenMgra = uniqueMgraSample[i]; + int freq = mgraSampleFreq[chosenMgra]; + + int sampleIndex = uniqueSampleIndices[i]; + double prob = sampleProbabilities[sampleIndex]; + + uniqueCorrectionFactors[i] = (float) Math.log((double) freq / prob); + } + + return uniqueCorrectionFactors; + } + + public void computeSoaProbabilities(int origTaz, int segmentTypeIndex) + { + + double[][] sizeProbs = mgraSizeProbs[segmentTypeIndex]; + double[] probs = new double[mgraManager.getMaxMgra() + 1]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray == null) continue; + + if (sizeProbs[taz - 1].length == 0) continue; + + double tazProb = 0; + if (taz > 1) tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][taz - 1] + - tazDistCumProbs[segmentTypeIndex][origTaz - 1][taz - 2]; + else tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; + + for (int mgraIndex = 0; mgraIndex < mgraArray.length; mgraIndex++) + { + double mgraProb = sizeProbs[taz - 1][mgraIndex]; + probs[mgraArray[mgraIndex]] = tazProb * mgraProb; + } + + } + + PrintWriter out = null; + try + { + out = new PrintWriter(new BufferedWriter(new FileWriter(new File("distSoaProbs.csv")))); + + for (int i = 1; i < probs.length; i++) + { + out.println(i + "," + probs[i]); + } + } catch (IOException e) + { + // TODO Auto-generated catch block + e.printStackTrace(); + } + + out.close(); + + } + + public void chooseSampleMgra(int sampleIndex, int origTaz, int sizePurposeIndex, + int segmentTypeIndex, double rn, boolean debug) + { + + // get the chosen TAZ array index for the 0-based cumulative TAZ + // distance probabilities array + int chosenTazIndex = Util.binarySearchDouble( + tazDistCumProbs[segmentTypeIndex][origTaz - 1], rn); + + if (mgraSizeProbs[segmentTypeIndex][chosenTazIndex].length == 0) + { + logger.error("The MGRA size probabilities array for chosen TAZ index = " + + chosenTazIndex + " has 0 length."); + logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); + logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex + + ", origTaz=" + origTaz + ", sizePurposeIndex=" + sizePurposeIndex + + ", segmentTypeIndex=" + segmentTypeIndex); + throw new RuntimeException(); + } + + // get the chosen TAZ distance probability from the taz distance + // cumulative probabilities array + // also initialize the 0 index cumulative MGRA probability to the + // cumulative taz distance propbaility + double tazProb = 0; + double cumProbabilityLowerBound = 0; + if (chosenTazIndex > 0) + { + tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex] + - tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; + cumProbabilityLowerBound = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; + } else + { + tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; + cumProbabilityLowerBound = 0; + } + + // get the array of MGRAs for the chosen TAZ (the chosen index + 1) + int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); + + // get the unscaled MGRA size probability, scale by the TAZ distance + // probability, and accumulate cumulative probabilities + tempMgraCumProbs[0] = cumProbabilityLowerBound + + (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][0] * tazProb); + for (int i = 1; i < mgraArray.length; i++) + tempMgraCumProbs[i] = tempMgraCumProbs[i - 1] + + (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][i] * tazProb); + + // get the chosen array index for the 0-based cumulative probabilities + // array + int chosenMgraIndex = Util.binarySearchDouble(cumProbabilityLowerBound, tempMgraCumProbs, + mgraArray.length, rn); + + // use the chosen mgra index to get the chosenMgra value from the + // 0-based array of MGRAs associated with the chosen TAZ + int chosenMgra = mgraArray[chosenMgraIndex]; + + // store the sampled mgra and its selection probability + sampleMgras[sampleIndex] = chosenMgra; + sampleProbabilities[sampleIndex] = (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][chosenMgraIndex] * tazProb); + + // if the sample freq is 0, this mgra has not been selected yet, so add + // it to the array of unique sampled mgras. + if (mgraSampleFreq[chosenMgra] == 0) + { + uniqueMgraSample[uniqueIndex] = chosenMgra; + uniqueSampleIndices[uniqueIndex] = sampleIndex; + uniqueIndex++; + } + + // increment the frequency of times this mgra was selected for the + // sample + mgraSampleFreq[chosenMgra]++; + + if (debug) + { + + double cumDistProb = 0; + double prevDistCumProb = 0; + if (chosenTazIndex > 1) + { + cumDistProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex]; + prevDistCumProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; + } else + { + cumDistProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; + prevDistCumProb = 0; + } + + double cumSizeProb = 0; + double prevSizeCumProb = 0; + if (chosenMgraIndex > 0) + { + cumSizeProb = tempMgraCumProbs[chosenMgraIndex]; + prevSizeCumProb = tempMgraCumProbs[chosenMgraIndex - 1]; + } else + { + cumSizeProb = tempMgraCumProbs[0]; + prevSizeCumProb = 0; + } + + logger.info(String.format( + "%-12d %10d %10.6f %16.8f %16.8f %18d %18.8f %18.8f %12d %18.8f", sampleIndex, + chosenTazIndex, rn, prevDistCumProb, cumDistProb, chosenMgraIndex, + prevSizeCumProb, cumSizeProb, chosenMgra, + ((cumSizeProb - prevSizeCumProb) * (cumDistProb - prevDistCumProb)))); + } + + } + + private void chooseSlcSampleMgraBinarySearch(int sampleIndex, int slcOrigTaz, int slcDestTaz, + int slcSizeSegmentIndex, double rn, boolean debug) + { + + // compute stop location sample probabilities from the pre-computed + // sample exponentiated utilities and taz size terms. + // first compute exponentiated utilites for each alternative from the + // pre-computed component exponentiated utilities + double totalExponentiatedUtility = 0; + for (int k = 0; k < maxTaz; k++) + { + slcTazProbs[k] = (slcTazDistExpUtils[slcOrigTaz - 1][k] + * slcTazDistExpUtils[k][slcDestTaz - 1] / slcTazDistExpUtils[slcOrigTaz - 1][slcDestTaz - 1]) + * slcTazSize[slcSizeSegmentIndex][k + 1]; + totalExponentiatedUtility += slcTazProbs[k]; + } + + // now compute alterantive probabilities and determine selected + // alternative + slcTazCumProbs[0] = slcTazProbs[0] / totalExponentiatedUtility; + for (int k = 1; k < maxTaz - 1; k++) + slcTazCumProbs[k] = slcTazCumProbs[k - 1] + + (slcTazProbs[k] / totalExponentiatedUtility); + slcTazCumProbs[maxTaz - 1] = 1.0; + + // get the chosen TAZ array index for the 0-based cumulative TAZ + // distance probabilities array + int chosenTazIndex = Util.binarySearchDouble(slcTazCumProbs, rn); + + /* + * // now compute alterantive probabilities and determine selected + * alternative int chosenTazIndex0 = -1; double sum = slcTazProbs[0] / + * totalExponentiatedUtility; if ( rn < sum ) { chosenTazIndex0 = 0; } + * else { for ( int k=1; k < maxTaz; k++ ) { slcTazProbs[k] /= + * totalExponentiatedUtility; sum += slcTazProbs[k]; if ( rn < sum ) { + * chosenTazIndex0 = k; break; } } } + * + * + * if ( chosenTazIndex0 != chosenTazIndex ) { logger.error ( + * "error - inconsistent choices made by two alternative monte carlo methods. " + * ); System.exit(-1); } + */ + + if (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex].length == 0) + { + logger.error("The MGRA size probabilities array for chosen stop location TAZ index = " + + chosenTazIndex + " has 0 length."); + logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); + logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex + + ", slcOrigTaz=" + slcOrigTaz + ", slcDestTaz=" + slcDestTaz + + ", slcSizeSegmentIndex=" + slcSizeSegmentIndex); + throw new RuntimeException(); + } + + // get the chosen SLC TAZ distance probability from the taz distance + // cumulative probabilities array + // also initialize the 0 index cumulative MGRA probability to the + // cumulative taz distance propbaility + double tazProb = 0; + double cumProbabilityLowerBound = 0; + if (chosenTazIndex > 0) + { + tazProb = slcTazCumProbs[chosenTazIndex] - slcTazCumProbs[chosenTazIndex - 1]; + cumProbabilityLowerBound = slcTazCumProbs[chosenTazIndex - 1]; + } else + { + tazProb = slcTazCumProbs[0]; + cumProbabilityLowerBound = 0; + } + + // get the array of MGRAs for the chosen TAZ (the chosen index + 1) + int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); + + // get the unscaled MGRA size probability, scale by the TAZ distance + // probability, and accumulate cumulative probabilities + tempMgraCumProbs[0] = cumProbabilityLowerBound + + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb); + for (int i = 1; i < mgraArray.length; i++) + tempMgraCumProbs[i] = tempMgraCumProbs[i - 1] + + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][i] * tazProb); + + // get the chosen array index for the 0-based cumulative probabilities + // array + int chosenMgraIndex = Util.binarySearchDouble(cumProbabilityLowerBound, tempMgraCumProbs, + mgraArray.length, rn); + + // use the chosen mgra index to get the chosenMgra value from the + // 0-based array of MGRAs associated with the chosen TAZ + int chosenMgra = mgraArray[chosenMgraIndex]; + + // store the sampled mgra and its selection probability + sampleMgras[sampleIndex] = chosenMgra; + sampleProbabilities[sampleIndex] = (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][chosenMgraIndex] * tazProb); + + // if the sample freq is 0, this mgra has not been selected yet, so add + // it to the array of unique sampled mgras. + if (mgraSampleFreq[chosenMgra] == 0) + { + uniqueMgraSample[uniqueIndex] = chosenMgra; + uniqueSampleIndices[uniqueIndex] = sampleIndex; + uniqueIndex++; + } + + // increment the frequency of times this mgra was selected for the + // sample + mgraSampleFreq[chosenMgra]++; + + } + + private void chooseSlcSampleMgraLinearWalk(int sampleIndex, int slcOrigTaz, int slcDestTaz, + int slcSizeSegmentIndex, double rn, boolean debug) + { + + // compute stop location sample probabilities from the pre-computed + // sample exponentiated utilities and taz size terms. + // first compute exponentiated utilites for each alternative from the + // pre-computed component exponentiated utilities + double totalExponentiatedUtility = 0; + for (int k = 0; k < maxTaz; k++) + { + slcTazProbs[k] = (slcTazDistExpUtils[slcOrigTaz - 1][k] + * slcTazDistExpUtils[k][slcDestTaz - 1] / slcTazDistExpUtils[slcOrigTaz - 1][slcDestTaz - 1]) + * slcTazSize[slcSizeSegmentIndex][k + 1]; + totalExponentiatedUtility += slcTazProbs[k]; + } + + /* + * // now compute alterantive probabilities and determine selected + * alternative slcTazCumProbs[0] = slcTazProbs[0] / + * totalExponentiatedUtility; for ( int k=1; k < maxTaz - 1; k++ ) + * slcTazCumProbs[k] = slcTazCumProbs[k-1] + (slcTazProbs[k] / + * totalExponentiatedUtility); slcTazCumProbs[maxTaz - 1] = 1.0; + * + * + * /* // get the chosen TAZ array index for the 0-based cumulative TAZ + * distance probabilities array int chosenTazIndex0 = + * Util.binarySearchDouble( slcTazCumProbs, rn ); + */ + + // now compute alterantive probabilities and determine selected + // alternative + int chosenTazIndex = -1; + double sum = slcTazProbs[0] / totalExponentiatedUtility; + double cumProbabilityLowerBound = 0; + double tazProb = 0; + if (rn < sum) + { + chosenTazIndex = 0; + tazProb = sum; + } else + { + for (int k = 1; k < maxTaz; k++) + { + tazProb = slcTazProbs[k] / totalExponentiatedUtility; + cumProbabilityLowerBound = sum; + sum += tazProb; + if (rn < sum) + { + chosenTazIndex = k; + break; + } + } + } + + /* + * if ( chosenTazIndex0 != chosenTazIndex ) { logger.error ( + * "error - inconsistent choices made by two alternative monte carlo methods. " + * ); System.exit(-1); } + */ + + if (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex].length == 0) + { + logger.error("The MGRA size probabilities array for chosen stop location TAZ index = " + + chosenTazIndex + " has 0 length."); + logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); + logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex + + ", slcOrigTaz=" + slcOrigTaz + ", slcDestTaz=" + slcDestTaz + + ", slcSizeSegmentIndex=" + slcSizeSegmentIndex); + throw new RuntimeException(); + } + + /* + * // get the chosen SLC TAZ distance probability from the taz distance + * cumulative probabilities array // also initialize the 0 index + * cumulative MGRA probability to the cumulative taz distance + * propbaility double tazProb = 0; double cumProbabilityLowerBound = 0; + * if ( chosenTazIndex > 0 ) { tazProb = slcTazCumProbs[chosenTazIndex] + * - slcTazCumProbs[chosenTazIndex-1]; cumProbabilityLowerBound = + * slcTazCumProbs[chosenTazIndex-1]; } else { tazProb = + * slcTazCumProbs[0]; cumProbabilityLowerBound = 0; } + */ + + // get the array of MGRAs for the chosen TAZ (the chosen index + 1) + int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); + + /* + * // get the unscaled MGRA size probability, scale by the TAZ distance + * probability, and accumulate cumulative probabilities + * tempMgraCumProbs[0] = cumProbabilityLowerBound + ( + * slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb ); for + * ( int i=1; i < mgraArray.length; i++ ) tempMgraCumProbs[i] = + * tempMgraCumProbs[i-1] + ( + * slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][i] * tazProb ); + */ + + // now compute alterantive probabilities and determine selected + // alternative + int chosenMgraIndex = -1; + sum = cumProbabilityLowerBound + + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb); + if (rn < sum) + { + chosenMgraIndex = 0; + } else + { + for (int k = 1; k < mgraArray.length; k++) + { + sum += (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][k] * tazProb); + if (rn < sum) + { + chosenMgraIndex = k; + break; + } + } + } + + /* + * // get the chosen array index for the 0-based cumulative + * probabilities array int chosenMgraIndex = Util.binarySearchDouble( + * cumProbabilityLowerBound, tempMgraCumProbs, mgraArray.length, rn ); + */ + + // use the chosen mgra index to get the chosenMgra value from the + // 0-based array of MGRAs associated with the chosen TAZ + int chosenMgra = mgraArray[chosenMgraIndex]; + + // store the sampled mgra and its selection probability + sampleMgras[sampleIndex] = chosenMgra; + sampleProbabilities[sampleIndex] = (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][chosenMgraIndex] * tazProb); + + // if the sample freq is 0, this mgra has not been selected yet, so add + // it to the array of unique sampled mgras. + if (mgraSampleFreq[chosenMgra] == 0) + { + uniqueMgraSample[uniqueIndex] = chosenMgra; + uniqueSampleIndices[uniqueIndex] = sampleIndex; + uniqueIndex++; + } + + // increment the frequency of times this mgra was selected for the + // sample + mgraSampleFreq[chosenMgra]++; + + } + + public void chooseSample(int origTaz, int sizeSegmentIndex, int segmentTypeIndex, + int numInSample, Random rand, boolean debug) + { + + long timeCheck = System.nanoTime(); + + if (debug) + { + computeSoaProbabilities(origTaz, segmentTypeIndex); + } + + resetSampleArrays(); + for (int i = 0; i < numInSample; i++) + { + chooseSampleMgra(i, origTaz, sizeSegmentIndex, segmentTypeIndex, rand.nextDouble(), + debug); + } + + soaRunTime += (System.nanoTime() - timeCheck); + + } + + public void chooseSlcSample(int origTaz, int destTaz, int sizeSegmentIndex, int numInSample, + Random rand, boolean debug) + { + + long timeCheck = System.nanoTime(); + + resetSampleArrays(); + for (int i = 0; i < numInSample; i++) + { + // chooseSlcSampleMgraBinarySearch( i, origTaz, destTaz, + // sizeSegmentIndex, rand.nextDouble(), debug ); + chooseSlcSampleMgraLinearWalk(i, origTaz, destTaz, sizeSegmentIndex, rand.nextDouble(), + debug); + } + + soaRunTime += (System.nanoTime() - timeCheck); + + } + + public void setSlcSoaProbsAndUtils(double[][] slcTazDistExpUtils, double[][][] slcSizeProbs, + double[][] slcTazSize) + { + this.slcSizeProbs = slcSizeProbs; + this.slcTazSize = slcTazSize; + this.slcTazDistExpUtils = slcTazDistExpUtils; + } + + public void setMgraSizeProbs(double[][][] probs) + { + mgraSizeProbs = probs; + } + + public void setTazDistProbs(double[][][] probs) + { + tazDistCumProbs = probs; + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java new file mode 100644 index 0000000..a23453c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java @@ -0,0 +1,408 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public abstract class DestChoiceTwoStageModelDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(DestChoiceTwoStageModelDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Person person; + protected Tour tour; + protected IndexValues dmuIndex = null; + + protected double workAccessibility; + protected double nonMandatoryAccessibility; + + protected double[] homeMgraNonMandatoryAccessibilityArray; + protected double[] homeMgraTotalEmploymentAccessibilityArray; + + protected int[] sampleMgras; + protected double[] modeChoiceLogsums; + protected double[] dcSoaCorrections; + + protected double[] mgraSizeArray; + protected double[] mgraDistanceArray; + + protected int toursLeftCount; + + protected ModelStructure modelStructure; + protected MgraDataManager mgraManager; + protected BuildAccessibilities aggAcc; + protected AccessibilitiesTable accTable; + + public DestChoiceTwoStageModelDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + initDmuObject(); + } + + public abstract void setMcLogsum(int mgra, double logsum); + + private void initDmuObject() + { + + dmuIndex = new IndexValues(); + + // create default objects - some choice models use these as place + // holders for values + person = new Person(null, -1, modelStructure); + hh = new Household(modelStructure); + + mgraManager = MgraDataManager.getInstance(); + + int maxMgra = mgraManager.getMaxMgra(); + + modeChoiceLogsums = new double[maxMgra + 1]; + dcSoaCorrections = new double[maxMgra + 1]; + + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setPersonObject(Person personObject) + { + person = personObject; + } + + public void setTourObject(Tour tour) + { + this.tour = tour; + } + + public void setAggAcc(BuildAccessibilities aggAcc) + { + this.aggAcc = aggAcc; + } + + public void setAccTable(AccessibilitiesTable myAccTable) + { + accTable = myAccTable; + } + + public void setMgraSizeArray(double[] mgraSizeArray) + { + this.mgraSizeArray = mgraSizeArray; + } + + public void setMgraDistanceArray(double[] mgraDistanceArray) + { + this.mgraDistanceArray = mgraDistanceArray; + } + + public void setSampleArray(int[] sampleArray) + { + sampleMgras = sampleArray; + } + + public void setDcSoaCorrections(double[] sampleCorrections) + { + dcSoaCorrections = sampleCorrections; + } + + public void setNonMandatoryAccessibility(double nonMandatoryAccessibility) + { + this.nonMandatoryAccessibility = nonMandatoryAccessibility; + } + + public void setToursLeftCount(int count) + { + toursLeftCount = count; + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug DC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public Household getHouseholdObject() + { + return hh; + } + + public Person getPersonObject() + { + return person; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + protected int getToursLeftCount() + { + return toursLeftCount; + } + + protected int getMaxContinuousAvailableWindow() + { + + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return hh + .getMaxJointTimeWindow(tour); + else return person.getMaximumContinuousAvailableWindow(); + } + + protected double getDcSoaCorrectionsAlt(int alt) + { + return dcSoaCorrections[alt - 1]; + } + + protected double getMcLogsumDestAlt(int alt) + { + return modeChoiceLogsums[alt - 1]; + } + + protected double getPopulationDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraPopulation(mgra); + } + + protected double getHouseholdsDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraHouseholds(mgra); + } + + protected double getGradeSchoolEnrollmentDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraGradeSchoolEnrollment(mgra); + } + + protected double getHighSchoolEnrollmentDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraHighSchoolEnrollment(mgra); + } + + protected double getUniversityEnrollmentDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraUniversityEnrollment(mgra); + } + + protected double getOtherCollegeEnrollmentDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraOtherCollegeEnrollment(mgra); + } + + protected double getAdultSchoolEnrollmentDestAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return aggAcc.getMgraAdultSchoolEnrollment(mgra); + } + + protected int getIncome() + { + return hh.getIncomeCategory(); + } + + protected int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + protected int getAutos() + { + return hh.getAutosOwned(); + } + + protected int getWorkers() + { + return hh.getWorkers(); + } + + protected int getNumberOfNonWorkingAdults() + { + return hh.getNumberOfNonWorkingAdults(); + } + + protected int getNumPreschool() + { + return hh.getNumPreschool(); + } + + public int getNumGradeSchoolStudents() + { + return hh.getNumGradeSchoolStudents(); + } + + public int getNumHighSchoolStudents() + { + return hh.getNumHighSchoolStudents(); + } + + protected int getNumChildrenUnder16() + { + return hh.getNumChildrenUnder16(); + } + + protected int getNumChildrenUnder19() + { + return hh.getNumChildrenUnder19(); + } + + protected int getAge() + { + return person.getAge(); + } + + protected int getFemaleWorker() + { + if (person.getPersonIsFemale() == 1) return 1; + else return 0; + } + + protected int getFemale() + { + if (person.getPersonIsFemale() == 1) return 1; + else return 0; + } + + protected int getFullTimeWorker() + { + if (person.getPersonIsFullTimeWorker() == 1) return 1; + else return 0; + } + + protected int getTypicalUniversityStudent() + { + return person.getPersonIsTypicalUniversityStudent(); + } + + protected int getPersonType() + { + return person.getPersonTypeNumber(); + } + + protected int getPersonHasBachelors() + { + return person.getHasBachelors(); + } + + protected int getPersonIsWorker() + { + return person.getPersonIsWorker(); + } + + protected int getWorkTaz() + { + return person.getWorkLocation(); + } + + protected int getWorkTourModeIsSOV() + { + boolean tourModeIsSov = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); + if (tourModeIsSov) return 1; + else return 0; + } + + protected int getTourIsJoint() + { + return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 + : 0; + } + + protected double getTotEmpAccessibilityAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return homeMgraTotalEmploymentAccessibilityArray[mgra]; + } + + protected double getNonMandatoryAccessibilityAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return accTable.getAggregateAccessibility("nonmotor", mgra); + } + + protected double getOpSovDistanceAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return mgraDistanceArray[mgra]; + } + + protected double getLnDcSizeAlt(int alt) + { + int mgra = sampleMgras[alt - 1]; + return Math.log(mgraSizeArray[mgra] + 1); + } + + protected void setWorkAccessibility(double accessibility) + { + workAccessibility = accessibility; + } + + protected double getWorkAccessibility() + { + return workAccessibility; + } + + protected double getNonMandatoryAccessibility() + { + return nonMandatoryAccessibility; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java new file mode 100644 index 0000000..c04c83e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java @@ -0,0 +1,159 @@ +package org.sandag.abm.ctramp; + +import java.util.Arrays; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class DestChoiceTwoStageSoaProbabilitiesCalculator +{ + + private transient Logger soaTwoStageProbsLogger = Logger.getLogger("soaTwoStageProbsLogger"); + + private TazDataManager tdm; + private int maxTaz; + + private ChoiceModelApplication cm; + + public DestChoiceTwoStageSoaProbabilitiesCalculator(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory, String soaDistUECPropertyKey, + String soaDistUECModelSheetPropertyKey, String soaDistUECDataSheetPropertyKey) + { + + tdm = TazDataManager.getInstance(propertyMap); + maxTaz = tdm.getMaxTaz(); + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String soaDistUecFileName = propertyMap.get(soaDistUECPropertyKey); + soaDistUecFileName = uecFileDirectory + soaDistUecFileName; + int soaModelPage = Integer.parseInt(propertyMap.get(soaDistUECModelSheetPropertyKey)); + int soaDataPage = Integer.parseInt(propertyMap.get(soaDistUECDataSheetPropertyKey)); + + DestChoiceTwoStageSoaTazDistanceUtilityDMU tazDistUtilityDmu = dmuFactory + .getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + // create a ChoiceModelApplication object for the filename, model page + // and data page. + cm = new ChoiceModelApplication(soaDistUecFileName, soaModelPage, soaDataPage, propertyMap, + (VariableTable) tazDistUtilityDmu); + + } + + /** + * @param dmuObject + * is the distance utility DMU object + */ + public double[][] computeDistanceUtilities(DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) + { + + double[][] tazDistUtils = new double[maxTaz][maxTaz]; + IndexValues iv = new IndexValues(); + dmuObject.setIndexValuesObject(iv); + + // Loop through combinations of orig/dest TAZs and compute OD utilities + for (int i = 0; i < tazDistUtils.length; i++) + { + iv.setOriginZone(i + 1); + iv.setZoneIndex(i + 1); + cm.computeUtilities(dmuObject, iv); + tazDistUtils[i] = Arrays.copyOf(cm.getUtilities(), tazDistUtils.length); + } + + return tazDistUtils; + } + + /** + * @param dmuObject + * is the distance utility DMU object + * @param distUtilityIndex + * is the distance utility segment index This method signature is + * the default, assuming that no distance probabilities logging + * is required + */ + public double[][] computeDistanceProbabilities( + DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) + { + + double[][] tazDistProbs = new double[maxTaz][maxTaz]; + IndexValues iv = new IndexValues(); + dmuObject.setIndexValuesObject(iv); + + // Loop through combinations of orig/dest TAZs and compute OD utilities + for (int i = 0; i < tazDistProbs.length; i++) + { + iv.setOriginZone(i + 1); + iv.setZoneIndex(i + 1); + cm.computeUtilities(dmuObject, iv); + double[] tempArray = Arrays + .copyOf(cm.getCumulativeProbabilities(), tazDistProbs.length); + tazDistProbs[i] = tempArray; + } + + return tazDistProbs; + } + + /** + * @param dmuObject + * is the distance utility DMU object + * @param distUtilityIndex + * is the distance utility segment index This alternative method + * signature allows distance probabilities logging to be written + */ + public double[][] computeDistanceProbabilities(int traceOrig, + DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) + { + + double[][] tazDistProbs = new double[maxTaz][maxTaz]; + IndexValues iv = new IndexValues(); + dmuObject.setIndexValuesObject(iv); + + // Loop through combinations of orig/dest TAZs and compute OD utilities + for (int i = 0; i < tazDistProbs.length; i++) + { + + iv.setOriginZone(i + 1); + iv.setZoneIndex(i + 1); + cm.computeUtilities(dmuObject, iv); + + if (i == traceOrig - 1) + { + int[] altsToLog = {0, 500, 1000, 2000, 2500, 3736, 3737, 3738, 3739, 3500, 4000}; + cm.logUECResultsSpecificAlts(soaTwoStageProbsLogger, + "Two stage SOA Dist Utilities from TAZ = " + (i + 1), altsToLog); + + double[] probs = cm.getProbabilities(); + double[] utils = cm.getUtilities(); + double total = 0; + for (int k = 0; k < probs.length; k++) + total += Math.exp(utils[k]); + + soaTwoStageProbsLogger.info(""); + for (int k = 1; k < altsToLog.length; k++) + soaTwoStageProbsLogger.info("alt=" + (altsToLog[k] - 1) + ", util=" + + utils[altsToLog[k] - 1] + ", prob=" + probs[altsToLog[k] - 1]); + + soaTwoStageProbsLogger.info("total exponentiated utility = " + total); + soaTwoStageProbsLogger.info(""); + soaTwoStageProbsLogger.info(""); + + } + + tazDistProbs[i] = Arrays.copyOf(cm.getCumulativeProbabilities(), tazDistProbs.length); + + } + + for (int i = 0; i < tazDistProbs.length; i++) + { + soaTwoStageProbsLogger.info("orig=" + (i + 1) + ", dest=3738, cumProb[3737]=" + + tazDistProbs[i][3736] + ", cumProb[3738]=" + tazDistProbs[i][3737] + + ", prob[3738]=" + (tazDistProbs[i][3737] - tazDistProbs[i][3736])); + } + + return tazDistProbs; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java new file mode 100644 index 0000000..24299ac --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java @@ -0,0 +1,156 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class DestChoiceTwoStageSoaTazDistanceUtilityDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(DestChoiceTwoStageSoaTazDistanceUtilityDMU.class); + + protected HashMap methodIndexMap; + + protected IndexValues dmuIndex = null; + + protected double[] dcSize; + protected double[] univEnrollment; + protected double[] gsEnrollment; + protected double[] hsEnrollment; + protected double[] numHhs; + protected int[] gsDistricts; + protected int[] hsDistricts; + + public DestChoiceTwoStageSoaTazDistanceUtilityDMU() + { + } + + public void setIndexValuesObject(IndexValues index) + { + dmuIndex = index; + } + + public void setDestChoiceTazSize(double[] size) + { + dcSize = size; + } + + public void setTazUnivEnrollment(double[] enrollment) + { + univEnrollment = enrollment; + } + + public void setTazGsEnrollment(double[] enrollment) + { + gsEnrollment = enrollment; + } + + public void setTazHsEnrollment(double[] enrollment) + { + hsEnrollment = enrollment; + } + + public void setNumHhs(double[] hhs) + { + numHhs = hhs; + } + + public void setTazGsDistricts(int[] districts) + { + gsDistricts = districts; + } + + public void setTazHsDistricts(int[] districts) + { + hsDistricts = districts; + } + + public double getLnDestChoiceSizeTazAlt(int taz) + { + return dcSize[taz] == 0 ? -999 : Math.log(dcSize[taz]); + } + + public double getSizeTazAlt(int taz) + { + return dcSize[taz]; + } + + public double getUniversityEnrollmentTazAlt(int taz) + { + return univEnrollment[taz]; + } + + public double getGradeSchoolEnrollmentTazAlt(int taz) + { + return gsEnrollment[taz]; + } + + public double getHighSchoolEnrollmentTazAlt(int taz) + { + return hsEnrollment[taz]; + } + + public double getHouseholdsTazAlt(int taz) + { + return numHhs[taz]; + } + + public int getHomeTazGradeSchoolDistrict() + { + return gsDistricts[dmuIndex.getZoneIndex()]; + } + + public int getGradeSchoolDistrictTazAlt(int taz) + { + return gsDistricts[taz]; + } + + public int getHomeTazHighSchoolDistrict() + { + return hsDistricts[dmuIndex.getZoneIndex()]; + } + + public int getHighSchoolDistrictTazAlt(int taz) + { + return hsDistricts[taz]; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java new file mode 100644 index 0000000..9bb1941 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java @@ -0,0 +1,612 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class DestinationSampleOfAlternativesModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(DestinationSampleOfAlternativesModel.class); + private transient Logger dcSoaLogger = Logger.getLogger("tourDcSoa"); + + // set to false to store probabilities in cache for re-use; true to disable + // probabilities cache. + private static final boolean ALWAYS_COMPUTE_PROBABILITIES = false; + private static final boolean ALLOW_DEBUG = true; + + private static final int DC_SOA_DATA_SHEET = 0; + private String dcSoaUecFileName; + private int sampleSize; + + private MgraDataManager mgraManager; + + private int currentOrigMgra; + private double[][] probabilitiesCache; + private double[][] cumProbabilitiesCache; + private int currentWorkMgra; + private double[][] subtourProbabilitiesCache; + private double[][] subtourCumProbabilitiesCache; + + // destsSample[] and destsAvailable[] are indexed by purpose and alternative + private boolean[] escortAvailable; + private int[] escortSample; + private boolean[][] destsAvailable; + private int[][] destsSample; + + private int[] sample; + private float[] corrections; + + private int[] dcSoaModelIndices; + private ChoiceModelApplication[] choiceModel; + + private int numberOfSoaChoiceAlternatives; + private int[] numberOfSoaChoiceAlternativesAvailable; + + private int soaProbabilitiesCalculationCount = 0; + private long soaRunTime = 0; + + public DestinationSampleOfAlternativesModel(String soaUecFile, int sampleSize, + HashMap propertyMap, MgraDataManager mgraManager, + double[][] dcSizeArray, DcSoaDMU dcSoaDmuObject, int[] soaUecIndices) + { + + this.sampleSize = sampleSize; + this.dcSoaUecFileName = soaUecFile; + this.mgraManager = mgraManager; + + // create an array of sample of alternative ChoiceModelApplication + // objects + // for each purpose + setupSampleOfAlternativesChoiceModelArrays(propertyMap, dcSizeArray, dcSoaDmuObject, + soaUecIndices); + + } + + private void setupSampleOfAlternativesChoiceModelArrays(HashMap propertyMap, + double[][] dcSizeArray, DcSoaDMU dcSoaDmuObject, int[] soaUecIndices) + { + + // create a HashMap to map purpose index to model index + dcSoaModelIndices = new int[soaUecIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int soaModelIndex = 0; + int sizeSegmentIndex = 0; + for (int uecIndex : soaUecIndices) + { + // if the uec sheet for the size segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, soaModelIndex); + dcSoaModelIndices[sizeSegmentIndex] = soaModelIndex++; + } else + { + dcSoaModelIndices[sizeSegmentIndex] = modelIndexMap.get(uecIndex); + } + + sizeSegmentIndex++; + } + // the value of soaModelIndex is the number of ChoiceModelApplication + // objects to create + // the modelIndexMap keys are the uec sheets to use in building + // ChoiceModelApplication objects + + choiceModel = new ChoiceModelApplication[modelIndexMap.size()]; + probabilitiesCache = new double[sizeSegmentIndex][]; + cumProbabilitiesCache = new double[sizeSegmentIndex][]; + subtourProbabilitiesCache = new double[sizeSegmentIndex][]; + subtourCumProbabilitiesCache = new double[sizeSegmentIndex][]; + + int i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + int modelIndex = -1; + try + { + modelIndex = modelIndexMap.get(uecIndex); + choiceModel[modelIndex] = new ChoiceModelApplication(dcSoaUecFileName, uecIndex, + DC_SOA_DATA_SHEET, propertyMap, (VariableTable) dcSoaDmuObject); + i++; + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up DC SOA ChoiceModelApplication[%d] for modelIndex=%d of %d models", + i, modelIndex, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + setAvailabilityForSampleOfAlternatives(dcSizeArray); + + } + + /** + * This method is called initially when the SOA choice models array is + * created. It would be called subsequently if a shadow pricing methodology + * is applied to reset the scaled size terms and corresponding + * availabilities and sample arrays. + */ + public void setAvailabilityForSampleOfAlternatives(double[][] dcSizeArray) + { + + int maxMgra = mgraManager.getMaxMgra(); + + // declare dimensions for the alternative availability array by purpose + // and + // number of alternaives + escortAvailable = new boolean[maxMgra + 1]; + escortSample = new int[maxMgra + 1]; + destsAvailable = new boolean[dcSizeArray.length][maxMgra + 1]; + destsSample = new int[dcSizeArray.length][maxMgra + 1]; + + numberOfSoaChoiceAlternativesAvailable = new int[dcSizeArray.length]; + + for (int i = 0; i < dcSizeArray.length; i++) + { + for (int k = 1; k <= maxMgra; k++) + { + if (dcSizeArray[i][k] > 0.0) + { + destsAvailable[i][k] = true; + destsSample[i][k] = 1; + numberOfSoaChoiceAlternativesAvailable[i]++; + } + } // k + } + + Arrays.fill(escortAvailable, true); + Arrays.fill(escortSample, 1); + + numberOfSoaChoiceAlternatives = maxMgra; + } + + public int getNumberOfAlternatives() + { + return numberOfSoaChoiceAlternatives; + } + + public void computeDestinationSampleOfAlternatives(DcSoaDMU dcSoaDmuObject, Tour tour, + Person person, String segmentName, int segmentIndex, int origMgra) + { + + long timeCheck = System.nanoTime(); + + // these will be dimensioned with the number of unique alternatives + // determined for the decision makers + int[] altList; + int[] altListFreq; + HashMap altFreqMap = new HashMap(); + + int modelIndex = dcSoaModelIndices[segmentIndex]; + + // if the flag is set to compute sample of alternative probabilities for + // every work/school location choice, + // or the tour's origin taz is different from the currentOrigTaz, reset + // the currentOrigTaz and clear the stored probabilities. + if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) + { + + if (ALWAYS_COMPUTE_PROBABILITIES || origMgra != currentWorkMgra) + { + + // clear the probabilities stored for the current origin mgra, + // for each DC segment + for (int i = 0; i < subtourProbabilitiesCache.length; i++) + { + subtourProbabilitiesCache[i] = null; + subtourCumProbabilitiesCache[i] = null; + } + currentWorkMgra = origMgra; + + } + + // If the sample of alternatives choice probabilities have not been + // computed for the current origin mgra + // and segment specified, compute them. + if (subtourProbabilitiesCache[segmentIndex] == null) + { + computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, + segmentName, segmentIndex, origMgra); + soaProbabilitiesCalculationCount++; + } + + } else if (tour != null + && tour.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) + { + + // always update probabilities for Escort tours + // clear the probabilities stored for the current origin mgra, for + // the Escort DC segment + probabilitiesCache[segmentIndex] = null; + cumProbabilitiesCache[segmentIndex] = null; + + destsAvailable[segmentIndex] = escortAvailable; + destsSample[segmentIndex] = escortSample; + + currentOrigMgra = origMgra; + + computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, + segmentName, segmentIndex, origMgra); + soaProbabilitiesCalculationCount++; + + } else + { + + if (ALWAYS_COMPUTE_PROBABILITIES || origMgra != currentOrigMgra) + { + + // clear the probabilities stored for the current origin mgra, + // for each DC segment + for (int i = 0; i < probabilitiesCache.length; i++) + { + probabilitiesCache[i] = null; + cumProbabilitiesCache[i] = null; + } + currentOrigMgra = origMgra; + + } + + // If the sample of alternatives choice probabilities have not been + // computed for the current origin taz + // and purpose specified, compute them. + if (probabilitiesCache[segmentIndex] == null) + { + computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, + segmentName, segmentIndex, origMgra); + soaProbabilitiesCalculationCount++; + } + + } + + Household hhObj = person.getHouseholdObject(); + Random hhRandom = hhObj.getHhRandom(); + int rnCount = hhObj.getHhRandomCount(); + // when household.getHhRandom() was applied, the random count was + // incremented, assuming a random number would be drawn right away. + // so let's decrement by 1, then increment the count each time a random + // number is actually drawn in this method. + rnCount--; + + // select sampleSize alternatives based on probabilitiesList[origTaz], + // and + // count frequency of alternatives chosen. + // final sample may include duplicate alternative selections. + for (int i = 0; i < sampleSize; i++) + { + + double rn = hhRandom.nextDouble(); + rnCount++; + + int chosenAlt = -1; + if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) chosenAlt = Util + .binarySearchDouble(subtourCumProbabilitiesCache[segmentIndex], rn) + 1; + else chosenAlt = Util.binarySearchDouble(cumProbabilitiesCache[segmentIndex], rn) + 1; + + // write choice model alternative info to log file + if (hhObj.getDebugChoiceModels()) + { + choiceModel[modelIndex] + .logSelectionInfo( + String.format( + "Sample Of Alternatives Choice for segmentName=%s, segmentIndex=%d, modelIndex=%d, origMgra=%d", + segmentName, segmentIndex, modelIndex, origMgra), String + .format("HHID=%d, rn=%.8f, rnCount=%d", hhObj.getHhId(), + rn, (rnCount + i)), rn, chosenAlt); + } + + int freq = 0; + if (altFreqMap.containsKey(chosenAlt)) freq = altFreqMap.get(chosenAlt); + altFreqMap.put(chosenAlt, (freq + 1)); + + } + + // sampleSize random number draws were made from the Random object for + // the + // current household, + // so update the count in the hh's Random. + hhObj.setHhRandomCount(rnCount); + + // create arrays of the unique chosen alternatives and the frequency + // with + // which those alternatives were chosen. + int numUniqueAlts = altFreqMap.keySet().size(); + altList = new int[numUniqueAlts]; + altListFreq = new int[numUniqueAlts]; + Iterator it = altFreqMap.keySet().iterator(); + int k = 0; + while (it.hasNext()) + { + int key = (Integer) it.next(); + int value = (Integer) altFreqMap.get(key); + altList[k] = key; + altListFreq[k] = value; + k++; + } + + // loop through these arrays, construct final sample[] and + // corrections[]. + sample = new int[numUniqueAlts + 1]; + corrections = new float[numUniqueAlts + 1]; + for (k = 0; k < numUniqueAlts; k++) + { + int alt = altList[k]; + int freq = altListFreq[k]; + + double prob = 0; + if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) prob = subtourProbabilitiesCache[segmentIndex][alt - 1]; + else prob = probabilitiesCache[segmentIndex][alt - 1]; + + sample[k + 1] = alt; + corrections[k + 1] = (float) Math.log((double) freq / prob); + } + + soaRunTime += (System.nanoTime() - timeCheck); + + } + + /** + * This method is used if the tour/person decision maker is the first one + * encountered for the purpose and origin taz. Once the sample of + * alternatives choice probabilities for a purpose and origin taz are + * computed, they are stored in an array and used by other decision makers + * with the same purpose and origin taz. + * + * @param probabilitiesList + * is the probabilities array for the given purpose in which + * choice probabilities will be saved for the origin of the + * tour/place to be selected. + * @param cumProbabilitiesList + * is the probabilities array for the given purpose in which + * choice cumulative probabilities will be saved for the origin + * of the tour/place to be selected. + * @param choiceModel + * the ChoiceModelApplication object for the purpose + * @param tour + * the tour object for whic destination choice is required, or + * null if a usual work/school location is being chosen + * @param person + * the person object for whom the choice is being made + * @param segmentName + * the name of the segment the choice is being made for - for + * logging + * @param segmentindex + * the index associated with the segment + * @param origMgra + * the index associated with the segment + * @param distances + * array frpm origin MGRA to all MGRAs + */ + private void computeSampleOfAlternativesChoiceProbabilities(DcSoaDMU dcSoaDmuObject, Tour tour, + Person person, String segmentName, int segmentIndex, int origMgra) + { + + Household hhObj = person.getHouseholdObject(); + + // set the hh, person, and tour objects for this DMU object + dcSoaDmuObject.setHouseholdObject(hhObj); + dcSoaDmuObject.setPersonObject(person); + dcSoaDmuObject.setTourObject(tour); + + // set sample of alternatives choice DMU attributes + dcSoaDmuObject.setDmuIndexValues(hhObj.getHhId(), hhObj.getHhMgra(), origMgra, 0); + + // prepare a trace log header that the choiceModel object will write + // prior to + // UEC trace logging + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + + int choiceModelIndex = dcSoaModelIndices[segmentIndex]; + + // If the person making the choice is from a household requesting trace + // information, + // create a trace logger header and write prior to the choiceModel + // computing + // utilities + if (hhObj.getDebugChoiceModels()) + { + + if (tour == null) + { + // null tour means the SOA choice is for a mandatory usual + // location choice + choiceModelDescription = String.format( + "Usual Location Sample of Alternatives Choice Model for: Segment=%s", + segmentName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person + .getHouseholdObject().getHhId(), person.getPersonNum(), person + .getPersonType()); + } else + { + choiceModelDescription = String.format( + "Destination Choice Model for: Segment=%s, TourId=%d", segmentName, + tour.getTourId()); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person + .getHouseholdObject().getHhId(), person.getPersonNum(), person + .getPersonType()); + } + + // log headers to traceLogger if the person making the choice is + // from a + // household requesting trace information + choiceModel[choiceModelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + } + + try + { + choiceModel[choiceModelIndex].computeUtilities(dcSoaDmuObject, + dcSoaDmuObject.getDmuIndexValues(), destsAvailable[segmentIndex], + destsSample[segmentIndex]); + } catch (Exception e) + { + logger.error("exception caught in DC SOA model for:"); + choiceModelDescription = String.format( + "Destination Choice Model for: Segment=%s, TourId=%d", segmentName, + tour.getTourId()); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person + .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); + logger.error("choiceModelDescription:" + choiceModelDescription); + logger.error("decisionMakerLabel:" + decisionMakerLabel); + throw new RuntimeException(e); + } + + // TODO: debug + if (choiceModel[choiceModelIndex].getAvailabilityCount() == 0) + { + + int j = 0; + int[] debugAlts = new int[5]; + for (int i = 0; i < destsSample[segmentIndex].length; i++) + { + if (destsSample[segmentIndex][i] == 1) + { + debugAlts[j++] = i; + if (j == 5) break; + } + } + + choiceModel[choiceModelIndex].logUECResultsSpecificAlts(dcSoaLogger, "debugging", + debugAlts); + } + + // the following order of assignment is important in mult-threaded + // context. + // probabilitiesCache[][] is a trigger variable - if it is not null for + // any thread, the cumProbabilitiesCache[][] values + // are used immediately, so the cumProbabilitiesCache values must be + // assigned before the probabilitiesCache + // are assigned, which indicates cumProbabilitiesCache[][] values are + // ready to be used. + if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) + { + + subtourCumProbabilitiesCache[segmentIndex] = Arrays.copyOf( + choiceModel[choiceModelIndex].getCumulativeProbabilities(), + choiceModel[choiceModelIndex].getNumberOfAlternatives()); + subtourProbabilitiesCache[segmentIndex] = Arrays.copyOf( + choiceModel[choiceModelIndex].getProbabilities(), + choiceModel[choiceModelIndex].getNumberOfAlternatives()); + } else + { + + cumProbabilitiesCache[segmentIndex] = Arrays.copyOf( + choiceModel[choiceModelIndex].getCumulativeProbabilities(), + choiceModel[choiceModelIndex].getNumberOfAlternatives()); + probabilitiesCache[segmentIndex] = Arrays.copyOf( + choiceModel[choiceModelIndex].getProbabilities(), + choiceModel[choiceModelIndex].getNumberOfAlternatives()); + + if (hhObj.getDebugChoiceModels()) + { + PrintWriter out = null; + try + { + out = new PrintWriter(new BufferedWriter(new FileWriter( + new File("soaProbs.csv")))); + + out.println("choiceModelDescription:" + choiceModelDescription); + out.println("decisionMakerLabel:" + decisionMakerLabel); + + for (int i = 0; i < probabilitiesCache[segmentIndex].length; i++) + { + out.println((i + 1) + "," + probabilitiesCache[segmentIndex][i]); + } + } catch (IOException e) + { + // TODO Auto-generated catch block + e.printStackTrace(); + } + + out.close(); + + } + + } + + // If the person making the choice is from a household requesting trace + // information, + // write choice model alternative info to the debug log file + if (hhObj.getDebugChoiceModels() && ALLOW_DEBUG) + { + // if ( dcSoaLogger.isDebugEnabled() ){ + int[] altsToLog = {0, 77, 78, 79, 80}; + choiceModel[choiceModelIndex].logAlternativesInfo(String.format( + "%s Sample Of Alternatives Choice for origTaz=%d", segmentName, origMgra), + String.format("HHID=%d", hhObj.getHhId()), dcSoaLogger); + choiceModel[choiceModelIndex].logUECResultsSpecificAlts(dcSoaLogger, + choiceModelDescription + ", " + decisionMakerLabel, altsToLog); + + double[] probs = choiceModel[choiceModelIndex].getProbabilities(); + double[] utils = choiceModel[choiceModelIndex].getUtilities(); + double total = 0; + for (int i = 0; i < probs.length; i++) + total += Math.exp(utils[i]); + + dcSoaLogger.info(""); + for (int i = 1; i < altsToLog.length; i++) + dcSoaLogger.info("alt=" + (altsToLog[i] + 1) + ", util=" + utils[altsToLog[i]] + + ", prob=" + probs[altsToLog[i]]); + + dcSoaLogger.info("total exponentiated utility = " + total); + dcSoaLogger.info(""); + dcSoaLogger.info(""); + // } + } + + } + + public int getSoaProbabilitiesCalculationCount() + { + return soaProbabilitiesCalculationCount; + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + + public int getCurrentOrigMgra() + { + return currentOrigMgra; + } + + public int[] getSampleOfAlternatives() + { + return sample; + } + + public float[] getSampleOfAlternativesCorrections() + { + return corrections; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java new file mode 100644 index 0000000..ba5b011 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java @@ -0,0 +1,1847 @@ +package org.sandag.abm.ctramp; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Random; + +import org.apache.log4j.Logger; + +public class Household + implements java.io.Serializable +{ + + private boolean debugChoiceModels; + + private int hhId; + private short hhIncomeCategory; + private int hhIncomeInDollars; + private short hhSize; + private short hhType; + private short unitType; + private short hhBldgsz; + private short hhWorkers; + + private int homeTaz; + private int homeMgra; + private int homeWalkSubzone; + + //added for BCA tool + private float autoOwnershipLogsum; + private float transponderLogsum; + private float cdapLogsum; + private float jtfLogsum; + + private Person[] persons; + + private Tour[] jointTours; + + private short aoModelAutos; + private short automatedVehicles; + private short conventionalVehicles; + private String cdapModelPattern; + private short imtfModelPattern; + private String jtfModelPattern; + private short tpChoice; + private short outboundEscortChoice; + private short inboundEscortChoice; + + + private Random hhRandom; + private int randomCount = 0; + private HashMap uwslRandomCountList; + private int preAoRandomCount; + private int aoRandomCount; + private int tpRandomCount; + private int fpRandomCount; + private int ieRandomCount; + private int cdapRandomCount; + private int imtfRandomCount; + private int imtodRandomCount; + private int awfRandomCount; + private int awlRandomCount; + private int awtodRandomCount; + private int jtfRandomCount; + private int jtlRandomCount; + private int jtodRandomCount; + private int inmtfRandomCount; + private int inmtlRandomCount; + private int inmtodRandomCount; + private int stfRandomCount; + private int stlRandomCount; + + private int maxAdultOverlaps; + private int maxChildOverlaps; + private int maxMixedOverlaps; + + private ModelStructure modelStructure; + + private long seed; + public Household(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + hhRandom = new Random(); + uwslRandomCountList = new HashMap(); + } + + /** + * Returns a 1-based array of persons in the household + * @return Person array + */ + public Person[] getPersons() + { + return persons; + } + + public void initializeWindows() + { + + // loop through the person array (1-based) + for (int i = 1; i < persons.length; ++i) + { + persons[i].initializeWindows(); + } + + } + + public void setDebugChoiceModels(boolean value) + { + debugChoiceModels = value; + } + + public void setHhId(int id, int baseSeed) + { + hhId = id; + randomCount = 0; + hhRandom.setSeed(baseSeed + hhId); + } + + public void setRandomObject(Random r) + { + hhRandom = r; + } + + public void setHhRandomCount(int count) + { + randomCount = count; + } + + // work/school location choice uses shadow pricing, so save randomCount per + // iteration + public void setUwslRandomCount(int iter, int count) + { + uwslRandomCountList.put(iter, count); + } + + public void setPreAoRandomCount(int count) + { + preAoRandomCount = count; + } + + public void setAoRandomCount(int count) + { + aoRandomCount = count; + } + + public void setTpRandomCount(int count) + { + tpRandomCount = count; + } + + public void setFpRandomCount(int count) + { + fpRandomCount = count; + } + + public void setIeRandomCount(int count) + { + ieRandomCount = count; + } + + public void setCdapRandomCount(int count) + { + cdapRandomCount = count; + } + + public void setImtfRandomCount(int count) + { + imtfRandomCount = count; + } + + public void setImtodRandomCount(int count) + { + imtodRandomCount = count; + } + + public void setAwfRandomCount(int count) + { + awfRandomCount = count; + } + + public void setAwlRandomCount(int count) + { + awlRandomCount = count; + } + + public void setAwtodRandomCount(int count) + { + awtodRandomCount = count; + } + + public void setJtfRandomCount(int count) + { + jtfRandomCount = count; + } + + public void setJtlRandomCount(int count) + { + jtlRandomCount = count; + } + + public void setJtodRandomCount(int count) + { + jtodRandomCount = count; + } + + public void setInmtfRandomCount(int count) + { + inmtfRandomCount = count; + } + + public void setInmtlRandomCount(int count) + { + inmtlRandomCount = count; + } + + public void setInmtodRandomCount(int count) + { + inmtodRandomCount = count; + } + + public void setStfRandomCount(int count) + { + stfRandomCount = count; + } + + public void setStlRandomCount(int count) + { + stlRandomCount = count; + } + + public void setHhTaz(int taz) + { + homeTaz = taz; + } + + public void setHhMgra(int mgra) + { + homeMgra = mgra; + } + + public void setHhWalkSubzone(int subzone) + { + homeWalkSubzone = (short) subzone; + } + + public void setHhAutos(int autos) + { + // this sets the variable that will be used in work/school location + // choice. + // after auto ownership runs, this variable gets updated with number of + // autos + // for result. + aoModelAutos = (short) autos; + } + + public void setTpChoice(int value) + { + tpChoice = (short) value; + } + + /** + * auto sufficiency: 1 if cars < workers, 2 if cars equal workers, 3 if cars + * > workers + * + * @return auto sufficiency value + */ + public int getAutoSufficiency() + { + if (aoModelAutos < hhWorkers) return 1; + else if (aoModelAutos == hhWorkers) return 2; + else return 3; + } + + public int getAutosOwned() + { + return (int) aoModelAutos; + } + + public int getAutomatedVehicles() { + return (int) automatedVehicles; + } + + public void setAutomatedVehicles(int automatedVehicles) { + this.automatedVehicles = (short) automatedVehicles; + } + + public int getConventionalVehicles() { + return (int) conventionalVehicles; + } + + public void setConventionalVehicles(int conventionalVehicles) { + this.conventionalVehicles = (short) conventionalVehicles; + } + + public int getTpChoice() + { + return (int) tpChoice; + } + + public void setCoordinatedDailyActivityPatternResult(String pattern) + { + cdapModelPattern = pattern; + } + + public String getCoordinatedDailyActivityPattern() + { + return cdapModelPattern; + } + + public void setJointTourFreqResult(int altIndex, String altName) + { + jtfModelPattern = String.format("%d_%s", altIndex, altName); + } + + public int getJointTourFreqChosenAlt() + { + int returnValue = 0; + if (jtfModelPattern == null) + { + returnValue = 0; + } else + { + int endIndex = jtfModelPattern.indexOf('_'); + returnValue = Integer.parseInt(jtfModelPattern.substring(0, endIndex)); + } + return returnValue; + } + + public String getJointTourFreqChosenAltName() + { + String returnValue = "none"; + if (jtfModelPattern != null) + { + int startIndex = jtfModelPattern.indexOf('_') + 1; + returnValue = jtfModelPattern.substring(startIndex); + } + return returnValue; + } + + public void setHhBldgsz(int code) + { + hhBldgsz = (short) code; + } + + public int getHhBldgsz() + { + return (int) hhBldgsz; + } + + public void setHhSize(int numPersons) + { + hhSize = (short) numPersons; + persons = new Person[numPersons + 1]; + for (int i = 1; i <= numPersons; i++) + persons[i] = new Person(this, i, modelStructure); + + } + + public void setHhIncomeCategory(int category) + { + hhIncomeCategory = (short) category; + } + + public void setHhIncomeInDollars(int dollars) + { + hhIncomeInDollars = dollars; + } + + public void setHhWorkers(int numWorkers) + { + hhWorkers = (short) numWorkers; + } + + public void setHhType(int type) + { + hhType = (short) type; + } + + // 0=Housing unit, 1=Institutional group quarters, 2=Noninstitutional group + // quarters + public void setUnitType(int type) + { + unitType = (short) type; + } + + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + public int getHhSize() + { + return (int) hhSize; + } + + public int getNumTotalIndivTours() + { + int count = 0; + for (int i = 1; i < persons.length; i++) + count += persons[i].getNumTotalIndivTours(); + return count; + } + + public int getNumberOfNonWorkingAdults() + { + int count = 0; + for (int i = 1; i < persons.length; i++) + count += persons[i].getPersonIsNonWorkingAdultUnder65() + + persons[i].getPersonIsNonWorkingAdultOver65(); + return count; + } + + public int getIsNonFamilyHousehold() + { + + if (hhType == HouseholdType.NON_FAMILY_MALE_ALONE.ordinal()) return (1); + if (hhType == HouseholdType.NON_FAMILY_MALE_NOT_ALONE.ordinal()) return (1); + if (hhType == HouseholdType.NON_FAMILY_FEMALE_ALONE.ordinal()) return (1); + if (hhType == HouseholdType.NON_FAMILY_FEMALE_NOT_ALONE.ordinal()) return (1); + + return (0); + } + + /** + * unitType: 0=Housing unit, 1=Institutional group quarters, + * 2=Noninstitutional group quarters + * + * @return 1 if household is group quarters, 0 for non-group quarters + */ + public int getIsGroupQuarters() + { + if (unitType == 0) return 0; + else return 1; + } + + public int getNumStudents() + { + int count = 0; + for (int i = 1; i < persons.length; ++i) + { + count += persons[i].getPersonIsStudent(); + } + return (count); + } + + public int getNumGradeSchoolStudents() + { + int count = 0; + for (int i = 1; i < persons.length; ++i) + { + count += persons[i].getPersonIsGradeSchool(); + } + return (count); + } + + public int getNumHighSchoolStudents() + { + int count = 0; + for (int i = 1; i < persons.length; ++i) + { + count += persons[i].getPersonIsHighSchool(); + } + return (count); + } + + public int getNumberOfChildren6To18WithoutMandatoryActivity() + { + + int count = 0; + + for (int i = 1; i < persons.length; ++i) + { + count += persons[i].getPersonIsChild6To18WithoutMandatoryActivity(); + } + + return (count); + } + + public int getNumberOfPreDrivingWithNonHomeActivity() + { + + int count = 0; + for (int i = 1; i < persons.length; ++i) + { + // count only predrving kids + if (persons[i].getPersonIsStudentDriving() == 1) + { + // count only if CDAP is M or N (i.e. not H) + if (!persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) + count++; + } + } + + return count; + } + + public int getNumberOfPreschoolWithNonHomeActivity() + { + + int count = 0; + for (int i = 1; i < persons.length; ++i) + { + // count only predrving kids + if (persons[i].getPersonIsPreschoolChild() == 1) + { + // count only if CDAP is M or N (i.e. not H) + if (!persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) + count++; + } + } + + return count; + } + + /** + * return the number of school age students this household has for the + * purpose index. + * + * @param purposeIndex + * is the DC purpose index to be compared to the usual school + * location index saved for this person upon reading synthetic + * population file. + * @return num, a value of the number of school age students in the + * household for this purpose index. + */ + public int getNumberOfDrivingAgedStudentsWithDcPurposeIndex(int segmentIndex) + { + int num = 0; + for (int j = 1; j < persons.length; j++) + { + if (persons[j].getPersonIsStudentDriving() == 1 + && persons[j].getSchoolLocationSegmentIndex() == segmentIndex) num++; + } + return num; + } + + public int getNumberOfNonDrivingAgedStudentsWithDcPurposeIndex(int segmentIndex) + { + int num = 0; + for (int j = 1; j < persons.length; j++) + { + if (persons[j].getPersonIsStudentNonDriving() == 1 + || persons[j].getPersonIsPreschoolChild() == 1 + && persons[j].getSchoolLocationSegmentIndex() == segmentIndex) num++; + } + return num; + } + + public Person getPerson(int persNum) + { + if (persNum < 1 || persNum > hhSize) + { + throw new RuntimeException(String.format( + "persNum value = %d is out of range for hhSize = %d", persNum, hhSize)); + } + + return persons[persNum]; + } + + // methods DMU will use to get info from household object + + public int getHhId() + { + return hhId; + } + + public Random getHhRandom() + { + randomCount++; + return hhRandom; + } + + public int getHhRandomCount() + { + return randomCount; + } + + public int getUwslRandomCount(int iter) + { + return uwslRandomCountList.get(iter); + } + + public int getPreAoRandomCount() + { + return preAoRandomCount; + } + + public int getAoRandomCount() + { + return aoRandomCount; + } + + public int getTpRandomCount() + { + return tpRandomCount; + } + + public int getFpRandomCount() + { + return fpRandomCount; + } + + public int getIeRandomCount() + { + return ieRandomCount; + } + + public int getCdapRandomCount() + { + return cdapRandomCount; + } + + public int getImtfRandomCount() + { + return imtfRandomCount; + } + + public int getImtodRandomCount() + { + return imtodRandomCount; + } + + public int getJtfRandomCount() + { + return jtfRandomCount; + } + + public int getAwfRandomCount() + { + return awfRandomCount; + } + + public int getAwlRandomCount() + { + return awlRandomCount; + } + + public int getAwtodRandomCount() + { + return awtodRandomCount; + } + + public int getJtlRandomCount() + { + return jtlRandomCount; + } + + public int getJtodRandomCount() + { + return jtodRandomCount; + } + + public int getInmtfRandomCount() + { + return inmtfRandomCount; + } + + public int getInmtlRandomCount() + { + return inmtlRandomCount; + } + + public int getInmtodRandomCount() + { + return inmtodRandomCount; + } + + public int getStfRandomCount() + { + return stfRandomCount; + } + + public int getStlRandomCount() + { + return stlRandomCount; + } + + public int getHhTaz() + { + return homeTaz; + } + + public int getHhMgra() + { + return homeMgra; + } + + public int getHhWalkSubzone() + { + return homeWalkSubzone; + } + + public int getIncomeCategory() + { + return (int) hhIncomeCategory; + } + + public int getIncomeInDollars() + { + return hhIncomeInDollars; + } + + public int getWorkers() + { + return (int) hhWorkers; + } + + public int getDrivers() + { + return getNumPersons16plus(); + } + + public int getSize() + { + return (int) hhSize; + } + + public int getChildunder16() + { + if (getNumChildrenUnder16() > 0) return 1; + else return 0; + } + + public int getChild16plus() + { + if (getNumPersons16plus() > 0) return 1; + else return 0; + } + + public int getNumChildrenUnder16() + { + int numChildrenUnder16 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() < 16) numChildrenUnder16++; + } + return numChildrenUnder16; + } + + public int getNumChildrenUnder19() + { + int numChildrenUnder19 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() < 19) numChildrenUnder19++; + } + return numChildrenUnder19; + } + + public int getNumPersons0to4() + { + int numPersons0to4 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() < 5) numPersons0to4++; + } + return numPersons0to4; + } + + /** + * used in AO choice utility + * + * @return number of persons age 6 to 15, inclusive + */ + public int getNumPersons6to15() + { + int numPersons6to15 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 6 && persons[i].getAge() <= 15) numPersons6to15++; + } + return numPersons6to15; + } + + /** + * used in Stop Frequency choice utility + * + * @return number of persons age 5 to 15, inclusive + */ + public int getNumPersons5to15() + { + int numPersons5to15 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 5 && persons[i].getAge() <= 15) numPersons5to15++; + } + return numPersons5to15; + } + + public int getNumPersons16to17() + { + int numPersons16to17 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 16 && persons[i].getAge() <= 17) numPersons16to17++; + } + return numPersons16to17; + } + + public int getNumPersons18to35(){ + + int numPersons18to35 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 18 && persons[i].getAge() <= 35) numPersons18to35++; + } + return numPersons18to35; + } + + public int getNumPersons16plus() + { + int numPersons16plus = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 16) numPersons16plus++; + } + return numPersons16plus; + } + + public int getNumPersons18plus() + { + int numPersons18plus = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 18) numPersons18plus++; + } + return numPersons18plus; + } + + public int getNumPersons80plus() + { + int numPersons80plus = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 80) numPersons80plus++; + } + return numPersons80plus; + } + + public int getNumPersons18to24() + { + int numPersons18to24 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 18 && persons[i].getAge() <= 24) numPersons18to24++; + } + return numPersons18to24; + } + + public int getNumPersons65to79() + { + int numPersons65to79 = 0; + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getAge() >= 65 && persons[i].getAge() <= 79) numPersons65to79++; + } + return numPersons65to79; + } + + public int getNumFtWorkers() + { + int numFtWorkers = 0; + for (int i = 1; i < persons.length; i++) + numFtWorkers += persons[i].getPersonIsFullTimeWorker(); + return numFtWorkers; + } + + public int getNumPtWorkers() + { + int numPtWorkers = 0; + for (int i = 1; i < persons.length; i++) + numPtWorkers += persons[i].getPersonIsPartTimeWorker(); + return numPtWorkers; + } + + public int getNumUnivStudents() + { + int numUnivStudents = 0; + for (int i = 1; i < persons.length; i++) + numUnivStudents += persons[i].getPersonIsUniversityStudent(); + return numUnivStudents; + } + + public int getNumNonWorkAdults() + { + int numNonWorkAdults = 0; + for (int i = 1; i < persons.length; i++) + numNonWorkAdults += persons[i].getPersonIsNonWorkingAdultUnder65(); + return numNonWorkAdults; + } + + public int getNumAdults() + { + int numAdults = 0; + for (int i = 1; i < persons.length; i++) + numAdults += (persons[i].getPersonIsFullTimeWorker() + + persons[i].getPersonIsPartTimeWorker() + + persons[i].getPersonIsUniversityStudent() + + persons[i].getPersonIsNonWorkingAdultUnder65() + persons[i] + .getPersonIsNonWorkingAdultOver65()); + return numAdults; + } + + public int getNumRetired() + { + int numRetired = 0; + for (int i = 1; i < persons.length; i++) + numRetired += persons[i].getPersonIsNonWorkingAdultOver65(); + return numRetired; + } + + public int getNumDrivingStudents() + { + int numDrivingStudents = 0; + for (int i = 1; i < persons.length; i++) + numDrivingStudents += persons[i].getPersonIsStudentDriving(); + return numDrivingStudents; + } + + public int getNumNonDrivingStudents() + { + int numNonDrivingStudents = 0; + for (int i = 1; i < persons.length; i++) + numNonDrivingStudents += persons[i].getPersonIsStudentNonDriving(); + return numNonDrivingStudents; + } + + public int getNumPreschool() + { + int numPreschool = 0; + for (int i = 1; i < persons.length; i++) + numPreschool += persons[i].getPersonIsPreschoolChild(); + return numPreschool; + } + + public int getNumHighSchoolGraduates() + { + int numGrads = 0; + for (int i = 1; i < persons.length; i++) + numGrads += persons[i].getPersonIsHighSchoolGraduate(); + return numGrads; + } + + /** + * Iterates through person array, adds up and returns number of children with school tours for preschool children, non-driving age students, and driving age students + * @return Number of children with school tours. + */ + public int getNumChildrenWithSchoolTours(){ + + int numChildrenWithSchoolTours = 0; + + for(int i=1; i < persons.length;++i){ + Person p = persons[i]; + if((p.getPersonIsPreschoolChild() == 1) || (p.getPersonIsStudentNonDriving() == 1)|| (p.getPersonIsStudentDriving() == 1)){ + if(p.getNumSchoolTours()>0) + ++numChildrenWithSchoolTours; + } + } + + return numChildrenWithSchoolTours; + } + /** + * joint tour frequency choice is not applied to a household unless it has: + * 2 or more persons, each with at least one out-of home activity, and at + * least 1 of the persons not a pre-schooler. + * */ + public int getValidHouseholdForJointTourFrequencyModel() + { + + // return one of the following condition codes for this household + // producing + // joint tours: + // 1: household eligible for joint tour production + // 2: household ineligible due to 1 person hh. + // 3: household ineligible due to fewer than 2 persons traveling out of + // home + // 4: household ineligible due to fewer than 1 non-preschool person + // traveling + // out of home + + // no joint tours for single person household + if (hhSize == 1) return 2; + + int leavesHome = 0; + int nonPreSchoolerLeavesHome = 0; + for (int i = 1; i < persons.length; i++) + { + if (!persons[i].getCdapActivity().equalsIgnoreCase("H")) + { + leavesHome++; + if (persons[i].getPersonIsPreschoolChild() == 0) nonPreSchoolerLeavesHome++; + } + } + + // if the number of persons leaving home during the day is not at least + // 2, no + // joint tours + if (leavesHome < 2) return 3; + + // if the number of non-preschool persons leaving home during the day is + // not + // at least 1, no joint tours + if (nonPreSchoolerLeavesHome < 1) return 4; + + // if all conditions are met, we can apply joint tour frequency model to + // this + // household + return 1; + + } + + /** + * return maximum periods of overlap between this person and other adult + * persons in the household. + * + * @return the most number of periods mutually available between this person + * and other adult household members + */ + public int getMaxAdultOverlaps() + { + return maxAdultOverlaps; + } + + /** + * return maximum periods of overlap between this person and other children + * in the household. + * + * @return the most number of periods mutually available between this person + * and other child household members + */ + public int getMaxChildOverlaps() + { + return maxChildOverlaps; + } + + /** + * return maximum periods of overlap between this person(adult/child) and + * other persons(child/adult) in the household. + * + * @return the most number of periods mutually available between this person + * and other type household members + */ + public int getMaxMixedOverlaps() + { + return maxMixedOverlaps; + } + + public int getMaxJointTimeWindow(Tour t) + { + // get array of person array indices participating in joint tour + int[] participatingPersonIndices = t.getPersonNumArray(); + + // create an array to hold time window arrays for each participant + short[][] personWindows = new short[participatingPersonIndices.length][]; + + // get time window arrays for each participant + int k = 0; + for (int i : participatingPersonIndices) + personWindows[k++] = persons[i].getTimeWindows(); + + int count = 0; + + int maxCount = 0; + // loop over time window intervals + for (int w = 1; w < personWindows[0].length; w++) + { + + // loop over party; determine if interval is available for everyone + // in party; + boolean available = true; + for (k = 0; k < personWindows.length; k++) + { + if (personWindows[k][w] > 0) + { + available = false; + break; + } + } + + // if available for whole party, increment count; determine maximum + // continous time window available to whole party. + if (available) + { + count++; + if (count > maxCount) maxCount = count; + } else + { + count = 0; + } + + } + + return maxCount; + } + + /** + * @return number of adults in household with "M" or "N" activity pattern - + * that is, traveling adults. + */ + public int getTravelActiveAdults() + { + + int adultsStayingHome = 0; + int adults = 0; + for (int p = 1; p < persons.length; p++) + { + // person is an adult + if (persons[p].getPersonIsAdult() == 1) + { + adults++; + if (persons[p].getCdapActivity().equalsIgnoreCase("H")) adultsStayingHome++; + } + } + + // return the number of adults traveling = number of adults minus the + // number + // of adults staying home. + return adults - adultsStayingHome; + + } + + /** + * @return number of children in household with "M" or "N" activity pattern + * - that is, traveling children. + */ + public int getTravelActiveChildren() + { + + int childrenStayingHome = 0; + int children = 0; + for (int p = 1; p < persons.length; p++) + { + // person is not an adult + if (persons[p].getPersonIsAdult() == 0) + { + children++; + if (persons[p].getCdapActivity().equalsIgnoreCase("H")) childrenStayingHome++; + } + } + + // return the number of adults traveling = number of adults minus the + // number + // of adults staying home. + return children - childrenStayingHome; + + } + + public int getOutboundEscortChoice() { + return outboundEscortChoice; + } + public void setOutboundEscortChoice(int outboundEscortChoice) { + this.outboundEscortChoice = (short) outboundEscortChoice; + } + public int getInboundEscortChoice() { + return inboundEscortChoice; + } + public void setInboundEscortChoice(int inboundEscortChoice) { + this.inboundEscortChoice = (short) inboundEscortChoice; + } + public void calculateTimeWindowOverlaps() + { + + boolean pAdult; + boolean qAdult; + + maxAdultOverlaps = 0; + maxChildOverlaps = 0; + maxMixedOverlaps = 0; + + int[] maxAdultOverlapsP = new int[persons.length]; + int[] maxChildOverlapsP = new int[persons.length]; + + // loop over persons in the household and count available time windows + for (int p = 1; p < persons.length; p++) + { + + // determine if person p is an adult -- that is, person is not any + // of the + // three child types + pAdult = persons[p].getPersonIsPreschoolChild() == 0 + && persons[p].getPersonIsStudentNonDriving() == 0 + && persons[p].getPersonIsStudentDriving() == 0; + + // loop over person indices to compute length of pairwise available + // time windows. + for (int q = 1; q < persons.length; q++) + { + + if (p == q) continue; + + // determine if person q is an adult -- that is, person is not + // any of the three child types + qAdult = persons[q].getPersonIsPreschoolChild() == 0 + && persons[q].getPersonIsStudentNonDriving() == 0 + && persons[q].getPersonIsStudentDriving() == 0; + + // get the length of the maximum pairwise available time window + // between persons p and q. + int maxWindow = persons[p].getMaximumContinuousPairwiseAvailableWindow(persons[q] + .getTimeWindows()); + + // determine max time window overlap between adult pairs, + // children pairs, and mixed pairs in the household + // for max windows in all pairs in hh, don't need to check q,p + // once we'alread done p,q, so skip q <= p. + if (q > p) + { + if (pAdult && qAdult) + { + if (maxWindow > maxAdultOverlaps) maxAdultOverlaps = maxWindow; + } else if (!pAdult && !qAdult) + { + if (maxWindow > maxChildOverlaps) maxChildOverlaps = maxWindow; + } else + { + if (maxWindow > maxMixedOverlaps) maxMixedOverlaps = maxWindow; + } + } + + // determine the max time window overlap between this person and + // other household adults and children. + if (qAdult) + { + if (maxWindow > maxAdultOverlapsP[p]) maxAdultOverlapsP[p] = maxWindow; + } else + { + if (maxWindow > maxChildOverlapsP[p]) maxChildOverlapsP[p] = maxWindow; + } + + } // end of person q + + // set person attributes + persons[p].setMaxAdultOverlaps(maxAdultOverlapsP[p]); + persons[p].setMaxChildOverlaps(maxChildOverlapsP[p]); + + } // end of person p + + } + + public boolean[] getAvailableJointTourTimeWindows(Tour t, int[] altStarts, int[] altEnds) + { + int[] participatingPersonIndices = t.getPersonNumArray(); + + // availability array for each person + boolean[][] availability = new boolean[participatingPersonIndices.length][]; + + for (int i = 0; i < participatingPersonIndices.length; i++) + { + + int personNum = participatingPersonIndices[i]; + Person person = persons[personNum]; + + // availability array is 1-based indexing + availability[i] = new boolean[altStarts.length + 1]; + + for (int k = 1; k <= altStarts.length; k++) + { + int start = altStarts[k - 1]; + int end = altEnds[k - 1]; + availability[i][k] = person.isWindowAvailable(start, end); + } + + } + + boolean[] jointAvailability = new boolean[availability[0].length]; + + for (int k = 0; k < jointAvailability.length; k++) + { + jointAvailability[k] = true; + for (int i = 0; i < participatingPersonIndices.length; i++) + { + if (!availability[i][k]) + { + jointAvailability[k] = false; + break; + } + } + } + + return jointAvailability; + + } + + public void scheduleJointTourTimeWindows(Tour t, int start, int end) + { + int[] participatingPersonIndices = t.getPersonNumArray(); + for (int i : participatingPersonIndices) + { + Person person = persons[i]; + person.scheduleWindow(start, end); + } + } + + public void createJointTourArray() + { + jointTours = new Tour[0]; + } + + public void createJointTourArray(Tour tour1) + { + jointTours = new Tour[1]; + tour1.setTourOrigMgra(homeMgra); + tour1.setTourDestMgra(0); + jointTours[0] = tour1; + } + + public void createJointTourArray(Tour tour1, Tour tour2) + { + jointTours = new Tour[2]; + tour1.setTourOrigMgra(homeMgra); + tour1.setTourDestMgra(0); + tour1.setTourId(0); + tour2.setTourOrigMgra(homeMgra); + tour2.setTourDestMgra(0); + tour2.setTourId(1); + jointTours[0] = tour1; + jointTours[1] = tour2; + } + + public Tour[] getJointTourArray() + { + return jointTours; + } + + public void initializeForAoRestart() + { + jointTours = null; + + aoModelAutos = 0; + cdapModelPattern = null; + imtfModelPattern = 0; + jtfModelPattern = null; + + tpRandomCount = 0; + fpRandomCount = 0; + ieRandomCount = 0; + cdapRandomCount = 0; + imtfRandomCount = 0; + imtodRandomCount = 0; + awfRandomCount = 0; + awlRandomCount = 0; + awtodRandomCount = 0; + jtfRandomCount = 0; + jtlRandomCount = 0; + jtodRandomCount = 0; + inmtfRandomCount = 0; + inmtlRandomCount = 0; + inmtodRandomCount = 0; + stfRandomCount = 0; + stlRandomCount = 0; + + maxAdultOverlaps = 0; + maxChildOverlaps = 0; + maxMixedOverlaps = 0; + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForAoRestart(); + + } + + public void initializeForImtfRestart() + { + jointTours = null; + + imtfModelPattern = 0; + jtfModelPattern = null; + + imtodRandomCount = 0; + jtfRandomCount = 0; + jtlRandomCount = 0; + jtodRandomCount = 0; + inmtfRandomCount = 0; + inmtlRandomCount = 0; + inmtodRandomCount = 0; + awfRandomCount = 0; + awlRandomCount = 0; + awtodRandomCount = 0; + stfRandomCount = 0; + stlRandomCount = 0; + + maxAdultOverlaps = 0; + maxChildOverlaps = 0; + maxMixedOverlaps = 0; + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForImtfRestart(); + + } + + public void initializeForJtfRestart() + { + + jtfModelPattern = null; + + jtfRandomCount = 0; + jtlRandomCount = 0; + jtodRandomCount = 0; + inmtfRandomCount = 0; + inmtlRandomCount = 0; + inmtodRandomCount = 0; + awfRandomCount = 0; + awlRandomCount = 0; + awtodRandomCount = 0; + stfRandomCount = 0; + stlRandomCount = 0; + + initializeWindows(); + + if (jointTours != null) + { + for (Tour t : jointTours) + { + t.clearStopModelResults(); + } + } + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForJtfRestart(); + + jointTours = null; + + } + + public void initializeForInmtfRestart() + { + + inmtfRandomCount = 0; + inmtlRandomCount = 0; + inmtodRandomCount = 0; + awfRandomCount = 0; + awlRandomCount = 0; + awtodRandomCount = 0; + stfRandomCount = 0; + stlRandomCount = 0; + + initializeWindows(); + + if (jointTours != null) + { + for (Tour t : jointTours) + { + for (int i : t.getPersonNumArray()) + persons[i].scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + } + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForInmtfRestart(); + + } + + public void initializeForAwfRestart() + { + + awfRandomCount = 0; + awlRandomCount = 0; + awtodRandomCount = 0; + stfRandomCount = 0; + stlRandomCount = 0; + + initializeWindows(); + + if (jointTours != null) + { + for (Tour t : jointTours) + { + for (int i : t.getPersonNumArray()) + persons[i].scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + } + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForAwfRestart(); + + } + + public void initializeForStfRestart() + { + + stfRandomCount = 0; + stlRandomCount = 0; + + for (int i = 1; i < persons.length; i++) + persons[i].initializeForStfRestart(); + + } + + public long getSeed() { + return seed; + } + public void setSeed(long seed) { + this.seed = seed; + } + + public float getAutoOwnershipLogsum() { + return autoOwnershipLogsum; + } + + public void setAutoOwnershipLogsum(float autoOwnershipLogsum) { + this.autoOwnershipLogsum = autoOwnershipLogsum; + } + + public float getTransponderLogsum() { + return transponderLogsum; + } + + public void setTransponderLogsum(float transponderLogsum) { + this.transponderLogsum = transponderLogsum; + } + + public float getCdapLogsum() { + return cdapLogsum; + } + + public void setCdapLogsum(float cdapLogsum) { + this.cdapLogsum = cdapLogsum; + } + + public float getJtfLogsum() { + return jtfLogsum; + } + + public void setJtfLogsum(float jtfLogsum) { + this.jtfLogsum = jtfLogsum; + } + + public void logHouseholdObject(String titleString, Logger logger) + { + + int totalChars = 72; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "H"; + + logger.info(separater); + logger.info(titleString); + logger.info(separater); + + Household.logHelper(logger, "hhId: ", hhId, totalChars); + Household.logHelper(logger, "debugChoiceModels: ", debugChoiceModels ? "True" : "False", + totalChars); + Household.logHelper(logger, "hhIncome: ", hhIncomeCategory, totalChars); + Household.logHelper(logger, "hhIncomeInDollars: ", hhIncomeInDollars, totalChars); + Household.logHelper(logger, "hhSize: ", hhSize, totalChars); + Household.logHelper(logger, "hhType: ", hhType, totalChars); + Household.logHelper(logger, "hhWorkers: ", hhWorkers, totalChars); + Household.logHelper(logger, "homeTaz: ", homeTaz, totalChars); + Household.logHelper(logger, "homeMgra: ", homeMgra, totalChars); + Household.logHelper(logger, "homeWalkSubzone: ", homeWalkSubzone, totalChars); + Household.logHelper(logger, "aoModelAutos: ", aoModelAutos, totalChars); + Household.logHelper(logger, "cdapModelPattern: ", cdapModelPattern, totalChars); + Household.logHelper(logger, "imtfModelPattern: ", imtfModelPattern, totalChars); + Household.logHelper(logger, "outboundEscortChoice: ", outboundEscortChoice, totalChars); + Household.logHelper(logger, "inboundEscortChoice: ", inboundEscortChoice, totalChars); + Household.logHelper(logger, "jtfModelPattern: ", jtfModelPattern, totalChars); + Household.logHelper(logger, "randomCount: ", randomCount, totalChars); + if (uwslRandomCountList.size() > 0) + { + for (int i : uwslRandomCountList.keySet()) + Household.logHelper(logger, String.format("uwslRandomCount[%d]: ", i), + uwslRandomCountList.get(i), totalChars); + } else + { + Household.logHelper(logger, "uwslRandomCount[0]: ", 0, totalChars); + } + Household.logHelper(logger, "aoRandomCount: ", aoRandomCount, totalChars); + Household.logHelper(logger, "tpRandomCount: ", tpRandomCount, totalChars); + Household.logHelper(logger, "fpRandomCount: ", fpRandomCount, totalChars); + Household.logHelper(logger, "ieRandomCount: ", ieRandomCount, totalChars); + Household.logHelper(logger, "cdapRandomCount: ", cdapRandomCount, totalChars); + Household.logHelper(logger, "imtfRandomCount: ", imtfRandomCount, totalChars); + Household.logHelper(logger, "imtodRandomCount: ", imtodRandomCount, totalChars); + Household.logHelper(logger, "awfRandomCount: ", awfRandomCount, totalChars); + Household.logHelper(logger, "awlRandomCount: ", awlRandomCount, totalChars); + Household.logHelper(logger, "awtodRandomCount: ", awtodRandomCount, totalChars); + Household.logHelper(logger, "jtfRandomCount: ", jtfRandomCount, totalChars); + Household.logHelper(logger, "jtlRandomCount: ", jtlRandomCount, totalChars); + Household.logHelper(logger, "jtodRandomCount: ", jtodRandomCount, totalChars); + Household.logHelper(logger, "inmtfRandomCount: ", inmtfRandomCount, totalChars); + Household.logHelper(logger, "inmtlRandomCount: ", inmtlRandomCount, totalChars); + Household.logHelper(logger, "inmtodRandomCount: ", inmtodRandomCount, totalChars); + Household.logHelper(logger, "stfRandomCount: ", stfRandomCount, totalChars); + Household.logHelper(logger, "stlRandomCount: ", stlRandomCount, totalChars); + Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); + Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); + Household.logHelper(logger, "maxMixedOverlaps: ", maxMixedOverlaps, totalChars); + + String tempString = String.format("Joint Tours[%s]:", + jointTours == null ? "" : String.valueOf(jointTours.length)); + logger.info(tempString); + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public void logPersonObject(String titleString, Logger logger, Person person) + { + + int totalChars = 114; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "P"; + + logger.info(separater); + logger.info(titleString); + logger.info(separater); + + person.logPersonObject(logger, totalChars); + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public void logTourObject(String titleString, Logger logger, Person person, Tour tour) + { + + int totalChars = 119; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "T"; + + logger.info(separater); + logger.info(titleString); + logger.info(separater); + + person.logTourObject(logger, totalChars, tour); + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public void logStopObject(String titleString, Logger logger, Stop stop, + ModelStructure modelStructure) + { + + int totalChars = 119; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "S"; + + logger.info(separater); + logger.info(titleString); + logger.info(separater); + + stop.logStopObject(logger, totalChars); + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public void logEntireHouseholdObject(String titleString, Logger logger) + { + + int totalChars = 60; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "="; + + logger.info(separater); + logger.info(titleString); + logger.info(separater); + + separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + Household.logHelper(logger, "hhId: ", hhId, totalChars); + Household.logHelper(logger, "debugChoiceModels: ", debugChoiceModels ? "True" : "False", + totalChars); + Household.logHelper(logger, "hhIncome: ", hhIncomeCategory, totalChars); + Household.logHelper(logger, "hhIncomeInDollars: ", hhIncomeInDollars, totalChars); + Household.logHelper(logger, "hhSize: ", hhSize, totalChars); + Household.logHelper(logger, "hhType: ", hhType, totalChars); + Household.logHelper(logger, "hhWorkers: ", hhWorkers, totalChars); + Household.logHelper(logger, "homeTaz: ", homeTaz, totalChars); + Household.logHelper(logger, "homeMgra: ", homeMgra, totalChars); + Household.logHelper(logger, "homeWalkSubzone: ", homeWalkSubzone, totalChars); + Household.logHelper(logger, "aoModelAutos: ", aoModelAutos, totalChars); + Household.logHelper(logger, "cdapModelPattern: ", cdapModelPattern, totalChars); + Household.logHelper(logger, "imtfModelPattern: ", imtfModelPattern, totalChars); + Household.logHelper(logger, "outboundEscortChoice: ", outboundEscortChoice, totalChars); + Household.logHelper(logger, "inboundEscortChoice: ", inboundEscortChoice, totalChars); + Household.logHelper(logger, "jtfModelPattern: ", jtfModelPattern, totalChars); + Household.logHelper(logger, "randomCount: ", randomCount, totalChars); + if (uwslRandomCountList.size() > 0) + { + for (int i : uwslRandomCountList.keySet()) + Household.logHelper(logger, String.format("uwslRandomCount[%d]: ", i), + uwslRandomCountList.get(i), totalChars); + } else + { + Household.logHelper(logger, "uwslRandomCount[0]: ", 0, totalChars); + } + Household.logHelper(logger, "aoRandomCount: ", aoRandomCount, totalChars); + Household.logHelper(logger, "tpRandomCount: ", tpRandomCount, totalChars); + Household.logHelper(logger, "fpRandomCount: ", fpRandomCount, totalChars); + Household.logHelper(logger, "ieRandomCount: ", ieRandomCount, totalChars); + Household.logHelper(logger, "cdapRandomCount: ", cdapRandomCount, totalChars); + Household.logHelper(logger, "imtfRandomCount: ", imtfRandomCount, totalChars); + Household.logHelper(logger, "imtodRandomCount: ", imtodRandomCount, totalChars); + Household.logHelper(logger, "awfRandomCount: ", awfRandomCount, totalChars); + Household.logHelper(logger, "awlRandomCount: ", awlRandomCount, totalChars); + Household.logHelper(logger, "awtodRandomCount: ", awtodRandomCount, totalChars); + Household.logHelper(logger, "jtfRandomCount: ", jtfRandomCount, totalChars); + Household.logHelper(logger, "jtlRandomCount: ", jtlRandomCount, totalChars); + Household.logHelper(logger, "jtodRandomCount: ", jtodRandomCount, totalChars); + Household.logHelper(logger, "inmtfRandomCount: ", inmtfRandomCount, totalChars); + Household.logHelper(logger, "inmtlRandomCount: ", inmtlRandomCount, totalChars); + Household.logHelper(logger, "inmtodRandomCount: ", inmtodRandomCount, totalChars); + Household.logHelper(logger, "stfRandomCount: ", stfRandomCount, totalChars); + Household.logHelper(logger, "stlRandomCount: ", stlRandomCount, totalChars); + Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); + Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); + Household.logHelper(logger, "maxMixedOverlaps: ", maxMixedOverlaps, totalChars); + + if (jointTours != null) + { + logger.info("Joint Tours:"); + if (jointTours.length > 0) + { + for (int i = 0; i < jointTours.length; i++) + jointTours[i].logEntireTourObject(logger); + } else logger.info(" No joint tours"); + } else logger.info(" No joint tours"); + + logger.info("Person Objects:"); + for (int i = 1; i < persons.length; i++) + persons[i].logEntirePersonObject(logger); + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public static void logHelper(Logger logger, String label, int value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } + + public static void logHelper(Logger logger, String label, String value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } + + public static void logHelper(Logger logger, String label, float value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%df", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } + public enum HouseholdType + { + nul, FAMILY_MARRIED, FAMILY_MALE_NO_WIFE, FAMILY_FEMALE_NO_HUSBAND, NON_FAMILY_MALE_ALONE, NON_FAMILY_MALE_NOT_ALONE, NON_FAMILY_FEMALE_ALONE, NON_FAMILY_FEMALE_NOT_ALONE + } + + /** + * Iterate through persons in household grab all adults and add them to an ArrayList of adults. Return the list. + * + * @return An ArrayList of adult household members. + */ + public List getAdultPersons() { + + List adultList = new ArrayList(); + + for(int i = 1; i < persons.length; ++i){ + Person p = persons[i]; + if((p.getPersonIsFullTimeWorker()==1)||(p.getPersonIsPartTimeWorker()==1)||(p.getPersonIsNonWorkingAdultUnder65()==1) + ||(p.getPersonIsUniversityStudent()==1)||(p.getPersonIsNonWorkingAdultOver65()==1)) + adultList.add(p); + + } + return adultList; + } + /** + * Iterate through persons in household grab all adults and add them to an ArrayList of adults. Return the list. + * + * @return An ArrayList of adult household members. + */ + public List getActiveAdultPersons() { + + List adultList = new ArrayList(); + + for(int i = 1; i < persons.length; ++i){ + Person p = persons[i]; + if(p.isActiveAdult()) + adultList.add(p); + + } + return adultList; + } + + /** + * Count and return the number of active adults in the household. + * + * @return The number of active adults. + */ + public int getNumberActiveAdults(){ + + int numberActiveAdults=0; + for(int i = 1; i < persons.length; ++i){ + if(persons[i].isActiveAdult()) + ++numberActiveAdults; + + } + return numberActiveAdults; + } + + + /** + * Iterate through persons in household grab all children and add them to an ArrayList of children. Return the list. + * + * @return An ArrayList of adult household members. + */ + public List getChildPersons() { + + List childList = new ArrayList(); + + for(int i = 1; i < persons.length; ++i){ + Person p = persons[i]; + if((p.getPersonIsPreschoolChild()==1)||(p.getPersonIsStudentNonDriving()==1)||(p.getPersonIsStudentDriving()==1)) + childList.add(p); + + } + return childList; + } + } diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java new file mode 100644 index 0000000..11daef6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java @@ -0,0 +1,334 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +// + +public class HouseholdAtWorkSubtourFrequencyModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdAtWorkSubtourFrequencyModel.class); + private transient Logger tourFreq = Logger.getLogger("tourFreq"); + + private static final String AWTF_CONTROL_FILE_TARGET = "awtf.uec.file"; + private static final String AWTF_DATA_SHEET_TARGET = "awtf.data.page"; + private static final String AWTF_MODEL_SHEET_TARGET = "awtf.model.page"; + + // model results + // private static final int NO_SUBTOURS = 1; + private static final int ONE_EAT = 2; + private static final int ONE_BUSINESS = 3; + private static final int ONE_OTHER = 4; + private static final int TWO_BUSINESS = 5; + private static final int TWO_OTHER = 6; + private static final int ONE_EAT_ONE_BUSINESS = 7; + + private AtWorkSubtourFrequencyDMU dmuObject; + private ChoiceModelApplication choiceModelApplication; + + private ModelStructure modelStructure; + private String[] alternativeNames; + + /** + * Constructor establishes the ChoiceModelApplication, which applies the + * logit model via the UEC spreadsheet. + * + * @param dmuObject + * is the UEC dmu object for this choice model + * @param uecFileName + * is the UEC control file name + * @param resourceBundle + * is the application ResourceBundle, from which a properties + * file HashMap will be created for the UEC + * @param tazDataManager + * is the object used to interact with the zonal data table + * @param modelStructure + * is the ModelStructure object that defines segmentation and + * other model structure relate atributes + */ + public HouseholdAtWorkSubtourFrequencyModel(HashMap propertyMap, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) + { + + this.modelStructure = modelStructure; + setUpModels(propertyMap, dmuFactory); + + } + + private void setUpModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up %s tour frequency choice model.", + ModelStructure.AT_WORK_CATEGORY)); + + // locate the individual mandatory tour frequency choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String awtfUecFile = propertyMap.get(AWTF_CONTROL_FILE_TARGET); + awtfUecFile = uecPath + awtfUecFile; + + int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, AWTF_DATA_SHEET_TARGET); + int modelPage = Util.getIntegerValueFromPropertyMap(propertyMap, AWTF_MODEL_SHEET_TARGET); + + dmuObject = dmuFactory.getAtWorkSubtourFrequencyDMU(); + + // set up the model + choiceModelApplication = new ChoiceModelApplication(awtfUecFile, modelPage, dataPage, + propertyMap, (VariableTable) dmuObject); + + } + + /** + * Applies the model for the array of households that are stored in the + * HouseholdDataManager. The results are summarized by person type. + * + * @param householdDataManager + * is the object containg the Household objects for which this + * model is to be applied. + */ + public void applyModel(Household household) + { + + int choice = -1; + String personTypeString = ""; + + Logger modelLogger = tourFreq; + if (household.getDebugChoiceModels()) + household.logHouseholdObject( + "Pre AtWork Subtour Frequency Choice HHID=" + household.getHhId() + " Object", + modelLogger); + + // get this household's person array + Person[] personArray = household.getPersons(); + + // set the household id, origin taz, hh taz, and debugFlag=false in the + // dmu + dmuObject.setHouseholdObject(household); + + // loop through the person array (1-based) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + // count the results by person type + personTypeString = person.getPersonType(); + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), personTypeString); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + // loop through the work tours for this person + ArrayList tourList = person.getListOfWorkTours(); + if (tourList == null) continue; + + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + int workTourIndex = 0; + for (Tour workTour : tourList) + { + + try + { + + // set the person and tour object + dmuObject.setPersonObject(person); + dmuObject.setTourObject(workTour); + + // write debug header + if (household.getDebugChoiceModels() || person.getPersonTypeNumber() == 7) + { + + choiceModelDescription = String + .format("At-work Subtour Frequency Choice Model:"); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, workTourId=%d", person + .getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), workTour.getTourId()); + choiceModelApplication.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = choiceModelDescription + " for " + decisionMakerLabel + + "."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + } + + // set the availability array for the tour frequency model + alternativeNames = choiceModelApplication.getAlternativeNames(); + int numberOfAlternatives = alternativeNames.length; + boolean[] availabilityArray = new boolean[numberOfAlternatives + 1]; + Arrays.fill(availabilityArray, true); + + // create the sample array + int[] sampleArray = new int[availabilityArray.length]; + Arrays.fill(sampleArray, 1); + + // compute the utilities + IndexValues index = dmuObject.getDmuIndexValues(); + index.setHHIndex(household.getHhId()); + index.setZoneIndex(household.getHhTaz()); + index.setOriginZone(workTour.getTourOrigMgra()); + index.setDestZone(workTour.getTourDestMgra()); + index.setDebug(household.getDebugChoiceModels()); + + if (household.getDebugChoiceModels()) + { + household.logTourObject(loggingHeader, modelLogger, person, workTour); + } + + float logsum = (float) choiceModelApplication.computeUtilities(dmuObject, index, availabilityArray, + sampleArray); + + workTour.setSubtourFreqLogsum(logsum); + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, + // make choice. + if (choiceModelApplication.getAvailabilityCount() > 0) choice = choiceModelApplication + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for j=%d, tourNum=%d, HHID=%d, no available at-work frequency alternatives to choose from in choiceModelApplication.", + j, workTourIndex, person.getHouseholdObject().getHhId())); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = choiceModelApplication.getUtilities(); + double[] probabilities = choiceModelApplication.getProbabilities(); + + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + + personTypeString); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("-------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < alternativeNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %-16s", k + 1, + alternativeNames[k]); + modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %-16s", choice, + alternativeNames[choice - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + choiceModelApplication.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + choiceModelApplication.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, choice); + + // write UEC calculation results to separate model + // specific + // log file + choiceModelApplication.logUECResults(modelLogger, loggingHeader); + + } + + workTour.setSubtourFreqChoice(choice); + + // set the person choices + if (choice == ONE_EAT) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkEatPurposeName()); + } else if (choice == ONE_BUSINESS) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkBusinessPurposeName()); + } else if (choice == ONE_OTHER) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkMaintPurposeName()); + } else if (choice == TWO_BUSINESS) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkBusinessPurposeName()); + id = workTourIndex * 10 + 2; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkBusinessPurposeName()); + } else if (choice == TWO_OTHER) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkMaintPurposeName()); + id = workTourIndex * 10 + 2; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkMaintPurposeName()); + } else if (choice == ONE_EAT_ONE_BUSINESS) + { + int id = workTourIndex * 10 + 1; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkEatPurposeName()); + id = workTourIndex * 10 + 2; + person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), + modelStructure.getAtWorkBusinessPurposeName()); + } + + } catch (Exception e) + { + logger.error(String.format("Exception caught for j=%d, tourNum=%d, HHID=%d.", + j, workTourIndex, household.getHhId())); + throw new RuntimeException(); + } + + workTourIndex++; + + } + + } // j (person loop) + + household.setAwfRandomCount(household.getHhRandomCount()); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java new file mode 100644 index 0000000..cfa8483 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java @@ -0,0 +1,343 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import com.pb.common.calculator.VariableTable; +import com.pb.common.model.ModelException; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class HouseholdAutoOwnershipModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdAutoOwnershipModel.class); + private transient Logger aoLogger = Logger.getLogger("ao"); + + private static final String AO_CONTROL_FILE_TARGET = "ao.uec.file"; + private static final String AO_MODEL_SHEET_TARGET = "ao.model.page"; + private static final String AO_DATA_SHEET_TARGET = "ao.data.page"; + + private static final int AUTO_SOV_TIME_INDEX = 10; + private static final int AUTO_LOGSUM_INDEX = 6; + private static final int TRANSIT_LOGSUM_INDEX = 8; + private static final int DT_RAIL_PROP_INDEX = 10; + + private AccessibilitiesTable accTable; + private MandatoryAccessibilitiesCalculator mandAcc; + private ChoiceModelApplication aoModel; + private AutoOwnershipChoiceDMU aoDmuObject; + + private int[] totalAutosByAlt; + private int[] automatedVehiclesByAlt; + private int[] conventionalVehiclesByAlt; + + public HouseholdAutoOwnershipModel(HashMap rbMap, + CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, + MandatoryAccessibilitiesCalculator myMandAcc) + { + + logger.info("setting up AO choice model."); + + // set the aggAcc class variable, which will serve as a flag: null -> no + // accessibilities, !null -> set accessibilities. + // if the BuildAccessibilities object is null, the AO utility does not + // need + // to use the accessibilities components. + accTable = myAccTable; + mandAcc = myMandAcc; + + // locate the auto ownership UEC + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String autoOwnershipUecFile = rbMap.get(AO_CONTROL_FILE_TARGET); + autoOwnershipUecFile = uecPath + autoOwnershipUecFile; + + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, AO_DATA_SHEET_TARGET); + int modelPage = Util.getIntegerValueFromPropertyMap(rbMap, AO_MODEL_SHEET_TARGET); + + // create the auto ownership choice model DMU object. + aoDmuObject = dmuFactory.getAutoOwnershipDMU(); + + // create the auto ownership choice model object + aoModel = new ChoiceModelApplication(autoOwnershipUecFile, modelPage, dataPage, rbMap, + (VariableTable) aoDmuObject); + + String[] alternativeNames = aoModel.getAlternativeNames(); + calculateAlternativeArrays(alternativeNames); + + } + + /** + * Set the dmu attributes, compute the pre-AO or AO utilities, and select an + * alternative + * + * @param hhObj + * for which to apply thye model + * @param preAutoOwnership + * is true if running pre-auto ownership, or false to run primary + * auto ownership model. + */ + + public void applyModel(Household hhObj, boolean preAutoOwnership) + { + + // update the AO dmuObject for this hh + aoDmuObject.setHouseholdObject(hhObj); + aoDmuObject.setDmuIndexValues(hhObj.getHhId(), hhObj.getHhMgra(), hhObj.getHhMgra(), 0); + + // set the non-mandatory accessibility values for the home MGRA. + // values used by both pre-ao and ao models. + aoDmuObject.setHomeTazAutoAccessibility(accTable.getAggregateAccessibility("auto", + hhObj.getHhMgra())); + aoDmuObject.setHomeTazTransitAccessibility(accTable.getAggregateAccessibility("transit", + hhObj.getHhMgra())); + aoDmuObject.setHomeTazNonMotorizedAccessibility(accTable.getAggregateAccessibility( + "nonmotor", hhObj.getHhMgra())); + aoDmuObject.setHomeTazMaasAccessibility(accTable.getAggregateAccessibility("maas", + hhObj.getHhMgra())); + + + + if (preAutoOwnership) + { + + aoDmuObject.setUseAccessibilities(false); + + } else + { + + aoDmuObject.setUseAccessibilities(true); + + // compute the disaggregate accessibilities for the home MGRA to + // work and + // school MGRAs summed accross workers and students + double workAutoDependency = 0.0; + double schoolAutoDependency = 0.0; + double workRailProp = 0.0; + double schoolRailProp = 0.0; + double workAutoTime = 0.0; + Person[] persons = hhObj.getPersons(); + for (int i = 1; i < persons.length; i++) + { + + // sum over all workers (full time or part time) + if (persons[i].getPersonIsWorker() == 1) + { + + int workMgra = persons[i].getWorkLocation(); + if (workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + + // Non-Motorized Factor = 0.5*MIN(MAX(DIST,1),3)-0.5 + // if 0 <= dist < 1, nmFactor = 0 + // if 1 <= dist <= 3, nmFactor = [0.0, 1.0] + // if 3 <= dist, nmFactor = 1.0 + double nmFactor = 0.5 * (Math.min( + Math.max(persons[i].getWorkLocationDistance(), 1.0), 3.0)) - 0.5; + + // if auto logsum < transit logsum, do not accumulate + // auto + // dependency + double[] workerAccessibilities = mandAcc + .calculateWorkerMandatoryAccessibilities(hhObj.getHhMgra(), + workMgra); + workAutoTime += workerAccessibilities[AUTO_SOV_TIME_INDEX]; + if (workerAccessibilities[AUTO_LOGSUM_INDEX] >= workerAccessibilities[TRANSIT_LOGSUM_INDEX]) + { + double logsumDiff = workerAccessibilities[AUTO_LOGSUM_INDEX] + - workerAccessibilities[TRANSIT_LOGSUM_INDEX]; + + // need to scale and cap logsum difference + logsumDiff = Math.min(logsumDiff / 3.0, 1.0); + workAutoDependency += (logsumDiff * nmFactor); + } + + workRailProp += workerAccessibilities[DT_RAIL_PROP_INDEX]; + + } + + } + + // sum over all students of driving age + if (persons[i].getPersonIsUniversityStudent() == 1 + || persons[i].getPersonIsStudentDriving() == 1) + { + + int schoolMgra = persons[i].getUsualSchoolLocation(); + if (schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) + { + + // Non-Motorized Factor = 0.5*MIN(MAX(DIST,1),3)-0.5 + // if 0 <= dist < 1, nmFactor = 0 + // if 1 <= dist <= 3, nmFactor = [0.0, 1.0] + // if 3 <= dist, nmFactor = 1.0 + double nmFactor = 0.5 * (Math.min( + Math.max(persons[i].getWorkLocationDistance(), 1.0), 3.0)) - 0.5; + + // if auto logsum < transit logsum, do not accumulate + // auto + // dependency + double[] studentAccessibilities = mandAcc + .calculateStudentMandatoryAccessibilities(hhObj.getHhMgra(), + schoolMgra); + if (studentAccessibilities[AUTO_LOGSUM_INDEX] >= studentAccessibilities[TRANSIT_LOGSUM_INDEX]) + { + double logsumDiff = studentAccessibilities[AUTO_LOGSUM_INDEX] + - studentAccessibilities[TRANSIT_LOGSUM_INDEX]; + + // need to scale and cap logsum difference + logsumDiff = Math.min(logsumDiff / 3.0, 1.0); + schoolAutoDependency += (logsumDiff * nmFactor); + } + + schoolRailProp += studentAccessibilities[DT_RAIL_PROP_INDEX]; + + } + } + + } + + aoDmuObject.setWorkAutoDependency(workAutoDependency); + aoDmuObject.setSchoolAutoDependency(schoolAutoDependency); + + aoDmuObject.setWorkersRailProportion(workRailProp); + aoDmuObject.setStudentsRailProportion(schoolRailProp); + + aoDmuObject.setWorkAutoTime(workAutoTime); + + } + + // compute utilities and choose auto ownership alternative. + float logsum = (float) aoModel.computeUtilities(aoDmuObject, aoDmuObject.getDmuIndexValues()); + + hhObj.setAutoOwnershipLogsum(logsum); + Random hhRandom = hhObj.getHhRandom(); + int randomCount = hhObj.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt = -1; + if (aoModel.getAvailabilityCount() > 0) + { + try + { + chosenAlt = aoModel.getChoiceResult(rn); + } catch (ModelException e) + { + logger.error(String.format( + "exception caught for HHID=%d in choiceModelApplication.", hhObj.getHhId())); + } + } else + { + logger.error(String + .format("error: HHID=%d has no available auto ownership alternatives to choose from in choiceModelApplication.", + hhObj.getHhId())); + throw new RuntimeException(); + } + + // write choice model alternative info to log file + if (hhObj.getDebugChoiceModels() || chosenAlt < 0) + { + + String loggerString = (preAutoOwnership ? "Pre-AO without" : "AO with") + + " accessibilities, Household " + hhObj.getHhId() + " Object"; + hhObj.logHouseholdObject(loggerString, aoLogger); + + double[] utilities = aoModel.getUtilities(); + double[] probabilities = aoModel.getProbabilities(); + + aoLogger.info("Alternative Utility Probability CumProb"); + aoLogger.info("-------------------- --------------- ------------ ------------"); + + double cumProb = 0.0; + for (int k = 0; k < aoModel.getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + aoLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", k + " autos", utilities[k], + probabilities[k], cumProb)); + } + + aoLogger.info(" "); + aoLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", chosenAlt, rn, + randomCount)); + + aoLogger.info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + aoLogger.info(""); + aoLogger.info(""); + + // write choice model alternative info to debug log file + aoModel.logAlternativesInfo("Household Auto Ownership Choice", + String.format("HH_%d", hhObj.getHhId())); + aoModel.logSelectionInfo("Household Auto Ownership Choice", + String.format("HH_%d", hhObj.getHhId()), rn, chosenAlt); + + // write UEC calculation results to separate model specific log file + aoModel.logUECResults(aoLogger, + String.format("Household Auto Ownership Choice, HH_%d", hhObj.getHhId())); + } + + if (preAutoOwnership) hhObj.setPreAoRandomCount(hhObj.getHhRandomCount()); + else hhObj.setAoRandomCount(hhObj.getHhRandomCount()); + + int autos = totalAutosByAlt[chosenAlt-1]; + int AVs = automatedVehiclesByAlt[chosenAlt-1]; + int CVs = conventionalVehiclesByAlt[chosenAlt-1]; + hhObj.setHhAutos(autos); + hhObj.setAutomatedVehicles(AVs); + hhObj.setConventionalVehicles(CVs); + + } + + + /** + * This is a helper method that iterates through the alternative names + * in the auto ownership UEC and searches through each name to collect + * the total number of autos (in the first position of the name character + * array), the number of AVs for the alternative (preceded by the "AV" substring) + * and the number of CVs for the alternative (preceded by the "CV" substring). The + * results are stored in the arrays: + * + * totalAutosByAlt + * automatedVehiclesByAlt + * conventionalVehiclesByAlt + * + * @param alternativeNames The array of alternative names. + */ + private void calculateAlternativeArrays(String[] alternativeNames){ + + totalAutosByAlt = new int[alternativeNames.length]; + automatedVehiclesByAlt = new int[alternativeNames.length]; + conventionalVehiclesByAlt = new int[alternativeNames.length]; + + + //iterate thru names + for(int i = 0; i < alternativeNames.length;++i){ + + String altName = alternativeNames[i]; + + //find the number of cars; first element of name (e.g. 0_CARS) + int autos = new Integer(altName.substring(0,1)).intValue(); + int AVs=0; + int HVs=0; + int AVPosition = altName.indexOf("AV"); + if(AVPosition>=0) + AVs = new Integer(altName.substring(AVPosition-1, AVPosition)).intValue(); + int HVPosition = altName.indexOf("HV"); + if(HVPosition>=0) + HVs = new Integer(altName.substring(HVPosition-1, HVPosition)).intValue(); + + totalAutosByAlt[i] = autos; + automatedVehiclesByAlt[i] = AVs; + conventionalVehiclesByAlt[i] = HVs; + + } + + } + + +} + diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java new file mode 100644 index 0000000..3a5d92a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java @@ -0,0 +1,254 @@ +package org.sandag.abm.ctramp; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; + +import org.apache.log4j.Logger; +import org.jppf.client.JPPFClient; +import org.jppf.client.JPPFJob; +import org.jppf.node.protocol.DataProvider; +import org.jppf.node.protocol.MemoryMapDataProvider; +import org.jppf.node.protocol.Task; + +import com.pb.common.calculator.MatrixDataServerIf; + +public class HouseholdChoiceModelRunner +{ + + private Logger logger = Logger.getLogger(HouseholdChoiceModelRunner.class); + + private static int PACKET_SIZE = 0; + + private static String PROPERTIES_NUM_INITIALIZATION_PACKETS = "number.initialization.packets"; + private static String PROPERTIES_INITIALIZATION_PACKET_SIZE = "initialization.packet.size"; + private static int NUM_INITIALIZATION_PACKETS = 0; + private static int INITIALIZATION_PACKET_SIZE = 0; + + private int ONE_HH_ID = -1; + + private static final String HOUSEHOLD_CHOICE_PACKET_SIZE = "distributed.task.packet.size"; + private static final String RUN_THIS_HOUSEHOLD_ONLY = "run.this.household.only"; + + private HashMap propertyMap; + private String restartModelString; + private MatrixDataServerIf ms; + private HouseholdDataManagerIf hhDataManager; + private ModelStructure modelStructure; + private CtrampDmuFactoryIf dmuFactory; + + private JPPFClient jppfClient; + private boolean logResults=false; + + // The number of initialization packets are the number of "small" packets + // submited at the beginning of a + // distributed task to minimize synchronization issues that significantly + // slow + // down model object setup. + // It is assumed that after theses small packets have run, all the model + // objects + // will have been setup, + // and the task objects can process much bigger chuncks of households. + + public HouseholdChoiceModelRunner(HashMap propertyMap, JPPFClient jppfClient, + String restartModelString, HouseholdDataManagerIf hhDataManager, MatrixDataServerIf ms, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) + { + setupHouseholdChoiceModelRunner(propertyMap, jppfClient, restartModelString, hhDataManager, + ms, modelStructure, dmuFactory); + } + + private void setupHouseholdChoiceModelRunner(HashMap propertyMap, + JPPFClient jppfClient, String restartModelString, HouseholdDataManagerIf hhDataManager, + MatrixDataServerIf ms, ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) + { + + this.propertyMap = propertyMap; + this.restartModelString = restartModelString; + this.hhDataManager = hhDataManager; + this.ms = ms; + this.modelStructure = modelStructure; + this.dmuFactory = dmuFactory; + this.jppfClient = jppfClient; + + String oneHhString = propertyMap.get(RUN_THIS_HOUSEHOLD_ONLY); + if (oneHhString != null) ONE_HH_ID = Integer.parseInt(oneHhString); + + String propertyValue = propertyMap.get(HOUSEHOLD_CHOICE_PACKET_SIZE); + if (propertyValue == null) PACKET_SIZE = 0; + else PACKET_SIZE = Integer.parseInt(propertyValue); + + propertyValue = propertyMap.get(PROPERTIES_NUM_INITIALIZATION_PACKETS); + if (propertyValue == null) NUM_INITIALIZATION_PACKETS = 0; + else NUM_INITIALIZATION_PACKETS = Integer.parseInt(propertyValue); + + propertyValue = propertyMap.get(PROPERTIES_INITIALIZATION_PACKET_SIZE); + if (propertyValue == null) INITIALIZATION_PACKET_SIZE = 0; + else INITIALIZATION_PACKET_SIZE = Integer.parseInt(propertyValue); + + logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + } + + /** + * + * JPPF framework based method + */ + public void runHouseholdChoiceModels() + { + + long initTime = System.currentTimeMillis(); + + submitTasks(); + + logger.info(String.format("household model runner finished %d households in %d minutes.", + hhDataManager.getNumHouseholds(), + ((System.currentTimeMillis() - initTime) / 1000) / 60)); + + } + + /** + * @param client + * is a JPPFClient object which is used to establish a connection + * to a computing node, submit tasks, and receive results. + */ + private void submitTasks() + { + + // if PACKET_SIZE was not specified, create a single task to use for all + // households + if (PACKET_SIZE == 0) PACKET_SIZE = hhDataManager.getNumHouseholds(); + + // Create a setup task object and submit it to the computing node. + // This setup task creates the HouseholdChoiceModelManager and causes it + // to + // create the necessary numuber + // of HouseholdChoiceModels objects which will operate in parallel on + // the + // computing node. + try + { + + JPPFJob job = new JPPFJob(); + job.setName("Household Choice Job"); + + DataProvider dataProvider = new MemoryMapDataProvider(); + dataProvider.setParameter("propertyMap", propertyMap); + dataProvider.setParameter("ms", ms); + dataProvider.setParameter("hhDataManager", hhDataManager); + dataProvider.setParameter("modelStructure", modelStructure); + dataProvider.setParameter("dmuFactory", dmuFactory); + dataProvider.setParameter("restartModelString", restartModelString); + job.setDataProvider(dataProvider); + + ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(hhDataManager + .getNumHouseholds()); + + int startIndex = 0; + int endIndex = 0; + int taskIndex = 1; + for (int[] startEndIndices : startEndTaskIndicesList) + { + startIndex = startEndIndices[0]; + endIndex = startEndIndices[1]; + + HouseholdChoiceModelsTaskJppf task = new HouseholdChoiceModelsTaskJppf(taskIndex, + startIndex, endIndex); + job.add(task); + taskIndex++; + } + + List> results = jppfClient.submitJob(job); + for (Task task : results) + { + if (task.getThrowable() != null) throw new Exception(task.getThrowable()); + + try + { + if(logResults){ + logger.info(String.format("HH TASK: %s returned: %s, maxAlts: %d.", + task.getId(), (String) task.getResult(), + ((HouseholdChoiceModelsTaskJppf) task).getMaxAlts())); + } + } catch (Exception e) + { + logger.error( + "Exception returned by computing node caught in HouseholdChoiceModelsTaskJppf.", + e); + throw new RuntimeException(); + } + + } + + } catch (Exception e) + { + logger.error( + "Exception caught creating/submitting/receiving HouseholdChoiceModelsTaskJppf.", + e); + throw new RuntimeException(); + } + + } + + private ArrayList getTaskHouseholdRanges(int numberOfHouseholds) + { + + ArrayList startEndIndexList = new ArrayList(); + + if (ONE_HH_ID < 0) + { + + int numInitializationHouseholds = NUM_INITIALIZATION_PACKETS + * INITIALIZATION_PACKET_SIZE; + + int startIndex = 0; + int endIndex = 0; + if (numInitializationHouseholds < numberOfHouseholds) + { + + while (endIndex < numInitializationHouseholds) + { + endIndex = startIndex + INITIALIZATION_PACKET_SIZE - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += INITIALIZATION_PACKET_SIZE; + } + + } + + while (endIndex < numberOfHouseholds - 1) + { + endIndex = startIndex + PACKET_SIZE - 1; + if (endIndex + PACKET_SIZE > numberOfHouseholds) endIndex = numberOfHouseholds - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += PACKET_SIZE; + } + + return startEndIndexList; + + } else + { + + // create a single task packet high one household id + int[] startEndIndices = new int[2]; + int index = hhDataManager.getArrayIndex(ONE_HH_ID); + startEndIndices[0] = index; + startEndIndices[1] = index; + startEndIndexList.add(startEndIndices); + + return startEndIndexList; + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java new file mode 100644 index 0000000..8bd78d8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java @@ -0,0 +1,980 @@ +package org.sandag.abm.ctramp; + +import java.util.Arrays; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.util.ObjectUtil; + +public class HouseholdChoiceModels +{ + + private transient Logger logger = Logger.getLogger(HouseholdChoiceModels.class); + + private static final String GLOBAL_MODEL_SEED_PROPERTY = "Model.Random.Seed"; + private static final int AO_SEED_OFFSET = 0; + private static final int TP_SEED_OFFSET = 1; + private static final int PP_SEED_OFFSET = 2; + private static final int CDAP_SEED_OFFSET = 3; + private static final int IMTF_SEED_OFFSET = 4; + private static final int IMTOD_SEED_OFFSET = 5; + private static final int JTF_SEED_OFFSET = 6; + private static final int JTDC_SEED_OFFSET = 7; + private static final int JTOD_SEED_OFFSET = 8; + private static final int INMTF_SEED_OFFSET = 9; + private static final int INMDC_SEED_OFFSET = 10; + private static final int INMTOD_SEED_OFFSET = 11; + private static final int AWTF_SEED_OFFSET = 12; + private static final int AWDC_SEED_OFFSET = 13; + private static final int AWTOD_SEED_OFFSET = 14; + private static final int STF_SEED_OFFSET = 15; + private static final int SLC_SEED_OFFSET = 16; + private static final int IE_SEED_OFFSET = 17; + + private static final String USE_NEW_SLC_SOA_METHOD_PROPERTY_KEY = "slc.use.new.soa"; + + private boolean runAutoOwnershipModel; + private boolean runTransponderModel; + private boolean runInternalExternalModel; + private boolean runParkingProvisionModel; + private boolean runCoordinatedDailyActivityPatternModel; + private boolean runIndividualMandatoryTourFrequencyModel; + private boolean runMandatoryTourModeChoiceModel; + private boolean runMandatoryTourDepartureTimeAndDurationModel; + private boolean runEscortModel; + private boolean runAtWorkSubTourFrequencyModel; + private boolean runAtWorkSubtourLocationChoiceModel; + private boolean runAtWorkSubtourModeChoiceModel; + private boolean runAtWorkSubtourDepartureTimeAndDurationModel; + private boolean runJointTourFrequencyModel; + private boolean runJointTourLocationChoiceModel; + private boolean runJointTourDepartureTimeAndDurationModel; + private boolean runJointTourModeChoiceModel; + private boolean runIndividualNonMandatoryTourFrequencyModel; + private boolean runIndividualNonMandatoryTourLocationChoiceModel; + private boolean runIndividualNonMandatoryTourModeChoiceModel; + private boolean runIndividualNonMandatoryTourDepartureTimeAndDurationModel; + private boolean runStopFrequencyModel; + private boolean runStopLocationModel; + + private String restartModelString; + + private HouseholdAutoOwnershipModel aoModel; + private TourVehicleTypeChoiceModel tvtcModel; + private TransponderChoiceModel tcModel; + private InternalExternalTripChoiceModel ieModel; + private ParkingProvisionModel ppModel; + private TelecommuteModel teModel; + private HouseholdCoordinatedDailyActivityPatternModel cdapModel; + private HouseholdIndividualMandatoryTourFrequencyModel imtfModel; + private HouseholdIndividualNonMandatoryTourFrequencyModel inmtfModel; + private SchoolEscortingModel escortModel; + private HouseholdAtWorkSubtourFrequencyModel awfModel; + private StopFrequencyModel stfModel; + private TourModeChoiceModel immcModel; + private HouseholdIndividualMandatoryTourDepartureAndDurationTime imtodModel; + private JointTourModels jtfModel; + private TourModeChoiceModel nmmcModel; + private NonMandatoryDestChoiceModel nmlcModel; + private NonMandatoryTourDepartureAndDurationTime nmtodModel; + private TourModeChoiceModel awmcModel; + private SubtourDestChoiceModel awlcModel; + private SubtourDepartureAndDurationTime awtodModel; + private IntermediateStopChoiceModels stlmcModel; + private MicromobilityChoiceModel mmModel; + + private long aoTime; + private long fpTime; + private long ieTime; + private long cdapTime; + private long escortTime; + private long imtfTime; + private long imtodTime; + private long imtmcTime; + private long jtfTime; + private long jtdcTime; + private long jtodTime; + private long jtmcTime; + private long inmtfTime; + private long inmtdcTime; + private long inmtdcSoaTime; + private long inmtodTime; + private long inmtmcTime; + private long awtfTime; + private long awtdcTime; + private long awtdcSoaTime; + private long awtodTime; + private long awtmcTime; + private long stfTime; + private long stdtmTime; + private long[] returnPartialTimes = new long[IntermediateStopChoiceModels.NUM_CPU_TIME_VALUES]; + + private int maxAlts; + private int modelIndex; + + private int globalSeed; + + private boolean useNewSlcSoaMethod; + + private double[][][] slcSizeProbs; + private double[][] slcTazSize; + private double[][] slcTazDistExpUtils; + + private double[] distanceToCordonsLogsums; + + private MgraDataManager mgraManager; + private TazDataManager tdm; + + public HouseholdChoiceModels(int modelIndex, String restartModelString, + HashMap propertyMap, ModelStructure modelStructure, + CtrampDmuFactoryIf dmuFactory, BuildAccessibilities aggAcc, + McLogsumsCalculator logsumHelper, MandatoryAccessibilitiesCalculator mandAcc, + double[] pctHighIncome, double[] pctMultipleAutos, double[] avgtts, + double[] transpDist, double[] pctDetour, double[][][] nonManSoaDistProbs, + double[][][] nonManSoaSizeProbs, double[][][] subTourSoaDistProbs, + double[][][] subTourSoaSizeProbs, double[] distanceToCordonsLogsums,AutoTazSkimsCalculator tazDistanceCalculator) + { + + this.modelIndex = modelIndex; + this.restartModelString = restartModelString; + + this.distanceToCordonsLogsums = distanceToCordonsLogsums; + + globalSeed = Integer.parseInt(propertyMap.get(GLOBAL_MODEL_SEED_PROPERTY)); + + mgraManager = MgraDataManager.getInstance(propertyMap); + tdm = TazDataManager.getInstance(propertyMap); + + runAutoOwnershipModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_AUTO_OWNERSHIP)); + runTransponderModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_TRANSPONDER_CHOICE)); + runInternalExternalModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP)); + runParkingProvisionModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_FREE_PARKING_AVAILABLE)); + runCoordinatedDailyActivityPatternModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN)); + runIndividualMandatoryTourFrequencyModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ)); + runMandatoryTourModeChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE)); + runMandatoryTourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR)); + runEscortModel = Boolean.parseBoolean(propertyMap.get(CtrampApplication.PROPERTIES_RUN_SCHOOL_ESCORT_MODEL)); + runAtWorkSubTourFrequencyModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ)); + runAtWorkSubtourLocationChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE)); + runAtWorkSubtourModeChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE)); + runAtWorkSubtourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR)); + runJointTourFrequencyModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_FREQ)); + runJointTourLocationChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_JOINT_LOCATION_CHOICE)); + runJointTourModeChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE)); + runJointTourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR)); + runIndividualNonMandatoryTourFrequencyModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ)); + runIndividualNonMandatoryTourLocationChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE)); + runIndividualNonMandatoryTourModeChoiceModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE)); + runIndividualNonMandatoryTourDepartureTimeAndDurationModel = Boolean + .parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR)); + runStopFrequencyModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_STOP_FREQUENCY)); + runStopLocationModel = Boolean.parseBoolean(propertyMap + .get(CtrampApplication.PROPERTIES_RUN_STOP_LOCATION)); + + boolean measureObjectSizes = false; + + try + { + useNewSlcSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, + USE_NEW_SLC_SOA_METHOD_PROPERTY_KEY); + + AccessibilitiesTable accTable = aggAcc.getAccessibilitiesTableObject(); + + // create the auto ownership choice model application object + if (runAutoOwnershipModel) + { + aoModel = new HouseholdAutoOwnershipModel(propertyMap, dmuFactory, accTable, + mandAcc); + tvtcModel = new TourVehicleTypeChoiceModel(propertyMap); + if ( measureObjectSizes ) logger.info ( "AO size: " + ObjectUtil.sizeOf( aoModel ) + ObjectUtil.sizeOf(tvtcModel)); + } + + if (runTransponderModel) + { + tcModel = new TransponderChoiceModel(propertyMap, dmuFactory, accTable, + pctHighIncome, pctMultipleAutos, avgtts, transpDist, pctDetour); + if (measureObjectSizes) + logger.info("TC size: " + ObjectUtil.sizeOf(tcModel)); + } + + if (runParkingProvisionModel) + { + ppModel = new ParkingProvisionModel(propertyMap, dmuFactory); + teModel = new TelecommuteModel(propertyMap, dmuFactory); + if (measureObjectSizes) { + logger.info("PP size: " + ObjectUtil.sizeOf(ppModel)); + logger.info("TE size: " + ObjectUtil.sizeOf(teModel)); + } + } + + if (runInternalExternalModel) + { + ieModel = new InternalExternalTripChoiceModel(propertyMap, modelStructure,dmuFactory); + if (measureObjectSizes) + logger.info("IE size: " + ObjectUtil.sizeOf(ieModel)); + } + + if (runCoordinatedDailyActivityPatternModel) + { + cdapModel = new HouseholdCoordinatedDailyActivityPatternModel(propertyMap, + modelStructure, dmuFactory, accTable); + if (measureObjectSizes) + logger.info("CDAP size: " + ObjectUtil.sizeOf(cdapModel)); + } + + if (runIndividualMandatoryTourFrequencyModel) + { + imtfModel = new HouseholdIndividualMandatoryTourFrequencyModel(propertyMap, + modelStructure, dmuFactory, accTable, mandAcc); + if (measureObjectSizes) + logger.info("IMTF size: " + ObjectUtil.sizeOf(imtfModel)); + } + + if (runMandatoryTourDepartureTimeAndDurationModel || runMandatoryTourModeChoiceModel) + { + immcModel = new TourModeChoiceModel(propertyMap, modelStructure, + TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + if (measureObjectSizes) + logger.info("IMMC size: " + ObjectUtil.sizeOf(immcModel)); + + imtodModel = new HouseholdIndividualMandatoryTourDepartureAndDurationTime( + propertyMap, modelStructure, aggAcc.getWorkSegmentNameList(), dmuFactory, + immcModel); + if (measureObjectSizes) + logger.info("IMTOD size: " + ObjectUtil.sizeOf(imtodModel)); + } + + if(runEscortModel){ + escortModel = new SchoolEscortingModel(propertyMap,mgraManager,tazDistanceCalculator); + if ( measureObjectSizes ) logger.info ( "SEM size: " + ObjectUtil.sizeOf( escortModel ) ); + + } + + if (runJointTourFrequencyModel) + { + jtfModel = new JointTourModels(propertyMap, accTable, modelStructure, dmuFactory); + if (measureObjectSizes) + logger.info("JTF size: " + ObjectUtil.sizeOf(jtfModel)); + } + + if (runIndividualNonMandatoryTourFrequencyModel) + { + inmtfModel = new HouseholdIndividualNonMandatoryTourFrequencyModel(propertyMap, + dmuFactory, accTable, mandAcc); + if (measureObjectSizes) + logger.info("INMTF size: " + ObjectUtil.sizeOf(inmtfModel)); + } + + if (runIndividualNonMandatoryTourLocationChoiceModel || runJointTourLocationChoiceModel + || runIndividualNonMandatoryTourDepartureTimeAndDurationModel + || runJointTourDepartureTimeAndDurationModel + || runIndividualNonMandatoryTourModeChoiceModel || runJointTourModeChoiceModel) + { + nmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + TourModeChoiceModel.NON_MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); + if (measureObjectSizes) + logger.info("INMMC size: " + ObjectUtil.sizeOf(nmmcModel)); + } + + if (runIndividualNonMandatoryTourLocationChoiceModel || runJointTourLocationChoiceModel) + { + nmlcModel = new NonMandatoryDestChoiceModel(propertyMap, modelStructure, aggAcc, + dmuFactory, nmmcModel); + nmlcModel.setNonMandatorySoaProbs(nonManSoaDistProbs, nonManSoaSizeProbs); + if (measureObjectSizes) + logger.info("INMLC size: " + ObjectUtil.sizeOf(nmlcModel)); + } + + if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel + || runJointTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) + { + nmtodModel = new NonMandatoryTourDepartureAndDurationTime(propertyMap, + modelStructure, dmuFactory, nmmcModel); + if (measureObjectSizes) + logger.info("INMTOD size: " + ObjectUtil.sizeOf(nmtodModel)); + } + + if (runAtWorkSubTourFrequencyModel) + { + awfModel = new HouseholdAtWorkSubtourFrequencyModel(propertyMap, modelStructure, + dmuFactory); + if (measureObjectSizes) + logger.info("AWTF size: " + ObjectUtil.sizeOf(awfModel)); + } + + if (runAtWorkSubtourLocationChoiceModel + || runAtWorkSubtourDepartureTimeAndDurationModel + || runAtWorkSubtourModeChoiceModel) + { + awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + TourModeChoiceModel.AT_WORK_SUBTOUR_MODEL_INDICATOR, dmuFactory, + logsumHelper); + if (measureObjectSizes) + logger.info("AWMC size: " + ObjectUtil.sizeOf(awmcModel)); + } + + if (runAtWorkSubtourLocationChoiceModel) + { + awlcModel = new SubtourDestChoiceModel(propertyMap, modelStructure, aggAcc, + dmuFactory, awmcModel); + awlcModel.setNonMandatorySoaProbs(subTourSoaDistProbs, subTourSoaSizeProbs); + if (measureObjectSizes) + logger.info("AWLC size: " + ObjectUtil.sizeOf(awlcModel)); + } + + if (runAtWorkSubtourDepartureTimeAndDurationModel) + { + awtodModel = new SubtourDepartureAndDurationTime(propertyMap, modelStructure, + dmuFactory, awmcModel); + if (measureObjectSizes) + logger.info("AWTOD size: " + ObjectUtil.sizeOf(awtodModel)); + } + + if (runStopFrequencyModel) + { + stfModel = new StopFrequencyModel(propertyMap, dmuFactory, modelStructure, accTable); + if (measureObjectSizes) + logger.info("STF size: " + ObjectUtil.sizeOf(stfModel)); + } + + if (runStopLocationModel) + { + stlmcModel = new IntermediateStopChoiceModels(propertyMap, modelStructure, + dmuFactory, logsumHelper); + + mmModel = new MicromobilityChoiceModel(propertyMap,modelStructure,dmuFactory); + + // if the slcTazDistProbs are not null, they have been already + // computed, and it is + // not necessary for the thread creating this + // HouseholdChoiceModels object to + // compute them also. If slcTazDistProbs is null, compute them. + if (useNewSlcSoaMethod && slcSizeProbs == null) + { + + // compute the array of cumulative taz distance based SOA + // probabilities for each origin taz. + DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu = dmuFactory + .getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + DestChoiceTwoStageSoaProbabilitiesCalculator slcSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, + dmuFactory, + IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY, + IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_MODEL_PAGE, + IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_DATA_PAGE); + + computeSlcSoaProbabilities(slcSoaDistProbsObject, locChoiceDistSoaDmu, + stlmcModel.getSizeSegmentNameIndexMap(), + stlmcModel.getSizeSegmentArray()); + + stlmcModel.setupSlcDistanceBaseSoaModel(propertyMap, slcTazDistExpUtils, + slcSizeProbs, slcTazSize); + } + + if (measureObjectSizes) + logger.info("SLMT size: " + ObjectUtil.sizeOf(stlmcModel)); + } + + } catch (RuntimeException e) + { + + String lastModel = ""; + if (runAutoOwnershipModel && aoModel != null) lastModel += " ao"; + + if (runParkingProvisionModel && ppModel != null) lastModel += " fp"; + + if (runInternalExternalModel && ieModel != null) lastModel += " ie"; + + if (runCoordinatedDailyActivityPatternModel && cdapModel != null) lastModel += " cdap"; + + if (runIndividualMandatoryTourFrequencyModel && imtfModel != null) + lastModel += " imtf"; + + if (runMandatoryTourModeChoiceModel && immcModel != null) lastModel += " immc"; + + if (runMandatoryTourDepartureTimeAndDurationModel && imtodModel != null) + lastModel += " imtod"; + + if (runJointTourFrequencyModel && jtfModel != null) lastModel += " jtf"; + + if (runJointTourModeChoiceModel && nmmcModel != null) lastModel += " jmc"; + + if (runJointTourLocationChoiceModel && nmlcModel != null) lastModel += " jlc"; + + if (runJointTourDepartureTimeAndDurationModel && nmtodModel != null) + lastModel += " jtod"; + + if (runIndividualNonMandatoryTourFrequencyModel && inmtfModel != null) + lastModel += " inmtf"; + + if (runIndividualNonMandatoryTourModeChoiceModel && nmmcModel != null) + lastModel += " inmmc"; + + if (runIndividualNonMandatoryTourLocationChoiceModel && nmlcModel != null) + lastModel += " inmlc"; + + if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel && nmtodModel != null) + lastModel += " inmtod"; + + if (runAtWorkSubTourFrequencyModel && awfModel != null) lastModel += " awf"; + + if (runAtWorkSubtourModeChoiceModel && awmcModel != null) lastModel += " awmc"; + + if (runAtWorkSubtourLocationChoiceModel && awlcModel != null) lastModel += " awlc"; + + if (runAtWorkSubtourDepartureTimeAndDurationModel && awtodModel != null) + lastModel += " awtod"; + + if (runStopFrequencyModel && stfModel != null) lastModel += " stf"; + + if (runStopLocationModel && stlmcModel != null) lastModel += " stlmc"; + + logger.error("RuntimeException setting up HouseholdChoiceModels."); + logger.error("Models setup = " + lastModel); + logger.error("", e); + + throw new RuntimeException(); + } + + } + + public void runModels(Household hhObject) + { + + // check to see if restartModel was set and reset random number sequence + // appropriately if so. + checkRestartModel(hhObject); + + if (runAutoOwnershipModel) aoModel.applyModel(hhObject, false); + + if (runTransponderModel) tcModel.applyModel(hhObject); + + if (runParkingProvisionModel) { + ppModel.applyModel(hhObject); + teModel.applyModel(hhObject); + } + + if (runInternalExternalModel) ieModel.applyModel(hhObject, distanceToCordonsLogsums); + + if (runCoordinatedDailyActivityPatternModel) cdapModel.applyModel(hhObject); + + if (runIndividualMandatoryTourFrequencyModel) { + imtfModel.applyModel(hhObject); + tvtcModel.applyModelToMandatoryTours(hhObject); + } + + if (runMandatoryTourDepartureTimeAndDurationModel||runMandatoryTourModeChoiceModel) + imtodModel.applyModel(hhObject, runMandatoryTourDepartureTimeAndDurationModel,runMandatoryTourModeChoiceModel); + + if(runEscortModel){ + try { + escortModel.applyModel(hhObject); + } catch (Exception e) { + logger.fatal("Error Attempting to run escort model for household "+hhObject.getHhId()); + throw new RuntimeException(e); + } + } + + if (runJointTourFrequencyModel) { + jtfModel.applyModel(hhObject); + tvtcModel.applyModelToJointTours(hhObject); + } + + if (runJointTourLocationChoiceModel) nmlcModel.applyJointModel(hhObject); + + if (runJointTourDepartureTimeAndDurationModel) + nmtodModel.applyJointModel(hhObject, runJointTourDepartureTimeAndDurationModel, runJointTourModeChoiceModel); + + if (runIndividualNonMandatoryTourFrequencyModel) { + inmtfModel.applyModel(hhObject); + tvtcModel.applyModelToNonMandatoryTours(hhObject); + } + + if (runIndividualNonMandatoryTourLocationChoiceModel) nmlcModel.applyIndivModel(hhObject); + + if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) + nmtodModel.applyIndivModel(hhObject, runIndividualNonMandatoryTourDepartureTimeAndDurationModel, runIndividualNonMandatoryTourModeChoiceModel); + + if (runAtWorkSubTourFrequencyModel) { + awfModel.applyModel(hhObject); + tvtcModel.applyModelToAtWorkSubTours(hhObject); + } + + if (runAtWorkSubtourLocationChoiceModel) awlcModel.applyModel(hhObject); + + if (runAtWorkSubtourDepartureTimeAndDurationModel) + awtodModel.applyModel(hhObject, runAtWorkSubtourModeChoiceModel); + + if (runStopFrequencyModel) stfModel.applyModel(hhObject); + + if (runStopLocationModel) { + stlmcModel.applyModel(hhObject, false); + mmModel.applyModel(hhObject); + } + + } + + public void runModelsWithTiming(Household hhObject) + { + + // check to see if restartModel was set and reset random number sequence + // appropriately if so. + checkRestartModel(hhObject); + + if (runAutoOwnershipModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + AO_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + aoModel.applyModel(hhObject, false); + aoTime += (System.nanoTime() - check); + } + + if (runTransponderModel) + { + // long hhSeed = globalSeed + hhObject.getHhId() + TP_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + tcModel.applyModel(hhObject); + } + + if (runParkingProvisionModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + PP_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + ppModel.applyModel(hhObject); + teModel.applyModel(hhObject); + fpTime += (System.nanoTime() - check); + } + + if (runInternalExternalModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + PP_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + ieModel.applyModel(hhObject, distanceToCordonsLogsums); + ieTime += (System.nanoTime() - check); + } + + if (runCoordinatedDailyActivityPatternModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + CDAP_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + cdapModel.applyModel(hhObject); + cdapTime += (System.nanoTime() - check); + } + + if (runIndividualMandatoryTourFrequencyModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + IMTF_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + imtfModel.applyModel(hhObject); + tvtcModel.applyModelToMandatoryTours(hhObject); + imtfTime += (System.nanoTime() - check); + } + + if (runMandatoryTourDepartureTimeAndDurationModel||runMandatoryTourModeChoiceModel); + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + + // IMTOD_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + imtodModel.applyModel(hhObject, runMandatoryTourDepartureTimeAndDurationModel,runMandatoryTourModeChoiceModel); + long mcTime = imtodModel.getModeChoiceTime(); + imtodTime += (System.nanoTime() - check - mcTime); + imtmcTime += mcTime; + } + if(runEscortModel){ + long check = System.nanoTime(); + try { + escortModel.applyModel(hhObject); + } catch (Exception e) { + logger.fatal("Error Attempting to run escort model for household "+hhObject.getHhId()); + e.printStackTrace(); + } + escortTime += ( System.nanoTime() - check ); + } + + + if (runJointTourFrequencyModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + JTF_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + jtfModel.applyModel(hhObject); + tvtcModel.applyModelToJointTours(hhObject); + jtfTime += (System.nanoTime() - check); + } + + if (runJointTourLocationChoiceModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + JTDC_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + nmlcModel.applyJointModel(hhObject); + jtdcTime += (System.nanoTime() - check); + } + + if (runJointTourDepartureTimeAndDurationModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + JTOD_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + nmtodModel.applyJointModel(hhObject, runJointTourDepartureTimeAndDurationModel,runJointTourModeChoiceModel); + long mcTime = nmtodModel.getJointModeChoiceTime(); + jtodTime += (System.nanoTime() - check - mcTime); + jtmcTime += mcTime; + } + + if (runIndividualNonMandatoryTourFrequencyModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + + // INMTF_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + inmtfModel.applyModel(hhObject); + tvtcModel.applyModelToNonMandatoryTours(hhObject); + inmtfTime += (System.nanoTime() - check); + } + + if (runIndividualNonMandatoryTourLocationChoiceModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + + // INMDC_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + nmlcModel.resetSoaRunTime(); + nmlcModel.applyIndivModel(hhObject); + inmtdcSoaTime += nmlcModel.getSoaRunTime(); + inmtdcTime += (System.nanoTime() - check); + } + + if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + + // INMTOD_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + nmtodModel.applyIndivModel(hhObject, runIndividualNonMandatoryTourDepartureTimeAndDurationModel,runIndividualNonMandatoryTourModeChoiceModel); + long mcTime = nmtodModel.getIndivModeChoiceTime(); + inmtodTime += (System.nanoTime() - check - mcTime); + inmtmcTime += mcTime; + } + + if (runAtWorkSubTourFrequencyModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + AWTF_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + awfModel.applyModel(hhObject); + tvtcModel.applyModelToAtWorkSubTours(hhObject); + awtfTime += (System.nanoTime() - check); + } + + if (runAtWorkSubtourLocationChoiceModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + AWDC_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + awlcModel.applyModel(hhObject); + awtdcSoaTime += awlcModel.getSoaRunTime(); + awtdcTime += (System.nanoTime() - check); + } + + if (runAtWorkSubtourDepartureTimeAndDurationModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + + // AWTOD_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + awtodModel.applyModel(hhObject, runAtWorkSubtourModeChoiceModel); + long mcTime = awtodModel.getModeChoiceTime(); + awtodTime += (System.nanoTime() - check - mcTime); + awtmcTime += mcTime; + } + + if (runStopFrequencyModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + STF_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + stfModel.applyModel(hhObject); + stfTime += (System.nanoTime() - check); + } + + if (runStopLocationModel) + { + long check = System.nanoTime(); + // long hhSeed = globalSeed + hhObject.getHhId() + SLC_SEED_OFFSET; + // hhObject.getHhRandom().setSeed( hhSeed ); + stlmcModel.applyModel(hhObject, true); + stdtmTime += (System.nanoTime() - check); + + long[] partials = stlmcModel.getStopTimes(); + for (int i = 0; i < returnPartialTimes.length; i++) + returnPartialTimes[i] += partials[i]; + + if (stlmcModel.getMaxAltsInSample() > maxAlts) + maxAlts = stlmcModel.getMaxAltsInSample(); + + mmModel.applyModel(hhObject); + } + + } + + private void checkRestartModel(Household hhObject) + { + + // none, ao, cdap, imtf, imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, + // inmtl, inmtod, stf, stl + // version 1.0.8.22 - changed model restart options - possible values + // for + // restart are now: none, uwsl, ao, imtf, jtf, inmtf, stf + + // if restartModel was specified, reset the random number sequence + // based on the cumulative count of random numbers drawn by the + // component + // preceding the one specified. + if (restartModelString.equalsIgnoreCase("") || restartModelString.equalsIgnoreCase("none")) return; + else if (restartModelString.equalsIgnoreCase("ao")) + { + hhObject.initializeForAoRestart(); + } else if (restartModelString.equalsIgnoreCase("imtf")) + { + hhObject.initializeForImtfRestart(); + } else if (restartModelString.equalsIgnoreCase("jtf")) + { + hhObject.initializeForJtfRestart(); + } else if (restartModelString.equalsIgnoreCase("inmtf")) + { + hhObject.initializeForInmtfRestart(); + } else if (restartModelString.equalsIgnoreCase("awf")) + { + hhObject.initializeForAwfRestart(); + } else if (restartModelString.equalsIgnoreCase("stf")) + { + hhObject.initializeForStfRestart(); + } + + } + + public int getModelIndex() + { + return modelIndex; + } + + public void zeroTimes() + { + aoTime = 0; + fpTime = 0; + ieTime = 0; + cdapTime = 0; + imtfTime = 0; + imtodTime = 0; + imtmcTime = 0; + jtfTime = 0; + jtdcTime = 0; + jtodTime = 0; + jtmcTime = 0; + inmtfTime = 0; + inmtdcTime = 0; + inmtdcSoaTime = 0; + inmtodTime = 0; + inmtmcTime = 0; + awtfTime = 0; + awtdcTime = 0; + awtdcSoaTime = 0; + awtodTime = 0; + awtmcTime = 0; + stfTime = 0; + stdtmTime = 0; + + Arrays.fill(returnPartialTimes, 0); + } + + public long[] getPartialStopTimes() + { + return returnPartialTimes; + } + + public long[] getTimes() + { + long[] returnTimes = new long[23]; + returnTimes[0] = aoTime; + returnTimes[1] = fpTime; + returnTimes[2] = ieTime; + returnTimes[3] = cdapTime; + returnTimes[4] = imtfTime; + returnTimes[5] = imtodTime; + returnTimes[6] = imtmcTime; + returnTimes[7] = jtfTime; + returnTimes[8] = jtdcTime; + returnTimes[9] = jtodTime; + returnTimes[10] = jtmcTime; + returnTimes[11] = inmtfTime; + returnTimes[12] = inmtdcSoaTime; + returnTimes[13] = inmtdcTime; + returnTimes[14] = inmtodTime; + returnTimes[15] = inmtmcTime; + returnTimes[16] = awtfTime; + returnTimes[17] = awtdcSoaTime; + returnTimes[18] = awtdcTime; + returnTimes[19] = awtodTime; + returnTimes[20] = awtmcTime; + returnTimes[21] = stfTime; + returnTimes[22] = stdtmTime; + return returnTimes; + } + + public int getMaxAlts() + { + return maxAlts; + } + + private void computeSlcSoaProbabilities( + DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, + DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu, + HashMap segmentNameIndexMap, double[][] dcSizeArray) + { + + // compute the exponentiated distance utilities that all segments of + // this tour purpose will share + slcTazDistExpUtils = computeTazDistanceExponentiatedUtilities(locChoiceSoaDistProbsObject, + locChoiceDistSoaDmu); + + slcTazSize = new double[dcSizeArray.length][]; + slcSizeProbs = new double[dcSizeArray.length][][]; + + // compute an array of SOA size probabilities for each segment + for (String segmentName : segmentNameIndexMap.keySet()) + { + + // compute the TAZ size values from the mgra values and the + // correspondence between mgras and tazs. + int segmentIndex = segmentNameIndexMap.get(segmentName); + slcTazSize[segmentIndex] = computeTazSize(dcSizeArray[segmentIndex]); + + slcSizeProbs[segmentIndex] = computeSizeSegmentProbabilities(dcSizeArray[segmentIndex], + slcTazSize[segmentIndex]); + + } + + } + + private double[][] computeSizeSegmentProbabilities(double[] size, double[] totalTazSize) + { + + int maxTaz = tdm.getMaxTaz(); + + // this is a 0-based array of cumulative probabilities + double[][] sizeProbs = new double[maxTaz][]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + + if (mgraArray == null) + { + sizeProbs[taz - 1] = new double[0]; + } else + { + + if (totalTazSize[taz] > 0) + { + sizeProbs[taz - 1] = new double[mgraArray.length]; + for (int i = 0; i < mgraArray.length; i++) + { + double mgraSize = size[mgraArray[i]]; + if (mgraSize > 0) mgraSize += 1; + sizeProbs[taz - 1][i] = mgraSize / totalTazSize[taz]; + } + } else + { + sizeProbs[taz - 1] = new double[0]; + } + } + + } + + return sizeProbs; + + } + + private double[][] computeTazDistanceExponentiatedUtilities( + DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, + DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu) + { + + // compute the TAZ x TAZ exponentiated utilities array for sample + // selection utilities. + double[][] tazDistExpUtils = locChoiceSoaDistProbsObject + .computeDistanceUtilities(locChoiceDistSoaDmu); + for (int i = 0; i < tazDistExpUtils.length; i++) + for (int j = 0; j < tazDistExpUtils[i].length; j++) + { + if (tazDistExpUtils[i][j] < -500) tazDistExpUtils[i][j] = 0; + else tazDistExpUtils[i][j] = Math.exp(tazDistExpUtils[i][j]); + } + + return tazDistExpUtils; + + } + + private double[] computeTazSize(double[] size) + { + + int maxTaz = tdm.getMaxTaz(); + + double[] tazSize = new double[maxTaz + 1]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray != null) + { + for (int mgra : mgraArray) + { + tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); + } + } + + } + + return tazSize; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java new file mode 100644 index 0000000..ef4223d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java @@ -0,0 +1,682 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Date; +import java.util.HashMap; +import java.util.LinkedList; +import java.util.StringTokenizer; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import org.sandag.abm.accessibilities.NonTransitUtilities; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MathUtil; + +public final class HouseholdChoiceModelsManager + implements Serializable +{ + + private static transient Logger logger = Logger.getLogger(HouseholdChoiceModelsManager.class); + + private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; + + private static final String TAZ_FIELD_NAME = "TAZ"; + private static final String TP_CHOICE_AVG_TTS_FILE = "tc.choice.avgtts.file"; + private static final String AVGTTS_COLUMN_NAME = "AVGTTS"; + private static final String TRANSP_DIST_COLUMN_NAME = "DIST"; + private static final String PCT_DETOUR_COLUMN_NAME = "PCTDETOUR"; + + private static final String IE_EXTERNAL_TAZS_KEY = "external.tazs"; + private static final String IE_DISTANCE_LOGSUM_COEFF_KEY = "ie.logsum.distance.coeff"; + + private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE = "nonSchool.soa.uec.file"; + private static String PROPERTIES_ESCORT_DC_SOA_UEC_MODEL_PAGE = "escort.soa.uec.model"; + private static String PROPERTIES_ESCORT_DC_SOA_UEC_DATA_PAGE = "escort.soa.uec.data"; + private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_MODEL_PAGE = "other.nonman.soa.uec.model"; + private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_DATA_PAGE = "other.nonman.soa.uec.data"; + private static String PROPERTIES_ATWORK_DC_SOA_UEC_MODEL_PAGE = "atwork.soa.uec.model"; + private static String PROPERTIES_ATWORK_DC_SOA_UEC_DATA_PAGE = "atwork.soa.uec.data"; + + private static HouseholdChoiceModelsManager objInstance = null; + + private LinkedList modelQueue = null; + + private HashMap propertyMap; + private String restartModelString; + private ModelStructure modelStructure; + private CtrampDmuFactoryIf dmuFactory; + + private MgraDataManager mgraManager; + private TazDataManager tdm; + + private int maxMgra; + private int maxTaz; + + private BuildAccessibilities aggAcc; + + private int completedHouseholds; + private int modelIndex; + + // store taz-taz exponentiated utilities (period, from taz, to taz) + private double[][][] sovExpUtilities; + private double[][][] hovExpUtilities; + private double[][][] nMotorExpUtilities; + private double[][][] maasExpUtilities; + + private double[] pctHighIncome; + private double[] pctMultipleAutos; + + private double[] avgtts; + private double[] transpDist; + private double[] pctDetour; + + private double[][][] nonMandatorySizeProbs; + private double[][][] nonMandatoryTazDistProbs; + private double[][][] subTourSizeProbs; + private double[][][] subTourTazDistProbs; + + private AutoTazSkimsCalculator tazDistanceCalculator; + + private boolean useNewSoaMethod; + private boolean logResults=false; + + private HouseholdChoiceModelsManager() + { + } + + public static synchronized HouseholdChoiceModelsManager getInstance() + { + // logger.info( + // "beginning of HouseholdChoiceModelsManager() - objInstance address = " + // + objInstance ); + if (objInstance == null) + { + objInstance = new HouseholdChoiceModelsManager(); + // logger.info( + // "after new HouseholdChoiceModelsManager() - objInstance address = " + // + objInstance ); + return objInstance; + } else + { + // logger.info( + // "returning current HouseholdChoiceModelsManager() - objInstance address = " + // + objInstance ); + return objInstance; + } + } + + // the task instances should call needToInitialize() first, then this method + // if necessary. + public synchronized void managerSetup(MatrixDataServerIf ms, + HouseholdDataManagerIf hhDataManager, HashMap propertyMap, + String restartModelString, ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) + { + + if (modelQueue != null) return; + + // get the DestChoiceModelManager instance and clear the objects that + // hold large memory references + DestChoiceModelManager.getInstance().clearDcModels(); + + modelIndex = 0; + completedHouseholds = 0; + + this.propertyMap = propertyMap; + this.restartModelString = restartModelString; + this.modelStructure = modelStructure; + this.dmuFactory = dmuFactory; + + logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + + mgraManager = MgraDataManager.getInstance(propertyMap); + maxMgra = mgraManager.getMaxMgra(); + + tdm = TazDataManager.getInstance(propertyMap); + maxTaz = tdm.getMaxTaz(); + + pctHighIncome = hhDataManager.getPercentHhsIncome100Kplus(); + pctMultipleAutos = hhDataManager.getPercentHhsMultipleAutos(); + readTpChoiceAvgTtsFile(); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + aggAcc = BuildAccessibilities.getInstance(); + if (!aggAcc.getAccessibilitiesAreBuilt()) + { + logger.info("creating Accessibilities Object for Household Choice Models."); + + aggAcc.setupBuildAccessibilities(propertyMap, false); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + + // assume that if the filename exists, at was created previously, + // either in another model run, or by the main client + // if the filename doesn't exist, then calculate the accessibilities + String projectDirectory = propertyMap + .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + boolean accFileReadFlag = Util.getBooleanValueFromPropertyMap(propertyMap, + CtrampApplication.READ_ACCESSIBILITIES); + + if (accFileReadFlag && (new File(accFileName)).canRead()) + { + + logger.info("filling Accessibilities Object in HouseholdChoiceModelManager by reading file: " + + accFileName + "."); + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + logger.info("filling Accessibilities Object HouseholdChoiceModelManager by calculating them."); + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + + } + + useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, + USE_NEW_SOA_METHOD_PROPERTY_KEY); + + if (useNewSoaMethod) + { + // compute the arrays of cumulative probabilities based on mgra size + // for mgras within each origin taz. + logger.info("pre-computing non-mandatory purpose SOA Distance and Size probabilities."); + computeNonMandatorySegmentSizeArrays(dmuFactory); + + logger.info("pre-computing at-work sub-tour purpose SOA Distance and Size probabilities."); + computeSubtourSegmentSizeArrays(modelStructure, dmuFactory); + } + + tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + // the first thread to reach this method initializes the modelQueue used + // to + // recycle hhChoiceModels objects. + modelQueue = new LinkedList(); + + mgraManager = MgraDataManager.getInstance(propertyMap); + + } + + /** + * @return DestChoiceModel object created if none is available from the + * queue. + * + */ + public synchronized HouseholdChoiceModels getHouseholdChoiceModelsObject(int taskIndex) + { + + String message = ""; + HouseholdChoiceModels hhChoiceModels = null; + + if (modelQueue.isEmpty()) + { + + NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, + hovExpUtilities, nMotorExpUtilities, maasExpUtilities); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( + propertyMap, ntUtilities, aggAcc.getExpConstants(), + logsumHelper.getBestTransitPathCalculator()); + + // calculate array of distanceToExternalCordon logsums by taz for + // use by internal-external model + double[] distanceToCordonsLogsums = computeTazDistanceToExternalCordonLogsums(); + + // create choice model object + hhChoiceModels = new HouseholdChoiceModels(++modelIndex, restartModelString, + propertyMap, modelStructure, dmuFactory, aggAcc, logsumHelper, mandAcc, + pctHighIncome, pctMultipleAutos, avgtts, transpDist, pctDetour, + nonMandatoryTazDistProbs, nonMandatorySizeProbs, subTourTazDistProbs, + subTourSizeProbs, distanceToCordonsLogsums, tazDistanceCalculator); + if(logResults){ + message = String.format("created hhChoiceModels=%d, task=%d, thread=%s.", modelIndex, + taskIndex, Thread.currentThread().getName()); + logger.info(message); + logger.info(""); + } + + } else + { + hhChoiceModels = modelQueue.remove(); + if(logResults){ + message = String.format("removed hhChoiceModels=%d from queue, task=%d, thread=%s.", + hhChoiceModels.getModelIndex(), taskIndex, Thread.currentThread().getName()); + logger.info(message); + logger.info(""); + } + } + + return hhChoiceModels; + + } + + /** + * return the HouseholdChoiceModels object to the manager's queue so that it + * may be used by another thread without it having to create one. + * + * @param hhModels + */ + public void returnHouseholdChoiceModelsObject(HouseholdChoiceModels hhModels, int startIndex, + int endIndex) + { + modelQueue.add(hhModels); + completedHouseholds += (endIndex - startIndex + 1); + if(logResults){ + logger.info("returned hhChoiceModels=" + hhModels.getModelIndex() + " to queue: thread=" + + Thread.currentThread().getName() + ", completedHouseholds=" + completedHouseholds + + "."); + } + } + + public synchronized void clearHhModels() + { + + if (modelQueue == null) return; + + logger.info(String.format("%s: clearing household choice models modelQueue, thread=%s.", + new Date(), Thread.currentThread().getName())); + while (!modelQueue.isEmpty()) + modelQueue.remove(); + + modelIndex = 0; + completedHouseholds = 0; + + modelQueue = null; + + } + + private void readTpChoiceAvgTtsFile() + { + + // construct input household file name from properties file values + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + + String inputFileName = propertyMap.get(TP_CHOICE_AVG_TTS_FILE); + String fileName = projectDirectory + inputFileName; + + TableDataSet table; + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + table = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading tp choice avgtts data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + int[] tazField = table.getColumnAsInt(TAZ_FIELD_NAME); + double[] avgttsField = table.getColumnAsDouble(AVGTTS_COLUMN_NAME); + double[] transpDistField = table.getColumnAsDouble(TRANSP_DIST_COLUMN_NAME); + double[] pctDetourField = table.getColumnAsDouble(PCT_DETOUR_COLUMN_NAME); + + avgtts = new double[tdm.getMaxTaz() + 1]; + transpDist = new double[tdm.getMaxTaz() + 1]; + pctDetour = new double[tdm.getMaxTaz() + 1]; + + // loop over the number of mgra records in the TableDataSet. + for (int k = 0; k < tdm.getMaxTaz(); k++) + { + + // get the mgra value for TableDataSet row k from the mgra field. + int taz = tazField[k]; + + avgtts[taz] = avgttsField[k]; + transpDist[taz] = transpDistField[k]; + pctDetour[taz] = pctDetourField[k]; + + } + + } + + private void computeNonMandatorySegmentSizeArrays(CtrampDmuFactoryIf dmuFactory) + { + + // compute the array of cumulative taz distance based SOA probabilities + // for each origin taz. + DestChoiceTwoStageSoaTazDistanceUtilityDMU dcDistSoaDmu = dmuFactory + .getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + // the size term array in aggAcc gives mgra*purpose - need an array of + // all mgras for one purpose + double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); + + String[] tourPurposeNames = {ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME}; + + int[] sizeSheetIndices = {BuildAccessibilities.ESCORT_INDEX, + BuildAccessibilities.SHOP_INDEX, BuildAccessibilities.OTH_MAINT_INDEX, + BuildAccessibilities.EATOUT_INDEX, BuildAccessibilities.VISIT_INDEX, + BuildAccessibilities.OTH_DISCR_INDEX}; + + HashMap nonMandatorySegmentNameIndexMap = new HashMap(); + HashMap nonMandatorySizeSegmentNameIndexMap = new HashMap(); + for (int k = 0; k < tourPurposeNames.length; k++) + { + nonMandatorySegmentNameIndexMap.put(tourPurposeNames[k], k); + nonMandatorySizeSegmentNameIndexMap.put(tourPurposeNames[k], sizeSheetIndices[k]); + } + + double[][] dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; + for (int i = 0; i < aggAccDcSizeArray.length; i++) + { + for (int m : nonMandatorySegmentNameIndexMap.values()) + { + int s = sizeSheetIndices[m]; + dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; + } + } + + // compute the arrays of cumulative probabilities based on mgra size for + // mgras within each origin taz. + nonMandatorySizeProbs = new double[tourPurposeNames.length][][]; + nonMandatoryTazDistProbs = new double[tourPurposeNames.length][][]; + + DestChoiceTwoStageSoaProbabilitiesCalculator nonManSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, + PROPERTIES_NON_MANDATORY_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_NON_MANDATORY_DC_SOA_UEC_DATA_PAGE); + + for (String tourPurpose : tourPurposeNames) + { + + int purposeSizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurpose); + + // compute the TAZ size values from the mgra values and the + // correspondence between mgras and tazs. + if (tourPurpose.equalsIgnoreCase(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) + { + + double[] mgraData = new double[maxMgra + 1]; + double[] tazData = null; + + // aggregate TAZ grade school enrollment and set array in DMU + for (int i = 1; i <= maxMgra; i++) + mgraData[i] = aggAcc.getMgraGradeSchoolEnrollment(i); + tazData = computeTazSize(mgraData); + dcDistSoaDmu.setTazGsEnrollment(tazData); + + // aggregate TAZ high school enrollment and set array in DMU + for (int i = 1; i <= maxMgra; i++) + mgraData[i] = aggAcc.getMgraHighSchoolEnrollment(i); + tazData = computeTazSize(mgraData); + dcDistSoaDmu.setTazHsEnrollment(tazData); + + // aggregate TAZ households and set array in DMU + for (int i = 1; i <= maxMgra; i++) + mgraData[i] = aggAcc.getMgraHouseholds(i); + tazData = computeTazSize(mgraData); + dcDistSoaDmu.setNumHhs(tazData); + + DestChoiceTwoStageSoaProbabilitiesCalculator escortSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, + PROPERTIES_ESCORT_DC_SOA_UEC_MODEL_PAGE, + PROPERTIES_ESCORT_DC_SOA_UEC_DATA_PAGE); + + logger.info(" " + tourPurpose + " probabilities"); + nonMandatoryTazDistProbs[purposeSizeIndex] = escortSoaDistProbsObject + .computeDistanceProbabilities(dcDistSoaDmu); + + } else + { + + // aggregate TAZ size for the non-mandatoy purpose and set array + // in DMU + double[] tazSize = computeTazSize(dcSizeArray[purposeSizeIndex]); + dcDistSoaDmu.setDestChoiceTazSize(tazSize); + + logger.info(" " + tourPurpose + " probabilities"); + nonMandatoryTazDistProbs[purposeSizeIndex] = nonManSoaDistProbsObject + .computeDistanceProbabilities(dcDistSoaDmu); + + } + + nonMandatorySizeProbs[purposeSizeIndex] = computeSizeSegmentProbabilities(dcSizeArray[purposeSizeIndex]); + + } + + } + + private void computeSubtourSegmentSizeArrays(ModelStructure modelStructure, + CtrampDmuFactoryIf dmuFactory) + { + + // compute the array of cumulative taz distance based SOA probabilities + // for each origin taz. + DestChoiceTwoStageSoaTazDistanceUtilityDMU dcDistSoaDmu = dmuFactory + .getDestChoiceSoaTwoStageTazDistUtilityDMU(); + + // the size term array in aggAcc gives mgra*purpose - need an array of + // all mgras for one purpose + double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); + + String[] tourPurposeNames = {modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME, + modelStructure.AT_WORK_EAT_PURPOSE_NAME, modelStructure.AT_WORK_MAINT_PURPOSE_NAME}; + + int[] sizeSheetIndices = {SubtourDestChoiceModel.PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET, + SubtourDestChoiceModel.PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET, + SubtourDestChoiceModel.PROPERTIES_AT_WORK_OTHER_SIZE_SHEET}; + + HashMap segmentNameIndexMap = new HashMap(); + HashMap sizeSegmentNameIndexMap = new HashMap(); + for (int k = 0; k < tourPurposeNames.length; k++) + { + segmentNameIndexMap.put(tourPurposeNames[k], k); + sizeSegmentNameIndexMap.put(tourPurposeNames[k], sizeSheetIndices[k]); + } + + double[][] dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; + for (int i = 0; i < aggAccDcSizeArray.length; i++) + { + for (int m : segmentNameIndexMap.values()) + { + int s = sizeSheetIndices[m]; + dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; + } + } + + // compute the arrays of cumulative probabilities based on mgra size for + // mgras within each origin taz. + subTourSizeProbs = new double[tourPurposeNames.length][][]; + subTourTazDistProbs = new double[tourPurposeNames.length][][]; + + DestChoiceTwoStageSoaProbabilitiesCalculator subTourSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( + propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, + PROPERTIES_ATWORK_DC_SOA_UEC_MODEL_PAGE, PROPERTIES_ATWORK_DC_SOA_UEC_DATA_PAGE); + + for (String tourPurpose : tourPurposeNames) + { + + int purposeSizeIndex = segmentNameIndexMap.get(tourPurpose); + + // aggregate TAZ size for the non-mandatoy purpose and set array in + // DMU + double[] tazSize = computeTazSize(dcSizeArray[purposeSizeIndex]); + dcDistSoaDmu.setDestChoiceTazSize(tazSize); + + logger.info(" " + tourPurpose + " probabilities"); + subTourTazDistProbs[purposeSizeIndex] = subTourSoaDistProbsObject + .computeDistanceProbabilities(dcDistSoaDmu); + + subTourSizeProbs[purposeSizeIndex] = computeSizeSegmentProbabilities(dcSizeArray[purposeSizeIndex]); + + } + + } + + private double[] computeTazSize(double[] size) + { + + // this is a 0-based array of cumulative probabilities + double[] tazSize = new double[maxTaz + 1]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + if (mgraArray != null) + { + for (int mgra : mgraArray) + { + tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); + } + } + + } + + return tazSize; + + } + + private double[][] computeSizeSegmentProbabilities(double[] size) + { + + // this is a 0-based array of cumulative probabilities + double[][] sizeProbs = new double[maxTaz][]; + + for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) + { + + int[] mgraArray = tdm.getMgraArray(taz); + + if (mgraArray == null) + { + sizeProbs[taz - 1] = new double[0]; + } else + { + double totalSize = 0; + for (int mgra : mgraArray) + totalSize += size[mgra] + (size[mgra] > 0 ? 1 : 0); + + if (totalSize > 0) + { + sizeProbs[taz - 1] = new double[mgraArray.length]; + for (int i = 0; i < mgraArray.length; i++) + { + double mgraSize = size[mgraArray[i]]; + if (mgraSize > 0) mgraSize += 1; + sizeProbs[taz - 1][i] = mgraSize / totalSize; + } + } else + { + sizeProbs[taz - 1] = new double[0]; + } + } + + } + + return sizeProbs; + + } + + private double[] computeTazDistanceToExternalCordonLogsums() + { + + int maxTaz = tdm.getMaxTaz(); + String uecPath = propertyMap.get("uec.path"); + String altFileName = uecPath + propertyMap.get("internalExternal.dc.uec.alts.file"); + TableDataSet altData = readFile(altFileName); + + int tazCol = altData.getColumnPosition("taz"); + int ieCol = altData.getColumnPosition("iePct"); + altData.buildIndex(tazCol); + + // get parameters used to develop distance to cordon logsums for IE + // model + String coeffString = propertyMap.get(IE_DISTANCE_LOGSUM_COEFF_KEY); + double coeff = Double.parseDouble(coeffString); + + ArrayList tazList = new ArrayList(); + String externalTazListString = propertyMap.get(IE_EXTERNAL_TAZS_KEY); + StringTokenizer st = new StringTokenizer(externalTazListString, ","); + while (st.hasMoreTokens()) + { + String listValue = st.nextToken(); + int tazValue = Integer.parseInt(listValue.trim()); + tazList.add(tazValue); + } + int[] externalTazs = new int[tazList.size()]; + for (int i = 0; i < externalTazs.length; i++) + externalTazs[i] = tazList.get(i); + + // get stored distance arrays + double[][][] periodDistanceMatrices = tazDistanceCalculator + .getStoredFromTazToAllTazsDistanceSkims(); + + // compute the TAZ x EXTERNAL TAZ distance based logsums. + double[] tazDistLogsums = new double[maxTaz + 1]; + for (int i = 1; i <= maxTaz; i++) + { + + double sum = 0; + for (int j = 0; j < externalTazs.length; j++) + { + double distanceToExternal = periodDistanceMatrices[ModelStructure.MD_SKIM_PERIOD_INDEX][i][externalTazs[j]]; + double iePct = altData.getValueAt(externalTazs[j], ieCol); + sum += iePct * MathUtil.exp(coeff * distanceToExternal); + } + + tazDistLogsums[i] = MathUtil.log(sum); + } + + return tazDistLogsums; + + } + + /** + * Read the file and return the TableDataSet. + * + * @param fileName + * @return data + */ + private TableDataSet readFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet data; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + data = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return data; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java new file mode 100644 index 0000000..5ca5dcc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java @@ -0,0 +1,207 @@ +package org.sandag.abm.ctramp; + +import java.net.UnknownHostException; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.jppf.node.protocol.AbstractTask; +import org.jppf.node.protocol.DataProvider; +import org.jppf.node.protocol.JPPFTask; + +import com.pb.common.calculator.MatrixDataServerIf; + +public class HouseholdChoiceModelsTaskJppf + extends AbstractTask +{ + + private transient HashMap propertyMap; + private transient MatrixDataServerIf ms; + private transient HouseholdDataManagerIf hhDataManager; + private transient ModelStructure modelStructure; + private transient CtrampDmuFactoryIf dmuFactory; + private transient String restartModelString; + + private int startIndex; + private int endIndex; + private int taskIndex; + + private int maxAlts; + + private boolean runWithTiming; + private boolean logResults=false; + + public HouseholdChoiceModelsTaskJppf(int taskIndex, int startIndex, int endIndex) + { + this.startIndex = startIndex; + this.endIndex = endIndex; + this.taskIndex = taskIndex; + runWithTiming = true; + } + + public void run() + { + + long startTime = System.nanoTime(); + + Logger logger = Logger.getLogger(this.getClass()); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " + + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + try + { + + DataProvider dataProvider = getDataProvider(); + + propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); + logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") + .equalsIgnoreCase("true"); + ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); + hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); + modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); + dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); + restartModelString = (String) dataProvider.getParameter("restartModelString"); + + } catch (Exception e) + { + e.printStackTrace(); + } + + // get the factory object used to create and recycle + // HouseholdChoiceModels objects. + HouseholdChoiceModelsManager modelManager = HouseholdChoiceModelsManager.getInstance(); + modelManager.managerSetup(ms, hhDataManager, propertyMap, restartModelString, + modelStructure, dmuFactory); + + HouseholdChoiceModels hhModel = modelManager.getHouseholdChoiceModelsObject(taskIndex); + + long setup1 = 0; + long setup2 = 0; + long setup3 = 0; + long setup4 = 0; + long setup5 = 0; + + setup1 = (System.nanoTime() - startTime) / 1000000; + + Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); + + setup2 = (System.nanoTime() - startTime) / 1000000; + + boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, + HouseholdDataManager.DEBUG_HHS_ONLY_KEY); + + if (runWithTiming) hhModel.zeroTimes(); + for (int i = 0; i < householdArray.length; i++) + { + + // for debugging only - process only household objects specified for + // debugging, if property key was set to true + if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; + + try + { + if (runWithTiming) hhModel.runModelsWithTiming(householdArray[i]); + else hhModel.runModels(householdArray[i]); + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught in taskIndex=%d hhModel index=%d applying hh model for i=%d, hhId=%d.", + taskIndex, hhModel.getModelIndex(), i, householdArray[i].getHhId())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + } + + long[] componentTimes = hhModel.getTimes(); + long[] partialStopTimes = hhModel.getPartialStopTimes(); + + if (hhModel.getMaxAlts() > maxAlts) maxAlts = hhModel.getMaxAlts(); + + setup3 = (System.nanoTime() - startTime) / 1000000; + + hhDataManager.setHhArray(householdArray, startIndex); + + setup4 = (System.nanoTime() - startTime) / 1000000; + + logger.info(String + .format("end of household choice model thread=%s, task[%d], hhModel[%d], startIndex=%d, endIndex=%d", + threadName, taskIndex, hhModel.getModelIndex(), startIndex, endIndex)); + + setResult(String.format("taskIndex=%d, hhModelInstance=%d, startIndex=%d, endIndex=%d", + taskIndex, hhModel.getModelIndex(), startIndex, endIndex)); + + setup5 = (System.nanoTime() - startTime) / 1000000; + + if(logResults){ + logger.info("task=" + taskIndex + ", setup=" + setup1 + ", getHhs=" + (setup2 - setup1) + + ", processHhs=" + (setup3 - setup2) + ", putHhs=" + (setup4 - setup3) + + ", return model=" + (setup5 - setup4) + "."); + } + + if (runWithTiming) + logModelComponentTimes(componentTimes, partialStopTimes, logger, + hhModel.getModelIndex()); + + // this has to be the last statement in this method. + // add this DestChoiceModel instance to the static queue shared by other + // tasks of this type + modelManager.returnHouseholdChoiceModelsObject(hhModel, startIndex, endIndex); + + } + + private void logModelComponentTimes(long[] componentTimes, long[] partialStopTimes, + Logger logger, int modelIndex) + { + + String[] label1 = {"AO", "FP", "IE", "CDAP", "IMTF", "IMTOD", "IMMC", "JTF", "JTDC", + "JTTOD", "JTMC", "INMTF", "INMTDCSOA", "INMTDCTOT", "INMTTOD", "INMTMC", "AWTF", + "AWTDCSOA", "AWTDCTOT", "AWTTOD", "AWTMC", "STF", "STDTM"}; + + logger.info("Household choice model component runtimes (in milliseconds) for task: " + + taskIndex + ", modelIndex: " + modelIndex + ", startIndex: " + startIndex + + ", endIndex: " + endIndex); + + float total = 0; + for (int i = 0; i < componentTimes.length; i++) + { + float time = (componentTimes[i] / 1000000); + logger.info(String.format("%-6d%30s:%15.1f", (i + 1), label1[i], time)); + total += time; + } + logger.info(String.format("%-6s%30s:%10.1f", "Total", "Total all components", total)); + logger.info(""); + + String[] label2 = {"SLC SOA AUTO", "SLC SOA OTHER", "SLC LS", "SLC DIST", "SLC", "SLC TOT", + "S TOD", "S MC", "TOTAL"}; + + logger.info(""); + logger.info("Times for parts of intermediate stop models:"); + for (int i = 0; i < partialStopTimes.length; i++) + { + float time = (partialStopTimes[i] / 1000000); + logger.info(String.format("%-6d%30s:%15.1f", (i + 1), label2[i], time)); + } + + } + + public String getId() + { + return Integer.toString(taskIndex); + } + + public int getMaxAlts() + { + return maxAlts; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java new file mode 100644 index 0000000..2b26011 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java @@ -0,0 +1,1423 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import com.pb.common.calculator.VariableTable; +import com.pb.common.model.Alternative; +import com.pb.common.model.ConcreteAlternative; +import com.pb.common.model.LogitModel; +import com.pb.common.model.ModelException; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * Implements a coordinated daily activity pattern model, which is a joint + * choice of activity types of each member of a household. The class builds and + * applies separate choice models for households of sizes 1, 2, 3, 4, and 5. For + * households larger than 5, the persons in the household are ordered such that + * the first 5 members include up to 2 workers and 3 children (youngest to + * oldest), the 5-person model is applied for these 5 household members, than a + * separate, simple cross-sectional distribution is looked up for the remaining + * household members. + * + * The utilities are computed using four separate UEC spreadsheets. The first + * computes the activity utility for each person individually; the second + * computes the activity utility for each person when paired with each other + * person; the third computes the activity utility for each person when paired + * with each group of two other people in the household; and the fourth computes + * the activity utility considering all the members of the household. These + * utilities are then aggregated to represent each possible activity pattern for + * the household, and the choice is made. For households larger than 5, a second + * model is applied after the first, which selects a pattern for the 5+ + * household members from a predefined distribution. + * + * @author D. Ory + * + */ +public class HouseholdCoordinatedDailyActivityPatternModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdCoordinatedDailyActivityPatternModel.class); + private transient Logger cdapLogger = Logger.getLogger("cdap"); + private transient Logger cdapUecLogger = Logger.getLogger("cdap_uec"); + private transient Logger cdapLogsumLogger = Logger.getLogger("cdap_logsum"); + + private static final String UEC_FILE_NAME_PROPERTY = "cdap.uec.file"; + private static final String UEC_DATA_PAGE_PROPERTY = "cdap.data.page"; + private static final String UEC_ONE_PERSON_UTILITY_PAGE_PROPERTY = "cdap.one.person.page"; + private static final String UEC_TWO_PERSON_UTILITY_PAGE_PROPERTY = "cdap.two.person.page"; + private static final String UEC_THREE_PERSON_UTILITY_PAGE_PROPERTY = "cdap.three.person.page"; + private static final String UEC_ALL_PERSON_UTILITY_PAGE_PROPERTY = "cdap.all.person.page"; + private static final String UEC_JOINT_UTILITY_PAGE_PROPERTY = "cdap.joint.page"; + + public static final int MAX_MODEL_HH_SIZE = 5; + + private static final String MANDATORY_PATTERN = Definitions.MANDATORY_PATTERN; + private static final String NONMANDATORY_PATTERN = Definitions.NONMANDATORY_PATTERN; + private static final String HOME_PATTERN = Definitions.HOME_PATTERN; + private static final String[] ACTIVITY_NAME_ARRAY = { + MANDATORY_PATTERN, NONMANDATORY_PATTERN, HOME_PATTERN }; + + private ModelStructure modelStructure; + private double[][] fixedCumulativeProportions; + + // collection of logit models - one for each household size + private ArrayList logitModelList; + + private AccessibilitiesTable accTable; + + // DMU for the UEC + private CoordinatedDailyActivityPatternDMU cdapDmuObject; + + // re-ordered collection of households + private Person[] cdapPersonArray; + + // Five separate UECs to compute segments of the utility + private UtilityExpressionCalculator onePersonUec, twoPeopleUec, threePeopleUec, + allMemberInteractionUec, jointUec; + + public HouseholdCoordinatedDailyActivityPatternModel(HashMap propertyMap, + ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory, + AccessibilitiesTable myAccTable) + { + + modelStructure = myModelStructure; + accTable = myAccTable; + + // setup the coordinated daily activity pattern choice model objects + createLogitModels(); + setupCoordinatedDailyActivityPatternModelApplication(propertyMap, dmuFactory); + + } + + private void setupCoordinatedDailyActivityPatternModelApplication( + HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + + logger.info("setting up CDAP choice model."); + + // locate the coordinated daily activity pattern choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String cdapUecFile = propertyMap.get(UEC_FILE_NAME_PROPERTY); + cdapUecFile = uecPath + cdapUecFile; + + int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, UEC_DATA_PAGE_PROPERTY); + int onePersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, + UEC_ONE_PERSON_UTILITY_PAGE_PROPERTY); + int twoPersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, + UEC_TWO_PERSON_UTILITY_PAGE_PROPERTY); + int threePersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, + UEC_THREE_PERSON_UTILITY_PAGE_PROPERTY); + int allPersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, + UEC_ALL_PERSON_UTILITY_PAGE_PROPERTY); + int jointPage = Util.getIntegerValueFromPropertyMap(propertyMap, + UEC_JOINT_UTILITY_PAGE_PROPERTY); + + // create the coordinated daily activity pattern choice model DMU + // object. + cdapDmuObject = dmuFactory.getCoordinatedDailyActivityPatternDMU(); + + // create the uecs + onePersonUec = new UtilityExpressionCalculator(new File(cdapUecFile), onePersonPage, + dataPage, propertyMap, (VariableTable) cdapDmuObject); + twoPeopleUec = new UtilityExpressionCalculator(new File(cdapUecFile), twoPersonPage, + dataPage, propertyMap, (VariableTable) cdapDmuObject); + threePeopleUec = new UtilityExpressionCalculator(new File(cdapUecFile), threePersonPage, + dataPage, propertyMap, (VariableTable) cdapDmuObject); + allMemberInteractionUec = new UtilityExpressionCalculator(new File(cdapUecFile), + allPersonPage, dataPage, propertyMap, (VariableTable) cdapDmuObject); + jointUec = new UtilityExpressionCalculator(new File(cdapUecFile), jointPage, dataPage, + propertyMap, (VariableTable) cdapDmuObject); + + // get the proportions by person type + double[][] fixedRelativeProportions = modelStructure.getCdap6PlusProps(); + fixedCumulativeProportions = new double[fixedRelativeProportions.length][]; + + // i loops over personTypes, 0 not used. + for (int i = 1; i < fixedRelativeProportions.length; i++) + { + fixedCumulativeProportions[i] = new double[fixedRelativeProportions[i].length]; + + // j loops over cdap patterns, can skip index 0. + fixedCumulativeProportions[i][0] = fixedRelativeProportions[i][0]; + for (int j = 1; j < fixedRelativeProportions[i].length; j++) + fixedCumulativeProportions[i][j] = fixedCumulativeProportions[i][j - 1] + + fixedRelativeProportions[i][j]; + + // calculate the difference between 1.0 and the cumulative + // proportion and + // add to the Mandatory category (j==0) + // to make sure the cumulative propbabilities sum to exactly 1.0. + double diff = 1.0 - fixedCumulativeProportions[i][fixedRelativeProportions[i].length - 1]; + fixedCumulativeProportions[i][0] += diff; + } + + } + + public void applyModel(Household hhObject) + { + + if (hhObject.getDebugChoiceModels()) + hhObject.logHouseholdObject("Pre CDAP Household " + hhObject.getHhId() + " Object", + cdapLogger); + + // get the activity pattern choice + String pattern = getCoordinatedDailyActivityPatternChoice(hhObject); + + // set the pattern for the household + hhObject.setCoordinatedDailyActivityPatternResult(pattern); + + // set the pattern for each person and count by person type + Person[] personArray = hhObject.getPersons(); + for (int j = 1; j < personArray.length; ++j) + { + String activityString = pattern.substring(j - 1, j); + personArray[j].setDailyActivityResult(activityString); + } // j (person loop) + + // log results for debug households + if (hhObject.getDebugChoiceModels()) + { + + cdapLogger.info(" "); + cdapLogger.info("CDAP Chosen Pattern by Person Type"); + cdapLogger + .info("(* indicates person was involved in coordinated choice; no * indicates choice by fixed proportions)"); + cdapLogger.info("CDAP # Type FT W PT W UNIV NONW RETR SCHD SCHN PRES"); + cdapLogger.info("------ ---- ---- ---- ---- ---- ---- ---- ---- ----"); + + String bString = ""; + for (int j = 1; j < personArray.length; ++j) + { + + Person[] tempPersonArray = getPersonsNotModeledByCdap(MAX_MODEL_HH_SIZE); + + boolean persNumMatch = false; + for (int jj = 1; jj < tempPersonArray.length; jj++) + { + if (tempPersonArray[jj].getPersonNum() == personArray[j].getPersonNum()) + persNumMatch = true; + } + + String persNumString = ""; + if (persNumMatch) persNumString = String.format("%d ", j); + else persNumString = String.format("%d *", j); + + String pString = pattern.substring(j - 1, j); + String stringToLog = ""; + + if (personArray[j].getPersonTypeIsFullTimeWorker() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "FT W", pString, bString, bString, bString, bString, bString, bString, + bString); + } else if (personArray[j].getPersonTypeIsPartTimeWorker() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "PT W", bString, pString, bString, bString, bString, bString, bString, + bString); + } else if (personArray[j].getPersonIsUniversityStudent() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "UNIV", bString, bString, pString, bString, bString, bString, bString, + bString); + } else if (personArray[j].getPersonIsNonWorkingAdultUnder65() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "NONW", bString, bString, bString, pString, bString, bString, bString, + bString); + } else if (personArray[j].getPersonIsNonWorkingAdultOver65() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "RETR", bString, bString, bString, bString, pString, bString, bString, + bString); + } else if (personArray[j].getPersonIsStudentDriving() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "SCHD", bString, bString, bString, bString, bString, pString, bString, + bString); + } else if (personArray[j].getPersonIsStudentNonDriving() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "SCHN", bString, bString, bString, bString, bString, bString, pString, + bString); + } else if (personArray[j].getPersonIsPreschoolChild() == 1) + { + stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, + "PRES", bString, bString, bString, bString, bString, bString, bString, + pString); + } + + cdapLogger.info(stringToLog); + + } // j (person loop) + + cdapLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + cdapLogger.info(""); + cdapLogger.info(""); + + } // if traceMe + + hhObject.setCdapRandomCount(hhObject.getHhRandomCount()); + + } + + /** + * Prepares a separate logit model for households of size 1, 2, 3, 4, and 5. + * Each model has 3^n alternatives, where n is the household size. The + * models are cleared and re-used for each household of the specified size. + * + */ + private void createLogitModels() + { + + // new the collection of logit models + logitModelList = new ArrayList(MAX_MODEL_HH_SIZE); + + // build a model for each HH size + for (int i = 0; i < MAX_MODEL_HH_SIZE; ++i) + { + + int hhSize = i + 1; + + // create the working model + LogitModel workingLogitModel = new LogitModel(hhSize + " Person HH"); + + // compute the number of alternatives + int numberOfAlternatives = 1; + for (int j = 0; j < hhSize; ++j) + numberOfAlternatives *= ACTIVITY_NAME_ARRAY.length; + + // create a counter for each of the people in the hh + int[] counterForEachPerson = new int[hhSize]; + Arrays.fill(counterForEachPerson, 0); + + // create the alternatives and add them to the logit model + int numberOfAltsCounter = 0; + int totalAltsCounter = 0; + while (numberOfAltsCounter < numberOfAlternatives) + { + + // set the string for the alternative + String alternativeName = ""; + int numOutOfHomeActivites = 0; + for (int j = 0; j < hhSize; ++j) + { + alternativeName += ACTIVITY_NAME_ARRAY[counterForEachPerson[j]]; + if (!ACTIVITY_NAME_ARRAY[counterForEachPerson[j]] + .equalsIgnoreCase(HOME_PATTERN)) numOutOfHomeActivites++; + } + + // create the alternative and add it to the model + if (numOutOfHomeActivites < 2) + { + ConcreteAlternative tempAlt = new ConcreteAlternative(alternativeName + "0", + totalAltsCounter); + workingLogitModel.addAlternative(tempAlt); + numberOfAltsCounter++; + totalAltsCounter++; + + } else + { + + ConcreteAlternative tempAlt = new ConcreteAlternative(alternativeName + "0", + totalAltsCounter); + workingLogitModel.addAlternative(tempAlt); + numberOfAltsCounter++; + totalAltsCounter++; + + tempAlt = new ConcreteAlternative(alternativeName + "j", totalAltsCounter); + workingLogitModel.addAlternative(tempAlt); + totalAltsCounter++; + + } + + // check increment the counters + for (int j = 0; j < hhSize; ++j) + { + counterForEachPerson[j]++; + if (counterForEachPerson[j] == ACTIVITY_NAME_ARRAY.length) counterForEachPerson[j] = 0; + else break; + } + + } + + // add the model to the array list + logitModelList.add(i, workingLogitModel); + + } // for i max hh size + + } + + /** + * Selects the coordinated daily activity pattern choice for the passed in + * Household. The method works for households of all sizes, though two + * separate models are applied for households with more than 5 members. + * + * @param householdObject + * @return a string of length household size, where each character in the + * string represents the activity pattern for that person, in order + * (see Household.reOrderPersonsForCdap method). + */ + public String getCoordinatedDailyActivityPatternChoice(Household householdObject) + { + + // set all household level dmu variables + cdapDmuObject.setHousehold(householdObject); + + // set the hh size (cap modeled size at MAX_MODEL_HH_SIZE) + int actualHhSize = householdObject.getSize(); + int modelHhSize = Math.min(MAX_MODEL_HH_SIZE, actualHhSize); + + // reorder persons for large households if need be + reOrderPersonsForCdap(householdObject); + + // get the logit model we need and clear it of any lingering probilities + LogitModel workingLogitModel = logitModelList.get(modelHhSize - 1); + workingLogitModel.clear(); + + // get the alternatives and reset the utilities to zero + ArrayList alternativeList = workingLogitModel.getAlternatives(); + for (int i = 0; i < alternativeList.size(); ++i) + { + Alternative tempAlt = (Alternative) alternativeList.get(i); + tempAlt.setUtility(0.0); + } + + // write the debug header if we have a trace household + if (householdObject.getDebugChoiceModels()) + { + + LogitModel.setLogger(cdapLogsumLogger); + + cdapLogger.info(" "); + cdapLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + cdapLogger.info("CDAP Model: Debug Statement for Household ID: " + + householdObject.getHhId()); + String firstHeader = "Utility Segment PersonA PersonB PersonC"; + String secondHeader = "------------------------------ -------- -------- --------"; + for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) + { + firstHeader += " " + ACTIVITY_NAME_ARRAY[j] + " util"; + secondHeader += " ---------"; + } + + cdapLogger.info(firstHeader); + cdapLogger.info(secondHeader); + + } + + // all the alternatives are available for all households (1-based, + // ignore 0 + // index and set other three to 1.) + int[] availability = {-1, 1, 1, 1}; + + String[] accStrings = {"", "hov0", "hov1", "hov2"}; + float retAccess = accTable.getAggregateAccessibility( + accStrings[householdObject.getAutoSufficiency()], householdObject.getHhMgra()); + + cdapDmuObject.setRetailAccessibility(retAccess); + + // loop through each person + for (int i = 0; i < modelHhSize; ++i) + { + + // get personA + Person personA = getCdapPerson(i + 1); + + // set the person level dmu variables + cdapDmuObject.setPersonA(personA); + + int workMgra = personA.getWorkLocation(); + if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) cdapDmuObject + .setWorkLocationModeChoiceLogsumA(personA.getWorkLocationLogsum()); + else cdapDmuObject.setWorkLocationModeChoiceLogsumA(0.0); + + int schoolMgra = personA.getPersonSchoolLocationZone(); + int studentType = getStudentTypeForThisCdapPerson(personA); + if (studentType > 0 && schoolMgra > 0) + { + cdapDmuObject.setSchoolLocationModeChoiceLogsumA(personA.getSchoolLocationLogsum()); + } else + { + cdapDmuObject.setSchoolLocationModeChoiceLogsumA(0.0); + } + + // compute the single person utilities + double[] firstPersonUtilities = onePersonUec.solve(cdapDmuObject.getIndexValues(), + cdapDmuObject, availability); + + // log these utilities for trace households + if (householdObject.getDebugChoiceModels()) + { + + String stringToLog = String.format("%-30s%9d%9s%9s", "OnePerson", (i + 1), "--", + "--"); + + for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) + { + stringToLog += String.format("%10.4f", firstPersonUtilities[j]); + } + cdapLogger.info(stringToLog); + + cdapUecLogger.info("PersonA:"); + personA.logEntirePersonObject(cdapUecLogger); + onePersonUec.logAnswersArray(cdapUecLogger, "ONE PERSON, personA personNum=" + + personA.getPersonNum()); + + } // debug trace + + // align the one person utilities with the alternatives for person i and calculate their individual logsum + float individualLogsum = 0; + for (int j = 0; j < alternativeList.size(); ++j) + { + + // get the name of the alternative + Alternative tempAlt = (Alternative) alternativeList.get(j); + String altName = tempAlt.getName(); + + // get the name of the activity for this person in the + // alternative + // string + String altNameForPersonA = altName.substring(i, i + 1); + + // align the utility results with this activity + for (int k = 0; k < ACTIVITY_NAME_ARRAY.length; ++k) + { + + if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[k])) + { + double currentUtility = tempAlt.getUtility(); + tempAlt.setUtility(currentUtility + firstPersonUtilities[k]); + individualLogsum += Math.exp(firstPersonUtilities[k]); + } + } // k + + } // j + + individualLogsum = (float) Math.log(individualLogsum); + personA.setCdapLogsum(individualLogsum); + + // loop through all possible person Bs + for (int j = 0; j < modelHhSize; ++j) + { + + // skip if same as person A + if (i == j) continue; + + Person personB = getCdapPerson(j + 1); + + // skip if i>j because if we have 1,2 for person 1, we don't + // also + // want 2,1; that's the + // same combination of two people + if (i > j) continue; + + // set the two person level dmu variables + cdapDmuObject.setPersonB(personB); + + // compute the two people utilities + double[] twoPersonUtilities = twoPeopleUec.solve(cdapDmuObject.getIndexValues(), + cdapDmuObject, availability); + + // log these utilities for trace households + if (householdObject.getDebugChoiceModels()) + { + + String stringToLog = String.format("%-30s%9d%9d%9s", "TwoPeople", (i + 1), + (j + 1), "--"); + + for (int k = 0; k < ACTIVITY_NAME_ARRAY.length; ++k) + { + stringToLog += String.format("%10.4f", twoPersonUtilities[k]); + } + cdapLogger.info(stringToLog); + + cdapUecLogger.info("PersonA:"); + personA.logEntirePersonObject(cdapUecLogger); + cdapUecLogger.info("PersonB:"); + personB.logEntirePersonObject(cdapUecLogger); + twoPeopleUec.logAnswersArray(cdapUecLogger, + "TWO PERSON, personA personNum=" + personA.getPersonNum() + + " personB personNum=" + personB.getPersonNum()); + + } // debug trace + + // align the two person utilities with the alternatives for + // person i + for (int k = 0; k < alternativeList.size(); ++k) + { + Alternative tempAlt = (Alternative) alternativeList.get(k); + String altName = tempAlt.getName(); + + // get the name of the activity for this person in the + // alternative string + String altNameForPersonA = altName.substring(i, i + 1); + String altNameForPersonB = altName.substring(j, j + 1); + + for (int l = 0; l < ACTIVITY_NAME_ARRAY.length; ++l) + { + if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[l]) + && altNameForPersonB.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[l])) + { + double currentUtility = tempAlt.getUtility(); + tempAlt.setUtility(currentUtility + twoPersonUtilities[l]); + } + } // l + } // k + + // loop through all possible person Cs + for (int k = 0; k < modelHhSize; ++k) + { + + // skip if same as person A + if (i == k) continue; + + // skip if same as person B + if (j == k) continue; + + // skip if j>k because if we have 1,2,3 for person 1, we + // don't + // also want 1,3,2; that's the + // same combination of three people + if (j > k) continue; + + Person personC = getCdapPerson(k + 1); + + // set the three level dmu variables + cdapDmuObject.setPersonC(personC); + + // compute the three person utilities + double[] threePersonUtilities = threePeopleUec.solve( + cdapDmuObject.getIndexValues(), cdapDmuObject, availability); + + // log these utilities for trace households + if (householdObject.getDebugChoiceModels()) + { + + String stringToLog = String.format("%-30s%9d%9d%9d", "ThreePeople", + (i + 1), (j + 1), (k + 1)); + + for (int l = 0; l < ACTIVITY_NAME_ARRAY.length; ++l) + { + stringToLog += String.format("%10.4f", threePersonUtilities[l]); + } + cdapLogger.info(stringToLog); + + cdapUecLogger.info("PersonA:"); + personA.logEntirePersonObject(cdapUecLogger); + cdapUecLogger.info("PersonB:"); + personB.logEntirePersonObject(cdapUecLogger); + cdapUecLogger.info("PersonC:"); + personC.logEntirePersonObject(cdapUecLogger); + threePeopleUec.logAnswersArray(cdapUecLogger, + "THREE PERSON, personA personNum=" + personA.getPersonNum() + + " personB personNum=" + personB.getPersonNum() + + " personC personNum=" + personC.getPersonNum()); + + } // debug trace + + // align the three person utilities with the alternatives + // for + // person i + for (int l = 0; l < alternativeList.size(); ++l) + { + Alternative tempAlt = (Alternative) alternativeList.get(l); + String altName = tempAlt.getName(); + + // get the name of the activity for this person in the + // alternative string + String altNameForPersonA = altName.substring(i, i + 1); + String altNameForPersonB = altName.substring(j, j + 1); + String altNameForPersonC = altName.substring(k, k + 1); + + for (int m = 0; m < ACTIVITY_NAME_ARRAY.length; ++m) + { + if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m]) + && altNameForPersonB.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m]) + && altNameForPersonC.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m])) + { + double currentUtility = tempAlt.getUtility(); + tempAlt.setUtility(currentUtility + threePersonUtilities[m]); + } + } // m + } // l + + } // k (person C loop) + + } // j (person B loop) + + } // i (person A loop) + + // compute the interaction utilities + double[] allMemberInteractionUtilities = allMemberInteractionUec.solve( + cdapDmuObject.getIndexValues(), cdapDmuObject, availability); + + // log these utilities for trace households + if (householdObject.getDebugChoiceModels()) + { + + String stringToLog = String.format("%-30s%9s%9s%9s", "AllMembers", "--", "--", "--"); + + for (int i = 0; i < ACTIVITY_NAME_ARRAY.length; ++i) + { + stringToLog += String.format("%10.4f", allMemberInteractionUtilities[i]); + } + cdapLogger.info(stringToLog); + + allMemberInteractionUec.logAnswersArray(cdapUecLogger, "ALL MEMBER INTERACTIONS"); + + } // debug trace + + // align the utilities with the proper alternatives + for (int i = 0; i < alternativeList.size(); ++i) + { + Alternative tempAlt = (Alternative) alternativeList.get(i); + String altName = tempAlt.getName(); + + for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) + { + + boolean samePattern = true; + for (int k = 0; k < modelHhSize; ++k) + { + + // alternative should have pattern j for each member k + String altNameForThisPerson = altName.substring(k, k + 1); + if (altNameForThisPerson.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[j])) continue; + else + { + samePattern = false; + break; + } + } // k + + // if all have the same pattern, add the new utilities + if (samePattern) + { + double currentUtility = tempAlt.getUtility(); + tempAlt.setUtility(currentUtility + allMemberInteractionUtilities[j]); + } + } // j + + } // i + + // compute the joint utilities to be added to alternatives with joint + // tour + // indicator + + int adultsWithMand = 0; + int adultsWithNonMand = 0; + int kidsWithMand = 0; + int kidsWithNonMand = 0; + int adultsLeaveHome = 0; + int workMgra = 0; + double workLocationAccessibilityForWorkers = 0.0; + for (int i = 0; i < alternativeList.size(); ++i) + { + + Alternative tempAlt = (Alternative) alternativeList.get(i); + String altName = tempAlt.getName(); + + adultsWithMand = 0; + adultsWithNonMand = 0; + kidsWithMand = 0; + kidsWithNonMand = 0; + adultsLeaveHome = 0; + workMgra = 0; + workLocationAccessibilityForWorkers = 0.0; + for (int k = 0; k < modelHhSize; ++k) + { + + // alternative should have pattern j for each member k + String altNameForThisPerson = altName.substring(k, k + 1); + if (altNameForThisPerson.equalsIgnoreCase(MANDATORY_PATTERN)) + { + if (isThisCdapPersonAnAdult(k + 1)) + { + adultsWithMand++; + adultsLeaveHome++; + } else + { + kidsWithMand++; + } + + workMgra = getWorkLocationForThisCdapPerson(k + 1); + if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + Person tempPerson = getCdapPerson(k + 1); + workLocationAccessibilityForWorkers += tempPerson.getWorkLocationLogsum(); + } + + } else if (altNameForThisPerson.equalsIgnoreCase(NONMANDATORY_PATTERN)) + { + if (isThisCdapPersonAnAdult(k + 1)) + { + adultsWithNonMand++; + adultsLeaveHome++; + } else + { + kidsWithNonMand++; + } + } + + } // k + + cdapDmuObject.setNumAdultsWithNonMandatoryDap(adultsWithNonMand); + cdapDmuObject.setNumAdultsWithMandatoryDap(adultsWithMand); + cdapDmuObject.setNumKidsWithNonMandatoryDap(kidsWithNonMand); + cdapDmuObject.setNumKidsWithMandatoryDap(kidsWithMand); + cdapDmuObject.setAllAdultsAtHome(adultsLeaveHome == 0 ? 1 : 0); + cdapDmuObject.setWorkAccessForMandatoryDap(workLocationAccessibilityForWorkers); + + double[] jointUtilities = jointUec.solve(cdapDmuObject.getIndexValues(), cdapDmuObject, + availability); + + // log these utilities for trace households + if (householdObject.getDebugChoiceModels()) + { + String stringToLog = String.format("%-13s%4d %-12s%9s%9s%9s", "Joint", (i + 1), + altName, "--", "--", "--"); + stringToLog += String.format("%10.4f", + (altName.indexOf("j") > 0 ? jointUtilities[0] : 0.0)); + cdapLogger.info(stringToLog); + + jointUec.logAnswersArray(cdapUecLogger, "JOINT Utility for CDAP Alt index = " + + (i + 1) + ", Alt name = " + altName); + } // debug trace + + // align the utilities with the proper alternatives + if (altName.indexOf("j") > 0) + { + double currentUtility = tempAlt.getUtility(); + tempAlt.setUtility(currentUtility + jointUtilities[0]); + } + + } // i + + // TODO: check this out - use computeAvailabilty() checks that an + // alternative + // is available + // all utilities are set - compute probabilities + // workingLogitModel.setAvailability(true); + workingLogitModel.computeAvailabilities(); + + // compute the exponentiated utility, logging debug if need be + if (householdObject.getDebugChoiceModels()) + { + workingLogitModel.setDebug(true); + workingLogitModel.writeUtilityHeader(); + } + float logsum = (float) workingLogitModel.getUtility(); + householdObject.setCdapLogsum(logsum); + + // compute the probabilities, logging debug if need be + if (householdObject.getDebugChoiceModels()) + { + workingLogitModel.writeProbabilityHeader(); + } + workingLogitModel.calculateProbabilities(); + + // turn debug off for the next guy + workingLogitModel.setDebug(false); + + // make a choice for the first five + Random hhRandom = householdObject.getHhRandom(); + + double randomNumber = hhRandom.nextDouble(); + + if (householdObject.getDebugChoiceModels()) + { + cdapLogger.info("randomNumber = " + randomNumber); + } + + String firstFiveChosenName = ""; + try + { + ConcreteAlternative chosenAlternative = (ConcreteAlternative) workingLogitModel + .chooseElementalAlternative(randomNumber); + firstFiveChosenName = chosenAlternative.getName(); + + if (householdObject.getDebugChoiceModels()) + { + + int chosenIndex = 0; + // get the index number for the alternative chosen + for (int i = 0; i < alternativeList.size(); ++i) + { + Alternative tempAlt = (Alternative) alternativeList.get(i); + String altName = tempAlt.getName(); + if (altName.equalsIgnoreCase(firstFiveChosenName)) + { + chosenIndex = i + 1; + break; + } + } + + cdapLogger.info("chosen pattern (5 or fewer hh members): Alt index = " + + chosenIndex + ", Alt name = " + firstFiveChosenName); + cdapLogger.info(""); + cdapLogger.info(""); + } + } catch (ModelException e) + { + logger.error(String.format( + "Exception caught for HHID=%d, no available CDAP alternatives to choose.", + householdObject.getHhId()), e); + throw new RuntimeException(); + } + + // make a choice for additional hh members if need be + if (actualHhSize > MAX_MODEL_HH_SIZE) + { + + String allMembersChosenPattern = applyModelForExtraHhMembers(householdObject, + firstFiveChosenName); + + String extraChar = ""; + int extraCharIndex = allMembersChosenPattern.indexOf("0"); + if (extraCharIndex > 0) + { + extraChar = "0"; + } else + { + extraCharIndex = allMembersChosenPattern.indexOf("j"); + if (extraCharIndex > 0) extraChar = "j"; + } + + allMembersChosenPattern = allMembersChosenPattern.substring(0, extraCharIndex) + + allMembersChosenPattern.substring(extraCharIndex + 1); + + // re-order the activities by the original order of persons + String finalHhPattern = ""; + String[] finalHhPatternActivities = new String[cdapPersonArray.length]; + for (int i = 1; i < cdapPersonArray.length; i++) + { + int k = cdapPersonArray[i].getPersonNum(); + finalHhPatternActivities[k] = allMembersChosenPattern.substring(i - 1, i); + } + + for (int i = 1; i < cdapPersonArray.length; i++) + finalHhPattern += finalHhPatternActivities[i]; + + String finalString = finalHhPattern + extraChar; + if (householdObject.getDebugChoiceModels()) + { + cdapLogger.info("final pattern (more than 5 hh members) = " + finalString); + cdapLogger.info(""); + cdapLogger.info(""); + } + return finalString; + + } + + if (householdObject.getDebugChoiceModels()) + { + // reset the logger to what it was before we changed it + LogitModel.setLogger(Logger.getLogger(LogitModel.class)); + } + + // no need to re-order the activities - hhsize <= MAX_MODEL_HH_SIZE have + // original order of persons + return firstFiveChosenName; + + } + + /** + * Applies a simple choice from fixed proportions by person type for members + * of households with more than 5 people who are not included in the CDAP + * model. The choices of the additional household members are independent of + * each other. + * + * @param householdObject + * @param patternStringForOtherHhMembers + * @return the pattern for the entire household, including the 5-member + * pattern chosen by the logit model and the additional members + * chosen by the fixed-distribution model. + * + */ + private String applyModelForExtraHhMembers(Household householdObject, + String patternStringForOtherHhMembers) + { + + String allMembersPattern = patternStringForOtherHhMembers; + + // get the persons not yet modeled + Person[] personArray = getPersonsNotModeledByCdap(MAX_MODEL_HH_SIZE); + + // person array is 1-based to be consistent with other person arrays + for (int i = 1; i < personArray.length; i++) + { + + int personType = personArray[i].getPersonTypeNumber(); + + // get choice index from fixed proportions for 6 plus persons + int chosen = ChoiceModelApplication.getMonteCarloSelection( + fixedCumulativeProportions[personType], householdObject.getHhRandom() + .nextDouble()); + + allMembersPattern += ACTIVITY_NAME_ARRAY[chosen]; + + } + + return allMembersPattern; + + } + + /** + * Method reorders the persons in the household for use with the CDAP model, + * which only explicitly models the interaction of five persons in a HH. + * Priority in the reordering is first given to full time workers (up to + * two), then to part time workers (up to two workers, of any type), then to + * children (youngest to oldest, up to three). If the method is called for a + * household with less than 5 people, the cdapPersonArray is the same as the + * person array. + * + */ + public void reOrderPersonsForCdap(Household household) + { + + // set the person array + Person[] persons = household.getPersons(); + + // if hh is not too big, set cdap equal to persons and return + int hhSize = household.getSize(); + if (hhSize <= MAX_MODEL_HH_SIZE) + { + cdapPersonArray = persons; + return; + } + + // create the end game array + cdapPersonArray = new Person[persons.length]; + + // keep track of which persons you count + boolean[] iCountedYou = new boolean[persons.length]; + Arrays.fill(iCountedYou, false); + + // define the persons we want to find among the five + int firstWorkerIndex = -99; + int secondWorkerIndex = -99; + + int youngestChildIndex = -99; + int secondYoungestChildIndex = -99; + int thirdYoungestChildIndex = -99; + + int youngestChildAge = 99; + int secondYoungestChildAge = 99; + int thirdYoungestChildAge = 99; + + // first: full-time workers (persons is 1-based array) + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + // is the person a full-time worker + if (persons[i].getPersonIsFullTimeWorker() == 1) + { + + // check our indices + if (firstWorkerIndex == -99) + { + firstWorkerIndex = i; + iCountedYou[i] = true; + } else if (secondWorkerIndex == -99) + { + secondWorkerIndex = i; + iCountedYou[i] = true; + } + } + + } // i (full time workers) + + // second: part-time workers (only if we don't have two workers) + if (firstWorkerIndex == -99 || secondWorkerIndex == -99) + { + + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + // is the person part-time worker + if (persons[i].getPersonIsPartTimeWorker() == 1) + { + + // check our indices + if (firstWorkerIndex == -99) + { + firstWorkerIndex = i; + iCountedYou[i] = true; + } else if (secondWorkerIndex == -99) + { + secondWorkerIndex = i; + iCountedYou[i] = true; + } + + } + + } // i (part-time workers) + } + + // third: youngest child loop + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + if (persons[i].getPersonIsPreschoolChild() == 1 + || persons[i].getPersonIsStudentNonDriving() == 1 + || persons[i].getPersonIsStudentDriving() == 1) + { + + // check our indices + if (youngestChildIndex == -99) + { + youngestChildIndex = i; + youngestChildAge = persons[i].getAge(); + iCountedYou[i] = true; + } else + { + + // see if this child is younger than the one on record + int age = persons[i].getAge(); + if (age < youngestChildAge) + { + + // reset iCountedYou for previous child + iCountedYou[youngestChildIndex] = false; + + // set variables for this child + youngestChildIndex = i; + youngestChildAge = age; + iCountedYou[i] = true; + + } + } + + } // if person is child + + } // i (youngest child loop) + + // fourth: second youngest child loop (skip if youngest child is not + // filled) + if (youngestChildIndex != -99) + { + + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + if (persons[i].getPersonIsPreschoolChild() == 1 + || persons[i].getPersonIsStudentNonDriving() == 1 + || persons[i].getPersonIsStudentDriving() == 1) + { + + // check our indices + if (secondYoungestChildIndex == -99) + { + secondYoungestChildIndex = i; + secondYoungestChildAge = persons[i].getAge(); + iCountedYou[i] = true; + } else + { + + // see if this child is younger than the one on record + int age = persons[i].getAge(); + if (age < secondYoungestChildAge) + { + + // reset iCountedYou for previous child + iCountedYou[secondYoungestChildIndex] = false; + + // set variables for this child + secondYoungestChildIndex = i; + secondYoungestChildAge = age; + iCountedYou[i] = true; + + } + } + + } // if person is child + + } // i (second youngest child loop) + } + + // fifth: third youngest child loop (skip if second kid not included) + if (secondYoungestChildIndex != -99) + { + + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + if (persons[i].getPersonIsPreschoolChild() == 1 + || persons[i].getPersonIsStudentNonDriving() == 1 + || persons[i].getPersonIsStudentDriving() == 1) + { + + // check our indices + if (thirdYoungestChildIndex == -99) + { + thirdYoungestChildIndex = i; + thirdYoungestChildAge = persons[i].getAge(); + iCountedYou[i] = true; + } else + { + + // see if this child is younger than the one on record + int age = persons[i].getAge(); + if (age < thirdYoungestChildAge) + { + + // reset iCountedYou for previous child + iCountedYou[thirdYoungestChildIndex] = false; + + // set variables for this child + thirdYoungestChildIndex = i; + thirdYoungestChildAge = age; + iCountedYou[i] = true; + + } + } + + } // if person is child + + } // i (third youngest child loop) + } + + // assign any missing spots among the top 5 to random members + int cdapPersonIndex; + + Random hhRandom = household.getHhRandom(); + + int randomCount = household.getHhRandomCount(); + // when household.getHhRandom() was applied, the random count was + // incremented, assuming a random number would be drawn right away. + // so let's decrement by 1, then increment the count each time a random + // number is actually drawn in this method. + randomCount--; + + // first worker + cdapPersonIndex = 1; // persons and cdapPersonArray are 1-based + if (firstWorkerIndex == -99) + { + + int randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + while (iCountedYou[randomIndex] || randomIndex == 0) + { + randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + } + + cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; + iCountedYou[randomIndex] = true; + + } else + { + cdapPersonArray[cdapPersonIndex] = persons[firstWorkerIndex]; + } + + // second worker + cdapPersonIndex = 2; + if (secondWorkerIndex == -99) + { + + int randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + while (iCountedYou[randomIndex] || randomIndex == 0) + { + randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + } + + cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; + iCountedYou[randomIndex] = true; + + } else + { + cdapPersonArray[cdapPersonIndex] = persons[secondWorkerIndex]; + } + + // youngest child + cdapPersonIndex = 3; + if (youngestChildIndex == -99) + { + + int randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + while (iCountedYou[randomIndex] || randomIndex == 0) + { + randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + } + + cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; + iCountedYou[randomIndex] = true; + + } else + { + cdapPersonArray[cdapPersonIndex] = persons[youngestChildIndex]; + } + + // second youngest child + cdapPersonIndex = 4; + if (secondYoungestChildIndex == -99) + { + + int randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + while (iCountedYou[randomIndex] || randomIndex == 0) + { + randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + } + + cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; + iCountedYou[randomIndex] = true; + + } else + { + cdapPersonArray[cdapPersonIndex] = persons[secondYoungestChildIndex]; + } + + // third youngest child + cdapPersonIndex = 5; + if (thirdYoungestChildIndex == -99) + { + + int randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + while (iCountedYou[randomIndex] || randomIndex == 0) + { + randomIndex = (int) (hhRandom.nextDouble() * hhSize); + randomCount++; + } + + cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; + iCountedYou[randomIndex] = true; + + } else + { + cdapPersonArray[cdapPersonIndex] = persons[thirdYoungestChildIndex]; + } + + // fill spots outside the top 5 + cdapPersonIndex = 6; + for (int i = 1; i < persons.length; ++i) + { + + if (iCountedYou[i]) continue; + + cdapPersonArray[cdapPersonIndex] = persons[i]; + cdapPersonIndex++; + } + + household.setHhRandomCount(randomCount); + + } + + protected Person getCdapPerson(int persNum) + { + if (persNum < 1 || persNum > cdapPersonArray.length - 1) + { + logger.fatal(String.format("persNum value = %d is out of range for hhSize = %d", + persNum, cdapPersonArray.length - 1)); + throw new RuntimeException(); + } + + return cdapPersonArray[persNum]; + } + + /** + * Method returns an array of persons not modeled by the CDAP model (i.e. + * persons 6 to X, when ordered by the reOrderPersonsForCdap method + * + * @param personsModeledByCdap + * @return + */ + public Person[] getPersonsNotModeledByCdap(int personsModeledByCdap) + { + + // create a 1-based person array to be consistent + Person[] personArray = null; + if (cdapPersonArray.length > personsModeledByCdap + 1) personArray = new Person[cdapPersonArray.length + - personsModeledByCdap]; + else personArray = new Person[0]; + + for (int i = 1; i < personArray.length; ++i) + { + personArray[i] = cdapPersonArray[personsModeledByCdap + i]; + } + + return personArray; + + } + + /** + * Returns true if this CDAP person number (meaning the number of persons + * for the purposes of the CDAP model) is an adult; false if not. + * + * @param cdapPersonNumber + * @return + */ + public boolean isThisCdapPersonAnAdult(int cdapPersonNumber) + { + + if (cdapPersonArray[cdapPersonNumber].getPersonIsAdult() == 1) return true; + return false; + } + + public int getStudentTypeForThisCdapPerson(int cdapPersonNumber) + { + + Person p = cdapPersonArray[cdapPersonNumber]; + int type = 0; + if (p.getPersonIsPreschoolChild() == 1) type = 1; + else if (p.getPersonIsGradeSchool() == 1) type = 2; + else if (p.getPersonIsHighSchool() == 1) type = 3; + else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) type = 4; + else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) type = 5; + + return type; + } + + public int getStudentTypeForThisCdapPerson(Person p) + { + + int type = 0; + if (p.getPersonIsPreschoolChild() == 1) type = 1; + else if (p.getPersonIsGradeSchool() == 1) type = 2; + else if (p.getPersonIsHighSchool() == 1) type = 3; + else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) type = 4; + else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) type = 5; + + return type; + } + + public int getWorkLocationForThisCdapPerson(int cdapPersonNumber) + { + int loc = cdapPersonArray[cdapPersonNumber].getWorkLocation(); + if (loc > 0 && loc != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return loc; + else return 0; + } + + public int getSchoolLocationForThisCdapPerson(int cdapPersonNumber) + { + return cdapPersonArray[cdapPersonNumber].getUsualSchoolLocation(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java new file mode 100644 index 0000000..eb6ebe6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java @@ -0,0 +1,2039 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Random; +import java.util.StringTokenizer; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import umontreal.iro.lecuyer.probdist.LognormalDist; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.IndexSort; +import com.pb.common.util.ObjectUtil; +import com.pb.common.util.SeededRandom; + +/** + * @author Jim Hicks + * + * Class for managing household and person object data read from + * synthetic population files. + */ +public abstract class HouseholdDataManager + implements HouseholdDataManagerIf, Serializable +{ + + protected transient Logger logger = Logger.getLogger(HouseholdDataManager.class); + + public static final String PROPERTIES_SYNPOP_INPUT_HH = "PopulationSynthesizer.InputToCTRAMP.HouseholdFile"; + public static final String PROPERTIES_SYNPOP_INPUT_PERS = "PopulationSynthesizer.InputToCTRAMP.PersonFile"; + + public static final String RANDOM_SEED_NAME = "Model.Random.Seed"; + + public static final String OUTPUT_HH_DATA_FILE_TARGET = "outputHouseholdData.file"; + public static final String OUTPUT_PERSON_DATA_FILE_TARGET = "outputPersonData.file"; + + public static final String READ_UWSL_RESULTS_FILE = "read.uwsl.results"; + public static final String READ_UWSL_RESULTS_FILENAME = "read.uwsl.filename"; + public static final String READ_PRE_AO_RESULTS_FILE = "read.pre.ao.results"; + public static final String READ_PRE_AO_RESULTS_FILENAME = "read.pre.ao.filename"; + + public static final String PROPERTIES_DISTRIBUTED_TIME = "distributedTimeCoefficients"; + + + // HHID,household_serial_no,TAZ,MGRA,VEH,PERSONS,HWORKERS,HINCCAT1,HINC,UNITTYPE,HHT,BLDGSZ + public static final String HH_ID_FIELD_NAME = "HHID"; + public static final String HH_HOME_TAZ_FIELD_NAME = "TAZ"; + public static final String HH_HOME_MGRA_FIELD_NAME = "MGRA"; + public static final String HH_INCOME_CATEGORY_FIELD_NAME = "HINCCAT1"; + public static final String HH_INCOME_DOLLARS_FIELD_NAME = "HINC"; + public static final String HH_WORKERS_FIELD_NAME = "HWORKERS"; + public static final String HH_AUTOS_FIELD_NAME = "VEH"; + public static final String HH_SIZE_FIELD_NAME = "PERSONS"; + public static final String HH_TYPE_FIELD_NAME = "HHT"; + public static final String HH_BLDGSZ_FIELD_NAME = "BLDGSZ"; + public static final String HH_UNITTYPE_FIELD_NAME = "UNITTYPE"; + + // HHID,PERID,AGE,SEX,OCCCEN1,INDCEN,PEMPLOY,PSTUDENT,PTYPE,EDUC,GRADE + public static final String PERSON_HH_ID_FIELD_NAME = "HHID"; + public static final String PERSON_PERSON_ID_FIELD_NAME = "PERID"; + public static final String PERSON_AGE_FIELD_NAME = "AGE"; + public static final String PERSON_GENDER_FIELD_NAME = "SEX"; + public static final String PERSON_MILITARY_FIELD_NAME = "MILTARY"; + public static final String PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME = "PEMPLOY"; + public static final String PERSON_STUDENT_CATEGORY_FIELD_NAME = "PSTUDENT"; + public static final String PERSON_TYPE_CATEGORY_FIELD_NAME = "PTYPE"; + public static final String PERSON_EDUCATION_ATTAINMENT_FIELD_NAME = "EDUC"; + public static final String PERSON_GRADE_ENROLLED_FIELD_NAME = "GRADE"; + public static final String PERSON_OCCCEN1_FIELD_NAME = "OCCCEN1"; + public static final String PERSON_SOC_FIELD_NAME = "OCCSOC5"; + public static final String PERSON_INDCEN_FIELD_NAME = "INDCEN"; + + public static final String PERSON_TIMEFACTOR_WORK_FIELD_NAME = "timeFactorWork"; + public static final String PERSON_TIMEFACTOR_NONWORK_FIELD_NAME = "timeFactorNonWork"; + + public static final String PROPERTIES_HOUSEHOLD_TRACE_LIST = "Debug.Trace.HouseholdIdList"; + public static final String DEBUG_HHS_ONLY_KEY = "Process.Debug.HHs.Only"; + + private static final String PROPERTIES_MIN_VALUE_OF_TIME_KEY = "HouseholdManager.MinValueOfTime"; + private static final String PROPERTIES_MAX_VALUE_OF_TIME_KEY = "HouseholdManager.MaxValueOfTime"; + private static final String PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY = "HouseholdManager.MeanValueOfTime.Values"; + private static final String PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY = "HouseholdManager.MeanValueOfTime.Income.Limits"; + private static final String PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY = "HouseholdManager.HH.ValueOfTime.Multiplier.Under18"; + private static final String PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY = "HouseholdManager.Mean.ValueOfTime.Multiplier.Mu"; + private static final String PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY = "HouseholdManager.ValueOfTime.Lognormal.Sigma"; + + private HashMap schoolSegmentNameIndexMap; + private HashMap gsDistrictSegmentMap; + private HashMap hsDistrictSegmentMap; + private int[] mgraGsDistrict; + private int[] mgraHsDistrict; + + // these are not used for sandag; instead sandag uses distributed time coefficients read in the person file + protected float hhValueOfTimeMultiplierForPersonUnder18; + protected double meanValueOfTimeMultiplierBeforeLogForMu; + protected double valueOfTimeLognormalSigma; + protected float minValueOfTime; + protected float maxValueOfTime; + protected float[] meanValueOfTime; + protected int[] incomeDollarLimitsForValueOfTime; + protected LognormalDist[] valueOfTimeDistribution; + + protected HashMap propertyMap; + + protected String projectDirectory; + protected String outputHouseholdFileName; + protected String outputPersonFileName; + + protected ModelStructure modelStructure; + + protected TableDataSet hhTable; + protected TableDataSet personTable; + + protected HashSet householdTraceSet; + + protected Household[] hhs; + protected int[] hhIndexArray; + + protected int inputRandomSeed; + protected int numPeriods; + protected int firstPeriod; + + protected float sampleRate; + protected int sampleSeed; + + protected int maximumNumberOfHouseholdsPerFile = 0; + protected int numberOfHouseholdDiskObjectFiles = 0; + + protected MgraDataManager mgraManager; + protected TazDataManager tazManager; + + protected double[] percentHhsIncome100Kplus; + protected double[] percentHhsMultipleAutos; + + protected boolean readTimeFactors; + public HouseholdDataManager() + { + } + + /** + * Associate data in hh and person TableDataSets read from synthetic + * population files with Household objects and Person objects with + * Households. + */ + protected abstract void mapTablesToHouseholdObjects(); + + public String testRemote() + { + System.out.println("testRemote() called by remote process."); + return String.format("testRemote() method in %s called.", this.getClass() + .getCanonicalName()); + } + + public void setDebugHhIdsFromHashmap() + { + + householdTraceSet = new HashSet(); + + // get the household ids for which debug info is required + String householdTraceStringList = propertyMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); + + if (householdTraceStringList != null) + { + StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); + while (householdTokenizer.hasMoreTokens()) + { + String listValue = householdTokenizer.nextToken(); + int idValue = Integer.parseInt(listValue.trim()); + householdTraceSet.add(idValue); + } + } + + } + + public void readPopulationFiles(String inputHouseholdFileName, String inputPersonFileName) + { + + TimeCoefficientDistributions timeDistributions = new TimeCoefficientDistributions(); + timeDistributions.createTimeDistributions(propertyMap); + timeDistributions.appendTimeDistributionsOnPersonFile(propertyMap); + + // read synthetic population files + readHouseholdData(inputHouseholdFileName); + + readPersonData(inputPersonFileName); + } + + public void setModelStructure(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + } + + public void setupHouseholdDataManager(ModelStructure modelStructure, + String inputHouseholdFileName, String inputPersonFileName) + { + + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + setModelStructure(modelStructure); + readPopulationFiles(inputHouseholdFileName, inputPersonFileName); + + // Set the seed for the JVM default SeededRandom object - should only be + // used + // to set the order for the + // HH index array so that hhs can be processed in an arbitrary order as + // opposed to the order imposed by + // the synthetic population generator. + // The seed was set as a command line argument for the model run, or the + // default if no argument supplied + SeededRandom.setSeed(sampleSeed); + + // the seed read from the properties file controls seeding the Household + // object random number generator objects. + inputRandomSeed = Integer.parseInt(propertyMap.get(HouseholdDataManager.RANDOM_SEED_NAME)); + + // map synthetic population table data to objects to be used by CT-RAMP + mapTablesToHouseholdObjects(); + hhTable = null; + personTable = null; + + logPersonSummary(); + + setTraceHouseholdSet(); + + // if read pre-ao results flag is set, read the results file and + // populate the + // household object ao result field from these values. + String readPreAoResultsString = propertyMap.get(READ_PRE_AO_RESULTS_FILE); + if (readPreAoResultsString != null) + { + boolean readResults = Boolean.valueOf(readPreAoResultsString); + if (readResults) readPreAoResults(); + } + + // if read uwsl results flag is set, read the results file and populate + // the + // household object work/school location result fields from these + // values. + String readUwslResultsString = propertyMap.get(READ_UWSL_RESULTS_FILE); + if (readUwslResultsString != null) + { + boolean readResults = Boolean.valueOf(readUwslResultsString); + if (readResults) readWorkSchoolLocationResults(); + } + + //check if we want to read distributed time factors from the person file + String readTimeFactorsString = propertyMap.get(PROPERTIES_DISTRIBUTED_TIME); + if (readTimeFactorsString != null) + { + readTimeFactors = Boolean.valueOf(readTimeFactorsString); + logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); + } + + + + } + + public void setPropertyFileValues(HashMap propertyMap) + { + + String propertyValue = ""; + this.propertyMap = propertyMap; + + // save the project specific parameters in class attributes + this.projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + + outputHouseholdFileName = propertyMap + .get(CtrampApplication.PROPERTIES_OUTPUT_HOUSEHOLD_FILE); + outputPersonFileName = propertyMap.get(CtrampApplication.PROPERTIES_OUTPUT_PERSON_FILE); + + setDebugHhIdsFromHashmap(); + + propertyValue = propertyMap + .get(CtrampApplication.PROPERTIES_SCHEDULING_NUMBER_OF_TIME_PERIODS); + if (propertyValue == null) numPeriods = 0; + else numPeriods = Integer.parseInt(propertyValue); + + propertyValue = propertyMap.get(CtrampApplication.PROPERTIES_SCHEDULING_FIRST_TIME_PERIOD); + if (propertyValue == null) firstPeriod = 0; + else firstPeriod = Integer.parseInt(propertyValue); + + //check if we want to read distributed time factors from the person file + String readTimeFactorsString = propertyMap.get(PROPERTIES_DISTRIBUTED_TIME); + if (readTimeFactorsString != null) + { + readTimeFactors = Boolean.valueOf(readTimeFactorsString); + logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); + } + + } + + public int[] getRandomOrderHhIndexArray(int numHhs) + { + + Random myRandom = new Random(); + myRandom.setSeed(numHhs + 1); + + int[] data = new int[numHhs]; + for (int i = 0; i < numHhs; i++) + data[i] = (int) (10000000 * myRandom.nextDouble()); + + int[] index = IndexSort.indexSort(data); + + return index; + } + + // this is called at the end of UsualWorkSchoolLocation model step. + public void setUwslRandomCount(int iter) + { + + for (int r = 0; r < hhs.length; r++) + hhs[r].setUwslRandomCount(iter, hhs[r].getHhRandomCount()); + + } + + private void resetRandom(Household h, int count) + { + // get the household's Random + Random r = h.getHhRandom(); + + int seed = inputRandomSeed + h.getHhId(); + r.setSeed(seed); + + // select count Random draws to reset this household's Random to it's + // state + // prior to + // the model run for which model results were stored in + // HouseholdDataManager. + for (int i = 0; i < count; i++) + r.nextDouble(); + + // reset the randomCount for the household's Random + h.setHhRandomCount(count); + } + + public void resetPreAoRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // The Pre Auto Ownership model is the first model component, so + // reset + // counts to 0. + resetRandom(hhs[i], 0); + } + } + + public void resetUwslRandom(int iter) + { + for (int i = 0; i < hhs.length; i++) + { + // get the current random count for the end of the shadow price + // iteration + // passed in. + // this value was set at the end of UsualWorkSchoolLocation model + // step + // for the given iter. + // if < 0, random count should be set to the count at end of pre + // auto + // ownership. + int uwslCount = hhs[i].getPreAoRandomCount(); + if (iter >= 0) + { + uwslCount = hhs[i].getUwslRandomCount(iter); + } + + // draw uwslCount random numbers from the household's Random + resetRandom(hhs[i], uwslCount); + } + } + + public void resetAoRandom(int iter) + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Auto Ownership model from the + // Household + // object. + // this value was set at the end of UsualWorkSchoolLocation model + // step. + + int aoCount = hhs[i].getUwslRandomCount(iter); + + // draw aoCount random numbers from the household's Random + resetRandom(hhs[i], aoCount); + } + } + + public void resetTpRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Auto Ownership model from the + // Household + // object. + // this value was set at the end of UsualWorkSchoolLocation model + // step. + int tpCount = hhs[i].getAoRandomCount(); + + // draw aoCount random numbers from the household's Random + resetRandom(hhs[i], tpCount); + } + } + + public void resetFpRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Auto Ownership model from the + // Household + // object. + // this value was set at the end of UsualWorkSchoolLocation model + // step. + int fpCount = hhs[i].getTpRandomCount(); + + // draw aoCount random numbers from the household's Random + resetRandom(hhs[i], fpCount); + } + } + + public void resetIeRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Auto Ownership model from the + // Household + // object. + // this value was set at the end of UsualWorkSchoolLocation model + // step. + int ieCount = hhs[i].getFpRandomCount(); + + // draw aoCount random numbers from the household's Random + resetRandom(hhs[i], ieCount); + } + } + + public void resetCdapRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Coordinated Daily Activity Pattern + // model from the Household object. + // this value was set at the end of Auto Ownership model step. + int cdapCount = hhs[i].getIeRandomCount(); + + // draw cdapCount random numbers + resetRandom(hhs[i], cdapCount); + } + } + + public void resetImtfRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Individual Mandatory Tour + // Frequency + // model from the Household object. + // this value was set at the end of Coordinated Daily Activity + // Pattern + // model step. + int imtfCount = hhs[i].getCdapRandomCount(); + + // draw imtfCount random numbers + resetRandom(hhs[i], imtfCount); + } + } + + public void resetImtodRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Individual Mandatory Tour + // Departure and + // duration model from the Household object. + // this value was set at the end of Individual Mandatory Tour + // Frequency + // model step. + int imtodCount = hhs[i].getImtfRandomCount(); + + // draw imtodCount random numbers + resetRandom(hhs[i], imtodCount); + } + } + + public void resetJtfRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Joint Tour Frequency model from + // the + // Household object. + // this value was set at the end of Individual Mandatory departure + // time + // Choice model step. + int jtfCount = hhs[i].getImtodRandomCount(); + + // draw jtfCount random numbers + resetRandom(hhs[i], jtfCount); + } + } + + public void resetJtlRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Joint Tour Location model from the + // Household object. + // this value was set at the end of Joint Tour Frequency model step. + int jtlCount = hhs[i].getJtfRandomCount(); + + // draw jtlCount random numbers + resetRandom(hhs[i], jtlCount); + } + } + + public void resetJtodRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Joint Tour Departure and duration + // model + // from the Household object. + // this value was set at the end of Joint Tour Location model step. + int jtodCount = hhs[i].getJtlRandomCount(); + + // draw jtodCount random numbers + resetRandom(hhs[i], jtodCount); + } + } + + public void resetInmtfRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Individual non-mandatory tour + // frequency + // model from the Household object. + // this value was set at the end of Joint Tour Departure and + // duration + // model step. + int inmtfCount = hhs[i].getJtodRandomCount(); + + // draw inmtfCount random numbers + resetRandom(hhs[i], inmtfCount); + } + } + + public void resetInmtlRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Individual non-mandatory tour + // location + // model from the Household object. + // this value was set at the end of Individual non-mandatory tour + // frequency model step. + int inmtlCount = hhs[i].getInmtfRandomCount(); + + // draw inmtlCount random numbers + resetRandom(hhs[i], inmtlCount); + } + } + + public void resetInmtodRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to Individual non-mandatory tour + // departure + // and duration model from the Household object. + // this value was set at the end of Individual non-mandatory tour + // location model step. + int inmtodCount = hhs[i].getInmtlRandomCount(); + + // draw inmtodCount random numbers + resetRandom(hhs[i], inmtodCount); + } + } + + public void resetAwfRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to At-work Subtour Frequency model + // from + // the Household object. + // this value was set at the end of Individual Non-Mandatory Tour + // Departure and duration model step. + int awfCount = hhs[i].getInmtodRandomCount(); + + // draw awfCount random numbers + resetRandom(hhs[i], awfCount); + } + } + + public void resetAwlRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to At-work Subtour Location Choice + // model + // from the Household object. + // this value was set at the end of At-work Subtour Frequency model + // step. + int awlCount = hhs[i].getAwfRandomCount(); + + // draw awlCount random numbers + resetRandom(hhs[i], awlCount); + } + } + + public void resetAwtodRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to At-work Subtour Time-of-day and + // mode + // choice model from the Household object. + // this value was set at the end of At-work Subtour Location Choice + // model + // step. + int awtodCount = hhs[i].getAwlRandomCount(); + + // draw awtodCount random numbers + resetRandom(hhs[i], awtodCount); + } + } + + public void resetStfRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to stop frequency model from the + // Household + // object. + // this value was set at the end of At-work Subtour Time-of-day and + // mode + // choice model step. + int stfCount = hhs[i].getAwtodRandomCount(); + + // draw stfCount random numbers + resetRandom(hhs[i], stfCount); + } + } + + public void resetStlRandom() + { + for (int i = 0; i < hhs.length; i++) + { + // get the current count prior to stop location model from the + // Household + // object. + // this value was set at the end of stop frequency model step. + int stlCount = hhs[i].getStfRandomCount(); + + // draw stlCount random numbers + resetRandom(hhs[i], stlCount); + } + } + + /** + * Sets the HashSet used to trace households for debug purposes and sets the + * debug switch for each of the listed households. Also sets + */ + public void setTraceHouseholdSet() + { + + // loop through the households in the set and set the trace switches + for (int i = 0; i < hhs.length; i++) + hhs[i].setDebugChoiceModels(false); + + for (int id : householdTraceSet) + { + int index = hhIndexArray[id]; + hhs[index].setDebugChoiceModels(true); + } + + } + + /** + * Sets the sample rate used to run the model for a portion of the + * households. + * + * @param sampleRate + * , proportion of total households for which to run the model + * [0.0, 1.0]. + */ + public void setHouseholdSampleRate(float sampleRate, int sampleSeed) + { + this.sampleRate = sampleRate; + this.sampleSeed = sampleSeed; + } + + public void setHhArray(Household[] hhArray) + { + hhs = hhArray; + } + + public void setHhArray(Household[] tempHhs, int startIndex) + { + // long startTime = System.currentTimeMillis(); + // logger.info(String.format("start setHhArray for startIndex=%d, startTime=%d.", + // startIndex, + // startTime)); + + synchronized(hhs) { + for (int i = 0; i < tempHhs.length; i++) + { + hhs[startIndex + i] = tempHhs[i]; + } + } + // long endTime = System.currentTimeMillis(); + // logger.info(String.format( + // "end setHhArray for startIndex=%d, endTime=%d, elapsed=%d millisecs.", + // startIndex, + // endTime, (endTime - startTime))); + } + + /** + * return the array of Household objects holding the synthetic population + * and choice model outcomes. + * + * @return hhs + */ + public Household[] getHhArray() + { + return hhs; + } + + public Household[] getHhArray(int first, int last) + { + // long startTime = System.currentTimeMillis(); + // logger.info(String.format("start getHhArray for first=%d, last=%d, startTime=%d.", + // first, last, startTime)); + Household[] tempHhs = new Household[last - first + 1]; + for (int i = 0; i < tempHhs.length; i++) + { + tempHhs[i] = hhs[first + i]; + } + // long endTime = System.currentTimeMillis(); + // logger.info(String.format( + // "end getHhArray for first=%d, last=%d, endTime=%d, elapsed=%d millisecs.", + // first, last, endTime, (endTime - startTime))); + return tempHhs; + } + + public int getArrayIndex(int hhId) + { + int i = hhIndexArray[hhId]; + return i; + } + + /** + * return the number of household objects read from the synthetic + * population. + * + * @return + */ + public int getNumHouseholds() + { + // hhs is dimesioned to number of households + 1. + return hhs.length; + } + + /** + * set walk segment (0-none, 1-short, 2-long walk to transit access) for the + * origin for this tour + */ + public int getInitialOriginWalkSegment(int taz, double randomNumber) + { + // double[] proportions = tazDataManager.getZonalWalkPercentagesForTaz( + // taz + // ); + // return ChoiceModelApplication.getMonteCarloSelection(proportions, + // randomNumber); + return 0; + } + + private void readHouseholdData(String inputHouseholdFileName) + { + + // construct input household file name from properties file values + String fileName = projectDirectory + "/" + inputHouseholdFileName; + + try + { + logger.info("reading popsyn household data file."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + hhTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading synthetic household data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + } + + private void readPersonData(String inputPersonFileName) + { + + // construct input person file name from properties file values + String fileName = projectDirectory + "/" + inputPersonFileName; + + try + { + logger.info("reading popsyn person data file."); + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet("," + reader.getDelimSet()); + personTable = reader.readFile(new File(fileName)); + } catch (Exception e) + { + logger.fatal(String + .format("Exception occurred reading synthetic person data file: %s into TableDataSet object.", + fileName)); + throw new RuntimeException(e); + } + + } + + public void logPersonSummary() + { + + HashMap> summaryResults; + + summaryResults = new HashMap>(); + + for (int i = 0; i < hhs.length; ++i) + { + + Household household = hhs[i]; + + Person[] personArray = household.getPersons(); + for (int j = 1; j < personArray.length; ++j) + { + Person person = personArray[j]; + String personType = person.getPersonType(); + + String employmentStatus = person.getPersonEmploymentCategory(); + String studentStatus = person.getPersonStudentCategory(); + int age = person.getAge(); + int ageCategory; + if (age <= 5) + { + ageCategory = 0; + } else if (age <= 15) + { + ageCategory = 1; + } else if (age <= 18) + { + ageCategory = 2; + } else if (age <= 24) + { + ageCategory = 3; + } else if (age <= 44) + { + ageCategory = 4; + } else if (age <= 64) + { + ageCategory = 5; + } else + { + ageCategory = 6; + } + + if (summaryResults.containsKey(personType)) + { + // have person type + if (summaryResults.get(personType).containsKey(employmentStatus)) + { + // have employment category + summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; + } else + { + // don't have employment category + summaryResults.get(personType).put(employmentStatus, new int[7]); + summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; + } + if (summaryResults.get(personType).containsKey(studentStatus)) + { + // have student category + summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; + } else + { + // don't have student category + summaryResults.get(personType).put(studentStatus, new int[7]); + summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; + } + } else + { + // don't have person type + summaryResults.put(personType, new HashMap()); + summaryResults.get(personType).put(studentStatus, new int[7]); + summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; + summaryResults.get(personType).put(employmentStatus, new int[7]); + summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; + } + } + } + String headerRow = String.format("%5s\t", "Age\t"); + for (String empCategory : Person.EMPLOYMENT_CATEGORY_NAME_ARRAY) + { + headerRow += String.format("%16s\t", empCategory); + } + for (String stuCategory : Person.STUDENT_CATEGORY_NAME_ARRAY) + { + headerRow += String.format("%16s\t", stuCategory); + } + String[] ageCategories = {"0-5", "6-15", "16-18", "19-24", "25-44", "45-64", "65+"}; + + for (String personType : summaryResults.keySet()) + { + + logger.info("Summary for person type: " + personType); + + logger.info(headerRow); + String row = ""; + + HashMap personTypeSummary = summaryResults.get(personType); + + for (int j = 0; j < ageCategories.length; ++j) + { + row = String.format("%5s\t", ageCategories[j]); + for (String empCategory : Person.EMPLOYMENT_CATEGORY_NAME_ARRAY) + { + if (personTypeSummary.containsKey(empCategory)) + { + row += String.format("%16d\t", personTypeSummary.get(empCategory)[j]); + } else row += String.format("%16d\t", 0); + } + for (String stuCategory : Person.STUDENT_CATEGORY_NAME_ARRAY) + { + if (personTypeSummary.containsKey(stuCategory)) + { + row += String.format("%16d\t", personTypeSummary.get(stuCategory)[j]); + } else row += String.format("%16d\t", 0); + } + logger.info(row); + } + + } + + } + + public int[][] getTourPurposePersonsByHomeMgra(String[] purposeList) + { + + int maxMgra = mgraManager.getMaxMgra(); + int[][] personsWithMandatoryPurpose = new int[purposeList.length][maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < hhs.length; r++) + { + + Person[] persons = hhs[r].getPersons(); + + int homeMgra = hhs[r].getHhMgra(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + int purposeIndex = -1; + try + { + + if (person.getPersonIsWorker() == 1) + { + + purposeIndex = person.getWorkLocationSegmentIndex(); + personsWithMandatoryPurpose[purposeIndex][homeMgra]++; + + } + + if (person.getPersonIsPreschoolChild() == 1 + || person.getPersonIsStudentDriving() == 1 + || person.getPersonIsStudentNonDriving() == 1 + || person.getPersonIsUniversityStudent() == 1) + { + + purposeIndex = person.getSchoolLocationSegmentIndex(); + personsWithMandatoryPurpose[purposeIndex][homeMgra]++; + + } + + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught summing workers/students by origin zone for household table record r=%d.", + r)); + throw e; + } + + } + + } // r (households) + + return personsWithMandatoryPurpose; + + } + + public double[] getPercentHhsIncome100Kplus() + { + return percentHhsIncome100Kplus; + } + + public double[] getPercentHhsMultipleAutos() + { + return percentHhsMultipleAutos; + } + + public void computeTransponderChoiceTazPercentArrays() + { + + PrintWriter out = null; + try + { + out = new PrintWriter(new BufferedWriter(new FileWriter(new File( + "./transpChoiceArrays.csv")))); + } catch (IOException e) + { + // TODO Auto-generated catch block + e.printStackTrace(); + } + + int maxTaz = tazManager.maxTaz; + int[] numHhs = new int[maxTaz + 1]; + + // for percent of households in TAZ with income > $100K + percentHhsIncome100Kplus = new double[maxTaz + 1]; + // for percent og households in TAZ with multiple autos + percentHhsMultipleAutos = new double[maxTaz + 1]; + + for (int r = 0; r < getNumHouseholds(); r++) + { + int homeMgra = hhs[r].getHhMgra(); + int homeTaz = mgraManager.getTaz(homeMgra); + numHhs[homeTaz]++; + if (hhs[r].getIncomeInDollars() > 100000) percentHhsIncome100Kplus[homeTaz]++; + if (hhs[r].getAutosOwned() > 1) percentHhsMultipleAutos[homeTaz]++; + } + + out.println("taz,numHhsTaz,numHhsIncome100KplusTaz,numHhsMultipleAutosTaz,proportionHhsIncome100KplusTaz,proportionHhsMultipleAutosTaz"); + + for (int i = 0; i <= maxTaz; i++) + { + + out.print(i + "," + numHhs[i] + "," + percentHhsIncome100Kplus[i] + "," + + percentHhsMultipleAutos[i]); + if (numHhs[i] > 0) + { + percentHhsIncome100Kplus[i] /= numHhs[i]; + percentHhsMultipleAutos[i] /= numHhs[i]; + out.println("," + percentHhsIncome100Kplus[i] + "," + percentHhsMultipleAutos[i]); + } else + { + out.println("," + 0.0 + "," + 0.0); + } + } + + out.close(); + } + + public int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap) + { + + int maxMgra = mgraManager.getMaxMgra(); + + int[][] workersByHomeMgra = new int[segmentValueIndexMap.size()][maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < getNumHouseholds(); r++) + { + + Person[] persons = hhs[r].getPersons(); + + int homeMgra = hhs[r].getHhMgra(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + if (person.getPersonIsFullTimeWorker() == 1 + || person.getPersonIsPartTimeWorker() == 1) + { + + int occup = -1; + int segmentIndex = -1; + try + { + + occup = person.getPersPecasOccup(); + segmentIndex = segmentValueIndexMap.get(occup); + workersByHomeMgra[segmentIndex][homeMgra]++; + + } catch (Exception e) + { + logger.error( + String.format( + "exception caught summing workers by origin MGRA for household table record r=%d, person=%d, homeMgra=%d, occup=%d, segmentIndex=%d.", + r, person.getPersonNum(), homeMgra, occup, segmentIndex), e); + throw new RuntimeException(); + } + + } + + } + + } // r (households) + + return workersByHomeMgra; + + } + + public int[][] getStudentsByHomeMgra() + { + + int maxMgra = mgraManager.getMaxMgra(); + + // there are 5 school types - preschool, K-8, HS, University with + // typical + // students, University with non-typical students. + int[][] studentsByHomeMgra = new int[schoolSegmentNameIndexMap.size()][maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < getNumHouseholds(); r++) + { + + Person[] persons = hhs[r].getPersons(); + + int homeMgra = hhs[r].getHhMgra(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + if (person.getPersonIsPreschoolChild() == 1 + || person.getPersonIsStudentNonDriving() == 1 + || person.getPersonIsStudentDriving() == 1 + || person.getPersonIsUniversityStudent() == 1) + { + + int segmentIndex = -1; + try + { + + if (person.getPersonIsPreschoolChild() == 1) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); + } else if (person.getPersonIsGradeSchool() == 1) + { + int gsDistrict = mgraGsDistrict[homeMgra]; + segmentIndex = gsDistrictSegmentMap.get(gsDistrict); + } else if (person.getPersonIsHighSchool() == 1) + { + int hsDistrict = mgraHsDistrict[homeMgra]; + segmentIndex = hsDistrictSegmentMap.get(hsDistrict); + } else if (person.getPersonIsUniversityStudent() == 1 + && person.getAge() < 30) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); + } else if (person.getPersonIsUniversityStudent() == 1 + && person.getAge() >= 30) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); + } + + // if person type is a student but segment index is -1, + // the person is not enrolled; assume home schooled and + // don't add to sum by home mgra + if (segmentIndex >= 0) studentsByHomeMgra[segmentIndex][homeMgra]++; + + } catch (Exception e) + { + logger.error( + String.format( + "exception caught summing students by origin MGRA for household table record r=%d, person=%d, homeMgra=%d, segmentIndex=%d.", + r, person.getPersonNum(), homeMgra, segmentIndex), e); + throw new RuntimeException(); + } + + } + + } + + } // r (households) + + return studentsByHomeMgra; + + } + + public int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString) + { + + // dimension the array + int maxMgra = mgraManager.getMaxMgra(); + int[] individualNonMandatoryTours = new int[maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + int count = 0; + for (int r = 0; r < hhs.length; r++) + { + + Person[] persons = hhs[r].getPersons(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + ArrayList it = person.getListOfIndividualNonMandatoryTours(); + + try + { + + if (it.size() == 0) continue; + + for (Tour tour : it) + { + // increment the segment count if it's the right purpose + String tourPurpose = tour.getTourPurpose(); + if (purposeString.startsWith(tourPurpose)) + { + int homeMgra = hhs[r].getHhMgra(); + individualNonMandatoryTours[homeMgra]++; + count++; + } + } + + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught counting number of individualNonMandatory tours for purpose: %s, for household table record r=%d, personNum=%d.", + purposeString, r, person.getPersonNum())); + throw e; + } + + } + + } // r (households) + + return individualNonMandatoryTours; + } + + public int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap) + { + + // dimension the array + int maxMgra = mgraManager.getMaxMgra(); + int destMgra = 0; + + int[][] workTours = new int[segmentValueIndexMap.size()][maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < getNumHouseholds(); r++) + { + + Person[] persons = hhs[r].getPersons(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + int occup = -1; + int segmentIndex = -1; + try + { + + if (person.getPersonIsWorker() == 1) + { + + destMgra = person.getWorkLocation(); + + if (destMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + occup = person.getPersPecasOccup(); + segmentIndex = segmentValueIndexMap.get(occup); + workTours[segmentIndex][destMgra]++; + } + + } + + } catch (Exception e) + { + logger.error( + String.format( + "exception caught summing workers by work location MGRA for household table record r=%d, person=%d, workMgra=%d, occup=%d, segmentIndex=%d.", + r, person.getPersonNum(), destMgra, occup, segmentIndex), e); + throw new RuntimeException(); + } + + } + + } // r (households) + + return workTours; + + } + + public int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap) + { + + int destMgra = 0; + + int[] workAtHome = new int[segmentValueIndexMap.size()]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < getNumHouseholds(); r++) + { + + Person[] persons = hhs[r].getPersons(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + int occup = -1; + int segmentIndex = -1; + try + { + + if (person.getPersonIsWorker() == 1) + { + + destMgra = person.getWorkLocation(); + + if (destMgra == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + occup = person.getPersPecasOccup(); + segmentIndex = segmentValueIndexMap.get(occup); + workAtHome[segmentIndex]++; + } + + } + + } catch (Exception e) + { + logger.error( + String.format( + "exception caught summing workers by work location MGRA for household table record r=%d, person=%d, workMgra=%d, occup=%d, segmentIndex=%d.", + r, person.getPersonNum(), destMgra, occup, segmentIndex), e); + throw new RuntimeException(); + } + + } + + } // r (households) + + return workAtHome; + + } + + public void setSchoolDistrictMappings(HashMap segmentNameIndexMap, + int[] mgraGsDist, int[] mgraHsDist, HashMap gsDistSegMap, + HashMap hsDistSegMap) + { + + schoolSegmentNameIndexMap = segmentNameIndexMap; + gsDistrictSegmentMap = gsDistSegMap; + hsDistrictSegmentMap = hsDistSegMap; + mgraGsDistrict = mgraGsDist; + mgraHsDistrict = mgraHsDist; + } + + public int[][] getSchoolToursByDestMgra() + { + + // dimension the array + int maxMgra = mgraManager.getMaxMgra(); + int destMgra = 0; + + int[][] schoolTours = new int[schoolSegmentNameIndexMap.size()][maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + for (int r = 0; r < getNumHouseholds(); r++) + { + + Person[] persons = hhs[r].getPersons(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + destMgra = person.getPersonSchoolLocationZone(); + if (destMgra == 0) continue; + + int segmentIndex = -1; + try + { + + if (person.getPersonIsPreschoolChild() == 1) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); + } else if (person.getPersonIsGradeSchool() == 1) + { + int gsDistrict = mgraGsDistrict[destMgra]; + segmentIndex = gsDistrictSegmentMap.get(gsDistrict); + } else if (person.getPersonIsHighSchool() == 1) + { + int hsDistrict = mgraHsDistrict[destMgra]; + segmentIndex = hsDistrictSegmentMap.get(hsDistrict); + } else if (person.getPersonIsUniversityStudent() == 1 && person.getAge() < 30) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); + } else if (person.getPersonIsUniversityStudent() == 1 && person.getAge() >= 30) + { + segmentIndex = schoolSegmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); + } + + // if person type is a student but segment index is -1, the + // person is not enrolled; assume home schooled and don't + // add to sum by home mgra + if (segmentIndex >= 0) schoolTours[segmentIndex][destMgra]++; + + } catch (Exception e) + { + logger.error( + String.format( + "exception caught summing students by origin MGRA for household table record r=%d, person=%d, schoolMgra=%d, segmentIndex=%d.", + r, person.getPersonNum(), destMgra, segmentIndex), e); + throw new RuntimeException(); + } + + } + + } // r (households) + + return schoolTours; + + } + + public int[] getJointToursByHomeZoneSubZone(String purposeString) + { + + // dimension the array + int maxMgra = mgraManager.getMaxMgra(); + + int[] jointTours = new int[maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + int count = 0; + for (int r = 0; r < hhs.length; r++) + { + + try + { + + Tour[] jt = hhs[r].getJointTourArray(); + + if (jt == null) continue; + + for (int i = 0; i < jt.length; i++) + { + // increment the segment count if it's the right purpose + if (jt[i].getTourPurpose().equalsIgnoreCase(purposeString)) + { + int homeMgra = hhs[r].getHhMgra(); + jointTours[homeMgra]++; + count++; + } + } + + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught counting number of joint tours for purpose: %s, for household table record r=%d.", + purposeString, r)); + throw e; + } + + } // r (households) + + return jointTours; + } + + public int[] getAtWorkSubtoursByWorkMgra(String purposeString) + { + + // dimension the array + int maxMgra = mgraManager.getMaxMgra(); + + int[] subtours = new int[maxMgra + 1]; + + // hhs is dimesioned to number of households + 1. + int count = 0; + for (int r = 0; r < hhs.length; r++) + { + + Person[] persons = hhs[r].getPersons(); + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + ArrayList subtourList = person.getListOfAtWorkSubtours(); + + try + { + + if (subtourList.size() == 0) continue; + + for (Tour tour : subtourList) + { + // increment the segment count if it's the right purpose + String tourPurpose = tour.getTourPurpose(); + if (tourPurpose.startsWith(purposeString)) + { + int workZone = tour.getTourOrigMgra(); + subtours[workZone]++; + count++; + } + } + + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught counting number of at-work subtours for purpose: %s, for household table record r=%d, personNum=%d, count=0%d.", + purposeString, r, person.getPersonNum(), count)); + throw e; + } + + } + + } // r (households) + + return subtours; + } + + public void readWorkSchoolLocationResults() + { + + String[] headings = {"HHID", "HomeMGRA", "Income", "PersonID", "PersonNum", "PersonType", + "PersonAge", "EmploymentCategory", "StudentCategory", "WorkSegment", + "SchoolSegment", "WorkLocation", "WorkLocationDistance", "WorkLocationLogsum", + "SchoolLocation", "SchoolLocationDistance", "SchoolLocationLogsum"}; + + String wsLocResultsFileName = propertyMap.get(READ_UWSL_RESULTS_FILENAME); + + // open the input stream + String delimSet = ","; + BufferedReader uwslStream = null; + String fileName = projectDirectory + wsLocResultsFileName; + try + { + uwslStream = new BufferedReader(new FileReader(new File(fileName))); + } catch (FileNotFoundException e) + { + e.printStackTrace(); + System.exit(-1); + } + + // first parse the results file field names from the first record and + // associate column position with fields used in model + HashMap indexHeadingMap = new HashMap(); + + String line = ""; + try + { + line = uwslStream.readLine(); + } catch (IOException e) + { + e.printStackTrace(); + System.exit(-1); + } + StringTokenizer st = new StringTokenizer(line, delimSet); + int col = 0; + while (st.hasMoreTokens()) + { + String label = st.nextToken(); + for (String heading : headings) + { + if (heading.equalsIgnoreCase(label)) + { + indexHeadingMap.put(col, heading); + break; + } + } + col++; + } + + Household hh = null; + Person person = null; + + try + { + + while ((line = uwslStream.readLine()) != null) + { + + // set the line number for the next line in the sample of + // households + // int sortedSampleIndex = + // sortedIndices[sortedSample[sampleCount]]; + + // get the household id and personNum first, before parsing + // other + // fields. Skip to next record if not in the sample. + col = 0; + int id = -1; + int personNum = -1; + int workLocation = -1; + int schoolLocation = -1; + st = new StringTokenizer(line, delimSet); + while (st.hasMoreTokens()) + { + String fieldValue = st.nextToken(); + if (indexHeadingMap.containsKey(col)) + { + String fieldName = indexHeadingMap.get(col++); + + if (fieldName.equalsIgnoreCase("HHID")) + { + id = Integer.parseInt(fieldValue); + + int index = getArrayIndex(id); + hh = hhs[index]; + } else if (fieldName.equalsIgnoreCase("PersonNum")) + { + personNum = Integer.parseInt(fieldValue); + person = hh.getPerson(personNum); + } else if (fieldName.equalsIgnoreCase("WorkSegment")) + { + int workSegment = Integer.parseInt(fieldValue); + person.setWorkLocationSegmentIndex(workSegment); + } else if (fieldName.equalsIgnoreCase("workLocation")) + { + workLocation = Integer.parseInt(fieldValue); + person.setWorkLocation(workLocation); + } else if (fieldName.equalsIgnoreCase("WorkLocationDistance")) + { + float distance = Float.parseFloat(fieldValue); + person.setWorkLocDistance(distance); + } else if (fieldName.equalsIgnoreCase("WorkLocationLogsum")) + { + float logsum = Float.parseFloat(fieldValue); + person.setWorkLocLogsum(logsum); + } else if (fieldName.equalsIgnoreCase("SchoolSegment")) + { + int schoolSegment = Integer.parseInt(fieldValue); + person.setSchoolLocationSegmentIndex(schoolSegment); + } else if (fieldName.equalsIgnoreCase("SchoolLocation")) + { + schoolLocation = Integer.parseInt(fieldValue); + person.setSchoolLoc(schoolLocation); + } else if (fieldName.equalsIgnoreCase("SchoolLocationDistance")) + { + float distance = Float.parseFloat(fieldValue); + person.setSchoolLocDistance(distance); + } else if (fieldName.equalsIgnoreCase("SchoolLocationLogsum")) + { + float logsum = Float.parseFloat(fieldValue); + person.setSchoolLocLogsum(logsum); + break; + } + + } else + { + col++; + } + + } + + } + + } catch (NumberFormatException e) + { + e.printStackTrace(); + System.exit(-1); + } catch (IOException e) + { + e.printStackTrace(); + System.exit(-1); + } + + } + + public void readPreAoResults() + { + + String[] headings = {"HHID", "AO"}; + + String preAoResultsFileName = propertyMap.get(READ_PRE_AO_RESULTS_FILENAME); + + // open the input stream + String delimSet = ","; + BufferedReader inStream = null; + String fileName = projectDirectory + preAoResultsFileName; + try + { + inStream = new BufferedReader(new FileReader(new File(fileName))); + } catch (FileNotFoundException e) + { + e.printStackTrace(); + System.exit(-1); + } + + // first parse the results file field names from the first record and + // associate column position with fields used in model + HashMap indexHeadingMap = new HashMap(); + + String line = ""; + try + { + line = inStream.readLine(); + } catch (IOException e) + { + e.printStackTrace(); + System.exit(-1); + } + StringTokenizer st = new StringTokenizer(line, delimSet); + int col = 0; + while (st.hasMoreTokens()) + { + String label = st.nextToken(); + for (String heading : headings) + { + if (heading.equalsIgnoreCase(label)) + { + indexHeadingMap.put(col, heading); + break; + } + } + col++; + } + + Household hh = null; + + try + { + + while ((line = inStream.readLine()) != null) + { + + // set the line number for the next line in the sample of + // households + // int sortedSampleIndex = + // sortedIndices[sortedSample[sampleCount]]; + + // get the household id first, before parsing other fields. Skip + // to + // next record if not in the sample. + col = 0; + int id = -1; + int ao = -1; + st = new StringTokenizer(line, delimSet); + while (st.hasMoreTokens()) + { + String fieldValue = st.nextToken(); + if (indexHeadingMap.containsKey(col)) + { + String fieldName = indexHeadingMap.get(col++); + + if (fieldName.equalsIgnoreCase("HHID")) + { + id = Integer.parseInt(fieldValue); + + int index = getArrayIndex(id); + hh = hhs[index]; + } else if (fieldName.equalsIgnoreCase("AO")) + { + ao = Integer.parseInt(fieldValue); + // pass in the ao model alternative number to this + // method + hh.setHhAutos(ao + 1); + break; + } + + } else + { + col++; + } + + } + + } + + } catch (NumberFormatException e) + { + e.printStackTrace(); + System.exit(-1); + } catch (IOException e) + { + e.printStackTrace(); + System.exit(-1); + } + + } + + public long getBytesUsedByHouseholdArray() + { + + long numBytes = 0; + for (int i = 0; i < hhs.length; i++) + { + Household hh = hhs[i]; + long size = ObjectUtil.sizeOf(hh); + numBytes += size; + } + + return numBytes; + } + + /** + * Assigns each individual person their own value of time, drawing from a + * lognormal distribution as a function of income. + */ + protected void setDistributedValuesOfTime() + { + + // read in values from property file + setValueOfTimePropertyFileValues(); + + // set up the probability distributions + for (int i = 0; i < valueOfTimeDistribution.length; i++) + { + double mu = Math.log(meanValueOfTime[i] * meanValueOfTimeMultiplierBeforeLogForMu); + valueOfTimeDistribution[i] = new LognormalDist(mu, valueOfTimeLognormalSigma); + } + + for (int i = 0; i < hhs.length; ++i) + { + Household household = hhs[i]; + + // each HH gets a VOT for consistency + double rnum = household.getHhRandom().nextDouble(); + int incomeCategory = getIncomeIndexForValueOfTime(household.getIncomeInDollars()); + double hhValueOfTime = valueOfTimeDistribution[incomeCategory - 1].inverseF(rnum); + + // constrain to logical min and max values + if (hhValueOfTime < minValueOfTime) hhValueOfTime = minValueOfTime; + if (hhValueOfTime > maxValueOfTime) hhValueOfTime = maxValueOfTime; + + // adults get the full value, and children 2/3 (1-based) + Person[] personArray = household.getPersons(); + for (int j = 1; j < personArray.length; ++j) + { + Person person = personArray[j]; + + int age = person.getAge(); + if (age < 18) person.setValueOfTime((float) hhValueOfTime + * hhValueOfTimeMultiplierForPersonUnder18); + else person.setValueOfTime((float) hhValueOfTime); + } + } + } + + /** + * Sets additional properties specific to MTC, included distributed + * value-of-time information + */ + private void setValueOfTimePropertyFileValues() + { + + boolean errorFlag = false; + String propertyValue = ""; + + propertyValue = propertyMap.get(PROPERTIES_MIN_VALUE_OF_TIME_KEY); + if (propertyValue == null) + { + logger.error("property file key missing: " + PROPERTIES_MIN_VALUE_OF_TIME_KEY + + ", not able to set min value of time value."); + errorFlag = true; + } else minValueOfTime = Float.parseFloat(propertyValue); + + propertyValue = propertyMap.get(PROPERTIES_MAX_VALUE_OF_TIME_KEY); + if (propertyValue == null) + { + logger.error("property file key missing: " + PROPERTIES_MAX_VALUE_OF_TIME_KEY + + ", not able to set max value of time value."); + errorFlag = true; + } else maxValueOfTime = Float.parseFloat(propertyValue); + + // mean values of time by income category are specified as a + // "comma-sparated" list of float values + // the number of mean values in the lsit determines the number of income + // categories for value of time + // the number of upper limit income dollar values is expected to be + // number of mean values - 1. + int numIncomeCategories = -1; + String meanValueOfTimesPropertyValue = propertyMap + .get(PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY); + if (meanValueOfTimesPropertyValue == null) + { + logger.error("property file key missing: " + PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY + + ", not able to set mean value of time values."); + errorFlag = true; + } else + { + + ArrayList valueList = new ArrayList(); + StringTokenizer valueTokenizer = new StringTokenizer(meanValueOfTimesPropertyValue, ","); + while (valueTokenizer.hasMoreTokens()) + { + String listValue = valueTokenizer.nextToken(); + float value = Float.parseFloat(listValue.trim()); + valueList.add(value); + } + + numIncomeCategories = valueList.size(); + meanValueOfTime = new float[numIncomeCategories]; + valueOfTimeDistribution = new LognormalDist[numIncomeCategories]; + + for (int i = 0; i < numIncomeCategories; i++) + meanValueOfTime[i] = valueList.get(i); + } + + // read the upper limit values for value of time income ranges. + // there should be exactly 1 less than the number of mean value of time + // values - any other value is an error. + String valueOfTimeIncomesPropertyValue = propertyMap + .get(PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY); + if (valueOfTimeIncomesPropertyValue == null) + { + logger.error("property file key missing: " + + PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY + + ", not able to set upper limits for value of time income ranges."); + errorFlag = true; + } else + { + + ArrayList valueList = new ArrayList(); + StringTokenizer valueTokenizer = new StringTokenizer(valueOfTimeIncomesPropertyValue, + ","); + while (valueTokenizer.hasMoreTokens()) + { + String listValue = valueTokenizer.nextToken(); + int value = Integer.parseInt(listValue.trim()); + valueList.add(value); + } + + int numIncomeValues = valueList.size(); + if (numIncomeValues != (numIncomeCategories - 1)) + { + Exception e = new RuntimeException(); + logger.error("an error occurred reading properties file values for distributed value of time calculations."); + logger.error("the mean value of time values property=" + + meanValueOfTimesPropertyValue + " specifies " + numIncomeCategories + + " mean values, thus " + numIncomeCategories + " income ranges."); + logger.error("the value of time income range values property=" + + valueOfTimeIncomesPropertyValue + " specifies " + numIncomeValues + + " income range limit values."); + logger.error("there should be exactly " + (numIncomeCategories - 1) + + " income range limit values for " + numIncomeCategories + + " mean value of time values.", e); + System.exit(-1); + } + + // set the income dollar value upper limits for value of time income + // ranges + incomeDollarLimitsForValueOfTime = new int[numIncomeValues + 1]; + for (int i = 0; i < numIncomeValues; i++) + incomeDollarLimitsForValueOfTime[i] = valueList.get(i); + + incomeDollarLimitsForValueOfTime[numIncomeValues] = Integer.MAX_VALUE; + } + + propertyValue = propertyMap.get(PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY); + if (propertyValue == null) + { + logger.error("property file key missing: " + + PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY + + ", not able to set hh value of time multiplier for kids in hh under age 18."); + errorFlag = true; + } else hhValueOfTimeMultiplierForPersonUnder18 = Float.parseFloat(propertyValue); + + propertyValue = propertyMap.get(PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY); + if (propertyValue == null) + { + logger.error("property file key missing: " + + PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY + + ", not able to set lognormal distribution mu parameter multiplier."); + errorFlag = true; + } else meanValueOfTimeMultiplierBeforeLogForMu = Float.parseFloat(propertyValue); + + propertyValue = propertyMap.get(PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY); + if (propertyValue == null) + { + logger.error("property file key missing: " + + PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY + + ", not able to set lognormal distribution sigma parameter."); + errorFlag = true; + } else valueOfTimeLognormalSigma = Float.parseFloat(propertyValue); + + if (errorFlag) + { + Exception e = new RuntimeException(); + logger.error( + "errors occurred reading properties file values for distributed value of time calculations.", + e); + System.exit(-1); + } + + } + + private int getIncomeIndexForValueOfTime(int incomeInDollars) + { + int returnValue = -1; + for (int i = 0; i < incomeDollarLimitsForValueOfTime.length; i++) + { + if (incomeInDollars < incomeDollarLimitsForValueOfTime[i]) + { + // return a 1s based index value + returnValue = i + 1; + break; + } + } + + return returnValue; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java new file mode 100644 index 0000000..c31368f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java @@ -0,0 +1,137 @@ +package org.sandag.abm.ctramp; + +import java.util.HashMap; + +/** + * @author Jim Hicks + * + * Class for managing household and person object data read from + * synthetic population files. + */ +public interface HouseholdDataManagerIf +{ + + String testRemote(); + + void setPropertyFileValues(HashMap propertyMap); + + void setDebugHhIdsFromHashmap(); + + void computeTransponderChoiceTazPercentArrays(); + + double[] getPercentHhsIncome100Kplus(); + + double[] getPercentHhsMultipleAutos(); + + int[] getRandomOrderHhIndexArray(int numHhs); + + int getArrayIndex(int hhId); + + void setHhArray(Household[] hhs); + + void setHhArray(Household[] tempHhs, int startIndex); + + void setSchoolDistrictMappings(HashMap segmentNameIndexMap, int[] mgraGsDist, + int[] mgraHsDist, HashMap gsDistSegMap, + HashMap hsDistSegMap); + + void setupHouseholdDataManager(ModelStructure modelStructure, String inputHouseholdFileName, + String inputPersonFileName); + + int[][] getTourPurposePersonsByHomeMgra(String[] purposeList); + + int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap); + + int[][] getStudentsByHomeMgra(); + + int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap); + + int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap); + + int[][] getSchoolToursByDestMgra(); + + int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString); + + int[] getJointToursByHomeMgra(String purposeString); + + int[] getAtWorkSubtoursByWorkMgra(String purposeString); + + void logPersonSummary(); + + void setUwslRandomCount(int iter); + + void resetUwslRandom(int iter); + + void resetPreAoRandom(); + + void resetAoRandom(int iter); + + void resetFpRandom(); + + void resetCdapRandom(); + + void resetImtfRandom(); + + void resetImtodRandom(); + + void resetAwfRandom(); + + void resetAwlRandom(); + + void resetAwtodRandom(); + + void resetJtfRandom(); + + void resetJtlRandom(); + + void resetJtodRandom(); + + void resetInmtfRandom(); + + void resetInmtlRandom(); + + void resetInmtodRandom(); + + void resetStfRandom(); + + void resetStlRandom(); + + /** + * Sets the HashSet used to trace households for debug purposes and sets the + * debug switch for each of the listed households. Also sets + */ + void setTraceHouseholdSet(); + + /** + * Sets the HashSet used to trace households for debug purposes and sets the + * debug switch for each of the listed households. Also sets + */ + void setHouseholdSampleRate(float sampleRate, int sampleSeed); + + /** + * return the array of Household objects holding the synthetic population + * and choice model outcomes. + * + * @return hhs + */ + Household[] getHhArray(); + + Household[] getHhArray(int firstHhIndex, int lastHhIndex); + + /** + * return the number of household objects read from the synthetic + * population. + * + * @return + */ + int getNumHouseholds(); + + /** + * set walk segment (0-none, 1-short, 2-long walk to transit access) for the + * origin for this tour + */ + int getInitialOriginWalkSegment(int taz, double randomNumber); + + long getBytesUsedByHouseholdArray(); + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java new file mode 100644 index 0000000..e98bdab --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java @@ -0,0 +1,373 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +/** + * @author Jim Hicks + * + * Class for managing household and person object data read from + * synthetic population files. + */ +public class HouseholdDataManagerRmi + implements HouseholdDataManagerIf, Serializable +{ + + UtilRmi remote; + String connectString; + + public HouseholdDataManagerRmi(String hostname, int port, String className) + { + + connectString = String.format("//%s:%d/%s", hostname, port, className); + remote = new UtilRmi(connectString); + + } + + public void setPropertyFileValues(HashMap propertyMap) + { + Object[] objArray = {propertyMap}; + remote.method("setPropertyFileValues", objArray); + } + + public void setDebugHhIdsFromHashmap() + { + Object[] objArray = {}; + remote.method("setDebugHhIdsFromHashmap", objArray); + } + + public void setupHouseholdDataManager(ModelStructure modelStructure, + String inputHouseholdFileName, String inputPersonFileName) + { + Object[] objArray = {modelStructure, inputHouseholdFileName, inputPersonFileName}; + remote.method("setupHouseholdDataManager", objArray); + } + + public void setSchoolDistrictMappings(HashMap segmentNameIndexMap, + int[] mgraGsDist, int[] mgraHsDist, HashMap gsDistSegMap, + HashMap hsDistSegMap) + { + Object[] objArray = {segmentNameIndexMap, mgraGsDist, mgraHsDist, gsDistSegMap, + hsDistSegMap}; + remote.method("setSchoolDistrictMappings", objArray); + } + + public void computeTransponderChoiceTazPercentArrays() + { + Object[] objArray = {}; + remote.method("computeTransponderChoiceTazPercentArrays", objArray); + } + + public double[] getPercentHhsIncome100Kplus() + { + Object[] objArray = {}; + return (double[]) remote.method("getPercentHhsIncome100Kplus", objArray); + } + + public double[] getPercentHhsMultipleAutos() + { + Object[] objArray = {}; + return (double[]) remote.method("getPercentHhsMultipleAutos", objArray); + } + + public void logPersonSummary() + { + Object[] objArray = {}; + remote.method("logPersonSummary", objArray); + } + + public int getArrayIndex(int hhId) + { + Object[] objArray = {hhId}; + return (Integer) remote.method("getArrayIndex", objArray); + } + + public int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap) + { + Object[] objArray = {segmentValueIndexMap}; + return (int[]) remote.method("getWorksAtHomeBySegment", objArray); + } + + public int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap) + { + Object[] objArray = {segmentValueIndexMap}; + return (int[][]) remote.method("getWorkToursByDestMgra", objArray); + } + + public int[][] getSchoolToursByDestMgra() + { + Object[] objArray = {}; + return (int[][]) remote.method("getSchoolToursByDestMgra", objArray); + } + + public int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap) + { + Object[] objArray = {segmentValueIndexMap}; + return (int[][]) remote.method("getWorkersByHomeMgra", objArray); + } + + public int[][] getStudentsByHomeMgra() + { + Object[] objArray = {}; + return (int[][]) remote.method("getStudentsByHomeMgra", objArray); + } + + public int[][] getTourPurposePersonsByHomeMgra(String[] purposeList) + { + Object[] objArray = {purposeList}; + return (int[][]) remote.method("getTourPurposePersonsByHomeMgra", objArray); + } + + public int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString) + { + Object[] objArray = {purposeString}; + return (int[]) remote.method("getIndividualNonMandatoryToursByHomeMgra", objArray); + } + + public int[] getJointToursByHomeMgra(String purposeString) + { + Object[] objArray = {purposeString}; + return (int[]) remote.method("getJointToursByHomeMgra", objArray); + } + + public int[] getAtWorkSubtoursByWorkMgra(String purposeString) + { + Object[] objArray = {purposeString}; + return (int[]) remote.method("getAtWorkSubtoursByWorkMgra", objArray); + } + + public String testRemote() + { + Object[] objArray = {}; + return (String) remote.method("testRemote", objArray); + } + + public void mapTablesToHouseholdObjects() + { + Object[] objArray = {}; + remote.method("mapTablesToHouseholdObjects", objArray); + } + + public void writeResultData() + { + Object[] objArray = {}; + remote.method("writeResultData", objArray); + } + + public int[] getRandomOrderHhIndexArray(int numHhs) + { + Object[] objArray = {numHhs}; + return (int[]) remote.method("getRandomOrderHhIndexArray", objArray); + } + + /** + * set the hh id for which debugging info from choice models applied to this + * household will be logged if debug logging. + */ + public void setDebugHouseholdId(int debugHhId, boolean value) + { + Object[] objArray = {debugHhId, value}; + remote.method("setDebugHouseholdId", objArray); + } + + /** + * Sets the HashSet used to trace households for debug purposes and sets the + * debug switch for each of the listed households. Also sets + */ + public void setTraceHouseholdSet() + { + Object[] objArray = {}; + remote.method("setTraceHouseholdSet", objArray); + } + + public void setHouseholdSampleRate(float sampleRate, int sampleSeed) + { + Object[] objArray = {sampleRate, sampleSeed}; + remote.method("setHouseholdSampleRate", objArray); + } + + public void resetUwslRandom(int iter) + { + Object[] objArray = {iter}; + remote.method("resetUwslRandom", objArray); + } + + public void resetPreAoRandom() + { + Object[] objArray = {}; + remote.method("resetPreAoRandom", objArray); + } + + public void setUwslRandomCount(int iter) + { + Object[] objArray = {iter}; + remote.method("setUwslRandomCount", objArray); + } + + public void resetAoRandom(int iter) + { + Object[] objArray = {iter}; + remote.method("resetAoRandom", objArray); + } + + public void resetFpRandom() + { + Object[] objArray = {}; + remote.method("resetFpRandom", objArray); + } + + public void resetCdapRandom() + { + Object[] objArray = {}; + remote.method("resetCdapRandom", objArray); + } + + public void resetImtfRandom() + { + Object[] objArray = {}; + remote.method("resetImtfRandom", objArray); + } + + public void resetImtodRandom() + { + Object[] objArray = {}; + remote.method("resetImtodRandom", objArray); + } + + public void resetAwfRandom() + { + Object[] objArray = {}; + remote.method("resetAwfRandom", objArray); + } + + public void resetAwlRandom() + { + Object[] objArray = {}; + remote.method("resetAwlRandom", objArray); + } + + public void resetAwtodRandom() + { + Object[] objArray = {}; + remote.method("resetAwtodRandom", objArray); + } + + public void resetJtfRandom() + { + Object[] objArray = {}; + remote.method("resetJtfRandom", objArray); + } + + public void resetJtlRandom() + { + Object[] objArray = {}; + remote.method("resetJtlRandom", objArray); + } + + public void resetJtodRandom() + { + Object[] objArray = {}; + remote.method("resetJtodRandom", objArray); + } + + public void resetInmtfRandom() + { + Object[] objArray = {}; + remote.method("resetInmtfRandom", objArray); + } + + public void resetInmtlRandom() + { + Object[] objArray = {}; + remote.method("resetInmtlRandom", objArray); + } + + public void resetInmtodRandom() + { + Object[] objArray = {}; + remote.method("resetInmtodRandom", objArray); + } + + public void resetStfRandom() + { + Object[] objArray = {}; + remote.method("resetStfRandom", objArray); + } + + public void resetStlRandom() + { + Object[] objArray = {}; + remote.method("resetStlRandom", objArray); + } + + /** + * return the array of Household objects holding the synthetic population + * and choice model outcomes. + * + * @return hhs + */ + public Household[] getHhArray() + { + Object[] objArray = {}; + return (Household[]) remote.method("getHhArray", objArray); + } + + public Household[] getHhArray(int first, int last) + { + Object[] objArray = {first, last}; + return (Household[]) remote.method("getHhArray", objArray); + } + + public void setHhArray(Household[] hhs) + { + Object[] objArray = {hhs}; + remote.method("setHhArray", objArray); + } + + public void setHhArray(Household[] tempHhs, int startIndex) + { + Object[] objArray = {tempHhs, startIndex}; + remote.method("setHhArray", objArray); + } + + /** + * return the array of Household objects holding the synthetic population + * and choice model outcomes. + * + * @return hhs + */ + public int[] getHhIndexArray() + { + Object[] objArray = {}; + return (int[]) remote.method("getHhIndexArray", objArray); + } + + /** + * return the number of household objects read from the synthetic + * population. + * + * @return number of households in synthetic population + */ + public int getNumHouseholds() + { + Object[] objArray = {}; + return (Integer) remote.method("getNumHouseholds", objArray); + } + + /** + * set walk segment (0-none, 1-short, 2-long walk to transit access) for the + * origin for this tour + */ + public int getInitialOriginWalkSegment(int taz, double randomNumber) + { + Object[] objArray = {taz, randomNumber}; + return (Integer) remote.method("getInitialOriginWalkSegment", objArray); + } + + public long getBytesUsedByHouseholdArray() + { + Object[] objArray = {}; + return (Long) remote.method("getBytesUsedByHouseholdArray", objArray); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java new file mode 100644 index 0000000..4a304b6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java @@ -0,0 +1,1871 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.IOException; +import java.io.PrintWriter; +import java.sql.Connection; +import java.sql.PreparedStatement; +import java.sql.SQLException; +import java.sql.Statement; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedList; +import java.util.List; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesDMU; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +// import com.pb.common.util.ObjectUtil; + +/** + * @author crf
+ * Started: Dec 31, 2008 11:46:36 AM + */ +public class HouseholdDataWriter +{ + + private transient Logger logger = Logger.getLogger(HouseholdDataWriter.class); + + private static final String PROPERTIES_HOUSEHOLD_DATA_FILE = "Results.HouseholdDataFile"; + private static final String PROPERTIES_PERSON_DATA_FILE = "Results.PersonDataFile"; + private static final String PROPERTIES_INDIV_TOUR_DATA_FILE = "Results.IndivTourDataFile"; + private static final String PROPERTIES_JOINT_TOUR_DATA_FILE = "Results.JointTourDataFile"; + private static final String PROPERTIES_INDIV_TRIP_DATA_FILE = "Results.IndivTripDataFile"; + private static final String PROPERTIES_JOINT_TRIP_DATA_FILE = "Results.JointTripDataFile"; + + private static final String PROPERTIES_HOUSEHOLD_TABLE = "Results.HouseholdTable"; + private static final String PROPERTIES_PERSON_TABLE = "Results.PersonTable"; + private static final String PROPERTIES_INDIV_TOUR_TABLE = "Results.IndivTourTable"; + private static final String PROPERTIES_JOINT_TOUR_TABLE = "Results.JointTourTable"; + private static final String PROPERTIES_INDIV_TRIP_TABLE = "Results.IndivTripTable"; + private static final String PROPERTIES_JOINT_TRIP_TABLE = "Results.JointTripTable"; + + private static final String PROPERTIES_WRITE_LOGSUMS = "Results.WriteLogsums"; + + private static final int NUM_WRITE_PACKETS = 2000; + + private final String intFormat = "%d"; + private final String floatFormat = "%f"; + private final String doubleFormat = "%f"; + private final String fileStringFormat = "%s"; + private final String databaseStringFormat = "'%s'"; + private String stringFormat = fileStringFormat; + + private boolean saveUtilsProbsFlag = false; + private boolean writeLogsums = false; + private int setNA = -1; + + + private HashMap rbMap; + + private MandatoryAccessibilitiesDMU dmu; + private UtilityExpressionCalculator autoSkimUEC; + private IndexValues iv; + private MgraDataManager mgraManager; + + private ModelStructure modelStructure; + private int iteration; + + private HashMap purposeIndexNameMap; + + public HouseholdDataWriter(HashMap rbMap, ModelStructure modelStructure, + int iteration) + { + logger.info("Writing data structures to files."); + this.modelStructure = modelStructure; + this.iteration = iteration; + this.rbMap = rbMap; + + // create a UEC to get highway distance traveled for tours + String uecFileName = rbMap.get("acc.mandatory.uec.file"); + int dataPage = Integer.parseInt(rbMap.get("acc.mandatory.data.page")); + int autoSkimPage = Integer.parseInt(rbMap.get("acc.mandatory.auto.page")); + File uecFile = new File(uecFileName); + dmu = new MandatoryAccessibilitiesDMU(); + autoSkimUEC = new UtilityExpressionCalculator(uecFile, autoSkimPage, dataPage, rbMap, dmu); + iv = new IndexValues(); + mgraManager = MgraDataManager.getInstance(rbMap); + + purposeIndexNameMap = this.modelStructure.getIndexPrimaryPurposeNameMap(); + + // default is to not save the tour mode choice utils and probs for each + // tour + String saveUtilsProbsString = rbMap + .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); + if (saveUtilsProbsString != null) + { + if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; + } + + String writeLogsumsString = rbMap.get(PROPERTIES_WRITE_LOGSUMS); + writeLogsums = Boolean.valueOf(writeLogsumsString); + + } + + // NOTE - this method should not be called simultaneously with the file one + // one + // as the string format is changed + public void writeDataToDatabase(HouseholdDataManagerIf householdData, String dbFileName) + { + logger.info("Writing data structures to database."); + long t = System.currentTimeMillis(); + stringFormat = databaseStringFormat; + writeData(householdData, new DatabaseDataWriter(dbFileName)); + float delta = ((Long) (System.currentTimeMillis() - t)).floatValue() / 60000.0f; + logger.info("Finished writing data structures to database (" + delta + " minutes)."); + } + + // NOTE - this method should not be called simultaneously with the database + // one + // one as the string format is changed + public void writeDataToFiles(HouseholdDataManagerIf householdData) + { + logger.info("Writing data structures to csv file."); + stringFormat = fileStringFormat; + FileDataWriter fdw = new FileDataWriter(); + writeData(householdData, fdw); + } + + private void writeData(HouseholdDataManagerIf householdDataManager, DataWriter writer) + { + int hhid = 0; + int persNum = 0; + int tourid = 0; + try + { + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + long maxSize = 0; + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (Household hh : householdArray) + { + if (hh == null) continue; + hhid = hh.getHhId(); + + // long size = ObjectUtil.sizeOf(hh); + // if (size > maxSize) maxSize = size; + + writer.writeHouseholdData(formHouseholdDataEntry(hh)); + for (Person p : hh.getPersons()) + { + if (p == null) continue; + persNum = p.getPersonNum(); + + writer.writePersonData(formPersonDataEntry(p)); + for (Tour t : p.getListOfWorkTours()) + writeIndivTourData(t, writer); + for (Tour t : p.getListOfSchoolTours()) + writeIndivTourData(t, writer); + for (Tour t : p.getListOfIndividualNonMandatoryTours()) + writeIndivTourData(t, writer); + for (Tour t : p.getListOfAtWorkSubtours()) + writeIndivTourData(t, writer); + } + Tour[] jointTours = hh.getJointTourArray(); + if (jointTours != null) for (Tour t : jointTours) + { + if (t == null) continue; + writeJointTourData(t, writer); + } + } + } + + // logger.info("max size for all Household objects after writing output files is " + // + maxSize + " bytes."); + + } catch (RuntimeException e) + { + logger.error(String.format("error writing hh=%d, persNum=%d", hhid, persNum), e); + throw new RuntimeException(); + } finally + { + writer.finishActions(); + } + } + + private void writeIndivTourData(Tour t, DataWriter writer) + { + writer.writeIndivTourData(formIndivTourDataEntry(t)); + + Stop[] outboundStops = t.getOutboundStops(); + if (outboundStops != null) + { + for (int i = 0; i < outboundStops.length; i++) + { + writer.writeIndivTripData(formIndivTripDataEntry(outboundStops[i])); + } + } else + { + writer.writeIndivTripData(formTourAsIndivTripDataEntry(t, false)); + } + + Stop[] inboundStops = t.getInboundStops(); + if (inboundStops != null) + { + for (Stop s : inboundStops) + writer.writeIndivTripData(formIndivTripDataEntry(s)); + } else + { + writer.writeIndivTripData(formTourAsIndivTripDataEntry(t, true)); + } + + } + + private void writeJointTourData(Tour t, DataWriter writer) + { + writer.writeJointTourData(formJointTourDataEntry(t)); + + Stop[] outboundStops = t.getOutboundStops(); + if (outboundStops != null) + { + for (Stop s : outboundStops) + writer.writeJointTripData(formJointTripDataEntry(s)); + } else + { + writer.writeJointTripData(formTourAsJointTripDataEntry(t, false)); + } + + Stop[] inboundStops = t.getInboundStops(); + if (inboundStops != null) + { + for (Stop s : inboundStops) + writer.writeJointTripData(formJointTripDataEntry(s)); + } else + { + writer.writeJointTripData(formTourAsJointTripDataEntry(t, true)); + } + + } + + private String string(int value) + { + return String.format(intFormat, value); + } + + private String string(float value) + { + return String.format(floatFormat, value); + } + + private String string(double value) + { + return String.format(doubleFormat, value); + } + + private String string(String value) + { + return String.format(stringFormat, value); + } + + private List formHouseholdColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("home_mgra"); + data.add("income"); + data.add("autos"); + data.add("HVs"); + data.add("AVs"); + data.add("transponder"); + data.add("cdap_pattern"); + data.add("out_escort_choice"); + data.add("inb_escort_choice"); + data.add("jtf_choice"); + + if(writeLogsums){ + data.add("aoLogsum"); + data.add("transponderLogsum"); + data.add("cdapLogsum"); + data.add("jtfLogsum"); + } + + return data; + } + + private List formHouseholdColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + + if(writeLogsums){ + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + } + return data; + } + + private List formHouseholdDataEntry(Household hh) + { + List data = new LinkedList(); + data.add(string(hh.getHhId())); + data.add(string(hh.getHhMgra())); + data.add(string(hh.getIncomeInDollars())); + data.add(string(hh.getAutosOwned())); + data.add(string(hh.getConventionalVehicles())); + data.add(string(hh.getAutomatedVehicles())); + data.add(string(hh.getTpChoice())); + data.add(string(hh.getCoordinatedDailyActivityPattern())); + data.add(string(hh.getOutboundEscortChoice())); + data.add(string(hh.getInboundEscortChoice())); + data.add(string(hh.getJointTourFreqChosenAlt())); + + if(writeLogsums){ + data.add(string(hh.getAutoOwnershipLogsum())); + data.add(string(hh.getTransponderLogsum())); + data.add(string(hh.getCdapLogsum())); + data.add(string(hh.getJtfLogsum())); + } + return data; + } + + private List formPersonColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("person_id"); + data.add("person_num"); + data.add("age"); + data.add("gender"); + data.add("type"); + data.add("value_of_time"); + data.add("activity_pattern"); + data.add("imf_choice"); + data.add("inmf_choice"); + data.add("fp_choice"); + data.add("reimb_pct"); + data.add("tele_choice"); + data.add("ie_choice"); + data.add("timeFactorWork"); + data.add("timeFactorNonWork"); + + if(writeLogsums){ + data.add("wfhLogsum"); + data.add("wlLogsum"); + data.add("slLogsum"); + data.add("fpLogsum"); + data.add("tcLogsum"); + data.add("ieLogsum"); + data.add("cdapLogsum"); + data.add("imtfLogsum"); + data.add("inmtfLogsum"); + } + + return data; + } + + private List formPersonColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + + if(writeLogsums){ + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + + } + return data; + } + + private List formPersonDataEntry(Person p) + { + List data = new LinkedList(); + data.add(string(p.getHouseholdObject().getHhId())); + data.add(string(p.getPersonId())); + data.add(string(p.getPersonNum())); + data.add(string(p.getAge())); + data.add(string(p.getPersonIsMale() == 1 ? "m" : "f")); + data.add(string(p.getPersonType())); + data.add(string(p.getValueOfTime())); + data.add(string(p.getCdapActivity())); + data.add(string(p.getImtfChoice())); + data.add(string(p.getInmtfChoice())); + data.add(string(p.getFreeParkingAvailableResult())); + data.add(string(p.getParkingReimbursement())); + data.add(string(p.getTelecommuteChoice())); + data.add(string(p.getInternalExternalTripChoiceResult())); + data.add(string(p.getTimeFactorWork())); + data.add(string(p.getTimeFactorNonWork())); + + if(writeLogsums){ + data.add(string(p.getWorksFromHomeLogsum())); + data.add(string(p.getWorkLocationLogsum())); + data.add(string(p.getSchoolLocationLogsum())); + data.add(string(p.getParkingProvisionLogsum())); + data.add(string(p.getTelecommuteLogsum())); + data.add(string(p.getIeLogsum())); + data.add(string(p.getCdapLogsum())); + data.add(string(p.getImtfLogsum())); + data.add(string(p.getInmtfLogsum())); + + } + return data; + } + + private List formIndivTourColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("person_id"); + data.add("person_num"); + data.add("person_type"); + data.add("tour_id"); + data.add("tour_category"); + data.add("tour_purpose"); + data.add("orig_mgra"); + data.add("dest_mgra"); + data.add("start_period"); + data.add("end_period"); + data.add("tour_mode"); + data.add("av_avail"); + data.add("tour_distance"); + data.add("atWork_freq"); + data.add("num_ob_stops"); + data.add("num_ib_stops"); + data.add("valueOfTime"); + + data.add("escort_type_out"); + data.add("escort_type_in"); + data.add("driver_num_out"); + data.add("driver_num_in"); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + for (int i = 1; i <= numModeAlts; i++) + { + String colName = String.format("util_%d", i); + data.add(colName); + } + + for (int i = 1; i <= numModeAlts; i++) + { + String colName = String.format("prob_%d", i); + data.add(colName); + } + } + + if(writeLogsums){ + data.add("timeOfDayLogsum"); + data.add("tourModeLogsum"); + data.add("subtourFreqLogsum"); + data.add("tourDestinationLogsum"); + data.add("stopFreqLogsum"); + + int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; + for(int i = 1; i<= numStopAlts;++i){ + String colName = String.format("outStopDCLogsum_%d", i); + data.add(colName); + } + for(int i = 1; i<= numStopAlts;++i){ + String colName = String.format("inbStopDCLogsum_%d", i); + data.add(colName); + } + } + + return data; + } + + private List formJointTourColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("tour_id"); + data.add("tour_category"); + data.add("tour_purpose"); + data.add("tour_composition"); + data.add("tour_participants"); + data.add("orig_mgra"); + data.add("dest_mgra"); + data.add("start_period"); + data.add("end_period"); + data.add("tour_mode"); + data.add("av_avail"); + data.add("tour_distance"); + data.add("num_ob_stops"); + data.add("num_ib_stops"); + data.add("valueOfTime"); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + for (int i = 1; i <= numModeAlts; i++) + { + String colName = String.format("util_%d", i); + data.add(colName); + } + + for (int i = 1; i <= numModeAlts; i++) + { + String colName = String.format("prob_%d", i); + data.add(colName); + } + } + if(writeLogsums){ + data.add("timeOfDayLogsum"); + data.add("tourModeLogsum"); + data.add("subtourFreqLogsum"); + data.add("tourDestinationLogsum"); + data.add("stopFreqLogsum"); + + int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; + for(int i = 1; i<= numStopAlts;++i){ + String colName = String.format("outStopDCLogsum_%d", i); + data.add(colName); + } + for(int i = 1; i<= numStopAlts;++i){ + String colName = String.format("inbStopDCLogsum_%d", i); + data.add(colName); + } + } + + return data; + } + + private List formIndivTourColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + for (int i = 1; i <= numModeAlts; i++) + { + data.add(SqliteDataTypes.REAL); + } + + for (int i = 1; i <= numModeAlts; i++) + { + data.add(SqliteDataTypes.REAL); + } + } + if(writeLogsums){ + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + + int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; + for(int i = 1; i<= numStopAlts;++i){ + data.add(SqliteDataTypes.REAL); + } + for(int i = 1; i<= numStopAlts;++i){ + data.add(SqliteDataTypes.REAL); + } + } + + + return data; + } + + private List formJointTourColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + for (int i = 1; i <= numModeAlts; i++) + { + data.add(SqliteDataTypes.REAL); + } + + for (int i = 1; i <= numModeAlts; i++) + { + data.add(SqliteDataTypes.REAL); + } + } + + if(writeLogsums){ + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + + int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; + for(int i = 1; i<= numStopAlts;++i){ + data.add(SqliteDataTypes.REAL); + } + for(int i = 1; i<= numStopAlts;++i){ + data.add(SqliteDataTypes.REAL); + } + } + return data; + } + + private List formIndivTourDataEntry(Tour t) + { + + List data = new LinkedList(); + data.add(string(t.getHhId())); + data.add(string(t.getPersonObject().getPersonId())); + data.add(string(t.getPersonObject().getPersonNum())); + data.add(string(t.getPersonObject().getPersonTypeNumber())); + data.add(string(t.getTourId())); + data.add(string(t.getTourCategory())); + data.add(string(t.getTourPurpose())); + data.add(string(t.getTourOrigMgra())); + data.add(string(t.getTourDestMgra())); + data.add(string(t.getTourDepartPeriod())); + data.add(string(t.getTourArrivePeriod())); + data.add(string(t.getTourModeChoice())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + data.add(string(calculateDistancesForAllMgras(t.getTourOrigMgra(), t.getTourDestMgra()))); + data.add(string(t.getSubtourFreqChoice())); + data.add(string(t.getNumOutboundStops() == 0 ? 0 : t.getNumOutboundStops() - 1)); + data.add(string(t.getNumInboundStops() == 0 ? 0 : t.getNumInboundStops() - 1)); + data.add(string(t.getValueOfTime())); + + data.add(string(t.getEscortTypeOutbound())); + data.add(string(t.getEscortTypeInbound())); + data.add(string(t.getDriverPnumOutbound())); + data.add(string(t.getDriverPnumInbound())); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + float[] utils = t.getTourModalUtilities(); + + if (utils != null){ + + for (int i = 0; i < utils.length; i++) + data.add(string(utils[i])); + for (int i = utils.length; i < numModeAlts; i++) + data.add("-999"); + + }else{ + for(int i =0;i outboundStopDCLogsums = t.getOutboundStopDestinationLogsums(); + for(int i = 0; i inboundStopDCLogsums = t.getInboundStopDestinationLogsums(); + for(int i = 0; i formJointTourDataEntry(Tour t) + { + List data = new LinkedList(); + data.add(string(t.getHhId())); + data.add(string(t.getTourId())); + data.add(string(t.getTourCategory())); + data.add(string(t.getTourPurpose())); + data.add(string(t.getJointTourComposition())); + data.add(string(formTourParticipationEntry(t))); + data.add(string(t.getTourOrigMgra())); + data.add(string(t.getTourDestMgra())); + data.add(string(t.getTourDepartPeriod())); + data.add(string(t.getTourArrivePeriod())); + data.add(string(t.getTourModeChoice())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + data.add(string(calculateDistancesForAllMgras(t.getTourOrigMgra(), t.getTourDestMgra()))); + data.add(string(t.getNumOutboundStops() == 0 ? 0 : t.getNumOutboundStops() - 1)); + data.add(string(t.getNumInboundStops() == 0 ? 0 : t.getNumInboundStops() - 1)); + data.add(string(t.getValueOfTime())); + + if (saveUtilsProbsFlag) + { + int numModeAlts = modelStructure.getMaxTourModeIndex(); + float[] utils = t.getTourModalUtilities(); + + int dummy = 0; + if (utils == null) dummy = 1; + + for (int i = 0; i < utils.length; i++) + data.add(string(utils[i])); + for (int i = utils.length; i < numModeAlts; i++) + data.add("-999"); + + float[] probs = t.getTourModalProbabilities(); + for (int i = 0; i < probs.length; i++) + data.add(string(probs[i])); + for (int i = probs.length; i < numModeAlts; i++) + data.add("0.0"); + } + + if(writeLogsums){ + data.add(string(t.getTimeOfDayLogsum())); + data.add(string(t.getTourModeLogsum())); + data.add(string(t.getSubtourFreqLogsum())); + data.add(string(t.getTourDestinationLogsum())); + data.add(string(t.getStopFreqLogsum())); + + int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; + ArrayList outboundStopDCLogsums = t.getOutboundStopDestinationLogsums(); + for(int i = 0; i inboundStopDCLogsums = t.getInboundStopDestinationLogsums(); + for(int i = 0; i formIndivTripColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("person_id"); + data.add("person_num"); + data.add("tour_id"); + data.add("stop_id"); + data.add("inbound"); + data.add("tour_purpose"); + data.add("orig_purpose"); + data.add("dest_purpose"); + data.add("orig_mgra"); + data.add("dest_mgra"); + data.add("parking_mgra"); + data.add("stop_period"); + data.add("trip_mode"); + data.add("av_avail"); + data.add("trip_board_tap"); + data.add("trip_alight_tap"); + data.add("set"); + data.add("tour_mode"); + data.add("driver_pnum"); + data.add("orig_escort_stoptype"); + data.add("orig_escortee_pnum"); + data.add("dest_escort_stoptype"); + data.add("dest_escortee_pnum"); + data.add("valueOfTime"); + data.add("transponder_avail"); + data.add("micro_walkMode"); + data.add("micro_trnAcc"); + data.add("micro_trnEgr"); + data.add("parkingCost"); + + if(writeLogsums) { + data.add("tripModeLogsum"); + data.add("microWalkModeLogsum"); + data.add("microTrnAccLogsum"); + data.add("microTrnEgrLogsum"); + } + return data; + } + + private List formJointTripColumnNames() + { + List data = new LinkedList(); + data.add("hh_id"); + data.add("tour_id"); + data.add("stop_id"); + data.add("inbound"); + data.add("tour_purpose"); + data.add("orig_purpose"); + data.add("dest_purpose"); + data.add("orig_mgra"); + data.add("dest_mgra"); + data.add("parking_mgra"); + data.add("stop_period"); + data.add("trip_mode"); + data.add("av_avail"); + data.add("num_participants"); + data.add("trip_board_tap"); + data.add("trip_alight_tap"); + data.add("set"); + data.add("tour_mode"); + data.add("valueOfTime"); + data.add("transponder_avail"); + //wsu remove micromobility columns, not applicable to joint trips + //data.add("micro_walkMode"); + //data.add("micro_trnAcc"); + //data.add("micro_trnEgr"); + data.add("parkingCost"); + + if(writeLogsums) { + data.add("tripModeLogsum"); + data.add("microWalkModeLogsum"); + data.add("microTrnAccLogsum"); + data.add("microTrnEgrLogsum"); + } + + return data; + } + + private List formIndivTripColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + + if(writeLogsums) { + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + } + return data; + } + + private List formJointTripColumnTypes() + { + List data = new LinkedList(); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.TEXT); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.INTEGER); + data.add(SqliteDataTypes.REAL); + + if(writeLogsums) { + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + data.add(SqliteDataTypes.REAL); + } + return data; + } + + private List formIndivTripDataEntry(Stop s) + { + Tour t = s.getTour(); + Household h = t.getPersonObject().getHouseholdObject(); + + List data = new LinkedList(); + data.add(string(t.getHhId())); + data.add(string(t.getPersonObject().getPersonId())); + data.add(string(t.getPersonObject().getPersonNum())); + data.add(string(t.getTourId())); + data.add(string(s.getStopId())); + data.add(string(s.isInboundStop() ? 1 : 0)); + data.add(string(t.getTourPurpose())); + + if (s.getStopId() == 0) + { + if (s.isInboundStop()) + { + // first trip on inbound half-tour with stops + data.add(s.getOrigPurpose()); + data.add(s.getDestPurpose()); + data.add(string(t.getTourDestMgra())); + data.add(string(s.getDest())); + } else + { + // first trip on outbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add("Work"); + data.add(s.getDestPurpose()); + } else + { + data.add("Home"); + data.add(s.getDestPurpose()); + } + data.add(string(t.getTourOrigMgra())); + data.add(string(s.getDest())); + } + } else if (s.isInboundStop() && s.getStopId() == t.getNumInboundStops() - 1) + { + // last trip on inbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(s.getOrigPurpose()); + data.add("Work"); + } else + { + data.add(s.getOrigPurpose()); + data.add("Home"); + } + data.add(string(s.getOrig())); + data.add(string(t.getTourOrigMgra())); + } else if (!s.isInboundStop() && s.getStopId() == t.getNumOutboundStops() - 1) + { + // last trip on outbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(s.getOrigPurpose()); + data.add(t.getTourPurpose()); + } else + { + data.add(s.getOrigPurpose()); + data.add(t.getTourPurpose()); + } + data.add(string(s.getOrig())); + data.add(string(t.getTourDestMgra())); + } else + { + data.add(s.getOrigPurpose()); + data.add(s.getDestPurpose()); + data.add(string(s.getOrig())); + data.add(string(s.getDest())); + } + + data.add(string(s.getPark())); + data.add(string(s.getStopPeriod())); + data.add(string(s.getMode())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + data.add(string(s.getBoardTap())); + data.add(string(s.getAlightTap())); + int set = setNA; + if(modelStructure.getTripModeIsTransit(s.getMode())) { + set = s.getSet(); + } + data.add(string(set)); + data.add(string(t.getTourModeChoice())); + data.add(string(s.isInboundStop() ? t.getDriverPnumInbound() : t.getDriverPnumOutbound())); + data.add(string(s.getEscortStopTypeOrig())); + data.add(string(s.getEscorteePnumOrig())); + data.add(string(s.getEscortStopTypeDest())); + data.add(string(s.getEscorteePnumDest())); + data.add(string(s.getValueOfTime())); + data.add(string(h.getTpChoice())); + data.add(string(s.getMicromobilityWalkMode())); + data.add(string(s.getMicromobilityAccessMode())); + data.add(string(s.getMicromobilityEgressMode())); + data.add(string(s.getParkingCost())); + + if(writeLogsums) { + data.add(string(s.getModeLogsum())); + data.add(string(s.getMicromobilityWalkLogsum())); + data.add(string(s.getMicromobilityAccessLogsum())); + data.add(string(s.getMicromobilityEgressLogsum())); + + } + return data; + } + + private List formJointTripDataEntry(Stop s) + { + Tour t = s.getTour(); + Household h = t.getPersonObject().getHouseholdObject(); + List data = new LinkedList(); + data.add(string(t.getHhId())); + data.add(string(t.getTourId())); + data.add(string(s.getStopId())); + data.add(string(s.isInboundStop() ? 1 : 0)); + data.add(string(t.getTourPurpose())); + + if (s.getStopId() == 0) + { + if (s.isInboundStop()) + { + // first trip on inbound half-tour with stops + data.add(s.getOrigPurpose()); + data.add(s.getDestPurpose()); + } else + { + // first trip on outbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add("Work"); + data.add(s.getDestPurpose()); + } else + { + data.add("Home"); + data.add(s.getDestPurpose()); + } + } + } else if (s.isInboundStop() && s.getStopId() == t.getNumInboundStops() - 1) + { + // last trip on inbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(s.getOrigPurpose()); + data.add("Work"); + } else + { + data.add(s.getOrigPurpose()); + data.add("Home"); + } + } else if (!s.isInboundStop() && s.getStopId() == t.getNumOutboundStops() - 1) + { + // last trip on outbound half-tour with stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(s.getOrigPurpose()); + data.add(t.getTourPurpose()); + } else + { + data.add(s.getOrigPurpose()); + data.add(t.getTourPurpose()); + } + } else + { + data.add(s.getOrigPurpose()); + data.add(s.getDestPurpose()); + } + + data.add(string(s.getOrig())); + data.add(string(s.getDest())); + data.add(string(s.getPark())); + data.add(string(s.getStopPeriod())); + data.add(string(s.getMode())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + + int[] participants = t.getPersonNumArray(); + if (participants == null) + { + logger.error("tour participants array is null, hhid=" + t.getHhId() + "."); + throw new RuntimeException(); + } + if (participants.length < 2) + { + logger.error("length of tour participants array is not null, but is < 2; should be >= 2 for joint tour, hhid=" + + t.getHhId() + "."); + throw new RuntimeException(); + } + + data.add(string(participants.length)); + data.add(string(s.getBoardTap())); + data.add(string(s.getAlightTap())); + int set = setNA; + if(modelStructure.getTripModeIsTransit(s.getMode())) { + set = s.getSet(); + } + data.add(string(set)); + data.add(string(t.getTourModeChoice())); + data.add(string(s.getValueOfTime())); + data.add(string(h.getTpChoice())); + //wsu, remove micromobility columns, not applicable to joint trips + //data.add(string(s.getMicromobilityWalkMode())); + //data.add(string(s.getMicromobilityAccessMode())); + //data.add(string(s.getMicromobilityEgressMode())); + data.add(string(s.getParkingCost())); + + if(writeLogsums) { + data.add(string(s.getModeLogsum())); + data.add(string(s.getMicromobilityWalkLogsum())); + data.add(string(s.getMicromobilityAccessLogsum())); + data.add(string(s.getMicromobilityEgressLogsum())); + + } + + + if(writeLogsums) + data.add(string(s.getModeLogsum())); + + return data; + } + + private List formTourAsIndivTripDataEntry(Tour t, boolean inbound) + { + List data = new LinkedList(); + Household h = t.getPersonObject().getHouseholdObject(); + + data.add(string(t.getHhId())); + data.add(string(t.getPersonObject().getPersonId())); + data.add(string(t.getPersonObject().getPersonNum())); + data.add(string(t.getTourId())); + data.add(string(-1)); + data.add(string((inbound ? 1 : 0))); + data.add(string(t.getTourPurpose())); + + if (inbound) + { + // inbound trip on half-tour with no stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(t.getTourPurpose()); + data.add("Work"); + } else + { + data.add(t.getTourPurpose()); + data.add("Home"); + } + } else + { + // outbound trip on half-tour with no stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add("Work"); + data.add(t.getTourPurpose()); + } else + { + data.add("Home"); + data.add(t.getTourPurpose()); + } + } + + data.add(string((inbound ? t.getTourDestMgra() : t.getTourOrigMgra()))); + data.add(string((inbound ? t.getTourOrigMgra() : t.getTourDestMgra()))); + data.add(string(t.getTourParkMgra())); + data.add(string(0)); + data.add(string(inbound ? t.getTourArrivePeriod() : t.getTourDepartPeriod())); + data.add(string(t.getTourModeChoice())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + data.add(string(inbound ? t.getDriverPnumInbound() : t.getDriverPnumOutbound())); + + /* //outbound chauffeured school tour no stops; origin stop type and escortee is zero, dest stop type is dropoff, escortee is pnum. + if(!inbound && t.getTourPurpose().equals("School") && t.getDriverPnumOutbound()>0){ + + data.add(string(0)); + data.add(string(0)); + data.add(string(ModelStructure.ESCORT_STOP_TYPE_DROPOFF)); + data.add(string(t.getPersonObject().getPersonNum())); + + }else if(inbound && t.getTourPurpose().equals("School") && t.getDriverPnumInbound()>0){ + + data.add(string(ModelStructure.ESCORT_STOP_TYPE_PICKUP)); + data.add(string(t.getPersonObject().getPersonNum())); + data.add(string(0)); + data.add(string(0)); + }else + */ + if (!inbound && t.getDriverPnumOutbound()>0){ //outbound + data.add(string(0)); //origin = home + data.add(string(0)); //origin = home + Stop[] stops = t.getInboundStops(); //there must be stops in inbound direction + int stopType = stops[0].getEscortStopTypeOrig(); //first inbound stop + int pnum = stops[0].getEscorteePnumOrig(); //first inbound stop + data.add(string(stopType)); //destination + data.add(string(pnum)); //destination + }else if (inbound && t.getDriverPnumInbound()>0){ //inbound + Stop[] stops = t.getOutboundStops(); + int stopType = stops[stops.length-1].getEscortStopTypeOrig(); //last outbound stop + int pnum = stops[stops.length-1].getEscorteePnumOrig(); //last outbound stop + data.add(string(stopType)); //origin + data.add(string(pnum)); //origin + data.add(string(0)); //destination = home + data.add(string(0)); //destination = home + } + else{ + data.add(string(0)); + data.add(string(0)); + data.add(string(0)); + data.add(string(0)); + } + + data.add(string(t.getTourModeChoice())); + + /* if(true){logger.error("Trying to write a tour as a trip"); + + logger.info("HHID: " +t.getHhId()); + logger.info("PERSNUM: "+ t.getPersonObject().getPersonNum()); + logger.info("TOURID: "+t.getTourId()); + logger.info(inbound ? "inbound" : "outbound"); + } + */ + + data.add(string(t.getValueOfTime())); + data.add(string(h.getTpChoice())); + + if(writeLogsums) + data.add(string(t.getTourModeLogsum())); + + return data; + } + + private List formTourAsJointTripDataEntry(Tour t, boolean inbound) + { + List data = new LinkedList(); + data.add(string(t.getHhId())); + data.add(string(t.getTourId())); + data.add(string(-1)); + data.add(string((inbound ? 1 : 0))); + data.add(string(t.getTourPurpose())); + + if (inbound) + { + // inbound trip on half-tour with no stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add(t.getTourPurpose()); + data.add("Work"); + } else + { + data.add(t.getTourPurpose()); + data.add("Home"); + } + } else + { + // outbound trip on half-tour with no stops + if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + data.add("Work"); + data.add(t.getTourPurpose()); + } else + { + data.add("Home"); + data.add(t.getTourPurpose()); + } + } + + data.add(string((inbound ? t.getTourDestMgra() : t.getTourOrigMgra()))); + data.add(string((inbound ? t.getTourOrigMgra() : t.getTourDestMgra()))); + data.add(string(t.getTourParkMgra())); + data.add(string(inbound ? t.getTourArrivePeriod() : t.getTourDepartPeriod())); + data.add(string(t.getTourModeChoice())); + data.add(string(t.getUseOwnedAV() ? 1 : 0)); + + int[] participants = t.getPersonNumArray(); + if (participants == null) + { + logger.error("tour participants array is null, hhid=" + t.getHhId() + "."); + throw new RuntimeException(); + } + if (participants.length < 2) + { + logger.error("length of tour participants array is not null, but is < 2; should be >= 2 for joint tour, hhid=" + + t.getHhId() + "."); + throw new RuntimeException(); + } + + data.add(string(participants.length)); + data.add(string(t.getTourModeChoice())); + data.add(string(t.getValueOfTime())); + + if(writeLogsums) + data.add(string(t.getTourModeLogsum())); + + return data; + } + + private static enum SqliteDataTypes + { + INTEGER, TEXT, REAL + } + + private interface DataWriter + { + void writeHouseholdData(List data); + + void writePersonData(List data); + + void writeIndivTourData(List data); + + void writeJointTourData(List data); + + void writeIndivTripData(List data); + + void writeJointTripData(List data); + + void finishActions(); + } + + private class DatabaseDataWriter + implements DataWriter + { + private final String householdTable = rbMap.get(PROPERTIES_HOUSEHOLD_TABLE); + private final String personTable = rbMap.get(PROPERTIES_PERSON_TABLE); + private final String indivTourTable = rbMap.get(PROPERTIES_INDIV_TOUR_TABLE); + private final String jointTourTable = rbMap.get(PROPERTIES_JOINT_TOUR_TABLE); + private final String indivTripTable = rbMap.get(PROPERTIES_INDIV_TRIP_TABLE); + private final String jointTripTable = rbMap.get(PROPERTIES_JOINT_TRIP_TABLE); + private Connection connection = null; + private PreparedStatement hhPreparedStatement = null; + private PreparedStatement personPreparedStatement = null; + private PreparedStatement indivTourPreparedStatement = null; + private PreparedStatement jointTourPreparedStatement = null; + private PreparedStatement indivTripPreparedStatement = null; + private PreparedStatement jointTripPreparedStatement = null; + + public DatabaseDataWriter(String dbFileName) + { + initializeTables(dbFileName); + } + + private void initializeTables(String dbFileName) + { + Statement s = null; + try + { + connection = ConnectionHelper.getConnection(dbFileName); + s = connection.createStatement(); + s.addBatch(getTableInitializationString(householdTable, formHouseholdColumnNames(), + formHouseholdColumnTypes())); + s.addBatch(getTableInitializationString(personTable, formPersonColumnNames(), + formPersonColumnTypes())); + s.addBatch(getTableInitializationString(indivTourTable, formIndivTourColumnNames(), + formIndivTourColumnTypes())); + s.addBatch(getTableInitializationString(jointTourTable, formJointTourColumnNames(), + formJointTourColumnTypes())); + s.addBatch(getTableInitializationString(indivTripTable, formIndivTripColumnNames(), + formIndivTripColumnTypes())); + s.addBatch(getTableInitializationString(jointTripTable, formJointTripColumnNames(), + formJointTripColumnTypes())); + s.addBatch(getClearTableString(householdTable)); + s.addBatch(getClearTableString(personTable)); + s.addBatch(getClearTableString(indivTourTable)); + s.addBatch(getClearTableString(jointTourTable)); + s.addBatch(getClearTableString(indivTripTable)); + s.addBatch(getClearTableString(jointTripTable)); + s.executeBatch(); + } catch (SQLException e) + { + try + { + if (connection != null) connection.close(); + } catch (SQLException ee) + { + // swallow + } + throw new RuntimeException(e); + } finally + { + closeStatement(s); + } + setupPreparedStatements(); + } + + private void setupPreparedStatements() + { + String psStart = "INSERT INTO "; + String psMiddle = " VALUES (?"; + StringBuilder hhp = new StringBuilder(psStart); + hhp.append(householdTable).append(psMiddle); + for (int i = 1; i < formHouseholdColumnNames().size(); i++) + hhp.append(",?"); + hhp.append(");"); + StringBuilder pp = new StringBuilder(psStart); + pp.append(personTable).append(psMiddle); + for (int i = 1; i < formPersonColumnNames().size(); i++) + pp.append(",?"); + pp.append(");"); + StringBuilder itp = new StringBuilder(psStart); + itp.append(indivTourTable).append(psMiddle); + for (int i = 1; i < formIndivTourColumnNames().size(); i++) + itp.append(",?"); + itp.append(");"); + StringBuilder jtp = new StringBuilder(psStart); + jtp.append(jointTourTable).append(psMiddle); + for (int i = 1; i < formJointTourColumnNames().size(); i++) + jtp.append(",?"); + jtp.append(");"); + StringBuilder itp2 = new StringBuilder(psStart); + itp2.append(indivTripTable).append(psMiddle); + for (int i = 1; i < formIndivTripColumnNames().size(); i++) + itp2.append(",?"); + itp2.append(");"); + StringBuilder jtp2 = new StringBuilder(psStart); + jtp2.append(jointTripTable).append(psMiddle); + for (int i = 1; i < formJointTripColumnNames().size(); i++) + jtp2.append(",?"); + jtp2.append(");"); + try + { + hhPreparedStatement = connection.prepareStatement(hhp.toString()); + personPreparedStatement = connection.prepareStatement(pp.toString()); + indivTourPreparedStatement = connection.prepareStatement(itp.toString()); + jointTourPreparedStatement = connection.prepareStatement(jtp.toString()); + indivTripPreparedStatement = connection.prepareStatement(itp2.toString()); + jointTripPreparedStatement = connection.prepareStatement(jtp2.toString()); + connection.setAutoCommit(false); + } catch (SQLException e) + { + throw new RuntimeException(e); + } + } + + private String getTableInitializationString(String table, List columns, + List types) + { + StringBuilder sb = new StringBuilder("CREATE TABLE IF NOT EXISTS "); + sb.append(table).append(" ("); + Iterator cols = columns.iterator(); + Iterator tps = types.iterator(); + sb.append(cols.next()).append(" ").append(tps.next().name()); + while (cols.hasNext()) + sb.append(",").append(cols.next()).append(" ").append(tps.next().name()); + sb.append(");"); + return sb.toString(); + } + + private String getClearTableString(String table) + { + return "DELETE FROM " + table + ";"; + } + + private void writeToTable(PreparedStatement ps, List values) + { + try + { + int counter = 1; + for (String value : values) + ps.setString(counter++, value); + ps.executeUpdate(); + } catch (SQLException e) + { + throw new RuntimeException(e); + } + } + + // private void writeToTable(String table, List values) { + // StringBuilder sb = new StringBuilder("INSERT INTO"); + // sb.append(" ").append(table).append(" VALUES("); + // Iterator vls = values.iterator(); + // sb.append(vls.next()); + // while (vls.hasNext()) + // sb.append(",").append(vls.next()); + // sb.append(");"); + // try { + // s.addBatch(sb.toString()); + // } catch (SQLException e) { + // try { + // throw new RuntimeException(e); + // } finally { + // try { + // if (s != null) + // s.close(); + // } catch (SQLException ee) { + // //swallow + // } + // try { + // if (connection != null) + // connection.close(); + // } catch (SQLException ee) { + // //swallow + // } + // } + // } + // } + + public void writeHouseholdData(List data) + { + writeToTable(hhPreparedStatement, data); + } + + public void writePersonData(List data) + { + writeToTable(personPreparedStatement, data); + } + + public void writeIndivTourData(List data) + { + writeToTable(indivTourPreparedStatement, data); + } + + public void writeJointTourData(List data) + { + writeToTable(jointTourPreparedStatement, data); + } + + public void writeIndivTripData(List data) + { + writeToTable(indivTripPreparedStatement, data); + } + + public void writeJointTripData(List data) + { + writeToTable(jointTripPreparedStatement, data); + } + + public void finishActions() + { + + try + { + connection.commit(); + } catch (SQLException e) + { + throw new RuntimeException(e); + } finally + { + closeStatement(hhPreparedStatement); + closeStatement(personPreparedStatement); + closeStatement(indivTourPreparedStatement); + closeStatement(jointTourPreparedStatement); + closeStatement(indivTripPreparedStatement); + closeStatement(jointTripPreparedStatement); + try + { + if (connection != null) connection.close(); + } catch (SQLException ee) + { + // swallow + } + } + } + + private void closeStatement(Statement s) + { + try + { + if (s != null) s.close(); + } catch (SQLException e) + { + // swallow + } + } + } + + private class FileDataWriter + implements DataWriter + { + private final PrintWriter hhWriter; + private final PrintWriter personWriter; + private final PrintWriter indivTourWriter; + private final PrintWriter jointTourWriter; + private final PrintWriter indivTripWriter; + private final PrintWriter jointTripWriter; + + public FileDataWriter() + { + String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + + String hhFile = formFileName(rbMap.get(PROPERTIES_HOUSEHOLD_DATA_FILE), iteration); + String personFile = formFileName(rbMap.get(PROPERTIES_PERSON_DATA_FILE), iteration); + String indivTourFile = formFileName(rbMap.get(PROPERTIES_INDIV_TOUR_DATA_FILE), + iteration); + String jointTourFile = formFileName(rbMap.get(PROPERTIES_JOINT_TOUR_DATA_FILE), + iteration); + String indivTripFile = formFileName(rbMap.get(PROPERTIES_INDIV_TRIP_DATA_FILE), + iteration); + String jointTripFile = formFileName(rbMap.get(PROPERTIES_JOINT_TRIP_DATA_FILE), + iteration); + + try + { + hhWriter = new PrintWriter(new File(baseDir + hhFile)); + personWriter = new PrintWriter(new File(baseDir + personFile)); + indivTourWriter = new PrintWriter(new File(baseDir + indivTourFile)); + jointTourWriter = new PrintWriter(new File(baseDir + jointTourFile)); + indivTripWriter = new PrintWriter(new File(baseDir + indivTripFile)); + jointTripWriter = new PrintWriter(new File(baseDir + jointTripFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + writeHouseholdData(formHouseholdColumnNames()); + writePersonData(formPersonColumnNames()); + writeIndivTourData(formIndivTourColumnNames()); + writeJointTourData(formJointTourColumnNames()); + writeIndivTripData(formIndivTripColumnNames()); + writeJointTripData(formJointTripColumnNames()); + } + + private String formFileName(String originalFileName, int iteration) + { + int lastDot = originalFileName.lastIndexOf('.'); + + String returnString = ""; + if (lastDot > 0) + { + String base = originalFileName.substring(0, lastDot); + String ext = originalFileName.substring(lastDot); + returnString = String.format("%s_%d%s", base, iteration, ext); + } else + { + returnString = String.format("%s_%d.csv", originalFileName, iteration); + } + + logger.info("writing household csv file to " + returnString); + + return returnString; + } + + public void writeHouseholdData(List data) + { + writeEntryToCsv(hhWriter, data); + } + + public void writePersonData(List data) + { + writeEntryToCsv(personWriter, data); + } + + public void writeIndivTourData(List data) + { + writeEntryToCsv(indivTourWriter, data); + } + + public void writeJointTourData(List data) + { + writeEntryToCsv(jointTourWriter, data); + } + + public void writeIndivTripData(List data) + { + writeEntryToCsv(indivTripWriter, data); + } + + public void writeJointTripData(List data) + { + writeEntryToCsv(jointTripWriter, data); + } + + private void writeEntryToCsv(PrintWriter pw, List data) + { + pw.println(formCsvString(data)); + } + + private String formCsvString(List data) + { + char delimiter = ','; + Iterator it = data.iterator(); + StringBuilder sb = new StringBuilder(it.next()); + while (it.hasNext()) + sb.append(delimiter).append(it.next()); + return sb.toString(); + } + + public void finishActions() + { + try + { + hhWriter.flush(); + personWriter.flush(); + indivTourWriter.flush(); + jointTourWriter.flush(); + indivTripWriter.flush(); + jointTripWriter.flush(); + } finally + { + hhWriter.close(); + personWriter.close(); + indivTourWriter.close(); + jointTourWriter.close(); + indivTripWriter.close(); + jointTripWriter.close(); + } + + } + } + + private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) + { + + ArrayList startEndIndexList = new ArrayList(); + + int startIndex = 0; + int endIndex = 0; + + while (endIndex < numberOfHouseholds - 1) + { + endIndex = startIndex + NUM_WRITE_PACKETS - 1; + if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) + endIndex = numberOfHouseholds - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += NUM_WRITE_PACKETS; + } + + return startEndIndexList; + + } + + /** + * Calculate auto skims for a given origin to all destination mgras, and + * return auto distance. + * + * @param oMgra + * The origin mgra + * @return An array of distances + */ + private double calculateDistancesForAllMgras(int oMgra, int dMgra) + { + + int oTaz = mgraManager.getTaz(oMgra); + int dTaz = mgraManager.getTaz(dMgra); + + iv.setOriginZone(oTaz); + iv.setDestZone(dTaz); + + // sov time in results[0] and distance in resuls[1] + double[] results = autoSkimUEC.solve(iv, dmu, null); + + return results[1]; + } +} + diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java new file mode 100644 index 0000000..369b6e9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java @@ -0,0 +1,1672 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; + +import java.io.Serializable; +import java.util.*; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import com.pb.common.newmodel.ChoiceModelApplication; + +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + + + + + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To + * change this template use File | Settings | File Templates. + */ +public class HouseholdIndividualMandatoryTourDepartureAndDurationTime + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdIndividualMandatoryTourDepartureAndDurationTime.class); + private transient Logger todLogger = Logger.getLogger("todLogger"); + private transient Logger tourMCManLogger = Logger.getLogger("tourMcMan"); + + private static final String IMTOD_UEC_FILE_TARGET = "departTime.uec.file"; + private static final String IMTOD_UEC_DATA_TARGET = "departTime.data.page"; + private static final String IMTOD_UEC_WORK_MODEL_TARGET = "departTime.work.page"; + private static final String IMTOD_UEC_SCHOOL_MODEL_TARGET = "departTime.school.page"; + private static final String IMTOD_UEC_UNIV_MODEL_TARGET = "departTime.univ.page"; + + private int[] workTourDepartureTimeChoiceSample; + private int[] schoolTourDepartureTimeChoiceSample; + + // DMU for the UEC + private TourDepartureTimeAndDurationDMU imtodDmuObject; + private TourModeChoiceDMU mcDmuObject; + + private String tourCategory = ModelStructure.MANDATORY_CATEGORY; + + private ModelStructure modelStructure; + + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private ChoiceModelApplication workTourChoiceModel; + private ChoiceModelApplication schoolTourChoiceModel; + private ChoiceModelApplication univTourChoiceModel; + private TourModeChoiceModel mcModel; + + private boolean[] needToComputeLogsum; + private double[] modeChoiceLogsums; + + private int[] altStarts; + private int[] altEnds; + + private int noAvailableWorkWindowCount = 0; + private int noAvailableSchoolWindowCount = 0; + + private int noUsualWorkLocationForMandatoryActivity = 0; + private int noUsualSchoolLocationForMandatoryActivity = 0; + + private HashMap rbMap; + + private long mcTime; + + public HouseholdIndividualMandatoryTourDepartureAndDurationTime( + HashMap propertyMap, ModelStructure modelStructure, + String[] tourPurposeList, CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) + { + + setupHouseholdIndividualMandatoryTourDepartureAndDurationTime(propertyMap, modelStructure, + tourPurposeList, dmuFactory, mcModel); + + } + + private void setupHouseholdIndividualMandatoryTourDepartureAndDurationTime( + HashMap propertyMap, ModelStructure modelStructure, + String[] tourPurposeList, CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) + { + + logger.info(String.format("setting up %s time-of-day choice model.", tourCategory)); + + // set the model structure + this.modelStructure = modelStructure; + this.mcModel = mcModel; + rbMap = propertyMap; + + tazs = TazDataManager.getInstance(); + mgraManager = MgraDataManager.getInstance(); + + // locate the individual mandatory tour frequency choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String imtodUecFile = propertyMap.get(IMTOD_UEC_FILE_TARGET); + imtodUecFile = uecPath + imtodUecFile; + + int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTOD_UEC_DATA_TARGET); + int workModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, + IMTOD_UEC_WORK_MODEL_TARGET); + int schoolModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, + IMTOD_UEC_SCHOOL_MODEL_TARGET); + int univModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, + IMTOD_UEC_UNIV_MODEL_TARGET); + + // get the dmu objects from the factory + imtodDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); + mcDmuObject = dmuFactory.getModeChoiceDMU(); + + // set up the models + workTourChoiceModel = new ChoiceModelApplication(imtodUecFile, workModelPage, dataPage, + propertyMap, (VariableTable) imtodDmuObject); + schoolTourChoiceModel = new ChoiceModelApplication(imtodUecFile, schoolModelPage, dataPage, + propertyMap, (VariableTable) imtodDmuObject); + univTourChoiceModel = new ChoiceModelApplication(imtodUecFile, univModelPage, dataPage, + propertyMap, (VariableTable) imtodDmuObject); + + // get the alternatives table from the work tod UEC. + TableDataSet altsTable = workTourChoiceModel.getUEC().getAlternativeData(); + altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); + altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); + altsTable = null; + + imtodDmuObject.setTodAlts(altStarts, altEnds); + + int numWorkDepartureTimeChoiceAlternatives = workTourChoiceModel.getNumberOfAlternatives(); + workTourDepartureTimeChoiceSample = new int[numWorkDepartureTimeChoiceAlternatives + 1]; + Arrays.fill(workTourDepartureTimeChoiceSample, 1); + + int numSchoolDepartureTimeChoiceAlternatives = schoolTourChoiceModel + .getNumberOfAlternatives(); + schoolTourDepartureTimeChoiceSample = new int[numSchoolDepartureTimeChoiceAlternatives + 1]; + Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); + + int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; + needToComputeLogsum = new boolean[numLogsumIndices]; + + modeChoiceLogsums = new double[numLogsumIndices]; + + } + + public void applyModel(Household household, boolean runTODChoice, boolean runModeChoice) + { + mcTime = 0; + + Logger modelLogger = todLogger; + if (household.getDebugChoiceModels()) + { + household.logHouseholdObject( + "Pre Individual Mandatory Departure Time Choice Model HHID=" + + household.getHhId(), modelLogger); + if (runModeChoice) + household.logHouseholdObject( + "Pre Individual Mandatory Tour Mode Choice Model HHID=" + + household.getHhId(), tourMCManLogger); + } + + // set the household id, origin taz, hh taz, and debugFlag=false in the + // dmu + imtodDmuObject.setHousehold(household); + + // get the array of persons for this household + Person[] personArray = household.getPersons(); + + + if(!runTODChoice) { + // loop through the persons (1-based array) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + if (runModeChoice) + household.logPersonObject(decisionMakerLabel, tourMCManLogger, person); + } + + try { + ArrayList workTours = person.getListOfWorkTours(); + if(workTours!=null) + if(workTours.size()>0) { + for(Tour tour: workTours) { + runModeChoice(household,person,tour,tour.getTourDepartPeriod(),tour.getTourArrivePeriod()); + } + } + ArrayList schoolTours = person.getListOfSchoolTours(); + if(schoolTours!=null) + if(schoolTours.size()>0) { + for(Tour tour: schoolTours) { + runModeChoice(household,person,tour,tour.getTourDepartPeriod(),tour.getTourArrivePeriod()); + } + } + }catch(Exception e) { + logger.error(String + .format("error mandatory mode choice model for j=%d, hhId=%d, persId=%d, persNum=%d, personType=%s.", + j, person.getHouseholdObject().getHhId(), person.getPersonId(), + person.getPersonNum(), person.getPersonType())); + throw new RuntimeException(e); + + } + + } + return; + } + + // loop through the persons (1-based array) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + person.resetTimeWindow(); + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + if (runModeChoice) + household.logPersonObject(decisionMakerLabel, tourMCManLogger, person); + } + + // mandatory tour departure time and dureation choice models for + // each + // worker/student require a specific order: + // 1. Work tours made by workers, school/university tours made by + // students. + // 2. Work tours made by students, school/university tours made by + // workers. + // TODO: check consistency of these definitions - + // TODO: workers can also be students (school-age and university)?, + // non-driving students can be workers?, + // TODO: cannot be school-age student and university? etc... + + try + { + + if (person.getPersonIsWorker() == 1) + { + applyDepartureTimeChoiceForWorkTours(person, runModeChoice); + if (person.getListOfSchoolTours().size() > 0) + { + if (person.getPersonIsUniversityStudent() == 1) + { + applyDepartureTimeChoiceForUnivTours(person, runModeChoice); + } else + { + applyDepartureTimeChoiceForSchoolTours(person, runModeChoice); + } + } + } else if (person.getPersonIsStudent() == 1 + || person.getPersonIsPreschoolChild() == 1) + { + if (person.getPersonIsUniversityStudent() == 1) + { + applyDepartureTimeChoiceForUnivTours(person, runModeChoice); + } else + { + applyDepartureTimeChoiceForSchoolTours(person, runModeChoice); + } + if (person.getListOfWorkTours().size() > 0) + applyDepartureTimeChoiceForWorkTours(person, runModeChoice); + } else + { + if (person.getListOfWorkTours().size() > 0 + || person.getListOfSchoolTours().size() > 0) + { + logger.error(String + .format("error mandatory departure time choice model for j=%d, hhId=%d, persNum=%d, personType=%s.", + j, person.getHouseholdObject().getHhId(), + person.getPersonNum(), person.getPersonType())); + logger.error(String + .format("person with type other than worker or student has %d work tours and %d school tours.", + person.getListOfWorkTours().size(), person + .getListOfSchoolTours().size())); + throw new RuntimeException(); + } + } + + } catch (Exception e) + { + logger.error(String + .format("error mandatory departure time choice model for j=%d, hhId=%d, persId=%d, persNum=%d, personType=%s.", + j, person.getHouseholdObject().getHhId(), person.getPersonId(), + person.getPersonNum(), person.getPersonType())); + throw new RuntimeException(e); + } + + } + + household.setImtodRandomCount(household.getHhRandomCount()); + + } + + /** + * + * @param person + * object for which time choice should be made + * @return the number of work tours this person had scheduled. + */ + private int applyDepartureTimeChoiceForWorkTours(Person person, boolean runModeChoice) + { + + Logger modelLogger = todLogger; + + // set the dmu object + imtodDmuObject.setPerson(person); + + Household household = person.getHouseholdObject(); + + ArrayList workTours = person.getListOfWorkTours(); + ArrayList schoolTours = person.getListOfSchoolTours(); + + for (int i = 0; i < workTours.size(); i++) + { + + Tour t = workTours.get(i); + t.setTourDepartPeriod(-1); + t.setTourArrivePeriod(-1); + + // dest taz was set from result of usual school location choice when + // tour + // object was created in mandatory tour frequency model. + // TODO: if the destMgra value is -1, then this mandatory tour was + // created for a non-student (retired probably) + // TODO: and we have to resolve this somehow - either genrate a + // work/school location for retired, or change activity type for + // person. + // TODO: for now, we'll just skip the tour, and keep count of them. + int destMgra = t.getTourDestMgra(); + if (destMgra <= 0) + { + noUsualWorkLocationForMandatoryActivity++; + continue; + } + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Mandatory Work Tour Departure Time Choice Model for: Purpose=%s", + t.getTourPurpose()); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), + person.getPersonNum(), person.getPersonType(), t.getTourId(), + workTours.size()); + + workTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "Individual Mandatory Work Tour Departure Time Choice Model: Debug Statement for Household ID: " + + household.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + + person.getPersonType() + + ", Work Tour Id: " + + t.getTourId() + " of " + workTours.size() + " work tours."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + } + + imtodDmuObject.setDestinationZone(destMgra); + imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); + + // set the dmu object + imtodDmuObject.setTour(t); + + int origMgra = t.getTourOrigMgra(); + imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); + imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); + + // set the choice availability and initialize sample array - + // choicemodelapplication will change sample[] according to + // availability[] + boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, + altEnds); + Arrays.fill(workTourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != workTourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in work departure time choice model for hhId=%d, persId=%d, persNum=%d, work tour %d of %d.", + person.getHouseholdObject().getHhId(), person.getPersonId(), + person.getPersonNum(), i, workTours.size())); + logger.error(String + .format("length of the availability array determined by the number of alternatiuves set in the person scheduler=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the work tour UEC=%d.", + workTourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if no time window is available for the tour, make the first and + // last + // alternatives available + // for that alternative, and keep track of the number of times this + // condition occurs. + boolean noAlternativeAvailable = true; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + noAlternativeAvailable = false; + break; + } + } + + if (noAlternativeAvailable) + { + noAvailableWorkWindowCount++; + departureTimeChoiceAvailability[1] = true; + departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; + } + + // check for multiple tours for this person + // set the first or second switch if multiple tours for person + if (workTours.size() == 1 && person.getListOfSchoolTours().size() == 0) + { + // not a multiple tour pattern + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else if (workTours.size() > 1 && person.getListOfSchoolTours().size() == 0) + { + // Two work tour multiple tour pattern + if (i == 0) + { + // first of 2 work tours + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(i + 1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(1); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else + { + // second of 2 work tours + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = workTours.get(0).getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this second work tour with depart + // <= first work tour departure AND arrive >= first work + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= workTours.get(0).getTourDepartPeriod() + && endPeriod >= workTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } else if (workTours.size() == 1 && schoolTours.size() == 1) + { + // One work tour, one school tour multiple tour pattern + if (person.getPersonIsWorker() == 1) + { + // worker, so work tour is first scheduled, school tour + // comes later. + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(1); + + } else + { + // student, so school tour was already scheduled, this work + // tour is the second. + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = person.getListOfSchoolTours().get(0) + .getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this work tour with depart <= + // first school tour departure AND arrive >= first school + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() + && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } + + // calculate and store the mode choice logsum for the usual work + // location + // for this worker at the various + // departure time and duration alternativees + setWorkTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, + departureTimeChoiceAvailability); + + if (household.getDebugChoiceModels()) + { + household.logTourObject(loggingHeader, modelLogger, person, t); + } + + float logsum=0; + try + { + logsum = (float) workTourChoiceModel.computeUtilities(imtodDmuObject, + imtodDmuObject.getIndexValues(), departureTimeChoiceAvailability, + workTourDepartureTimeChoiceSample); + } catch (Exception e) + { + logger.error("exception caught computing work tour TOD choice utilities."); + throw new RuntimeException(); + } + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has no available alternatives, choose between + // the + // first and last alternative. + int chosen; + if (workTourChoiceModel.getAvailabilityCount() > 0) chosen = workTourChoiceModel + .getChoiceResult(rn); + else chosen = rn < 0.5 ? 1 : altStarts.length; + + // schedule the chosen alternative + int chosenStartPeriod = altStarts[chosen - 1]; + int chosenEndPeriod = altEnds[chosen - 1]; + try + { + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + } catch (Exception e) + { + logger.error("exception caught updating work tour TOD choice time windows."); + throw new RuntimeException(); + } + + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = workTourChoiceModel.getUtilities(); + double[] probabilities = workTourChoiceModel.getProbabilities(); + boolean[] availabilities = workTourChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString + + ", Tour Id: " + t.getTourId()); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < workTourChoiceModel.getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], + altEnds[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, + altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + workTourChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + workTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, + rn, chosen); + + // write UEC calculation results to separate model specific log + // file + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + workTourChoiceModel.logUECResults(modelLogger, loggingHeader); + + } + + if (runModeChoice) + { + runModeChoice(household, person, t, chosenStartPeriod, chosenEndPeriod); + } + + } + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format( + "Final Work Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + return workTours.size(); + + } + + private void runModeChoice(Household household, Person person, Tour t, int chosenStartPeriod, int chosenEndPeriod) { + + long check = System.nanoTime(); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); + + // use the mcModel object already setup for computing logsums + // and get + // the mode choice, where the selected + // worklocation and subzone an departure time and duration are + // set + // for this work tour. + int chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); + t.setTourModeChoice(chosenMode); + + mcTime += (System.nanoTime() - check); + } + + + + private void setWorkTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Person person, + Tour tour, boolean[] altAvailable) + { + + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todLogger; + String choiceModelDescription = String.format( + "Work Tour Mode Choice Logsum calculation for %s Departure Time Choice", + tour.getTourPurpose()); + String decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person + .getPersonNum(), person.getPersonType(), tour.getTourId(), person + .getListOfWorkTours().size()); + String loggingHeader = String + .format("%s %s", choiceModelDescription, decisionMakerLabel); + + for (int a = 1; a <= altStarts.length; a++) + { + + // if the depart/arrive alternative is unavailable, no need to check + // to see if a logsum has been calculated + if (!altAvailable[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) + + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " work tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + throw new RuntimeException(e); + } + needToComputeLogsum[index] = false; + } + + } + + imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); + + mcDmuObject.getTourObject().setTourDepartPeriod(0); + mcDmuObject.getTourObject().setTourArrivePeriod(0); + } + + private void setSchoolTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives( + Person person, Tour tour, boolean[] altAvailable) + { + + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todLogger; + String choiceModelDescription = String.format( + "School Tour Mode Choice Logsum calculation for %s Departure Time Choice", + tour.getTourPurpose()); + String decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person + .getPersonNum(), person.getPersonType(), tour.getTourId(), person + .getListOfSchoolTours().size()); + String loggingHeader = String + .format("%s %s", choiceModelDescription, decisionMakerLabel); + + for (int a = 1; a <= altStarts.length; a++) + { + + // if the depart/arrive alternative is unavailable, no need to check + // to see if a logsum has been calculated + if (!altAvailable[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) + + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.error(e); + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " school tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + throw new RuntimeException(); + } + needToComputeLogsum[index] = false; + } + + } + + imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); + + } + + /** + * + * @param person + * object for which time choice should be made + * @return the number of school tours this person had scheduled. + */ + private int applyDepartureTimeChoiceForSchoolTours(Person person, boolean runModeChoice) + { + + Logger modelLogger = todLogger; + + // set the dmu object + imtodDmuObject.setPerson(person); + + Household household = person.getHouseholdObject(); + + ArrayList workTours = person.getListOfWorkTours(); + ArrayList schoolTours = person.getListOfSchoolTours(); + + for (int i = 0; i < schoolTours.size(); i++) + { + + Tour t = schoolTours.get(i); + t.setTourDepartPeriod(-1); + t.setTourArrivePeriod(-1); + + // dest taz was set from result of usual school location choice when + // tour + // object was created in mandatory tour frequency model. + // TODO: if the destMgra value is -1, then this mandatory tour was + // created for a non-student (retired probably) + // TODO: and we have to resolve this somehow - either genrate a + // work/school location for retired, or change activity type for + // person. + // TODO: for now, we'll just skip the tour, and keep count of them. + int destMgra = t.getTourDestMgra(); + if (destMgra <= 0) + { + noUsualSchoolLocationForMandatoryActivity++; + continue; + } + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Mandatory School Tour Departure Time Choice Model for: Purpose=%s", + t.getTourPurpose()); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), + person.getPersonNum(), person.getPersonType(), t.getTourId(), + schoolTours.size()); + + schoolTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "Individual Mandatory School Tour Departure Time Choice Model: Debug Statement for Household ID: " + + household.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + + person.getPersonType() + + ", Tour Id: " + + t.getTourId() + " of " + schoolTours.size() + " school tours."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + imtodDmuObject.setDestinationZone(destMgra); + imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); + + // set the dmu object + imtodDmuObject.setTour(t); + + int origMgra = t.getTourOrigMgra(); + imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); + imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); + + // set the choice availability and sample + boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, + altEnds); + Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != schoolTourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in school departure time choice model for hhId=%d, persId=%d, persNum=%d, school tour %d of %d.", + person.getHouseholdObject().getHhId(), person.getPersonId(), + person.getPersonNum(), i, schoolTours.size())); + logger.error(String + .format("length of the availability array determined by the number of alternatiuves set in the person scheduler=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the school tour UEC=%d.", + schoolTourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if no time window is available for the tour, make the first and + // last + // alternatives available + // for that alternative, and keep track of the number of times this + // condition occurs. + boolean noAlternativeAvailable = true; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + noAlternativeAvailable = false; + break; + } + } + + if (noAlternativeAvailable) + { + noAvailableSchoolWindowCount++; + departureTimeChoiceAvailability[1] = true; + schoolTourDepartureTimeChoiceSample[1] = 1; + departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; + schoolTourDepartureTimeChoiceSample[schoolTourDepartureTimeChoiceSample.length - 1] = 1; + } + + // check for multiple tours for this person + // set the first or second switch if multiple tours for person + if (schoolTours.size() == 1 && person.getListOfWorkTours().size() == 0) + { + // not a multiple tour pattern + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else if (schoolTours.size() > 1 && person.getListOfWorkTours().size() == 0) + { + // Two school tour multiple tour pattern + if (i == 0) + { + // first of 2 school tours + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(i + 1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(1); + } else + { + // second of 2 school tours + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = schoolTours.get(0).getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this 2nd school tour with depart + // <= first school tour departure AND arrive >= first school + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() + && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } else if (schoolTours.size() == 1 && workTours.size() == 1) + { + // One school tour, one work tour multiple tour pattern + if (person.getPersonIsStudent() == 1) + { + // student, so school tour is first scheduled, work comes + // later. + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(1); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else + { + // worker, so work tour was already scheduled, this school + // tour is the second. + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = person.getListOfWorkTours().get(0) + .getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this 2nd school tour with depart + // <= first work tour departure AND arrive >= first work + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= workTours.get(0).getTourDepartPeriod() + && endPeriod >= workTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } + + // calculate and store the mode choice logsum for the usual school + // location for this student at the various + // departure time and duration alternativees + setSchoolTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, + departureTimeChoiceAvailability); + + if (household.getDebugChoiceModels()) + { + household.logTourObject(loggingHeader, modelLogger, person, t); + } + + float logsum = (float) schoolTourChoiceModel.computeUtilities(imtodDmuObject, imtodDmuObject.getIndexValues(), + departureTimeChoiceAvailability, schoolTourDepartureTimeChoiceSample); + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has no available alternatives, choose between + // the + // first and last alternative. + int chosen; + if (schoolTourChoiceModel.getAvailabilityCount() > 0) chosen = schoolTourChoiceModel + .getChoiceResult(rn); + else chosen = rn < 0.5 ? 1 : altStarts.length; + + // schedule the chosen alternative + int chosenStartPeriod = altStarts[chosen - 1]; + int chosenEndPeriod = altEnds[chosen - 1]; + try + { + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + } catch (Exception e) + { + logger.error("exception caught updating school tour TOD choice time windows."); + throw new RuntimeException(); + } + + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = schoolTourChoiceModel.getUtilities(); + double[] probabilities = schoolTourChoiceModel.getProbabilities(); + boolean[] availabilities = schoolTourChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString + + ", Tour Id: " + t.getTourId()); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < schoolTourChoiceModel.getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], + altEnds[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, + altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + schoolTourChoiceModel.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + schoolTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, + rn, chosen); + + // write UEC calculation results to separate model specific log + // file + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + schoolTourChoiceModel.logUECResults(modelLogger, loggingHeader, 200); + + } + + if (runModeChoice) + { + + long check = System.nanoTime(); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); + + // use the mcModel object already setup for computing logsums + // and get + // the mode choice, where the selected + // school location and subzone and departure time and duration + // are + // set for this school tour. + int chosenMode = -1; + chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); + + t.setTourModeChoice(chosenMode); + + mcTime += (System.nanoTime() - check); + } + + } + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String + .format("Final School Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + return schoolTours.size(); + + } + + private void setUnivTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Person person, + Tour tour, boolean[] altAvailable) + { + + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todLogger; + String choiceModelDescription = String.format( + "University Tour Mode Choice Logsum calculation for %s Departure Time Choice", + tour.getTourPurpose()); + String decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person + .getPersonNum(), person.getPersonType(), tour.getTourId(), person + .getListOfSchoolTours().size()); + String loggingHeader = String + .format("%s %s", choiceModelDescription, decisionMakerLabel); + + for (int a = 1; a <= altStarts.length; a++) + { + + // if the depart/arrive alternative is unavailable, no need to check + // to see if a logsum has been calculated + if (!altAvailable[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) + + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.error(e); + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " university tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + throw new RuntimeException(); + } + needToComputeLogsum[index] = false; + } + + } + + imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); + + } + + /** + * + * @param person + * object for which time choice should be made + * @return the number of school tours this person had scheduled. + */ + private int applyDepartureTimeChoiceForUnivTours(Person person, boolean runModeChoice) + { + + Logger modelLogger = todLogger; + + // set the dmu object + imtodDmuObject.setPerson(person); + + Household household = person.getHouseholdObject(); + + ArrayList workTours = person.getListOfWorkTours(); + ArrayList schoolTours = person.getListOfSchoolTours(); + + for (int i = 0; i < schoolTours.size(); i++) + { + + Tour t = schoolTours.get(i); + t.setTourDepartPeriod(-1); + t.setTourArrivePeriod(-1); + + // dest taz was set from result of usual school location choice when + // tour + // object was created in mandatory tour frequency model. + // TODO: if the destMgra value is -1, then this mandatory tour was + // created for a non-student (retired probably) + // TODO: and we have to resolve this somehow - either genrate a + // work/school location for retired, or change activity type for + // person. + // TODO: for now, we'll just skip the tour, and keep count of them. + int destMgra = t.getTourDestMgra(); + if (destMgra <= 0) + { + noUsualSchoolLocationForMandatoryActivity++; + continue; + } + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Mandatory University Tour Departure Time Choice Model for: Purpose=%s", + t.getTourPurpose()); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), + person.getPersonNum(), person.getPersonType(), t.getTourId(), + schoolTours.size()); + + univTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "Individual Mandatory University Tour Departure Time Choice Model: Debug Statement for Household ID: " + + household.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + + person.getPersonType() + + ", Tour Id: " + + t.getTourId() + " of " + schoolTours.size() + " school tours."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + imtodDmuObject.setDestinationZone(destMgra); + imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); + + // set the dmu object + imtodDmuObject.setTour(t); + + int origMgra = t.getTourOrigMgra(); + imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); + imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); + + // set the choice availability and sample + boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, + altEnds); + Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != schoolTourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in university departure time choice model for hhId=%d, persId=%d, persNum=%d, school tour %d of %d.", + person.getHouseholdObject().getHhId(), person.getPersonId(), + person.getPersonNum(), i, schoolTours.size())); + logger.error(String + .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the university tour UEC=%d.", + schoolTourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if no time window is available for the tour, make the first and + // last + // alternatives available + // for that alternative, and keep track of the number of times this + // condition occurs. + boolean noAlternativeAvailable = true; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + noAlternativeAvailable = false; + break; + } + } + + if (noAlternativeAvailable) + { + noAvailableSchoolWindowCount++; + departureTimeChoiceAvailability[1] = true; + schoolTourDepartureTimeChoiceSample[1] = 1; + departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; + schoolTourDepartureTimeChoiceSample[schoolTourDepartureTimeChoiceSample.length - 1] = 1; + } + + // check for multiple tours for this person + // set the first or second switch if multiple tours for person + if (schoolTours.size() == 1 && person.getListOfWorkTours().size() == 0) + { + // not a multiple tour pattern + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else if (schoolTours.size() > 1 && person.getListOfWorkTours().size() == 0) + { + // Two school tour multiple tour pattern + if (i == 0) + { + // first of 2 school tours + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(i + 1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(1); + } else + { + // second of 2 school tours + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = schoolTours.get(0).getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this 2nd school tour with depart + // <= first school tour departure AND arrive >= first school + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() + && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } else if (schoolTours.size() == 1 && workTours.size() == 1) + { + // One school tour, one work tour multiple tour pattern + if (person.getPersonIsStudent() == 1) + { + // student, so school tour is first scheduled, work comes + // later. + imtodDmuObject.setFirstTour(1); + imtodDmuObject.setSubsequentTour(0); + imtodDmuObject.setTourNumber(1); + imtodDmuObject.setEndOfPreviousScheduledTour(0); + imtodDmuObject.setSubsequentTourIsWork(1); + imtodDmuObject.setSubsequentTourIsSchool(0); + } else + { + // worker, so work tour was already scheduled, this school + // tour is the second. + imtodDmuObject.setFirstTour(0); + imtodDmuObject.setSubsequentTour(1); + imtodDmuObject.setTourNumber(i + 1); + int otherTourArrivePeriod = person.getListOfWorkTours().get(0) + .getTourArrivePeriod(); + imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); + imtodDmuObject.setSubsequentTourIsWork(0); + imtodDmuObject.setSubsequentTourIsSchool(0); + + // block alternatives for this 2nd school tour with depart + // <= first work tour departure AND arrive >= first work + // tour arrival. + for (int a = 1; a <= altStarts.length; a++) + { + // if the depart/arrive alternative is unavailable, no + // need to check to see if a logsum has been calculated + if (!departureTimeChoiceAvailability[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + if (startPeriod <= workTours.get(0).getTourDepartPeriod() + && endPeriod >= workTours.get(0).getTourArrivePeriod()) + departureTimeChoiceAvailability[a] = false; + } + } + } + + // calculate and store the mode choice logsum for the usual school + // location for this student at the various + // departure time and duration alternativees + setUnivTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, + departureTimeChoiceAvailability); + + if (household.getDebugChoiceModels()) + { + household.logTourObject(loggingHeader, modelLogger, person, t); + } + + float logsum = (float) univTourChoiceModel.computeUtilities(imtodDmuObject, imtodDmuObject.getIndexValues(), + departureTimeChoiceAvailability, schoolTourDepartureTimeChoiceSample); + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has no available alternatives, choose between + // the + // first and last alternative. + int chosen; + if (univTourChoiceModel.getAvailabilityCount() > 0) chosen = univTourChoiceModel + .getChoiceResult(rn); + else chosen = rn < 0.5 ? 1 : altStarts.length; + + // schedule the chosen alternative + int chosenStartPeriod = altStarts[chosen - 1]; + int chosenEndPeriod = altEnds[chosen - 1]; + try + { + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + } catch (Exception e) + { + logger.error("exception caught updating school tour TOD choice time windows."); + throw new RuntimeException(); + } + + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = univTourChoiceModel.getUtilities(); + double[] probabilities = univTourChoiceModel.getProbabilities(); + boolean[] availabilities = univTourChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString + + ", Tour Id: " + t.getTourId()); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < schoolTourChoiceModel.getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], + altEnds[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, + altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + univTourChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + univTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, + rn, chosen); + + // write UEC calculation results to separate model specific log + // file + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + univTourChoiceModel.logUECResults(modelLogger, loggingHeader, 200); + + } + + if (runModeChoice) + { + long check = System.nanoTime(); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); + + // use the mcModel object already setup for computing logsums + // and get + // the mode choice, where the selected + // school location and subzone and departure time and duration + // are + // set for this school tour. + int chosenMode = -1; + chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); + + t.setTourModeChoice(chosenMode); + + mcTime += (System.nanoTime() - check); + } + + } + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String + .format("Final University Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + return schoolTours.size(); + + } + + private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, + int startPeriod, int endPeriod) + { + + t.setTourDepartPeriod(startPeriod); + t.setTourArrivePeriod(endPeriod); + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(t); + mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), + t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); + + + + mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(t.getTourOrigMgra())); + mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(t.getTourOrigMgra())); + mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(t.getTourOrigMgra())); + + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(t.getTourDestMgra())); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(t.getTourDestMgra())); + + mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t + .getTourOrigMgra()))); + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t + .getTourDestMgra()))); + + mcDmuObject.setOriginMgra(t.getTourOrigMgra()); + mcDmuObject.setDestMgra(t.getTourDestMgra()); + + } + + public long getModeChoiceTime() + { + return mcTime; + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + TourModeChoiceModel mcModel = null; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + String[] tourPurposeList = {"White Collar", "Services", "Health", "Retail and Food", + "Blue Collar", "Military"}; + + HouseholdIndividualMandatoryTourDepartureAndDurationTime testObject = new HouseholdIndividualMandatoryTourDepartureAndDurationTime( + propertyMap, modelStructure, tourPurposeList, dmuFactory, mcModel); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java new file mode 100644 index 0000000..bce9e2c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java @@ -0,0 +1,414 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +/** + * Implements an invidual mandatory tour frequency model, which selects the + * number of work, school, or work and school tours for each person who selects + * a mandatory activity. There are essentially seven separate models, one for + * each person type (full-time worker, part-time worker, university student, non + * working adults, retired, driving students, and non-driving students), except + * pre-school students. The choices are one work tour, two work tours, one + * school tour, two school tours, and one work and school tour. Availability + * arrays are defined for each person type. + * + * The UEC for the model has two additional matrix calcuation tabs, which + * computes the one-way walk distance and the round-trip auto time to work + * and/or school for the model. This allows us to compute the work and/or school + * time, by setting the DMU destination index, just using the UEC. + * + * @author D. Ory + * + */ +public class HouseholdIndividualMandatoryTourFrequencyModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdIndividualMandatoryTourFrequencyModel.class); + private transient Logger tourFreq = Logger.getLogger("tourFreq"); + + private static final String IMTF_CONTROL_FILE_TARGET = "imtf.uec.file"; + private static final String IMTF_DATA_SHEET_TARGET = "imtf.data.page"; + private static final String IMTF_MODEL_SHEET_TARGET = "imtf.model.page"; + + private static final String MANDATORY_ACTIVITY = Definitions.MANDATORY_PATTERN; + + // model results + public static final int CHOICE_ONE_WORK = 1; + public static final int CHOICE_TWO_WORK = 2; + public static final int CHOICE_ONE_SCHOOL = 3; + public static final int CHOICE_TWO_SCHOOL = 4; + public static final int CHOICE_WORK_AND_SCHOOL = 5; + + public static final String[] CHOICE_RESULTS = {"1 Work", "2 Work", + "1 School", "2 School", "Wrk & Schl", "Worker Works At Home", "Student Works At Home", + "Worker School At Home", "Student School At Home" }; + + private IndividualMandatoryTourFrequencyDMU imtfDmuObject; + private ChoiceModelApplication choiceModelApplication; + + private AccessibilitiesTable accTable; + private MandatoryAccessibilitiesCalculator mandAcc; + + /** + * Constructor establishes the ChoiceModelApplication, which applies the + * logit model via the UEC spreadsheet, and it also establishes the UECs + * used to compute the one-way walk distance to work and/or school and the + * round-trip auto time to work and/or school. The model must be the first + * UEC tab, the one-way distance calculations must be the second UEC tab, + * round-trip time must be the third UEC tab. + * + * @param dmuObject + * is the UEC dmu object for this choice model + * @param uecFileName + * is the UEC control file name + * @param resourceBundle + * is the application ResourceBundle, from which a properties + * file HashMap will be created for the UEC + * @param tazDataManager + * is the object used to interact with the zonal data table + * @param modelStructure + * is the ModelStructure object that defines segmentation and + * other model structure relate atributes + */ + public HouseholdIndividualMandatoryTourFrequencyModel(HashMap propertyMap, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, + AccessibilitiesTable accTable, MandatoryAccessibilitiesCalculator myMandAcc) + { + + setupHouseholdIndividualMandatoryTourFrequencyModel(propertyMap, modelStructure, + dmuFactory, accTable, myMandAcc); + + } + + private void setupHouseholdIndividualMandatoryTourFrequencyModel( + HashMap propertyMap, ModelStructure modelStructure, + CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, + MandatoryAccessibilitiesCalculator myMandAcc) + { + + logger.info("setting up IMTF choice model."); + + accTable = myAccTable; + + // locate the individual mandatory tour frequency choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String imtfUecFile = propertyMap.get(IMTF_CONTROL_FILE_TARGET); + imtfUecFile = uecPath + imtfUecFile; + + int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTF_DATA_SHEET_TARGET); + int modelPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTF_MODEL_SHEET_TARGET); + + // get the dmu object from the factory + imtfDmuObject = dmuFactory.getIndividualMandatoryTourFrequencyDMU(); + + // set up the model + choiceModelApplication = new ChoiceModelApplication(imtfUecFile, modelPage, dataPage, + propertyMap, (VariableTable) imtfDmuObject); + + mandAcc = myMandAcc; + + } + + /** + * Applies the model for the array of households that are stored in the + * HouseholdDataManager. The results are summarized by person type. + * + * @param householdDataManager + * is the object containg the Household objects for which this + * model is to be applied. + */ + public void applyModel(Household household) + { + + Logger modelLogger = tourFreq; + if (household.getDebugChoiceModels()) + household.logHouseholdObject("Pre Individual Mandatory Tour Frequency Choice HHID=" + + household.getHhId() + " Object", modelLogger); + + int choice = -1; + + // get this household's person array + Person[] personArray = household.getPersons(); + + // set the household id, origin taz, hh taz, and debugFlag=false in the + // dmu + imtfDmuObject.setHousehold(household); + + // set the auto sufficiency dependent escort accessibility value for the + // household + String[] types = {"", "escort0", "escort1", "escort2"}; + int autoSufficiency = household.getAutoSufficiency(); + float accessibility = accTable.getAggregateAccessibility(types[autoSufficiency], + household.getHhMgra()); + imtfDmuObject.setEscortAccessibility(accessibility); + + // loop through the person array (1-based) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + String activity = person.getCdapActivity(); + + try + { + + // only apply the model for those with mandatory activities and + // not + // preschool children + if (person.getPersonIsPreschoolChild() == 0 + && activity.equalsIgnoreCase(MANDATORY_ACTIVITY)) + { + + // set the person + imtfDmuObject.setPerson(person); + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Mandatory Tour Frequency Choice Model:"); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s.", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + + choiceModelApplication.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "Individual Mandatory Tour Frequency Choice Model: Debug Statement for Household ID: " + + household.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + person.getPersonType() + "."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + } + + double distance = 999.0; + double time = 999.0; + if (person.getPersonIsWorker() == 1) + { + + int workMgra = person.getWorkLocation(); + if (workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + + double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( + household.getHhMgra(), workMgra, + household.getDebugChoiceModels(), tourFreq); + + distance = person.getWorkLocationDistance(); + time = accessibilities[0]; // sov time + // wt time + if (accessibilities[2] > 0.0 && accessibilities[2] < time) + time = accessibilities[2]; + // dt time + if (accessibilities[3] > 0.0 && accessibilities[3] < time) + time = accessibilities[3]; + + } else + { + // no work location; skip the rest if no school + // location. + int schoolMgra = person.getUsualSchoolLocation(); + if (schoolMgra <= 0 + || schoolMgra == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) + continue; + } + + } + imtfDmuObject.setDistanceToWorkLoc(distance); + imtfDmuObject.setBestTimeToWorkLoc(time); + + distance = 999.0; + if (person.getPersonIsUniversityStudent() == 1 + || person.getPersonIsStudentDriving() == 1 + || person.getPersonIsStudentNonDriving() == 1) + { + + int schoolMgra = person.getUsualSchoolLocation(); + if (schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) + { + distance = person.getSchoolLocationDistance(); + } else + { + // no school location; skip the rest if no work + // location. + int workMgra = person.getWorkLocation(); + if (workMgra <= 0 + || workMgra == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + continue; + } + + } + imtfDmuObject.setDistanceToSchoolLoc(distance); + + // compute the utilities + IndexValues index = imtfDmuObject.getIndexValues(); + float logsum = (float) choiceModelApplication.computeUtilities(imtfDmuObject, index); + person.setImtfLogsum(logsum); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, + // make choice. + if (choiceModelApplication.getAvailabilityCount() > 0) choice = choiceModelApplication + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for j=%d, activity=%s, HHID=%d, no available alternatives to choose from in choiceModelApplication.", + j, activity, household.getHhId())); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = choiceModelApplication.getUtilities(); + double[] probabilities = choiceModelApplication.getProbabilities(); + + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + + person.getPersonType()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < probabilities.length; ++k) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %10s", k + 1, CHOICE_RESULTS[k]); + modelLogger.info(String.format("%-15s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %10s", choice, + CHOICE_RESULTS[choice - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + choiceModelApplication.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + choiceModelApplication.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, choice); + + // write UEC calculation results to separate model + // specific + // log file + choiceModelApplication.logUECResults(modelLogger, loggingHeader); + + } + + person.setImtfChoice(choice); + + // set the person choices + if (choice == CHOICE_ONE_WORK) + { + person.createWorkTours(1, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, + ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); + } else if (choice == CHOICE_TWO_WORK) + { + person.createWorkTours(2, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, + ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); + } else if (choice == CHOICE_ONE_SCHOOL) + { + if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(1, + 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, + ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); + else person.createSchoolTours(1, 0, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); + } else if (choice == CHOICE_TWO_SCHOOL) + { + if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(2, + 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, + ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); + else person.createSchoolTours(2, 0, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); + } else if (choice == CHOICE_WORK_AND_SCHOOL) + { + person.createWorkTours(1, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, + ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); + if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(1, + 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, + ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); + else person.createSchoolTours(1, 0, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); + } + + } else if (activity.equalsIgnoreCase(MANDATORY_ACTIVITY) + && person.getPersonIsPreschoolChild() == 1) + { + // mandatory activity if + // pre-school child with mandatory activity type is assigned + // choice = 3 (1 school tour). + choice = 3; + + person.setImtfChoice(choice); + + // get the school purpose name for a non-driving age person + // to + // use for preschool tour purpose + person.createSchoolTours(1, 0, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); + } + + } catch (Exception e) + { + logger.error(String.format("Exception caught for j=%d, activity=%s, HHID=%d", j, + activity, household.getHhId())); + throw new RuntimeException(); + } + + } // j (person loop) + + household.setImtfRandomCount(household.getHhRandomCount()); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java new file mode 100644 index 0000000..a0cf96e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java @@ -0,0 +1,823 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Map; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +/** + * Implements an invidual mandatory tour frequency model, which selects the + * number of work, school, or work and school tours for each person who selects + * a mandatory activity. There are essentially seven separate models, one for + * each person type (full-time worker, part-time worker, university student, non + * working adults, retired, driving students, and non-driving students), except + * pre-school students. The choices are one work tour, two work tours, one + * school tour, two school tours, and one work and school tour. Availability + * arrays are defined for each person type. + * + * The UEC for the model has two additional matrix calcuation tabs, which + * computes the one-way walk distance and the round-trip auto time to work + * and/or school for the model. This allows us to compute the work and/or school + * time, by setting the DMU destination index, just using the UEC. + * + * @author D. Ory + * + */ +public class HouseholdIndividualNonMandatoryTourFrequencyModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(HouseholdIndividualNonMandatoryTourFrequencyModel.class); + private transient Logger tourFreq = Logger.getLogger("tourFreq"); + + private static final String UEC_DATA_PAGE_KEY = "inmtf.data.page"; + private static final String UEC_PERSONTYPE_1_PAGE_KEY = "inmtf.perstype1.page"; + private static final String UEC_PERSONTYPE_2_PAGE_KEY = "inmtf.perstype2.page"; + private static final String UEC_PERSONTYPE_3_PAGE_KEY = "inmtf.perstype3.page"; + private static final String UEC_PERSONTYPE_4_PAGE_KEY = "inmtf.perstype4.page"; + private static final String UEC_PERSONTYPE_5_PAGE_KEY = "inmtf.perstype5.page"; + private static final String UEC_PERSONTYPE_6_PAGE_KEY = "inmtf.perstype6.page"; + private static final String UEC_PERSONTYPE_7_PAGE_KEY = "inmtf.perstype7.page"; + private static final String UEC_PERSONTYPE_8_PAGE_KEY = "inmtf.perstype8.page"; + + private static final String HOME_ACTIVITY = Definitions.HOME_PATTERN; + + private static final String PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ = "inmtf.uec.file"; + + private static final String PROPERTIES_TOUR_FREQUENCY_EXTENSION_PROBABILITIES_FILE = "inmtf.FrequencyExtension.ProbabilityFile"; + + private static final int AUTO_LOGSUM_INDEX = 6; + + private AccessibilitiesTable accTable; + private MandatoryAccessibilitiesCalculator mandAcc; + + private HashMap purposeIndexToNameMap; + private HashMap personTypeIndexToModelIndexMap; + + private IndividualNonMandatoryTourFrequencyDMU dmuObject; + private ChoiceModelApplication[] choiceModelApplication; + private TableDataSet alternativesTable; + + private Map tourFrequencyIncreaseProbabilityMap; + private int[] maxTourFrequencyChoiceList; + + private static String[] escortTypes = { + "", "escort0", "escort1", "escort2" }; + private static String[] shopTypes = { + "", "shop0", "shop1", "shop2" }; + private static String[] maintTypes = { + "", "maint0", "maint1", "maint2" }; + private static String[] discrTypes = { + "", "discr0", "discr1", "discr2" }; + private static String[] eatOutTypes = { + "", "eatOut0", "eatOut1", "eatOut2" }; + private static String[] visitTypes = { + "", "visit0", "visit1", "visit2" }; + + /** + * Constructor establishes the ChoiceModelApplication, which applies the + * logit model via the UEC spreadsheet, and it also establishes the UECs + * used to compute the one-way walk distance to work and/or school and the + * round-trip auto time to work and/or school. The model must be the first + * UEC tab, the one-way distance calculations must be the second UEC tab, + * round-trip time must be the third UEC tab. + * + * @param dmuObject + * is the UEC dmu object for this choice model + * @param uecFileName + * is the UEC control file name + * @param resourceBundle + * is the application ResourceBundle, from which a properties + * file HashMap will be created for the UEC + * @param tazDataManager + * is the object used to interact with the zonal data table + * @param modelStructure + * is the ModelStructure object that defines segmentation and + * other model structure relate atributes + */ + + public HouseholdIndividualNonMandatoryTourFrequencyModel(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, + MandatoryAccessibilitiesCalculator myMandAcc) + { + + accTable = myAccTable; + mandAcc = myMandAcc; + + setUpModels(propertyMap, dmuFactory); + + } + + private void setUpModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up %s tour frequency choice model.", + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String inmtfUecFile = propertyMap.get(PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ); + String uecFileName = uecPath + inmtfUecFile; + + dmuObject = dmuFactory.getIndividualNonMandatoryTourFrequencyDMU(); + + personTypeIndexToModelIndexMap = new HashMap(); + + int dataSheet = Integer.parseInt(propertyMap.get(UEC_DATA_PAGE_KEY)); + int sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_1_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(1, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_2_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(2, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_3_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(3, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_4_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(4, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_5_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(5, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_6_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(6, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_7_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(7, sheet); + sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_8_PAGE_KEY)); + personTypeIndexToModelIndexMap.put(8, sheet); + + choiceModelApplication = new ChoiceModelApplication[personTypeIndexToModelIndexMap.size() + 1]; // one + // choice + // model + // for + // each + // person + // type + // that + // has model specified; Ones indexing for + // personType. + + // set up the model + for (int i : personTypeIndexToModelIndexMap.keySet()) + choiceModelApplication[i] = new ChoiceModelApplication(uecFileName, + personTypeIndexToModelIndexMap.get(i), dataSheet, propertyMap, + (VariableTable) dmuObject); + + // the alternatives are the same for each person type; use the first + // choiceModelApplication to get its uec and from it, get the + // TableDataSet + // of alternatives + // to use to determine which tour purposes should be generated for the + // chose + // alternative. + int ftIndex = personTypeIndexToModelIndexMap.get(1); + alternativesTable = choiceModelApplication[ftIndex].getUEC().getAlternativeData(); + + // check the field names in the alternatives table; make sure the their + // order + // is as expected. + String[] fieldNames = alternativesTable.getColumnLabels(); + + // create a mapping between names used in lookup file and purpose names + // used + // in model + HashMap primaryPurposeMap = new HashMap(); + primaryPurposeMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME); + primaryPurposeMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME); + primaryPurposeMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME); + primaryPurposeMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME); + primaryPurposeMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME); + primaryPurposeMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME, + dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME); + + purposeIndexToNameMap = new HashMap(); + purposeIndexToNameMap.put(1, ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); + purposeIndexToNameMap.put(2, ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); + purposeIndexToNameMap.put(3, ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); + purposeIndexToNameMap.put(4, ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); + purposeIndexToNameMap.put(5, ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); + purposeIndexToNameMap.put(6, ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); + + if (!fieldNames[0].equalsIgnoreCase("a") && !fieldNames[0].equalsIgnoreCase("alt")) + { + logger.error("error while checking order of fields in IndividualNonMandatoryTourFrequencyModel alternatives file."); + logger.error(String + .format("first field expected to be 'a' or 'alt' (case insensitive), but %s was found instead.", + fieldNames[0])); + throw new RuntimeException(); + } else + { + + for (int i : purposeIndexToNameMap.keySet()) + { + String primaryName = purposeIndexToNameMap.get(i).trim(); + String name = primaryPurposeMap.get(primaryName).trim(); + if (!fieldNames[i].equalsIgnoreCase(name)) + { + logger.error("error while checking order of fields in IndividualNonMandatoryTourFrequencyModel alternatives file."); + logger.error(String + .format("field %d expected to be '%s' (case insensitive), but %s was found instead.", + i, name, fieldNames[i])); + throw new RuntimeException(); + } + } + } + + // load data used for tour frequency extension model + loadIndividualNonMandatoryIncreaseModelData(uecPath + + propertyMap.get(PROPERTIES_TOUR_FREQUENCY_EXTENSION_PROBABILITIES_FILE)); + + } + + /** + * Applies the model for the array of households that are stored in the + * HouseholdDataManager. The results are summarized by person type. + * + * @param householdDataManager + * is the object containg the Household objects for which this + * model is to be applied. + */ + public void applyModel(Household household) + { + + int modelIndex = -1; + int choice = -1; + String personTypeString = "Missing"; + + Logger modelLogger = tourFreq; + if (household.getDebugChoiceModels()) + household.logHouseholdObject("Pre Individual Non-Mandatory Tour Frequency Choice HHID=" + + household.getHhId() + " Object", modelLogger); + + // this will be an array with values 1 -> tours.length being the number + // of + // non-mandatory tours in each category + // this keeps it consistent with the way the alternatives are held in + // the + // alternatives file/arrays + float[] tours = null; + + // get this household's person array + Person[] personArray = household.getPersons(); + + // set the household id, origin taz, hh taz, and debugFlag=false in the + // dmu + dmuObject.setHouseholdObject(household); + + // set the auto sufficiency dependent non-mandatory accessibility values + // for + // the household + int autoSufficiency = household.getAutoSufficiency(); + dmuObject.setShopAccessibility(accTable.getAggregateAccessibility( + shopTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setMaintAccessibility(accTable.getAggregateAccessibility( + maintTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setEatOutAccessibility(accTable.getAggregateAccessibility( + eatOutTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setVisitAccessibility(accTable.getAggregateAccessibility( + visitTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setDiscrAccessibility(accTable.getAggregateAccessibility( + discrTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setEscortAccessibility(accTable.getAggregateAccessibility( + escortTypes[autoSufficiency], household.getHhMgra())); + + dmuObject.setNmDcLogsum(accTable.getAggregateAccessibility("nonmotor", + household.getHhMgra())); + + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + // loop through the person array (1-based) + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + String activity = person.getCdapActivity(); + + try + { + + // only apply the model if person does not have H daily activity + // pattern + if (!activity.equalsIgnoreCase(HOME_ACTIVITY)) + { + + // set the person + dmuObject.setPersonObject(person); + + if (household.getDebugChoiceModels()) + { + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + // set the availability array for the tour frequency model + // same number of alternatives for each person type, so use + // person type 1 to get num alts. + int numberOfAlternatives = choiceModelApplication[1].getNumberOfAlternatives(); + boolean[] availabilityArray = new boolean[numberOfAlternatives + 1]; + Arrays.fill(availabilityArray, true); + + modelIndex = personTypeIndexToModelIndexMap.get(person.getPersonTypeNumber()); + personTypeString = person.getPersonType(); + + int workMgra = person.getWorkLocation(); + double accessibility = 0.0; + if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( + household.getHhMgra(), workMgra, household.getDebugChoiceModels(), + tourFreq); + accessibility = accessibilities[AUTO_LOGSUM_INDEX]; + } + dmuObject.setWorkAccessibility(accessibility); + + int schoolMgra = person.getUsualSchoolLocation(); + accessibility = 0.0; + if (schoolMgra > 0 && schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) + { + double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( + household.getHhMgra(), schoolMgra, + household.getDebugChoiceModels(), tourFreq); + accessibility = accessibilities[AUTO_LOGSUM_INDEX]; + } + dmuObject.setSchoolAccessibility(accessibility); + + // person.computeIdapResidualWindows(); + + // create the sample array + int[] sampleArray = new int[availabilityArray.length]; + Arrays.fill(sampleArray, 1); + + // compute the utilities + dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), + household.getHhTaz(), -1); + + if (household.getDebugChoiceModels()) + { + + // write debug header + choiceModelDescription = String + .format("Individual Non-Mandatory Tour Frequency Orignal Choice Model:"); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType()); + choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = choiceModelDescription + " for " + decisionMakerLabel + + "."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, + dmuObject.getDmuIndexValues(), availabilityArray, sampleArray); + person.setInmtfLogsum(logsum); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, + // make choice. + if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for j=%d, activity=%s, HHID=%d, no Non-Mandatory Tour Frequency alternatives available to choose from in choiceModelApplication.", + j, activity, person.getHouseholdObject().getHhId())); + throw new RuntimeException(); + } + + // create the non-mandatory tour objects for the choice + // made. + // createIndividualNonMandatoryTours ( person, choice ); + tours = runIndividualNonMandatoryToursIncreaseModel(person, choice); + createIndividualNonMandatoryTours_new(person, tours); + + // debug output + if (household.getDebugChoiceModels()) + { + + String[] alternativeNames = choiceModelApplication[modelIndex] + .getAlternativeNames(); + double[] utilities = choiceModelApplication[modelIndex].getUtilities(); + double[] probabilities = choiceModelApplication[modelIndex] + .getProbabilities(); + + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + + personTypeString); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("--------------------------------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < alternativeNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %-66s", k + 1, + getAlternativeNameFromChoice(k + 1)); + modelLogger.info(String.format("%-72s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %s", choice, + getAlternativeNameFromChoice(choice)); + modelLogger.info(String.format( + "Original Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + altString = String.format("%-3d %s", choice, + getAlternativeNameFromModifiedChoice(tours)); + modelLogger.info(String.format("Revised Choice After Increase: %s", + altString)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + choiceModelApplication[modelIndex].logAlternativesInfo( + choiceModelDescription, decisionMakerLabel); + choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, choice); + + loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; + + // write UEC calculation results to separate model + // specific + // log file + choiceModelApplication[modelIndex] + .logUECResults(modelLogger, loggingHeader); + + } + + person.setInmtfChoice(choice); + + } + + } catch (Exception e) + { + logger.error(String.format("Exception caught for j=%d, activity=%s, HHID=%d", j, + activity, household.getHhId())); + throw new RuntimeException(e); + } + + } // j (person loop) + + household.setInmtfRandomCount(household.getHhRandomCount()); + + } + + private String getAlternativeNameFromChoice(int choice) + { + + // use the 1s based choice value as the table row number + float[] rowValues = alternativesTable.getRowValues(choice); + + String altName = ""; + + // rowValues is a 0s based indexed array, but the first field is the + // alternative number, + // and subsequent fields indicate the number of tours to be generated + // for the + // purpose corresponding to the field. + for (int i = 1; i < rowValues.length; i++) + { + + int numTours = (int) rowValues[i]; + if (numTours == 0) continue; + + String purposeName = purposeIndexToNameMap.get(i); + if (altName.length() == 0) altName = String.format(", %d %s", numTours, purposeName); + else altName += String.format(", %d %s", numTours, purposeName); + } + + if (altName.length() == 0) altName = "no tours"; + + return altName; + } + + private String getAlternativeNameFromModifiedChoice(float[] rowValues) + { + + String altName = ""; + + // rowValues is a 0s based indexed array, but the first field is the + // alternative number, + // and subsequent fields indicate the nuimber of tours to be generated + // for + // the purpose corresponding to the field. + for (int i = 1; i < rowValues.length; i++) + { + + int numTours = (int) rowValues[i]; + if (numTours == 0) continue; + + String purposeName = purposeIndexToNameMap.get(i); + if (altName.length() == 0) altName = String.format(", %d %s", numTours, purposeName); + else altName += String.format(", %d %s", numTours, purposeName); + } + + if (altName.length() == 0) altName = "no tours"; + + return altName; + } + + /** + * Logs the results of the model. + * + * public void logResults(){ + * + * logger.info(" "); logger.info( + * "~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" + * ); logger.info("Individual Non-Mandatory Tour Frequency Model Results"); + * + * // count of model results logger.info(" "); String firstHeader = + * "Person type "; String secondHeader = + * "----------------------------- "; + * + * + * + * String[] purposeNames = alternativesTable.getColumnLabels(); int[] + * columnTotals = new int[purposeNames.length-1]; + * + * for( int i=0; i < columnTotals.length; i++ ) { firstHeader += + * String.format("%12s", purposeNames[i+1]); secondHeader += "----------- "; + * } + * + * firstHeader += String.format("%12s","Total"); secondHeader += + * "-----------"; + * + * logger.info(firstHeader); logger.info(secondHeader); + * + * + * + * int lineTotal = 0; for(int i=0;i(); + TableDataSet probabilityTable; + try + { + probabilityTable = (new CSVFileReader().readFile(new File(filePath))); + } catch (IOException e) + { + logger.error( + "Exception caught reading Individual Non-Mandatory Tour Frequency extension probability table.", + e); + throw new RuntimeException(e); + } + + String personTypeColumnName = "person_type"; + String mandatoryTourParticipationColumnName = "mandatory_tour"; + String jointTourParticipationColumnName = "joint_tour"; + String nonMandatoryTourTypeColumn = "nonmandatory_tour_type"; + String zeroAdditionalToursColumnName = "0_tours"; + String oneAdditionalToursColumnName = "1_tours"; + String twoAdditionalToursColumnName = "2_tours"; + + for (int i = 1; i <= probabilityTable.getRowCount(); i++) + { + int key = getTourIncreaseTableKey( + (int) probabilityTable.getValueAt(i, nonMandatoryTourTypeColumn), + (int) probabilityTable.getValueAt(i, personTypeColumnName), + ((int) probabilityTable.getValueAt(i, mandatoryTourParticipationColumnName)) == 1, + ((int) probabilityTable.getValueAt(i, jointTourParticipationColumnName)) == 1); + tourFrequencyIncreaseProbabilityMap.put(key, + new float[] {probabilityTable.getValueAt(i, zeroAdditionalToursColumnName), + probabilityTable.getValueAt(i, oneAdditionalToursColumnName), + probabilityTable.getValueAt(i, twoAdditionalToursColumnName)}); + } + } + + private float[] runIndividualNonMandatoryToursIncreaseModel(Person person, int choice) + { + // use the 1s based choice value as the table row number + // rowValues is a 0s based indexed array, but the first field is the + // alternative number, + // and subsequent fields indicate the nuimber of tours to be generated + // for + // the purpose corresponding to the field. + + Household household = person.getHouseholdObject(); + + int personType = person.getPersonTypeNumber(); + boolean participatedInMandatoryTour = person.getListOfWorkTours().size() > 0 + || person.getListOfSchoolTours().size() > 0; + + boolean participatedInJointTour = false; + Tour[] jointTours = person.getHouseholdObject().getJointTourArray(); + if (jointTours != null) + { + for (Tour t : jointTours) + { + if (t.getPersonInJointTour(person)) + { + participatedInJointTour = true; + break; + } + } + } + + float[] rowValues = null; + + boolean notDone = true; + while (notDone) + { + + rowValues = alternativesTable.getRowValues(choice); + + int firstCount = tourCountSum(rowValues); + if (firstCount == 0 || firstCount >= 5) // if 0 or 5+ tours already, + // we + // are done + break; + + for (int i = 1; i < rowValues.length; i++) + { + + if (rowValues[i] < maxTourFrequencyChoiceList[i]) continue; + + int newChoice = -1; + int key = getTourIncreaseTableKey(i, personType, participatedInMandatoryTour, + participatedInJointTour); + float[] probabilities = tourFrequencyIncreaseProbabilityMap.get(key); + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + for (int j = 0; j < probabilities.length; j++) + { + if (rn <= probabilities[j]) + { + rowValues[i] += j; + newChoice = j; + break; + } + } + + // debug output + if (household.getDebugChoiceModels()) + { + + Logger modelLogger = tourFreq; + String[] alternativeNames = {"no additional", "1 additional", "2 additional"}; + + modelLogger + .info("Individual Non-Mandatory Tour Frequency Increase Choice for tour purposeName=" + + purposeIndexToNameMap.get(i) + ", purposeIndex=" + i); + modelLogger.info("Alternative Probability CumProb"); + modelLogger.info("--------------- -------------- --------------"); + + // probabilities array has tour frequency extension + // probabilities alread stored as cumulative probabilities. + // calculate alternative probabilities from cumulative for + // logging. + double prob = probabilities[0]; + double cumProb = probabilities[0]; + for (int k = 1; k < alternativeNames.length; k++) + { + cumProb = probabilities[k]; + prob = probabilities[k] - probabilities[k - 1]; + String altString = String.format("%-3d %-15s", k + 1, alternativeNames[k]); + modelLogger.info(String.format("%-20s%18.6e%18.6e", altString, prob, + cumProb)); + } + + modelLogger.info(String.format("choice: %s, with rn=%.8f, randomCount=%d", + newChoice + 1, rn, randomCount)); + + modelLogger.info(""); + modelLogger.info(""); + + } + + } + notDone = tourCountSum(rowValues) > 5; + } + + return rowValues; + } + + private int tourCountSum(float[] tours) + { + // tours are located in indices 1 -> tours.length + int sum = 0; + for (int i = 1; i < tours.length; i++) + sum += tours[i]; + return sum; + } + + private int getTourIncreaseTableKey(int nonMandatoryTourType, int personType, + boolean participatedInMandatoryTour, boolean participatedInJointTour) + { + return nonMandatoryTourType + 100 * personType + 10000 + * (participatedInMandatoryTour ? 1 : 0) + 100000 + * (participatedInJointTour ? 1 : 0); + } + + private void createIndividualNonMandatoryTours_new(Person person, float[] tours) + { + // person.clearIndividualNonMandatoryToursArray(); + + for (int i = 1; i < tours.length; i++) + { + int numTours = (int) tours[i]; + if (numTours > 0) + person.createIndividualNonMandatoryTours(numTours, purposeIndexToNameMap.get(i)); + } + } + + // private void createIndividualNonMandatoryTours ( Person person, int + // choice ) { + // + // // use the 1s based choice value as the table row number + // float[] rowValues = alternativesTable.getRowValues( choice ); + // + // // rowValues is a 0s based indexed array, but the first field is the + // alternative number, + // // and subsequent fields indicate the nuimber of tours to be generated + // for the + // purpose corresponding to the field. + // for ( int i=1; i < rowValues.length; i++ ) { + // + // int numTours = (int)rowValues[i]; + // if ( numTours == 0 ) + // continue; + // + // String purposeName = purposeIndexToNameMap.get(i); + // + // person.createIndividualNonMandatoryTours( numTours, i, purposeName ); + // } + // + // } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java new file mode 100644 index 0000000..9b6d466 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java @@ -0,0 +1,82 @@ +package org.sandag.abm.ctramp; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagTourBasedModel; + +public class HouseholdValidator +{ + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + + public static boolean vailidateWorkLocChoices(Household hh) + { + boolean result = true; + + for (int j = 1; j <= hh.getHhSize(); j++) + { + Person p = hh.getPerson(j); + int windex = p.getWorkLocationSegmentIndex(); + int wmgra = p.getWorkLocation(); + if (windex != -1 && wmgra == 0) + { + result = false; + logger.info("Invalid work location choice for " + p.getPersonId() + " a " + + p.getAge() + " years old with work segment index " + + p.getWorkLocationSegmentIndex() + " resubmitting job......"); + break; + } + } + return result; + } + + public static boolean vailidateSchoolLocChoices(Household hh) + { + boolean result = true; + + for (int j = 1; j <= hh.getHhSize(); j++) + { + Person p = hh.getPerson(j); + int sindex = p.getSchoolLocationSegmentIndex(); + int smgra = p.getPersonSchoolLocationZone(); + if (sindex != -1 && smgra == 0) + { + result = false; + logger.fatal("Invalid school location choice for " + p.getPersonId() + " a " + + p.getAge() + " years old with school segment index " + + p.getSchoolLocationSegmentIndex() + " resubmitting job....."); + break; + } + } + return result; + } + + public static boolean validateMandatoryDestinationChoices(Household[] householdArray, + String type) + { + boolean result = true; + if (type.equalsIgnoreCase("work")) + { + for (int i = 0; i < householdArray.length; i++) + { + if (!vailidateWorkLocChoices(householdArray[i])) + { + result = false; + break; + } + } + } else if (type.equalsIgnoreCase("school")) + { + for (int i = 0; i < householdArray.length; i++) + { + if (!vailidateSchoolLocChoices(householdArray[i])) + { + result = false; + break; + } + } + } else + { + logger.fatal("invalide mandatory destination choice type:" + type); + } + return result; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java new file mode 100644 index 0000000..6098701 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java @@ -0,0 +1,301 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class IndividualMandatoryTourFrequencyDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(IndividualMandatoryTourFrequencyDMU.class); + + protected HashMap methodIndexMap; + + protected IndexValues dmuIndex; + protected Household household; + protected Person person; + + protected double walkDistanceToWork, walkDistanceToSchool; + protected double roundTripAutoTimeToWork, roundTripAutoTimeToSchool; + + protected double distanceToWork; + protected double timeToWork; + protected double distanceToSchool; + protected double escortAccessibility; + + private int homeTazAreaType; + + public IndividualMandatoryTourFrequencyDMU() + { + dmuIndex = new IndexValues(); + } + + public IndexValues getIndexValues() + { + return dmuIndex; + } + + public void setHousehold(Household passedInHousehold) + { + household = passedInHousehold; + + // set the origin and zone indices + dmuIndex.setOriginZone(household.getHhMgra()); + dmuIndex.setZoneIndex(household.getHhMgra()); + dmuIndex.setHHIndex(household.getHhId()); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (household.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug IMTF UEC"); + } + + } + + public void setHomeTazAreaType(int at) + { + homeTazAreaType = at; + } + + public void setDestinationZone(int destinationZone) + { + dmuIndex.setDestZone(destinationZone); + } + + public void setPerson(Person passedInPerson) + { + person = passedInPerson; + } + + public void setDistanceToWorkLoc(double distance) + { + distanceToWork = distance; + } + + public void setBestTimeToWorkLoc(double time) + { + timeToWork = time; + } + + public void setDistanceToSchoolLoc(double distance) + { + distanceToSchool = distance; + } + + public void setEscortAccessibility(double accessibility) + { + escortAccessibility = accessibility; + } + + public double getDistanceToWorkLocation() + { + return distanceToWork; + } + + public double getBestTimeToWorkLocation() + { + return timeToWork; + } + + public double getDistanceToSchoolLocation() + { + return distanceToSchool; + } + + public double getEscortAccessibility() + { + return escortAccessibility; + } + + public int getFullTimeWorker() + { + return (person.getPersonTypeIsFullTimeWorker()); + } + + public int getPartTimeWorker() + { + return (person.getPersonTypeIsPartTimeWorker()); + } + + public int getUniversityStudent() + { + return (person.getPersonIsUniversityStudent()); + } + + public int getNonWorkingAdult() + { + return (person.getPersonIsNonWorkingAdultUnder65()); + } + + public int getRetired() + { + return (person.getPersonIsNonWorkingAdultOver65()); + } + + public int getDrivingAgeSchoolChild() + { + return (person.getPersonIsStudentDriving()); + } + + public int getPreDrivingAgeSchoolChild() + { + return (person.getPersonIsStudentNonDriving()); + } + + public int getFemale() + { + return (person.getPersonIsFemale()); + } + + public int getPersonType() + { + return person.getPersonTypeNumber(); + } + + public int getAge() + { + return (person.getAge()); + } + + public int getStudentIsEmployed() + { + + if (person.getPersonIsUniversityStudent() == 1 || person.getPersonIsStudentDriving() == 1) + { + return (person.getPersonIsWorker()); + } + + return (0); + } + + public int getNonStudentGoesToSchool() + { + + if (person.getPersonTypeIsFullTimeWorker() == 1 + || person.getPersonTypeIsPartTimeWorker() == 1 + || person.getPersonIsNonWorkingAdultUnder65() == 1 + || person.getPersonIsNonWorkingAdultOver65() == 1) + { + + return (person.getPersonIsStudent()); + } + + return (0); + + } + + public int getNotEmployed() + { + return person.notEmployed(); + } + + public int getNumberOfChildren6To18WithoutMandatoryActivity() + { + return household.getNumberOfChildren6To18WithoutMandatoryActivity(); + } + + public int getAutos() + { + return (household.getAutosOwned()); + } + + public int getDrivers() + { + return (household.getDrivers()); + } + + public int getPreschoolChildren() + { + return household.getNumPreschool(); + } + + public int getNonWorkers() + { + return (household.getNumberOfNonWorkingAdults()); + } + + public int getIncomeInDollars() + { + return (household.getIncomeInDollars()); + } + + public int getIncomeHigherThan50k() + { + if (household.getIncomeCategory() > 2) return (1); + return (0); + } + + public int getNonFamilyHousehold() + { + if (household.getIsNonFamilyHousehold() == 1 || household.getIsGroupQuarters() == 1) return 1; + else return 0; + } + + public int getAreaType() + { + return homeTazAreaType; + } + + public int getUsualWorkLocation() + { + return person.getWorkLocation(); + } + + public int getWorkAtHome() + { + return person.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR ? 1 + : 0; + } + + public int getSchoolAtHome() + { + return person.getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX ? 1 + : 0; + } + + public int getTelecommuteFrequency() { + return person.getTelecommuteChoice(); + } + + + public int getUsualSchoolLocation() + { + return person.getUsualSchoolLocation(); + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java new file mode 100644 index 0000000..317e940 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java @@ -0,0 +1,773 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class IndividualNonMandatoryTourFrequencyDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(IndividualNonMandatoryTourFrequencyDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Person person; + protected IndexValues dmuIndex; + + protected int homeTazIsUrban; + protected double distanceToWork; + protected double distanceToSchool; + protected double escortAccessibility; + protected double shopAccessibility; + protected double maintAccessibility; + protected double eatOutAccessibility; + protected double visitAccessibility; + protected double discrAccessibility; + protected double nmDcLogsum; + protected double workAccessibility; + protected double schoolAccessibility; + + public String TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME = "Escort"; + public String TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME = "Shopping"; + public String TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME = "Maint"; + public String TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME = "EatOut"; + public String TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME = "Visit"; + public String TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME = "Discr"; + + public IndividualNonMandatoryTourFrequencyDMU() + { + dmuIndex = new IndexValues(); + } + + public Household getHouseholdObject() + { + return hh; + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setPersonObject(Person persObject) + { + person = persObject; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug INMTF UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return household income in dollars + */ + public int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + /** + * @return the number of persons in the "decision making" household. + */ + public int getHouseholdSize() + { + // 1-based indexing, so the array is dimensioned 1 more than the number + // of + // persons. + return hh.getPersons().length - 1; + } + + public int getNumAutos() + { + return hh.getAutosOwned(); + } + + /** + * @return 1 if household has at least 1 car, and the number of cars equals + * the number of workers + */ + public int getCarsEqualsWorkers() + { + int numAutos = hh.getAutosOwned(); + int numWorkers = hh.getWorkers(); + + // household must have at least 1 car, otherwise return 0. + if (numAutos > 0) + { + // if at least one car and numWorkers == numAutos, return 1, + // otherwise 0. + if (numAutos == numWorkers) return 1; + else return 0; + } else + { + return 0; + } + } + + /** + * @return 1 if household has at least 1 car, and the number of cars equals + * the number of workers + */ + public int getMoreCarsThanWorkers() + { + int numAutos = hh.getAutosOwned(); + int numWorkers = hh.getWorkers(); + + if (numAutos > numWorkers) return 1; + else return 0; + } + + public int getNumAdults() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + num += persons[i].getPersonIsAdult(); + return num; + } + + public int getNumChildren() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + num += (persons[i].getPersonIsAdult() == 0 ? 1 : 0); + return num; + } + + public int getPersonIsAdult() + { + return person.getPersonIsAdult(); + } + + public int getPersonIsChild() + { + return person.getPersonIsAdult() == 0 ? 1 : 0; + } + + public int getPersonIsFullTimeWorker() + { + return person.getPersonIsFullTimeWorker(); + } + + public int getPersonIsPartTimeWorker() + { + return person.getPersonIsPartTimeWorker(); + } + + public int getPersonIsUniversity() + { + return person.getPersonIsUniversityStudent(); + } + + public int getPersonIsNonworker() + { + return person.getPersonIsNonWorkingAdultUnder65() + + person.getPersonIsNonWorkingAdultOver65(); + } + + public int getPersonIsPreschool() + { + return person.getPersonIsPreschoolChild(); + } + + public int getPersonIsStudentNonDriving() + { + return person.getPersonIsStudentNonDriving(); + } + + public int getPersonIsStudentDriving() + { + return person.getPersonIsStudentDriving(); + } + + public int getPersonStaysHome() + { + return person.getCdapActivity().equalsIgnoreCase("H") ? 1 : 0; + } + + public int getWorksAtHome() + { + if (person.getPersonIsWorker() == 1 + && person.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 1; + else return 0; + } + + public int getFemale() + { + return person.getPersonIsFemale(); + } + + /** + * determines the number of persons in the "decision making" household of + * type: full-time worker. returns the count, or 3, if count is 3 or more. + * + * @return count (up to a max of 3) of the number of full-time workers. + */ + public int getFullTimeWorkers() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: full time worker; if more than 3, + // return + // 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsFullTimeWorker(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: part-time worker. returns the count, or 3, if count is 3 or more. + * + * @return count (up to a max of 3) of the number of part-time workers. + */ + public int getPartTimeWorkers() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: part-time worker; if more than 3, + // return + // 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsPartTimeWorker(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: university student. returns the count, or 3, if count is 3 or more. + * + * @return count (up to a max of 3) of the number of university students. + */ + public int getUniversityStudents() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: university student; if more than 3, + // return 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsUniversityStudent(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: non-worker. returns the count, or 3, if count is 3 or more. + * + * @return count (up to a max of 3) of the number of non-workers. + */ + public int getNonWorkers() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: nonworker + retired; if more than + // 3, + // return 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsNonWorkingAdultUnder65() + + p[i].getPersonIsNonWorkingAdultOver65(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: driving-age student. returns the count, or 3, if count is 3 or + * more. + * + * @return count (up to a max of 3) of the number of driving-age students. + */ + public int getDrivingAgeStudents() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: driving-age student; if more than + // 3, + // return 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsStudentDriving(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: non-driving-age student. returns the count, or 3, if count is 3 or + * more. + * + * @return count (up to a max of 3) of the number of non-driving-age + * students. + */ + public int getNonDrivingAgeStudents() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: non-driving-age student; if more + // than 3, + // return 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsStudentNonDriving(); + if (count == 3) break; + } + + return count; + } + + /** + * determines the number of persons in the "decision making" household of + * type: pre-school age. returns the count, or 3, if count is 3 or more. + * + * @return count (up to a max of 3) of the number of pre-school age + * children. + */ + public int getPreSchoolers() + { + Person[] p = hh.getPersons(); + + // get the count of persons of type: pre-school; if more than 3, return + // 3. + int count = 0; + for (int i = 1; i < p.length; i++) + { + count += p[i].getPersonIsPreschoolChild(); + if (count == 3) break; + } + + return count; + } + + public void setEscortAccessibility(double accessibility) + { + escortAccessibility = accessibility; + } + + public void setShopAccessibility(double accessibility) + { + shopAccessibility = accessibility; + } + + public void setMaintAccessibility(double accessibility) + { + maintAccessibility = accessibility; + } + + public void setEatOutAccessibility(double accessibility) + { + eatOutAccessibility = accessibility; + } + + public void setVisitAccessibility(double accessibility) + { + visitAccessibility = accessibility; + } + + public void setDiscrAccessibility(double accessibility) + { + discrAccessibility = accessibility; + } + + public void setNmDcLogsum(double logsum) + { + nmDcLogsum = logsum; + } + + public void setWorkAccessibility(double accessibility) + { + workAccessibility = accessibility; + } + + public void setSchoolAccessibility(double accessibility) + { + schoolAccessibility = accessibility; + } + + /** + * called by methods invoked by UEC.solve() + * + * @return maximum number of hours mutually available between pairs of + * adults in household + */ + public int getMaxAdultOverlaps() + { + return hh.getMaxAdultOverlaps(); + } + + /** + * called by methods invoked by UEC.solve() + * + * @return maximum number of hours mutually available between pairs of + * children in household + */ + public int getMaxChildOverlaps() + { + return hh.getMaxChildOverlaps(); + } + + /** + * called by methods invoked by UEC.solve() + * + * @return maximum number of hours mutually available between pairs or + * adults/children where pairs consist of different types in + * household + */ + public int getMaxMixedOverlaps() + { + return hh.getMaxMixedOverlaps(); + } + + public int getMaxPairwiseOverlapAdult() + { + int maxOverlap = 0; + + // get array of person objects for the decision making household + Person[] dmuPersons = hh.getPersons(); + + for (int i = 1; i < dmuPersons.length; i++) + { + if (dmuPersons[i].getPersonIsAdult() == 1) + { + int overlap = getOverlap(person, dmuPersons[i]); + if (overlap > maxOverlap) maxOverlap = overlap; + } + } + + return maxOverlap; + } + + public int getMaxPairwiseOverlapChild() + { + int maxOverlap = 0; + + // get array of person objects for the decision making household + Person[] dmuPersons = hh.getPersons(); + + for (int i = 1; i < dmuPersons.length; i++) + { + if (dmuPersons[i].getPersonIsAdult() == 0) + { + int overlap = getOverlap(person, dmuPersons[i]); + if (overlap > maxOverlap) maxOverlap = overlap; + } + } + + return maxOverlap; + } + + // TODO: find out if this is suposed to be total pairwise available hours, + // or + // largest consecutive hours available for persons. + // TODO: right now, assuming total pairwise available hours + private int getOverlap(Person dmuPerson, Person otherPerson) + { + short[] dmuWindow = dmuPerson.getTimeWindows(); + short[] otherWindow = otherPerson.getTimeWindows(); + + int overlap = 0; + for (int i = 0; i < dmuWindow.length; i++) + { + if (dmuWindow[i] == 0 && otherWindow[i] == 0) overlap++; + } + + return overlap; + } + + public int getWindowBeforeFirstMandJointTour() + { + return person.getWindowBeforeFirstMandJointTour(); + } + + public int getWindowBetweenFirstLastMandJointTour() + { + return person.getWindowBetweenFirstLastMandJointTour(); + } + + public int getWindowAfterLastMandJointTour() + { + return person.getWindowAfterLastMandJointTour(); + } + + public int getNumHhFtWorkers() + { + return hh.getNumFtWorkers(); + } + + public int getNumHhPtWorkers() + { + return hh.getNumPtWorkers(); + } + + public int getNumHhUnivStudents() + { + return hh.getNumUnivStudents(); + } + + public int getNumHhNonWorkAdults() + { + return hh.getNumNonWorkAdults(); + } + + public int getNumHhRetired() + { + return hh.getNumRetired(); + } + + public int getNumHhDrivingStudents() + { + return hh.getNumDrivingStudents(); + } + + public int getNumHhNonDrivingStudents() + { + return hh.getNumNonDrivingStudents(); + } + + public int getNumHhPreschool() + { + return hh.getNumPreschool(); + } + + public int getTravelActiveAdults() + { + return hh.getTravelActiveAdults(); + } + + public int getTravelActiveChildren() + { + return hh.getTravelActiveChildren(); + } + + public int getNumMandatoryTours() + { + return person.getNumMandatoryTours(); + } + + public int getNumJointShoppingTours() + { + return person.getNumJointShoppingTours(); + } + + public int getNumJointOthMaintTours() + { + return person.getNumJointOthMaintTours(); + } + + public int getNumJointEatOutTours() + { + return person.getNumJointEatOutTours(); + } + + public int getNumJointSocialTours() + { + return person.getNumJointSocialTours(); + } + + public int getNumJointOthDiscrTours() + { + return person.getNumJointOthDiscrTours(); + } + + public int getJTours() + { + return hh.getJointTourArray().length; + } + + public int getPreDrivingAtHome() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getPersonIsStudentNonDriving() == 1 + && persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) + num++; + } + return num; + } + + public int getPreschoolAtHome() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + { + if (persons[i].getPersonIsPreschoolChild() == 1 + && persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) + num++; + } + return num; + } + + public double getDistanceToWorkLocation() + { + return person.getWorkLocationDistance(); + } + + public double getDistanceToSchoolLocation() + { + return person.getSchoolLocationDistance(); + } + + public double getEscortAccessibility() + { + return escortAccessibility; + } + + public double getShopAccessibility() + { + return shopAccessibility; + } + + public double getMaintAccessibility() + { + return maintAccessibility; + } + + public double getEatOutAccessibility() + { + return eatOutAccessibility; + } + + public double getVisitAccessibility() + { + return visitAccessibility; + } + + public double getDiscrAccessibility() + { + return discrAccessibility; + } + + public int getCdapIndex() + { + return person.getCdapIndex(); + } + + public double getNonMotorizedDcLogsum() + { + return nmDcLogsum; + } + + public int getNumPredrivingKidsGoOut() + { + return hh.getNumberOfPreDrivingWithNonHomeActivity(); + } + + public int getNumPreschoolKidsGoOut() + { + return hh.getNumberOfPreschoolWithNonHomeActivity(); + } + + public int getCollegeEducation() + { + return person.getHasBachelors(); + } + + public int getLowEducation() + { + return person.getPersonIsHighSchoolGraduate() == 1 ? 0 : 1; + } + + public int getDetachedHh() + { + return hh.getHhBldgsz() == 1 ? 1 : 0; + } + + public double getWorkAccessibility() + { + return workAccessibility; + } + + public double getSchoolAccessibility() + { + return schoolAccessibility; + } + + public int getTelecommuteFrequency() { + return person.getTelecommuteChoice(); + } + + + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java new file mode 100644 index 0000000..bd1e1db --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java @@ -0,0 +1,5075 @@ +package org.sandag.abm.ctramp; + +import org.sandag.abm.ctramp.CtrampDmuFactoryIf; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Stop; +import org.sandag.abm.ctramp.StopLocationDMU; +import org.sandag.abm.ctramp.Tour; +import org.sandag.abm.ctramp.TripModeChoiceDMU; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Random; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * This class will be used for determining the trip departure time for outbound + * stops, trip arrival time for inbound stops, location of stops, and trip mode + * for trips between stops on individual mandatory, individual non-mandatory and + * joint tours. + * + * @author Jim Hicks + * @version Oct 2010 + */ +public class IntermediateStopChoiceModels + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(IntermediateStopChoiceModels.class); + private transient Logger slcLogger = Logger.getLogger("slcLogger"); + private transient Logger slcSoaLogger = Logger.getLogger("slcSoaLogger"); + private transient Logger smcLogger = Logger.getLogger("tripMcLog"); + private transient Logger tripDepartLogger = Logger.getLogger("tripDepartLog"); + private transient Logger parkLocLogger = Logger.getLogger("parkLocLog"); + + public static final int WTW = 0; + public static final int WTD = 1; + public static final int DTW = 2; + public static final int[] ACC_EGR = {WTW,WTD,DTW}; + private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "slc.use.new.soa"; + + public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; + + private static final String PROPERTIES_UEC_SLC_CHOICE = "slc.uec.file"; + private static final String PROPERTIES_UEC_SLC_DATA_PAGE = "slc.uec.data.page"; + private static final String PROPERTIES_UEC_MAND_SLC_MODEL_PAGE = "slc.mandatory.uec.model.page"; + private static final String PROPERTIES_UEC_MAINT_SLC_MODEL_PAGE = "slc.maintenance.uec.model.page"; + private static final String PROPERTIES_UEC_DISCR_SLC_MODEL_PAGE = "slc.discretionary.uec.model.page"; + + public static final String PROPERTIES_UEC_SLC_SOA_CHOICE = "slc.soa.uec.file"; + public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY = "auto.slc.soa.distance.uec.file"; + public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_DATA_PAGE = "auto.slc.soa.distance.data.page"; + public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_MODEL_PAGE = "auto.slc.soa.distance.model.page"; + + private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE = "slc.soa.size.uec.file"; + private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE_DATA = "slc.soa.size.uec.data.page"; + private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE_MODEL = "slc.soa.size.uec.model.page"; + + private static final String PROPERTIES_UEC_PARKING_LOCATION_CHOICE = "plc.uec.file"; + private static final String PROPERTIES_UEC_PLC_DATA_PAGE = "plc.uec.data.page"; + private static final String PROPERTIES_UEC_PLC_MODEL_PAGE = "plc.uec.model.page"; + + private static final String PROPERTIES_UEC_PARKING_LOCATION_CHOICE_ALTERNATIVES = "plc.alts.corresp.file"; + + public static final int WORK_SHEET = 1; + public static final int UNIVERSITY_SHEET = 2; + public static final int SCHOOL_SHEET = 3; + public static final int MAINTENANCE_SHEET = 4; + public static final int DISCRETIONARY_SHEET = 5; + public static final int SUBTOUR_SHEET = 6; + public static final int[] MC_PURPOSE_SHEET_INDICES = { + -1, WORK_SHEET, UNIVERSITY_SHEET, SCHOOL_SHEET, MAINTENANCE_SHEET, MAINTENANCE_SHEET, + MAINTENANCE_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, + SUBTOUR_SHEET }; + + public static final int WORK_CATEGORY = 0; + public static final int UNIVERSITY_CATEGORY = 1; + public static final int SCHOOL_CATEGORY = 2; + public static final int MAINTENANCE_CATEGORY = 3; + public static final int DISCRETIONARY_CATEGORY = 4; + public static final int SUBTOUR_CATEGORY = 5; + public static final String[] PURPOSE_CATEGORY_LABELS = { + "work", "university", "school", "maintenance", "discretionary", "subtour" }; + public static final int[] PURPOSE_CATEGORIES = { + -1, WORK_CATEGORY, UNIVERSITY_CATEGORY, SCHOOL_CATEGORY, MAINTENANCE_CATEGORY, + MAINTENANCE_CATEGORY, MAINTENANCE_CATEGORY, DISCRETIONARY_CATEGORY, + DISCRETIONARY_CATEGORY, DISCRETIONARY_CATEGORY, SUBTOUR_CATEGORY }; + + private static final String PARK_MGRA_COLUMN = "mgra"; + private static final String PARK_AREA_COLUMN = "parkarea"; + private static final int MAX_PLC_SAMPLE_SIZE = 1200; + + private static final int WORK_STOP_PURPOSE_INDEX = 1; + private static final int UNIV_STOP_PURPOSE_INDEX = 2; + private static final int ESCORT_STOP_PURPOSE_INDEX = 4; + private static final int SHOP_STOP_PURPOSE_INDEX = 5; + private static final int MAINT_STOP_PURPOSE_INDEX = 6; + private static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; + private static final int VISIT_STOP_PURPOSE_INDEX = 8; + private static final int DISCR_STOP_PURPOSE_INDEX = 9; + + private static final int OTHER_STOP_LOC_SOA_SHEET_INDEX = 2; + private static final int WALK_STOP_LOC_SOA_SHEET_INDEX = 3; + private static final int BIKE_STOP_LOC_SOA_SHEET_INDEX = 4; + private static final int MAX_STOP_LOC_SOA_SHEET_INDEX = 4; + + private static final int WORK_STOP_PURPOSE_SOA_SIZE_INDEX = 0; + private static final int UNIV_STOP_PURPOSE_SOA_SIZE_INDEX = 1; + private static final int ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX = 2; + private static final int ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX = 3; + private static final int ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 4; + private static final int ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 5; + private static final int ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 6; + private static final int ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 7; + private static final int ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 8; + private static final int ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 9; + private static final int SHOP_STOP_PURPOSE_SOA_SIZE_INDEX = 10; + private static final int MAINT_STOP_PURPOSE_SOA_SIZE_INDEX = 11; + private static final int EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX = 12; + private static final int VISIT_STOP_PURPOSE_SOA_SIZE_INDEX = 13; + private static final int DISCR_STOP_PURPOSE_SOA_SIZE_INDEX = 14; + + public static final int[] SLC_SIZE_SEGMENT_INDICES = { + WORK_STOP_PURPOSE_SOA_SIZE_INDEX, UNIV_STOP_PURPOSE_SOA_SIZE_INDEX, + ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX, + ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX, + ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, + ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, + SHOP_STOP_PURPOSE_SOA_SIZE_INDEX, MAINT_STOP_PURPOSE_SOA_SIZE_INDEX, + EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX, VISIT_STOP_PURPOSE_SOA_SIZE_INDEX, + DISCR_STOP_PURPOSE_SOA_SIZE_INDEX }; + + public static final String[] SLC_SIZE_SEGMENT_NAMES = { + "work", "univ", "escort_0", "escort_ps", "escort_gs", "escort_hs", "escort_ps_gs", + "escort_ps_hs", "escort_gs_hs", "escort_ps_gs_hs", "shop", "maint", "eatout", "visit", + "discr" }; + + private static final int MAND_SLC_MODEL_INDEX = 0; + private static final int MAINT_SLC_MODEL_INDEX = 1; + private static final int DISCR_SLC_MODEL_INDEX = 2; + + private static final String PROPERTIES_TRIP_UTILITY_IVT_COEFFS = "trip.utility.ivt.coeffs"; + private static final String PROPERTIES_TRIP_UTILITY_INCOME_COEFFS = "trip.utility.income.coeffs"; + private static final String PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS = "trip.utility.income.exponents"; + + + private boolean[] sampleAvailability; + private int[] inSample; + private boolean[] soaAvailability; + private int[] soaSample; + private boolean[] soaAvailabilityBackup; + private int[] soaSampleBackup; + private int[] finalSample; + private double[] sampleCorrectionFactors; + private double[] tripModeChoiceLogsums; + private boolean[] sampleMgraInBoardingTapShed; + private boolean[] sampleMgraInAlightingTapShed; + private boolean earlierTripWasLocatedInAlightingTapShed; + + private double[] tourOrigToAllMgraDistances; + private double[] tourDestToAllMgraDistances; + private double[] ikDistance; + private double[] kjDistance; + private double[] okDistance; + private double[] kdDistance; + + private AutoAndNonMotorizedSkimsCalculator anm; + private McLogsumsCalculator logsumHelper; + private ModelStructure modelStructure; + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private int sampleSize; + private HashMap altFreqMap; + private StopLocationDMU stopLocDmuObj; + private TripModeChoiceDMU mcDmuObject; + private ParkingChoiceDMU parkingChoiceDmuObj; + + private double[][] slcSizeTerms; + private int[][] slcSizeSample; + private boolean[][] slcSizeAvailable; + + private double[] distanceFromStopOrigToAllMgras; + private double[] distanceToFinalDestFromAllMgras; + + private final BikeLogsum bls; + private BikeLogsumSegment segment; + + private double[] bikeLogsumFromStopOrigToAllMgras; + private double[] bikeLogsumToFinalDestFromAllMgras; + + private double[][] mcCumProbsSegmentIk; + private double[][] mcCumProbsSegmentKj; + + private double[] mcLogsumsSegmentIk; + private double[] mcLogsumsSegmentKj; + + private double[][] mcVOTsSegmentIk; //by sample and occupancy (0=non-SR,1=S2,2=S3) + private double[][] mcVOTsSegmentKj; + + private float[] parkingCostSegmentIk; //by sample + private float[] parkingCostSegmentKj; + + private double[][][][] segmentIkBestTapPairs; + private double[][][][] segmentKjBestTapPairs; + + + private ChoiceModelApplication[] mcModelArray; + private ChoiceModelApplication[] slcSoaModel; + private ChoiceModelApplication[] slcModelArray; + private ChoiceModelApplication plcModel; + + private int[] altMgraIndices; + private double[] altOsDistances; + private double[] altSdDistances; + private boolean[] altParkAvail; + private int[] altParkSample; + + private int numAltsInSample; + private int maxAltsInSample; + + private TableDataSet plcAltsTable; + private HashMap mgraAltLocationIndex; + private HashMap mgraAltParkArea; + private int[] parkMgras; + private int[] parkAreas; + + private int[] mgraParkArea; + private int[] numfreehrs; + private int[] hstallsoth; + private int[] hstallssam; + private float[] hparkcost; + private int[] dstallsoth; + private int[] dstallssam; + private float[] dparkcost; + private int[] mstallsoth; + private int[] mstallssam; + private float[] mparkcost; + + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + + private double[] altParkingCostsM; + private double[] altParkingCostsD; + private double[] altParkingCostsH; + private int[] altMstallsoth; + private int[] altMstallssam; + private float[] altMparkcost; + private int[] altDstallsoth; + private int[] altDstallssam; + private float[] altDparkcost; + private int[] altHstallsoth; + private int[] altHstallssam; + private float[] altHparkcost; + private int[] altNumfreehrs; + + private HashMap sizeSegmentNameIndexMap; + + private StopDepartArrivePeriodModel stopTodModel; + + private int availAltsToLog = 55; + // this + private TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator; + // logging + // private int availAltsToLog = 5; + + // set this constant for checking the number of times depart/arrive period + // selection is made so that no infinite loop occurs. + private static final int MAX_INVALID_FIRST_ARRIVAL_COUNT = 100; + + public static final int NUM_CPU_TIME_VALUES = 9; + private long soaAutoTime; + private long soaOtherTime; + private long slsTime; + private long sldTime; + private long slcTime; + private long todTime; + private long smcTime; + private long[] hhTimes = new long[NUM_CPU_TIME_VALUES]; + + private DestChoiceTwoStageModel dcTwoStageModelObject; + + private boolean useNewSoaMethod; + + private String loggerSeparator = ""; + + // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose + double[] ivtCoeffs; + double[] incomeCoeffs; + double[] incomeExponents; + + /** + * Constructor that will be used to set up the ChoiceModelApplications for + * each type of tour + * + * @param projectDirectory + * - name of root level project directory + * @param resourceBundle + * - properties file with paths identified + * @param dmuObject + * - decision making unit for stop frequency + * @param modelStructure + * - holds the model structure info + */ + public IntermediateStopChoiceModels(HashMap propertyMap, + ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory, + McLogsumsCalculator myLogsumHelper) + { + + tazs = TazDataManager.getInstance(propertyMap); + mgraManager = MgraDataManager.getInstance(propertyMap); + + mgraParkArea = mgraManager.getMgraParkAreas(); + numfreehrs = mgraManager.getNumFreeHours(); + lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); + lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); + lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); + mstallsoth = mgraManager.getMStallsOth(); + mstallssam = mgraManager.getMStallsSam(); + mparkcost = mgraManager.getMParkCost(); + dstallsoth = mgraManager.getDStallsOth(); + dstallssam = mgraManager.getDStallsSam(); + dparkcost = mgraManager.getDParkCost(); + hstallsoth = mgraManager.getHStallsOth(); + hstallssam = mgraManager.getHStallsSam(); + hparkcost = mgraManager.getHParkCost(); + + modelStructure = myModelStructure; + logsumHelper = myLogsumHelper; + + setupStopLocationChoiceModels(propertyMap, dmuFactory); + setupParkingLocationModel(propertyMap, dmuFactory); + + bls = BikeLogsum.getBikeLogsum(propertyMap); + + //get the coefficients for ivt and the coefficients to calculate the cost coefficient + ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_IVT_COEFFS); + incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_COEFFS); + incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS); + + tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); + tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); + + + } + + private void setupStopLocationChoiceModels(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up stop location choice models.")); + + useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, + USE_NEW_SOA_METHOD_PROPERTY_KEY); + + stopLocDmuObj = dmuFactory.getStopLocationDMU(); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String slcSoaUecFile = propertyMap.get(PROPERTIES_UEC_SLC_SOA_CHOICE); + slcSoaUecFile = uecPath + slcSoaUecFile; + + String slcUecFile = propertyMap.get(PROPERTIES_UEC_SLC_CHOICE); + slcUecFile = uecPath + slcUecFile; + + slcSoaModel = new ChoiceModelApplication[MAX_STOP_LOC_SOA_SHEET_INDEX + 1]; + slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, + OTHER_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); + slcSoaModel[WALK_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, + WALK_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); + slcSoaModel[BIKE_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, + BIKE_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); + + int numSlcSoaAlternatives = slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX] + .getNumberOfAlternatives(); + stopLocDmuObj = dmuFactory.getStopLocationDMU(); + + sizeSegmentNameIndexMap = new HashMap(); + for (int k = 0; k < SLC_SIZE_SEGMENT_INDICES.length; k++) + { + sizeSegmentNameIndexMap.put(SLC_SIZE_SEGMENT_NAMES[k], k); + sizeSegmentNameIndexMap.put(SLC_SIZE_SEGMENT_NAMES[k], SLC_SIZE_SEGMENT_INDICES[k]); + } + + // set the second argument to a positive, non-zero mgra value to get + // logging for the size term calculation for the specified mgra. + slcSizeTerms = calculateLnSlcSizeTerms(propertyMap, -1); + + String mcUecFile = propertyMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + mcDmuObject = dmuFactory.getTripModeChoiceDMU(); + + // logsumHelper.setupSkimCalculators(propertyMap); + anm = logsumHelper.getAnmSkimCalculator(); + mcDmuObject.setParkingCostInfo(mgraParkArea, lsWgtAvgCostM, lsWgtAvgCostD, lsWgtAvgCostH); + + mcModelArray = new ChoiceModelApplication[5 + 1]; + mcModelArray[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, + propertyMap, (VariableTable) mcDmuObject); + mcModelArray[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, + 0, propertyMap, (VariableTable) mcDmuObject); + mcModelArray[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, + propertyMap, (VariableTable) mcDmuObject); + mcModelArray[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, + MAINTENANCE_SHEET, 0, propertyMap, (VariableTable) mcDmuObject); + mcModelArray[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, + DISCRETIONARY_SHEET, 0, propertyMap, (VariableTable) mcDmuObject); + mcModelArray[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, + propertyMap, (VariableTable) mcDmuObject); + + // set up the stop location choice model object + int slcDataPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_SLC_DATA_PAGE)); + int slcMandModelPage = Integer + .parseInt(propertyMap.get(PROPERTIES_UEC_MAND_SLC_MODEL_PAGE)); + int slcMaintModelPage = Integer.parseInt(propertyMap + .get(PROPERTIES_UEC_MAINT_SLC_MODEL_PAGE)); + int slcDiscrModelPage = Integer.parseInt(propertyMap + .get(PROPERTIES_UEC_DISCR_SLC_MODEL_PAGE)); + slcModelArray = new ChoiceModelApplication[3]; + slcModelArray[MAND_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, + slcMandModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); + slcModelArray[MAINT_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, + slcMaintModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); + slcModelArray[DISCR_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, + slcDiscrModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); + + sampleSize = slcModelArray[MAND_SLC_MODEL_INDEX].getNumberOfAlternatives(); + altFreqMap = new HashMap(sampleSize); + finalSample = new int[sampleSize + 1]; + sampleCorrectionFactors = new double[sampleSize + 1]; + + // decalre the arrays for storing stop location choice ik and kj segment + // mode choice probability arrays + mcCumProbsSegmentIk = new double[sampleSize + 1][]; + mcCumProbsSegmentKj = new double[sampleSize + 1][]; + + //declare the arrays for storing the VOTs calculated by trip mode choice for ik and kj segment + mcVOTsSegmentIk = new double[sampleSize+1][3]; + mcVOTsSegmentKj = new double[sampleSize+1][3]; + + //declare arrays for storing the trip mode choice parking cost for ik and kj segment + parkingCostSegmentIk = new float[sampleSize+1]; + parkingCostSegmentKj = new float[sampleSize+1]; + + //declare the arrays for storing the stop location choice ik and kj segment + //mode choice logsum arrays + mcLogsumsSegmentIk = new double[sampleSize + 1]; + mcLogsumsSegmentKj = new double[sampleSize + 1]; + + // declare the arrays for storing stop location choice ik and kj segment best tap pair arrays + segmentIkBestTapPairs = new double[sampleSize+1][ACC_EGR.length][][]; + segmentKjBestTapPairs = new double[sampleSize+1][ACC_EGR.length][][]; + + // declare the arrays that holds ik and kj segment logsum values for + // each location choice sample alternative + tripModeChoiceLogsums = new double[sampleSize + 1]; + + // declare the arrays that holds ik and kj distance values for each + // location choice sample alternative + // these are set as stop location dmu attributes for stop orig to stop + // alt, and stop alt to half-tour final dest. + ikDistance = new double[sampleSize + 1]; + kjDistance = new double[sampleSize + 1]; + + // declare the arrays that holds ik and kj distance values for each + // location choice sample alternative + // these are set as stop location dmu attributes for tour orig to stop + // alt, and stop alt to tour dest. + okDistance = new double[sampleSize + 1]; + kdDistance = new double[sampleSize + 1]; + + // this array has elements with values of 0 if utility is not to be + // computed for alternative, or 1 if utility is to be computed. + inSample = new int[sampleSize + 1]; + + // this array has elements that are boolean that set availability of + // alternative - unavailable altrnatives do not get utility computed. + sampleAvailability = new boolean[sampleSize + 1]; + + soaSample = new int[numSlcSoaAlternatives + 1]; + soaAvailability = new boolean[numSlcSoaAlternatives + 1]; + soaSampleBackup = new int[numSlcSoaAlternatives + 1]; + soaAvailabilityBackup = new boolean[numSlcSoaAlternatives + 1]; + + sampleMgraInBoardingTapShed = new boolean[mgraManager.getMaxMgra() + 1]; + sampleMgraInAlightingTapShed = new boolean[mgraManager.getMaxMgra() + 1]; + + distanceFromStopOrigToAllMgras = new double[mgraManager.getMaxMgra() + 1]; + distanceToFinalDestFromAllMgras = new double[mgraManager.getMaxMgra() + 1]; + + bikeLogsumFromStopOrigToAllMgras = new double[mgraManager.getMaxMgra() + 1]; + bikeLogsumToFinalDestFromAllMgras = new double[mgraManager.getMaxMgra() + 1]; + + tourOrigToAllMgraDistances = new double[mgraManager.getMaxMgra() + 1]; + tourDestToAllMgraDistances = new double[mgraManager.getMaxMgra() + 1]; + + // create the array of 1s for MGRAs that have a non-empty set of TAPs + // within walk egress distance of them + // for the setting walk transit available for teh stop location + // alternatives. + // createWalkTransitAvailableArray(); + + setupTripDepartTimeModel(propertyMap, dmuFactory); + + loggerSeparator += "-"; + + } + + public void setupSlcDistanceBaseSoaModel(HashMap propertyMap, + double[][] soaExpUtils, double[][][] soaSizeProbs, double[][] soaTazSize) + { + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String slcSoaDistanceUecFile = propertyMap.get(PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY); + slcSoaDistanceUecFile = uecPath + slcSoaDistanceUecFile; + + dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, sampleSize); + dcTwoStageModelObject.setSlcSoaProbsAndUtils(soaExpUtils, soaSizeProbs, soaTazSize); + + } + + private void setupTripDepartTimeModel(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + stopTodModel = new StopDepartArrivePeriodModel(propertyMap, modelStructure); + } + + private void setupParkingLocationModel(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + logger.info("setting up parking location choice models."); + + // locate the UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String plcUecFile = propertyMap.get(PROPERTIES_UEC_PARKING_LOCATION_CHOICE); + plcUecFile = uecPath + plcUecFile; + + int plcDataPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_PLC_DATA_PAGE)); + int plcModelPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_PLC_MODEL_PAGE)); + + altMgraIndices = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altOsDistances = new double[MAX_PLC_SAMPLE_SIZE + 1]; + altSdDistances = new double[MAX_PLC_SAMPLE_SIZE + 1]; + altParkingCostsM = new double[MAX_PLC_SAMPLE_SIZE + 1]; + altParkingCostsD = new double[MAX_PLC_SAMPLE_SIZE + 1]; + altParkingCostsH = new double[MAX_PLC_SAMPLE_SIZE + 1]; + altMstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altMstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altMparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; + altDstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altDstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altDparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; + altHstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altHstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; + altHparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; + altNumfreehrs = new int[MAX_PLC_SAMPLE_SIZE + 1]; + + altParkAvail = new boolean[MAX_PLC_SAMPLE_SIZE + 1]; + altParkSample = new int[MAX_PLC_SAMPLE_SIZE + 1]; + + parkingChoiceDmuObj = dmuFactory.getParkingChoiceDMU(); + + plcModel = new ChoiceModelApplication(plcUecFile, plcModelPage, plcDataPage, propertyMap, + (VariableTable) parkingChoiceDmuObj); + + // read the parking choice alternatives data file to get alternatives + // names + String plcAltsFile = propertyMap.get(PROPERTIES_UEC_PARKING_LOCATION_CHOICE_ALTERNATIVES); + plcAltsFile = uecPath + plcAltsFile; + + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + plcAltsTable = reader.readFile(new File(plcAltsFile)); + } catch (IOException e) + { + logger.error("problem reading table of cbd zones for parking location choice model.", e); + System.exit(1); + } + + parkMgras = plcAltsTable.getColumnAsInt(PARK_MGRA_COLUMN); + parkAreas = plcAltsTable.getColumnAsInt(PARK_AREA_COLUMN); + + parkingChoiceDmuObj.setParkAreaMgraArray(parkMgras); + parkingChoiceDmuObj.setSampleIndicesArray(altMgraIndices); + parkingChoiceDmuObj.setDistancesOrigAlt(altOsDistances); + parkingChoiceDmuObj.setDistancesAltDest(altSdDistances); + parkingChoiceDmuObj.setParkingCostsM(altParkingCostsM); + parkingChoiceDmuObj.setMstallsoth(altMstallsoth); + parkingChoiceDmuObj.setMstallssam(altMstallssam); + parkingChoiceDmuObj.setMparkCost(altMparkcost); + parkingChoiceDmuObj.setDstallsoth(altDstallsoth); + parkingChoiceDmuObj.setDstallssam(altDstallssam); + parkingChoiceDmuObj.setDparkCost(altDparkcost); + parkingChoiceDmuObj.setHstallsoth(altHstallsoth); + parkingChoiceDmuObj.setHstallssam(altHstallssam); + parkingChoiceDmuObj.setHparkCost(altHparkcost); + parkingChoiceDmuObj.setNumfreehrs(altNumfreehrs); + + mgraAltLocationIndex = new HashMap(); + mgraAltParkArea = new HashMap(); + + for (int i = 0; i < parkMgras.length; i++) + { + mgraAltLocationIndex.put(parkMgras[i], i); + mgraAltParkArea.put(parkMgras[i], parkAreas[i]); + } + + } + + private double[][] calculateLnSlcSizeTerms(HashMap rbMap, int logMgra) + { + + logger.info("calculating Stop Location SOA Size Terms"); + + String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String slcSizeUecFile = rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE); + slcSizeUecFile = uecPath + slcSizeUecFile; + int slcSizeUecData = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE_DATA)); + int slcSizeUecModel = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE_MODEL)); + + IndexValues iv = new IndexValues(); + UtilityExpressionCalculator slcSizeUec = new UtilityExpressionCalculator(new File( + slcSizeUecFile), slcSizeUecModel, slcSizeUecData, rbMap, null); + + ArrayList mgras = mgraManager.getMgras(); + int maxMgra = mgraManager.getMaxMgra(); + int numSizeSegments = slcSizeUec.getNumberOfAlternatives(); + + // create the array for storing logged size term values - to be returned + // by this method + double[][] lnSlcSoaSize = new double[numSizeSegments][maxMgra + 1]; + slcSizeSample = new int[numSizeSegments][maxMgra + 1]; + slcSizeAvailable = new boolean[numSizeSegments][maxMgra + 1]; + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + iv.setZoneIndex(mgra); + double[] size = slcSizeUec.solve(iv, null, null); + + // if a logMgra values > 0 was specified, log the size term utility + // calculation for that mgra + if (mgra == logMgra) + slcSizeUec.logAnswersArray(slcSoaLogger, "Stop Location SOA Size Terms, MGRA = " + + mgra); + + // store the logged size terms + for (int i = 0; i < numSizeSegments; i++) + { + lnSlcSoaSize[i][mgra] = Math.log(size[i] + 1); + if (size[i] > 0) + { + slcSizeSample[i][mgra] = 1; + slcSizeAvailable[i][mgra] = true; + } + } + + } + + return lnSlcSoaSize; + + } + + private double[] getLnSlcSizeTermsForStopPurpose(int stopPurpose, Household hh) + { + + double[] lnSlcSizeTerms = null; + + switch (stopPurpose) + { + + case WORK_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case UNIV_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case ESCORT_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = getLnSlcSizeTermsForEscortStopPurpose(hh); + break; + case SHOP_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case MAINT_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case EAT_OUT_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case VISIT_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + case DISCR_STOP_PURPOSE_INDEX: + lnSlcSizeTerms = slcSizeTerms[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; + break; + } + + // save backup arrays with oroginal availability and sample values. + // the procedure to generate availabilty for transit tours overwrites + // the arrays used by the UECs, + // so they need to be restored after that happens. + for (int i = 0; i < soaSample.length; i++) + { + soaSampleBackup[i] = soaSample[i]; + soaAvailabilityBackup[i] = soaAvailability[i]; + } + + return lnSlcSizeTerms; + + } + + private double[] getLnSlcSizeTermsForEscortStopPurpose(Household hh) + { + + double[] lnSlcSizeTermsForEscort = null; + + // set booleans for presence of preschool, grade school, and high school + // students in the hh + boolean psInHh = hh.getNumPreschool() > 0; + boolean gsInHh = hh.getNumGradeSchoolStudents() > 0; + boolean hsInHh = hh.getNumHighSchoolStudents() > 0; + + if (!psInHh && !gsInHh && !hsInHh) + { + // if hh has no preschool, grade school or high school children, set + // the array to that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (psInHh && !gsInHh && !hsInHh) + { + // if hh has a preschool child and no gs or hs, set the array to + // that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (!psInHh && gsInHh && !hsInHh) + { + // if hh has a grade school child and no ps or hs, set the array to + // that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (!psInHh && !gsInHh && hsInHh) + { + // if hh has a high school child and no ps or gs, set the array to + // that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (psInHh && gsInHh && !hsInHh) + { + // if hh has a preschool and a grade school child and no hs, set the + // array to that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (psInHh && !gsInHh && hsInHh) + { + // if hh has a preschool and a high school child and no gs, set the + // array to that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (!psInHh && gsInHh && hsInHh) + { + // if hh has a grade school and a high school child and no ps, set + // the array to that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } else if (psInHh && gsInHh && hsInHh) + { + // if hh has a preschool a grade school and a high school child, set + // the array to that specific size term field + lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaSample = slcSizeSample[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + soaAvailability = slcSizeAvailable[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; + } + + return lnSlcSizeTermsForEscort; + } + + public void applyModel(Household household, boolean withTiming) + { + + if (withTiming) zeroOutCpuTimes(); + + if (household.getDebugChoiceModels()) + { + slcLogger.info("applying SLC model for hhid=" + household.getHhId()); + } + + // get this household's person array + Person[] personArray = household.getPersons(); + + // loop through the person array (1-based) + for (int j = 1; j < personArray.length; ++j) + { + + ArrayList tours = new ArrayList(); + + Person person = personArray[j]; + + // apply stop location and mode choice for all individual tours. + tours.addAll(person.getListOfWorkTours()); + tours.addAll(person.getListOfSchoolTours()); + tours.addAll(person.getListOfIndividualNonMandatoryTours()); + tours.addAll(person.getListOfAtWorkSubtours()); + + for (Tour tour : tours) + { + + if (withTiming) applyForOutboundStopsWithTiming(tour, person, household); + else applyForOutboundStops(tour, person, household); + + if (withTiming) applyForInboundStopsWithTiming(tour, person, household); + else applyForInboundStops(tour, person, household); + + } // tour loop + + } // j (person loop) + + // apply stop location and mode choice for all joint tours. + Tour[] jointTours = household.getJointTourArray(); + if (jointTours != null) + { + + for (Tour tour : jointTours) + { + + if (withTiming) applyForOutboundStopsWithTiming(tour, null, household); + else applyForOutboundStops(tour, null, household); + + if (withTiming) applyForInboundStopsWithTiming(tour, null, household); + else applyForInboundStops(tour, null, household); + + } // tour loop + + } + + household.setStlRandomCount(household.getHhRandomCount()); + + } + + private void applyForOutboundStops(Tour tour, Person person, Household household) + { + + //don't apply if the outbound direction is escort + if((tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE)) + return; + + Stop[] stops = tour.getOutboundStops(); + + // select trip depart periods for outbound stops + if (stops != null) + { + setOutboundTripDepartTimes(stops); + } + + int origMgra = tour.getTourOrigMgra(); + int destMgra = tour.getTourDestMgra(); + + applySlcModel(household, person, tour, stops, origMgra, destMgra, false); + + } + + private void applyForOutboundStopsWithTiming(Tour tour, Person person, Household household) + { + + //don't apply if the outbound direction is escort + if((tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE)) + return; + + long check = System.nanoTime(); + + Stop[] stops = tour.getOutboundStops(); + + // select trip depart periods for outbound stops + if (stops != null) + { + setOutboundTripDepartTimes(stops); + } + + int origMgra = tour.getTourOrigMgra(); + int destMgra = tour.getTourDestMgra(); + + todTime += (System.nanoTime() - check); + + applySlcModelWithTiming(household, person, tour, stops, origMgra, destMgra, false); + + } + + private void applyForInboundStops(Tour tour, Person person, Household household) + { + + //don't apply if the inbound direction is escort + if((tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE)) + return; + + Stop[] stops = tour.getInboundStops(); + + // select trip arrive periods for inbound stops + if (stops != null) + { + int lastOutboundTripDeparts = -1; + + // get the outbound stops array - note, if there were no outbound + // stops for half-tour, one stop object would have been generated + // to hold information about the half-tour trip, so this array + // should never be null. + Stop[] obStops = tour.getOutboundStops(); + if (obStops == null) + { + logger.error("error getting last outbound stop object for setting lastOutboundTripDeparts attribute for inbound stop arrival time choice."); + logger.error("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); + throw new RuntimeException(); + } else + { + Stop lastStop = obStops[obStops.length - 1]; + lastOutboundTripDeparts = lastStop.getStopPeriod(); + } + + setInboundTripDepartTimes(stops, lastOutboundTripDeparts); + } + + int origMgra = tour.getTourDestMgra(); + int destMgra = tour.getTourOrigMgra(); + + applySlcModel(household, person, tour, stops, origMgra, destMgra, true); + + } + + private void applyForInboundStopsWithTiming(Tour tour, Person person, Household household) + { + + //don't apply if the inbound direction is escort + if((tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE)) + return; + + long check = System.nanoTime(); + + Stop[] stops = tour.getInboundStops(); + + // select trip arrive periods for inbound stops + if (stops != null) + { + int lastOutboundTripDeparts = -1; + + // get the outbound stops array - note, if there were no outbound + // stops for half-tour, one stop object would have been generated + // to hold information about the half-tour trip, so this array + // should never be null. + Stop[] obStops = tour.getOutboundStops(); + if (obStops == null) + { + logger.error("error getting last outbound stop object for setting lastOutboundTripDeparts attribute for inbound stop arrival time choice."); + logger.error("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); + throw new RuntimeException(); + } else + { + Stop lastStop = obStops[obStops.length - 1]; + lastOutboundTripDeparts = lastStop.getStopPeriod(); + } + + setInboundTripDepartTimes(stops, lastOutboundTripDeparts); + } + + int origMgra = tour.getTourDestMgra(); + int destMgra = tour.getTourOrigMgra(); + + todTime += (System.nanoTime() - check); + + applySlcModelWithTiming(household, person, tour, stops, origMgra, destMgra, true); + + } + + private void applySlcModel(Household household, Person person, Tour tour, Stop[] stops, + int origMgra, int destMgra, boolean directionIsInbound) + { + + int lastDest = -1; + int newOrig = -1; + + // get the array of distances from the tour origin mgra to all MGRAs. + // use these distances for tour orig to stop alt distances + anm.getDistancesFromMgra(origMgra, tourOrigToAllMgraDistances, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + anm.getDistancesFromMgra(destMgra, tourDestToAllMgraDistances, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + // anm.getDistancesToMgra( destMgra, tourDestToAllMgraDistances, + // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) ); + + // if there are stops on this half-tour, determine their destinations, + // depart hours, trip modes, and parking tazs. + if (stops != null) + { + + int oldSelectedIndex = -1; + for (int i = 0; i < stops.length; i++) + { + + Stop stop = stops[i]; + + // if i is 0, the stop origin is set to origMgra; otherwise stop + // orig is the chosen dest from the previous stop. + if (i == 0){ + newOrig = origMgra; + } + else{ + newOrig = lastDest; + } + stop.setOrig(newOrig); + + stopLocDmuObj.setStopObject(stop); + stopLocDmuObj.setDmuIndexValues(household.getHhId(), household.getHhMgra(), + newOrig, destMgra); + + int choice = -1; + int selectedIndex = -1; + int modeAlt = -1; + float modeLogsum = 0; + double vot = -1.0; + // if not the last stop object, make a destination choice and a + // mode choice from IK MC probabilities; + // otherwise stop dest is set to destMgra, and make a mode + // choice from KJ MC probabilities. + if (i < stops.length - 1) + { + + //new code - depart period to and from stop + int departPeriodToStop = stop.getStopPeriod(); + int departPeriodFromStop = -1; + if(!directionIsInbound) + departPeriodFromStop = tour.getOutboundStops()[i+1].getStopPeriod(); + else + departPeriodFromStop = tour.getInboundStops()[i+1].getStopPeriod(); + + try + { + + selectedIndex = selectDestination(stop, departPeriodToStop, departPeriodFromStop); + choice = finalSample[selectedIndex]; + stop.setDest(choice); + lastDest = choice; + + if (household.getDebugChoiceModels()) + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + } + + modeAlt = selectModeFromProbabilities(stop, + mcCumProbsSegmentIk[selectedIndex]); + modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; + + if (modeAlt < 0) + { + logger.info("error getting trip mode choice for IK proportions, i=" + i); + logger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + logger.info("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + throw new RuntimeException(); + } + + //value of time; lookup vot, votS2, or votS3 depending on chosen mode + if(modelStructure.getTripModeIsS2(modeAlt)){ + vot = mcVOTsSegmentIk[selectedIndex][1]; + }else if (modelStructure.getTripModeIsS3(modeAlt)){ + vot = mcVOTsSegmentIk[selectedIndex][2]; + }else{ + vot = mcVOTsSegmentIk[selectedIndex][0]; + } + + int park = -1; + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + if (park > 0) lastDest = park; + float parkingCost = parkingCostSegmentIk[selectedIndex]; + stop.setParkingCost(parkingCost); + + } + + } catch (Exception e) + { + logger.error(String + .format("Exception caught processing %s stop location choice model for %s type tour %s stop: HHID=%d, personNum=%s, stop=%d.", + (stopLocDmuObj.getInboundStop() == 1 ? "inbound" + : "outbound"), + tour.getTourCategory(), + tour.getTourPurpose(), + household.getHhId(), + (person == null ? "N/A" : Integer.toString(person + .getPersonNum())), (i + 1))); + throw new RuntimeException(e); + } + + stop.setMode(modeAlt); + stop.setModeLogsum(modeLogsum); + stop.setValueOfTime(vot); + + // if the trip is a transit mode, set the boarding and + // alighting tap pairs in the stop object based on the ik + // segment pairs + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + + int accEgr = -1; + if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { + accEgr = WTW; + } else { + if (stop.isInboundStop()) { + accEgr = WTD; + } else { + accEgr = DTW; + } + } + + if ( segmentIkBestTapPairs[selectedIndex][accEgr] == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + //pick transit path from N-paths + double rn = household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, segmentIkBestTapPairs[selectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][0] ); + stop.setAlightTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][1] ); + stop.setSet( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][2] ); + + } + + } + + oldSelectedIndex = selectedIndex; + + } else + { + // last stop on half-tour, so dest is the half-tour final + // dest, and oldSelectedIndex was + // the selectedIndex determined for the previous stop + // location choice. + stop.setDest(destMgra); + + if (household.getDebugChoiceModels()) + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=End of " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " half-tour, stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + + stop.getOrig() + ", stopDest=" + stop.getDest()); + } + + modeAlt = selectModeFromProbabilities(stop, + mcCumProbsSegmentKj[oldSelectedIndex]); + modeLogsum = (float) mcLogsumsSegmentKj[oldSelectedIndex]; + + if (modeAlt < 0) + { + logger.error("error getting trip mode choice for KJ proportions, i=" + i); + logger.error("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + logger.error("StopID=End of " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " half-tour, stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + + stop.getOrig() + ", stopDest=" + stop.getDest()); + throw new RuntimeException(); + } + + //value of time; lookup vot, votS2, or votS3 depending on chosen mode + if(modelStructure.getTripModeIsS2(modeAlt)){ + vot = mcVOTsSegmentKj[oldSelectedIndex][1]; + }else if (modelStructure.getTripModeIsS3(modeAlt)){ + vot = mcVOTsSegmentKj[oldSelectedIndex][2]; + }else{ + vot = mcVOTsSegmentKj[oldSelectedIndex][0]; + } + + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) { + float parkingCost = parkingCostSegmentKj[oldSelectedIndex]; + stop.setParkingCost(parkingCost); + } + // last stop on tour, so if inbound, only need park location + // choice if tour is work-based subtour; + // otherwise dest is home. + int park = -1; + if (directionIsInbound) + { + if (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + } else + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + + stop.setMode(modeAlt); + stop.setModeLogsum(modeLogsum); + stop.setValueOfTime(vot); + + // if the last trip is a transit mode, set the boarding and + // alighting tap pairs in the stop object based on the kj + // segment pairs + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + + int accEgr = -1; + if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { + accEgr = WTW; + } else { + if (stop.isInboundStop()) { + accEgr = WTD; + } else { + accEgr = DTW; + } + } + + if ( segmentKjBestTapPairs[oldSelectedIndex][accEgr] == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + + //pick transit path from N-paths + float rn = (float)household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, segmentKjBestTapPairs[oldSelectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][0] ); + stop.setAlightTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][1] ); + stop.setSet( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][2] ); + } + + } + + } + + } + + } else + { + // create a stop object to hold attributes for orig, dest, mode, + // departtime, etc. + // for the half-tour with no stops. + + // create a Stop object for use in applying trip mode choice for + // this half tour without stops + String origStopPurpose = ""; + String destStopPurpose = ""; + if (!directionIsInbound) + { + origStopPurpose = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + destStopPurpose = tour.getTourPrimaryPurpose(); + } else + { + origStopPurpose = tour.getTourPrimaryPurpose(); + destStopPurpose = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + } + + Stop stop = null; + try + { + stop = tour.createStop(origStopPurpose, destStopPurpose, + directionIsInbound, + tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)); + } catch (Exception e) + { + logger.info("exception creating stop."); + throw new RuntimeException(e); + } + + // set stop origin and destination mgra, the stop period based on + // direction, then calculate the half-tour trip mode choice + stop.setOrig(origMgra); + stop.setDest(destMgra); + + if (directionIsInbound) stop.setStopPeriod(tour.getTourArrivePeriod()); + else stop.setStopPeriod(tour.getTourDepartPeriod()); + + int modeAlt = getHalfTourModeChoice(stop); + if (modeAlt < 0) + { + logger.info("error getting trip mode choice for half-tour with no stops."); + logger.info("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId()); + logger.info("StopID=" + (stop.getStopId() + 1) + " of no stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + throw new RuntimeException(); + } + + stop.setMode(modeAlt); + + double[][] bestTaps = null; + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) ) { + if ( directionIsInbound ) + bestTaps = tour.getBestWtwTapPairsIn(); + else + bestTaps = tour.getBestWtwTapPairsOut(); + } + else if ( modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + if ( directionIsInbound ) + bestTaps = tour.getBestWtdTapPairsIn(); + else + bestTaps = tour.getBestDtwTapPairsOut(); + } + + if ( bestTaps == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + //pick transit path from N-paths + float rn = (float)household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, bestTaps, household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)bestTaps[pathindex][0] ); + stop.setAlightTap( (int)bestTaps[pathindex][1] ); + stop.setSet( (int)bestTaps[pathindex][2] ); + } + + // inbound half-tour, only need park location choice if tour is + // work-based subtour; + // otherwise dest is home. + int park = -1; + if (directionIsInbound) + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + } else + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + + } + + } + + private void applySlcModelWithTiming(Household household, Person person, Tour tour, + Stop[] stops, int origMgra, int destMgra, boolean directionIsInbound) + { + + int lastDest = -1; + int newOrig = -1; + + // get the array of distances from the tour origin mgra to all MGRAs. + // use these distances for tour orig to stop alt distances + long check = System.nanoTime(); + anm.getDistancesFromMgra(origMgra, tourOrigToAllMgraDistances, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + anm.getDistancesFromMgra(destMgra, tourDestToAllMgraDistances, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + // anm.getDistancesToMgra( destMgra, tourDestToAllMgraDistances, + // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) ); + sldTime += (System.nanoTime() - check); + + // if there are stops on this half-tour, determine their destinations, + // depart hours, trip modes, and parking tazs. + if (stops != null) + { + + int oldSelectedIndex = -1; + earlierTripWasLocatedInAlightingTapShed = false; + + for (int i = 0; i < stops.length; i++) + { + + Stop stop = stops[i]; + + // if i is 0, the stop origin is set to origMgra; otherwise stop + // orig is the chosen dest from the previous stop. + if (i == 0) newOrig = origMgra; + else newOrig = lastDest; + stop.setOrig(newOrig); + + stopLocDmuObj.setStopObject(stop); + stopLocDmuObj.setDmuIndexValues(household.getHhId(), household.getHhMgra(), + newOrig, destMgra); + + int choice = -1; + int selectedIndex = -1; + int modeAlt = -1; + float modeLogsum = 0; + double vot= -1; + + // if not the last stop object, make a destination choice and a + // mode choice from IK MC probabilities; + // otherwise stop dest is set to destMgra, and make a mode + // choice from KJ MC probabilities. + if (i < stops.length - 1) + { + + //new code - depart period to and from stop + int departPeriodToStop = stop.getStopPeriod(); + int departPeriodFromStop = -1; + if(!directionIsInbound) + departPeriodFromStop = tour.getOutboundStops()[i+1].getStopPeriod(); + else + departPeriodFromStop = tour.getInboundStops()[i+1].getStopPeriod(); + + try + { + + check = System.nanoTime(); + + selectedIndex = selectDestinationWithTiming(stop,departPeriodToStop,departPeriodFromStop); + //close small probability logical hole, reset stop destination as intrazonal stop, log out reset cases + if(selectedIndex<0) { + selectedIndex=0; + choice=origMgra; + stop.setDest(choice); + modeAlt=tour.getTourModeChoice(); + modeLogsum=0; + logger.warn("Stop ID"+stop.id+" :destination set as intrazonal stop"); + }else { + choice = finalSample[selectedIndex]; + stop.setDest(choice); + modeAlt = selectModeFromProbabilities(stop, + mcCumProbsSegmentIk[selectedIndex]); + modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; + } + + if (sampleMgraInAlightingTapShed[choice]) + earlierTripWasLocatedInAlightingTapShed = true; + lastDest = choice; + slcTime += (System.nanoTime() - check); + + if (household.getDebugChoiceModels()) + { + if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " for joint tour stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + "N/A" + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + + ", tourId=" + tour.getTourId() + ", tourMode=" + + tour.getTourModeChoice()); + smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + } else + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + + tour.getTourModeChoice()); + smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + } + } + + check = System.nanoTime(); + /* + modeAlt = selectModeFromProbabilities(stop, + mcCumProbsSegmentIk[selectedIndex]); + modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; + */ + + if (modeAlt < 0) + { + logger.info("error getting trip mode choice for IK proportions, i=" + i); + logger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + logger.info("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + throw new RuntimeException(); + } + + //value of time; lookup vot, votS2, or votS3 depending on chosen mode + if(modelStructure.getTripModeIsS2(modeAlt)){ + vot = mcVOTsSegmentIk[selectedIndex][1]; + }else if (modelStructure.getTripModeIsS3(modeAlt)){ + vot = mcVOTsSegmentIk[selectedIndex][2]; + }else{ + vot = mcVOTsSegmentIk[selectedIndex][0]; + } + + int park = -1; + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + if (park > 0) lastDest = park; + float parkingCost = parkingCostSegmentIk[selectedIndex]; + stop.setParkingCost(parkingCost); + + } + + smcTime += (System.nanoTime() - check); + } catch (Exception e) + { + logger.error(String.format( + "Exception caught processing %s stop location choice model.", + (stopLocDmuObj.getInboundStop() == 1 ? "inbound" : "outbound"))); + logger.error("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tour category=" + + tour.getTourCategory() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + logger.error("StopID=" + (stop.getStopId() + 1) + " of " + + (stops.length - 1) + " stops, inbound=" + stop.isInboundStop() + + ", stopPurpose=" + stop.getDestPurpose() + ", stopDepart=" + + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + logger.error(String + .format("origMgra=%d, destMgra=%d, newOrig=%d, lastDest=%d, modeAlt=%d, selectedIndex=%d, choice=%d.", + origMgra, destMgra, newOrig, lastDest, modeAlt, + selectedIndex, choice)); + throw new RuntimeException(e); + } + + stop.setMode(modeAlt); + stop.setModeLogsum(modeLogsum); + stop.setValueOfTime(vot); + + // if the trip is a transit mode, set the boarding and + // alighting tap pairs in the stop object based on the ik + // segment pairs + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + + int accEgr = -1; + if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { + accEgr = WTW; + } else { + if (stop.isInboundStop()) { + accEgr = WTD; + } else { + accEgr = DTW; + } + } + + if ( segmentIkBestTapPairs[selectedIndex] == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + //pick transit path from N-paths + double rn = household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, segmentIkBestTapPairs[selectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][0] ); + stop.setAlightTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][1] ); + stop.setSet( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][2] ); + + } + + } + + oldSelectedIndex = selectedIndex; + + } else + { + + // last stop on half-tour, so dest is the half-tour final + // dest, and oldSelectedIndex was + // the selectedIndex determined for the previous stop + // location choice. + stop.setDest(destMgra); + + if (household.getDebugChoiceModels()) + { + if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " joint tour stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + "N/A" + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=End of " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " half-tour, stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + + stop.getOrig() + ", stopDest=" + stop.getDest()); + } else + { + smcLogger + .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " stop."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=End of " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " half-tour, stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + + stop.getOrig() + ", stopDest=" + stop.getDest()); + } + } + check = System.nanoTime(); + //Wu added + if(mcCumProbsSegmentKj[oldSelectedIndex]!=null&&mcLogsumsSegmentKj!=null) { + modeAlt = selectModeFromProbabilities(stop, + mcCumProbsSegmentKj[oldSelectedIndex]); + modeLogsum = (float) mcLogsumsSegmentKj[oldSelectedIndex]; + }else { + modeAlt = tour.getTourModeChoice(); + modeLogsum = 0; + logger.warn("Stop ID"+stop.id+" :mode and mode logsum reset."); + } + + if (modeAlt < 0) + { + logger.error("error getting trip mode choice for KJ proportions, i=" + i); + logger.error("HHID=" + household.getHhId() + ", persNum=" + + person.getPersonNum() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() + + ", tourMode=" + tour.getTourModeChoice()); + logger.error("StopID=End of " + + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") + + " half-tour, stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + + stop.getOrig() + ", stopDest=" + stop.getDest()); + throw new RuntimeException(); + } + + //value of time; lookup vot, votS2, or votS3 depending on chosen mode + if(modelStructure.getTripModeIsS2(modeAlt)){ + vot = mcVOTsSegmentKj[oldSelectedIndex][1]; + }else if (modelStructure.getTripModeIsS3(modeAlt)){ + vot = mcVOTsSegmentKj[oldSelectedIndex][2]; + }else{ + vot = mcVOTsSegmentKj[oldSelectedIndex][0]; + } + + + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) { + float parkingCost = parkingCostSegmentKj[oldSelectedIndex]; + stop.setParkingCost(parkingCost); + } + // last stop on tour, so if inbound, only need park location + // choice if tour is work-based subtour; + // otherwise dest is home. + int park = -1; + if (directionIsInbound) + { + if (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + } else + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + + smcTime += (System.nanoTime() - check); + + stop.setMode(modeAlt); + stop.setModeLogsum(modeLogsum); + stop.setValueOfTime(vot); + + // if the last trip is a transit mode, set the boarding and + // alighting tap pairs in the stop object based on the kj + // segment pairs + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) || modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + + int accEgr = -1; + if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { + accEgr = WTW; + } else { + if (stop.isInboundStop()) { + accEgr = WTD; + } else { + accEgr = DTW; + } + } + + if ( segmentKjBestTapPairs[oldSelectedIndex] == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + //pick transit path from N-paths + float rn = (float)household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, segmentKjBestTapPairs[oldSelectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][0] ); + stop.setAlightTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][1] ); + stop.setSet( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][2] ); + + } + + } + + } + + } + + } else + { // create a stop object to hold attributes for orig, dest, mode, + // departtime, etc. + // for the half-tour with no stops. + + check = System.nanoTime(); + + // create a Stop object for use in applying trip mode choice for + // this half tour without stops + String origStopPurpose = ""; + String destStopPurpose = ""; + if (!directionIsInbound) + { + origStopPurpose = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + destStopPurpose = tour.getTourPrimaryPurpose(); + } else + { + origStopPurpose = tour.getTourPrimaryPurpose(); + destStopPurpose = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + } + + Stop stop = null; + try + { + stop = tour.createStop(origStopPurpose, destStopPurpose, + directionIsInbound, + tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)); + } catch (Exception e) + { + logger.info("exception creating stop."); + } + + stop.setOrig(origMgra); + stop.setDest(destMgra); + + if (directionIsInbound) stop.setStopPeriod(tour.getTourArrivePeriod()); + else stop.setStopPeriod(tour.getTourDepartPeriod()); + + int modeAlt = getHalfTourModeChoice(stop); + if (modeAlt < 0) + { + logger.info("error getting trip mode choice for half-tour with no stops."); + logger.info("HHID=" + household.getHhId() + ", tourPurpose=" + + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId()); + logger.info("StopID=" + (stop.getStopId() + 1) + " of no stops, inbound=" + + stop.isInboundStop() + ", stopPurpose=" + stop.getDestPurpose() + + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() + + ", stopDest=" + stop.getDest()); + + modeAlt = stop.getTour().getTourModeChoice(); + // throw new RuntimeException(); + } + + stop.setMode(modeAlt); + + double[][] bestTaps = null; + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) ) { + if ( directionIsInbound ) + bestTaps = tour.getBestWtwTapPairsIn(); + else + bestTaps = tour.getBestWtwTapPairsOut(); + } + else if ( modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + if ( directionIsInbound ) + bestTaps = tour.getBestWtdTapPairsIn(); + else + bestTaps = tour.getBestDtwTapPairsOut(); + } + + if ( bestTaps == null ) { + stop.setBoardTap( 0 ); + stop.setAlightTap( 0 ); + stop.setSet( 0 ); + } + else { + + // set taps + if ( modelStructure.getTripModeIsWalkTransit(modeAlt) || modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { + + //pick transit path from N-paths + float rn = (float)household.getHhRandom().nextDouble(); + int pathindex = logsumHelper.chooseTripPath(rn, bestTaps, household.getDebugChoiceModels(), smcLogger); + + stop.setBoardTap( (int)bestTaps[pathindex][0] ); + stop.setAlightTap( (int)bestTaps[pathindex][1] ); + stop.setSet( (int)bestTaps[pathindex][2] ); + + } + + } + + // inbound half-tour, only need park location choice if tour is + // work-based subtour; + // otherwise dest is home. + int park = -1; + if (directionIsInbound) + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + } else + { + if (modelStructure.getTripModeIsSovOrHov(modeAlt)) + { + park = selectParkingLocation(household, tour, stop); + stop.setPark(park); + } + } + + smcTime += (System.nanoTime() - check); + } + + } + + private int selectDestination(Stop s, int departPeriodToStop, int departPeriodFromStop) + { + + Logger modelLogger = slcLogger; + + int[] loggingSample = null; + int[] debugLoggingSample = null; + // int[] debugLoggingSample = { 0, 16569 }; + // int[] debugLoggingSample = { 0, 4886, 16859, 18355, 3222, 14879, + // 26894, 16512, 9908, 18287, 14989 }; + + Tour tour = s.getTour(); + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + + if (household.getDebugChoiceModels()) + { + household.logHouseholdObject( + "Pre Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" + + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" + + tour.getTourPurpose() + ", Tour_" + tour.getTourId() + + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" + + (s.getStopId() + 1), modelLogger); + household.logPersonObject("Pre Stop Location Choice for person " + + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); + household.logTourObject("Pre Stop Location Choice for tour " + tour.getTourId(), + modelLogger, tour.getPersonObject(), tour); + household.logStopObject("Pre Stop Location Choice for stop " + (s.getStopId() + 1), + modelLogger, s, modelStructure); + + loggingSample = debugLoggingSample; + } + + int numAltsInSample = -1; + + stopLocDmuObj.setTourModeIndex(tour.getTourModeChoice()); + + // set the size terms array for the stop purpose in the dmu object + stopLocDmuObj + .setLogSize(getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household)); + + // get the array of distances from the stop origin mgra to all MGRAs and + // set in the dmu object + anm.getDistancesFromMgra(s.getOrig(), distanceFromStopOrigToAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesFromOrigMgra(distanceFromStopOrigToAllMgras); + + + // bike logsums from origin to all destinations + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumFromStopOrigToAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int dMgra = 1; dMgra <= mgraManager.getMaxMgra(); dMgra++) + { + bikeLogsumFromStopOrigToAllMgras[dMgra] = bls.getLogsum(segment,s.getOrig(),dMgra); + } + stopLocDmuObj.setBikeLogsumsFromOrigMgra(bikeLogsumFromStopOrigToAllMgras); + } + + + + // if tour mode is transit, set availablity of location alternatives + // based on transit accessibility relative to best transit TAP pair for + // tour + if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) + { + + Arrays.fill(sampleMgraInBoardingTapShed, false); + Arrays.fill(sampleMgraInAlightingTapShed, false); + + int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, tour, household.getDebugChoiceModels()); + if (numAvailableAlternatives == 0) + { + logger.error("no available locations - empty sample."); + logger.error("best tap pair which is empty: " + Arrays.deepToString(s.isInboundStop() ? tour.getBestWtwTapPairsIn() : tour.getBestWtwTapPairsOut())); + throw new RuntimeException(); + } + } + + // get the array of distances to the half-tour final destination mgra + // from all MGRAs and set in the dmu object + if (s.isInboundStop()) + { + // if inbound, final half-tour destination is the tour origin + anm.getDistancesToMgra(tour.getTourOrigMgra(), distanceToFinalDestFromAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); + + + // set the distance from the stop origin to the final half-tour + // destination + stopLocDmuObj + .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()]); + + + // bike logsums from all MGRAs back to tour origin + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) + { + bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourOrigMgra()); + } + stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); + } + + + + + + } else + { + // if outbound, final half-tour destination is the tour destination + anm.getDistancesToMgra(tour.getTourDestMgra(), distanceToFinalDestFromAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); + + // set the distance from the stop origin to the final half-tour + // destination + stopLocDmuObj + .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); + + // bike logsums from all MGRAs back to tour origin + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) + { + bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourDestMgra()); + } + stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); + } + + + + + } + + if (useNewSoaMethod) + { + if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) selectSampleOfAlternativesAutoTourNew( + s, tour, person, household, loggingSample); + else selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); + + numAltsInSample = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); + } else + { + if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) selectSampleOfAlternativesAutoTour( + s, tour, person, household, loggingSample); + else selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); + + numAltsInSample = altFreqMap.size(); + } + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + String separator = ""; + + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = "Stop Location Choice"; + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1)); + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + setupStopLocationChoiceAlternativeArrays(numAltsInSample, s, departPeriodToStop,departPeriodFromStop); + + int slcModelIndex = -1; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) slcModelIndex = MAND_SLC_MODEL_INDEX; + else if (tour.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX + || tour.getTourPrimaryPurposeIndex() == ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX + || tour.getTourPrimaryPurposeIndex() == ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX) slcModelIndex = MAINT_SLC_MODEL_INDEX; + else slcModelIndex = DISCR_SLC_MODEL_INDEX; + + float logsum = (float) slcModelArray[slcModelIndex].computeUtilities(stopLocDmuObj, + stopLocDmuObj.getDmuIndexValues(), sampleAvailability, inSample); + + if(s.isInboundStop()) + tour.addInboundStopDestinationLogsum(logsum); + else + tour.addOutboundStopDestinationLogsum(logsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + int selectedIndex = -1; + if (slcModelArray[slcModelIndex].getAvailabilityCount() > 0) + { + chosen = slcModelArray[slcModelIndex].getChoiceResult(rn); + selectedIndex = chosen; + }else{ + //wu's tempory fix to set chosen stop alternative to origin mgra if no alternative is available-8/27/2014 + //instead of this method, seems selectDestinationWithTiming(Stop s) is called (similar change made there too) + //tempory fix is put in here just in case. + chosen=tour.getTourOrigMgra(); + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels() || chosen < 0) + { + + if (chosen < 0) + { + + modelLogger + .error("ERROR selecting stop location choice due to no alternatives available."); + modelLogger + .error("setting debug to true and recomputing sample of alternatives selection."); + modelLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourPurpose(), + tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1), s.getOrig())); + + choiceModelDescription = "Stop Location Choice"; + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1), s.getOrig()); + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + + modelLogger.error(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.error(loggingHeader); + modelLogger.error(separator); + modelLogger.error(""); + modelLogger.error(""); + + // utilities and probabilities are 0 based. + double[] utilities = slcModelArray[slcModelIndex].getUtilities(); + double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); + + // availabilities is 1 based. + boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger + .error("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .error("Alternative Availability Utility Probability CumProb"); + modelLogger + .error("--------------------- ------------ ----------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 1; j <= numAltsInSample; j++) + { + + int alt = finalSample[j]; + + if (j == chosen) selectedIndex = j; + + cumProb += probabilities[j - 1]; + String altString = String.format("%-3d %5d", j, alt); + modelLogger.error(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); + } + + modelLogger.error(" "); + String altString = String.format("%-3d %5d", selectedIndex, -1); + modelLogger.error(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.error(separator); + modelLogger.error(""); + modelLogger.error(""); + + slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log + // file + slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); + + logger.error(String + .format("Error for HHID=%d, PersonNum=%d, no available %s stop destination choice alternatives to choose from in choiceModelApplication.", + tour.getHhId(), tour.getPersonObject().getPersonNum(), + tour.getTourPurpose())); + throw new RuntimeException(); + + } + + // utilities and probabilities are 0 based. + double[] utilities = slcModelArray[slcModelIndex].getUtilities(); + double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); + + // availabilities is 1 based. + boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- ------------ ----------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 1; j <= numAltsInSample; j++) + { + + int alt = finalSample[j]; + + if (j == chosen) selectedIndex = j; + + cumProb += probabilities[j - 1]; + String altString = String.format("%-3d %5d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %5d", selectedIndex, finalSample[selectedIndex]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); + + } + + return selectedIndex; + } + + private int selectDestinationWithTiming(Stop s,int departPeriodToStop,int departPeriodFromStop) + { + + Logger modelLogger = slcLogger; + + int[] loggingSample = null; + int[] debugLoggingSample = null; + // int[] debugLoggingSample = { 0, 16569 }; + // int[] debugLoggingSample = { 0, 4886, 16859, 18355, 3222, 14879, + // 26894, 16512, 9908, 18287, 14989 }; + + Tour tour = s.getTour(); + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + + if (household.getDebugChoiceModels()) + { + household.logHouseholdObject( + "Pre Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" + + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" + + tour.getTourPurpose() + ", Tour_" + tour.getTourId() + + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" + + (s.getStopId() + 1), modelLogger); + household.logPersonObject("Pre Stop Location Choice for person " + + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); + household.logTourObject("Pre Stop Location Choice for tour " + tour.getTourId(), + modelLogger, tour.getPersonObject(), tour); + household.logStopObject("Pre Stop Location Choice for stop " + (s.getStopId() + 1), + modelLogger, s, modelStructure); + + loggingSample = debugLoggingSample; + } + + int numAltsInSample = -1; + + stopLocDmuObj.setTourModeIndex(tour.getTourModeChoice()); + + // set the size terms array for the stop purpose in the dmu object + stopLocDmuObj + .setLogSize(getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household)); + + // get the array of distances from the stop origin mgra to all MGRAs and + // set in the dmu object + anm.getDistancesFromMgra(s.getOrig(), distanceFromStopOrigToAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesFromOrigMgra(distanceFromStopOrigToAllMgras); + + + // bike logsums from origin to all destinations + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumFromStopOrigToAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int dMgra = 1; dMgra <= mgraManager.getMaxMgra(); dMgra++) + { + bikeLogsumFromStopOrigToAllMgras[dMgra] = bls.getLogsum(segment,s.getOrig(),dMgra); + } + stopLocDmuObj.setBikeLogsumsFromOrigMgra(bikeLogsumFromStopOrigToAllMgras); + } + + // if tour mode is transit, set availablity of location alternatives + // based on transit accessibility relative to best transit TAP pair for + // tour + if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) + { + + Arrays.fill(sampleMgraInBoardingTapShed, false); + Arrays.fill(sampleMgraInAlightingTapShed, false); + + int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, tour, household.getDebugChoiceModels()); + if (numAvailableAlternatives == 0) + { + logger.error("no available locations - empty sample."); + logger.error("best tap pair which is empty: " + Arrays.deepToString(s.isInboundStop() ? tour.getBestWtwTapPairsIn() : tour.getBestWtwTapPairsOut())); + throw new RuntimeException(); + } + } + + // get the array of distances to the half-tour final destination mgra + // from all MGRAs and set in the dmu object + if (s.isInboundStop()) + { + // if inbound, final half-tour destination is the tour origin + anm.getDistancesToMgra(tour.getTourOrigMgra(), distanceToFinalDestFromAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); + + // set the distance from the stop origin to the final half-tour + // destination + stopLocDmuObj + .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()]); + + // bike logsums from all MGRAs back to tour origin + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) + { + bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourOrigMgra()); + } + stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); + } + + + } else + { + // if outbound, final half-tour destination is the tour destination + anm.getDistancesToMgra(tour.getTourDestMgra(), distanceToFinalDestFromAllMgras, + modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); + stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); + + // set the distance from the stop origin to the final half-tour + // destination + stopLocDmuObj + .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); + + // bike logsums from all MGRAs back to tour origin + if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ + + Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); + segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); + + for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) + { + bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourDestMgra()); + } + stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); + } + + + } + + long check = System.nanoTime(); + if (useNewSoaMethod) + { + if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) + { + selectSampleOfAlternativesAutoTourNew(s, tour, person, household, loggingSample); + soaAutoTime += (System.nanoTime() - check); + numAltsInSample = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); + } else + { + selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); + soaOtherTime += (System.nanoTime() - check); + numAltsInSample = altFreqMap.size(); + } + } else + { + if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) + { + selectSampleOfAlternativesAutoTour(s, tour, person, household, loggingSample); + soaAutoTime += (System.nanoTime() - check); + } else + { + selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); + soaOtherTime += (System.nanoTime() - check); + } + numAltsInSample = altFreqMap.size(); + } + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + String separator = ""; + + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = "Stop Location Choice"; + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1)); + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + check = System.nanoTime(); + setupStopLocationChoiceAlternativeArrays(numAltsInSample, s,departPeriodToStop,departPeriodFromStop); + slsTime += (System.nanoTime() - check); + + int slcModelIndex = -1; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) slcModelIndex = MAND_SLC_MODEL_INDEX; + else if (tour.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX + || tour.getTourPrimaryPurposeIndex() == ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX + || tour.getTourPrimaryPurposeIndex() == ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX) slcModelIndex = MAINT_SLC_MODEL_INDEX; + else slcModelIndex = DISCR_SLC_MODEL_INDEX; + + float logsum = (float) slcModelArray[slcModelIndex].computeUtilities(stopLocDmuObj, + stopLocDmuObj.getDmuIndexValues(), sampleAvailability, inSample); + if(s.isInboundStop()) + tour.addInboundStopDestinationLogsum(logsum); + else + tour.addOutboundStopDestinationLogsum(logsum); + + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + int selectedIndex = -1; + if (slcModelArray[slcModelIndex].getAvailabilityCount() > 0) + { + selectedIndex = slcModelArray[slcModelIndex].getChoiceResult(rn); + chosen = finalSample[selectedIndex]; + }else{ + //wu's tempory fix to set chosen stop alternative to origin mgra if no alternative is available-8/27/2014 + chosen=tour.getTourOrigMgra(); + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels() || chosen < 0) + { + + if (chosen < 0) + { + + modelLogger + .error("ERROR selecting stop location choice due to no alternatives available."); + modelLogger + .error("setting debug to true and recomputing sample of alternatives selection."); + modelLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourPurpose(), + tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1), s.getOrig())); + + choiceModelDescription = "Stop Location Choice"; + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), + s.getDestPurpose(), (s.getStopId() + 1), s.getOrig()); + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + + modelLogger.error(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.error(loggingHeader); + modelLogger.error(separator); + modelLogger.error(""); + modelLogger.error(""); + + // utilities and probabilities are 0 based. + double[] utilities = slcModelArray[slcModelIndex].getUtilities(); + double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); + + // availabilities is 1 based. + boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger + .error("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .error("Alternative Availability Utility Probability CumProb"); + modelLogger + .error("--------------------- ------------ ----------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 1; j <= numAltsInSample; j++) + { + + int alt = finalSample[j]; + + if (j == chosen) selectedIndex = j; + + cumProb += probabilities[j - 1]; + String altString = String.format("%-3d %5d", j, alt); + modelLogger.error(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); + } + + modelLogger.error(" "); + String altString = String.format("%-3d %5d", selectedIndex, -1); + modelLogger.error(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.error(separator); + modelLogger.error(""); + modelLogger.error(""); + + slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log + // file + slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); + + logger.error(String + .format("Error for HHID=%d, PersonNum=%d, no available %s stop destination choice alternatives to choose from in choiceModelApplication.", + tour.getHhId(), tour.getPersonObject().getPersonNum(), + tour.getTourPurpose())); + throw new RuntimeException(); + + } + + // utilities and probabilities are 0 based. + double[] utilities = slcModelArray[slcModelIndex].getUtilities(); + double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); + + // availabilities is 1 based. + boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- ------------ ----------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 1; j <= numAltsInSample; j++) + { + + int alt = finalSample[j]; + + if (j == chosen) selectedIndex = j; + + cumProb += probabilities[j - 1]; + String altString = String.format("%-3d %5d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %5d", selectedIndex, finalSample[selectedIndex]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); + + } + + return selectedIndex; + } + + private void setupStopLocationChoiceAlternativeArrays(int numAltsInSample, Stop s,int departPeriodToStop, int departPeriodFromStop) + { + + stopLocDmuObj.setNumberInSample(numAltsInSample); + stopLocDmuObj.setSampleOfAlternatives(finalSample); + stopLocDmuObj.setSlcSoaCorrections(sampleCorrectionFactors); + + // create arrays for ik and kj mode choice logsums for the stop origin, + // the sample stop location, and the half-tour final destination. + setupLogsumCalculation(s); + + int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; + ChoiceModelApplication mcModel = mcModelArray[category]; + + Household household = s.getTour().getPersonObject().getHouseholdObject(); + double income = (double) household.getIncomeInDollars(); + double ivtCoeff = ivtCoeffs[category]; + double incomeCoeff = incomeCoeffs[category]; + double incomeExpon = incomeExponents[category]; + double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); + double timeFactor = 1.0f; + if(s.getTour().getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + timeFactor = mcDmuObject.getJointTourTimeFactor(); + else if(s.getTour().getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) + timeFactor = mcDmuObject.getWorkTimeFactor(); + else + timeFactor = mcDmuObject.getNonWorkTimeFactor(); + + mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); + mcDmuObject.setCostCoeff(costCoeff); + + int halfTourFinalDest = s.isInboundStop() ? s.getTour().getTourOrigMgra() : s.getTour() + .getTourDestMgra(); + + // set the land use data items in the DMU for the stop origin + mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(s.getOrig())); + mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(s.getOrig())); + mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(s.getOrig())); + + for (int i = 1; i <= numAltsInSample; i++) + { + + int altMgra = finalSample[i]; + mcDmuObject.getDmuIndexValues().setDestZone(altMgra); + + // set distances to/from stop anchor points to stop location alternative. + ikDistance[i] = distanceFromStopOrigToAllMgras[altMgra]; + kjDistance[i] = distanceToFinalDestFromAllMgras[altMgra]; + + // set distances from tour anchor points to stop location + // alternative. + okDistance[i] = tourOrigToAllMgraDistances[altMgra]; + kdDistance[i] = tourDestToAllMgraDistances[altMgra]; + + // set the land use data items in the DMU for the sample location + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(altMgra)); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(altMgra)); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(altMgra)); + + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(altMgra))); + + // for walk-transit tours - if half-tour direction is outbound and + // stop alternative is in the walk shed, walk and walk-transit + // should be allowed for ik segments + // if half-tour direction is inbound and stop alternative is in the + // walk shed, walk and walk-transit should be allowed for kj + // segments + // if half-tour direction is outbound and stop alternative is in the + // walk shed, walk and walk-transit should be allowed for kj + // segments + // if half-tour direction is inbound and stop alternative is in the + // walk shed, walk and walk-transit should be allowed for ik + // segments + + // for drive-transit tours - if half-tour direction is outbound and + // stop alternative is in the drive shed, auto should be allowed for + // ik segments + // if half-tour direction is inbound and stop alternative is in the + // drive shed, auto should be allowed for kj segments + // if half-tour direction is outbound and stop alternative is in the + // walk shed, walk and walk-transit should be allowed for kj + // segments + // if half-tour direction is inbound and stop alternative is in the + // walk shed, walk and walk-transit should be allowed for ik + // segments + + // set values for walk-transit and drive-transit tours according to + // logic for IK segments + mcDmuObject.setAutoModeRequiredForTripSegment(false); + mcDmuObject.setWalkModeAllowedForTripSegment(false); + + mcDmuObject.setSegmentIsIk(true); + + double ikSegment = -999; + // drive transit tours are handled differently than walk transit + // tours + if (modelStructure.getTourModeIsDriveTransit(s.getTour().getTourModeChoice())) + { + + // if the direction is outbound + if (!s.isInboundStop()) + { + + // if the sampled mgra is in the outbound half-tour boarding tap shed (near tour origin) + if ( sampleMgraInBoardingTapShed[altMgra] ) { + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + } + + // if the sampled mgra is in the outbound half-tour alighting tap shed (near tour primary destination) + if ( sampleMgraInAlightingTapShed[altMgra] ) { + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject,s.getOrig(),altMgra,departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); + } + + + // if the trip origin and sampled mgra are in the outbound half-tour alighting tap shed (near tour origin) + if ( sampleMgraInAlightingTapShed[s.getOrig()] && sampleMgraInAlightingTapShed[altMgra] ) { + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, s.getOrig(), altMgra, departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + } + + } else + { + // if the sampled mgra is in the inbound half-tour boarding tap shed (near tour primary destination) + if ( sampleMgraInBoardingTapShed[altMgra] ) { + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, s.getOrig(), altMgra, departPeriodToStop, s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels() ); + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + } + + // if the sampled mgra is in the inbound half-tour alighting tap shed (near tour origin) + if ( sampleMgraInAlightingTapShed[altMgra] ) { + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject,s.getOrig(),altMgra,departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); + } + + // if the trip origin and sampled mgra are in the inbound half-tour alighting tap shed (near tour origin) + if ( sampleMgraInAlightingTapShed[s.getOrig()] && sampleMgraInAlightingTapShed[altMgra] ) { + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + } + + } + + } else if (modelStructure.getTourModeIsWalkTransit(s.getTour().getTourModeChoice())) + { // tour mode is walk-transit + + logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, s.getOrig(), altMgra, + departPeriodToStop, s.getTour().getPersonObject().getHouseholdObject() + .getDebugChoiceModels()); + + } else + { + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + } + ikSegment = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, departPeriodToStop, + mcModel, mcDmuObject, slcLogger); + + if (ikSegment < -900) + { + slcLogger.error("ERROR calculating trip mode choice logsum for " + + (s.isInboundStop() ? "inbound" : "outbound") + + " stop location choice - ikLogsum = " + ikSegment + "."); + slcLogger + .error("setting debug to true and recomputing ik segment logsum in order to log utility expression results."); + + if (s.isInboundStop()) slcLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, TourOrigMGRA=%d, TourDestMGRA=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumIBStops=%d, StopOrig=%d, AltStopLoc=%d", + s.getTour().getPersonObject().getHouseholdObject() + .getHhId(), s.getTour().getPersonObject() + .getPersonNum(), s.getTour().getPersonObject() + .getPersonType(), s.getTour().getTourPurpose(), s + .getTour().getTourModeChoice(), s.getTour() + .getTourId(), s.getTour().getTourOrigMgra(),s.getTour().getTourDestMgra(),s.getDestPurpose(), "inbound", (s + .getStopId() + 1), + s.getTour().getNumInboundStops() - 1, s.getOrig(), altMgra)); + else slcLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d,TourOrigMGRA=%d,TourDestMGRA=%d,StopPurpose=%s, StopDirection=%s, StopId=%d, NumOBStops=%d, StopOrig=%d, AltStopLoc=%d", + s.getTour().getPersonObject().getHouseholdObject() + .getHhId(), s.getTour().getPersonObject() + .getPersonNum(), s.getTour().getPersonObject() + .getPersonType(), s.getTour().getTourPurpose(), s + .getTour().getTourModeChoice(), s.getTour() + .getTourId(),s.getTour().getTourOrigMgra(),s.getTour().getTourDestMgra(),s.getDestPurpose(), "outbound", (s + .getStopId() + 1), s.getTour() + .getNumOutboundStops() - 1, s.getOrig(), altMgra)); + + mcDmuObject.getDmuIndexValues().setDebug(true); + mcDmuObject.getHouseholdObject().setDebugChoiceModels(true); + /* suppress log: Wu + mcDmuObject.getHouseholdObject().setDebugChoiceModels(true); + */ + mcDmuObject.getDmuIndexValues().setHHIndex( + s.getTour().getPersonObject().getHouseholdObject().getHhId()); + ikSegment = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, + departPeriodToStop, mcModel, mcDmuObject, slcLogger); + mcDmuObject.getDmuIndexValues().setDebug(false); + mcDmuObject.getHouseholdObject().setDebugChoiceModels(false); + + } + + // store the mode choice probabilities for the segment + mcCumProbsSegmentIk[i] = logsumHelper.getStoredSegmentCumulativeProbabilities(); + mcVOTsSegmentIk[i] = logsumHelper.getStoredSegmentVOTs(); + parkingCostSegmentIk[i] = logsumHelper.getTripModeChoiceSegmentStoredParkingCost(); + + // Store the mode choice logsum for the segment + mcLogsumsSegmentIk[i] = ikSegment; + + // store the best tap pairs for the segment + for(int j=0; j it = altFreqMap.keySet().iterator(); + int k = 0; + while (it.hasNext()) + { + + int alt = it.next(); + int freq = altFreqMap.get(alt); + + double prob = 0; + prob = probabilitiesList[alt - 1]; + + finalSample[k + 1] = alt; + sampleCorrectionFactors[k + 1] = Math.log((double) freq / prob); + + k++; + } + + while (k < sampleSize) + { + finalSample[k + 1] = -1; + sampleCorrectionFactors[k + 1] = Double.NaN; + sampleAvailability[k + 1] = false; + inSample[k + 1] = 0; + k++; + } + + } + + private void selectSampleOfAlternativesOther(Stop s, Tour tour, Person person, + Household household, int[] loggingSample) + { + + Logger soaLogger = Logger.getLogger("slcSoaLogger"); + + altFreqMap.clear(); + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + ChoiceModelApplication cm; + if (modelStructure.getTourModeIsWalk(tour.getTourModeChoice())) cm = slcSoaModel[WALK_STOP_LOC_SOA_SHEET_INDEX]; + else if (modelStructure.getTourModeIsBike(tour.getTourModeChoice())) cm = slcSoaModel[BIKE_STOP_LOC_SOA_SHEET_INDEX]; + else cm = slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX]; + + if (household.getDebugChoiceModels()) + { + choiceModelDescription = String + .format("Stop Location SOA Choice Model for: stop purpose=%s, direction=%s, stopId=%d, stopOrig=%d", + s.getDestPurpose(), s.isInboundStop() ? "inbound" : "outbound", + (s.getStopId() + 1), s.getOrig()); + decisionMakerLabel = String + .format("HH=%d, persNum=%d, persType=%s, tourId=%d, tourPurpose=%s, tourOrig=%d, tourDest=%d, tourMode=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourId(), tour.getTourPrimaryPurpose(), tour.getTourOrigMgra(), + tour.getTourDestMgra(), tour.getTourModeChoice()); + cm.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, decisionMakerLabel); + } + + IndexValues dmuIndex = stopLocDmuObj.getDmuIndexValues(); + dmuIndex.setDebug(household.getDebugChoiceModels()); + + // stopLocDmuObj.setTourModeIndex( tour.getTourModeChoice() ); + // + // // set the size terms array for the stop purpose in the dmu object + // stopLocDmuObj.setLogSize( + // getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household) + // ); + // + // // get the array of distances from the stop origin mgra to all MGRAs + // and set in the dmu object + // anm.getDistancesFromMgra( s.getOrig(), + // distanceFromStopOrigToAllMgras, modelStructure.getTourModeIsSovOrHov( + // tour.getTourModeChoice() ) ); + // stopLocDmuObj.setDistancesFromOrigMgra( + // distanceFromStopOrigToAllMgras ); + // + // // if tour mode is transit, set availablity of location alternatives + // based on transit accessibility relative to best transit TAP pair for + // tour + // if ( modelStructure.getTourModeIsTransit( tour.getTourModeChoice() ) + // ) { + // int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, + // tour); + // if ( numAvailableAlternatives == 0 ) { + // logger.error( "no available locations - empty sample." ); + // throw new RuntimeException(); + // } + // } + // + // // get the array of distances to the half-tour final destination mgra + // from all MGRAs and set in the dmu object + if (s.isInboundStop()) + { + // // if inbound, final half-tour destination is the tour origin + // anm.getDistancesToMgra( tour.getTourOrigMgra(), + // distanceToFinalDestFromAllMgras, + // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) + // ); + // stopLocDmuObj.setDistancesToDestMgra( + // distanceToFinalDestFromAllMgras ); + // + // // set the distance from the stop origin to the final half-tour + // destination + // stopLocDmuObj.setOrigDestDistance( + // distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()] ); + // + // // not used in UEC to reference matrices, but may be for + // debugging using $ORIG and $DEST as an expression + dmuIndex.setOriginZone(mgraManager.getTaz(s.getOrig())); + dmuIndex.setDestZone(mgraManager.getTaz(tour.getTourOrigMgra())); + } else + { + // // if outbound, final half-tour destination is the tour + // destination + // anm.getDistancesToMgra( tour.getTourDestMgra(), + // distanceToFinalDestFromAllMgras, + // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) + // ); + // stopLocDmuObj.setDistancesToDestMgra( + // distanceToFinalDestFromAllMgras ); + // + // // set the distance from the stop origin to the final half-tour + // destination + // stopLocDmuObj.setOrigDestDistance( + // distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); + // + // // not used in UEC to reference matrices, but may be for + // debugging using $ORIG and $DEST as an expression + dmuIndex.setOriginZone(mgraManager.getTaz(s.getOrig())); + dmuIndex.setDestZone(mgraManager.getTaz(tour.getTourDestMgra())); + } + + cm.computeUtilities(stopLocDmuObj, dmuIndex, soaAvailability, soaSample); + double[] probabilitiesList = cm.getProbabilities(); + double[] cumProbabilitiesList = cm.getCumulativeProbabilities(); + + // debug output + if (household.getDebugChoiceModels()) + { + + // write choice model alternative info to debug log file + cm.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + + // write UEC calculation results to separate model specific log file + loggingHeader = choiceModelDescription + ", " + decisionMakerLabel; + + if (loggingSample == null) + { + cm.logUECResultsSpecificAlts(soaLogger, loggingHeader, new int[] {0, s.getOrig(), + tour.getTourOrigMgra(), tour.getTourDestMgra()}); + // cm.logUECResults( soaLogger, loggingHeader, 10 ); + } else + { + cm.logUECResultsSpecificAlts(soaLogger, loggingHeader, loggingSample); + } + + } + + // loop over sampleSize, select alternatives based on probabilitiesList, + // and count frequency of alternatives chosen. + // may include duplicate alternative selections. + + Random hhRandom = household.getHhRandom(); + int rnCount = household.getHhRandomCount(); + // when household.getHhRandom() was applied, the random count was + // incremented, assuming a random number would be drawn right away. + // so let's decrement by 1, then increment the count each time a random + // number is actually drawn in this method. + rnCount--; + + // log degenerative cases + if (cm.getAvailabilityCount() == 0) + { + Logger badSlcLogger = Logger.getLogger("badSlc"); + + choiceModelDescription = String + .format("Stop Location SOA Choice Model for: stop purpose=%s, direction=%s, stopId=%d, stopOrig=%d", + s.getDestPurpose(), s.isInboundStop() ? "inbound" : "outbound", + (s.getStopId() + 1), s.getOrig()); + decisionMakerLabel = String + .format("HH=%d, persNum=%d, persType=%s, tourId=%d, tourPurpose=%s, tourOrig=%d, tourDest=%d, tourMode=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourId(), tour.getTourPrimaryPurpose(), tour.getTourOrigMgra(), + tour.getTourDestMgra(), tour.getTourModeChoice()); + loggingHeader = choiceModelDescription + ", " + decisionMakerLabel; + + badSlcLogger.info("....... Start Logging ......."); + badSlcLogger + .info("setting stop location sample to be an array with 1 element - just the stop origin mgra."); + badSlcLogger.info(""); + + household.logHouseholdObject( + "Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" + + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" + + tour.getTourPurpose() + ", Tour_" + tour.getTourId() + + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" + + (s.getStopId() + 1), badSlcLogger); + household.logPersonObject("Stop Location Choice for person " + + tour.getPersonObject().getPersonNum(), badSlcLogger, tour.getPersonObject()); + household.logTourObject("Stop Location Choice for tour " + tour.getTourId(), + badSlcLogger, tour.getPersonObject(), tour); + household.logStopObject("Stop Location Choice for stop " + (s.getStopId() + 1), + badSlcLogger, s, modelStructure); + + badSlcLogger.info(decisionMakerLabel + " has no available alternatives for " + + choiceModelDescription + "."); + badSlcLogger.info("Logging StopLocation SOA Choice utility calculations for: stopOrig=" + + s.getOrig() + ", tourOrig=" + tour.getTourOrigMgra() + ", and tourDest=" + + tour.getTourDestMgra() + "."); + cm.logUECResultsSpecificAlts(badSlcLogger, loggingHeader, new int[] {0, s.getOrig(), + tour.getTourOrigMgra(), tour.getTourDestMgra()}); + + int chosenAlt = s.getOrig(); + probabilitiesList[chosenAlt - 1] = 1.0; + for (int j = chosenAlt - 1; j < cumProbabilitiesList.length; j++) + cumProbabilitiesList[j] = 1.0; + + double sum = 0; + double epsilon = .0000001; + for (int j = 0; j < probabilitiesList.length; j++) + { + sum += probabilitiesList[j]; + if (!(Math.abs(sum - cumProbabilitiesList[j]) < epsilon) || sum > 1.0) + { + badSlcLogger.info("error condition found! sum=" + sum + ", j=" + j + + ", cumProbabilitiesList[j]=" + cumProbabilitiesList[j]); + badSlcLogger.info("....... End Logging ......."); + throw new RuntimeException(); + } + } + + badSlcLogger.info("....... End Logging ......."); + badSlcLogger.info(""); + badSlcLogger.info(""); + } + + int chosenAlt = -1; + for (int i = 0; i < sampleSize; i++) + { + + double rn = hhRandom.nextDouble(); + rnCount++; + chosenAlt = Util.binarySearchDouble(cumProbabilitiesList, rn) + 1; + + // write choice model alternative info to log file + if (household.getDebugChoiceModels()) + { + cm.logSelectionInfo(loggingHeader, String.format("rnCount=%d", rnCount), rn, + chosenAlt); + } + + int freq = 0; + if (altFreqMap.containsKey(chosenAlt)) freq = altFreqMap.get(chosenAlt); + altFreqMap.put(chosenAlt, (freq + 1)); + + } + + // sampleSize random number draws were made from this Random object, so + // update the count in the hh's Random. + household.setHhRandomCount(rnCount); + + Arrays.fill(sampleAvailability, true); + Arrays.fill(inSample, 1); + + // create arrays of the unique chosen alternatives and the frequency + // with which those alternatives were chosen. + Iterator it = altFreqMap.keySet().iterator(); + int k = 0; + while (it.hasNext()) + { + + int alt = it.next(); + int freq = altFreqMap.get(alt); + + double prob = 0; + prob = probabilitiesList[alt - 1]; + + finalSample[k + 1] = alt; + sampleCorrectionFactors[k + 1] = Math.log((double) freq / prob); + + k++; + } + + while (k < sampleSize) + { + finalSample[k + 1] = -1; + sampleCorrectionFactors[k + 1] = Double.NaN; + sampleAvailability[k + 1] = false; + inSample[k + 1] = 0; + k++; + } + + // if the sample was determined for a transit tour, the sample and + // availability arrays for the full set of SOA alternatives need to be + // restored. + if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) + { + for (int i = 0; i < soaSample.length; i++) + { + soaSample[i] = soaSampleBackup[i]; + soaAvailability[i] = soaAvailabilityBackup[i]; + } + } + + } + + private void setupLogsumCalculation(Stop s) + { + + Tour t = s.getTour(); + Person p = t.getPersonObject(); + Household hh = p.getHouseholdObject(); + + mcDmuObject.setHouseholdObject(hh); + mcDmuObject.setPersonObject(p); + mcDmuObject.setTourObject(t); + + int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; + double income = (double) hh.getIncomeInDollars(); + double ivtCoeff = ivtCoeffs[category]; + double incomeCoeff = incomeCoeffs[category]; + double incomeExpon = incomeExponents[category]; + double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); + double timeFactor = 1.0f; + if(t.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + timeFactor = mcDmuObject.getJointTourTimeFactor(); + else if(t.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) + timeFactor = mcDmuObject.getWorkTimeFactor(); + else + timeFactor = mcDmuObject.getNonWorkTimeFactor(); + + mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); + mcDmuObject.setCostCoeff(costCoeff); + + int tourMode = t.getTourModeChoice(); + int origMgra = s.getOrig(); + + mcDmuObject.getDmuIndexValues().setHHIndex(hh.getHhId()); + mcDmuObject.getDmuIndexValues().setZoneIndex(hh.getHhMgra()); + mcDmuObject.getDmuIndexValues().setOriginZone(origMgra); + mcDmuObject.getDmuIndexValues().setDebug(hh.getDebugChoiceModels()); + + mcDmuObject.setOutboundStops(t.getOutboundStops() == null ? 0 + : t.getOutboundStops().length - 1); + mcDmuObject.setInboundStops(t.getInboundStops() == null ? 0 + : t.getInboundStops().length - 1); + + mcDmuObject.setTripOrigIsTourDest(s.isInboundStop() && s.getStopId() == 0 ? 1 : 0); + mcDmuObject.setTripDestIsTourDest(!s.isInboundStop() + && ((s.getStopId() + 1) == (t.getNumOutboundStops() - 1)) ? 1 : 0); + + mcDmuObject.setInbound(s.isInboundStop()); + + mcDmuObject.setFirstTrip(0); + mcDmuObject.setLastTrip(0); + if (s.isInboundStop()) + { + mcDmuObject.setOutboundHalfTourDirection(0); + // compare stopId (0-based, so add 1) with number of stops (stops + // array length - 1); if last stop, set flag to 1, otherwise 0. + mcDmuObject.setLastTrip(((s.getStopId() + 1) == (t.getNumInboundStops() - 1)) ? 1 : 0); + } else + { + mcDmuObject.setOutboundHalfTourDirection(1); + // if first stopId (0-based), set flag to 1, otherwise 0. + mcDmuObject.setFirstTrip(s.getStopId() == 0 ? 1 : 0); + } + + mcDmuObject.setJointTour(t.getTourCategory().equalsIgnoreCase( + ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 : 0); + mcDmuObject + .setEscortTour(t.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 + : 0); + + mcDmuObject.setIncomeInDollars(hh.getIncomeInDollars()); + mcDmuObject.setAdults(hh.getNumPersons18plus()); + mcDmuObject.setAutos(hh.getAutosOwned()); + mcDmuObject.setAge(p.getAge()); + mcDmuObject.setHhSize(hh.getHhSize()); + mcDmuObject.setPersonIsFemale(p.getPersonIsFemale()); + + mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(tourMode) ? 1 : 0); + mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(tourMode) ? 1 : 0); + + mcDmuObject + .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); + + mcDmuObject.setDepartPeriod(t.getTourDepartPeriod()); + mcDmuObject.setArrivePeriod(t.getTourArrivePeriod()); + mcDmuObject.setTripPeriod(s.getStopPeriod()); + + double reimbursePct = mcDmuObject.getPersonObject().getParkingReimbursement(); + mcDmuObject.setReimburseProportion( reimbursePct ); + + + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(origMgra); + float waitTimeSingleTNC=0; + float waitTimeSharedTNC=0; + float waitTimeTaxi=0; + + Random hhRandom = hh.getHhRandom(); + double rnum = hhRandom.nextDouble(); + waitTimeSingleTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + waitTimeSharedTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + waitTimeTaxi = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + mcDmuObject.setWaitTimeSingleTNC(waitTimeSingleTNC); + mcDmuObject.setWaitTimeSharedTNC(waitTimeSharedTNC); + mcDmuObject.setWaitTimeTaxi(waitTimeTaxi); + + + } + + /** + * determine if each indexed mgra has transit access to the best tap pairs + * for the tour create an array with 1 if the mgra indexed has at least one + * TAP within walk egress distance of the mgra or zero if no walk TAPS exist + * for the mgra. + */ + private int setSoaAvailabilityForTransitTour(Stop s, Tour t, boolean debug) + { + + int availableCount = 0; + double[][] bestTaps = null; + + if (s.isInboundStop()) + { + + if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) + bestTaps = t.getBestWtwTapPairsIn(); + else + bestTaps = t.getBestWtdTapPairsIn(); + + // loop through mgras and determine if they are available as a stop + // location + ArrayList mgras = mgraManager.getMgras(); + for (int alt : mgras) + { + // if alternative mgra is unavailable because it has no size, no + // need to check its accessibility + // if ( ! soaAvailability[alt] ) + // continue; + + boolean accessible = false; + int i=-1; + for (double[] tapPair : bestTaps) + { + if (tapPair == null) continue; + + ++i; + if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) { + // if alternative location mgra is accessible by walk to any of the best inbound boarding taps, AND it's accessible by walk to the stop origin, it's available. + if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) + && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) + && earlierTripWasLocatedInAlightingTapShed == false ) { + accessible = true; + sampleMgraInBoardingTapShed[alt] = true; + } + // if alternative location mgra is accessible by walk to any of the best inbound alighting taps, AND it's accessible by walk to the tour origin, it's available. + else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourOrigMgra()) ) { + accessible = true; + sampleMgraInAlightingTapShed[alt] = true; + } + } + else { + // if alternative location mgra is accessible by walk to any of the best origin taps, AND it's accessible by walk to the stop origin, it's available. + if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) + && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) + && earlierTripWasLocatedInAlightingTapShed == false ) { + accessible = true; + sampleMgraInBoardingTapShed[alt] = true; + } + // if alternative location mgra is accessible by drive to any of the best destination taps it's available. + else if ( mgraManager.getTapIsDriveAccessibleFromMgra(alt, (int)tapPair[1]) ) { + accessible = true; + sampleMgraInAlightingTapShed[alt] = true; + } + } + + if ( accessible ){ + if(debug){ + slcSoaLogger.info(""); + if(sampleMgraInBoardingTapShed[alt]==true){ + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in boarding shed of TAP "+tapPair[0]); + }else if(sampleMgraInAlightingTapShed[alt]==true){ + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in alighting shed of TAP "+tapPair[1]); + } + if((sampleMgraInBoardingTapShed[alt]==true) && (sampleMgraInAlightingTapShed[alt]==true)) //should not happen + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in both boarding shed of TAP "+tapPair[0]+" and alighting shed of TAP "+tapPair[1]); + } + + break; + } + } + + if (accessible) + { + availableCount++; + } else + { + soaSample[alt] = 0; + soaAvailability[alt] = false; + } + + } + + } else + { + if (modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice())) + bestTaps = t.getBestWtwTapPairsOut(); + else + bestTaps = t.getBestDtwTapPairsOut(); + + // loop through mgras and determine if they have walk egress + ArrayList mgras = mgraManager.getMgras(); + for (int alt : mgras) + { + // if alternative mgra is unavailable because it has no size, no + // need to check its accessibility + // if ( ! soaAvailability[alt] ) + // continue; + + // check whether any of the outbound dtw boarding taps or best + // wtw alighting taps are in the set of walk accessible TAPs for + // the alternative mgra. + // if not, the alternative is not available. + boolean accessible = false; + int i=-1; + for (double[] tapPair : bestTaps) + { + if (tapPair == null) continue; + + ++i; + if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) { + // if alternative location mgra is accessible by walk to any of the best origin taps, AND it's accessible by walk to the stop origin, it's available. + if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) + && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) + && earlierTripWasLocatedInAlightingTapShed == false ) { + accessible = true; + sampleMgraInBoardingTapShed[alt] = true; + } + // if alternative location mgra is accessible by walk to any of the best destination taps, AND it's accessible by walk to the tour primary destination, it's available. + else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourDestMgra()) ) { + accessible = true; + sampleMgraInAlightingTapShed[alt] = true; + } + } + else { + // if alternative location mgra is accessible by drive to any of the best origin taps, it's available. + if ( mgraManager.getTapIsDriveAccessibleFromMgra(alt, (int)tapPair[0]) + && earlierTripWasLocatedInAlightingTapShed == false ) { + accessible = true; + sampleMgraInBoardingTapShed[alt] = true; + } + // if alternative location mgra is accessible by walk to any of the best destination taps, AND it's accessible by walk to the tour primary destination, it's available. + else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourDestMgra()) ) { + accessible = true; + sampleMgraInAlightingTapShed[alt] = true; + } + } + + if ( accessible ){ + if(debug){ + slcSoaLogger.info(""); + if(sampleMgraInBoardingTapShed[alt]==true){ + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in boarding shed of TAP "+tapPair[0]); + }else if(sampleMgraInAlightingTapShed[alt]==true){ + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in alighting shed of TAP "+tapPair[1]); + } + if((sampleMgraInBoardingTapShed[alt]==true) && (sampleMgraInAlightingTapShed[alt]==true)) //should not happen + slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in both boarding shed of TAP "+tapPair[0]+" and alighting shed of TAP "+tapPair[1]); + } + + break; + } + + } + if (accessible) + { + availableCount++; + } else + { + soaSample[alt] = 0; + soaAvailability[alt] = false; + } + + } + + } + + return availableCount; + } + + /** + * create an array with 1 if the mgra indexed has at least one TAP within + * walk egress distance of the mgra or zero if no walk TAPS exist for the + * mgra. private void createWalkTransitAvailableArray() { + * + * ArrayList mgras = mgraManager.getMgras(); int maxMgra = + * mgraManager.getMaxMgra(); + * + * walkTransitAvailable = new int[maxMgra+1]; + * + * // loop through mgras and determine if they have walk egress for (int alt + * : mgras) { + * + * // get the TAP set within walk egress distance of the stop location + * alternative. int[] aMgraSet = + * mgraManager.getMgraWlkTapsDistArray()[alt][0]; + * + * // set to 1 if the list of TAPS with walk accessible egress to alt is not + * empty; 0 otherwise if ( aMgraSet != null && aMgraSet.length > 0 ) + * walkTransitAvailable[alt] = 1; + * + * } + * + * stopLocDmuObj.setWalkTransitAvailable( walkTransitAvailable ); } + */ + + /** + * Do a monte carlo selection from the array of stored mode choice + * cumulative probabilities (0 based array). The probabilities were saved at + * the time the stop location alternative segment mode choice logsums were + * calculated. If the stop is not the last stop for the half-tour, the IK + * segment probabilities are passed in. If the stop is the last stop, the KJ + * probabilities are passeed in. + * + * @param household + * object frim which to get the Random object. + * @param props + * is the array of stored mode choice probabilities - 0s based + * array. + * + * @return the selected mode choice alternative from [1,...,numMcAlts]. + */ + private int selectModeFromProbabilities(Stop s, double[] cumProbs) + { + + Household household = s.getTour().getPersonObject().getHouseholdObject(); + + int selectedModeAlt = -1; + double rn = household.getHhRandom().nextDouble(); + int randomCount = household.getHhRandomCount(); + + int numAvailAlts = 0; + double sumProb = 0.0; + for (int i = 0; i < cumProbs.length; i++) + { + double tempProb = cumProbs[i] - sumProb; + sumProb += tempProb; + + if (tempProb > 0) numAvailAlts++; + + if (rn < cumProbs[i]) + { + selectedModeAlt = i + 1; + break; + } + } + + if (household.getDebugChoiceModels() || selectedModeAlt < 0 + || numAvailAlts >= availAltsToLog) + { + + // set the number of available alts to log value to a large number + // so no more get logged. + if (numAvailAlts >= availAltsToLog) + { + Person person = s.getTour().getPersonObject(); + Tour tour = s.getTour(); + smcLogger + .info("Monte Carlo selection for determining Mode Choice from Probabilities for stop with more than " + + availAltsToLog + " mode alts available."); + smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=" + + (s.getStopId() + 1) + + " of " + + (s.isInboundStop() ? tour.getNumInboundStops() - 1 : tour + .getNumOutboundStops() - 1) + " stops, inbound=" + + s.isInboundStop() + ", stopPurpose=" + s.getDestPurpose() + + ", stopDepart=" + s.getStopPeriod() + ", stopOrig=" + s.getOrig() + + ", stopDest=" + s.getDest()); + availAltsToLog = 9999; + } + + smcLogger.info(""); + smcLogger.info(""); + String separator = ""; + for (int k = 0; k < 60; k++) + separator += "+"; + smcLogger.info(separator); + + smcLogger + .info("Alternative Availability Utility Probability CumProb"); + smcLogger + .info("--------------------- ------------ ----------- -------------- --------------"); + + sumProb = 0.0; + for (int j = 0; j < cumProbs.length; j++) + { + String altString = String.format("%-3d %-25s", j + 1, ""); + double tempProb = cumProbs[j] - sumProb; + smcLogger.info(String.format("%-30s%15s%18s%18.6e%18.6e", altString, "", "", + tempProb, cumProbs[j])); + sumProb += tempProb; + } + + if (selectedModeAlt < 0) + { + smcLogger.info(" "); + String altString = String.format("%-3d %-25s", selectedModeAlt, + "no MC alt available"); + smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + throw new RuntimeException(); + } else + { + smcLogger.info(" "); + String altString = String.format("%-3d %-25s", selectedModeAlt, ""); + smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + } + + smcLogger.info(separator); + smcLogger.info(""); + smcLogger.info(""); + + } + + // if this statement is reached, there's a problem with the cumulative + // probabilities array, so return -1. + return selectedModeAlt; + } + + /** + * This method is taken from the setupStopLocationChoiceAlternativeArrays(), + * except that the stop location choice dmu attributes are not set and the + * logsum calculation setup is done only for the selected stop location + * alternative. + * + * @param stop + * object representing the half-tour. + * + * @return the selected mode choice alternative from [1,...,numMcAlts]. + */ + private int getHalfTourModeChoice(Stop s) + { + + Household hh = s.getTour().getPersonObject().getHouseholdObject(); + + // create arrays for ik and kj mode choice logsums for the stop origin, + // the sample stop location, and the half-tour final destination. + setupLogsumCalculation(s); + + int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; + ChoiceModelApplication mcModel = mcModelArray[category]; + + int altMgra = s.getDest(); + mcDmuObject.getDmuIndexValues().setDestZone(altMgra); + + // set the mode choice attributes for the sample location + mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(altMgra)); + mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(altMgra)); + mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(altMgra)); + + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(altMgra))); + + mcDmuObject.setAutoModeRequiredForTripSegment(false); + mcDmuObject.setWalkModeAllowedForTripSegment(false); + + if (hh.getDebugChoiceModels()) + { + smcLogger.info("LOGSUM calculation for determining Mode Choice Probabilities for " + + (s.isInboundStop() ? "INBOUND" : "OUTBOUND") + " half-tour with no stops."); + + hh.logHouseholdObject( + "Half Tour Mode Choice: HH_" + hh.getHhId() + ", Pers_" + + s.getTour().getPersonObject().getPersonNum() + ", Tour Purpose_" + + s.getTour().getTourPurpose() + ", Tour_" + s.getTour().getTourId() + + ", Tour Purpose_" + s.getTour().getTourPurpose() + ", Stop_" + + (s.getStopId() + 1), smcLogger); + hh.logPersonObject("Half Tour Mode Choice for person " + + s.getTour().getPersonObject().getPersonNum(), smcLogger, s.getTour().getPersonObject()); + hh.logTourObject("Half Tour Mode Choice for tour " + s.getTour().getTourId(), + smcLogger, s.getTour().getPersonObject(), s.getTour()); + hh.logStopObject("Half Tour Mode Choice for stop " + (s.getStopId() + 1), + smcLogger, s, modelStructure); + } + + if (modelStructure.getTourModeIsDriveTransit(s.getTour().getTourModeChoice())) + { + + logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); + + if (s.isInboundStop()) logsumHelper.setWtdTripMcDmuAttributesForBestTapPairs( + mcDmuObject, s.getOrig(), altMgra, s.getStopPeriod(), s.getTour() + .getBestWtdTapPairsIn(), s.getTour().getPersonObject() + .getHouseholdObject().getDebugChoiceModels()); + else logsumHelper.setDtwTripMcDmuAttributesForBestTapPairs(mcDmuObject, s.getOrig(), + altMgra, s.getStopPeriod(), s.getTour().getBestDtwTapPairsOut(), s.getTour() + .getPersonObject().getHouseholdObject().getDebugChoiceModels()); + + } else + { + + logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); + + if (s.isInboundStop()) logsumHelper.setWtwTripMcDmuAttributesForBestTapPairs( + mcDmuObject, s.getOrig(), altMgra, s.getStopPeriod(), s.getTour() + .getBestWtwTapPairsIn(), s.getTour().getPersonObject() + .getHouseholdObject().getDebugChoiceModels()); + else logsumHelper.setWtwTripMcDmuAttributesForBestTapPairs(mcDmuObject, s.getOrig(), + altMgra, s.getStopPeriod(), s.getTour().getBestWtwTapPairsOut(), s.getTour() + .getPersonObject().getHouseholdObject().getDebugChoiceModels()); + + } + double logsum = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, s.getStopPeriod(), + mcModel, mcDmuObject, smcLogger); + + s.setModeLogsum((float) logsum); + + double rn = hh.getHhRandom().nextDouble(); + int randomCount = hh.getHhRandomCount(); + + int selectedModeAlt = -1; + if (mcModel.getAvailabilityCount() > 0) + { + selectedModeAlt = mcModel.getChoiceResult(rn); + } + + if (hh.getDebugChoiceModels() || selectedModeAlt < 0 + || mcModel.getAvailabilityCount() >= availAltsToLog) + { + + // set the number of available alts to log value to a large number + // so no more get logged. + if (selectedModeAlt < 0 || mcModel.getAvailabilityCount() >= availAltsToLog) + { + Person person = s.getTour().getPersonObject(); + Tour tour = s.getTour(); + if (mcModel.getAvailabilityCount() >= availAltsToLog) + { + availAltsToLog = 9999; + smcLogger + .info("Logsum calculation for determining Mode Choice for half-tour more than " + + availAltsToLog + " mode alts available."); + } else + { + smcLogger + .info("Logsum calculation for determining Mode Choice for half-tour with no stops."); + } + smcLogger.info("HHID=" + hh.getHhId() + ", persNum=" + person.getPersonNum() + + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" + + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); + smcLogger.info("StopID=" + + (s.getStopId() + 1) + + " of " + + (s.isInboundStop() ? tour.getNumInboundStops() - 1 : tour + .getNumOutboundStops() - 1) + " stops, inbound=" + + s.isInboundStop() + ", stopPurpose=" + s.getDestPurpose() + + ", stopDepart=" + s.getStopPeriod() + ", stopOrig=" + s.getOrig() + + ", stopDest=" + s.getDest()); + } + + // altNames, utilities and probabilities are 0 based. + String[] altNames = mcModel.getAlternativeNames(); + double[] utilities = mcModel.getUtilities(); + double[] probabilities = mcModel.getProbabilities(); + + // availabilities is 1 based. + boolean[] availabilities = mcModel.getAvailabilities(); + + smcLogger.info(""); + smcLogger.info(""); + String separator = ""; + for (int k = 0; k < 60; k++) + separator += "+"; + smcLogger.info(separator); + + smcLogger + .info("Alternative Availability Utility Probability CumProb"); + smcLogger + .info("--------------------- ------------ ----------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < utilities.length; j++) + { + cumProb += probabilities[j]; + String altString = String.format("%-3d %-25s", j + 1, altNames[j]); + smcLogger.info(String.format("%-30s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + if (selectedModeAlt < 0) + { + smcLogger.info(" "); + String altString = String.format("%-3d %-25s", selectedModeAlt, + "no MC alt available"); + smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + } else + { + smcLogger.info(" "); + String altString = String.format("%-3d %-25s", selectedModeAlt, + altNames[selectedModeAlt - 1]); + smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + } + + smcLogger.info(separator); + smcLogger.info(""); + smcLogger.info(""); + + if (logsum < -900 || selectedModeAlt < 0) + { + if (logsum < -900 || selectedModeAlt < 0) smcLogger + .error("ERROR calculating trip mode choice logsum for " + + (s.isInboundStop() ? "inbound" : "outbound") + + " half-tour with no stops - ikLogsum = " + logsum + "."); + else smcLogger.error("No half-tour mode choice alternatives available " + + (s.isInboundStop() ? "inbound" : "outbound") + + " half-tour with no stops - ikLogsum = " + logsum + "."); + smcLogger + .error("setting debug to true and recomputing half-tour logsum in order to log utility expression results."); + + if (s.isInboundStop()) smcLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumIBStops=%d, StopOrig=%d, AltStopLoc=%d", + s.getTour().getPersonObject().getHouseholdObject() + .getHhId(), s.getTour().getPersonObject() + .getPersonNum(), s.getTour().getPersonObject() + .getPersonType(), s.getTour().getTourPurpose(), s + .getTour().getTourModeChoice(), s.getTour() + .getTourId(), s.getDestPurpose(), "inbound", (s + .getStopId() + 1), + s.getTour().getNumInboundStops() - 1, s.getOrig(), altMgra)); + else smcLogger + .error(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumOBStops=%d, StopOrig=%d, AltStopLoc=%d", + s.getTour().getPersonObject().getHouseholdObject() + .getHhId(), s.getTour().getPersonObject() + .getPersonNum(), s.getTour().getPersonObject() + .getPersonType(), s.getTour().getTourPurpose(), s + .getTour().getTourModeChoice(), s.getTour() + .getTourId(), s.getDestPurpose(), "outbound", (s + .getStopId() + 1), s.getTour() + .getNumOutboundStops() - 1, s.getOrig(), altMgra)); + + mcDmuObject.getDmuIndexValues().setDebug(true); + mcDmuObject.getDmuIndexValues().setHHIndex( + s.getTour().getPersonObject().getHouseholdObject().getHhId()); + logsum = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, + s.getStopPeriod(), mcModel, mcDmuObject, smcLogger); + mcDmuObject.getDmuIndexValues().setDebug(false); + + // throw new RuntimeException(); + + } + + } + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = mcModel.getUEC(); + + double vot = 0.0; + + if(modelStructure.getTripModeIsS2(selectedModeAlt)){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = uec.getValueForIndex(votIndex); + }else if (modelStructure.getTripModeIsS3(selectedModeAlt)){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = uec.getValueForIndex(votIndex); + } + s.setValueOfTime(vot); + + return selectedModeAlt; + } + + private void setOutboundTripDepartTimes(Stop[] stops) + { + + // these stops are in outbound direction + int halfTourDirection = 0; + + for (int i = 0; i < stops.length; i++) + { + + // if tour depart and arrive periods are the same, set same values + // for the stops + Stop stop = stops[i]; + Tour tour = stop.getTour(); + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + if (tour.getTourArrivePeriod() == tour.getTourDepartPeriod()) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Depart Time Model Not Run Since Tour Depart and Arrive Periods are Equal; Stop Depart Period set to Tour Depart Period = " + + tour.getTourDepartPeriod() + " for outbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourModeChoice(), + tour.getTourCategory(), tour.getTourPrimaryPurpose(), + tour.getTourId(), stop.getOrigPurpose(), + stop.getDestPurpose(), (stop.getStopId() + 1), + stops.length)); + tripDepartLogger.info(String.format("tourDepartPeriod=%d, tourArrivePeriod=%d", + tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourDepartPeriod()); + + } else + { + + int tripIndex = i + 1; + + if (tripIndex == 1) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Depart Time Model Not Run Since Trip is first trip in sequence, departing from " + + stop.getOrigPurpose() + + "; Stop Depart Period set to Tour Depart Period = " + + tour.getTourDepartPeriod() + " for outbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), tour.getTourId(), + stop.getOrigPurpose(), stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger.info(String.format( + "tourDepartPeriod=%d, tourArrivePeriod=%d", + tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourDepartPeriod()); + + } else + { + + int prevTripPeriod = stops[i - 1].getStopPeriod(); + + if (prevTripPeriod == tour.getTourArrivePeriod()) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Depart Time Model Not Run Since Previous Trip Depart and Tour Arrive Periods are Equal; Stop Depart Period set to Tour Arrive Period = " + + tour.getTourArrivePeriod() + + " for outbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), tour.getTourId(), + stop.getOrigPurpose(), stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger.info(String.format( + "prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", + prevTripPeriod, tour.getTourDepartPeriod(), + tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourDepartPeriod()); + + } else + { + + int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); + + double[] proportions = stopTodModel.getStopTodIntervalProportions( + tourPrimaryPurposeIndex, halfTourDirection, prevTripPeriod, + tripIndex); + + // for inbound trips, the first trip cannot arrive + // earlier than the last outbound trip departs + // if such a case is chosen, re-select. + int invalidCount = 0; + boolean validTripDepartPeriodSet = false; + while (validTripDepartPeriodSet == false) + { + + double rn = household.getHhRandom().nextDouble(); + int choice = getMonteCarloSelection(proportions, rn); + + // check that this stop depart time departs at same + // time or later than the stop object preceding this + // one in the stop sequence. + if (choice >= prevTripPeriod && choice <= tour.getTourArrivePeriod()) + { + validTripDepartPeriodSet = true; + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Depart Time Model for outbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), + person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), + tour.getTourId(), + stop.getOrigPurpose(), + stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger + .info(String + .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", + prevTripPeriod, + tour.getTourDepartPeriod(), + tour.getTourArrivePeriod())); + tripDepartLogger.info("tourPrimaryPurposeIndex=" + + tourPrimaryPurposeIndex + ", halfTourDirection=" + + halfTourDirection + ", tripIndex=" + tripIndex); + tripDepartLogger.info(""); + + tripDepartLogger.info(loggerSeparator); + tripDepartLogger.info(String.format("%-4s %-8s %10s %10s", + "alt", "time", "prob", "cumProb")); + double cumProb = 0.0; + for (int p = 1; p < proportions.length; p++) + { + int hr = 4 + (p / 2); + int min = (p % 2) * 30; + cumProb += proportions[p]; + String timeString = ((hr < 10) ? ("0" + hr) + : ("" + hr + ":")) + ((min == 30) ? min : "00"); + tripDepartLogger.info(String.format( + "%-4d %-8s %10.8f %10.8f", p, timeString, + proportions[p], cumProb)); + } + tripDepartLogger.info(loggerSeparator); + tripDepartLogger.info("rn=" + rn + ", choice=" + choice + + ", try=" + invalidCount); + tripDepartLogger.info(""); + } + stop.setStopPeriod(choice); + + } else + { + invalidCount++; + } + + if (invalidCount > MAX_INVALID_FIRST_ARRIVAL_COUNT) + { + tripDepartLogger.warn("Problem in Outbound Trip Depart Time Model."); + tripDepartLogger + .warn("Outbound trip depart time less than previous trip depart time for " + + invalidCount + " times."); + tripDepartLogger.warn("Possible infinite loop?"); + tripDepartLogger + .warn(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), + tour.getTourId(), stop.getOrigPurpose(), + stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger + .warn(String + .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d, last choice=%d", + prevTripPeriod, tour.getTourDepartPeriod(), + tour.getTourArrivePeriod(), choice)); + tripDepartLogger.warn("=" + invalidCount + " times."); + + //throw new RuntimeException(); + //instead of throwing an exception, set the stop period to the same period as the last stop + stop.setStopPeriod(prevTripPeriod); + } + + } + + } + + } + + } + + } + + } + + private void setInboundTripDepartTimes(Stop[] stops, int lastOutboundTripDeparts) + { + + // these stops are in inbound direction + int halfTourDirection = 1; + + for (int i = stops.length - 1; i >= 0; i--) + { + + // if tour depart and arrive periods are the same, set same values + // for the stops + Stop stop = stops[i]; + Tour tour = stop.getTour(); + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + if (tour.getTourArrivePeriod() == tour.getTourDepartPeriod()) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Arrive Time Model Not Run Since Tour Depart and Arrive Periods are Equal; Stop Arrive Period set to Tour Arrive Period = " + + tour.getTourDepartPeriod() + " for inbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourModeChoice(), + tour.getTourCategory(), tour.getTourPrimaryPurpose(), + tour.getTourId(), stop.getOrigPurpose(), + stop.getDestPurpose(), (stop.getStopId() + 1), + stops.length)); + tripDepartLogger.info(String.format("tourDepartPeriod=%d, tourArrivePeriod=%d", + tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourArrivePeriod()); + + } else + { + + int tripIndex = stops.length - i; + + if (tripIndex == 1) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Arrive Time Model Not Run Since Trip is last trip in sequence, arriving at " + + stop.getDestPurpose() + + "; Stop Arrive Period set to Tour Arrive Period = " + + tour.getTourArrivePeriod() + " for inbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), tour.getTourId(), + stop.getOrigPurpose(), stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger.info(String.format( + "tourDepartPeriod=%d, tourArrivePeriod=%d", + tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourArrivePeriod()); + + } else + { + + int prevTripPeriod = stops[i + 1].getStopPeriod(); + + if (prevTripPeriod == tour.getTourArrivePeriod()) + { + + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Arrive Time Model Not Run Since Previous Trip Arrive and Tour Arrive Periods are Equal; Stop Arrive Period set to Tour Arrive Period = " + + tour.getTourArrivePeriod() + + " for intbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), tour.getTourId(), + stop.getOrigPurpose(), stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger.info(String.format( + "prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", + prevTripPeriod, tour.getTourDepartPeriod(), + tour.getTourArrivePeriod())); + tripDepartLogger.info(""); + } + stop.setStopPeriod(tour.getTourArrivePeriod()); + + } else + { + + int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); + + double[] proportions = stopTodModel.getStopTodIntervalProportions( + tourPrimaryPurposeIndex, halfTourDirection, prevTripPeriod, + tripIndex); + + // for inbound trips, the first trip cannot arrive + // earlier than the last outbound trip departs + // if such a case is chosen, re-select. + int invalidCount = 0; + boolean validTripArrivePeriodSet = false; + while (validTripArrivePeriodSet == false) + { + + double rn = household.getHhRandom().nextDouble(); + int choice = getMonteCarloSelection(proportions, rn); + + // check that this stop arrival time arrives at same + // time or earlier than the stop object following + // this one in the stop sequence. + // also check that this stop arrival is after the + // depart time for the last outbound stop. + if (choice <= prevTripPeriod && choice >= lastOutboundTripDeparts) + { + validTripArrivePeriodSet = true; + if (household.getDebugChoiceModels()) + { + tripDepartLogger + .info("Trip Arrive Time Model for inbound half-tour."); + tripDepartLogger + .info(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), + person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), + tour.getTourId(), + stop.getOrigPurpose(), + stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger.info("tourPrimaryPurposeIndex=" + + tourPrimaryPurposeIndex + ", halfTourDirection=" + + halfTourDirection + ", tripIndex=" + tripIndex + + ", prevTripPeriod=" + prevTripPeriod); + tripDepartLogger + .info(String + .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", + prevTripPeriod, + tour.getTourDepartPeriod(), + tour.getTourArrivePeriod())); + tripDepartLogger.info(loggerSeparator); + tripDepartLogger.info(""); + + tripDepartLogger.info(String.format("%-4s %-8s %10s %10s", + "alt", "time", "prob", "cumProb")); + double cumProb = 0.0; + for (int p = 1; p < proportions.length; p++) + { + int hr = 4 + (p / 2); + int min = (p % 2) * 30; + cumProb += proportions[p]; + String timeString = ((hr < 10) ? ("0" + hr) + : ("" + hr + ":")) + ((min == 30) ? min : "00"); + tripDepartLogger.info(String.format( + "%-4d %-8s %10.8f %10.8f", p, timeString, + proportions[p], cumProb)); + } + tripDepartLogger.info(loggerSeparator); + tripDepartLogger.info("rn=" + rn + ", choice=" + choice + + ", try=" + invalidCount); + tripDepartLogger.info(""); + } + stop.setStopPeriod(choice); + } else + { + invalidCount++; + } + + if (invalidCount > MAX_INVALID_FIRST_ARRIVAL_COUNT) + { + tripDepartLogger.warn("Problem in Inbound Trip Arrival Time Model."); + tripDepartLogger + .warn("Inbound trip arrive time greater than tour arrive time chosen for " + + invalidCount + " times."); + tripDepartLogger.warn("Possible infinite loop?"); + tripDepartLogger + .warn(String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), + tour.getTourModeChoice(), + tour.getTourCategory(), + tour.getTourPrimaryPurpose(), + tour.getTourId(), stop.getOrigPurpose(), + stop.getDestPurpose(), + (stop.getStopId() + 1), stops.length)); + tripDepartLogger + .warn(String + .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d, last choice=%d", + prevTripPeriod, tour.getTourDepartPeriod(), + tour.getTourArrivePeriod(), choice)); + tripDepartLogger.warn("=" + invalidCount + " times."); + + //throw new RuntimeException(); + //instead of throwing an exception, set the stop period to the same period as the previous stop + stop.setStopPeriod(lastOutboundTripDeparts); + } + + } + + } + + } + + } + + } + + } + + /** + * + * @param probabilities + * has 1s based indexing + * @param randomNumber + * @return + */ + private int getMonteCarloSelection(double[] probabilities, double randomNumber) + { + + int returnValue = 0; + double sum = probabilities[1]; + // probabilities array passded into this method is 1s based. + for (int i = 1; i < probabilities.length - 1; i++) + { + if (randomNumber <= sum) + { + returnValue = i; + break; + } else + { + sum += probabilities[i + 1]; + returnValue = i + 1; + } + } + return returnValue; + } + + private void zeroOutCpuTimes() + { + soaAutoTime = 0; + soaOtherTime = 0; + slsTime = 0; + sldTime = 0; + slcTime = 0; + todTime = 0; + smcTime = 0; + } + + public long[] getStopTimes() + { + hhTimes[0] = soaAutoTime; + hhTimes[1] = soaOtherTime; + hhTimes[2] = slsTime; + hhTimes[3] = sldTime; + hhTimes[4] = slcTime - (soaAutoTime + soaOtherTime + slsTime); + hhTimes[5] = slcTime; + hhTimes[6] = todTime; + hhTimes[7] = smcTime; + hhTimes[8] = slcTime + sldTime + todTime + smcTime; + + return hhTimes; + } + + // this method is called to determine the parking mgra location if the stop + // location is in parkarea 1 and chosen mode is sov or hov. + private int selectParkingLocation(Household household, Tour tour, Stop stop) + { + + Logger modelLogger = parkLocLogger; + + // if the trip destination mgra is not in parking area 1, it's not + // necessary to make a parking location choice + if (mgraAltLocationIndex.containsKey(stop.getDest()) == false + || mgraAltParkArea.get(stop.getDest()) != 1) return -1; + + // if person worked at home, no reason to make a parking location choice + if (tour.getPersonObject().getFreeParkingAvailableResult() == ParkingProvisionModel.FP_MODEL_NO_REIMBURSEMENT_CHOICE) + return -1; + + // if the person has free parking, set the parking location + if (tour.getPersonObject().getFreeParkingAvailableResult() == 1) return stop.getDest(); + + parkingChoiceDmuObj.setDmuIndexValues(household.getHhId(), stop.getOrig(), stop.getDest(), + household.getDebugChoiceModels()); + + parkingChoiceDmuObj.setPersonType(tour.getPersonObject().getPersonTypeNumber()); + + Stop[] stops = null; + if (stop.isInboundStop()) stops = tour.getInboundStops(); + else stops = tour.getOutboundStops(); + + // determine activity duration in number od departure time intervals + // if no stops on halftour, activity duration is tour duration + int activityIntervals = 0; + if (stops.length == 1) + { + activityIntervals = tour.getTourArrivePeriod() - tour.getTourDepartPeriod(); + } else + { + int stopId = stop.getStopId(); + if (stopId == stops.length - 1) activityIntervals = tour.getTourArrivePeriod() + - stop.getStopPeriod(); + else activityIntervals = stops[stopId + 1].getStopPeriod() - stop.getStopPeriod(); + } + + parkingChoiceDmuObj.setActivityIntervals(activityIntervals); + + parkingChoiceDmuObj.setDestPurpose(stop.getStopPurposeIndex()); + + parkingChoiceDmuObj.setReimbPct(tour.getPersonObject().getParkingReimbursement()); + + int[] sampleIndices = setupParkLocationChoiceAlternativeArrays(stop.getOrig(), + stop.getDest()); + + // if no alternatives in the sample, it's not necessary to make a + // parking location choice + if (sampleIndices == null) return -1; + + if (household.getDebugChoiceModels()) + { + household.logHouseholdObject( + "Pre Parking Location Choice for trip: HH_" + household.getHhId() + ", Pers_" + + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" + + tour.getTourPurpose() + ", Tour_" + tour.getTourId() + + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" + + stop.getStopId(), modelLogger); + household.logPersonObject("Pre Parking Location Choice for person " + + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); + household.logTourObject("Pre Parking Location Choice for tour " + tour.getTourId(), + modelLogger, tour.getPersonObject(), tour); + household.logStopObject("Pre Parking Location Choice for stop " + stop.getStopId(), + modelLogger, stop, modelStructure); + } + + Person person = tour.getPersonObject(); + + String choiceModelDescription = ""; + String separator = ""; + String loggerString = ""; + String decisionMakerLabel = ""; + + // log headers to traceLogger if the person making the destination + // choice is from a household requesting trace information + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = "Parking Location Choice Model for trip"; + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourId=%d, StopPurpose=%s, StopId=%d", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourPurpose(), tour.getTourId(), tour.getTourPurpose(), + stop.getStopId()); + + modelLogger.info(" "); + loggerString = choiceModelDescription + " for " + decisionMakerLabel + "."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + plcModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + plcModel.computeUtilities(parkingChoiceDmuObj, parkingChoiceDmuObj.getDmuIndexValues(), + altParkAvail, altParkSample); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + int chosenIndex = -1; + int parkMgra = 0; + if (plcModel.getAvailabilityCount() > 0) + { + // get the mgra number associated with the chosen alternative + chosen = plcModel.getChoiceResult(rn); + // sampleIndices is 1-based, but the values returned are 0-based, + // parkMgras is 0-based + chosenIndex = sampleIndices[chosen]; + parkMgra = parkMgras[chosenIndex]; + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels() || chosen < 0) + { + + double[] utilities = plcModel.getUtilities(); + double[] probabilities = plcModel.getProbabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("-------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + + for (int k = 1; k <= numAltsInSample; k++) + { + int index = sampleIndices[k]; + int altMgra = parkMgras[index]; + cumProb += probabilities[k - 1]; + String altString = String.format("k=%d, index=%d, altMgra=%d", k, index, altMgra); + modelLogger.info(String.format("%-35s%18.6e%18.6e%18.6e", altString, + utilities[k - 1], probabilities[k - 1], cumProb)); + } + + modelLogger.info(" "); + if (chosen < 0) + { + modelLogger.info(String.format("No Alternatives Available For Choice !!!")); + } else + { + String altString = String.format("chosen=%d, chosenIndex=%d, chosenMgra=%d", + chosen, chosenIndex, parkMgra); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + } + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + plcModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + plcModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + plcModel.logUECResults(modelLogger, loggerString); + + } + + if (chosen > 0) return parkMgra; + else + { + logger.error(String + .format("Exception caught for HHID=%d, personNum=%d, no available parking location alternatives in tourId=%d to choose from in plcModelApplication.", + household.getHhId(), person.getPersonNum(), tour.getTourId())); + return stop.getDest(); + //throw new RuntimeException(); + } + + } + + // this method is called for trips that require a park location choice -- + // trip destination in parkarea 1 and not a work trip with free onsite + // parking + // return false if no parking location alternatives are in walk distance of + // trip destination; true otherwise. + private int[] setupParkLocationChoiceAlternativeArrays(int tripOrigMgra, int tripDestMgra) + { + + // get the array of mgras within walking distance of the trip + // destination + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(tripDestMgra); + + // set the distance values for the mgras walkable to the destination + if (walkMgras != null) + { + + // get distances, in feet, and convert to miles + // get distances from destMgra since this is the direction of + // distances read from the data file + int altCount = 0; + for (int wMgra : walkMgras) + { + // if wMgra is in the set of parkarea==1 MGRAs, add to list of + // alternatives for this park location choice + if (mgraAltLocationIndex.containsKey(wMgra)) + { + + double curWalkDist = mgraManager.getMgraToMgraWalkDistTo(wMgra, tripDestMgra) / 5280.0; + + if (curWalkDist > MgraDataManager.MAX_PARKING_WALK_DISTANCE) continue; + + // the hashMap stores a 0-based index + int altIndex = mgraAltLocationIndex.get(wMgra); + //int m = wMgra - 1; + int m=wMgra; + + altSdDistances[altCount + 1] = curWalkDist; + altMgraIndices[altCount + 1] = altIndex; + + altParkingCostsM[altCount + 1] = lsWgtAvgCostM[m]; + altParkingCostsD[altCount + 1] = lsWgtAvgCostD[m]; + altParkingCostsH[altCount + 1] = lsWgtAvgCostH[m]; + altMstallsoth[altCount + 1] = mstallsoth[m]; + altMstallssam[altCount + 1] = mstallssam[m]; + altMparkcost[altCount + 1] = mparkcost[m]; + altDstallsoth[altCount + 1] = dstallsoth[m]; + altDstallssam[altCount + 1] = dstallssam[m]; + altDparkcost[altCount + 1] = dparkcost[m]; + altHstallsoth[altCount + 1] = hstallsoth[m]; + altHstallssam[altCount + 1] = hstallssam[m]; + altHparkcost[altCount + 1] = hparkcost[m]; + altNumfreehrs[altCount + 1] = numfreehrs[m]; + + altParkAvail[altCount + 1] = true; + altParkSample[altCount + 1] = 1; + + altCount++; + } + } + + if (altCount > 0) + { + + for (int i = altCount; i < MAX_PLC_SAMPLE_SIZE; i++) + { + altOsDistances[i + 1] = Double.NaN; + altSdDistances[i + 1] = Double.NaN; + altMgraIndices[i + 1] = Integer.MAX_VALUE; + + altParkingCostsM[i + 1] = Double.NaN; + altParkingCostsD[i + 1] = Double.NaN; + altParkingCostsH[i + 1] = Double.NaN; + altMstallsoth[i + 1] = Integer.MAX_VALUE; + altMstallssam[i + 1] = Integer.MAX_VALUE; + altMparkcost[i + 1] = Float.MAX_VALUE; + altDstallsoth[i + 1] = Integer.MAX_VALUE; + altDstallssam[i + 1] = Integer.MAX_VALUE; + altDparkcost[i + 1] = Float.MAX_VALUE; + altHstallsoth[i + 1] = Integer.MAX_VALUE; + altHstallssam[i + 1] = Integer.MAX_VALUE; + altHparkcost[i + 1] = Float.MAX_VALUE; + altNumfreehrs[i + 1] = Integer.MAX_VALUE; + + altParkAvail[i + 1] = false; + altParkSample[i + 1] = 0; + } + numAltsInSample = altCount; + if (numAltsInSample > maxAltsInSample) maxAltsInSample = numAltsInSample; + } + + return altMgraIndices; + + } + + return null; + + } + + public int getMaxAltsInSample() + { + return maxAltsInSample; + } + + public HashMap getSizeSegmentNameIndexMap() + { + return sizeSegmentNameIndexMap; + } + + public double[][] getSizeSegmentArray() + { + return slcSizeTerms; + } + /** + * This method calculates a cost coefficient based on the following formula: + * + * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) + * + * + * @param incomeCoeff + * @param incomeExponent + * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice + */ + public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ + + return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java new file mode 100644 index 0000000..3b00f69 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java @@ -0,0 +1,113 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * @author crf
+ * Started: Apr 14, 2009 11:09:58 AM + */ +public class InternalExternalTripChoiceDMU + implements Serializable, VariableTable +{ + + protected HashMap methodIndexMap; + + private Household hh; + private Person person; + + private double distanceToCordonsLogsum; + private double vehiclesPerHouseholdMember; + + private IndexValues iv; + + public InternalExternalTripChoiceDMU() + { + iv = new IndexValues(); + } + + public void setDmuIndexValues(int hhid, int hhtaz) + { + iv.setHHIndex(hhid); + iv.setZoneIndex(hhtaz); + iv.setDebug(hh.getDebugChoiceModels()); + } + + public IndexValues getDmuIndexValues() + { + return iv; + } + + public void setHouseholdObject(Household hhObj) + { + hh = hhObj; + } + + public void setPersonObject(Person persObj) + { + person = persObj; + } + + public void setDistanceToCordonsLogsum(double value) + { + distanceToCordonsLogsum = value; + } + + public double getDistanceToCordonsLogsum() + { + return distanceToCordonsLogsum; + } + + public void setVehiclesPerHouseholdMember(double value) + { + vehiclesPerHouseholdMember = value; + } + + public double getVehiclesPerHouseholdMember() + { + return vehiclesPerHouseholdMember; + } + + public int getHhIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + public int getAge() + { + return person.getAge(); + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java new file mode 100644 index 0000000..eb856d0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java @@ -0,0 +1,120 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; + +import org.apache.log4j.Logger; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class InternalExternalTripChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("ie"); + + private static final String IE_CONTROL_FILE_TARGET = "ie.uec.file"; + private static final String IE_DATA_SHEET_TARGET = "ie.data.page"; + private static final String IE_MODEL_SHEET_TARGET = "ie.model.page"; + + public static final int IE_MODEL_NO_ALT = 1; + public static final int IE_MODEL_YES_ALT = 2; + + private ChoiceModelApplication ieModel; + private InternalExternalTripChoiceDMU ieDmuObject; + private ModelStructure modelStructure; + + public InternalExternalTripChoiceModel(HashMap propertyMap,ModelStructure myModelStructure, + CtrampDmuFactoryIf dmuFactory) + { + + logger.info("setting up internal-external trip choice model."); + + + modelStructure = myModelStructure; + // locate the IE choice UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String ieUecFile = uecFileDirectory + propertyMap.get(IE_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, IE_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, IE_MODEL_SHEET_TARGET); + + // create the choice model DMU object. + ieDmuObject = dmuFactory.getInternalExternalTripChoiceDMU(); + + // create the choice model object + ieModel = new ChoiceModelApplication(ieUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) ieDmuObject); + + } + + public void applyModel(Household hhObject, double[] distanceToCordonsLogsums) + { + + int homeTaz = hhObject.getHhTaz(); + ieDmuObject.setDistanceToCordonsLogsum(distanceToCordonsLogsums[homeTaz]); + + ieDmuObject.setHouseholdObject(hhObject); + double vehiclesPerHouseholdMember = hhObject.getAutosOwned() + / hhObject.getHhSize(); + ieDmuObject.setVehiclesPerHouseholdMember(vehiclesPerHouseholdMember); + + Random hhRandom = hhObject.getHhRandom(); + + // person array is 1-based + Person[] person = hhObject.getPersons(); + for (int i = 1; i < person.length; i++) + { + + ieDmuObject.setPersonObject(person[i]); + ieDmuObject.setDmuIndexValues(hhObject.getHhId(), hhObject.getHhTaz()); + + double randomNumber = hhRandom.nextDouble(); + + // compute utilities and choose alternative. + float logsum = (float) ieModel.computeUtilities(ieDmuObject, ieDmuObject.getDmuIndexValues()); + person[i].setIeLogsum(logsum); + + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + if (ieModel.getAvailabilityCount() > 0) + { + chosenAlt = ieModel.getChoiceResult(randomNumber); + } else + { + String decisionMaker = String.format("HHID=%d, PersonNum=%d", hhObject.getHhId(), + person[i].getPersonNum()); + String errorMessage = String + .format("Exception caught for %s, no available internal-external trip choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + ieModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + // write choice model alternative info to log file + if (hhObject.getDebugChoiceModels()) + { + String decisionMaker = String.format("HHID=%d, PersonNum=%d", hhObject.getHhId(), + person[i].getPersonNum()); + ieModel.logAlternativesInfo("Internal-External Trip Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d with rn %.8f", + "Internal-External Trip Choice", decisionMaker, chosenAlt, randomNumber)); + ieModel.logUECResults(logger, decisionMaker); + } + + person[i].setInternalExternalTripChoiceResult(chosenAlt); + + + } + + hhObject.setIeRandomCount(hhObject.getHhRandomCount()); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java new file mode 100644 index 0000000..c886c22 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java @@ -0,0 +1,916 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.net.UnknownHostException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To + * change this template use File | Settings | File Templates. + */ +public class JointTourModels + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(JointTourModels.class); + private transient Logger tourFreq = Logger.getLogger("tourFreq"); + + // these are public because CtrampApplication creates a + // ChoiceModelAppplication in order to get the alternatives for a log report + public static final String UEC_FILE_PROPERTIES_TARGET = "jtfcp.uec.file"; + public static final String UEC_DATA_PAGE_TARGET = "jtfcp.data.page"; + public static final String UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE = "jtfcp.freq.comp.page"; + + private static final String UEC_JOINT_TOUR_PARTIC_MODEL_PAGE = "jtfcp.participate.page"; + + public static final int JOINT_TOUR_COMPOSITION_ADULTS = 1; + public static final int JOINT_TOUR_COMPOSITION_CHILDREN = 2; + public static final int JOINT_TOUR_COMPOSITION_MIXED = 3; + + public static final String[] JOINT_TOUR_COMPOSITION_NAMES = {"", "adult", "child", + "mixed" }; + + public static final int PURPOSE_1_FIELD = 2; + public static final int PURPOSE_2_FIELD = 3; + public static final int PARTY_1_FIELD = 4; + public static final int PARTY_2_FIELD = 5; + + // DMU for the UEC + private JointTourModelsDMU dmuObject; + private AccessibilitiesTable accTable; + + private ChoiceModelApplication jointTourFrequencyModel; + private ChoiceModelApplication jointTourParticipation; + private TableDataSet jointModelsAltsTable; + private HashMap purposeIndexNameMap; + + private int[] invalidCount = new int[5]; + + private String threadName = null; + + public JointTourModels(HashMap propertyMap, AccessibilitiesTable myAccTable, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) + { + + accTable = myAccTable; + + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + ": " + + Thread.currentThread().getName() + "]"; + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + setUpModels(propertyMap, modelStructure, dmuFactory); + } + + public void setUpModels(HashMap propertyMap, ModelStructure modelStructure, + CtrampDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up %s tour frequency model on %s", + ModelStructure.JOINT_NON_MANDATORY_CATEGORY, threadName)); + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + String uecFileName = propertyMap.get(UEC_FILE_PROPERTIES_TARGET); + uecFileName = uecFileDirectory + uecFileName; + + dmuObject = dmuFactory.getJointTourModelsDMU(); + + purposeIndexNameMap = new HashMap(); + purposeIndexNameMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX, + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); + purposeIndexNameMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX, + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); + purposeIndexNameMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_INDEX, + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); + purposeIndexNameMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_INDEX, + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); + purposeIndexNameMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_INDEX, + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); + + int dataSheet = Integer.parseInt(propertyMap.get(UEC_DATA_PAGE_TARGET)); + int freqCompSheet = Integer.parseInt(propertyMap.get(UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE)); + int particSheet = Integer.parseInt(propertyMap.get(UEC_JOINT_TOUR_PARTIC_MODEL_PAGE)); + + // set up the models + jointTourFrequencyModel = new ChoiceModelApplication(uecFileName, freqCompSheet, dataSheet, + propertyMap, (VariableTable) dmuObject); + jointModelsAltsTable = jointTourFrequencyModel.getUEC().getAlternativeData(); + modelStructure.setJtfAltLabels(jointTourFrequencyModel.getAlternativeNames()); + + jointTourParticipation = new ChoiceModelApplication(uecFileName, particSheet, dataSheet, + propertyMap, (VariableTable) dmuObject); + } + + public void applyModel(Household household) + { + + // this household does not make joint tours if the CDAP pattern does not + // contain "j". + if (!household.getCoordinatedDailyActivityPattern().contains("j")) return; + + household.calculateTimeWindowOverlaps(); + + try + { + + // joint tour frequency choice is not applied to a household unless + // it has: + // 2 or more persons, each with at least one out-of home activity, + // and at + // least 1 of the persons not a pre-schooler. + + Logger modelLogger = tourFreq; + if (household.getDebugChoiceModels()) + household.logHouseholdObject( + "Pre Joint Tour Frequency Choice HHID=" + household.getHhId() + " Object", + modelLogger); + + // if it's not a valid household for joint tour frequency, keep + // track of + // count for logging later, and return. + int validIndex = household.getValidHouseholdForJointTourFrequencyModel(); + if (validIndex != 1) + { + invalidCount[validIndex]++; + switch (validIndex) + { + case 2: + household.setJointTourFreqResult(-2, "-2_1 person"); + break; + case 3: + household.setJointTourFreqResult(-3, "-3_< 2 travel"); + break; + case 4: + household.setJointTourFreqResult(-4, "-4_only preschool travel"); + break; + } + return; + } + + // set the household id, origin taz, hh taz, and debugFlag=false in + // the dmu + dmuObject.setHouseholdObject(household); + + // set the accessibility values needed based on auto sufficiency + // category for the hh. + if (household.getAutoSufficiency() == 1) + { + dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov0", + household.getHhMgra())); + dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov0", + household.getHhMgra())); + dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov0", + household.getHhMgra())); + } else if (household.getAutoSufficiency() == 2) + { + dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov1", + household.getHhMgra())); + dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov1", + household.getHhMgra())); + dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov1", + household.getHhMgra())); + } else if (household.getAutoSufficiency() == 3) + { + dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov2", + household.getHhMgra())); + dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov2", + household.getHhMgra())); + dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov2", + household.getHhMgra())); + } + + IndexValues index = dmuObject.getDmuIndexValues(); + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Joint Non-Mandatory Tour Frequency Choice Model:"); + decisionMakerLabel = String.format("HH=%d, hhSize=%d.", household.getHhId(), + household.getHhSize()); + + jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + } + + float logsum = (float) jointTourFrequencyModel.computeUtilities(dmuObject, index); + household.setJtfLogsum(logsum); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosenFreqAlt = -1; + if (jointTourFrequencyModel.getAvailabilityCount() > 0) + { + chosenFreqAlt = jointTourFrequencyModel.getChoiceResult(rn); + household.setJointTourFreqResult(chosenFreqAlt, + jointTourFrequencyModel.getAlternativeNames()[chosenFreqAlt - 1]); + } else + { + logger.error(String + .format("Exception caught for HHID=%d, no available joint tour frequency alternatives to choose from in choiceModelApplication.", + household.getHhId())); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + String[] altNames = jointTourFrequencyModel.getAlternativeNames(); + + double[] utilities = jointTourFrequencyModel.getUtilities(); + double[] probabilities = jointTourFrequencyModel.getProbabilities(); + + modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " + + household.getHhSize()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %10s", k + 1, altNames[k]); + modelLogger.info(String.format("%-15s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %10s", chosenFreqAlt, + altNames[chosenFreqAlt - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + jointTourFrequencyModel.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + jointTourFrequencyModel.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosenFreqAlt); + + // write UEC calculation results to separate model specific log + // file + jointTourFrequencyModel.logUECResults(modelLogger, loggingHeader); + + } + + createJointTours(household, chosenFreqAlt); + + } catch (Exception e) + { + logger.error(String.format("error joint tour choices model for hhId=%d.", + household.getHhId())); + throw new RuntimeException(); + } + + household.setJtfRandomCount(household.getHhRandomCount()); + + } + + private void jointTourParticipation(Tour jointTour) + { + + // get the Household object for this joint tour + Household household = dmuObject.getHouseholdObject(); + + // get the array of Person objects for this hh + Person[] persons = household.getPersons(); + + // define an ArrayList to hold indices of person objects participating + // in the joint tour + ArrayList jointTourPersonList = null; + + // make sure each joint tour has a valid composition before going to the + // next one. + boolean validParty = false; + + int adults = 0; + int children = 0; + + Logger modelLogger = tourFreq; + + int loopCount = 0; + while (!validParty) + { + + dmuObject.setTourObject(jointTour); + + adults = 0; + children = 0; + + jointTourPersonList = new ArrayList(); + + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + jointTour.setPersonObject(persons[p]); + + if (household.getDebugChoiceModels() || loopCount == 1000) + { + decisionMakerLabel = String.format( + "HH=%d, hhSize=%d, PersonNum=%d, PersonType=%s, tourId=%d.", + household.getHhId(), household.getHhSize(), person.getPersonNum(), + person.getPersonType(), jointTour.getTourId()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + // if person type is inconsistent with tour composition, + // person's + // participation is by definition no, + // so skip making the choice and go to next person + switch (jointTour.getJointTourComposition()) + { + + // adults only in joint tour + case 1: + if (persons[p].getPersonIsAdult() == 1) + { + + // write debug header + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + choiceModelDescription = String + .format("Adult Party Joint Tour Participation Choice Model:"); + jointTourParticipation.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + + decisionMakerLabel + "."; + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + } + + jointTourParticipation.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + int chosen = -1; + if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for HHID=%d, person p=%d, no available adults only joint tour participation alternatives to choose from in choiceModelApplication.", + jointTour.getHhId(), p)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + String[] altNames = jointTourParticipation.getAlternativeNames(); + + double[] utilities = jointTourParticipation.getUtilities(); + double[] probabilities = jointTourParticipation.getProbabilities(); + + modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " + + household.getHhSize() + ", tourId: " + + jointTour.getTourId() + ", jointFreqChosen: " + + household.getJointTourFreqChosenAltName()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %13s", k + 1, + altNames[k]); + modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", + altString, utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %13s", chosen, + altNames[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug + // log + // file + jointTourParticipation.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + jointTourParticipation.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate + // model + // specific log file + jointTourParticipation.logUECResults(modelLogger, loggingHeader); + + if (loopCount == 1000) + { + jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + jointTourFrequencyModel.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + jointTourFrequencyModel.logUECResults(modelLogger, + loggingHeader); + } + + } + + // particpate is alternative 1, not participating is + // alternative 2. + if (chosen == 1) + { + jointTourPersonList.add(p); + adults++; + } + } + break; + + // children only in joint tour + case 2: + if (persons[p].getPersonIsAdult() == 0) + { + + // write debug header + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + choiceModelDescription = String + .format("Child Party Joint Tour Participation Choice Model:"); + jointTourParticipation.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + + decisionMakerLabel + "."; + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + } + + jointTourParticipation.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + int chosen = -1; + if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for HHID=%d, person p=%d, no available children only joint tour participation alternatives to choose from in choiceModelApplication.", + jointTour.getHhId(), p)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + String[] altNames = jointTourParticipation.getAlternativeNames(); + + double[] utilities = jointTourParticipation.getUtilities(); + double[] probabilities = jointTourParticipation.getProbabilities(); + + modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " + + household.getHhSize() + ", tourId: " + + jointTour.getTourId()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %13s", k + 1, + altNames[k]); + modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", + altString, utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %13s", chosen, + altNames[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug + // log + // file + jointTourParticipation.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + jointTourParticipation.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate + // model + // specific log file + jointTourParticipation.logUECResults(modelLogger, loggingHeader); + + if (loopCount == 1000) + { + jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + jointTourFrequencyModel.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + jointTourFrequencyModel.logUECResults(modelLogger, + loggingHeader); + } + } + + // particpate is alternative 1, not participating is + // alternative 2. + if (chosen == 1) + { + jointTourPersonList.add(p); + children++; + } + } + break; + + // mixed, adults and children in joint tour + case 3: + + // write debug header + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + choiceModelDescription = String + .format("Mixed Party Joint Tour Participation Choice Model:"); + jointTourParticipation.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + decisionMakerLabel + + "."; + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + } + + jointTourParticipation.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + int chosen = -1; + if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught for HHID=%d, person p=%d, no available mixed adult/children joint tour participation alternatives to choose from in choiceModelApplication.", + jointTour.getHhId(), p)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels() || loopCount == 1000) + { + + String[] altNames = jointTourParticipation.getAlternativeNames(); + + double[] utilities = jointTourParticipation.getUtilities(); + double[] probabilities = jointTourParticipation.getProbabilities(); + + modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " + + household.getHhSize() + ", tourId: " + jointTour.getTourId()); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %13s", k + 1, altNames[k]); + modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", + altString, utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %13s", chosen, + altNames[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log + // file + jointTourParticipation.logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + jointTourParticipation.logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model + // specific log file + jointTourParticipation.logUECResults(modelLogger, loggingHeader); + + if (loopCount == 1000) + { + jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + jointTourFrequencyModel.computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + jointTourFrequencyModel.logUECResults(modelLogger, loggingHeader); + } + } + + // particpate is alternative 1, not participating is + // alternative 2. + if (chosen == 1) + { + jointTourPersonList.add(p); + if (persons[p].getPersonIsAdult() == 1) adults++; + else children++; + } + break; + + } + + } + + // done with all persons, so see if the chosen participation is a + // valid + // composition, and if not, repeat the participation choice. + switch (jointTour.getJointTourComposition()) + { + + case 1: + if (adults > 1 && children == 0) validParty = true; + break; + + case 2: + if (adults == 0 && children > 1) validParty = true; + break; + + case 3: + if (adults > 0 && children > 0) validParty = true; + break; + + } + + if (!validParty) loopCount++; + + if (loopCount > 1000) + { + logger.warn("loop count in joint tour participation model is " + loopCount); + if (loopCount > 2000) + { + logger.warn("joint tour party composition-terminating on excessive loop count."); + //logger.error("terminating on excessive loop count."); + //throw new RuntimeException(); + } + } + + } // end while + + // create an array of person indices for participation in the tour + int[] personNums = new int[jointTourPersonList.size()]; + for (int i = 0; i < personNums.length; i++) + personNums[i] = jointTourPersonList.get(i); + jointTour.setPersonNumArray(personNums); + + if (household.getDebugChoiceModels()) + { + for (int i = 0; i < personNums.length; i++) + { + Person person = household.getPersons()[personNums[i]]; + String decisionMakerLabel = String + .format("Person in Party, HH=%d, hhSize=%d, PersonNum=%d, PersonType=%s, tourId=%d.", + household.getHhId(), household.getHhSize(), person.getPersonNum(), + person.getPersonType(), jointTour.getTourId()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + } + + } + + /** + * creates the tour objects in the Household object given the chosen joint + * tour frequency alternative. + * + * @param chosenAlt + */ + private void createJointTours(Household household, int chosenAlt) + { + + int purposeIndex1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PURPOSE_1_FIELD); + int purposeIndex2 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PURPOSE_2_FIELD); + + if (purposeIndex1 > 0 && purposeIndex2 > 0) + { + + Tour t1 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex1), + ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex1); + int party1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_1_FIELD); + t1.setJointTourComposition(party1); + + Tour t2 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex2), + ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex2); + int party2 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_2_FIELD); + t2.setJointTourComposition(party2); + + household.createJointTourArray(t1, t2); + + jointTourParticipation(t1); + jointTourParticipation(t2); + + } else + { + + Tour t1 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex1), + ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex1); + int party1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_1_FIELD); + t1.setJointTourComposition(party1); + + household.createJointTourArray(t1); + + jointTourParticipation(t1); + + } + + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + /* + * + */ + String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); + + MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, + Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TazDataManager tazManager = TazDataManager.getInstance(propertyMap); + + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + AccessibilitiesTable accTable = new AccessibilitiesTable(accFileName); + + String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); + int hhServerPort = Integer.parseInt((String) propertyMap + .get("RunModel.HouseholdServerPort")); + + HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, + hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + + householdDataManager.setPropertyFileValues(propertyMap); + householdDataManager.setHouseholdSampleRate(1.0f, 0); + householdDataManager.setDebugHhIdsFromHashmap(); + householdDataManager.setTraceHouseholdSet(); + + int id = householdDataManager.getArrayIndex(423804); + Household[] hh = householdDataManager.getHhArray(id, id); + + JointTourModels jtfModel = new JointTourModels(propertyMap, accTable, modelStructure, + dmuFactory); + jtfModel.applyModel(hh[0]); + + /* + * Use this block to instantiate a UEC for the joint freq/comp model and + * a UEC for the joint participate model. After checking the UECs are + * instantiated correctly (spelling/typos/dmu methods implemented/etc.), + * test model implementation. String uecFileDirectory = propertyMap.get( + * CtrampApplication.PROPERTIES_UEC_PATH ); + * + * ModelStructure modelStructure = new SandagModelStructure(); + * SandagCtrampDmuFactory dmuFactory = new + * SandagCtrampDmuFactory(modelStructure); + * + * JointTourModelsDMU dmuObject = dmuFactory.getJointTourModelsDMU(); + * File uecFile = new File(uecFileDirectory + propertyMap.get( + * UEC_FILE_PROPERTIES_TARGET )); UtilityExpressionCalculator uec = new + * UtilityExpressionCalculator(uecFile, 1, 0, propertyMap, + * (VariableTable)dmuObject); + * System.out.println("Jount tour freq/comp choice UEC passed."); + * + * uec = new UtilityExpressionCalculator(uecFile, 2, 0, propertyMap, + * (VariableTable)dmuObject); + * System.out.println("Joint tour participation choice UEC passed."); + */ + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java new file mode 100644 index 0000000..efadb79 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java @@ -0,0 +1,310 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class JointTourModelsDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); + + protected HashMap methodIndexMap; + + protected Household hh; + protected Tour tour; + protected IndexValues dmuIndex; + + private float shopHOVAccessibility; + private float maintHOVAccessibility; + private float discrHOVAccessibility; + + public JointTourModelsDMU() + { + dmuIndex = new IndexValues(); + } + + public Household getHouseholdObject() + { + return hh; + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public void setTourObject(Tour tourObject) + { + tour = tourObject; + } + + // DMU methods - define one of these for every @var in the mode choice + // control + // file. + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getActiveCountFullTimeWorkers() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsFullTimeWorker() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountPartTimeWorkers() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsPartTimeWorker() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountUnivStudents() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsUniversityStudent() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountNonWorkers() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsNonWorkingAdultUnder65() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountRetirees() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsNonWorkingAdultOver65() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountDrivingAgeSchoolChildren() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsStudentDriving() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountPreDrivingAgeSchoolChildren() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsStudentNonDriving() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getActiveCountPreSchoolChildren() + { + int count = 0; + for (Person p : hh.getPersons()) + if (p != null && p.getPersonIsPreschoolChild() == 1) + if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; + return count; + } + + public int getMaxPairwiseAdultOverlapsHh() + { + return hh.getMaxAdultOverlaps(); + } + + public int getMaxPairwiseChildOverlapsHh() + { + return hh.getMaxChildOverlaps(); + } + + public int getMaxPairwiseMixedOverlapsHh() + { + return hh.getMaxMixedOverlaps(); + } + + public int getMaxPairwiseOverlapOtherAdults() + { + return tour.getPersonObject().getMaxAdultOverlaps(); + } + + public int getMaxPairwiseOverlapOtherChildren() + { + return tour.getPersonObject().getMaxChildOverlaps(); + } + + public int getTravelActiveAdults() + { + return hh.getTravelActiveAdults(); + } + + public int getTravelActiveChildren() + { + return hh.getTravelActiveChildren(); + } + + public int getPersonStaysHome() + { + Person p = tour.getPersonObject(); + return p.getCdapActivity().equalsIgnoreCase("H") ? 1 : 0; + } + + public int getIncomeLessThan30K() + { + return hh.getIncomeInDollars() < 30000 ? 1 : 0; + } + + public int getIncome30Kto60K() + { + int income = hh.getIncomeInDollars(); + return (income >= 30000 && income < 60000) ? 1 : 0; + } + + public int getIncomeMoreThan100K() + { + return hh.getIncomeInDollars() >= 100000 ? 1 : 0; + } + + public int getNumAdults() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + num += (persons[i].getPersonIsAdult() == 1 ? 1 : 0); + return num; + } + + public int getNumChildren() + { + int num = 0; + Person[] persons = hh.getPersons(); + for (int i = 1; i < persons.length; i++) + num += (persons[i].getPersonIsAdult() == 0 ? 1 : 0); + return num; + } + + public int getHhWorkers() + { + return hh.getWorkers(); + } + + public int getAutoOwnership() + { + return hh.getAutosOwned(); + } + + public int getTourPurposeIsMaint() + { + return tour.getTourPurpose() + .equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME) ? 1 : 0; + } + + public int getTourPurposeIsEat() + { + return tour.getTourPurpose().equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) ? 1 + : 0; + } + + public int getTourPurposeIsVisit() + { + return tour.getTourPurpose().equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) ? 1 + : 0; + } + + public int getTourPurposeIsDiscr() + { + return tour.getTourPurpose() + .equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME) ? 1 : 0; + } + + public int getPersonType() + { + return tour.getPersonObject().getPersonTypeNumber(); + } + + public int getJointTourComposition() + { + return tour.getJointTourComposition(); + } + + public int getJTours() + { + return hh.getJointTourArray().length; + } + + public void setShopHOVAccessibility(float accessibility) + { + shopHOVAccessibility = accessibility; + } + + public float getShopHOVAccessibility() + { + return shopHOVAccessibility; + } + + public void setMaintHOVAccessibility(float accessibility) + { + maintHOVAccessibility = accessibility; + } + + public float getMaintHOVAccessibility() + { + return maintHOVAccessibility; + } + + public void setDiscrHOVAccessibility(float accessibility) + { + discrHOVAccessibility = accessibility; + } + + public float getDiscrHOVAccessibility() + { + return discrHOVAccessibility; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java new file mode 100644 index 0000000..ebc8c62 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java @@ -0,0 +1,885 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.IndexSort; + +public class MandatoryDestChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(MandatoryDestChoiceModel.class); + private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); + + // this constant used as a dimension for saving distance and logsums for + // alternatives in samples + private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; + + private static final int DC_DATA_SHEET = 0; + private static final int DC_WORK_AT_HOME_SHEET = 1; + + private MgraDataManager mgraManager; + private DestChoiceSize dcSizeObj; + + private DestChoiceDMU dcDmuObject; + private DcSoaDMU dcSoaDmuObject; + + private TourModeChoiceModel mcModel; + private DestinationSampleOfAlternativesModel dcSoaModel; + + private String[] segmentNameList; + private HashMap segmentNameIndexMap; + private HashMap workOccupValueSegmentIndexMap; + + private int[] dcModelIndices; + + // A ChoiceModelApplication object and modeAltsAvailable[] is needed for + // each purpose + private ChoiceModelApplication[] locationChoiceModels; + private ChoiceModelApplication locationChoiceModel; + private ChoiceModelApplication worksAtHomeModel; + + private int[] uecSheetIndices; + private int[] soaUecSheetIndices; + + int origMgra; + + private int modelIndex; + private int shadowPricingIteration; + + private double[] sampleAlternativeDistances; + private double[] sampleAlternativeLogsums; + + private double[] mgraDistanceArray; + + private BuildAccessibilities aggAcc; + + public MandatoryDestChoiceModel(int index, HashMap propertyMap, + DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, MgraDataManager mgraManager, + String dcUecFileName, String soaUecFile, int soaSampleSize, String modeChoiceUecFile, + CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) + { + + // set the model structure and the tour purpose list + this.mgraManager = mgraManager; + this.aggAcc = aggAcc; + this.dcSizeObj = dcSizeObj; + this.mcModel = mcModel; + + modelIndex = index; + + dcDmuObject = dmuFactory.getDestChoiceDMU(); + dcDmuObject.setAggAcc(aggAcc); + + dcSoaDmuObject = dmuFactory.getDcSoaDMU(); + dcSoaDmuObject.setAggAcc(aggAcc); + + shadowPricingIteration = 0; + + sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + + workOccupValueSegmentIndexMap = aggAcc.getWorkOccupValueIndexMap(); + + } + + public void setupWorkSegments(int[] myUecSheetIndices, int[] mySoaUecSheetIndices) + { + uecSheetIndices = myUecSheetIndices; + soaUecSheetIndices = mySoaUecSheetIndices; + segmentNameList = aggAcc.getWorkSegmentNameList(); + } + + public void setupSchoolSegments() + { + aggAcc.createSchoolSegmentNameIndices(); + uecSheetIndices = aggAcc.getSchoolDcUecSheets(); + soaUecSheetIndices = aggAcc.getSchoolDcSoaUecSheets(); + segmentNameList = aggAcc.getSchoolSegmentNameList(); + } + + public void setupDestChoiceModelArrays(HashMap propertyMap, + String dcUecFileName, String soaUecFile, int soaSampleSize) + { + + segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); + + // create a sample of alternatives choice model object for use in + // selecting a sample + // of all possible destination choice alternatives. + dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFile, soaSampleSize, + propertyMap, mgraManager, dcSizeObj.getDcSizeArray(), dcSoaDmuObject, + soaUecSheetIndices); + + // create the works-at-home ChoiceModelApplication object + worksAtHomeModel = new ChoiceModelApplication(dcUecFileName, DC_WORK_AT_HOME_SHEET, + DC_DATA_SHEET, propertyMap, (VariableTable) dcDmuObject); + + dcSoaModel.setAvailabilityForSampleOfAlternatives(dcSizeObj.getDcSizeArray()); + + // create a lookup array to map purpose index to model index + dcModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int dcModelIndex = 0; + int dcSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, dcModelIndex); + dcModelIndices[dcSegmentIndex] = dcModelIndex++; + } else + { + dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); + } + + dcSegmentIndex++; + } + // the value of dcModelIndex is the number of ChoiceModelApplication + // objects to create + // the modelIndexMap keys are the uec sheets to use in building + // ChoiceModelApplication objects + + locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; + + int i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + + int modelIndex = -1; + try + { + modelIndex = modelIndexMap.get(uecIndex); + locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, + uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcDmuObject); + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", + i, modelIndex, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + throw new RuntimeException(); + } + + } + + mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; + + } + + public void applyWorkLocationChoice(Household hh) + { + + if (hh.getDebugChoiceModels()) + { + String label = String.format("Pre Work Location Choice HHId=%d Object", hh.getHhId()); + hh.logHouseholdObject(label, dcManLogger); + } + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + int occupSegmentIndex = -1; + int occup = -1; + String occupSegmentName = ""; + + Person[] persons = hh.getPersons(); + + int tourNum = 0; + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + // skip person if they are not a worker + if (p.getPersonIsWorker() != 1) + { + p.setWorkLocationSegmentIndex(-1); + p.setWorkLocation(0); + p.setWorkLocDistance(0); + p.setWorkLocLogsum(-999); + continue; + } + + // skip person if their work at home choice was work in the home + // (alternative 2 in choice model) + int worksAtHomeChoice = selectWorksAtHomeChoice(dcDmuObject, hh, p); + if (worksAtHomeChoice == ModelStructure.WORKS_AT_HOME_ALTERNATUVE_INDEX) + { + p.setWorkLocationSegmentIndex(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); + p.setWorkLocation(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); + p.setWorkLocDistance(0); + p.setWorkLocLogsum(-999); + continue; + } + + // save person information in decision maker label, and log person + // object + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p + .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); + hh.logPersonObject(decisionMakerLabel, dcManLogger, p); + } + + double[] results = null; + try + { + + int homeMgra = hh.getHhMgra(); + origMgra = homeMgra; + + occup = p.getPersPecasOccup(); + occupSegmentIndex = workOccupValueSegmentIndexMap.get(occup); + occupSegmentName = segmentNameList[occupSegmentIndex]; + + p.setWorkLocationSegmentIndex(occupSegmentIndex); + + // update the DC dmuObject for this person + dcDmuObject.setHouseholdObject(hh); + dcDmuObject.setPersonObject(p); + dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); + + double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[occupSegmentIndex]; + mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, + mgraDistanceArray); + + // set size array for the tour segment and distance array from + // the home mgra to all destinaion mgras. + dcSoaDmuObject.setDestChoiceSize(homeMgraSizeArray); + dcSoaDmuObject.setDestDistance(mgraDistanceArray); + + dcDmuObject.setDestChoiceSize(homeMgraSizeArray); + dcDmuObject.setDestChoiceDistance(mgraDistanceArray); + + int choiceModelIndex = dcModelIndices[occupSegmentIndex]; + locationChoiceModel = locationChoiceModels[choiceModelIndex]; + + // get the work location alternative chosen from the sample + results = selectLocationFromSampleOfAlternatives("work", -1, p, occupSegmentName, + occupSegmentIndex, tourNum++, homeMgraSizeArray, mgraDistanceArray); + + } catch (RuntimeException e) + { + logger.fatal(String + .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in work location choice, occup=%d, segmentIndex=%d, segmentName=%s", + i, hh.getHhId(), i, occup, occupSegmentIndex, occupSegmentName)); + logger.fatal("Exception caught:", e); + logger.fatal("calling System.exit(-1) to terminate."); + System.exit(-1); + } + + p.setWorkLocation((int) results[0]); + p.setWorkLocDistance((float) results[1]); + p.setWorkLocLogsum((float) results[2]); + + } + + } + + public void applySchoolLocationChoice(Household hh) + { + + if (hh.getDebugChoiceModels()) + { + String label = String.format("Pre school Location Choice HHId=%d Object", hh.getHhId()); + hh.logHouseholdObject(label, dcManLogger); + } + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + + int homeMgra = hh.getHhMgra(); + Person[] persons = hh.getPersons(); + + int tourNum = 0; + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + int segmentIndex = -1; + int segmentType = -1; + if (p.getPersonIsPreschoolChild() == 1 || p.getPersonIsStudentNonDriving() == 1 + || p.getPersonIsStudentDriving() == 1 || p.getPersonIsUniversityStudent() == 1) + { + + if (p.getPersonIsPreschoolChild() == 1) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.PRESCHOOL_ALT_INDEX; + } else if (p.getPersonIsGradeSchool() == 1) + { + segmentIndex = aggAcc.getMgraGradeSchoolSegmentIndex(homeMgra); + segmentType = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; + } else if (p.getPersonIsHighSchool() == 1) + { + segmentIndex = aggAcc.getMgraHighSchoolSegmentIndex(homeMgra); + segmentType = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; + } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; + } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; + } + + // if person type is a student but segment index is -1, the + // person is not enrolled + // assume home schooled + if (segmentIndex < 0) + { + p.setSchoolLocationSegmentIndex(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); + p.setSchoolLoc(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); + p.setSchoolLocDistance(0); + p.setSchoolLocLogsum(-999); + continue; + } else + { + // if the segment is in the skip shadow pricing set, and the + // iteration is > 0, dont' compute new choice + if (shadowPricingIteration == 0 + || !dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) + p.setSchoolLocationSegmentIndex(segmentIndex); + } + + if (segmentType < 0) + { + segmentType = ModelStructure.NOT_ENROLLED_SEGMENT_INDEX; + } + } else // not a student person type + { + p.setSchoolLocationSegmentIndex(-1); + p.setSchoolLoc(0); + p.setSchoolLocDistance(0); + p.setSchoolLocLogsum(-999); + continue; + } + + // save person information in decision maker label, and log person + // object + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p + .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); + hh.logPersonObject(decisionMakerLabel, dcManLogger, p); + } + + // if the segment is in the skip shadow pricing set, and the + // iteration is > 0, dont' compute new choice + if (shadowPricingIteration > 0 && dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) + continue; + + double[] results = null; + int modelIndex = 0; + try + { + + origMgra = homeMgra; + + // update the DC dmuObject for this person + dcDmuObject.setHouseholdObject(hh); + dcDmuObject.setPersonObject(p); + dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); + + /** + * remove following - don't need non-mandatory accessibility + * since we're doing shadow pricing for school tours // set the + * auto sufficiency dependent non-mandatory accessibility value + * for the household int autoSufficiency = + * hh.getAutoSufficiency(); float accessibility = + * aggAcc.getAggregateAccessibility( + * nonMandatoryAccessibilityTypes[autoSufficiency], + * hh.getHhMgra() ); dcDmuObject.setNonMandatoryAccessibility( + * accessibility ); + */ + + double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[segmentIndex]; + mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, + mgraDistanceArray); + + // set size array for the tour segment and distance array from + // the home mgra to all destinaion mgras. + dcSoaDmuObject.setDestChoiceSize(homeMgraSizeArray); + dcSoaDmuObject.setDestDistance(mgraDistanceArray); + + dcDmuObject.setDestChoiceSize(homeMgraSizeArray); + dcDmuObject.setDestChoiceDistance(mgraDistanceArray); + + modelIndex = dcModelIndices[segmentIndex]; + locationChoiceModel = locationChoiceModels[modelIndex]; + + // get the school location alternative chosen from the sample + results = selectLocationFromSampleOfAlternatives("school", segmentType, p, + segmentNameList[segmentIndex], segmentIndex, tourNum++, homeMgraSizeArray, + mgraDistanceArray); + + } catch (RuntimeException e) + { + logger.fatal(String + .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in school location choice, modelIndex=%d, segmentIndex=%d, segmentName=%s", + i, hh.getHhId(), i, modelIndex, segmentIndex, + segmentNameList[segmentIndex])); + logger.fatal("Exception caught:", e); + logger.fatal("calling System.exit(-1) to terminate."); + System.exit(-1); + } + + p.setSchoolLoc((int) results[0]); + p.setSchoolLocDistance((float) results[1]); + p.setSchoolLocLogsum((float) results[2]); + + } + + } + + /** + * + * @return an array with chosen mgra, distance to chosen mgra, and logsum to + * chosen mgra. + */ + private double[] selectLocationFromSampleOfAlternatives(String segmentType, + int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, + int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcManLogger; + + Household household = person.getHouseholdObject(); + + // compute the sample of alternatives set for the person + dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, null, person, + segmentName, sizeSegmentIndex, household.getHhMgra()); + + // get sample of locations and correction factors for sample + int[] finalSample = dcSoaModel.getSampleOfAlternatives(); + float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); + + int numAlts = locationChoiceModel.getNumberOfAlternatives(); + + // set the destAltsAvailable array to true for all destination choice + // alternatives for each purpose + boolean[] destAltsAvailable = new boolean[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsAvailable[k] = false; + + // set the destAltsSample array to 1 for all destination choice + // alternatives + // for each purpose + int[] destAltsSample = new int[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsSample[k] = 0; + + int[] sampleValues = new int[finalSample.length]; + + dcDmuObject.setDestChoiceSize(homeMgraSizeArray); + dcDmuObject.setDestChoiceDistance(homeMgraDistanceArray); + + // for the destinations and sub-zones in the sample, compute mc logsums + // and + // save in DC dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 1; i < finalSample.length; i++) + { + + int destMgra = finalSample[i]; + sampleValues[i] = finalSample[i]; + + // get the mode choice logsum for the destination choice sample + // alternative + double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); + + sampleAlternativeLogsums[i] = logsum; + sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + dcDmuObject.setMcLogsum(destMgra, logsum); + + // set sample of alternatives correction factor used in destination + // choice utility for the sampled alternative. + dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); + + // set availaibility and sample values for the purpose, dcAlt. + destAltsAvailable[finalSample[i]] = true; + destAltsSample[finalSample[i]] = 1; + + } + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tourNum); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" + + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " + + person.getPersonType() + ", TourNum=" + tourNum); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + locationChoiceModel.computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), + destAltsAvailable, destAltsSample); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (locationChoiceModel.getAvailabilityCount() > 0) + { + chosen = locationChoiceModel.getChoiceResult(rn); + } else + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), segmentName)); + } + + // write choice model alternative info to log file + int selectedIndex = -1; + for (int j = 1; j < finalSample.length; j++) + { + if (finalSample[j] == chosen) + { + selectedIndex = j; + break; + } + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = locationChoiceModel.getUtilities(); + double[] probabilities = locationChoiceModel.getProbabilities(); + boolean[] availabilities = locationChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb Distance Logsum"); + modelLogger + .info("--------------------- -------------- -------------- -------------- -------------- -------------- --------------"); + + int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); + + int sortedSelectedIndex = 0; + double cumProb = 0.0; + for (int j = 1; j < finalSample.length; j++) + { + int k = sortedSampleValueIndices[j]; + int alt = finalSample[k]; + + if (alt == chosen) sortedSelectedIndex = j; + + cumProb += probabilities[alt - 1]; + String altString = String.format("j=%-2d, k=%-2d, mgra=%-5d", j, k, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e%18.6e%18.6e", + altString, availabilities[alt], utilities[alt - 1], probabilities[alt - 1], + cumProb, sampleAlternativeDistances[k], sampleAlternativeLogsums[k])); + } + + if (sortedSelectedIndex >= 0) + { + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", sortedSelectedIndex, chosen); + modelLogger.info(String.format( + "Choice: %s, dist=%.6e, logsum=%.6e with rn=%.8f, randomCount=%d", + altString, sampleAlternativeDistances[selectedIndex], + sampleAlternativeLogsums[selectedIndex], rn, randomCount)); + } else + { + modelLogger.info(" "); + modelLogger.info(String.format( + "j=%d, mgra=None selected, no alternatives available", selectedIndex)); + modelLogger.info(String.format("Choice: %s, rn=%.8f, randomCount=%d", "N/A", rn, + randomCount)); + } + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + + // write UEC calculation results to separate model specific log file + locationChoiceModel.logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.fatal(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), segmentName)); + logger.fatal("calling System.exit(-1) to terminate."); + System.exit(-1); + } + + } + + double[] returnArray = new double[3]; + + returnArray[0] = chosen; + returnArray[1] = sampleAlternativeDistances[selectedIndex]; + returnArray[2] = sampleAlternativeLogsums[selectedIndex]; + + return returnArray; + + } + + private int selectWorksAtHomeChoice(DestChoiceDMU dcDmuObject, Household household, + Person person) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcManLogger; + + dcDmuObject.setHouseholdObject(household); + dcDmuObject.setPersonObject(person); + dcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), origMgra, 0); + + double accessibility = aggAcc.getAccessibilitiesTableObject().getAggregateAccessibility( + "totEmp", household.getHhMgra()); + dcDmuObject.setWorkAccessibility(accessibility); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format("Usual Work Location Is At Home Choice Model"); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person + .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Usual Work Location Is At Home Choice Model: Person Num: " + + person.getPersonNum() + ", Person Type: " + person.getPersonType()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + worksAtHomeModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float logsum = (float) worksAtHomeModel.computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues()); + person.setWorksFromHomeLogsum(logsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (worksAtHomeModel.getAvailabilityCount() > 0) + { + chosen = worksAtHomeModel.getChoiceResult(rn); + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels() || chosen < 0) + { + + double[] utilities = worksAtHomeModel.getUtilities(); + double[] probabilities = worksAtHomeModel.getProbabilities(); + boolean[] availabilities = worksAtHomeModel.getAvailabilities(); + + String[] altNames = worksAtHomeModel.getAlternativeNames(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < utilities.length; j++) + { + cumProb += probabilities[j]; + String altString = String.format("%d, %s", j + 1, altNames[j]); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, alt=%s", chosen, + (chosen < 0 ? "N/A, no available alternatives" : altNames[chosen - 1])); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + worksAtHomeModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + worksAtHomeModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + + // write UEC calculation results to separate model specific log file + worksAtHomeModel.logUECResults(modelLogger, loggingHeader); + + } + + if (chosen < 0) + { + logger.fatal(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available works at home alternatives to choose from in choiceModelApplication.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum())); + logger.fatal("calling System.exit(-1) to terminate."); + System.exit(-1); + } + + return chosen; + + } + + private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, + int segmentTypeIndex) + { + + int purposeIndex = 0; + String purpose = ""; + if (segmentTypeIndex < 0) + { + purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } + + // create a temporary tour to use to calculate mode choice logsum + Tour mcLogsumTour = new Tour(person, 0, purposeIndex); + mcLogsumTour.setTourPurpose(purpose); + mcLogsumTour.setTourOrigMgra(household.getHhMgra()); + mcLogsumTour.setTourDestMgra(sampleDestMgra); + mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); + mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + + if (household.getDebugChoiceModels()) + { + dcManLogger.info(""); + dcManLogger.info(""); + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, + dcManLogger, person); + } + + double logsum = -1; + try + { + logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, + choiceModelDescription, decisionMakerLabel); + } catch (Exception e) + { + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + logger.fatal("Exception caught:", e); + System.exit(-1); + } + + return logsum; + } + + public int getModelIndex() + { + return modelIndex; + } + + public void setDcSizeObject(DestChoiceSize dcSizeObj) + { + this.dcSizeObj = dcSizeObj; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java new file mode 100644 index 0000000..aa90422 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java @@ -0,0 +1,209 @@ +package org.sandag.abm.ctramp; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagTourBasedModel; + +import com.pb.common.calculator.DataEntry; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; + +/** + * @author Jim Hicks + * + * Class for managing matrix data in a remote process and accessed by + * UECs using RMI. + */ +public class MatrixDataServer + implements MatrixDataServerIf, Serializable +{ + + private static Logger logger = Logger.getLogger(MatrixDataServer.class); + + private Object objectLock; + + private static final String VERSION = "2.3_OMX_Only"; + + // These are used if the server is started manually by running this class's + // main(). If so, these must be defined consistently with + // any class that acts as a client to the server, i.e. the client must know + // the + // address and port as well. + private static final String MATRIX_DATA_SERVER_ADDRESS = "127.0.0.1"; + private static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final String MATRIX_DATA_SERVER_NAME = MatrixDataServer.class + .getCanonicalName(); + private static final String MATRIX_DATA_SERVER_LABEL = "matrix server"; + + private HashMap matrixEntryMap; + private HashMap matrixMap; + + public MatrixDataServer() + { + + // create the HashMap objects to keep track of matrix data read by the server + matrixEntryMap = new HashMap(); + matrixMap = new HashMap(); + + objectLock = new Object(); + } + + public String testRemote(String remoteObjectName) + { + logger.info("testRemote() called by remote process: " + remoteObjectName + "."); + return String.format("testRemote() method in %s called by %s.", this.getClass() + .getCanonicalName(), remoteObjectName); + } + + public String testRemote() + { + logger.info("testRemote() called by remote process."); + return String.format("testRemote() method in %s called.", this.getClass() + .getCanonicalName()); + } + + /* + * Read a matrix. + * + * @param matrixEntry a DataEntry describing the matrix to read + * + * @return a Matrix + */ + public Matrix getMatrix(DataEntry matrixEntry) + { + + Matrix matrix; + + synchronized (objectLock) + { + + String name = matrixEntry.name; + + if (matrixEntryMap.containsKey(name)) + { + matrix = matrixMap.get(name); + } else + { + + //create 64bit matrix reader + String fileName = matrixEntry.fileName; + MatrixReader mr = MatrixReader.createReader(MatrixType.OMX, new File(fileName)); + matrix = mr.readMatrix(matrixEntry.matrixName); + logger.info("Read " + matrixEntry.matrixName + " as " + name + " from " + fileName); + + // Use token name from control file for matrix name (not name + // from underlying matrix) + matrix.setName(matrixEntry.name); + + matrixMap.put(name, matrix); + matrixEntryMap.put(name, matrixEntry); + } + + } + + return matrix; + } + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m) + { + + File outFile = new File(fileName); + MatrixWriter writer = MatrixWriter.createWriter(MatrixType.OMX, outFile); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + } + + writer.writeMatrices(names, m); + } + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m, MatrixType mt) + { + writeMatrixFile(fileName, m); + } + + public void clear() + { + if (matrixMap != null) + { + matrixMap = new HashMap(); + logger.info("MatrixDataServer matrixMap object is cleared."); + } else + { + logger.info("MatrixDataServer.clear() called, but matrixMap object is null."); + } + + if (matrixEntryMap != null) + { + matrixEntryMap = new HashMap(); + logger.info("MatrixDataServer matrixEntryMap object is cleared."); + } else + { + logger.info("MatrixDataServer.clear() called, but matrixEntryMap object is null."); + } + } + + //Empty methods to maintain compatibility + public void start32BitMatrixIoServer(MatrixType mType) {} + public void stop32BitMatrixIoServer() {} + public void setRam(int ram) {} + + public static void main(String[] args) throws Exception + { + + String serverAddress = MATRIX_DATA_SERVER_ADDRESS; + int serverPort = MATRIX_DATA_SERVER_PORT; + String className = MATRIX_DATA_SERVER_NAME; + String serverLabel = MATRIX_DATA_SERVER_LABEL; + int ram = -1; + + for (int i = 0; i < args.length; i++) + { + if (args[i].equalsIgnoreCase("-hostname")) serverAddress = args[i + 1]; + else if (args[i].equalsIgnoreCase("-port")) serverPort = Integer.parseInt(args[i + 1]); + else if (args[i].equalsIgnoreCase("-label")) serverLabel = args[i + 1]; + else if (args[i].equalsIgnoreCase("-ram")) ram = Integer.parseInt(args[i + 1]); + } + + MatrixDataServer matrixServer = new MatrixDataServer(); + + // bind this concrete object with the cajo library objects for managing RMI + Remote.config(serverAddress, serverPort, null, 0); + ItemServer.bind(matrixServer, className); + + // log that the server started + logger.info("server starting on " + (System.getProperty("os.arch")) + + " operating system."); + logger.info(String.format("%s version %s started on: %s:%d", serverLabel, VERSION, + serverAddress, serverPort)); + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java new file mode 100644 index 0000000..767b869 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java @@ -0,0 +1,98 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; + +import com.pb.common.calculator.DataEntry; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; + +/** + * @author Jim Hicks + * + * Class for managing matrix data in a remote process and accessed by + * UECs using RMI. + */ +public class MatrixDataServerRmi + implements MatrixDataServerIf, Serializable +{ + + // protected static Logger logger = + // Logger.getLogger(MatrixDataServerRmi.class); + + UtilRmi remote; + String connectString; + + public MatrixDataServerRmi(String hostname, int port, String className) + { + + connectString = String.format("//%s:%d/%s", hostname, port, className); + remote = new UtilRmi(connectString); + + } + + public void clear() + { + Object[] objArray = {}; + remote.method("clear", objArray); + } + + public void writeMatrixFile(String fileName, Matrix[] m, MatrixType mt) + { + Object[] objArray = {fileName, m, mt}; + remote.method("writeMatrixFile", objArray); + } + + public Matrix getMatrix(DataEntry dataEntry) + { + Object[] objArray = {dataEntry}; + return (Matrix) remote.method("getMatrix", objArray); + } + + public void start32BitMatrixIoServer(MatrixType mType) + { + Object[] objArray = {mType}; + remote.method("start32BitMatrixIoServer", objArray); + } + + public void start32BitMatrixIoServer(MatrixType mType, String label) + { + Object[] objArray = {mType, label}; + remote.method("start32BitMatrixIoServer", objArray); + } + + public void stop32BitMatrixIoServer() + { + Object[] objArray = {}; + remote.method("stop32BitMatrixIoServer", objArray); + } + + public void stop32BitMatrixIoServer(String label) + { + Object[] objArray = {label}; + remote.method("stop32BitMatrixIoServer", objArray); + } + + public String testRemote(String remoteObjectName) + { + Object[] objArray = {remoteObjectName}; + return (String) remote.method("testRemote", objArray); + } + + public String testRemote() + { + Object[] objArray = {}; + return (String) remote.method("testRemote", objArray); + } + + /** + * This method is included in the Interface this class implements, but is not used anywhere by the SANDAG model. + * It is included hear to satisfy the interface only. + */ + @Override + public void writeMatrixFile(String fileName, Matrix[] m) { + // TODO Auto-generated method stub + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java new file mode 100644 index 0000000..2699a7b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java @@ -0,0 +1,754 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; + +import com.pb.common.newmodel.Alternative; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.ConcreteAlternative; +import com.pb.common.newmodel.LogitModel; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.calculator.IndexValues; + + +public class McLogsumsCalculator implements Serializable +{ + + private transient Logger autoSkimLogger = Logger.getLogger(McLogsumsCalculator.class); + + public static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; + public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; + + + public static final int WTW = 0; + public static final int WTD = 1; + public static final int DTW = 2; + public static final int NUM_ACC_EGR = 3; + + public static final int OUT = 0; + public static final int IN = 1; + public static final int NUM_DIR = 2; + + private BestTransitPathCalculator bestPathUEC; + private double[] tripModeChoiceSegmentStoredProbabilities; + private double[] tripModeChoiceSegmentStoredVOTs; + private float tripModeChoiceSegmentStoredParkingCost; + + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + private int[] parkingArea; + + private double[][] bestWtwTapPairsOut; + private double[][] bestWtwTapPairsIn; + private double[][] bestWtdTapPairsOut; + private double[][] bestWtdTapPairsIn; + private double[][] bestDtwTapPairsOut; + private double[][] bestDtwTapPairsIn; + + private double[][] bestWtwTripTapPairs; + private double[][] bestWtdTripTapPairs; + private double[][] bestDtwTripTapPairs; + + private AutoAndNonMotorizedSkimsCalculator anm; + + private int setTourMcLogsumDmuAttributesTotalTime = 0; + private int setTripMcLogsumDmuAttributesTotalTime = 0; + + + + public McLogsumsCalculator() + { + if (mgraManager == null) + mgraManager = MgraDataManager.getInstance(); + + if (tazManager == null) + tazManager = TazDataManager.getInstance(); + + this.lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); + this.lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); + this.lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); + this.parkingArea = mgraManager.getMgraParkAreas(); + + tripModeChoiceSegmentStoredVOTs = new double[3]; + } + + + public BestTransitPathCalculator getBestTransitPathCalculator() + { + return bestPathUEC; + } + + + public void setupSkimCalculators(HashMap rbMap) + { + bestPathUEC = new BestTransitPathCalculator(rbMap); + anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); + } + + public void setTazDistanceSkimArrays( double[][][] storedFromTazDistanceSkims, double[][][] storedToTazDistanceSkims ) { + anm.setTazDistanceSkimArrays( storedFromTazDistanceSkims, storedToTazDistanceSkims ); + } + + + public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() + { + return anm; + } + + public void setTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean debug ) + { + + setNmTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); + setWtwTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); + setDtwTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); + setWtdTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); + + // set the land use data items in the DMU for the origin + mcDmuObject.setOrigDuDen( mgraManager.getDuDenValue( origMgra ) ); + mcDmuObject.setOrigEmpDen( mgraManager.getEmpDenValue( origMgra ) ); + mcDmuObject.setOrigTotInt( mgraManager.getTotIntValue( origMgra ) ); + + // set the land use data items in the DMU for the destination + mcDmuObject.setDestDuDen( mgraManager.getDuDenValue( destMgra ) ); + mcDmuObject.setDestEmpDen( mgraManager.getEmpDenValue( destMgra ) ); + mcDmuObject.setDestTotInt( mgraManager.getTotIntValue( destMgra ) ); + + mcDmuObject.setLsWgtAvgCostM( lsWgtAvgCostM[destMgra] ); + mcDmuObject.setLsWgtAvgCostD( lsWgtAvgCostD[destMgra] ); + mcDmuObject.setLsWgtAvgCostH( lsWgtAvgCostH[destMgra] ); + + int tourOrigTaz = mgraManager.getTaz(origMgra); + int tourDestTaz = mgraManager.getTaz(destMgra); + + mcDmuObject.setPTazTerminalTime( tazManager.getOriginTazTerminalTime(tourOrigTaz) ); + mcDmuObject.setATazTerminalTime( tazManager.getDestinationTazTerminalTime(tourDestTaz) ); + + Person person = mcDmuObject.getPersonObject(); + + double reimbursePct=0; + if(person!=null) { + reimbursePct = person.getParkingReimbursement(); + } + + mcDmuObject.setReimburseProportion( reimbursePct ); + mcDmuObject.setParkingArea(parkingArea[destMgra]); + + + } + + + public double calculateTourMcLogsum(int origMgra, int destMgra, int departPeriod, int arrivePeriod, + ChoiceModelApplication mcModel, TourModeChoiceDMU mcDmuObject) + { + + long currentTime = System.currentTimeMillis(); + setTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, mcDmuObject.getDmuIndexValues().getDebug() ); + setTourMcLogsumDmuAttributesTotalTime += ( System.currentTimeMillis() - currentTime ); + + // mode choice UEC references highway skim matrices directly, so set index orig/dest to O/D TAZs. + IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); + int tourOrigTaz = mgraManager.getTaz(origMgra); + int tourDestTaz = mgraManager.getTaz(destMgra); + mcDmuIndex.setOriginZone(tourOrigTaz); + mcDmuIndex.setDestZone(tourDestTaz); + mcDmuObject.setOriginMgra(origMgra); + mcDmuObject.setDestMgra(destMgra); + + mcModel.computeUtilities(mcDmuObject, mcDmuIndex); + double logsum = mcModel.getLogsum(); + + return logsum; + + } + + public void setWalkTransitLogSumUnavailable( TripModeChoiceDMU tripMcDmuObject ) { + tripMcDmuObject.setTransitLogSum( WTW, bestPathUEC.NA ); + } + + public void setDriveTransitLogSumUnavailable( TripModeChoiceDMU tripMcDmuObject, boolean isInbound ) { + + // set drive transit skim attributes to unavailable + if ( ! isInbound ) { + tripMcDmuObject.setTransitLogSum( DTW, bestPathUEC.NA); + } + else { + tripMcDmuObject.setTransitLogSum( WTD, bestPathUEC.NA); + } + + } + + + public double calculateTripMcLogsum(int origMgra, int destMgra, int departPeriod, ChoiceModelApplication mcModel, TripModeChoiceDMU mcDmuObject, Logger myLogger) + { + long currentTime = System.currentTimeMillis(); + setNmTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, mcDmuObject.getHouseholdObject().getDebugChoiceModels() ); + + mcDmuObject.setTripPeriod(departPeriod); + + // set the land use data items in the DMU for the origin + mcDmuObject.setOrigDuDen( mgraManager.getDuDenValue( origMgra ) ); + mcDmuObject.setOrigEmpDen( mgraManager.getEmpDenValue( origMgra ) ); + mcDmuObject.setOrigTotInt( mgraManager.getTotIntValue( origMgra ) ); + + // set the land use data items in the DMU for the destination + mcDmuObject.setDestDuDen( mgraManager.getDuDenValue( destMgra ) ); + mcDmuObject.setDestEmpDen( mgraManager.getEmpDenValue( destMgra ) ); + mcDmuObject.setDestTotInt( mgraManager.getTotIntValue( destMgra ) ); + + // mode choice UEC references highway skim matrices directly, so set index orig/dest to O/D TAZs. + IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); + mcDmuIndex.setOriginZone(mgraManager.getTaz(origMgra)); + mcDmuIndex.setDestZone(mgraManager.getTaz(destMgra)); + mcDmuObject.setOriginMgra(origMgra); + mcDmuObject.setDestMgra(destMgra); + + setTripMcLogsumDmuAttributesTotalTime += ( System.currentTimeMillis() - currentTime ); + mcDmuObject.setPTazTerminalTime( tazManager.getOriginTazTerminalTime(mgraManager.getTaz(origMgra)) ); + mcDmuObject.setATazTerminalTime( tazManager.getDestinationTazTerminalTime(mgraManager.getTaz(destMgra)) ); + + + mcModel.computeUtilities(mcDmuObject, mcDmuIndex); + double logsum = mcModel.getLogsum(); + tripModeChoiceSegmentStoredProbabilities = Arrays.copyOf( mcModel.getCumulativeProbabilities(), mcModel.getNumberOfAlternatives() ); + + //also save the VOTs from the model + UtilityExpressionCalculator uec = mcModel.getUEC(); + + ModelStructure modelStructure = mcDmuObject.modelStructure; + + tripModeChoiceSegmentStoredVOTs[0] = uec.getValueForIndex(uec.lookupVariableIndex("vot")); + tripModeChoiceSegmentStoredVOTs[1] = uec.getValueForIndex(uec.lookupVariableIndex("votS2")); + tripModeChoiceSegmentStoredVOTs[2] = uec.getValueForIndex(uec.lookupVariableIndex("votS3")); + + tripModeChoiceSegmentStoredParkingCost = (float) uec.getValueForIndex(uec.lookupVariableIndex("parkingCost")); + + if ( mcDmuObject.getHouseholdObject().getDebugChoiceModels() ) + mcModel.logUECResults(myLogger, "Trip Mode Choice Utility Expressions for mgras: " + origMgra + " to " + destMgra + " for HHID: " + mcDmuIndex.getHHIndex() ); + + return logsum; + + } + + + /** + * return the array of mode choice model cumulative probabilities determined while + * computing the mode choice logsum for the trip segmen during stop location choice. + * These probabilities arrays are stored for each sampled stop location so that when + * the selected sample stop location is known, the mode choice can be drawn from the + * already computed probabilities. + * + * @return mode choice cumulative probabilities array + */ + public double[] getStoredSegmentCumulativeProbabilities() { + return tripModeChoiceSegmentStoredProbabilities; + } + + public double[] getStoredSegmentVOTs() { + return tripModeChoiceSegmentStoredVOTs; + } + public double[][] getBestWtwTapsOut() + { + return bestWtwTapPairsOut; + } + + public double[][] getBestWtwTapsIn() + { + return bestWtwTapPairsIn; + } + + public double[][] getBestWtdTapsOut() + { + return bestWtdTapPairsOut; + } + + public double[][] getBestWtdTapsIn() + { + return bestWtdTapPairsIn; + } + + public double[][] getBestDtwTapsOut() + { + return bestDtwTapPairsOut; + } + + public double[][] getBestDtwTapsIn() + { + return bestDtwTapPairsIn; + } + + public double[][] getBestWtwTripTaps() + { + return bestWtwTripTapPairs; + } + + public double[][] getBestDtwTripTaps() + { + return bestDtwTripTapPairs; + } + + public double[][] getBestWtdTripTaps() + { + return bestWtdTripTapPairs; + } + + + private void setNmTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) + { + // non-motorized, outbound then inbound + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + departPeriod = skimPeriodIndex; + double[] nmSkimsOut = anm.getNonMotorizedSkims(origMgra, destMgra, departPeriod, loggingEnabled, autoSkimLogger); + if (loggingEnabled) + anm.logReturnedSkims(origMgra, destMgra, departPeriod, nmSkimsOut, "non-motorized outbound", autoSkimLogger); + + skimPeriodIndex = ModelStructure.getSkimPeriodIndex(arrivePeriod); + arrivePeriod = skimPeriodIndex; + double[] nmSkimsIn = anm.getNonMotorizedSkims(destMgra, origMgra, arrivePeriod, loggingEnabled, autoSkimLogger); + if (loggingEnabled) anm.logReturnedSkims(destMgra, origMgra, arrivePeriod, nmSkimsIn, "non-motorized inbound", autoSkimLogger); + + int walkIndex = anm.getNmWalkTimeSkimIndex(); + mcDmuObject.setNmWalkTimeOut( nmSkimsOut[walkIndex] ); + mcDmuObject.setNmWalkTimeIn( nmSkimsIn[walkIndex] ); + + int bikeIndex = anm.getNmBikeTimeSkimIndex(); + mcDmuObject.setNmBikeTimeOut( nmSkimsOut[bikeIndex] ); + mcDmuObject.setNmBikeTimeIn( nmSkimsIn[bikeIndex] ); + + } + + private void setWtwTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) + { + + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // walk access, walk egress transit, outbound + int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + bestWtwTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); + + if (bestWtwTapPairsOut[0] == null) { + mcDmuObject.setTransitLogSum( WTW, false, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(mcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); + } + + bestWtwTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestWtwTapPairsOut, walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); + double logsumOut = bestPathUEC.calcTripLogSum(bestWtwTapPairsOut, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( WTW, false, logsumOut); + } + + //setup best path dmu variables + walkDmu = new TransitWalkAccessDMU(); + driveDmu = new TransitDriveAccessDMU(); + + // walk access, walk egress transit, inbound + int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); + bestWtwTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); + + if (bestWtwTapPairsIn[0] == null) { + mcDmuObject.setTransitLogSum( WTW, true, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(mcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); + } + + bestWtwTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestWtwTapPairsIn, walkDmu, driveDmu, WTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); + double logsumIn = bestPathUEC.calcTripLogSum(bestWtwTapPairsIn, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( WTW, true, logsumIn); + } + } + + private void setWtdTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) + { + + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // logsum for WTD outbound is never used -> set to NA + mcDmuObject.setTransitLogSum( WTD, false, bestPathUEC.NA ); + /* TODO: - remove this section of code after successful testing + // walk access, drive egress transit, outbound + int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); + bestWtdTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger); + + if (bestWtdTapPairsOut[0] == null) { + mcDmuObject.setTransitLogSum( WTD, false, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.personType : mcDmuObject.getPersonType()); + walkDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.personType : mcDmuObject.getPersonType()); + driveDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); + + bestWtdTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestWtdTapPairsOut, walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger); + double logsumOut = bestPathUEC.calcTripLogSum(bestWtdTapPairsOut, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( WTD, false, logsumOut); + } + */ + + // walk access, drive egress transit, inbound + int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + + bestWtdTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); + + if (bestWtdTapPairsIn[0] == null) { + mcDmuObject.setTransitLogSum( WTD, true, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(mcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); + } + bestWtdTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestWtdTapPairsIn, walkDmu, driveDmu, WTD, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); + double logsumIn = bestPathUEC.calcTripLogSum(bestWtdTapPairsIn, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( WTD, true, logsumIn); + } + } + + private void setDtwTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) + { + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // drive access, walk egress transit, outbound + int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + + bestDtwTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); + + if (bestDtwTapPairsOut[0] == null) { + mcDmuObject.setTransitLogSum( DTW, false, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(mcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); + } + + bestDtwTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestDtwTapPairsOut, walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); + double logsumOut = bestPathUEC.calcTripLogSum(bestDtwTapPairsOut, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( DTW, false, logsumOut); + } + + // logsum for DTW inbound is never used -> set to NA + mcDmuObject.setTransitLogSum( DTW, true, bestPathUEC.NA ); + + /* TODO: remove this section of code after successful testing + //setup best path dmu variables + walkDmu = new TransitWalkAccessDMU(); + driveDmu = new TransitDriveAccessDMU(); + + // drive access, walk egress transit, inbound + int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); + bestDtwTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger); + + if (bestDtwTapPairsIn[0] == null) { + mcDmuObject.setTransitLogSum( DTW, true, bestPathUEC.NA ); + } else { + // calculate logsum + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + walkDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); + walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.personType : mcDmuObject.getPersonType()); + walkDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); + driveDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); + driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.personType : mcDmuObject.getPersonType()); + driveDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); + + bestDtwTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestDtwTapPairsIn, walkDmu, driveDmu, DTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger); + double logsumIn = bestPathUEC.calcTripLogSum(bestDtwTapPairsIn, loggingEnabled, autoSkimLogger); + mcDmuObject.setTransitLogSum( DTW, true, logsumIn); + } + */ + } + + public void setNmTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) + { + + double[] nmSkims = null; + + // non-motorized, outbound then inbound + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + departPeriod = skimPeriodIndex; + nmSkims = anm.getNonMotorizedSkims(origMgra, destMgra, departPeriod, loggingEnabled, autoSkimLogger); + if (loggingEnabled) + anm.logReturnedSkims(origMgra, destMgra, departPeriod, nmSkims, "non-motorized trip mode choice skims", autoSkimLogger); + + int walkIndex = anm.getNmWalkTimeSkimIndex(); + tripMcDmuObject.setNonMotorizedWalkTime(nmSkims[walkIndex] ); + + int bikeIndex = anm.getNmBikeTimeSkimIndex(); + tripMcDmuObject.setNonMotorizedBikeTime(nmSkims[bikeIndex] ); + + } + + public void setWtwTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled) + { + + if (bestTapPairs == null) { + if(loggingEnabled){ + autoSkimLogger.info("Attempting to set WTW Trip MC DMU Attributes for null best TAP pairs array"); + } + tripMcDmuObject.setTransitLogSum( WTW, bestPathUEC.NA ); + bestWtwTripTapPairs = bestTapPairs; + return; + } + + // calculate logsum + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); + tripMcDmuObject.setTransitLogSum( WTW, logsum); + bestWtwTripTapPairs = bestTapPairs; + + } + + public void setDtwTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled ) + { + + if (bestTapPairs == null) { + if(loggingEnabled){ + autoSkimLogger.info("Attempting to set DTW Trip MC DMU Attributes for null best TAP pairs array"); + } + tripMcDmuObject.setTransitLogSum( DTW, bestPathUEC.NA ); + bestDtwTripTapPairs = bestTapPairs; + return; + } + + // calculate logsum + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); + + if(loggingEnabled) + autoSkimLogger.info("Setting DTW logsum in trip MC DMU object to "+logsum); + + tripMcDmuObject.setTransitLogSum( DTW, logsum); + bestDtwTripTapPairs = bestTapPairs; + + } + + public void setWtdTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled ) + { + + if (bestTapPairs == null) { + if(loggingEnabled){ + autoSkimLogger.info("Attempting to set WTD Trip MC DMU Attributes for null best TAP pairs array"); + } + tripMcDmuObject.setTransitLogSum( WTD, bestPathUEC.NA ); + bestWtdTripTapPairs = bestTapPairs; + return; + } + + // calculate logsum + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); + tripMcDmuObject.setTransitLogSum( WTD, logsum); + bestWtdTripTapPairs = bestTapPairs; + + } + + public void setWtwTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) + { + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // walk access and walk egress for transit segment + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + + // store best tap pairs for walk-transit-walk + bestWtwTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance ); + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(tripMcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); + driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); + } + // calculate logsum + bestWtwTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestWtwTripTapPairs, walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); + double logsum = bestPathUEC.calcTripLogSum(bestWtwTripTapPairs, loggingEnabled, autoSkimLogger); + tripMcDmuObject.setTransitLogSum( WTW, logsum); + + } + + public void setWtdTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) + { + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // walk access, drive egress transit, outbound + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + + // store best tap pairs using outbound direction array + bestWtdTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(tripMcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); + driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); + } + + // calculate logsum + bestWtdTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestWtdTripTapPairs, walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); + double logsum = bestPathUEC.calcTripLogSum(bestWtdTripTapPairs, loggingEnabled, autoSkimLogger); + tripMcDmuObject.setTransitLogSum( WTD, logsum); + + } + + public void setDtwTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) + { + //setup best path dmu variables + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + + // drive access, walk egress transit, outbound + int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); + int pTaz = mgraManager.getTaz(origMgra); + int aTaz = mgraManager.getTaz(destMgra); + float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; + + // store best tap pairs using outbound direction array + bestDtwTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); + + //set person specific variables and re-calculate best tap pair utilities + walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); + driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); + driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); + + //catch issues where the trip mode choice DMU was set up without a household or person object + if(tripMcDmuObject.getHouseholdObject()!=null){ + walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); + driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); + } + // calculate logsum + bestDtwTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestDtwTripTapPairs, walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); + double logsum = bestPathUEC.calcTripLogSum(bestDtwTripTapPairs, loggingEnabled, autoSkimLogger); + tripMcDmuObject.setTransitLogSum( DTW, logsum); + + } + + //select best transit path from N-path for trip + public int chooseTripPath(double rnum, double[][] bestTapPairs, boolean myTrace, Logger myLogger) { + return bestPathUEC.chooseTripPath(rnum, bestTapPairs, myTrace, myLogger); + } + + + public float getTripModeChoiceSegmentStoredParkingCost() { + return tripModeChoiceSegmentStoredParkingCost; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java new file mode 100644 index 0000000..356b914 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java @@ -0,0 +1,119 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + */ +public class MicromobilityChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(MicromobilityChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + protected double ivtCoeff; + protected double costCoeff; + protected float walkTime; + protected boolean isTransit; + protected boolean microTransitAvailable; + + + public MicromobilityChoiceDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + } + + + public double getIvtCoeff() { + return ivtCoeff; + } + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public double getCostCoeff() { + return costCoeff; + } + + public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; + } + + public float getWalkTime() { + return walkTime; + } + + public void setWalkTime(float walkTime) { + this.walkTime = walkTime; + } + + public boolean isTransit() { + return isTransit; + } + + public void setTransit(boolean isTransit) { + this.isTransit = isTransit; + } + + public boolean isMicroTransitAvailable() { + return microTransitAvailable; + } + + public void setMicroTransitAvailable(boolean microTransitAvailable) { + this.microTransitAvailable = microTransitAvailable; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java new file mode 100644 index 0000000..724da7a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java @@ -0,0 +1,449 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class MicromobilityChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("micromobility"); + + private static final String MM_CONTROL_FILE_TARGET = "micromobility.uec.file"; + private static final String MM_DATA_SHEET_TARGET = "micromobility.data.page"; + private static final String MM_MODEL_SHEET_TARGET = "micromobility.model.page"; + private static final String MT_TAP_FILE_TARGET = "active.microtransit.tap.file"; + private static final String MT_MAZ_FILE_TARGET = "active.microtransit.mgra.file"; + + public static final int MM_MODEL_WALK_ALT = 0; + public static final int MM_MODEL_MICROMOBILITY_ALT = 1; + public static final int MM_MODEL_MICROTRANSIT_ALT = 2; + + + private ChoiceModelApplication mmModel; + private MicromobilityChoiceDMU mmDmuObject; + + // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose + double[] ivtCoeffs; + double[] incomeCoeffs; + double[] incomeExponents; + + private static final String PROPERTIES_TRIP_UTILITY_IVT_COEFFS = "trip.utility.ivt.coeffs"; + private static final String PROPERTIES_TRIP_UTILITY_INCOME_COEFFS = "trip.utility.income.coeffs"; + private static final String PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS = "trip.utility.income.exponents"; + private ModelStructure modelStructure; + private MgraDataManager mgraDataManager; + + private HashSet microtransitTaps; + private HashSet microtransitMazs; + + + public MicromobilityChoiceModel(HashMap propertyMap, + ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory) + { + + setupMicromobilityChoiceModelApplication(propertyMap, myModelStructure, dmuFactory); + } + + private void setupMicromobilityChoiceModelApplication(HashMap propertyMap, + ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory) + { + logger.info("setting up micromobility choice model."); + + modelStructure = myModelStructure; + + // locate the micromobility choice UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mmUecFile = uecFileDirectory + propertyMap.get(MM_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_MODEL_SHEET_TARGET); + + // create the micromobility choice model DMU object. + mmDmuObject = dmuFactory.getMicromobilityChoiceDMU(); + + // create the transponder choice model object + mmModel = new ChoiceModelApplication(mmUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) mmDmuObject); + + + //get the coefficients for ivt and the coefficients to calculate the cost coefficient + ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_IVT_COEFFS); + incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_COEFFS); + incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS); + + mgraDataManager = MgraDataManager.getInstance(); + + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String microTransitTapFile = projectDirectory + propertyMap.get(MT_TAP_FILE_TARGET); + String microTransitMazFile = projectDirectory + propertyMap.get(MT_MAZ_FILE_TARGET); + + TableDataSet microTransitTapData = Util.readTableDataSet(microTransitTapFile); + TableDataSet microTransitMazData = Util.readTableDataSet(microTransitMazFile); + + microtransitTaps = new HashSet(); + microtransitMazs = new HashSet(); + + for(int i=1;i<=microTransitTapData.getRowCount();++i) { + + int tap = (int) microTransitTapData.getValueAt(i,"TAP"); + microtransitTaps.add(tap); + } + + for(int i=1;i<=microTransitMazData.getRowCount();++i) { + + int maz = (int) microTransitMazData.getValueAt(i,"MGRA"); + microtransitMazs.add(maz); + } + + + } + + + /** + * Apply model to all trips for the household. + * + * @param household + */ + public void applyModel(Household household) { + + for(Person person : household.getPersons()) { + + if(person==null) + continue; + + //work tours + if(person.getListOfWorkTours()!=null) { + + for(Tour tour:person.getListOfWorkTours()) + applyModel(household, person, tour); + } + + //school tours + if(person.getListOfSchoolTours()!=null) { + + for(Tour tour:person.getListOfSchoolTours()) + applyModel(household, person, tour); + } + + //non-mandatory tours + if(person.getListOfIndividualNonMandatoryTours()!=null) { + + for(Tour tour:person.getListOfIndividualNonMandatoryTours()) + applyModel(household, person, tour); + } + + //at-work sub tours + if(person.getListOfAtWorkSubtours()!=null) { + + for(Tour tour:person.getListOfAtWorkSubtours()) + applyModel(household, person, tour); + } + + } + + + } + + public void applyModel(Household household, Person person, Tour tour) { + + //apply to outbound stops + if(tour.getOutboundStops()!=null) { + + for(Stop s: tour.getOutboundStops()) + applyModel(household, person, tour, s); + } + + //apply to inbound stops + if(tour.getInboundStops()!=null) { + + for(Stop s: tour.getInboundStops()) + applyModel(household, person, tour, s); + } + + + } + + public void applyModel(Household household, Person person, Tour tour, Stop s) + { + + if(tour==null) + return; + + + if(!modelStructure.getTourModeIsWalk(s.getMode()) && !modelStructure.getTourModeIsWalkTransit(s.getMode())&& !modelStructure.getTourModeIsDriveTransit(s.getMode())) + return; + + int homeMaz = household.getHhMgra(); + double income = (double) household.getIncomeInDollars(); + + int category = IntermediateStopChoiceModels.PURPOSE_CATEGORIES[tour.getTourPrimaryPurposeIndex()]; + double ivtCoeff = ivtCoeffs[category]; + double incomeCoeff = incomeCoeffs[category]; + double incomeExpon = incomeExponents[category]; + double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); + double timeFactor = 1.0f; + if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + timeFactor = tour.getJointTourTimeFactor(); + else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) + timeFactor = person.getTimeFactorWork(); + else + timeFactor = person.getTimeFactorNonWork(); + + mmDmuObject.setIvtCoeff(ivtCoeff * timeFactor); + mmDmuObject.setCostCoeff(costCoeff); + int originMaz = s.getOrig(); + int destMaz = s.getDest(); + + if(modelStructure.getTourModeIsWalk(s.getMode())) + mmDmuObject.setTransit(false); + else + mmDmuObject.setTransit(true); + + if(modelStructure.getTourModeIsWalk(s.getMode())) { + + float walkTime = mgraDataManager.getMgraToMgraWalkTime(originMaz, destMaz); + mmDmuObject.setWalkTime(walkTime); + + if(microtransitMazs.contains(originMaz) && microtransitMazs.contains(destMaz)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + s.setMicromobilityWalkLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice(household, person, tour, s); + s.setMicromobilityWalkMode(chosenAlt); + + // write choice model alternative info to log file + if (household.getDebugChoiceModels()) + { + String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Micromobility Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + + }else if(modelStructure.getTourModeIsWalkTransit(s.getMode())) { + + //access + int tapPosition = mgraDataManager.getTapPosition(originMaz, s.boardTap); + if(tapPosition==-1) { + logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); + logger.warn("Can't find walk connection from origin to board TAP; skipping micromobility choice"); + return; + } + float walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); + mmDmuObject.setWalkTime(walkTime); + + if(microtransitMazs.contains(originMaz) && microtransitTaps.contains(s.boardTap)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + s.setMicromobilityAccessLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice(household, person, tour, s); + s.setMicromobilityAccessMode(chosenAlt); + + // write choice model alternative info to log file + if (household.getDebugChoiceModels()) + { + String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()+ " access choice"); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Micromobility Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + //egress + tapPosition = mgraDataManager.getTapPosition(destMaz, s.alightTap); + if(tapPosition==-1) { + logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); + logger.warn("Can't find walk connection from alight TAP to destination; skipping micromobility choice"); + return; + } + walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); + mmDmuObject.setWalkTime(walkTime); + + if(microtransitMazs.contains(destMaz) && microtransitTaps.contains(s.alightTap)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC + int closestMazToAlightTap = mgraDataManager.getClosestMgra(s.alightTap); + mmDmuObject.setDmuIndexValues(household.getHhId(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); + + // compute utilities and choose micromobility choice alternative. + logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + s.setMicromobilityEgressLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + chosenAlt = (byte) getChoice(household, person, tour, s); + s.setMicromobilityEgressMode(chosenAlt); + + // write choice model alternative info to log file + if (household.getDebugChoiceModels()) + { + String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()+ " egress choice"); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Micromobility Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + + + } else if( modelStructure.getTourModeIsDriveTransit(s.getMode()) ) { //drive-transit. Choose non-drive direction + + int tapPosition = 0; + float walkTime = 9999; + + if(s.isInboundStop()) { //inbound, so access mode is walk + tapPosition = mgraDataManager.getTapPosition(originMaz, s.boardTap); + if(tapPosition==-1) { + logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); + logger.warn("Can't find walk connection from origin to board TAP; skipping micromobility choice"); + return; + } + + walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); + + if(microtransitMazs.contains(originMaz) && microtransitTaps.contains(s.boardTap)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + }else { //outbound so egress mode is walk. + tapPosition = mgraDataManager.getTapPosition(destMaz, s.alightTap); + if(tapPosition==-1) { + logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); + logger.warn("Can't find walk connection from destination MAZ to alight TAP; skipping micromobility choice"); + return; + } + walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); + //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC + int closestMazToAlightTap = mgraDataManager.getClosestMgra(s.alightTap); + mmDmuObject.setDmuIndexValues(household.getHhId(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); + + if(microtransitMazs.contains(destMaz) && microtransitTaps.contains(s.alightTap)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + } + mmDmuObject.setWalkTime(walkTime); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice(household, person, tour, s); + + if(s.isInboundStop()) { //inbound, set access + s.setMicromobilityAccessMode(chosenAlt); + s.setMicromobilityAccessLogsum(logsum); + }else { //outound, set egress + s.setMicromobilityEgressMode(chosenAlt); + s.setMicromobilityEgressLogsum(logsum); + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels()) + { + String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Micromobility Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + + } + + } + + + /** + * Select the micromobility mode from the UEC. This is helper code for applyModel(), where utilities have already been calculated. + * + * @param household + * @param person + * @param tour + * @param s + * @return The micromobility mode. + */ + private int getChoice(Household household, Person person, Tour tour, Stop s) { + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + Random hhRandom = household.getHhRandom(); + if (mmModel.getAvailabilityCount() > 0) + { + double randomNumber = hhRandom.nextDouble(); + chosenAlt = mmModel.getChoiceResult(randomNumber); + return chosenAlt; + } else + { + String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); + String errorMessage = String + .format("Exception caught for %s, no available micromobility choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.info(errorMessage); + logger.info("Setting mode to walk"); + + mmModel.logUECResults(logger, decisionMaker); + return MM_MODEL_WALK_ALT; + } + + } + + /** + * This method calculates a cost coefficient based on the following formula: + * + * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) + * + * + * @param incomeCoeff + * @param incomeExponent + * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice + */ + public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ + + return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java new file mode 100644 index 0000000..db7b60e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java @@ -0,0 +1,600 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +/** + * Holds the tour purpose list as well as the market segments for each tour. + * + * @author D. Ory + * + */ +public abstract class ModelStructure + implements Serializable +{ + + public static final String[] DC_SIZE_AREA_TYPE_BASED_SEGMENTS = {"CBD", + "URBAN", "SUBURBAN", "RURAL" }; + + public static final String MANDATORY_CATEGORY = "MANDATORY"; + public static final String JOINT_NON_MANDATORY_CATEGORY = "JOINT_NON_MANDATORY"; + public static final String INDIVIDUAL_NON_MANDATORY_CATEGORY = "INDIVIDUAL_NON_MANDATORY"; + public static final String AT_WORK_CATEGORY = "AT_WORK"; + + public static final String MANDATORY_PATTERN = "M"; + public static final String NONMANDATORY_PATTERN = "N"; + public static final String HOME_PATTERN = "H"; + + public static final int FIRST_DEPART_HOUR = 4; + public static final int LAST_DEPART_HOUR = 24; + public static final int FIRST_TOD_INTERVAL_HOUR = 430; + public static final int LAST_TOD_INTERVAL_HOUR = 2400; + public static final float TOD_INTERVAL_IN_MINUTES = 30.0f; + public static final int MAX_TOD_INTERVAL = 40; + protected String[] TOD_INTERVAL_LABELS; + + public static final int EA_SKIM_PERIOD_INDEX = 0; + public static final int AM_SKIM_PERIOD_INDEX = 1; + public static final int MD_SKIM_PERIOD_INDEX = 2; + public static final int PM_SKIM_PERIOD_INDEX = 3; + public static final int EV_SKIM_PERIOD_INDEX = 4; + public static final int[] SKIM_PERIOD_INDICES = { + EA_SKIM_PERIOD_INDEX, AM_SKIM_PERIOD_INDEX, MD_SKIM_PERIOD_INDEX, PM_SKIM_PERIOD_INDEX, + EV_SKIM_PERIOD_INDEX }; + + public static final int[] PERIODCODES = { EA_SKIM_PERIOD_INDEX, AM_SKIM_PERIOD_INDEX, + MD_SKIM_PERIOD_INDEX, PM_SKIM_PERIOD_INDEX, EV_SKIM_PERIOD_INDEX }; + + public static final String[] SKIM_PERIOD_STRINGS = {"EA", + "AM", "MD", "PM", "EV" }; + + // define indices associated with valid skim period combinations + public static final int EA_EA = 0; + public static final int EA_AM = 1; + public static final int EA_MD = 2; + public static final int EA_PM = 3; + public static final int EA_EV = 4; + // AM cannot be before EA + public static final int AM_EA = -1; + public static final int AM_AM = 5; + public static final int AM_MD = 6; + public static final int AM_PM = 7; + public static final int AM_EV = 8; + // MD cannot be before EA or AM + public static final int MD_EA = -1; + public static final int MD_AM = -1; + public static final int MD_MD = 9; + public static final int MD_PM = 10; + public static final int MD_EV = 11; + // PM cannot be before EA, AM or PM + public static final int PM_EA = -1; + public static final int PM_AM = -1; + public static final int PM_MD = -1; + public static final int PM_PM = 12; + public static final int PM_EV = 13; + // EV cannot be before EA, AM, MD or PM + public static final int EV_EA = -1; + public static final int EV_AM = -1; + public static final int EV_MD = -1; + public static final int EV_PM = -1; + public static final int EV_EV = 14; + + // define an array that contains the set of the valid skim period + // combination indices + public static final int[] SKIM_PERIOD_COMBINATION_INDICES = {EA_EA, + EA_AM, EA_MD, EA_PM, EA_EV, AM_AM, AM_MD, AM_PM, AM_EV, MD_MD, MD_PM, MD_EV, PM_PM, + PM_EV, EV_EV }; + + // define a 2-D array for the set of skim period combinations associatedf + // with each skim period index value + public static final int[][] SKIM_PERIOD_COMBINATIONS = { + {EA_EA, EA_AM, EA_MD, EA_PM, EA_EV}, {AM_EA, AM_AM, AM_MD, AM_PM, AM_EV}, + {MD_EA, MD_AM, MD_MD, MD_PM, MD_EV}, {PM_EA, PM_AM, PM_MD, PM_PM, PM_EV}, + {EV_EA, EV_AM, EV_MD, EV_PM, EV_EV} }; + + // define model period labels associated with each model period index + public static final String[] MODEL_PERIOD_LABELS = {"EA", + "AM", "MD", "PM", "EV" }; + + // the upper TOD interval index for each model period (EA:1-3, AM:6-9, + // MD:10-22, PM:23-29, EV:30-40) + public static final int UPPER_EA = 3; + public static final int UPPER_AM = 9; + public static final int UPPER_MD = 22; + public static final int UPPER_PM = 29; + + public static final int[] PERIOD_ENDS = {UPPER_EA,UPPER_AM,UPPER_MD,UPPER_PM, 40}; + + private HashMap indexTimePeriodMap; + private HashMap timePeriodIndexMap; + + public static final int WORKS_AT_HOME_ALTERNATUVE_INDEX = 2; + public static final int WORKS_AT_HOME_LOCATION_INDICATOR = 99999; + public static final int NOT_ENROLLED_SEGMENT_INDEX = 88888; + + private HashMap primaryTourPurposeNameIndexMap = new HashMap(); + private HashMap indexPrimaryTourPurposeNameMap = new HashMap(); + + public static final String WORK_PRIMARY_PURPOSE_NAME = "Work"; + public static final String UNIVERSITY_PRIMARY_PURPOSE_NAME = "University"; + public static final String SCHOOL_PRIMARY_PURPOSE_NAME = "School"; + public static final String ESCORT_PRIMARY_PURPOSE_NAME = "Escort"; + public static final String SHOPPING_PRIMARY_PURPOSE_NAME = "Shop"; + public static final String OTH_MAINT_PRIMARY_PURPOSE_NAME = "Maintenance"; + public static final String EAT_OUT_PRIMARY_PURPOSE_NAME = "Eating Out"; + public static final String VISITING_PRIMARY_PURPOSE_NAME = "Visiting"; + public static final String OTH_DISCR_PRIMARY_PURPOSE_NAME = "Discretionary"; + public static final String WORK_BASED_PRIMARY_PURPOSE_NAME = "Work-Based"; + + public static final int WORK_PRIMARY_PURPOSE_INDEX = 1; + public static final int UNIVERSITY_PRIMARY_PURPOSE_INDEX = 2; + public static final int SCHOOL_PRIMARY_PURPOSE_INDEX = 3; + public static final int ESCORT_PRIMARY_PURPOSE_INDEX = 4; + public static final int SHOPPING_PRIMARY_PURPOSE_INDEX = 5; + public static final int OTH_MAINT_PRIMARY_PURPOSE_INDEX = 6; + public static final int EAT_OUT_PRIMARY_PURPOSE_INDEX = 7; + public static final int VISITING_PRIMARY_PURPOSE_INDEX = 8; + public static final int OTH_DISCR_PRIMARY_PURPOSE_INDEX = 9; + public static final int WORK_BASED_PRIMARY_PURPOSE_INDEX = 10; + public static final int NUM_PRIMARY_PURPOSES = 10; + + public static final int WORK_STOP_PURPOSE_INDEX = 1; + public static final int UNIV_STOP_PURPOSE_INDEX = 2; + public static final int ESCORT_STOP_PURPOSE_INDEX = 4; + public static final int SHOP_STOP_PURPOSE_INDEX = 5; + public static final int MAINT_STOP_PURPOSE_INDEX = 6; + public static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; + public static final int VISIT_STOP_PURPOSE_INDEX = 8; + public static final int DISCR_STOP_PURPOSE_INDEX = 9; + + public static final byte ESCORT_STOP_TYPE_DROPOFF = 1; + public static final byte ESCORT_STOP_TYPE_PICKUP = 2; + public static final int RIDE_SHARING_TYPE = 1; + public static final int PURE_ESCORTING_TYPE = 2; + + public static final int MAX_STOPS_PER_DIRECTION = 4; + + public String WORK_PURPOSE_NAME; + public String UNIVERSITY_PURPOSE_NAME; + public String SCHOOL_PURPOSE_NAME; + public String ESCORT_PURPOSE_NAME; + public String SHOPPING_PURPOSE_NAME; + public String EAT_OUT_PURPOSE_NAME; + public String OTH_MAINT_PURPOSE_NAME; + public String SOCIAL_PURPOSE_NAME; + public String OTH_DISCR_PURPOSE_NAME; + public String AT_WORK_PURPOSE_NAME; + public String AT_WORK_EAT_PURPOSE_NAME; + public String AT_WORK_BUSINESS_PURPOSE_NAME; + public String AT_WORK_MAINT_PURPOSE_NAME; + + public int AT_WORK_PURPOSE_INDEX_EAT; + public int AT_WORK_PURPOSE_INDEX_BUSINESS; + public int AT_WORK_PURPOSE_INDEX_MAINT; + + public String[] ESCORT_SEGMENT_NAMES; + public String[] AT_WORK_SEGMENT_NAMES; + + protected HashMap workSegmentNameIndexMap; + protected HashMap schoolSegmentNameIndexMap; + protected HashMap workSegmentIndexNameMap; + protected HashMap schoolSegmentIndexNameMap; + + // TODO: Determine which of the following can be eliminated + protected HashMap dcSoaUecIndexMap; + protected HashMap dcUecIndexMap; + protected HashMap tourModeChoiceUecIndexMap; + + protected HashMap dcSizeDcModelPurposeMap; + protected HashMap dcModelDcSizePurposeMap; + + protected HashMap dcModelPurposeIndexMap; // segments + // for + // which + // dc + // soa alternative models + // are applied + protected HashMap dcModelIndexPurposeMap; // segments + // for + // which + // dc + // soa alternative models + // are applied + + protected HashMap dcSizeSegmentIndexMap; // segments + // for + // which + // separate dc size + // coefficients are + // specified + protected HashMap dcSizeIndexSegmentMap; + protected HashMap dcSizeArrayPurposeIndexMap; // segments + // for + // which + // dc + // size terms are stored + protected HashMap dcSizeArrayIndexPurposeMap; + protected HashMap> dcSizePurposeSegmentMap; + + private String dcSizeCoeffPurposeFieldName = "purpose"; + private String dcSizeCoeffSegmentFieldName = "segment"; + + // TODO meld with what jim is doing on this front + protected String[] mandatoryDcModelPurposeNames; + protected String[] jointDcModelPurposeNames; + protected String[] nonMandatoryDcModelPurposeNames; + protected String[] atWorkDcModelPurposeNames; + + protected String workPurposeName; + protected String universityPurposeName; + protected String schoolPurposeName; + + protected String[] workPurposeSegmentNames; + protected String[] universityPurposeSegmentNames; + protected String[] schoolPurposeSegmentNames; + + protected HashMap stopFreqUecIndexMap; + protected HashMap stopLocUecIndexMap; + protected HashMap tripModeChoiceUecIndexMap; + + protected String[] jtfAltLabels; + protected String[] awfAltLabels; + + /** + * Assume name of the columns in the destination size coefficients file that + * contain the purpose strings is "purpose" and the column that contains the + * segment strings is "segment" + */ + public ModelStructure() + { + + workSegmentNameIndexMap = new HashMap(); + schoolSegmentNameIndexMap = new HashMap(); + workSegmentIndexNameMap = new HashMap(); + schoolSegmentIndexNameMap = new HashMap(); + + dcModelPurposeIndexMap = new HashMap(); + dcModelIndexPurposeMap = new HashMap(); + dcSoaUecIndexMap = new HashMap(); + dcUecIndexMap = new HashMap(); + tourModeChoiceUecIndexMap = new HashMap(); + stopFreqUecIndexMap = new HashMap(); + stopLocUecIndexMap = new HashMap(); + tripModeChoiceUecIndexMap = new HashMap(); + + // create a mapping between primary purpose + // names and purpose indices + createPrimaryPurposeMappings(); + + createIndexTimePeriodMap(); + + } + + public abstract HashMap getWorkSegmentNameIndexMap(); + + public abstract HashMap getSchoolSegmentNameIndexMap(); + + public abstract HashMap getWorkSegmentIndexNameMap(); + + public abstract HashMap getSchoolSegmentIndexNameMap(); + + // a derived class must implement these methods to retrieve purpose names + // for + // various personTypes making mandatory tours. + public abstract String getWorkPurpose(int incomeCategory); + + public abstract String getWorkPurpose(boolean isPtWorker, int incomeCategory); + + public abstract String getUniversityPurpose(); + + public abstract String getSchoolPurpose(int age); + + public abstract boolean getTourModeIsSov(int tourMode); + + public abstract boolean getTourModeIsSovOrHov(int tourMode); + + public abstract boolean getTourModeIsS2(int tourMode); + + public abstract boolean getTourModeIsS3(int tourMode); + + public abstract boolean getTourModeIsHov(int tourMode); + + public abstract boolean getTourModeIsNonMotorized(int tourMode); + + public abstract boolean getTourModeIsBike(int tourMode); + + public abstract boolean getTourModeIsWalk(int tourMode); + + public abstract boolean getTourModeIsTransit(int tourMode); + + public abstract boolean getTourModeIsWalkTransit(int tourMode); + + public abstract boolean getTourModeIsDriveTransit(int tourMode); + + public abstract boolean getTourModeIsPnr(int tourMode); + + public abstract boolean getTourModeIsKnr(int tourMode); + + public abstract boolean getTourModeIsSchoolBus(int tourMode); + + public abstract boolean getTourModeIsTncTransit(int tripMode); + + public abstract boolean getTourModeIsMaas(int tripMode); + + public abstract boolean getTripModeIsSovOrHov(int tripMode); + + public abstract boolean getTripModeIsWalkTransit(int tripMode); + + public abstract boolean getTripModeIsPnrTransit(int tripMode); + + public abstract boolean getTripModeIsKnrTransit(int tripMode); + + public abstract boolean getTripModeIsTransit(int tripMode); + + public abstract boolean getTripModeIsS2(int tripMode); + + public abstract boolean getTripModeIsS3(int tripMode); + + public abstract double[][] getCdap6PlusProps(); + + public abstract int getDefaultAmPeriod(); + + public abstract int getDefaultPmPeriod(); + + public abstract int getDefaultMdPeriod(); + + public abstract int getMaxTourModeIndex(); + + public abstract String getModelPeriodLabel(int period); + + public abstract int[] getSkimPeriodCombinationIndices(); + + public abstract int getSkimPeriodCombinationIndex(int startPeriod, int endPeriod); + + public abstract String getSkimMatrixPeriodString(int period); + + public abstract HashMap> getDcSizePurposeSegmentMap(); + + public abstract String[] getJtfAltLabels(); + + public abstract void setJtfAltLabels(String[] labels); + + private void createPrimaryPurposeMappings() + { + + primaryTourPurposeNameIndexMap.put(WORK_PRIMARY_PURPOSE_NAME, WORK_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(WORK_PRIMARY_PURPOSE_INDEX, WORK_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(UNIVERSITY_PRIMARY_PURPOSE_NAME, + UNIVERSITY_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(UNIVERSITY_PRIMARY_PURPOSE_INDEX, + UNIVERSITY_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(SCHOOL_PRIMARY_PURPOSE_NAME, + SCHOOL_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(SCHOOL_PRIMARY_PURPOSE_INDEX, + SCHOOL_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(ESCORT_PRIMARY_PURPOSE_NAME, + ESCORT_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(ESCORT_PRIMARY_PURPOSE_INDEX, + ESCORT_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(SHOPPING_PRIMARY_PURPOSE_NAME, + SHOPPING_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(SHOPPING_PRIMARY_PURPOSE_INDEX, + SHOPPING_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(OTH_MAINT_PRIMARY_PURPOSE_NAME, + OTH_MAINT_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(OTH_MAINT_PRIMARY_PURPOSE_INDEX, + OTH_MAINT_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(EAT_OUT_PRIMARY_PURPOSE_NAME, + EAT_OUT_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(EAT_OUT_PRIMARY_PURPOSE_INDEX, + EAT_OUT_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(VISITING_PRIMARY_PURPOSE_NAME, + VISITING_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(VISITING_PRIMARY_PURPOSE_INDEX, + VISITING_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(OTH_DISCR_PRIMARY_PURPOSE_NAME, + OTH_DISCR_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(OTH_DISCR_PRIMARY_PURPOSE_INDEX, + OTH_DISCR_PRIMARY_PURPOSE_NAME); + primaryTourPurposeNameIndexMap.put(WORK_BASED_PRIMARY_PURPOSE_NAME, + WORK_BASED_PRIMARY_PURPOSE_INDEX); + indexPrimaryTourPurposeNameMap.put(WORK_BASED_PRIMARY_PURPOSE_INDEX, + WORK_BASED_PRIMARY_PURPOSE_NAME); + + } + + /** + * @return the HashMap object that maps primary tour purpose + * names common to all CTRAMP implementations to indices (1-10). + */ + public HashMap getPrimaryPurposeNameIndexMap() + { + return primaryTourPurposeNameIndexMap; + } + + /** + * @return the HashMap object that maps indices (1-10) to + * primary tour purpose names common to all CTRAMP implementations. + */ + public HashMap getIndexPrimaryPurposeNameMap() + { + return indexPrimaryTourPurposeNameMap; + } + + /** + * @param purposeKey + * is the "purpose" name used as a key for the map to get the + * associated UEC tab number. + * @return the tab number of the UEC control file for the purpose + */ + public int getSoaUecIndexForPurpose(String purposeKey) + { + return dcSoaUecIndexMap.get(purposeKey); + } + + /** + * @param purposeKey + * is the "purpose" name used as a key for the map to get the + * associated UEC tab number. + * @return the tab number of the UEC control file for the purpose + */ + public int getDcUecIndexForPurpose(String purposeKey) + { + return dcUecIndexMap.get(purposeKey); + } + + /** + * @param purposeKey + * is the "purpose" name used as a key for the map to get the + * associated UEC tab number. + * @return the tab number of the UEC control file for the purpose + */ + public int getTourModeChoiceUecIndexForPurpose(String purposeKey) + { + return tourModeChoiceUecIndexMap.get(purposeKey); + } + + public String[] getDcModelPurposeList(String tourCategory) + { + if (tourCategory.equalsIgnoreCase(MANDATORY_CATEGORY)) return mandatoryDcModelPurposeNames; + else if (tourCategory.equalsIgnoreCase(JOINT_NON_MANDATORY_CATEGORY)) return jointDcModelPurposeNames; + else if (tourCategory.equalsIgnoreCase(INDIVIDUAL_NON_MANDATORY_CATEGORY)) return nonMandatoryDcModelPurposeNames; + else if (tourCategory.equalsIgnoreCase(AT_WORK_CATEGORY)) return atWorkDcModelPurposeNames; + else return null; + } + + public String getDcSizeCoeffPurposeFieldName() + { + return dcSizeCoeffPurposeFieldName; + } + + public String getDcSizeCoeffSegmentFieldName() + { + return this.dcSizeCoeffSegmentFieldName; + } + + public String getAtWorkEatPurposeName() + { + return AT_WORK_EAT_PURPOSE_NAME; + } + + public String[] getAtWorkSegmentNames() + { + return AT_WORK_SEGMENT_NAMES; + } + + public String getAtWorkBusinessPurposeName() + { + return AT_WORK_BUSINESS_PURPOSE_NAME; + } + + public String getAtWorkMaintPurposeName() + { + return AT_WORK_MAINT_PURPOSE_NAME; + } + + public int getAtWorkEatPurposeIndex() + { + return AT_WORK_PURPOSE_INDEX_EAT; + } + + public int getAtWorkBusinessPurposeIndex() + { + return AT_WORK_PURPOSE_INDEX_BUSINESS; + } + + public int getAtWorkMaintPurposeIndex() + { + return AT_WORK_PURPOSE_INDEX_MAINT; + } + + /** + * @param departPeriod + * is the model TOD interval for the departure period (for tour + * or trip) + * @return the skim period index associated with the departure interval + */ + public static int getSkimPeriodIndex(int departPeriod) + { + + int skimPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) skimPeriodIndex = EA_SKIM_PERIOD_INDEX; + else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM_SKIM_PERIOD_INDEX; + else if (departPeriod <= UPPER_MD) skimPeriodIndex = MD_SKIM_PERIOD_INDEX; + else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM_SKIM_PERIOD_INDEX; + else skimPeriodIndex = EV_SKIM_PERIOD_INDEX; + + return skimPeriodIndex; + + } + + /** + * @param departPeriod + * is the model TOD interval for the departure period (for tour + * or trip) + * @return the model period index associated with the departure interval + * Model periods: 0=EA, 1=AM, 2=MD, 3=PM, 4=EV + */ + public static int getModelPeriodIndex(int departPeriod) + { + + int modelPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; + else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; + else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; + else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; + else modelPeriodIndex = 4; + + return modelPeriodIndex; + + } + + private void createIndexTimePeriodMap() + { + indexTimePeriodMap = new HashMap(); + timePeriodIndexMap = new HashMap(); + + int numHours = LAST_DEPART_HOUR - FIRST_DEPART_HOUR; + int numHalfHours = numHours * 2; + + TOD_INTERVAL_LABELS = new String[numHalfHours + 1]; + + for (int i = 1; i <= numHalfHours; i++) + { + int time = ((int) (i / 2) + FIRST_DEPART_HOUR) * 100 + (i % 2) * 30; + indexTimePeriodMap.put(i, time); + timePeriodIndexMap.put(time, i); + TOD_INTERVAL_LABELS[i] = Integer.toString(time); + } + } + + public String[] getTimePeriodLabelArray() + { + return TOD_INTERVAL_LABELS; + } + + public String getTimePeriodLabel(int timePeriodIndex) + { + return TOD_INTERVAL_LABELS[timePeriodIndex]; + } + + // time argument is specified as: 500 for 5 am, 530 for 5:30 am, 1530 for + // 3:30 pm, etc. + public int getTimePeriodIndexForTime(int time) + { + return timePeriodIndexMap.get(time); + } + + public int getNumberOfTimePeriods() + { + return TOD_INTERVAL_LABELS.length - 1; + } + + public String[] getAwfAltLabels() + { + return awfAltLabels; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java new file mode 100644 index 0000000..29fc670 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java @@ -0,0 +1,115 @@ +package org.sandag.abm.ctramp; + +import com.pb.common.math.MathUtil; +import com.pb.common.model.Alternative; +import com.pb.common.model.LogitModel; +import com.pb.common.model.ModelException; + +public class MyLogit + extends LogitModel +{ + + private static final int MAX_EXP_ARGUMENT = 400; + + private double[] utilities; + private double[] util; + private double[] constant; + private String[] altName; + + public MyLogit(String n, int numberOfAlternatives) + { + super(n, numberOfAlternatives); + + utilities = new double[numberOfAlternatives]; + util = new double[numberOfAlternatives]; + constant = new double[numberOfAlternatives]; + altName = new String[numberOfAlternatives]; + + nf.setMaximumFractionDigits(8); + nf.setMinimumFractionDigits(8); + } + + /** + * Overrides the base class getUtility() method to call a method to return + * the array of exponentiated utilities, having passed to it an array of + * utilities. + * + * @return The composite utility (logsum value) of all the alternatives. + */ + public double getUtility() throws ModelException + { + + double sum = 0; + double base = 0; + + // get the array of utility values to be exponentiated from the + // alternatives + // objects. + int i = 0; + for (int alt = 0; alt < alternatives.size(); ++alt) + { + Alternative thisAlt = (Alternative) alternatives.get(alt); + if (thisAlt.isAvailable()) + { + + // assign attributes of the alternatives + util[i] = thisAlt.getUtility(); + constant[i] = thisAlt.getConstant(); + altName[i] = thisAlt.getName(); + + // if alternative has a very large negative utility, it isn't + // available + if (util[i] + constant[i] < -MAX_EXP_ARGUMENT) + { + utilities[i] = -MAX_EXP_ARGUMENT; + } else + { + utilities[i] = dispersionParameter * (util[i] + constant[i]); + setAvailability(true); + } + + i++; + } else + { + utilities[i++] = -MAX_EXP_ARGUMENT; + } + } + + // exponentiate the utilities array and save result in expUtilities. + MathUtil.expArray(utilities, expUtilities); + + // sum the exponentiated utilities + for (i = 0; i < expUtilities.length; i++) + sum += expUtilities[i]; + + // if debug, and the alternatives is elemental, log the utility values + if (debug) + { + for (i = 0; i < expUtilities.length; i++) + { + Boolean elemental = (Boolean) isElementalAlternative.get(i); + if (elemental.equals(Boolean.TRUE)) + logger.info(String.format("%-20s", altName[i]) + "\t\t" + nf.format(util[i]) + + "\t\t\t" + nf.format(constant[i]) + "\t\t\t" + + nf.format(Math.exp(utilities[i]))); + } + } + + if (isAvailable()) + { + base = (1 / dispersionParameter) * MathUtil.log(sum); + + if (Double.isNaN(base)) throw new ModelException(ModelException.INVALID_UTILITY); + + if (debug) + logger.info(String.format("%-20s", getName() + " logsum:") + "\t\t" + + nf.format(base)); + + return base; + } + + // if nothing avaiable, return a bad utilty + return -999; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java new file mode 100644 index 0000000..04a5ecc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java @@ -0,0 +1,1527 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.Random; +import java.util.ResourceBundle; + +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.IndexSort; +import com.pb.common.util.ResourceUtil; +import com.pb.common.newmodel.ChoiceModelApplication; +import org.apache.log4j.Logger; + +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import org.sandag.abm.accessibilities.NonTransitUtilities; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampDmuFactoryIf; +import org.sandag.abm.ctramp.DcSoaDMU; +import org.sandag.abm.ctramp.DestChoiceDMU; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Tour; +import org.sandag.abm.ctramp.DestinationSampleOfAlternativesModel; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import org.sandag.abm.ctramp.TourModeChoiceModel; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.visitor.VisitorTour; +public class NonMandatoryDestChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(NonMandatoryDestChoiceModel.class); + private transient Logger dcNonManLogger = Logger.getLogger("tourDcNonMan"); + private transient Logger todMcLogger = Logger.getLogger("todMcLogsum"); + + // TODO eventually remove this target + private static final String PROPERTIES_DC_UEC_FILE = "nmdc.uec.file"; + private static final String PROPERTIES_DC_UEC_FILE2 = "nmdc.uec.file2"; + private static final String PROPERTIES_DC_SOA_UEC_FILE = "nmdc.soa.uec.file"; + + private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; + + private static final String PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY = "nmdc.soa.SampleSize"; + + private static final String PROPERTIES_DC_DATA_SHEET = "nmdc.data.page"; + + private static final String PROPERTIES_DC_ESCORT_MODEL_SHEET = "nmdc.escort.model.page"; + private static final String PROPERTIES_DC_SHOP_MODEL_SHEET = "nmdc.shop.model.page"; + private static final String PROPERTIES_DC_MAINT_MODEL_SHEET = "nmdc.maint.model.page"; + private static final String PROPERTIES_DC_EATOUT_MODEL_SHEET = "nmdc.eat.model.page"; + private static final String PROPERTIES_DC_VISIT_MODEL_SHEET = "nmdc.visit.model.page"; + private static final String PROPERTIES_DC_DISCR_MODEL_SHEET = "nmdc.discr.model.page"; + + private static final String PROPERTIES_DC_SOA_ESCORT_MODEL_SHEET = "nmdc.soa.escort.model.page"; + private static final String PROPERTIES_DC_SOA_SHOP_MODEL_SHEET = "nmdc.soa.shop.model.page"; + private static final String PROPERTIES_DC_SOA_MAINT_MODEL_SHEET = "nmdc.soa.maint.model.page"; + private static final String PROPERTIES_DC_SOA_EATOUT_MODEL_SHEET = "nmdc.soa.eat.model.page"; + private static final String PROPERTIES_DC_SOA_VISIT_MODEL_SHEET = "nmdc.soa.visit.model.page"; + private static final String PROPERTIES_DC_SOA_DISCR_MODEL_SHEET = "nmdc.soa.discr.model.page"; + + private static final String PROPERTIES_DC_SAMPLE_TOD_PERIOD = "nmdc.SampleTODPeriod"; + private static final String PROPERTIES_SAMPLE_TOD_PERIOD_FILE = "nmdc.SampleTODPeriod.file"; + private boolean sampleTODPeriod = false; + private double[][] cumProbability; // by purpose, alternative: cumulative probability distribution + private int[][] outboundPeriod; // by purpose, alternative: outbound period + private int[][] returnPeriod; // by purpose, alternative: return period + + + private static final String[] TOUR_PURPOSE_NAMES = { + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME }; + + // set priority ranking for non-mandatory purposes - include 0 values for 0 + // element and mandatory purposes + private static final int[] TOUR_PURPOSE_PRIORITIES = {0, + 0, 0, 0, 1, 3, 2, 6, 4, 5 }; + + private static final String[] DC_MODEL_SHEET_KEYS = { + PROPERTIES_DC_ESCORT_MODEL_SHEET, PROPERTIES_DC_SHOP_MODEL_SHEET, + PROPERTIES_DC_MAINT_MODEL_SHEET, PROPERTIES_DC_EATOUT_MODEL_SHEET, + PROPERTIES_DC_VISIT_MODEL_SHEET, PROPERTIES_DC_DISCR_MODEL_SHEET }; + + private static final String[] DC_SOA_MODEL_SHEET_KEYS = { + PROPERTIES_DC_SOA_ESCORT_MODEL_SHEET, PROPERTIES_DC_SOA_SHOP_MODEL_SHEET, + PROPERTIES_DC_SOA_MAINT_MODEL_SHEET, PROPERTIES_DC_SOA_EATOUT_MODEL_SHEET, + PROPERTIES_DC_SOA_VISIT_MODEL_SHEET, PROPERTIES_DC_SOA_DISCR_MODEL_SHEET }; + + // all three subtour purposes use the same SOA sheet + private final int[] sizeSheetIndices = { + BuildAccessibilities.ESCORT_INDEX, BuildAccessibilities.SHOP_INDEX, + BuildAccessibilities.OTH_MAINT_INDEX, BuildAccessibilities.EATOUT_INDEX, + BuildAccessibilities.VISIT_INDEX, BuildAccessibilities.OTH_DISCR_INDEX }; + + // set default depart periods that represents each model period + private static final int EA = 1; + private static final int AM = 8; + private static final int MD = 16; + private static final int PM = 26; + private static final int EV = 36; + + private static final int[][][] PERIOD_COMBINATIONS = { + { {AM, AM}, {MD, MD}, {PM, PM}}, { {MD, MD}, {PM, PM}, {EV, EV}}, + { {AM, MD}, {MD, PM}, {PM, EV}}, { {MD, MD}, {PM, PM}, {EV, EV}}, + { {MD, MD}, {PM, PM}, {EV, EV}}, { {AM, MD}, {MD, PM}, {PM, EV}} }; + + private static final double[][] PERIOD_COMBINATION_COEFFICIENTS = { + {-1.065820, -0.871051, -1.439514}, {-0.467154, -1.411351, -2.044826}, + {-0.941865, -0.813977, -1.789714}, {-1.007316, -0.968856, -1.365375}, + {-1.081531, -1.121260, -1.093461}, {-1.258919, -1.155085, -0.913773} }; + + private ModelStructure modelStructure; + + private int[] dcModelIndices; + private HashMap purposeNameIndexMap; + HashMap nonMandatorySegmentNameIndexMap; + HashMap nonMandatorySizeSegmentNameIndexMap; + + private double[][] dcSizeArray; + + private TourModeChoiceDMU mcDmuObject; + private DestChoiceDMU dcDmuObject; + private DestChoiceTwoStageModelDMU dcDistSoaDmuObject; + private DcSoaDMU dcSoaDmuObject; + + private boolean[] needToComputeLogsum; + private double[] modeChoiceLogsums; + + private TourModeChoiceModel mcModel; + private DestinationSampleOfAlternativesModel dcSoaModel; + private ChoiceModelApplication[] dcModel; + private ChoiceModelApplication[] dcModel2; + + private boolean[] dcModel2AltsAvailable; + private int[] dcModel2AltsSample; + private int[] dcModel2SampleValues; + + private double[] mgraDistanceArray; + + private BuildAccessibilities aggAcc; + + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private DestChoiceTwoStageModel dcSoaTwoStageObject; + + private boolean useNewSoaMethod; + + private int soaSampleSize; + + private long soaRunTime; + + public NonMandatoryDestChoiceModel(HashMap propertyMap, + ModelStructure myModelStructure, BuildAccessibilities myAggAcc, + CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel myMcModel) + { + + logger.info("setting up Non-Mandatory tour destination choice model."); + + // set the model structure and the tour purpose list + this.modelStructure = myModelStructure; + this.mcModel = myMcModel; + aggAcc = myAggAcc; + + mgraManager = MgraDataManager.getInstance(); + tazs = TazDataManager.getInstance(); + + soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, + PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); + + useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, + USE_NEW_SOA_METHOD_PROPERTY_KEY); + + if (useNewSoaMethod) + dcSoaTwoStageObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); + + // create an array of ChoiceModelApplication objects for each choice + // purpose + setupDestChoiceModelArrays(propertyMap, dmuFactory); + + sampleTODPeriod = Util.getBooleanValueFromPropertyMap(propertyMap, PROPERTIES_DC_SAMPLE_TOD_PERIOD); + String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); + String diurnalFile = Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_SAMPLE_TOD_PERIOD_FILE); + diurnalFile = directory + diurnalFile; + + if(sampleTODPeriod) + readTODFile(diurnalFile); + } + + /** + * Read the TOD distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readTODFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating tour TOD probability distribution"); + + int purposes = PERIOD_COMBINATIONS.length; // start at 0 + int periods = ModelStructure.MAX_TOD_INTERVAL; // start at 1 + int periodCombinations = periods * (periods + 1) / 2; + + cumProbability = new double[purposes][periodCombinations]; + outboundPeriod = new int[purposes][periodCombinations]; + returnPeriod = new int[purposes][periodCombinations]; + + // fill up arrays + int rowCount = probabilityTable.getRowCount(); + int lastPurpose = -99; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose") - 4; //4 mandatory purposes, first non-mand purpose is escort - 4 + int outPer = (int) probabilityTable.getValueAt(row, "OutboundPeriod"); + int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); + + // continue if return period before outbound period + if (retPer < outPer) continue; + + // reset if new purpose + if (purpose != lastPurpose) + { + + // log cumulative probability just in case + if (lastPurpose != -99) + logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); + cumProb = 0; + alt = 0; + } + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + outboundPeriod[purpose][alt] = outPer; + returnPeriod[purpose][alt] = retPer; + + ++alt; + + lastPurpose = purpose; + } + + logger.info("End calculating tour TOD probability distribution"); + + } + + /** + * Calculate tour time of day for the tour. + * + * @param tour + * A tour (with purpose) + */ + public double sampleTODPeriodAndCalculateDCLogsum(Person person, Tour tour, int sampleDestMgra) + { + + Logger modelLogger = todMcLogger; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + Household household = person.getHouseholdObject(); + + if (household.getDebugChoiceModels()) + { + choiceModelDescription = String + .format("Non-Mandatory sample TOD logsum calculations for %s Location Choice", + tour.getTourPurpose()); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourId(), person.getListOfIndividualNonMandatoryTours().size()); + + } + + + double random = household.getHhRandom().nextDouble(); + int purpose = purposeNameIndexMap.get(tour.getTourPurpose()); + + int depart = -1; + int arrive = -1; + if (household.getDebugChoiceModels()) + { + logger.info("Choosing tour time of day for purpose " + + tour.getTourPurpose() + " using random number " + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + //Wu added to prevent large random number resulting in invalid choice + if (random>0.999999) { + depart = outboundPeriod[purpose][cumProbability[purpose].length-1]; + arrive = returnPeriod[purpose][cumProbability[purpose].length-1]; + break; + } + if (random < cumProbability[purpose][i]) + { + depart = outboundPeriod[purpose][i]; + arrive = returnPeriod[purpose][i]; + break; + } + } + if((depart ==-1)||(arrive==-1)){ + logger.fatal("Error: did not find outbound or return period for tour"); + logger.fatal("Depart period, arrive period = "+depart+","+arrive); + logger.fatal("Random number: "+random); + tour.logTourObject(logger,100); + throw new RuntimeException(); + } + + String periodString = modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(depart)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(arrive)); + + if (household.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose depart period " + depart + " and arrival period " + + arrive); + logger.info(""); + } + + // set the mode choice attributes needed by @variables in the UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, depart, arrive, sampleDestMgra); + + double logsum = -999; + try + { + logsum = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel+","+periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + tour.getTourPrimaryPurpose() + " tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + e.printStackTrace(); + throw new RuntimeException(e); + } + + if (household.getDebugChoiceModels()) + modelLogger.info("Mode choice logsum for sampled mgra = " + logsum); + + return logsum; + + } + + + private void setupDestChoiceModelArrays(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + String dcUecFileName = propertyMap.get(PROPERTIES_DC_UEC_FILE); + dcUecFileName = uecFileDirectory + dcUecFileName; + + String dcUecFileName2 = propertyMap.get(PROPERTIES_DC_UEC_FILE2); + dcUecFileName2 = uecFileDirectory + dcUecFileName2; + + String soaUecFileName = propertyMap.get(PROPERTIES_DC_SOA_UEC_FILE); + soaUecFileName = uecFileDirectory + soaUecFileName; + + int dcModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, + PROPERTIES_DC_DATA_SHEET); + + dcDmuObject = dmuFactory.getDestChoiceDMU(); + dcDmuObject.setAggAcc(aggAcc); + dcDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); + + if (useNewSoaMethod) + { + dcDistSoaDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); + dcDistSoaDmuObject.setAggAcc(aggAcc); + dcDistSoaDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); + } + + dcSoaDmuObject = dmuFactory.getDcSoaDMU(); + dcSoaDmuObject.setAggAcc(aggAcc); + + mcDmuObject = dmuFactory.getModeChoiceDMU(); + + int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; + needToComputeLogsum = new boolean[numLogsumIndices]; + modeChoiceLogsums = new double[numLogsumIndices]; + + // create the arrays of dc model and soa model indices + int[] uecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; + int[] soaUecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; + + purposeNameIndexMap = new HashMap(TOUR_PURPOSE_NAMES.length); + + int i = 0; + for (String purposeName : TOUR_PURPOSE_NAMES) + { + int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); + int soaUecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, + DC_SOA_MODEL_SHEET_KEYS[i]); + purposeNameIndexMap.put(purposeName, i); + uecSheetIndices[i] = uecIndex; + soaUecSheetIndices[i] = soaUecIndex; + i++; + } + + // create a lookup array to map purpose index to model index + dcModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int dcModelIndex = 0; + int dcSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, dcModelIndex); + dcModelIndices[dcSegmentIndex] = dcModelIndex++; + } else + { + dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); + } + + dcSegmentIndex++; + } + + // the size term array in aggAcc gives mgra*purpose - need an array of + // all mgras for one purpose + double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); + nonMandatorySegmentNameIndexMap = new HashMap(); + nonMandatorySizeSegmentNameIndexMap = new HashMap(); + for (int k = 0; k < TOUR_PURPOSE_NAMES.length; k++) + { + nonMandatorySegmentNameIndexMap.put(TOUR_PURPOSE_NAMES[k], k); + nonMandatorySizeSegmentNameIndexMap.put(TOUR_PURPOSE_NAMES[k], sizeSheetIndices[k]); + } + + dcSizeArray = new double[TOUR_PURPOSE_NAMES.length][aggAccDcSizeArray.length]; + for (i = 0; i < aggAccDcSizeArray.length; i++) + { + for (int m : nonMandatorySegmentNameIndexMap.values()) + { + int s = sizeSheetIndices[m]; + dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; + } + } + + dcModel = new ChoiceModelApplication[modelIndexMap.size()]; + + if (useNewSoaMethod) + { + dcModel2 = new ChoiceModelApplication[modelIndexMap.size()]; + dcModel2AltsAvailable = new boolean[soaSampleSize + 1]; + dcModel2AltsSample = new int[soaSampleSize + 1]; + dcModel2SampleValues = new int[soaSampleSize]; + } else + { + // create a sample of alternatives choice model object for use in + // selecting a sample + // of all possible destination choice alternatives. + dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFileName, soaSampleSize, + propertyMap, mgraManager, dcSizeArray, dcSoaDmuObject, soaUecSheetIndices); + } + + i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + + try + { + dcModel[i] = new ChoiceModelApplication(dcUecFileName, uecIndex, dcModelDataSheet, + propertyMap, (VariableTable) dcDmuObject); + + if (useNewSoaMethod) + { + dcModel2[i] = new ChoiceModelApplication(dcUecFileName2, uecIndex, + dcModelDataSheet, propertyMap, (VariableTable) dcDistSoaDmuObject); + } + + i++; + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up DC ChoiceModelApplication[%d] for model index=%d of %d models", + i, i, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; + } + + public void applyIndivModel(Household hh) + { + + soaRunTime = 0; + + if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); + else dcSoaModel.resetSoaRunTime(); + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + + Person[] persons = hh.getPersons(); + + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + // get the individual non-mandatory tours for this person and choose + // a destination for each. + ArrayList tourList = getPriorityOrderedTourList(p + .getListOfIndividualNonMandatoryTours()); + + int currentTourNum = 0; + for (Tour tour : tourList) + { + + if(tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE||tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE|| + tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE||tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE) + continue; + + int chosen = -1; + try + { + + int homeTaz = hh.getHhTaz(); + int origMgra = tour.getTourOrigMgra(); + + // update the MC dmuObject for this person + mcDmuObject.setHouseholdObject( hh ); + mcDmuObject.setPersonObject( p ); + mcDmuObject.setTourObject( tour ); + mcDmuObject.setDmuIndexValues( hh.getHhId(), homeTaz, origMgra, 0, hh.getDebugChoiceModels() ); + mcDmuObject.setOriginMgra(origMgra); + + // update the DC dmuObject for this person + dcDmuObject.setHouseholdObject(hh); + dcDmuObject.setPersonObject(p); + dcDmuObject.setTourObject(tour); + dcDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); + + if (useNewSoaMethod) + { + dcDistSoaDmuObject.setHouseholdObject(hh); + dcDistSoaDmuObject.setPersonObject(p); + dcDistSoaDmuObject.setTourObject(tour); + dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); + } + + // for individual non-mandatory DC, just count remaining + // individual non-mandatory tours + int toursLeftCount = tourList.size() - currentTourNum; + dcDmuObject.setToursLeftCount(toursLeftCount); + if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); + + // get the tour location alternative chosen from the sample + if (useNewSoaMethod) + { + chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); + soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); + } else + { + chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, + dcSoaDmuObject, mcDmuObject); + soaRunTime += dcSoaModel.getSoaRunTime(); + } + + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught selecting individual non-mandatory tour destination choice for hh.hhid=%d, personNum=%d, tourId=%d, purposeName=%s", + hh.getHhId(), p.getPersonNum(), tour.getTourId(), + tour.getTourPurpose())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + // set chosen values in tour object + tour.setTourDestMgra(chosen); + + currentTourNum++; + } + + } + + hh.setInmtlRandomCount(hh.getHhRandomCount()); + + } + + public void applyJointModel(Household hh) + { + + soaRunTime = 0; + + if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); + else dcSoaModel.resetSoaRunTime(); + + // if no joint non-mandatory tours, nothing to do for this household. + Tour[] jointTours = hh.getJointTourArray(); + if (jointTours == null || jointTours.length == 0) return; + + // get the individual non-mandatory tours for this person and choose a + // destination for each. + ArrayList tourList = getPriorityOrderedTourList(jointTours); + + int currentTourNum = 0; + for (Tour tour : tourList) + { + + int chosen = -1; + try + { + + int homeTaz = hh.getHhTaz(); + int origMgra = tour.getTourOrigMgra(); + + // update the MC dmuObject for this person + mcDmuObject.setHouseholdObject( hh ); + mcDmuObject.setPersonObject( null ); + mcDmuObject.setTourObject( tour ); + mcDmuObject.setDmuIndexValues( hh.getHhId(), homeTaz, origMgra, 0, hh.getDebugChoiceModels() ); + mcDmuObject.setOriginMgra(origMgra); + + // update the DC dmuObject for this person + dcDmuObject.setHouseholdObject(hh); + dcDmuObject.setPersonObject(null); + dcDmuObject.setTourObject(tour); + dcDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); + + if (useNewSoaMethod) + { + dcDistSoaDmuObject.setHouseholdObject(hh); + dcDistSoaDmuObject.setPersonObject(null); + dcDistSoaDmuObject.setTourObject(tour); + dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); + } + + // for individual non-mandatory DC, just count remaining + // individual non-mandatory tours + int toursLeftCount = tourList.size() - currentTourNum; + dcDmuObject.setToursLeftCount(toursLeftCount); + if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); + + // get the tour location alternative chosen from the sample + if (useNewSoaMethod) + { + chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); + soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); + } else + { + chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, + dcSoaDmuObject, mcDmuObject); + soaRunTime += dcSoaModel.getSoaRunTime(); + } + + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught selecting joint non-mandatory tour destination choice for hh.hhid=%d, tourId=%d, purposeName=%s", + hh.getHhId(), tour.getTourId(), tour.getTourPurpose())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + // set chosen values in tour object + tour.setTourDestMgra(chosen); + + currentTourNum++; + } + + hh.setJtlRandomCount(hh.getHhRandomCount()); + + } + + /** + * + * @return chosen mgra. + */ + private int selectLocationFromSampleOfAlternatives(Tour tour, DestChoiceDMU dcDmuObject, + DcSoaDMU dcSoaDmuObject, TourModeChoiceDMU mcDmuObject) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcNonManLogger; + + // get the Household object for the person making this non-mandatory + // tour + Person person = tour.getPersonObject(); + + // get the Household object for the person making this non-mandatory + // tour + Household household = person.getHouseholdObject(); + + // get the tour purpose name + String tourPurposeName = tour.getTourPurpose(); + int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); + + int sizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurposeName); + dcSoaDmuObject.setDestChoiceSize(dcSizeArray[sizeIndex]); + + // double[] homeMgraDistanceArray = + // mandAcc.calculateDistancesForAllMgras( household.getHhMgra() ); + mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(household.getHhMgra(), + mgraDistanceArray); + dcSoaDmuObject.setDestDistance(mgraDistanceArray); + + dcDmuObject.setDestChoiceSize(dcSizeArray[sizeIndex]); + dcDmuObject.setDestChoiceDistance(mgraDistanceArray); + + // compute the sample of alternatives set for the person + dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, tour, person, + tourPurposeName, tourPurposeIndex, household.getHhMgra()); + + // get sample of locations and correction factors for sample + int[] finalSample = dcSoaModel.getSampleOfAlternatives(); + float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); + + int m = dcModelIndices[tourPurposeIndex]; + int numAlts = dcModel[m].getNumberOfAlternatives(); + + // set the destAltsAvailable array to true for all destination choice + // alternatives for each purpose + boolean[] destAltsAvailable = new boolean[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsAvailable[k] = false; + + // set the destAltsSample array to 1 for all destination choice + // alternatives + // for each purpose + int[] destAltsSample = new int[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsSample[k] = 0; + + int[] sampleValues = new int[finalSample.length]; + + // for the destinations and sub-zones in the sample, compute mc logsums + // and + // save in DC dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 1; i < finalSample.length; i++) + { + + int destMgra = finalSample[i]; + sampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = -999; + if(sampleTODPeriod) + logsum = sampleTODPeriodAndCalculateDCLogsum(person, tour, destMgra); + else + logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); + + dcDmuObject.setMcLogsum(destMgra, logsum); + + // set sample of alternatives correction factor used in destination + // choice utility for the sampled alternative. + dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); + + // set availaibility and sample values for the purpose, dcAlt. + destAltsAvailable[finalSample[i]] = true; + destAltsSample[finalSample[i]] = 1; + + } + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" + + tourPurposeName + ", Person Num: " + person.getPersonNum() + + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + dcModel[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float modelLogsum = (float) dcModel[m].computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), + destAltsAvailable, destAltsSample); + + tour.setTourDestinationLogsum(modelLogsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (dcModel[m].getAvailabilityCount() > 0) + { + try + { + chosen = dcModel[m].getChoiceResult(rn); + } catch (Exception e) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), tour.getTourId(), + tourPurposeName)); + throw new RuntimeException(); + } + } + + // write choice model alternative info to log file + int selectedIndex = -1; + for (int j = 1; j < finalSample.length; j++) + { + if (finalSample[j] == chosen) + { + selectedIndex = j; + break; + } + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = dcModel[m].getUtilities(); + double[] probabilities = dcModel[m].getProbabilities(); + boolean[] availabilities = dcModel[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); + + double cumProb = 0.0; + for (int j = 1; j < finalSample.length; j++) + { + int k = sortedSampleValueIndices[j]; + int alt = finalSample[k]; + + if (finalSample[k] == chosen) selectedIndex = j; + + cumProb += probabilities[alt - 1]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[alt], utilities[alt - 1], probabilities[alt - 1], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", selectedIndex, chosen); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + dcModel[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + dcModel[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + dcModel[m].logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), tour.getTourId(), + tourPurposeName)); + throw new RuntimeException(); + } + + } + + return chosen; + + } + + /** + * + * @return chosen mgra. + */ + private int selectLocationFromTwoStageSampleOfAlternatives(Tour tour, + TourModeChoiceDMU mcDmuObject) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcNonManLogger; + + // get the Household object for the person making this non-mandatory + // tour + Person person = tour.getPersonObject(); + + // get the Household object for the person making this non-mandatory + // tour + Household household = person.getHouseholdObject(); + + // get the tour purpose name + String tourPurposeName = tour.getTourPurpose(); + int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); + + // get sample of locations and correction factors for sample using the + // alternate method + // for non-mandatory tour destination choice, the sizeSegmentType INdex + // and sizeSegmentIndex are the same values. + dcSoaTwoStageObject.chooseSample(mgraManager.getTaz(tour.getTourOrigMgra()), + tourPurposeIndex, tourPurposeIndex, soaSampleSize, household.getHhRandom(), + household.getDebugChoiceModels()); + int[] finalSample = dcSoaTwoStageObject.getUniqueSampleMgras(); + double[] sampleCorrectionFactors = dcSoaTwoStageObject + .getUniqueSampleMgraCorrectionFactors(); + int numUniqueAlts = dcSoaTwoStageObject.getNumberofUniqueMgrasInSample(); + + int m = dcModelIndices[tourPurposeIndex]; + int numAlts = dcModel2[m].getNumberOfAlternatives(); + + Arrays.fill(dcModel2AltsAvailable, false); + Arrays.fill(dcModel2AltsSample, 0); + Arrays.fill(dcModel2SampleValues, 999999); + + mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(household.getHhMgra(), + mgraDistanceArray); + dcDistSoaDmuObject.setMgraDistanceArray(mgraDistanceArray); + + int sizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurposeName); + dcDistSoaDmuObject.setMgraSizeArray(dcSizeArray[sizeIndex]); + + // set sample of alternatives correction factors used in destination + // choice utility for the sampled alternatives. + dcDistSoaDmuObject.setDcSoaCorrections(sampleCorrectionFactors); + + // for the destination mgras in the sample, compute mc logsums and save + // in dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 0; i < numUniqueAlts; i++) + { + + int destMgra = finalSample[i]; + dcModel2SampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = -999; + if(sampleTODPeriod) + logsum = sampleTODPeriodAndCalculateDCLogsum(person, tour, destMgra); + else + logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); + dcDistSoaDmuObject.setMcLogsum(i, logsum); + + // set availaibility and sample values for the purpose, dcAlt. + dcModel2AltsAvailable[i + 1] = true; + dcModel2AltsSample[i + 1] = 1; + + } + + dcDistSoaDmuObject.setSampleArray(dcModel2SampleValues); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" + + tourPurposeName + ", Person Num: " + person.getPersonNum() + + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + dcModel2[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float logsum = (float) dcModel2[m].computeUtilities(dcDistSoaDmuObject, dcDistSoaDmuObject.getDmuIndexValues(), + dcModel2AltsAvailable, dcModel2AltsSample); + + tour.setTourDestinationLogsum(logsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (dcModel2[m].getAvailabilityCount() > 0) + { + try + { + chosen = dcModel2[m].getChoiceResult(rn); + } catch (Exception e) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", + dcDistSoaDmuObject.getHouseholdObject().getHhId(), + dcDistSoaDmuObject.getPersonObject().getPersonNum(), + tour.getTourId(), tourPurposeName)); + throw new RuntimeException(); + } + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = dcModel2[m].getUtilities(); + double[] probabilities = dcModel2[m].getProbabilities(); + boolean[] availabilities = dcModel2[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < finalSample.length; j++) + { + int alt = finalSample[j]; + cumProb += probabilities[j]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + dcModel2[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + dcModel2[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + dcModel2[m].logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", + dcDistSoaDmuObject.getHouseholdObject().getHhId(), + dcDistSoaDmuObject.getPersonObject().getPersonNum(), + tour.getTourId(), tourPurposeName)); + throw new RuntimeException(); + } + + } + + return chosen; + + } + + private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, + int startPeriod, int endPeriod, int sampleDestMgra) + { + + t.setTourDestMgra(sampleDestMgra); + t.setTourDepartPeriod(startPeriod); + t.setTourArrivePeriod(endPeriod); + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(t); + mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), + t.getTourOrigMgra(), sampleDestMgra, household.getDebugChoiceModels()); + + mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t + .getTourOrigMgra()))); + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(sampleDestMgra))); + mcDmuObject.setOriginMgra(t.getTourOrigMgra()); + mcDmuObject.setDestMgra(t.getTourDestMgra()); + + } + + + /** + * This method calculates TOD choice logsum for the person, tour and sampled destination. + * @param person + * @param tour + * @param sampleDestMgra + * @param sampleNum + * @return The logsum. + */ + private double calculateSimpleTODChoiceLogsum(Person person, Tour tour, int sampleDestMgra, + int sampleNum) + { + + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todMcLogger; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + choiceModelDescription = String + .format("Non-Mandatory Simplified TOD logsum calculations for %s Location Choice, Sample Number %d", + tour.getTourPurpose(), sampleNum); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourId(), person.getListOfIndividualNonMandatoryTours().size()); + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + + } + + int i = 0; + int tourPurposeIndex = purposeNameIndexMap.get(tour.getTourPurpose()); + double totalExpUtility = 0.0; + for (int[] combo : PERIOD_COMBINATIONS[tourPurposeIndex]) + { + int startPeriod = combo[0]; + int endPeriod = combo[1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod, + sampleDestMgra); + + if (household.getDebugChoiceModels()) + { + modelLogger.info(""); + modelLogger.info(""); + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, tour); + } + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " " + tour.getTourPrimaryPurpose() + " tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + e.printStackTrace(); + //System.exit(-1); + throw new RuntimeException(e); + } + needToComputeLogsum[index] = false; + } + + double expUtil = Math.exp(modeChoiceLogsums[index] + + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]); + totalExpUtility += expUtil; + + if (household.getDebugChoiceModels()) + modelLogger + .info("i = " + + i + + ", purpose = " + + tourPurposeIndex + + ", " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)) + + " MCLS = " + + modeChoiceLogsums[index] + + ", ASC = " + + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i] + + ", (MCLS + ASC) = " + + (modeChoiceLogsums[index] + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]) + + ", exp(MCLS + ASC) = " + expUtil + ", cumExpUtility = " + + totalExpUtility); + + i++; + } + + double logsum = Math.log(totalExpUtility); + + if (household.getDebugChoiceModels()) + modelLogger.info("final simplified TOD logsum = " + logsum); + + return logsum; + } + + /** + * takes an ArrayList of tours + * + * @return a new ArrayList ordered by priority + */ + private ArrayList getPriorityOrderedTourList(ArrayList toursIn) + { + + int[] tourPriorities = new int[toursIn.size()]; + + int i = 0; + for (Tour tour : toursIn) + { + String purposeName = tour.getTourPurpose(); + int purposeIndex = purposeNameIndexMap.get(purposeName); + int purposePriority = TOUR_PURPOSE_PRIORITIES[purposeIndex]; + tourPriorities[i] = purposePriority; + } + + int[] sortedIndices = IndexSort.indexSort(tourPriorities); + ArrayList toursOut = new ArrayList(toursIn.size()); + + for (i = 0; i < toursIn.size(); i++) + toursOut.add(toursIn.get(sortedIndices[i])); + + return toursOut; + } + + /** + * takes an ArrayList of tours + * + * @return a new ArrayList ordered by priority + */ + private ArrayList getPriorityOrderedTourList(Tour[] toursIn) + { + + int[] tourPriorities = new int[toursIn.length]; + + int i = 0; + for (Tour tour : toursIn) + { + String purposeName = tour.getTourPurpose(); + int purposeIndex = purposeNameIndexMap.get(purposeName); + int purposePriority = TOUR_PURPOSE_PRIORITIES[purposeIndex]; + tourPriorities[i] = purposePriority; + } + + int[] sortedIndices = IndexSort.indexSort(tourPriorities); + ArrayList toursOut = new ArrayList(toursIn.length); + + for (i = 0; i < toursIn.length; i++) + toursOut.add(toursIn[sortedIndices[i]]); + + return toursOut; + } + + public void setNonMandatorySoaProbs(double[][][] soaDistProbs, double[][][] soaSizeProbs) + { + if (useNewSoaMethod) + { + dcSoaTwoStageObject.setTazDistProbs(soaDistProbs); + dcSoaTwoStageObject.setMgraSizeProbs(soaSizeProbs); + } + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); + + MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, + Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TazDataManager tazManager = TazDataManager.getInstance(propertyMap); + + BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); + if (!aggAcc.getAccessibilitiesAreBuilt()) + { + aggAcc.setupBuildAccessibilities(propertyMap, false); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + // aggAcc.buildAccessibilityComponents(propertyMap); + + boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, + CtrampApplication.READ_ACCESSIBILITIES); + if (readAccessibilities) + { + + // output data + String projectDirectory = propertyMap + .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + + } + + double[][] expConstants = aggAcc.getExpConstants(); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + + double[][][] sovExpUtilities = null; + double[][][] hovExpUtilities = null; + double[][][] nMotorExpUtilities = null; + double[][][] maasExpUtilities = null; + + NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, + hovExpUtilities, nMotorExpUtilities,maasExpUtilities); + + MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( + propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); + + HouseholdIndividualNonMandatoryTourFrequencyModel inmtfModel = new HouseholdIndividualNonMandatoryTourFrequencyModel( + propertyMap, dmuFactory, aggAcc.getAccessibilitiesTableObject(), mandAcc); + + TourModeChoiceModel inmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + "Non-Mandatory", dmuFactory, logsumHelper); + + NonMandatoryDestChoiceModel testObject = new NonMandatoryDestChoiceModel(propertyMap, + modelStructure, aggAcc, dmuFactory, inmmcModel); + + String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); + int hhServerPort = Integer.parseInt((String) propertyMap + .get("RunModel.HouseholdServerPort")); + + HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, + hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + boolean restartHhServer = false; + try + { + // possible values for the following can be none, ao, cdap, imtf, + // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, + // stf, stl + String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); + if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; + else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") + || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") + || restartModel.equalsIgnoreCase("imtf") + || restartModel.equalsIgnoreCase("imtod") + || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") + || restartModel.equalsIgnoreCase("awtod") + || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") + || restartModel.equalsIgnoreCase("jtod") + || restartModel.equalsIgnoreCase("inmtf") + || restartModel.equalsIgnoreCase("inmtl") + || restartModel.equalsIgnoreCase("inmtod") + || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) + restartHhServer = false; + } catch (MissingResourceException e) + { + restartHhServer = true; + } + + if (restartHhServer) + { + + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(0.2f, 0); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, + inputPersonFileName); + + } else + { + + householdDataManager.setHouseholdSampleRate(0.2f, 0); + householdDataManager.setDebugHhIdsFromHashmap(); + householdDataManager.setTraceHouseholdSet(); + + } + + int id = householdDataManager.getArrayIndex(1033380); + Household[] hh = householdDataManager.getHhArray(id, id); + + testObject.applyIndivModel(hh[0]); + testObject.applyJointModel(hh[0]); + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java new file mode 100644 index 0000000..fb4cfec --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java @@ -0,0 +1,1448 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To + * change this template use File | Settings | File Templates. + */ +public class NonMandatoryTourDepartureAndDurationTime + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(NonMandatoryTourDepartureAndDurationTime.class); + private transient Logger todLogger = Logger.getLogger("todLogger"); + private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); + + private static final String IMTOD_UEC_FILE_TARGET = "departTime.uec.file"; + private static final String IMTOD_UEC_DATA_TARGET = "departTime.data.page"; + private static final String IMTOD_UEC_ESCORT_MODEL_TARGET = "departTime.escort.page"; + private static final String IMTOD_UEC_SHOP_MODEL_TARGET = "departTime.shop.page"; + private static final String IMTOD_UEC_MAINT_MODEL_TARGET = "departTime.maint.page"; + private static final String IMTOD_UEC_EAT_MODEL_TARGET = "departTime.eat.page"; + private static final String IMTOD_UEC_VISIT_MODEL_TARGET = "departTime.visit.page"; + private static final String IMTOD_UEC_DISCR_MODEL_TARGET = "departTime.discr.page"; + + private static final String[] TOUR_PURPOSE_NAMES = { + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME }; + + private static final String[] DC_MODEL_SHEET_KEYS = { + IMTOD_UEC_ESCORT_MODEL_TARGET, IMTOD_UEC_SHOP_MODEL_TARGET, + IMTOD_UEC_MAINT_MODEL_TARGET, IMTOD_UEC_EAT_MODEL_TARGET, IMTOD_UEC_VISIT_MODEL_TARGET, + IMTOD_UEC_DISCR_MODEL_TARGET }; + + // process non-mandatory tours in order by priority purpose: + // 4=escort, 6=oth maint, 5=shop, 8=visiting, 9=oth discr, 7=eat out, + private static final int[] TOUR_PURPOSE_INDEX_ORDER = {4, 6, 5, 8, 9, 7}; + + private ArrayList[] purposeTourLists; + + private int[] todModelIndices; + private HashMap purposeNameIndexMap; + + private int[] tourDepartureTimeChoiceSample; + + // DMU for the UEC + private TourDepartureTimeAndDurationDMU todDmuObject; + private TourModeChoiceDMU mcDmuObject; + + // model structure to compare the .properties time of day with the UECs + private ModelStructure modelStructure; + + // private double[][] dcSizeArray; + + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private ChoiceModelApplication[] todModels; + private TourModeChoiceModel mcModel; + + private int[] altStarts; + private int[] altEnds; + + private boolean[] needToComputeLogsum; + private double[] modeChoiceLogsums; + + private int noAltChoice = 1; + + private long jointModeChoiceTime; + private long indivModeChoiceTime; + + public NonMandatoryTourDepartureAndDurationTime(HashMap propertyMap, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, + TourModeChoiceModel mcModel) + { + + // set the model structure + this.modelStructure = modelStructure; + this.mcModel = mcModel; + + logger.info("setting up Non-Mandatory time-of-day choice model."); + + setupTodChoiceModels(propertyMap, dmuFactory); + } + + private void setupTodChoiceModels(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + tazs = TazDataManager.getInstance(); + mgraManager = MgraDataManager.getInstance(); + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + String todUecFileName = propertyMap.get(IMTOD_UEC_FILE_TARGET); + todUecFileName = uecFileDirectory + todUecFileName; + + todDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); + + mcDmuObject = dmuFactory.getModeChoiceDMU(); + + int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; + needToComputeLogsum = new boolean[numLogsumIndices]; + modeChoiceLogsums = new double[numLogsumIndices]; + + // create the array of tod model indices + int[] uecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; + + purposeNameIndexMap = new HashMap(TOUR_PURPOSE_NAMES.length); + + int i = 0; + for (String purposeName : TOUR_PURPOSE_NAMES) + { + int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); + purposeNameIndexMap.put(purposeName, i); + uecSheetIndices[i] = uecIndex; + i++; + } + + // create a lookup array to map purpose index to model index + todModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int todModelIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, todModelIndex); + todModelIndices[todModelIndex] = todModelIndex++; + } else + { + todModelIndices[todModelIndex++] = modelIndexMap.get(uecIndex); + } + } + + todModels = new ChoiceModelApplication[modelIndexMap.size()]; + int todModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, + IMTOD_UEC_DATA_TARGET); + + for (int uecIndex : modelIndexMap.keySet()) + { + int modelIndex = modelIndexMap.get(uecIndex); + try + { + todModels[modelIndex] = new ChoiceModelApplication(todUecFileName, uecIndex, + todModelDataSheet, propertyMap, (VariableTable) todDmuObject); + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up NonMandatory TOD ChoiceModelApplication[%d] for modelIndex=%d, num choice models=%d", + modelIndex, modelIndex, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + // get the alternatives table from the work tod UEC. + TableDataSet altsTable = todModels[0].getUEC().getAlternativeData(); + + altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); + altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); + todDmuObject.setTodAlts(altStarts, altEnds); + + int numDepartureTimeChoiceAlternatives = todModels[0].getNumberOfAlternatives(); + tourDepartureTimeChoiceSample = new int[numDepartureTimeChoiceAlternatives + 1]; + Arrays.fill(tourDepartureTimeChoiceSample, 1); + + // allocate an array of ArrayList objects to hold tour lists by purpose + // - tour lists will be processed + // in priority purpose order. + int maxPurposeIndex = 0; + for (i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) + if (TOUR_PURPOSE_INDEX_ORDER[i] > maxPurposeIndex) + maxPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; + + purposeTourLists = new ArrayList[maxPurposeIndex + 1]; + for (i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) + { + int index = TOUR_PURPOSE_INDEX_ORDER[i]; + purposeTourLists[index] = new ArrayList(); + } + + } + + public void applyIndivModel(Household hh, boolean runTODChoice, boolean runModeChoice) + { + + indivModeChoiceTime = 0; + + Logger modelLogger = todLogger; + + // get the person objects for this household + Person[] persons = hh.getPersons(); + + if(!runTODChoice) { + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + // if no individual non-mandatory tours, nothing to do. + if (person.getListOfIndividualNonMandatoryTours().size() == 0) continue; + + // arrange the individual non-mandatory tours for this person in an + // array of ArrayLists by purpose + getPriorityOrderedTourList(person.getListOfIndividualNonMandatoryTours()); + + for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) { + int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; + for (Tour t : purposeTourLists[tourPurposeIndex]) + try { + runModeChoice(hh,person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); + + }catch(Exception e) { + String errorMessage = String + .format("Exception caught for HHID=%d, personNum=%d, individual non-mandatory mode choice, tour ArrayList index=%d.", + hh.getHhId(), person.getPersonNum(), tourPurposeIndex); + String decisionMakerLabel = String + .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), person.getPersonNum(), person.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, person); + logger.error(errorMessage, e); + throw new RuntimeException(e); + + } + } + } + return; + + } + + + + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + // if no individual non-mandatory tours, nothing to do. + if (person.getListOfIndividualNonMandatoryTours().size() == 0) continue; + + // arrange the individual non-mandatory tours for this person in an + // array of ArrayLists by purpose + getPriorityOrderedTourList(person.getListOfIndividualNonMandatoryTours()); + + // define variables to hold depart/arrive periods selected for the + // most recent tour. + // if a tour has no non-overlapping period available, set the + // periods to either the depart or arrive of the most recently + // determined + // if none has been selected yet, set to the first of last interval + int previouslySelectedDepartPeriod = -1; + int previouslySelectedArrivePeriod = -1; + + for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) + { + + int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; + + // process each individual non-mandatory tour from the list + int m = -1; + int tourPurpNum = 1; + int noWindowCountFirstTemp = 0; + int noWindowCountLastTemp = 0; + int noLaterAlternativeCountTemp = 0; + + for (Tour t : purposeTourLists[tourPurposeIndex]) + { + + //store the tour depart time and arrival time if it is an escort tour; that way mode choice + //logsums can be calculated for the tour and stored when the actual tour dep/arr period isn't chosen. + int escortTourDepartPeriod=0; + int escortTourArrivePeriod=0; + if(t.getEscortTypeOutbound()>0 || t.getEscortTypeInbound()>0){ + escortTourDepartPeriod = t.getTourDepartPeriod(); + escortTourArrivePeriod = t.getTourArrivePeriod(); + continue; + } + try + { + + // get the choice model for the tour purpose + String tourPurposeName = t.getTourPurpose(); + m = todModelIndices[purposeNameIndexMap.get(tourPurposeName)]; + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (hh.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Non-Mandatory Tour Departure Time Choice Model for: Purpose=%s", + tourPurposeName); + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, tourId=%d, num=%d of %d %s tours", + hh.getHhId(), person.getPersonNum(), + person.getPersonType(), t.getTourId(), tourPurpNum, + purposeTourLists[tourPurposeIndex].size(), + tourPurposeName); + + todModels[m].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "Individual Non-Mandatory Tour Departure Time Choice Model: Debug Statement for Household ID: " + + hh.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + + person.getPersonType() + + ", Tour Id: " + + t.getTourId() + + ", num " + + tourPurpNum + + " of " + + purposeTourLists[tourPurposeIndex].size() + + " " + + tourPurposeName + " tours."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + + } + + // set the dmu object + todDmuObject.setHousehold(hh); + todDmuObject.setPerson(person); + todDmuObject.setTour(t); + + // check for multiple tours for this person, by purpose + // set the first or second switch if multiple tours for + // person, by purpose + if (purposeTourLists[tourPurposeIndex].size() == 1) + { + // not a multiple tour pattern + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(1); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else if (purposeTourLists[tourPurposeIndex].size() > 1) + { + // Two-plus tour multiple tour pattern + if (tourPurpNum == 1) + { + // first of 2+ tours + todDmuObject.setFirstTour(1); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(tourPurpNum); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else + { + // 2nd or greater tours + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(1); + todDmuObject.setTourNumber(tourPurpNum); + // the ArrayList is 0-based, and we want the + // previous tour, so subtract 2 from tourPurpNum + // to get the right index + int otherTourEndHour = purposeTourLists[tourPurposeIndex].get( + tourPurpNum - 2).getTourArrivePeriod(); + todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); + } + } + + // set the choice availability and sample + boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows( + altStarts, altEnds); + Arrays.fill(tourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in individual non-mandatory departure time choice model for hhId=%d, personNum=%d, tour purpose index=%d, tour ArrayList index=%d.", + hh.getHhId(), person.getPersonNum(), tourPurposeIndex, + tourPurpNum - 1)); + logger.error(String + .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the individual non-mandatory tour UEC=%d.", + tourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if all time windows for this person have already been + // scheduled, choose either the first and last + // alternatives and keep track of the number of times + // this condition occurs. + int alternativeAvailable = -1; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + alternativeAvailable = a; + break; + } + } + + int chosen = -1; + int chosenStartPeriod = -1; + int chosenEndPeriod = -1; + + // alternate making the first and last periods chosen if + // no alternatives are available + if (alternativeAvailable < 0) + { + + if (noAltChoice == 1) + { + if (previouslySelectedDepartPeriod < 0) + { + chosenStartPeriod = altStarts[noAltChoice - 1]; + chosenEndPeriod = altEnds[noAltChoice - 1]; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, and no non-mandatory tour scheduled yet, depart AND arrive set to first period=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + "."); + } else + { + chosenStartPeriod = previouslySelectedArrivePeriod; + chosenEndPeriod = previouslySelectedArrivePeriod; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled non-mandatory tour=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + "."); + } + noWindowCountFirstTemp++; + noAltChoice = departureTimeChoiceAvailability.length - 1; + } else + { + if (previouslySelectedDepartPeriod < 0) + { + chosenStartPeriod = altStarts[noAltChoice - 1]; + chosenEndPeriod = altEnds[noAltChoice - 1]; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, and no non-mandatory tour scheduled yet, depart AND arrive set to last period=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + "."); + } else + { + chosenStartPeriod = previouslySelectedArrivePeriod; + chosenEndPeriod = previouslySelectedArrivePeriod; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled non-mandatory tour=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + "."); + } + noWindowCountLastTemp++; + noAltChoice = 1; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to work tour arrive period=" + + chosenEndPeriod + "."); + } + + // schedule the chosen alternative + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + previouslySelectedDepartPeriod = chosenStartPeriod; + previouslySelectedArrivePeriod = chosenEndPeriod; + + if (runModeChoice) + { + + runModeChoice( hh, person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); + } + + } else + { + + // calculate and store the mode choice logsum for + // the usual work location for this worker at the + // various + // departure time and duration alternativees + setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, + departureTimeChoiceAvailability); + + if (hh.getDebugChoiceModels()) + { + hh.logTourObject(loggingHeader, modelLogger, person, t); + } + + todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); + todDmuObject + .setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); + + float logsum = (float) todModels[m].computeUtilities(todDmuObject, + todDmuObject.getIndexValues(), departureTimeChoiceAvailability, + tourDepartureTimeChoiceSample); + + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = hh.getHhRandom(); + int randomCount = hh.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + if (todModels[m].getAvailabilityCount() > 0) + { + chosen = todModels[m].getChoiceResult(rn); + + // debug output + if (hh.getDebugChoiceModels()) + { + + double[] utilities = todModels[m].getUtilities(); + double[] probabilities = todModels[m].getProbabilities(); + boolean[] availabilities = todModels[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + + personTypeString + ", Tour Id: " + t.getTourId() + + ", Tour num: " + tourPurpNum + " of " + + purposeTourLists[tourPurposeIndex].size() + " " + + tourPurposeName + " tours."); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", + k + 1, altStarts[k], altEnds[k]); + modelLogger.info(String.format( + "%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], + probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", + chosen, altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to + // debug log file + todModels[m].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + todModels[m].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate + // model specific log file + loggingHeader = String.format("%s for %s", + choiceModelDescription, decisionMakerLabel); + todModels[m].logUECResults(modelLogger, loggingHeader); + + } + + } else + { + + // since there were no alternatives with valid + // utility, assuming previous + // tour of this type scheduled up to the last + // period, so no periods left + // for this tour. + + // TODO: do a formal check for this so we can + // still flag other reasons why there's + // no valid utility for any alternative + chosen = departureTimeChoiceAvailability.length - 1; + noLaterAlternativeCountTemp++; + + } + + // schedule the chosen alternative + chosenStartPeriod = altStarts[chosen - 1]; + chosenEndPeriod = altEnds[chosen - 1]; + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + previouslySelectedDepartPeriod = chosenStartPeriod; + previouslySelectedArrivePeriod = chosenEndPeriod; + + if (runModeChoice) + { + + runModeChoice(hh,person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); + } + + } + + } catch (Exception e) + { + String errorMessage = String + .format("Exception caught for HHID=%d, personNum=%d, individual non-mandatory Departure time choice, tour ArrayList index=%d.", + hh.getHhId(), person.getPersonNum(), tourPurpNum - 1); + String decisionMakerLabel = String + .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), person.getPersonNum(), person.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, person); + todModels[m].logUECResults(modelLogger, errorMessage); + + logger.error(errorMessage, e); + throw new RuntimeException(e); + } + + tourPurpNum++; + + } + + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String + .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), person.getPersonNum(), person.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + } + + } + + hh.setInmtodRandomCount(hh.getHhRandomCount()); + + } + + + public void runModeChoice(Household hh, Person person, Tour t, int chosenStartPeriod, int chosenEndPeriod) { + if (hh.getDebugChoiceModels()) + hh.logHouseholdObject( + "Pre Non-Mandatory Tour Mode Choice Household " + + hh.getHhId() + + ", Tour " + + t.getTourId() + + " of " + + person.getListOfIndividualNonMandatoryTours() + .size(), tourMCNonManLogger); + + // set the mode choice attributes needed by + // @variables in the UEC spreadsheets + setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, + chosenEndPeriod); + + // use the mcModel object already setup for + // computing logsums and get + // the mode choice, where the selected + // worklocation and subzone an departure time + // and duration are set + // for this work tour. + int chosenMode = mcModel.getModeChoice(mcDmuObject, + t.getTourPrimaryPurpose()); + t.setTourModeChoice(chosenMode); + + } + public void applyJointModel(Household hh,boolean runTODChoice, boolean runModeChoice) + { + + jointModeChoiceTime = 0; + + // if no joint non-mandatory tours, nothing to do for this household. + Tour[] jointTours = hh.getJointTourArray(); + if (jointTours == null || jointTours.length == 0) return; + + Logger modelLogger = todLogger; + + // arrange the joint non-mandatory tours for this househol in an array + // of ArrayLists by purpose + getPriorityOrderedTourList(jointTours); + + // process tour lists by priority purpose + if(!runTODChoice) { + for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) + { + + int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; + for (Tour t : purposeTourLists[tourPurposeIndex]) + runModeChoice(hh,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); + } + return; + } + + + // define variables to hold depart/arrive periods selected for the most + // recent tour. + // if a tour has no non-overlapping period available, set the periods to + // either the depart or arrive of the most recently determined + // if none has been selected yet, set to the first of last interval + int previouslySelectedDepartPeriod = -1; + int previouslySelectedArrivePeriod = -1; + + for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) + { + + int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; + + // process each individual non-mandatory tour from the list + int m = -1; + int tourPurpNum = 1; + int noWindowCountFirstTemp = 0; + int noWindowCountLastTemp = 0; + int noLaterAlternativeCountTemp = 0; + for (Tour t : purposeTourLists[tourPurposeIndex]) + { + + try + { + + // get the choice model for the tour purpose + String tourPurposeName = t.getTourPurpose(); + m = todModelIndices[purposeNameIndexMap.get(tourPurposeName)]; + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (hh.getDebugChoiceModels()) + { + + String personNumsInJointTour = "Person Nums: ["; + for (int n : t.getPersonNumArray()) + personNumsInJointTour += " " + n; + personNumsInJointTour += " ]"; + + choiceModelDescription = String + .format("Joint Non-Mandatory Tour Departure Time Choice Model for: Purpose=%s", + tourPurposeName); + decisionMakerLabel = String.format( + "HH=%d, tourId=%d, %s, num=%d of %d %s tours", hh.getHhId(), + t.getTourId(), personNumsInJointTour, tourPurpNum, + purposeTourLists[tourPurposeIndex].size(), tourPurposeName); + + todModels[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + // set the dmu object + todDmuObject.setHousehold(hh); + todDmuObject.setTour(t); + + // check for multiple tours for this person, by purpose + // set the first or second switch if multiple tours for + // person, by purpose + if (purposeTourLists[tourPurposeIndex].size() == 1) + { + // not a multiple tour pattern + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(1); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else if (purposeTourLists[tourPurposeIndex].size() > 1) + { + // Two-plus tour multiple tour pattern + if (tourPurpNum == 1) + { + // first of 2+ tours + todDmuObject.setFirstTour(1); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(tourPurpNum); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else + { + // 2nd or greater tours + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(1); + todDmuObject.setTourNumber(tourPurpNum); + // the ArrayList is 0-based, and we want the + // previous tour, so subtract 2 from tourPurpNum to + // get the right index + int otherTourEndHour = purposeTourLists[tourPurposeIndex].get( + tourPurpNum - 2).getTourArrivePeriod(); + todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); + } + } + + // set the choice availability and sample + boolean[] departureTimeChoiceAvailability = hh + .getAvailableJointTourTimeWindows(t, altStarts, altEnds); + Arrays.fill(tourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in joint non-mandatory departure time choice model for hhId=%d, tour purpose index=%d, tour ArrayList index=%d.", + hh.getHhId(), tourPurposeIndex, tourPurpNum - 1)); + logger.error(String + .format("length of the availability array determined by the number of alternatives set in the person schedules=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the joint non-mandatory tour UEC=%d.", + tourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if all time windows for this person have already been + // scheduled, choose either the first and last + // alternatives and keep track of the number of times this + // condition occurs. + int alternativeAvailable = -1; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + alternativeAvailable = a; + break; + } + } + + int chosen = -1; + int chosenStartPeriod = -1; + int chosenEndPeriod = -1; + + // alternate making the first and last periods chosen if no + // alternatives are available + if (alternativeAvailable < 0) + { + + if (noAltChoice == 1) + { + if (previouslySelectedDepartPeriod < 0) + { + chosenStartPeriod = altStarts[noAltChoice - 1]; + chosenEndPeriod = altEnds[noAltChoice - 1]; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, and no joint non-mandatory tour scheduled yet, depart AND arrive set to first period=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + + "."); + } else + { + chosenStartPeriod = previouslySelectedArrivePeriod; + chosenEndPeriod = previouslySelectedArrivePeriod; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled joint non-mandatory tour=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + + "."); + } + noWindowCountFirstTemp++; + noAltChoice = departureTimeChoiceAvailability.length - 1; + } else + { + if (previouslySelectedDepartPeriod < 0) + { + chosenStartPeriod = altStarts[noAltChoice - 1]; + chosenEndPeriod = altEnds[noAltChoice - 1]; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, and no joint non-mandatory tour scheduled yet, depart AND arrive set to last period=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + + "."); + } else + { + chosenStartPeriod = previouslySelectedArrivePeriod; + chosenEndPeriod = previouslySelectedArrivePeriod; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled joint non-mandatory tour=" + + chosenStartPeriod + + ", " + + chosenEndPeriod + + "."); + } + noWindowCountLastTemp++; + noAltChoice = 1; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to work tour arrive period=" + + chosenEndPeriod + "."); + } + + // schedule the chosen alternative + hh.scheduleJointTourTimeWindows(t, chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + previouslySelectedDepartPeriod = chosenStartPeriod; + previouslySelectedArrivePeriod = chosenEndPeriod; + + if (runModeChoice) + { + runModeChoice(hh,t,chosenStartPeriod,chosenEndPeriod); + + } + + } else + { + + // calculate and store the mode choice logsum for the + // usual work location for this worker at the various + // departure time and duration alternativees + setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, + departureTimeChoiceAvailability); + + if (hh.getDebugChoiceModels()) + { + for (int p = 1; p < hh.getPersons().length; p++) + { + Person pers = hh.getPersons()[p]; + hh.logTourObject(loggingHeader, modelLogger, pers, t); + } + } + + todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); + todDmuObject.setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); + + float logsum = (float) todModels[m].computeUtilities(todDmuObject, todDmuObject.getIndexValues(), + departureTimeChoiceAvailability, tourDepartureTimeChoiceSample); + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = hh.getHhRandom(); + int randomCount = hh.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + if (todModels[m].getAvailabilityCount() > 0) + { + chosen = todModels[m].getChoiceResult(rn); + + // debug output + if (hh.getDebugChoiceModels()) + { + + double[] utilities = todModels[m].getUtilities(); + double[] probabilities = todModels[m].getProbabilities(); + boolean[] availabilities = todModels[m].getAvailabilities(); + + modelLogger.info("Tour Id: " + t.getTourId() + ", Tour num: " + + tourPurpNum + " of " + + purposeTourLists[tourPurposeIndex].size() + " " + + tourPurposeName + " tours."); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", + k + 1, altStarts[k], altEnds[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", + altString, availabilities[k + 1], utilities[k], + probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, + altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug + // log file + todModels[m].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + todModels[m].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate + // model specific log file + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + todModels[m].logUECResults(modelLogger, loggingHeader); + + } + + } else + { + + // since there were no alternatives with valid + // utility, assuming previous + // tour of this type scheduled up to the last + // period, so no periods left + // for this tour. + + // TODO: do a formal check for this so we can still + // flag other reasons why there's + // no valid utility for any alternative + chosen = departureTimeChoiceAvailability.length - 1; + noLaterAlternativeCountTemp++; + + } + + // schedule the chosen alternative + chosenStartPeriod = altStarts[chosen - 1]; + chosenEndPeriod = altEnds[chosen - 1]; + hh.scheduleJointTourTimeWindows(t, chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + previouslySelectedDepartPeriod = chosenStartPeriod; + previouslySelectedArrivePeriod = chosenEndPeriod; + + if (runModeChoice) + { + + runModeChoice(hh,t,chosenStartPeriod,chosenEndPeriod); + + } + + } + + } catch (Exception e) + { + String errorMessage = String + .format("Exception caught for HHID=%d, joint non-mandatory Departure time choice, tour ArrayList index=%d.", + hh.getHhId(), tourPurpNum - 1); + String decisionMakerLabel = "Final Joint Non-Mandatory Departure Time Person Objects:"; + for (int p = 1; p < hh.getPersons().length; p++) + { + Person pers = hh.getPersons()[p]; + hh.logPersonObject(decisionMakerLabel, modelLogger, pers); + todModels[m].logUECResults(modelLogger, errorMessage); + } + + logger.error(errorMessage, e); + throw new RuntimeException(); + } + + tourPurpNum++; + + } + + if (hh.getDebugChoiceModels()) + { + for (int p = 1; p < hh.getPersons().length; p++) + { + Person pers = hh.getPersons()[p]; + String decisionMakerLabel = String + .format("Final Joint Non-Mandatory Departure Time Person Objects: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), pers.getPersonNum(), pers.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, pers); + } + } + + } + + hh.setJtodRandomCount(hh.getHhRandomCount()); + + } + + /** + * For joint tours + * @param hh + * @param t + * @param chosenStartPeriod + * @param chosenEndPeriod + */ + private void runModeChoice(Household hh, Tour t, int chosenStartPeriod, int chosenEndPeriod) { + + long check = System.nanoTime(); + + if (hh.getDebugChoiceModels()) + hh.logHouseholdObject( + "Pre Joint Non-Mandatory Tour Mode Choice Household " + + hh.getHhId() + ", Tour " + t.getTourId()+1 + " of " + + hh.getJointTourArray().length, + tourMCNonManLogger); + + // set the mode choice attributes needed by + // @variables in the UEC spreadsheets + setModeChoiceDmuAttributes(hh, null, t, chosenStartPeriod, + chosenEndPeriod); + + // use the mcModel object already setup for + // computing logsums and get + // the mode choice, where the selected + // worklocation and subzone an departure time and + // duration are set + // for this work tour. + int chosenMode = mcModel.getModeChoice(mcDmuObject, + t.getTourPrimaryPurpose()); + t.setTourModeChoice(chosenMode); + jointModeChoiceTime += (System.nanoTime() - check); + + } + + + + + + private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, + int startPeriod, int endPeriod) + { + + t.setTourDepartPeriod(startPeriod); + t.setTourArrivePeriod(endPeriod); + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(t); + mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), + t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); + + + + mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t + .getTourOrigMgra()))); + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t + .getTourDestMgra()))); + + mcDmuObject.setOriginMgra(t.getTourOrigMgra()); + mcDmuObject.setDestMgra(t.getTourDestMgra()); + + } + + private void setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Tour tour, + boolean[] altAvailable) + { + + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todLogger; + String choiceModelDescription = String.format( + "NonMandatory Tour Mode Choice Logsum calculation for %s Departure Time Choice", + tour.getTourPurpose()); + String decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person + .getPersonNum(), person.getPersonType(), tour.getTourId(), person + .getListOfWorkTours().size()); + String loggingHeader = String + .format("%s %s", choiceModelDescription, decisionMakerLabel); + + for (int a = 1; a <= altStarts.length; a++) + { + + // if the depart/arrive alternative is unavailable, no need to check + // to see if a logsum has been calculated + if (!altAvailable[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " work tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + throw new RuntimeException(e); + } + needToComputeLogsum[index] = false; + } + + } + + todDmuObject.setModeChoiceLogsums(modeChoiceLogsums); + + mcDmuObject.getTourObject().setTourDepartPeriod(0); + mcDmuObject.getTourObject().setTourArrivePeriod(0); + } + + /** + * takes an ArrayList of individual non-mandatory tours creates an array of + * ArrayLists of tours by purpose + */ + private void getPriorityOrderedTourList(ArrayList toursIn) + { + + // clear the ArrayLists + for (int i = 0; i < purposeTourLists.length; i++) + { + if (purposeTourLists[i] != null) purposeTourLists[i].clear(); + } + + // go through the list of non-mandatory tours, and put each into an + // array of ArrayLists, by purpose. + for (Tour tour : toursIn) + { + int purposeIndex = tour.getTourPrimaryPurposeIndex(); + purposeTourLists[purposeIndex].add(tour); + } + + } + + /** + * takes an array of joint non-mandatory tours creates an array of + * ArrayLists of tours by purpose + */ + private void getPriorityOrderedTourList(Tour[] toursIn) + { + + // clear the ArrayLists + for (int i = 0; i < purposeTourLists.length; i++) + { + if (purposeTourLists[i] != null) purposeTourLists[i].clear(); + } + + // go through the list of non-mandatory tours, and put each into an + // array of ArrayLists, by purpose. + for (Tour tour : toursIn) + { + int purposeIndex = tour.getTourPrimaryPurposeIndex(); + purposeTourLists[purposeIndex].add(tour); + } + + } + + public long getJointModeChoiceTime() + { + return jointModeChoiceTime; + } + + public long getIndivModeChoiceTime() + { + return indivModeChoiceTime; + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); + + MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, + Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TazDataManager tazManager = TazDataManager.getInstance(propertyMap); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + + String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); + int hhServerPort = Integer.parseInt((String) propertyMap + .get("RunModel.HouseholdServerPort")); + + HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, + hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + boolean restartHhServer = false; + try + { + // possible values for the following can be none, ao, cdap, imtf, + // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, + // stf, stl + String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); + if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; + else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") + || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") + || restartModel.equalsIgnoreCase("imtf") + || restartModel.equalsIgnoreCase("imtod") + || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") + || restartModel.equalsIgnoreCase("awtod") + || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") + || restartModel.equalsIgnoreCase("jtod") + || restartModel.equalsIgnoreCase("inmtf") + || restartModel.equalsIgnoreCase("inmtl") + || restartModel.equalsIgnoreCase("inmtod") + || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) + restartHhServer = false; + } catch (MissingResourceException e) + { + restartHhServer = true; + } + + if (restartHhServer) + { + + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(1.0f, 0); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, + inputPersonFileName); + + } else + { + + householdDataManager.setHouseholdSampleRate(1.0f, 0); + householdDataManager.setDebugHhIdsFromHashmap(); + householdDataManager.setTraceHouseholdSet(); + + } + + // int id = householdDataManager.getArrayIndex( 1033380 ); + // int id = householdDataManager.getArrayIndex( 1033331 ); + int id = householdDataManager.getArrayIndex(423804); + Household[] hh = householdDataManager.getHhArray(id, id); + + TourModeChoiceModel inmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, dmuFactory, logsumHelper); + + NonMandatoryTourDepartureAndDurationTime testObject = new NonMandatoryTourDepartureAndDurationTime( + propertyMap, modelStructure, dmuFactory, inmmcModel); + + testObject.applyIndivModel(hh[0], true, true); + testObject.applyJointModel(hh[0], true, true); + + /** + * used this block of code to test for typos and implemented dmu + * methiods in the TOD choice UECs + * + * String uecFileDirectory = propertyMap.get( + * CtrampApplication.PROPERTIES_UEC_PATH ); String todUecFileName = + * propertyMap.get( IMTOD_UEC_FILE_TARGET ); todUecFileName = + * uecFileDirectory + todUecFileName; + * + * ModelStructure modelStructure = new SandagModelStructure(); + * SandagCtrampDmuFactory dmuFactory = new + * SandagCtrampDmuFactory(modelStructure); + * TourDepartureTimeAndDurationDMU todDmuObject = + * dmuFactory.getTourDepartureTimeAndDurationDMU(); + * + * int[] indices = { 0, 1, 2, 3, 4, 5 }; for (int i : indices ) { int + * uecIndex = i + 4; File uecFile = new File(todUecFileName); + * UtilityExpressionCalculator uec = new + * UtilityExpressionCalculator(uecFile, uecIndex, 0, propertyMap, + * (VariableTable) todDmuObject); } + */ + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java new file mode 100644 index 0000000..8cbfa08 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java @@ -0,0 +1,332 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * @author crf
+ * Started: Apr 14, 2009 1:34:03 PM + */ +public class ParkingChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(ParkingChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + + private int personType; + private int activityIntervals; + private int destPurpose; + private double reimbPct; + + private double[] distancesOrigAlt; + private double[] distancesAltDest; + + private double[] altParkingCostsM; + private int[] altMstallsoth; + private int[] altMstallssam; + private float[] altMparkcost; + private int[] altDstallsoth; + private int[] altDstallssam; + private float[] altDparkcost; + private int[] altHstallsoth; + private int[] altHstallssam; + private float[] altHparkcost; + private int[] altNumfreehrs; + + private int[] parkAreaMgras; + private int[] altMgraIndices; + + public ParkingChoiceDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int hhId, int origMgra, int destMgra, boolean hhDebug) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setOriginZone(origMgra); + dmuIndex.setDestZone(destMgra); + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + + if (hhDebug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug Parking Choice UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public void setPersonType(int value) + { + personType = value; + } + + public void setActivityIntervals(int value) + { + activityIntervals = value; + } + + public void setDestPurpose(int value) + { + destPurpose = value; + } + + public void setReimbPct(double value) + { + reimbPct = value; + } + + /** + * @param mgras + * is the array of MGRAs in parking area from + * "plc.alts.corresp.file". This is a 0-based array + */ + public void setParkAreaMgraArray(int[] mgras) + { + parkAreaMgras = mgras; + } + + /** + * @param indices + * is an array of indices representing this person's park + * location choice sample. the index value in this array points + * to the parkAreaMgras element, and the corresponding mgra + * value. This is a 0-based array + */ + public void setSampleIndicesArray(int[] indices) + { + altMgraIndices = indices; + } + + public void setDistancesOrigAlt(double[] distances) + { + distancesOrigAlt = distances; + } + + public void setDistancesAltDest(double[] distances) + { + distancesAltDest = distances; + } + + public void setParkingCostsM(double[] values) + { + altParkingCostsM = values; + } + + public void setMstallsoth(int[] values) + { + altMstallsoth = values; + } + + public void setMstallssam(int[] values) + { + altMstallssam = values; + } + + public void setMparkCost(float[] values) + { + altMparkcost = values; + } + + public void setDstallsoth(int[] values) + { + altDstallsoth = values; + } + + public void setDstallssam(int[] values) + { + altDstallssam = values; + } + + public void setDparkCost(float[] values) + { + altDparkcost = values; + } + + public void setHstallsoth(int[] values) + { + altHstallsoth = values; + } + + public void setHstallssam(int[] values) + { + altHstallssam = values; + } + + public void setHparkCost(float[] values) + { + altHparkcost = values; + } + + public void setNumfreehrs(int[] values) + { + altNumfreehrs = values; + } + + public int getPersonType() + { + return personType; + } + + public int getActivityIntervals() + { + return activityIntervals; + } + + public int getTripDestPurpose() + { + return destPurpose; + } + + public double getReimbPct() + { + return reimbPct; + } + + /** + * @param alt + * is the index value in the alternatives array (0,...,num alts). + * @return the distance between the trip origin mgra and the alternative + * park mgra. + */ + public double getDistanceTripOrigToParkAlt(int alt) + { + return distancesOrigAlt[alt]; + } + + /** + * @param alt + * is the index value in the alternatives array (0,...,num alts). + * @return the distance between the alternative park mgra and the trip + * destination mgra. + */ + public double getDistanceTripDestFromParkAlt(int alt) + { + return distancesAltDest[alt]; + } + + /** + * @param alt + * is the index value in the alternatives array (0,...,num alts). + * @return the cost for this person to park at the alternative park mgra. + */ + public double getLsWgtAvgCostM(int alt) + { + return altParkingCostsM[alt]; + } + + public int getMstallsoth(int alt) + { + return altMstallsoth[alt]; + } + + public int getMstallssam(int alt) + { + return altMstallssam[alt]; + } + + public float getMparkcost(int alt) + { + return altMparkcost[alt]; + } + + public int getDstallsoth(int alt) + { + return altDstallsoth[alt]; + } + + public int getDstallssam(int alt) + { + return altDstallssam[alt]; + } + + public float getDparkcost(int alt) + { + return altDparkcost[alt]; + } + + public int getHstallsoth(int alt) + { + return altHstallsoth[alt]; + } + + public int getHstallssam(int alt) + { + return altHstallssam[alt]; + } + + public float getHparkcost(int alt) + { + return altHparkcost[alt]; + } + + public int getNumfreehrs(int alt) + { + return altNumfreehrs[alt]; + } + + /** + * @return 1 if the altMgra attribute that was set equals the trip + * destination + */ + public int getDestSameAsParkAlt(int alt) + { + int index = altMgraIndices[alt]; + int altMgra = parkAreaMgras[index]; + return altMgra == dmuIndex.getDestZone() ? 1 : 0; + } + + /** + * @return the altMgra attribute for this alternative + */ + public int getParkMgraAlt(int alt) + { + int index = altMgraIndices[alt]; + int altMgra = parkAreaMgras[index]; + return altMgra; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java new file mode 100644 index 0000000..cde7ed7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java @@ -0,0 +1,256 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * @author crf
+ * Started: Apr 14, 2009 11:09:58 AM + */ +public class ParkingProvisionChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(ParkingProvisionChoiceDMU.class); + + protected HashMap methodIndexMap; + + private Household hh; + private Person person; + private IndexValues dmuIndex; + + private int mgraParkArea; + private int numFreeHours; + private int mstallsoth; + private int mstallssam; + private float mparkcost; + private int dstallsoth; + private int dstallssam; + private float dparkcost; + private int hstallsoth; + private int hstallssam; + private float hparkcost; + + private double lsWgtAvgCostM; + private double lsWgtAvgCostD; + private double lsWgtAvgCostH; + + public ParkingProvisionChoiceDMU() + { + dmuIndex = new IndexValues(); + } + + /** need to set hh and home taz before using **/ + public void setPersonObject(Person person) + { + hh = person.getHouseholdObject(); + this.person = person; + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug Free Parking UEC"); + } + } + + public void setMgraParkArea(int value) + { + mgraParkArea = value; + } + + public void setNumFreeHours(int value) + { + numFreeHours = value; + } + + public void setLsWgtAvgCostM(double cost) + { + lsWgtAvgCostM = cost; + } + + public void setLsWgtAvgCostD(double cost) + { + lsWgtAvgCostD = cost; + } + + public void setLsWgtAvgCostH(double cost) + { + lsWgtAvgCostH = cost; + } + + public void setMStallsOth(int value) + { + mstallsoth = value; + } + + public void setMStallsSam(int value) + { + mstallssam = value; + } + + public void setMParkCost(float value) + { + mparkcost = value; + } + + public void setDStallsOth(int value) + { + dstallsoth = value; + } + + public void setDStallsSam(int value) + { + dstallssam = value; + } + + public void setDParkCost(float value) + { + dparkcost = value; + } + + public void setHStallsOth(int value) + { + hstallsoth = value; + } + + public void setHStallsSam(int value) + { + hstallssam = value; + } + + public void setHParkCost(float value) + { + hparkcost = value; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /* dmu @ functions */ + + public int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + public double getLsWgtAvgCostM() + { + return lsWgtAvgCostM; + } + + public double getLsWgtAvgCostD() + { + return lsWgtAvgCostD; + } + + public double getLsWgtAvgCostH() + { + return lsWgtAvgCostH; + } + + public int getMgraParkArea() + { + return mgraParkArea; + } + + public int getNumFreeHours() + { + return numFreeHours; + } + + public int getMStallsOth() + { + return mstallsoth; + } + + public int getMStallsSam() + { + return mstallssam; + } + + public float getMParkCost() + { + return mparkcost; + } + + public int getDStallsOth() + { + return dstallsoth; + } + + public int getDStallsSam() + { + return dstallssam; + } + + public float getDParkCost() + { + return dparkcost; + } + + public int getHStallsOth() + { + return hstallsoth; + } + + public int getHStallsSam() + { + return hstallssam; + } + + public float getHParkCost() + { + return hparkcost; + } + + public int getWorkLocMgra() + { + return person.getWorkLocation(); + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java new file mode 100644 index 0000000..e283792 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java @@ -0,0 +1,216 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class ParkingProvisionModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("fp"); + + private static final String FP_CONTROL_FILE_TARGET = "fp.uec.file"; + private static final String FP_DATA_SHEET_TARGET = "fp.data.page"; + private static final String FP_MODEL_SHEET_TARGET = "fp.model.page"; + + public static final int FP_MODEL_NO_REIMBURSEMENT_CHOICE = -1; + public static final int FP_MODEL_FREE_ALT = 1; + public static final int FP_MODEL_PAY_ALT = 2; + public static final int FP_MODEL_REIMB_ALT = 3; + + private static final String REIMBURSEMENT_MEAN = "park.cost.reimb.mean"; + private static final String REIMBURSEMENT_STD_DEV = "park.cost.reimb.std.dev"; + + private MgraDataManager mgraManager; + + private double meanReimb; + private double stdDevReimb; + + private int[] mgraParkArea; + private int[] numfreehrs; + private int[] hstallsoth; + private int[] hstallssam; + private float[] hparkcost; + private int[] dstallsoth; + private int[] dstallssam; + private float[] dparkcost; + private int[] mstallsoth; + private int[] mstallssam; + private float[] mparkcost; + + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + + private ChoiceModelApplication fpModel; + private ParkingProvisionChoiceDMU fpDmuObject; + + public ParkingProvisionModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + mgraManager = MgraDataManager.getInstance(propertyMap); + setupFreeParkingChoiceModelApplication(propertyMap, dmuFactory); + } + + private void setupFreeParkingChoiceModelApplication(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + logger.info("setting up free parking choice model."); + + // locate the free parking UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String fpUecFile = uecFileDirectory + propertyMap.get(FP_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, FP_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, FP_MODEL_SHEET_TARGET); + + // create the auto ownership choice model DMU object. + fpDmuObject = dmuFactory.getFreeParkingChoiceDMU(); + + // create the auto ownership choice model object + fpModel = new ChoiceModelApplication(fpUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) fpDmuObject); + + meanReimb = Float.parseFloat(propertyMap.get(REIMBURSEMENT_MEAN)); + stdDevReimb = Float.parseFloat(propertyMap.get(REIMBURSEMENT_STD_DEV)); + + mgraParkArea = mgraManager.getMgraParkAreas(); + numfreehrs = mgraManager.getNumFreeHours(); + lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); + lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); + lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); + mstallsoth = mgraManager.getMStallsOth(); + mstallssam = mgraManager.getMStallsSam(); + mparkcost = mgraManager.getMParkCost(); + dstallsoth = mgraManager.getDStallsOth(); + dstallssam = mgraManager.getDStallsSam(); + dparkcost = mgraManager.getDParkCost(); + hstallsoth = mgraManager.getHStallsOth(); + hstallssam = mgraManager.getHStallsSam(); + hparkcost = mgraManager.getHParkCost(); + + } + + public void applyModel(Household hhObject) + { + + Random hhRandom = hhObject.getHhRandom(); + + // person array is 1-based + Person[] person = hhObject.getPersons(); + for (int i = 1; i < person.length; i++) + { + + int workLoc = person[i].getWorkLocation(); + if (workLoc == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + + person[i].setFreeParkingAvailableResult(FP_MODEL_NO_REIMBURSEMENT_CHOICE); + person[i].setParkingReimbursement(FP_MODEL_NO_REIMBURSEMENT_CHOICE); + + } else if (workLoc > 0 && mgraParkArea[workLoc] == MgraDataManager.PARK_AREA_ONE) + { + + double randomNumber = hhRandom.nextDouble(); + int chosen = getParkingChoice(person[i], randomNumber); + person[i].setFreeParkingAvailableResult(chosen); + + if (chosen == FP_MODEL_REIMB_ALT) + { + double logReimbPct = meanReimb + hhRandom.nextGaussian() * stdDevReimb; + person[i].setParkingReimbursement(Math.exp(logReimbPct)); + } else if (chosen == FP_MODEL_FREE_ALT) + { + person[i].setParkingReimbursement(0.0); + } else if (chosen == FP_MODEL_PAY_ALT) + { + person[i].setParkingReimbursement(0.0); + } + + } else + { + + person[i].setFreeParkingAvailableResult(FP_MODEL_NO_REIMBURSEMENT_CHOICE); + person[i].setParkingReimbursement(0.0); + + } + } + + hhObject.setFpRandomCount(hhObject.getHhRandomCount()); + } + + private int getParkingChoice(Person personObj, double randomNumber) + { + + // get the corresponding household object + Household hhObj = personObj.getHouseholdObject(); + fpDmuObject.setPersonObject(personObj); + + fpDmuObject.setMgraParkArea(mgraParkArea[personObj.getWorkLocation()]); + fpDmuObject.setNumFreeHours(numfreehrs[personObj.getWorkLocation()]); + fpDmuObject.setLsWgtAvgCostM(lsWgtAvgCostM[personObj.getWorkLocation()]); + fpDmuObject.setLsWgtAvgCostD(lsWgtAvgCostD[personObj.getWorkLocation()]); + fpDmuObject.setLsWgtAvgCostH(lsWgtAvgCostH[personObj.getWorkLocation()]); + fpDmuObject.setMStallsOth(mstallsoth[personObj.getWorkLocation()]); + fpDmuObject.setMStallsSam(mstallssam[personObj.getWorkLocation()]); + fpDmuObject.setMParkCost(mparkcost[personObj.getWorkLocation()]); + fpDmuObject.setDStallsSam(dstallssam[personObj.getWorkLocation()]); + fpDmuObject.setDStallsOth(dstallsoth[personObj.getWorkLocation()]); + fpDmuObject.setDParkCost(dparkcost[personObj.getWorkLocation()]); + fpDmuObject.setHStallsOth(hstallsoth[personObj.getWorkLocation()]); + fpDmuObject.setHStallsSam(hstallssam[personObj.getWorkLocation()]); + fpDmuObject.setHParkCost(hparkcost[personObj.getWorkLocation()]); + + // set the zone and dest attributes to the person's work location + fpDmuObject.setDmuIndexValues(hhObj.getHhId(), personObj.getWorkLocation(), + hhObj.getHhTaz(), personObj.getWorkLocation()); + + // compute utilities and choose auto ownership alternative. + float logsum = (float) fpModel.computeUtilities(fpDmuObject, fpDmuObject.getDmuIndexValues()); + personObj.setParkingProvisionLogsum(logsum); + + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + if (fpModel.getAvailabilityCount() > 0) + { + try { + chosenAlt = fpModel.getChoiceResult(randomNumber); + }catch(Exception e) { + + logger.fatal("Error trying to get parking location for HHID="+hhObj.getHhId()+", PERSID="+ + personObj.getPersonId() +" Destination MGRA="+personObj.getWorkLocation()); + throw new RuntimeException(e); + } + } else + { + String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), + personObj.getPersonId()); + String errorMessage = String + .format("Exception caught for %s, no available free parking alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + fpModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + // write choice model alternative info to log file + if (hhObj.getDebugChoiceModels()) + { + String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), + personObj.getPersonId()); + fpModel.logAlternativesInfo("Free parking Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d with rn %.8f", + "Free parking Choice", decisionMaker, chosenAlt, randomNumber)); + fpModel.logUECResults(logger, decisionMaker); + } + + return chosenAlt; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java new file mode 100644 index 0000000..9ce7486 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java @@ -0,0 +1,2032 @@ +package org.sandag.abm.ctramp; + +import java.util.ArrayList; +import org.apache.log4j.Logger; + +public class Person + implements java.io.Serializable +{ + + // 8 am default departure period + public static final int DEFAULT_MANDATORY_START_PERIOD = 8; + // 10 am default departure period + public static final int DEFAULT_NON_MANDATORY_START_PERIOD = 12; + // 12 pm default departure period + public static final int DEFAULT_AT_WORK_SUBTOUR_START_PERIOD = 16; + // 5 pm default arrival period + public static final int DEFAULT_MANDATORY_END_PERIOD = 26; + // 3 pm default arrival period + public static final int DEFAULT_NON_MANDATORY_END_PERIOD = 22; + // 2 pm default arrival period + public static final int DEFAULT_AT_WORK_SUBTOUR_END_PERIOD = 20; + + public static final int MIN_ADULT_AGE = 19; + public static final int MIN_STUDENT_AGE = 5; + + + // person type strings used for data summaries + public static final String PERSON_TYPE_FULL_TIME_WORKER_NAME = "Full-time worker"; + public static final String PERSON_TYPE_PART_TIME_WORKER_NAME = "Part-time worker"; + public static final String PERSON_TYPE_UNIVERSITY_STUDENT_NAME = "University student"; + public static final String PERSON_TYPE_NON_WORKER_NAME = "Non-worker"; + public static final String PERSON_TYPE_RETIRED_NAME = "Retired"; + public static final String PERSON_TYPE_STUDENT_DRIVING_NAME = "Student of driving age"; + public static final String PERSON_TYPE_STUDENT_NON_DRIVING_NAME = "Student of non-driving age"; + public static final String PERSON_TYPE_PRE_SCHOOL_CHILD_NAME = "Child too young for school"; + public static final int PERSON_TYPE_FULL_TIME_WORKER_INDEX = 1; + public static final int PERSON_TYPE_PART_TIME_WORKER_INDEX = 2; + public static final int PERSON_TYPE_UNIVERSITY_STUDENT_INDEX = 3; + public static final int PERSON_TYPE_NON_WORKER_INDEX = 4; + public static final int PERSON_TYPE_RETIRED_INDEX = 5; + public static final int PERSON_TYPE_STUDENT_DRIVING_INDEX = 6; + public static final int PERSON_TYPE_STUDENT_NON_DRIVING_INDEX = 7; + public static final int PERSON_TYPE_PRE_SCHOOL_CHILD_INDEX = 8; + public static final int MALE_INDEX = 1; + public static final int FEMALE_INDEX = 2; + public static final String[] PERSON_TYPE_NAME_ARRAY = { + PERSON_TYPE_FULL_TIME_WORKER_NAME, PERSON_TYPE_PART_TIME_WORKER_NAME, + PERSON_TYPE_UNIVERSITY_STUDENT_NAME, PERSON_TYPE_NON_WORKER_NAME, + PERSON_TYPE_RETIRED_NAME, PERSON_TYPE_STUDENT_DRIVING_NAME, + PERSON_TYPE_STUDENT_NON_DRIVING_NAME, PERSON_TYPE_PRE_SCHOOL_CHILD_NAME}; + + // Employment category (1-employed FT, 2-employed PT, 3-not employed, + // 4-under age + // 16) + // Student category (1 - student in grade or high school; 2 - student in + // college + // or higher; 3 - not a student) + + public static final String EMPLOYMENT_CATEGORY_FULL_TIME_WORKER_NAME = "Full-time worker"; + public static final String EMPLOYMENT_CATEGORY_PART_TIME_WORKER_NAME = "Part-time worker"; + public static final String EMPLOYMENT_CATEGORY_NOT_EMPLOYED_NAME = "Not employed"; + public static final String EMPLOYMENT_CATEGORY_UNDER_AGE_16_NAME = "Under age 16"; + + public static final String[] EMPLOYMENT_CATEGORY_NAME_ARRAY = { + EMPLOYMENT_CATEGORY_FULL_TIME_WORKER_NAME, EMPLOYMENT_CATEGORY_PART_TIME_WORKER_NAME, + EMPLOYMENT_CATEGORY_NOT_EMPLOYED_NAME, EMPLOYMENT_CATEGORY_UNDER_AGE_16_NAME}; + + public static final String STUDENT_CATEGORY_GRADE_OR_HIGH_SCHOOL_NAME = "Grade or high school"; + public static final String STUDENT_CATEGORY_COLLEGE_OR_HIGHER_NAME = "College or higher"; + public static final String STUDENT_CATEGORY_NOT_STUDENT_NAME = "Not student"; + + public static final String[] STUDENT_CATEGORY_NAME_ARRAY = { + STUDENT_CATEGORY_GRADE_OR_HIGH_SCHOOL_NAME, STUDENT_CATEGORY_COLLEGE_OR_HIGHER_NAME, + STUDENT_CATEGORY_NOT_STUDENT_NAME }; + + private Household hhObj; + + private int persNum; + private int persId; + private short persAge; + private short persGender; + private short persPecasOccup; + private short persActivityCode; + private short persEmploymentCategory; + private short persStudentCategory; + private short personType; + private boolean gradeSchool; + private boolean highSchool; + private boolean highSchoolGraduate; + private boolean hasBachelors; + + // individual value-of-time in $/hr + private float persValueOfTime; + + private int workLocation; + private int workLocSegmentIndex; + private float workLocDistance; + private float workLocLogsum; + private int schoolLoc; + private int schoolLocSegmentIndex; + private float schoolLocDistance; + private float schoolLocLogsum; + + private float timeFactorWork; + private float timeFactorNonWork; + + private short freeParkingAvailable; + private short internalExternalTripChoice = 1; + private float reimbursePercent; + + private float worksFromHomeLogsum; + private float parkingProvisionLogsum; + private float telecommuteLogsum; + private float ieLogsum; + private float cdapLogsum; + private float imtfLogsum; + private float inmtfLogsum; + + private String cdapActivity; + private short imtfChoice; + private short inmtfChoice; + + private int maxAdultOverlaps; + private int maxChildOverlaps; + + private short telecommuteChoice; + + private ArrayList workTourArrayList; + private ArrayList schoolTourArrayList; + private ArrayList indNonManTourArrayList; + private ArrayList atWorkSubtourArrayList; + + // private Scheduler scheduler; + // windows[] is 1s based - indexed from 1 to number of intervals. + private short[] windows; + + private int windowBeforeFirstMandJointTour; + private int windowBetweenFirstLastMandJointTour; + private int windowAfterLastMandJointTour; + + private ModelStructure modelStructure; + + public Person(Household hhObj, int persNum, ModelStructure modelStructure) + { + this.hhObj = hhObj; + this.persNum = persNum; + this.workTourArrayList = new ArrayList(); + this.schoolTourArrayList = new ArrayList(); + this.indNonManTourArrayList = new ArrayList(); + this.atWorkSubtourArrayList = new ArrayList(); + this.modelStructure = modelStructure; + + initializeWindows(); + + freeParkingAvailable = ParkingProvisionModel.FP_MODEL_REIMB_ALT; + reimbursePercent = 0.43f; + } + + public Household getHouseholdObject() + { + return hhObj; + } + + public ArrayList getListOfWorkTours() + { + return workTourArrayList; + } + + public ArrayList getListOfSchoolTours() + { + return schoolTourArrayList; + } + + public ArrayList getListOfIndividualNonMandatoryTours() + { + return indNonManTourArrayList; + } + + public ArrayList getListOfAtWorkSubtours() + { + return atWorkSubtourArrayList; + } + + public short[] getTimeWindows() + { + return windows; + } + + public String getTimePeriodLabel(int windowIndex) + { + return modelStructure.getTimePeriodLabel(windowIndex); + } + + public void initializeWindows() + { + windows = new short[modelStructure.getNumberOfTimePeriods() + 1]; + } + + public void resetTimeWindow(int startPeriod, int endPeriod) + { + for (int i = startPeriod; i <= endPeriod; i++) + { + windows[i] = 0; + } + } + + public void resetTimeWindow() + { + for (int i = 0; i < windows.length; i++) + { + windows[i] = 0; + } + } + + /** + * code the time window array for this tour being scheduled. 0: unscheduled, + * 1: scheduled, middle of tour, 2: scheduled, start of tour, 3: scheduled, + * end of tour, 4: scheduled, end of previous tour, start of current tour or + * end of current tour, start of subsequent tour; or current tour start/end + * same period. + * + * @param start + * is the departure period index for the tour + * @param end + * is the arrival period index for the tour + */ + public void scheduleWindow(int start, int end) + { + + /* + * This is the logic used in ARC/MTC, but for SANDAG, we don't allow + * overlapping tours + * + * + * if (start == end) { windows[start] = 4; } else { if (windows[start] + * == 3) windows[start] = 4; else if (windows[start] == 0) + * windows[start] = 2; + * + * if (windows[end] == 2) windows[end] = 4; else if (windows[end] == 0) + * windows[end] = 3; } + * + * for (int h = start + 1; h < end; h++) { windows[h] = 1; } + */ + + for (int h = start; h <= end; h++) + { + windows[h] = 1; + } + + } + + public boolean[] getAvailableTimeWindows(int[] altStarts, int[] altEnds) + { + + // availability array is used by UEC based choice model, which uses + // 1-based + // indexing + boolean[] availability = new boolean[altStarts.length + 1]; + + for (int i = 1; i <= altStarts.length; i++) + { + int start = altStarts[i - 1]; + int end = altEnds[i - 1]; + availability[i] = isWindowAvailable(start, end); + } + + return availability; + } + + public boolean isWindowAvailable(int start, int end) + { + + /* + * This is the logic used in ARC/MTC, but for SANDAG, we don't allow + * overlapping tours + * + * + * // check start period, if window is 0, it is unscheduled; // if + * window is 3, it is the last period of another tour, and available // + * as the first period of this tour. if (windows[start] == 1) return + * false; else if (windows[start] == 2 && start != end) return false; + * + * // check end period, if window is 0, it is unscheduled; // if window + * is 2, it is the first period of another tour, and available // as the + * last period of this tour. if (windows[end] == 1) return false; else + * if (windows[end] == 3 && start != end) return false; + * + * // the alternative is available if start and end are available, and + * all periods // from start+1,...,end-1 are available. for (int h = + * start + 1; h < end; h++) { if (windows[h] > 0) return false; } + * + * return true; + */ + + // the alternative is available if all intervals between start and end, + // inclusive, are available + for (int h = start; h <= end; h++) + { + if (windows[h] > 0) return false; + } + + return true; + + } + + /** + * @return true if the window for the argument is the end of a previously + * scheduled tour and this period does not overlap with any other + * tour. + */ + public boolean isPreviousArrival(int period) + { + + if (windows[period] == 3 || windows[period] == 4) return true; + else return false; + + } + + /** + * @return true if the window for the argument is the start of a previously + * scheduled tour and this period does not overlap with any other + * tour. + */ + public boolean isPreviousDeparture(int period) + { + + if (windows[period] == 2 || windows[period] == 4) return true; + else return false; + + } + + public boolean isPeriodAvailable(int period) + { + // if windows[index] == 0, the period is available. + + // if window[index] is 0 (available), 2 (start of another tour), 3 (end + // of + // another tour), 4 available for this period only, the period is + // available; + // otherwise, if window[index] is 1 (middle of another tour), it is not + // available. + if (windows[period] == 1) return false; + else return true; + } + + public void setPersId(int id) + { + persId = id; + } + + public void setFreeParkingAvailableResult(int chosenAlt) + { + freeParkingAvailable = (short) chosenAlt; + } + + /** + * set the chosen alternative number: 1=no, 2=yes + * + * @param chosenAlt + */ + public void setInternalExternalTripChoiceResult(int chosenAlt) + { + internalExternalTripChoice = (short) chosenAlt; + } + + public void setParkingReimbursement(double pct) + { + reimbursePercent = (float) pct; + } + + public void setWorkLocationSegmentIndex(int workSegment) + { + workLocSegmentIndex = workSegment; + } + + public void setSchoolLocationSegmentIndex(int schoolSegment) + { + schoolLocSegmentIndex = schoolSegment; + } + + public void setPersAge(int age) + { + persAge = (short) age; + } + + public void setPersGender(int gender) + { + persGender = (short) gender; + } + + public void setPersPecasOccup(int occup) + { + persPecasOccup = (short) occup; + } + + public void setPersActivityCode(int actCode) + { + persActivityCode = (short) actCode; + } + + public void setPersEmploymentCategory(int category) + { + persEmploymentCategory = (short) category; + } + + public void setPersStudentCategory(int category) + { + persStudentCategory = (short) category; + } + + public void setPersonTypeCategory(int personTypeCategory) + { + personType = (short) personTypeCategory; + } + + public void setValueOfTime(float vot) + { + persValueOfTime = vot; + } + + public void setWorkLocation(int aWorkLocationMgra) + { + workLocation = aWorkLocationMgra; + } + + public void setWorkLocDistance(float distance) + { + workLocDistance = distance; + } + + public void setWorkLocLogsum(float logsum) + { + workLocLogsum = logsum; + } + + public void setSchoolLoc(int loc) + { + schoolLoc = loc; + } + + public void setSchoolLocDistance(float distance) + { + schoolLocDistance = distance; + } + + public void setSchoolLocLogsum(float logsum) + { + schoolLocLogsum = logsum; + } + + public void setImtfChoice(int choice) + { + imtfChoice = (short) choice; + } + + public void setInmtfChoice(int choice) + { + inmtfChoice = (short) choice; + } + + public int getImtfChoice() + { + return imtfChoice; + } + + public int getInmtfChoice() + { + return inmtfChoice; + } + + public void clearIndividualNonMandatoryToursArray() + { + indNonManTourArrayList.clear(); + } + + public void createIndividualNonMandatoryTours(int numberOfTours, String primaryPurposeName) + { + + /* + * // if purpose is escort, need to determine if household has kids or + * not String purposeName = primaryPurposeName; if ( + * purposeName.equalsIgnoreCase( modelStructure.ESCORT_PURPOSE_NAME ) ) + * { if ( hhObj.getNumChildrenUnder19() > 0 ) purposeName += "_" + + * modelStructure.ESCORT_SEGMENT_NAMES[0]; else purposeName += "_" + + * modelStructure.ESCORT_SEGMENT_NAMES[1]; } int purposeIndex = + * modelStructure.getDcModelPurposeIndex( purposeName ); + */ + + int id = indNonManTourArrayList.size(); + + int primaryIndex = modelStructure.getPrimaryPurposeNameIndexMap().get(primaryPurposeName); + + for (int i = 0; i < numberOfTours; i++) + { + Tour tempTour = new Tour(id++, this.hhObj, this, primaryPurposeName, + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, primaryIndex); + + tempTour.setTourOrigMgra(this.hhObj.getHhMgra()); + tempTour.setTourDestMgra(0); + + tempTour.setTourPurpose(primaryPurposeName); + + tempTour.setTourDepartPeriod(DEFAULT_NON_MANDATORY_START_PERIOD); + tempTour.setTourArrivePeriod(DEFAULT_NON_MANDATORY_END_PERIOD); + + indNonManTourArrayList.add(tempTour); + } + + } + + public void createWorkTours(int numberOfTours, int startId, String tourPurpose, + int tourPurposeIndex) + { + + workTourArrayList.clear(); + + for (int i = 0; i < numberOfTours; i++) + { + int id = startId + i; + Tour tempTour = new Tour(this, id, tourPurposeIndex); + + tempTour.setTourOrigMgra(hhObj.getHhMgra()); + + if (workLocation == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) tempTour + .setTourDestMgra(hhObj.getHhMgra()); + else tempTour.setTourDestMgra(workLocation); + + tempTour.setTourPurpose(tourPurpose); + + tempTour.setTourDepartPeriod(-1); + tempTour.setTourArrivePeriod(-1); + // tempTour.setTourDepartPeriod(DEFAULT_MANDATORY_START_PERIOD); + // tempTour.setTourArrivePeriod(DEFAULT_MANDATORY_END_PERIOD); + + workTourArrayList.add(tempTour); + } + + } + + public void clearAtWorkSubtours() + { + + atWorkSubtourArrayList.clear(); + + } + + public void createAtWorkSubtour(int id, int choice, int workMgra, String subtourPurpose) + { + + /* + * String segmentedPurpose = modelStructure.AT_WORK_PURPOSE_NAME + "_" + + * tourPurpose; int purposeIndex = + * modelStructure.getDcModelPurposeIndex( segmentedPurpose ); + */ + + Tour tempTour = new Tour(id, this.hhObj, this, + ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME, ModelStructure.AT_WORK_CATEGORY, + ModelStructure.WORK_BASED_PRIMARY_PURPOSE_INDEX); + + tempTour.setTourOrigMgra(workMgra); + tempTour.setTourDestMgra(0); + + tempTour.setTourPurpose(ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME); + tempTour.setSubTourPurpose(subtourPurpose); + + tempTour.setTourDepartPeriod(DEFAULT_AT_WORK_SUBTOUR_START_PERIOD); + tempTour.setTourArrivePeriod(DEFAULT_AT_WORK_SUBTOUR_END_PERIOD); + + atWorkSubtourArrayList.add(tempTour); + + } + + public void createSchoolTours(int numberOfTours, int startId, String tourPurpose, + int tourPurposeIndex) + { + + schoolTourArrayList.clear(); + + for (int i = 0; i < numberOfTours; i++) + { + int id = startId + i; + Tour tempTour = new Tour(this, id, tourPurposeIndex); + + tempTour.setTourOrigMgra(this.hhObj.getHhMgra()); + + if (schoolLoc == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) tempTour + .setTourDestMgra(hhObj.getHhMgra()); + else tempTour.setTourDestMgra(schoolLoc); + + tempTour.setTourPurpose(tourPurpose); + + tempTour.setTourDepartPeriod(-1); + tempTour.setTourArrivePeriod(-1); + // tempTour.setTourDepartPeriod(DEFAULT_MANDATORY_START_PERIOD); + // tempTour.setTourArrivePeriod(DEFAULT_MANDATORY_END_PERIOD); + + schoolTourArrayList.add(tempTour); + } + } + + public int getWorkLocationSegmentIndex() + { + return workLocSegmentIndex; + } + + public int getSchoolLocationSegmentIndex() + { + return schoolLocSegmentIndex; + } + + public void setDailyActivityResult(String activity) + { + this.cdapActivity = activity; + } + + public int getPersonIsChildUnder16WithHomeOrNonMandatoryActivity() + { + + // check the person type + if (persIsStudentNonDrivingAge() == 1 || persIsPreschoolChild() == 1) + { + + // check the activity type + if (cdapActivity.equalsIgnoreCase(ModelStructure.HOME_PATTERN)) return (1); + + if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return (1); + + } + + return (0); + } + + /** + * @return 1 if M, 2 if N, 3 if H + */ + public int getCdapIndex() + { + + // return the activity type + if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return 1; + + if (cdapActivity.equalsIgnoreCase(ModelStructure.NONMANDATORY_PATTERN)) return 2; + + if (cdapActivity.equalsIgnoreCase(ModelStructure.HOME_PATTERN)) return 3; + + return (0); + } + + public int getPersonIsChild6To18WithoutMandatoryActivity() + { + + // check the person type + if (persIsStudentDrivingAge() == 1 || persIsStudentNonDrivingAge() == 1) + { + + // check the activity type + if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return 0; + else return 1; + + } + + return 0; + } + + // methods DMU will use to get info from household object + + public int getAge() + { + return (int) persAge; + } + + public int getHomemaker() + { + return persIsHomemaker(); + } + + public int getGender() + { + return (int) persGender; + } + + public int getPersonIsFemale() + { + if (persGender == 2) return 1; + return 0; + } + + public int getPersonIsMale() + { + if (persGender == 1) return 1; + return 0; + } + + public int getPersonId() + { + return this.persId; + } + + public int getPersonNum() + { + return this.persNum; + } + + public String getPersonType() + { + return PERSON_TYPE_NAME_ARRAY[personType - 1]; + } + + public void setPersonIsHighSchool(boolean flag) + { + highSchool = flag; + } + + public int getPersonIsHighSchool() + { + return highSchool ? 1 : 0; + } + + public void setPersonIsGradeSchool(boolean flag) + { + gradeSchool = flag; + } + + public int getPersonIsGradeSchool() + { + return gradeSchool ? 1 : 0; + } + + public int getPersonIsHighSchoolGraduate() + { + return highSchoolGraduate ? 1 : 0; + } + + public void setPersonIsHighSchoolGraduate(boolean hsGrad) + { + highSchoolGraduate = hsGrad; + } + + public void setPersonHasBachelors(boolean hasBS) + { + hasBachelors = hasBS; + } + + public int getPersonTypeNumber() + { + return personType; + } + + public int getPersPecasOccup() + { + return (int) persPecasOccup; + } + + public int getPersActivityCode() + { + return (int) persActivityCode; + } + + public int getPersonEmploymentCategoryIndex() + { + return (int) persEmploymentCategory; + } + + public String getPersonEmploymentCategory() + { + return EMPLOYMENT_CATEGORY_NAME_ARRAY[persEmploymentCategory - 1]; + } + + public int getPersonStudentCategoryIndex() + { + return persStudentCategory; + } + + public String getPersonStudentCategory() + { + return STUDENT_CATEGORY_NAME_ARRAY[persStudentCategory - 1]; + } + + public float getValueOfTime() + { + return persValueOfTime; + } + + public int getWorkLocation() + { + return workLocation; + } + + public int getPersonSchoolLocationZone() + { + return schoolLoc; + } + + public int getFreeParkingAvailableResult() + { + return freeParkingAvailable; + } + + public int getInternalExternalTripChoiceResult() + { + return internalExternalTripChoice; + } + + public double getParkingReimbursement() + { + return reimbursePercent; + } + + public String getCdapActivity() + { + return cdapActivity; + } + + public float getWorkLocationDistance() + { + return workLocDistance; + } + + public float getWorkLocationLogsum() + { + return workLocLogsum; + } + + public int getUsualSchoolLocation() + { + return schoolLoc; + } + + public float getSchoolLocationDistance() + { + return schoolLocDistance; + } + + public float getSchoolLocationLogsum() + { + return schoolLocLogsum; + } + + public int getHasBachelors() + { + return hasBachelors ? 1 : 0; + } + + public int getNumWorkTours() + { + ArrayList workTours = getListOfWorkTours(); + if (workTours != null) return workTours.size(); + else return 0; + } + + public int getNumSchoolTours() + { + ArrayList schoolTours = getListOfSchoolTours(); + if (schoolTours != null) return schoolTours.size(); + else return 0; + } + + public int getNumIndividualEscortTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.ESCORT_PURPOSE_NAME)) num++; + return num; + } + + public int getNumIndividualShoppingTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SHOPPING_PURPOSE_NAME)) + num++; + return num; + } + + public int getNumIndividualEatOutTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.EAT_OUT_PURPOSE_NAME)) num++; + return num; + } + + public int getNumIndividualOthMaintTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_MAINT_PURPOSE_NAME)) + num++; + return num; + } + + public int getNumIndividualSocialTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SOCIAL_PURPOSE_NAME)) num++; + return num; + } + + public int getNumIndividualOthDiscrTours() + { + int num = 0; + for (Tour tour : getListOfIndividualNonMandatoryTours()) + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_DISCR_PURPOSE_NAME)) + num++; + return num; + } + + public int getNumMandatoryTours() + { + int numTours = 0; + ArrayList workTours = getListOfWorkTours(); + if (workTours != null) numTours += workTours.size(); + + ArrayList schoolTours = getListOfSchoolTours(); + if (schoolTours != null) numTours += schoolTours.size(); + + return numTours; + } + + public int getNumNonMandatoryTours() + { + ArrayList nonMandTours = getListOfIndividualNonMandatoryTours(); + if (nonMandTours == null) return 0; + else return nonMandTours.size(); + } + + public int getNumSubtours() + { + ArrayList subtours = getListOfAtWorkSubtours(); + if (subtours == null) return 0; + else return subtours.size(); + } + + public int getNumTotalIndivTours() + { + return getNumMandatoryTours() + getNumNonMandatoryTours() + getNumSubtours(); + } + + public int getNumJointShoppingTours() + { + return getNumJointToursForPurpose(modelStructure.SHOPPING_PURPOSE_NAME); + } + + public int getNumJointOthMaintTours() + { + return getNumJointToursForPurpose(modelStructure.OTH_MAINT_PURPOSE_NAME); + } + + public int getNumJointEatOutTours() + { + return getNumJointToursForPurpose(modelStructure.EAT_OUT_PURPOSE_NAME); + } + + public int getNumJointSocialTours() + { + return getNumJointToursForPurpose(modelStructure.SOCIAL_PURPOSE_NAME); + } + + public int getNumJointOthDiscrTours() + { + return getNumJointToursForPurpose(modelStructure.OTH_DISCR_PURPOSE_NAME); + } + + private int getNumJointToursForPurpose(String purposeName) + { + int count = 0; + Tour[] jt = hhObj.getJointTourArray(); + if (jt == null) return 0; + + for (int i = 0; i < jt.length; i++) + { + if (jt[i] == null) continue; + String jtPurposeName = jt[i].getTourPurpose(); + int[] personNumsParticipating = jt[i].getPersonNumArray(); + for (int p : personNumsParticipating) + { + if (p == persNum) + { + if (jtPurposeName.equalsIgnoreCase(purposeName)) count++; + break; + } + } + } + + return count; + } + + public void computeIdapResidualWindows() + { + + // find the start of the earliest mandatory or joint tour for this + // person + // and end of last one. + int firstTourStart = 9999; + int lastTourEnd = -1; + int firstTourEnd = 0; + int lastTourStart = 0; + + // first check mandatory tours + for (Tour tour : workTourArrayList) + { + int tourDeparts = tour.getTourDepartPeriod(); + int tourArrives = tour.getTourArrivePeriod(); + + if (tourDeparts < firstTourStart) + { + firstTourStart = tourDeparts; + firstTourEnd = tourArrives; + } + + if (tourArrives > lastTourEnd) + { + lastTourStart = tourDeparts; + lastTourEnd = tourArrives; + } + } + + for (Tour tour : schoolTourArrayList) + { + int tourDeparts = tour.getTourDepartPeriod(); + int tourArrives = tour.getTourArrivePeriod(); + + if (tourDeparts < firstTourStart) + { + firstTourStart = tourDeparts; + firstTourEnd = tourArrives; + } + + if (tourArrives > lastTourEnd) + { + lastTourStart = tourDeparts; + lastTourEnd = tourArrives; + } + } + + // now check joint tours + Tour[] jointTourArray = hhObj.getJointTourArray(); + if (jointTourArray != null) + { + for (Tour tour : jointTourArray) + { + + if (tour == null) continue; + + // see if this person is in the joint tour or not + if (tour.getPersonInJointTour(this)) + { + + int tourDeparts = tour.getTourDepartPeriod(); + int tourArrives = tour.getTourArrivePeriod(); + + if (tourDeparts < firstTourStart) + { + firstTourStart = tourDeparts; + firstTourEnd = tourArrives; + } + + if (tourArrives > lastTourEnd) + { + lastTourStart = tourDeparts; + lastTourEnd = tourArrives; + } + + } + + } + } + + if (firstTourStart > modelStructure.getNumberOfTimePeriods() - 1 && lastTourEnd < 0) + { + int numPeriods = windows.length; + windowBeforeFirstMandJointTour = numPeriods; + windowAfterLastMandJointTour = numPeriods; + windowBetweenFirstLastMandJointTour = numPeriods; + } else + { + + // since first tour first period and last tour last period are + // available, + // account for them. + windowBeforeFirstMandJointTour = firstTourStart + 1; + windowAfterLastMandJointTour = modelStructure.getNumberOfTimePeriods() - lastTourEnd; + + // find the number of unscheduled periods between end of first tour + // and + // start of last tour + windowBetweenFirstLastMandJointTour = 0; + for (int i = firstTourEnd; i <= lastTourStart; i++) + { + if (isPeriodAvailable(i)) windowBetweenFirstLastMandJointTour++; + } + } + + } + + public int getWindowBeforeFirstMandJointTour() + { + return windowBeforeFirstMandJointTour; + } + + public int getWindowBetweenFirstLastMandJointTour() + { + return windowBetweenFirstLastMandJointTour; + } + + public int getWindowAfterLastMandJointTour() + { + return windowAfterLastMandJointTour; + } + + // public int getNumberOfMandatoryWorkTours( String workPurposeName ){ + // + // int numberOfTours = 0; + // for(int i=0;i= MIN_ADULT_AGE + && persEmploymentCategory == EmployStatus.NOT_EMPLOYED.ordinal()) return 1; + else return 0; + } + + public int notEmployed() + { + if (persEmploymentCategory == EmployStatus.NOT_EMPLOYED.ordinal()) return 1; + else return 0; + } + + private int persIsWorker() + { + if (persEmploymentCategory == EmployStatus.FULL_TIME.ordinal() + || persEmploymentCategory == EmployStatus.PART_TIME.ordinal()) return 1; + else return 0; + } + + private int persIsStudent() + { + if (persStudentCategory == StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal() + || persStudentCategory == StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal()) + { + return 1; + } else + { + return 0; + } + } + + private int persIsFullTimeWorker() + { + if (persEmploymentCategory == EmployStatus.FULL_TIME.ordinal()) return 1; + else return 0; + } + + private int persIsPartTimeWorker() + { + if (persEmploymentCategory == EmployStatus.PART_TIME.ordinal()) return 1; + else return 0; + } + + private int persTypeIsFullTimeWorker() + { + if (personType == PersonType.FT_worker_age_16plus.ordinal()) return 1; + else return 0; + } + + private int persTypeIsPartTimeWorker() + { + if (personType == PersonType.PT_worker_nonstudent_age_16plus.ordinal()) return 1; + else return 0; + } + + private int persIsUniversity() + { + if (personType == PersonType.University_student.ordinal()) return 1; + else return 0; + } + + private int persIsStudentDrivingAge() + { + if (personType == PersonType.Student_age_16_19_not_FT_wrkr_or_univ_stud.ordinal()) return 1; + else return 0; + } + + private int persIsStudentNonDrivingAge() + { + if (personType == PersonType.Student_age_6_15_schpred.ordinal()) return 1; + else return 0; + } + + private int persIsPreschoolChild() + { + if (personType == PersonType.Preschool_under_age_6.ordinal()) return 1; + else return 0; + + } + + private int persIsNonWorkingAdultUnder65() + { + if (personType == PersonType.Nonworker_nonstudent_age_16_64.ordinal()) return 1; + else return 0; + } + + private int persIsNonWorkingAdultOver65() + { + if (personType == PersonType.Nonworker_nonstudent_age_65plus.ordinal()) + { + return 1; + } else + { + return 0; + } + } + + /** + * return maximum periods of overlap between this person and other adult + * persons in the household. + * + * @return the most number of periods mutually available between this person + * and other adult household members + */ + public int getMaxAdultOverlaps() + { + return maxAdultOverlaps; + } + + /** + * set maximum periods of overlap between this person and other adult + * persons in the household. + * + * @param overlaps + * are the most number of periods mutually available between this + * person and other adult household members + */ + public void setMaxAdultOverlaps(int overlaps) + { + maxAdultOverlaps = overlaps; + } + + /** + * return maximum periods of overlap between this person and other children + * in the household. + * + * @return the most number of periods mutually available between this person + * and other child household members + */ + public int getMaxChildOverlaps() + { + return maxChildOverlaps; + } + + /** + * set maximum periods of overlap between this person and other children in + * the household. + * + * @param overlaps + * are the most number of periods mutually available between this + * person and other child household members + */ + public void setMaxChildOverlaps(int overlaps) + { + maxChildOverlaps = overlaps; + } + + /** + * return available time window for this person in the household. + * + * @return the total number of periods available for this person + */ + public int getAvailableWindow() + { + int numPeriodsAvailable = 0; + for (int i = 1; i < windows.length; i++) + if (windows[i] != 1) numPeriodsAvailable++; + + return numPeriodsAvailable; + } + + /** + * determine the maximum consecutive available time window for the person + * + * @return the length of the maximum available window in units of time + * intervals + */ + public int getMaximumContinuousAvailableWindow() + { + int maxWindow = 0; + int currentWindow = 0; + for (int i = 1; i < windows.length; i++) + { + if (windows[i] == 0) + { + currentWindow++; + } else + { + if (currentWindow > maxWindow) maxWindow = currentWindow; + currentWindow = 0; + } + } + if (currentWindow > maxWindow) maxWindow = currentWindow; + + return maxWindow; + } + + /** + * determine the maximum consecutive pairwise available time window for this + * person and the person for which a window was passed + * + * @return the length of the maximum pairwise available window in units of + * time intervals + */ + public int getMaximumContinuousPairwiseAvailableWindow(short[] otherWindow) + { + int maxWindow = 0; + int currentWindow = 0; + for (int i = 1; i < windows.length; i++) + { + if (windows[i] == 0 && otherWindow[i] == 0) + { + currentWindow++; + } else + { + if (currentWindow > maxWindow) maxWindow = currentWindow; + currentWindow = 0; + } + } + if (currentWindow > maxWindow) maxWindow = currentWindow; + + return maxWindow; + } + + public void setTimeWindows(short[] win) + { + windows = win; + } + + public void initializeForAoRestart() + { + + cdapActivity = "-"; + imtfChoice = 0; + inmtfChoice = 0; + + maxAdultOverlaps = 0; + maxChildOverlaps = 0; + + workTourArrayList.clear(); + schoolTourArrayList.clear(); + indNonManTourArrayList.clear(); + atWorkSubtourArrayList.clear(); + + initializeWindows(); + + windowBeforeFirstMandJointTour = 0; + windowBetweenFirstLastMandJointTour = 0; + windowAfterLastMandJointTour = 0; + + } + + public void initializeForImtfRestart() + { + + imtfChoice = 0; + inmtfChoice = 0; + + maxAdultOverlaps = 0; + maxChildOverlaps = 0; + + workTourArrayList.clear(); + schoolTourArrayList.clear(); + indNonManTourArrayList.clear(); + atWorkSubtourArrayList.clear(); + + initializeWindows(); + + windowBeforeFirstMandJointTour = 0; + windowBetweenFirstLastMandJointTour = 0; + windowAfterLastMandJointTour = 0; + + } + + /** + * initialize the person attributes and tour objects for restarting the + * model at joint tour frequency + */ + public void initializeForJtfRestart() + { + + inmtfChoice = 0; + + indNonManTourArrayList.clear(); + atWorkSubtourArrayList.clear(); + + for (int i = 0; i < workTourArrayList.size(); i++) + { + Tour t = workTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + for (int i = 0; i < schoolTourArrayList.size(); i++) + { + Tour t = schoolTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + + windowBeforeFirstMandJointTour = 0; + windowBetweenFirstLastMandJointTour = 0; + windowAfterLastMandJointTour = 0; + + } + + /** + * initialize the person attributes and tour objects for restarting the + * model at individual non-mandatory tour frequency. + */ + public void initializeForInmtfRestart() + { + + inmtfChoice = 0; + + indNonManTourArrayList.clear(); + atWorkSubtourArrayList.clear(); + + for (int i = 0; i < workTourArrayList.size(); i++) + { + Tour t = workTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + for (int i = 0; i < schoolTourArrayList.size(); i++) + { + Tour t = schoolTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + + windowBeforeFirstMandJointTour = 0; + windowBetweenFirstLastMandJointTour = 0; + windowAfterLastMandJointTour = 0; + + } + + /** + * initialize the person attributes and tour objects for restarting the + * model at at-work sub-tour frequency. + */ + public void initializeForAwfRestart() + { + + atWorkSubtourArrayList.clear(); + + for (int i = 0; i < workTourArrayList.size(); i++) + { + Tour t = workTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + for (int i = 0; i < schoolTourArrayList.size(); i++) + { + Tour t = schoolTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + for (int i = 0; i < indNonManTourArrayList.size(); i++) + { + Tour t = indNonManTourArrayList.get(i); + scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); + t.clearStopModelResults(); + } + + } + + /** + * initialize the person attributes and tour objects for restarting the + * model at stop frequency. + */ + public void initializeForStfRestart() + { + + for (int i = 0; i < workTourArrayList.size(); i++) + { + Tour t = workTourArrayList.get(i); + t.clearStopModelResults(); + } + for (int i = 0; i < schoolTourArrayList.size(); i++) + { + Tour t = schoolTourArrayList.get(i); + t.clearStopModelResults(); + } + for (int i = 0; i < atWorkSubtourArrayList.size(); i++) + { + Tour t = atWorkSubtourArrayList.get(i); + t.clearStopModelResults(); + } + for (int i = 0; i < indNonManTourArrayList.size(); i++) + { + Tour t = indNonManTourArrayList.get(i); + t.clearStopModelResults(); + } + + } + + public float getParkingProvisionLogsum() { + return parkingProvisionLogsum; + } + + public void setParkingProvisionLogsum(float parkingProvisionLogsum) { + this.parkingProvisionLogsum = parkingProvisionLogsum; + } + + public float getIeLogsum() { + return ieLogsum; + } + + public void setIeLogsum(float ieLogsum) { + this.ieLogsum = ieLogsum; + } + + public float getCdapLogsum() { + return cdapLogsum; + } + + public void setCdapLogsum(float cdapLogsum) { + this.cdapLogsum = cdapLogsum; + } + + + public float getImtfLogsum() { + return imtfLogsum; + } + + public void setImtfLogsum(float imtfLogsum) { + this.imtfLogsum = imtfLogsum; + } + + public float getInmtfLogsum() { + return inmtfLogsum; + } + + public void setInmtfLogsum(float inmtfLogsum) { + this.inmtfLogsum = inmtfLogsum; + } + + public float getWorksFromHomeLogsum() { + return worksFromHomeLogsum; + } + + public void setWorksFromHomeLogsum(float worksFromHomeLogsum) { + this.worksFromHomeLogsum = worksFromHomeLogsum; + } + + public void logPersonObject(Logger logger, int totalChars) + { + + Household.logHelper(logger, "persNum: ", persNum, totalChars); + Household.logHelper(logger, "persId: ", persId, totalChars); + Household.logHelper(logger, "persAge: ", persAge, totalChars); + Household.logHelper(logger, "persGender: ", persGender, totalChars); + Household.logHelper(logger, "persEmploymentCategory: ", persEmploymentCategory, totalChars); + Household.logHelper(logger, "persStudentCategory: ", persStudentCategory, totalChars); + Household.logHelper(logger, "personType: ", personType, totalChars); + Household.logHelper(logger, "workLoc: ", workLocation, totalChars); + Household.logHelper(logger, "schoolLoc: ", schoolLoc, totalChars); + Household.logHelper(logger, "workLocSegmentIndex: ", workLocSegmentIndex, totalChars); + Household.logHelper(logger, "schoolLocSegmentIndex: ", schoolLocSegmentIndex, totalChars); + + Household.logHelper(logger, "timeFactorWork: ", String.format("%.2f",timeFactorWork), totalChars); + Household.logHelper(logger, "timeFactorNonWork: ", String.format("%.2f",timeFactorNonWork), totalChars); + Household.logHelper(logger, "freeParkingAvailable: ", freeParkingAvailable, totalChars); + Household.logHelper(logger, "reimbursementPct: ", + String.format("%.2f%%", (100 * reimbursePercent)), totalChars); + Household.logHelper(logger, "cdapActivity: ", cdapActivity, totalChars); + Household.logHelper(logger, "imtfChoice: ", imtfChoice, totalChars); + Household.logHelper(logger, "inmtfChoice: ", inmtfChoice, totalChars); + Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); + Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); + Household.logHelper(logger, "windowBeforeFirstMandJointTour: ", + windowBeforeFirstMandJointTour, totalChars); + Household.logHelper(logger, "windowBetweenFirstLastMandJointTour: ", + windowBetweenFirstLastMandJointTour, totalChars); + Household.logHelper(logger, "windowAfterLastMandJointTour: ", windowAfterLastMandJointTour, + totalChars); + + String header1 = " Index: |"; + String header2 = " Period: |"; + String windowString = " Window: |"; + String periodString = ""; + for (int i = 1; i < windows.length; i++) + { + header1 += String.format(" %2d |", i); + header2 += String.format("%4s|", modelStructure.getTimePeriodLabel(i)); + switch (windows[i]) + { + case 0: + periodString = " "; + break; + case 1: + periodString = "XXXX"; + break; + } + windowString += String.format("%4s|", periodString); + } + + logger.info(header1); + logger.info(header2); + logger.info(windowString); + + if (workTourArrayList.size() > 0) + { + for (Tour tour : workTourArrayList) + { + int id = tour.getTourId(); + logger.info(tour.getTourWindow(String.format("W%d", id))); + } + } + if (atWorkSubtourArrayList.size() > 0) + { + for (Tour tour : atWorkSubtourArrayList) + { + int id = tour.getTourId(); + String alias = ""; + String purposeName = tour.getSubTourPurpose(); + if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME)) alias = "aB"; + else if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_EAT_PURPOSE_NAME)) alias = "aE"; + else if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_MAINT_PURPOSE_NAME)) + alias = "aM"; + logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); + } + } + if (schoolTourArrayList.size() > 0) + { + for (Tour tour : schoolTourArrayList) + { + int id = tour.getTourId(); + String alias = "S"; + logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); + } + } + if (hhObj.getJointTourArray() != null && hhObj.getJointTourArray().length > 0) + { + for (Tour tour : hhObj.getJointTourArray()) + { + if (tour == null) continue; + + // log this persons time window if they are in the joint tour + // party. + int[] persNumArray = tour.getPersonNumArray(); + if (persNumArray != null) + { + for (int num : persNumArray) + { + if (num == persNum) + { + + Person person = hhObj.getPersons()[num]; + tour.setPersonObject(person); + + int id = tour.getTourId(); + String alias = ""; + if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) alias = "jE"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) alias = "jS"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) alias = "jM"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) alias = "jV"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) alias = "jD"; + logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); + } + } + } + } + } + if (indNonManTourArrayList.size() > 0) + { + for (Tour tour : indNonManTourArrayList) + { + int id = tour.getTourId(); + String alias = ""; + if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) alias = "ie"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) alias = "iE"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) alias = "iS"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) alias = "iM"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) alias = "iV"; + else if (tour.getTourPurpose().equalsIgnoreCase( + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) alias = "iD"; + logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); + } + } + + } + + public void logTourObject(Logger logger, int totalChars, Tour tour) + { + tour.logTourObject(logger, totalChars); + } + + public void logEntirePersonObject(Logger logger) + { + + int totalChars = 60; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + Household.logHelper(logger, "persNum: ", persNum, totalChars); + Household.logHelper(logger, "persId: ", persId, totalChars); + Household.logHelper(logger, "persAge: ", persAge, totalChars); + Household.logHelper(logger, "persGender: ", persGender, totalChars); + Household.logHelper(logger, "persEmploymentCategory: ", persEmploymentCategory, totalChars); + Household.logHelper(logger, "persStudentCategory: ", persStudentCategory, totalChars); + Household.logHelper(logger, "personType: ", personType, totalChars); + Household.logHelper(logger, "workLoc: ", workLocation, totalChars); + Household.logHelper(logger, "schoolLoc: ", schoolLoc, totalChars); + Household.logHelper(logger, "workLocSegmentIndex: ", workLocSegmentIndex, totalChars); + Household.logHelper(logger, "schoolLocSegmentIndex: ", schoolLocSegmentIndex, totalChars); + Household.logHelper(logger, "freeParkingAvailable: ", freeParkingAvailable, totalChars); + Household.logHelper(logger, "reimbursementPct: ", + String.format("%.2f%%", (100 * reimbursePercent)), totalChars); + Household.logHelper(logger, "cdapActivity: ", cdapActivity, totalChars); + Household.logHelper(logger, "imtfChoice: ", imtfChoice, totalChars); + Household.logHelper(logger, "inmtfChoice: ", inmtfChoice, totalChars); + Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); + Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); + Household.logHelper(logger, "windowBeforeFirstMandJointTour: ", + windowBeforeFirstMandJointTour, totalChars); + Household.logHelper(logger, "windowBetweenFirstLastMandJointTour: ", + windowBetweenFirstLastMandJointTour, totalChars); + Household.logHelper(logger, "windowAfterLastMandJointTour: ", windowAfterLastMandJointTour, + totalChars); + + String header = " Period: |"; + String windowString = " Window: |"; + for (int i = 1; i < windows.length; i++) + { + header += String.format("%4s|", modelStructure.getTimePeriodLabel(i)); + windowString += String.format("%4s|", windows[i] == 0 ? " " : "XXXX"); + } + + logger.info(header); + logger.info(windowString); + + if (workTourArrayList.size() > 0) + { + for (Tour tour : workTourArrayList) + { + int id = tour.getTourId(); + logger.info(tour.getTourWindow(String.format("W%d", id))); + } + } + if (schoolTourArrayList.size() > 0) + { + for (Tour tour : schoolTourArrayList) + { + logger.info(tour + .getTourWindow(tour.getTourPurpose().equalsIgnoreCase("university") ? "U" + : "S")); + } + } + if (indNonManTourArrayList.size() > 0) + { + for (Tour tour : indNonManTourArrayList) + { + logger.info(tour.getTourWindow("N")); + } + } + if (atWorkSubtourArrayList.size() > 0) + { + for (Tour tour : atWorkSubtourArrayList) + { + logger.info(tour.getTourWindow("A")); + } + } + if (hhObj.getJointTourArray() != null && hhObj.getJointTourArray().length > 0) + { + for (Tour tour : hhObj.getJointTourArray()) + { + if (tour != null) logger.info(tour.getTourWindow("J")); + } + } + + logger.info(separater); + + logger.info("Work Tours:"); + if (workTourArrayList.size() > 0) + { + for (Tour tour : workTourArrayList) + { + tour.logEntireTourObject(logger); + } + } else + { + logger.info(" No work tours"); + } + + logger.info("School Tours:"); + if (schoolTourArrayList.size() > 0) + { + for (Tour tour : schoolTourArrayList) + { + tour.logEntireTourObject(logger); + } + } else + { + logger.info(" No school tours"); + } + + logger.info("Individual Non-mandatory Tours:"); + if (indNonManTourArrayList.size() > 0) + { + for (Tour tour : indNonManTourArrayList) + { + tour.logEntireTourObject(logger); + } + } else + { + logger.info(" No individual non-mandatory tours"); + } + + logger.info("Work based subtours Tours:"); + if (atWorkSubtourArrayList.size() > 0) + { + for (Tour tour : atWorkSubtourArrayList) + { + tour.logEntireTourObject(logger); + } + } else + { + logger.info(" No work based subtours"); + } + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public double getTimeFactorWork() { + return (double) timeFactorWork; + } + + public void setTimeFactorWork(double timeFactorWork) { + this.timeFactorWork = (float) timeFactorWork; + } + + public double getTimeFactorNonWork() { + return (double) timeFactorNonWork; + } + + public void setTimeFactorNonWork(double timeFactorNonWork) { + this.timeFactorNonWork = (float) timeFactorNonWork; + } + + public enum EmployStatus + { + nul, FULL_TIME, PART_TIME, NOT_EMPLOYED, UNDER16 + } + + public enum StudentStatus + { + nul, STUDENT_HIGH_SCHOOL_OR_LESS, STUDENT_COLLEGE_OR_HIGHER, NON_STUDENT + } + + public enum PersonType + { + nul, FT_worker_age_16plus, PT_worker_nonstudent_age_16plus, University_student, Nonworker_nonstudent_age_16_64, Nonworker_nonstudent_age_65plus, Student_age_16_19_not_FT_wrkr_or_univ_stud, Student_age_6_15_schpred, Preschool_under_age_6 + } + + /** + * Returns true if this person is an active adult, else returns false. Active adult + * is defined as full-time worker, part-time worker, university student, + * non-working adult or retired person who has an activity pattern other than H. + * + * @return true if active adult, else false. + */ + public boolean isActiveAdult(){ + boolean activeAdult=false; + if((getPersonTypeNumber()==Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX)||(getPersonTypeNumber()==Person.PERSON_TYPE_PART_TIME_WORKER_INDEX)|| + (getPersonTypeNumber()==Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX)||(getPersonTypeNumber()==Person.PERSON_TYPE_NON_WORKER_INDEX)|| + (getPersonTypeNumber()==Person.PERSON_TYPE_RETIRED_INDEX)) + if(!getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) + activeAdult=true; + return activeAdult; + } + + public short getTelecommuteChoice() { + return telecommuteChoice; + } + + public void setTelecommuteChoice(short telecommuteChoice) { + this.telecommuteChoice = telecommuteChoice; + } + + public float getTelecommuteLogsum() { + return telecommuteLogsum; + } + + public void setTelecommuteLogsum(float telecommuteLogsum) { + this.telecommuteLogsum = telecommuteLogsum; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java new file mode 100644 index 0000000..32f93ef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java @@ -0,0 +1,67 @@ +/* +* The school-escort model was designed by PB (Gupta, Vovsha, et al) +* as part of the Maricopa Association of Governments (MAG) +* Activity-based Travel Model Development project. +* +* This source code, which implements the school escort model, +* was written exclusively for and funded by MAG as part of the +* same project; therefore, per their contract, the +* source code belongs to MAG and can only be used with their +* permission. +* +* It is being adapted for the Southern Oregon ABM by PB & RSG +* with permission from MAG and all references to +* the school escort model as well as source code adapted from this +* original code should credit MAG's role in its development. +* +* The escort model and source code should not be transferred to or +* adapted for other agencies or used in other projects without +* expressed permission from MAG. +* +* The source code has been substantially revised to fit within the +* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). +*/ + +package org.sandag.abm.ctramp; + +import java.io.Serializable; + +public class SchoolEscortChauffeurResult implements Serializable { + + private static final long serialVersionUID = 1L; + + private final int pid; + private final short dir; + private final short bundle; + private final short escortType; + private final short[] childPnums; + + public SchoolEscortChauffeurResult( int pid, short dir, short bundle, short escortType, short[] childPnums ) { + this.pid = pid; + this.dir = dir; + this.bundle = bundle; + this.escortType = escortType; + this.childPnums = childPnums; + } + + public int getPid() { + return pid; + } + + public short getDirection() { + return dir; + } + + public short getBundle() { + return bundle; + } + + public short getEscortType() { + return escortType; + } + + public short[] getChildPnums() { + return childPnums; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java new file mode 100644 index 0000000..cfa6098 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java @@ -0,0 +1,68 @@ +/* +* The school-escort model was designed by PB (Gupta, Vovsha, et al) +* as part of the Maricopa Association of Governments (MAG) +* Activity-based Travel Model Development project. +* +* This source code, which implements the school escort model, +* was written exclusively for and funded by MAG as part of the +* same project; therefore, per their contract, the +* source code belongs to MAG and can only be used with their +* permission. +* +* It is being adapted for the Southern Oregon ABM by PB & RSG +* with permission from MAG and all references to +* the school escort model as well as source code adapted from this +* original code should credit MAG's role in its development. +* +* The escort model and source code should not be transferred to or +* adapted for other agencies or used in other projects without +* expressed permission from MAG. +* +* The source code has been substantially revised to fit within the +* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). +* +*/ + +package org.sandag.abm.ctramp; + +import java.io.Serializable; + +public class SchoolEscortChildResult implements Serializable { + + private static final long serialVersionUID = 1L; + + private final int pid; + private final short dir; + private final short bundle; + private final short escortType; + private final short adultPnum; + + public SchoolEscortChildResult( int pid, short dir, short bundle, short escortType, short adultPnum ) { + this.pid = pid; + this.dir = dir; + this.bundle = bundle; + this.escortType = escortType; + this.adultPnum = adultPnum; + } + + public int getPid() { + return pid; + } + + public short getDirection() { + return dir; + } + + public short getBundle() { + return bundle; + } + + public short getEscortType() { + return escortType; + } + + public short getAdultPnum() { + return adultPnum; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java new file mode 100644 index 0000000..ee37952 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java @@ -0,0 +1,499 @@ +/* +* The school-escort model was designed by PB (Gupta, Vovsha, et al) +* as part of the Maricopa Association of Governments (MAG) +* Activity-based Travel Model Development project. +* +* This source code, which implements the school escort model, +* was written exclusively for and funded by MAG as part of the +* same project; therefore, per their contract, the +* source code belongs to MAG and can only be used with their +* permission. +* +* It is being adapted for the Southern Oregon ABM by PB & RSG +* with permission from MAG and all references to +* the school escort model as well as source code adapted from this +* original code should credit MAG's role in its development. +* +* The escort model and source code should not be transferred to or +* adapted for other agencies or used in other projects without +* expressed permission from MAG. +* +* The source code has been substantially revised to fit within the +* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). +*/ + +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.List; +import java.util.StringTokenizer; + +import org.apache.log4j.Logger; + + +public class SchoolEscortingBundle implements Serializable { + + private static final long serialVersionUID = 1L; + + private int id; + private int dir; + private final int alt; + private final int bundle; + private final int type; + private final int chaufId; + private int chaufPnum; + private int chaufPersType; + private int chaufPid; + private int[] childIds; + private int[] childPnums; + private int[] schoolMazs; + private float[] schoolDists; + private int workOrSchoolMaz; + private int departHome; + private int arriveWork; + private int departWork; + private int arriveHome; + private int departPrimaryInterval = -1; + + private SchoolEscortingBundle( int alt, int bundle, int chaufId, int type, int[] childIds, int[] childPnums ) { + this.alt = alt; + this.bundle = bundle; + this.chaufId = chaufId; + this.type = type; + this.childIds = childIds; + this.childPnums = childPnums; + } + + + /** + * Get an Arraylist of SchoolEscortingBundles, dimensioned by: + * 0: max chauffeurs (2) + * 1: max bundles (3) + * + * @param alt The alternative number + * @param altBundleIncidence + * @return An array of SchoolEscortingBundles (dimensioned by 2, for each chauffeur) + */ + public static List[] constructAltBundles( int alt, int[][] altBundleIncidence ) { + + //first dimension of results array is dimensioned by number of chauffeurs + 1 + List[] results = new List[ SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1 ]; + + //for each potential bundle (3) + for ( int i=1; i <= SchoolEscortingModel.NUM_BUNDLES; i++ ) { + + //for each potential chauffeur (2) + for ( int j=1; j <= SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH; j++ ) { + + // for each escort type (rideshare vs pure escort) + for ( int k=1; k <= SchoolEscortingModel.NUM_ESCORT_TYPES; k++ ) { + + //if an arraylist hasn't been created for this chauffeur, create one. + if ( results[j] == null ) + results[j] = new ArrayList(SchoolEscortingModel.NUM_BUNDLES); + + //childIdList initial capacity is max escortees (3); for each potential escortee + List childIdList = new ArrayList(SchoolEscortingModel.NUM_ESCORTEES_PER_HH); + for ( int l=1; l <= SchoolEscortingModel.NUM_ESCORTEES_PER_HH; l++ ) { + int columnIndex = (i-1) * (SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH * SchoolEscortingModel.NUM_ESCORT_TYPES * SchoolEscortingModel.NUM_ESCORTEES_PER_HH) + + (j-1) * (SchoolEscortingModel.NUM_ESCORT_TYPES * SchoolEscortingModel.NUM_ESCORTEES_PER_HH) + + (k-1) * (SchoolEscortingModel.NUM_ESCORTEES_PER_HH) + + (l-1) + 1; + //if child number l belongs to this bundle\chauffeur\escort type combination, add l to the childIdList + if ( altBundleIncidence[alt][columnIndex] > 0 ) + childIdList.add( l ); + } + + //if children are in this bundle\chauffeur\escort type combination + if ( childIdList.size() > 0 ) { + int[] childIds = new int[childIdList.size()]; + int[] childPnums = new int[childIdList.size()]; + for ( int l=0; l < childIdList.size(); l++ ) { + childIds[l] = childIdList.get( l ); + } + //add a new bundle to the chauffeur element. The bundle contains the number of the bundle, the number of the chauffeur, the + //escort type (rideshare versus pure), the child ids (1 through 3) and an empty array of person numbers for each child. + results[j].add( new SchoolEscortingBundle( alt, i, j, k, childIds, childPnums ) ); + } + + } + + } + + } + + return results; + + } + + public void setId( int id ) { + this.id = id; + } + + public int getId() { + return id; + } + + public void setDir( int dir ) { + this.dir = dir; + } + + public int getDir() { + return dir; + } + + public int getAlt() { + return alt; + } + + public int getBundle() { + return bundle; + } + + public int getChaufId() { + return chaufId; + } + + public void setChaufPnum( int pnum ) { + chaufPnum = pnum; + } + + public int getChaufPnum() { + return chaufPnum; + } + + public void setChaufPersType( int ptype ) { + chaufPersType = ptype; + } + + public int getChaufPersType() { + return chaufPersType; + } + + public void setChaufPid( int pid ) { + chaufPid = pid; + } + + public int getChaufPid() { + return chaufPid; + } + + public int getEscortType() { + return type; + } + + public void setSchoolMazs( int[] schoolMazs ) { + this.schoolMazs = schoolMazs; + } + + public int[] getSchoolMazs() { + return schoolMazs; + } + + public void setSchoolDists( float[] schoolDists ) { + this.schoolDists = schoolDists; + } + + public float[] getSchoolDists() { + return schoolDists; + } + + public void setChildIds( int[] childIds ) { + this.childIds = childIds; + } + + public int[] getChildIds() { + return childIds; + } + + public void setChildPnums( int[] childPnums ) { + this.childPnums = childPnums; + } + + public int[] getChildPnums() { + return childPnums; + } + + public void setDepartHome( int depart ) { + departHome = depart; + } + + public int getDepartHome() { + return departHome; + } + + /* + public void setArriveHome( int arrive ) { + arriveHome = Math.min( arrive, TourTodDmu.NUM_TOD_INTERVALS ); + } +*/ + /** + * Arrive home; modified JEF to remove taking the minimum of arrive and number of TOD intervals. + * @param arrive + */ + public void setArriveHome( int arrive ) { + arriveHome = arrive; + } + + + public int getArriveHome() { + return arriveHome; + } + + public void setDepartWork( int depart ) { + departWork = depart; + } + + public int getDepartWork() { + return departWork; + } + + public void setArriveWork( int arrive ) { + arriveWork = arrive; + } + + public int getArriveWork() { + return arriveWork; + } + + public void setWorkOrSchoolMaz( int maz ) { + workOrSchoolMaz = maz; + } + + public int getWorkOrSchoolMaz() { + return workOrSchoolMaz; + } + + public void setDepartPrimaryInterval( int interval ) { + departPrimaryInterval = interval; + } + + public int getDepartPrimaryInterval() { + return departPrimaryInterval; + } + + + public String toString() { + + String childIdString = "["; + String childPnumString = "["; + String schoolString = "["; + String distsString = "["; + if ( childIds.length > 0 ) { + childIdString += childIds[0]; + childPnumString += childPnums[0]; + schoolString += schoolMazs[0]; + distsString += String.format( "%.5f", schoolDists[0] ); + for ( int i=1; i < childIds.length; i++ ) { + childIdString += "," + childIds[i]; + childPnumString += "," + childPnums[i]; + schoolString += "," + schoolMazs[i]; + distsString += "," + String.format( "%.5f", schoolDists[i] ); + } + } + childIdString += "]"; + childPnumString += "]"; + schoolString += "]"; + distsString += "]"; + + String outputString = + "\tid = " + id + "\n" + + "\tdir = " + (dir == SchoolEscortingModel.DIR_OUTBOUND ? "outbound" : "inbound" ) + "\n" + + "\talt = " + alt + "\n" + + "\tbundle = " + bundle + "\n" + + "\tchaufPnum = " + chaufPnum + "\n" + + "\tchaufPid = " + chaufPid + "\n" + + "\tchaufPtype = " + chaufPersType + "\n" + + "\tescort type = " + (type == ModelStructure.RIDE_SHARING_TYPE ? "ride sharing" : "pure escort" ) + "\n" + + "\tchildIds = " + childIdString + "\n" + + "\tchildPnums = " + childPnumString + "\n" + + "\tschoolMazs = " + schoolString + "\n" + + "\tschoolDists = " + distsString + "\n" + + "\tdepartHome = " + departHome + "\n" + + "\tarriveHome = " + arriveHome + "\n" + + "\tdepartWork = " + departWork + "\n" + + "\tarriveWork = " + arriveWork + "\n\n"; + + return outputString; + + } + + public static String getExportHeaderString() { + String header = "id,dir,alt,bundle,type,chaufId,chaufPnum,chaufPersType,chaufPid,departHome,arriveHome,departWork,arriveWork,childIds,childPnums,schoolMazs,schoolDists"; + return header; + } + + public String getExportString() { + + String childIdString = "["; + String childPnumString = "["; + String schoolString = "["; + String distsString = "["; + if ( childIds.length > 0 ) { + childIdString += childIds[0]; + childPnumString += childPnums[0]; + schoolString += schoolMazs[0]; + distsString += String.format( "%.5f", schoolDists[0] ); + for ( int i=1; i < childIds.length; i++ ) { + childIdString += "," + childIds[i]; + childPnumString += "," + childPnums[i]; + schoolString += "," + schoolMazs[i]; + distsString += "," + String.format( "%.5f", schoolDists[i] ); + } + } + childIdString += "]"; + childPnumString += "]"; + schoolString += "]"; + distsString += "]"; + + String outputString = + id + "," + + dir + "," + + alt + "," + + bundle + "," + + type + "," + + chaufId + "," + + chaufPnum + "," + + chaufPersType + "," + + chaufPid + "," + + departHome + "," + + arriveHome + "," + + departWork + "," + + arriveWork + "," + + childIdString + "," + + childPnumString + "," + + schoolString + "," + + distsString; + + return outputString; + + } + + public void logBundle(Logger logger){ + + String childIdString = "["; + String childPnumString = "["; + String schoolString = "["; + String distsString = "["; + if ( childIds.length > 0 ) { + childIdString += childIds[0]; + childPnumString += childPnums[0]; + schoolString += schoolMazs[0]; + distsString += String.format( "%.5f", schoolDists[0] ); + for ( int i=1; i < childIds.length; i++ ) { + childIdString += "," + childIds[i]; + childPnumString += "," + childPnums[i]; + schoolString += "," + schoolMazs[i]; + distsString += "," + String.format( "%.5f", schoolDists[i] ); + } + } + childIdString += "]"; + childPnumString += "]"; + schoolString += "]"; + distsString += "]"; + logger.info("***********************************************"); + logger.info("id = " + id); + logger.info("dir = " + (dir == SchoolEscortingModel.DIR_OUTBOUND ? "outbound" : "inbound" ) ); + logger.info("alt = " + alt ); + logger.info("bundle = " + bundle ); + logger.info("chaufPnum = " + chaufPnum ); + logger.info("chaufPid = " + chaufPid ); + logger.info("chaufPtype = " + chaufPersType ); + logger.info("escort type = " + (type == ModelStructure.RIDE_SHARING_TYPE ? "ride sharing" : "pure escort" ) ); + logger.info("childIds = " + childIdString ); + logger.info("childPnums = " + childPnumString ); + logger.info("schoolMazs = " + schoolString ); + logger.info("schoolDists = " + distsString ); + logger.info("departHome = " + departHome ); + logger.info("arriveHome = " + arriveHome ); + logger.info("departWork = " + departWork ); + logger.info("arriveWork = " + arriveWork ); + logger.info("***********************************************"); + + + } + +/* + public static SchoolEscortingBundle restoreSchoolEscortingBundleFromExportString( String exportString ) throws Exception { + + StringTokenizer st = new StringTokenizer( exportString, "," ); + + String stringValue = st.nextToken().trim(); + int idValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int dirValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int altValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int bundleValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int typeValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int chaufIdValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int chaufPnumValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int chaufPtypeValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int chaufPidValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int departHomeValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int arriveHomeValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int departWorkValue = Integer.parseInt( stringValue ); + + stringValue = st.nextToken().trim(); + int arriveWorkValue = Integer.parseInt( stringValue ); + + int startCharIndex = exportString.indexOf("[") + 1; + int endCharIndex = exportString.indexOf("]"); + String valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); + int[] childIdValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); + Integer.par + startCharIndex = exportString.indexOf("[", endCharIndex) + 1; + endCharIndex = exportString.indexOf("]", startCharIndex); + valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); + int[] childPnumValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); + + startCharIndex = exportString.indexOf("[", endCharIndex) + 1; + endCharIndex = exportString.indexOf("]", startCharIndex); + valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); + int[] schoolMazsValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); + + startCharIndex = exportString.indexOf("[", endCharIndex) + 1; + endCharIndex = exportString.indexOf("]", startCharIndex); + valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); + float[] schoolDistValues = Parsing.getOneDimensionalFloatArrayValuesFromExportString( valuesOnlyString ); + + + SchoolEscortingBundle newBundle = new SchoolEscortingBundle( altValue, bundleValue, chaufIdValue, typeValue, childIdValues, childPnumValues ); + newBundle.setId( idValue ); + newBundle.setDir( dirValue ); + newBundle.setChaufPnum( chaufPnumValue ); + newBundle.setChaufPersType( chaufPtypeValue ); + newBundle.setChaufPid( chaufPidValue ); + newBundle.setSchoolMazs( schoolMazsValues ); + newBundle.setSchoolDists( schoolDistValues ); + newBundle.setDepartHome( departHomeValue ); + newBundle.setArriveHome( arriveHomeValue ); + newBundle.setDepartWork( departWorkValue ); + newBundle.setArriveWork( arriveWorkValue ); + + return newBundle; + + } + */ +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java new file mode 100644 index 0000000..f8c76fb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java @@ -0,0 +1,1716 @@ +/* +* The school-escort model was designed by PB (Gupta, Vovsha, et al) +* as part of the Maricopa Association of Governments (MAG) +* Activity-based Travel Model Development project. +* +* This source code, which implements the school escort model, +* was written exclusively for and funded by MAG as part of the +* same project; therefore, per their contract, the +* source code belongs to MAG and can only be used with their +* permission. +* +* It is being adapted for the Southern Oregon ABM by PB & RSG +* with permission from MAG and all references to +* the school escort model as well as source code adapted from this +* original code should credit MAG's role in its development. +* +* The escort model and source code should not be transferred to or +* adapted for other agencies or used in other projects without +* expressed permission from MAG. +*/ + +package org.sandag.abm.ctramp; + +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.matrix.Matrix; +import com.pb.common.util.IndexSort; + + + + +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; + + + +public class SchoolEscortingDmu implements VariableTable { + + private Logger logger = Logger.getLogger( SchoolEscortingDmu.class ); + + private static final float DROP_OFF_DURATION = 5.0f; + private static final float PICK_UP_DURATION = 10.0f; + private static final float MINUTES_PER_MILE = 2.0f; + + + private Household hhObj; + + private float[] distHomeSchool; + private float[] timeHomeSchool; + private float[] distSchoolHome; + private float[] timeSchoolHome; + + //for each cheaffeur + private float[] distHomeMandatory; + private float[] timeHomeMandatory; + private float[] distMandatoryHome; + private float[] timeMandatoryHome; + + private float[][] distSchoolSchool; + private float[][] distSchoolMandatory; + private float[][] distMandatorySchool; + + private Person[] escortees; + + + private int[] escorteeIds; + private int[] escorteePnums; + private int[] escorteeAge; + + private int[] escorteeSchoolLoc; + private int[] escorteeSchoolAtHome; + private int[] escorteeDepartForSchool; + private int[] escorteeDepartFromSchool; + private int numChildrenTravelingToSchool; + + private Person[] chauffers; + private int[] chauffeurPnums; + private int[] chauffeurPids; + private int[] chauffeurAge; + private int[] chauffeurGender; + private int[] chauffeurPersonType; + private int[] chauffeurDap; + private int[] chauffeurMandatoryLoc; + private int[] chauffeurDepartForMandatory; + private int[] chauffeurDepartFromMandatory; + private int numPotentialChauffeurs; + + private int[][][] chaufExtents; + + private int chosenObEscortType1; + private int chosenObEscortType2; + private int chosenObEscortType3; + private int chosenObChauf1; + private int chosenObChauf2; + private int chosenObChauf3; + private int potentialObChauf1; + private int potentialObChauf2; + + private int chosenIbEscortType1; + private int chosenIbEscortType2; + private int chosenIbEscortType3; + private int chosenIbChauf1; + private int chosenIbChauf2; + private int chosenIbChauf3; + private int potentialIbChauf1; + private int potentialIbChauf2; + + private MgraDataManager mgraDataManager; + + private double[][] distanceArray; + + private int[][] altBundleIncidence; + + private Map methodIndexMap; + + + + /** + * Create the DMU by passing in... + * @param MgraDataManager mgraDataManager + * @param Matrix distanceMatrix + */ + public SchoolEscortingDmu(MgraDataManager mgraDataManager, double[][] distanceArray) { + + this.mgraDataManager = mgraDataManager; + this.distanceArray = distanceArray; + + chauffeurPnums = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurPids = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurAge = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurGender = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurPersonType = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurDap = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurMandatoryLoc = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurDepartForMandatory = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + chauffeurDepartFromMandatory = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + distHomeMandatory = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + timeHomeMandatory = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + distMandatoryHome = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + timeMandatoryHome = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + + escorteeIds = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteePnums = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteeAge = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteeSchoolLoc = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteeSchoolAtHome = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteeDepartForSchool = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + escorteeDepartFromSchool = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + distHomeSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + timeHomeSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + distSchoolHome = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + timeSchoolHome = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + + distSchoolSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1][SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + distSchoolMandatory = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1][SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; + distMandatorySchool = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1][SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; + + setupMethodIndexMap(); + } + + + public void setAltTableBundleIncidence( int[][] altBundleIncidence ) { + + this.altBundleIncidence = altBundleIncidence; + + } + + /** + * Set attributes of the potential chauffeurs + * @param numPotential Number of potential chauffeurs (size of adults array) + * @param adults Ordered array of chauffeurs + * @param mandatoryMazs Array size of chauffeurs, holding MAZ of last mandatory tour + * @param mandatoryDeparts Array size of chauffeurs, holding home departure period of last mandatory tour + * @param mandatoryReturns Array size of chauffeurs, holding work departure period of last mandatory tour + * @param chaufExtents Array size of chauffeurs, + */ + public void setChaufferAttributes( int numPotential, Person[] adults, int[] mandatoryMazs, int[] mandatoryDeparts, int[] mandatoryReturns, int[][][] chaufExtents ) { + + this.chaufExtents = chaufExtents; + + chauffers = adults; + numPotentialChauffeurs = numPotential; + + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) { + chauffeurAge[i] = 0; + chauffeurGender[i] = 0; + chauffeurPersonType[i] = 0; + chauffeurDap[i] = 0; + chauffeurMandatoryLoc[i] = 0; + chauffeurDepartForMandatory[i] = 0; + chauffeurDepartFromMandatory[i] = 0; + chauffeurPnums[i] = 0; + chauffeurPids[i] = 0; + } + else { + chauffeurPnums[i] = chauffers[i].getPersonNum(); + chauffeurPids[i] = chauffers[i].getPersonId(); + chauffeurAge[i] = chauffers[i].getAge(); + chauffeurGender[i] = chauffers[i].getGender(); + chauffeurPersonType[i] = chauffers[i].getPersonTypeNumber(); + chauffeurDap[i] = chauffers[i].getCdapIndex(); + if ( mandatoryMazs[i] == 0 ) { + chauffeurMandatoryLoc[i] = 0; + chauffeurDepartForMandatory[i] = 0; + chauffeurDepartFromMandatory[i] = 0; + } + else { + chauffeurMandatoryLoc[i] = mandatoryMazs[i]; + chauffeurDepartForMandatory[i] = mandatoryDeparts[i]; + chauffeurDepartFromMandatory[i] = mandatoryReturns[i]; + } + } + } + } + + /** + * Set attributes for escortees. + * + * @param numPotential Number of potential escortees (children traveling to school). + * @param children Person array of potential escortees + * @param schoolAtHome An array for each person indicating if they are schooled at home + * @param schoolMazs An array of school MAZs for each person + * @param schoolDeparts An array of school tour outbound periods + * @param schoolReturns an array of school tour return periods + */ + public void setEscorteeAttributes( int numPotential, Person[] children, int[] schoolAtHome, int[] schoolMazs, int[] schoolDeparts, int[] schoolReturns ) { + + escortees = children; + numChildrenTravelingToSchool = numPotential; + + for ( int i=1; i < escortees.length; i++ ) { + if ( escortees[i] == null || schoolMazs[i] == 0 ) { + escorteeIds[i] = 0; + escorteePnums[i] = 0; + escorteeAge[i] = 0; + escorteeSchoolLoc[i] = 0; + escorteeSchoolAtHome[i] = 0; + escorteeDepartForSchool[i] = 0; + escorteeDepartFromSchool[i] = 0; + } + else { + escorteeIds[i] = i; + escorteePnums[i] = escortees[i].getPersonNum(); + escorteeAge[i] = escortees[i].getAge(); + escorteeSchoolLoc[i] = schoolMazs[i]; + escorteeSchoolAtHome[i] = schoolAtHome[i]; + escorteeDepartForSchool[i] = schoolDeparts[i]; + escorteeDepartFromSchool[i] = schoolReturns[i]; + } + + } + + } + + /** + * Sets distance time attributes for combinations of chauffeur mandatory locations and escortee school locations. + * @param hhObj + * @param distanceArray + */ + public void setDistanceTimeAttributes( Household hhObj, double[][] distanceArray ) { + + this.hhObj = hhObj; + + int homeMaz = hhObj.getHhTaz(); + int homeTaz = mgraDataManager.getTaz(homeMaz); + + // compute times and distances from "home to work" and from "work to home" by traversing the chain of business locations for work tours + // to/from the primary work location for work tours and the chain that may include a work location for school tours. + + //for each chauffeur + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) { + distHomeMandatory[i] = 0; + timeHomeMandatory[i] = 0; + distMandatoryHome[i] = 0; + timeMandatoryHome[i] = 0; + } + else { + if ( chauffeurMandatoryLoc[i] > 0 ) { + + distHomeMandatory[i] = 0; + timeHomeMandatory[i] = 0; + distMandatoryHome[i] = 0; + timeMandatoryHome[i] = 0; + + //the MAG model would traverse all the activities on the tour, skipping non-work and non-school tours, and skipping + //non-work activities, and sum up the distance from home to each work activity. In the case of ORRAMP, only the work + //primary destination is known, so the method has been re-written accordingly to use work primary destination for workers. + + int mandatoryMaz = chauffeurMandatoryLoc[i]; + int mandatoryTaz = mgraDataManager.getTaz(mandatoryMaz); + + distHomeMandatory[i] = (float) distanceArray[homeTaz][ mandatoryTaz]; + timeHomeMandatory[i] = MINUTES_PER_MILE * (float) distanceArray[homeTaz][mandatoryTaz]; + distMandatoryHome[i] = (float) distanceArray[mandatoryTaz][homeTaz]; + timeMandatoryHome[i] = MINUTES_PER_MILE * (float) distanceArray[mandatoryTaz][homeTaz]; + + } + else { + distHomeMandatory[i] = 0; + timeHomeMandatory[i] = 0; + distMandatoryHome[i] = 0; + timeMandatoryHome[i] = 0; + } + } + } + + //iterating through potential escortees (i) + for ( int i=1; i < escortees.length; i++ ) { + if ( escortees[i] == null ) { + distHomeSchool[i] = 0; + timeHomeSchool[i] = 0; + distSchoolHome[i] = 0; + timeSchoolHome[i] = 0; + for ( int j=1; j < chauffeurPnums.length; j++ ) { + distSchoolMandatory[i][j] = 0; + distMandatorySchool[j][i] = 0; + } + for ( int j=1; j < escorteePnums.length; j++ ) + distSchoolSchool[i][j] = 0; + } + else { + if ( escorteeSchoolLoc[i] > 0 ) { + + int schoolTaz = mgraDataManager.getTaz(escorteeSchoolLoc[i]); + + distHomeSchool[i] = (float) distanceArray[homeTaz][schoolTaz]; + timeHomeSchool[i] = MINUTES_PER_MILE * (float) distanceArray[homeTaz][schoolTaz]; + distSchoolHome[i] = (float) distanceArray[schoolTaz][ homeTaz]; + timeSchoolHome[i] = MINUTES_PER_MILE * (float) distanceArray[schoolTaz][homeTaz]; + + //iterating through potential chauffeurs (j) + for ( int j=1; j < chauffeurPnums.length; j++ ) { + distSchoolMandatory[i][j] = 0; + distMandatorySchool[j][i] = 0; + if ( chauffeurMandatoryLoc[j] > 0 ) { + int mandatoryMaz = chauffeurMandatoryLoc[j]; + int mandatoryTaz = mgraDataManager.getTaz(mandatoryMaz); + distSchoolMandatory[i][j] = (float) distanceArray[schoolTaz][mandatoryTaz]; + distMandatorySchool[j][i] = (float) distanceArray[mandatoryTaz][schoolTaz]; + } + } + + for ( int j=1; j < escorteePnums.length; j++ ) { + distSchoolSchool[i][j] = 0; + if ( escorteeSchoolLoc[j] > 0 && escorteeSchoolLoc[j] != escorteeSchoolLoc[i] ) { + int schoolTazJ = mgraDataManager.getTaz(escorteeSchoolLoc[j]); + distSchoolSchool[i][j] = (float) distanceArray[schoolTaz][schoolTazJ]; + } + } + } + else { + distHomeSchool[i] = 0; + timeHomeSchool[i] = 0; + distSchoolHome[i] = 0; + timeSchoolHome[i] = 0; + for ( int j=1; j < chauffeurPnums.length; j++ ) { + distSchoolMandatory[i][j] = 0; + distMandatorySchool[j][i] = 0; + } + for ( int j=1; j < escorteePnums.length; j++ ) + distSchoolSchool[i][j] = 0; + } + } + + } + + } + + + public void setOutboundEscortType1( int chosenObEscortType ) { + chosenObEscortType1 = chosenObEscortType; + } + + public void setOutboundEscortType2( int chosenObEscortType ) { + chosenObEscortType2 = chosenObEscortType; + } + + public void setOutboundEscortType3( int chosenObEscortType ) { + chosenObEscortType3 = chosenObEscortType; + } + + public void setOutboundChauffeur1( int chosenObChauf ) { + chosenObChauf1 = chosenObChauf; + } + + public void setOutboundChauffeur2( int chosenObChauf ) { + chosenObChauf2 = chosenObChauf; + } + + public void setOutboundChauffeur3( int chosenObChauf ) { + chosenObChauf3 = chosenObChauf; + } + + public void setOutboundPotentialChauffeur1( int chaufPnum ) { + potentialObChauf1 = chaufPnum; + } + + public void setOutboundPotentialChauffeur2( int chaufPnum ) { + potentialObChauf2 = chaufPnum; + } + + + + public void setInboundEscortType1( int chosenIbEscortType ) { + chosenIbEscortType1 = chosenIbEscortType; + } + + public void setInboundEscortType2( int chosenIbEscortType ) { + chosenIbEscortType2 = chosenIbEscortType; + } + + public void setInboundEscortType3( int chosenIbEscortType ) { + chosenIbEscortType3 = chosenIbEscortType; + } + + public void setInboundChauffeur1( int chosenIbChauf ) { + chosenIbChauf1 = chosenIbChauf; + } + + public void setInboundChauffeur2( int chosenIbChauf ) { + chosenIbChauf2 = chosenIbChauf; + } + + public void setInboundChauffeur3( int chosenIbChauf ) { + chosenIbChauf3 = chosenIbChauf; + } + + public void setInboundPotentialChauffeur1( int chaufPnum ) { + potentialIbChauf1 = chaufPnum; + } + + public void setInboundPotentialChauffeur2( int chaufPnum ) { + potentialIbChauf2 = chaufPnum; + } + + public int[] getChauffeurPnums() { + return chauffeurPnums; + } + + public int[] getChauffeurDepartForMandatory() { + return chauffeurDepartForMandatory; + } + + public int[] getChauffeurDepartFromMandatory() { + return chauffeurDepartFromMandatory; + } + + public int[] getEscorteePnums() { + return escorteePnums; + } + + public int[] getEscorteeDepartForSchool() { + return escorteeDepartForSchool; + } + + public int[] getEscorteeDepartFromSchool() { + return escorteeDepartFromSchool; + } + + public int[] getEscorteeSchoolAtHome() { + return escorteeSchoolAtHome; + } + + public int[] getEscorteeDistToSchool() { + int[] tempDist = new int[distHomeSchool.length]; + for ( int i=1; i < distHomeSchool.length; i++ ) + tempDist[i] = (int)(distHomeSchool[i] * 100); + return tempDist; + } + + public int[] getEscorteeDistFromSchool() { + int[] tempDist = new int[distSchoolHome.length]; + for ( int i=1; i < distSchoolHome.length; i++ ) + tempDist[i] = (int)(distSchoolHome[i] * 100); + return tempDist; + } + + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put( "getChild1Pnum", 1 ); + methodIndexMap.put( "getChild2Pnum", 2 ); + methodIndexMap.put( "getChild3Pnum", 3 ); + methodIndexMap.put( "getAdult1Pnum", 4 ); + methodIndexMap.put( "getAdult2Pnum", 5 ); + methodIndexMap.put( "getAgeChild1", 6 ); + methodIndexMap.put( "getAgeChild2", 7 ); + methodIndexMap.put( "getAgeChild3", 8 ); + methodIndexMap.put( "getSchoolMazChild1", 9 ); + methodIndexMap.put( "getSchoolMazChild2", 10 ); + methodIndexMap.put( "getSchoolMazChild3", 11 ); + methodIndexMap.put( "getDistHomeSchool1", 12 ); + methodIndexMap.put( "getDistHomeSchool2", 13 ); + methodIndexMap.put( "getDistHomeSchool3", 14 ); + methodIndexMap.put( "getDistSchoolHome1", 15 ); + methodIndexMap.put( "getDistSchoolHome2", 16 ); + methodIndexMap.put( "getDistSchoolHome3", 17 ); + methodIndexMap.put( "getTimeHomeSchool1", 18 ); + methodIndexMap.put( "getTimeHomeSchool2", 19 ); + methodIndexMap.put( "getTimeHomeSchool3", 20 ); + methodIndexMap.put( "getTimeSchoolHome1", 21 ); + methodIndexMap.put( "getTimeSchoolHome2", 22 ); + methodIndexMap.put( "getTimeSchoolHome3", 23 ); + methodIndexMap.put( "getDepartHomeSchool1", 24 ); + methodIndexMap.put( "getDepartHomeSchool2", 25 ); + methodIndexMap.put( "getDepartHomeSchool3", 26 ); + methodIndexMap.put( "getDepartSchoolHome1", 27 ); + methodIndexMap.put( "getDepartSchoolHome2", 28 ); + methodIndexMap.put( "getDepartSchoolHome3", 29 ); + methodIndexMap.put( "getGenderAdult1", 30 ); + methodIndexMap.put( "getGenderAdult2", 31 ); + methodIndexMap.put( "getPersonTypeAdult1", 32 ); + methodIndexMap.put( "getPersonTypeAdult2", 33 ); + methodIndexMap.put( "getAgeAdult1", 34 ); + methodIndexMap.put( "getAgeAdult2", 35 ); + methodIndexMap.put( "getDepartHomeWorkAdult1", 36 ); + methodIndexMap.put( "getDepartHomeWorkAdult2", 37 ); + methodIndexMap.put( "getDepartWorkHomeAdult1", 38 ); + methodIndexMap.put( "getDepartWorkHomeAdult2", 39 ); + methodIndexMap.put( "getDapAdult1", 40 ); + methodIndexMap.put( "getDapAdult2", 41 ); + methodIndexMap.put( "getDistHomeWork1", 42 ); + methodIndexMap.put( "getDistHomeWork2", 43 ); + methodIndexMap.put( "getTimeHomeWork1", 44 ); + methodIndexMap.put( "getTimeHomeWork2", 45 ); + methodIndexMap.put( "getDistWorkHome1", 46 ); + methodIndexMap.put( "getDistWorkHome2", 47 ); + methodIndexMap.put( "getTimeWorkHome1", 48 ); + methodIndexMap.put( "getTimeWorkHome2", 49 ); + methodIndexMap.put( "getDistSchool1School2", 50 ); + methodIndexMap.put( "getDistSchool1School3", 51 ); + methodIndexMap.put( "getDistSchool2School3", 52 ); + methodIndexMap.put( "getDistSchool1Work1", 53 ); + methodIndexMap.put( "getDistSchool1Work2", 54 ); + methodIndexMap.put( "getDistSchool2Work1", 55 ); + methodIndexMap.put( "getDistSchool2Work2", 56 ); + methodIndexMap.put( "getDistSchool3Work1", 57 ); + methodIndexMap.put( "getDistSchool3Work2", 58 ); + methodIndexMap.put( "getDistWork1School1", 59 ); + methodIndexMap.put( "getDistWork2School1", 60 ); + methodIndexMap.put( "getDistWork1School2", 61 ); + methodIndexMap.put( "getDistWork2School2", 62 ); + methodIndexMap.put( "getDistWork1School3", 63 ); + methodIndexMap.put( "getDistWork2School3", 64 ); + methodIndexMap.put( "getIncome", 65 ); + methodIndexMap.put( "getNumAutosInHH", 66 ); + methodIndexMap.put( "getNumWorkersInHH", 67 ); + methodIndexMap.put( "getNumChildrenWithSchoolOutsideOfHomeAndDap1", 68 ); + methodIndexMap.put( "getNumAdultsinHHDap12", 69 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild1Chauffeur1", 70 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild1Chauffeur2", 71 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild2Chauffeur1", 72 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild2Chauffeur2", 73 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild3Chauffeur1", 74 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild3Chauffeur2", 75 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild12Chauffeur1", 76 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild12Chauffeur2", 77 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild13Chauffeur1", 78 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild13Chauffeur2", 79 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild23Chauffeur1", 80 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild23Chauffeur2", 81 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild123Chauffeur1", 82 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild123Chauffeur2", 83 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild1Chauffeur1", 84 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild1Chauffeur2", 85 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild2Chauffeur1", 86 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild2Chauffeur2", 87 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild3Chauffeur1", 88 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild3Chauffeur2", 89 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild12Chauffeur1", 90 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild12Chauffeur2", 91 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild13Chauffeur1", 92 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild13Chauffeur2", 93 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild23Chauffeur1", 94 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild23Chauffeur2", 95 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild123Chauffeur1", 96 ); + methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild123Chauffeur2", 97 ); + methodIndexMap.put( "getInboundEscortType1", 98 ); + methodIndexMap.put( "getInboundEscortType2", 99 ); + methodIndexMap.put( "getInboundEscortType3", 100 ); + methodIndexMap.put( "getInboundChauffeur1", 101 ); + methodIndexMap.put( "getInboundChauffeur2", 102 ); + methodIndexMap.put( "getInboundChauffeur3", 103 ); + methodIndexMap.put( "getOutboundEscortType1", 104 ); + methodIndexMap.put( "getOutboundEscortType2", 105 ); + methodIndexMap.put( "getOutboundEscortType3", 106 ); + methodIndexMap.put( "getOutboundChauffeur1", 107 ); + methodIndexMap.put( "getOutboundChauffeur2", 108 ); + methodIndexMap.put( "getOutboundChauffeur3", 109 ); + methodIndexMap.put( "getInboundPotentialChauffeur1", 110 ); + methodIndexMap.put( "getInboundPotentialChauffeur2", 111 ); + methodIndexMap.put( "getOutboundPotentialChauffeur1", 112 ); + methodIndexMap.put( "getOutboundPotentialChauffeur2", 113 ); + methodIndexMap.put( "getTravelTimeWork1School1", 114 ); + methodIndexMap.put( "getTravelTimeWork2School1", 115 ); + methodIndexMap.put( "getTravelTimeWork1School2", 116 ); + methodIndexMap.put( "getTravelTimeWork2School2", 117 ); + methodIndexMap.put( "getTravelTimeWork1School3", 118 ); + methodIndexMap.put( "getTravelTimeWork2School3", 119 ); + methodIndexMap.put( "getTravelTimeWork1Home", 120 ); + methodIndexMap.put( "getTravelTimeWork2Home", 121 ); + methodIndexMap.put( "getAvailabilityForMultipleBundlesOutbound", 122 ); + methodIndexMap.put( "getAvailabilityForMultipleBundlesInbound", 123 ); + methodIndexMap.put( "getAvailabilityForInboundChauf1WithOutboundBundles", 124); + methodIndexMap.put( "getAvailabilityForInboundChauf2WithOutboundBundles", 125); + methodIndexMap.put( "getTravelTimeHomeSchool1", 126 ); + methodIndexMap.put( "getTravelTimeHomeSchool2", 127 ); + methodIndexMap.put( "getTravelTimeHomeSchool3", 128 ); + methodIndexMap.put( "getTravelTimeSchool1Home", 129 ); + methodIndexMap.put( "getTravelTimeSchool2Home", 130 ); + methodIndexMap.put( "getTravelTimeSchool3Home", 131 ); + methodIndexMap.put( "getAvailabilityForOutboundChauf1WithInboundBundles", 132); + methodIndexMap.put( "getAvailabilityForOutboundChauf2WithInboundBundles", 133); + + } + + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return escorteePnums[1]; + case 2: + return escorteePnums[2]; + case 3: + return escorteePnums[3]; + case 4: + return chauffeurPnums[1]; + case 5: + return chauffeurPnums[2]; + case 6: + return escorteeAge[1]; + case 7: + return escorteeAge[2]; + case 8: + return escorteeAge[3]; + case 9: + return escorteeSchoolLoc[1]; + case 10: + return escorteeSchoolLoc[2]; + case 11: + return escorteeSchoolLoc[3]; + case 12: + return distHomeSchool[1]; + case 13: + return distHomeSchool[2]; + case 14: + return distHomeSchool[3]; + case 15: + return distSchoolHome[1]; + case 16: + return distSchoolHome[2]; + case 17: + return distSchoolHome[3]; + case 18: + return timeHomeSchool[1]; + case 19: + return timeHomeSchool[2]; + case 20: + return timeHomeSchool[3]; + case 21: + return timeSchoolHome[1]; + case 22: + return timeSchoolHome[2]; + case 23: + return timeSchoolHome[3]; + case 24: + return escorteeDepartForSchool[1]; + case 25: + return escorteeDepartForSchool[2]; + case 26: + return escorteeDepartForSchool[3]; + case 27: + return escorteeDepartFromSchool[1]; + case 28: + return escorteeDepartFromSchool[2]; + case 29: + return escorteeDepartFromSchool[3]; + case 30: + return chauffeurGender[1]; + case 31: + return chauffeurGender[2]; + case 32: + return chauffeurPersonType[1]; + case 33: + return chauffeurPersonType[2]; + case 34: + return chauffeurAge[1]; + case 35: + return chauffeurAge[2]; + case 36: + return chauffeurDepartForMandatory[1]; + case 37: + return chauffeurDepartForMandatory[2]; + case 38: + return chauffeurDepartFromMandatory[1]; + case 39: + return chauffeurDepartFromMandatory[2]; + case 40: + return chauffeurDap[1]; + case 41: + return chauffeurDap[2]; + case 42: + return distHomeMandatory[1]; + case 43: + return distHomeMandatory[2]; + case 44: + return timeHomeMandatory[1]; + case 45: + return timeHomeMandatory[2]; + case 46: + return distMandatoryHome[1]; + case 47: + return distMandatoryHome[2]; + case 48: + return timeMandatoryHome[1]; + case 49: + return timeMandatoryHome[2]; + case 50: + return distSchoolSchool[1][2]; + case 51: + return distSchoolSchool[1][3]; + case 52: + return distSchoolSchool[2][3]; + case 53: + return distSchoolMandatory[1][1]; + case 54: + return distSchoolMandatory[1][2]; + case 55: + return distSchoolMandatory[2][1]; + case 56: + return distSchoolMandatory[2][2]; + case 57: + return distSchoolMandatory[3][1]; + case 58: + return distSchoolMandatory[3][2]; + case 59: + return distMandatorySchool[1][1]; + case 60: + return distMandatorySchool[1][2]; + case 61: + return distMandatorySchool[1][3]; + case 62: + return distMandatorySchool[2][1]; + case 63: + return distMandatorySchool[2][2]; + case 64: + return distMandatorySchool[2][3]; + case 65: + return hhObj.getIncomeInDollars(); + case 66: + return hhObj.getAutosOwned(); + case 67: + return hhObj.getWorkers(); + case 68: + return numChildrenTravelingToSchool; + case 69: + return numPotentialChauffeurs; + case 70: + return Math.max( distHomeSchool[1] + distSchoolMandatory[1][1] - distHomeMandatory[1], 0 ); + case 71: + return Math.max( distHomeSchool[1] + distSchoolMandatory[1][2] - distHomeMandatory[2], 0 ); + case 72: + return Math.max( distHomeSchool[2] + distSchoolMandatory[2][1] - distHomeMandatory[1], 0 ); + case 73: + return Math.max( distHomeSchool[2] + distSchoolMandatory[2][2] - distHomeMandatory[2], 0 ); + case 74: + return Math.max( distHomeSchool[3] + distSchoolMandatory[3][1] - distHomeMandatory[1], 0 ); + case 75: + return Math.max( distHomeSchool[3] + distSchoolMandatory[3][2] - distHomeMandatory[2], 0 ); + case 76: + return getAbsoluteDeviationDistanceOutboundChild12Chauffeur1(); + case 77: + return getAbsoluteDeviationDistanceOutboundChild12Chauffeur2(); + case 78: + return getAbsoluteDeviationDistanceOutboundChild13Chauffeur1(); + case 79: + return getAbsoluteDeviationDistanceOutboundChild13Chauffeur2(); + case 80: + return getAbsoluteDeviationDistanceOutboundChild23Chauffeur1(); + case 81: + return getAbsoluteDeviationDistanceOutboundChild23Chauffeur2(); + case 82: + return getAbsoluteDeviationDistanceOutboundChild123Chauffeur1(); + case 83: + return getAbsoluteDeviationDistanceOutboundChild123Chauffeur2(); + case 84: + return Math.max( distMandatorySchool[1][1] + distSchoolHome[1] - distMandatoryHome[1], 0 ); + case 85: + return Math.max( distMandatorySchool[2][1] + distSchoolHome[1] - distMandatoryHome[2], 0 ); + case 86: + return Math.max( distMandatorySchool[1][2] + distSchoolHome[2] - distMandatoryHome[1], 0 ); + case 87: + return Math.max( distMandatorySchool[2][2] + distSchoolHome[2] - distMandatoryHome[2], 0 ); + case 88: + return Math.max( distMandatorySchool[1][3] + distSchoolHome[3] - distMandatoryHome[1], 0 ); + case 89: + return Math.max( distMandatorySchool[2][3] + distSchoolHome[3] - distMandatoryHome[2], 0 ); + case 90: + return getAbsoluteDeviationDistanceInboundChild12Chauffeur1(); + case 91: + return getAbsoluteDeviationDistanceInboundChild12Chauffeur2(); + case 92: + return getAbsoluteDeviationDistanceInboundChild13Chauffeur1(); + case 93: + return getAbsoluteDeviationDistanceInboundChild13Chauffeur2(); + case 94: + return getAbsoluteDeviationDistanceInboundChild23Chauffeur1(); + case 95: + return getAbsoluteDeviationDistanceInboundChild23Chauffeur2(); + case 96: + return getAbsoluteDeviationDistanceInboundChild123Chauffeur1(); + case 97: + return getAbsoluteDeviationDistanceInboundChild123Chauffeur2(); + case 98: + return chosenIbEscortType1; + case 99: + return chosenIbEscortType2; + case 100: + return chosenIbEscortType3; + case 101: + return chosenIbChauf1; + case 102: + return chosenIbChauf2; + case 103: + return chosenIbChauf3; + case 104: + return chosenObEscortType1; + case 105: + return chosenObEscortType2; + case 106: + return chosenObEscortType3; + case 107: + return chosenObChauf1; + case 108: + return chosenObChauf2; + case 109: + return chosenObChauf3; + case 110: + return potentialIbChauf1; + case 111: + return potentialIbChauf2; + case 112: + return potentialObChauf1; + case 113: + return potentialObChauf2; + case 114: + return (int)( ( ( distMandatorySchool[1][1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 115: + return (int)( ( ( distMandatorySchool[2][1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 116: + return (int)( ( ( distMandatorySchool[1][2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 117: + return (int)( ( ( distMandatorySchool[2][2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 118: + return (int)( ( ( distMandatorySchool[1][3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 119: + return (int)( ( ( distMandatorySchool[2][3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 120: + return (int)( ( ( distMandatoryHome[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 121: + return (int)( ( ( distMandatoryHome[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 122: + return getAvailabilityForMultipleBundlesOutbound( arrayIndex ); + case 123: + return getAvailabilityForMultipleBundlesInbound( arrayIndex ); + case 124: + return getAvailabilityForInboundChauf1WithOutboundBundles( arrayIndex ); + case 125: + return getAvailabilityForInboundChauf2WithOutboundBundles( arrayIndex ); + case 126: + return (int)( ( ( distHomeSchool[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 127: + return (int)( ( ( distHomeSchool[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 128: + return (int)( ( ( distHomeSchool[3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 129: + return (int)( ( ( distSchoolHome[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 130: + return (int)( ( ( distSchoolHome[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 131: + return (int)( ( ( distSchoolHome[3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); + case 132: + return getAvailabilityForOutboundChauf1WithInboundBundles( arrayIndex ); + case 133: + return getAvailabilityForOutboundChauf2WithInboundBundles( arrayIndex ); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + + + private float getAbsoluteDeviationDistanceOutboundChild12Chauffeur1() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolMandatory[2][1]; + float d2 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolMandatory[1][1]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceInboundChild12Chauffeur1() { + float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][2] + distSchoolHome[2]; + float d2 = distMandatorySchool[1][2] + distSchoolSchool[2][1] + distSchoolHome[1]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceOutboundChild12Chauffeur2() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolMandatory[2][2]; + float d2 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolMandatory[1][2]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceInboundChild12Chauffeur2() { + float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][2] + distSchoolHome[2]; + float d2 = distMandatorySchool[2][2] + distSchoolSchool[2][1] + distSchoolHome[1]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceOutboundChild13Chauffeur1() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolMandatory[3][1]; + float d2 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolMandatory[1][1]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceInboundChild13Chauffeur1() { + float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[1][3] + distSchoolSchool[3][1] + distSchoolHome[1]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceOutboundChild13Chauffeur2() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolMandatory[3][2]; + float d2 = distHomeSchool[3] + distSchoolSchool[3][1] - distSchoolMandatory[1][2]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceInboundChild13Chauffeur2() { + float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceOutboundChild23Chauffeur1() { + float d1 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolMandatory[3][1]; + float d2 = distHomeSchool[3] + distSchoolSchool[3][2] - distSchoolMandatory[2][1]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceInboundChild23Chauffeur1() { + float d1 = distMandatorySchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; + return Math.min( d1, d2 ) - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceOutboundChild23Chauffeur2() { + float d1 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolMandatory[3][2]; + float d2 = distHomeSchool[3] + distSchoolSchool[3][2] - distSchoolMandatory[2][2]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceInboundChild23Chauffeur2() { + float d1 = distMandatorySchool[2][2] + distSchoolSchool[2][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[2][3] + distSchoolSchool[3][2] + distSchoolHome[2]; + return Math.min( d1, d2 ) - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceOutboundChild123Chauffeur1() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolMandatory[3][1]; + float d2 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolMandatory[2][1]; + float d3 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolMandatory[3][1]; + float d4 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolMandatory[1][1]; + float d5 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolMandatory[2][1]; + float d6 = distHomeSchool[3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolMandatory[1][1]; + float d = Math.min( d1, d2 ); + d = Math.min( d, d3 ); + d = Math.min( d, d4 ); + d = Math.min( d, d5 ); + d = Math.min( d, d6 ); + return d - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceInboundChild123Chauffeur1() { + float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[1][1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; + float d3 = distMandatorySchool[1][2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; + float d4 = distMandatorySchool[1][2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; + float d5 = distMandatorySchool[1][3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolHome[2]; + float d6 = distMandatorySchool[1][3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolHome[1]; + float d = Math.min( d1, d2 ); + d = Math.min( d, d3 ); + d = Math.min( d, d4 ); + d = Math.min( d, d5 ); + d = Math.min( d, d6 ); + return d - distHomeMandatory[1]; + } + + private float getAbsoluteDeviationDistanceOutboundChild123Chauffeur2() { + float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolMandatory[3][2]; + float d2 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolMandatory[2][2]; + float d3 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolMandatory[3][2]; + float d4 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolMandatory[1][2]; + float d5 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolMandatory[2][2]; + float d6 = distHomeSchool[3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolMandatory[1][2]; + float d = Math.min( d1, d2 ); + d = Math.min( d, d3 ); + d = Math.min( d, d4 ); + d = Math.min( d, d5 ); + d = Math.min( d, d6 ); + return d - distHomeMandatory[2]; + } + + private float getAbsoluteDeviationDistanceInboundChild123Chauffeur2() { + float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; + float d2 = distMandatorySchool[2][1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; + float d3 = distMandatorySchool[2][2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; + float d4 = distMandatorySchool[2][2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; + float d5 = distMandatorySchool[2][3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolHome[2]; + float d6 = distMandatorySchool[2][3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolHome[1]; + float d = Math.min( d1, d2 ); + d = Math.min( d, d3 ); + d = Math.min( d, d4 ); + d = Math.min( d, d5 ); + d = Math.min( d, d6 ); + return d - distHomeMandatory[2]; + } + + + /** + * This method should only be called for relevant alternatives - those with multiple bundles for a single chauffeur. + */ + private int getAvailabilityForMultipleBundlesOutbound( int alt ) { + + // set availability to 0 if unavailable, or 1 if available + int availabilityForMultipleBundles = 1; + + List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); + + //check the number of bundles + int chaufIndex = 0; + if ( altChaufBundles[1].size() > 1 ) //more than one bundle for the first chauffeur + chaufIndex = 1; + else if ( altChaufBundles[2].size() > 1 ) //more than one bundle for the second chauffeur + chaufIndex = 2; + else { + logger.fatal( "UEC method getAvailabilityForMultipleBundlesOutbound( alt=" + alt + " ) was called, but neither chauf has multiple escort bundles." ); + logger.fatal("Size of altChaufBundles[1] = "+altChaufBundles[1].size()); + logger.fatal("Size of altChaufBundles[2] = "+altChaufBundles[2].size()); + throw new RuntimeException( ); + } + + + // set the bundle depart intervals for all bundles and arrive back home intervals for pure escort only + for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { + + int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; + Arrays.fill( sortData, 999999999 ); + int[] children = bundleObj.getChildIds(); + for ( int j=0; j < children.length; j++ ) + sortData[children[j]] = escorteeDepartForSchool[children[j]]; + + int[] childrenOrder = IndexSort.indexSort( sortData ); + + int departHomeInterval = escorteeDepartForSchool[ childrenOrder[0] ]; + bundleObj.setDepartHome( departHomeInterval ); + + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( childrenOrder, children.length, DROP_OFF_DURATION ); + float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); + bundleObj.setArriveHome( arriveHomeInterval ); + } + + } + + + int[] chaufBundlesOrder = getChaufBundlesOrderOutbound( altChaufBundles[chaufIndex] ); + + for( int j=1; j < chaufBundlesOrder.length; j++ ) { + SchoolEscortingBundle bundleSubsequent = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); + SchoolEscortingBundle bundlePrevious = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j-1] ); + if ( bundleSubsequent.getDepartHome() <= bundlePrevious.getArriveHome() ) { + availabilityForMultipleBundles = 0; + break; + } + } + + return availabilityForMultipleBundles; + + } + + float convertIntervalToMinutes(int interval){ + + float minutes = interval * ModelStructure.TOD_INTERVAL_IN_MINUTES; + return minutes; + } + + int convertMinutesToInterval(float minutes){ + + int interval = (int) (minutes/ModelStructure.TOD_INTERVAL_IN_MINUTES); + interval = Math.min(interval,ModelStructure.MAX_TOD_INTERVAL); + return interval; + + } + + + /** + * This method should only be called for relevant alternatives - those with multiple bundles for a single chauffeur. + */ + private int getAvailabilityForMultipleBundlesInbound( int alt ) { + + // set availability to 0 if unavailable, or 1 if available + int availabilityForMultipleBundles = 1; + + List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); + + int chaufIndex = 0; + if ( altChaufBundles[1].size() > 1 ) + chaufIndex = 1; + else if ( altChaufBundles[2].size() > 1 ) + chaufIndex = 2; + else { + logger.error( "UEC method getAvailabilityForMultipleBundlesInbound( alt=" + alt + " ) was called, but neither chauf has multiple escort bundles." ); + throw new RuntimeException( ); + } + + + // set the bundle arrive intervals for all bundles and depart from home intervals for pure escort only + for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { + + int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; + Arrays.fill( sortData, 999999999 ); + int[] children = bundleObj.getChildIds(); + for ( int j=0; j < children.length; j++ ) + sortData[children[j]] = escorteeDepartFromSchool[children[j]]; + + int[] childrenOrder = IndexSort.indexSort( sortData ); + + int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; + + float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); + float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); + bundleObj.setArriveHome( arriveHomeInterval ); + + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[childrenOrder[0]] ); + float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; + int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); + bundleObj.setDepartHome( departHomeInterval ); + } + + } + + + int[] chaufBundlesOrder = getChaufBundlesOrderInbound( altChaufBundles[chaufIndex] ); + + for( int j=1; j < chaufBundlesOrder.length; j++ ) { + SchoolEscortingBundle bundleSubsequent = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); + SchoolEscortingBundle bundlePrevious = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j-1] ); + if ( bundleSubsequent.getDepartHome() <= bundlePrevious.getArriveHome() ) { + availabilityForMultipleBundles = 0; + break; + } + } + + return availabilityForMultipleBundles; + + } + + private int getAvailabilityForInboundChauf1WithOutboundBundles( int alt ) { + return getAvailabilityForChaufWithPreviousDirectionBundles( 1, SchoolEscortingModel.DIR_INBOUND, alt ); + } + + private int getAvailabilityForInboundChauf2WithOutboundBundles( int alt ) { + return getAvailabilityForChaufWithPreviousDirectionBundles( 2, SchoolEscortingModel.DIR_INBOUND, alt ); + } + + private int getAvailabilityForOutboundChauf1WithInboundBundles( int alt ) { + return getAvailabilityForChaufWithPreviousDirectionBundles( 1, SchoolEscortingModel.DIR_OUTBOUND, alt ); + } + + private int getAvailabilityForOutboundChauf2WithInboundBundles( int alt ) { + return getAvailabilityForChaufWithPreviousDirectionBundles( 2, SchoolEscortingModel.DIR_OUTBOUND, alt ); + } + + /** + * This method should only be called for for relevant alternatives - those with multiple bundles for a single chauffeur. + */ + private int getAvailabilityForChaufWithPreviousDirectionBundles( int chaufid, int dir, int alt ) { + + // set availability to 0 if unavailable, or 1 if available + int availability = 1; + + List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); + + // set the bundle depart and arrive intervals for all bundles for the alternative + for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufid] ) { + + // order the children by earliest pickup time and set arriveHome + int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; + Arrays.fill( sortData, 999999999 ); + int[] children = bundleObj.getChildIds(); + for ( int j=0; j < children.length; j++ ) { + sortData[children[j]] = escorteeDepartFromSchool[children[j]]; + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) + sortData[children[j]] = escorteeDepartForSchool[children[j]]; + else + sortData[children[j]] = escorteeDepartFromSchool[children[j]]; + } + + int[] childrenOrder = IndexSort.indexSort( sortData ); + + int chaufPnum = chauffeurPnums[chaufid]; + + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + + // set OB depart to earliest child's depart from home or either pure escort or ride sharing + int departHomeInterval = escorteeDepartForSchool[ childrenOrder[0] ]; + bundleObj.setDepartHome( departHomeInterval ); + + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + + float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( childrenOrder, children.length, DROP_OFF_DURATION ); + float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); + bundleObj.setArriveHome( arriveHomeInterval ); + + // neither end of the alternative window can overlap the reserved window + if ( ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) || + ( arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) ) + availability = 0; + // if the start of the alternative window is before the start of the reserved window, the end of the alternative window must also be before the start of the reserved window. + else if ( departHomeInterval < chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] ) + availability = 0; + // the start of the alternative window cannot be after the start of the reserved window + else if ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] ) + availability = 0; + } + else { + + if ( chauffeurDap[chaufid] != 1 ) { + availability = 0; + } + else { + + float numMinutes = getTimeInMinutesFromHomeThruAllSchoolsToWork( chauffeurMandatoryLoc[chaufid], childrenOrder, children.length, DROP_OFF_DURATION ); + float arriveWorkMinute = convertIntervalToMinutes( departHomeInterval ) + numMinutes; + int arriveWorkInterval = convertMinutesToInterval( arriveWorkMinute ); + + if ( ( departHomeInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) || + ( arriveWorkInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && arriveWorkInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) ) + availability = 0; + } + } + + } + else { + + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + + int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; + + float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[childrenOrder[0]] ); + float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; + int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); + + float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); + float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); + + // neither end of the alternative window can overlap the reserved window + if ( ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) || + ( arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) ) + availability = 0; + // the start of the alternative window cannot be before the end of the reserved window + else if ( departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) + availability = 0; + } + else { + + if ( chauffeurDap[chaufid] != 1 ) { + availability = 0; + } + else { + + int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; + + float workToFirstSchoolMinutes = getMazToMazTimeInMinutes( chauffeurMandatoryLoc[chaufid], escorteeSchoolLoc[childrenOrder[0]] ); + float departWorkMinute = convertIntervalToMinutes( departFromFirstSchoolInterval ) - workToFirstSchoolMinutes; + int departWorkInterval = convertMinutesToInterval( departWorkMinute ); + + float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); + float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); + + if ( ( departWorkInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && departWorkInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) || + ( arriveHomeInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) ) + availability = 0; + } + } + + } + + } + + + return availability; + + } + + + /** + * This method creates and returns the chosen bundles given the attributes of the chauffeur and escortees on the bundle and the type of bundle selected (pure escort vs rideshare) + * and the direction for the bundle (outbound versus return). + * + * @param alt The chosen alternative. + * @param chaufIndex The chauffeur to get the bundle for. + * @param dir Outbound or inbound. + * @return A fully coded bundle for the chauffeur given the chosen alternative. + */ + public SchoolEscortingBundle[] getChosenBundles( int alt, int chaufIndex, int dir ) { + + //an arraylist of school escorting bundles, dimensioned by each chauffeur (2) + List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); + + // set the bundle depart intervals for all bundles and arrive back home intervals for pure escort only + for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { + + //if the bundle direction is outbound, sort the escortees by departure period, else sort by arrival period + int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; + Arrays.fill( sortData, 999999999 ); + int[] altBundleChildIds = bundleObj.getChildIds(); + for ( int j=0; j < altBundleChildIds.length; j++ ) { + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) + sortData[altBundleChildIds[j]] = escorteeDepartForSchool[altBundleChildIds[j]]; + else + sortData[altBundleChildIds[j]] = escorteeDepartFromSchool[altBundleChildIds[j]]; + } + int[] altBundleChildrenOrder = IndexSort.indexSort( sortData ); + + //set the school locations and person numbers for each escortee in the order of escortees set above + int[] altBundleChildSchools = new int[altBundleChildIds.length]; + int[] altBundleChildPnums = new int[altBundleChildIds.length]; + for ( int j=0; j < altBundleChildIds.length; j++ ) { + int k = altBundleChildrenOrder[j]; + altBundleChildSchools[j] = escorteeSchoolLoc[k]; + altBundleChildPnums[j] = escorteePnums[k]; + } + + //set other elements of the bundle for this choice\household + bundleObj.setDir( dir ); + bundleObj.setChaufPnum( chauffeurPnums[chaufIndex] ); + bundleObj.setChaufPid( chauffeurPids[chaufIndex] ); + bundleObj.setChaufPersType( chauffeurPersonType[chaufIndex] ); + bundleObj.setChildPnums( altBundleChildPnums ); + bundleObj.setSchoolMazs( altBundleChildSchools ); + bundleObj.setWorkOrSchoolMaz( chauffeurMandatoryLoc[chaufIndex] ); + + + //if the bundle is outbound + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + + //get an array of distances to each child's school starting from home. + float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( hhObj.getHhMgra() ), altBundleChildrenOrder, altBundleChildIds.length ); + bundleObj.setSchoolDists( altBundleSchoolDistances ); + + // set OB depart to earliest child's depart from home for either pure escort or ride sharing + int departHomeInterval = escorteeDepartForSchool[ altBundleChildrenOrder[0] ]; + bundleObj.setDepartHome( departHomeInterval ); + + //if the bundle is pure escort, then the total trip time in minutes is from home through all passengers back to home and the arrival time back at home is the departure + //period plus the time and some time for stops. Otherwise the time arriving to work is the time period departing home + travel time through all escortees plus dwell. + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + + float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( altBundleChildrenOrder, altBundleChildIds.length, DROP_OFF_DURATION ); + float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); + bundleObj.setArriveHome( arriveHomeInterval ); + } + else { + float numMinutes = getTimeInMinutesFromHomeThruAllSchoolsToWork( chauffeurMandatoryLoc[chaufIndex], altBundleChildrenOrder, altBundleChildIds.length, DROP_OFF_DURATION ); + float arriveWorkMinute = convertIntervalToMinutes( departHomeInterval ) + numMinutes; + int arriveWorkInterval = convertMinutesToInterval( arriveWorkMinute ); + bundleObj.setArriveWork( arriveWorkInterval ); + } + + } + else { + + //on the return direction, the departure time from the first school is the time for the first child being escorted. + int departFromFirstSchoolInterval = escorteeDepartFromSchool[altBundleChildrenOrder[0]]; + + // if the bundle is a pure escort, set the departure time period from home to the time that the first child departs from school minus the time required to get to school. + if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[altBundleChildrenOrder[0]] ); + float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; + int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); + bundleObj.setDepartHome( departHomeInterval ); + + float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( hhObj.getHhMgra() ), altBundleChildrenOrder, altBundleChildIds.length ); + bundleObj.setSchoolDists( altBundleSchoolDistances ); + } + else { //else if the bundle is rideshare, the time departing work is the time the first child leave school minus the time required to get to school from work. + float workToFirstSchoolMinutes = getMazToMazTimeInMinutes( chauffeurMandatoryLoc[chaufIndex], escorteeSchoolLoc[altBundleChildrenOrder[0]] ); + float departWorkMinute = convertIntervalToMinutes( departFromFirstSchoolInterval ) - workToFirstSchoolMinutes; + int departWorkInterval = convertMinutesToInterval( departWorkMinute ); + bundleObj.setDepartWork( departWorkInterval ); + + float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( chauffeurMandatoryLoc[chaufIndex] ), altBundleChildrenOrder, altBundleChildIds.length ); + bundleObj.setSchoolDists( altBundleSchoolDistances ); + } + + //on the return direction, the arrival time back home is the time from the first child's departure time plus the time required to get home plus dwell time for each escortee. + float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( altBundleChildrenOrder, altBundleChildIds.length, PICK_UP_DURATION ); + float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; + int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); + bundleObj.setArriveHome( arriveHomeInterval ); + + bundleObj.setDepartPrimaryInterval( departFromFirstSchoolInterval ); + } + + } + + + int[] chaufBundlesOrder = null; + if ( altChaufBundles[chaufIndex].size() > 1 ) { + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) + chaufBundlesOrder = getChaufBundlesOrderOutbound( altChaufBundles[chaufIndex] ); + else + chaufBundlesOrder = getChaufBundlesOrderInbound( altChaufBundles[chaufIndex] ); + } + else { + chaufBundlesOrder = new int[]{ 0 }; + } + + SchoolEscortingBundle[] result = new SchoolEscortingBundle[chaufBundlesOrder.length]; + for( int j=0; j < chaufBundlesOrder.length; j++ ) + result[j] = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); + + + return result; + + } + + + /** + * Get an array of distances to each child's school. + * + * @param origTaz The originTaz is the origin of the first child's trip (home for outbound direction, primary destination for return direction) + * @param childrenOrder The order of each escortee, can be by departure time + * @param numChildren Number of children + * @return A float array of distances to each child's school. + */ + private float[] getDistancesToSchools( int origTaz, int[] childrenOrder, int numChildren ) { + + float[] distances = new float[numChildren]; + + for ( int j=0; j < numChildren; j++ ) { + int k = childrenOrder[j]; + int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); + distances[j] = (float) distanceArray[origTaz][schoolTaz]; + origTaz = schoolTaz; + } + + return distances; + + } + + + private float getMinutesFromFirstSchoolToHome( int[] childrenOrder, int numChildren, float minDuration ) { + + int homeTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); + int firstSchool = escorteeSchoolLoc[ childrenOrder[0] ]; + int firstSchoolTaz = mgraDataManager.getTaz( firstSchool ); + int originTaz = firstSchoolTaz; + + // distance is the cumulative distance from the school where the child is picked up to home, through other schools . + float distance = 0; + float duration = 0; + + for ( int j=1; j < numChildren; j++ ) { + int k = childrenOrder[j]; + int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); + distance += (float) distanceArray[originTaz][ schoolTaz ]; + originTaz = schoolTaz; + + duration += minDuration; + } + + distance += (float) distanceArray[originTaz][ homeTaz ]; + + float timeInMinutes = distance * MINUTES_PER_MILE + duration; + + return timeInMinutes; + + } + + + private float getMazToMazTimeInMinutes( int fromMaz, int toMaz ) { + + int fromTaz = mgraDataManager.getTaz( fromMaz ); + int toTaz = mgraDataManager.getTaz( toMaz ); + + float timeInMinutes = MINUTES_PER_MILE * (float) distanceArray[fromTaz][ toTaz]; + return timeInMinutes; + + } + + + private float getRoundTripMinutesFromHomeThruAllSchoolsToHome( int[] childrenOrder, int numChildren, float minDuration ) { + + int originTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); + + // cumulative distance and duration + float distance = 0; + float duration = 0; + + for ( int j=0; j < numChildren; j++ ) { + int k = childrenOrder[j]; + int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); + distance += (float) distanceArray[originTaz][ schoolTaz]; + originTaz = schoolTaz ; + + duration += minDuration; + } + int destTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); + distance += (float) distanceArray[originTaz][ destTaz]; + + float timeInMinutes = MINUTES_PER_MILE * distance + duration; + return timeInMinutes; + + } + + + private float getTimeInMinutesFromHomeThruAllSchoolsToWork( int workMaz, int[] childrenOrder, int numChildren, float minDuration ) { + + int homeTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); + int workTaz = mgraDataManager.getTaz( workMaz ); + + int originTaz = homeTaz; + + float distance = 0; + float duration = 0; + + for ( int j=0; j < numChildren; j++ ) { + int k = childrenOrder[j]; + int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); + distance += distanceArray[originTaz][ schoolTaz ]; + originTaz = schoolTaz; + duration += minDuration; + } + + distance += (float) distanceArray[originTaz][ workTaz ]; + + float timeInMinutes = MINUTES_PER_MILE * distance + duration; + return timeInMinutes; + + } + + + // Create an array to hold the list indices for the order in which escort activities should be performed. + // This method only gets called while checking availability for a chauffeur to have multiple bundles, so the + // list altChaufBundles has either 2 or 3 escort activities. + private int[] getChaufBundlesOrderOutbound( List altChaufBundles ) { + + int[] chaufBundlesOrder = null; + + // number of escort activities is 2. + if ( altChaufBundles.size() == 2 ) { + // [RS,PE]: if the first activity for the alternative is ride sharing, the second must be pure escort, and should be ordered before the ride sharing escort activity; + if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + chaufBundlesOrder = new int[]{ 1, 0 }; + // [PE,RS]: likewise if the second activity is ride sharing - the first must be pure escort, and must be ordered first + else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + chaufBundlesOrder = new int[]{ 0, 1 }; + // [PE,PE]: otherwise, both activities are pure escort, and the depart times will determine the order + else if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 1, 0 }; + else + chaufBundlesOrder = new int[]{ 0, 1 }; + } + // number of escort activities is 3. + else { + // [RS,PE,PE]: if the first activity for the alternative is ride sharing, the second and third must also be pure escort, and should be ordered before the ride sharing escort activity; + if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 1 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 2, 1, 0 }; + else + chaufBundlesOrder = new int[]{ 1, 2, 0 }; + } + // [PE,RS,PE]: likewise if the second activity is ride sharing - the first and third must be pure escort, and must be ordered before ride sharing + else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 2, 0, 1 }; + else + chaufBundlesOrder = new int[]{ 0, 2, 1 }; + } + // [PE,PE,RS]: likewise if the third activity is ride sharing - the first and second must be pure escort, and must be ordered before ride sharing + else if ( altChaufBundles.get( 2 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 1, 0, 2 }; + else + chaufBundlesOrder = new int[]{ 0, 1, 2 }; + } + // [PE,PE,PE]: otherwise, all three activities are pure escort, and the depart times will determine the order + else { + int[] sortData = new int[]{ altChaufBundles.get( 0 ).getDepartHome(), altChaufBundles.get( 1 ).getDepartHome(), altChaufBundles.get( 2 ).getDepartHome() }; + chaufBundlesOrder = IndexSort.indexSort( sortData ); + } + } + + return chaufBundlesOrder; + + } + + + // Create an array to hold the list indices for the order in which escort activities should be performed. + // This method only gets called while checking availability for a chauffeur to have multiple bundles, so the + // list altChaufBundles has either 2 or 3 escort activities. + private int[] getChaufBundlesOrderInbound( List altChaufBundles ) { + + int[] chaufBundlesOrder = null; + + // number of escort activities is 2. + if ( altChaufBundles.size() == 2 ) { + // [RS,PE]: if the first activity for the alternative is ride sharing, the second must be pure escort, and should be ordered after the ride sharing escort activity; + if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + chaufBundlesOrder = new int[]{ 0, 1 }; + // [PE,RS]: likewise if the second activity is ride sharing - the first must be pure escort, and must be ordered second + else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + chaufBundlesOrder = new int[]{ 1, 0 }; + // [PE,PE]: otherwise, both activities are pure escort, and the depart times will determine the order + else if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 1, 0 }; + else + chaufBundlesOrder = new int[]{ 0, 1 }; + } + // number of escort activities is 3. + else { + // [RS,PE,PE]: if the first activity for the alternative is ride sharing, the second and third must also be pure escort, and should be ordered after the ride sharing escort activity; + if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 1 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 0, 2, 1 }; + else + chaufBundlesOrder = new int[]{ 0, 1, 2 }; + } + // [PE,RS,PE]: likewise if the second activity is ride sharing - the first and third must be pure escort, and must be ordered after ride sharing + else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 1, 2, 0 }; + else + chaufBundlesOrder = new int[]{ 1, 0, 2 }; + } + // [PE,PE,RS]: likewise if the third activity is ride sharing - the first and second must be pure escort, and must be ordered after ride sharing + else if ( altChaufBundles.get( 2 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) + chaufBundlesOrder = new int[]{ 2, 1, 0 }; + else + chaufBundlesOrder = new int[]{ 2, 0, 1 }; + } + // [PE,PE,PE]: otherwise, all three activities are pure escort, and the depart times will determine the order + else { + int[] sortData = new int[]{ altChaufBundles.get( 0 ).getDepartHome(), altChaufBundles.get( 1 ).getDepartHome(), altChaufBundles.get( 2 ).getDepartHome() }; + chaufBundlesOrder = IndexSort.indexSort( sortData ); + } + } + + return chaufBundlesOrder; + + } + + + + + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java new file mode 100644 index 0000000..a79ed0b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java @@ -0,0 +1,2154 @@ +/* +* The school-escort model was designed by PB (Gupta, Vovsha, et al) +* as part of the Maricopa Association of Governments (MAG) +* Activity-based Travel Model Development project. +* +* This source code, which implements the school escort model, +* was written exclusively for and funded by MAG as part of the +* same project; therefore, per their contract, the +* source code belongs to MAG and can only be used with their +* permission. +* +* It is being adapted for the Southern Oregon ABM by PB & RSG +* with permission from MAG and all references to +* the school escort model as well as source code adapted from this +* original code should credit MAG's role in its development. +* +* The escort model and source code should not be transferred to or +* adapted for other agencies or used in other projects without +* expressed permission from MAG. +* +* The source code has been substantially revised to fit within the +* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). +*/ + +package org.sandag.abm.ctramp; +import java.util.ArrayList; +import java.util.Collections; +import java.util.Comparator; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Random; +import java.util.Set; +import java.util.TreeSet; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.calculator.IndexValues; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.IndexSort; +import com.pb.common.util.PropertyMap; + + +public class SchoolEscortingModel { + + public static final String ALT_TABLE_BUNDLE1_NAME = "bundle1"; + public static final String ALT_TABLE_BUNDLE2_NAME = "bundle2"; + public static final String ALT_TABLE_BUNDLE3_NAME = "bundle3"; + public static final String ALT_TABLE_CHAUF1_NAME = "chauf1"; + public static final String ALT_TABLE_CHAUF2_NAME = "chauf2"; + public static final String ALT_TABLE_CHAUF3_NAME = "chauf3"; + public static final String ALT_TABLE_NBUNDLES_NAME = "nbundles"; + public static final String ALT_TABLE_NBUNDLES_RS1_NAME = "nrs1"; + public static final String ALT_TABLE_NBUNDLES_ES1_NAME = "npe1"; + public static final String ALT_TABLE_NBUNDLES_RS2_NAME = "nrs2"; + public static final String ALT_TABLE_NBUNDLES_ES2_NAME = "npe2"; + public static final String ALT_TABLE_BUNDLE_INCIDENCE_FIRST_COLUMN_NAME = "b1_rs1_1"; + + + public static final int NUM_ESCORTEES_PER_HH = 3; + public static final int NUM_CHAUFFEURS_PER_HH = 2; + public static final int NUM_ESCORT_TYPES = 2; + public static final int NUM_BUNDLES = 3; + + + public static final int ESCORT_ELIGIBLE = 1; + + public static final int CHAUFFEUR_1 = 1; + public static final int CHAUFFEUR_2 = 2; + + public static final int DIR_OUTBOUND = 1; + public static final int DIR_INBOUND = 2; + + public static final int RIDE_SHARING_CHAUFFEUR_1 = 1; + public static final int PURE_ESCORTING_CHAUFFEUR_1 = 2; + public static final int RIDE_SHARING_CHAUFFEUR_2 = 3; + public static final int PURE_ESCORTING_CHAUFFEUR_2 = 4; + + + + // chauffeur priority lookup values determined by 100*PersonTypesOld(1-8) + 10*gender(1-2) + 1*age > 25 + private static final int PT_WEIGHT = 100; + private static final int G_WEIGHT = 10; + private static final int A_WEIGHT = 1; + + public static final int RESULT_CHILD_DIRECTION_FIELD = 0; + public static final int RESULT_CHILD_CHOSEN_ALT_FIELD = 1; + public static final int RESULT_CHILD_HHID_FIELD = 2; + public static final int RESULT_CHILD_PNUM_FIELD = 3; + public static final int RESULT_CHILD_PID_FIELD = 4; + public static final int RESULT_CHILD_PERSON_TYPE_FIELD = 5; + public static final int RESULT_CHILD_AGE_FIELD = 6; + public static final int RESULT_CHILD_CDAP_FIELD = 7; + public static final int RESULT_CHILD_SCHOOL_AT_HOME_FIELD = 8; + public static final int RESULT_CHILD_SCHOOL_LOC_FIELD = 9; + public static final int RESULT_CHILD_ESCORT_ELIGIBLE_FIELD = 10; + public static final int RESULT_CHILD_DEPART_FROM_HOME_FIELD = 11; + public static final int RESULT_CHILD_DEPART_TO_HOME_FIELD = 12; + public static final int RESULT_CHILD_DIST_TO_SCHOOL_FIELD = 13; + public static final int RESULT_CHILD_DIST_FROM_SCHOOL_FIELD = 14; + public static final int RESULT_CHILD_ADULT1_DEPART_FROM_HOME_FIELD = 15; + public static final int RESULT_CHILD_ADULT1_DEPART_TO_HOME_FIELD = 16; + public static final int RESULT_CHILD_ADULT2_DEPART_FROM_HOME_FIELD = 17; + public static final int RESULT_CHILD_ADULT2_DEPART_TO_HOME_FIELD = 18; + public static final int RESULT_CHILD_ESCORT_TYPE_FIELD = 19; + public static final int RESULT_CHILD_BUNDLE_ID_FIELD = 20; + public static final int RESULT_CHILD_CHILD_ID_FIELD = 21; + public static final int RESULT_CHILD_CHAUFFEUR_ID_FIELD = 22; + public static final int RESULT_CHILD_CHAUFFEUR_PNUM_FIELD = 23; + public static final int RESULT_CHILD_CHAUFFEUR_PID_FIELD = 24; + public static final int RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD = 25; + public static final int RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD = 26; + public static final int RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD = 27; + public static final int RESULT_CHILD_RANDOM_NUM_FIELD = 28; + public static final int NUM_RESULTS_BY_CHILD_FIELDS = 29; + + + public static final int RESULT_CHAUF_BUNDLE_ID_FIELD = 0; + public static final int RESULT_CHAUF_DIRECTION_FIELD = 1; + public static final int RESULT_CHAUF_CHOSEN_ALT_FIELD = 2; + public static final int RESULT_CHAUF_HHID_FIELD = 3; + public static final int RESULT_CHAUF_PNUM_FIELD = 4; + public static final int RESULT_CHAUF_PID_FIELD = 5; + public static final int RESULT_CHAUF_PERSON_TYPE_FIELD = 6; + public static final int RESULT_CHAUF_AGE_FIELD = 7; + public static final int RESULT_CHAUF_CDAP_FIELD = 8; + public static final int RESULT_CHAUF_ESCORT_ELIGIBLE_FIELD = 9; + public static final int RESULT_CHAUF_DEPART_HOME_FIELD = 10; + public static final int RESULT_CHAUF_ID_FIELD = 11; + public static final int RESULT_CHAUF_ESCORT_TYPE_FIELD = 12; + public static final int RESULT_CHAUF_CHILD1_PNUM_FIELD = 13; + public static final int RESULT_CHAUF_CHILD1_PERSON_TYPE_FIELD = 14; + public static final int RESULT_CHAUF_CHILD1_DEPART_HOME_FIELD = 15; + public static final int RESULT_CHAUF_CHILD2_PNUM_FIELD = 16; + public static final int RESULT_CHAUF_CHILD2_PERSON_TYPE_FIELD = 17; + public static final int RESULT_CHAUF_CHILD2_DEPART_HOME_FIELD = 18; + public static final int RESULT_CHAUF_CHILD3_PNUM_FIELD = 19; + public static final int RESULT_CHAUF_CHILD3_PERSON_TYPE_FIELD = 20; + public static final int RESULT_CHAUF_CHILD3_DEPART_HOME_FIELD = 21; + public static final int NUM_RESULTS_BY_CHAUF_FIELDS = 22; + + public static final int DRIVE_ALONE_MODE = 1; + public static final int SHARED_RIDE_2_MODE = 2; + public static final int SHARED_RIDE_3_MODE = 3; + + private Map chauffeurPriorityOutboundMap; + private Map chauffeurPriorityInboundMap; + + + private TableDataSet altTable; + private String[] altTableNames; + private int[] altTableBundle1; + private int[] altTableBundle2; + private int[] altTableBundle3; + private int[] altTableChauf1; + private int[] altTableChauf2; + private int[] altTableChauf3; + private int[] altTableNumBundles; + private int[] altTableNumRideSharing1Bundles; + private int[] altTableNumPureEscort1Bundles; + private int[] altTableNumRideSharing2Bundles; + private int[] altTableNumPureEscort2Bundles; + + private float defaultTourVOT = (float)7.5; //default VOT for pure escort tours generated by this model. + private float defaultTripVOT = (float)15.0; //default VOT for pure escort trips generated by this model. + + private int[][] altBundleIncidence; + + private int[] previousChoiceChauffeurs; + + private SchoolEscortingDmu decisionMaker; + private transient Logger logger = Logger.getLogger(SchoolEscortingModel.class); + + private static final String UEC_PATH_KEY = "uec.path"; + private static final String OUTBOUND_UEC_FILENAME_KEY = "school.escort.uec.filename"; + private static final String OUTBOUND_UEC_MODEL_SHEET_KEY = "school.escort.outbound.model.sheet"; + private static final String OUTBOUND_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; + private static final String OUTBOUND_CHOICE_MODEL_DESCRIPTION = "School Escorting - Outbound unconditional Choice"; + + private static final String INBOUND_CONDITIONAL_UEC_FILENAME_KEY = "school.escort.uec.filename"; + private static final String INBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY = "school.escort.inbound.conditonal.model.sheet"; + private static final String INBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; + private static final String INBOUND_CONDITIONAL_CHOICE_MODEL_DESCRIPTION = "School Escorting - inbound Conditional Choice"; + + private static final String OUTBOUND_CONDITIONAL_UEC_FILENAME_KEY = "school.escort.uec.filename"; + private static final String OUTBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY = "school.escort.outbound.conditonal.model.sheet"; + private static final String OUTBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; + private static final String OUTBOUND_CONDITIONAL_CHOICE_MODEL_DESCRIPTION = "School Escorting - Outbound Conditional Choice"; + + private static final String PROPERTIES_MODEL_OFFSET = "school.escort.RNG.offset"; + + private ChoiceModelApplication outboundModel; + private ChoiceModelApplication inboundConditionalModel; + private ChoiceModelApplication outboundConditionalModel; + + private MgraDataManager mgraDataManager; + private double[][] distanceArray; + + private IndexValues indexValues; + + private long randomOffset = 110001; + private MersenneTwister random; + + + public SchoolEscortingModel( HashMap propertyMap, MgraDataManager mgraDataManager, AutoTazSkimsCalculator tazDistanceCalculator ) { + + this.mgraDataManager = mgraDataManager; + + random = new MersenneTwister(); + randomOffset = PropertyMap.getLongValueFromPropertyMap(propertyMap,PROPERTIES_MODEL_OFFSET); + + //to do: set distance and time matrix + + createChauffeurPriorityOutboundMap(); + createChauffeurPriorityInboundMap(); + + double[][][] storedFromTazToAllTazsDistanceSkims = tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(); + distanceArray = storedFromTazToAllTazsDistanceSkims[ModelStructure.AM_SKIM_PERIOD_INDEX]; + decisionMaker = new SchoolEscortingDmu( mgraDataManager, distanceArray ); + + + // Create the choice model applications + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + //1. outbound model + String outboundUecFile = uecPath + propertyMap.get(OUTBOUND_UEC_FILENAME_KEY); + int outboundDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_UEC_DATA_SHEET_KEY); + int outboundModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_UEC_MODEL_SHEET_KEY); + + // create the outbound choice model object + outboundModel = new ChoiceModelApplication(outboundUecFile, outboundModelPage, outboundDataPage, propertyMap, + (VariableTable) decisionMaker); + + //2. inbound conditional model + String inboundConditionalUecFile = uecPath + propertyMap.get(INBOUND_CONDITIONAL_UEC_FILENAME_KEY); + int inboundConditionalDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, INBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY); + int inboundConditionalModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, INBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY); + + // create the inbound conditional choice model object + inboundConditionalModel = new ChoiceModelApplication(inboundConditionalUecFile, inboundConditionalModelPage, + inboundConditionalDataPage, propertyMap, + (VariableTable) decisionMaker); + + //3. outbound conditional model + String outboundConditionalUecFile = uecPath + propertyMap.get(OUTBOUND_CONDITIONAL_UEC_FILENAME_KEY); + int outboundConditionalDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY); + int outboundConditionalModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY); + + // create the outbound conditional choice model object + outboundConditionalModel = new ChoiceModelApplication(outboundConditionalUecFile, outboundConditionalModelPage, + outboundConditionalDataPage, propertyMap, + (VariableTable) decisionMaker); + + indexValues = new IndexValues(); + + // get the 0-index columns from the alternatives table, and save in 1-base indexed arrays for lookup by alternative number (1,...,numAlts). + UtilityExpressionCalculator outboundUEC = outboundModel.getUEC(); + altTable = outboundUEC.getAlternativeData(); + altTableNames = outboundUEC.getAlternativeNames(); + + int[] temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE1_NAME ); + altTableBundle1 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableBundle1[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE2_NAME ); + altTableBundle2 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableBundle2[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE3_NAME ); + altTableBundle3 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableBundle3[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF1_NAME ); + altTableChauf1 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableChauf1[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF2_NAME ); + altTableChauf2 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableChauf2[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF3_NAME ); + altTableChauf3 = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableChauf3[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_NAME ); + altTableNumBundles = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableNumBundles[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_RS1_NAME ); + altTableNumRideSharing1Bundles = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableNumRideSharing1Bundles[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_ES1_NAME ); + altTableNumPureEscort1Bundles = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableNumPureEscort1Bundles[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_RS2_NAME ); + altTableNumRideSharing2Bundles = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableNumRideSharing2Bundles[i+1] = temp[i]; + + temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_ES2_NAME ); + altTableNumPureEscort2Bundles = new int[temp.length+1]; + for ( int i=0; i < temp.length; i++ ) + altTableNumPureEscort2Bundles[i+1] = temp[i]; + + int index = altTable.getColumnPosition( ALT_TABLE_BUNDLE_INCIDENCE_FIRST_COLUMN_NAME ); + int numColumns = NUM_BUNDLES * NUM_CHAUFFEURS_PER_HH * NUM_ESCORT_TYPES * NUM_ESCORTEES_PER_HH; + altBundleIncidence = new int[temp.length+1][ numColumns + 1 ]; + for ( int i=0; i < numColumns; i++ ) { + temp = altTable.getColumnAsInt( index+i ); + for ( int j=0; j < temp.length; j++ ) + altBundleIncidence[j+1][i+1] = temp[j]; + } + + decisionMaker.setAltTableBundleIncidence( altBundleIncidence ); + + previousChoiceChauffeurs = new int[NUM_CHAUFFEURS_PER_HH+1]; + + } + + + + /** + * Solve the school escort model for an array of households. The method sets DMU objects, calls the outbound choice model, the + * inbound conditional choice model, and another outbound conditional choice model. The method returns an ArrayList of results with + * 3 member lists: + * 0: results for escortees + * 1: results for chauffeurs + * 2: + * @param logger + * @param hhs + * @throws Exception + */ + public void applyModel( Household household ) throws Exception { + + List childResultList = new ArrayList(); + List chaufResultList = new ArrayList(); + + // apply model only if at least 1 child goes to school + // output: + // child - each direction, escortType(0=none, 1=ride sharing, 2=pure escort), pnum of chauffeur, bundle id, preferred departure time + // chauffeur - each direction, each bundle, pnums of children, preferred departure times + // household - bundles + + //TODO: apply schedule synchronization step after choice according to rules + + + int bundleListId = 0; + List escortBundleList = new ArrayList(); + List obPidList = new ArrayList(); + List obPeList = new ArrayList(); + List obRsList = new ArrayList(); + List ibPidList = new ArrayList(); + List ibPeList = new ArrayList(); + List ibRsList = new ArrayList(); + + try { + //there has to be at least one child with a school tour and one active adult + if ( (household.getNumChildrenWithSchoolTours() > 0) && (household.getNumberActiveAdults() > 0)) { + + boolean debug = false; + if ( household.getDebugChoiceModels() ) { + household.logEntireHouseholdObject("Escort Model trace for Household "+household.getHhId(), logger); + debug = true; + } + + long seed = household.getSeed() + randomOffset; + random.setSeed(seed); + + previousChoiceChauffeurs[1] = 0; + previousChoiceChauffeurs[2] = 0; + + setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_OUTBOUND ); + setDmuAttributesForAdultsOutbound( household, null ); + int[][][] ob0BundleResults = applyOutboundChoiceModel( logger, household, random, debug ); + + + int[][][] chaufExtentsReservedForIb = getEscortBundlesExtent( ob0BundleResults[1], SchoolEscortingModel.DIR_OUTBOUND, household.getSize() ); + setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_INBOUND ); + setDmuAttributesForAdultsInbound( household, chaufExtentsReservedForIb ); + + //note: first dimension = escortees versus chauffeurs. Second dimension = size of household or size * bundles. Third dimension = results + int[][][] ibBundleResults = applyInboundConditionalChoiceModel( logger, household, random, debug ); + + try { + bundleListId = createInboundEscortBundleObjects( ibBundleResults[1], bundleListId, escortBundleList, ibPidList, ibPeList, ibRsList ); + } + catch (Exception e) { + logger.error ( "exception caught saving inbound school escort bundle objects for hhid = " + household.getHhId() + ".", e ); + throw new RuntimeException(e); + } + + + int[][][] chaufExtentsReservedForOb = getEscortBundlesExtent( ibBundleResults[1], SchoolEscortingModel.DIR_INBOUND, household.getHhSize() ); + setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_OUTBOUND ); + setDmuAttributesForAdultsOutbound( household, chaufExtentsReservedForOb ); + //note: first dimension = escortees versus chauffeurs. Second dimension = size of household or size * bundles. Third dimension = results + int[][][] obBundleResults = applyOutboundConditionalChoiceModel( logger, household, debug ); + + try { + bundleListId = createOutboundEscortBundleObjects( obBundleResults[1], bundleListId, escortBundleList, obPidList, obPeList, obRsList ); + } + catch (Exception e) { + logger.error ( "exception caught saving outbound school escort bundle objects for hhid = " + household.getHhId() + ".", e ); + throw new RuntimeException(e); + } + + for ( int j=1; j < ibBundleResults[0].length; j++ ) { + if ( ( ibBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 4 || ibBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 5 ) && ibBundleResults[0][j][RESULT_CHILD_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) + logger.info( "inbound child has ridesharing with non-working chauffeur, j=" + j ); + else if ( ( obBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 4 || obBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 5 ) && obBundleResults[0][j][RESULT_CHILD_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) + logger.info( "outbound child has ridesharing with non-working chauffeur, j=" + j ); + else if ( ( ibBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 4 || ibBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 5 ) && ibBundleResults[1][j][RESULT_CHAUF_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) + logger.info( "inbound non-working chauffeur has ridesharing, j=" + j ); + else if ( ( obBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 4 || obBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 5 ) && obBundleResults[1][j][RESULT_CHAUF_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) + logger.info( "outbound non-working chauffeur has ridesharing, j=" + j ); + } + + childResultList.add( obBundleResults[0] ); //outbound results for escortees + childResultList.add( ibBundleResults[0] ); //inbound results for escortees + chaufResultList.add( obBundleResults[1] ); //outbound results for chauffeurs + chaufResultList.add( ibBundleResults[1] ); //inbound results for chauffeurs + + + } + createTours(household, escortBundleList); + recodeSchoolTours(household); + + } + catch ( Exception e ) { + logger.error( "exception caught applying escort choice model for hh id:" + household.getHhId(), e ); + household.logEntireHouseholdObject("Escort model trace for problem household", logger); + int bundleNumber=0; + for(SchoolEscortingBundle bundle : escortBundleList){ + logger.error("School Escort Bundle "+bundleNumber); + bundle.logBundle(logger); + ++bundleNumber; + } + throw new RuntimeException(e); + } + + if(household.getDebugChoiceModels()){ + logger.info("Logging escort model results for household: "+household.getHhId()); + household.logEntireHouseholdObject("Escort model logging", logger); + } + + } + + + + private int[][][] getEscortBundlesExtent( int[][] bundleResults, int dir, int numPersons ) { + + Set processedChauffSet = new TreeSet(); + + int[][][] chaufExtent = new int[NUM_ESCORT_TYPES+1][numPersons+1][2]; + + for ( int j=1; j < bundleResults.length; j++ ) { + + int chaufid = bundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; + if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { + + int chaufPnum = bundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; + SchoolEscortingBundle[] escortBundles = decisionMaker.getChosenBundles( bundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, dir ); + + if ( escortBundles[0].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + chaufExtent[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] = escortBundles[0].getDepartHome(); + chaufExtent[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] = escortBundles[0].getArriveWork(); + } + else if ( escortBundles[0].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + chaufExtent[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] = escortBundles[0].getDepartHome(); + + int lastBundleIndex = 0; + for ( int i=0; i < escortBundles.length; i++ ) + if ( escortBundles[i].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) + lastBundleIndex = i; + chaufExtent[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] = escortBundles[lastBundleIndex].getArriveHome(); + } + + processedChauffSet.add( chaufid ); + + } + + } + + return chaufExtent; + + } + + + + /** + * Create outbound escort bundle objects and return the bundle list id incremented by the number of new bundles created. The method also adds the bundles to the escortBundleList and + * increments bundleListId by the number of outbound escort bundles. + * + * @param obBundleResults An integer array of results for escortees dimensioned by: household size + 1, escort results fields. + * @param bundleListId The starting bundle ID; will be incremented for each additional bundle, first for rideshare bundles, then escort bundles. + * @param escortBundleList A List of SchoolEscortingBundle objects that will be added to by this method. + * @param obPidList A List of chauffeur IDs that will be added to by this method, one for each outbound bundle + * @param obPeList A List of the outbound pure escort bundle IDs, one for each outbound pure escort bundle + * @param obRsList A List of the outbound rideshare bundle IDs, one for each outbound rideshare bundle + * @return The updated number of chosen bundles (last ID set in the ID lists) + */ + private int createOutboundEscortBundleObjects( int[][] obBundleResults, int bundleListId, List escortBundleList, List obPidList, List obPeList, List obRsList ) { + + int[] obRsIds = null; + int[] obPeIds = null; + + Set processedChauffSet = new TreeSet(); + + for ( int j=1; j < obBundleResults.length; j++ ) { + + int chaufid = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; + if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { + + int chaufPid = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PID_FIELD]; + int chaufPnum = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; + int chaufPtype = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PERSON_TYPE_FIELD]; + + SchoolEscortingBundle[] obEscortBundles = decisionMaker.getChosenBundles( obBundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, SchoolEscortingModel.DIR_OUTBOUND ); + + int numRs = 0; + int numPe = 0; + for ( int k=0; k < obEscortBundles.length; k++ ) { + if ( obEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + numRs++; + else if ( obEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) + numPe++; + } + + obRsIds = new int[numRs]; + obPeIds = new int[numPe]; + + int n = 0; + for ( int k=0; k < obEscortBundles.length; k++ ) { + if ( obEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + obEscortBundles[k].setId(bundleListId); + obEscortBundles[k].setChaufPnum( chaufPnum ); + obEscortBundles[k].setChaufPersType( chaufPtype ); + obEscortBundles[k].setChaufPid( chaufPid ); + escortBundleList.add( obEscortBundles[k] ); + obRsIds[n++] = bundleListId++; + } + } + n = 0; + for ( int k=0; k < obEscortBundles.length; k++ ) { + if ( obEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + obEscortBundles[k].setId(bundleListId); + obEscortBundles[k].setChaufPnum( chaufPnum ); + obEscortBundles[k].setChaufPersType( chaufPtype ); + obEscortBundles[k].setChaufPid( chaufPid ); + escortBundleList.add( obEscortBundles[k] ); + obPeIds[n++] = bundleListId++; + } + } + + obPidList.add( chaufPid ); + obPeList.add( obPeIds ); + obRsList.add( obRsIds ); + + processedChauffSet.add( chaufid ); + + } + + } + + return bundleListId; + + } + + + /** + * Create inbound escort bundle objects and return the bundle list id incremented by the number of new bundles created. The method also adds the bundles to the escortBundleList and + * increments bundleListId by the number of inbound escort bundles. + * + * @param ibBundleResults An integer array of results for escortees dimensioned by: household size + 1, escort results fields. + * @param bundleListId The starting bundle ID; will be incremented for each additional bundle, first for rideshare bundles, then escort bundles. + * @param escortBundleList A List of SchoolEscortingBundle objects that will be added to by this method. + * @param ibPidList A List of chauffeur IDs that will be added to by this method, one for each inbound bundle + * @param ibPeList A List of the inbound pure escort bundle IDs, one for each inbound pure escort bundle + * @param ibRsList A List of the inbound rideshare bundle IDs, one for each inbound rideshare bundle + * @return The updated number of chosen bundles (last ID in the ID lists) + */ + private int createInboundEscortBundleObjects( int[][] ibBundleResults, int bundleListId, List escortBundleList, List ibPidList, List ibPeList, List ibRsList ) { + + int[] ibRsIds = null; + int[] ibPeIds = null; + + Set processedChauffSet = new TreeSet(); + + //for each person in the household + for ( int j=1; j < ibBundleResults.length; j++ ) { + + //the id of the chauffeur for this escortee + int chaufid = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; + + // If this household member is escorted and we haven't created a bundle for the chauffeur yet + if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { + + int chaufPid = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PID_FIELD]; + int chaufPnum = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; + int chaufPtype = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PERSON_TYPE_FIELD]; + + //get the chosen bundles for this chauffeur in the inbound direction + SchoolEscortingBundle[] ibEscortBundles = decisionMaker.getChosenBundles( ibBundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, SchoolEscortingModel.DIR_INBOUND ); + + int numRs = 0; //number rideshare bundles + int numPe = 0; //number pure escort bundles + for ( int k=0; k < ibEscortBundles.length; k++ ) { + if ( ibEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) + numRs++; + else if ( ibEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) + numPe++; + } + + ibRsIds = new int[numRs]; //id of each inbound rideshare bundle + ibPeIds = new int[numPe]; //id of each inbound pure escort bundle + + //for each inbound bundle for this chauffeur, set the bundle chauffeur attributes + int n = 0; + for ( int k=0; k < ibEscortBundles.length; k++ ) { + if ( ibEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { + ibEscortBundles[k].setId(bundleListId); + ibEscortBundles[k].setChaufPnum( chaufPnum ); + ibEscortBundles[k].setChaufPersType( chaufPtype ); + ibEscortBundles[k].setChaufPid( chaufPid ); + escortBundleList.add( ibEscortBundles[k] ); + ibRsIds[n++] = bundleListId++; + } + } + n = 0; + for ( int k=0; k < ibEscortBundles.length; k++ ) { + if ( ibEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { + ibEscortBundles[k].setId(bundleListId); + ibEscortBundles[k].setChaufPnum( chaufPnum ); + ibEscortBundles[k].setChaufPersType( chaufPtype ); + ibEscortBundles[k].setChaufPid( chaufPid ); + escortBundleList.add( ibEscortBundles[k] ); + ibPeIds[n++] = bundleListId++; + } + } + + ibPidList.add( chaufPid ); + ibPeList.add( ibPeIds ); + ibRsList.add( ibRsIds ); + + processedChauffSet.add( chaufid ); + + } + + } + + return bundleListId; + + } + + + /** + * Apply the outbound choice model for the household. Returns a three-dimensional integer array where: + * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs + * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 + * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. + * Ever hear of an object? + * @param logger A logger to write messages and debug statements to. + * @param hh The household to solve the outbound model for. + * @param debug If true calculations will be logged to the logger. + * @return A ragged 3d integer array containing results for the outbound choice model for both escortees and chauffeurs. + * @throws Exception Will be thrown if no alternative is chosen. + */ + private int[][][] applyOutboundChoiceModel( Logger logger, Household hh, Random randObject, boolean debug ) throws Exception { + + + double rn = randObject.nextDouble(); + outboundModel.computeUtilities(decisionMaker, indexValues); + //double[] utilities = outboundModel. + int chosenObAlt = outboundModel.getChoiceResult(rn); + if ( chosenObAlt < 0 || debug ) { + + logger.info("Logging Escort Outbound Model Results for household "+hh.getHhId()); + + //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; + int[] altvaluesToLog = new int[50]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+1; + outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+51; + outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+101; + outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); + altvaluesToLog = new int[altTableNames.length - 150]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+151; + outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); + if ( chosenObAlt < 0 ) { + logger.error( hh.toString() ); + throw new Exception( "chosenObAlt = " + chosenObAlt + " for hhid=" + hh.getHhId() ); + } + + //outboundModel.logInfo("outbound unconditional model", "HHID "+ hh.getHhId(), logger); + + if(debug) + logger.info("Chose outbound unconditional alternative "+ chosenObAlt); + } + + // get field valuess from alternatives table associated with chosen alternative + int chosenObBundle1 = altTableBundle1[chosenObAlt]; + int chosenObBundle2 = altTableBundle2[chosenObAlt]; + int chosenObBundle3 = altTableBundle3[chosenObAlt]; + int chosenObChauf1 = altTableChauf1[chosenObAlt]; + int chosenObChauf2 = altTableChauf2[chosenObAlt]; + int chosenObChauf3 = altTableChauf3[chosenObAlt]; + + + // set the dmu attributes associated with the chosen alternative needed by the Inbound Conditional Choice model + decisionMaker.setOutboundEscortType1( chosenObChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setOutboundEscortType2( chosenObChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setOutboundEscortType3( chosenObChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setOutboundChauffeur1( chosenObChauf1 ); + decisionMaker.setOutboundChauffeur2( chosenObChauf2 ); + decisionMaker.setOutboundChauffeur3( chosenObChauf3 ); + + int[][][] results = new int[2][][]; + results[0] = getResultsByChildArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, (int)(rn*1000000000), chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); + results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); + + return results; + + } + + + /** + * Apply the inbound conditional choice model for the household. Returns a three-dimensional integer array where: + * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs + * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 + * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. + * Ever hear of an object? + * @param logger A logger to write messages and debug statements to. + * @param hh The household to solve the inbound conditional model for. + * @param debug If true calculations will be logged to the logger. + * @return A ragged 3d integer array containing results for the inbound conditional choice model for both escortees and chauffeurs. + * @throws Exception Will be thrown if no alternative is chosen. + */ + private int[][][] applyInboundConditionalChoiceModel( Logger logger, Household hh, Random randObject, boolean debug ) throws Exception { + + double rn = randObject.nextDouble(); + inboundConditionalModel.computeUtilities(decisionMaker, indexValues); + int chosenIbAlt = inboundConditionalModel.getChoiceResult(rn); + if ( chosenIbAlt < 0 || debug ) { + + logger.info("Logging Escort Inbound Conditional Model Results for household "+hh.getHhId()); + + //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; + int[] altvaluesToLog = new int[50]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+1; + inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+51; + inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+101; + inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); + altvaluesToLog = new int[altTableNames.length - 150]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+151; + inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); + if ( chosenIbAlt < 0 ) { + logger.error( hh.toString() ); + throw new Exception( "chosenIbAlt = " + chosenIbAlt + " for hhid=" + hh.getHhId() ); + } + + //inboundConditionalModel.logInfo("inbound conditional model", "HHID "+ hh.getHhId(), logger); + + if(debug) + logger.info("Chose inbound conditional alternative "+ chosenIbAlt); + } + + hh.setInboundEscortChoice(chosenIbAlt); + + // get field values from alternatives table associated with chosen alternative + int chosenIbBundle1 = altTableBundle1[chosenIbAlt]; + int chosenIbBundle2 = altTableBundle2[chosenIbAlt]; + int chosenIbBundle3 = altTableBundle3[chosenIbAlt]; + int chosenIbChauf1 = altTableChauf1[chosenIbAlt]; + int chosenIbChauf2 = altTableChauf2[chosenIbAlt]; + int chosenIbChauf3 = altTableChauf3[chosenIbAlt]; + + if ( debug ) { + int[] escortees = decisionMaker.getEscorteePnums(); + String escorteeString = String.format( "[%s%s%s]", String.valueOf(escortees[1]), (escortees[2] > 0 ? "," + String.valueOf(escortees[2]) : ""), (escortees[3] > 0 ? "," + String.valueOf(escortees[3]) : "") ); + int[] chaufs = decisionMaker.getChauffeurPnums(); + String chaufString = String.format( "[%s%s]", String.valueOf(chaufs[1]), (chaufs[2] > 0 ? "," + String.valueOf(chaufs[2]) : "") ); + logger.info( "hhid=" + hh.getHhId() + ", escortees=" + escorteeString + ", chaufs=" + chaufString ); + logger.info( "chosenIbAlt=" + chosenIbAlt + ", chosenIbChauf1=" + chosenIbChauf1 + ", chosenIbChauf2=" + chosenIbChauf2 + ", chosenIbChauf3=" + chosenIbChauf3 ); + logger.info( "chosenIbBundle1=" + chosenIbBundle1 + ", chosenIbBundle2=" + chosenIbBundle2 + ", chosenIbBundle3=" + chosenIbBundle3 ); + } + + + // set the dmu attributes associated with the chosen alternative needed by the Outbound Conditional Choice model + decisionMaker.setInboundEscortType1( chosenIbChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setInboundEscortType2( chosenIbChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setInboundEscortType3( chosenIbChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); + decisionMaker.setInboundChauffeur1( chosenIbChauf1 ); + decisionMaker.setInboundChauffeur2( chosenIbChauf2 ); + decisionMaker.setInboundChauffeur3( chosenIbChauf3 ); + + + int[][][] results = new int[2][][]; + results[0] = getResultsByChildArray( hh, decisionMaker, DIR_INBOUND, chosenIbAlt, (int)(rn*1000000000), chosenIbBundle1, chosenIbBundle2, chosenIbBundle3, chosenIbChauf1, chosenIbChauf2, chosenIbChauf3 ); + results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_INBOUND, chosenIbAlt, chosenIbBundle1, chosenIbBundle2, chosenIbBundle3, chosenIbChauf1, chosenIbChauf2, chosenIbChauf3 ); + + return results; + + } + + + /** + * Apply the outbound conditional choice model for the household. Returns a three-dimensional integer array where: + * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs + * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 + * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. + * Ever hear of an object? + * @param logger A logger to write messages and debug statements to. + * @param hh The household to solve the outbound conditional model for. + * @param debug If true calculations will be logged to the logger. + * @return A ragged 3d integer array containing results for the outbound conditional choice model for both escortees and chauffeurs. + * @throws Exception Will be thrown if no alternative is chosen. + */ + private int[][][] applyOutboundConditionalChoiceModel( Logger logger, Household hh, boolean debug ) throws Exception { + + double rn = random.nextDouble(); + outboundConditionalModel.computeUtilities(decisionMaker, indexValues); + int chosenObAlt = outboundConditionalModel.getChoiceResult(rn); + if ( chosenObAlt < 0 || debug ) { + + logger.info("Logging Escort Outbound Conditional Model Results for household "+hh.getHhId()); + + //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; + int[] altvaluesToLog = new int[50]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+1; + outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+51; + outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+101; + outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); + altvaluesToLog = new int[altTableNames.length - 150]; + for ( int i=0; i < altvaluesToLog.length; i++ ) + altvaluesToLog[i] = i+151; + outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); + if ( chosenObAlt < 0 ) { + logger.error( hh.toString() ); + throw new Exception( "chosenObAlt = " + chosenObAlt + " for hhid=" + hh.getHhId() ); + } + + //outboundConditionalModel.logInfo("outbound conditional model", "HHID "+ hh.getHhId(), logger); + + if(debug) + logger.info("Chose outbound conditional alternative "+ chosenObAlt); + } + + hh.setOutboundEscortChoice(chosenObAlt); + + // get field valuess from alternatives table associated with chosen alternative + int chosenObBundle1 = altTableBundle1[chosenObAlt]; + int chosenObBundle2 = altTableBundle2[chosenObAlt]; + int chosenObBundle3 = altTableBundle3[chosenObAlt]; + int chosenObChauf1 = altTableChauf1[chosenObAlt]; + int chosenObChauf2 = altTableChauf2[chosenObAlt]; + int chosenObChauf3 = altTableChauf3[chosenObAlt]; + + if ( debug ) { + int[] escortees = decisionMaker.getEscorteePnums(); + String escorteeString = String.format( "[%s%s%s]", String.valueOf(escortees[1]), (escortees[2] > 0 ? "," + String.valueOf(escortees[2]) : ""), (escortees[3] > 0 ? "," + String.valueOf(escortees[3]) : "") ); + int[] chaufs = decisionMaker.getChauffeurPnums(); + String chaufString = String.format( "[%s%s]", String.valueOf(chaufs[1]), (chaufs[2] > 0 ? "," + String.valueOf(chaufs[2]) : "") ); + logger.info( "hhid=" + hh.getHhId() + ", escortees=" + escorteeString + ", chaufs=" + chaufString ); + logger.info( "chosenObAlt=" + chosenObAlt + ", chosenObChauf1=" + chosenObChauf1 + ", chosenObChauf2=" + chosenObChauf2 + ", chosenObChauf3=" + chosenObChauf3 ); + logger.info( "chosenObBundle1=" + chosenObBundle1 + ", chosenObBundle2=" + chosenObBundle2 + ", chosenObBundle3=" + chosenObBundle3 ); + } + + int[][][] results = new int[2][][]; + results[0] = getResultsByChildArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, (int)(rn*1000000000), chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); + results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); + + return results; + + } + + + private void setDmuAttributesForChildren( Household hh, int dir ) { + + List children = hh.getChildPersons(); + List cList = new ArrayList(); + for ( Person child : children ) { + if (child.getUsualSchoolLocation() > 0 && child.getNumSchoolTours() > 0 ) + cList.add( child ); + } + Person[] escortees = getOrderedSchoolChildrenForEscorting( cList, dir ); + + int[] schoolAtHome = new int[NUM_ESCORTEES_PER_HH+1]; + int[] schoolMazs = new int[NUM_ESCORTEES_PER_HH+1]; + int[] schoolDeparts = new int[NUM_ESCORTEES_PER_HH+1]; + int[] schoolReturns = new int[NUM_ESCORTEES_PER_HH+1]; + for ( int i=1; i < escortees.length; i++ ) { + if ( escortees[i] == null ) + continue; + ArrayList schoolTours = escortees[i].getListOfSchoolTours(); + if ( schoolTours != null ) { + Tour tour = schoolTours.get(0); + schoolAtHome[i] = 0; + schoolMazs[i] = tour.getTourDestMgra(); + schoolDeparts[i] = tour.getTourDepartPeriod(); + schoolReturns[i] = tour.getTourArrivePeriod(); + } + } + + decisionMaker.setEscorteeAttributes( cList.size(), escortees, schoolAtHome, schoolMazs, schoolDeparts, schoolReturns ); + + } + + + private void setDmuAttributesForAdultsInbound( Household hh, int[][][] chaufExtents ) { + + List activeAdults = hh.getActiveAdultPersons(); + + Person[] chauffers = getOrderedAdultsForChauffeuringInbound( activeAdults ); + + int[] mandatoryMazs = new int[chauffers.length]; + int[] mandatoryDeparts = new int[chauffers.length]; + int[] mandatoryReturns = new int[chauffers.length]; + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) + continue; + + ArrayList mandatoryTours = getMandatoryTours(chauffers[i]); + + if ( ! mandatoryTours.isEmpty() ) { + Tour tour2 = mandatoryTours.size() == 2 ? mandatoryTours.get(1) : mandatoryTours.get(0); + mandatoryMazs[i] = tour2.getTourDestMgra(); + mandatoryDeparts[i] = tour2.getTourDepartPeriod(); + mandatoryReturns[i] = tour2.getTourArrivePeriod(); + } + } + + decisionMaker.setChaufferAttributes( activeAdults.size(), chauffers, mandatoryMazs, mandatoryDeparts, mandatoryReturns, chaufExtents ); + decisionMaker.setDistanceTimeAttributes( hh, distanceArray ); + + // set the potential chauffeur pnums from the outbound unconditional choice for use with the inbound conditional choice + decisionMaker.setOutboundPotentialChauffeur1( previousChoiceChauffeurs[1] ); + decisionMaker.setOutboundPotentialChauffeur2( previousChoiceChauffeurs[2] ); + + // set the potential chauffeur pnum values in the inbound direction for eventual use in the outbound conditional choice + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) + previousChoiceChauffeurs[i] = 0; + else + previousChoiceChauffeurs[i] = chauffers[i].getPersonNum(); + } + + } + + + /** + * Get a list of mandatory tours for this person. + * + * @param p A person. + * @return An ArrayList of mandatory tours. Empty if no mandatory tours. + */ + ArrayList getMandatoryTours(Person p){ + + ArrayList mandatoryTours = new ArrayList(); + + if(p.getListOfWorkTours()!=null) + mandatoryTours.addAll(p.getListOfWorkTours()); + + if(p.getListOfSchoolTours()!=null) + mandatoryTours.addAll(p.getListOfSchoolTours()); + + return mandatoryTours; + + } + /** + * Set DMU attributes for adults in the outbound direction. + * + * @param hh + * @param chaufExtents + */ + private void setDmuAttributesForAdultsOutbound( Household hh, int[][][] chaufExtents ) { + + List activeAdults = hh.getActiveAdultPersons(); + + Person[] chauffers = getOrderedAdultsForChauffeuringOutbound( activeAdults ); + + int[] mandatoryMazs = new int[NUM_CHAUFFEURS_PER_HH+1]; + int[] mandatoryDeparts = new int[NUM_CHAUFFEURS_PER_HH+1]; + int[] mandatoryReturns = new int[NUM_CHAUFFEURS_PER_HH+1]; + + + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) + continue; + + ArrayList mandatoryTours = getMandatoryTours(chauffers[i]); + + if ( ! mandatoryTours.isEmpty() ) { + Tour tour1 = mandatoryTours.get(0); + Tour tour2 = mandatoryTours.size() == 2 ? mandatoryTours.get(1) : mandatoryTours.get(0); + mandatoryMazs[i] = tour1.getTourDestMgra(); + mandatoryDeparts[i] = tour1.getTourDepartPeriod(); + mandatoryReturns[i] = tour2.getTourArrivePeriod(); + } + + } + + decisionMaker.setChaufferAttributes( activeAdults.size(), chauffers, mandatoryMazs, mandatoryDeparts, mandatoryReturns, chaufExtents ); + decisionMaker.setDistanceTimeAttributes( hh, distanceArray ); + + // set the potential chauffeur pnums from the inbound conditional choice for use with the outbound conditional choice + decisionMaker.setInboundPotentialChauffeur1( previousChoiceChauffeurs[1] ); + decisionMaker.setInboundPotentialChauffeur2( previousChoiceChauffeurs[2] ); + + // set the potential chauffeur pnum values in the outbound direction for eventual use in the inbound conditional choice + for ( int i=1; i < chauffers.length; i++ ) { + if ( chauffers[i] == null ) + previousChoiceChauffeurs[i] = 0; + else + previousChoiceChauffeurs[i] = chauffers[i].getPersonNum(); + } + + } + + + /** + * Order children in list according to age, distance and time period, and return the result in an array. + * + * @param childList List of children to escort + * @param dir Direction of travel (outbound or return) + * @return + */ + private Person[] getOrderedSchoolChildrenForEscorting( List childList, int dir) { + + Household hh = childList.get( 0 ).getHouseholdObject(); + int homeTaz = mgraDataManager.getTaz( hh.getHhMgra() ); + + // sort the eligible children by age so that the 3 youngest are the final candidates + Collections.sort( childList, + new Comparator() { + @Override + public int compare( Person p1, Person p2 ) { + return p1.getAge() - p2.getAge(); + } + } + ); + + Person[] returnArray = new Person[NUM_ESCORTEES_PER_HH+1]; + + //if there is only one child to escort, its easy - just return him/her. + if ( childList.size() == 1 ) { + returnArray[1] = childList.get( 0 ); + } + //if there are two or three children sort them according to distance and departure time time + else if ( childList.size() == 2 ) { + + Person child0 = childList.get( 0 ); + ArrayList schoolTours = child0.getListOfSchoolTours(); + Tour tour0 = schoolTours.get(0); + int timeInterval0 = 0; + int schoolTaz = mgraDataManager.getTaz( tour0.getTourDestMgra()); + float dist0 = (float) distanceArray[homeTaz][ schoolTaz]; + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + timeInterval0 = tour0.getTourDepartPeriod(); + } + else { + timeInterval0 = tour0.getTourArrivePeriod(); + } + + Person child1 = childList.get( 1 ); + schoolTours = child1.getListOfSchoolTours(); + Tour tour1 = schoolTours.get(0); + int timeInterval1 = 0; + schoolTaz = mgraDataManager.getTaz( tour1.getTourDestMgra()); + float dist1 = (float) distanceArray[homeTaz][ schoolTaz]; + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + timeInterval1 = tour1.getTourDepartPeriod(); + } + else { + timeInterval1 = tour1.getTourArrivePeriod(); + } + + int[] sortData = new int[2]; + sortData[0] = timeInterval0*10000000 + (int)(dist0*1000); + sortData[1] = timeInterval1*10000000 + (int)(dist1*1000); + int[] sortIndices = IndexSort.indexSort( sortData ); + + returnArray[1] = childList.get( sortIndices[0] ); + returnArray[2] = childList.get( sortIndices[1] ); + + } + else if ( childList.size() >= 3 ) { + + Person child0 = childList.get( 0 ); + ArrayList schoolTours = child0.getListOfSchoolTours(); + Tour tour0 = schoolTours.get(0); + int timeInterval0 = 0; + int schoolTaz = mgraDataManager.getTaz( tour0.getTourDestMgra()); + float dist0 = (float) distanceArray[homeTaz][ schoolTaz]; + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + timeInterval0 = tour0.getTourDepartPeriod(); + } + else { + timeInterval0 = tour0.getTourArrivePeriod(); + } + + Person child1 = childList.get( 1 ); + schoolTours = child1.getListOfSchoolTours(); + Tour tour1 = schoolTours.get(0); + int timeInterval1 = 0; + schoolTaz = mgraDataManager.getTaz( tour1.getTourDestMgra()); + float dist1 = (float) distanceArray[homeTaz][ schoolTaz]; + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + timeInterval1 = tour1.getTourDepartPeriod(); + } + else { + timeInterval1 = tour1.getTourArrivePeriod(); + } + + Person child2 = childList.get( 2 ); + schoolTours = child2.getListOfSchoolTours(); + Tour tour2 = schoolTours.get(0); + int timeInterval2 = 0; + schoolTaz = mgraDataManager.getTaz( tour2.getTourDestMgra()); + float dist2 = (float) distanceArray[homeTaz][ schoolTaz]; + + if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { + timeInterval2 = tour2.getTourDepartPeriod(); + } + else { + timeInterval2 = tour2.getTourArrivePeriod(); + } + + int[] sortData = new int[3]; + sortData[0] = timeInterval0*10000000 + (int)(dist0*1000); + sortData[1] = timeInterval1*10000000 + (int)(dist1*1000); + sortData[2] = timeInterval2*10000000 + (int)(dist2*1000); + int[] sortIndices = IndexSort.indexSort( sortData ); + + returnArray[1] = childList.get( sortIndices[0] ); + returnArray[2] = childList.get( sortIndices[1] ); + returnArray[3] = childList.get( sortIndices[2] ); + + } + + return returnArray; + + } + + + private Person[] getOrderedAdultsForChauffeuringOutbound( List adultList ) { + + Collections.sort( adultList, + new Comparator() { + @Override + public int compare( Person p1, Person p2 ) { + int p1LookupValue = PT_WEIGHT*p1.getPersonTypeNumber() + G_WEIGHT*p1.getGender() + A_WEIGHT*(p1.getAge() > 25 ? 1 : 0); + int p2LookupValue = PT_WEIGHT*p2.getPersonTypeNumber() + G_WEIGHT*p2.getGender() + A_WEIGHT*(p2.getAge() > 25 ? 1 : 0); + if(!chauffeurPriorityOutboundMap.containsKey(p1LookupValue)){ + logger.fatal("Cannot find lookup value "+p1LookupValue+" in outbound chauffeur priority map"); + throw new RuntimeException(); + } + if(!chauffeurPriorityOutboundMap.containsKey(p2LookupValue)){ + logger.fatal("Cannot find lookup value "+p2LookupValue+" in outbound chauffeur priority map"); + throw new RuntimeException(); + } + int p1Score = chauffeurPriorityOutboundMap.get(p1LookupValue); + int p2Score = chauffeurPriorityOutboundMap.get(p2LookupValue); + return p1Score - p2Score; + } + } + ); + + Person[] returnArray = new Person[NUM_CHAUFFEURS_PER_HH+1]; + for ( int i=0; i < adultList.size() && i < NUM_CHAUFFEURS_PER_HH; i++ ) + returnArray[i+1] = adultList.get( i ); + + return returnArray; + + } + + + /** + * Orders the list of adults passed to the method by person type, gender, and age combination. + * + * @param adultList A list of potential chauffeurs + * @return An ordered array of the chauffeurs. + */ + private Person[] getOrderedAdultsForChauffeuringInbound( List adultList ) { + + Collections.sort( adultList, + new Comparator() { + @Override + public int compare( Person p1, Person p2 ) { + int p1LookupValue = PT_WEIGHT*p1.getPersonTypeNumber() + G_WEIGHT*p1.getGender() + A_WEIGHT*(p1.getAge() >= 25 ? 1 : 0); + int p2LookupValue = PT_WEIGHT*p2.getPersonTypeNumber() + G_WEIGHT*p2.getGender() + A_WEIGHT*(p2.getAge() >= 25 ? 1 : 0); + return chauffeurPriorityInboundMap.get(p1LookupValue) - chauffeurPriorityInboundMap.get(p2LookupValue); + } + } + ); + + Person[] returnArray = new Person[NUM_CHAUFFEURS_PER_HH+1]; + for ( int i=0; i < adultList.size() && i < NUM_CHAUFFEURS_PER_HH; i++ ) + returnArray[i+1] = adultList.get( i ); + + return returnArray; + + } + + /** + * Create the priority map for outbound chauffeurs, according to person type, gender, and age bin. + */ + private void createChauffeurPriorityOutboundMap() { + + chauffeurPriorityOutboundMap = new HashMap(); + + int lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 1 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 2 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 3 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 4 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 5 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 6 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 7 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 8 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 9 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 10 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 11 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 12 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 13 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 13 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 14 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 14 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 15 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 15 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityOutboundMap.put( lookupValue, 16 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityOutboundMap.put( lookupValue, 16 ); + + } + + + /** + * Create the priority map for inbound chauffeurs, according to person type, gender, and age bin. + */ + private void createChauffeurPriorityInboundMap() { + + chauffeurPriorityInboundMap = new HashMap(); + + int lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 1 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 1 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 2 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 2 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 3 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 3 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 4 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 5 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 5 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 6 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 6 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 7 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 8 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 8 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 9 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 9 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 10 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 10 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; + chauffeurPriorityInboundMap.put( lookupValue, 11 ); + + lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; + chauffeurPriorityInboundMap.put( lookupValue, 11 ); + + } + + /** + * Gets the results for each child in the household. The method returns a two dimensional array where the first + * dimension is sized by persons in household + 1, and the second dimension is the total number of fields for child results. + * The first dimension is indexed into by person number and the second dimension is indexed into by result field number. + * Results include household and person attributes for the child being escorted and the chauffeur who is escorting them, + * as well as the attributes of the choice. + * + * @param hh + * @param decisionMaker + * @param direction + * @param chosenAlt Number of chosen alternative. + * @param intRandNum + * @param chosenBundle1 The bundle for child 1: 0 = not escorted. Max 3 bundles + * @param chosenBundle2 The bundle for child 2: 0 = not escorted. Max 3 bundles + * @param chosenBundle3 The bundle for child 3: 0 = not escorted. Max 3 bundles + * @param chosenChauf1 The chauffeur for child 1: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @param chosenChauf2 The chauffeur for child 2: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @param chosenChauf3 The chauffeur for child 3: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @return An integer array of results for each person in the household. + */ + private int[][] getResultsByChildArray( Household hh, SchoolEscortingDmu decisionMaker, int direction, int chosenAlt, int intRandNum, int chosenBundle1, int chosenBundle2, int chosenBundle3, int chosenChauf1, int chosenChauf2, int chosenChauf3 ) { + + int[][] resultsByChild = new int[hh.getHhSize()+1][NUM_RESULTS_BY_CHILD_FIELDS]; + + + int[] adultPnums = decisionMaker.getChauffeurPnums(); + int[] childPnums = decisionMaker.getEscorteePnums(); + + + // set result attributes for all children in the Household + for ( int pnum=1; pnum <= hh.getHhSize(); pnum++ ) { + Person person = hh.getPerson( pnum ); + resultsByChild[pnum][RESULT_CHILD_HHID_FIELD] = hh.getHhId(); + resultsByChild[pnum][RESULT_CHILD_PNUM_FIELD] = pnum; + resultsByChild[pnum][RESULT_CHILD_PID_FIELD] = person.getPersonId(); + resultsByChild[pnum][RESULT_CHILD_PERSON_TYPE_FIELD] = person.getPersonTypeNumber(); + resultsByChild[pnum][RESULT_CHILD_AGE_FIELD] = person.getAge(); + resultsByChild[pnum][RESULT_CHILD_CDAP_FIELD] = person.getCdapIndex(); + resultsByChild[pnum][RESULT_CHILD_SCHOOL_AT_HOME_FIELD] = 0; + resultsByChild[pnum][RESULT_CHILD_SCHOOL_LOC_FIELD] = person.getPersonSchoolLocationZone(); + } + + // set result attributes for children that are to be escorted + for ( Person child : hh.getChildPersons() ) { + int pnum = child.getPersonNum(); + resultsByChild[pnum][RESULT_CHILD_DIRECTION_FIELD] = direction; + resultsByChild[pnum][RESULT_CHILD_CHOSEN_ALT_FIELD] = chosenAlt; + resultsByChild[pnum][RESULT_CHILD_RANDOM_NUM_FIELD] = intRandNum; + } + + // for each escorted child + for ( int i=1; i < childPnums.length; i++ ) { + if ( childPnums[i] > 0 ) { + resultsByChild[childPnums[i]][RESULT_CHILD_ESCORT_ELIGIBLE_FIELD] = ESCORT_ELIGIBLE; + resultsByChild[childPnums[i]][RESULT_CHILD_DEPART_FROM_HOME_FIELD] = decisionMaker.getEscorteeDepartForSchool()[i]; + resultsByChild[childPnums[i]][RESULT_CHILD_DEPART_TO_HOME_FIELD] = decisionMaker.getEscorteeDepartFromSchool()[i]; + resultsByChild[childPnums[i]][RESULT_CHILD_DIST_TO_SCHOOL_FIELD] = decisionMaker.getEscorteeDistToSchool()[i]; + resultsByChild[childPnums[i]][RESULT_CHILD_DIST_FROM_SCHOOL_FIELD] = decisionMaker.getEscorteeDistFromSchool()[i]; + if ( adultPnums[1] > 0 ) { + resultsByChild[childPnums[i]][RESULT_CHILD_ADULT1_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[1]; + resultsByChild[childPnums[i]][RESULT_CHILD_ADULT1_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[1]; + } + if ( adultPnums[2] > 0 ) { + resultsByChild[childPnums[i]][RESULT_CHILD_ADULT2_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[2]; + resultsByChild[childPnums[i]][RESULT_CHILD_ADULT2_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[2]; + } + } + } + + // for each adult + for ( int i=1; i < adultPnums.length; i++ ) { + if ( adultPnums[i] > 0 ) { + resultsByChild[adultPnums[i]][RESULT_CHILD_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[i]; + resultsByChild[adultPnums[i]][RESULT_CHILD_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[i]; + } + } + + if ( chosenChauf1 > 0 ) { + int childid = 1; + int chaufid = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; + resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle1; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; + } + + if ( chosenChauf2 > 0 ) { + int childid = 2; + int chaufid = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; + resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle2; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; + } + + if ( chosenChauf3 > 0 ) { + int childid = 3; + int chaufid = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; + resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle3; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; + resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; + } + + return resultsByChild; + + } + + + /** + * Get the results for each chauffeur in the household and escort bundles undertaken. The method returns a two dimensional integer array + * where the first dimension is sized by household size * total possible bundles (3) + 1, and the second dimension is sized by number of + * results fields for chauffeurs. + * + * @param hh + * @param decisionMaker + * @param direction + * @param chosenAlt + * @param chosenBundle1 The bundle for child 1: 0 = not escorted. Max 3 bundles + * @param chosenBundle2 The bundle for child 2: 0 = not escorted. Max 3 bundles + * @param chosenBundle3 The bundle for child 3: 0 = not escorted. Max 3 bundles + * @param chosenChauf1 The chauffeur for child 1: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @param chosenChauf2 The chauffeur for child 2: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @param chosenChauf3 The chauffeur for child 3: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort + * @return + */ + private int[][] getResultsByChauffeurArray( Household hh, SchoolEscortingDmu decisionMaker, int direction, int chosenAlt, int chosenBundle1, int chosenBundle2, int chosenBundle3, int chosenChauf1, int chosenChauf2, int chosenChauf3 ) { + + int[][] resultsByChauffeurBundle = new int[hh.getHhSize()*NUM_BUNDLES+1][NUM_RESULTS_BY_CHAUF_FIELDS]; + + + int[] adultPnums = decisionMaker.getChauffeurPnums(); + int[] childPnums = decisionMaker.getEscorteePnums(); + + + // set result attributes for chauffeurs + for ( int pnum=1; pnum <= hh.getHhSize(); pnum++ ) { + Person person = hh.getPerson( pnum ); + for ( int i=1; i <= NUM_BUNDLES; i++ ) { + int rowIndex = (pnum-1)*NUM_BUNDLES+i; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_BUNDLE_ID_FIELD] = i; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_HHID_FIELD] = hh.getHhId(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PNUM_FIELD] = pnum; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PID_FIELD] = person.getPersonId(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PERSON_TYPE_FIELD] = person.getPersonTypeNumber(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_AGE_FIELD] = person.getAge(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CDAP_FIELD] = person.getCdapIndex(); + } + } + + for ( Person adult : hh.getAdultPersons() ) { + int pnum = adult.getPersonNum(); + for ( int i=1; i <= NUM_BUNDLES; i++ ) { + int rowIndex = (pnum-1)*NUM_BUNDLES+i; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_DIRECTION_FIELD] = direction; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHOSEN_ALT_FIELD] = chosenAlt; + } + } + + for ( int i=1; i < adultPnums.length; i++ ) { + if ( adultPnums[i] > 0 ) { + for ( int j=1; j <= NUM_BUNDLES; j++ ) { + int rowIndex = (adultPnums[i]-1)*NUM_BUNDLES+j; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_ELIGIBLE_FIELD] = ESCORT_ELIGIBLE; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getChauffeurDepartForMandatory()[i] : decisionMaker.getChauffeurDepartFromMandatory()[i]; + } + } + } + + + if ( chosenChauf1 > 0 ) { + int childid = 1; + int chaufid = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + int bundle = chosenBundle1; + int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; + + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_PNUM_FIELD] = childPnums[childid]; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; + } + + if ( chosenChauf2 > 0 ) { + int childid = 2; + int chaufid = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + int bundle = chosenBundle2; + int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; + + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_PNUM_FIELD] = childPnums[childid]; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; + } + + if ( chosenChauf3 > 0 ) { + int childid = 3; + int chaufid = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; + int escortType = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; + int bundle = chosenBundle3; + int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; + + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_PNUM_FIELD] = childPnums[childid]; + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); + resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; + } + + return resultsByChauffeurBundle; + + } + + /** + * Modify and/or create tours based on the results of the school escort model. + * + * @param household + * @param escortBundleList An array of escort bundles. + */ + public void createTours(Household household, List escortBundleList){ + + + //for each bundle + for(SchoolEscortingBundle escortBundle : escortBundleList){ + + //get chauffeur + int chauffeurPnum = escortBundle.getChaufPnum(); + if(chauffeurPnum==0) + continue; + Person chauffeur = household.getPerson(chauffeurPnum); + + //get the list of ordered children in the bundle + int[] childPnums = escortBundle.getChildPnums(); + int[] schoolMAZs = escortBundle.getSchoolMazs(); + + Tour chauffeurTour = null; + int escortType = escortBundle.getEscortType(); + int numStops = 0; + //************************************************************************************************************** + // + // Pure escort tour : Need to create the chauffeur tour + // Also, number of stops(trips) is equal to children + // + //************************************************************************************************************** + if(escortType==ModelStructure.PURE_ESCORTING_TYPE){ + + ArrayList existingTours = chauffeur.getListOfIndividualNonMandatoryTours(); + int id=0; + if(existingTours!=null) + id=existingTours.size(); + else + existingTours = new ArrayList(); + + //generate a non-mandatory escort tour + chauffeurTour = new Tour(id++, household, chauffeur, "Escort", + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX); + + chauffeurTour.setTourOrigMgra(household.getHhMgra()); + chauffeurTour.setTourPurpose("Escort"); + + if(escortBundle.getDir() == DIR_OUTBOUND){ + //the destination of the outbound tour is the school MAZ of the last child to drop off + int destMAZ = schoolMAZs[schoolMAZs.length-1]; + chauffeurTour.setTourDestMgra(destMAZ); + }else{ + //the destination of the inbound tour is the school MAZ of the first child to pick up + int destMAZ = schoolMAZs[0]; + chauffeurTour.setTourDestMgra(destMAZ); + } + + int departPeriod = escortBundle.getDepartHome(); + int arrivePeriod = escortBundle.getArriveHome(); + chauffeurTour.setTourDepartPeriod(departPeriod); + chauffeurTour.setTourArrivePeriod(arrivePeriod); + chauffeur.scheduleWindow(departPeriod, arrivePeriod); + chauffeurTour.setValueOfTime(defaultTourVOT); + existingTours.add(chauffeurTour); + numStops = escortBundle.getChildPnums().length; + + } + //************************************************************************************************************** + // + // Ridesharing tour: Need to find chauffeur tour in existing mandatory tour array + // Also, number of stops is equal to children + 1 + // + //************************************************************************************************************** + if(escortType==ModelStructure.RIDE_SHARING_TYPE){ + + // ******************************************************************************************** + // Change the mandatory tour of the chauffeur + // ******************************************************************************************** + ArrayList chauffeurMandatoryTours = getMandatoryTours(chauffeur); + if(chauffeurMandatoryTours.isEmpty()){ + logger.fatal("Error: trying to get mandatory tours for person "+chauffeurPnum+" in household "+household.getHhId()+" for ride-sharing bundle"); + household.logEntireHouseholdObject("Escort model debug", logger); + throw new RuntimeException(); + } + + //number of stops is number of children needing to be dropped off plus one for the primary destination/tour origin + numStops = childPnums.length+1; + + //get tour and find existing tour mode (if already set by this method) + if(escortBundle.getDir()==DIR_OUTBOUND) + chauffeurTour = chauffeurMandatoryTours.get(0); + else{ + chauffeurTour = chauffeurMandatoryTours.get(chauffeurMandatoryTours.size() - 1 ); + } + } // end if rideshare type + + //set tour mode to max of existing tour mode and mode for occupancy + int occupancy = childPnums.length + 1; + + //check for stops in the opposite direction, in order to set the occupancy to the max for the tour + int occupancyInOppositeDirection = 0; + if(escortBundle.getDir()==DIR_OUTBOUND){ + Stop[] stops = chauffeurTour.getInboundStops(); + if(stops!=null) + occupancyInOppositeDirection = stops.length; + }else{ + Stop[] stops = chauffeurTour.getOutboundStops(); + if(stops!=null) + occupancyInOppositeDirection = stops.length; + } + + if(Math.max(occupancy,occupancyInOppositeDirection)==2) + chauffeurTour.setTourModeChoice(SHARED_RIDE_2_MODE); + else + chauffeurTour.setTourModeChoice(SHARED_RIDE_3_MODE); + + + //create stops for each child on the chauffeurs tour if ride-share type, or children - 1 if pure escort + if(escortBundle.getDir()==DIR_OUTBOUND){ + + chauffeurTour.setEscortTypeOutbound(escortType); + chauffeurTour.setDriverPnumOutbound(chauffeurPnum); + + //if this is a pure rideshare tour, and there's only one child, create one stop for the outbound direction and move on. + if(escortType==ModelStructure.PURE_ESCORTING_TYPE && numStops == 1){ + Stop stop = chauffeurTour.createStop( "Home", "Escort", false, false); + stop.setOrig(household.getHhMgra()); + stop.setDest(chauffeurTour.getTourDestMgra()); + stop.setStopPeriod(chauffeurTour.getTourDepartPeriod()); + stop.setMode(SHARED_RIDE_2_MODE); + stop.setValueOfTime(defaultTripVOT); + stop.setEscorteePnumDest(childPnums[0]); + stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); + }//more than one stop on the outbound direction on pure escort or its a rideshare tour + else{ + //insert stops on tour for each child to be escorted + String[] stopOrigPurposes = new String[numStops]; + String[] stopDestPurposes = new String[numStops]; + int[] stopPurposeIndices = new int[numStops]; + stopOrigPurposes[0] = "Home"; + + for(int i = 0; i < numStops-1; ++i){ + if (i > 0) + stopOrigPurposes[i] = stopDestPurposes[i - 1]; + stopPurposeIndices[i] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; + stopDestPurposes[i] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; + + } + stopOrigPurposes[numStops-1] = stopDestPurposes[numStops - 2]; + stopDestPurposes[numStops-1] = chauffeurTour.getTourPrimaryPurpose(); + chauffeurTour.createOutboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); + + Stop[] stops = chauffeurTour.getOutboundStops(); + int origMAZ = household.getHhMgra(); + int escorteePnumOrig=0; + byte escortStopTypeOrig=0; + for (int i = 0; i < stops.length; ++i) { + Stop stop = stops[i]; + + //mode to stop is the occupancy - number of stops so far + int mode = 0; + if(occupancy==1) + mode = DRIVE_ALONE_MODE; + else if(occupancy==2) + mode = SHARED_RIDE_2_MODE; + else + mode = SHARED_RIDE_3_MODE; + stop.setMode(mode); + stop.setValueOfTime(defaultTripVOT); + + //decrement the occupancy + occupancy--; + + //set other information for the stop (really the trip to the school) + stop.setEscorteePnumOrig(escorteePnumOrig); + stop.setEscortStopTypeOrig(escortStopTypeOrig); + stop.setOrig(origMAZ); + stop.setStopPeriod(chauffeurTour.getTourDepartPeriod()); + + if(i 0) + stopOrigPurposes[i] = stopDestPurposes[i - 1]; + stopPurposeIndices[i] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; + stopDestPurposes[i] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; + + } + stopOrigPurposes[numStops-1] = stopDestPurposes[numStops - 2]; + stopDestPurposes[numStops-1] = "HOME"; + chauffeurTour.createInboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); + + Stop[] stops = chauffeurTour.getInboundStops(); + int origMAZ = chauffeurTour.getTourDestMgra(); + + + + //if it is a pure escort tour, the origin of the first inbound stop is escort + int escorteePnumOrig=0; + byte escortStopTypeOrig=0; + if(escortType==ModelStructure.PURE_ESCORTING_TYPE){ + escorteePnumOrig = childPnums[0]; + escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; + occupancy = 2; //driver + child + }else{ + occupancy=1; //driver + } + for (int i = 0; i < stops.length; ++i) { + Stop stop = stops[i]; + int mode = 0; + if(occupancy==1) + mode = DRIVE_ALONE_MODE; + else if(occupancy==2) + mode = SHARED_RIDE_2_MODE; + else + mode = SHARED_RIDE_3_MODE; + stop.setMode(mode); + stop.setValueOfTime(defaultTripVOT); + + //increment the occupancy + occupancy++; + + //set the person being escorted + stop.setEscorteePnumOrig(escorteePnumOrig); + stop.setEscortStopTypeOrig(escortStopTypeOrig); + stop.setStopPeriod(chauffeurTour.getTourArrivePeriod()); + escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; //origin stop type of next stop to this stop type + stop.setOrig(origMAZ); + + + int childIndex = i; + if(escortType==ModelStructure.PURE_ESCORTING_TYPE) + ++childIndex; + + if(i adults = household.getActiveAdultPersons(); + List children = household.getChildPersons(); + + if(adults == null || children == null) + return; + + //cycle thru adults in household + for(Person adult : adults){ + ArrayList mandatoryTours = getMandatoryTours(adult); + ArrayList nonMandatoryTours = adult.getListOfIndividualNonMandatoryTours(); + + ArrayList chauffeurTours = new ArrayList(); + + //add mandatory tours to the potential list of chauffeur tours + if(mandatoryTours!=null) + if(mandatoryTours.size()>0) + chauffeurTours.addAll(mandatoryTours); + + //add non-mandatory tours to the potential list of chauffeur tours + if(nonMandatoryTours!=null) + if(nonMandatoryTours.size()>0) + chauffeurTours.addAll(nonMandatoryTours); + + //if there are no possible chauffeur tours, we're done. + if(chauffeurTours.size()==0){ + return; + } + + //cycle thru mandatory tours for each adult + for(Tour chauffeurTour : chauffeurTours){ + + if(chauffeurTour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE||chauffeurTour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE){ + + //cycle thru children in household + for(Person child : children){ + + if(child.getListOfSchoolTours() !=null) + recodeSchoolTour(household, child.getPersonNum(), chauffeurTour, DIR_OUTBOUND); + } + } + if(chauffeurTour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE||chauffeurTour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE){ + + //cycle thru children in household + for(Person child : children){ + + if(child.getListOfSchoolTours() !=null) + recodeSchoolTour(household, child.getPersonNum(), chauffeurTour, DIR_INBOUND); + } + } + + } + } + } + + /** + * Recode the child's school tour to be consistent with the chauffeurTour for the given direction. If the + * child does not have any school tours, the method will simply return. If there is a school tour, and the + * chauffeur is escorting the child, then the child's stop sequence, tour and trip modes, and other + * relevant data is made consistent with the chauffeur's tour. + * + * @param household Household object for the given child pnum. + * @param childPnum The person number of the child. + * @param chauffeurTour The chauffeurTour to use for checking\coding. + * @param direction Outbound or inbound direction. + */ + public void recodeSchoolTour(Household household, int childPnum, Tour chauffeurTour, int direction){ + + //get child's school tour + Person child = household.getPerson(childPnum); + ArrayList schoolTours = child.getListOfSchoolTours(); + + // no school tours for this child + if(schoolTours.isEmpty()){ + return; + } + Tour schoolTour = schoolTours.get(0); + schoolTour.setTourModeChoice(SHARED_RIDE_2_MODE); + + if(direction==DIR_OUTBOUND){ + Stop[] chauffeurStops = chauffeurTour.getOutboundStops(); + + int driverPnum = chauffeurTour.getDriverPnumOutbound(); + + //loop through chauffeur tour stops + for(int i = 0; i < chauffeurStops.length; ++i){ + Stop chauffeurStop = chauffeurStops[i]; + + int occupancy=0; + if(chauffeurTour.getTourPurpose().equals("Escort")) + occupancy = chauffeurStops.length+1; //occupancy of last trip is equal to number of stops + 1 for first child picked up + else + occupancy = chauffeurStops.length; + + if(chauffeurStop.getEscorteePnumDest()==childPnum){ + + int existingTourMode = schoolTour.getTourModeChoice(); + schoolTour.setTourModeChoice(Math.max(existingTourMode,chauffeurTour.getTourModeChoice())); + schoolTour.setDriverPnumOutbound(driverPnum); + schoolTour.setEscortTypeOutbound(chauffeurTour.getEscortTypeOutbound()); + + //child is first stop; no intermediate stops on this child's school tour in the outbound direction + if(i==0){ + Stop stop = schoolTour.createStop( "Home", "School", false, false); + stop.setOrig(household.getHhMgra()); + stop.setDest(schoolTour.getTourDestMgra()); + stop.setStopPeriod(schoolTour.getTourDepartPeriod()); + if(occupancy==2) + stop.setMode(SHARED_RIDE_2_MODE); + else + stop.setMode(SHARED_RIDE_3_MODE); + stop.setValueOfTime(defaultTripVOT); + stop.setEscorteePnumDest(childPnum); + stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); + break; + } + + //child is second or third stop; create outbound stops array with one or two previous stops. + if(i>0){ + //insert stops on tour for each child to be escorted + String[] stopOrigPurposes = new String[i + 1]; + String[] stopDestPurposes = new String[i + 1]; + int[] stopPurposeIndices = new int[i + 1]; + stopOrigPurposes[0] = "Home"; + + for(int j = 0; j < i; ++j){ + if (j > 0) + stopOrigPurposes[j] = stopDestPurposes[j - 1]; + stopPurposeIndices[j] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; + stopDestPurposes[j] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; + } + stopOrigPurposes[i] = stopDestPurposes[i - 1]; + stopDestPurposes[i] = schoolTour.getTourPrimaryPurpose(); + schoolTour.createOutboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); + + Stop[] stops = schoolTour.getOutboundStops(); + int origMAZ = household.getHhMgra(); + int escorteePnumOrig=0; + byte escortStopTypeOrig=0; + for (int j = 0; j < stops.length; ++j) { + Stop stop = stops[j]; + stop.setOrig(origMAZ); + stop.setDest(chauffeurStops[j].getDest()); + origMAZ = stop.getDest(); + stop.setStopPeriod(schoolTour.getTourDepartPeriod()); + if(occupancy==2) + stop.setMode(SHARED_RIDE_2_MODE); + else + stop.setMode(SHARED_RIDE_3_MODE); + stop.setValueOfTime(defaultTripVOT); + stop.setEscorteePnumOrig(escorteePnumOrig); + stop.setEscortStopTypeOrig(escortStopTypeOrig); + stop.setEscorteePnumDest(chauffeurStops[j].getEscorteePnumDest()); + stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); + escorteePnumOrig = chauffeurStops[j].getEscorteePnumDest(); + escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_DROPOFF; + } + break; + } + + } //end if found child in chauffeur stop array + --occupancy; + + } //end cycling through stops in outbound direction + + + } //end if in outbound direction + + // things are more complicated in the inbound direction. in this case, the child who is the last to be + // picked up has the simple tour. + if(direction==DIR_INBOUND){ + Stop[] chauffeurStops = chauffeurTour.getInboundStops(); + + int driverPnum = chauffeurTour.getDriverPnumInbound(); + + //loop through chauffeur tour stops from last to first + for(int i = chauffeurStops.length - 1; i >=0; --i){ + Stop chauffeurStop = chauffeurStops[i]; + if(chauffeurStop.getEscorteePnumOrig()==childPnum){ + + int existingTourMode = schoolTour.getTourModeChoice(); + schoolTour.setTourModeChoice(Math.max(existingTourMode,chauffeurTour.getTourModeChoice())); + schoolTour.setDriverPnumInbound(driverPnum); + schoolTour.setEscortTypeInbound(chauffeurTour.getEscortTypeInbound()); + + //child is last stop; no intermediate stops on this child's school tour in the inbound direction + if(i==chauffeurStops.length-1){ + Stop stop = schoolTour.createStop( "School", "Home", true, false); + stop.setOrig(schoolTour.getTourDestMgra()); + stop.setDest(household.getHhMgra()); + stop.setStopPeriod(schoolTour.getTourArrivePeriod()); + int occupancy=0; + if(chauffeurTour.getTourPurpose().equals("Escort")) + occupancy = chauffeurStops.length+1; //occupancy of last trip is equal to number of stops + 1 for first child picked up + else + occupancy = chauffeurStops.length; + if(occupancy==2) + stop.setMode(SHARED_RIDE_2_MODE); + else + stop.setMode(SHARED_RIDE_3_MODE); + stop.setValueOfTime(defaultTripVOT); + stop.setEscorteePnumOrig(childPnum); + stop.setEscortStopTypeOrig(ModelStructure.ESCORT_STOP_TYPE_PICKUP); + break; + } + + //child is not last stop; create inbound stops array with one or two subsequent stops. + if(i 0) + stopOrigPurposes[j] = stopDestPurposes[j - 1]; + stopPurposeIndices[j] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; + stopDestPurposes[j] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; + } + stopOrigPurposes[numberOfOtherChildrenToPickup] = stopDestPurposes[numberOfOtherChildrenToPickup - 1]; + stopDestPurposes[numberOfOtherChildrenToPickup] = "Home"; + schoolTour.createInboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); + + Stop[] stops = schoolTour.getInboundStops(); + int origMAZ = chauffeurTour.getTourDestMgra(); + int escorteePnumOrig=0; + byte escortStopTypeOrig=0; + for (int j = 0; j < stops.length; ++j) { + Stop stop = stops[j]; + stop.setOrig(origMAZ); + stop.setDest(chauffeurStops[j].getDest()); + origMAZ = stop.getDest(); + stop.setStopPeriod(schoolTour.getTourDepartPeriod()); + stop.setMode(chauffeurStops[j].getMode()); + stop.setValueOfTime(defaultTripVOT); + stop.setEscorteePnumOrig(escorteePnumOrig); + stop.setEscortStopTypeOrig(escortStopTypeOrig); + stop.setEscorteePnumDest(chauffeurStops[j].getEscorteePnumDest()); + stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_PICKUP); + escorteePnumOrig = chauffeurStops[j].getEscorteePnumDest(); + escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; + } + break; + } + + } //end if found child in chauffeur stop array + } //end cycling through stops in inbound direction + + } //end if inbound direction + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java new file mode 100644 index 0000000..deb2c80 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java @@ -0,0 +1,603 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class SchoolLocationChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(SchoolLocationChoiceModel.class); + private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); + + // this constant used as a dimension for saving distance and logsums for + // alternatives in samples + private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; + + private static final int DC_DATA_SHEET = 0; + + private MgraDataManager mgraManager; + private DestChoiceSize dcSizeObj; + + private DestChoiceTwoStageModelDMU dcTwoStageDmuObject; + + private DestChoiceTwoStageModel dcTwoStageModelObject; + private TourModeChoiceModel mcModel; + + private String[] segmentNameList; + private HashMap segmentNameIndexMap; + + private int[] dcModelIndices; + + // A ChoiceModelApplication object and modeAltsAvailable[] is needed for + // each purpose + private ChoiceModelApplication[] locationChoiceModels; + private ChoiceModelApplication locationChoiceModel; + + private boolean[] dcModelAltsAvailable; + private int[] dcModelAltsSample; + private int[] dcModelSampleValues; + + private int[] uecSheetIndices; + + int origMgra; + + private int modelIndex; + private int shadowPricingIteration; + + private double[] sampleAlternativeDistances; + private double[] sampleAlternativeLogsums; + + private double[] mgraDistanceArray; + + private BuildAccessibilities aggAcc; + + private int soaSampleSize; + + private long soaRunTime; + + public SchoolLocationChoiceModel(int index, HashMap propertyMap, + DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, String dcUecFileName, + String soaUecFile, int soaSampleSize, String modeChoiceUecFile, + CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel, + double[][][] schoolSizeProbs, double[][][] schoolTazDistProbs) + { + + this.aggAcc = aggAcc; + this.dcSizeObj = dcSizeObj; + this.mcModel = mcModel; + this.soaSampleSize = soaSampleSize; + + modelIndex = index; + + mgraManager = MgraDataManager.getInstance(); + + dcTwoStageDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); + dcTwoStageDmuObject.setAggAcc(this.aggAcc); + + dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); + dcTwoStageModelObject.setTazDistProbs(schoolTazDistProbs); + dcTwoStageModelObject.setMgraSizeProbs(schoolSizeProbs); + + shadowPricingIteration = 0; + + sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + + } + + public void setupSchoolSegments() + { + aggAcc.createSchoolSegmentNameIndices(); + uecSheetIndices = aggAcc.getSchoolDcUecSheets(); + segmentNameList = aggAcc.getSchoolSegmentNameList(); + } + + public void setupDestChoiceModelArrays(HashMap propertyMap, + String dcUecFileName, String soaUecFile, int soaSampleSize) + { + + segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); + + // create a lookup array to map purpose index to model index + dcModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int dcModelIndex = 0; + int dcSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, dcModelIndex); + dcModelIndices[dcSegmentIndex] = dcModelIndex++; + } else + { + dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); + } + + dcSegmentIndex++; + } + // the value of dcModelIndex is the number of ChoiceModelApplication + // objects to create + // the modelIndexMap keys are the uec sheets to use in building + // ChoiceModelApplication objects + + locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; + + int i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + + int modelIndex = -1; + try + { + modelIndex = modelIndexMap.get(uecIndex); + locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, + uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", + i, modelIndex, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + dcModelAltsAvailable = new boolean[soaSampleSize + 1]; + dcModelAltsSample = new int[soaSampleSize + 1]; + dcModelSampleValues = new int[soaSampleSize]; + + mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; + + } + + public void applySchoolLocationChoice(Household hh) + { + + if (hh.getDebugChoiceModels()) + { + String label = String.format("Pre school Location Choice HHId=%d Object", hh.getHhId()); + hh.logHouseholdObject(label, dcManLogger); + } + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + + int homeMgra = hh.getHhMgra(); + Person[] persons = hh.getPersons(); + + int tourNum = 0; + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + int segmentIndex = -1; + int segmentType = -1; + if (p.getPersonIsPreschoolChild() == 1 || p.getPersonIsStudentNonDriving() == 1 + || p.getPersonIsStudentDriving() == 1 || p.getPersonIsUniversityStudent() == 1) + { + + if (p.getPersonIsPreschoolChild() == 1) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.PRESCHOOL_ALT_INDEX; + } else if (p.getPersonIsGradeSchool() == 1) + { + segmentIndex = aggAcc.getMgraGradeSchoolSegmentIndex(homeMgra); + segmentType = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; + } else if (p.getPersonIsHighSchool() == 1) + { + segmentIndex = aggAcc.getMgraHighSchoolSegmentIndex(homeMgra); + segmentType = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; + } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; + } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) + { + segmentIndex = segmentNameIndexMap + .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); + segmentType = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; + } + + // if person type is a student but segment index is -1, the + // person is not enrolled + // assume home schooled + if (segmentIndex < 0) + { + p.setSchoolLocationSegmentIndex(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); + p.setSchoolLoc(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); + p.setSchoolLocDistance(0); + p.setSchoolLocLogsum(-999); + continue; + } else + { + // if the segment is in the skip shadow pricing set, and the + // iteration is > 0, dont' compute new choice + if (shadowPricingIteration == 0 + || !dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) + p.setSchoolLocationSegmentIndex(segmentIndex); + } + + if (segmentType < 0) + { + segmentType = ModelStructure.NOT_ENROLLED_SEGMENT_INDEX; + } + } else // not a student person type + { + p.setSchoolLocationSegmentIndex(-1); + p.setSchoolLoc(0); + p.setSchoolLocDistance(0); + p.setSchoolLocLogsum(-999); + continue; + } + + // save person information in decision maker label, and log person + // object + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p + .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); + hh.logPersonObject(decisionMakerLabel, dcManLogger, p); + } + + // if the segment is in the skip shadow pricing set, and the + // iteration is > 0, dont' compute new choice + if (shadowPricingIteration > 0 && dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) + continue; + + double[] results = null; + int modelIndex = 0; + try + { + + origMgra = homeMgra; + + // update the DC dmuObject for this person + dcTwoStageDmuObject.setHouseholdObject(hh); + dcTwoStageDmuObject.setPersonObject(p); + dcTwoStageDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); + + double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[segmentIndex]; + mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, + mgraDistanceArray); + + // set size array for the tour segment and distance array from + // the home mgra to all destinaion mgras. + dcTwoStageDmuObject.setMgraSizeArray(homeMgraSizeArray); + dcTwoStageDmuObject.setMgraDistanceArray(mgraDistanceArray); + + modelIndex = dcModelIndices[segmentIndex]; + locationChoiceModel = locationChoiceModels[modelIndex]; + + // get the school location alternative chosen from the sample + results = selectLocationFromSampleOfAlternatives("school", segmentType, p, + segmentNameList[segmentIndex], segmentIndex, tourNum++, homeMgraSizeArray, + mgraDistanceArray); + + } catch (RuntimeException e) + { + logger.fatal(String + .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in school location choice, modelIndex=%d, segmentType=%d, segmentIndex=%d, segmentName=%s", + i, hh.getHhId(), i, modelIndex, segmentType, segmentIndex, + segmentNameList[segmentIndex])); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + p.setSchoolLoc((int) results[0]); + p.setSchoolLocDistance((float) results[1]); + p.setSchoolLocLogsum((float) results[2]); + + } + + } + + /** + * + * @return an array with chosen mgra, distance to chosen mgra, and logsum to + * chosen mgra. + */ + private double[] selectLocationFromSampleOfAlternatives(String segmentType, + int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, + int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcManLogger; + + Household household = person.getHouseholdObject(); + + // get sample of locations and correction factors for sample using the + // alternate method + dcTwoStageModelObject.chooseSample(household.getHhTaz(), sizeSegmentIndex, + segmentTypeIndex, soaSampleSize, household.getHhRandom(), + household.getDebugChoiceModels()); + int[] finalSample = dcTwoStageModelObject.getUniqueSampleMgras(); + double[] sampleCorrectionFactors = dcTwoStageModelObject + .getUniqueSampleMgraCorrectionFactors(); + int numUniqueAlts = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); + + Arrays.fill(dcModelAltsAvailable, false); + Arrays.fill(dcModelAltsSample, 0); + Arrays.fill(dcModelSampleValues, 999999); + + // set sample of alternatives correction factors used in destination + // choice utility for the sampled alternatives. + dcTwoStageDmuObject.setDcSoaCorrections(sampleCorrectionFactors); + + // for the destination mgras in the sample, compute mc logsums and save + // in dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 0; i < numUniqueAlts; i++) + { + + int destMgra = finalSample[i]; + dcModelSampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); + dcTwoStageDmuObject.setMcLogsum(i, logsum); + + sampleAlternativeLogsums[i] = logsum; + sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; + + // set availaibility and sample values for the purpose, dcAlt. + dcModelAltsAvailable[i + 1] = true; + dcModelAltsSample[i + 1] = 1; + + } + + dcTwoStageDmuObject.setSampleArray(dcModelSampleValues); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tourNum); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" + + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " + + person.getPersonType() + ", TourNum=" + tourNum); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + locationChoiceModel.computeUtilities(dcTwoStageDmuObject, + dcTwoStageDmuObject.getDmuIndexValues(), dcModelAltsAvailable, dcModelAltsSample); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (locationChoiceModel.getAvailabilityCount() > 0) + { + try + { + chosen = locationChoiceModel.getChoiceResult(rn); + } catch (Exception e) + { + } + } else + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", + dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject + .getPersonObject().getPersonNum(), segmentName)); + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = locationChoiceModel.getUtilities(); + double[] probabilities = locationChoiceModel.getProbabilities(); + boolean[] availabilities = locationChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb Distance Logsum"); + modelLogger + .info("--------------------- -------------- -------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 1; j < finalSample.length; j++) + { + int alt = finalSample[j]; + cumProb += probabilities[j]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + if (chosen > 0) + { + String altString = String.format("j=%d, mgra=%d", chosen - 1, + finalSample[chosen - 1]); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + } else + { + String altString = String.format("No Chosen Alt, availability count = %d", + locationChoiceModel.getAvailabilityCount()); + modelLogger.info(altString); + } + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + + // write UEC calculation results to separate model specific log file + locationChoiceModel.logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", + dcTwoStageDmuObject.getHouseholdObject().getHhId(), + dcTwoStageDmuObject.getPersonObject().getPersonNum(), segmentName)); + System.exit(-1); + } + + } + + double[] returnArray = new double[3]; + + returnArray[0] = finalSample[chosen - 1]; + returnArray[1] = sampleAlternativeDistances[chosen - 1]; + returnArray[2] = sampleAlternativeLogsums[chosen - 1]; + + return returnArray; + + } + + private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, + int segmentTypeIndex) + { + + int purposeIndex = 0; + String purpose = ""; + if (segmentTypeIndex < 0) + { + purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } + + // create a temporary tour to use to calculate mode choice logsum + Tour mcLogsumTour = new Tour(person, 0, purposeIndex); + mcLogsumTour.setTourPurpose(purpose); + mcLogsumTour.setTourOrigMgra(household.getHhMgra()); + mcLogsumTour.setTourDestMgra(sampleDestMgra); + mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); + mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + + if (household.getDebugChoiceModels()) + { + dcManLogger.info(""); + dcManLogger.info(""); + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, + dcManLogger, person); + } + + double logsum = -1; + try + { + logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, + choiceModelDescription, decisionMakerLabel); + } catch (Exception e) + { + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + e.printStackTrace(); + System.exit(-1); + } + + return logsum; + } + + public int getModelIndex() + { + return modelIndex; + } + + public void setDcSizeObject(DestChoiceSize dcSizeObj) + { + this.dcSizeObj = dcSizeObj; + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java new file mode 100644 index 0000000..b162282 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java @@ -0,0 +1,238 @@ +package org.sandag.abm.ctramp; + +import java.net.UnknownHostException; +import java.util.Date; +import java.util.HashMap; +import org.jppf.node.protocol.AbstractTask; +import org.jppf.node.protocol.DataProvider; +import com.pb.common.calculator.MatrixDataServerIf; + +public class SchoolLocationChoiceTaskJppf + extends AbstractTask +{ + + private static String VERSION = "Task.1.0.3"; + + private transient HashMap propertyMap; + private transient MatrixDataServerIf ms; + private transient HouseholdDataManagerIf hhDataManager; + private transient ModelStructure modelStructure; + private transient String tourCategory; + private transient DestChoiceSize dcSizeObj; + private transient String dcUecFileName; + private transient String soaUecFileName; + private transient int soaSampleSize; + private transient CtrampDmuFactoryIf dmuFactory; + private transient String restartModelString; + + private int iteration; + private int startIndex; + private int endIndex; + private int taskIndex = -1; + + public SchoolLocationChoiceTaskJppf(int taskIndex, int startIndex, int endIndex, int iteration) + { + this.startIndex = startIndex; + this.endIndex = endIndex; + this.taskIndex = taskIndex; + this.iteration = iteration; + } + + public void run() + { + + String start = (new Date()).toString(); + long startTime = System.currentTimeMillis(); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " + + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + // logger.info( String.format( + // "startTime=%d, task=%d run(), thread=%s, start=%d, end=%d.", + // startTime, + // taskIndex, threadName, startIndex, + // endIndex ) ); + + try + { + DataProvider dataProvider = getDataProvider(); + + this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); + this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); + this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); + this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); + this.tourCategory = (String) dataProvider.getParameter("tourCategory"); + this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); + this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); + this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); + this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); + this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); + this.restartModelString = (String) dataProvider.getParameter("restartModelString"); + + } catch (Exception e) + { + e.printStackTrace(); + } + + // HouseholdChoiceModelsManager hhModelManager = + // HouseholdChoiceModelsManager.getInstance( + // propertyMap, restartModelString, modelStructure, dmuFactory); + // hhModelManager.clearHhModels(); + // hhModelManager = null; + + // get the factory object used to create and recycle dcModel objects. + DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); + + // one of tasks needs to initialize the manager object by passing + // attributes + // needed to create a destination choice model object. + modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, + soaSampleSize, dmuFactory, restartModelString); + + // get a dcModel object from manager, which either creates one or + // returns one + // for re-use. + MandatoryDestChoiceModel dcModel = modelManager.getDcSchoolModelObject(taskIndex, + iteration, dcSizeObj); + + // logger.info( String.format( + // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, + // taskIndex, + // threadName, startIndex, endIndex ) ); + System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", + new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); + + long setup1 = (System.currentTimeMillis() - startTime) / 1000; + + Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); + + long setup2 = (System.currentTimeMillis() - startTime) / 1000; + // logger.info( String.format( + // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", + // taskIndex, startIndex, endIndex, + // threadName, setup1, setup2 ) ); + System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", + new Date(), taskIndex, startIndex, endIndex, threadName)); + + int i = -1; + try + { + + boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, + HouseholdDataManager.DEBUG_HHS_ONLY_KEY); + + for (i = 0; i < householdArray.length; i++) + { + // for debugging only - process only household objects specified + // for debugging, if property key was set to true + if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; + + dcModel.applySchoolLocationChoice(householdArray[i]); + } + + hhDataManager.setHhArray(householdArray, startIndex); + + //check to make sure hh array got set in hhDataManager + boolean allHouseholdsAreSame = false; + while(!allHouseholdsAreSame) { + Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); + for(int j = 0; j< householdArrayRemote.length;++j) { + + Household remoteHousehold = householdArrayRemote[j]; + Household localHousehold = householdArray[j]; + + allHouseholdsAreSame = checkIfSameSchoolLocationResults(remoteHousehold, localHousehold); + + if(!allHouseholdsAreSame) + break; + } + if(!allHouseholdsAreSame) { + System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with school location choice results; updating"); + hhDataManager.setHhArray(householdArray, startIndex); + + } + } + + + } catch (Exception e) + { + if (i >= 0 && i < householdArray.length) System.out + .println(String + .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", + taskIndex, i, householdArray[i].getHhId(), startIndex)); + else System.out.println(String.format( + "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", + taskIndex, i, startIndex)); + System.out.println("Exception caught:"); + e.printStackTrace(); + System.out.println("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; + long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; + // logger.info( String.format( + // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, + // threadName, getHhs, processHhs ) ); + System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, + threadName)); + + long total = (System.currentTimeMillis() - startTime) / 1000; + String resultString = String + .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", + threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, + setup2, getHhs, processHhs, total); + // logger.info( resultString ); + setResult(resultString); + + modelManager.returnDcSchoolModelObject(dcModel, taskIndex, startIndex, endIndex); + + } + + /** + * Returns true if school location results are the same, else returns false. + * + * @param thisHousehold + * @param thatHousehold + * @return true or false + */ + public boolean checkIfSameSchoolLocationResults(Household thisHousehold, Household thatHousehold) { + + Person[] thisPersons = thisHousehold.getPersons(); + Person[] thatPersons = thatHousehold.getPersons(); + + if(thisPersons.length!=thatPersons.length) + return false; + + for(int k=1;k +{ + + private static String VERSION = "Task.1.0.3"; + + private transient HashMap propertyMap; + private transient MatrixDataServerIf ms; + private transient HouseholdDataManagerIf hhDataManager; + private transient ModelStructure modelStructure; + private transient String tourCategory; + private transient DestChoiceSize dcSizeObj; + private transient String dcUecFileName; + private transient String soaUecFileName; + private transient int soaSampleSize; + private transient CtrampDmuFactoryIf dmuFactory; + private transient String restartModelString; + + private int iteration; + private int startIndex; + private int endIndex; + private int taskIndex = -1; + + public SchoolLocationChoiceTaskJppfNew(int taskIndex, int startIndex, int endIndex, + int iteration) + { + this.startIndex = startIndex; + this.endIndex = endIndex; + this.taskIndex = taskIndex; + this.iteration = iteration; + } + + public void run() + { + + String start = (new Date()).toString(); + long startTime = System.currentTimeMillis(); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " + + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + // logger.info( String.format( + // "startTime=%d, task=%d run(), thread=%s, start=%d, end=%d.", + // startTime, + // taskIndex, threadName, startIndex, + // endIndex ) ); + + try + { + DataProvider dataProvider = getDataProvider(); + + this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); + this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); + this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); + this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); + this.tourCategory = (String) dataProvider.getParameter("tourCategory"); + this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); + this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); + this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); + this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); + this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); + this.restartModelString = (String) dataProvider.getParameter("restartModelString"); + + } catch (Exception e) + { + e.printStackTrace(); + } + + // HouseholdChoiceModelsManager hhModelManager = + // HouseholdChoiceModelsManager.getInstance( + // propertyMap, restartModelString, modelStructure, dmuFactory); + // hhModelManager.clearHhModels(); + // hhModelManager = null; + + // get the factory object used to create and recycle dcModel objects. + DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); + + // one of tasks needs to initialize the manager object by passing + // attributes + // needed to create a destination choice model object. + modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, + soaSampleSize, dmuFactory, restartModelString); + + // get a dcModel object from manager, which either creates one or + // returns one + // for re-use. + SchoolLocationChoiceModel dcModel = modelManager.getSchoolLocModelObject(taskIndex, + iteration, dcSizeObj); + + // logger.info( String.format( + // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, + // taskIndex, + // threadName, startIndex, endIndex ) ); + System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", + new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); + + long setup1 = (System.currentTimeMillis() - startTime) / 1000; + + Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); + + long setup2 = (System.currentTimeMillis() - startTime) / 1000; + // logger.info( String.format( + // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", + // taskIndex, startIndex, endIndex, + // threadName, setup1, setup2 ) ); + System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", + new Date(), taskIndex, startIndex, endIndex, threadName)); + + int i = -1; + try + { + + boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, + HouseholdDataManager.DEBUG_HHS_ONLY_KEY); + + for (i = 0; i < householdArray.length; i++) + { + // for debugging only - process only household objects specified + // for debugging, if property key was set to true + if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; + + dcModel.applySchoolLocationChoice(householdArray[i]); + } + + hhDataManager.setHhArray(householdArray, startIndex); + + //check to make sure hh array got set in hhDataManager + boolean allHouseholdsAreSame = false; + while(!allHouseholdsAreSame) { + Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); + for(int j = 0; j< householdArrayRemote.length;++j) { + + Household remoteHousehold = householdArrayRemote[j]; + Household localHousehold = householdArray[j]; + + allHouseholdsAreSame = checkIfSameSchoolLocationResults(remoteHousehold, localHousehold); + + if(!allHouseholdsAreSame) + break; + } + if(!allHouseholdsAreSame) { + System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with school location choice results; updating"); + hhDataManager.setHhArray(householdArray, startIndex); + + } + } + + } catch (Exception e) + { + if (i >= 0 && i < householdArray.length) System.out + .println(String + .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", + taskIndex, i, householdArray[i].getHhId(), startIndex)); + else System.out.println(String.format( + "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", + taskIndex, i, startIndex)); + System.out.println("Exception caught:"); + e.printStackTrace(); + System.out.println("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; + long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; + // logger.info( String.format( + // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, + // threadName, getHhs, processHhs ) ); + System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, + threadName)); + + long total = (System.currentTimeMillis() - startTime) / 1000; + String resultString = String + .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", + threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, + setup2, getHhs, processHhs, total); + // logger.info( resultString ); + setResult(resultString); + + modelManager.returnSchoolLocModelObject(dcModel, taskIndex, startIndex, endIndex); + + clearClassAttributes(); + } + + /** + * Returns true if school location results are the same, else returns false. + * + * @param thisHousehold + * @param thatHousehold + * @return true or false + */ + public boolean checkIfSameSchoolLocationResults(Household thisHousehold, Household thatHousehold) { + + Person[] thisPersons = thisHousehold.getPersons(); + Person[] thatPersons = thatHousehold.getPersons(); + + if(thisPersons.length!=thatPersons.length) + return false; + + for(int k=1;k + * The type that the matrices are segmented against. + */ +public interface SegmentedSparseMatrix { + /** + * Get the value of the matrix for a specified segment and row/column ids. + * + * @param segment + * The segment. + * + * @param rowId + * The row id. + * + * @param columnId + * The column id. + * + * @return the matrix value at (rowId,columnId) for {@code segment}. + */ + double getValue(S segment, int rowId, int columnId); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java new file mode 100644 index 0000000..740cbb0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java @@ -0,0 +1,10 @@ +package org.sandag.abm.ctramp; + +/** + * @author crf
+ * Started: Nov 15, 2008 3:25:49 PM + */ +public interface SoaDMU +{ + Household getHouseholdObject(); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java new file mode 100644 index 0000000..73c7178 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java @@ -0,0 +1,118 @@ +package org.sandag.abm.ctramp; + +import java.sql.Connection; +import java.sql.ResultSet; +import java.sql.SQLException; +import java.sql.Statement; + +public class SqliteService +{ + + Connection c = null; + String databaseFile; + + public void connect(String fileName, String tableName) throws DAOException + { + + try + { + c = ConnectionHelper.getConnection(fileName); + Statement s = c.createStatement(); + + s.execute("CREATE TABLE IF NOT EXISTS " + tableName + " (" + " id INTEGER, " + + " numProcessed INTEGER, " + " totalToProcess INTEGER, " + + " startUp INTEGER, " + " runTime INTEGER, " + " shutDown INTEGER " + + ")"); + + s.execute("DELETE FROM " + tableName); + + } catch (SQLException e) + { + e.printStackTrace(); + throw new DAOException(e); + } + + } + + public void listRecords(String tableName) throws DAOException + { + + try + { + + Statement s = c.createStatement(); + + ResultSet rs = s + .executeQuery("SELECT id, numProcessed, totalToProcess, startUp, runTime, shutDown FROM " + + tableName + " ORDER BY id"); + while (rs.next()) + { + System.out.println(rs.getInt("id") + ", " + rs.getInt("numProcessed") + ", " + + rs.getInt("totalToProcess") + ", " + rs.getInt("startUp") + ", " + + rs.getInt("runTime") + ", " + rs.getInt("shutDown")); + } + + } catch (SQLException e) + { + e.printStackTrace(); + throw new DAOException(e); + } + + } + + public void insertRecord(String tableName, int id, int numProcessed, int totalToProcess, + int startUp, int runTime, int shutDown) throws DAOException + { + + try + { + + Statement s = c.createStatement(); + String query = String + .format("INSERT INTO %s (id, numProcessed, totalToProcess, startUp, runTime, shutDown) VALUES (%d, %d, %d, %d, %d, %d)", + tableName, id, numProcessed, totalToProcess, startUp, runTime, shutDown); + s.execute(query); + + } catch (SQLException e) + { + e.printStackTrace(); + throw new DAOException(e); + } + + } + + public void updateRecord(String tableName, int id, int numProcessed, int totalToProcess, + int startUp, int runTime, int shutDown) throws DAOException + { + + try + { + + Statement s = c.createStatement(); + String query = String + .format("UPDATE %s SET numProcessed=%d, totalToProcess=%d, startUp=%d, runTime=%d, shutDown=%d WHERE id=%d", + tableName, numProcessed, totalToProcess, startUp, runTime, shutDown, id); + s.execute(query); + + } catch (SQLException e) + { + e.printStackTrace(); + throw new DAOException(e); + } + + } + + public static void main(String[] args) + { + + SqliteService s = new SqliteService(); + s.connect("c:/jim/status.db", "uwsl"); + + s.insertRecord("uwsl", 0, 27, 1250, 99, 102, 10); + s.insertRecord("uwsl", 2, 29, 1250, 58, 101, 9); + s.insertRecord("uwsl", 1, 32, 1250, 77, 99, 8); + s.listRecords("uwsl"); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java new file mode 100644 index 0000000..882b5b8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java @@ -0,0 +1,304 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import org.apache.log4j.Logger; + +public class Stop + implements Serializable +{ + + static final byte STOP_TYPE_PICKUP = 1; + static final byte STOP_TYPE_DROPOFF = 2; + static final byte STOP_TYPE_OTHER = 3; + int id; + int orig; + int dest; + int park; + int mode; + byte micromobilityWalkMode; + byte micromobilityAccessMode; + byte micromobilityEgressMode; + float micromobilityWalkLogsum; + float micromobilityAccessLogsum; + float micromobilityEgressLogsum; + private float modeLogsum; + private float parkingCost; + + int stopPeriod; + int boardTap; + int alightTap; + boolean inbound; + int set = -1; + + private int escorteePnumOrig; + private byte escortStopTypeOrig; + private int escorteePnumDest; + private byte escortStopTypeDest; + + String origPurpose; + String destPurpose; + int stopPurposeIndex; + + Tour parentTour; + + private double valueOfTime; + + public Stop(Tour parentTour, String origPurpose, String destPurpose, int id, boolean inbound, + int stopPurposeIndex) + { + this.parentTour = parentTour; + this.origPurpose = origPurpose; + this.destPurpose = destPurpose; + this.stopPurposeIndex = stopPurposeIndex; + this.id = id; + this.inbound = inbound; + } + + public void setOrig(int orig) + { + this.orig = orig; + } + + public void setDest(int dest) + { + this.dest = dest; + } + + public void setPark(int park) + { + this.park = park; + } + + public void setMode(int mode) + { + this.mode = mode; + } + + public void setSet(int Skimset) + { + set = Skimset; + } + + public void setBoardTap(int tap) + { + boardTap = tap; + } + + public void setAlightTap(int tap) + { + alightTap = tap; + } + + public void setStopPeriod(int period) + { + stopPeriod = period; + } + + public int getOrig() + { + return orig; + } + + public int getDest() + { + return dest; + } + + public int getPark() + { + return park; + } + + public String getOrigPurpose() + { + return origPurpose; + } + + public String getDestPurpose() + { + return destPurpose; + } + + public void setOrigPurpose(String purpose) + { + origPurpose=purpose; + } + public void setDestPurpose(String purpose){ + destPurpose = purpose; + } + public int getStopPurposeIndex() + { + return stopPurposeIndex; + } + + public int getMode() + { + return mode; + } + public int getSet() + { + return set; + } + + public int getBoardTap() + { + return boardTap; + } + + public int getAlightTap() + { + return alightTap; + } + + public int getStopPeriod() + { + return stopPeriod; + } + + public Tour getTour() + { + return parentTour; + } + + public boolean isInboundStop() + { + return inbound; + } + + public int getStopId() + { + return id; + } + + public int getEscorteePnumOrig() { + return escorteePnumOrig; + } + + public void setEscorteePnumOrig(int escorteePnum) { + this.escorteePnumOrig = escorteePnum; + } + + public byte getEscortStopTypeOrig() { + return escortStopTypeOrig; + } + + public void setEscortStopTypeOrig(byte stopType) { + this.escortStopTypeOrig = stopType; + } + + public int getEscorteePnumDest() { + return escorteePnumDest; + } + + public void setEscorteePnumDest(int escorteePnum) { + this.escorteePnumDest = escorteePnum; + } + + public byte getEscortStopTypeDest() { + return escortStopTypeDest; + } + + public void setEscortStopTypeDest(byte stopType) { + this.escortStopTypeDest = stopType; + } + + public float getModeLogsum() { + return modeLogsum; + } + + public void setModeLogsum(float modeLogsum) { + this.modeLogsum = modeLogsum; + } + public double getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(double valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public void setMicromobilityWalkMode(byte micromobilityWalkMode) { + this.micromobilityWalkMode=micromobilityWalkMode; + } + + public byte getMicromobilityWalkMode() { + return micromobilityWalkMode; + } + public float getMicromobilityWalkLogsum() { + return micromobilityWalkLogsum; + } + + public void setMicromobilityWalkLogsum(float micromobilityWalkLogsum) { + this.micromobilityWalkLogsum = micromobilityWalkLogsum; + } + + public byte getMicromobilityAccessMode() { + return micromobilityAccessMode; + } + + public void setMicromobilityAccessMode(byte micromobilityAccessMode) { + this.micromobilityAccessMode = micromobilityAccessMode; + } + + public byte getMicromobilityEgressMode() { + return micromobilityEgressMode; + } + + public void setMicromobilityEgressMode(byte micromobilityEgressMode) { + this.micromobilityEgressMode = micromobilityEgressMode; + } + + public float getMicromobilityAccessLogsum() { + return micromobilityAccessLogsum; + } + + public void setMicromobilityAccessLogsum(float micromobilityAccessLogsum) { + this.micromobilityAccessLogsum = micromobilityAccessLogsum; + } + + public float getMicromobilityEgressLogsum() { + return micromobilityEgressLogsum; + } + + public void setMicromobilityEgressLogsum(float micromobilityEgressLogsum) { + this.micromobilityEgressLogsum = micromobilityEgressLogsum; + } + + public float getParkingCost() { + return parkingCost; + } + + public void setParkingCost(float parkingCost) { + this.parkingCost = parkingCost; + } + + public void logStopObject(Logger logger, int totalChars) + { + + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + Household.logHelper(logger, "stopId: ", id, totalChars); + Household.logHelper(logger, "origPurpose: ", origPurpose, totalChars); + Household.logHelper(logger, "destPurpose: ", destPurpose, totalChars); + Household.logHelper(logger, "orig: ", orig, totalChars); + Household.logHelper(logger, "dest: ", dest, totalChars); + Household.logHelper(logger, "mode: ", mode, totalChars); + Household.logHelper(logger, "value of time: ", ((float)valueOfTime), totalChars); + Household.logHelper(logger, "boardTap: ", boardTap, totalChars); + Household.logHelper(logger, "alightTap: ", alightTap, totalChars); + Household.logHelper(logger, "TapSet: ", set, totalChars); + Household.logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); + Household.logHelper( logger, "stopPeriod: ", stopPeriod, totalChars ); + Household.logHelper( logger, "orig escort stop type: ",escortStopTypeOrig, totalChars); + Household.logHelper( logger, "orig escortee pnum: ",escorteePnumOrig, totalChars); + Household.logHelper( logger, "dest escort stop type: ",escortStopTypeDest, totalChars); + Household.logHelper( logger, "dest escortee pnum: ",escorteePnumDest, totalChars); + logger.info(separater); + logger.info(""); + logger.info(""); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java new file mode 100644 index 0000000..ab675da --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java @@ -0,0 +1,167 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import com.pb.common.calculator.VariableTable; + +/** + * @author crf
+ * Started: Nov 14, 2008 3:32:58 PM + */ +public class StopDCSoaDMU + implements Serializable, VariableTable +{ + + protected HashMap methodIndexMap; + + protected int tourModeIndex; + protected int[] walkTransitAvailableAtMgra; + protected double origDestDistance; + protected double[] distancesFromOrigMgra; + protected double[] distancesToDestMgra; + protected double[] logSizeTerms; + protected ModelStructure modelStructure; + + public StopDCSoaDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + } + + /** + * set the array of distance values from the origin MGRA of the stop to all + * MGRAs. + * + * @param distances + */ + public void setDistancesFromOrigMgra(double[] distances) + { + distancesFromOrigMgra = distances; + } + + /** + * set the array of distance values from all MGRAs to the final destination + * MGRA of the stop. + * + * @param distances + */ + public void setDistancesToDestMgra(double[] distances) + { + distancesToDestMgra = distances; + } + + /** + * set the OD distance value from the stop origin MGRA to the final + * destination MGRA of the stop. + * + * @param distances + */ + public void setOrigDestDistance(double distance) + { + origDestDistance = distance; + } + + /** + * set the tour mode index value for the tour of the stop being located + * + * @param tour + */ + public void setTourModeIndex(int index) + { + tourModeIndex = index; + } + + /** + * set the array of attributes for all MGRAs that says their is walk transit + * access for the indexed mgra + * + * @param tour + */ + public void setWalkTransitAvailable(int[] avail) + { + walkTransitAvailableAtMgra = avail; + } + + /** + * set the array of logged size terms for all MGRAs for the stop being + * located + * + * @param size + */ + public void setLnSlcSizeAlt(double[] size) + { + logSizeTerms = size; + } + + public double getOrigToMgraDistanceAlt(int alt) + { + return distancesFromOrigMgra[alt]; + } + + public double getMgraToDestDistanceAlt(int alt) + { + return distancesToDestMgra[alt]; + } + + public double getOdDistance() + { + return origDestDistance; + } + + public int getTourModeIsWalk() + { + boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tourModeIndex); + return tourModeIsWalk ? 1 : 0; + } + + public int getTourModeIsBike() + { + boolean tourModeIsBike = modelStructure.getTourModeIsBike(tourModeIndex); + return tourModeIsBike ? 1 : 0; + } + + public int getTourModeIsWalkTransit() + { + return (modelStructure.getTourModeIsWalkTransit(tourModeIndex) ? 1 : 0); + } + + public int getWalkTransitAvailableAlt(int alt) + { + return walkTransitAvailableAtMgra[alt]; + } + + public double getLnSlcSizeAlt(int alt) + { + return logSizeTerms[alt]; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java new file mode 100644 index 0000000..3c1e437 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java @@ -0,0 +1,144 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class will be used for determining the number of stops on individual + * mandatory, individual non-mandatory and joint tours. + * + * @author Christi Willison + * @version Nov 4, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public class StopDepartArrivePeriodModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(StopDepartArrivePeriodModel.class); + + private static final String PROPERTIES_STOP_TOD_LOOKUP_FILE = "stop.depart.arrive.proportions"; + + // define names used in lookup file + private static final String TOUR_PURPOSE_INDEX_COLUMN_HEADING = "tourpurp"; + private static final String HALF_TOUR_DIRECTION_COLUMN_HEADING = "isInbound"; + private static final String TOUR_TOD_PERIOD_HEADING = "interval"; + private static final String TRIP_NUMBER_COLUMN_HEADING = "trip"; + private static final String INTERVAL_1_PROPORTION_COLUMN_HEADING = "p1"; + + private static final int NUM_DIRECTIONS = 2; + private static final int NUM_TRIPS = 4; + + private double[][][][][] proportions; + + private ModelStructure modelStructure; + + /** + * Constructor + * + * @param propertyMap + * - properties HashMap + * @param modelStructure + * - model definitions helper class + */ + public StopDepartArrivePeriodModel(HashMap propertyMap, + ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + setupModels(propertyMap); + } + + private void setupModels(HashMap propertyMap) + { + + logger.info(String.format("setting up stop depart/arrive choice model.")); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String propsFile = uecPath + propertyMap.get(PROPERTIES_STOP_TOD_LOOKUP_FILE); + + // read the stop purpose lookup table data and populate the maps used to + // assign stop purposes + readLookupProportions(propsFile); + + } + + private void readLookupProportions(String propsLookupFilename) + { + + // read the stop purpose proportions into a TableDataSet + TableDataSet propsLookupTable = null; + String fileName = ""; + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + propsLookupTable = reader.readFile(new File(propsLookupFilename)); + } catch (Exception e) + { + logger.error(String.format( + "Exception occurred reading stop purpose lookup proportions file: %s.", + fileName), e); + throw new RuntimeException(); + } + + // allocate an array for storing proportions arrays. + int lastInterval = modelStructure + .getTimePeriodIndexForTime(ModelStructure.LAST_TOD_INTERVAL_HOUR); + proportions = new double[ModelStructure.NUM_PRIMARY_PURPOSES + 1][NUM_DIRECTIONS][lastInterval + 1][NUM_TRIPS + 1][lastInterval + 1]; + + // fields in lookup file are: + // tourpurp isInbound interval trip p1-p40 (alternative interval + // proportions) + + // populate the outProportionsMaps and inProportionsMaps arrays of maps + // from data in the TableDataSet. + // when stops are generated, they can lookup the proportions for stop + // depart or arrive interval determined + // by tour purpose, outbound/inbound direction and interval of previous + // trip. From these proportions, + // a stop tod interval can be drawn. + + // loop over rows in the TableDataSet + for (int i = 0; i < propsLookupTable.getRowCount(); i++) + { + + // get the tour primary purpose index (1-10) + int tourPrimaryPurposeIndex = (int) propsLookupTable.getValueAt(i + 1, + TOUR_PURPOSE_INDEX_COLUMN_HEADING); + + // get the half tour direction (0 for outbound or 1 for inbound) + int direction = (int) propsLookupTable.getValueAt(i + 1, + HALF_TOUR_DIRECTION_COLUMN_HEADING); + + // get the tod interval (1-40) + int todInterval = (int) propsLookupTable.getValueAt(i + 1, TOUR_TOD_PERIOD_HEADING); + + // get the trip number (1-4) + int tripNumber = (int) propsLookupTable.getValueAt(i + 1, TRIP_NUMBER_COLUMN_HEADING); + + // get the index of the first alternative TOD interval proportion. + int firstPropColumn = propsLookupTable + .getColumnPosition(INTERVAL_1_PROPORTION_COLUMN_HEADING); + + // starting at this column, read the proportions for all TOD + // interval proportions. + // Create the array of proportions for this table record. + for (int j = 1; j <= lastInterval; j++) + proportions[tourPrimaryPurposeIndex][direction][todInterval][tripNumber][j] = propsLookupTable + .getValueAt(i + 1, firstPropColumn + j - 1); + + } + + } + + public double[] getStopTodIntervalProportions(int tourPrimaryPurposeIndex, int direction, + int prevTripTodInterval, int tripNumber) + { + return proportions[tourPrimaryPurposeIndex][direction][prevTripTodInterval][tripNumber]; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java new file mode 100644 index 0000000..a09ac84 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java @@ -0,0 +1,214 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.Serializable; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import org.apache.log4j.Logger; +import com.pb.common.datafile.CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * @author crf
+ * Started: Nov 15, 2008 4:17:57 PM + */ +public class StopDestChoiceSize + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(StopDestChoiceSize.class); + + public static final String PROPERTIES_STOP_DC_SIZE_INPUT = "StopDestinationChoice.SizeCoefficients.InputFile"; + + private final Map>> sizeMap; // map + // of + // purpose,purpose + // segment, + // and + // zone/subzone + // to + // size + private final TazDataIf tazDataManager; + private final ModelStructure modelStructure; + private Map>> sizeCoefficients; + + public StopDestChoiceSize(HashMap propertyMap, TazDataIf tazDataManager, + ModelStructure modelStructure) + { + this.tazDataManager = tazDataManager; + this.modelStructure = modelStructure; + sizeMap = new HashMap>>(); + + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String coeffsFileName = propertyMap.get(PROPERTIES_STOP_DC_SIZE_INPUT); + coeffsFileName = projectDirectory + coeffsFileName; + + loadSizeData(coeffsFileName); + } + + public double getDcSize(String purpose, String purposeSegment, int zone, int subzone) + { + return sizeMap.get(purpose).get(purposeSegment).get(getZoneSubzoneMapping(zone, subzone)); + } + + private int getZoneSubzoneMapping(int zone, int subzone) + { + return zone * 10 + subzone; + } + + private void loadSizeData(String coeffsFileName) + { + loadSizeCoefficientTableInformation(readSizeCoefficientTable(coeffsFileName)); + determineSizeCoefficients(); + } + + private TableDataSet readSizeCoefficientTable(String coeffsFileName) + { + try + { + CSVFileReader reader = new CSVFileReader(); + return reader.readFile(new File(coeffsFileName)); + } catch (Exception e) + { + logger.fatal(String.format( + "Exception occurred reading DC Stop Size coefficients data file = %s.", + coeffsFileName), e); + throw new RuntimeException(); + } + } + + private Set getValidPurposes() + { + Set validPurposes = new HashSet(); + validPurposes.add(modelStructure.WORK_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.ESCORT_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.SHOPPING_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.EAT_OUT_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.OTH_MAINT_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.SOCIAL_PURPOSE_NAME.toLowerCase()); + validPurposes.add(modelStructure.OTH_DISCR_PURPOSE_NAME.toLowerCase()); + return validPurposes; + } + + private Set getValidSegments(String purpose) + { + Set validSegments = new HashSet(); + validSegments.add(purpose); + if (purpose.equals(modelStructure.ESCORT_PURPOSE_NAME.toLowerCase())) + for (String segment : modelStructure.ESCORT_SEGMENT_NAMES) + validSegments.add(segment.toLowerCase()); + return validSegments; + } + + private void loadSizeCoefficientTableInformation(TableDataSet coefficients) + { + Set sizeTazColumns = new HashSet(); + String[] coefficientTableColumns = coefficients.getColumnLabels(); + String purposeColumn = modelStructure.getDcSizeCoeffPurposeFieldName(); + String segmentColumn = modelStructure.getDcSizeCoeffSegmentFieldName(); + boolean foundPurposeColumn = false; + boolean foundSegmentColumn = false; + boolean errors = false; + for (String label : coefficientTableColumns) + { + if (label.equals(purposeColumn)) + { + foundPurposeColumn = true; + continue; + } + if (label.equals(segmentColumn)) + { + foundSegmentColumn = true; + continue; + } + + if (!tazDataManager.isValidZoneTableField(label)) + { + logger.fatal("Stop size coefficient table column does not correspond to taz data column: " + + label); + errors = true; + } + sizeTazColumns.add(label); + } + if (!foundPurposeColumn) + { + logger.fatal("Purpose column (" + purposeColumn + + ") not found in stop size coefficient table"); + errors = true; + } + if (!foundSegmentColumn) + { + logger.fatal("Purpose segment column (" + segmentColumn + + ") not found in stop size coefficient table"); + errors = true; + } + + if (!errors) + { + sizeCoefficients = new HashMap>>(); + Set validPurposes = getValidPurposes(); + for (int i = 1; i <= coefficients.getRowCount(); i++) + { + String purpose = coefficients.getStringValueAt(i, purposeColumn).toLowerCase(); + String segment = coefficients.getStringValueAt(i, segmentColumn).toLowerCase(); + if (validPurposes.contains(purpose)) + { + if (!sizeCoefficients.containsKey(purpose)) + sizeCoefficients.put(purpose, new HashMap>()); + if (getValidSegments(purpose).contains(segment)) + { + Map coefficientMap = new HashMap(); + for (String column : sizeTazColumns) + coefficientMap.put(column, (double) coefficients.getValueAt(i, column)); + sizeCoefficients.get(purpose).put(segment, coefficientMap); + } else + { + logger.fatal("Invalid segment for purpose " + purpose + + " found in stop destination choice size coefficient table: " + + segment); + errors = true; + } + + } else + { + logger.fatal("Invalid purpose found in stop destination choice size coefficient table: " + + purpose); + errors = true; + } + } + } + + if (errors) + { + throw new RuntimeException( + "Errors in stop destination choice size coefficient file; see log file for details."); + } + } + + private void determineSizeCoefficients() + { + sizeMap.clear(); + for (String purpose : sizeCoefficients.keySet()) + { + sizeMap.put(purpose, new HashMap>()); + for (String segment : sizeCoefficients.get(purpose).keySet()) + { + Map zoneSizeMap = new HashMap(); + for (int i = 1; i <= tazDataManager.getNumberOfZones(); i++) + { + double size = 0.0d; + Map coefficients = sizeCoefficients.get(purpose).get(segment); + for (String column : sizeCoefficients.get(purpose).get(segment).keySet()) + size += tazDataManager.getZoneTableValue(i, column) + * coefficients.get(column); + double[] walkPercentages = tazDataManager.getZonalWalkPercentagesForTaz(i); + for (int j = 0; j < tazDataManager.getNumberOfSubZones(); j++) + zoneSizeMap.put(getZoneSubzoneMapping(i, j), size * walkPercentages[j]); + } + sizeMap.get(purpose).put(segment, zoneSizeMap); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java new file mode 100644 index 0000000..51403ef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java @@ -0,0 +1,428 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Nov 4, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public abstract class StopFrequencyDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(StopFrequencyDMU.class); + + protected HashMap methodIndexMap; + + public static final int[] NUM_OB_STOPS_FOR_ALT = {-99999999, 0, 0, 0, 0, + 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3 }; + public static final int[] NUM_IB_STOPS_FOR_ALT = {-99999999, 0, 1, 2, 3, + 0, 1, 2, 3, 0, 1, 2, 3, 0, 1, 2, 3 }; + + public String STOP_PURPOSE_FILE_WORK_NAME = "Work"; + public String STOP_PURPOSE_FILE_UNIVERSITY_NAME = "University"; + public String STOP_PURPOSE_FILE_SCHOOL_NAME = "School"; + public String STOP_PURPOSE_FILE_ESCORT_NAME = "Escort"; + public String STOP_PURPOSE_FILE_SHOPPING_NAME = "Shopping"; + public String STOP_PURPOSE_FILE_MAINT_NAME = "Maint"; + public String STOP_PURPOSE_FILE_EAT_OUT_NAME = "EatOut"; + public String STOP_PURPOSE_FILE_VISIT_NAME = "Visit"; + public String STOP_PURPOSE_FILE_DISCR_NAME = "Discr"; + public String STOP_PURPOSE_FILE_WORK_BASED_NAME = "WorkBased"; + + protected IndexValues dmuIndex; + protected Household household; + protected Person person; + protected Tour tour; + + protected ModelStructure modelStructure; + + private double shoppingAccessibility; + private double maintenanceAccessibility; + private double discretionaryAccessibility; + + public StopFrequencyDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + dmuIndex = new IndexValues(); + } + + public abstract HashMap getTourPurposeChoiceModelIndexMap(); + + public abstract int[] getModelSheetValuesArray(); + + public void setDmuIndexValues(int hhid, int homeTaz, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhid); + dmuIndex.setZoneIndex(homeTaz); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (household.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug SF UEC"); + } + + } + + public void setHouseholdObject(Household household) + { + this.household = household; + } + + public void setPersonObject(Person person) + { + this.person = person; + } + + public void setTourObject(Tour tour) + { + this.tour = tour; + } + + public int getTourIsJoint() + { + return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 + : 0; + } + + public void setShoppingAccessibility(double shoppingAccessibility) + { + this.shoppingAccessibility = shoppingAccessibility; + } + + public void setMaintenanceAccessibility(double maintenanceAccessibility) + { + this.maintenanceAccessibility = maintenanceAccessibility; + } + + public void setDiscretionaryAccessibility(double discretionaryAccessibility) + { + this.discretionaryAccessibility = discretionaryAccessibility; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the household income in dollars + */ + public int getIncomeInDollars() + { + return household.getIncomeInDollars(); + } + + /** + * @return the number of full time workers in the household + */ + public int getNumFtWorkers() + { + return household.getNumFtWorkers(); + } + + /** + * @return the number of part time workers in the household + */ + public int getNumPtWorkers() + { + return household.getNumPtWorkers(); + } + + /** + * @return the person type index for the person making the tour + */ + public int getPersonType() + { + return person.getPersonTypeNumber(); + } + + /** + * @return the number of persons in the household + */ + public int getHhSize() + { + return household.getHhSize(); + } + + /** + * @return the number of driving age students in the household + */ + public int getNumHhDrivingStudents() + { + return household.getNumDrivingStudents(); + } + + /** + * @return the number of non-driving age students in the household + */ + public int getNumHhNonDrivingStudents() + { + return household.getNumNonDrivingStudents(); + } + + /** + * @return the number of preschool age students in the household + */ + public int getNumHhPreschool() + { + return household.getNumPreschool(); + } + + /** + * @return the number of work tours made by this person + */ + public int getWorkTours() + { + return person.getNumWorkTours(); + } + + /** + * + * @return 1 if the outbound portion of the tour is escort, in which case stops are already determined, else 0 + */ + public int getOutboundIsEscort(){ + + return ((tour.getEscortTypeOutbound() == ModelStructure.RIDE_SHARING_TYPE) ||(tour.getEscortTypeOutbound() == ModelStructure.PURE_ESCORTING_TYPE)) + ? 1 : 0; + } + + /** + * + * @return 1 if the inbound portion of the tour is escort, in which case stops are already determined, else 0 + */ + public int getInboundIsEscort(){ + + return ((tour.getEscortTypeInbound() == ModelStructure.RIDE_SHARING_TYPE) ||(tour.getEscortTypeInbound() == ModelStructure.PURE_ESCORTING_TYPE)) + ? 1 : 0; + } + + /** + * @return the total number of tours made by this person + */ + public int getTotalTours() + { + return person.getNumTotalIndivTours(); + } + + /** + * @return the total number of tours made by this person + */ + public int getTotalHouseholdTours() + { + return household.getNumTotalIndivTours(); + } + + /** + * @return the distance from the home mgra to the work mgra for this person + */ + public double getWorkLocationDistance() + { + return person.getWorkLocationDistance(); + } + + /** + * @return the distance from the home mgra to the school mgra for this + * person + */ + public double getSchoolLocationDistance() + { + return person.getSchoolLocationDistance(); + } + + /** + * @return the age of this person + */ + public int getAge() + { + return person.getAge(); + } + + /** + * @return the number of school tours made by this person + */ + public int getSchoolTours() + { + return person.getNumSchoolTours(); + } + + /** + * @return the number of escort tours made by this person + */ + public int getEscortTours() + { + return person.getNumIndividualEscortTours(); + } + + /** + * @return the number of shopping tours made by this person + */ + public int getShoppingTours() + { + return person.getNumIndividualShoppingTours(); + } + + /** + * @return the number of maintenance tours made by this person + */ + public int getMaintenanceTours() + { + return person.getNumIndividualOthMaintTours(); + } + + /** + * @return the number of eating out tours made by this person + */ + public int getEatTours() + { + return person.getNumIndividualEatOutTours(); + } + + /** + * @return the number of visit tours made by this person + */ + public int getVisitTours() + { + return person.getNumIndividualSocialTours(); + } + + /** + * @return the number of discretionary tours made by this person + */ + public int getDiscretionaryTours() + { + return person.getNumIndividualOthDiscrTours(); + } + + /** + * @return the shopping accessibility for the household (alts 28-30) + */ + public double getShoppingAccessibility() + { + return shoppingAccessibility; + } + + /** + * @return the maintenance accessibility for the household (alts 31-33) + */ + public double getMaintenanceAccessibility() + { + return maintenanceAccessibility; + } + + /** + * @return the discretionary accessibility for the household (alts 40-42) + */ + public double getDiscretionaryAccessibility() + { + return discretionaryAccessibility; + } + + /** + * @return the number of inbound stops that correspond to the chosen stop + * frequency alternative + */ + public int getNumIbStopsAlt(int alt) + { + return NUM_IB_STOPS_FOR_ALT[alt]; + } + + /** + * @return the number of outbound stops that correspond to the chosen stop + * frequency alternative + */ + public int getNumObStopsAlt(int alt) + { + return NUM_OB_STOPS_FOR_ALT[alt]; + } + + /** + * get the tour duration, measured in hours + * + * @return duration of tour in hours - number of half-hour intervals - + * arrive period - depart period divided by 2. + */ + public float getTourDurationInHours() + { + return (tour.getTourArrivePeriod() - tour.getTourDepartPeriod()) / 2; + } + + public int getTourModeIsAuto() + { + return modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice()) ? 1 : 0; + } + + public int getTourModeIsTransit() + { + return modelStructure.getTourModeIsTransit(tour.getTourModeChoice()) ? 1 : 0; + } + + public int getTourModeIsNonMotorized() + { + return modelStructure.getTourModeIsNonMotorized(tour.getTourModeChoice()) ? 1 : 0; + } + + public int getTourModeIsSchoolBus() + { + return modelStructure.getTourModeIsSchoolBus(tour.getTourModeChoice()) ? 1 : 0; + } + + public int getTourDepartPeriod() + { + return tour.getTourDepartPeriod(); + } + + public int getTourArrivePeriod() + { + return tour.getTourArrivePeriod(); + } + + public int getTelecommuteFrequency() { + return person.getTelecommuteChoice(); + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java new file mode 100644 index 0000000..a5778c5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java @@ -0,0 +1,811 @@ +package org.sandag.abm.ctramp; + +import java.io.File; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Random; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +/** + * This class will be used for determining the number of stops on individual + * mandatory, individual non-mandatory and joint tours. + * + * @author Christi Willison + * @version Nov 4, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public class StopFrequencyModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(StopFrequencyModel.class); + private transient Logger stopFreqLogger = Logger.getLogger("stopFreqLog"); + + private static final String PROPERTIES_UEC_STOP_FREQ = "stf.uec.file"; + private static final String PROPERTIES_STOP_PURPOSE_LOOKUP_FILE = "stf.purposeLookup.proportions"; + + private static String[] shopTypes = {"", + "shopSov0", "shopSov1", "shopSov2" }; + private static String[] maintTypes = {"", + "maintSov0", "maintSov1", "maintSov2" }; + private static String[] discrTypes = {"", + "discrSov0", "discrSov1", "discrSov2" }; + + private static final int UEC_DATA_PAGE = 0; + + // define names used in lookup file + private static final String TOUR_PRIMARY_PURPOSE_COLUMN_HEADING = "PrimPurp"; + private static final String HALF_TOUR_DIRECTION_COLUMN_HEADING = "Direction"; + private static final String TOUR_DEPARTURE_START_RANGE_COLUMN_HEADING = "DepartRangeStart"; + private static final String TOUR_DEPARTURE_END_RANGE_COLUMN_HEADING = "DepartRangeEnd"; + private static final String PERSON_TYPE_COLUMN_HEADING = "Ptype"; + + private static final String OUTBOUND_DIRECTION_NAME = "Outbound"; + private static final String INBOUND_DIRECTION_NAME = "Inbound"; + + private static final String FT_WORKER_PERSON_TYPE_NAME = "FT Worker"; + private static final String PT_WORKER_PERSON_TYPE_NAME = "PT Worker"; + private static final String UNIVERSITY_PERSON_TYPE_NAME = "University Student"; + private static final String NONWORKER_PERSON_TYPE_NAME = "Homemaker"; + private static final String RETIRED_PERSON_TYPE_NAME = "Retired"; + private static final String DRIVING_STUDENT_PERSON_TYPE_NAME = "Driving-age Child"; + private static final String NONDRIVING_STUDENT_PERSON_TYPE_NAME = "Pre-Driving Child"; + private static final String PRESCHOOL_PERSON_TYPE_NAME = "Preschool"; + private static final String ALL_PERSON_TYPE_NAME = "All"; + + private StopFrequencyDMU dmuObject; + private ChoiceModelApplication[] choiceModelApplication; + + HashMap tourPurposeModelIndexMap; + HashMap tourPrimaryPurposeIndexNameMap; + + private HashMap indexPurposeMap; + private HashMap[] outProportionsMaps; + private HashMap[] inProportionsMaps; + + private AccessibilitiesTable accTable; + private ModelStructure modelStructure; + private MgraDataManager mgraManager; + + /** + * Constructor that will be used to set up the ChoiceModelApplications for + * each type of tour + * + * @param projectDirectory + * - name of root level project directory + * @param resourceBundle + * - properties file with paths identified + * @param dmuObject + * - decision making unit for stop frequency + * @param tazDataManager + * - holds information about TAZs in the model. + */ + public StopFrequencyModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory, + ModelStructure myModelStructure, AccessibilitiesTable myAccTable) + { + accTable = myAccTable; + modelStructure = myModelStructure; + setupModels(propertyMap, dmuFactory); + } + + private void setupModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + + mgraManager = MgraDataManager.getInstance(propertyMap); + + logger.info(String.format("setting up stop frequency choice models.")); + + // String projectDirectory = propertyMap.get( + // CtrampApplication.PROPERTIES_PROJECT_DIRECTORY ); + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String stfUecFile = propertyMap.get(PROPERTIES_UEC_STOP_FREQ); + String uecFileName = uecPath + stfUecFile; + + dmuObject = dmuFactory.getStopFrequencyDMU(); + + tourPrimaryPurposeIndexNameMap = modelStructure.getIndexPrimaryPurposeNameMap(); + + tourPurposeModelIndexMap = dmuObject.getTourPurposeChoiceModelIndexMap(); + int[] modelSheetsArray = dmuObject.getModelSheetValuesArray(); + + // one choice model for each model sheet specified + choiceModelApplication = new ChoiceModelApplication[modelSheetsArray.length]; + for (int i = 0; i < modelSheetsArray.length; i++) + choiceModelApplication[i] = new ChoiceModelApplication(uecFileName, + modelSheetsArray[i], UEC_DATA_PAGE, propertyMap, (VariableTable) dmuObject); + + String purposeLookupFileName = uecPath + + propertyMap.get(PROPERTIES_STOP_PURPOSE_LOOKUP_FILE); + + // read the stop purpose lookup table data and populate the maps used to + // assign stop purposes + readPurposeLookupProportionsTable(purposeLookupFileName); + + } + + public void applyModel(Household household) + { + + int totalStops = 0; + int totalTours = 0; + + Logger modelLogger = stopFreqLogger; + if (household.getDebugChoiceModels()) + household.logHouseholdObject("Pre Stop Frequency Choice: HH=" + household.getHhId(), + stopFreqLogger); + + // get this household's person array + Person[] personArray = household.getPersons(); + + // set the household id, origin taz, hh taz, and debugFlag=false in the + // dmu + dmuObject.setHouseholdObject(household); + + // set the auto sufficiency dependent non-mandatory accessibility values + // for + // the household + int autoSufficiency = household.getAutoSufficiency(); + dmuObject.setShoppingAccessibility(accTable.getAggregateAccessibility( + shopTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setMaintenanceAccessibility(accTable.getAggregateAccessibility( + maintTypes[autoSufficiency], household.getHhMgra())); + dmuObject.setDiscretionaryAccessibility(accTable.getAggregateAccessibility( + discrTypes[autoSufficiency], household.getHhMgra())); + + // process the joint tours for the household first + Tour[] jt = household.getJointTourArray(); + if (jt != null) + { + + List tourList = new ArrayList(); + for (Tour t : jt) + tourList.add(t); + + int tourCount = 0; + for (Tour tour : tourList) + { + + try + { + + //tour.clearStopModelResults(); + + int modelIndex = tourPurposeModelIndexMap + .get(tour.getTourPrimaryPurposeIndex()); + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + if (household.getDebugChoiceModels()) + { + choiceModelDescription = String + .format("Joint Tour Stop Frequency Choice Model:"); + decisionMakerLabel = String.format( + "HH=%d, TourType=%s, TourId=%d, TourPurpose=%s.", + household.getHhId(), tour.getTourCategory(), tour.getTourId(), + tour.getTourPurpose()); + choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; + + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + } + + // set the tour object + dmuObject.setTourObject(tour); + + // set the tour orig/dest TAZs associated with the tour + // orig/dest MGRAs in the IndexValues object. + dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), + mgraManager.getTaz(tour.getTourOrigMgra()), + mgraManager.getTaz(tour.getTourDestMgra())); + + // compute the utilities + float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + tour.setStopFreqLogsum(logsum); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + int choice = -1; + if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught applying joint tour stop frequency choice model for %s type tour: HHID=%d, tourCount=%d, randomCount=%f -- no avaialable stop frequency alternative to choose.", + tour.getTourCategory(), household.getHhId(), tourCount, + randomCount)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = choiceModelApplication[modelIndex].getUtilities(); + double[] probabilities = choiceModelApplication[modelIndex] + .getProbabilities(); + String[] altNames = choiceModelApplication[modelIndex] + .getAlternativeNames(); + + // 0s-indexing + modelLogger.info(decisionMakerLabel); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %15s", k + 1, altNames[k]); + modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %s", choice, altNames[choice - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + choiceModelApplication[modelIndex].logAlternativesInfo( + choiceModelDescription, decisionMakerLabel); + choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, choice); + + // write UEC calculation results to separate model + // specific log file + choiceModelApplication[modelIndex] + .logUECResults(modelLogger, loggingHeader); + } + + // save the chosen alternative and create and populate the + // arrays of inbound/outbound + // stops in the tour object + totalStops += setStopFreqChoice(tour, choice); + + totalTours++; + tourCount++; + + } catch (Exception e) + { + logger.error(String + .format("Exception caught processing joint tour stop frequency choice model for %s type tour: HHID=%d, tourCount=%d.", + tour.getTourCategory(), household.getHhId(), tourCount)); + throw new RuntimeException(e); + } + + } + + } + + // now loop through the person array (1-based), and process all tours + // for + // each person + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + if (household.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", + household.getHhId(), person.getPersonNum(), person.getPersonType()); + household.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + // set the person + dmuObject.setPersonObject(person); + + List tourList = new ArrayList(); + + // apply stop frequency for all person tours + tourList.addAll(person.getListOfWorkTours()); + tourList.addAll(person.getListOfSchoolTours()); + tourList.addAll(person.getListOfIndividualNonMandatoryTours()); + tourList.addAll(person.getListOfAtWorkSubtours()); + + int tourCount = 0; + for (Tour tour : tourList) + { + + try + { + + //tour.clearStopModelResults(); + + int modelIndex = tourPurposeModelIndexMap + .get(tour.getTourPrimaryPurposeIndex()); + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + choiceModelDescription = String + .format("Individual Tour Stop Frequency Choice Model:"); + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, TourType=%s, TourId=%d, TourPurpose=%s, modelIndex=%d.", + household.getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourCategory(), + tour.getTourId(), tour.getTourPurpose(), modelIndex); + + choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + } + + // set the tour object + dmuObject.setTourObject(tour); + + // compute the utilities + dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), + mgraManager.getTaz(tour.getTourOrigMgra()), + mgraManager.getTaz(tour.getTourDestMgra())); + + float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, + dmuObject.getDmuIndexValues()); + tour.setStopFreqLogsum(logsum); + + // get the random number from the household + Random random = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = random.nextDouble(); + + // if the choice model has at least one available + // alternative, + // make choice. + int choice = -1; + if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] + .getChoiceResult(rn); + else + { + logger.error(String + .format("Exception caught applying Individual Tour stop frequency choice model for %s type tour: j=%d, HHID=%d, personNum=%d, tourCount=%d, randomCount=%f -- no avaialable stop frequency alternative to choose.", + tour.getTourCategory(), j, household.getHhId(), + person.getPersonNum(), tourCount, randomCount)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = choiceModelApplication[modelIndex].getUtilities(); + double[] probabilities = choiceModelApplication[modelIndex] + .getProbabilities(); + String[] altNames = choiceModelApplication[modelIndex] + .getAlternativeNames(); // 0s-indexing + + modelLogger.info(decisionMakerLabel); + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("------------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < altNames.length; ++k) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %15s", k + 1, altNames[k]); + modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, + utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %s", choice, altNames[choice - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", + altString, rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug log file + choiceModelApplication[modelIndex].logAlternativesInfo( + choiceModelDescription, decisionMakerLabel); + choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, choice); + + // write UEC calculation results to separate model + // specific + // log file + choiceModelApplication[modelIndex] + .logUECResults(modelLogger, loggingHeader); + + } + + // choiceResultsFreq[choice][modelIndex]++; + + // save the chosen alternative and create and populate the + // arrays + // of inbound/outbound stops in the tour object + totalStops += setStopFreqChoice(tour, choice); + totalTours++; + + tourCount++; + + } catch (Exception e) + { + logger.error(String + .format("Exception caught processing Individual Tour stop frequency choice model for %s type tour: j=%d, HHID=%d, personNum=%d, tourCount=%d.", + tour.getTourCategory(), j, household.getHhId(), + person.getPersonNum(), tourCount)); + throw new RuntimeException(e); + } + + } + + } // j (person loop) + + household.setStfRandomCount(household.getHhRandomCount()); + + } + + private int setStopFreqChoice(Tour tour, int stopFreqChoice) + { + + tour.setStopFreqChoice(stopFreqChoice); + + // set argument values for method call to get stop purpose + Household hh = tour.getPersonObject().getHouseholdObject(); + int tourDepartPeriod = tour.getTourDepartPeriod(); + int tourArrivePeriod = tour.getTourArrivePeriod(); + + //log out tour details if invalid tour departure and arrival time periods are found + if(tourDepartPeriod==-1||tourArrivePeriod==-1) tour.logTourObject(logger, 100); + + int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); + String tourPrimaryPurpose = tourPrimaryPurposeIndexNameMap.get(tourPrimaryPurposeIndex); + String personType = tour.getPersonObject().getPersonType(); + + int numObStops = dmuObject.getNumObStopsAlt(stopFreqChoice); + if ((numObStops > 0) && (tour.getEscortTypeOutbound()!=ModelStructure.RIDE_SHARING_TYPE) && (tour.getEscortTypeOutbound()!=ModelStructure.PURE_ESCORTING_TYPE)) + { + // get a stop purpose for each outbound stop generated, plus the + // stop at + // the primary destination + String[] obStopOrigPurposes = new String[numObStops + 1]; + String[] obStopDestPurposes = new String[numObStops + 1]; + int[] obStopPurposeIndices = new int[numObStops + 1]; + obStopOrigPurposes[0] = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + for (int i = 0; i < numObStops; i++) + { + if (i > 0) obStopOrigPurposes[i] = obStopDestPurposes[i - 1]; + obStopPurposeIndices[i] = getStopPurpose(hh, OUTBOUND_DIRECTION_NAME, + tourDepartPeriod, tourPrimaryPurpose, personType); + obStopDestPurposes[i] = indexPurposeMap.get(obStopPurposeIndices[i]); + } + obStopOrigPurposes[numObStops] = obStopDestPurposes[numObStops - 1]; + obStopDestPurposes[numObStops] = tourPrimaryPurpose; + // the last stop record is for the trip from stop to destination + + // pass in the array of stop purposes; length of array determines + // number + // of outbound stop objects created. + if (tour.getOutboundStops() != null) + { + Exception e = new RuntimeException(); + logger.error("outbound stops array for hhid=" + tour.getHhId() + ", person=" + + tour.getPersonObject().getPersonNum() + ", tour=" + tour.getTourId() + + ", purpose=" + tour.getTourPurpose(), e); + try + { + throw e; + } catch (Exception e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + } + tour.createOutboundStops(obStopOrigPurposes, obStopDestPurposes, obStopPurposeIndices); + } + + int numIbStops = dmuObject.getNumIbStopsAlt(stopFreqChoice); + if ((numIbStops > 0) && (tour.getEscortTypeInbound()!=ModelStructure.RIDE_SHARING_TYPE) && (tour.getEscortTypeInbound()!=ModelStructure.PURE_ESCORTING_TYPE)) + { + // get a stop purpose for each inbound stop generated + String[] ibStopOrigPurposes = new String[numIbStops + 1]; + String[] ibStopDestPurposes = new String[numIbStops + 1]; + int[] ibStopPurposeIndices = new int[numIbStops + 1]; + ibStopOrigPurposes[0] = tour.getTourPrimaryPurpose(); + for (int i = 0; i < numIbStops; i++) + { + if (i > 0) ibStopOrigPurposes[i] = ibStopDestPurposes[i - 1]; + ibStopPurposeIndices[i] = getStopPurpose(hh, INBOUND_DIRECTION_NAME, + tourArrivePeriod, tourPrimaryPurpose, personType); + ibStopDestPurposes[i] = indexPurposeMap.get(ibStopPurposeIndices[i]); + } + ibStopOrigPurposes[numIbStops] = ibStopDestPurposes[numIbStops - 1]; + ibStopDestPurposes[numIbStops] = tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; + // the last stop record is for the trip from stop to home or work + + // pass in the array of stop purposes; length of array determines + // number + // of inbound stop objects created. + if (tour.getInboundStops() != null) + { + Exception e = new RuntimeException(); + logger.error("inbound stops array for hhid=" + tour.getHhId() + ", person=" + + tour.getPersonObject().getPersonNum() + ", tour=" + tour.getTourId() + + ", purpose=" + tour.getTourPurpose(), e); + try + { + throw e; + } catch (Exception e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + } + + tour.createInboundStops(ibStopOrigPurposes, ibStopDestPurposes, ibStopPurposeIndices); + } + + return numObStops + numIbStops; + + } + + private void readPurposeLookupProportionsTable(String purposeLookupFilename) + { + + // read the stop purpose proportions into a TableDataSet + TableDataSet purposeLookupTable = null; + String fileName = ""; + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + purposeLookupTable = reader.readFile(new File(purposeLookupFilename)); + } catch (Exception e) + { + logger.error(String.format( + "Exception occurred reading stop purpose lookup proportions file: %s.", + fileName), e); + throw new RuntimeException(); + } + + // allocate a HashMap array for each direction, dimensioned to maximum + // departure hour, to map keys determined by combination of categories + // to + // proportions arrays. + int numDepartPeriods = modelStructure.getNumberOfTimePeriods(); + outProportionsMaps = new HashMap[numDepartPeriods + 1]; + inProportionsMaps = new HashMap[numDepartPeriods + 1]; + for (int i = 0; i <= numDepartPeriods; i++) + { + outProportionsMaps[i] = new HashMap(); + inProportionsMaps[i] = new HashMap(); + } + + // create a mapping between names used in lookup file and purpose names + // used + // in model + HashMap primaryPurposeMap = new HashMap(); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_WORK_NAME, + ModelStructure.WORK_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_UNIVERSITY_NAME, + ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_SCHOOL_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_ESCORT_NAME, + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_SHOPPING_NAME, + ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_EAT_OUT_NAME, + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_MAINT_NAME, + ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_VISIT_NAME, + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_DISCR_NAME, + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); + primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_WORK_BASED_NAME, + ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME); + + // create a mapping between stop purpose alternative indices selected + // from + // monte carlo process and stop purpose names used in model + // the indices are the order of the proportions columns in the table + indexPurposeMap = new HashMap(); + indexPurposeMap.put(1, "work related"); + indexPurposeMap.put(2, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(3, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(4, ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(5, ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(6, ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(7, ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(8, ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); + indexPurposeMap.put(9, ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); + + // create a mapping between names used in lookup file and person type + // names + // used in model + HashMap personTypeMap = new HashMap(); + personTypeMap.put(FT_WORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_FULL_TIME_WORKER_NAME); + personTypeMap.put(PT_WORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_PART_TIME_WORKER_NAME); + personTypeMap.put(UNIVERSITY_PERSON_TYPE_NAME, Person.PERSON_TYPE_UNIVERSITY_STUDENT_NAME); + personTypeMap.put(NONWORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_NON_WORKER_NAME); + personTypeMap.put(RETIRED_PERSON_TYPE_NAME, Person.PERSON_TYPE_RETIRED_NAME); + personTypeMap + .put(DRIVING_STUDENT_PERSON_TYPE_NAME, Person.PERSON_TYPE_STUDENT_DRIVING_NAME); + personTypeMap.put(NONDRIVING_STUDENT_PERSON_TYPE_NAME, + Person.PERSON_TYPE_STUDENT_NON_DRIVING_NAME); + personTypeMap.put(PRESCHOOL_PERSON_TYPE_NAME, Person.PERSON_TYPE_PRE_SCHOOL_CHILD_NAME); + personTypeMap.put(ALL_PERSON_TYPE_NAME, ALL_PERSON_TYPE_NAME); + + // fields in lookup file are: + // PrimPurp Direction DepartRangeStart DepartRangeEnd Ptype Work + // University + // School Escort Shop Maintenance Eating Out Visiting Discretionary + + // populate the outProportionsMaps and inProportionsMaps arrays of maps + // from + // data in the TableDataSet. + // when stops are generated, they can lookup the proportions for stop + // purpose + // selection from a map determined + // by tour purpose, person type, outbound/inbound direction and tour + // departure time. From these proportions, + // a stop purpose can be drawn. + + // loop over rows in the TableDataSet + for (int i = 0; i < purposeLookupTable.getRowCount(); i++) + { + + // get the tour primary purpose + String tourPrimPurp = primaryPurposeMap.get(purposeLookupTable.getStringValueAt(i + 1, + TOUR_PRIMARY_PURPOSE_COLUMN_HEADING)); + + // get the half tour direction + String direction = purposeLookupTable.getStringValueAt(i + 1, + HALF_TOUR_DIRECTION_COLUMN_HEADING); + + // get the beginning of the range of departure hours + int departPeriodRangeStart = (int) purposeLookupTable.getValueAt(i + 1, + TOUR_DEPARTURE_START_RANGE_COLUMN_HEADING); + + // get the end of the range of departure hours + int arriveperiodRangeEnd = (int) purposeLookupTable.getValueAt(i + 1, + TOUR_DEPARTURE_END_RANGE_COLUMN_HEADING); + + int startRange = modelStructure.getTimePeriodIndexForTime(departPeriodRangeStart); + int endRange = modelStructure.getTimePeriodIndexForTime(arriveperiodRangeEnd); + + // get the person type + String personType = personTypeMap.get(purposeLookupTable.getStringValueAt(i + 1, + PERSON_TYPE_COLUMN_HEADING)); + + // columns following person type are proportions by stop purpose. + // Get the + // index of the first stop purpose proportion. + int firstPropColumn = purposeLookupTable.getColumnPosition(PERSON_TYPE_COLUMN_HEADING) + 1; + + // starting at this column, read the proportions for all stop + // purposes. + // Create the array of proportions for this table record. + double[] props = new double[indexPurposeMap.size()]; + for (int j = 0; j < props.length; j++) + { + props[j] = purposeLookupTable.getValueAt(i + 1, firstPropColumn + j); + } + + // get a HashMap for the direction and each hour in the start/end + // range, + // and store the proportions in that map for the key. + // the key to use for any of these HashMaps is created consisting of + // "TourPrimPurp_PersonType" + // if the person type for the record is "All", a key is defined for + // each + // person type, and the proportions stored for each key. + if (personType.equalsIgnoreCase(ALL_PERSON_TYPE_NAME)) + { + for (String ptype : personTypeMap.values()) + { + String key = tourPrimPurp + "_" + ptype; + if (direction.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) + { + for (int k = startRange; k <= endRange; k++) + { + outProportionsMaps[k].put(key, props); + } + } else if (direction.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) + { + for (int k = startRange; k <= endRange; k++) + inProportionsMaps[k].put(key, props); + } + } + } else + { + String key = tourPrimPurp + "_" + personType; + if (direction.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) + { + for (int k = startRange; k <= endRange; k++) + outProportionsMaps[k].put(key, props); + } else if (direction.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) + { + for (int k = startRange; k <= endRange; k++) + inProportionsMaps[k].put(key, props); + } + } + + } + + } + + private int getStopPurpose(Household household, String halfTourDirection, int tourDepartPeriod, + String tourPrimaryPurpose, String personType) + { + + double[] props = null; + String key = tourPrimaryPurpose + "_" + personType; + + try + { + if (halfTourDirection.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) props = outProportionsMaps[tourDepartPeriod] + .get(key); + else if (halfTourDirection.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) + props = inProportionsMaps[tourDepartPeriod].get(key); + + double rn = household.getHhRandom().nextDouble(); + int choice = ChoiceModelApplication.getMonteCarloSelection(props, rn); + + return (choice + 1); + + } catch (Exception e) + { + logger.error("exception caught trying to determine stop purpose."); + logger.error("key=" + key + ", tourPrimaryPurpose=" + tourPrimaryPurpose + + ", personType=" + personType + ", halfTourDirection=" + halfTourDirection + + ", tourDepartPeriod=" + tourDepartPeriod); + throw new RuntimeException(); + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java new file mode 100644 index 0000000..180f181 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java @@ -0,0 +1,411 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Nov 4, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public class StopLocationDMU + implements Serializable, VariableTable +{ + + protected HashMap methodIndexMap; + + protected IndexValues dmuIndex; + protected Household household; + protected Person person; + protected Tour tour; + protected Stop stop; + protected ModelStructure modelStructure; + + protected int numberInSample; + protected int tourModeIndex; + protected double origDestDistance; + + // these arrays are dimensioned to the total number of location choice + // alternatives (number of MGRAs) + protected int[] walkTransitAvailableAtMgra; + protected double[] distancesFromOrigMgra; + protected double[] distancesFromTourOrigMgra; + protected double[] distancesToDestMgra; + protected double[] distancesToTourDestMgra; + protected double[] logSizeTerms; + + protected double[] bikeLogsumsFromOrigMgra; + protected double[] bikeLogsumsToDestMgra; + + + + // these arrays are dimensioned to the maximum number of alternatives in the + // sample + protected double[] mcLogsums; + protected double[] slcSoaCorrections; + protected int[] sampleArray; + + public StopLocationDMU(ModelStructure modelStructure) + { + dmuIndex = new IndexValues(); + + this.modelStructure = modelStructure; + } + + public void setDmuIndexValues(int hhid, int homeTaz, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhid); + dmuIndex.setZoneIndex(homeTaz); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (household.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug SL UEC"); + } + + } + + public void setStopObject(Stop myStop) + { + stop = myStop; + tour = stop.getTour(); + person = tour.getPersonObject(); + household = person.getHouseholdObject(); + } + + /** + * set the value for the number of unique alternatives in the sample. + * sampleArray can be indexed as i=1; i <= numberInSample. + * sampleArray.length - 1 is the maximum number of locations in the sample. + * + * @param num + * - number of unique alternatives in the sample. + */ + public void setNumberInSample(int num) + { + numberInSample = num; + } + + /** + * set the array of sample MGRA values from which the stop location MGRA + * will be selected. + * + * @param sample + * - the sample array of MGRA location choice alternatives. - use + * numberInSample as upperbound of relevant choices in sample + */ + public void setSampleOfAlternatives(int[] sample) + { + sampleArray = sample; + } + + public void setSlcSoaCorrections(double[] corrections) + { + slcSoaCorrections = corrections; + } + + public void setMcLogsums(double[] logsums) + { + mcLogsums = logsums; + } + + public void setLogSize(double[] size) + { + logSizeTerms = size; + } + + /** + * set the array of distance values from the origin MGRA of the stop to all + * MGRAs. + * + * @param distances + */ + public void setDistancesFromOrigMgra(double[] distances) + { + distancesFromOrigMgra = distances; + } + + /** + * set the array of distance values from the tour origin MGRA to all MGRAs. + * + * @param distances + */ + public void setDistancesFromTourOrigMgra(double[] distances) + { + distancesFromTourOrigMgra = distances; + } + + /** + * set the array of distance values from all MGRAs to the final destination + * MGRA of the stop. + * + * @param distances + */ + public void setDistancesToDestMgra(double[] distances) + { + distancesToDestMgra = distances; + } + + /** + * set the array of distance values from all MGRAs to the tour destination + * MGRA. + * + * @param distances + */ + public void setDistancesToTourDestMgra(double[] distances) + { + distancesToTourDestMgra = distances; + } + + /** + * @param bikeLogsumsFromOrigMgra the bikeLogsumsFromOrigMgra to set + */ + public void setBikeLogsumsFromOrigMgra(double[] bikeLogsumsFromOrigMgra) { + this.bikeLogsumsFromOrigMgra = bikeLogsumsFromOrigMgra; + } + + /** + * @param bikeLogsumsToDestMgra the bikeLogsumsToDestMgra to set + */ + public void setBikeLogsumsToDestMgra(double[] bikeLogsumsToDestMgra) { + this.bikeLogsumsToDestMgra = bikeLogsumsToDestMgra; + } + + /** + * set the OD distance value from the stop origin MGRA to the final + * destination MGRA of the stop. + * + * @param distances + */ + public void setOrigDestDistance(double distance) + { + origDestDistance = distance; + } + + /** + * set the tour mode index value for the tour of the stop being located + * + * @param tour + */ + public void setTourModeIndex(int index) + { + tourModeIndex = index; + } + + /** + * set the array of attributes for all MGRAs that says their is walk transit + * access for the indexed mgra + * + * @param tour + */ + public void setWalkTransitAvailable(int[] avail) + { + walkTransitAvailableAtMgra = avail; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getSampleMgraAlt(int alt) + { + return sampleArray[alt]; + } + + public double getSlcSoaCorrectionsAlt(int alt) + { + return slcSoaCorrections[alt]; + } + + public double getMcLogsumAlt(int alt) + { + return mcLogsums[alt]; + } + + /** + * get the logged size term from the full set of size terms for all mgra + * associated with the sample alternative + * + * @param alt + * - element number for the sample array + * @return logged size term for mgra associated with the sample element + */ + public double getLnSlcSizeSampleAlt(int alt) + { + int mgra = sampleArray[alt]; + return logSizeTerms[mgra]; + } + + /** + * get the logged size term ffrom the full set of size terms for all mgra + * alternatives + * + * @param mgra + * - mgra location alternive + * @return logged size term for mgra + */ + public double getLnSlcSizeAlt(int mgra) + { + return logSizeTerms[mgra]; + } + + protected int getTourIsJoint() + { + return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 + : 0; + } + + public int getTourMode() + { + return tour.getTourModeChoice(); + } + + public int getTourPurpose() + { + return tour.getTourPrimaryPurposeIndex(); + } + + public int getFemale() + { + return person.getPersonIsFemale(); + } + + public int getIncomeInDollars() + { + return household.getIncomeInDollars(); + } + + public int getAge() + { + return person.getAge(); + } + + public int getStopPurpose() + { + return stop.getStopPurposeIndex(); + } + + public int getStopNumber() + { + return (stop.getStopId() + 1); + } + + public int getInboundStop() + { + return stop.isInboundStop() ? 1 : 0; + } + + public int getStopsOnHalfTour() + { + return stop.isInboundStop() ? tour.getInboundStops().length + : tour.getOutboundStops().length; + } + + public double getOrigToMgraDistanceAlt(int alt) + { + // int dummy=0; + // double dist = Math.abs(distancesFromOrigMgra[alt] - + // distancesToDestMgra[alt]); + // double maxSegDist = Math.max(distancesFromOrigMgra[alt], + // distancesToDestMgra[alt]); + // if ( dist > 0 && dist < 1 && origDestDistance > 40 ) + // dummy = 1; + + return distancesFromOrigMgra[alt]; + } + + public double getTourOrigToMgraDistanceAlt(int alt) + { + return distancesFromTourOrigMgra[alt]; + } + + public double getMgraToDestDistanceAlt(int alt) + { + return distancesToDestMgra[alt]; + } + + public double getMgraToTourDestDistanceAlt(int alt) + { + return distancesToTourDestMgra[alt]; + } + + public double getOrigToMgraBikeLogsumAlt(int alt) + { + return bikeLogsumsFromOrigMgra[alt]; + } + + public double getMgraToDestBikeLogsumAlt(int alt) + { + return bikeLogsumsToDestMgra[alt]; + } + + + + public double getOdDistance() + { + return origDestDistance; + } + + public int getTourModeIsWalk() + { + boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tourModeIndex); + return tourModeIsWalk ? 1 : 0; + } + + public int getTourModeIsBike() + { + boolean tourModeIsBike = modelStructure.getTourModeIsBike(tourModeIndex); + return tourModeIsBike ? 1 : 0; + } + + public int getTourModeIsWalkTransit() + { + return (modelStructure.getTourModeIsWalkTransit(tourModeIndex) ? 1 : 0); + } + + public int getWalkTransitAvailableAlt(int alt) + { + return walkTransitAvailableAtMgra[alt]; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java new file mode 100644 index 0000000..736fd49 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java @@ -0,0 +1,956 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; + +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import org.sandag.abm.accessibilities.NonTransitUtilities; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.ResourceUtil; + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To + * change this template use File | Settings | File Templates. + */ +public class SubtourDepartureAndDurationTime + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(SubtourDepartureAndDurationTime.class); + private transient Logger todLogger = Logger.getLogger("todLogger"); + private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); + + private static final String TOD_UEC_FILE_TARGET = "departTime.uec.file"; + private static final String TOD_UEC_DATA_TARGET = "departTime.data.page"; + private static final String TOD_UEC_AT_WORK_MODEL_TARGET = "departTime.atwork.page"; + + private static String[] tourPurposeNames; + + private int[] todModelIndices; + private HashMap purposeNameIndexMap; + + private static final String[] DC_MODEL_SHEET_KEYS = { + TOD_UEC_AT_WORK_MODEL_TARGET, TOD_UEC_AT_WORK_MODEL_TARGET, + TOD_UEC_AT_WORK_MODEL_TARGET }; + + private int[] tourDepartureTimeChoiceSample; + + // DMU for the UEC + private TourDepartureTimeAndDurationDMU todDmuObject; + private TourModeChoiceDMU mcDmuObject; + + // model structure to compare the .properties time of day with the UECs + private ModelStructure modelStructure; + + private TazDataManager tazs; + private MgraDataManager mgraManager; + + // private double[][] dcSizeArray; + + private ChoiceModelApplication[] todModels; + private TourModeChoiceModel mcModel; + + private int[] altStarts; + private int[] altEnds; + + private boolean[] needToComputeLogsum; + private double[] modeChoiceLogsums; + + private short[] tempWindow; + + // create an array to count the subtours propcessed within work tours + // there are at most 2 work tours per person + private int[] subtourNumForWorkTours = new int[2]; + + private int noAltChoice = 1; + + private long mcTime; + + public SubtourDepartureAndDurationTime(HashMap propertyMap, + ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, + TourModeChoiceModel mcModel) + { + + // set the model structure + this.modelStructure = modelStructure; + this.mcModel = mcModel; + + logger.info(String.format("setting up %s time-of-day choice model.", + ModelStructure.AT_WORK_CATEGORY)); + + setupTodChoiceModels(propertyMap, dmuFactory); + } + + private void setupTodChoiceModels(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + tazs = TazDataManager.getInstance(); + mgraManager = MgraDataManager.getInstance(); + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + String todUecFileName = propertyMap.get(TOD_UEC_FILE_TARGET); + todUecFileName = uecFileDirectory + todUecFileName; + + todDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); + + mcDmuObject = dmuFactory.getModeChoiceDMU(); + + int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; + needToComputeLogsum = new boolean[numLogsumIndices]; + modeChoiceLogsums = new double[numLogsumIndices]; + + tourPurposeNames = new String[3]; + tourPurposeNames[0] = modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME; + tourPurposeNames[1] = modelStructure.AT_WORK_EAT_PURPOSE_NAME; + tourPurposeNames[2] = modelStructure.AT_WORK_MAINT_PURPOSE_NAME; + + // create the array of tod model indices + int[] uecSheetIndices = new int[tourPurposeNames.length]; + + purposeNameIndexMap = new HashMap(tourPurposeNames.length); + + int i = 0; + for (String purposeName : tourPurposeNames) + { + int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); + purposeNameIndexMap.put(purposeName, i); + uecSheetIndices[i] = uecIndex; + i++; + } + + // create a lookup array to map purpose index to model index + todModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int todModelIndex = 0; + int todSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, todModelIndex); + todModelIndices[todSegmentIndex] = todModelIndex++; + } else + { + todModelIndices[todSegmentIndex] = modelIndexMap.get(uecIndex); + } + + todSegmentIndex++; + } + + todModels = new ChoiceModelApplication[modelIndexMap.size()]; + int todModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, + TOD_UEC_DATA_TARGET); + + for (int uecIndex : modelIndexMap.keySet()) + { + int modelIndex = modelIndexMap.get(uecIndex); + try + { + todModels[modelIndex] = new ChoiceModelApplication(todUecFileName, uecIndex, + todModelDataSheet, propertyMap, (VariableTable) todDmuObject); + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up At-work Subtour TOD ChoiceModelApplication[%d] for model index=%d of %d models", + i, i, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + // get the alternatives table from the work tod UEC. + TableDataSet altsTable = todModels[0].getUEC().getAlternativeData(); + + altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); + altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); + todDmuObject.setTodAlts(altStarts, altEnds); + + int numDepartureTimeChoiceAlternatives = todModels[0].getNumberOfAlternatives(); + tourDepartureTimeChoiceSample = new int[numDepartureTimeChoiceAlternatives + 1]; + Arrays.fill(tourDepartureTimeChoiceSample, 1); + + tempWindow = new short[modelStructure.getNumberOfTimePeriods() + 1]; + + } + + public void applyModel(Household hh, boolean runModeChoice) + { + + mcTime = 0; + + Logger modelLogger = todLogger; + + // get the peron objects for this household + Person[] persons = hh.getPersons(); + for (int p = 1; p < persons.length; p++) + { + + Person person = persons[p]; + + // get the work tours for the person + ArrayList subtourList = person.getListOfAtWorkSubtours(); + + // if no work subtours for person, nothing to do. + if (subtourList.size() == 0) continue; + + ArrayList workTourList = person.getListOfWorkTours(); + int numWorkTours = workTourList.size(); + + // save a copy of this person's original time windows + short[] personWindow = person.getTimeWindows(); + for (int w = 0; w < personWindow.length; w++) + tempWindow[w] = personWindow[w]; + + for (int i = 0; i < subtourNumForWorkTours.length; i++) + subtourNumForWorkTours[i] = 0; + + int m = -1; + int tourPurpNum = 1; + int noWindowCountFirstTemp = 0; + int noWindowCountLastTemp = 0; + int noLaterAlternativeCountTemp = 0; + for (Tour t : subtourList) + { + + Tour workTour = null; + int workTourIndex = 0; + + try + { + + workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); + subtourNumForWorkTours[workTourIndex]++; + workTour = workTourList.get(workTourIndex); + + // if the first subtour for a work tour, make window of work + // tour available, and other windows not available + if (subtourNumForWorkTours[workTourIndex] == 1) + { + person.resetTimeWindow(workTour.getTourDepartPeriod(), + workTour.getTourArrivePeriod()); + person.scheduleWindow(0, workTour.getTourDepartPeriod() - 1); + person.scheduleWindow(workTour.getTourArrivePeriod() + 1, + modelStructure.getNumberOfTimePeriods()); + } else if (subtourNumForWorkTours[workTourIndex] > 2) + { + logger.error("too many subtours for a work tour. workTourIndex=" + + workTourIndex + ", subtourNumForWorkTours[workTourIndex]" + + subtourNumForWorkTours[workTourIndex]); + logger.error("hhid=" + hh.getHhId() + ", persNum=" + person.getPersonNum()); + throw new RuntimeException(); + } + + // get the choice model for the tour purpose + String tourPurposeName = t.getSubTourPurpose(); + + int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); + m = todModelIndices[tourPurposeIndex]; + + // write debug header + String separator = ""; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (hh.getDebugChoiceModels()) + { + + choiceModelDescription = String.format( + "AtWork Subtour Departure Time Choice Model for: Purpose=%s", + tourPurposeName); + decisionMakerLabel = String + .format("HH=%d, PersonNum=%d, PersonType=%s, tourId=%d, num=%d of %d %s tours", + hh.getHhId(), person.getPersonNum(), + person.getPersonType(), t.getTourId(), tourPurpNum, + subtourList.size(), tourPurposeName); + todModels[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + String loggerString = "AtWork Subtour Departure Time Choice Model: Debug Statement for Household ID: " + + hh.getHhId() + + ", Person Num: " + + person.getPersonNum() + + ", Person Type: " + + person.getPersonType() + + ", Tour Id: " + + t.getTourId() + + ", num " + + tourPurpNum + + " of " + + subtourList.size() + " " + tourPurposeName + " tours."; + for (int k = 0; k < loggerString.length(); k++) + separator += "+"; + modelLogger.info(loggerString); + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + + } + + // set the dmu object + todDmuObject.setHousehold(hh); + todDmuObject.setPerson(person); + todDmuObject.setTour(t); + + int otherTourEndHour = -1; + + // check for multiple tours for this person, by purpose + // set the first or second switch if multiple tours for + // person, by purpose + if (subtourList.size() == 1) + { + // not a multiple tour pattern + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(1); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else if (subtourList.size() > 1) + { + // Two-plus tour multiple tour pattern + if (tourPurpNum == 1) + { + // first of 2+ tours + todDmuObject.setFirstTour(1); + todDmuObject.setSubsequentTour(0); + todDmuObject.setTourNumber(tourPurpNum); + todDmuObject.setEndOfPreviousScheduledTour(0); + } else + { + // 2nd or greater tours + todDmuObject.setFirstTour(0); + todDmuObject.setSubsequentTour(1); + todDmuObject.setTourNumber(tourPurpNum); + // the ArrayList is 0-based, and we want the + // previous tour, so subtract 2 from tourPurpNum to + // get the right index + otherTourEndHour = subtourList.get(tourPurpNum - 2) + .getTourArrivePeriod(); + todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); + } + } + + // set the choice availability and sample + boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows( + altStarts, altEnds); + Arrays.fill(tourDepartureTimeChoiceSample, 1); + + if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) + { + logger.error(String + .format("error in at-work subtour departure time choice model for hhId=%d, personNum=%d, tour purpose index=%d, tour ArrayList index=%d.", + hh.getHhId(), person.getPersonNum(), tourPurposeIndex, + tourPurpNum - 1)); + logger.error(String + .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", + departureTimeChoiceAvailability.length)); + logger.error(String + .format("does not equal the length of the sample array determined by the number of alternatives in the at-work subtour UEC=%d.", + tourDepartureTimeChoiceSample.length)); + throw new RuntimeException(); + } + + // if no time window is available for the tour, make the + // first and last alternatives available + // for that alternative, and keep track of the number of + // times this condition occurs. + int alternativeAvailable = -1; + for (int a = 0; a < departureTimeChoiceAvailability.length; a++) + { + if (departureTimeChoiceAvailability[a]) + { + alternativeAvailable = a; + break; + } + } + + int chosen = -1; + int chosenStartPeriod = -1; + int chosenEndPeriod = -1; + + // alternate making the first and last periods chosen if no + // alternatives are available + if (alternativeAvailable < 0) + { + + if (noAltChoice == 1) + { + if (subtourList.size() > 1 && tourPurpNum > 1) + { + chosenStartPeriod = otherTourEndHour; + chosenEndPeriod = otherTourEndHour; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to previous sub-tour arrive period=" + + chosenStartPeriod + "."); + } else + { + chosenStartPeriod = workTour.getTourDepartPeriod(); + chosenEndPeriod = workTour.getTourDepartPeriod(); + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled and no previous sub-tour, depart AND arrive set to work tour depart period=" + + chosenStartPeriod + "."); + } + noWindowCountFirstTemp++; + noAltChoice = departureTimeChoiceAvailability.length - 1; + } else + { + if (subtourList.size() > 1 && tourPurpNum > 1) + { + chosenStartPeriod = otherTourEndHour; + chosenEndPeriod = otherTourEndHour; + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled, depart AND arrive set to previous sub-tour arrive period=" + + chosenStartPeriod + "."); + } else + { + chosenStartPeriod = workTour.getTourArrivePeriod(); + chosenEndPeriod = workTour.getTourArrivePeriod(); + if (hh.getDebugChoiceModels()) + modelLogger + .info("All alternatives already scheduled and no previous sub-tour, depart AND arrive set to work tour arrive period=" + + chosenStartPeriod + "."); + } + noWindowCountLastTemp++; + noAltChoice = 1; + } + + // schedule the chosen alternative + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + + if (runModeChoice) + { + + long check = System.nanoTime(); + + if (hh.getDebugChoiceModels()) + hh.logHouseholdObject( + "Pre At-work Subtour Tour Mode Choice Household " + + hh.getHhId() + ", Tour " + tourPurpNum + " of " + + subtourList.size(), tourMCNonManLogger); + + // set the mode choice attributes needed by + // @variables in the UEC spreadsheets + setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, + chosenEndPeriod); + + // use the mcModel object already setup for + // computing logsums and get + // the mode choice, where the selected + // worklocation and subzone an departure time and + // duration are set + // for this work tour. + int chosenMode = mcModel.getModeChoice(mcDmuObject, + t.getTourPrimaryPurpose()); + t.setTourModeChoice(chosenMode); + + mcTime += (System.nanoTime() - check); + + } + + } else + { + + // calculate and store the mode choice logsum for the + // usual work location for this worker at the various + // departure time and duration alternativees + setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, + departureTimeChoiceAvailability); + + if (hh.getDebugChoiceModels()) + { + hh.logTourObject(loggingHeader, modelLogger, person, t); + } + + todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); + todDmuObject.setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); + + float logsum = (float) todModels[m].computeUtilities(todDmuObject, todDmuObject.getIndexValues(), + departureTimeChoiceAvailability, tourDepartureTimeChoiceSample); + t.setTimeOfDayLogsum(logsum); + + Random hhRandom = hh.getHhRandom(); + int randomCount = hh.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available + // alternative, make choice. + if (todModels[m].getAvailabilityCount() > 0) + { + chosen = todModels[m].getChoiceResult(rn); + + // debug output + if (hh.getDebugChoiceModels()) + { + + double[] utilities = todModels[m].getUtilities(); + double[] probabilities = todModels[m].getProbabilities(); + boolean[] availabilities = todModels[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + + personTypeString + ", Tour Id: " + t.getTourId() + + ", Tour num: " + tourPurpNum + " of " + + subtourList.size() + " " + tourPurposeName + " tours."); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("-------------------- ------------ -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d out=%-3d, in=%-3d", + k + 1, altStarts[k], altEnds[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", + altString, availabilities[k + 1], utilities[k], + probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, + altStarts[chosen - 1], altEnds[chosen - 1]); + modelLogger.info(String.format( + "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, + randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to debug + // log file + todModels[m].logAlternativesInfo(choiceModelDescription, + decisionMakerLabel); + todModels[m].logSelectionInfo(choiceModelDescription, + decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate + // model specific log file + loggingHeader = String.format("%s for %s", choiceModelDescription, + decisionMakerLabel); + todModels[m].logUECResults(modelLogger, loggingHeader); + + } + + } else + { + + // since there were no alternatives with valid + // utility, assuming previous + // tour of this type scheduled up to the last + // period, so no periods left + // for this tour. + + // TODO: do a formal check for this so we can still + // flag other reasons why there's + // no valid utility for any alternative + chosen = departureTimeChoiceAvailability.length - 1; + noLaterAlternativeCountTemp++; + + } + + // schedule the chosen alternative + chosenStartPeriod = altStarts[chosen - 1]; + chosenEndPeriod = altEnds[chosen - 1]; + person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); + t.setTourDepartPeriod(chosenStartPeriod); + t.setTourArrivePeriod(chosenEndPeriod); + + if (runModeChoice) + { + + if (hh.getDebugChoiceModels()) + hh.logHouseholdObject( + "Pre At-work Subtour Tour Mode Choice Household " + + hh.getHhId() + ", Tour " + tourPurpNum + " of " + + subtourList.size(), tourMCNonManLogger); + + // set the mode choice attributes needed by + // @variables in the UEC spreadsheets + setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, + chosenEndPeriod); + + // use the mcModel object already setup for + // computing logsums and get + // the mode choice, where the selected + // worklocation and subzone an departure time and + // duration are set + // for this work tour. + int chosenMode = mcModel.getModeChoice(mcDmuObject, + t.getTourPrimaryPurpose()); + t.setTourModeChoice(chosenMode); + + } + + } + + } catch (Exception e) + { + String errorMessage = String + .format("Exception caught for HHID=%d, personNum=%d, At-work Subtour Departure time choice, tour ArrayList index=%d.", + hh.getHhId(), person.getPersonNum(), tourPurpNum - 1); + String decisionMakerLabel = String + .format("Final At-work Subtour Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), person.getPersonNum(), person.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, person); + todModels[m].logUECResults(modelLogger, errorMessage); + + logger.error(errorMessage, e); + throw new RuntimeException(); + } + + tourPurpNum++; + + } + + for (int w = 0; w < person.getTimeWindows().length; w++) + person.getTimeWindows()[w] = tempWindow[w]; + + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String + .format("Final At-work Subtour Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", + hh.getHhId(), person.getPersonNum(), person.getPersonType()); + hh.logPersonObject(decisionMakerLabel, modelLogger, person); + } + + } + + hh.setAwtodRandomCount(hh.getHhRandomCount()); + + } + + private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, + int startPeriod, int endPeriod) + { + + t.setTourDepartPeriod(startPeriod); + t.setTourArrivePeriod(endPeriod); + + int workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); + Tour workTour = person.getListOfWorkTours().get(workTourIndex); + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(t); + mcDmuObject.setWorkTourObject(workTour); + mcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), + t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); + + + mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t + .getTourOrigMgra()))); + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t + .getTourDestMgra()))); + + mcDmuObject.setOriginMgra(t.getTourOrigMgra()); + mcDmuObject.setDestMgra(t.getTourDestMgra()); + + } + + private void setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Tour tour, + boolean[] altAvailable) + { + + Person person = tour.getPersonObject(); + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todLogger; + String choiceModelDescription = String.format( + "At-work Subtour Mode Choice Logsum calculation for %s Departure Time Choice", + tour.getTourPurpose()); + String decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person + .getPersonNum(), person.getPersonType(), tour.getTourId(), person + .getListOfWorkTours().size()); + String loggingHeader = String + .format("%s %s", choiceModelDescription, decisionMakerLabel); + + for (int a = 1; a <= altStarts.length; a++) + { + + // if the depart/arrive alternative is unavailable, no need to check + // to see if a logsum has been calculated + if (!altAvailable[a]) continue; + + int startPeriod = altStarts[a - 1]; + int endPeriod = altEnds[a - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " work tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + throw new RuntimeException(e); + } + needToComputeLogsum[index] = false; + } + + } + + todDmuObject.setModeChoiceLogsums(modeChoiceLogsums); + + mcDmuObject.getTourObject().setTourDepartPeriod(0); + mcDmuObject.getTourObject().setTourArrivePeriod(0); + } + + public long getModeChoiceTime() + { + return mcTime; + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); + + MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, + Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + /* + */ + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TazDataManager tazManager = TazDataManager.getInstance(propertyMap); + + BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); + if (!aggAcc.getAccessibilitiesAreBuilt()) + { + aggAcc.setupBuildAccessibilities(propertyMap, false); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + // aggAcc.buildAccessibilityComponents(propertyMap); + + boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, + CtrampApplication.READ_ACCESSIBILITIES); + if (readAccessibilities) + { + + // output data + String projectDirectory = propertyMap + .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + + } + + double[][] expConstants = aggAcc.getExpConstants(); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + + double[][][] sovExpUtilities = null; + double[][][] hovExpUtilities = null; + double[][][] nMotorExpUtilities = null; + double[][][] maasExpUtilities = null; + NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, + hovExpUtilities, nMotorExpUtilities, maasExpUtilities); + + MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( + propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); + + TourModeChoiceModel awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + ModelStructure.AT_WORK_CATEGORY, dmuFactory, logsumHelper); + + SubtourDestChoiceModel testObject = new SubtourDestChoiceModel(propertyMap, modelStructure, + aggAcc, dmuFactory, awmcModel); + System.out.println("SubtourDestChoiceModel object creation passed."); + + SubtourDepartureAndDurationTime testObject2 = new SubtourDepartureAndDurationTime( + propertyMap, modelStructure, dmuFactory, awmcModel); + System.out.println("SubtourDepartureAndDurationTime object creation passed."); + + // String hhHandlerAddress = (String) + // propertyMap.get("RunModel.HouseholdServerAddress"); + // int hhServerPort = Integer.parseInt((String) + // propertyMap.get("RunModel.HouseholdServerPort")); + // + // HouseholdDataManagerIf householdDataManager = new + // HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, + // SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + // + // + // householdDataManager.setPropertyFileValues(propertyMap); + // + // // have the household data manager read the synthetic population + // // files and apply its tables to objects mapping method. + // boolean restartHhServer = false; + // try + // { + // // possible values for the following can be none, ao, cdap, imtf, + // // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, + // // stf, stl + // String restartModel = (String) + // propertyMap.get("RunModel.RestartWithHhServer"); + // if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; + // else if (restartModel.equalsIgnoreCase("uwsl") + // || restartModel.equalsIgnoreCase("ao") + // || restartModel.equalsIgnoreCase("fp") + // || restartModel.equalsIgnoreCase("cdap") + // || restartModel.equalsIgnoreCase("imtf") + // || restartModel.equalsIgnoreCase("imtod") + // || restartModel.equalsIgnoreCase("awf") + // || restartModel.equalsIgnoreCase("awl") + // || restartModel.equalsIgnoreCase("awtod") + // || restartModel.equalsIgnoreCase("jtf") + // || restartModel.equalsIgnoreCase("jtl") + // || restartModel.equalsIgnoreCase("jtod") + // || restartModel.equalsIgnoreCase("inmtf") + // || restartModel.equalsIgnoreCase("inmtl") + // || restartModel.equalsIgnoreCase("inmtod") + // || restartModel.equalsIgnoreCase("stf") + // || restartModel.equalsIgnoreCase("stl")) restartHhServer = false; + // } catch (MissingResourceException e) + // { + // restartHhServer = true; + // } + // + // if (restartHhServer) + // { + // + // householdDataManager.setDebugHhIdsFromHashmap(); + // + // String inputHouseholdFileName = (String) + // propertyMap.get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + // String inputPersonFileName = (String) + // propertyMap.get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + // householdDataManager.setHouseholdSampleRate( 1.0f, 0 ); + // householdDataManager.setupHouseholdDataManager(modelStructure, null, + // inputHouseholdFileName, inputPersonFileName); + // + // } else + // { + // + // householdDataManager.setHouseholdSampleRate( 1.0f, 0 ); + // householdDataManager.setDebugHhIdsFromHashmap(); + // householdDataManager.setTraceHouseholdSet(); + // + // } + + // int id = householdDataManager.getArrayIndex( 1033380 ); + // int id = householdDataManager.getArrayIndex( 1033331 ); + // int id = householdDataManager.getArrayIndex( 423804 ); + // Household[] hh = householdDataManager.getHhArray( id, id ); + // testObject.applyModel( hh[0] ); + // testObject2.applyModel( hh[0], true ); + + /** + * used this block of code to test for typos and implemented dmu + * methiods in the TOD choice UECs + * + * String uecFileDirectory = propertyMap.get( + * CtrampApplication.PROPERTIES_UEC_PATH ); String todUecFileName = + * propertyMap.get( TOD_UEC_FILE_TARGET ); todUecFileName = + * uecFileDirectory + todUecFileName; + * + * ModelStructure modelStructure = new SandagModelStructure(); + * SandagCtrampDmuFactory dmuFactory = new + * SandagCtrampDmuFactory(modelStructure); + * TourDepartureTimeAndDurationDMU todDmuObject = + * dmuFactory.getTourDepartureTimeAndDurationDMU(); + * + * File uecFile = new File(todUecFileName); UtilityExpressionCalculator + * uec = new UtilityExpressionCalculator(uecFile, 10, 0, propertyMap, + * (VariableTable) todDmuObject); + * System.out.println("Subtour departure and duration UEC passed"); + */ + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java new file mode 100644 index 0000000..1d44542 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java @@ -0,0 +1,1185 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.Random; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; + +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; +import org.sandag.abm.accessibilities.NonTransitUtilities; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.util.IndexSort; +import com.pb.common.util.ResourceUtil; + +public class SubtourDestChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(SubtourDestChoiceModel.class); + private transient Logger dcNonManLogger = Logger.getLogger("tourDcNonMan"); + private transient Logger todMcLogger = Logger.getLogger("todMcLogsum"); + + // TODO eventually remove this target + private static final String PROPERTIES_DC_UEC_FILE = "nmdc.uec.file"; + private static final String PROPERTIES_DC_UEC_FILE2 = "nmdc.uec.file2"; + private static final String PROPERTIES_DC_SOA_UEC_FILE = "nmdc.soa.uec.file"; + + private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; + + private static final String PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY = "nmdc.soa.SampleSize"; + + private static final String PROPERTIES_DC_DATA_SHEET = "nmdc.data.page"; + + private static final String PROPERTIES_DC_AT_WORK_MODEL_SHEET = "nmdc.atwork.model.page"; + + private static final String PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET = "nmdc.soa.atwork.model.page"; + + // size term array indices for purposes are 0-based + public static final int PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET = 11; + public static final int PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET = 10; + public static final int PROPERTIES_AT_WORK_OTHER_SIZE_SHEET = 9; + + private static String[] tourPurposeNames; + private static int[] tourPurposeIndices; + + // all three subtour purposes use the same DC sheet + private static final String[] DC_MODEL_SHEET_KEYS = { + PROPERTIES_DC_AT_WORK_MODEL_SHEET, PROPERTIES_DC_AT_WORK_MODEL_SHEET, + PROPERTIES_DC_AT_WORK_MODEL_SHEET }; + + // all three subtour purposes use the same SOA sheet + private static final String[] DC_SOA_MODEL_SHEET_KEYS = { + PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET, PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET, + PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET }; + + // all three subtour purposes use the same SOA sheet + private final int[] sizeSheetKeys = { + PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET, PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET, + PROPERTIES_AT_WORK_OTHER_SIZE_SHEET }; + + // set default depart periods that represents each model period + private static final int EA = 1; + private static final int AM = 8; + private static final int MD = 16; + private static final int PM = 26; + private static final int EV = 36; + + private static final int[][][] PERIOD_COMBINATIONS = { + { {AM, AM}, {MD, MD}, {PM, PM}}, { {AM, AM}, {MD, MD}, {PM, PM}}, + { {AM, AM}, {MD, MD}, {PM, PM}} }; + + private static final double[][] PERIOD_COMBINATION_COEFFICIENTS = { + {-3.1453, -0.1029, -2.9056}, {-3.1453, -0.1029, -2.9056}, {-3.1453, -0.1029, -2.9056}}; + + private String tourCategory; + private ModelStructure modelStructure; + + private int[] dcModelIndices; + private HashMap purposeNameIndexMap; + + private HashMap subtourSegmentNameIndexMap; + + private double[][] dcSizeArray; + + private TourModeChoiceDMU mcDmuObject; + private DestChoiceDMU dcDmuObject; + private DestChoiceTwoStageModelDMU dcDistSoaDmuObject; + private DcSoaDMU dcSoaDmuObject; + + private boolean[] needToComputeLogsum; + private double[] modeChoiceLogsums; + + private TourModeChoiceModel mcModel; + private DestinationSampleOfAlternativesModel dcSoaModel; + private ChoiceModelApplication[] dcModel; + private ChoiceModelApplication[] dcModel2; + + private boolean[] dcModel2AltsAvailable; + private int[] dcModel2AltsSample; + private int[] dcModel2SampleValues; + + private BuildAccessibilities aggAcc; + + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private double[] mgraDistanceArray; + + private DestChoiceTwoStageModel dcSoaTwoStageObject; + + private boolean useNewSoaMethod; + + private int soaSampleSize; + + private long soaRunTime; + + public SubtourDestChoiceModel(HashMap propertyMap, + ModelStructure myModelStructure, BuildAccessibilities myAggAcc, + CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel myMcModel) + { + + tourCategory = ModelStructure.AT_WORK_CATEGORY; + modelStructure = myModelStructure; + mcModel = myMcModel; + aggAcc = myAggAcc; + + tourPurposeIndices = new int[3]; + tourPurposeIndices[0] = modelStructure.AT_WORK_PURPOSE_INDEX_BUSINESS; + tourPurposeIndices[1] = modelStructure.AT_WORK_PURPOSE_INDEX_EAT; + tourPurposeIndices[2] = modelStructure.AT_WORK_PURPOSE_INDEX_MAINT; + + tourPurposeNames = new String[3]; + tourPurposeNames[0] = modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME; + tourPurposeNames[1] = modelStructure.AT_WORK_EAT_PURPOSE_NAME; + tourPurposeNames[2] = modelStructure.AT_WORK_MAINT_PURPOSE_NAME; + + logger.info(String.format("creating %s subtour dest choice mode instance", tourCategory)); + + mgraManager = MgraDataManager.getInstance(); + tazs = TazDataManager.getInstance(); + + soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, + PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); + + useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, + USE_NEW_SOA_METHOD_PROPERTY_KEY); + + if (useNewSoaMethod) + dcSoaTwoStageObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); + + // create an array of ChoiceModelApplication objects for each choice + // purpose + setupDestChoiceModelArrays(propertyMap, dmuFactory); + + } + + private void setupDestChoiceModelArrays(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + + String dcUecFileName = propertyMap.get(PROPERTIES_DC_UEC_FILE); + dcUecFileName = uecFileDirectory + dcUecFileName; + + String dcUecFileName2 = propertyMap.get(PROPERTIES_DC_UEC_FILE2); + dcUecFileName2 = uecFileDirectory + dcUecFileName2; + + String soaUecFileName = propertyMap.get(PROPERTIES_DC_SOA_UEC_FILE); + soaUecFileName = uecFileDirectory + soaUecFileName; + + int dcModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, + PROPERTIES_DC_DATA_SHEET); + int soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, + PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); + + dcDmuObject = dmuFactory.getDestChoiceDMU(); + dcDmuObject.setAggAcc(aggAcc); + dcDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); + + if (useNewSoaMethod) + { + dcDistSoaDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); + dcDistSoaDmuObject.setAggAcc(aggAcc); + dcDistSoaDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); + } + + dcSoaDmuObject = dmuFactory.getDcSoaDMU(); + dcSoaDmuObject.setAggAcc(aggAcc); + + mcDmuObject = dmuFactory.getModeChoiceDMU(); + + int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; + needToComputeLogsum = new boolean[numLogsumIndices]; + modeChoiceLogsums = new double[numLogsumIndices]; + + // create the arrays of dc model and soa model indices + int[] uecSheetIndices = new int[tourPurposeNames.length]; + int[] soaUecSheetIndices = new int[tourPurposeNames.length]; + + purposeNameIndexMap = new HashMap(tourPurposeNames.length); + + int i = 0; + for (String purposeName : tourPurposeNames) + { + int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); + int soaUecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, + DC_SOA_MODEL_SHEET_KEYS[i]); + purposeNameIndexMap.put(purposeName, i); + uecSheetIndices[i] = uecIndex; + soaUecSheetIndices[i] = soaUecIndex; + i++; + } + + // create a lookup array to map purpose index to model index + dcModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int dcModelIndex = 0; + int dcSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, dcModelIndex); + dcModelIndices[dcSegmentIndex] = dcModelIndex++; + } else + { + dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); + } + + dcSegmentIndex++; + } + + // the size term array in aggAcc gives mgra*purpose - need an array of + // all mgras for one purpose + double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); + subtourSegmentNameIndexMap = new HashMap(); + for (int k = 0; k < tourPurposeIndices.length; k++) + { + subtourSegmentNameIndexMap.put(tourPurposeNames[k], k); + } + + dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; + for (i = 0; i < aggAccDcSizeArray.length; i++) + { + for (int m : subtourSegmentNameIndexMap.values()) + { + int s = sizeSheetKeys[m]; + dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; + } + } + + // create a sample of alternatives choice model object for use in + // selecting a sample + // of all possible destination choice alternatives. + dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFileName, soaSampleSize, + propertyMap, mgraManager, dcSizeArray, dcSoaDmuObject, soaUecSheetIndices); + + dcModel = new ChoiceModelApplication[modelIndexMap.size()]; + + if (useNewSoaMethod) + { + dcModel2 = new ChoiceModelApplication[modelIndexMap.size()]; + dcModel2AltsAvailable = new boolean[soaSampleSize + 1]; + dcModel2AltsSample = new int[soaSampleSize + 1]; + dcModel2SampleValues = new int[soaSampleSize]; + } + + i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + + try + { + dcModel[i] = new ChoiceModelApplication(dcUecFileName, uecIndex, dcModelDataSheet, + propertyMap, (VariableTable) dcDmuObject); + + if (useNewSoaMethod) + { + dcModel2[i] = new ChoiceModelApplication(dcUecFileName2, uecIndex, + dcModelDataSheet, propertyMap, (VariableTable) dcDistSoaDmuObject); + } + + i++; + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up ATWork DC ChoiceModelApplication[%d] for model index=%d of %d models", + i, i, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; + + } + + public void applyModel(Household hh) + { + + soaRunTime = 0; + + if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); + else dcSoaModel.resetSoaRunTime(); + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + + Logger modelLogger = dcNonManLogger; + if (hh.getDebugChoiceModels()) + hh.logHouseholdObject("Pre Subtour Location Choice Household " + hh.getHhId() + + " Object", modelLogger); + + Person[] persons = hh.getPersons(); + + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + // get the at-work subtours for this person and choose a destination + // for each. + ArrayList tourList = p.getListOfAtWorkSubtours(); + + int currentTourNum = 0; + for (Tour tour : tourList) + { + + Tour workTour = null; + int workTourIndex = 0; + workTourIndex = tour.getWorkTourIndexFromSubtourId(tour.getTourId()); + workTour = p.getListOfWorkTours().get(workTourIndex); + + int chosen = -1; + try + { + + int homeMgra = hh.getHhMgra(); + int homeTaz = hh.getHhTaz(); + int origMgra = workTour.getTourDestMgra(); + tour.setTourOrigMgra(origMgra); + + // update the MC dmuObject for this person + mcDmuObject.setHouseholdObject(hh); + mcDmuObject.setPersonObject(p); + mcDmuObject.setTourObject(tour); + mcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0, + hh.getDebugChoiceModels()); + mcDmuObject.setOriginMgra(origMgra); + + // update the DC dmuObject for this person + dcDmuObject.setHouseholdObject(hh); + dcDmuObject.setPersonObject(p); + dcDmuObject.setTourObject(tour); + dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); + + if (useNewSoaMethod) + { + dcDistSoaDmuObject.setHouseholdObject(hh); + dcDistSoaDmuObject.setPersonObject(p); + dcDistSoaDmuObject.setTourObject(tour); + dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); + } + + // for At-work Subtour DC, just count remaining At-work + // Subtour tours + int toursLeftCount = tourList.size() - currentTourNum; + dcDmuObject.setToursLeftCount(toursLeftCount); + if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); + + // get the tour location alternative chosen from the sample + if (useNewSoaMethod) + { + chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); + soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); + } else + { + chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, + dcSoaDmuObject, mcDmuObject); + soaRunTime += dcSoaModel.getSoaRunTime(); + } + + } catch (RuntimeException e) + { + logger.fatal(String + .format("exception caught selecting %s tour destination choice for hh.hhid=%d, personNum=%d, tourId=%d, purposeName=%s", + tourCategory, hh.getHhId(), p.getPersonNum(), tour.getTourId(), + tour.getSubTourPurpose())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + // set chosen values in tour object + tour.setTourDestMgra(chosen); + + currentTourNum++; + } + + } + + hh.setAwlRandomCount(hh.getHhRandomCount()); + + } + + /** + * + * @return chosen mgra. + */ + private int selectLocationFromSampleOfAlternatives(Tour tour, DestChoiceDMU dcDmuObject, + DcSoaDMU dcSoaDmuObject, TourModeChoiceDMU mcDmuObject) + { + + Logger modelLogger = dcNonManLogger; + + // get the Household object for the person making this subtour + Person person = tour.getPersonObject(); + + // get the Household object for the person making this subtour + Household household = person.getHouseholdObject(); + + // get the tour purpose name + String tourPurposeName = tour.getSubTourPurpose(); + int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); + + dcSoaDmuObject.setDestChoiceSize(dcSizeArray[tourPurposeIndex]); + + // the originMgra in the tour object is already set to the work tour + // dest mgra + // double[] workMgraDistanceArray = + // mandAcc.calculateDistancesForAllMgras( tour.getTourOrigMgra() ); + mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(tour.getTourOrigMgra(), + mgraDistanceArray); + dcSoaDmuObject.setDestDistance(mgraDistanceArray); + + dcDmuObject.setDestChoiceSize(dcSizeArray[tourPurposeIndex]); + dcDmuObject.setDestChoiceDistance(mgraDistanceArray); + + // compute the sample of alternatives set for the person + dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, tour, person, + tourPurposeName, tourPurposeIndex, tour.getTourOrigMgra()); + + // get sample of locations and correction factors for sample + int[] finalSample = dcSoaModel.getSampleOfAlternatives(); + float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); + + int m = dcModelIndices[tourPurposeIndex]; + int numAlts = dcModel[m].getNumberOfAlternatives(); + + // set the destAltsAvailable array to true for all destination choice + // alternatives for each purpose + boolean[] destAltsAvailable = new boolean[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsAvailable[k] = false; + + // set the destAltsSample array to 1 for all destination choice + // alternatives + // for each purpose + int[] destAltsSample = new int[numAlts + 1]; + for (int k = 0; k <= numAlts; k++) + destAltsSample[k] = 0; + + int[] sampleValues = new int[finalSample.length]; + + // for the destinations and sub-zones in the sample, compute mc logsums + // and + // save in DC dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 1; i < finalSample.length; i++) + { + + int destMgra = finalSample[i]; + sampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); + dcDmuObject.setMcLogsum(destMgra, logsum); + + // set sample of alternatives correction factor used in destination + // choice utility for the sampled alternative. + dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); + + // set availaibility and sample values for the purpose, dcAlt. + destAltsAvailable[finalSample[i]] = true; + destAltsSample[finalSample[i]] = 1; + + } + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "At-work Subtour Location Choice Model for: tour purpose=%s", tourPurposeName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("At-work Subtour Location Choice Model for tour purpose=" + + tourPurposeName + ", Person Num: " + person.getPersonNum() + + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + dcModel[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float logsum = (float) dcModel[m].computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), + destAltsAvailable, destAltsSample); + + tour.setTourDestinationLogsum(logsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (dcModel[m].getAvailabilityCount() > 0) + { + try + { + chosen = dcModel[m].getChoiceResult(rn); + } catch (Exception e) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), tour.getTourId(), + tourPurposeName)); + throw new RuntimeException(); + } + } + + // write choice model alternative info to log file + int selectedIndex = -1; + for (int j = 1; j < finalSample.length; j++) + { + if (finalSample[j] == chosen) + { + selectedIndex = j; + break; + } + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = dcModel[m].getUtilities(); + double[] probabilities = dcModel[m].getProbabilities(); + boolean[] availabilities = dcModel[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); + + double cumProb = 0.0; + for (int j = 1; j < finalSample.length; j++) + { + int k = sortedSampleValueIndices[j]; + int alt = finalSample[k]; + + if (finalSample[k] == chosen) selectedIndex = j; + + cumProb += probabilities[alt - 1]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[alt], utilities[alt - 1], probabilities[alt - 1], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", selectedIndex, chosen); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + dcModel[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + dcModel[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + dcModel[m].logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", + dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject + .getPersonObject().getPersonNum(), tour.getTourId(), + tourPurposeName)); + throw new RuntimeException(); + } + + } + + return chosen; + + } + + /** + * + * @return chosen mgra. + */ + private int selectLocationFromTwoStageSampleOfAlternatives(Tour tour, + TourModeChoiceDMU mcDmuObject) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcNonManLogger; + + // get the Household object for the person making this non-mandatory + // tour + Person person = tour.getPersonObject(); + + // get the Household object for the person making this non-mandatory + // tour + Household household = person.getHouseholdObject(); + + // get the tour purpose name + String tourPurposeName = tour.getSubTourPurpose(); + int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); + + // get sample of locations and correction factors for sample using the + // alternate method + // for non-mandatory tour destination choice, the sizeSegmentType INdex + // and sizeSegmentIndex are the same values. + dcSoaTwoStageObject.chooseSample(mgraManager.getTaz(tour.getTourOrigMgra()), + tourPurposeIndex, tourPurposeIndex, soaSampleSize, household.getHhRandom(), + household.getDebugChoiceModels()); + int[] finalSample = dcSoaTwoStageObject.getUniqueSampleMgras(); + double[] sampleCorrectionFactors = dcSoaTwoStageObject + .getUniqueSampleMgraCorrectionFactors(); + int numUniqueAlts = dcSoaTwoStageObject.getNumberofUniqueMgrasInSample(); + + int m = dcModelIndices[tourPurposeIndex]; + int numAlts = dcModel2[m].getNumberOfAlternatives(); + + Arrays.fill(dcModel2AltsAvailable, false); + Arrays.fill(dcModel2AltsSample, 0); + Arrays.fill(dcModel2SampleValues, 999999); + + mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(tour.getTourOrigMgra(), + mgraDistanceArray); + dcDistSoaDmuObject.setMgraDistanceArray(mgraDistanceArray); + + int sizeIndex = subtourSegmentNameIndexMap.get(tourPurposeName); + dcDistSoaDmuObject.setMgraSizeArray(dcSizeArray[sizeIndex]); + + // set sample of alternatives correction factors used in destination + // choice utility for the sampled alternatives. + dcDistSoaDmuObject.setDcSoaCorrections(sampleCorrectionFactors); + + // for the destination mgras in the sample, compute mc logsums and save + // in dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 0; i < numUniqueAlts; i++) + { + + int destMgra = finalSample[i]; + dcModel2SampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); + dcDistSoaDmuObject.setMcLogsum(i, logsum); + + // set availaibility and sample values for the purpose, dcAlt. + dcModel2AltsAvailable[i + 1] = true; + dcModel2AltsSample[i + 1] = 1; + + } + + dcDistSoaDmuObject.setSampleArray(dcModel2SampleValues); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" + + tourPurposeName + ", Person Num: " + person.getPersonNum() + + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + dcModel2[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float modelLogsum = (float) dcModel2[m].computeUtilities(dcDistSoaDmuObject, dcDistSoaDmuObject.getDmuIndexValues(), + dcModel2AltsAvailable, dcModel2AltsSample); + tour.setTourDestinationLogsum(modelLogsum); + + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (dcModel2[m].getAvailabilityCount() > 0) + { + try + { + chosen = dcModel2[m].getChoiceResult(rn); + } catch (Exception e) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", + dcDistSoaDmuObject.getHouseholdObject().getHhId(), + dcDistSoaDmuObject.getPersonObject().getPersonNum(), + tour.getTourId(), tourPurposeName)); + throw new RuntimeException(); + } + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = dcModel2[m].getUtilities(); + double[] probabilities = dcModel2[m].getProbabilities(); + boolean[] availabilities = dcModel2[m].getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < finalSample.length; j++) + { + int alt = finalSample[j]; + cumProb += probabilities[j]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + dcModel2[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + dcModel2[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + + // write UEC calculation results to separate model specific log file + dcModel2[m].logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", + dcDistSoaDmuObject.getHouseholdObject().getHhId(), + dcDistSoaDmuObject.getPersonObject().getPersonNum(), + tour.getTourId(), tourPurposeName)); + throw new RuntimeException(); + } + + } + + return chosen; + + } + + private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, + int startPeriod, int endPeriod, int sampleDestMgra) + { + + t.setTourDestMgra(sampleDestMgra); + t.setTourDepartPeriod(startPeriod); + t.setTourArrivePeriod(endPeriod); + + int workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); + Tour workTour = person.getListOfWorkTours().get(workTourIndex); + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(t); + mcDmuObject.setWorkTourObject(workTour); + mcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), + t.getTourOrigMgra(), sampleDestMgra, household.getDebugChoiceModels()); + + + mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t + .getTourOrigMgra()))); + mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager + .getTaz(sampleDestMgra))); + + mcDmuObject.setOriginMgra(t.getTourOrigMgra()); + mcDmuObject.setDestMgra(t.getTourDestMgra()); + + } + + private double calculateSimpleTODChoiceLogsum(Person person, Tour tour, int sampleDestMgra, + int sampleNum) + { + + Household household = person.getHouseholdObject(); + + Arrays.fill(needToComputeLogsum, true); + Arrays.fill(modeChoiceLogsums, -999); + + Logger modelLogger = todMcLogger; + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + choiceModelDescription = String + .format("At-work Subtour Simplified TOD logsum calculations for %s Location Choice, Sample Number %d", + tour.getSubTourPurpose(), sampleNum); + decisionMakerLabel = String.format( + "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", + household.getHhId(), person.getPersonNum(), person.getPersonType(), + tour.getTourId(), person.getListOfAtWorkSubtours().size()); + loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); + } + + int i = 0; + int tourPurposeIndex = purposeNameIndexMap.get(tour.getSubTourPurpose()); + double totalExpUtility = 0.0; + for (int[] combo : PERIOD_COMBINATIONS[tourPurposeIndex]) + { + int startPeriod = combo[0]; + int endPeriod = combo[1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + if (needToComputeLogsum[index]) + { + + String periodString = modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)); + + // set the mode choice attributes needed by @variables in the + // UEC spreadsheets + setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod, + sampleDestMgra); + + if (household.getDebugChoiceModels()) + household.logTourObject(loggingHeader + ", " + periodString, modelLogger, + person, mcDmuObject.getTourObject()); + + try + { + modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, + modelLogger, choiceModelDescription, decisionMakerLabel + ", " + + periodString); + } catch (Exception e) + { + logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " + + periodString + " " + tour.getTourPrimaryPurpose() + " tour."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + e.printStackTrace(); + System.exit(-1); + // throw new RuntimeException(e); + } + needToComputeLogsum[index] = false; + } + + double expUtil = Math.exp(modeChoiceLogsums[index] + + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]); + totalExpUtility += expUtil; + + if (household.getDebugChoiceModels()) + modelLogger + .info("i = " + + i + + ", purpose = " + + tourPurposeIndex + + ", " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(startPeriod)) + + " to " + + modelStructure.getModelPeriodLabel(modelStructure + .getModelPeriodIndex(endPeriod)) + + " MCLS = " + + modeChoiceLogsums[index] + + ", ASC = " + + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i] + + ", (MCLS + ASC) = " + + (modeChoiceLogsums[index] + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]) + + ", exp(MCLS + ASC) = " + expUtil + ", cumExpUtility = " + + totalExpUtility); + + i++; + } + + double logsum = Math.log(totalExpUtility); + + if (household.getDebugChoiceModels()) + modelLogger.info("final simplified TOD logsum = " + logsum); + + return logsum; + } + + public void setNonMandatorySoaProbs(double[][][] soaDistProbs, double[][][] soaSizeProbs) + { + if (useNewSoaMethod) + { + dcSoaTwoStageObject.setTazDistProbs(soaDistProbs); + dcSoaTwoStageObject.setMgraSizeProbs(soaSizeProbs); + } + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + + public static void main(String[] args) + { + + // set values for these arguments so an object instance can be created + // and setup run to test integrity of UEC files before running full + // model. + HashMap propertyMap; + + if (args.length == 0) + { + System.out + .println("no properties file base name (without .properties extension) was specified as an argument."); + return; + } else + { + ResourceBundle rb = ResourceBundle.getBundle(args[0]); + propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); + } + + String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); + String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); + + MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, + Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(ms); + + MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); + TazDataManager tazManager = TazDataManager.getInstance(propertyMap); + + /* + * + */ + ModelStructure modelStructure = new SandagModelStructure(); + SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); + if (!aggAcc.getAccessibilitiesAreBuilt()) + { + aggAcc.setupBuildAccessibilities(propertyMap, false); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + // aggAcc.buildAccessibilityComponents(propertyMap); + + boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, + CtrampApplication.READ_ACCESSIBILITIES); + if (readAccessibilities) + { + + // output data + String projectDirectory = propertyMap + .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String accFileName = projectDirectory + + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + + } + + double[][] expConstants = aggAcc.getExpConstants(); + + McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + + double[][][] sovExpUtilities = null; + double[][][] hovExpUtilities = null; + double[][][] nMotorExpUtilities = null; + double[][][] maasExpUtilities = null; + NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, + hovExpUtilities, nMotorExpUtilities, maasExpUtilities); + + MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( + propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); + + String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); + int hhServerPort = Integer.parseInt((String) propertyMap + .get("RunModel.HouseholdServerPort")); + + HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, + hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + boolean restartHhServer = false; + try + { + // possible values for the following can be none, ao, cdap, imtf, + // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, + // stf, stl + String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); + if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; + else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") + || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") + || restartModel.equalsIgnoreCase("imtf") + || restartModel.equalsIgnoreCase("imtod") + || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") + || restartModel.equalsIgnoreCase("awtod") + || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") + || restartModel.equalsIgnoreCase("jtod") + || restartModel.equalsIgnoreCase("inmtf") + || restartModel.equalsIgnoreCase("inmtl") + || restartModel.equalsIgnoreCase("inmtod") + || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) + restartHhServer = false; + } catch (MissingResourceException e) + { + restartHhServer = true; + } + + if (restartHhServer) + { + + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = (String) propertyMap + .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(1.0f, 0); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, + inputPersonFileName); + + } else + { + + householdDataManager.setHouseholdSampleRate(1.0f, 0); + householdDataManager.setDebugHhIdsFromHashmap(); + householdDataManager.setTraceHouseholdSet(); + + } + + // int id = householdDataManager.getArrayIndex( 1033380 ); + // int id = householdDataManager.getArrayIndex( 1033331 ); + int id = householdDataManager.getArrayIndex(423804); + Household[] hh = householdDataManager.getHhArray(id, id); + + TourModeChoiceModel awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, + ModelStructure.AT_WORK_CATEGORY, dmuFactory, logsumHelper); + + SubtourDestChoiceModel testObject = new SubtourDestChoiceModel(propertyMap, modelStructure, + aggAcc, dmuFactory, awmcModel); + + testObject.applyModel(hh[0]); + + /** + * used this block of code to test for typos and implemented dmu methods + * in the TOD choice UECs + * + * String uecFileDirectory = propertyMap.get( + * CtrampApplication.PROPERTIES_UEC_PATH ); + * + * ModelStructure modelStructure = new SandagModelStructure(); + * SandagCtrampDmuFactory dmuFactory = new + * SandagCtrampDmuFactory(modelStructure); + * + * String dcUecFileName = propertyMap.get( PROPERTIES_DC_UEC_FILE ); + * DestChoiceDMU dcDmuObject = dmuFactory.getDestChoiceDMU(); File + * uecFile = new File(uecFileDirectory + dcUecFileName); + * UtilityExpressionCalculator uec = new + * UtilityExpressionCalculator(uecFile, 13, 0, propertyMap, + * (VariableTable) dcDmuObject); + * System.out.println("Subtour destination choice UEC passed"); + * + * String soaUecFileName = propertyMap.get( PROPERTIES_DC_SOA_UEC_FILE + * ); DcSoaDMU dcSoaDmuObject = dmuFactory.getDcSoaDMU(); uecFile = new + * File(uecFileDirectory + soaUecFileName); uec = new + * UtilityExpressionCalculator(uecFile, 7, 0, propertyMap, + * (VariableTable) dcSoaDmuObject); + * System.out.println("Subtour destination choice SOA UEC passed"); + */ + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java new file mode 100644 index 0000000..3731e65 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java @@ -0,0 +1,264 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; + +import umontreal.iro.lecuyer.probdist.LognormalDist; + + + +public class TNCAndTaxiWaitTimeCalculator implements Serializable{ + + private static Logger logger = Logger.getLogger(TNCAndTaxiWaitTimeCalculator.class); + + private float[] startPopEmpPerSqMi; + + private LognormalDist[] TNCSingleWaitTimeDistribution; + private LognormalDist[] TNCSharedWaitTimeDistribution; + private LognormalDist[] TaxiWaitTimeDistribution; + float[] meanTNCSingleWaitTime ; + float[] meanTNCSharedWaitTime ; + float[] meanTaxiWaitTime ; + + /** + * Constructor; doesn't do anything (call @createTimeDistributions method next) + */ + public TNCAndTaxiWaitTimeCalculator(){ + + } + + /** + * Reads the propertyMap and finds values for property arrays + * TNC.waitTime.mean, Taxi.waitTime.mean, + * TNC.waitTime.sd, and Taxi.waitTime.sd - containing arrays + * of wait time and standard deviations for TNCs and Taxis by area type, + * plus an array of end ranges for area type (pop+emp)/sq miles + * with an implied start value of 0. + * Creates and stores umontreal.iro.lecuyer.probdist.LognormalDist + * TNCWaitTimeDistribution[] and TaxiWaitTimeDistribution[] where + * each element of the distribution corresponds to the areatype range. + * @param propertyMap + */ + + public void createWaitTimeDistributions(HashMap propertyMap){ + + //read properties + meanTNCSingleWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.single.waitTime.mean"); + float[] sdTNCSingleWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.single.waitTime.sd"); + + meanTNCSharedWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.shared.waitTime.mean"); + float[] sdTNCSharedWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.shared.waitTime.sd"); + + meanTaxiWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "Taxi.waitTime.mean"); + float[] sdTaxiWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "Taxi.waitTime.sd"); + + startPopEmpPerSqMi = Util.getFloatArrayFromPropertyMap(propertyMap, "WaitTimeDistribution.EndPopEmpPerSqMi"); + + // create the distribution arrays + TNCSingleWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; + TNCSharedWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; + TaxiWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; + + //iterate through area types + for(int i = 0; i< startPopEmpPerSqMi.length;++i){ + + // calculate the location and scale parameters from the mean and standard deviations + double locationTNCSingleWaitTime = calculateLocation(meanTNCSingleWaitTime[i], sdTNCSingleWaitTime[i]); + double scaleTNCSingleWaitTime = calculateScale(meanTNCSingleWaitTime[i], sdTNCSingleWaitTime[i]); + + + double locationTNCSharedWaitTime = calculateLocation(meanTNCSharedWaitTime[i], sdTNCSharedWaitTime[i]); + double scaleTNCSharedWaitTime = calculateScale(meanTNCSharedWaitTime[i], sdTNCSharedWaitTime[i]); + + // create the TNC wait time distribution for this area type + TNCSingleWaitTimeDistribution[i] = new LognormalDist(locationTNCSingleWaitTime, scaleTNCSingleWaitTime); + TNCSharedWaitTimeDistribution[i] = new LognormalDist(locationTNCSharedWaitTime, scaleTNCSharedWaitTime); + + double locationTaxiWaitTime = calculateLocation(meanTaxiWaitTime[i], sdTaxiWaitTime[i]); + double scaleTaxiWaitTime = calculateScale(meanTaxiWaitTime[i], sdTaxiWaitTime[i]); + + TaxiWaitTimeDistribution[i] = new LognormalDist(locationTaxiWaitTime, scaleTaxiWaitTime); + + } + } + + /** + * Calculate the lognormal distribution location given + * the mean and standard deviation of the distribution + * according to the formula: + * + * location = ln(mean/sqrt(1 + variance/mean^2)) + * + * @param mean + * @param standardDeviation + * @return Location variable (u) + */ + public double calculateLocation(double mean, double standardDeviation){ + + double variance = standardDeviation * standardDeviation; + double meanSquared = mean * mean; + double denom = Math.sqrt(1.0 + (variance/meanSquared)); + double location = mean/denom; + if(location<=0){ + logger.error("Error: Trying to calculation location for mean "+mean + +" and standard deviation "+standardDeviation); + throw new RuntimeException(); + } + + return Math.log(location); + + } + + /** + * Calculate the lognormal distribution scale given + * the mean and standard deviation of the distribution + * according to the formula: + * + * scale = sqrt(ln(1 + variance/mean^2)); + * + * @param mean + * @param standardDeviation + * @return Scale variable (sigma) + */ + public double calculateScale(double mean, double standardDeviation){ + + double variance = standardDeviation * standardDeviation; + double meanSquared = mean * mean; + return Math.sqrt(Math.log(1 + variance/meanSquared)); + } + + /** + * Sample from the Single TNC wait time distribution and return the wait time. + * @param rnum A unit-distributed random number. + * @param popEmpPerSqMi The population plus employment divided by square miles + * @return The sampled TNC wait time. + */ + public double sampleFromSingleTNCWaitTimeDistribution(double rnum, double popEmpPerSqMi){ + + for(int i = 0; i < startPopEmpPerSqMi.length;++i){ + + if(popEmpPerSqMi zoneValues = new HashMap(); + + for (int i = 1; i <= zoneTable.getRowCount(); i++) + { + int zone = (int) zoneTable.getValueAt(i, zoneCol); + if (zoneValues.containsKey(zone)) + { + logger.fatal(String + .format("zone employment table read from %s has duplicate value for ZONE=%d in column %d at record number %d", + zoneDataFileName, zone, zoneCol, (i + 1))); + throw new RuntimeException(); + } else + { + zoneValues.put(zone, i); + numZones++; + if (zone > maxZone) maxZone = zone; + } + + } + + } catch (IOException e) + { + logger.error(String + .format("Exception occurred reading zonal employment data file: %s into TableDataSet object.", + zoneDataFileName)); + throw new RuntimeException(); + } + + NUM_ZONES = numZones; + + // store the row numbers for each zone so that zonal attributes can be + // retrieved later using given a zone number + indexToZone = new int[numZones + 1]; + zoneTableRow = new int[maxZone + 1]; + for (int i = 1; i <= zoneTable.getRowCount(); i++) + { + int zone = (int) zoneTable.getValueAt(i, zoneCol); + zoneTableRow[zone] = i; + indexToZone[i] = zone; + } + + return zoneTable; + + } + + private void readWalkPercentagesFile(String fileName) + { + + int taz; + float[] shrtArray = new float[NUM_ZONES + 1]; + float[] longArray = new float[NUM_ZONES + 1]; + zonalWalkPctArray = new float[3][NUM_ZONES + 1]; + Arrays.fill(zonalWalkPctArray[0], 1.0f); + Arrays.fill(zonalWalkPctArray[1], 0.0f); + Arrays.fill(zonalWalkPctArray[2], 0.0f); + + if (fileName != null) + { + + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet(" " + reader.getDelimSet()); + TableDataSet wa = reader.readFile(new File(fileName)); + + int tazPosition = wa.getColumnPosition(walkPctZoneFieldName); + if (tazPosition <= 0) + { + logger.fatal(String + .format("expected zone field name=%s was not a field in the walk access file: %s.", + WALK_PERCENTAGE_FILE_ZONE_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int shrtPosition = wa.getColumnPosition(walkPctShortFieldName); + if (shrtPosition <= 0) + { + logger.fatal(String + .format("expected short field name=%s was not a field in the walk access file: %s.", + WALK_PERCENTAGE_FILE_SHORT_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int longPosition = wa.getColumnPosition(walkPctLongFieldName); + if (longPosition <= 0) + { + logger.fatal(String + .format("expected long field name=%s was not a field in the walk access file: %s.", + WALK_PERCENTAGE_FILE_LONG_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + for (int j = 1; j <= wa.getRowCount(); j++) + { + taz = (int) wa.getValueAt(j, tazPosition); + shrtArray[taz] = wa.getValueAt(j, shrtPosition); + longArray[taz] = wa.getValueAt(j, longPosition); + zonalWalkPctArray[1][taz] = shrtArray[taz]; + zonalWalkPctArray[2][taz] = longArray[taz]; + zonalWalkPctArray[0][taz] = (float) (1.0 - (shrtArray[taz] + longArray[taz])); + } + + } catch (IOException e) + { + logger.fatal( + String.format("exception caught reading walk access file: %s", fileName), e); + } + + } else + { + + logger.fatal("no zonal walk access data file was named in properties file with target: 'WalkPercentages.file ='."); + throw new RuntimeException(); + + } + + } + + private void readZonalAccessibilitiesFile(String fileName) + { + + int taz; + pkAutoRetail = new float[NUM_ZONES + 1]; + pkAutoTotal = new float[NUM_ZONES + 1]; + opAutoRetail = new float[NUM_ZONES + 1]; + opAutoTotal = new float[NUM_ZONES + 1]; + pkTransitRetail = new float[NUM_ZONES + 1]; + pkTransitTotal = new float[NUM_ZONES + 1]; + opTransitRetail = new float[NUM_ZONES + 1]; + opTransitTotal = new float[NUM_ZONES + 1]; + nonMotorizedRetail = new float[NUM_ZONES + 1]; + nonMotorizedTotal = new float[NUM_ZONES + 1]; + + if (fileName != null) + { + + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + reader.setDelimSet(" " + reader.getDelimSet()); + TableDataSet acc = reader.readFile(new File(fileName)); + + int tazPosition = acc.getColumnPosition(ACCESSIBILITIES_FILE_ZONE_FIELD_NAME); + if (tazPosition <= 0) + { + logger.fatal(String + .format("expected zone field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_FILE_ZONE_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int pkAutoRetailPosition = acc + .getColumnPosition(ACCESSIBILITIES_PEAK_AUTO_RETAIL_FIELD_NAME); + if (pkAutoRetailPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_PEAK_AUTO_RETAIL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int pkAutoTotalPosition = acc + .getColumnPosition(ACCESSIBILITIES_PEAK_AUTO_TOTAL_FIELD_NAME); + if (pkAutoTotalPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_PEAK_AUTO_TOTAL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int pkTransitRetailPosition = acc + .getColumnPosition(ACCESSIBILITIES_PEAK_TRANSIT_RETAIL_FIELD_NAME); + if (pkTransitRetailPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_PEAK_TRANSIT_RETAIL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int pkTransitTotalPosition = acc + .getColumnPosition(ACCESSIBILITIES_PEAK_TRANSIT_TOTAL_FIELD_NAME); + if (pkTransitTotalPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_PEAK_TRANSIT_TOTAL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int opAutoRetailPosition = acc + .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_AUTO_RETAIL_FIELD_NAME); + if (opAutoRetailPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_OFF_PEAK_AUTO_RETAIL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int opAutoTotalPosition = acc + .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_AUTO_TOTAL_FIELD_NAME); + if (opAutoTotalPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_OFF_PEAK_AUTO_TOTAL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int opTransitRetailPosition = acc + .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_TRANSIT_RETAIL_FIELD_NAME); + if (opTransitRetailPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_OFF_PEAK_TRANSIT_RETAIL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int opTransitTotalPosition = acc + .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_TRANSIT_TOTAL_FIELD_NAME); + if (opTransitTotalPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_OFF_PEAK_TRANSIT_TOTAL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int nonMotorizedRetailPosition = acc + .getColumnPosition(ACCESSIBILITIES_NON_MOTORIZED_RETAIL_FIELD_NAME); + if (nonMotorizedRetailPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_NON_MOTORIZED_RETAIL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + int nonMotorizedTotalPosition = acc + .getColumnPosition(ACCESSIBILITIES_NON_MOTORIZED_TOTAL_FIELD_NAME); + if (nonMotorizedTotalPosition <= 0) + { + logger.fatal(String + .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", + ACCESSIBILITIES_NON_MOTORIZED_TOTAL_FIELD_NAME, fileName)); + throw new RuntimeException(); + } + + for (int i = 1; i <= acc.getRowCount(); i++) + { + taz = (int) acc.getValueAt(i, tazPosition); + pkAutoRetail[taz] = acc.getValueAt(i, pkAutoRetailPosition); + pkAutoTotal[taz] = acc.getValueAt(i, pkAutoTotalPosition); + pkTransitRetail[taz] = acc.getValueAt(i, pkTransitRetailPosition); + pkTransitTotal[taz] = acc.getValueAt(i, pkTransitTotalPosition); + opAutoRetail[taz] = acc.getValueAt(i, opAutoRetailPosition); + opAutoTotal[taz] = acc.getValueAt(i, opAutoTotalPosition); + opTransitRetail[taz] = acc.getValueAt(i, opTransitRetailPosition); + opTransitTotal[taz] = acc.getValueAt(i, opTransitTotalPosition); + nonMotorizedRetail[taz] = acc.getValueAt(i, nonMotorizedRetailPosition); + nonMotorizedTotal[taz] = acc.getValueAt(i, nonMotorizedTotalPosition); + } + + } catch (IOException e) + { + logger.fatal(String.format("exception caught reading accessibilities file: %s", + fileName), e); + } + + } else + { + + logger.fatal("no zonal accessibilities data file was named in properties file with target: " + + fileName); + throw new RuntimeException(); + + } + + } + + /** + * @param alt + * is the DC alternaive number + * @return zone number for the DC alt. + */ + private int getZoneFromAlt(int alt) + { + int zone = (int) ((alt - 1) / NUM_SUBZONES) + 1; + if (zone < 1 || zone > NUM_ZONES) + { + logger.fatal(String.format( + "invalid value for zone index = %d, determined for alt = %d.", zone, alt)); + logger.fatal(String.format("NUM_ZONES = %d, NUM_SUBZONES = %d.", NUM_ZONES, + NUM_SUBZONES)); + throw new RuntimeException(); + } + return zone; + } + + /** + * @param alt + * is the DC alternaive number + * @return walk subzone index for the DC alt. + */ + private int getWalkSubzoneFromAlt(int alt) + { + int zone = getZoneFromAlt(alt); + int subzone = alt - (zone - 1) * NUM_SUBZONES - 1; + if (subzone < 0 || subzone >= NUM_SUBZONES) + { + logger.fatal(String + .format("invalid value for walk subzone index = %d, zone = %d, determined for alt = %d.", + subzone, zone, alt)); + logger.fatal(String.format("NUM_ZONES = %d, NUM_SUBZONES = %d.", NUM_ZONES, + NUM_SUBZONES)); + throw new RuntimeException(); + } + return subzone; + } + + public String testRemote() + { + return String.format("testRemote() method in %s called.", this.getClass() + .getCanonicalName()); + } + + public int[] getAltToZoneArray() + { + return altToZone; + } + + public int[] getAltToSubZoneArray() + { + return altToSubZone; + } + + public int[] getIndexToZoneArray() + { + return indexToZone; + } + + public int[] getZoneTableRowArray() + { + return zoneTableRow; + } + + /** + * + * @param field + * is the field name to be checked against the column names in + * the zone data table. + * @return true if field matches one of the zone data table column names, + * otherwise false. + */ + public boolean isValidZoneTableField(String field) + { + return zoneDataTable.getColumnPosition(field) >= 0; + } + + public String[] getZoneDataTableColumnLabels() + { + return zoneDataTable.getColumnLabels(); + } + + public int getNumberOfZones() + { + return NUM_ZONES; + } + + public int getNumberOfSubZones() + { + return NUM_SUBZONES; + } + + public String[] getSubZoneNames() + { + return subZoneNames; + } + + public double[] getZonalWalkPercentagesForTaz(int taz) + { + double[] percentages = new double[NUM_SUBZONES]; + for (int i = 0; i < NUM_SUBZONES; i++) + percentages[i] = zonalWalkPctArray[i][taz]; + return percentages; + } + + public float getZoneTableValue(int taz, String fieldName) + { + // get the table row number for the TAZ passed in + int rowIndex = zoneTableRow[taz]; + + // get the table value from the rowIndex and fieldname passed in + return zoneDataTable.getValueAt(rowIndex, fieldName); + } + + // get the table column from the fieldname passed in + public int[] getZoneTableIntColumn(String fieldName) + { + return zoneDataTable.getColumnAsInt(fieldName); + } + + // get the table column from the fieldname passed in + public float[] getZoneTableFloatColumn(String fieldName) + { + return zoneDataTable.getColumnAsFloat(fieldName); + } + + /** + * @param tableRowNumber + * is the zone table row number + * @return zone number for the table row. + */ + public int getTazNumber(int tableRowNumber) + { + return (int) zoneDataTable.getValueAt(tableRowNumber, tazDataZoneFieldName); + } + + /** + * @return area type array from the zone data table. + */ + public int[] getZonalAreaType() + { + int atFieldPosition = zoneDataTable.getColumnPosition(tazDataAtFieldName); + if (atFieldPosition < 0) + { + logger.error(String + .format("The area type field name = %s defined in %s is not found as a field name in the zone data table.", + tazDataAtFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsInt(atFieldPosition); + } + + /** + * @return district array from the zone data table. + */ + public int[] getZonalDistrict() + { + int districtFieldPosition = zoneDataTable.getColumnPosition(tazDataDistFieldName); + if (districtFieldPosition < 0) + { + logger.error(String + .format("The district field name = %s defined in %s is not found as a field name in the zone data table.", + tazDataDistFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsInt(districtFieldPosition); + } + + /** + * @return county array from the zone data table. + */ + public int[] getZonalCounty() + { + int countyFieldPosition = zoneDataTable.getColumnPosition(tazDataCountyFieldName); + if (countyFieldPosition < 0) + { + logger.error(String + .format("The county field name = %s defined in %s is not found as a field name in the zone data table.", + tazDataCountyFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsInt(countyFieldPosition); + } + + public int getZoneIsCbd(int taz) + { + return getZoneIsInAreaType(taz, areaTypes[cbdAreaTypesArrayIndex]); + } + + public int getZoneIsUrban(int taz) + { + return getZoneIsInAreaType(taz, areaTypes[urbanAreaTypesArrayIndex]); + } + + public int getZoneIsSuburban(int taz) + { + return getZoneIsInAreaType(taz, areaTypes[suburbanAreaTypesArrayIndex]); + } + + public int getZoneIsRural(int taz) + { + return getZoneIsInAreaType(taz, areaTypes[ruralAreaTypesArrayIndex]); + } + + private int getZoneIsInAreaType(int taz, int[] areaTypes) + { + int returnValue = 0; + int tazAreaType = (int) getZoneTableValue(taz, tazDataAtFieldName); + for (int atIndex : areaTypes) + { + if (tazAreaType == atIndex) + { + returnValue = 1; + break; + } + } + return returnValue; + } + + /** + * @return parkTot array from the zone data table. + */ + public int[] getZonalParkTot() + { + int parkTotFieldPosition = zoneDataTable.getColumnPosition(parkTotFieldName); + if (parkTotFieldPosition < 0) + { + logger.error(String + .format("The parkTot field name = %s defined in %s is not found as a field name in the zone data table.", + parkTotFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsInt(parkTotFieldPosition); + } + + /** + * @return parkLong array from the zone data table. + */ + public int[] getZonalParkLong() + { + int parkLongFieldPosition = zoneDataTable.getColumnPosition(parkLongFieldName); + if (parkLongFieldPosition < 0) + { + logger.error(String + .format("The parkLong field name = %s defined in %s is not found as a field name in the zone data table.", + parkLongFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsInt(parkLongFieldPosition); + } + + /** + * @return propFree array from the zone data table. + */ + public float[] getZonalPropFree() + { + int propFreeFieldPosition = zoneDataTable.getColumnPosition(propFreeFieldName); + if (propFreeFieldPosition < 0) + { + logger.error(String + .format("The propFree field name = %s defined in %s is not found as a field name in the zone data table.", + propFreeFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsFloat(propFreeFieldPosition); + } + + /** + * @return parkRate array from the zone data table. + */ + public float[] getZonalParkRate() + { + int parkRateFieldPosition = zoneDataTable.getColumnPosition(parkRateFieldName); + if (parkRateFieldPosition < 0) + { + logger.error(String + .format("The parkRate field name = %s defined in %s is not found as a field name in the zone data table.", + parkRateFieldName, this.getClass().getName())); + throw new RuntimeException(); + } + return zoneDataTable.getColumnAsFloat(parkRateFieldPosition); + } + + public float[] getPkAutoRetailAccessibity() + { + return pkAutoRetail; + } + + public float[] getPkAutoTotalAccessibity() + { + return pkAutoTotal; + } + + public float[] getPkTransitRetailAccessibity() + { + return pkTransitRetail; + } + + public float[] getPkTransitTotalAccessibity() + { + return pkTransitTotal; + } + + public float[] getOpAutoRetailAccessibity() + { + return opAutoRetail; + } + + public float[] getOpAutoTotalAccessibity() + { + return opAutoTotal; + } + + public float[] getOpTransitRetailAccessibity() + { + return opTransitRetail; + } + + public float[] getOpTransitTotalAccessibity() + { + return opTransitTotal; + } + + public float[] getNonMotorizedRetailAccessibity() + { + return nonMotorizedRetail; + } + + public float[] getNonMotorizedTotalAccessibity() + { + return nonMotorizedTotal; + } + + public enum AreaType + { + CBD, URBAN, SUBURBAN, RURAL + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java new file mode 100644 index 0000000..0e051c5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java @@ -0,0 +1,297 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; + +/** + * This class provides methods defined in the TazDataIf interface for accessing + * zonal data stored in its TazDataManager object. + * + * A CT-RAMP tour based model application could create an instance of a subclass + * of this class, where additional project specific varaible definitions and + * methods are defined and pass that instance to its model component objects. + * + * Alternatively, an application could use TazDataHandlerRmi as the base class + * instead and create a "remoteable" subclass. The TazDataHandlerRmi class + * implements the same interface, so the model component classes can be unaware + * of whether the taz data handler object accesses zonal data from its member + * object or remotely from a server. Those methods in the rmi class access zonal + * data from a TazDataManager object contained in a "taz data server" object + * which must exist in a separate JVM on the same machine or on another + * addressable machine over the network. + * + * The flexibility provided by this design is intended to allow the "local" + * instance to be declared and passed within a single JVM to model components + * for possibly greater performance (yet to be tested and proven) at production + * run time. The "rmi" instance however allows the model components to access + * zonal data from a "long-running process" (the server class may execute for + * weeks or months). This approach aids in model development as during + * development, model applications can be written to skip startup procedures for + * reading zonal data, and access them directly from the server that is already + * running. + * + * A similar approach is planned for managing objects such as Household objects + * and ModelResults objects so that model components, for example individual + * non-mandatory tour related models which occur well into the tour based model + * stream, can be run in a "hot-start" fasion, where the model component of + * interest is executed immediately where all the preliminary data and prior + * model results it requires are stored in long-running server objects. Testing + * and debugging of these model components can occur without the time required + * to run through all preliminary steps. + * + * + */ + +public class TazDataHandlerRmi + implements TazDataIf, Serializable +{ + + UtilRmi remote; + String connectString; + + public TazDataHandlerRmi(String hostname, int port, String className) + { + + connectString = String.format("//%s:%d/%s", hostname, port, className); + remote = new UtilRmi(connectString); + + } + + public String testRemote() + { + Object[] objArray = {}; + return (String) remote.method("testRemote", objArray); + } + + public int[] getAltToZoneArray() + { + Object[] objArray = {}; + return (int[]) remote.method("getAltToZoneArray", objArray); + } + + public int[] getAltToSubZoneArray() + { + Object[] objArray = {}; + return (int[]) remote.method("getAltToSubZoneArray", objArray); + } + + public int[] getIndexToZoneArray() + { + Object[] objArray = {}; + return (int[]) remote.method("getIndexToZoneArray", objArray); + } + + public int[] getZoneTableRowArray() + { + Object[] objArray = {}; + return (int[]) remote.method("getZoneTableRowArray", objArray); + } + + /** + * @param field + * is the field name to be checked against the column names in + * the zone data table. + * @return true if field matches one of the zone data table column names, + * otherwise false. + */ + public boolean isValidZoneTableField(String field) + { + Object[] objArray = {field}; + return (Boolean) remote.method("isValidZoneTableField", objArray); + } + + public String[] getZoneDataTableColumnLabels() + { + Object[] objArray = {}; + return (String[]) remote.method("getZoneDataTableColumnLabels", objArray); + } + + public int getNumberOfZones() + { + Object[] objArray = {}; + return (Integer) remote.method("getNumberOfZones", objArray); + } + + public int getNumberOfSubZones() + { + Object[] objArray = {}; + return (Integer) remote.method("getNumberOfSubZones", objArray); + } + + public String[] getSubZoneNames() + { + Object[] objArray = {}; + return (String[]) remote.method("getSubZoneNames", objArray); + } + + public double[] getZonalWalkPercentagesForTaz(int taz) + { + Object[] objArray = {taz}; + return (double[]) remote.method("getZonalWalkPercentagesForTaz", objArray); + } + + public float getZoneTableValue(int taz, String fieldName) + { + Object[] objArray = {taz, fieldName}; + return (Float) remote.method("getZoneTableValue", objArray); + } + + public int[] getZoneTableIntColumn(String fieldName) + { + Object[] objArray = {fieldName}; + return (int[]) remote.method("getZoneTableIntColumn", objArray); + } + + // get the table column from the fieldname passed in + public float[] getZoneTableFloatColumn(String fieldName) + { + Object[] objArray = {fieldName}; + return (float[]) remote.method("getZoneTableFloatColumn", objArray); + } + + /** + * @param tableRowNumber + * is the zone table row number + * @return zone number for the table row. + */ + public int getTazNumber(int tableRowNumber) + { + Object[] objArray = {tableRowNumber}; + return (Integer) remote.method("getTazNumber", objArray); + } + + /** + * @return area type from the zone data table for the zone. + */ + public int[] getZonalAreaType() + { + Object[] objArray = {}; + return (int[]) remote.method("getZonalAreaType", objArray); + } + + /** + * @return district from the zone data table for the zone. + */ + public int[] getZonalDistrict() + { + Object[] objArray = {}; + return (int[]) remote.method("getZonalDistrict", objArray); + } + + public int[] getZonalParkTot() + { + Object[] objArray = {}; + return (int[]) remote.method("getZonalParkTot", objArray); + } + + public int[] getZonalParkLong() + { + Object[] objArray = {}; + return (int[]) remote.method("getZonalParkLong", objArray); + } + + public float[] getZonalPropFree() + { + Object[] objArray = {}; + return (float[]) remote.method("getZonalPropFree", objArray); + } + + public float[] getZonalParkRate() + { + Object[] objArray = {}; + return (float[]) remote.method("getZonalParkRate", objArray); + } + + /** + * @return integer county value from the zone data table for the zone. + */ + public int[] getZonalCounty() + { + Object[] objArray = {}; + return (int[]) remote.method("getZonalCounty", objArray); + } + + public int getZoneIsCbd(int taz) + { + Object[] objArray = {taz}; + return (Integer) remote.method("getZoneIsCbd", objArray); + } + + public int getZoneIsUrban(int taz) + { + Object[] objArray = {taz}; + return (Integer) remote.method("getZoneIsUrban", objArray); + } + + public int getZoneIsSuburban(int taz) + { + Object[] objArray = {taz}; + return (Integer) remote.method("getZoneIsSuburban", objArray); + } + + public int getZoneIsRural(int taz) + { + Object[] objArray = {taz}; + return (Integer) remote.method("getZoneIsRural", objArray); + } + + public float[] getPkAutoRetailAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getPkAutoRetailAccessibity", objArray); + } + + public float[] getPkAutoTotalAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getPkAutoTotalAccessibity", objArray); + } + + public float[] getPkTransitRetailAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getPkTransitRetailAccessibity", objArray); + } + + public float[] getPkTransitTotalAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getPkTransitTotalAccessibity", objArray); + } + + public float[] getOpAutoRetailAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getOpAutoRetailAccessibity", objArray); + } + + public float[] getOpAutoTotalAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getOpAutoTotalAccessibity", objArray); + } + + public float[] getOpTransitRetailAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getOpTransitRetailAccessibity", objArray); + } + + public float[] getOpTransitTotalAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getOpTransitTotalAccessibity", objArray); + } + + public float[] getNonMotorizedRetailAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getNonMotorizedRetailAccessibity", objArray); + } + + public float[] getNonMotorizedTotalAccessibity() + { + Object[] objArray = {}; + return (float[]) remote.method("getNonMotorizedTotalAccessibity", objArray); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java new file mode 100644 index 0000000..f43c846 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java @@ -0,0 +1,152 @@ +package org.sandag.abm.ctramp; + +/** + * Created by IntelliJ IDEA. User: Jim Date: Jul 1, 2008 Time: 9:58:21 AM + * + * Interface for accessing zonal information used by CT-RAMP modules + */ +public interface TazDataIf +{ + + String testRemote(); + + int[] getAltToZoneArray(); + + int[] getAltToSubZoneArray(); + + int[] getIndexToZoneArray(); + + int[] getZoneTableRowArray(); + + int getZoneIsCbd(int taz); + + int getZoneIsUrban(int taz); + + int getZoneIsSuburban(int taz); + + int getZoneIsRural(int taz); + + float[] getPkAutoRetailAccessibity(); + + float[] getPkAutoTotalAccessibity(); + + float[] getPkTransitRetailAccessibity(); + + float[] getPkTransitTotalAccessibity(); + + float[] getOpAutoRetailAccessibity(); + + float[] getOpAutoTotalAccessibity(); + + float[] getOpTransitRetailAccessibity(); + + float[] getOpTransitTotalAccessibity(); + + float[] getNonMotorizedRetailAccessibity(); + + float[] getNonMotorizedTotalAccessibity(); + + /** + * + * @param field + * is the field name to be checked against the column names in + * the zone data table. + * @return true if field matches one of the zone data table column names, + * otherwise false. + */ + boolean isValidZoneTableField(String field); + + /** + * @return a String[] of the column labels in the zone data table + */ + String[] getZoneDataTableColumnLabels(); + + /** + * @return an int value for the number of zones, i.e. rows in the zone data + * table + */ + int getNumberOfZones(); + + /** + * @return an int value for the number of subZones, i.e. number of + * walkTransit accessible segments defined in model for zones. + * Typical value might be 3, "no walk access", "short walk access", + * "long walk access". + */ + int getNumberOfSubZones(); + + /** + * @return a String[] for the subZone names, e.g. "no walk access", + * "short walk access", "long walk access". + */ + String[] getSubZoneNames(); + + /** + * @param taz + * is the taz index for the zonalWalkPctArray which is + * dimensioned to ZONES+1, assuming taz index values range from 1 + * to NUM_ZONES. + * @return a double[], dimensioned to NUM_SIBZONES, with the subzone + * proportions for the TAZ passed in + */ + double[] getZonalWalkPercentagesForTaz(int taz); + + /** + * @param taz + * is the taz index for the zone data table which is dimensioned + * to ZONES+1, assuming taz index values range from 1 to + * NUM_ZONES. + * @param fieldName + * is the column label in the zone data table. + * @return a float value from the zone data table at the specified row index + * and column label. + */ + float getZoneTableValue(int taz, String fieldName); + + int[] getZoneTableIntColumn(String fieldName); + + float[] getZoneTableFloatColumn(String fieldName); + + /** + * @param tableRowNumber + * is the zone table row number + * @return zone number for the table row. + */ + int getTazNumber(int tableRowNumber); + + /** + * @return area type from the zone data table for the zone index. + */ + int[] getZonalAreaType(); + + /** + * @return district from the zone data table for the zone index. + */ + int[] getZonalDistrict(); + + /** + * @return integer county value from the zone data table for the zone index. + */ + int[] getZonalCounty(); + + /** + * @return the parking rate array + */ + float[] getZonalParkRate(); + + /** + * @return the proportion of free parking array + */ + float[] getZonalPropFree(); + + /** + * @return the number of long parking spots array + */ + int[] getZonalParkLong(); + + /** + * @return the number of parking spots array + */ + int[] getZonalParkTot(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java new file mode 100644 index 0000000..5a1aed8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java @@ -0,0 +1,172 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + * @author jef
+ * Started: Jun 2019 + */ +public class TelecommuteDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TelecommuteDMU.class); + + protected HashMap methodIndexMap; + + private Household hh; + private Person person; + private IndexValues dmuIndex; + + public TelecommuteDMU() + { + dmuIndex = new IndexValues(); + } + + /** need to set hh and home taz before using **/ + public void setPersonObject(Person person) + { + this.hh = person.getHouseholdObject(); + this.person = person; + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug Telecommute UEC"); + } + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /* dmu @ functions */ + + public int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + public int getNumberOfAdults() + { + Person[] persons = hh.getPersons(); + int adults=0; + for(int i=1;i=18) + ++adults; + } + return adults; + } + + public int getHasKids_0_5() + { + Person[] persons = hh.getPersons(); + int hasKids_0_5=0; + for(int i=1;i=0) && persons[i].getAge()<=5) { + hasKids_0_5=1; + break; + } + } + return hasKids_0_5; + } + + + public int getHasKids_6_12() + { + Person[] persons = hh.getPersons(); + int hasKids_6_12=0; + for(int i=1;i=6) && persons[i].getAge()<=12) { + hasKids_6_12=1; + break; + } + } + return hasKids_6_12; + } + + public int getFemale() + { + return person.getPersonIsFemale(); + } + + public int getPersonType() + { + return person.getPersonTypeNumber(); + } + + public int getNumberOfAutos() + { + return hh.getAutosOwned(); + + } + + public int getOccupation() + { + return person.getPersPecasOccup(); + } + + public int getPaysToPark() + { + + int freeParkingChoice = person.getFreeParkingAvailableResult(); + if((freeParkingChoice==ParkingProvisionModel.FP_MODEL_PAY_ALT)|| + (freeParkingChoice==ParkingProvisionModel.FP_MODEL_REIMB_ALT)) + + return 1; + return 0; + } + + public float getWorkDistance() { + + return person.getWorkLocationDistance(); + } + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java new file mode 100644 index 0000000..5f47355 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java @@ -0,0 +1,144 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class TelecommuteModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("tc"); + + private static final String TC_CONTROL_FILE_TARGET = "te.uec.file"; + private static final String TC_DATA_SHEET_TARGET = "te.data.page"; + private static final String TC_MODEL_SHEET_TARGET = "te.model.page"; + + public static final short TC_MODEL_NO_TC_CHOICE = -1; + public static final short TC_MODEL_NO_TELECOMMUTE = 0; + public static final short TC_MODEL_1_DAY_WEEK_CHOICE = 1; + public static final short TC_MODEL_2_3_DAYS_WEEK_CHOICE = 2; + public static final short TC_MODEL_4P_DAYS_WEEK_CHOICE = 3; + public static final short WORK_AT_HOME_CHOICE = 9; + + + private MgraDataManager mgraManager; + + private ChoiceModelApplication tcModel; + private TelecommuteDMU tcDmuObject; + + public TelecommuteModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) + { + mgraManager = MgraDataManager.getInstance(propertyMap); + setupTelecommuteChoiceModelApplication(propertyMap, dmuFactory); + } + + private void setupTelecommuteChoiceModelApplication(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + logger.info("Setting up telecommute choice model."); + + // locate the telecommute UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tcUecFile = uecFileDirectory + propertyMap.get(TC_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TC_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TC_MODEL_SHEET_TARGET); + + // create the telecommute model DMU object. + tcDmuObject = dmuFactory.getTelecommuteDMU(); + + // create the telecommute model object + tcModel = new ChoiceModelApplication(tcUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) tcDmuObject); + + } + + public void applyModel(Household hhObject) + { + + Random hhRandom = hhObject.getHhRandom(); + + // person array is 1-based + Person[] person = hhObject.getPersons(); + for (int i = 1; i < person.length; i++) + { + + int workLoc = person[i].getWorkLocation(); + if (workLoc == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) + { + + person[i].setTelecommuteChoice(WORK_AT_HOME_CHOICE); + + } else if (workLoc > 0 ) + { + + double randomNumber = hhRandom.nextDouble(); + short chosen = (short) ((short) getTelecommuteChoice(person[i], randomNumber) - (short)1); + person[i].setTelecommuteChoice(chosen ); + + + } else + { + + person[i].setTelecommuteChoice(TC_MODEL_NO_TC_CHOICE); + + } + } + + hhObject.setFpRandomCount(hhObject.getHhRandomCount()); + } + + private int getTelecommuteChoice(Person personObj, double randomNumber) + { + + // get the corresponding household object + Household hhObj = personObj.getHouseholdObject(); + tcDmuObject.setPersonObject(personObj); + + // set the zone and dest attributes to the person's work location + tcDmuObject.setDmuIndexValues(hhObj.getHhId(), personObj.getWorkLocation(), + hhObj.getHhTaz(), personObj.getWorkLocation()); + + // compute utilities and choose telecommute alternative. + float logsum = (float) tcModel.computeUtilities(tcDmuObject, tcDmuObject.getDmuIndexValues()); + personObj.setTelecommuteLogsum(logsum); + + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + if (tcModel.getAvailabilityCount() > 0) + { + chosenAlt = tcModel.getChoiceResult(randomNumber); + } else + { + String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), + personObj.getPersonId()); + String errorMessage = String + .format("Exception caught for %s, no available telecommute alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + tcModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + // write choice model alternative info to log file + if (hhObj.getDebugChoiceModels()) + { + String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), + personObj.getPersonId()); + tcModel.logAlternativesInfo("Telecommute Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d with rn %.8f", + "Telecommute Choice", decisionMaker, chosenAlt, randomNumber)); + tcModel.logUECResults(logger, decisionMaker); + } + + return chosenAlt; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java new file mode 100644 index 0000000..b2fb242 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java @@ -0,0 +1,284 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedWriter; +import java.nio.charset.Charset; +import java.nio.charset.StandardCharsets; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ResourceBundle; +import java.util.Scanner; +import java.util.Set; + +import org.apache.log4j.Logger; + +import com.pb.common.math.MersenneTwister; +import com.pb.common.util.ResourceUtil; + +import umontreal.iro.lecuyer.probdist.LognormalDist; + +public class TimeCoefficientDistributions { + + private static Logger logger = Logger.getLogger(TimeCoefficientDistributions.class); + protected LognormalDist timeDistributionWork; + protected LognormalDist timeDistributionNonWork; + + + /** + * Constructor; doesn't do anything (call @createTimeDistributions method next) + */ + public TimeCoefficientDistributions(){ + + } + + /** + * Reads the propertyMap and finds values for properties + * timeDistributionMean.work, and + * timeDistributionStandardDistribution.work. + * Creates and stores umontreal.iro.lecuyer.probdist.LognormalDist + * workDistribution for work tours & trips. + * Reads the propertyMap and finds values for properties + * timeDistributionMean.nonwork, + * timeDistributionStandardDistribution.nonwork. + * Creates and stores a umontreal.iro.lecuyer.probdist.LognormalDist + * nonworkDistribution for non-work tours & trips. + * @param propertyMap + */ + + public void createTimeDistributions(HashMap propertyMap){ + + double meanWork = new Double(propertyMap.get("timeDistribution.mean.work" )); + double sdWork = new Double(propertyMap.get("timeDistribution.standardDeviation.work" )); + + double locationWork = calculateLocation(meanWork, sdWork); + double scaleWork = calculateScale(meanWork, sdWork); + + timeDistributionWork = new LognormalDist(locationWork, scaleWork); + + double meanNonWork = new Double(propertyMap.get("timeDistribution.mean.nonWork" )); + double sdNonWork = new Double(propertyMap.get("timeDistribution.standardDeviation.nonWork" )); + + double locationNonWork = calculateLocation(meanNonWork, sdNonWork); + double scaleNonWork = calculateScale(meanNonWork, sdNonWork); + + timeDistributionNonWork = new LognormalDist(locationNonWork, scaleNonWork); + + } + + /** + * Calculate the lognormal distribution location given + * the mean and standard deviation of the distribution + * according to the formula: + * + * location = ln(mean/sqrt(1 + variance/mean^2)) + * + * @param mean + * @param standardDeviation + * @return Location variable (u) + */ + public double calculateLocation(double mean, double standardDeviation){ + + double variance = standardDeviation * standardDeviation; + double meanSquared = mean * mean; + double denom = Math.sqrt(1.0 + (variance/meanSquared)); + double location = mean/denom; + if(location<=0){ + logger.error("Error: Trying to calculation location for mean "+mean + +" and standard deviation "+standardDeviation); + throw new RuntimeException(); + } + + return Math.log(location); + + } + + /** + * Calculate the lognormal distribution scale given + * the mean and standard deviation of the distribution + * according to the formula: + * + * scale = sqrt(ln(1 + variance/mean^2)); + * + * @param mean + * @param standardDeviation + * @return Scale variable (sigma) + */ + public double calculateScale(double mean, double standardDeviation){ + + double variance = standardDeviation * standardDeviation; + double meanSquared = mean * mean; + return Math.sqrt(Math.log(1 + variance/meanSquared)); + + + } + + /** + * Sample from the work distribution and return the factor to apply to work + * travel time coefficient. + * @param rnum A unit-distributed random number. + * @return The sampled time factor for work tours & trips. + */ + public double sampleFromWorkDistribution(double rnum){ + + return timeDistributionWork.inverseF(rnum); + } + + /** + * Sample from the non-work distribution and return the factor to apply + * to the non non-work travel time coefficient. + * + * @param rnum A unit-distributed random number. + * @return The sampled time factor for non-work tours and trips. + */ + public double sampleFromNonWorkDistribution(double rnum){ + + return timeDistributionNonWork.inverseF(rnum); + } + + /** + * Get the time distribution for work. + * + * @return The lognormal distribution for work. + */ + public LognormalDist getTimeDistributionWork() { + return timeDistributionWork; + } + + /** + * Get the time distribution for non-work. + * + * @return The lognormal distribution for non-work. + */ + public LognormalDist getTimeDistributionNonWork() { + return timeDistributionNonWork; + } + + + /** + * This method reads the input person file, samples from the lognormal + * time distributions for work and for non-work tours and trips, and + * appends the two fields for each person on the person file, over-writing the + * input person file with the results. If the fields already exist, nothing is + * done. The fields added to the person file are: + * + * timeFactorWork + * timeFactorNonWork + * + * @param propertyMap A property map with the following properties: + * Project.Directory: the path to directory to read the person file from. + * PopulationSynthesizer.InputToCTRAMP.PersonFile: the input person file + * timeDistribution.randomSeed: a random seed for sampling from the distributions for each person + */ + public void appendTimeDistributionsOnPersonFile(HashMap propertyMap){ + + logger.info("Appending time factors to person file"); + String directory = propertyMap.get("Project.Directory"); + String personFile = directory + propertyMap.get("PopulationSynthesizer.InputToCTRAMP.PersonFile"); + + long seed = new Long(propertyMap.get("timeDistribution.randomSeed")); + MersenneTwister random = new MersenneTwister(seed); + + Charset ENCODING = StandardCharsets.UTF_8; + + logger.info(""); + logger.info("Reading person file "+personFile); + Path path = Paths.get(personFile); + ArrayList personData = new ArrayList(5000000); + String header = null; + + + int persons = 0; + + + try (Scanner scanner = new Scanner(path, ENCODING.name())){ + + header = scanner.nextLine(); + + // does first row of person file contain field names for time factors? + if(header.contains("timeFactorWork")){ + + logger.info("File "+personFile+ " contains time factor fields already"); + scanner.close(); + return; + + }else{ + while (scanner.hasNextLine()){ + //add the row to the person data array list + personData.add(scanner.nextLine()); + ++persons; + + if(persons % 100000 == 0) + logger.info("Reading person file line "+persons); + } + } + scanner.close(); + }catch(Exception e){ + logger.fatal("Error while reading "+personFile); + throw new RuntimeException(); + } + + logger.info("Appending time factors to person file "+personFile); + header = header +",timeFactorWork,timeFactorNonWork"; + + try (BufferedWriter writer = Files.newBufferedWriter(path, ENCODING)){ + + //write the header with the additional fields + writer.write(header); + writer.newLine(); + + //write each person line, sampling from the work and non-work distributions for each and + //appending the results onto the initial data + for(int person = 0; person pMap; + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else + { + rb = ResourceBundle.getBundle(args[0]); + pMap = ResourceUtil.getResourceBundleAsHashMap(args[0]); + } + + TimeCoefficientDistributions timeDistributions = new TimeCoefficientDistributions(); + timeDistributions.createTimeDistributions(pMap); + timeDistributions.appendTimeDistributionsOnPersonFile(pMap); + + logger.info("All done!"); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java new file mode 100644 index 0000000..65f09c5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java @@ -0,0 +1,105 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class TimeDMU + implements Serializable, VariableTable +{ + + IndexValues dmuIndex = null; + + // switches used in the Individual Mandatory Tour Frequency Model + int imtfWorkSwitch, imtfSchoolSwitch; + + public TimeDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz, boolean debugUec) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debugUec) + { + dmuIndex.setDebug(true); + // dmuIndex.setDebugLabel ( "Debug IMTF Time UEC" ); + dmuIndex.setDebugLabel("Debug AO Time UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + // /** + // * Used in the Individual Mandatory Tour Frequency model; set to true + // * when the model is applied for a worker (to get round trip time to work, + // * which uses peak skims) + // * @param workOn + // */ + // public void setImtfWorkSwitch(int workOn){ + // imtfWorkSwitch = workOn; + // } + // + // /** + // * Used in the Individual Mandatory Tour Frequency model; set to true + // * when the model is applied for a student (to get round trip time to + // school, + // * which uses peak skims in the o/d direction and off-peak skims in the + // d/o + // * direction) + // * @param schoolOn + // */ + // public void setImtfSchoolSwitch(int schoolOn){ + // imtfSchoolSwitch = schoolOn; + // } + // + // public int getImtfWorkSwitch(){ + // return this.imtfWorkSwitch; + // } + // + // public int getImtfSchoolSwitch(){ + // return this.imtfSchoolSwitch; + // } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public int getIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java new file mode 100644 index 0000000..0646c5c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java @@ -0,0 +1,875 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.ArrayList; + +import org.apache.log4j.Logger; + +public class Tour + implements Serializable +{ + + private Person perObj; + private Household hhObj; + + private String tourCategory; + private String tourPurpose; + private String subtourPurpose; + + // use this array to hold personNum (i.e. index values for Household.persons + // array) for persons in tour. + // for individual tour types, this array is null. + // for joint tours, there will be an entry for each participating person. + private int[] personNumArray; + + // alternative number chosen by the joint tour composition model ( 1=adults, + // 2=children, 3=mixed ). + private int jointTourComposition; + + private int tourId; + private int tourOrigMgra; + private int tourDestMgra; + private int tourOrigWalkSubzone; + private int tourDestWalkSubzone; + private int tourDepartPeriod; + private int tourArrivePeriod; + private int tourMode; + private int subtourFreqChoice; + private int tourParkMgra; + + private float timeOfDayLogsum; + private float tourModeLogsum; + private float subtourFreqLogsum; + private float tourDestinationLogsum; + private float stopFreqLogsum; + + + private int tourPrimaryPurposeIndex; + + private float[] tourModalProbabilities; + private float[] tourModalUtilities; + + private int stopFreqChoice; + private Stop[] outboundStops; + private Stop[] inboundStops; + + private ArrayList outboundStopDestinationLogsums; + private ArrayList inboundStopDestinationLogsums; + + //Dimension N-path by 3 - 0=btap, 1=atap, 2=skim set , 3=utility + private double[][] bestWtwTapPairsOut; + + + private double[][] bestWtwTapPairsIn; + private double[][] bestWtdTapPairsOut; + private double[][] bestWtdTapPairsIn; + private double[][] bestDtwTapPairsOut; + private double[][] bestDtwTapPairsIn; + + private int choosenTransitPathIn; + + private int choosenTransitPathOut; + + private boolean useOwnedAV; + + private double valueOfTime; + + private int escortTypeOutbound; + private int escortTypeInbound; + private int driverPnumOutbound; + private int driverPnumInbound; + + // this constructor used for mandatory tour creation + public Tour(Person perObj, int tourId, int primaryIndex) + { + hhObj = perObj.getHouseholdObject(); + this.perObj = perObj; + this.tourId = tourId; + tourCategory = ModelStructure.MANDATORY_CATEGORY; + tourPrimaryPurposeIndex = primaryIndex; + + outboundStopDestinationLogsums = new ArrayList(); + inboundStopDestinationLogsums = new ArrayList(); + } + + // this constructor used for joint tour creation + public Tour(Household hhObj, String tourPurpose, String category, int primaryIndex) + { + this.hhObj = hhObj; + this.tourPurpose = tourPurpose; + tourCategory = category; + tourPrimaryPurposeIndex = primaryIndex; + outboundStopDestinationLogsums = new ArrayList(); + inboundStopDestinationLogsums = new ArrayList(); + } + + // this constructor used for individual non-mandatory or at-work subtour + // creation + public Tour(int id, Household hhObj, Person persObj, String tourPurpose, String category, + int primaryIndex) + { + this.hhObj = hhObj; + this.perObj = persObj; + tourId = id; + this.tourPurpose = tourPurpose; + tourCategory = category; + tourPrimaryPurposeIndex = primaryIndex; + outboundStopDestinationLogsums = new ArrayList(); + inboundStopDestinationLogsums = new ArrayList(); + } + + public Person getPersonObject() + { + return perObj; + } + + public void setPersonObject(Person p) + { + perObj = p; + } + + public void setPersonNumArray(int[] personNums) + { + personNumArray = personNums; + } + + public int[] getPersonNumArray() + { + return personNumArray; + } + + public boolean getPersonInJointTour(Person person) + { + boolean inTour = false; + for (int num : personNumArray) + { + if (person.getPersonNum() == num) + { + inTour = true; + break; + } + } + return inTour; + } + + public void setJointTourComposition(int compositionAlternative) + { + jointTourComposition = compositionAlternative; + } + + public int getJointTourComposition() + { + return jointTourComposition; + } + + public void setTourPurpose(String name) + { + tourPurpose = name; + } + + public void setSubTourPurpose(String name) + { + subtourPurpose = name; + } + + public String getSubTourPurpose() + { + return subtourPurpose; + } + + public String getTourCategory() + { + return tourCategory; + } + + public String getTourPurpose() + { + return tourPurpose; + } + + public String getTourPrimaryPurpose() + { + int index = tourPurpose.indexOf('_'); + if (index < 0) return tourPurpose; + else return tourPurpose.substring(0, index); + } + + // public int getTourPurposeIndex() { + // return tourPurposeIndex; + // } + + public int getTourPrimaryPurposeIndex() + { + return tourPrimaryPurposeIndex; + } + + public int getTourModeChoice() + { + return tourMode; + } + + public void setTourId(int id) + { + tourId = id; + } + + public void setTourOrigMgra(int origMgra) + { + tourOrigMgra = origMgra; + } + + public void setTourDestMgra(int destMgra) + { + tourDestMgra = destMgra; + } + + public void setTourOrigWalkSubzone(int subzone) + { + tourOrigWalkSubzone = subzone; + } + + public void setTourDestWalkSubzone(int subzone) + { + tourDestWalkSubzone = subzone; + } + + public void setTourDepartPeriod(int departPeriod) + { + tourDepartPeriod = departPeriod; + } + + public void setTourArrivePeriod(int arrivePeriod) + { + tourArrivePeriod = arrivePeriod; + } + + public void setTourModeChoice(int modeIndex) + { + tourMode = modeIndex; + } + + public void setTourParkMgra(int parkMgra) + { + tourParkMgra = parkMgra; + } + + // methods DMU will use to get info from household object + + public int getTourOrigMgra() + { + return tourOrigMgra; + } + + public int getTourDestMgra() + { + return tourDestMgra; + } + + public int getTourOrigWalkSubzone() + { + return tourOrigWalkSubzone; + } + + public int getTourDestWalkSubzone() + { + return tourDestWalkSubzone; + } + + public int getTourDepartPeriod() + { + return tourDepartPeriod; + } + + public int getTourArrivePeriod() + { + return tourArrivePeriod; + } + + public int getTourParkMgra() + { + return tourParkMgra; + } + + public int getHhId() + { + return hhObj.getHhId(); + } + + public int getHhMgra() + { + return hhObj.getHhMgra(); + } + + public int getTourId() + { + return tourId; + } + + public int getWorkTourIndexFromSubtourId(int subtourIndex) + { + // when subtour was created, it's purpose index was set to 10*work + // purpose + // index + at-work subtour index + return subtourIndex / 10; + } + + public int getSubtourIndexFromSubtourId(int subtourIndex) + { + // when subtour was created, it's purpose index was set to 10*work + // purpose + // index + at-work subtour index + int workTourIndex = subtourIndex / 10; + return subtourIndex - 10 * workTourIndex; + } + + public void setSubtourFreqChoice(int choice) + { + subtourFreqChoice = choice; + } + + public int getSubtourFreqChoice() + { + return subtourFreqChoice; + } + + public void setStopFreqChoice(int chosenAlt) + { + stopFreqChoice = chosenAlt; + } + + public int getStopFreqChoice() + { + return stopFreqChoice; + } + + public void createOutboundStops(String[] stopOrigPurposes, String[] stopDestPurposes, + int[] stopPurposeIndex) + { + outboundStops = new Stop[stopOrigPurposes.length]; + for (int i = 0; i < stopOrigPurposes.length; i++) + outboundStops[i] = new Stop(this, stopOrigPurposes[i], stopDestPurposes[i], i, false, + stopPurposeIndex[i]); + } + + public void createInboundStops(String[] stopOrigPurposes, String[] stopDestPurposes, + int[] stopPurposeIndex) + { + // needs outbound stops to be created first to get id numbering correct + + inboundStops = new Stop[stopOrigPurposes.length]; + for (int i = 0; i < stopOrigPurposes.length; i++) + inboundStops[i] = new Stop(this, stopOrigPurposes[i], stopDestPurposes[i], i, true, + stopPurposeIndex[i]); + } + + /** + * Create a Stop object to represent a half-tour where no stops were + * generated. The id for the stop is set to -1 so that trips for half-tours + * without stops can be distinguished in the output trip files from turs + * that have stops. Trips for these tours come from stop objects with ids in + * the range 0,...,3. + * + * @param origPurp + * is "home" or "work" (for at-work subtours) if outbound, or the + * primary tour purpose if inbound + * @param destPurp + * is "home" or "work" (for at-work subtours) if inbound, or the + * primary tour purpose if outbound + * @param inbound + * is true if the half-tour is inbound, or false if outbound. + * @return the created Stop object. + */ + public Stop createStop(String origPurp, String destPurp, + boolean inbound, boolean subtour) + { + Stop stop = null; + int id = -1; + if (inbound) + { + inboundStops = new Stop[1]; + inboundStops[0] = new Stop(this, origPurp, destPurp, id, inbound, 0); + stop = inboundStops[0]; + } else + { + outboundStops = new Stop[1]; + outboundStops[0] = new Stop(this, origPurp, destPurp, id, inbound, 0); + stop = outboundStops[0]; + } + return stop; + } + + public int getNumOutboundStops() + { + if (outboundStops == null) return 0; + else return outboundStops.length; + } + + public int getNumInboundStops() + { + if (inboundStops == null) return 0; + else return inboundStops.length; + } + + public Stop[] getOutboundStops() + { + return outboundStops; + } + + public Stop[] getInboundStops() + { + return inboundStops; + } + + public void clearStopModelResults() + { + stopFreqChoice = 0; + outboundStops = null; + inboundStops = null; + } + + public String getTourWindow(String purposeAbbreviation) + { + String returnString = String.format(" %5s: |", purposeAbbreviation); + short[] windows = perObj.getTimeWindows(); + for (int i = 1; i < windows.length; i++) + { + String tempString = String.format("%s", + i >= tourDepartPeriod && i <= tourArrivePeriod ? purposeAbbreviation : " "); + if (tempString.length() == 2 || tempString.length() == 3) + tempString = " " + tempString; + returnString += String.format("%4s|", tempString); + } + return returnString; + } + + public int getEscortTypeOutbound() { + return escortTypeOutbound; + } + public void setEscortTypeOutbound(int escortType) { + this.escortTypeOutbound = escortType; + } + public int getEscortTypeInbound() { + return escortTypeInbound; + } + public void setEscortTypeInbound(int escortType) { + this.escortTypeInbound = escortType; + } + public int getDriverPnumOutbound() { + return driverPnumOutbound; + } + public void setDriverPnumOutbound(int driverPnum) { + this.driverPnumOutbound = driverPnum; + } + public int getDriverPnumInbound() { + return driverPnumInbound; + } + public void setDriverPnumInbound(int driverPnum) { + this.driverPnumInbound = driverPnum; + } + public void logTourObject(Logger logger, int totalChars) + { + + String personNumArrayString = "-"; + if (personNumArray != null) + { + personNumArrayString = "[ "; + personNumArrayString += String.format("%d", personNumArray[0]); + for (int i = 1; i < personNumArray.length; i++) + personNumArrayString += String.format(", %d", personNumArray[i]); + personNumArrayString += " ]"; + } + + Household.logHelper(logger, "tourId: ", tourId, totalChars); + Household.logHelper(logger, "tourCategory: ", tourCategory, totalChars); + Household.logHelper(logger, "tourPurpose: ", tourPurpose, totalChars); + Household.logHelper(logger, "tourPurposeIndex: ", tourPrimaryPurposeIndex, totalChars); + Household.logHelper(logger, "personNumArray: ", personNumArrayString, totalChars); + Household.logHelper(logger, "jointTourComposition: ", jointTourComposition, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", tourOrigMgra, totalChars); + Household.logHelper(logger, "tourDestMgra: ", tourDestMgra, totalChars); + Household.logHelper(logger, "tourOrigWalkSubzone: ", tourOrigWalkSubzone, totalChars); + Household.logHelper(logger, "tourDestWalkSubzone: ", tourDestWalkSubzone, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", tourDepartPeriod, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", tourArrivePeriod, totalChars); + Household.logHelper(logger, "tourMode: ", tourMode, totalChars); + Household.logHelper(logger, "escortTypeOutbound: ", escortTypeOutbound, totalChars); + Household.logHelper(logger, "driverPnumOutbound: ", driverPnumOutbound, totalChars); + Household.logHelper(logger, "escortTypeInbound: ", escortTypeInbound, totalChars); + Household.logHelper(logger, "driverPnumInbound: ", driverPnumInbound, totalChars); + Household.logHelper(logger, "stopFreqChoice: ", stopFreqChoice, totalChars); + + String tempString = String.format("outboundStops[%s]:", + outboundStops == null ? "" : String.valueOf(outboundStops.length)); + logger.info(tempString); + + tempString = String.format("inboundStops[%s]:", + inboundStops == null ? "" : String.valueOf(inboundStops.length)); + logger.info(tempString); + + if ( bestWtwTapPairsOut == null ) { + tempString = "bestWtwTapPairsOut: no tap pairs saved"; + } + else { + if ( bestWtwTapPairsOut[0] == null ) + tempString = "bestWtwTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestWtwTapPairsOut: " + 0 + "[" + bestWtwTapPairsOut[0][0] + "," + bestWtwTapPairsOut[0][1] + "," + bestWtwTapPairsOut[0][2] + "]"; + for (int i=1; i < bestWtwTapPairsOut.length; i++) + if ( bestWtwTapPairsOut[i] == null ) + tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestWtwTapPairsOut[i][0] + "," + bestWtwTapPairsOut[i][1] + "," + bestWtwTapPairsOut[0][2] + "]"; + + tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; + } + logger.info(tempString); + + if ( bestWtwTapPairsIn == null ) { + tempString = "bestWtwTapPairsIn: no tap pairs saved"; + } + else { + if ( bestWtwTapPairsIn[0] == null ) + tempString = "bestWtwTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestWtwTapPairsIn: " + 0 + "[" + bestWtwTapPairsIn[0][0] + "," + bestWtwTapPairsIn[0][1] + "," + bestWtwTapPairsIn[0][2] + "]"; + for (int i=1; i < bestWtwTapPairsIn.length; i++) + if ( bestWtwTapPairsIn[i] == null ) + tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestWtwTapPairsIn[i][0] + "," + bestWtwTapPairsIn[i][1] + "," + bestWtwTapPairsIn[0][2] + "]"; + + tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; + } + logger.info(tempString); + + if ( bestWtdTapPairsOut == null ) { + tempString = "bestWtdTapPairsOut: no tap pairs saved"; + } + else { + if ( bestWtdTapPairsOut[0] == null ) + tempString = "bestWtdTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestWtdTapPairsOut: " + 0 + "[" + bestWtdTapPairsOut[0][0] + "," + bestWtdTapPairsOut[0][1] + "," + bestWtdTapPairsOut[0][2] + "]"; + for (int i=1; i < bestWtdTapPairsOut.length; i++) + if ( bestWtdTapPairsOut[i] == null ) + tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestWtdTapPairsOut[i][0] + "," + bestWtdTapPairsOut[i][1] + "," + bestWtdTapPairsOut[0][2] + "]"; + + tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; + } + logger.info(tempString); + + if ( bestWtdTapPairsIn == null ) { + tempString = "bestWtdTapPairsIn: no tap pairs saved"; + } + else { + if ( bestWtdTapPairsIn[0] == null ) + tempString = "bestWtdTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestWtdTapPairsIn: " + 0 + "[" + bestWtdTapPairsIn[0][0] + "," + bestWtdTapPairsIn[0][1] + "," + bestWtdTapPairsIn[0][2] + "]"; + for (int i=1; i < bestWtdTapPairsIn.length; i++) + if ( bestWtdTapPairsIn[i] == null ) + tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestWtdTapPairsIn[i][0] + "," + bestWtdTapPairsIn[i][1] + "," + bestWtdTapPairsIn[0][2] + "]"; + + tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; + } + logger.info(tempString); + + if ( bestDtwTapPairsOut == null ) { + tempString = "bestDtwTapPairsOut: no tap pairs saved"; + } + else { + if ( bestDtwTapPairsOut[0] == null ) + tempString = "bestDtwTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestDtwTapPairsOut: " + 0 + "[" + bestDtwTapPairsOut[0][0] + "," + bestDtwTapPairsOut[0][1] + "," + bestDtwTapPairsOut[0][2] + "]"; + for (int i=1; i < bestDtwTapPairsOut.length; i++) + if ( bestDtwTapPairsOut[i] == null ) + tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestDtwTapPairsOut[i][0] + "," + bestDtwTapPairsOut[i][1] + "," + bestDtwTapPairsOut[0][2] + "]"; + + tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; + } + logger.info(tempString); + + if ( bestDtwTapPairsIn == null ) { + tempString = "bestDtwTapPairsIn: no tap pairs saved"; + } + else { + if ( bestDtwTapPairsIn[0] == null ) + tempString = "bestDtwTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString = "bestDtwTapPairsIn: " + 0 + "[" + bestDtwTapPairsIn[0][0] + "," + bestDtwTapPairsIn[0][1] + "," + bestDtwTapPairsIn[0][2] + "]"; + for (int i=1; i < bestDtwTapPairsIn.length; i++) + if ( bestDtwTapPairsIn[i] == null ) + tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; + else + tempString += ", " + i + "[" + bestDtwTapPairsIn[i][0] + "," + bestDtwTapPairsIn[i][1] + "," + bestDtwTapPairsIn[0][2] + "]"; + + tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; + } + logger.info(tempString); + + } + + public void logEntireTourObject(Logger logger) + { + + int totalChars = 60; + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + String personNumArrayString = "-"; + if (personNumArray != null) + { + personNumArrayString = "[ "; + personNumArrayString += String.format("%d", personNumArray[0]); + for (int i = 1; i < personNumArray.length; i++) + personNumArrayString += String.format(", %d", personNumArray[i]); + personNumArrayString += " ]"; + } + + Household.logHelper(logger, "tourId: ", tourId, totalChars); + Household.logHelper(logger, "tourCategory: ", tourCategory, totalChars); + Household.logHelper(logger, "tourPurpose: ", tourPurpose, totalChars); + Household.logHelper(logger, "tourPurposeIndex: ", tourPrimaryPurposeIndex, totalChars); + Household.logHelper(logger, "personNumArray: ", personNumArrayString, totalChars); + Household.logHelper(logger, "jointTourComposition: ", jointTourComposition, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", tourOrigMgra, totalChars); + Household.logHelper(logger, "tourDestMgra: ", tourDestMgra, totalChars); + Household.logHelper(logger, "tourOrigWalkSubzone: ", tourOrigWalkSubzone, totalChars); + Household.logHelper(logger, "tourDestWalkSubzone: ", tourDestWalkSubzone, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", tourDepartPeriod, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", tourArrivePeriod, totalChars); + Household.logHelper(logger, "driverPnumOutbound: ", driverPnumOutbound, totalChars); + Household.logHelper(logger, "escortTypeInbound: ", escortTypeInbound, totalChars); + Household.logHelper(logger, "driverPnumInbound: ", driverPnumInbound, totalChars); + Household.logHelper(logger, "tourMode: ", tourMode, totalChars); + Household.logHelper(logger, "stopFreqChoice: ", stopFreqChoice, totalChars); + + if (outboundStops != null) + { + logger.info("Outbound Stops:"); + if (outboundStops.length > 0) + { + for (int i = 0; i < outboundStops.length; i++) + outboundStops[i].logStopObject(logger, totalChars); + } else + { + logger.info(" No outbound stops"); + } + } else + { + logger.info(" No outbound stops"); + } + + if (inboundStops != null) + { + logger.info("Inbound Stops:"); + if (inboundStops.length > 0) + { + for (int i = 0; i < inboundStops.length; i++) + inboundStops[i].logStopObject(logger, totalChars); + } else + { + logger.info(" No inbound stops"); + } + } else + { + logger.info(" No inbound stops"); + } + + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public void setTourModalUtilities(float[] utils) + { + tourModalUtilities = utils; + } + + public float[] getTourModalUtilities() + { + return tourModalUtilities; + } + + public void setTourModalProbabilities(float[] probs) + { + tourModalProbabilities = probs; + } + + public float[] getTourModalProbabilities() + { + return tourModalProbabilities; + } + + public void setBestWtwTapPairsOut(double[][] tapPairArray) + { + bestWtwTapPairsOut = tapPairArray; + } + + public void setBestWtwTapPairsIn(double[][] tapPairArray) + { + bestWtwTapPairsIn = tapPairArray; + } + + public void setBestWtdTapPairsOut(double[][] tapPairArray) + { + bestWtdTapPairsOut = tapPairArray; + } + + public void setBestWtdTapPairsIn(double[][] tapPairArray) + { + bestWtdTapPairsIn = tapPairArray; + } + + public void setBestDtwTapPairsOut(double[][] tapPairArray) + { + bestDtwTapPairsOut = tapPairArray; + } + + public void setBestDtwTapPairsIn(double[][] tapPairArray) + { + bestDtwTapPairsIn = tapPairArray; + } + + public void setChoosenTransitPathIn( int path ) + { + choosenTransitPathIn = path; + } + public void setChoosenTransitPathOut( int path ) + { + choosenTransitPathOut = path; + } + public double[][] getBestWtwTapPairsOut() + { + return bestWtwTapPairsOut; + } + + public double[][] getBestWtwTapPairsIn() + { + return bestWtwTapPairsIn; + } + + public double[][] getBestWtdTapPairsOut() + { + return bestWtdTapPairsOut; + } + + public double[][] getBestWtdTapPairsIn() + { + return bestWtdTapPairsIn; + } + + public double[][] getBestDtwTapPairsOut() + { + return bestDtwTapPairsOut; + } + + public double[][] getBestDtwTapPairsIn() + { + return bestDtwTapPairsIn; + } + + public double getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(double valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public float getTimeOfDayLogsum() { + return timeOfDayLogsum; + } + + public void setTimeOfDayLogsum(float timeOfDayLogsum) { + this.timeOfDayLogsum = timeOfDayLogsum; + } + + public float getTourModeLogsum() { + return tourModeLogsum; + } + + public void setTourModeLogsum(float tourModeLogsum) { + this.tourModeLogsum = tourModeLogsum; + } + + public float getSubtourFreqLogsum() { + return subtourFreqLogsum; + } + + public void setSubtourFreqLogsum(float subtourFreqLogsum) { + this.subtourFreqLogsum = subtourFreqLogsum; + } + + public float getTourDestinationLogsum() { + return tourDestinationLogsum; + } + + public void setTourDestinationLogsum(float tourDestinationLogsum) { + this.tourDestinationLogsum = tourDestinationLogsum; + } + + public float getStopFreqLogsum() { + return stopFreqLogsum; + } + + public void setStopFreqLogsum(float stopFreqLogsum) { + this.stopFreqLogsum = stopFreqLogsum; + } + + public ArrayList getOutboundStopDestinationLogsums(){ + return outboundStopDestinationLogsums; + } + public ArrayList getInboundStopDestinationLogsums(){ + return inboundStopDestinationLogsums; + } + + public void addOutboundStopDestinationLogsum(float logsum){ + outboundStopDestinationLogsums.add(logsum); + } + + public void addInboundStopDestinationLogsum(float logsum){ + inboundStopDestinationLogsums.add(logsum); + } + + public boolean getUseOwnedAV() { + return useOwnedAV; + } + + public void setUseOwnedAV(boolean useOwnedAV) { + this.useOwnedAV = useOwnedAV; + } + + /** + * Iterate through persons on tour and return non-work time factor + * for oldest person. If the person array is null then return 1.0. + * + * @return Time factor for oldest person on joint tour. + */ + public double getJointTourTimeFactor() { + int[] personNumArray = getPersonNumArray(); + int oldestAge = -999; + Person oldestPerson = null; + for (int num : personNumArray){ + Person p = hhObj.getPerson(num); + if(p.getAge() > oldestAge){ + oldestPerson = p; + oldestAge = p.getAge(); + } + } + if(oldestPerson != null) + return oldestPerson.getTimeFactorNonWork(); + + return 1.0; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java new file mode 100644 index 0000000..6e55920 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java @@ -0,0 +1,864 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class TourDepartureTimeAndDurationDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TourDepartureTimeAndDurationDMU.class); + + protected HashMap methodIndexMap; + + protected IndexValues dmuIndex; + + protected Person person; + protected Household household; + protected Tour tour; + + protected double destEmpDen; + protected int subsequentTourIsWork; + protected int subsequentTourIsSchool; + + protected double[] modeChoiceLogsums; + + private int[] altStarts; + private int[] altEnds; + + protected int originAreaType, destinationAreaType; + + protected int tourNumber; + + protected int firstTour; + protected int subsequentTour; + protected int endOfPreviousScheduledTour; + + protected ModelStructure modelStructure; + + public TourDepartureTimeAndDurationDMU(ModelStructure modelStructure) + { + this.modelStructure = modelStructure; + dmuIndex = new IndexValues(); + } + + public void setPerson(Person passedInPerson) + { + person = passedInPerson; + } + + public void setHousehold(Household passedInHousehold) + { + household = passedInHousehold; + + // set the origin and zone indices + dmuIndex.setZoneIndex(household.getHhMgra()); + dmuIndex.setHHIndex(household.getHhId()); + + // set the debug flag that can be used in the UEC + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (household.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug DepartTime UEC"); + } + + } + + public void setTour(Tour passedInTour) + { + tour = passedInTour; + } + + public void setOriginZone(int zone) + { + dmuIndex.setOriginZone(zone); + } + + public void setDestinationZone(int zone) + { + dmuIndex.setDestZone(zone); + } + + public void setOriginAreaType(int areaType) + { + originAreaType = areaType; + } + + public void setDestinationAreaType(int areaType) + { + destinationAreaType = areaType; + } + + public void setDestEmpDen(double arg) + { + destEmpDen = arg; + } + + public void setFirstTour(int trueOrFalse) + { + firstTour = trueOrFalse; + } + + public void setSubsequentTour(int trueOrFalse) + { + subsequentTour = trueOrFalse; + } + + public void setSubsequentTourIsWork(int trueOrFalse) + { + subsequentTourIsWork = trueOrFalse; + } + + public void setSubsequentTourIsSchool(int trueOrFalse) + { + subsequentTourIsSchool = trueOrFalse; + } + + /** + * Set the sequence number of this tour among all scheduled + * + * @param tourNum + */ + public void setTourNumber(int tourNum) + { + tourNumber = tourNum; + } + + public void setEndOfPreviousScheduledTour(int endHr) + { + endOfPreviousScheduledTour = endHr; + } + + public void setModeChoiceLogsums(double[] logsums) + { + modeChoiceLogsums = logsums; + } + + public void setTodAlts(int[] altStarts, int[] altEnds) + { + this.altStarts = altStarts; + this.altEnds = altEnds; + } + + public IndexValues getIndexValues() + { + return (dmuIndex); + } + + public Household getDmuHouseholdObject() + { + return household; + } + + public int getOriginZone() + { + return (dmuIndex.getOriginZone()); + } + + public int getDestinationZone() + { + return (dmuIndex.getDestZone()); + } + + public int getOriginAreaType() + { + return (originAreaType); + } + + public int getDestinationAreaType() + { + return (destinationAreaType); + } + + public int getPreDrivingAgeChild() + { + return (person.getPersonIsStudentNonDriving() == 1 || person.getPersonIsPreschoolChild() == 1) ? 1 + : 0; + } + + public int getPersonAge() + { + return person.getAge(); + } + + public int getPersonIsFemale() + { + return person.getGender() == 2 ? 1 : 0; + } + + public int getHouseholdSize() + { + return household.getHhSize(); + } + + public int getNumPreschoolChildrenInHh() + { + return household.getNumPreschool(); + } + + public int getNumChildrenUnder16InHh() + { + return household.getNumChildrenUnder16(); + } + + public int getNumNonWorkingAdultsInHh() + { + return household.getNumberOfNonWorkingAdults(); + } + + public int getFullTimeWorker() + { + return (this.person.getPersonTypeIsFullTimeWorker()); + } + + public int getPartTimeWorker() + { + return (this.person.getPersonTypeIsPartTimeWorker()); + } + + public int getUniversityStudent() + { + return (this.person.getPersonIsUniversityStudent()); + } + + public int getStudentDrivingAge() + { + return (this.person.getPersonIsStudentDriving()); + } + + public int getStudentNonDrivingAge() + { + return (this.person.getPersonIsStudentNonDriving()); + } + + public int getNonWorker() + { + return (this.person.getPersonIsNonWorkingAdultUnder65()); + } + + public int getRetired() + { + return (this.person.getPersonIsNonWorkingAdultOver65()); + } + + public int getAllAdultsFullTimeWorkers() + { + Person[] p = household.getPersons(); + boolean allAdultsAreFullTimeWorkers = true; + for (int i = 1; i < p.length; i++) + { + if (p[i].getPersonIsAdult() == 1 && p[i].getPersonIsFullTimeWorker() == 0) + { + allAdultsAreFullTimeWorkers = false; + break; + } + } + + if (allAdultsAreFullTimeWorkers) return 1; + else return 0; + } + + public int getSubtourPurposeIsEatOut() + { + if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_EAT_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getSubtourPurposeIsBusiness() + { + if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getSubtourPurposeIsOther() + { + if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_MAINT_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourPurposeIsShopping() + { + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SHOPPING_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourPurposeIsEatOut() + { + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.EAT_OUT_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourPurposeIsMaint() + { + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_MAINT_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourPurposeIsVisit() + { + if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SOCIAL_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourPurposeIsDiscr() + { + if (tour.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getNumIndivShopTours() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) + count++; + + return count; + } + + public int getNumIndivMaintTours() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) + count++; + + return count; + } + + public int getNumIndivVisitTours() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) + count++; + + return count; + } + + public int getNumIndivDiscrTours() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) + count++; + + return count; + } + + /* + * if ( tour.getTourCategory() == ModelStructure.AT_WORK_CATEGORY ) { return + * tour.getTourPurposeIndex(); } else { return 0; } } + */ + + public int getAdultsInTour() + { + + int count = 0; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person[] persons = household.getPersons(); + + int[] personNums = tour.getPersonNumArray(); + for (int i = 0; i < personNums.length; i++) + { + int p = personNums[i]; + if (persons[p].getPersonIsAdult() == 1) count++; + } + } else if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) + { + if (person.getPersonIsAdult() == 1) count = 1; + } + + return count; + } + + public int getJointTourPartySize() + { + int count = 0; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + count = tour.getPersonNumArray().length; + + return count; + } + + public int getKidsOnJointTour() + { + + int count = 0; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person[] persons = household.getPersons(); + + int[] personNums = tour.getPersonNumArray(); + for (int i = 0; i < personNums.length; i++) + { + int p = personNums[i]; + if ((persons[p].getPersonIsPreschoolChild() + + persons[p].getPersonIsStudentNonDriving() + persons[p] + .getPersonIsStudentDriving()) > 0) count++; + } + } + + return count > 0 ? 1 : 0; + + } + + // return 1 if at least one preschool or pre-driving child is in joint tour, + // otherwise 0. + public int getPreschoolPredrivingInTour() + { + + int count = 0; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person[] persons = household.getPersons(); + int[] personNums = tour.getPersonNumArray(); + for (int i = 0; i < personNums.length; i++) + { + int p = personNums[i]; + if (persons[p].getPersonIsPreschoolChild() == 1 + || persons[p].getPersonIsStudentNonDriving() == 1) return 1; + } + } else if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) + { + if (person.getPersonIsPreschoolChild() == 1 + || person.getPersonIsStudentNonDriving() == 1) count = 1; + } + + return count; + + } + + // return 1 if the person is preschool + public int getPreschool() + { + return person.getPersonIsPreschoolChild() == 1 ? 1 : 0; + + } + + // return 1 if at least one university student is in joint tour, otherwise + // 0. + public int getUnivInTour() + { + + int count = 0; + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person[] persons = household.getPersons(); + int[] personNums = tour.getPersonNumArray(); + for (int i = 0; i < personNums.length; i++) + { + int p = personNums[i]; + if (persons[p].getPersonIsUniversityStudent() == 1) return 1; + } + } else if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) + { + if (person.getPersonIsUniversityStudent() == 1) count = 1; + } + + return count; + + } + + // return 1 if all adults in joint tour are fulltime workers, 0 otherwise; + public int getAllWorkFull() + { + + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + int adultCount = 0; + int ftWorkerAdultCount = 0; + + Person[] persons = household.getPersons(); + int[] personNums = tour.getPersonNumArray(); + for (int i = 0; i < personNums.length; i++) + { + int p = personNums[i]; + if (persons[p].getPersonIsAdult() == 1) + { + adultCount++; + if (persons[p].getPersonIsFullTimeWorker() == 1) ftWorkerAdultCount++; + } + } + + if (adultCount > 0 && adultCount == ftWorkerAdultCount) return 1; + else return 0; + } + + return 0; + + } + + public int getPartyComp() + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + return tour.getJointTourComposition(); + } else + { + return 0; + } + } + + /** + * @return number of individual non-mandatory tours, including escort, for + * the person + */ + public int getPersonNonMandatoryTotalWithEscort() + { + return person.getListOfIndividualNonMandatoryTours().size(); + } + + /** + * @return number of individual non-mandatory tours, excluding escort, for + * the person + */ + public int getPersonNonMandatoryTotalNoEscort() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (!t.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) count++; + return count; + } + + /** + * @return number of individual non-mandatory discretionary tours for the + * person + */ + public int getPersonDiscrToursTotal() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + { + if (t.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) + || t.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) + || t.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) count++; + } + return count; + } + + /** + * @return number of individual non-mandatory tours, excluding escort, for + * the person + */ + public int getPersonEscortTotal() + { + int count = 0; + for (Tour t : person.getListOfIndividualNonMandatoryTours()) + if (t.getTourPurpose().startsWith("escort")) count++; + return count; + } + + public int getHhJointTotal() + { + Tour[] jt = household.getJointTourArray(); + if (jt == null) return 0; + else return jt.length; + } + + public int getPersonMandatoryTotal() + { + return person.getListOfWorkTours().size() + person.getListOfSchoolTours().size(); + } + + public int getPersonJointTotal() + { + Tour[] jtArray = household.getJointTourArray(); + if (jtArray == null) + { + return 0; + } else + { + int numJtParticipations = 0; + for (Tour jt : jtArray) + { + int[] personJtIndices = jt.getPersonNumArray(); + for (int pNum : personJtIndices) + { + if (pNum == person.getPersonNum()) + { + numJtParticipations++; + break; + } + } + } + return numJtParticipations; + } + } + + public int getPersonJointAndIndivDiscrToursTotal() + { + + int totDiscr = getPersonDiscrToursTotal(); + + Tour[] jtArray = household.getJointTourArray(); + if (jtArray == null) + { + return totDiscr; + } else + { + // count number of joint discretionary tours person participates in + int numJtParticipations = 0; + for (Tour jt : jtArray) + { + if (jt.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) + || jt.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) + || jt.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) + { + int[] personJtIndices = jt.getPersonNumArray(); + for (int pNum : personJtIndices) + { + if (pNum == person.getPersonNum()) + { + numJtParticipations++; + break; + } + } + } + } + return numJtParticipations + totDiscr; + } + } + + public int getFirstTour() + { + return firstTour; + } + + public int getSubsequentTour() + { + return subsequentTour; + } + + public int getWorkAndSchoolToursByWorker() + { + int returnValue = 0; + if (person.getPersonIsWorker() == 1) + { + if (person.getImtfChoice() == HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_WORK_AND_SCHOOL) + returnValue = 1; + } + return returnValue; + } + + public int getWorkAndSchoolToursByStudent() + { + int returnValue = 0; + if (person.getPersonIsStudent() == 1) + { + if (person.getImtfChoice() == HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_WORK_AND_SCHOOL) + returnValue = 1; + } + return returnValue; + } + + public double getModeChoiceLogsumAlt(int alt) + { + + int startPeriod = altStarts[alt - 1]; + int endPeriod = altEnds[alt - 1]; + + int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); + + return modeChoiceLogsums[index]; + + } + + public int getPrevTourEndsThisDeparturePeriodAlt(int alt) + { + + // get the departure period for the current alternative + int thisTourStartsPeriod = altStarts[alt - 1]; + + if (person.isPreviousArrival(thisTourStartsPeriod)) return 1; + else return 0; + + } + + public int getPrevTourBeginsThisArrivalPeriodAlt(int alt) + { + + // get the arrival period for the current alternative + int thisTourEndsPeriod = altStarts[alt - 1]; + + if (person.isPreviousDeparture(thisTourEndsPeriod)) return 1; + else return 0; + + } + + public int getAdjWindowBeforeThisPeriodAlt(int alt) + { + + int thisTourStartsPeriod = altStarts[alt - 1]; + + int numAdjacentPeriodsAvailable = 0; + for (int i = thisTourStartsPeriod - 1; i >= 0; i--) + { + if (person.isPeriodAvailable(i)) numAdjacentPeriodsAvailable++; + else break; + } + + return numAdjacentPeriodsAvailable; + + } + + public int getAdjWindowAfterThisPeriodAlt(int alt) + { + + int thisTourEndsPeriod = altEnds[alt - 1]; + + int numAdjacentPeriodsAvailable = 0; + for (int i = thisTourEndsPeriod + 1; i < modelStructure.getNumberOfTimePeriods(); i++) + { + if (person.isPeriodAvailable(i)) numAdjacentPeriodsAvailable++; + else break; + } + + return numAdjacentPeriodsAvailable; + + } + + public int getRemainingPeriodsAvailableAlt(int alt) + { + + int periodsAvail = person.getAvailableWindow(); + + int start = altStarts[alt - 1]; + int end = altEnds[alt - 1]; + + // determine the availabilty of each period after the alternative time + // window + // is hypothetically scheduled + // if start == end, the availability won't change, so no need to + // compute. + if (start != end) + { + + // the start and end periods will always be available after + // scheduling, so + // don't need to check them. + // the periods between start/end must be 0 or the alternative could + // not + // have been available, + // so count them all as unavailable after scheduling this window. + periodsAvail -= (end - start - 1); + + } + + return periodsAvail; + + } + + public float getRemainingInmToursToAvailablePeriodsRatioAlt(int alt) + { + int periodsAvail = getRemainingPeriodsAvailableAlt(alt); + if (periodsAvail > 0) + { + float ratio = (float) (person.getListOfIndividualNonMandatoryTours().size() - tourNumber) + / periodsAvail; + return ratio; + } else return -999; + } + + public int getMaximumAvailableTimeWindow() + { + return person.getMaximumContinuousAvailableWindow(); + } + + public int getMaxJointTimeWindow() + { + return household.getMaxJointTimeWindow(tour); + } + + /** + * get the number of tours left to be scheduled, including the current tour + * + * @return number of tours left to be scheduled, including the current tour + */ + public int getToursLeftToSchedule() + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Tour[] jt = household.getJointTourArray(); + return jt.length - tourNumber + 1; + } else return person.getListOfIndividualNonMandatoryTours().size() - tourNumber + 1; + } + + public int getEndOfPreviousTour() + { + return endOfPreviousScheduledTour; + } + + public int getTourCategoryIsJoint() + { + return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 + : 0; + } + + public float getOpSovTimeOd() + { + return 1; + } + + public float getOpSovTimeDo() + { + return 1; + } + + public int getDestMgraInCbd() + { + return 0; + } + + public int getOrigMgraInRural() + { + return 0; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java new file mode 100644 index 0000000..7e83278 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java @@ -0,0 +1,523 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class TourModeChoiceDMU implements Serializable, + VariableTable { + + protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); + + public static final int WTW = McLogsumsCalculator.WTW; + public static final int WTD = McLogsumsCalculator.WTD; + public static final int DTW = McLogsumsCalculator.DTW; + protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + protected float origTaxiWaitTime; + protected float destTaxiWaitTime; + protected float origSingleTNCWaitTime; + protected float destSingleTNCWaitTime; + protected float origSharedTNCWaitTime; + protected float destSharedTNCWaitTime; + + + protected Household hh; + protected Tour tour; + protected Tour workTour; + protected Person person; + + protected ModelStructure modelStructure; + + protected double origDuDen; + protected double origEmpDen; + protected double origTotInt; + protected double destDuDen; + protected double destEmpDen; + protected double destTotInt; + + protected double lsWgtAvgCostM; + protected double lsWgtAvgCostD; + protected double lsWgtAvgCostH; + protected double reimburseProportion; + protected int parkingArea; + + protected float pTazTerminalTime; + protected float aTazTerminalTime; + + protected double nmWalkTimeOut; + protected double nmWalkTimeIn; + protected double nmBikeTimeOut; + protected double nmBikeTimeIn; + + protected int originMgra; + protected int destMgra; + + protected double ivtCoeff; + protected double costCoeff; + + protected double[][] transitLogSum; + + + public TourModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) { + this.modelStructure = modelStructure; + dmuIndex = new IndexValues(); + + //accEgr by in/outbound + transitLogSum = new double[NUM_ACC_EGR][NUM_DIR]; + + } + + public void setHouseholdObject(Household hhObject) { + hh = hhObject; + } + + + public Household getHouseholdObject() { + return hh; + } + + public void setPersonObject(Person personObject) { + person = personObject; + } + + public Person getPersonObject() { + return person; + } + + public void setWorkTourObject(Tour tourObject) { + workTour = tourObject; + } + + public void setTourObject(Tour tourObject) { + tour = tourObject; + } + + public Tour getTourObject() { + return tour; + } + + public int getParkingArea() { + return parkingArea; + } + + public void setParkingArea(int parkingArea) { + this.parkingArea = parkingArea; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, + int destIndex, boolean debug) { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public int getPersonType() { + return person.getPersonTypeNumber(); + } + + public void setOrigDuDen(double arg) { + origDuDen = arg; + } + + public void setOrigEmpDen(double arg) { + origEmpDen = arg; + } + + public void setOrigTotInt(double arg) { + origTotInt = arg; + } + + public void setDestDuDen(double arg) { + destDuDen = arg; + } + + public void setDestEmpDen(double arg) { + destEmpDen = arg; + } + + public void setDestTotInt(double arg) { + destTotInt = arg; + } + + public void setReimburseProportion(double proportion) { + reimburseProportion = proportion; + } + + public void setLsWgtAvgCostM(double cost) { + lsWgtAvgCostM = cost; + } + + public void setLsWgtAvgCostD(double cost) { + lsWgtAvgCostD = cost; + } + + public void setLsWgtAvgCostH(double cost) { + lsWgtAvgCostH = cost; + } + + public void setPTazTerminalTime(float time) { + pTazTerminalTime = time; + } + + public void setATazTerminalTime(float time) { + aTazTerminalTime = time; + } + + public IndexValues getDmuIndexValues() { + return dmuIndex; + } + + public void setIndexDest(int d) { + dmuIndex.setDestZone(d); + } + + public void setTransitLogSum(int accEgr, boolean inbound, double value){ + transitLogSum[accEgr][inbound == true ? 1 : 0] = value; + } + + protected double getTransitLogSum(int accEgr,boolean inbound){ + return transitLogSum[accEgr][inbound == true ? 1 : 0]; + } + + public int getWorkTourModeIsSov() { + boolean tourModeIsSov = modelStructure.getTourModeIsSov(workTour + .getTourModeChoice()); + return tourModeIsSov ? 1 : 0; + } + + public int getWorkTourModeIsHov() { + boolean tourModeIsHov = modelStructure.getTourModeIsHov(workTour + .getTourModeChoice()); + return tourModeIsHov ? 1 : 0; + } + + public int getWorkTourModeIsBike() { + boolean tourModeIsBike = modelStructure.getTourModeIsBike(workTour + .getTourModeChoice()); + return tourModeIsBike ? 1 : 0; + } + + public int getTourCategoryJoint() { + if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + return 1; + else + return 0; + } + + public int getTourCategoryEscort() { + if (tour.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) + return 1; + else + return 0; + } + + public int getTourCategorySubtour() { + if (tour.getTourCategory().equalsIgnoreCase( + ModelStructure.AT_WORK_CATEGORY)) + return 1; + else + return 0; + } + + public int getNumberOfParticipantsInJointTour() { + int[] participants = tour.getPersonNumArray(); + int returnValue = 0; + if (participants != null) + returnValue = participants.length; + return returnValue; + } + + public int getHhSize() { + return hh.getHhSize(); + } + + public int getAutos() { + return hh.getAutosOwned(); + } + + public int getAge() { + return person.getAge(); + } + + public int getIncomeCategory() { + return hh.getIncomeCategory(); + } + + public int getIncomeInDollars() { + return hh.getIncomeInDollars(); + } + + + public void setNmWalkTimeOut(double nmWalkTime) { + nmWalkTimeOut = nmWalkTime; + } + + public double getNmWalkTimeOut() { + return nmWalkTimeOut; + } + + public void setNmWalkTimeIn(double nmWalkTime) { + nmWalkTimeIn = nmWalkTime; + } + + public double getNmWalkTimeIn() { + return nmWalkTimeIn; + } + + public void setNmBikeTimeOut(double nmBikeTime) { + nmBikeTimeOut = nmBikeTime; + } + + public double getNmBikeTimeOut() { + return nmBikeTimeOut; + } + + public void setNmBikeTimeIn(double nmBikeTime) { + nmBikeTimeIn = nmBikeTime; + } + + public double getNmBikeTimeIn() { + return nmBikeTimeIn; + } + + public double getWorkTimeFactor() { + return person.getTimeFactorWork(); + } + + public double getNonWorkTimeFactor(){ + return person.getTimeFactorNonWork(); + } + + /** + * Iterate through persons on tour and return non-work time factor + * for oldest person. If the person array is null then return 1.0. + * + * @return Time factor for oldest person on joint tour. + */ + public double getJointTourTimeFactor() { + int[] personNumArray = tour.getPersonNumArray(); + int oldestAge = -999; + Person oldestPerson = null; + for (int num : personNumArray){ + Person p = hh.getPerson(num); + if(p.getAge() > oldestAge){ + oldestPerson = p; + oldestAge = p.getAge(); + } + } + if(oldestPerson != null) + return oldestPerson.getTimeFactorNonWork(); + + return 1.0; + } + + + public int getFreeParkingEligibility() { + return person.getFreeParkingAvailableResult(); + } + + public double getReimburseProportion() { + return reimburseProportion; + } + + public double getMonthlyParkingCost() { + return lsWgtAvgCostM; + } + + public double getDailyParkingCost() { + return lsWgtAvgCostD; + } + + public double getHourlyParkingCost() { + return lsWgtAvgCostH; + } + + public double getPTazTerminalTime() { + return pTazTerminalTime; + } + + public double getATazTerminalTime() { + return aTazTerminalTime; + } + + public void setOriginMgra( int value ) { + originMgra = value; + } + + public void setDestMgra( int value ) { + destMgra = value; + } + + public int getOriginMgra() { + return originMgra; + } + + public int getDestMgra() { + return destMgra; + } + + /** + * 1 if household owns transponder, else 0 + * @return 1 if household owns transponder, else 0 + */ + public int getTransponderOwnership(){ + return hh.getTpChoice(); + } + + public double getIvtCoeff() { + return ivtCoeff; +} + +public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; +} + +public double getCostCoeff() { + return costCoeff; +} + +public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; +} + + public int getIndexValue(String variableName) { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) { + throw new UnsupportedOperationException(); + } + public int getUseOwnedAV(){ + + if(tour==null) + return 0; + + return (tour.getUseOwnedAV() ? 1: 0); + } + + + + public float getOrigTaxiWaitTime() { + return origTaxiWaitTime; + } + + + + public void setOrigTaxiWaitTime(float origTaxiWaitTime) { + this.origTaxiWaitTime = origTaxiWaitTime; + } + + + + public float getDestTaxiWaitTime() { + return destTaxiWaitTime; + } + + + + public void setDestTaxiWaitTime(float destTaxiWaitTime) { + this.destTaxiWaitTime = destTaxiWaitTime; + } + + + + public float getOrigSingleTNCWaitTime() { + return origSingleTNCWaitTime; + } + + + + public void setOrigSingleTNCWaitTime(float origSingleTNCWaitTime) { + this.origSingleTNCWaitTime = origSingleTNCWaitTime; + } + + + + public float getDestSingleTNCWaitTime() { + return destSingleTNCWaitTime; + } + + + + public void setDestSingleTNCWaitTime(float destSingleTNCWaitTime) { + this.destSingleTNCWaitTime = destSingleTNCWaitTime; + } + + + + public float getOrigSharedTNCWaitTime() { + return origSharedTNCWaitTime; + } + + + + public void setOrigSharedTNCWaitTime(float origSharedTNCWaitTime) { + this.origSharedTNCWaitTime = origSharedTNCWaitTime; + } + + + + public float getDestSharedTNCWaitTime() { + return destSharedTNCWaitTime; + } + + + + public void setDestSharedTNCWaitTime(float destSharedTNCWaitTime) { + this.destSharedTNCWaitTime = destSharedTNCWaitTime; + } + + + +} + diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java new file mode 100644 index 0000000..48122f6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java @@ -0,0 +1,762 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; + + + + + + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; +import org.sandag.abm.modechoice.MgraDataManager; + +public class TourModeChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(TourModeChoiceModel.class); + private transient Logger tourMCManLogger = Logger.getLogger("tourMcMan"); + private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); + + public static final String MANDATORY_MODEL_INDICATOR = ModelStructure.MANDATORY_CATEGORY; + public static final String NON_MANDATORY_MODEL_INDICATOR = "Non-Mandatory"; + public static final String AT_WORK_SUBTOUR_MODEL_INDICATOR = ModelStructure.AT_WORK_CATEGORY; + + public static final boolean DEBUG_BEST_PATHS = true; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + private static final int MC_DATA_SHEET = 0; + private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; + private static final String PROPERTIES_UEC_MAINT_TOUR_MODE_SHEET = "tourModeChoice.maint.model.page"; + private static final String PROPERTIES_UEC_DISCR_TOUR_MODE_SHEET = "tourModeChoice.discr.model.page"; + private static final String PROPERTIES_UEC_AT_WORK_TOUR_MODE_SHEET = "tourModeChoice.atwork.model.page"; + + + private static final String PROPERTIES_TOUR_UTILITY_IVT_COEFFS = "tour.utility.ivt.coeffs"; + private static final String PROPERTIES_TOUR_UTILITY_INCOME_COEFFS = "tour.utility.income.coeffs"; + private static final String PROPERTIES_TOUR_UTILITY_INCOME_EXPONENTS = "tour.utility.income.exponents"; + + // A MyChoiceModelApplication object and modeAltsAvailable[] is needed for + // each purpose + private ChoiceModelApplication[] mcModel; + private TourModeChoiceDMU mcDmuObject; + private McLogsumsCalculator logsumHelper; + + private ModelStructure modelStructure; + + private String tourCategory; + private String[] tourPurposeList; + + private HashMap purposeModelIndexMap; + + private String[][] modeAltNames; + + private boolean saveUtilsProbsFlag = false; + + // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose + double[] ivtCoeffs; + double[] incomeCoeffs; + double[] incomeExponents; + + private MgraDataManager mgraManager; + + //added for TNC and Taxi modes + TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; + + public TourModeChoiceModel(HashMap propertyMap, ModelStructure myModelStructure, + String myTourCategory, CtrampDmuFactoryIf dmuFactory, McLogsumsCalculator myLogsumHelper) + { + + modelStructure = myModelStructure; + tourCategory = myTourCategory; + logsumHelper = myLogsumHelper; + // logsumHelper passed in, but if it were instantiated here, it woul be + // as follows + // logsumHelper = new McLogsumsAppender(); + // logsumHelper.setupSkimCalculators(propertyMap); + + mcDmuObject = dmuFactory.getModeChoiceDMU(); + setupModeChoiceModelApplicationArray(propertyMap, tourCategory); + + mgraManager = MgraDataManager.getInstance(); + + //get the coefficients for ivt and the coefficients to calculate the cost coefficient + ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_IVT_COEFFS); + incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_INCOME_COEFFS); + incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_INCOME_EXPONENTS); + + tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); + tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); + + } + + public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() + { + return logsumHelper.getAnmSkimCalculator(); + } + + private void setupModeChoiceModelApplicationArray(HashMap propertyMap, + String tourCategory) + { + + logger.info(String.format("setting up %s tour mode choice model.", tourCategory)); + + // locate the individual mandatory tour mode choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = propertyMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + // default is to not save the tour mode choice utils and probs for each + // tour + String saveUtilsProbsString = propertyMap + .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); + if (saveUtilsProbsString != null) + { + if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; + } + + // get the number of purposes and declare the array dimension to be this + // size. + HashMap modelIndexMap = new HashMap(); + + // create a HashMap to map purposeName to model index + purposeModelIndexMap = new HashMap(); + + if (tourCategory.equalsIgnoreCase(MANDATORY_MODEL_INDICATOR)) + { + tourPurposeList = new String[3]; + tourPurposeList[0] = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; + tourPurposeList[1] = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + tourPurposeList[2] = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + + int uecIndex = 1; + int mcModelIndex = 0; + for (String purposeName : tourPurposeList) + { + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, mcModelIndex); + purposeModelIndexMap.put(purposeName, mcModelIndex++); + } else + { + purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); + } + uecIndex++; + } + + } else if (tourCategory.equalsIgnoreCase(NON_MANDATORY_MODEL_INDICATOR)) + { + tourPurposeList = new String[6]; + tourPurposeList[0] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; + tourPurposeList[1] = ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME; + tourPurposeList[2] = ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME; + tourPurposeList[3] = ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME; + tourPurposeList[4] = ModelStructure.VISITING_PRIMARY_PURPOSE_NAME; + tourPurposeList[5] = ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME; + + int maintSheet = Integer + .parseInt(propertyMap.get(PROPERTIES_UEC_MAINT_TOUR_MODE_SHEET)); + int discrSheet = Integer + .parseInt(propertyMap.get(PROPERTIES_UEC_DISCR_TOUR_MODE_SHEET)); + + int uecIndex = 1; + int mcModelIndex = 0; + int i = 0; + for (String purposeName : tourPurposeList) + { + + uecIndex = -1; + if (purposeName.equalsIgnoreCase(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME) + || purposeName + .equalsIgnoreCase(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME) + || purposeName + .equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) uecIndex = maintSheet; + else if (purposeName.equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) + || purposeName + .equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) + || purposeName + .equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) + uecIndex = discrSheet; + + // if the uec sheet for the model segment is not in the map, add + // it, otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, mcModelIndex); + purposeModelIndexMap.put(purposeName, mcModelIndex++); + } else + { + purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); + } + i++; + } + + } else if (tourCategory.equalsIgnoreCase(AT_WORK_SUBTOUR_MODEL_INDICATOR)) + { + tourPurposeList = new String[1]; + tourPurposeList[0] = ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME; + + int[] uecSheets = new int[1]; + uecSheets[0] = Integer + .parseInt(propertyMap.get(PROPERTIES_UEC_AT_WORK_TOUR_MODE_SHEET)); + + int mcModelIndex = 0; + int i = 0; + for (String purposeName : tourPurposeList) + { + int uecIndex = uecSheets[i]; + + // if the uec sheet for the model segment is not in the map, add + // it, otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, mcModelIndex); + purposeModelIndexMap.put(purposeName, mcModelIndex++); + } else + { + purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); + } + i++; + } + + } + + mcModel = new ChoiceModelApplication[modelIndexMap.size()]; + + // declare dimensions for the array of choice alternative availability + // by + // purpose + modeAltNames = new String[purposeModelIndexMap.size()][]; + + // for each unique model index, create the ChoiceModelApplication object + // and + // the availabilty array + int i = 0; + for (int m : modelIndexMap.keySet()) + { + mcModel[i] = new ChoiceModelApplication(mcUecFile, m, MC_DATA_SHEET, propertyMap, + (VariableTable) mcDmuObject); + modeAltNames[i] = mcModel[i].getAlternativeNames(); + i++; + } + + + } + + public double getModeChoiceLogsum(Household household, Person person, Tour tour, + Logger modelLogger, String choiceModelDescription, String decisionMakerLabel) + { + + // update the MC dmuObjects for this person + mcDmuObject.setHouseholdObject(household); + mcDmuObject.setPersonObject(person); + mcDmuObject.setTourObject(tour); + mcDmuObject.setDmuIndexValues(household.getHhId(), tour.getTourDestMgra(), + tour.getTourOrigMgra(), tour.getTourDestMgra(), household.getDebugChoiceModels()); + mcDmuObject.setOriginMgra(tour.getTourOrigMgra()); + mcDmuObject.setDestMgra(tour.getTourDestMgra()); + + float SingleTNCWaitTimeOrig = 0; + float SingleTNCWaitTimeDest = 0; + float SharedTNCWaitTimeOrig = 0; + float SharedTNCWaitTimeDest = 0; + float TaxiWaitTimeOrig = 0; + float TaxiWaitTimeDest = 0; + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); + float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); + + if(household!=null){ + Random hhRandom = household.getHhRandom(); + double rnum = hhRandom.nextDouble(); + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); + }else{ + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); + } + + mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); + mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); + mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); + mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); + mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); + mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); + + return getModeChoiceLogsum(mcDmuObject, tour, modelLogger, choiceModelDescription, + decisionMakerLabel); + } + + public double getModeChoiceLogsum(TourModeChoiceDMU mcDmuObject, Tour tour, Logger modelLogger, + String choiceModelDescription, String decisionMakerLabel) + { + + int modelIndex = purposeModelIndexMap.get(tour.getTourPrimaryPurpose()); + + Household household = tour.getPersonObject().getHouseholdObject(); + double income = (double) household.getIncomeInDollars(); + double timeFactor = 1.0f; + if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + timeFactor = mcDmuObject.getJointTourTimeFactor(); + else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) + timeFactor = mcDmuObject.getWorkTimeFactor(); + else + timeFactor = mcDmuObject.getNonWorkTimeFactor(); + + double ivtCoeff = ivtCoeffs[modelIndex]; + double incomeCoeff = incomeCoeffs[modelIndex]; + double incomeExpon = incomeExponents[modelIndex]; + double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); + + mcDmuObject.setIvtCoeff(ivtCoeff*timeFactor); + mcDmuObject.setCostCoeff(costCoeff); + + float SingleTNCWaitTimeOrig = 0; + float SingleTNCWaitTimeDest = 0; + float SharedTNCWaitTimeOrig = 0; + float SharedTNCWaitTimeDest = 0; + float TaxiWaitTimeOrig = 0; + float TaxiWaitTimeDest = 0; + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); + float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); + + if(household!=null){ + Random hhRandom = household.getHhRandom(); + double rnum = hhRandom.nextDouble(); + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); + }else{ + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); + } + + mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); + mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); + mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); + mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); + mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); + mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); + + // log headers to traceLogger + if (household.getDebugChoiceModels()) + { + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + } + + double logsum = logsumHelper.calculateTourMcLogsum(tour.getTourOrigMgra(), + tour.getTourDestMgra(), tour.getTourDepartPeriod(), tour.getTourArrivePeriod(), + mcModel[modelIndex], mcDmuObject); + + // write UEC calculation results to separate model specific log file + if (household.getDebugChoiceModels()) + { + String loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); + modelLogger.info(""); + modelLogger.info(""); + } + + return logsum; + + } + + public int getModeChoice(TourModeChoiceDMU mcDmuObject, String purposeName) + { + + int modelIndex = purposeModelIndexMap.get(purposeName); + + Household household = mcDmuObject.getHouseholdObject(); + Tour tour = mcDmuObject.getTourObject(); + double income = (double) household.getIncomeInDollars(); + double ivtCoeff = ivtCoeffs[modelIndex]; + double incomeCoeff = incomeCoeffs[modelIndex]; + double incomeExpon = incomeExponents[modelIndex]; + double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); + double timeFactor = 1.0f; + if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + timeFactor = mcDmuObject.getJointTourTimeFactor(); + else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) + timeFactor = mcDmuObject.getWorkTimeFactor(); + else + timeFactor = mcDmuObject.getNonWorkTimeFactor(); + + mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); + mcDmuObject.setCostCoeff(costCoeff); + + float SingleTNCWaitTimeOrig = 0; + float SingleTNCWaitTimeDest = 0; + float SharedTNCWaitTimeOrig = 0; + float SharedTNCWaitTimeDest = 0; + float TaxiWaitTimeOrig = 0; + float TaxiWaitTimeDest = 0; + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); + float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); + + if(household!=null){ + Random hhRandom = household.getHhRandom(); + double rnum = hhRandom.nextDouble(); + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); + }else{ + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); + } + + mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); + mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); + mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); + mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); + mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); + mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); + + Logger modelLogger = null; + if (tourCategory.equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) modelLogger = tourMCManLogger; + else modelLogger = tourMCNonManLogger; + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + String separator = ""; + + + if (household.getDebugChoiceModels()) + { + + if (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person person = null; + Person[] persons = mcDmuObject.getHouseholdObject().getPersons(); + int[] personNums = tour.getPersonNumArray(); + for (int n = 0; n < personNums.length; n++) + { + int p = personNums[n]; + person = persons[p]; + + choiceModelDescription = String.format( + "%s Tour Mode Choice Model for: Purpose=%s, Home=%d, Dest=%d", + tourCategory, purposeName, household.getHhMgra(), + tour.getTourDestMgra()); + decisionMakerLabel = String + .format("HH=%d, person record %d of %d in joint tour, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), p, personNums.length, + person.getPersonNum(), person.getPersonType(), tour.getTourId()); + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + household.logTourObject(loggingHeader, modelLogger, person, tour); + } + } else + { + Person person = mcDmuObject.getPersonObject(); + + choiceModelDescription = String.format( + "%s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", + tourCategory, purposeName, tour.getTourOrigMgra(), tour.getTourDestMgra()); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + household.logTourObject(loggingHeader, modelLogger, person, tour); + } + + } + + logsumHelper.setTourMcDmuAttributes(mcDmuObject, tour.getTourOrigMgra(), + tour.getTourDestMgra(), tour.getTourDepartPeriod(), tour.getTourArrivePeriod(), + (household.getDebugChoiceModels() && DEBUG_BEST_PATHS)); + + // mode choice UEC references highway skim matrices directly, so set + // index orig/dest to O/D TAZs. + IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); + mcDmuIndex.setOriginZone(mgraManager.getTaz(tour.getTourOrigMgra())); + mcDmuIndex.setDestZone(mgraManager.getTaz(tour.getTourDestMgra())); + mcDmuIndex.setZoneIndex(tour.getTourDestMgra()); + mcDmuObject.setOriginMgra(tour.getTourOrigMgra()); + mcDmuObject.setDestMgra(tour.getTourDestMgra()); + + float logsum = (float) mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuIndex); + tour.setTourModeLogsum(logsum); + + mcDmuIndex.setOriginZone(tour.getTourOrigMgra()); + mcDmuIndex.setDestZone(tour.getTourDestMgra()); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen; + if (mcModel[modelIndex].getAvailabilityCount() > 0) + { + + chosen = mcModel[modelIndex].getChoiceResult(rn); + + // best tap pairs were determined and saved in mcDmuObject while + // setting dmu skim attributes + // if chosen mode is a transit mode, save these tap pairs in the + // tour object; if not transit tour attributes remain null. + if (modelStructure.getTourModeIsTransit(chosen)) + { + tour.setBestWtwTapPairsOut(logsumHelper.getBestWtwTapsOut()); + tour.setBestWtwTapPairsIn(logsumHelper.getBestWtwTapsIn()); + tour.setBestWtdTapPairsOut(logsumHelper.getBestWtdTapsOut()); + tour.setBestWtdTapPairsIn(logsumHelper.getBestWtdTapsIn()); + tour.setBestDtwTapPairsOut(logsumHelper.getBestDtwTapsOut()); + tour.setBestDtwTapPairsIn(logsumHelper.getBestDtwTapsIn()); + } + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = mcModel[modelIndex].getUEC(); + + double vot = 0.0; + + if(modelStructure.getTourModeIsS2(chosen)){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = uec.getValueForIndex(votIndex); + }else if (modelStructure.getTourModeIsS3(chosen)){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = uec.getValueForIndex(votIndex); + } + tour.setValueOfTime(vot); + + + } else + { + + if (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + Person person = null; + Person[] persons = mcDmuObject.getHouseholdObject().getPersons(); + int[] personNums = tour.getPersonNumArray(); + for (int n = 0; n < personNums.length; n++) + { + int p = personNums[n]; + person = persons[p]; + + choiceModelDescription = String + .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Home=%d, Dest=%d", + tourCategory, purposeName, household.getHhMgra(), + tour.getTourDestMgra()); + decisionMakerLabel = String + .format("HH=%d, person record %d of %d in joint tour, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), p, personNums.length, + person.getPersonNum(), person.getPersonType(), tour.getTourId()); + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading( + choiceModelDescription, decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + household.logTourObject(loggingHeader, modelLogger, person, tour); + } + } else + { + Person person = mcDmuObject.getPersonObject(); + + choiceModelDescription = String + .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", + tourCategory, purposeName, tour.getTourOrigMgra(), + tour.getTourDestMgra()); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tour.getTourId()); + loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + + mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + modelLogger.info(" "); + for (int k = 0; k < loggingHeader.length(); k++) + separator += "+"; + modelLogger.info(loggingHeader); + modelLogger.info(separator); + + household.logTourObject(loggingHeader, modelLogger, person, tour); + } + + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + modelLogger.info(""); + modelLogger.info(""); + + logger.error(String + .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", + household.getHhId(), tourCategory)); + throw new RuntimeException(); + } + + // debug output + if (household.getDebugChoiceModels()) + { + + double[] utilities = mcModel[modelIndex].getUtilities(); // 0s-indexing + double[] probabilities = mcModel[modelIndex].getProbabilities(); // 0s-indexing + boolean[] availabilities = mcModel[modelIndex].getAvailabilities(); // 1s-indexing + String[] altNames = mcModel[modelIndex].getAlternativeNames(); // 0s-indexing + + if (tour.getTourCategory() + .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) + { + modelLogger.info("Joint Tour Id: " + tour.getTourId()); + } else + { + Person person = mcDmuObject.getPersonObject(); + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString + + ", Tour Id: " + tour.getTourId()); + } + modelLogger + .info("Alternative Utility Probability CumProb"); + modelLogger + .info("-------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < mcModel[modelIndex].getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %s", k + 1, altNames[k]); + modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger.info(separator); + modelLogger.info(""); + modelLogger.info(""); + + // write choice model alternative info to log file + mcModel[modelIndex].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + mcModel[modelIndex].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + mcModel[modelIndex].logLogitCalculations(choiceModelDescription, decisionMakerLabel); + + // write UEC calculation results to separate model specific log file + mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); + } + + if (saveUtilsProbsFlag) + { + + // get the utilities and probabilities arrays for the tour mode + // choice + // model for this tour and save them to the tour object + double[] dUtils = mcModel[modelIndex].getUtilities(); + double[] dProbs = mcModel[modelIndex].getProbabilities(); + + float[] utils = new float[dUtils.length]; + float[] probs = new float[dUtils.length]; + for (int k = 0; k < dUtils.length; k++) + { + utils[k] = (float) dUtils[k]; + probs[k] = (float) dProbs[k]; + } + + tour.setTourModalUtilities(utils); + tour.setTourModalProbabilities(probs); + + } + + return chosen; + + } + + public String[] getModeAltNames(int purposeIndex) + { + int modelIndex = purposeModelIndexMap.get(tourPurposeList[purposeIndex]); + return modeAltNames[modelIndex]; + } + + /** + * This method calculates a cost coefficient based on the following formula: + * + * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) + * + * + * @param incomeCoeff + * @param incomeExponent + * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice + */ + public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ + + return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java new file mode 100644 index 0000000..bd6ef9b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java @@ -0,0 +1,189 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import com.pb.common.calculator.VariableTable; +import com.pb.common.model.ModelException; + +import org.apache.log4j.Logger; + +public class TourVehicleTypeChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(TourVehicleTypeChoiceModel.class); + float probabilityBoostAutosLTDrivers = 0; + float probabilityBoostAutosGEDrivers = 0; + + + public TourVehicleTypeChoiceModel(HashMap rbMap) + { + + logger.info("setting up tour tNCVehicle type choice model."); + probabilityBoostAutosLTDrivers = Util.getFloatValueFromPropertyMap(rbMap,"Mobility.AV.ProbabilityBoost.AutosLTDrivers"); + probabilityBoostAutosGEDrivers = Util.getFloatValueFromPropertyMap(rbMap,"Mobility.AV.ProbabilityBoost.AutosGEDrivers"); + + + } + + /** + * Calculate the probability of the tour using the AV in the household. If AVs owned =0, returns 0, else + * returns a probability equal to the share of AVs in the household, boosted by the parameters in the properties file. + * The parameters are named Mobility.AV.ProbabilityBoost.AutosLTDrivers and Mobility.AV.ProbabilityBoost.AutosGEDrivers + * and are read in the object constructor. + * + * @param hhObj + * @return The probability of using one of the household AVs for the tour. + */ + public double calculateProbability(Household hhObj){ + + float numberOfAVs = (float) hhObj.getAutomatedVehicles(); + + if(numberOfAVs==0) + return 0; + + float numberOfCVs = (float) hhObj.getConventionalVehicles(); + float numberOfDrivers = (float) hhObj.getDrivers(); + float totalVehicles = numberOfAVs + numberOfCVs; + float probability = numberOfAVs/totalVehicles; + if(totalVehicles + * Started: Apr 14, 2009 11:09:58 AM + */ +public class TransponderChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TransponderChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + + private Household hh; + + private double percentTazIncome100Kplus; + private double percentTazMultpleAutos; + private double expectedTravelTimeSavings; + private double transpDist; + private double pctDetour; + private double accessibility; + + public TransponderChoiceDMU() + { + dmuIndex = new IndexValues(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (hh.getDebugChoiceModels()) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug Free Parking UEC"); + } + } + + public void setHouseholdObject(Household hhObj) + { + hh = hhObj; + } + + public void setPctIncome100Kplus(double value) + { + percentTazIncome100Kplus = value; + } + + public void setPctTazMultpleAutos(double value) + { + percentTazMultpleAutos = value; + } + + public void setExpectedTravelTimeSavings(double value) + { + expectedTravelTimeSavings = value; + } + + public void setTransponderDistance(double value) + { + transpDist = value; + } + + public void setPctDetour(double value) + { + pctDetour = value; + } + + public void setAccessibility(double value) + { + accessibility = value; + } + + public double getPctIncome100Kplus() + { + return percentTazIncome100Kplus; + } + + public double getPctTazMultpleAutos() + { + return percentTazMultpleAutos; + } + + public double getExpectedTravelTimeSavings() + { + return expectedTravelTimeSavings; + } + + public double getTransponderDistance() + { + return transpDist; + } + + public double getPctDetour() + { + return pctDetour; + } + + public double getAccessibility() + { + return accessibility; + } + + public int getAutoOwnership() + { + return hh.getAutosOwned(); + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java new file mode 100644 index 0000000..35eda9f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java @@ -0,0 +1,131 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AccessibilitiesTable; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class TransponderChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("tp"); + + private static final String TP_CONTROL_FILE_TARGET = "tc.uec.file"; + private static final String TP_DATA_SHEET_TARGET = "tc.data.page"; + private static final String TP_MODEL_SHEET_TARGET = "tc.model.page"; + + public static final int TP_MODEL_NO_ALT = 1; + public static final int TP_MODEL_YES_ALT = 2; + + private ChoiceModelApplication tpModel; + private TransponderChoiceDMU tpDmuObject; + + private AccessibilitiesTable accTable; + + private double[] pctHighIncome; + private double[] pctMultipleAutos; + private double[] avgtts; + private double[] transpDist; + private double[] pctDetour; + + public TransponderChoiceModel(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable accTable, double[] pctHighIncome, + double[] pctMultipleAutos, double[] avgtts, double[] transpDist, double[] pctDetour) + { + this.accTable = accTable; + this.pctHighIncome = pctHighIncome; + this.pctMultipleAutos = pctMultipleAutos; + this.avgtts = avgtts; + this.transpDist = transpDist; + this.pctDetour = pctDetour; + + setupTransponderChoiceModelApplication(propertyMap, dmuFactory); + } + + private void setupTransponderChoiceModelApplication(HashMap propertyMap, + CtrampDmuFactoryIf dmuFactory) + { + logger.info("setting up transponder choice model."); + + // locate the transponder choice UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tpUecFile = uecFileDirectory + propertyMap.get(TP_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TP_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TP_MODEL_SHEET_TARGET); + + // create the transponder choice model DMU object. + tpDmuObject = dmuFactory.getTransponderChoiceDMU(); + + // create the transponder choice model object + tpModel = new ChoiceModelApplication(tpUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) tpDmuObject); + + } + + public void applyModel(Household hhObject) + { + + int homeTaz = hhObject.getHhTaz(); + + tpDmuObject.setHouseholdObject(hhObject); + + // set the zone, orig and dest attributes + tpDmuObject.setDmuIndexValues(hhObject.getHhId(), hhObject.getHhTaz(), hhObject.getHhTaz(), + 0); + + tpDmuObject.setPctIncome100Kplus(pctHighIncome[homeTaz]); + tpDmuObject.setPctTazMultpleAutos(pctMultipleAutos[homeTaz]); + tpDmuObject.setExpectedTravelTimeSavings(avgtts[homeTaz]); + tpDmuObject.setTransponderDistance(transpDist[homeTaz]); + tpDmuObject.setPctDetour(pctDetour[homeTaz]); + + float accessibility = accTable.getAggregateAccessibility("transit", hhObject.getHhMgra()); + tpDmuObject.setAccessibility(accessibility); + + Random hhRandom = hhObject.getHhRandom(); + double randomNumber = hhRandom.nextDouble(); + + // compute utilities and choose transponder choice alternative. + float logsum = (float) tpModel.computeUtilities(tpDmuObject, tpDmuObject.getDmuIndexValues()); + + hhObject.setTransponderLogsum(logsum); + + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + if (tpModel.getAvailabilityCount() > 0) + { + chosenAlt = tpModel.getChoiceResult(randomNumber); + } else + { + String decisionMaker = String.format("HHID=%d", hhObject.getHhId()); + String errorMessage = String + .format("Exception caught for %s, no available transponder choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + tpModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + // write choice model alternative info to log file + if (hhObject.getDebugChoiceModels()) + { + String decisionMaker = String.format("HHID=%d", hhObject.getHhId()); + tpModel.logAlternativesInfo("Transponder Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d with rn %.8f", + "Transponder Choice", decisionMaker, chosenAlt, randomNumber)); + tpModel.logUECResults(logger, decisionMaker); + } + + hhObject.setTpChoice(chosenAlt - 1); + + hhObject.setTpRandomCount(hhObject.getHhRandomCount()); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java new file mode 100644 index 0000000..b8ca3e7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java @@ -0,0 +1,846 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class TripModeChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TripModeChoiceDMU.class); + + protected static final int WTW = McLogsumsCalculator.WTW; + protected static final int WTD = McLogsumsCalculator.WTD; + protected static final int DTW = McLogsumsCalculator.DTW; + protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + protected HashMap methodIndexMap; + + protected Tour tour; + protected Person person; + protected Household hh; + protected IndexValues dmuIndex; + + protected double nmWalkTime; + protected double nmBikeTime; + + protected ModelStructure modelStructure; + + protected double origDuDen; + protected double origEmpDen; + protected double origTotInt; + protected double destDuDen; + protected double destEmpDen; + protected double destTotInt; + + protected int tripOrigIsTourDest; + protected int tripDestIsTourDest; + + protected int tripTime; + protected int firstTrip; + protected int lastTrip; + protected int outboundStops; + protected int inboundStops; + + protected int incomeInDollars; + protected int age; + protected int adults; + protected int autos; + protected int hhSize; + protected int personIsFemale; + + protected int departPeriod; + protected int arrivePeriod; + protected int tripPeriod; + + protected int escortTour; + protected int jointTour; + protected int partySize; + + protected int outboundHalfTourDirection; + protected float waitTimeTaxi; + protected float waitTimeSingleTNC; + protected float waitTimeSharedTNC; + + protected int tourModeIsDA; + protected int tourModeIsS2; + protected int tourModeIsS3; + protected int tourModeIsWalk; + protected int tourModeIsBike; + protected int tourModeIsWTran; + protected int tourModeIsPnr; + protected int tourModeIsKnr; + protected int tourModeIsSchBus; + + protected double reimburseAmount; + + protected float pTazTerminalTime; + protected float aTazTerminalTime; + + protected int[] mgraParkArea; + + protected double[] lsWgtAvgCostM; + protected double[] lsWgtAvgCostD; + protected double[] lsWgtAvgCostH; + + protected boolean segmentIsIk; + protected boolean autoModeRequiredForDriveTransit; + protected boolean walkModeAllowedForDriveTransit; + + protected double ivtCoeff; + protected double costCoeff; + + protected double[] transitLogSum; + + protected boolean inbound; + + protected int originMgra; + protected int destMgra; + + + public TripModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) + { + this.modelStructure = modelStructure; + dmuIndex = new IndexValues(); + + transitLogSum = new double[McLogsumsCalculator.NUM_ACC_EGR]; + } + + + + public void setParkingCostInfo(int[] mgraParkArea, double[] lsWgtAvgCostM, + double[] lsWgtAvgCostD, double[] lsWgtAvgCostH) + { + this.mgraParkArea = mgraParkArea; + this.lsWgtAvgCostM = lsWgtAvgCostM; + this.lsWgtAvgCostD = lsWgtAvgCostD; + this.lsWgtAvgCostH = lsWgtAvgCostH; + } + + public void setHouseholdObject(Household hhObject) + { + hh = hhObject; + } + + public Household getHouseholdObject() + { + return hh; + } + + public void setPersonObject(Person personObject) + { + person = personObject; + } + + public Person getPersonObject() + { + return person; + } + + public void setTourObject(Tour tourObject) + { + tour = tourObject; + } + + public Tour getTourObject() + { + return tour; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public float getTimeOutbound() + { + return tour.getTourDepartPeriod(); + } + + public float getTimeInbound() + { + return tour.getTourArrivePeriod(); + } + + public void setSegmentIsIk(boolean flag) + { + segmentIsIk = flag; + } + + public int getSegmentIsIk() + { + return segmentIsIk ? 1 : 0; + } + + public void setIncomeInDollars(int arg) + { + incomeInDollars = arg; + } + + public void setAutos(int arg) + { + autos = arg; + } + + public void setAdults(int arg) + { + adults = arg; + } + + public void setHhSize(int arg) + { + hhSize = arg; + } + + public void setAge(int arg) + { + age = arg; + } + + public void setPersonIsFemale(int arg) + { + personIsFemale = arg; + } + + public void setEscortTour(int arg) + { + escortTour = arg; + } + + public void setJointTour(int arg) + { + jointTour = arg; + } + + public void setPartySize(int arg) + { + partySize = arg; + } + + public void setOutboundHalfTourDirection(int arg) + { + outboundHalfTourDirection = arg; + } + + public void setDepartPeriod(int period) + { + departPeriod = period; + } + + public void setArrivePeriod(int period) + { + arrivePeriod = period; + } + + public void setTripPeriod(int period) + { + tripPeriod = period; + } + + public void setOutboundStops(int stops) + { + outboundStops = stops; + } + + public void setInboundStops(int stops) + { + inboundStops = stops; + } + + public void setFirstTrip(int trip) + { + firstTrip = trip; + } + + public void setLastTrip(int trip) + { + lastTrip = trip; + } + + public void setTourModeIsDA(int arg) + { + tourModeIsDA = arg; + } + + public void setTourModeIsS2(int arg) + { + tourModeIsS2 = arg; + } + + public void setTourModeIsS3(int arg) + { + tourModeIsS3 = arg; + } + + public void setTourModeIsWalk(int arg) + { + tourModeIsWalk = arg; + } + + public void setTourModeIsBike(int arg) + { + tourModeIsBike = arg; + } + + public void setTourModeIsWTran(int arg) + { + tourModeIsWTran = arg; + } + + public void setTourModeIsPnr(int arg) + { + tourModeIsPnr = arg; + } + + public void setTourModeIsKnr(int arg) + { + tourModeIsKnr = arg; + } + + public void setTourModeIsSchBus(int arg) + { + tourModeIsSchBus = arg; + } + + public void setOrigDuDen(double arg) + { + origDuDen = arg; + } + + public void setOrigEmpDen(double arg) + { + origEmpDen = arg; + } + + public void setOrigTotInt(double arg) + { + origTotInt = arg; + } + + public void setDestDuDen(double arg) + { + destDuDen = arg; + } + + public void setDestEmpDen(double arg) + { + destEmpDen = arg; + } + + public void setDestTotInt(double arg) + { + destTotInt = arg; + } + + public void setReimburseProportion(double prop) + { + reimburseAmount = prop; + } + + public void setPTazTerminalTime(float time) + { + pTazTerminalTime = time; + } + + public void setATazTerminalTime(float time) + { + aTazTerminalTime = time; + } + + public void setTripOrigIsTourDest(int value) + { + tripOrigIsTourDest = value; + } + + public void setTripDestIsTourDest(int value) + { + tripDestIsTourDest = value; + } + + public void setBikeLogsum(int origin, int dest, boolean inbound) { + //do nothing - this is a stub to allow SANDAG to work correctly + // see SandagTripModeChoiceModelDMU for actual implementation + } + + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public void setAutoModeRequiredForTripSegment(boolean flag) + { + autoModeRequiredForDriveTransit = flag; + } + + public void setWalkModeAllowedForTripSegment(boolean flag) + { + walkModeAllowedForDriveTransit = flag; + } + + public void setIndexDest(int d) + { + dmuIndex.setDestZone(d); + } + + public void setNonMotorizedWalkTime(double walkTime) + { + nmWalkTime = walkTime; + } + + public void setNonMotorizedBikeTime(double bikeTime) + { + nmBikeTime = bikeTime; + } + + public int getAutoModeAllowedForTripSegment() + { + return autoModeRequiredForDriveTransit ? 1 : 0; + } + + public int getWalkModeAllowedForTripSegment() + { + return walkModeAllowedForDriveTransit ? 1 : 0; + } + + public int getTourModeIsDA() + { + boolean tourModeIsDa = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); + return tourModeIsDa ? 1 : 0; + } + + public int getTourModeIsS2() + { + boolean tourModeIsS2 = modelStructure.getTourModeIsS2(tour.getTourModeChoice()); + return tourModeIsS2 ? 1 : 0; + } + + public int getTourModeIsS3() + { + boolean tourModeIsS3 = modelStructure.getTourModeIsS3(tour.getTourModeChoice()); + return tourModeIsS3 ? 1 : 0; + } + + public int getTourModeIsSchBus() + { + boolean tourModeIsSchBus = modelStructure.getTourModeIsSchoolBus(tour.getTourModeChoice()); + return tourModeIsSchBus ? 1 : 0; + } + + public int getTourModeIsWalk() + { + boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tour.getTourModeChoice()); + return tourModeIsWalk ? 1 : 0; + } + + public int getTourModeIsBike() + { + boolean tourModeIsBike = modelStructure.getTourModeIsBike(tour.getTourModeChoice()); + return tourModeIsBike ? 1 : 0; + } + + public int getTourModeIsWTran() + { + boolean tourModeIsWTran = modelStructure.getTourModeIsWalkTransit(tour.getTourModeChoice()); + return tourModeIsWTran ? 1 : 0; + } + + public int getTourModeIsPnr() + { + boolean tourModeIsPnr = modelStructure.getTourModeIsPnr(tour.getTourModeChoice()); + return tourModeIsPnr ? 1 : 0; + } + + public int getTourModeIsKnr() + { + boolean tourModeIsKnr = modelStructure.getTourModeIsKnr(tour.getTourModeChoice()); + return tourModeIsKnr ? 1 : 0; + } + + public int getTourModeIsTncTransit() + { + boolean tourModeIsTncTransit = modelStructure.getTourModeIsTncTransit(tour.getTourModeChoice()); + return tourModeIsTncTransit ? 1 : 0; + } + + public int getTourModeIsMaas() + { + boolean tourModeIsMaas = modelStructure.getTourModeIsMaas(tour.getTourModeChoice()); + return tourModeIsMaas ? 1 : 0; + } + + public void setTransitLogSum(int accEgr, double value){ + transitLogSum[accEgr] = value; + } + + public double getTransitLogSum(int accEgr){ + return transitLogSum[accEgr]; + } + + + public double getODUDen() + { + return origDuDen; + } + + public double getOEmpDen() + { + return origEmpDen; + } + + public double getOTotInt() + { + return origTotInt; + } + + public double getDDUDen() + { + return destDuDen; + } + + public double getDEmpDen() + { + return destEmpDen; + } + + public double getDTotInt() + { + return destTotInt; + } + + public int getFirstTrip() + { + return firstTrip; + } + + public int getLastTrip() + { + return lastTrip; + } + + public int getTourCategoryJoint() + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return 1; + else return 0; + } + + public int getTourCategoryEscort() + { + if (tour.getTourPrimaryPurpose().equalsIgnoreCase( + ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) return 1; + else return 0; + } + + public int getTourCategorySubtour() + { + if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) return 1; + else return 0; + } + + public int getNumberOfParticipantsInJointTour() + { + int[] participants = tour.getPersonNumArray(); + int returnValue = 0; + if (participants != null) returnValue = participants.length; + return returnValue; + } + + public int getHhSize() + { + return hh.getHhSize(); + } + + public int getAutos() + { + return hh.getAutosOwned(); } + + public int getAge() + { + return person.getAge(); + } + + public int getFemale() + { + return person.getPersonIsFemale(); + } + + public int getIncomeCategory() + { + return hh.getIncomeCategory(); + } + + public double getNm_walkTime() + { + return nmWalkTime; + } + + public double getNm_bikeTime() + { + return nmBikeTime; + } + + public double getReimburseAmount() + { + return reimburseAmount; + } + + public double getMonthlyParkingCostTourDest() + { + return lsWgtAvgCostM[tour.getTourDestMgra()]; + } + + public double getDailyParkingCostTourDest() + { + return lsWgtAvgCostD[tour.getTourDestMgra()]; + } + + public double getHourlyParkingCostTourDest() + { + return lsWgtAvgCostH[tour.getTourDestMgra()]; + } + + public double getHourlyParkingCostTripOrig() + { + return lsWgtAvgCostH[originMgra]; + } + + public double getHourlyParkingCostTripDest() + { + return lsWgtAvgCostH[destMgra]; + } + + public int getTripOrigIsTourDest() + { + return tripOrigIsTourDest; + } + + public int getTripDestIsTourDest() + { + return tripDestIsTourDest; + } + + public void setOriginMgra( int value ) { + originMgra = value; + } + + public void setDestMgra( int value ) { + destMgra = value; + } + + public int getFreeOnsite() + { + return person.getFreeParkingAvailableResult() == ParkingProvisionModel.FP_MODEL_FREE_ALT ? 1 + : 0; + } + + public int getPersonType() + { + return person.getPersonTypeNumber(); + } + + public double getWorkTimeFactor() { + return person.getTimeFactorWork(); + } + + public double getNonWorkTimeFactor(){ + return person.getTimeFactorNonWork(); + } + + /** + * Iterate through persons on tour and return non-work time factor + * for oldest person. If the person array is null then return 1.0. + * + * @return Time factor for oldest person on joint tour. + */ + public double getJointTourTimeFactor() { + int[] personNumArray = tour.getPersonNumArray(); + int oldestAge = -999; + Person oldestPerson = null; + for (int num : personNumArray){ + Person p = hh.getPerson(num); + if(p.getAge() > oldestAge){ + oldestPerson = p; + oldestAge = p.getAge(); + } + } + if(oldestPerson != null) + return oldestPerson.getTimeFactorNonWork(); + + return 1.0; + } + + public double getPTazTerminalTime() + { + return pTazTerminalTime; + } + + public double getATazTerminalTime() + { + return aTazTerminalTime; + } + + /** + * @return the originMgra + */ + public int getOriginMgra() { + return originMgra; + } + + /** + * @return the destMgra + */ + public int getDestMgra() { + return destMgra; + } + + public boolean isInbound() { + return inbound; + } + + public int getInbound() { + return inbound ? 1 : 0 ; + } + + + public void setInbound(boolean inbound) { + this.inbound = inbound; + } + + /** + * 1 if household owns transponder, else 0 + * @return 1 if household owns transponder, else 0 + */ + public int getTransponderOwnership(){ + return hh.getTpChoice(); + } + + + + + public double getIvtCoeff() { + return ivtCoeff; + } + + + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + + + public double getCostCoeff() { + return costCoeff; + } + + + + public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; + } + + public int getIncomeInDollars() + { + return hh.getIncomeInDollars(); + } + + public int getUseOwnedAV(){ + + if(tour==null) + return 0; + + return (tour.getUseOwnedAV() ? 1: 0); + } + + + + + public float getWaitTimeTaxi() { + return waitTimeTaxi; + } + + public void setWaitTimeTaxi(float waitTimeTaxi) { + this.waitTimeTaxi = waitTimeTaxi; + } + + public float getWaitTimeSingleTNC() { + return waitTimeSingleTNC; + } + + public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { + this.waitTimeSingleTNC = waitTimeSingleTNC; + } + + public float getWaitTimeSharedTNC() { + return waitTimeSharedTNC; + } + + public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { + this.waitTimeSharedTNC = waitTimeSharedTNC; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java new file mode 100644 index 0000000..5b04c22 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java @@ -0,0 +1,1003 @@ +package org.sandag.abm.ctramp; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.MissingResourceException; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; +import org.jppf.client.JPPFClient; +import org.jppf.client.JPPFJob; +import org.jppf.node.protocol.DataProvider; +import org.jppf.node.protocol.JPPFTask; +import org.jppf.node.protocol.MemoryMapDataProvider; +import org.jppf.node.protocol.Task; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; + +public class UsualWorkSchoolLocationChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(UsualWorkSchoolLocationChoiceModel.class); + + private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "uwsl.use.new.soa"; + + private static final String PROPERTIES_DC_SOA_WORK_SAMPLE_SIZE = "uwsl.work.soa.SampleSize"; + private static final String PROPERTIES_DC_SOA_SCHOOL_SAMPLE_SIZE = "uwsl.school.soa.SampleSize"; + private static final String PROPERTIES_UEC_USUAL_LOCATION = "uwsl.dc.uec.file"; + private static final String PROPERTIES_UEC_USUAL_LOCATION_NEW = "uwsl.dc2.uec.file"; + private static final String PROPERTIES_UEC_USUAL_LOCATION_SOA = "uwsl.soa.uec.file"; + + private static final String PROPERTIES_RESULTS_WORK_SCHOOL_LOCATION_CHOICE = "Results.UsualWorkAndSchoolLocationChoice"; + + private static final String PROPERTIES_WORK_SCHOOL_LOCATION_CHOICE_PACKET_SIZE = "distributed.task.packet.size"; + + private static final String WORK_SCHOOL_SEGMENTS_FILE_NAME = "workSchoolSegments.definitions"; + + private static final int[] WORK_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX = {1, 1, 1, + 1, 1, 1 }; + private static final int[] WORK_LOC_SEGMENT_TO_UEC_SHEET_INDEX = {2, 2, 2, + 2, 2, 2 }; + private static int PACKET_SIZE = 0; + + // TODO: see if we can eliminate the setup synchronization issues - + // otherwise the + // number of these small + // packets can be fine-tuned and set in properties file.. + + // The number of initialization packets are the number of "small" packets + // submited at the beginning of a + // distributed task to minimize synchronization issues that significantly + // slow + // down model object setup. + // It is assumed that after theses small packets have run, all the model + // objects + // will have been setup, + // and the task objects can process much bigger chuncks of households. + private static String PROPERTIES_NUM_INITIALIZATION_PACKETS = "number.initialization.packets"; + private static String PROPERTIES_INITIALIZATION_PACKET_SIZE = "initialization.packet.size"; + private static int NUM_INITIALIZATION_PACKETS = 0; + private static int INITIALIZATION_PACKET_SIZE = 0; + + private static final int NUM_WRITE_PACKETS = 1000; + + private String wsLocResultsFileName; + + private transient ResourceBundle resourceBundle; + + private MgraDataManager mgraManager; + private TazDataManager tdm; + + private int maxTaz; + + private MatrixDataServerIf ms; + private ModelStructure modelStructure; + private CtrampDmuFactoryIf dmuFactory; + + private String workLocUecFileName; + private String schoolLocUecFileName; + private String soaUecFileName; + private int soaWorkSampleSize; + private int soaSchoolSampleSize; + + private HashSet skipSegmentIndexSet; + + private BuildAccessibilities aggAcc; + private DestChoiceSize workerDcSizeObj; + private DestChoiceSize schoolDcSizeObj; + + private String restartModelString; + + private JPPFClient jppfClient = null; + + private boolean useNewSoaMethod; + + public UsualWorkSchoolLocationChoiceModel(ResourceBundle resourceBundle, + String restartModelString, JPPFClient jppfClient, ModelStructure modelStructure, + MatrixDataServerIf ms, CtrampDmuFactoryIf dmuFactory, BuildAccessibilities aggAcc) + { + + // set the local variables + this.resourceBundle = resourceBundle; + this.modelStructure = modelStructure; + this.dmuFactory = dmuFactory; + this.ms = ms; + this.jppfClient = jppfClient; + this.restartModelString = restartModelString; + this.aggAcc = aggAcc; + + try + { + PACKET_SIZE = Integer.parseInt(resourceBundle + .getString(PROPERTIES_WORK_SCHOOL_LOCATION_CHOICE_PACKET_SIZE)); + } catch (MissingResourceException e) + { + PACKET_SIZE = 0; + } + + try + { + NUM_INITIALIZATION_PACKETS = Integer.parseInt(resourceBundle + .getString(PROPERTIES_NUM_INITIALIZATION_PACKETS)); + } catch (MissingResourceException e) + { + NUM_INITIALIZATION_PACKETS = 0; + } + + try + { + INITIALIZATION_PACKET_SIZE = Integer.parseInt(resourceBundle + .getString(PROPERTIES_INITIALIZATION_PACKET_SIZE)); + } catch (MissingResourceException e) + { + INITIALIZATION_PACKET_SIZE = 0; + } + + try + { + wsLocResultsFileName = resourceBundle + .getString(PROPERTIES_RESULTS_WORK_SCHOOL_LOCATION_CHOICE); + } catch (MissingResourceException e) + { + wsLocResultsFileName = null; + } + + String uecPath = ResourceUtil.getProperty(resourceBundle, + CtrampApplication.PROPERTIES_UEC_PATH); + + // get the sample-of-alternatives sample size + soaWorkSampleSize = ResourceUtil.getIntegerProperty(resourceBundle, + PROPERTIES_DC_SOA_WORK_SAMPLE_SIZE); + soaSchoolSampleSize = ResourceUtil.getIntegerProperty(resourceBundle, + PROPERTIES_DC_SOA_SCHOOL_SAMPLE_SIZE); + + useNewSoaMethod = ResourceUtil.getBooleanProperty(resourceBundle, + USE_NEW_SOA_METHOD_PROPERTY_KEY); + + // locate the UECs for destination choice, sample of alts, and mode + // choice + String usualWorkLocationUecFileName; + String usualSchoolLocationUecFileName; + if (useNewSoaMethod) + { + usualWorkLocationUecFileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_UEC_USUAL_LOCATION_NEW); + usualSchoolLocationUecFileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_UEC_USUAL_LOCATION_NEW); + } else + { + usualWorkLocationUecFileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_UEC_USUAL_LOCATION); + usualSchoolLocationUecFileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_UEC_USUAL_LOCATION); + } + workLocUecFileName = uecPath + usualWorkLocationUecFileName; + schoolLocUecFileName = uecPath + usualSchoolLocationUecFileName; + + String usualLocationSoaUecFileName = ResourceUtil.getProperty(resourceBundle, + PROPERTIES_UEC_USUAL_LOCATION_SOA); + soaUecFileName = uecPath + usualLocationSoaUecFileName; + + mgraManager = MgraDataManager.getInstance(); + + } + + public void runWorkLocationChoiceModel(HouseholdDataManagerIf householdDataManager, + double[][] workerSizeTerms) + { + + HashMap propertyMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + + // get the map of size term segment values to names + HashMap occupValueIndexMap = aggAcc.getWorkOccupValueIndexMap(); + + HashMap workSegmentIndexNameMap = aggAcc.getWorkSegmentIndexNameMap(); + HashMap workSegmentNameIndexMap = aggAcc.getWorkSegmentNameIndexMap(); + + int maxShadowPriceIterations = Integer.parseInt(propertyMap + .get(DestChoiceSize.PROPERTIES_WORK_DC_SHADOW_NITER)); + + // create an object for calculating destination choice attraction size + // terms + // and managing shadow price calculations. + workerDcSizeObj = new DestChoiceSize(propertyMap, workSegmentIndexNameMap, + workSegmentNameIndexMap, workerSizeTerms, maxShadowPriceIterations); + + int[][] originLocationsByHomeMgra = householdDataManager + .getWorkersByHomeMgra(occupValueIndexMap); + + // balance the size variables + workerDcSizeObj.balanceSizeVariables(originLocationsByHomeMgra); + + if (PACKET_SIZE == 0) PACKET_SIZE = householdDataManager.getNumHouseholds(); + + int currentIter = 0; + String fileName = propertyMap + .get(CtrampApplication.PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); + if (fileName != null) + { + if (fileName.length() > 2) + { + String projectDirectory = ResourceUtil.getProperty(resourceBundle, + CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + workerDcSizeObj.restoreShadowPricingInfo(projectDirectory + fileName); + int underScoreIndex = fileName.lastIndexOf('_'); + int dotIndex = fileName.lastIndexOf('.'); + currentIter = Integer.parseInt(fileName.substring(underScoreIndex + 1, dotIndex)); + currentIter++; + } + } + + // String restartFlag = propertyMap.get( + // CtrampApplication.PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); + // if ( restartFlag == null ) + // restartFlag = "none"; + // if ( restartFlag.equalsIgnoreCase("none") ) + // currentIter = 0; + + long initTime = System.currentTimeMillis(); + + // shadow pricing iterations + for (int iter = 0; iter < workerDcSizeObj.getMaxShadowPriceIterations(); iter++) + { + + logger.info("Work location choice shadow pricing iteration "+iter); + int innerLoop=0; + + while(true) { + ++innerLoop; + try + { + JPPFJob job = new JPPFJob(); + job.setName("Work Location Choice Job"); + + ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(householdDataManager + .getNumHouseholds()); + + DataProvider dataProvider = new MemoryMapDataProvider(); + dataProvider.setParameter("propertyMap", propertyMap); + dataProvider.setParameter("ms", ms); + dataProvider.setParameter("hhDataManager", householdDataManager); + dataProvider.setParameter("modelStructure", modelStructure); + dataProvider.setParameter("uecIndices", WORK_LOC_SEGMENT_TO_UEC_SHEET_INDEX); + dataProvider.setParameter("soaUecIndices", WORK_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX); + dataProvider.setParameter("tourCategory", ModelStructure.MANDATORY_CATEGORY); + dataProvider.setParameter("dcSizeObj", workerDcSizeObj); + dataProvider.setParameter("dcUecFileName", workLocUecFileName); + dataProvider.setParameter("soaUecFileName", soaUecFileName); + dataProvider.setParameter("soaSampleSize", soaWorkSampleSize); + dataProvider.setParameter("dmuFactory", dmuFactory); + dataProvider.setParameter("restartModelString", restartModelString); + + job.setDataProvider(dataProvider); + + int startIndex = 0; + int endIndex = 0; + int taskIndex = 1; + WorkLocationChoiceTaskJppf myTask = null; + WorkLocationChoiceTaskJppfNew myTaskNew = null; + for (int[] startEndIndices : startEndTaskIndicesList) + { + startIndex = startEndIndices[0]; + endIndex = startEndIndices[1]; + + if (innerLoop == 1) { + resetWorkLocationResults(startIndex,endIndex,householdDataManager); + } + + //check if there's work to do (JEF) + if(workLocationResultsOK(startIndex, endIndex, householdDataManager)) + continue; + + if (useNewSoaMethod) + { + myTaskNew = new WorkLocationChoiceTaskJppfNew(taskIndex, startIndex, + endIndex, iter); + job.add(myTaskNew); + } else + { + myTask = new WorkLocationChoiceTaskJppf(taskIndex, startIndex, endIndex, + iter); + job.add(myTask); + } + taskIndex++; + } + //nothing to do, so break out of this loop (JEF) + if(job.getJobTasks().isEmpty()) { + logger.info("Work location choice tasks completed successfully after "+(innerLoop-1)+" loops"); + break; + }else { + logger.info("Work location choice tasks need to be executed in loop "+innerLoop); + } + logger.info("Usual work location choice model submitting tasks to jppf job"); + List> results = jppfClient.submitJob(job); + for (Task task : results) + { + //if (task.getException() != null) throw task.getException(); + //wu modefied for jppf 6.1.4 + if (task.getThrowable() != null) { + Throwable t = task.getThrowable(); + t.printStackTrace(); + } + try + { + String stringResult = (String) task.getResult(); + logger.info(stringResult); + System.out.println(stringResult); + } catch (Exception e) + { + logger.error("", e); + throw new RuntimeException(); + } + + } + + } catch (Exception e) + { + e.printStackTrace(); + } + } + + // sum the chosen destinations by purpose, dest zone and subzone for + // shadow pricing adjustment + int[][] finalModeledDestChoiceLocationsByDestMgra = householdDataManager + .getWorkToursByDestMgra(occupValueIndexMap); + + int[] numChosenDests = new int[workSegmentNameIndexMap.size()]; + + for (int i = 0; i < numChosenDests.length; i++) + { + for (int j = 1; j <= mgraManager.getMaxMgra(); j++) + numChosenDests[i] += finalModeledDestChoiceLocationsByDestMgra[i][j]; + } + + logger.info(String + .format("Usual work location choice tasks completed for shadow price iteration %d in %d seconds.", + iter, ((System.currentTimeMillis() - initTime) / 1000))); + logger.info(String.format("Chosen dests by segment:")); + double total = 0; + for (int i = 0; i < numChosenDests.length; i++) + { + String segmentString = workSegmentIndexNameMap.get(i); + logger.info(String.format("\t%-8d%-15s = %10d", i + 1, segmentString, + numChosenDests[i])); + total += numChosenDests[i]; + } + logger.info(String.format("\t%-8s%-15s = %10.0f", "total", "", total)); + + // apply the shadow price adjustments + workerDcSizeObj.reportMaxDiff(iter, finalModeledDestChoiceLocationsByDestMgra); + workerDcSizeObj.saveWorkMaxDiffValues(iter, finalModeledDestChoiceLocationsByDestMgra); + workerDcSizeObj.updateShadowPrices(finalModeledDestChoiceLocationsByDestMgra); + workerDcSizeObj.updateSizeVariables(); + workerDcSizeObj.updateShadowPricingInfo(currentIter, originLocationsByHomeMgra, + finalModeledDestChoiceLocationsByDestMgra, "work"); + + householdDataManager.setUwslRandomCount(currentIter); + + currentIter++; + + } // iter + + logger.info("Usual work location choices computed in " + + ((System.currentTimeMillis() - initTime) / 1000) + " seconds."); + + } + + public void runSchoolLocationChoiceModel(HouseholdDataManagerIf householdDataManager, + double[][] schoolSizeTerms, double[][] schoolFactors) + { + + HashMap propertyMap = ResourceUtil + .changeResourceBundleIntoHashMap(resourceBundle); + + // get the maps of segment names and indices for school location choice + // size + HashMap schoolSegmentIndexNameMap = aggAcc.getSchoolSegmentIndexNameMap(); + HashMap schoolSegmentNameIndexMap = aggAcc.getSchoolSegmentNameIndexMap(); + + int maxShadowPriceIterations = Integer.parseInt(propertyMap + .get(DestChoiceSize.PROPERTIES_SCHOOL_DC_SHADOW_NITER)); + + // create an object for calculating destination choice attraction size + // terms + // and managing shadow price calculations. + schoolDcSizeObj = new DestChoiceSize(propertyMap, schoolSegmentIndexNameMap, + schoolSegmentNameIndexMap, schoolSizeTerms, maxShadowPriceIterations); + + // get the set of segment indices for which shadow pricing should be + // skipped. + skipSegmentIndexSet = aggAcc.getNoShadowPriceSchoolSegmentIndexSet(); + schoolDcSizeObj.setNoShadowPriceSchoolSegmentIndices(skipSegmentIndexSet); + + // set the school segment external factors calculated for university + // segment in the method that called this one. + schoolDcSizeObj.setExternalFactors(schoolFactors); + + householdDataManager.setSchoolDistrictMappings(schoolSegmentNameIndexMap, + aggAcc.getMgraGsDistrict(), aggAcc.getMgraHsDistrict(), + aggAcc.getGsDistrictIndexMap(), aggAcc.getHsDistrictIndexMap()); + + int[][] originLocationsByHomeMgra = householdDataManager.getStudentsByHomeMgra(); + + // balance the size variables + schoolDcSizeObj.balanceSizeVariables(originLocationsByHomeMgra); + + if (PACKET_SIZE == 0) PACKET_SIZE = householdDataManager.getNumHouseholds(); + + int currentIter = 0; + String fileName = propertyMap + .get(CtrampApplication.PROPERTIES_SCHOOL_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); + if (fileName != null) + if (fileName.length() > 2) + { + { + String projectDirectory = ResourceUtil.getProperty(resourceBundle, + CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + schoolDcSizeObj.restoreShadowPricingInfo(projectDirectory + fileName); + int underScoreIndex = fileName.lastIndexOf('_'); + int dotIndex = fileName.lastIndexOf('.'); + currentIter = Integer.parseInt(fileName + .substring(underScoreIndex + 1, dotIndex)); + currentIter++; + } + } + // String restartFlag = propertyMap.get( + // CtrampApplication.PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); + // if ( restartFlag == null ) + // restartFlag = "none"; + // if ( restartFlag.equalsIgnoreCase("none") ) + // currentIter = 0; + + long initTime = System.currentTimeMillis(); + + // shadow pricing iterations + for (int iter = 0; iter < schoolDcSizeObj.getMaxShadowPriceIterations(); iter++) + { + + // logger.info( String.format( "Size of Household[] in bytes = %d.", + // householdDataManager.getBytesUsedByHouseholdArray() ) ); + logger.info("School location choice shadow pricing iteration "+iter); + int innerLoop=0; + + while(true) { + ++innerLoop; + + try + { + JPPFJob job = new JPPFJob(); + job.setName("School Location Choice Job"); + + ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(householdDataManager + .getNumHouseholds()); + + DataProvider dataProvider = new MemoryMapDataProvider(); + dataProvider.setParameter("propertyMap", propertyMap); + dataProvider.setParameter("ms", ms); + dataProvider.setParameter("hhDataManager", householdDataManager); + dataProvider.setParameter("modelStructure", modelStructure); + dataProvider.setParameter("tourCategory", ModelStructure.MANDATORY_CATEGORY); + dataProvider.setParameter("dcSizeObj", schoolDcSizeObj); + dataProvider.setParameter("dcUecFileName", schoolLocUecFileName); + dataProvider.setParameter("soaUecFileName", soaUecFileName); + dataProvider.setParameter("soaSampleSize", soaSchoolSampleSize); + dataProvider.setParameter("dmuFactory", dmuFactory); + dataProvider.setParameter("restartModelString", restartModelString); + + job.setDataProvider(dataProvider); + + int startIndex = 0; + int endIndex = 0; + int taskIndex = 1; + SchoolLocationChoiceTaskJppf myTask = null; + SchoolLocationChoiceTaskJppfNew myTaskNew = null; + for (int[] startEndIndices : startEndTaskIndicesList) + { + startIndex = startEndIndices[0]; + endIndex = startEndIndices[1]; + + if (innerLoop == 1) { + resetSchoolLocationResults(startIndex,endIndex,householdDataManager); + } + //check if there's work to do (JEF) + if( schoolLocationResultsOK(startIndex, endIndex, householdDataManager)) + continue; + + if (useNewSoaMethod) + { + myTaskNew = new SchoolLocationChoiceTaskJppfNew(taskIndex, startIndex, + endIndex, iter); + job.add(myTaskNew); + } else + { + myTask = new SchoolLocationChoiceTaskJppf(taskIndex, startIndex, endIndex, + iter); + job.add(myTask); + } + taskIndex++; + } + + //nothing to do, so break out of this loop (JEF) + if(job.getJobTasks().isEmpty()) { + logger.info("School location choice tasks completed successfully after "+(innerLoop-1)+" loops"); + break; + }else { + logger.info("School location choice tasks need to be executed in loop "+innerLoop); + } + + List> results = jppfClient.submitJob(job); + for (Task task : results) + { + //if (task.getException() != null) throw task.getException(); + //wu modefied for jppf 6.1.4 + if (task.getThrowable() != null) { + Throwable t = task.getThrowable(); + t.printStackTrace(); + } + try + { + String stringResult = (String) task.getResult(); + logger.info(stringResult); + System.out.println(stringResult); + } catch (Exception e) + { + logger.error("", e); + throw new RuntimeException(); + } + + } + + } catch (Exception e) + { + e.printStackTrace(); + } + } + // sum the chosen destinations by purpose, dest zone and subzone for + // shadow pricing adjustment + int[][] finalModeledDestChoiceLocationsByDestMgra = householdDataManager + .getSchoolToursByDestMgra(); + + int[] numChosenDests = new int[schoolSegmentIndexNameMap.size()]; + + for (int i = 0; i < numChosenDests.length; i++) + { + for (int j = 1; j <= mgraManager.getMaxMgra(); j++) + numChosenDests[i] += finalModeledDestChoiceLocationsByDestMgra[i][j]; + } + + logger.info(String.format( + "Usual school location choice tasks completed for shadow price iteration %d.", + iter)); + logger.info(String.format("Chosen dests by segment:")); + double total = 0; + for (int i = 0; i < numChosenDests.length; i++) + { + String segmentString = schoolSegmentIndexNameMap.get(i); + logger.info(String.format("\t%-8d%-20s = %10d", i + 1, segmentString, + numChosenDests[i])); + total += numChosenDests[i]; + } + logger.info(String.format("\t%-8s%-20s = %10.0f", "total", "", total)); + + logger.info(String + .format("Usual school location choice tasks completed for shadow price iteration %d in %d seconds.", + iter, ((System.currentTimeMillis() - initTime) / 1000))); + + // apply the shadow price adjustments + schoolDcSizeObj.reportMaxDiff(iter, finalModeledDestChoiceLocationsByDestMgra); + schoolDcSizeObj + .saveSchoolMaxDiffValues(iter, finalModeledDestChoiceLocationsByDestMgra); + schoolDcSizeObj.updateShadowPrices(finalModeledDestChoiceLocationsByDestMgra); + schoolDcSizeObj.updateSizeVariables(); + schoolDcSizeObj.updateShadowPricingInfo(currentIter, originLocationsByHomeMgra, + finalModeledDestChoiceLocationsByDestMgra, "school"); + + householdDataManager.setUwslRandomCount(currentIter); + + currentIter++; + + } // iter + + logger.info("Usual school location choices computed in " + + ((System.currentTimeMillis() - initTime) / 1000) + " seconds."); + + } + + /** + * Loops through the households in the HouseholdDataManager, gets the + * households and persons and writes a row with detail on each of these in a + * file. + * + * @param householdDataManager + * is the object from which the array of household objects can be + * retrieved. + * @param projectDirectory + * is the root directory for the output file named + */ + public void saveResults(HouseholdDataManagerIf householdDataManager, String projectDirectory, + int globalIteration) + { + + HashMap workSegmentNameIndexMap = modelStructure + .getWorkSegmentNameIndexMap(); + HashMap schoolSegmentNameIndexMap = modelStructure + .getSchoolSegmentNameIndexMap(); + + FileWriter writer; + PrintWriter outStream = null; + + if (wsLocResultsFileName != null) + { + + // insert '_' and the global iteration number at end of filename or + // before '.' if there is a file extension in the name. + int dotIndex = wsLocResultsFileName.indexOf('.'); + if (dotIndex < 0) + { + wsLocResultsFileName = String + .format("%s_%d", wsLocResultsFileName, globalIteration); + } else + { + String base = wsLocResultsFileName.substring(0, dotIndex); + String extension = wsLocResultsFileName.substring(dotIndex); + wsLocResultsFileName = String.format("%s_%d%s", base, globalIteration, extension); + } + + wsLocResultsFileName = projectDirectory + wsLocResultsFileName; + + try + { + writer = new FileWriter(new File(wsLocResultsFileName)); + outStream = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format("Exception occurred opening wsLoc results file: %s.", + wsLocResultsFileName)); + throw new RuntimeException(e); + } + + // write header + outStream + .println("HHID,HomeMGRA,Income,PersonID,PersonNum,PersonType,PersonAge,EmploymentCategory,StudentCategory,WorkSegment,SchoolSegment,WorkLocation,WorkLocationDistance,WorkLocationLogsum,SchoolLocation,SchoolLocationDistance,SchoolLocationLogsum"); + + ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager + .getNumHouseholds()); + + for (int[] startEndIndices : startEndTaskIndicesList) + { + + int startIndex = startEndIndices[0]; + int endIndex = startEndIndices[1]; + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + for (int i = 0; i < householdArray.length; ++i) + { + + Household household = householdArray[i]; + + int hhId = household.getHhId(); + int homeMgra = household.getHhMgra(); + int income = household.getIncomeInDollars(); + + Person[] personArray = household.getPersons(); + + for (int j = 1; j < personArray.length; ++j) + { + + Person person = personArray[j]; + + int personId = person.getPersonId(); + int personNum = person.getPersonNum(); + int personType = person.getPersonTypeNumber(); + int personAge = person.getAge(); + int employmentCategory = person.getPersonEmploymentCategoryIndex(); + int studentCategory = person.getPersonStudentCategoryIndex(); + + int schoolSegmentIndex = person.getSchoolLocationSegmentIndex(); + int workSegmentIndex = person.getWorkLocationSegmentIndex(); + + int workLocation = person.getWorkLocation(); + int schoolLocation = person.getUsualSchoolLocation(); + + // write data record + outStream.println(String.format( + "%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%.5e,%.5e,%d,%.5e,%.5e", hhId, + homeMgra, income, personId, personNum, personType, personAge, + employmentCategory, studentCategory, workSegmentIndex, + schoolSegmentIndex, workLocation, person.getWorkLocationDistance(), + person.getWorkLocationLogsum(), schoolLocation, + person.getSchoolLocationDistance(), + person.getSchoolLocationLogsum())); + + } + + } + + } + + outStream.close(); + + } + + // save the mappings between segment index and segment labels to a file + // for + // workers and students + String fileName = projectDirectory + WORK_SCHOOL_SEGMENTS_FILE_NAME; + + try + { + writer = new FileWriter(new File(fileName)); + outStream = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) + { + logger.fatal(String.format( + "Exception occurred opening work/school segment definitions file: %s.", + fileName)); + throw new RuntimeException(e); + } + + outStream + .println("Correspondence table for work location segment indices and work location segment names"); + outStream.println(String.format("%-15s %-20s", "Index", "Segment Name")); + + String[] names = new String[workSegmentNameIndexMap.size() + 1]; + for (String key : workSegmentNameIndexMap.keySet()) + { + int index = workSegmentNameIndexMap.get(key); + names[index] = key; + } + + for (int i = 0; i < names.length; i++) + { + if (names[i] != null) outStream.println(String.format("%-15d %-20s", i, names[i])); + } + + outStream.println(""); + outStream.println(""); + outStream.println(""); + + outStream + .println("Correspondence table for school location segment indices and school location segment names"); + outStream.println(String.format("%-15s %-20s", "Index", "Segment Name")); + + names = new String[schoolSegmentNameIndexMap.size() + 1]; + for (String key : schoolSegmentNameIndexMap.keySet()) + { + int index = schoolSegmentNameIndexMap.get(key); + names[index] = key; + } + + for (int i = 0; i < names.length; i++) + { + if (names[i] != null) outStream.println(String.format("%-15d %-20s", i, names[i])); + } + + outStream.println(""); + + outStream.close(); + + } + + private ArrayList getTaskHouseholdRanges(int numberOfHouseholds) + { + + ArrayList startEndIndexList = new ArrayList(); + + int numInitializationHouseholds = NUM_INITIALIZATION_PACKETS * INITIALIZATION_PACKET_SIZE; + + int startIndex = 0; + int endIndex = 0; + if (numInitializationHouseholds < numberOfHouseholds) + { + + while (endIndex < numInitializationHouseholds) + { + endIndex = startIndex + INITIALIZATION_PACKET_SIZE - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += INITIALIZATION_PACKET_SIZE; + } + + } + + while (endIndex < numberOfHouseholds - 1) + { + endIndex = startIndex + PACKET_SIZE - 1; + if (endIndex + PACKET_SIZE > numberOfHouseholds) endIndex = numberOfHouseholds - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += PACKET_SIZE; + } + + return startEndIndexList; + + } + + private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) + { + + ArrayList startEndIndexList = new ArrayList(); + + int startIndex = 0; + int endIndex = 0; + + while (endIndex < numberOfHouseholds - 1) + { + endIndex = startIndex + NUM_WRITE_PACKETS - 1; + if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) + endIndex = numberOfHouseholds - 1; + + int[] startEndIndices = new int[2]; + startEndIndices[0] = startIndex; + startEndIndices[1] = endIndex; + startEndIndexList.add(startEndIndices); + + startIndex += NUM_WRITE_PACKETS; + } + + return startEndIndexList; + + } + + + /** + * Returns true if work location results are OK, else returns false. + * + * @param startIndex starting range of households + * @param endIndex ending range of households + * @param HouseholdDataManagerIf household data manager + * @return true or false + */ + public boolean workLocationResultsOK(int startIndex, int endIndex, HouseholdDataManagerIf householdDataManager) { + + + // get the array of households + Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); + + //iterate through hhs + for(Household thisHousehold:householdArray) { + + Person[] persons = thisHousehold.getPersons(); + + //iterate through persons + for(int k=1;k rbMap, String key) + { + boolean returnValue; + String value = rbMap.get(key); + if (value.equalsIgnoreCase("true") || value.equalsIgnoreCase("false")) + { + returnValue = Boolean.parseBoolean(value); + } else + { + logger.info("property file key: " + key + " = " + value + + " should be either 'true' or 'false'."); + throw new RuntimeException(); + } + + return returnValue; + } + + public static String getStringValueFromPropertyMap(HashMap rbMap, String key) + { + String returnValue = rbMap.get(key); + if (returnValue == null) returnValue = ""; + + return returnValue; + } + + public static String[] getStringArrayFromPropertyMap(HashMap rbMap, String key) { + String[] values = getStringValueFromPropertyMap(rbMap,key).split(","); + return values; + } + + + public static int getIntegerValueFromPropertyMap(HashMap rbMap, String key) + { + String value = rbMap.get(key); + if (value != null) + { + return Integer.parseInt(value); + } else + { + logger.info("property file key: " + key + + " missing. No integer value can be determined."); + throw new RuntimeException(); + } + } + + public static float getFloatValueFromPropertyMap(HashMap rbMap, String key) + { + String value = rbMap.get(key); + if (value != null) + { + return Float.parseFloat(value); + } else + { + logger.info("property file key: " + key + + " missing. No float value can be determined."); + throw new RuntimeException(); + } + } + + public static int[] getIntegerArrayFromPropertyMap(HashMap rbMap, String key) + { + + int[] returnArray; + String valueList = rbMap.get(key); + if (valueList != null) + { + + ArrayList valueSet = new ArrayList(); + + if (valueSet != null) + { + StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); + while (valueTokenizer.hasMoreTokens()) + { + String listValue = valueTokenizer.nextToken(); + int intValue = Integer.parseInt(listValue.trim()); + valueSet.add(intValue); + } + } + + returnArray = new int[valueSet.size()]; + int i = 0; + for (int v : valueSet) + returnArray[i++] = v; + + } else + { + logger.info("property file key: " + key + + " missing. No integer value can be determined."); + throw new RuntimeException(); + } + + return returnArray; + + } + + public static float[] getFloatArrayFromPropertyMap(HashMap rbMap, String key) + { + + float[] returnArray; + String valueList = rbMap.get(key); + if (valueList != null) + { + + ArrayList valueSet = new ArrayList(); + + StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); + while (valueTokenizer.hasMoreTokens()) + { + String listValue = valueTokenizer.nextToken(); + float floatValue = Float.parseFloat(listValue.trim()); + valueSet.add(floatValue); + } + + returnArray = new float[valueSet.size()]; + int i = 0; + for (float v : valueSet) + returnArray[i++] = v; + + } else + { + logger.info("property file key: " + key + + " missing. No float value can be determined."); + throw new RuntimeException(); + } + + return returnArray; + + } + + public static double[] getDoubleArrayFromPropertyMap(HashMap rbMap, String key) + { + + double[] returnArray; + String valueList = rbMap.get(key); + if (valueList != null) + { + + ArrayList valueSet = new ArrayList(); + + StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); + while (valueTokenizer.hasMoreTokens()) + { + String listValue = valueTokenizer.nextToken(); + double doubleValue = Double.parseDouble(listValue.trim()); + valueSet.add(doubleValue); + } + + returnArray = new double[valueSet.size()]; + int i = 0; + for (double v : valueSet) + returnArray[i++] = v; + + } else + { + logger.info("property file key: " + key + + " missing. No double value can be determined."); + throw new RuntimeException(); + } + + return returnArray; + + } + + /** + * + * @param cumProbabilities + * cumulative probabilities array + * @param entry + * target to search for in array + * @return the array index i where cumProbabilities[i] < entry and + * cumProbabilities[i-1] <= entry. + */ + public static int binarySearchDouble(double[] cumProbabilities, double entry) + { + + // lookup index for 0 <= entry < 1.0 in cumProbabilities + // cumProbabilities values are assumed to be in range: [0,1], and + // cumProbabilities[cumProbabilities.length-1] must equal 1.0 + + // if entry is outside the allowed range, return -1 + if (entry < 0 || entry >= 1.0) + { + System.out.println("entry = " + entry + + " is outside of allowable range for cumulative distribution [0,...,1.0)"); + return -1; + } + + // if cumProbabilities[cumProbabilities.length-1] is not equal to 1.0, + // return -1 + double epsilon = .0000001; + if (!(Math.abs(cumProbabilities[cumProbabilities.length - 1] - 1.0) < epsilon)) + { + System.out.println("cumProbabilities[cumProbabilities.length-1] = " + + cumProbabilities[cumProbabilities.length - 1] + " must equal 1.0"); + return -1; + } + + int hi = cumProbabilities.length; + int lo = 0; + int mid = (hi - lo) / 2; + + int safetyCount = 0; + + // if mid is 0, + if (mid == 0) + { + if (entry < cumProbabilities[0]) return 0; + else return 1; + } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) + { + return mid; + } + + while (true) + { + + if (entry < cumProbabilities[mid]) + { + hi = mid; + mid = (hi + lo) / 2; + } else + { + lo = mid; + mid = (hi + lo) / 2; + } + + // if mid is 0, + if (mid == 0) + { + if (entry < cumProbabilities[0]) return 0; + else return 1; + } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) + { + return mid; + } + + if (safetyCount++ > cumProbabilities.length) + { + logger.info("binary search stuck in the while loop"); + throw new RuntimeException("binary search stuck in the while loop"); + } + + } + + } + + /** + * + * @param cumProbabilities + * cumulative probabilities array + * @param numIndices + * are the number of probability values to consider in the + * cumulative probabilities array + * @param entry + * target to search for in array between indices 1 and numValues. + * @return the array index i where cumProbabilities[i] < entry and + * cumProbabilities[i-1] <= entry. + */ + public static int binarySearchDouble(double cumProbabilityLowerBound, + double[] cumProbabilities, int numIndices, double entry) + { + + // search for 0-based index i for cumProbabilities such that + // cumProbabilityLowerBound <= entry < cumProbabilities[0], i = 0; + // or + // cumProbabilities[i-1] <= entry < cumProbabilities[i], for i = + // 1,...numIndices-1; + + // if entry is outside the allowed range, return -1 + if (entry < cumProbabilityLowerBound || entry >= cumProbabilities[numIndices - 1]) + { + logger.info("entry = " + entry + + " is outside of allowable range of cumulative probabilities."); + logger.info("cumProbabilityLowerBound = " + cumProbabilityLowerBound + + ", cumProbabilities[numIndices-1] = " + cumProbabilities[numIndices - 1] + + ", numIndices = " + numIndices); + return -1; + } + + int hi = numIndices; + int lo = 0; + int mid = (hi - lo) / 2; + + int safetyCount = 0; + + // if mid is 0, + if (mid == 0) + { + if (entry < cumProbabilities[0]) return 0; + else return 1; + } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) + { + return mid; + } + + while (true) + { + + if (entry < cumProbabilities[mid]) + { + hi = mid; + mid = (hi + lo) / 2; + } else + { + lo = mid; + mid = (hi + lo) / 2; + } + + // if mid is 0, + if (mid == 0) + { + if (entry < cumProbabilities[0]) return 0; + else return 1; + } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) + { + return mid; + } + + if (safetyCount++ > numIndices) + { + logger.info("binary search stuck in the while loop"); + throw new RuntimeException("binary search stuck in the while loop"); + } + + } + + } + + /** + * REad a tabledataset from a CSV file and return it. + * + * @param fileName + * @return + */ + public static TableDataSet readTableDataSet(String fileName) { + + + TableDataSet tableData; + + try + { + CSVFileReader csvFile = new CSVFileReader(); + tableData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + logger.fatal("Error trying to read table data set from csv file: "+ fileName); + throw new RuntimeException(e); + } + + return tableData; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java new file mode 100644 index 0000000..a3f10a0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java @@ -0,0 +1,137 @@ +package org.sandag.abm.ctramp; + +import gnu.cajo.invoke.Remote; +import java.io.IOException; +import java.lang.reflect.InvocationTargetException; +import java.net.MalformedURLException; +import java.rmi.ConnectIOException; +import java.rmi.NotBoundException; +import java.rmi.RemoteException; +import org.apache.log4j.Logger; + +/** + * User: Jim Date: Jul 3, 2008 Time: 2:27:02 PM + * + * Utility class for applying remote methods of various types + * + */ + +public class UtilRmi + implements java.io.Serializable +{ + + private transient Logger logger = Logger.getLogger(UtilRmi.class); + private String connectString; + + private static int MAX_RETRY_COUNT = 100; + private static int MAX_RETRY_TIME = 1000; // milliseconds + + public UtilRmi(String connectString) + { + this.connectString = connectString; + } + + public Object method(String name, Object[] args) + { + + int connectExceptionCount = 0; + + Object itemObject = null; + Object returnObject = null; + + while (connectExceptionCount < MAX_RETRY_COUNT) + { + + try + { + itemObject = Remote.getItem(connectString); + break; + } catch (ConnectIOException e) + { + + try + { + Thread.currentThread().wait(MAX_RETRY_TIME); + } catch (InterruptedException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + connectExceptionCount++; + + } catch (RemoteException e) + { + logger.error("RemoteException exception making RMI method call: " + connectString + + "." + name + "().", e); + throw new RuntimeException(); + } catch (MalformedURLException e) + { + logger.error("MalformedURLException exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } catch (NotBoundException e) + { + logger.error("NotBoundException exception making RMI method call: " + connectString + + "." + name + "().", e); + throw new RuntimeException(); + } catch (IOException e) + { + logger.error("IOException exception making RMI method call: " + connectString + "." + + name + "().", e); + throw new RuntimeException(); + } catch (ClassNotFoundException e) + { + logger.error("ClassNotFoundException exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } catch (InstantiationException e) + { + logger.error("InstantiationException exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } catch (IllegalAccessException e) + { + logger.error("IllegalAccessException exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } catch (UnsatisfiedLinkError e) + { + logger.error("UnsatisfiedLinkError exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } + + } + + if (connectExceptionCount > 0) + { + logger.warn("UtilRmi.method() timed out " + connectExceptionCount + + "times connecting to: " + connectString + "." + name + "()."); + } + if (connectExceptionCount == MAX_RETRY_COUNT) + { + logger.error("UtilRmi.method() connection was never made."); + throw new RuntimeException(); + } + + try + { + returnObject = Remote.invoke(itemObject, name, args); + } catch (InvocationTargetException e) + { + logger.error("InvocationTargetException exception making RMI method call: " + + connectString + "." + name + "().", e); + throw new RuntimeException(); + } catch (Exception e) + { + logger.error("Exception exception making RMI method call: " + connectString + "." + + name + "().", e); + throw new RuntimeException(); + } + + return returnObject; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java new file mode 100644 index 0000000..18af3b5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java @@ -0,0 +1,692 @@ +package org.sandag.abm.ctramp; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Random; +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.modechoice.MgraDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class WorkLocationChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(MandatoryDestChoiceModel.class); + private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); + + // this constant used as a dimension for saving distance and logsums for + // alternatives in samples + private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; + + private static final int DC_DATA_SHEET = 0; + private static final int DC_WORK_AT_HOME_SHEET = 1; + + // private TazDataManager tazs; + private MgraDataManager mgraManager; + private DestChoiceSize dcSizeObj; + + private DestChoiceTwoStageModelDMU dcTwoStageDmuObject; + + private DestChoiceTwoStageModel dcTwoStageModelObject; + private TourModeChoiceModel mcModel; + + private String[] segmentNameList; + private HashMap workOccupValueSegmentIndexMap; + + private int[] dcModelIndices; + + // A ChoiceModelApplication object and modeAltsAvailable[] is needed for + // each purpose + private ChoiceModelApplication[] locationChoiceModels; + private ChoiceModelApplication locationChoiceModel; + private ChoiceModelApplication worksAtHomeModel; + + private boolean[] dcModelAltsAvailable; + private int[] dcModelAltsSample; + private int[] dcModelSampleValues; + + private int[] uecSheetIndices; + + int origMgra; + + private int modelIndex; + + private double[] sampleAlternativeDistances; + private double[] sampleAlternativeLogsums; + + private BuildAccessibilities aggAcc; + + private double[] mgraDistanceArray; + + private int soaSampleSize; + + private long soaRunTime; + + public WorkLocationChoiceModel(int index, HashMap propertyMap, + DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, String dcUecFileName, + String soaUecFile, int soaSampleSize, String modeChoiceUecFile, + CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel, double[][][] workSizeProbs, + double[][][] workTazDistProbs) + { + + this.aggAcc = aggAcc; + this.dcSizeObj = dcSizeObj; + this.mcModel = mcModel; + this.soaSampleSize = soaSampleSize; + + modelIndex = index; + + mgraManager = MgraDataManager.getInstance(); + + dcTwoStageDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); + dcTwoStageDmuObject.setAggAcc(this.aggAcc); + + dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); + dcTwoStageModelObject.setTazDistProbs(workTazDistProbs); + dcTwoStageModelObject.setMgraSizeProbs(workSizeProbs); + + sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; + + workOccupValueSegmentIndexMap = aggAcc.getWorkOccupValueIndexMap(); + + } + + public void setupWorkSegments(int[] myUecSheetIndices, int[] mySoaUecSheetIndices) + { + uecSheetIndices = myUecSheetIndices; + segmentNameList = aggAcc.getWorkSegmentNameList(); + } + + public void setupDestChoiceModelArrays(HashMap propertyMap, + String dcUecFileName, String soaUecFile, int soaSampleSize) + { + + // create the works-at-home ChoiceModelApplication object + worksAtHomeModel = new ChoiceModelApplication(dcUecFileName, DC_WORK_AT_HOME_SHEET, + DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); + + // create a lookup array to map purpose index to model index + dcModelIndices = new int[uecSheetIndices.length]; + + // get a set of unique model sheet numbers so that we can create + // ChoiceModelApplication objects once for each model sheet used + // also create a HashMap to relate size segment index to SOA Model + // objects + HashMap modelIndexMap = new HashMap(); + int dcModelIndex = 0; + int dcSegmentIndex = 0; + for (int uecIndex : uecSheetIndices) + { + // if the uec sheet for the model segment is not in the map, add it, + // otherwise, get it from the map + if (!modelIndexMap.containsKey(uecIndex)) + { + modelIndexMap.put(uecIndex, dcModelIndex); + dcModelIndices[dcSegmentIndex] = dcModelIndex++; + } else + { + dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); + } + + dcSegmentIndex++; + } + // the value of dcModelIndex is the number of ChoiceModelApplication + // objects to create + // the modelIndexMap keys are the uec sheets to use in building + // ChoiceModelApplication objects + + locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; + + int i = 0; + for (int uecIndex : modelIndexMap.keySet()) + { + + int modelIndex = -1; + try + { + modelIndex = modelIndexMap.get(uecIndex); + locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, + uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); + } catch (RuntimeException e) + { + logger.error(String + .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", + i, modelIndex, modelIndexMap.size())); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + } + + dcModelAltsAvailable = new boolean[soaSampleSize + 1]; + dcModelAltsSample = new int[soaSampleSize + 1]; + dcModelSampleValues = new int[soaSampleSize]; + + mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; + + } + + public boolean applyWorkLocationChoice(Household hh) + { + boolean result=true; + + if (hh.getDebugChoiceModels()) + { + String label = String.format("Pre Work Location Choice HHId=%d Object", hh.getHhId()); + hh.logHouseholdObject(label, dcManLogger); + } + + // declare these variables here so their values can be logged if a + // RuntimeException occurs. + int i = -1; + int occupSegmentIndex = -1; + int occup = -1; + String occupSegmentName = ""; + + int homeMgra = hh.getHhMgra(); + Person[] persons = hh.getPersons(); + + int tourNum = 0; + for (i = 1; i < persons.length; i++) + { + + Person p = persons[i]; + + // skip person if they are not a worker + if (p.getPersonIsWorker() != 1) + { + p.setWorkLocationSegmentIndex(-1); + p.setWorkLocation(0); + p.setWorkLocDistance(0); + p.setWorkLocLogsum(-999); + continue; + } + + // skip person if their work at home choice was work in the home + // (alternative 2 in choice model) + int worksAtHomeChoice = selectWorksAtHomeChoice(dcTwoStageDmuObject, hh, p); + if (worksAtHomeChoice == ModelStructure.WORKS_AT_HOME_ALTERNATUVE_INDEX) + { + p.setWorkLocationSegmentIndex(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); + p.setWorkLocation(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); + p.setWorkLocDistance(0); + p.setWorkLocLogsum(-999); + continue; + } + + // save person information in decision maker label, and log person + // object + if (hh.getDebugChoiceModels()) + { + String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p + .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); + hh.logPersonObject(decisionMakerLabel, dcManLogger, p); + } + + double[] results = null; + int modelIndex = 0; + try + { + + origMgra = homeMgra; + + occup = p.getPersPecasOccup(); + occupSegmentIndex = workOccupValueSegmentIndexMap.get(occup); + occupSegmentName = segmentNameList[occupSegmentIndex]; + + p.setWorkLocationSegmentIndex(occupSegmentIndex); + + // update the DC dmuObject for this person + dcTwoStageDmuObject.setHouseholdObject(hh); + dcTwoStageDmuObject.setPersonObject(p); + dcTwoStageDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); + + double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[occupSegmentIndex]; + mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, + mgraDistanceArray); + + // set size array for the tour segment and distance array from + // the home mgra to all destinaion mgras. + dcTwoStageDmuObject.setMgraSizeArray(homeMgraSizeArray); + dcTwoStageDmuObject.setMgraDistanceArray(mgraDistanceArray); + + modelIndex = dcModelIndices[occupSegmentIndex]; + locationChoiceModel = locationChoiceModels[modelIndex]; + + // get the work location alternative chosen from the sample + results = selectLocationFromSampleOfAlternatives("work", -1, p, occupSegmentName, + occupSegmentIndex, tourNum++, homeMgraSizeArray, mgraDistanceArray); + + soaRunTime += dcTwoStageModelObject.getSoaRunTime(); + + } catch (RuntimeException e) + { + logger.fatal(String + .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in work location choice, modelIndex=%d, occup=%d, segmentIndex=%d, segmentName=%s", + i, hh.getHhId(), i, modelIndex, occup, occupSegmentIndex, + occupSegmentName)); + logger.fatal("Exception caught:", e); + logger.fatal("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(); + } + + p.setWorkLocation((int) results[0]); + p.setWorkLocDistance((float) results[1]); + p.setWorkLocLogsum((float) results[2]); + + if(p.getWorkLocationSegmentIndex()>-1 && p.getWorkLocation()==0){ + logger.error("***********************************************************************************************************************************"); + logger.error("!!!!!!!Worker in worksegmetn "+p.getWorkLocationSegmentIndex()+" can't find a work location!!!!! RESTART Work location choice!!!!"); + result=false; + } + + } + return result; + + } + /** + * + * @return an array with chosen mgra, distance to chosen mgra, and logsum to + * chosen mgra. + */ + private double[] selectLocationFromSampleOfAlternatives(String segmentType, + int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, + int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcManLogger; + + Household household = person.getHouseholdObject(); + + // get sample of locations and correction factors for sample using the + // alternate method + // for work location, the sizeSegmentType INdex and sizeSegmentIndex are + // the same values. + dcTwoStageModelObject.chooseSample(household.getHhTaz(), sizeSegmentIndex, + sizeSegmentIndex, soaSampleSize, household.getHhRandom(), + household.getDebugChoiceModels()); + int[] finalSample = dcTwoStageModelObject.getUniqueSampleMgras(); + double[] sampleCorrectionFactors = dcTwoStageModelObject + .getUniqueSampleMgraCorrectionFactors(); + int numUniqueAlts = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); + + Arrays.fill(dcModelAltsAvailable, false); + Arrays.fill(dcModelAltsSample, 0); + Arrays.fill(dcModelSampleValues, 999999); + + // set sample of alternatives correction factors used in destination + // choice utility for the sampled alternatives. + dcTwoStageDmuObject.setDcSoaCorrections(sampleCorrectionFactors); + + // for the destination mgras in the sample, compute mc logsums and save + // in dmuObject. + // also save correction factor and set availability and sample value for + // the + // sample alternative to true. 1, respectively. + for (int i = 0; i < numUniqueAlts; i++) + { + + int destMgra = finalSample[i]; + dcModelSampleValues[i] = finalSample[i]; + + // set logsum value in DC dmuObject for the logsum index, sampled + // zone and subzone. + double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); + dcTwoStageDmuObject.setMcLogsum(i, logsum); + + sampleAlternativeLogsums[i] = logsum; + sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; + + // set availaibility and sample values for the purpose, dcAlt. + dcModelAltsAvailable[i + 1] = true; + dcModelAltsSample[i + 1] = 1; + + } + + dcTwoStageDmuObject.setSampleArray(dcModelSampleValues); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format( + "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", + person.getHouseholdObject().getHhId(), person.getPersonNum(), + person.getPersonType(), tourNum); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" + + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " + + person.getPersonType() + ", TourNum=" + tourNum); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + locationChoiceModel.computeUtilities(dcTwoStageDmuObject, + dcTwoStageDmuObject.getDmuIndexValues(), dcModelAltsAvailable, dcModelAltsSample); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (locationChoiceModel.getAvailabilityCount() > 0) + { + try + { + chosen = locationChoiceModel.getChoiceResult(rn); + } catch (Exception e) + { + } + } else + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", + dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject + .getPersonObject().getPersonNum(), segmentName)); + } + + if (household.getDebugChoiceModels() || chosen <= 0) + { + + double[] utilities = locationChoiceModel.getUtilities(); + double[] probabilities = locationChoiceModel.getProbabilities(); + boolean[] availabilities = locationChoiceModel.getAvailabilities(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < finalSample.length; j++) + { + int alt = finalSample[j]; + cumProb += probabilities[j]; + String altString = String.format("j=%d, mgra=%d", j, alt); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); + modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + + // write UEC calculation results to separate model specific log file + locationChoiceModel.logUECResults(modelLogger, loggingHeader); + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, workSegment=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", + dcTwoStageDmuObject.getHouseholdObject().getHhId(), + dcTwoStageDmuObject.getPersonObject().getPersonNum(), segmentName)); + throw new RuntimeException(); + } + + } + + double[] returnArray = new double[3]; + + returnArray[0] = finalSample[chosen - 1]; + returnArray[1] = sampleAlternativeDistances[chosen - 1]; + returnArray[2] = sampleAlternativeLogsums[chosen - 1]; + + return returnArray; + + } + + private int selectWorksAtHomeChoice(DestChoiceTwoStageModelDMU dcTwoStageDmuObject, + Household household, Person person) + { + + // set tour origin taz/subzone and start/end times for calculating mode + // choice logsum + Logger modelLogger = dcManLogger; + + dcTwoStageDmuObject.setHouseholdObject(household); + dcTwoStageDmuObject.setPersonObject(person); + dcTwoStageDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), origMgra, + 0); + + double accessibility = aggAcc.getAccessibilitiesTableObject().getAggregateAccessibility( + "totEmp", household.getHhMgra()); + dcTwoStageDmuObject.setWorkAccessibility(accessibility); + + // log headers to traceLogger if the person making the destination + // choice is + // from a household requesting trace information + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + if (household.getDebugChoiceModels()) + { + + // null tour means the DC is a mandatory usual location choice + choiceModelDescription = String.format("Usual Work Location Is At Home Choice Model"); + decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person + .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); + + modelLogger.info(" "); + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info("Usual Work Location Is At Home Choice Model: Person Num: " + + person.getPersonNum() + ", Person Type: " + person.getPersonType()); + + loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); + + worksAtHomeModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, + decisionMakerLabel); + + } + + // compute destination choice proportions and choose alternative + float logsum = (float) worksAtHomeModel.computeUtilities(dcTwoStageDmuObject, + dcTwoStageDmuObject.getDmuIndexValues()); + person.setWorksFromHomeLogsum(logsum); + + Random hhRandom = household.getHhRandom(); + int randomCount = household.getHhRandomCount(); + double rn = hhRandom.nextDouble(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen = -1; + if (worksAtHomeModel.getAvailabilityCount() > 0) + { + chosen = worksAtHomeModel.getChoiceResult(rn); + } + + // write choice model alternative info to log file + if (household.getDebugChoiceModels() || chosen < 0) + { + + double[] utilities = worksAtHomeModel.getUtilities(); + double[] probabilities = worksAtHomeModel.getProbabilities(); + boolean[] availabilities = worksAtHomeModel.getAvailabilities(); + + String[] altNames = worksAtHomeModel.getAlternativeNames(); + + String personTypeString = person.getPersonType(); + int personNum = person.getPersonNum(); + + modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); + modelLogger + .info("Alternative Availability Utility Probability CumProb"); + modelLogger + .info("--------------------- -------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int j = 0; j < utilities.length; j++) + { + cumProb += probabilities[j]; + String altString = String.format("%d, %s", j + 1, altNames[j]); + modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, + availabilities[j + 1], utilities[j], probabilities[j], cumProb)); + } + + modelLogger.info(" "); + String altString = String.format("j=%d, alt=%s", chosen, + (chosen < 0 ? "N/A, no available alternatives" : altNames[chosen - 1])); + modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, + rn, randomCount)); + + modelLogger + .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); + modelLogger.info(" "); + + worksAtHomeModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + worksAtHomeModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, + chosen); + + // write UEC calculation results to separate model specific log file + worksAtHomeModel.logUECResults(modelLogger, loggingHeader); + + } + + if (chosen < 0) + { + logger.error(String + .format("Exception caught for HHID=%d, PersonNum=%d, no available works at home alternatives to choose from in choiceModelApplication.", + dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject + .getPersonObject().getPersonNum())); + throw new RuntimeException(); + } + + return chosen; + + } + + private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, + int segmentTypeIndex) + { + + int purposeIndex = 0; + String purpose = ""; + if (segmentTypeIndex < 0) + { + purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) + { + purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) + { + purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; + purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; + } + + // create a temporary tour to use to calculate mode choice logsum + Tour mcLogsumTour = new Tour(person, 0, purposeIndex); + mcLogsumTour.setTourPurpose(purpose); + mcLogsumTour.setTourOrigMgra(household.getHhMgra()); + mcLogsumTour.setTourDestMgra(sampleDestMgra); + mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); + mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + + if (household.getDebugChoiceModels()) + { + dcManLogger.info(""); + dcManLogger.info(""); + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, + dcManLogger, person); + } + + double logsum = -1; + try + { + logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, + choiceModelDescription, decisionMakerLabel); + } catch (Exception e) + { + choiceModelDescription = "location choice logsum for segmentTypeIndex=" + + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; + decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " + + person.getPersonNum(); + logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); + logger.fatal("choiceModelDescription = " + choiceModelDescription); + logger.fatal("decisionMakerLabel = " + decisionMakerLabel); + e.printStackTrace(); + System.exit(-1); + } + + return logsum; + } + + public int getModelIndex() + { + return modelIndex; + } + + public void setDcSizeObject(DestChoiceSize dcSizeObj) + { + this.dcSizeObj = dcSizeObj; + } + + public long getSoaRunTime() + { + return soaRunTime; + } + + public void resetSoaRunTime() + { + soaRunTime = 0; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java new file mode 100644 index 0000000..99ccb09 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java @@ -0,0 +1,246 @@ +package org.sandag.abm.ctramp; + +import java.net.UnknownHostException; +import java.util.Date; +import java.util.HashMap; +import org.jppf.node.protocol.AbstractTask; +import org.jppf.node.protocol.DataProvider; +import com.pb.common.calculator.MatrixDataServerIf; + +import nl.tudelft.simulation.logger.Logger; + +public class WorkLocationChoiceTaskJppf + extends AbstractTask +{ + + private static String VERSION = "Task.1.0.3"; + + private transient HashMap propertyMap; + private transient MatrixDataServerIf ms; + private transient ModelStructure modelStructure; + private transient HouseholdDataManagerIf hhDataManager; + private transient String tourCategory; + private transient DestChoiceSize dcSizeObj; + private transient int[] uecIndices; + private transient int[] soaUecIndices; + private transient String dcUecFileName; + private transient String soaUecFileName; + private transient int soaSampleSize; + private transient CtrampDmuFactoryIf dmuFactory; + private transient String restartModelString; + + private int iteration; + private int startIndex; + private int endIndex; + private int taskIndex = -1; + + public WorkLocationChoiceTaskJppf(int taskIndex, int startIndex, int endIndex, int iteration) + { + this.startIndex = startIndex; + this.endIndex = endIndex; + this.taskIndex = taskIndex; + this.iteration = iteration; + } + + public void run() + { + + String start = (new Date()).toString(); + long startTime = System.currentTimeMillis(); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " + + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + try + { + DataProvider dataProvider = getDataProvider(); + + this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); + this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); + this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); + this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); + this.uecIndices = (int[]) dataProvider.getParameter("uecIndices"); + this.soaUecIndices = (int[]) dataProvider.getParameter("soaUecIndices"); + this.tourCategory = (String) dataProvider.getParameter("tourCategory"); + this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); + this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); + this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); + this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); + this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); + this.restartModelString = (String) dataProvider.getParameter("restartModelString"); + + } catch (Exception e) + { + e.printStackTrace(); + throw new RuntimeException(e); + } + + // get the factory object used to create and recycle dcModel objects. + DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); + + // one of tasks needs to initialize the manager object by passing + // attributes + // needed to create a destination choice model object. + modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, + soaSampleSize, dmuFactory, restartModelString); + + // get a dcModel object from manager, which either creates one or + // returns one + // for re-use. + MandatoryDestChoiceModel dcModel = modelManager.getDcWorkModelObject(taskIndex, iteration, + dcSizeObj, uecIndices, soaUecIndices); + + // logger.info( String.format( + // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, + // taskIndex, + // threadName, startIndex, endIndex ) ); + System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", + new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); + + long setup1 = (System.currentTimeMillis() - startTime) / 1000; + + Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); + + long setup2 = (System.currentTimeMillis() - startTime) / 1000; + // logger.info( String.format( + // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", + // taskIndex, startIndex, endIndex, + // threadName, setup1, setup2 ) ); + System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", + new Date(), taskIndex, startIndex, endIndex, threadName)); + + int i = -1; + try + { + + boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, + HouseholdDataManager.DEBUG_HHS_ONLY_KEY); + + for (i = 0; i < householdArray.length; i++) + { + // for debugging only - process only household objects specified + // for debugging, if property key was set to true + if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; + + dcModel.applyWorkLocationChoice(householdArray[i]); + } + + boolean worked=false; + int tries=0; + do{ + ++tries; + try{ + hhDataManager.setHhArray(householdArray, startIndex); + worked=true; + }catch(Exception e) { + System.out.println("Error trying to set households in hh manager for start index "+startIndex+" (tried "+tries+" times"); + if(tries<1000) + System.out.println("Trying again!"); + } + }while(!worked && (tries<1000)); + + //check to make sure hh array got set in hhDataManager + boolean allHouseholdsAreSame = false; + while(!allHouseholdsAreSame) { + Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); + for(int j = 0; j< householdArrayRemote.length;++j) { + + Household remoteHousehold = householdArrayRemote[j]; + Household localHousehold = householdArray[j]; + + allHouseholdsAreSame = checkIfSameWorkLocationResults(remoteHousehold, localHousehold); + + if(!allHouseholdsAreSame) + break; + } + if(!allHouseholdsAreSame) { + System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with work location choice results; updating"); + hhDataManager.setHhArray(householdArray, startIndex); + + } + } + + + } catch (Exception e) + { + if (i >= 0 && i < householdArray.length) System.out + .println(String + .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", + taskIndex, i, householdArray[i].getHhId(), startIndex)); + else System.out.println(String.format( + "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", + taskIndex, i, startIndex)); + System.out.println("Exception caught:"); + e.printStackTrace(); + System.out.println("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; + long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; + // logger.info( String.format( + // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, + // threadName, getHhs, processHhs ) ); + System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, + threadName)); + + long total = (System.currentTimeMillis() - startTime) / 1000; + String resultString = String + .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", + threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, + setup2, getHhs, processHhs, total); + // logger.info( resultString ); + setResult(resultString); + + modelManager.returnDcWorkModelObject(dcModel, taskIndex, startIndex, endIndex); + + } + + /** + * Returns true if work location results are the same, else returns false. + * + * @param thisHousehold + * @param thatHousehold + * @return true or false + */ + public boolean checkIfSameWorkLocationResults(Household thisHousehold, Household thatHousehold) { + + Person[] thisPersons = thisHousehold.getPersons(); + Person[] thatPersons = thatHousehold.getPersons(); + + if(thisPersons.length!=thatPersons.length) + return false; + + for(int k=1;k +{ + + private static String VERSION = "Task.1.0.3"; + + private transient HashMap propertyMap; + private transient MatrixDataServerIf ms; + private transient ModelStructure modelStructure; + private transient HouseholdDataManagerIf hhDataManager; + private transient String tourCategory; + private transient DestChoiceSize dcSizeObj; + private transient int[] uecIndices; + private transient int[] soaUecIndices; + private transient String dcUecFileName; + private transient String soaUecFileName; + private transient int soaSampleSize; + private transient CtrampDmuFactoryIf dmuFactory; + private transient String restartModelString; + + private int iteration; + private int startIndex; + private int endIndex; + private int taskIndex = -1; + + public WorkLocationChoiceTaskJppfNew(int taskIndex, int startIndex, int endIndex, int iteration) + { + this.startIndex = startIndex; + this.endIndex = endIndex; + this.taskIndex = taskIndex; + this.iteration = iteration; + } + + public void run() + { + + String start = (new Date()).toString(); + long startTime = System.currentTimeMillis(); + + String threadName = null; + try + { + threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " + + Thread.currentThread().getName(); + } catch (UnknownHostException e1) + { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + try + { + DataProvider dataProvider = getDataProvider(); + + this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); + this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); + this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); + this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); + this.uecIndices = (int[]) dataProvider.getParameter("uecIndices"); + this.soaUecIndices = (int[]) dataProvider.getParameter("soaUecIndices"); + this.tourCategory = (String) dataProvider.getParameter("tourCategory"); + this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); + this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); + this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); + this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); + this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); + this.restartModelString = (String) dataProvider.getParameter("restartModelString"); + + } catch (Exception e) + { + e.printStackTrace(); + } + + // get the factory object used to create and recycle dcModel objects. + DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); + + // one of tasks needs to initialize the manager object by passing + // attributes + // needed to create a destination choice model object. + modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, + soaSampleSize, dmuFactory, restartModelString); + + // get a dcModel object from manager, which either creates one or + // returns one for re-use. + WorkLocationChoiceModel dcModel = modelManager.getWorkLocModelObject(taskIndex, iteration, + dcSizeObj, uecIndices, soaUecIndices); + + // logger.info( String.format( + // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, + // taskIndex, + // threadName, startIndex, endIndex ) ); + System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", + new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); + + long setup1 = (System.currentTimeMillis() - startTime) / 1000; + + Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); + + long setup2 = (System.currentTimeMillis() - startTime) / 1000; + // logger.info( String.format( + // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", + // taskIndex, startIndex, endIndex, + // threadName, setup1, setup2 ) ); + System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", + new Date(), taskIndex, startIndex, endIndex, threadName)); + + int i = -1; + try + { + + boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, + HouseholdDataManager.DEBUG_HHS_ONLY_KEY); + + for (i = 0; i < householdArray.length; i++) + { + // for debugging only - process only household objects specified + // for debugging, if property key was set to true + if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) + // if ( householdArray[i].getHhTaz() % 200 != 0 ) + continue; + + if(!dcModel.applyWorkLocationChoice(householdArray[i])){ + i=0; + String restartMsg="A Worker in this HH batch didn't get valid work location. REPROCESSING HH batch, startIndex:"+startIndex+" endIndex="+endIndex; + setResult(restartMsg); + System.out.println(restartMsg); + } + } + + hhDataManager.setHhArray(householdArray, startIndex); + + //check to make sure hh array got set in hhDataManager + boolean allHouseholdsAreSame = false; + while(!allHouseholdsAreSame) { + Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); + for(int j = 0; j< householdArrayRemote.length;++j) { + + Household remoteHousehold = householdArrayRemote[j]; + Household localHousehold = householdArray[j]; + + allHouseholdsAreSame = checkIfSameWorkLocationResults(remoteHousehold, localHousehold); + + if(!allHouseholdsAreSame) + break; + } + if(!allHouseholdsAreSame) { + System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with work location choice results; updating"); + hhDataManager.setHhArray(householdArray, startIndex); + } + } + + + } catch (Exception e) + { + if (i >= 0 && i < householdArray.length) System.out + .println(String + .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", + taskIndex, i, householdArray[i].getHhId(), startIndex)); + else System.out.println(String.format( + "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", + taskIndex, i, startIndex)); + System.out.println("Exception caught:"); + e.printStackTrace(); + System.out.println("Throwing new RuntimeException() to terminate."); + throw new RuntimeException(e); + } + + long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; + long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; + // logger.info( String.format( + // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, + // threadName, getHhs, processHhs ) ); + System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, + threadName)); + + long total = (System.currentTimeMillis() - startTime) / 1000; + String resultString = String + .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", + threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, + setup2, getHhs, processHhs, total); + // logger.info( resultString ); + setResult(resultString); + + modelManager.returnWorkLocModelObject(dcModel, taskIndex, startIndex, endIndex); + + clearClassAttributes(); + } + + /** + * Returns true if work location results are the same, else returns false. + * + * @param thisHousehold + * @param thatHousehold + * @return true or false + */ + public boolean checkIfSameWorkLocationResults(Household thisHousehold, Household thatHousehold) { + + Person[] thisPersons = thisHousehold.getPersons(); + Person[] thatPersons = thatHousehold.getPersons(); + + if(thisPersons.length!=thatPersons.length) + return false; + + for(int k=1;k rbMap; + public HashSet householdTraceSet; + public HashSet originTraceSet; + + public String outputsPath; + public String disaggTODPath; + public String todType; + public String outputFile; + + public String inputFile; + public String marketSegment; + public double SampleRate; + + public PrintWriter tripWriter; + + private static Logger logger = Logger.getLogger("postprocessModel"); + + + /** + * Default constructor. + */ + public PostprocessModel(HashMap rbMap, String timeType, double sampleRate, String inputFile, String marketSegment){ + + this.rbMap = rbMap; + this.SampleRate = sampleRate; + this.inputFile = inputFile; + this.marketSegment = marketSegment; + this.todType = timeType; + + } + + + public void runModel(){ + + disaggTODPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_DISAGGPATHTOD); + outputFile = disaggTODPath + Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_TRIPOUT); + outputsPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_OUTPUTSPATH); + outputFile = outputsPath + Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_TRIPOUT); + + setDebugHouseholdsFromPropertyMap(); + setDebugOrigZonesFromPropertyMap(); + + logger.info("Trip file being written to "+outputFile); + // Write the trip header to the output file + boolean fileExists = new File(outputFile).isFile(); + + FileWriter writer; + + if(fileExists){ + logger.info("Output file already exists. New data will be appended."); + try { + writer = new FileWriter(new File(outputFile), true); + tripWriter = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) { + logger.fatal(String.format("Exception occurred opening Postprocessing output file: %s.", + outputFile)); + throw new RuntimeException(e); + } + }else{ + logger.info("Output file does not exist. New file being created."); + try { + writer = new FileWriter(new File(outputFile)); + tripWriter = new PrintWriter(new BufferedWriter(writer)); + } catch (IOException e) { + logger.fatal(String.format("Exception occurred opening Postprocessing output file: %s.", + outputFile)); + throw new RuntimeException(e); + } + dtaTrip Trip = new dtaTrip(); + Trip.writeHeader(tripWriter); + } + + if(todType.equalsIgnoreCase("broad")){ + logger.info("Processing Broad TOD Model"); + TableDataSet broadFiles = TableDataSet.readFile(disaggTODPath+inputFile); + int numFiles = broadFiles.getRowCount(); + for (int i=0; i(); + + // get the household ids for which debug info is required + String householdTraceStringList = rbMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); + + if (householdTraceStringList != null) + { + StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); + while(householdTokenizer.hasMoreTokens()) + { + String listValue = householdTokenizer.nextToken(); + int idValue = Integer.parseInt(listValue.trim()); + householdTraceSet.add(idValue); + } + } + + } + + /** + * Set the HashSet for debugging households, which contains the IDs of the households to debug. + */ + private void setDebugOrigZonesFromPropertyMap() + { + originTraceSet = new HashSet(); + + // get the household ids for which debug info is required + String originTraceStringList = rbMap.get(PROPERTIES_ORIGIN_TRACE_LIST); + + if (originTraceStringList != null) + { + StringTokenizer originTokenizer = new StringTokenizer(originTraceStringList, ","); + while(originTokenizer.hasMoreTokens()) + { + String listValue = originTokenizer.nextToken(); + int idValue = Integer.parseInt(listValue.trim()); + originTraceSet.add(idValue); + } + } + + } + + /** + * Check if this is a trace household. + * + * @param householdId + * @return True if a trace household, else false + */ + public boolean isTraceHousehold(int householdId){ + + return householdTraceSet.contains(householdId); + + } + + /** + * Check if this is a trace origin. + * + * @param householdId + * @return True if a trace household, else false + */ + public boolean isTraceOrigin(int origTAZ){ + + return originTraceSet.contains(origTAZ); + + } + + /** + * @param args + */ + public static void main(String[] args) { + + String propertiesFile = null; + HashMap pMap; + String todType = null; + String inputFile = null; + String marketSegment = null; + double sampleRate = 1.0; + + if (args.length == 0) { + logger.error( String.format("no properties file base name (without .properties extension) was specified as an argument.") ); + return; + } else{ + propertiesFile = args[0]; + } + for (int i = 1; i< args.length;++i){ + if (args[i].equalsIgnoreCase("-todType")){ + todType = (String) args[i + 1]; + } + if (args[i].equalsIgnoreCase("-sampleRate")){ + sampleRate = new Double(args[i+1]); + } + if (args[i].equalsIgnoreCase("-inputFile")){ + inputFile = (String) args[i+1]; + } + if (args[i].equalsIgnoreCase("-marketSegment")){ + marketSegment = (String) args[i+1]; + } + } + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + + logger.info("Running SANDAG Trip TOD Disaggregation Model"); + + logger.info("todType = "+todType); + logger.info("Sample Rate = "+sampleRate); + logger.info("Input File = "+inputFile); + logger.info("Market Segment = "+marketSegment); + + PostprocessModel postprocessingModel = new PostprocessModel(pMap,todType,sampleRate,inputFile,marketSegment); + postprocessingModel.runModel(); + } + +} + + diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java new file mode 100644 index 0000000..855582c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java @@ -0,0 +1,224 @@ +package org.sandag.abm.dta.postprocessing; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; + +import java.io.File; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.HashSet; + +import org.apache.log4j.Logger; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; + +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.dta.postprocessing.dtaTrip; +import org.sandag.abm.dta.postprocessing.todDisaggregationModel; +import org.sandag.abm.dta.postprocessing.PostprocessModel; +import org.sandag.abm.dta.postprocessing.spatialDisaggregationModel; + +public class broadTODProcessing { + + private static final String PROPERTIES_DISAGGPATHTOD = "dta.postprocessing.disaggregateTOD.path"; + private static final String PROPERTIES_DISAGGPATHZONE = "dta.postprocessing.disaggregateZone.path"; + private static final String PROPERTIES_DISAGGPATHNODE = "dta.postprocessing.disaggregateNode.path"; + private static final String PROPERTIES_BROADTODPROBABILITIES = "dta.postprocessing.BroadTODFile"; + private static final String PROPERTIES_ZONEPROBABILITIES = "dta.postprocessing.ZoneFile"; + private static final String PROPERTIES_NODEPROBABILITIES = "dta.postprocessing.NodeFile"; + private static final String PROPERTIES_RANDOMSEED = "dta.postprocessing.RandomSeed"; + + private HashMap rbMap; + private HashSet originTraceSet; + private MersenneTwister random; + private long randomSeed; + + private String disaggTODPath; + private String disaggZonePath; + private String disaggNodePath; + + private double sampleRate; + private String inputFile; + private String marketSegment; + + private int[] broadTODMap; + private int[] tazMap; + private int[] mgraTAZMap; + private int[] mgraNodeMap; + private int[] nodeMap; + private double[] broadProbabilities; + private double[] mgraProdProbabilities; + private double[] mgraAttrProbabilities; + private double[] nodeProbabilities; + + private todDisaggregationModel todDisaggregationModel; + private spatialDisaggregationModel spatialDisaggregationModel; + + private dtaTrip Trip; + private PrintWriter tripWriter; + + + private transient Logger logger = Logger.getLogger("postprocessModel"); + + + /** + * Default constructor. + */ + public broadTODProcessing(HashMap rbMap, double sampleRate, String inputFile, String marketSegment, HashSet originTraceSet, PrintWriter tripWriter){ + + this.rbMap = rbMap; + this.sampleRate = sampleRate; + this.inputFile = inputFile; + this.marketSegment = marketSegment; + this.tripWriter = tripWriter; + this.originTraceSet = originTraceSet; + + todDisaggregationModel = new todDisaggregationModel(rbMap); + spatialDisaggregationModel = new spatialDisaggregationModel(rbMap); + + randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); + random = new MersenneTwister(); + random.setSeed(randomSeed); + + //Read in factors and maps to aggregate time periods + String broadFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_BROADTODPROBABILITIES); + + TableDataSet BroadData = TableDataSet.readFile(broadFactorsFile); + int numPeriods = BroadData.getRowCount(); + broadProbabilities = todDisaggregationModel.getTODProbabilities(BroadData, numPeriods, marketSegment); + broadTODMap = todDisaggregationModel.getTODMap(BroadData, numPeriods); + + String mgraFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_ZONEPROBABILITIES); + TableDataSet MGRAData = TableDataSet.readFile(mgraFactorsFile); + int numMGRAs = MGRAData.getRowCount(); + mgraProdProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Prods", marketSegment); + mgraAttrProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Attrs", marketSegment); + tazMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "taz"); + mgraTAZMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "mgra"); + + String nodeFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_NODEPROBABILITIES); + TableDataSet NodeData = TableDataSet.readFile(nodeFactorsFile); + int numNodes = NodeData.getRowCount(); + nodeProbabilities = spatialDisaggregationModel.getSpatialProbabilities(NodeData, numNodes, "Probability", null); + nodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "NodeId"); + mgraNodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "MGRA"); + + } + + + /** + * Create trip record from and disaggregate tod, mgra, and node for broad tod files + */ + public void createBroadTODTrips(String inputFileName,String MarketSegment,String matrixName,int broadTOD,String vehType,int occ,int toll){ + + //Read TransCAD matrix and create a trip for each record in each cell + File inputFile = new File(inputFileName); + Matrix m = MatrixReader.readMatrix(inputFile,matrixName); + int intTrips = 0; + double expansionFactor=1.0; + int totalTrips = 0; + boolean debug=false; + + logger.info("*************************************"); + logger.info("Summary info for TransCAD Matrix"); + logger.info("Market Segment = "+MarketSegment); + logger.info("TNCVehicle Type = "+vehType); + logger.info("TNCVehicle Occupancy = "+occ); + logger.info("Toll Eligibility = "+toll); + logger.info("Number of Trips = "+m.getSum()); + logger.info("*************************************"); + + + for (int i=1; i<=m.getRowCount();++i){ + for (int j=1; j<=m.getColumnCount();++j){ + + double numTrips = m.getValueAt(i,j); + + if (numTrips==0.0) + continue; + + intTrips = (int) Math.floor(numTrips); + double fracTrips = numTrips - intTrips; + double rn = random.nextDouble(); + // Check if a trip should be created for the fractional trip value + if (rn rbMap; + private HashSet householdTraceSet; + private HashSet originTraceSet; + private MgraDataManager mgraManager; + //private final AutoAndNonMotorizedSkimsCalculator autoNonMotSkims; + private MersenneTwister random; + private long randomSeed; + + private todDisaggregationModel todDisaggregationModel; + private spatialDisaggregationModel spatialDisaggregationModel; + + private String disaggTODPath; + private String disaggNodePath; + private String outputsPath; + + private double sampleRate; + private String inputFile; + private String marketSegment; + + private int[] detailTODMap; + private int[] mgraNodeMap; + private int[] nodeMap; + private int[] tazMap; + private int[] mgraTAZMap; + + private double[] detailProbabilities; + private double[] nodeProbabilities; + private double[] mgraProdProbabilities; + private double[] mgraAttrProbabilities; + + private dtaTrip Trip; + private PrintWriter tripWriter; + + + private transient Logger logger = Logger.getLogger("postprocessModel"); + + + /** + * Default constructor. + */ + public detailedTODProcessing(HashMap rbMap, double sampleRate, String inputFile, String marketSegment, HashSet householdTraceSet, HashSet originTraceSet, PrintWriter tripWriter){ + + this.sampleRate = sampleRate; + this.rbMap = rbMap; + this.inputFile = inputFile; + this.marketSegment = marketSegment; + this.tripWriter = tripWriter; + this.householdTraceSet = householdTraceSet; + this.originTraceSet = originTraceSet; + + todDisaggregationModel = new todDisaggregationModel(rbMap); + + spatialDisaggregationModel = new spatialDisaggregationModel(rbMap); + + mgraManager = MgraDataManager.getInstance(rbMap); + //autoNonMotSkims = new AutoAndNonMotorizedSkimsCalculator(rbMap); + + outputsPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_OUTPUTSPATH); + + //Read in factors and maps to aggregate time periods + String detailedFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_DETAILPROBABILITIES); + + TableDataSet DetailedData = TableDataSet.readFile(detailedFactorsFile); + int numDetailedPeriods = DetailedData.getRowCount(); + detailProbabilities = todDisaggregationModel.getTODProbabilities(DetailedData, numDetailedPeriods, null); + detailTODMap = todDisaggregationModel.getTODMap(DetailedData, numDetailedPeriods); + + String nodeFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_NODEPROBABILITIES); + TableDataSet NodeData = TableDataSet.readFile(nodeFactorsFile); + int numNodes = NodeData.getRowCount(); + nodeProbabilities = spatialDisaggregationModel.getSpatialProbabilities(NodeData, numNodes, "Probability", null); + nodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "NodeId"); + mgraNodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "MGRA"); + + randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); + random = new MersenneTwister(); + random.setSeed(randomSeed); + + // read skims - AshishK + String skimPath = Util.getStringValueFromPropertyMap(rbMap, "skims.path"); + String skimPrefix = Util.getStringValueFromPropertyMap(rbMap, "dta.skims.prefix"); + String skimSuffix = Util.getStringValueFromPropertyMap(rbMap, "dta.skims.mat.name.suffix"); + skimMatrix = new Matrix[ModelStructure.MODEL_PERIOD_LABELS.length]; + + for(int p=0; p rbMap){ + SandagModelStructure modelStructure = new SandagModelStructure(); + TableDataSet tripRecords = TableDataSet.readFile(inputFile); + int numRecords = tripRecords.getRowCount(); + int period = -1; + int periodLast = -1; + int hhidLast = -1; + int persidLast = -1; + int touridLast = -1; + int modeLast = -1; + + double expansionFactor=1.0; + boolean debug = false; + boolean addSOVTrip = false; + int tripExp=0; + int offset = 0; + int scheduleCount = 0; + int tripsCount = 0; + int [] dtaTimes; + int [] dtaTimesPrev; + dtaTimes = new int[10]; + dtaTimesPrev = new int[10]; + + Arrays.fill(dtaTimesPrev, 0); + + if(inputFile.contains("TripMatrices.csv")) { + String mgraFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_ZONEPROBABILITIES); + TableDataSet MGRAData = TableDataSet.readFile(mgraFactorsFile); + int numMGRAs = MGRAData.getRowCount(); + mgraProdProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Prods", marketSegment); + mgraAttrProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Attrs", marketSegment); + tazMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "taz"); + mgraTAZMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "mgra"); + + } + + + logger.info("Reading "+numRecords+" trip records from disaggregate file: "+inputFile); + + // Create a trip record for each record in the input file + for (int i=0; i0){ + destMGRA = parkingMGRA; + } + } + if (tripRecords.containsColumn("originMGRA")){ + origMGRA = (int) tripRecords.getValueAt(i+1, "originMGRA"); + } + if (tripRecords.containsColumn("destinationMGRA")){ + destMGRA = (int) tripRecords.getValueAt(i+1, "destinationMGRA"); + } + + if (tripRecords.containsColumn("originTAZ")){ + origTAZ = (int) tripRecords.getValueAt(i+1, "originTAZ"); + }else{ + origTAZ = mgraManager.getTaz(origMGRA); + } + if (tripRecords.containsColumn("destinationTAZ")){ + destTAZ = (int) tripRecords.getValueAt(i+1, "destinationTAZ"); + }else{ + destTAZ = mgraManager.getTaz(destMGRA); + } + + if(origMGRA==0) + origMGRA = spatialDisaggregationModel.selectMGRAfromTAZ(Trip.getOriginTaz(),mgraTAZMap,tazMap,mgraProdProbabilities,debug); + + if(destMGRA==0) + destMGRA = spatialDisaggregationModel.selectMGRAfromTAZ(Trip.getOriginTaz(),mgraTAZMap,tazMap,mgraAttrProbabilities,debug); + + if (origMGRA==0){ + origMGRA = 50000+Trip.getOriginTaz(); + } + if (destMGRA==0){ + destMGRA = 50000+Trip.getDestinationTaz(); + } + + //JEF 2021-04-21: not sure what the following code intended - removing + /* + if (tripRecords.containsColumn("arrivalMode")){ + int arriveMode = (int) tripRecords.getValueAt(i+1, "arrivalMode"); + if((occ>1 && arriveMode==5)||(mode>=16 && mode<26)){ + addSOVTrip = true; + } + } + */ + if (tripRecords.containsColumn("driver")){ + tourDriver = (int) tripRecords.getValueAt(i+1, "driver"); + } + + if (tripRecords.containsColumn("stop_period")){ + period = (int) tripRecords.getValueAt(i+1,"stop_period"); + } + if (tripRecords.containsColumn("period")){ + period = (int) tripRecords.getValueAt(i+1,"period"); + } + if (tripRecords.containsColumn("departTime")){ + period = (int) tripRecords.getValueAt(i+1,"departTime"); + } + if(tripRecords.containsColumn("departTimeAbmHalfHour")) { + period = (int) tripRecords.getValueAt(i+1,"departTimeAbmHalfHour"); + + } + + if(period==0) + period=1; + + //Calculate number of trips to generate from the record (at a tour level where possible) + + if (tourid!=touridLast || persid!=persidLast || hhid!=hhidLast || (tourid==0 && hhid==0)){ + tripExp = (int) Math.floor(expansionFactor); + double tripsFrac = expansionFactor - tripExp; + double rn = random.nextDouble(); + if (rnfractionalTrips) + tripExp += 0; + }else if(fractionalTrips<1.0 && mode==modeLast){ + double rn = random.nextDouble(); + if (rn>fractionalTrips) + tripExp += 0; + }else if((fractionalTrips<1.0 && mode!=modeLast) || (fractionalTrips==1.0 && modelStructure.getTourModeIsHov(modeLast))){ + tripExp = (int) Math.floor(expansionFactor); + double tripsFrac = expansionFactor - tripExp; + double rn = random.nextDouble(); + if (rnfractionalTrips) + tripExp += 0; + } + + //logger.info("expansionFactor " + expansionFactor + ", fractionalTrips " + fractionalTrips + ", tripExp " + tripExp); + // Reset the dtaTimes array + Arrays.fill(dtaTimes,0); + + // Create a number of integer trip instances based on the expansion factor + for (int k=0; k=1 & dtaPer<=36){ + tod = EA_SKIM_PERIOD_INDEX; + }else if(dtaPer>36 & dtaPer<=72){ + tod = AM_SKIM_PERIOD_INDEX; + }else if(dtaPer>72 & dtaPer<=150){ + tod = MD_SKIM_PERIOD_INDEX; + }else if(dtaPer>150 & dtaPer<=192){ + tod = PM_SKIM_PERIOD_INDEX; + }else{ + tod = EV_SKIM_PERIOD_INDEX; + } + + //dtaPeriod = todDisaggregationModel.calculateDisaggregateTOD(period, detailTODMap, detailProbabilities,debug); + + //double[] autoSkims = autoNonMotSkims.getAutoSkims(tripOrig, tripDest, tod, false, logger); + //double travTime = autoSkims[DA_TIME_INDEX]; + + //changed the code to directly read the values from skim matrices - AshishK + double travTime = skimMatrix[tod].getValueAt(origTAZ, destTAZ); + + int travPer = (int) Math.ceil(travTime/5.0); + + if (direction==1){ + dtaPeriod = dtaPer + travPer + 2; + }else{ + dtaPeriod = dtaPer - (travPer + 2); + } + + // limit the dta period between 1 and 288 - AshishK + if(dtaPeriod < 1) { + dtaPeriod = 1; + } + if(dtaPeriod > detailProbabilities.length) { + dtaPeriod = detailProbabilities.length; + } + + int origNode = spatialDisaggregationModel.selectNodeFromMGRA(tripOrig, nodeMap, mgraNodeMap, nodeProbabilities, debug); + int destNode = spatialDisaggregationModel.selectNodeFromMGRA(tripDest, nodeMap, mgraNodeMap, nodeProbabilities, debug); + + Trip = new dtaTrip(); + Trip.setMarketSegment(marketSegment); + Trip.setOriginMGRA(tripOrig); + Trip.setDestinationMGRA(tripDest); + Trip.setOriginTaz(origTAZ); + Trip.setDestinationTaz(destTAZ); + Trip.setOriginNode(origNode); + Trip.setDestinationNode(destNode); + Trip.setDetailedPeriod(period); + Trip.setDTAPeriod(dtaPeriod); + Trip.setVehicleType("passengerCar"); + Trip.setVehicleOccupancy(1); + Trip.setTollEligible(toll); + Trip.setExpansionFactor(1.0); + tripWriter.print("\r\n"); + Trip.writeTrip(tripWriter); + + } + + /** + * Check if this is a trace household. + * + * @param householdId + * @return True if a trace household, else false + */ + public boolean isTraceHousehold(int householdId){ + + return householdTraceSet.contains(householdId); + + } + + /** + * Check if this is a trace origin. + * + * @param householdId + * @return True if a trace household, else false + */ + public boolean isTraceOrigin(int origTAZ){ + + return originTraceSet.contains(origTAZ); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java new file mode 100644 index 0000000..44b6e34 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java @@ -0,0 +1,386 @@ +package org.sandag.abm.dta.postprocessing; + +import java.io.PrintWriter; +import java.io.Serializable; + +public class dtaTrip implements Serializable { + + private int id; + private int hhid; + private int persid; + private int tourid; + private int originTaz; + private int destinationTaz; + private int originMGRA; + private int destinationMGRA; + private int originNode; + private int destinationNode; + private String vehicleType; + private int vehOccupancy; + private int tollEligible; + private String marketSegment; + private int broadPeriod; + private int detailedPeriod; + private int dtaPeriod; + private int driver; + private double expansionFactor; + + + + /** + * Default constructor; nothing initialized. + */ + public dtaTrip(){ + + } + /** + * Initialize a trip will zero values for all fields + */ + public void initializeTrip() { + this.id=0; + this.hhid=0; + this.persid=0; + this.tourid=0; + this.originTaz=0; + this.destinationTaz=0; + this.originMGRA=0; + this.destinationMGRA=0; + this.originNode=0; + this.destinationNode=0; + this.vehicleType="na"; + this.vehOccupancy=0; + this.tollEligible=0; + this.marketSegment="na"; + this.broadPeriod=0; + this.detailedPeriod=0; + this.dtaPeriod=0; + this.driver=-1; + this.expansionFactor=1.0; + + } + /** + * @return the household id + */ + public int getHHId() { + return hhid; + } + /** + * @param hhid the household id to set + */ + public void setHHId(int hhid) { + this.hhid = hhid; + } + /** + * @return the person id + */ + public int getPersonId() { + return persid; + } + /** + * @param persid the person id to set + */ + public void setPersonId(int persid) { + this.persid = persid; + } + /** + * @return the tour id + */ + public int getTourId() { + return tourid; + } + /** + * @param tourid the tour id to set + */ + public void setTourId(int tourid) { + this.tourid = tourid; + } + /** + * @return the id + */ + public int getId() { + return id; + } + /** + * @param id the id to set + */ + public void setId(int id) { + this.id = id; + } + + /** + * @return the originTaz + */ + public int getOriginTaz() { + return originTaz; + } + /** + * @param originTaz the originTaz to set + */ + public void setOriginTaz(int originTaz) { + this.originTaz = originTaz; + } + /** + * @return the destinationTaz + */ + public int getDestinationTaz() { + return destinationTaz; + } + /** + * @param destinationTaz the destinationTaz to set + */ + public void setDestinationTaz(int destinationTaz) { + this.destinationTaz = destinationTaz; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() { + return originMGRA; + } + /** + * @param originMGRA the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) { + this.originMGRA = originMGRA; + } + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() { + return destinationMGRA; + } + /** + * @param destinationMGRA the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) { + this.destinationMGRA = destinationMGRA; + } + + /** + * @return the originNode + */ + public int getOriginNode() { + return originNode; + } + /** + * @param originNode the originNode to set + */ + public void setOriginNode(int originNode) { + this.originNode = originNode; + } + /** + * @return the destinationNode + */ + public int getDestinationNode() { + return destinationNode; + } + /** + * @param destinationMGRA the destinationMGRA to set + */ + public void setDestinationNode(int destinationNode) { + this.destinationNode = destinationNode; + } + /** + * set trip mode values based on trip mode in input file + */ + public void setTripMode(int mode) { + if(mode<=8||mode>=27){ + setVehicleType("passengerCar"); + setVehicleOccupancy(1); + setTollEligible(0); + if(mode==2){ + setTollEligible(1); + } + if((mode>=3 && mode<=5)||mode==27){ + setVehicleOccupancy(2); + } + if(mode==5){ + setTollEligible(1); + } + if(mode>=6 && mode<=8){ + setVehicleOccupancy(3); + } + if(mode==8){ + setTollEligible(1); + } + } + if(mode>8 && mode<11){ + setVehicleType("nonMotorized"); + setVehicleOccupancy(0); + setTollEligible(0); + } + if(mode>=11 && mode<16){ + setVehicleType("WalkTransit"); + setVehicleOccupancy(0); + setTollEligible(0); + } + if(mode>=16 && mode<26){ + setVehicleType("DriveTransit"); + setVehicleOccupancy(1); + setTollEligible(0); + } + if(mode==26){ + setVehicleType("SchoolBus"); + } + } + /** + * @return the person number of the driver + */ + public int getTourDriver() { + return driver; + } + /** + * @param driver the tour driver to set + */ + public void setTourDriver(int tourDriver) { + this.driver = tourDriver; + } + + /** + * @return the vehicleType + */ + public String getVehicleType() { + return vehicleType; + } + /** + * @param vehicleType the vehicleType to set + */ + public void setVehicleType(String vehicleType) { + this.vehicleType = vehicleType; + } + /** + * @return the vehicleOccupancy + */ + public int getVehicleOccupancy() { + return vehOccupancy; + } + /** + * @param vecOccupancy the vehOccupancy to set + */ + public void setVehicleOccupancy(int vehOccupancy) { + this.vehOccupancy = vehOccupancy; + } + /** + * @return the tollEligibility + */ + public int getTollEligible() { + return tollEligible; + } + /** + * @param tollEligible the tollEligible to set + */ + public void setTollEligible(int tollEligible) { + this.tollEligible = tollEligible; + } + /** + * @return the market segment + */ + public String getMarketSegment() { + return marketSegment; + } + + /** + * @param marketSegment the marketSegment to set + */ + public void setMarketSegment(String marketSegment){ + this.marketSegment = marketSegment; + } + + /** + * @return the broad time period + */ + public int getBroadPeriod() { + return broadPeriod; + } + /** + * @param broadPeriod the broadPeriod to set + */ + public void setBroadPeriod(int Period) { + this.broadPeriod = Period; + } + + /** + * @return the detailed time period + */ + public int getDetailedPeriod() { + return detailedPeriod; + } + /** + * @param detailedPeriod the detailedPeriod to set + */ + public void setDetailedPeriod(int Period) { + this.detailedPeriod = Period; + } + + /** + * @return the dta time period + */ + public int getDTAPeriod() { + return dtaPeriod; + } + /** + * @param dtaPeriod the dtaPeriod to set + */ + public void setDTAPeriod(int Period) { + this.dtaPeriod = Period; + } + /** + * @return the trip expansion factor + */ + public double getExpansionFactor() { + return expansionFactor; + } + /** + * @param dtaPeriod the dtaPeriod to set + */ + public void setExpansionFactor(double expansionFactor) { + this.expansionFactor = expansionFactor; + } + + /** + * Write the trip + * + * @param writer + */ + public void writeTrip(PrintWriter writer){ + String record = new String( + hhid + "," + + persid + "," + + tourid + "," + + id + "," + + originTaz + "," + + destinationTaz + "," + + originMGRA + "," + + destinationMGRA + "," + + originNode + "," + + destinationNode + "," + + vehicleType + "," + + vehOccupancy + "," + + tollEligible + "," + + marketSegment + "," + + detailedPeriod + "," + + broadPeriod + "," + + dtaPeriod + "," + + driver + "," + + expansionFactor + ); + writer.print(record); + } + + /** + * Write a header record + * + * @param writer + */ + /** + * Write a header record + * + * @param writer + */ + public void writeHeader(PrintWriter writer){ + String header = "hh_id,person_id,tour_id,trip_id,originTaz,destinationTaz,originMGRA,destinationMGRA,originNode,destinationNode,vehicleType,vehicleOccupancy,tollEligibility,marketSegment,detailedPeriod,broadPeriod,dtaPeriod,driver,expansionFactor"; + writer.print(header); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java new file mode 100644 index 0000000..fd22ba1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java @@ -0,0 +1,144 @@ +package org.sandag.abm.dta.postprocessing; + +import java.io.PrintWriter; +import java.util.HashMap; + +import org.apache.log4j.Logger; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import org.sandag.abm.ctramp.Util; + +public class spatialDisaggregationModel { + + private static final String PROPERTIES_RANDOMSEED = "dta.postprocessing.RandomSeed"; + + private HashMap rbMap; + private MersenneTwister random; + private long randomSeed; + + public PrintWriter tripWriter; + + + private transient Logger logger = Logger.getLogger("postprocessModel"); + + + /** + * Default constructor. + */ + public spatialDisaggregationModel(HashMap rbMap){ + + this.rbMap = rbMap; + randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); + random = new MersenneTwister(); + random.setSeed(randomSeed); + } + + /** + * Read the probability by spatial data file, return an array of probabilities. + */ + public double[] getSpatialProbabilities(TableDataSet SpatialData, int numRecords, String inputField, String marketSegment){ + + // read the spatial factors file + double [] probabilities; + probabilities = new double [numRecords]; + + String fieldName = null; + if (marketSegment==null){ + fieldName = inputField; + }else{ + fieldName = marketSegment+inputField; + } + //fill in probabilities array + for(int i = 0;i rbMap; + private MersenneTwister random; + private long randomSeed; + + + public PrintWriter tripWriter; + + + private transient Logger logger = Logger.getLogger("postprocessModel"); + + + /** + * Default constructor. + */ + public todDisaggregationModel(HashMap rbMap){ + this.rbMap = rbMap; + randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); + random = new MersenneTwister(); + random.setSeed(randomSeed); + } + + /** + * Read the probability by tod data file, return an array of probabilities. + */ + public double[] getTODProbabilities(TableDataSet TODData, int numPeriods, String marketSegment){ + + // read the tod factors file + double [] probabilities; + probabilities = new double [numPeriods]; + + //fill in probabilities array + for(int i = 0;i1) + startLoc = (period-1)*6 + 18; + + // loop through the array of probabilities + for (int i=startLoc;i rbMap; + private McLogsumsCalculator logsumsCalculator; + private AutoTazSkimsCalculator tazDistanceCalculator; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + + private boolean seek; + private int traceId; + + private static float sampleRate=0; + private int iteration = 1; + + /** + * Constructor + * + * @param rbMap + */ + public InternalExternalModel(HashMap rbMap) + { + this.rbMap = rbMap; + mgraManager = MgraDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trace")); + + } + + public int getIteration() + { + return iteration; + } + + public void setIteration(int iteration) + { + this.iteration = iteration; + } + + /** + * Run InternalExternal model. + */ + public void runModel() + { + + InternalExternalModelStructure modelStructure = new InternalExternalModelStructure(); + + InternalExternalDmuFactoryIf dmuFactory = new InternalExternalDmuFactory(modelStructure); + + InternalExternalTourManager tourManager = new InternalExternalTourManager(rbMap, iteration); + + tourManager.generateTours(); + + InternalExternalTour[] tours = tourManager.getTours(); + + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + logsumsCalculator = new McLogsumsCalculator(); + logsumsCalculator.setupSkimCalculators(rbMap); + logsumsCalculator.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + InternalExternalTourTimeOfDayChoiceModel todChoiceModel = new InternalExternalTourTimeOfDayChoiceModel( + rbMap); + InternalExternalTourDestChoiceModel destChoiceModel = new InternalExternalTourDestChoiceModel( + rbMap, modelStructure, dmuFactory); + destChoiceModel.calculateTazProbabilities(dmuFactory); + + InternalExternalTripModeChoiceModel tripModeChoiceModel = new InternalExternalTripModeChoiceModel( + rbMap, modelStructure, dmuFactory); + + // Run models for array of tours + for (int i = 0; i < tours.length; ++i) + { + + InternalExternalTour tour = tours[i]; + + if (i < 10 || i % 1000 == 0) logger.info("Processing tour " + i); + + if (seek && tour.getID() != traceId) continue; + + if (tour.getID() == traceId) tour.setDebugChoiceModels(true); + + todChoiceModel.calculateTourTOD(tour); + destChoiceModel.chooseDestination(tour); + + // generate trips and choose mode for them - note this assumes two + // trips per tour + InternalExternalTrip[] trips = new InternalExternalTrip[2]; + int tripNumber = 0; + + // generate an outbound trip from the tour origin to the destination + // and choose a mode + trips[tripNumber] = new InternalExternalTrip(tour, true, mgraManager); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + // generate an inbound trip from the tour destination to the origin + // and choose a mode + trips[tripNumber] = new InternalExternalTrip(tour, false, mgraManager); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + // set the trips in the tour object + tour.setTrips(trips); + + } + + tourManager.writeOutputFile(rbMap); + + logger.info("Internal-External Model successfully completed!"); + + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @param args + */ + public static void main(String[] args) + { + Runtime gfg = Runtime.getRuntime(); + long memory1; + // checking the total memeory + System.out.println("Total memory is: "+ gfg.totalMemory()); + // checking free memory + memory1 = gfg.freeMemory(); + System.out.println("Initial free memory at IE model: "+ memory1); + // calling the garbage collector on demand + gfg.gc(); + memory1 = gfg.freeMemory(); + System.out.println("Free memory after garbage "+ "collection: " + memory1); + + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running InternalExternal Model")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + + // sampleRate is not relevant for internal-external model, since + // sampling + // would have been applied in CT-RAMP model + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + + logger.info("IE Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + InternalExternalModel internalExternalModel = new InternalExternalModel(pMap); + internalExternalModel.setIteration(iteration); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = internalExternalModel.startMatrixServerProcess( + matrixServerAddress, serverPort, mt); + internalExternalModel.ms = matrixServer; + } else + { + internalExternalModel.ms = new MatrixDataServerRmi(matrixServerAddress, + serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + internalExternalModel.ms.testRemote("InternalExternalModel"); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(internalExternalModel.ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + internalExternalModel.runModel(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java new file mode 100644 index 0000000..f97bed3 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java @@ -0,0 +1,95 @@ +package org.sandag.abm.internalexternal; + +import org.sandag.abm.application.SandagModelStructure; + +public class InternalExternalModelStructure + extends SandagModelStructure +{ + + public static final byte NUMBER_VISITOR_PURPOSES = 6; + public static final byte WORK = 0; + public static final byte RECREATION = 1; + public static final byte DINING = 2; + + public static final String[] VISITOR_PURPOSES = {"WORK", "RECREATE", "DINING"}; + + // override on max tour mode, since we have taxi in this model. + public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 27; + + public static final byte NUMBER_VISITOR_SEGMENTS = 2; + public static final byte BUSINESS = 0; + public static final byte PERSONAL = 1; + + public static final String[] VISITOR_SEGMENTS = {"BUSINESS", "PERSONAL"}; + public static final byte DEPARTURE = 0; + public static final byte ARRIVAL = 1; + + public static final byte INCOME_SEGMENTS = 5; + + // note that time periods start at 1 and go to 40 + public static final byte TIME_PERIODS = 40; + + public static final int AM = 0; + public static final int PM = 1; + public static final int OP = 2; + public static final int[] SKIM_PERIODS = {AM, PM, OP}; + public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; + public static final int UPPER_EA = 3; + public static final int UPPER_AM = 9; + public static final int UPPER_MD = 22; + public static final int UPPER_PM = 29; + public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; + + public static final byte TAXI = 13; + + /** + * Taxi tour mode + * + * @param tourMode + * @return + */ + public boolean getTourModeIsTaxi(int tourMode) + { + + if (tourMode == TAXI) return true; + else return false; + + } + + /** + * return the Skim period index 0=am, 1=pm, 2=off-peak + */ + public static int getSkimPeriodIndex(int departPeriod) + { + + int skimPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; + else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; + else skimPeriodIndex = OP; + + return skimPeriodIndex; + + } + + /** + * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV + */ + public static int getModelPeriodIndex(int departPeriod) + { + + int modelPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; + else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; + else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; + else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; + else modelPeriodIndex = 4; + + return modelPeriodIndex; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java new file mode 100644 index 0000000..97a7dc8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java @@ -0,0 +1,307 @@ +package org.sandag.abm.internalexternal; + +import java.io.Serializable; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Household; +import com.pb.common.math.MersenneTwister; + +public class InternalExternalTour + implements Serializable +{ + + private MersenneTwister random; + private int ID; + private int hhID; + private int personID; + private int pnum; + + // following variables set from household and person objects + private int income; + private int autos; + private int age; + private int female; + private double nonWorkTimeFactor; + + private boolean avAvailable; + + // private InternalExternalStop[] outboundStops; + // private InternalExternalStop[] inboundStops; + + private InternalExternalTrip[] trips; + + private int departTime; + private int arriveTime; + + private boolean debugChoiceModels; + + // following variables chosen via choice models + private int originMGRA; + private int destinationMGRA; + private int destinationTAZ; // the external TAZ may be + // different from the + // external MGRA + + /** + * Public constructor. + * + * @param seed + * A seed for the random number generator. + */ + public InternalExternalTour(long seed) + { + + random = new MersenneTwister(seed); + } + + /** + * @return the destinationTAZ + */ + public int getDestinationTAZ() + { + return destinationTAZ; + } + + /** + * @param destinationTAZ + * the destinationTAZ to set + */ + public void setDestinationTAZ(int destinationTAZ) + { + this.destinationTAZ = destinationTAZ; + } + + /** + * @return the iD + */ + public int getID() + { + return ID; + } + + /** + * @param iD + * the iD to set + */ + public void setID(int iD) + { + ID = iD; + } + + /** + * @return the departTime + */ + public int getDepartTime() + { + return departTime; + } + + /** + * @param departTime + * the departTime to set + */ + public void setDepartTime(int departTime) + { + this.departTime = departTime; + } + + public InternalExternalTrip[] getTrips() + { + return trips; + } + + public void setTrips(InternalExternalTrip[] trips) + { + this.trips = trips; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() + { + return originMGRA; + } + + /** + * @param originMGRA + * the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) + { + this.originMGRA = originMGRA; + } + + /** + * Get a random number from the parties random class. + * + * @return A random number. + */ + public double getRandom() + { + return random.nextDouble(); + } + + /** + * @return the debugChoiceModels + */ + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + /** + * @param debugChoiceModels + * the debugChoiceModels to set + */ + public void setDebugChoiceModels(boolean debugChoiceModels) + { + this.debugChoiceModels = debugChoiceModels; + } + + /** + * Get the number of outbound stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberOutboundStops() + { + return 0; + + } + + /** + * Get the number of return stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberInboundStops() + { + return 0; + + } + + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() + { + return destinationMGRA; + } + + /** + * @param destinationMGRA + * the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) + { + this.destinationMGRA = destinationMGRA; + } + + public void setArriveTime(int arriveTime) + { + this.arriveTime = arriveTime; + } + + public int getArriveTime() + { + return arriveTime; + } + + /** + * @return the income + */ + public int getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(int income) + { + this.income = income; + } + + public int getAutos() + { + return autos; + } + + public void setAutos(int autos) + { + this.autos = autos; + } + + public int getAge() + { + return age; + } + + public void setAge(int age) + { + this.age = age; + } + + public int getFemale() + { + return female; + } + + public void setFemale(int female) + { + this.female = female; + } + + public int getHhID() { + return hhID; + } + + public void setHhID(int hhID) { + this.hhID = hhID; + } + + public int getPersonID() { + return personID; + } + + public void setPersonID(int personID) { + this.personID = personID; + } + + public int getPnum() { + return pnum; + } + + public void setPnum(int pnum) { + this.pnum = pnum; + } + + public double getNonWorkTimeFactor() { + return nonWorkTimeFactor; + } + + public void setNonWorkTimeFactor(double nonWorkTimeFactor) { + this.nonWorkTimeFactor = nonWorkTimeFactor; + } + + public boolean isAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(boolean avAvailable) { + this.avAvailable = avAvailable; + } + + public void logTourObject(Logger logger, int totalChars) + { + + Household.logHelper(logger, "tourId: ", ID, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); + Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java new file mode 100644 index 0000000..05886cc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java @@ -0,0 +1,114 @@ +package org.sandag.abm.internalexternal; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class InternalExternalTourDestChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("internalExternalModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + public InternalExternalTourDestChoiceDMU(InternalExternalModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + // methodIndexMap.put("getTimeOutbound", 0); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + /* + * + * switch (variableIndex) { + * + * case 0: returnValue = getTimeOutbound(); break; + * + * default: logger.error("method number = " + variableIndex + + * " not found"); throw new RuntimeException("method number = " + + * variableIndex + " not found"); + * + * } + */ + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java new file mode 100644 index 0000000..83e952a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java @@ -0,0 +1,172 @@ +package org.sandag.abm.internalexternal; + +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * This class is used for external station destination choice model for IE + * tours. + * + * + * @author Freedman + * + */ +public class InternalExternalTourDestChoiceModel +{ + + private transient Logger logger = Logger.getLogger("internalExternalModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication destModel; + + private HashMap rbMap; + + private InternalExternalTourDestChoiceDMU dcDmu; + + private Matrix tazProbabilities; + private TableDataSet altData; + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of airport model DMUs + */ + public InternalExternalTourDestChoiceModel(HashMap rbMap, + InternalExternalModelStructure modelStructure, InternalExternalDmuFactoryIf dmuFactory) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + + // initiate a DMU + dcDmu = dmuFactory.getInternalExternalTourDestChoiceDMU(); + + // create the full model UECs + // read the model pages from the property file, create one choice model + // for each full model + String internalExternalDCFileName = Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.dc.uec.file"); + internalExternalDCFileName = uecFileDirectory + internalExternalDCFileName; + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, + "internalExternal.dc.uec.data.page"); + int destModelPage = Util.getIntegerValueFromPropertyMap(rbMap, + "internalExternal.dc.uec.model.page"); + destModel = new ChoiceModelApplication(internalExternalDCFileName, destModelPage, dataPage, + rbMap, (VariableTable) dcDmu); + + } + + /** + * Calculate taz probabilities. This method initializes and calculates the + * tazProbabilities array. + */ + public void calculateTazProbabilities(InternalExternalDmuFactoryIf dmuFactory) + { + + logger.info("Calculating IE Model TAZ Probabilities Arrays"); + + // iterate through the alternatives in the alternatives file and set the + // size term and station logsum for each alternative + UtilityExpressionCalculator soaModelUEC = destModel.getUEC(); + altData = soaModelUEC.getAlternativeData(); + + // initialize the arrays + int maxTaz = tazManager.getMaxTaz(); + + tazProbabilities = new Matrix("Prob_Matrix", "Probability Matrix", maxTaz + 1, maxTaz + 1); + + // iterate through origin zones, solve the UEC and store the results in + // the matrix + for (int taz = 1; taz <= maxTaz; ++taz) + { + + int originTaz = taz; + + // set origin taz in dmu (destination set in UEC by alternative) + dcDmu.setDmuIndexValues(originTaz, originTaz, originTaz, originTaz, false); + + // Calculate utilities & probabilities + destModel.computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); + + // Store probabilities (by purpose) + double[] probabilities = destModel.getCumulativeProbabilities(); + + for (int i = 0; i < probabilities.length; ++i) + { + + double cumProb = probabilities[i]; + int destTaz = (int) altData.getValueAt(i + 1, "taz"); + tazProbabilities.setValueAt(originTaz, destTaz, (float) cumProb); + } + } + logger.info("Finished Calculating IE Model TAZ Probabilities Arrays"); + } + + /** + * Choose a destination TAZ and MGRA for the tour. + * + * @param tour + * An IE tour with a tour origin. + */ + public void chooseDestination(InternalExternalTour tour) + { + + double random = tour.getRandom(); + int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Choosing destination alternative"); + tour.logTourObject(logger, 1000); + + } + + // cycle through probability array for origin taz and find destination + // station & corresponding MGRA + int chosenTaz = -1; + int chosenMgra = -1; + for (int i = 1; i <= altData.getRowCount(); ++i) + { + int destTaz = (int) altData.getValueAt(i, "taz"); + if (random < tazProbabilities.getValueAt(originTaz, destTaz)) + { + chosenTaz = destTaz; + chosenMgra = (int) altData.getValueAt(i, "mgraOut"); + break; + } + } + + if (chosenTaz == -1) + { + logger.error("Error: IE Tour Destination Choice Model for tour " + tour.getID()); + throw new RuntimeException(); + } + + tour.setDestinationMGRA(chosenMgra); + tour.setDestinationTAZ(chosenTaz); + + if (tour.getDebugChoiceModels()) + logger.info("Chose taz " + chosenTaz + " mgra " + chosenMgra + " with random " + random); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java new file mode 100644 index 0000000..3d5d669 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java @@ -0,0 +1,377 @@ +package org.sandag.abm.internalexternal; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.util.ResourceUtil; + +public class InternalExternalTourManager +{ + + private static Logger logger = Logger.getLogger("internalExternalModel"); + + private InternalExternalTour[] tours; + public static final String PROPERTIES_DISTRIBUTED_TIME = "distributedTimeCoefficients"; + protected boolean readTimeFactors; + public static final String PERSON_TIMEFACTOR_NONWORK_FIELD_NAME = "timeFactorNonWork"; + + InternalExternalModelStructure modelStructure; + + TableDataSet personData; + + private boolean seek; + private int traceId; + + private MersenneTwister random; + + private class HouseholdClass + { + + int autos; + int income; + int homeMGRA; + int autonomousVehicles; + } + + private HashMap householdData; + + /** + * Constructor. Reads properties file and opens/stores all probability + * distributions for sampling. Estimates number of airport travel parties + * and initializes parties[]. + * + * @param resourceFile + * Property file. + * + * Creates the array of cross-border tours. + */ + public InternalExternalTourManager(HashMap rbMap, int iteration) + { + + modelStructure = new InternalExternalModelStructure(); + + // append _iteration to file + String iterationString = "_" + new Integer(iteration).toString(); + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + + String personFile = Util.getStringValueFromPropertyMap(rbMap, "Results.PersonDataFile"); + // Remove extension from filename + String extension = getFileExtension(personFile); + personFile = removeFileExtension(personFile) + iterationString + extension; + + personFile = directory + personFile; + + String householdFile = Util.getStringValueFromPropertyMap(rbMap, + "Results.HouseholdDataFile"); + + householdFile = directory + householdFile; + // Remove extension from filename + extension = getFileExtension(householdFile); + householdFile = removeFileExtension(householdFile) + iterationString + extension; + + readHouseholdFile(householdFile); + personData = readFile(personFile); + + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trace")); + + random = new MersenneTwister(1000001); + //check if we want to read distributed time factors from the person file + String readTimeFactorsString = rbMap.get(PROPERTIES_DISTRIBUTED_TIME); + if (readTimeFactorsString != null) + { + readTimeFactors = Boolean.valueOf(readTimeFactorsString); + logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); + } + + } + + /** + * Get the file extension + * + * @param fileName + * with the extension + * @return The extension + */ + public String getFileExtension(String fileName) + { + + int index = fileName.lastIndexOf("."); + int length = fileName.length(); + + String extension = fileName.substring(index, length); + + return extension; + + } + + /** + * Get the file name without the extension + * + * @param fileName + * The filename with the extension + * @return The filename without the extension + */ + public String removeFileExtension(String fileName) + { + int index = fileName.lastIndexOf("."); + String name = fileName.substring(0, index); + + return name; + + } + + /** + * Read household records and store autos owned. + * + * @param fileName + * household file path/name. + */ + public void readHouseholdFile(String fileName) + { + + householdData = new HashMap(); + + logger.info("Begin reading the data in file " + fileName); + + TableDataSet hhData; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + hhData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + // iterate through the table and save number of autos + for (int i = 1; i <= hhData.getRowCount(); ++i) + { + long hhID = (long) hhData.getValueAt(i, "hh_id"); + int autos = (int) hhData.getValueAt(i, "autos"); + int income = (int) hhData.getValueAt(i, "income"); + int mgra = (int) hhData.getValueAt(i, "home_mgra"); + + int AVs = (int) hhData.getValueAt(i,"AVs"); + + // new household + HouseholdClass hh = new HouseholdClass(); + hh.autos = autos; + hh.income = income; + hh.homeMGRA = mgra; + hh.autonomousVehicles = AVs; + + // store in HashMap + householdData.put(hhID, hh); + } + logger.info("End reading the data in file " + fileName); + } + + /** + * Read the file and return the TableDataSet. + * + * @param fileName + * @return data + */ + private TableDataSet readFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet data; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + data = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return data; + } + + /** + * Generate and attribute IE tours + */ + public void generateTours() + { + + ArrayList tourList = new ArrayList(); + + int rows = personData.getRowCount(); + + int tourCount = 0; + for (int i = 1; i <= rows; ++i) + { + + // TODO: generate IE tours here + if (((int) personData.getValueAt(i, "ie_choice")) == 2) + { + + InternalExternalTour tour = new InternalExternalTour(i + 100001); + tour.setID(i); + + // get the household for the person + long ID = (long) personData.getValueAt(i, "hh_id"); + HouseholdClass hh = householdData.get(ID); + tour.setHhID((int)ID); + + int pID = (int) personData.getValueAt(i, "person_id"); + tour.setPersonID(pID); + + int pnum=(int) personData.getValueAt(i, "person_num"); + tour.setPnum(pnum); + + int age = (int) personData.getValueAt(i, "age"); + String gender = (String) personData.getStringValueAt(i, "gender"); + + tour.setOriginMGRA(hh.homeMGRA); + tour.setIncome(hh.income); + tour.setAutos(hh.autos); + tour.setAge(age); + + if(hh.autonomousVehicles>0) + tour.setAvAvailable(true); + else + tour.setAvAvailable(false); + + if (gender.equals("f")) tour.setFemale(1); + else tour.setFemale(0); + + double timeFactorNonWork = 1.0; + if(readTimeFactors){ + timeFactorNonWork = (double) personData.getValueAt(i, + personData.getColumnPosition(PERSON_TIMEFACTOR_NONWORK_FIELD_NAME)); + } + tour.setNonWorkTimeFactor(timeFactorNonWork); + + tourList.add(tour); + + ++tourCount; + } + + } + if (tourList.isEmpty()) + { + logger.error("Internal-external tour list is empty!!"); + throw new RuntimeException(); + } + + tours = new InternalExternalTour[tourList.size()]; + for (int i = 0; i < tours.length; ++i) + tours[i] = tourList.get(i); + + logger.info("Total IE tours: " + tourCount); + + } + + /** + * Create a text file and write all records to the file. + * + */ + public void writeOutputFile(HashMap rbMap) + { + + // Open file and print header + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String tripFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trip.output.file"); + + logger.info("Writing IE trips to file " + tripFileName); + + PrintWriter tripWriter = null; + try + { + tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tripFileName + " for writing\n"); + throw new RuntimeException(); + } + String tripHeaderString = new String( + "hhID,pnum,personID,tourID,originMGRA,destinationMGRA,originTAZ,destinationTAZ,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,av_avail,boardingTap,alightingTap,set,valueOfTime\n"); + tripWriter.print(tripHeaderString); + + for (int i = 0; i < tours.length; ++i) + { + InternalExternalTrip[] trips = tours[i].getTrips(); + for (int j = 0; j < trips.length; ++j) + writeTrip(tours[i].getHhID(), tours[i].getPnum(),tours[i].getPersonID(), tours[i].getID(), tours[i],trips[j], tripWriter); + } + + tripWriter.close(); + + } + + /** + * Write the trip to the PrintWriter + * + * @param tour + * @param trip + * @param tripNumber + * @param writer + */ + private void writeTrip(int hhID, int pnum, int personID, int tourID, InternalExternalTour tour, InternalExternalTrip trip, PrintWriter writer) + { + + String record = new String(hhID+","+pnum+","+personID+","+tourID+","+trip.getOriginMgra() + "," + trip.getDestinationMgra() + "," + + trip.getOriginTaz() + "," + trip.getDestinationTaz() + "," + trip.isInbound() + + "," + trip.isOriginIsTourDestination() + "," + + trip.isDestinationIsTourDestination() + "," + trip.getPeriod() + "," + + trip.getTripMode() + "," + (tour.isAvAvailable() ? 1 : 0) + "," + + trip.getBoardTap() + "," + trip.getAlightTap() + "," + trip.getSet()+ "," + +String.format("%9.2f",trip.getValueOfTime()) + "\n"); + writer.print(record); + } + + /** + * @return the trips + */ + public InternalExternalTour[] getTours() + { + return tours; + } + + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running IE Model Trip Manager")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + InternalExternalTourManager apm = new InternalExternalTourManager(pMap, 1); + apm.generateTours(); + apm.writeOutputFile(pMap); + + logger.info("IE Trip Manager successfully completed!"); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java new file mode 100644 index 0000000..189a548 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java @@ -0,0 +1,182 @@ +package org.sandag.abm.internalexternal; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the TOD choice model for IE tours. It is currently based on a + * static probability distribution stored in an input file, and indexed into by + * purpose. Since there are no IE purposes, the purpose is 0. + * + * @author Freedman + * + */ +public class InternalExternalTourTimeOfDayChoiceModel +{ + private transient Logger logger = Logger.getLogger("internalExternalModel"); + + private double[][] cumProbability; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + private int[][] outboundPeriod; // by + // purpose, + // alternative: + // outbound + // period + private int[][] returnPeriod; // by + // purpose, + // alternative: + // return + // period + InternalExternalModelStructure modelStructure; + + /** + * Constructor. + */ + public InternalExternalTourTimeOfDayChoiceModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.tour.tod.file"); + stationDiurnalFile = directory + stationDiurnalFile; + + modelStructure = new InternalExternalModelStructure(); + + readTODFile(stationDiurnalFile); + + } + + /** + * Read the TOD distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readTODFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating tour TOD probability distribution"); + + int purposes = 1; // start at 0 + int periods = modelStructure.TIME_PERIODS; // start at 1 + int periodCombinations = periods * (periods + 1) / 2; + + cumProbability = new double[purposes][periodCombinations]; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + outboundPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // outbound + // period + returnPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // return period + + // fill up arrays + int rowCount = probabilityTable.getRowCount(); + int lastPurpose = -99; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); + int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); + + // continue if return period before outbound period + if (retPer < outPer) continue; + + // reset if new purpose + if (purpose != lastPurpose) + { + + // log cumulative probability just in case + if (lastPurpose != -99) + logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); + cumProb = 0; + alt = 0; + } + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + outboundPeriod[purpose][alt] = outPer; + returnPeriod[purpose][alt] = retPer; + + ++alt; + + lastPurpose = purpose; + } + + logger.info("End calculating tour TOD probability distribution"); + + } + + /** + * Calculate tour time of day for the tour. + * + * @param tour + * An IE tour + */ + public void calculateTourTOD(InternalExternalTour tour) + { + + int purpose = 0; + double random = tour.getRandom(); + + if (tour.getDebugChoiceModels()) + { + logger.info("Choosing tour time of day for tour ID " + tour.getID() + + " using random number " + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + if (random < cumProbability[purpose][i]) + { + int depart = outboundPeriod[purpose][i]; + int arrive = returnPeriod[purpose][i]; + tour.setDepartTime(depart); + tour.setArriveTime(arrive); + break; + } + } + + if (tour.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " + + tour.getArriveTime()); + logger.info(""); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java new file mode 100644 index 0000000..c347392 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java @@ -0,0 +1,313 @@ +package org.sandag.abm.internalexternal; + +import java.io.Serializable; +import org.sandag.abm.modechoice.MgraDataManager; + +public class InternalExternalTrip + implements Serializable +{ + + private int originMgra; + private int destinationMgra; + + + private int originTaz; + private int destinationTaz; + + private int tripMode; + private byte period; + private boolean inbound; + private boolean firstTrip; + private boolean lastTrip; + private boolean originIsTourDestination; + private boolean destinationIsTourDestination; + + private int boardTap; + private int alightTap; + private int set = -1; + + private double valueOfTime; + + /** + * Default constructor; nothing initialized. + */ + public InternalExternalTrip() + { + + } + + /** + * Create a cross border trip from a tour leg (no stops). + * + * @param tour + * The tour. + * @param outbound + * Outbound direction + */ + public InternalExternalTrip(InternalExternalTour tour, boolean outbound, + MgraDataManager mgraManager) + { + + initializeFromTour(tour, outbound, mgraManager); + } + + /** + * Initilize from the tour. + * + * @param tour + * The tour. + * @param outbound + * Outbound direction. + */ + public void initializeFromTour(InternalExternalTour tour, boolean outbound, + MgraDataManager mgraManager) + { + // Note: mode is unknown + if (outbound) + { + this.originMgra = tour.getOriginMGRA(); + this.originTaz = mgraManager.getTaz(tour.getOriginMGRA()); + this.destinationMgra = tour.getDestinationMGRA(); + this.destinationTaz = tour.getDestinationTAZ(); + this.period = (byte) tour.getDepartTime(); + this.inbound = false; + this.firstTrip = true; + this.lastTrip = false; + this.originIsTourDestination = false; + this.destinationIsTourDestination = true; + } else + { + this.originMgra = tour.getDestinationMGRA(); + this.originTaz = tour.getDestinationTAZ(); + this.destinationMgra = tour.getOriginMGRA(); + this.destinationTaz = mgraManager.getTaz(tour.getOriginMGRA()); + this.period = (byte) tour.getArriveTime(); + this.inbound = true; + this.firstTrip = false; + this.lastTrip = true; + this.originIsTourDestination = true; + this.destinationIsTourDestination = false; + } + + } + /** + * @param destinationTaz + * the destinationTaz to set + */ + public void setDestinationTaz(int destinationTaz) + { + this.destinationTaz = destinationTaz; + } + + + /** + * @return the period + */ + public byte getPeriod() + { + return period; + } + + /** + * @param period + * the period to set + */ + public void setPeriod(byte period) + { + this.period = period; + } + + /** + * @return the originMgra + */ + public int getOriginMgra() + { + return originMgra; + } + + /** + * @param originMgra + * the originMgra to set + */ + public void setOriginMgra(int originMgra) + { + this.originMgra = originMgra; + } + + /** + * @return the destinationMgra + */ + public int getDestinationMgra() + { + return destinationMgra; + } + + /** + * @param destinationMgra + * the destinationMgra to set + */ + public void setDestinationMgra(int destinationMgra) + { + this.destinationMgra = destinationMgra; + } + + /** + * @return the tripMode + */ + public int getTripMode() + { + return tripMode; + } + + /** + * @param tripMode + * the tripMode to set + */ + public void setTripMode(int tripMode) + { + this.tripMode = tripMode; + } + public int getBoardTap() { + return boardTap; + } + + public void setBoardTap(int boardTap) { + this.boardTap = boardTap; + } + + public int getAlightTap() { + return alightTap; + } + + public void setAlightTap(int alightTap) { + this.alightTap = alightTap; + } + + public int getSet() { + return set; + } + + public void setSet(int set) { + this.set = set; + } + + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the firstTrip + */ + public boolean isFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(boolean firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public boolean isLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(boolean lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the originIsTourDestination + */ + public boolean isOriginIsTourDestination() + { + return originIsTourDestination; + } + + /** + * @param originIsTourDestination + * the originIsTourDestination to set + */ + public void setOriginIsTourDestination(boolean originIsTourDestination) + { + this.originIsTourDestination = originIsTourDestination; + } + + /** + * @return the destinationIsTourDestination + */ + public boolean isDestinationIsTourDestination() + { + return destinationIsTourDestination; + } + + /** + * @param destinationIsTourDestination + * the destinationIsTourDestination to set + */ + public void setDestinationIsTourDestination(boolean destinationIsTourDestination) + { + this.destinationIsTourDestination = destinationIsTourDestination; + } + + /** + * @return the originTaz + */ + public int getOriginTaz() + { + return originTaz; + } + + /** + * @param originTaz + * the originTaz to set + */ + public void setOriginTaz(int originTaz) + { + this.originTaz = originTaz; + } + + /** + * @return the destinationTaz + */ + public int getDestinationTaz() + { + return destinationTaz; + } + + public double getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(double valueOfTime) { + this.valueOfTime = valueOfTime; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java new file mode 100644 index 0000000..1f1f864 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java @@ -0,0 +1,551 @@ +package org.sandag.abm.internalexternal; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class InternalExternalTripModeChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(InternalExternalTripModeChoiceDMU.class); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + + protected int tourDepartPeriod; + protected int tourArrivePeriod; + protected int tripPeriod; + protected int outboundStops; + protected int returnStops; + protected int firstTrip; + protected int lastTrip; + + protected int income; + protected int female; + protected int age; + protected int autos; + protected int hhSize; + protected int tripOrigIsTourDest; + protected int tripDestIsTourDest; + + protected double nonWorkTimeFactor; + + protected double nmWalkTime; + protected double nmBikeTime; + + + protected double ivtCoeff; + protected double costCoeff; + + protected double walkTransitLogsum; + protected double pnrTransitLogsum; + protected double knrTransitLogsum; + + protected int outboundHalfTourDirection; + + protected int avAvailable; + + public InternalExternalTripModeChoiceDMU(InternalExternalModelStructure modelStructure, + Logger aLogger) + { + if (aLogger == null) + { + aLogger = Logger.getLogger("internalExternalModel"); + } + logger = aLogger; + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the tripPeriod + */ + public int getTripPeriod() + { + return tripPeriod; + } + + /** + * @param tripPeriod + * the tripPeriod to set + */ + public void setTripPeriod(int tripPeriod) + { + this.tripPeriod = tripPeriod; + } + + /** + * @return the outboundStops + */ + public int getOutboundStops() + { + return outboundStops; + } + + /** + * @param outboundStops + * the outboundStops to set + */ + public void setOutboundStops(int outboundStops) + { + this.outboundStops = outboundStops; + } + + /** + * @return the returnStops + */ + public int getReturnStops() + { + return returnStops; + } + + /** + * @param returnStops + * the returnStops to set + */ + public void setReturnStops(int returnStops) + { + this.returnStops = returnStops; + } + + /** + * @return the firstTrip + */ + public int getFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(int firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public int getLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(int lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the tripOrigIsTourDest + */ + public int getTripOrigIsTourDest() + { + return tripOrigIsTourDest; + } + + /** + * @param tripOrigIsTourDest + * the tripOrigIsTourDest to set + */ + public void setTripOrigIsTourDest(int tripOrigIsTourDest) + { + this.tripOrigIsTourDest = tripOrigIsTourDest; + } + + /** + * @return the tripDestIsTourDest + */ + public int getTripDestIsTourDest() + { + return tripDestIsTourDest; + } + + /** + * @param tripDestIsTourDest + * the tripDestIsTourDest to set + */ + public void setTripDestIsTourDest(int tripDestIsTourDest) + { + this.tripDestIsTourDest = tripDestIsTourDest; + } + + /** + * @return the outboundHalfTourDirection + */ + public int getOutboundHalfTourDirection() + { + return outboundHalfTourDirection; + } + + /** + * @param outboundHalfTourDirection + * the outboundHalfTourDirection to set + */ + public void setOutboundHalfTourDirection(int outboundHalfTourDirection) + { + this.outboundHalfTourDirection = outboundHalfTourDirection; + } + + /** + * @return the tourDepartPeriod + */ + public int getTourDepartPeriod() + { + return tourDepartPeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(int tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(int tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + /** + * @return the tourArrivePeriod + */ + public int getTourArrivePeriod() + { + return tourArrivePeriod; + } + + public double getNm_walkTime() + { + return nmWalkTime; + } + + public void setNonMotorizedWalkTime(double nmWalkTime) + { + this.nmWalkTime = nmWalkTime; + } + + public void setNonMotorizedBikeTime(double nmBikeTime) + { + this.nmBikeTime = nmBikeTime; + } + + public double getNm_bikeTime() + { + return nmBikeTime; + } + + /** + * @return the income + */ + public int getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(int income) + { + this.income = income; + } + + public int getFemale() + { + return female; + } + + public void setFemale(int female) + { + this.female = female; + } + + public int getAge() + { + return age; + } + + public void setAge(int age) + { + this.age = age; + } + + public int getAutos() + { + return autos; + } + + public void setAutos(int autos) + { + this.autos = autos; + } + + public int getHhSize() + { + return hhSize; + } + + public void setHhSize(int hhSize) + { + this.hhSize = hhSize; + } + public double getNonWorkTimeFactor(){ + return nonWorkTimeFactor; + } + + public void setNonWorkTimeFactor(double nonWorkTimeFactor){ + this.nonWorkTimeFactor=nonWorkTimeFactor; + } + + public double getIvtCoeff() { + return ivtCoeff; + } + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public double getCostCoeff() { + return costCoeff; + } + + public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; + } + + public double getWalkTransitLogsum() { + return walkTransitLogsum; + } + + public void setWalkTransitLogsum(double walkTransitLogsum) { + this.walkTransitLogsum = walkTransitLogsum; + } + + public double getPnrTransitLogsum() { + return pnrTransitLogsum; + } + + public void setPnrTransitLogsum(double pnrTransitLogsum) { + this.pnrTransitLogsum = pnrTransitLogsum; + } + + public double getKnrTransitLogsum() { + return knrTransitLogsum; + } + + public void setKnrTransitLogsum(double knrTransitLogsum) { + this.knrTransitLogsum = knrTransitLogsum; + } + + + public int getAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(int avAvailable) { + this.avAvailable = avAvailable; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTourDepartPeriod", 0); + methodIndexMap.put("getTourArrivePeriod", 1); + methodIndexMap.put("getTripPeriod", 2); + methodIndexMap.put("getOutboundStops", 5); + methodIndexMap.put("getReturnStops", 6); + methodIndexMap.put("getFirstTrip", 7); + methodIndexMap.put("getLastTrip", 8); + methodIndexMap.put("getIncome", 9); + methodIndexMap.put("getFemale", 10); + methodIndexMap.put("getAutos", 11); + methodIndexMap.put("getHhSize", 12); + methodIndexMap.put("getAge", 13); + methodIndexMap.put("getNonWorkTimeFactor", 14); + + methodIndexMap.put("getTripOrigIsTourDest", 23); + methodIndexMap.put("getTripDestIsTourDest", 24); + + methodIndexMap.put("getIvtCoeff", 60); + methodIndexMap.put("getCostCoeff", 61); + + methodIndexMap.put("getWalkSetLogSum", 62); + methodIndexMap.put("getPnrSetLogSum", 63); + methodIndexMap.put("getKnrSetLogSum", 64); + + methodIndexMap.put("getAvAvailable",70); + + methodIndexMap.put("getNm_walkTime", 90); + methodIndexMap.put("getNm_bikeTime", 91); + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTourDepartPeriod(); + break; + case 1: + returnValue = getTourArrivePeriod(); + break; + case 2: + returnValue = getTripPeriod(); + break; + case 5: + returnValue = getOutboundStops(); + break; + case 6: + returnValue = getReturnStops(); + break; + case 7: + returnValue = getFirstTrip(); + break; + case 8: + returnValue = getLastTrip(); + break; + case 9: + returnValue = getIncome(); + break; + case 10: + returnValue = getFemale(); + break; + case 11: + returnValue = getAutos(); + break; + case 12: + returnValue = getHhSize(); + break; + case 13: + returnValue = getAge(); + break; + case 14: + returnValue = getNonWorkTimeFactor(); + break; + case 23: + returnValue = getTripOrigIsTourDest(); + break; + case 24: + returnValue = getTripDestIsTourDest(); + break; + + case 60: + returnValue = getIvtCoeff(); + break; + case 61: + returnValue = getCostCoeff(); + break; + case 62: + returnValue = getWalkTransitLogsum(); + break; + case 63: + returnValue = getPnrTransitLogsum(); + break; + case 64: + returnValue = getKnrTransitLogsum(); + break; + case 70: + returnValue = getAvAvailable(); + break; + case 90: + returnValue = getNm_walkTime(); + break; + case 91: + returnValue = getNm_bikeTime(); + break; + default: + logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + } + + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java new file mode 100644 index 0000000..f9dc39d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java @@ -0,0 +1,277 @@ +package org.sandag.abm.internalexternal; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.airport.AirportModelStructure; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class InternalExternalTripModeChoiceModel +{ + + private transient Logger logger = Logger.getLogger("internalExternalModel"); + + private AutoTazSkimsCalculator tazDistanceCalculator; + + private McLogsumsCalculator logsumHelper; + private InternalExternalModelStructure modelStructure; + private TazDataManager tazs; + private MgraDataManager mgraManager; + private InternalExternalTripModeChoiceDMU dmu; + private ChoiceModelApplication tripModeChoiceModel; + double logsum = 0; + + private static final String PROPERTIES_UEC_DATA_SHEET = "internalExternal.trip.mc.data.page"; + private static final String PROPERTIES_UEC_MODEL_SHEET = "internalExternal.trip.mc.model.page"; + private static final String PROPERTIES_UEC_FILE = "internalExternal.trip.mc.uec.file"; + private TripModeChoiceDMU mcDmuObject; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public InternalExternalTripModeChoiceModel(HashMap propertyMap, + InternalExternalModelStructure myModelStructure, + InternalExternalDmuFactoryIf dmuFactory) + { + tazs = TazDataManager.getInstance(propertyMap); + mgraManager = MgraDataManager.getInstance(propertyMap); + + modelStructure = myModelStructure; + + tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + SandagModelStructure modelStructure = new SandagModelStructure(); + mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); + + setupTripModeChoiceModel(propertyMap, dmuFactory); + + } + + /** + * Read the UEC file and set up the trip mode choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupTripModeChoiceModel(HashMap propertyMap, + InternalExternalDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up IE trip mode choice model.")); + + dmu = dmuFactory.getInternalExternalTripModeChoiceDMU(); + + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_DATA_SHEET)); + int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_MODEL_SHEET)); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); + tripModeUecFile = uecPath + tripModeUecFile; + + tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, + propertyMap, (VariableTable) dmu); + + } + + /** + * Calculate utilities and return logsum for the tour and stop. + * + * @param tour + * @param trip + */ + public double computeUtilities(InternalExternalTour tour, InternalExternalTrip trip) + { + + setDmuAttributes(tour, trip); + + tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + tour.logTourObject(logger, 100); + tripModeChoiceModel.logUECResults(logger, "IE trip mode choice model"); + + } + + logsum = tripModeChoiceModel.getLogsum(); + + if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); + + return logsum; + + } + + /** + * Choose a mode and store in the trip object. + * + * @param tour + * InternalExternalTour + * @param trip + * InternalExternalTrip + * + */ + public void chooseMode(InternalExternalTour tour, InternalExternalTrip trip) + { + + computeUtilities(tour, trip); + + double rand = tour.getRandom(); + int mode = tripModeChoiceModel.getChoiceResult(rand); + + trip.setTripMode(mode); + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + + double vot = 0.0; + + if(modelStructure.getTripModeIsS2(mode)){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = uec.getValueForIndex(votIndex); + }else if (modelStructure.getTripModeIsS3(mode)){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = uec.getValueForIndex(votIndex); + } + trip.setValueOfTime(vot); + + + if(modelStructure.getTripModeIsTransit(mode)){ + double[][] bestTapPairs = null; + + if (modelStructure.getTripModeIsWalkTransit(mode)){ + bestTapPairs = logsumHelper.getBestWtwTripTaps(); + } + else if (modelStructure.getTripModeIsPnrTransit(mode)||modelStructure.getTripModeIsKnrTransit(mode)){ + if (!trip.isInbound()) + bestTapPairs = logsumHelper.getBestDtwTripTaps(); + else + bestTapPairs = logsumHelper.getBestWtdTripTaps(); + } + double rn = tour.getRandom(); + int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); + int boardTap = (int) bestTapPairs[pathIndex][0]; + int alightTap = (int) bestTapPairs[pathIndex][1]; + int set = (int) bestTapPairs[pathIndex][2]; + trip.setBoardTap(boardTap); + trip.setAlightTap(alightTap); + trip.setSet(set); + } + + + + + + } + + /** + * Set DMU attributes. + * + * @param tour + * @param trip + */ + public void setDmuAttributes(InternalExternalTour tour, InternalExternalTrip trip) + { + + int tripOriginTaz = trip.getOriginTaz(); + int tripDestinationTaz = trip.getDestinationTaz(); + + dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, + tour.getDebugChoiceModels()); + + dmu.setTourDepartPeriod(tour.getDepartTime()); + dmu.setTourArrivePeriod(tour.getArriveTime()); + dmu.setTripPeriod(trip.getPeriod()); + + dmu.setAutos(tour.getAutos()); + dmu.setIncome(tour.getIncome()); + dmu.setAge(tour.getAge()); + dmu.setFemale(tour.getFemale()); + + dmu.setNonWorkTimeFactor(tour.getNonWorkTimeFactor()); + + // set trip mc dmu values for transit logsum (gets replaced below by uec values) + double c_ivt = -0.03; + double c_cost = - 0.003; + + // Solve trip mode level utilities + mcDmuObject.setIvtCoeff(c_ivt * tour.getNonWorkTimeFactor()); + mcDmuObject.setCostCoeff(c_cost); + + dmu.setIvtCoeff(c_ivt * tour.getNonWorkTimeFactor()); + dmu.setCostCoeff(c_cost); + double walkTransitLogsum = -999.0; + double driveTransitLogsum = -999.0; + + logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); + dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); + + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); + walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + dmu.setWalkTransitLogsum(walkTransitLogsum); + if (!trip.isInbound()) + { + logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); + } else + { + logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); + } + + dmu.setPnrTransitLogsum(driveTransitLogsum); + dmu.setKnrTransitLogsum(driveTransitLogsum); + + dmu.setOutboundStops(tour.getNumberInboundStops()); + dmu.setReturnStops(tour.getNumberInboundStops()); + + if (trip.isFirstTrip()) dmu.setFirstTrip(1); + else dmu.setFirstTrip(0); + + if (trip.isLastTrip()) dmu.setLastTrip(1); + else dmu.setLastTrip(0); + + if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); + else dmu.setTripOrigIsTourDest(0); + + if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); + else dmu.setTripDestIsTourDest(0); + + if(tour.isAvAvailable()) + dmu.setAvAvailable(1); + else + dmu.setAvAvailable(0); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java new file mode 100644 index 0000000..3594539 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java @@ -0,0 +1,698 @@ +package org.sandag.abm.internalexternal; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class InternalExternalTripTables +{ + + private static Logger logger = Logger.getLogger("tripTables"); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TableDataSet tripData; + + // Some parameters + private int[] modeIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // total + // modes, + // returns + // 0=auto + // modes, + // 1=non-motor, + // 2=transit, + // 3= + // other + private int[] matrixIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // modes, + // returns + // the + // element + // of + // the + // matrix + // array + // to + // store + // value + + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private HashMap rbMap; + private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; + private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; + + // matrices are indexed by modes, votbins, tables + private Matrix[][][] matrix; + private float averageOcc3Plus = 3.5f; + + private ResourceBundle rb; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + + private MatrixDataServerRmi ms; + private float sampleRate = 1; + private static int iteration=1; + private float valueOfTimeThresholdLow = 0; + private float valueOfTimeThresholdMed = 0; + //value of time bins by mode group + int[] votBins = {3,1,1,1}; + + public int numSkimSets; + + + /** + * @return the sampleRate + */ + public float getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(float sampleRate) + { + this.sampleRate = sampleRate; + } + + public InternalExternalTripTables(HashMap rbMap) + { + + this.rbMap = rbMap; + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTourModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + } else if (modelStructure.getTourModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + } else if (modelStructure.getTourModeIsWalkTransit(i) + || modelStructure.getTourModeIsDriveTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + } + } + //value of time thresholds + valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); + valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][][]; + for (int i = 0; i < numberOfModes; ++i) + { + matrix[i] = new Matrix[votBins[i]][]; + + String modeName; + for(int j = 0; j< votBins[i];++j){ + + if (i == 0) + { + matrix[i][j] = new Matrix[autoModes]; + for (int k = 0; k < autoModes; ++k) + { + modeName = modelStructure.getModeName(k + 1); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 1) + { + matrix[i][j] = new Matrix[nmotModes]; + for (int k = 0; k < nmotModes; ++k) + { + modeName = modelStructure.getModeName(k + 1 + autoModes); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 2) + { + matrix[i][j] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l1) + votBin = getValueOfTimeBin(valueOfTime); + + if (mode == 0) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); + } else if (mode == 1) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } else if (mode == 2) + { + + if (boardTap == 0 || alightTap == 0) continue; + + //store transit trips in matrices + mat = (matrixIndex[tripMode]*numSkimSets)+set; + float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); + matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); + + // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) + if (modelStructure.getTourModeIsDriveTransit(tripMode)) + { + + // add the tNCVehicle trip portion to the trip table + if (!inbound) + { // from origin to lot (boarding tap) + int PNRTAZ = tapManager.getTazForTap(boardTap); + value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); + matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); + + } else + { // from lot (alighting tap) to destination + int PNRTAZ = tapManager.getTazForTap(alightTap); + value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); + matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); + } + + } + } else + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } + + //logger.info("End creating trip tables for period " + timePeriod); + } + } + + /** + * Get the output trip table file names from the properties file, and write + * trip tables for all modes for the given time period. + * + * @param period + * Time period, which will be used to find the period time string + * to append to each trip table matrix file + */ + public void writeTrips(int period, MatrixType mt) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "scenario.path"); + String per = modelStructure.getModelPeriodLabel(period); + String[][] end = new String[4][]; + String[] fileName = new String[4]; + + fileName[0] = directory + + Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.results.autoTripMatrix"); + fileName[1] = directory + + Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.results.nMotTripMatrix"); + fileName[2] = directory + + Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.results.tranTripMatrix"); + fileName[3] = directory + + Util.getStringValueFromPropertyMap(rbMap, + "internalExternal.results.othrTripMatrix"); + + //the end of the name depends on whether there are multiple vot bins or not + String[] votBinName = {"low","med","high"}; + + for(int i = 0; i<4;++i){ + end[i] = new String[votBins[i]]; + for(int j = 0; j < votBins[i];++j){ + if(votBins[i]>1) + end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; + else + end[i][j] = "_" + per + ".omx"; + } + } + + for (int i = 0; i < 4; ++i) + { + for(int j = 0; j < votBins[i];++j){ + try + { + //Delete the file if it exists + File f = new File(fileName[i]+end[i][j]); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); + f.delete(); + } + + if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); + else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); + } catch (Exception e) + { + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName[i] +end[i][j] + ", for mode index = " + i, e); + throw new RuntimeException(); + } + } + } + } + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + /** + * Start matrix server + * + * @param serverAddress + * @param serverPort + * @param mt + * @return + */ + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @param args + */ + public static void main(String[] args) + { + + HashMap pMap; + String propertiesFile = null; + + logger.info(String.format( + "SANDAG IE Model Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + InternalExternalTripTables tripTables = new InternalExternalTripTables(pMap); + float sampleRate = 1.0f; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + + logger.info("IE Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + tripTables.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + tripTables.ms = matrixServer; + } else + { + tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + tripTables.ms.testRemote("InternalExternalTripTables"); + + // mdm = MatrixDataManager.getInstance(); + // mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + tripTables.createTripTables(mt); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java new file mode 100644 index 0000000..120dbef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java @@ -0,0 +1,1496 @@ +package org.sandag.abm.maas; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Iterator; +import java.util.Set; +import java.util.StringTokenizer; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.PropertyMap; + +public class HouseholdAVAllocationManager { + + HashMap householdMap; //by hh_id + private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModelRunner.class); + protected HashMap propertyMap = null; + protected MersenneTwister random; + protected ModelStructure modelStructure; + protected int iteration; + protected static final String ModelSeedProperty = "Model.Random.Seed"; + protected static final String DirectoryProperty = "Project.Directory"; + protected static final String HouseholdDataFileProperty = "Results.HouseholdDataFile"; + protected static final String PersonDataFileProperty = "Results.PersonDataFile"; + protected static final String IndivTripDataFileProperty = "Results.IndivTripDataFile"; + protected static final String JointTripDataFileProperty = "Results.JointTripDataFile"; + protected static final String VEHICLETRIP_OUTPUT_FILE_PROPERTY = "Maas.AVAllocationModel.vehicletrip.output.file"; + protected static final String VEHICLETRIP_OUTPUT_MATRIX_PROPERTY = "Maas.AVAllocationModel.vehicletrip.output.matrix"; + protected static final String REMOTE_PARKING_COST_PROPERTY = "Mobility.AV.RemoteParkingCostPerHour"; + + protected HashSet householdTraceSet; + public static final String PROPERTIES_HOUSEHOLD_TRACE_LIST = "Debug.Trace.HouseholdIdList"; + // one file per time period + // matrices are indexed by periods + private Matrix[] emptyVehicleTripMatrix; + MgraDataManager mazManager; + TazDataManager tazManager; + + protected static final int[] AutoModes = {1,2,3}; + protected static final int MaxAutoMode = 3; + private long randomSeed = 198761; + protected String vehicleTripOutputFile; + + protected float remoteParkingCostAtDest; + + boolean sortByPerson; + + HashMap personTypeMap; + String[] personTypes = {"Full-time worker","Part-time worker","University student", + "Non-worker", "Retired","Student of driving age","Student of non-driving age", + "Child too young for school"}; + + class Household { + + HashMap personMap; //by person_num + int id; + int homeMaz; + int income; + int autos; + int HVs; + int AVs; + ArrayList trips; + ArrayList autonomousVehicles; + int seed; + boolean debug; + + public void writeDebug(Logger logger, boolean logAVs) { + + logger.info("******** HH DEBUG **************"); + logger.info("HH ID: "+ id); + logger.info("Home MAZ: "+homeMaz); + logger.info("Income: "+income); + logger.info("Autos: "+ autos); + logger.info("HVs: "+HVs); + logger.info("AVs: "+AVs); + logger.info("Seed: "+seed); + + //log persons + if(personMap.size()==0) { + logger.info(" No persons to log"); + }else { + Set keySet = personMap.keySet(); + for(Integer key: keySet) { + Person person = personMap.get(key); + person.writeDebug(logger); + } + } + + //log trips + if(trips.size()==0) { + logger.info(" No trips to log"); + }else { + + for(int i=0;i0 && logAVs) { + for(int i=0;i(); + personMap = new HashMap(); + autonomousVehicles = new ArrayList(); + + } + + public ArrayList getTrips() { + return trips; + } + public void setTrips(ArrayList trips) { + this.trips = trips; + } + public int getId() { + return id; + } + + public void setId(int id) { + this.id = id; + } + + public int getHomeMaz() { + return homeMaz; + } + + public void setHomeMaz(int homeMaz) { + this.homeMaz = homeMaz; + } + + public int getIncome() { + return income; + } + + public void setIncome(int income) { + this.income = income; + } + + public int getAutos() { + return autos; + } + + public void setAutos(int autos) { + this.autos = autos; + } + + public int getHVs() { + return HVs; + } + + public void setHVs(int hVs) { + HVs = hVs; + } + + public int getAVs() { + return AVs; + } + + public void setAVs(int aVs) { + AVs = aVs; + } + + public ArrayList getAutonomousVehicles() { + return autonomousVehicles; + } + + public void setAutonomousVehicles(ArrayList autonomousVehicles) { + this.autonomousVehicles = autonomousVehicles; + } + + public int getSeed() { + return seed; + } + + public void setSeed(int seed) { + this.seed = seed; + } + + public boolean isDebug() { + return debug; + } + + public void setDebug(boolean debug) { + this.debug = debug; + } + + + } + + class Person { + + int hh_id; + int person_id; + int person_num; + int age; + int gender; + int type; + float value_of_time; + float reimb_pct; + float timeFactorWork; + float timeFactorNonWork; + int fp_choice; + + public void writeDebug(Logger logger) { + + logger.info("******** PERSON DEBUG **************"); + logger.info("HH ID: "+ hh_id); + logger.info("Person ID: "+person_id); + logger.info("Person Num: "+person_num); + logger.info("Age: "+age); + logger.info("Gender: "+gender); + logger.info("Type: "+type); + logger.info("Value of time: "+value_of_time); + logger.info("Reimb percent: "+reimb_pct); + logger.info("Time factor work: "+timeFactorWork); + logger.info("Time factor nonwork: "+timeFactorNonWork); + logger.info("Free parking choice: "+fp_choice); + + } + + + public int getHh_id() { + return hh_id; + } + public void setHh_id(int hh_id) { + this.hh_id = hh_id; + } + public int getPerson_id() { + return person_id; + } + public void setPerson_id(int person_id) { + this.person_id = person_id; + } + public int getPerson_num() { + return person_num; + } + public void setPerson_num(int person_num) { + this.person_num = person_num; + } + public int getAge() { + return age; + } + public void setAge(int age) { + this.age = age; + } + public int getGender() { + return gender; + } + public void setGender(int gender) { + this.gender = gender; + } + public int getType() { + return type; + } + public void setType(int type) { + this.type = type; + } + public float getValue_of_time() { + return value_of_time; + } + public void setValue_of_time(float value_of_time) { + this.value_of_time = value_of_time; + } + public float getReimb_pct() { + return reimb_pct; + } + public void setReimb_pct(float reimb_pct) { + this.reimb_pct = reimb_pct; + } + public float getTimeFactorWork() { + return timeFactorWork; + } + public void setTimeFactorWork(float timeFactorWork) { + this.timeFactorWork = timeFactorWork; + } + public float getTimeFactorNonWork() { + return timeFactorNonWork; + } + public void setTimeFactorNonWork(float timeFactorNonWork) { + this.timeFactorNonWork = timeFactorNonWork; + } + public int getFp_choice() { + return fp_choice; + } + public void setFp_choice(int fp_choice) { + this.fp_choice = fp_choice; + } + } + + class Trip implements Comparable{ + + int hh_id; + int person_id; + int person_num; + int tour_id; + int stop_id; + int inbound; + String tour_purpose; + String orig_purpose; + String dest_purpose; + int orig_maz; + int dest_maz; + int parking_maz; + int stop_period; + int periodsUntilNextTrip; + int trip_mode; + int av_avail; + int tour_mode; + int driver_pnum; + float valueOfTime; + int transponder_avail; + int num_participants; //for joint trips + ArrayList persons; + int veh_used; //the number of the vehicle used (1,2,3 or 0 for no AV used) + + public Trip() { + + tour_purpose=""; + orig_purpose=""; + dest_purpose=""; + + } + public void writeDebug(Logger logger) { + + logger.info("******** TRIP DEBUG *******"); + logger.info("HH ID: "+ hh_id); + logger.info("Person ID: "+person_id); + logger.info("Person Num: "+person_num); + logger.info("Tour ID: "+tour_id); + logger.info("Stop ID: "+stop_id); + logger.info("Inbound: "+inbound); + logger.info("Tour purpose: "+tour_purpose); + logger.info("Orig purpose: "+orig_purpose); + logger.info("Dest purpose: "+dest_purpose); + logger.info("Orig MAZ : "+orig_maz); + logger.info("Dest MAZ : "+dest_maz); + logger.info("Parking MAZ: "+parking_maz); + logger.info("Stop Period: "+stop_period); + logger.info("Periods Until Next Trip: "+periodsUntilNextTrip); + logger.info("Trip mode: "+trip_mode); + logger.info("AV avail: "+av_avail); + logger.info("Tour mode: "+tour_mode); + logger.info("Driver pnum: "+driver_pnum); + logger.info("Value Of Time: "+valueOfTime); + logger.info("Transponder Avail: "+transponder_avail); + logger.info("Num participants: "+num_participants); //for joint trips + logger.info("Veh used: "+veh_used); //the number of the vehicle used (1,2,3 or 0 for no AV used) + + } + + + /** + * Return true if its the same person and the same tour id and purpose + * @param thatTrip + * @return true or false + */ + public boolean sameTour(Trip thatTrip) { + + if((person_id==thatTrip.getPerson_id()) && + (tour_id==thatTrip.getTour_id()) && + (num_participants==thatTrip.getNum_participants()) && + (tour_purpose.compareTo(thatTrip.getTour_purpose())==0)) + return true; + return false; + } + + @Override + public int compareTo(Trip thatTrip) { + + if (this == thatTrip) return 0; + + if(sortByPerson) { + + if(person_idthatTrip.getPerson_id()) + return 1; + else if(person_id==thatTrip.getPerson_id()) { + if(stop_periodthatTrip.getStop_period()) + return 1; + else if(stop_period==thatTrip.getStop_period()) { + if(tour_purpose.compareTo("Work")==0 && thatTrip.getTour_purpose().compareTo("Work-Based")==0) { + if(inbound==0) + return -1; + else + return 1; + }else if(tour_purpose.compareTo("Work-based")==0 && thatTrip.getTour_purpose().compareTo("Work")==0) { + if(thatTrip.getInbound()==0) + return 1; + else + return -1; + } + if(tour_idthatTrip.getTour_id()) + return 1; + if(inbound==0 && thatTrip.getInbound()==1) + return -1; + } + + } + return 0; + } + + /* + //if its the same person and the same tour, use the stop ID + if(sameTour(thatTrip)){ + int thisTourTripSeq = inbound * 1000+stop_id; + int thatTourTripSeq = thatTrip.getInbound() * 1000+thatTrip.getStop_id(); + + if(thisTourTripSeqthatTourTripSeq) + return 1; + else + return 0; + } + + //its not the same tour + */ if(stop_periodthatTrip.getStop_period()) + return 1; + else if(stop_period==thatTrip.getStop_period()) { //its the same stop period + if((person_id==thatTrip.getPerson_id())) { //same person + if(tour_purpose.compareTo("Work")==0 && thatTrip.getTour_purpose().compareTo("Work-Based")==0) { + if(inbound==0) + return -1; + else + return 1; + }else if(tour_purpose.compareTo("Work-based")==0 && thatTrip.getTour_purpose().compareTo("Work")==0) { + if(thatTrip.getInbound()==0) + return 1; + else + return -1; + } + + } + /* + * if(periodsUntilNextTripthatTrip.getPeriodsUntilNextTrip()) + return 1; + */ + } + + return 0; + } + + public int getHh_id() { + return hh_id; + } + public void setHh_id(int hh_id) { + this.hh_id = hh_id; + } + public int getPerson_id() { + return person_id; + } + public void setPerson_id(int person_id) { + this.person_id = person_id; + } + public int getPerson_num() { + return person_num; + } + public void setPerson_num(int person_num) { + this.person_num = person_num; + } + public int getTour_id() { + return tour_id; + } + public void setTour_id(int tour_id) { + this.tour_id = tour_id; + } + public int getStop_id() { + return stop_id; + } + public void setStop_id(int stop_id) { + this.stop_id = stop_id; + } + public int getInbound() { + return inbound; + } + public void setInbound(int inbound) { + this.inbound = inbound; + } + public String getTour_purpose() { + return tour_purpose; + } + public void setTour_purpose(String tour_purpose) { + this.tour_purpose = tour_purpose; + } + public String getOrig_purpose() { + return orig_purpose; + } + public void setOrig_purpose(String orig_purpose) { + this.orig_purpose = orig_purpose; + } + public String getDest_purpose() { + return dest_purpose; + } + public void setDest_purpose(String dest_purpose) { + this.dest_purpose = dest_purpose; + } + public int getOrig_maz() { + return orig_maz; + } + public void setOrig_maz(int orig_maz) { + this.orig_maz = orig_maz; + } + public int getDest_maz() { + return dest_maz; + } + public void setDest_maz(int dest_maz) { + this.dest_maz = dest_maz; + } + public int getParking_maz() { + return parking_maz; + } + public void setParking_maz(int parking_maz) { + this.parking_maz = parking_maz; + } + public int getStop_period() { + return stop_period; + } + public void setStop_period(int stop_period) { + this.stop_period = stop_period; + } + public int getTrip_mode() { + return trip_mode; + } + public void setTrip_mode(int trip_mode) { + this.trip_mode = trip_mode; + } + public int getAv_avail() { + return av_avail; + } + public void setAv_avail(int av_avail) { + this.av_avail = av_avail; + } + public int getTour_mode() { + return tour_mode; + } + public void setTour_mode(int tour_mode) { + this.tour_mode = tour_mode; + } + public int getDriver_pnum() { + return driver_pnum; + } + public void setDriver_pnum(int driver_pnum) { + this.driver_pnum = driver_pnum; + } + public float getValueOfTime() { + return valueOfTime; + } + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + public int getTransponder_avail() { + return transponder_avail; + } + public void setTransponder_avail(int transponder_avail) { + this.transponder_avail = transponder_avail; + } + public int getNum_participants() { + return num_participants; + } + public void setNum_participants(int num_participants) { + this.num_participants = num_participants; + } + public ArrayList getPersons() { + return persons; + } + public void setPersons(ArrayList persons) { + this.persons = persons; + } + + public int getPeriodsUntilNextTrip() { + return periodsUntilNextTrip; + } + + public void setPeriodsUntilNextTrip(int periodsUntilNextTrip) { + this.periodsUntilNextTrip = periodsUntilNextTrip; + } + + public int getVeh_used() { + return veh_used; + } + + public void setVeh_used(int veh_used) { + this.veh_used = veh_used; + } + + } + + public class VehicleTrip{ + + int origMaz; + int destMaz; + int period; + int occupants; + boolean originIsHome; + boolean destinationIsHome; + boolean originIsRemoteParking; + boolean destinationIsRemoteParking; + int parkingChoiceAtDestination; + + Trip tripServed; + + public void writeDebug(Logger logger) { + + logger.info("*** VEHICLE TRIP DEBUG ***"); + logger.info("Orig MAZ: "+origMaz); + logger.info("Dest MAZ: "+destMaz); + logger.info("Period: "+period); + logger.info("Occupants: "+occupants); + logger.info("Orig is home: "+originIsHome); + logger.info("Dest is home: "+destinationIsHome); + logger.info("Orig is remote park: "+originIsRemoteParking); + logger.info("Dest is remote park: "+destinationIsRemoteParking); + logger.info("Parking choice at dest: "+parkingChoiceAtDestination); + + } + + public VehicleTrip() { + } + + public int getOrigMaz() { + return origMaz; + } + + public void setOrigMaz(int origMaz) { + this.origMaz = origMaz; + } + + public int getDestMaz() { + return destMaz; + } + + public void setDestMaz(int destMaz) { + this.destMaz = destMaz; + } + + public int getPeriod() { + return period; + } + + public void setPeriod(int period) { + this.period = period; + } + + public int getOccupants() { + return occupants; + } + + public void setOccupants(int occupants) { + this.occupants = occupants; + } + + public boolean isOriginIsHome() { + return originIsHome; + } + + public void setOriginIsHome(boolean originIsHome) { + this.originIsHome = originIsHome; + } + + public boolean isDestinationIsHome() { + return destinationIsHome; + } + + public void setDestinationIsHome(boolean destinationIsHome) { + this.destinationIsHome = destinationIsHome; + } + + public boolean isOriginIsRemoteParking() { + return originIsRemoteParking; + } + + public void setOriginIsRemoteParking(boolean originIsRemoteParking) { + this.originIsRemoteParking = originIsRemoteParking; + } + + public boolean isDestinationIsRemoteParking() { + return destinationIsRemoteParking; + } + + public void setDestinationIsRemoteParking(boolean destinationIsRemoteParking) { + this.destinationIsRemoteParking = destinationIsRemoteParking; + } + + public int getParkingChoiceAtDestination() { + return parkingChoiceAtDestination; + } + + public void setParkingChoiceAtDestination(int parkingChoiceAtDestination) { + this.parkingChoiceAtDestination = parkingChoiceAtDestination; + } + + public Trip getTripServed() { + return tripServed; + } + + public void setTripServed(Trip tripServed) { + this.tripServed = tripServed; + } + + } + + /** + * This method writes AV vehicle trips to the output file. + * + */ + public void writeVehicleTrips(float sampleRate){ + + logger.info("Writing AV trips to file " + vehicleTripOutputFile); + PrintWriter printWriter = null; + try + { + printWriter = new PrintWriter(new BufferedWriter(new FileWriter(vehicleTripOutputFile))); + } catch (IOException e) + { + logger.fatal("Could not open file " + vehicleTripOutputFile + " for writing\n"); + throw new RuntimeException(); + } + + printHeader(printWriter); + Set keySet = householdMap.keySet(); + for(Integer key: keySet) { + Household hh = householdMap.get(key); + printVehicleTrips(printWriter,hh, sampleRate); + printWriter.flush(); + } + + printWriter.close(); + + } + + + public void printHeader(PrintWriter writer) { + + writer.println("hh_id,veh_id,vehicleTrip_id,orig_mgra,dest_gra,period,occupants," + + "originIsHome,destinationIsHome,originIsRemoteParking,destinationIsRemoteParking," + + "parkingChoiceAtDestination,remoteParkingCostAtDest," + + "person_id,person_num,tour_id,stop_id,inbound,tour_purpose,orig_purpose,dest_purpose," + + "trip_orig_mgra,trip_dest_mgra,stop_period,periodsUntilNextTrip,trip_mode"); + + } + + /** + * Write output to the printwriter. + * + * @param writer + * @param hh + */ + public void printVehicleTrips(PrintWriter writer, Household hh, float sampleRate) { + + int hhid=hh.getId(); + ArrayList vehicles = hh.getAutonomousVehicles(); + if(vehicles==null) + return; + for(int i=0;i vehicleTrips = vehicle.getVehicleTrips(); + + if(vehicleTrips==null) + continue; + + if(vehicleTrips.size()==0) + continue; + + for(int j=0;j vehicleTrips; + + public Vehicle() { + + vehicleTrips = new ArrayList(); + } + + public void writeDebug(Logger logger) { + + logger.info("*** Vehicle debug **"); + logger.info("MAZ: "+maz); + logger.info("Is home: "+isHome); + logger.info("Period available: "+periodAvailable); + + if(vehicleTrips.size()>0) { + for(int i =0;i0) { + VehicleTrip lastTrip = vehicleTrips.get(vehicleTrips.size()-1); + vehicleTrip.setOrigMaz(lastTrip.getDestMaz()); + vehicleTrip.setOriginIsHome(lastTrip.isDestinationIsHome()); + }else { + vehicleTrip.setOriginIsHome(true); + } + + vehicleTrips.add(vehicleTrip); + + return vehicleTrip; + } + + public VehicleTrip createNewVehicleTrip() { + return new VehicleTrip(); + } + + public int getMaz() { + return maz; + } + public void setMaz(int maz) { + this.maz = maz; + } + public boolean isHome() { + return isHome; + } + public void setHome(boolean isHome) { + this.isHome = isHome; + } + public int getPeriodAvailable() { + return periodAvailable; + } + public void setPeriodAvailable(int periodAvailable) { + this.periodAvailable = periodAvailable; + } + + public ArrayList getVehicleTrips() { + return vehicleTrips; + } + + public void setVehicleTrips(ArrayList vehicleTrips) { + this.vehicleTrips = vehicleTrips; + } + + public int getWithPersonId() { + return withPersonId; + } + + public void setWithPersonId(int withPersonId) { + this.withPersonId = withPersonId; + } + + } + + public HouseholdAVAllocationManager(HashMap propertyMap, int iteration,MgraDataManager mazManager,TazDataManager tazManager){ + this.iteration = iteration; + this.propertyMap = propertyMap; + this.tazManager = tazManager; + this.mazManager = mazManager; + modelStructure = new SandagModelStructure(); + + } + + + public void setup() { + + random = new MersenneTwister(); + random.setSeed(randomSeed); + String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); + vehicleTripOutputFile = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_FILE_PROPERTY); + + householdMap = new HashMap(); + + personTypeMap = new HashMap(); + + for(int i = 0;i keySet = householdMap.keySet(); + + for(Integer key: keySet) { + + Household hh = householdMap.get(key); + if(hh.isDebug()) { + + logger.info("***********************************************"); + logger.info("AV allocation model trace (After reading) for household "+hh.getId()); + logger.info(""); + hh.writeDebug(logger, false); + logger.info("***********************************************"); + + } + } + + + } + + /* + * Drop households from the map that don't have AVs + */ + public void dropHouseholdsWithoutAVs() { + + logger.info("Dropping non-AV households"); + + Set keys = householdMap.keySet(); + ArrayList hhIdsToRemove = new ArrayList(); + + for(Integer key: keys) { + + Household hh = householdMap.get(key); + if(hh.getAVs()<=0) { + hhIdsToRemove.add(key); + } + } + if(hhIdsToRemove.size()>0) { + for(Integer hhId : hhIdsToRemove) { + householdMap.remove(hhId); + } + } + logger.info("Completed dropping non-AV households"); + + } + + /* + * Drop trips from the map that aren't auto trips with AVs + */ + public void dropNonAVTrips() { + + logger.info("Dropping non-AV trips from households"); + Set keys = householdMap.keySet(); + + for(Integer key: keys) { + + Household hh = householdMap.get(key); + ArrayList trips = hh.getTrips(); + if(trips.size()==0) + continue; + + Iterator itr = trips.iterator(); + while (itr.hasNext()){ + Trip trip = itr.next(); + + if(trip.getAv_avail()==0) + itr.remove(); + + else if(trip.trip_mode>MaxAutoMode) + itr.remove(); + } + } + logger.info("Completed dropping non-AV trips from households"); + + } + + + public void sortTrips() { + + Set keys = householdMap.keySet(); + for(Integer key: keys) { + + Household hh = householdMap.get(key); + ArrayList trips = hh.getTrips(); + if(trips.size()==0) + continue; + + sortByPerson=true; + Collections.sort(trips); + + //first calculate time before next AV trip made by same person + for(int i = 0 ; i< trips.size();++i) { + Trip trip = trips.get(i); + if(i<(trips.size()-1)) { + Trip nextTrip = trips.get(i+1); + if(trip.getPerson_id()==nextTrip.getPerson_id()) { + int periods = nextTrip.getStop_period()-trip.getStop_period(); + trip.setPeriodsUntilNextTrip(periods); + }else + trip.setPeriodsUntilNextTrip(99);//last trip of the day + }else + trip.setPeriodsUntilNextTrip(99); //last trip of the household + + } + sortByPerson=false; + + Collections.sort(trips); + + } + + } + + + public void readHouseholds() { + + + setDebugHhIdsFromHashmap(); + + String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); + String householdFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, HouseholdDataFileProperty); + householdFile = insertIterationNumber(householdFile,iteration); + + //get the household table and fill up the householdMap with households. + TableDataSet householdDataSet = readTableData(householdFile); + + for(int row=1;row<=householdDataSet.getRowCount();++row) { + + int seed = Math.abs(random.nextInt()); + + //read data + int hhId = (int) householdDataSet.getValueAt(row, "hh_id"); + int hhMgra = (int) householdDataSet.getValueAt(row,"home_mgra"); + int income = (int) householdDataSet.getValueAt(row,"income"); + int autos = (int) householdDataSet.getValueAt(row,"autos"); + int HVs = (int) householdDataSet.getValueAt(row,"HVs"); + int AVs = (int) householdDataSet.getValueAt(row,"AVs"); + + //create household object + Household hh = new Household(); + hh.setId(hhId); + hh.setHomeMaz(hhMgra); + hh.setIncome(income); + hh.setAutos(autos); + hh.setHVs(HVs); + hh.setAVs(AVs); + hh.setSeed(seed); + + if(householdTraceSet.contains(hhId)) + hh.setDebug(true); + else + hh.setDebug(false); + + //generate a set of vehicles and store in h + if(AVs>0) { + for(int i=0;i vehicles = hh.getAutonomousVehicles(); + vehicles.add(AV); + } + } + + //put hh in map + householdMap.put(hhId, hh); + + } + + + } + + public void readPersons() { + + + String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); + String personFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, PersonDataFileProperty); + personFile = insertIterationNumber(personFile,iteration); + + //get the household table and fill up the householdMap with households. + TableDataSet personDataSet = readTableData(personFile); + + for(int row=1;row<=personDataSet.getRowCount();++row) { + + + int hh_id = (int) personDataSet.getValueAt(row,"hh_id"); + int person_id = (int) personDataSet.getValueAt(row,"person_id"); + int person_num = (int) personDataSet.getValueAt(row,"person_num"); + int age = (int) personDataSet.getValueAt(row,"age"); + String gender = personDataSet.getStringValueAt(row,"gender"); + String type = personDataSet.getStringValueAt(row,"type"); + float value_of_time = personDataSet.getValueAt(row,"value_of_time"); + float reimb_pct = personDataSet.getValueAt(row,"reimb_pct"); + int parkingChoice = (int) personDataSet.getValueAt(row, "fp_choice"); + float timeFactorWork = personDataSet.getValueAt(row,"timeFactorWork"); + float timeFactorNonWork = personDataSet.getValueAt(row,"timeFactorNonWork"); + + Person person = new Person(); + person.setHh_id(hh_id); + person.setPerson_id(person_id); + person.setPerson_num(person_num); + person.setAge(age); + person.setGender(gender.compareToIgnoreCase("m")==0 ? 1 : 2); + person.setType(personTypeMap.get(type)); + person.setValue_of_time(value_of_time); + person.setReimb_pct(reimb_pct); + person.setFp_choice(parkingChoice); + person.setTimeFactorWork(timeFactorWork); + person.setTimeFactorNonWork(timeFactorNonWork); + + if(householdMap.containsKey(hh_id)) { + Household hh = householdMap.get(hh_id); + hh.personMap.put(person_num,person); + }else { + logger.fatal("Error: No household ID "+hh_id+" in householdMap. Cannot add person object"); + throw new RuntimeException(); + } + + } + + } + + public void readTrips(String filename, boolean isJoint) { + + + //get the household table and fill up the householdMap with households. + TableDataSet tripDataSet = readTableData(filename); + + for(int row=1;row<=tripDataSet.getRowCount();++row) { + + + int hh_id = (int) tripDataSet.getValueAt(row, "hh_id"); + int person_id=-9; + int person_num=-9; + int num_participants = 1; + int driver_pnum=-9; + if(!isJoint) { + person_id = (int) tripDataSet.getValueAt(row,"person_id"); + person_num = (int) tripDataSet.getValueAt(row,"person_num"); + driver_pnum = (int) tripDataSet.getValueAt(row,"driver_pnum"); + }else { + num_participants = (int) tripDataSet.getValueAt(row,"num_participants"); + person_id = hh_id*100+num_participants; + } + int tour_id = (int) tripDataSet.getValueAt(row,"tour_id"); + int stop_id = (int) tripDataSet.getValueAt(row,"stop_id"); + int inbound = (int) tripDataSet.getValueAt(row,"inbound"); + String tour_purpose = tripDataSet.getStringValueAt(row,"tour_purpose"); + String orig_purpose = tripDataSet.getStringValueAt(row,"orig_purpose"); + String dest_purpose = tripDataSet.getStringValueAt(row,"dest_purpose"); + int orig_maz= (int) tripDataSet.getValueAt(row,"orig_mgra"); + int dest_maz= (int) tripDataSet.getValueAt(row,"dest_mgra"); + int parking_maz = (int) tripDataSet.getValueAt(row,"parking_mgra"); + int stop_period = (int) tripDataSet.getValueAt(row,"stop_period"); + int trip_mode = (int) tripDataSet.getValueAt(row,"trip_mode"); + int av_avail = (int) tripDataSet.getValueAt(row,"av_avail"); + int tour_mode = (int) tripDataSet.getValueAt(row,"tour_mode"); + float valueOfTime = tripDataSet.getValueAt(row,"valueOfTime"); + int transponder_avail = (int) tripDataSet.getValueAt(row,"transponder_avail"); + + Trip trip = new Trip(); + trip.setHh_id(hh_id); + trip.setPerson_id(person_id); + trip.setPerson_num(person_num); + trip.setTour_id(tour_id); + trip.setStop_id(stop_id); + trip.setInbound(inbound); + trip.setTour_purpose(tour_purpose); + trip.setOrig_purpose(orig_purpose); + trip.setDest_purpose(dest_purpose); + trip.setOrig_maz(orig_maz); + trip.setDest_maz(dest_maz); + trip.setParking_maz(parking_maz); + trip.setStop_period(stop_period); + trip.setTrip_mode(trip_mode); + trip.setAv_avail(av_avail); + trip.setTour_mode(tour_mode); + trip.setDriver_pnum(driver_pnum); + trip.setValueOfTime(valueOfTime); + trip.setTransponder_avail(transponder_avail); + trip.setNum_participants(num_participants); + + if(householdMap.containsKey(hh_id)){ + Household hh = householdMap.get(hh_id); + + //following code only handles individual trips right now + //TODO: Revise trip file to include participants, and modify code to + //add all participants. + if(person_num!=-99) { + HashMap personMap = hh.personMap; + Person person = personMap.get(person_num); + ArrayList personsOnTrip = trip.getPersons(); + if(personsOnTrip==null) + personsOnTrip = new ArrayList(); + personsOnTrip.add(person); + } + hh.trips.add(trip); + }else { + logger.fatal("Error: No household ID "+hh_id+" in householdMap. Cannot add trip object"); + throw new RuntimeException(); + + } + + } + + + + } + + /** + * Read data into inputDataTable tabledataset. + * + */ + private TableDataSet readTableData(String inputFile){ + + TableDataSet tableDataSet = null; + + logger.info("Begin reading the data in file " + inputFile); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + tableDataSet = csvFile.readFile(new File(inputFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End reading the data in file " + inputFile); + + return tableDataSet; + } + /** + * A simple helper function to insert the iteration number into the file name. + * + * @param filename The input file name (ex: inputFile.csv) + * @param iteration The iteration number (ex: 3) + * @return The new string (ex: inputFile_3.csv) + */ + private String insertIterationNumber(String filename, int iteration){ + + String newFileName = filename.replace(".csv", "_"+new Integer(iteration).toString()+".csv"); + return newFileName; + } + + public HashMap getHouseholdMap() { + return householdMap; + } + + public void setDebugHhIdsFromHashmap() + { + + householdTraceSet = new HashSet(); + + // get the household ids for which debug info is required + String householdTraceStringList = propertyMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); + + if (householdTraceStringList != null) + { + StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); + while (householdTokenizer.hasMoreTokens()) + { + String listValue = householdTokenizer.nextToken(); + int idValue = Integer.parseInt(listValue.trim()); + householdTraceSet.add(idValue); + } + } + + } + + /** + * Get the output trip table file names from the properties file, and write + * trip tables for all modes for the given time period. + * + * @param period + * Time period, which will be used to find the period time string + * to append to each trip table matrix file + */ + public void writeTripTable(MatrixDataServerRmi ms) + { + + String directory = Util.getStringValueFromPropertyMap(propertyMap, "scenario.path"); + String matrixTypeName = Util.getStringValueFromPropertyMap(propertyMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + String fileName = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_MATRIX_PROPERTY) + ".omx"; + try{ + //Delete the file if it exists + File f = new File(fileName); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName); + f.delete(); + } + + if (ms != null) + ms.writeMatrixFile(fileName, emptyVehicleTripMatrix, mt); + else + writeMatrixFile(fileName, emptyVehicleTripMatrix); + } catch (Exception e){ + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName, e); + throw new RuntimeException(); + } + + } + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + private void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java new file mode 100644 index 0000000..8101a57 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java @@ -0,0 +1,671 @@ +package org.sandag.abm.maas; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ListIterator; +import java.util.Random; +import java.util.Set; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.MicromobilityChoiceDMU; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Household; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Person; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Trip; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Vehicle; +import org.sandag.abm.maas.HouseholdAVAllocationManager.VehicleTrip; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.ResourceUtil; + +public class HouseholdAVAllocationModel { + + private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModel.class); + private HashMap propertyMap = null; + HouseholdAVAllocationManager avManager; + private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; + private static final String MINUTES_PER_SIMULATION_PERIOD_PROPERTY = "Maas.RoutingModel.minutesPerSimulationPeriod"; + + private static final String AV_CONTROL_FILE_TARGET = "Maas.AVAllocation.uec.file"; + private static final String AV_DATA_SHEET_TARGET = "Maas.AVAllocation.data.page"; + private static final String AV_VEHICLECHOICE_SHEET_TARGET = "Maas.AVAllocation.vehiclechoice.model.page"; + private static final String AV_PARKINGCHOICE_SHEET_TARGET = "Maas.AVAllocation.parkingchoice.model.page"; + private static final String AV_TRIPUTILITY_SHEET_TARGET = "Maas.AVAllocation.triputility.model.page"; + + private static final int parkingChoiceStay =1; + private static final int parkingChoiceRemote=2; + private static final int parkingChoiceHome=3; + + private static final int vehicleChoiceOther=4; + + private MersenneTwister random; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + + private ChoiceModelApplication parkingChoiceModel; + private ChoiceModelApplication vehicleChoiceModel; + private ChoiceModelApplication tripUtilityModel; + + private HouseholdAVAllocationModelParkingChoiceDMU parkingChoiceDMU; + private HouseholdAVAllocationModelVehicleChoiceDMU vehicleChoiceDMU; + private HouseholdAVAllocationModelTripUtilityDMU tripUtilityDMU; + + int vehicleChoiceOffset = 23942345; + int parkingChoiceOffset =984388432; + int[] closestRemoteLotToMaz; + + + /** + * Constructor. + * + * @param propertyMap + * @param iteration + */ + public HouseholdAVAllocationModel(HashMap propertyMap, MgraDataManager mgraManager, + TazDataManager tazManager, int[] closestRemoteLotToMaz){ + this.propertyMap = propertyMap; + this.mgraManager = mgraManager; + this.tazManager = tazManager; + this.closestRemoteLotToMaz=closestRemoteLotToMaz; + } + + /** + * Initialize all the data members. + * + */ + public void initialize(){ + + + //seed the random number generator so that results can be replicated if desired. + int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); + + random = new MersenneTwister(seed + 4292); + + //create the model UECs + // locate the micromobility choice UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String avUecFile = uecFileDirectory + propertyMap.get(AV_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_DATA_SHEET_TARGET); + int vehicleModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_VEHICLECHOICE_SHEET_TARGET); + int parkingModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_PARKINGCHOICE_SHEET_TARGET); + int tripModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_TRIPUTILITY_SHEET_TARGET); + + // create the DMU objects. + vehicleChoiceDMU = new HouseholdAVAllocationModelVehicleChoiceDMU(); + parkingChoiceDMU = new HouseholdAVAllocationModelParkingChoiceDMU(); + tripUtilityDMU = new HouseholdAVAllocationModelTripUtilityDMU(); + + // create the choice model objects + vehicleChoiceModel= new ChoiceModelApplication(avUecFile, vehicleModelSheet, dataSheet, propertyMap, + (VariableTable) vehicleChoiceDMU); + parkingChoiceModel= new ChoiceModelApplication(avUecFile, parkingModelSheet, dataSheet, propertyMap, + (VariableTable) parkingChoiceDMU); + tripUtilityModel= new ChoiceModelApplication(avUecFile, tripModelSheet, dataSheet, propertyMap, + (VariableTable) tripUtilityDMU); + + + } + + /** + * Run the AV allocation model for all households in HashMap. + * + * @param hhMap A hashmap of households + * @return the completed hashmap + */ + public HashMap runModel(HashMap hhMap){ + + //iterate through map + Set keySet = hhMap.keySet(); + for(Integer key : keySet) { + + Household hh = hhMap.get(key); + ArrayList trips = hh.getTrips(); + if(trips==null) + continue; + if(trips.size()==0) + continue; + + ArrayList hhVehicles = hh.getAutonomousVehicles(); + + //iterate through trips, choose vehicle + for(int i =0;i vehicleTrips = vehicle.getVehicleTrips(); + VehicleTrip newVehicleTrip = vehicle.createNewVehicleTrip(); + + if(trip.getOrig_purpose().compareToIgnoreCase("Home")==0) + newVehicleTrip.setOriginIsHome(true); + else + newVehicleTrip.setOriginIsHome(false); + + if(trip.getDest_purpose().compareToIgnoreCase("Home")==0) + newVehicleTrip.setDestinationIsHome(true); + else + newVehicleTrip.setDestinationIsHome(false); + + newVehicleTrip.setOrigMaz(trip.getOrig_maz()); + newVehicleTrip.setDestMaz(trip.getDest_maz()); + newVehicleTrip.setOccupants(trip.getNum_participants()); + newVehicleTrip.setPeriod(trip.getStop_period()); + newVehicleTrip.setTripServed(trip); + vehicleTrips.add(newVehicleTrip); + + //update the time for which the vehicle will be available if it + //is an AV, and update the vehicle location + int period = trip.getStop_period(); + if(period>1 && period<40) { + float timeInMinutes = getTravelTime(hh.getId(),trip.getOrig_maz(),trip.getDest_maz(),period); + int additionalPeriods = (int) (timeInMinutes/30); + vehicle.setPeriodAvailable(period + additionalPeriods); + }else { + vehicle.setPeriodAvailable(period); + } + vehicle.setMaz(trip.getDest_maz()); + if(trip.getDest_purpose().compareToIgnoreCase("Home")==0) + vehicle.setHome(true); + else + vehicle.setHome(false); + + vehicle.setWithPersonId(trip.getPerson_id()); + + } + + if(hh.isDebug()) { + + logger.info("***********************************************"); + logger.info("AV allocation model trace (After vehicle choice) for household "+hh.getId()); + logger.info(""); + hh.writeDebug(logger, true); + logger.info("***********************************************"); + + } + + //iterate through vehicles, choose parking location for each trip in each vehicle. + for(int i =0;i vehicleTrips = veh.getVehicleTrips(); + if(vehicleTrips.size()==0) + continue; + + for(int j =0; j persons = trip.getPersons(); + Person person=null; + if(persons!=null) + person = persons.get(0); + + setParkingChoiceDMUAttributes(hh,person,trip, nextTrip); + parkingChoiceModel.computeUtilities(parkingChoiceDMU, parkingChoiceDMU.getDmuIndexValues()); + int parkingChoice = getParkingChoice(hh,trip); + vehicleTrip.setParkingChoiceAtDestination(parkingChoice); + } + } + + if(hh.isDebug()) { + + logger.info("***********************************************"); + logger.info("AV allocation model trace (After parking choice) for household "+hh.getId()); + logger.info(""); + hh.writeDebug(logger, true); + logger.info("***********************************************"); + + } + + //iterate through vehicles, generate empty vehicle trips. + for(int i =0;i vehicleTrips = vehicle.getVehicleTrips(); + if(vehicleTrips.size()==0) + return; + + int homeMaz = hh.getHomeMaz(); + + ArrayList newVehicleTrips = new ArrayList(); + for(int i =0; i < vehicleTrips.size();++i) { + VehicleTrip vehicleTrip = vehicleTrips.get(i); + Trip tripServed = vehicleTrip.getTripServed(); + int personId = tripServed.getPerson_id(); + + //first vehicle trip + if(i==0) { + + //vehicle is home, but first trip is NOT home, create a trip to it + if(tripServed.orig_purpose.compareToIgnoreCase("Home")!=0) { + VehicleTrip newTrip = vehicle.createNewVehicleTrip(); + newTrip.setOriginIsHome(true); + newTrip.setOrigMaz(homeMaz); + newTrip.setDestMaz(tripServed.getOrig_maz()); + newTrip.setPeriod(tripServed.getStop_period()); + newTrip.setOccupants(0); + newTrip.setOriginIsRemoteParking(false); + newTrip.setDestinationIsRemoteParking(false); + newTrip.setDestinationIsHome(false); + newTrip.setOccupants(0); + newVehicleTrips.add(newTrip); + } + } + newVehicleTrips.add(vehicleTrip); + + //generate empty trip to remote lot if parking is remote lot + int parkingChoice = vehicleTrip.getParkingChoiceAtDestination(); + if(parkingChoice==parkingChoiceRemote) { + VehicleTrip newTrip = vehicle.createNewVehicleTrip(); + newTrip.setOriginIsHome(false); + newTrip.setOrigMaz(vehicleTrip.getDestMaz()); + newTrip.setDestMaz(closestRemoteLotToMaz[vehicleTrip.getDestMaz()]); + newTrip.setPeriod(tripServed.getStop_period()); + newTrip.setDestinationIsHome(false); + newTrip.setOccupants(0); + newTrip.setOriginIsRemoteParking(false); + newTrip.setDestinationIsRemoteParking(true); + newTrip.setDestinationIsHome(false); + newVehicleTrips.add(newTrip); + //or a trip to home if parking choice is home + }else if(parkingChoice==parkingChoiceHome) { + VehicleTrip newTrip = vehicle.createNewVehicleTrip(); + newTrip.setOriginIsHome(false); + newTrip.setOrigMaz(vehicleTrip.getDestMaz()); + newTrip.setDestMaz(homeMaz); + newTrip.setPeriod(tripServed.getStop_period()); + newTrip.setDestinationIsHome(true); + newTrip.setOccupants(0); + newTrip.setOriginIsRemoteParking(false); + newTrip.setDestinationIsRemoteParking(false); + newTrip.setDestinationIsHome(true); + newVehicleTrips.add(newTrip); + } + //next trip + if(i<(vehicleTrips.size()-1)) { + VehicleTrip nextTrip = vehicleTrips.get(i+1); + Trip nextTripServed = nextTrip.getTripServed(); + VehicleTrip lastTrip = newVehicleTrips.get(newVehicleTrips.size()-1); + //if the trip is not already in the same MAZ generate an empty trip + if(lastTrip.getDestMaz()!=nextTrip.getOrigMaz()) { + VehicleTrip newTrip = vehicle.createNewVehicleTrip(); + newTrip.setOriginIsHome(lastTrip.isDestinationIsHome()); + newTrip.setOrigMaz(lastTrip.getDestMaz()); + newTrip.setDestMaz(nextTripServed.getOrig_maz()); + newTrip.setPeriod(nextTripServed.getStop_period()); + newTrip.setDestinationIsHome(nextTripServed.getOrig_purpose().compareToIgnoreCase("home")==0); + newTrip.setOccupants(0); + newTrip.setOriginIsRemoteParking(lastTrip.isDestinationIsRemoteParking()); + newTrip.setDestinationIsRemoteParking(false); + newTrip.setDestinationIsHome(nextTripServed.orig_purpose.compareToIgnoreCase("Home")==0); + newVehicleTrips.add(newTrip); + } + } + } + vehicle.setVehicleTrips(newVehicleTrips); + } + + + /** + * Set attributes for the vehicle allocation model. + * @param hh + * @param thisTrip + */ + public void setVehicleChoiceDMUAttributes(Household hh,Trip thisTrip) { + int[] vehicleIsAvailable= {0,0,0}; + float[] travelUtilityToPerson= {0,0,0}; + int[] vehicleIsWithPerson = {0,0,0}; + + int[] avail = {1,1,1}; + + ArrayList vehicles = hh.getAutonomousVehicles(); + for(int i = 0;iveh.getPeriodAvailable()) + vehicleIsAvailable[i]=1; + else { + vehicleIsAvailable[i]=0; + continue; + } + + //if the vehicle is with the person + if(thisTrip.getPerson_id()==veh.getWithPersonId()) + vehicleIsWithPerson[i]=1; + + int origMgra=veh.getMaz(); + int origTaz = mgraManager.getTaz(origMgra); + int destMgra=thisTrip.getDest_maz(); + int destTaz=mgraManager.getTaz(destMgra); + vehicleChoiceDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); + + //the utility to the person is 0, so don't calculate anything + if(veh.isHome() && thisTrip.orig_purpose.compareToIgnoreCase("Home")==0) + continue; + + float utility= getTravelUtility(hh, origMgra,destMgra,period); + + travelUtilityToPerson[i]=utility; + + } + + vehicleChoiceDMU.setVehicle1IsAvailable(vehicleIsAvailable[0]); + vehicleChoiceDMU.setVehicle2IsAvailable(vehicleIsAvailable[1]); + vehicleChoiceDMU.setVehicle3IsAvailable(vehicleIsAvailable[2]); + + vehicleChoiceDMU.setPersonWithVehicle1(vehicleIsWithPerson[0]); + vehicleChoiceDMU.setPersonWithVehicle2(vehicleIsWithPerson[1]); + vehicleChoiceDMU.setPersonWithVehicle3(vehicleIsWithPerson[2]); + + vehicleChoiceDMU.setTravelUtilityToPersonVeh1(travelUtilityToPerson[0]); + vehicleChoiceDMU.setTravelUtilityToPersonVeh2(travelUtilityToPerson[1]); + vehicleChoiceDMU.setTravelUtilityToPersonVeh3(travelUtilityToPerson[2]); + + vehicleChoiceDMU.setMinutesUntilNextTrip(thisTrip.getPeriodsUntilNextTrip()*30); + + + } + + /** + * After the trip is complete, a parking choice is made for the vehicle used for + * the trip. This should only be called for trips for which an AV was used. + * + * @param hh + * @param person + * @param thisTrip + * @param nextTrip + */ + public void setParkingChoiceDMUAttributes(Household hh, Person person, Trip thisTrip, Trip nextTrip) { + + int[] parkArea = mgraManager.getMgraParkAreas(); + float[] monthlyCosts =mgraManager.getMParkCost(); + float[] dailyCosts = mgraManager.getDParkCost(); + float[] hourlyCosts = mgraManager.getHParkCost(); + + int id = hh.getId(); + int hhMaz = hh.getHomeMaz(); + int destMaz = thisTrip.getDest_maz(); + int period = thisTrip.getStop_period(); + + float durationBeforeNextTrip=8*60; //assume 8 hrs before the next trip if there isn't one + if(nextTrip!=null) + durationBeforeNextTrip=(nextTrip.getStop_period()- thisTrip.getStop_period())*30; + + int personType=0; + int atWork=0; + float reimburseProportion=0; + int freeParkingEligibility=0; + if(person!=null) { + personType = person.getType(); + atWork = thisTrip.getDest_purpose().compareToIgnoreCase("Work")==0? 1 : 0; + reimburseProportion = person.getReimb_pct(); + freeParkingEligibility = person.getFp_choice()==1 ? 1 : 0; + } + int parkingArea= parkArea[destMaz]; + float dailyParkingCost= dailyCosts[destMaz]; + float hourlyParkingCost=hourlyCosts[destMaz]; + float monthlyParkingCost=monthlyCosts[destMaz]; + + float utilityToClosestRemoteLot = getTravelUtility(hh, destMaz,destMaz,period); + float utilityToHome = getTravelUtility(hh, destMaz,hhMaz,period); + + float utilityFromHomeToNextTrip =0; + if(nextTrip!=null) { + int nextMaz = nextTrip.getOrig_maz(); + utilityFromHomeToNextTrip = getTravelUtility(hh, destMaz,nextMaz,period); + } + + parkingChoiceDMU.setDurationBeforeNextTrip(durationBeforeNextTrip); + parkingChoiceDMU.setPersonType(personType); + parkingChoiceDMU.setAtWork(atWork); + parkingChoiceDMU.setFreeParkingEligibility(freeParkingEligibility); + parkingChoiceDMU.setReimburseProportion(reimburseProportion); + parkingChoiceDMU.setDailyParkingCost(dailyParkingCost); + parkingChoiceDMU.setHourlyParkingCost(hourlyParkingCost); + parkingChoiceDMU.setMonthlyParkingCost(monthlyParkingCost); + parkingChoiceDMU.setParkingArea(parkingArea); + parkingChoiceDMU.setUtilityToClosestRemoteLot(utilityToClosestRemoteLot); + parkingChoiceDMU.setUtilityToHome(utilityToHome); + parkingChoiceDMU.setUtilityFromHomeToNextTrip(utilityFromHomeToNextTrip); + + } + + /** + * Get the travel time for the origin and destination. + * + * @param id An ID for the trip + * @param originMaz origin MAZ + * @param destMaz destination MAZ + * @param period departure period + * @return Time from the tripUtilityUEC (2nd alternative) + */ + public float getTravelTime(int id, int originMaz,int destMaz,int period) { + + int[] avail = {1,1,1}; + + //calculate when the vehicle would be available for the next trip given the current + //period and the travel time to the next trip. + int origTaz = mgraManager.getTaz(originMaz); + int destTaz=mgraManager.getTaz(destMaz); + tripUtilityDMU.setDmuIndexValues(id, origTaz, origTaz, destTaz); + vehicleChoiceDMU.setDmuIndexValues(id, origTaz, origTaz, destTaz); + tripUtilityDMU.setTimeTrip(period); + + UtilityExpressionCalculator tripUtilityUEC = tripUtilityModel.getUEC(); + double[] util = tripUtilityUEC.solve(tripUtilityDMU.getDmuIndexValues(), tripUtilityDMU, avail); + + return (float) util[1]; + } + + /** + * Get the travel utility for the origin and destination. + * + * @param id An ID for the trip + * @param originMaz origin MAZ + * @param destMaz destination MAZ + * @param period departure period + * @return Utility from the tripUtilityUEC (1st alternative) + */ + public float getTravelUtility(Household hh, int originMaz,int destMaz,int period) { + + int[] avail = {1,1,1}; + + //calculate when the vehicle would be available for the next trip given the current + //period and the travel time to the next trip. + int origTaz = mgraManager.getTaz(originMaz); + int destTaz=mgraManager.getTaz(destMaz); + tripUtilityDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); + vehicleChoiceDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); + tripUtilityDMU.setTimeTrip(period); + + UtilityExpressionCalculator tripUtilityUEC = tripUtilityModel.getUEC(); + double[] util = tripUtilityUEC.solve(tripUtilityDMU.getDmuIndexValues(), tripUtilityDMU, avail); + + // write choice model alternative info to log file + if (hh.isDebug()) + { + logger.info("Calculating travel utility calculation for household " + hh.getId()+ " trip in period " + period +" from MAZ "+originMaz+" to MAZ "+destMaz); + tripUtilityUEC.logAnswersArray( logger, "Trip utility"); + } + + + return (float) util[0]; + } + + /** + * Select the vehicle choice from the UEC. This is helper code for applyModel(), where utilities have already been calculated. + + * @return The vehicle alternative 1,2,3,4.1-3 are AV veh numbers, 4 is other + */ + private int getVehicleChoice(Household hh, Trip trip) { + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + long seed = hh.getSeed() + (vehicleChoiceOffset + trip.getStop_id()*23 +trip.getPerson_id()*34 + trip.getStop_period()*23); + random.setSeed(seed); + + if (vehicleChoiceModel.getAvailabilityCount() > 0) + { + double randomNumber = random.nextDouble(); + chosenAlt = vehicleChoiceModel.getChoiceResult(randomNumber); + + // write choice model alternative info to log file + if (hh.isDebug()) + { + String decisionMaker = String.format("Household " + hh.getId()+ " trip in period " + trip.stop_period +" from MAZ "+trip.getOrig_maz()+" to MAZ "+trip.getDest_maz()); + vehicleChoiceModel.logAlternativesInfo("Vehicle Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Vehicle Choice", decisionMaker, chosenAlt)); + vehicleChoiceModel.logUECResults(logger, decisionMaker); + } + + + return chosenAlt; + } else + { + String decisionMaker = String.format("Household " + hh.getId()+" "+trip.getStop_id()); + String errorMessage = String + .format("Exception caught for %s, no available vehicle choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + vehicleChoiceModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + } + + + /** + * Select the parking choice from the UEC. This is helper code for applyModel(), where utilities have already been calculated. + + * @return The parking alternative + */ + private int getParkingChoice(Household hh, Trip trip) { + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + long seed = hh.getSeed() + (parkingChoiceOffset + trip.getStop_id()*123 +trip.getPerson_id()*23 + trip.getStop_period()*18); + random.setSeed(seed); + + if (parkingChoiceModel.getAvailabilityCount() > 0) + { + double randomNumber = random.nextDouble(); + chosenAlt = parkingChoiceModel.getChoiceResult(randomNumber); + + // write choice model alternative info to log file + if (hh.isDebug()) + { + String decisionMaker = String.format("Household " + hh.getId()+ " trip in period " + trip.stop_period +" from MAZ "+trip.getOrig_maz()+" to MAZ "+trip.getDest_maz()); + parkingChoiceModel.logAlternativesInfo("Parking Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Parking Choice", decisionMaker, chosenAlt)); + parkingChoiceModel.logUECResults(logger, decisionMaker); + } + + + + return chosenAlt; + } else + { + String decisionMaker = String.format("Household " + hh.getId()+" "+trip.getStop_id()); + String errorMessage = String + .format("Exception caught for %s, no available vehicle choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.error(errorMessage); + + parkingChoiceModel.logUECResults(logger, decisionMaker); + throw new RuntimeException(); + } + + } + + + + + /** + * Main run method + * @param args + */ + public static void main(String[] args) { + + + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java new file mode 100644 index 0000000..f9d0f26 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java @@ -0,0 +1,239 @@ +package org.sandag.abm.maas; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + */ +public class HouseholdAVAllocationModelParkingChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelParkingChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + float durationBeforeNextTrip; + int personType; + int atWork; + int freeParkingEligibility; + float reimburseProportion; + float dailyParkingCost; + float hourlyParkingCost; + float monthlyParkingCost; + int parkingArea; + float utilityToClosestRemoteLot; + float utilityToHome; + float utilityFromHomeToNextTrip; + + + public HouseholdAVAllocationModelParkingChoiceDMU() + { + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + } + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getDurationBeforeNextTrip",1); + methodIndexMap.put("getPersonType",2); + methodIndexMap.put("getAtWork",3); + methodIndexMap.put("getFreeParkingEligibility",4); + methodIndexMap.put("getReimburseProportion",5); + methodIndexMap.put("getDailyParkingCost",6); + methodIndexMap.put("getHourlyParkingCost",7); + methodIndexMap.put("getMonthlyParkingCost",8); + methodIndexMap.put("getParkingArea",9); + methodIndexMap.put("getUtilityToClosestRemoteLot",10); + methodIndexMap.put("getUtilityToHome",11); + methodIndexMap.put("getUtilityFromHomeToNextTrip",12); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getDurationBeforeNextTrip(); + case 2: + return getPersonType(); + case 3: + return getAtWork(); + case 4: + return getFreeParkingEligibility(); + case 5: + return getReimburseProportion(); + case 6: + return getDailyParkingCost(); + case 7: + return getHourlyParkingCost(); + case 8: + return getMonthlyParkingCost(); + case 9: + return getParkingArea(); + case 10: + return getUtilityToClosestRemoteLot(); + case 11: + return getUtilityToHome(); + case 12: + return getUtilityFromHomeToNextTrip(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public float getDurationBeforeNextTrip() { + return durationBeforeNextTrip; + } + + public void setDurationBeforeNextTrip(float durationBeforeNextTrip) { + this.durationBeforeNextTrip = durationBeforeNextTrip; + } + + public int getPersonType() { + return personType; + } + + public void setPersonType(int personType) { + this.personType = personType; + } + + public int getAtWork() { + return atWork; + } + + public void setAtWork(int atWork) { + this.atWork = atWork; + } + + public int getFreeParkingEligibility() { + return freeParkingEligibility; + } + + public void setFreeParkingEligibility(int freeParkingEligibility) { + this.freeParkingEligibility = freeParkingEligibility; + } + + public float getReimburseProportion() { + return reimburseProportion; + } + + public void setReimburseProportion(float reimburseProportion) { + this.reimburseProportion = reimburseProportion; + } + + public float getDailyParkingCost() { + return dailyParkingCost; + } + + public void setDailyParkingCost(float dailyParkingCost) { + this.dailyParkingCost = dailyParkingCost; + } + + public float getHourlyParkingCost() { + return hourlyParkingCost; + } + + public void setHourlyParkingCost(float hourlyParkingCost) { + this.hourlyParkingCost = hourlyParkingCost; + } + + public float getMonthlyParkingCost() { + return monthlyParkingCost; + } + + public void setMonthlyParkingCost(float monthlyParkingCost) { + this.monthlyParkingCost = monthlyParkingCost; + } + + public int getParkingArea() { + return parkingArea; + } + + public void setParkingArea(int parkingArea) { + this.parkingArea = parkingArea; + } + + public float getUtilityToClosestRemoteLot() { + return utilityToClosestRemoteLot; + } + + public void setUtilityToClosestRemoteLot(float utilityToClosestRemoteLot) { + this.utilityToClosestRemoteLot = utilityToClosestRemoteLot; + } + + public float getUtilityToHome() { + return utilityToHome; + } + + public void setUtilityToHome(float utilityToHome) { + this.utilityToHome = utilityToHome; + } + + public float getUtilityFromHomeToNextTrip() { + return utilityFromHomeToNextTrip; + } + + public void setUtilityFromHomeToNextTrip(float utilityFromHomeToNextTrip) { + this.utilityFromHomeToNextTrip = utilityFromHomeToNextTrip; + } + + + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java new file mode 100644 index 0000000..cc4c03f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java @@ -0,0 +1,260 @@ +package org.sandag.abm.maas; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Set; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.MicromobilityChoiceDMU; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Household; +import org.sandag.abm.maas.HouseholdAVAllocationManager.Trip; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.matrix.MatrixType; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.ResourceUtil; + +public class HouseholdAVAllocationModelRunner { + + private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModelRunner.class); + private HashMap propertyMap = null; + HouseholdAVAllocationManager avManager; + private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; + + private int iteration; + private float sampleRate; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private HouseholdAVAllocationModel AVAllocationModel; //one for now, can multi-thread later + private int[] closestMazWithRemoteLot; + MatrixDataServerRmi ms; + private UtilityExpressionCalculator distanceUEC; + protected VariableTable dmu = null; + protected float[][] tazDistanceSkims; //travel distance + + /** + * Constructor. + * + * @param propertyMap + * @param iteration + */ + public HouseholdAVAllocationModelRunner(HashMap propertyMap, int iteration, float sampleRate){ + this.propertyMap = propertyMap; + this.iteration = iteration; + this.sampleRate = sampleRate; + } + + /** + * Initialize all the data members. + * + */ + public void initialize(){ + + startMatrixServer(propertyMap); + + //managers for MAZ and TAZ data + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + + //create a household AV manager, read trips + avManager = new HouseholdAVAllocationManager(propertyMap, iteration, mgraManager, tazManager); + avManager.setup(); + avManager.readInputFiles(); + + calculateDistanceSkims(); + calculateClosestRemoteLotMazs(); + + } + + /** + * Creates a midday distance UEC, solves it for all zones, stores results in tazDistanceSkims[][]. + */ + public void calculateDistanceSkims() { + + logger.info("Calculating distance skims"); + // Create the distance UEC + String uecPath = Util.getStringValueFromPropertyMap(propertyMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = uecPath + + Util.getStringValueFromPropertyMap(propertyMap, "taz.distance.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, "taz.distance.data.page"); + int distancePage = Util.getIntegerValueFromPropertyMap(propertyMap, "taz.od.distance.md.page"); + distanceUEC = new UtilityExpressionCalculator(new File(uecFileName), distancePage, dataPage, + propertyMap, dmu); + IndexValues iv = new IndexValues(); + + int maxTaz = tazManager.getMaxTaz(); + tazDistanceSkims = new float[maxTaz+1][maxTaz+1]; + + for (int oTaz = 1; oTaz <= maxTaz; oTaz++){ + + iv.setOriginZone(oTaz); + + double[] autoDist = distanceUEC.solve(iv, dmu, null); + for (int d = 0; d < maxTaz; d++){ + tazDistanceSkims[oTaz][d + 1] = (float) autoDist[d]; + } + } + logger.info("Completed calculating distance skims"); + + } + + public void runModel(){ + + //iterate through map + HashMap hhMap = avManager.getHouseholdMap(); + AVAllocationModel = new HouseholdAVAllocationModel(propertyMap, mgraManager,tazManager,closestMazWithRemoteLot); + AVAllocationModel.initialize(); + + AVAllocationModel.runModel(hhMap); + + avManager.writeVehicleTrips(sampleRate); + + avManager.writeTripTable(ms); + + } + + + + /** + * Start a matrix server + * + * @param properties + */ + private void startMatrixServer(HashMap properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + logger.error("could not connect to matrix server", e); + throw new RuntimeException(e); + + } + + } + + + /** + * Iterate through the zones for each MAZ and find the closest MAZ with a remote parking lot. + * + */ + public void calculateClosestRemoteLotMazs() { + + int maxMaz = mgraManager.getMaxMgra(); + + //initialize the array + closestMazWithRemoteLot = new int[maxMaz+1]; + + //iterate through origin MAZs + for(int originMaz=1;originMaz<=maxMaz;++originMaz) { + + float minDist = 99999; //initialize to a really high value + + int originTaz = mgraManager.getTaz(originMaz); + if(originTaz<=0) + continue; + + //iterate through destination MAZs + for(int destinationMaz=1;destinationMaz<=maxMaz;++destinationMaz) { + + //no refueling stations in the destination, keep going + if(mgraManager.getRemoteParkingLot(originMaz)==0) + continue; + + int destinationTaz = mgraManager.getTaz(destinationMaz); + if(destinationTaz<=0) + continue; + + float dist = getDistance(originTaz, destinationTaz); + + //lowest distance, so reset the closest MAZ + if(dist pMap; + + logger.info(String.format("Household AV Fleet Allocation Program using CT-RAMP version ", + CtrampApplication.VERSION)); + + int iteration=0; + float sampleRate=1; + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else { + propertiesFile = args[0]; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.valueOf(args[i + 1]); + } + + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.valueOf(args[i + 1]); + } + + } + } + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + HouseholdAVAllocationModelRunner householdAVModel = new HouseholdAVAllocationModelRunner(pMap, iteration, sampleRate); + householdAVModel.initialize(); + householdAVModel.runModel(); + + + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java new file mode 100644 index 0000000..fa1c6bb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java @@ -0,0 +1,101 @@ +package org.sandag.abm.maas; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + */ +public class HouseholdAVAllocationModelTripUtilityDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelTripUtilityDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + int timeTrip; //trip period + + public HouseholdAVAllocationModelTripUtilityDMU() + { + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + } + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + methodIndexMap.put("getTimeTrip", 1); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getTimeTrip(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public int getTimeTrip() { + return timeTrip; + } + + public void setTimeTrip(int timeTrip) { + this.timeTrip = timeTrip; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java new file mode 100644 index 0000000..91373a0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java @@ -0,0 +1,212 @@ +package org.sandag.abm.maas; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + */ +public class HouseholdAVAllocationModelVehicleChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelVehicleChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + int vehicle1IsAvailable; + int vehicle2IsAvailable; + int vehicle3IsAvailable; + int personWithVehicle1; + int personWithVehicle2; + int personWithVehicle3; + float travelUtilityToPersonVeh1; + float travelUtilityToPersonVeh2; + float travelUtilityToPersonVeh3; + float minutesUntilNextTrip; + + public HouseholdAVAllocationModelVehicleChoiceDMU() + { + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + } + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + methodIndexMap.put("getVehicle1IsAvailable",1); + methodIndexMap.put("getVehicle2IsAvailable",2); + methodIndexMap.put("getVehicle3IsAvailable",3); + methodIndexMap.put("getPersonWithVehicle1",4); + methodIndexMap.put("getPersonWithVehicle2",5); + methodIndexMap.put("getPersonWithVehicle3",6); + methodIndexMap.put("getTravelUtilityToPersonVeh1",7); + methodIndexMap.put("getTravelUtilityToPersonVeh2",8); + methodIndexMap.put("getTravelUtilityToPersonVeh3",9); + methodIndexMap.put("getMinutesUntilNextTrip",10); + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 1: + return getVehicle1IsAvailable(); + case 2: + return getVehicle2IsAvailable(); + case 3: + return getVehicle3IsAvailable(); + case 4: + return getPersonWithVehicle1(); + case 5: + return getPersonWithVehicle2(); + case 6: + return getPersonWithVehicle3(); + case 7: + return getTravelUtilityToPersonVeh1(); + case 8: + return getTravelUtilityToPersonVeh2(); + case 9: + return getTravelUtilityToPersonVeh3(); + case 10: + return getMinutesUntilNextTrip(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + + + public float getTravelUtilityToPersonVeh1() { + return travelUtilityToPersonVeh1; + } + + public void setTravelUtilityToPersonVeh1(float travelUtilityToPersonVeh1) { + this.travelUtilityToPersonVeh1 = travelUtilityToPersonVeh1; + } + + public float getTravelUtilityToPersonVeh2() { + return travelUtilityToPersonVeh2; + } + + public void setTravelUtilityToPersonVeh2(float travelUtilityToPersonVeh2) { + this.travelUtilityToPersonVeh2 = travelUtilityToPersonVeh2; + } + + public float getTravelUtilityToPersonVeh3() { + return travelUtilityToPersonVeh3; + } + + public void setTravelUtilityToPersonVeh3(float travelUtilityToPersonVeh3) { + this.travelUtilityToPersonVeh3 = travelUtilityToPersonVeh3; + } + + public int getVehicle1IsAvailable() { + return vehicle1IsAvailable; + } + + public void setVehicle1IsAvailable(int vehicle1IsAvailable) { + this.vehicle1IsAvailable = vehicle1IsAvailable; + } + + public int getVehicle2IsAvailable() { + return vehicle2IsAvailable; + } + + public void setVehicle2IsAvailable(int vehicle2IsAvailable) { + this.vehicle2IsAvailable = vehicle2IsAvailable; + } + + public int getVehicle3IsAvailable() { + return vehicle3IsAvailable; + } + + public void setVehicle3IsAvailable(int vehicle3IsAvailable) { + this.vehicle3IsAvailable = vehicle3IsAvailable; + } + + public int getPersonWithVehicle1() { + return personWithVehicle1; + } + + public void setPersonWithVehicle1(int personWithVehicle1) { + this.personWithVehicle1 = personWithVehicle1; + } + + public int getPersonWithVehicle2() { + return personWithVehicle2; + } + + public void setPersonWithVehicle2(int personWithVehicle2) { + this.personWithVehicle2 = personWithVehicle2; + } + + public int getPersonWithVehicle3() { + return personWithVehicle3; + } + + public void setPersonWithVehicle3(int personWithVehicle3) { + this.personWithVehicle3 = personWithVehicle3; + } + + public float getMinutesUntilNextTrip() { + return minutesUntilNextTrip; + } + + public void setMinutesUntilNextTrip(float minutesUntilNextTrip) { + this.minutesUntilNextTrip = minutesUntilNextTrip; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java new file mode 100644 index 0000000..b1e0c6a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java @@ -0,0 +1,277 @@ +package org.sandag.abm.maas; + +/** + * A holder class for trips. + * @author joel.freedman + * + */ +class PersonTrip implements Comparable, Cloneable{ + protected String uniqueId; + protected long hhid; + protected long personId; + protected int personNumber; + protected int tourid; + protected int stopid; + protected int inbound; + protected int joint; + + protected int originMaz; + protected int destinationMaz; + protected short departPeriod; + protected float departTime; //minutes after 3 AM + protected float sampleRate; + protected int mode; + protected int parkingMaz; + protected byte avAvailable; + protected boolean rideSharer; + protected int pickupMaz; + protected int dropoffMaz; + + public PersonTrip(String uniqueId,long hhid,long personId,int personNumber, int tourid,int stopid,int inbound,int joint,int originMaz, int destinationMaz, int departPeriod, float departTime, float sampleRate, int mode, int boardingTap, int alightingTap, int set, boolean rideSharer){ + this.uniqueId = uniqueId; + this.hhid = hhid; + this.personId = personId; + this.personNumber = personNumber; + this.tourid = tourid; + this.stopid = stopid; + this.inbound = inbound; + this.joint = joint; + + this.originMaz = originMaz; + this.destinationMaz = destinationMaz; + this.departPeriod = (short) departPeriod; + this.departTime = departTime; + this.sampleRate = sampleRate; + this.mode = mode; + this.rideSharer = rideSharer; + + //set the pickup MAZ to the originMaz and dropoff MAZ to the destinationMaz + this.pickupMaz = originMaz; + this.dropoffMaz = destinationMaz; + + + + + } + + public String getUniqueId() { + return uniqueId; + } + + public void setUniqueId(String uniqueId) { + this.uniqueId = uniqueId; + } + public long getHhid() { + return hhid; + } + + public void setHhid(long hhid) { + this.hhid = hhid; + } + + public long getPersonId() { + return personId; + } + + public void setPersonId(long personId) { + this.personId = personId; + } + + public int getPersonNumber() { + return personNumber; + } + + public void setPersonNumber(int personNumber) { + this.personNumber = personNumber; + } + + public int getTourid() { + return tourid; + } + + public void setTourid(int tourid) { + this.tourid = tourid; + } + + public int getStopid() { + return stopid; + } + + public void setStopid(int stopid) { + this.stopid = stopid; + } + + public int getInbound() { + return inbound; + } + + public void setInbound(int inbound) { + this.inbound = inbound; + } + + public int getJoint() { + return joint; + } + + public void setJoint(int joint) { + this.joint = joint; + } + + public int getOriginMaz() { + return originMaz; + } + + public void setOriginMaz(int originMaz) { + this.originMaz = originMaz; + } + + public int getPickupMaz() { + return pickupMaz; + } + + public void setPickupMaz(int pickupMaz) { + this.pickupMaz = pickupMaz; + } + + public int getDestinationMaz() { + return destinationMaz; + } + + public void setDestinationMaz(int destinationMaz) { + this.destinationMaz = destinationMaz; + } + + public int getDropoffMaz() { + return dropoffMaz; + } + + public void setDropoffMaz(int dropoffMaz) { + this.dropoffMaz = dropoffMaz; + } + + public short getDepartPeriod() { + return departPeriod; + } + + public void setDepartPeriod(short departPeriod) { + this.departPeriod = departPeriod; + } + + public float getDepartTime() { + return departTime; + } + + public void setDepartTime(float departTime) { + this.departTime = departTime; + } + + public float getSampleRate() { + return sampleRate; + } + + public void setSampleRate(float sampleRate) { + this.sampleRate = sampleRate; + } + + public int getMode() { + return mode; + } + + public void setMode(int mode) { + this.mode = mode; + } + + + public int getParkingMaz() { + return parkingMaz; + } + + public void setParkingMaz(int parkingMaz) { + this.parkingMaz = parkingMaz; + } + + public int getAvAvailable() { + return avAvailable; + } + + public void setAvAvailable(byte avAvailable) { + this.avAvailable = avAvailable; + } + + public boolean isRideSharer() { + return rideSharer; + } + + public void setRideSharer(boolean rideSharer) { + this.rideSharer=rideSharer; + } + + + /** + * Compare based on departure time. + */ + public int compareTo(Object aThat) { + final int BEFORE = -1; + final int EQUAL = 0; + final int AFTER = 1; + + final PersonTrip that = (PersonTrip)aThat; + + //primitive numbers follow this form + if (this.departTime < that.departTime) return BEFORE; + if (this.departTime > that.departTime) return AFTER; + + return EQUAL; + } + + /** + * Override equals. + */ + public boolean equals(Object aThat){ + final PersonTrip that = (PersonTrip)aThat; + + if(this.uniqueId.compareTo(that.getUniqueId())==0) + return true; + + return false; + } + + + /** + * If this trip and thatTrip are both joint trips from the same travel party, return true, else return false + * the comparison is done based on uniqueID, where the end of the unique id is the participant number, + * for example "_1" versus "_2". The method compares the part of the unique ID before the participant number, + * and returns true if they are the same. + * + * @param thatTrip + * @return True if both from same party, else false + */ + public boolean sameParty(PersonTrip thatTrip) { + + if(joint==0) + return false; + + if(thatTrip.getJoint()==0) + return false; + + int lastIndex = uniqueId.lastIndexOf("_"); + String thisUniqueIdMinusParticipantID = uniqueId.substring(0, lastIndex); + + lastIndex = thatTrip.getUniqueId().lastIndexOf("_"); + String thatUniqueIdMinusParticipantID = thatTrip.getUniqueId().substring(0, lastIndex); + + if(thisUniqueIdMinusParticipantID.compareTo(thatUniqueIdMinusParticipantID)==0) + return true; + + return false; + + } + + public Object clone() throws + CloneNotSupportedException + { + PersonTrip trip = (PersonTrip) super.clone(); + trip.uniqueId= new String(); + return trip; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java new file mode 100644 index 0000000..5cffb2c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java @@ -0,0 +1,1087 @@ +package org.sandag.abm.maas; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; + +public class PersonTripManager { + + protected static final Logger logger = Logger.getLogger(PersonTripManager.class); + protected HashMap propertyMap = null; + protected MersenneTwister random; + protected ModelStructure modelStructure; + protected HashMap personTripMap; + protected ArrayList[][] personTripArrayByDepartureBinAndMaz; //an array of PersonTrips by departure time increment and origin MAZ. + protected ArrayList[] personTripArrayByDepartureBin; //an array of PersonTrips by departure time increment + protected double[] endTimeMinutes; // the period end time in number of minutes past 3 AM , starting in period 1 (index 1) + protected int iteration; + protected MgraDataManager mgraManager; + protected TazDataManager tazManager; + protected int idNumber; + protected int[] modesToKeep; + protected int[] rideShareEligibleModes; + protected int numberOfTimeBins; + protected int periodLengthInMinutes; + protected int minTaz; //the minimum taz number with mazs; any origin or destination person trip less than this will be skipped. + protected float maxWalkDistance; + + protected static final String ModelSeedProperty = "Model.Random.Seed"; + protected static final String DirectoryProperty = "Project.Directory"; + protected static final String IndivTripDataFileProperty = "Results.IndivTripDataFile"; + protected static final String JointTripDataFileProperty = "Results.JointTripDataFile"; + protected static final String ModesToKeepProperty = "Maas.RoutingModel.Modes"; + protected static final String SharedEligibleProperty = "Maas.RoutingModel.SharedEligible"; + protected static final String MaxWalkDistance = "Maas.RoutingModel.maxWalkDistance"; + + protected static final String MexResTripDataFileProperty ="crossBorder.trip.output.file"; + protected static final String VisitorTripDataFileProperty ="visitor.trip.output.file"; + protected static final String AirportSANTripDataFileProperty ="airport.SAN.output.file"; + protected static final String AirportCBXTripDataFileProperty ="airport.CBX.output.file"; + protected static final String IETripDataFileProperty ="internalExternal.trip.output.file"; + + + /** + * Constructor. + * + * @param propertyMap + * @param iteration + */ + public PersonTripManager(HashMap propertyMap, int iteration){ + this.iteration = iteration; + this.propertyMap = propertyMap; + + modelStructure = new SandagModelStructure(); + + } + + /** + * Initialize (not done by default). + * Initializes array to simulate actual departure time + * Reads in individual and joint person trips + */ + public void initialize(int periodLengthInMinutes){ + logger.info("Initializing PersonTripManager"); + + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + //find minimum TAZ with mazs + int maxTaz = tazManager.getMaxTaz(); + minTaz = -1; + for(int i=1;i<=maxTaz;++i){ + int[] mazs = tazManager.getMgraArray(i); + + if(mazs==null|| mazs.length==0) + minTaz = Math.max(i, minTaz); + } + + logger.info("Minimum TAZ number is "+minTaz); + logger.info("Maximum TAZ number is "+maxTaz); + + //initialize the end time in minutes (stored in double so no overlap between periods) + endTimeMinutes = new double[40+1]; + endTimeMinutes[1]=119.999999; //first period is 3-3:59:99:99 + for(int period=2;period 0, implement hotspots + if(maxWalkDistance>0) + moveRidesharersToHotspots(); + + logger.info("Completed Initializing PersonTripManager"); + + + } + + /** + * Read the input individual and joint trip files. This function calls the method + * @readTripList for each table. This method is called from {@initialize()} + */ + private void readInputFiles(){ + + String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); + String indivTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, IndivTripDataFileProperty); + indivTripFile = insertIterationNumber(indivTripFile,iteration); + String jointTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, JointTripDataFileProperty); + jointTripFile = insertIterationNumber(jointTripFile,iteration); + + //start with individual trips + TableDataSet indivTripDataSet = readTableData(indivTripFile); + personTripMap = readResidentTripList(personTripMap, indivTripDataSet, false); + int tripsSoFar=personTripMap.size(); + + logger.info("Read "+tripsSoFar+" individual person trips"); + + //now read joint trip data + TableDataSet jointTripDataSet = readTableData(jointTripFile); + personTripMap = readResidentTripList(personTripMap, jointTripDataSet, true); + + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" joint person trips"); + tripsSoFar=personTripMap.size(); + + + String mexicanResidentTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, MexResTripDataFileProperty); + TableDataSet mexicanResidentTripDataSet = readTableData(mexicanResidentTripFile); + personTripMap = readMexicanResidentTripList(personTripMap, mexicanResidentTripDataSet); + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" mexican resident person trips"); + tripsSoFar=personTripMap.size(); + + + String visitorTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, VisitorTripDataFileProperty); + TableDataSet visitorTripDataSet = readTableData(visitorTripFile); + personTripMap = readVisitorTripList(personTripMap, visitorTripDataSet); + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" visitor person trips"); + tripsSoFar=personTripMap.size(); + + String airportSANTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, AirportSANTripDataFileProperty); + TableDataSet airportSANTripDataSet = readTableData(airportSANTripFile); + personTripMap = readAirportTripList(personTripMap, airportSANTripDataSet, -6,"SAN"); + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" SAN airport person trips"); + tripsSoFar=personTripMap.size(); + + String airportCBXTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, AirportCBXTripDataFileProperty); + TableDataSet airportCBXTripDataSet = readTableData(airportCBXTripFile); + personTripMap = readAirportTripList(personTripMap, airportCBXTripDataSet, -5,"CBX"); + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" CBX airport person trips"); + tripsSoFar=personTripMap.size(); + + String ieTripFile = directory + + Util.getStringValueFromPropertyMap(propertyMap, IETripDataFileProperty); + TableDataSet ieTripDataSet = readTableData(ieTripFile); + personTripMap = readIETripList(personTripMap, ieTripDataSet); + logger.info("Read "+(personTripMap.size()-tripsSoFar)+" IE person trips"); + tripsSoFar=personTripMap.size(); + + logger.info("Read "+personTripMap.size()+" total person trips"); + + } + + /** + * Read the CTRAMP trip list in the TableDataSet. + * + * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. + * @param inputTripTableData The TableDataSet containing the CT-RAMP output trip file. + * @param jointTripData A boolean indicating whether the data is for individual or joint trips. + */ + public HashMap readResidentTripList(HashMap personTripMap, TableDataSet inputTripTableData, boolean jointTripData){ + + if(personTripMap==null) + personTripMap = new HashMap(); + + for(int row = 1; row <= inputTripTableData.getRowCount();++row){ + + + int mode = (int) inputTripTableData.getValueAt(row,"trip_mode"); + if(modesToKeep[mode]!=1) + continue; + + boolean rideShare=false; + if(rideShareEligibleModes[mode]==1) + rideShare=true; + + int oMaz = (int) inputTripTableData.getValueAt(row,"orig_mgra"); + int dMaz = (int) inputTripTableData.getValueAt(row,"dest_mgra"); + + int oTaz = mgraManager.getTaz(oMaz); + int dTaz = mgraManager.getTaz(dMaz); + + if((oTaz1) { + personTrip.setJoint(1); + personTrip.setUniqueId(uniqueID+"_1"); + } + personTripMap.put(idNumber, personTrip); + + //replicate joint trips + if(num_participants>1) + for(int i=2;i<=num_participants;++i){ + ++idNumber; + PersonTrip newTrip = null; + try { + newTrip = (PersonTrip) personTrip.clone(); + }catch(Exception e) { + + logger.fatal("Error attempting to clone joint trip object "+uniqueID); + throw new RuntimeException(e); + } + newTrip.setUniqueId(uniqueID+"_"+i); + personTripMap.put(idNumber, newTrip); + } + } + + return personTripMap; + } + + + /** + * Read the visitor trip list in the TableDataSet. + * + * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. + * @param inputTripTableData The TableDataSet containing the visitor output trip file. + */ + public HashMap readVisitorTripList(HashMap personTripMap, TableDataSet inputTripTableData){ + + if(personTripMap==null) + personTripMap = new HashMap(); + + for(int row = 1; row <= inputTripTableData.getRowCount();++row){ + + + int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); + if(modesToKeep[mode]!=1) + continue; + + boolean rideShare=false; + if(rideShareEligibleModes[mode]==1) + rideShare=true; + + int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); + int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); + + int oTaz = mgraManager.getTaz(oMaz); + int dTaz = mgraManager.getTaz(dMaz); + + if((oTaz1) { + personTrip.setJoint(1); + personTrip.setUniqueId(uniqueID+"_1"); + } + personTripMap.put(idNumber, personTrip); + + //replicate joint trips + if(num_participants>1) + for(int i=2;i<=num_participants;++i){ + ++idNumber; + PersonTrip newTrip = null; + try { + newTrip = (PersonTrip) personTrip.clone(); + }catch(Exception e) { + + logger.fatal("Error attempting to clone joint trip object "+uniqueID); + throw new RuntimeException(e); + } + newTrip.setUniqueId(uniqueID+"_"+i); + personTripMap.put(idNumber, newTrip); + } + + } + + return personTripMap; + } + + /** + * Read the Mexican resident trip list in the TableDataSet. + * + * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. + * @param inputTripTableData The TableDataSet containing the visitor output trip file. + */ + public HashMap readMexicanResidentTripList(HashMap personTripMap, TableDataSet inputTripTableData){ + + if(personTripMap==null) + personTripMap = new HashMap(); + + for(int row = 1; row <= inputTripTableData.getRowCount();++row){ + + int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); + if(modesToKeep[mode]!=1) + continue; + + boolean rideShare=false; + if(rideShareEligibleModes[mode]==1) + rideShare=true; + + int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); + int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); + + int oTaz = mgraManager.getTaz(oMaz); + int dTaz = mgraManager.getTaz(dMaz); + + if((oTaz1) { + personTrip.setJoint(1); + personTrip.setUniqueId(uniqueID+"_1"); + } + personTripMap.put(idNumber, personTrip); + + //replicate joint trips + if(num_participants>1) + for(int i=2;i<=num_participants;++i){ + ++idNumber; + PersonTrip newTrip = null; + try { + newTrip = (PersonTrip) personTrip.clone(); + }catch(Exception e) { + + logger.fatal("Error attempting to clone joint trip object "+uniqueID); + throw new RuntimeException(e); + } + newTrip.setUniqueId(uniqueID+"_"+i); + personTripMap.put(idNumber, newTrip); + } + + + } + + return personTripMap; + } + + /** + * Read the airport trip list in the TableDataSet. + * + * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. + * @param inputTripTableData The TableDataSet containing the visitor output trip file. + */ + public HashMap readAirportTripList(HashMap personTripMap, TableDataSet inputTripTableData, int default_id, String airportCode){ + + if(personTripMap==null) + personTripMap = new HashMap(); + + for(int row = 1; row <= inputTripTableData.getRowCount();++row){ + + int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); + if(modesToKeep[mode]!=1) + continue; + + boolean rideShare=false; + if(rideShareEligibleModes[mode]==1) + rideShare=true; + + int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); + int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); + + int oTaz = mgraManager.getTaz(oMaz); + int dTaz = mgraManager.getTaz(dMaz); + + if((oTaz1) { + personTrip.setJoint(1); + personTrip.setUniqueId(uniqueID+"_1"); + + } + personTripMap.put(idNumber, personTrip); + + //replicate joint trips + if(num_participants>1) + for(int i=2;i<=num_participants;++i){ + ++idNumber; + PersonTrip newTrip = null; + try { + newTrip = (PersonTrip) personTrip.clone(); + }catch(Exception e) { + + logger.fatal("Error attempting to clone joint trip object "+uniqueID); + throw new RuntimeException(e); + } + newTrip.setUniqueId(uniqueID+"_"+i); + personTripMap.put(idNumber, newTrip); + } + + + } + + return personTripMap; + } + + + /** + * Read the IE trip list in the TableDataSet. + * + * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. + * @param inputTripTableData The TableDataSet containing the visitor output trip file. + */ + public HashMap readIETripList(HashMap personTripMap, TableDataSet inputTripTableData){ + + if(personTripMap==null) + personTripMap = new HashMap(); + + for(int row = 1; row <= inputTripTableData.getRowCount();++row){ + + int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); + if(modesToKeep[mode]!=1) + continue; + + boolean rideShare=false; + if(rideShareEligibleModes[mode]==1) + rideShare=true; + + int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); + int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); + + int oTaz = (int) inputTripTableData.getValueAt(row,"originTAZ"); + int dTaz = (int) inputTripTableData.getValueAt(row,"destinationTAZ"); + + if((oTaz1) { + personTrip.setJoint(1); + personTrip.setUniqueId(uniqueID+"_1"); + } + personTripMap.put(idNumber, personTrip); + + //replicate joint trips + if(num_participants>1) + for(int i=2;i<=num_participants;++i){ + ++idNumber; + PersonTrip newTrip = null; + try { + newTrip = (PersonTrip) personTrip.clone(); + }catch(Exception e) { + + logger.fatal("Error attempting to clone joint trip object "+uniqueID); + throw new RuntimeException(e); + } + newTrip.setUniqueId(uniqueID+"_"+i); + personTripMap.put(idNumber, newTrip); + } + + } + + return personTripMap; + } + /** + * Simulate the exact time for the period. + * + * @param period The time period (1->40) + * @return The exact time in double precision (number of minutes past 3 AM) + */ + public float simulateExactTime(int period){ + + double lowerEnd = endTimeMinutes[period-1]; + double upperEnd = endTimeMinutes[period]; + double randomNumber = random.nextDouble(); + + float time = (float) ((upperEnd - lowerEnd) * randomNumber + lowerEnd); + + return time; + } + + /** + * A simple helper function to insert the iteration number into the file name. + * + * @param filename The input file name (ex: inputFile.csv) + * @param iteration The iteration number (ex: 3) + * @return The new string (ex: inputFile_3.csv) + */ + private String insertIterationNumber(String filename, int iteration){ + + String newFileName = filename.replace(".csv", "_"+new Integer(iteration).toString()+".csv"); + return newFileName; + } + + /** + * Read data into inputDataTable tabledataset. + * + */ + private TableDataSet readTableData(String inputFile){ + + TableDataSet tableDataSet = null; + + logger.info("Begin reading the data in file " + inputFile); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + tableDataSet = csvFile.readFile(new File(inputFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End reading the data in file " + inputFile); + + return tableDataSet; + } + + /** + * Go through the person trip list, sort the person trips by departure time and MAZ. + * + */ + @SuppressWarnings("unchecked") + public void groupPersonTripsByDepartureTimePeriodAndOrigin(){ + + numberOfTimeBins = ((24*60)/periodLengthInMinutes); + int maxMaz = mgraManager.getMaxMgra(); + + logger.info("Calculated "+numberOfTimeBins+" simulation periods using a period length of "+periodLengthInMinutes+" minutes"); + personTripArrayByDepartureBinAndMaz = new ArrayList[numberOfTimeBins][maxMaz+1]; + personTripArrayByDepartureBin = new ArrayList[numberOfTimeBins]; + + //initialize + for(int i = 0; i < numberOfTimeBins;++i){ + personTripArrayByDepartureBin[i] = new ArrayList(); + for(int j = 0; j <=maxMaz;++j){ + personTripArrayByDepartureBinAndMaz[i][j] = new ArrayList(); + } + + } + + Collection personTripList = personTripMap.values(); + for(PersonTrip personTrip : personTripList){ + + int originMaz = personTrip.getPickupMaz(); + + float departTime = personTrip.getDepartTime(); + int bin = (int) Math.floor(departTime/((float) periodLengthInMinutes)); + + personTripArrayByDepartureBinAndMaz[bin][originMaz].add(personTrip); + personTripArrayByDepartureBin[bin].add(personTrip); + + + } + } + + + /** + * Get the person trips for the period bin (indexed from 0) and the origin MAZ. + * + * @param periodBin The number of the departure time period bin based on the period length used to group person trips. + * @param maz The number of the origin MAZ. + * + * @return An arraylist of person trips. + */ + public ArrayList getPersonTripsByDepartureTimePeriodAndMaz(int periodBin, int maz){ + + return personTripArrayByDepartureBinAndMaz[periodBin][maz]; + + } + + /** + * Sample a person trip from the array for the given period. REMOVE IT from the person trip arrays. + * + * @param simulationPeriod The simulation period to sample a trip from. + * @param rnum A random number to be used in sampling. + * @return A person trip, or null if the ArrayList is null or empty. + */ + PersonTrip samplePersonTrip(int simulationPeriod, double rnum){ + + ArrayList personTripArray = personTripArrayByDepartureBin[simulationPeriod]; + + if(personTripArray==null) + return null; + + int listSize = personTripArray.size(); + + if(listSize==0) + return null; + + int element = (int) Math.floor(rnum * listSize); + PersonTrip personTrip = personTripArray.get(element); + personTripArrayByDepartureBin[simulationPeriod].remove(personTrip); + personTripArrayByDepartureBinAndMaz[simulationPeriod][personTrip.getPickupMaz()].remove(personTrip); + + return personTrip; + } + + /** + * Sample a person trip from the array for the given period. REMOVE IT from the array. + * + * @param simulationPeriod The period to sample a trip from. + * @param maz The maz to sample a trip from. + * @param rnum A random number to be used in sampling. + * @return A person trip, or null if the ArrayList is null or empty. + */ + PersonTrip samplePersonTrip(int simulationPeriod, int maz, double rnum){ + + ArrayList personTripArray = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; + + if(personTripArray==null) + return null; + + int listSize = personTripArray.size(); + + if(listSize==0) + return null; + + int element = (int) Math.floor(rnum * listSize); + PersonTrip personTrip = personTripArray.get(element); + personTripArrayByDepartureBinAndMaz[simulationPeriod][maz].remove(personTrip); + personTripArrayByDepartureBin[simulationPeriod].remove(personTrip); + + return personTrip; + } + + /** + * Check if there are more person trips in this simulation period. + * + * @param simulationPeriod + * @return true if there are more person trips, false if not. + */ + public boolean morePersonTripsInSimulationPeriod(int simulationPeriod){ + ArrayList personTripArray = personTripArrayByDepartureBin[simulationPeriod]; + + if(personTripArray==null) + return false; + + int listSize = personTripArray.size(); + + if(listSize==0) + return false; + + return true; + + } + + /** + * Check if there are more person trips in this simulation period and maz. + * + * @param simulationPeriod + * @return true if there are more person trips, false if not. + */ + public boolean morePersonTripsInSimulationPeriodAndMaz(int simulationPeriod, int maz){ + ArrayList personTripArray = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; + + if(personTripArray==null) + return false; + + int listSize = personTripArray.size(); + + if(listSize==0) + return false; + + return true; + + } + /** + * Remove the person trip. + * + * @param trip + * @param simulationPeriod + */ + public void removePersonTrip(PersonTrip trip, int simulationPeriod){ + + int originMaz = trip.getPickupMaz(); + personTripArrayByDepartureBin[simulationPeriod].remove(trip); + personTripArrayByDepartureBinAndMaz[simulationPeriod][originMaz].remove(trip); + + } + + /** + * Pre-process the array of person trips by moving nearby ridesharers to hotspots. + * The algorithm finds the maz in each TAZ and simulation period with the most ridesharers. + * It moves the ridesharers within the maximum walking distance to that MAZ. + */ + public void moveRidesharersToHotspots() { + + logger.info("Hotspots - moving ride-sharers to high demand MAZs"); + + int[] tazs = tazManager.getTazs(); + int maxTaz = tazManager.getMaxTaz(); + + //store the hotspot Maz for each period and taz + int[][] hotspotMazs = new int[numberOfTimeBins][maxTaz+1]; + + //track the number of ridesharers moved + int totalRidesharersMoved = 0; + + for(int simulationPeriod=0;simulationPeriod personTrips = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; + if(personTrips==null) + continue; + + //set maz and max ridesharers + if(personTrips.size()>maxRideSharers) { + maxRideSharers= personTrips.size(); + hotspotMaz = maz; + hotspotMazs[simulationPeriod][taz] = hotspotMaz; + } + } // end mazs + + //no mazs with ridesharers in this taz and simulation period + if(maxRideSharers==-1) + continue; + else { + + //get nearby ridesharers and move them to hotspot + ArrayList nearbySharers = findNearbyRideSharersByOriginMaz(hotspotMaz, simulationPeriod, mgraManager.getTaz(hotspotMaz)); + + if(nearbySharers==null) + continue; + + if(nearbySharers.size()==0) + continue; + + // logger.info("TAZ "+taz+": found "+nearbySharers.size()+" to move to hotspot MAZ "+hotspotMaz+" in period "+simulationPeriod); + + //add each ridesharer to the person trip array at the hotspot maz, and remove them from their origin + for(PersonTrip personTrip : nearbySharers) { + int originMaz = personTrip.getOriginMaz(); + personTrip.setPickupMaz(hotspotMaz); + personTripArrayByDepartureBinAndMaz[simulationPeriod][originMaz].remove(personTrip); + personTripArrayByDepartureBinAndMaz[simulationPeriod][hotspotMaz].add(personTrip); + ++ridesharersMoved; + ++totalRidesharersMoved; + } + } + } //end for zones + + + + logger.info("Simulation period "+ simulationPeriod+" moved "+ridesharersMoved+" ridesharers"); + } // end for simulation periods + + //now move dropoffs to hotspot locations + int movedDropoffs = 0; + Collection personTripList = personTripMap.values(); + for(PersonTrip personTrip : personTripList){ + + //skip non-ride shareres + if(!personTrip.isRideSharer()) + continue; + + int destinationMaz = personTrip.getDestinationMaz(); + int destinationTaz = mgraManager.getTaz(destinationMaz); + float departTime = personTrip.getDepartTime(); + int departBin = (int) Math.floor(departTime/((float) periodLengthInMinutes)); + int hotspotMaz = hotspotMazs[departBin][destinationTaz]; + + //no hotspot for this person's destination taz + if(hotspotMaz==0) + continue; + + float distance = ((float) mgraManager.getMgraToMgraWalkDistFrom(destinationMaz,hotspotMaz))/((float)5280.0); + + if(distance==0) + continue; + + //distance between destination and hotspot is less than max walk distance, so move this person + if(distance<=maxWalkDistance) { + personTrip.setDropoffMaz(hotspotMaz); + ++movedDropoffs; + } + } + + logger.info("Hotspots moved "+totalRidesharersMoved+" ride-share pickups and "+movedDropoffs+" dropoffs"); + } + + + + /** + * Cycle through all the MAZs within maximum walk distance of the origin MAZ, and find + * rideshare passengers departing within the same period. Add them to an ArrayList and + * return it. + * + * @param originMaz The origin for searching + * @param simulationPeriod The simulation period + * @return The ArrayList of ridesharers. + */ + public ArrayList findNearbyRideSharersByOriginMaz(int originMaz, int simulationPeriod, int constraintTaz) { + + int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(originMaz); + + if(walkMgras==null) + return null; + + ArrayList nearbyRideSharers = new ArrayList(); + + //cycle through walk mgras + for(int walkMgra : walkMgras) { + + //skip intrazonal + if(walkMgra==originMaz) + continue; + + if(constraintTaz>0) + if(mgraManager.getTaz(walkMgra)!=constraintTaz) + continue; + + //walk mgra is less than max walk distance + float distance = ((float) mgraManager.getMgraToMgraWalkDistFrom(originMaz,walkMgra))/((float)5280.0); + + if(distance==0) + continue; + if(distance<=maxWalkDistance) { + + ArrayList personTrips = personTripArrayByDepartureBinAndMaz[simulationPeriod][walkMgra]; + + //cycle through person trips in this mgra and add them to the array if they are willing to rideshare + for(PersonTrip personTrip : personTrips) { + + if(personTrip.isRideSharer()) { + nearbyRideSharers.add(personTrip); + } + } + } + } + + return nearbyRideSharers; + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java new file mode 100644 index 0000000..66e2cd1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java @@ -0,0 +1,448 @@ +package org.sandag.abm.maas; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +public class TNCFleetModel { + + private static final Logger logger = Logger.getLogger(TNCFleetModel.class); + private HashMap propertyMap = null; + TransportCostManager transportCostManager; //manages transport costs! + PersonTripManager personTripManager; //manages person trips! + TNCVehicleManager tNCVehicleManager; //manages vehicles! + + private int iteration; + private float sampleRate; + private int minutesPerSimulationPeriod; + private int numberOfSimulationPeriods; + private MersenneTwister random; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private byte maxSharedTNCPassengers; + + MatrixDataServerRmi ms; + private byte[] skimPeriodLookup; //an array indexed by number of periods that corresponds to the skim period + + private boolean routeIntrazonal; + + private static final String MAX_PICKUP_DISTANCE_PROPERTY = "Maas.RoutingModel.maxDistanceForPickup"; + private static final String MAX_PICKUP_DIVERSON_TIME_PROPERTY = "Maas.RoutingModel.maxDiversionTimeForPickup"; + private static final String MINUTES_PER_SIMULATION_PERIOD_PROPERTY = "Maas.RoutingModel.minutesPerSimulationPeriod"; + private static final String MAX_SHARED_TNC_PASSENGERS_PROPERTY = "Maas.RoutingModel.maxPassengers"; + private static final String ROUTE_INTRAZONAL_PROPERTY = "Maas.RoutingModel.routeIntrazonal"; + private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; + + int vehicleDebug; + /** + * Constructor. + * + * @param propertyMap + * @param iteration + */ + public TNCFleetModel(HashMap propertyMap, int iteration, float sampleRate){ + this.propertyMap = propertyMap; + this.iteration = iteration; + this.sampleRate = sampleRate; + } + + /** + * Initialize all the data members. + * + */ + public void initialize(){ + + startMatrixServer(propertyMap); + + //managers for MAZ and TAZ data + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + //some controlling properties + float maxPickupDistance = Util.getFloatValueFromPropertyMap(propertyMap, MAX_PICKUP_DISTANCE_PROPERTY); + float maxDiversionTime = Util.getFloatValueFromPropertyMap(propertyMap, MAX_PICKUP_DIVERSON_TIME_PROPERTY); + maxSharedTNCPassengers = (byte) Util.getIntegerValueFromPropertyMap(propertyMap, MAX_SHARED_TNC_PASSENGERS_PROPERTY); + routeIntrazonal = Util.getBooleanValueFromPropertyMap(propertyMap, ROUTE_INTRAZONAL_PROPERTY); + + //set the length of a simulation period + minutesPerSimulationPeriod = Util.getIntegerValueFromPropertyMap(propertyMap, MINUTES_PER_SIMULATION_PERIOD_PROPERTY); + numberOfSimulationPeriods = ((24*60)/minutesPerSimulationPeriod); + logger.info("Running "+numberOfSimulationPeriods+" simulation periods using a period length of "+minutesPerSimulationPeriod+" minutes"); + calculateSkimPeriods(); + + //create a new transport cost manager and create data structures + transportCostManager = new TransportCostManager(propertyMap,maxDiversionTime,maxPickupDistance); + transportCostManager.initialize(); + transportCostManager.calculateTazsByTimeFromOrigin(); + + //create a person trip manager, read person trips + personTripManager = new PersonTripManager(propertyMap, iteration); + personTripManager.initialize(minutesPerSimulationPeriod); + + + //create a tNCVehicle manager + tNCVehicleManager = new TNCVehicleManager(propertyMap, transportCostManager, maxSharedTNCPassengers, minutesPerSimulationPeriod); + tNCVehicleManager.initialize(); + vehicleDebug = tNCVehicleManager.vehicleDebug; + + //seed the random number generator so that results can be replicated if desired. + int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); + random = new MersenneTwister(seed + 4292); + + } + + /** + * Relate simulation periods to skim periods. + * + */ + public void calculateSkimPeriods(){ + + skimPeriodLookup = new byte[numberOfSimulationPeriods]; + int numberSkimPeriods = ModelStructure.SKIM_PERIOD_INDICES.length; + int[] endSkimPeriod = new int[numberSkimPeriods]; + + int lastPeriodEnd = 0; + int lastEndSkimPeriod = 0; + for(int skimPeriod = 0;skimPeriod=0;--skimPeriod){ + if(period pMap; + + logger.info(String.format("TNC Fleet Simulation Program using CT-RAMP version ", + CtrampApplication.VERSION)); + + int iteration=0; + float sampleRate=1; + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else { + propertiesFile = args[0]; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.valueOf(args[i + 1]); + } + + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.valueOf(args[i + 1]); + } + + + + } + } + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + TNCFleetModel fleetModel = new TNCFleetModel(pMap, iteration, sampleRate); + fleetModel.initialize(); + fleetModel.runModel(); + + + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java new file mode 100644 index 0000000..837d3f9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java @@ -0,0 +1,156 @@ +package org.sandag.abm.maas; + +import java.util.ArrayList; + +public class TNCVehicle { + + + protected ArrayList personTripList; + + protected ArrayList tNCVehicleTrips; + protected byte maxPassengers; + protected short generationTaz; + protected short generationPeriod; + protected int id; + protected float maxDistanceBeforeRefuel; + protected float distanceSinceRefuel; + protected int periodsRefueling; + + /** + * Create a new tNCVehicle. + * + * @param id + * @param maxPassengers + * @param maxDistanceBeforeRefuel + */ + public TNCVehicle(int id, byte maxPassengers, float maxDistanceBeforeRefuel){ + this.id= id; + this.maxPassengers = maxPassengers; + personTripList = new ArrayList(); + tNCVehicleTrips = new ArrayList(); + this.maxDistanceBeforeRefuel = maxDistanceBeforeRefuel; + } + + /** + * Add a tNCVehicle trip to this tNCVehicle. + * + * @param tNCVehicleTrip + */ + public void addVehicleTrip(TNCVehicleTrip tNCVehicleTrip){ + tNCVehicleTrips.add(tNCVehicleTrip); + } + + /** + * Add an ArrayList of tNCVehicle trips to this tNCVehicle. + * + * @param tNCVehicleTrips + */ + public void addVehicleTrips(ArrayList tNCVehicleTrips){ + this.tNCVehicleTrips.addAll(tNCVehicleTrips); + } + + /** + * Clear all the person trips from this tNCVehicle. Used after routing the tNCVehicle. + * + */ + public void clearPersonTrips(){ + this.personTripList.clear(); + } + + /** + * Get all the tNCVehicle trips for this tNCVehicle. + * + * @return VehicleTrips + */ + public ArrayList getVehicleTrips(){ + + return tNCVehicleTrips; + } + + /** + * Get the tNCVehicle ID. + * + * @return + */ + public int getId(){ + return this.id; + } + + /** + * Add a passenger to the tNCVehicle. + * + * @param personTrip + */ + public void addPersonTrip(PersonTrip personTrip){ + + personTripList.add(personTrip); + + } + + /** + * Remove one person trip from the tNCVehicle. Used after routing. + * + * @param personTrip + */ + public void removePersonTrip(PersonTrip personTrip){ + + personTripList.remove(personTrip); + } + + /** + * Get number of passengers. + * + * @return The number of passengers + */ + public byte getNumberPassengers(){ + + return (byte) personTripList.size(); + } + + public byte getMaxPassengers() { + return maxPassengers; + } + + public void setMaxPassengers(byte maxPassengers) { + this.maxPassengers = maxPassengers; + } + + public short getGenerationTaz() { + return generationTaz; + } + + public void setGenerationTaz(short generationTaz) { + this.generationTaz = generationTaz; + } + + public short getGenerationPeriod() { + return generationPeriod; + } + + public void setGenerationPeriod(short generationPeriod) { + this.generationPeriod = generationPeriod; + } + + public ArrayList getPersonTripList() { + return personTripList; + } + + public float getDistanceSinceRefuel() { + return distanceSinceRefuel; + } + + public void setDistanceSinceRefuel(float distanceSinceRefuel) { + this.distanceSinceRefuel = distanceSinceRefuel; + } + + public int getPeriodsRefueling() { + return periodsRefueling; + } + + public void setPeriodsRefueling(int periodsRefueling) { + this.periodsRefueling = periodsRefueling; + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java new file mode 100644 index 0000000..78e411f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java @@ -0,0 +1,878 @@ +package org.sandag.abm.maas; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.maas.TNCVehicleTrip.Purpose; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.math.MersenneTwister; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; + +public class TNCVehicleManager { + + protected static final Logger logger = Logger.getLogger(TNCVehicleManager.class); + protected HashMap propertyMap = null; + protected ArrayList[] emptyVehicleList; //by taz + protected ArrayList vehiclesToRouteList; + protected ArrayList activeVehicleList; + protected ArrayList refuelingVehicleList; + + protected MersenneTwister random; + protected static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; + protected static final String VEHICLETRIP_OUTPUT_FILE_PROPERTY = "Maas.RoutingModel.vehicletrip.output.file"; + protected static final String VEHICLETRIP_OUTPUT_MATRIX_PROPERTY = "Maas.RoutingModel.vehicletrip.output.matrix"; + protected static final String MAX_DISTANCE_BEFORE_REFUEL_PROPERTY = "Maas.RoutingModel.maxDistanceBeforeRefuel"; + protected static final String TIME_REQUIRED_FOR_REFUEL_PROPERTY = "Maas.RoutingModel.timeRequiredForRefuel"; + + + protected String vehicleTripOutputFile; + protected TazDataManager tazManager; + protected MgraDataManager mazManager; + protected int maxTaz; + protected int maxMaz; + protected byte maxPassengers; + protected TransportCostManager transportCostManager; + protected int totalVehicles; + protected int minutesPerSimulationPeriod; + protected int vehicleDebug; + protected int totalVehicleTrips; + private byte[] skimPeriodLookup; //an array indexed by number of periods that corresponds to the skim period + private int numberOfSimulationPeriods; + protected float maxDistanceBeforeRefuel; + protected float timeRequiredForRefuel; + protected int periodsRequiredForRefuel; + protected int[] closestMazWithRefeulingStation ; //the closest MAZ with a refueling station + + // one file per time period + // matrices are indexed by periods, occupants + private Matrix[][] TNCTripMatrix; + + + + + /** + * + * @param propertyMap + * @param transportCostManager + */ + public TNCVehicleManager(HashMap propertyMap, TransportCostManager transportCostManager, byte maxPassengers, int minutesPerSimulationPeriod){ + + this.propertyMap = propertyMap; + this.transportCostManager = transportCostManager; + this.maxPassengers = maxPassengers; + this.minutesPerSimulationPeriod = minutesPerSimulationPeriod; + numberOfSimulationPeriods = ((24*60)/minutesPerSimulationPeriod); + + } + + @SuppressWarnings("unchecked") + public void initialize(){ + + int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); + random = new MersenneTwister(seed + 234324); + + tazManager = TazDataManager.getInstance(); + maxTaz = tazManager.getMaxTaz(); + + mazManager = MgraDataManager.getInstance(); + maxMaz = mazManager.getMaxMgra(); + + emptyVehicleList = new ArrayList[maxTaz+1]; + + activeVehicleList = new ArrayList(); + vehiclesToRouteList = new ArrayList(); + refuelingVehicleList = new ArrayList(); + + String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); + vehicleTripOutputFile = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_FILE_PROPERTY); + + maxDistanceBeforeRefuel = Util.getFloatValueFromPropertyMap(propertyMap, MAX_DISTANCE_BEFORE_REFUEL_PROPERTY); + timeRequiredForRefuel = Util.getFloatValueFromPropertyMap(propertyMap, TIME_REQUIRED_FOR_REFUEL_PROPERTY); + periodsRequiredForRefuel = (int) Math.ceil(timeRequiredForRefuel/minutesPerSimulationPeriod); + + vehicleDebug = 1; + + calculateClosestRefuelingMazs(); + + calculateSkimPeriods(); + + //initialize the matrices for writing trips + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + TNCTripMatrix = new Matrix[transportCostManager.NUM_PERIODS][]; + + for(int i =0;i0){ + + TNCVehicle tNCVehicle = null; + double rnum = random.nextDouble(); + tNCVehicle = getRandomVehicleFromList(emptyVehicleList[taz], rnum); + + //generate an empty trip for the tNCVehicle + ArrayList trips = tNCVehicle.getVehicleTrips(); + + if(trips.size()==0){ + logger.warn("Weird: got an empty tNCVehicle (id:"+tNCVehicle.getId()+") but no trips in tNCVehicle trip list"); + return tNCVehicle; + } + + TNCVehicleTrip lastTrip = trips.get(trips.size()-1); + int originTaz = lastTrip.getDestinationTaz(); + int originMaz = lastTrip.getDestinationMaz(); + ArrayList pickupIdsAtOrigin = lastTrip.getPickupIdsAtDestination(); + ArrayList dropoffIdsAtOrigin = lastTrip.getDropoffIdsAtDestination(); + + ++totalVehicleTrips; + TNCVehicleTrip newTrip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); + + newTrip.setOriginMaz(originMaz); + newTrip.setOriginTaz((short) originTaz); + newTrip.setDestinationMaz(departureMaz); + newTrip.setDestinationTaz((short)departureTaz); + newTrip.setStartPeriod(simulationPeriod); + newTrip.setEndPeriod(simulationPeriod); //instantaneous arrivals? Need traveling tNCVehicle queue... + if(lastTrip.getDestinationPurpose()==TNCVehicleTrip.Purpose.REFUEL) + newTrip.setOriginPurpose(TNCVehicleTrip.Purpose.REFUEL); + else { + newTrip.setPickupIdsAtOrigin(pickupIdsAtOrigin); + newTrip.setDropoffIdsAtOrigin(dropoffIdsAtOrigin); + } + newTrip.setDestinationPurpose(TNCVehicleTrip.Purpose.PICKUP_ONLY); + tNCVehicle.addVehicleTrip(newTrip); + + return tNCVehicle; + } + + } + + //Iterated through all TAZs, could not find an empty tNCVehicle. Return a new tNCVehicle. + return generateVehicle(simulationPeriod, departureTaz); + } + + /** + * Get a tNCVehicle at random from the arraylist of vehicles, remove it from the list, and return it. + * + * @param emptyVehicleList + * @param rnum a random number used to draw a tNCVehicle from the list. + * @return The tNCVehicle chosen. + */ + private TNCVehicle getRandomVehicleFromList(ArrayList vehicleList, double rnum){ + + + if(vehicleList==null) + return null; + + int listSize = vehicleList.size(); + int element = (int) Math.floor(rnum * listSize); + TNCVehicle tNCVehicle = vehicleList.get(element); + vehicleList.remove(element); + return tNCVehicle; + + } + + /** + * Encapsulating in method so that vehicles and some statistics can be tracked. + */ + private synchronized TNCVehicle generateVehicle(int simulationPeriod, int taz){ + ++totalVehicles; + TNCVehicle tNCVehicle = new TNCVehicle(totalVehicles, maxPassengers, maxDistanceBeforeRefuel); + tNCVehicle.setGenerationPeriod((short)simulationPeriod); + tNCVehicle.setGenerationTaz((short) taz); + return tNCVehicle; + + } + + public int getTotalVehicles(){ + return totalVehicles; + } + + /** + * Add empty tNCVehicle to the empty tNCVehicle list. + * + * @param tNCVehicle + * @param taz + */ + public void storeEmptyVehicle(TNCVehicle tNCVehicle, int taz){ + + if(emptyVehicleList[taz] == null) + emptyVehicleList[taz] = new ArrayList(); + + emptyVehicleList[taz].add(tNCVehicle); + + } + + public void addActiveVehicle(TNCVehicle tNCVehicle){ + + activeVehicleList.add(tNCVehicle); + } + + public void addVehicleToRoute(TNCVehicle tNCVehicle){ + + ArrayList personTrips = tNCVehicle.getPersonTripList(); + if(personTrips.size()==0){ + logger.info("Adding tNCVehicle "+tNCVehicle.getId()+" to vehicles to route list but no person trips"); + throw new RuntimeException(); + } + vehiclesToRouteList.add(tNCVehicle); + } + + /** + * All active vehicles are assigned passengers, now they must be routed through all pickups and dropoffs. + * THe method iterates through the vehiclesToRouteList and adds passengers based on the out-direction + * time required to pick them up and drop them off. + * + * @param skimPeriod + * @param simulationPeriod + * @param transportCostManager + */ + public synchronized void routeActiveVehicles(int skimPeriod, int simulationPeriod, TransportCostManager transportCostManager){ + + logger.info("Routing "+vehiclesToRouteList.size()+" vehicles in period "+simulationPeriod); + ArrayList vehiclesToRemove = new ArrayList(); + + + //iterate through vehicles to route list + for(TNCVehicle tNCVehicle: vehiclesToRouteList){ + + // get the person list, if it is empty throw a warning (should never be empty) + ArrayList personTrips = tNCVehicle.getPersonTripList(); + if(personTrips==null||personTrips.size()==0){ + logger.error("Attempting to route empty tNCVehicle "+tNCVehicle.getId()); + } + + if(tNCVehicle.getId()==vehicleDebug){ + logger.info("***********************************************************************************"); + logger.info("Debugging Vehicle routing for vehicle ID "+tNCVehicle.getId()); + logger.info("***********************************************************************************"); + logger.info("There are "+personTrips.size()+" person trips in vehicle ID "+tNCVehicle.getId()); + for(PersonTrip pTrip: personTrips){ + logger.info("Vehicle ID "+tNCVehicle.getId()+" person trip id: "+pTrip.getUniqueId()+" from pickup MAZ: "+pTrip.getPickupMaz()+ " to dropoff MAZ "+pTrip.getDropoffMaz()); + } + } + //some information on the first passenger + PersonTrip firstTrip = personTrips.get(0); + int firstOriginMaz = firstTrip.getPickupMaz(); + int firstOriginTaz = mazManager.getTaz(firstOriginMaz); + int firstDestinationMaz = firstTrip.getDropoffMaz(); + int firstDestinationTaz = mazManager.getTaz(firstDestinationMaz); + + // get the arraylist of tNCVehicle trips for this tNCVehicle + ArrayList existingVehicleTrips = tNCVehicle.getVehicleTrips(); + ArrayList newVehicleTrips = new ArrayList(); + + if(tNCVehicle.getId()==vehicleDebug) + logger.info("There are "+existingVehicleTrips.size()+" existing vehicle trips in vehicle ID "+tNCVehicle.getId()); + + //iterate through person list and save HashMap of other passenger pickups and dropoffs by MAZ + HashMap> pickupsByMaz = new HashMap>(); + HashMap> dropoffsByMaz = new HashMap>(); + + //save the dropoff location of the first passenger in the dropoffsByMaz array (the pickup location must be the trip origin) + ArrayList firstDropoffArray = new ArrayList(); + firstDropoffArray.add(personTrips.get(0)); + dropoffsByMaz.put(firstDestinationMaz, firstDropoffArray); + + //iterate through the rest of the person trips other than the first passenger + for(int i = 1; i < personTrips.size();++i){ + + PersonTrip personTrip = personTrips.get(i); + + int pickupMaz = personTrip.getPickupMaz(); + int dropoffMaz = personTrip.getDropoffMaz(); + + //only add pickup maz for passengers other than first passenger + if(!pickupsByMaz.containsKey(pickupMaz) ){ + ArrayList pickups = new ArrayList(); + pickups.add(personTrip); + pickupsByMaz.put(pickupMaz,pickups); + }else{ + ArrayList pickups = pickupsByMaz.get(pickupMaz); + pickups.add(personTrip); + pickupsByMaz.put(pickupMaz,pickups); + } + + if(!dropoffsByMaz.containsKey(dropoffMaz)){ + ArrayList dropoffs = new ArrayList(); + dropoffs.add(personTrip); + dropoffsByMaz.put(dropoffMaz,dropoffs); + }else{ + ArrayList dropoffs = dropoffsByMaz.get(dropoffMaz); + dropoffs.add(personTrip); + dropoffsByMaz.put(dropoffMaz,dropoffs); + } + } + + if(tNCVehicle.getId()==vehicleDebug){ + logger.info("There are "+pickupsByMaz.size()+" pickup mazs in vehicle ID "+tNCVehicle.getId()); + logger.info("There are "+dropoffsByMaz.size()+" dropoff mazs in vehicle ID "+tNCVehicle.getId()); + } + + // the list of TAZs in order from closest to furthest, that will determine tNCVehicle routing. + // any TAZ in the list with an origin or destination by a passenger will be visited. + short[] tazs = transportCostManager.getZonesWithinMaxDiversionTime(skimPeriod, firstOriginTaz, firstDestinationTaz); + + //create a new tNCVehicle trip, and populate it with information from the first passenger + ++totalVehicleTrips; + TNCVehicleTrip trip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); + trip.setStartPeriod(simulationPeriod); + trip.addPickupAtOrigin(firstTrip.getUniqueId()); + trip.setOriginMaz(firstOriginMaz); + trip.setOriginTaz((short) firstOriginTaz); + trip.setPassengers(1); + + //iterate through tazs sorted by time from first passenger's origin, and + //assign person trips to pickup and dropoff arrays based on diversion time. + for(int i=0;i pickups = pickupsByMaz.get(maz); + for(int p = 0; p< pickups.size();++p){ + PersonTrip pTrip = pickups.get(p); + trip.addPickupAtDestination(pTrip.getUniqueId()); + } + } + + //there are dropoffs in this maz + if(dropoffsByMaz.containsKey(maz)){ + ArrayList dropoffs = dropoffsByMaz.get(maz); + for(int p = 0; p< dropoffs.size();++p){ + PersonTrip pTrip = dropoffs.get(p); + trip.addDropoffAtDestination(pTrip.getUniqueId()); + + //remove this person trip from the list of persons in this tNCVehicle since they are getting dropped off. + tNCVehicle.removePersonTrip(pTrip); + + } + } + + // this is not the first tNCVehicle trip for this tNCVehicle. So we need to find the last tNCVehicle trip + // occupancy and destination pickups and dropoffs to set the trip occupancy and origin pickups & dropoffs accordingly. + int lastTripPassengers=0; + TNCVehicleTrip lastTrip = null; + if(newVehicleTrips.size()==0 && existingVehicleTrips.size()>0){ + lastTrip = existingVehicleTrips.get(existingVehicleTrips.size()-1); + }else if(newVehicleTrips.size()>0){ + lastTrip = newVehicleTrips.get(newVehicleTrips.size()-1); + } + //set the origin and other values for the trip + if(lastTrip!=null){ + lastTripPassengers = lastTrip.getPassengers(); + ArrayList dropoffsAtDestinationOfLastTrip = lastTrip.getDropoffIdsAtDestination(); + ArrayList pickupsAtDestinationOfLastTrip = lastTrip.getPickupIdsAtDestination(); + + //add pickups and dropoffs at origin from last trip + trip.addDropoffIdsAtOrigin(dropoffsAtDestinationOfLastTrip); + trip.addPickupIdsAtOrigin(pickupsAtDestinationOfLastTrip); + + //add pickup and dropoffs at origin of this trip to destination of last trip. (commenting to test write problem) + //lastTrip.addDropoffIdsAtDestination(trip.getDropoffIdsAtOrigin()); + //lastTrip.addPickupIdsAtDestination(trip.getPickupIdsAtOrigin()); + + + + } + + int passengers = lastTripPassengers + trip.getNumberOfPickupsAtOrigin() - trip.getNumberOfDropoffsAtOrigin(); + trip.setPassengers(passengers); + trip.setDestinationMaz(maz); + trip.setDestinationTaz((short) tazs[i]); + + //measure time from first trip to destination (current) or track time in tNCVehicle explicitly for each trip? + float time = transportCostManager.getTime(skimPeriod, firstOriginTaz, trip.getDestinationTaz()); + float periods = time/(float)minutesPerSimulationPeriod; + int endPeriod = (int) Math.floor(simulationPeriod + periods); //currently measuring time as simulation period + straight time to dest. + trip.setEndPeriod(endPeriod); + + //measure distance for current trip origin and destination + float distance = transportCostManager.getDistance(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz()); + trip.setDistance(distance); + tNCVehicle.setDistanceSinceRefuel(tNCVehicle.getDistanceSinceRefuel()+distance); + + if(tNCVehicle.getId()==vehicleDebug){ + logger.info("Vehicle ID "+tNCVehicle.getId()+" now has vehicle trip ID "+trip.getId()); + trip.writeTrace(); + } + + newVehicleTrips.add(trip); + + //more trips to go! + if(tNCVehicle.getPersonTripList().size()>0){ + ++totalVehicleTrips; + TNCVehicleTrip newTrip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); + newTrip.setOriginMaz(maz); + newTrip.setOriginTaz((short) tazs[i]); + newTrip.setStartPeriod(trip.getEndPeriod()); + trip = newTrip; + } + + } //end mazs + + } //end tazs + + //add tNCVehicle trips to tNCVehicle + tNCVehicle.addVehicleTrips(newVehicleTrips); + + //add tNCVehicle to active vehicles + activeVehicleList.add(tNCVehicle); + + //track tNCVehicle in vehicles to remove from route list. + vehiclesToRemove.add(tNCVehicle); + } //end vehicles + + //Remove vehicles that have been routed + vehiclesToRouteList.removeAll(vehiclesToRemove); + } + + /** + * Free vehicles from the active tNCVehicle list and put them in the free tNCVehicle list if + * the last trip in the tNCVehicle ends in the current simulation period. + * + * @param simulationPeriod + */ + public void freeVehicles(int simulationPeriod){ + + int freedVehicles=0; + + //no active vehicles in the simulation period + if(activeVehicleList.size()==0){ + logger.warn("Trying to free vehicles from active vehicle list in simulation period "+simulationPeriod+" but there are no active vehicles."); + }else{ + logger.info("There are "+activeVehicleList.size()+" active vehicles in period "+simulationPeriod); + } + + //track the vehicles to remove + ArrayList vehiclesToRemove = new ArrayList(); + // go through active vehicles (vehicles that have been routed and are picking up/dropping off passengers) + for(int i = 0; i< activeVehicleList.size();++i){ + TNCVehicle tNCVehicle = activeVehicleList.get(i); + + ArrayList trips = tNCVehicle.getVehicleTrips(); + + //this tNCVehicle has no trips; why is it in the active tNCVehicle list?? + if(trips.size()==0){ + logger.error("Vehicle ID "+tNCVehicle.getId()+" has no vehicle trips but is in active vehicle list"); + continue; + } + + //Find out when the last dropoff occurs (the end period of the last trip) + TNCVehicleTrip lastTrip = trips.get(trips.size()-1); + if(lastTrip.endPeriod==simulationPeriod){ + int taz = lastTrip.getDestinationTaz(); + vehiclesToRemove.add(tNCVehicle); + ++freedVehicles; + + //store the empty tNCVehicle in the last dropoff location (the last trip destination TAZ) + if(emptyVehicleList[taz]==null) + emptyVehicleList[taz]= new ArrayList(); + + emptyVehicleList[taz].add(tNCVehicle); + } + } + activeVehicleList.removeAll(vehiclesToRemove); + logger.info("Freed "+freedVehicles+" vehicles from active tNCVehicle list"); + logger.info("There are now "+activeVehicleList.size()+" vehicles in the active tNCVehicle list"); + } + + + /** + * First find vehicles that need to refuel, generate a trip to the closest refueling station, then + * remove them from the empty tNCVehicle list, and add them to the refueling tNCVehicle list. + * Next, for all refueling vehicles, check if they are done refueling, and if so, remove them + * from the refueling list and add them to the empty tNCVehicle list. + * + * @param skimPeriod + * @param simulationPeriod + */ + public synchronized void checkForRefuelingVehicles(int skimPeriod, int simulationPeriod) { + + //iterate through zones + for(int i = 1; i <= maxTaz; ++ i){ + if(emptyVehicleList[i]==null) + continue; + + //track the vehicles to remove + ArrayList vehiclesToRemove = new ArrayList(); + + //iterate through vehicles in this zone + for(TNCVehicle tNCVehicle : emptyVehicleList[i]) { + + //if distance since refueling is greater than max, generate a new trip to the closest refueling station. + if(tNCVehicle.getDistanceSinceRefuel()>=maxDistanceBeforeRefuel) { + + ArrayList currentTrips = tNCVehicle.getVehicleTrips(); + TNCVehicleTrip lastTrip = currentTrips.get(currentTrips.size()-1); + + TNCVehicleTrip trip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips+1); + trip.setStartPeriod(lastTrip.endPeriod); + trip.setOriginMaz(lastTrip.destinationMaz); + trip.setOriginTaz(lastTrip.originTaz); + trip.setPassengers(0); + trip.setOriginPurpose(lastTrip.destinationPurpose); + trip.setDestinationPurpose(Purpose.REFUEL); + + int refeulingMaz = closestMazWithRefeulingStation[trip.getOriginMaz()]; + trip.setDestinationMaz(refeulingMaz); + trip.setDestinationTaz((short) mazManager.getTaz(refeulingMaz)); + float time = transportCostManager.getTime(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz() ); + float distance = transportCostManager.getDistance(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz()); + float periods = time/(float)minutesPerSimulationPeriod; + int endPeriod = (int) Math.floor(simulationPeriod + periods); + trip.setEndPeriod(endPeriod); + trip.setDistance(distance); + + //add the tNCVehicle trip to the tNCVehicle + tNCVehicle.addVehicleTrip(trip); + + vehiclesToRemove.add(tNCVehicle); + + if(tNCVehicle.getId()==vehicleDebug) { + logger.info("*************"); + logger.info("Vehicle ID "+tNCVehicle.getId()+" refueling trip generated"); + logger.info("From origin MAZ "+trip.getOriginMaz()+" to refueling MAZ "+trip.getDestinationMaz()+" in TAZ "+trip.getDestinationTaz()); + logger.info("Start period "+trip.getStartPeriod()+" end period "+trip.getEndPeriod()); + logger.info("*************"); + + } + + } + + } + //remove all the refueling vehicles from the empty tNCVehicle list + emptyVehicleList[i].removeAll(vehiclesToRemove); + + //add them to the refueling tNCVehicle list + refuelingVehicleList.addAll(vehiclesToRemove); + } + + //track the vehicles to remove + ArrayList vehiclesToRemove = new ArrayList(); + + //iterate through the refueling vehicles + for(TNCVehicle tNCVehicle : refuelingVehicleList ) { + + ArrayList currentTrips = tNCVehicle.getVehicleTrips(); + TNCVehicleTrip lastTrip = currentTrips.get(currentTrips.size()-1); + + //trip is not refueling + if(lastTrip.destinationPurpose!=Purpose.REFUEL) + continue; + + //trip is still en-route to refueling + if(lastTrip.endPeriod>simulationPeriod) + continue; + + + //if its been refueling for appropriate periods, add to empty tNCVehicle list and remove it from the refueling tNCVehicle list + if(tNCVehicle.periodsRefueling==periodsRequiredForRefuel) { + tNCVehicle.setDistanceSinceRefuel(0); + vehiclesToRemove.add(tNCVehicle); + short refuelTaz = lastTrip.destinationTaz; + + if(emptyVehicleList[refuelTaz] == null) + emptyVehicleList[refuelTaz] = new ArrayList(); + + emptyVehicleList[refuelTaz].add(tNCVehicle); + + if(tNCVehicle.getId()==vehicleDebug) { + logger.info("*************"); + logger.info("Vehicle ID "+tNCVehicle.getId()+" has completed refueling in period "+simulationPeriod); + logger.info("Distance to refuel is reset to 0 and vehicle added to empty vehicle list in TAZ "+refuelTaz); + logger.info("*************"); + } + + + // else increment up the number of periods refueling + }else { + tNCVehicle.setPeriodsRefueling(tNCVehicle.getPeriodsRefueling()+1); + if(tNCVehicle.getId()==vehicleDebug) { + logger.info("*************"); + logger.info("Vehicle ID "+tNCVehicle.getId()+" is still refueling in period "+simulationPeriod); + logger.info("*************"); + } + + } + } + + refuelingVehicleList.removeAll(vehiclesToRemove); + } + + + /** + * This method writes tNCVehicle trips to the output file. + * + */ + public void writeVehicleTrips(float sampleRate){ + + logger.info("Writing tNCVehicle trips to file " + vehicleTripOutputFile); + PrintWriter printWriter = null; + try + { + printWriter = new PrintWriter(new BufferedWriter(new FileWriter(vehicleTripOutputFile))); + } catch (IOException e) + { + logger.fatal("Could not open file " + vehicleTripOutputFile + " for writing\n"); + throw new RuntimeException(); + } + + TNCVehicleTrip.printHeader(printWriter); + + //count the total empty vehicles + int totalEmptyVehicles =0; + for(int i = 1; i <= maxTaz; ++ i){ + if(emptyVehicleList[i]==null) + continue; + + totalEmptyVehicles+=emptyVehicleList[i].size(); + } + logger.info("Writing "+totalEmptyVehicles+" total vehicles to file"); + + //reset trip id;wsu + int tripid=0; + for(int i = 1; i <= maxTaz; ++ i){ + if(emptyVehicleList[i]==null) + continue; + + if(emptyVehicleList[i].size()==0) + continue; + + for(TNCVehicle tNCVehicle : emptyVehicleList[i] ){ + if(tNCVehicle.getId()==vehicleDebug) { + logger.info("Writing "+tNCVehicle.getVehicleTrips().size()+" vehicle trips for vehicle ID "+tNCVehicle.getId()); + } + + for(TNCVehicleTrip tNCVehicleTrip : tNCVehicle.getVehicleTrips()){ + + tripid++; + //reorder trip id by wsu + tNCVehicleTrip.setId(tripid); + tNCVehicleTrip.printData(printWriter); + + //save the data in the trip matrix + int startPeriod = tNCVehicleTrip.getStartPeriod(); + int skimPeriod = skimPeriodLookup[startPeriod]; + int origTaz = tNCVehicleTrip.getOriginTaz(); + int destTaz = tNCVehicleTrip.getDestinationTaz(); + int occ = Math.min(tNCVehicleTrip.getPassengers(),3); + + float existingTrips = TNCTripMatrix[skimPeriod][occ].getValueAt(origTaz,destTaz); + TNCTripMatrix[skimPeriod][occ].setValueAt(origTaz,destTaz,existingTrips + (1*(1/sampleRate))); + + + } + } + } + printWriter.close(); + } + + /** + * Get the output trip table file names from the properties file, and write + * trip tables for all modes for the given time period. + * + * @param period + * Time period, which will be used to find the period time string + * to append to each trip table matrix file + */ + public void writeTripTable(MatrixDataServerRmi ms) + { + + String directory = Util.getStringValueFromPropertyMap(propertyMap, "scenario.path"); + String matrixTypeName = Util.getStringValueFromPropertyMap(propertyMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + + for(int i =0;i< transportCostManager.NUM_PERIODS;++i) { + String fileName = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_MATRIX_PROPERTY) + "_"+transportCostManager.PERIODS[i]+".omx"; + try{ + //Delete the file if it exists + File f = new File(fileName); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName); + f.delete(); + } + + if (ms != null) + ms.writeMatrixFile(fileName, TNCTripMatrix[i], mt); + else + writeMatrixFile(fileName, TNCTripMatrix[i]); + } catch (Exception e){ + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName, e); + throw new RuntimeException(); + } + } + + } + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + private void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + /** + * Relate simulation periods to skim periods. + * + */ + public void calculateSkimPeriods(){ + + skimPeriodLookup = new byte[numberOfSimulationPeriods]; + int numberSkimPeriods = ModelStructure.SKIM_PERIOD_INDICES.length; + int[] endSkimPeriod = new int[numberSkimPeriods]; + + int lastPeriodEnd = 0; + int lastEndSkimPeriod = 0; + for(int skimPeriod = 0;skimPeriod=0;--skimPeriod){ + if(period pickupIdsAtOrigin; + protected ArrayList dropoffIdsAtOrigin; + protected ArrayList pickupIdsAtDestination; + protected ArrayList dropoffIdsAtDestination; + protected Purpose originPurpose; + protected Purpose destinationPurpose; + protected float distance; + + protected enum Purpose { HOME, PICKUP_ONLY, DROPOFF_ONLY, PICKUP_AND_DROPOFF, REFUEL } + + + public TNCVehicleTrip(TNCVehicle tNCVehicle, int id){ + + this.id=id; + this.tNCVehicle = tNCVehicle; + pickupIdsAtOrigin = new ArrayList(); + dropoffIdsAtOrigin = new ArrayList(); + pickupIdsAtDestination = new ArrayList(); + dropoffIdsAtDestination = new ArrayList(); + originPurpose=Purpose.HOME; + destinationPurpose=Purpose.HOME; + + } + + public ArrayList getPickupIdsAtOrigin() { + return pickupIdsAtOrigin; + } + + public void setPickupIdsAtOrigin(ArrayList pickupIdsAtOrigin) { + this.pickupIdsAtOrigin = pickupIdsAtOrigin; + + if(pickupIdsAtOrigin.isEmpty()) + return; + + if(originPurpose==Purpose.DROPOFF_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.PICKUP_ONLY; + } + + public void addPickupIdsAtOrigin(ArrayList pickupIdsAtOrigin) { + this.pickupIdsAtOrigin.addAll(pickupIdsAtOrigin); + + if(pickupIdsAtOrigin.isEmpty()) + return; + + if(originPurpose==Purpose.DROPOFF_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.PICKUP_ONLY; + + } + public void addPickupIdsAtDestination(ArrayList pickupIdsAtDestination) { + this.pickupIdsAtDestination.addAll(pickupIdsAtDestination); + + if(pickupIdsAtDestination.isEmpty()) + return; + + if(destinationPurpose==Purpose.DROPOFF_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.PICKUP_ONLY; + + } + public ArrayList getDropoffIdsAtOrigin() { + return dropoffIdsAtOrigin; + } + + public void setDropoffIdsAtOrigin(ArrayList dropoffIdsAtOrigin) { + this.dropoffIdsAtOrigin = dropoffIdsAtOrigin; + + if(dropoffIdsAtOrigin.isEmpty()) + return; + + if(originPurpose==Purpose.PICKUP_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.DROPOFF_ONLY; + + } + + public void addDropoffIdsAtOrigin(ArrayList dropoffIdsAtOrigin) { + this.dropoffIdsAtOrigin.addAll(dropoffIdsAtOrigin); + + if(dropoffIdsAtOrigin.isEmpty()) + return; + + if(originPurpose==Purpose.PICKUP_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.DROPOFF_ONLY; + + } + public void addDropoffIdsAtDestination(ArrayList dropoffIdsAtDestination) { + this.dropoffIdsAtDestination.addAll(dropoffIdsAtDestination); + + if(dropoffIdsAtDestination.isEmpty()) + return; + + if(destinationPurpose==Purpose.PICKUP_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.DROPOFF_ONLY; + + } + public ArrayList getPickupIdsAtDestination() { + return pickupIdsAtDestination; + } + + public void setPickupIdsAtDestination(ArrayList pickupIdsAtDestination) { + this.pickupIdsAtDestination = pickupIdsAtDestination; + + if(pickupIdsAtDestination.isEmpty()) + return; + + if(destinationPurpose==Purpose.DROPOFF_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.PICKUP_ONLY; + + } + + public ArrayList getDropoffIdsAtDestination() { + return dropoffIdsAtDestination; + } + + public void setDropoffIdsAtDestination( + ArrayList dropoffIdsAtDestination) { + this.dropoffIdsAtDestination = dropoffIdsAtDestination; + + if(dropoffIdsAtDestination.isEmpty()) + return; + + if(destinationPurpose==Purpose.PICKUP_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.DROPOFF_ONLY; + + } + + + + public void addPickupAtOrigin(String id){ + pickupIdsAtOrigin.add(id); + + if(originPurpose==Purpose.DROPOFF_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.PICKUP_ONLY; + } + + public void addPickupAtDestination(String id){ + pickupIdsAtDestination.add(id); + + if(destinationPurpose==Purpose.DROPOFF_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.PICKUP_ONLY; + } + + public void addDropoffAtOrigin(String id){ + dropoffIdsAtOrigin.add(id); + + if(originPurpose==Purpose.PICKUP_ONLY) + originPurpose = Purpose.PICKUP_AND_DROPOFF; + else + originPurpose = Purpose.DROPOFF_ONLY; + + + } + + public void addDropoffAtDestination(String id){ + dropoffIdsAtDestination.add(id); + + if(destinationPurpose==Purpose.PICKUP_ONLY) + destinationPurpose = Purpose.PICKUP_AND_DROPOFF; + else + destinationPurpose = Purpose.DROPOFF_ONLY; +} + + public int getNumberOfPickupsAtOrigin(){ + return pickupIdsAtOrigin.size(); + } + + public int getNumberOfDropoffsAtOrigin(){ + return dropoffIdsAtOrigin.size(); + } + + public int getNumberOfPickupsAtDestination(){ + return pickupIdsAtDestination.size(); + } + + public int getNumberOfDropoffsAtDestination(){ + return dropoffIdsAtDestination.size(); + } + + public TNCVehicle getVehicle() { + return tNCVehicle; + } + + + public void setVehicle(TNCVehicle tNCVehicle) { + this.tNCVehicle = tNCVehicle; + } + + + public short getOriginTaz() { + return originTaz; + } + + + public void setOriginTaz(short originTaz) { + this.originTaz = originTaz; + } + + + public short getDestinationTaz() { + return destinationTaz; + } + + + public void setDestinationTaz(short destinationTaz) { + this.destinationTaz = destinationTaz; + } + + + public int getOriginMaz() { + return originMaz; + } + + + public void setOriginMaz(int originMaz) { + this.originMaz = originMaz; + } + + + public int getDestinationMaz() { + return destinationMaz; + } + + + public void setDestinationMaz(int destinationMaz) { + this.destinationMaz = destinationMaz; + } + + + public int getPassengers() { + return passengers; + } + + + public void setPassengers(int passengers) { + this.passengers = passengers; + } + + public int getStartPeriod() { + return startPeriod; + } + + public void setStartPeriod(int startPeriod) { + this.startPeriod = startPeriod; + } + + public int getEndPeriod() { + return endPeriod; + } + + public void setEndPeriod(int endPeriod) { + this.endPeriod = endPeriod; + } + + public int getId() { + return id; + } + + public void setId(int id) { + this.id = id; + } + + public static void printHeader(PrintWriter writer){ + String record = new String("trip_ID,vehicle_ID,originTaz,destinationTaz,originMgra,destinationMgra,totalPassengers,startPeriod,endPeriod,pickupIdsAtOrigin,dropoffIdsAtOrigin,pickupIdsAtDestination,dropoffIdsAtDestination, originPurpose, destinationPurpose"); + writer.println(record); + writer.flush(); + } + + public void printData(PrintWriter writer){ + + String pickupIdsAtOriginString = ""; + String dropoffIdsAtOriginString = ""; + String pickupIdsAtDestinationString = ""; + String dropoffIdsAtDestinationString = ""; + + if(pickupIdsAtOrigin.size()>0) + for(String pid : pickupIdsAtOrigin) + pickupIdsAtOriginString += (pid + " "); + + if(dropoffIdsAtOrigin.size()>0) + for(String pid : dropoffIdsAtOrigin) + dropoffIdsAtOriginString += (pid + " "); + + if(pickupIdsAtDestination.size()>0) + for(String pid : pickupIdsAtDestination) + pickupIdsAtDestinationString += (pid + " "); + + if(dropoffIdsAtDestination.size()>0) + for(String pid : dropoffIdsAtDestination) + dropoffIdsAtDestinationString += (pid + " "); + + String record = new String( + id + "," + + tNCVehicle.getId() +"," + + originTaz + "," + + destinationTaz + "," + + originMaz + "," + + destinationMaz + "," + + passengers + "," + + startPeriod + "," + + endPeriod + "," + + pickupIdsAtOriginString + "," + + dropoffIdsAtOriginString + "," + + pickupIdsAtDestinationString + "," + + dropoffIdsAtDestinationString + "," + + originPurpose.ordinal() + "," + + destinationPurpose.ordinal()); + + writer.println(record); + writer.flush(); + } + + public void writeTrace(){ + + String pickupIdsAtOriginString = ""; + String dropoffIdsAtOriginString = ""; + String pickupIdsAtDestinationString = ""; + String dropoffIdsAtDestinationString = ""; + + if(pickupIdsAtOrigin.size()>0) + for(String pid : pickupIdsAtOrigin) + pickupIdsAtOriginString += (pid + " "); + + if(dropoffIdsAtOrigin.size()>0) + for(String pid : dropoffIdsAtOrigin) + dropoffIdsAtOriginString += (pid + " "); + + if(pickupIdsAtDestination.size()>0) + for(String pid : pickupIdsAtDestination) + pickupIdsAtDestinationString += (pid + " "); + + if(dropoffIdsAtDestination.size()>0) + for(String pid : dropoffIdsAtDestination) + dropoffIdsAtDestinationString += (pid + " "); + + logger.info("*********************************************************"); + logger.info("Trace for tNCVehicle trip "+id+" in tNCVehicle "+tNCVehicle.getId()); + logger.info("Trip ID: " + id); + logger.info("TNCVehicle ID: "+tNCVehicle.getId()); + logger.info("Origin TAZ: "+originTaz); + logger.info("Destination TAZ: "+destinationTaz); + logger.info("Origin MAZ: "+originMaz); + logger.info("Destination MAZ: "+destinationMaz); + logger.info("Passengers: "+passengers); + logger.info("Start period: "+startPeriod); + logger.info("End period: "+endPeriod); + logger.info("Pickups at Origin: "+ pickupIdsAtOriginString); + logger.info("Dropoffs at Origin: "+ dropoffIdsAtOriginString); + logger.info("Pickups at Destination: "+ pickupIdsAtDestinationString); + logger.info("Dropoffs at Destination: "+ dropoffIdsAtDestinationString); + logger.info("Origin Purpose: "+ originPurpose); + logger.info("Destination Purpose: "+ destinationPurpose); + + logger.info("*********************************************************"); + + + } + + public Purpose getOriginPurpose() { + return originPurpose; + } + + public void setOriginPurpose(Purpose originPurpose) { + this.originPurpose = originPurpose; + } + + public Purpose getDestinationPurpose() { + return destinationPurpose; + } + + public void setDestinationPurpose(Purpose destinationPurpose) { + this.destinationPurpose = destinationPurpose; + } + + + public float getDistance() { + return distance; + } + + public void setDistance(float distance) { + this.distance = distance; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java new file mode 100644 index 0000000..e0f64db --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java @@ -0,0 +1,438 @@ +package org.sandag.abm.maas; + +import java.io.File; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.concurrent.Executors; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +import drasys.or.util.Array; + +public class TransportCostManager { + + protected transient Logger logger = Logger.getLogger(TransportCostManager.class); + + protected static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; + protected static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; + protected static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; + protected static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; + protected static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; + public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; + protected static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; + + protected static int TAZ_CALCULATOR_THREADS = 20; //default + + //by period, origin, destination - ragged array of zone numbers of zones within max time diversion + //sorted by time from origin (assuming pickups would be en-route) + protected short[][][][] tazsWithinOriginAndDestination; + // private float[][][][] addTimeWithinOriginAndDestination; + + //by period, origin, destination + protected float[][][] tazTimeSkims; //travel time + protected float[][][] tazDistanceSkims; //travel distance + + protected short[][][] tazsByTimeFromOrigin; //array of TAZs sorted by time from origin, by period and origin TAZ + + protected float maxTimeDiversion; + protected float maxDistanceToPickup; + protected int maxTaz; + + // declare an array of UEC objects, 1 for each time period + protected UtilityExpressionCalculator[] autoDistOD_UECs; + protected UtilityExpressionCalculator[] autoTimeOD_UECs; + + // The simple auto skims UEC does not use any DMU variables + protected VariableTable dmu = null; + protected TazDataManager tazManager; + int totalThreads; + + + /** + * Instantiate transport cost manager. + * + * @param rbMap + * @param maxTimeDiversion + */ + public TransportCostManager(HashMap rbMap, float maxTimeDiversion, float maxDistanceToPickup) + { + + this.maxTimeDiversion=maxTimeDiversion; + this.maxDistanceToPickup=maxDistanceToPickup; + + // Create the UECs + String uecPath = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String uecFileName = uecPath + + Util.getStringValueFromPropertyMap(rbMap, "taz.distance.uec.file"); + int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "taz.distance.data.page"); + + + //iterate thru settings in properties file and create time and distance UECs + autoDistOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; + autoTimeOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; + File uecFile = new File(uecFileName); + + for(int i =0; i stopTazList = new ArrayList(); + + for (int oTaz = startOriginTaz; oTaz <= endOriginTaz; oTaz++){ + + if((oTaz==startOriginTaz)||(oTaz % 100 == 0)) + logger.info("Thread "+threadName + " Period "+period+" Origin TAZ "+oTaz); + + for (int dTaz = 1; dTaz <= maxTaz; dTaz++){ + + stopTazList.clear(); + + //Stop TAZs + for(int kTaz = 1; kTaz <= maxTaz; ++kTaz){ + + //Calculate additional time to stop + float ikTime = tazTimeSkims[period][oTaz][kTaz]; + float kjTime = tazTimeSkims[period][kTaz][dTaz]; + float totalIKJTime = ikTime + kjTime; + float divertTime = totalIKJTime - tazTimeSkims[period][oTaz][dTaz]; + + //if time is less than max diversion time (or the stop zone is the origin or destination zone), add zone and time to arraylist + if( (divertTime < maxTimeDiversion) || (kTaz==oTaz) || (kTaz==dTaz)){ + StopTaz stopTaz = new StopTaz(); + stopTaz.tazNumber = kTaz; + stopTaz.diversionTime = divertTime; + stopTaz.originStopTime = ikTime; + stopTazList.add(stopTaz); + } + + } //end for stops + + //initialize arrays for saving tazs, time and set the values in the ragged arrays + if(!stopTazList.isEmpty()){ + Collections.sort(stopTazList); + int numberOfStops = stopTazList.size(); + tazsWithinOriginAndDestination[period][oTaz][dTaz] = new short[numberOfStops]; + //addTimeWithinOriginAndDestination[period][oTaz][dTaz] = new float[numberOfStops]; + + for(int k = 0; k < numberOfStops; ++k){ + StopTaz stopTaz = stopTazList.get(k); + tazsWithinOriginAndDestination[period][oTaz][dTaz][k] = (short) stopTaz.tazNumber; + //addTimeWithinOriginAndDestination[period][oTaz][dTaz][k] = stopTaz.diversionTime; + } + } + } + + } + + + + + } + } + + /** + * This method finds stop zones for each origin-destination zone pair and saves the zone number + * and diversion time, sorted by distance from origin. + * + */ + private void calculateTazsWithinDistanceThreshold(){ + + + tazsWithinOriginAndDestination = new short[NUM_PERIODS][maxTaz+1][maxTaz+1][]; + //addTimeWithinOriginAndDestination = new float[NUM_PERIODS][maxTaz+1][maxTaz+1][]; + int processors = Runtime.getRuntime().availableProcessors(); + //use 80% of the machine's processing power + TAZ_CALCULATOR_THREADS = totalThreads; + int chunkSize = (int) Math.floor(maxTaz / TAZ_CALCULATOR_THREADS); + + logger.info("...Calculating TAZs within distance thresholds with "+TAZ_CALCULATOR_THREADS+ " threads ("+processors+" processors)"); + + for( int period = 0; period < NUM_PERIODS;++period ){ + + int endZone = 0; + + ThreadPoolExecutor executor = (ThreadPoolExecutor) Executors.newFixedThreadPool(TAZ_CALCULATOR_THREADS); + + for(int i = 0; i < TAZ_CALCULATOR_THREADS; ++ i){ + + int startZone = endZone + 1; + + if(i==(TAZ_CALCULATOR_THREADS-1)) + endZone = maxTaz; + else + endZone = startZone+chunkSize; + + executor.execute(new TazDistanceCalculatorThread( "Thread-"+i,period,startZone,endZone)); + + } + executor.shutdown(); + try{ + executor.awaitTermination(60, TimeUnit.MINUTES); + }catch(InterruptedException e){ + throw new RuntimeException(e); + } + } + } + + /** + * Calculate zones sorted by time from origin. Always include intrazonal as within the maximum distance range. + * + */ + public void calculateTazsByTimeFromOrigin(){ + + ArrayList stopTazList = new ArrayList(); + + tazsByTimeFromOrigin = new short[NUM_PERIODS][maxTaz+1][]; + + for(int period = 0; period that.originStopTime) return AFTER; + + return EQUAL; + } + + + } + + /** + * Get the array of zones that are within the diversion time from the origin to the + * destination, sorted by time from origin. + * + * @param skimPeriod + * @param origTaz + * @param destTaz + * @return The array of zones, or null if there are no zones within the max diversion time. + */ + public short[] getZonesWithinMaxDiversionTime(int skimPeriod, int origTaz, int destTaz){ + + return tazsWithinOriginAndDestination[skimPeriod][origTaz][destTaz]; + + } + + /** + * Is the zone within the set of zones that is within maximum diversion time from the origin to the destination? + * + * @param skimPeriod + * @param origTaz The origin TAZ + * @param destTaz The destination TAZ + * @param taz The stop TAZ + * @return A boolean indicating whether the zone is within the maximum deviation time from the origin to the destination. + */ + public boolean stopZoneIsWithinMaxDiversionTime(int skimPeriod, int origTaz, int destTaz, int taz){ + + short[] tazArray = getZonesWithinMaxDiversionTime(skimPeriod, origTaz, destTaz); + for(int i = 0; i < tazArray.length; ++i) + if(tazArray[i]==taz) + return true; + return false; + + } + + + + /** + * Get the diversion times for the zones that are within the diversion time from the origin to the + * destination, sorted by time from origin. + * + * @param period + * @param origTaz + * @param destTaz + * @return The array of diversion times, or null if there are no zones within the max diversion time. + */ + public float[] getDiversionTimes(int period, int origTaz, int destTaz){ + + //return addTimeWithinOriginAndDestination[period][origTaz][destTaz]; + logger.fatal("Error trying to call getDiversionTimes when additional time array not initialized"); + throw new RuntimeException(); + } + + /** + * Get a ragged array of zone numbers sorted by time from the origin. The array is ragged + * because it is capped by the maximum distance for hailing a TNC\TAXI. + * + * @param period + * @param origTaz + * @return A sorted array of zone numbers, or null if there are no zones within the maximum distance. + */ + public short[] getZoneNumbersSortedByTime(int period, int origTaz){ + + return tazsByTimeFromOrigin[period][origTaz]; + } + + public float getTime(int period, int origTaz, int destTaz){ + + return tazTimeSkims[period][origTaz][destTaz]; + } + + public float getDistance(int period, int origTaz, int destTaz){ + + return tazDistanceSkims[period][origTaz][destTaz]; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java new file mode 100644 index 0000000..59823a0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java @@ -0,0 +1,136 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Mar 9, 2009 + *

+ * Created by IntelliJ IDEA. + */ +public class AutoDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(AutoDMU.class); + + protected HashMap methodIndexMap; + + private double avgHourlyParkingCostAtDestTaz; + private float pTazTerminalTime; + private float aTazTerminalTime; + + public AutoDMU() + { + setupMethodIndexMap(); + } + + public double getAvgHourlyParkingCostAtDestTaz() + { + return avgHourlyParkingCostAtDestTaz; + } + + public void setAvgHourlyParkingCostAtDestTaz(double cost) + { + avgHourlyParkingCostAtDestTaz = cost; + } + + public float getPTazTerminalTime() + { + return pTazTerminalTime; + } + + public void setPTazTerminalTime(float pTazTerminalTime) + { + this.pTazTerminalTime = pTazTerminalTime; + } + + public float getATazTerminalTime() + { + return aTazTerminalTime; + } + + public void setATazTerminalTime(float aTazTerminalTime) + { + this.aTazTerminalTime = aTazTerminalTime; + } + + /** + * Log the DMU values. + * + * @param localLogger + * The logger to use. + */ + public void logValues(Logger localLogger) + { + + localLogger.info(""); + localLogger.info("Auto DMU Values:"); + localLogger.info(""); + localLogger.info(String.format("Average TAZ Parking cost at destination: %9f", + avgHourlyParkingCostAtDestTaz)); + localLogger.info(String.format("Production/Origin Terminal Time: %9.4f", pTazTerminalTime)); + localLogger.info(String.format("Attraction/Destin Terminal Time: %9.4f", aTazTerminalTime)); + + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getAvgHourlyParkingCostAtDestTaz", 0); + methodIndexMap.put("getATazTerminalTime", 1); + methodIndexMap.put("getPTazTerminalTime", 2); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getAvgHourlyParkingCostAtDestTaz(); + case 1: + return getATazTerminalTime(); + case 2: + return getPTazTerminalTime(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java new file mode 100644 index 0000000..2481944 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java @@ -0,0 +1,138 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.LogitModel; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Mar 9, 2009 + *

+ * Created by IntelliJ IDEA. + */ +public class AutoUEC + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(AutoUEC.class); + private TazDataManager tazs; + private UtilityExpressionCalculator uec; + private LogitModel model; + private ChoiceModelApplication modelApp; + + private IndexValues index = new IndexValues(); + private int[] availFlag; + private AutoDMU dmu; + + // seek and trace + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + protected Tracer tracer; + + /** + * Constructor. + * + * @param rb + * ResourceBundle + * @param UECFileName + * The path/name of the UEC containing the auto model. + * @param modelSheet + * The sheet (0-indexed) containing the model specification. + * @param dataSheet + * The sheet (0-indexed) containing the data specification. + */ + public AutoUEC(HashMap rbHashMap, String uecFileName, int modelSheet, + int dataSheet) + { + + dmu = new AutoDMU(); + + // use the choice model application to set up the model structure + modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); + + // but return the logit model itself, so we can use compound utilities + model = modelApp.getRootLogitModel(); + uec = modelApp.getUEC(); + + tazs = TazDataManager.getInstance(); + trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + } + + /** + * Solve auto utilities for a given zone-pair + * + * @param pTaz + * Production/Origin TAZ. + * @param aTaz + * Attraction/Destination TAZ. + * @return The root utility. + */ + public double calculateUtilitiesForTazPair(int pTaz, int aTaz, double avgTazHourlyParkingCost) + { + + trace = false; + if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) + { + trace = true; + } + index.setOriginZone(pTaz); + index.setDestZone(aTaz); + availFlag = new int[uec.getNumberOfAlternatives() + 1]; + Arrays.fill(availFlag, 1); + + dmu.setAvgHourlyParkingCostAtDestTaz(avgTazHourlyParkingCost); + dmu.setPTazTerminalTime(tazs.getOriginTazTerminalTime(pTaz)); + dmu.setATazTerminalTime(tazs.getDestinationTazTerminalTime(aTaz)); + + // log DMU values + if (trace) + { + TapDataManager tapManager = TapDataManager.getInstance(); + if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 + && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) + uec.logDataValues(logger, pTaz, aTaz, aTaz); + dmu.logValues(logger); + } + + modelApp.computeUtilities(dmu, index); + double utility = modelApp.getLogsum(); + + // logging + if (trace) + { + uec.logAnswersArray(logger, "Auto UEC"); + uec.logResultsArray(logger, pTaz, aTaz); + modelApp.logLogitCalculations("Auto UEC", "Trace"); + logger.info("Logsum = " + utility); + trace = false; + } + + return utility; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java new file mode 100644 index 0000000..46c933d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java @@ -0,0 +1,56 @@ +package org.sandag.abm.modechoice; + +/** + * This class is used for storing constants. Many of these are listed in the + * sandag.inc file associated with the FORTRAN code. We should eventually move + * these into a properties file and have this class set them from the prop file. + * + * I am just trying to not get bogged down in the details. + * + * @author Christi Willison + * @version Nov 6, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public final class Constants +{ + + public static int MAX_EXTERNAL = 12; + public static float AutoCostPerMile = 10.0f; + + public static float[][] parkingCost = { {0.0f, 50.0f, 200.0f, 300.0f, 400.0f}, + {0.0f, 50.0f, 125.0f, 200.0f, 400.0f}, {0.0f, 50.0f, 100.0f, 200.0f, 400.0f}}; + + public static float walkMinutesPerMile = 20.0f; // 20 + // minutes + // per + // mile + // (dist + // is + // in + // feet) + // or + // 3 + // mph. + public static float bikeMinutesPerMile = 5.0f; // 5 + + // minutes + // per + // mile + // (dist + // is + // in + // feet) + // or + // 12 + // mph. + + public static float feetPerMile = 5280.0f; + public static double walkMinutesPerFoot = walkMinutesPerMile/feetPerMile; + public static double bikeMinutesPerFoot = bikeMinutesPerMile/feetPerMile; + + private Constants() + { + // Not Implemented + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java new file mode 100644 index 0000000..52cd382 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java @@ -0,0 +1,128 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +/** + * This class is the DMU object for MAAS + * joel freedman + * RSG 2019-07-08 + **/ + +public class MaasDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(MaasDMU.class); + + protected HashMap methodIndexMap; + + protected float waitTimeTaxi; + protected float waitTimeSingleTNC; + protected float waitTimeSharedTNC; + + public MaasDMU() + { + setupMethodIndexMap(); + } + + public float getWaitTimeTaxi() { + return waitTimeTaxi; + } + + public void setWaitTimeTaxi(float waitTimeTaxi) { + this.waitTimeTaxi = waitTimeTaxi; + } + + public float getWaitTimeSingleTNC() { + return waitTimeSingleTNC; + } + + public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { + this.waitTimeSingleTNC = waitTimeSingleTNC; + } + + public float getWaitTimeSharedTNC() { + return waitTimeSharedTNC; + } + + public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { + this.waitTimeSharedTNC = waitTimeSharedTNC; + } + + + /** + * Log the DMU values. + * + * @param localLogger + * The logger to use. + */ + public void logValues(Logger localLogger) + { + + localLogger.info(""); + localLogger.info("Maas DMU Values:"); + localLogger.info(""); + localLogger.info(String.format("Taxi wait time: %9.2f", waitTimeTaxi)); + localLogger.info(String.format("Single TNC wait time: %9.2f", waitTimeSingleTNC)); + localLogger.info(String.format("Shared TNC wait time: %9.2f", waitTimeSharedTNC)); + + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getWaitTimeTaxi", 0); + methodIndexMap.put("getWaitTimeSingleTNC", 1); + methodIndexMap.put("getWaitTimeSharedTNC", 2); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getWaitTimeTaxi(); + case 1: + return getWaitTimeSingleTNC(); + case 2: + return getWaitTimeSharedTNC(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java new file mode 100644 index 0000000..682f7b8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java @@ -0,0 +1,141 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; +import org.sandag.abm.ctramp.Util; +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.LogitModel; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +/** + * This class is the UEC for MAAS + * Joel Freedman + * RSG 2019-07-08 + */ +public class MaasUEC + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(MaasUEC.class); + private TazDataManager tazs; + private UtilityExpressionCalculator uec; + private LogitModel model; + private ChoiceModelApplication modelApp; + + private IndexValues index = new IndexValues(); + private int[] availFlag; + private MaasDMU dmu; + + // seek and trace + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + protected Tracer tracer; + + + + + /** + * Constructor. + * + * @param rb + * ResourceBundle + * @param UECFileName + * The path/name of the UEC containing the auto model. + * @param modelSheet + * The sheet (0-indexed) containing the model specification. + * @param dataSheet + * The sheet (0-indexed) containing the data specification. + */ + public MaasUEC(HashMap rbHashMap, String uecFileName, int modelSheet, + int dataSheet) + { + + dmu = new MaasDMU(); + + // use the choice model application to set up the model structure + modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); + + // but return the logit model itself, so we can use compound utilities + model = modelApp.getRootLogitModel(); + uec = modelApp.getUEC(); + + tazs = TazDataManager.getInstance(); + trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + if (trace) + { + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + } + } + } + + + } + + /** + * Solve auto utilities for a given zone-pair + * + * @param pTaz + * Production/Origin TAZ. + * @param aTaz + * Attraction/Destination TAZ. + * @return The root utility. + */ + public double calculateUtilitiesForTazPair(int pTaz, int aTaz, float avgTaxiWaitTime, float avgSingleTNCWaitTime,float avgSharedTNCWaitTime) + { + + trace = false; + if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) + { + trace = true; + } + index.setOriginZone(pTaz); + index.setDestZone(aTaz); + availFlag = new int[uec.getNumberOfAlternatives() + 1]; + Arrays.fill(availFlag, 1); + + dmu.setWaitTimeTaxi(avgTaxiWaitTime); + dmu.setWaitTimeSingleTNC(avgSingleTNCWaitTime); + dmu.setWaitTimeSharedTNC(avgSharedTNCWaitTime); + + // log DMU values + if (trace) + { + TapDataManager tapManager = TapDataManager.getInstance(); + if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 + && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) + uec.logDataValues(logger, pTaz, aTaz, aTaz); + dmu.logValues(logger); + } + + modelApp.computeUtilities(dmu, index); + double utility = modelApp.getLogsum(); + + // logging + if (trace) + { + uec.logAnswersArray(logger, "Maas UEC"); + uec.logResultsArray(logger, pTaz, aTaz); + modelApp.logLogitCalculations("Maas UEC", "Trace"); + logger.info("Logsum = " + utility); + trace = false; + } + + return utility; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java new file mode 100644 index 0000000..06a2bee --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java @@ -0,0 +1,1551 @@ +package org.sandag.abm.modechoice; + + +import org.sandag.abm.active.sandag.SandagWalkPathAlternativeListGenerationConfiguration; +import org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication; +import org.sandag.abm.ctramp.BikeLogsum; +import org.sandag.abm.ctramp.BikeLogsumSegment; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.Modes.AccessMode; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.io.Serializable; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import java.util.ResourceBundle; +import java.util.Set; +import java.util.StringTokenizer; +import java.util.TreeSet; + +import org.apache.log4j.Logger; + +import com.pb.common.datafile.CSVFileReader; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Sep 4, 2008 + *

+ * Created by IntelliJ IDEA. + * + * Edited JEF May 2009 + */ +public final class MgraDataManager + implements Serializable +{ + + private static MgraDataManager instance; + protected transient Logger logger = Logger.getLogger(MgraDataManager.class); + + public static final double MAX_PARKING_WALK_DISTANCE = 0.75; + + private static final int LOG_MGRA = -4502; + private static final String LOG_MGRA_FILE = LOG_MGRA + + "debug"; + private static final String MGRA_MGRA_WALK_FILE_PROPERTY = "active.micromobility.file.walk.mgra"; + private static final String MGRA_TAP_WALK_FILE_PROPERTY = "active.micromobility.file.walk.mgratap"; + + + // create Strubg variables for the 4D land use data file field names + public static final String MGRA_4DDENSITY_DU_DEN_FIELD = "DUDen"; + public static final String MGRA_4DDENSITY_EMP_DEN_FIELD = "EmpDen"; + public static final String MGRA_4DDENSITY_TOT_INT_FIELD = "TotInt"; + + // public static final String MGRA_FIELD_NAME = "MGRASR10"; + public static final String MGRA_FIELD_NAME = "mgra"; + public static final String MGRA_TAZ_FIELD_NAME = "TAZ"; + public static final String MGRA_LUZ_FIELD_NAME = "luz_id"; + private static final String MGRA_POPULATION_FIELD_NAME = "pop"; + private static final String MGRA_HOUSEHOLDS_FIELD_NAME = "hh"; + private static final String MGRA_GRADE_SCHOOL_ENROLLMENT_FIELD_NAME = "EnrollGradeKto8"; + private static final String MGRA_HIGH_SCHOOL_ENROLLMENT_FIELD_NAME = "EnrollGrade9to12"; + private static final String MGRA_UNIVERSITY_ENROLLMENT_FIELD_NAME = "collegeEnroll"; + private static final String MGRA_OTHER_COLLEGE_ENROLLMENT_FIELD_NAME = "otherCollegeEnroll"; + private static final String MGRA_ADULT_SCHOOL_ENROLLMENT_FIELD_NAME = "AdultSchEnrl"; + private static final String MGRA_GRADE_SCHOOL_DISTRICT_FIELD_NAME = "ech_dist"; + private static final String MGRA_HIGH_SCHOOL_DISTRICT_FIELD_NAME = "hch_dist"; + private static final String MGRA_REFUELING_STATIONS_FIELD_NAME = "refueling_stations"; + private static final String MGRA_REMOTE_PARKING_LOT_FIELD_NAME = "remoteAVParking"; + + private static final String PROPERTIES_PARKING_COST_OUTPUT_FILE = "mgra.avg.cost.output.file"; + + + public static final String PROPERTIES_MGRA_DATA_FILE = "mgra.socec.file"; + private static final String MGRA_DISTANCE_COEFF_WORK = "mgra.avg.cost.dist.coeff.work"; + private static final String MGRA_DISTANCE_COEFF_OTHER = "mgra.avg.cost.dist.coeff.other"; + + public static final int PARK_AREA_ONE = 1; + private static final String MGRA_PARKAREA_FIELD = "parkarea"; + private static final String MGRA_HSTALLSOTH_FIELD = "hstallsoth"; + private static final String MGRA_HSTALLSSAM_FIELD = "hstallssam"; + private static final String MGRA_HPARKCOST_FIELD = "hparkcost"; + private static final String MGRA_NUMFREEHRS_FIELD = "numfreehrs"; + private static final String MGRA_DSTALLSOTH_FIELD = "dstallsoth"; + private static final String MGRA_DSTALLSSAM_FIELD = "dstallssam"; + private static final String MGRA_DPARKCOST_FIELD = "dparkcost"; + private static final String MGRA_MSTALLSOTH_FIELD = "mstallsoth"; + private static final String MGRA_MSTALLSSAM_FIELD = "mstallssam"; + private static final String MGRA_MPARKCOST_FIELD = "mparkcost"; + + //for TNC and Taxi wait time calculations + private static final String MGRA_POPEMPPERSQMI_FIELD = "PopEmpDenPerMi"; + private ArrayList mgras = new ArrayList(); + private int maxMgra; + private int maxLuz; + + private int maxTap; + private int nMgrasWithWlkTaps; + // [mgra], [0=tapID, 1=Distance], [tap number (0-number of taps)] + private int[][][] mgraWlkTapsDistArray; + private int[] mgraTaz; + private int[] mgraLuz; + + // An array of Hashmaps dimensioned by origin mgra, with distance in feet, + // in a ragged + // array (no key for mgra means no other mgras in walk distance) + private HashMap[] oMgraWalkDistance; + + // An array of Hashmaps dimensioned by destination mgra, with distance in + // feet, in a ragged + // array (no key for mgra means no other mgras in walk distance) + private HashMap[] dMgraWalkDistance; + + private BikeLogsum bls; + //segment doesn't matter as it is now just a passthrough + private BikeLogsumSegment defaultSegment = new BikeLogsumSegment(true,true,true); + + // An array dimensioned to maxMgra of ragged arrays of lists of TAPs + // accessible by driving + private Set[] driveAccessibleTaps; + private Set[] walkAccessibleTaps; + + //by TAP, closest mgra to the tap by walking distance. + private int[] closestMgraToTap; + + private TableDataSet mgraTableDataSet; + + private HashMap mgraDataTableMgraRowMap; + + private double[] duDen; + private double[] empDen; + private double[] totInt; + private double[] popEmpDenPerSqMi; + + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + + private int[] mgraParkArea; + private int[] numfreehrs; + private int[] hstallsoth; + private int[] hstallssam; + private float[] hparkcost; + private int[] dstallsoth; + private int[] dstallssam; + private float[] dparkcost; + private int[] mstallsoth; + private int[] mstallssam; + private float[] mparkcost; + + private TableDataSet tapLinesTable; + private HashMap taplines; + + /** + * Constructor. + * + * @param rbMap + * A HashMap created from a resourcebundle with model properties. + * + */ + private MgraDataManager(HashMap rbMap) + { + System.out.println("I'm the MgraDataManager"); + readMgraTableData(rbMap); + readMgraWlkTaps(rbMap); + readMgraWlkDist(rbMap); + + readTapLines(rbMap); + trimTapSet(); + + + + bls = BikeLogsum.getBikeLogsum(rbMap); + + // pre-process the list of TAPS reachable by drive access for each MGRA + mapDriveAccessTapsToMgras(TazDataManager.getInstance(rbMap)); + + // create arrays from 4ddensity fields added to MGRA table used by + // TourModeChoice DMU methods + process4ddensityData(rbMap); + + calculateMgraAvgParkingCosts(rbMap); + + calculateClosestMgraToTap(); + + printMgraStats(); + } + + /** + * Find the closest mgra by walk time to the tap. + */ + public void calculateClosestMgraToTap() { + + closestMgraToTap = new int[maxTap+1]; + float[] minTimeToTap = new float[maxTap+1]; + Arrays.fill(minTimeToTap, 999999); + + for(int mgra = 1; mgra taps = walkAccessibleTaps[mgra]; + for(int tap : taps) { + int pos = getTapPosition(mgra, tap); + float time = getMgraToTapWalkTime(mgra,pos); + if(time rbMap) + { + if (instance == null) + { + instance = new MgraDataManager(rbMap); + return instance; + } else return instance; + } + + /** + * This method should only be used after the getInstance(ResourceBundle rb) + * method has been called since the rb is needed to read in all the data and + * populate the object. This method will return the instance that has + * already been populated. + * + * @return instance + * @throws RuntimeException + */ + public static MgraDataManager getInstance() + { + if (instance == null) + { + throw new RuntimeException( + "Must instantiate MgraDataManager with the getInstance(rb) method first"); + } else + { + return instance; + } + } + + /** + * Read the walk-transit taps for mgras. + * + * @param rb + * The resourcebundle with the scenario.path and + * mgra.wlkacc.taps.and.distance.file properties. + */ + public void readMgraWlkTaps(HashMap rbMap) + { + String mgraWlkTapTimeFile = rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT) + +rbMap.get(MGRA_TAP_WALK_FILE_PROPERTY); + + TableDataSet mgraTapData = Util.readTableDataSet(mgraWlkTapTimeFile); + + Map> mgraWlkTapList = new HashMap<>(); //mgra -> tap -> distance + + //mgra,tap,walkTime,dist,mmTime,mmCost,mtTime,mtCost,mmGenTime,mtGenTime,minTime + for(int row = 1; row <= mgraTapData.getRowCount();++row) { + + int mgra = (int) mgraTapData.getValueAt(row, "mgra"); + int tap = (int) mgraTapData.getValueAt(row, "tap"); + if (tap > maxTap) maxTap = tap; + float minTime = mgraTapData.getValueAt(row,"minTime"); + + int distance = Math.round(minTime / Constants.walkMinutesPerMile * Constants.feetPerMile); + + //reset 0 distances to 0.1 miles, and log potential error + if(distance==0){ + //logger.info("Potential error: Distance from mgra "+mgra+" to tap "+tap+" is 0; resetting to 0.1 miles"); + distance = Math.round(Constants.feetPerMile * (float)0.1); + } + + if (!mgraWlkTapList.containsKey(mgra)) + mgraWlkTapList.put(mgra,new HashMap()); + mgraWlkTapList.get(mgra).put(tap,distance); + } + + // now go thru the array of ArrayLists and convert the lists to arrays + // and + // store in the class variable mgraWlkTapsDistArrays. + mgraWlkTapsDistArray = new int[maxMgra + 1][2][]; + nMgrasWithWlkTaps = mgraWlkTapList.size(); + for (int mgra : mgraWlkTapList.keySet()) { + Map wlkTapList = mgraWlkTapList.get(mgra); + mgraWlkTapsDistArray[mgra][0] = new int[wlkTapList.size()]; + mgraWlkTapsDistArray[mgra][1] = new int[wlkTapList.size()]; + int counter = 0; + for (int tap : new TreeSet(wlkTapList.keySet())) { //get the taps in ascending order - not sure if this matters, but it is cleaner + int distance = wlkTapList.get(tap); + mgraWlkTapsDistArray[mgra][0][counter] = tap; + mgraWlkTapsDistArray[mgra][1][counter] = distance; + counter++; + } + } + } + + /** + * read tap lines table (tap, line names served) + * @param rbMap + */ + public void readTapLines(HashMap rbMap) { + + File tapLinesTableFile = Paths.get(Util.getStringValueFromPropertyMap(rbMap, "scenario.path"), + Util.getStringValueFromPropertyMap(rbMap, "maz.tap.tapLines")).toFile(); + try { + CSVFileReader csvReader = new CSVFileReader(); + tapLinesTable = csvReader.readFile( tapLinesTableFile); + } catch (IOException e) { + throw new RuntimeException(); + } + + //get tap lines table field names + int[] tapLinesTapIds = tapLinesTable.getColumnAsInt("TAP"); + String[] linesForTap = tapLinesTable.getColumnAsString("LINES"); + + //create lookups + taplines = new HashMap(); + for(int i=0; i maz2TapData = new ArrayList(); + for (int j=0; j linesServed = new HashMap(); + for (Maz2Tap m2t : maz2TapData) { + + //skip if no lines served + if(m2t.lines != null) { + + for (int k=0; k tapsToRemove = new ArrayList(); + for (Maz2Tap m2t : maz2TapData) { + mazToTaps = mazToTaps + 1; + if( m2t.servesNewLines == false) { + tapsToRemove.add(m2t.tap); + trimmedTaps = trimmedTaps + 1; + } + } + + int[] finalTaps = new int[taps.length-tapsToRemove.size()]; + int[] finalDistances = new int[taps.length-tapsToRemove.size()]; + + int tapCounter = 0; + for (int m=0; m rbMap) + { + String mgraWlkTimeFile = rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT) + + rbMap.get(MGRA_MGRA_WALK_FILE_PROPERTY); + oMgraWalkDistance = new HashMap[maxMgra + 1]; + dMgraWalkDistance = new HashMap[maxMgra + 1]; + + TableDataSet mgraWalkData = Util.readTableDataSet(mgraWlkTimeFile); + + //i,j,walkTime,dist,mmTime,mmCost,mtTime,mtCost,mmGenTime,mtGenTime,minTime + for(int row = 1; row <= mgraWalkData.getRowCount();++row) { + + int oMgra = (int) mgraWalkData.getValueAt(row, "i"); + int dMgra = (int) mgraWalkData.getValueAt(row, "j"); + int distance = Math.round( mgraWalkData.getValueAt(row, "minTime") / Constants.walkMinutesPerMile * Constants.feetPerMile); + + if (oMgraWalkDistance[oMgra] == null) + oMgraWalkDistance[oMgra] = new HashMap(); + oMgraWalkDistance[oMgra].put(dMgra, distance); + + if (dMgraWalkDistance[dMgra] == null) + dMgraWalkDistance[dMgra] = new HashMap(); + dMgraWalkDistance[dMgra].put(oMgra, distance); + } + + } + + /** + * Return an int array of mgras within walking distance of this mgra. + * + * @param mgra + * The mgra to look up + * @return The mgras within walking distance. Null is returned if no mgras + * are within walk distance. + */ + public int[] getMgrasWithinWalkDistanceFrom(int mgra) + { + + if (oMgraWalkDistance[mgra] == null) return null; + + Set keySet = oMgraWalkDistance[mgra].keySet(); + int[] walkMgras = new int[keySet.size()]; + Iterator it = keySet.iterator(); + int i = 0; + while (it.hasNext()) + { + walkMgras[i] = it.next(); + ++i; + } + return walkMgras; + + } + + /** + * Return an int array of mgras within walking distance of this mgra. + * + * @param mgra + * The mgra to look up + * @return The mgras within walking distance. Null is returned if no mgras + * are within walk distance. + */ + public int[] getMgrasWithinWalkDistanceTo(int mgra) + { + + if (dMgraWalkDistance[mgra] == null) return null; + + Set keySet = dMgraWalkDistance[mgra].keySet(); + int[] walkMgras = new int[keySet.size()]; + Iterator it = keySet.iterator(); + int i = 0; + while (it.hasNext()) + { + walkMgras[i] = it.next(); + ++i; + } + return walkMgras; + + } + + /** + * Return true if mgras are within walking distance of each other. + * + * @param oMgra + * The from mgra + * @param dMgra + * The to mgra + * @return The mgras are within walking distance - true or false. + */ + public boolean getMgrasAreWithinWalkDistance(int oMgra, int dMgra) + { + + if (dMgraWalkDistance[dMgra] == null) return false; + + return dMgraWalkDistance[dMgra].containsKey(oMgra); + + } + + + + /** + * Get the position of the tap in the mgra walk tap array. + * + * @param mgra + * The mgra to lookup + * @param tap + * The tap to lookup + * @return The position of the tap in the mgra array. -1 is returned if it + * is an invalid tap for the mgra, or if the tap is not within + * walking distance. + */ + public int getTapPosition(int mgra, int tap) + { + + if (mgraWlkTapsDistArray[mgra] != null) + { + if (mgraWlkTapsDistArray[mgra][0] != null) + { + for (int i = 0; i < mgraWlkTapsDistArray[mgra][0].length; ++i) + if (mgraWlkTapsDistArray[mgra][0][i] == tap) return i; + } + } + + return -1; + + } + + /** + * Get the walk board time from an MGRA to a TAP. + * + * @param mgra + * The number of the destination MGRA. + * @param pos + * The position of the TAP in the MGRA array (0+) + * @return The walk time in minutes. + */ + public float getMgraToTapWalkBoardTime(int mgra, int pos) + { + float distanceInFeet = (float) mgraWlkTapsDistArray[mgra][1][pos]; + float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; + return time; + } + + + //todo: delete this method: currently retained for compatibility (namely: abm_reports) + /** + * Get the walk time from an MGRA to a TAP. + * + * @param mgra The number of the destination MGRA. + * @param pos The position of the TAP in the MGRA array (0+) + * @return The walk time in minutes. + */ + public float getMgraToTapWalkTime(int mgra, int pos) + { + float distanceInFeet = (float) mgraWlkTapsDistArray[mgra][1][pos]; + float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; + return time; + } + + /** + * Get the walk distance from an MGRA to an MGRA. Return 0 if not within walking + * distance. + * + * @param oMgra + * The number of the production/origin MGRA. + * @param dMgra + * The number of the attraction/destination MGRA. + * @return The walk distance in feet. + */ + public int getMgraToMgraWalkDistFrom(int oMgra, int dMgra) + { + + if (oMgraWalkDistance[oMgra] == null) return 0; + else if (oMgraWalkDistance[oMgra].containsKey(dMgra)) + //return oMgraWalkDistance[oMgra].get(dMgra)[0]; + + return oMgraWalkDistance[oMgra].get(dMgra); + + return 0; + } + + /** + * Get the walk time from an MGRA to a TAP. + * + * @param mgra The MGRA + * @param tap The TAP + * @return The walk time in minutes, else -1 if there is no walk link between the MGRA and the TAP. + */ + public float getWalkTimeFromMgraToTap(int mgra, int tap){ + + int tapPosition = getTapPosition(mgra, tap); + float time = 0; + + if(tapPosition==-1){ + logger.info("Bad Tap Position for Walk Access From MAZ: "+mgra+" to TAP: "+tap); + return -1; + } + else{ + time = (float) (mgraWlkTapsDistArray[mgra][1][tapPosition] * Constants.walkMinutesPerFoot); + } + return time; + } + /** + * Get the walk distance from an MGRA to a TAP. + * + * @param mgra The MGRA + * @param tap The TAP + * @return The walk distance in miles, else -1 if there is no walk link between the MGRA and the TAP. + */ + public float getWalkDistanceFromMgraToTap(int mgra, int tap){ + + int tapPosition = getTapPosition(mgra, tap); + float distance = 0; + + if(tapPosition==-1){ + logger.info("Bad Tap Position for Walk Access From MAZ: "+mgra+" to TAP: "+tap); + return -1; + } + else{ + distance = (float) (mgraWlkTapsDistArray[mgra][1][tapPosition]/Constants.feetPerMile); + } + return distance; + } + /** + * Get the walk distance from an MGRA to an MGRA. Return 0 if not within + * walking distance. + * + * @param oMgra + * The number of the production/origin MGRA. + * @param dMgra + * The number of the attraction/destination MGRA. + * @return The walk distance in feet. + */ + public int getMgraToMgraWalkDistTo(int oMgra, int dMgra) + { + + if (dMgraWalkDistance[dMgra] == null) return 0; + else if (dMgraWalkDistance[dMgra].containsKey(oMgra)) + + return dMgraWalkDistance[dMgra].get(oMgra); + + return 0; + } + + /** + * Get the walk time from an MGRA to an MGRA. Return 0 if not within walking + * distance. + * + * @param oMgra + * The number of the production/origin MGRA. + * @param dMgra + * The number of the attraction/destination MGRA. + * @return The walk time in minutes. + */ + public float getMgraToMgraWalkTime(int oMgra, int dMgra) + { + + if (oMgraWalkDistance[oMgra] == null) return 0f; + else if (oMgraWalkDistance[oMgra].containsKey(dMgra)){ + float distanceInFeet = (float) oMgraWalkDistance[oMgra].get(dMgra); + float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; + return time; + } + return 0f; + } + + /** + * Get the bike time from an MGRA to an MGRA. Return 0 if not within walking + * distance. + * + * @param oMgra + * The number of the production/origin MGRA. + * @param dMgra + * The number of the attraction/destination MGRA. + * @return The bike time in minutes. + */ + public float getMgraToMgraBikeTime(int oMgra, int dMgra) + { + double time = bls.getTime(defaultSegment,oMgra,dMgra); + return (time == Double.POSITIVE_INFINITY) ? 0f : (float) time; + } + + /** + * Print mgra data to the log file for debugging purposes. + * + */ + public void printMgraStats() + { + logger.info("Number of MGRAs: " + mgras.size()); + logger.info("Max MGRA: " + maxMgra); + + // logger.info("Number of MGRAs with WalkAccessTaps: " + + // nMgrasWithWlkTaps); + // logger.info("Number of TAPs in MGRA 18 (should be 3): " + // + mgraWlkTapsDistArray[18][0].length); + // logger.info("Distance between MGRA 18 and TAP 1648 (should be 2728): " + // + mgraWlkTapsDistArray[18][1][1]); + // logger.info("MGRA 28435 is in what TAZ? (Should be 995)" + + // mgraTaz[28435]); + // logger.info("Number of mgras within walk distance of mgra 22573 (Should be 67)" + // + getMgrasWithinWalkDistanceFrom(22573).length); + + } + + /** + * + * @param mgra + * - the zone + * @return the taz that the tmgra is contained in + */ + public int getTaz(int mgra) + { + return mgraTaz[mgra]; + } + + /** + * + * @param mgra + * - the zone + * @return the luz that the mgra is contained in + */ + public int getMgraLuz(int mgra) + { + return mgraLuz[mgra]; + } + + /** + * Get the maximum LUZ. + * + * @return The highest LUZ number + */ + public int getMaxLuz() + { + return maxLuz; + } + + /** + * Get the maximum MGRA. + * + * @return The highest MGRA number + */ + public int getMaxMgra() + { + return maxMgra; + } + + /** + * Get the maximum TAP. + * + * @return The highest TAP number + */ + public int getMaxTap() + { + return maxTap; + } + + /** + * Get the ArrayList of MGRAs + * + * @return ArrayList mgras. + */ + public ArrayList getMgras() + { + return mgras; + } + + /** + * Get the MgraTaz correspondence array. Given an MGRA, returns its TAZ. + * + * @return int[] mgraTaz correspondence array. + */ + public int[] getMgraTaz() + { + return mgraTaz; + } + + /** + * Get the array of Taps within walk distance + * + * @return The int[][][] array of Taps within walk distance of MGRAs + */ + public int[][][] getMgraWlkTapsDistArray() + { + return mgraWlkTapsDistArray; + } + + /** + * get arrays of drive accessible TAPS for each MGRA and populate an array + * of sets so that later one can determine, for a given mgra, if a tap is + * contained in the set. + * + * @param args + * TazDataManager to get TAPs with drive access from TAZs + */ + public void mapDriveAccessTapsToMgras(TazDataManager tazDataManager) + { + + walkAccessibleTaps = new TreeSet[maxMgra + 1]; + driveAccessibleTaps = new TreeSet[maxMgra + 1]; + + for (int mgra = 1; mgra <= maxMgra; mgra++) + { + + // get the TAZ associated with this MGRA + int taz = getTaz(mgra); + + // store the array of walk accessible TAPS for this MGRA as a set so + // that contains can be called on it later + // to determine, for a given mgra, if a tap is contained in the set. + int[] mgraSet = getMgraWlkTapsDistArray()[mgra][0]; + if (mgraSet != null) + { + walkAccessibleTaps[mgra] = new TreeSet(); + for (int i = 0; i < mgraSet.length; i++) + walkAccessibleTaps[mgra].add(mgraSet[i]); + } + + // store the array of drive accessible TAPS for this MGRA as a set + // so that contains can be called on it later + // to determine, for a given mgra, if a tap is contained in the set. + int[] tapItems = tazDataManager.getParkRideOrKissRideTapsForZone(taz, + AccessMode.PARK_N_RIDE); + driveAccessibleTaps[mgra] = new TreeSet(); + for (int item : tapItems) + driveAccessibleTaps[mgra].add(item); + + } + + } + + /** + * @param mgra + * for which we want to know if TAP can be reached by drive + * access + * @param tap + * for which we want to know if the mgra can reach it by drive + * access + * @return true if reachable; false otherwise + */ + public boolean getTapIsDriveAccessibleFromMgra(int mgra, int tap) + { + if (driveAccessibleTaps[mgra] == null) return false; + else return driveAccessibleTaps[mgra].contains(tap); + } + + /** + * @param mgra + * for which we want to know if TAP can be reached by walk access + * @param tap + * for which we want to know if the mgra can reach it by walk + * access + * @return true if reachable; false otherwise + */ + public boolean getTapIsWalkAccessibleFromMgra(int mgra, int tap) + { + if (walkAccessibleTaps[mgra] == null) return false; + else return walkAccessibleTaps[mgra].contains(tap); + } + + /** + * return the duDen value for the mgra + * + * @param mgra + * is the MGRA value for which the duDen value is needed + * @return duDen[mgra] + */ + public double getDuDenValue(int mgra) + { + return duDen[mgra]; + } + + /** + * return the empDen value for the mgra + * + * @param mgra + * is the MGRA value for which the empDen value is needed + * @return empDen[mgra] + */ + public double getEmpDenValue(int mgra) + { + return empDen[mgra]; + } + + /** + * return the totInt value for the mgra + * + * @param mgra + * is the MGRA value for which the totInt value is needed + * @return totInt[mgra] + */ + public double getTotIntValue(int mgra) + { + return totInt[mgra]; + } + + + public double getPopEmpPerSqMi( int mgra ) { + return popEmpDenPerSqMi[mgra]; + } + + /** + * Process the 4D density land use data file and store the selected fields + * as arrays indexed by the mgra value. The data fields are in the mgra + * TableDataSet read from the MGRA csv file. + * + * @param rbMap + * is a HashMap for the resource bundle generated from the + * properties file. + */ + public void process4ddensityData(HashMap rbMap) + { + + try + { + + // allocate arrays for the land use data fields + duDen = new double[maxMgra + 1]; + empDen = new double[maxMgra + 1]; + totInt = new double[maxMgra + 1]; + + //added for Taxi/TNC + popEmpDenPerSqMi = new double[maxMgra+1]; + + // get the data fields needed for the mode choice utilities as + // 0-based double[] + double[] duDenField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_DU_DEN_FIELD); + double[] empDenField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_EMP_DEN_FIELD); + double[] totIntField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_TOT_INT_FIELD); + double[] popEmpField = mgraTableDataSet.getColumnAsDouble( MGRA_POPEMPPERSQMI_FIELD ); + + // create a HashMap to convert MGRA values to array indices for the + // data + // arrays above + int mgraCol = mgraTableDataSet.getColumnPosition(MGRA_FIELD_NAME); + + for (int row = 1; row <= mgraTableDataSet.getRowCount(); row++) + { + + int mgra = (int) mgraTableDataSet.getValueAt(row, mgraCol); + duDen[mgra] = duDenField[row - 1]; + empDen[mgra] = empDenField[row - 1]; + totInt[mgra] = totIntField[row - 1]; + popEmpDenPerSqMi [mgra] = popEmpField[row-1]; + + } + + } catch (Exception e) + { + logger.error( + String.format("Exception occurred processing 4ddensity data file from mgraData TableDataSet object."), + e); + throw new RuntimeException(e); + } + + } + + private void readMgraTableData(HashMap rbMap) + { + + // get the mgra data table from one of these UECs. + String projectPath = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String mgraFile = rbMap.get(PROPERTIES_MGRA_DATA_FILE); + mgraFile = projectPath + mgraFile; + + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + mgraTableDataSet = reader.readFile(new File(mgraFile)); + } catch (IOException e) + { + logger.error("problem reading mgra data table for MgraDataManager.", e); + System.exit(1); + } + + HashMap tazs = new HashMap(); + HashMap luzs = new HashMap(); + + // create a HashMap between mgra values and the corresponding row number + // in the mgra TableDataSet. + mgraDataTableMgraRowMap = new HashMap(); + maxMgra = 0; + maxLuz = 0; + for (int i = 1; i <= mgraTableDataSet.getRowCount(); i++) + { + int mgra = (int) mgraTableDataSet.getValueAt(i, MGRA_FIELD_NAME); + int taz = (int) mgraTableDataSet.getValueAt(i, MGRA_TAZ_FIELD_NAME); + + int luz = (int) mgraTableDataSet.getValueAt(i, MGRA_LUZ_FIELD_NAME); + + mgraDataTableMgraRowMap.put(mgra, i); + + if (mgra > maxMgra) maxMgra = mgra; + mgras.add(mgra); + + tazs.put(mgra, taz); + + if (luz > 0) + { + if (luz > maxLuz) maxLuz = luz; + luzs.put(mgra, luz); + } + } + + mgraTaz = new int[maxMgra + 1]; + for (int mgra : mgras) + mgraTaz[mgra] = tazs.get(mgra); + + mgraLuz = new int[maxMgra + 1]; + for (int mgra : mgras) + mgraLuz[mgra] = luzs.get(mgra); + + } + + /** + * @param mgra + * for which table data is desired + * @return population for the specified mgra. + */ + public double getMgraPopulation(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_POPULATION_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return households for the specified mgra. + */ + public double getMgraHouseholds(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_HOUSEHOLDS_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return grade school enrollment for the specified mgra. + */ + public double getMgraGradeSchoolEnrollment(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_GRADE_SCHOOL_ENROLLMENT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return high school enrollment for the specified mgra. + */ + public double getMgraHighSchoolEnrollment(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_HIGH_SCHOOL_ENROLLMENT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return university enrollment for the specified mgra. + */ + public double getMgraUniversityEnrollment(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_UNIVERSITY_ENROLLMENT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return other college enrollment for the specified mgra. + */ + public double getMgraOtherCollegeEnrollment(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_OTHER_COLLEGE_ENROLLMENT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return adult school enrollment for the specified mgra. + */ + public double getMgraAdultSchoolEnrollment(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_ADULT_SCHOOL_ENROLLMENT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return grade school district for the specified mgra. + */ + public int getMgraGradeSchoolDistrict(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return (int) mgraTableDataSet.getValueAt(row, MGRA_GRADE_SCHOOL_DISTRICT_FIELD_NAME); + } + + /** + * @param mgra + * for which table data is desired + * @return high school district for the specified mgra. + */ + public int getMgraHighSchoolDistrict(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return (int) mgraTableDataSet.getValueAt(row, MGRA_HIGH_SCHOOL_DISTRICT_FIELD_NAME); + } + + public HashMap getMgraDataTableMgraRowMap() + { + return mgraDataTableMgraRowMap; + } + + private void calculateMgraAvgParkingCosts(HashMap propertyMap) + { + + // open output file to write average parking costs for each mgra + PrintWriter out = null; + + String projectPath = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String outFile = propertyMap.get(PROPERTIES_PARKING_COST_OUTPUT_FILE); + outFile = projectPath + outFile; + + try + { + out = new PrintWriter(new BufferedWriter(new FileWriter(new File(outFile)))); + } catch (IOException e) + { + logger.error("Exception caught trying to create file " + outFile); + System.out.println("Exception caught trying to create file " + outFile); + e.printStackTrace(); + throw new RuntimeException(); + } + + // write the header record + out.println("mgra,mgraParkArea,lsWgtAvgCostM,lsWgtAvgCostD,lsWgtAvgCostH"); + + // open output files for writing debug info for a specific mgra + PrintWriter outM = null; + PrintWriter outD = null; + PrintWriter outH = null; + + if (LOG_MGRA > 0) + { + try + { + outM = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath + + "output/" + LOG_MGRA_FILE + "M.csv")))); + outD = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath + + "output/" + LOG_MGRA_FILE + "D.csv")))); + outH = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath + + "output/" + LOG_MGRA_FILE + "H.csv")))); + } catch (IOException e) + { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + + float workDistCoeff = Float.parseFloat(propertyMap.get(MGRA_DISTANCE_COEFF_WORK)); + float otherDistCoeff = Float.parseFloat(propertyMap.get(MGRA_DISTANCE_COEFF_OTHER)); + + int[] mgraField = mgraTableDataSet.getColumnAsInt(MGRA_FIELD_NAME); + int[] mgraParkAreaField = mgraTableDataSet.getColumnAsInt(MGRA_PARKAREA_FIELD); + int[] hstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_HSTALLSOTH_FIELD); + int[] hstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_HSTALLSSAM_FIELD); + float[] hparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_HPARKCOST_FIELD); + int[] numfreehrsField = mgraTableDataSet.getColumnAsInt(MGRA_NUMFREEHRS_FIELD); + int[] dstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_DSTALLSOTH_FIELD); + int[] dstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_DSTALLSSAM_FIELD); + float[] dparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_DPARKCOST_FIELD); + int[] mstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_MSTALLSOTH_FIELD); + int[] mstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_MSTALLSSAM_FIELD); + float[] mparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_MPARKCOST_FIELD); + + mgraParkArea = new int[maxMgra + 1]; + numfreehrs = new int[maxMgra + 1]; + hstallsoth = new int[maxMgra + 1]; + hstallssam = new int[maxMgra + 1]; + hparkcost = new float[maxMgra + 1]; + dstallsoth = new int[maxMgra + 1]; + dstallssam = new int[maxMgra + 1]; + dparkcost = new float[maxMgra + 1]; + mstallsoth = new int[maxMgra + 1]; + mstallssam = new int[maxMgra + 1]; + mparkcost = new float[maxMgra + 1]; + + lsWgtAvgCostM = new double[maxMgra + 1]; + lsWgtAvgCostD = new double[maxMgra + 1]; + lsWgtAvgCostH = new double[maxMgra + 1]; + + // loop over the number of mgra records in the TableDataSet. + for (int k = 0; k < maxMgra; k++) + { + + // get the mgra value for TableDataSet row k from the mgra field. + int mgra = mgraField[k]; + + mgraParkArea[mgra] = mgraParkAreaField[k]; + numfreehrs[mgra] = numfreehrsField[k]; + hstallsoth[mgra] = hstallsothField[k]; + hstallssam[mgra] = hstallssamField[k]; + hparkcost[mgra] = hparkcostField[k]; + dstallsoth[mgra] = dstallsothField[k]; + dstallssam[mgra] = dstallssamField[k]; + dparkcost[mgra] = dparkcostField[k]; + mstallsoth[mgra] = mstallsothField[k]; + mstallssam[mgra] = mstallssamField[k]; + mparkcost[mgra] = mparkcostField[k]; + + // get the array of mgras within walking distance of m + int[] walkMgras = getMgrasWithinWalkDistanceFrom(mgra); + + // park area 1. + if (mgraParkArea[mgra] == PARK_AREA_ONE) + { + + // calculate weighted average cost from monthly costs + double dist = getMgraToMgraWalkDistFrom(mgra, mgra) / 5280.0; + + double numeratorM = mstallssam[mgra] * Math.exp(workDistCoeff * dist) + * mparkcost[mgra]; + double denominatorM = mstallssam[mgra] * Math.exp(workDistCoeff * dist); + + double numeratorD = dstallssam[mgra] * Math.exp(workDistCoeff * dist) + * dparkcost[mgra]; + double denominatorD = dstallssam[mgra] * Math.exp(workDistCoeff * dist); + + double discountFactor = Math.max(1 - (numfreehrs[mgra] / 4), 0); + double numeratorH = hstallssam[mgra] * Math.exp(workDistCoeff * dist) + * discountFactor * hparkcost[mgra]; + double denominatorH = hstallssam[mgra] * Math.exp(workDistCoeff * dist); + + if (mgra == LOG_MGRA) + { + // log the file header + outM.println("wMgra" + "," + "mgraParkArea" + "," + "workDistCoeff*dist" + "," + + "exp(workDistCoeff*dist)" + "," + "mstallsoth" + "," + "mparkcost" + + "," + "numeratorM" + "," + "denominatorM"); + outD.println("wMgra" + "," + "mgraParkArea" + "," + "otherDistCoeff*dist" + "," + + "exp(otherDistCoeff*dist)" + "," + "dstallsoth" + "," + "dparkcost" + + "," + "numeratorD" + "," + "denominatorD"); + outH.println("wMgra" + "," + "mgraParkArea" + "," + "otherDistCoeff*dist" + "," + + "exp(otherDistCoeff*dist)" + "," + "discountFactor" + "," + + "hstallsoth" + "," + "hparkcost" + "," + "numeratorH" + "," + + "denominatorH"); + + outM.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," + + Math.exp(workDistCoeff * dist) + "," + mstallsoth[mgra] + "," + + mparkcost[mgra] + "," + numeratorM + "," + denominatorM); + outD.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," + + Math.exp(workDistCoeff * dist) + "," + dstallsoth[mgra] + "," + + dparkcost[mgra] + "," + numeratorD + "," + denominatorD); + outH.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," + + Math.exp(workDistCoeff * dist) + "," + discountFactor + "," + + hstallsoth[mgra] + "," + hparkcost[mgra] + "," + numeratorH + "," + + denominatorH); + } + + if (walkMgras != null) + { + + for (int wMgra : walkMgras) + { + + // skip mgra if not in park area 1 or 2. + if (mgraParkArea[wMgra] > 2) + { + if (mgra == LOG_MGRA) + { + outM.println(wMgra + "," + mgraParkArea[wMgra]); + outD.println(wMgra + "," + mgraParkArea[wMgra]); + outH.println(wMgra + "," + mgraParkArea[wMgra]); + } + continue; + } + + if (wMgra != mgra) + { + dist = getMgraToMgraWalkDistFrom(mgra, wMgra) / 5280.0; + + if (dist > MAX_PARKING_WALK_DISTANCE) + { + if (mgra == LOG_MGRA) + { + outM.println(wMgra + "," + mgraParkArea[wMgra]); + outD.println(wMgra + "," + mgraParkArea[wMgra]); + outH.println(wMgra + "," + mgraParkArea[wMgra]); + } + continue; + } + + numeratorM += mstallsoth[wMgra] * Math.exp(workDistCoeff * dist) + * mparkcost[wMgra]; + denominatorM += mstallsoth[wMgra] * Math.exp(workDistCoeff * dist); + + numeratorD += dstallsoth[wMgra] * Math.exp(otherDistCoeff * dist) + * dparkcost[wMgra]; + denominatorD += dstallsoth[wMgra] * Math.exp(otherDistCoeff * dist); + + discountFactor = Math.max(1 - (numfreehrs[wMgra] / 4), 0); + numeratorH += hstallsoth[wMgra] * Math.exp(otherDistCoeff * dist) + * discountFactor * hparkcost[wMgra]; + denominatorH += hstallsoth[wMgra] * Math.exp(otherDistCoeff * dist); + + if (mgra == LOG_MGRA) + { + outM.println(wMgra + "," + mgraParkArea[wMgra] + "," + + workDistCoeff * dist + "," + + Math.exp(workDistCoeff * dist) + "," + mstallsoth[wMgra] + + "," + mparkcost[wMgra] + "," + numeratorM + "," + + denominatorM); + outD.println(wMgra + "," + mgraParkArea[wMgra] + "," + + otherDistCoeff * dist + "," + + Math.exp(otherDistCoeff * dist) + "," + dstallsoth[wMgra] + + "," + dparkcost[wMgra] + "," + numeratorD + "," + + denominatorD); + outH.println(wMgra + "," + mgraParkArea[wMgra] + "," + + otherDistCoeff * dist + "," + + Math.exp(otherDistCoeff * dist) + "," + discountFactor + + "," + hstallsoth[wMgra] + "," + hparkcost[wMgra] + "," + + numeratorH + "," + denominatorH); + } + + } + + } + + } + // jef: storing by mgra since they are indexed into by mgra + // wsu added if clauses. If denominators are 0, read costs directly from input file + if(denominatorM>0) + lsWgtAvgCostM[mgra] = numeratorM / denominatorM; + else + lsWgtAvgCostM[mgra] = mparkcost[mgra]; + if(denominatorD>0) + lsWgtAvgCostD[mgra] = numeratorD / denominatorD; + else + lsWgtAvgCostD[mgra] = dparkcost[mgra]; + if(denominatorH>0) + lsWgtAvgCostH[mgra] = numeratorH / denominatorH; + else + lsWgtAvgCostH[mgra] = hparkcost[mgra]; + } else + { + + lsWgtAvgCostM[mgra] = mparkcost[mgra]; + lsWgtAvgCostD[mgra] = dparkcost[mgra]; + lsWgtAvgCostH[mgra] = hparkcost[mgra]; + + } + + // write the data record + out.println(mgra + "," + mgraParkArea[mgra] + "," + lsWgtAvgCostM[mgra] + "," + + lsWgtAvgCostD[mgra] + "," + lsWgtAvgCostH[mgra]); + } + + if (LOG_MGRA > 0) + { + outM.close(); + outD.close(); + outH.close(); + } + + out.close(); + + } + + public double[] getLsWgtAvgCostM() + { + return lsWgtAvgCostM; + } + + public double[] getLsWgtAvgCostD() + { + return lsWgtAvgCostD; + } + + public double[] getLsWgtAvgCostH() + { + return lsWgtAvgCostH; + } + + public int[] getMgraParkAreas() + { + return mgraParkArea; + } + + public int[] getNumFreeHours() + { + return numfreehrs; + } + + public int[] getMStallsOth() + { + return mstallsoth; + } + + public int[] getMStallsSam() + { + return mstallssam; + } + + public float[] getMParkCost() + { + return mparkcost; + } + + public int[] getDStallsOth() + { + return dstallsoth; + } + + public int[] getDStallsSam() + { + return dstallssam; + } + + public float[] getDParkCost() + { + return dparkcost; + } + + public int[] getHStallsOth() + { + return hstallsoth; + } + + public int[] getHStallsSam() + { + return hstallssam; + } + + public float[] getHParkCost() + { + return hparkcost; + } + + /** + * @param mgra + * for which table data is desired + * @return high school district for the specified mgra. + */ + public int getMgraHourlyParkingCost(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return (int) mgraTableDataSet.getValueAt(row, MGRA_HPARKCOST_FIELD); + } + + public float getRefeulingStations(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_REFUELING_STATIONS_FIELD_NAME); + } + + public float getRemoteParkingLot(int mgra) + { + int row = mgraDataTableMgraRowMap.get(mgra); + return mgraTableDataSet.getValueAt(row, MGRA_REMOTE_PARKING_LOT_FIELD_NAME); + } + private class Maz2Tap implements Comparable, Serializable + { + public int maz; + public int tap; + public double dist; + public String[] lines; + public boolean servesNewLines = false; + + @Override + public int compareTo(Maz2Tap o) { + if ( this.dist < o.dist ) { + return -1; + } else if (this.dist==o.dist) { + return 0; + } else { + return 1; + } + } + } + public TableDataSet getMgraTableDataSet() { + return mgraTableDataSet; + } + + public static void main(String[] args) + { + ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); + MgraDataManager mdm = MgraDataManager.getInstance(ResourceUtil + .changeResourceBundleIntoHashMap(rb)); + mdm.printMgraStats(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java new file mode 100644 index 0000000..2204762 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java @@ -0,0 +1,115 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; + +/** + * This class is used for specifying modes. + * + * @author Christi Willison + * @version Sep 8, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public final class Modes + implements Serializable +{ + + public enum AutoMode + { + DRIVE_ALONE_TOLL("dat"), DRIVE_ALONE_NONTOLL("dan"), TWO_NONTOLL("2nt"), TWO_TOLL( + "2t"), THREEPLUS_NONTOLL("3+nt"), THREEPLUS_TOLL( + "3+t"); + + private final String name; + + AutoMode(String s) + { + this.name = s; + } + + public AutoMode[] getAutoModes() + { + return AutoMode.values(); + } + + public String toString() + { + return name; + } + } + + public enum TransitMode + { + COMMUTER_RAIL("cr", true), // label and true = premium + LIGHT_RAIL("lr", true), BRT("brt", true), EXPRESS_BUS("eb", true), LOCAL_BUS("lb", false); + + private final String name; + private final boolean premium; + + TransitMode(String name, boolean premium) + { + this.name = name; + this.premium = premium; + } + + public TransitMode[] getTransitModes() + { + return TransitMode.values(); + } + + public boolean isPremiumMode(TransitMode transitMode) + { + return transitMode.premium; + } + + public String toString() + { + return name; + } + + } + + public enum AccessMode + { + WALK("WLK"), PARK_N_RIDE("PNR"), KISS_N_RIDE("KNR"); + private final String name; + + AccessMode(String name) + { + this.name = name; + } + + public AccessMode[] getAccessModes() + { + return AccessMode.values(); + } + + public String toString() + { + return name; + } + + } + + public enum NonMotorizedMode + { + WALK, BIKE + } + + public enum OtherMode + { + SCHOOL_BUS + } + + private Modes() + { + // Not implemented in utility classes + } + + + public static void main(String[] args) + { + System.out.println(AutoMode.DRIVE_ALONE_NONTOLL.toString()); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java new file mode 100644 index 0000000..01a7a4b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java @@ -0,0 +1,146 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +/** + * This class is used for non-motorized DMU attributes. + * + * @author Joel Freedman + * @version May 28,2009 + *

+ */ +public class NonMotorDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(NonMotorDMU.class); + + protected HashMap methodIndexMap; + + private float mgraWalkTime; + private float mgraBikeTime; + + public NonMotorDMU() + { + setupMethodIndexMap(); + } + + /** + * Get MGRA-MGRA walk time. + * + * @return Mgra-mgra walk time in minutes. + */ + public float getMgraWalkTime() + { + return mgraWalkTime; + } + + /** + * Set Mgra-Mgra walk time in minutes. + * + * @param mgraWalkTime + * Mgra walk time in minutes. + */ + public void setMgraWalkTime(float mgraWalkTime) + { + this.mgraWalkTime = mgraWalkTime; + } + + /** + * Get MGRA-MGRA bike time. + * + * @return Mgra-mgra bike time in minutes. + */ + public float getMgraBikeTime() + { + return mgraBikeTime; + } + + /** + * Set Mgra-Mgra bike time in minutes. + * + * @param mgraBikeTime + * Mgra bike time in minutes. + */ + public void setMgraBikeTime(float mgraBikeTime) + { + this.mgraBikeTime = mgraBikeTime; + } + + /** + * Log the DMU values. + * + * @param localLogger + * The logger to use. + */ + public void logValues(Logger localLogger) + { + + localLogger.info(""); + localLogger.info("Non-Motorized DMU Values:"); + localLogger.info(""); + localLogger.info(String.format("MGRA-MGRA Walk Time: %9.4f", mgraWalkTime)); + localLogger.info(String.format("MGRA-MGRA Bike Time: %9.4f", mgraBikeTime)); + + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getMgraBikeTime", 0); + methodIndexMap.put("getMgraWalkTime", 1); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = 0; + + switch (variableIndex) + { + case 0: + returnValue = getMgraBikeTime(); + break; + case 1: + returnValue = getMgraWalkTime(); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java new file mode 100644 index 0000000..4e80181 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java @@ -0,0 +1,189 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.Arrays; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.calculator.IndexValues; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.LogitModel; +import com.pb.common.newmodel.UtilityExpressionCalculator; +import com.pb.common.util.Tracer; + +/** + * This class is used for ... + * + * @author Christi Willison + * @version Mar 9, 2009 + *

+ * Created by IntelliJ IDEA. + */ +public class NonMotorUEC + implements Serializable +{ + + private transient Logger logger = Logger.getLogger(NonMotorUEC.class); + private TazDataManager tazManager; + private MgraDataManager mgraManager; + private UtilityExpressionCalculator uec; + private IndexValues index = new IndexValues(); + private int[] availFlag; + private NonMotorDMU dmu; + private LogitModel model; + private ChoiceModelApplication modelApp; + + // seek and trace + private boolean trace; + private int[] traceOtaz; + private int[] traceDtaz; + protected Tracer tracer; + + /** + * Default Constructor. + * + * @param rb + * @param uecFileName + * @param modelSheet + * @param dataSheet + */ + public NonMotorUEC(HashMap rbHashMap, String uecFileName, int modelSheet, + int dataSheet) + { + + dmu = new NonMotorDMU(); + + // use the choice model application to set up the model structure + modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); + + // but return the logit model itself, so we can use compound utilities + model = modelApp.getRootLogitModel(); + uec = modelApp.getUEC(); + availFlag = new int[uec.getNumberOfAlternatives() + 1]; + + tazManager = TazDataManager.getInstance(); + mgraManager = MgraDataManager.getInstance(); + + trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); + traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); + traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); + + // set up the tracer object + tracer = Tracer.getTracer(); + tracer.setTrace(trace); + for (int i = 0; i < traceOtaz.length; i++) + { + for (int j = 0; j < traceDtaz.length; j++) + { + tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); + logger.info("Setting trace zone pair in NonMotorUEC Object for i: "+ traceOtaz[i] + " j: " + traceDtaz[j]); + } + } + } + + /** + * Calculate utilities for a given TAZ pair. + * + * @param pTaz + * Production/Origin TAZ. + * @param aTaz + * Attraction/Destination TAZ. + * @return The root utility. + */ + public double calculateUtilitiesForTazPair(int pTaz, int aTaz) + { + + index.setOriginZone(pTaz); + index.setDestZone(aTaz); + + Arrays.fill(availFlag, 1); + dmu.setMgraWalkTime(0); + dmu.setMgraBikeTime(0); + + trace = false; + if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) + { + trace = true; + } + + // log DMU values + if (trace) + { + TapDataManager tapManager = TapDataManager.getInstance(); + if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 + && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) + uec.logDataValues(logger, pTaz, aTaz, 0); + dmu.logValues(logger); + } + + modelApp.computeUtilities(dmu, index); + double utility = modelApp.getLogsum(); + if (utility == 0) utility = -999; + + // logging + if (trace) + { + uec.logAnswersArray(logger, "NonMotorized UEC"); + uec.logResultsArray(logger, pTaz, aTaz); + modelApp.logLogitCalculations("NonMotorized UEC", "Zone Trace"); + logger.info("Logsum = " + utility); + trace = false; + } + + return utility; + } + + /** + * Calculate utilities for a given TAZ pair. + * + * @param oMgra + * Production/Origin Mgra. + * @param dMgra + * Attraction/Destination Mgra. + * @return The root utility. + */ + public double calculateUtilitiesForMgraPair(int oMgra, int dMgra) + { + + Arrays.fill(availFlag, 1); + + trace = false; + int pTaz = mgraManager.getTaz(oMgra); + int aTaz = mgraManager.getTaz(dMgra); + index.setOriginZone(pTaz); + index.setDestZone(aTaz); + + if (tracer.isTraceOn() && tracer.isTraceZone(pTaz)) + { + trace = true; + } + + dmu.setMgraWalkTime(mgraManager.getMgraToMgraWalkTime(oMgra, dMgra)); + dmu.setMgraBikeTime(mgraManager.getMgraToMgraBikeTime(oMgra, dMgra)); + + // log DMU values + if (trace) + { + logger.info("MGRA-MGRA non-motorized calculations for " + oMgra + " to " + dMgra); + dmu.logValues(logger); + } + + modelApp.computeUtilities(dmu, index); + double utility = modelApp.getLogsum(); + if (utility == 0) utility = -999; + + // logging + if (trace) + { + uec.logAnswersArray(logger, "NonMotorized UEC"); + uec.logResultsArray(logger, pTaz, aTaz); + modelApp.logLogitCalculations("NonMotorized UEC", "Mgra Trace"); + logger.info("Logsum = " + utility); + trace = false; + } + dmu.setMgraWalkTime(0); + dmu.setMgraWalkTime(0); + return utility; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java new file mode 100644 index 0000000..5f42ed1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java @@ -0,0 +1,321 @@ +package org.sandag.abm.modechoice; + +import java.io.BufferedReader; +import java.io.File; +import java.io.FileReader; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.Iterator; +import java.util.List; +import java.util.Map; +import java.util.ResourceBundle; +import java.util.StringTokenizer; +import java.util.TreeMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.active.sandag.SandagWalkPathAlternativeListGenerationConfiguration; +import org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.util.ResourceUtil; + +public final class TapDataManager + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(TapDataManager.class); + private static volatile TapDataManager instance = null; + private static final Object LOCK = new Object(); + + // tapID, [tapNum, lotId, ??, taz][list of values] + private float[][][] tapParkingInfo; + + // an array that stores parking lot use by lot ID. + private int[] lotUse; + + // an array of taps + private int[] taps; + private int maxTap; + + public int getMaxTap() + { + return maxTap; + } + + private TapDataManager(HashMap rbMap) + { + + System.out.println("I'm the TapDataManager"); + readTap(rbMap); + getTapList(rbMap); + intializeLotUse(); + printStats(); + } + + /** + * This method should only be used after the getInstance(HashMap rbMap) method has been called since the rbMap is needed to read + * in all the data and populate the object. This method will return the + * instance that has already been populated. + * + * @return instance + * @throws RuntimeException + */ + public static TapDataManager getInstance() + { + if (instance == null) + { + throw new RuntimeException( + "Must instantiate TapDataManager with the getInstance(rbMap) method first"); + } else + { + return instance; + } + } + + /** + * This method will read in the tapParkingInfo.ptype file and store the info + * in a TreeMap where key equals the iTap and value equals an array of [][3] + * elements. The TreeMap will be passed to the populateTap function which + * will transpose the array of [][3] elements to an array of [4][] elements + * and attaches it to the this.tapParkingInfo[key] + * + * //TODO: Test this and see if there is only a single lot associated // + * TODO with each tap. + * + * The file has 6 columns - tap, lotId, parking type, taz, capacity and mode + * + * @param rb + * - the resource bundle that lists the tap.ptype file and + * scenario.path. + */ + private void readTap(HashMap rbMap) + { + + File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") + + Util.getStringValueFromPropertyMap(rbMap, "tap.ptype.file")); + String s; + TreeMap> map = new TreeMap>(); + StringTokenizer st; + try + { + BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile)); + while ((s = br.readLine()) != null) + { + st = new StringTokenizer(s, " "); + float[] tapList = new float[6]; + int key = Integer.parseInt(st.nextToken()); // tap number + tapList[0] = Float.parseFloat(st.nextToken()); // lot id + tapList[3] = Float.parseFloat(st.nextToken()); // ptype + tapList[1] = Float.parseFloat(st.nextToken()); // taz + tapList[2] = (Math.max(Float.parseFloat(st.nextToken()), 15)) * 2.5f; // lot capacity + tapList[4] = Float.parseFloat(st.nextToken()); // distance from lot to TAP + tapList[5] = Float.parseFloat(st.nextToken()); /* Transit mode {4: CR, + 5: LRT, + 6: BRT, + 7: BRT, + 8:Limited Express Bus, + 9:Express bus, + 10: local}*/ + + if (map.get(key) == null) + { + List newList = new ArrayList(); + newList.add(tapList); + map.put(key, newList); + } else + { + map.get(key).add(tapList); + } + } + br.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + populateTap(map); + } + + /** + * The function will get a TreeMap having with iTaps as keys and [][4] + * arrays. For each iTap in the TreeMap it will transpose the [][4] array + * associated with it and attach it to the this.tapParkingInfo[key] element. + * + * @param map + * - a TreeMap containing all the records of the + * tapParkingInfo.ptype file + */ + private void populateTap(TreeMap> map) + { + + this.tapParkingInfo = new float[map.lastKey() + 1][6][]; + Iterator iterKeys = map.keySet().iterator(); + while (iterKeys.hasNext()) + { + int key = iterKeys.next(); + int numElem = map.get(key).size(); + for (int i = 0; i < 6; i++) + this.tapParkingInfo[key][i] = new float[numElem]; + for (int i = 0; i < numElem; i++) + { + for (int j = 0; j < 6; j++) + { + this.tapParkingInfo[key][j][i] = map.get(key).get(i)[j]; + } + } + } + } + + // TODO: test this. + public void intializeLotUse() + { + + float maxLotId = 0; + for (int i = 0; i < tapParkingInfo.length; i++) + { + float[] lotIds = tapParkingInfo[i][0]; + if (lotIds != null) + { + for (int j = 0; j < tapParkingInfo[i][0].length; j++) + { + if (maxLotId < tapParkingInfo[i][0][j]) maxLotId = tapParkingInfo[i][0][j]; + + } + } + } + + lotUse = new int[(int) maxLotId + 1]; + } + + /** + * Set the array of tap numbers (taps[]), indexed at 1. + * + * @param rb + * A Resourcebundle with skims.path and tap.skim.file properties. + */ + public void getTapList(HashMap rbMap) + { + ArrayList tapList = new ArrayList(); + + File mgraWlkTapTimeFile = new File(rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT), + rbMap.get(SandagWalkPathChoiceLogsumMatrixApplication.WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY)); + Map> mgraWlkTapList = new HashMap<>(); //mgra -> tap -> [board dist,alight dist] + String s; + try ( BufferedReader br = new BufferedReader(new FileReader(mgraWlkTapTimeFile))) + { + // read the first data file line containing column names + s = br.readLine(); + + // read the data records + while ((s = br.readLine()) != null) + { + StringTokenizer st = new StringTokenizer(s, ","); + int mgra = Integer.parseInt(st.nextToken().trim()); + int tap = Integer.parseInt(st.nextToken().trim()); + if (tap > maxTap) maxTap = tap; + if (!tapList.contains(tap)) tapList.add(tap); + } + } catch (IOException e) { + logger.error(e); + throw new RuntimeException(e); + } + + // read taps from park-and-ride file + File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") + + Util.getStringValueFromPropertyMap(rbMap, "tap.ptype.file")); + + try (BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile))) + { + + while ((s = br.readLine()) != null) + { + StringTokenizer st = new StringTokenizer(s, " "); + int tap = Integer.parseInt(st.nextToken()); // tap number + if (!tapList.contains(tap)) tapList.add(tap); + } + br.close(); + } catch (IOException e) { + logger.error(e); + throw new RuntimeException(e); + } + + Collections.sort(tapList); + // now go thru the array of ArrayLists and convert the lists to arrays + // and + taps = new int[tapList.size() + 1]; + + for (int i = 0; i < tapList.size(); ++i) + taps[i + 1] = tapList.get(i); + + } + + public int getLotUse(int lotId) + { + return lotUse[lotId]; + } + + public void printStats() + { + /* + * logger.info("Tap 561 is in zone: " + tapParkingInfo[561][1][0]); + * logger.info("Tap 298 lot capacity: " + tapParkingInfo[298][2][0]); + */ + } + + public int getTazForTap(int tap) + { + return (int) tapParkingInfo[tap][1][0]; + } + + public static TapDataManager getInstance(HashMap rbMap) + { + if (instance == null) { + synchronized (LOCK) { + if (instance == null) { + instance = new TapDataManager(rbMap); + } + } + } + return instance; + } + + public float[][][] getTapParkingInfo() + { + if (instance != null) + { + return this.tapParkingInfo; + } else + { + throw new RuntimeException(); + } + } + + public float getCarToStationWalkTime(int tap) + { + return 0.0f; + } + + public float getEscalatorTime(int tap) + { + return 0.0f; + } + + public int[] getTaps() + { + return taps; + } + + public static void main(String[] args) + { + ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); + + TapDataManager tdm = TapDataManager.getInstance(ResourceUtil + .changeResourceBundleIntoHashMap(rb)); + tdm.printStats(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java new file mode 100644 index 0000000..ab697d1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java @@ -0,0 +1,924 @@ +package org.sandag.abm.modechoice; + +import java.io.BufferedReader; +import java.io.File; +import java.io.FileReader; +import java.io.IOException; +import java.io.Serializable; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; +import java.util.ResourceBundle; +import java.util.StringTokenizer; +import java.util.TreeSet; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.Modes.AccessMode; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +/** + * This class is used for storing the TAZ data for the mode choice model. + * + * @author Christi Willison + * @version Sep 2, 2008 + *

+ * Created by IntelliJ IDEA. + */ +public final class TazDataManager + implements Serializable +{ + + protected transient Logger logger = Logger.getLogger(TazDataManager.class); + private static TazDataManager instance; + public int[] tazs; + protected int[] tazsOneBased; + + // arrays for the MGRA and TAZ fields from the MGRA table data file. + private int[] mgraTableMgras; + private int[] mgraTableTazs; + + // list of TAZs in numerical order + public TreeSet tazSet = new TreeSet(); + public int maxTaz; + + private int nTazsWithMgras; + private int[][] tazMgraArray; + // private int[][] tazXYCoordArray; + private float[] tazDestinationTerminalTime; + private float[] tazOriginTerminalTime; + // private int[] tazSuperDistrict; + // private int[] pmsa; + // private int[] cmsa; + // This might be a poor name for this array. + // Please change if you know better. + private int[] tazAreaType; + // they are being read from same file but might be different - + // [tdz][id,time,dist][tap position] + // in the future. + private float[][][] tazParkNRideTaps; + // They are floats because they store time and distance + private float[][][] tazKissNRideTaps; + + // added from tdz manager + private int[] tazParkingType; + + /** + * Get an array of tazs, indexed sequentially from 0 + * + * @return Taz array indexed from 0 + */ + public int[] getTazs() + { + return tazs; + } + + /** + * Get an array of tazs, indexed sequentially from 1 + * + * @return taz array indexed from 1 + */ + public int[] getTazsOneBased() + { + return tazsOneBased; + } + + private TazDataManager(HashMap rbMap) + { + System.out.println("I'm the TazDataManager"); + + // read the MGRA data file into a TableDataSet and get the MGRA and TAZ + // fields from it for setting TAZ correspondence. + readMgraTableData(rbMap); + setTazMgraCorrespondence(); + readTazTerminalTimeCorrespondence(rbMap); + readPnRTapsInfo(rbMap); + + //printTazStats(); + } + + /** + * This method reads in the taz.tdz file which has 2 columns. The first + * column is the taz and the second column is corresponding tdz. The + * correspondence will be stored in the tdz class. The only data captured + * here is the list of TAZs. + * + * This method will also set the maxTaz value. + * + * @param rb + * the properties file that lists the taz.tdz file and the + * generic.path. + */ + private void readTazs(HashMap rbMap) + { + File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") + + Util.getStringValueFromPropertyMap(rbMap, "taz.tdz.correspondence.file")); + String s; + int taz; + StringTokenizer st; + try + { + BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile)); + while ((s = br.readLine()) != null) + { + st = new StringTokenizer(s, " "); + taz = Integer.parseInt(st.nextToken()); + tazSet.add(taz); + st.nextToken(); + } + br.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + maxTaz = tazSet.last(); + tazs = new int[tazSet.size()]; + tazsOneBased = new int[tazSet.size() + 1]; + int i = 0; + for (Integer tazNumber : tazSet) + { + tazs[i] = tazNumber; + tazsOneBased[i + 1] = tazNumber; + ++i; + } + + } + + /** + * This method will set the TAZ/MGRA correspondence. Two columns from the + * MGRA data table are used. The first column is the MGRA and the second + * column is the TAZ. The goal of this method is to populate the + * tazMgraArray array and the tazs treeset, plus set maxTaz. + * + */ + private void setTazMgraCorrespondence() + { + + HashMap> tazMgraMap = new HashMap>(); + + int mgra; + int taz; + + for (int i = 0; i < mgraTableMgras.length; i++) + { + + mgra = mgraTableMgras[i]; + taz = mgraTableTazs[i]; + if (!tazSet.contains(taz)) tazSet.add(taz); + + maxTaz = Math.max(taz, maxTaz); + + if (!tazMgraMap.containsKey(taz)) + { + ArrayList tazMgraList = new ArrayList(); + tazMgraList.add(mgra); + tazMgraMap.put(taz, tazMgraList); + } else + { + ArrayList tazMgraList = tazMgraMap.get(taz); + tazMgraList.add(mgra); + } + + } + + // now go thru the array of ArrayLists and convert the lists to arrays + // and + // store in the class variable tazMgraArrays. + tazMgraArray = new int[maxTaz + 1][]; + for (Iterator it = tazMgraMap.entrySet().iterator(); it.hasNext();) + { // elements + // in + // the + // array + // of + // arraylists + Map.Entry entry = (Map.Entry) it.next(); + taz = (Integer) entry.getKey(); + ArrayList tazMgraList = (ArrayList) entry.getValue(); + if (tazMgraList != null) + { // if the list isn't null + tazMgraArray[taz] = new int[tazMgraList.size()]; // initialize + // the class + // variable + for (int j = 0; j < tazMgraList.size(); j++) + tazMgraArray[taz][j] = (Integer) tazMgraList.get(j); + nTazsWithMgras++; + } + } + tazs = new int[tazSet.size()]; + + tazsOneBased = new int[tazSet.size() + 1]; + int i = 0; + for (Integer tazNumber : tazSet) + { + tazs[i] = tazNumber; + tazsOneBased[i + 1] = tazNumber; + ++i; + } + } + + /** + * This method will initialize the class variable tazSuperDistrict. The + * taz.district file has 2 columns, the first is the taz and the second is + * the superdistrict + * + * @param rb + * the resource bundle that specifies the taz.district file and + * the generic.path public void + * readTazDistrictCorrespondence(HashMap rbMap) { + * tazSuperDistrict = new int[maxTaz + 1]; File tazTdzCorresFile + * = new File(Util.getStringValueFromPropertyMap(rbMap, + * "generic.path") + Util.getStringValueFromPropertyMap(rbMap, + * "taz.district.correspondence.file")); String s; int taz; int + * sd; StringTokenizer st; try { BufferedReader br = new + * BufferedReader(new FileReader(tazTdzCorresFile)); while ((s = + * br.readLine()) != null) { st = new StringTokenizer(s, " "); + * taz = Integer.parseInt(st.nextToken()); sd = + * Integer.parseInt(st.nextToken()); tazSuperDistrict[taz] = sd; + * } br.close(); } catch (IOException e) { e.printStackTrace(); } + * } + */ + + /** + * This method will read the zone.avrzone file and store the location area + * (0-3) for each taz. + * + * + * @param rb + * - resourceBundle That specifies the zone.avrzone file and the + * generic.path private void + * readZoneAvrZoneCorrespondence(HashMap rbMap) { + * File zoneAvrZoneCorresFile = new + * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") + * + Util.getStringValueFromPropertyMap(rbMap, + * "taz.avrzone.correspondence.file")); tazAreaType = new + * int[maxTaz + 1]; + * + * // read the file to get the location area (0 - 3) for each TDZ + * String s; int taz; StringTokenizer st; int location; try { + * BufferedReader br = new BufferedReader(new + * FileReader(zoneAvrZoneCorresFile)); while ((s = br.readLine()) + * != null) { st = new StringTokenizer(s, " "); taz = + * Integer.parseInt(st.nextToken()); location = + * Integer.parseInt(st.nextToken()); tazAreaType[taz] = location; + * } br.close(); } catch (IOException e) { e.printStackTrace(); } + * + * } + */ + + /** + * This method reads in the zone.pmsa file which has 2 columns. The first + * column is the taz and the second column is corresponding pmsa. The + * correspondence will be stored in the pmsa list. The only data captured + * here is the list of pmsas. + * + * This method will also set the maxTaz value. + * + * @param rb + * the properties file that lists the taz.tdz file and the + * generic.path private void readZonePMSA(HashMap + * rbMap) { + * + * pmsa = new int[maxTaz + 1]; File zonePmsaFileName = new + * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") + * + Util.getStringValueFromPropertyMap(rbMap, "taz.pmsa.file")); + * String s; int taz; int tazPmsa; StringTokenizer st; try { + * BufferedReader br = new BufferedReader(new + * FileReader(zonePmsaFileName)); // BufferedReader br = new + * BufferedReader(new // + * FileReader("/Users/michalis/Documents/Fortran2Java/data/zone.pmsa" + * )); while ((s = br.readLine()) != null) { st = new + * StringTokenizer(s, " "); taz = + * Integer.parseInt(st.nextToken()); tazPmsa = + * Integer.parseInt(st.nextToken()); pmsa[taz] = tazPmsa; } + * br.close(); } catch (IOException e) { e.printStackTrace(); } + * + * } + */ + + /** + * This method reads in the zone.cmsa file which has 2 columns. The first + * column is the taz and the second column is corresponding cmsa. The + * correspondence will be stored in the cmsa list. The only data captured + * here is the list of cmsas. + * + * This method will also set the maxTaz value. + * + * @param rb + * the properties file that lists the zone.cmsa file and the + * generic.path + * + * private void readZoneCMSA(HashMap rbMap) { + * + * cmsa = new int[maxTaz + 1]; File zoneCmsaFile = new + * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") + * + Util.getStringValueFromPropertyMap(rbMap, "taz.cmsa.file")); + * String s; int taz; int tazCmsa; StringTokenizer st; try { + * BufferedReader br = new BufferedReader(new + * FileReader(zoneCmsaFile)); // BufferedReader br = new + * BufferedReader(new // + * FileReader("/Users/michalis/Documents/Fortran2Java/data/zone.cmsa" + * )); while ((s = br.readLine()) != null) { st = new + * StringTokenizer(s, " "); taz = + * Integer.parseInt(st.nextToken()); tazCmsa = + * Integer.parseInt(st.nextToken()); cmsa[taz] = tazCmsa; } + * br.close(); } catch (IOException e) { e.printStackTrace(); } + * + * } + * */ + + /** + * This method will read the zone.term file and store the terminal time for + * each taz. + * + * @param rb + * the properties file that lists the zone.term file and the + * scenario.path + */ + private void readTazTerminalTimeCorrespondence(HashMap rbMap) + { + File tdzTerminalTimeCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, + "scenario.path") + + Util.getStringValueFromPropertyMap(rbMap, "taz.terminal.time.file")); + + tazDestinationTerminalTime = new float[maxTaz + 1]; + tazOriginTerminalTime = new float[maxTaz + 1]; + + // read the file to get the terminal time for each TDZ + String s; + int taz; + StringTokenizer st; + float terminalTime; + try + { + BufferedReader br = new BufferedReader(new FileReader(tdzTerminalTimeCorresFile)); + while ((s = br.readLine()) != null) + { + st = new StringTokenizer(s, " "); + taz = Integer.parseInt(st.nextToken()); + terminalTime = Float.parseFloat(st.nextToken()); + tazDestinationTerminalTime[taz] = terminalTime; + tazOriginTerminalTime[taz] = terminalTime; + } + br.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + + } + + /** + * This method will read the zone.pterm file and store the production + * terminal time for each tdz. + * + * @param rb + * the properties file that lists the zone.pterm file and the + * scenario.path + * + * + * private void + * readTazProductionTerminalTimeCorrespondence(HashMap rbMap) { File tdzProductionTerminalTimeCorresFile = + * new File(Util.getStringValueFromPropertyMap( rbMap, + * "scenario.path") + Util.getStringValueFromPropertyMap(rbMap, + * "taz.prod.terminal.time.file")); + * + * tazOriginTerminalTime = new float[maxTaz + 1]; + * + * // read the file to get the production terminal time for each + * TDZ String s; int taz; StringTokenizer st; float + * productionTerminalTime; try { BufferedReader br = new + * BufferedReader(new FileReader( + * tdzProductionTerminalTimeCorresFile)); while ((s = + * br.readLine()) != null) { st = new StringTokenizer(s, " "); + * taz = Integer.parseInt(st.nextToken()); productionTerminalTime + * = Float.parseFloat(st.nextToken()); tazOriginTerminalTime[taz] + * = productionTerminalTime; } br.close(); } catch (IOException + * e) { e.printStackTrace(); } + * + * } + */ + + /** + * This method will read the zone.park file and store the parking type for + * each taz. Only types 2 - 5 are given. Rest assumed to be type 1. + * + * @param rb + * the properties file that lists the taz.parkingtype.file and + * the scenario.path private void + * readTAZParkingTypeCorrespondence(HashMap + * rbMap) { File tazParkingTypeCorresFile = new + * File(Util.getStringValueFromPropertyMap(rbMap, + * "scenario.path") + Util.getStringValueFromPropertyMap(rbMap, + * "taz.parkingtype.file")); + * + * tazParkingType = new int[maxTaz + 1]; + * Arrays.fill(tazParkingType, 1); + * + * // read the file to get the parking type (2 - 5) for each TAZ + * String s; int taz; StringTokenizer st; int parkingType; try { + * BufferedReader br = new BufferedReader(new + * FileReader(tazParkingTypeCorresFile)); while ((s = + * br.readLine()) != null) { st = new StringTokenizer(s, " "); + * taz = Integer.parseInt(st.nextToken()); parkingType = + * Integer.parseInt(st.nextToken()); tazParkingType[taz] = + * parkingType; } br.close(); } catch (IOException e) { + * e.printStackTrace(); } + * + * } + */ + + /** + * This method read in the access061.prp file that lists the taz and the # + * of taps that have drive access. Then the taps are listed along with the + * time and the distance to those taps from the taz. + * + * @param rb + * the properties file that lists the taz.driveaccess.taps.file + * and the scenario.path + */ + public void readPnRTapsInfo(HashMap rbMap) + { + File tdzDATapFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") + + Util.getStringValueFromPropertyMap(rbMap, "taz.driveaccess.taps.file")); + tazParkNRideTaps = new float[maxTaz + 1][3][]; // tapId, time, distance + tazKissNRideTaps = new float[maxTaz + 1][3][]; // tapId, time, distance + + String s, s1; + StringTokenizer st, st1; + int taz; + int tapId; + float tapTime; + float tapDist; + + //Shove into hash at first, then decompose into float array + HashMap< Integer, HashMap> tazTapMap = new HashMap>(); + + try + { + BufferedReader br = new BufferedReader(new FileReader(tdzDATapFile)); + while ((s = br.readLine()) != null) + { + st = new StringTokenizer(s, ","); + + taz = Integer.parseInt(st.nextToken()); + tapId = Integer.parseInt(st.nextToken()); + tapTime = Float.parseFloat(st.nextToken()); + tapDist = Float.parseFloat(st.nextToken()); + + if(tazTapMap.get(taz) != null){ + + HashMap< Integer, float[] > tapVals = tazTapMap.get(taz); + if(tapVals.get(tapId) != null){ + //something wrong, since there should only be unique taps for a taz + throw new RuntimeException("There should not be any duplicate TAPs for a TAZ"); + }else{ + float[] timeDist = new float[2]; + timeDist[0] = tapTime; + timeDist[1] = tapDist; + tapVals.put(tapId, timeDist); + } + + }else{ + HashMap< Integer, float[] > tapVals = new HashMap(); + float[] timeDist = new float[2]; + timeDist[0] = tapTime; + timeDist[1] = tapDist; + tapVals.put(tapId, timeDist); + tazTapMap.put(taz, tapVals); + } + } + + Iterator it = tazTapMap.entrySet().iterator(); + while (it.hasNext()) { + Map.Entry pair = (Map.Entry)it.next(); + taz = (int) pair.getKey(); + HashMap< Integer, float[] > tapVals = tazTapMap.get(taz); + int nTaps = tapVals.keySet().size(); + + tazParkNRideTaps[taz][0] = new float[nTaps]; + tazParkNRideTaps[taz][1] = new float[nTaps]; + tazParkNRideTaps[taz][2] = new float[nTaps]; + tazKissNRideTaps[taz][0] = new float[nTaps]; + tazKissNRideTaps[taz][1] = new float[nTaps]; + tazKissNRideTaps[taz][2] = new float[nTaps]; + + Iterator it2 = tapVals.entrySet().iterator(); + int i = 0; + while (it2.hasNext()) { + Map.Entry pair2 = (Map.Entry)it2.next(); + tapId = (int) pair2.getKey(); + float[] vals = (float[]) pair2.getValue(); + tazParkNRideTaps[taz][0][i] = tapId; + tazParkNRideTaps[taz][1][i] = vals[0]; + tazParkNRideTaps[taz][2][i] = vals[1]; + tazKissNRideTaps[taz][0][i] = tapId; + tazKissNRideTaps[taz][1][i] = vals[0]; + tazKissNRideTaps[taz][2][i] = vals[1]; + i++; + } + } + + br.close(); + } catch (IOException e) + { + e.printStackTrace(); + } + + } + + /** + * This method will return the Area Type (Location?) for the TAZ. + * + * + * @param taz + * - TAZ that AreaType is wanted for. + * @return area type that the taz corresponds to + */ + public int getTAZAreaType(int taz) + { + return tazAreaType[taz]; + } + + /** + * Write taz data manager data to logger for debugging. + * + */ + public void printTazStats() + { + + logger.info("Number of TAZs: " + tazSet.size()); + logger.info("Max TAZ: " + maxTaz); + logger.info("Number of TAZs with MGRAs: " + nTazsWithMgras); + } + + /** + * Get a static instance of the Taz Data Manager. One is created if it + * doesn't exist already. + * + * @param rb + * A resourcebundle with properties for the TazDataManager. + * @return A static instance of this class. + */ + public static TazDataManager getInstance(HashMap rbMap) + { + if (instance == null) + { + instance = new TazDataManager(rbMap); + return instance; + } else return instance; + } + + /** + * This method should only be used after the getInstance(HashMap rbMap) method has been called since the rbMap is needed to read + * in all the data and populate the object. This method will return the + * instance that has already been populated. + * + * @return instance + * @throws RuntimeException + */ + public static TazDataManager getInstance() throws RuntimeException + { + if (instance == null) + { + throw new RuntimeException( + "Must instantiate TazDataManager with the getInstance(rbMap) method first"); + } else + { + return instance; + } + } + + /** + * Get the number of TAZs with MGRAs. + * + * @return The number of TAZs with MGRAs. + */ + public int getNTazsWithMgras() + { + if (instance != null) + { + return nTazsWithMgras; + } else + { + throw new RuntimeException(); + } + } + + public int[][] getTazMgraArray() + { + if (instance != null) + { + return tazMgraArray; + } else + { + throw new RuntimeException(); + } + } + + /** + * Return the list of MGRAs within this TAZ. + * + * @param taz + * The TAZ number + * @return An array of MGRAs within the TAZ. + */ + public int[] getMgraArray(int taz) + { + if (instance != null) + { + return tazMgraArray[taz]; + } else + { + throw new RuntimeException(); + } + } + + /* + * public int[][] getTazXYCoordArray() { if (instance != null) { return + * tazXYCoordArray; } else { throw new RuntimeException(); } } + * + * public int[] getTazSuperDistrict() { if (instance != null) { return + * tazSuperDistrict; } else { throw new RuntimeException(); } + * + * } + * + * public int[] getPmsa() { if (instance != null) { return this.pmsa; } else + * { throw new RuntimeException(); } } + * + * public int[] getCmsa() { if (instance != null) { return this.cmsa; } else + * { throw new RuntimeException(); } } + */ + + /** + * This method will return the Parking Type for the TAZ. + * + * @param taz + * - TAZ that Parking Type is wanted for. + * @return Parking Type + */ + public int getTazParkingType(int taz) + { + return tazParkingType[taz]; + } + + /** + * Get the list of Park and Ride Taps for this TAZ. + * + * @param Taz + * @return An array of PNR taps for the TAZ. + */ + public int[] getParkRideTapsForZone(int taz) + { + if (tazParkNRideTaps[taz][0] == null) return null; + + int[] parkTaps = new int[tazParkNRideTaps[taz][0].length]; + for (int i = 0; i < tazParkNRideTaps[taz][0].length; i++) + { + parkTaps[i] = (int) tazParkNRideTaps[taz][0][i]; + } + return parkTaps; + } + + /** + * Get the list of Kiss and Ride Taps for this TAZ. + * + * @param Taz + * @return An array of KNR taps for the TAZ. + */ + public int[] getKissRideTapsForZone(int taz) + { + if (tazKissNRideTaps[taz][0] == null) return null; + int[] kissTaps = new int[tazKissNRideTaps[taz][0].length]; + for (int i = 0; i < tazKissNRideTaps[taz][0].length; i++) + { + kissTaps[i] = (int) tazKissNRideTaps[taz][0][i]; + } + return kissTaps; + } + + public int[] getParkRideOrKissRideTapsForZone(int taz, AccessMode aMode) + { + + switch (aMode) + { + case WALK: + return null; + case PARK_N_RIDE: + return getParkRideTapsForZone(taz); + case KISS_N_RIDE: + return getKissRideTapsForZone(taz); + default: + throw new RuntimeException( + "Error trying to get ParkRideOrKissRideTaps for unknown access mode: " + + aMode); + } + } + + /** + * Get the position of the tap in the taz tap array. + * + * @param taz + * The taz to lookup + * @param tap + * The tap to lookup + * @param aMode + * The access mode + * @return The position of the tap in the taz array. -1 is returned if it is + * an invalid tap for the taz. + */ + public int getTapPosition(int taz, int tap, AccessMode aMode) + { + + int[] taps = getParkRideOrKissRideTapsForZone(taz, aMode); + + if (taps == null) return -1; + + for (int i = 0; i < taps.length; ++i) + if (taps[i] == tap) return i; + + return -1; + + } + + /** + * Get the taz to tap time in minutes. + * + * @param taz + * Origin/Production TAZ + * @param pos + * Position of the TAP in this TAZ + * @param mode + * Park and Ride or Kiss and Ride + * @return The TAZ to TAP time in minutes. + */ + public float getTapTime(int taz, int pos, AccessMode aMode) + { + // only expecting this method for Park and Ride and Kiss and Ride modes. + switch (aMode) + { + case PARK_N_RIDE: + return (tazParkNRideTaps[taz][1][pos]); + case KISS_N_RIDE: + return (tazKissNRideTaps[taz][1][pos]); + default: + throw new RuntimeException( + "Error trying to get ParkRideOrKissRideTaps for invalid access mode: " + + aMode); + } + } + + /** + * Get the taz to tap distance in miles. + * + * @param taz + * Origin/Production TAZ + * @param pos + * Position of the TAP in this TAZ + * @param mode + * Park and Ride or Kiss and Ride + * @return The TAZ to TAP distance in miles. + */ + public float getTapDist(int taz, int pos, AccessMode aMode) + { + // only expecting this method for Park and Ride and Kiss and Ride modes. + switch (aMode) + { + case PARK_N_RIDE: + return (tazParkNRideTaps[taz][2][pos]); + case KISS_N_RIDE: + return (tazKissNRideTaps[taz][2][pos]); + default: + throw new RuntimeException( + "Error trying to get ParkRideOrKissRideTaps for invalid access mode: " + + aMode); + } + } + + /** + * Get the time from the TAZ to the TAP in minutes. + * + * @param taz The origin TAZ + * @param tap The destination TAP + * @param aMode The access model (PNR or KNR) + * @return The time in minutes, or -1 if there isn't an access link from the TAZ to the TAP. + */ + public float getTimeToTapFromTaz(int taz, int tap, AccessMode aMode){ + + int btapPosition = getTapPosition(taz,tap,aMode); + float time; + + if(btapPosition==-1){ + logger.info("Bad tap position for " + (aMode==Modes.AccessMode.PARK_N_RIDE ? "PNR" : "KNR") +" access board tap"); + return -1; + }else{ + time = getTapTime(taz,btapPosition,Modes.AccessMode.PARK_N_RIDE); + } + + return time; + + } + /** + * Get the distance from the TAZ to the TAP in miles. + * + * @param taz The origin TAZ + * @param tap The destination TAP + * @param aMode The access model (PNR or KNR) + * @return The distance in miles, or -1 if there isn't an access link from the TAZ to the TAP. + */ + public float getDistanceToTapFromTaz(int taz, int tap, AccessMode aMode){ + + int btapPosition = getTapPosition(taz,tap,aMode); + float distance; + + if(btapPosition==-1){ + logger.info("Bad tap position for " + (aMode==Modes.AccessMode.PARK_N_RIDE ? "PNR" : "KNR") +" access board tap"); + return -1; + }else{ + distance = getTapDist(taz,btapPosition,Modes.AccessMode.PARK_N_RIDE); + } + + return distance; + + } /** + * Returns the max TAZ value + * + * @return the max TAZ value + */ + public int getMaxTaz() + { + return maxTaz; + } + + private void readMgraTableData(HashMap rbMap) + { + + // get the mgra data table from one of these UECs. + String projectPath = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String mgraFile = rbMap.get(MgraDataManager.PROPERTIES_MGRA_DATA_FILE); + mgraFile = projectPath + mgraFile; + + TableDataSet mgraTableDataSet = null; + try + { + OLD_CSVFileReader reader = new OLD_CSVFileReader(); + mgraTableDataSet = reader.readFile(new File(mgraFile)); + } catch (IOException e) + { + logger.error("problem reading mgra data table for TazDataManager.", e); + System.exit(1); + } + + // get 0-based arrays from the specified fields in the MGRA table + mgraTableMgras = mgraTableDataSet.getColumnAsInt(MgraDataManager.MGRA_FIELD_NAME); + mgraTableTazs = mgraTableDataSet.getColumnAsInt(MgraDataManager.MGRA_TAZ_FIELD_NAME); + + } + + /** + * Test an instance of the class by instantiating and reporting. + * + * @param args + * [0] The properties file name/path. + */ + public static void main(String[] args) + { + ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); + + TazDataManager tdm = TazDataManager.getInstance(ResourceUtil + .changeResourceBundleIntoHashMap(rb)); + + } + + /** + * This method will return the Origin Terminal Time for the TDZ. + * + * @param taz + * - TAZ that Terminal Time is wanted for. + * @return Origin Terminal Time + */ + public float getOriginTazTerminalTime(int taz) + { + return tazOriginTerminalTime[taz]; + } + + /** + * This method will return the Destination Terminal Time for the TDZ. + * + * @param taz + * - TAZ that Terminal Time is wanted for. + * @return Destination Terminal Time + */ + public float getDestinationTazTerminalTime(int taz) + { + return tazDestinationTerminalTime[taz]; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java new file mode 100644 index 0000000..6e102b6 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java @@ -0,0 +1,465 @@ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.modechoice.Modes.AccessMode; + +import com.pb.common.calculator.VariableTable; +/** + * This class is used for ... + * + * @author Joel Freedman + * @version Mar 20, 2009 + *

+ * Created by IntelliJ IDEA. + */ +public class TransitDriveAccessDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TransitDriveAccessDMU.class); + + protected HashMap methodIndexMap; + + double driveTimeToTap; + double driveDistToTap; + double driveDistFromTap; + double driveTimeFromTap; + double OrigDestDistance; + double tapToMgraWalkTime; + double mgraToTapWalkTime; + double carToStationWalkTime; + double escalatorTime; + int accessMode; + int period; + int set; + + //default values for generic application + int applicationType = 0; + int personType = 1; + float ivtCoeff; + float costCoeff; + int joint = 0; //added for consistency with MTC + float valueOfTime = 0; //added for consistency with MTC + + + public TransitDriveAccessDMU() + { + setupMethodIndexMap(); + } + + /** + * Set the joint indicator. + * + * @param joint + */ + public void setTourCategoryIsJoint(int joint){ + this.joint = joint; + } + + /** + * Get the joint indicator. + * + * @return joint + */ + public int getTourCategoryIsJoint(){ + return joint; + } + + /** + * Set the value of time. + * + * @param the value of time + */ + public void setValueOfTime(float valueOfTime){ + this.valueOfTime = valueOfTime; + } + + /** + * Get the value of time. + * + * @return value of time. + */ + public float getValueOfTime(){ + return valueOfTime; + } + + + /** + * Get the walk time from the alighting TAP to the destination MGRA. + * + * @return The walk time from the alighting TAP to the destination MGRA. + */ + public double getTapMgraWalkTime() + { + return tapToMgraWalkTime; + } + + /** + * Set the walk time from the alighting TAP to the destination MGRA. + * + * @param walkTime The walk time from the alighting TAP to the destination MGRA. + */ + public void setTapMgraWalkTime(double walkTime) + { + tapToMgraWalkTime = walkTime; + } + + /** + * Get the walk time to the boarding TAP from the origin MGRA. + * + * @return The walk time from the origin MGRA to the boarding TAP. + */ + public double getMgraTapWalkTime() + { + return mgraToTapWalkTime; + } + + /** + * Set the walk time to the boarding TAP from the origin MGRA + * + * @param walkTime The walk time to the boarding TAP from the origin MGRA. + */ + public void setMgraTapWalkTime(double walkTime) + { + mgraToTapWalkTime = walkTime; + } + + /** + * Get the walk time from the lot to the station. + * + * @return The time in minutes. + */ + public double getCarToStationWalkTime() + { + return carToStationWalkTime; + } + + /** + * Set the walk time from the lot to the station. + * + * @param carToStationWalkTime The time in minutes. + */ + public void setCarToStationWalkTime(double carToStationWalkTime) + { + this.carToStationWalkTime = carToStationWalkTime; + } + + /** + * Get the time to get to the platform. + * + * @return The time in minutes. + */ + public double getEscalatorTime() + { + return escalatorTime; + } + + /** + * Set the time to get to the platform. + * + * @param escalatorTime The time in minutes. + */ + public void setEscalatorTime(double escalatorTime) + { + this.escalatorTime = escalatorTime; + } + + /** + * Get the access mode for this DMU. + * + * @return The access mode. + */ + public int getAccessMode() + { + return accessMode; + } + + /** + * Set the access mode for this DMU. + * + * @param accessMode The access mode. + */ + public void setAccessMode(int accessMode) + { + this.accessMode = accessMode; + } + + /** + * Get the drive time from the origin/production TDZ/TAZ to the TAP. + * + * @return The drive time in minutes. + */ + public double getDriveTimeToTap() + { + return driveTimeToTap; + } + + /** + * Set the drive time from the origin/production TDZ/TAZ to the TAP. + * + * @param driveTimeToTap The drive time in minutes. + */ + public void setDriveTimeToTap(double driveTimeToTap) + { + this.driveTimeToTap = driveTimeToTap; + } + + /** + * Get the drive distance from the origin/production TDZ/TAZ to the TAP. + * + * @return The drive distance in miles. + */ + public double getDriveDistToTap() + { + return driveDistToTap; + } + + /** + * Set the drive distance from the origin/production TDZ/TAZ to the TAP. + * + * @param driveDistToTap The drive distance in miles. + */ + public void setDriveDistToTap(double driveDistToTap) + { + this.driveDistToTap = driveDistToTap; + } + + /** + * Get the drive time from the TAP to the destination/attraction TDZ/TAZ. + * + * @return The drive time in minutes. + */ + public double getDriveTimeFromTap() + { + return driveTimeFromTap; + } + + /** + * Set the drive time from the TAP to the destination/attraction TDZ/TAZ. + * + * @param driveTime The drive time in minutes. + */ + public void setDriveTimeFromTap(double driveTime) + { + driveTimeFromTap = driveTime; + } + + /** + * Get the drive distance from the TAP to the destination/attraction TDZ/TAZ. + * + * @return The drive distance in miles. + */ + public double getDriveDistFromTap() + { + return driveDistFromTap; + } + + /** + * Set the drive distance from the TAP to the destination/attraction TDZ/TAZ. + * + * @param driveDist The drive distance in miles. + */ + public void setDriveDistFromTap(double driveDist) + { + driveDistFromTap = driveDist; + } + + public double getOrigDestDistance() { + return OrigDestDistance; + } + + public void setOrigDestDistance(double origDestDistance) { + OrigDestDistance = origDestDistance; + } + + public void setTOD(int period) { + this.period = period; + } + + public int getTOD() { + return period; + } + + public void setSet(int set) { + this.set = set; + } + + public int getSet() { + return set; + } + + + public void setApplicationType(int applicationType) { + this.applicationType = applicationType; + } + + public int getApplicationType() { + return applicationType; + } + + public void setPersonType(int personType) { + this.personType = personType; + } + + public int getPersonType() { + return personType; + } + + public void setIvtCoeff(float ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public void setCostCoeff(float costCoeff) { + this.costCoeff = costCoeff; + } + + public float getIvtCoeff() { + return ivtCoeff; + } + + public float getCostCoeff() { + return costCoeff; + } + + /** + * Log the DMU values. + * + * @param localLogger The logger to use. + */ + public void logValues(Logger localLogger) + { + + localLogger.info(""); + localLogger.info("Drive-Transit Auto Access DMU Values:"); + localLogger.info(""); + localLogger.info(String.format("Drive Time To Tap: %9.4f", driveTimeToTap)); + localLogger.info(String.format("Drive Dist To Tap: %9.4f", driveDistToTap)); + localLogger.info(String.format("Drive Time From Tap: %9.4f", driveTimeFromTap)); + localLogger.info(String.format("Drive Dist From Tap: %9.4f", driveDistFromTap)); + localLogger.info(String.format("TAP to MGRA walk time: %9.4f", tapToMgraWalkTime)); + localLogger.info(String.format("MGRA to TAP walk time: %9.4f", mgraToTapWalkTime)); + localLogger.info(String.format("Car to station walk time: %9.4f", carToStationWalkTime)); + localLogger.info(String.format("Escalator time: %9.4f", escalatorTime)); + localLogger.info(String.format("Period: %9s", period)); + localLogger.info(String.format("Set: %9s", set)); + localLogger.info(String.format("applicationType: %9s", applicationType)); + localLogger.info(String.format("personType: %9s", personType)); + localLogger.info(String.format("ivtCoeff %9.4f", ivtCoeff)); + localLogger.info(String.format("costCoeff %9.4f", costCoeff)); + localLogger.info(String.format("origDestDistance %9.4f, origDestDistance")); + localLogger.info(String.format("joint : %9s", joint)); + localLogger.info(String.format("value of time : %9.4f", valueOfTime)); + + + AccessMode[] accessModes = AccessMode.values(); + localLogger.info(String.format("Access Mode: %5s", accessModes[accessMode] + .toString())); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getAccessMode", 0); + methodIndexMap.put("getCarToStationWalkTime", 1); + methodIndexMap.put("getDriveDistToTap", 2); + methodIndexMap.put("getDriveTimeToTap", 3); + methodIndexMap.put("getDriveDistFromTap", 4); + methodIndexMap.put("getDriveTimeFromTap", 5); + methodIndexMap.put("getEscalatorTime", 6); + methodIndexMap.put("getTapMgraWalkTime", 7); + methodIndexMap.put("getMgraTapWalkTime", 8); + methodIndexMap.put("getTOD", 9); + methodIndexMap.put("getSet", 10); + + methodIndexMap.put("getApplicationType", 12); + methodIndexMap.put("getTourCategoryIsJoint", 13); + methodIndexMap.put("getPersonType", 14); + methodIndexMap.put("getIvtCoeff", 15); + methodIndexMap.put("getCostCoeff", 16); + methodIndexMap.put("getOrigDestDistance",17); + methodIndexMap.put("getTourCategoryIsJoint", 18); + methodIndexMap.put("getValueOfTime", 19); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getAccessMode(); + case 1: + return getCarToStationWalkTime(); + case 2: + return getDriveDistToTap(); + case 3: + return getDriveTimeToTap(); + case 4: + return getDriveDistFromTap(); + case 5: + return getDriveTimeFromTap(); + case 6: + return getEscalatorTime(); + case 7: + return getTapMgraWalkTime(); + case 8: + return getMgraTapWalkTime(); + case 9: + return getTOD(); + case 10: + return getSet(); + + case 12: + return getApplicationType(); + case 14: + return getPersonType(); + case 15: + return getIvtCoeff(); + case 16: + return getCostCoeff(); + case 17: + return getOrigDestDistance(); + case 18: + return getTourCategoryIsJoint(); + case 19: + return getValueOfTime(); + + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java new file mode 100644 index 0000000..781d4d1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java @@ -0,0 +1,332 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.modechoice; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.VariableTable; + +/** + * WalkDMU is the Decision-Making Unit class for the Walk-transit choice. The class + * contains getter and setter methods for the variables used in the WalkPathUEC. + * + * @author Joel Freedman + * @version 1.0, March, 2009 + * + */ +public class TransitWalkAccessDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(TransitWalkAccessDMU.class); + + protected HashMap methodIndexMap; + + double tapToMgraWalkTime; + double mgraToTapWalkTime; + double escalatorTime; + int period; + int set; + + //default values for generic application + int applicationType = 0; + int personType = 1; //defaults to full-time worker + float ivtCoeff; + float costCoeff; + int accessEgressMode=0; //this is called a walk-access DMU but it is used in the TAP-to-TAP UEC, so it is + //possible that it is being called for a drive-access path! + int joint = 0; //added for consistency with MTC + float valueOfTime = 0; //added for consistency with MTC + public TransitWalkAccessDMU() + { + setupMethodIndexMap(); + } + + + /** + * Set the joint indicator. + * + * @param joint + */ + public void setTourCategoryIsJoint(int joint){ + this.joint = joint; + } + + /** + * Get the joint indicator. + * + * @return joint + */ + public int getTourCategoryIsJoint(){ + return joint; + } + + /** + * Set the value of time. + * + * @param the value of time + */ + public void setValueOfTime(float valueOfTime){ + this.valueOfTime = valueOfTime; + } + + /** + * Get the value of time. + * + * @return value of time. + */ + public float getValueOfTime(){ + return valueOfTime; + } + + + /** + * Set the access/egress mode + * + * @param accessEgressMode + */ + public void setAccessEgressMode(int accessEgressMode){ + this.accessEgressMode = accessEgressMode; + } + + + /** + * Get the access/egress mode + * + * @return accessEgressMode + */ + public int getAccessEgressMode(){ + return accessEgressMode; + } + + + /** + * Get the time from the production/origin MGRA to the boarding TAP. + * + * @return The time from the production/origin MGRA to the boarding TAP. + */ + public double getMgraTapWalkTime() + { + return mgraToTapWalkTime; + } + + /** + * Set the time from the production/origin MGRA to the boarding TAP. + * + * @param walkTime The time from the production/origin MGRA to the boarding TAP. + */ + public void setMgraTapWalkTime(double walkTime) + { + this.mgraToTapWalkTime = walkTime; + } + + /** + * Get the time from the alighting TAP to the attraction/destination MGRA. + * + * @return The time from the alighting TAP to the attraction/destination MGRA. + */ + public double getTapMgraWalkTime() + { + return tapToMgraWalkTime; + } + + /** + * Set the time from the alighting TAP to the attraction/destination MGRA. + * + * @param walkTime The time from the alighting TAP to the attraction/destination + * MGRA. + */ + public void setTapMgraWalkTime(double walkTime) + { + this.tapToMgraWalkTime = walkTime; + } + + /** + * Get the time to get to the platform. + * + * @return The time in minutes. + */ + public double getEscalatorTime() + { + return escalatorTime; + } + + /** + * Set the time to get to the platform. + * + * @param escalatorTime The time in minutes. + */ + public void setEscalatorTime(double escalatorTime) + { + this.escalatorTime = escalatorTime; + } + + public void setTOD(int period) { + this.period = period; + } + + public int getTOD() { + return period; + } + + public void setSet(int set) { + this.set = set; + } + + public int getSet() { + return set; + } + + + public void setApplicationType(int applicationType) { + this.applicationType = applicationType; + } + + public int getApplicationType() { + return applicationType; + } + + + public void setPersonType(int personType) { + this.personType = personType; + } + + public int getPersonType() { + return personType; + } + + public void setIvtCoeff(float ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public void setCostCoeff(float costCoeff) { + this.costCoeff = costCoeff; + } + + public float getIvtCoeff() { + return ivtCoeff; + } + + public float getCostCoeff() { + return costCoeff; + } + + /** + * Log the DMU values. + * + * @param localLogger The logger to use. + */ + public void logValues(Logger localLogger) + { + + localLogger.info(""); + localLogger.info("Walk DMU Values:"); + localLogger.info(""); + localLogger.info(String.format("MGRA to TAP walk time: %9.4f", mgraToTapWalkTime)); + localLogger.info(String.format("TAP to MGRA walk time: %9.4f", tapToMgraWalkTime)); + localLogger.info(String.format("Escalator time: %9.4f", escalatorTime)); + localLogger.info(String.format("Period: %9s", period)); + localLogger.info(String.format("Set: %9s", set)); + localLogger.info(String.format("applicationType: %9s", applicationType)); + localLogger.info(String.format("personType: %9s", personType)); + localLogger.info(String.format("ivtCoeff : %9.4f", ivtCoeff)); + localLogger.info(String.format("costCoeff : %9.4f", costCoeff)); + localLogger.info(String.format("accessEgressMode : %9s", accessEgressMode)); + localLogger.info(String.format("joint : %9s", joint)); + localLogger.info(String.format("value of time : %9.4f", valueOfTime)); + + + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getEscalatorTime", 0); + methodIndexMap.put("getMgraTapWalkTime", 1); + methodIndexMap.put("getTapMgraWalkTime", 2); + methodIndexMap.put("getTOD", 3); + methodIndexMap.put("getSet", 4); + + methodIndexMap.put("getApplicationType", 6); + methodIndexMap.put("getPersonType", 8); + methodIndexMap.put("getIvtCoeff", 9); + methodIndexMap.put("getCostCoeff", 10); + methodIndexMap.put("getAccessEgressMode", 11); + methodIndexMap.put("getTourCategoryIsJoint", 12); + methodIndexMap.put("getValueOfTime", 13); + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getEscalatorTime(); + case 1: + return getMgraTapWalkTime(); + case 2: + return getTapMgraWalkTime(); + case 3: + return getTOD(); + case 4: + return getSet(); + case 6: + return getApplicationType(); + case 8: + return getPersonType(); + case 9: + return getIvtCoeff(); + case 10: + return getCostCoeff(); + case 11: + return getAccessEgressMode(); + case 12: + return getTourCategoryIsJoint(); + case 13: + return getValueOfTime(); + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java new file mode 100644 index 0000000..a757c38 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java @@ -0,0 +1,32 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.util.Properties; +import org.apache.log4j.Logger; + +public abstract class AbstractCsvExporter + implements IExporter +{ + private final File file; + private final IMatrixDao matrixDao; + private final String reportFolder = "report.path"; + + protected static final Logger LOGGER = Logger.getLogger(AbstractCsvExporter.class); + + public AbstractCsvExporter(Properties properties, IMatrixDao aMatrixDao, String aBaseFileName) + { + this.file = new File(properties.getProperty(reportFolder), aBaseFileName + ".csv"); + this.matrixDao = aMatrixDao; + } + + public IMatrixDao getMatrixDao() + { + return this.matrixDao; + } + + public File getFile() + { + return this.file; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java new file mode 100644 index 0000000..ecc5632 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java @@ -0,0 +1,304 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Properties; +import java.util.Set; + +import org.apache.log4j.Logger; + +import com.pb.common.datafile.DataTypes; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; + +public class CVMExporter { + private static final Logger logger = Logger.getLogger(DataExporter.class); + + public final String[] cvmPeriodNames = {"OE","AM","MD","PM","OL"}; + public final String[] periodNames = {"EA","AM","MD","PM","EV"}; + public final String[] cvmClassNames = {"L","M","I","H"}; + public final String[] skimTollClassNames = {"SOV_NT_H","TRK_L","TRK_M","TRK_H"}; + public final String[] skimNonTollClassNames = {"SOV_TR_H","TRK_L","TRK_M","TRK_H"}; + public final String[] nonTollSkims = {"TIME","DIST"}; + public final String[] tollSkims = {"TIME","DIST","TOLLCOST"}; + public final String[] cvmModeNames = {"NT","T"}; + public final String[] modelModeNames = {"GP","TOLL"}; + + public final String[] segmentNames = {"FA","RE","GO","IN","SV","WH","TH"}; + + protected Properties properties; + protected String projectPath; + protected String reportPath; + protected HashMap cvmSkimMap; + protected HashMap periodMap; //lookup cvm period, return model period + protected HashMap tollClassMap; //lookup cvm class, return toll skim class + protected HashMap nonTollClassMap; //lookup cvm class, return non-toll skim class + + protected HashMap modeMap; //lookup cvm mode, return model mode + + private final OMXMatrixDao mtxDao; + protected float autoOperatingCost; + + + public CVMExporter(Properties theProperties, OMXMatrixDao aMtxDao){ + this.properties = theProperties; + this.mtxDao = aMtxDao; + projectPath = properties.getProperty("scenario.path"); + reportPath = properties.getProperty("report.path"); + float fuelCost = new Float(properties.getProperty("aoc.fuel")); + float mainCost = new Float(properties.getProperty("aoc.maintenance")); + autoOperatingCost = (fuelCost + mainCost) * 0.01f; + + } + + private void createPeriodMap(){ + + periodMap = new HashMap(); + for(int i = 0; i(); + tollClassMap = new HashMap(); + for(int i = 0; i(); + for (int i = 0; i < cvmModeNames.length;++i) + modeMap.put(cvmModeNames[i], modelModeNames[i]); + } + + public void export(){ + createPeriodMap(); + createClassMap(); + createModeMap(); + readSkims(); + TableDataSet inputData = readCVMTrips(); + int totalRows = inputData.getRowCount(); + float[] timeCol = new float[totalRows]; + float[] distCol = new float[totalRows]; + float[] aocCol = new float[totalRows]; + float[] tollCol = new float[totalRows]; + + for(int row = 1; row<=totalRows;++row){ + + int otaz = (int) inputData.getValueAt(row, "I"); + int dtaz = (int) inputData.getValueAt(row, "J"); + String cvmPeriod = inputData.getStringValueAt(row,"OriginalTimePeriod"); + String cvmClass = inputData.getStringValueAt(row,"Mode"); + String cvmMode = inputData.getStringValueAt(row,"TripMode"); + + Matrix timeMatrix = null; + Matrix distMatrix = null; + Matrix tollMatrix = null; + + String modelPeriod = periodMap.get(cvmPeriod); + String modelClass = null; + if(cvmMode.equals("NT")){ + modelClass = nonTollClassMap.get(cvmClass); + + }else{ + modelClass = tollClassMap.get(cvmClass); + } + + timeMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"TIME"); + distMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"DIST"); + if(cvmSkimMap.containsKey(modelPeriod+"_"+modelClass+"_"+"TOLLCOST")) + tollMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"TOLLCOST"); + + timeCol[row-1] = timeMatrix.getValueAt(otaz, dtaz); + distCol[row-1] = distMatrix.getValueAt(otaz, dtaz); + aocCol[row-1] = distMatrix.getValueAt(otaz, dtaz) * autoOperatingCost; + if(tollMatrix != null) + tollCol[row-1] = tollMatrix.getValueAt(otaz, dtaz); + + } + + //append the columns + inputData.appendColumn(timeCol, "TIME"); + inputData.appendColumn(distCol, "DIST"); + inputData.appendColumn(aocCol, "AOC"); + inputData.appendColumn(tollCol, "TOLLCOST"); + + //write the data + TableDataSet.writeFile(reportPath+"cvm_trips.csv", inputData); + + } + + /** + * Read data into inputDataTable tabledataset. + * + */ + private TableDataSet readTableDataSet(String inputFile){ + + logger.info("Begin reading the data in file " + inputFile); + TableDataSet inputDataTable = null; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + inputDataTable = csvFile.readFile(new File(inputFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End reading the data in file " + inputFile); + + return inputDataTable; + } + + private void readSkims(){ + + cvmSkimMap = new HashMap(); + + for(String period: periodNames){ + + String fileName = "traffic_skims_"+period+".omx"; + + for(String nonTollClass: skimNonTollClassNames){ + + for(String skim:nonTollSkims){ + + String skimName = period+"_"+nonTollClass+"_"+skim; + + Matrix m = mtxDao.getMatrix(fileName, skimName); + cvmSkimMap.put(skimName, m); + + } + } + for(String tollClass: skimTollClassNames){ + for(String skim:tollSkims){ + String skimName = period+"_"+tollClass+"_"+skim; + + Matrix m = mtxDao.getMatrix(fileName, skimName); + cvmSkimMap.put(skimName, m); + + } + } + } + + + + + } + + /** + * Helper method to read in all the CVM files and concatenate into one TableDataSet. + * + * @return the concatenated data + */ + private TableDataSet readCVMTrips(){ + + int tables = 0; + String[] header = null; + int[] columnType = null; + HashMap> floatCols = new HashMap>(); + HashMap> stringCols = new HashMap>(); + + //first read all the data, and store arraylists of data in the two hashmaps + for(String segment: segmentNames){ + + for(String period:cvmPeriodNames){ + + String fileName = projectPath+"output\\Trip_"+segment+"_"+period+".csv"; + TableDataSet inData = readTableDataSet(fileName); + if(tables==0){ + columnType = inData.getColumnType(); + header = inData.getColumnLabels(); + } + ++tables; + for(int i = 0; i< inData.getColumnCount();++i){ + + String colName = header[i]; + if(columnType[i]==DataTypes.NUMBER){ + float[] data = inData.getColumnAsFloat(colName); + ArrayList colArray = null; + if(floatCols.containsKey(colName)) + colArray = floatCols.get(colName); + else + colArray = new ArrayList(); + + for(int j=0;j colArray = null; + if(stringCols.containsKey(colName)) + colArray = stringCols.get(colName); + else + colArray = new ArrayList(); + + for(int j=0;j keySet = floatCols.keySet(); + String[] colNames = new String[keySet.size()]; + float[][] data = null; + int colNumber=0; + for(String colName:keySet){ + colNames[colNumber] = colName; + ArrayList col = floatCols.get(colName); + if(colNumber==0){ + + data = new float[col.size()][colNames.length]; + } + for(int i = 0; i < col.size();++i){ + data[i][colNumber] = col.get(i); + } + ++colNumber; + } + + TableDataSet allTrips = TableDataSet.create(data,colNames); + + //logger.info("Created table data set with "+ allTrips.getColumnCount()+" columns"); + //String outputColNames = ""; + //for(String colName : allTrips.getColumnLabels()) + // outputColNames += (colName + " "); + //logger.info("Columns: "+outputColNames); + + //now append the string columns + keySet = stringCols.keySet(); + colNames = new String[keySet.size()]; + colNumber=0; + for(String colName:keySet){ + colNames[colNumber] = colName; + ArrayList col = stringCols.get(colName); + String[] stringData = new String[col.size()]; + stringData = col.toArray(stringData); + logger.info("Appending string column "+colName+" with "+stringData.length+" elements"); + allTrips.appendColumn(stringData, colName); + } + + //logger.info("Final table data set has "+ allTrips.getColumnCount()+" columns"); + //outputColNames = ""; + //for(String colName : allTrips.getColumnLabels()) + // outputColNames += (colName + " "); + //logger.info("Columns: "+outputColNames); + + return allTrips; + + + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java new file mode 100644 index 0000000..d7d595e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java @@ -0,0 +1,179 @@ +package org.sandag.abm.reporting; +import java.io.File; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.List; +import java.util.Properties; + +import org.apache.log4j.Logger; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +public class CVMScaler { + private static final Logger logger= Logger.getLogger(DataExporter.class); + protected Properties properties; + protected String projectPath; + protected String reportPath; + protected String [] lightscalers; + protected String [] mediumscalers; + protected String [] heavyscalers; + protected float lightshare; + protected float mediumshare; + protected float heavyshare; + + public CVMScaler(Properties theProperties){ + this.properties = theProperties; + projectPath = properties.getProperty("scenario.path"); + reportPath = properties.getProperty("report.path"); + String delims = "[,]"; + lightscalers = properties.getProperty("cvm.scale_light").split(delims); + mediumscalers = properties.getProperty("cvm.scale_medium").split(delims); + heavyscalers = properties.getProperty("cvm.scale_heavy").split(delims); + lightshare = new Float(properties.getProperty("cvm.share.light")); + mediumshare = new Float(properties.getProperty("cvm.share.medium")); + heavyshare = new Float(properties.getProperty("cvm.share.heavy")); + } + + public void scale(){ + logger.info("Running CVM scaler ... "); + String fileName = reportPath+"cvm_trips.csv"; + TableDataSet inData = readTableDataSet(fileName); + int totalRows = inData.getRowCount(); + String[] timeCol=new String[totalRows]; + + int tod=lightscalers.length; + float [] lscaler=new float[tod]; + float [] mscaler=new float[tod]; + float [] hscaler=new float[tod]; + + for(int i=0; i0) { + if (colname.equals("Mode")){ + value = value.replaceAll(vehicle, "I"); + }else if (colname.equals("TripTime")){ + value = value.replaceAll(":"+vehicle, ":I"); + } + + if (line_new==null) line_new = value; + else line_new = line_new + "," + value; + } + } + + //write existing lines + scaler = getScaler(scalerArray, str); + value_new = scaler * (1-share); + line = line + "," + Float.toString(value_new); + writer.println(line.trim()); + + //write new lines + if (share>0) { + value_new = scaler * (share); + line_new = line_new + "," + Float.toString(value_new); + writer.println(line_new.trim()); + } + } + + private float getScaler(float [] scalerArray, String str) { + float scaler = 1.0f; + + if(str.contains("_EA")){ + scaler = scalerArray[0]; + }else if(str.contains("_AM")){ + scaler = scalerArray[1]; + }else if(str.contains("_MD")){ + scaler = scalerArray[2]; + }else if(str.contains("_PM")){ + scaler = scalerArray[3]; + }else if(str.contains("_EV")){ + scaler = scalerArray[4]; + }else { + scaler = 1.0f; + } + + return scaler; + + } + + private TableDataSet readTableDataSet(String inputFile){ + + logger.info("Begin reading the data in file " + inputFile); + TableDataSet inputDataTable = null; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + inputDataTable = csvFile.readFile(new File(inputFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End reading the data in file " + inputFile); + + return inputDataTable; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java new file mode 100644 index 0000000..fb93a89 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java @@ -0,0 +1,22 @@ +package org.sandag.abm.reporting; + +public class CsvRow +{ + private final String row; + + public CsvRow(String[] values) + { + StringBuilder sb = new StringBuilder(32); + sb.append(values[0]); + for (int i = 1; i < values.length; i++) + sb.append(',').append(values[i]); + sb.append(System.lineSeparator()); + + row = sb.toString(); + } + + public String getRow() + { + return row; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java new file mode 100644 index 0000000..9ee01e2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java @@ -0,0 +1,86 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.nio.ByteBuffer; +import java.nio.channels.FileChannel; +import java.util.concurrent.BlockingQueue; +import org.apache.log4j.Logger; + +public class CsvWriterThread + implements Runnable +{ + private final BlockingQueue queue; + private final File file; + private final String[] header; + + private static final Logger LOGGER = Logger.getLogger(CsvWriterThread.class); + + private int maxBuffer = 1024 * 1024 * 1024; + private static final String ENCODING = "UTF-8"; + + public static final CsvRow POISON_PILL = new CsvRow(new String[] {"ALL_DONE"}); + + public CsvWriterThread(BlockingQueue aRowQueue, File anOutputLocation, String[] aHeader) + { + this.queue = aRowQueue; + this.file = anOutputLocation; + this.header = aHeader; + } + + public int getMaxBuffer() + { + return this.maxBuffer; + } + + public void setMaxBuffer(int aMaxBuffer) + { + this.maxBuffer = aMaxBuffer; + } + + @Override + public void run() + { + FileOutputStream outStream = null; + try + { + outStream = new FileOutputStream(file, false); + FileChannel outChannel = outStream.getChannel(); + ByteBuffer buffer = ByteBuffer.allocateDirect(getMaxBuffer()); + + CsvRow headerRow = new CsvRow(header); + buffer.put(headerRow.getRow().getBytes(ENCODING)); + + CsvRow row = null; + while ((row = queue.take()) != CsvWriterThread.POISON_PILL) + { + byte[] rowBytes = row.getRow().getBytes(ENCODING); + if ((buffer.position() + rowBytes.length) > buffer.capacity()) + { + buffer.flip(); + outChannel.write(buffer); + buffer.clear(); + } + buffer.put(rowBytes); + } + LOGGER.info("End of records found. Clearing Buffer and Writing Remains."); + buffer.flip(); + outChannel.write(buffer); + } catch (IOException | InterruptedException e) + { + LOGGER.fatal(e); + throw new RuntimeException(e); + } finally + { + if (null != outStream) try + { + outStream.close(); + LOGGER.info("CSV Writer Stream Closed."); + } catch (IOException e) + { + LOGGER.error(e); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java new file mode 100644 index 0000000..30bc37e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java @@ -0,0 +1,2688 @@ +package org.sandag.abm.reporting; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileInputStream; +import java.io.FileReader; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.rmi.RemoteException; +import java.text.DecimalFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.HashSet; +import java.util.LinkedHashMap; +import java.util.LinkedHashSet; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; +import java.util.Properties; +import java.util.Set; +import java.util.TreeSet; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.LinkedBlockingQueue; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.matrix.OMXMatrixWriter; +import com.pb.common.util.ResourceUtil; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +/** + * The {@code DataExporter} ... + * + * @author crf Started 9/20/12 8:36 AM + */ +public final class DataExporter +{ + private static final Logger LOGGER = Logger.getLogger(DataExporter.class); + + private static final String NUMBER_FORMAT_NAME = "NUMBER"; + private static final String STRING_FORMAT_NAME = "STRING"; + private static final String PROJECT_PATH_PROPERTY_TOKEN = "%project.folder%"; + private static final String TOD_TOKEN = "%TOD%"; + + private final Properties properties; + private final OMXMatrixDao mtxDao; + private final File projectPathFile; + private final int feedbackIterationNumber; + private final Set tables; + private final String[] timePeriods = ModelStructure.MODEL_PERIOD_LABELS; + private final String FUEL_COST_PROPERTY = "aoc.fuel"; + private final String MAINTENANCE_COST_PROPERTY = "aoc.maintenance"; + private static final String WRITE_LOGSUMS_PROPERTY = "Results.WriteLogsums"; + private static final String WRITE_TRANSIT_IVT_PROPERTY = "Report.writeTransitIVT"; + private static final String WRITE_UTILS_PROPERTY = "TourModeChoice.Save.UtilsAndProbs"; + + private float autoOperatingCost; + + private boolean writeCSV = false; + private boolean writeLogsums = false; + private boolean writeUtilities = true; + //private boolean writeTransitIVTs = false; + private MatrixDataServerIf ms; + + + public DataExporter(Properties theProperties, OMXMatrixDao aMtxDao, String projectPath, + int feedbackIterationNumber) + { + this.properties = theProperties; + this.mtxDao = aMtxDao; + + projectPathFile = new File(theProperties.getProperty("Project.Directory")); + this.feedbackIterationNumber = feedbackIterationNumber; + + float fuelCost = new Float(theProperties.getProperty(FUEL_COST_PROPERTY)); + float mainCost = new Float(theProperties.getProperty(MAINTENANCE_COST_PROPERTY)); + writeLogsums = new Boolean(theProperties.getProperty(WRITE_LOGSUMS_PROPERTY)); + writeUtilities = new Boolean(theProperties.getProperty(WRITE_UTILS_PROPERTY)); + //writeTransitIVTs = new Boolean(theProperties.getProperty(WRITE_TRANSIT_IVT_PROPERTY)); + + autoOperatingCost = (fuelCost + mainCost) * 0.01f; + + tables = new LinkedHashSet(); + + } + + private void addTable(String table) + { + tables.add(table); + LOGGER.info("exporting data: " + table); + } + + private String getPath(String path) + { + if (properties.containsKey(path)) return getPathFromProperty(path); + File ff = new File(path); + if (!ff.exists()) ff = new File(projectPathFile, path); + return ff.getAbsolutePath(); + } + + private String getPathFromProperty(String propertyToken) + { + String path = (String) properties.get(propertyToken); + if (!path.startsWith(projectPathFile.getAbsolutePath())) + path = new File(projectPathFile, path).getAbsolutePath(); + return path; + } + + /** + * Takes an input file name, returns the path to the directory + * with that name. + * + * @param The name of the file to create + * @return The path of the file. + */ + private String getOutputPath(String file) + { + return new File(properties.getProperty("report.path"), file).getAbsolutePath(); + } + + private String getData(TableDataSet data, int row, int column, FieldType type) + { + switch (type) + { + case INT: + return "" + Math.round(data.getValueAt(row, column)); + case FLOAT: + return "" + data.getValueAt(row, column); + case STRING: + return data.getStringValueAt(row, column); + case BIT: + return Boolean.parseBoolean(data.getStringValueAt(row, column)) ? "1" : "0"; + default: + throw new IllegalStateException("undefined field type: " + type); + } + } + + private String getPreferredColumnName(String columnName) + { + if (columnName.equalsIgnoreCase("hh_id")) return "HH_ID"; + if (columnName.equalsIgnoreCase("person_id")) return "PERSON_ID"; + if (columnName.toLowerCase().contains("maz")) + return columnName.toLowerCase().replace("maz", "mgra").toUpperCase(); + return columnName.toUpperCase(); + } + + private void exportData(TableDataSet data, String outputFileName, + Map outputMapping, Map outputTypes) + { + int[] outputIndices = new int[outputMapping.size()]; + FieldType[] outputFieldTypes = new FieldType[outputIndices.length]; + String[] header = new String[outputMapping.size()]; + + int counter = 0; + for (String column : outputMapping.keySet()) + { + header[counter] = column; + outputIndices[counter] = data.getColumnPosition(outputMapping.get(column)); + outputFieldTypes[counter++] = outputTypes.get(column); + } + + BlockingQueue queue = new LinkedBlockingQueue(); + Thread writerProcess = null; + try + { + CsvWriterThread writerThread = new CsvWriterThread(queue, new File( + getOutputPath(outputFileName + ".csv")), header); + writerProcess = new Thread(writerThread); + writerProcess.start(); + + for (int i = 1; i <= data.getRowCount(); i++) + { + String[] row = new String[outputMapping.size()]; + row[0] = getData(data, i, outputIndices[0], outputFieldTypes[0]); + + for (int j = 1; j < outputIndices.length; j++) + { + row[j] = getData(data, i, outputIndices[j], outputFieldTypes[j]); + } + queue.add(new CsvRow(row)); + } + } finally + { + queue.add(CsvWriterThread.POISON_PILL); + if (null != writerProcess) + { + try + { + writerProcess.join(); + } catch (InterruptedException e) + { + LOGGER.error(e); + System.exit(-1); + } + } + } + } + + private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, + boolean includeFeedbackIteration, String[] formats, Set floatColumns, + Set stringColumns, Set intColumns, Set bitColumns, + FieldType defaultFieldType, Set primaryKey, + TripStructureDefinition tripStructureDefinition,boolean isCB) + { + return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, + formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, + primaryKey, tripStructureDefinition, null,isCB); + } + + private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, + boolean includeFeedbackIteration, String[] formats, Set floatColumns, + Set stringColumns, Set intColumns, Set bitColumns, + FieldType defaultFieldType, Set primaryKey, + TripStructureDefinition tripStructureDefinition, JoinData joinData,boolean isCB) + { + return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, + formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, + primaryKey, new HashMap(), tripStructureDefinition, joinData,isCB); + } + + private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, + boolean includeFeedbackIteration, String[] formats, Set floatColumns, + Set stringColumns, Set intColumns, Set bitColumns, + FieldType defaultFieldType, Set primaryKey, + Map overridingFieldMappings, + TripStructureDefinition tripStructureDefinition,boolean isCB) + { + return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, + formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, + primaryKey, overridingFieldMappings, tripStructureDefinition, null,isCB); + } + + private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, + boolean includeFeedbackIteration, String[] formats, Set floatColumns, + Set stringColumns, Set intColumns, Set bitColumns, + FieldType defaultFieldType, Set primaryKey, + Map overridingFieldMappings, + TripStructureDefinition tripStructureDefinition, JoinData joinData,boolean isCB) + { + TableDataSet table; + try + { + String f = includeFeedbackIteration ? getPath(filePropertyToken).replace(".csv", + "_" + feedbackIterationNumber + ".csv") : getPath(filePropertyToken); + table = formats == null ? new CSVFileReader().readFile(new File(f)) + : new CSVFileReader().readFileWithFormats(new File(f), formats); + } catch (IOException e) + { + throw new RuntimeException(e); + } + if (joinData != null) joinData.joinDataToTable(table); + exportDataGeneric(table, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, defaultFieldType, primaryKey, overridingFieldMappings, + tripStructureDefinition,isCB); + return table; + } + + private class JoinData + { + private final Map> data; + private final Map dataType; + private final String idColumn; + + public JoinData(String idColumn) + { + this.idColumn = idColumn; + data = new LinkedHashMap>(); + dataType = new HashMap(); + } + + public void addJoinData(Map joinData, FieldType type, String columnName) + { + data.put(columnName, joinData); + dataType.put(columnName, type); + } + + public void joinDataToTable(TableDataSet table) + { + int[] ids = table.getColumnAsInt(idColumn); + for (String column : data.keySet()) + table.appendColumn(getData(ids, column), column); + } + + private Object getData(int[] ids, String column) + { + switch (dataType.get(column)) + { + case INT: + { + int[] columnData = new int[ids.length]; + @SuppressWarnings("unchecked") + // this is correct + Map dataMap = (Map) data.get(column); + for (int i = 0; i < ids.length; i++) + columnData[i] = dataMap.get(ids[i]); + return columnData; + } + case FLOAT: + { + float[] columnData = new float[ids.length]; + @SuppressWarnings("unchecked") + // this is correct + Map dataMap = (Map) data.get(column); + for (int i = 0; i < ids.length; i++) + columnData[i] = dataMap.get(ids[i]); + return columnData; + } + case STRING: + { + String[] columnData = new String[ids.length]; + @SuppressWarnings("unchecked") + // this is correct + Map dataMap = (Map) data.get(column); + for (int i = 0; i < ids.length; i++) + columnData[i] = dataMap.get(ids[i]); + return columnData; + } + case BIT: + { + boolean[] columnData = new boolean[ids.length]; + @SuppressWarnings("unchecked") + // this is correct + Map dataMap = (Map) data.get(column); + for (int i = 0; i < ids.length; i++) + columnData[i] = dataMap.get(ids[i]); + return columnData; + } + default: + throw new IllegalStateException("shouldn't be here: " + dataType.get(column)); + } + } + } + + private void exportDataGeneric(TableDataSet table, String outputFileBase, + Set floatColumns, Set stringColumns, Set intColumns, + Set bitColumns, FieldType defaultFieldType, Set primaryKey, + TripStructureDefinition tripStructureDefinition,boolean isCB) + { + exportDataGeneric(table, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, defaultFieldType, primaryKey, new HashMap(), + tripStructureDefinition,isCB); + + } + + private void exportDataGeneric(TableDataSet table, String outputFileBase, + Set floatColumns, Set stringColumns, Set intColumns, + Set bitColumns, FieldType defaultFieldType, Set primaryKey, + Map overridingFieldMappings, + TripStructureDefinition tripStructureDefinition,boolean isCB) + { + Map fieldMappings = new LinkedHashMap(); + Map fieldTypes = new HashMap(); + + if (tripStructureDefinition != null) + { + appendTripData(table, tripStructureDefinition,isCB); + floatColumns.add("AUTO_IVT"); + floatColumns.add("AUTO_AOC"); + floatColumns.add("AUTO_STD"); + floatColumns.add("AUTO_TOLL"); + floatColumns.add("TRAN_IVT"); + floatColumns.add("TRAN_WAIT"); + floatColumns.add("TRAN_WALK"); + floatColumns.add("TRAN_FARE"); + floatColumns.add("TRAN_ACCDIST"); + floatColumns.add("TRAN_EGRDIST"); + floatColumns.add("TRAN_AUXTIME"); + floatColumns.add("TRAN_ACCTIME"); + floatColumns.add("TRAN_EGRTIME"); + floatColumns.add("TRAN_TRANSFERS"); + floatColumns.add("WALK_TIME"); + floatColumns.add("BIKE_TIME"); + floatColumns.add("TRIP_DIST"); + stringColumns.add("TRIP_PURPOSE_NAME"); + stringColumns.add("TRIP_MODE_NAME"); + intColumns.add("RECID"); + floatColumns.add("LOC_IVT"); + floatColumns.add("EXP_IVT"); + floatColumns.add("BRT_IVT"); + floatColumns.add("LRT_IVT"); + floatColumns.add("CR_IVT"); + floatColumns.add("TRAN_DIST"); + floatColumns.add("PARK_WALK_TIME"); + floatColumns.add("PARK_WALK_DIST"); + + } + + if (primaryKey.size() == 0) + { + // have to add in a key - call it ID + int[] id = new int[table.getRowCount()]; + for (int i = 0; i < id.length; i++) + id[i] = i + 1; + table.appendColumn(id, "ID"); + + primaryKey.add("ID"); + intColumns.add("ID"); + } + + outer: for (String column : table.getColumnLabels()) + { + String c = overridingFieldMappings.containsKey(column) ? overridingFieldMappings + .get(column) : getPreferredColumnName(column); + fieldMappings.put(c, column); + for (String fc : floatColumns) + { + if (fc.equalsIgnoreCase(column)) + { + fieldTypes.put(c, FieldType.FLOAT); + continue outer; + } + } + for (String sc : stringColumns) + { + if (sc.equalsIgnoreCase(column)) + { + fieldTypes.put(c, FieldType.STRING); + continue outer; + } + } + for (String sc : intColumns) + { + if (sc.equalsIgnoreCase(column)) + { + fieldTypes.put(c, FieldType.INT); + continue outer; + } + } + for (String sc : bitColumns) + { + if (sc.equalsIgnoreCase(column)) + { + fieldTypes.put(c, FieldType.BIT); + continue outer; + } + } + fieldTypes.put(c, defaultFieldType); + } + Set pKey = new LinkedHashSet(); + for (String column : primaryKey) + pKey.add(getPreferredColumnName(column)); + exportData(table, outputFileBase, fieldMappings, fieldTypes); + } + + private PrintWriter getBufferedPrintWriter(String fileName) throws IOException + { + return new PrintWriter(new BufferedWriter(new FileWriter(fileName))); + } + + /** + * Appends trip data to table including skim attributes. + * + * @param table + * @param tripStructureDefinition + */ + private void appendTripData(TableDataSet table, TripStructureDefinition tripStructureDefinition, boolean isCB) + { + // id triptype recid partysize orig_maz dest_maz trip_board_tap + // trip_alight_tap trip_depart_time trip_time trip_distance trip_cost + // trip_purpose_name trip_mode_name vot + int rowCount = table.getRowCount(); + + float[] autoInVehicleTime = new float[rowCount]; + float[] autoOperatingCost = new float[rowCount]; + float[] autoStandardDeviation = new float[rowCount]; + float[] autoTollCost = new float[rowCount]; + float[] transitInVehicleTime = new float[rowCount]; + float[] transitWaitTime = new float[rowCount]; + float[] transitWalkTime = new float[rowCount]; + float[] transitFare = new float[rowCount]; + float[] walkModeTime = new float[rowCount]; + float[] bikeModeTime = new float[rowCount]; + float[] tripDistance = new float[rowCount]; + String[] tripPurpose = new String[rowCount]; + String[] tripMode = new String[rowCount]; + int[] tripId = new int[rowCount]; + int[] tripDepartTime = new int[rowCount]; + int[] tripBoardTaz = new int[rowCount]; + int[] tripAlightTaz = new int[rowCount]; + float[] tripParkingTime = new float[rowCount]; + float[] tripParkingDistance = new float[rowCount]; + + + float[] transitAccessDist = new float[rowCount]; + float[] transitEgressDist = new float[rowCount]; + float[] transitAuxTime = new float[rowCount]; + float[] transitAccTime = new float[rowCount]; + float[] transitEgrTime = new float[rowCount]; + float[] transitTransfers = new float[rowCount]; + + //these are only set if writeTransitIVTs is true + float[] locIVT = new float[rowCount]; + float[] expIVT = new float[rowCount]; + float[] brtIVT = new float[rowCount]; + float[] lrtIVT = new float[rowCount]; + float[] crIVT = new float[rowCount]; + + float[] tranDist = new float[rowCount]; + + SkimBuilder skimBuilder = new SkimBuilder(properties); + boolean hasPurposeColumn = tripStructureDefinition.originPurposeColumn > -1; + for (int i = 0; i < rowCount; i++) + { + int row = i + 1; + + double epsilon = .000001; + boolean inbound = tripStructureDefinition.booleanIndicatorVariables ? table + .getBooleanValueAt(row, tripStructureDefinition.inboundColumn) : Math.abs(table + .getValueAt(row, tripStructureDefinition.inboundColumn) - 1.0) < epsilon; + + int transponderOwnership=0; + if(tripStructureDefinition.transponderOwnershipColumn>0) + transponderOwnership = (int) table.getValueAt(row, tripStructureDefinition.transponderOwnershipColumn); + + SkimBuilder.TripAttributes attributes = skimBuilder.getTripAttributes( + (int) table.getValueAt(row, tripStructureDefinition.originMgraColumn), + (int) table.getValueAt(row, tripStructureDefinition.destMgraColumn), + (int) table.getValueAt(row, tripStructureDefinition.modeColumn), + (int) table.getValueAt(row, tripStructureDefinition.boardTapColumn), + (int) table.getValueAt(row, tripStructureDefinition.alightTapColumn), + (int) table.getValueAt(row, tripStructureDefinition.todColumn), + inbound, + table.getValueAt(row,tripStructureDefinition.valueOfTimeColumn), + (int) table.getValueAt(row, tripStructureDefinition.setColumn),isCB,transponderOwnership); + + autoInVehicleTime[i] = attributes.getAutoInVehicleTime(); + autoOperatingCost[i] = attributes.getAutoOperatingCost(); + autoStandardDeviation[i] = attributes.getAutoStandardDeviationTime(); + autoTollCost[i] = attributes.getAutoTollCost(); + transitInVehicleTime[i] = attributes.getTransitInVehicleTime(); + transitWaitTime[i] = attributes.getTransitWaitTime(); + transitWalkTime[i] = attributes.getTransitWalkTime(); + transitFare[i] = attributes.getTransitFare(); + walkModeTime[i] = attributes.getWalkModeTime(); + bikeModeTime[i] = attributes.getBikeModeTime(); + tripDistance[i] = attributes.getTripDistance(); + transitAccessDist[i] = attributes.getTransitAccessDistance(); + transitEgressDist[i] = attributes.getTransitEgressDistance(); + transitAuxTime[i] = attributes.getTransitAuxiliaryTime(); + transitAccTime[i] = attributes.getTransitAccessTime(); + transitEgrTime[i] = attributes.getTransitEgressTime(); + transitTransfers[i] = attributes.getTransitTransfers(); + + + //get parking walk time + if(tripStructureDefinition.parkingMazColumn>-1) { + int parkingMaz = (int) table.getValueAt(row, tripStructureDefinition.parkingMazColumn); + + if(parkingMaz>0) { + + int destMaz = (int) table.getValueAt(row, tripStructureDefinition.destMgraColumn); + float parkingWalkTime = skimBuilder.getLotWalkTime(parkingMaz,destMaz); + float parkingWalkDistance = skimBuilder.getLotWalkDistance(parkingMaz,destMaz); + tripParkingTime[i]= parkingWalkTime; + tripParkingDistance[i] = parkingWalkDistance; + } + } + + + if (hasPurposeColumn) + { + tripPurpose[i] = table.getStringValueAt(row, + tripStructureDefinition.destinationPurposeColumn); + } else + { + if (!inbound) // going out + tripPurpose[i] = tripStructureDefinition.destinationName; + else tripPurpose[i] = tripStructureDefinition.homeName; + } + tripMode[i] = attributes.getTripModeName(); + tripId[i] = i; + tripDepartTime[i] = attributes.getTripStartTime(); + tripBoardTaz[i] = attributes.getTripBoardTaz(); + tripAlightTaz[i] = attributes.getTripAlightTaz(); + + locIVT[i] = attributes.getLocTime(); + expIVT[i] = attributes.getExpTime(); + brtIVT[i] = attributes.getBrtTime(); + lrtIVT[i] = attributes.getLrtTime(); + crIVT[i] = attributes.getCrTime(); + + tranDist[i] = attributes.getTransitDistance(); + } + table.appendColumn(autoInVehicleTime, "AUTO_IVT"); + table.appendColumn(autoOperatingCost, "AUTO_AOC"); + table.appendColumn(autoStandardDeviation, "AUTO_STD"); + table.appendColumn(autoTollCost, "AUTO_TOLL"); + table.appendColumn(transitInVehicleTime, "TRAN_IVT"); + table.appendColumn(transitWaitTime, "TRAN_WAIT"); + table.appendColumn(transitWalkTime, "TRAN_WALK"); + table.appendColumn(transitFare, "TRAN_FARE"); + table.appendColumn(transitAccessDist, "TRAN_ACCDIST"); + table.appendColumn(transitEgressDist, "TRAN_EGRDIST"); + table.appendColumn(transitAuxTime, "TRAN_AUXTIME"); + table.appendColumn(transitAccTime, "TRAN_ACCTIME"); + table.appendColumn(transitEgrTime, "TRAN_EGRTIME"); + table.appendColumn(transitTransfers, "TRAN_TRANSFERS"); + + table.appendColumn(walkModeTime, "WALK_TIME"); + table.appendColumn(bikeModeTime, "BIKE_TIME"); + table.appendColumn(tripDistance, "TRIP_DIST"); + table.appendColumn(tripPurpose, "TRIP_PURPOSE_NAME"); + table.appendColumn(tripMode, "TRIP_MODE_NAME"); + table.appendColumn(tripId, "RECID"); + table.appendColumn(tripBoardTaz, "TRIP_BOARD_TAZ"); + table.appendColumn(tripAlightTaz, "TRIP_ALIGHT_TAZ"); + + table.appendColumn(locIVT, "LOC_IVT"); + table.appendColumn(expIVT, "EXP_IVT"); + table.appendColumn(brtIVT, "BRT_IVT"); + table.appendColumn(lrtIVT, "LRT_IVT"); + table.appendColumn(crIVT, "CR_IVT"); + + table.appendColumn(tranDist, "TRAN_DIST"); + table.appendColumn(tripParkingTime, "PARK_WALK_TIME"); + table.appendColumn(tripParkingDistance,"PARK_WALK_DIST"); + +// table.appendColumn(valueOfTime, "VALUE_OF_TIME"); + } + + private void exportAccessibilities(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(Arrays.asList("mgra")); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("mgra")); + exportDataGeneric(outputFileBase, "acc.output.file", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private void exportMazData(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(Arrays.asList("mgra", "TAZ", "ZIP09")); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("mgra")); + exportDataGeneric(outputFileBase, "mgra.socec.file", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private void nullifyFile(String file) + { + String tempFile = file + ".temp"; + File f = new File(file); + if (!f.renameTo(new File(tempFile))) + throw new RuntimeException("Couldn't rename to file: " + f); + BufferedReader reader = null; + PrintWriter writer = null; + try + { + reader = new BufferedReader(new FileReader(tempFile)); + writer = getBufferedPrintWriter(file); + String line; + while ((line = reader.readLine()) != null) + writer.println(line.replace(NULL_VALUE, "")); + } catch (IOException e) + { + throw new RuntimeException(e); + } finally + { + if (reader != null) + { + try + { + reader.close(); + } catch (IOException e) + { + // ignore + } + } + if (writer != null) writer.close(); + } + new File(tempFile).delete(); + } + + public static int NULL_INT_VALUE = -98765; + public static float NULL_FLOAT_VALUE = NULL_INT_VALUE; + public static String NULL_VALUE = "" + NULL_FLOAT_VALUE; + + private void exportTapData(String outputFileBase) + { + addTable(outputFileBase); + Map ptype = readSpaceDelimitedData(getPath("tap.ptype.file"), + Arrays.asList("TAP", "LOTID", "PTYPE", "TAZ", "CAPACITY", "DISTANCE")); + Map pelev = readSpaceDelimitedData( + getPath("tap.ptype.file").replace("ptype", "elev"), Arrays.asList("TAP", "ELEV")); + float[] taps = ptype.get("TAP"); + float[] etaps = pelev.get("TAP"); + ptype.put("ELEV", getPartialData(taps, etaps, pelev.get("ELEV"))); + + TableDataSet finalData = new TableDataSet(); + for (String columnName : ptype.keySet()) + finalData.appendColumn(ptype.get(columnName), columnName); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("TAP")); + exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, FieldType.INT, primaryKey, null,false); + nullifyFile(getOutputPath(outputFileBase + ".csv")); + } + + private void exportMgraToTapData(String outputFileBase) + { + addTable(outputFileBase); + String walkdistanceFile=PROJECT_PATH_PROPERTY_TOKEN+"\\input\\"+properties.getProperty("active.logsum.matrix.file.walk.mgratap"); + Map mgraToTap = readSpaceDelimitedData(walkdistanceFile, Arrays.asList("MGRA", "TAP", "DISTANCE")); + TableDataSet finalData = new TableDataSet(); + for (String columnName : mgraToTap.keySet()) + finalData.appendColumn(mgraToTap.get(columnName), columnName); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("MGRA", "TAP")); + exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, FieldType.INT, primaryKey, null,false); + nullifyFile(getOutputPath(outputFileBase + ".csv")); + } + + private void exportMgraToMgraData(String outputFileBase) + { + addTable(outputFileBase); + //wu modified to get the updated walk distance between MGRAs + String walkdistanceFile=PROJECT_PATH_PROPERTY_TOKEN+"\\input\\"+properties.getProperty("active.logsum.matrix.file.walk.mgra"); + Map mgraToMgra = readSpaceDelimitedData(walkdistanceFile, Arrays.asList("ORIG_MGRA", "DEST_MGRA", "DISTANCE")); + Map> actualData = new LinkedHashMap>(); + for (String column : Arrays.asList("TAZ", "ORIG_MGRA", "DEST_MGRA", "DISTANCE")) + actualData.put(column, new LinkedList()); + float[] dcolumn = mgraToMgra.get("DISTANCE"); + float[] origColumn = mgraToMgra.get("ORIG_MGRA"); + float[] destColumn = mgraToMgra.get("DEST_MGRA"); + for (int i = 0; i < dcolumn.length; i++) + { + int count = 0; + if (dcolumn[i] < 0) count = (int) destColumn[i]; + int taz = (int) origColumn[i]; + while (count-- > 0) + { + i++; + actualData.get("TAZ").add(taz); + actualData.get("ORIG_MGRA").add((int) origColumn[i]); + actualData.get("DEST_MGRA").add((int) destColumn[i]); + actualData.get("DISTANCE").add(dcolumn[i]); + } + } + + TableDataSet finalData = new TableDataSet(); + for (String columnName : actualData.keySet()) + { + Object data; + if (columnName.equals("DISTANCE")) + { + float[] dd = new float[actualData.get(columnName).size()]; + int counter = 0; + for (Number n : actualData.get(columnName)) + dd[counter++] = n.floatValue(); + data = dd; + } else + { + int[] dd = new int[actualData.get(columnName).size()]; + int counter = 0; + for (Number n : actualData.get(columnName)) + dd[counter++] = n.intValue(); + data = dd; + } + finalData.appendColumn(data, columnName); + } + + Set intColumns = new HashSet(Arrays.asList("TAZ", "ORIG_MGRA", "DEST_MGRA")); + Set floatColumns = new HashSet(Arrays.asList("DISTANCE")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("ORIG_MGRA", "DEST_MGRA")); + exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, FieldType.INT, primaryKey, null,false); + nullifyFile(getOutputPath(outputFileBase + ".csv")); + } + + private void exportTazToTapData(String outputFileBase) + { + addTable(outputFileBase); + Map tazToTap = readSpaceDelimitedData( + getPath("taz.driveaccess.taps.file"), Arrays.asList("TAZ", "TAP", "TIME", "DISTANCE", "MODE")); + + Map> actualData = new LinkedHashMap>(); + for (String column : Arrays.asList("TAZ", "TAP", "TIME", "DISTANCE", "MODE")) + actualData.put(column, new LinkedList()); + + float[] taz = tazToTap.get("TAZ"); + float[] tap = tazToTap.get("TAP"); + float[] time = tazToTap.get("TIME"); + float[] dist = tazToTap.get("DISTANCE"); + float[] mode = tazToTap.get("MODE"); + + for (int i = 0; i < taz.length; i++) + { + actualData.get("TAZ").add((int) taz[i]); + actualData.get("TAP").add((int) tap[i]); + actualData.get("TIME").add(time[i]); + actualData.get("DISTANCE").add(dist[i]); + actualData.get("MODE").add(mode[i]); + } + + TableDataSet finalData = new TableDataSet(); + for (String columnName : actualData.keySet()) + { + Object data; + if (columnName.equals("DISTANCE") || columnName.equals("TIME")) + { + float[] dd = new float[actualData.get(columnName).size()]; + int counter = 0; + for (Number n : actualData.get(columnName)) + dd[counter++] = n.floatValue(); + data = dd; + } else + { + int[] dd = new int[actualData.get(columnName).size()]; + int counter = 0; + for (Number n : actualData.get(columnName)) + dd[counter++] = n.intValue(); + data = dd; + } + finalData.appendColumn(data, columnName); + } + + Set intColumns = new HashSet(Arrays.asList("TAZ", "TAP")); + Set floatColumns = new HashSet(Arrays.asList("TIME", "DISTANCE")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("TAZ", "TAP")); + exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, FieldType.INT, primaryKey, null,false); + nullifyFile(getOutputPath(outputFileBase + ".csv")); + } + + private float[] toFloatArray(int[] data) + { + float[] f = new float[data.length]; + for (int i = 0; i < f.length; i++) + f[i] = data[i]; + return f; + } + + private float[] getPartialData(float[] fullKey, float[] partialKey, float[] partialData) + { + float[] data = new float[fullKey.length]; + Arrays.fill(data, NULL_FLOAT_VALUE); + int counter = 0; + for (float key : fullKey) + { + for (int i = 0; i < partialKey.length; i++) + { + if (partialKey[i] == key) + { + data[counter] = partialData[i]; + } + } + counter++; + } + return data; + } + + private void exportTazData(String outputFileBase) + { + addTable(outputFileBase); + int[] tazs = getTazList(); + TableDataSet data = new TableDataSet(); + data.appendColumn(tazs, "TAZ"); + Map term = readSpaceDelimitedData(getPath("taz.terminal.time.file"), + Arrays.asList("TAZ", "TERM")); + Map park = readSpaceDelimitedData(getPath("taz.parkingtype.file"), + Arrays.asList("TAZ", "PARK")); + data.appendColumn(getPartialData(toFloatArray(tazs), term.get("TAZ"), term.get("TERM")), + "TERM"); + data.appendColumn(getPartialData(toFloatArray(tazs), park.get("TAZ"), park.get("PARK")), + "PARK"); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("TAZ")); + exportDataGeneric(data, outputFileBase, floatColumns, stringColumns, intColumns, + bitColumns, FieldType.INT, primaryKey, null,false); + nullifyFile(getOutputPath(outputFileBase + ".csv")); + } + + private int[] getTazList() + { + Set tazs = new TreeSet(); + TableDataSet mgraData; + try + { + mgraData = new CSVFileReader().readFile(new File(getPath("mgra.socec.file"))); + } catch (IOException e) + { + throw new RuntimeException(e); + } + boolean first = true; + for (int taz : mgraData.getColumnAsInt("taz")) + { + if (first) + { + first = false; + continue; + } + tazs.add(taz); + } + int[] finalTazs = new int[tazs.size()]; + int counter = 0; + for (int taz : tazs) + finalTazs[counter++] = taz; + return finalTazs; + } + + private Map readSpaceDelimitedData(String location, List columnNames) + { + Map> data = new LinkedHashMap>(); + for (String columnName : columnNames) + data.put(columnName, new LinkedList()); + BufferedReader reader = null; + try + { + reader = new BufferedReader(new FileReader(location)); + String line; + while ((line = reader.readLine()) != null) + { + String[] d = line.trim().split(","); + int counter = 0; + for (String columnName : columnNames) + { + if (counter < d.length) + { + data.get(columnName).add(Float.parseFloat(d[counter++])); + } else + { + data.get(columnName).add(NULL_FLOAT_VALUE); // if missing + // entry/entries, + // then put in + // null value + } + } + } + } catch (IOException e) + { + throw new RuntimeException(e); + } finally + { + if (reader != null) + { + try + { + reader.close(); + } catch (IOException e) + { + // ignore + } + } + } + Map d = new LinkedHashMap(); + for (String columnName : columnNames) + { + float[] f = new float[data.get(columnName).size()]; + int counter = 0; + for (Float i : data.get(columnName)) + f[counter++] = i; + d.put(columnName, f); + } + return d; + } + + private void exportHouseholdData(String outputFileBase) + { + addTable(outputFileBase); + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // hh_id + formatList.add(NUMBER_FORMAT_NAME); // home_mgra + formatList.add(NUMBER_FORMAT_NAME); // income + formatList.add(NUMBER_FORMAT_NAME); // autos + formatList.add(NUMBER_FORMAT_NAME); // HVs + formatList.add(NUMBER_FORMAT_NAME); // AVs + formatList.add(NUMBER_FORMAT_NAME); // transponder + formatList.add(STRING_FORMAT_NAME); // cdap_pattern + formatList.add(NUMBER_FORMAT_NAME); // jtf_choice + + + if(writeLogsums){ + formatList.add(NUMBER_FORMAT_NAME); //aoLogsum + formatList.add(NUMBER_FORMAT_NAME); //transponderLogsum + formatList.add(NUMBER_FORMAT_NAME); //cdapLogsum + formatList.add(NUMBER_FORMAT_NAME); //jtfLogsum + } + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + + if(writeLogsums){ + floatColumns.add("aoLogsum"); //aoLogsum + floatColumns.add("transponderLogsum"); //transponderLogsum + floatColumns.add("cdapLogsum"); //cdapLogsum + floatColumns.add("jtfLogsum"); //jtfLogsum + } + + Set stringColumns = new HashSet(Arrays.asList("cdap_pattern")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id")); + exportDataGeneric(outputFileBase, "Results.HouseholdDataFile", true, formats, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, null,false); + } + + private void exportPersonData(String outputFileBase) + { + addTable(outputFileBase); + + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // hh_id + formatList.add(NUMBER_FORMAT_NAME); // person_id + formatList.add(NUMBER_FORMAT_NAME); // person_num + formatList.add(NUMBER_FORMAT_NAME); // age + formatList.add(STRING_FORMAT_NAME); // gender + formatList.add(STRING_FORMAT_NAME); // type + formatList.add(NUMBER_FORMAT_NAME); // value_of_time (float) + formatList.add(STRING_FORMAT_NAME); // activity_pattern + formatList.add(NUMBER_FORMAT_NAME); // imf_choice + formatList.add(NUMBER_FORMAT_NAME); // inmf_choice + formatList.add(NUMBER_FORMAT_NAME); // fp_choice + formatList.add(NUMBER_FORMAT_NAME); // reimb_pct (float) + formatList.add(NUMBER_FORMAT_NAME); // ie_choice + formatList.add(NUMBER_FORMAT_NAME); // timeFactorWork + formatList.add(NUMBER_FORMAT_NAME); // timeFactorNonWork + + if(writeLogsums){ + formatList.add(NUMBER_FORMAT_NAME); //wfhLogsum + formatList.add(NUMBER_FORMAT_NAME); //wlLogsum + formatList.add(NUMBER_FORMAT_NAME); //slLogsum + formatList.add(NUMBER_FORMAT_NAME); //fpLogsum + formatList.add(NUMBER_FORMAT_NAME); //ieLogsum + formatList.add(NUMBER_FORMAT_NAME); //cdapLogsum + formatList.add(NUMBER_FORMAT_NAME); //imtfLogsum + formatList.add(NUMBER_FORMAT_NAME);//inmtfLogsum + } + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("value_of_time", "reimb_pct")); + + if(writeLogsums){ + floatColumns.add("wfhLogsum"); //wfhLogsum + floatColumns.add("wlLogsum"); //wlLogsum + floatColumns.add("slLogsum"); //slLogsum + floatColumns.add("fpLogsum"); //fpLogsum + floatColumns.add("ieLogsum"); //ieLogsum + floatColumns.add("cdapLogsum"); //cdapLogsum + floatColumns.add("imtfLogsum"); //imtfLogsum + floatColumns.add("inmtfLogsum");//inmtfLogsum + } + + Set stringColumns = new HashSet(Arrays.asList("gender", "type", + "activity_pattern")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("person_id")); + exportDataGeneric(outputFileBase, "Results.PersonDataFile", true, formats, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, null,false); + } + + private void exportSyntheticHouseholdData(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("HHID")); + exportDataGeneric(outputFileBase, "PopulationSynthesizer.InputToCTRAMP.HouseholdFile", + false, null, floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, + primaryKey, null,false); + } + + private void exportSyntheticPersonData(String outputFileBase) + { + addTable(outputFileBase); + String[] formats = {NUMBER_FORMAT_NAME, // HHID + NUMBER_FORMAT_NAME, // PERID + NUMBER_FORMAT_NAME, // household_serial_no + NUMBER_FORMAT_NAME, // PNUM + NUMBER_FORMAT_NAME, // AGE + NUMBER_FORMAT_NAME, // SEX + NUMBER_FORMAT_NAME, // MILTARY + NUMBER_FORMAT_NAME, // PEMPLOY + NUMBER_FORMAT_NAME, // PSTUDENT + NUMBER_FORMAT_NAME, // PTYPE + NUMBER_FORMAT_NAME, // EDUC + NUMBER_FORMAT_NAME, // GRADE + NUMBER_FORMAT_NAME, // OCCCEN5 + STRING_FORMAT_NAME, // OCCSOC5 + NUMBER_FORMAT_NAME, // INDCEN + NUMBER_FORMAT_NAME, // WEEKS + NUMBER_FORMAT_NAME, // HOURS + }; + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(Arrays.asList("OCCSOC5")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("PERID")); + exportDataGeneric(outputFileBase, "PopulationSynthesizer.InputToCTRAMP.PersonFile", false, + formats, floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, + primaryKey, null,false); + } + + private void exportWorkSchoolLocation(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("WorkLocationDistance", + "WorkLocationLogsum", "SchoolLocation", "SchoolLocationDistance", + "SchoolLocationLogsum")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("PERSON_ID")); + Map overridingNames = new HashMap(); + overridingNames.put("PersonID", "PERSON_ID"); + exportDataGeneric(outputFileBase, "Results.UsualWorkAndSchoolLocationChoice", true, null, + floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + overridingNames, null,false); + } + + private void exportIndivToursData(String outputFileBase) + { + addTable(outputFileBase); + + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // hh_id + formatList.add(NUMBER_FORMAT_NAME); // person_id + formatList.add(NUMBER_FORMAT_NAME); // person_num + formatList.add(NUMBER_FORMAT_NAME); // person_type + formatList.add(NUMBER_FORMAT_NAME); // tour_id + formatList.add(STRING_FORMAT_NAME); // tour_category + formatList.add(STRING_FORMAT_NAME); // tour_purpose + formatList.add(NUMBER_FORMAT_NAME); // orig_maz + formatList.add(NUMBER_FORMAT_NAME); // dest_maz + formatList.add(NUMBER_FORMAT_NAME); // start_period + formatList.add(NUMBER_FORMAT_NAME); // end_period + formatList.add(NUMBER_FORMAT_NAME); // tour_mode + formatList.add(NUMBER_FORMAT_NAME); // av_avail + formatList.add(NUMBER_FORMAT_NAME); // tour_distance + formatList.add(NUMBER_FORMAT_NAME); // atWork_freq + formatList.add(NUMBER_FORMAT_NAME); // num_ob_stops + formatList.add(NUMBER_FORMAT_NAME); // num_ib_stops + formatList.add(NUMBER_FORMAT_NAME); // valueOfTime + + if(writeUtilities){ + for(int i=1;i<=26;++i) + formatList.add(NUMBER_FORMAT_NAME); // util_i + for(int i=1;i<=26;++i) + formatList.add(NUMBER_FORMAT_NAME); // prob_i + } + + if(writeLogsums){ + formatList.add(NUMBER_FORMAT_NAME); //timeOfDayLogsum + formatList.add(NUMBER_FORMAT_NAME);//tourModeLogsum + formatList.add(NUMBER_FORMAT_NAME);//subtourFreqLogsum + formatList.add(NUMBER_FORMAT_NAME);//tourDestinationLogsum + formatList.add(NUMBER_FORMAT_NAME);//stopFreqLogsum + + for(int i = 1; i<=4;++i) + formatList.add(NUMBER_FORMAT_NAME);//outStopDCLogsum_i + + for(int i = 1; i<=4;++i) + formatList.add(NUMBER_FORMAT_NAME);//inbStopDCLogsum_i + } + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(Arrays.asList("hh_id", "person_id", + "person_num", "person_type", "tour_id", "orig_mgra", "dest_mgra", "start_period", + "end_period", "tour_mode", "av_avail", "atWork_freq", "num_ob_stops", "num_ib_stops")); + + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + + if(writeUtilities){ + for(int i=1;i<=12;++i) + floatColumns.add("util_"+i); // util_i + for(int i=1;i<=12;++i) + floatColumns.add("prob_"+i); // prob_i + } + if(writeLogsums){ + floatColumns.add("timeOfDayLogsum"); + floatColumns.add("tourModeLogsum"); + floatColumns.add("subtourFreqLogsum"); + floatColumns.add("tourDestinationLogsum"); + floatColumns.add("stopFreqLogsum"); + + for(int i = 1; i<=4;++i) + floatColumns.add("outStopDCLogsum_"+i);//outStopDCLogsum_i + + for(int i = 1; i<=4;++i) + floatColumns.add("inbStopDCLogsum_"+i);//inbStopDCLogsum_i + } + + Set stringColumns = new HashSet(Arrays.asList("tour_category", + "tour_purpose")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "person_id", + "tour_category", "tour_id", "tour_purpose")); + exportDataGeneric(outputFileBase, "Results.IndivTourDataFile", true, formats, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private void exportJointToursData(String outputFileBase) + { + addTable(outputFileBase); + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // hh_id + formatList.add(NUMBER_FORMAT_NAME); // tour_id + formatList.add(STRING_FORMAT_NAME); // tour_category + formatList.add(STRING_FORMAT_NAME); // tour_purpose + formatList.add(NUMBER_FORMAT_NAME); // tour_composition + formatList.add(STRING_FORMAT_NAME); // tour_participants + formatList.add(NUMBER_FORMAT_NAME); // orig_maz + formatList.add(NUMBER_FORMAT_NAME); // dest_maz + formatList.add(NUMBER_FORMAT_NAME); // start_period + formatList.add(NUMBER_FORMAT_NAME); // end_period + formatList.add(NUMBER_FORMAT_NAME); // tour_mode + formatList.add(NUMBER_FORMAT_NAME); // av_avail + formatList.add(NUMBER_FORMAT_NAME); // tour_distance + formatList.add(NUMBER_FORMAT_NAME); // num_ob_stops + formatList.add(NUMBER_FORMAT_NAME); // num_ib_stops + formatList.add(NUMBER_FORMAT_NAME); // valueOfTime + + if(writeUtilities){ + for(int i=1;i<=26;++i) + formatList.add(NUMBER_FORMAT_NAME); // util_i + for(int i=1;i<=26;++i) + formatList.add(NUMBER_FORMAT_NAME); // prob_i + } + + if(writeLogsums){ + formatList.add(NUMBER_FORMAT_NAME); //timeOfDayLogsum + formatList.add(NUMBER_FORMAT_NAME);//tourModeLogsum + formatList.add(NUMBER_FORMAT_NAME);//subtourFreqLogsum + formatList.add(NUMBER_FORMAT_NAME);//tourDestinationLogsum + formatList.add(NUMBER_FORMAT_NAME);//stopFreqLogsum + + for(int i = 1; i<=4;++i) + formatList.add(NUMBER_FORMAT_NAME);//outStopDCLogsum_i + + for(int i = 1; i<=4;++i) + formatList.add(NUMBER_FORMAT_NAME);//inbStopDCLogsum_i + } + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(Arrays.asList("hh_id", "tour_id", + "tour_composition", "orig_mgra", "dest_mgra", "start_period", "end_period", + "tour_mode", "av_avail", "num_ob_stops", "num_ib_stops")); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + + if(writeUtilities){ + for(int i=1;i<=12;++i) + floatColumns.add("util_"+i); // util_i + for(int i=1;i<=12;++i) + floatColumns.add("prob_"+i); // prob_i + } + if(writeLogsums){ + floatColumns.add("timeOfDayLogsum"); + floatColumns.add("tourModeLogsum"); + floatColumns.add("subtourFreqLogsum"); + floatColumns.add("tourDestinationLogsum"); + floatColumns.add("stopFreqLogsum"); + + for(int i = 1; i<=4;++i) + floatColumns.add("outStopDCLogsum_"+i);//outStopDCLogsum_i + + for(int i = 1; i<=4;++i) + floatColumns.add("inbStopDCLogsum_"+i);//inbStopDCLogsum_i + } + Set stringColumns = new HashSet(Arrays.asList("tour_category", + "tour_purpose", "tour_participants")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "tour_category", + "tour_id", "tour_purpose")); + exportDataGeneric(outputFileBase, "Results.JointTourDataFile", true, formats, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private void exportIndivTripData(String outputFileBase) + { + addTable(outputFileBase); + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // 1 hh_id + formatList.add(NUMBER_FORMAT_NAME); // 2 person_id + formatList.add(NUMBER_FORMAT_NAME); // 3 person_num + formatList.add(NUMBER_FORMAT_NAME); // 4 tour_id + formatList.add(NUMBER_FORMAT_NAME); // 5 stop_id + formatList.add(NUMBER_FORMAT_NAME); // 6 inbound + formatList.add(STRING_FORMAT_NAME); // 7 tour_purpose + formatList.add(STRING_FORMAT_NAME); // 8 orig_purpose + formatList.add(STRING_FORMAT_NAME); // 9 dest_purpose + formatList.add(NUMBER_FORMAT_NAME); // 10 orig_maz + formatList.add(NUMBER_FORMAT_NAME); // 11 dest_maz + formatList.add(NUMBER_FORMAT_NAME); // 12 parking_maz + formatList.add(NUMBER_FORMAT_NAME); // 13 stop_period + formatList.add(NUMBER_FORMAT_NAME); // 14 trip_mode + formatList.add(NUMBER_FORMAT_NAME); // 15 av_avail + formatList.add(NUMBER_FORMAT_NAME); // 16 trip_board_tap + formatList.add(NUMBER_FORMAT_NAME); // 17 trip_alight_tap + formatList.add(NUMBER_FORMAT_NAME); // 18 set + formatList.add(NUMBER_FORMAT_NAME); // 19 tour_mode + formatList.add(NUMBER_FORMAT_NAME); // 20 driver_pnum + formatList.add(NUMBER_FORMAT_NAME); // 21 orig_escort_stoptype + formatList.add(NUMBER_FORMAT_NAME); // 22 orig_escortee_pnum + formatList.add(NUMBER_FORMAT_NAME); // 23 dest_escort_stoptype + formatList.add(NUMBER_FORMAT_NAME); // 24 dest_escortee_pnum + formatList.add(NUMBER_FORMAT_NAME); // 25 value of time + formatList.add(NUMBER_FORMAT_NAME); // 26 transponder availability + formatList.add(NUMBER_FORMAT_NAME); // 27 micro_walkMode + formatList.add(NUMBER_FORMAT_NAME); // 28 micro_trnAcc + formatList.add(NUMBER_FORMAT_NAME); // 29 micro_trnEgr + + if(writeLogsums) + formatList.add(NUMBER_FORMAT_NAME);//tripModeLogsum + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(); + + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + + if(writeLogsums) + floatColumns = new HashSet(Arrays.asList("valueOfTime","tripModeLogsum")); + + Set stringColumns = new HashSet(Arrays.asList("tour_purpose", + "orig_purpose", "dest_purpose")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "person_id", + "tour_id", "tour_purpose", "inbound", "stop_id")); + exportDataGeneric( + outputFileBase, + "Results.IndivTripDataFile", + true, + formats, + floatColumns, + stringColumns, + intColumns, + bitColumns, + FieldType.INT, + primaryKey, + new TripStructureDefinition(10, 11, 8, 9, 13, 14, 16, 17, 12, -1, 29, "INDIV", 6, false, 25, 18,26),false); + } + + private void exportJointTripData(String outputFileBase) + { + addTable(outputFileBase); + ArrayList formatList = new ArrayList(); + + formatList.add(NUMBER_FORMAT_NAME); // 1 hh_id + formatList.add(NUMBER_FORMAT_NAME); // 2 tour_id + formatList.add(NUMBER_FORMAT_NAME); // 3 stop_id + formatList.add(NUMBER_FORMAT_NAME); // 4 inbound + formatList.add(STRING_FORMAT_NAME); // 5 tour_purpose + formatList.add(STRING_FORMAT_NAME); // 6 orig_purpose + formatList.add(STRING_FORMAT_NAME); // 7 dest_purpose + formatList.add(NUMBER_FORMAT_NAME); // 8 orig_maz + formatList.add(NUMBER_FORMAT_NAME); // 9 dest_maz + formatList.add(NUMBER_FORMAT_NAME); // 10 parking_maz + formatList.add(NUMBER_FORMAT_NAME); // 11 stop_period + formatList.add(NUMBER_FORMAT_NAME); // 12 trip_mode + formatList.add(NUMBER_FORMAT_NAME); // 13 av_avail + formatList.add(NUMBER_FORMAT_NAME); // 14 num_participants + formatList.add(NUMBER_FORMAT_NAME); // 15 trip_board_tap + formatList.add(NUMBER_FORMAT_NAME); // 16 trip_alight_tap + formatList.add(NUMBER_FORMAT_NAME); // 17 set + formatList.add(NUMBER_FORMAT_NAME); // 18 tour_mode + formatList.add(NUMBER_FORMAT_NAME); // 19 value of time + formatList.add(NUMBER_FORMAT_NAME); // 20 transponder availability + formatList.add(NUMBER_FORMAT_NAME); // 21 micro_walkMode + formatList.add(NUMBER_FORMAT_NAME); // 22 micro_trnAcc + formatList.add(NUMBER_FORMAT_NAME); // 23 micro_trnEgr + + if(writeLogsums) + formatList.add(NUMBER_FORMAT_NAME);//tripModeLogsum + + String[] formats = new String[formatList.size()]; + formats = formatList.toArray(formats); + + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + + if(writeLogsums) + floatColumns = new HashSet(Arrays.asList("valueOfTime","tripModeLogsum")); + + Set stringColumns = new HashSet(Arrays.asList("tour_purpose", + "orig_purpose", "dest_purpose")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "tour_id", + "tour_purpose", "inbound", "stop_id")); + exportDataGeneric(outputFileBase, "Results.JointTripDataFile", true, formats, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + new TripStructureDefinition(8, 9, 6, 7, 11, 12, 15, 16, 10, 14, 23, "JOINT", 4, false, 19, 17,20),false); + } + + private void exportAirportTripsSAN(String outputFileBase) + { + + //id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); + + addTable(outputFileBase); + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("id")); + Map overridingNames = new HashMap(); + // overridingNames.put("id","PARTYID"); + exportDataGeneric(outputFileBase, "airport.SAN.output.file", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, overridingNames, + new TripStructureDefinition(8, 9, 7, 12, 15, 16, -1, 4, 18, "AIRPORT", "HOME", + "AIRPORT", 2, false, 18, 17,-1),false); + } + + private void exportAirportTripsCBX(String outputFileBase) + { + + //id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); + addTable(outputFileBase); + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("id")); + Map overridingNames = new HashMap(); + // overridingNames.put("id","PARTYID"); + exportDataGeneric(outputFileBase, "airport.CBX.output.file", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, overridingNames, + new TripStructureDefinition(8, 9, 7, 12, 15, 16, -1, 4, 18, "AIRPORT", "HOME", + "AIRPORT", 2, false, 18, 17,-1),false); + } + + private void exportCrossBorderTourData(String outputFileBase) + { + addTable(outputFileBase); + String[] formats = {NUMBER_FORMAT_NAME, // id + NUMBER_FORMAT_NAME, // purpose + STRING_FORMAT_NAME, // sentri + NUMBER_FORMAT_NAME, // poe + NUMBER_FORMAT_NAME, // departTime + NUMBER_FORMAT_NAME, // arriveTime + NUMBER_FORMAT_NAME, // originMGRA + NUMBER_FORMAT_NAME, // destinationMGRA + NUMBER_FORMAT_NAME, // origTaz + NUMBER_FORMAT_NAME, // destTaz + NUMBER_FORMAT_NAME, // tourMode + NUMBER_FORMAT_NAME, // av_avail + NUMBER_FORMAT_NAME, // workTimeFactor + NUMBER_FORMAT_NAME, // nonWorkTimeFactor + NUMBER_FORMAT_NAME // valueOfTime + }; + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(Arrays.asList("sentri")); + Set primaryKey = new LinkedHashSet(Arrays.asList("TOURID")); + Map overridingNames = new HashMap(); + overridingNames.put("id", "TOURID"); + exportDataGeneric(outputFileBase, "crossBorder.tour.output.file", false, formats, + floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + overridingNames, null,true); + } + + private void exportCrossBorderTripData(String outputFileBase) + { + addTable(outputFileBase); + String[] formats = { + NUMBER_FORMAT_NAME, // 1 tourID + NUMBER_FORMAT_NAME, // 2 tripID + NUMBER_FORMAT_NAME, // 3 originPurp + NUMBER_FORMAT_NAME, // 4 destPurp + NUMBER_FORMAT_NAME, // 5 originMGRA + NUMBER_FORMAT_NAME, // 6 destinationMGRA + NUMBER_FORMAT_NAME, // 7 originTAZ + NUMBER_FORMAT_NAME, // 8 destinationTAZ + STRING_FORMAT_NAME, // 9 inbound + STRING_FORMAT_NAME, // 10 originIsTourDestination + STRING_FORMAT_NAME, // 11 destinationIsTourDestination + NUMBER_FORMAT_NAME, // 12 period + NUMBER_FORMAT_NAME, // 13 tripMode + NUMBER_FORMAT_NAME, // 14 av_avail + NUMBER_FORMAT_NAME, // 15 boardingTap + NUMBER_FORMAT_NAME, // 16 alightingTap + NUMBER_FORMAT_NAME, // 17 set + NUMBER_FORMAT_NAME, // 18 workTimeFactor + NUMBER_FORMAT_NAME, // 19 nonWorkTimeFactor + NUMBER_FORMAT_NAME // 20 valueOfTime + }; + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("workTimeFactor","nonWorkTimeFactor","valueOfTime")); + Set stringColumns = new HashSet(Arrays.asList("inbound", + "originIsTourDestination", "destinationIsTourDestination")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("tourID", "tripID")); + Map overridingNames = new HashMap(); + overridingNames.put("id", "TOURID"); + exportDataGeneric(outputFileBase, "crossBorder.trip.output.file", false, formats, + floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + overridingNames, new TripStructureDefinition(5, 6, 3, 4, 12, 13, 15, 16, -1, -1, 20, + "CB", 9, true, 20, 17,-1),true); + } + + private void exportVisitorData(String outputTourFileBase, String outputTripFileBase) + { + TableDataSet tourData = exportVisitorTourData(outputTourFileBase); + String tourIdField = "id"; + String partySizeField = "partySize"; + Map tourIdToPartySize = new HashMap(); + int[] ids = tourData.getColumnAsInt(tourIdField); + int[] partySize = tourData.getColumnAsInt(partySizeField); + for (int i = 0; i < ids.length; i++) + tourIdToPartySize.put(ids[i], partySize[i]); + exportVisitorTripData(outputTripFileBase, tourIdToPartySize); + } + + private TableDataSet exportVisitorTourData(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("id", "segment")); + Map overridingNames = new HashMap(); + // overridingNames.put("id","PARTYID"); + return exportDataGeneric(outputFileBase, "visitor.tour.output.file", false, null, + floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + overridingNames, null,false); + } + + private void exportVisitorTripData(String outputFileBase, Map tourIdToPartyMap) + { + addTable(outputFileBase); + String[] formats = { + NUMBER_FORMAT_NAME, // 1 tourID + NUMBER_FORMAT_NAME, // 2 tripID + NUMBER_FORMAT_NAME, // 3 originPurp + NUMBER_FORMAT_NAME, // 4 destPurp + NUMBER_FORMAT_NAME, // 5 originMGRA + NUMBER_FORMAT_NAME, // 6 destinationMGRA + STRING_FORMAT_NAME, // 7 inbound + STRING_FORMAT_NAME, // 8 originIsTourDestination + STRING_FORMAT_NAME, // 9 destinationIsTourDestination + NUMBER_FORMAT_NAME, // 10 period + NUMBER_FORMAT_NAME, // 11 tripMode + NUMBER_FORMAT_NAME, // 12 avAvailable + NUMBER_FORMAT_NAME, // 13 boardingTap + NUMBER_FORMAT_NAME, // 14 alightingTap + NUMBER_FORMAT_NAME, // 15 set + NUMBER_FORMAT_NAME, // 16 valueOfTime + NUMBER_FORMAT_NAME, // 17 partySize (added) + NUMBER_FORMAT_NAME, // 18 micro_walkMode + NUMBER_FORMAT_NAME, // 19 micro_trnAcc + NUMBER_FORMAT_NAME // 20 micro_trnEgr + + }; + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(Arrays.asList("inbound", + "originIsTourDestination", "destinationIsTourDestination")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("tourID", "tripId")); + primaryKey = new LinkedHashSet(Arrays.asList("RECID")); // todo: temporary until bugfix + //JoinData joinData = new JoinData("tourID"); + //joinData.addJoinData(tourIdToPartyMap, FieldType.INT, "partySize"); + exportDataGeneric( + outputFileBase, + "visitor.trip.output.file", + false, + formats, + floatColumns, + stringColumns, + intColumns, + bitColumns, + FieldType.INT, + primaryKey, + new TripStructureDefinition(5, 6, 3, 4, 10, 11, 13, 14, -1, 17, 20, "VISITOR", 7, true, 16,15,-1),false); + //, joinData); + } + + private void exportInternalExternalTripData(String outputFileBase) + { + addTable(outputFileBase); + String[] formats = { + NUMBER_FORMAT_NAME, // 1 hh_id + NUMBER_FORMAT_NAME, // 2 pnum + NUMBER_FORMAT_NAME, // 3 person_id + NUMBER_FORMAT_NAME, // 4 tour_id + NUMBER_FORMAT_NAME, // 5 originMGRA + NUMBER_FORMAT_NAME, // 6 destinationMGRA + NUMBER_FORMAT_NAME, // 7 originTAZ + NUMBER_FORMAT_NAME, // 8 destinationTAZ + STRING_FORMAT_NAME, // 9 inbound + STRING_FORMAT_NAME, // 10 originIsTourDestination + STRING_FORMAT_NAME, // 11 destinationIsTourDestination + NUMBER_FORMAT_NAME, // 12 period + NUMBER_FORMAT_NAME, // 13 tripMode + NUMBER_FORMAT_NAME, // 14 av_avail + NUMBER_FORMAT_NAME, // 15 boardingTap + NUMBER_FORMAT_NAME, // 16 alightingTap + NUMBER_FORMAT_NAME, // 17 set + NUMBER_FORMAT_NAME // 18 value of time + }; + Set intColumns = new HashSet(); + Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); + Set stringColumns = new HashSet(Arrays.asList("inbound", + "originIsTourDestination", "destinationIsTourDestination")); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(); + exportDataGeneric(outputFileBase, "internalExternal.trip.output.file", false, formats, + floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, + new TripStructureDefinition(5, 6, 12, 13, 15, 16, -1, -1, 18, "IE", "HOME", "EXTERNAL", + 9, true,18,17,-1),false); + } + + private Set getExternalZones() + { + Set externalZones = new LinkedHashSet(); + for (String zone : ((String) properties.get("external.tazs")).trim().split(",")) + externalZones.add(Integer.parseInt(zone.trim())); + return externalZones; + } + + /** + * Export commercial tNCVehicle data. + * + * @param outputFileBase + * @throws IOException + */ + private void exportCommVehData(String outputFileBase) throws IOException + { + addTable(outputFileBase); + Set internalZones = new LinkedHashSet(); + DecimalFormat formatter = new DecimalFormat("#.######"); + + BufferedWriter writer = null; + try + { + writer = new BufferedWriter(new FileWriter(new File(getOutputPath(outputFileBase + + ".csv"))), 1024 * 1024 * 1024); + + CsvRow headerRow = new CsvRow(new String[] {"ORIG_TAZ", "DEST_TAZ", "TOD", + "TRIPS_COMMVEH"}); + writer.write(headerRow.getRow()); + + for (String period : timePeriods) + { + Matrix matrixData = mtxDao.getMatrix("commVehTODTrips", period + " Trips"); + + // This doesn't make sense + if (internalZones.isEmpty()) for (int zone : matrixData.getExternalColumnNumbers()) + internalZones.add(zone); + + for (int i : internalZones) + { + for (int j : internalZones) + { + float value = matrixData.getValueAt(i, j); + if (value > .00001) + { + String[] rowValue = new String[4]; + rowValue[0] = String.valueOf(i); + rowValue[1] = String.valueOf(j); + rowValue[2] = period; + rowValue[3] = formatter.format(value); + CsvRow dataRow = new CsvRow(rowValue); + writer.write(dataRow.getRow()); + } + } + } + } + } finally + { + if (writer != null) writer.close(); + } + } + + /** + * Export commercial tNCVehicle data to OMX Format. + * + * @param outputFileBase + * @throws IOException + */ + private void exportCommVehDataToOmx(String outputFileBase) throws IOException + { + String[] modes = {"Toll","NonToll"}; + + addTable(outputFileBase); + for (String period : timePeriods){ + + Matrix[] matrices = new Matrix[modes.length]; + int counter = 0; + for(String mode : modes){ + + matrices[counter] = mtxDao.getMatrix("commVehTODTrips", period + " " + mode); + ++counter; + } + File outMatrixFile = new File(getOutputPath("commVeh_" + period + ".omx")); + MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX,outMatrixFile); + matrixWriter.writeMatrices(modes,matrices); + } + + } + private void exportExternalInternalTripData(String outputFileBase) + { + addTable(outputFileBase); + Set internalZones = new LinkedHashSet(); + Set externalZones = getExternalZones(); + List cores = Arrays.asList("DAN", "S2N", "S3N", "DAT", "S2T", "S3T"); + Map purposeMap = new HashMap(); + purposeMap.put("WORK", "Wrk"); + purposeMap.put("NONWORK", "Non"); + + Matrix[] matrixData = new Matrix[cores.size()]; + + PrintWriter writer = null; + try + { + writer = getBufferedPrintWriter(getOutputPath(outputFileBase + ".csv")); + + StringBuilder sb = new StringBuilder(); + sb.append("ORIG_TAZ,DEST_TAZ,TOD,PURPOSE"); + for (String core : cores) + sb.append(",").append("TRIPS_").append(core); + writer.println(sb.toString()); + + for (String period : timePeriods) + { + for (String purpose : purposeMap.keySet()) + { + int counter = 0; + for (String core : cores) + matrixData[counter++] = mtxDao.getMatrix("usSd" + purposeMap.get(purpose) + + "_" + period, core); + + if (internalZones.size() == 0) + { // only need to form internal zones once + for (int zone : matrixData[0].getExternalColumnNumbers()) + internalZones.add(zone); + internalZones.removeAll(externalZones); + } + + for (int i : internalZones) + { + for (int e : externalZones) + { + StringBuilder sbie = new StringBuilder(); + StringBuilder sbei = new StringBuilder(); + sbie.append(i).append(",").append(e).append(",").append(period) + .append(",").append(purpose); + sbei.append(e).append(",").append(i).append(",").append(period) + .append(",").append(purpose); + float ie = 0; + float ei = 0; + + for (Matrix matrix : matrixData) + { + float vie = matrix.getValueAt(i, e); + float vei = matrix.getValueAt(e, i); + ie += vie; + ei += vei; + sbie.append(",").append(vie); + sbei.append(",").append(vei); + } + if (ie > 0) writer.println(sbie.toString()); + if (ei > 0) writer.println(sbei.toString()); + } + } + } + } + + } catch (IOException e) + { + throw new RuntimeException(e); + } finally + { + if (writer != null) writer.close(); + } + } + + /** + * Export the external-internal trips to OMX format. Collapse out purposes. + * @param outputFileBase + */ + private void exportExternalInternalTripDataToOMX(String outputFileBase) + { + addTable(outputFileBase); + String[] cores = {"DAN", "S2N", "S3N", "DAT", "S2T", "S3T"}; + String[] purposes = {"Wrk","Non"}; + + Matrix[] outMatrixData = new Matrix[cores.length]; + + for (String period : timePeriods) + { + for(int p = 0; p externalZones = getExternalZones(); + + BufferedWriter writer = null; + MatrixWriter matrixWriter = null; + try + { + if(writeCSV){ + writer = new BufferedWriter(new FileWriter(new File(getOutputPath(outputFileBase + + ".csv"))), 1024 * 1024 * 1024); + + CsvRow headerRow = new CsvRow(new String[] {"ORIG_TAZ", "DEST_TAZ", "TRIPS_EE"}); + writer.write(headerRow.getRow()); + }else{ + matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, new File(getOutputPath(outputFileBase + ".omx"))); + } + + Matrix m = mtxDao.getMatrix("externalExternalTrips", "Trips"); + + if(writeCSV){ + for (int o : externalZones) + { + for (int d : externalZones) + { + String[] values = new String[3]; + values[0] = String.valueOf(o); + values[1] = String.valueOf(d); + values[2] = String.valueOf(m.getValueAt(o, d)); + CsvRow dataRow = new CsvRow(values); + writer.write(dataRow.getRow()); + } + } + }else{ + matrixWriter.writeMatrix(m); + } + } finally + { + if (writer != null) writer.close(); + } + + } + + /** + * A private helper class to organize skims + * + * @author joel.freedman + * + */ + private class AutoSkimSet{ + + String fileName; + String[] skimNames; + + AutoSkimSet(String fileName, String[] skimNames){ + this.fileName = fileName; + this.skimNames = skimNames; + } + } + + /** + * Return a map containing a number of elements where key is the name of the skim file and + * value is the name of a matrix core in the skim file. The map includes length and time for + * "free" path skims and length, time and toll for toll skims. + * + * @return The map. + */ + private HashMap getVehicleSkimFileCoreNameMapping() + { + HashMap map = new HashMap(); + + String[] votBins = {"L","M","H"}; + + for(int i = 1; i< votBins.length;++i){ + + // DA Non-Toll + AutoSkimSet SOVGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_SOVGP"+votBins[i]+"_DIST", + TOD_TOKEN+"_SOVGP"+votBins[i]+"_TIME", + TOD_TOKEN+"_SOVGP"+votBins[i]+"_REL"}); + map.put("SOVGP"+votBins[i], SOVGP); + + // DA Toll + AutoSkimSet SOVTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_DIST", + TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_TIME", + TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_TOLLCOST", + TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_REL"}); + map.put("SOTOLL"+votBins[i], SOVTOLL); + + // S2 Non-Toll + AutoSkimSet HOV2HOV = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_DIST", + TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_TIME", + TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_REL"}); + map.put("HOV2HOV"+votBins[i], HOV2HOV); + + // S2 Toll + AutoSkimSet HOV2TOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_DIST", + TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_TIME", + TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_TOLLCOST", + TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_REL"}); + map.put("HOV2TOLL"+votBins[i], HOV2TOLL); + + // S3+ Non-Toll + AutoSkimSet HOV3HOV = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_DIST", + TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_TIME", + TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_REL"}); + map.put("HOV3HOV"+votBins[i], HOV3HOV); + + // S3+ Toll + AutoSkimSet HOV3TOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_DIST", + TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_TIME", + TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_TOLLCOST", + TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_REL"}); + map.put("HOV3TOLL"+votBins[i], HOV3TOLL); + + } + + // Light Truck GP + AutoSkimSet TRKLGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKLGP_DIST", + TOD_TOKEN+"_TRKLGP_TIME"}); + map.put("TRKLGP", TRKLGP); + + // Light Truck Toll + AutoSkimSet TRKLTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKLTOLL_DIST", + TOD_TOKEN+"_TRKLTOLL_TIME", + TOD_TOKEN+"_TRKLTOLL_TOLLCOST"}); + map.put("TRKLTOLL", TRKLTOLL); + + + // Medium Truck GP + AutoSkimSet TRKMGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKMGP_DIST", + TOD_TOKEN+"_TRKMGP_TIME"}); + map.put("TRKMGP", TRKMGP); + + // Medium Truck Toll + AutoSkimSet TRKMTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKMTOLL_DIST", + TOD_TOKEN+"_TRKMTOLL_TIME", + TOD_TOKEN+"_TRKMTOLL_TOLLCOST"}); + map.put("TRKMTOLL", TRKMTOLL); + + // Heavy Truck GP + AutoSkimSet TRKHGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKHGP_DIST", + TOD_TOKEN+"_TRKHGP_TIME"}); + map.put("TRKHGP", TRKHGP); + + // Heavy Truck Toll + AutoSkimSet TRKHTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { + TOD_TOKEN+"_TRKHTOLL_DIST", + TOD_TOKEN+"_TRKHTOLL_TIME", + TOD_TOKEN+"_TRKHTOLL_TOLLCOST"}); + map.put("TRKHTOLL", TRKHTOLL); + + + return map; + } + + /** + * Export auto skims to the directory using both csv and omx formats. The CSV file will be + * written if writeCSV is true. Otherwise OMX files will be written. The OMX files will also + * contain an auto operating cost matrix. + * + * @param outputFileBase The name of output csv file to write to the reports directory. + */ + private void exportAutoSkims(String outputFileBase) + { + addTable(outputFileBase); + String[] includedTimePeriods = getTimePeriodsForSkims(); + Set internalZones = new LinkedHashSet(); + String path = properties.getProperty("report.path"); + + BlockingQueue queue = new LinkedBlockingQueue(); + try + { + + Map vehicleSkimCores = getVehicleSkimFileCoreNameMapping(); + + boolean first = true; + + ArrayList modeNames = new ArrayList(); + + for (String period : includedTimePeriods) + { + Map lengthMatrix = new LinkedHashMap(); + Map timeMatrix = new LinkedHashMap(); + Map tollMatrix = new LinkedHashMap(); + Map stdMatrix = new LinkedHashMap(); + + + //iterate through the auto modes + for (String key : vehicleSkimCores.keySet()) + { + // String name = vehicleSkimFiles.get(key); + AutoSkimSet skimSet = vehicleSkimCores.get(key); + String skimFileName = skimSet.fileName; + String[] inputMatrixNames = skimSet.skimNames; + + // the first skim is always distance. Remove it to get the name of the mode. + modeNames.add(key+"_"+TOD_TOKEN); //store all the modes + + int stdMatrixNumber=-1; + int tollMatrixNumber=-1; + + //need to replace the TOD token with the period name for matrices to output to OMX + String[] outCores = new String[inputMatrixNames.length+1]; + for(int i =0; i < (outCores.length-1);++i){ + outCores[i] = inputMatrixNames[i].replace(TOD_TOKEN, period); + if(inputMatrixNames[i].contains("REL")) + stdMatrixNumber=i; + if(inputMatrixNames[i].contains("TOLLCOST")) + tollMatrixNumber=i; + } + + skimFileName = skimFileName.replace(TOD_TOKEN,period); + Matrix length = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[0].replace(TOD_TOKEN, period)); + Matrix time = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[1].replace(TOD_TOKEN, period)); + Matrix aoc = length.multiply(autoOperatingCost); + + String aocName = outCores[0].replace("_DIST", "_AOC"); + outCores[outCores.length-1] = aocName; + + String outputFileName = path+key+"_"+period+".omx"; + + MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, new File(outputFileName)); + + Matrix[] matrices = new Matrix[inputMatrixNames.length+1]; + matrices[0] = length; + matrices[1] = time; + + lengthMatrix.put(inputMatrixNames[0],length); + timeMatrix.put(inputMatrixNames[1], time); + + + int matrixNumber=2; + if(stdMatrixNumber>-1){ + Matrix std = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[stdMatrixNumber].replace(TOD_TOKEN, period)); + matrices[matrixNumber]= std; + stdMatrix.put(inputMatrixNames[matrixNumber], std); + ++matrixNumber; + } + if(tollMatrixNumber>-1){ + Matrix cost = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[tollMatrixNumber].replace(TOD_TOKEN, period)); + matrices[matrixNumber]= cost; + + ++matrixNumber; + } + + matrices[matrixNumber] = aoc; + + LOGGER.info("Writing "+outCores.length+" skims to file "+outputFileName); + matrixWriter.writeMatrices(outCores, matrices); + + if(writeCSV){ + if (internalZones.size() == 0) + { + boolean f = true; + for (int zone : lengthMatrix.get(inputMatrixNames[0]).getExternalColumnNumbers()) + { + if (f) + { + f = false; + continue; + } + internalZones.add(zone); + } + } + + // put data into arrays for faster access + Matrix[] orderedData = new Matrix[lengthMatrix.size() + timeMatrix.size() + + stdMatrix.size() + tollMatrix.size()]; + int counter = 0; + for (String mode : modeNames) + { + orderedData[counter++] = lengthMatrix.get(mode); + orderedData[counter++] = timeMatrix.get(mode); + orderedData[counter++] = stdMatrix.get(mode); + if (tollMatrix.containsKey(mode)) + orderedData[counter++] = tollMatrix.get(mode); + } + + if (first) + { + List header = new ArrayList(); + header.add("ORIG_TAZ"); + header.add("DEST_TAZ"); + header.add("TOD"); + + for (String modeName : modeNames) + { + header.add("DIST_" + modeName); + header.add("TIME_" + modeName); + header.add("STD_TIME_" + modeName); + if (tollMatrix.containsKey(modeName)) + { + header.add("COST_" + modeName); + } + } + + CsvWriterThread writerThread = new CsvWriterThread(queue, new File( + getOutputPath(outputFileBase + ".csv")), + header.toArray(new String[header.size()])); + new Thread(writerThread).start(); + first = false; + } + + int rowSize = 3 + orderedData.length; + + for (int i : internalZones) + { + for (int j : internalZones) + { + String[] values = new String[rowSize]; + values[0] = String.valueOf(i); + values[1] = String.valueOf(j); + values[2] = period; + int position = 3; + for (Matrix matrix : orderedData) + values[position++] = DoubleFormatUtil.formatDouble( + matrix.getValueAt(i, j), 4, 4); + queue.add(new CsvRow(values)); + } + } + } + } + } + } finally + { + queue.add(CsvWriterThread.POISON_PILL); + } + } + + private Map getTransitSkimFileNameMapping() + { + Map map = new LinkedHashMap(); + // map.put("implocl_" + TOD_TOKEN + "o", "LOCAL_TRANSIT"); + map.put("impprem_" + TOD_TOKEN + "o", "PREMIUM_TRANSIT"); + return map; + } + + private String getTransitSkimFileFareCoreName() + { + return "Fare"; + } + + private Map getTransitSkimFileInVehicleTimeCoreNameMapping() + { // distance,time,cost + Map map = new LinkedHashMap(); + map.put("impprem_" + TOD_TOKEN + "o", new String[] {"IVT:CR", "IVT:LR", "IVT:BRT", + "IVT:EXP", "IVT:LB"}); + return map; + } + + private String[] getTimePeriodsForSkims() + { + return IExporter.TOD_TOKENS; + } + + /** + * This method reads the transit skims and exports them to OMX format. It will also write + * csv file of skim values if the writeCSVSkims attribute is set to true. + * + * @param outputFileBase + */ + private void exportTransitSkims(String outputFileBase) + { + addTable(outputFileBase); + String[] includedTimePeriods = getTimePeriodsForSkims(); + + Set internalZones = new LinkedHashSet(); + + BlockingQueue queue = new LinkedBlockingQueue(); + try + { + Map transitSkimFiles = getTransitSkimFileNameMapping(); + Map transitSkimTimeCores = getTransitSkimFileInVehicleTimeCoreNameMapping(); + String fareCore = getTransitSkimFileFareCoreName(); + String initialWaitCore = "Initial Wait Time"; + String transferTimeCore = "Transfer Wait Time"; + String walkTimeCore = "Walk Time"; + Set modeNames = new LinkedHashSet(); + for (String n : transitSkimFiles.keySet()) + modeNames.add(transitSkimFiles.get(n)); + boolean first = true; + int numOfColumns = 3 + 5 * modeNames.size(); + for (String period : includedTimePeriods) + { + Map timeMatrix = new LinkedHashMap(); + Map fareMatrix = new LinkedHashMap(); + Map initialMatrix = new LinkedHashMap(); + Map transferMatrix = new LinkedHashMap(); + Map walkTimeMatrix = new LinkedHashMap(); + + for (String key : transitSkimFiles.keySet()) + { + String name = transitSkimFiles.get(key); + String[] timeCores = transitSkimTimeCores.get(key); + String file = key.replace(TOD_TOKEN, period); + Matrix[] timeMatrices = new Matrix[timeCores.length]; + for (int i = 0; i < timeCores.length; i++) + timeMatrices[i] = mtxDao.getMatrix(file, + timeCores[i].replace(TOD_TOKEN, period)); + timeMatrix.put(name, timeMatrices); + fareMatrix.put(name, + mtxDao.getMatrix(file, fareCore.replace(TOD_TOKEN, period))); + initialMatrix.put(name, mtxDao.getMatrix(file, initialWaitCore)); + transferMatrix.put(name, mtxDao.getMatrix(file, transferTimeCore)); + walkTimeMatrix.put(name, mtxDao.getMatrix(file, walkTimeCore)); + if (internalZones.size() == 0) + { + boolean f = true; + for (int zone : fareMatrix.get(name).getExternalColumnNumbers()) + { + if (f) + { + f = false; + continue; + } + internalZones.add(zone); + } + } + } + + // put data into arrays for faster access + Matrix[][] orderedTimeData = new Matrix[timeMatrix.size()][]; + Matrix[] fareData = new Matrix[orderedTimeData.length]; + Matrix[] initialWaitData = new Matrix[orderedTimeData.length]; + Matrix[] transferTimeData = new Matrix[orderedTimeData.length]; + Matrix[] walkTimeData = new Matrix[orderedTimeData.length]; + + int counter = 0; + for (String mode : modeNames) + { + orderedTimeData[counter] = timeMatrix.get(mode); + fareData[counter] = fareMatrix.get(mode); + initialWaitData[counter] = initialMatrix.get(mode); + transferTimeData[counter] = transferMatrix.get(mode); + walkTimeData[counter++] = walkTimeMatrix.get(mode); + } + + if (first) + { + String[] header = new String[numOfColumns]; + + header[0] = "ORIG_TAP"; + header[1] = "DEST_TAP"; + header[2] = "TOD"; + int column = 3; + + for (String modeName : modeNames) + { + header[column++] = "TIME_INIT_WAIT_" + modeName; + header[column++] = "TIME_IVT_TIME_" + modeName; + header[column++] = "TIME_WALK_TIME_" + modeName; + header[column++] = "TIME_TRANSFER_TIME_" + modeName; + header[column++] = "FARE_" + modeName; + } + + CsvWriterThread writerThread = new CsvWriterThread(queue, new File( + getOutputPath(outputFileBase + ".csv")), header); + new Thread(writerThread).start(); + + first = false; + } + + for (int i : internalZones) + { + for (int j : internalZones) + { + String[] values = new String[numOfColumns]; + values[0] = String.valueOf(i); + values[1] = String.valueOf(j); + values[2] = period; + + int column = 3; + float runningTotal = 0.0f; + + for (int m = 0; m < orderedTimeData.length; m++) + { + float time = 0.0f; + float initTime = initialWaitData[m].getValueAt(i, j); + for (Matrix tm : orderedTimeData[m]) + time += tm.getValueAt(i, j); + float walkTime = walkTimeData[m].getValueAt(i, j); + float transferTime = transferTimeData[m].getValueAt(i, j); + float fare = fareData[m].getValueAt(i, j); + runningTotal += fare + time; + values[column++] = DoubleFormatUtil.formatDouble(initTime, 4, 4); + values[column++] = DoubleFormatUtil.formatDouble(time, 4, 4); + values[column++] = DoubleFormatUtil.formatDouble(walkTime, 4, 4); + values[column++] = DoubleFormatUtil.formatDouble(transferTime, 4, 4); + values[column++] = DoubleFormatUtil.formatDouble(fare, 2, 2); + } + if (runningTotal > 0.0f) queue.add(new CsvRow(values)); + } + } + } + + } finally + { + queue.add(CsvWriterThread.POISON_PILL); + } + } + + private void exportDefinitions(String outputFileBase) + { + addTable(outputFileBase); + Map tripPurposes = new LinkedHashMap(); + Map modes = new LinkedHashMap(); + Map ejCategories = new LinkedHashMap(); + + PrintWriter writer = null; + try + { + writer = getBufferedPrintWriter(getOutputPath(outputFileBase + ".csv")); + writer.println("type,code,description"); + writer.println("nothing,placeholder,this describes nothing"); + for (String tripPurpose : tripPurposes.keySet()) + writer.println("trip_purpose," + tripPurpose + "," + tripPurposes.get(tripPurpose)); + for (String mode : modes.keySet()) + writer.println("mode," + mode + "," + modes.get(mode)); + for (String ejCategory : ejCategories.keySet()) + writer.println("ej_category," + ejCategory + "," + ejCategories.get(ejCategory)); + } catch (IOException e) + { + throw new RuntimeException(e); + } finally + { + if (writer != null) writer.close(); + } + } + + private void exportPnrVehicleData(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(Arrays.asList("TAP")); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("TAP")); + exportDataGeneric(outputFileBase, "Results.PNRFile", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private void exportCbdVehicleData(String outputFileBase) + { + addTable(outputFileBase); + Set intColumns = new HashSet(Arrays.asList("MGRA")); + Set floatColumns = new HashSet(); + Set stringColumns = new HashSet(); + Set bitColumns = new HashSet(); + Set primaryKey = new LinkedHashSet(Arrays.asList("MGRA")); + exportDataGeneric(outputFileBase, "Results.CBDFile", false, null, floatColumns, + stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); + } + + private static enum FieldType + { + INT, FLOAT, STRING, BIT + } + + private final class TripStructureDefinition + { + private final int originMgraColumn; + private final int destMgraColumn; + private final int originPurposeColumn; + private final int destinationPurposeColumn; + private final int todColumn; + private final int modeColumn; + private final int boardTapColumn; + private final int alightTapColumn; + + private final int parkingMazColumn; + + private final String homeName; + private final String destinationName; + private final int inboundColumn; + private final boolean booleanIndicatorVariables; + private final int valueOfTimeColumn; + private final int setColumn; + private final int transponderOwnershipColumn; + + private TripStructureDefinition(int originMgraColumn, int destMgraColumn, + int originPurposeColumn, int destinationPurposeColumn, int todColumn, + int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, + int tripTimeColumn, int outVehicleTimeColumn, int tripDistanceColumn, + int tripCostColumn, int tripPurposeNameColumn, int tripModeNameColumn, + int recIdColumn, int boardTazColumn, int alightTazColumn, String tripType, + String homeName, String destinationName, int inboundColumn, + boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) + { + this.originMgraColumn = originMgraColumn; + this.destMgraColumn = destMgraColumn; + this.originPurposeColumn = originPurposeColumn; + this.destinationPurposeColumn = destinationPurposeColumn; + this.todColumn = todColumn; + this.modeColumn = modeColumn; + this.boardTapColumn = boardTapColumn; + this.alightTapColumn = alightTapColumn; + this.parkingMazColumn = parkingMazColumn; + this.homeName = homeName; + this.destinationName = destinationName; + this.inboundColumn = inboundColumn; + + this.booleanIndicatorVariables = booleanIndicatorVariables; + this.valueOfTimeColumn = valueOfTimeColumn; + this.setColumn = setColumn; + this.transponderOwnershipColumn = transponderOwnershipColumn; + } + + private TripStructureDefinition(int originMgraColumn, int destMgraColumn, + int originPurposeColumn, int destinationPurposeColumn, int todColumn, + int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, + int columnCount, String tripType, int inboundColumn, + boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) + { + this(originMgraColumn, destMgraColumn, originPurposeColumn, destinationPurposeColumn, + todColumn, modeColumn, boardTapColumn, alightTapColumn, parkingMazColumn, partySizeColumn, + columnCount + 1, columnCount + 2, columnCount + 3, columnCount + 4, + columnCount + 5, columnCount + 6, columnCount + 7, columnCount + 8, + columnCount + 9, tripType, "", "", inboundColumn, booleanIndicatorVariables, valueOfTimeColumn,setColumn,transponderOwnershipColumn); + } + + private TripStructureDefinition(int originMgraColumn, int destMgraColumn, int todColumn, + int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, + int columnCount, String tripType, String homeName, String destinationName, + int inboundColumn, boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) + { + this(originMgraColumn, destMgraColumn, -1, -1, todColumn, modeColumn, boardTapColumn, + alightTapColumn, parkingMazColumn, partySizeColumn, columnCount + 1, columnCount + 2, + columnCount + 3, columnCount + 4, columnCount + 5, columnCount + 6, + columnCount + 7, columnCount + 8, columnCount + 9, tripType, homeName, + destinationName, inboundColumn, booleanIndicatorVariables, valueOfTimeColumn,setColumn,transponderOwnershipColumn); + } + } + + public static void main(String... args) throws Exception + { + String propertiesFile = null; + propertiesFile = args[0]; + + Properties properties = new Properties(); + properties.load(new FileInputStream("conf/sandag_abm.properties")); + HashMap pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + + int feedbackIteration = Integer.valueOf(properties.getProperty("Report.iteration").trim()); + + List definedTables = new ArrayList(); + for (String table : properties.getProperty("Report.tables").trim().split(",")) + definedTables.add(table.trim().toLowerCase()); + + String path = ClassLoader.getSystemResource("").getPath(); + path = path.substring(1, path.length() - 2); + String appPath = path.substring(0, path.lastIndexOf("/")); + + for (Object key : properties.keySet()) + { + String value = (String) properties.get(key); + properties.setProperty((String) key, + value.replace(PROJECT_PATH_PROPERTY_TOKEN, appPath)); + } + + OMXMatrixDao mtxDao = new OMXMatrixDao(properties); + + DataExporter dataExporter = new DataExporter(properties, mtxDao, appPath, feedbackIteration); + dataExporter.startMatrixServer(pMap); + + if (definedTables.contains("accessibilities")) + dataExporter.exportAccessibilities("accessibilities"); + if (definedTables.contains("mgra")) dataExporter.exportMazData("mgra"); + if (definedTables.contains("taz")) dataExporter.exportTazData("taz"); + if (definedTables.contains("tap")) dataExporter.exportTapData("tap"); + if (definedTables.contains("mgratotap")) dataExporter.exportMgraToTapData("mgratotap"); + if (definedTables.contains("mgratomgra")) dataExporter.exportMgraToMgraData("mgratomgra"); + if (definedTables.contains("taztotap")) dataExporter.exportTazToTapData("taztotap"); + if (definedTables.contains("hhdata")) dataExporter.exportHouseholdData("hhdata"); + if (definedTables.contains("persondata")) dataExporter.exportPersonData("persondata"); + if (definedTables.contains("wslocation")) + dataExporter.exportWorkSchoolLocation("wslocation"); + if (definedTables.contains("synhh")) dataExporter.exportSyntheticHouseholdData("synhh"); + if (definedTables.contains("synperson")) + dataExporter.exportSyntheticPersonData("synperson"); + if (definedTables.contains("indivtours")) dataExporter.exportIndivToursData("indivtours"); + if (definedTables.contains("jointtours")) dataExporter.exportJointToursData("jointtours"); + if (definedTables.contains("indivtrips")) dataExporter.exportIndivTripData("indivtrips"); + if (definedTables.contains("jointtrips")) dataExporter.exportJointTripData("jointtrips"); + if (definedTables.contains("airporttripssan")) + dataExporter.exportAirportTripsSAN("airporttripssan"); + if (definedTables.contains("airporttripscbx")) + dataExporter.exportAirportTripsCBX("airporttripscbx"); + if (definedTables.contains("cbtours")) dataExporter.exportCrossBorderTourData("cbtours"); + if (definedTables.contains("cbtrips")) dataExporter.exportCrossBorderTripData("cbtrips"); + if (definedTables.contains("visitortours") && definedTables.contains("visitortrips")) + dataExporter.exportVisitorData("visitortours", "visitortrips"); + if (definedTables.contains("ietrip")) + dataExporter.exportInternalExternalTripData("ietrip"); + if (definedTables.contains("commtrip")){ + CVMExporter cvmExporter = new CVMExporter(properties,mtxDao); + cvmExporter.export(); + CVMScaler cvmScaler = new CVMScaler(properties); + cvmScaler.scale(); + } + + + if (definedTables.contains("trucktrip")) + { + if(dataExporter.writeCSV){ + IExporter truckExporter = new TruckCsvExporter(properties, mtxDao, "trucktrip"); + truckExporter.export(); + }else{ + IExporter truckExporter = new TruckOmxExporter(properties, mtxDao, "trucktrip"); + truckExporter.export(); + } + } + if (definedTables.contains("eetrip")) + dataExporter.exportExternalExternalTripData("eetrip"); + + if (definedTables.contains("eitrip")) + if(dataExporter.writeCSV) + dataExporter.exportExternalInternalTripData("eitrip"); + else + dataExporter.exportExternalInternalTripDataToOMX("eitrip"); + + if (definedTables.contains("tazskim")) dataExporter.exportAutoSkims("tazskim"); + if (definedTables.contains("tapskim")) dataExporter.exportTransitSkims("tapskim"); + if (definedTables.contains("definition")) dataExporter.exportDefinitions("definition"); + if (definedTables.contains("pnrvehicles")) + dataExporter.exportPnrVehicleData("pnrvehicles"); + if (definedTables.contains("cbdvehicles")) + dataExporter.exportCbdVehicleData("cbdvehicles"); + } + + private void startMatrixServer(HashMap properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + LOGGER.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + LOGGER.error("could not connect to matrix server"); + LOGGER.info("Running Data Exporter with internal matrix class"); + // throw new RuntimeException(e); + + } + + } + + /** + * Startup a connection to the matrix manager. + * + * @param serverAddress + * @param serverPort + * @param mt + * @return + */ + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + try + { + // create the concrete data server object + matrixServer.start32BitMatrixIoServer(mt); + } catch (RuntimeException e) + { + matrixServer.stop32BitMatrixIoServer(); + LOGGER.error( + "RuntimeException caught making remote method call to start 32 bit mitrix in remote MatrixDataServer.", + e); + } + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + LOGGER.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + matrixServer.stop32BitMatrixIoServer(); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + LOGGER.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + matrixServer.stop32BitMatrixIoServer(); + throw new RuntimeException(); + } + + return matrixServer; + + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java new file mode 100644 index 0000000..19f2c7d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java @@ -0,0 +1,484 @@ +package org.sandag.abm.reporting; +/* + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with this + * work for additional information regarding copyright ownership. The ASF + * licenses this file to You 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. + */ + +/* $Id$ */ + +/** + * This class implements fast, thread-safe format of a double value with a given + * number of decimal digits. + *

+ * The contract for the format methods is this one: if the source is greater + * than or equal to 1 (in absolute value), use the decimals parameter to define + * the number of decimal digits; else, use the precision parameter to define the + * number of decimal digits. + *

+ * A few examples (consider decimals being 4 and precision being 8): + *

    + *
  • 0.0 should be rendered as "0" + *
  • 0.1 should be rendered as "0.1" + *
  • 1234.1 should be rendered as "1234.1" + *
  • 1234.1234567 should be rendered as "1234.1235" (note the trailing 5! + * Rounding!) + *
  • 1234.00001 should be rendered as "1234" + *
  • 0.00001 should be rendered as "0.00001" (here you see the effect of the + * "precision" parameter) + *
  • 0.00000001 should be rendered as "0.00000001" + *
  • 0.000000001 should be rendered as "0" + *
+ * + * Originally authored by Julien Aymé. + */ +public final class DoubleFormatUtil +{ + + private DoubleFormatUtil() + { + } + + public static String formatDouble(double source, int decimals, int precision) + { + StringBuffer target = new StringBuffer(); + int scale = (Math.abs(source) >= 1.0) ? decimals : precision; + if (tooManyDigitsUsed(source, scale) || tooCloseToRound(source, scale)) + { + formatDoublePrecise(source, decimals, precision, target); + } else + { + formatDoubleFast(source, decimals, precision, target); + } + + return target.toString(); + } + + /** + * Rounds the given source value at the given precision and writes the + * rounded value into the given target + * + * @param source + * the source value to round + * @param decimals + * the decimals to round at (use if abs(source) ≥ 1.0) + * @param precision + * the precision to round at (use if abs(source) < 1.0) + * @param target + * the buffer to write to + */ + public static void formatDouble(double source, int decimals, int precision, StringBuffer target) + { + int scale = (Math.abs(source) >= 1.0) ? decimals : precision; + if (tooManyDigitsUsed(source, scale) || tooCloseToRound(source, scale)) + { + formatDoublePrecise(source, decimals, precision, target); + } else + { + formatDoubleFast(source, decimals, precision, target); + } + } + + /** + * Rounds the given source value at the given precision and writes the + * rounded value into the given target + *

+ * This method internally uses the String representation of the source + * value, in order to avoid any double precision computation error. + * + * @param source + * the source value to round + * @param decimals + * the decimals to round at (use if abs(source) ≥ 1.0) + * @param precision + * the precision to round at (use if abs(source) < 1.0) + * @param target + * the buffer to write to + */ + public static void formatDoublePrecise(double source, int decimals, int precision, + StringBuffer target) + { + if (isRoundedToZero(source, decimals, precision)) + { + // Will always be rounded to 0 + target.append('0'); + return; + } else if (Double.isNaN(source) || Double.isInfinite(source)) + { + // Cannot be formated + target.append(Double.toString(source)); + return; + } + + boolean negative = source < 0.0; + if (negative) + { + source = -source; + // Done once and for all + target.append('-'); + } + int scale = (source >= 1.0) ? decimals : precision; + + // The only way to format precisely the double is to use the String + // representation of the double, and then to do mathematical integer + // operation on it. + String s = Double.toString(source); + if (source >= 1e-3 && source < 1e7) + { + // Plain representation of double: "intPart.decimalPart" + int dot = s.indexOf('.'); + String decS = s.substring(dot + 1); + int decLength = decS.length(); + if (scale >= decLength) + { + if ("0".equals(decS)) + { + // source is a mathematical integer + target.append(s.substring(0, dot)); + } else + { + target.append(s); + // Remove trailing zeroes + for (int l = target.length() - 1; l >= 0 && target.charAt(l) == '0'; l--) + { + target.setLength(l); + } + } + return; + } else if (scale + 1 < decLength) + { + // ignore unnecessary digits + decLength = scale + 1; + decS = decS.substring(0, decLength); + } + long intP = Long.parseLong(s.substring(0, dot)); + long decP = Long.parseLong(decS); + format(target, scale, intP, decP); + } else + { + // Scientific representation of double: "x.xxxxxEyyy" + int dot = s.indexOf('.'); + assert dot >= 0; + int exp = s.indexOf('E'); + assert exp >= 0; + int exposant = Integer.parseInt(s.substring(exp + 1)); + String intS = s.substring(0, dot); + String decS = s.substring(dot + 1, exp); + int decLength = decS.length(); + if (exposant >= 0) + { + int digits = decLength - exposant; + if (digits <= 0) + { + // no decimal part, + // no rounding involved + target.append(intS); + target.append(decS); + for (int i = -digits; i > 0; i--) + { + target.append('0'); + } + } else if (digits <= scale) + { + // decimal part precision is lower than scale, + // no rounding involved + target.append(intS); + target.append(decS.substring(0, exposant)); + target.append('.'); + target.append(decS.substring(exposant)); + } else + { + // decimalDigits > scale, + // Rounding involved + long intP = Long.parseLong(intS) * tenPow(exposant) + + Long.parseLong(decS.substring(0, exposant)); + long decP = Long.parseLong(decS.substring(exposant, exposant + scale + 1)); + format(target, scale, intP, decP); + } + } else + { + // Only a decimal part is supplied + exposant = -exposant; + int digits = scale - exposant + 1; + if (digits < 0) + { + target.append('0'); + } else if (digits == 0) + { + long decP = Long.parseLong(intS); + format(target, scale, 0L, decP); + } else if (decLength < digits) + { + long decP = Long.parseLong(intS) * tenPow(decLength + 1) + Long.parseLong(decS) + * 10; + format(target, exposant + decLength, 0L, decP); + } else + { + long subDecP = Long.parseLong(decS.substring(0, digits)); + long decP = Long.parseLong(intS) * tenPow(digits) + subDecP; + format(target, scale, 0L, decP); + } + } + } + } + + /** + * Returns true if the given source value will be rounded to zero + * + * @param source + * the source value to round + * @param decimals + * the decimals to round at (use if abs(source) ≥ 1.0) + * @param precision + * the precision to round at (use if abs(source) < 1.0) + * @return true if the source value will be rounded to zero + */ + private static boolean isRoundedToZero(double source, int decimals, int precision) + { + // Use 4.999999999999999 instead of 5 since in some cases, 5.0 / 1eN > + // 5e-N (e.g. for N = 37, 42, 45, 66, ...) + return source == 0.0 + || Math.abs(source) < 4.999999999999999 / tenPowDouble(Math + .max(decimals, precision) + 1); + } + + /** + * Most used power of ten (to avoid the cost of Math.pow(10, n) + */ + private static final long[] POWERS_OF_TEN_LONG = new long[19]; + private static final double[] POWERS_OF_TEN_DOUBLE = new double[30]; + static + { + POWERS_OF_TEN_LONG[0] = 1L; + for (int i = 1; i < POWERS_OF_TEN_LONG.length; i++) + { + POWERS_OF_TEN_LONG[i] = POWERS_OF_TEN_LONG[i - 1] * 10L; + } + for (int i = 0; i < POWERS_OF_TEN_DOUBLE.length; i++) + { + POWERS_OF_TEN_DOUBLE[i] = Double.parseDouble("1e" + i); + } + } + + /** + * Returns ten to the power of n + * + * @param n + * the nth power of ten to get + * @return ten to the power of n + */ + public static long tenPow(int n) + { + assert n >= 0; + return n < POWERS_OF_TEN_LONG.length ? POWERS_OF_TEN_LONG[n] : (long) Math.pow(10, n); + } + + private static double tenPowDouble(int n) + { + assert n >= 0; + return n < POWERS_OF_TEN_DOUBLE.length ? POWERS_OF_TEN_DOUBLE[n] : Math.pow(10, n); + } + + /** + * Helper method to do the custom rounding used within formatDoublePrecise + * + * @param target + * the buffer to write to + * @param scale + * the expected rounding scale + * @param intP + * the source integer part + * @param decP + * the source decimal part, truncated to scale + 1 digit + */ + private static void format(StringBuffer target, int scale, long intP, long decP) + { + if (decP != 0L) + { + // decP is the decimal part of source, truncated to scale + 1 digit. + // Custom rounding: add 5 + decP += 5L; + decP /= 10L; + if (decP >= tenPowDouble(scale)) + { + intP++; + decP -= tenPow(scale); + } + if (decP != 0L) + { + // Remove trailing zeroes + while (decP % 10L == 0L) + { + decP = decP / 10L; + scale--; + } + } + } + target.append(intP); + if (decP != 0L) + { + target.append('.'); + // Use tenPow instead of tenPowDouble for scale below 18, + // since the casting of decP to double may cause some imprecisions: + // E.g. for decP = 9999999999999999L and scale = 17, + // decP < tenPow(16) while (double) decP == tenPowDouble(16) + while (scale > 0 + && (scale > 18 ? decP < tenPowDouble(--scale) : decP < tenPow(--scale))) + { + // Insert leading zeroes + target.append('0'); + } + target.append(decP); + } + } + + /** + * Rounds the given source value at the given precision and writes the + * rounded value into the given target + *

+ * This method internally uses double precision computation and rounding, so + * the result may not be accurate (see formatDouble method for conditions). + * + * @param source + * the source value to round + * @param decimals + * the decimals to round at (use if abs(source) ≥ 1.0) + * @param precision + * the precision to round at (use if abs(source) < 1.0) + * @param target + * the buffer to write to + */ + public static void formatDoubleFast(double source, int decimals, int precision, + StringBuffer target) + { + if (isRoundedToZero(source, decimals, precision)) + { + // Will always be rounded to 0 + target.append('0'); + return; + } else if (Double.isNaN(source) || Double.isInfinite(source)) + { + // Cannot be formated + target.append(Double.toString(source)); + return; + } + + boolean isPositive = source >= 0.0; + source = Math.abs(source); + int scale = (source >= 1.0) ? decimals : precision; + + long intPart = (long) Math.floor(source); + double tenScale = tenPowDouble(scale); + double fracUnroundedPart = (source - intPart) * tenScale; + long fracPart = Math.round(fracUnroundedPart); + if (fracPart >= tenScale) + { + intPart++; + fracPart = Math.round(fracPart - tenScale); + } + if (fracPart != 0L) + { + // Remove trailing zeroes + while (fracPart % 10L == 0L) + { + fracPart = fracPart / 10L; + scale--; + } + } + + if (intPart != 0L || fracPart != 0L) + { + // non-zero value + if (!isPositive) + { + // negative value, insert sign + target.append('-'); + } + // append integer part + target.append(intPart); + if (fracPart != 0L) + { + // append fractional part + target.append('.'); + // insert leading zeroes + while (scale > 0 && fracPart < tenPowDouble(--scale)) + { + target.append('0'); + } + target.append(fracPart); + } + } else + { + target.append('0'); + } + } + + /** + * Returns the exponent of the given value + * + * @param value + * the value to get the exponent from + * @return the value's exponent + */ + public static int getExponant(double value) + { + // See Double.doubleToRawLongBits javadoc or IEEE-754 spec + // to have this algorithm + long exp = Double.doubleToRawLongBits(value) & 0x7ff0000000000000L; + exp = exp >> 52; + return (int) (exp - 1023L); + } + + /** + * Returns true if the rounding is considered to use too many digits of the + * double for a fast rounding + * + * @param source + * the source to round + * @param scale + * the scale to round at + * @return true if the rounding will potentially use too many digits + */ + private static boolean tooManyDigitsUsed(double source, int scale) + { + // if scale >= 308, 10^308 ~= Infinity + double decExp = Math.log10(source); + return scale >= 308 || decExp + scale >= 14.5; + } + + /** + * Returns true if the given source is considered to be too close of a + * rounding value for the given scale. + * + * @param source + * the source to round + * @param scale + * the scale to round at + * @return true if the source will be potentially rounded at the scale + */ + private static boolean tooCloseToRound(double source, int scale) + { + source = Math.abs(source); + long intPart = (long) Math.floor(source); + double fracPart = (source - intPart) * tenPowDouble(scale); + double decExp = Math.log10(source); + double range = decExp + scale >= 12 ? .1 : .001; + double distanceToRound1 = Math.abs(fracPart - Math.floor(fracPart)); + double distanceToRound2 = Math.abs(fracPart - Math.floor(fracPart) - 0.5); + return distanceToRound1 <= range || distanceToRound2 <= range; + // .001 range: Totally arbitrary range, + // I never had a failure in 10e7 random tests with this value + // May be JVM dependent or architecture dependent + } +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java new file mode 100644 index 0000000..ef96f80 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java @@ -0,0 +1,11 @@ +package org.sandag.abm.reporting; + +import java.io.IOException; + +public interface IExporter +{ + static final String[] TOD_TOKENS = {"EA", "AM", "MD", "PM", "EV"}; + static final String TOD_TOKEN = "${TOD_TOKEN}"; + + void export() throws IOException; +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java new file mode 100644 index 0000000..12567c9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java @@ -0,0 +1,8 @@ +package org.sandag.abm.reporting; + +import com.pb.common.matrix.Matrix; + +public interface IMatrixDao +{ + Matrix getMatrix(String matrixName, String coreName); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java new file mode 100644 index 0000000..2ba5261 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java @@ -0,0 +1,36 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.util.HashMap; +import java.util.Properties; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.MatrixType; + +public class OMXMatrixDao + implements IMatrixDao +{ + private final String outputFolderToken = "skims.path"; + private final String matrixLocation; + + public OMXMatrixDao(Properties properties) + { + matrixLocation = properties.getProperty(outputFolderToken); + } + + public OMXMatrixDao(HashMap properties) + { + matrixLocation = (String)properties.get(outputFolderToken); + } + + + public Matrix getMatrix(String matrixName, String coreName) + { + String matrixPath = matrixLocation + File.separator + matrixName; + + MatrixReader mr = MatrixReader.createReader(MatrixType.OMX, new File(matrixPath)); + + return mr.readMatrix(coreName); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java new file mode 100644 index 0000000..ff3e8df --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java @@ -0,0 +1,721 @@ +package org.sandag.abm.reporting; + +import java.util.HashMap; +import java.util.Map; +import java.util.Properties; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +/** + * The {@code SkimBuilder} ... + * + * @author crf Started 10/17/12 9:12 AM + */ +public class SkimBuilder +{ + private static final Logger logger = Logger.getLogger(SkimBuilder.class); + + private static final int WALK_TIME_INDEX = 0; + private static final int BIKE_TIME_INDEX = 0; + + private static final int DA_NT_TIME_INDEX = 0; + private static final int DA_NT_FF_TIME_INDEX = 1; + private static final int DA_NT_DIST_INDEX = 2; + private static final int DA_NT_TOLL_INDEX = 3; + private static final int DA_NT_TOLLDIST_INDEX = 4; + private static final int DA_NT_STD_INDEX = 5; + private static final int DA_TR_TIME_INDEX = 6; + private static final int DA_TR_FF_TIME_INDEX = 7; + private static final int DA_TR_DIST_INDEX = 8; + private static final int DA_TR_TOLL_INDEX = 9; + private static final int DA_TR_TOLLDIST_INDEX = 10; + private static final int DA_TR_STD_INDEX = 11; + private static final int SR2_TIME_INDEX = 12; + private static final int SR2_FF_TIME_INDEX = 13; + private static final int SR2_DIST_INDEX = 14; + private static final int SR2_HOVDIST_INDEX = 15; + private static final int SR2_TOLL_INDEX = 16; + private static final int SR2_TOLLDIST_INDEX = 17; + private static final int SR2_STD_INDEX = 18; + private static final int SR3_TIME_INDEX = 19; + private static final int SR3_FF_TIME_INDEX = 20; + private static final int SR3_DIST_INDEX = 21; + private static final int SR3_HOVDIST_INDEX = 22; + private static final int SR3_TOLL_INDEX = 23; + private static final int SR3_TOLLDIST_INDEX = 24; + private static final int SR3_STD_INDEX = 25; + + + + private static final int TRANSIT_SET_ACCESS_TIME_INDEX = 0; + private static final int TRANSIT_SET_EGRESS_TIME_INDEX = 1; + private static final int TRANSIT_SET_AUX_WALK_TIME_INDEX = 2; + private static final int TRANSIT_SET_LOCAL_BUS_TIME_INDEX = 3; + private static final int TRANSIT_SET_EXPRESS_BUS_TIME_INDEX = 4; + private static final int TRANSIT_SET_BRT_TIME_INDEX = 5; + private static final int TRANSIT_SET_LRT_TIME_INDEX = 6; + private static final int TRANSIT_SET_CR_TIME_INDEX = 7; + private static final int TRANSIT_SET_FIRST_WAIT_TIME_INDEX = 8; + private static final int TRANSIT_SET_TRANSFER_WAIT_TIME_INDEX = 9; + private static final int TRANSIT_SET_FARE_INDEX = 10; + private static final int TRANSIT_SET_MAIN_MODE_INDEX = 11; + private static final int TRANSIT_SET_XFERS_INDEX = 12; + private static final int TRANSIT_SET_DIST_INDEX = 13; + + private static final double FEET_IN_MILE = 5280.0; + + private final TapDataManager tapManager; + private final TazDataManager tazManager; + private final MgraDataManager mgraManager; + private final AutoTazSkimsCalculator tazDistanceCalculator; + private final AutoAndNonMotorizedSkimsCalculator autoNonMotSkims; + private final WalkTransitWalkSkimsCalculator wtw; + private final WalkTransitDriveSkimsCalculator wtd; + private final DriveTransitWalkSkimsCalculator dtw; + + private final String FUEL_COST_PROPERTY = "aoc.fuel"; + private final String MAINTENANCE_COST_PROPERTY = "aoc.maintenance"; + private float autoOperatingCost; + + + public SkimBuilder(Properties properties) + { + + HashMap rbMap = new HashMap( + (Map) (Map) properties); + tapManager = TapDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + autoNonMotSkims = new AutoAndNonMotorizedSkimsCalculator(rbMap); + autoNonMotSkims.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + BestTransitPathCalculator bestPathUEC = new BestTransitPathCalculator(rbMap); + wtw = new WalkTransitWalkSkimsCalculator(rbMap); + wtw.setup(rbMap, logger, bestPathUEC); + wtd = new WalkTransitDriveSkimsCalculator(rbMap); + wtd.setup(rbMap, logger, bestPathUEC); + dtw = new DriveTransitWalkSkimsCalculator(rbMap); + dtw.setup(rbMap, logger, bestPathUEC); + + float fuelCost = new Float(properties.getProperty(FUEL_COST_PROPERTY)); + float mainCost = new Float(properties.getProperty(MAINTENANCE_COST_PROPERTY)); + autoOperatingCost = (fuelCost + mainCost) * 0.01f; + + } + + // todo: hard coding these next two lookups because it is convenient, but + // probably should move to a lookup file + private final String[] modeNameLookup = { + "UNKNOWN", // ids start at one + "DRIVEALONE", "SHARED2", "SHARED3", "WALK", "BIKE", "WALK_SET", "PNR_SET", + "KNR_TRN", "TNC_TRN", "TAXI", "TNC_SINGLE", "TNC_SHARED","SCHBUS"}; + + private final TripModeChoice[] modeChoiceLookup = {TripModeChoice.UNKNOWN, + TripModeChoice.DRIVE_ALONE, TripModeChoice.SR2,TripModeChoice.SR3, + TripModeChoice.WALK, TripModeChoice.BIKE, TripModeChoice.WALK_SET, + TripModeChoice.PNR_SET, TripModeChoice.KNR_SET, TripModeChoice.KNR_SET, + TripModeChoice.SR2, + TripModeChoice.SR2, TripModeChoice.SR2, TripModeChoice.SR2 }; + + private int getTod(int tripTimePeriod) + { + return ModelStructure.getSkimPeriodIndex(tripTimePeriod); + } + + private int getStartTime(int tripTimePeriod) + { + return (tripTimePeriod - 1) * 30 + 270; // starts at 4:30 and goes half + // hour intervals after that + } + + public TripAttributes getTripAttributes(int origin, int destination, int tripModeIndex, + int boardTap, int alightTap, int tripTimePeriod, boolean inbound, float valueOfTime, int set, boolean isCB, int transponderOwnership) + { + int tod = getTod(tripTimePeriod); + TripModeChoice tripMode = modeChoiceLookup[tripModeIndex < 0 ? 0 : tripModeIndex]; + + TripAttributes attributes = getTripAttributes(tripMode, origin, destination, boardTap, + alightTap, tod, inbound, valueOfTime, set,isCB,transponderOwnership); + attributes.setTripModeName(modeNameLookup[tripModeIndex < 0 ? 0 : tripModeIndex]); + attributes.setTripStartTime(getStartTime(tripTimePeriod)); + return attributes; + } + + private TripAttributes getTripAttributesUnknown() + { + return new TripAttributes(0,0,0,0,0,0,0,0,0,0,0,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); + } + + private final float DEFAULT_BIKE_SPEED = 12; + private final float DEFAULT_WALK_SPEED = 3; + private int omeMgra=7123; + + private TripAttributes getTripAttributes(TripModeChoice modeChoice, int origin, + int destination, int boardTap, int alightTap, int tod, boolean inbound, float vot, int set,boolean isCB, int transponderOwnership) + { + int timeIndex = -1; + int distIndex = -1; + int costIndex = -1; + int stdIndex = -1; + int otaz=-1; + int dtaz=-1; + + double [] autoSkims; + + switch (modeChoice) + { + case UNKNOWN: + return getTripAttributesUnknown(); + case DRIVE_ALONE: + { + if(transponderOwnership>0) { + timeIndex = DA_TR_TIME_INDEX; + distIndex = DA_TR_DIST_INDEX; + costIndex = DA_TR_TOLL_INDEX; + stdIndex = DA_TR_STD_INDEX; + }else { + timeIndex = DA_NT_TIME_INDEX; + distIndex = DA_NT_DIST_INDEX; + costIndex = DA_NT_TOLL_INDEX; + stdIndex = DA_NT_STD_INDEX; + + } + if(isCB && (origin==omeMgra||destination==omeMgra)) { + int[] tazPair=setTazOD(origin, destination); + otaz=tazPair[0]; + dtaz=tazPair[1]; + autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, + logger); + }else { + autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, + logger); + } + return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); + } + + case SR2: // wu added + { + timeIndex = SR2_TIME_INDEX; + distIndex = SR2_DIST_INDEX; + costIndex = SR2_TOLL_INDEX; + stdIndex = SR2_STD_INDEX; + if(isCB && (origin==omeMgra||destination==omeMgra)) { + int[] tazPair=setTazOD(origin, destination); + otaz=tazPair[0]; + dtaz=tazPair[1]; + autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, + logger); + }else { + autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, + logger); + } + + return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); + } + case SR3: + { + timeIndex = SR3_TIME_INDEX; + distIndex = SR3_DIST_INDEX; + costIndex = SR3_TOLL_INDEX; + stdIndex = SR3_STD_INDEX; + if(isCB && (origin==omeMgra||destination==omeMgra)) { + int[] tazPair=setTazOD(origin, destination); + otaz=tazPair[0]; + dtaz=tazPair[1]; + autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, + logger); + }else { + autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, + logger); + } + return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); + } + case WALK: + { + // first, look in mgra manager, otherwise default to auto skims + double distance = mgraManager.getMgraToMgraWalkDistFrom(origin, destination) / FEET_IN_MILE; + double time =0; + if (distance > 0) + { + time = mgraManager.getMgraToMgraWalkTime(origin, destination); + }else{ + distance = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, logger)[DA_NT_DIST_INDEX]; + time = distance * 60 / DEFAULT_WALK_SPEED; + } + return new TripAttributes(0, 0, 0, 0, 0, 0, 0, 0, time, 0, distance, -1, -1, 0,0,0,0,0,0,0,0,0,0,0,0,0,0); + } + case BIKE: + { + double time = mgraManager.getMgraToMgraBikeTime(origin, destination); + double distance = 0; + if (time > 0) + { + distance = time * DEFAULT_BIKE_SPEED / 60; + + }else{ + distance = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, + logger)[DA_NT_DIST_INDEX]; + time = distance * 60 / DEFAULT_BIKE_SPEED; + } + return new TripAttributes(0, 0, 0, 0, 0, 0, 0, 0, 0, time, distance, -1, -1, 0,0,0,0,0,0,0,0,0,0,0,0,0,0); + } + case WALK_SET : + case PNR_SET : + case KNR_SET : { + boolean isDrive = modeChoice.isDrive; + double walkTime = 0.0; + double driveTime = 0.0; + + double[] skims; + int boardTaz = -1; + int alightTaz = -1; + double boardAccessTime = 0.0; + double alightEgressTime = 0.0; + double accessDistance = 0.0; + double egressDistance = 0.0; + int originTaz = mgraManager.getTaz(origin); + int destTaz = mgraManager.getTaz(destination); + if (isDrive) { + if (!inbound) { //outbound: drive to transit stop at origin, then transit to destination + boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + accessDistance = tazManager.getDistanceToTapFromTaz(originTaz,boardTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + alightEgressTime = mgraManager.getWalkTimeFromMgraToTap(destination,alightTap); + egressDistance = mgraManager.getWalkDistanceFromMgraToTap(destination,alightTap); + + if (boardAccessTime ==-1) { + logger.info("Error: TAP not accessible from origin TAZ by "+ (modeChoice==TripModeChoice.PNR_SET ? "PNR" : "KNR" )+" access"); + logger.info("mc: " + modeChoice); + logger.info("origin MAZ: " + origin); + logger.info("origin TAZ" + originTaz); + logger.info("dest MAZ: " + destination); + logger.info("board tap: " + boardTap); + logger.info("alight tap: " + alightTap); + logger.info("tod: " + tod); + logger.info("inbound: " + inbound); + logger.info("set: " + set); + } + + if (alightEgressTime == -1){ + logger.info("Error: TAP not accessible from destination MAZ by walk access"); + logger.info("mc: " + modeChoice); + logger.info("origin MAZ: " + origin); + logger.info("origin TAZ" + originTaz); + logger.info("dest MAZ: " + destination); + logger.info("board tap: " + boardTap); + logger.info("alight tap: " + alightTap); + logger.info("tod: " + tod); + logger.info("inbound: " + inbound); + logger.info("set: " + set); + + } + skims = dtw.getDriveTransitWalkSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); + walkTime = alightEgressTime; + driveTime= boardAccessTime; + + } else { //inbound: transit from origin to destination, then drive + boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(origin,boardTap); + accessDistance = mgraManager.getWalkDistanceFromMgraToTap(origin,boardTap); + alightEgressTime = tazManager.getTimeToTapFromTaz(destTaz,alightTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + egressDistance = tazManager.getDistanceToTapFromTaz(destTaz,alightTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + if (boardAccessTime ==-1) { + logger.info("Error: TAP not accessible from origin MAZ by walk access"); + logger.info("mc: " + modeChoice); + logger.info("origin MAZ: " + origin); + logger.info("origin TAZ" + originTaz); + logger.info("dest MAZ: " + destination); + logger.info("board tap: " + boardTap); + logger.info("alight tap: " + alightTap); + logger.info("tod: " + tod); + logger.info("inbound: " + inbound); + logger.info("set: " + set); + } + + if (alightEgressTime == -1){ + logger.info("Error: TAP not accessible from destination TAZ by "+ (modeChoice==TripModeChoice.PNR_SET ? "PNR" : "KNR" )+" access"); + logger.info("mc: " + modeChoice); + logger.info("origin MAZ: " + origin); + logger.info("origin TAZ" + originTaz); + logger.info("dest MAZ: " + destination); + logger.info("board tap: " + boardTap); + logger.info("alight tap: " + alightTap); + logger.info("tod: " + tod); + logger.info("inbound: " + inbound); + logger.info("set: " + set); + + } + skims = wtd.getWalkTransitDriveSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); + walkTime = boardAccessTime ; + driveTime= alightEgressTime; + } + } else { + int bt = mgraManager.getTapPosition(origin,boardTap); + int at = mgraManager.getTapPosition(destination,alightTap); + if (bt < 0 || at < 0) { + logger.info("bad tap position: " + bt + " " + at); + logger.info("mc: " + modeChoice); + logger.info("origin: " + origin); + logger.info("dest: " + destination); + logger.info("board tap: " + boardTap); + logger.info("alight tap: " + alightTap); + logger.info("tod: " + tod); + logger.info("inbound: " + inbound); + logger.info("set: " + set); + logger.info("board tap position: " + bt); + logger.info("alight tap position: " + at); + } else { + boardAccessTime = mgraManager.getMgraToTapWalkTime(origin,bt); + accessDistance = mgraManager.getWalkDistanceFromMgraToTap(origin,boardTap); + alightEgressTime = mgraManager.getMgraToTapWalkTime(destination,at); + egressDistance = mgraManager.getWalkDistanceFromMgraToTap(destination,alightTap); + } + walkTime = boardAccessTime + alightEgressTime; + skims = wtw.getWalkTransitWalkSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); + } + + double transitInVehicleTime = 0.0; + + transitInVehicleTime += skims[TRANSIT_SET_CR_TIME_INDEX]; + transitInVehicleTime += skims[TRANSIT_SET_LRT_TIME_INDEX]; + transitInVehicleTime += skims[TRANSIT_SET_BRT_TIME_INDEX]; + transitInVehicleTime += skims[TRANSIT_SET_EXPRESS_BUS_TIME_INDEX]; + transitInVehicleTime += skims[TRANSIT_SET_LOCAL_BUS_TIME_INDEX]; + + double crTime = skims[TRANSIT_SET_CR_TIME_INDEX]; + double lrtTime = skims[TRANSIT_SET_LRT_TIME_INDEX]; + double brtTime = skims[TRANSIT_SET_BRT_TIME_INDEX]; + double expTime = skims[TRANSIT_SET_EXPRESS_BUS_TIME_INDEX]; + double locTime = skims[TRANSIT_SET_LOCAL_BUS_TIME_INDEX]; + + //wsu 9/17/18, walkTime already set + //walkTime += skims[TRANSIT_SET_ACCESS_TIME_INDEX]; + //walkTime += skims[TRANSIT_SET_EGRESS_TIME_INDEX ]; + walkTime += skims[TRANSIT_SET_AUX_WALK_TIME_INDEX]; + + double auxiliaryTime = skims[TRANSIT_SET_AUX_WALK_TIME_INDEX]; + + double waitTime = 0.0; + waitTime += skims[TRANSIT_SET_FIRST_WAIT_TIME_INDEX]; + waitTime += skims[TRANSIT_SET_TRANSFER_WAIT_TIME_INDEX]; + + double transfers = skims[TRANSIT_SET_XFERS_INDEX]; + + double transitFare = 0.0; + transitFare += skims[TRANSIT_SET_FARE_INDEX]; + + double transitDist = skims[TRANSIT_SET_DIST_INDEX]; + /* + int modeIndex = 0; + for(modeIndex = TRANSIT_SET_LOCAL_BUS_TIME_INDEX; modeIndex <= TRANSIT_SET_CR_TIME_INDEX; modeIndex++){ + if(skims[modeIndex] > 0) + break; + } + */ + double dist = autoNonMotSkims.getAutoSkims(origin,destination,tod,vot,false,logger)[DA_NT_DIST_INDEX]; //todo: is this correct enough? + return new TripAttributes(driveTime, driveTime/60*35*autoOperatingCost, 0, 0, transitInVehicleTime, + waitTime, walkTime, transitFare, 0, 0, dist, boardTaz, alightTaz, vot, set, + accessDistance,egressDistance,auxiliaryTime,boardAccessTime,alightEgressTime,transfers,locTime,expTime,brtTime,lrtTime,crTime,transitDist); + } + default: + throw new IllegalStateException("Should not be here: " + modeChoice); + } + } + + private int[] setTazOD(int omgra, int dmgra) { + int [] result=new int[2]; + //int omeMgra=7123; + result [0]=mgraManager.getTaz(omgra); + result [1]=mgraManager.getTaz(dmgra); + if (omgra==omeMgra) result[0]=3; + if (dmgra==omeMgra) result[1]=3; + return result; + } + + public static enum TripModeChoice + { + UNKNOWN(false), + DRIVE_ALONE(true), + SR2(true), + SR3(true), + WALK(false), + BIKE(false), + WALK_SET(false), + PNR_SET(true), + KNR_SET(true); + + private final boolean isDrive; + + private TripModeChoice(boolean drive) + { + isDrive = drive; + } + + } + + public static class TripAttributes + { + private final float autoInVehicleTime; + private final float autoOperatingCost; + private final float autoStandardDeviationTime; + private final float autoTollCost; + private final float transitInVehicleTime; + private final float transitWaitTime; + private final float transitWalkTime; + private final float transitFare; + private final float walkModeTime; + private final float bikeModeTime; + private final float tripDistance; + private final int tripBoardTaz; + private final int tripAlightTaz; + private final int set; + private final float valueOfTime; + private final float transitAccessDistance; + private final float transitEgressDistance; + private final float transitAuxiliaryTime; + private final float transitAccessTime; + private final float transitEgressTime; + private final float transitTransfers; + private final float locTime; + private final float expTime; + private final float brtTime; + private final float lrtTime; + private final float crTime; + private final float transitDistance; + + private String tripModeName; + + public int getTripStartTime() + { + return tripStartTime; + } + + public void setTripStartTime(int tripStartTime) + { + this.tripStartTime = tripStartTime; + } + + private int tripStartTime; + + public TripAttributes(double autoInVehicleTime, double autoOperatingCost, double autoStandardDeviationTime, double autoTollCost, double transitInVehicleTime, + double transitWaitTime, double transitWalkTime, double transitFare, double walkModeTime, double bikeModeTime, double tripDistance, + int tripBoardTaz, int tripAlightTaz, float valueOfTime, int set, double accessDistance, + double egressDistance, double auxiliaryTime, double accessTime,double egressTime, double transfers, double locTime, double expTime, double brtTime, double lrtTime, double crTime, double trnDist) + { + this.autoInVehicleTime = (float) autoInVehicleTime; + this.autoOperatingCost = (float) autoOperatingCost; + this.autoStandardDeviationTime = (float) autoStandardDeviationTime; + this.autoTollCost = (float) autoTollCost; + this.transitInVehicleTime = (float) transitInVehicleTime; + this.transitWaitTime = (float) transitWaitTime; + this.transitWalkTime = (float) transitWalkTime; + this.transitFare = (float) transitFare; + this.walkModeTime = (float) walkModeTime; + this.bikeModeTime = (float) bikeModeTime; + this.tripDistance = (float) tripDistance; + this.tripBoardTaz = tripBoardTaz; + this.tripAlightTaz = tripAlightTaz; + this.set = set; + this.valueOfTime = valueOfTime; + this.transitAccessDistance = (float) accessDistance; + this.transitEgressDistance = (float) egressDistance; + this.transitAuxiliaryTime = (float) auxiliaryTime; + this.transitAccessTime = (float) accessTime; + this.transitEgressTime = (float) egressTime; + this.transitTransfers = (float) transfers; + this.locTime = (float) locTime; + this.expTime = (float) expTime; + this.brtTime = (float) brtTime; + this.lrtTime = (float) lrtTime; + this.crTime = (float) crTime; + this.transitDistance = (float) trnDist; + + } + + + /** + * A method to set create trip attributes for a non-toll auto choice. + * + * @param autoInVehicleTime + * @param tripDistance + * @param autoOperatingCost + */ + public TripAttributes(double autoInVehicleTime, double tripDistance, double autoOperatingCost, double stdDevTime) + { + this(autoInVehicleTime, autoOperatingCost, stdDevTime, 0,0,0,0,0,0,0,tripDistance,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); + } + + /** + * A method to create trip attributes for a toll auto choice. + * + * @param autoInVehicleTime + * @param tripDistance + * @param autoOperatingCost + * @param tollCost + */ + public TripAttributes(double autoInVehicleTime, double tripDistance, double autoOperatingCost, double stdDevTime, double tollCost) + { + this(autoInVehicleTime, autoOperatingCost, stdDevTime, tollCost,0,0,0,0,0,0,tripDistance,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); + } + + + + public void setTripModeName(String tripModeName) + { + this.tripModeName = tripModeName; + } + + public float getAutoInVehicleTime() { + return autoInVehicleTime; + } + + public float getAutoOperatingCost() { + return autoOperatingCost; + } + + public float getAutoStandardDeviationTime() { + return autoStandardDeviationTime; + } + + public float getAutoTollCost() { + return autoTollCost; + } + + public float getTransitInVehicleTime() { + return transitInVehicleTime; + } + + public float getTransitWaitTime() { + return transitWaitTime; + } + + public float getTransitFare() { + return transitFare; + } + + public float getTransitWalkTime() { + return transitWalkTime; + } + + public float getWalkModeTime() { + return walkModeTime; + } + + public float getBikeModeTime() { + return bikeModeTime; + } + + public float getTripDistance() { + return tripDistance; + } + + public String getTripModeName() + { + return tripModeName; + } + + public int getTripBoardTaz() + { + return tripBoardTaz; + } + + public int getTripAlightTaz() + { + return tripAlightTaz; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public int getSet() { + return set; + } + + public float getTransitAccessDistance() { + return transitAccessDistance; + } + + public float getTransitEgressDistance() { + return transitEgressDistance; + } + + public float getTransitAuxiliaryTime() { + return transitAuxiliaryTime; + } + + public float getTransitAccessTime() { + return transitAccessTime; + } + + public float getTransitEgressTime() { + return transitEgressTime; + } + + public float getTransitTransfers() { + return transitTransfers; + } + + public float getLocTime() { + return locTime; + } + + public float getExpTime() { + return expTime; + } + + public float getBrtTime() { + return brtTime; + } + + public float getLrtTime() { + return lrtTime; + } + + public float getCrTime() { + return crTime; + } + + public float getTransitDistance(){ + return transitDistance; + } + } + + public float getLotWalkTime(int parkingLotMaz, int destinationMaz) { + + // first, look in mgra manager, otherwise default to auto skims + double distance = mgraManager.getMgraToMgraWalkDistFrom(parkingLotMaz, destinationMaz) / FEET_IN_MILE; + if (distance <= 0) { + distance = autoNonMotSkims.getAutoSkims(parkingLotMaz, destinationMaz, SandagModelStructure.EA_SKIM_PERIOD_INDEX +1, (float)15.0,false, logger)[DA_NT_DIST_INDEX]; + } + + return (float) (distance * 60 / DEFAULT_WALK_SPEED); + } + + + public float getLotWalkDistance(int parkingLotMaz, int destinationMaz) { + + // first, look in mgra manager, otherwise default to auto skims + double distance = mgraManager.getMgraToMgraWalkDistFrom(parkingLotMaz, destinationMaz) / FEET_IN_MILE; + if (distance <= 0) { + distance = autoNonMotSkims.getAutoSkims(parkingLotMaz, destinationMaz, SandagModelStructure.EA_SKIM_PERIOD_INDEX +1, (float)15.0,false, logger)[DA_NT_DIST_INDEX]; + } + + return (float) distance; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java new file mode 100644 index 0000000..f5aafc2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java @@ -0,0 +1,28 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.util.Properties; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.MatrixType; + +public class TranscadMatrixDao + implements IMatrixDao +{ + private final String outputFolderToken = "skims.path"; + private final String matrixLocation; + + public TranscadMatrixDao(Properties properties) + { + matrixLocation = properties.getProperty(outputFolderToken); + } + + public Matrix getMatrix(String matrixName, String coreName) + { + String matrixPath = matrixLocation + File.separator + matrixName + ".mtx"; + + MatrixReader mr = MatrixReader.createReader(MatrixType.TRANSCAD, new File(matrixPath)); + + return mr.readMatrix(coreName); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java new file mode 100644 index 0000000..bb5628f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java @@ -0,0 +1,449 @@ +package org.sandag.abm.reporting; + +import java.io.BufferedWriter; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.Modes; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; + +public class TransitTimeReporter { + private static final Logger logger = Logger.getLogger(TransitTimeReporter.class); + private BestTransitPathCalculator bestPathCalculator; + protected WalkTransitWalkSkimsCalculator wtw; + protected WalkTransitDriveSkimsCalculator wtd; + protected DriveTransitWalkSkimsCalculator dtw; + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + private MatrixDataServerRmi ms; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + AutoTazSkimsCalculator tazDistanceCalculator; + + //skim locations in WalkTransitWalkSkims UEC + private static final int WLK_WALKACCESSTIME = 0; + private static final int WLK_WALKEGRESSTIME = 1; + private static final int WLK_AUXWALKTIME = 2; + private static final int WLK_LOCALBUSIVT = 3; + private static final int WLK_EXPRESSBUSIVT = 4; + private static final int WLK_BRTIVT = 5; + private static final int WLK_LRTIVT = 6; + private static final int WLK_CRIVT = 7; + private static final int WLK_T1IVT = 8; + private static final int WLK_FIRSTWAITTIME = 9; + private static final int WLK_TRWAITTIME = 10; + private static final int WLK_FARE = 11; + private static final int WLK_TOTALIVT = 12; + private static final int WLK_XFERS = 13; + private static final int WLK_DIST = 14; + + //skim locations in DriveTransitWalkSkims UEC + private static final int DRV_DRIVEACCESSTIME = 0; + private static final int DRV_WALKEGRESSTIME = 1; + private static final int DRV_AUXWALKTIME = 2; + private static final int DRV_LOCALBUSIVT = 3; + private static final int DRV_EXPRESSBUSIVT = 4; + private static final int DRV_BRTIVT = 5; + private static final int DRV_LRTIVT = 6; + private static final int DRV_CRIVT = 7; + private static final int DRV_T1IVT = 8; + private static final int DRV_FIRSTWAITTIME = 9; + private static final int DRV_TRWAITTIME = 10; + private static final int DRV_FARE = 11; + private static final int DRV_TOTALIVT = 12; + private static final int DRV_XFERS = 13; + private static final int DRV_DIST = 14; + + String period; //should be "AM" or "MD" + float threshold; //tested at 30 minutes + boolean inbound = false; + + private PrintWriter walkAccessWriter; + private PrintWriter driveAccessWriter; + private String outWalkFile; + private String outDriveFile; + private boolean createDriveFile=false; + + public TransitTimeReporter(HashMap propertyMap, float threshold, String period,String outWalkFileName,String outDriveFileName){ + + startMatrixServer(propertyMap); + + this.threshold = threshold; + this.period = period; + this.outWalkFile = outWalkFileName; + this.outDriveFile= outDriveFileName; + if(outDriveFile!=null) { + this.outDriveFile = outDriveFileName; + createDriveFile=true; + } + + initialize(propertyMap); + } + + /** + * Initialize best path builders. + * + * @param propertyMap A property map with relevant properties. + */ + public void initialize(HashMap propertyMap){ + + String path=System.getProperty("user.dir"); + outWalkFile=path+"\\output\\"+outWalkFile; + if(createDriveFile) { + outDriveFile=path+"\\output\\"+outDriveFile; + } + + logger.info("Initializing Transit Time Reporter"); + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + bestPathCalculator = new BestTransitPathCalculator(propertyMap); + + tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + wtw = new WalkTransitWalkSkimsCalculator(propertyMap); + wtw.setup(propertyMap, logger, bestPathCalculator); + wtd = new WalkTransitDriveSkimsCalculator(propertyMap); + wtd.setup(propertyMap, logger, bestPathCalculator); + dtw = new DriveTransitWalkSkimsCalculator(propertyMap); + dtw.setup(propertyMap, logger, bestPathCalculator); + + walkAccessWriter = createOutputFile(outWalkFile); + if(createDriveFile) { + driveAccessWriter = createOutputFile(outDriveFile); + } + } + + /** + * Create the output file. + */ + private PrintWriter createOutputFile(String fileName){ + + logger.info("Creating file " + fileName); + PrintWriter writer; + try + { + writer = new PrintWriter(new BufferedWriter(new FileWriter(fileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + fileName + " for writing\n"); + throw new RuntimeException(); + } + + return writer; + + } + + private ArrayList getWalkTransitTimeComponents(HashMap propertyMap){ + String timeElements = (String) propertyMap.get("transitShed.walkTransitTimeComponents"); + String delims = "[,]"; + String[] elements = timeElements.split(delims); + ArrayList components=new ArrayList(); + for (int i=0; i getDriveTransitTimeComponents(HashMap propertyMap){ + String timeElements = (String) propertyMap.get("transitShed.driveTransitTimeComponents"); + String delims = "[,]"; + String[] elements = timeElements.split(delims); + ArrayList components=new ArrayList(); + for (int i=0; i pMap){ + + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + double boardAccessTime; + double alightAccessTime; + + int skimPeriod = -1; + + if(period.compareTo("EA")==0){ + skimPeriod=ModelStructure.EA_SKIM_PERIOD_INDEX; + inbound = false; + }else if(period.compareTo("AM")==0){ + skimPeriod=ModelStructure.AM_SKIM_PERIOD_INDEX; + inbound = false; + }else if(period.compareTo("MD")==0){ + skimPeriod=ModelStructure.MD_SKIM_PERIOD_INDEX; + inbound = false; + }else if(period.compareTo("PM")==0){ + skimPeriod=ModelStructure.PM_SKIM_PERIOD_INDEX; + inbound = true; + }else if(period.compareTo("EV")==0){ + skimPeriod=ModelStructure.EV_SKIM_PERIOD_INDEX; + inbound = true; + }else{ + logger.fatal("Skim period "+period+" not recognized"); + throw new RuntimeException(); + } + + //iterate through mazs and calculate time + ArrayList mazs = mgraManager.getMgras(); + + //origins + for(int originMaz: mazs ){ + + if((originMaz<=100) || ((originMaz % 100) == 0)) + logger.info("Processing origin mgra "+originMaz); + + int originTaz = mgraManager.getTaz(originMaz); + + //for saving results + String outWalkString = null; + String outDriveString = null; + + //destinations + for(int destinationMaz:mazs){ + + int destinationTaz = mgraManager.getTaz(destinationMaz); + + float odDistance = (float) tazDistanceCalculator.getTazToTazDistance(skimPeriod, originTaz, destinationTaz); + + //walk calculations + double[][] bestWalkTaps = bestPathCalculator.getBestTapPairs(walkDmu, driveDmu, bestPathCalculator.WTW, originMaz, destinationMaz, skimPeriod, false, logger, odDistance); + double[] bestWalkUtilities = bestPathCalculator.getBestUtilities(); + + //only look at best utility path; continue if MGRA isn't available by walk. + if(bestWalkUtilities[0]>-500){ + + //Best walk TAP pair + int boardTap = (int) bestWalkTaps[0][0]; + int alightTap = (int) bestWalkTaps[0][1]; + int set = (int) bestWalkTaps[0][2]; + + // get walk skims + boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); + alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); + double[] walkSkims = wtw.getWalkTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); + + //calculate total time + double totalTime=0; + ArrayList wtelements=getWalkTransitTimeComponents(pMap); + for (int i=0; i-500){ + + //best drive TAP pair + int boardTap = (int) bestDriveTaps[0][0]; + int alightTap = (int) bestDriveTaps[0][1]; + int set = (int) bestDriveTaps[0][2]; + + //skims for best drive pair + double[] driveSkims = null; + if(inbound==false){ + boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( Modes.AccessMode.PARK_N_RIDE )); + alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); + driveSkims = dtw.getDriveTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); + }else{ + boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); + alightAccessTime = tazManager.getTimeToTapFromTaz(destinationTaz,alightTap,( Modes.AccessMode.PARK_N_RIDE )); + driveSkims = wtd.getWalkTransitDriveSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); + + } + //total drive-transit time + //calculate total time + double totalTime=0; + ArrayList dtelements=getDriveTransitTimeComponents(pMap); + for (int i=0; i properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + MatrixDataServerIf ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + logger.error("could not connect to matrix server", e); + throw new RuntimeException(e); + + } + + } + + /** + * Main run method + * @param args + */ + public static void main(String[] args) { + + String propertiesFile = null; + float threshold = 0; + String period = null; + String outWalkFileName = null; + String outDriveFileName = null; + String delims = "[.]"; + + HashMap pMap; + + logger.info(String.format("Report MAZs within transit time threshold. Using CT-RAMP version ", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else { + propertiesFile = args[0]; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-threshold")) + { + threshold = Float.valueOf(args[i + 1]); + } + + if (args[i].equalsIgnoreCase("-period")) + { + period = args[i + 1]; + } + + if (args[i].equalsIgnoreCase("-outWalkFileName")) + { + String[] elements = args[i + 1].split(delims); + outWalkFileName = elements[0]+"_"+period+".csv"; + } + if (args[i].equalsIgnoreCase("-outDriveFileName")) + { + String[] elements = args[i + 1].split(delims); + outDriveFileName = elements[0]+"_"+period+".csv"; + } + } + } + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + TransitTimeReporter transitTimeReporter = new TransitTimeReporter(pMap, threshold, period,outWalkFileName,outDriveFileName); + + + transitTimeReporter.run(pMap); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java new file mode 100644 index 0000000..ebd5cf2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java @@ -0,0 +1,58 @@ +package org.sandag.abm.reporting; + +import java.io.IOException; +import java.util.Properties; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.LinkedBlockingQueue; + +public class TruckCsvExporter + extends AbstractCsvExporter +{ + private static final String MATRIX_BASE_NAME = "dailyDistributionMatricesTruck" + TOD_TOKEN; + private static final String[] CORE_NAMES = {"lhdn", "lhdt", "mhdn", "mhdt", "hhdn", + "hhdt" }; + private static final String[] COLUMN_HEADERS = {"ORIG", "DEST", "TOD", "CLASS", "TRIPS"}; + + public TruckCsvExporter(Properties properties, IMatrixDao aMatrixServerWrapper, + String aBaseFileName) + { + super(properties, aMatrixServerWrapper, aBaseFileName); + } + + @Override + public void export() throws IOException + { + BlockingQueue queue = new LinkedBlockingQueue(); + + Thread[] threads = new Thread[TOD_TOKENS.length]; + + LOGGER.info("Initializing Truck Writer Thread. Output Location: " + + getFile().getAbsoluteFile()); + CsvWriterThread writerThread = new CsvWriterThread(queue, getFile(), COLUMN_HEADERS); + new Thread(writerThread).start(); + + for (int i = 0; i < TOD_TOKENS.length; i++) + { + String matrixName = MATRIX_BASE_NAME.replace(TOD_TOKEN, TOD_TOKENS[i]); + LOGGER.info("Initializing Truck Reader Thread. Matrix: " + matrixName); + TruckCsvPublisherThread publisherThread = new TruckCsvPublisherThread(queue, + getMatrixDao(), matrixName, TOD_TOKENS[i], CORE_NAMES); + threads[i] = new Thread(publisherThread); + threads[i].start(); + } + + for (Thread thread : threads) + { + try + { + thread.join(); + } catch (InterruptedException e) + { + e.printStackTrace(System.err); + } + } + + LOGGER.info("Initializing Truck Reader Threads Complete. Issuing Poison Pill to Writer."); + queue.add(CsvWriterThread.POISON_PILL); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java new file mode 100644 index 0000000..2120d6e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java @@ -0,0 +1,67 @@ +package org.sandag.abm.reporting; + +import java.text.DecimalFormat; +import java.util.concurrent.BlockingQueue; +import org.apache.log4j.Logger; +import com.pb.common.matrix.Matrix; + +public class TruckCsvPublisherThread + implements Runnable +{ + private static final Logger LOGGER = Logger.getLogger(TruckCsvPublisherThread.class); + + private BlockingQueue queue; + private IMatrixDao mtxDao; + private String matrixName; + private String tod; + private String[] cores; + + private final double sizeThreshold = 0.00001; + + private static final DecimalFormat FORMATTER = new DecimalFormat("#.######"); + + public TruckCsvPublisherThread(BlockingQueue aQueue, + IMatrixDao aMtxDao, String aMatrixName, String aTod, String[] theCores) + { + this.queue = aQueue; + this.mtxDao = aMtxDao; + this.matrixName = aMatrixName; + this.tod = aTod; + this.cores = theCores; + } + + @Override + public void run() + { + for (String core : cores) + { + Matrix matrix = mtxDao.getMatrix(matrixName, core); + try + { + addRowsToQueue(core, matrix); + } catch (InterruptedException e) + { + LOGGER.fatal(e); + throw new RuntimeException(e); + } + } + + } + + public void addRowsToQueue(String core, Matrix matrix) throws InterruptedException + { + for (int origin : matrix.getExternalNumbers()) + { + for (int dest : matrix.getExternalColumnNumbers()) + { + float trips = matrix.getValueAt(origin, dest); + if (trips > sizeThreshold) + { + CsvRow row = new CsvRow(new String[] {String.valueOf(origin), + String.valueOf(dest), tod, core, FORMATTER.format(trips)}); + queue.put(row); + } + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java new file mode 100644 index 0000000..a7272ef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java @@ -0,0 +1,56 @@ +package org.sandag.abm.reporting; + +import java.io.File; +import java.io.IOException; +import java.util.Properties; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.LinkedBlockingQueue; + +import org.apache.log4j.Logger; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; + +public class TruckOmxExporter implements IExporter +{ + private static final String MATRIX_BASE_NAME = "DailyDistributionMatricesTruck" + TOD_TOKEN; + private static final String[] CORE_NAMES = {"lhdn", "lhdt", "mhdn", "mhdt", "hhdn", + "hhdt" }; + + private IMatrixDao matrixDao; + private String reportFolder = "report.path"; + private Properties properties; + + protected static final Logger LOGGER = Logger.getLogger(AbstractCsvExporter.class); + + + public TruckOmxExporter(Properties properties, IMatrixDao aMatrixServerWrapper, + String aBaseFileName) + { + this.matrixDao = aMatrixServerWrapper; + this.properties = properties; + } + + @Override + public void export() throws IOException + { + + for (int i = 0; i < TOD_TOKENS.length; i++) + { + String matrixName = MATRIX_BASE_NAME.replace(TOD_TOKEN, TOD_TOKENS[i]); + + File outMatrixFile = new File(properties.getProperty(reportFolder), matrixName+".omx"); + + MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, outMatrixFile); + Matrix[] inMatrix = new Matrix[CORE_NAMES.length]; + + for(int j = 0; j < CORE_NAMES.length; ++j) + inMatrix[j] = matrixDao.getMatrix(matrixName, CORE_NAMES[j]); + + + matrixWriter.writeMatrices(CORE_NAMES, inMatrix); + + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java new file mode 100644 index 0000000..e8ec218 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java @@ -0,0 +1,47 @@ +package org.sandag.abm.reporting.emfac2011; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.pb.sawdust.util.property.PropertyDeluxe; + +/** + * The {@code SandagAquavisInputBuilder} ... + * + * @author Wu.Sun@sandag.org 1/21/2014 + */ +public class AquavisDataBuilder { + private static final Logger LOGGER = LoggerFactory + .getLogger(AquavisDataBuilder.class); + private final PropertyDeluxe properties; + private Emfac2011SqlUtil sqlUtil = null; + + public AquavisDataBuilder(PropertyDeluxe properties, + Emfac2011SqlUtil sqlUtil) { + this.sqlUtil = sqlUtil; + this.properties = properties; + } + + public void createAquavisInputs() { + String scenarioId = properties + .getString(Emfac2011Properties.SCENARIO_ID); + String scenarioToken = properties + .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY); + + LOGGER.info("Step 1.1: Creating intrazonal Aquavis table..."); + sqlUtil.detemplifyAndRunScript( + properties + .getPath(Emfac2011Properties.CREATE_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY), + scenarioId, scenarioToken); + LOGGER.info("Step 1.2: Creating network Aquavis table..."); + sqlUtil.detemplifyAndRunScript( + properties + .getPath(Emfac2011Properties.CREATE_AQUAVIS_NETWORK_TEMPLATE_PROPERTY), + scenarioId, scenarioToken); + LOGGER.info("Step 1.3: Creating trips Aquavis table..."); + sqlUtil.detemplifyAndRunScript( + properties + .getPath(Emfac2011Properties.CREATE_AQUAVIS_TRIPS_TEMPLATE_PROPERTY), + scenarioId, scenarioToken); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java new file mode 100644 index 0000000..81fb2b9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java @@ -0,0 +1,45 @@ +package org.sandag.abm.reporting.emfac2011; +/** + * + * @author Wu.Sun@sandag.org + * + */ +public class Emfac2011AquavisIntrazonal { + protected int zone; + protected double distance; + protected double speed; + protected String region; + protected String aType; + public int getZone() { + return zone; + } + public void setZone(int zone) { + this.zone = zone; + } + public double getDistance() { + return distance; + } + public void setDistance(double distance) { + this.distance = distance; + } + public double getSpeed() { + return speed; + } + public void setSpeed(double speed) { + this.speed = speed; + } + public String getRegion() { + return region; + } + public void setRegion(String region) { + this.region = region; + } + public String getaType() { + return aType; + } + public void setaType(String aType) { + this.aType = aType; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java new file mode 100644 index 0000000..76a6fd4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java @@ -0,0 +1,302 @@ +package org.sandag.abm.reporting.emfac2011; + +import java.util.ArrayList; +import java.util.EnumMap; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.pb.sawdust.tabledata.DataRow; +import com.pb.sawdust.tabledata.DataTable; +import com.pb.sawdust.tabledata.TableIndex; +import com.pb.sawdust.tabledata.basic.BasicTableIndex; +import com.pb.sawdust.tabledata.basic.RowDataTable; +import com.pb.sawdust.tabledata.metadata.DataType; +import com.pb.sawdust.tabledata.metadata.TableSchema; +import com.pb.sawdust.util.property.PropertyDeluxe; + +/** + * The {@code AbstractEmfac2011Data} class is used to provide data used to + * modify the EMFAC2011 SG input file. It essentially reads in the generic + * AquaVis data (which represents the travel demand model results) and refactors + * it into a data table that is used (by {@link Emfac2011InputFileCreator}) to + * create an adjusted EMFAC2011 input file. + * + * @author crf Started 2/8/12 9:13 AM + * Modified by Wu.Sun@sandag.org 1/21/2014 + */ +public abstract class Emfac2011Data { + private static final Logger LOGGER = LoggerFactory + .getLogger(Emfac2011Data.class); + private final PropertyDeluxe properties; + private Emfac2011SqlUtil sqlUtil = null; + + /** + * Get a mapping from the tNCVehicle types listed in a model's AquaVis results + * to their corresponding EMFAC2011 tNCVehicle types. The returned map should + * have the (exact) names listed in the AquaVis results as keys, and the set + * of EMFAC2011 tNCVehicle types that the represent the AquaVis type. A single + * EMFAC2011 may be used in the mappings of multiple AquaVis types (there is + * no functional mapping requirement), and only mutable EMFAC2011 tNCVehicle + * types may be used in the mapping (see + * {@link com.pb.aquavis.emfac2011.Emfac2011VehicleType#getMutableVehicleTypes()} + * . Also, all aquavis tNCVehicle types must be represented in the map + * (even if it is an empty mapping). + * + * @return a map representing the relationship between the AquaVis and + * EMFAC2011 tNCVehicle types. + */ + protected abstract Map> getAquavisVehicleTypeToEmfacTypeMapping(); + + public Emfac2011Data(PropertyDeluxe properties, Emfac2011SqlUtil sqlUtil) { + this.sqlUtil = sqlUtil; + this.properties = properties; + } + + public DataTable processAquavisData(Emfac2011Properties properties) { + + String scenario = properties.getString(Emfac2011Properties.SCENARIO_ID); + ArrayList> network = queryNetwork(sqlUtil, scenario); + ArrayList> trips = queryTrips(sqlUtil, scenario); + ArrayList> intrazonal = queryIntrazonal(sqlUtil, + scenario); + + Map> areas = new HashMap<>( + properties + .> getMap(Emfac2011Properties.AREAS_PROPERTY)); + Map districtsToSubareas = new HashMap<>(); + for (String subarea : areas.keySet()) + for (String district : areas.get(subarea)) + districtsToSubareas.put(district, subarea); + + DataTable outputTable = buildEmfacDataTableShell(areas); + TableIndex index = new BasicTableIndex<>(outputTable, + Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD, + Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, + Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); + index.buildIndex(); + + // need to spread out speed fractions - by auto class + Map> aquavisVehicleTypeToEmfacTypeMapping = getAquavisVehicleTypeToEmfacTypeMapping(); + Map> vehicleFractions = buildVehicleFractioning(aquavisVehicleTypeToEmfacTypeMapping); + + LOGGER.info("Step 2.1: Aggregating aquavis network VMT data"); + for (ArrayList row : network) { + double len = new Double(row.get(3)).doubleValue(); + String vehicleType = row.get(6); + double speed = new Double(row.get(7)).doubleValue(); + double vol = new Double(row.get(8)).doubleValue(); + String district = row.get(9); + + if (districtsToSubareas.containsKey(district)) { + String subarea = districtsToSubareas.get(district); + for (Emfac2011VehicleType emfacVehicle : aquavisVehicleTypeToEmfacTypeMapping + .get(vehicleType)) { + double fraction = vehicleFractions.get(emfacVehicle).get( + vehicleType); + for (int r : index.getRowNumbers(Emfac2011SpeedCategory + .getSpeedCategory(speed).getName(), subarea, + emfacVehicle.getName())) + outputTable + .setCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, + (Double) outputTable + .getCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) + + fraction * vol * len); + } + } + } + + // need to collect intrazonal vmt and add it to network vmt - by auto + // class + LOGGER.info("Step 2.2: Aggregating aquavis intrazonal VMT data"); + HashMap intrazonalMap = convertIntrazonal(intrazonal); + for (ArrayList row : trips) { + int zone = new Integer(row.get(1)).intValue(); + String vClass = row.get(5); + int vol = new Integer(row.get(6)).intValue(); + String district = (String) intrazonalMap.get(zone).getRegion(); + if (districtsToSubareas.containsKey(district)) { + String subarea = districtsToSubareas.get(district); + double speed = intrazonalMap.get(zone).getSpeed(); + double vmt = intrazonalMap.get(zone).getDistance() * vol; + for (Emfac2011VehicleType emfacVehicle : aquavisVehicleTypeToEmfacTypeMapping + .get(vClass)) { + double fraction = vehicleFractions.get(emfacVehicle).get( + vClass); + for (int r : index.getRowNumbers(Emfac2011SpeedCategory + .getSpeedCategory(speed).getName(), subarea, + emfacVehicle.getName())) + outputTable + .setCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, + (Double) outputTable + .getCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) + + fraction * vmt); + } + } + } + + LOGGER.info("Step 2.3: Building speed fractions"); + // build fractions + index = new BasicTableIndex<>(outputTable, + Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, + Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); + index.buildIndex(); + for (Emfac2011VehicleType emfacVehicle : Emfac2011VehicleType + .getMutableVehicleTypes()) { + for (String subarea : areas.keySet()) { + double sum = 0.0; + int count = 0; + for (DataRow row : outputTable.getIndexedRows(index, subarea, + emfacVehicle.getName())) { + sum += row + .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD); + count++; + } + for (int r : index.getRowNumbers(subarea, + emfacVehicle.getName())) + outputTable + .setCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD, + sum == 0.0 ? 1.0 / count + : (Double) outputTable + .getCellValue( + r, + Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) + / sum); + } + } + + return outputTable; + } + + private DataTable buildEmfacDataTableShell(Map> areas) { + LOGGER.debug("Building EMFAC data table shell"); + TableSchema schema = new TableSchema("Emfac Data"); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, + DataType.STRING); + schema.addColumn( + Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, + DataType.DOUBLE); + schema.addColumn( + Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD, + DataType.DOUBLE); + DataTable outputTable = new RowDataTable(schema); + + // first, add rows for everything + for (Emfac2011VehicleType vehicle : Emfac2011VehicleType + .getMutableVehicleTypes()) + for (String subArea : areas.keySet()) + for (Emfac2011SpeedCategory speed : Emfac2011SpeedCategory + .values()) + outputTable.addRow(speed.getName(), subArea, + vehicle.getName(), 0.0, -1.0); // -1 + // is + // for + // error + // checking + // - + // 0 + // might + // pass + // through + return outputTable; // unnoticed + } + + private Map> buildVehicleFractioning( + Map> aquavisVehicleTypeToEmfacTypeMapping) { + // returns a map which says for every emfac tNCVehicle type, what aquavis + // tNCVehicle types should have their vmt added + // to it, and by what fraction + + Map> vehicleFractionMap = new EnumMap<>( + Emfac2011VehicleType.class); + for (Emfac2011VehicleType type : Emfac2011VehicleType + .getMutableVehicleTypes()) + vehicleFractionMap.put(type, new HashMap()); + for (String aquavisVehicleType : aquavisVehicleTypeToEmfacTypeMapping + .keySet()) { + double fraction = 1.0 / aquavisVehicleTypeToEmfacTypeMapping.get( + aquavisVehicleType).size(); + for (Emfac2011VehicleType type : aquavisVehicleTypeToEmfacTypeMapping + .get(aquavisVehicleType)) { + if (!vehicleFractionMap.containsKey(type)) + throw new IllegalStateException( + "Emfac tNCVehicle type is not mutable (" + + type + + ") and should not be component for aquavis type " + + aquavisVehicleType); + vehicleFractionMap.get(type).put(aquavisVehicleType, fraction); + } + } + return vehicleFractionMap; + } + + private HashMap convertIntrazonal( + ArrayList> intrazonal) { + HashMap result = new HashMap(); + for (ArrayList row : intrazonal) { + Emfac2011AquavisIntrazonal rec = new Emfac2011AquavisIntrazonal(); + int zone = new Integer(row.get(1)).intValue(); + rec.setZone(zone); + rec.setDistance(new Double(row.get(2))); + rec.setSpeed(new Double(row.get(3))); + rec.setRegion(row.get(4)); + rec.setaType(row.get(5)); + result.put(zone, rec); + } + return result; + } + + private ArrayList> queryNetwork(Emfac2011SqlUtil sqlUtil, + String schema) { + ArrayList> result = sqlUtil + .queryAquavisTables( + properties + .getPath(Emfac2011Properties.QUERY_AQUAVIS_NETWORK_TEMPLATE_PROPERTY), + schema, + properties + .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); + return result; + } + + private ArrayList> queryTrips(Emfac2011SqlUtil sqlUtil, + String schema) { + ArrayList> result = sqlUtil + .queryAquavisTables( + properties + .getPath(Emfac2011Properties.QUERY_AQUAVIS_TRIPS_TEMPLATE_PROPERTY), + schema, + properties + .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); + return result; + } + + private ArrayList> queryIntrazonal( + Emfac2011SqlUtil sqlUtil, String schema) { + ArrayList> result = sqlUtil + .queryAquavisTables( + properties + .getPath(Emfac2011Properties.QUERY_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY), + schema, + properties + .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); + return result; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java new file mode 100644 index 0000000..bdc94b2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java @@ -0,0 +1,86 @@ +package org.sandag.abm.reporting.emfac2011; + +/** + * + * @author Wu.Sun@sandag.org 1/20/2014 + * + */ + +public class Emfac2011Definitions { + + //Aquavis table content defintions + public static final String AQUAVIS_NETWORK_FROM_NODE_FIELD = "from_node"; + public static final String AQUAVIS_NETWORK_TO_NODE_FIELD = "to_node"; + public static final String AQUAVIS_NETWORK_LENGTH_FIELD = "length"; + public static final String AQUAVIS_NETWORK_LINK_CLASS_FIELD = "link_class"; + public static final String AQUAVIS_NETWORK_TIME_PERIOD_FIELD = "time_period"; + public static final String AQUAVIS_NETWORK_VEHICLE_CLASS_FIELD = "vehicle_class"; + public static final String AQUAVIS_NETWORK_SPEED_FIELD = "assigned_speed"; + public static final String AQUAVIS_NETWORK_VOLUME_FIELD = "volume"; + public static final String AQUAVIS_NETWORK_REGION_FIELD = "region"; + + public static final String AQUAVIS_INTRAZONAL_ZONE_FIELD = "zone"; + public static final String AQUAVIS_INTRAZONAL_LENGTH_FIELD = "distance"; + public static final String AQUAVIS_INTRAZONAL_SPEED_FIELD = "speed"; + public static final String AQUAVIS_INTRAZONAL_REGION_FIELD = "region"; + public static final String AQUAVIS_INTRAZONAL_AREA_TYPE_FIELD = "area_type"; + + public static final String AQUAVIS_TRIPS_ORIGIN_ZONE_FIELD = "origin_zone"; + public static final String AQUAVIS_TRIPS_DESTINATION_ZONE_FIELD = "destination_zone"; + public static final String AQUAVIS_TRIPS_HOUR_FIELD = "hour"; + public static final String AQUAVIS_TRIPS_TIME_PERIOD_FIELD = "time_period"; + public static final String AQUAVIS_TRIPS_VEHICLE_CLASS_FIELD = "vehicle_class"; + public static final String AQUAVIS_TRIPS_TRIPS_FIELD = "trips"; + + public static final String VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN = "EMFAC2011_MODE"; + + // Emfac2011 Data Table Definitions + public static final String EMFAC_2011_DATA_SUB_AREA_FIELD = "subarea"; + public static final String EMFAC_2011_DATA_SPEED_FIELD = "speed"; + public static final String EMFAC_2011_DATA_VEHICLE_TYPE_FIELD = "vehicle_type"; + public static final String EMFAC_2011_DATA_VMT_FIELD = "vmt"; + public static final String EMFAC_2011_DATA_SPEED_FRACTION_FIELD = "fraction"; + + // Emfac2011 Excel Input Sheets definitions + public static final String EMFAC_2011_SCENARIO_TABLE_NAME = "Regional_Scenarios"; + public static final String EMFAC_2011_SCENARIO_TABLE_GROUP_FIELD = "Group"; + public static final String EMFAC_2011_SCENARIO_TABLE_AREA_TYPE_FIELD = "Area Type"; + public static final String EMFAC_2011_SCENARIO_TABLE_AREA_FIELD = "Area"; + public static final String EMFAC_2011_SCENARIO_TABLE_YEAR_FIELD = "CalYr"; + public static final String EMFAC_2011_SCENARIO_TABLE_SEASON_FIELD = "Season"; + + public static final String EMFAC_2011_VMT_TABLE_NAME = "Scenario_Base_Inputs"; + public static final String EMFAC_2011_VMT_TABLE_GROUP_FIELD = "Group"; + public static final String EMFAC_2011_VMT_TABLE_AREA_FIELD = "Area"; + public static final String EMFAC_2011_VMT_TABLE_SCENARIO_FIELD = "Scenario"; + public static final String EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD = "Sub-Area"; + public static final String EMFAC_2011_VMT_TABLE_YEAR_FIELD = "CalYr"; + public static final String EMFAC_2011_VMT_TABLE_SEASON_FIELD = "Season"; + public static final String EMFAC_2011_VMT_TABLE_TITLE_FIELD = "Title"; + public static final String EMFAC_2011_VMT_TABLE_VMT_PROFILE_FIELD = "VMT Profile"; + public static final String EMFAC_2011_VMT_TABLE_VMT_BY_VEH_FIELD = "VMT by TNCVehicle Category"; + public static final String EMFAC_2011_VMT_TABLE_SPEED_PROFILE_FIELD = "Speed Profile"; + public static final String EMFAC_2011_VMT_TABLE_VMT_FIELD = "New Total VMT"; + + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_NAME = "Scenario_VMT_by_VehCat"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_GROUP_FIELD = "Group"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_AREA_FIELD = "Area"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SCENARIO_FIELD = "Scenario"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD = "Sub-Area"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_YEAR_FIELD = "CalYr"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SEASON_FIELD = "Season"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_TITLE_FIELD = "Title"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD = "Veh & Tech"; + public static final String EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD = "New VMT"; + + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_NAME = "Scenario_Speed_Profiles"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_GROUP_FIELD = "Group"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_AREA_FIELD = "Area"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SCENARIO_FIELD = "Scenario"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD = "Sub-Area"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_YEAR_FIELD = "CalYr"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SEASON_FIELD = "Season"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_TITLE_FIELD = "Title"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD = "Veh & Tech"; + public static final String EMFAC_2011_SPEED_FRACTION_TABLE_2007_VEHICLE_FIELD = "EMFAC2007 Veh & Tech"; +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java new file mode 100644 index 0000000..85a5cef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java @@ -0,0 +1,577 @@ +package org.sandag.abm.reporting.emfac2011; + +import static com.pb.sawdust.util.Range.range; +import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Arrays; +import java.util.EnumMap; +import java.util.HashMap; +import java.util.LinkedHashMap; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; +import java.util.Set; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import com.pb.sawdust.excel.tabledata.read.ExcelTableReader; +import com.pb.sawdust.excel.tabledata.write.ExcelTableWriter; +import com.pb.sawdust.tabledata.DataRow; +import com.pb.sawdust.tabledata.DataTable; +import com.pb.sawdust.tabledata.TableIndex; +import com.pb.sawdust.tabledata.basic.BasicTableIndex; +import com.pb.sawdust.tabledata.basic.RowDataTable; +import com.pb.sawdust.tabledata.metadata.DataType; +import com.pb.sawdust.tabledata.metadata.TableSchema; +import com.pb.sawdust.tabledata.write.TableWriter; +import com.pb.sawdust.util.ProcessUtil; +import com.pb.sawdust.util.exceptions.RuntimeIOException; + +/** + * The {@code InputTemplateCreator} class is used to build an adjusted input + * file used for running the EMFAC2011 SG model. + * + * + * @author crf Started 2/7/12 1:48 PM Modified by Wu.Sun@sandag.org 1/21/2014 + */ +public class Emfac2011InputFileCreator +{ + private static final Logger LOGGER = LoggerFactory.getLogger(Emfac2011Data.class); + + private static final String[] SEASONS = {"ANNUAL", "SUMMER", "WINTER"}; + + /** + * Create an input file that can be used with the EMFAC2011 SG model. The + * constructed input file will use the default EMFAC2011 + * parameters/specifications, with adjustments based on a travel demand + * model's results. + * + * @param properties + * The properties specific to the model run. + * + * @param emfacModelData + * A data table (obtained from + * {@link Emfac2011Data#processAquavisData(Emfac2011Properties)} + * )) holding the results of the travel demand model. + * + * @return an EMFAC2011 SG input file, adjusted using {@code emfacModelData} + * . + */ + public Path createInputFile(Emfac2011Properties properties, DataTable emfacModelData) + { + String areaType = properties.getString(Emfac2011Properties.AREA_TYPE_PROPERTY); + String region = properties.getString(Emfac2011Properties.REGION_NAME_PROPERTY); + // Set areas = new + // HashSet<>(properties.getList(Emfac2011Properties.AREAS_PROPERTY)); + // Map> areas = new + // HashMap<>(properties.>getMap(Emfac2011Properties.AREAS_PROPERTY)); + Set areas = new HashMap<>( + properties.>getMap(Emfac2011Properties.AREAS_PROPERTY)) + .keySet(); + int oriYear = properties.getInt(Emfac2011Properties.YEAR_PROPERTY); + int year = Math.min(oriYear, 2035); + String inventoryDir = Paths.get( + properties.getString(Emfac2011Properties.EMFAC2011_INSTALLATION_DIR_PROPERTY), + "Application Files/Inventory Files").toString(); + String outputDir = properties.getString(Emfac2011Properties.OUTPUT_DIR_PROPERTY); + String converterProgram = properties + .getString(Emfac2011Properties.XLS_CONVERTER_PROGRAM_PROPERTY); + boolean preserveEmfacVehicleFractions = properties + .getBoolean(Emfac2011Properties.PRESERVE_EMFAC_VEHICLE_FRACTIONS_PROPERTY); + boolean modelVmtIncludesNonMutableVehicleTypes = properties + .getBoolean(Emfac2011Properties.MODEL_VMT_INCLUDES_NON_MUTABLES_PROPERTY); + return createInputFile(areaType, region, areas, year, oriYear, emfacModelData, + preserveEmfacVehicleFractions, modelVmtIncludesNonMutableVehicleTypes, + inventoryDir, outputDir, converterProgram); + } + + private String formInventoryFileName(String area, String season, int year) + { + return "EMFAC2011-SG Inventory - " + area + " - " + year + " (" + season + ").xls"; + } + + private void convertFile(String inventoryFile, String outputInventoryFile, + String converterProgram) + { + ProcessUtil.runProcess(Arrays.asList(converterProgram, inventoryFile, outputInventoryFile)); + } + + private DataTable readInventoryTable(String inventoryFile) + { + return new RowDataTable(ExcelTableReader.excelTableReader(inventoryFile)); + } + + private Path createInputFile(String areaType, String region, Set areas, int year, + int oriYear, DataTable emfacModelData, boolean preserveEmfacVehicleFractions, + boolean modelVmtIncludesNonMutableVehicleTypes, String inventoryDir, String outputDir, + String converterProgram) + { + Path outputFile = Paths.get(outputDir, formOutputFileName(region, oriYear)); + try + { + Files.deleteIfExists(outputFile); + } catch (IOException e) + { + throw new RuntimeIOException(e); + } + + TableWriter writer = new ExcelTableWriter(outputFile.toFile()); + LOGGER.debug("Initializing input excel file: " + outputFile); + writer.writeTable(formScenarioTable(areaType, region, year, SEASONS)); + + DataTable masterVmtTable = initVmtTable(); + DataTable masterVehicleVmtTable = initVehicleVmtTable(); + DataTable masterVmtSpeedTable = initVmtSpeedTable(); + + for (int i=0;i inputTables = new LinkedHashMap<>(); + for (String area : areas) + { + String inventoryFile = Paths.get(inventoryDir, + formInventoryFileName(area, SEASONS[i], year)).toString(); + Path outputInventoryFile = Paths.get(outputDir, + formInventoryFileName(area, SEASONS[i], year)); + convertFile(inventoryFile, outputInventoryFile.toString(), converterProgram); + inputTables.put(area, readInventoryTable(outputInventoryFile.toString())); + try + { + Files.delete(outputInventoryFile); + } catch (IOException e) + { + throw new RuntimeIOException(e); + } + } + + LOGGER.debug("Building vmt table"); + DataTable vmtTable = extractVmtTables(inputTables, i+1, region, year, SEASONS[i]); + LOGGER.debug("Building vmt by tNCVehicle type table"); + DataTable vehicleVmtTable = extractVmtVehicleTables(inputTables, i+1, region, year, SEASONS[i]); + LOGGER.debug("Building speed fraction table"); + DataTable vmtSpeedTable = extractVmtSpeedTables(inputTables, i+1, region, year, SEASONS[i]); + LOGGER.debug("Shifting tables using model data"); + shiftVmtTables(vmtTable, vehicleVmtTable, areas, emfacModelData, + preserveEmfacVehicleFractions, modelVmtIncludesNonMutableVehicleTypes); + shiftSpeedFractionTable(vmtSpeedTable, emfacModelData); + LOGGER.debug("Writing tables"); + + appendDataTable(masterVmtTable, vmtTable); + appendDataTable(masterVehicleVmtTable, vehicleVmtTable); + appendDataTable(masterVmtSpeedTable, vmtSpeedTable); + } + + writer.writeTable(masterVmtTable); + writer.writeTable(masterVehicleVmtTable); + writer.writeTable(masterVmtSpeedTable); + return outputFile; + } + + private void appendDataTable(DataTable master, DataTable fragment) + { + for(DataRow row : fragment) + { + master.addRow(row); + } + } + + private String formOutputFileName(String region, int year) + { + return "EMFAC2011-" + region + "-" + year + ".xls"; + } + + private DataTable formScenarioTable(String areaType, String area, int year, String[] seasons) + { + TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_NAME); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_GROUP_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_AREA_TYPE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_AREA_FIELD, DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_YEAR_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_SEASON_FIELD, + DataType.STRING); + + DataTable table = new RowDataTable(schema); + + for (int i = 0; i < seasons.length; i++) + { + List row = new LinkedList<>(); + row.add(i+1); + row.add(areaType); + row.add(area); + row.add(year); + row.add(seasons[i]); + table.addRow(row.toArray(new Object[row.size()])); + } + return table; + } + + private DataTable initVmtTable() + { + TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_NAME); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_GROUP_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_AREA_FIELD, DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SCENARIO_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD, DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_YEAR_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SEASON_FIELD, DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_TITLE_FIELD, DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_PROFILE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_BY_VEH_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SPEED_PROFILE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD, DataType.DOUBLE); + return new RowDataTable(schema); + } + + private DataTable initVehicleVmtTable() + { + TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_NAME); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_GROUP_FIELD, + DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_AREA_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SCENARIO_FIELD, + DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_YEAR_FIELD, DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SEASON_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_TITLE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD, + DataType.DOUBLE); + return new RowDataTable(schema); + } + + private DataTable initVmtSpeedTable() + { + TableSchema schema = new TableSchema( + Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_NAME); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_GROUP_FIELD, + DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_AREA_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SCENARIO_FIELD, + DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_YEAR_FIELD, + DataType.INT); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SEASON_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_TITLE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD, + DataType.STRING); + schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_2007_VEHICLE_FIELD, + DataType.STRING); + for (Emfac2011SpeedCategory category : Emfac2011SpeedCategory.values()) + schema.addColumn(category.getName(), DataType.DOUBLE); + return new RowDataTable(schema); + } + + private DataTable extractVmtTables(Map inputTables, int group, String area, int year, + String season) + { + DataTable vmtTable = initVmtTable(); + + int counter = 1; + for (String subArea : inputTables.keySet()) + { + List row = new LinkedList<>(); + row.add(group); + row.add(area); + row.add(counter); + row.add(subArea); + row.add(year); + row.add(season); + row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter++, + subArea, year, season)); + row.add("User"); + row.add("User"); + row.add("User"); + + double totalVmt = 0.0; + for (DataRow r : inputTables.get(subArea)) + if (r.getCellAsString("Tech").equals("TOT")) totalVmt += r.getCellAsDouble("VMT"); + row.add(totalVmt); + vmtTable.addRow(row.toArray(new Object[row.size()])); + } + return vmtTable; + } + + private DataTable extractVmtVehicleTables(Map inputTables, int group, String area, + int year, String season) + { + DataTable vehicleVmtTable = initVehicleVmtTable(); + + int counter = 1; + for (String subArea : inputTables.keySet()) + { + for (DataRow r : inputTables.get(subArea)) + { + String tech = r.getCellAsString("Tech"); + if (tech.equals("DSL") || tech.equals("GAS")) + { + List row = new LinkedList<>(); + row.add(group); + row.add(area); + row.add(counter); + row.add(subArea); + row.add(year); + row.add(season); + row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter, + subArea, year, season)); + row.add(r.getCellAsString("Veh & Tech")); + row.add(r.getCellAsDouble("VMT")); + vehicleVmtTable.addRow(row.toArray(new Object[row.size()])); + } + } + counter++; + } + return vehicleVmtTable; + } + + private DataTable extractVmtSpeedTables(Map inputTables, int group, String area, + int year, String season) + { + DataTable vmtSpeedTable = initVmtSpeedTable(); + + int counter = 1; + for (String subArea : inputTables.keySet()) + { + // loop over everything once to get sums and types, and then second + // time to generate fractions + Map techTotals = new LinkedHashMap<>(); + for (DataRow r : inputTables.get(subArea)) + { + String tech = r.getCellAsString("Tech"); + if (tech.startsWith("Spd") && !tech.endsWith("TOT")) + { + String vehNTech = r.getCellAsString("Veh & Tech").toLowerCase(); + if (!techTotals.containsKey(vehNTech)) techTotals.put(vehNTech, 0.0); + techTotals.put(vehNTech, techTotals.get(vehNTech) + r.getCellAsDouble("VMT")); + } + } + Map techRows = new HashMap<>(); + // loop over each type and add in the rows + for (Emfac2011VehicleType type : Emfac2011VehicleType.values()) + { + List row = new LinkedList<>(); + row.add(group); + row.add(area); + row.add(counter); + row.add(subArea); + row.add(year); + row.add(season); + row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter, + subArea, year, season)); + row.add(type.getName()); + row.add(type.getEmfac2007Name()); + for (int i : range(5, 71, 5)) + row.add(0.0); + vmtSpeedTable.addRow(row.toArray(new Object[row.size()])); + techRows.put(type.getName().toLowerCase(), vmtSpeedTable.getRowCount() - 1); + } + // now reloop over table to get fractions + for (DataRow r : inputTables.get(subArea)) + { + String tech = r.getCellAsString("Tech"); + if (tech.startsWith("Spd") && !tech.endsWith("TOT")) + { + String vehNTech = r.getCellAsString("Veh & Tech").toLowerCase(); + double fraction = techTotals.get(vehNTech) == 0.0 ? 0.0 : r + .getCellAsDouble("VMT") / techTotals.get(vehNTech); + vmtSpeedTable.setCellValue(techRows.get(vehNTech), + Integer.parseInt(tech.substring(3, 5)) + "MPH", fraction); + } + } + counter++; + } + // ensure that we sum up to one + counter = 0; + for (DataRow r : vmtSpeedTable) + { + double sum = 1.0; + Emfac2011SpeedCategory[] speeds = Emfac2011SpeedCategory.values(); + for (int i : range(speeds.length - 1)) + sum -= r.getCellAsDouble(speeds[i].getName()); + vmtSpeedTable.setCellValue(counter++, Emfac2011SpeedCategory.SPEED_65_70plus.getName(), + sum); + } + return vmtSpeedTable; + } + + // private void shiftSpeedFractionTable(DataTable speedVmtTable, DataTable + // modelData) { + private void shiftSpeedFractionTable(DataTable speedVmtTable, DataTable modelData) + { + TableIndex index = new BasicTableIndex<>(speedVmtTable, + Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD, + Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD); + index.buildIndex(); + + for (DataRow row : modelData) + { + String subArea = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD); + String vehicleType = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); + String category = Emfac2011SpeedCategory.getTypeForName( + row.getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD)) + .getName(); + speedVmtTable.setCellValue(index.getRowNumbers(subArea, vehicleType).iterator().next(), + category, + row.getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD)); + } + } + + private void shiftVmtTables(DataTable vmtTable, DataTable vehicleVmtTable, Set areas, + DataTable modelData, boolean preserveEmfacVehicleFractions, + boolean modelVmtIncludesNonMutableVehicleTypes) + { + Map> modelVmtByAreaAndVehicleType = new HashMap<>(); + Map> emfacMutableVmtByAreaAndVehicleType = new HashMap<>(); + Map> emfacImmutableVmtByAreaAndVehicleType = new HashMap<>(); + + Set mutableVehicleTypes = Emfac2011VehicleType + .getMutableVehicleTypes(); + + for (String subarea : areas) + { + Map m1 = new EnumMap<>(Emfac2011VehicleType.class); + Map m2 = new EnumMap<>(Emfac2011VehicleType.class); + Map m3 = new EnumMap<>(Emfac2011VehicleType.class); + for (Emfac2011VehicleType vehicleType : Emfac2011VehicleType.values()) + { + if (mutableVehicleTypes.contains(vehicleType)) + { + m1.put(vehicleType, 0.0); + m2.put(vehicleType, 0.0); + } else + { + if (modelVmtIncludesNonMutableVehicleTypes) m1.put(vehicleType, 0.0); + m3.put(vehicleType, 0.0); + } + } + modelVmtByAreaAndVehicleType.put(subarea, m1); + emfacMutableVmtByAreaAndVehicleType.put(subarea, m2); + emfacImmutableVmtByAreaAndVehicleType.put(subarea, m3); + } + + for (DataRow row : vehicleVmtTable) + { + String subArea = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD); + Emfac2011VehicleType vehicleType = Emfac2011VehicleType + .getVehicleType(row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD)); + double vmt = row + .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD); + if (mutableVehicleTypes.contains(vehicleType)) emfacMutableVmtByAreaAndVehicleType.get( + subArea).put(vehicleType, + emfacMutableVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); + else emfacImmutableVmtByAreaAndVehicleType.get(subArea).put(vehicleType, + emfacImmutableVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); + } + + for (DataRow row : modelData) + { + String subArea = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD); + Emfac2011VehicleType vehicleType = Emfac2011VehicleType.getVehicleType(row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD)); + double vmt = row.getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD); + modelVmtByAreaAndVehicleType.get(subArea).put(vehicleType, + modelVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); + } + + if (preserveEmfacVehicleFractions) + { + // need to reshift data + for (String subarea : modelVmtByAreaAndVehicleType.keySet()) + { + double totalVmt = 0.0; + Map modelVmt = modelVmtByAreaAndVehicleType + .get(subarea); + for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) + totalVmt += modelVmt.get(vehicleType); + double totalEmfacVmt = 0.0; + Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType + .get(subarea); + Map emfacImutableVmt = emfacImmutableVmtByAreaAndVehicleType + .get(subarea); + for (Emfac2011VehicleType vehicleType : emfacMutableVmt.keySet()) + totalEmfacVmt += emfacMutableVmt.get(vehicleType); + if (modelVmtIncludesNonMutableVehicleTypes) + for (Emfac2011VehicleType vehicleType : emfacImutableVmt.keySet()) + totalEmfacVmt += emfacImutableVmt.get(vehicleType); + for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) + { + if (emfacMutableVmt.containsKey(vehicleType)) modelVmt.put(vehicleType, + totalVmt * emfacMutableVmt.get(vehicleType) / totalEmfacVmt); + else modelVmt.put(vehicleType, totalVmt * emfacImutableVmt.get(vehicleType) + / totalEmfacVmt); + } + } + } + + // shift overall vmt + vmtTable.setPrimaryKey(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD); + for (DataRow row : vmtTable) + { + String subArea = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD); + double originalVmt = row + .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD); + Map modelVmt = modelVmtByAreaAndVehicleType.get(subArea); + Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType + .get(subArea); + Map emfacImutableVmt = emfacImmutableVmtByAreaAndVehicleType + .get(subArea); + for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) + { + originalVmt += modelVmt.get(vehicleType); // add corrected vmt + originalVmt -= emfacMutableVmt.containsKey(vehicleType) ? emfacMutableVmt + .get(vehicleType) : emfacImutableVmt.get(vehicleType); // subtract + // old + // vmt + } + vmtTable.setCellValueByKey(subArea, + Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD, originalVmt); + } + + // replace tNCVehicle vmts + TableIndex index = new BasicTableIndex<>(vehicleVmtTable, + Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD, + Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD); + index.buildIndex(); + for (DataRow row : vehicleVmtTable) + { + String subArea = row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD); + Map modelVmt = modelVmtByAreaAndVehicleType.get(subArea); + Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType + .get(subArea); + Emfac2011VehicleType vehicleType = Emfac2011VehicleType + .getVehicleType(row + .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD)); + if (modelVmt.containsKey(vehicleType)) + { + double vmt = row + .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD); + if (modelVmtIncludesNonMutableVehicleTypes) vmt = modelVmt.get(vehicleType); + else vmt += modelVmt.get(vehicleType) - emfacMutableVmt.get(vehicleType); + vehicleVmtTable.setCellValue(index.getRowNumbers(subArea, vehicleType.getName()) + .iterator().next(), + Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD, vmt); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java new file mode 100644 index 0000000..b3a2031 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java @@ -0,0 +1,101 @@ +package org.sandag.abm.reporting.emfac2011; + +import com.pb.sawdust.util.property.PropertyDeluxe; + +/** + * The {@code Emfac2011Properties} holds the properties used by the various + * classes in this package. The property keys listed in this class (as + * {@code public static final} constants) must all be present in the + * instantiated properties object, or it will throw an error during + * construction. Additionally, {@link #AREAS_PROPERTY} must be typed correctly + * as a map from a string to a list holding strings (see its documentation for + * more information). + * + * @author crf Started 2/9/12 8:40 AM + * Modified by Wu.Sun@sanag.org 1/28/2014 + */ +public class Emfac2011Properties + extends PropertyDeluxe +{ + //database properties + public static final String REPORTS_DATABASE_IPADDRESS_PROPERTY = "reports.database.ipaddress"; + public static final String REPORTS_DATABASE_PORT_PROPERTY = "reports.database.port"; + public static final String REPORTS_DATABASE_NAME_PROPERTY = "reports.database.name"; + public static final String REPORTS_DATABASE_USERNAME_PROPERTY = "reports.database.username"; + public static final String REPORTS_DATABASE_PASSWORD_PROPERTY = "reports.database.password"; + public static final String REPORTS_DATABASE_INSTANCE_PROPERTY = "reports.database.instance"; + + //San Diego Emfac2011 properties + public static final String AREA_TYPE_PROPERTY = "emfac.2011.area.type"; + public static final String REGION_NAME_PROPERTY = "emfac.2011.region.name"; + public static final String AREAS_PROPERTY = "emfac.2011.area"; + public static final String SEASON_PROPERTY = "emfac.2011.season"; + public static final String YEAR_PROPERTY = "emfac.2011.year"; + public static final String EMFAC2011_INSTALLATION_DIR_PROPERTY = "emfac.2011.installation.dir"; + public static final String OUTPUT_DIR_PROPERTY = "emfac.2011.output.dir"; + public static final String AQUAVIS_INTRAZONAL_FILE_PROPERTY = "emfac.2011.aquavis.intrazonal"; + + // Aquavis table creation templates + public static final String CREATE_AQUAVIS_NETWORK_TEMPLATE_PROPERTY = "aquavis.network.sql.template"; + public static final String CREATE_AQUAVIS_TRIPS_TEMPLATE_PROPERTY = "aquavis.trips.sql.template"; + public static final String CREATE_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY = "aquavis.intrazonal.sql.template"; + + // Aquavis table query templates + public static final String QUERY_AQUAVIS_NETWORK_TEMPLATE_PROPERTY = "aquavis.network.query.template"; + public static final String QUERY_AQUAVIS_TRIPS_TEMPLATE_PROPERTY = "aquavis.trips.query.template"; + public static final String QUERY_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY = "aquavis.intrazonal.query.template"; + + // inputs, outputs, and tokens + public static final String AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY= "aquavis.template.scenarioId.token"; + public static final String SCENARIO_ID = "scenario.id"; + public static final String VEHICLE_CODE_MAPPING_FILE_PROPERTY = "emfac.2011.to.sandag.vehicle.code.mapping.file"; + + // switch if EMFAC is executed + public static final String EXECUTE_EMFAC="execute.emfac"; + + /** + * The property key for the boolean indicating if the (default) EMFAC + * tNCVehicle fractions should be preserved in the EMFAC input file (value is + * {@code true}), or if the model tNCVehicle fractions should be used (value is + * {@code false}). + */ + public static final String PRESERVE_EMFAC_VEHICLE_FRACTIONS_PROPERTY = "emfac.2011.preserve.emfac.vehicle.fractions"; + /** + * The property key for the boolean indicating whether or not the model + * (travel demand, not EMFAC) VMT includes totals for non-mutable tNCVehicle + * types. If it does, then the VMT will be scaled before adjusting the EMFAC + * input file. + */ + public static final String MODEL_VMT_INCLUDES_NON_MUTABLES_PROPERTY = "emfac.2011.model.vmt.includes.non.mutable.vehicles"; + + /** + * The property key for the location of the xls converter program. This + * program converts the malformed EMFAC2011 reference files to a (strictly) + * valid format, which can then be used by the rest of the model. + */ + public static final String XLS_CONVERTER_PROGRAM_PROPERTY = "emfac.2011.xls.converter.program"; + + /** + * The property key for the location (full path) of the AquaVis output + * intrazonal file. + */ + + /** + * Constructor specifying the resources used to build the properties. + * + * @param firstResource + * The first properties resource. + * + * @param additionalResources + * Any additional properties resources. + * + * @throws IllegalArgumentException + * if any of the required properties is missing, or if they are + * typed incorrectly. + */ + public Emfac2011Properties(String firstResource) + { + super(firstResource); + System.out.println("first property="+firstResource); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java new file mode 100644 index 0000000..4622be2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java @@ -0,0 +1,216 @@ +package org.sandag.abm.reporting.emfac2011; + +import java.io.IOException; +import java.nio.file.FileSystems; +import java.nio.file.FileVisitOption; +import java.nio.file.FileVisitResult; +import java.nio.file.FileVisitor; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.PathMatcher; +import java.nio.file.Paths; +import java.nio.file.SimpleFileVisitor; +import java.nio.file.attribute.BasicFileAttributes; +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumMap; +import java.util.EnumSet; +import java.util.HashMap; +import java.util.LinkedHashSet; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.pb.sawdust.tabledata.DataRow; +import com.pb.sawdust.tabledata.DataTable; +import com.pb.sawdust.tabledata.basic.RowDataTable; +import com.pb.sawdust.tabledata.read.CsvTableReader; +import com.pb.sawdust.util.ProcessUtil; +import com.pb.sawdust.util.exceptions.RuntimeIOException; + +/** + * The {@code Emfac2011Runner} class is used to generate an EMFAC2011 SG input + * file adjusted for travel demand model results, and then run (via the end + * user) the EMFAC2011 SG model using those inputs. + * + * @author crf Started 2/9/12 9:17 AM + * Modified by Wu.Sun@sandag.org 1/24/2014 + */ +public class Emfac2011Runner { + private static final Logger LOGGER = LoggerFactory + .getLogger(Emfac2011Runner.class); + private final Emfac2011Properties properties; + private Emfac2011SqlUtil sqlUtil;; + + /** + * Constructor specifying the resources used to build the properties used + * for the EMFAC2011 SG run. + * + * @param propertyResource + * The first properties resource. + * + * @param additionalResources + * Any additional properties resources. + */ + public Emfac2011Runner(String propertyResource) { + properties = new Emfac2011Properties(propertyResource); + sqlUtil = new Emfac2011SqlUtil(properties); + } + + public void runEmfac2011() { + LOGGER.info("***************Running Emfac2011 for SANDAG***********************"); + LOGGER.info("Step 0: Setting up mutable tNCVehicle types"); + // have to call this first because it sets the mutable types, which are, + // used throughout the EMFAC2011 process + Path path = Paths + .get(properties.getString(Emfac2011Properties.VEHICLE_CODE_MAPPING_FILE_PROPERTY)); + + final Map> aquavisVehicleTypeToEmfacMapping = buildAquavisVehicleTypeToEmfacMapping(path); + + runEmfac2011(new Emfac2011Data(properties, sqlUtil) { + @Override + protected Map> getAquavisVehicleTypeToEmfacTypeMapping() { + return aquavisVehicleTypeToEmfacMapping; + } + }); + } + + /** + * Run the EMFAC2011 model. This method will process the model results (via + * AquaVis outputs), create an adjusted EMFAC2011 input file, and initiate + * the EMFAC2011 SG model. Because of the way it is set up, the user must + * actually set up and run the EMFAC2011 SG model, but this method will + * create a dialog window which will walk the user through the steps + * required to do that. + * + * @param emfac2011Data + * The {@code Emfac2011Data} instance corresponding to the model + * results/run. + */ + public void runEmfac2011(Emfac2011Data emfac2011Data) { + LOGGER.info("Step 1: Building Aquavis inputs from scenario: " + + properties.getString(Emfac2011Properties.SCENARIO_ID)); + AquavisDataBuilder builder = new AquavisDataBuilder(properties, sqlUtil); + builder.createAquavisInputs(); + LOGGER.info("Step 2: Processing aquavis data"); + DataTable data = emfac2011Data.processAquavisData(properties); + LOGGER.info("Step 3: Creating EMFAC2011 input file"); + Emfac2011InputFileCreator inputFileCreator = new Emfac2011InputFileCreator(); + Path inputfile = inputFileCreator.createInputFile(properties, data); + if((properties.getString(Emfac2011Properties.EXECUTE_EMFAC)).equalsIgnoreCase("true")){ + LOGGER.info("Step 4: Initiating EMFAC2011"); + RunEmfacDialog.createAndShowGUI(inputfile, this); + }else{ + LOGGER.info("Sipped--Step 4: Initiating EMFAC2011"); + } + LOGGER.info("EMFAC2011 run finished"); + } + + private Map> buildAquavisVehicleTypeToEmfacMapping( + Path vehicleCodeMappingFile) { + Map> mapping = new EnumMap<>( + SandagAutoModes.class); + for (SandagAutoModes type : SandagAutoModes.values()) + mapping.put(type, EnumSet.noneOf(Emfac2011VehicleType.class)); + + // file has one column = + // VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN + // the rest have names which, when made uppercase, should match + // VehicleType enum + DataTable vehicleCodeMapping = new RowDataTable(new CsvTableReader( + vehicleCodeMappingFile.toString())); + vehicleCodeMapping.setDataCoersion(true); + Set vehicleCodeColumns = new LinkedHashSet<>(); + for (String column : vehicleCodeMapping.getColumnLabels()) { + try { + SandagAutoModes.valueOf(column.toUpperCase()); + vehicleCodeColumns.add(column); + } catch (IllegalArgumentException e) { + // absorb - not a valid type column + } + } + Set mutableVehicleType = EnumSet + .noneOf(Emfac2011VehicleType.class); + for (DataRow row : vehicleCodeMapping) { + Emfac2011VehicleType emfac2011VehicleType = Emfac2011VehicleType + .getVehicleType(row + .getCellAsString(Emfac2011Definitions.VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN)); + // now dynamically setting mutable tNCVehicle types, so we need to not + // rely on the defaults + // if (!emfac2011VehicleType.isMutableType()) + // continue; //skip any non-mutable types, as they can't be used + for (String column : vehicleCodeColumns) { + if (row.getCellAsBoolean(column)) { + mutableVehicleType.add(emfac2011VehicleType); // if a + // mapping + // exists, + // then the + // EMFAC + // tNCVehicle + // type is + // assumed + // to + // be + // mutable + mapping.get(SandagAutoModes.valueOf(column.toUpperCase())) + .add(emfac2011VehicleType); + } + } + } + Emfac2011VehicleType.setMutableTypes(mutableVehicleType); + Map> finalMapping = new HashMap<>(); + for (SandagAutoModes type : mapping.keySet()) + finalMapping.put(type.name(), mapping.get(type)); + return finalMapping; + } + + void runEmfac2011Program() { + final Path emfacInstallationDir = Paths + .get(properties + .getString(Emfac2011Properties.EMFAC2011_INSTALLATION_DIR_PROPERTY)); + final PathMatcher matcher = FileSystems.getDefault().getPathMatcher( + "glob:*.lnk"); + final List link = new LinkedList<>(); + FileVisitor visitor = new SimpleFileVisitor() { + + @Override + public FileVisitResult visitFile(Path file, + BasicFileAttributes attrs) throws IOException { + Path name = file.getFileName(); + if (name != null && matcher.matches(name)) { + link.add(file); + return FileVisitResult.TERMINATE; + } + return FileVisitResult.CONTINUE; + } + }; + + try { + Files.walkFileTree(emfacInstallationDir, + Collections. emptySet(), 1, visitor); + } catch (IOException e) { + throw new RuntimeIOException(e); + } + + if (link.size() == 0) + throw new IllegalStateException( + "Cannot find Emfac2011 shortcut in " + emfacInstallationDir); + ProcessUtil.runProcess(Arrays.asList("cmd", "/c", link.get(0) + .toString())); + } + + public static void main(String... args) { + double startTime = System.currentTimeMillis(); + // do work + new Emfac2011Runner(args[0]).runEmfac2011(); + // time stamp + LOGGER.info("Emfac2011 completed in: " + + (float) (((System.currentTimeMillis() - startTime) / 1000.0) / 60.0) + + " minutes."); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java new file mode 100644 index 0000000..1d76b98 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java @@ -0,0 +1,73 @@ +package org.sandag.abm.reporting.emfac2011; + +/** + * The {@code EmfacSpeedClass} enum represents the speed categories used in the + * EMFAC2011 model. + * + * @author crf Started 2/9/12 5:58 AM + */ +public enum Emfac2011SpeedCategory +{ + SPEED_0_5(5, "5MPH"), SPEED_5_10(10, "10MPH"), SPEED_10_15(15, "15MPH"), SPEED_15_20(20, + "20MPH"), SPEED_20_25(25, "25MPH"), SPEED_25_30(30, "30MPH"), SPEED_30_35(35, "35MPH"), SPEED_35_40( + 40, "40MPH"), SPEED_40_45(45, "45MPH"), SPEED_45_50(50, "50MPH"), SPEED_50_55(55, + "55MPH"), SPEED_55_60(60, "60MPH"), SPEED_60_65(65, "65MPH"), SPEED_65_70plus(1000, + "70MPH"); // big upper bound + + private final String name; + private final double upperBound; + + private Emfac2011SpeedCategory(double upperBound, String name) + { + this.upperBound = upperBound; + this.name = name; + } + + /** + * Get the EMFAC name for this speed category. + * + * @return this speed category's name. + */ + public String getName() + { + return name; + } + + /** + * Get the {@code SpeedCategory} for a given speed. + * + * @param speed + * The speed in question. + * + * @return the {@code SpeedCategory} for the given speed. + * + * @throws IllegalArgumentException + * if {@code speed} < 0. + */ + public static Emfac2011SpeedCategory getSpeedCategory(double speed) + { + if (speed < 0) + throw new IllegalArgumentException("Negative speeds are not allowed: " + speed); + for (Emfac2011SpeedCategory sc : values()) + if (speed <= sc.upperBound) return sc; + throw new IllegalStateException("Couldn't find speed category for " + speed); + } + + /** + * Get the {@code SpeedCategory} corresponding to an EMFAC name. + * + * @param name + * The EMFAC speed category name. + * + * @return the {@code SpeedCategory} corresponding to {@code name}. + * + * @throws IllegalArgumentException + * if {@code name} is not a valid EMFAC speed category name. + */ + public static Emfac2011SpeedCategory getTypeForName(String name) + { + for (Emfac2011SpeedCategory type : values()) + if (type.name.equals(name)) return type; + throw new IllegalArgumentException("Type for name not found: " + name); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java new file mode 100644 index 0000000..56a719e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java @@ -0,0 +1,145 @@ +package org.sandag.abm.reporting.emfac2011; + +import java.io.FileReader; +import java.io.IOException; +import java.nio.file.Path; +import java.sql.Connection; +import java.sql.DriverManager; +import java.sql.ResultSet; +import java.sql.SQLException; +import java.sql.Statement; +import java.util.ArrayList; + +import nl.tudelft.simulation.logger.Logger; + +import com.pb.sawdust.util.exceptions.RuntimeIOException; +import com.pb.sawdust.util.exceptions.RuntimeWrappingException; +import com.pb.sawdust.util.property.PropertyDeluxe; + +/** + * @author crf Started 2/9/12 9:17 AM + * Modified by Wu.Sun@sandag.org 1/21/2014 + */ + +public class Emfac2011SqlUtil { + + private final String connectionUrl; + + public Emfac2011SqlUtil(PropertyDeluxe properties) { + connectionUrl = formConnectionUrl( + properties + .getString(Emfac2011Properties.REPORTS_DATABASE_IPADDRESS_PROPERTY), + properties + .getInt(Emfac2011Properties.REPORTS_DATABASE_PORT_PROPERTY), + properties + .getString(Emfac2011Properties.REPORTS_DATABASE_NAME_PROPERTY), + properties + .hasKey(Emfac2011Properties.REPORTS_DATABASE_USERNAME_PROPERTY) ? properties + .getString(Emfac2011Properties.REPORTS_DATABASE_USERNAME_PROPERTY) + : null, + properties + .hasKey(Emfac2011Properties.REPORTS_DATABASE_PASSWORD_PROPERTY) ? properties + .getString(Emfac2011Properties.REPORTS_DATABASE_PASSWORD_PROPERTY) + : null, + properties + .hasKey(Emfac2011Properties.REPORTS_DATABASE_INSTANCE_PROPERTY) ? properties + .getString(Emfac2011Properties.REPORTS_DATABASE_INSTANCE_PROPERTY) + : null); + } + + public void detemplifyAndRunScript(Path script, String scenarioId, + String scenarioIdToken) { + String s = readFile(script).replace(scenarioIdToken, scenarioId); + try (Connection connection = getConnection(); + Statement statement = connection.createStatement()) { + connection.setAutoCommit(false); + statement.execute(s); + connection.commit(); + } catch (SQLException e) { + throw new RuntimeWrappingException(e); + } + } + + public ArrayList> queryAquavisTables(Path script, + String scenarioId, String scenarioIdToken) { + ArrayList> table = null; + String s = readFile(script).replace(scenarioIdToken, scenarioId); + try (Connection connection = getConnection(); + Statement statement = connection.createStatement()) { + System.out.println("query="+s); + table = extract(statement.executeQuery(s)); + connection.close(); + } catch (SQLException e) { + throw new RuntimeWrappingException(e); + } + return table; + } + + private ArrayList> extract(ResultSet resultSet) + throws SQLException { + ArrayList> table; + int columnCount = resultSet.getMetaData().getColumnCount(); + + if (resultSet.getType() == ResultSet.TYPE_FORWARD_ONLY) + table = new ArrayList>(); + else { + resultSet.last(); + table = new ArrayList>(resultSet.getRow()); + resultSet.beforeFirst(); + } + + for (ArrayList row; resultSet.next(); table.add(row)) { + row = new ArrayList(columnCount); + + for (int c = 1; c <= columnCount; ++c) + row.add(resultSet.getString(c).intern()); + } + return table; + } + + private String readFile(Path file) { + try (FileReader reader = new FileReader(file.toFile())) { + StringBuilder sb = new StringBuilder(); + char[] buffer = new char[8192]; + int readCount; + while ((readCount = reader.read(buffer, 0, buffer.length)) > 0) + sb.append(buffer, 0, readCount); + return sb.toString(); + } catch (IOException e) { + throw new RuntimeIOException(e); + } + } + + private String formConnectionUrl(String ipAddress, int port, + String databaseName, String username, String password, + String instance) { + String url = "jdbc:jtds:sqlserver://" + ipAddress + ":" + port + "/" + + databaseName; + if (username != null) + url += ";user=" + username + ";" + password; // not + // super + // secure, + // btu + // ok + // for + // now + // - + // probably + // will + // use + // SSO + // normally + if (instance != null) + url += ";instance=" + instance; + return url; + } + + private Connection getConnection() throws SQLException { + try { + Class.forName("net.sourceforge.jtds.jdbc.Driver"); + } catch (ClassNotFoundException e) { + throw new RuntimeWrappingException(e); + } + return DriverManager.getConnection(connectionUrl); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java new file mode 100644 index 0000000..0ec7a68 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java @@ -0,0 +1,146 @@ +package org.sandag.abm.reporting.emfac2011; + +import java.util.Collections; +import java.util.EnumSet; +import java.util.Set; + +/** + * The {@code EmfacVehicleType} enum represents the EMFAC2011 tNCVehicle types. + * + * @author crf Started 2/8/12 8:54 AM + */ +public enum Emfac2011VehicleType +{ + OTHER_BUSES_DSL("All Other Buses - DSL", "OBUS - DSL", false), LDA_DSL("LDA - DSL", + "LDA - DSL", true), LDA_GAS("LDA - GAS", "LDA - GAS", true), LDT1_DSL("LDT1 - DSL", + "LDT1 - DSL", true), LDT1_GAS("LDT1 - GAS", "LDT1 - GAS", true), LDT2_DSL("LDT2 - DSL", + "LDT2 - DSL", true), LDT2_GAS("LDT2 - GAS", "LDT2 - GAS", true), LHD1_DSL("LHD1 - DSL", + "LHDT1 - DSL", true), LHE1_GAS("LHD1 - GAS", "LHDT1 - GAS", true), LHD2_DSL( + "LHD2 - DSL", "LHDT2 - DSL", true), LHD2_GAS("LHD2 - GAS", "LHDT2 - GAS", true), MCY_GAS( + "MCY - GAS", "MCY - GAS", true), MDV_DSL("MDV - DSL", "MDV - DSL", true), MDV_GAS( + "MDV - GAS", "MDV - GAS", true), MH_DSL("MH - DSL", "MH - DSL", false), MH_GAS( + "MH - GAS", "MH - GAS", false), MOTOR_COACH_DSL("Motor Coach - DSL", "OBUS - DSL", + false), OBUS_GAS("OBUS - GAS", "OBUS - GAS", false), PTO_DSL("PTO - DSL", "HHDT - DSL", + false), SBUS_DSL("SBUS - DSL", "SBUS - DSL", false), SBUS_GAS("SBUS - GAS", + "SBUS - GAS", false), T6_AG_DSL("T6 Ag - DSL", "MHDT - DSL", false), T6_CAIRP_HEAVY_DSL( + "T6 CAIRP heavy - DSL", "MHDT - DSL", false), T6_CAIRP_SMALL_DSL( + "T6 CAIRP small - DSL", "MHDT - DSL", false), T6_INSTATE_CONSTRUCTION_HEAVY_DSL( + "T6 instate construction heavy - DSL", "MHDT - DSL", false), T6_INSTATE_CONSTRUCTION_SMALL_DSL( + "T6 instate construction small - DSL", "MHDT - DSL", false), T6_INSTATE_HEAVY_DSL( + "T6 instate heavy - DSL", "MHDT - DSL", false), T6_INSTATE_SMALL_DSL( + "T6 instate small - DSL", "MHDT - DSL", false), T6_OOS_HEAVY_DSL("T6 OOS heavy - DSL", + "MHDT - DSL", false), T6_OOS_SMALL_DSL("T6 OOS small - DSL", "MHDT - DSL", false), T6_PUBLIC_DSL( + "T6 public - DSL", "MHDT - DSL", false), T6_UTILITY_DSL("T6 utility - DSL", + "MHDT - DSL", false), T6TS_GAS("T6TS - GAS", "MHDT - GAS", false), T7_AG_DSL( + "T7 Ag - DSL", "HHDT - DSL", false), T7_CAIRP_DSL("T7 CAIRP - DSL", "HHDT - DSL", false), T7_CAIRP_CONSTRUCTION_DSL( + "T7 CAIRP construction - DSL", "HHDT - DSL", false), T7_NNOOS_DSL("T7 NNOOS - DSL", + "HHDT - DSL", false), T7_NOOS_DSL("T7 NOOS - DSL", "HHDT - DSL", false), T7_OTHER_PORT_DSL( + "T7 other port - DSL", "HHDT - DSL", false), T7_POAK_DSL("T7 POAK - DSL", "HHDT - DSL", + false), T7_POLA_DSL("T7 POLA - DSL", "HHDT - DSL", false), T7_PUBLIC_DSL( + "T7 public - DSL", "HHDT - DSL", false), T7_SINGLE_DSL("T7 Single - DSL", "HHDT - DSL", + false), T7_SINGLE_CONSTRUCTION_DSL("T7 single construction - DSL", "HHDT - DSL", false), T7_SWCV_DSL( + "T7 SWCV - DSL", "HHDT - DSL", false), T7_TRACTOR_DSL("T7 tractor - DSL", "HHDT - DSL", + false), T7_TRACTOR_CONSTRUCTION_DSL("T7 tractor construction - DSL", "HHDT - DSL", + false), T7_UTILITY_DSL("T7 utility - DSL", "HHDT - DSL", false), T7IS_GAS("T7IS - GAS", + "HHDT - GAS", false), UBUS_DSL("UBUS - DSL", "UBUS - DSL", false), UBUS_GAS( + "UBUS - GAS", "UBUS - GAS", false); + + private final String name; + private final String emfac2007Name; + private final boolean isMutableType; + + private Emfac2011VehicleType(String name, String emfac2007Name, boolean isMutableType) + { + this.name = name; + this.emfac2007Name = emfac2007Name; + this.isMutableType = isMutableType; + } + + /** + * Get the name of the tNCVehicle type. This is the name used by the EMFAC2011 + * model. + * + * @return the name of the tNCVehicle type. + */ + public String getName() + { + return name; + } + + /** + * Get this tNCVehicle type's equivalent EMFAC2007 tNCVehicle type name. + * + * @return the EMFAC2007 tNCVehicle type corresponding to this tNCVehicle type. + */ + public String getEmfac2007Name() + { + return emfac2007Name; + } + + /** + * Determine if this tNCVehicle type is mutable. If a tNCVehicle type is mutable, + * then its EMFAC2011 SG inputs may be adjusted to account for travel demand + * model results. + * + * @return {@code true} if this tNCVehicle type is mutable, {@code false} if + * not. + */ + public boolean isMutableType() + { + return mutableTypes.contains(this); + } + + private static Set mutableTypes; + + static + { + Set set = EnumSet.noneOf(Emfac2011VehicleType.class); + for (Emfac2011VehicleType type : Emfac2011VehicleType.values()) + if (type.isMutableType) set.add(type); + setMutableTypes(set); + } + + /** + * Set the mutable tNCVehicle types. This need only be called if the default + * mutable types are unsatisfactory for the particular application; however, + * if it needs to be called, then it should be before any of the + * EMFAC2011/Aquavis processing commences. + * + * @param mutableTypes + * The set of mutable tNCVehicle types for processing. + */ + public static void setMutableTypes(Set mutableTypes) + { + Emfac2011VehicleType.mutableTypes = Collections.unmodifiableSet(mutableTypes); + } + + /** + * Get the tNCVehicle type corresponding to the given (EMFAC2011) name. + * + * @param name + * The tNCVehicle type name used in the EMFAC2011 tNCVehicle type. + * + * @return the tNCVehicle type corresponding to {@code name}. + * + * @throws IllegalArgumentException + * if {@code name} does not correspond to any tNCVehicle type. + */ + public static Emfac2011VehicleType getVehicleType(String name) + { + for (Emfac2011VehicleType type : values()) + if (type.getName().equals(name)) return type; + throw new IllegalArgumentException("No EMFAC tNCVehicle type corresponding to: " + name); + } + + /** + * Get the set of EMFAC2011 tNCVehicle types which are mutable. If a tNCVehicle + * type is mutable, then its EMFAC2011 SG inputs may be adjusted to account + * for travel demand model results. + * + * @return a set holding the mutable EMFAC2011 tNCVehicle types. + */ + public static Set getMutableVehicleTypes() + { + return mutableTypes; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java new file mode 100644 index 0000000..a19b83d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java @@ -0,0 +1,174 @@ +package org.sandag.abm.reporting.emfac2011; + +import java.awt.GridBagConstraints; +import java.awt.GridBagLayout; +import java.awt.event.ActionEvent; +import java.awt.event.ActionListener; +import java.awt.event.WindowAdapter; +import java.awt.event.WindowEvent; +import java.nio.file.Path; +import javax.swing.JButton; +import javax.swing.JFrame; +import javax.swing.JLayeredPane; +import javax.swing.JPanel; +import javax.swing.JScrollPane; +import javax.swing.JTextArea; +import com.pb.sawdust.util.exceptions.RuntimeInterruptedException; + +/** + * The {@code HelpDialog} ... + * + * @author crf Started 2/9/12 3:57 PM + */ +class RunEmfacDialog + extends JPanel + implements ActionListener +{ + private static final long serialVersionUID = -3645537208340049132L; + private final JButton button; + private final JLayeredPane pane; + private final Emfac2011Runner emfac2011Runner; + + public RunEmfacDialog(Path outputPath, Emfac2011Runner emfac2011Runner) + { + super(new GridBagLayout()); + + StringBuilder sb = new StringBuilder(); + sb.append("\n ") + .append("Almost there! Press the button to start EMFAC2011, then follow these directions.") + .append("\n\n"); + sb.append(" ") + .append("EMFAC2011 should have booted up. If not, check that the installation directory is defined correctly in the properties file.") + .append("\n\n"); + sb.append(" ").append("The following steps will take you through running the program.") + .append("\n\n"); + sb.append(" ") + .append("1) In the \"Regional Scenarios\" box, hit the \"Load Regional Scenarios (External Files)\" button.") + .append("\n\n"); + sb.append(" ").append("2) Browse to and select: ").append(outputPath).append("\n\n"); + sb.append(" ") + .append("3) The EMFAC2011-SG-Scenario Builder window should appear. Press the \"Save and Continue\" button.") + .append("\n\n"); + sb.append(" ").append("4) A message box will appear. Click \"Yes\"").append("\n\n"); + sb.append(" ") + .append("5) In the EMFAC2011-SG model window, hit the \"Verify Speed Data Quality\" button.") + .append("\n\n"); + sb.append(" ") + .append("6) If there are no errors, hit the \"Continue\" button in the \"Verify Speed Inputs\" window.") + .append("\n\n"); + sb.append(" ") + .append("7) In the EMFAC2011-SG model window, hit the \"Save Scenarios\" button.") + .append("\n\n"); + sb.append(" ") + .append("8) Select the same file that we loaded in step (2). Say \"Yes\" to the question about replacing the file.") + .append("\n\n"); + sb.append(" ").append("9) EMFAC2011 will tell you it saved the input file. Click \"OK\"") + .append("\n\n"); + sb.append(" ") + .append("10) In the EMFAC2011-SG model window, hit the \"Execute Model\" button.") + .append("\n\n"); + sb.append(" ") + .append("11) In the EMFAC2011-SG-Model Execution Options window do the following") + .append("\n"); + sb.append(" ") + .append("\ta) In the \"Input Parameters\" box, the \"Export Default Input Parameters\" check box should NOT be checked.") + .append("\n"); + sb.append(" ") + .append("\tb) In the \"Model Outputs\" box, choose \"XLS\" as the output format, and check the \"Create Additional Summary Outputs\" \n\t checkbox. Leave the \"Create Separate Output Files for Each Regional Scenario\" checkbox unchecked.") + .append("\n"); + sb.append(" ").append("\tc) Hit the \"Start\" button.").append("\n\n"); + sb.append(" ") + .append("12) The EMFAC2011 model should run, and then pop up a dialog box saying it finished. Click \"OK\"") + .append("\n\n"); + sb.append(" ").append("13) Click the \"Exit EMFAC2011-SG\" button.").append("\n\n"); + sb.append(" ").append("14) All done! Close this window when you are finished."); + + JTextArea textArea = new JTextArea(40, 80); + textArea.setEditable(false); + textArea.setText(sb.toString()); + textArea.setCaretPosition(0); + textArea.setLineWrap(true); + textArea.setWrapStyleWord(true); + JScrollPane scrollPane = new JScrollPane(textArea); + GridBagConstraints c = new GridBagConstraints(); + c.gridwidth = GridBagConstraints.REMAINDER; + c.fill = GridBagConstraints.HORIZONTAL; + c.fill = GridBagConstraints.BOTH; + c.weightx = 1.0; + c.weighty = 1.0; + + pane = new JLayeredPane(); + scrollPane.setSize(850, 675); + button = new JButton("Start EMFAC2011"); + button.addActionListener(this); + button.setLocation(500, 15); + button.setSize(150, 20); + pane.add(button, 0, -1); + pane.add(scrollPane); + add(pane); + add(pane, c); + + this.emfac2011Runner = emfac2011Runner; + } + + public static void createAndShowGUI(Path inputFile, Emfac2011Runner emfac2011Runner) + { + final Object lock = new Object(); + final JFrame frame = new JFrame("Run EMFAC2011"); + frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); + frame.add(new RunEmfacDialog(inputFile, emfac2011Runner)); + frame.setSize(800, 700); + frame.setVisible(true); + + Thread thread = new Thread() + { + @Override + public void run() + { + synchronized (lock) + { + while (frame.isVisible()) + { + try + { + lock.wait(); + } catch (InterruptedException e) + { + throw new RuntimeInterruptedException(e); + } + } + } + } + }; + thread.start(); + + frame.addWindowListener(new WindowAdapter() + { + @Override + public void windowClosing(WindowEvent arg0) + { + synchronized (lock) + { + frame.setVisible(false); + lock.notify(); + } + } + }); + try + { + thread.join(); + } catch (InterruptedException e) + { + throw new RuntimeInterruptedException(e); + } + + } + + @Override + public void actionPerformed(ActionEvent e) + { + pane.remove(button); + pane.repaint(); + emfac2011Runner.runEmfac2011Program(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java new file mode 100644 index 0000000..4cdb52a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java @@ -0,0 +1,11 @@ +package org.sandag.abm.reporting.emfac2011; + +/** + * The {@code VehicleType} ... + * + * @author crf Started 12/7/12 3:28 PM + */ +public enum SandagAutoModes +{ + SOV_GP, SOV_PAY, SR2_GP, SR2_HOV, SR2_PAY, SR3_GP, SR3_HOV, SR3_PAY, LHDN, MHDN, HHDN, LHDT, MHDT, HHDT, UBUS +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java new file mode 100644 index 0000000..2eb422a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java @@ -0,0 +1,42 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.specialevent; + +import java.io.Serializable; +import org.sandag.abm.application.SandagModelStructure; + +/** + * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects + * + * @author Joel Freedman + */ +public class SpecialEventDmuFactory + implements SpecialEventDmuFactoryIf, Serializable +{ + + private SandagModelStructure sandagModelStructure; + + public SpecialEventDmuFactory(SandagModelStructure modelStructure) + { + this.sandagModelStructure = modelStructure; + } + + public SpecialEventTripModeChoiceDMU getSpecialEventTripModeChoiceDMU() + { + return new SpecialEventTripModeChoiceDMU(sandagModelStructure, null); + } + + public SpecialEventOriginChoiceDMU getSpecialEventOriginChoiceDMU() + { + return new SpecialEventOriginChoiceDMU(sandagModelStructure); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java new file mode 100644 index 0000000..33e5633 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java @@ -0,0 +1,13 @@ +package org.sandag.abm.specialevent; + +/** + * A DMU factory interface + */ +public interface SpecialEventDmuFactoryIf +{ + + SpecialEventTripModeChoiceDMU getSpecialEventTripModeChoiceDMU(); + + SpecialEventOriginChoiceDMU getSpecialEventOriginChoiceDMU(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java new file mode 100644 index 0000000..ddc0782 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java @@ -0,0 +1,347 @@ +package org.sandag.abm.specialevent; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +/** + * Trips for special events. + * + * This programs models trips to special events using the Event Model framework. + * The Event Model framework generates and distributes trips. These trips are + * put through a mode choice model. User benefits are optionally calculated and + * written to a SUMMIT formatted file. + * + */ +public final class SpecialEventModel +{ + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + + private MatrixDataServerRmi ms; + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + private HashMap rbMap; + private McLogsumsCalculator logsumsCalculator; + private AutoTazSkimsCalculator tazDistanceCalculator; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private boolean seek; + private int traceId; + + private TableDataSet eventData; + private double sampleRate = 1; + + /** + * Default Constructor. + */ + private SpecialEventModel(HashMap rbMap) + { + this.rbMap = rbMap; + mgraManager = MgraDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + + // read the event data + String eventFile = Util.getStringValueFromPropertyMap(rbMap, "specialEvent.event.file"); + eventFile = directory + eventFile; + eventData = readFile(eventFile); + + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "specialEvent.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "specialEvent.trace")); + + } + + /** + * Read the file and return the TableDataSet. + * + * @param fileName + * @return data + */ + private TableDataSet readFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet data; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + data = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return data; + } + + /** + * @return the sampleRate + */ + public double getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(double sampleRate) + { + this.sampleRate = sampleRate; + } + + /** + * Run the Event Model. + * + * The Event Model is run for each mgra specified in the events.file + */ + public void runModel() + { + + SandagModelStructure modelStructure = new SandagModelStructure(); + + SpecialEventDmuFactoryIf dmuFactory = new SpecialEventDmuFactory(modelStructure); + + SpecialEventTourManager tourManager = new SpecialEventTourManager(rbMap, eventData); + + tourManager.generateTours(); + SpecialEventTour[] tours = tourManager.getTours(); + + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + logsumsCalculator = new McLogsumsCalculator(); + logsumsCalculator.setupSkimCalculators(rbMap); + logsumsCalculator.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + SpecialEventOriginChoiceModel originChoiceModel = new SpecialEventOriginChoiceModel(rbMap, + dmuFactory, eventData); + originChoiceModel.calculateSizeTerms(dmuFactory); + originChoiceModel.calculateTazProbabilities(dmuFactory); + + SpecialEventTripModeChoiceModel tripModeChoiceModel = new SpecialEventTripModeChoiceModel( + rbMap, modelStructure, dmuFactory, logsumsCalculator, eventData); + + // Run models for array of tours + for (int i = 0; i < tours.length; ++i) + { + + SpecialEventTour tour = tours[i]; + + // Wu added for sampling tours + double rand = tour.getRandom(); + if (rand > sampleRate) continue; + + if (i < 10 || i % 1000 == 0) logger.info("Processing tour " + (i + 1)); + + if (seek && tour.getID() != traceId) continue; + + if (tour.getID() == traceId) tour.setDebugChoiceModels(true); + originChoiceModel.chooseOrigin(tour); + // generate trips and choose mode for them + SpecialEventTrip[] trips = new SpecialEventTrip[2]; + int tripNumber = 0; + + // generate an outbound trip from the tour origin to the destination + // and choose a mode + trips[tripNumber] = new SpecialEventTrip(tour, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + // generate an inbound trip from the tour destination to the origin + // and choose a mode + trips[tripNumber] = new SpecialEventTrip(tour, false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + // set the trips in the tour object + tour.setTrips(trips); + + } + + tourManager.writeOutputFile(rbMap); + + logger.info("Special Event Model successfully completed!"); + + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * Run Special Events Model. + * + * The Special Events Model generates, distributes, and chooses modes for + * attendees of sporting events and similar activities. + */ + public static void main(String[] args) + { + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Special Event Model")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + SpecialEventModel specialEventModel = new SpecialEventModel(pMap); + + //Wu added for sampling special event tours based on sample rate + float sampleRate = 1.0f; + int iteration = 1; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + logger.info("Special Event Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + specialEventModel.setSampleRate(sampleRate); + + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = specialEventModel.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + specialEventModel.ms = matrixServer; + } else + { + specialEventModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + specialEventModel.ms.testRemote("SpecialEventModel"); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(specialEventModel.ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + specialEventModel.runModel(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java new file mode 100644 index 0000000..0f418bc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java @@ -0,0 +1,205 @@ +package org.sandag.abm.specialevent; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class SpecialEventOriginChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("specialEventModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected float tourDepartPeriod; + protected float tourArrivePeriod; + protected int purpose; + protected double[][] sizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgras) + + protected double nmWalkTimeOut; + protected double nmWalkTimeIn; + protected double nmBikeTimeOut; + protected double nmBikeTimeIn; + protected double lsWgtAvgCostM; + protected double lsWgtAvgCostD; + protected double lsWgtAvgCostH; + + public SpecialEventOriginChoiceDMU(SandagModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + /** + * @return the sizeTerm. The size term is the size of the origin taz. + */ + public double getSizeTerm(int alt) + { + return sizeTerms[purpose][alt]; + } + + /** + * @param sizeTerms + * the sizeTerms to set. The size term is the array of origin taz + * sizes. + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the purpose + */ + public int getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(int purpose) + { + this.purpose = purpose; + } + + public float getTimeOutbound() + { + return tourDepartPeriod; + } + + public float getTimeInbound() + { + return tourArrivePeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(float tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(float tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getSizeTerm", 2); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 2: + returnValue = getSizeTerm(arrayIndex); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java new file mode 100644 index 0000000..690775a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java @@ -0,0 +1,384 @@ +package org.sandag.abm.specialevent; + +import java.io.File; +import java.util.ArrayList; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class SpecialEventOriginChoiceModel +{ + + private double[][] mgraSizeTerms; // by + // purpose, + // MGRA + private double[][] tazSizeTerms; // by + // purpose, + // TAZ + private double[][][] mgraProbabilities; // by + // purpose, + // tazNumber, + // mgra + // index + // (sequential, + // 0-based) + private Matrix[] tazProbabilities; // by + // purpose, + // origin + // TAZ, + // destination + // TAZ + + private TableDataSet alternativeData; // the + // alternatives, + // with + // a + // dest + // field + // indicating + // tazNumber + + private transient Logger logger = Logger.getLogger("specialEventModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication destModel; + private UtilityExpressionCalculator sizeTermUEC; + private HashMap rbMap; + private HashMap purposeMap; // string + // is + // purpose, + // int + // is + // alternative + // for + // size + // terms + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of airport model DMUs + */ + public SpecialEventOriginChoiceModel(HashMap rbMap, + SpecialEventDmuFactoryIf dmuFactory, TableDataSet eventData) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String destUecFileName = Util.getStringValueFromPropertyMap(rbMap, + "specialEvent.dc.uec.file"); + destUecFileName = uecFileDirectory + destUecFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "specialEvent.dc.data.page")); + int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "specialEvent.dc.size.page")); + int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "specialEvent.dc.model.page")); + + // read the model pages from the property file, create one choice model + // for each + // get page from property file + + SpecialEventOriginChoiceDMU dcDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); + + // create a ChoiceModelApplication object for the filename, model page + // and data page. + destModel = new ChoiceModelApplication(destUecFileName, modelPage, dataPage, rbMap, + (VariableTable) dcDmu); + + // get the alternative data from the first segment + UtilityExpressionCalculator uec = destModel.getUEC(); + alternativeData = uec.getAlternativeData(); + + // create a UEC to solve size terms for each MGRA + SpecialEventOriginChoiceDMU dmu = dmuFactory.getSpecialEventOriginChoiceDMU(); + sizeTermUEC = new UtilityExpressionCalculator(new File(destUecFileName), sizePage, + dataPage, rbMap, dmu); + + } + + /** + * Calculate size terms + */ + public void calculateSizeTerms(SpecialEventDmuFactoryIf dmuFactory) + { + + logger.info("Calculating Special Event Origin Choice Model Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int maxTaz = tazManager.getMaxTaz(); + int purposes = sizeTermUEC.getNumberOfAlternatives(); + + // set up the map of string purpose to integer purpose + String[] altNames = sizeTermUEC.getAlternativeNames(); + purposeMap = new HashMap(); + for (int i = 0; i < altNames.length; ++i) + { + purposeMap.put(altNames[i], i); + } + + mgraSizeTerms = new double[purposes][maxMgra + 1]; + tazSizeTerms = new double[purposes][maxTaz + 1]; + IndexValues iv = new IndexValues(); + SpecialEventOriginChoiceDMU aDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = sizeTermUEC.solve(iv, aDmu, null); + + // store the size terms + for (int purpose = 0; purpose < purposes; ++purpose) + { + + mgraSizeTerms[purpose][mgra] = utilities[purpose]; + tazSizeTerms[purpose][taz] += utilities[purpose]; + } + + } + + // now calculate probability of selecting each MGRA within each TAZ for + // SOA + mgraProbabilities = new double[purposes][maxTaz + 1][]; + int[] tazs = tazManager.getTazs(); + + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazs.length; ++taz) + { + int tazNumber = tazs[taz]; + int[] mgraArray = tazManager.getMgraArray(tazNumber); + + // initialize the vector of mgras for this purpose-taz + mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; + + // now calculate the cumulative probability distribution + double lastProb = 0.0; + for (int mgra = 0; mgra < mgraArray.length; ++mgra) + { + + int mgraNumber = mgraArray[mgra]; + if (tazSizeTerms[purpose][tazNumber] > 0.0) + mgraProbabilities[purpose][tazNumber][mgra] = lastProb + + mgraSizeTerms[purpose][mgraNumber] + / tazSizeTerms[purpose][tazNumber]; + lastProb = mgraProbabilities[purpose][tazNumber][mgra]; + } + if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) + logger.info("Error: purpose " + purpose + " taz " + tazNumber + + " cum prob adds up to " + lastProb); + } + + } + + // calculate logged size terms for mgra and taz vectors to be used in + // dmu + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) + if (tazSizeTerms[purpose][taz] > 0.0) + tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); + + for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) + if (mgraSizeTerms[purpose][mgra] > 0.0) + mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); + + } + logger.info("Finished Calculating Special Event Tour Origin Choice Model Size Terms"); + } + + /** + * Calculate taz probabilities. This method initializes and calculates the + * tazProbabilities array. + */ + public void calculateTazProbabilities(SpecialEventDmuFactoryIf dmuFactory) + { + + if (tazSizeTerms == null) + { + logger.error("Error: attemping to execute SpecialEventTourOriginChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); + throw new RuntimeException(); + } + + logger.info("Calculating Special Event Model TAZ Probabilities Arrays"); + + // initialize taz probabilities array + int purposes = tazSizeTerms.length; + + // initialize the arrays + tazProbabilities = new Matrix[purposes]; + + // iterate through the alternatives in the alternatives file and set the + // size term for each alternative + UtilityExpressionCalculator modelUEC = destModel.getUEC(); + TableDataSet altData = modelUEC.getAlternativeData(); + + SpecialEventOriginChoiceDMU dcDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); + dcDmu.setSizeTerms(tazSizeTerms); + + // iterate through purposes + for (int purpose = 0; purpose < purposes; ++purpose) + { + + tazProbabilities[purpose] = new Matrix("Prob_Matrix", "Probability Matrix", + altData.getRowCount() + 1, altData.getRowCount() + 1); + int[] tazs = altData.getColumnAsInt("dest"); + tazProbabilities[purpose].setExternalNumbersZeroBased(tazs); + + // iterate through destination zones, solve the UEC for all origins + // and store the results in the matrix + for (int taz = 0; taz < tazs.length; ++taz) + { + + int destinationTaz = (int) tazs[taz]; + + // set origin taz in dmu (destination set in UEC by alternative) + dcDmu.setDmuIndexValues(0, 0, 0, destinationTaz, false); + + dcDmu.setPurpose(purpose); + + // Calculate utilities & probabilities + destModel.computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); + + // Store probabilities (by purpose) + double[] probabilities = destModel.getCumulativeProbabilities(); + + for (int i = 0; i < probabilities.length; ++i) + { + + double cumProb = probabilities[i]; + int originTaz = (int) altData.getValueAt(i + 1, "dest"); + tazProbabilities[purpose] + .setValueAt(originTaz, destinationTaz, (float) cumProb); + } + } + } + logger.info("Finished Calculating Special Event Model TAZ Probabilities Arrays"); + } + + /** + * Choose an MGRA + * + * @param eventType + * Event type corresponding to size term + * @param destinationMgra + * MGRA of destination + * @param random + * Random number + * @return The chosen MGRA number + */ + public int chooseMGRA(String eventType, int destinationMgra, double random, boolean debug) + { + + int destinationTaz = mgraManager.getTaz(destinationMgra); + int purpose = purposeMap.get(eventType); + + if (debug) + { + logger.info("Random number " + random); + logger.info("Purpose " + purpose); + logger.info("Destination TAZ " + destinationTaz); + + } + + // first find a TAZ and station + Matrix tazCumProb = tazProbabilities[purpose]; + double altProb = 0; + double cumProb = 0; + int originTaz = -1; + for (int i = 0; i < tazCumProb.getColumnCount(); ++i) + { + originTaz = (int) tazCumProb.getExternalColumnNumber(i); + if (tazCumProb.getValueAt(originTaz, destinationTaz) > random) + { // the + // probabilities + // are + // stored + // column-wise + if (i != 0) + { + cumProb = tazCumProb.getValueAt(originTaz, + tazCumProb.getExternalColumnNumber(i - 1)); + altProb = tazCumProb.getValueAt(originTaz, destinationTaz) + - tazCumProb.getValueAt(originTaz, + tazCumProb.getExternalColumnNumber(i - 1)); + } else + { + altProb = tazCumProb.getValueAt(originTaz, destinationTaz); + } + break; + } + } + + // get the taz number of the alternative, and an array of mgras in that + // taz + int[] mgraArray = tazManager.getMgraArray(originTaz); + + // now find an MGRA in the taz corresponding to the random number drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives probability + int mgraNumber = 0; + + if (debug) + { + logger.info("Chosen origin TAZ " + originTaz); + } + + double[] mgraCumProb = mgraProbabilities[purpose][originTaz]; + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > random) + { + mgraNumber = mgraArray[i]; + } + } + if (debug) logger.info("Chose origin MGRA " + mgraNumber); + // return the chosen MGRA number + return mgraNumber; + } + + /** + * Choose origin MGRAs for a special event tour. + * + * @param tour + * A Special Event tour + */ + public void chooseOrigin(SpecialEventTour tour) + { + + String eventType = tour.getEventType(); + double random = tour.getRandom(); + int destinationMgra = tour.getDestinationMGRA(); + int mgra = chooseMGRA(eventType, destinationMgra, random, tour.getDebugChoiceModels()); + tour.setOriginMGRA(mgra); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java new file mode 100644 index 0000000..ba05b9b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java @@ -0,0 +1,291 @@ +package org.sandag.abm.specialevent; + +import java.io.Serializable; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Household; +import com.pb.common.math.MersenneTwister; + +public class SpecialEventTour + implements Serializable +{ + + private MersenneTwister random; + private int ID; + + private byte eventNumber; + private String eventType; + // following variables determined via simulation + private int income; + private int partySize; + + private SpecialEventTrip[] trips; + + private int departTime; + private int arriveTime; + + private boolean debugChoiceModels; + + // following variables chosen via choice models + private int originMGRA; + private int destinationMGRA; + private byte tourMode; + + private float valueOfTime; + + /** + * Public constructor. + * + * @param seed + * A seed for the random number generator. + */ + public SpecialEventTour(long seed) + { + + random = new MersenneTwister(seed); + } + + /** + * @return the iD + */ + public int getID() + { + return ID; + } + + /** + * @param iD + * the iD to set + */ + public void setID(int iD) + { + ID = iD; + } + + /** + * @return the eventNumber + */ + public byte getEventNumber() + { + return eventNumber; + } + + /** + * @param eventNumber + * the eventNumber to set + */ + public void setEventNumber(byte eventNumber) + { + this.eventNumber = eventNumber; + } + + /** + * @return the departTime + */ + public int getDepartTime() + { + return departTime; + } + + /** + * @param departTime + * the departTime to set + */ + public void setDepartTime(int departTime) + { + this.departTime = departTime; + } + + public SpecialEventTrip[] getTrips() + { + return trips; + } + + /** + * @return the eventType + */ + public String getEventType() + { + return eventType; + } + + /** + * @param eventType + * the eventType to set + */ + public void setEventType(String eventType) + { + this.eventType = eventType; + } + + /** + * @return the income + */ + public int getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(int income) + { + this.income = income; + } + + /** + * @return the partySize + */ + public int getPartySize() + { + return partySize; + } + + /** + * @param partySize + * the partySize to set + */ + public void setPartySize(int partySize) + { + this.partySize = partySize; + } + + public void setTrips(SpecialEventTrip[] trips) + { + this.trips = trips; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() + { + return originMGRA; + } + + /** + * @param originMGRA + * the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) + { + this.originMGRA = originMGRA; + } + + /** + * @return the tour mode + */ + public byte getTourMode() + { + return tourMode; + } + + /** + * @param mode + * the tour mode to set + */ + public void setTourMode(byte mode) + { + this.tourMode = mode; + } + + /** + * Get a random number from the parties random class. + * + * @return A random number. + */ + public double getRandom() + { + return random.nextDouble(); + } + + /** + * @return the debugChoiceModels + */ + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + /** + * @param debugChoiceModels + * the debugChoiceModels to set + */ + public void setDebugChoiceModels(boolean debugChoiceModels) + { + this.debugChoiceModels = debugChoiceModels; + } + + + + /** + * Get the number of outbound stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberOutboundStops() + { + return 0; + + } + + /** + * Get the number of return stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberInboundStops() + { + return 0; + + } + + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() + { + return destinationMGRA; + } + + /** + * @param destinationMGRA + * the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) + { + this.destinationMGRA = destinationMGRA; + } + + public void setArriveTime(int arriveTime) + { + this.arriveTime = arriveTime; + } + + public int getArriveTime() + { + return arriveTime; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public void logTourObject(Logger logger, int totalChars) + { + + Household.logHelper(logger, "tourId: ", ID, totalChars); + Household.logHelper(logger, "Event type: ", eventType, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); + Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); + Household.logHelper(logger, "tourMode: ", tourMode, totalChars); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java new file mode 100644 index 0000000..9e38021 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java @@ -0,0 +1,342 @@ +package org.sandag.abm.specialevent; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; + +public class SpecialEventTourManager +{ + + private TableDataSet eventData; + private TableDataSet incomeData; + private TableDataSet partySizeData; + private HashMap rbMap; + private static Logger logger = Logger.getLogger("specialEventModel"); + + private SpecialEventTour[] tours; + private MersenneTwister randomGenerator; + private SandagModelStructure sandagStructure; + private boolean saveUtilsAndProbs; + + /** + * Default Constructor. + */ + public SpecialEventTourManager(HashMap rbMap, TableDataSet eventData) + { + this.rbMap = rbMap; + randomGenerator = new MersenneTwister(10001); + saveUtilsAndProbs = Util.getBooleanValueFromPropertyMap(rbMap, + "specialEvent.saveUtilsAndProbs"); + this.eventData = eventData; + sandagStructure = new SandagModelStructure(); + + } + + /** + * Generate special event tours. + */ + public void generateTours() + { + + // read the party size data + String partySizeFile = Util.getStringValueFromPropertyMap(rbMap, + "specialEvent.partySize.file"); + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + partySizeFile = directory + partySizeFile; + partySizeData = readFile(partySizeFile); + + // read the income data + String incomeFile = Util.getStringValueFromPropertyMap(rbMap, "specialEvent.income.file"); + incomeFile = directory + incomeFile; + incomeData = readFile(incomeFile); + + ArrayList eventTourList = new ArrayList(); + + int eventTours = 0; + for (int i = 1; i <= eventData.getRowCount(); ++i) + { + + int eventMgra = (int) eventData.getValueAt(i, "MGRA"); + int attendance = (int) eventData.getValueAt(i, "Attendance"); + String eventType = eventData.getStringValueAt(i, "EventType"); + int startPeriod = (int) eventData.getValueAt(i, "StartPeriod"); + int endPeriod = (int) eventData.getValueAt(i, "EndPeriod"); + + // generate tours for the event + for (int j = 0; j < attendance; ++j) + { + + ++eventTours; + + long randomSeed = getRandomSeed(eventTours); + SpecialEventTour tour = new SpecialEventTour(randomSeed); + tour.setID(eventTours); + tour.setEventNumber((byte) i); + tour.setDestinationMGRA(eventMgra); + tour.setDepartTime(startPeriod); + tour.setArriveTime(endPeriod); + tour.setEventType(eventType); + + // choose income and party size for the tour + int income = chooseIncome(randomGenerator.nextDouble(), eventType); + int partySize = choosePartySize(randomGenerator.nextDouble(), eventType); + + tour.setIncome(income); + tour.setPartySize(partySize); + + eventTourList.add(tour); + } + } + + // convert the ArrayList to an array + tours = new SpecialEventTour[eventTourList.size()]; + for (int i = 0; i < tours.length; ++i) + tours[i] = eventTourList.get(i); + + } + + /** + * Simulate income from the income data table. + * + * @param random + * a uniformly-distributed random number. + * @param eventType + * a string identifying the type of event, which should be a + * column in the income data table + * @return income chosen + */ + public int chooseIncome(double random, String eventType) + { + + int income = -1; + double cumProb = 0; + for (int i = 1; i <= incomeData.getRowCount(); ++i) + { + cumProb += incomeData.getValueAt(i, eventType); + if (random < cumProb) + { + income = (int) incomeData.getValueAt(i, "Income"); + break; + } + } + return income; + } + + /** + * Simulate party size from the party size data table. + * + * @param random + * a uniformly-distributed random number. + * @param eventType + * a string identifying the type of event, which should be a + * column in the party size data table + * @return party size chosen + */ + public int choosePartySize(double random, String eventType) + { + + int partySize = -1; + double cumProb = 0; + for (int i = 1; i <= partySizeData.getRowCount(); ++i) + { + cumProb += partySizeData.getValueAt(i, eventType); + if (random < cumProb) + { + partySize = (int) partySizeData.getValueAt(i, "PartySize"); + break; + } + } + return partySize; + } + + /** + * Read the file and return the TableDataSet. + * + * @param fileName + * @return data + */ + private TableDataSet readFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet data; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + data = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return data; + } + + /** + * Create a text file and write all records to the file. + * + */ + public void writeOutputFile(HashMap rbMap) + { + + // Open file and print header + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String tourFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "specialEvent.tour.output.file"); + String tripFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "specialEvent.trip.output.file"); + + logger.info("Writing special event tours to file " + tourFileName); + logger.info("Writing special event trips to file " + tripFileName); + + PrintWriter tourWriter = null; + try + { + tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tourFileName + " for writing\n"); + throw new RuntimeException(); + } + String tourHeaderString = new String( + "id,eventNumber,eventType,income,partySize,departTime,arriveTime,originMGRA,destinationMGRA,tourMode,valueOfTime\n"); + tourWriter.print(tourHeaderString); + + PrintWriter tripWriter = null; + try + { + tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tripFileName + " for writing\n"); + throw new RuntimeException(); + } + String tripHeaderString = new String( + "tourID,tripID,originMGRA,destinationMGRA,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,boardingTap,alightingTap,set,valueOfTime"); + + // Iterate through the array, printing records to the file + for (int i = 0; i < tours.length; ++i) + { + + SpecialEventTour tour = tours[i]; + + SpecialEventTrip[] trips = tours[i].getTrips(); + + if (trips == null) continue; + + writeTour(tour, tourWriter); + + // if this is the first record, and we are saving utils and probs, + // append a line to the trip file header + if (i == 0) + { + + if (saveUtilsAndProbs) + { + float[] utils = trips[0].getModeUtilities(); + String header = ""; + for (int j = 0; j < utils.length; ++j) + header += ",util_" + j; + for (int j = 0; j < utils.length; ++j) + header += ",prob_" + j; + tripWriter.print(tripHeaderString + header + "\n"); + } else tripWriter.print(tripHeaderString + "\n"); + } + for (int j = 0; j < trips.length; ++j) + { + writeTrip(tour, trips[j], j + 1, tripWriter); + } + } + + tourWriter.close(); + tripWriter.close(); + + } + + /** + * Write the tour to the PrintWriter + * + * @param tour + * @param writer + */ + private void writeTour(SpecialEventTour tour, PrintWriter writer) + { + String record = new String(tour.getID() + "," + tour.getEventNumber() + "," + + tour.getEventType() + "," + tour.getIncome() + "," + tour.getPartySize() + "," + + tour.getDepartTime() + "," + tour.getArriveTime() + "," + tour.getOriginMGRA() + + "," + tour.getDestinationMGRA() + "," + tour.getTourMode() + "," + + String.format("%9.2f",tour.getValueOfTime()) + "\n"); + writer.print(record); + + } + + /** + * Write the trip to the PrintWriter + * + * @param tour + * @param trip + * @param tripNumber + * @param writer + */ + private void writeTrip(SpecialEventTour tour, SpecialEventTrip trip, int tripNumber, + PrintWriter writer) + { + + String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginMgra() + + "," + trip.getDestinationMgra() + "," + trip.isInbound() + "," + + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() + + "," + trip.getPeriod() + "," + trip.getTripMode() + "," + + trip.getBoardTap() + "," + trip.getAlightTap() +"," + trip.getSet()+ "," + + String.format("%9.2f",tour.getValueOfTime())); + + + if (saveUtilsAndProbs) + { + + String utilRecord = new String(); + float[] utils = trip.getModeUtilities(); + for (int i = 0; i < utils.length; ++i) + utilRecord += ("," + String.format("%9.5f", utils[i])); + float[] probs = trip.getModeProbabilities(); + for (int i = 0; i < probs.length; ++i) + utilRecord += ("," + String.format("%9.5f", probs[i])); + record = record + utilRecord; + } + writer.print(record + "\n"); + } + /** + * get special event tours. + * + * @return + */ + public SpecialEventTour[] getTours() + { + return tours; + } + + /** + * Calculate and return a random number seed for the tour. + * + * @param eventID + * @return + */ + public long getRandomSeed(int eventID) + { + + long seed = (eventID * 10 + 100001); + return seed; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java new file mode 100644 index 0000000..cc34fa1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java @@ -0,0 +1,292 @@ +package org.sandag.abm.specialevent; + +public class SpecialEventTrip +{ + + private int originMgra; + private int destinationMgra; + private int tripMode; + private byte period; + private boolean inbound; + private boolean firstTrip; + private boolean lastTrip; + private boolean originIsTourDestination; + private boolean destinationIsTourDestination; + + private float[] modeProbabilities; + private float[] modeUtilities; + + private int boardTap; + private int alightTap; + private int set = -1; + + /** + * Default constructor; nothing initialized. + */ + public SpecialEventTrip() + { + + } + + /** + * Create a cross border trip from a tour leg (no stops). + * + * @param tour + * The tour. + * @param outbound + * Outbound direction + */ + public SpecialEventTrip(SpecialEventTour tour, boolean outbound) + { + + initializeFromTour(tour, outbound); + + } + + /** + * Initilize from the tour. + * + * @param tour + * The tour. + * @param outbound + * Outbound direction. + */ + public void initializeFromTour(SpecialEventTour tour, boolean outbound) + { + // Note: mode is unknown + if (outbound) + { + this.originMgra = tour.getOriginMGRA(); + this.destinationMgra = tour.getDestinationMGRA(); + this.period = (byte) tour.getDepartTime(); + this.inbound = false; + this.firstTrip = true; + this.lastTrip = false; + this.originIsTourDestination = false; + this.destinationIsTourDestination = true; + } else + { + this.originMgra = tour.getDestinationMGRA(); + this.destinationMgra = tour.getOriginMGRA(); + this.period = (byte) tour.getArriveTime(); + this.inbound = true; + this.firstTrip = false; + this.lastTrip = true; + this.originIsTourDestination = true; + this.destinationIsTourDestination = false; + } + + } + + /** + * @return the period + */ + public byte getPeriod() + { + return period; + } + + /** + * @param period + * the period to set + */ + public void setPeriod(byte period) + { + this.period = period; + } + + /** + * @return the originMgra + */ + public int getOriginMgra() + { + return originMgra; + } + + /** + * @param originMgra + * the originMgra to set + */ + public void setOriginMgra(int originMgra) + { + this.originMgra = originMgra; + } + + /** + * @return the destinationMgra + */ + public int getDestinationMgra() + { + return destinationMgra; + } + + /** + * @param destinationMgra + * the destinationMgra to set + */ + public void setDestinationMgra(int destinationMgra) + { + this.destinationMgra = destinationMgra; + } + + /** + * @return the tripMode + */ + public int getTripMode() + { + return tripMode; + } + + /** + * @param tripMode + * the tripMode to set + */ + public void setTripMode(int tripMode) + { + this.tripMode = tripMode; + } + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the firstTrip + */ + public boolean isFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(boolean firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public boolean isLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(boolean lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the originIsTourDestination + */ + public boolean isOriginIsTourDestination() + { + return originIsTourDestination; + } + + /** + * @param originIsTourDestination + * the originIsTourDestination to set + */ + public void setOriginIsTourDestination(boolean originIsTourDestination) + { + this.originIsTourDestination = originIsTourDestination; + } + + /** + * @return the destinationIsTourDestination + */ + public boolean isDestinationIsTourDestination() + { + return destinationIsTourDestination; + } + + /** + * @param destinationIsTourDestination + * the destinationIsTourDestination to set + */ + public void setDestinationIsTourDestination(boolean destinationIsTourDestination) + { + this.destinationIsTourDestination = destinationIsTourDestination; + } + + /** + * @return the modeProbabilities + */ + public float[] getModeProbabilities() + { + return modeProbabilities; + } + + /** + * @param modeProbabilities + * the modeProbabilities to set + */ + public void setModeProbabilities(float[] modeProbabilities) + { + this.modeProbabilities = modeProbabilities; + } + + /** + * @return the modeUtilities + */ + public float[] getModeUtilities() + { + return modeUtilities; + } + + /** + * @param modeUtilities + * the modeUtilities to set + */ + public void setModeUtilities(float[] modeUtilities) + { + this.modeUtilities = modeUtilities; + } + + public int getBoardTap() { + return boardTap; + } + + public void setBoardTap(int boardTap) { + this.boardTap = boardTap; + } + + public int getAlightTap() { + return alightTap; + } + + public void setAlightTap(int alightTap) { + this.alightTap = alightTap; + } + + public int getSet() { + return set; + } + + public void setSet(int set) { + this.set = set; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java new file mode 100644 index 0000000..2ac573b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java @@ -0,0 +1,451 @@ +package org.sandag.abm.specialevent; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class SpecialEventTripModeChoiceDMU + implements Serializable, VariableTable +{ + protected transient Logger logger = Logger.getLogger(SpecialEventTripModeChoiceDMU.class); + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected int tourDepartPeriod; + protected int tourArrivePeriod; + protected int tripPeriod; + protected int tripOrigIsTourDest; + protected int tripDestIsTourDest; + protected float parkingCost; + protected float parkingTime; + protected int income; + protected int partySize; + + protected double nmWalkTime; + protected double nmBikeTime; + + protected double ivtCoeff; + protected double costCoeff; + + protected double walkTransitLogsum; + protected double pnrTransitLogsum; + protected double knrTransitLogsum; + + protected int outboundHalfTourDirection; + + public SpecialEventTripModeChoiceDMU(SandagModelStructure modelStructure, Logger aLogger) + { + if (aLogger == null) + { + aLogger = Logger.getLogger("specialEventModel"); + } + logger = aLogger; + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the tripPeriod + */ + public int getTripPeriod() + { + return tripPeriod; + } + + /** + * @param tripPeriod + * the tripPeriod to set + */ + public void setTripPeriod(int tripPeriod) + { + this.tripPeriod = tripPeriod; + } + + /** + * @return the tripOrigIsTourDest + */ + public int getTripOrigIsTourDest() + { + return tripOrigIsTourDest; + } + + /** + * @param tripOrigIsTourDest + * the tripOrigIsTourDest to set + */ + public void setTripOrigIsTourDest(int tripOrigIsTourDest) + { + this.tripOrigIsTourDest = tripOrigIsTourDest; + } + + /** + * @return the tripDestIsTourDest + */ + public int getTripDestIsTourDest() + { + return tripDestIsTourDest; + } + + /** + * @param tripDestIsTourDest + * the tripDestIsTourDest to set + */ + public void setTripDestIsTourDest(int tripDestIsTourDest) + { + this.tripDestIsTourDest = tripDestIsTourDest; + } + + /** + * @return the outboundHalfTourDirection + */ + public int getOutboundHalfTourDirection() + { + return outboundHalfTourDirection; + } + + /** + * @param outboundHalfTourDirection + * the outboundHalfTourDirection to set + */ + public void setOutboundHalfTourDirection(int outboundHalfTourDirection) + { + this.outboundHalfTourDirection = outboundHalfTourDirection; + } + + /** + * @return the tourDepartPeriod + */ + public int getTourDepartPeriod() + { + return tourDepartPeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(int tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(int tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + /** + * @return the tourArrivePeriod + */ + public int getTourArrivePeriod() + { + return tourArrivePeriod; + } + + public double getNm_walkTime() + { + return nmWalkTime; + } + + public void setNonMotorizedWalkTime(double nmWalkTime) + { + this.nmWalkTime = nmWalkTime; + } + + public void setNonMotorizedBikeTime(double nmBikeTime) + { + this.nmBikeTime = nmBikeTime; + } + + public double getNm_bikeTime() + { + return nmBikeTime; + } + + /** + * @return the parkingCost + */ + public float getParkingCost() + { + return parkingCost; + } + + /** + * @param parkingCost + * the parkingCost to set + */ + public void setParkingCost(float parkingCost) + { + this.parkingCost = parkingCost; + } + + /** + * @return the parkingTime + */ + public float getParkingTime() + { + return parkingTime; + } + + /** + * @param parkingTime + * the parkingTime to set + */ + public void setParkingTime(float parkingTime) + { + this.parkingTime = parkingTime; + } + + /** + * @return the income + */ + public int getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(int income) + { + this.income = income; + } + + /** + * @return the partySize + */ + public int getPartySize() + { + return partySize; + } + + /** + * @param partySize + * the partySize to set + */ + public void setPartySize(int partySize) + { + this.partySize = partySize; + } + + public double getNmWalkTime() { + return nmWalkTime; + } + + public void setNmWalkTime(double nmWalkTime) { + this.nmWalkTime = nmWalkTime; + } + + public double getNmBikeTime() { + return nmBikeTime; + } + + public void setNmBikeTime(double nmBikeTime) { + this.nmBikeTime = nmBikeTime; + } + + public double getIvtCoeff() { + return ivtCoeff; + } + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public double getCostCoeff() { + return costCoeff; + } + + public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; + } + + public double getWalkTransitLogsum() { + return walkTransitLogsum; + } + + public void setWalkTransitLogsum(double walkTransitLogsum) { + this.walkTransitLogsum = walkTransitLogsum; + } + + public double getPnrTransitLogsum() { + return pnrTransitLogsum; + } + + public void setPnrTransitLogsum(double pnrTransitLogsum) { + this.pnrTransitLogsum = pnrTransitLogsum; + } + + public double getKnrTransitLogsum() { + return knrTransitLogsum; + } + + public void setKnrTransitLogsum(double knrTransitLogsum) { + this.knrTransitLogsum = knrTransitLogsum; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTourDepartPeriod", 0); + methodIndexMap.put("getTourArrivePeriod", 1); + methodIndexMap.put("getTripPeriod", 2); + methodIndexMap.put("getParkingCost", 3); + methodIndexMap.put("getParkingTime", 4); + methodIndexMap.put("getTripOrigIsTourDest", 5); + methodIndexMap.put("getTripDestIsTourDest", 6); + methodIndexMap.put("getIncome", 7); + methodIndexMap.put("getPartySize", 8); + + methodIndexMap.put("getIvtCoeff", 60); + methodIndexMap.put("getCostCoeff", 61); + + methodIndexMap.put("getWalkSetLogSum", 62); + methodIndexMap.put("getPnrSetLogSum", 63); + methodIndexMap.put("getKnrSetLogSum", 64); + + methodIndexMap.put("getNm_walkTime", 90); + methodIndexMap.put("getNm_bikeTime", 91); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTourDepartPeriod(); + break; + case 1: + returnValue = getTourArrivePeriod(); + break; + case 2: + returnValue = getTripPeriod(); + break; + case 3: + returnValue = getParkingCost(); + break; + case 4: + returnValue = getParkingTime(); + break; + case 5: + returnValue = getTripOrigIsTourDest(); + break; + case 6: + returnValue = getTripDestIsTourDest(); + break; + case 7: + returnValue = getIncome(); + break; + case 8: + returnValue = getPartySize(); + break; + case 60: + returnValue = getIvtCoeff(); + break; + case 61: + returnValue = getCostCoeff(); + break; + case 62: + returnValue = getWalkTransitLogsum(); + break; + case 63: + returnValue = getPnrTransitLogsum(); + break; + case 64: + returnValue = getKnrTransitLogsum(); + break; + case 90: + returnValue = getNm_walkTime(); + break; + case 91: + returnValue = getNm_bikeTime(); + break; + default: + logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + + + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java new file mode 100644 index 0000000..8680d17 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java @@ -0,0 +1,284 @@ +package org.sandag.abm.specialevent; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class SpecialEventTripModeChoiceModel +{ + + private transient Logger logger = Logger.getLogger("specialEventModel"); + + private AutoAndNonMotorizedSkimsCalculator anm; + private McLogsumsCalculator logsumHelper; + private ModelStructure modelStructure; + private TazDataManager tazs; + private MgraDataManager mgraManager; + + private SpecialEventTripModeChoiceDMU dmu; + private ChoiceModelApplication tripModeChoiceModel; + private boolean saveUtilsAndProbs; + double logsum = 0; + TableDataSet eventData; + private TripModeChoiceDMU mcDmuObject; + + private static final String PROPERTIES_UEC_DATA_SHEET = "specialEvent.trip.mc.data.page"; + private static final String PROPERTIES_UEC_MODEL_SHEET = "specialEvent.trip.mc.model.page"; + private static final String PROPERTIES_UEC_FILE = "specialEvent.trip.mc.uec.file"; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public SpecialEventTripModeChoiceModel(HashMap propertyMap, + ModelStructure myModelStructure, SpecialEventDmuFactoryIf dmuFactory, + McLogsumsCalculator myLogsumHelper, TableDataSet eventData) + { + tazs = TazDataManager.getInstance(propertyMap); + mgraManager = MgraDataManager.getInstance(propertyMap); + + modelStructure = myModelStructure; + logsumHelper = myLogsumHelper; + this.eventData = eventData; + + SandagModelStructure modelStructure = new SandagModelStructure(); + mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); + + setupTripModeChoiceModel(propertyMap, dmuFactory); + saveUtilsAndProbs = Util.getBooleanValueFromPropertyMap(propertyMap, + "specialEvent.saveUtilsAndProbs"); + + } + + /** + * Read the UEC file and set up the trip mode choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupTripModeChoiceModel(HashMap propertyMap, + SpecialEventDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up Special Event trip mode choice model.")); + + dmu = dmuFactory.getSpecialEventTripModeChoiceDMU(); + + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_DATA_SHEET)); + int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_MODEL_SHEET)); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); + tripModeUecFile = uecPath + tripModeUecFile; + logger.info(tripModeUecFile); + + tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, + propertyMap, (VariableTable) dmu); + logger.info(String.format("Finished setting up Special Event trip mode choice model.")); + } + + /** + * Calculate utilities and return logsum for the tour and stop. + * + * @param tour + * @param trip + */ + public double computeUtilities(SpecialEventTour tour, SpecialEventTrip trip) + { + + setDmuAttributes(tour, trip); + + tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + tour.logTourObject(logger, 100); + tripModeChoiceModel.logUECResults(logger, "Special Event trip mode choice model"); + + } + + logsum = tripModeChoiceModel.getLogsum(); + + if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); + + return logsum; + + } + + /** + * Choose a mode and store in the trip object. + * + * @param tour + * SpecialEventTour + * @param trip + * SpecialEventTrip + * + */ + public void chooseMode(SpecialEventTour tour, SpecialEventTrip trip) + { + + computeUtilities(tour, trip); + + double rand = tour.getRandom(); + int mode = tripModeChoiceModel.getChoiceResult(rand); + + trip.setTripMode(mode); + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + + float vot = 0.0f; + + if(modelStructure.getTourModeIsS2(mode)){ + int votIndex = uec.lookupVariableIndex("votS2"); + vot = (float) uec.getValueForIndex(votIndex); + }else if (modelStructure.getTourModeIsS3(mode)){ + int votIndex = uec.lookupVariableIndex("votS3"); + vot = (float) uec.getValueForIndex(votIndex); + }else{ + int votIndex = uec.lookupVariableIndex("vot"); + vot = (float) uec.getValueForIndex(votIndex); + } + tour.setValueOfTime(vot); + + if(mode>=9){ + double[][] bestTapPairs = null; + + if (mode == 9){ + bestTapPairs = logsumHelper.getBestWtwTripTaps(); + } + else if (mode==10||mode==11){ + if (!trip.isInbound()) + bestTapPairs = logsumHelper.getBestDtwTripTaps(); + else + bestTapPairs = logsumHelper.getBestWtdTripTaps(); + } + double rn = tour.getRandom(); + int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); + int boardTap = (int) bestTapPairs[pathIndex][0]; + int alightTap = (int) bestTapPairs[pathIndex][1]; + int set = (int) bestTapPairs[pathIndex][2]; + trip.setBoardTap(boardTap); + trip.setAlightTap(alightTap); + trip.setSet(set); + } + + + if (tour.getDebugChoiceModels()) + { + logger.info("Chose mode " + mode + " with random number " + rand); + } + + if (saveUtilsAndProbs) + { + double[] probs = tripModeChoiceModel.getProbabilities(); + float[] localProbs = new float[probs.length]; + for (int i = 0; i < probs.length; ++i) + localProbs[i] = (float) probs[i]; + + double[] utils = tripModeChoiceModel.getUtilities(); + float[] localUtils = new float[utils.length]; + for (int i = 0; i < utils.length; ++i) + localUtils[i] = (float) utils[i]; + + trip.setModeUtilities(localUtils); + trip.setModeProbabilities(localProbs); + } + + } + + /** + * Set DMU attributes. + * + * @param tour + * @param trip + */ + public void setDmuAttributes(SpecialEventTour tour, SpecialEventTrip trip) + { + + int tourDestinationMgra = tour.getDestinationMGRA(); + int tripOriginMgra = trip.getOriginMgra(); + int tripDestinationMgra = trip.getDestinationMgra(); + + int tripOriginTaz = mgraManager.getTaz(tripOriginMgra); + int tripDestinationTaz = mgraManager.getTaz(tripDestinationMgra); + + dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, + tour.getDebugChoiceModels()); + + dmu.setTourDepartPeriod(tour.getDepartTime()); + dmu.setTourArrivePeriod(tour.getArriveTime()); + dmu.setTripPeriod(trip.getPeriod()); + dmu.setIncome(tour.getIncome()); + dmu.setPartySize(tour.getPartySize()); + if (trip.isInbound()) dmu.setOutboundHalfTourDirection(0); + else dmu.setOutboundHalfTourDirection(1); + + // set trip mc dmu values for transit logsum (gets replaced below by uec values) + double c_ivt = -0.03; + double c_cost = - 0.0033; + + // Solve trip mode level utilities + mcDmuObject.setIvtCoeff(c_ivt); + mcDmuObject.setCostCoeff(c_cost); + double walkTransitLogsum = -999.0; + double driveTransitLogsum = -999.0; + + logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); + dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); + + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); + walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + dmu.setWalkTransitLogsum(walkTransitLogsum); + if (!trip.isInbound()) + { + logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); + } else + { + logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); + } + + dmu.setPnrTransitLogsum(driveTransitLogsum); + dmu.setKnrTransitLogsum(driveTransitLogsum); + + int eventNumber = tour.getEventNumber(); + + float parkingCost = eventData.getValueAt(eventNumber, "ParkingCost"); + float parkingTime = eventData.getValueAt(eventNumber, "ParkingTime"); + + dmu.setParkingCost(parkingCost); + dmu.setParkingTime(parkingTime); + + if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); + else dmu.setTripOrigIsTourDest(0); + + if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); + else dmu.setTripDestIsTourDest(0); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java new file mode 100644 index 0000000..f3edf5a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java @@ -0,0 +1,595 @@ +package org.sandag.abm.specialevent; +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class SpecialEventTripTables { + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TableDataSet tripData; + + // Some parameters + private int[] modeIndex; // an index array,dimensioned by number of total modes, returns 0=auto modes, 1=non-motor, 2=transit, 3=other + private int[] matrixIndex; // an index array, dimensioned by number of modes, returns the element of the matrix array to store value + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private HashMap rbMap; + + // matrices are indexed by modes + private Matrix[][] matrix; + + private ResourceBundle rb; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + + private MatrixDataServerRmi ms; + private float sampleRate; + private static int iteration=1; + public int numSkimSets; + + public SpecialEventTripTables(HashMap rbMap) + { + + this.rbMap = rbMap; + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTourModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + } else if (modelStructure.getTourModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + } else if (modelStructure.getTourModeIsWalkTransit(i) + || modelStructure.getTourModeIsDriveTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + } + } + + logger.info("autoModes="+autoModes+" nmotModes="+nmotModes+" tranModes="+tranModes+" othrModes="+othrModes); + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][]; + for (int i = 0; i < numberOfModes; ++i) + { + + String modeName; + + if (i == 0) + { + matrix[i] = new Matrix[autoModes]; + for (int j = 0; j < autoModes; ++j) + { + modeName = modelStructure.getModeName(j + 1); + matrix[i][j] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j].setExternalNumbers(tazIndex); + } + } else if (i == 1) + { + matrix[i] = new Matrix[nmotModes]; + for (int j = 0; j < nmotModes; ++j) + { + modeName = modelStructure.getModeName(j + 1 + autoModes); + matrix[i][j] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j].setExternalNumbers(tazIndex); + } + } else if (i == 2) + { + matrix[i] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l pMap; + String propertiesFile = null; + + logger.info(String.format( + "SANDAG Special Event Model Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + float sampleRate = 1.0f; + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + logger.info("Special Event Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + SpecialEventTripTables tripTables = new SpecialEventTripTables(pMap); + tripTables.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + tripTables.ms = matrixServer; + } else + { + tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + tripTables.ms.testRemote("SpecialEventTripTables"); + + // mdm = MatrixDataManager.getInstance(); + // mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + tripTables.createTripTables(mt); + + } + + /** + * @return the sampleRate + */ + public double getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(float sampleRate) + { + this.sampleRate = sampleRate; + } + +} + + diff --git a/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java b/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java new file mode 100644 index 0000000..912a87c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java @@ -0,0 +1,363 @@ +package org.sandag.abm.survey; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.rmi.RemoteException; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.accessibilities.BestTransitPathCalculator; +import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; +import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; +import org.sandag.abm.modechoice.TransitDriveAccessDMU; +import org.sandag.abm.modechoice.TransitWalkAccessDMU; +import org.sandag.abm.modechoice.Modes; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +/** + * This class reads and processes on-board transit survey data. It writes out + * all TAP-pairs given the origin/destination MAZ, the time period, and the + * access/egress mode sequence for the observation. + * + * @author joel.freedman + * + */ +public class OutputTapPairs { + private static final Logger logger = Logger.getLogger(OutputTapPairs.class); + private BestTransitPathCalculator bestPathCalculator; + protected WalkTransitWalkSkimsCalculator wtw; + protected WalkTransitDriveSkimsCalculator wtd; + protected DriveTransitWalkSkimsCalculator dtw; + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + private MatrixDataServerRmi ms; + private String inputFile; + private String outputFile; + private TableDataSet inputDataTable; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private HashMap sequentialMaz; + + AutoTazSkimsCalculator tazDistanceCalculator; + + protected PrintWriter writer; + + + public OutputTapPairs(HashMap propertyMap, String inputFile, String outputFile){ + this.inputFile = inputFile; + this.outputFile = outputFile; + + startMatrixServer(propertyMap); + initialize(propertyMap); + } + + + /** + * Initialize best path builders. + * + * @param propertyMap A property map with relevant properties. + */ + public void initialize(HashMap propertyMap){ + + logger.info("Initializing OutputTapPairs"); + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + bestPathCalculator = new BestTransitPathCalculator(propertyMap); + + tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); + tazDistanceCalculator.computeTazDistanceArrays(); + + wtw = new WalkTransitWalkSkimsCalculator(propertyMap); + wtw.setup(propertyMap, logger, bestPathCalculator); + wtd = new WalkTransitDriveSkimsCalculator(propertyMap); + wtd.setup(propertyMap, logger, bestPathCalculator); + dtw = new DriveTransitWalkSkimsCalculator(propertyMap); + dtw.setup(propertyMap, logger, bestPathCalculator); + + readData(); + createOutputFile(); + + } + + /** + * Read data into inputDataTable tabledataset. + * + */ + private void readData(){ + + logger.info("Begin reading the data in file " + inputFile); + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + inputDataTable = csvFile.readFile(new File(inputFile)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End reading the data in file " + inputFile); + } + + /** + * Create the output file and write a header record. + */ + private void createOutputFile(){ + + logger.info("Creating file " + outputFile); + + try + { + writer = new PrintWriter(new BufferedWriter(new FileWriter(outputFile))); + } catch (IOException e) + { + logger.fatal("Could not open file " + outputFile + " for writing\n"); + throw new RuntimeException(); + } + String headerString = new String( + "rownames,npath,access_mode_recode,period,set,boardTap,alightTap,bestUtility,accessTime,egressTime,auxWalkTime," + + "localBusIvt,expressBusIvt,brtIvt,lrtIvt,crIvt,firstWaitTime,trfWaitTime,fare,totalIVT,xfers\n"); + + + writer.print(headerString); + + } + + + /** + * Iterate through input data and process\write taps. + */ + private void run(){ + + TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); + TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); + double[][] bestTaps = null; + double[] skims = null; + double boardAccessTime; + double alightAccessTime; + //iterate through data and calculate + for(int row = 1; row<=inputDataTable.getRowCount();++row ){ + + if((row<=100) || ((row % 100) == 0)) + logger.info("Processing input record "+row); + + String label=inputDataTable.getStringValueAt(row, "id"); + int originMaz = (int) inputDataTable.getValueAt(row, "orig_maz"); + int destinationMaz = (int) inputDataTable.getValueAt(row, "dest_maz"); + int period = (int) inputDataTable.getValueAt(row, "period")-1; //Input is 1=EA, 2=AM, 3=MD, 4=PM, 5=EV + int accessMode = (int) inputDataTable.getValueAt(row, "accessEgress"); // 1 walk, 2 PNR, 3 KNR\bike + int inbound = (int) inputDataTable.getValueAt(row, "inbound"); // 1 if inbound, else 0 + + int accessEgressMode = -1; + + if(accessMode ==1) + accessEgressMode=bestPathCalculator.WTW; + else if ((accessMode == 2||accessMode==3) && inbound==0) + accessEgressMode = bestPathCalculator.DTW; + else if ((accessMode == 2||accessMode==3) && inbound==1) + accessEgressMode = bestPathCalculator.WTD; + + if(originMaz==0||destinationMaz==0||accessEgressMode==-1) + continue; + + + int originTaz = mgraManager.getTaz(originMaz); + int destinationTaz = mgraManager.getTaz(destinationMaz); + + float odDistance = (float) tazDistanceCalculator.getTazToTazDistance(ModelStructure.AM_SKIM_PERIOD_INDEX, originTaz, destinationTaz); + bestTaps = bestPathCalculator.getBestTapPairs(walkDmu, driveDmu, accessEgressMode, originMaz, destinationMaz, period, false, logger, odDistance); + double[] bestUtilities = bestPathCalculator.getBestUtilities(); + + //iterate through n-best paths + for (int i = 0; i < bestTaps.length; i++) + { + if(bestUtilities[i]<-500) + continue; + + writer.print(label); + + //write transit TAP pairs and utility + int boardTap = (int) bestTaps[i][0]; + int alightTap = (int) bestTaps[i][1]; + int set = (int) bestTaps[i][2]; + + writer.format(",%d,%d,%d,%d,%d,%d,%9.4f",i,accessMode,period,set,boardTap,alightTap,bestUtilities[i]); + + // System.out.println(label+String.format(",%d,%d,%d,%d,%d,%d,%9.4f",i,accessEgressMode,period,set,boardTap,alightTap,bestUtilities[i])); + //write skims + if(accessEgressMode==bestPathCalculator.WTW){ + boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); + alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); + skims = wtw.getWalkTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); + }else if (accessEgressMode==bestPathCalculator.DTW){ + boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( accessMode==2? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); + skims = dtw.getDriveTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); + }else if(accessEgressMode==bestPathCalculator.WTD){ + boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); + alightAccessTime = tazManager.getTimeToTapFromTaz(destinationTaz,alightTap,( accessMode==2? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); + skims = wtd.getWalkTransitDriveSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); + } + + for(int j=0; j < skims.length; ++j) + writer.format(",%9.2f",skims[j]); + + writer.format("\n"); + + } + writer.flush(); + } + + + } + + /** + * Startup a connection to the matrix manager. + * + * @param serverAddress + * @param serverPort + * @param mt + * @return + */ + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + try + { + // create the concrete data server object + matrixServer.start32BitMatrixIoServer(mt); + } catch (RuntimeException e) + { + matrixServer.stop32BitMatrixIoServer(); + logger.error( + "RuntimeException caught making remote method call to start 32 bit mitrix in remote MatrixDataServer.", + e); + } + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + matrixServer.stop32BitMatrixIoServer(); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + matrixServer.stop32BitMatrixIoServer(); + throw new RuntimeException(); + } + + return matrixServer; + + } + + + /** + * Main run method + * @param args + */ + public static void main(String[] args) { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("Best Tap Pairs Program using CT-RAMP version ", + CtrampApplication.VERSION)); + + logger.info(String.format("Outputting TAP pairs and utilities for on-board survey data")); + + + String inputFile = null; + String outputFile = null; + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else { + propertiesFile = args[0]; + + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-inputFile")) + { + inputFile = args[i + 1]; + } + if (args[i].equalsIgnoreCase("-outputFile")) + { + outputFile = args[i + 1]; + } + } + } + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + OutputTapPairs outputTapPairs = new OutputTapPairs(pMap, inputFile, outputFile); + + + outputTapPairs.run(); + + + + + } + private void startMatrixServer(HashMap properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + MatrixDataServerIf ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + logger.error("could not connect to matrix server", e); + throw new RuntimeException(e); + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java new file mode 100644 index 0000000..ae898e5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java @@ -0,0 +1,410 @@ +package org.sandag.abm.utilities; + +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.jppf.client.JPPFClient; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.HouseholdChoiceModelRunner; +import org.sandag.abm.ctramp.HouseholdDataManager; +import org.sandag.abm.ctramp.HouseholdDataManagerIf; +import org.sandag.abm.ctramp.HouseholdDataManagerRmi; +import org.sandag.abm.ctramp.HouseholdDataWriter; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.UsualWorkSchoolLocationChoiceModel; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; + +public class CreateLogsums { + + + private BuildAccessibilities aggAcc; + private JPPFClient jppfClient; + private static Logger logger = Logger.getLogger(CreateLogsums.class); + private HouseholdDataManagerIf householdDataManager; + private HashMap propertyMap; + private ResourceBundle resourceBundle; + // are used if no command line arguments are specified. + private int globalIterationNumber = 0; + private float iterationSampleRate = 0f; + private int sampleSeed = 0; + private SandagModelStructure modelStructure; + private SandagCtrampDmuFactory dmuFactory; + private MatrixDataServerIf ms; + private ModelOutputReader modelOutputReader; + + + /** + * Constructor. + * + * @param propertiesFile + * @param globalIterationNumber + * @param globalSampleRate + * @param sampleSeed + */ + public CreateLogsums(String propertiesFile, int globalIterationNumber, float globalSampleRate, int sampleSeed){ + + this.resourceBundle = ResourceBundle.getBundle(propertiesFile); + propertyMap = ResourceUtil.getResourceBundleAsHashMap ( propertiesFile); + this.globalIterationNumber = globalIterationNumber; + this.iterationSampleRate = globalSampleRate; + this.sampleSeed = sampleSeed; + + } + + /** + * Initialize data members + */ + public void initialize(){ + + startMatrixServer(propertyMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + householdDataManager = getHouseholdDataManager(); + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after getting household manager"); + + // create a factory object to pass to various model components from which + // they can create DMU objects + dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + modelOutputReader = new ModelOutputReader(propertyMap,modelStructure, globalIterationNumber); + } + + + /** + * Run all components. + * + */ + public void run(){ + + initialize(); + readModelOutputsAndCreateTours(); + createWorkLogsums(); + createNonWorkLogsums(); + + HouseholdDataWriter dataWriter = new HouseholdDataWriter( propertyMap, modelStructure, globalIterationNumber ); + dataWriter.writeDataToFiles(householdDataManager); + + } + + /** + * Read the model outputs and create tours. + */ + public void readModelOutputsAndCreateTours(){ + + modelOutputReader.readHouseholdDataOutput(); + modelOutputReader.readPersonDataOutput(); + modelOutputReader.readTourDataOutput(); + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before reading model output"); + + Household[] households = householdDataManager.getHhArray(); + for(Household household : households){ + + modelOutputReader.setHouseholdAndPersonAttributes(household); + + if(modelOutputReader.hasJointTourFile()) + modelOutputReader.createJointTours(household); + + if(modelOutputReader.hasIndividualTourFile()) + modelOutputReader.createIndividualTours(household); + } + householdDataManager.setHhArray(households); + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after reading model output"); + + } + + + + /** + * Calculate and write work destination choice logsums for the synthetic population. + * + * @param propertyMap + */ + public void createWorkLogsums(){ + + jppfClient = new JPPFClient(); + + if (aggAcc == null) + { + logger.info("creating Accessibilities Object for UWSL."); + aggAcc = BuildAccessibilities.getInstance(); + aggAcc.setupBuildAccessibilities(propertyMap,false); +// aggAcc.setJPPFClient(jppfClient); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + + boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, "acc.read.input.file"); + if (readAccessibilities) + { + String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap,"Project.Directory"); + String accFileName = Paths.get(projectDirectory,Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file")).toString(); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + } + + // new the usual school and location choice model object + UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( + resourceBundle, "none", jppfClient, modelStructure, ms, dmuFactory, aggAcc); + + // calculate and get the array of worker size terms table - MGRAs by + // occupations + aggAcc.createWorkSegmentNameIndices(); + aggAcc.calculateWorkerSizeTerms(); + double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); + + // run the model + logger.info("Starting usual work location choice for logsum calculations."); + usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, workerSizeTerms); + logger.info("Finished with usual work location choice for logsum calculations."); + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after running school and work location choice"); + + } + + public void createNonWorkLogsums(){ + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before running non-work logsums"); + + HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( propertyMap, jppfClient, "False", householdDataManager, ms, modelStructure, dmuFactory ); + runner.runHouseholdChoiceModels(); + + } + + + + /** + * Create the household data manager. Based on the code in MTCTM2TourBasedModel.runTourBasedModel() + * @return The household data manager interface. + */ + public HouseholdDataManagerIf getHouseholdDataManager( ){ + + + boolean localHandlers = false; + + String testString; + + HouseholdDataManagerIf householdDataManager; + String hhHandlerAddress = ""; + int hhServerPort = 0; + try + { + // get household server address. if none is specified a local server in + // the current process will be started. + hhHandlerAddress = resourceBundle.getString("RunModel.HouseholdServerAddress"); + try + { + // get household server port. + hhServerPort = Integer.parseInt(resourceBundle.getString("RunModel.HouseholdServerPort")); + localHandlers = false; + } catch (MissingResourceException e) + { + // if no household data server address entry is found, the object + // will be created in the local process + localHandlers = true; + } + } catch (MissingResourceException e) + { + localHandlers = true; + } + + + try + { + + if (localHandlers) + { + + // create a new local instance of the household array manager + householdDataManager = new SandagHouseholdDataManager(); + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + String inputHouseholdFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); + + } else + { + + householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, + SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + testString = householdDataManager.testRemote(); + logger.info("HouseholdDataManager test: " + testString); + + householdDataManager.setPropertyFileValues(propertyMap); + } + + //always starting from scratch (RunModel.RestartWithHhServer=none) + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = resourceBundle + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = resourceBundle + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); + + }catch (Exception e) + { + + logger.error(String + .format("Exception caught setting up household data manager."), e); + throw new RuntimeException(); + + } + + return householdDataManager; + } + + + /** + * Start a new matrix server connection. + * + * @param properties + */ + private void startMatrixServer(HashMap properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + logger.error("could not connect to matrix server", e); + throw new RuntimeException(e); + + } + + } + + + public JPPFClient getJppfClient() { + return jppfClient; + } + + public void setJppfClient(JPPFClient jppfClient) { + this.jppfClient = jppfClient; + } + + public static void main(String[] args) { + + long startTime = System.currentTimeMillis(); + int globalIterationNumber = -1; + float iterationSampleRate = -1.0f; + int sampleSeed = -1; + + ResourceBundle rb = null; + + logger.info( String.format( "Generating Logsums from MTC Tour Based Model using CT-RAMP version %s, 22feb2011 build %s", CtrampApplication.VERSION, 2 ) ); + + if ( args.length == 0 ) { + logger.error( String.format( "no properties file base name (without .properties extension) was specified as an argument." ) ); + return; + } + else { + rb = ResourceBundle.getBundle( args[0] ); + + // optional arguments + for (int i=1; i < args.length; i++) { + + if (args[i].equalsIgnoreCase("-iteration")) { + globalIterationNumber = Integer.parseInt( args[i+1] ); + logger.info( String.format( "-iteration %d.", globalIterationNumber ) ); + } + + if (args[i].equalsIgnoreCase("-sampleRate")) { + iterationSampleRate = Float.parseFloat( args[i+1] ); + logger.info( String.format( "-sampleRate %.4f.", iterationSampleRate ) ); + } + + if (args[i].equalsIgnoreCase("-sampleSeed")) { + sampleSeed = Integer.parseInt( args[i+1] ); + logger.info( String.format( "-sampleSeed %d.", sampleSeed ) ); + } + + } + + if ( globalIterationNumber < 0 ) { + globalIterationNumber = 1; + logger.info( String.format( "no -iteration flag, default value %d used.", globalIterationNumber ) ); + } + + if ( iterationSampleRate < 0 ) { + iterationSampleRate = 1; + logger.info( String.format( "no -sampleRate flag, default value %.4f used.", iterationSampleRate ) ); + } + + if ( sampleSeed < 0 ) { + sampleSeed = 0; + logger.info( String.format( "no -sampleSeed flag, default value %d used.", sampleSeed ) ); + } + + } + + + String baseName; + if ( args[0].endsWith(".properties") ) { + int index = args[0].indexOf(".properties"); + baseName = args[0].substring(0, index); + } + else { + baseName = args[0]; + } + + + // create an instance of this class for main() to use. + CreateLogsums mainObject = new CreateLogsums( args[0], globalIterationNumber, iterationSampleRate, sampleSeed ); + + // Create logsums + try { + + logger.info ("Creating logsums."); + mainObject.run(); + + } + catch ( RuntimeException e ) { + logger.error ( "RuntimeException caught in com.pb.mtctm2.abm.reports.CreateLogsums.main() -- exiting.", e ); + System.exit(2); + } + + + logger.info (""); + logger.info (""); + logger.info ("CreateLogsums finished in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); + + System.exit(0); + + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java new file mode 100644 index 0000000..92ee34d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java @@ -0,0 +1,41 @@ +package org.sandag.abm.utilities; + +import java.util.HashMap; +/** + * ErrorLogging definitions + * + * @author wsu + * @mail Wu.Sun@sandag.org + * @version 13.3.0 + * @since 2016-06-20 + * + */ +public class ErrorLogging { + public static final String AtTapNotInTransitNetwork = "A TAP in AT network is not in TRANSIT network!"; + public static final String TransitTapNotInAt = "A TAP in TRANSIT network is not in AT network!"; + public static final String InconsistentTapPostions = "Positions of a TAP are different in AT and TRANSIT networks!"; + public static final String NoAtTaps = "No valid TAPs in AT network!"; + public static final String NoTransitTaps = "No valid TAPs in TRANSIT network!"; + protected HashMap atErrorIndexMap; + + public ErrorLogging() + { + + atErrorIndexMap = new HashMap(); + createAtErrorIndexMap(); + } + private void createAtErrorIndexMap() + { + atErrorIndexMap.put("AT1", AtTapNotInTransitNetwork); + atErrorIndexMap.put("AT2", TransitTapNotInAt ); + atErrorIndexMap.put("AT3", InconsistentTapPostions); + atErrorIndexMap.put("AT4", NoAtTaps); + atErrorIndexMap.put("AT5", NoTransitTaps); + } + + public String getAtError(String index) + { + return atErrorIndexMap.get(index); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java new file mode 100644 index 0000000..23ef9fa --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java @@ -0,0 +1,836 @@ +package org.sandag.abm.utilities; + +import java.io.File; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.ModelStructure; +import org.sandag.abm.ctramp.Person; +import org.sandag.abm.ctramp.Tour; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +public class ModelOutputReader { + + private transient Logger logger = Logger.getLogger(ModelOutputReader.class); + + private static final String PROPERTIES_HOUSEHOLD_DATA_FILE = "Accessibilities.HouseholdDataFile"; + private static final String PROPERTIES_PERSON_DATA_FILE = "Accessibilities.PersonDataFile"; + private static final String PROPERTIES_INDIV_TOUR_DATA_FILE = "Accessibilities.IndivTourDataFile"; + private static final String PROPERTIES_JOINT_TOUR_DATA_FILE = "Accessibilities.JointTourDataFile"; + private static final String PROPERTIES_INDIV_TRIP_DATA_FILE = "Accessibilities.IndivTripDataFile"; + private static final String PROPERTIES_JOINT_TRIP_DATA_FILE = "Accessibilities.JointTripDataFile"; + private ModelStructure modelStructure; + private int iteration; + private HashMap rbMap; + private HashMap householdFileAttributesMap; + private HashMap personFileAttributesMap; + private HashMap> individualTourAttributesMap; //by person_id + private HashMap> jointTourAttributesMap; //by hh_id + + private boolean readIndividualTourFile = false; + private boolean readJointTourFile = false; + + /** + * Default constructor. + * @param rbMap Hashmap of properties + * @param modelStructure Model structure object + * @param iteration Iteration number used for file names + */ + public ModelOutputReader(HashMap rbMap, ModelStructure modelStructure, + int iteration) + { + logger.info("Writing data structures to files."); + this.modelStructure = modelStructure; + this.iteration = iteration; + this.rbMap = rbMap; + } + + + /** + * Read household data and store records in householdFileAttributesMap + */ + public void readHouseholdDataOutput(){ + + String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String hhFile = rbMap.get(PROPERTIES_HOUSEHOLD_DATA_FILE); + + TableDataSet householdData = readTableData(baseDir+hhFile); + + householdFileAttributesMap = new HashMap(); + + //hh_id,home_mgra,income,HVs,AVs,transponder,cdap_pattern,out_escort_choice,inb_escort_choice,jtf_choice + for(int row = 1; row<=householdData.getRowCount();++row){ + + long hhid = (long) householdData.getValueAt(row,"hh_id"); + int home_mgra = (int)householdData.getValueAt(row,"home_mgra"); + int income = (int) householdData.getValueAt(row,"income"); + int automated_vehicles = (int) householdData.getValueAt(row,"AVs"); + int human_vehicles = (int) householdData.getValueAt(row,"HVs"); + int autos = automated_vehicles + human_vehicles; + int transponder = (int) householdData.getValueAt(row,"transponder"); + String cdap_pattern = householdData.getStringValueAt(row,"cdap_pattern"); + int jtf_choice = (int) householdData.getValueAt(row,"jtf_choice"); + int out_escort_choice = (int) householdData.getValueAt(row,"out_escort_choice"); + int inb_escort_choice = (int) householdData.getValueAt(row,"inb_escort_choice"); + + + // float sampleRate = householdData.getValueAt(row,"sampleRate"); + HouseholdFileAttributes hhAttributes = new HouseholdFileAttributes(hhid, + home_mgra, income, autos, automated_vehicles, human_vehicles,transponder,cdap_pattern, + jtf_choice,out_escort_choice,inb_escort_choice); + + householdFileAttributesMap.put(hhid, hhAttributes); + + } + } + + + /** + * Read the data from the Results.PersonDataFile. + * Data is stored in HashMap personFileAttributesMap + * so that it can be retrieved quickly for a household object. + * + */ + public void readPersonDataOutput(){ + + //read person data + String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String personFile = baseDir + rbMap.get(PROPERTIES_PERSON_DATA_FILE); + TableDataSet personData = readTableData(personFile); + + personFileAttributesMap = new HashMap(); + //hh_id,person_id,person_num,age,gender,type,value_of_time,activity_pattern,imf_choice,inmf_choice, + // fp_choice,reimb_pct,tele_choice,ie_choice,timeFactorWork,timeFactorNonWork + + for(int row = 1; row<=personData.getRowCount();++row){ + + //get the values for this person + long hhid = (long) personData.getValueAt(row, "hh_id"); + long person_id = (long) personData.getValueAt(row,"person_id"); + long personNumber = (long) personData.getValueAt(row,"person_num"); + int age = (int) personData.getValueAt(row,"age"); + + String genderString = personData.getStringValueAt(row,"gender"); + int gender = (genderString.compareTo("m")==0 ? 1 : 2); + + float valueOfTime = personData.getValueAt(row,"value_of_time"); + String activityPattern = personData.getStringValueAt(row,"activity_pattern"); + String type = personData.getStringValueAt(row,"type"); + int personType = getPersonType(type); + + // int occup = (int) personData.getValueAt(row,"occp"); + + + int imfChoice = (int) personData.getValueAt(row, "imf_choice"); + int inmfChoice = (int) personData.getValueAt(row, "inmf_choice"); + int fp_choice = (int) personData.getValueAt(row,"fp_choice"); + float reimb_pct = personData.getValueAt(row,"reimb_pct"); + int tele_choice = (int) personData.getValueAt(row,"tele_choice"); + int ie_choice = (int) personData.getValueAt(row,"ie_choice"); + float timeFactorWork = personData.getValueAt(row,"timeFactorWork"); + float timeFactorNonWork = personData.getValueAt(row,"timeFactorNonWork"); + + //float sampleRate = personData.getValueAt(row,"sampleRate"); + + PersonFileAttributes personFileAttributes = new PersonFileAttributes(hhid,person_id,personNumber,age,gender,valueOfTime, + activityPattern,personType, imfChoice,inmfChoice,fp_choice,reimb_pct,tele_choice, + ie_choice, timeFactorWork, timeFactorNonWork); + + personFileAttributesMap.put(person_id,personFileAttributes); + + } + + } + + /** + * Read both tour files. + * + */ + public void readTourDataOutput(){ + + String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + + if(rbMap.containsKey(PROPERTIES_INDIV_TOUR_DATA_FILE)){ + String indivTourFile = rbMap.get(PROPERTIES_INDIV_TOUR_DATA_FILE); + if(indivTourFile != null){ + if(indivTourFile.length()>0){ + individualTourAttributesMap = readTourData(baseDir+indivTourFile, false, individualTourAttributesMap); + readIndividualTourFile = true; + } + } + } + if(rbMap.containsKey(PROPERTIES_JOINT_TOUR_DATA_FILE)){ + String jointTourFile = rbMap.get(PROPERTIES_JOINT_TOUR_DATA_FILE); + if(jointTourFile != null){ + if(jointTourFile.length()>0){ + jointTourAttributesMap = readTourData(baseDir+jointTourFile, true, jointTourAttributesMap); + readJointTourFile = true; + } + } + } + if(readIndividualTourFile==false){ + logger.info("No individual tour file to read in MtcModelOutputReader class"); + } + if(readJointTourFile==false){ + logger.info("No joint tour file to read in MtcModelOutputReader class"); + } + } + + + /** + * Read the data from the Results.IndivTourDataFile or Results.JointTourDataFile. + * Data is stored in HashMap passed into method as an argument. Method handles + * both individual and joint data. Joint tour data is indexed by hh_id + * so that it can be retrieved quickly for a household object. Individual tour data is + * indexed by person_id. + * + */ + public HashMap> readTourData(String filename, boolean isJoint, HashMap> tourFileAttributesMap ){ + + TableDataSet tourData = readTableData(filename); + + tourFileAttributesMap = new HashMap>(); + //hh_id,person_id,person_num,person_type,tour_id,tour_category,tour_purpose, + //orig_mgra,dest_mgra,start_period,end_period,tour_mode,av_avail,tour_distance,atwork_freq, + //num_ob_stops,num_ib_stops,valueOfTime,escort_type_out,escort_type_in,driver_num_out,driver_num_in + + for(int row = 1; row<=tourData.getRowCount();++row){ + + long hh_id = (long) tourData.getValueAt(row,"hh_id"); + long person_id = 0; + int person_num=0; + int person_type=0; + int escort_type_out=0; + int escort_type_in=0; + int driver_num_out=0; + int driver_num_in=0; + if(!isJoint){ + person_id = (long) tourData.getValueAt(row,"person_id");; + person_num = (int) tourData.getValueAt(row,"person_num"); + person_type = (int) tourData.getValueAt(row,"person_type"); + escort_type_out = (int) tourData.getValueAt(row,"escort_type_out"); + escort_type_in = (int) tourData.getValueAt(row,"escort_type_in"); + driver_num_out = (int) tourData.getValueAt(row,"driver_num_out"); + driver_num_in = (int) tourData.getValueAt(row,"driver_num_in"); + } + int tour_id = (int) tourData.getValueAt(row,"tour_id"); + String tour_category = tourData.getStringValueAt(row,"tour_category"); + String tour_purpose = tourData.getStringValueAt(row,"tour_purpose"); + + int tour_composition = 0; + String tour_participants = null; + if(isJoint){ + tour_composition = (int) tourData.getValueAt(row,"tour_composition"); + tour_participants = tourData.getStringValueAt(row,"tour_participants"); + } + + int orig_mgra = (int) tourData.getValueAt(row,"orig_mgra"); + int dest_mgra = (int) tourData.getValueAt(row,"dest_mgra"); + int start_period = (int) tourData.getValueAt(row,"start_period"); + int end_period = (int) tourData.getValueAt(row,"end_period"); + int tour_mode = (int) tourData.getValueAt(row,"tour_mode"); + int av_avail = (int) tourData.getValueAt(row,"av_avail"); + float tour_distance = tourData.getValueAt(row,"tour_distance"); + // float tour_time = tourData.getValueAt(row,"tour_time"); + int atWork_freq = (int) tourData.getValueAt(row,"atWork_freq"); + int num_ob_stops = (int) tourData.getValueAt(row,"num_ob_stops"); + int num_ib_stops = (int) tourData.getValueAt(row,"num_ib_stops"); + float valueOfTime = tourData.getValueAt(row, "valueOfTime"); + /* + int out_btap = (int) tourData.getValueAt(row,"out_btap"); + int out_atap = (int) tourData.getValueAt(row,"out_atap"); + int in_btap = (int) tourData.getValueAt(row,"in_btap"); + int in_atap = (int) tourData.getValueAt(row,"in_atap"); + int out_set = (int) tourData.getValueAt(row,"out_set"); + int in_set = (int) tourData.getValueAt(row,"in_set"); +// float sampleRate = tourData.getValueAt(row,"sampleRate"); +// int avAvailable = (int) tourData.getValueAt(row,"avAvailable"); + */ float[] util = new float[modelStructure.getMaxTourModeIndex()]; + float[] prob = new float[modelStructure.getMaxTourModeIndex()]; + + TourFileAttributes tourFileAttributes = new TourFileAttributes(hh_id, person_id, person_num, person_type, + tour_id, tour_category, tour_purpose, orig_mgra,dest_mgra, + start_period, end_period, tour_mode, av_avail, tour_distance, + atWork_freq, num_ob_stops, num_ib_stops, valueOfTime, + escort_type_out,escort_type_in,driver_num_out,driver_num_in, + tour_composition, tour_participants,util,prob); + + //if individual tour, map key is person_id, else it is hh_id + long key = -1; + if(!isJoint) + key = person_id; + else + key = hh_id; + + //if the not the first tour for this person or hh, add the tour to the existing + //arraylist; else create a new arraylist and add the tour attributes to it, + //then add the arraylist to the map + if(tourFileAttributesMap.containsKey(key)){ + ArrayList tourArray = tourFileAttributesMap.get(key); + tourArray.add(tourFileAttributes); + }else{ + ArrayList tourArray = new ArrayList(); + tourArray.add(tourFileAttributes); + tourFileAttributesMap.put(key, tourArray); + } + + } + + return tourFileAttributesMap; + } + + /** + * Create individual tour objects for all persons in the household object based + * on the data read in the individual tour file. + * + * @param household + */ + public void createIndividualTours(Household household){ + + HashMap purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); + Person[] persons = household.getPersons(); + for(int pnum=1;pnum tourAttributesArray = individualTourAttributesMap.get(personId); + + //store tours by type + ArrayList workTours = new ArrayList(); + ArrayList universityTours = new ArrayList(); + ArrayList schoolTours = new ArrayList(); + ArrayList atWorkSubtours = new ArrayList(); + ArrayList nonMandTours = new ArrayList(); + + for(int i=0;i0){ + p.createWorkTours(workTours.size(), 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, + ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); + ArrayList workTourArrayList = p.getListOfWorkTours(); + for(int i=0;i0){ + p.createSchoolTours(schoolTours.size(), 0, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, + ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); + ArrayList schoolTourArrayList = p.getListOfSchoolTours(); + for(int i=0;i0){ + p.createSchoolTours(universityTours.size(), 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, + ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); + ArrayList universityTourArrayList = p.getListOfSchoolTours(); + for(int i=0;i nonMandTourArrayList = p.getListOfIndividualNonMandatoryTours(); + for(int i =0; i purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); + + //joint tours + long hhid = household.getHhId(); + if(jointTourAttributesMap.containsKey(hhid)){ + ArrayList tourArray = jointTourAttributesMap.get(hhid); + int numberOfJointTours = tourArray.size(); + + //get the first joint tour + TourFileAttributes tourAttributes = tourArray.get(0); + String purposeString = tourAttributes.tour_purpose; + int purpose = purposeIndexMap.get(purposeString); + int composition = tourAttributes.tour_composition; + int[] tourParticipants = getTourParticipantsArray(tourAttributes.tour_participants); + + Tour t1 = new Tour(household,purposeString, ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purpose); + t1.setJointTourComposition(composition); + t1.setPersonNumArray(tourParticipants); + + //if the household has two joint tours, get the second + if(numberOfJointTours==2){ + tourAttributes = tourArray.get(2); + purposeString = tourAttributes.tour_purpose; + purpose = purposeIndexMap.get(purposeString); + composition = tourAttributes.tour_composition; + tourParticipants = getTourParticipantsArray(tourAttributes.tour_participants); + + Tour t2 = new Tour(household,purposeString, ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purpose); + t2.setJointTourComposition(composition); + t2.setPersonNumArray(tourParticipants); + + //set in hh object + household.createJointTourArray(t1, t2); + tourAttributes.setModeledTourAttributes(t1); + tourAttributes.setModeledTourAttributes(t2); + }else{ + household.createJointTourArray(t1); + tourAttributes.setModeledTourAttributes(t1); + } + } + + + } + +// HELPER METHODS AND CLASSES + + /** + * Split the participants string around spaces and return the + * integer array of participant numbers. + * + * @param tourParticipants + * @return + */ + public int[] getTourParticipantsArray(String tourParticipants){ + + String[] values = tourParticipants.split(" "); + int[] array = new int[values.length]; + for (int i = 0; i < array.length; i++) + array[i] = Integer.parseInt(values[i]); + return array; + } + + /** + * Set household and person attributes for this household object. This method uses + * the data in the personFileAttributesMap to set the data members of the + * Person objects for all persons in the household. + * + * @param hhObject + */ + public void setHouseholdAndPersonAttributes(Household hhObject){ + + long hhid = (long) hhObject.getHhId(); + HouseholdFileAttributes hhAttributes = householdFileAttributesMap.get(hhid); + hhAttributes.setHouseholdAttributes(hhObject); + Person[] persons = hhObject.getPersons(); + for(int i=1;i 0) + { + String base = originalFileName.substring(0, lastDot); + String ext = originalFileName.substring(lastDot); + returnString = String.format("%s_%d%s", base, iteration, ext); + } else + { + returnString = String.format("%s_%d.csv", originalFileName, iteration); + } + + logger.info("writing " + originalFileName + " file to " + returnString); + + return returnString; + } + + public HashMap getHouseholdFileAttributesMap() { + return householdFileAttributesMap; + } + + public HashMap getPersonFileAttributesMap() { + return personFileAttributesMap; + } + + public HashMap> getIndividualTourAttributesMap() { + return individualTourAttributesMap; + } + + public HashMap> getJointTourAttributesMap() { + return jointTourAttributesMap; + } + + public boolean hasIndividualTourFile() { + return readIndividualTourFile; + } + + public boolean hasJointTourFile() { + return readJointTourFile; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java new file mode 100644 index 0000000..883786d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java @@ -0,0 +1,402 @@ +package org.sandag.abm.utilities; + +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.jppf.client.JPPFClient; +import org.sandag.abm.accessibilities.BuildAccessibilities; +import org.sandag.abm.application.SandagCtrampDmuFactory; +import org.sandag.abm.application.SandagHouseholdDataManager; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Household; +import org.sandag.abm.ctramp.HouseholdChoiceModelRunner; +import org.sandag.abm.ctramp.HouseholdDataManager; +import org.sandag.abm.ctramp.HouseholdDataManagerIf; +import org.sandag.abm.ctramp.HouseholdDataManagerRmi; +import org.sandag.abm.ctramp.HouseholdDataWriter; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.UsualWorkSchoolLocationChoiceModel; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.calculator.MatrixDataServerIf; +import com.pb.common.util.ResourceUtil; + +public class RunModeChoice { + + + private BuildAccessibilities aggAcc; + private JPPFClient jppfClient; + private static Logger logger = Logger.getLogger(RunModeChoice.class); + private HouseholdDataManagerIf householdDataManager; + private HashMap propertyMap; + private ResourceBundle resourceBundle; + // are used if no command line arguments are specified. + private int globalIterationNumber = 0; + private float iterationSampleRate = 0f; + private int sampleSeed = 0; + private SandagModelStructure modelStructure; + private SandagCtrampDmuFactory dmuFactory; + private MatrixDataServerIf ms; + private ModelOutputReader modelOutputReader; + + + /** + * Constructor. + * + * @param propertiesFile + * @param globalIterationNumber + * @param globalSampleRate + * @param sampleSeed + */ + public RunModeChoice(String propertiesFile, int globalIterationNumber, float globalSampleRate, int sampleSeed){ + + this.resourceBundle = ResourceBundle.getBundle(propertiesFile); + propertyMap = ResourceUtil.getResourceBundleAsHashMap ( propertiesFile); + this.globalIterationNumber = globalIterationNumber; + this.iterationSampleRate = globalSampleRate; + this.sampleSeed = sampleSeed; + + } + + /** + * Initialize data members + */ + public void initialize(){ + + startMatrixServer(propertyMap); + + // create modelStructure object + modelStructure = new SandagModelStructure(); + + householdDataManager = getHouseholdDataManager(); + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after getting household manager"); + + // create a factory object to pass to various model components from which + // they can create DMU objects + dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); + + modelOutputReader = new ModelOutputReader(propertyMap,modelStructure, globalIterationNumber); + + jppfClient = new JPPFClient(); + + } + + + /** + * Run all components. + * + */ + public void run(){ + + initialize(); + readModelOutputsAndCreateTours(); + runHouseholdModels(); + HouseholdDataWriter dataWriter = new HouseholdDataWriter( propertyMap, modelStructure, globalIterationNumber ); + dataWriter.writeDataToFiles(householdDataManager); + + } + + /** + * Read the model outputs and create tours. + */ + public void readModelOutputsAndCreateTours(){ + + modelOutputReader.readHouseholdDataOutput(); + modelOutputReader.readPersonDataOutput(); + modelOutputReader.readTourDataOutput(); + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before reading model output"); + + Household[] households = householdDataManager.getHhArray(); + for(Household household : households){ + + modelOutputReader.setHouseholdAndPersonAttributes(household); + + if(modelOutputReader.hasJointTourFile()) + modelOutputReader.createJointTours(household); + + if(modelOutputReader.hasIndividualTourFile()) + modelOutputReader.createIndividualTours(household); + } + householdDataManager.setHhArray(households); + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after reading model output"); + + } + + + + /** + * Calculate and write work destination choice logsums for the synthetic population. + * + * @param propertyMap + */ + public void createWorkLogsums(){ + + + if (aggAcc == null) + { + logger.info("creating Accessibilities Object for UWSL."); + aggAcc = BuildAccessibilities.getInstance(); + aggAcc.setupBuildAccessibilities(propertyMap,false); +// aggAcc.setJPPFClient(jppfClient); + + aggAcc.calculateSizeTerms(); + aggAcc.calculateConstants(); + + boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, "acc.read.input.file"); + if (readAccessibilities) + { + String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap,"Project.Directory"); + String accFileName = Paths.get(projectDirectory,Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file")).toString(); + + aggAcc.readAccessibilityTableFromFile(accFileName); + + } else + { + + aggAcc.calculateDCUtilitiesDistributed(propertyMap); + + } + } + + // new the usual school and location choice model object + UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( + resourceBundle, "none", jppfClient, modelStructure, ms, dmuFactory, aggAcc); + + // calculate and get the array of worker size terms table - MGRAs by + // occupations + aggAcc.createWorkSegmentNameIndices(); + aggAcc.calculateWorkerSizeTerms(); + double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); + + // run the model + logger.info("Starting usual work location choice for logsum calculations."); + usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, workerSizeTerms); + logger.info("Finished with usual work location choice for logsum calculations."); + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after running school and work location choice"); + + } + + public void runHouseholdModels(){ + + logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before running non-work logsums"); + + HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( propertyMap, jppfClient, "False", householdDataManager, ms, modelStructure, dmuFactory ); + runner.runHouseholdChoiceModels(); + + } + + + + /** + * Create the household data manager. Based on the code in MTCTM2TourBasedModel.runTourBasedModel() + * @return The household data manager interface. + */ + public HouseholdDataManagerIf getHouseholdDataManager( ){ + + + boolean localHandlers = false; + + String testString; + + HouseholdDataManagerIf householdDataManager; + String hhHandlerAddress = ""; + int hhServerPort = 0; + try + { + // get household server address. if none is specified a local server in + // the current process will be started. + hhHandlerAddress = resourceBundle.getString("RunModel.HouseholdServerAddress"); + try + { + // get household server port. + hhServerPort = Integer.parseInt(resourceBundle.getString("RunModel.HouseholdServerPort")); + localHandlers = false; + } catch (MissingResourceException e) + { + // if no household data server address entry is found, the object + // will be created in the local process + localHandlers = true; + } + } catch (MissingResourceException e) + { + localHandlers = true; + } + + + try + { + + if (localHandlers) + { + + // create a new local instance of the household array manager + householdDataManager = new SandagHouseholdDataManager(); + householdDataManager.setPropertyFileValues(propertyMap); + + // have the household data manager read the synthetic population + // files and apply its tables to objects mapping method. + String inputHouseholdFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); + + } else + { + + householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, + SandagHouseholdDataManager.HH_DATA_SERVER_NAME); + testString = householdDataManager.testRemote(); + logger.info("HouseholdDataManager test: " + testString); + + householdDataManager.setPropertyFileValues(propertyMap); + } + + //always starting from scratch (RunModel.RestartWithHhServer=none) + householdDataManager.setDebugHhIdsFromHashmap(); + + String inputHouseholdFileName = resourceBundle + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); + String inputPersonFileName = resourceBundle + .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); + householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); + householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); + + }catch (Exception e) + { + + logger.error(String + .format("Exception caught setting up household data manager."), e); + throw new RuntimeException(); + + } + + return householdDataManager; + } + + + /** + * Start a new matrix server connection. + * + * @param properties + */ + private void startMatrixServer(HashMap properties) { + String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); + int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); + logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); + + try{ + + MatrixDataManager mdm = MatrixDataManager.getInstance(); + ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + ms.testRemote(Thread.currentThread().getName()); + mdm.setMatrixDataServerObject(ms); + + } catch (Exception e) { + logger.error("could not connect to matrix server", e); + throw new RuntimeException(e); + + } + + } + + + public static void main(String[] args) { + + long startTime = System.currentTimeMillis(); + int globalIterationNumber = -1; + float iterationSampleRate = -1.0f; + int sampleSeed = -1; + + ResourceBundle rb = null; + + logger.info( String.format( "Generating Logsums from MTC Tour Based Model using CT-RAMP version %s, 22feb2011 build %s", CtrampApplication.VERSION, 2 ) ); + + if ( args.length == 0 ) { + logger.error( String.format( "no properties file base name (without .properties extension) was specified as an argument." ) ); + return; + } + else { + rb = ResourceBundle.getBundle( args[0] ); + + // optional arguments + for (int i=1; i < args.length; i++) { + + if (args[i].equalsIgnoreCase("-iteration")) { + globalIterationNumber = Integer.parseInt( args[i+1] ); + logger.info( String.format( "-iteration %d.", globalIterationNumber ) ); + } + + if (args[i].equalsIgnoreCase("-sampleRate")) { + iterationSampleRate = Float.parseFloat( args[i+1] ); + logger.info( String.format( "-sampleRate %.4f.", iterationSampleRate ) ); + } + + if (args[i].equalsIgnoreCase("-sampleSeed")) { + sampleSeed = Integer.parseInt( args[i+1] ); + logger.info( String.format( "-sampleSeed %d.", sampleSeed ) ); + } + + } + + if ( globalIterationNumber < 0 ) { + globalIterationNumber = 1; + logger.info( String.format( "no -iteration flag, default value %d used.", globalIterationNumber ) ); + } + + if ( iterationSampleRate < 0 ) { + iterationSampleRate = 1; + logger.info( String.format( "no -sampleRate flag, default value %.4f used.", iterationSampleRate ) ); + } + + if ( sampleSeed < 0 ) { + sampleSeed = 0; + logger.info( String.format( "no -sampleSeed flag, default value %d used.", sampleSeed ) ); + } + + } + + + String baseName; + if ( args[0].endsWith(".properties") ) { + int index = args[0].indexOf(".properties"); + baseName = args[0].substring(0, index); + } + else { + baseName = args[0]; + } + + + // create an instance of this class for main() to use. + RunModeChoice mainObject = new RunModeChoice( args[0], globalIterationNumber, iterationSampleRate, sampleSeed ); + + // Create logsums + try { + + logger.info ("Creating logsums."); + mainObject.run(); + + } + catch ( RuntimeException e ) { + logger.error ( "RuntimeException caught in com.pb.mtctm2.abm.reports.CreateLogsums.main() -- exiting.", e ); + System.exit(2); + } + + + logger.info (""); + logger.info (""); + logger.info ("CreateLogsums finished in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); + + System.exit(0); + + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java new file mode 100644 index 0000000..f930cca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java @@ -0,0 +1,183 @@ +package org.sandag.abm.utilities; + +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; + +import com.linuxense.javadbf.DBFException; +import com.linuxense.javadbf.DBFReader; + +/** + * The TapAtConsistencyCheck program implements consistency checkng between TAPs in AT and Transit networks. + * There are three types of inconsistencies: + * 1) A TAP in AT network is not in Tranist network + * 2) A TAP in Transit network is not in AT network + * 3) Positions of a TAP are different in AT and Transit networks. + * + * @author wsu + * @mail Wu.Sun@sandag.org + * @version 13.3.0 + * @since 2016-06-20 + * + */ +public class TapAtConsistencyCheck { + private static Logger logger = Logger.getLogger(TapAtConsistencyCheck.class); + private HashMap > atMap; + private HashMap > tranMap; + private float xThreshold; + private float yThreshold; + private String message; + + public TapAtConsistencyCheck(ResourceBundle aRb, String folder){ + xThreshold=Float.parseFloat(aRb.getString("AtTransitConsistency.xThreshold")); + yThreshold=Float.parseFloat(aRb.getString("AtTransitConsistency.yThreshold")); + readAtTaps(folder); + logger.info("Finished reading TAPs in AT network."); + readTranTaps(folder); + logger.info("Finished reading TAPs in transit network."); + } + + public boolean validate(){ + boolean result=false; + message=compareMap(atMap,tranMap); + if(message.equalsIgnoreCase("OK")){ + result=true; + } + return result; + } + + + private void readAtTaps(String folder){ + atMap=new HashMap>(); + Object [] atTapObjects; + + try { + InputStream inputStream = new FileInputStream(folder+"\\SANDAG_Bike_Node.dbf"); + DBFReader atTapReader = new DBFReader( inputStream); + while( (atTapObjects = atTapReader.nextRecord()) != null) { + double tap_at = (double)atTapObjects[3]; + if(tap_at>0){ + ArrayList xy=new ArrayList(); + xy.add((float)atTapObjects[4]); + xy.add((float)atTapObjects[5]); + atMap.put((int)tap_at, xy); + } + } + inputStream.close(); + }catch( DBFException e) { + System.out.println( e.getMessage()); + logger.fatal(e.getMessage()); + System.exit(-1); + }catch( IOException e) { + System.out.println( e.getMessage()); + logger.fatal(e.getMessage()); + System.exit(-1); + } + } + + private void readTranTaps(String folder){ + tranMap=new HashMap>(); + Object [] tranTapObjects; + + try { + InputStream inputStream = new FileInputStream(folder+"\\tapcov.dbf"); + DBFReader tranTapReader = new DBFReader( inputStream); + while( (tranTapObjects = tranTapReader.nextRecord()) != null) { + double tap_tran = (double)tranTapObjects[16]; + ArrayList xy=new ArrayList(); + xy.add((double)tranTapObjects[7]); + xy.add((double)tranTapObjects[8]); + tranMap.put((int)tap_tran, xy); + } + inputStream.close(); + }catch( DBFException e) { + System.out.println( e.getMessage()); + logger.fatal(e.getMessage()); + System.exit(-1); + }catch( IOException e) { + System.out.println( e.getMessage()); + logger.fatal(e.getMessage()); + System.exit(-1); + } + } + +public String compareMap(HashMap> map1, HashMap> map2) { + + String message="OK"; + + if (map1.size()==0){ + message=new ErrorLogging().getAtError("AT4"); + logger.fatal(message); + return message; + } + + if (map2.size()==0){ + message=new ErrorLogging().getAtError("AT5"); + logger.fatal(message); + return message; + } + + for (Integer ch1 : map1.keySet()) { + Float x1=map1.get(ch1).get(0); + Float y1=map1.get(ch1).get(1); + + if(map2.get(ch1)==null||map2.get(ch1)==null){ + message=new ErrorLogging().getAtError("AT1")+"(in SANDAG_Bike_Node.dbf "+"TAP="+ch1+" x_at="+x1+" y_at="+y1+")"; + logger.fatal(message); + }else{ + Double x2=map2.get(ch1).get(0); + Double y2=map2.get(ch1).get(1); + if((Math.abs(x1-x2)>xThreshold)&&(Math.abs(y1-y2)>yThreshold)){ + message=new ErrorLogging().getAtError("AT3")+"("+"TAP="+ch1+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2+")"; + logger.fatal(message); + } + } + + } + + for (Integer ch2 : map2.keySet()) { + Double x2=map2.get(ch2).get(0); + Double y2=map2.get(ch2).get(1); + + if(map1.get(ch2)==null||map1.get(ch2)==null){ + message=new ErrorLogging().getAtError("AT2")+"(in tapcov.dbf "+"TAP="+ch2+" x_tran="+x2+" y_tran="+y2+")"; + logger.fatal(message); + }else{ + Float x1=map1.get(ch2).get(0); + Float y1=map1.get(ch2).get(1); + //System.out.println("TAP="+ch2+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2); + if((Math.abs(x1-x2)>xThreshold)&&(Math.abs(y1-y2)>yThreshold)){ + message=new ErrorLogging().getAtError("AT3")+"("+"TAP="+ch2+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2+")"; + logger.fatal(message); + } + } + } + + return message; +} + + public static void main(String[] args) + { + ResourceBundle rb = null; + logger.info("Checking AT and Transit Network Consistency..."); + + if (args.length == 0) + { + logger.error(String.format("no properties file base name (without .properties extension) was specified as an argument.")); + System.exit(-1); + } else + { + rb = ResourceBundle.getBundle(args[0]); + TapAtConsistencyCheck mainObject = new TapAtConsistencyCheck(rb, args[1]); + if(!mainObject.validate()){ + logger.fatal(mainObject.message); + System.exit(-1); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java b/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java new file mode 100644 index 0000000..f596b5c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java @@ -0,0 +1,437 @@ +package org.sandag.abm.validation; + +import java.io.*; +import java.nio.file.Files; +import org.apache.poi.xssf.usermodel.XSSFCell; +import org.apache.poi.xssf.usermodel.XSSFRow; +import org.apache.poi.xssf.usermodel.XSSFSheet; +import org.apache.poi.xssf.usermodel.XSSFWorkbook; +import java.sql.Connection; +import java.sql.DriverManager; +import java.sql.ResultSet; +import java.sql.ResultSetMetaData; +import java.sql.SQLException; +import java.sql.Statement; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.ResourceBundle; + +public class MainApplication { + + public static List getHighwayFlowData(int scenario, ResourceBundle rb) throws SQLException { + + System.out.println("Getting highway flow data for scenario " + scenario + " from database ..."); + + String query = "SELECT " + + " flow.scenario_id," + + " link.hwycov_id " + + " ,sum(CONVERT(bigint,(flow.flow + link_ab_tod.preload / 3.0))) as total_flow " + + "FROM " + + " abm_13_2_3.abm.hwy_flow flow " + + "JOIN abm_13_2_3.abm.hwy_link_ab_tod link_ab_tod " + + " ON flow.scenario_id = link_ab_tod.scenario_id AND flow.hwy_link_ab_tod_id = link_ab_tod.hwy_link_ab_tod_id " + + "JOIN abm_13_2_3.abm.hwy_link_tod link_tod " + + " ON link_ab_tod.scenario_id = link_tod.scenario_id AND link_ab_tod.hwy_link_tod_id = link_tod.hwy_link_tod_id " + + "JOIN abm_13_2_3.abm.hwy_link link " + + " ON link_tod.scenario_id = link.scenario_id AND link_tod.hwy_link_id = link.hwy_link_id " + + "JOIN ref.ifc ifc " + + " ON link.ifc = ifc.ifc " + + "WHERE " + + " flow.scenario_id = " + scenario + + "GROUP BY " + + " flow.scenario_id, link.hwycov_id " + + "ORDER BY " + + " hwycov_id"; + + System.out.println(""); + return getSqlData(query, rb); + } + + public static List getTransitBoardingByModeData(int scenario, ResourceBundle rb) throws SQLException { + + System.out.println("Getting transit boarding by mode data for scenario " + scenario + " from database ..."); + + String query = "SELECT transit_mode_desc,transit_access_mode_desc,sum(boardings) " + + "FROM ws.dbo.transitBoardingSummary(" + scenario + ") " + + "GROUP BY transit_mode_desc,transit_access_mode_desc " + + "ORDER BY transit_mode_desc,transit_access_mode_desc"; + System.out.println(""); + return getSqlData(query, rb); + } + + public static List getTransitBoardingByRouteData(int scenario, ResourceBundle rb) throws SQLException { + + System.out.println("Getting transit boarding by route data for scenario " + scenario + " from database ..."); + + String query = "SELECT rtt, sum(boardings) as boardings " + + "FROM " + + "(SELECT (config/1000) as rtt, boardings " + + "FROM ws.dbo.transitBoardingSummary(" + scenario + ")" + " AS boarding " + + "JOIN abm_13_2_3.abm.transit_route route " + + " ON boarding.route_id = route.transit_route_id " + + "WHERE scenario_id = " + scenario + ") as t " + + "GROUP BY rtt " + + "ORDER BY rtt"; + System.out.println(""); + return getSqlData(query, rb); + } + + public static List getSqlData(String query, ResourceBundle rb) throws SQLException { + + String dbHost = getProperty(rb, "database.host", null); + String dbName = getProperty(rb, "database.name", null); + String user = getProperty(rb, "database.user", null); + String password = getProperty(rb, "database.pwd", null); + + // System.out.println( dbHost ); + // System.out.println( user ); + // System.out.println( password ); + // System.out.println( dbName ); + + Connection conn = null; + conn = getConnectionToDatabase(dbHost, dbName, user, password); + + Statement stmt = conn.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); + ResultSet rs = stmt.executeQuery(query); + ResultSetMetaData rsmd = rs.getMetaData(); + + int nCols = rsmd.getColumnCount(); + + List data = new ArrayList(); + StringBuilder row = null; + + while(rs.next()){ + row = new StringBuilder(); + for(int c = 1; c <= nCols; c++){ + row.append(rs.getString(c)); + + if(c < nCols) + row.append("|"); + } + data.add(row.toString()); + } + + conn.close(); + + return data; + } + + public static void printData(List data){ + for(String str: data) + System.out.println(str); + } + + public static void printRow(String[] vals) { + for (String i : vals) { + System.out.print(i); + System.out.print("\t"); + } + System.out.println(); + } + + //Wu modified to allow writing out files to a different location other than input file location + public static void writeHighwayDataToExcel(String input_file_name, String sheet_to_modify, List data, int scenario, String outputDir) throws IOException { + try { + InputStream input_file = new FileInputStream(new File(input_file_name)); + XSSFWorkbook wb = new XSSFWorkbook(input_file); + input_file.close(); + + XSSFSheet ws = wb.getSheet(sheet_to_modify); + int prevRows = ws.getLastRowNum(); + + XSSFRow row = null; + XSSFCell cell = null; + + // writing data to excel sheet + int r = 1; + for (String str : data) { + row = ws.getRow(r); + + if(row == null){ + // this will happen when the number of links in the template sheet + // are less than the number of highway links for the data obtained from database + ws.createRow(r); + row = ws.getRow(r); + } + + //System.out.println(str); + + String[] vals = str.split("\\|"); + + for(int c = 0; c < vals.length; c++){ + cell = row.getCell(c); + + if(cell == null){ + row.createCell(c); + cell = row.getCell(c); + } + + cell.setCellValue(Integer.valueOf(vals[c])); + } + r++; + } + + // delete additional existing rows, if any + // this is to remove excel rows if the highway links in the template sheet + // are more than the number of highway links for the data obtained from database) + for(int d = data.size() + 1; d < prevRows; d++){ + row = ws.getRow(d); + ws.removeRow(row); + } + + //update all formula calculation + wb.getCreationHelper().createFormulaEvaluator().evaluateAll(); + wb.setForceFormulaRecalculation(true); + + // FormulaEvaluator evaluator = wb.getCreationHelper().createFormulaEvaluator(); + // + // for (int i = 0; i < wb.getNumberOfSheets(); i++) + // { + // Sheet sheet = wb.getSheetAt(i); + // System.out.println("Sheet name " + sheet.getSheetName()); + // for (Row ro : sheet) { + // if(sheet.getSheetName().equalsIgnoreCase("all_nb")){ + // System.out.println("row num " + ro.getRowNum()); + // } + // for (Cell c : ro) { + // if (c.getCellTypeEnum() == CellType.FORMULA) { + // if(sheet.getSheetName().equalsIgnoreCase("all_nb")){ + // System.out.println("col index " + c.getColumnIndex()); + // } + // try { + // evaluator.evaluateFormulaCellEnum(c); + // } catch (Exception e) { + // // TODO Auto-generated catch block + // e.printStackTrace(); + // } + // } + // } + // } + // } + + //forming output file name + String name = input_file_name.substring(0, input_file_name.lastIndexOf(".")); + String ext = input_file_name.substring(input_file_name.lastIndexOf(".") + 1); + String output_file_name = outputDir+name + "_s" + String.valueOf(scenario) + "." + ext; + + //writing out revised excel file + System.out.println("Writing highway data to excel file : " + output_file_name); + + File file = new File(output_file_name); + Files.deleteIfExists(file.toPath()); + + OutputStream output_file = new FileOutputStream(new File(output_file_name)); + wb.write(output_file); + wb.close(); + output_file.close(); + System.out.println(""); + + } catch (FileNotFoundException e) { + e.printStackTrace(); + } catch (IOException e) { + e.printStackTrace(); + } + } + + public static void writeBoardingDataToExcel(String input_file_name, String sheet_to_modify, List data, int scenario, String output_file_name) throws IOException { + try { + InputStream input_file = new FileInputStream(new File(input_file_name)); + XSSFWorkbook wb = new XSSFWorkbook(input_file); + input_file.close(); + + XSSFSheet ws = wb.getSheet(sheet_to_modify); + int prevRows = ws.getLastRowNum(); + + XSSFRow row = null; + XSSFCell cell = null; + + boolean[] colIsNumeric = null; + + // writing data to excel sheet + int r = 1; + for (String str : data) { + row = ws.getRow(r); + + if(row == null){ + // this will happen when the number of routes/modes in the template sheet + // are less than the number of routes/modes for the data obtained from database + ws.createRow(r); + row = ws.getRow(r); + } + + String[] vals = str.split("\\|"); + + //identify columns as string or int/double value + if(r == 1){ + colIsNumeric = new boolean[vals.length]; + Arrays.fill(colIsNumeric, Boolean.TRUE); + + for(int c = 0; c < vals.length; c++) + colIsNumeric[c] = isNumeric(vals[c]); + } + + for(int c = 0; c < vals.length; c++){ + + cell = row.getCell(c); + + if(cell == null){ + row.createCell(c); + cell = row.getCell(c); + } + if(colIsNumeric[c]) + cell.setCellValue(Double.valueOf(vals[c]).intValue()); + else + cell.setCellValue(vals[c]); + } + r++; + } + + // delete additional existing rows, if any + // this is to remove excel rows if the transit routes/modes in the template sheet + // are more than the number of transit routes/modes for the data obtained from database) + for(int d = data.size() + 1; d < prevRows; d++){ + row = ws.getRow(d); + ws.removeRow(row); + } + + //update all formula calculation + wb.getCreationHelper().createFormulaEvaluator().evaluateAll(); + wb.setForceFormulaRecalculation(true); + + //writing out revised excel file + System.out.println("Writing boarding data to excel file : " + output_file_name); + + File file = new File(output_file_name); + Files.deleteIfExists(file.toPath()); + + OutputStream output_file = new FileOutputStream(new File(output_file_name)); + wb.write(output_file); + output_file.close(); + System.out.println(""); + + } catch (FileNotFoundException e) { + e.printStackTrace(); + + } catch (IOException e) { + e.printStackTrace(); + } + } + + public static boolean isNumeric(String str) + { + try{ + double d = Double.parseDouble(str); + } + catch(NumberFormatException nfe) { + return false; + } + return true; + } + + public static String getProperty(ResourceBundle rb, String keyName, String defaultValue) { + + String keyValue = defaultValue; + + try { + keyValue = rb.getString(keyName); + } catch (RuntimeException e) { + //key was not found or resource bundle is null + if(rb == null) throw new RuntimeException("ResourceBundle is null", e); + } + + if(keyValue == null) return keyValue; //you can't trim a null. + + return keyValue.trim(); + } + + /** + * @param dbHost is database server name + * @param dbName is the name of the database we want to use in the server + * @param user is the name of the user used to login to access the database. Note for MS_SQL, user can be null, + * which indicates that standard MS Windows authentication will be used. In this case sqljdbc_auth.dll must be in the java library path. + * @param password is the user's password - not necessary if user is null. + * @return the connection instance for the URL formed for the specific database server specified + */ + + public static Connection getConnectionToDatabase(String dbHost, String dbName, String user, String password) { + Connection conn = null; + + try { + Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver"); + } catch (ClassNotFoundException e1) { + throw new RuntimeException("Class not found ", e1); + } + + try { + String urlToConnect = "jdbc:sqlserver://" + dbHost + ";" + "database=" + dbName + ";" + + (user != null ? ("user=" + user + ";" + "password=" + password) : "integratedSecurity=True"); + + //System.out.println("urlToConnect " + urlToConnect); + conn = DriverManager.getConnection(urlToConnect); + } + catch (SQLException e) { + throw new RuntimeException("Cannot connect to database ", e); + } + return conn; + } + + public static void main(String[] args) throws Exception { + long startTime = System.currentTimeMillis(); + + + // Collection supportedFuncs = WorkbookEvaluator.getSupportedFunctionNames(); + // System.out.println(supportedFuncs); + // + // Collection unsupportedFuncs = WorkbookEvaluator.getNotSupportedFunctionNames(); + // System.err.println(unsupportedFuncs); + // System.exit(-1); + + if ( args.length < 2 ) { + System.out.println( "invalid number of command line arguments." ); + System.out.println( "two argument must be specified, 1) basename of the properties file and 2) scenario number"); + System.exit(-1); + } + + ResourceBundle rb = ResourceBundle.getBundle( args[0] ); + + int scenario = Integer.valueOf(args[1]); + + //Wu modified to allow writing out files to a different location other than input file location + String outputDir=args[2]; + + String sheet_to_modify = null; + + List flow_data = getHighwayFlowData(scenario, rb); + + // revise summary by class workbook + String summary_by_class_file = rb.getString("highway.summary.class.template"); + sheet_to_modify = rb.getString("sheet.to.modify.highway.summary"); + writeHighwayDataToExcel(summary_by_class_file, sheet_to_modify, flow_data, scenario,outputDir); + + // revise summary by corridor workbook + String summary_by_corridor_file = rb.getString("highway.summary.corridor.template"); + sheet_to_modify = rb.getString("sheet.to.modify.highway.summary"); + writeHighwayDataToExcel(summary_by_corridor_file, sheet_to_modify, flow_data, scenario, outputDir); + + // revise transit validation workbook + String transit_validation_file = rb.getString("transit.validation.template"); + + //forming transit output file name + String name = transit_validation_file.substring(0, transit_validation_file.lastIndexOf(".")); + String ext = transit_validation_file.substring(transit_validation_file.lastIndexOf(".") + 1); + String output_file_name = name + "_s" + String.valueOf(scenario) + "." + ext; + String transit_output_file =outputDir+output_file_name; + + List boarding_by_route_data = getTransitBoardingByRouteData(scenario, rb); + sheet_to_modify = rb.getString("sheet.to.modify.transit.boarding.route"); + writeBoardingDataToExcel(transit_validation_file, sheet_to_modify, boarding_by_route_data, scenario, transit_output_file); + + List boarding_by_mode_data = getTransitBoardingByModeData(scenario, rb); + sheet_to_modify = rb.getString("sheet.to.modify.transit.boarding.mode"); + writeBoardingDataToExcel(transit_output_file, sheet_to_modify, boarding_by_mode_data, scenario, transit_output_file); + + System.out.println( String.format( "%s%.1f%s", "Total Run Time - ", ( ( System.currentTimeMillis() - startTime ) / 1000.0 ), " seconds." ) ); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java new file mode 100644 index 0000000..acf7f8c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java @@ -0,0 +1,55 @@ +/* + * Copyright 2005 PB Consult 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. + */ +package org.sandag.abm.visitor; + +import java.io.Serializable; + +/** + * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects + * + * @author Joel Freedman + */ +public class VisitorDmuFactory + implements VisitorDmuFactoryIf, Serializable +{ + + private VisitorModelStructure visitorModelStructure; + + public VisitorDmuFactory(VisitorModelStructure modelStructure) + { + this.visitorModelStructure = modelStructure; + } + + public VisitorTourModeChoiceDMU getVisitorTourModeChoiceDMU() + { + return new VisitorTourModeChoiceDMU(visitorModelStructure, null); + } + + public VisitorTourDestChoiceDMU getVisitorTourDestChoiceDMU() + { + return new VisitorTourDestChoiceDMU(visitorModelStructure); + } + + public VisitorStopLocationChoiceDMU getVisitorStopLocationChoiceDMU() + { + return new VisitorStopLocationChoiceDMU(visitorModelStructure); + } + + public VisitorTripModeChoiceDMU getVisitorTripModeChoiceDMU() + { + return new VisitorTripModeChoiceDMU(visitorModelStructure, null); + } + + public VisitorMicromobilityChoiceDMU getVisitorMicromobilityChoiceDMU() + { + return new VisitorMicromobilityChoiceDMU(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java new file mode 100644 index 0000000..d25e087 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java @@ -0,0 +1,18 @@ +package org.sandag.abm.visitor; + +/** + * A DMU factory interface + */ +public interface VisitorDmuFactoryIf +{ + + VisitorTourModeChoiceDMU getVisitorTourModeChoiceDMU(); + + VisitorTourDestChoiceDMU getVisitorTourDestChoiceDMU(); + + VisitorStopLocationChoiceDMU getVisitorStopLocationChoiceDMU(); + + VisitorTripModeChoiceDMU getVisitorTripModeChoiceDMU(); + + VisitorMicromobilityChoiceDMU getVisitorMicromobilityChoiceDMU(); +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java new file mode 100644 index 0000000..8634f41 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java @@ -0,0 +1,139 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +/** + */ +public class VisitorMicromobilityChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(VisitorMicromobilityChoiceDMU.class); + + protected HashMap methodIndexMap; + + private IndexValues dmuIndex; + protected int income; + protected float walkTime; + protected boolean isTransit; + protected boolean microTransitAvailable; + + + public VisitorMicromobilityChoiceDMU() + { + dmuIndex = new IndexValues(); + setupMethodIndexMap(); + + } + + public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) + { + dmuIndex.setHHIndex(hhId); + dmuIndex.setZoneIndex(zoneId); + dmuIndex.setOriginZone(origTaz); + dmuIndex.setDestZone(destTaz); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + } + + + public int getIncome() + { + return income; + } + + public void setIncome(int income) { + this.income = income; + } + + public float getWalkTime() { + return walkTime; + } + + public void setWalkTime(float walkTime) { + this.walkTime = walkTime; + } + + public boolean isTransit() { + return isTransit; + } + + public void setTransit(boolean isTransit) { + this.isTransit = isTransit; + } + + public boolean isMicroTransitAvailable() { + return microTransitAvailable; + } + + public void setMicroTransitAvailable(boolean microTransitAvailable) { + this.microTransitAvailable = microTransitAvailable; + } + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getIncome", 0); + methodIndexMap.put("getWalkTime", 1); + methodIndexMap.put("getIsTransit", 3); + methodIndexMap.put("getMicroTransitAvailable", 4); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + switch (variableIndex) + { + case 0: + return getIncome(); + case 1: + return getWalkTime(); + + case 3: + return isTransit()? 1 : 0; + case 4: + return isMicroTransitAvailable() ? 1 : 0; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java new file mode 100644 index 0000000..135bc8f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java @@ -0,0 +1,325 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.HashSet; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; + +public class VisitorMicromobilityChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("micromobility"); + + private static final String MM_CONTROL_FILE_TARGET = "visitor.micromobility.uec.file"; + private static final String MM_DATA_SHEET_TARGET = "visitor.micromobility.data.page"; + private static final String MM_MODEL_SHEET_TARGET = "visitor.micromobility.model.page"; + private static final String MT_TAP_FILE_TARGET = "active.microtransit.tap.file"; + private static final String MT_MAZ_FILE_TARGET = "active.microtransit.mgra.file"; + + public static final int MM_MODEL_WALK_ALT = 0; + public static final int MM_MODEL_MICROMOBILITY_ALT = 1; + public static final int MM_MODEL_MICROTRANSIT_ALT = 2; + + private ChoiceModelApplication mmModel; + private VisitorMicromobilityChoiceDMU mmDmuObject; + + private VisitorModelStructure modelStructure; + private MgraDataManager mgraDataManager; + + private HashSet microtransitTaps; + private HashSet microtransitMazs; + + public VisitorMicromobilityChoiceModel(HashMap propertyMap, + VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory) + { + + setupMicromobilityChoiceModelApplication(propertyMap, myModelStructure, dmuFactory); + } + + private void setupMicromobilityChoiceModelApplication(HashMap propertyMap, + VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory) + { + // logger.info("setting up micromobility choice model."); + + modelStructure = myModelStructure; + + // locate the micromobility choice UEC + String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mmUecFile = uecFileDirectory + propertyMap.get(MM_CONTROL_FILE_TARGET); + + int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_DATA_SHEET_TARGET); + int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_MODEL_SHEET_TARGET); + + // create the micromobility choice model DMU object. + mmDmuObject = dmuFactory.getVisitorMicromobilityChoiceDMU(); + + // create the transponder choice model object + mmModel = new ChoiceModelApplication(mmUecFile, modelSheet, dataSheet, propertyMap, + (VariableTable) mmDmuObject); + + mgraDataManager = MgraDataManager.getInstance(); + String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); + String microTransitTapFile = projectDirectory + propertyMap.get(MT_TAP_FILE_TARGET); + String microTransitMazFile = projectDirectory + propertyMap.get(MT_MAZ_FILE_TARGET); + + TableDataSet microTransitTapData = Util.readTableDataSet(microTransitTapFile); + TableDataSet microTransitMazData = Util.readTableDataSet(microTransitMazFile); + + microtransitTaps = new HashSet(); + microtransitMazs = new HashSet(); + + for(int i=1;i<=microTransitTapData.getRowCount();++i) { + + int tap = (int) microTransitTapData.getValueAt(i,"TAP"); + microtransitTaps.add(tap); + } + + for(int i=1;i<=microTransitMazData.getRowCount();++i) { + + int maz = (int) microTransitMazData.getValueAt(i,"MGRA"); + microtransitMazs.add(maz); + } + + } + + + public void applyModel(VisitorTour tour) { + + //apply to trips on tour + if(tour.getTrips()!=null) { + + for(VisitorTrip trip: tour.getTrips()) + applyModel(tour, trip); + } + + + } + + public void applyModel(VisitorTour tour, VisitorTrip trip) + { + + + if(!modelStructure.getTourModeIsWalk(trip.getTripMode()) && !modelStructure.getTourModeIsWalkTransit(trip.getTripMode())&& !modelStructure.getTourModeIsDriveTransit(trip.getTripMode())) + return; + + mmDmuObject.setIncome(tour.getIncome()); + int originMaz = trip.getOriginMgra(); + int destMaz = trip.getDestinationMgra(); + if(modelStructure.getTourModeIsWalk(trip.getTripMode())) + mmDmuObject.setTransit(false); + else + mmDmuObject.setTransit(true); + + + if(modelStructure.getTourModeIsWalk(trip.getTripMode())) { + + float walkTime = mgraDataManager.getMgraToMgraWalkTime(originMaz, destMaz); + mmDmuObject.setWalkTime(walkTime); + + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); + + if(microtransitMazs.contains(originMaz) && microtransitMazs.contains(destMaz)) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + trip.setMicromobilityWalkLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice(tour, trip); + trip.setMicromobilityWalkMode(chosenAlt); + + // write choice model alternative info to log file + if (tour.getDebugChoiceModels()) + { + String decisionMaker = String.format("Tour "+tour.getID()+ " mode " +trip.getTripMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Transponder Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + + + }else if(modelStructure.getTourModeIsWalkTransit(trip.getTripMode())) { + + //access + int tapPosition = mgraDataManager.getTapPosition(originMaz, trip.getBoardTap()); + float walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); + mmDmuObject.setWalkTime(walkTime); + + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); + + if(microtransitTaps.contains(trip.getBoardTap())) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + trip.setMicromobilityAccessLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice( tour, trip); + trip.setMicromobilityAccessMode(chosenAlt); + + // write choice model alternative info to log file + if (tour.getDebugChoiceModels()) + { + String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Transponder Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + //egress + tapPosition = mgraDataManager.getTapPosition(destMaz, trip.getAlightTap()); + walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); + mmDmuObject.setWalkTime(walkTime); + + if(microtransitTaps.contains(trip.getAlightTap())) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC + int closestMazToAlightTap = mgraDataManager.getClosestMgra(trip.getAlightTap()); + mmDmuObject.setDmuIndexValues(tour.getID(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); + + // compute utilities and choose micromobility choice alternative. + logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + trip.setMicromobilityEgressLogsum(logsum); + + // if the choice model has at least one available alternative, make choice + chosenAlt = (byte) getChoice( tour, trip); + trip.setMicromobilityEgressMode(chosenAlt); + + // write choice model alternative info to log file + if (tour.getDebugChoiceModels()) + { + String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Transponder Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + } else if( modelStructure.getTourModeIsDriveTransit(trip.getTripMode()) ) { //drive-transit. Choose non-drive direction + + int tapPosition = 0; + float walkTime = 9999; + + if(trip.isInbound()) { //inbound, so access mode is walk + tapPosition = mgraDataManager.getTapPosition(originMaz, trip.getBoardTap()); + walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); + //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC + mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); + + if(microtransitTaps.contains(trip.getBoardTap())) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + }else { //outbound so egress mode is walk. + tapPosition = mgraDataManager.getTapPosition(destMaz, trip.getAlightTap()); + walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); + //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC + int closestMazToAlightTap = mgraDataManager.getClosestMgra(trip.getAlightTap()); + mmDmuObject.setDmuIndexValues(tour.getID(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); + } + mmDmuObject.setWalkTime(walkTime); + + if(microtransitTaps.contains(trip.getAlightTap())) + mmDmuObject.setMicroTransitAvailable(true); + else + mmDmuObject.setMicroTransitAvailable(false); + + // compute utilities and choose micromobility choice alternative. + float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); + + // if the choice model has at least one available alternative, make choice + byte chosenAlt = (byte) getChoice(tour, trip); + + if(trip.isInbound()) { //inbound, set access + trip.setMicromobilityAccessMode(chosenAlt); + trip.setMicromobilityAccessLogsum(logsum); + }else { //outound, set egress + trip.setMicromobilityEgressMode(chosenAlt); + trip.setMicromobilityEgressLogsum(logsum); + } + + // write choice model alternative info to log file + if (tour.getDebugChoiceModels()) + { + String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); + mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); + logger.info(String.format("%s result chosen for %s is %d", + "Transponder Choice", decisionMaker, chosenAlt)); + mmModel.logUECResults(logger, decisionMaker); + } + + } + + } + + + /** + * Select the micromobility mode from the UEC. This is helper code for applyModel(), where utilities have already been calculated. + * + * @param household + * @param person + * @param tour + * @param s + * @return The micromobility mode. + */ + private int getChoice(VisitorTour tour, VisitorTrip trip) { + // if the choice model has at least one available alternative, make + // choice. + int chosenAlt; + if (mmModel.getAvailabilityCount() > 0) + { + double randomNumber = tour.getRandom(); + chosenAlt = mmModel.getChoiceResult(randomNumber); + return chosenAlt; + } else + { + String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); + String errorMessage = String + .format("Exception caught for %s, no available micromobility choice alternatives to choose from in choiceModelApplication.", + decisionMaker); + logger.info(errorMessage); + + mmModel.logUECResults(logger, decisionMaker); + return MM_MODEL_WALK_ALT; + } + + } + + /** + * This method calculates a cost coefficient based on the following formula: + * + * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) + * + * + * @param incomeCoeff + * @param incomeExponent + * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice + */ + public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ + + return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java new file mode 100644 index 0000000..a45b6f5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java @@ -0,0 +1,499 @@ +package org.sandag.abm.visitor; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.concurrent.ForkJoinPool; +import java.util.concurrent.atomic.AtomicBoolean; +import java.util.concurrent.atomic.AtomicInteger; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.crossborder.CrossBorderTripModeChoiceModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; +import com.pb.sawdust.util.concurrent.DnCRecursiveAction; + + +public class VisitorModel +{ + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + public static final String RUN_MODEL_CONCURRENT_PROPERTY_KEY = "visitor.run.concurrent"; + public static final String CONCURRENT_PARALLELISM_PROPERTY_KEY = "visitor.concurrent.parallelism"; + + private MatrixDataServerRmi ms; + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + private static final Object INITIALIZATION_LOCK = new Object(); + + private HashMap rbMap; + private McLogsumsCalculator logsumsCalculator; + private AutoTazSkimsCalculator tazDistanceCalculator; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + + private boolean seek; + private int traceId; + private double sampleRate = 1; + private static int iteration=1; + + /** + * Constructor + * + * @param rbMap + */ + public VisitorModel(HashMap rbMap) + { + this.rbMap = rbMap; + synchronized (INITIALIZATION_LOCK) + { // lock to make sure only one of + // these actually initializes + // things so we don't cross + // threads + mgraManager = MgraDataManager.getInstance(rbMap); + tazManager = TazDataManager.getInstance(rbMap); + } + + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "visitor.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "visitor.trace")); + + } + + // global variable used for reporting + private static final AtomicInteger TOUR_COUNTER = new AtomicInteger(0); + private final AtomicBoolean calculatorsInitialized = new AtomicBoolean(false); + + /** + * Run visitor model. + */ + private void runModel(VisitorTour[] tours, int start, int end){ + + + VisitorModelStructure modelStructure = new VisitorModelStructure(); + + VisitorDmuFactoryIf dmuFactory = new VisitorDmuFactory(modelStructure); + + if (!calculatorsInitialized.get()) + { + // only let one thread in to initialize + synchronized (calculatorsInitialized) + { + // if still not initialized, then this is the first in so do the + // initialization (otherwise skip) + if (!calculatorsInitialized.get()) + { + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + calculatorsInitialized.set(true); + } + } + } + + + VisitorTourTimeOfDayChoiceModel todChoiceModel = new VisitorTourTimeOfDayChoiceModel(rbMap); + VisitorTourDestChoiceModel destChoiceModel = new VisitorTourDestChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + VisitorTourModeChoiceModel tourModeChoiceModel = destChoiceModel.getTourModeChoiceModel(); + //VisitorTripModeChoiceModel tripModeChoiceModel = tourModeChoiceModel.getTripModeChoiceModel(); + destChoiceModel.calculateSizeTerms(dmuFactory); + destChoiceModel.calculateTazProbabilities(dmuFactory); + + VisitorStopFrequencyModel stopFrequencyModel = new VisitorStopFrequencyModel(rbMap); + VisitorStopPurposeModel stopPurposeModel = new VisitorStopPurposeModel(rbMap); + VisitorStopTimeOfDayChoiceModel stopTodChoiceModel = new VisitorStopTimeOfDayChoiceModel(rbMap); + VisitorStopLocationChoiceModel stopLocationChoiceModel = new VisitorStopLocationChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + VisitorTripModeChoiceModel tripModeChoiceModel = new VisitorTripModeChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + VisitorMicromobilityChoiceModel micromobilityChoiceModel = new VisitorMicromobilityChoiceModel(rbMap,modelStructure, dmuFactory); + + double[][] mgraSizeTerms = destChoiceModel.getMgraSizeTerms(); + double[][] tazSizeTerms = destChoiceModel.getTazSizeTerms(); + double[][][] mgraProbabilities = destChoiceModel.getMgraProbabilities(); + stopLocationChoiceModel.setMgraSizeTerms(mgraSizeTerms); + stopLocationChoiceModel.setTazSizeTerms(tazSizeTerms); + stopLocationChoiceModel.setMgraProbabilities(mgraProbabilities); + stopLocationChoiceModel.setTripModeChoiceModel(tripModeChoiceModel); + + // Run models for array of tours + for (int i = start; i < end; i++) + { + VisitorTour tour = tours[i]; + + // sample tours + double rand = tour.getRandom(); + if (rand > sampleRate) continue; + + int tourCount = TOUR_COUNTER.incrementAndGet(); + if (tourCount % 1000 == 0) logger.info("Processing tour " + tourCount); + + if (seek && tour.getID() != traceId) continue; + + if (tour.getID() == traceId) + tour.setDebugChoiceModels(true); + + + todChoiceModel.calculateTourTOD(tour); + destChoiceModel.chooseDestination(tour); + tourModeChoiceModel.chooseTourMode(tour); + + stopFrequencyModel.calculateStopFrequency(tour); + stopPurposeModel.calculateStopPurposes(tour); + + int outboundStops = tour.getNumberOutboundStops(); + int inboundStops = tour.getNumberInboundStops(); + + // choose TOD for stops and location of each + if (outboundStops > 0) + { + VisitorStop[] stops = tour.getOutboundStops(); + for (VisitorStop stop : stops) + { + stopTodChoiceModel.chooseTOD(tour, stop); + stopLocationChoiceModel.chooseStopLocation(tour, stop); + } + } + if (inboundStops > 0) + { + VisitorStop[] stops = tour.getInboundStops(); + for (VisitorStop stop : stops) + { + stopTodChoiceModel.chooseTOD(tour, stop); + stopLocationChoiceModel.chooseStopLocation(tour, stop); + } + } + + // generate trips and choose mode for them + VisitorTrip[] trips = new VisitorTrip[outboundStops + inboundStops + 2]; + int tripNumber = 0; + + // outbound stops + if (outboundStops > 0) + { + VisitorStop[] stops = tour.getOutboundStops(); + for (VisitorStop stop : stops) + { + // generate a trip to the stop and choose a mode for it + trips[tripNumber] = new VisitorTrip(tour, stop, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + // generate a trip from the last stop to the tour destination + trips[tripNumber] = new VisitorTrip(tour, stops[stops.length - 1], false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + + } else + { + // generate an outbound trip from the tour origin to the + // destination and choose a mode + trips[tripNumber] = new VisitorTrip(tour, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + + // inbound stops + if (inboundStops > 0) + { + VisitorStop[] stops = tour.getInboundStops(); + for (VisitorStop stop : stops) + { + // generate a trip to the stop and choose a mode for it + trips[tripNumber] = new VisitorTrip(tour, stop, true); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + // generate a trip from the last stop to the tour origin + trips[tripNumber] = new VisitorTrip(tour, stops[stops.length - 1], false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } else + { + + // generate an inbound trip from the tour destination to the + // origin and choose a mode + trips[tripNumber] = new VisitorTrip(tour, false); + tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); + ++tripNumber; + } + + // set the trips in the tour object + tour.setTrips(trips); + micromobilityChoiceModel.applyModel(tour); + + } + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @return the sampleRate + */ + public double getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(double sampleRate) + { + this.sampleRate = sampleRate; + } + + /** + * This class is the divide-and-conquer action (void return task) for + * running the visitor model using the fork-join framework. The + * divisible problem is an array of tours, and the actual work is the + * {@link VisitorModel#runModel(VisitorTour[],int,int)} method, + * applied to a section of the array. + */ + private class VisitorModelAction + extends DnCRecursiveAction + { + private final HashMap rbMap; + private final VisitorTour[] tours; + + private VisitorModelAction(HashMap rbMap, VisitorTour[] tours) + { + super(0, tours.length); + this.rbMap = rbMap; + this.tours = tours; + } + + private VisitorModelAction(HashMap rbMap, VisitorTour[] tours, + long start, long length, DnCRecursiveAction next) + { + super(start, length, next); + this.rbMap = rbMap; + this.tours = tours; + } + + @Override + protected void computeAction(long start, long length) + { + runModel(tours, (int) start, (int) (start + length)); + } + + @Override + protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) + { + return new VisitorModelAction(rbMap, tours, start, length, next); + } + + @Override + protected boolean continueDividing(long length) + { + // if there are 3 extra tasks queued up, then start executing + // if there are 1000 or less tours to process, then start executing + // otherwise, keep dividing to build up tasks for the threads to + // process + return getSurplusQueuedTaskCount() < 3 && length > 5000; + } + } + + /** + * Run visitor model. + */ + public void runModel() + { + VisitorTourManager tourManager = new VisitorTourManager(rbMap); + tourManager.generateVisitorTours(); + VisitorTour[] tours = tourManager.getTours(); + + // get new keys to see if we want to run in concurrent mode, and the + // parallelism + // (defaults to single threaded and parallelism = # of processors) + // note that concurrent can use up memory very quickly, so setting the + // parallelism might be prudent + boolean concurrent = rbMap.containsKey(RUN_MODEL_CONCURRENT_PROPERTY_KEY) + && Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, + RUN_MODEL_CONCURRENT_PROPERTY_KEY)); + int parallelism = rbMap.containsKey(CONCURRENT_PARALLELISM_PROPERTY_KEY) ? Integer + .valueOf(Util.getStringValueFromPropertyMap(rbMap, + CONCURRENT_PARALLELISM_PROPERTY_KEY)) : Runtime.getRuntime() + .availableProcessors(); + + if (concurrent) + { // use fork-join + VisitorModelAction action = new VisitorModelAction(rbMap, tours); + new ForkJoinPool(parallelism).execute(action); + action.getResult(); // wait for finish + } else + { // single-threaded: call the model runner in this thread + runModel(tours, 0, tours.length); + } + + tourManager.writeOutputFile(rbMap); + logger.info("Visitor Model successfully completed!"); + } + + /** + * @param args + */ + public static void main(String[] args) + { + Runtime gfg = Runtime.getRuntime(); + long memory1; + // checking the total memeory + System.out.println("Total memory is: "+ gfg.totalMemory()); + // checking free memory + memory1 = gfg.freeMemory(); + System.out.println("Initial free memory at Visitor model: "+ memory1); + // calling the garbage collector on demand + gfg.gc(); + memory1 = gfg.freeMemory(); + System.out.println("Free memory after garbage "+ "collection: " + memory1); + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Visitor Model")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + VisitorModel visitorModel = new VisitorModel(pMap); + + float sampleRate = 1.0f; + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + logger.info("Visitor Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + visitorModel.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = visitorModel.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + visitorModel.ms = matrixServer; + } else + { + visitorModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + visitorModel.ms.testRemote("VisitorModel"); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(visitorModel.ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + visitorModel.runModel(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java new file mode 100644 index 0000000..920f59a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java @@ -0,0 +1,95 @@ +package org.sandag.abm.visitor; + +import org.sandag.abm.application.SandagModelStructure; + +public class VisitorModelStructure + extends SandagModelStructure +{ + + public static final byte NUMBER_VISITOR_PURPOSES = 6; + public static final byte WORK = 0; + public static final byte RECREATION = 1; + public static final byte DINING = 2; + + public static final String[] VISITOR_PURPOSES = {"WORK", "RECREATE", "DINING"}; + + // override on max tour mode, since we have taxi in this model. + public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 13; + + public static final byte NUMBER_VISITOR_SEGMENTS = 2; + public static final byte BUSINESS = 0; + public static final byte PERSONAL = 1; + + public static final String[] VISITOR_SEGMENTS = {"BUSINESS", "PERSONAL"}; + public static final byte DEPARTURE = 0; + public static final byte ARRIVAL = 1; + + public static final byte INCOME_SEGMENTS = 5; + + // note that time periods start at 1 and go to 40 + public static final byte TIME_PERIODS = 40; + + public static final int AM = 0; + public static final int PM = 1; + public static final int OP = 2; + public static final int[] SKIM_PERIODS = {AM, PM, OP}; + public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; + public static final int UPPER_EA = 3; + public static final int UPPER_AM = 9; + public static final int UPPER_MD = 22; + public static final int UPPER_PM = 29; + public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; + + public static final byte TAXI = 13; + + /** + * Taxi tour mode + * + * @param tourMode + * @return + */ + public boolean getTourModeIsTaxi(int tourMode) + { + + if (tourMode == TAXI) return true; + else return false; + + } + + /** + * return the Skim period index 0=am, 1=pm, 2=off-peak + */ + public static int getSkimPeriodIndex(int departPeriod) + { + + int skimPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; + else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; + else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; + else skimPeriodIndex = OP; + + return skimPeriodIndex; + + } + + /** + * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV + */ + public static int getModelPeriodIndex(int departPeriod) + { + + int modelPeriodIndex = 0; + + if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; + else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; + else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; + else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; + else modelPeriodIndex = 4; + + return modelPeriodIndex; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java new file mode 100644 index 0000000..86e0499 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java @@ -0,0 +1,176 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import org.apache.log4j.Logger; + +public class VisitorStop + implements Serializable +{ + + private int id; + private int mode; + private int period; + private boolean inbound; + private int mgra; + private byte purpose; + + VisitorTour parentTour; + + public VisitorStop(VisitorTour parentTour, int id, boolean inbound) + { + this.parentTour = parentTour; + this.id = id; + this.inbound = inbound; + } + + /** + * @return the mgra + */ + public int getMgra() + { + return mgra; + } + + /** + * @param mgra + * the mgra to set + */ + public void setMgra(int mgra) + { + this.mgra = mgra; + } + + public void setMode(int mode) + { + this.mode = mode; + } + + public void setPeriod(int period) + { + this.period = period; + } + + /** + * @return the id + */ + public int getId() + { + return id; + } + + /** + * @param id + * the id to set + */ + public void setId(int id) + { + this.id = id; + } + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the parentTour + */ + public VisitorTour getParentTour() + { + return parentTour; + } + + /** + * @param parentTour + * the parentTour to set + */ + public void setParentTour(VisitorTour parentTour) + { + this.parentTour = parentTour; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(byte stopPurposeIndex) + { + this.purpose = stopPurposeIndex; + } + + public byte getPurpose() + { + return purpose; + } + + public int getMode() + { + return mode; + } + + public int getStopPeriod() + { + return period; + } + + public VisitorTour getTour() + { + return parentTour; + } + + public int getStopId() + { + return id; + } + + public void logStopObject(Logger logger, int totalChars) + { + + String separater = ""; + for (int i = 0; i < totalChars; i++) + separater += "-"; + + String purposeString = VisitorModelStructure.VISITOR_PURPOSES[purpose]; + logHelper(logger, "stopId: ", id, totalChars); + logHelper(logger, "mgra: ", mgra, totalChars); + logHelper(logger, "mode: ", mode, totalChars); + logHelper(logger, "purpose: ", purposeString, totalChars); + logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); + logHelper(logger, inbound ? "outbound departPeriod: " : "inbound arrivePeriod: ", period, + totalChars); + logger.info(separater); + logger.info(""); + logger.info(""); + + } + + public static void logHelper(Logger logger, String label, int value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } + + public static void logHelper(Logger logger, String label, String value, int totalChars) + { + int labelChars = label.length() + 2; + int remainingChars = totalChars - labelChars - 4; + String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); + String logString = String.format(formatString, label, value); + logger.info(logString); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java new file mode 100644 index 0000000..ed33801 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java @@ -0,0 +1,325 @@ +package org.sandag.abm.visitor; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the stop frequency model for visitor tours. It is currently + * based on a static probability distribution stored in an input file, and + * indexed into by tour purpose and duration. + * + * @author Freedman + * + */ +public class VisitorStopFrequencyModel +{ + private transient Logger logger = Logger.getLogger("visitorModel"); + + private double[][] cumProbability; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + private int[][] lowerBoundDurationHours; // by + // purpose, + // alternative: + // lower + // bound + // in + // hours + private int[][] upperBoundDurationHours; // by + // purpose, + // alternative: + // upper + // bound + // in + // hours + private int[][] outboundStops; // by + // purpose, + // alternative: + // number + // of + // outbound + // stops + private int[][] inboundStops; // by + // purpose, + // alternative: + // number + // of + // inbound + // stops + VisitorModelStructure modelStructure; + + /** + * Constructor. + */ + public VisitorStopFrequencyModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.stop.frequency.file"); + stopFrequencyFile = directory + stopFrequencyFile; + + modelStructure = new VisitorModelStructure(); + + readStopFrequencyFile(stopFrequencyFile); + + } + + /** + * Read the stop frequency distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readStopFrequencyFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating stop frequency probability distribution"); + + int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 + + int[] alts = new int[purposes]; + + // take a pass through the data and see how many alternatives there are + // for each purpose + int rowCount = probabilityTable.getRowCount(); + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + ++alts[purpose]; + } + + // initialize all the arrays + cumProbability = new double[purposes][]; + lowerBoundDurationHours = new int[purposes][]; + upperBoundDurationHours = new int[purposes][]; + outboundStops = new int[purposes][]; + inboundStops = new int[purposes][]; + + for (int i = 0; i < purposes; ++i) + { + cumProbability[i] = new double[alts[i]]; + lowerBoundDurationHours[i] = new int[alts[i]]; + upperBoundDurationHours[i] = new int[alts[i]]; + outboundStops[i] = new int[alts[i]]; + inboundStops[i] = new int[alts[i]]; + } + + // fill up arrays + int lastPurpose = 0; + int lastLowerBound = 0; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + int lowerBound = (int) probabilityTable.getValueAt(row, "DurationLo"); + int upperBound = (int) probabilityTable.getValueAt(row, "DurationHi"); + int outStops = (int) probabilityTable.getValueAt(row, "Outbound"); + int inbStops = (int) probabilityTable.getValueAt(row, "Inbound"); + + // reset cumulative probability if new purpose or lower-bound + if (purpose != lastPurpose || lowerBound != lastLowerBound) + { + + // log cumulative probability just in case + logger.info("Cumulative probability for purpose " + purpose + " lower bound " + + lowerBound + " is " + cumProb); + cumProb = 0; + } + + if (purpose != lastPurpose) alt = 0; + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + lowerBoundDurationHours[purpose][alt] = lowerBound; + upperBoundDurationHours[purpose][alt] = upperBound; + outboundStops[purpose][alt] = outStops; + inboundStops[purpose][alt] = inbStops; + + ++alt; + + lastPurpose = purpose; + lastLowerBound = lowerBound; + } + + logger.info("End calculating stop frequency probability distribution"); + + for (int purp = 0; purp < purposes; ++purp) + { + for (int a = 0; a < cumProbability[purp].length; ++a) + { + logger.info("Purpose " + purp + " lower " + lowerBoundDurationHours[purp][a] + + " upper " + upperBoundDurationHours[purp][a] + " cumProb " + + cumProbability[purp][a]); + } + } + + } + + /** + * Calculate number of stops for the tour. + * + * @param tour + * A visitor tour (with purpose and mode chosen) + */ + public void calculateStopFrequency(VisitorTour tour) + { + + int purpose = tour.getPurpose(); + double random = tour.getRandom(); + + int tourMode = tour.getTourMode(); + + if (!modelStructure.getTourModeIsSovOrHov(tourMode) + && !modelStructure.getTourModeIsTaxi(tourMode)) return; + + if (tour.getDebugChoiceModels()) + { + logger.info("Choosing stop frequency for purpose " + + modelStructure.VISITOR_PURPOSES[purpose] + " using random number " + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + if (!tourIsInRange(tour, lowerBoundDurationHours[purpose][i], + upperBoundDurationHours[purpose][i])) continue; + + if (tour.getDebugChoiceModels()) + { + logger.info("lower bound " + lowerBoundDurationHours[purpose][i] + " upper bound " + + upperBoundDurationHours[purpose][i]); + } + + if (random < cumProbability[purpose][i]) + { + int outStops = outboundStops[purpose][i]; + int inbStops = inboundStops[purpose][i]; + + if (outStops > 0) + { + VisitorStop[] stops = generateOutboundStops(tour, outStops); + tour.setOutboundStops(stops); + } + + if (inbStops > 0) + { + VisitorStop[] stops = generateInboundStops(tour, inbStops); + tour.setInboundStops(stops); + } + if (tour.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose " + outStops + " outbound stops and " + inbStops + + " inbound stops"); + logger.info(""); + } + break; + } + } + + } + + /** + * Check if the tour duration is in range + * + * @param tour + * @param lowerBound + * @param upperBound + * @return True if tour duration is greater than or equal to lower and + */ + private boolean tourIsInRange(VisitorTour tour, int lowerBound, int upperBound) + { + + float depart = (float) tour.getDepartTime(); + float arrive = (float) tour.getArriveTime(); + + float halfHours = arrive + 1 - depart; // at least 30 minutes + float tourDurationInHours = halfHours * (float) 0.5; + + if ((tourDurationInHours >= (float) lowerBound) + && (tourDurationInHours <= (float) upperBound)) return true; + + return false; + } + + /** + * Generate an array of outbound stops, from tour origin to primary + * destination, in order. + * + * @param tour + * The parent tour. + * @param numberOfStops + * Number of stops from stop frequency model. + * @return The array of outbound stops. + */ + private VisitorStop[] generateOutboundStops(VisitorTour tour, int numberOfStops) + { + + VisitorStop[] stops = new VisitorStop[numberOfStops]; + + for (int i = 0; i < stops.length; ++i) + { + VisitorStop stop = new VisitorStop(tour, i, false); + stops[i] = stop; + stop.setInbound(false); + stop.setParentTour(tour); + } + + return stops; + } + + /** + * Generate an array of inbound stops, from primary dest back to tour + * origin, in order. + * + * @param tour + * Parent tour. + * @param numberOfStops + * Number of stops from stop frequency model. + * @return The array of inbound stops. + */ + private VisitorStop[] generateInboundStops(VisitorTour tour, int numberOfStops) + { + + VisitorStop[] stops = new VisitorStop[numberOfStops]; + + for (int i = 0; i < stops.length; ++i) + { + VisitorStop stop = new VisitorStop(tour, i, true); + stops[i] = stop; + stop.setInbound(true); + stop.setParentTour(tour); + + } + + return stops; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java new file mode 100644 index 0000000..8fabb9c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java @@ -0,0 +1,406 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class VisitorStopLocationChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("visitorModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected int purpose; + protected int stopsOnHalfTour; + protected int stopNumber; + protected int inboundStop; + protected int tourDuration; + + protected double[][] sizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgra) + protected double[] correctionFactors; // by + // alternative + // (sampled + // mgra, + // for + // full + // model + // only) + + protected int[] sampleNumber; // by + // alternative + // (taz + // or + // sampled + // mgra) + + protected double[] osMcLogsumAlt; + protected double[] sdMcLogsumAlt; + + protected double[] tourOrigToStopDistanceAlt; + protected double[] stopToTourDestDistanceAlt; + + public VisitorStopLocationChoiceDMU(VisitorModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + /** + * @return the stopsOnHalfTour + */ + public int getStopsOnHalfTour() + { + return stopsOnHalfTour; + } + + /** + * @param stopsOnHalfTour + * the stopsOnHalfTour to set + */ + public void setStopsOnHalfTour(int stopsOnHalfTour) + { + this.stopsOnHalfTour = stopsOnHalfTour; + } + + /** + * @return the stopNumber + */ + public int getStopNumber() + { + return stopNumber; + } + + /** + * @param stopNumber + * the stopNumber to set + */ + public void setStopNumber(int stopNumber) + { + this.stopNumber = stopNumber; + } + + /** + * @return the inboundStop + */ + public int getInboundStop() + { + return inboundStop; + } + + /** + * @param inboundStop + * the inboundStop to set + */ + public void setInboundStop(int inboundStop) + { + this.inboundStop = inboundStop; + } + + /** + * @return the tourDuration + */ + public int getTourDuration() + { + return tourDuration; + } + + /** + * @param tourDuration + * the tourDuration to set + */ + public void setTourDuration(int tourDuration) + { + this.tourDuration = tourDuration; + } + + /** + * @return the sampleNumber + */ + public int getSampleNumber(int alt) + { + return sampleNumber[alt]; + } + + /** + * @param sampleNumber + * the sampleNumber to set + */ + public void setSampleNumber(int[] sampleNumber) + { + this.sampleNumber = sampleNumber; + } + + /** + * @return the osMcLogsumAlt + */ + public double getOsMcLogsumAlt(int alt) + { + return osMcLogsumAlt[alt]; + } + + /** + * @param osMcLogsumAlt + * the osMcLogsumAlt to set + */ + public void setOsMcLogsumAlt(double[] osMcLogsumAlt) + { + this.osMcLogsumAlt = osMcLogsumAlt; + } + + /** + * @return the sdMcLogsumAlt + */ + public double getSdMcLogsumAlt(int alt) + { + return sdMcLogsumAlt[alt]; + } + + /** + * @param sdMcLogsumAlt + * the sdMcLogsumAlt to set + */ + public void setSdMcLogsumAlt(double[] sdMcLogsumAlt) + { + this.sdMcLogsumAlt = sdMcLogsumAlt; + } + + /** + * @return the tourOrigToStopDistanceAlt + */ + public double getTourOrigToStopDistanceAlt(int alt) + { + return tourOrigToStopDistanceAlt[alt]; + } + + /** + * @param tourOrigToStopDistanceAlt + * the tourOrigToStopDistanceAlt to set + */ + public void setTourOrigToStopDistanceAlt(double[] tourOrigToStopDistanceAlt) + { + this.tourOrigToStopDistanceAlt = tourOrigToStopDistanceAlt; + } + + /** + * @return the stopToTourDestDistanceAlt + */ + public double getStopToTourDestDistanceAlt(int alt) + { + return stopToTourDestDistanceAlt[alt]; + } + + /** + * @param stopToTourDestDistanceAlt + * the stopToTourDestDistanceAlt to set + */ + public void setStopToTourDestDistanceAlt(double[] stopToTourDestDistanceAlt) + { + this.stopToTourDestDistanceAlt = stopToTourDestDistanceAlt; + } + + /** + * @return the sizeTerms. The size term is the size of the alternative north + * of the border. It is indexed by alternative, where alternative is + * either taz-station pair or mgra-station pair, depending on + * whether the DMU is being used for the SOA model or the actual + * model. + */ + public double getSizeTerm(int alt) + { + return sizeTerms[purpose][alt]; + } + + /** + * @param sizeTerms + * the sizeTerms to set. The size term is the size of the + * alternative north of the border. It is indexed by alternative, + * where alternative is either taz-station pair or mgra-station + * pair, depending on whether the DMU is being used for the SOA + * model or the actual model. + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + /** + * @return the correctionFactors + */ + public double getCorrectionFactor(int alt) + { + return correctionFactors[alt]; + } + + /** + * @param correctionFactors + * the correctionFactors to set + */ + public void setCorrectionFactors(double[] correctionFactors) + { + this.correctionFactors = correctionFactors; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the purpose + */ + public int getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(int purpose) + { + this.purpose = purpose; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + methodIndexMap.put("getPurpose", 0); + methodIndexMap.put("getStopsOnHalfTour", 1); + methodIndexMap.put("getStopNumber", 2); + methodIndexMap.put("getInboundStop", 3); + methodIndexMap.put("getTourDuration", 4); + + methodIndexMap.put("getSizeTerm", 5); + methodIndexMap.put("getCorrectionFactor", 6); + methodIndexMap.put("getSampleNumber", 7); + methodIndexMap.put("getOsMcLogsumAlt", 8); + methodIndexMap.put("getSdMcLogsumAlt", 9); + methodIndexMap.put("getTourOrigToStopDistanceAlt", 10); + methodIndexMap.put("getStopToTourDestDistanceAlt", 11); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getPurpose(); + break; + case 1: + returnValue = getStopsOnHalfTour(); + break; + case 2: + returnValue = getStopNumber(); + break; + case 3: + returnValue = getInboundStop(); + break; + case 4: + returnValue = getTourDuration(); + break; + case 5: + returnValue = getSizeTerm(arrayIndex); + break; + case 6: + returnValue = getCorrectionFactor(arrayIndex); + break; + case 7: + returnValue = getSampleNumber(arrayIndex); + break; + case 8: + returnValue = getOsMcLogsumAlt(arrayIndex); + break; + case 9: + returnValue = getSdMcLogsumAlt(arrayIndex); + break; + case 10: + returnValue = getTourOrigToStopDistanceAlt(arrayIndex); + break; + case 11: + returnValue = getStopToTourDestDistanceAlt(arrayIndex); + break; + + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java new file mode 100644 index 0000000..4a6bde0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java @@ -0,0 +1,538 @@ +package org.sandag.abm.visitor; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.ConcreteAlternative; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class VisitorStopLocationChoiceModel +{ + + private transient Logger logger = Logger.getLogger("visitorModel"); + + private McLogsumsCalculator logsumHelper; + private VisitorModelStructure modelStructure; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private VisitorStopLocationChoiceDMU dmu; + private VisitorTripModeChoiceModel tripModeChoiceModel; + double logsum = 0; + private ChoiceModelApplication soaModel; + private ChoiceModelApplication destModel; + + // the following arrays are calculated in the station-destination choice + // model and passed in the constructor. + private double[][] mgraSizeTerms; // by + // purpose, + // MGRA + private double[][][] mgraProbabilities; // by + // purpose, + // TAZ, + // MGRA + + private TableDataSet alternativeData; // the + // alternatives, + // with + // a + // "dest" + // - + // indicating + // the + // destination + // TAZ + // in + // San + // Diego + // County + + // following are used for each taz alternative + private double[] soaTourOrigToStopDistanceAlt; // by + // TAZ + private double[] soaStopToTourDestDistanceAlt; // by + // TAZ + private double[][] tazSizeTerms; // by + // purpose, + // TAZ + // - + // set + // by + // constructor + + // following are used for sampled mgras + private int sampleRate; + private double[][] sampledSizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgra) + private double[] correctionFactors; // by + // alternative + // (sampled + // mgra, + // for + // full + // model + // only) + private int[] sampledTazs; // by + // alternative + // (sampled + // taz) + private int[] sampledMgras; // by + // alternative(sampled + // mgra) + private double[] tourOrigToStopDistanceAlt; + private double[] stopToTourDestDistanceAlt; + private double[] osMcLogsumAlt; + private double[] sdMcLogsumAlt; + + HashMap frequencyChosen; + + private VisitorTrip trip; + + private int originMgra; // the + // origin + // MGRA + // of + // the + // stop + // (originMgra + // -> + // stopMgra + // -> + // destinationMgra) + private int destinationMgra; // the + // destination + // MGRA + // of + // the + // stop + // (originMgra + // -> + // stopMgra + // -> + // destinationMgra) + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public VisitorStopLocationChoiceModel(HashMap propertyMap, + VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + mgraManager = MgraDataManager.getInstance(propertyMap); + tazManager = TazDataManager.getInstance(propertyMap); + + modelStructure = myModelStructure; + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + // this sets by thread, so do it outside of initialization + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + setupStopLocationChoiceModel(propertyMap, dmuFactory); + + frequencyChosen = new HashMap(); + + trip = new VisitorTrip(); + + } + + /** + * Read the UEC file and set up the stop destination choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupStopLocationChoiceModel(HashMap rbMap, + VisitorDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up visitor stop location choice model.")); + + dmu = dmuFactory.getVisitorStopLocationChoiceDMU(); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String visitorStopLocationSoaFileName = Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.soa.uec.file"); + visitorStopLocationSoaFileName = uecFileDirectory + visitorStopLocationSoaFileName; + + int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.soa.data.page")); + int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.soa.model.page")); + + String visitorStopLocationFileName = Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.uec.file"); + visitorStopLocationFileName = uecFileDirectory + visitorStopLocationFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.data.page")); + int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.slc.model.page")); + + // create a ChoiceModelApplication object for the SOA model. + soaModel = new ChoiceModelApplication(visitorStopLocationSoaFileName, soaModelPage, + soaDataPage, rbMap, (VariableTable) dmu); + + // create a ChoiceModelApplication object for the full model. + destModel = new ChoiceModelApplication(visitorStopLocationFileName, modelPage, dataPage, + rbMap, (VariableTable) dmu); + sampleRate = destModel.getAlternativeNames().length; + + // get the alternative data + UtilityExpressionCalculator uec = soaModel.getUEC(); + alternativeData = uec.getAlternativeData(); + int purposes = modelStructure.VISITOR_PURPOSES.length; + + sampledSizeTerms = new double[purposes][sampleRate + 1]; // by purpose, + // alternative + // (taz or + // sampled + // mgra) + correctionFactors = new double[sampleRate + 1]; // by alternative + // (sampled mgra, for + // full model only) + sampledTazs = new int[sampleRate + 1]; // by alternative (sampled taz) + sampledMgras = new int[sampleRate + 1]; // by alternative (sampled mgra) + tourOrigToStopDistanceAlt = new double[sampleRate + 1]; + stopToTourDestDistanceAlt = new double[sampleRate + 1]; + osMcLogsumAlt = new double[sampleRate + 1]; + sdMcLogsumAlt = new double[sampleRate + 1]; + + } + + /** + * Create a sample for the tour and stop. + * + * @param tour + * @param stop + */ + private void createSample(VisitorTour tour, VisitorStop stop) + { + + int purpose = tour.getPurpose(); + int origTaz = 0; + int destTaz = 0; + int period = modelStructure.AM; + + dmu.setPurpose(purpose); + boolean inbound = stop.isInbound(); + if (inbound) + { + dmu.setInboundStop(1); + dmu.setStopsOnHalfTour(tour.getNumberInboundStops()); + + // destination for inbound stops is always tour origin + destinationMgra = tour.getOriginMGRA(); + destTaz = mgraManager.getTaz(destinationMgra); + + // origin for inbound stops is tour destination if first stop, or + // last chosen stop location + if (stop.getId() == 0) + { + originMgra = tour.getDestinationMGRA(); + origTaz = mgraManager.getTaz(originMgra); + } else + { + VisitorStop[] stops = tour.getInboundStops(); + originMgra = stops[stop.getId() - 1].getMgra(); + origTaz = mgraManager.getTaz(originMgra); + } + + } else + { + dmu.setInboundStop(0); + dmu.setStopsOnHalfTour(tour.getNumberOutboundStops()); + + // destination for outbound stops is always tour destination + destinationMgra = tour.getDestinationMGRA(); + destTaz = mgraManager.getTaz(destinationMgra); + + // origin for outbound stops is tour origin if first stop, or last + // chosen stop location + if (stop.getId() == 0) + { + originMgra = tour.getOriginMGRA(); + origTaz = mgraManager.getTaz(originMgra); + } else + { + VisitorStop[] stops = tour.getOutboundStops(); + originMgra = stops[stop.getId() - 1].getMgra(); + origTaz = mgraManager.getTaz(originMgra); + } + } + dmu.setStopNumber(stop.getId() + 1); + dmu.setDmuIndexValues(origTaz, origTaz, origTaz, 0, false); + + // distances + soaTourOrigToStopDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceFromTaz( + origTaz, period); + soaStopToTourDestDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceToTaz( + destTaz, period); + dmu.setTourOrigToStopDistanceAlt(soaTourOrigToStopDistanceAlt); + dmu.setStopToTourDestDistanceAlt(soaStopToTourDestDistanceAlt); + + dmu.setSizeTerms(tazSizeTerms); + + // solve for each sample + frequencyChosen.clear(); + for (int sample = 1; sample <= sampleRate; ++sample) + { + + // solve the UEC + soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + // choose a TAZ + double random = tour.getRandom(); + ConcreteAlternative[] alts = soaModel.getAlternatives(); + double cumProb = 0; + double altProb = 0; + int sampledTaz = -1; + for (int i = 0; i < alts.length; ++i) + { + cumProb += alts[i].getProbability(); + if (random < cumProb) + { + sampledTaz = (int) alternativeData.getValueAt(i + 1, "dest"); + altProb = alts[i].getProbability(); + break; + } + } + + // set the sampled taz in the array + sampledTazs[sample] = sampledTaz; + + // now find an MGRA in the taz corresponding to the random number + // drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives + // probability + int[] mgraArray = tazManager.getMgraArray(sampledTaz); + int mgraNumber = 0; + double[] mgraCumProb = mgraProbabilities[purpose][sampledTaz]; + + if (mgraCumProb == null) + { + logger.error("Error: mgraCumProb array is null for purpose " + purpose + + " sampledTaz " + sampledTaz + " hhID " + tour.getID()); + throw new RuntimeException(); + } + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > random && mgraCumProb[i] > 0) + { + mgraNumber = mgraArray[i]; + sampledMgras[sample] = mgraNumber; + + // for now, store the probability in the correction factors + // array + correctionFactors[sample] = mgraCumProb[i] * altProb; + + break; + } + } + + // store frequency chosen + if (!frequencyChosen.containsKey(mgraNumber)) + { + frequencyChosen.put(mgraNumber, 1); + } else + { + int freq = frequencyChosen.get(mgraNumber); + frequencyChosen.put(mgraNumber, freq + 1); + } + + // set the size terms for the sample + sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; + + // set the distances for the sample + tourOrigToStopDistanceAlt[sample] = soaTourOrigToStopDistanceAlt[sampledTaz]; + stopToTourDestDistanceAlt[sample] = soaStopToTourDestDistanceAlt[sampledTaz]; + + } + // calculate correction factors + for (int sample = 1; sample <= sampleRate; ++sample) + { + int mgra = sampledMgras[sample]; + int freq = frequencyChosen.get(mgra); + correctionFactors[sample] = (float) Math.log((double) freq / correctionFactors[sample]); + + } + + } + + /** + * Choose a stop location from the sample. + * + * @param tour + * The visitor tour. + * @param stop + * The visitor stop. + */ + public void chooseStopLocation(VisitorTour tour, VisitorStop stop) + { + + // create a sample of mgras and set all of the dmu properties + createSample(tour, stop); + dmu.setCorrectionFactors(correctionFactors); + dmu.setSizeTerms(sampledSizeTerms); + dmu.setTourOrigToStopDistanceAlt(stopToTourDestDistanceAlt); + dmu.setStopToTourDestDistanceAlt(stopToTourDestDistanceAlt); + dmu.setSampleNumber(sampledMgras); + + // calculate trip mode choice logsums to and from stop + for (int i = 1; i <= sampleRate; ++i) + { + + // to stop (originMgra -> stopMgra ) + trip.initializeFromStop(tour, stop, true); + trip.setOriginMgra(trip.getOriginMgra()); + trip.setDestinationMgra(sampledMgras[i]); + double logsum = tripModeChoiceModel.computeUtilities(tour, trip); + osMcLogsumAlt[i] = logsum; + + // from stop (stopMgra -> destinationMgra) + trip.initializeFromStop(tour, stop, false); + trip.setOriginMgra(sampledMgras[i]); + trip.setDestinationMgra(trip.getDestinationMgra()); + logsum = tripModeChoiceModel.computeUtilities(tour, trip); + sdMcLogsumAlt[i] = logsum; + + } + dmu.setOsMcLogsumAlt(osMcLogsumAlt); + dmu.setSdMcLogsumAlt(sdMcLogsumAlt); + + // log headers to traceLogger + if (tour.getDebugChoiceModels()) + { + String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() + + " purpose " + modelStructure.VISITOR_PURPOSES[stop.getPurpose()]; + destModel.choiceModelUtilityTraceLoggerHeading( + "Intermediate stop location choice model", decisionMakerLabel); + } + + destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + double random = tour.getRandom(); + int alt = destModel.getChoiceResult(random); + int destMgra = sampledMgras[alt]; + stop.setMgra(destMgra); + + // write UEC calculation results and choice + if (tour.getDebugChoiceModels()) + { + String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() + + " purpose " + modelStructure.VISITOR_PURPOSES[stop.getPurpose()]; + String loggingHeader = String.format("%s %s", + "Intermediate stop location choice model", decisionMakerLabel); + destModel.logUECResults(logger, loggingHeader); + logger.info("Chose alternative " + alt + " mgra " + destMgra + " with random number " + + random); + logger.info(""); + logger.info(""); + } + + } + + /** + * @return the mgraSizeTerms + */ + public double[][] getMgraSizeTerms() + { + return mgraSizeTerms; + } + + /** + * @return the mgraProbabilities + */ + public double[][][] getMgraProbabilities() + { + return mgraProbabilities; + } + + /** + * @return the tazSizeTerms + */ + public double[][] getTazSizeTerms() + { + return tazSizeTerms; + } + + /** + * Set mgra size terms: must call before choosing location. + * + * @param mgraSizeTerms + */ + public void setMgraSizeTerms(double[][] mgraSizeTerms) + { + + if (mgraSizeTerms == null) + { + logger.error("Error attempting to set MGRASizeTerms in VisitorStopLocationChoiceModel: MGRASizeTerms are null"); + throw new RuntimeException(); + } + this.mgraSizeTerms = mgraSizeTerms; + } + + /** + * Set taz size terms: must call before choosing location. + * + * @param tazSizeTerms + */ + public void setTazSizeTerms(double[][] tazSizeTerms) + { + if (tazSizeTerms == null) + { + logger.error("Error attempting to set TazSizeTerms in VisitorStopLocationChoiceModel: TazSizeTerms are null"); + throw new RuntimeException(); + } + this.tazSizeTerms = tazSizeTerms; + } + + /** + * Set the mgra probabilities. Must call before choosing location. + * + * @param mgraProbabilities + */ + public void setMgraProbabilities(double[][][] mgraProbabilities) + { + if (mgraProbabilities == null) + { + logger.error("Error attempting to set mgraProbabilities in VisitorStopLocationChoiceModel: mgraProbabilities are null"); + throw new RuntimeException(); + } + this.mgraProbabilities = mgraProbabilities; + } + + /** + * Set trip mode choice model. Must call before choosing location. + * + * @param tripModeChoiceModel + */ + public void setTripModeChoiceModel(VisitorTripModeChoiceModel tripModeChoiceModel) + { + this.tripModeChoiceModel = tripModeChoiceModel; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java new file mode 100644 index 0000000..2693a44 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java @@ -0,0 +1,233 @@ +package org.sandag.abm.visitor; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the stop purpose choice model for visitor tours. It is + * currently based on a static probability distribution stored in an input file, + * and indexed into by purpose, tour leg direction (inbound or outbound), the + * stop number, and whether there is just one or multiple stops on the tour leg. + * + * @author Freedman + * + */ +public class VisitorStopPurposeModel +{ + private transient Logger logger = Logger.getLogger("visitorModel"); + + private double[][] cumProbability; // by + // alternative, + // stop + // purpose: + // cumulative + // probability + // distribution + VisitorModelStructure modelStructure; + + HashMap arrayElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + // tour + // purpose, + // tour + // leg + // direction, + // stop + // number, + // and + // stop + // complexity. + + /** + * Constructor. + */ + public VisitorStopPurposeModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.stop.purpose.file"); + stopFrequencyFile = directory + stopFrequencyFile; + + modelStructure = new VisitorModelStructure(); + + arrayElementMap = new HashMap(); + readStopPurposeFile(stopFrequencyFile); + + } + + /** + * Read the stop frequency distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readStopPurposeFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating stop purpose probability distribution"); + + // take a pass through the data and see how many alternatives there are + // for each purpose + int rowCount = probabilityTable.getRowCount(); + int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 + + cumProbability = new double[rowCount][purposes]; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "TourPurp"); + + int inbound = (int) probabilityTable.getValueAt(row, "Inbound"); + int stopNumber = (int) probabilityTable.getValueAt(row, "StopNum"); + int multiple = (int) probabilityTable.getValueAt(row, "Multiple"); + + // store cumulative probabilities + float cumProb = 0; + for (int p = 0; p < purposes; ++p) + { + String label = "StopPurp" + p; + cumProb += probabilityTable.getValueAt(row, label); + cumProbability[row - 1][p] += cumProb; + } + + if (Math.abs(cumProb - 1.0) > 0.00001) + logger.info("Cumulative probability for tour purpose " + purpose + " inbound " + + inbound + " stopNumber " + stopNumber + " multiple " + multiple + " is " + + cumProb); + + int key = getKey(purpose, inbound, stopNumber, multiple); + arrayElementMap.put(key, row - 1); + + } + + logger.info("End calculating stop purpose probability distribution"); + + } + + /** + * Get the key for the arrayElementMap. + * + * @param tourPurp + * Tour purpose + * @param isInbound + * 1 if the stop is on the inbound direction, else 0. + * @param stopNumber + * The number of the stop. + * @param multipleStopsOnLeg + * 1 if multiple stops on leg, else 0. + * @return arrayElementMap key. + */ + private int getKey(int tourPurp, int isInbound, int stopNumber, int multipleStopsOnLeg) + { + + return tourPurp * 1000 + isInbound * 100 + stopNumber * 10 + multipleStopsOnLeg; + } + + /** + * Calculate purposes all stops on the tour + * + * @param tour + * A cross border tour (with tour purpose) + */ + public void calculateStopPurposes(VisitorTour tour) + { + + // outbound stops first + if (tour.getNumberOutboundStops() != 0) + { + + int tourPurp = tour.getPurpose(); + VisitorStop[] stops = tour.getOutboundStops(); + int multiple = 0; + if (stops.length > 1) multiple = 1; + + // iterate through stop list and calculate purpose for each + for (int i = 0; i < stops.length; ++i) + { + int key = getKey(tourPurp, 0, i + 1, multiple); + int element = arrayElementMap.get(key); + double[] cumProb = cumProbability[element]; + double rand = tour.getRandom(); + int purpose = chooseFromDistribution(rand, cumProb); + stops[i].setPurpose((byte) purpose); + } + } + // inbound stops last + if (tour.getNumberInboundStops() != 0) + { + + int tourPurp = tour.getPurpose(); + VisitorStop[] stops = tour.getInboundStops(); + int multiple = 0; + if (stops.length > 1) multiple = 1; + + // iterate through stop list and calculate purpose for each + for (int i = 0; i < stops.length; ++i) + { + int key = getKey(tourPurp, 1, i + 1, multiple); + int element = arrayElementMap.get(key); + double[] cumProb = cumProbability[element]; + double rand = tour.getRandom(); + int purpose = chooseFromDistribution(rand, cumProb); + stops[i].setPurpose((byte) purpose); + } + } + } + + /** + * Choose purpose from the cumulative probability distribution + * + * @param random + * Uniformly distributed random number + * @param cumProb + * Cumulative probability distribution + * @return Stop purpose (0 init). + */ + private int chooseFromDistribution(double random, double[] cumProb) + { + + int choice = -1; + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + choice = i; + break; + } + + } + return choice; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java new file mode 100644 index 0000000..e5a2bd7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java @@ -0,0 +1,365 @@ +package org.sandag.abm.visitor; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the TOD choice model for visitor tours. It is currently based + * on a static probability distribution stored in an input file, and indexed + * into by purpose. + * + * @author Freedman + * + */ +public class VisitorStopTimeOfDayChoiceModel +{ + private transient Logger logger = Logger.getLogger("visitorModel"); + + private double[][] outboundCumProbability; // by + // alternative: + // outbound + // cumulative + // probability + // distribution + private int[] outboundOffsets; // by + // alternative: + // offsets + // for + // outbound + // stop + // duration + // choice + + private double[][] inboundCumProbability; // by + // alternative: + // inbound + // cumulative + // probability + // distribution + private int[] inboundOffsets; // by + // alternative: + // offsets + // for + // inbound + // stop + // duration + // choice + private VisitorModelStructure modelStructure; + + private HashMap outboundElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + // tour duration and stop number. + + private HashMap inboundElementMap; // Hashmap + // used + // to + // get + // the + // element + // number + // of + // the + // cumProbability + // array + // based + // on + // the + + // tour duration and stop number. + + /** + * Constructor. + */ + public VisitorStopTimeOfDayChoiceModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String outboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.stop.outbound.duration.file"); + String inboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.stop.inbound.duration.file"); + + outboundDurationFile = directory + outboundDurationFile; + inboundDurationFile = directory + inboundDurationFile; + + modelStructure = new VisitorModelStructure(); + + outboundElementMap = new HashMap(); + readOutboundFile(outboundDurationFile); + + inboundElementMap = new HashMap(); + readInboundFile(inboundDurationFile); + } + + /** + * Read the outbound stop duration file and store the cumulative probability + * distribution as well as the offsets and set the key map to index into the + * probability array. + * + * @param fileName + */ + public void readOutboundFile(String fileName) + { + TableDataSet outboundTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + outboundTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + int columns = outboundTable.getColumnCount(); + int rows = outboundTable.getRowCount(); + outboundCumProbability = new double[rows][columns - 3]; + + // first three columns are index fields, rest are offsets + outboundOffsets = new int[columns - 3]; + for (int i = 4; i <= columns; ++i) + { + String offset = outboundTable.getColumnLabel(i); + outboundOffsets[i - 4] = new Integer(offset); + } + + // now fill in cumulative probability array + for (int row = 1; row <= rows; ++row) + { + + int lowerBound = (int) outboundTable.getValueAt(row, "RemainingLow"); + int upperBound = (int) outboundTable.getValueAt(row, "RemainingHigh"); + int stopNumber = (int) outboundTable.getValueAt(row, "Stop"); + + for (int duration = lowerBound; duration <= upperBound; ++duration) + { + int key = getKey(stopNumber, duration); + outboundElementMap.put(key, row - 1); + } + + // cumulative probability distribution + double cumProb = 0; + for (int col = 4; col <= columns; ++col) + { + cumProb += outboundTable.getValueAt(row, col); + outboundCumProbability[row - 1][col - 4] = cumProb; + } + + } + + } + + /** + * Read the inbound stop duration file and store the cumulative probability + * distribution as well as the offsets and set the key map to index into the + * probability array. + * + * @param fileName + */ + public void readInboundFile(String fileName) + { + TableDataSet inboundTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + inboundTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + int columns = inboundTable.getColumnCount(); + int rows = inboundTable.getRowCount(); + inboundCumProbability = new double[rows][columns - 3]; + + // first three columns are index fields, rest are offsets + inboundOffsets = new int[columns - 3]; + for (int i = 4; i <= columns; ++i) + { + String offset = inboundTable.getColumnLabel(i); + inboundOffsets[i - 4] = new Integer(offset); + } + + // now fill in cumulative probability array + for (int row = 1; row <= rows; ++row) + { + + int lowerBound = (int) inboundTable.getValueAt(row, "RemainingLow"); + int upperBound = (int) inboundTable.getValueAt(row, "RemainingHigh"); + int stopNumber = (int) inboundTable.getValueAt(row, "Stop"); + + for (int duration = lowerBound; duration <= upperBound; ++duration) + { + int key = getKey(stopNumber, duration); + inboundElementMap.put(key, row - 1); + } + // cumulative probability distribution + double cumProb = 0; + for (int col = 4; col <= columns; ++col) + { + cumProb += inboundTable.getValueAt(row, col); + inboundCumProbability[row - 1][col - 4] = cumProb; + } + + } + + } + + /** + * Get the key for the arrayElementMap. + * + * @param stopNumber + * stop number + * @param periodsRemaining + * Remaining time periods + * @return arrayElementMap key. + */ + private int getKey(int stopNumber, int periodsRemaining) + { + + return periodsRemaining * 10 + stopNumber; + } + + /** + * Choose the stop time of day period. + * + * @param tour + * @param stop + */ + public void chooseTOD(VisitorTour tour, VisitorStop stop) + { + + boolean inbound = stop.isInbound(); + int stopNumber = stop.getId() + 1; + int arrivalPeriod = tour.getArriveTime(); + + if (!inbound) + { + + // find the departure time + int departPeriod = 0; + if (stop.getId() == 0) departPeriod = tour.getDepartTime(); + else + { + VisitorStop[] stops = tour.getOutboundStops(); + departPeriod = stops[stop.getId() - 1].getStopPeriod(); + } + + int periodsRemaining = arrivalPeriod - departPeriod; + + int key = getKey(stopNumber, periodsRemaining); + int element = outboundElementMap.get(key); + double[] cumProb = outboundCumProbability[element]; + double random = tour.getRandom(); + + // iterate through the offset distribution, choose an offset, and + // set in the stop + if (tour.getDebugChoiceModels()) + { + logger.info("Stop TOD Choice Model for tour " + tour.getID() + " outbound stop " + + stop.getId() + " periods remaining " + periodsRemaining); + logger.info(" random number " + random); + } + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + int offset = outboundOffsets[i]; + int period = departPeriod + offset; + stop.setPeriod(period); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Chose alt " + i + " offset " + offset + " from depart period " + + departPeriod); + logger.info("Stop period is " + stop.getStopPeriod()); + + } + break; + + } + } + } else + { + // inbound stop + + // find the departure time + int departPeriod = 0; + + // first inbound stop + if (stop.getId() == 0) + { + + // there were outbound stops + if (tour.getOutboundStops() != null) + { + VisitorStop[] outboundStops = tour.getOutboundStops(); + departPeriod = outboundStops[outboundStops.length - 1].getStopPeriod(); + } else + { + // no outbound stops + departPeriod = tour.getDepartTime(); + } + } else + { + // not first inbound stop + VisitorStop[] stops = tour.getInboundStops(); + departPeriod = stops[stop.getId() - 1].getStopPeriod(); + } + + int periodsRemaining = arrivalPeriod - departPeriod; + + int key = getKey(stopNumber, periodsRemaining); + int element = inboundElementMap.get(key); + double[] cumProb = inboundCumProbability[element]; + double random = tour.getRandom(); + if (tour.getDebugChoiceModels()) + { + logger.info("Stop TOD Choice Model for tour " + tour.getID() + " inbound stop " + + stop.getId() + " periods remaining " + periodsRemaining); + logger.info("Random number " + random); + } + for (int i = 0; i < cumProb.length; ++i) + { + if (random < cumProb[i]) + { + int offset = inboundOffsets[i]; + int arrivePeriod = tour.getArriveTime(); + int period = arrivePeriod + offset; + stop.setPeriod(period); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Chose alt " + i + " offset " + offset + " from arrive period " + + arrivePeriod); + logger.info("Stop period is " + stop.getStopPeriod()); + + } + break; + } + } + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java new file mode 100644 index 0000000..1de6057 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java @@ -0,0 +1,350 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Household; +import com.pb.common.math.MersenneTwister; + +public class VisitorTour + implements Serializable +{ + + private MersenneTwister random; + private int ID; + + // following variables determined via simulation + private byte segment; // 0 + // = + // business, + // 1 + // = + // personal + private byte purpose; + private byte numberOfParticipants; + private int income; + private int autoAvailable; + + private VisitorStop[] outboundStops; + private VisitorStop[] inboundStops; + + private VisitorTrip[] trips; + + private int departTime; + private int arriveTime; + + private boolean debugChoiceModels; + + // following variables chosen via choice models + private int originMGRA; + private int destinationMGRA; + private byte tourMode; + + private float valueOfTime; + /** + * Public constructor. + * + * @param seed + * A seed for the random number generator. + */ + public VisitorTour(long seed) + { + + random = new MersenneTwister(seed); + } + + /** + * @return the iD + */ + public int getID() + { + return ID; + } + + /** + * @param iD + * the iD to set + */ + public void setID(int iD) + { + ID = iD; + } + + /** + * @return the purpose + */ + public byte getPurpose() + { + return purpose; + } + + /** + * @return the outboundStops + */ + public VisitorStop[] getOutboundStops() + { + return outboundStops; + } + + /** + * @param outboundStops + * the outboundStops to set + */ + public void setOutboundStops(VisitorStop[] outboundStops) + { + this.outboundStops = outboundStops; + } + + /** + * @return the inboundStops + */ + public VisitorStop[] getInboundStops() + { + return inboundStops; + } + + /** + * @param inboundStops + * the inboundStops to set + */ + public void setInboundStops(VisitorStop[] inboundStops) + { + this.inboundStops = inboundStops; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(byte purpose) + { + this.purpose = purpose; + } + + /** + * @return the departTime + */ + public int getDepartTime() + { + return departTime; + } + + /** + * @param departTime + * the departTime to set + */ + public void setDepartTime(int departTime) + { + this.departTime = departTime; + } + + public VisitorTrip[] getTrips() + { + return trips; + } + + public void setTrips(VisitorTrip[] trips) + { + this.trips = trips; + } + + /** + * @return the originMGRA + */ + public int getOriginMGRA() + { + return originMGRA; + } + + /** + * @param originMGRA + * the originMGRA to set + */ + public void setOriginMGRA(int originMGRA) + { + this.originMGRA = originMGRA; + } + + /** + * @return the tour mode + */ + public byte getTourMode() + { + return tourMode; + } + + /** + * @param mode + * the tour mode to set + */ + public void setTourMode(byte mode) + { + this.tourMode = mode; + } + + /** + * Get a random number from the parties random class. + * + * @return A random number. + */ + public double getRandom() + { + return random.nextDouble(); + } + + /** + * @return the debugChoiceModels + */ + public boolean getDebugChoiceModels() + { + return debugChoiceModels; + } + + /** + * @param debugChoiceModels + * the debugChoiceModels to set + */ + public void setDebugChoiceModels(boolean debugChoiceModels) + { + this.debugChoiceModels = debugChoiceModels; + } + + /** + * Get the number of outbound stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberOutboundStops() + { + if (outboundStops == null) return 0; + else return outboundStops.length; + + } + + /** + * Get the number of return stops + * + * @return 0 if not initialized, else number of stops + */ + public int getNumberInboundStops() + { + if (inboundStops == null) return 0; + else return inboundStops.length; + + } + + /** + * @return the destinationMGRA + */ + public int getDestinationMGRA() + { + return destinationMGRA; + } + + /** + * @param destinationMGRA + * the destinationMGRA to set + */ + public void setDestinationMGRA(int destinationMGRA) + { + this.destinationMGRA = destinationMGRA; + } + + public void setArriveTime(int arriveTime) + { + this.arriveTime = arriveTime; + } + + public int getArriveTime() + { + return arriveTime; + } + + /** + * @return the numberOfParticipants + */ + public byte getNumberOfParticipants() + { + return numberOfParticipants; + } + + /** + * @param numberOfParticipants + * the numberOfParticipants to set + */ + public void setNumberOfParticipants(byte numberOfParticipants) + { + this.numberOfParticipants = numberOfParticipants; + } + + /** + * @return the income + */ + public int getIncome() + { + return income; + } + + /** + * @param income + * the income to set + */ + public void setIncome(int income) + { + this.income = income; + } + + /** + * @return the autoAvailable + */ + public int getAutoAvailable() + { + return autoAvailable; + } + + /** + * @param autoAvailable + * the autoAvailable to set + */ + public void setAutoAvailable(int autoAvailable) + { + this.autoAvailable = autoAvailable; + } + + /** + * @return the segment + */ + public byte getSegment() + { + return segment; + } + + /** + * @param segment + * the segment to set + */ + public void setSegment(byte segment) + { + this.segment = segment; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + + public void logTourObject(Logger logger, int totalChars) + { + + Household.logHelper(logger, "tourId: ", ID, totalChars); + Household.logHelper(logger, "tourPurpose: ", purpose, totalChars); + Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); + Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); + Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); + Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); + Household.logHelper(logger, "tourMode: ", tourMode, totalChars); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java new file mode 100644 index 0000000..405ab33 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java @@ -0,0 +1,317 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; +import org.apache.log4j.Logger; +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class VisitorTourDestChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger("visitorModel"); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected float tourDepartPeriod; + protected float tourArrivePeriod; + protected int purpose; + protected double[][] sizeTerms; // by + // purpose, + // alternative + // (taz + // or + // sampled + // mgras) + protected double[] correctionFactors; // by + // alternative + // (sampled + // mgra, + // for + // full + // model + // only) + protected double[] tourModeLogsums; // by + // alternative + // (sampled + // mgra + // pair, + // for + // full + // model + // only) + protected int[] sampleMGRA; // by + // alternative + // (sampled + // mgra) + protected int[] sampleTAZ; // by + // alternative + // (sampled + // taz) + + protected double nmWalkTimeOut; + protected double nmWalkTimeIn; + protected double nmBikeTimeOut; + protected double nmBikeTimeIn; + protected double lsWgtAvgCostM; + protected double lsWgtAvgCostD; + protected double lsWgtAvgCostH; + + public VisitorTourDestChoiceDMU(VisitorModelStructure modelStructure) + { + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Get the tour mode choice logsum for the sampled station-mgra pair. + * + * @param alt + * Sampled station-mgra + * @return + */ + public double getTourModeLogsum(int alt) + { + return tourModeLogsums[alt]; + } + + /** + * Set the tour mode choice logsums + * + * @param poeNumbers + * An array of tour mode choice logsums, one for each alternative + * (sampled station-mgra) + */ + public void setTourModeLogsums(double[] logsums) + { + this.tourModeLogsums = logsums; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + /** + * @return the sizeTerms. The size term is the size of the alternative north + * of the border. It is indexed by alternative, where alternative is + * either taz-station pair or mgra-station pair, depending on + * whether the DMU is being used for the SOA model or the actual + * model. + */ + public double getSizeTerm(int alt) + { + return sizeTerms[purpose][alt]; + } + + /** + * @param sizeTerms + * the sizeTerms to set. The size term is the size of the + * alternative north of the border. It is indexed by alternative, + * where alternative is either taz-station pair or mgra-station + * pair, depending on whether the DMU is being used for the SOA + * model or the actual model. + */ + public void setSizeTerms(double[][] sizeTerms) + { + this.sizeTerms = sizeTerms; + } + + /** + * @return the correctionFactors + */ + public double getCorrectionFactor(int alt) + { + return correctionFactors[alt]; + } + + /** + * @param correctionFactors + * the correctionFactors to set + */ + public void setCorrectionFactors(double[] correctionFactors) + { + this.correctionFactors = correctionFactors; + } + + public int getSampleMgra(int alt) + { + return sampleMGRA[alt]; + } + + public void setSampleMgra(int[] sampleNumber) + { + this.sampleMGRA = sampleNumber; + } + + public int getSampleTaz(int alt) + { + return sampleTAZ[alt]; + } + + public void setSampleTaz(int[] sampleNumber) + { + this.sampleTAZ = sampleNumber; + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the purpose + */ + public int getPurpose() + { + return purpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setPurpose(int purpose) + { + this.purpose = purpose; + } + + public float getTimeOutbound() + { + return tourDepartPeriod; + } + + public float getTimeInbound() + { + return tourArrivePeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(float tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(float tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getSizeTerm", 2); + methodIndexMap.put("getCorrectionFactor", 3); + methodIndexMap.put("getPurpose", 4); + methodIndexMap.put("getTourModeLogsum", 5); + methodIndexMap.put("getSampleMgra", 6); + methodIndexMap.put("getSampleTaz", 7); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 2: + returnValue = getSizeTerm(arrayIndex); + break; + case 3: + returnValue = getCorrectionFactor(arrayIndex); + break; + case 4: + returnValue = getPurpose(); + break; + case 5: + returnValue = getTourModeLogsum(arrayIndex); + break; + case 6: + returnValue = getSampleMgra(arrayIndex); + break; + case 7: + returnValue = getSampleTaz(arrayIndex); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java new file mode 100644 index 0000000..800b310 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java @@ -0,0 +1,586 @@ +package org.sandag.abm.visitor; + +import java.io.File; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +/** + * This class is used for both the sample of alternatives and the full + * destination choice model for visitor tours. + * + * + * @author Freedman + * + */ +public class VisitorTourDestChoiceModel +{ + + private double[][] mgraSizeTerms; // by + // purpose, + // MGRA + private double[][] tazSizeTerms; // by + // purpose, + // TAZ + private double[][][] mgraProbabilities; // by + // purpose, + // tazNumber, + // mgra + // index + // (sequential, + // 0-based) + private Matrix[] tazProbabilities; // by + // purpose, + // origin + // TAZ, + // destination + // TAZ + private TableDataSet alternativeData; // the + // alternatives, + // with + // a + // dest + // field + // indicating + // tazNumber + private int[] sampleMgras; // numbers + // of + // mgra + // for + // the + // sample + private int[] sampleTazs; // numbers + // of + // taz + // for + // the + // sample + private double[] sampleCorrectionFactors; // correction + // factors + // for + // sample + private double[][] sampleSizeTerms; // size + // terms + // for + // sample + private double[] sampleLogsums; // tour + // mc + // logsums + + private transient Logger logger = Logger.getLogger("visitorModel"); + + private TazDataManager tazManager; + private MgraDataManager mgraManager; + + private ChoiceModelApplication[] soaModel; + private ChoiceModelApplication[] destModel; + private UtilityExpressionCalculator sizeTermUEC; + private HashMap rbMap; + + private VisitorTourDestChoiceDMU dcDmu; + private VisitorTourModeChoiceModel tourModeChoiceModel; + + private HashMap frequencyChosen; // by + // mgra, + // number + // of + // times + // chosen + private int sampleRate; + + /** + * Constructor + * + * @param propertyMap + * Resource properties file map. + * @param dmuFactory + * Factory object for creation of airport model DMUs + */ + public VisitorTourDestChoiceModel(HashMap rbMap, + VisitorModelStructure modelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + + this.rbMap = rbMap; + + tazManager = TazDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, + CtrampApplication.PROPERTIES_UEC_PATH); + String visitorDCSoaFileName = Util.getStringValueFromPropertyMap(rbMap, + "visitor.dc.soa.uec.file"); + visitorDCSoaFileName = uecFileDirectory + visitorDCSoaFileName; + + int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.dc.soa.data.page")); + int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, + "visitor.dc.soa.size.page")); + + // initiate a DMU + dcDmu = dmuFactory.getVisitorTourDestChoiceDMU(); + + // read the model pages from the property file, create one choice model + // for each soa model + soaModel = new ChoiceModelApplication[VisitorModelStructure.VISITOR_PURPOSES.length]; + for (int i = 0; i < soaModel.length; ++i) + { + + // get page from property file + String purpose = VisitorModelStructure.VISITOR_PURPOSES[i].toLowerCase(); + String purposeName = "visitor.dc.soa." + purpose + ".page"; + String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); + purposeString.replaceAll(" ", ""); + int destModelPage = Integer.parseInt(purposeString); + + // create a ChoiceModelApplication object for the filename, model + // page and data page. + soaModel[i] = new ChoiceModelApplication(visitorDCSoaFileName, destModelPage, dataPage, + rbMap, (VariableTable) dcDmu); + } + + // get the alternative data from the first segment + UtilityExpressionCalculator uec = soaModel[0].getUEC(); + alternativeData = uec.getAlternativeData(); + + // create a UEC to solve size terms for each MGRA + sizeTermUEC = new UtilityExpressionCalculator(new File(visitorDCSoaFileName), sizePage, + dataPage, rbMap, (VariableTable) dcDmu); + + // create the full model UECs + // read the model pages from the property file, create one choice model + // for each full model + String visitorDCFileName = Util.getStringValueFromPropertyMap(rbMap, "visitor.dc.uec.file"); + visitorDCFileName = uecFileDirectory + visitorDCFileName; + destModel = new ChoiceModelApplication[VisitorModelStructure.VISITOR_PURPOSES.length]; + for (int i = 0; i < destModel.length; ++i) + { + + // get page from property file + String purpose = VisitorModelStructure.VISITOR_PURPOSES[i].toLowerCase(); + String purposeName = "visitor.dc." + purpose + ".page"; + String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); + purposeString.replaceAll(" ", ""); + int destModelPage = Integer.parseInt(purposeString); + + // create a ChoiceModelApplication object for the filename, model + // page and data page. + destModel[i] = new ChoiceModelApplication(visitorDCFileName, destModelPage, dataPage, + rbMap, (VariableTable) dcDmu); + if (i == 0) sampleRate = destModel[i].getNumberOfAlternatives(); + } + + frequencyChosen = new HashMap(); + sampleMgras = new int[sampleRate + 1]; + sampleTazs = new int[sampleRate + 1]; + sampleCorrectionFactors = new double[sampleRate + 1]; + sampleSizeTerms = new double[destModel.length][sampleRate + 1]; + sampleLogsums = new double[sampleRate + 1]; + + tourModeChoiceModel = new VisitorTourModeChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); + + } + + /** + * Calculate size terms + */ + public void calculateSizeTerms(VisitorDmuFactoryIf dmuFactory) + { + + logger.info("Calculating Visitor Tour Destination Choice Model MGRA Size Terms"); + + ArrayList mgras = mgraManager.getMgras(); + int[] mgraTaz = mgraManager.getMgraTaz(); + int maxMgra = mgraManager.getMaxMgra(); + int maxTaz = tazManager.getMaxTaz(); + int purposes = sizeTermUEC.getNumberOfAlternatives(); + + mgraSizeTerms = new double[purposes][maxMgra + 1]; + tazSizeTerms = new double[purposes][maxTaz + 1]; + IndexValues iv = new IndexValues(); + VisitorTourDestChoiceDMU aDmu = dmuFactory.getVisitorTourDestChoiceDMU(); + + // loop through mgras and calculate size terms + for (int mgra : mgras) + { + + int taz = mgraTaz[mgra]; + iv.setZoneIndex(mgra); + double[] utilities = sizeTermUEC.solve(iv, aDmu, null); + + // store the size terms + for (int purpose = 0; purpose < purposes; ++purpose) + { + + mgraSizeTerms[purpose][mgra] = utilities[purpose]; + tazSizeTerms[purpose][taz] += utilities[purpose]; + } + + } + + // now calculate probability of selecting each MGRA within each TAZ for + // SOA + mgraProbabilities = new double[purposes][maxTaz + 1][]; + int[] tazs = tazManager.getTazs(); + + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazs.length; ++taz) + { + int tazNumber = tazs[taz]; + int[] mgraArray = tazManager.getMgraArray(tazNumber); + + // initialize the vector of mgras for this purpose-taz + mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; + + // now calculate the cumulative probability distribution + double lastProb = 0.0; + for (int mgra = 0; mgra < mgraArray.length; ++mgra) + { + + int mgraNumber = mgraArray[mgra]; + if (tazSizeTerms[purpose][tazNumber] > 0.0) + mgraProbabilities[purpose][tazNumber][mgra] = lastProb + + mgraSizeTerms[purpose][mgraNumber] + / tazSizeTerms[purpose][tazNumber]; + lastProb = mgraProbabilities[purpose][tazNumber][mgra]; + } + if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) + logger.info("Error: purpose " + purpose + " taz " + tazNumber + + " cum prob adds up to " + lastProb); + } + + } + + // calculate logged size terms for mgra and taz vectors to be used in + // dmu + for (int purpose = 0; purpose < purposes; ++purpose) + { + for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) + if (tazSizeTerms[purpose][taz] > 0.0) + tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); + + for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) + if (mgraSizeTerms[purpose][mgra] > 0.0) + mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); + + } + logger.info("Finished Calculating Visitor Tour Destination Choice Model MGRA Size Terms"); + } + + /** + * Calculate taz probabilities. This method initializes and calculates the + * tazProbabilities array. + */ + public void calculateTazProbabilities(VisitorDmuFactoryIf dmuFactory) + { + + if (tazSizeTerms == null) + { + logger.error("Error: attemping to execute VisitorTourDestChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); + throw new RuntimeException(); + } + + logger.info("Calculating Visitor Model TAZ Probabilities Arrays"); + + // initialize taz probabilities array + int purposes = tazSizeTerms.length; + + // initialize the arrays + tazProbabilities = new Matrix[purposes]; + + // iterate through the alternatives in the alternatives file and set the + // size term and station logsum for each alternative + UtilityExpressionCalculator soaModelUEC = soaModel[0].getUEC(); + TableDataSet altData = soaModelUEC.getAlternativeData(); + + dcDmu.setSizeTerms(tazSizeTerms); + + // iterate through purposes + for (int purpose = 0; purpose < soaModel.length; ++purpose) + { + + tazProbabilities[purpose] = new Matrix("Prob_Matrix", "Probability Matrix", + altData.getRowCount() + 1, altData.getRowCount() + 1); + int[] tazs = altData.getColumnAsInt("dest"); + tazProbabilities[purpose].setExternalNumbersZeroBased(tazs); + + // iterate through origin zones, solve the UEC and store the results + // in the matrix + for (int taz = 0; taz < tazs.length; ++taz) + { + + int originTaz = (int) tazs[taz]; + + // set origin taz in dmu (destination set in UEC by alternative) + dcDmu.setDmuIndexValues(originTaz, originTaz, originTaz, originTaz, false); + + dcDmu.setPurpose(purpose); + + // Calculate utilities & probabilities + soaModel[purpose].computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); + + // Store probabilities (by purpose) + double[] probabilities = soaModel[purpose].getCumulativeProbabilities(); + + for (int i = 0; i < probabilities.length; ++i) + { + + double cumProb = probabilities[i]; + int destTaz = (int) altData.getValueAt(i + 1, "dest"); + tazProbabilities[purpose].setValueAt(originTaz, destTaz, (float) cumProb); + } + } + } + logger.info("Finished Calculating Visitor Model TAZ Probabilities Arrays"); + } + + /** + * Choose a MGRA alternative for sampling + * + * @param tour + * VisitorTour with purpose and Random + */ + private void chooseMgraSample(VisitorTour tour) + { + + frequencyChosen.clear(); + + // choose sample, set station logsums and mgra size terms + int purpose = tour.getPurpose(); + int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); + + for (int sample = 1; sample <= sampleRate; ++sample) + { + + // first find a TAZ and station + int alt = 0; + Matrix tazCumProb = tazProbabilities[purpose]; + double altProb = 0; + double cumProb = 0; + double random = tour.getRandom(); + int destinationTaz = -1; + for (int i = 0; i < tazCumProb.getColumnCount(); ++i) + { + destinationTaz = (int) tazCumProb.getExternalColumnNumber(i); + if (tazCumProb.getValueAt(originTaz, destinationTaz) > random) + { + alt = i; + if (i != 0) + { + cumProb = tazCumProb.getValueAt(originTaz, + tazCumProb.getExternalColumnNumber(i - 1)); + altProb = tazCumProb.getValueAt(originTaz, destinationTaz) + - tazCumProb.getValueAt(originTaz, + tazCumProb.getExternalColumnNumber(i - 1)); + } else + { + altProb = tazCumProb.getValueAt(originTaz, destinationTaz); + } + break; + } + } + + // get the taz number of the alternative, and an array of mgras in + // that taz + + int[] mgraArray = tazManager.getMgraArray(destinationTaz); + + // now find an MGRA in the taz corresponding to the random number + // drawn: + // note that the indexing needs to be offset by the cumulative + // probability of the chosen taz and the + // mgra probabilities need to be scaled by the alternatives + // probability + int mgraNumber = 0; + double[] mgraCumProb = mgraProbabilities[purpose][destinationTaz]; + for (int i = 0; i < mgraCumProb.length; ++i) + { + cumProb += mgraCumProb[i] * altProb; + if (cumProb > random && mgraCumProb[i] > 0) + { + mgraNumber = mgraArray[i]; + sampleMgras[sample] = mgraNumber; + sampleTazs[sample] = mgraManager.getTaz(mgraNumber); + + // for now, store the probability in the correction factors + // array + sampleCorrectionFactors[sample] = mgraCumProb[i] * altProb; + + break; + } + } + // store frequency chosen + if (!frequencyChosen.containsKey(mgraNumber)) + { + frequencyChosen.put(mgraNumber, 1); + } else + { + int freq = frequencyChosen.get(mgraNumber); + frequencyChosen.put(mgraNumber, freq + 1); + } + // set the size terms for the sample + sampleSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; + } + // calculate correction factors + for (int sample = 1; sample <= sampleRate; ++sample) + { + int mgra = sampleMgras[sample]; + int freq = frequencyChosen.get(mgra); + sampleCorrectionFactors[sample] = (float) Math.log((double) freq + / sampleCorrectionFactors[sample]); + + } + + } + + /** + * Use the tour mode choice model to calculate the logsum for each sampled + * mgra and store in the array. + * + * @param tour + * The visitor tour. + */ + private void calculateLogsumsForSample(VisitorTour tour) + { + + for (int sample = 1; sample <= sampleRate; ++sample) + { + + if (sampleMgras[sample] > 0) + { + + int destinationMgra = sampleMgras[sample]; + tour.setDestinationMGRA(destinationMgra); + + double logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, + "Sample logsum " + sample, "tour " + tour.getID() + " dest " + + destinationMgra); + sampleLogsums[sample] = logsum; + } else sampleLogsums[sample] = 0; + + } + + } + + /** + * Choose a destination MGRA for the tour. + * + * @param tour + * A cross border tour with a tour origin, purpose, attributes, + * and departure\arrival time and SENTRI availability members. + */ + public void chooseDestination(VisitorTour tour) + { + + chooseMgraSample(tour); + calculateLogsumsForSample(tour); + + double random = tour.getRandom(); + int purpose = tour.getPurpose(); + dcDmu.setPurpose(purpose); + + // set origin taz in dmu (destination set in UEC by alternative) + int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); + dcDmu.setDmuIndexValues(0, 0, originTaz, 0, false); + + // set size terms for each sampled station-mgra pair corresponding to + // mgra + dcDmu.setSizeTerms(sampleSizeTerms); + + // set the correction factors + dcDmu.setCorrectionFactors(sampleCorrectionFactors); + + // set the tour mode choice logsums + dcDmu.setTourModeLogsums(sampleLogsums); + + // sampled mgra + dcDmu.setSampleMgra(sampleMgras); + + // sampled taz + dcDmu.setSampleTaz(sampleTazs); + + if (tour.getDebugChoiceModels()) + { + logger.info("***"); + logger.info("Choosing destination alternative from sample"); + tour.logTourObject(logger, 100); + + // log the sample + destModel[purpose].choiceModelUtilityTraceLoggerHeading( + "Visitor tour destination model", "tour " + tour.getID()); + } + + destModel[purpose].computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + destModel[purpose].logUECResults(logger, "Visitor tour destination model"); + } + int alt = destModel[purpose].getChoiceResult(random); + + int primaryDestination = sampleMgras[alt]; + + if (tour.getDebugChoiceModels()) + { + logger.info("Chose destination MGRA " + primaryDestination); + } + + tour.setDestinationMGRA(primaryDestination); + } + + /** + * @return the tourModeChoiceModel + */ + public VisitorTourModeChoiceModel getTourModeChoiceModel() + { + return tourModeChoiceModel; + } + + /** + * @param tourModeChoiceModel + * the tourModeChoiceModel to set + */ + public void setTourModeChoiceModel(VisitorTourModeChoiceModel tourModeChoiceModel) + { + this.tourModeChoiceModel = tourModeChoiceModel; + } + + /** + * @return the mgraSizeTerms + */ + public double[][] getMgraSizeTerms() + { + return mgraSizeTerms; + } + + /** + * @return the tazSizeTerms + */ + public double[][] getTazSizeTerms() + { + return tazSizeTerms; + } + + /** + * @return the mgraProbabilities + */ + public double[][][] getMgraProbabilities() + { + return mgraProbabilities; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java new file mode 100644 index 0000000..31acfca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java @@ -0,0 +1,382 @@ +package org.sandag.abm.visitor; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.MatrixDataManager; +import com.pb.common.datafile.CSVFileWriter; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.MatrixType; +import com.pb.common.util.ResourceUtil; + +public class VisitorTourEstimationFile +{ + + public static final int MATRIX_DATA_SERVER_PORT = 1171; + public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; + + private MatrixDataServerRmi ms; + private String inputFileName; + private String outputFileName; + private TableDataSet estimationData; + + private VisitorModelStructure myModelStructure; + private VisitorDmuFactoryIf dmuFactory; + private McLogsumsCalculator logsumsCalculator; + private VisitorTourModeChoiceModel tourModeChoiceModel; + private HashMap rbMap; + private AutoTazSkimsCalculator tazDistanceCalculator; + + private static Logger logger = Logger.getLogger(VisitorTourEstimationFile.class); + private static final int SAMPLE_SIZE = 30; + private MgraDataManager mgraManager; + + /** + * Default constructor + */ + public VisitorTourEstimationFile(HashMap propertyMap) + { + this.rbMap = propertyMap; + mgraManager = MgraDataManager.getInstance(propertyMap); + myModelStructure = new VisitorModelStructure(); + + dmuFactory = new VisitorDmuFactory(myModelStructure); + + } + + public void createEstimationFile() + { + tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); + tazDistanceCalculator.computeTazDistanceArrays(); + logsumsCalculator = new McLogsumsCalculator(); + logsumsCalculator.setupSkimCalculators(rbMap); + logsumsCalculator.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + tourModeChoiceModel = new VisitorTourModeChoiceModel(rbMap, myModelStructure, dmuFactory, tazDistanceCalculator); + + // open file + estimationData = openFile(inputFileName); + + // iterate through file and calculate logsums + calculateTourMCLogsums(); + + // write the file + writeFile(outputFileName, estimationData); + } + + /** + * Open a trip file and return the Tabledataset. + * + * @fileName The name of the trip file + * @return The tabledataset + */ + public TableDataSet openFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet tripData; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + tripData = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return tripData; + } + + /** + * Write the file to disk. + * + * @param fileName + * Name of file + * @param data + * TableDataSet to write + */ + public void writeFile(String fileName, TableDataSet data) + { + logger.info("Begin writing the data to file " + fileName); + + try + { + CSVFileWriter csvFile = new CSVFileWriter(); + csvFile.writeFile(data, new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + logger.info("End writing the data to file " + fileName); + + } + + /** + * Calculate mode choice logsums + * + * TOURNUM: Unique ID SURVNUM: Number associated with traveler PersonType: 0 + * - Business, 1 - Personal Persons: Number of persons (in addition to + * traveler) on tour HHIncome: 1= $0-29,999; 2 = 30,000-59,999; 3 = + * 60,000-99,999; 4 = 100,000-149,000; 5=$150,000 or more; 6 = DK/Refused + * AutoAvail: 0 - No auto, 1 - Auto Available PURPOSE: 1 - Work, 2 - Other + * (non-dining) 3 - Dining originMGRA - origin of tour (hotel/overnight + * location) destMGRA - primary destination MGRA PeriodDepart: Period of + * Tour Departure from hotel/overnight location PeriodArrive: Period of Tour + * Arrival at primary destination SAMPLE_: 1:30 sampled alternatives + * + */ + public void calculateTourMCLogsums() + { + + int[] sample = new int[SAMPLE_SIZE]; + float[][] sampleLogsum = new float[SAMPLE_SIZE][estimationData.getRowCount()]; + float[] chosenLogsum = new float[estimationData.getRowCount()]; + String fieldName = null; + + for (int i = 0; i < estimationData.getRowCount(); ++i) + { + + if ((i + 1) <= 10 || (i + 1) % 100 == 0) + { + logger.info("Processing record " + (i + 1)); + } + int ID = (int) estimationData.getValueAt(i + 1, "TOURNUM"); + byte segment = (byte) estimationData.getValueAt(i + 1, "PersonType"); + byte purpose = (byte) estimationData.getValueAt(i + 1, "PURPOSE"); + byte income = (byte) estimationData.getValueAt(i + 1, "HHIncome"); + byte autoAvailable = (byte) estimationData.getValueAt(i + 1, "AutoAvail"); + byte participants = (byte) (estimationData.getValueAt(i + 1, "Persons") + 1); + int departTime = (int) estimationData.getValueAt(i + 1, "PeriodDepart"); + int arriveTime = (int) estimationData.getValueAt(i + 1, "PeriodArrive"); + int originMGRA = (int) estimationData.getValueAt(i + 1, "originMGRA"); + int destinationMGRA = (int) estimationData.getValueAt(i + 1, "destMGRA"); + + for (int j = 0; j < SAMPLE_SIZE; ++j) + { + fieldName = "SAMPLE_" + new Integer(j + 1).toString(); + sample[j] = (int) estimationData.getValueAt(i + 1, fieldName); + } + + VisitorTour tour = new VisitorTour(ID + 10000); + tour.setID(ID); + tour.setSegment(segment); + tour.setPurpose(purpose); + tour.setIncome(income); + tour.setAutoAvailable(autoAvailable); + tour.setNumberOfParticipants(participants); + tour.setDepartTime(departTime); + tour.setArriveTime(arriveTime); + tour.setOriginMGRA(originMGRA); + + if ((i + 1) == 1 || (i + 1) == 500) tour.setDebugChoiceModels(true); + else tour.setDebugChoiceModels(false); + + double logsum = 0; + // for each sampled destination + for (int j = 0; j < SAMPLE_SIZE; ++j) + { + + tour.setDestinationMGRA(sample[j]); + + // some of the samples are 0 + if (sample[j] > 0 && originMGRA > 0) + { + logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, + "DCEstimationFileLogsum", "ID " + ID); + sampleLogsum[j][i] = (float) logsum; + } + } + + // for the chosen destination + tour.setDestinationMGRA(destinationMGRA); + if (originMGRA > 0 && destinationMGRA > 0) + logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, + "DCEstimationFileLogsum", "ID " + ID); + chosenLogsum[i] = (float) logsum; + + } + + // append the logsum fields to the tabledata + estimationData.appendColumn(chosenLogsum, "CHSN_LS"); + + for (int j = 0; j < SAMPLE_SIZE; ++j) + { + fieldName = "SAMPLE_" + (j + 1) + "_LS"; + estimationData.appendColumn(sampleLogsum[j], fieldName); + } + + } + + /** + * Write the destination choice estimation file with logsums appended. + */ + public void writeDCEstimationFile(String name) + { + + } + + /** + * @param args + */ + public static void main(String[] args) + { + // TODO Auto-generated method stub + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + 2.0)); + + logger.info(String.format("Running Visitor Tour Estimation File Model")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + String inFile = args[1]; + String outFile = args[2]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + VisitorTourEstimationFile visitorEstimationFile = new VisitorTourEstimationFile(pMap); + + visitorEstimationFile.inputFileName = inFile; + visitorEstimationFile.outputFileName = outFile; + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = visitorEstimationFile.startMatrixServerProcess( + matrixServerAddress, serverPort, mt); + visitorEstimationFile.ms = matrixServer; + } else + { + visitorEstimationFile.ms = new MatrixDataServerRmi(matrixServerAddress, + serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); + visitorEstimationFile.ms.testRemote(Thread.currentThread().getName()); + + // these methods need to be called to set the matrix data + // manager in the matrix data server + MatrixDataManager mdm = MatrixDataManager.getInstance(); + mdm.setMatrixDataServerObject(visitorEstimationFile.ms); + } + + } + + } catch (Exception e) + { + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + visitorEstimationFile.createEstimationFile(); + + if (args.length < 3) + { + System.out + .println("Error: please specifiy inputFileName and outputFileName on command line"); + throw new RuntimeException(); + } + + } + + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java new file mode 100644 index 0000000..3bf9f9b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java @@ -0,0 +1,531 @@ +package org.sandag.abm.visitor; + +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.ArrayList; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.application.SandagTourBasedModel; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.Util; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.math.MersenneTwister; +import com.pb.common.util.ResourceUtil; + +public class VisitorTourManager +{ + + private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); + + private VisitorTour[] tours; + + VisitorModelStructure modelStructure; + SandagModelStructure sandagStructure; + + TableDataSet businessTourFrequency; + TableDataSet personalTourFrequency; + TableDataSet partySizeFrequency; + TableDataSet autoAvailableFrequency; + TableDataSet incomeFrequency; + + TableDataSet mgraData; + + float occupancyRate; + float householdRate; + + float businessHotelPercent; + float businessHouseholdPercent; + + private boolean seek; + private int traceId; + + private MersenneTwister random; + + private boolean avAvailable; + + /** + * Constructor. Reads properties file and opens/stores all probability + * distributions for sampling. Estimates number of airport travel parties + * and initializes parties[]. + * + * @param resourceFile + * Property file. + * + * Creates the array of cross-border tours. + */ + public VisitorTourManager(HashMap rbMap) + { + + modelStructure = new VisitorModelStructure(); + sandagStructure = new SandagModelStructure(); + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String mgraFile = Util.getStringValueFromPropertyMap(rbMap, "mgra.socec.file"); + mgraFile = directory + mgraFile; + + occupancyRate = new Float(Util.getStringValueFromPropertyMap(rbMap, + "visitor.hotel.occupancyRate")); + householdRate = new Float(Util.getStringValueFromPropertyMap(rbMap, + "visitor.household.occupancyRate")); + + businessHotelPercent = new Float(Util.getStringValueFromPropertyMap(rbMap, + "visitor.hotel.businessPercent")); + businessHouseholdPercent = new Float(Util.getStringValueFromPropertyMap(rbMap, + "visitor.household.businessPercent")); + + String businessTourFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.business.tour.file"); + businessTourFile = directory + businessTourFile; + + String personalTourFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.personal.tour.file"); + personalTourFile = directory + personalTourFile; + + String partySizeFile = Util.getStringValueFromPropertyMap(rbMap, "visitor.partySize.file"); + partySizeFile = directory + partySizeFile; + + String autoAvailableFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.autoAvailable.file"); + autoAvailableFile = directory + autoAvailableFile; + + String incomeFile = Util.getStringValueFromPropertyMap(rbMap, "visitor.income.file"); + incomeFile = directory + incomeFile; + + businessTourFrequency = readFile(businessTourFile); + personalTourFrequency = readFile(personalTourFile); + partySizeFrequency = readFile(partySizeFile); + autoAvailableFrequency = readFile(autoAvailableFile); + incomeFrequency = readFile(incomeFile); + + mgraData = readFile(mgraFile); + + seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "visitor.seek")); + traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "visitor.trace")); + + float avShare = new Float(Util.getFloatValueFromPropertyMap(rbMap, "Mobility.AV.Share")); + if(avShare>0) + avAvailable=true; + + random = new MersenneTwister(1000001); + + } + + /** + * Read the file and return the TableDataSet. + * + * @param fileName + * @return data + */ + private TableDataSet readFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet data; + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + data = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + return data; + } + + /** + * Generate and attribute visitor tours + */ + public void generateVisitorTours() + { + + // calculate total number of cross border tours + ArrayList tourList = new ArrayList(); + + int rows = mgraData.getRowCount(); + + int tourCount = 0; + int personalCount = 0; + int businessCount = 0; + for (int i = 1; i <= rows; ++i) + { + + float hotelRooms = mgraData.getValueAt(i, "HotelRoomTotal"); + float households = mgraData.getValueAt(i, "hh"); + int mgraNumber = (int) mgraData.getValueAt(i, "mgra"); + + float hotelVisitorParties = hotelRooms * occupancyRate; + float householdVisitorParties = households * householdRate; + + int businessParties = Math.round(hotelVisitorParties * businessHotelPercent); + int personalParties = Math.round(hotelVisitorParties * (1.0f - businessHotelPercent)); + + businessParties += Math.round(householdVisitorParties * businessHouseholdPercent); + personalParties += Math.round(householdVisitorParties + * (1.0f - businessHouseholdPercent)); + + personalCount += personalParties; + businessCount += businessParties; + + // generate a tour for each business party + for (int j = 0; j < businessParties; ++j) + { + + int[] tourPurposes = simulateTours(businessTourFrequency); + + for (int k = 0; k < tourPurposes.length; ++k) + { + VisitorTour tour = new VisitorTour(tourCount + 1000001); + tour.setID(tourCount + 1); + tour.setOriginMGRA(mgraNumber); + tour.setSegment(modelStructure.BUSINESS); + tour.setPurpose((byte) tourPurposes[k]); + calculateSize(tour); + calculateAutoAvailability(tour); + calculateIncome(tour); + tourList.add(tour); + ++tourCount; + } + } + + // generate a tour for each personal party + for (int j = 0; j < personalParties; ++j) + { + + int[] tourPurposes = simulateTours(personalTourFrequency); + + for (int k = 0; k < tourPurposes.length; ++k) + { + VisitorTour tour = new VisitorTour(tourCount + 1000001); + tour.setID(tourCount + 1); + tour.setOriginMGRA(mgraNumber); + tour.setSegment(modelStructure.PERSONAL); + tour.setPurpose((byte) tourPurposes[k]); + calculateSize(tour); + calculateAutoAvailability(tour); + calculateIncome(tour); + tourList.add(tour); + ++tourCount; + } + } + + } + + if (tourList.isEmpty()) + { + logger.error("Visitor tour list is empty!!"); + throw new RuntimeException(); + } + + tours = new VisitorTour[tourList.size()]; + for (int i = 0; i < tours.length; ++i) + tours[i] = tourList.get(i); + + logger.info("Total personal parties: " + personalCount); + logger.info("Total business parties: " + businessCount); + + logger.info("Total visitor tours: " + tourCount); + + } + + /** + * Calculate the number of tours for this travel party, by purpose. Return + * an array whose length equals the number of tours, where each element is + * the purpose of the tour. + * + * @param tourFrequency + * A tableDataSet with the following fields + * WorkTours,RecreationTours,DiningTours,Percent + * @return An array dimensioned to number of tours to generate, with the + * purpose of each. + */ + private int[] simulateTours(TableDataSet tourFrequency) + { + + int[] tourPurposes; + double rand = random.nextDouble(); + + double cumProb = 0.0; + int row = -1; + for (int i = 0; i < tourFrequency.getRowCount(); ++i) + { + + float percent = tourFrequency.getValueAt(i + 1, "Percent"); + cumProb += percent; + if (rand < cumProb) + { + row = i + 1; + break; + } + } + int workTours = (int) tourFrequency.getValueAt(row, "WorkTours"); + int recTours = (int) tourFrequency.getValueAt(row, "RecreationTours"); + int diningTours = (int) tourFrequency.getValueAt(row, "DiningTours"); + + int totalTours = workTours + recTours + diningTours; + tourPurposes = new int[totalTours]; + + int workSet = 0; + int recSet = 0; + int diningSet = 0; + for (int j = 0; j < tourPurposes.length; ++j) + { + + if (workTours > 0 && workSet < workTours) + { + tourPurposes[j] = modelStructure.WORK; + ++workSet; + } else if (recTours > 0 && recSet < recTours) + { + tourPurposes[j] = modelStructure.RECREATION; + ++recSet; + } else if (diningTours > 0 && diningSet < diningTours) + { + tourPurposes[j] = modelStructure.DINING; + ++diningSet; + } + } + return tourPurposes; + } + + /** + * Calculate the size of the tour and store in tour object. + * + * @param tour + */ + private void calculateSize(VisitorTour tour) + { + + byte purp = tour.getPurpose(); + String purpString = modelStructure.VISITOR_PURPOSES[purp]; + String columnName = purpString.toLowerCase(); + + double cumProb = 0; + double rand = tour.getRandom(); + byte size = -1; + int rowCount = partySizeFrequency.getRowCount(); + for (int i = 1; i <= rowCount; ++i) + { + cumProb += partySizeFrequency.getValueAt(i, columnName); + if (rand < cumProb) + { + size = (byte) partySizeFrequency.getValueAt(i, "PartySize"); + break; + } + } + if (size == -1) + { + logger.error("Error attempting to choose party size for visitor tour " + tour.getID()); + throw new RuntimeException(); + } + tour.setNumberOfParticipants(size); + } + + /** + * Calculate whether autos are available for this tour. + * + * @param tour + */ + private void calculateAutoAvailability(VisitorTour tour) + { + + byte purp = tour.getPurpose(); + String purpString = modelStructure.VISITOR_PURPOSES[purp]; + String columnName = purpString.toLowerCase(); + + double rand = tour.getRandom(); + boolean autoAvailable = false; + double probability = autoAvailableFrequency.getValueAt(1, columnName); + if (rand < probability) autoAvailable = true; + + tour.setAutoAvailable(autoAvailable ? 1 : 0); + } + + /** + * Calculate the income of the tour + * + * @param tour + */ + private void calculateIncome(VisitorTour tour) + { + byte segment = tour.getSegment(); + String segmentString = modelStructure.VISITOR_SEGMENTS[segment]; + String columnName = segmentString.toLowerCase(); + + double rand = tour.getRandom(); + int income = -1; + double cumProb = 0; + int rowCount = incomeFrequency.getRowCount(); + for (int i = 1; i <= rowCount; ++i) + { + cumProb += incomeFrequency.getValueAt(i, columnName); + if (rand < cumProb) + { + income = (int) incomeFrequency.getValueAt(i, "Income"); + break; + } + } + if (income == -1) + { + logger.error("Error attempting to choose party size for visitor tour " + tour.getID()); + throw new RuntimeException(); + } + tour.setIncome(income); + + } + + /** + * Create a text file and write all records to the file. + * + */ + public void writeOutputFile(HashMap rbMap) + { + + // Open file and print header + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String tourFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "visitor.tour.output.file"); + String tripFileName = directory + + Util.getStringValueFromPropertyMap(rbMap, "visitor.trip.output.file"); + + logger.info("Writing visitor tours to file " + tourFileName); + logger.info("Writing visitor trips to file " + tripFileName); + + PrintWriter tourWriter = null; + try + { + tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tourFileName + " for writing\n"); + throw new RuntimeException(); + } + String tourHeaderString = new String( + "id,segment,purpose,autoAvailable,partySize,income,departTime,arriveTime,originMGRA,destinationMGRA,tourMode,avAvailable,outboundStops,inboundStops,valueOfTime\n"); + tourWriter.print(tourHeaderString); + + PrintWriter tripWriter = null; + try + { + tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); + } catch (IOException e) + { + logger.fatal("Could not open file " + tripFileName + " for writing\n"); + throw new RuntimeException(); + } + String tripHeaderString = new String( + "tourID,tripID,originPurp,destPurp,originMGRA,destinationMGRA,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,avAvailable,boardingTap,alightingTap,set,valueOfTime,partySize," + +"micro_walkMode,micro_trnAcc,micro_trnEgr,parkingCost\n"); + tripWriter.print(tripHeaderString); + + // Iterate through the array, printing records to the file + for (int i = 0; i < tours.length; ++i) + { + + VisitorTour tour = tours[i]; + + if (seek && tour.getID() != traceId) continue; + + VisitorTrip[] trips = tours[i].getTrips(); + + if (trips == null) continue; + + writeTour(tour, tourWriter); + + for (int j = 0; j < trips.length; ++j) + { + writeTrip(tour, trips[j], j + 1, tripWriter); + } + } + + tourWriter.close(); + tripWriter.close(); + + } + + /** + * Write the tour to the PrintWriter + * + * @param tour + * @param writer + */ + private void writeTour(VisitorTour tour, PrintWriter writer) + { + String record = new String(tour.getID() + "," + tour.getSegment() + "," + tour.getPurpose() + + "," + tour.getAutoAvailable() + "," + tour.getNumberOfParticipants() + "," + + tour.getIncome() + "," + tour.getDepartTime() + "," + tour.getArriveTime() + "," + + tour.getOriginMGRA() + "," + tour.getDestinationMGRA() + "," + tour.getTourMode() + "," + + (avAvailable?1:0) + + "," + tour.getNumberOutboundStops() + "," + tour.getNumberInboundStops() + + "," + String.format("%9.2f",tour.getValueOfTime())+ "\n"); + writer.print(record); + + } + + /** + * Write the trip to the PrintWriter + * + * @param tour + * @param trip + * @param tripNumber + * @param writer + */ + private void writeTrip(VisitorTour tour, VisitorTrip trip, int tripNumber, PrintWriter writer) + { + String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginPurpose() + + "," + trip.getDestinationPurpose() + "," + trip.getOriginMgra() + "," + + trip.getDestinationMgra() + "," + trip.isInbound() + "," + + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() + + "," + trip.getPeriod() + "," + trip.getTripMode() + "," + (avAvailable?1:0) +"," + + trip.getBoardTap() + "," + trip.getAlightTap() + "," + trip.getSet() + + "," + String.format("%9.2f",trip.getValueOfTime())+ "," + tour.getNumberOfParticipants()+"," + +trip.getMicromobilityWalkMode()+"," +trip.getMicromobilityAccessMode()+"," +trip.getMicromobilityEgressMode() + + "," + String.format("%9.2f", trip.getParkingCost()) +"\n"); + writer.print(record); + } + + + /** + * @return the parties + */ + public VisitorTour[] getTours() + { + return tours; + } + + public static void main(String[] args) + { + + String propertiesFile = null; + HashMap pMap; + + logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", + CtrampApplication.VERSION)); + + logger.info(String.format("Running Cross Border Model Tour Manager")); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + VisitorTourManager apm = new VisitorTourManager(pMap); + apm.generateVisitorTours(); + apm.writeOutputFile(pMap); + + logger.info("Cross-Border Tour Manager successfully completed!"); + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java new file mode 100644 index 0000000..3fa9b3c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java @@ -0,0 +1,570 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.TourModeChoiceDMU; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class VisitorTourModeChoiceDMU + implements Serializable, VariableTable +{ + protected transient Logger logger = Logger.getLogger(VisitorTourModeChoiceDMU.class); + + public static final int WTW = McLogsumsCalculator.WTW; + public static final int WTD = McLogsumsCalculator.WTD; + public static final int DTW = McLogsumsCalculator.DTW; + protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected float tourDepartPeriod; + protected float tourArrivePeriod; + protected double origDuDen; + protected double origEmpDen; + protected double origTotInt; + protected double destDuDen; + protected double destEmpDen; + protected double destTotInt; + + protected int partySize; + protected int autoAvailable; + protected int income; + protected int tourPurpose; + + protected float pTazTerminalTime; + protected float aTazTerminalTime; + + protected double nmWalkTimeOut; + protected double nmWalkTimeIn; + protected double nmBikeTimeOut; + protected double nmBikeTimeIn; + protected double lsWgtAvgCostM; + protected double lsWgtAvgCostD; + protected double lsWgtAvgCostH; + + protected double[][] transitLogSum; + protected float origTaxiWaitTime; + protected float destTaxiWaitTime; + protected float origSingleTNCWaitTime; + protected float destSingleTNCWaitTime; + protected float origSharedTNCWaitTime; + protected float destSharedTNCWaitTime; + + public VisitorTourModeChoiceDMU(VisitorModelStructure modelStructure, Logger aLogger) + { + if (aLogger == null) aLogger = Logger.getLogger(TourModeChoiceDMU.class); + logger = aLogger; + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + //accEgr by in/outbound + transitLogSum = new double[NUM_ACC_EGR][NUM_DIR]; + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + public void setLsWgtAvgCostM(double cost) + { + lsWgtAvgCostM = cost; + } + + public void setLsWgtAvgCostD(double cost) + { + lsWgtAvgCostD = cost; + } + + public void setLsWgtAvgCostH(double cost) + { + lsWgtAvgCostH = cost; + } + + public double getMonthlyParkingCost() + { + return lsWgtAvgCostM; + } + + public double getDailyParkingCost() + { + return lsWgtAvgCostD; + } + + public double getHourlyParkingCost() + { + return lsWgtAvgCostH; + } + + public float getTimeOutbound() + { + return tourDepartPeriod; + } + + public float getTimeInbound() + { + return tourArrivePeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(float tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(float tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + public void setOrigDuDen(double arg) + { + origDuDen = arg; + } + + public void setOrigEmpDen(double arg) + { + origEmpDen = arg; + } + + public void setOrigTotInt(double arg) + { + origTotInt = arg; + } + + public void setDestDuDen(double arg) + { + destDuDen = arg; + } + + public void setDestEmpDen(double arg) + { + destEmpDen = arg; + } + + public void setDestTotInt(double arg) + { + destTotInt = arg; + } + + public int getTourPurpose() + { + return tourPurpose; + } + + public void setTourPurpose(int tourPurpose) + { + this.tourPurpose = tourPurpose; + } + + public double getODUDen() + { + return origDuDen; + } + + public double getOEmpDen() + { + return origEmpDen; + } + + public double getOTotInt() + { + return origTotInt; + } + + public double getDDUDen() + { + return destDuDen; + } + + public double getDEmpDen() + { + return destEmpDen; + } + + public double getDTotInt() + { + return destTotInt; + } + + public void setNmWalkTimeOut(double nmWalkTime) + { + nmWalkTimeOut = nmWalkTime; + } + + public double getNm_walkTime_out() + { + return nmWalkTimeOut; + } + + public void setNmWalkTimeIn(double nmWalkTime) + { + nmWalkTimeIn = nmWalkTime; + } + + public double getNm_walkTime_in() + { + return nmWalkTimeIn; + } + + public void setNmBikeTimeOut(double nmBikeTime) + { + nmBikeTimeOut = nmBikeTime; + } + + public double getNm_bikeTime_out() + { + return nmBikeTimeOut; + } + + public void setNmBikeTimeIn(double nmBikeTime) + { + nmBikeTimeIn = nmBikeTime; + } + + public double getNm_bikeTime_in() + { + return nmBikeTimeIn; + } + + public void setPTazTerminalTime(float time) + { + pTazTerminalTime = time; + } + + public void setATazTerminalTime(float time) + { + aTazTerminalTime = time; + } + + public double getPTazTerminalTime() + { + return pTazTerminalTime; + } + + public double getATazTerminalTime() + { + return aTazTerminalTime; + } + + public int getPartySize() + { + return partySize; + } + + public void setPartySize(int partySize) + { + this.partySize = partySize; + } + + public int getAutoAvailable() + { + return autoAvailable; + } + + public void setAutoAvailable(int autoAvailable) + { + this.autoAvailable = autoAvailable; + } + + public int getIncome() + { + return income; + } + + public void setIncome(int income) + { + this.income = income; + } + + public void setTransitLogSum(int accEgr, boolean inbound, double value){ + transitLogSum[accEgr][inbound == true ? 1 : 0] = value; + } + + protected double getTransitLogSum(int accEgr,boolean inbound){ + return transitLogSum[accEgr][inbound == true ? 1 : 0]; + } + + + + public float getOrigTaxiWaitTime() { + return origTaxiWaitTime; + } + + + + public void setOrigTaxiWaitTime(float origTaxiWaitTime) { + this.origTaxiWaitTime = origTaxiWaitTime; + } + + + + public float getDestTaxiWaitTime() { + return destTaxiWaitTime; + } + + + + public void setDestTaxiWaitTime(float destTaxiWaitTime) { + this.destTaxiWaitTime = destTaxiWaitTime; + } + + + + public float getOrigSingleTNCWaitTime() { + return origSingleTNCWaitTime; + } + + + + public void setOrigSingleTNCWaitTime(float origSingleTNCWaitTime) { + this.origSingleTNCWaitTime = origSingleTNCWaitTime; + } + + + + public float getDestSingleTNCWaitTime() { + return destSingleTNCWaitTime; + } + + + + public void setDestSingleTNCWaitTime(float destSingleTNCWaitTime) { + this.destSingleTNCWaitTime = destSingleTNCWaitTime; + } + + + + public float getOrigSharedTNCWaitTime() { + return origSharedTNCWaitTime; + } + + + + public void setOrigSharedTNCWaitTime(float origSharedTNCWaitTime) { + this.origSharedTNCWaitTime = origSharedTNCWaitTime; + } + + + + public float getDestSharedTNCWaitTime() { + return destSharedTNCWaitTime; + } + + + + public void setDestSharedTNCWaitTime(float destSharedTNCWaitTime) { + this.destSharedTNCWaitTime = destSharedTNCWaitTime; + } + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTimeOutbound", 0); + methodIndexMap.put("getTimeInbound", 1); + methodIndexMap.put("getPTazTerminalTime", 14); + methodIndexMap.put("getATazTerminalTime", 15); + methodIndexMap.put("getODUDen", 16); + methodIndexMap.put("getOEmpDen", 17); + methodIndexMap.put("getOTotInt", 18); + methodIndexMap.put("getDDUDen", 19); + methodIndexMap.put("getDEmpDen", 20); + methodIndexMap.put("getDTotInt", 21); + methodIndexMap.put("getMonthlyParkingCost", 23); + methodIndexMap.put("getDailyParkingCost", 24); + methodIndexMap.put("getHourlyParkingCost", 25); + methodIndexMap.put("getPartySize", 30); + methodIndexMap.put("getAutoAvailable", 31); + methodIndexMap.put("getIncome", 32); + methodIndexMap.put("getTourPurpose", 33); + + methodIndexMap.put("getIvtCoeff", 56); + methodIndexMap.put("getCostCoeff", 57); + methodIndexMap.put("getWalkSetLogSum", 59); + methodIndexMap.put("getPnrSetLogSum", 60); + methodIndexMap.put("getKnrSetLogSum", 61); + + methodIndexMap.put( "getOrigTaxiWaitTime", 70 ); + methodIndexMap.put( "getDestTaxiWaitTime", 71 ); + methodIndexMap.put( "getOrigSingleTNCWaitTime", 72 ); + methodIndexMap.put( "getDestSingleTNCWaitTime", 73 ); + methodIndexMap.put( "getOrigSharedTNCWaitTime", 74 ); + methodIndexMap.put( "getDestSharedTNCWaitTime", 75 ); + + methodIndexMap.put("getNm_walkTime_out", 90); + methodIndexMap.put("getNm_walkTime_in", 91); + methodIndexMap.put("getNm_bikeTime_out", 92); + methodIndexMap.put("getNm_bikeTime_in", 93); + + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + case 0: + returnValue = getTimeOutbound(); + break; + case 1: + returnValue = getTimeInbound(); + break; + case 14: + returnValue = getPTazTerminalTime(); + break; + case 15: + returnValue = getATazTerminalTime(); + break; + case 16: + returnValue = getODUDen(); + break; + case 17: + returnValue = getOEmpDen(); + break; + case 18: + returnValue = getOTotInt(); + break; + case 19: + returnValue = getDDUDen(); + break; + case 20: + returnValue = getDEmpDen(); + break; + case 21: + returnValue = getDTotInt(); + break; + case 23: + returnValue = getMonthlyParkingCost(); + break; + case 24: + returnValue = getDailyParkingCost(); + break; + case 25: + returnValue = getHourlyParkingCost(); + break; + case 30: + returnValue = getPartySize(); + break; + case 31: + returnValue = getAutoAvailable(); + break; + case 32: + returnValue = getIncome(); + break; + case 33: + returnValue = getTourPurpose(); + break; + case 59: + returnValue = getTransitLogSum(WTW, true) + getTransitLogSum(WTW, false); + break; + case 60: + returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); + break; + case 61: + returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); + break; + case 70: return getOrigTaxiWaitTime(); + case 71: return getDestTaxiWaitTime(); + case 72: return getOrigSingleTNCWaitTime(); + case 73: return getDestSingleTNCWaitTime(); + case 74: return getOrigSharedTNCWaitTime(); + case 75: return getDestSharedTNCWaitTime(); + case 90: + returnValue = getNm_walkTime_out(); + break; + case 91: + returnValue = getNm_walkTime_in(); + break; + case 92: + returnValue = getNm_bikeTime_out(); + break; + case 93: + returnValue = getNm_bikeTime_in(); + break; + default: + logger.error("method number = " + variableIndex + " not found"); + throw new RuntimeException("method number = " + variableIndex + " not found"); + } + + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java new file mode 100644 index 0000000..87572d1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java @@ -0,0 +1,399 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; +import java.util.Random; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; +import org.sandag.abm.ctramp.TourModeChoiceDMU; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class VisitorTourModeChoiceModel + implements Serializable +{ + + private transient Logger logger = Logger.getLogger("visitorModel"); + + public static final boolean DEBUG_BEST_PATHS = false; + + private MgraDataManager mgraManager; + + protected static final int OUT = McLogsumsCalculator.OUT; + protected static final int IN = McLogsumsCalculator.IN; + protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; + + private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "visitor.mc.uec.file"; + private static final String PROPERTIES_UEC_TOUR_DATA_SHEET = "visitor.mc.data.page"; + private static final String PROPERTIES_UEC_TOUR_MODEL_SHEET = "visitor.mc.model.page"; + + private ChoiceModelApplication mcModel; + private VisitorTourModeChoiceDMU mcDmuObject; + private TripModeChoiceDMU tripDmuObject; + private McLogsumsCalculator logsumHelper; + + private VisitorModelStructure modelStructure; + + private String tourCategory; + + private String[] modeAltNames; + + private boolean saveUtilsProbsFlag = false; + private AutoTazSkimsCalculator tazDistanceCalculator; + + //added for TNC and Taxi modes + TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public VisitorTourModeChoiceModel(HashMap propertyMap, + VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + + mgraManager = MgraDataManager.getInstance(propertyMap); + modelStructure = myModelStructure; + this.tazDistanceCalculator = tazDistanceCalculator; + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + tripDmuObject = new TripModeChoiceDMU(modelStructure,logger); + mcDmuObject = dmuFactory.getVisitorTourModeChoiceDMU(); + setupModeChoiceModelApplicationArray(propertyMap); + + tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); + tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); + + } + + /** + * Set up the mode choice model. + * + * @param propertyMap + */ + private void setupModeChoiceModelApplicationArray(HashMap propertyMap) + { + + logger.info(String.format("setting up visitor tour mode choice model.")); + + // locate the individual mandatory tour mode choice model UEC + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String mcUecFile = Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_TOUR_MODE_CHOICE); + mcUecFile = uecPath + mcUecFile; + + logger.info("Will read mcUECFile " + mcUecFile); + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_TOUR_DATA_SHEET)); + int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_TOUR_MODEL_SHEET)); + + // default is to not save the tour mode choice utils and probs for each + // tour + String saveUtilsProbsString = propertyMap + .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); + if (saveUtilsProbsString != null) + { + if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; + } + + mcModel = new ChoiceModelApplication(mcUecFile, modelPage, dataPage, propertyMap, + (VariableTable) mcDmuObject); + modeAltNames = mcModel.getAlternativeNames(); + + } + + public double getModeChoiceLogsum(VisitorTour tour, Logger modelLogger, + String choiceModelDescription, String decisionMakerLabel) + { + + setDmuAttributes(tour); + + // log headers to traceLogger + if (tour.getDebugChoiceModels()) + { + mcModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, decisionMakerLabel); + } + + + mcModel.computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); + double logsum = mcModel.getLogsum(); + + // write UEC calculation results to separate model specific log file + if (tour.getDebugChoiceModels()) + { + String loggingHeader = String.format("%s %s", choiceModelDescription, + decisionMakerLabel); + mcModel.logUECResults(modelLogger, loggingHeader); + modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); + modelLogger.info(""); + modelLogger.info(""); + } + + return logsum; + + } + + /** + * Set the DMU attributes for the tour. + * + * @param tour + */ + private void setDmuAttributes(VisitorTour tour) + { + + // update the MC dmuObjects for this person + int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); + int destinationTaz = mgraManager.getTaz(tour.getDestinationMGRA()); + mcDmuObject.setDmuIndexValues(tour.getID(), originTaz, originTaz, destinationTaz, + tour.getDebugChoiceModels()); + + mcDmuObject.setTourDepartPeriod(tour.getDepartTime()); + mcDmuObject.setTourArrivePeriod(tour.getArriveTime()); + mcDmuObject.setIncome((byte) tour.getIncome()); + mcDmuObject.setAutoAvailable(tour.getAutoAvailable()); + mcDmuObject.setPartySize(tour.getNumberOfParticipants()); + mcDmuObject.setTourPurpose(tour.getPurpose()); + double ivtCoeff = -0.015; + double costCoeff = -0.0017; + tripDmuObject.setIvtCoeff(ivtCoeff); + tripDmuObject.setCostCoeff(costCoeff); + + logsumHelper.setNmTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), + tour.getDepartTime(),tour.getDebugChoiceModels()); + double nmWalkTimeOut = tripDmuObject.getNm_walkTime(); + double nmBikeTimeOut = tripDmuObject.getNm_bikeTime(); + mcDmuObject.setNmWalkTimeOut(nmWalkTimeOut); + mcDmuObject.setNmBikeTimeOut(nmBikeTimeOut); + logsumHelper.setNmTripMcDmuAttributes( tripDmuObject, tour.getDestinationMGRA(), tour.getOriginMGRA(), + tour.getArriveTime(),tour.getDebugChoiceModels()); + double nmWalkTimeIn = tripDmuObject.getNm_walkTime(); + double nmBikeTimeIn = tripDmuObject.getNm_bikeTime(); + mcDmuObject.setNmWalkTimeOut(nmWalkTimeIn); + mcDmuObject.setNmBikeTimeOut(nmBikeTimeIn); + + + double walkTransitLogsumOut = -999.0; + double driveTransitLogsumOut = -999.0; + double walkTransitLogsumIn = -999.0; + double driveTransitLogsumIn = -999.0; + + // walk-transit out logsum + logsumHelper.setWtwTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), + tour.getDepartTime(),tour.getDebugChoiceModels()); + + walkTransitLogsumOut = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + // walk-transit in logsum + logsumHelper.setWtwTripMcDmuAttributes( tripDmuObject,tour.getDestinationMGRA(), tour.getOriginMGRA(), + tour.getArriveTime(),tour.getDebugChoiceModels()); + + walkTransitLogsumIn = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + //drive-transit out logsum + logsumHelper.setDtwTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), + tour.getDepartTime(),tour.getDebugChoiceModels()); + + driveTransitLogsumOut = tripDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); + + //drive-transit in logsum + logsumHelper.setWtdTripMcDmuAttributes( tripDmuObject, tour.getDestinationMGRA(),tour.getOriginMGRA(), + tour.getArriveTime(),tour.getDebugChoiceModels()); + + driveTransitLogsumIn = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); + + mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTW,false,walkTransitLogsumOut); + mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTW,true,walkTransitLogsumIn); + mcDmuObject.setTransitLogSum(McLogsumsCalculator.DTW,false,driveTransitLogsumOut); + mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTD,true,driveTransitLogsumIn); + + float SingleTNCWaitTimeOrig = 0; + float SingleTNCWaitTimeDest = 0; + float SharedTNCWaitTimeOrig = 0; + float SharedTNCWaitTimeDest = 0; + float TaxiWaitTimeOrig = 0; + float TaxiWaitTimeDest = 0; + float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getOriginMGRA()); + float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getDestinationMGRA()); + + double rnum = tour.getRandom(); + SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); + SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); + SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); + TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); + TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); + + mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); + mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); + mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); + mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); + mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); + mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); + + + } + + /** + * Use to choose tour mode and set result in tour object. + * + * @param tour + * The crossborder tour + */ + public void chooseTourMode(VisitorTour tour) + { + + byte tourMode = (byte) getModeChoice(tour); + tour.setTourMode(tourMode); + } + + /** + * Get the choice of mode for the tour, and return as an integer (don't + * store in tour object) + * + * @param tour + * @return + */ + private int getModeChoice(VisitorTour tour) + { + + String choiceModelDescription = ""; + String decisionMakerLabel = ""; + String loggingHeader = ""; + String separator = ""; + String purposeName = modelStructure.VISITOR_PURPOSES[tour.getPurpose()]; + + if (tour.getDebugChoiceModels()) + { + + tour.logTourObject(logger, 100); + logger.info("Logging tour mode choice model"); + } + + setDmuAttributes(tour); + + mcModel.computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + mcModel.logUECResults(logger, "Visitor tour mode choice model"); + + double rn = tour.getRandom(); + + // if the choice model has at least one available alternative, make + // choice. + int chosen; + if (mcModel.getAvailabilityCount() > 0) + { + + chosen = mcModel.getChoiceResult(rn); + + //value of time; lookup vot from the UEC + UtilityExpressionCalculator uec = mcModel.getUEC(); + int votIndex = uec.lookupVariableIndex("vot"); + float vot = (float) uec.getValueForIndex(votIndex); + + tour.setValueOfTime(vot); + + } else + { + + tour.logTourObject(logger, loggingHeader.length()); + + mcModel.logUECResults(logger, loggingHeader); + logger.info(""); + logger.info(""); + + logger.error(String + .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", + tour.getID(), tourCategory)); + throw new RuntimeException(); + } + + // debug output + if (tour.getDebugChoiceModels()) + { + + double[] utilities = mcModel.getUtilities(); // 0s-indexing + double[] probabilities = mcModel.getProbabilities(); // 0s-indexing + boolean[] availabilities = mcModel.getAvailabilities(); // 1s-indexing + String[] altNames = mcModel.getAlternativeNames(); // 0s-indexing + + logger.info("Tour Id: " + tour.getID()); + logger.info("Alternative Utility Probability CumProb"); + logger.info("-------------------- -------------- -------------- --------------"); + + double cumProb = 0.0; + for (int k = 0; k < mcModel.getNumberOfAlternatives(); k++) + { + cumProb += probabilities[k]; + String altString = String.format("%-3d %s", k + 1, altNames[k]); + logger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, + availabilities[k + 1], utilities[k], probabilities[k], cumProb)); + } + + logger.info(" "); + String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); + logger.info(String.format("Choice: %s, with rn=%.8f", altString, rn)); + + logger.info(separator); + logger.info(""); + logger.info(""); + + // write choice model alternative info to log file + mcModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); + mcModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); + mcModel.logLogitCalculations(choiceModelDescription, decisionMakerLabel); + + } + + if (saveUtilsProbsFlag) + { + + // get the utilities and probabilities arrays for the tour mode + // choice + // model for this tour and save them to the tour object + double[] dUtils = mcModel.getUtilities(); + double[] dProbs = mcModel.getProbabilities(); + + float[] utils = new float[dUtils.length]; + float[] probs = new float[dUtils.length]; + for (int k = 0; k < dUtils.length; k++) + { + utils[k] = (float) dUtils[k]; + probs[k] = (float) dProbs[k]; + } + + // tour.setTourModalUtilities(utils); + // tour.setTourModalProbabilities(probs); + + } + + return chosen; + + } + + public String[] getModeAltNames(int purposeIndex) + { + return modeAltNames; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java new file mode 100644 index 0000000..f12d379 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java @@ -0,0 +1,192 @@ +package org.sandag.abm.visitor; + +import java.io.File; +import java.io.IOException; +import java.util.HashMap; +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.Util; +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; + +/** + * This class is the TOD choice model for cross border tours. It is currently + * based on a static probability distribution stored in an input file, and + * indexed into by purpose. + * + * @author Freedman + * + */ +public class VisitorTourTimeOfDayChoiceModel +{ + private transient Logger logger = Logger.getLogger("visitorModel"); + + private double[][] cumProbability; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + private int[][] outboundPeriod; // by + // purpose, + // alternative: + // outbound + // period + private int[][] returnPeriod; // by + // purpose, + // alternative: + // return + // period + VisitorModelStructure modelStructure; + + /** + * Constructor. + */ + public VisitorTourTimeOfDayChoiceModel(HashMap rbMap) + { + + String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); + String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, + "visitor.tour.tod.file"); + stationDiurnalFile = directory + stationDiurnalFile; + + modelStructure = new VisitorModelStructure(); + + readTODFile(stationDiurnalFile); + + } + + /** + * Read the TOD distribution in the file and populate the arrays. + * + * @param fileName + */ + private void readTODFile(String fileName) + { + + logger.info("Begin reading the data in file " + fileName); + TableDataSet probabilityTable; + + try + { + OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); + probabilityTable = csvFile.readFile(new File(fileName)); + } catch (IOException e) + { + throw new RuntimeException(e); + } + + logger.info("End reading the data in file " + fileName); + + logger.info("Begin calculating tour TOD probability distribution"); + + int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 + int periods = modelStructure.TIME_PERIODS; // start at 1 + int periodCombinations = periods * (periods + 1) / 2; + + cumProbability = new double[purposes][periodCombinations]; // by + // purpose, + // alternative: + // cumulative + // probability + // distribution + outboundPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // outbound + // period + returnPeriod = new int[purposes][periodCombinations]; // by purpose, + // alternative: + // return period + + // fill up arrays + int rowCount = probabilityTable.getRowCount(); + int lastPurpose = -99; + double cumProb = 0; + int alt = 0; + for (int row = 1; row <= rowCount; ++row) + { + + int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); + int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); + int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); + + // continue if return period before outbound period + if (retPer < outPer) continue; + + // reset if new purpose + if (purpose != lastPurpose) + { + + // log cumulative probability just in case + if (lastPurpose != -99) + logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); + cumProb = 0; + alt = 0; + } + + // calculate cumulative probability and store in array + cumProb += probabilityTable.getValueAt(row, "Percent"); + cumProbability[purpose][alt] = cumProb; + outboundPeriod[purpose][alt] = outPer; + returnPeriod[purpose][alt] = retPer; + + ++alt; + + lastPurpose = purpose; + } + + logger.info("End calculating tour TOD probability distribution"); + + } + + /** + * Calculate tour time of day for the tour. + * + * @param tour + * A cross border tour (with purpose) + */ + public void calculateTourTOD(VisitorTour tour) + { + + int purpose = tour.getPurpose(); + double random = tour.getRandom(); + + int depart = -1; + int arrive = -1; + if (tour.getDebugChoiceModels()) + { + logger.info("Choosing tour time of day for purpose " + + modelStructure.VISITOR_PURPOSES[purpose] + " using random number " + random); + tour.logTourObject(logger, 100); + } + + for (int i = 0; i < cumProbability[purpose].length; ++i) + { + + if (random < cumProbability[purpose][i]) + { + depart = outboundPeriod[purpose][i]; + arrive = returnPeriod[purpose][i]; + tour.setDepartTime(depart); + tour.setArriveTime(arrive); + break; + } + } + if((depart ==-1)||(arrive==-1)){ + logger.fatal("Error: did not find outbound or return period for tour"); + logger.fatal("Depart period, arrive period = "+depart+","+arrive); + logger.fatal("Random number: "+random); + tour.logTourObject(logger,100); + throw new RuntimeException(); + } + + + if (tour.getDebugChoiceModels()) + { + logger.info(""); + logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " + + tour.getArriveTime()); + logger.info(""); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java new file mode 100644 index 0000000..32f7dc8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java @@ -0,0 +1,518 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; + +public class VisitorTrip + implements Serializable +{ + + private int originMgra; + private int destinationMgra; + private int tripMode; + private byte originPurpose; + private byte destinationPurpose; + private byte period; + private boolean inbound; + private boolean firstTrip; + private boolean lastTrip; + private boolean originIsTourDestination; + private boolean destinationIsTourDestination; + byte micromobilityWalkMode; + byte micromobilityAccessMode; + byte micromobilityEgressMode; + float micromobilityWalkLogsum; + float micromobilityAccessLogsum; + float micromobilityEgressLogsum; + + float parkingCost; + + private int boardTap; + private int alightTap; + private int set = -1; + + private float valueOfTime; + + /** + * Default constructor; nothing initialized. + */ + public VisitorTrip() + { + + } + + /** + * Create a cross border trip from a tour leg (no stops). + * + * @param tour + * The tour. + * @param outbound + * Outbound direction + */ + public VisitorTrip(VisitorTour tour, boolean outbound) + { + + initializeFromTour(tour, outbound); + } + + /** + * Initilize from the tour. + * + * @param tour + * The tour. + * @param outbound + * Outbound direction. + */ + public void initializeFromTour(VisitorTour tour, boolean outbound) + { + // Note: mode is unknown + if (outbound) + { + this.originMgra = tour.getOriginMGRA(); + this.destinationMgra = tour.getDestinationMGRA(); + this.originPurpose = -1; + this.destinationPurpose = tour.getPurpose(); + this.period = (byte) tour.getDepartTime(); + this.inbound = false; + this.firstTrip = true; + this.lastTrip = false; + this.originIsTourDestination = false; + this.destinationIsTourDestination = true; + } else + { + this.originMgra = tour.getDestinationMGRA(); + this.destinationMgra = tour.getOriginMGRA(); + this.originPurpose = tour.getPurpose(); + this.destinationPurpose = -1; + this.period = (byte) tour.getArriveTime(); + this.inbound = true; + this.firstTrip = false; + this.lastTrip = true; + this.originIsTourDestination = true; + this.destinationIsTourDestination = false; + } + + } + + /** + * Create a visitor trip from a tour\stop. Note: trip mode is unknown. Stop + * period is only known for first, last stop on tour. + * + * @param tour + * The tour. + * @param stop + * The stop + */ + public VisitorTrip(VisitorTour tour, VisitorStop stop, boolean toStop) + { + + initializeFromStop(tour, stop, toStop); + } + + /** + * Initialize from stop attributes. A trip will be created to the stop if + * toStop is true, else a trip will be created from the stop. Use after all + * stop locations are known, or else reset the stop origin and destination + * mgras accordingly after using. + * + * @param tour + * @param stop + * @param toStop + */ + public void initializeFromStop(VisitorTour tour, VisitorStop stop, boolean toStop) + { + + this.inbound = stop.isInbound(); + this.destinationIsTourDestination = false; + this.originIsTourDestination = false; + + // if trip to stop, destination is stop mgra; else origin is stop mgra + if (toStop) + { + this.destinationMgra = stop.getMgra(); + this.destinationPurpose = stop.getPurpose(); + } else + { + this.originMgra = stop.getMgra(); + this.originPurpose = stop.getPurpose(); + } + VisitorStop[] stops; + + if (!inbound) stops = tour.getOutboundStops(); + else stops = tour.getInboundStops(); + + // if outbound, and trip is to stop + if (!inbound && toStop) + { + + // first trip on outbound journey, origin is tour origin + if (stop.getId() == 0) + { + this.originMgra = tour.getOriginMGRA(); + this.originPurpose = -1; + this.period = (byte) tour.getDepartTime(); + } else + { + // not first trip on outbound journey, origin is last stop + this.originMgra = stops[stop.getId() - 1].getMgra(); // last + // stop + // location + this.originPurpose = stops[stop.getId() - 1].getPurpose(); // last + // stop + // location + this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); + } + } else if (!inbound && !toStop) + { + // outbound and trip is from stop to either next stop or tour + // destination. + + // last trip on outbound journey, destination is tour destination + if (stop.getId() == (stops.length - 1)) + { + this.destinationMgra = tour.getDestinationMGRA(); + this.destinationPurpose = tour.getPurpose(); + this.destinationIsTourDestination = true; + } else + { + // not last trip on outbound journey, destination is next stop + this.destinationMgra = stops[stop.getId() + 1].getMgra(); + this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); + } + + // the period for the trip is the origin for the trip + if (stop.getId() == 0) this.period = (byte) tour.getDepartTime(); + else this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); + + } else if (inbound && toStop) + { + // inbound, trip is to stop from either tour destination or last + // stop. + + // first inbound trip; origin is tour destination + if (stop.getId() == 0) + { + this.originMgra = tour.getDestinationMGRA(); + this.originPurpose = tour.getPurpose(); + this.originIsTourDestination = true; + } else + { + // not first inbound trip; origin is last stop + this.originMgra = stops[stop.getId() - 1].getMgra(); // last + // stop + // location + this.originPurpose = stops[stop.getId() - 1].getPurpose(); + } + + // the period for the trip is the destination for the trip + if (stop.getId() == stops.length - 1) this.period = (byte) tour.getArriveTime(); + else this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); + } else + { + // inbound, trip is from stop to either next stop or tour origin. + + // last trip, destination is back to tour origin + if (stop.getId() == (stops.length - 1)) + { + this.destinationMgra = tour.getOriginMGRA(); + this.destinationPurpose = -1; + this.period = (byte) tour.getArriveTime(); + } else + { + // not last trip, destination is next stop + this.destinationMgra = stops[stop.getId() + 1].getMgra(); + this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); + this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); + } + } + + // code period for first trip on tour + if (toStop && !inbound && stop.getId() == 0) + { + this.firstTrip = true; + this.lastTrip = false; + this.period = (byte) tour.getDepartTime(); + } + // code period for last trip on tour + if (!toStop && inbound && stop.getId() == (stops.length - 1)) + { + this.firstTrip = false; + this.lastTrip = true; + this.period = (byte) tour.getArriveTime(); + } + + } + + /** + * @return the period + */ + public byte getPeriod() + { + return period; + } + + /** + * @param period + * the period to set + */ + public void setPeriod(byte period) + { + this.period = period; + } + + /** + * @return the origin purpose + */ + public byte getOriginPurpose() + { + return originPurpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setOriginPurpose(byte purpose) + { + this.originPurpose = purpose; + } + + /** + * @return the destination purpose + */ + public byte getDestinationPurpose() + { + return destinationPurpose; + } + + /** + * @param purpose + * the purpose to set + */ + public void setDestinationPurpose(byte purpose) + { + this.destinationPurpose = purpose; + } + + /** + * @return the originMgra + */ + public int getOriginMgra() + { + return originMgra; + } + + /** + * @param originMgra + * the originMgra to set + */ + public void setOriginMgra(int originMgra) + { + this.originMgra = originMgra; + } + + /** + * @return the destinationMgra + */ + public int getDestinationMgra() + { + return destinationMgra; + } + + /** + * @param destinationMgra + * the destinationMgra to set + */ + public void setDestinationMgra(int destinationMgra) + { + this.destinationMgra = destinationMgra; + } + + /** + * @return the tripMode + */ + public int getTripMode() + { + return tripMode; + } + + /** + * @param tripMode + * the tripMode to set + */ + public void setTripMode(int tripMode) + { + this.tripMode = tripMode; + } + + public int getBoardTap() { + return boardTap; + } + + public void setBoardTap(int boardTap) { + this.boardTap = boardTap; + } + + public int getAlightTap() { + return alightTap; + } + + public void setAlightTap(int alightTap) { + this.alightTap = alightTap; + } + + public int getSet() { + return set; + } + + public void setSet(int set) { + this.set = set; + } + + /** + * @return the inbound + */ + public boolean isInbound() + { + return inbound; + } + + /** + * @param inbound + * the inbound to set + */ + public void setInbound(boolean inbound) + { + this.inbound = inbound; + } + + /** + * @return the firstTrip + */ + public boolean isFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(boolean firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public boolean isLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(boolean lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the originIsTourDestination + */ + public boolean isOriginIsTourDestination() + { + return originIsTourDestination; + } + + /** + * @param originIsTourDestination + * the originIsTourDestination to set + */ + public void setOriginIsTourDestination(boolean originIsTourDestination) + { + this.originIsTourDestination = originIsTourDestination; + } + + /** + * @return the destinationIsTourDestination + */ + public boolean isDestinationIsTourDestination() + { + return destinationIsTourDestination; + } + + /** + * @param destinationIsTourDestination + * the destinationIsTourDestination to set + */ + public void setDestinationIsTourDestination(boolean destinationIsTourDestination) + { + this.destinationIsTourDestination = destinationIsTourDestination; + } + + public float getValueOfTime() { + return valueOfTime; + } + + public void setValueOfTime(float valueOfTime) { + this.valueOfTime = valueOfTime; + } + public float getParkingCost() { + return parkingCost; + } + + public void setParkingCost(float parkingCost) { + this.parkingCost = parkingCost; + } + + public void setMicromobilityWalkMode(byte micromobilityWalkMode) { + this.micromobilityWalkMode=micromobilityWalkMode; + } + + public byte getMicromobilityWalkMode() { + return micromobilityWalkMode; + } + public float getMicromobilityWalkLogsum() { + return micromobilityWalkLogsum; + } + + public void setMicromobilityWalkLogsum(float micromobilityWalkLogsum) { + this.micromobilityWalkLogsum = micromobilityWalkLogsum; + } + + public byte getMicromobilityAccessMode() { + return micromobilityAccessMode; + } + + public void setMicromobilityAccessMode(byte micromobilityAccessMode) { + this.micromobilityAccessMode = micromobilityAccessMode; + } + + public byte getMicromobilityEgressMode() { + return micromobilityEgressMode; + } + + public void setMicromobilityEgressMode(byte micromobilityEgressMode) { + this.micromobilityEgressMode = micromobilityEgressMode; + } + + public float getMicromobilityAccessLogsum() { + return micromobilityAccessLogsum; + } + + public void setMicromobilityAccessLogsum(float micromobilityAccessLogsum) { + this.micromobilityAccessLogsum = micromobilityAccessLogsum; + } + + public float getMicromobilityEgressLogsum() { + return micromobilityEgressLogsum; + } + + public void setMicromobilityEgressLogsum(float micromobilityEgressLogsum) { + this.micromobilityEgressLogsum = micromobilityEgressLogsum; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java new file mode 100644 index 0000000..c984730 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java @@ -0,0 +1,883 @@ +package org.sandag.abm.visitor; + +import java.io.Serializable; +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.internalexternal.InternalExternalTripModeChoiceDMU; + +import com.pb.common.calculator.IndexValues; +import com.pb.common.calculator.VariableTable; + +public class VisitorTripModeChoiceDMU + implements Serializable, VariableTable +{ + + protected transient Logger logger = Logger.getLogger(InternalExternalTripModeChoiceDMU.class); + + protected HashMap methodIndexMap; + protected IndexValues dmuIndex; + + protected int tourDepartPeriod; + protected int tourArrivePeriod; + protected int tripPeriod; + protected int outboundStops; + protected int returnStops; + protected int firstTrip; + protected int lastTrip; + protected int tourPurpose; + protected int segment; + protected int partySize; + protected int autoAvailable; + protected int income; + + // tour mode + protected int tourModeIsDA; + protected int tourModeIsS2; + protected int tourModeIsS3; + protected int tourModeIsWalk; + protected int tourModeIsBike; + protected int tourModeIsWalkTransit; + protected int tourModeIsPNRTransit; + protected int tourModeIsKNRTransit; + protected int tourModeIsMaas; + protected int tourModeIsTNCTransit; + + + protected float hourlyParkingCostTourDest; + protected float dailyParkingCostTourDest; + protected float monthlyParkingCostTourDest; + protected int tripOrigIsTourDest; + protected int tripDestIsTourDest; + protected float hourlyParkingCostTripOrig; + protected float hourlyParkingCostTripDest; + + protected double nmWalkTime; + protected double nmBikeTime; + + protected double ivtCoeff; + protected double costCoeff; + protected double walkTransitLogsum; + protected double pnrTransitLogsum; + protected double knrTransitLogsum; + + protected float waitTimeTaxi; + protected float waitTimeSingleTNC; + protected float waitTimeSharedTNC; + + + protected int outboundHalfTourDirection; + + public VisitorTripModeChoiceDMU(VisitorModelStructure modelStructure, Logger aLogger) + { + if (aLogger == null) aLogger = Logger.getLogger("visitorModel"); + logger = aLogger; + setupMethodIndexMap(); + dmuIndex = new IndexValues(); + + } + + /** + * Set this index values for this tour mode choice DMU object. + * + * @param hhIndex + * is the DMU household index + * @param zoneIndex + * is the DMU zone index + * @param origIndex + * is the DMU origin index + * @param destIndex + * is the DMU desatination index + */ + public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, + boolean debug) + { + dmuIndex.setHHIndex(hhIndex); + dmuIndex.setZoneIndex(zoneIndex); + dmuIndex.setOriginZone(origIndex); + dmuIndex.setDestZone(destIndex); + + dmuIndex.setDebug(false); + dmuIndex.setDebugLabel(""); + if (debug) + { + dmuIndex.setDebug(true); + dmuIndex.setDebugLabel("Debug MC UEC"); + } + + } + + public IndexValues getDmuIndexValues() + { + return dmuIndex; + } + + /** + * @return the tripPeriod + */ + public int getTripPeriod() + { + return tripPeriod; + } + + /** + * @param tripPeriod + * the tripPeriod to set + */ + public void setTripPeriod(int tripPeriod) + { + this.tripPeriod = tripPeriod; + } + + /** + * @return the outboundStops + */ + public int getOutboundStops() + { + return outboundStops; + } + + /** + * @param outboundStops + * the outboundStops to set + */ + public void setOutboundStops(int outboundStops) + { + this.outboundStops = outboundStops; + } + + /** + * @return the returnStops + */ + public int getReturnStops() + { + return returnStops; + } + + /** + * @param returnStops + * the returnStops to set + */ + public void setReturnStops(int returnStops) + { + this.returnStops = returnStops; + } + + /** + * @return the firstTrip + */ + public int getFirstTrip() + { + return firstTrip; + } + + /** + * @param firstTrip + * the firstTrip to set + */ + public void setFirstTrip(int firstTrip) + { + this.firstTrip = firstTrip; + } + + /** + * @return the lastTrip + */ + public int getLastTrip() + { + return lastTrip; + } + + /** + * @param lastTrip + * the lastTrip to set + */ + public void setLastTrip(int lastTrip) + { + this.lastTrip = lastTrip; + } + + /** + * @return the tourModeIsDA + */ + public int getTourModeIsDA() + { + return tourModeIsDA; + } + + /** + * @param tourModeIsDA + * the tourModeIsDA to set + */ + public void setTourModeIsDA(int tourModeIsDA) + { + this.tourModeIsDA = tourModeIsDA; + } + + /** + * @return the tourModeIsS2 + */ + public int getTourModeIsS2() + { + return tourModeIsS2; + } + + /** + * @param tourModeIsS2 + * the tourModeIsS2 to set + */ + public void setTourModeIsS2(int tourModeIsS2) + { + this.tourModeIsS2 = tourModeIsS2; + } + + /** + * @return the tourModeIsS3 + */ + public int getTourModeIsS3() + { + return tourModeIsS3; + } + + /** + * @param tourModeIsS3 + * the tourModeIsS3 to set + */ + public void setTourModeIsS3(int tourModeIsS3) + { + this.tourModeIsS3 = tourModeIsS3; + } + + /** + * @return the tourModeIsWalk + */ + public int getTourModeIsWalk() + { + return tourModeIsWalk; + } + + /** + * @param tourModeIsWalk + * the tourModeIsWalk to set + */ + public void setTourModeIsWalk(int tourModeIsWalk) + { + this.tourModeIsWalk = tourModeIsWalk; + } + + /** + * @return the tourModeIsBike + */ + public int getTourModeIsBike() + { + return tourModeIsBike; + } + + /** + * @param tourModeIsBike + * the tourModeIsBike to set + */ + public void setTourModeIsBike(int tourModeIsBike) + { + this.tourModeIsBike = tourModeIsBike; + } + + /** + * @return the tourModeIsWalkTransit + */ + public int getTourModeIsWalkTransit() + { + return tourModeIsWalkTransit; + } + + /** + * @param tourModeIsWalkTransit + * the tourModeIsWalkTransit to set + */ + public void setTourModeIsWalkTransit(int tourModeIsWalkTransit) + { + this.tourModeIsWalkTransit = tourModeIsWalkTransit; + } + + /** + * @return the tourModeIsPNRTransit + */ + public int getTourModeIsPNRTransit() + { + return tourModeIsPNRTransit; + } + + /** + * @param tourModeIsPNRTransit + * the tourModeIsPNRTransit to set + */ + public void setTourModeIsPNRTransit(int tourModeIsPNRTransit) + { + this.tourModeIsPNRTransit = tourModeIsPNRTransit; + } + + /** + * @return the tourModeIsKNRTransit + */ + public int getTourModeIsKNRTransit() + { + return tourModeIsKNRTransit; + } + + /** + * @param tourModeIsKNRTransit + * the tourModeIsKNRTransit to set + */ + public void setTourModeIsKNRTransit(int tourModeIsKNRTransit) + { + this.tourModeIsKNRTransit = tourModeIsKNRTransit; + } + + /** + * @return the tourModeIsMaas + */ + public int getTourModeIsMaas() + { + return tourModeIsMaas; + } + + /** + * @param tourModeIsMaas + * the tourModeIsMaas to set + */ + public void setTourModeIsMaas(int tourModeIsMaas) + { + this.tourModeIsMaas = tourModeIsMaas; + } + + /** + * @return the hourlyParkingCostTourDest + */ + public float getHourlyParkingCostTourDest() + { + return hourlyParkingCostTourDest; + } + + /** + * @param hourlyParkingCostTourDest + * the hourlyParkingCostTourDest to set + */ + public void setHourlyParkingCostTourDest(float hourlyParkingCostTourDest) + { + this.hourlyParkingCostTourDest = hourlyParkingCostTourDest; + } + + /** + * @return the dailyParkingCostTourDest + */ + public float getDailyParkingCostTourDest() + { + return dailyParkingCostTourDest; + } + + /** + * @param dailyParkingCostTourDest + * the dailyParkingCostTourDest to set + */ + public void setDailyParkingCostTourDest(float dailyParkingCostTourDest) + { + this.dailyParkingCostTourDest = dailyParkingCostTourDest; + } + + /** + * @return the monthlyParkingCostTourDest + */ + public float getMonthlyParkingCostTourDest() + { + return monthlyParkingCostTourDest; + } + + /** + * @param monthlyParkingCostTourDest + * the monthlyParkingCostTourDest to set + */ + public void setMonthlyParkingCostTourDest(float monthlyParkingCostTourDest) + { + this.monthlyParkingCostTourDest = monthlyParkingCostTourDest; + } + + /** + * @return the tripOrigIsTourDest + */ + public int getTripOrigIsTourDest() + { + return tripOrigIsTourDest; + } + + /** + * @param tripOrigIsTourDest + * the tripOrigIsTourDest to set + */ + public void setTripOrigIsTourDest(int tripOrigIsTourDest) + { + this.tripOrigIsTourDest = tripOrigIsTourDest; + } + + /** + * @return the tripDestIsTourDest + */ + public int getTripDestIsTourDest() + { + return tripDestIsTourDest; + } + + /** + * @param tripDestIsTourDest + * the tripDestIsTourDest to set + */ + public void setTripDestIsTourDest(int tripDestIsTourDest) + { + this.tripDestIsTourDest = tripDestIsTourDest; + } + + /** + * @return the hourlyParkingCostTripOrig + */ + public float getHourlyParkingCostTripOrig() + { + return hourlyParkingCostTripOrig; + } + + /** + * @param hourlyParkingCostTripOrig + * the hourlyParkingCostTripOrig to set + */ + public void setHourlyParkingCostTripOrig(float hourlyParkingCostTripOrig) + { + this.hourlyParkingCostTripOrig = hourlyParkingCostTripOrig; + } + + /** + * @return the hourlyParkingCostTripDest + */ + public float getHourlyParkingCostTripDest() + { + return hourlyParkingCostTripDest; + } + + /** + * @param hourlyParkingCostTripDest + * the hourlyParkingCostTripDest to set + */ + public void setHourlyParkingCostTripDest(float hourlyParkingCostTripDest) + { + this.hourlyParkingCostTripDest = hourlyParkingCostTripDest; + } + + /** + * @return the outboundHalfTourDirection + */ + public int getOutboundHalfTourDirection() + { + return outboundHalfTourDirection; + } + + /** + * @param outboundHalfTourDirection + * the outboundHalfTourDirection to set + */ + public void setOutboundHalfTourDirection(int outboundHalfTourDirection) + { + this.outboundHalfTourDirection = outboundHalfTourDirection; + } + + /** + * @return the tourDepartPeriod + */ + public int getTourDepartPeriod() + { + return tourDepartPeriod; + } + + /** + * @param tourDepartPeriod + * the tourDepartPeriod to set + */ + public void setTourDepartPeriod(int tourDepartPeriod) + { + this.tourDepartPeriod = tourDepartPeriod; + } + + /** + * @param tourArrivePeriod + * the tourArrivePeriod to set + */ + public void setTourArrivePeriod(int tourArrivePeriod) + { + this.tourArrivePeriod = tourArrivePeriod; + } + + /** + * @return the tourArrivePeriod + */ + public int getTourArrivePeriod() + { + return tourArrivePeriod; + } + + public double getNm_walkTime() + { + return nmWalkTime; + } + + public void setNonMotorizedWalkTime(double nmWalkTime) + { + this.nmWalkTime = nmWalkTime; + } + + public void setNonMotorizedBikeTime(double nmBikeTime) + { + this.nmBikeTime = nmBikeTime; + } + + public double getNm_bikeTime() + { + return nmBikeTime; + } + + /** + * @return the tourPurpose + */ + public int getTourPurpose() + { + return tourPurpose; + } + + /** + * @param tourPurpose + * the tourPurpose to set + */ + public void setTourPurpose(int tourPurpose) + { + this.tourPurpose = tourPurpose; + } + + /** + * @return the segment + */ + public int getSegment() + { + return segment; + } + + /** + * @param segment + * the segment to set + */ + public void setSegment(int segment) + { + this.segment = segment; + } + + public int getPartySize() + { + return partySize; + } + + public void setPartySize(int partySize) + { + this.partySize = partySize; + } + + public int getAutoAvailable() + { + return autoAvailable; + } + + public void setAutoAvailable(int autoAvailable) + { + this.autoAvailable = autoAvailable; + } + + public int getIncome() + { + return income; + } + + public void setIncome(int income) + { + this.income = income; + } + public double getIvtCoeff() { + return ivtCoeff; + } + + public void setIvtCoeff(double ivtCoeff) { + this.ivtCoeff = ivtCoeff; + } + + public double getCostCoeff() { + return costCoeff; + } + + public void setCostCoeff(double costCoeff) { + this.costCoeff = costCoeff; + } + + public double getWalkTransitLogsum() { + return walkTransitLogsum; + } + + public void setWalkTransitLogsum(double walkTransitLogsum) { + this.walkTransitLogsum = walkTransitLogsum; + } + + public double getPnrTransitLogsum() { + return pnrTransitLogsum; + } + + public void setPnrTransitLogsum(double pnrTransitLogsum) { + this.pnrTransitLogsum = pnrTransitLogsum; + } + + public double getKnrTransitLogsum() { + return knrTransitLogsum; + } + + public void setKnrTransitLogsum(double knrTransitLogsum) { + this.knrTransitLogsum = knrTransitLogsum; + } + + public float getWaitTimeTaxi() { + return waitTimeTaxi; + } + + public void setWaitTimeTaxi(float waitTimeTaxi) { + this.waitTimeTaxi = waitTimeTaxi; + } + + public float getWaitTimeSingleTNC() { + return waitTimeSingleTNC; + } + + public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { + this.waitTimeSingleTNC = waitTimeSingleTNC; + } + + public float getWaitTimeSharedTNC() { + return waitTimeSharedTNC; + } + + public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { + this.waitTimeSharedTNC = waitTimeSharedTNC; + } + + + private void setupMethodIndexMap() + { + methodIndexMap = new HashMap(); + + methodIndexMap.put("getTourDepartPeriod", 0); + methodIndexMap.put("getTourArrivePeriod", 1); + methodIndexMap.put("getTripPeriod", 2); + methodIndexMap.put("getSegment", 3); + methodIndexMap.put("getTourPurpose", 4); + methodIndexMap.put("getOutboundStops", 5); + methodIndexMap.put("getReturnStops", 6); + methodIndexMap.put("getFirstTrip", 7); + methodIndexMap.put("getLastTrip", 8); + methodIndexMap.put("getTourModeIsDA", 9); + methodIndexMap.put("getTourModeIsS2", 10); + methodIndexMap.put("getTourModeIsS3", 11); + methodIndexMap.put("getTourModeIsWalk", 12); + methodIndexMap.put("getTourModeIsBike", 13); + methodIndexMap.put("getTourModeIsWalkTransit", 14); + methodIndexMap.put("getTourModeIsPNRTransit", 15); + methodIndexMap.put("getTourModeIsKNRTransit", 16); + methodIndexMap.put("getTourModeIsMaas", 17); + methodIndexMap.put("getTourModeIsTNCTransit", 18); + + methodIndexMap.put("getHourlyParkingCostTourDest", 20); + methodIndexMap.put("getDailyParkingCostTourDest", 21); + methodIndexMap.put("getMonthlyParkingCostTourDest", 22); + methodIndexMap.put("getTripOrigIsTourDest", 23); + methodIndexMap.put("getTripDestIsTourDest", 24); + methodIndexMap.put("getHourlyParkingCostTripOrig", 25); + methodIndexMap.put("getHourlyParkingCostTripDest", 26); + + methodIndexMap.put("getPartySize", 30); + methodIndexMap.put("getAutoAvailable", 31); + methodIndexMap.put("getIncome", 32); + + methodIndexMap.put("getIvtCoeff", 60); + methodIndexMap.put("getCostCoeff", 61); + + methodIndexMap.put("getWalkSetLogSum", 62); + methodIndexMap.put("getPnrSetLogSum", 63); + methodIndexMap.put("getKnrSetLogSum", 64); + + methodIndexMap.put("getWaitTimeTaxi", 70); + methodIndexMap.put("getWaitTimeSingleTNC", 71); + methodIndexMap.put("getWaitTimeSharedTNC", 72); + + methodIndexMap.put("getNm_walkTime", 90); + methodIndexMap.put("getNm_bikeTime", 91); + + } + + public double getValueForIndex(int variableIndex, int arrayIndex) + { + + double returnValue = -1; + + switch (variableIndex) + { + case 0: + returnValue = getTourDepartPeriod(); + break; + case 1: + returnValue = getTourArrivePeriod(); + break; + case 2: + returnValue = getTripPeriod(); + break; + case 3: + returnValue = getSegment(); + break; + case 4: + returnValue = getTourPurpose(); + break; + case 5: + returnValue = getOutboundStops(); + break; + case 6: + returnValue = getReturnStops(); + break; + case 7: + returnValue = getFirstTrip(); + break; + case 8: + returnValue = getLastTrip(); + break; + case 9: + returnValue = getTourModeIsDA(); + break; + case 10: + returnValue = getTourModeIsS2(); + break; + case 11: + returnValue = getTourModeIsS3(); + break; + case 12: + returnValue = getTourModeIsWalk(); + break; + case 13: + returnValue = getTourModeIsBike(); + break; + case 14: + returnValue = getTourModeIsWalkTransit(); + break; + case 15: + returnValue = getTourModeIsPNRTransit(); + break; + case 16: + returnValue = getTourModeIsKNRTransit(); + break; + case 17: + returnValue = getTourModeIsMaas(); + break; + case 18: + returnValue = getTourModeIsTNCTransit(); + break; + case 20: + returnValue = getHourlyParkingCostTourDest(); + break; + case 21: + returnValue = getDailyParkingCostTourDest(); + break; + case 22: + returnValue = getMonthlyParkingCostTourDest(); + break; + case 23: + returnValue = getTripOrigIsTourDest(); + break; + case 24: + returnValue = getTripDestIsTourDest(); + break; + case 25: + returnValue = getHourlyParkingCostTripOrig(); + break; + case 26: + returnValue = getHourlyParkingCostTripDest(); + break; + case 30: + returnValue = getPartySize(); + break; + case 31: + returnValue = getAutoAvailable(); + break; + case 32: + returnValue = getIncome(); + break; + case 60: + returnValue = getIvtCoeff(); + break; + case 61: + returnValue = getCostCoeff(); + break; + case 62: + returnValue = getWalkTransitLogsum(); + break; + case 63: + returnValue = getPnrTransitLogsum(); + break; + case 64: + returnValue = getKnrTransitLogsum(); + break; + case 70: return getWaitTimeTaxi(); + case 71: return getWaitTimeSingleTNC(); + case 72: return getWaitTimeSharedTNC(); + case 90: + returnValue = getNm_walkTime(); + break; + case 91: + returnValue = getNm_bikeTime(); + break; + default: + logger.error( "method number = " + variableIndex + " not found" ); + throw new RuntimeException( "method number = " + variableIndex + " not found" ); + } + return returnValue; + + } + + public int getIndexValue(String variableName) + { + return methodIndexMap.get(variableName); + } + + public int getAssignmentIndexValue(String variableName) + { + throw new UnsupportedOperationException(); + } + + public double getValueForIndex(int variableIndex) + { + throw new UnsupportedOperationException(); + } + + public void setValue(String variableName, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public void setValue(int variableIndex, double variableValue) + { + throw new UnsupportedOperationException(); + } + + public int getTourModeIsTNCTransit() { + return tourModeIsTNCTransit; + } + + public void setTourModeIsTNCTransit(int tourModeIsTNCTransit) { + this.tourModeIsTNCTransit = tourModeIsTNCTransit; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java new file mode 100644 index 0000000..bd2196c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java @@ -0,0 +1,327 @@ +package org.sandag.abm.visitor; + +import java.util.HashMap; + +import org.apache.log4j.Logger; +import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; +import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.McLogsumsCalculator; +import org.sandag.abm.ctramp.TripModeChoiceDMU; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.calculator.VariableTable; +import com.pb.common.newmodel.ChoiceModelApplication; +import com.pb.common.newmodel.UtilityExpressionCalculator; + +public class VisitorTripModeChoiceModel +{ + + private transient Logger logger = Logger.getLogger("visitorModel"); + + private AutoAndNonMotorizedSkimsCalculator anm; + private McLogsumsCalculator logsumHelper; + private VisitorModelStructure modelStructure; + private TazDataManager tazs; + private MgraDataManager mgraManager; + private double[] lsWgtAvgCostM; + private double[] lsWgtAvgCostD; + private double[] lsWgtAvgCostH; + private VisitorTripModeChoiceDMU dmu; + private ChoiceModelApplication tripModeChoiceModel; + double logsum = 0; + + private static final String PROPERTIES_UEC_DATA_SHEET = "visitor.trip.mc.data.page"; + private static final String PROPERTIES_UEC_MODEL_SHEET = "visitor.trip.mc.model.page"; + private static final String PROPERTIES_UEC_FILE = "visitor.trip.mc.uec.file"; + private TripModeChoiceDMU mcDmuObject; + private AutoTazSkimsCalculator tazDistanceCalculator; + + /** + * Constructor. + * + * @param propertyMap + * @param myModelStructure + * @param dmuFactory + * @param myLogsumHelper + */ + public VisitorTripModeChoiceModel(HashMap propertyMap, + VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) + { + tazs = TazDataManager.getInstance(propertyMap); + mgraManager = MgraDataManager.getInstance(propertyMap); + + lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); + lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); + lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); + + modelStructure = myModelStructure; + this.tazDistanceCalculator = tazDistanceCalculator; + + logsumHelper = new McLogsumsCalculator(); + logsumHelper.setupSkimCalculators(propertyMap); + logsumHelper.setTazDistanceSkimArrays( + tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), + tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); + + SandagModelStructure modelStructure = new SandagModelStructure(); + mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); + + setupTripModeChoiceModel(propertyMap, dmuFactory); + + } + + /** + * Read the UEC file and set up the trip mode choice model. + * + * @param propertyMap + * @param dmuFactory + */ + private void setupTripModeChoiceModel(HashMap propertyMap, + VisitorDmuFactoryIf dmuFactory) + { + + logger.info(String.format("setting up visitor trip mode choice model.")); + + dmu = dmuFactory.getVisitorTripModeChoiceDMU(); + + int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_DATA_SHEET)); + int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, + PROPERTIES_UEC_MODEL_SHEET)); + + String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); + String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); + tripModeUecFile = uecPath + tripModeUecFile; + + tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, + propertyMap, (VariableTable) dmu); + + } + + /** + * Calculate utilities and return logsum for the tour and stop. + * + * @param tour + * @param trip + */ + public double computeUtilities(VisitorTour tour, VisitorTrip trip) + { + + setDmuAttributes(tour, trip); + + tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); + + if (tour.getDebugChoiceModels()) + { + tour.logTourObject(logger, 100); + tripModeChoiceModel.logUECResults(logger, "Visitor trip mode choice model"); + + } + + logsum = tripModeChoiceModel.getLogsum(); + + if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); + + return logsum; + + } + + /** + * Choose a mode and store in the trip object. + * + * @param tour + * VisitorTour + * @param trip + * VisitorTrip + * + */ + public void chooseMode(VisitorTour tour, VisitorTrip trip) + { + + computeUtilities(tour, trip); + + double rand = tour.getRandom(); + try{ + int mode = tripModeChoiceModel.getChoiceResult(rand); + trip.setTripMode(mode); + + //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + + int votIndex = uec.lookupVariableIndex("vot"); + double vot = uec.getValueForIndex(votIndex); + trip.setValueOfTime((float)vot); + + float parkingCost = getTripParkingCost(mode); + trip.setParkingCost(parkingCost); + + if(modelStructure.getTripModeIsTransit(mode)){ + double[][] bestTapPairs = null; + + if (modelStructure.getTripModeIsWalkTransit(mode)){ + bestTapPairs = logsumHelper.getBestWtwTripTaps(); + } + else if (modelStructure.getTripModeIsPnrTransit(mode)||modelStructure.getTripModeIsKnrTransit(mode)){ + if (!trip.isInbound()) + bestTapPairs = logsumHelper.getBestDtwTripTaps(); + else + bestTapPairs = logsumHelper.getBestWtdTripTaps(); + } + double rn = tour.getRandom(); + int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); + int boardTap = (int) bestTapPairs[pathIndex][0]; + int alightTap = (int) bestTapPairs[pathIndex][1]; + int set = (int) bestTapPairs[pathIndex][2]; + trip.setBoardTap(boardTap); + trip.setAlightTap(alightTap); + trip.setSet(set); + } + + }catch(Exception e){ + logger.info("Error calculating visitor trip mode choice with rand="+rand); + tour.logTourObject(logger, 100); + logger.error(e.getMessage()); + } + + } + + /** + * Return parking cost from UEC if auto trip, else return 0. + * + * @param tripMode + * @return Parking cost if auto mode, else 0 + */ + public float getTripParkingCost(int tripMode) { + + float parkingCost=0; + + if(modelStructure.getTripModeIsSovOrHov(tripMode)) { + UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); + int parkingCostIndex = uec.lookupVariableIndex("parkingCost"); + parkingCost = (float) uec.getValueForIndex(parkingCostIndex); + return parkingCost; + } + return parkingCost; + } + + + /** + * Set DMU attributes. + * + * @param tour + * @param trip + */ + public void setDmuAttributes(VisitorTour tour, VisitorTrip trip) + { + + int tourDestinationMgra = tour.getDestinationMGRA(); + int tripOriginMgra = trip.getOriginMgra(); + int tripDestinationMgra = trip.getDestinationMgra(); + + int tripOriginTaz = mgraManager.getTaz(tripOriginMgra); + int tripDestinationTaz = mgraManager.getTaz(tripDestinationMgra); + + int tourMode = tour.getTourMode(); + + dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, + tour.getDebugChoiceModels()); + + dmu.setTourDepartPeriod(tour.getDepartTime()); + dmu.setTourArrivePeriod(tour.getArriveTime()); + dmu.setTripPeriod(trip.getPeriod()); + + // set trip mc dmu values for transit logsum (gets replaced below by uec values) + double c_ivt = -0.03; + double c_cost = - 0.0033; + + // Solve trip mode level utilities + mcDmuObject.setIvtCoeff(c_ivt); + mcDmuObject.setCostCoeff(c_cost); + double walkTransitLogsum = -999.0; + double driveTransitLogsum = -999.0; + + logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); + dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); + + logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); + walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); + + dmu.setWalkTransitLogsum(walkTransitLogsum); + if (!trip.isInbound()) + { + logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); + } else + { + logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); + driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); + } + + dmu.setPnrTransitLogsum(driveTransitLogsum); + dmu.setKnrTransitLogsum(driveTransitLogsum); + + dmu.setTourPurpose(tour.getPurpose()); + + dmu.setOutboundStops(tour.getNumberInboundStops()); + dmu.setReturnStops(tour.getNumberInboundStops()); + + if (trip.isFirstTrip()) dmu.setFirstTrip(1); + else dmu.setFirstTrip(0); + + if (trip.isLastTrip()) dmu.setLastTrip(1); + else dmu.setLastTrip(0); + + if (modelStructure.getTourModeIsSov(tourMode)) dmu.setTourModeIsDA(1); + else dmu.setTourModeIsDA(0); + + if (modelStructure.getTourModeIsS2(tourMode)) dmu.setTourModeIsS2(1); + else dmu.setTourModeIsS2(0); + + if (modelStructure.getTourModeIsS3(tourMode)) dmu.setTourModeIsS3(1); + else dmu.setTourModeIsS3(0); + + if (modelStructure.getTourModeIsWalk(tourMode)) dmu.setTourModeIsWalk(1); + else dmu.setTourModeIsWalk(0); + + if (modelStructure.getTourModeIsBike(tourMode)) dmu.setTourModeIsBike(1); + else dmu.setTourModeIsBike(0); + + if (modelStructure.getTourModeIsWalkTransit(tourMode)) dmu.setTourModeIsWalkTransit(1); + else dmu.setTourModeIsWalkTransit(0); + + if (modelStructure.getTourModeIsPnr(tourMode)) dmu.setTourModeIsPNRTransit(1); + else dmu.setTourModeIsPNRTransit(0); + + if (modelStructure.getTourModeIsKnr(tourMode)) dmu.setTourModeIsKNRTransit(1); + else dmu.setTourModeIsKNRTransit(0); + + if (modelStructure.getTourModeIsMaas(tourMode)) dmu.setTourModeIsMaas(1); + else dmu.setTourModeIsMaas(0); + + if (modelStructure.getTourModeIsTncTransit(tourMode)) dmu.setTourModeIsTNCTransit(1); + else dmu.setTourModeIsTNCTransit(0); + + if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); + else dmu.setTripOrigIsTourDest(0); + + if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); + else dmu.setTripDestIsTourDest(0); + + dmu.setIncome((byte) tour.getIncome()); + dmu.setAutoAvailable(tour.getAutoAvailable()); + dmu.setPartySize(tour.getNumberOfParticipants()); + + dmu.setHourlyParkingCostTourDest((float) lsWgtAvgCostH[tourDestinationMgra]); + dmu.setDailyParkingCostTourDest((float) lsWgtAvgCostD[tourDestinationMgra]); + dmu.setMonthlyParkingCostTourDest((float) lsWgtAvgCostM[tourDestinationMgra]); + dmu.setHourlyParkingCostTripOrig((float) lsWgtAvgCostH[tripOriginMgra]); + dmu.setHourlyParkingCostTripDest((float) lsWgtAvgCostH[tripDestinationMgra]); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java new file mode 100644 index 0000000..7a715bf --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java @@ -0,0 +1,682 @@ +package org.sandag.abm.visitor; + +import gnu.cajo.invoke.Remote; +import gnu.cajo.utils.ItemServer; + +import java.io.File; +import java.io.IOException; +import java.rmi.RemoteException; +import java.util.HashMap; +import java.util.MissingResourceException; +import java.util.ResourceBundle; + +import org.apache.log4j.Logger; +import org.sandag.abm.application.SandagModelStructure; +import org.sandag.abm.ctramp.CtrampApplication; +import org.sandag.abm.ctramp.MatrixDataServer; +import org.sandag.abm.ctramp.MatrixDataServerRmi; +import org.sandag.abm.ctramp.Util; +import org.sandag.abm.modechoice.MgraDataManager; +import org.sandag.abm.modechoice.TapDataManager; +import org.sandag.abm.modechoice.TazDataManager; + +import com.pb.common.datafile.OLD_CSVFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.MatrixWriter; +import com.pb.common.util.ResourceUtil; + +public class VisitorTripTables +{ + + private static Logger logger = Logger.getLogger("tripTables"); + public static final int MATRIX_DATA_SERVER_PORT = 1171; + + private TableDataSet tripData; + + // Some parameters + private int[] modeIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // total + // modes, + // returns + // 0=auto + // modes, + // 1=non-motor, + // 2=transit, + // 3= + // other + private int[] matrixIndex; // an + // index + // array, + // dimensioned + // by + // number + // of + // modes, + // returns + // the + // element + // of + // the + // matrix + // array + // to + // store + // value + + // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER + private int autoModes = 0; + private int tranModes = 0; + private int nmotModes = 0; + private int othrModes = 0; + + // one file per time period + private int numberOfPeriods; + + private HashMap rbMap; + + // matrices are indexed by modes, vot bins, submodes + private Matrix[][][] matrix; + + private ResourceBundle rb; + private MgraDataManager mgraManager; + private TazDataManager tazManager; + private TapDataManager tapManager; + private SandagModelStructure modelStructure; + + private MatrixDataServerRmi ms; + private float sampleRate; + private static int iteration=1; + private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; + private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; + private float valueOfTimeThresholdLow = 0; + private float valueOfTimeThresholdMed = 0; + //value of time bins by mode group + int[] votBins = {3,1,1,1}; + public int numSkimSets; + + public VisitorTripTables(HashMap rbMap) + { + + this.rbMap = rbMap; + tazManager = TazDataManager.getInstance(rbMap); + tapManager = TapDataManager.getInstance(rbMap); + mgraManager = MgraDataManager.getInstance(rbMap); + + modelStructure = new SandagModelStructure(); + + // Time period limits + numberOfPeriods = modelStructure.getNumberModelPeriods(); + + // number of modes + modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; + matrixIndex = new int[modeIndex.length]; + + numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); + + // set the mode arrays + for (int i = 1; i < modeIndex.length; ++i) + { + if (modelStructure.getTourModeIsSovOrHov(i)) + { + modeIndex[i] = 0; + matrixIndex[i] = autoModes; + ++autoModes; + } else if (modelStructure.getTourModeIsNonMotorized(i)) + { + modeIndex[i] = 1; + matrixIndex[i] = nmotModes; + ++nmotModes; + } else if (modelStructure.getTourModeIsWalkTransit(i) + || modelStructure.getTourModeIsDriveTransit(i)) + { + modeIndex[i] = 2; + matrixIndex[i] = tranModes; + ++tranModes; + } else + { + modeIndex[i] = 3; + matrixIndex[i] = othrModes; + ++othrModes; + } + } + //value of time thresholds + valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); + valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); + } + + /** + * Initialize all the matrices for the given time period. + * + * @param periodName + * The name of the time period. + */ + public void initializeMatrices(String periodName) + { + + /* + * This won't work because external stations aren't listed in the MGRA + * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = + * tazIndex.length-1; + */ + // Instead, use maximum taz number + int maxTaz = tazManager.getMaxTaz(); + int[] tazIndex = new int[maxTaz + 1]; + + // assume zone numbers are sequential + for (int i = 1; i < tazIndex.length; ++i) + tazIndex[i] = i; + + // get the tap index + int[] tapIndex = tapManager.getTaps(); + int taps = tapIndex.length - 1; + + // Initialize matrices; one for each mode group (auto, non-mot, tran, + // other) + // All matrices will be dimensioned by TAZs except for transit, which is + // dimensioned by TAPs + int numberOfModes = 4; + matrix = new Matrix[numberOfModes][][]; + for (int i = 0; i < numberOfModes; ++i) + { + + String modeName; + + matrix[i] = new Matrix[votBins[i]][]; + + for(int j = 0; j< votBins[i];++j){ + if (i == 0) + { + matrix[i][j] = new Matrix[autoModes]; + for (int k = 0; k < autoModes; ++k) + { + modeName = modelStructure.getModeName(k + 1); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 1) + { + matrix[i][j] = new Matrix[nmotModes]; + for (int k = 0; k < nmotModes; ++k) + { + modeName = modelStructure.getModeName(k + 1 + autoModes); + matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); + matrix[i][j][k].setExternalNumbers(tazIndex); + } + } else if (i == 2) + { + matrix[i][j] = new Matrix[tranModes*numSkimSets]; + for (int k = 0; k < tranModes; ++k) + { + for(int l=0;l1) + votBin = getValueOfTimeBin(valueOfTime); + + if (mode == 0) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); + } else if (mode == 1) + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } else if (mode == 2) + { + + if (boardTap == 0 || alightTap == 0) continue; + + //store transit trips in matrices + mat = (matrixIndex[tripMode]*numSkimSets)+set; + float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); + matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); + + // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) + if (modelStructure.getTourModeIsDriveTransit(tripMode)) + { + + // add the tNCVehicle trip portion to the trip table + if (!inbound) + { // from origin to lot (boarding tap) + int PNRTAZ = tapManager.getTazForTap(boardTap); + value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); + matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); + + } else + { // from lot (alighting tap) to destination + int PNRTAZ = tapManager.getTazForTap(alightTap); + value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); + matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); + } + + } + } else + { + float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); + matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); + } + + //logger.info("End creating trip tables for period " + timePeriod); + } + } + + /** + * Return the value of time bin 0 through 2 based on the thresholds provided in the property map + * @param valueOfTime + * @return value of time bin 0 through 2 + */ + public int getValueOfTimeBin(float valueOfTime){ + + if(valueOfTime1) + end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; + else + end[i][j] = "_" + per + ".omx"; + } + } + for (int i = 0; i < 4; ++i){ + for(int j = 0; j < votBins[i];++j){ + try + { + //Delete the file if it exists + File f = new File(fileName[i]+end[i][j]); + if(f.exists()){ + logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); + f.delete(); + } + if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); + else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); + } catch (Exception e) + { + logger.error("exception caught writing " + mt.toString() + " matrix file = " + + fileName[i] +end[i][j] + ", for mode index = " + i, e); + throw new RuntimeException(); + } + } + } + + } + + /** + * Utility method to write a set of matrices to disk. + * + * @param fileName + * The file name to write to. + * @param m + * An array of matrices + */ + public void writeMatrixFile(String fileName, Matrix[] m) + { + + // auto trips + MatrixWriter writer = MatrixWriter.createWriter(fileName); + String[] names = new String[m.length]; + + for (int i = 0; i < m.length; i++) + { + names[i] = m[i].getName(); + logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " + + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); + } + + writer.writeMatrices(names, m); + } + + /** + * Start matrix server + * + * @param serverAddress + * @param serverPort + * @param mt + * @return + */ + private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, + MatrixType mt) + { + + String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; + MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + + // bind this concrete object with the cajo library objects for managing + // RMI + try + { + Remote.config(serverAddress, serverPort, null, 0); + } catch (Exception e) + { + logger.error(String.format( + "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + try + { + ItemServer.bind(matrixServer, className); + } catch (RemoteException e) + { + logger.error(String.format( + "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", + serverAddress, serverPort), e); + throw new RuntimeException(); + } + + return matrixServer; + + } + + /** + * @param args + */ + public static void main(String[] args) + { + + HashMap pMap; + String propertiesFile = null; + + logger.info(String.format( + "SANDAG Visitor Model Trip Table Generation Program using CT-RAMP version %s", + CtrampApplication.VERSION)); + + if (args.length == 0) + { + logger.error(String + .format("no properties file base name (without .properties extension) was specified as an argument.")); + return; + } else propertiesFile = args[0]; + + float sampleRate = 1.0f; + for (int i = 1; i < args.length; ++i) + { + if (args[i].equalsIgnoreCase("-sampleRate")) + { + sampleRate = Float.parseFloat(args[i + 1]); + } + if (args[i].equalsIgnoreCase("-iteration")) + { + iteration = Integer.parseInt(args[i + 1]); + } + } + logger.info("Visitor Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); + pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); + VisitorTripTables tripTables = new VisitorTripTables(pMap); + tripTables.setSampleRate(sampleRate); + + String matrixServerAddress = ""; + int serverPort = 0; + try + { + // get matrix server address. if "none" is specified, no server will + // be + // started, and matrix io will ocurr within the current process. + matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, + "RunModel.MatrixServerAddress"); + try + { + // get matrix server port. + serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, leave undefined + // -- + // it's eithe not needed or show could create an error. + } + } catch (MissingResourceException e) + { + // if no matrix server address entry is found, set to localhost, and + // a + // separate matrix io process will be started on localhost. + matrixServerAddress = "localhost"; + serverPort = MATRIX_DATA_SERVER_PORT; + } + + MatrixDataServerRmi matrixServer = null; + String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); + MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); + + try + { + + if (!matrixServerAddress.equalsIgnoreCase("none")) + { + + if (matrixServerAddress.equalsIgnoreCase("localhost")) + { + matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, + serverPort, mt); + tripTables.ms = matrixServer; + } else + { + tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, + MatrixDataServer.MATRIX_DATA_SERVER_NAME); + tripTables.ms.testRemote("VisitorTripTables"); + + // mdm = MatrixDataManager.getInstance(); + // mdm.setMatrixDataServerObject(ms); + } + + } + + } catch (Exception e) + { + + logger.error( + String.format("exception caught running ctramp model components -- exiting."), + e); + throw new RuntimeException(); + + } + + tripTables.createTripTables(mt); + + } + + /** + * @return the sampleRate + */ + public double getSampleRate() + { + return sampleRate; + } + + /** + * @param sampleRate + * the sampleRate to set + */ + public void setSampleRate(float sampleRate) + { + this.sampleRate = sampleRate; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java new file mode 100644 index 0000000..d6cda4b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java @@ -0,0 +1,42 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public interface AlternativeUsesMatrices extends CodedAlternative { + + void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError; + + + /** + * Method readMatrices. + * @param matrixCacheReader + */ + void readMatrices(MatrixCacheReader matrixCacheReader); + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java new file mode 100644 index 0000000..f69c70c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java @@ -0,0 +1,52 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +/* + * An interface that determines the travel time between two points, given a time (represented by a double) and + * a tNCVehicle type (represented by a char). + */ +package org.sandag.cvm.activityTravel; + +/** + * @author jabraham + * + * An interface that determines a travel attribute between two points, given a time (represented by a double) and + * a tNCVehicle type (represented by a char). For instance if there are matrices of travel time for peak and off-peak, + * and implementation of this interface would be able to determine whether time was in the peak or off-peak, and + * then return the travel conditions appropriately. + */ +public interface ChangingTravelAttributeGetter { + /** + * + * This method returns the travel attribute associated with travelling from origin to destination at time time by + * tNCVehicle type vehicleType + * + * @param origin + * @param destination + * @param time + * @param vehicleType + * @return + */ + public abstract double getTravelAttribute( + int origin, + int destination, + double time, + char vehicleType); +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java new file mode 100644 index 0000000..738846d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java @@ -0,0 +1,35 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + + +import org.sandag.cvm.common.model.Alternative; + + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public interface CodedAlternative extends Alternative { + + public String getCode(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java new file mode 100644 index 0000000..fc937a4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java @@ -0,0 +1,61 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CoefficientFormatError extends Exception { + + /** + * Constructor for CoefficientFormatError. + */ + public CoefficientFormatError() { + super(); + } + + /** + * Constructor for CoefficientFormatError. + * @param message + */ + public CoefficientFormatError(String message) { + super(message); + } + + /** + * Constructor for CoefficientFormatError. + * @param message + * @param cause + */ + public CoefficientFormatError(String message, Throwable cause) { + super(message, cause); + } + + /** + * Constructor for CoefficientFormatError. + * @param cause + */ + public CoefficientFormatError(Throwable cause) { + super(cause); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java new file mode 100644 index 0000000..9009825 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java @@ -0,0 +1,91 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +//import org.sandag.cvm.calgary.commercial.GenerateCommercialTours; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class DurationModel implements ModelUsesMatrices, RealNumberDistribution { + + protected double a=0; + protected double b=0; + protected double c=0; + protected double d=0; + protected double e=0; + protected double f=0; + + + /** @return stop duration in hours + */ + public double sampleValue() { + double x = -Math.random(); + double y = a*Math.exp(b*x)+c*Math.exp(d*x)+f; + if (y<0) y=0; + if (y>24) y = 24; + return y; + } + + /** + * Method addCoefficient. + * @param alternative + * @param index1 + * @param index2 + * @param matrix + * @param coefficient + */ + public void addCoefficient ( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError { + if (index1.equals("a")) a = coefficient; + else if(index1.equals("b")) b = coefficient; + else if(index1.equals("c")) c = coefficient; + else if(index1.equals("d")) d = coefficient; + else if (index1.equals("e")) e = coefficient; + else if(index1.equals("f")) f = coefficient; + else throw new CoefficientFormatError("Duration model coefficients must have index1 as a,b,c or d"); + } + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader matrixReader) {} + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { +// readMatrices(GenerateCommercialTours.matrixReader); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java new file mode 100644 index 0000000..9449313 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java @@ -0,0 +1,46 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on 24-Mar-2005 + * + */ +package org.sandag.cvm.activityTravel; + +import java.util.Collection; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public interface HouseholdInterface { + + /** + * @return the ID of the household + */ + public int getId(); + + /** + * @return a collection containing the Persons in the household + */ + public Collection getPersons(); + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java new file mode 100644 index 0000000..6bd9ac2 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java @@ -0,0 +1,79 @@ +package org.sandag.cvm.activityTravel; + +import org.apache.log4j.Logger; + +public class LoggingStopAlternative extends StopAlternative { + + + static Logger logger = Logger.getLogger(LoggingStopAlternative.class); + static double[] co = new double[10]; + + static boolean loggedParts = false; + +// @Override + public double getUtility() { + if (!loggedParts) { + logger.info("Parts of utility function are travel,destination,returnHomeTravel,returnHomeDisutility,travelDisutility,returnHomeTime,travelTime,angle,zoneType,sizeTerm"); + loggedParts=true; + } + + co[0] = c.travelUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); + co[1] = c.destinationUtilityFunction.calcForIndex(location,1); + // TODO should be logsum of trip mode + co[2] = c.returnHomeUtilityFunction.calcForIndex(location,c.myTour.getOriginZone()); + if (c.disutilityToOriginCoefficient!=0) { + co[3] = c.disutilityToOriginCoefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + } else co[3]=0; + if (c.disutilityToNextStopCoefficient!=0) { + co[4] = c.disutilityToNextStopCoefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + } else co[4] = 0; + if (c.timeToOriginCoefficient!=0) { + double timeToOriginUtility = c.timeToOriginCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + // Doug and Kevin Hack of Jan 5 2004 +// if (myTour.getTotalElapsedTime()>240.0) timeToOriginUtility*=3; + co[5] = timeToOriginUtility; + } else co[5]=0; + if (c.timeToNextStopCoefficient!=0) { + double timeToNextStopUtility = c.timeToNextStopCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + // Doug and Kevin Hack of Jan 5 2004 +// if (myTour.getTotalElapsedTime()>240.0) timeToNextStopUtility*=3; + co[6] = timeToNextStopUtility; + } else co[6]=0; + + if (c.xMatrix !=null && c.yMatrix != null) { + + double xOrig = c.xMatrix.getValueAt(c.myTour.getOriginZone(),1); + double yOrig = c.yMatrix.getValueAt(c.myTour.getOriginZone(),1); + double xNow = c.xMatrix.getValueAt(c.myTour.getCurrentLocation(),1); + double yNow = c.yMatrix.getValueAt(c.myTour.getCurrentLocation(),1); + double xMaybe = c.xMatrix.getValueAt(location,1); + double yMaybe = c.yMatrix.getValueAt(location,1); + double angle1 = Math.atan2(yNow-yOrig,xNow-xOrig); + double angle2 = Math.atan2(yMaybe-yNow,xMaybe-xNow); + double angle = (angle2-angle1)+Math.PI; + if (angle > Math.PI*2) angle -= Math.PI*2; + if (angle <0) angle += Math.PI*2; + if (angle > Math.PI) angle =2*Math.PI-angle; + co[7]= c.angleCoefficient*angle*180/Math.PI; + } else co[7]=0; + co[8]= c.zoneTypeUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); + if (c.sizeTermCoefficient !=0) { + double sizeTermValue = Math.log(c.sizeTerm.calcForIndex(location,1)); + co[9]= c.sizeTermCoefficient*sizeTermValue; + } else co[9]=0; + StringBuffer logStatement = new StringBuffer("OD "+c.getTour().getCurrentLocation()+","+location+" :"); + double uti=0; + for (int index=0;indexduration in hours + */ + public float duration; + /** + * location represents the location of the stop + */ + public int location; + /** + * purpose represents the purpose of the stop + */ + public int purpose; + public double travelTimeMinutes; + public String tripMode = "NA"; + public final int previousLocation; + public final double departureTimeFromPreviousStop; + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java new file mode 100644 index 0000000..9249773 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java @@ -0,0 +1,106 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + + +public class StopAlternative implements CodedAlternative { + final StopChoice c; + public final int location; + public double getUtility() { + + if (c.maxDist >0) { + assert c.distanceMatrix!=null : "Distance Matrix didn't get initialized, yet maxDist set to "+c.maxDist; + double distance = c.distanceMatrix.getValueAt(c.myTour.getOriginZone(),location); + + if((distance<-99999) || (distance>99999)) + distance=0; + + if (distance >c.maxDist) { + return Double.NEGATIVE_INFINITY; + } + } + + boolean firstStop = (c.myTour.getStopCounts()[0]==0); + + double utility = c.travelUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); + utility += c.destinationUtilityFunction.calcForIndex(location,1); + utility += c.returnHomeUtilityFunction.calcForIndex(location,c.myTour.getOriginZone()); + + if (c.disutilityToOriginCoefficient!=0 || c.disutilityToOriginAdditionalCoefficientForStopGT1!=0) { + double coefficient = c.disutilityToOriginCoefficient + (firstStop ? 0 : c.disutilityToOriginAdditionalCoefficientForStopGT1); + utility += coefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + } + if (c.disutilityToNextStopCoefficient!=0 || c.disutilityToNextStopAdditionalCoefficientForStopGT1 !=0) { + double coefficient = c.disutilityToNextStopCoefficient + (firstStop ? 0 : c.disutilityToNextStopAdditionalCoefficientForStopGT1); + utility += coefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + } + if (c.timeToOriginCoefficient!=0) { + double timeToOriginUtility = c.timeToOriginCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + // Doug and Kevin Hack of Jan 5 2004 +// if (myTour.getTotalElapsedTime()>240.0) timeToOriginUtility*=3; + utility += timeToOriginUtility; + } + if (c.timeToNextStopCoefficient!=0) { + double timeToNextStopUtility = c.timeToNextStopCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); + // Doug and Kevin Hack of Jan 5 2004 +// if (myTour.getTotalElapsedTime()>240.0) timeToNextStopUtility*=3; + utility += timeToNextStopUtility; + } + + + if (c.xMatrix !=null && c.yMatrix != null) { + // angle calculation and max distance + + double xOrig = c.xMatrix.getValueAt(c.myTour.getOriginZone(),1); + double yOrig = c.yMatrix.getValueAt(c.myTour.getOriginZone(),1); + double xNow = c.xMatrix.getValueAt(c.myTour.getCurrentLocation(),1); + double yNow = c.yMatrix.getValueAt(c.myTour.getCurrentLocation(),1); + double xMaybe = c.xMatrix.getValueAt(location,1); + double yMaybe = c.yMatrix.getValueAt(location,1); + + double angle1 = Math.atan2(yNow-yOrig,xNow-xOrig); + double angle2 = Math.atan2(yMaybe-yNow,xMaybe-xNow); + double angle = (angle2-angle1)+Math.PI; + if (angle > Math.PI*2) angle -= Math.PI*2; + if (angle <0) angle += Math.PI*2; + if (angle > Math.PI) angle =2*Math.PI-angle; + utility += c.angleCoefficient*angle*180/Math.PI; + } + utility += c.zoneTypeUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); + if (c.sizeTermCoefficient !=0) { + double sizeTermValue = Math.log(c.sizeTerm.calcForIndex(location,1)); + utility += c.sizeTermCoefficient*sizeTermValue; + } + return utility; + } + + public StopAlternative(StopChoice choice, int stopLocation) { + this.location = stopLocation; + this.c = choice; + } + + public String getCode() { + return String.valueOf(location); + } + + + + + } \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java new file mode 100644 index 0000000..38f419a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java @@ -0,0 +1,230 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.common.emme2.IndexConditionFunction; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.LogitModel; +import com.pb.common.matrix.*; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + * Modified 2014 for trip mode choice + */ +public abstract class StopChoice extends LogitModel implements ModelUsesMatrices { + + static int angleCalculationCounter = 50; + public static MatrixCacheReader myMatrixCacheReader=null; + + static Logger logger = Logger.getLogger(StopChoice.class); + /*private static Boolean useTripModeChoice = null;*/ + + + + protected double angleCoefficient; + protected double maxDist; + + protected IndexLinearFunction destinationUtilityFunction = new IndexLinearFunction(); + + protected double disutilityToNextStopAdditionalCoefficientForStopGT1; + protected double disutilityToNextStopCoefficient; + protected double disutilityToOriginAdditionalCoefficientForStopGT1; + protected double disutilityToOriginCoefficient; + protected Tour myTour; + protected IndexLinearFunction returnHomeUtilityFunction = new IndexLinearFunction(); + protected IndexLinearFunction sizeTerm = null; + protected double sizeTermCoefficient = 0; + protected double timeToNextStopCoefficient; + protected double timeToOriginCoefficient; + protected IndexLinearFunction travelUtilityFunction = new IndexLinearFunction(); + protected Matrix xMatrix; + private String xMatrixName; + protected Matrix yMatrix; + private String yMatrixName; + protected IndexConditionFunction zoneTypeUtilityFunction = new IndexConditionFunction(); + public Matrix distanceMatrix; + private String distanceMatrixName; + /** + * Method addCoefficient. + * @param alternative + * @param index1 + * @param index2 + * @param matrix + * @param coefficient + */ + public void addCoefficient( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError { + if (!alternative.equals("zone")) throw new RuntimeException("StopAlternative coefficients must have \"zone\" as alternative"); + + //If Index1 is "cstop" then Index2 must equal "nstop", and the term in the utility function is the entry from the mf matrix identified in the Matrix field, indexed with i being the current stop location and j being the next stop location whose utility is being evaluated. + if(index1.equals("cstop")) { + if(index2.equals("nstop")) { + travelUtilityFunction.addCoefficient(matrix,coefficient); + } else throw new CoefficientFormatError("cstop coefficients for next stop location must index an mf matrix using nstop as the J"); + } + + + else if (index1.equals("nstop")) { + //If Index1 is "nstop" and Index2 is "origin" then Matrix identifies an mf matrix, and the term in the utility function is the matrix value indexed with i being the next stop location whose utility is being evaluated and j is the origin of the tour. + if(index2.equals("origin")) { + returnHomeUtilityFunction.addCoefficient(matrix,coefficient); + //If Index1 is "nstop" and Index2 is blank, then Matrix identifies an mo or md matrix. The next stop location under consideration is used to retrieve the appropriate entry from the matrix. + } else if (index2.equals("") ||index2.equals("none")) { + destinationUtilityFunction.addCoefficient(matrix,coefficient); + } else throw new CoefficientFormatError("nstop coefficients for next stop location must have index2=\"\" or index2 = origin"); + } + else if (index1.equals("angle")) { + angleCoefficient = coefficient; + setXYNames(matrix, index2); + } + else if (index1.equals("travelDisutility")) { + if (index2.equals("nstop")) { + disutilityToNextStopCoefficient+=coefficient; + } else if (index2.equals("origin")) { + disutilityToOriginCoefficient+=coefficient; + } else if (index2.equalsIgnoreCase("nstopx")){ + disutilityToNextStopAdditionalCoefficientForStopGT1 += coefficient; + } else if (index2.equalsIgnoreCase("originx")) { + disutilityToOriginAdditionalCoefficientForStopGT1 += coefficient; + } else { + throw new CoefficientFormatError("travelDisutility coefficients for next stop must have be to either \"origin\" or to \"nstop\""); + } + } + else if (index1.equals("travelTime")) { + if (index2.equals("nstop")) { + timeToNextStopCoefficient+=coefficient; + } else if (index2.equals("origin")) { + timeToOriginCoefficient+=coefficient; + } else { + throw new CoefficientFormatError("travelDisutility coefficients for next stop must have be to either \"origin\" or to \"nstop\""); + } + } + else if (index1.equals("sizeTerm1")) { + sizeTermCoefficient += coefficient; + if (sizeTerm== null) sizeTerm = new IndexLinearFunction(); + sizeTerm.addCoefficient(matrix,1.0); + } + else if (index1.equals("sizeTerm2") || index1.equals("sizeTermx")) { + if (sizeTerm== null) sizeTerm = new IndexLinearFunction(); + sizeTerm.addCoefficient(matrix,coefficient); + } + else if (index1.equals("maxDist")) { + maxDist = coefficient; + distanceMatrixName = matrix; + } + else { + int destinationCondition; + int originCondition; + try { + destinationCondition = Integer.valueOf(index1).intValue(); + } catch (NumberFormatException e) { + throw new CoefficientFormatError("Can't convert "+index1+" to a number, not allowed as an index type for stop location choice"); + } + boolean twoTypeCondition=false; + try { + originCondition = Integer.valueOf(index2).intValue(); + zoneTypeUtilityFunction.addCoefficient(matrix,destinationCondition,originCondition,coefficient); + twoTypeCondition= true; + } catch (NumberFormatException e) { + } + if (!twoTypeCondition) { + zoneTypeUtilityFunction.addCoefficient(matrix,destinationCondition,coefficient); + } + } + } + private void setXYNames(String xName, String yName) { + if (xMatrixName==null) { + xMatrixName = xName; + } else{ + if (!xMatrixName.equals(xName)) { + String msg = "xName for angle and maxdist needs to be the same, "+xMatrixName+"!="+xName; + logger.fatal(msg); + throw new RuntimeException(msg); + } + } + if (yMatrixName == null) { + yMatrixName = yName; + } else { + if (!yMatrixName.equals(yName)) { + String msg = "yName for angle and maxdist needs to be the same, "+yMatrixName+"!="+yName; + logger.fatal(msg); + throw new RuntimeException(msg); + } + } + } + /** + * Returns the myTour. + * @return CommercialTour + */ + public Tour getTour() { + return myTour; + } + public void init() { + if (myMatrixCacheReader==null) throw new RuntimeException("StopChoice needs an initialized Emme2MatrixReader before it can be initialized"); + readMatrices(myMatrixCacheReader); + } + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader mr) { + travelUtilityFunction.readMatrices(mr); + destinationUtilityFunction.readMatrices(mr); + returnHomeUtilityFunction.readMatrices(mr); + zoneTypeUtilityFunction.readMatrices(mr); + if (sizeTerm != null) sizeTerm.readMatrices(mr); + if (xMatrixName != null) xMatrix = mr.readMatrix(xMatrixName); + if (yMatrixName!=null) yMatrix = mr.readMatrix(yMatrixName); + if (distanceMatrixName!=null) { + distanceMatrix = mr.readMatrix(distanceMatrixName); + } else { + logger.warn("Distance matrix name not specified, no maximum distance set"); + } + } + + /** + * Sets the myTour. + * @param myTour The myTour to set + */ + public void setTour(Tour myTour) { + this.myTour = myTour; + } + /*public static Boolean getUseTripModeChoice() { + return useTripModeChoice; + } + public static void setUseTripModeChoice(Boolean useTripModeChoice) { + StopChoice.useTripModeChoice = useTripModeChoice; + }*/ + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java new file mode 100644 index 0000000..7ed78ef --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java @@ -0,0 +1,290 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on Feb 4, 2005 + * + */ +package org.sandag.cvm.activityTravel; + +import java.util.ArrayList; +import java.util.Iterator; + +import org.apache.log4j.Logger; + +//import org.sandag.cvm.calgary.commercial.TourStartTimeModel; +//import org.sandag.cvm.calgary.commercial.WeekendTravelTimeTracker; +import org.sandag.cvm.activityTravel.cvm.TourStartTimeModel; +import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; +import org.sandag.cvm.common.model.NoAlternativeAvailable; + + +/** + * @author jabraham + * + * This is a representation of a tour. + */ +public abstract class Tour implements TourInterface { + + public abstract ChangingTravelAttributeGetter getElapsedTravelTimeCalculator(); + public abstract TourStartTimeModel getTourStartTimeModel(); + +// @Override + public String toString() { + StringBuffer buff = new StringBuffer("Tour from "+originZone+" via "); + for (int i=0;istops is an ArrayList containing instances of Tour.Stop, one for each stop made on the tour + */ + protected ArrayList stops = new ArrayList(); + + private double tourStartTimeHrs = 0; + private double travelTimeMinutes; + + static Logger logger = Logger.getLogger(Tour.class); + + /** + * Adds a stop to the stops and alsu updates travelTimeMinutes and currentTimeHrs + * @param newStop + */ + protected void addStop(Stop newStop) { + int lastStopLocation = getOrigin(); + if (stops.size()!=0) lastStopLocation = ((Stop) stops.get(stops.size()-1)).location; + stops.add(newStop); + // TODO need to use trip mode not tour mode for travel time tracking. + // TODO should account for toll/non toll for travel time tracking (e.g. if toll is a trip mode.) + double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastStopLocation,newStop.location,currentTimeHrs,myVehicleTourType.getVehicleType()); + if (legTravelTime == 0 ) { + // a problem + logger.warn("Leg travel time is zero for "+lastStopLocation + " to " + newStop.location); + } + newStop.travelTimeMinutes=legTravelTime; + travelTimeMinutes+=legTravelTime; + currentTimeHrs += (legTravelTime/60 + newStop.duration); + } + + protected double calcCurrentTime() { + double travelTime =0; + double currentTimeCalc = getTourStartTimeHrs(); + Iterator stopIt = stops.iterator(); + int lastStop = getOriginZone(); + while (stopIt.hasNext()) { + Stop stop = (Stop) stopIt.next(); + // TODO need to use trip mode not tour mode for travel time tracking. + double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastStop,stop.location,currentTimeCalc,myVehicleTourType.getVehicleType()); + travelTime += legTravelTime; + currentTimeCalc += legTravelTime/60; + currentTimeCalc += stop.duration; + lastStop = stop.location; + } + return currentTimeCalc; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getCurrentTimeHrs() + */ + public double getCurrentTimeHrs() { + return currentTimeHrs; + } + + /** + * Method getLastStopType. + * @return int + */ + public int getLastStopType() { + if (stops.size()==0) return 0; + Stop theLastStop = (Stop) stops.get(stops.size()-1); + return theLastStop.purpose; + } + + /** + * @return Returns the myVehicleTourType. + */ + public TourType getMyVehicleTourType() { + return myVehicleTourType; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getOrigin() + */ + public int getOrigin() { + return getOriginZone(); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getOriginZone() + */ + public int getOriginZone() { + return originZone; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getStopCount() + */ + public int[] getStopCounts() { + int[] stopCounter = new int[getMaxTourTypes()+1]; + Iterator stopIt = stops.iterator(); + while (stopIt.hasNext()) { + Stop stop = (Stop) stopIt.next(); + stopCounter[0]++; + stopCounter[stop.purpose]++; + } + return stopCounter; + } + + public abstract int getMaxTourTypes(); + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getTotalElapsedTime() + */ + public double getTotalElapsedTimeHrs() { + return getCurrentTimeHrs()-getTourStartTimeHrs(); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getTotalTravelTimeMinutes() + */ + public double getTotalTravelTimeMinutes() { + return travelTimeMinutes; + } + + /** + * Method getTourStartTime. + * @return double + */ + protected double getTourStartTimeHrs() { + return tourStartTimeHrs; + } + + /** + * Method getTourTypeCode. + * @return A string grepresenting the type of tour -- perhaps representing the types + * of activities that occur in the tour. + */ + protected String getTourTypeCode() { + return myVehicleTourType.getCode().substring(1); + } + + /** + * Method getVehicleCode. + * @return a string representing the tNCVehicle code used for the tour + */ + protected String getVehicleCode() { + return myVehicleTourType.getCode().substring(0,1); + } + + /** + * @return Returns the vehicleTourTypeChoice. + */ + public abstract VehicleTourTypeChoice getVehicleTourTypeChoice(); + + /** + * Method sampleStartTime. + */ + public void sampleStartTime() { + tourStartTimeHrs = getTourStartTimeModel().sampleValue(); + if (stops.size()==0) currentTimeHrs = tourStartTimeHrs; + else currentTimeHrs = calcCurrentTime(); + + } + + /** + * This method uses a random number generator to sample the stops along the tour -- their location, + * duration and purpose. + */ + public abstract void sampleStops(); + + /** + * This method uses a random number generator to sample the type of tour, including + * the type of tNCVehicle(s) used for the tour and perhaps some information about the types + * of activities that occur along the tour. + */ + public void sampleVehicleAndTourType() { + getVehicleTourTypeChoice().setMyTour(this); + try { + myVehicleTourType = (TourType) getVehicleTourTypeChoice().monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + myVehicleTourType = null; + //Leave it null for an error to occur when it's actually needed + } + } + + /** + * @param currentTimeHrs The currentTimeHrs to set. + */ + protected void setCurrentTimeHrs(double currentTimeHrs) { + this.currentTimeHrs = currentTimeHrs; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#setOrigin(int) + */ + public void setOrigin(int z) { + setOriginZone(z); + } + + void setOriginZone(int originZone) { + this.originZone = originZone; + } + + /** + * @param tourStartTimeHrs The time when the tour starts. + */ + protected void setTourStartTimeHrs(double tourStartTimeHrs) { + this.tourStartTimeHrs = tourStartTimeHrs; + } + + /** + * @param vehicleTourTypeChoice The vehicleTourTypeChoice to set. + */ + public abstract void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoice); + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.TourInterface#getCurrentLocation() + */ + public int getCurrentLocation() { + if (stops.size()==0) return getOriginZone(); + Stop stop = (Stop) stops.get(stops.size()-1); + return stop.location; + } + + public abstract ChangingTravelAttributeGetter getTravelDisutilityTracker() ; + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java new file mode 100644 index 0000000..c755573 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java @@ -0,0 +1,61 @@ +package org.sandag.cvm.activityTravel; + +public interface TourInterface { + + /** + * Method getCurrentTime + * @deprecated + * @return double current time of day in hours + */ + public double getCurrentTimeHrs(); + + /** + * Method getCurrentTime + * @return double current time of day in minutes + */ + public double getCurrentTimeMinutes(); + + /** + * Method getOrigin. + * @return an int representing the origin location of the tour. + */ + public int getOrigin(); + + /** + * @return the integer representing the origin of the tour + */ + public int getOriginZone(); + + /** + * Method getStopCount. + * @return int[] an integer array counting the stops that occur by type. Element 0 + * is the total number of stops; other elements correspond to different stop purposes + */ + public int[] getStopCounts(); + + /** + * Method getTotalElapsedTime. + * @deprecated + * @return double total elapsed time in hours + */ + public double getTotalElapsedTimeHrs(); + + /** + * Method getTotalElapsedTime. + * @return double total elapsed time in hours + */ + public double getTotalElapsedTimeMinutes(); + + /** + * Method getTotalTravelTime. + * @return double total travel time in minutes + */ + public double getTotalTravelTimeMinutes(); + + /** + * Method getCurrentLocation. + * @return int + */ + public int getCurrentLocation(); + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java new file mode 100644 index 0000000..1ddaeaa --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java @@ -0,0 +1,45 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on 25-Feb-2005 + * + */ +package org.sandag.cvm.activityTravel; + +import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; + + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public interface TourNextStopPurposeChoice extends DiscreteChoiceModelInterface { + /** + * sets the tour to be used for independent variables to influence the choice of next stop purpose + * @param myTour + */ + public abstract void setMyTour(Tour myTour); + /** + * @return the tour currently being used for information that influences the next stop purpose choice + */ + public abstract Tour getMyTour(); +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java new file mode 100644 index 0000000..fefb6a1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java @@ -0,0 +1,115 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on Feb 4, 2005 + * + */ +package org.sandag.cvm.activityTravel; + +/** + * @author jabraham + * + * This class is a representation of a type of tour. Contained it in + * are counters for the number of instances of the type of tour and trips + * as well as a choice model for choosing a tNCVehicle type for the tour. + * The class is identified by a name which is a String. + */ +public abstract class TourType { + + /** + * @param tourType a string representing the type of tour, which may also have information on tNCVehicle types + * @param vehicleType a char representing the type of tNCVehicle + * @param theChoiceModel the choice model associated with the tour type + */ + public TourType(String tourType, char vehicleType, VehicleTourTypeChoice theChoiceModel){ + myChoice = theChoiceModel; + this.tourTypeName = tourType; + this.vehicleType= vehicleType; + } + + /** + * tourCount is a counter for the number of tours of this type. + */ + protected int tourCount = 0; + protected int tripCount = 0; + + /** + * This class can serve as a place to keep track of the number of tours and trips of different types. + * This method increments the number of tours and trips. + * @param tours the number of tours to increment the tour counter by + * @param trips the number of trips to increment the trip counter by + */ + public void incrementTourAndTripCount(int tours, int trips) { + tourCount += tours; + tripCount += trips; + } + + /** + * tourTypeName is the unique identifier of the type of tour + */ + public final String tourTypeName; + /** + * vehicleType is the identifier for the tNCVehicle type used in the tour + */ + public final char vehicleType; + + /** + * @return the tourTypeName + */ + public String getCode() { + return getTourTypeName(); + } + + /** + * @return the tourTypeName + */ + public String getTourTypeName() { + return tourTypeName; + } + + /** + * @return the char representing the tNCVehicle type + */ + public char getVehicleType() { + return vehicleType; + } + + /** + * @return the utility of this alternative + */ + public abstract double getUtility(); + + /** + * @return Returns the tourCount. + */ + public int getTourCount() { + return tourCount; + } + + /** + * @return Returns the tripCount. + */ + public int getTripCount() { + return tripCount; + } + + protected final VehicleTourTypeChoice myChoice; + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java new file mode 100644 index 0000000..14ef56a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java @@ -0,0 +1,117 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel; + +import java.util.ArrayList; +import java.util.Iterator; + +import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; +import org.sandag.cvm.activityTravel.ModelUsesMatrices; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public abstract class TravelTimeTracker + implements ChangingTravelAttributeGetter, ModelUsesMatrices { + + protected final ArrayList travelTimeMatrices = new ArrayList(); + + + + public static class TravelTimeMatrixSpec { + final String name; + Matrix matrix; + final float startTime; + final float endTime; + final char vehicleType; + + public TravelTimeMatrixSpec(String name, float startTime, float endTime, char vehicleType) { + this.name = name; + this.startTime = startTime; + this.endTime=endTime; + this.vehicleType = vehicleType; + } + + /** + * Method readMatrices. + * @param matrixReader + */ + void readMatrices(MatrixCacheReader matrixReader) { + matrix = matrixReader.readMatrix(name); + } + + /** + * Method getTimeFromMatrix. + * @param origin + * @param destination + * @return double + * + * Note: modified to return 0 if value is greater than 99999 + */ + double getTimeFromMatrix(int origin, int destination) { + double value= matrix.getValueAt(origin,destination); + if((value>(-99999)) && (value<99999)) + return value; + else + return 0; + } + + + + } + + /** + * Method get. + * @param lastStop + * @param i + * @param currentTime + * @return double + */ + public double getTravelAttribute(int origin, int destination, double timeOfDay, char vehicleType) { + TravelTimeMatrixSpec defaultMatrix = null; + while (timeOfDay>=24.00) timeOfDay -=24.00; + for (int i =0; i=s.startTime && timeOfDay < s.endTime && s.vehicleType == vehicleType) return s.getTimeFromMatrix(origin,destination); + else if (s.startTime <0) defaultMatrix = s; + } + } + if (defaultMatrix==null) throw new RuntimeException("no default travel time matrix for tNCVehicle type "+vehicleType); + return defaultMatrix.getTimeFromMatrix(origin,destination); + } + + public void readMatrices(MatrixCacheReader matrixReader) { + Iterator it = travelTimeMatrices.iterator(); + while (it.hasNext()) { + TravelTimeMatrixSpec s = (TravelTimeMatrixSpec) it.next(); + s.readMatrices(matrixReader); + } + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java new file mode 100644 index 0000000..4c0f205 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java @@ -0,0 +1,74 @@ +package org.sandag.cvm.activityTravel; + +import org.sandag.cvm.common.emme2.MatrixCacheReader; + +public abstract class TripMode { + + /** + * First part of string is tNCVehicle type aka tour mode. + * Followed by a colon and then the trip mode. + * e.g L:T means tour "Light" and trip-mode "toll" + */ + protected final String myType; + protected final TripModeChoice myChoiceModel; + public final char vehicleType; + public final String tripMode; + protected int origin; + protected int destination; + protected double timeOfDay; + + public TripMode( + TripModeChoice choiceModel, + String type) { + myType = type; + myChoiceModel = choiceModel; + vehicleType = myType.split(":")[0].charAt(0); + tripMode = myType.split(":")[1]; + + } + + + public abstract double getUtility(); + + + public String getCode() { + return myType; + } + + public abstract void readMatrices(MatrixCacheReader matrixReader); + + public abstract void addCoefficient(String index1, String index2, + String matrix, double coefficient) throws CoefficientFormatError; + + public int getDestination() { + return destination; + } + + public void setDestination(int destination) { + this.destination = destination; + } + + public int getOrigin() { + return origin; + } + + public void setOrigin(int origin) { + this.origin = origin; + } + + public void setTime(double time) { + timeOfDay = time; + + } + + public String logOriginDestination() { + return String.valueOf(origin)+" to "+destination; + } + + public String getTripMode() { + return tripMode; + } + + + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java new file mode 100644 index 0000000..391a6eb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java @@ -0,0 +1,96 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on Feb 4, 2005 + * + */ +package org.sandag.cvm.activityTravel; + +import java.util.Iterator; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.activityTravel.cvm.CommercialTripMode; +import org.sandag.cvm.common.emme2.MatrixAndTAZTableCache; +import org.sandag.cvm.common.model.Alternative; +import org.sandag.cvm.common.model.ChoiceModelOverflowException; +import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; +import org.sandag.cvm.common.model.LogitModel; +import org.sandag.cvm.common.model.NoAlternativeAvailable; + + +/** + * @author jabraham + * + * A model of tNCVehicle type together with tour type + */ +public abstract class TripModeChoice implements ChangingTravelAttributeGetter { + + protected LogitModel myLogitModel = new LogitModel(); + + protected static Logger logger = Logger.getLogger(TripModeChoice.class); + + protected Tour theTour; + + /** + * @param myTour the tour to consider when making the tour type choice + */ + public void setMyTour(Tour myTour) { + theTour = myTour; + } + + + + /** + * @return the tour associated with the tour type choice + */ + public Tour getMyTour() { + return theTour; + } + + + + @Override + public double getTravelAttribute(int origin, int destination, double time, + char vehicleType) { + Iterator m = myLogitModel.getAlternativesIterator(); + while (m.hasNext()) { + TripMode tm = (TripMode) m.next(); + tm.setOrigin(origin); + tm.setDestination(destination); + tm.setTime(time); + } + return myLogitModel.getUtility(); + } + + + + public void readMatrices(MatrixAndTAZTableCache matrixReader) { + Iterator m = myLogitModel.getAlternativesIterator(); + while (m.hasNext()) { + CommercialTripMode tm = (CommercialTripMode) m.next(); + tm.readMatrices(matrixReader); + } + } + + + public abstract TripMode chooseTripModeForDestination(int location) throws ChoiceModelOverflowException, NoAlternativeAvailable; + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java new file mode 100644 index 0000000..d87ec63 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java @@ -0,0 +1,51 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +/* + * Created on Feb 4, 2005 + * + */ +package org.sandag.cvm.activityTravel; + +import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; + + +/** + * @author jabraham + * + * A model of tNCVehicle type together with tour type + */ +public interface VehicleTourTypeChoice extends DiscreteChoiceModelInterface { + /** + * @param myTour the tour to consider when making the tour type choice + */ + public void setMyTour(Tour myTour); + + /** + * + */ + public void writeTourAndTripSummary(); + + /** + * @return the tour associated with the tour type choice + */ + Tour getMyTour(); + + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java new file mode 100644 index 0000000..a2ffddd --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java @@ -0,0 +1,7 @@ +package org.sandag.cvm.activityTravel; + +public interface ZonePairDisutility { + + public abstract double calcForIndex(int i, int j); + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java new file mode 100644 index 0000000..6c04cf1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java @@ -0,0 +1,21 @@ +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.common.model.LogitModel; + +public class AlogitLogitModelNest extends LogitModel { + + double nestingCoefficient = 1.0; + + public void setAlogitNestingCoefficient(double coefficient) { + nestingCoefficient = coefficient; + } + + @Override + public double getUtility() { + if (getDispersionParameter()!=1.0) { + throw new RuntimeException("Alogit nesting always needs a dispersion parameter of 1.0 in the lower level nests"); + } + return super.getUtility()*nestingCoefficient; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java new file mode 100644 index 0000000..c937cad --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java @@ -0,0 +1,64 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.activityTravel.StopAlternative; +import org.sandag.cvm.activityTravel.StopChoice; +import com.pb.common.matrix.Matrix; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CommercialNextStopChoice extends StopChoice { + + private String segmentId; + + /** + * Constructor for VehicleTypeChoice. + */ + + public CommercialNextStopChoice(int[] zoneNums, int notEqualToOrLowerThan, int notEqualToOrHigherThan, String segmentID) { + super(); + this.segmentId = segmentID; + for (int z = 1;z notEqualToOrLowerThan && theNumber < notEqualToOrHigherThan) { + this.addAlternative(new StopAlternative(this, zoneNums[z])); + } + } + } + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateCommercialTours.matrixReader); + } + + @Override + public String toString() { + return "Stop choice for "+segmentId; + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java new file mode 100644 index 0000000..7409cc7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java @@ -0,0 +1,249 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.LogitModel; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +import java.util.*; + +import org.apache.log4j.Logger; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CommercialNextStopPurposeChoice extends LogitModel implements ModelUsesMatrices, TourNextStopPurposeChoice { + + /** + * Constructor for VehicleTypeChoice. + */ + + final static int SERVICE = 1; + final static int GOODS = 2; + final static int OTHER = 3; + final static int RETURNTOORIGIN = 4; + private static Logger logger = Logger.getLogger(CommercialNextStopPurposeChoice.class); + + final char tourType; + + /** + * Constructor CommercialNextStopPurposeChoice. + * @param c + */ + public CommercialNextStopPurposeChoice(char c) { + if (c == 'S' || c == 's') this.addAlternative(new NextStopPurpose(SERVICE)); + if (c == 'G' || c == 'g') this.addAlternative(new NextStopPurpose(GOODS)); + this.addAlternative(new NextStopPurpose(OTHER)); + this.addAlternative(new NextStopPurpose(RETURNTOORIGIN)); + tourType = c; + } + + static String decodeStopPurpose(int s) { + if (s==SERVICE) return "Srv"; + if (s==GOODS) return "Gds"; + if (s==OTHER) return "Oth"; + if (s==RETURNTOORIGIN) return "Est"; + String msg = "Bad stop purpose code "+s; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + + private CommercialTour myTour; + + + public class NextStopPurpose implements AlternativeUsesMatrices { + double[] transitionConstants = {0,0,0,0}; + double[] stopCountCoefficients = {0,0,0,0}; + /** + * Constructor VehicleTypeAlternative. + * @param stopType + */ + public NextStopPurpose(int stopType) { + this.stopType = stopType; + } + + double constant = 0; + + final int stopType; + IndexLinearFunction previousStopUtility = new IndexLinearFunction(); + IndexLinearFunction originUtility = new IndexLinearFunction(); + IndexLinearFunction returnToOriginUtility = new IndexLinearFunction(); + double timeToOriginCoefficient = 0; + double disutilityToOriginCoefficient = 0; + double totalTravelTimeCoefficient = 0; + double totalTripTimeCoefficient = 0; + public double getUtility() { + double utility = previousStopUtility.calcForIndex(myTour.getCurrentLocation(),1); + utility += originUtility.calcForIndex(getMyTour().getOriginZone(),1); + int previousStopType= myTour.getLastStopType(); + utility += transitionConstants[previousStopType]; + int[] stopCounts = getMyTour().getStopCounts(); + // can't return home on first stop + if (stopCounts[0]==0 && stopType==RETURNTOORIGIN) utility += Double.NEGATIVE_INFINITY; + utility += stopCountCoefficients[0]*Math.log(stopCounts[0] +1) + + stopCountCoefficients[1]*Math.log(stopCounts[1]+1) + + stopCountCoefficients[2]*Math.log(stopCounts[2]+1) + + stopCountCoefficients[3]*Math.log(stopCounts[3]+1); + double returnHomeUtility = returnToOriginUtility.calcForIndex(myTour.getCurrentLocation(),getMyTour().getOriginZone()); + + // make people return home more -- Doug and Kevin Hack of Jan 5th + //if (myTour.getTotalElapsedTime()>240.0) returnHomeUtility *=3; + utility += returnHomeUtility; + + utility += totalTravelTimeCoefficient*getMyTour().getTotalTravelTimeMinutes(); + utility += totalTripTimeCoefficient*getMyTour().getTotalElapsedTimeHrs(); + utility += timeToOriginCoefficient*getMyTour().getElapsedTravelTimeCalculator().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); + utility += disutilityToOriginCoefficient*getMyTour().getTravelDisutilityTracker().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); + utility += constant; + return utility; + } + + /** + * Method addParameter. + * @param matrix + * @param coefficient + */ + public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { + if(index1.equals("origin")) { + originUtility.addCoefficient(matrix,coefficient); + } else if (index1.equals("cstop")) { + if (index2.equals("origin")) returnToOriginUtility.addCoefficient(matrix,coefficient); + else previousStopUtility.addCoefficient(matrix,coefficient); + } else if (index1.equals("prevStopType")) { + if (index2.equals("goods")) transitionConstants[GOODS] = coefficient; + else if (index2.equals("service")) transitionConstants[SERVICE] = coefficient; + else if (index2.equals("other")) transitionConstants[OTHER] = coefficient; + else if (index2.equals("return")) transitionConstants[RETURNTOORIGIN]= coefficient; + else throw new RuntimeException("previous stop type not known: "+index2); + } else if (index1.equals("logStopCount")) { + if (index2.equals("goods")) stopCountCoefficients[GOODS] = coefficient; + else if (index2.equals("service")) stopCountCoefficients[SERVICE] = coefficient; + else if (index2.equals("other")) stopCountCoefficients[OTHER] = coefficient; + else if (index2.equals("all")) stopCountCoefficients[0] = coefficient; + else throw new RuntimeException("stop count type not known: "+index2); + } else if (index1.equals("timeAccumulator")) { + totalTravelTimeCoefficient += coefficient; + } else if (index1.equals("totalAccumulator")) { + totalTripTimeCoefficient += coefficient; + } else if (index1.equals("travelDisutility") && index2.equals("origin")) { + disutilityToOriginCoefficient += coefficient; + } else if (index1.equals("travelTime") && index2.equals("origin")) { + timeToOriginCoefficient += coefficient; + } else if (index1.equals("") && index2.equals("")) { + constant += coefficient; + } else { + throw new CoefficientFormatError("invalid indexing "+index1+ ","+index2+" in matrix "+matrix +" for next stop purpose model "); + } + } + + + + /** + * Method readMatrices. + * @param mr + */ + public void readMatrices(MatrixCacheReader mr) { + previousStopUtility.readMatrices(mr); + originUtility.readMatrices(mr); + returnToOriginUtility.readMatrices(mr); + } + + /** + * Method getStopPurposeCode. + * @return String + */ + public String getCode() { + if (stopType == SERVICE) return "S"; + if (stopType == OTHER) return "O"; + if (stopType == GOODS) return "G"; + if (stopType == RETURNTOORIGIN) return "R"; + return null; + } + + @Override + public String toString() { + return "StopPurpose:"+getCode(); + } + + + +} + + /** + * Method addParameter. + * @param alternative + * @param matrix + * @param coefficient + */ + public void addCoefficient( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError { + Iterator alternativeIterator = alternatives.iterator(); + boolean found = false; + while (alternativeIterator.hasNext()) { + AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); + if (alternative.equals(alt.getCode())) { + alt.addCoefficient(index1,index2,matrix,coefficient); + found = true; + } + } + if (!found) throw new CoefficientFormatError("Bad alternative in next stop purpose choice model: "+alternative); + } + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader matrixCacheReader) { + Iterator alternativeIterator = alternatives.iterator(); + while (alternativeIterator.hasNext()) { + AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); + alt.readMatrices(matrixCacheReader); + } + } + + public void setMyTour(Tour myTour) { + this.myTour = (CommercialTour) myTour; + } + + public Tour getMyTour() { + return myTour; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateCommercialTours.matrixReader); + } + + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java new file mode 100644 index 0000000..279c3ba --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java @@ -0,0 +1,328 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel.cvm; +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.*; + +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.*; + +import org.apache.log4j.Logger; + +import com.pb.common.matrix.*; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CommercialTour extends Tour { + + static Logger logger = Logger.getLogger(CommercialTour.class); + + + private final PrintWriter tripLog; + private Hashtable models; + private final GenerateCommercialTours generator; + + + +// private String myTripOutputMatrixName; +// Matrix emme2OutputMatrix; + + static class TripOutputMatrixSpec { + + final String name; + boolean write; + Matrix matrix; + final float startTime; + final float endTime; + final char vehicleType; + public final String tripMode; + + TripOutputMatrixSpec(String name, String type, float startTime, float endTime, char vehicleType, String tripMode) { + if (type.equals("tripOut")) { + write = true; + } else if (type.equals("timeDef")) { + write = false; + } else { + String msg = "Invalid type in index1 in TripMatrix "+type; + logger.fatal(msg); + throw new RuntimeException(msg); + } + this.name = name; + this.startTime = startTime; + this.endTime=endTime; + this.vehicleType = vehicleType; + this.tripMode = tripMode; + } + + @Override + public String toString() { + return "Outmatrix "+name+":"+vehicleType+"("+startTime+":"+endTime+")"; + } + + /** + * Method readMatrices. + * @param matrixReader + */ + void readMatrices(MatrixCacheReader matrixReader) { + matrix = null; + if (write) matrix = matrixReader.readMatrix(name); + } + + public void createMatrix(int size,int[] externalNumbers) { + matrix = new Matrix(name, "Trips for "+vehicleType+" between "+startTime+" and "+endTime,size,size); + matrix.setExternalNumbers(externalNumbers); + } + + } + + + + public CommercialTour(Hashtable models, GenerateCommercialTours generator, PrintWriter tripLog, int tourNumber) { + super(); + this.tripLog = tripLog; + this.generator = generator; + this.models = models; + tourNum = tourNumber; + } + + /** + * Method sampleVehicleAndTourType. + */ + public void sampleVehicleAndTourType() { + synchronized (generator.vehicleTourTypeChoice) { + generator.vehicleTourTypeChoice.setMyTour(this); + try { + myVehicleTourType = (TourType) generator.vehicleTourTypeChoice.monteCarloElementalChoice(); + myNextStopPurposeChoice = (CommercialNextStopPurposeChoice) models.get(myVehicleTourType.getCode()+"StopType"); + ((CommercialNextStopPurposeChoice) myNextStopPurposeChoice).setMyTour(this); + } catch (NoAlternativeAvailable e) { + throw new RuntimeException(e); + } + } + } + + /** + * Method addTripsToMatrix. + */ + public void addTripsToMatrix() { + int trips = 0; + Iterator stopIterator = stops.iterator(); + int lastLocation = getOriginZone(); + double currentTime = getTourStartTimeHrs(); + String prevStopType = "Est"; + while (stopIterator.hasNext()) { + Stop s = (Stop) stopIterator.next(); + trips ++; + int location = s.location; + // FIXME should take into account trip mode. + double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastLocation,location,currentTime,myVehicleTourType.getVehicleType()); + float midPointOfTripTime = (float) (currentTime + legTravelTime/60/2); + + String newStopType = CommercialNextStopPurposeChoice.decodeStopPurpose(s.purpose); + + String mNameForTripLog = "None"; + + TripOutputMatrixSpec spec = generator.getTripOutputMatrixSpec(getVehicleCode().charAt(0),s.tripMode,midPointOfTripTime); + if (spec!=null) { + mNameForTripLog = spec.name; + if (spec.write) { + synchronized(spec.matrix) { + // increment stop count in the matrix if this matrix is going to be written out + float mTripCountEntry = spec.matrix.getValueAt(lastLocation,location); + mTripCountEntry++; + spec.matrix.setValueAt(lastLocation,location,mTripCountEntry); + } + } + } + // write to trip log + if (tripLog !=null) { + tripLog.print("3,"+tourNum+",1,"+trips+",1,"+getOriginZone()+","+generator.segmentString1+","+prevStopType+","+newStopType+","+lastLocation+","+location+","+mNameForTripLog+","+getVehicleCode()+","+currentTime+","); + } + boolean tollAvailable = isTollAvailable(lastLocation,location,currentTime); + if (s.tripMode.equals("T") && !tollAvailable) { + logger.error("No toll available but toll chosen for "+s); + } + + currentTime += legTravelTime/60; + currentTime += s.duration; + if (tripLog !=null) { + tripLog.println(currentTime+","+s.duration+","+getTourTypeCode()+","+generator.segmentString2+","+s.tripMode+","+tollAvailable); + } + lastLocation = location; + prevStopType = newStopType; + } + myVehicleTourType.incrementTourAndTripCount(1,trips); + } + + /** + * Method sampleStops. + */ + public void sampleStops() { + ((CommercialNextStopPurposeChoice) myNextStopPurposeChoice).setMyTour(this); + CommercialNextStopChoice myNextStopModel; + do { + Stop thisStop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); + Alternative temp; + try { + temp = myNextStopPurposeChoice.monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + logger.fatal("no valid purpose alternative available for "+this, e); + throw new RuntimeException("no valid purpose alternative available for "+this, e); + } catch (RuntimeException e) { + logger.fatal("Cannot sample stops for "+this, e); + throw new RuntimeException("Cannot sample stops for "+this, e); + } + CommercialNextStopPurposeChoice.NextStopPurpose nextStopPurpose = (CommercialNextStopPurposeChoice.NextStopPurpose) temp; + thisStop.purpose=nextStopPurpose.stopType; + if (thisStop.purpose==CommercialNextStopPurposeChoice.RETURNTOORIGIN) { + thisStop.location=getOriginZone(); + chooseTripMode(thisStop); + addStop(thisStop); + break; + } + String nextStopModelCode = getVehicleCode()+getTourTypeCode()+nextStopPurpose.getCode() + "StopLocation"; + myNextStopModel = (CommercialNextStopChoice) models.get(nextStopModelCode); + if (myNextStopModel == null) throw new RuntimeException("Can't find stop model "+nextStopModelCode); + myNextStopModel.setTour(this); + try { + temp = myNextStopModel.monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + logger.error("no valid location alternative available from "+this.toString()+" for "+generator.segmentString+" stop purpose "+nextStopModelCode); + throw new RuntimeException("no valid location alternative available from "+this.toString()); + } + thisStop.location = ((StopAlternative) temp).location; + DurationModel myDurationModel = (DurationModel) models.get(myVehicleTourType.getCode()+"Duration"); + thisStop.duration = (float) myDurationModel.sampleValue(); + chooseTripMode(thisStop); + addStop(thisStop); + } while (true); + } + + private void chooseTripMode(Stop thisStop) { + if (generator.isUseTripModes()) { + CommercialVehicleTripModeChoice tmc = (CommercialVehicleTripModeChoice) generator.getTravelDisutilityTracker(getVehicleCode()); + tmc.setMyTour(this); + TripMode m; + try { + m = (TripMode) tmc.chooseTripModeForDestination(thisStop.location); + } catch (ChoiceModelOverflowException | NoAlternativeAvailable e) { + String msg = "Can't sample trip mode for "+this; + logger.fatal(msg); + throw new RuntimeException(msg); + } + thisStop.tripMode=m.getCode().split(":")[1]; + } + } + + private boolean isTollAvailable(int origin, int destination, double timeOfDay) { + if (!generator.isUseTripModes()) { + return false; + } + CommercialVehicleTripModeChoice tmc = (CommercialVehicleTripModeChoice) generator.getTravelDisutilityTracker(getVehicleCode()); + tmc.setMyTour(this); + + return tmc.isTollAvailable(origin,destination,timeOfDay); + } + + /** + * Method getTourTypeCode. + * @return String + */ + public String getTourTypeCode() { + return myVehicleTourType.getCode().substring(1); + } + + + /** + * Method getVehicleCode. + */ + public String getVehicleCode() { + return myVehicleTourType.getCode().substring(0,1); + } + + + private int tourNum; + + + + + /** + * @return + */ + public int getOriginZoneType() { + return Math.round(generator.landUseTypeMatrix.getValueAt(getOriginZone(),1)); + } + + public ChangingTravelAttributeGetter getTravelDisutilityTracker() { + if (generator.isUseTripModes()) { + return generator.getTravelDisutilityTracker(getVehicleCode()); + } + return generator.getTravelDisutilityTracker(); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.Tour#getMaxTourTypes() + */ + public int getMaxTourTypes() { + return 4; + } + + public double getCurrentTimeMinutes() { + return getCurrentTimeHrs()*60; + } + + public double getTotalElapsedTimeMinutes() { + return getTotalElapsedTimeHrs()*60; + } + + @Override + public VehicleTourTypeChoice getVehicleTourTypeChoice() { + // TODO Auto-generated method stub + return null; + } + + @Override + public void setVehicleTourTypeChoice( + VehicleTourTypeChoice vehicleTourTypeChoice) { + // TODO Auto-generated method stub + + } + + @Override + public ChangingTravelAttributeGetter getElapsedTravelTimeCalculator() { + // FIXME should use trip modes, like getTravelDisutilityTracker() does. + return generator.getElapsedTravelTimeCalculator(); + } + + @Override + public TourStartTimeModel getTourStartTimeModel() { + return generator.getTourStartTimeModel(); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java new file mode 100644 index 0000000..3a4be51 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java @@ -0,0 +1,77 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; +import org.sandag.cvm.activityTravel.CoefficientFormatError; +import org.sandag.cvm.activityTravel.ModelUsesMatrices; +import org.sandag.cvm.activityTravel.TravelTimeTracker; + + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CommercialTravelTimeTracker extends TravelTimeTracker implements ModelUsesMatrices, ChangingTravelAttributeGetter { + + /** + * Method addCoefficient. + * @param alternative + * @param index1 + * @param index2 + * @param matrix + * @param coefficient + */ + public void addCoefficient ( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError + { + if (alternative.length()!=1) { + throw new CoefficientFormatError("Alternative must be L, M, I or H for TravelTimeMatrix"); + } + char vehicleType = alternative.charAt(0); + if (vehicleType!='L' && vehicleType!='M' && vehicleType!='H' && vehicleType != 'I') { + throw new CoefficientFormatError("Alternative must be L, M, I or H for TravelTimeMatrix"); + } + if (index1.equals("default")) { + travelTimeMatrices.add(new TravelTimeMatrixSpec(matrix, -1, -1, vehicleType)); + } else if (index1.equals("")) { + travelTimeMatrices.add(new TravelTimeMatrixSpec(matrix, Float.valueOf(index2).floatValue(), (float) coefficient, vehicleType)); + } else throw new CoefficientFormatError("Index1 must be \"default\" or blank for TravelTimeMatrices"); + } + + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateCommercialTours.matrixReader); + } + + + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java new file mode 100644 index 0000000..fb7816c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java @@ -0,0 +1,256 @@ +package org.sandag.cvm.activityTravel.cvm; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.activityTravel.AlternativeUsesMatrices; +import org.sandag.cvm.activityTravel.CodedAlternative; +import org.sandag.cvm.activityTravel.CoefficientFormatError; +import org.sandag.cvm.activityTravel.TripMode; +import org.sandag.cvm.activityTravel.TripModeChoice; +import org.sandag.cvm.activityTravel.TravelTimeTracker.TravelTimeMatrixSpec; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; + +public class CommercialTripMode extends TripMode implements CodedAlternative , AlternativeUsesMatrices { + + CommercialTravelTimeTracker noTollDistance = new CommercialTravelTimeTracker(); + CommercialTravelTimeTracker noTollTime = new CommercialTravelTimeTracker(); + CommercialTravelTimeTracker tollDistance = new CommercialTravelTimeTracker(); + CommercialTravelTimeTracker totalDistance = new CommercialTravelTimeTracker(); + CommercialTravelTimeTracker tollTime = new CommercialTravelTimeTracker(); + CommercialTravelTimeTracker tollCost = new CommercialTravelTimeTracker(); + private static Logger logger = Logger.getLogger(CommercialTripMode.class); + + CommercialVehicleTripModeChoice getMyCM() { + return (CommercialVehicleTripModeChoice) myChoiceModel; + } + + public CommercialTripMode( + CommercialVehicleTripModeChoice choiceModel, + String type) { + super(choiceModel, type); + try { + switch (myType) { + case "M:T": + case "M:NT": //uses light-heavy truck skims + noTollDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); + noTollDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); + noTollDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); + noTollTime.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TIME", 0); + noTollTime.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TIME", 9); + noTollTime.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TIME", 19); + tollDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); + tollDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); + tollDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); + totalDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); + totalDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); + totalDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); + tollTime.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TIME", 0); + tollTime.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TIME", 9); + tollTime.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TIME", 19); + tollCost.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TOLLCOST", 0); + tollCost.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TOLLCOST", 9); + tollCost.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TOLLCOST", 19); + break; + case "H:T": + case "H:NT": //uses heavy-heavy truck skims + noTollDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); + noTollDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); + noTollDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); + noTollTime.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TIME", 0); + noTollTime.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TIME", 9); + noTollTime.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TIME", 19); + tollDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); + tollDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); + tollDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); + totalDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); + totalDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); + totalDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); + tollTime.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TIME", 0); + tollTime.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TIME", 9); + tollTime.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TIME", 19); + tollCost.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TOLLCOST", 0); + tollCost.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TOLLCOST", 9); + tollCost.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TOLLCOST", 19); + break; + case "I:T": + case "I:NT": //use medium-heavy truck skims + noTollDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); + noTollDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); + noTollDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); + noTollTime.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TIME", 0); + noTollTime.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TIME", 9); + noTollTime.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TIME", 19); + tollDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); + tollDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); + tollDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); + totalDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); + totalDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); + totalDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); + tollTime.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TIME", 0); + tollTime.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TIME", 9); + tollTime.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TIME", 19); + tollCost.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TOLLCOST", 0); + tollCost.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TOLLCOST", 9); + tollCost.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TOLLCOST", 19); + break; + case "L:T": + case "L:NT": //light vehicles use high-VOT auto skims + noTollDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_NT_H_DIST", 0); + noTollDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_NT_H_DIST", 9); + noTollDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_NT_H_DIST", 19); + noTollTime.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_NT_H_TIME", 0); + noTollTime.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_NT_H_TIME", 9); + noTollTime.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_NT_H_TIME", 19); + tollDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_DIST", 0); + tollDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_DIST", 9); + tollDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_DIST", 19); + totalDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_DIST", 0); + totalDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_DIST", 9); + totalDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_DIST", 19); + tollTime.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_TIME", 0); + tollTime.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_TIME", 9); + tollTime.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_TIME", 19); + tollCost.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_TOLLCOST", 0); + tollCost.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_TOLLCOST", 9); + tollCost.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_TOLLCOST", 19); + break; + default: + logger.fatal("Invalid tNCVehicle type in trip mode choice model "+myType); + throw new RuntimeException("Invalid tNCVehicle type in trip mode choice model "+myType); + } + } catch (CoefficientFormatError e) { + logger.fatal("Problem setting up coefficeint for trip modes", e); + throw new RuntimeException("Problem setting up coefficient for trip modes", e); + } + } + + @Override + public double getUtility() { + double tollOptTotalDistance = totalDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + double tollPortion = 0; + if (tollOptTotalDistance >0) { + //tollPortion = tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)/tollOptTotalDistance; + tollPortion=0; + } + double tollDisutility = 0; + switch (myType) { + // FIXME parameterize them in the .CSV file + case "L:NT": + return getMyCM().dispersionParam*( + -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.138*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); + + + case "I:NT": + case "M:NT": + return getMyCM().dispersionParam*( + -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.492*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); + + case "H:NT": + return getMyCM().dispersionParam*( + -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.580*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); + // return -999.0; + case "L:T": + + //if(toll<0.01 || toll>99999) + return -999.0; + /* + double toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + + tollDisutility = + -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.138*tollOptTotalDistance+ + -0.01 * toll; + return getMyCM().dispersionParam * tollDisutility + + getMyCM().portionParam * tollPortion; */ + case "I:T": + case "M:T": + + //if(toll<0.01 || toll>99999) + return -999.0; + /* + toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + + tollDisutility = + -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.492*tollOptTotalDistance+ + -0.01 * tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + return getMyCM().dispersionParam * tollDisutility + + getMyCM().portionParam * tollPortion; + */ + case "H:T": + + //if(toll<0.01 || toll>99999) + return -999.0; + /* toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + + tollDisutility = + -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + -0.580*tollOptTotalDistance+ + -0.01 * tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + return getMyCM().dispersionParam * tollDisutility + + getMyCM().portionParam * tollPortion; + */ + } + String msg = "Invalid tNCVehicle toll trip mode "+myType; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + @Override + public void readMatrices(MatrixCacheReader matrixReader) { + tollCost.readMatrices(matrixReader); + noTollTime.readMatrices(matrixReader); + noTollDistance.readMatrices(matrixReader); + tollDistance.readMatrices(matrixReader); + totalDistance.readMatrices(matrixReader); + tollTime.readMatrices(matrixReader); + } + + @Override + public void addCoefficient(String index1, String index2, String matrix, + double coefficient) throws CoefficientFormatError { + logger.warn("Ignoring coefficeiint "+index1+" "+index2+" "+matrix+" "+coefficient+", trip mode matrix coefficients are still hardcoded"); + } + + public double getTollDistance() { + return tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + } + + public Double getTollTime() { + return tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + } + + public Double getNonTollTime() { + return noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + } + + public String reportAttributes() { + switch (myType) { + // FIXME parameterize them in the .CSV file + case "L:NT": + case "I:NT": + case "M:NT": + case "H:NT": + return "time:"+noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + " dist:"+noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + case "L:T": + case "I:T": + case "M:T": + case "H:T": + return "time:"+tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + " dist:"+totalDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + " tolldist:"+tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ + " toll:"+tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); + } + logger.fatal("Oops invalid trip mode for reportATtributes()"); + throw new RuntimeException("Oops invalid trip mode for reportATtributes()"); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java new file mode 100644 index 0000000..9a15850 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java @@ -0,0 +1,112 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +/* + * Created on Feb 4, 2005 + * + */ +package org.sandag.cvm.activityTravel.cvm; + +import java.util.Enumeration; +import java.util.Hashtable; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; + + +public class CommercialVehicleTourType extends TourType implements AlternativeUsesMatrices { + public CommercialVehicleTourType(CommercialVehicleTourTypeChoice choice, String vehicleTourType) { + super (vehicleTourType, vehicleTourType.charAt(0), (VehicleTourTypeChoice) choice); + myId = vehicleTourType; + } + + final String myId; + + + + @Override + public String toString() { + return "VehicleTourType:"+myId; + + } + + /** + * Method readMatrices. + * @param mr + */ + public void readMatrices(MatrixCacheReader mr) { + utilityFunction.readMatrices(mr); + Enumeration it = conditionalUtilityFunctions.elements(); + while (it.hasMoreElements()) { + IndexLinearFunction bob = (IndexLinearFunction) it.nextElement(); + bob.readMatrices(mr); + } + } + + + + + IndexLinearFunction utilityFunction = new IndexLinearFunction(); + Hashtable conditionalUtilityFunctions = new Hashtable(); + + class TripOutputMatrixSpec { + String name; + Matrix matrix; + float startTime; + float endTime; + } + + public double getUtility() { + double utility = utilityFunction.calcForIndex(this.myChoice.getMyTour().getOriginZone(),1); + Integer myZoneType = new Integer(((CommercialTour) this.myChoice.getMyTour()).getOriginZoneType()); + ZonePairDisutility uf = (ZonePairDisutility) conditionalUtilityFunctions.get(myZoneType); + if (uf != null) { + utility += uf.calcForIndex(this.myChoice.getMyTour().getOriginZone(),1); + } + return utility; + } + + public void addCoefficient(String index1, String index2, String matrix, double coefficient) { + if (index1.equals("origin")) { + utilityFunction.addCoefficient(matrix,coefficient); + } else if (index1.equals("originLU") || index1.equals("originConstant")){ + Integer luCondition = Integer.valueOf(index2); + IndexLinearFunction conditionalUtilityFunction = (IndexLinearFunction) conditionalUtilityFunctions.get(luCondition); + if (conditionalUtilityFunction == null) { + conditionalUtilityFunction = new IndexLinearFunction(); + conditionalUtilityFunctions.put(luCondition,conditionalUtilityFunction); + } + if (index1.equals("originConstant")) { + conditionalUtilityFunction.addConstant(coefficient); + } else if (index1.equals("originLU")) { + conditionalUtilityFunction.addCoefficient(matrix,coefficient); + } + }else if (index1.equals("")) { + utilityFunction.addConstant(coefficient); + } else { + throw new RuntimeException("Invalid index1 for alternative "+getCode()); + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java new file mode 100644 index 0000000..1715143 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java @@ -0,0 +1,238 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.activityTravel.cvm; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Enumeration; +import java.util.Hashtable; +import java.util.Iterator; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.Alternative; +import org.sandag.cvm.common.model.LogitModel; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class CommercialVehicleTourTypeChoice extends LogitModel implements ModelUsesMatrices, VehicleTourTypeChoice { + + ArrayList alternativesWaitingToBeAddedToNestingStructure = new ArrayList(); + ArrayList allElementalAlternatives = new ArrayList(); + + private void addAlternativeInitially(CommercialVehicleTourType a) { + alternativesWaitingToBeAddedToNestingStructure.add(a); + allElementalAlternatives.add(a); + } + + /** + * Constructor for VehicleTypeChoice. + * @param types + */ + + // tour types are LS,LG,LO,MS,MG,MO,HS,HG,HO + public CommercialVehicleTourTypeChoice(String[] types, boolean nesting) { + super(); + for (String type : types) { + addAlternativeInitially(new CommercialVehicleTourType(this, type)); + } + if (!nesting) { + // no nesting structure specified + for (String type : types ) { + setUpNestingElement(type,"top",1.0); + } + } + + } + + private CommercialTour myTour; + private static Logger logger = Logger.getLogger(CommercialVehicleTourTypeChoice.class); + + /** + * Method addParameter. + * + * @param alternative + * @param matrix + * @param coefficient + * @throws CoefficientFormatError + */ + public void addCoefficient(String alternative, String index1, + String index2, String matrix, double coefficient) + throws CoefficientFormatError { + // if index1 is "nest" we will set up a nesting structure. + if (index1.equalsIgnoreCase("nest")) { + setUpNestingElement(alternative, matrix, coefficient); + } else { + boolean found = false; + Iterator myAltsIterator = allElementalAlternatives.iterator(); + while (myAltsIterator.hasNext()) { + Alternative alt = (Alternative) myAltsIterator.next(); + CommercialVehicleTourType vtt = (CommercialVehicleTourType) alt; + if (vtt.getCode().equals(alternative)) { + vtt.addCoefficient(index1, index2, matrix, coefficient); + found = true; + } + } + if (!found) throw new RuntimeException("can't find alternative "+alternative+" in vehicleTourType model"); + } + } + + private void setUpNestingElement(String alternativeName, String nestName, double coefficient) { + boolean found = false; + // first check if an elemental alternative; + for (int i=0;i myAltsIterator = myLogitModel.getAlternativesIterator(); + while (myAltsIterator.hasNext()) { + Alternative alt = myAltsIterator.next(); + CommercialTripMode tm = (CommercialTripMode) alt; + if (tm.getCode().equals(alternative)) { + tm.addCoefficient(index1, index2, matrix, coefficient); + found = true; + } + } + if (!found) { + if (index1.equalsIgnoreCase("dispersion")) { + dispersionParam = coefficient; + } else if (index1.equalsIgnoreCase("portion")) { + portionParam = coefficient; + } else { + throw new RuntimeException("can't find alternative "+alternative+" in TripMode model"); + } + } + } + + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader matrixCacheReader) { + Iterator myAltsIterator = myLogitModel.getAlternativesIterator(); + while (myAltsIterator.hasNext()) { + CommercialTripMode tm = (CommercialTripMode) myAltsIterator.next(); + tm.readMatrices(matrixCacheReader); + } + } + + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateCommercialTours.matrixReader); + } + + @Override + public TripMode chooseTripModeForDestination(int location) + throws ChoiceModelOverflowException, NoAlternativeAvailable { + CommercialTripMode choice = (CommercialTripMode) debugChooseTripModeForDestination(location); + if (choice.getTripMode().equals("T")) { + // going to check if any tolls are actually paid + double tollDistance = choice.getTollDistance(); + if (tollDistance ==0) { + // get the non toll alternative + Iterator it = myLogitModel.getAlternativesIterator(); + while (it.hasNext()) { + CommercialTripMode tm = (CommercialTripMode) it.next(); + if (tm.getTripMode().equals("NT")) { + choice = tm; + break; + } + } + } else { + // logTollTripChoice(); turn this off for now + } + } + return choice; + } + + + public TripMode debugChooseTripModeForDestination(int location) + throws ChoiceModelOverflowException, NoAlternativeAvailable { + /*// delete this debug stuff + Double tollTime = null; + Double nTollTime = null; + */ + + Iterator m = myLogitModel.getAlternativesIterator(); + while (m.hasNext()) { + CommercialTripMode tm = (CommercialTripMode) m.next(); + tm.setOrigin(getMyTour().getCurrentLocation()); + tm.setDestination(location); + tm.setTime(getMyTour().getCurrentTimeHrs()); + + /* delete this debug stuff + if (tm.getTripMode().equals("T")) { + tollTime = tm.getTollTime(); + } + if (tm.getTripMode().equals("NT")) { + nTollTime = tm.getNonTollTime(); + }*/ + } + + /* delete this debug stuff + if (tollTime==null || nTollTime==null) { + String msg = "Oops couldn't extract toll and non toll times for debugging purposes"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + if (nTollTime < tollTime ) { + logger.warn("nTollTime is less than tollTime for "+getMyTour().getCurrentLocation()+" to "+location); + } + if (nTollTime - tollTime > 5) { + // 5 minute savings! Check it out + logger.info("5 minute savings taking the toll road from "+getMyTour().getCurrentLocation()+" to "+location); + } + */ + return (TripMode) myLogitModel.monteCarloChoice(); + } + + + + protected void logTollTripChoice() { + + StringBuffer msg = new StringBuffer("Toll chosen:"); + msg.append(((CommercialTripMode) myLogitModel.alternativeAt(0)).logOriginDestination()); + msg.append(" for "); + msg.append(theTour.getMyVehicleTourType()); + msg.append(" "); + double utility1= ((CommercialTripMode) myLogitModel.alternativeAt(0)).getUtility(); + double utility2 = ((CommercialTripMode) myLogitModel.alternativeAt(1)).getUtility(); + double prob = Math.exp(utility1)/(Math.exp(utility2)+Math.exp(utility1)); + msg.append(((CommercialTripMode)myLogitModel.alternativeAt(0)).getTripMode()+" prob "+prob+" {"); + msg.append(((CommercialTripMode) myLogitModel.alternativeAt(0)).reportAttributes()); + msg.append(" or "); + msg.append(((CommercialTripMode) myLogitModel.alternativeAt(1)).reportAttributes()); + msg.append("}"); + logger.info(msg); + + } + + + public boolean isTollAvailable(int origin, int destination, double timeOfDay) { + Iterator it = myLogitModel.getAlternativesIterator(); + while (it.hasNext()) { + CommercialTripMode tm = (CommercialTripMode) it.next(); + tm.setOrigin(origin); + tm.setDestination(destination); + tm.setTime(timeOfDay); + if (tm.tripMode.equals("T")) { + if (tm.getTollDistance() ==0) return false; + } else { + return true; + } + } + throw new RuntimeException("No toll option for trip, can't report whether toll was actually available "); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java new file mode 100644 index 0000000..068aeb5 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java @@ -0,0 +1,49 @@ +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.activityTravel.CoefficientFormatError; +import org.sandag.cvm.activityTravel.DurationModel; + +public class DurationModel2 extends DurationModel { + + int functionalForm; + static boolean durationInMinutes = false; + + @Override + public void addCoefficient(String alternative, String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { + if(index1.equals("functionForm")) { + if (index2.equals("power")) functionalForm =1; + else if (index2.equals("cubic")) functionalForm = 2; + else if (index2.equals("exponential")) functionalForm = 3; + else if (index2.equals("addedexponential")) functionalForm = 0; + else throw new CoefficientFormatError("functionalForm for tour start model must have index2 as \"power\", \"cubic\" or \"exponential\""); + } else { + super.addCoefficient(alternative, index1, index2, matrix, coefficient); + } + } + + @Override + public double sampleValue() { + double y=0; + double x = Math.random(); + switch (functionalForm) { + case 0: + y = super.sampleValue(); + break; + case 1: + y = a * Math.pow(x,b)+c*Math.pow(x,d) + e * x + f; + break; + case 2: + y = a+b*x+c*x*x+d*x*x*x; + break; + case 3: + y = c*Math.exp(a*x+b)+d; + break; + default: + throw new RuntimeException("Functional form for duration model must be 0,1,2 or 3"); + } + if (durationInMinutes) y=y/60; + if (y<0) y =0; + if (y>24) y = 24; + return y; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java new file mode 100644 index 0000000..6be0c18 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java @@ -0,0 +1,855 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + + */ + +package org.sandag.cvm.activityTravel.cvm; + +/** + * @author John Abraham + * + * (c) 2003-2010 John Abraham + */ + +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixAndTAZTableCache; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.skims.HDF5MatrixReader; +import org.sandag.cvm.common.skims.OMXMatrixCollectionReader; +import org.sandag.cvm.common.skims.TranscadMatrixCollectionReader; +import org.sandag.cvm.common.datafile.CSVFileReader; +import org.sandag.cvm.common.datafile.TableDataSet; + +import com.pb.common.matrix.*; +import com.pb.common.util.ResourceUtil; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.activityTravel.cvm.CommercialTour.TripOutputMatrixSpec; + +import java.io.*; +import java.sql.*; +import java.util.*; +import java.util.concurrent.Callable; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; + +import org.apache.log4j.Logger; + + +public class GenerateCommercialTours implements Runnable, Callable { + + private static Logger logger = Logger.getLogger(GenerateCommercialTours.class); + public static MatrixAndTAZTableCache matrixReader; + public static ResourceBundle propsResource; + + final ArrayList tripOutputMatrices = new ArrayList(); + final String segmentString; + final String segmentString1; + final String segmentString2; + + + public GenerateCommercialTours(String genColumn, boolean useTripModes) { + segmentString = genColumn; + String[] strings = genColumn.split("_"); + if (strings.length>0) segmentString1 = strings[0]; + else segmentString1 = ""; + if (strings.length>1) segmentString2 = strings[1]; + else segmentString2 = ""; + this.useTripModes = useTripModes; + } + + /** + * Generate commercial tours for Calgary EMME/2 model + * @param args is six strings, location of databank, location of coefficients file, name of coefficients file, minZone,maxZone, anyOldMatrixName + */ + public static void main(String[] args) { + + // check arguments + if (args.length != 1) { + System.out.println("Usage: java "+GenerateCommercialTours.class.getCanonicalName()+" propertiesFileName"); + throw new RuntimeException("Usage: java "+GenerateCommercialTours.class.getCanonicalName()+" propertiesFileName"); + } + try { + + try { + // basic setup of connections, variables, readers and the like + setupStaticData(args); + + //DurationModel2.durationInMinutes=ResourceUtil.getBooleanProperty(propsResource,"DurationInMinutes", false); + //TourStartTimeModel.startTimeInMinutes=ResourceUtil.getBooleanProperty(propsResource, "StartTimeInMinutes", false); + DurationModel2.durationInMinutes=silentlyCheckBooleanProperty(propsResource,"DurationInMinutes",false); + TourStartTimeModel.startTimeInMinutes=silentlyCheckBooleanProperty(propsResource, "StartTimeInMinutes",false); + + zones = zonalAttributes.getColumnAsInt(zonalAttributes.checkColumnPosition("TAZ")); + + String[] vehTypes = ResourceUtil.checkAndGetProperty(propsResource, "FirstPart").split(","); + String[] timePeriods = ResourceUtil.checkAndGetProperty(propsResource, "SecondPart").split(","); + ArrayList segmentRunners = new ArrayList(); + + if (ResourceUtil.getProperty(propsResource, "RunZones")!=null) { + logger.info("Setting up Run Zones "+ResourceUtil.getProperty(propsResource, "RunZones")); + setUpValidZones(ResourceUtil.getProperty(propsResource, "RunZones")); + } + /*if (ResourceUtil.getProperty(propsResource, "ExcludeZones")!=null) { + logger.info("Excluding Zones "+ResourceUtil.getProperty(propsResource, "ExcludeZones"); + excludeZones(ResourceUtil.getProperty(propsResource, "ExcludeZones")); + }*/ + TreeMap createdMatrices = new TreeMap(); + for (String vehType : vehTypes) { + for (String timePeriod : timePeriods) { + String genColumn = vehType + "_"+ timePeriod; + if (timePeriod.equals("")) { + genColumn = vehType; + timePeriod = null; + } + GenerateCommercialTours generator = new GenerateCommercialTours(genColumn, ResourceUtil.getBooleanProperty(propsResource, "UseTripModes")); + segmentRunners.add(generator); + generator.buildModelStructure(zones,minZone,maxZone,propsResource); + generator.generationEquation = new IndexLinearFunction(); + generator.prevGenerationEquation = new IndexLinearFunction(); + + // read the coefficients and make sure all the data is read in. + try { + readCoefficientFiles(csvInputFileReader, vehType, timePeriod); + } catch (IOException e) { + String msg = "Error reading coefficeint files "+vehType+" and "+timePeriod+ " from "+ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation"); + logger.fatal(msg,e); + throw new RuntimeException(msg,e); + } + generator.setUpCoefficients(args, coefficients); + if (coefficients2!=null) generator.setUpCoefficients(args, coefficients2); + + // TODO move this to coefficent file + generator.setUpDisutilityGetters(); + + if (ResourceUtil.getBooleanProperty(propsResource, "UseSegmentNameInGeneration",true)) { + generator.generationEquation.addCoefficient(genColumn,1); + } + + String tripLogPath = ResourceUtil.getProperty(propsResource, "TripLogPath"); + if (tripLogPath !=null) { + if (!tripLogPath.equals("")) { + String tripLogFile = tripLogPath + + "Trip_"+genColumn+".csv"; + + generator.openTripLog(new File(tripLogFile)); + } + } + generator.readMatrices(); + if (ResourceUtil.getBooleanProperty(propsResource,"ReadOutputMatrices",true)) { + generator.readOutputMatrices(matrixReader); + } else { + generator.createEmptyOutputMatrices(matrixReader.getDim2MatrixSize(),matrixReader.getDim2MatrixExternalNumbers(),createdMatrices); + } + generator.landUseTypeMatrix = matrixReader.readMatrix(generator.landUseTypeMatrixName); + + + } + } + + int nThreads = ResourceUtil.getIntegerProperty(propsResource, "nThreads",8); + + threadpool = Executors.newFixedThreadPool(nThreads); + + List> runners; + try { + runners = threadpool.invokeAll(segmentRunners); + for (Future runner : runners) { + runner.get(); + } + } catch (InterruptedException e) { + logger.fatal("Thread was interrupted",e); + throw new RuntimeException("Thread was interrupted",e); + } catch (ExecutionException e) { + logger.fatal("Exception in one segment",e); + throw new RuntimeException("Exception in one segment",e); + } + + int totalTours = 0; + int totalTrips = 0; + for (GenerateCommercialTours generator : segmentRunners) { + totalTours += generator.totalTours; + totalTrips += generator.totalTrips; + logger.info("Trips for segment "+generator.segmentString); + generator.vehicleTourTypeChoice.writeTourAndTripSummary(); + } + + + logger.info("finished generating "+totalTours+" tours and "+totalTrips+" trips"); + + MatrixWriter writer = null; + if (ResourceUtil.getBooleanProperty(propsResource, "WriteEmmeApiMatrices",false)) { + try { + writer = setUpEmmeApiWriter(); + writeAllMatrices(writer,segmentRunners); + } finally { + if (writer !=null) { + try { + ((EmmeApiMatrixWriter) writer).close(); + } catch (IOException e) { + // oh well + } + } + } + } else if (ResourceUtil.getBooleanProperty(propsResource, "WriteDirectEmmeMatrices",false)) { + //writer = new Emme2MatrixWriter(file) + //writeMatrices(writer); + logger.error("WriteDirectEmmeMatrices is no longer supported, please use WriteEmmeApiMatrices instead. Matrices not written out!"); + } else if (ResourceUtil.getProperty(propsResource, "CSVOutputFileLocation")!=null) { + writer = new CSVMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "CSVOutputFileLocation"))); + writeAllMatrices(writer, segmentRunners); + } else if (ResourceUtil.getProperty(propsResource, "TranscadCVMMatrixFile")!=null) { + writer = new TranscadMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "TranscadCVMMatrixFile"))); + writeAllMatrices(writer, segmentRunners); + } else { + writer = new CSVMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "CSVFileLocation")+File.pathSeparator+"TripMatrices.csv")); + writeAllMatrices(writer,segmentRunners); + } + + logger.info("Finished writing matrices"); + + + } catch (Throwable e) { + logger.fatal("Error in CVM program", e); + } + } finally { + if (threadpool!=null) { + threadpool.shutdown(); + } + if (matrixReader!=null) { + MatrixReader actualReader = matrixReader.getActualReader(); + if (actualReader instanceof EmmeApiMatrixReader) { + try { + ((EmmeApiMatrixReader) actualReader).close(); + } catch (IOException e) { + logger.warn("IOException trying to close EmmeApiMatrixReader",e); + } + } + } + } + } + + + + private static boolean silentlyCheckBooleanProperty( + ResourceBundle propsResource2, String name, boolean defaultVal) { + String str = ResourceUtil.getProperty(propsResource2, name); + if (str==null) return defaultVal; + if (str.equals("")) return defaultVal; + if (str.equalsIgnoreCase("True")) return true; + if (str.equalsIgnoreCase("False")) return false; + logger.error(name + "property should be boolean, it is "+name); + return defaultVal; + } + + private HashMap travelDisutilityTrackers = null; + + private void readMatrices() { + Iterator modelIterator = models.values().iterator(); + while (modelIterator.hasNext()) { + ModelUsesMatrices model = (ModelUsesMatrices) modelIterator.next(); + model.readMatrices(matrixReader); + } + generationEquation.readMatrices(matrixReader); + prevGenerationEquation.readMatrices(matrixReader); + } + + VehicleTourTypeChoice vehicleTourTypeChoice; + private final boolean useTripModes; + /** + * @return Returns the vehicleTourTypeChoice. + */ + public VehicleTourTypeChoice getVehicleTourTypeChoice() { + return vehicleTourTypeChoice; + } + + /** + * @param vehicleTourTypeChoice The vehicleTourTypeChoice to set. + */ + public void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoiceParam) { + vehicleTourTypeChoice = vehicleTourTypeChoiceParam; + } + + + + private void generateTheTours() { + vehicleTourTypeChoice=(CommercialVehicleTourTypeChoice) models.get("VehicleTourType"); + setTourStartTimeModel((TourStartTimeModel) models.get("TourStartTime")); + setElapsedTravelTimeCalculator((CommercialTravelTimeTracker) models.get("TravelTimeMatrix")); + totalTours = 0; + int oldTotalTours = 0; + totalTrips = 0; + System.out.println("Generating tours..."); + for (int z = 1;z 100) { + String msg = null; + if (totalTours - oldTotalTours == tours) + msg = "Generating "+tours+" "+segmentString+" tours in zone "+zone+" ... "; + else + msg = "Generated "+(totalTours - oldTotalTours)+" "+segmentString+"tours, including "+tours+" currently being generated in zone "+zone; + logger.info(msg); + oldTotalTours = totalTours; + } + int tripCount = 0; + for (int tour = 0; tour < tours; tour ++) { + CommercialTour t = new CommercialTour(models,this,tripLog,getNextTourNumber()); + t.setOrigin(zone); + t.sampleStartTime(); + t.sampleVehicleAndTourType(); + t.sampleStops(); + t.addTripsToMatrix(); + tripCount += t.getStopCounts()[0]; + } + if (logger.isDebugEnabled()) logger.debug(tripCount+" trips generated from "+segmentString+" tours in zone "+zone+" ... "); + totalTrips += tripCount; + } + + } + } + + private void setUpDisutilityGetters() { + if (isUseTripModes()) { + // TODO this shouldn't be hardcoded + HashMap trackers = new HashMap(); + trackers.put("L", (ChangingTravelAttributeGetter) models.get("TripModeL")); + trackers.put("I", (ChangingTravelAttributeGetter) models.get("TripModeI")); + trackers.put("M", (ChangingTravelAttributeGetter) models.get("TripModeM")); + trackers.put("H", (ChangingTravelAttributeGetter) models.get("TripModeH")); + setTravelDisutilityTrackers(trackers); + } else { + HashMap trackers = new HashMap(); + trackers.put("",((CommercialTravelTimeTracker) models.get("TravelDisutilityMatrix"))); + } + } + + static TreeSet zoneSet = null; + + private boolean isValidZone(int zone) { + if (zone maxZone) return false; + if (zoneSet==null) return true; + if (zoneSet.contains(zone)) return true; + return false; + } + + + private static void setUpValidZones(String property) { + String[] zoneIds = property.split(","); + zoneSet = new TreeSet(); + for (String zoneId : zoneIds) { + zoneSet.add(Integer.valueOf(zoneId)); + } + } + + + + private static void setupStaticData(String[] args) { + try { + propsResource = new PropertyResourceBundle(new FileInputStream(args[0])); + } catch (FileNotFoundException e2) { + logger.fatal("Can't find file "+args[0]); + throw new RuntimeException("Can't find file "+args[0], e2); + } catch (IOException e2) { + logger.fatal("Can't read file "+args[1]); + throw new RuntimeException("Can't read file "+args[0], e2); + } + csvInputFileReader = new CSVFileReader(); + csvInputFileReader.setPadNulls(true); + csvInputFileReader.setMyDirectory(ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation")); + try { + logger.info("Reading ZonalProperties"); + zonalAttributes = csvInputFileReader.readTable(ResourceUtil.getProperty(propsResource, "ZonalPropertiesFileName", "ZonalProperties.csv ")); + String file2 = ResourceUtil.getProperty(propsResource, "ZonalPropertiesFileName2"); + if (file2 !=null) { + logger.info("Reading ZonalProperties file 2"); + TableDataSet attributes2 = csvInputFileReader.readTable(file2); + appendNewDataSet(zonalAttributes, attributes2, "TAZ"); + } else { + logger.info("No ZonalPropertiesFileName2, just using one zonal properties file"); + } + } catch (IOException e1) { + String msg = "Error reading zonal properties files from "+ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation")+" this might be ok if your zonal properties are stored with your matrices"; + logger.warn(msg,e1); + } + + minZone = ResourceUtil.getIntegerProperty(propsResource, "StartZone"); + maxZone = ResourceUtil.getIntegerProperty(propsResource, "EndZone"); + + if(ResourceUtil.getProperty(propsResource, "SkimDatabase") !=null) setUpDatabaseSkims(); + else if (ResourceUtil.getProperty(propsResource, "EmmeUserInitials") != null) setUpEmmeApiSkims(); + else if (ResourceUtil.getProperty(propsResource, "TranscadSkimLocation") != null) setUpTranscadSkims(); + else if (ResourceUtil.getProperty(propsResource, "OMXSkimLocation") != null) setUpOMXSkims(); + else setUpHDF5Skims(); + + } + + private static void setUpTranscadSkims() { + matrixReader = new MatrixAndTAZTableCache( + new TranscadMatrixCollectionReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "TranscadSkimLocation"))), zonalAttributes); + } + + private static void setUpOMXSkims() { + matrixReader = new MatrixAndTAZTableCache( + new OMXMatrixCollectionReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "OMXSkimLocation"))), zonalAttributes); + } + + private static void setUpHDF5Skims() { + String skimNameString = ResourceUtil.checkAndGetProperty(propsResource, "SkimNames"); + String[] skimNames = skimNameString.split(" *, *"); + + String nodeNameString = ResourceUtil.checkAndGetProperty(propsResource, "SkimFileNodeNames"); + String[] nodeNames = nodeNameString.split(" *, *"); + matrixReader = new MatrixAndTAZTableCache( + new HDF5MatrixReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "SkimFile")), + nodeNames, + skimNames), zonalAttributes); + } + + private static void setUpEmmeApiSkims() { + String initials = ResourceUtil.checkAndGetProperty(propsResource, "EmmeUserInitials"); + String emmeBank = ResourceUtil.checkAndGetProperty(propsResource, "EmmeBank"); + String iks = ResourceUtil.getProperty(propsResource, "EmmeIKS"); + try { + if (iks==null) { + matrixReader = new MatrixAndTAZTableCache( + new EmmeApiMatrixReader(initials, emmeBank), zonalAttributes); + } else { + matrixReader = new MatrixAndTAZTableCache( + new EmmeApiMatrixReader(initials, emmeBank, iks, "", false), zonalAttributes); + } + } catch (IOException e) { + String msg = "Couldn't open the emme 2 databank "+emmeBank; + logger.fatal(msg,e); + throw new RuntimeException(msg,e); + } + } + + private static EmmeApiMatrixWriter setUpEmmeApiWriter() { + if (matrixReader.getActualReader() instanceof EmmeApiMatrixReader) { + return new EmmeApiMatrixWriter((EmmeApiMatrixReader) matrixReader.getActualReader()); + } else { + EmmeApiMatrixWriter writer; + String initials = ResourceUtil.checkAndGetProperty(propsResource, "EmmeUserInitials"); + String emmeBank = ResourceUtil.checkAndGetProperty(propsResource, "EmmeBank"); + String iks = ResourceUtil.getProperty(propsResource, "EmmeIKS"); + try { + if (iks==null) { + writer = new EmmeApiMatrixWriter(initials, emmeBank); + } else { + writer = new EmmeApiMatrixWriter(initials, emmeBank, iks, "", false); + } + } catch (IOException e) { + String msg = "Couldn't open the emme 2 databank "+emmeBank; + logger.fatal(msg,e); + throw new RuntimeException(msg,e); + } + return writer; + } + } + + + private static void setUpDatabaseSkims() { + Connection conn; + try { + conn = conn=DriverManager.getConnection( + ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabase"), + ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabaseUser"), + ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabasePassword")); + } catch (SQLException e1) { + String msg = "Can't open skim database"; + logger.fatal(msg,e1); + throw new RuntimeException(msg,e1); + } + logger.info("Reading SQL Matrices"); + matrixReader=new MatrixAndTAZTableCache(new SQLMatrixReader(conn, + ResourceUtil.checkAndGetProperty(propsResource, "SkimQuery"), + ResourceUtil.checkAndGetProperty(propsResource, "OriginQuery"), + ResourceUtil.checkAndGetProperty(propsResource, "DestinationQuery") + ), zonalAttributes); + } + + private static void readCoefficientFiles(CSVFileReader reader, String name1, String name2) + throws IOException { + coefficients = reader.readTable(name1); + if (name2 != null) { + coefficients2 = reader.readTable(name2); + } else { + logger.info("No CoefficientFileName2, just using one coefficient file"); + } + } + + /** + * Takes the columns from one dataset and appends them to another dataset based on a common integer index column. + * This uses the indexing feature in the TableDataSet and so creates a new row index on the dataset using + * the common column name. If you are relying on the row indexing feature from a different column, you'll + * need to reindex it after. + * @param datasetToAppendTo the dataset to be modified by appending the new columns + * @param newData the dataset containing the new data to be appended to the other data set + * @param commonIndexColumn the name of the common index column + */ + private static void appendNewDataSet(TableDataSet datasetToAppendTo, TableDataSet newData, String commonIndexColumn) { + int originalTazColumnNum = datasetToAppendTo.checkColumnPosition(commonIndexColumn); + datasetToAppendTo.buildIndex(originalTazColumnNum); + int tazColumn = newData.checkColumnPosition(commonIndexColumn); + for (int column = 1; column <= newData.getColumnCount(); column++) { + if (column != tazColumn) { + String columnName = newData.getColumnLabel(column); + if (datasetToAppendTo.getColumnPosition(columnName)!=-1) { + String msg = "Duplicate column name "+columnName+" in first and second dataset"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + float[] newColumnVals = new float[datasetToAppendTo.getRowCount()]; + datasetToAppendTo.appendColumn(newColumnVals, columnName); + int newColumnNumber = datasetToAppendTo.checkColumnPosition(columnName); + for (int row = 1; row <= newData.getRowCount();row++) { + datasetToAppendTo.setIndexedValueAt(((int) newData.getValueAt(row, tazColumn)),newColumnNumber,newData.getValueAt(row,column)); + } + } + } + } + + private void setUpCoefficients(String[] args, + TableDataSet coefficients) { + try { + for (int row=1;row<=coefficients.getRowCount();row++) { + String modelName = coefficients.getStringValueAt(row,"Model"); + String matrix = coefficients.getStringValueAt(row,"Matrix"); + String alternative = coefficients.getStringValueAt(row,"Alternative"); + String index1 = coefficients.getStringValueAt(row,"Index1"); + String index2 = coefficients.getStringValueAt(row,"Index2"); + double coefficient = coefficients.getValueAt(row,"Value"); + if (modelName.equals("Generation")) { + if (index1.equalsIgnoreCase("origin")) { + generationEquation.addCoefficient(matrix,coefficient); + } else if (index1.equalsIgnoreCase("moms")) { + generationEquation.addCoefficient(matrix,index2); + } else { + throw new CoefficientFormatError("generation equations must have origin or MOMS in Index1"); + } + } else if (modelName.equalsIgnoreCase("PrevGeneration")) { + if (index1.equalsIgnoreCase("origin")) { + prevGenerationEquation.addCoefficient(matrix,coefficient); + } else if (index1.equalsIgnoreCase("moms")) { + prevGenerationEquation.addCoefficient(matrix,index2); + } else throw new CoefficientFormatError("generation equations must have origin or MOMS in Index1"); + + } else if (modelName.equals("TripMatrix")) { + if (alternative.contains(":")) { + String[] vehicleTour = alternative.split(":"); + tripOutputMatrices.add(new CommercialTour.TripOutputMatrixSpec(matrix, index1, Float.valueOf(index2).floatValue(), (float) coefficient, vehicleTour[0].charAt(0),vehicleTour[1])); + } else { + tripOutputMatrices.add(new CommercialTour.TripOutputMatrixSpec(matrix, index1, Float.valueOf(index2).floatValue(), (float) coefficient, alternative.charAt(0),"")); + } + } else if (modelName.equals("LandUseTypeCode")) { + setLandUseTypeMatrixName(matrix); + } else { + ModelUsesMatrices model = (ModelUsesMatrices) models.get(modelName); + if (model == null) throw new CoefficientFormatError("Bad model type in coefficients: "+modelName); + model.addCoefficient(alternative,index1,index2,matrix,coefficient); + } + + } + } catch (CoefficientFormatError e) { + System.out.println("Coefficient format error -- you have an invalid coefficient"); + System.out.println(e.toString()); + System.out.println("Aborting..."); + throw new RuntimeException("Coefficient format error -- you have an invalid coefficient",e); + } + } + + void setLandUseTypeMatrixName(String landUseTypeMatrixName) { + this.landUseTypeMatrixName = landUseTypeMatrixName; + } + + + /** + * Method buildModelStructure. + * @param propsResource2 + */ + private void buildModelStructure(int[] zoneNums, int lowNumber, int highNumber, ResourceBundle propsResource) { + models.put("VehicleTourType",new CommercialVehicleTourTypeChoice( + ResourceUtil.getProperty(propsResource,"VehicleTourTypes", + "LS,LG,LO,MS,MG,MO,IS,IG,IO,HS,HG,HO").split(","), + ResourceUtil.getBooleanProperty(propsResource,"NestingVTTChoice",true))); + + // don't use these anymore because tNCVehicle and tour type are joint in the same alternative. + // models.put("LTour", new TourTypeChoice()); + // models.put("MTour", new TourTypeChoice()); + // models.put("HTour", new TourTypeChoice()); + String stopTypeModelString = ResourceUtil.getProperty(propsResource, "StopTypeModels", + "LSStopType, LGStopType, LOStopType, MSStopType, MGStopType, MOStopType, ISStopType, IGStopType, IOStopType, HSStopType, HGStopType, HOStopType"); + String[] stopTypeModels = stopTypeModelString.split(","); + for (String stopTypeModel : stopTypeModels) { + models.put(stopTypeModel.trim(), new CommercialNextStopPurposeChoice(stopTypeModel.trim().charAt(1))); + } + + String[] stopLocationModels = ResourceUtil.getProperty(propsResource, "StopLocationModels", + "LSSStopLocation, LSOStopLocation, LGGStopLocation,LGOStopLocation,LOOStopLocation,MSSStopLocation,MSOStopLocation,MGGStopLocation,MGOStopLocation,MGOStopLocation,MOOStopLocation,ISSStopLocation,ISOStopLocation,IGGStopLocation,IGOStopLocation,IOOStopLocation,HSSStopLocation,HSOStopLocation,HGGStopLocation,HGOStopLocation,HOOStopLocation") + .split(","); + for (String stopLocationModel : stopLocationModels) { + models.put(stopLocationModel.trim(), new CommercialNextStopChoice(zoneNums, lowNumber-1, highNumber+1, stopLocationModel.trim())); + } + + String[] durationModels = ResourceUtil.getProperty(propsResource, "DurationModels", + "LODuration,LSDuration,LGDuration,MODuration,MSDuration,MGDuration,IODuration,ISDuration,IGDuration,HODuration,HSDuration,HGDuration") + .split(","); + for (String durationModel : durationModels) { + models.put(durationModel.trim(), new DurationModel2()); + } + + //TODO here is the place where we parameterize the trip mode choice + if (isUseTripModes()) { + /*String[] tripModeChoiceModels = ResourceUtil.getProperty(propsResource, "TripModeL,TripModelM,TripModeI,TripModeH") + .split(","); + for (String tripModeChoiceModel : tripModeChoiceModels) { + models.put(tripModeChoiceModel.trim(), new CommercialVehicleTripModeChoice(tripModeChoiceModels)); + }*/ + models.put("TripModeL", new CommercialVehicleTripModeChoice(new String[] {"L:T", "L:NT"})); + models.put("TripModeM", new CommercialVehicleTripModeChoice(new String[] {"M:T", "M:NT"})); + models.put("TripModeI", new CommercialVehicleTripModeChoice(new String[] {"I:T", "I:NT"})); + models.put("TripModeH", new CommercialVehicleTripModeChoice(new String[] {"H:T", "H:NT"})); + + } + + models.put("TourStartTime", new TourStartTimeModel()); + models.put("TravelTimeMatrix", new CommercialTravelTimeTracker()); + models.put("TravelDisutilityMatrix", new CommercialTravelTimeTracker()); + + ModelUsesMatrices dummyModel = new ModelUsesMatrices() { + public void readMatrices(MatrixCacheReader matrixReader) { + // nothing + } + public void addCoefficient(String alternative, String index1, + String index2, String matrixName, double coefficient) + throws CoefficientFormatError { + // nothing + } + public void init() { + // nothing + } + + }; + + // dummy models, placeholders for coefficients handled elsewhere (in python) + models.put("ShipNoShip", dummyModel); + models.put("GenPerEmployee", dummyModel); + models.put("TourTOD", dummyModel); + + } + + final Hashtable models = new Hashtable(); + private static int minZone; + private static int maxZone; + static TableDataSet coefficients; + static TableDataSet coefficients2; + static TableDataSet zonalAttributes; + private int totalTours; + private int totalTrips; + private IndexLinearFunction generationEquation; + private IndexLinearFunction prevGenerationEquation; + private static int globalTourNumber = 0; + private static int[] zones; + private static CSVFileReader csvInputFileReader; + private PrintWriter tripLog; + + + public static int getNextTourNumber() { + synchronized(GenerateCommercialTours.class){ + return globalTourNumber ++; + } + } + + void openTripLog(File tripLogFile) { + try { + logger.info("Opening log file "+tripLogFile); + tripLog = new PrintWriter(new FileWriter(tripLogFile)); + tripLog.println("Model,SerialNo,Person,Trip,Tour,HomeZone,ActorType,OPurp,DPurp,I,J,TripTime,Mode,StartTime,EndTime,StopDuration,TourType,OriginalTimePeriod,TripMode,TollAvailable"); + } catch (IOException e) { + logger.fatal("Can't open trip log file "+tripLogFile,e); + throw new RuntimeException("Can't open trip log file "+tripLogFile, e); + } + } + + void closeTripLog() { + if (tripLog !=null) { + tripLog.close(); + logger.info("Closing trip log file for "+segmentString); + } + } + + + String landUseTypeMatrixName; + Matrix landUseTypeMatrix; + private TourStartTimeModel tourStartTimeModel; + private ChangingTravelAttributeGetter elapsedTravelTimeCalculator; + private static ExecutorService threadpool; + + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readOutputMatrices(MatrixCacheReader matrixReader) { + for (int i =0; i createdOnes) { + for (TripOutputMatrixSpec s : tripOutputMatrices) { + if (s.write) { + Matrix m = createdOnes.get(s.name); + if (m==null) { + s.createMatrix(size, externalNumbers); + createdOnes.put(s.name, s.matrix); + } else { + s.matrix = m; + } + } + } + } + + /** + * Method writeMatrices. + * @param matrixWriter + */ + void writeMatrices(MatrixWriter matrixWriter) { + ArrayList names = new ArrayList(); + ArrayList m = new ArrayList(); + for (int i =0; i0) { + matrixWriter.writeMatrices(names.toArray(new String[names.size()]), m.toArray(new Matrix[m.size()])); + } + } + + static void writeAllMatrices(MatrixWriter matrixWriter, ArrayList segmentRunners ) { + ArrayList names = new ArrayList(); + ArrayList m = new ArrayList(); + for (GenerateCommercialTours segment : segmentRunners) { + ArrayList tripSpecs = segment.tripOutputMatrices; + for (int i =0; i0) { + StringBuffer msg = new StringBuffer("Writing out matrices "); + for (String name : names) msg.append(","+name); + logger.info(msg); + matrixWriter.writeMatrices(names.toArray(new String[names.size()]), m.toArray(new Matrix[m.size()])); + } + } + + + + TripOutputMatrixSpec getTripOutputMatrixSpec(char vehicleType, String tripMode, float time) { + while (time>=24.00) time-=24.00; + for (int i =0; i=s.startTime && time < s.endTime && s.vehicleType==vehicleType && s.tripMode.equals(tripMode)) return s; + } + return null; + } + + @Override + public void run() { + logger.info("Starting segment "+segmentString); + generateTheTours(); + closeTripLog(); + System.out.println("DonE segement "+segmentString +"!"); + logger.info("DonE segement "+segmentString +"!"); + if (tripLog!=null) tripLog.close(); + } + + private void setElapsedTravelTimeCalculator( + // TODO should use trip modes like getTravelDisutilityTracker does + ChangingTravelAttributeGetter elapsedTravelTimeCalculator) { + this.elapsedTravelTimeCalculator = elapsedTravelTimeCalculator; + } + + public ChangingTravelAttributeGetter getElapsedTravelTimeCalculator() { + return elapsedTravelTimeCalculator; + } + + private void setTourStartTimeModel(TourStartTimeModel tourStartTimeModel) { + this.tourStartTimeModel = tourStartTimeModel; + } + + public TourStartTimeModel getTourStartTimeModel() { + return tourStartTimeModel; + } + + @Override + public Object call() throws Exception { + run(); + return null; + } + + ChangingTravelAttributeGetter getTravelDisutilityTracker() { + if (!isUseTripModes()) + return travelDisutilityTrackers.get(""); + else { + String msg = "Trip mode is being used, no defualt travel disutility tracker"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + } + + void setTravelDisutilityTrackers(HashMap trackers) { + this.travelDisutilityTrackers = trackers; + } + + public ChangingTravelAttributeGetter getTravelDisutilityTracker( + String vehicleCode) { + return travelDisutilityTrackers.get(vehicleCode); + + } + + public boolean isUseTripModes() { + return useTripModes; + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java new file mode 100644 index 0000000..0dfc727 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java @@ -0,0 +1,115 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class TourStartTimeModel implements ModelUsesMatrices, RealNumberDistribution { + + /** returns tour start time in hours since midnight + */ + static boolean startTimeInMinutes = false; + + double periodStart; // base time for the TourStartTimeModel + int functionalForm; // power =1, cubic=2 or exponential =3 + double a; + double b; + double c; + double d; + double e; + double f; + + + public double sampleValue() { + double x = Math.random(); + double y=0; + switch (functionalForm) { + case 1: + y = a * Math.pow(x,b)+c*Math.pow(x,d) + e * x + f; + break; + case 2: + y = a+b*x+c*x*x+d*x*x*x; + break; + case 3: + y = c*Math.exp(a*x+b)+d; + break; + } + if (startTimeInMinutes) y = y/60; + if (y<0) y =0; + if (y>24) y = 24; + y += periodStart; + return y; + } + + + /** + * Method addCoefficient. + * @param alternative + * @param index1 + * @param index2 + * @param matrix + * @param coefficient + */ + public void addCoefficient ( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError { + if (index1.equals("a")) a = coefficient; + else if(index1.equals("b")) b = coefficient; + else if(index1.equals("c")) c = coefficient; + else if(index1.equals("d")) d = coefficient; + else if(index1.equals("e")) e = coefficient; + else if(index1.equals("f")) f = coefficient; + else if(index1.equals("functionForm")) { + if (index2.equals("power")) functionalForm =1; + else if (index2.equals("cubic")) functionalForm = 2; + else if (index2.equals("exponential")) functionalForm = 3; + else throw new CoefficientFormatError("functionalForm for tour start model must have index2 as \"power\", \"cubic\" or \"exponential\""); + } + else if (index1.equals("periodStart")) periodStart=coefficient; + else throw new CoefficientFormatError("Tour start time model model coefficients must have index1 as a,b,c,d,e,f, periodStart or functionalForm"); + } + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader matrixReader) {} + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateCommercialTours.matrixReader); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java new file mode 100644 index 0000000..25df5d7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java @@ -0,0 +1,16 @@ +package org.sandag.cvm.activityTravel.cvm; + +import org.sandag.cvm.common.model.LogitModel; + +public class VehicleTourTypeNest extends AlogitLogitModelNest { + final String myCode; + + public VehicleTourTypeNest(String code) { + myCode = code; + } + + public String getCode() { + return myCode; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java new file mode 100644 index 0000000..2c97491 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java @@ -0,0 +1,292 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.calgary.weekend; + +import com.pb.common.matrix.*; +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.*; + +import java.io.*; +import java.sql.*; +import java.util.*; +import org.sandag.cvm.common.datafile.*; +import org.apache.log4j.Logger; + +public class GenerateWeekendTours { + + private static Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); + public static MatrixCacheReader matrixReader; + public static TableDataSetCollection inputData; + static Properties props = new Properties(); + + + /** + * Generate commercial tours for Calgary EMME/2 model + * @param args is six strings, location of databank, location of coefficients file, name of coefficients file, minZone,maxZone, anyOldMatrixName + */ + public static void main(String[] args) { + + FileInputStream fin = null; + try { + fin = new FileInputStream(args[0]); + } catch (FileNotFoundException e) { + System.out.println("error: be sure to put the location of the properties file on the command line"); + e.printStackTrace(); + System.exit(-1); + } + try { + props.load(fin); + } catch (IOException e) { + System.out.println("Error reading properties file "+args[0]); + System.exit(-1); + } + + // get emme/2 matrix + try { + matrixReader = new MatrixCacheReader(new Emme2MatrixReader(new File(props.getProperty("databank")))); + } catch (Exception e) { + System.out.println("Error opening emme2 databank \""+props.getProperty("databank")+"\""); + System.out.println(e); + e.printStackTrace(); + } + Matrix anyOldMatrix = matrixReader.readMatrix(props.getProperty("anyOldMatrixName")); + minZone = Integer.valueOf(props.getProperty("minZone")).intValue(); + maxZone = Integer.valueOf(props.getProperty("maxZone")).intValue(); + //int[] zones = anyOldMatrix.getExternalNumbers(); + buildModelStructure(matrixReader,anyOldMatrix,minZone,maxZone); + + // These next lines are just for testing in the case that you don't have an emme2 databank to load +// int[] zones = new int[10]; +// int minZone = 0; +// int maxZone = 10; +// buildModelStructure(null,minZone,maxZone); + + String inputLocation = props.getProperty("inputLocation"); + CSVFileReader inputDataReader = new CSVFileReader(); + inputDataReader.setMyDirectory(inputLocation); + // no output location for now + inputData = new TableDataSetCollection(inputDataReader, null); + + + TableDataSet coefficients = inputData.getTableDataSet(props.getProperty("coefficientsTable")); + + + try { + for (int cn= 1; cn <= coefficients.getRowCount(); cn++) { //cn is "coefficientNumber" + String modelName = coefficients.getStringValueAt(cn, "Model"); + String matrix = coefficients.getStringValueAt(cn,"Matrix"); + String alternative = coefficients.getStringValueAt(cn,"Alternative"); + String index1 = coefficients.getStringValueAt(cn,"Index1"); + String index2 = coefficients.getStringValueAt(cn,"Index2"); + double coefficient = coefficients.getValueAt(cn,"Value"); + ModelWithCoefficients model = (ModelWithCoefficients) models.get(modelName); + if (model == null) throw new CoefficientFormatError("Bad model type in coefficients: "+modelName); + model.addCoefficient(alternative,index1,index2,matrix,coefficient); + + } + } catch (CoefficientFormatError e) { + System.out.println("Coefficient format error -- you have an invalid coefficient"); + System.out.println(e.toString()); + e.printStackTrace(); + System.out.println("Aborting..."); + System.exit(-1); + } + + Iterator modelIterator = models.values().iterator(); + while (modelIterator.hasNext()) { + ModelWithCoefficients model = (ModelWithCoefficients) modelIterator.next(); + model.init(); + } + +// generationEquation.readMatrices(matrixReader); +// WeekendTour.readMatrices(matrixReader); +// +// CommercialTour.tourTypeChoiceModel=(CommercialVehicleTourTypeChoice) models.get("VehicleTourType"); +// CommercialTour.setTourStartTimeModel((TourStartTimeModel) models.get("TourStartTime")); +// CommercialTour.setElapsedTravelTimeCalculator((WeekendTravelTimeTracker) models.get("TravelTimeMatrix")); +// CommercialTour.travelDisutilityTracker = (WeekendTravelTimeTracker) models.get("TravelDisutilityMatrix"); + + //TODO the household dataset is probably too big to load in the whole thing, so instead step through the file one record at a time + TableDataSet households = inputData.getTableDataSet(props.getProperty("householdsTable")); + TableDataSet householdDetails = inputData.getTableDataSet(props.getProperty("householdDetailsTable")); + householdDetails.buildIndex(householdDetails.checkColumnPosition("hh_ID")); + // TODO read in person file + + int totalTours = 0; + int totalTrips = 0; + System.out.println("Generating tours..."); + for (int hhnum = 1;hhnum<=households.getRowCount();hhnum++) { + if (hhnum %10 == 0) System.out.print(hhnum+" "); + if (hhnum %200 == 0) System.out.println(); + + + WeekendHousehold household = new WeekendHousehold(households, hhnum, householdDetails); + // TODO should add people by reading them from a dataset + household.addPeople(); + + int zone = household.getHomeZone(); + + // TODO check if to generate tours from externals? Probably not. + if (zone>=minZone && zone <= maxZone) { + household.resetCurrentTime(); + WeekendTour tour = household.sampleNextWeekendTour(); + while (tour != null) { + totalTours++; + // TODO add the trips to the matrix; + // t.addTripsToMatrix(); + // TODO write the tour to the tour database + totalTrips += tour.getStopCounts()[0]; + tour = household.sampleNextWeekendTour(); + } + } + + } + System.out.println(); + + + System.out.println("finished generating "+totalTours+" tours and "+totalTrips+" trips , now writing trip matrices out to emme2 databank"); +// WeekendTour.tourTypeChoiceModel.writeTourAndTripSummary(); + Emme2MatrixWriter matrixWriter = new Emme2MatrixWriter(new File(props.getProperty("databank"))); + // TODO write out all of the trip matrices back into the emme2 databank +// WeekendTour.writeMatrices(matrixWriter); + System.out.println("done!"); + } + + /** + * Method buildModelStructure. + */ + private static void buildModelStructure(MatrixCacheReader matrixReader2, Matrix anyOldMatrix, int lowNumber, int highNumber) { + + TourInTimeBand titb = new TourInTimeBand(); + WeekendTour.setTourInTimeBand(titb); + models.put("tourInBand",titb); + NextWeekendTourStartTime ttnt = new NextWeekendTourStartTime(); + WeekendTour.setTourStartTimeModel(ttnt); + models.put("tourStart", ttnt); + WeekendTourTypeChoice wttc = new WeekendTourTypeChoice(); + wttc.setMatrixReader(matrixReader2); + models.put("tourType", wttc); + WeekendTour.tourTypeChoiceModel= wttc; + models.put("workPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); + models.put("schoolPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("relCivicPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("exercisePrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("outOfTownPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("workIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); + models.put("schoolIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("relCivicIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("exerciseIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("outOfTownIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("workReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); + models.put("schoolReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("relCivicReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("exerciseReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("outOfTownReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("SELSEReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("chaufReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); + models.put("workStopType", new WeekendStopPurposeChoice("workStopType")); + models.put("schoolStopType", new WeekendStopPurposeChoice("schoolStopType")); + models.put("exerciseStopType", new WeekendStopPurposeChoice("exerciseStopType")); + models.put("relCivicStopType", new WeekendStopPurposeChoice("relCivicStopType")); + models.put("outOfTownStopType", new WeekendStopPurposeChoice("outOfTownStopType")); + models.put("chaufStopType", new WeekendStopPurposeChoice("chaufStopType")); + models.put("SELSEStopType", new WeekendStopPurposeChoice("SELSEStopType")); + models.put("workDuration", new DurationModel()); + models.put("shopDuration", new DurationModel()); + models.put("relCivicDuration", new DurationModel()); + models.put("eatDuration", new DurationModel()); + models.put("entLeisureDuration", new DurationModel()); + models.put("socialDuration", new DurationModel()); + models.put("exerciseDuration", new DurationModel()); + models.put("schoolDuration", new DurationModel()); + models.put("outOfTownDuration", new DurationModel()); + models.put("pickUpDuration", new DurationModel()); + models.put("dropOffDuration", new DurationModel()); + + // models.put("VehicleTourType",new CommercialVehicleTourTypeChoice()); +//// models.put("LTour", new TourTypeChoice()); +//// models.put("MTour", new TourTypeChoice()); +//// models.put("HTour", new TourTypeChoice()); +// models.put("LSStopType", new CommercialNextStopPurposeChoice('S')); +// models.put("LGStopType", new CommercialNextStopPurposeChoice('G')); +// models.put("LOStopType", new CommercialNextStopPurposeChoice('O')); +// models.put("MSStopType", new CommercialNextStopPurposeChoice('S')); +// models.put("MGStopType", new CommercialNextStopPurposeChoice('G')); +// models.put("MOStopType", new CommercialNextStopPurposeChoice('O')); +// models.put("HSStopType", new CommercialNextStopPurposeChoice('S')); +// models.put("HGStopType", new CommercialNextStopPurposeChoice('G')); +// models.put("HOStopType", new CommercialNextStopPurposeChoice('O')); +// models.put("LSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("LSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("LGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("LGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("LOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("MSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("MSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("MGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("MGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("MOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("HSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("HSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("HGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("HGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("HOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); +// models.put("LODuration", new DurationModel()); +// models.put("LSDuration", new DurationModel()); +// models.put("LGDuration", new DurationModel()); +// models.put("MODuration", new DurationModel()); +// models.put("MSDuration", new DurationModel()); +// models.put("MGDuration", new DurationModel()); +// models.put("HODuration", new DurationModel()); +// models.put("HSDuration", new DurationModel()); +// models.put("HGDuration", new DurationModel()); + + WeekendTravelTimeTracker timeTracker = new WeekendTravelTimeTracker(); + models.put("TravelTimeMatrix",timeTracker); + WeekendTour.setElapsedTravelTimeCalculator(timeTracker); + + WeekendTravelTimeTracker disutilTracker = new WeekendTravelTimeTracker(); + models.put("TravelDisutilityMatrix", disutilTracker); + WeekendTour.setTravelDisutilityTracker(disutilTracker); + } + + + static final Hashtable models = new Hashtable(); + private static int minZone; + private static int maxZone; + +// public static ResultSet readCoefficients(String parameterFileLocation, String tableName) { +// ResultSet results = null; +// try { +// Class.forName("org.relique.jdbc.csv.CsvDriver"); +// Connection conn = DriverManager.getConnection("jdbc:relique:csv:" + parameterFileLocation); +// Statement stmt = conn.createStatement(); +// results = stmt.executeQuery("SELECT * FROM "+tableName); +// return results; +// } +// catch(Exception e) +// { +// System.out.println("JDBC Error connecting to "+parameterFileLocation+" "+ e); +// e.printStackTrace(); +// System.exit(-1); +// } +// return results; +// } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java new file mode 100644 index 0000000..95d72e7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java @@ -0,0 +1,101 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.calgary.weekend; + +import org.sandag.cvm.activityTravel.*; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public class NextWeekendTourStartTime + implements ModelWithCoefficients, RealNumberDistribution { + + double startTime = 0; + double endTime = 1; + private WeekendHousehold myHousehold; + + /** + * + */ + public NextWeekendTourStartTime() { + super(); + // TODO Auto-generated constructor stub + } + + /* (non-Javadoc) + * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#addCoefficient(java.lang.String, java.lang.String, java.lang.String, java.lang.String, double) + */ + public void addCoefficient( + String alternative, + String index1, + String index2, + String matrixName, + double coefficient) + throws CoefficientFormatError { + if (alternative.equalsIgnoreCase("EndTime")) { + endTime = coefficient; + } else if (alternative.equalsIgnoreCase("StartTime")) { + startTime = coefficient; + } else { + throw new CoefficientFormatError("Valid coefficients for tour start time model are \"StartTime\" and \"EndTime\""); + } + + } + + /* (non-Javadoc) + * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#init() + */ + public void init() { + // Nothing to do here. + + } + + public void shiftTime(double shift) { + startTime+=shift; + endTime +=shift; + } + + public void resetStart() { + endTime = endTime-startTime; + startTime = 0; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.common.model.SingleValueRealDistribution#sampleValue() + */ + public double sampleValue() { + // TODO use a different distribution? + double timeToNextTour = Math.random()*(endTime-startTime); + return timeToNextTour+startTime; + } + + public void setMyHousehold(WeekendHousehold myHousehold) { + this.myHousehold = myHousehold; + } + + public WeekendHousehold getMyHousehold() { + return myHousehold; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java new file mode 100644 index 0000000..09a2f9a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java @@ -0,0 +1,185 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.calgary.weekend; + +import java.util.ArrayList; + +import org.sandag.cvm.activityTravel.AlternativeUsesMatrices; +import org.sandag.cvm.activityTravel.CoefficientFormatError; +import org.sandag.cvm.activityTravel.ModelWithCoefficients; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.Alternative; +import org.sandag.cvm.common.model.LogitModel; +import org.sandag.cvm.common.model.NoAlternativeAvailable; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +/** + * @author jabraham + * + * Controls the time bands, increments through them, and has the model as + * to whether another tour occurs in the time band + */ +public class TourInTimeBand extends LogitModel implements ModelWithCoefficients { + + ArrayList bandStarts = new ArrayList(); + double dayEnd = 24*60; + final Alternative noTour; + final Alternative makeATour; + int currentBand = 0; + double makeATourConstant = 0; + WeekendHousehold currentHousehold=null; + + double getCurrentBandStart() { + if (currentBand >= bandStarts.size()) { + return dayEnd; + } else { + return ((Double)bandStarts.get(currentBand)).doubleValue(); + } + } + + double getCurrentBandEnd() { + if (currentBand +1 >= bandStarts.size()) { + return dayEnd; + } else { + return ((Double)bandStarts.get(currentBand+1)).doubleValue(); + } + } + + /* (non-Javadoc) + * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#addCoefficient(java.lang.String, java.lang.String, java.lang.String, java.lang.String, double) + */ + public void addCoefficient(String alternative, String index1, String index2, String matrixName, double coefficient) throws CoefficientFormatError { + Double lastBandStart = null; + if (bandStarts.size() == 0) lastBandStart = new Double(Double.NEGATIVE_INFINITY); + else lastBandStart =(Double)(bandStarts.get(bandStarts.size()-1)); + if (alternative.equalsIgnoreCase("bandStart")) { + if (coefficient <= lastBandStart.doubleValue()) { + throw new CoefficientFormatError("Start bands for tour time band model need to be specified in increasing order"); + } + bandStarts.add(new Double(coefficient)); + } else if (alternative.equalsIgnoreCase("dayEnd")) { + if (coefficient <= lastBandStart.doubleValue()) { + throw new CoefficientFormatError("End of day for tour time band model need to be greater than last time band"); + } + dayEnd = coefficient; + } else if (alternative.equalsIgnoreCase("constant")) { + makeATourConstant = coefficient; + } else { + throw new CoefficientFormatError("Bad coefficient for tour band model "+alternative+" "+index1+" "+index2+" "+matrixName); + } + + // TODO other coefficients + + } + + /* (non-Javadoc) + * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#init() + */ + public void init() { + // TODO read matrices if necessary + } + + public TourInTimeBand() { + super(); + noTour = new AlternativeUsesMatrices() { + public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { + throw new CoefficientFormatError("The NoTour alternative has no coefficients and baseline utility zero"); + } + + public void readMatrices(MatrixCacheReader mr) { + // nothing to do + } + + public String getCode() { + return "noTour"; + } + + public double getUtility() { + return 0; + } + }; + makeATour = new AlternativeUsesMatrices() { + public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { + throw new CoefficientFormatError("The NoTour alternative has no coefficients and baseline utility zero"); + } + + public void readMatrices(MatrixCacheReader mr) { + // read matrices if we need to. + } + + public String getCode() { + return "makeATour"; + } + + public double getUtility() { + if (getCurrentBandEnd()<=getCurrentBandStart()){ + return Double.NEGATIVE_INFINITY; + } + return makeATourConstant + Math.log(getCurrentBandEnd()-getCurrentBandStart()) + Math.log(currentHousehold.countPeopleAtHome()); + // TODO add in other parameters related to number of people at home and household size etc. + } + }; + addAlternative(noTour); + addAlternative(makeATour); + } + + public boolean beyondLastBand() { + if (currentBand>=bandStarts.size()) return true; + return false; + } + + /** + * @return + */ + public boolean tourStartsInBand() { + Alternative chosen=null; + try { + chosen = this.monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + e.printStackTrace(); + throw new RuntimeException("Error in TourStartsInBand module",e); + } + return chosen == makeATour; + } + + /** + * @param currentTime + */ + public void setBandBasedOnTime(double currentTime) { + if (bandStarts.size() == 0) { + currentBand = 0; + return; + } + if (currentTime > dayEnd) { + currentBand = bandStarts.size(); + return; + } + if (currentTime < ((Double)bandStarts.get(0)).doubleValue()) { + throw new RuntimeException("Current household time isn't in any time band"); + } + for (currentBand = bandStarts.size()-1;currentBand >=0;currentBand--) { + if (currentTime >= ((Double)bandStarts.get(currentBand)).doubleValue()) { + return; + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java new file mode 100644 index 0000000..d27d19f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java @@ -0,0 +1,366 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.calgary.weekend; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +import org.sandag.cvm.activityTravel.HouseholdInterface; +import org.sandag.cvm.activityTravel.PersonInterface; +import org.sandag.cvm.common.datafile.TableDataSet; + + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public class WeekendHousehold implements HouseholdInterface { + + ArrayList myPeople =null; + private static int incrementalId = 0; + int id; + private boolean incomeValid; + private float annualIncome; + private int numberOfVehicles; + private int adultOtherCount; // ao + private int adultWorkerNeedingCarCount; // awnc + private int adultWorkerNotNeedingCarCount; // awnnc + private int studentKto9Count; //kejs + private int postSecondaryStudentCount; //pss + private int seniorCount; // Sen + private int student10to12Count; //SHS + private int youthOtherCount; //YO + private double currentTime; + private boolean homelessHousehold = true; + private int homeZone = 0; + + static NextWeekendTourStartTime myTourStartTimeModel = null; + + public WeekendHousehold(TableDataSet populationHouseholds, int rowNum, TableDataSet sampleHouseholds) { + homeZone = (int) populationHouseholds.getValueAt(rowNum,"Zone"); + homelessHousehold = false; + int hhid = (int) populationHouseholds.getValueAt(rowNum,"HHID"); + int sampleRowNum = sampleHouseholds.getIndexedRowNumber(hhid); + fillInDataFromDataSet(sampleHouseholds, sampleRowNum); + id = incrementalId++; + } + + + + public WeekendHousehold(TableDataSet tds, int rowNum) { + // For integration with household synthesis, need to get household attributes + // and people information from another file. + id=(int) tds.getValueAt(rowNum,"hh_ID"); + fillInDataFromDataSet(tds,rowNum); + } + + void fillInDataFromDataSet(TableDataSet tds, int rowNum) { + //TODO check to see if value targets in synthesis deal with missing values properly + annualIncome = tds.getValueAt(rowNum,"Value"); + if (annualIncome < -100000) { + annualIncome = 0; + incomeValid = false; + } else { + incomeValid = true; + } + numberOfVehicles = (int) tds.getValueAt(rowNum,"CountOfveh_id"); + adultOtherCount = (int) tds.getValueAt(rowNum,"AO"); + adultWorkerNeedingCarCount = (int) tds.getValueAt(rowNum,"AWNC"); + adultWorkerNotNeedingCarCount = (int) tds.getValueAt(rowNum,"AWNNC"); + studentKto9Count = (int) tds.getValueAt(rowNum,"KEJS"); + postSecondaryStudentCount = (int) tds.getValueAt(rowNum,"PSS"); + seniorCount = (int)tds.getValueAt(rowNum,"Sen"); + student10to12Count = (int) tds.getValueAt(rowNum,"SHS"); + youthOtherCount = (int)tds.getValueAt(rowNum,"YO"); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.HouseholdInterface#getId() + */ + public int getId() { + return id; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.HouseholdInterface#getPersons() + */ + public Collection getPersons() { + if (myPeople == null) { + makeMyPeople(); + } + return myPeople; + } + + private void makeMyPeople() { + // TODO add in other attributes of people as necessary + // TODO get RID of this method, as people should be read from the database. use AddPeople instead + myPeople = new ArrayList(); + int peopleToMake = getPersonCount(); + for (int p=0;p person.returnTime) person.atHome = true; + } + } + } + + + + double getCurrentTime() { + return currentTime; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java new file mode 100644 index 0000000..2b4ee77 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java @@ -0,0 +1,105 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.calgary.weekend; + +import org.sandag.cvm.activityTravel.HouseholdInterface; +import org.sandag.cvm.activityTravel.PersonInterface; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public class WeekendPerson implements PersonInterface { + + WeekendHousehold myHousehold; + /** + * @param household + */ + public WeekendPerson(WeekendHousehold household) { + myHousehold = household; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.PersonInterface#getMyHousehold() + */ + public HouseholdInterface getMyHousehold() { + return myHousehold; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.PersonInterface#getAge() + */ + public int getAge() { + // TODO add age attribute + throw new RuntimeException("getAge() is not yet implemented for WeekendPerson"); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.PersonInterface#getPersonID() + */ + public long getPersonID() { + // TODO Auto-generated method stub + throw new RuntimeException("getPersonID() is not yet implemented for WeekendPerson"); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.PersonInterface#isFemale() + */ + public boolean isFemale() { + // TODO Auto-generated method stub + throw new RuntimeException("isFemale() is not yet implemented for WeekendPerson"); + } + + public boolean atHome= true; + + public double returnTime=0; + + /** + * @return Returns the atHome. + */ + public boolean isAtHome() { + return atHome; + } + + /** + * @param atHome The atHome to set. + */ + public void setAtHome(boolean atHome) { + this.atHome = atHome; + } + + /** + * @return Returns the returnTime. + */ + public double getReturnTime() { + return returnTime; + } + + /** + * @param returnTime The returnTime to set. + */ + public void setReturnTime(double returnTime) { + this.returnTime = returnTime; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java new file mode 100644 index 0000000..1b6ba0d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java @@ -0,0 +1,54 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.calgary.weekend; + +import org.sandag.cvm.activityTravel.StopAlternative; +import org.sandag.cvm.activityTravel.StopChoice; +import com.pb.common.matrix.Matrix; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class WeekendStopChoice extends StopChoice { + + public WeekendStopChoice(Matrix r, int minZone, int maxZone) { + super(); + int[] zoneNums = r.getExternalNumbers(); + for (int z = 1;z= minZone && theNumber <= maxZone) { + this.addAlternative(new StopAlternative(this, zoneNums[z])); + } + } + } + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateWeekendTours.matrixReader); + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java new file mode 100644 index 0000000..c6c238c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java @@ -0,0 +1,217 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + +package org.sandag.cvm.calgary.weekend; + +import org.sandag.cvm.activityTravel.*; +import org.sandag.cvm.common.emme2.IndexLinearFunction; +import org.sandag.cvm.common.emme2.MatrixCacheReader; +import org.sandag.cvm.common.model.LogitModel; +import org.sandag.cvm.common.model.NoAlternativeAvailable; +import com.pb.common.matrix.Emme2MatrixReader; +import com.pb.common.matrix.MatrixReader; + +import java.util.*; + +import org.apache.log4j.Logger; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2005 + */ +public class WeekendStopPurposeChoice extends LogitModel implements ModelUsesMatrices, TourNextStopPurposeChoice { + + + private static Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); + Tour myTour; + static final int RETURN=WeekendTour.convertPurposeToInt("return"); + + public class NextStopPurpose implements AlternativeUsesMatrices { + public NextStopPurpose(int stopType) { + this.stopType = stopType; + } + + double[] transitionConstants = new double[14]; + double[] stopCountCoefficients = new double[14]; + double constant = 0; + + final int stopType; + IndexLinearFunction previousStopUtility = new IndexLinearFunction(); + IndexLinearFunction originUtility = new IndexLinearFunction(); + IndexLinearFunction returnToOriginUtility = new IndexLinearFunction(); + double timeToOriginCoefficient = 0; + double disutilityToOriginCoefficient = 0; + double totalTravelTimeCoefficient = 0; + double totalTripTimeCoefficient = 0; + public double getUtility() { + double utility = previousStopUtility.calcForIndex(myTour.getCurrentLocation(),1); + utility += originUtility.calcForIndex(getMyTour().getOriginZone(),1); + int previousStopType= myTour.getLastStopType(); + // TODO make sure the stop count code works properly for weekend model; default implementation just counts total stops, not by type + int[] stopCounts = getMyTour().getStopCounts(); + // can't return home on first stop + if (stopCounts[0]==0 && stopType==RETURN) utility += Double.NEGATIVE_INFINITY; + for (int type =0;type < stopCountCoefficients.length;type++) { + utility += stopCountCoefficients[type]*Math.log(stopCounts[type]+1); + } + double returnHomeUtility = returnToOriginUtility.calcForIndex(myTour.getCurrentLocation(),getMyTour().getOriginZone()); + + // make people return home more -- Doug and Kevin Hack of Jan 5th + //if (myTour.getTotalElapsedTime()>240.0) returnHomeUtility *=3; + utility += returnHomeUtility; + + utility += totalTravelTimeCoefficient*getMyTour().getTotalTravelTimeMinutes(); + utility += totalTripTimeCoefficient*getMyTour().getTotalElapsedTimeHrs(); + utility += timeToOriginCoefficient*myTour.getElapsedTravelTimeCalculator().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); + utility += disutilityToOriginCoefficient*getMyTour().getTravelDisutilityTracker().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); + utility += constant; + return utility; + } + + /** + * Method addParameter. + * @param matrix + * @param coefficient + */ + public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { + if(index1.equals("origin")) { + originUtility.addCoefficient(matrix,coefficient); + } else if (index1.equals("cstop")) { + if (index2.equals("origin")) returnToOriginUtility.addCoefficient(matrix,coefficient); + else previousStopUtility.addCoefficient(matrix,coefficient); + } else if (index1.equals("prevStopType")) { + int type2 = WeekendTour.convertPurposeToInt(index2); + transitionConstants[type2] = coefficient; + } else if (index1.equals("logStopCount")) { + int type2 = WeekendTour.convertPurposeToInt(index2); + stopCountCoefficients[type2] = coefficient; + } else if (index1.equals("timeAccumulator")) { + totalTravelTimeCoefficient += coefficient; + } else if (index1.equals("totalAccumulator")) { + totalTripTimeCoefficient += coefficient; + } else if (index1.equals("travelDisutility") && index2.equals("origin")) { + disutilityToOriginCoefficient += coefficient; + } else if (index1.equals("travelTime") && index2.equals("origin")) { + timeToOriginCoefficient += coefficient; + } else if ((index1.equals("") || index1.equals("none")) && (index2.equals("") ||index2.equals("none"))) { + constant += coefficient; + } else { + throw new CoefficientFormatError("invalid indexing "+index1+ ","+index2+" in matrix "+matrix +" for next stop purpose model "); + } + } + + + + /** + * Method readMatrices. + * @param mr + */ + public void readMatrices(MatrixCacheReader mr) { + previousStopUtility.readMatrices(mr); + originUtility.readMatrices(mr); + returnToOriginUtility.readMatrices(mr); + } + + /** + * Method getStopPurposeCode. + * @return String + */ + public String getCode() { + return WeekendTour.convertPurposeToString(stopType); + } + +} + + /** + * Method addParameter. + * @param alternative + * @param matrix + * @param coefficient + */ + public void addCoefficient( + String alternative, + String index1, + String index2, + String matrix, + double coefficient) throws CoefficientFormatError { + Iterator alternativeIterator = alternatives.iterator(); + boolean found = false; + while (alternativeIterator.hasNext()) { + AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); + if (alternative.equals(alt.getCode())) { + alt.addCoefficient(index1,index2,matrix,coefficient); + found = true; + } + } + if (!found) { + logger.info("adding alternative "+alternative+" to "+name); + NextStopPurpose newPurpose = new NextStopPurpose(WeekendTour.convertPurposeToInt(alternative)); + addAlternative(newPurpose); + newPurpose.addCoefficient(index1,index2,matrix,coefficient); + } + } + + /** + * Method readMatrices. + * @param matrixReader + */ + public void readMatrices(MatrixCacheReader mr) { + Iterator alternativeIterator = alternatives.iterator(); + while (alternativeIterator.hasNext()) { + AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); + alt.readMatrices(mr); + } + } + + public void setMyTour(Tour myTour) { + this.myTour = myTour; + } + + public Tour getMyTour() { + return myTour; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() + */ + public void init() { + readMatrices(GenerateWeekendTours.matrixReader); + } + + + final String name; + + public WeekendStopPurposeChoice(String myName) { + super(); + this.name = myName; + } + + int monteCarloSamplePurpose() { + NextStopPurpose purpose; + try { + purpose = (NextStopPurpose) monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + e.printStackTrace(); + throw new RuntimeException("No valid purposes available",e); + } + return purpose.stopType; + } +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java new file mode 100644 index 0000000..58e439c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java @@ -0,0 +1,345 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.calgary.weekend; + +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; +import org.sandag.cvm.activityTravel.RealNumberDistribution; +import org.sandag.cvm.activityTravel.Stop; +import org.sandag.cvm.activityTravel.StopAlternative; +import org.sandag.cvm.activityTravel.StopChoice; +import org.sandag.cvm.activityTravel.Tour; +import org.sandag.cvm.activityTravel.VehicleTourTypeChoice; +import org.sandag.cvm.activityTravel.cvm.TourStartTimeModel; +import org.sandag.cvm.common.model.NoAlternativeAvailable; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public class WeekendTour extends Tour { + + static final Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); + + + /** + * The primaryPerson is the person who makes a full tour, from origin and back again. + */ + public WeekendPerson primaryPerson; + ArrayList otherPeople = new ArrayList(); // the other people in the tour + + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.Tour#sampleStops() + */ + public void sampleStops() { + String tourType = myVehicleTourType.getTourTypeName(); + if (tourType.equals("SELSE") || tourType.equals("chauf")) { + sampleReturnStops(); + } + else { + samplePrimaryStop(); + sampleIntermediateOutboundStop(); + samplePrimaryAndIntermediateDurationAndAddToStopList(); + sampleReturnStops(); + } + } + + private void samplePrimaryAndIntermediateDurationAndAddToStopList() { + if (intermediateStop !=null) { + String stopTypeCode = convertPurposeToString(intermediateStop.purpose); + RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(stopTypeCode+"Duration"); + if (myDurationModel == null) throw new RuntimeException("Can't find duration model "+stopTypeCode+"Duration for stop type number "+intermediateStop.purpose); + intermediateStop.duration = (float) myDurationModel.sampleValue(); + addStop(intermediateStop); + } + String stopTypeCode = convertPurposeToString(primaryStop.purpose); + RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(stopTypeCode+"Duration"); + if (myDurationModel == null) throw new RuntimeException("Can't find duration model "+stopTypeCode+"Duration for stop type number "+primaryStop.purpose); + primaryStop.duration = (float) myDurationModel.sampleValue(); + addStop(primaryStop); + } + + public static int convertPurposeToInt(String stopPurpose) { + if (stopPurpose.equals("work")) return 1; + if (stopPurpose.equals("school")) return 2; + if (stopPurpose.equals("exercise")) return 3; + if (stopPurpose.equals("relCivic")) return 4; + if (stopPurpose.equals("social")) return 5; + if (stopPurpose.equals("entLeisure")) return 6; + if (stopPurpose.equals("shop")) return 7; + if (stopPurpose.equals("eat")) return 8; + if (stopPurpose.equals("dropOff")) return 9; + if (stopPurpose.equals("outOfTown")) return 10; + if (stopPurpose.equals("return")) return 11; + if (stopPurpose.equals("pickUp")) return 12; + if (stopPurpose.equals("dropOff")) return 13; + throw new RuntimeException("stop purpose "+stopPurpose+" is not a valid stop purpose type"); + } + + + /** + * @return String representing the stop type + */ + public static String convertPurposeToString(int purpose) { + switch (purpose) { + case 1: + return "work"; + case 2: + return "school"; + case 3: + return "exercise"; + case 4: + return "relCivic"; + case 5: + return "social"; + case 6: + return "entLeisure"; + case 7: + return "shop"; + case 8: + return "eat"; + case 9: + return "dropOff"; + case 10: + return "outOfTown"; + case 11: + return "return"; + case 12: + return "pickUp"; + case 13: + return "dropOff"; + } + throw new RuntimeException("invalid stop purpose code "+purpose); + } + + Stop intermediateStop = null; + + private void sampleIntermediateOutboundStop() { + + //TODO smarter intermediate stop choice existance model + if (Math.random() > 0.3) return; // 70% chance of no intermediate stop + + StopChoice theModel; + String stopModelStringCode = getTourTypeCode()+"IntermediateStop"; + theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); + if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); + theModel.setTour(this); + intermediateStop = new Stop(this, getCurrentLocation(),getTotalElapsedTimeHrs()); + try { + intermediateStop.location = ((StopAlternative) theModel.monteCarloChoice()).location; + } catch (NoAlternativeAvailable e) { + e.printStackTrace(); + throw new RuntimeException("Can't find a viable intermediate stop alternative",e); + } + + WeekendStopPurposeChoice purposeModel = (WeekendStopPurposeChoice) GenerateWeekendTours.models.get(getTourTypeCode()+"StopType"); + if (purposeModel == null)throw new RuntimeException("Can't find stop purpose model for "+getTourTypeCode()); + purposeModel.setMyTour(this); + intermediateStop.purpose = purposeModel.monteCarloSamplePurpose(); + + } + + Stop primaryStop = null; + + private void samplePrimaryStop() { + StopChoice theModel; + String stopModelStringCode = getTourTypeCode()+"PrimaryStop"; + theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); + if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); + theModel.setTour(this); + // FIXME will need to rewrite start location and start time if there are any intermediate stops outbound + primaryStop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); + try { + primaryStop.location= ((StopAlternative) theModel.monteCarloChoice()).location; + } catch (NoAlternativeAvailable e) { + e.printStackTrace(); + throw new RuntimeException("Can't find a viable primary stop alternative",e); + } + primaryStop.purpose = convertPurposeToInt(getTourTypeCode()); // assume tour type code with primary stops are subset of stop type codes + } + + final int returnStopTypeCode = convertPurposeToInt("return"); + + private void sampleReturnStops() { + StopChoice theModel; + String stopModelStringCode = getTourTypeCode()+"ReturnStop"; + theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); + if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); + final int maxReturnStops = 100; + int returnStops = 0; + Stop stop; + do { + theModel.setTour(this); + stop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); + WeekendStopPurposeChoice purposeModel = (WeekendStopPurposeChoice) GenerateWeekendTours.models.get(getTourTypeCode()+"StopType"); + if (purposeModel == null)throw new RuntimeException("Can't find stop purpose model for "+getTourTypeCode()); + purposeModel.setMyTour(this); + stop.purpose = purposeModel.monteCarloSamplePurpose(); + if (stop.purpose == returnStopTypeCode) { + stop.location = getOriginZone(); + } + try { + stop.location= ((StopAlternative) theModel.monteCarloChoice()).location; + } catch (NoAlternativeAvailable e) { + e.printStackTrace(); + throw new RuntimeException("Can't find a viable return stop alternative",e); + } + if (stop.purpose != returnStopTypeCode) { + RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(WeekendTour.convertPurposeToString(stop.purpose)+"Duration"); + if (myDurationModel == null) throw new RuntimeException("no "+WeekendTour.convertPurposeToString(stop.purpose)+"Duration model"); + stop.duration = (float) myDurationModel.sampleValue(); + } + addStop(stop); + returnStops ++; + } while (stop.purpose != returnStopTypeCode && returnStops <= maxReturnStops); + if (returnStops >= maxReturnStops) { + logger.warn("Return stops hit maximum, "+maxReturnStops); + } + } + + /** + * tourTypeChoiceModel is the choice model for the tNCVehicle tour type. It is currently + * a static variable, but it is accesssed by getters and setters so could be an instance variable instead, if + * different tours need different models for choosing the tour type + */ + static WeekendTourTypeChoice tourTypeChoiceModel; + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.Tour#getVehicleTourTypeChoice() + */ + public VehicleTourTypeChoice getVehicleTourTypeChoice() { + return tourTypeChoiceModel; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.activityTravel.Tour#setVehicleTourTypeChoice(org.sandag.cvm.activityTravel.VehicleTourTypeChoice) + */ + public void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoice) { + WeekendTour.tourTypeChoiceModel = (WeekendTourTypeChoice) vehicleTourTypeChoice; + + } + + + static TourInTimeBand tourInTimeBand; + + /** + * @param titb + */ + public static void setTourInTimeBand(TourInTimeBand titb) { + tourInTimeBand = titb; + + } + + /** + * + */ + public void buildTourGroupAndFlagPeopleOutOfHome() { + primaryPerson.atHome = false; + List peopleInHousehold = (List) primaryPerson.getMyHousehold().getPersons(); + for (int p=0;p0.5) { + person.atHome = false; + otherPeople.add(person); + } + } + } + + } + + /** + * + */ + public void setReturnHomeTimesForGroupMembers() { + primaryPerson.returnTime = getCurrentTimeHrs(); + for (int p=0;p MAX_KEY_LENGTH) { + throw new IllegalArgumentException("Key is larger than permitted size of " + + MAX_KEY_LENGTH + " bytes"); + } + + file.seek(indexPositionToKeyFp(currentNumRecords)); + temp.writeTo(file); + file.seek(indexPositionToDataHeaderFp(currentNumRecords)); + newRecord.write(file); + newRecord.setIndexPosition(currentNumRecords); + writeNumRecordsHeader(currentNumRecords + 1); + } + + + /** + * Removes the record from the index. Replaces the target with the entry at the + * end of the index. + */ + protected void deleteEntryFromIndex(String key, DataHeader header, int currentNumRecords) + throws IOException { + + if (header.indexPosition != (currentNumRecords - 1)) { + String lastKey = readKeyFromIndex(currentNumRecords - 1); + DataHeader last = keyToDataHeader(lastKey); + + last.setIndexPosition(header.indexPosition); + file.seek(indexPositionToKeyFp(last.indexPosition)); + file.writeUTF(lastKey); + file.seek(indexPositionToDataHeaderFp(last.indexPosition)); + last.write(file); + } + + writeNumRecordsHeader(currentNumRecords - 1); + } + + + /** + * Adds the given record to the database. + */ + public synchronized void insertRecord(DataWriter rw) throws IOException { + String key = rw.getKey(); + + if (recordExists(key)) { + throw new IllegalArgumentException("Key exists: " + key); + } + + insureIndexSpace(getNumRecords() + 1); + + DataHeader newRecord = allocateRecord(key, rw.getDataLength()); + + writeRecordData(newRecord, rw); + addEntryToIndex(key, newRecord, getNumRecords()); + } + + + /** + * Updates an existing record. If the new contents do not fit in the original record, + * then the update is handled by deleting the old record and adding the new. + */ + public synchronized void updateRecord(DataWriter rw) throws IOException { + DataHeader header = keyToDataHeader(rw.getKey()); + + if (rw.getDataLength() > header.dataCapacity) { + deleteRecord(rw.getKey()); + insertRecord(rw); + } else { + writeRecordData(header, rw); + writeDataHeaderToIndex(header); + } + } + + + /** + * Reads a record. + */ + public synchronized DataReader readRecord(String key) throws IOException { + byte[] data = readRecordData(key); + + return new DataReader(key, data); + } + + + /** + * Reads the data for the record with the given key. + */ + protected byte[] readRecordData(String key) throws IOException { + return readRecordData(keyToDataHeader(key)); + } + + + /** + * Reads the record data for the given record header. + */ + protected byte[] readRecordData(DataHeader header) throws IOException { + byte[] buf = new byte[header.dataCount]; + + file.seek(header.dataPointer); + file.readFully(buf); + + return buf; + } + + + /** + * Updates the contents of the given record. An IOException is thrown if the + * new data does not fit in the space allocated to the record. The header's + * data count is updated, but not written to the file. + */ + protected void writeRecordData(DataHeader header, DataWriter rw) throws IOException { + if (rw.getDataLength() > header.dataCapacity) { + throw new IOException("Record data does not fit, header.dataCapacity="+ + header.dataCapacity+ + ", dataLength="+rw.getDataLength()); + } + + header.dataCount = rw.getDataLength(); + file.seek(header.dataPointer); + rw.writeTo((DataOutput) file); + } + + + /** + * Updates the contents of the given record. A DataFileException is thrown if + * the new data does not fit in the space allocated to the record. The header's + * data count is updated, but not written to the file. + */ + protected void writeRecordData(DataHeader header, byte[] data) throws IOException { + if (data.length > header.dataCapacity) { + throw new IOException("Record data does not fit, header.dataCapacity="+ + header.dataCapacity+ + ", dataLength="+data.length); + } + + header.dataCount = data.length; + file.seek(header.dataPointer); + file.write(data, 0, data.length); + } + + + /** + * Deletes a record. + */ + public synchronized void deleteRecord(String key) throws IOException { + DataHeader delRec = keyToDataHeader(key); + int currentNumRecords = getNumRecords(); + + if (getFileLength() == (delRec.dataPointer + delRec.dataCapacity)) { + // shrink file since this is the last record in the file + setFileLength(delRec.dataPointer); + } else { + DataHeader previous = getRecordAt(delRec.dataPointer - 1); + + if (previous != null) { + // append space of deleted record onto previous record + previous.dataCapacity += delRec.dataCapacity; + writeDataHeaderToIndex(previous); + } else { + // target record is first in the file and is deleted by adding its + // space to the second record. + DataHeader secondRecord = getRecordAt(delRec.dataPointer + + (long) delRec.dataCapacity); + byte[] data = readRecordData(secondRecord); + + secondRecord.dataPointer = delRec.dataPointer; + secondRecord.dataCapacity += delRec.dataCapacity; + writeRecordData(secondRecord, data); + writeDataHeaderToIndex(secondRecord); + } + } + + deleteEntryFromIndex(key, delRec, currentNumRecords); + } + + + // Checks to see if there is space for and additional index entry. If + // not, space is created by moving records to the end of the file. + protected void insureIndexSpace(int requiredNumRecords) throws IOException { + int originalFirstDataCapacity; + int currentNumRecords = getNumRecords(); + long endIndexPtr = indexPositionToKeyFp(requiredNumRecords); + + if (endIndexPtr > getFileLength() && currentNumRecords == 0) { + setFileLength(endIndexPtr); + dataStartPtr = endIndexPtr; + writeDataStartPtrHeader(dataStartPtr); + + return; + } + + // If first.dataCapacity is set to the actual data count BEFORE resetting + // dataStartPtr, and there is free space in 'first', then dataStartPtr will + // not be reset to the start of the second record. Capture the capacity + // first and use it to perform the reset. + while (endIndexPtr > dataStartPtr) { + DataHeader first = getRecordAt(dataStartPtr); + byte[] data = readRecordData(first); + first.dataPointer = getFileLength(); + originalFirstDataCapacity = first.dataCapacity; + first.dataCapacity = data.length; + setFileLength(first.dataPointer + data.length); + writeRecordData(first, data); + writeDataHeaderToIndex(first); + dataStartPtr += originalFirstDataCapacity; + writeDataStartPtrHeader(dataStartPtr); + } + } + + + /** + * Closes the file. + */ + public synchronized void close() throws IOException { + try { + file.close(); + } finally { + file = null; + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java new file mode 100644 index 0000000..761b4b0 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java @@ -0,0 +1,114 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.ObjectInput; +import java.io.ObjectInputStream; +import java.util.ArrayList; +import org.apache.log4j.Logger; + +/** + * Reads a binary file containing a seriazlied TableDataSet class and creates + * an instance of a TableDataSet class. + * + * @author Tim Heier + * @version 1.0, 5/08/2004 + * + */ +public class BinaryFileReader extends TableDataFileReader implements DataTypes { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + //These attributes are initialized on each call to readFile() + private int columnCount; + private int rowCount; + private ArrayList columnData = new ArrayList(); + private ArrayList columnLabels = new ArrayList(); + private int[] columnType; + + + public BinaryFileReader () { + } + + + /** + * Reads a binary file containing serialized objects that make up a TableDataSet + * object. + * + * @param file name of file which contains the binary data + * @return a TableDataSet object + * + * @throws IOException when the file is not found + */ + public TableDataSet readFile(File file) throws IOException { + + TableDataSet table = null; + + try { + logger.debug("Opening file: "+file); + + //Open the file + ObjectInput inStream = new ObjectInputStream(new FileInputStream(file)); + + //Read magic number + int magicNumber = inStream.readInt(); + + //Read number of columns + int nCols = inStream.readInt(); + columnType = new int[nCols]; + + //Read titles + for (int c=0; c < nCols; c++) { + columnLabels.add( inStream.readUTF() ); + } + + //Read column data + for (int c=0; c < nCols; c++) { + Object colObj = inStream.readObject(); + columnData.add( colObj ); + } + inStream.close(); + + table = new TableDataSet(); + table.setName(file.toString()); + for (int i=0; i < nCols; i++) { + table.appendColumn(columnData.get(i), (String) columnLabels.get(i)); + } + } + catch (ClassNotFoundException e) { + logger.error("", e); + } + + return table; + } + + + public TableDataSet readTable(String tableName) throws IOException { + File fileName = new File (getMyDirectory() + File.separator + tableName + ".binTable"); + TableDataSet myTable = readFile(fileName); + myTable.setName(tableName); + return myTable; + } + + + public void close() { + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java new file mode 100644 index 0000000..0d1f0fb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java @@ -0,0 +1,106 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.ObjectOutput; +import java.io.ObjectOutputStream; +import java.util.ArrayList; +import org.apache.log4j.Logger; + + +/** + * Writes a TableDataSet class to a binary file. + * + * @author Tim Heier + * @version 1.0, 5/08/2004 + * + */ +public class BinaryFileWriter extends TableDataFileWriter implements DataTypes { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + + public BinaryFileWriter () { + } + + + /** + * Writes a binary file containing serialized objects that make up a TableDataSet + * object. + * + * @param tableData the tabledataset to be written out to file + * @param file the file where the data should be written + * + * @throws IOException is thrown when the file cannot be written to + */ + public void writeFile(TableDataSet tableData, File file) throws IOException { + + //Pull data out of tableData object for convenience + int nCols = tableData.getColumnCount(); + String[] columnLabels = tableData.getColumnLabels(); + ArrayList columnData = tableData.getColumnData(); + + try { + logger.debug("Opening file: "+file); + + //Create file + ObjectOutput outStream = new ObjectOutputStream(new FileOutputStream(file)); + + //Write magic number + outStream.writeInt(1); + + //Write number of columns + outStream.writeInt( nCols ); + + //Write titles + for (int c=0; c < nCols; c++) { + outStream.writeUTF( columnLabels[c] ); + } + + //Write column data + for (int c=0; c < nCols; c++) { + outStream.writeObject( columnData.get(c) ); + } + + outStream.close(); + } + catch (IOException e) { + throw e; + } + + } + + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataWriter#writeTable(org.sandag.cvm.common.datafile.TableDataSet, java.lang.String) + */ + public void writeTable(TableDataSet tableData, String tableName) throws IOException { + File file = new File (getMyDirectory().getPath() + File.separator + tableName + ".binTable"); + writeFile(tableData, file); + } + + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataWriter#close() + */ + public void close() { + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java new file mode 100644 index 0000000..7f5637b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java @@ -0,0 +1,810 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import org.apache.log4j.Logger; + +import java.io.*; +import java.net.URL; +import java.net.URLConnection; +import java.util.ArrayList; +import java.util.List; +import java.util.regex.Pattern; + + +/** + * Creates a TableData class from a CSV file. The default delimiter character is a comma. + * + * @author Tim Heier + * @version 1.0, 2/07/2004 + * + */ +public class CSVFileReader extends TableDataFileReader implements DataTypes { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + //Can be set by caller + private char delimiter = ','; + + //Pattern composed of regular expression used to parse CSV fields + private String pattern = ",(?=(?:[^\"]*\"[^\"]*\")*(?![^\"]*\"))"; + private Pattern regexPattern = Pattern.compile(pattern); + + //These attributes are initialized on each call to readFile() + private int columnCount; + private int rowCount; + private List columnData; + private ArrayList columnLabels; + private int[] columnType; + + private boolean padNulls=false; + + + public boolean isPadNulls() { + return padNulls; + } + + + public void setPadNulls(boolean padNulls) { + this.padNulls = padNulls; + } + + + public CSVFileReader () { + } + + + /** + * Sets the delimiters used by the StringTokenizer when reading column values. + * + * @param delimiter character separating fields in CSV file, default is a comma + */ + public void setDelimiter(char delimiter) { + this.delimiter = delimiter; + + //Update the pattern string with the new delimiter + pattern = Character.toString(delimiter) + pattern.substring(1); + regexPattern = Pattern.compile(pattern); + } + + + /** + * + * @return the delimiter in use + */ + public char getDelimiter() { + return delimiter; + } + + /** + * + * @return the pattern string used to parse CSV fields + */ + public String getPattern() { + return pattern; + } + + /** + * + * @param pattern new pattern string to be used when parsing CSV fields + */ + public void setPattern(String pattern) { + this.pattern = pattern; + regexPattern = Pattern.compile(pattern); + } + + + public TableDataSet readFile(File file) throws IOException { + return readFile(file, true); + } + + public TableDataSet readFile(String urlString) throws IOException { + return readFile(urlString, true); + } + + + /** + * Convenience method to load a CSV file into a table data class. + * + * @param file name of file to read + * @param columnLabelsPresent determines whether first line is treated + * as column titles + * @throws IOException + * + */ + public TableDataSet readFile(File file, boolean columnLabelsPresent) throws IOException { + return readFile(file, columnLabelsPresent, null); + } + + public TableDataSet readFile(String urlString, boolean columnLabelsPresent) throws IOException { + return readFile(urlString, columnLabelsPresent, null); + } + + + + + /** + * Convenience method to load a CSV file into a table data class. + * + * @param file name of file to read + * @param columnsToRead list of column labels that should be read - all other + * columns will be dropped from the table data set + * @throws IOException + * + */ + public TableDataSet readFile(File file, String[] columnsToRead) throws IOException { + return readFile(file, true, columnsToRead); + } + + + /** + * Main method which loads a CSV file into a table data class. + * + * @param file name of file to read + * @param columnLabelsPresent determines whether first line is treated + * as column titles + * @param columnsToRead list of column labels that should be read - all other + * columns will be dropped from the table data set + * @throws IOException + * + */ + public TableDataSet readFile(File file, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { + + if ((columnsToRead != null) && (columnLabelsPresent == false)) { + throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); + } + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + BufferedReader inStream = openFile(file); + + boolean[] readColumnFlag = null; + + if (columnLabelsPresent) { + readColumnFlag = readColumnLabels(inStream, columnsToRead); + boolean readAColumn=false; + for (boolean b: readColumnFlag) { + readAColumn = readAColumn || b; + } + if (!readAColumn) { + logger.fatal("No columns read when reading file "+file); + throw new RuntimeException("No columns read when reading file "+file); + } + } + readData(file, inStream, columnLabelsPresent, readColumnFlag); + + TableDataSet tds = makeTableDataSet(); + tds.setName(file.toString()); + return tds; + } + + /** + * Main method which loads a CSV file into a table data class. + * + * @param urlString http address of file to read + * @param columnLabelsPresent determines whether first line is treated + * as column titles + * @param columnsToRead list of column labels that should be read - all other + * columns will be dropped from the table data set + * @throws IOException + * + */ + public TableDataSet readFile(String urlString, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { + + if ((columnsToRead != null) && (columnLabelsPresent == false)) { + throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); + } + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + URL url; + URLConnection urlConn; + DataInputStream dis; + + url = new URL(urlString); + urlConn = url.openConnection(); + dis = new DataInputStream(urlConn.getInputStream()); + BufferedReader inStream = new BufferedReader(new InputStreamReader(dis)); + + + boolean[] readColumnFlag = null; + + if (columnLabelsPresent) { + readColumnFlag = readColumnLabels(inStream, columnsToRead); + boolean readAColumn=false; + for (boolean b: readColumnFlag) { + readAColumn = readAColumn || b; + } + if (!readAColumn) { + logger.fatal("No columns read when reading file "+ urlString); + throw new RuntimeException("No columns read when reading file "+ urlString); + } + } + readData(urlString, inStream, columnLabelsPresent, readColumnFlag); + + TableDataSet tds = makeTableDataSet(); + tds.setName(urlString.substring((urlString.lastIndexOf("/"))+1, urlString.length())); + System.out.println("Table Name is: " + tds.getName()); + return tds; + } + + + /* + * Read the csv file with a String[] of specified column formats (NUMBER or STRING), + * where the format is specified for all columns, all columns are read, + * and column headings must be present on the first line. + */ + public TableDataSet readFileWithFormats(File file, String[] columnFormats) throws IOException { + + boolean columnLabelsPresent = true; + String[] columnsToRead = null; + + if ((columnsToRead != null) && (columnLabelsPresent == false)) { + throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); + } + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + BufferedReader inStream = openFile(file); + + boolean[] readColumnFlag = null; + + if (columnLabelsPresent) { + readColumnFlag = readColumnLabels(inStream, columnsToRead); + } + readData(file, inStream, columnLabelsPresent, readColumnFlag, columnFormats); + + TableDataSet tds = makeTableDataSet(); + tds.setName(file.toString()); + return tds; + } + + + private BufferedReader openFile(File file) throws IOException { + logger.debug("Opening file: "+file); + + BufferedReader inStream = null; + try { + inStream = new BufferedReader( new FileReader(file) ); + } + catch (IOException e) { + throw e; + } + + return inStream; + } + + + /** + * Read and parse the column titles from the first line of file. + */ + private boolean[] readColumnLabels(BufferedReader inStream, String[] columnsToRead) + throws IOException { + //Read the first line + String line = inStream.readLine(); + + //Test for an empty file + if (line == null) { + throw new IOException("Error: file looks like it's empty"); + } + + //Tokenize the first line + String[] tokens = parseTokens(line); + int count = tokens.length; + + boolean[] readColumnFlag = new boolean[count]; + + //Initialize the readColumnFlag to false if the caller has supplied a + //list of columns. It will be turned to true basedon a comparison of the + //column labels found in the file. Otherwise initialize to true. + for (int i=0; i < count; i++) { + if (columnsToRead != null) + readColumnFlag[i] = false; + else + readColumnFlag[i] = true; + } + + //Read column titles + int c = 0; + for (int i=0; i < count; i++) { + String column_name = tokens[i]; + + //Check if column should be read based on list supplied by caller + if (columnsToRead != null) { + for (int j=0; j < columnsToRead.length; j++) { + if (columnsToRead[j].equalsIgnoreCase(column_name)) { + readColumnFlag[c] = true; + + columnLabels.add(column_name); + columnCount++; + break; + } + } + } + else { + columnLabels.add(column_name); + columnCount++; + } + c++; //the actual columnn number in the file being read + } + + //Debugging output + String msg = "column read flag = "; + for (int i=0; i < readColumnFlag.length; i++) { + if (readColumnFlag[i] == true) + msg += "true"; + else + msg += "false"; + if (i < (readColumnFlag.length-1)) + msg += ", "; + } + msg += "\n"; + logger.debug(msg); + + + return readColumnFlag; + } + + + /** + * Read and parse data portion of file. + */ + private void readData(File file, BufferedReader inStream, boolean columnLabelsPresent, + boolean[] readColumnFlag) + throws IOException { + + int rowNumber = 0; + + //Determine the number of lines in the file + rowCount = findNumberOfLinesInFile(file); + + readRows(file.toString(), inStream, columnLabelsPresent, readColumnFlag, rowNumber); + inStream.close(); + } + + + private void readRows(String source, BufferedReader inStream, + boolean columnLabelsPresent, boolean[] readColumnFlag, int rowNumber) + throws IOException { + logger.debug("number of lines in file: " + rowCount); + if (columnLabelsPresent) { + rowCount--; + } + + //Process each line in the file + String line; + if (rowCount == 0) { + columnType = new int[columnCount]; + readColumnFlag = new boolean[columnCount]; + for (int col =0; col #.00 and fieldWidth = 8 + * %6.0f --> #.# and fieldWidth = 6 + * %.3f --> #.000 and fieldWidth = 3 + * + * @param tableData the TableDataSet to write + * @param file the destination file to write to + * @param fieldFormat an array of PaddedDecimalFormat objects, one for each column + * + */ + public void writeFile(TableDataSet tableData, File file, PaddedDecimalFormat[] fieldFormat) throws IOException { + String formatString; + PrintWriter outStream = null; + + //Pull data out of tableData object for convenience + int nCols = tableData.getColumnCount(); + int nRows = tableData.getRowCount(); + int[] columnType = tableData.getColumnType(); + String[] columnLabels = tableData.getColumnLabels(); + ArrayList columnData = tableData.getColumnData(); + + if (fieldFormat.length != nCols) { + throw new RuntimeException("Length of format array is " + fieldFormat.length + + " should be " + nCols); + } + + try { + outStream = new PrintWriter (new BufferedWriter( new FileWriter(file) ) ); + + //Print titles + for (int i = 0; i < columnLabels.length; i++) { + if (i != 0) + outStream.print(","); + outStream.print( columnLabels[i] ); + } + outStream.println(); + + //Print data + for (int r=0; r < nRows; r++) { + //float[] rowValues = getRowValues(r, 0); + + for (int c=0; c < nCols; c++) { + if (c != 0) + outStream.print(","); + + switch(columnType[c]) { + case STRING: + String[] s = (String[]) columnData.get(c); + if (quoteStrings) { + outStream.print("\""+ s[r]+"\"" ); + } else { + outStream.print(s[r]); + } + break; + case NUMBER: + float[] f = (float[]) columnData.get(c); + if (Float.isInfinite(f[r])) { + // don't want the infinity figure (sideways 8) in output file, want Infinity or -Infinity. + outStream.print(f[r]); + } else { + outStream.print( fieldFormat[c].format(f[r]) ); + } + break; + default: + throw new RuntimeException("unknown column data type: " + columnType[c]); + } + } + outStream.println(); + } + outStream.close(); + } + catch (IOException e) { + throw e; + } + + //Update dirty flag + tableData.setDirty(false); + } + + + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataWriter#writeTable(org.sandag.cvm.common.datafile.TableDataSet, java.lang.String) + */ + public void writeTable(TableDataSet tableData, String tableName) throws IOException { + File file = new File (getMyDirectory().getPath() + File.separator + tableName + ".csv"); + writeFile(tableData, file); + } + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataWriter#close() + */ + public void close() { + } + + public boolean isQuoteStrings() { + return quoteStrings; + } + + public void setQuoteStrings(boolean quoteStrings) { + this.quoteStrings = quoteStrings; + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java new file mode 100644 index 0000000..55b45ca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java @@ -0,0 +1,304 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileReader; +import java.io.BufferedReader; +import java.io.IOException; +import java.io.Serializable; +import java.util.StringTokenizer; +import java.util.ArrayList; +import org.apache.log4j.Logger; + + +/** + * Reads a standard Emme/2 d211 text format file containing node and link records + * for a transportation network. + * + * @author Jim Hicks + * @version 1.0, 5/12/2004 + * + */ +public class D211FileReader implements Serializable { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + + + public D211FileReader () { + } + + + + public TableDataSet readNodeTable (File file) throws IOException { + + int record = 0; + boolean found_t_nodes_init = false; + boolean found_t_links_init = false; + + ArrayList nList = new ArrayList(); + ArrayList xList = new ArrayList(); + ArrayList yList = new ArrayList(); + + float[][] dataTable = null; + + TableDataSet table = null; + + + + try { + logger.debug( "Opening d211 file to read node records: " + file.getName() ); + + //Open the file + BufferedReader in = new BufferedReader(new FileReader(file)); + + String s = new String(); + while ((s = in.readLine()) != null) { + + record++; + + if ( s.indexOf("t nodes") >= 0 ) { + + found_t_nodes_init = true; + + } + else if ( s.indexOf("t links") >= 0 ) { + + found_t_links_init = true; + + } + else if (found_t_nodes_init && !found_t_links_init) { + + parseNode(s, nList, xList, yList); + + } + + } + + } + catch (Exception e) { + System.out.println ("IO Exception caught reading node table data from d211 format file: " + file.getName() + ", record number=" + record ); + e.printStackTrace(); + } + + + + dataTable = new float[nList.size()][3]; + for (int i=0; i < nList.size(); i++) { + try { + dataTable[i][0] = Integer.parseInt ( (String)nList.get(i) ); + dataTable[i][1] = Float.parseFloat ( (String)xList.get(i) ); + dataTable[i][2] = Float.parseFloat ( (String)yList.get(i) ); + } catch (Exception e) { + String msg = "Can't parse "+i+"th node"; + logger.fatal(msg,e); + throw new RuntimeException(msg,e); + } + } + + + ArrayList tableHeadings = new ArrayList(); + tableHeadings.add ("node"); + tableHeadings.add ("x"); + tableHeadings.add ("y"); + + + table = TableDataSet.create( dataTable, tableHeadings ); + + return table; + + } + + + + public TableDataSet readLinkTable (File file) throws IOException { + return readLinkTable ( file, 'a' ); + } + + + public TableDataSet readLinkTableMods (File file) throws IOException { + return readLinkTable ( file, 'm' ); + } + + + private TableDataSet readLinkTable (File file, char action) throws IOException { + + int record = 0; + boolean found_t_links_init = false; + + ArrayList anList = new ArrayList(); + ArrayList bnList = new ArrayList(); + ArrayList distList = new ArrayList(); + ArrayList modeList = new ArrayList(); + ArrayList typeList = new ArrayList(); + ArrayList lanesList = new ArrayList(); + ArrayList vdfList = new ArrayList(); + ArrayList ul1List = new ArrayList(); + ArrayList ul2List = new ArrayList(); + ArrayList ul3List = new ArrayList(); + ArrayList ul4List = new ArrayList(); + + float[][] dataTable = null; + String[] stringColumn = null; + + TableDataSet table = null; + + + + try { + logger.debug( "Opening d211 file to read link records: " + file.getName() ); + + //Open the file + BufferedReader in = new BufferedReader(new FileReader(file)); + + String s = new String(); + while ((s = in.readLine()) != null) { + + record++; + + if ( s.indexOf("t links") >= 0 ) { + + found_t_links_init = true; + + } + else if (found_t_links_init) { + + parseLink( s, action, anList, bnList, distList, modeList, typeList, lanesList, vdfList, ul1List, ul2List, ul3List, ul4List ); + + } + + } + + } + catch (Exception e) { + System.out.println ("IO Exception caught reading link table data from d211 format file: " + file.getName() + ", record number=" + record ); + e.printStackTrace(); + } + + + + dataTable = new float[anList.size()][10]; + stringColumn = new String[anList.size()]; + for (int i=0; i < anList.size(); i++) { + dataTable[i][0] = Integer.parseInt ( (String)anList.get(i) ); + dataTable[i][1] = Integer.parseInt ( (String)bnList.get(i) ); + dataTable[i][2] = Float.parseFloat ( (String)distList.get(i) ); + stringColumn[i] = (String)modeList.get(i); + dataTable[i][3] = Integer.parseInt ( (String)typeList.get(i) ); + dataTable[i][4] = Float.parseFloat ( (String)lanesList.get(i) ); + dataTable[i][5] = Integer.parseInt ( (String)vdfList.get(i) ); + + try { + dataTable[i][6] = Float.parseFloat ( (String)ul1List.get(i) ); + } + catch (Exception e) { + dataTable[i][6] = 0.0f; + } + try { + dataTable[i][7] = Float.parseFloat ( (String)ul2List.get(i) ); + } + catch (Exception e) { + dataTable[i][7] = 0.0f; + } + try { + dataTable[i][8] = Float.parseFloat ( (String)ul3List.get(i) ); + } + catch (Exception e) { + dataTable[i][8] = 0.0f; + } + try { + dataTable[i][9] = Float.parseFloat ( (String)ul4List.get(i) ); + } + catch (Exception e) { + dataTable[i][9] = 0.0f; + } + } + + + ArrayList tableHeadings = new ArrayList(); + tableHeadings.add ("anode"); + tableHeadings.add ("bnode"); + tableHeadings.add ("dist"); + tableHeadings.add ("type"); + tableHeadings.add ("lanes"); + tableHeadings.add ("vdf"); + tableHeadings.add ("ul1"); + tableHeadings.add ("ul2"); + tableHeadings.add ("ul3"); + tableHeadings.add ("ul4"); + + table = TableDataSet.create( dataTable, tableHeadings ); + table.appendColumn (stringColumn, "mode"); + + + + return table; + + } + + + + void parseNode ( String InputString, ArrayList n, ArrayList x, ArrayList y ) { + + StringTokenizer st = new StringTokenizer(InputString); + + if (st.hasMoreTokens()) { + + if ((st.nextToken()).charAt(0) == 'a') { // read only add records + + n.add ( st.nextToken() ); + x.add ( st.nextToken() ); + y.add ( st.nextToken() ); + + } + + } + + } + + + + + void parseLink ( String InputString, char action, ArrayList anList, ArrayList bnList, ArrayList distList, ArrayList modeList, ArrayList typeList, ArrayList lanesList, ArrayList vdfList, ArrayList ul1List, ArrayList ul2List, ArrayList ul3List, ArrayList ul4List ) { + + StringTokenizer st = new StringTokenizer(InputString); + int count = st.countTokens(); + + while (st.hasMoreTokens()) { + + if ( (st.nextToken()).charAt(0) == action ) { // process add or mod records as requested + + anList.add ( st.nextToken() ); + bnList.add ( st.nextToken() ); + distList.add ( st.nextToken() ); + modeList.add ( st.nextToken() ); + typeList.add ( st.nextToken() ); + lanesList.add ( st.nextToken() ); + vdfList.add ( st.nextToken() ); + ul1List.add ( st.nextToken() ); + if (st.hasMoreTokens()) ul2List.add ( st.nextToken() ); + if (st.hasMoreTokens()) ul3List.add ( st.nextToken() ); + if (st.hasMoreTokens()) ul4List.add ( st.nextToken() ); + + } + + } + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java new file mode 100644 index 0000000..f991620 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java @@ -0,0 +1,129 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileReader; +import java.io.BufferedReader; +import java.io.IOException; +import java.io.Serializable; +import java.util.StringTokenizer; +import java.util.ArrayList; +import org.apache.log4j.Logger; + + +/** + * Reads a standard Emme/2 d231 text format file containing turn definitions + * for a transportation network. + * + * @author Jim Hicks + * @version 1.0, 5/12/2004 + * + */ +public class D231FileReader implements Serializable { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + + + public D231FileReader () { + } + + + + public float[][] readTurnTable (File file) throws IOException { + + int record = 0; + boolean found_t_turns_init = false; + + ArrayList values = new ArrayList(); + + float[][] dataTable = null; + + + try { + logger.debug( "Opening d231 file to read turn records: " + file.getName() ); + + //Open the file + BufferedReader in = new BufferedReader(new FileReader(file)); + + String s = new String(); + while ((s = in.readLine()) != null) { + + record++; + + if ( s.indexOf("t turns") >= 0 ) { + + found_t_turns_init = true; + + } + else if (found_t_turns_init) { + + values.add ( parseRecord ( s ) ); + + } + + } + + } + catch (Exception e) { + System.out.println ("IO Exception caught reading node table data from d211 format file: " + file.getName() + ", record number=" + record ); + e.printStackTrace(); + } + + + + dataTable = new float[values.size()][((float[])values.get(0)).length]; + for (int i=0; i < values.size(); i++) { + dataTable[i] = (float[])values.get(i); + } + + + return dataTable; + + } + + + + private float[] parseRecord ( String InputString ) { + + float[] values = new float[8]; + + StringTokenizer st = new StringTokenizer(InputString); + + + int i = 0; + + if (st.hasMoreTokens()) { + + if ((st.nextToken()).charAt(0) == 'a') { // read only add records + + while (st.hasMoreTokens()) { + + values[i] = Float.parseFloat ( st.nextToken() ); + i++; + + } + } + + } + + return values; + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java new file mode 100644 index 0000000..b27c645 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java @@ -0,0 +1,225 @@ +/* + * Copyright 2006 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.util.ArrayList; +import java.util.List; + +import org.apache.log4j.Logger; + +import com.linuxense.javadbf.DBFException; +import com.linuxense.javadbf.DBFField; +import com.linuxense.javadbf.DBFReader; +import org.sandag.cvm.common.datafile.TableDataFileReader; +import org.sandag.cvm.common.datafile.TableDataSet; + +import static org.sandag.cvm.common.datafile.DataTypes.*; + +/** + * This class reads DBF files and stores them as TableDataSets. It uses the third party + * classes from com.linuxense.javadbf + * + * @author Erhardt + * @version 1.0 Nov 27, 2006 + * + */ +public class DBFFileReader extends TableDataFileReader { + + protected static transient Logger logger = Logger.getLogger(DBFFileReader.class); + + //These attributes are initialized on each call to readFile() + private int columnCount; + private int rowCount; + private List columnData; + private ArrayList columnLabels; + private int[] columnType; + private DBFField[] columnDBF; + + /** + * Default constructor. + * + */ + public DBFFileReader() { + + } + + /** + * @see org.sandag.cvm.common.datafile.TableDataFileReader#readFile(java.io.File) + */ + @Override + public TableDataSet readFile(File file) throws IOException { + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + // open file + logger.debug("Opening file: "+file); + InputStream inStream = new FileInputStream(file); + DBFReader reader = new DBFReader(inStream); + + // read data + readHeader(reader); + readData(reader); + inStream.close(); + + // make table + TableDataSet tds = makeTableDataSet(); + tds.setName(file.toString()); + return tds; + } + + /** + * Reads the header information and sets up the table structure. + * + * @param reader + * @throws DBFException + */ + private void readHeader(DBFReader reader) throws DBFException { + + // Determine the number of columns in the file + columnCount = reader.getFieldCount(); + logger.debug("number of columns in file: " + columnCount); + + //Determine the number of lines in the file + rowCount = reader.getRecordCount(); + logger.debug("number of lines in file: " + rowCount); + + //Get the field descriptions + columnType = new int[columnCount]; + columnDBF = new DBFField[columnCount]; + for (int col=0; col 0) { + if (dataLength > maxElementSize) { + throw new IllegalArgumentException( + "Size of entry="+dataLength+" bytes is larger than maxElementSize=" + maxElementSize); + } + setFileLength(fp + maxElementSize); //all elements are maxElementSize in length + newRecord = new DataHeader(fp, maxElementSize); + } + else { + setFileLength(fp + dataLength); + newRecord = new DataHeader(fp, dataLength); + } + } + + return newRecord; + } + + + /** + * Returns the record to which the target file pointer belongs - meaning the specified location + * in the file is part of the record data of the DataHeader which is returned. Returns null if + * the location is not part of a record. (O(n) mem accesses) + */ + protected DataHeader getRecordAt(long targetFp) { + Enumeration e = memIndex.elements(); + + while (e.hasMoreElements()) { + DataHeader next = (DataHeader) e.nextElement(); + + if ((targetFp >= next.dataPointer) && (targetFp < (next.dataPointer + (long) next.dataCapacity))) { + return next; + } + } + + return null; + } + + + /** + * Closes the database. + */ + public synchronized void close() throws IOException { + try { + super.close(); + } finally { + memIndex.clear(); + memIndex = null; + } + } + + + /** + * Adds the new record to the in-memory index and calls the super class add + * the index entry to the file. + */ + protected void addEntryToIndex(String key, DataHeader newRecord, int currentNumRecords) throws IOException { + super.addEntryToIndex(key, newRecord, currentNumRecords); + memIndex.put(key, newRecord); + } + + + /** + * Removes the record from the index. Replaces the target with the entry at the + * end of the index. + */ + protected void deleteEntryFromIndex(String key, DataHeader header, int currentNumRecords) throws IOException { + super.deleteEntryFromIndex(key, header, currentNumRecords); + + DataHeader deleted = (DataHeader) memIndex.remove(key); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java new file mode 100644 index 0000000..091015c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java @@ -0,0 +1,114 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.DataInput; +import java.io.DataOutput; +import java.io.IOException; + +public class DataHeader { + + /** + * File pointer to the first byte of record data (8 bytes). + */ + protected long dataPointer; + + /** + * Actual number of bytes of data held in this record (4 bytes). + */ + protected int dataCount; + + /** + * Number of bytes of data that this record can hold (4 bytes). + */ + protected int dataCapacity; + + /** + * Indicates this header's position in the file index. + */ + protected int indexPosition; + + protected DataHeader() { + } + + + protected DataHeader(long dataPointer, int dataCapacity) { + if (dataCapacity < 1) { + throw new IllegalArgumentException("Bad record size: " + dataCapacity); + } + + this.dataPointer = dataPointer; + this.dataCapacity = dataCapacity; + this.dataCount = 0; + } + + protected int getIndexPosition() { + return indexPosition; + } + + + protected void setIndexPosition(int indexPosition) { + this.indexPosition = indexPosition; + } + + + protected int getDataCapacity() { + return dataCapacity; + } + + + protected int getFreeSpace() { + return dataCapacity - dataCount; + } + + + protected void read(DataInput in) throws IOException { + dataPointer = in.readLong(); + dataCapacity = in.readInt(); + dataCount = in.readInt(); + } + + + protected void write(DataOutput out) throws IOException { + out.writeLong(dataPointer); + out.writeInt(dataCapacity); + out.writeInt(dataCount); + } + + + protected static DataHeader readHeader(DataInput in) throws IOException { + DataHeader r = new DataHeader(); + + r.read(in); + + return r; + } + + + /** + * Returns a new record header which occupies the free space of this record. + * Shrinks this record size by the size of its free space. + */ + protected DataHeader split() { + long newFp = dataPointer + (long) dataCount; + DataHeader newData = new DataHeader(newFp, getFreeSpace()); + + dataCapacity = dataCount; + + return newData; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java new file mode 100644 index 0000000..58bb994 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java @@ -0,0 +1,94 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.*; +import org.apache.log4j.Logger; + +public class DataReader { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + String key; + byte[] data; + ByteArrayInputStream in; + ObjectInputStream objIn; + + public DataReader(String key, byte[] data) { + this.key = key; + this.data = data; + in = new ByteArrayInputStream(data); + } + + public String getKey() { + return key; + } + + + public byte[] getData() { + return data; + } + + + public InputStream getInputStream() throws IOException { + return in; + } + + + public ObjectInputStream getObjectInputStream() throws IOException { + if (objIn == null) { + objIn = new ObjectInputStream(in); + } + + return objIn; + } + + + /** + * Reads the next object in the record using an ObjectInputStream. + */ + public Object readObject() throws IOException, OptionalDataException, ClassNotFoundException { + return getObjectInputStream().readObject(); + } + + + + /** + * Reads the serialized object from filename on disk using the specified key. + */ + public static Object readDiskObject ( String filename, String key ) { + + Object obj=null; + DataFile dataFile=null; + + try { + dataFile = new DataFile( filename, "r" ); + DataReader dr = dataFile.readRecord( key ); + obj = dr.readObject(); + } + catch (Exception e) { + logger.error("Exception thrown when reading DiskObject file: " + filename ); + e.printStackTrace(); + } + + return obj; + } + + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java new file mode 100644 index 0000000..2de42ff --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java @@ -0,0 +1,38 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.Serializable; + +/** + * Supported data types in table oriented data set classes. + * + * @author Tim Heier + * @version 1.0, 1/30/2003 + * + */ + +public interface DataTypes extends Serializable { + + //Supported data types + public final static int NULL = 0; + public final static int BOOLEAN = 1; + public final static int STRING = 2; + public final static int NUMBER = 3; + public final static int DOUBLE = 4; + public final static int OTHER = 1111; +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java new file mode 100644 index 0000000..6a4420a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java @@ -0,0 +1,105 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.DataOutput; +import java.io.IOException; +import java.io.ObjectOutputStream; +import org.apache.log4j.Logger; + +public class DataWriter { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + private String key; + private DbByteArrayOutputStream out; + private ObjectOutputStream objOut; + + + public DataWriter(String key) { + this.key = key; + out = new DbByteArrayOutputStream(); + try { + objOut = new ObjectOutputStream(out); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + + public String getKey() { + return key; + } + + + public void writeObject(Object o) throws IOException { + + //Reset the size of the underlying ByteArrayOutputStream so it can be reused + //out.reset(); + objOut.reset(); + objOut.writeObject(o); + objOut.flush(); + } + + + /** + * Returns the number of bytes in the data. + */ + public int getDataLength() { + return out.size(); + } + + + /** + * Writes the data out to the stream without re-allocating the buffer. + */ + public void writeTo(DataOutput str) throws IOException { + out.writeTo(str); + } + + + /** + * Writes the serialized contents of the object to filename on disk using the specified key. + */ + public static void writeDiskObject ( Object obj, String filename, String key ) { + + DataFile dataFile=null; + + try { + dataFile = new DataFile( filename, 1 ); + DataWriter dw = new DataWriter( key ); + dw.writeObject( obj ); + dataFile.insertRecord(dw); + } + catch (IOException e) { + logger.error( "IO Exception thrown when writing DiskObject file: " + filename ); + e.printStackTrace(); + } + + try { + dataFile.close(); + } + catch (IOException e) { + logger.error( "IO Exception thrown when closing DiskObject file: " + filename ); + e.printStackTrace(); + } + } + + +} + + diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java new file mode 100644 index 0000000..4e4133c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java @@ -0,0 +1,47 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.ByteArrayOutputStream; +import java.io.DataOutput; +import java.io.IOException; + +/** + * Extends ByteArrayOutputStream to provide a way of writing the buffer to + * a DataOutput without re-allocating it. + */ +public class DbByteArrayOutputStream extends ByteArrayOutputStream { + + public DbByteArrayOutputStream() { + super(); + } + + + public DbByteArrayOutputStream(int size) { + super(size); + } + + /** + * Writes the full contents of the buffer a DataOutput stream. + */ + public synchronized void writeTo(DataOutput dstr) throws IOException { + byte[] data = super.buf; + int l = super.size(); + + dstr.write(data, 0, l); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java new file mode 100644 index 0000000..5bbb092 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java @@ -0,0 +1,227 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.IOException; +import java.io.Serializable; + +/** + * + * @author Tim Heier + * @version 1.0, 8/13/2003 + * + */ + +public class DiskObjectArray implements Serializable { + + private String fileName; + private int arraySize; + private int maxElementSize; + + private String[] indexArray; + + private transient DataFile dataFile; + + private DiskObjectArray() { + } + + + /** + * Constructor called when creating/overwriting an existing file. + * + * @param fileName fully quallified file name of file used to + * @param arraySize number of elements in the array + * @param maxElementSize maximum size of an element that will be stored (in bytes) + * @throws IOException + */ + public DiskObjectArray(String fileName, int arraySize, int maxElementSize) throws IOException { + this.fileName = fileName; + this.arraySize = arraySize; + this.maxElementSize = maxElementSize; + + dataFile = new DataFile(fileName, arraySize, maxElementSize); + + //Number of elements in array - used when the data file is opened later + DataWriter dw = new DataWriter("arraySize"); + dw.writeObject( new Integer(arraySize) ); + dataFile.insertRecord(dw); + + //maximum size of an element that will be stored - used when the data file is opened later + dw = new DataWriter("maxElementSize"); + dw.writeObject( new Integer(maxElementSize) ); + dataFile.insertRecord(dw); + + indexArray = new String[arraySize+1]; + for (int i=0; i <=arraySize; i++) { + indexArray[i] = i + ""; + } + + } + + + /** + * Constructor called when opening an existing file. + * + * @param fileName fully qualified file name of data-file + * @throws IOException + */ + public DiskObjectArray(String fileName) throws IOException, FileNotFoundException { + this.fileName = fileName; + + File f = new File(fileName); + + if (! f.exists()) { + throw new FileNotFoundException("Database file could not be found: " + fileName); + } + + //Read size of array from data file and initialize indexArray + try { + dataFile = new DataFile(fileName, "rw"); + DataReader dr = dataFile.readRecord("arraySize"); + Integer i = (Integer) dr.readObject(); + this.arraySize = i.intValue(); + + dr = dataFile.readRecord("maxElementSize"); + i = (Integer) dr.readObject(); + this.maxElementSize = i.intValue(); + + //This is a small hack to set the max element size without calling a constructor + //dataFile.maxElementSize = this.maxElementSize; + + } + catch (IOException e) { + throw e; + } + catch (ClassNotFoundException e) { + e.printStackTrace(); + } + + indexArray = new String[arraySize+1]; + for (int i=0; i <=arraySize; i++) { + indexArray[i] = i + ""; + } + + } + + + public void add(int index, Object element) { + //Skip size check for maximum performance + //if (index > size || index < 0) { + // throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size); + //} + + DataWriter dw = new DataWriter( indexArray[index] ); + try { + dw.writeObject( element ); + } + catch (IOException e) { + e.printStackTrace(); + } + + //Check size of element against the maximum allowed record size + if (dw.getDataLength() > maxElementSize) { + throw new IllegalArgumentException( + "Size of entry="+dw.getDataLength()+" bytes is larger than maxElementSize=" + maxElementSize); + } + + try { + //Update record if it exists already + if (dataFile.recordExists(indexArray[index])) { + dataFile.updateRecord( dw ); + } + else { + dataFile.insertRecord( dw ); + } + } + catch (Exception e) { + e.printStackTrace(); + } + } + + + /** + * Returns an element from the array. + * + * @param index array index + */ + public Object get(int index) { + //Skip size check for maximum performance + //if (index > size || index < 0) { + // throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size); + //} + + DataReader dr = null; + Object obj = null; + try { + dr = dataFile.readRecord(indexArray[index]); + obj = dr.readObject(); + } + catch (Exception e) { + e.printStackTrace(); + } + + return obj; + } + + + /** + * Removes an element from the array. Can be an expensive operation so only + * delete when necessary. + * + * @param index array index + */ + public void remove(int index) { + try { + dataFile.deleteRecord( indexArray[index] ); + } + catch (IOException e) { + e.printStackTrace(); + } + + } + + + public String getFileName() { + return fileName; + } + + + public int getArraySize() { + return arraySize; + } + + + public int getMaxElementSize() { + return maxElementSize; + } + + /** + * Closes the underlying file. + * + */ + public void close() { + try { + dataFile.close(); + } + catch (IOException e) { + e.printStackTrace(); + } + } +} + diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java new file mode 100644 index 0000000..f79a397 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java @@ -0,0 +1,692 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +import jxl.CellType; +import jxl.NumberCell; +import jxl.NumberFormulaCell; +import jxl.Workbook; +import jxl.Cell; +import jxl.Sheet; + +import org.apache.log4j.Logger; + + +/** + * Creates a TableData class from an Excel file. + * + * @author Joel Freedman + * @author John Abraham + * @version 1.1, 10/10/2007 + * + * New version checks for cell types and throws an error + * if it can't get a number value from a cell when it thinks + * it should (J. Abraham, Sept-Oct 2007) + */ +public class ExcelFileReader extends TableDataFileReader implements DataTypes { + + /** + * + */ + private static final long serialVersionUID = 1L; + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + //These attributes are initialized on each call to readFile() + private int columnCount; + private int rowCount; + private List columnData; + private ArrayList columnLabels; + private int[] columnType; + private String worksheetName; + private File workbookFile=null; + + + public ExcelFileReader () { + } + + /** + * Set the name of the worksheet to read in this excel workbook. + * @param name THe name of the worksheet to read. + */ + public void setWorksheetName(String name){ + worksheetName = name; + } + + public TableDataSet readFile(File file) throws IOException { + + if(worksheetName == null){ + logger.fatal("Error: must set worksheet to read using setWorksheetName method before reading"); + throw new RuntimeException(); + } + + return readFile(file, worksheetName, true); + } + + + /** + * Convenience method to load a Excel file into a table data class. + * + * @param file name of file to read + * @param columnLabelsPresent determines whether first line is treated + * as column titles + * @throws IOException + * + */ + public TableDataSet readFile(File file, String worksheetName, boolean columnLabelsPresent) throws IOException { + return readFile(file, worksheetName, columnLabelsPresent, null); + } + + + /** + * Convenience method to load a Excel file into a table data class. + * + * @param file name of file to read + * @param columnsToRead list of column labels that should be read - all other + * columns will be dropped from the table data set + * @throws IOException + * + */ + public TableDataSet readFile(File file, String worksheetName, String[] columnsToRead) throws IOException { + return readFile(file, worksheetName,true, columnsToRead); + } + + + /** + * Main method which loads an Excel file into a table data class. + * + * @param file name of file to read + * @param columnLabelsPresent determines whether first line is treated + * as column titles + * @param columnsToRead list of column labels that should be read - all other + * columns will be dropped from the table data set + * @throws IOException + * + */ + public TableDataSet readFile(File file, String worksheetName, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { + + if ((columnsToRead != null) && (columnLabelsPresent == false)) { + throw new RuntimeException("Column lables provided as filter but there are no column labels in Excel file"); + } + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + Workbook workbook = null; + //open workbook + try { + workbook = Workbook.getWorkbook( file); + } + catch (Throwable t) { + logger.error("Error attemting to open excel file " + file); + t.printStackTrace(); + } + + Sheet worksheet = workbook.getSheet(worksheetName); + if (worksheet==null) return null; + boolean[] readColumnFlag = null; + + if (columnLabelsPresent) { + readColumnFlag = readColumnLabels(worksheet, columnsToRead); + } + readData(worksheet, columnLabelsPresent, readColumnFlag); + + TableDataSet tds = makeTableDataSet(); + tds.setName(file.toString()); + return tds; + } + + + /** + * Read the excel file with a String[] of specified column formats (NUMBER or STRING), + * where the format is specified for all columns, all columns are read, + * and column headings must be present on the first line. + * + * @param file File object of excel workbook + * @param worksheetName the name of the worksheet to read + * @param columnFormats An array of column formats. + * + * @return A tableDataSet object containing the data in the worksheet. + */ + public TableDataSet readFileWithFormats(File file, String worksheetName, String[] columnFormats) throws IOException { + + boolean columnLabelsPresent = true; + String[] columnsToRead = null; + + if ((columnsToRead != null) && (columnLabelsPresent == false)) { + throw new RuntimeException("Column lables provided as filter but there are no column labels in Excel file"); + } + + //Initialize class attributes + columnCount = 0; + rowCount = 0; + columnData = new ArrayList(); + columnLabels = new ArrayList(); + columnType = null; + + Workbook workbook = openFile(file); + Sheet worksheet = workbook.getSheet(worksheetName); + + boolean[] readColumnFlag = null; + + if (columnLabelsPresent) { + readColumnFlag = readColumnLabels(worksheet, columnsToRead); + } + + readData( worksheet, columnLabelsPresent, readColumnFlag, columnFormats); + + TableDataSet tds = makeTableDataSet(); + tds.setName(file.toString()); + return tds; + } + + + /** + * Open file method. + * @param file The file. + * @return The workbook + * @throws IOException + */ + private Workbook openFile(File file) throws IOException { + logger.debug("Opening excel file: "+file); + + Workbook workbook = null; + //open workbook + try { + workbook = Workbook.getWorkbook( file ); + } + catch (Throwable t) { + logger.error("Error attemting to open excel file " + file); + t.printStackTrace(); + } + return workbook; + } + + /** + * Read and parse the column titles from the first line of file. + */ + private boolean[] readColumnLabels(Sheet worksheet, String[] columnsToRead) + throws IOException { + + + //Read the first cell + Cell cell = worksheet.getCell(0,0); + + //Test for an empty file + if (cell.getContents().length()==0) { + throw new IOException("Error: first row in sheet looks like it's empty"); + } + + int count = countNumberOfColumns(worksheet); + + boolean[] readColumnFlag = new boolean[count]; + + //Initialize the readColumnFlag to false if the caller has supplied a + //list of columns. It will be turned to true based on a comparison of the + //column labels found in the file. Otherwise initialize to true. + for (int i=0; i < count; i++) { + if (columnsToRead != null) + readColumnFlag[i] = false; + else + readColumnFlag[i] = true; + } + + //Read column titles + int c = 0; + for (int i=0; i < count; i++) { + cell = worksheet.getCell(i, 0); + String column_name = cell.getContents(); + + //Check if column should be read based on list supplied by caller + if (columnsToRead != null) { + for (int j=0; j < columnsToRead.length; j++) { + if (columnsToRead[j].equalsIgnoreCase(column_name)) { + readColumnFlag[c] = true; + + columnLabels.add(column_name); + columnCount++; + break; + } + } + } + else { + columnLabels.add(column_name); + columnCount++; + } + c++; //the actual columnn number in the file being read + } + + //Debugging output + String msg = "column read flag = "; + for (int i=0; i < readColumnFlag.length; i++) { + if (readColumnFlag[i] == true) + msg += "true"; + else + msg += "false"; + if (i < (readColumnFlag.length-1)) + msg += ", "; + } + msg += "\n"; + logger.debug(msg); + + + return readColumnFlag; + } + + + /** + * Read and parse data portion of file. + */ + private void readData(Sheet worksheet, boolean columnLabelsPresent, + boolean[] readColumnFlag) + throws IOException { + + //Determine the number of lines in the file + rowCount = countNumberOfRows(worksheet); + + logger.debug("number of rows in file: " + rowCount); + if (columnLabelsPresent) { + rowCount--; + } + + //Process each line in the file + if (rowCount == 0) { + columnType = new int[columnCount]; + readColumnFlag = new boolean[columnCount]; + for (int col =0; col 0) { + inStream.readLine(); + } + + // read the data + ArrayList[] columnData = readData(inStream, dictionary); + inStream.close(); + + // make the table + TableDataSet tds = makeTableDataSet(columnData, dictionary); + tds.setName(file.toString()); + + return tds; + } + + /** + * Opens the file, and creates a buffered file reader to that file. + * + * @param file The file to open. + * @return Reader for the file + * @throws IOException + */ + private BufferedReader openFile(File file) throws IOException { + logger.debug("Opening file: "+file); + + BufferedReader inStream = null; + try { + inStream = new BufferedReader( new FileReader(file) ); + } + catch (IOException e) { + throw e; + } + + return inStream; + } + + /** + * Read and parse data portion of file. + * + * @param inStream The stream of input data. + * @param dictionary The dictionary defining how the columns are set up. + * + * @return An ArrayList of data for each column read. + * + * @throws IOException + */ + private ArrayList[] readData(BufferedReader inStream, TableDataSet dictionary) throws IOException { + + // set up the start, end and type arrays + int start[] = new int[dictionary.getRowCount()]; + int end[] = new int[dictionary.getRowCount()]; + String type[] = new String[dictionary.getRowCount()]; + for (int i=0; i(); + } + if (type[i].equals("STRING")) { + columnData[i] = new ArrayList(); + } + } + + //Process each line in the file + String line; + int lineNum = 1; + while ((line = inStream.readLine()) != null) { + String[] lineData = new String[dictionary.getRowCount()]; + for (int i=0; i end) { + throw new RuntimeException("Start greater than end position: "+start+">"+end); + } + + String type = dictionary.getStringValueAt(i, "TYPE"); + if (!type.equals("NUMBER") && !type.equalsIgnoreCase("STRING")) { + throw new RuntimeException("Column type must be NUMBER or STRING."); + } + + int labelInFile = (int) dictionary.getValueAt(i, "LABELINFILE"); + if (labelInFile!=0 && labelInFile!=1) { + throw new RuntimeException("LABELINFILE must be 0 or 1."); + } + } + + return dictionary; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataReader#readTable(java.lang.String) + */ + public TableDataSet readTable(String tableName) throws IOException { + File fileName = new File (getMyDirectory() + File.separator + tableName + ".dat"); + TableDataSet me= readFile(fileName); + me.setName(tableName); + return me; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataReader#close() + */ + public void close() { + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java new file mode 100644 index 0000000..14dc2bb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java @@ -0,0 +1,53 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.text.*; + +/** + * This class provides a way to use scientific notation when necessary. + */ +public class GeneralDecimalFormat extends DecimalFormat { + final DecimalFormat nonScientific; + double useScientificAbove; + double useScientificBelow; + + public GeneralDecimalFormat(String scientificString, double useScientificAbove, double useScientificBelow) { + super(scientificString); + this.useScientificAbove = useScientificAbove; + this.useScientificBelow = useScientificBelow; + String nonScientificString = scientificString.replaceAll("E0+",""); + nonScientific = new DecimalFormat(nonScientificString); + } + + public StringBuffer format(double val, StringBuffer buffer, FieldPosition f) { + if (val ==0.0 || (Math.abs(val) <= useScientificAbove && Math.abs(val) > useScientificBelow)) { + return nonScientific.format(val, buffer, f); + } else { + return super.format(val,buffer,f); + } + } + + public StringBuffer format(long val, StringBuffer buffer, FieldPosition f) { + if (val ==0.0 || (Math.abs(val) <= useScientificAbove && Math.abs(val) > useScientificBelow)) { + return nonScientific.format(val, buffer, f); + } else { + return super.format(val,buffer,f); + } + } + +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java new file mode 100644 index 0000000..a010bfc --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java @@ -0,0 +1,221 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.IOException; +import java.sql.*; +import java.util.ArrayList; +import java.util.List; +import org.apache.log4j.Logger; + +import com.pb.common.sql.JDBCConnection; +import com.pb.common.sql.SQLExecute; + +/** + * Creates a TableData class from a table in a JDBC data source. + * + * @author Tim Heier + * @version 1.0, 2/07/2004 + * + */ +public class JDBCTableReader extends TableDataReader { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + private JDBCConnection jdbcConn = null; + + private boolean mangleTableNamesForExcel = false; + + + public JDBCTableReader (JDBCConnection jdbcConn) { + if (jdbcConn == null) { + throw new RuntimeException("Database connection is null"); + } + + this.jdbcConn = jdbcConn; + } + + + /** + * Load all columns and rows of table. + */ + public TableDataSet readTable(String tableName) throws IOException { + String mangledTableName = tableName; + if (mangleTableNamesForExcel) { + logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); + mangledTableName = "["+tableName+"$]"; + } + TableDataSet theTable = null; + try { + theTable = loadTable(tableName, "SELECT * FROM " + mangledTableName); + theTable.setName(tableName); + } catch (RuntimeException e) { + logger.warn("Table "+tableName+" can not be read by JDBCTableReader, "+e.toString()); + // want to return null object if table doesn't exist, to be consistent with other table readers. + theTable = null; + } + return theTable; + } + + + /** + * Load table using specified query string. + */ + private TableDataSet loadTable(String tableName, String sqlString) throws IOException { + List columnData = new ArrayList(); + String[] columnLabels; + int columnCount; + + logger.debug("JDBCTableReader, table name: " + tableName); + logger.debug("JDBCTableReader, SQL String: " + sqlString); + + if (mangleTableNamesForExcel) { + logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); + tableName = "["+tableName+"$]"; + } + + Statement stmt = null; + ResultSet rs = null; + ResultSetMetaData metaData = null; + + SQLExecute sqlExecute = new SQLExecute(jdbcConn); + try { + int rowCount = sqlExecute.getRowCount(tableName); + + //Run main query + rs = sqlExecute.executeQuery(sqlString); + metaData = rs.getMetaData(); + + columnCount = sqlExecute.getColumnCount(); + columnLabels = sqlExecute.getColumnLabels(); + int[] columnType = new int[columnCount]; + + //Set up a vector of arrays to hold the result set. Store the + //column type at the same time. Each vector holds a column of data. + for (int c = 0; c < columnCount; c++) { + int type = metaData.getColumnType(c+1); + + switch(type) { + //Map these types to STRING + case Types.CHAR: + case Types.VARCHAR: + case Types.LONGVARCHAR: + case Types.DATE: + case Types.TIME: + case Types.TIMESTAMP: + case Types.BIT: + columnData.add(new String[rowCount]); + columnType[c] = DataTypes.STRING; + break; + //Map these types to NUMBER + case Types.TINYINT: + case Types.SMALLINT: + case Types.INTEGER: + case Types.BIGINT: + case Types.FLOAT: + case Types.REAL: + case Types.DOUBLE: + case Types.DECIMAL: + case Types.NUMERIC: + columnData.add(new float[rowCount]); + columnType[c] = DataTypes.NUMBER; + break; + default: + System.err.println("**error** unknown column data type, column=" + c + + ", type=" + type); + break; + } + } + + //Read result set and store the data into column-wise arrays. + //Column data in a ResultSet starts in 1,2... + int row = 0; + while (rs.next()) { + for (int c=0; c < columnCount; c++) { + int type = columnType[c]; + + switch(type) { + case DataTypes.STRING: + String[] s = (String[]) columnData.get(c); + s[row] = rs.getString(c+1); + break; + case DataTypes.NUMBER: + float[] f = (float[]) columnData.get(c); + f[row] = rs.getFloat(c+1); + break; + default: + System.err.println("**error** unknown column data type - should not be here"); + break; + } + } + row++; + } + } + catch (SQLException e) { + throw new IOException(e.getMessage()); + } +// finally { +// try { +// rs.close(); +// stmt.close(); +// } +// catch (SQLException e) { +// e.printStackTrace(); +// } +// } + + TableDataSet tds = makeTableDataSet(columnData, columnLabels, columnCount); + tds.setName(tableName); + return tds; + } + + + private TableDataSet makeTableDataSet(List columnData, String[] columnLabels, int columnCount) { + + TableDataSet table = new TableDataSet(); + + for (int i=0; i < columnCount; i++) { + table.appendColumn(columnData.get(i), columnLabels[i]); + } + + return table; + } + + + /** + * @return Returns the mangleTableNamesForExcel. + */ + public boolean isMangleTableNamesForExcel() { + return mangleTableNamesForExcel; + } + + /** + * @param mangleTableNamesForExcel The mangleTableNamesForExcel to set. + */ + public void setMangleTableNamesForExcel(boolean mangleTableNamesForExcel) { + this.mangleTableNamesForExcel = mangleTableNamesForExcel; + } + + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataReader#close() + */ + public void close() { + jdbcConn.close(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java new file mode 100644 index 0000000..7c4a487 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java @@ -0,0 +1,136 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.IOException; +import org.apache.log4j.Logger; + +import com.pb.common.sql.JDBCConnection; +import com.pb.common.sql.SQLExecute; + +/** + * @author jabraham + * + * This class writes TableDataSets to an SQL Database + */ +public class JDBCTableWriter extends TableDataWriter { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); + + private JDBCConnection jdbcConn = null; + + private boolean mangleTableNamesForExcel = false; + + + + public void writeTable(TableDataSet tableData, String tableName) + throws IOException { + String insertMangledTableName = tableName; + if (mangleTableNamesForExcel) { + logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); + insertMangledTableName = "["+tableName+"$]"; + } + String createMangledTableName = tableName; + if (mangleTableNamesForExcel) { + logger.debug("Mangling table name "+tableName+" to ["+tableName+"]"); + createMangledTableName = "["+tableName+"]"; + } + saveTable(createMangledTableName, insertMangledTableName, tableData); + } + + + /** + * @param mangledTableName + * @param tableData + */ + private void saveTable(String createMangledTableName, String insertMangledTableName, TableDataSet tableData) { + + //TODO: First we need to check to see if the table already exists and then delete it. + + // first create the table + StringBuffer createStatement = new StringBuffer("CREATE TABLE "+createMangledTableName+ " ("); + int[] columnTypes = tableData.getColumnType(); + for (int c =0; c nCols)) { + String msg = "Column number out of range: " + column; + msg += (", number of columns: " + nCols); + + throw new RuntimeException(msg); + } + + if (columnType[column] != type) { + throw new RuntimeException("column " + column + " is type " + + columnType[column] + " not type " + type); + } + } + + /** + * Return the name of the column given a postion. Column numbers are 1-based. + * + * @return The name of the column or an empty string if the requested + * field is out of bounds. + */ + public String getColumnLabel(int column) { + //Data is 0 based so subtract 1 from what the user supplies + column = column - 1; + + if ((columnLabels != null) && (column >= 0) && + (column < columnLabels.size())) { + return (String) columnLabels.get(column); + } else { + return ""; + } + } + + /** + * Return the postion of a column given the name. Column numbers are 1-based. + * + * @return -1 if the requested column name is not found. + * + */ + public int getColumnPosition(String columnName) { + int position = -1; + + for (int col = 0; col < columnLabels.size(); col++) { + String currentColumn = (String) columnLabels.get(col); + if (currentColumn.equalsIgnoreCase(columnName)) { + position = col + 1; + break; + } + } + + return position; + } + + public int checkColumnPosition(String columnName) throws RuntimeException { + int position = getColumnPosition(columnName); + if (position <0) throw new RuntimeException("Column "+columnName+" does not exist in TableDataSet "+getName()); + return position; + } + + /** + * Indicates whether or not the table constains the specified column. + * + * @param columnName Name of the column to check. + * @return boolean indicating if the column is present. + */ + public boolean containsColumn(String columnName) { + int position = getColumnPosition(columnName); + if (position>=0) return true; + else return false; + } + + /** + * Return the values in a specified row as a float[] + * + * @param row row number to retrieve, values are 0-based + * + * @throws RuntimeException when one of the columns is of type STRING + * + */ + public float[] getRowValues(int row) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + + int startPosition = 0; //position to start in array + + float[] rowValues = new float[nCols + startPosition]; + + for (int c = 0; c < nCols; c++) { + if (columnType[c] == STRING) { + throw new RuntimeException("column " + c + 1 + + " is of type STRING"); + } + + float[] f = (float[]) columnData.get(c); + + rowValues[c + startPosition] = f[row]; + } + + return rowValues; + } + + /** + * Return a values from a specified row using the indexed column. + * + * @param row indexed row number + */ + public float[] getIndexedRowValuesAt(int row) { + if (columnIndex == null) { + throw new RuntimeException("No index defined."); + } + + row = columnIndex[row] + 1; // getRowValues will subtract 1 + + return getRowValues(row); + } + + /** + * Return the row number from the index. + * + * @param index indexed row number + * @return the row number associated with the index + */ + public int getIndexedRowNumber(int index) { + if (columnIndex == null) { + throw new RuntimeException("No index defined."); + } + + return columnIndex[index] + 1; // getRowValues will subtract 1 + } + + + /** + * Return the values in a specified row as a String[] + * + * @param row row number to retrieve, values are 0-based + * + * @throws RuntimeException when one of the columns is of type STRING + * + */ + public String[] getRowValuesAsString(int row) { + //TODO lookup based on indexed column + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + + int startPosition = 0; //position to start in array + + String[] rowValues = new String[nCols + startPosition]; + + for (int c = 0; c < nCols; c++) { + switch (columnType[c]) { + case STRING: + String[] s = (String[]) columnData.get(c); + rowValues[c + startPosition] = s[row]; + break; + case NUMBER: + float[] f = (float[]) columnData.get(c); + rowValues[c + startPosition] = valueFormat.format(f[row]); + break; + } + } + + return rowValues; + } + + /** + * Returns a copy of the values in the table as a float[][] + */ + public float[][] getValues() { + float[][] tableValues = new float[nRows][nCols]; + + for (int c = 0; c < nCols; c++) { + if (columnType[c] == STRING) { + throw new RuntimeException("column " + c + 1 + + " is of type STRING"); + } + + float[] f = (float[]) columnData.get(c); + + for (int r = 0; r < nRows; r++) { + tableValues[r][c] = f[r]; + } + } + + return tableValues; + } + + /** + * Return a value from a specified row and column. For speed, the column + * type is not checked. A RuntimeException will be thrown if the column is + * of type NUMBER. + * + */ + public float getValueAt(int row, int column) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + column = column - 1; + + + //TODO lookup based on indexed column + float[] f = null; + try { + f = (float[]) columnData.get(column); + } catch (ClassCastException e) { + throw new RuntimeException("Column "+column+" in TableDataSet is not float values",e); + } + + return f[row]; + } + + public float getValueAt(int row, String columnName) { + int columnNumber = getColumnPosition(columnName); + + if (columnNumber <= 0) { + logger.error("no column named " + columnName + " in TableDataSet"); + throw new RuntimeException("no column named " + columnName + + " in TableDataSet"); + } + + return getValueAt(row, columnNumber); + } + + /** + * Return a value from a specified row and column. If the column type is + * not STRING then the numeric value will be converted to string before + * it is returned. + * + */ + public String getStringValueAt(int row, int column) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + column = column - 1; + + String value; + + if (columnType[column] == NUMBER) { + float[] f = (float[]) columnData.get(column); + value = valueFormat.format(f[row]); + } else { + String[] s = (String[]) columnData.get(column); + value = s[row]; + } + + return value; + } + + public boolean getBooleanValueAt(int row, String columnName) { + return getBooleanValueAt(row,checkColumnPosition(columnName)); + } + /** + * Return a value from a specified row and column. + * + */ + public boolean getBooleanValueAt(int row, int column) { + String boolString = getStringValueAt(row,column); + boolString = boolString.trim(); + if (boolString == null) throw new RuntimeException("Boolean value in TableDataSet "+name+" is blank (null)"); + if (use1sAnd0sForTrueFalse ) { + if (boolString.equalsIgnoreCase("1")) return true; + if (boolString.equalsIgnoreCase("0")) return false; + } + /* ABDEL M.:I added boolString.equalsIgnoreCase("t") and boolString.equalsIgnoreCase("f") to the condition below. + The method getStringValueAt(row,column) above returns t or f rather than true or false. It is not the best way to fix the issue but this is a work around + for now till John A. comes and check it. + */ + if (boolString.equalsIgnoreCase("true") || boolString.equalsIgnoreCase("t")) return true; + if (boolString.equalsIgnoreCase("false") || boolString.equalsIgnoreCase("f")) return false; + throw new RuntimeException("Boolean value in table dataset "+name+" column "+ column+ " is neither true nor false, but ('"+boolString+"')."); + } + + public void setBooleanValueAt(int row, String columnName, boolean value) { + setBooleanValueAt(row,checkColumnPosition(columnName), value); + } + + public void setBooleanValueAt(int row, int column, boolean value) { + if (value) setStringValueAt(row,column,"true"); + else setStringValueAt(row,column,"false"); + } + + /** + * Return a value from a specified row and column. For speed, the column + * type is not checked. A RuntimeException will be thrown if the column is + * of type STRING. + * + */ + public String getStringValueAt(int row, String columnName) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + + int columnNumber = getColumnPosition(columnName); + + if (columnNumber <= 0) { + logger.error("no column named " + columnName + " in TableDataSet"); + + throw new RuntimeException("no column named " + columnName + + " in TableDataSet"); + } + + //Call with 1-based row and column numbers + return getStringValueAt(row + 1, columnNumber); + } + + /** + * Set at a value using the column name. + */ + public void setValueAt(int row, String colName, float newValue) { + int col = getColumnPosition(colName); + setValueAt(row, col, newValue); + } + + /** + * Return a value from a specified row and column. For speed, the column + * type is not checked. A RuntimeException will be thrown if the column is + * of type NUMBER. + * + */ + public void setValueAt(int row, int column, float newValue) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + column = column - 1; + + float[] f = (float[]) columnData.get(column); + + f[row] = newValue; + + // any TableDataSetIndex that uses this column will have to regenerate its index. + if (indexColumns[column]==true) { + fireIndexValuesChanged(); + } + setDirty(true); + } + + /** + * update the column specified with the int values specified. + * + */ + public void setColumnAsInt ( int column, int[] newValues ) { + //Data is 0 based so subtract 1 from what the user supplies + column = column - 1; + + float[] f = new float[newValues.length]; + for (int i=0; i < newValues.length; i++) + f[i] = (float)newValues[i]; + + columnData.set( column, f ); + if (indexColumns[column]==true) { + fireIndexValuesChanged(); + } + + setDirty(true); + } + + /** + * update the column specified with the float values specified. + * + */ + public void setColumnAsFloat ( int column, float[] newValues ) { + //Data is 0 based so subtract 1 from what the user supplies + column = column - 1; + + columnData.set( column, newValues ); + if (indexColumns[column]==true) { + fireIndexValuesChanged(); + } + + setDirty(true); + } + + /** + * update the column specified with the double values specified. + * + */ + public void setColumnAsDouble ( int column, double[] newValues ) { + //Data is 0 based so subtract 1 from what the user supplies + column = column - 1; + + float[] f = new float[newValues.length]; + for (int i=0; i < newValues.length; i++) + f[i] = (float)newValues[i]; + + columnData.set( column, f ); + if (indexColumns[column]==true) { + fireIndexValuesChanged(); + } + + setDirty(true); + } + + public void setIndexedValueAt(int row, String colName, float newValue) { + if (columnIndex == null) { + throw new RuntimeException("No index defined."); + } + row = columnIndex[row] + 1; + + int col = getColumnPosition(colName); + setValueAt(row, col, newValue); + } + + /** + * Return a value from an indexed row and column. For speed, the column + * type is not checked. A RuntimeException will be thrown if the column is + * of type NUMBER. + * + */ + public void setIndexedValueAt(int row, int column, float newValue) { + if (columnIndex == null) { + throw new RuntimeException("No index defined."); + } + + row = columnIndex[row] + 1; // getRowValues will subtract 1 + setValueAt(row, column, newValue); + } + + /** + * Return a value from a specified row and column. For speed, the column + * type is not checked. A RuntimeException will be thrown if the column is + * of type STRING. + * + */ + public void setStringValueAt(int row, int column, String newValue) { + //Data is 0 based so subtract 1 from what the user supplies + row = row - 1; + column = column - 1; + + //TODO lookup based on indexed column + String[] s = (String[]) columnData.get(column); + + s[row] = newValue; + setDirty(true); + if (indexColumns[column]==true) { + fireIndexValuesChanged(); + } + if (column == stringIndexColumn) + stringIndexDirty = true; + } + + /** + * Set the value of a {@code STRING} column at a specified row. + * + * @param row + * The (1-based) row number. + * + * @param column + * The name of the column. + * + * @param value + * The value to place in {@code column} at {@code row}. + * + * @throws ClassCastException if {@code column} is not a {@code STRING} column. + * @throws RuntimeException if {@code column} is not found in this table. + */ + public void setStringValueAt(int row, String column, String value) { + setStringValueAt(row,checkColumnPosition(column),value); + } + + public void setIndexColumnNames(String[] indexColumnNames) { + for (int i=0;i= 0 ) + continue; + + logger.info("Adding column " + tdsInHeadings[c] + " to TableDataSet due to merge"); + if ( tdsInTypes[c] == NUMBER ) { + float[] newColumn = new float[nRows]; + + for (int r = 0; r < nRows; r++) + newColumn[r] = tdsIn.getValueAt( r+1, c+1 ); + + appendColumn( newColumn, tdsInHeadings[c] ); + } + else if ( tdsInTypes[c] == STRING ) { + String[] newColumn = new String[nRows]; + + for (int r = 0; r < nRows; r++) + newColumn[r] = tdsIn.getStringValueAt( r+1, c+1 ); + + appendColumn( newColumn, tdsInHeadings[c] ); + } + } + setDirty(true); + + } + + /** + * Static method to log the frequency of values in + * a column specified by the user. + * + * @param tableName String identifying contents of TableDataSet + * @param tds containing column for creating frequency table + * @param columnPosition position of desired column + * + */ + public static void logColumnFreqReport(String tableName, TableDataSet tds, + int columnPosition) { + if (tds.getRowCount() == 0) { + logger.info(tableName + " Table is empty - no data to summarize"); + + return; + } + + float[] columnData = new float[tds.getRowCount()]; + int[] sortValues = new int[tds.getRowCount()]; + + for (int r = 1; r <= tds.getRowCount(); r++) { + columnData[r - 1] = tds.getValueAt(r, columnPosition); + sortValues[r - 1] = (int) (columnData[r - 1] * 10000); + } + + // sort the column elements + int[] index = IndexSort.indexSort(sortValues); + + ArrayList bucketValues = new ArrayList(); + ArrayList bucketSizes = new ArrayList(); + + // count the number of identical elements into buckets + float oldValue = columnData[index[0]]; + int count = 1; + + for (int r = 1; r < tds.getRowCount(); r++) { + if (columnData[index[r]] > oldValue) { + bucketValues.add(Float.toString(oldValue)); + bucketSizes.add(Integer.toString(count)); + count = 0; + oldValue = columnData[index[r]]; + } + + count++; + } + + bucketValues.add(Float.toString(oldValue)); + bucketSizes.add(Integer.toString(count)); + + // print a simple summary table + logger.info("Frequency Report table: " + tableName); + logger.info("Frequency for column " + columnPosition + ": " + + (tds.getColumnLabel(columnPosition))); + logger.info(String.format("%8s", "Value") + + String.format("%11s", "Frequency")); + + int total = 0; + + for (int i = 0; i < bucketValues.size(); i++) { + float value = Float.parseFloat((String) (bucketValues.get(i))); + logger.info(String.format("%8.0f", value) + + String.format("%11d", Integer.parseInt((String) (bucketSizes.get(i))))); + total += Integer.parseInt((String) (bucketSizes.get(i))); + } + + logger.info(String.format("%8s", "Total") + + String.format("%11d\n\n\n", total)); + } + + /** + * Logs the frequency of values in a column specified by the user. + * The array list argument can hold descriptions of the values + * if they are known. For example if a column lists alternatives 1-5, you + * might know that alt 1= 0_autos, alt2=1_auto, etc. + * + * @param tableName String identifying contents of TableDataSet + * @param tds containing column for creating frequency table + * @param columnPosition position of desired column + * + */ + public static void logColumnFreqReport(String tableName, TableDataSet tds, + int columnPosition, String[] descriptions) { + if (tds.getRowCount() == 0) { + logger.info(tableName + " Table is empty - no data to summarize"); + + return; + } + + float[] columnData = new float[tds.getRowCount()]; + int[] sortValues = new int[tds.getRowCount()]; + + for (int r = 1; r <= tds.getRowCount(); r++) { + columnData[r - 1] = tds.getValueAt(r, columnPosition); + sortValues[r - 1] = (int) (columnData[r - 1] * 10000); + } + + // sort the column elements + int[] index = IndexSort.indexSort(sortValues); + + ArrayList bucketValues = new ArrayList(); + ArrayList bucketSizes = new ArrayList(); + + // count the number of identical elements into buckets + float oldValue = columnData[index[0]]; + int count = 1; + + for (int r = 1; r < tds.getRowCount(); r++) { + if (columnData[index[r]] > oldValue) { + bucketValues.add(Float.toString(oldValue)); + bucketSizes.add(Integer.toString(count)); + count = 0; + oldValue = columnData[index[r]]; + } + + count++; + } + + bucketValues.add(Float.toString(oldValue)); + bucketSizes.add(Integer.toString(count)); + + // print a simple summary table + logger.info("Frequency Report table: " + tableName); + logger.info("Frequency for column " + columnPosition + ": " + + (tds.getColumnLabel(columnPosition))); + logger.info(String.format("%8s", "Value") + + String.format("%13s", "Description") + + String.format("%11s", "Frequency")); + + if(descriptions!=null) + if(bucketValues.size() != descriptions.length) + logger.fatal("The number of descriptions does not match the number of values in your data"); + + int total = 0; + + for (int i = 0; i < bucketValues.size(); i++) { + float value = Float.parseFloat((String) (bucketValues.get(i))); + String description = ""; //default value as sometime certain columns don't have descriptions + if(descriptions !=null) { + description = descriptions[i]; + } + logger.info(String.format("%8.0f", value) + " " + String.format("%-11s", description) + + String.format("%11d", Integer.parseInt((String) (bucketSizes.get(i))))); + total += Integer.parseInt((String) (bucketSizes.get(i))); + } + + logger.info(String.format("%23s", "Total") + + String.format("%9d\n\n\n", total)); + } + + + /** + * @param index + */ + public void removeChangeListener(ChangeListener index) { + changeListeners.remove(index); + } + + + public void setName(String name) { + this.name = name; + } + + + public String getName() { + return name; + } + + /* (non-Javadoc) + * @see java.lang.Object#toString() + */ + public String toString() { + return "TableDataSet "+name; + } + + + public interface TableDataSetWatcher { + public void isBeingForgotten(TableDataSet s); + public void isDirty(TableDataSet s); + } + + private ArrayList myWatchers = null; + + public void addFinalizingListener(TableDataSetWatcher watcher) { + if (myWatchers ==null) { + myWatchers = new ArrayList(); + } + myWatchers.add(watcher); + } + + + void tellWatchersImBeingForgotten() { + if (myWatchers != null) { + for (int w=0;w stringIndex = null; //map of string index keys to 1-based row numbers + private int stringIndexColumn = -1; + private boolean stringIndexDirty; + + /** + * Build a string index on the specified column. The values in the column must be unique. If the column is + * modified after the index is built, the index must be rebuilt (using this function) before it can be used again. + * + * @param column + * The column to build the index on. + * + * @throws IllegalStateException if the column contains repeated values. + * @throws RuntimeException if the column is not of type {@code STRING}. + */ + public void buildStringIndex(int column) { + checkColumnNumber(column, STRING); + column--; //changed to a zero based index + String[] columnValues = (String[]) columnData.get(column); + + stringIndex = new HashMap(columnValues.length); + Set repeatedValues = new HashSet(); + + for (int i = 0; i < columnValues.length; i++) + if (stringIndex.put(columnValues[i],i+1) != null) + repeatedValues.add(columnValues[i]); + + if (repeatedValues.size() > 0) { + //nulify the index, as it is invalid anyway, in case somebody catches this exception and tries to carry on + stringIndex = null; + stringIndexColumn = -1; + throw new IllegalStateException("String index cannot be built on a column with non-unique values." + + "The following values have been repeated: " + Arrays.toString(repeatedValues.toArray(new String[repeatedValues.size()]))); + } + + stringIndexColumn = column; + stringIndexDirty = false; + } + + private void checkStringIndexValue(String index) { + if (stringIndex == null) + throw new IllegalStateException("No string index exists for this table."); + if (stringIndexDirty) + throw new IllegalStateException("String index column changed, must be rebuilt."); + if (!stringIndex.containsKey(index)) + throw new IllegalArgumentException("String value not found in index: " + index); + } + + /** + * Get the (0-based) row number for the given string index value. + * + * @param index + * The string index value. + * + * @return the row number corresponding to index {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + */ + public int getStringIndexedRowNumber(String index) { + checkStringIndexValue(index); + return stringIndex.get(index)-1; + } + + /** + * Get the value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + * @throws RuntimeException if {@code column} does not hold {@code float}s. + */ + public float getStringIndexedValueAt(String index, int column) { + checkStringIndexValue(index); + return getValueAt(stringIndex.get(index),column); + } + + /** + * Get the value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table, or does not hold {@code float}s. + */ + public float getStringIndexedValueAt(String index, String column) { + checkStringIndexValue(index); + return getValueAt(stringIndex.get(index),column); + } + + /** + * Get the boolean value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + * @throws RuntimeException if {@code column} does not hold {@code boolean}s. + */ + public boolean getStringIndexedBooleanValueAt(String index, int column) { + checkStringIndexValue(index); + return getBooleanValueAt(stringIndex.get(index),column); + } + + /** + * Get the boolean value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table, or does not hold {@code boolean}s. + */ + public boolean getStringIndexedBooleanValueAt(String index, String column) { + checkStringIndexValue(index); + return getBooleanValueAt(stringIndex.get(index),column); + } + + /** + * Get the string value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + */ + public String getStringIndexedStringValueAt(String index, int column) { + checkStringIndexValue(index); + return getStringValueAt(stringIndex.get(index),column); + } + + /** + * Get the string value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @return the value in {@code column} at the row specified by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table. + */ + public String getStringIndexedStringValueAt(String index, String column) { + checkStringIndexValue(index); + return getStringValueAt(stringIndex.get(index),column); + } + + /** + * Set the value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + * @throws ClassCastException if {@code column} does not hold {@code float}s. + */ + public void setStringIndexedValueAt(String index, int column, float value) { + checkStringIndexValue(index); + setValueAt(stringIndex.get(index),column,value); + } + + /** + * Set the value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table. + * @throws ClassCastException if {@code column} does not hold {@code float}s. + */ + public void setStringIndexedValueAt(String index, String column, float value) { + checkStringIndexValue(index); + setValueAt(stringIndex.get(index),column,value); + } + + /** + * Set the boolean value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + * @throws ClassCastException if {@code column} does not hold {@code boolean}s. + */ + public void setStringIndexedBooleanValueAt(String index, int column, boolean value) { + checkStringIndexValue(index); + setBooleanValueAt(stringIndex.get(index),column,value); + } + + /** + * Set the boolean value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table. + * @throws ClassCastException if {@code column} does not hold {@code boolean}s. + */ + public void setStringIndexedBooleanValueAt(String index, String column, boolean value) { + checkStringIndexValue(index); + setBooleanValueAt(stringIndex.get(index),column,value); + } + + /** + * Set the string value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The (1-based) column number. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. + * @throws ClassCastException if {@code column} does not hold {@code String}s. + */ + public void setStringIndexedStringValueAt(String index, int column, String value) { + checkStringIndexValue(index); + setStringValueAt(stringIndex.get(index),column,value); + } + + /** + * Set the string value for the specified column and row (specified by string index). + * + * @param index + * The index specifying the row. + * + * @param column + * The column name. + * + * @param value + * The value to place in {@code column} at the row indexed by {@code index}. + * + * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be + * rebuilt (because the index column has been modified after the index was built). + * @throws IllegalArgumentException if {@code index} is not found in the index column. + * @throws RuntimeException if {@code column} is not found in the table. + * @throws ClassCastException if {@code column} does not hold {@code String}s. + */ + public void setStringIndexedStringValueAt(String index, String column, String value) { + checkStringIndexValue(index); + setStringValueAt(stringIndex.get(index),column,value); + } + + //The user must ensure that the keys in the HashMap correspond + //to the headers in the table and that each column in the table + //has a value in the HashMap that is of the correct type. This + //method will not do a lot of error checking or handling. + // + // + public void appendRow(HashMap rowData){ + + int type; + String[] headers = getColumnLabels(); + int columnNum = 1; + for(String header : headers){ + System.out.println("Header Value to Append: " + header); + type = getColumnType()[getColumnPosition(header)-1]; + if(type == DataTypes.NUMBER){ + float[] col = getColumnAsFloat(header); + float[] newCol = new float[col.length+1]; + System.arraycopy(col, 0, newCol, 0, col.length); + newCol[newCol.length-1] = (Float) rowData.get(header); + replaceFloatColumn(columnNum, newCol); + }else if(type == DataTypes.STRING){ + String[] col = getColumnAsString(header); + String[] newCol = new String[col.length+1]; + System.arraycopy(col, 0, newCol, 0, col.length); + newCol[newCol.length-1] = (String) rowData.get(header); + replaceStringColumn(columnNum, newCol); + } + columnNum++; + } + nRows++; + + } + + //column positions are 1-number of columns but columnData ArrayList + //is zero-based so subtract 1 from the supplied colNumber + public void replaceFloatColumn(int colNumber, float[] newData){ + columnData.remove(colNumber-1); + columnData.add(colNumber-1, newData); + } + + //column positions are 1-number of columns but columnData ArrayList + //is zero-based so subtract 1 from the supplied colNumber + public void replaceStringColumn(int colNumber, String[] newData){ + columnData.remove(colNumber-1); + columnData.add(colNumber-1, newData); + } + +} + + diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java new file mode 100644 index 0000000..1270832 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java @@ -0,0 +1,218 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.IOException; +import java.lang.ref.Reference; +import java.lang.ref.ReferenceQueue; +import java.lang.ref.SoftReference; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; + +import org.apache.log4j.Logger; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to Window - + * Preferences - Java - Code Generation - Code and Comments + */ +public class TableDataSetCacheCollection extends TableDataSetCollection implements TableDataSet.TableDataSetWatcher { + + private static Logger logger = Logger.getLogger(TableDataSetCollection.class); + private HashMap readDataSoftReferences = new HashMap(); + + private HashMap dirtyDataSetMap = new HashMap(); + + ReferenceQueue myReferenceQueue = new ReferenceQueue(); + + public TableDataSetCacheCollection(TableDataReader reader, TableDataWriter writer) { + super(reader,writer); +// Runnable cleanUp = new Runnable() { +// public void run() { +// while (true) { +// try { +// TableDataSetSoftReference fred = (TableDataSetSoftReference) myReferenceQueue.remove(); +// logger.info("TableDataSet "+fred.name+" is gone"); +// readDataSoftReferences.remove(fred.name); +// } catch (InterruptedException e) { +// throw new RuntimeException("TableDataSetCacheCollection cleanup thread is being interrupted",e); +// } +// } +// } +// }; +// Thread cleanUpThread = new Thread(cleanUp); +// cleanUpThread.start(); + } + + + + /** + * @param name + * @return the TableDataSet requested + */ + public synchronized TableDataSet getTableDataSet(String name) { + // first see if we can remove references to datasets that we don't use + TableDataSetSoftReference fred; + fred = (TableDataSetSoftReference) myReferenceQueue.poll(); + while (fred !=null) { + if (logger.isDebugEnabled()) logger.debug("Removing key "+fred.name+" from list of TableDataSets read"); + readDataSoftReferences.remove(fred.name); + fred = (TableDataSetSoftReference) myReferenceQueue.poll(); + } + SoftReference wr = (SoftReference) readDataSoftReferences.get(name); + TableDataSet theTable = null; + if (wr!=null) { + theTable = (TableDataSet) wr.get(); + } + // could be the case that the soft reference was cleared before it became dirty, check + // to see if we can find it in our dirty list. + if (theTable == null) { + theTable = (TableDataSet) dirtyDataSetMap.get(name); + if (theTable!=null) { + addTableToTempStorage(theTable); + } + } + if (theTable == null) { + try { + if (logger.isDebugEnabled()) logger.debug("reading table "+name); + theTable = getMyReader().readTable(name); + double freeMem = Runtime.getRuntime().freeMemory()/1000000.0; + if (logger.isDebugEnabled()) logger.debug("Memory is "+freeMem); + } catch (IOException e) { + e.printStackTrace(); + } + if (theTable == null) + throw new RuntimeException("Can't read in table " + name); + addTableToTempStorage(theTable); + theTable.addFinalizingListener(this); + } + return theTable; + } + + + + private void addTableToTempStorage(TableDataSet theTable) { + if (logger.isDebugEnabled()) logger.debug("Creating temporary references for "+theTable.getName()); + TableDataSetSoftReference fred = new TableDataSetSoftReference(theTable, myReferenceQueue); + readDataSoftReferences.put(theTable.getName(), fred); + } + + public synchronized void flushAndForget(TableDataSet me) { + if (me.isDirty()) { + writeTableToDisk(me); + } + //TODO remove this next line, shouldn't need to manually remove it from the SoftReferences. + readDataSoftReferences.remove(me.getName()); + dirtyDataSetMap.remove(me.getName()); + } + + + private void writeTableToDisk(TableDataSet me) { + try { + if (logger.isDebugEnabled()) logger.debug("writing table "+me.getName() + ". Table has " + me.getRowCount() + " rows"); + getMyWriter().writeTable(me, me.getName()); + me.setDirty(false); + dirtyDataSetMap.remove(me.getName()); + } catch (IOException e1) { + e1.printStackTrace(); + throw new RuntimeException("Can't write out table " + me.getName()); + } + } + + /* + * (non-Javadoc) + * + * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#flush() + */ + public synchronized void flush() { + Iterator it = dirtyDataSetMap.entrySet().iterator(); + while (it.hasNext()) { + Map.Entry e = (Map.Entry) it.next(); + TableDataSet t = (TableDataSet) e.getValue(); + if (t.isDirty()) { + writeTableToDisk(t); + } else { + // if (logger.isDebugEnabled()) logger.debug("*don't need to flush* table "+t.getName()+" as it's not dirty"); + } + } + dirtyDataSetMap.clear(); + it = readDataSoftReferences.entrySet().iterator(); + while (it.hasNext()) { + Map.Entry e = (Map.Entry) it.next(); + TableDataSet t = (TableDataSet) ((Reference) e.getValue()).get(); + if (t == null) { + if (logger.isDebugEnabled()) logger.debug("soft reference to table "+e.getKey()+" has been cleared"); + } else { + if (t.isDirty()) { + logger.error("Dirty table not in dirtyDataSetMap"); + writeTableToDisk(t); + } else { + // if (logger.isDebugEnabled()) logger.debug("*don't need to flush* table "+t.getName()+" as it's not dirty"); + } + } + } + } + + /* + * Call flush first if any changes need to be written out + * + * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#invalidate() + */ + public synchronized void invalidate() throws IOException { + //TODO should we do something to remove them from the reference queue? + readDataSoftReferences.clear(); + dirtyDataSetMap.clear(); + super.invalidate(); + } + /** + * @param aTable the TableDataSet to add to the colleciton + */ + public synchronized void addTableDataSet(TableDataSet aTable) { + addTableToTempStorage(aTable); + aTable.setDirty(true); + aTable.addFinalizingListener(this); + } + + + public synchronized void isBeingForgotten(TableDataSet t) { + if (logger.isDebugEnabled()) { + double freeMem = Runtime.getRuntime().freeMemory() / 1000000.0; + logger.debug("getting ready to forget about table " + t.getName() + " freeMem=" + + freeMem); + } + } + + public synchronized void isDirty(TableDataSet s) { + if (logger.isDebugEnabled()) logger.debug("Table "+s+" is now dirty, creating a new soft reference to it now"); + dirtyDataSetMap.put(s.getName(),s); + addTableToTempStorage(s); + } + + @Override + protected void finalize() throws Throwable { + // this could be called at the end of the run, before all of the TableDatasets have been finalized + // So we have to write out any dirty datasets. + try { + flush(); + } finally { + super.finalize(); + } + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java new file mode 100644 index 0000000..1c43810 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java @@ -0,0 +1,200 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile; + +import org.apache.log4j.Logger; + +import java.io.IOException; +import java.lang.ref.WeakReference; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Iterator; +import java.util.Map; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to Window - + * Preferences - Java - Code Generation - Code and Comments + */ +public class TableDataSetCollection { + + private static Logger logger = Logger.getLogger(TableDataSetCollection.class); + + private HashMap readTableDataSets = new HashMap(); + private ArrayList cacheOfCurrentlyUsedIndices = new ArrayList(); + private TableDataReader myReader = null; + private TableDataWriter myWriter = null; + + public TableDataSetCollection(TableDataReader reader, TableDataWriter writer) { + setMyReader(reader); + setMyWriter(writer); + } + + public synchronized TableDataSetIndex getTableDataSetIndex(String tableName, String[] stringKeyColumnNames, String[] intKeyColumnNames) { + TableDataSetIndex theIndex = null; + TableDataSet theTableDataSet = getTableDataSet(tableName); + Iterator it = cacheOfCurrentlyUsedIndices.iterator(); + while (it.hasNext() && theIndex == null) { + WeakReference r = (WeakReference) it.next(); + TableDataSetIndex anIndex = (TableDataSetIndex) r.get(); + if (anIndex != null) { + if (anIndex.getTableName().equals(tableName)) { + theIndex = anIndex; // assume they match then prove + // otherwise + if (anIndex.getStringColumnNumbers().length != stringKeyColumnNames.length + || anIndex.getIntColumnNumbers().length != intKeyColumnNames.length) { + theIndex = null; + } else { + for (int j = 0; j < stringKeyColumnNames.length; j++) { + int column = theTableDataSet.getColumnPosition(stringKeyColumnNames[j]); + if (column != anIndex.getStringColumnNumbers()[j]) { + theIndex = null; + } + } + for (int j = 0; j < intKeyColumnNames.length; j++) { + int column = theTableDataSet.getColumnPosition(intKeyColumnNames[j]); + if (column != anIndex.getIntColumnNumbers()[j]) { + theIndex = null; + } + } + } + + } + } else { + // null weak reference + it.remove(); + } + } + if (theIndex == null) { + theIndex = new TableDataSetIndex(this, tableName); + theIndex.setIndexColumns(stringKeyColumnNames, intKeyColumnNames); + cacheOfCurrentlyUsedIndices.add(new WeakReference(theIndex)); + } + return theIndex; + } + + /** + * @param name + * @return the TableDataSet requested + */ + public synchronized TableDataSet getTableDataSet(String name) { + TableDataSet theTable = (TableDataSet) readTableDataSets.get(name); + if (theTable == null) { + try { + logger.info("reading table "+name); + theTable = getMyReader().readTable(name); + } catch (IOException e) { + e.printStackTrace(); + } + if (theTable == null) { + logger.fatal("Can't read in table " + name); + throw new RuntimeException("Can't read in table " + name); + } + readTableDataSets.put(name, theTable); + } + return theTable; + } + + public synchronized void flushAndForget(TableDataSet me) { + if (me.isDirty()) { + try { + logger.info("writing table "+me.getName() + ". Table has " + me.getRowCount() + " rows"); + getMyWriter().writeTable(me, me.getName()); + } catch (IOException e1) { + e1.printStackTrace(); + throw new RuntimeException("Can't write out table " + me.getName()); + } + + } + readTableDataSets.remove(me.getName()); + } + + /* + * (non-Javadoc) + * + * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#flush() + */ + public synchronized void flush() { + Iterator it = readTableDataSets.entrySet().iterator(); + while (it.hasNext()) { + Map.Entry e = (Map.Entry) it.next(); + TableDataSet t = (TableDataSet) e.getValue(); + if (t.isDirty()) { + try { + logger.info("writing table "+t.getName() + ". Table has " + t.getRowCount() + " rows"); + getMyWriter().writeTable(t, (String) e.getKey()); + } catch (IOException e1) { + e1.printStackTrace(); + throw new RuntimeException("Can't write out table " + e.getKey()); + } + } + } + getMyWriter().close(); + + } + + /* + * Call flush first if any changes need to be written out + * + * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#invalidate() + */ + public synchronized void invalidate() throws IOException { + readTableDataSets.clear(); +// cacheOfCurrentlyUsedIndices.clear(); + Iterator it = cacheOfCurrentlyUsedIndices.iterator(); + while (it.hasNext()) { + TableDataSetIndex x = (TableDataSetIndex) ((WeakReference) it.next()).get(); + if (x!=null) x.tableDataSetShouldBeReloaded(); + else it.remove(); + } + } + + synchronized void setMyReader(TableDataReader myReader) { + this.myReader = myReader; + } + + TableDataReader getMyReader() { + return myReader; + } + + synchronized void setMyWriter(TableDataWriter myWriter) { + this.myWriter = myWriter; + } + + TableDataWriter getMyWriter() { + return myWriter; + } + + /** + * @param landInventoryTable + */ + public synchronized void addTableDataSet(TableDataSet aTable) { + readTableDataSets.put(aTable.getName(),aTable); + aTable.setDirty(true); + } + + /** + * + */ + public synchronized void close() { + myReader.close(); + myWriter.close(); + + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java new file mode 100644 index 0000000..8fbdee9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java @@ -0,0 +1,407 @@ +/* + * Created on 13-Oct-2005 + * + * Copyright 2005 JE Abraham and others + * + * 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. + * + */ +package org.sandag.cvm.common.datafile; + +import java.io.File; +import java.util.Iterator; +import java.util.TreeSet; + +public class TableDataSetCrosstabber { + + + + public TableDataSetCrosstabber() { + super(); + } + + public static void main(String args[]) { + if (args.length!=5) { + System.out.println("usage: java org.sandag.cvm.common.datafile.TableDataSetCrosstabber directory table rows columns, values"); + System.exit(1); + } + CSVFileReader myReader= new CSVFileReader(); + myReader.setMyDirectory(args[0]); + CSVFileWriter myWriter = new CSVFileWriter(); + myWriter.setMyDecimalFormat(new GeneralDecimalFormat("0.#########E0",10000000,.001)); + myWriter.setMyDirectory(new File(args[0])); + TableDataSetCollection myCollection = new TableDataSetCollection(myReader,myWriter); + TableDataSet results = crossTabDataset(myCollection,args[1],args[2],args[3],args[4]); + results.setName(args[1]+"_"+args[2]+args[3]+"_"+args[4]+"_crossTab"); + myCollection.addTableDataSet(results); + myCollection.flush(); + } + +// this will be the new way to avoid code duplication +// public static TableDataSet crossTabDataset(TableDataSetCollection aCollection, String inputTableName, String rowColumnName, String columnColumnName, String valuesColumnName) { +// String[] rowNames = new String[1]; +// rowNames[0] = rowColumnName; +// return crossTabDataset(aCollection, inputTableName, rowNames, columnColumnName, valuesColumnName); +// } + +// this is the method that should be able to handle multiple row headers + // TODO test this + public static TableDataSet crossTabDataset(TableDataSetCollection aCollection, String inputTableName, String[] rowColumnNames, String columnColumnName, String valuesColumnName) { + // preliminaries + TableDataSet inputTable = aCollection.getTableDataSet(inputTableName); + int[] columnTypes = inputTable.getColumnType(); + TableDataSet out = new TableDataSet(); + + // set up rows + TreeSet[] rowHeaders = new TreeSet[rowColumnNames.length]; + int totalOutputRows = 1; + for (int rowColumnId=0;rowColumnId < rowColumnNames.length;rowColumnId++) { + int forRows = inputTable.checkColumnPosition(rowColumnNames[rowColumnId]); + rowHeaders[rowColumnId] = new TreeSet(); + if (columnTypes[forRows-1]==TableDataSet.STRING) { + String[] rowColumn = inputTable.getColumnAsString(forRows); + for (int i=0;ivalueMode controls behaviour when multiple records match the indices + */ + private int valueMode=SINGLE_VALUE_MODE; + /** + * With SUM_MODE the value returned is the sum of all the values + * that match the indices. When a value is put it is divided by the number of + * records that match the indices + */ + public static final int SUM_MODE=1; + /** + * With AVERAGE_MODE the value returned is the average of all the values + * that match the indices. When a value is put the value is put into all of the records + * that match the indices + */ + public static final int AVERAGE_MODE=2; + /** + * With SINGLE_VALUE_MODE a runtime exception is thrown if there is + * more than one record in the table that matches the indices + */ + public static final int SINGLE_VALUE_MODE=3; + + private boolean errorOnMissingValues = false; + + public boolean isErrorOnMissingValues() { + return errorOnMissingValues; + } + + public void setErrorOnMissingValues(boolean errorOnMissingValues) { + this.errorOnMissingValues = errorOnMissingValues; + } + + public TableDataSetIndexedValue( + String tableName, + String[] stringKeyNames, + String[] intKeyNames, + String[][] stringIndexValues, + int[][] intIndexValues, + String columnName) { + + stringKeyNameValues = new String[1+stringIndexValues.length][]; + stringKeyNameValues[0] = stringKeyNames; + for (int i=0;i1 && valueMode == SINGLE_VALUE_MODE) { + throw new MultipleValueException("Multiple matching index values in SINGLE_VALUE_MODE "+this); + } + if (lastRowNumbers.length<1 && isErrorOnMissingValues()) { + throw new MissingValueException("No matching index values for "+this); + } + float sum=0; + float denominator= 0; + for (int r=0;r1 && valueMode == SINGLE_VALUE_MODE) { + throw new RuntimeException("Multiple matching index values in SINGLE_VALUE_MODE "+this); + } + float value= compositeValue; + if (valueMode == SUM_MODE) { + value = compositeValue/lastRowNumbers.length; + } + for (int r=0;r0) { + intKeyValues[r-1][i] = Integer.valueOf(intKeyNameValues[r][i]).intValue(); + } else { + intKeyValues[r-1][i]=0; + } + } catch (java.lang.NumberFormatException e) { + lastExceptionFound = e;; + } + + } + } + } + + public void setMyFieldName(String myFieldName) { + this.myFieldName = myFieldName; + //myLastCollection = null; + //myLastIndex = null; + lastDataColumnNumber = -1; + } + + /** + * To manually set the data column number. You have to know what column number + * has the data you are interested in; if you don't use setMyFieldName instead. + * @param myDataColumnNumber + */ + public void setMyDataColumn(String dataColumnName, int dataColumnNumber) { + myFieldName = dataColumnName; + lastDataColumnNumber = dataColumnNumber; + } + + public String getMyFieldName() { + return myFieldName; + } + + + public String toString() { + StringBuffer myInfo = new StringBuffer(); + myInfo.append(getMyTableName()); + myInfo.append(" "); + myInfo.append(getMyFieldName()); + myInfo.append(" ("); + for (int s=0;s4) myInfo.append("..."+(stringKeyNameValues.length-4)+" more..."); + myInfo.append(" "); + } + for (int i=0;i4) myInfo.append("..."+(intKeyNameValues.length-4)+" more..."); + myInfo.append(" "); + } + myInfo.append(")"); + return myInfo.toString(); + } + /** + * @return Returns the intKeyNameValues. + */ + public String[][] getIntKeyNameValues() { + return intKeyNameValues; + } + + /** + * @param intKeyNameValues The intKeyNameValues to set. + */ + public void setIntKeyNameValues(String[][] intKeyNameValues) { + this.intKeyNameValues = intKeyNameValues; + updateIntKeys(); + myLastCollection= null; + myLastIndex = null; + } + + /** + * @return Returns the stringKeyNameValues. + */ + public String[][] getStringKeyNameValues() { + return stringKeyNameValues; + } + + /** + * @param stringKeyNameValues The stringKeyNameValues to set. + */ + public void setStringKeyNameValues(String[][] stringKeyNameValues) { + this.stringKeyNameValues = stringKeyNameValues; + myLastCollection= null; + myLastIndex = null; + } + + public void setValueMode(int valueMode) { + this.valueMode = valueMode; + } + + public int getValueMode() { + return valueMode; + } + + /* (non-Javadoc) + * @see org.sandag.cvm.common.datafile.TableDataSetIndex.ChangeListener#indexChanged() + */ + public void indexChanged(TableDataSetIndex r) { + lastRowNumbers=null; + } + + /** + * @param newFieldName + * @param newFieldValues + */ + public void addNewStringKey(String newFieldName, String[] newFieldValues) { + if (newFieldValues.length >0) { + String[][] oldKeyValues = stringKeyNameValues; + + // number of permutations and combinations + int numRows = (oldKeyValues.length-1)*newFieldValues.length; + stringKeyNameValues = new String[numRows+1][]; + stringKeyNameValues[0] = new String[oldKeyValues[0].length+1]; + System.arraycopy(oldKeyValues[0],0,stringKeyNameValues[0],0,oldKeyValues[0].length); + stringKeyNameValues[0][oldKeyValues[0].length] = newFieldName; + + //also need to make more int key rows + String[][] oldIntKeyNameValues = intKeyNameValues; + intKeyNameValues = new String[numRows+1][]; + intKeyNameValues[0] = oldIntKeyNameValues[0]; + + for (int originalRow=1;originalRow0) { + String[][] oldKeyValues = intKeyNameValues; + + // number of permutations and combinations + int numRows = (oldKeyValues.length-1)*newFieldValues.length; + intKeyNameValues = new String[numRows+1][]; + intKeyNameValues[0] = new String[oldKeyValues[0].length+1]; + System.arraycopy(oldKeyValues[0],0,intKeyNameValues[0],0,oldKeyValues[0].length); + intKeyNameValues[0][oldKeyValues[0].length] = newFieldName; + + //also need to make more string key rows + String[][] oldStringKeyNameValues = stringKeyNameValues; + stringKeyNameValues = new String[numRows+1][]; + stringKeyNameValues[0] = oldStringKeyNameValues[0]; + + for (int originalRow=1;originalRow1 && valueMode == SINGLE_VALUE_MODE) { + return false; + } + return true; + } catch (RuntimeException e) { + return false; + } + } + + public boolean hasValidLinks() { + return hasValidLinks(myLastCollection); + } + + /** + * @param collection + * @return String indicating the retrieval status + */ + public String retrieveValueStatusString(TableDataSetCollection collection) { + try { + updateLinks(collection); + } catch (RuntimeException e) { + return "Error updating links "+e; + } + if (lastRowNumbers.length==0) { + return "no matching rows"; + } + if (lastRowNumbers.length>1 && valueMode == SINGLE_VALUE_MODE) { + return "Multiple matching index values in SINGLE_VALUE_MODE"; + } + return String.valueOf(retrieveValue(collection)); + } + + /** + * @param e + */ + public void updateIntKeys(TableModelEvent e, String[][] newIntKeyNameValues) { + boolean updateAll = true; + if (e.getType()== e.UPDATE) { + if (e.getFirstRow()>0 && e.getLastRow()<=intKeyValues.length) { + updateAll=false; + for (int i = e.getFirstRow(); i<= e.getLastRow();i++ ) { + for (int c=0;c { + + //Holds end-of-line separator for current platform + public static String EOL; + + //Get end-of-line separator from operating system + static { + EOL = System.getProperty("line.separator"); + } + + String fileName; + + + /** + * Constructor to create a brand new text file. + * + */ + public TextFile() { + + } + + /** + * Basic constructor which reads file line by line. + * + * @param fileName name of file to open and read + */ + public TextFile(String fileName) { + this(fileName, EOL); + } + + /** + * Read a file and split by any regular expression. Default splitter + * is the platform dependent end-of-line character. + * + * @param fileName name of file to open and read + * @param splitter end-of-line separator, default is EOL for system + */ + public TextFile(String fileName, String splitter) { + super(Arrays.asList(readFrom(fileName).split(splitter))); + + //Remember file name for write() method + this.fileName = fileName; + + //Regular expression split() often leaves an empty + //String at the first position: + if (get(0).equals("")) + remove(0); + } + + /** + * Adds a string to the current TextFile object. Wrapper for underlying + * ArrayList.add() method. + * + * @param line string to add to file + */ + public void addLine(String line) { + add(line); + } + + /** + * Returns a line from the current TextFile object given a line number. Line + * numbers start at 0. Wrapper for underlying ArrayList.get() method. + * + * @param lineNumber string to add to file + */ + public String getLine(int lineNumber) { + return get(lineNumber); + } + + /** + * Updates a line in the current TextFile object given a line number. Line + * numbers start at 0. The old line is returned. Wrapper for underlying + * ArrayList.set() method. + * + * @param lineNumber string to add to file + * @param newLine represents the new line character + * @return the old line + */ + public String setLine(int lineNumber, String newLine) { + return set(lineNumber, newLine); + } + + /** + * Read a file as a single string. This method is static and can + * be called directly without creating a TextFile object. + * + * @param fileName name of the file to read from + */ + public static String readFrom(String fileName) { + StringBuilder sb = new StringBuilder(4096); + try { + BufferedReader in = new BufferedReader( + new FileReader( + new File(fileName).getAbsoluteFile())); + try { + String s; + while ((s = in.readLine()) != null) { + sb.append(s); + sb.append(EOL); + } + } finally { + in.close(); + } + } catch (IOException e) { + throw new RuntimeException(e); + } + return sb.toString(); + } + + synchronized public void write() { + writeTo(this.fileName, false); + } + + synchronized public void write(boolean append) { + writeTo(this.fileName, append); + } + + /** + * Write a single file in one method call. This method is static and can + * be called directly without creating a TextFile object. + * + * @param fileName name of file to create and write to + * @param text contents of file + */ + synchronized public static void writeTo(String fileName, String text) { + try { + PrintWriter out = new PrintWriter( + new File(fileName).getAbsoluteFile()); + try { + out.print(text); + } finally { + out.close(); + } + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + /** + * Writes the contents of a TextFile object to a specified file. The + * destination file can be opened in append mode otherwise it will be + * created if it does not exist. + * + * @param fileName name of file to write to + * @param append flag, true to open file for appending + */ + synchronized public void writeTo(String fileName, boolean append) { + try { + FileWriter fWriter = + new FileWriter(new File(fileName).getAbsoluteFile(), append); + BufferedWriter bWriter = new BufferedWriter(fWriter); + PrintWriter out = new PrintWriter(bWriter); + try { + for (String item : this) + out.println(item); + } finally { + out.close(); + } + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + synchronized public void writeTo(String fileName) { + writeTo(fileName, false); + } + + //Used for testing + public static void main(String[] args) { + + //Create a text file in memory and then write it to a file + TextFile newFile = new TextFile(); + newFile.addLine("line #1"); + newFile.addLine("line #2"); + newFile.addLine("line #3"); + newFile.addLine("line #4"); + newFile.setLine(2, "line #3 was replaced with this line"); + newFile.writeTo("testfile.txt"); + + //Example #1: Read previous file into one string + String fileContents = TextFile.readFrom("testfile.txt"); + + //Example #2: Write a string to a new file + TextFile.writeTo("copy of test.txt", fileContents); + + //Example #3: Read a file into an Arraylist based on separator + TextFile txtFile = new TextFile("testfile.txt"); + + System.out.println("number of lines="+ txtFile.size()); + + //Example #3 (cont'd): Change a line by directly accessing it + txtFile.setLine(2, "line #3 has been changed"); + + //Print out to console to check + for (String s : txtFile) { + System.out.println(s); + } + } +} +/*Output: +number of lines=10 +49 35 91 41 82 58 63 46 32 21 +68 33 20 17 43 58 49 89 21 37 +line #2 has been changed +17 30 58 86 83 42 43 50 41 18 +75 20 17 88 49 46 68 60 58 23 +61 31 36 58 42 74 42 72 71 44 +30 47 67 18 94 51 61 78 72 58 +35 84 15 97 98 20 49 61 70 63 +67 39 12 87 34 88 47 47 12 43 +70 15 87 95 77 55 76 55 93 36 +*/ diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java new file mode 100644 index 0000000..fdc8bc1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java @@ -0,0 +1,105 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.File; +import java.io.IOException; + +import org.sandag.cvm.common.datafile.BinaryFileReader; +import org.sandag.cvm.common.datafile.BinaryFileWriter; +import org.sandag.cvm.common.datafile.CSVFileReader; +import org.sandag.cvm.common.datafile.TableDataSet; + + +/** + * Tests the BinaryFileReader and BinaryFile Writer classes. + * + * @author Tim Heier + * @version 1.0, 5/08/2004 + */ +public class BinaryFileTest { + + + public static void main(String[] args) { + + BinaryFileTest.testWrite(); + BinaryFileTest.testRead(); + } + + + public static void testWrite() { + TableDataSet table = null; + + //Read sample file from common-base/src/sql directory + long startTime = System.currentTimeMillis(); + try { + CSVFileReader reader = new CSVFileReader(); + table = reader.readFile(new File("src/sql/zonedata.csv")); + } catch (IOException e) { + e.printStackTrace(); + } + long stopTime = System.currentTimeMillis(); + System.out.println("Time used to read CSV file: " + (stopTime-startTime)); + + //Write the table out to a new file name + BinaryFileWriter writer = new BinaryFileWriter(); + try { + writer.writeFile(table, new File("src/sql/zonedata.bin")); + } catch (IOException e) { + e.printStackTrace(); + } + } + + + public static void testRead() { + TableDataSet table = null; + + //Read sample file from common-base/src/sql directory + long startTime = System.currentTimeMillis(); + try { + BinaryFileReader reader = new BinaryFileReader(); + table = reader.readFile(new File("src/sql/zonedata.bin")); + } catch (IOException e) { + e.printStackTrace(); + } + long stopTime = System.currentTimeMillis(); + System.out.println("Time used to read Binary file: " + (stopTime-startTime)); + + //Display some statistics about the file + System.out.println("Number of columns: " + table.getColumnCount()); + System.out.println("Number of rows: " + table.getRowCount()); + + //Display column titles + String[] labels = table.getColumnLabels(); + for (int i = 0; i < labels.length; i++) { + System.out.print( String.format(" %10s", labels[i]) ); + } + System.out.println(); + + //Print data + for (int i=1; i <= 10; i++) { + //Get a row from table + String row[] = table.getRowValuesAsString(i); + + for (int j=0; j < row.length; j++) { + System.out.print( String.format(" %10s", row[j]) ); + } + System.out.println(); + } + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java new file mode 100644 index 0000000..cc52aed --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java @@ -0,0 +1,83 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.File; +import java.io.IOException; + +import org.sandag.cvm.common.datafile.CSVFileReader; +import org.sandag.cvm.common.datafile.TableDataSet; +import org.sandag.cvm.common.datafile.OLD_CSVFileReader; +import com.pb.common.util.PerformanceTimer; +import com.pb.common.util.PerformanceTimerType; + +/** + * Tests the CSVFileReader class. + * + * @author Tim Heier + * @version 1.0, 2/7/2004 + */ +public class CSVFileReaderTest { + + public static void main(String[] args) { + + CSVFileReaderTest.testRead(); + } + + public static void testRead() { + + CSVFileReader reader = new CSVFileReader(); + // Can set the delimiter set if needed + // reader.setDelimSet(" ,\t\n\r\f\""); + + PerformanceTimer timer = PerformanceTimer.createNewTimer("CsvFileTest", + PerformanceTimerType.CSV_READ); + timer.start(); + + // Read sample file from common-base/src/sql directory + TableDataSet table = null; + try { + table = reader.readFile(new File("test.csv")); + timer.stop(); + } catch (IOException e) { + e.printStackTrace(); + } + + // Display some statistics about the file + System.out.println("Number of columns: " + table.getColumnCount()); + System.out.println("Number of rows: " + table.getRowCount()); + + // Display column titles + String[] labels = table.getColumnLabels(); + for (int i = 0; i < labels.length; i++) { + System.out.print(String.format(" %10s", labels[i])); + } + System.out.println(); + + // Print data + for (int i = 1; i <= 10; i++) { + // Get a row from table + String row[] = table.getRowValuesAsString(i); + + for (int j = 0; j < row.length; j++) { + System.out.print(String.format(" %10s", row[j])); + } + System.out.println(); + } + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java new file mode 100644 index 0000000..4f20c1f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java @@ -0,0 +1,64 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.File; +import java.io.IOException; + +import org.sandag.cvm.common.datafile.CSVFileReader; +import org.sandag.cvm.common.datafile.CSVFileWriter; +import org.sandag.cvm.common.datafile.TableDataSet; + + +/** + * Tests the CSVFileWriter class. + * + * @author Tim Heier + * @version 1.0, 2/7/2004 + */ +public class CSVFileWriterTest { + + + public static void main(String[] args) { + + CSVFileWriterTest.testWrite(); + } + + + public static void testWrite() { + + //Read a CSV file first + TableDataSet table = null; + try { + CSVFileReader reader = new CSVFileReader(); + table = reader.readFile(new File("src/sql/zonedata.csv")); + } catch (IOException e) { + e.printStackTrace(); + } + + //Write the table out to a new file name + CSVFileWriter writer = new CSVFileWriter(); + try { + writer.writeFile(table, new File("src/sql/zonedata_new.csv")); + } catch (IOException e) { + e.printStackTrace(); + } + + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java new file mode 100644 index 0000000..722fb0e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java @@ -0,0 +1,85 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.File; + +import java.util.Date; + +import org.sandag.cvm.common.datafile.DataFile; +import org.sandag.cvm.common.datafile.DataReader; +import org.sandag.cvm.common.datafile.DataWriter; + +/** + * Tests the records package. + * class. + * + * @author Tim Heier + * @version 1.0, 4/15/2000 + * + */ +public class DataFileTest { + + public static void main(String[] args) throws Exception { + int numRecords = 100; + String fileName = "sample.db"; + + //Delete file if it exists as part of the test + File file = new File(fileName); + + if (file.exists()) { + file.delete(); + } + + System.out.println("creating data file: sample.db"); + + DataFile dataFile = new DataFile("sample.db", 100); + + System.out.println("adding data: lastAccessTime"); + + DataWriter dw = new DataWriter("lastAccessTime"); + + dw.writeObject(new Date()); + dataFile.insertRecord(dw); + + DataReader dr = dataFile.readRecord("lastAccessTime"); + Date d = (Date) dr.readObject(); + + System.out.println("lastAccessTime = " + d.toString()); + + System.out.println("updating data: lastAccessTime"); + dw = new DataWriter("lastAccessTime"); + dw.writeObject(new Date()); + dataFile.updateRecord(dw); + + System.out.println("reading data: lastAccessTime"); + dr = dataFile.readRecord("lastAccessTime"); + d = (Date) dr.readObject(); + System.out.println("lastAccessTime = " + d.toString()); + + System.out.println("deleting data: lastAccessTime"); + dataFile.deleteRecord("lastAccessTime"); + + if (dataFile.recordExists("lastAccessTime")) { + throw new Exception("data not deleted"); + } else { + System.out.println("data successfully deleted."); + } + + System.out.println("test completed."); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java new file mode 100644 index 0000000..2d3585f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java @@ -0,0 +1,193 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.IOException; + +import org.sandag.cvm.common.datafile.DiskObjectArray; +import com.pb.common.util.ObjectUtil; + +/** + * + * @author Tim Heier + * @version 1.0, 8/13/2003 + * + */ + +public class DiskObjectArrayTest { + + public static final int ARRAY_SIZE = 500000; + public static final int DATA_SIZE = 1000; + + + public static void main(String[] args) { + + DiskObjectArrayTest test = new DiskObjectArrayTest(); + test.testCreate(); + test.testAddElements(); + test.testFillArray(); + test.testUpdateArray(); + test.testAddLargeElement(); + } + + + public void testCreate() { + + //Create a large object array + DiskObjectArray ba = null; + try { + ba = new DiskObjectArray("test.array", ARRAY_SIZE, DATA_SIZE); + } + catch (IOException e) { + e.printStackTrace(); + } + + System.out.println("testCreate() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); + + ba.close(); + } + + + public void testAddElements() { + + //Create a large object array + DiskObjectArray ba = null; + try { + ba = new DiskObjectArray("test.array"); + } + catch (IOException e) { + e.printStackTrace(); + } + + //----- Add elements to array + + ba.add( 1, new Integer(1) ); + ba.add( 1, new Integer(2) ); //try adding to same location + + int[] intArray = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; + ba.add( 2, intArray); + + ba.add( 1000, new Integer(1000) ); + + //----- Read elements from array + + System.out.println("element 1 = " + ba.get( 1 )); + + intArray = (int[]) ba.get( 2 ); + for (int i=0; i < 10; i++) { + System.out.println("element 2["+i+"]" + " = " + intArray[i]); + } + + System.out.println("element 1000 = " + ba.get( 1000 )); + + System.out.println("testAddElements() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); + + ba.close(); + } + + + public void testFillArray() { + + long startTime=System.currentTimeMillis(), endTime; + + //Create a large object array + DiskObjectArray ba = null; + try { + ba = new DiskObjectArray("test.array"); + } + catch (IOException e) { + e.printStackTrace(); + } + + //----- Add a small object to each location in array + for (int i=0; i < ARRAY_SIZE; i++) { + //for (int i=0; i < 5000; i++) { + if ((i % 500) == 0) { + endTime = System.currentTimeMillis(); + System.out.println("adding="+ i + ", time="+(endTime-startTime)); + startTime = System.currentTimeMillis(); + } + ba.add( i, new Integer(i)); + } + + System.out.println("testFillArray() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); + + //Close it + ba.close(); + } + + + public void testUpdateArray() { + + long startTime=System.currentTimeMillis(), endTime; + + //Create a large object array + DiskObjectArray ba = null; + try { + ba = new DiskObjectArray("test.array"); + } + catch (IOException e) { + e.printStackTrace(); + } + + //Update an element in the middle of the array + int[] intArray = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19 }; + ba.add( 250000, intArray); + + intArray = (int[]) ba.get( 250000 ); + + System.out.println("element 250000[5]" + " = " + intArray[5]); + + System.out.println("testUpdateArray() done."); + + //Close it + ba.close(); + } + + + public void testAddLargeElement() { + + //Create a large object array + DiskObjectArray ba = null; + try { + ba = new DiskObjectArray("test.array"); + } + catch (IOException e) { + e.printStackTrace(); + } + + //----- Add elements to array + + ba.add( 1, new Integer(1) ); + ba.add( 2, new Integer(2) ); + ba.add( 3, new Integer(3) ); + + int[] intArray = new int[210]; + System.out.println("size of new element 2 = " + ObjectUtil.sizeOf(intArray)); + + ba.add( 2, intArray); + + System.out.println("Trying to read element 3..."); + System.out.println("element 3 = " + ba.get( 3 )); + + System.out.println("testAddLargeElement() done."); + + ba.close(); + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java new file mode 100644 index 0000000..f342d72 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java @@ -0,0 +1,87 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import java.io.IOException; + +import org.sandag.cvm.common.datafile.JDBCTableReader; +import org.sandag.cvm.common.datafile.TableDataSet; +import com.pb.common.sql.JDBCConnection; + +/** + * Tests the JDBCTableReader class. + * + * @author Tim Heier + * @version 1.0, 2/7/2004 + */ +public class JDBCTableReaderTest { + + public static String HOST = "localhost"; + public static String DATABASE = "test"; + public static String USER = ""; + public static String PASSWD = ""; + public static String TABLE = "zonedata"; + public static String DRIVER = "com.mysql.jdbc.Driver"; + + + //Database url string - specific to vendor + public static String URL = "jdbc:mysql://" + HOST + "/" + DATABASE + "?user=" + USER + "&password=" + PASSWD; + + + public static void main(String[] args) { + + JDBCTableReaderTest.testLoad(); + } + + + public static void testLoad() { + JDBCConnection jdbcConn = new JDBCConnection(URL, "com.mysql.jdbc.Driver", USER, PASSWD); + + JDBCTableReader reader = new JDBCTableReader(jdbcConn); + + TableDataSet table = null; + try { + table = reader.readTable(TABLE); + } catch (IOException e) { + e.printStackTrace(); + } + jdbcConn.close(); + + //Display some statistics about the file + System.out.println("Number of columns: " + table.getColumnCount()); + System.out.println("Number of rows: " + table.getRowCount()); + + //Display column titles + String[] labels = table.getColumnLabels(); + for (int i = 0; i < labels.length; i++) { + System.out.print( String.format("%10s", labels[i]) ); + } + System.out.println(); + + //Print data + for (int i=1; i <= 10; i++) { + + //Get a row from table + String row[] = table.getRowValuesAsString(i); + + for (int j=0; j < row.length; j++) { + System.out.print( String.format(" %10s", row[j]) ); + } + System.out.println(); + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java new file mode 100644 index 0000000..97fe080 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java @@ -0,0 +1,322 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.datafile.tests; + +import org.sandag.cvm.common.datafile.CSVFileReader; +import org.sandag.cvm.common.datafile.CSVFileWriter; +import org.sandag.cvm.common.datafile.TableDataSet; + +import java.io.File; +import java.io.IOException; +import org.apache.log4j.Logger; + +/** + * Tests the TableDataSet class. + * + * @author Tim Heier + * @version 1.0, 2/8/2003 + */ +public class TableDataSetTest { + + protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile.test"); + + public static void main(String[] args) { + TableDataSet table = TableDataSetTest.readCSVFile(); + TableDataSetTest.testGetRowValues( table ); + TableDataSetTest.testGetValueAt( table ); + TableDataSetTest.testColmnNameRetrieval( table ); + TableDataSetTest.testSetValueAt( table ); + TableDataSetTest.testSaveFile( table ); + TableDataSetTest.testBuildIndex( table ); + TableDataSetTest.readCSVFileNoLabels(); + TableDataSetTest.readCSVFileFilteringColumns(); + //TableDataSetTest.testPerformance(); + } + + +/* TCH - Test method for my machine. + + public static void testPerformance() { + TableDataSetTest.printMemory(); + + TableDataSet[] tables = new TableDataSet[3]; + long startTime, stopTime = 0; + try { + for (int i=0; i < 3; i++) { + startTime = System.currentTimeMillis(); + tables[i] = new TableDataSet(new File("c:/temp/morpc/M5678.csv")); + stopTime = System.currentTimeMillis(); + logger.info("Finished reading " + i + ", " + (stopTime-startTime)); + TableDataSetTest.printMemory(); + } + startTime = System.currentTimeMillis(); + tables[2].saveFile(new File("c:/temp/morpc/M5678_new.csv"), 0, new DecimalFormat("#.000000")); + stopTime = System.currentTimeMillis(); + logger.info("Finished writing " + (stopTime-startTime)); + TableDataSetTest.printMemory(); + } + catch (IOException e) { + e.printStackTrace(); + return; + } + } + + public static void printMemory() { + logger.info("Total memory : " + Runtime.getRuntime().totalMemory()); + logger.info("Max memory : " + Runtime.getRuntime().maxMemory()); + logger.info("Free memory : " + Runtime.getRuntime().freeMemory()); + } +*/ + + /** + * Read a table data set. + * + * @return a fully populated TableDataSet + */ + public static TableDataSet readCSVFile() { + System.out.println("executing readCSVFile()"); + TableDataSet table = null; + + try { + CSVFileReader reader = new CSVFileReader(); + table = reader.readFile(new File("src/sql/bufcrl1.txt")); + } + catch (IOException e) { + e.printStackTrace(); + } + + return table; + } + + + /** + * Tests the getRowValues() method. + * + */ + public static void testGetRowValues(TableDataSet table) { + System.out.println("executing testGetRowValues()"); + + //Display some statistics about the file + System.out.println("Number of columns: " + table.getColumnCount()); + System.out.println("Number of rows: " + table.getRowCount()); + + //Display column titles + String[] titles = table.getColumnLabels(); + for (int i = 0; i < titles.length; i++) { + System.out.print( String.format("%10s", titles[i]) ); + } + System.out.println(); + + try { + //Print the first 10 rows + for (int i=1; i <= 10; i++) { + + //Get a row from table + float row[] = (float[]) table.getRowValues(i); + + for (int j=0; j < row.length; j++) { + System.out.print( String.format(" %9.2f", row[j]) ); + } + System.out.println(); + } + } + catch (Throwable e) { + System.out.println("Exception in testGetRowValues()"); + e.printStackTrace(); + } + } + + + /** + * Tests the getValueAt() and getValues() methods. + * + */ + public static void testGetValueAt(TableDataSet table) { + System.out.println("executing testGetValueAt()"); + + //Get the name of a column + float value1 = table.getValueAt( 1, 10 ); + float value2 = table.getValueAt( 2, 10 ); + String strValue = table.getStringValueAt( 2, 19 ); + + System.out.println( String.format("value at (1,10) =%7.2f", value1) ); + System.out.println( String.format("value at (2,10) =%7.2f", value2) ); + + System.out.println( String.format("value at (2,19) =%s", strValue) ); + System.exit(1); + + //Ask for data as a float[][]. This can be used to create a Matrix + float[][] values = null; + try { + values = table.getValues(); + + for (int i = 0; i < 10; i++) { + + for (int j = 0; j < table.getColumnCount(); j++) { + System.out.print( String.format(" %9.2f", values[i][j]) ); + } + System.out.println(); + } + } + catch (Exception e) { + System.out.println("Exception in testGetRowValues()"); + e.printStackTrace(); + } + } + + + /** + * Test the column look-up features of the table dataset. + * + */ + public static void testColmnNameRetrieval(TableDataSet table) { + System.out.println("executing testColmnNameRetrieval()"); + + //Get the name of a column + String name1 = table.getColumnLabel( 1 ); + String name2 = table.getColumnLabel( 2 ); + + System.out.println( "column 1 = " + name1 ); + System.out.println( "column 2 = " + name2 ); + + //Get the position of a column given it's name - case is ignored + int position1 = table.getColumnPosition("zone"); + int position2 = table.getColumnPosition("totpop"); + + System.out.println( "position of zone = " + position1 ); + System.out.println( "position of totpop = " + position2 ); + + String hhInc = table.getStringValueAt(1, "totpop"); + System.out.println("totpop on row 1 = " + hhInc); + } + + + /** + * Test the column look-up features of the table dataset. + * + */ + public static void testSetValueAt(TableDataSet table) { + System.out.println("executing testSetValueAt()"); + + float value1 = table.getValueAt( 1, 10 ); + System.out.println( String.format("value before update (1,10) =%7.2f", value1) ); + + table.setValueAt(1, 10, (float)300.1); + + System.out.println( String.format("value after update (1,10) =%7.2f", table.getValueAt( 1, 10 )) ); + + } + + + /** + * Test the indexing feature of the table dataset. + * + */ + public static void testSaveFile(TableDataSet table) { + System.out.println("executing testSaveFile()"); + + try { + CSVFileWriter writer = new CSVFileWriter(); + writer.writeFile(table, new File("testtable_new.csv")); + } + catch (IOException e) { + e.printStackTrace(); + } + } + + + /** + * Test the indexing feature of the table dataset. + * + */ + public static void testBuildIndex(TableDataSet table) { + System.out.println("executing testBuildIndex()"); + + //Build an index on column 1 + table.buildIndex(1); + float value1 = table.getIndexedValueAt( 25, 1 ); + + System.out.println( String.format("looking up indexed value=25 in column=1: %7.2f", value1) ); + } + + + /** + * Read a table data set. + */ + public static void readCSVFileNoLabels() { + System.out.println("executing readCSVFileNoLabels()"); + + TableDataSet table = null; + + try { + CSVFileReader reader = new CSVFileReader(); + table = reader.readFile(new File("src/sql/zonedata_nolabels.csv"), false); + } + catch (IOException e) { + e.printStackTrace(); + System.exit(1); + } + + //Display column titles + String[] titles = table.getColumnLabels(); + for (int i = 0; i < titles.length; i++) { + System.out.print( String.format("%10s", titles[i]) ); + } + System.out.println(); + + } + + + /** + * Read a table data set. + */ + public static void readCSVFileFilteringColumns() { + System.out.println("executing readCSVFileFilteringColumns()"); + TableDataSet table = null; + + String[] columnsToRead = { "tothh", "hhinc1" }; + + try { + CSVFileReader reader = new CSVFileReader(); + table = reader.readFile(new File("src/sql/zonedata.csv"), columnsToRead); + + //Display column titles + String[] titles = table.getColumnLabels(); + for (int i = 0; i < titles.length; i++) { + System.out.print( String.format("%10s", titles[i]) ); + } + System.out.println(); + + //Print a couple of values + float value1 = table.getValueAt( 1, 1 ); + float value2 = table.getValueAt( 1, 2 ); + float value3 = table.getValueAt( 2, 1 ); + float value4 = table.getValueAt( 2, 2 ); + + System.out.println( String.format("value at (1,1) =%7.2f", value1) ); + System.out.println( String.format("value at (1,2) =%7.2f", value2) ); + System.out.println( String.format("value at (2,1) =%7.2f", value3) ); + System.out.println( String.format("value at (2,2) =%7.2f", value4) ); + } + catch (Exception e) { + e.printStackTrace(); + } + + } + + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java new file mode 100644 index 0000000..2441f6c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java @@ -0,0 +1,33 @@ +package org.sandag.cvm.common.discreteEvent; + +import java.util.NoSuchElementException; + +import org.apache.log4j.Logger; + +public class EventDispatcher { + static final Logger logger = Logger.getLogger(EventDispatcher.class); + EventQueue myQueue = new EventQueue(); + + public EventDispatcher() { + + } + + public void dispatchEvents() { + try { + while(true) { + TimedEvent nextEvent = myQueue.popNextEvent(); + nextEvent.handleEvent(this); + } + } catch (NoSuchElementException e) { + logger.info("Event queue is empty"); + } + } + + public EventQueue getMyQueue() { + return myQueue; + } + + public void setMyQueue(EventQueue myQueue) { + this.myQueue = myQueue; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java new file mode 100644 index 0000000..0dba10e --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java @@ -0,0 +1,34 @@ +package org.sandag.cvm.common.discreteEvent; + +import java.util.NoSuchElementException; +import java.util.TreeSet; + +public class EventQueue { + + //ENHANCEMENT look at using other storage for events. + // For instance we could have 3600*24 buckets, one for + // each second in the day, and then optionally sort the events + // within the buckets. + // Then we could use an array or linked list for each bucket? + TreeSet futureEvents; + + public EventQueue() { + futureEvents = new TreeSet (); + } + + public void enqueue(TimedEvent e) { + futureEvents.add(e); + } + + public TimedEvent popNextEvent() throws NoSuchElementException { + TimedEvent nextEvent = futureEvents.first(); + futureEvents.remove(nextEvent); + return nextEvent; + + } + + public int size() { + return futureEvents.size(); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java new file mode 100644 index 0000000..083db80 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java @@ -0,0 +1,65 @@ +package org.sandag.cvm.common.discreteEvent; + +import org.junit.BeforeClass; +import org.junit.Test; + +public class TestEventDispatcher { + + class RequeueEvent extends TimedEvent { + + public RequeueEvent(double myTime) { + super(myTime); + } + + @Override + public void handleEvent(EventDispatcher dispatch) { + numOfEvents++; + if (numOfEvents + averageQueueSize < numberOfEventsToSimulate) { + double selector = (int) (Math.random()*3); + // queue up 0, 1 or 2 new events. + for (int i=0;i ((TimedEvent)arg0).myTime) { + return 2; + } + if (myTime < ((TimedEvent)arg0).myTime) { + return -2; + } + if (arg0==this) return 0; + // ok, our times are identical! + if (myRandomNumber ==0) myRandomNumber = (Math.random()+.001); + TimedEvent other = (TimedEvent) arg0; + if (other.myRandomNumber == 0) other.myRandomNumber = (Math.random()+.001); + if (myRandomNumber > other.myRandomNumber) return 1; + if (myRandomNumber < other.myRandomNumber) return -1; + //TODO maybe do something smarter? + logger.error("randomly generated event sortings are the same"); + return 0; + } + + @Override + public String toString() { + return "Event at "+myTime; + } + + public boolean isProcessEvenIfAfterSimulationTime() { + return processEvenIfAfterSimulationTime; + } + + public void setProcessEvenIfAfterSimulationTime( + boolean processEvenIfAfterSimulationTime) { + this.processEvenIfAfterSimulationTime = processEvenIfAfterSimulationTime; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java new file mode 100644 index 0000000..877efca --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java @@ -0,0 +1,65 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + +package org.sandag.cvm.common.emme2; + +import java.io.File; +import java.util.*; + +import com.pb.common.matrix.*; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class Emme2MatrixHashtableReader extends Emme2MatrixReader { + + private Hashtable readMatrices = new Hashtable(); + + /** Reads an entire matrix from an Emme2 databank, but if it's already been + * read once before returns the previously read one hashtable + * + * @param index the short name of the matrix, eg. "mf10" + * @return a complete matrix + * @throws MatrixException + */ + public Matrix readMatrix(String index) throws MatrixException { + + if (readMatrices.containsKey(index)) { + return (Matrix) readMatrices.get(index); + } + Matrix m = super.readMatrix(index ); + readMatrices.put(index,m); + return m; + + } + + + + /** + * Constructor for Emme2MatrixHashtableReader. + * @param file + */ + public Emme2MatrixHashtableReader(File file) { + super(file); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java new file mode 100644 index 0000000..6d05feb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java @@ -0,0 +1,124 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + +package org.sandag.cvm.common.emme2; + +import com.pb.common.matrix.*; + +import java.util.*; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class IndexConditionFunction { + + public void readMatrices(MatrixCacheReader mr) { + coefficients = new double[coefficientValues.size()]; + matrices = new Matrix[matrixNames.size()]; +// conditions = new int[conditionValues.size()]; + for (int i=0;i + * MSValue * MFValue + *
Where MSValue is constant and MFValue is the value from the MF matrix + * @param matrix the MF matrix where the value is looked up based on the indices + * @param coeffMSMatrix the MS matrix where the coefficient value comes from + */ + public void addCoefficient(String matrix, String coeffMSMatrix) { + matrixNames.add(matrix); + coefficientValues.add(coeffMSMatrix); + coefficients = null; + matrices = null; + } + + /** + * Method addCoefficient, adds a term into the utility function
+ * coeffValue * MFValue + * @param matrix the emme/2 name of the matrix to be used (if + * blank or "none" the coefficient will be taken as part of the constant term) + * @param coeffValue value of the coefficient to be multiplied by the emme/2 matrix term + */ + public void addCoefficient(String matrix, double coeffValue) { + if (matrix.length() ==0 || matrix.equalsIgnoreCase("none")) addConstant(coeffValue); + else { + matrixNames.add(matrix); + coefficientValues.add(new Double(coeffValue)); + coefficients = null; + matrices = null; + } + } + + /** Changes the value of a specific coefficient (normally use addCoefficient instead) + * @param coeffIndex the previously added coefficient to change the value of + * @param matrix the matrix that is multiplied by the coefficient value + * @param coeffValue the coefficient value + */ + void setCoefficient(int coeffIndex,String matrix,double coeffValue) { + coefficientValues.set(coeffIndex,new Double(coeffValue)); + matrixNames.set(coeffIndex,matrix); + coefficients = null; + matrices = null; + } + + /** + * Reads in the emme2matrices into memory, also initializes some internal data. Call this method after the + * terms have been added to the utility function but before the utility function is evaluated. + * @param matrixReader the matrix reader to use to read in the matrices from the emme2 databank + */ + public void readMatrices(MatrixCacheReader matrixReader) { + coefficients = new double[coefficientValues.size()]; + matrices = new Matrix[matrixNames.size()]; + for (int i=0;i=4.712999999 || m.getValueAt(1619,1610)<=4.7130000001); + assertTrue("name should be mf22 but is "+m.getName(),m.getName().equals("mf22")); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java new file mode 100644 index 0000000..45d4549 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java @@ -0,0 +1,26 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public interface AggregateAlternative extends Alternative { + void setAggregateQuantity(double amount, double derivative); +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java new file mode 100644 index 0000000..3d6712b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java @@ -0,0 +1,26 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public interface Alternative { + double getUtility(); +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java new file mode 100644 index 0000000..3f36357 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java @@ -0,0 +1,20 @@ +package org.sandag.cvm.common.model; + +public class ChoiceModelOverflowException extends RuntimeException { + + public ChoiceModelOverflowException() { + } + + public ChoiceModelOverflowException(String message) { + super(message); + } + + public ChoiceModelOverflowException(Throwable cause) { + super(cause); + } + + public ChoiceModelOverflowException(String message, Throwable cause) { + super(message, cause); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java new file mode 100644 index 0000000..ddd447c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java @@ -0,0 +1,63 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + + +package org.sandag.cvm.common.model; + +import java.util.Random; + +public abstract class DiscreteChoiceModel implements DiscreteChoiceModelInterface { + /** Picks one of the alternatives based on the logit model probabilities + * @throws ChoiceModelOverflowException */ + public abstract Alternative monteCarloChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException ; + + /** Picks one of the alternatives based on the logit model probabilities and random number given*/ + public abstract Alternative monteCarloChoice(double r) throws NoAlternativeAvailable ; + + public Alternative monteCarloElementalChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException { + Alternative a = monteCarloChoice(); + while (a instanceof DiscreteChoiceModel) { + a = ((DiscreteChoiceModel) a).monteCarloChoice(); + } + return a; + } + /** Use this method if you want to give a random number */ + public Alternative monteCarloElementalChoice(double r) throws NoAlternativeAvailable { + Alternative a = monteCarloChoice(r ); + Random newRandom = new Random(new Double(r*1000).longValue()); + while (a instanceof DiscreteChoiceModel) { + a = ((DiscreteChoiceModel) a).monteCarloChoice(newRandom.nextDouble()); + } + return a; + } + + /** @param a the alternative to add into the choice set */ + public abstract void addAlternative(Alternative a); + + public abstract Alternative alternativeAt(int i); + + public abstract double[] getChoiceProbabilities(); + + abstract public void allocateQuantity(double amount); + + +} + diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java new file mode 100644 index 0000000..24c59ed --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java @@ -0,0 +1,47 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + +package org.sandag.cvm.common.model; + +/** + * @author jabraham + * + * To change the template for this generated type comment go to + * Window - Preferences - Java - Code Generation - Code and Comments + */ +public interface DiscreteChoiceModelInterface { + /** Picks one of the alternatives based on the logit model probabilities + * @throws ChoiceModelOverflowException */ + public abstract Alternative monteCarloChoice() + throws NoAlternativeAvailable, ChoiceModelOverflowException; + /** Picks one of the alternatives based on the logit model probabilities and random number given*/ + public abstract Alternative monteCarloChoice(double r) + throws NoAlternativeAvailable; + public abstract Alternative monteCarloElementalChoice() + throws NoAlternativeAvailable, ChoiceModelOverflowException; + /** Use this method if you want to give a random number */ + public abstract Alternative monteCarloElementalChoice(double r) + throws NoAlternativeAvailable; + /** @param a the alternative to add into the choice set */ + public abstract void addAlternative(Alternative a); + public abstract Alternative alternativeAt(int i); + public abstract double[] getChoiceProbabilities(); + public abstract void allocateQuantity(double amount); +} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java new file mode 100644 index 0000000..b772896 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java @@ -0,0 +1,37 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public class FixedUtilityAlternative implements Alternative { + public FixedUtilityAlternative(double utilityValue) { + this.utilityValue=utilityValue; + } + + public double getUtility() {return utilityValue;} + + public double getUtilityValue(){ return utilityValue; } + + public void setUtilityValue(double utilityValue){ this.utilityValue = utilityValue; } + + private double utilityValue; + public String toString() {return "FixedUtility - "+utilityValue;}; +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java new file mode 100644 index 0000000..eb38dfa --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java @@ -0,0 +1,77 @@ +/** + * + */ +package org.sandag.cvm.common.model; + +import java.util.Random; + +/** + * This class is used to sample and store Weibull distributed values + * @author John Abraham + * + */ +public class GumbelErrorTerm extends RandomVariable { + + static final double eulersConstant = 0.5772156649015328606; + + /** + * Static method to sample and return + * @param dispersionParameter + * @return a sample value from the Gumbel distribution with mean 0. + */ + public static double sample(double dispersionParameter) { + double sample = Math.random(); + return eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); + } + + /** + * Static method to sample and return + * @param dispersionParameter + * @param r The random number generator to use + * @return a sample value from the Gumbel distribution with mean 0. + */ + public static double sample(double dispersionParameter, Random r) { + double sample = r.nextDouble(); + return eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); + } + + private double dispersionParameter; + + /** + * Constructor + */ + public GumbelErrorTerm(double dispersionParameter) { + value = 0; + this.dispersionParameter = dispersionParameter; + } + + @Override + public Object clone() throws CloneNotSupportedException { + return super.clone(); + } + + public double getDispersionParameter() { + return dispersionParameter; + } + + /** (non-Javadoc) + * @see org.sandag.cvm.common.model.RandomVariable#sample() + * Changes the value to a new random term by sampling from the Probability + * Density Function z*exp(-z)/beta where z=exp(-(x)/beta)) + */ + @Override + public double sample() { + double sample = Math.random(); + value = eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); + return value; + } + + public static double transformUniformToGumble(double uniform, double dispersionParameter) { + return eulersConstant - dispersionParameter*Math.log(-Math.log(uniform)); + } + + public void setDispersionParameter(double dispersionParameter) { + this.dispersionParameter = dispersionParameter; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java new file mode 100644 index 0000000..9091d4c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java @@ -0,0 +1,72 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + +package org.sandag.cvm.common.model; + +/** + * @author John Abraham + * + * A cool class created by John Abraham (c) 2003 + */ +public class LinearInParametersFunction { + + /** + * Constructor for LinearInParametersFunction. + */ + public LinearInParametersFunction() { + new LinearInParametersFunction(0); + } + + public LinearInParametersFunction(int size) { + coefficients = new double[size]; + } + + private double[] coefficients; + + public void addCoefficient(double coeffValue) { + double[] oldCoefficients = coefficients; + coefficients = new double[oldCoefficients.length+1]; + System.arraycopy(oldCoefficients,0,coefficients,0,oldCoefficients.length); + coefficients[coefficients.length-1]=coeffValue; + } + + public void setCoefficient(int coeffIndex,double coeffValue) { + if (coefficients.length <= coeffIndex) { + double[] oldCoefficients = coefficients; + coefficients = new double[coeffIndex+1]; + System.arraycopy(oldCoefficients,0,coefficients,0,oldCoefficients.length); + } + coefficients[coeffIndex]=coeffValue; + } + + public double getCoefficient(int coeffIndex) { + return coefficients[coeffIndex]; + } + + public double calcProduct(double[] values) { + double value = 0; + for (int i=0; i(); + dispersionParameter = 1.0; + } + //use this constructor if you know how many alternatives + public LogitModel(int numberOfAlternatives) { + alternatives = new ArrayList(numberOfAlternatives); + dispersionParameter = 1.0; + } + + + /** @return the composite utility (log sum value) of all the alternatives */ + public double getUtility() { + double sum = 0; + int i = 0; + while (i it = alternatives.listIterator(); + int i = 0; + while (it.hasNext()) { + Alternative a = (Alternative) it.next(); + double utility = a.getUtility(); + weights[i] = Math.exp(dispersionParameter * utility); + if (Double.isNaN(weights[i])) { + System.out.println("hmm, alternative "+a+" was such that LogitModel weight was NaN"); + System.out.println("dispersionParameter ="+dispersionParameter+", utility ="+utility); + throw new Error("NAN in weight for alternative "+a); + } + sum += weights[i]; + i++; + } + if (sum!=0) { + for (i = 0; i < weights.length; i++) { + weights[i] /= sum; + } + } + return weights; + } + } + + public Alternative alternativeAt(int i) { return (Alternative) alternatives.get(i);}// should throw an error if out of range + + + /** Picks one of the alternatives based on the logit model probabilities + * @throws ChoiceModelOverflowException */ + public Alternative monteCarloChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException { + // synchronized(alternatives) { + double[] weights = new double[alternatives.size()]; + double sum = 0; + Iterator it = alternatives.listIterator(); + int i = 0; + while (it.hasNext()) { + Alternative a = it.next(); + double utility = a.getUtility(); + weights[i] = Math.exp(dispersionParameter * utility); + if (Double.isNaN(weights[i])) { + System.out.println("hmm, alternative "+a+" was such that LogitModel weight was NaN"); + System.out.println("dispersionParameter ="+dispersionParameter+", utility ="+utility); + } + sum += weights[i]; + i++; + } + if (Double.isInfinite(sum)) { + logger .fatal("Overflow error in choice model, list of alternatives follows"); + it = alternatives.listIterator(); + while (it.hasNext()) { + Alternative a = (Alternative) it.next(); + double utility = a.getUtility(); + System.out.println(" U:"+utility+", W:"+Math.exp(dispersionParameter * utility)+" for "+a); + } + + throw new ChoiceModelOverflowException("Infinite weight(s) in logit model choice function"); + } + if (sum==0) throw new NoAlternativeAvailable(); + double selector = myRandom.nextDouble() * sum; + sum = 0; + for (i = 0; i < weights.length; i++) { + sum += weights[i]; + if (selector <= sum) return (Alternative)alternatives.get(i); + } + //yikes! + System.out.println("Error: problem with logit model. sum is "+sum+", rand is "+selector); + System.out.println("Alternative,weight"); + for (i=0; i < weights.length; i++){ + System.out.println((Alternative)alternatives.get(i)+","+weights[i]); + } + throw new Error("Random Number Generator in Logit Model didn't return value between 0 and 1"); + // } + } + + /** Picks one of the alternatives based on the logit model probabilities; + use this if you want to give method random number */ + public Alternative monteCarloChoice(double randomNumber) throws NoAlternativeAvailable { + synchronized(alternatives) { + double[] weights = new double[alternatives.size()]; + double sum = 0; + Iterator it = alternatives.listIterator(); + int i = 0; + while (it.hasNext()) { + double utility = ((Alternative)it.next()).getUtility(); + weights[i] = Math.exp(dispersionParameter * utility); + if (Double.isNaN(weights[i])) { + System.out.println("hmm, alternative was such that LogitModel weight was NaN"); + } + sum += weights[i]; + i++; + } + if (sum==0) throw new NoAlternativeAvailable(); + double selector = randomNumber * sum; + sum = 0; + for (i = 0; i < weights.length; i++) { + sum += weights[i]; + if (selector <= sum) return (Alternative)alternatives.get(i); + } + //yikes! + System.out.println("Error: problem with logit model. sum is "+sum+", rand is "+randomNumber); + System.out.println("Alternative,weight"); + for (i=0; i < weights.length; i++){ + System.out.println((Alternative)alternatives.get(i)+","+weights[i]); + } + throw new Error("Random Number Generator in Logit Model didn't return value between 0 and 1"); + } + } + + + private double dispersionParameter; + private double constantUtility=0; + protected ArrayList alternatives; + + public String toString() { + StringBuffer altsString = new StringBuffer(); + int alternativeCounter = 0; + if (alternatives.size() > 5) { altsString.append("LogitModel with " + alternatives.size() + "alternatives {"); } + else altsString.append("LogitModel, choice between "); + Iterator it = alternatives.iterator(); + while (it.hasNext() && alternativeCounter < 5) { + altsString.append(it.next()); + altsString.append(","); + alternativeCounter ++; + } + if (it.hasNext()) altsString.append("...}"); else altsString.append("}"); + return new String(altsString); + } + + public double getConstantUtility(){ return constantUtility; } + + public void setConstantUtility(double constantUtility){ this.constantUtility = constantUtility; } + + /** + * Method arrayCoefficientSimplifiedChoice. + * @param theCoefficients + * @param theAttributes + * @return int + */ + public static int arrayCoefficientSimplifiedChoice( + double[][] theCoefficients, + double[] theAttributes) { + + double[] utilities = new double[theCoefficients.length]; + int alt; + for (alt =0; alt < theCoefficients.length; alt++){ + utilities[alt] = 0; + for (int c=0;c getAlternativesIterator() { + return alternatives.iterator(); + } + +} + + diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java new file mode 100644 index 0000000..3e377ed --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java @@ -0,0 +1,28 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public class NoAlternativeAvailable extends Exception { + + public NoAlternativeAvailable() { + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java new file mode 100644 index 0000000..3962bbe --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java @@ -0,0 +1,34 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public abstract class NumericalDerivativeSingleParameterFunction implements SingleParameterFunction { + double delta; + public NumericalDerivativeSingleParameterFunction(double delta) { + this.delta=delta; + } + + public double derivative(double point){ + double perturbed = evaluate(point+delta); + return (perturbed-evaluate(point))/delta; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java new file mode 100644 index 0000000..b5d0f59 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java @@ -0,0 +1,35 @@ +package org.sandag.cvm.common.model; + +public abstract class RandomVariable implements Cloneable { + + double value; + + private boolean validValue; + + public RandomVariable() { + super(); + value = 0; + validValue = false; + } + + @Override + public Object clone() throws CloneNotSupportedException { + return super.clone(); + } + + abstract public double sample(); + + public double sampleIfNecessaryAndSave() { + if (validValue) { + return value; + } + value = sample(); + validValue = true; + return value; + } + + public void setInvalid() { + validValue = false; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java new file mode 100644 index 0000000..9063198 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java @@ -0,0 +1,28 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + +package org.sandag.cvm.common.model; + +public interface SingleParameterFunction { + double evaluate(double point); + + double derivative(double point); +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java new file mode 100644 index 0000000..9476ff7 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java @@ -0,0 +1,63 @@ +/* + Travel Model Microsimulation library + Copyright (C) 2005 PbConsult, JE Abraham and others + + + 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. + +*/ + + + + +package org.sandag.cvm.common.model; + +public class TestLogitModel { + public static void main(String[] args) throws ChoiceModelOverflowException { + try { + LogitModel lm = new LogitModel(); + Alternative a = new FixedUtilityAlternative(1.0); + Alternative b = new FixedUtilityAlternative(2.0); + lm.addAlternative(a); + lm.addAlternative(b); + for (double dp=.001;dp<100000 ;dp*=2 ) + { + lm.setDispersionParameter(dp); + int acount=0; + int bcount=0; + for (int i=0;i<1000 ;i++ ) + { + if (lm.monteCarloChoice()==a) acount++; else bcount++; + } + System.out.println("DispersionParameter="+dp+" acount="+acount+" bcount"+bcount); + } + lm=new LogitModel(); + lm.addAlternative(new FixedUtilityAlternative(Math.exp(50000))); + System.out.println("Composite utility of one infinite utility alternative is "+lm.getUtility()); + lm.monteCarloChoice(); + lm=new LogitModel(); + lm.addAlternative(new FixedUtilityAlternative(Math.log(0))); + lm.addAlternative(new FixedUtilityAlternative(5.0)); + + System.out.println("Composite utility of one negative infinite utility alternative and a '5' alt is "+lm.getUtility()); + lm.monteCarloChoice(); + + lm.addAlternative(new FixedUtilityAlternative(Math.log(0))); + System.out.println("Composite utility of one negative infinite utility alternative and a '5' alt and an alternative with zero size is "+lm.getUtility()); + lm.monteCarloChoice(); + } catch (NoAlternativeAvailable e) { + System.out.println("No alternative available somewhere here..."); + } + + } +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java new file mode 100644 index 0000000..8d312b8 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java @@ -0,0 +1,64 @@ +package org.sandag.cvm.common.model; + +import java.util.ArrayList; + +import org.apache.log4j.Logger; + + +public class UtilityMaximizingChoiceModel extends + DiscreteChoiceModel { + + ArrayList alternatives = new ArrayList(); + + private static Logger logger = Logger.getLogger(UtilityMaximizingChoiceModel.class); + + public UtilityMaximizingChoiceModel(RandomVariable myRandomVariable) { + super(); + } + + @Override + public void addAlternative(Alternative a) { + alternatives.add(a); + } + + @Override + public void allocateQuantity(double amount) { + String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + @Override + public Alternative alternativeAt(int i) { + return alternatives.get(i); + } + + @Override + public double[] getChoiceProbabilities() { + String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + + public Alternative monteCarloChoice() { + int maxAlternative = 0; + double maxUtility = Double.NEGATIVE_INFINITY; + for (int i=0;imaxUtility) { + maxUtility = utility; + maxAlternative=i; + } + } + return alternatives.get(maxAlternative); + } + + @Override + public Alternative monteCarloChoice(double r) throws NoAlternativeAvailable { + String msg = this.getClass().toString()+" can't take a random number parameter"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java new file mode 100644 index 0000000..59f15f1 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java @@ -0,0 +1,84 @@ +package org.sandag.cvm.common.model; + +import java.util.ArrayList; + +import org.apache.log4j.Logger; + + +public class UtilityMaximizingChoiceModelWithErrorTermVector extends + DiscreteChoiceModel { + + ArrayList alternatives = new ArrayList(); + double[] errorTerms = null; + RandomVariable myRandomVariable; + + private static Logger logger = Logger.getLogger(UtilityMaximizingChoiceModelWithErrorTermVector.class); + + public UtilityMaximizingChoiceModelWithErrorTermVector(RandomVariable myRandomVariable) { + super(); + this.myRandomVariable = myRandomVariable; + } + + @Override + public void addAlternative(Alternative a) { + alternatives.add(a); + errorTerms = null; + } + + @Override + public void allocateQuantity(double amount) { + String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + @Override + public Alternative alternativeAt(int i) { + return alternatives.get(i); + } + + @Override + public double[] getChoiceProbabilities() { + String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + @Override + public Alternative monteCarloChoice() throws NoAlternativeAvailable { + if (errorTerms !=null) { + if (errorTerms.length != alternatives.size()) { + errorTerms = null; + } + } + if (errorTerms ==null) { + errorTerms = new double[alternatives.size()]; + for (int i =0;imaxUtility) { + maxUtility = utility; + maxAlternative=i; + } + } + return alternatives.get(maxAlternative); + } + + @Override + public Alternative monteCarloChoice(double r) throws NoAlternativeAvailable { + String msg = this.getClass().toString()+" can't take a random number parameter"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java new file mode 100644 index 0000000..7edb73f --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java @@ -0,0 +1,395 @@ +package org.sandag.cvm.common.skims; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.Enumeration; +import java.util.List; +import java.util.TreeSet; + +import javax.swing.tree.DefaultMutableTreeNode; + +import ncsa.hdf.object.CompoundDS; +import ncsa.hdf.object.FileFormat; +import ncsa.hdf.object.HObject; +import ncsa.hdf.object.h5.H5File; + +import org.apache.log4j.Logger; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixException; +import com.pb.common.matrix.MatrixReader; + + +public class HDF5MatrixReader extends MatrixReader { + + public static void main(String args[]) { + String[] skimsToGet = new String[] {"Light_Off","Light_AM","Light_Mid","Light_PM","Medium_Off", + "Medium_AM","Medium_Mid","Medium_PM","Heavy_Off","Heavy_AM","Heavy_Mid","Heavy_PM"}; + String[] nodes = new String[] {"cvm"}; + HDF5MatrixReader r = new HDF5MatrixReader(new File("/ProjectWork/CSTDM2009 105073x4/Technical/Skims and OD/skims.h5"), nodes, skimsToGet); + +// r.testMatrixFile(); + r.readMatrices(); + } + + static Logger logger = Logger.getLogger(HDF5MatrixReader.class); + + File hdf5File; + + private List nodeNames; + + private String[] initialMatrixNames; + + public HDF5MatrixReader(File hdf5File, String[] nodeNames, String[] matrixNames) { + this.nodeNames = Arrays.asList(nodeNames); + this.hdf5File = hdf5File; + this.initialMatrixNames = matrixNames; + } + + public HDF5MatrixReader(File file, String node) { + throw new RuntimeException("Not implemented yet"); + //FIXME implement read all skims from node + } + + /** + * Builds the user to sequential lookup table and the sequential to user lookup table + * @param origins + * @param destinations + * @return [0] is sequentialToUser, to lookup user zone numbers, [1] is userToSequential to lookup index. + */ + private static int[][] buildCrossLookups(int[] origins, int[] destinations) { + TreeSet zoneSet = new TreeSet(); + int mod = 0; + for (int o : origins) { + zoneSet.add(o); + if (++mod % 250000 == 0) logger.info(" Processed line "+mod+" origin "+o); + } + mod = 0; + for (int d : destinations) { + zoneSet.add(d); + if (++mod % 250000 == 0) logger.info(" Processed line "+mod+" destination "+d); + } + int maxZone = Collections.max(zoneSet); + int[][] userSequentialCrossLookup= new int[2][]; + int[] sequentialToUserLookup = new int[zoneSet.size()]; + userSequentialCrossLookup[0] = sequentialToUserLookup; + int[] userToSequentialLookup = new int[maxZone+1]; + userSequentialCrossLookup[1] = userToSequentialLookup; + int z=0; + for (int z1 : userToSequentialLookup) { + userToSequentialLookup[z++] = -1; + } + z=0; + for (int z2 : zoneSet) { + sequentialToUserLookup[z] = z2; + userToSequentialLookup[z2] = z++; + } + return userSequentialCrossLookup; + } + + + /** + * Check if the dataset contains a column with the correct name + * If it does return the index number, else return -1 + * Also if it does contain the column, select the column for retrieval + * @param hdfDataset the dataset to be checked + * @param name the name of the column to be checked and marked for retrieval + * @return the index of the column, -1 if the column was not present + */ + static int selectHDFFieldByName(CompoundDS hdfDataset, String name) { + List nameList = Arrays.asList(hdfDataset.getMemberNames()); + int index = nameList.indexOf(name); + if (index != -1) { + hdfDataset.selectMember(index); + } + return index; + } + + + static class intKeyString implements Comparable { + int myInt; + String myString; + boolean found = false; + + intKeyString(int i, String s) { + myInt = i; + myString = s; + } + + @Override + public int compareTo(Object o) { + int otherInt = ((intKeyString) o).myInt; + if (otherInt > myInt) return -1; + if (otherInt < myInt) return 1; + if (otherInt == myInt) return 0; + assert false; + return 0; + } + } + + + @Override + public Matrix[] readMatrices() throws MatrixException { + return readMatrices(initialMatrixNames); + } + + public Matrix[] readMatrices(String[] matrixNames) { + ArrayList matrixList = new ArrayList(); + FileFormat f = null; + + intKeyString[] skimIndices = new intKeyString[matrixNames.length]; + int j=0; + for (String skimName : matrixNames) { + skimIndices[j] = new intKeyString(-1, skimName); + j++; + } + + try { + f = new H5File(hdf5File.getAbsolutePath(), H5File.READ); + f.open(); + DefaultMutableTreeNode theRoot = (DefaultMutableTreeNode) f.getRootNode(); + if (theRoot == null) { + String msg= "Null root in HDF5 skim file "+hdf5File; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + + Enumeration local_enum = ((DefaultMutableTreeNode) theRoot).breadthFirstEnumeration(); + while (local_enum.hasMoreElements()) { + DefaultMutableTreeNode theNode = (DefaultMutableTreeNode) local_enum.nextElement(); + HObject theObj = (HObject) theNode.getUserObject(); + String theName = theObj.getName(); + if (nodeNames.contains(theName)){ + logger.info("Found object \""+theName+"\" in HDF5File "+hdf5File+", reading skims"); + if (!(theObj instanceof CompoundDS)) { + String msg = "object \""+theName+"\" in HDF5File "+hdf5File+" is not a compound dataset, can't read skims"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + theObj.getMetadata(); + CompoundDS skims = (CompoundDS) theObj; + assert skims.getRank() ==1 : "Skim object in HDF5 file should only be of rank 1"; + skims.setMemberSelection(false); + int originFieldIndex = selectHDFFieldByName(skims, "origin"); + assert originFieldIndex == 0 : "Origin needs to be first member in HDF skim dataset"; + int destinationFieldIndex = selectHDFFieldByName(skims, "destination"); + assert destinationFieldIndex == 1 : "Destination needs to be second member in HDF skim dataset"; + + // get origin and destination zones, get all rows from file but no content yet. + long[] selected = skims.getSelectedDims(); + logger.info("getting zone numbers from "+selected[0]+" rows of origins and destinations"); + + List odNumbers = (List) skims.read(); + assert odNumbers.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; + int[] origins = (int[]) odNumbers.get(0); + assert odNumbers.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; + int[] destinations = (int[]) odNumbers.get(1); + int[][] userSequentialCrossLookup = buildCrossLookups( + origins, destinations); + + odNumbers = null; // forget it so we can collect the memory with garbage collection + + int count = 0; + for (intKeyString identifier : skimIndices) { + int index = selectHDFFieldByName(skims,identifier.myString); + if (index >=0 ) { + // found + if (identifier.found) { + String msg = "found "+identifier.myString+" in more than one node in skim file, not sure which one to use"; + logger.fatal(msg); + throw new RuntimeException(msg); + } else { + logger.info("Reading "+identifier.myString+" from node "+theName); + identifier.myInt = index; + identifier.found = true; + count ++; + } + } else { + identifier.myInt = -1; + } + } + Arrays.sort(skimIndices); // important to sort them so we get them in the correct order below, -1 should be first; + + if (count ==0) { + logger.warn("No relevant skims in node "+theName+" skipping"); + } else { + + // allocate the storage for the arrays based on the number of skims and the number of zones + float[][][] matrixArrays = new float[count][userSequentialCrossLookup[0].length][userSequentialCrossLookup[0].length]; + + // now get content 100 rows at a time + final long HOWMANY = 250000; + long[] start = skims.getStartDims(); + selected[0] = HOWMANY; + long size = skims.getDims()[0]; + boolean verbose = false; + int startingCol = 0; + while (skimIndices[startingCol].myInt<0) startingCol++; + assert skimIndices.length-startingCol == count; + + for (long beginAt = 0; beginAt <= size; beginAt += HOWMANY) { + logger.info("Processing line "+beginAt+" from skims"); + start[0] = beginAt; + if (beginAt+HOWMANY >= size) // should be >=? + { + //verbose = true; + selected[0] = size - beginAt; + } + List skimData = (List) skims.read(); + assert skimData.size() == count+2; + assert skimData.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; + origins = (int[]) skimData.get(0); + assert skimData.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; + destinations = (int[]) skimData.get(1); + for (int r = 0; r < origins.length; r++ ) { + int originArrayIndex = userSequentialCrossLookup[1][origins[r]]; + int destinationArrayIndex = userSequentialCrossLookup[1][destinations[r]]; + if (verbose) System.out.println("processing origin "+origins[r]+","+destinations[r]+" (index "+originArrayIndex+","+destinationArrayIndex); + for (int col = 0; col + 2 < skimData.size(); col ++) { + matrixArrays[col][originArrayIndex][destinationArrayIndex] = ((float[]) skimData.get(col+2))[r]; + } + } + verbose = false; + + } + + + int[] externalZoneNumbers = new int[userSequentialCrossLookup[0].length+1]; + for(int k=1;k99999)) + matrixValue=0; + utility += coefficients[i]*matrixValue; + } + return utility; + } + if (travelConditions instanceof SomeSkims) { + lastSkims = (SomeSkims) travelConditions; + matrixIndices = new int[namesList.size()]; + for (int i=0;i99999)) + matrixValue=0; + components[i] = coefficients[i]*matrixValue; + } + return components; + } + if (travelConditions instanceof SomeSkims) { + lastSkims = (SomeSkims) travelConditions; + matrixIndices = new int[namesList.size()]; + for (int i=0;i2) { + String msg = "Matrix name "+name+" has more than 1 part"; + } + OMXMatrixReader r = new OMXMatrixReader(new File(directoryOfMatrices,split[0]+".omx")); + // if (split.length == 1) return r.readMatrix(0); + return r.readMatrix(split[1]); + } + + /* (non-Javadoc) + * @see com.pb.common.matrix.MatrixReader#readMatrix() + */ + @Override + public Matrix readMatrix() throws MatrixException { + throw new RuntimeException("Can't read OMX Matrix without specifying file_name:matrix_name"); + } + + /* (non-Javadoc) + * @see com.pb.common.matrix.MatrixReader#readMatrices() + */ + @Override + public Matrix[] readMatrices() throws MatrixException { + throw new RuntimeException("Can't read OMX Matrices without specifying the file name, this java class is to be used for an entire directory of files."); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java new file mode 100644 index 0000000..206e589 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java @@ -0,0 +1,437 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +package org.sandag.cvm.common.skims; + +import org.sandag.cvm.common.datafile.TableDataSet; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.MatrixType; +import com.pb.common.matrix.ZipMatrixReader; + +import drasys.or.util.Array; + +import ncsa.hdf.object.h5.H5CompoundDS; +import ncsa.hdf.object.FileFormat; +import ncsa.hdf.object.HObject; +import ncsa.hdf.object.h5.H5File; + +import org.apache.log4j.Logger; + +import java.io.File; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.Enumeration; +import java.util.List; +import java.util.TreeSet; + +import javax.swing.tree.DefaultMutableTreeNode; + +/** + * A class that reads in peak auto skims and facilitates zone pair disutility calculations + * @author John Abraham, Joel Freedman + * + */ +public class SomeSkims implements TravelAttributesInterface { + protected static Logger logger = Logger.getLogger("com.pb.models.pecas"); + + private ArrayList matrixList = new ArrayList(); + public Matrix[] matrices = new Matrix[0]; + private ArrayList matrixNameList = new ArrayList(); + + String my1stPath; + String my2ndPath; + + + public SomeSkims() { + my1stPath = System.getProperty("user.dir"); + } + + public SomeSkims(String firstPath, String secondPath) { + my1stPath = firstPath; + my2ndPath =secondPath; + }; + + public Matrix getMatrix(String name) { + int place = matrixNameList.indexOf(name); + if (place >=0) return matrices[place]; + return null; + } + + public void addZipMatrix(String matrixName) { + if (matrixNameList.contains(matrixName)) { + logger.info("SomeSkims already contains matrix named "+matrixName+", not reading it in again"); + } else { + File skim = new File(my1stPath+matrixName+".zip"); + if (!skim.exists()) skim = new File(my1stPath+matrixName+".zipMatrix"); + if(!skim.exists()) skim = new File(my1stPath+matrixName+".zmx"); + if(!skim.exists()){ + skim = new File(my2ndPath+matrixName+".zip"); + if(!skim.exists()) skim = new File(my2ndPath+matrixName+".zipMatrix"); + if(!skim.exists()) skim = new File(my2ndPath+matrixName+".zmx"); + if (!skim.exists()) { + logger.fatal("Could not find "+ matrixName+".zip, .zipMatrix or .zmx in either directory"); + throw new RuntimeException("Could not find "+ matrixName+".zip, .zipMatrix or .zmx in either directory"); + } + } + matrixList.add(new ZipMatrixReader(skim).readMatrix()); + matrixNameList.add(matrixName); + matrices = (Matrix[]) matrixList.toArray(matrices); + } + + if(logger.isDebugEnabled()) logger.debug("finished reading zipmatrix of skims "+matrixName+" into memory"); + } + + public void addTableDataSetSkims(TableDataSet s, String[] fieldsToAdd, int maxZoneNumber) { + addTableDataSetSkims(s, fieldsToAdd, maxZoneNumber, "origin", "destination"); + + } + + public void addMatrixFromFile(String fileName, String matrixName) { + File f = new File(fileName); + MatrixType type = MatrixReader.determineMatrixType(f); + if (type == null) { + logger.error("Can't determine matrix type for "+fileName); + } else { + MatrixReader r = MatrixReader.createReader(type,f); + Matrix m = r.readMatrix(); + matrixNameList.add(matrixName); + matrixList.add(m); + m.setName(matrixName); + matrices = (Matrix[]) matrixList.toArray(matrices); + } + + } + + public void addMatrix(Matrix m, String name) { + matrixNameList.add(name); + matrixList.add(m); + m.setName(name); + matrices = (Matrix[]) matrixList.toArray(matrices); + } + + public void addMatrixCSVSkims(TableDataSet s, String name) { + int rows = s.getRowCount(); + int columns = s.getColumnCount()-1; + if (rows!=columns) { + logger.fatal("Trying to add CSV Matrix Skims and number of columns does not equal number of rows"); + throw new RuntimeException("Trying to add CSV Matrix Skims and number of columns does not equal number of rows"); + } + float[][] tempArray = new float[rows][columns]; + int[] userToSequentialLookup = new int[rows+1]; + // check order of rows and columns + for (int check = 1;check < s.getRowCount();check++) { + if (!(s.getColumnLabel(check+1).equals(String.valueOf((int) (s.getValueAt(check,1)))))) { + logger.fatal("CSVMatrixSkims have columns out of order (needs to be the same as rows)"); + throw new RuntimeException("CSVMatrixSkims have columns out of order (needs to be the same as rows)"); + } + } + // TODO check for missing skims when using CSV format + for (int tdsRow = 1;tdsRow <= s.getRowCount();tdsRow++) { + userToSequentialLookup[tdsRow]=(int) s.getValueAt(tdsRow,1); + for (int tdsCol=2;tdsCol<=s.getColumnCount();tdsCol++) { + tempArray[tdsRow-1][tdsCol-2]=s.getValueAt(tdsRow,tdsCol); + } + } + Matrix m = new Matrix(name,"",tempArray); + matrixNameList.add(name); + m.setExternalNumbers(userToSequentialLookup); + this.matrixList.add(m); + matrices = (Matrix[]) matrixList.toArray(matrices); + } + + /** Adds a table data set of skims into the set of skims that are available + * + * @param s the table dataset of skims. There must be a column called "origin" + * and another column called "destination" + * @param fieldsToAdd the names of the fields from which to create matrices from, all other fields + * will be ignored. + */ + public void addTableDataSetSkims(TableDataSet s, String[] fieldsToAdd, int maxZoneNumber, String originFieldName, String destinationFieldName) { + int originField = s.checkColumnPosition(originFieldName); + int destinationField = s.checkColumnPosition(destinationFieldName); + int[] userToSequentialLookup = new int[maxZoneNumber]; + int[] sequentialToUserLookup = new int[maxZoneNumber]; + for (int i =0; i0) { + matrixArrays[entry][userToSequentialLookup[origin]][userToSequentialLookup[destination]] = s.getValueAt(row,fieldIds[entry]); + } + } + } + + for (int matrixToBeAdded =0; matrixToBeAdded < fieldsToAdd.length; matrixToBeAdded++) { + if (fieldIds[matrixToBeAdded]>0) { + matrixNameList.add(fieldsToAdd[matrixToBeAdded]); + Matrix m = new Matrix(fieldsToAdd[matrixToBeAdded],"",matrixArrays[matrixToBeAdded]); + m.setExternalNumbers(externalZoneNumbers); + this.matrixList.add(m); + } + } + + matrices = (Matrix[]) matrixList.toArray(matrices); + + logger.info("Finished reading TableDataSet skims "+s+" into memory"); + } + + + static int selectHDFFieldByName(H5CompoundDS hdfDataset, String name) { + List nameList = Arrays.asList(hdfDataset.getMemberNames()); + int index = nameList.indexOf(name); + if (index == -1) { + String msg = "No field of name "+name+" in HDF5 file node"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + hdfDataset.selectMember(index); + return index; + } + + + static class intKeyString implements Comparable { + int myInt; + String myString; + + intKeyString(int i, String s) { + myInt = i; + myString = s; + } + + @Override + public int compareTo(Object o) { + int otherInt = ((intKeyString) o).myInt; + if (otherInt > myInt) return -1; + if (otherInt < myInt) return 1; + if (otherInt == myInt) return 0; + assert false; + return 0; + } + } + + /** + * Adds some skims from an HDF5 file of skims into the set of skims that are available + * @param hdf5File + * @param nodeName + * @param fieldsToAdd + * @param maxZoneNumber + * @param originFieldName + * @param destinationFieldName + */ + public void addHDF5Skims(File hdf5File, String nodeName, String[] fieldsToAdd, int maxZoneNumber, String originFieldName, String destinationFieldName) { + logger.error("HDF5 Skims have not been tested yet, test SomeSkims.addHDF5Skims before using it"); + FileFormat f = null; + try { + f = new H5File(hdf5File.getAbsolutePath(), H5File.READ); + f.open(); + DefaultMutableTreeNode theRoot = (DefaultMutableTreeNode) f.getRootNode(); + if (theRoot == null) { + String msg= "Null root in HDF5 skim file "+hdf5File; + logger.fatal(msg); + throw new RuntimeException(msg); + } + + Enumeration local_enum = ((DefaultMutableTreeNode) theRoot).breadthFirstEnumeration(); + while (local_enum.hasMoreElements()) { + DefaultMutableTreeNode theNode = (DefaultMutableTreeNode) local_enum.nextElement(); + HObject theObj = (HObject) theNode.getUserObject(); + String theName = theObj.getName(); + if (theName.equals(nodeName)){ + logger.info("Found object \""+theName+"\" in HDF5File "+hdf5File+", reading skims"); + if (!(theObj instanceof H5CompoundDS)) { + String msg = "object \""+theName+"\" in HDF5File "+hdf5File+" is not a compound dataset, can't read skims"; + logger.fatal(msg); + throw new RuntimeException(msg); + } + H5CompoundDS skims = (H5CompoundDS) theObj; + assert skims.getRank() ==1 : "Skim object in HDF5 file should only be of rank 1"; + skims.setMemberSelection(false); + int originFieldIndex = selectHDFFieldByName(skims, originFieldName); + assert originFieldIndex == 0 : "Origin needs to be first member in HDF skim dataset"; + int destinationFieldIndex = selectHDFFieldByName(skims, destinationFieldName); + assert destinationFieldIndex == 1 : "Destination needs to be second member in HDF skim dataset"; + + // get origin and destination zones, get all rows from file but no content yet. + long[] selected = skims.getSelectedDims(); + logger.info("getting zone numbers from "+selected[0]+" rows of origins and destinations"); + + List odNumbers = (List) skims.read(); + assert odNumbers.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; + int[] origins = (int[]) odNumbers.get(0); + assert odNumbers.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; + int[] destinations = (int[]) odNumbers.get(1); + int[][] userSequentialCrossLookup = buildCrossLookups( + origins, destinations); + + odNumbers = null; // forget it so we can collect the memory with garbage collection + + intKeyString[] skimIndices = new intKeyString[fieldsToAdd.length]; + int i = 0; + for (String skimName : fieldsToAdd) { + skimIndices[i] = new intKeyString(selectHDFFieldByName(skims,skimName), skimName); + i++; + } + Arrays.sort(skimIndices); // important to sort them so we get them in the correct order below. + + float[][][] matrixArrays = new float[skimIndices.length][userSequentialCrossLookup[0].length][userSequentialCrossLookup[0].length]; + + // now get content 100 rows at a time + final long HOWMANY = 100; + long[] start = skims.getStartDims(); + selected[0] = HOWMANY; + long size = skims.getDims()[0]; + for (long beginAt = 0; beginAt <= size; beginAt += HOWMANY) { + if (beginAt+HOWMANY >= size) // should be >=? + { + selected[0] = size - beginAt; + } + List skimData = (List) skims.read(); + assert skimData.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; + origins = (int[]) skimData.get(0); + assert skimData.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; + destinations = (int[]) skimData.get(1); + for (int r = 0; r < origins.length; r++ ) { + int originArrayIndex = userSequentialCrossLookup[0][origins[r]]; + int destinationArrayIndex = userSequentialCrossLookup[0][destinations[r]]; + for (int col = 0; col + 2 < skimData.size(); col ++) { + matrixArrays[col][originArrayIndex][destinationArrayIndex] = ((float[]) skimData.get(col+2))[r]; + } + } + + } + + + int[] externalZoneNumbers = new int[userSequentialCrossLookup[0].length+1]; + for(int k=1;k zoneSet = new TreeSet(); + for (int o : origins) { + zoneSet.add(o); + } + for (int d : destinations) { + zoneSet.add(d); + } + int maxZone = Collections.max(zoneSet); + int[][] userSequentialCrossLookup= new int[2][]; + int[] sequentialToUserLookup = new int[zoneSet.size()]; + userSequentialCrossLookup[0] = sequentialToUserLookup; + int[] userToSequentialLookup = new int[maxZone+1]; + userSequentialCrossLookup[1] = userToSequentialLookup; + int z=0; + for (int z1 : userToSequentialLookup) { + userToSequentialLookup[z++] = -1; + } + z=0; + for (int z2 : zoneSet) { + sequentialToUserLookup[z] = z2; + userToSequentialLookup[z2] = z++; + } + return userSequentialCrossLookup; + } + + + public int getMatrixId(String string) { + + return matrixNameList.indexOf(string); + } + + + /** + * @param my1stPath The my1stPath to set. + */ + public void setMy1stPath(String my1stPath) { + this.my1stPath = my1stPath; + } + + /** + * @param my2ndPath The my2ndPath to set. + */ + public void setMy2ndPath(String my2ndPath) { + this.my2ndPath = my2ndPath; + } + +}; diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java new file mode 100644 index 0000000..64611ab --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java @@ -0,0 +1,67 @@ +/** + * + */ +package org.sandag.cvm.common.skims; + +import java.io.File; + +import org.apache.log4j.Logger; + +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixException; +import com.pb.common.matrix.MatrixReader; +import com.pb.common.matrix.TranscadMatrixReader; + +/** + * Reads a matrix by name with two parts separated by a colon. The first part is the file + * name (without the .mtx extension), the second part is the "core" name within the file. + * + * Transcad creates matrix files, but they are three dimensional matrices, with the z dimension being + * referred to as "cores". This allows treating these files as a list of two dimensional matrices + * @author johna + * + */ +public class TranscadMatrixCollectionReader extends MatrixReader { + + protected static Logger logger = Logger.getLogger(TranscadMatrixCollectionReader.class); + + protected File directoryOfMatrices; + + /** + * + */ + public TranscadMatrixCollectionReader(File directory) { + directoryOfMatrices = directory; + } + + /* (non-Javadoc) + * @see com.pb.common.matrix.MatrixReader#readMatrix(java.lang.String) + */ + @Override + public Matrix readMatrix(String name) throws MatrixException { + String[] split = name.split(":"); + if (split.length>2) { + String msg = "Matrix name "+name+" has more than 1 part"; + } + TranscadMatrixReader r = new TranscadMatrixReader(new File(directoryOfMatrices,split[0]+".mtx")); + if (split.length == 1) return r.readMatrix(0); + return r.readMatrix(split[1]); + } + + /* (non-Javadoc) + * @see com.pb.common.matrix.MatrixReader#readMatrix() + */ + @Override + public Matrix readMatrix() throws MatrixException { + throw new RuntimeException("Can't read Transcad Matrix without specifying file_name:matrix_name"); + } + + /* (non-Javadoc) + * @see com.pb.common.matrix.MatrixReader#readMatrices() + */ + @Override + public Matrix[] readMatrices() throws MatrixException { + throw new RuntimeException("Can't read Transcad Matrices without specifying the file name, this java class is to be used for an entire directory of files."); + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java new file mode 100644 index 0000000..208960a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java @@ -0,0 +1,23 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ +/* Generated by Together */ + +package org.sandag.cvm.common.skims; + +public interface TravelAttributesInterface { + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java new file mode 100644 index 0000000..15d787b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java @@ -0,0 +1,28 @@ +/* + * Copyright 2005 PB Consult 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. + * + */ + +package org.sandag.cvm.common.skims; + + +/** A class that represents how the preferences for travel by different modes and different times of day + * + * @author J. Abraham + */ +public interface TravelUtilityCalculatorInterface { + public double getUtility(Location origin, Location destination, TravelAttributesInterface travelConditions); + +} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java b/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java new file mode 100644 index 0000000..625e869 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java @@ -0,0 +1,121 @@ +package org.sandag.cvm.model.patternDetail; + +import java.util.Random; + +import org.apache.log4j.Logger; + +import org.sandag.cvm.common.model.GumbelErrorTerm; + +public class DestinationRandomTerms { + + protected static Logger logger = Logger.getLogger(DestinationRandomTerms.class); + + + static final Random random = new Random(); + static final double log4pi = Math.log(4*Math.PI); + + final int normalPoolSize = 100000; + /** + * This is the pool of standard normals, used for zones with size < useGumbel + */ + double[] normalPool = null; + + final int uniformPoolSize = 100000; + /** + * This is the pool of uniform distributed numbers, used with size > useGumbel + */ + double[] uniformPool = null; + + // TODO set useGumbel to something > 1 + /** + * if n > useGumbel, we'll use the Hall approximation to the Gumbel Distribution + */ + int useGumbel = 200; + + /** + * Gets a extreme normal random variable for a zone. If zone, poolOffset, poolSkip and n + * are the same it will return the same value. + * @param zone the destination zone under consideration + * @param poolOffset something stored in Preferences to distinguish decision makers + * @param poolSkip something stored in Preferences to distinguish decision makers + * @param n how large the population is in the zone + * @param stdDev the variance of the underlying normal distribution + * @return the sample from the extreme normal distribution + */ + public double getExtremeNormal(int zone, int poolOffset, int poolSkip, int n, double stdDev) { + if (poolSkip ==0) { + logger.warn("poolSkip is zero, setting it to 1"); + poolSkip = 1; + } + checkPools(); + if (stdDev == 0) return 0; + if (n 1.9999); + } + // let's check for n=20, see if we get expected within a certain tolerance + // these are the expected histograms from another experiment, with 4777 maximum random draws + // they are 0.1 apart and the first upper bound is 0.2 + int[] expectedHistograms = new int[]{ + 0, + 0, + 1, + 0, + 5, + 13, + 26, + 38, + 74, + 91, + 155, + 217, + 254, + 327, + 355, + 350, + 370, + 394, + 330, + 330, + 305, + 229, + 205, + 163, + 116, + 114, + 97, + 61, + 47, + 31, + 17, + 22, + 15, + 3, + 10, + 2, + 3, + 4, + 1, + 0, + 0, + 0, + 0, + 0, + 2, + 0, + 0, + 0, + 0}; + int offSet = (int) (Math.random()*10000); + int skip = (int) (Math.random()*10); + for (int i=0;i<4777;i++) { + int fakeZone = (int) (Math.random()*15000); + double sample1 = sampler.getExtremeNormal(fakeZone, offSet, skip, 20, 1); + int bin = (int) ((sample1-0.1)/0.1); + if (bin <0) bin =0; + if (bin >= expectedHistograms.length) bin = expectedHistograms.length-1; + expectedHistograms[bin]--; + } + // now check to see if maximum diff < 70; + int maximumDiff = 0; + for (int i=0;i= expectedHistograms.length) bin = expectedHistograms.length-1; + expectedHistograms[bin]--; + } + // now check to see if maximum diff < 100; + maximumDiff = 0; + for (int i=0;i= expectedHistograms.length) bin = expectedHistograms.length-1; + expectedHistograms[bin]--; + } + // now check to see if maximum diff < 5; + maximumDiff = 0; + for (int i=0;i cntFlows; + private Matrix[] truckFlowsSUT; + private Matrix[] truckFlowsMUT; + private Matrix emptySUT; + private Matrix emptyMUT; + + + public SandagCountyModel(ReadFAF4 faf4, disaggregateFlows df) { + // constructor + this.faf4 = faf4; + this.df = df; + + } + + + public void runSandagCountyModel () { + // run model to disaggregate flows from FAF zones to counties + + logger.info("Model to disaggregate flows from FAF zones to counties"); + + df.getUScountyEmploymentByIndustry(utilities.getRb()); + utilities.createZoneList(); + faf4.readAllData(utilities.getRb(), utilities.getYear(), "tons"); + + faf4.definePortsOfEntry(utilities.getRb()); + if (utilities.getBooleanProperty("read.in.raw.faf.data", true)) extractTruckData(); + disaggregateFromFafToCounties(); + + convertTonsToTrucks cttt = new convertTonsToTrucks(utilities.getRb()); + cttt.readData(); + convertTonsToTrucks(cttt); + addEmptyTrucks(); + writeCountyTripTables(); + } + + + private void extractTruckData() { + // extract truck data and write flows to file + logger.info("Extracting FAF truck data"); + String[] scaleTokens = ResourceUtil.getArray(utilities.getRb(), "scaling.truck.trips.tokens"); + double[] scaleValues = ResourceUtil.getDoubleArray(utilities.getRb(), "scaling.truck.trips.values"); + HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); + String truckFileNameT = ResourceUtil.getProperty(utilities.getRb(), "processed.truck.faf.data") + "_" + + utilities.getYear(); + + // create output directory if it does not exist yet + File file = new File ("output/temp"); + if (!file.exists()) { + boolean outputDirectorySuccessfullyCreated = file.mkdir(); + if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); + } + faf4.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); + } + + + private void disaggregateFromFafToCounties() { + // disaggregates freight flows from FAF zoneArray to counties + + logger.info(" Disaggregating FAF data from FAF zones to counties for year " + utilities.getYear() + "."); + + int matrixSize = utilities.countyFips.length; + cntFlows = new HashMap<>(); + + float globalScale = (float) ResourceUtil.getDoubleProperty(utilities.getRb(), "overall.scaling.factor.truck"); + + // regular method + for (String com: ReadFAF4.sctgStringCommodities) { + float[][] dummy = new float[matrixSize][matrixSize]; + cntFlows.put(com, dummy); + } + df.prepareCountyDataForFAFwithDetailedEmployment(utilities.getRb(), utilities.getYear(), false); + df.scaleSelectedCounties(utilities.getRb()); + + java.util.concurrent.ForkJoinPool pool = new java.util.concurrent.ForkJoinPool(); + DnCRecursiveAction action = new DissaggregateFafAction(globalScale); + pool.execute(action); + action.getResult(); + } + + + private class DissaggregateFafAction extends DnCRecursiveAction { + private final float globalScale; + + private DissaggregateFafAction(float globalScale) { + super(0,ReadFAF4.sctgStringCommodities.length); + this.globalScale = globalScale; + } + + private DissaggregateFafAction(float globalScale, long start, long length, DnCRecursiveAction next) { + super(start,length,next); + this.globalScale = globalScale; + } + + @Override + protected void computeAction(long start, long length) { + long end = start + length; + for (int comm = (int) start; comm < end; comm++) { + int cm = ReadFAF4.sctgCommodities[comm]; + + String fileName = ResourceUtil.getProperty(utilities.getRb(), "processed.truck.faf.data") + "_" + utilities.getYear(); + if (cm < 10) fileName = fileName + "_SCTG0" + cm + ".csv"; + else fileName = fileName + "_SCTG" + cm + ".csv"; + logger.info(" Working on " + fileName); + String sctg = ReadFAF4.getSCTGname(cm); + float[][] values = cntFlows.get(sctg); + TableDataSet tblFlows = fafUtils.importTable(fileName); + for (int row = 1; row <= tblFlows.getRowCount(); row++) { + float shortTons = tblFlows.getValueAt(row, "shortTons"); + if (shortTons == 0) continue; + String dir = tblFlows.getStringValueAt(row, "flowDirection"); + int orig = (int) tblFlows.getValueAt(row, "originFAF"); + int dest = (int) tblFlows.getValueAt(row, "destinationFAF"); + TableDataSet singleFlow; + if (dir.startsWith("import") || dir.startsWith("export")) { + TableDataSet poe = null; + // Entry through land border + switch (dir) { + case "import": + poe = ReadFAF4.getPortsOfEntry(orig); + break; + // Entry through marine port + case "import_port": + poe = ReadFAF4.getMarinePortsOfEntry(orig); + break; + // Entry through airport + case "import_airport": + poe = ReadFAF4.getAirPortsOfEntry(orig); + break; + // Exit through land border + case "export": + poe = ReadFAF4.getPortsOfEntry(dest); + break; + // Exit through marine port + case "export_port": + poe = ReadFAF4.getMarinePortsOfEntry(dest); + break; + // Exit through airport + case "export_airport": + poe = ReadFAF4.getAirPortsOfEntry(dest); + break; + } + singleFlow = df.disaggregateSingleFAFFlowThroughPOE(dir, poe, orig, dest, sctg, shortTons, 1); + } else singleFlow = df.disaggregateSingleFAFFlow(orig, dest, sctg, shortTons, 1); + for (int i = 1; i <= singleFlow.getRowCount(); i++) { + int oFips = (int) singleFlow.getValueAt(i, "oFips"); + int oZone = utilities.countyFipsIndex[oFips]; + int dFips = (int) singleFlow.getValueAt(i, "dFips"); + int dZone = utilities.countyFipsIndex[dFips]; + float thisFlow = singleFlow.getValueAt(i, "Tons") * globalScale; + values[oZone][dZone] += thisFlow; + } + } + } + } + + @Override + protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) { + return new DissaggregateFafAction(globalScale,start,length,next); + } + + @Override + protected boolean continueDividing(long length) { + return getSurplusQueuedTaskCount() < 3 && length > 1; + } + } + + + private void convertTonsToTrucks (convertTonsToTrucks cttt) { + // convert flows in tons into flows in trucks using average payload factors + + logger.info(" Converting tons into trucks"); + + int highestGroupCode = utilities.getHighestVal(utilities.getCommodityGroupOfSCTG()); + truckFlowsSUT = new Matrix[highestGroupCode + 1]; + truckFlowsMUT = new Matrix[highestGroupCode + 1]; + float aawdtFactor = (float) ResourceUtil.getDoubleProperty(utilities.getRb(), "AADT.to.AAWDT.factor"); + for (int i = 0; i <= highestGroupCode; i++) { + truckFlowsSUT[i] = createCountyMatrix(); + truckFlowsMUT[i] = createCountyMatrix(); + } + + for (String com: ReadFAF4.sctgStringCommodities) { + int comGroup = utilities.getCommodityGroupOfSCTG()[Integer.parseInt(com.substring(4))]; + float[][] flowsThisCommodity = cntFlows.get(com); + for (int oFips: utilities.countyFips) { + for (int dFips: utilities.countyFips) { + int oZone = utilities.countyFipsIndex[oFips]; + int dZone = utilities.countyFipsIndex[dFips]; + float distance = df.getCountyDistance(oFips, dFips); + float truckByType[] = cttt.convertThisFlowFromTonsToTrucks(com, distance, flowsThisCommodity[oZone][dZone]); + float oldValueSUT = truckFlowsSUT[comGroup].getValueAt(oFips, dFips); + float newValueSut = truckByType[0] / 365.25f * aawdtFactor; + truckFlowsSUT[comGroup].setValueAt(oFips, dFips, oldValueSUT + newValueSut); + float oldValueMUT = truckFlowsMUT[comGroup].getValueAt(oFips, dFips); + float newValueMUT = (truckByType[1] + truckByType[2] + truckByType[3]) / 365.25f * aawdtFactor; + truckFlowsMUT[comGroup].setValueAt(oFips, dFips, oldValueMUT + newValueMUT); + } + } + } + } + + + private Matrix createCountyMatrix() { + Matrix mat = new Matrix(utilities.countyFips.length, utilities.countyFips.length); + mat.setExternalNumbersZeroBased(utilities.countyFips); + return mat; + } + + private void addEmptyTrucks() { + // Empty truck model to ensure balanced truck volumes entering and leaving every zone + + double emptyRate = 1f - ResourceUtil.getDoubleProperty(utilities.getRb(), "empty.truck.rate"); + + int highestGroupCode = utilities.getHighestVal(utilities.getCommodityGroupOfSCTG()); + double[] balSut = new double[utilities.countyFips.length]; + double[] balMut = new double[utilities.countyFips.length]; + Matrix loadedSutTot = createCountyMatrix(); + Matrix loadedMutTot = createCountyMatrix(); + for (int orig = 0; orig < utilities.countyFips.length; orig++) { + for (int dest = 0; dest < utilities.countyFips.length; dest++) { + for (int comGroup = 0; comGroup <= highestGroupCode; comGroup++) { + float sut = truckFlowsSUT[comGroup].getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]); + float mut = truckFlowsMUT[comGroup].getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]); + balSut[orig] -= sut; + balSut[dest] += sut; + balMut[orig] -= mut; + balMut[dest] += mut; + loadedSutTot.setValueAt(utilities.countyFips[orig], utilities.countyFips[dest], + (loadedSutTot.getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]) + sut)); + loadedMutTot.setValueAt(utilities.countyFips[orig], utilities.countyFips[dest], + (loadedMutTot.getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]) + mut)); + } + } + } + Matrix emptyBalancedSut = balanceEmpties(balSut); + Matrix emptyBalancedMut = balanceEmpties(balMut); + double targetSut = loadedSutTot.getSum() / emptyRate; + double targetMut = loadedMutTot.getSum() / emptyRate; + double emptySutRetTot = emptyBalancedSut.getSum(); + double emptyMutRetTot = emptyBalancedMut.getSum(); + + logger.info(" Trucks generated by commodity flows: " + Math.round(loadedSutTot.getSum()) + " SUT and " + + Math.round(loadedMutTot.getSum()) + " MUT."); + logger.info(" Empty trucks generated by balancing: " + Math.round((float) emptySutRetTot) + " SUT and " + + Math.round((float) emptyMutRetTot) + " MUT."); + double correctedEmptyTruckRate = emptyRate + (emptySutRetTot + emptyMutRetTot) / (targetSut + targetMut); + if (correctedEmptyTruckRate < 0) logger.warn("Empty truck rate for returning trucks is with " + + utilities.rounder(((emptySutRetTot + emptyMutRetTot) / (targetSut + targetMut)), 2) + + " greater than global empty-truck rate of " + utilities.rounder(emptyRate, 2)); + logger.info(" Empty trucks added by statistics: " + Math.round((float) ((1 - correctedEmptyTruckRate) * targetSut)) + + " SUT and " + Math.round((float) ((1 - correctedEmptyTruckRate) * targetMut)) + " MUT."); + + emptySUT = createCountyMatrix(); + emptyMUT = createCountyMatrix(); + for (int origin : utilities.countyFips) { + for (int destination : utilities.countyFips) { + float emptySutReturn = emptyBalancedSut.getValueAt(destination, origin); // note: orig and dest are switched to get return trip + float emptyMutReturn = emptyBalancedMut.getValueAt(destination, origin); // note: orig and dest are switched to get return trip + double emptySutStat = (loadedSutTot.getValueAt(origin, destination) + emptySutReturn) / correctedEmptyTruckRate - + (loadedSutTot.getValueAt(origin, destination) + emptySutReturn); + double emptyMutStat = (loadedMutTot.getValueAt(origin, destination) + emptyMutReturn) / correctedEmptyTruckRate - + (loadedMutTot.getValueAt(origin, destination) + emptyMutReturn); + + emptySUT.setValueAt(origin, destination, (float) (emptySutReturn + emptySutStat)); + emptyMUT.setValueAt(origin, destination, (float) (emptyMutReturn + emptyMutStat)); + } + } + } + + + private Matrix balanceEmpties(double[] trucks) { + // generate empty truck trips + + RowVector emptyTruckDest = new RowVector(utilities.countyFips.length); + emptyTruckDest.setExternalNumbersZeroBased(utilities.countyFips); + ColumnVector emptyTruckOrig = new ColumnVector(utilities.countyFips.length); + emptyTruckOrig.setExternalNumbersZeroBased(utilities.countyFips); + for (int zn = 0; zn < utilities.countyFips.length; zn++) { + if (trucks[zn] > 0) { + emptyTruckDest.setValueAt(utilities.countyFips[zn], (float) trucks[zn]); + emptyTruckOrig.setValueAt(utilities.countyFips[zn], 0f); + } + else { + emptyTruckOrig.setValueAt(utilities.countyFips[zn], (float) trucks[zn]); + emptyTruckDest.setValueAt(utilities.countyFips[zn], 0f); + } + } + Matrix seed = createCountyMatrix(); + for (int o: utilities.countyFips) { + for (int d: utilities.countyFips) { + float friction = (float) Math.exp(-0.001 * df.getCountyDistance(o, d)); + seed.setValueAt(o, d, friction); + } + } + MatrixBalancerRM mb = new MatrixBalancerRM(seed, emptyTruckOrig, emptyTruckDest, 0.001, 10, MatrixBalancerRM.ADJUST.BOTH_USING_AVERAGE); + return mb.balance(); + } + + + private void writeCountyTripTables() { + // write out county-to-county trip tables + + logger.info(" Writing county-to-county truck trip table"); + String fileName = utilities.getRb().getString("county.to.county.trip.table") + "_" + utilities.getYear() + ".csv"; + PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); + pw.println("origFips,destFips,sut1,sut2,sut3,sut4,sut5,sut6,emptySut,mut1,mut2,mut3,mut4,mut5,mut6,emptyMut"); + for (int oFips: utilities.countyFips) { + for (int dFips: utilities.countyFips) { + pw.print(oFips+","+dFips); + for (int i = 1; i < truckFlowsSUT.length; i++) pw.print("," + truckFlowsSUT[i].getValueAt(oFips,dFips)); + pw.print("," + emptySUT.getValueAt(oFips, dFips)); + for (int i = 1; i < truckFlowsMUT.length; i++) pw.print("," + truckFlowsMUT[i].getValueAt(oFips,dFips)); + pw.println("," + emptyMUT.getValueAt(oFips, dFips)); + } + } + pw.close(); + } + + + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java b/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java new file mode 100644 index 0000000..0cd3513 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java @@ -0,0 +1,118 @@ +package org.sandag.htm.applications; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import org.sandag.htm.processFAF.countyTruckModel; +import org.sandag.htm.processFAF.fafUtils; +import org.sandag.htm.processFAF.readFAF3; + +import org.apache.log4j.Logger; + +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; + +/** + * Application of countyTruckModel for Ohio State + * Author: Rolf Moeckel, PB Albuquerque + * Date: June 15, 2012 (Chicago IL) + + */ +public class ohio { + + private static Logger logger = Logger.getLogger(ohio.class); + private static String[] listOfRailRegions; + private static String[] railRegionReference; + private static boolean[] relevantCommodities; + + public static void summarizeFAFData (ResourceBundle appRb, int[] countyFips) { + // Summarize commodity flows of FAF data + +// if (!ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data")) { +// logger.error("Cannot summarize data for Ohio, set \"read.in.raw.faf.data\" to true."); +// return; +// } + if (ResourceUtil.getBooleanProperty(appRb, "summarize.by.ohio.rail.zones")) { + readOhioRailRegions(appRb, countyFips); + readRelevantCommodities(appRb); + } + + } + + + private static void readOhioRailRegions (ResourceBundle appRb, int[] countyFips) { + // create reference between fips code and Ohio Rail Region + + logger.info("Reading Ohio Rail Regions"); + TableDataSet railRegions = fafUtils.importTable(appRb.getString("rail.zone.definition")); + int highestFips = fafUtils.getHighestVal(countyFips); + + railRegionReference = new String[highestFips + 1]; + for (int row = 1; row <= railRegions.getRowCount(); row++) { + int fips = (int) railRegions.getValueAt(row, "fips"); + String reg = railRegions.getStringValueAt(row, "ohioRailRegion"); + railRegionReference[fips] = reg; + } + listOfRailRegions = fafUtils.getUniqueListOfValues(railRegionReference); + } + + + private static void readRelevantCommodities (ResourceBundle appRb) { + // Read how commodities are grouped by SCTG cagegory + + TableDataSet comGroups = fafUtils.importTable(appRb.getString("commodity.grouping")); + relevantCommodities = new boolean[fafUtils.getHighestVal(readFAF3.sctgCommodities) + 1]; + for (int i = 0; i < relevantCommodities.length; i++) relevantCommodities[i] = false; + for (int row = 1; row <= comGroups.getRowCount(); row++) { + int sctg = (int) comGroups.getValueAt(row, "SCTG"); + String truckType = comGroups.getStringValueAt(row, "MainTruckType"); + relevantCommodities[sctg] = truckType.equals("Van"); // set relevantCommodities to true if truckType equals Van + } + } + + + public static void sumFlowByRailZone(ResourceBundle appRb, int year, int[] countyFips, int[] countyIndex, HashMap cntFlows) { + // summarize flows by rail regions + + // Step 1: Initialize counter + HashMap railRegionIndex = new HashMap<>(); + int regionCounter = 0; + for (String txt: listOfRailRegions) { + railRegionIndex.put(txt,regionCounter); + regionCounter++; + } + double[][] summaryRailRegions = new double[listOfRailRegions.length][listOfRailRegions.length]; + + // Step 2: Summarize flows + String[] commodities = readFAF3.sctgStringCommodities; + for (String com: commodities) { + if (!relevantCommodities[Integer.parseInt(com.substring(4))]) continue; + float[][] flows = cntFlows.get(com); + for (int oFips: countyFips) { + if (railRegionReference[oFips] != null) { + int origRailRegion = railRegionIndex.get(railRegionReference[oFips]); + for (int dFips: countyFips) { + if (railRegionReference[dFips] != null) { + int destRailRegion = railRegionIndex.get(railRegionReference[dFips]); + summaryRailRegions[origRailRegion][destRailRegion] += flows[countyIndex[oFips]][countyIndex[dFips]]; + } + } + } + } + } + + PrintWriter pw = fafUtils.openFileForSequentialWriting(appRb.getString("rail.zone.output") + "_" + year + ".csv"); + + pw.print("Region"); + for (String txt: listOfRailRegions) pw.print("," + txt); + pw.println(); + for (String orig: listOfRailRegions) { + pw.print(orig); + for (String dest: listOfRailRegions) { + pw.print("," + summaryRailRegions[railRegionIndex.get(orig)][railRegionIndex.get(dest)]); + } + pw.println(); + } + pw.close(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java b/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java new file mode 100644 index 0000000..1e5e07d --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java @@ -0,0 +1,338 @@ +package org.sandag.htm.applications; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import org.sandag.htm.processFAF.disaggregateFlows; +import org.sandag.htm.processFAF.fafUtils; +import com.pb.sawdust.calculator.Function1; +import com.pb.sawdust.util.array.ArrayUtil; +import com.pb.sawdust.util.concurrent.ForkJoinPoolFactory; +import com.pb.sawdust.util.concurrent.IteratorAction; +import org.apache.log4j.Logger; + +import java.io.PrintWriter; +import java.util.*; +import java.util.concurrent.ForkJoinPool; + +/** + * SANDAG external truck model + * Class to disaggregate FAF flows from counties to SANDAG zones + * Author: Rolf Moeckel, PB Albuquerque + * Date: 6 March 2013 (Santa Fe, NM) + * Version 1.0 + */ + +public class sandagZonalModel { + + private static Logger logger = Logger.getLogger(sandagZonalModel.class); + private int[] zones; + private HashMap useHshLocal; + private HashMap makeHshLocal; + private String[] industries; + private float[][] zonalEmployment; + private TableDataSet countyFlows; + private int[] sandagExtStatID; + private boolean[] sandagExtStat_border; + private int[] sanDiegoNodes; + private final HashMap disaggregatedFlows = new HashMap<>(); + private String[] todNames; + private float[][] todSutShare; + private float[][] todMutShare; + + + public sandagZonalModel() { + // constructor + } + + + public void runSandagZonalModel () { + // run model to disaggregate flows from counties to SANDAG zones + + logger.info("Model to disaggregate flows from counties to zones"); + + readInputData(); + disaggregateFlowsFromCountiesToZones(); + writeOutDisaggregatedFlows(); + } + + + private void readInputData() { + // read input data + + logger.info(" Reading input data"); + // define TAZ and MGRA system + TableDataSet zonesTbl = fafUtils.importTable(utilities.getRb().getString("local.zones")); + zones = zonesTbl.getColumnAsInt("TAZ"); + TableDataSet mgraZones = fafUtils.importTable(utilities.getRb().getString("local.mgra.zones")); + int[] mgras = mgraZones.getColumnAsInt("mgra13"); + int[] mgraTazReference = new int[fafUtils.getHighestVal(mgras) + 1]; + for (int row = 1; row <= mgraZones.getRowCount(); row++) { + mgraTazReference[(int) mgraZones.getValueAt(row, "mgra13")] = (int) mgraZones.getValueAt(row, "taz13"); + } + + // process local make/use coefficients + String useToken = "faf.use.coefficients.local"; + String makeToken = "faf.make.coefficients.local"; + useHshLocal = disaggregateFlows.createMakeUseHashMap(utilities.getRb(), useToken); + makeHshLocal = disaggregateFlows.createMakeUseHashMap(utilities.getRb(), makeToken); + TableDataSet useCoeff = fafUtils.importTable(utilities.getRb().getString(useToken)); + industries = useCoeff.getColumnAsString("Industry"); + + // read local employment data + TableDataSet employmentMGRA = fafUtils.importTable(utilities.getRb().getString("local.employment.data") + + utilities.getYear() + ".csv"); + zonalEmployment = new float[utilities.getHighestVal(zones)+1][industries.length]; + for (int row = 1; row <= employmentMGRA.getRowCount(); row++) { + int mgra = (int) employmentMGRA.getValueAt(row, "mgra"); + int taz = mgraTazReference[mgra]; + for (int ind = 0; ind < industries.length; ind++) { + zonalEmployment[taz][ind] += employmentMGRA.getValueAt(row, industries[ind]); + } + } + + // read external station IDs + TableDataSet extStations = fafUtils.importTable(utilities.getRb().getString("external.station.definition")); + sandagExtStatID = new int[utilities.getHighestVal(extStations.getColumnAsInt("natExtStat")) + 1]; + sandagExtStat_border = new boolean[utilities.getHighestVal(zones) + 1]; + for (int row = 1; row <= extStations.getRowCount(); row++) { + sandagExtStatID[(int) extStations.getValueAt(row, "natExtStat")] = + (int) extStations.getValueAt(row, "sandagExtStat"); + sandagExtStat_border[(int) extStations.getValueAt(row, "sandagExtStat")] = + extStations.getBooleanValueAt(row, "borderCrossing"); + } + + // read flows at external stations + String fileName = utilities.getRb().getString("external.station.flows"); + countyFlows = utilities.importTableFromDBF(fileName); + if (countyFlows.getColumnCount() != 16) logger.error("Excepted 16 but found " + countyFlows.getColumnCount() + + " columns in " + fileName); + String[] expectedLabels = {"SUBAREA_NO","SUBAREA_N1","DEMAND_SUT","DEMAND_SU1","DEMAND_SU2","DEMAND_SU3","DEMAND_SU4","DEMAND_SU5","DEMAND_EMP","DEMAND_MUT","DEMAND_MU1","DEMAND_MU2","DEMAND_MU3","DEMAND_MU4","DEMAND_MU5","DEMAND_EM1"}; + String[] actualLabels = countyFlows.getColumnLabels(); + boolean wrongHeader = false; + for (int col = 0; col < countyFlows.getColumnCount(); col++) { + if (!actualLabels[col].equalsIgnoreCase(expectedLabels[col])) + wrongHeader = true; + } + if (wrongHeader) { + logger.error("File " + fileName + " has unexpected headers:"); + logger.info("Column,ExpectedLabel,ActualLabel"); + for (int col = 0; col < countyFlows.getColumnCount(); col++) + logger.info(col+1 + "," + expectedLabels[col] + "," + actualLabels[col] + "," + (expectedLabels[col].equals(actualLabels[col]))); + System.exit(1); + } + countyFlows.setColumnLabels(new String[]{"orig","dest","sut1","sut2","sut3","sut4","sut5","sut6","sutEmpty", + "mut1","mut2","mut3","mut4","mut5","mut6","mutEmpty"}); + + sanDiegoNodes = ResourceUtil.getIntegerArray(utilities.getRb(), "internal.nodes.san.diego"); + + // read time-of-day shares + TableDataSet TODValuesGeneral = fafUtils.importTable(utilities.getRb().getString("time.of.day.shares.general")); + TableDataSet TODValuesBorder = fafUtils.importTable(utilities.getRb().getString("time.of.day.shares.border")); + todNames = TODValuesGeneral.getColumnAsString("DESCRIPTION"); + todSutShare = new float[2][TODValuesGeneral.getRowCount()]; + todMutShare = new float[2][TODValuesGeneral.getRowCount()]; + float[] checkSum = new float[4] ; + for (int row = 1; row <= TODValuesGeneral.getRowCount(); row++) { + todSutShare[0][row-1] = TODValuesGeneral.getValueAt(row, "ShareSUT"); + todMutShare[0][row-1] = TODValuesGeneral.getValueAt(row, "ShareMUT"); + todSutShare[1][row-1] = TODValuesBorder.getValueAt(row, "ShareSUT"); + todMutShare[1][row-1] = TODValuesBorder.getValueAt(row, "ShareMUT"); + checkSum[0] += todSutShare[0][row-1]; + checkSum[1] += todMutShare[0][row-1]; + checkSum[2] += todSutShare[1][row-1]; + checkSum[3] += todMutShare[1][row-1]; + } + if (checkSum[0] > 1.001 || checkSum[0] < 0.999) logger.warn("Time of day share for SUT (general) does not add up to 1 but " + checkSum[0]); + if (checkSum[1] > 1.001 || checkSum[1] < 0.999) logger.warn("Time of day share for MUT (general) does not add up to 1 but " + checkSum[1]); + if (checkSum[2] > 1.001 || checkSum[2] < 0.999) logger.warn("Time of day share for SUT (border) does not add up to 1 but " + checkSum[2]); + if (checkSum[3] > 1.001 || checkSum[3] < 0.999) logger.warn("Time of day share for MUT (border) does not add up to 1 but " + checkSum[3]); + } + + + private void disaggregateFlowsFromCountiesToZones() { + // calculate local weights and disaggregate flows from counties/external stations to zones/external stations + + final HashMap weights = prepareZonalWeights(); + logger.info(" Disaggregating flows to zones"); + + int[] listOfCommodityGroupsPlusEmpties = new int[utilities.getListOfCommodityGroups().length + 1]; + listOfCommodityGroupsPlusEmpties[0] = 0; // category for empty trucks + System.arraycopy(utilities.getListOfCommodityGroups(), 0, listOfCommodityGroupsPlusEmpties, 1, utilities.getListOfCommodityGroups().length); + Integer[] list = new Integer[listOfCommodityGroupsPlusEmpties.length]; + for (int i = 0; i < listOfCommodityGroupsPlusEmpties.length; i++) list[i] = listOfCommodityGroupsPlusEmpties[i]; + Function1 commodityDisaggregationFunctionFAF = new Function1() { + public Void apply(Integer com) { + processCommodityDisaggregation(com, weights); + return null; + } + }; + + Iterator commodityIterator = ArrayUtil.getIterator(list); + IteratorAction itTask = new IteratorAction<>(commodityIterator, commodityDisaggregationFunctionFAF); + ForkJoinPool pool = ForkJoinPoolFactory.getForkJoinPool(); + pool.execute(itTask); + itTask.waitForCompletion(); + } + + + + + private HashMap prepareZonalWeights() { + // prepare zonal weights based on employment by industry + + logger.info(" Calculating zonal weights"); + HashMap weights = new HashMap<>(); + + double[] makeEmpty = new double[zones.length]; + double[] useEmpty = new double[zones.length]; + + for (int comGrp: utilities.getListOfCommodityGroups()) { + double[] makeWeight = new double[zones.length]; + double[] useWeight = new double[zones.length]; + for (int com: utilities.getComGroupDefinition().get(comGrp)) { + for (int iz = 0; iz < zones.length; iz++) { + int zn = zones[iz]; + for (int ind = 0; ind < industries.length; ind++) { + String industry = industries[ind]; + String code; + if (com <= 9) code = industry + "_SCTG0" + com; + else code = industry + "_SCTG" + com; + makeWeight[iz] += zonalEmployment[zn][ind] * makeHshLocal.get(code); + useWeight[iz] += zonalEmployment[zn][ind] * useHshLocal.get(code); + makeEmpty[iz] += zonalEmployment[zn][ind] * makeHshLocal.get(code); + useEmpty[iz] += zonalEmployment[zn][ind] * useHshLocal.get(code); + } + } + } + String mCode = comGrp + "_make"; + weights.put(mCode, makeWeight); + String uCode = comGrp + "_use"; + weights.put(uCode, useWeight); + } + weights.put("0_make", makeEmpty); + weights.put("0_use", useEmpty); + return weights; + } + + + + private boolean checkIfCountyInSanDiego(int node) { + // check if county is either San Diego County of San Diego Phantom County + + boolean insideSandag = false; + for (int i: sanDiegoNodes) if (i == node) insideSandag = true; + return insideSandag; + } + + + private void processCommodityDisaggregation(Integer comGroup, Map weights) { + // Disaggregate a single commodity from county-to-county flows to zone-to-zone flows + logger.info(" Processing commodity group " + comGroup); + + ArrayList origAL = new ArrayList<>(); + ArrayList destAL = new ArrayList<>(); + ArrayList flowSutAL = new ArrayList<>(); + ArrayList flowMutAL = new ArrayList<>(); + for (int row = 1; row <= countyFlows.getRowCount(); row ++) { + int orig = (int) countyFlows.getValueAt(row, "orig"); + int dest = (int) countyFlows.getValueAt(row, "dest"); + String sutLabel; + if (comGroup == 0) sutLabel = "sutEmpty"; + else sutLabel = "sut" + comGroup; + float sutTrk = countyFlows.getValueAt(row, sutLabel); + String mutLabel; + if (comGroup == 0) mutLabel = "mutEmpty"; + else mutLabel = "mut" + comGroup; + float mutTrk = countyFlows.getValueAt(row, mutLabel); + + if (checkIfCountyInSanDiego(orig) && checkIfCountyInSanDiego(dest)) { + // flows from San Diego County to San Diego Phantom County or vice versa + // ignore as these flows are internal to SANDAG and known to be underestimated + } else if (checkIfCountyInSanDiego(orig)) { + // flows from San Diego to elsewhere + double[] makeShare = weights.get(comGroup + "_make"); + double makeShareSum = utilities.getSum(makeShare); + for (int zone = 0; zone < zones.length; zone++) { + origAL.add(zones[zone]); + destAL.add(sandagExtStatID[dest]); + flowSutAL.add((float) (sutTrk * makeShare[zone] / makeShareSum)); + flowMutAL.add((float) (mutTrk * makeShare[zone] / makeShareSum)); + } + } else if (checkIfCountyInSanDiego(dest)) { + // flows from elsewhere to San Diego + double[] useShare = weights.get(comGroup + "_use"); + double useShareSum = utilities.getSum(useShare); + for (int zone = 0; zone < zones.length; zone++) { + origAL.add(sandagExtStatID[orig]); + destAL.add(zones[zone]); + flowSutAL.add((float) (sutTrk * useShare[zone] / useShareSum)); + flowMutAL.add((float) (mutTrk * useShare[zone] / useShareSum)); + } + } else { + // through flows through San Diego + origAL.add(sandagExtStatID[orig]); + destAL.add(sandagExtStatID[dest]); + flowSutAL.add(sutTrk); + flowMutAL.add(mutTrk); + } + + TableDataSet disFlows = new TableDataSet(); + disFlows.appendColumn(utilities.convertIntArrayListToArray(origAL), "orig"); + disFlows.appendColumn(utilities.convertIntArrayListToArray(destAL), "dest"); + disFlows.appendColumn(utilities.convertFloatArrayListToArray(flowSutAL), "sut"); + disFlows.appendColumn(utilities.convertFloatArrayListToArray(flowMutAL), "mut"); + + synchronized (disaggregatedFlows) { + disaggregatedFlows.put(comGroup, disFlows); + } + } + } + + + private void writeOutDisaggregatedFlows () { + // write out disaggregated flows to csv file + + logger.info(" Writing zone-to-external station truck trip table"); + String fileName = utilities.getRb().getString("zone.to.ext.stat.trip.table") + "_" + utilities.getYear() + ".csv"; + PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); + pw.print("orig,dest"); + for (String tod: todNames) pw.print(",SUT " + tod); + for (String tod: todNames) pw.print(",MUT " + tod); + pw.println(); + + HashMap summarizedFlows = new HashMap<>(); + + for (int comGrp = 0; comGrp <= utilities.getHighestVal(utilities.getListOfCommodityGroups()); comGrp++) { + TableDataSet flows = disaggregatedFlows.get(comGrp); + + for (int row = 1; row <= flows.getRowCount(); row++) { + String key = (int) flows.getValueAt(row, "orig") + "," + (int) flows.getValueAt(row, "dest"); + float[] values; + if (summarizedFlows.containsKey(key)) { + values = summarizedFlows.get(key); + } else { + values = new float[]{0,0}; + } + values[0] += flows.getValueAt(row, "sut"); + values[1] += flows.getValueAt(row, "mut"); + summarizedFlows.put(key, values); + } + } + + for (String key: summarizedFlows.keySet()) { + float[] values = summarizedFlows.get(key); + pw.print(key); + + // check if flow crosses border with Mexico, in which case different time-of-day split is used + String[] odPair = key.split(","); + int border = 0; + if (sandagExtStat_border[Integer.parseInt(odPair[0])] || + sandagExtStat_border[Integer.parseInt(odPair[1])]) border = 1; + + for (int tod = 0; tod < todNames.length; tod++) pw.print("," + values[0] * todSutShare[border][tod]); + for (int tod = 0; tod < todNames.length; tod++) pw.print("," + values[1] * todMutShare[border][tod]); + pw.println(); + } + pw.close(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java b/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java new file mode 100644 index 0000000..fcd4ed9 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java @@ -0,0 +1,45 @@ +package org.sandag.htm.applications; + +import org.sandag.htm.processFAF.disaggregateFlows; +import org.sandag.htm.processFAF.readFAF3; +import org.sandag.htm.processFAF.ReadFAF4; + +import org.apache.log4j.Logger; + +/** + * Program to model SANDAG external truck flows based on FAF3 data + * Author: Rolf Moeckel, PB Albuquerque + * Date: 6 March 2013 (Santa Fe, NM) + * Version 1.0 + * + * Modified 2017-12-28 to use FAF4 data by JEF, RSG + */ + +public class sandag_tm { + + private static Logger logger = Logger.getLogger(sandag_tm.class); + + + public static void main(String[] args) { + + long startTime = System.currentTimeMillis(); + + ReadFAF4 faf4 = new ReadFAF4(); + disaggregateFlows df = new disaggregateFlows(); + utilities.truckModelInitialization(args, faf4, df); + + if (utilities.getModel().equalsIgnoreCase("counties")) { + SandagCountyModel scm = new SandagCountyModel(faf4, df); + scm.runSandagCountyModel(); + } else { + sandagZonalModel szm = new sandagZonalModel(); + szm.runSandagZonalModel(); + } + + logger.info("Finished SANDAG Truck Model for year " + utilities.getYear()); + float endTime = utilities.rounder(((System.currentTimeMillis() - startTime) / 60000), 1); + int hours = (int) (endTime / 60); + int min = (int) (endTime - 60 * hours); + logger.info("Runtime: " + hours + " hours and " + min + " minutes."); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java b/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java new file mode 100644 index 0000000..330aca4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java @@ -0,0 +1,217 @@ +package org.sandag.htm.applications; + +import com.pb.common.datafile.DBFFileReader; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import org.sandag.htm.processFAF.disaggregateFlows; +import org.sandag.htm.processFAF.fafUtils; +import org.sandag.htm.processFAF.readFAF3; +import org.sandag.htm.processFAF.ReadFAF4; + +import org.apache.log4j.Logger; + +import java.io.File; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ResourceBundle; + +/** + * Utility methods for SANDAG external truck model + * Author: Rolf Moeckel, PB Albuquerque + * Date: 6 March 2013 (Santa Fe, NM) + * Version 1.0 + */ + +public class utilities { + + + private static ResourceBundle rb; + private static String model; + private static int year; + public static int[] countyFips; + public static int[] countyFipsIndex; + private static int[] commodityGroupOfSCTG; + private static HashMap comGroupDefinition; + private static int[] listOfCommodityGroups; + private static Logger logger = Logger.getLogger(utilities.class); + + + public static void truckModelInitialization(String[] args, readFAF3 faf3, disaggregateFlows df) { + // read in properties file and define basic variables, such as model year + + String rbName = args[0]; + File propFile = new File(rbName); + rb = ResourceUtil.getPropertyBundle(propFile); + model = args[1]; + if (!model.equalsIgnoreCase("counties") && !model.equalsIgnoreCase("zones")) { + logger.error("Call program with parameters "); + logger.error("Geography " + args[1] + " not understood. Choose \"counties\" or \"zones\""); + System.exit(1); + } + year = Integer.parseInt(args[2]); + logger.info("Starting SANDAG Truck Model for year " + utilities.getYear()); + readCommodityGrouping(); + } + + + public static void truckModelInitialization(String[] args, ReadFAF4 faf4, disaggregateFlows df) { + // read in properties file and define basic variables, such as model year + + String rbName = args[0]; + File propFile = new File(rbName); + rb = ResourceUtil.getPropertyBundle(propFile); + model = args[1]; + if (!model.equalsIgnoreCase("counties") && !model.equalsIgnoreCase("zones")) { + logger.error("Call program with parameters "); + logger.error("Geography " + args[1] + " not understood. Choose \"counties\" or \"zones\""); + System.exit(1); + } + year = Integer.parseInt(args[2]); + logger.info("Starting SANDAG Truck Model for year " + utilities.getYear()); + readCommodityGrouping(); + } + + private static void readCommodityGrouping () { + // read in commodity grouping + + TableDataSet commodityGrouping = fafUtils.importTable(utilities.getRb().getString("commodity.grouping")); + int highestValue = utilities.getHighestVal(commodityGrouping.getColumnAsInt("SCTG")); + commodityGroupOfSCTG = new int[highestValue + 1]; + comGroupDefinition = new HashMap<>(); + for (int row = 1; row <= commodityGrouping.getRowCount(); row++) { + int sctg = (int) commodityGrouping.getValueAt(row, "SCTG"); + int grp = (int) commodityGrouping.getValueAt(row, "CommodityGroup"); + commodityGroupOfSCTG[sctg] = grp; + if (comGroupDefinition.containsKey(grp)) { + int[] commodities = comGroupDefinition.get(grp); + comGroupDefinition.put(grp, expandArrayByOneElement(commodities, sctg)); + } else { + comGroupDefinition.put(grp, new int[]{sctg}); + } + } + listOfCommodityGroups = new int[comGroupDefinition.size()]; + int count = 0; + for (int comGrp: comGroupDefinition.keySet()) { + listOfCommodityGroups[count] = comGrp; + count++; + } + } + + + public static int[] getCommodityGroupOfSCTG() { + return commodityGroupOfSCTG; + } + + + public static HashMap getComGroupDefinition() { + return comGroupDefinition; + } + + + public static int[] getListOfCommodityGroups() { + return listOfCommodityGroups; + } + + + public static float rounder(float value, int digits) { + // rounds value to digits behind the decimal point + return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); + } + + + public static int getYear() { + return year; + } + + public static ResourceBundle getRb() { + return rb; + } + + public static String getModel() { + return model; + } + + public static boolean getBooleanProperty (String token, boolean defaultIfNotAvailable) { + return ResourceUtil.getBooleanProperty(rb, token, defaultIfNotAvailable); + } + + + public static int getHighestVal(int[] array) { + // return highest number in array + + int high = Integer.MIN_VALUE; + for (int num: array) high = Math.max(high, num); + return high; + } + + + public static float rounder(double value, int digits) { + // rounds value to digits behind the decimal point + return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); + } + + + public static int[] expandArrayByOneElement (int[] existing, int addElement) { + // create new array that has length of existing.length + 1 and copy values into new array + int[] expanded = new int[existing.length + 1]; + System.arraycopy(existing, 0, expanded, 0, existing.length); + expanded[expanded.length - 1] = addElement; + return expanded; + } + + + public static void createZoneList() { + // Create array with specialRegions that serve as port of entry/exit + + int[] poeLand = fafUtils.importTable(rb.getString("ports.of.entry")).getColumnAsInt("pointOfEntry"); + int[] poeSea = fafUtils.importTable(rb.getString("marine.ports.of.entry")).getColumnAsInt("pointOfEntry"); + int[] poeAir = fafUtils.importTable(rb.getString("air.ports.of.entry")).getColumnAsInt("pointOfEntry"); + int[] list = new int[poeLand.length + poeSea.length + poeAir.length]; + System.arraycopy(poeLand, 0, list, 0, poeLand.length); + System.arraycopy(poeSea, 0, list, poeLand.length, poeSea.length); + System.arraycopy(poeAir, 0, list, poeLand.length + poeSea.length, poeAir.length); + countyFips = fafUtils.createCountyFipsArray(list); + countyFipsIndex = new int[fafUtils.getHighestVal(countyFips) + 1]; + for (int i = 0; i < countyFips.length; i++) { + countyFipsIndex[countyFips[i]] = i; + } + } + + + public static TableDataSet importTableFromDBF(String filePath) { + // read a dbf file into a TableDataSet + + TableDataSet tblData; + DBFFileReader dbfReader = new DBFFileReader(); + try { + tblData = dbfReader.readFile(new File( filePath )); + } catch (Exception e) { + throw new RuntimeException("File not found: <" + filePath + ">.", e); + } + dbfReader.close(); + return tblData; + } + + + public static double getSum (double[] array) { + // return sum of all elements in array + double sum = 0; + for (double val: array) sum += val; + return sum; + } + + + public static float[] convertIntArrayListToArray(ArrayList al) { + float[] array = new float[al.size()]; + for (int i = 0; i < al.size(); i++) array[i] = al.get(i); + return array; + } + + + public static float[] convertFloatArrayListToArray(ArrayList al) { + float[] array = new float[al.size()]; + for (int i = 0; i < al.size(); i++) array[i] = al.get(i); + return array; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java b/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java new file mode 100644 index 0000000..72a67eb --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java @@ -0,0 +1,180 @@ +package org.sandag.htm.applications; + +import com.pb.common.datafile.TableDataSet; +import org.sandag.htm.processFAF.fafUtils; +import org.sandag.htm.processFAF.readFAF3; + +import java.io.PrintWriter; +import java.util.ResourceBundle; + +/** + * Application of countyTruckModel for Yuma, AZ MPO + * Author: Rolf Moeckel, PB Albuquerque + * Date: April 5, 2012 (Santa Fe NM) + + */ +public class yumaMPO { + + private static float[][][] tons_yuma_border; + private static float[][][] tons_yuma_fafzones; + private static float[][][] tons_yuma_counties; + private static float[][] tons_yuma_mex; + private static TableDataSet counties; + private static int[] countyIndex; + + + public static void initializeVariables(ResourceBundle appRb) { + // set up variables for summaries + int maxCom = fafUtils.getHighestVal(readFAF3.sctgCommodities); + tons_yuma_border = new float[2][560 + 1][maxCom + 1]; // by direction, FAF zone and commodity + tons_yuma_fafzones = new float[2][560 + 1][maxCom + 1]; // by direction, FAF zone and commodity + tons_yuma_counties = new float[2][15 + 1][maxCom + 1]; // by direction, county and commodity + counties = fafUtils.importTable(appRb.getString("county.ID")); + counties.buildIndex(counties.getColumnPosition("COUNTYFIPS")); + countyIndex = new int[fafUtils.getHighestVal(counties.getColumnAsInt("COUNTYFIPS")) + 1]; + for (int i = 0; i < countyIndex.length; i++) countyIndex[i] = -1; + countyIndex[4001] = 0; // index counties within Arizona + countyIndex[4003] = 1; + countyIndex[4005] = 2; + countyIndex[4007] = 3; + countyIndex[4009] = 4; + countyIndex[4011] = 5; + countyIndex[4012] = 6; + countyIndex[4013] = 7; + countyIndex[4015] = 8; + countyIndex[4017] = 9; + countyIndex[4019] = 10; + countyIndex[4021] = 11; + countyIndex[4023] = 12; + countyIndex[4025] = 13; + countyIndex[4027] = 14; + countyIndex[6025] = 15; // index Imperial County in California + tons_yuma_mex = new float[2][2]; + } + + + public static void saveForYuma(int com, int oFips, int dFips, float tons) { + // save relevant data for Yuma-specific summaries + +// try { + int oFAF = 0; + if (oFips < 60000) oFAF = (int) counties.getIndexedValueAt(oFips, "FAF3region"); + int dFAF = 0; + if (dFips < 60000) dFAF = (int) counties.getIndexedValueAt(dFips, "FAF3region"); + + // Flows through Yuma border crossing + if (oFips == 61934 && dFAF != 0) { + tons_yuma_border[0][dFAF][com] += tons; + } else if (dFips == 61934 && oFAF != 0) { + tons_yuma_border[1][oFAF][com] += tons; + } + // Flows to/from Yuma + if (oFips == 4027 && dFAF != 0) { + tons_yuma_fafzones[0][dFAF][com] += tons; + if (countyIndex[dFips] >= 0) tons_yuma_counties[0][countyIndex[dFips]][com] += tons; + } else if (dFips == 4027 && oFAF != 0) { + tons_yuma_fafzones[1][oFAF][com] += tons; + if (countyIndex[oFips] >= 0) tons_yuma_counties[1][countyIndex[oFips]][com] += tons; + } + // Flows between Yuma and Mexico + if (oFips == 4027) tons_yuma_mex[0][0] += tons; + if (oFips == 4027 && dFips > 60000) tons_yuma_mex[0][1] += tons; + if (dFips == 4027) tons_yuma_mex[1][0] += tons; + if (dFips == 4027 && oFips > 60000) tons_yuma_mex[1][1] += tons; + +// } catch (Exception e) { +// System.out.println("Error: " + com+" "+oFips+" "+dFips); +// } + } + + + public static void writeOutResults(ResourceBundle appRb, int year) { + // write results to summary file + + PrintWriter pw = fafUtils.openFileForSequentialWriting(appRb.getString("yuma.summary") + year + ".csv"); + + pw.println("Total tons leaving Yuma : " + tons_yuma_mex[0][0]); + pw.println("Tons from Yuma to Mexico: " + tons_yuma_mex[0][1]); + pw.println("Total tons entering Yuma: " + tons_yuma_mex[1][0]); + pw.println("Tons from Mexico to Yuma: " + tons_yuma_mex[1][1]); + pw.println(); + + pw.println("SUMMARY BY FAF ZONE"); + pw.print("FlowsFromYumaToFAFZone"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int faf = 1; faf <= 560; faf++) { + float sum = 0; + for (int i: readFAF3.sctgCommodities) sum += tons_yuma_fafzones[0][faf][i]; + if (sum > 0) { + pw.print(faf); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_fafzones[0][faf][i]); + pw.println(); + } + } + pw.println(); + pw.print("FlowsToYumaFromFAFZone"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int faf = 1; faf <= 560; faf++) { + float sum = 0; + for (int i: readFAF3.sctgCommodities) sum += tons_yuma_fafzones[1][faf][i]; + if (sum > 0) { + pw.print(faf); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_fafzones[1][faf][i]); + pw.println(); + } + } + pw.println(); + + pw.println("SUMMARY BY COUNTY"); + pw.print("FlowsFromYumaToCounty"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int county: counties.getColumnAsInt("COUNTYFIPS")) { + if (countyIndex[county] == -1) continue; + pw.print(county); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_counties[0][countyIndex[county]][i]); + pw.println(); + } + pw.println(); + pw.print("FlowsToYumaFromCounty"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int county: counties.getColumnAsInt("COUNTYFIPS")) { + if (countyIndex[county] == -1) continue; + pw.print(county); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_counties[1][countyIndex[county]][i]); + pw.println(); + } + pw.println(); + + pw.println("SUMMARY BY BORDER ZONE"); + pw.print("FlowsFromYumaBorderToFAFZone"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int faf = 1; faf <= 560; faf++) { + float sum = 0; + for (int i: readFAF3.sctgCommodities) sum += tons_yuma_border[0][faf][i]; + if (sum > 0) { + pw.print(faf); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_border[0][faf][i]); + pw.println(); + } + } + pw.println(); + pw.print("FlowsToYumaBorderFromFAFZone"); + for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); + pw.println(); + for (int faf = 1; faf <= 560; faf++) { + float sum = 0; + for (int i: readFAF3.sctgCommodities) sum += tons_yuma_border[1][faf][i]; + if (sum > 0) { + pw.print(faf); + for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_border[1][faf][i]); + pw.println(); + } + } + pw.close(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java new file mode 100644 index 0000000..ef5e308 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java @@ -0,0 +1,19 @@ +package org.sandag.htm.processFAF; + +/** + * Defines modes available in FAF + * Author: Rolf Moeckel (PB Albuquerque) + * Date: September 9, 2010 + * Edited jef 2017-12-28 to remove reference to FAF3 since FAF3 modes are same as FAF4 + */ + +public enum ModesFAF { + + Truck, + Rail, + Water, + Air, + MultipleModesAndMail, + Pipeline, + OtherAndUnknown +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java new file mode 100644 index 0000000..8984f5c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java @@ -0,0 +1,522 @@ +package org.sandag.htm.processFAF; + +import org.apache.log4j.Logger; + +import java.util.ResourceBundle; +import java.util.HashMap; +import java.io.PrintWriter; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +/** + * This class reads FAF4 data and stores data in a TableDataSet + * Author: Joel Freedman, RSG - based on readFAF3 by Rolf Moeckel, PB + * Date: Dec 27, 2017 + */ + +public class ReadFAF4 { + Logger logger = Logger.getLogger(ReadFAF4.class); + private int factor; + private String[] valueColumnName; + private TableDataSet faf4commodityFlows; + public static TableDataSet fafRegionList; + private String[] regionState; + private static int[] domRegionIndex; + static public int[] sctgCommodities; + static public String[] sctgStringCommodities; + static private int[] sctgStringIndex; + private static HashMap portsOfEntry; + private static HashMap marinePortsOfEntry; + private static HashMap railPortsOfEntry; + private static HashMap airPortsOfEntry; + private static int[] listOfBorderPortOfEntries; + + + public void readAllData (ResourceBundle appRb, int year, String unit) { + // read input data + + if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) + readAllFAF4DataSets(appRb, unit, year); + readCommodityList(appRb); + readFAF4ReferenceLists(appRb); + } + + + public void readCommodityList(ResourceBundle appRb) { + // read commodity names + TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf4.sctg.commodity.list")); + sctgCommodities = new int[sctgComList.getRowCount()]; + sctgStringCommodities = new String[sctgCommodities.length]; + for (int i = 1; i <= sctgComList.getRowCount(); i++) { + sctgCommodities[i-1] = (int) sctgComList.getValueAt(i, "SCTG"); + if (sctgCommodities[i-1] < 10) sctgStringCommodities[i-1] = "SCTG0" + sctgCommodities[i-1]; + else sctgStringCommodities[i-1] = "SCTG" + sctgCommodities[i-1]; + } + sctgStringIndex = new int[fafUtils.getHighestVal(sctgCommodities) + 1]; + for (int num = 0; num < sctgCommodities.length; num++) sctgStringIndex[sctgCommodities[num]] = num; + } + + + public int getIndexOfCommodity (int commodity) { + return sctgStringIndex[commodity]; + } + + + public static String getSCTGname(int sctgInt) { + // get String name from sctg number + return sctgStringCommodities[sctgStringIndex[sctgInt]]; + } + + + public static String getFAFzoneName(int fafInt) { + // get String name from int FAF zone code number + return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "FAF4 Zones -Short Description"); + } + + + public static String getFAFzoneState(int fafInt) { + // get String two-letter abbreviation of state of fafInt + return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "State"); + } + + + public void definePortsOfEntry(ResourceBundle appRb) { + // read data to translate ports of entry in network links + + // Border crossings + portsOfEntry = new HashMap<>(); + TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); + for (int row = 1; row <= poe.getRowCount(); row++) { + int fafID = (int) poe.getValueAt(row, "faf4id"); + int node = (int) poe.getValueAt(row, "pointOfEntry"); + float weight = poe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (portsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = portsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + portsOfEntry.put(fafID, newPortsOfEntry); + } + listOfBorderPortOfEntries = poe.getColumnAsInt("pointOfEntry"); + + // Marine ports + marinePortsOfEntry = new HashMap<>(); + if (appRb.containsKey("marine.ports.of.entry")) { + TableDataSet mpoe = fafUtils.importTable(appRb.getString("marine.ports.of.entry")); + for (int row = 1; row <= mpoe.getRowCount(); row++) { + int fafID = (int) mpoe.getValueAt(row, "faf4id"); + int node = (int) mpoe.getValueAt(row, "pointOfEntry"); + float weight = mpoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (marinePortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = marinePortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + marinePortsOfEntry.put(fafID, newPortsOfEntry); + } + } + // Rail ports (railyards) + railPortsOfEntry = new HashMap<>(); + if (appRb.containsKey("rail.ports.of.entry")) { + TableDataSet rpoe = fafUtils.importTable(appRb.getString("rail.ports.of.entry")); + for (int row = 1; row <= rpoe.getRowCount(); row++) { + int fafID = (int) rpoe.getValueAt(row, "faf4id"); + int node = (int) rpoe.getValueAt(row, "pointOfEntry"); + float weight = rpoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (railPortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = railPortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + railPortsOfEntry.put(fafID, newPortsOfEntry); + } + } + // Airports + airPortsOfEntry = new HashMap<>(); + if (appRb.containsKey("air.ports.of.entry")) { + TableDataSet apoe = fafUtils.importTable(appRb.getString("air.ports.of.entry")); + for (int row = 1; row <= apoe.getRowCount(); row++) { + int fafID = (int) apoe.getValueAt(row, "faf4id"); + int node = (int) apoe.getValueAt(row, "pointOfEntry"); + float weight = apoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (airPortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = airPortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + airPortsOfEntry.put(fafID, newPortsOfEntry); + } + } + } + + + public static int[] getListOfBorderPortOfEntries() { + return listOfBorderPortOfEntries; + } + + + + public static TableDataSet getPortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (portsOfEntry.containsKey(fafZone)) return portsOfEntry.get(fafZone); + else return null; + } + + + public static TableDataSet getMarinePortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (marinePortsOfEntry.containsKey(fafZone)) return marinePortsOfEntry.get(fafZone); + else return null; + } + + + public static TableDataSet getAirPortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (airPortsOfEntry.containsKey(fafZone)) return airPortsOfEntry.get(fafZone); + else return null; + } + + + public void readAllFAF4dataSets2015(ResourceBundle appRb, String unit) { + // read all FAF4 data into TableDataSets in unit (= tons or dollars) + + logger.info ("Reading domestic FAF4 data in " + unit); + if (unit.equals("tons")) { + factor = 1000; // tons are in 1,000s + valueColumnName = new String[]{"tons_2015"}; + } else if (unit.equals("dollars")) { + factor = 1000000; // dollars are in 1,000,000s + valueColumnName = new String[]{"value_2015"}; + } else { + logger.fatal ("Wrong token " + unit + " in method readAllFAF4dataSets2015. Use tons or dollars."); + } + faf4commodityFlows = readFAF4CommodityFlows(appRb, unit); + } + + + public double[] summarizeFlowByCommodity (ModesFAF fafMode) { + // sum commodity flows by commodity and return array with total tons + + double[] totalFlows = new double[sctgCommodities.length]; + int modeNum = fafUtils.getEnumOrderNumber(fafMode); + for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { + if (faf4commodityFlows.getValueAt(row, "dms_mode") != modeNum) continue; + int com = sctgStringIndex[(int) faf4commodityFlows.getValueAt(row, "sctg2")]; + if (valueColumnName.length == 1) { + // use year provided by user + totalFlows[com] += faf4commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); + totalFlows[com] += val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + } + return totalFlows; + } + + + public void readAllFAF4DataSets(ResourceBundle appRb, String unit, int year) { + // read all FAF4 data into TableDataSets in unit (= tons or dollars) + + logger.info (" Reading FAF4 data in " + unit); + switch (unit) { + case "tons": + factor = 1000; // tons are provided in 1,000s + break; + case "dollars": + factor = 1000000; // dollars are provided in 1,000,000s + break; + default: + logger.fatal("Wrong token " + unit + " in method readAllFAF4DataSets. Use tons or dollars."); + throw new RuntimeException(); + } + int[] availYears = {2012, 2013, 2014, 2015, 2020, 2025, 2030, 2035, 2040, 2045}; + boolean yearInFaf = false; + for (int y: availYears) if (year == y) yearInFaf = true; + if (!yearInFaf) { // interpolate between two years + logger.info(" Year " + year + " does not exist in FAF4 data."); + int year1 = availYears[0]; + int year2 = availYears[availYears.length-1]; + for (int availYear : availYears) if (availYear < year) year1 = availYear; + for (int i = availYears.length - 1; i >= 0; i--) if (availYears[i] > year) year2 = availYears[i]; + logger.info(" FAF4 data are interpolated between " + year1 + " and " + year2 + "."); + // first position: lower year, second position: higher year, third position: steps away from lower year + valueColumnName = new String[]{unit + "_" + year1, unit + "_" + year2, String.valueOf((1f * (year - year1)) / (1f * (year2 - year1)))}; + } else { // use year provided by user + valueColumnName = new String[]{unit + "_" + year}; + } + faf4commodityFlows = readFAF4CommodityFlows(appRb, unit); + } + + + private TableDataSet readFAF4CommodityFlows(ResourceBundle appRb, String unit) { + // read FAF4 data and return TableDataSet with flow data + String fileName = ResourceUtil.getProperty(appRb, ("faf4.data")); + return fafUtils.importTable(fileName); + } + + + public HashMap createScaler(String[] tokens, double[] values) { + // create HashMap with state O-D pairs that need to be scaled + + HashMap scaler = new HashMap(); + if (tokens.length != values.length) { + throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); + } + for (int i=0; i scaler) { + // extract truck flows for year yr and scale flows according to scaler HashMap (no special regions specified) + + PrintWriter outFile = fafUtils.openFileForSequentialWriting(outFileName); + outFile.println("originFAF,destinationFAF,flowDirection," + commodityClassType.SCTG + "_commodity,shortTons"); + int modeNum = fafUtils.getEnumOrderNumber(mode); + for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { + int type = (int) faf4commodityFlows.getValueAt(row, "trade_type"); + double val; + if (valueColumnName.length == 1) { + // use year provided by user + val = faf4commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); + val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + val *= factor * odScaler(row, type, scaler); + if (val == 0) continue; + if (type == 1) writeDomesticFlow(modeNum, val, row, outFile); + else if (type == 2) writeImportFlow(modeNum, val, row, outFile, repF); + else if (type == 3) writeExportFlow(modeNum, val, row, outFile, repF); + else if (type == 4) writeThroughFlow(modeNum, val, row, outFile, repF); + else{ + logger.info("Invalid trade_type in FAF4 dataset in row " + row + ": " + type); + } + } + outFile.close(); + } + + + public void writeFlowsByModeAndCommodity (String outFileName, ModesFAF mode, reportFormat repF, + HashMap scaler) { + // extract truck flows for year yr and scale flows according to scaler HashMap, including special regions + + PrintWriter outFile[] = new PrintWriter[sctgCommodities.length]; + for (int com: sctgCommodities) { + String fileName; + if (com < 10) fileName = outFileName + "_SCTG0" + com + ".csv"; + else fileName = outFileName + "_SCTG" + com + ".csv"; + outFile[sctgStringIndex[com]] = fafUtils.openFileForSequentialWriting(fileName); + outFile[sctgStringIndex[com]].println("originFAF,destinationFAF,flowDirection,SCTG_commodity,shortTons"); + } + int modeNum = fafUtils.getEnumOrderNumber(mode); + for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { + int type = (int) faf4commodityFlows.getValueAt(row, "trade_type"); + double val; + if (valueColumnName.length == 1) { + // use year provided by user + val = faf4commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); + val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + val *= factor * odScaler(row, type, scaler); + if (val == 0) continue; + int comIndex = getIndexOfCommodity((int) faf4commodityFlows.getValueAt(row, "sctg2")); + if (type == 1) writeDomesticFlow(modeNum, val, row, outFile[comIndex]); + else if (type == 2) writeImportFlow(modeNum, val, row, outFile[comIndex], repF); + else if (type == 3) writeExportFlow(modeNum, val, row, outFile[comIndex], repF); + else if (type == 4) writeThroughFlow(modeNum, val, row, outFile[comIndex], repF); + else logger.info("Invalid trade_type in FAF4 dataset in row " + row + ": " + type); + } + for (int com: sctgCommodities) outFile[sctgStringIndex[com]].close(); + } + + + public float odScaler (int row, int type, HashMap scaler) { + // find scaler for origin destination pair in row + + int orig; + int dest; + if (type == 1) { + orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + } else if (type == 2) { + orig = (int) faf4commodityFlows.getValueAt(row, "fr_orig"); + dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + } else if (type == 3) { + orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + dest = (int) faf4commodityFlows.getValueAt(row, "fr_dest"); + } else { + orig = (int) faf4commodityFlows.getValueAt(row, "fr_orig"); + dest = (int) faf4commodityFlows.getValueAt(row, "fr_dest"); + } + String stateLevelToken = regionState[orig] + "_" + regionState[dest]; + String combo1Token = orig + "_" + regionState[dest]; + String combo2Token = regionState[orig] + "_" + dest; + String fafLevelToken = orig + "_" + dest; + float adj = 1; + if (scaler.containsKey(stateLevelToken)) adj = scaler.get(stateLevelToken); + if (scaler.containsKey(combo1Token)) adj = scaler.get(combo1Token); + if (scaler.containsKey(combo2Token)) adj = scaler.get(combo2Token); + if (scaler.containsKey(fafLevelToken)) adj = scaler.get(fafLevelToken); + return adj; + } + + + private int tryGettingThisValue(int row, String token) { + // for some flows, international zones/modes are empty -> catch this case and set zone to 0 + + int region; + try { + region = (int) faf4commodityFlows.getValueAt(row, token); + } catch (Exception e) { + region = 0; + } + return region; + } + + + public void writeDomesticFlow (int modeNum, double val, int row, PrintWriter outFile) { + // internal US flow + if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + int dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); + outFile.println(orig + "," + dest + ",domestic," + comm + "," + val); + } + } + + + public void writeImportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // from abroad to US + + int frInMode = (int) faf4commodityFlows.getValueAt(row, "fr_inmode"); + int borderZone = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); + if (frInMode == modeNum && repF != reportFormat.internat_domesticPart) { + int orig = tryGettingThisValue(row, "fr_orig"); + outFile.println(orig + "," + borderZone + ",import," + comm + "," + val); + } + if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + String txt; + if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",import_port,"; + else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",import_rail,"; + else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",import_airport,"; + else txt = ",import,"; + outFile.println(borderZone + "," + dest + txt + comm + "," + val); + } + } + + + public void writeExportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // from US to abroad + int frOutMode = tryGettingThisValue(row, "fr_outmode"); + int borderZone = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); + if (frOutMode == modeNum && repF != reportFormat.internat_domesticPart) { + int dest = tryGettingThisValue(row, "fr_dest"); + outFile.println(borderZone + "," + dest + ",export," + comm + "," + val); + } + if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + String txt = ",export,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",export_port,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",export_rail,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",export_airport,"; + outFile.println(orig + "," + borderZone + txt + comm + "," + val); + } + } + + + public void writeThroughFlow(int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // flows in transit through US + if ((int) faf4commodityFlows.getValueAt(row, "dms_mode") != modeNum) return; + int borderInZone = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); + int borderOutZone = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); + logger.warn("Through flows not yet implemented. This flow from " + borderInZone + " to " + borderOutZone + " is lost."); + } + + + public void readFAF4ReferenceLists(ResourceBundle rb) { + // read list of regions for FAF4 + String regFileName = rb.getString("faf4.region.list"); + fafRegionList = fafUtils.importTable(regFileName); + int[] reg = fafRegionList.getColumnAsInt(fafRegionList.getColumnPosition("ZoneID")); + domRegionIndex = new int[fafUtils.getHighestVal(reg) + 1]; + for (int num = 0; num < reg.length; num++) domRegionIndex[reg[num]] = num + 1; + regionState = new String[fafUtils.getHighestVal(reg) + 1]; + for (int row = 1; row <= fafRegionList.getRowCount(); row++) { + int zone = (int) fafRegionList.getValueAt(row, "ZoneID"); + regionState[zone] = fafRegionList.getStringValueAt(row, "State"); + } + } + + + public int[] getFAFzoneIDs () { + return fafRegionList.getColumnAsInt("ZoneID"); + } + + + public TableDataSet getFAF4Flows() { + return faf4commodityFlows; + } + + + public int getFactor() { + return factor; + } + + + public TableDataSet getFAF4CommodityFlows() { + return faf4commodityFlows; + } + + + public String[] getValueColumnName() { + return valueColumnName; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java new file mode 100644 index 0000000..17e8314 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java @@ -0,0 +1,13 @@ +package org.sandag.htm.processFAF; + +/** + * Defines commodity classification + * STCC + * SCTG + * User: Moeckel + * Date: May 7, 2009 + */ + +public enum commodityClassType { + STCC, SCTG +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java new file mode 100644 index 0000000..3d578cd --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java @@ -0,0 +1,113 @@ +package org.sandag.htm.processFAF; + +import com.pb.common.datafile.TableDataSet; +import org.apache.log4j.Logger; + +import java.util.ResourceBundle; + +/** + * Class to convert flows from tons into trucks by type + * Author: Rolf Moeckel, PB Albuquerque + * Data: 11 March 2013 (Santa Fe) + */ + +public class convertTonsToTrucks { + + private static Logger logger = Logger.getLogger(convertTonsToTrucks.class); + private String[] truckTypes = new String[] {"SingleUnit","TruckTrailer","CombinationSemitrailer","CombinationDoubleTriple"}; + private int[] fromMiles; + private float[][] truckShareByDistance; + private TableDataSet[] payloadByTruckType; + private ResourceBundle appRb; + + public convertTonsToTrucks(ResourceBundle appRb) { + this.appRb = appRb; + } + + + public void readData () { + // read data to convert tons into trucks + + logger.info(" Reading FAF3 payload factors"); + // truck types by distance class + TableDataSet truckTypeByDist = fafUtils.importTable(appRb.getString("truck.type.share.by.distance")); + fromMiles = truckTypeByDist.getColumnAsInt("fromMiles"); + truckShareByDistance = new float[fromMiles.length][truckTypes.length]; + // todo: allow adjustment of SUT/MUT share + for (int row = 1; row <= truckTypeByDist.getRowCount(); row++) { + for (int col = 0; col < truckTypes.length; col++) { + truckShareByDistance[row-1][col] = truckTypeByDist.getValueAt(row, truckTypes[col]); + } + } + + // truck body type by commodity + payloadByTruckType = new TableDataSet[4]; + payloadByTruckType[0] = fafUtils.importTable(appRb.getString("truck.body.share.single.unit")); + payloadByTruckType[0].buildIndex(payloadByTruckType[0].getColumnPosition("sctg")); + payloadByTruckType[1] = fafUtils.importTable(appRb.getString("truck.body.share.tractor.trl")); + payloadByTruckType[1].buildIndex(payloadByTruckType[1].getColumnPosition("sctg")); + payloadByTruckType[2] = fafUtils.importTable(appRb.getString("truck.body.share.comb.semi.t")); + payloadByTruckType[2].buildIndex(payloadByTruckType[2].getColumnPosition("sctg")); + payloadByTruckType[3] = fafUtils.importTable(appRb.getString("truck.body.share.comb.dbl.tr")); + payloadByTruckType[3].buildIndex(payloadByTruckType[3].getColumnPosition("sctg")); + } + + + public float[] convertThisFlowFromTonsToTrucks (String commodity, float distance, float flowInTons) { + // convert commodity flow in tons into number of trucks by truck type + + // find correct distance class + int distClass = 0; + for (int i = 0; i < fromMiles.length; i++) if (distance > fromMiles[i]) distClass = i; + + // calculate trucks by truck type + float[] trucksByType = new float[truckTypes.length]; + for (int i = 0; i < truckTypes.length; i++) trucksByType[i] = loadTruckShare(i, distClass, commodity, flowInTons); + + return trucksByType; + } + + + private float loadTruckShare (int truckType, int distClass, String commodity, float flowInTons) { + // calculate payload for with for + + int com = Integer.parseInt(commodity.substring(4)); + float tonsOnThisTruckType = flowInTons * truckShareByDistance[distClass][truckType]; + + float trucks = 0; + for (int col = 2; col <= payloadByTruckType[truckType].getColumnCount(); col++) { + trucks += tonsOnThisTruckType * payloadByTruckType[truckType].getIndexedValueAt(com, col); + } + return trucks; + } + + + public double[] convertThisFlowFromTonsToTrucks (String commodity, float distance, double flowInTons) { + // convert commodity flow in tons into number of trucks by truck type + + // find correct distance class + int distClass = 0; + for (int i = 0; i < fromMiles.length; i++) if (distance > fromMiles[i]) distClass = i; + + // calculate trucks by truck type + double[] trucksByType = new double[truckTypes.length]; + for (int i = 0; i < truckTypes.length; i++) trucksByType[i] = loadTruckShare(i, distClass, commodity, flowInTons); + + return trucksByType; + } + + + private double loadTruckShare (int truckType, int distClass, String commodity, double flowInTons) { + // calculate payload for with for + + int com = Integer.parseInt(commodity.substring(4)); + double tonsOnThisTruckType = flowInTons * truckShareByDistance[distClass][truckType]; + + double trucks = 0; + for (int col = 2; col <= payloadByTruckType[truckType].getColumnCount(); col++) { + trucks += tonsOnThisTruckType * payloadByTruckType[truckType].getIndexedValueAt(com, col); + } + return trucks; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java new file mode 100644 index 0000000..0be8116 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java @@ -0,0 +1,597 @@ +package org.sandag.htm.processFAF; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import org.sandag.htm.applications.ohio; +import org.sandag.htm.applications.yumaMPO; +import org.apache.log4j.Logger; +import com.pb.sawdust.util.concurrent.DnCRecursiveAction; + +import java.io.File; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.ResourceBundle; + +/** + * Reads FAF3 data, disaggregates them to county-to-county flows and converts them into truck trips + * Author: Rolf Moeckel, PB Albuquerque + * Date: August 22, 2011 (Santa Fe NM) + * Revised for Yuma County: March 29, 2012 (Albuquerque, NM) + */ + +public class countyTruckModel { + + private static Logger logger = Logger.getLogger(countyTruckModel.class); + private ResourceBundle appRb; + private int year; + private disaggregateFlows df; + private HashMap cntFlows; + private int[] countyFips; + private int[] countyFipsIndex; + private int[] commodityGrouping; + private int numCommodityGroups; + private boolean yumaSummary; + + + public countyTruckModel(ResourceBundle rb, int yr) { + this.appRb = rb; + this.year = yr; + } + + + public static void main(String[] args) { + // construct main model + + long startTime = System.currentTimeMillis(); + ResourceBundle appRb = fafUtils.getResourceBundle(args[0]); + int year = Integer.parseInt(args[1]); + countyTruckModel ctm = new countyTruckModel(appRb, year); + ctm.run(); + logger.info("County Truck Model completed."); + float endTime = fafUtils.rounder(((System.currentTimeMillis() - startTime) / 60000), 1); + logger.info("Runtime: " + endTime + " minutes."); + } + + + private void run() { + // main run method + + logger.info("Started FAF4 model to generate county-to-county truck flows for " + year); + ReadFAF4 faf4 = new ReadFAF4(); + df = new disaggregateFlows(); + df.getUScountyEmploymentByIndustry(appRb); + faf4.readAllData(appRb, year, "tons"); + faf4.definePortsOfEntry(appRb); + if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) extractTruckData(faf4); + createZoneList(); + String truckTypeDefinition = "sut_mut"; + df.defineTruckTypes(truckTypeDefinition, appRb); + if (ResourceUtil.getBooleanProperty(appRb, "save.results.for.ohio", false)) ohio.summarizeFAFData(appRb, countyFips); + yumaSummary = ResourceUtil.getBooleanProperty(appRb, "save.results.for.yuma", false); + if (yumaSummary) yumaMPO.initializeVariables(appRb); + disaggregateFromFafToCounties(); + if (yumaSummary) yumaMPO.writeOutResults(appRb, year); + if (ResourceUtil.getBooleanProperty(appRb, "report.by.employment")) { + convertFromCommoditiesToTrucksByEmpType(); + } else { + convertFromCommoditiesToTrucks(); + } + if (ResourceUtil.getBooleanProperty(appRb, "analyze.commodity.groups", false)) { + readCommodityGrouping(); + convertFromCommoditiesToTrucksByComGroup(); + } + } + + private void extractTruckData(readFAF3 faf3) { + // extract truck data and write flows to file + logger.info("Extracting FAF3 truck data"); + String[] scaleTokens = ResourceUtil.getArray(appRb, "scaling.truck.trips.tokens"); + double[] scaleValues = ResourceUtil.getDoubleArray(appRb, "scaling.truck.trips.values"); + HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); + String truckFileNameT = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; + + // create output directory if it does not exist yet + File file = new File ("output/temp"); + if (!file.exists()) { + boolean outputDirectorySuccessfullyCreated = file.mkdir(); + if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); + } + + faf3.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); + } + + private void extractTruckData(ReadFAF4 faf4) { + // extract truck data and write flows to file + logger.info("Extracting FAF4 truck data"); + String[] scaleTokens = ResourceUtil.getArray(appRb, "scaling.truck.trips.tokens"); + double[] scaleValues = ResourceUtil.getDoubleArray(appRb, "scaling.truck.trips.values"); + HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); + String truckFileNameT = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; + + // create output directory if it does not exist yet + File file = new File ("output/temp"); + if (!file.exists()) { + boolean outputDirectorySuccessfullyCreated = file.mkdir(); + if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); + } + + faf4.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); + } + + private void createZoneList() { + // Create array with specialRegions that serve as port of entry/exit + + TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); + countyFips = fafUtils.createCountyFipsArray(poe.getColumnAsInt("pointOfEntry")); + countyFipsIndex = new int[fafUtils.getHighestVal(countyFips) + 1]; + for (int i = 0; i < countyFips.length; i++) { + countyFipsIndex[countyFips[i]] = i; + } + } + + + private void disaggregateFromFafToCounties() { + // disaggregates freight flows from FAF zoneArray to counties + + logger.info("Disaggregating FAF3 data from FAF zones to counties for year " + year + "."); + + cntFlows = new HashMap<>(); + + String[] commodities; + commodities = readFAF3.sctgStringCommodities; + int matrixSize = countyFips.length; + cntFlows = new HashMap<>(); + + float globalScale = (float) ResourceUtil.getDoubleProperty(appRb, "overall.scaling.factor.truck"); + + // regular method + for (String com: commodities) { + float[][] dummy = new float[matrixSize][matrixSize]; + cntFlows.put(com, dummy); + } + boolean keepTrackOfEmplType = ResourceUtil.getBooleanProperty(appRb, "report.by.employment", false); + df.prepareCountyDataForFAFwithDetailedEmployment(appRb, year, keepTrackOfEmplType); + + java.util.concurrent.ForkJoinPool pool = new java.util.concurrent.ForkJoinPool(); + DnCRecursiveAction action = new DissaggregateFafAction(globalScale); + pool.execute(action); + action.getResult(); + + if (ResourceUtil.getBooleanProperty(appRb, "summarize.by.ohio.rail.zones", false)) + ohio.sumFlowByRailZone(appRb, year, countyFips, countyFipsIndex, cntFlows); + } + + + private class DissaggregateFafAction extends DnCRecursiveAction { + private final float globalScale; + + private DissaggregateFafAction(float globalScale) { + super(0,readFAF3.sctgStringCommodities.length); + this.globalScale = globalScale; + } + + private DissaggregateFafAction(float globalScale, long start, long length, DnCRecursiveAction next) { + super(start,length,next); + this.globalScale = globalScale; + } + + @Override + protected void computeAction(long start, long length) { + long end = start + length; + for (int comm = (int) start; comm < end; comm++) { + int cm = readFAF3.sctgCommodities[comm]; + String fileName = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; + if (cm < 10) fileName = fileName + "_SCTG0" + cm + ".csv"; + else fileName = fileName + "_SCTG" + cm + ".csv"; + logger.info(" Working on " + fileName); + String sctg = readFAF3.getSCTGname(cm); + float[][] values = cntFlows.get(sctg); + TableDataSet tblFlows = fafUtils.importTable(fileName); + for (int row = 1; row <= tblFlows.getRowCount(); row++) { + float shortTons = tblFlows.getValueAt(row, "shortTons"); + if (shortTons == 0) continue; + String dir = tblFlows.getStringValueAt(row, "flowDirection"); + int orig = (int) tblFlows.getValueAt(row, "originFAF"); + int dest = (int) tblFlows.getValueAt(row, "destinationFAF"); + TableDataSet singleFlow; + if (dir.equals("import") || dir.equals("export")) { + TableDataSet poe; + if (dir.equals("import")) poe = readFAF3.getPortsOfEntry(orig); + else poe = readFAF3.getPortsOfEntry(dest); + singleFlow = df.disaggregateSingleFAFFlowThroughPOE(dir, poe, orig, dest, sctg, shortTons, 1); + } else singleFlow = df.disaggregateSingleFAFFlow(orig, dest, sctg, shortTons, 1); + for (int i = 1; i <= singleFlow.getRowCount(); i++) { + int oFips = (int) singleFlow.getValueAt(i, "oFips"); + int oZone = getCountyId(oFips); + int dFips = (int) singleFlow.getValueAt(i, "dFips"); + int dZone = getCountyId(dFips); + float thisFlow = singleFlow.getValueAt(i, "Tons") * globalScale; + values[oZone][dZone] += thisFlow; + if (yumaSummary) yumaMPO.saveForYuma(cm, oFips, dFips, thisFlow); + } + } + } + } + + @Override + protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) { + return new DissaggregateFafAction(globalScale,start,length,next); + } + + @Override + protected boolean continueDividing(long length) { + return getSurplusQueuedTaskCount() < 3 && length > 1; + } + } + + + private int getCountyId(int fips) { + // Return region code of regName + return countyFipsIndex[fips]; + } + + + private void readCommodityGrouping () { + // Read how commodities are grouped by SCTG cagegory + + TableDataSet comGroups = fafUtils.importTable(appRb.getString("commodity.grouping")); + commodityGrouping = new int[fafUtils.getHighestVal(readFAF3.sctgCommodities) + 1]; + for (int row = 1; row <= comGroups.getRowCount(); row++) { + int sctg = (int) comGroups.getValueAt(row, "SCTG"); + int group = (int) comGroups.getValueAt(row, "Group"); + commodityGrouping[sctg] = group; + } + numCommodityGroups = 0; + for (int i = 0; i < commodityGrouping.length; i++) { + if (commodityGrouping[i] == 0) continue; + boolean alreadyExists = false; + for (int j = 0; j < i; j++) { + if (j == i) continue; + if (commodityGrouping[j] == commodityGrouping[i]) alreadyExists = true; + } + if (!alreadyExists) numCommodityGroups++; + } + } + + + private void convertFromCommoditiesToTrucks() { + // generate truck flows based on commodity flows + logger.info("Converting flows in tons into truck trips"); + float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); + float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); + double[][]sluTrucks = new double[countyFips.length][countyFips.length]; + double[][]mtuTrucks = new double[countyFips.length][countyFips.length]; + for (String com: readFAF3.sctgStringCommodities) { + double avPayload = fafUtils.findAveragePayload(com, "SCTG"); + double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; + double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; + float[][] tonFlows = cntFlows.get(com); + for (int i: countyFips) { + for (int j: countyFips) { + float dist = df.getCountyDistance(i, j); + if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. + int orig = getCountyId(i); + int dest = getCountyId(j); + if (tonFlows[orig][dest] == 0) continue; + double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); + // add empty trucks + trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); + trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); + // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor + trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); + trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); + sluTrucks[orig][dest] += trucksByType[0]; + mtuTrucks[orig][dest] += trucksByType[1]; + } + } + } + if (ResourceUtil.getBooleanProperty(appRb, "write.cnty.to.cnty.truck.trps", false)) + writeOutDisaggregatedTruckTrips(sluTrucks, mtuTrucks); + if (ResourceUtil.getBooleanProperty(appRb, "write.ii.ei.ee.trips", false)) + writeOutDisaggregatedTruckTripsByDirection(sluTrucks, mtuTrucks); + } + + + private void convertFromCommoditiesToTrucksByEmpType() { + // generate truck flows based on commodity flows, keeping track of employment type generating/attracting trucks + + logger.info("Converting flows in tons into truck trips, keeping track of employment types generating trucks"); + float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); + float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); + HashMap countyWeights = getCountyWeightsWithDetEmpl(); + + String[] emplCat = df.getEmpCats(); + + double[][][]sluTrucks = new double[emplCat.length][countyFips.length][countyFips.length]; + double[][][]mtuTrucks = new double[emplCat.length][countyFips.length][countyFips.length]; + + for (String com: readFAF3.sctgStringCommodities) { + double avPayload = fafUtils.findAveragePayload(com, "SCTG"); + double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; + double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; + float[][] tonFlows = cntFlows.get(com); + for (int i: countyFips) { + float[] weights = countyWeights.get(i); + for (int j: countyFips) { + float dist = df.getCountyDistance(i, j); + if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. + int orig = getCountyId(i); + int dest = getCountyId(j); + if (tonFlows[orig][dest] == 0) continue; + double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); + // add empty trucks + trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); + trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); + // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor + trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); + trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); + for (int eCat = 0; eCat < emplCat.length; eCat++) { + if (trucksByType[0] + trucksByType[1] == 0) continue; + try { + sluTrucks[eCat][orig][dest] += trucksByType[0] * weights[eCat]; + mtuTrucks[eCat][orig][dest] += trucksByType[1] * weights[eCat]; + } catch (Exception e){ + logger.warn(eCat+" "+orig+" "+dest+" "+i+" "+j+" "+com+": "+trucksByType[0]+trucksByType[1]); + } + } + } + } + } + writeOutDisaggregatedTruckTripsDetEmpl(sluTrucks, mtuTrucks); + } + + + private HashMap getCountyWeightsWithDetEmpl() { + // look up employment weight for each county + + HashMap weights = new HashMap<>(); + TableDataSet counties = fafUtils.importTable(ResourceUtil.getProperty(appRb, "county.ID")); + String[] emplCats = df.getEmpCats(); + for (int row = 1; row <= counties.getRowCount(); row++) { + int fips = (int) counties.getValueAt(row, "COUNTYFIPS"); + int faf = (int) counties.getValueAt(row, "FAF3region"); + if (faf == -99) continue; + for (String com: readFAF3.sctgStringCommodities) { + TableDataSet countyWeightsInOrigInFafZone = df.getCountyWeights("orig", faf, com); + for (int rw = 1; rw <= countyWeightsInOrigInFafZone.getRowCount(); rw++) { + if (countyWeightsInOrigInFafZone.getValueAt(rw, "COUNTYFIPS") == fips) { + float[] wght = new float[emplCats.length]; + for (int eCat = 0; eCat < emplCats.length; eCat++) { + wght[eCat] = countyWeightsInOrigInFafZone.getValueAt(rw, emplCats[eCat]); + } + float sum = fafUtils.getSum(wght); + // normalize weights from 0 - 1 + for (int eCat = 0; eCat < emplCats.length; eCat++) { + wght[eCat] = wght[eCat] / sum; + } + weights.put(fips, wght); + } + } + } + } + return weights; + } + + + private void writeOutDisaggregatedTruckTrips(double[][]sluTrucks, double[][]mtuTrucks) { + // write out disaggregated truck trips + + double[] truckProd = new double[fafUtils.getHighestVal(countyFips) + 1]; + + String fileName = appRb.getString("cnt.to.cnt.truck.flows") + year + ".csv"; + logger.info("Writing results to file " + fileName); + PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); + pw.println("OrigFips,DestFips,sut,mut"); + for (int i: countyFips) { + for (int j: countyFips) { + int orig = getCountyId(i); + int dest = getCountyId(j); + double slu = sluTrucks[orig][dest]; + double mtu = mtuTrucks[orig][dest]; + if (slu + mtu >= 0.00001) { + pw.format ("%d,%d,%.5f,%.5f", i, j, slu, mtu); + pw.println(); + truckProd[i] += slu + mtu; + } + } + } + pw.close(); + + if (ResourceUtil.getBooleanProperty(appRb, "report.truck.prod.by.county", false)) { + String fileNameProd = appRb.getString("truck.prod.by.county") + year + ".csv"; + PrintWriter pwp = fafUtils.openFileForSequentialWriting(fileNameProd); + pwp.println("CountyFips,TrucksGenerated"); + for (int i: countyFips) { + if (truckProd[i] > 0) pwp.println(i + "," + truckProd[i]); + } + pwp.close(); + } + } + + + private void writeOutDisaggregatedTruckTripsDetEmpl(double[][][]sluTrucks, double[][][]mtuTrucks) { + // write out disaggregated truck trips + + String fileName = appRb.getString("cnt.to.cnt.truck.flows") + "ByEmployment" + year + ".csv"; + logger.info("Writing results to file " + fileName); + PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); + pw.print("OrigFips,DestFips"); + String[] emplCats = df.getEmpCats(); + for (String emplCat : emplCats) { + if (emplCat.contains("Trade") || emplCat.contains("Financial") || + emplCat.contains("Education") || emplCat.contains("Other")) continue; + pw.print("," + emplCat + "_SUT," + emplCat + "_MUT"); + } + pw.println(); + for (int i: countyFips) { + for (int j: countyFips) { + int orig = getCountyId(i); + int dest = getCountyId(j); + pw.print(i + "," + j); + for (int eCat = 0; eCat < emplCats.length; eCat++) { + if (emplCats[eCat].contains("Trade") || emplCats[eCat].contains("Financial") || + emplCats[eCat].contains("Education") || emplCats[eCat].contains("Other")) continue; + double slu = sluTrucks[eCat][orig][dest]; + double mtu = mtuTrucks[eCat][orig][dest]; + pw.print("," + slu + "," + mtu); + } + pw.println(); + } + } + pw.close(); + } + + + private void writeOutDisaggregatedTruckTripsByDirection(double[][]sluTrucks, double[][]mtuTrucks) { + // write out disaggregated truck trips distinguishing II, EI/IE and EE trips + + TableDataSet counties = disaggregateFlows.countyIDsWithEmployment; + boolean[] relevantCounty = new boolean[fafUtils.getHighestVal(countyFips) + 1]; + String state = appRb.getString("ii.state"); + + for (int row = 1; row <= counties.getRowCount(); row++) { + relevantCounty[(int) counties.getValueAt(row, "COUNTYFIPS")] = counties.getStringValueAt(row, "StateCode").equals(state); + } + + String fileName = appRb.getString("cnt.to.cnt.truck.flows.by.dir") + year + ".csv"; + logger.info("Writing ii/ei/ie/ee flows to file " + fileName); + PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); + pw.println("OrigFips,DestFips,iiSut,iiMut,eiSut,eiMut,eeSut,eeMut,totSut,totMut"); + for (int i: countyFips) { + for (int j: countyFips) { + int orig = getCountyId(i); + int dest = getCountyId(j); + double slu = sluTrucks[orig][dest]; + double mtu = mtuTrucks[orig][dest]; + if (slu + mtu >= 0.00001) { + if (relevantCounty[i] && relevantCounty[j]) { + pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, slu, mtu, 0., 0., 0., 0., slu, mtu); + pw.println(); + } else if ((relevantCounty[i] && !relevantCounty[j]) || (!relevantCounty[i] && relevantCounty[j])) { + pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, 0., 0., slu, mtu, 0., 0., slu, mtu); + pw.println(); + } else { + pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, 0., 0., 0., 0., slu, mtu, slu, mtu); + pw.println(); + } + } + } + } + pw.close(); + } + + + private void convertFromCommoditiesToTrucksByComGroup () { + // read flows in tons and convert into flows in trucks, distinguishing commodity groups + + float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); + float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); + double[][][] sluTrucks = new double[numCommodityGroups+1][countyFips.length][countyFips.length]; + double[][][] mtuTrucks = new double[numCommodityGroups+1][countyFips.length][countyFips.length]; + for (String com: readFAF3.sctgStringCommodities) { + int iCom = Integer.parseInt(com.substring(4, 6)); // convert "SCTG00" into "00" + double avPayload = fafUtils.findAveragePayload(com, "SCTG"); + double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; + double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; + float[][] tonFlows = cntFlows.get(com); + for (int i: countyFips) { + for (int j: countyFips) { + int orig = getCountyId(i); + int dest = getCountyId(j); + if (tonFlows[orig][dest] == 0) continue; + float dist = df.getCountyDistance(i, j); + if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. + double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); + // add empty trucks + trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); + trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); + // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor + trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); + trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); + sluTrucks[commodityGrouping[iCom]][orig][dest] += trucksByType[0]; + mtuTrucks[commodityGrouping[iCom]][orig][dest] += trucksByType[1]; + } + } + } + writeOutDisaggregatedTruckTrips(sluTrucks, mtuTrucks); + } + + + private void writeOutDisaggregatedTruckTrips(double sluTrucks[][][], double[][][] mutTrucks) { + // Write out truck trips by commodity group + + TableDataSet counties = disaggregateFlows.countyIDsWithEmployment; + boolean[] relevantCounty = new boolean[fafUtils.getHighestVal(countyFips) + 1]; + String state = appRb.getString("ii.state"); + double[][] prodByCounty = new double[countyFips.length][numCommodityGroups + 1]; + double[][] attrByCounty = new double[countyFips.length][numCommodityGroups + 1]; + + for (int row = 1; row <= counties.getRowCount(); row++) { + relevantCounty[(int) counties.getValueAt(row, "COUNTYFIPS")] = + counties.getStringValueAt(row, "StateCode").equals(state); + } + + boolean seperateEEflows = ResourceUtil.getBooleanProperty(appRb, "report.ext.flows.separately"); + String cgFile = appRb.getString("trucks.by.commodity.group") + year + ".csv"; + PrintWriter pwc = fafUtils.openFileForSequentialWriting(cgFile); + pwc.print("orig,dest"); + for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print(",com" + cg); + if (seperateEEflows) { + pwc.println(",external"); + } else { + pwc.println(); + } + for (int i = 0; i < countyFips.length; i++) { + for (int j = 0; j < countyFips.length; j++) { + + double sm = 0.; + for (int cg = 1; cg <= numCommodityGroups; cg++) sm += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; + + if (sm > 0) { + pwc.print(countyFips[i] + "," + countyFips[j]); + if (seperateEEflows) { + if (!relevantCounty[countyFips[i]] && !relevantCounty[countyFips[j]]) { + // E-E flows + for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print(",0"); + pwc.println("," + sm); + } else { + // I-I, I-E and E-I flows + for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print("," + + (sluTrucks[cg][i][j] + mutTrucks[cg][i][j])); + pwc.println(",0"); + } + } else { + // I-I, I-E, E-I and E-E flows + for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print("," + + (sluTrucks[cg][i][j] + mutTrucks[cg][i][j])); + pwc.println(); + } + for (int cg = 1; cg <= numCommodityGroups; cg++) { + prodByCounty[i][cg] += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; + attrByCounty[j][cg] += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; + } + } + } + } + pwc.close(); + + String cgFileAgg = appRb.getString("trucks.by.commodity.group") + year + "_prodAttr.csv"; + PrintWriter pwca = fafUtils.openFileForSequentialWriting(cgFileAgg); + pwca.print("countyFips"); + for (int cg = 1; cg <= numCommodityGroups; cg++) pwca.print(",prod_" + cg + ",attr_" + cg); + pwca.println(); + for (int i = 0; i < countyFips.length; i++) { + float sm = 0; + for (int cg = 1; cg <= numCommodityGroups; cg++) { + sm += prodByCounty[i][cg] + attrByCounty[i][cg]; + } + if (sm > 0) { + pwca.print(countyFips[i]); + for (int cg = 1; cg <= numCommodityGroups; cg++) pwca.print("," + prodByCounty[i][cg] + "," + + attrByCounty[i][cg]); + pwca.println(); + } + } + pwca.close(); + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java new file mode 100644 index 0000000..004554c --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java @@ -0,0 +1,67 @@ +package org.sandag.htm.processFAF; + +import com.pb.common.datafile.TableDataSet; + +import java.util.HashMap; +import java.util.ResourceBundle; + +/** + * This class disaggregates and aggregates FAF flows from FAF zones to model zones (which may be smaller or larger than FAF zones) + * User: Rolf Moeckel, PB Albuquerque + * Date: November 17, 2011 (Santa Fe, NM) + */ + +// todo: Started this class for NCSTM, but then realized that FAF zones do not nest in RMZ of NCSTM. Some FAF zones +// todo: would have to be split in parts and aggregated in other parts. +// todo: May be useful in other projects. + +public class disaggregateAndAggregateFlows { + + private ResourceBundle appRb; + private HashMap zonesToDisaggregate; + private int[] zonesToAggregate; + + public disaggregateAndAggregateFlows(ResourceBundle appRb) { + this.appRb = appRb; + } + + + public void defineZonesToAggregate(String fileName) { + // define which FAF zones need to be aggregated to larger model zones + + TableDataSet fafZones = fafUtils.importTable(fileName); + int highestFAF = -1; + for (int row = 1; row <= fafZones.getRowCount(); row++) { + if (fafZones.getValueAt(row, "modelZone") != -1) highestFAF = (int) Math.max(highestFAF, fafZones.getValueAt(row, "modelZone")); + } + zonesToAggregate = new int[highestFAF + 1]; + for (int row = 1; row <= fafZones.getRowCount(); row++) { + if (fafZones.getValueAt(row, "modelZone") == -1) { + zonesToAggregate[(int) fafZones.getValueAt(row, "ZoneID")] = -1; + } else { + zonesToAggregate[(int) fafZones.getValueAt(row, "ZoneID")] = (int) fafZones.getValueAt(row, "modelZone"); + } + } + } + + + public void defineZonesToDisaggregate(TableDataSet zoneSystem) { + // define which FAF zones need to be disaggregated to smaller model zones + + zonesToDisaggregate = new HashMap<>(); + for (int row = 1; row <= zoneSystem.getRowCount(); row++) { + int taz = (int) zoneSystem.getValueAt(row, "TAZ"); + int faf = (int) zoneSystem.getValueAt(row, "FAFzone"); + if (!zoneSystem.getBooleanValueAt(row, "disaggregate")) continue; + if (zonesToDisaggregate.containsKey(faf)) { + int[] zones = zonesToDisaggregate.get(faf); + int[] zonesNew = new int[zones.length + 1]; + System.arraycopy(zones, 0, zonesNew, 0, zones.length); + zonesNew[zones.length] = taz; + zonesToDisaggregate.put(faf, zonesNew); + } else { + zonesToDisaggregate.put(faf, new int[]{taz}); + } + } + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java new file mode 100644 index 0000000..5045b90 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java @@ -0,0 +1,1357 @@ +package org.sandag.htm.processFAF; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; +import com.pb.common.matrix.Matrix; +import com.pb.common.matrix.MatrixReader; +import java.util.HashMap; +import java.util.ResourceBundle; +import java.io.File; +import org.apache.log4j.Logger; + + +/** + * This class disaggregates FAF flows from FAF zones to counties + * User: Rolf Moeckel + * Date: May 6, 2009 + * + * Updated 2017-12-27 JEF + */ + +public class disaggregateFlows { + + static Logger logger = Logger.getLogger(disaggregateFlows.class); + public static TableDataSet countyIDsWithEmployment; + private HashMap countyShares; + private TableDataSet truckTypeShares; + private int[] truckTypeShareDistBin; + private float[] truckTypeShareMT; + private Matrix distCounties; + private static int[] countyFips; + private int FAFVersion=4; + + + private static String[] empCats = {"Agriculture", "Construction Natural Resources and Mining", "Manufacturing", + "Trade Transportation and Utilities", "Information", "Financial Activities", + "Professional and Business Services", "Education and Health Services", "Leisure and Hospitality", + "Other Services", "coalProduction"}; //, "Government"}; (no government employment available at county-level nationwide at this point + + + public void prepareCountyData(ResourceBundle rb) { + // prepare county data to provide total employment as a weight + logger.info("Preparing county data for flow disaggregation"); + + getUScountyEmployment(rb); + if (readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); + //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share + countyShares = new HashMap<>(); + for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { + for (String com: readFAF2.sctgCommodities) { + TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, null, null); + // dir is not really needed here, but kept for consistency reason as it is required + // in prepareCountyDataForFAF(ResourceBundle rb, TableDataSet detailEmployment) + String codeo = "orig_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; + countyShares.put(codeo, CountyList); + String coded = "dest_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; + countyShares.put(coded, CountyList); + } + } + } + + + public void prepareCountyData(ResourceBundle rb, int yr, TableDataSet detailEmployment, String truckType) { + // prepare county data to provide detailed employment given in detailEmployment and total employment elsewhere + // as a weight + logger.info("Preparing county data for flow disaggregation for year " + yr + "..."); + + HashMap useC = createMakeUseHashMap(rb, "use.coefficients"); + HashMap makeC = createMakeUseHashMap(rb, "make.coefficients"); + + truckTypeShares = fafUtils.importTable(rb.getString("truck.type.by.distance")); + if (truckType.equalsIgnoreCase("weight")) createTruckShareArraysWeight(); + else createTruckShareArraysUnit(); + distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); + + getUScountyEmployment(rb); + + if (readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); + //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share + countyShares = new HashMap<>(); + String[] direction = {"orig", "dest"}; + for (String dir: direction) { + HashMap factors; + if (dir.equals("orig")) factors = makeC; + else factors = useC; + for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { + for (String com: readFAF2.sctgCommodities) { + TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, detailEmployment, factors); + String code = dir + "_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; + countyShares.put(code, CountyList); + } + } + } + } + + + private void createTruckShareArraysWeight() { + // create array with distance bins and shares or medium-heavy trucks + truckTypeShareDistBin = new int[truckTypeShares.getRowCount()]; + truckTypeShareMT = new float[truckTypeShares.getRowCount()]; + for (int row = 1; row <= truckTypeShares.getRowCount(); row++) { + truckTypeShareDistBin[row - 1] = (int) truckTypeShares.getValueAt(row, "DistanceGreaterThan"); + truckTypeShareMT[row - 1] = truckTypeShares.getValueAt(row, "MT(<26k_lbs)"); + // data quality check + float htShare = truckTypeShares.getValueAt(row, "HT(>26k_lbs)"); + if (htShare + truckTypeShareMT[row - 1] != 1) logger.warn("Shares of Medium and Heavy Trucks add up to " + + htShare + truckTypeShareMT[row - 1] + " instead of 1 for distance class " + truckTypeShareDistBin[row - 1]); + } + } + + + private void createTruckShareArraysUnit() { + // create array with distance bins and shares or Single-Unit/Multi-Unit trucks + truckTypeShareDistBin = new int[truckTypeShares.getRowCount()]; + truckTypeShareMT = new float[truckTypeShares.getRowCount()]; + for (int row = 1; row <= truckTypeShares.getRowCount(); row++) { + truckTypeShareDistBin[row - 1] = (int) truckTypeShares.getValueAt(row, "DistanceGreaterThan"); + truckTypeShareMT[row - 1] = truckTypeShares.getValueAt(row, "SUT"); + // data quality check + float htShare = truckTypeShares.getValueAt(row, "MUT"); + if (htShare + truckTypeShareMT[row - 1] != 1) logger.warn("Shares of SUT and MUT Trucks add up to " + + htShare + truckTypeShareMT[row - 1] + " instead of 1 for distance class " + truckTypeShareDistBin[row - 1]); + } + } + + + public void prepareCountyDataForFAF(ResourceBundle rb, int yr, TableDataSet detailEmployment, + TableDataSet specialRegions, int fafVersion) { + // prepare county data to provide detailed employment given in detailEmployment and total employment elsewhere + // as a weight using the "fafVersion" zone system + logger.info("Preparing county data for FAF" + fafVersion + " flow disaggregation for year " + yr + "..."); + + String useToken = "faf" + fafVersion + ".use.coefficients"; + String makeToken = "faf" + fafVersion + ".make.coefficients"; + HashMap useC = createMakeUseHashMap(rb, useToken); + HashMap makeC = createMakeUseHashMap(rb, makeToken); + + distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); + + if (fafVersion == 2 && readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); +// if (fafVersion == 3 && readFAF3.fafRegionList == null) readFAF3.readFAF3referenceLists(rb); // has been read earlier for FAF3 + + //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share + countyShares = new HashMap<>(); + String[] direction = {"orig", "dest"}; + for (String dir: direction) { + HashMap factors; + if (dir.equals("orig")) factors = makeC; + else factors = useC; + + if (fafVersion == 2) { + for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { + for (String com: readFAF2.sctgCommodities) { + TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, detailEmployment, factors); + String code = dir + "_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; + countyShares.put(code, CountyList); + } + } + } else if (fafVersion==3){ // fafVersion == 3 + for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { + for (String com: readFAF3.sctgStringCommodities) { + int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); + TableDataSet CountyList = getCountySpecificDataByFAF(zoneNum, com, detailEmployment, factors); + String code = dir + "_" + zoneNum + "_" + com; + countyShares.put(code, CountyList); + } + } + } + else{ // fafVersion == 4 + for (int fafNum = 1; fafNum <= ReadFAF4.fafRegionList.getRowCount(); fafNum++) { + for (String com: ReadFAF4.sctgStringCommodities) { + int zoneNum = (int) ReadFAF4.fafRegionList.getValueAt(fafNum, "ZoneID"); + TableDataSet CountyList = getCountySpecificDataByFAF(zoneNum, com, detailEmployment, factors); + String code = dir + "_" + zoneNum + "_" + com; + countyShares.put(code, CountyList); + } + } + } + } + + // create fake county lists for special regions such as airports or seaports that shall be kept separate from the FAF regions + if (specialRegions == null) return; + for (int row = 1; row <= specialRegions.getRowCount(); row++) { + int[] fips = {(int) specialRegions.getValueAt(row, "modelCode")}; + String[] name; + if (fafVersion == 2) { + name = new String[]{specialRegions.getStringValueAt(row, "Region")}; + }else if(fafVersion==3){ + name = new String[]{specialRegions.getStringValueAt(row, "faf3code")}; + }else{ + name = new String[]{specialRegions.getStringValueAt(row, "faf4code")}; + } + + TableDataSet CountyList = new TableDataSet(); + float[] emplDummy = {1}; + CountyList.appendColumn(name, "Name"); + CountyList.appendColumn(fips, "COUNTYFIPS"); + CountyList.appendColumn(name, "FAFRegion"); + CountyList.appendColumn(emplDummy, "Employment"); + for (String dir: direction) { + if (fafVersion == 2) { + for (String com: readFAF2.sctgCommodities) { + String code = dir + "_" + name[0] + "_" + com; + countyShares.put(code, CountyList); + } + } else if (fafVersion == 3){ + for (String com: readFAF3.sctgStringCommodities) { + String code = dir + "_" + fips[0] + "_" + com; + countyShares.put(code, CountyList); + } + }else{ + for (String com: ReadFAF4.sctgStringCommodities) { + String code = dir + "_" + fips[0] + "_" + com; + countyShares.put(code, CountyList); + } + + } + + } + } + } + + + public static HashMap createMakeUseHashMap (ResourceBundle rb, String token) { + // create HashMap with make/use coefficients + TableDataSet coeff = fafUtils.importTable(ResourceUtil.getProperty(rb, token)); + HashMap hsm = new HashMap<>(); + for (int i = 1; i <= coeff.getRowCount(); i++) { + String industry = coeff.getStringValueAt(i, "Industry"); + for (int j = 2; j <= coeff.getColumnCount(); j++) { + String sctg = coeff.getColumnLabel(j); + String code = industry + "_" + sctg; + hsm.put(code, coeff.getValueAt(i, j)); + } + } + return hsm; + } + + + public TableDataSet disaggregateSingleFAF2flow(String orig, String dest, String sctg, float tons) { + // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF2 + + TableDataSet flows = new TableDataSet(); + // get county specific data for origin FAF and destination FAF + TableDataSet CountyDatI = getCountyTableDataSet("orig", orig, sctg); + TableDataSet CountyDatJ = getCountyTableDataSet("dest", dest, sctg); + + // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf + double EmplICplusJCtotal = 0; + int count = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + EmplICplusJCtotal += CountyDatI.getValueAt(origCounty, "Employment") + + CountyDatJ.getValueAt(destCounty, "Employment"); + count++; + } + } + int[] oFips = new int[count]; + int[] dFips = new int[count]; + String[] codes = new String[count]; + float[] tonShare = new float[count]; + int k = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); + double EmplICplusJC = CountyDatI.getValueAt(origCounty, "Employment") + + CountyDatJ.getValueAt(destCounty, "Employment"); + tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); + oFips[k] = origFips; + dFips[k] = destFips; + codes[k] = String.valueOf(origFips) + "_" + String.valueOf(destFips); + k++; + } + } + flows.appendColumn(oFips, "oFips"); + flows.appendColumn(dFips, "dFips"); + flows.appendColumn(codes, "Codes"); + flows.appendColumn(tonShare, "Tons"); + return flows; + } + + + public TableDataSet disaggregateSingleFAFFlow(int orig, int dest, String sctg, float tons, float centerDamper) { + // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF3 + + TableDataSet flows = new TableDataSet(); + // get county specific data for origin FAF and destination FAF + + TableDataSet CountyDatI = getCountyWeights("orig", orig, sctg); + TableDataSet CountyDatJ = getCountyWeights("dest", dest, sctg); + if (CountyDatI == null) { + logger.error("Could not find table for disaggregating county " + orig + " with commodity " + sctg); + return null; + } + if (CountyDatJ == null) { + logger.error("Could not find table for disaggregating county " + dest + " with commodity " + sctg); + return null; + } + // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf + double EmplICplusJCtotal = 0; + int count = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + EmplICplusJCtotal += Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * + CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); + count++; + } + } + int[] oFips = new int[count]; + int[] dFips = new int[count]; + String[] codes = new String[count]; + float[] tonShare = new float[count]; + int k = 0; + + if (EmplICplusJCtotal == 0) logger.error("Could not find weight for FAF zone " + orig + " to " + dest + " for " + sctg); + + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); + double EmplICplusJC = Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * + CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); + tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); + + oFips[k] = origFips; + dFips[k] = destFips; + codes[k] = origFips + "_" + destFips; + k++; + } + } + + flows.appendColumn(oFips, "oFips"); + flows.appendColumn(dFips, "dFips"); + flows.appendColumn(codes, "Codes"); + flows.appendColumn(tonShare, "Tons"); + return flows; + } + + + public TableDataSet disaggregateSingleFAFFlowThroughPOE(String dir, TableDataSet poe, int orig, int dest, String sctg, + float tons, float centerDamper) { + // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF3 for import/exports through ports + // of entry/exit + + // get county specific data for origin FAF and destination FAF + TableDataSet CountyDatI = getCountyWeights("orig", orig, sctg); + if (dir.startsWith("import") && poe != null) CountyDatI = poe; + TableDataSet CountyDatJ = getCountyWeights("dest", dest, sctg); + if (dir.startsWith("export") && poe != null) CountyDatJ = poe; + if (CountyDatI == null) { + logger.error("Could not find table for " + dir + " disaggregating county " + orig + " with commodity " + sctg); + return null; + } + if (CountyDatJ == null) { + logger.error("Could not find table for " + dir + " disaggregating county " + dest + " with commodity " + sctg); + return null; + } + // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf + double EmplICplusJCtotal = 0; + int count = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + EmplICplusJCtotal += Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * + CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); + count++; + } + } + int[] oFips = new int[count]; + int[] dFips = new int[count]; + String[] codes = new String[count]; + float[] tonShare = new float[count]; + int k = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); + double EmplICplusJC = Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * + CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); + tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); + oFips[k] = origFips; + dFips[k] = destFips; + codes[k] = origFips + "_" + destFips; + k++; + } + } + TableDataSet flows = new TableDataSet(); + flows.appendColumn(oFips, "oFips"); + flows.appendColumn(dFips, "dFips"); + flows.appendColumn(codes, "Codes"); + flows.appendColumn(tonShare, "Tons"); + return flows; + } + + + public float getCountyDistance(int origFips, int destFips) { + // return distance from origFips to destFips + + try { + return distCounties.getValueAt(origFips, destFips); + } catch (Exception e){ + return -1; + } + } + + + public float[] splitToTruckTypes(int origFips, int destFips, float trucks) { + // split trucks for this OD pair into two truck types + float dist; + try { + dist = distCounties.getValueAt(origFips, destFips); + } catch (Exception e) { + dist = 999; + } + float[] splitTrucks = {0, 0}; + for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) { + if (dist >= truckTypeShareDistBin[dClass]) { + splitTrucks[0] = trucks * truckTypeShareMT[dClass]; + splitTrucks[1] = trucks * (1 - truckTypeShareMT[dClass]); + return splitTrucks; } + } + logger.warn("Could not find truck share for distance " + dist); + return null; + } + + + public double[] splitToTruckTypes(int origFips, int destFips, double trucks) { + // split trucks for this OD pair into two truck types + int dist; + try { + dist = (int) distCounties.getValueAt(origFips, destFips); + if (dist == -999) dist = 9999; // destination unreachable on network, therefore skim is negative, but there might be trips in reality, including ferry + } catch (Exception e) { + dist = 999; + } + double[] splitTrucks = {0, 0}; + for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) + if (dist >= truckTypeShareDistBin[dClass]) { + splitTrucks[0] = trucks * truckTypeShareMT[dClass]; + splitTrucks[1] = trucks * (1 - truckTypeShareMT[dClass]); + return splitTrucks; + } + logger.warn("Could not find truck share for distance " + dist); + return null; + } + + + public double[] splitToTruckTypes(int origFips, int destFips) { + // split trucks for this OD pair into two truck types + int dist; + try { + dist = (int) distCounties.getValueAt(origFips, destFips); + if (dist == -999) dist = 9999; // destination unreachable on network, therefore skim is negative, but there might be trips in reality, including ferry + } catch (Exception e) { + dist = 999; + } + double[] splitTrucks = {0, 0}; + for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) + if (dist >= truckTypeShareDistBin[dClass]) { + splitTrucks[0] = truckTypeShareMT[dClass]; + splitTrucks[1] = (1 - truckTypeShareMT[dClass]); + return splitTrucks; + } + logger.warn("Could not find truck share for distance " + dist); + return null; + } + + + public double[] getTrucksByType(float dist, double sutPL, double mutPL, double tonFlow) { + // convert tons into trucks and split trucks into single-unit and multi-unit trucks + + double[] truckShares = splitToTruckTypes(dist); + double[] trucksByType = new double[2]; + trucksByType[0] = tonFlow / (sutPL + truckShares[1]/truckShares[0] * mutPL); + trucksByType[1] = tonFlow / (truckShares[0]/truckShares[1] * sutPL + mutPL); + return trucksByType; + } + + + public double[] getTrucksByType(float dist, double sutPL, double mutPL, double tonFlow, float adjustmentSUT) { + // convert tons into trucks and split trucks into single-unit and multi-unit trucks + + double[] truckShares = splitToTruckTypes(dist); + truckShares[0] = truckShares[0] * (1. + adjustmentSUT); + truckShares[1] = 1. - truckShares[0]; + double[] trucksByType = new double[2]; + trucksByType[0] = tonFlow / (sutPL + truckShares[1]/truckShares[0] * mutPL); + trucksByType[1] = tonFlow / (truckShares[0]/truckShares[1] * sutPL + mutPL); + return trucksByType; + } + + + public double[] splitToTruckTypes(float distance) { + // split trucks for this OD pair into two truck types + double[] splitTrucks = {0, 0}; + for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) + if (distance >= truckTypeShareDistBin[dClass]) { + splitTrucks[0] = truckTypeShareMT[dClass]; + splitTrucks[1] = (1 - truckTypeShareMT[dClass]); + return splitTrucks; + } + logger.warn("Could not find truck share for distance " + distance); + return null; + } + + + public float[] splitToTruckTypes(String code, float trucks) { + // split trucks for this OD pair into two truck types + String[] origDestFips = code.split("_"); + int origFips = Integer.parseInt(origDestFips[0]); + int destFips = Integer.parseInt(origDestFips[1]); + float dist; + try { + dist = distCounties.getValueAt(origFips, destFips); + } catch (Exception e) { + dist = 1; + } + int distClass = 1; + for (int row = 1; row <= truckTypeShares.getRowCount(); row++) + if (dist > truckTypeShares.getValueAt(row, "DistanceGreaterThan")) distClass = row; + float[] splitTrucks = {0, 0}; + splitTrucks[0] = trucks * truckTypeShares.getValueAt(distClass, "MT(<26k_lbs)"); + splitTrucks[1] = trucks * truckTypeShares.getValueAt(distClass, "HT(>26k_lbs)"); + return splitTrucks; + } + + + private TableDataSet getCountyTableDataSet(String direction, String FAFregion, String sctgCode) { + // check if this FAFregion equals a special generator + TableDataSet tbl; + if (FAFregion.equals("OR Portl_Airport") || FAFregion.equals("OR Portl_Port") || + FAFregion.equals("OR rem_Airport") || FAFregion.equals("OR rem_Port") || FAFregion.equals("Other")) { + tbl = new TableDataSet(); + String[] a = {FAFregion}; + float[] b = new float[1]; + if (FAFregion.equals("OR Portl_Airport")) b[0] = 90001f; + else if (FAFregion.equals("OR Portl_Port")) b[0] = 90002f; + else if (FAFregion.equals("OR rem_Airport")) b[0] = 90003f; + else if (FAFregion.equals("OR rem_Port")) b[0] = 90004f; + else if (FAFregion.equals("Other")) b[0] = 90005f; + float[] c = {1f}; + tbl.appendColumn(a, "Name"); + tbl.appendColumn(b, "COUNTYFIPS"); + tbl.appendColumn(a, "FAFRegion"); + tbl.appendColumn(c, "Employment"); + tbl.setName(FAFregion); + } else { + String code = direction + "_" + FAFregion + "_" + sctgCode; + tbl = countyShares.get(code); + } + return tbl; + } + + + public TableDataSet getCountyWeights(String direction, int FAFregion, String sctgCode) { + // check if this FAFregion equals a special generator + String code = direction + "_" + FAFregion + "_" + sctgCode; + return countyShares.get(code); + } + + + public TableDataSet disaggregateSingleFlowInOregon(String orig, String dest, String sctg, float trucks) { + // disaggregate trucks from orig FAFzone to dest FAFzone to the county level + + if (!orig.contains("OR Portl") && !orig.contains("OR rem")) orig = "Other"; + if (!dest.contains("OR Portl") && !dest.contains("OR rem")) dest = "Other"; + TableDataSet flows = new TableDataSet(); + // get county specific data for origin FAF and destination FAF + TableDataSet CountyDatI = getCountyTableDataSet("orig", orig, sctg); + TableDataSet CountyDatJ = getCountyTableDataSet("dest", dest, sctg); + + // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf + double EmplICplusJCtotal = 0; + int count = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + EmplICplusJCtotal += CountyDatI.getValueAt(origCounty, "Employment") + + CountyDatJ.getValueAt(destCounty, "Employment"); + count++; + } + } + String[] codes = new String[count]; + float[] tonShare = new float[count]; + int k = 0; + for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { + int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); + for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { + int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); + double EmplICplusJC = CountyDatI.getValueAt(origCounty, "Employment") + + CountyDatJ.getValueAt(destCounty, "Employment"); + tonShare[k] = (float) (trucks * EmplICplusJC / EmplICplusJCtotal); + codes[k] = String.valueOf(origFips) + "_" + String.valueOf(destFips); + k++; + } + } + flows.appendColumn(codes, "Codes"); + flows.appendColumn(tonShare, "ShortTons"); + return flows; + } + + + public static void getUScountyEmployment(ResourceBundle rb) { + // read employment and county id data + + logger.info("Reading County Employment Data..."); + countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); + + TableDataSet StatesTable = fafUtils.importTable(ResourceUtil.getProperty(rb, "state.list")); + String[] StateNames = StatesTable.getColumnAsString("StateName"); + TableDataSet[] StateEmployment = new TableDataSet[StateNames.length]; + for (int st = 0; st < StateNames.length; st++) { + String fileName = ResourceUtil.getProperty(rb, "state.employment.prefix") + StateNames[st] + ".csv"; + StateEmployment[st] = fafUtils.importTable(fileName); + StateEmployment[st].setName(StateNames[st]); + } + + String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; + int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; + countyIDsWithEmployment.appendColumn(tempString, "stateName"); + countyIDsWithEmployment.appendColumn(tempInt, "Employment"); + + // assign employment to every county + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; + String CountyStateAbbreviation = countyIDsWithEmployment.getStringValueAt(i, "StateCode"); + String CountyName = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + + // find full state name + String StateNameOfCounty = " "; + for (int j = 1; j <= StatesTable.getRowCount(); j++) + if (StatesTable.getStringValueAt(j, "StateAbbreviation").equals(CountyStateAbbreviation)) + StateNameOfCounty = StatesTable.getStringValueAt(j, "stateName"); + + // find right employment table for this state + int StateNumber = 0; + for (int j = 0; j < StateEmployment.length; j++) { + if (StateEmployment[j].getName().equals(StateNameOfCounty)) StateNumber = j; + } + + // delete state code from county name + StringBuffer sb = new StringBuffer(); + sb.append(CountyName); + int n = sb.lastIndexOf(CountyStateAbbreviation); + sb.delete(n-1, n+2); + // add ", StateAbbreviation" + sb.append(", "); + sb.append(CountyStateAbbreviation); + CountyName = sb.toString(); + + // find employment of current county + boolean found = false; + for (int k = 1; k <= StateEmployment[StateNumber].getRowCount(); k++) { + if (StateEmployment[StateNumber].getStringValueAt(k, "Region").equalsIgnoreCase(CountyName)) { + countyIDsWithEmployment.setValueAt + (i, "Employment", StateEmployment[StateNumber].getValueAt(k, "Employment")); + countyIDsWithEmployment.setStringValueAt + (i, countyIDsWithEmployment.getColumnPosition("stateName"), StateNameOfCounty); + found = true; + } + } + + // County Broomfield, Colorado has no polygon in the county layer. Employment is added to Boulder County: + if (CountyName.equals("BOULDER, CO")) { + for (int k = 1; k <= StateEmployment[StateNumber].getRowCount(); k++) { + if (StateEmployment[StateNumber].getStringValueAt(k, "Region").equals("Broomfield, CO")) { + float BoulderBroomfieldEmpl = countyIDsWithEmployment.getValueAt(i, "Employment") + + StateEmployment[StateNumber].getValueAt(k, "Employment"); + countyIDsWithEmployment.setValueAt(i, "Employment", BoulderBroomfieldEmpl); + } + } + } + + // write error message (unless it is an island of UM or Guam) + if (!found && !CountyStateAbbreviation.equals("UM") && !CountyStateAbbreviation.equals("GU")) + logger.warn("Not found: " + CountyName + " (" + + countyIDsWithEmployment.getStringValueAt(i, "TYPE") + ")"); + } + } + + + public static void getUScountyEmploymentFromOneFile (ResourceBundle rb) { + // read file with county employment in the US + + logger.info("Reading County Employment Data..."); + countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); + TableDataSet countyEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "us.county.employment")); + + String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; + int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; + countyIDsWithEmployment.appendColumn(tempString, "stateName"); + countyIDsWithEmployment.appendColumn(tempInt, "Employment"); + + // assign employment to every county + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; + // skip Guam and United States Minor Outlying Islands + if (countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("GU") || + countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("UM")) continue; + float fips = countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + // find employment of current county + boolean found = false; + for (int row = 1; row <= countyEmployment.getRowCount(); row++) { + if (countyEmployment.getValueAt(row, "fips") == fips) { + countyIDsWithEmployment.setValueAt + (i, "Employment", countyEmployment.getValueAt(row, "totalAnnualAverageEmployment")); + found = true; + } + } + + // write error message if county was not found + if (!found) { + String CountyName = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + logger.warn("Not found: " + CountyName + " (" + + countyIDsWithEmployment.getStringValueAt(i, "TYPE") + ")"); + } + } + } + + + public void getUScountyEmploymentByIndustry (ResourceBundle rb) { + // read file with employment by county by industry + + logger.info(" Reading County Employment Data..."); + countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); + TableDataSet countyEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "us.county.employment.by.ind")); + countyEmployment.buildIndex(countyEmployment.getColumnPosition("FIPS")); + TableDataSet agEmpl = fafUtils.importTable(rb.getString("us.county.employment.agricult")); + agEmpl.buildIndex(agEmpl.getColumnPosition("FIPS")); + + String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; + int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; + countyIDsWithEmployment.appendColumn(tempString, "stateName"); + for (String emp: empCats) countyIDsWithEmployment.appendColumn(tempInt, emp); + + // assign employment to every county + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && + !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; + // skip Guam and United States Minor Outlying Islands + if (countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("GU") || + countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("UM")) continue; + int fips = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + // find employment of current county + float consNatResMinEmployment = countyEmployment.getIndexedValueAt(fips, "Natural Resources and Mining") + + countyEmployment.getIndexedValueAt(fips, "Construction"); + countyIDsWithEmployment.setValueAt(i, "Construction Natural Resources and Mining", consNatResMinEmployment); + countyIDsWithEmployment.setValueAt(i, "Manufacturing", countyEmployment.getIndexedValueAt(fips, "Manufacturing")); + countyIDsWithEmployment.setValueAt(i, "Trade Transportation and Utilities", countyEmployment.getIndexedValueAt(fips, "Trade, Transportation, and Utilities")); + countyIDsWithEmployment.setValueAt(i, "Information", countyEmployment.getIndexedValueAt(fips, "Information")); + countyIDsWithEmployment.setValueAt(i, "Financial Activities", countyEmployment.getIndexedValueAt(fips, "Financial Activities")); + countyIDsWithEmployment.setValueAt(i, "Professional and Business Services", countyEmployment.getIndexedValueAt(fips, "Professional and Business Services")); + countyIDsWithEmployment.setValueAt(i, "Education and Health Services", countyEmployment.getIndexedValueAt(fips, "Education and Health Services")); + countyIDsWithEmployment.setValueAt(i, "Leisure and Hospitality", countyEmployment.getIndexedValueAt(fips, "Leisure and Hospitality")); + countyIDsWithEmployment.setValueAt(i, "Other Services", countyEmployment.getIndexedValueAt(fips, "Other Services")); + + // agricultural employment is missing in most of Alaska and in independent cities (deemed not to be relevant) + try { + countyIDsWithEmployment.setValueAt(i, "Agriculture", agEmpl.getIndexedValueAt(fips, "agEmployment")); + } catch (Exception e) { + // Set to minor value to ensure that all FAF ag production can be allocated. If the FAF zone of this + // county has other counties with ag production, this miniWeight will be irrelevant as it is small. + // If no county in this FAF zone has ag production, the employment "Construction Natural Resources and Mining" + // will be used to disaggregate flows (with all counties being multiplied by 0.0001, which doesn't affect the result. + float miniWeight = 0.0001f * countyIDsWithEmployment.getValueAt(i, "Construction Natural Resources and Mining"); + countyIDsWithEmployment.setValueAt(i, "Agriculture", miniWeight); + + } + } + // try to add coal production as a weight + try { + TableDataSet coalProd = fafUtils.importTable(rb.getString("mining.production")); + coalProd.buildIndex(coalProd.getColumnPosition("FIPS")); + countyIDsWithEmployment.appendColumn(tempInt, "coalProduction"); + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + int fips = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + try { + countyIDsWithEmployment.setValueAt(i, "coalProduction", coalProd.getIndexedValueAt(fips, "Production_total")); + } catch (Exception e) { + // Set to minor value to ensure that all FAF coal production can be allocated. If the FAF zone of this + // county has other counties with coal production, this miniWeight will be irrelevant as it is small. + // If no county in this FAF zone has coal production, the employment "Construction Natural Resources and Mining" + // will be used to disaggregate flows (with all counties being multiplied by 0.0001, which doesn't affect the result. + float miniWeight = 0.0001f * countyIDsWithEmployment.getValueAt(i, "Construction Natural Resources and Mining"); + countyIDsWithEmployment.setValueAt(i, "coalProduction", miniWeight); + } + } + } catch (Exception e) { + logger.warn("Coal production not defined."); + } + countyFips = countyIDsWithEmployment.getColumnAsInt( + countyIDsWithEmployment.getColumnPosition("COUNTYFIPS")); + } + + + public int[] getCountyFips () { + return countyFips; + } + + + public int getStateNumberOfCounty (int fips) { + for (int row = 1; row <= countyIDsWithEmployment.getRowCount(); row ++) { + if (countyIDsWithEmployment.getValueAt(row, "COUNTYFIPS") == fips) return (int) countyIDsWithEmployment.getValueAt(row, "State"); + } + logger.warn ("State of county " + fips + " was not found."); + return -1; + } + + + public void defineTruckTypes (String truckType, ResourceBundle rb) { + // define truck types by weight or size + + truckTypeShares = fafUtils.importTable(rb.getString("truck.type.by.distance")); + if (truckType.equalsIgnoreCase("weight")) createTruckShareArraysWeight(); + else createTruckShareArraysUnit(); + } + + + private TableDataSet getCountySpecificDataByFAF2(int FAFNumber, String com, TableDataSet detEmpl, + HashMap factors) { + // get employment as a weight for FAF zone FAFNumber in FAF2 + // where detailed employment is available in detEmpl, use make/use factors for commodity-specific weights + + TableDataSet CountySpecifics = new TableDataSet(); + String nameOfFAF = readFAF2.domRegionList.getStringValueAt(FAFNumber, "RegionName"); + CountySpecifics.setName(nameOfFAF); + + int NoCountiesInFAF = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getStringValueAt(i, "FafRegion").equals(nameOfFAF)) NoCountiesInFAF += 1; + } + String[] CountyName = new String[NoCountiesInFAF]; + int[] CountyFips = new int[NoCountiesInFAF]; + String[] CountyFAFRegion = new String[NoCountiesInFAF]; + float[] countyEmpl = new float[NoCountiesInFAF]; + boolean hasDetailedEmployment = checkIfFAFzoneHasDetailedEmployment(nameOfFAF, NoCountiesInFAF, detEmpl, 2); + + int k = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getStringValueAt(i, "FafRegion").equals(nameOfFAF)){ + CountyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + CountyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + CountyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, "FafRegion"); + if (hasDetailedEmployment) { + for (int row = 1; row <= detEmpl.getRowCount(); row++) + if (detEmpl.getValueAt(row, "CountyFips") == CountyFips[k]) + countyEmpl[k] = getWeightedEmpl(row, com, detEmpl, factors); + } else { + countyEmpl[k] = countyIDsWithEmployment.getValueAt(i, "Employment"); + } + k++; + } + } + CountySpecifics.appendColumn(CountyName, "Name"); + CountySpecifics.appendColumn(CountyFips, "COUNTYFIPS"); + CountySpecifics.appendColumn(CountyFAFRegion, "FAFRegion"); + CountySpecifics.appendColumn(countyEmpl, "Employment"); + return CountySpecifics; + } + + + private TableDataSet getCountySpecificDataByFAF(int FAFNumber, String com, TableDataSet detEmployment, + HashMap factors) { + + String fieldName = "FAF3region"; + if(FAFVersion==4) + fieldName = "FAF4region"; + + // get employment as a weight for FAF zone FAFNumber in FAF3 + // where detailed employment is available in detEmpl, use make/use factors for commodity-specific weights + + if (FAFNumber > 800) { + // foreign FAF zone + TableDataSet foreignCountry = new TableDataSet(); + String[] countyName = {String.valueOf(FAFNumber)}; + int[] countyFips = {-1}; + String[] countyFAFRegion = {String.valueOf(FAFNumber)}; + int[] countyEmpl = {1}; + foreignCountry.appendColumn(countyName, "Name"); + foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); + foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); + foreignCountry.appendColumn(countyEmpl, "Employment"); + return foreignCountry; + } + TableDataSet CountySpecifics = new TableDataSet(); + String nameOfFAF = null; + if(FAFVersion==3) + nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); + else + nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); + CountySpecifics.setName(nameOfFAF); + + int NoCountiesInFAF = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; + } + if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); + String[] countyName = new String[NoCountiesInFAF]; + int[] countyFips = new int[NoCountiesInFAF]; + String[] countyFAFRegion = new String[NoCountiesInFAF]; + float[] countyEmpl = new float[NoCountiesInFAF]; + boolean hasDetailedEmployment = checkIfFAFzoneHasDetailedEmployment(nameOfFAF, NoCountiesInFAF, detEmployment, 3); + + int k = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ + countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); + if (hasDetailedEmployment) { + for (int row = 1; row <= detEmployment.getRowCount(); row++) + if (detEmployment.getValueAt(row, "CountyFips") == countyFips[k]) + countyEmpl[k] = getWeightedEmpl(row, com, detEmployment, factors); + } else { + countyEmpl[k] = countyIDsWithEmployment.getValueAt(i, "Employment"); + } + k++; + } + } + CountySpecifics.appendColumn(countyName, "Name"); + CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); + CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); + CountySpecifics.appendColumn(countyEmpl, "Employment"); + return CountySpecifics; + } + + + private TableDataSet getCountySpecificDataByFAFwithDetailedEmployment (int FAFNumber, String com, + HashMap factors, float officeReduction) { + // get employment as a weight for FAF zone FAFNumber in FAF3, use make/use factors for commodity-specific weights + String fieldName = "FAF3region"; + if(FAFVersion==4) + fieldName = "FAF4region"; + + if (FAFNumber > 800) { + // foreign FAF zone + TableDataSet foreignCountry = new TableDataSet(); + String[] countyName = {String.valueOf(FAFNumber)}; + int[] countyFips = {-1}; + String[] countyFAFRegion = {String.valueOf(FAFNumber)}; + int[] countyEmpl = {1}; + foreignCountry.appendColumn(countyName, "Name"); + foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); + foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); + foreignCountry.appendColumn(countyEmpl, "Employment"); + return foreignCountry; + } + TableDataSet CountySpecifics = new TableDataSet(); + String nameOfFAF = null; + + if(FAFVersion==3) + nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); + else + nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); + + CountySpecifics.setName(nameOfFAF); + + int NoCountiesInFAF = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; + } + if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); + String[] countyName = new String[NoCountiesInFAF]; + int[] countyFips = new int[NoCountiesInFAF]; + String[] countyFAFRegion = new String[NoCountiesInFAF]; + float[] countyEmpl = new float[NoCountiesInFAF]; + + int k = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ + countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); + countyEmpl[k] = getWeightedEmpl(i, com, factors, officeReduction); +// if(countyFips[k]==37082 || countyFips[k]==37058 || countyFips[k]==37152 ){//|| destFips==37082 || destFips==37058 || destFips==37152){ +// logger.info("FIPS: ("+countyFips[k]+") WeightedEmp: "+countyEmpl[k]); +// } + k++; + } + } + CountySpecifics.appendColumn(countyName, "Name"); + CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); + CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); + CountySpecifics.appendColumn(countyEmpl, "Employment"); + return CountySpecifics; + } + + + private TableDataSet getCountySpecificDataByFAFwithDetailedEmploymentByType (int FAFNumber, String com, + HashMap factors, float officeReduction) { + // get employment as a weight for FAF zone FAFNumber in FAF3, use make/use factors for commodity-specific weights + // keep track of single employment types + + String fieldName = "FAF3region"; + if(FAFVersion==4) + fieldName = "FAF4region"; + + if (FAFNumber > 800) { + // foreign FAF zone + TableDataSet foreignCountry = new TableDataSet(); + String[] countyName = {String.valueOf(FAFNumber)}; + int[] countyFips = {-1}; + String[] countyFAFRegion = {String.valueOf(FAFNumber)}; + int[] countyEmpl = {1}; + foreignCountry.appendColumn(countyName, "Name"); + foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); + foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); + foreignCountry.appendColumn(countyEmpl, "Employment"); + return foreignCountry; + } + TableDataSet CountySpecifics = new TableDataSet(); + + String nameOfFAF=null; + if(FAFVersion==3) + nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); + else + nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); + + CountySpecifics.setName(nameOfFAF); + + int NoCountiesInFAF = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; + } + if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); + String[] countyName = new String[NoCountiesInFAF]; + int[] countyFips = new int[NoCountiesInFAF]; + String[] countyFAFRegion = new String[NoCountiesInFAF]; + float[][] countyEmpl = new float[NoCountiesInFAF][empCats.length+1]; + + int k = 0; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { + if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ + countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); + countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); + countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); + HashMap detEmpl = getWeightedEmplByEmpl(i, com, factors, officeReduction); + countyEmpl[k][empCats.length] = detEmpl.get("total"); + for (int emp = 0; emp < empCats.length; emp++) countyEmpl[k][emp] = detEmpl.get(empCats[emp]); + k++; + } + } + CountySpecifics.appendColumn(countyName, "Name"); + CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); + CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); + for (int emp = 0; emp <= empCats.length; emp++) { + float[] copy = new float[NoCountiesInFAF]; + for (int i = 0; i < copy.length; i++) copy[i] = countyEmpl[i][emp]; + if (emp < empCats.length) CountySpecifics.appendColumn(copy, empCats[emp]); + else CountySpecifics.appendColumn(copy, "Employment"); + } + return CountySpecifics; + } + + + public void prepareCountyDataForFAFwithDetailedEmployment(ResourceBundle rb, int yr, TableDataSet specialRegions) { + // prepare county data to provide detailed employment as a weight using the "fafVersion" zone system + + logger.info("Preparing county data with detailed employment for FAF flow disaggregation for year " + yr + "..."); + HashMap useC = createMakeUseHashMap(rb, "faf.use.coefficients"); + HashMap makeC = createMakeUseHashMap(rb, "faf.make.coefficients"); + distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); + float reduction = (float) ResourceUtil.getDoubleProperty(rb, "reduction.local.office.weight", 1d); + + //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share + countyShares = new HashMap<>(); + String[] direction = {"orig", "dest"}; + for (String dir: direction) { + HashMap factors; + if (dir.equals("orig")) factors = makeC; + else factors = useC; + + for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { + for (String com: readFAF3.sctgStringCommodities) { + int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); + TableDataSet CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); + String code = dir + "_" + zoneNum + "_" + com; + countyShares.put(code, CountyList); + } + } + } + + // create fake county lists for special regions such as airports or seaports that shall be kept separate from the FAF regions + if (specialRegions == null) return; // OK truck flows + for (int row = 1; row <= specialRegions.getRowCount(); row++) { + int[] fips = {(int) specialRegions.getValueAt(row, "modelCode")}; + String[] name = new String[]{specialRegions.getStringValueAt(row, "faf3code")}; + TableDataSet CountyList = new TableDataSet(); + float[] emplDummy = {1}; + CountyList.appendColumn(name, "Name"); + CountyList.appendColumn(fips, "COUNTYFIPS"); + CountyList.appendColumn(name, "FAFRegion"); + CountyList.appendColumn(emplDummy, "Employment"); + for (String dir: direction) { + for (String com: readFAF3.sctgStringCommodities) { + String code = dir + "_" + fips[0] + "_" + com; + countyShares.put(code, CountyList); + } + } + } + } + + + public void prepareCountyDataForFAFwithDetailedEmployment(ResourceBundle rb, int yr, boolean keepTrackOfEmpType) { + // prepare county data to provide detailed employment as a weight + + logger.info(" Preparing county data with detailed employment for FAF flow disaggregation for year " + yr + "..."); + HashMap useC = createMakeUseHashMap(rb, "faf.use.coefficients"); + HashMap makeC = createMakeUseHashMap(rb, "faf.make.coefficients"); + distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); + float reduction = (float) ResourceUtil.getDoubleProperty(rb, "reduction.local.office.weight", 1d); + + //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share + countyShares = new HashMap<>(); + String[] direction = {"orig", "dest"}; + for (String dir: direction) { + HashMap factors; + if (dir.equals("orig")) factors = makeC; + else factors = useC; + + Object readFAF = null; + if(FAFVersion==3){ + + for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { + for (String com: readFAF3.sctgStringCommodities) { + int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); + TableDataSet CountyList; + if (!keepTrackOfEmpType) { + CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); + } else { + CountyList = getCountySpecificDataByFAFwithDetailedEmploymentByType(zoneNum, com, factors, reduction); + } + String code = dir + "_" + zoneNum + "_" + com; + countyShares.put(code, CountyList); + } + } + }else{ + for (int fafNum = 1; fafNum <= ReadFAF4.fafRegionList.getRowCount(); fafNum++) { + for (String com: ReadFAF4.sctgStringCommodities) { + int zoneNum = (int) ReadFAF4.fafRegionList.getValueAt(fafNum, "ZoneID"); + TableDataSet CountyList; + if (!keepTrackOfEmpType) { + CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); + } else { + CountyList = getCountySpecificDataByFAFwithDetailedEmploymentByType(zoneNum, com, factors, reduction); + } + String code = dir + "_" + zoneNum + "_" + com; + countyShares.put(code, CountyList); + } + } + + } + } + } + + + public void scaleSelectedCounties (ResourceBundle rb) { + // scale weight of employment up or down for selected counties + String fieldName = "FAF3region"; + if(FAFVersion==4) + fieldName = "FAF4region"; + + TableDataSet countyScaler = fafUtils.importTable(rb.getString("county.scaler")); + for (int row = 1; row <= countyScaler.getRowCount(); row++) { + int fips= (int) countyScaler.getValueAt(row, "countyFips"); + float scaler = countyScaler.getValueAt(row, "scaler"); + // find FAF zone + int fafZone = 0; + for (int countyRow = 1; countyRow <= countyIDsWithEmployment.getRowCount(); countyRow++) { + if (countyIDsWithEmployment.getValueAt(countyRow, "COUNTYFIPS") == fips) fafZone = + (int) countyIDsWithEmployment.getValueAt(countyRow, fieldName); + } + if (fafZone == 0) logger.error("Could not find FAF zone of FIPS " + fips + " in file " + rb.getString("county.ID")); + String[] direction = {"orig", "dest"}; + for (String dir: direction) { + + if(FAFVersion==3){ + for (String com: readFAF3.sctgStringCommodities) { + String code = dir + "_" + fafZone + "_" + com; + TableDataSet scalerThisCounty = countyShares.get(code); + for (int scalerRow = 1; scalerRow <= scalerThisCounty.getRowCount(); scalerRow++) { + if (scalerThisCounty.getValueAt(scalerRow, "COUNTYFIPS") == fips) { + float value = scalerThisCounty.getValueAt(scalerRow, "Employment") * scaler; + scalerThisCounty.setValueAt(scalerRow, "Employment", value); + } + } + } + }else{ + for (String com: ReadFAF4.sctgStringCommodities) { + String code = dir + "_" + fafZone + "_" + com; + TableDataSet scalerThisCounty = countyShares.get(code); + for (int scalerRow = 1; scalerRow <= scalerThisCounty.getRowCount(); scalerRow++) { + if (scalerThisCounty.getValueAt(scalerRow, "COUNTYFIPS") == fips) { + float value = scalerThisCounty.getValueAt(scalerRow, "Employment") * scaler; + scalerThisCounty.setValueAt(scalerRow, "Employment", value); + } + } + } + + + } + } + } + } + + + private boolean checkIfFAFzoneHasDetailedEmployment(String nameOfFAF, int noCountiesInFAF, TableDataSet detEmpl, + int fafVersion) { + // Check if detEmpl has detailed employment for all counties in FAF zone FAFNumber + + if (detEmpl == null) return false; + boolean[] countyHasDet = new boolean[noCountiesInFAF]; + int k = 0; + String colLabel; + if (fafVersion == 2) colLabel = "FafRegion"; + else colLabel = "FAF3region"; + for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) + if (countyIDsWithEmployment.getStringValueAt(i, colLabel).equals(nameOfFAF)) + for (int row = 1; row <= detEmpl.getRowCount(); row++) { + if (countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS") == detEmpl.getValueAt(row, "CountyFips")) { + countyHasDet[k] = true; + k++; + } + } + int count = 0; + for (boolean county: countyHasDet) if (county) count++; + if (count == 0) return false; + else if (count == noCountiesInFAF) return true; + else logger.fatal("Detailed employment covers FAF zone " + nameOfFAF + " only partly."); + return false; + } + + + private float getWeightedEmpl(int row, String comFAF2, TableDataSet detEmpl, HashMap factors) { + // calculate weighted employment based on make/use coefficients + + float empl = 0; + String[] emplCategoriesTemp = detEmpl.getColumnLabels(); + String[] emplCategories = new String[emplCategoriesTemp.length-2]; + System.arraycopy(emplCategoriesTemp, 2, emplCategories, 0, emplCategories.length); + for (String emplCat: emplCategories) { + String code = emplCat + "_" + comFAF2; + float coeff = factors.get(code); + for (int col = 3; col <= detEmpl.getColumnCount(); col++) { + empl += detEmpl.getValueAt(row, col) * coeff; + } + } + return empl; + } + + + private float getWeightedEmpl(int row, int comFAF3, TableDataSet detEmpl, HashMap factors) { + // calculate weighted employment based on make/use coefficients + + float empl = 0; + String[] emplCategoriesTemp = detEmpl.getColumnLabels(); + String[] emplCategories = new String[emplCategoriesTemp.length-2]; + System.arraycopy(emplCategoriesTemp, 2, emplCategories, 0, emplCategories.length); + for (String emplCat: emplCategories) { + String code = emplCat + "_" + String.valueOf(comFAF3); + float coeff = factors.get(code); + for (int col = 3; col <= detEmpl.getColumnCount(); col++) { + empl += detEmpl.getValueAt(row, col) * coeff; + } + } + return empl; + } + + + private float getWeightedEmpl(int row, String comFAF, HashMap factors, float officeReduction) { + // calculate weighted employment based on make/use coefficients + float empl = 0; + for (String emp: empCats) { + String code = emp + "_" + comFAF; + float coeff = factors.get(code); + float factor = 1; + if (fafUtils.arrayContainsElement(emp, new String[]{"Information", "Financial Activities", + "Professional and Business Services", "Education and Health Services", + "Leisure and Hospitality"})) factor = officeReduction; + empl += countyIDsWithEmployment.getValueAt(row, emp) * coeff * factor; + } + return empl; + } + + + private HashMap getWeightedEmplByEmpl(int row, String comFAF, HashMap factors, float officeReduction) { + // calculate weighted employment based on make/use coefficients, keep information by employment type + + HashMap empl = new HashMap<>(); + float total = 0; + for (String emp: empCats) { + String code = emp + "_" + comFAF; + float coeff = factors.get(code); + float factor = 1; + if (fafUtils.arrayContainsElement(emp, new String[]{"Information", "Financial Activities", + "Professional and Business Services", "Education and Health Services", + "Leisure and Hospitality"})) factor = officeReduction; + float thisVal = countyIDsWithEmployment.getValueAt(row, emp) * coeff * factor; + empl.put(emp, thisVal); + total += thisVal; + } + empl.put("total", total); + return empl; + } + + + public double[][] disaggCountyToZones(float flow, double[] weightsA, double[] weightsB) { + // disaggregate flow from orig county A to orig zones i and dest county B to dest zones j + + // sum up weight products + double sm = 0; + for (double thisWeightsA : weightsA) { + for (double thisWeightsB : weightsB) { + sm += thisWeightsA * thisWeightsB; + } + } + // disaggregate flow + double[][] disFlow = new double[weightsA.length][weightsB.length]; + for (int i = 0; i < weightsA.length; i++) { + for (int j = 0; j < weightsB.length; j++) { + disFlow[i][j] = flow * weightsA[i] * weightsB[j] / sm; + } + } + return disFlow; + } + + + public static int getCountyId(int fips) { + // Return region code of regName + for (int i = 0; i < countyFips.length; i++) { + if (countyFips[i] == fips) return i; + } + logger.error("Could not find county FIPS code " + fips); + return -1; + } + + + public String[] getEmpCats() { + return empCats; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java new file mode 100644 index 0000000..ee20bc4 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java @@ -0,0 +1,331 @@ +package org.sandag.htm.processFAF; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.datafile.CSVFileReader; +import com.pb.common.util.ResourceUtil; + +import java.io.*; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.ResourceBundle; +import org.apache.log4j.Logger; + +/** + * Utilities to process FAF2 data + * User: Rolf Moeckel + * Date: May 6, 2009 + */ +public class fafUtils { + + private static Logger logger = Logger.getLogger(fafUtils.class); + private static ResourceBundle rb; + private static TableDataSet payloadSTCC; + private static TableDataSet payloadSCTG; + + public static TableDataSet importTable(String filePath) { + // read a csv file into a TableDataSet + TableDataSet tblData; + CSVFileReader cfrReader = new CSVFileReader(); + try { + tblData = cfrReader.readFile(new File( filePath )); + } catch (Exception e) { + throw new RuntimeException("File not found: <" + filePath + ">.", e); + } + cfrReader.close(); + return tblData; + } + + + public static ResourceBundle getResourceBundle(String pathToRb) { + File propFile = new File(pathToRb); + rb = ResourceUtil.getPropertyBundle(propFile); + if (rb == null) logger.fatal ("Problem loading resource bundle: " + pathToRb); + return rb; + } + + + public static void setResourceBundle (ResourceBundle appRb) { + rb = appRb; + } + + + public static PrintWriter openFileForSequentialWriting(String fileName) { + File outputFile = new File(fileName); + FileWriter fw = null; + try { + fw = new FileWriter(outputFile); + } catch (IOException e) { + logger.error("Could not open file <" + fileName + ">."); + } + BufferedWriter bw = new BufferedWriter(fw); + return new PrintWriter(bw); + } + + + public static HashMap createScalerHashMap (String[] tokens, double[] values) { + // create HashMap with state O-D pairs that need to be scaled + + HashMap scaler = new HashMap<>(); + if (tokens.length != values.length) { + throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); + } + for (int i=0; i createScalerHashMap (TableDataSet scaleTable, String[] columnNames) { + // create HashMap with state O-D pairs that need to be scaled + + HashMap scaler = new HashMap<>(); + for (int row = 1; row < scaleTable.getRowCount(); row++) { + String token = String.valueOf((int) scaleTable.getValueAt(row, columnNames[0])) + "_" + + String.valueOf((int) scaleTable.getValueAt(row, columnNames[1])); + scaler.put(token, scaleTable.getValueAt(row, columnNames[2])); + } + return scaler; + } + + + public static int isOnPosition (String txt, String[] txtArray) { + // checks if txt is a value of txtArray + int position = -1; + for (int i = 0; i < txtArray.length; i++) if (txtArray[i].equals(txt)) position = i; + return position; + } + + + public static int[] createCountyFipsArray (int[] specRegCodes) { + // create array with county FIPS codes including special regions (such as airports) + + // Note: Used to readFAF2 employment from here, but should be done separately for each project +// disaggregateFlows.getUScountyEmploymentFromOneFile(appRb); + int[] countyFipsS = disaggregateFlows.countyIDsWithEmployment.getColumnAsInt( + disaggregateFlows.countyIDsWithEmployment.getColumnPosition("COUNTYFIPS")); + int[] countyFips = new int[countyFipsS.length + specRegCodes.length]; + System.arraycopy(countyFipsS, 0, countyFips, 0, countyFipsS.length); + System.arraycopy(specRegCodes, 0, countyFips, countyFipsS.length, specRegCodes.length); + return countyFips; + } + + + public static float findAveragePayload(String comm, String comClass) { + // returns average payload in tons per truck for commodity comm + + if (comm.equals("SCTG99")) comm = "SCTG42"; // FAF3 calls unknown SCTG99 instead of SCTG42 + TableDataSet payload = new TableDataSet(); + if (comClass.equals("STCC")) { + if (payloadSTCC == null) payloadSTCC = importTable(ResourceUtil.getProperty(rb, "truck.commodity.payload")); + payload = payloadSTCC; + } else if (comClass.equals("SCTG")) { + if (payloadSCTG == null) payloadSCTG = importTable(ResourceUtil.getProperty(rb, "truck.SCTG.commodity.payload")); + payload = payloadSCTG; + } + int n = -1; + for (int k = 1; k <= payload.getRowCount(); k++) { + if (payload.getStringValueAt(k, "Commodity").equals(comm)) n = k; + } + if (n == -1) { + logger.fatal("Commodity " + comm + " not found in payload factor file."); + System.exit(1); + } + // weight of each truck type, using an average for all commodities, derived from fhwa website + if (payload.containsColumn("Single Unit Trucks")) { + float[] weights = new float[] {0.307f, 0.155f, 0.269f, 0.269f}; // including pickups, minivans, other light vans, SUVs: 0.93642251f,0.022471435f,0.010687433f,0.030418621f + return (payload.getValueAt(n, "Single Unit Trucks") * weights[0]) + + (payload.getValueAt(n, "Semi Trailer") * weights[1]) + + (payload.getValueAt(n, "Double Trailers") * weights[2]) + + (payload.getValueAt(n, "Triples") * weights[3]); + } else { + return (payload.getValueAt(n, "Payload (lbs)") * (float) 0.0005); + } + } + + + public static void readPayloadFactors (ResourceBundle appRb) { + payloadSCTG = importTable(ResourceUtil.getProperty(appRb, "truck.SCTG.commodity.payload")); + } + + + public static float findAveragePayload(String comm) { + // returns average payload in tons per truck for commodity comm + + if (comm.equals("SCTG99")) comm = "SCTG42"; // FAF3 calls unknown SCTG99 instead of SCTG42 + int n = -1; + for (int k = 1; k <= payloadSCTG.getRowCount(); k++) { + if (payloadSCTG.getStringValueAt(k, "Commodity").equals(comm)) n = k; + } + if (n == -1) { + logger.fatal("Commodity " + comm + " not found in payload factor file."); + System.exit(1); + } + return (payloadSCTG.getValueAt(n, "Payload (lbs)") * (float) 0.0005); + } + + + public static int getEnumOrderNumber(ModesFAF mode) { + // return order number of mode + int num = 1; + for (ModesFAF thisMode: ModesFAF.values()) { + if (thisMode == mode) return num; + num++; + } + logger.warn("Could not find mode " + mode.toString()); + return 0; + } + + + public static ModesFAF getModeName (int mode) { + return ModesFAF.values()[mode - 1]; + } + + + public static int getHighestVal(int[] array) { + // return highest number in array + int high = Integer.MIN_VALUE; + for (int num: array) high = Math.max(high, num); + return high; + } + + + public static double sumArray (double[][] arr) { + // sum a two-dimensional double array + + double sum = 0; + for (double[] anArr : arr) { + for (int j = 0; j < arr[1].length; j++) { + sum += anArr[j]; + } + } + return sum; + } + + + public static TableDataSet createSpecialRegions(String[] specRegNames, String[] specRegModes, int[] specRegCodes, + int [] specRegZones, int[] specRegFAFCodes) { + // create TableDataSet with special regions + + if (specRegCodes.length != specRegNames.length || specRegCodes.length != specRegModes.length|| specRegCodes.length != specRegFAFCodes.length) { + logger.error ("Names of special regions and modes of special regions and codes of special regions and " + + "FAF codes of special regions all need to be of same length. No special regions created."); + return null; + } + TableDataSet specRegions = new TableDataSet(); + specRegions.appendColumn(specRegNames, "Name"); + specRegions.appendColumn(specRegCodes, "modelCode"); + specRegions.appendColumn(specRegZones, "modelZone"); + specRegions.appendColumn(specRegFAFCodes, "faf3code"); + specRegions.appendColumn(specRegModes, "mode"); + float[] dummyEmployment = new float[specRegCodes.length]; + for (int i = 0; i < dummyEmployment.length; i++) dummyEmployment[i] = 1; + specRegions.appendColumn(dummyEmployment, "Employment"); + return specRegions; + } + + + public static boolean countyFlowConnectsWithHawaii (int orig, int dest) { + // check if flow connects with Hawaii county + int oState = (int) (orig / 1000f); + int dState = (int) (dest / 1000f); + return oState == 15 || dState == 15; + } + + + public static boolean arrayContainsElement (String element, String[] array) { + // Check if Array contains Element + + boolean result = false; + for (String t: array) if (t.equals(element)) result = true; + return result; + } + + + public static int[] expandArrayByOneElement (int[] existing, int addElement) { + // create new array that has length of existing.length + 1 and copy values into new array + int[] expanded = new int[existing.length + 1]; + System.arraycopy(existing, 0, expanded, 0, existing.length); + expanded[expanded.length - 1] = addElement; + return expanded; + } + + + public static float[] expandArrayByOneElement (float[] existing, float addElement) { + // create new array that has length of existing.length + 1 and copy values into new array + float[] expanded = new float[existing.length + 1]; + System.arraycopy(existing, 0, expanded, 0, existing.length); + expanded[expanded.length - 1] = addElement; + return expanded; + } + + + public static String[] expandArrayByOneElement (String[] existing, String addElement) { + // create new array that has length of existing.length + 1 and copy values into new array + String[] expanded = new String[existing.length + 1]; + System.arraycopy(existing, 0, expanded, 0, existing.length); + expanded[expanded.length - 1] = addElement; + return expanded; + } + + + public static float rounder(float value, int digits) { + // rounds value to digits behind the decimal point + return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); + } + + + public static String[] findUniqueElements (String[] list) { + // find unique elements in list[] and return string[] with these elements + + ArrayList unique = new ArrayList(); + for (String txt: list) { + if (!unique.contains(txt)) unique.add(txt); + } + String[] elements = new String[unique.size()]; + for (int i = 0; i < unique.size(); i++) elements[i] = unique.get(i); + return elements; + } + + + public static float getSum (float[] array) { + float sum = 0; + for (float val: array) sum += val; + return sum; + } + + public static double getSum (double[] array) { + double sum = 0; + for (double val: array) sum += val; + return sum; + } + + public static float getSum (float[][] array) { + float sum = 0; + for (float[] anArray : array) { + for (int j = 0; j < array[0].length; j++) sum += anArray[j]; + } + return sum; + } + + public static double getSum (double[][] array) { + double sum = 0; + for (double[] anArray : array) { + for (int j = 0; j < array[0].length; j++) sum += anArray[j]; + } + return sum; + } + + public static String[] getUniqueListOfValues (String[] list) { + // itentify unique list of value in list[] and return as shortened string list + ArrayList al = new ArrayList<>(); + for (String txt: list) { + if (!al.contains(txt) && txt != null) al.add(txt); + } + String[] shortenedList = new String[al.size()]; + for (int i = 0; i < al.size(); i++) { + shortenedList[i] = al.get(i); + } + return shortenedList; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java new file mode 100644 index 0000000..658cc10 --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java @@ -0,0 +1,547 @@ +package org.sandag.htm.processFAF; + +import org.apache.log4j.Logger; +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +import java.util.ResourceBundle; +import java.util.HashMap; + +/** + * This class reads FAF2 data and stores data in TableDataSets + * User: Rolf Moeckel + * Date: May 6, 2009 + */ + +public class readFAF2 { + + Logger logger = Logger.getLogger(readFAF2.class); + static public TableDataSet domRegionList; + static public TableDataSet rowRegionList; + static public String[] FAFzones; + static public String[] sctgCommodities; + static public String[] stccCommodities; + static private int highestDomRegion; + static public HashMap SCTGtoSTCCconversion; + static private HashMap sctgCode; + + private boolean referenceListsAreRead = false; + private TableDataSet domesticTonCommodityFlows; + private TableDataSet intBorderTonCommodityFlows; + private TableDataSet intSeaTonCommodityFlows; + private TableDataSet intAirTonCommodityFlows; + private TableDataSet domesticDollarCommodityFlows; + private TableDataSet intBorderDollarCommodityFlows; + private TableDataSet intSeaDollarCommodityFlows; + private TableDataSet intAirDollarCommodityFlows; + private int factor; + + + public void readCommodityList(ResourceBundle appRb) { + // read commodity names + TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.sctg.commodity.list")); + sctgCommodities = new String[sctgComList.getRowCount()]; + for (int i = 1; i <= sctgComList.getRowCount(); i++) sctgCommodities[i-1] = sctgComList.getStringValueAt(i, "SCTG"); + } + + + public void readAllFAF2dataSets(ResourceBundle appRb, String unit) { + // read all FAF2 data into TableDataSets in unit (= tons or dollars) + if (unit.equals("tons")) { + domesticTonCommodityFlows = readDomesticCommodityFlows(appRb, unit); + intBorderTonCommodityFlows = readInternationalCommodityFlowsThroughLandBorder(appRb, unit); + intSeaTonCommodityFlows = readInternationalCommodityFlowsBySea(appRb, unit); + intAirTonCommodityFlows = readInternationalCommodityFlowsByAir(appRb, unit); + factor = 1000; // tons are in 1,000s + } else if (unit.equals("dollars")) { + domesticDollarCommodityFlows = readDomesticCommodityFlows(appRb, unit); + intBorderDollarCommodityFlows = readInternationalCommodityFlowsThroughLandBorder(appRb, unit); + intSeaDollarCommodityFlows = readInternationalCommodityFlowsBySea(appRb, unit); + intAirDollarCommodityFlows = readInternationalCommodityFlowsByAir(appRb, unit); + factor = 1000000; // dollars are in 1,000,000s + } else { + logger.error("Wrong token " + unit + " in method readAllFAF2dataSets. Use tons or dollars."); + } + } + + + public void adjustTruckFAF2data(ResourceBundle appRb, String mode, int[] years) { + // adjust FAF2 growth to exogenous adjustment + String fileName = appRb.getString("adjustment.of.faf." + mode); + logger.info("Adjusting FAF2 " + mode + " forecast to growth rate set in " + fileName); + TableDataSet adjust = fafUtils.importTable(fileName); + adjust.buildIndex(adjust.getColumnPosition("Year")); + for (int yr: years) { + float fafForecast = adjust.getIndexedValueAt(yr, "FAF2forecast"); + float adjForecast = adjust.getIndexedValueAt(yr, "AdjustedNumber"); + if (fafForecast == adjForecast) continue; + for (int row = 1; row <= domesticTonCommodityFlows.getRowCount(); row++) { + if (!domesticTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; + float value = domesticTonCommodityFlows.getValueAt(row, Integer.toString(yr)); + value = value * adjForecast / fafForecast; + domesticTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); + } + for (int row = 1; row <= intBorderTonCommodityFlows.getRowCount(); row++) { + if (!intBorderTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; + float value = intBorderTonCommodityFlows.getValueAt(row, Integer.toString(yr)); + value = value * adjForecast / fafForecast; + intBorderTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); + } + for (int row = 1; row <= intSeaTonCommodityFlows.getRowCount(); row++) { + if (!intSeaTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; + float value = intSeaTonCommodityFlows.getValueAt(row, Integer.toString(yr)); + value = value * adjForecast / fafForecast; + intSeaTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); + } + for (int row = 1; row <= intAirTonCommodityFlows.getRowCount(); row++) { + if (!intAirTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase("Air & " + mode)) continue; + float value = intAirTonCommodityFlows.getValueAt(row, Integer.toString(yr)); + value = value * adjForecast / fafForecast; + intAirTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); + } + } + } + + + public TableDataSet readDomesticCommodityFlows(ResourceBundle appRb, String unit) { + // read domestic FAF2 flows in unit (tons or dollars) + logger.info ("Reading domestic FAF2 data in " + unit); + + String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.domestic." + unit)); + TableDataSet flows = fafUtils.importTable(fileName); + if (!referenceListsAreRead) { + readFAF2ReferenceLists(appRb); + referenceListsAreRead = true; + } + int[] originCodes = new int[flows.getRowCount()]; + int[] destinationCodes = new int[flows.getRowCount()]; + for (int i = 1; i <= flows.getRowCount(); i++) { + //find Origin and Destination codes + originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); + destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); + } + flows.appendColumn(originCodes, "OriginCode"); + flows.appendColumn(destinationCodes, "DestinationCode"); + return flows; + } + + + public static void readFAF2ReferenceLists(ResourceBundle appRb) { + // read reference lists for zones and commodities + domRegionList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.region.list")); + rowRegionList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.row.region.list")); + + highestDomRegion = 0; + for (int k = 1; k <= domRegionList.getRowCount(); k++) + highestDomRegion = Math.max((int) domRegionList.getValueAt(k, "RegionCode"), highestDomRegion); + int highestRegionOverAll = highestDomRegion; + for (int k = 1; k <= rowRegionList.getRowCount(); k++) + highestRegionOverAll = Math.max((int) rowRegionList.getValueAt(k, "ROWCode"), highestRegionOverAll); + FAFzones = new String[highestRegionOverAll + 1]; + for (int k = 1; k <= domRegionList.getRowCount(); k++) + FAFzones[(int) domRegionList.getValueAt(k, "RegionCode")] = domRegionList.getStringValueAt(k, "RegionName"); + for (int k = 1; k <= rowRegionList.getRowCount(); k++) + FAFzones[(int) rowRegionList.getValueAt(k, "ROWCode")] = rowRegionList.getStringValueAt(k, "ROWRegionName"); + + domRegionList.buildIndex(1); + rowRegionList.buildIndex(1); + + TableDataSet sctgNumber = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.commodity.reference")); + sctgCode = new HashMap(); + for (int i = 1; i <= sctgNumber.getRowCount(); i++) + sctgCode.put(sctgNumber.getStringValueAt(i, "FlowTableCategories"), (int) sctgNumber.getValueAt(i, "SCTG")); + + } + + + private int findZoneCode(String strZone) { + // find code of zone with name strZone + int zoneCode = -1; + // Domestic FAF zones + for (int k = 1; k <= domRegionList.getRowCount(); k++) + if (domRegionList.getStringValueAt(k, "RegionName").equals(strZone)) + zoneCode = (int) domRegionList.getValueAt(k, "RegionCode"); + // International FAF zones + if (zoneCode == -1) { + for (int k = 1; k <= rowRegionList.getRowCount(); k++) + if (rowRegionList.getStringValueAt(k, "ROWRegionName").equals(strZone)) + zoneCode = (int) rowRegionList.getValueAt(k, "ROWCode"); + } + if (zoneCode == -1) { + logger.error ("Unknown Zone in FAF2 data: " + strZone); + System.exit(1); + } + return zoneCode; + } + + + public HashMap getDomesticFlows(String mode, commodityClassType comClass, int yr, String unit) { + // extract domestic FAF2 flows + TableDataSet flowTbl; + if (unit.equals("tons")) { + flowTbl = domesticTonCommodityFlows; + } else { + flowTbl = domesticDollarCommodityFlows; + } + + HashMap hshFlows = new HashMap(); + for (int i = 1; i <= flowTbl.getRowCount(); i++) { + String thisMode = flowTbl.getStringValueAt(i, "Mode"); + if (!mode.equals("all") && !thisMode.equalsIgnoreCase(mode)) continue; + int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); + int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); + float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; + String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); + int intSctgCommmodity = sctgCode.get(fafDataCommodity); + String sctgCommodity; + if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; + else sctgCommodity = "SCTG" + intSctgCommmodity; + // report flows by STCC commodity classification + if (comClass.equals(commodityClassType.STCC)) { + for (String com: stccCommodities) { + float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; + String code; + if (flowShare > 0) { + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + com + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + com; + } + + if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); + hshFlows.put(code, flowShare); + } + } + } else { + // report flows by SCTG commodity classification + String code; + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + sctgCommodity; + } + if (hshFlows.containsKey(code)) flows += hshFlows.get(code); + hshFlows.put(code, flows); + } + } + return hshFlows; + } + + + public TableDataSet readInternationalCommodityFlowsThroughLandBorder(ResourceBundle appRb, String unit) { + // read international FAF2 flows that cross the U.S. border by land in unit (tons or dollars) + logger.info ("Reading international FAF2 data crossing borders by land in " + unit); + + String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.border." + unit)); + TableDataSet flows = fafUtils.importTable(fileName); + if (!referenceListsAreRead) { + readFAF2ReferenceLists(appRb); + referenceListsAreRead = true; + } + int[] originCodes = new int[flows.getRowCount()]; + int[] portOfEntryCodes = new int[flows.getRowCount()]; + int[] destinationCodes = new int[flows.getRowCount()]; + for (int i = 1; i <= flows.getRowCount(); i++) { + //find Origin and Destination codes + originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); + portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "PortOfEntryExit")); + destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); + } + flows.appendColumn(originCodes, "OriginCode"); + flows.appendColumn(portOfEntryCodes, "PortOfEntryCode"); + flows.appendColumn(destinationCodes, "DestinationCode"); + return flows; + } + + + public HashMap getIntBorderFlows(reportFormat repform, String mode, commodityClassType comClass, + int yr, String unit) { + // extract international FAF2 flows (international mode truck or train) + + TableDataSet flowTbl; + if (unit.equals("tons")) { + flowTbl = intBorderTonCommodityFlows; + } else { + flowTbl = intBorderDollarCommodityFlows; + } + + HashMap hshFlows = new HashMap(); + for (int i = 1; i <= flowTbl.getRowCount(); i++) { + String thisMode = flowTbl.getStringValueAt(i, "Mode"); + if (!mode.equals("all") && !thisMode.equals(mode)) continue; + int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); + int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); + String strOrig = flowTbl.getStringValueAt(i, "Origin"); + String strDest = flowTbl.getStringValueAt(i, "Destination"); + int borderCode = (int) flowTbl.getValueAt(i, "PortOfEntryCode"); + // if flow goes from WA Blain to Canada with port of exit WA Blain, don't report domestic part. - Changed my mind, do report this flow +// if (repform == reportFormat.internat_domesticPart && (destCode == borderCode || origCode == borderCode)) continue; + if (repform == reportFormat.internat_domesticPart) { + boolean isCanadaOrMexico = checkIfMexicoOrCanada(strOrig); + if (isCanadaOrMexico) origCode = borderCode; + isCanadaOrMexico = checkIfMexicoOrCanada(strDest); + if (isCanadaOrMexico) destCode = borderCode; + } else if (repform == reportFormat.internat_internationalPart) { + boolean isCanadaOrMexico = checkIfMexicoOrCanada(strOrig); + if (isCanadaOrMexico) destCode = borderCode; + isCanadaOrMexico = checkIfMexicoOrCanada(strDest); + if (isCanadaOrMexico) origCode = borderCode; + } + // note that border flows are done differently than sea and air flows. Oregon has port and airports, that's + // why those flows are extracted separately. Oregon has no international border, therefore border flows are + // added to the US FAF region where the goods cross the border (unless reportFormat.internatOrigToDest has + // been chosen). + float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; + String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); + int intSctgCommmodity = sctgCode.get(fafDataCommodity); + String sctgCommodity; + if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; + else sctgCommodity = "SCTG" + intSctgCommmodity; + if (comClass.equals(commodityClassType.STCC)) { + // report by STCC commodity classification + for (String com: stccCommodities) { + float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; + if (flowShare > 0) { + String code; + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + sctgCommodity; + } + if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); + hshFlows.put(code, flowShare); + } + } + } else { + // report by SCTG commodity classification + String code; + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + sctgCommodity; + } + if (hshFlows.containsKey(code)) flows += hshFlows.get(code); + hshFlows.put(code, flows); + } + } + return hshFlows; + } + + + public TableDataSet readInternationalCommodityFlowsBySea(ResourceBundle appRb, String unit) { + // read international FAF2 flows that cross the U.S. border by sea in unit (tons or dollars) + logger.info ("Reading international FAF2 data by sea in " + unit); + + String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.sea." + unit)); + TableDataSet flows = fafUtils.importTable(fileName); + if (!referenceListsAreRead) { + readFAF2ReferenceLists(appRb); + referenceListsAreRead = true; + } + int[] originCodes = new int[flows.getRowCount()]; + int[] portOfEntryCodes = new int[flows.getRowCount()]; + int[] destinationCodes = new int[flows.getRowCount()]; + for (int i = 1; i <= flows.getRowCount(); i++) { + //find Origin and Destination codes + originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); + portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Port")); + destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); + } + flows.appendColumn(originCodes, "OriginCode"); + flows.appendColumn(portOfEntryCodes, "PortCode"); + flows.appendColumn(destinationCodes, "DestinationCode"); + return flows; + } + + + public HashMap getIntSeaFlows (reportFormat repform, String mode, commodityClassType comClass, + int yr, String unit) { + // extract international FAF2 flows (international mode sea) + // If repform is set to internat_domesticPart port entry and exit points are set as negative origin or destination codes + + TableDataSet flowTbl; + if (unit.equals("tons")) { + flowTbl = intSeaTonCommodityFlows; + } else { + flowTbl = intSeaDollarCommodityFlows; + } + + HashMap hshFlows = new HashMap(); + for (int i = 1; i <= flowTbl.getRowCount(); i++) { + String thisMode = flowTbl.getStringValueAt(i, "Mode"); + if (!mode.equals("all") && !thisMode.equalsIgnoreCase(mode) && + repform != reportFormat.internat_internationalPart) continue; + if (!mode.equals("all") && !mode.equalsIgnoreCase("Water") && + repform == reportFormat.internat_internationalPart) continue; + int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); + int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); + int portCode = -1 * (int) flowTbl.getValueAt(i, "PortCode"); + // if flow goes from Savannah GA to Europe with port of exit Savannah GA, don't report domestic part - changed my mind: do report this flow +// if (repform == reportFormat.internat_domesticPart && (destCode == -portCode || origCode == -portCode)) continue; + if (repform == reportFormat.internat_domesticPart) { + if (origCode > highestDomRegion && destCode <= highestDomRegion) + // from abroad to U.S. -> set origin to entry port + origCode = portCode; + else if (origCode <= highestDomRegion && destCode > highestDomRegion) + // from U.S. to abroad -> set destination to exit port + destCode = portCode; + else if (origCode > highestDomRegion && destCode > highestDomRegion) { + // from abroad through U.S. port to abroad + boolean origMexCan = checkIfMexicoOrCanada(flowTbl.getStringValueAt(i, "Origin")); + // from Mexico or Canada through U.S. port to Overseas -> set destination to exit port + if (origMexCan) destCode = portCode; + // from Overseas through U.S. port to Mexico or Canada -> set origin to entry port + else origCode = portCode; + // Note: if both origin and destination are MEX/CAN or both origin and destination are overseas, + // it is impossible to determine which part of the trips was made by mode mode, therefore it is + // reported as from abroad to abroad without noting the entry/exit port [seems not to exist in FAF2 data] + } + } else if (repform == reportFormat.internat_internationalPart) { + if (origCode > highestDomRegion && destCode <= highestDomRegion) + // from abroad to U.S. -> set destination to entry port + destCode = portCode; + else if (origCode <= highestDomRegion && destCode > highestDomRegion) + // from U.S. to abroad -> set origin to exit port + origCode = portCode; + else if (origCode > highestDomRegion && destCode > highestDomRegion) { + // from abroad through U.S. port to abroad + boolean origMexCan = checkIfMexicoOrCanada(flowTbl.getStringValueAt(i, "Origin")); + // from Mexico or Canada through U.S. port to Overseas -> set origin to exit port + if (origMexCan) origCode = portCode; + // from Overseas through U.S. port to Mexico or Canada -> set destination to entry port + else destCode = portCode; + // Note: if both origin and destination are MEX/CAN or both origin and destination are overseas, + // it is impossible to determine which part of the trips was made by mode mode, therefore it is + // reported as from abroad to abroad without noting the entry/exit port [seems not to exist in FAF2 data] + } + } + float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; + String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); + int intSctgCommmodity = sctgCode.get(fafDataCommodity); + String sctgCommodity; + if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; + else sctgCommodity = "SCTG" + intSctgCommmodity; + if (comClass.equals(commodityClassType.STCC)) { + for (String com: stccCommodities) { + float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; + if (flowShare > 0) { + String code; + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + sctgCommodity; + } + if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); + hshFlows.put(code, flowShare); + } + } + } else { + // report by SCTG commodity classification + String code; + if (mode.equals("all")) { + code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; + } else { + code = origCode + "_" + destCode + "_" + sctgCommodity; + } + if (hshFlows.containsKey(code)) flows += hshFlows.get(code); + hshFlows.put(code, flows); + } + } + return hshFlows; + } + + + private boolean checkIfMexicoOrCanada (String name) { + // check if name is Canada or Mexico and return true or false + boolean contains = false; + if (name.equals("Canada") || name.equals("Mexico")) contains = true; + return contains; + } + + + public TableDataSet readInternationalCommodityFlowsByAir(ResourceBundle appRb, String unit) { + // read international FAF2 flows that cross the U.S. border by air in unit (tons or dollars) + logger.info ("Reading international FAF2 data by air in " + unit); + + String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.air." + unit)); + TableDataSet flows = fafUtils.importTable(fileName); + if (!referenceListsAreRead) { + readFAF2ReferenceLists(appRb); + referenceListsAreRead = true; + } + int[] originCodes = new int[flows.getRowCount()]; + int[] portOfEntryCodes = new int[flows.getRowCount()]; + int[] destinationCodes = new int[flows.getRowCount()]; + for (int i = 1; i <= flows.getRowCount(); i++) { + //find Origin and Destination codes + originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); + portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Coast")); + destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); + } + flows.appendColumn(originCodes, "OriginCode"); + flows.appendColumn(portOfEntryCodes, "PortCode"); + flows.appendColumn(destinationCodes, "DestinationCode"); + return flows; + } + + + public HashMap getIntAirFlows(reportFormat repform, String mode, commodityClassType comClass, + int yr, String unit) { + // extract international FAF2 flows (international mode air) + // If repform is set to internat_domesticPart port entry and exit points are set as negative origin or destination codes + + TableDataSet flowTbl; + if (unit.equals("tons")) { + flowTbl = intAirTonCommodityFlows; + } else { + flowTbl = intAirDollarCommodityFlows; + } + + HashMap hshFlows = new HashMap(); + for (int i = 1; i <= flowTbl.getRowCount(); i++) { + // Note: Only mode "Air & Truck" is available in this data set + if (repform == reportFormat.internat_domesticPart && + !flowTbl.getStringValueAt(i, "Mode").equals(mode)) continue; + if (repform == reportFormat.internat_internationalPart && + !flowTbl.getStringValueAt(i, "Mode").equals("Air & Truck")) continue; + int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); + int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); + int portCode = -1 * (int) flowTbl.getValueAt(i, "PortCode"); + // if flow goes from Houston TX to Europe with port of exit Houston TX, don't report domestic part - changed my mind: do report this flow +// if (repform == reportFormat.internat_domesticPart && (destCode == -portCode || origCode == -portCode)) continue; + if (origCode <= highestDomRegion || destCode <= highestDomRegion) { + // this should be the case for every record, there should be no international to international records + if (repform == reportFormat.internat_domesticPart) { + if (origCode > highestDomRegion) origCode = portCode; + if (destCode > highestDomRegion) destCode = portCode; + } else if (repform == reportFormat.internat_internationalPart) { + if (origCode > highestDomRegion) destCode = portCode; + if (destCode > highestDomRegion) origCode = portCode; + } + } + float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; + String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); + int intSctgCommmodity = sctgCode.get(fafDataCommodity); + String sctgCommodity; + if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; + else sctgCommodity = "SCTG" + intSctgCommmodity; + if (comClass.equals(commodityClassType.STCC)) { + for (String com: stccCommodities) { + float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; + if (flowShare > 0) { + String code = origCode + "_" + destCode + "_" + com; + if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); + hshFlows.put(code, flowShare); + } + } + } else { + // report by SCTG commodity classification + String code = origCode + "_" + destCode + "_" + sctgCommodity; + if (hshFlows.containsKey(code)) flows += hshFlows.get(code); + hshFlows.put(code, flows); + } + + } + return hshFlows; + } +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java new file mode 100644 index 0000000..615d08b --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java @@ -0,0 +1,520 @@ +package org.sandag.htm.processFAF; + +import org.apache.log4j.Logger; + +import java.util.ResourceBundle; +import java.util.HashMap; +import java.io.PrintWriter; + +import com.pb.common.datafile.TableDataSet; +import com.pb.common.util.ResourceUtil; + +/** + * This class reads FAF3 data and stores data in a TableDataSet + * Author: Rolf Moeckel, PB + * Date: September 9, 2010 + */ + +public class readFAF3 { + Logger logger = Logger.getLogger(readFAF3.class); + private int factor; + private String[] valueColumnName; + private TableDataSet faf3commodityFlows; + public static TableDataSet fafRegionList; + private String[] regionState; + private static int[] domRegionIndex; + static public int[] sctgCommodities; + static public String[] sctgStringCommodities; + static private int[] sctgStringIndex; + private static HashMap portsOfEntry; + private static HashMap marinePortsOfEntry; + private static HashMap railPortsOfEntry; + private static HashMap airPortsOfEntry; + private static int[] listOfBorderPortOfEntries; + + + public void readAllData (ResourceBundle appRb, int year, String unit) { + // read input data + + if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) + readAllFAF3dataSets(appRb, unit, year); + readCommodityList(appRb); + readFAF3referenceLists(appRb); + } + + + public void readCommodityList(ResourceBundle appRb) { + // read commodity names + TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf3.sctg.commodity.list")); + sctgCommodities = new int[sctgComList.getRowCount()]; + sctgStringCommodities = new String[sctgCommodities.length]; + for (int i = 1; i <= sctgComList.getRowCount(); i++) { + sctgCommodities[i-1] = (int) sctgComList.getValueAt(i, "SCTG"); + if (sctgCommodities[i-1] < 10) sctgStringCommodities[i-1] = "SCTG0" + sctgCommodities[i-1]; + else sctgStringCommodities[i-1] = "SCTG" + sctgCommodities[i-1]; + } + sctgStringIndex = new int[fafUtils.getHighestVal(sctgCommodities) + 1]; + for (int num = 0; num < sctgCommodities.length; num++) sctgStringIndex[sctgCommodities[num]] = num; + } + + + public int getIndexOfCommodity (int commodity) { + return sctgStringIndex[commodity]; + } + + + public static String getSCTGname(int sctgInt) { + // get String name from sctg number + return sctgStringCommodities[sctgStringIndex[sctgInt]]; + } + + + public static String getFAFzoneName(int fafInt) { + // get String name from int FAF zone code number + return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "FAF3 Zones -Short Description"); + } + + + public static String getFAFzoneState(int fafInt) { + // get String two-letter abbreviation of state of fafInt + return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "State"); + } + + + public void definePortsOfEntry(ResourceBundle appRb) { + // read data to translate ports of entry in network links + + // Border crossings + portsOfEntry = new HashMap<>(); + TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); + for (int row = 1; row <= poe.getRowCount(); row++) { + int fafID = (int) poe.getValueAt(row, "faf3id"); + int node = (int) poe.getValueAt(row, "pointOfEntry"); + float weight = poe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (portsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = portsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + portsOfEntry.put(fafID, newPortsOfEntry); + } + listOfBorderPortOfEntries = poe.getColumnAsInt("pointOfEntry"); + + // Marine ports + marinePortsOfEntry = new HashMap<>(); + if (appRb.containsKey("marine.ports.of.entry")) { + TableDataSet mpoe = fafUtils.importTable(appRb.getString("marine.ports.of.entry")); + for (int row = 1; row <= mpoe.getRowCount(); row++) { + int fafID = (int) mpoe.getValueAt(row, "faf3id"); + int node = (int) mpoe.getValueAt(row, "pointOfEntry"); + float weight = mpoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (marinePortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = marinePortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + marinePortsOfEntry.put(fafID, newPortsOfEntry); + } + } + // Rail ports (railyards) + railPortsOfEntry = new HashMap<>(); + if (appRb.containsKey("rail.ports.of.entry")) { + TableDataSet rpoe = fafUtils.importTable(appRb.getString("rail.ports.of.entry")); + for (int row = 1; row <= rpoe.getRowCount(); row++) { + int fafID = (int) rpoe.getValueAt(row, "faf3id"); + int node = (int) rpoe.getValueAt(row, "pointOfEntry"); + float weight = rpoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (railPortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = railPortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + railPortsOfEntry.put(fafID, newPortsOfEntry); + } + } + // Airports + airPortsOfEntry = new HashMap<>(); + if (appRb.containsKey("air.ports.of.entry")) { + TableDataSet apoe = fafUtils.importTable(appRb.getString("air.ports.of.entry")); + for (int row = 1; row <= apoe.getRowCount(); row++) { + int fafID = (int) apoe.getValueAt(row, "faf3id"); + int node = (int) apoe.getValueAt(row, "pointOfEntry"); + float weight = apoe.getValueAt(row, "weight"); + TableDataSet newPortsOfEntry = new TableDataSet(); + if (airPortsOfEntry.containsKey(fafID)) { + TableDataSet existingNodes = airPortsOfEntry.get(fafID); + int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for + float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation + int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); + float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); + newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(newWeights, "Employment"); + } else { + newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); + newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); + } + airPortsOfEntry.put(fafID, newPortsOfEntry); + } + } + } + + + public static int[] getListOfBorderPortOfEntries() { + return listOfBorderPortOfEntries; + } + + + + public static TableDataSet getPortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (portsOfEntry.containsKey(fafZone)) return portsOfEntry.get(fafZone); + else return null; + } + + + public static TableDataSet getMarinePortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (marinePortsOfEntry.containsKey(fafZone)) return marinePortsOfEntry.get(fafZone); + else return null; + } + + + public static TableDataSet getAirPortsOfEntry (int fafZone) { + // return list of ports of entry if available, otherwise return fafZone + if (airPortsOfEntry.containsKey(fafZone)) return airPortsOfEntry.get(fafZone); + else return null; + } + + + public void readAllFAF3dataSets2007(ResourceBundle appRb, String unit) { + // read all FAF3 data into TableDataSets in unit (= tons or dollars) + + logger.info ("Reading domestic FAF3 data in " + unit); + if (unit.equals("tons")) { + factor = 1000; // tons are in 1,000s + valueColumnName = new String[]{"tons_2007"}; + } else if (unit.equals("dollars")) { + factor = 1000000; // dollars are in 1,000,000s + valueColumnName = new String[]{"value_2007"}; + } else { + logger.fatal ("Wrong token " + unit + " in method readAllFAF3dataSets2007. Use tons or dollars."); + } + faf3commodityFlows = readFAF3commodityFlows(appRb, unit); + } + + + public double[] summarizeFlowByCommodity (ModesFAF fafMode) { + // sum commodity flows by commodity and return array with total tons + + double[] totalFlows = new double[sctgCommodities.length]; + int modeNum = fafUtils.getEnumOrderNumber(fafMode); + for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { + if (faf3commodityFlows.getValueAt(row, "dms_mode") != modeNum) continue; + int com = sctgStringIndex[(int) faf3commodityFlows.getValueAt(row, "sctg2")]; + if (valueColumnName.length == 1) { + // use year provided by user + totalFlows[com] += faf3commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); + totalFlows[com] += val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + } + return totalFlows; + } + + + public void readAllFAF3dataSets(ResourceBundle appRb, String unit, int year) { + // read all FAF3 data into TableDataSets in unit (= tons or dollars) + + logger.info (" Reading FAF3 data in " + unit); + switch (unit) { + case "tons": + factor = 1000; // tons are provided in 1,000s + break; + case "dollars": + factor = 1000000; // dollars are provided in 1,000,000s + break; + default: + logger.fatal("Wrong token " + unit + " in method readAllFAF3dataSets. Use tons or dollars."); + System.exit(0); + } + int[] availYears = {2007, 2015, 2020, 2025, 2030, 2035, 2040}; + boolean yearInFaf = false; + for (int y: availYears) if (year == y) yearInFaf = true; + if (!yearInFaf) { // interpolate between two years + logger.info(" Year " + year + " does not exist in FAF3 data."); + int year1 = 2007; + int year2 = 2040; + for (int availYear : availYears) if (availYear < year) year1 = availYear; + for (int i = availYears.length - 1; i >= 0; i--) if (availYears[i] > year) year2 = availYears[i]; + logger.info(" FAF3 data are interpolated between " + year1 + " and " + year2 + "."); + // first position: lower year, second position: higher year, third position: steps away from lower year + valueColumnName = new String[]{unit + "_" + year1, unit + "_" + year2, String.valueOf((1f * (year - year1)) / (1f * (year2 - year1)))}; + } else { // use year provided by user + valueColumnName = new String[]{unit + "_" + year}; + } + faf3commodityFlows = readFAF3commodityFlows(appRb, unit); + } + + + private TableDataSet readFAF3commodityFlows(ResourceBundle appRb, String unit) { + // read FAF3 data and return TableDataSet with flow data + String fileName = ResourceUtil.getProperty(appRb, ("faf3.data")); + return fafUtils.importTable(fileName); + } + + + public HashMap createScaler(String[] tokens, double[] values) { + // create HashMap with state O-D pairs that need to be scaled + + HashMap scaler = new HashMap(); + if (tokens.length != values.length) { + throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); + } + for (int i=0; i scaler) { + // extract truck flows for year yr and scale flows according to scaler HashMap (no special regions specified) + + PrintWriter outFile = fafUtils.openFileForSequentialWriting(outFileName); + outFile.println("originFAF,destinationFAF,flowDirection," + commodityClassType.SCTG + "_commodity,shortTons"); + int modeNum = fafUtils.getEnumOrderNumber(mode); + for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { + int type = (int) faf3commodityFlows.getValueAt(row, "trade_type"); + double val; + if (valueColumnName.length == 1) { + // use year provided by user + val = faf3commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); + val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + val *= factor * odScaler(row, type, scaler); + if (val == 0) continue; + if (type == 1) writeDomesticFlow(modeNum, val, row, outFile); + else if (type == 2) writeImportFlow(modeNum, val, row, outFile, repF); + else if (type == 3) writeExportFlow(modeNum, val, row, outFile, repF); + else if (type == 4) writeThroughFlow(modeNum, val, row, outFile, repF); + else logger.info("Invalid trade_type in FAF3 dataset in row " + row + ": " + type); + } + outFile.close(); + } + + + public void writeFlowsByModeAndCommodity (String outFileName, ModesFAF mode, reportFormat repF, + HashMap scaler) { + // extract truck flows for year yr and scale flows according to scaler HashMap, including special regions + + PrintWriter outFile[] = new PrintWriter[sctgCommodities.length]; + for (int com: sctgCommodities) { + String fileName; + if (com < 10) fileName = outFileName + "_SCTG0" + com + ".csv"; + else fileName = outFileName + "_SCTG" + com + ".csv"; + outFile[sctgStringIndex[com]] = fafUtils.openFileForSequentialWriting(fileName); + outFile[sctgStringIndex[com]].println("originFAF,destinationFAF,flowDirection,SCTG_commodity,shortTons"); + } + int modeNum = fafUtils.getEnumOrderNumber(mode); + for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { + int type = (int) faf3commodityFlows.getValueAt(row, "trade_type"); + double val; + if (valueColumnName.length == 1) { + // use year provided by user + val = faf3commodityFlows.getValueAt(row, valueColumnName[0]); + } else { + // interpolate between two years + float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); + float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); + val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); + } + val *= factor * odScaler(row, type, scaler); + if (val == 0) continue; + int comIndex = getIndexOfCommodity((int) faf3commodityFlows.getValueAt(row, "sctg2")); + if (type == 1) writeDomesticFlow(modeNum, val, row, outFile[comIndex]); + else if (type == 2) writeImportFlow(modeNum, val, row, outFile[comIndex], repF); + else if (type == 3) writeExportFlow(modeNum, val, row, outFile[comIndex], repF); + else if (type == 4) writeThroughFlow(modeNum, val, row, outFile[comIndex], repF); + else logger.info("Invalid trade_type in FAF3 dataset in row " + row + ": " + type); + } + for (int com: sctgCommodities) outFile[sctgStringIndex[com]].close(); + } + + + public float odScaler (int row, int type, HashMap scaler) { + // find scaler for origin destination pair in row + + int orig; + int dest; + if (type == 1) { + orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + } else if (type == 2) { + orig = (int) faf3commodityFlows.getValueAt(row, "fr_orig"); + dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + } else if (type == 3) { + orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + dest = (int) faf3commodityFlows.getValueAt(row, "fr_dest"); + } else { + orig = (int) faf3commodityFlows.getValueAt(row, "fr_orig"); + dest = (int) faf3commodityFlows.getValueAt(row, "fr_dest"); + } + String stateLevelToken = regionState[orig] + "_" + regionState[dest]; + String combo1Token = orig + "_" + regionState[dest]; + String combo2Token = regionState[orig] + "_" + dest; + String fafLevelToken = orig + "_" + dest; + float adj = 1; + if (scaler.containsKey(stateLevelToken)) adj = scaler.get(stateLevelToken); + if (scaler.containsKey(combo1Token)) adj = scaler.get(combo1Token); + if (scaler.containsKey(combo2Token)) adj = scaler.get(combo2Token); + if (scaler.containsKey(fafLevelToken)) adj = scaler.get(fafLevelToken); + return adj; + } + + + private int tryGettingThisValue(int row, String token) { + // for some flows, international zones/modes are empty -> catch this case and set zone to 0 + + int region; + try { + region = (int) faf3commodityFlows.getValueAt(row, token); + } catch (Exception e) { + region = 0; + } + return region; + } + + + public void writeDomesticFlow (int modeNum, double val, int row, PrintWriter outFile) { + // internal US flow + if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + int dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); + outFile.println(orig + "," + dest + ",domestic," + comm + "," + val); + } + } + + + public void writeImportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // from abroad to US + + int frInMode = (int) faf3commodityFlows.getValueAt(row, "fr_inmode"); + int borderZone = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); + if (frInMode == modeNum && repF != reportFormat.internat_domesticPart) { + int orig = tryGettingThisValue(row, "fr_orig"); + outFile.println(orig + "," + borderZone + ",import," + comm + "," + val); + } + if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + String txt; + if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",import_port,"; + else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",import_rail,"; + else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",import_airport,"; + else txt = ",import,"; + outFile.println(borderZone + "," + dest + txt + comm + "," + val); + } + } + + + public void writeExportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // from US to abroad + int frOutMode = tryGettingThisValue(row, "fr_outmode"); + int borderZone = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); + if (frOutMode == modeNum && repF != reportFormat.internat_domesticPart) { + int dest = tryGettingThisValue(row, "fr_dest"); + outFile.println(borderZone + "," + dest + ",export," + comm + "," + val); + } + if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { + int orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + String txt = ",export,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",export_port,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",export_rail,"; + if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",export_airport,"; + outFile.println(orig + "," + borderZone + txt + comm + "," + val); + } + } + + + public void writeThroughFlow(int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { + // flows in transit through US + if ((int) faf3commodityFlows.getValueAt(row, "dms_mode") != modeNum) return; + int borderInZone = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); + int borderOutZone = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); + logger.warn("Through flows not yet implemented. This flow from " + borderInZone + " to " + borderOutZone + " is lost."); + } + + + public void readFAF3referenceLists(ResourceBundle rb) { + // read list of regions for FAF3 + String regFileName = rb.getString("faf3.region.list"); + fafRegionList = fafUtils.importTable(regFileName); + int[] reg = fafRegionList.getColumnAsInt(fafRegionList.getColumnPosition("ZoneID")); + domRegionIndex = new int[fafUtils.getHighestVal(reg) + 1]; + for (int num = 0; num < reg.length; num++) domRegionIndex[reg[num]] = num + 1; + regionState = new String[fafUtils.getHighestVal(reg) + 1]; + for (int row = 1; row <= fafRegionList.getRowCount(); row++) { + int zone = (int) fafRegionList.getValueAt(row, "ZoneID"); + regionState[zone] = fafRegionList.getStringValueAt(row, "State"); + } + } + + + public int[] getFAFzoneIDs () { + return fafRegionList.getColumnAsInt("ZoneID"); + } + + + public TableDataSet getFAF3flows() { + return faf3commodityFlows; + } + + + public int getFactor() { + return factor; + } + + + public TableDataSet getFaf3commodityFlows() { + return faf3commodityFlows; + } + + + public String[] getValueColumnName() { + return valueColumnName; + } + +} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java new file mode 100644 index 0000000..a33927a --- /dev/null +++ b/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java @@ -0,0 +1,23 @@ +package org.sandag.htm.processFAF; + +/** + * Defines reporting method + * internationalByEntryPoint reports international flows + * internat_domesticPart = International flow from port of entry to final domestic destination or from + * domestic origin to port of exit + * internat_internationalPart = International flow from foreign origin to port of entry or from port of exit to + * foreign destination + * internatOrigToDest = International flow from foreign origin to domestic destination or from domestic + * origin to foreign destination + * internatOrigToBorderToDest = International flow from origin to port of entry/port of exit to destination (2 flows) + * + * User: Rolf Moeckel, PB New York + * Date: May 7, 2009 + */ + +public enum reportFormat { + internat_domesticPart, + internat_internationalPart, + internatOrigToDest, + internatOrigToBorderToDest, +} diff --git a/sandag_abm/src/main/python/assignScenarioID.py b/sandag_abm/src/main/python/assignScenarioID.py new file mode 100644 index 0000000..0571fe4 --- /dev/null +++ b/sandag_abm/src/main/python/assignScenarioID.py @@ -0,0 +1,27 @@ +import os +import glob +import pandas as pd +import pyodbc + +toolpath = os.getcwd()[3:-7] + +conn = pyodbc.connect("DRIVER={SQL Server};" + "SERVER=DDAMWSQL16;" + "DATABASE=abm_14_2_0;" + "Trusted_Connection=yes;") +sql = ("SELECT * FROM [abm_14_2_0].[dimension].[scenario] where RIGHT(path, len(path)-23) = '%s'" % toolpath) +df_sql = pd.read_sql_query(sql, conn) +scenid = df_sql['scenario_id'].max() +list = glob.glob(os.getcwd()[:-6]+'report\\hwyload*') +list_shape = glob.glob(os.getcwd()[:-6]+'report\\hwyload*.shp') + +if len(list_shape) and len(df_sql): + for item in list: + if 'csv' not in item: + try: + os.rename(item, os.getcwd()[:-6]+'report\\hwyLoad_'+ str(scenid) + item[-4:]) + except Exception as error: + print('Caught this error: ' + repr(error)) + print ('The scenaio ID has been added to the shapefile.') +else: + print ("Cannot find the scenario in the SQ database or hwyloadshape file is not available. Please check...") \ No newline at end of file diff --git a/sandag_abm/src/main/python/calculate_micromobility.py b/sandag_abm/src/main/python/calculate_micromobility.py new file mode 100644 index 0000000..a277359 --- /dev/null +++ b/sandag_abm/src/main/python/calculate_micromobility.py @@ -0,0 +1,273 @@ +#!/usr/bin/env python3 +# -*- coding: utf-8 -*- +# Author: RSG Inc. +"""calculate_micromobility.py + +This Python 3 script calculates Generalized Time between MGRAs for micromobility +and microtransit modes in the SANDAG activity-based model. + +Constants are read from the properties file specified at the command line. + +Currently this script uses three files to perform its calculations: + - mgra.socec.file, contains MicroAccessTime for origin MRGAs + - active.logsum.matrix.file.walk.mgra, contains the pre-calculated + walk times between MGRAs + - active.logsum.matrix.file.walk.mgratap, contains the pre-calculated + walk times from MGRAs to TAPs + - active.microtransit.tap.file, contains a list of TAPs with microtransit availability + - active.microtransit.mgra.file, contains a list of MGRAs with microtransit availability + + +The script then writes a fresh MGRA file with the newly calculated micromobility calculations: + - walkTime: the original walk time + - mmTime: the micro-mobility time, including travel time, rental time, access time + - mmCost: micro-mobility variable cost * travel time + fixed cost + - mmGenTime: mmTime + mmCost converted to time + constant + - mtTime: the micro-transit time, including travel time, wait time, access time + - mtCost: micro-transit variable cost * travel time + fixed cost + - minTime: minimum of walkTime, mmGenTime, and mtGenTime + +Run `python calculate_micromobility.py -h` for more command-line usage. +""" + +import argparse +import os +import pandas as pd + + +def process_file(config, zone): + """Performs micromobility calculations using given output_file + and attributes from the provided MGRA file and properties file + + Writes newly calculated micromobility time and intermediate calculations + """ + + filename = config.walk_mgra_output_file if zone == 'mgra' else config.walk_mgra_tap_output_file + + output_file = os.path.join(config.cli.outputs_directory, filename) + config.validate_file(output_file) + + if zone == 'mgra': + walk_time_col = 'actual' + orig_col = 'i' + dest_col = 'j' + + else: + walk_time_col = 'boardingActual' + orig_col = 'mgra' + dest_col = 'tap' + + print('Processing %s ...' % output_file) + df = pd.read_csv(output_file, usecols=[walk_time_col, orig_col, dest_col]) + df.rename(columns={walk_time_col:'walkTime'}, inplace=True) + + # OD vectors + length = df['walkTime'] / config.walk_coef + + # availability masks + if zone == 'mgra': + mt_avail = \ + (df[orig_col].isin(config.mt_mgras) & df[dest_col].isin(config.mt_mgras)) & \ + (length <= config.mt_max_dist_mgra) + + walk_avail = length <= config.walk_max_dist_mgra + mm_avail = length <= config.mm_max_dist_mgra + + else: + mt_avail = \ + df[orig_col].isin(config.mt_mgras) & df[dest_col].isin(config.mt_taps) & \ + (length <= config.mt_max_dist_tap) + walk_avail = length <= config.walk_max_dist_tap + mm_avail = length <= config.mm_max_dist_tap + + all_rows = df.shape[0] + df = df[mt_avail | walk_avail | mm_avail] + print('Filtered out %s unavailable pairs' % str(all_rows - df.shape[0])) + + # micro-mobility + mm_ivt = length * 60 / config.mm_speed # micro-mobility in-vehicle time + orig_mat = df[orig_col].map(config.mat) # micro-access time at origin + mm_time = mm_ivt + config.mm_rental_time + orig_mat # total mm time + mm_cost = config.mm_variable_cost * mm_ivt + config.mm_fixed_cost + mm_cost_as_time = mm_cost * 60 / config.vot + + # micro-transit + mt_ivt = length * 60 / config.mt_speed + mt_time = mt_ivt + 2 * config.mt_wait_time + config.mt_access_time + mt_cost = mt_time * config.mt_variable_cost + config.mt_fixed_cost + mt_cost_as_time = mt_cost * 60 / config.vot + + # save intermediate calculations + df['dist'] = length + df['mmTime'] = mm_time + df['mmCost'] = mm_cost + df['mtTime'] = mt_time + df['mtCost'] = mt_cost + + # calculate micromobility and microtransit Generalized Time + df['mmGenTime'] = mm_time + mm_cost_as_time + config.mm_constant + df['mtGenTime'] = mt_time + mt_cost_as_time + config.mt_constant + + # update zones with unavailable walk, micromobility, and microtransit + df.loc[~walk_avail, ['walkTime']] = config.mt_not_avail + df.loc[~mm_avail, ['mmTime', 'mmCost', 'mmGenTime']] = config.mt_not_avail + df.loc[~mt_avail, ['mtTime', 'mtCost', 'mtGenTime']] = config.mt_not_avail + + # calculate the minimum of walk time vs. generalized time + df['minTime'] = df[['walkTime', 'mmGenTime', 'mtGenTime']].min(axis=1) + + # write output + outfile = os.path.join( + config.cli.outputs_directory, + os.path.basename(output_file).replace('walk', 'micro') + ) + + print("Writing final table to %s" % outfile) + df.to_csv(outfile, index=False) + print("Done.") + + +class Config(): + + def __init__(self): + + self.init_cli_args() + self.init_properties() + self.init_micro_access_time() + self.init_tap_mgra_lists() + + def init_cli_args(self): + """Use argparse to set command-line args + + """ + + self.parser = argparse.ArgumentParser(formatter_class=argparse.ArgumentDefaultsHelpFormatter) + + self.parser.add_argument( + '-p', '--properties_file', + default=os.path.join('..', 'conf', 'sandag_abm.properties'), + help="Java properties file.") + + self.parser.add_argument( + '-o', '--outputs_directory', + default=os.path.join('..', 'output'), + help="Directory containing walk MGRA output files.") + + self.parser.add_argument( + '-i', '--inputs_parent_directory', + default='..', + help="Directory containing 'input' folder") + + self.cli = self.parser.parse_args() + + def validate_file(self, filename): + if not os.path.isfile(filename): + self.parser.print_help() + raise IOError("Could not locate %s" % filename) + + def init_properties(self): + """Parses attributes from a Java properties file + + """ + + filename = self.cli.properties_file + self.validate_file(filename) + print('Parsing tokens from %s ...' % filename) + + all_props = {} + with open(filename, 'r') as f: + for line in f: + if line.startswith("#"): + continue + if '=' in line: + atr, val = list(map(str.strip, line.split('='))) + all_props[atr] = val + + def parse(property_name): + if property_name not in all_props: + raise KeyError("Could not find %s in %s" % (property_name, filename)) + + return all_props.get(property_name) + + self.mgra_file = parse('mgra.socec.file') + self.walk_mgra_output_file = parse('active.logsum.matrix.file.walk.mgra') + self.walk_mgra_tap_output_file = parse('active.logsum.matrix.file.walk.mgratap') + self.mt_tap_file = parse('active.microtransit.tap.file') + self.mt_mgra_file = parse('active.microtransit.mgra.file') + + self.walk_coef = float(parse('active.walk.minutes.per.mile')) + self.walk_max_dist_mgra = float(parse('active.maxdist.walk.mgra')) + self.walk_max_dist_tap = float(parse('active.maxdist.walk.tap')) + + self.vot = float(parse('active.micromobility.vot')) + + self.mm_speed = float(parse('active.micromobility.speed')) + self.mm_rental_time = float(parse('active.micromobility.rentalTime')) + self.mm_constant = float(parse('active.micromobility.constant')) + self.mm_variable_cost = float(parse('active.micromobility.variableCost')) + self.mm_fixed_cost = float(parse('active.micromobility.fixedCost')) + self.mm_max_dist_mgra = float(parse('active.maxdist.micromobility.mgra')) + self.mm_max_dist_tap = float(parse('active.maxdist.micromobility.tap')) + + self.mt_speed = float(parse('active.microtransit.speed')) + self.mt_wait_time = float(parse('active.microtransit.waitTime')) + self.mt_access_time = float(parse('active.microtransit.accessTime')) + self.mt_constant = float(parse('active.microtransit.constant')) + self.mt_variable_cost = float(parse('active.microtransit.variableCost')) + self.mt_fixed_cost = float(parse('active.microtransit.fixedCost')) + self.mt_not_avail = float(parse('active.microtransit.notAvailable')) + self.mt_max_dist_mgra = float(parse('active.maxdist.microtransit.mgra')) + self.mt_max_dist_tap = float(parse('active.maxdist.microtransit.tap')) + + def init_micro_access_time(self): + """Reads the MicroAccessTime for each origin MGRA from + the provided MGRA file. If no MicroAccessTime is found, + a simple calculation is performed instead. + + """ + mgra_file_path = os.path.join(self.cli.inputs_parent_directory, self.mgra_file) + self.validate_file(mgra_file_path) + + with open(mgra_file_path, 'r') as f: + use_dummy = 'MicroAccessTime' not in f.readline() + + if use_dummy: + print('No MicroAccessTime column found in %s, using 2 minute ' + 'default for PARKAREA==1, 15 minutes otherwise.' % mgra_file_path) + park_area = pd.read_csv(mgra_file_path, usecols=['mgra', 'parkarea'], + index_col='mgra', dtype='Int64', squeeze=True) + mat = pd.Series(index=park_area.index, data=15.0, name='MicroAccessTime') + mat.loc[park_area == 1] = 2.0 + else: + mat = pd.read_csv(mgra_file_path, usecols=['mgra', 'MicroAccessTime'], + index_col='mgra', squeeze=True) + + self.mat = mat + + def init_tap_mgra_lists(self): + """Reads in lists of ids that identify micro-transit accessibility TAPs/MGRAs + + """ + mt_tap_file_path = os.path.join(self.cli.inputs_parent_directory, self.mt_tap_file) + mt_mgra_file_path = os.path.join(self.cli.inputs_parent_directory, self.mt_mgra_file) + self.validate_file(mt_tap_file_path) + self.validate_file(mt_mgra_file_path) + + self.mt_taps = \ + pd.read_csv(mt_tap_file_path, + usecols=lambda x: x.strip().lower() == 'tap', + squeeze=True).values + + self.mt_mgras = \ + pd.read_csv(mt_mgra_file_path, + usecols=lambda x: x.strip().lower() == 'mgra', + squeeze=True).values + + +if __name__ == '__main__': + + config = Config() + process_file(config, zone='tap') + process_file(config, zone='mgra') + + print('Finished!') diff --git a/sandag_abm/src/main/python/checkFreeSpace.py b/sandag_abm/src/main/python/checkFreeSpace.py new file mode 100644 index 0000000..d94812f --- /dev/null +++ b/sandag_abm/src/main/python/checkFreeSpace.py @@ -0,0 +1,33 @@ +__author__ = 'wsu' +import sys +import ctypes +""" +ctypes is a foreign function library for Python. +It provides C compatible data types, and allows calling functions in DLLs or shared libraries. + +sys provides access to operating system. +It provides access to some variables used by the interpreter. +""" +path=sys.argv[1] +minSpace=sys.argv[2] +_, total, free = ctypes.c_ulonglong(), ctypes.c_ulonglong(), ctypes.c_ulonglong() +if sys.version_info >= (3,) or isinstance(path, unicode): + fun = ctypes.windll.kernel32.GetDiskFreeSpaceExW +else: + fun = ctypes.windll.kernel32.GetDiskFreeSpaceExA +ret = fun(path, ctypes.byref(_), ctypes.byref(total), ctypes.byref(free)) +if ret == 0: + raise ctypes.WinError() +totalMB=total.value/1024.0/1024.0 +freeMB=free.value/2014.0/1024.0 +usedMB = totalMB- freeMB + +if freeMB < int(minSpace): + print "free space on C <",minSpace,"MB!" + sys.exit() +else: + print "Total MB on C:",totalMB + print "Used MB on C:",usedMB + print "Free MB on C:",freeMB + + diff --git a/sandag_abm/src/main/python/check_output.py b/sandag_abm/src/main/python/check_output.py new file mode 100644 index 0000000..b0dc43b --- /dev/null +++ b/sandag_abm/src/main/python/check_output.py @@ -0,0 +1,139 @@ +""" Output Checker + +Checks that ABM components successfully generate required files. + +""" + +# Import libraries +import os +import sys + + +# Define model-output dictionary +output_dict = { + "Setup": [ + "walkMgraTapEquivMinutes.csv", + "microMgraTapEquivMinutes.csv", + "microMgraEquivMinutes.csv", + "bikeTazLogsum.csv", + "bikeMgraLogsum.csv", + "walkMgraEquivMinutes.csv" + ], + "SDRM": [ + "wsLocResults_ITER.csv", + "aoResults.csv", + "householdData_ITER.csv", + "indivTourData_ITER.csv", + "indivTripData_ITER.csv", + "jointTourData_ITER.csv", + "jointTripData_ITER.csv", + "personData_ITER.csv" + ], + "IE": [ + "internalExternalTrips.csv" + ], + "SAN": [ + "airport_out.SAN.csv" + ], + "CBX": [ + "airport_out.CBX.csv" + ], + "CBM": [ + "crossBorderTrips.csv", + "crossBorderTours.csv" + ], + "Visitor": [ + "visitorTrips.csv", + "visitorTours.csv" + ], + "TNC": [ + "TNCTrips.csv" + ], + "AV": [ + "householdAVTrips.csv" + ], + "CVM": [ + "Gen and trip sum.csv" + ], + "Exporter": [ + # NOTE: This is an incomplete list of the 40+ output files. + # These shapefiles are last to be generated and their + # existence indicates a successful Data Export. + "hwyLoad.prj", + "hwyLoad.cpg", + "hwyLoad.shx", + "hwyLoad.shp", + "hwyLoad.dbf" + ] +} + + +def check_output(scenario_fp, component, iteration=None): + """ + Checks that a specific ABM component generated + required files. + + :param component: String representing ABM component + :param scenario_fp: String representing scenario file path + :param iteration: Integer representing ABM iteration + :returns: Exit code + """ + + # Get required files + files = output_dict[component] + + # Construct output file path + if component == 'Exporter': + out_dir = 'report' + else: + out_dir = 'output' + output_dir = os.path.join(scenario_fp, out_dir) + + # Check that required files were generated + missing = [] + for file in files: + + # Append iteration integer if needed + if 'ITER' in file: + file = file.replace('ITER', iteration) + + file_path = os.path.join(output_dir, file) + if os.path.exists(file_path): + continue + else: + missing.append(file+'\n') + + # Write out missing files to log file + if len(missing) > 0: + create_log(scenario_fp, component, missing) + return sys.exit(2) + + return sys.exit(0) + + +def create_log(scenario_fp, component, lst): + """ + Creates a log file containing the files an ABM component + failed to generate. + + :param scenario_fp: String representing scenario file path + :param component: ABM component + :param lst: List of missing file names + """ + + # Create log file path + log_fp = os.path.join(scenario_fp, 'logFiles', 'missing_files.log') + + # Create and write to log file + with open(log_fp, "w") as log: + header = "'{}' failed to generate the following files:\n".format(component) + log.write(header) + log.writelines(lst) + + return + + +if __name__ == '__main__': + targets = sys.argv[1:] + check_output(*targets) + diff --git a/sandag_abm/src/main/python/cvm_analysis.zip b/sandag_abm/src/main/python/cvm_analysis.zip new file mode 100644 index 0000000000000000000000000000000000000000..dd5fd7bcfbcda20b66279f6646d20f9efa93128a GIT binary patch literal 29456 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 BF;f5l literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/python/cvm_input_create.py b/sandag_abm/src/main/python/cvm_input_create.py new file mode 100644 index 0000000..16868e9 --- /dev/null +++ b/sandag_abm/src/main/python/cvm_input_create.py @@ -0,0 +1,462 @@ +''' +PURPOSE: +Commercial Vehicle Model (CVM) input file creation + +INSTALL (LIBRARY): +Numpy +https://pypi.python.org/pypi/numpy + +Pandas +https://pypi.python.org/pypi/pandas + +HOW TO RUN: +[script_filepath] [Project Directory] [MGRA filename] [TAZ Centroid flename] [output filename] + +example: cvm_input_create.py "C:\Projects\SANDAG_CTM_Validation\_Tasks" "input/mgra13_based_input2012.csv" "tazcentroids_cvm.csv" "Zonal Properties CVM.csv" + +Note: The script name is sufficient if the run folder is the same as the script folder. Otherwise, a full path of the script would be needed. Also, if the output file does not contain spaces, the command line does not need to have quotation marks around the arguments. + +STEPS: +STEP 0: read mgra socio-economic file +STEP 1: find min and max tazids +STEP 2: read taz centroids. +STEP 3: calculate taz level variables +STEP 4. write to output + +REFERENCES: +Following reference are used for calculations: +T:\devel\CVM\sr13\2012_calib5\input\mgra13_based_input2012_CVM.xlsx +T:\devel\CVM\sr13\2012_calib5\CVM\Zonal Properties SDCVM_SR13 KJS rcu_check.xlsx +The final example product: T:\devel\CVM\sr13\2012_calib5\CVM\Inputs\Zonal Properties CVM.csv + +CREATED BY: +nagendra.dhakar@rsginc.com + +LAST MODIFIED: +03/13/2018 + +Updates: +03/13/2018 - nagendra.dhakar@rsginc.com +Updated to remove transformation of coordinates +Instead an input file is provided with taz centroids (tazcentroids_cvm.csv +the script now reads the input centroid file that already has transformed coordinates that are reported in the output +no need for GDAL library +''' + +import sys +import os +import numpy as np +import datetime +import math +import csv +import pandas as pd + +class Constant: + """ Represents constants in the script + constants: FEET_TO_MILE, COORDS_EPSG_SOURCE,COORDS_EPSG_TARGET + """ + ACRES_TO_SQMILE = 0.0015625 + #COORDS_EPSG_SOURCE = 2230 #EPSG: 2230 - NAD83/ California zone 6 (ftUS) + #COORDS_EPSG_TARGET = 3310 #EPSG: 3310 - NAD83/ California Albers + +class Header: + """ Represents header for output files """ + temptazdatafile = ['taz','pop','hh','i1','i2','i3','i4','i5','i6','i7','i8','i9','i10','emp_total','emp_fed_mil','sqmile','land_sqmile','Emp_IN','Emp_RE','Emp_SV','Emp_TU','Emp_WH','Emp_OFF', + 'HHIncome','EmpDens','PopDens','Per/Emp','Emp_ServRet_Pct','Ret_ServRet','Emp_Office','Low Density','Residential','Commercial','Industrial','Employment Node', + 'Industrial_pct','TU_pct','Wholesale_pct','Retail_pct','Service_pct','Office_pct','E500_Industrial','E500_TU','E500_Wholesale','E500_Retail','E500_Service', + 'E500_Office','RetailZone','ZoneType'] + + temptazcentroidsfile = ["hnode","x_coord_spft","y_coord_spft","x_coord_albers","y_coord_albers"] + + outfile = ['TAZ','Pop','Income','Area_SqMi','x-meters','y-meters','EmpDens','PopDens','TotEmp','Military', + 'CVM_IN','CVM_RE','CVM_SV','CVM_TH','CVM_WH','CVM_GO','CVM_LU_Type','SqrtArea','CVM_LU_Low','CVM_LU_Res', + 'CVM_LU_Ret','CVM_LU_Ind','CVM_LU_Emp','Emp_LU_Lo','Emp_LU_Re','Emp_LU_RC','Emp_LU_In','Emp_LU_EN'] + +class TazId: + """ Represents Taz Id range + attributes: min, max + """ + +class Input: + """ Represents input settings + attributes: dir, mgrafile, nodebinfile. + """ +class Output: + """ Represents output settings + attributes: dir, outfile, temp_centroidfile, temp_tazfile + """ + +def read_node_header(node_file): + """Returns name and type of the fields in the node bin file """ + + # node header file + nodeheaderfile = os.path.join(os.path.splitext(node_file)[0]+".DCB") + + # first row is blank + # second row is total bytes in a row + # start reading fields names from third row + # header file format - field_name, type, start_byte, length, .. + + fields_info=[] + with open(nodeheaderfile) as headerfile: + reader = csv.reader(headerfile) + i=0 + for row in reader: + if i >= 2: + field_name = row[0] + field_length = row[3] + + # integer + if row[1] == 'I': + field_type = int + + # character/string + elif row[1] == 'C': + field_type = 'S' + field_length + + # first field + if i == 2: + fields_info.append([field_name]) + fields_info.append([field_type]) + + # remaining fields + else: + fields_info[0].append(field_name) + fields_info[1].append(field_type) + + i=i+1 + + return fields_info + +def get_tazid_range(data): + """Returns min and maz taz ids""" + + # max and min tazid + + id_list = np.array(data.keys()) + id_list = id_list.astype(np.float) + + id_max = int(max(id_list)) + id_min = int(min(id_list)) + + return([id_min,id_max]) + +def read_node_file(tazcentroid_file): + """ + Reads taz centroids from node bin file + Transforms them into the coordinate system expected by the CTM + Returns transformed (projected) coordinates + """ + + centroids = pd.read_csv(tazcentroid_file) + centroids = centroids[centroids['taz']>0] + centroids = centroids.reset_index() + + coords_proj = {} + for i in range(0,max(centroids['taz'])): + coords_proj[str(centroids['taz'][i])] = [float(centroids['x_coord_albers'][i]), float(centroids['y_coord_albers'][i])] + + return coords_proj + +def read_mgra_input(mgrafile): + """ + Reads MGRA socia-economic file + Calculates some variables at MGRA level + Aggregates data by TAZ + Returns TAZ level data + """ + + data={} + + with open (mgrafile) as csvfile: + reader = csv.DictReader(csvfile) # read in dictionary format + i=1 + for row in reader: + # calculate new variables at MGRA + row['sqmile'] = float(row['acres'])*Constant.ACRES_TO_SQMILE + row['land_sqmile'] = float(row['land_acres'])*Constant.ACRES_TO_SQMILE + + row['CVM_IN'] = float(row['emp_ag']) + float(row['emp_const_non_bldg_prod']) + float(row['emp_const_non_bldg_office']) + \ + float(row['emp_const_bldg_prod']) + float(row['emp_const_bldg_office']) + float(row['emp_mfg_prod']) + float(row['emp_mfg_office']) + + row['CVM_RE'] = float(row['emp_retail']) + + row['CVM_SV'] = float(row['emp_pvt_ed_k12']) + float(row['emp_pvt_ed_post_k12_oth']) + float(row['emp_health']) + \ + float(row['emp_personal_svcs_office']) + float(row['emp_amusement']) + float(row['emp_hotel']) + \ + float(row['emp_restaurant_bar']) + float(row['emp_personal_svcs_retail']) + float(row['emp_religious']) + \ + float(row['emp_pvt_hh']) + float(row['emp_public_ed']) + + row['CVM_TH'] = float(row['emp_utilities_prod']) + float(row['emp_utilities_office']) + float(row['emp_trans']) + + row['CVM_WH'] = float(row['emp_whsle_whs']) + + + row['CVM_OFF'] = float(row['emp_prof_bus_svcs']) + float(row['emp_prof_bus_svcs_bldg_maint']) + \ + float(row['emp_state_local_gov_ent']) + float(row['emp_fed_non_mil']) + float(row['emp_state_local_gov_blue']) + \ + float(row['emp_state_local_gov_white']) + float(row['emp_own_occ_dwell_mgmt']) + + # aggregate data by TAZ + if row['taz'] not in data: + data[row['taz']] = [int(row['taz']),int(row['pop']), int(row['hh']), float(row['i1']), float(row['i2']), float(row['i3']), float(row['i4']), float(row['i5']), float(row['i6']), float(row['i7']), + float(row['i8']), float(row['i9']), float(row['i10']), float(row['emp_total']), float(row['emp_fed_mil']), float(row['sqmile']), + float(row['land_sqmile']), float(row['CVM_IN']), float(row['CVM_RE']), float(row['CVM_SV']), float(row['CVM_TH']), float(row['CVM_WH']), float(row['CVM_OFF'])] + + else: + #taz_data[row['TAZ']][0] = int(row['TAZ']) + data[row['taz']][1] += int(row['pop']) + data[row['taz']][2] += int(row['hh']) + data[row['taz']][3] += float(row['i1']) + data[row['taz']][4] += float(row['i2']) + data[row['taz']][5] += float(row['i3']) + data[row['taz']][6] += float(row['i4']) + data[row['taz']][7] += float(row['i5']) + data[row['taz']][8] += float(row['i6']) + data[row['taz']][9] += float(row['i7']) + data[row['taz']][10] += float(row['i8']) + data[row['taz']][11] += float(row['i9']) + data[row['taz']][12] += float(row['i10']) + data[row['taz']][13] += float(row['emp_total']) + data[row['taz']][14] += float(row['emp_fed_mil']) + data[row['taz']][15] += float(row['sqmile']) + data[row['taz']][16] += float(row['land_sqmile']) + data[row['taz']][17] += float(row['CVM_IN']) + data[row['taz']][18] += float(row['CVM_RE']) + data[row['taz']][19] += float(row['CVM_SV']) + data[row['taz']][20] += float(row['CVM_TH']) + data[row['taz']][21] += float(row['CVM_WH']) + data[row['taz']][22] += float(row['CVM_OFF']) + + return data + +# calculate taz level variables +def calculate_taz_variables(data, coords, taz_ids, outfile): + """ + Reads TAZ data stored in read_mgra_input + Calculates variables + Returns calculated taz variables that would be in the output file + """ + + data_calc = {} + with open(outfile,"wb") as csvfile: + fieldnames = Header.temptazdatafile + + writer = csv.writer(csvfile) + writer.writerow(fieldnames) + + for taz in range(1, taz_ids.max+1): + + if taz>=taz_ids.min: + + # initialize variables to 0 + emp_dens, pop_dens, cvm_emp_dens, emp_cvm_total=(0,)*4 + per_emp, emp_servret_pct, ret_servret, emp_office, low_dens, residential, commercial, industrial=(0,)*8 + industrial_pct, transport_pct, wholesale_pct, retail_pct, service_pct, office_pct=(0,)*6 + emp_office, retail_zone=(0,)*2 + industrial_e500, transport_e500, wholesale_e500, retail_e500, service_e500, office_e500=(0,)*6 + cvm_lu_low, cvm_lu_res, cvm_lu_ret, cvm_lu_ind, cvm_lu_emp=(0,)*5 + + # get data + [taz_id,pop,hh,inc1,inc2,inc3,inc4,inc5,inc6,inc7,inc8,inc9,inc10,emp_total,emp_fed_mil,sqmile,land_sqmile,cvm_in,cvm_re,cvm_sv,cvm_th,cvm_wh,cvm_off]=data[str(taz)] + + # average hh income + if (hh>0): + hh_income = (inc1*7500+inc2*22500+inc3*37500+inc4*52500+inc5*67500+inc6*87500+inc7*112500+inc8*137500+inc9*175000+inc10*225)/hh + else: + hh_income=64678 + + # total CVM employment = industrial + retail + service + transport + wholesale + office + emp_cvm_total = cvm_in + cvm_re + cvm_sv + cvm_th + cvm_wh + cvm_off + + # calculate densities + if (land_sqmile > 0): + emp_dens = emp_total/land_sqmile + pop_dens = pop/land_sqmile + cvm_emp_dens = emp_cvm_total/land_sqmile + + # share of employment in each sector + if (emp_total > 0): + per_emp = pop/emp_total + emp_servret_pct = (cvm_re + cvm_sv + cvm_off)/emp_total + + if ((cvm_re+cvm_sv+cvm_off)/emp_total) < 0.8: + emp_office = 1 + + # additional shares/variables - not for the final output + industrial_pct = cvm_in/emp_total + transport_pct = cvm_th/emp_total + wholesale_pct = cvm_wh/emp_total + retail_pct = cvm_re/emp_total + service_pct = cvm_sv/emp_total + office_pct = cvm_off/emp_total + + if cvm_re/emp_total > 0.5: + retail_zone = 1 + + # end of additional variables + + # calculate flags + if ((cvm_re+cvm_sv) > 0): + if ((cvm_re/(cvm_re+cvm_sv+cvm_off)) > 0.25): + ret_servret = 1 + + # landuse flags + + if (emp_dens < 250 and pop_dens < 250): + # low density + low_dens = 1 + + if (low_dens == 0 and pop_dens > 250 and per_emp > 2): + # residential + residential = 1 + + if (low_dens == 0 and residential == 0 and emp_servret_pct > 0.6 and emp_dens > 1500 and ret_servret == 1): + # retail/commercial + commercial = 1 + + if (low_dens == 0 and residential == 0 and commercial == 0 and emp_dens < 15000 and emp_office == 1): + # industrial + industrial = 1 + + if (low_dens == 1 or residential == 1 or commercial == 1 or industrial == 1): + employment_node = 0 + else: + # other + employment_node = 1 + + # employment more than 500 flags - additional, not for the final output + if cvm_in > 500: + industrial_e500 = 1 + if cvm_th > 500: + transport_e500 = 1 + if cvm_wh > 500: + wholesale_e500 = 1 + if cvm_re > 500: + retail_e500 = 1 + if cvm_sv > 500: + service_e500 = 1 + if cvm_off > 500: + office_e500 = 1 + + # end of additional variables + + # zone type + zone_type = 1*low_dens + 2*residential + 3*commercial + 4*industrial + 5*employment_node + sqrt_area = math.sqrt(land_sqmile) + + # TAZ centroids + taz_x_meters = coords[str(taz)][0] + taz_y_meters = coords[str(taz)][1] + + # landuse flags + if zone_type == 1: + # low density + cvm_lu_low = 1 + elif zone_type == 2: + # residential + cvm_lu_res = 1 + elif zone_type == 3: + # retail/commercial + cvm_lu_ret = 1 + elif zone_type == 4: + # industrial + cvm_lu_ind = 1 + elif zone_type == 5: + # other + cvm_lu_emp = 1 + + # employment by land use + emp_lu_low = cvm_lu_low * emp_cvm_total + emp_lu_res = cvm_lu_res * emp_cvm_total + emp_lu_ret = cvm_lu_ret * emp_cvm_total + emp_lu_ind = cvm_lu_ind * emp_cvm_total + emp_lu_emp = cvm_lu_emp * emp_cvm_total + + # write all taz data to a temp file + data[str(taz)].extend([hh_income,emp_dens,pop_dens,per_emp,emp_servret_pct,ret_servret,emp_office, + low_dens,residential,commercial,industrial,employment_node,industrial_pct, + transport_pct,wholesale_pct,retail_pct,service_pct,office_pct,industrial_e500, + transport_e500,wholesale_e500,retail_e500,service_e500,office_e500,retail_zone,zone_type]) + writer.writerow(data[str(taz)]) + + # store calculated variables + data_calc[str(taz)]=[taz_id,pop,hh_income,land_sqmile,taz_x_meters,taz_y_meters,cvm_emp_dens,pop_dens,emp_cvm_total, + emp_fed_mil,cvm_in,cvm_re,cvm_sv,cvm_th,cvm_wh,cvm_off,zone_type,sqrt_area, + cvm_lu_low,cvm_lu_res,cvm_lu_ret,cvm_lu_ind,cvm_lu_emp, + emp_lu_low,emp_lu_res,emp_lu_ret,emp_lu_ind,emp_lu_emp] + return data_calc + +def write_output(data, taz_ids, outfile): + """ + Writes taz data to an output + """ + + with open(outfile,"wb") as csvfile: + fieldnames = Header.outfile + + writer = csv.writer(csvfile) + writer.writerow(fieldnames) + + for taz in range(1, taz_ids.max+1): + if taz None: + self.scenario_path = scenario_path + + @property + @lru_cache(maxsize=1) + def mgra_xref(self) -> pd.DataFrame: + """ Cross reference of Master Geographic Reference Area (MGRA) model + geography to Transportation Analysis Zone (TAZ) and Land Use Zone + (LUZ) model geographies. Cross reference is stored in each ABM + scenario input MGRA file (input/mgra13_based_input<>.csv). + """ + + # load the mgra based input file + fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" + + mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), + usecols=["mgra", # MGRA geography + "taz", # TAZ geography + "luz_id"], + dtype={"mgra": "int16", + "taz": "int16", + "luz_id": "int16"}) # LUZ geography + + # genericize column names + mgra.rename(columns={"mgra": "MGRA", + "taz": "TAZ", + "luz_id": "LUZ"}, + inplace=True) + + return mgra + + @property + @lru_cache(maxsize=1) + def pnr_taps(self) -> pd.DataFrame: + """ Create the transit TAP park and ride lot data-set. + + Read in and combine the transit TAP parking lot type information and + parking lot vehicles by time of day information. + + Returns: + A Pandas DataFrame of the transit TAP park and ride lot data-set """ + # load parking lot type data-set + lots = pd.read_fwf( + os.path.join(self.scenario_path, "input", "tap.ptype"), + names=["TAP", + "lotID", + "parkingType", + "lotTAZ", + "capacity", + "distance", + "mode"], + header=None, + widths=[5, 6, 6, 5, 5, 5, 3]) + + # replicate parking lot data by ABM five time of day + five_tod = pd.DataFrame( + {"key": [0]*5, + "timeFiveTod": ["EA", "AM", "MD", "PM", "EV"]}) + + lots["key"] = 0 + lots = lots.merge(five_tod) + + # load parking lot vehicles by time of day + vehicles = pd.read_csv( + os.path.join(self.scenario_path, "output", "PNRByTAP_Vehicles.csv"), + usecols=["TAP", + "EA", + "AM", + "MD", + "PM", + "EV"]) + + # restructure vehicle data from wide to long by ABM five time of day + vehicles = pd.melt( + vehicles, + id_vars=["TAP"], + value_vars=["EA", "AM", "MD", "PM", "EV"], + var_name="timeFiveTod", + value_name="vehicles") + + # merge parking lot and vehicle data + lots = lots.merge(vehicles, how="left") + + # set missing vehicle fields to 0 + lots["vehicles"] = lots["vehicles"].fillna(0) + + # convert distance field from feet to miles + lots["distance"] = lots["distance"] / 5280 + + # apply exhaustive field mappings where applicable + mappings = { + "timeFiveTod": {"EA": 1, + "AM": 2, + "MD": 3, + "PM": 4, + "EV": 5}, + "parkingType": {1: "Formal Parking", + 2: "Other Parking", + 3: "Other Light Rail Trolley Parking", + 4: "Non-formal parking area based on the on-board survey", + 5: "Non-formal parking area based on the on-board survey"} + } + + for field in mappings: + lots[field] = lots[field].map(mappings[field]) + + # rename columns to standard/generic ABM naming conventions + lots.rename(columns={"TAP": "tapID"}, inplace=True) + + return lots[["tapID", + "lotID", + "lotTAZ", + "timeFiveTod", + "parkingType", + "capacity", + "distance", + "vehicles"]] + + @property + @lru_cache(maxsize=1) + def properties(self) -> dict: + """ Get the ABM scenario properties from the ABM scenario + properties file (conf/sandag_abm.properties). + + The return dictionary contains the following ABM scenario properties: + cvmScaleLight - commercial vehicle model trip scaling factor for + light vehicles for each ABM five time of day + cvmScaleMedium - commercial vehicle model trip scaling factor for + intermediate and medium vehicles for each ABM five time of day + cvmScaleHeavy - commercial vehicle model trip scaling factor for + heavy vehicles for each ABM five time of day + cvmShareLight - commercial vehicle model trip intermediate vehicle + share factor for light vehicles + cvmShareMedium - commercial vehicle model trip intermediate vehicle + share factor for medium vehicles + cvmShareHeavy - commercial vehicle model trip intermediate vehicle + share factor for heavy vehicles + iterations - number of model iteration + nonPooledTNCPassengers - average number of passengers to assume for + Non-Pooled TNC mode in models without party size specifications + pooledTNCPassengers - average number of passengers to assume for + Pooled TNC mode in models without party size specifications + sr2Passengers - average number of passengers to assume for Shared + Ride 2 mode in models without party size specifications + sr3Passengers - average number of passengers to assume for Shared + Ride 3+ mode in models without party size specifications + taxiPassengers - average number of passengers to assume for Taxi + mode in models without party size specifications + timePeriodWidthTNC - time period width (in minutes) for custom + fixed-width time periods used in TNC routing model, note that + this is not currently restricted to nest within ABM model time + periods + sampleRate - sample rate of final iteration + valueOfTimeLow - upper limit of 'Low' value of time category + valueOfTimeMedium - upper limit of 'Medium' value of time category + year - analysis year of the ABM scenario + + Returns: + A dictionary defining the ABM scenario properties. """ + + # create dictionary holding ABM properties file information + # each property contains a dictionary {line, value} where the line + # is the string to match in the properties file to + # return the value of the property + lookup = { + "cvmScaleLight": { + "line": "cvm.scale_light=", + "type": "list", + "value": None}, + "cvmScaleMedium": { + "line": "cvm.scale_medium=", + "type": "list", + "value": None}, + "cvmScaleHeavy": { + "line": "cvm.scale_heavy=", + "type": "list", + "value": None}, + "cvmShareLight": { + "line": "cvm.share.light=", + "type": "float", + "value": None}, + "cvmShareMedium": { + "line": "cvm.share.medium=", + "type": "float", + "value": None}, + "cvmShareHeavy": { + "line": "cvm.share.heavy=", + "type": "float", + "value": None}, + "iterations": { + "line": None, + "type": "int", + "value": None}, + "nonPooledTNCPassengers": { + "line": "TNC.single.passengersPerVehicle=", + "type": "float", + "value": None}, + "pooledTNCPassengers": { + "line": "TNC.shared.passengersPerVehicle=", + "type": "float", + "value": None}, + "sr2Passengers": { + "line": None, + "type": "int", + "value": 2}, + "sr3Passengers": { + "line": None, + "type": "float", + "value": 3.34}, + "taxiPassengers": { + "line": "Taxi.passengersPerVehicle=", + "type": "float", + "value": None}, + "timePeriodWidthTNC": { + "line": "Maas.RoutingModel.minutesPerSimulationPeriod=", + "type": "int", + "value": None}, + "sampleRate": { + "line": "sample_rates=", + "type": "float", + "value": None}, + "valueOfTimeLow": { + "line": "valueOfTime.threshold.low=", + "type": "float", + "value": None}, + "valueOfTimeMedium": { + "line": "valueOfTime.threshold.med=", + "type": "float", + "value": None}, + "year": { + "line": "scenarioYear=", + "type": "int", + "value": None} + } + + # open the ABM scenario properties file + file = open(os.path.join(self.scenario_path, "conf", "sandag_abm.properties"), "r") + + # loop through each line of the properties file + for line in file: + # strip all white space from the line + line = line.replace(" ", "") + + # for each element of the properties dictionary + for name in lookup: + item = lookup[name] + + # if the properties file contains the matching line + if item["line"] is not None: + match = re.compile(item["line"]).match(line) + else: + match = False + + if match: + # if the match is for the sample rate element + # then take the portion of the line after the matching string + # and split by the comma character + if name == "sampleRate": + line = line[match.end():].split(",") + + # set number of iterations to number of sample rates + # that are specified + lookup["iterations"]["value"] = len(line) + + # if the split line contains a single element + # return that element otherwise return the final element + if len(line) == 1: + value = line[0] + else: + value = line[-1] + # if the match is for a cvm scale element then take the + # portion of the line after the matching string and split + # by the comma character into a list of floats + elif "cvmScale" in name: + value = line[match.end():].split(",") + value = list(map(float, value)) + # otherwise take the final element of the line + else: + value = line[match.end():] + + # update the dictionary value using the appropriate data type + if item["type"] == "float": + value = float(value) + elif item["type"] == "int": + value = int(value) + else: + pass + + item["value"] = value + + break + + file.close() + + # convert the property name and value to a non-nested dictionary + results = {} + for name in lookup: + results[name] = lookup[name]["value"] + + return results + + @property + def time_periods(self) -> dict: + """ Dictionary of ABM model time resolution periods with start and + end times where the start time is inclusive and the end time is + exclusive. Dictionary is of the form: + {"period": "startTime": "endTime":} + + Returns: + A Dictionary of the ABM model time resolution periods. + """ + periods = { + "abmHalfHour": [ + {"period": 1, + "startTime": time(3, 0), + "endTime": time(5, 0)}, + {"period": 2, + "startTime": time(5, 0), + "endTime": time(5, 30)}, + {"period": 3, + "startTime": time(5, 30), + "endTime": time(6, 0)}, + {"period": 4, + "startTime": time(6, 0), + "endTime": time(6, 30)}, + {"period": 5, + "startTime": time(6, 30), + "endTime": time(7, 0)}, + {"period": 6, + "startTime": time(7, 0), + "endTime": time(7, 30)}, + {"period": 7, + "startTime": time(7, 30), + "endTime": time(8, 0)}, + {"period": 8, + "startTime": time(8, 0), + "endTime": time(8, 30)}, + {"period": 9, + "startTime": time(8, 30), + "endTime": time(9, 0)}, + {"period": 10, + "startTime": time(9, 0), + "endTime": time(9, 30)}, + {"period": 11, + "startTime": time(9, 30), + "endTime": time(10, 0)}, + {"period": 12, + "startTime": time(10, 0), + "endTime": time(10, 30)}, + {"period": 13, + "startTime": time(10, 30), + "endTime": time(11, 0)}, + {"period": 14, + "startTime": time(11, 0), + "endTime": time(11, 30)}, + {"period": 15, + "startTime": time(11, 30), + "endTime": time(12, 0)}, + {"period": 16, + "startTime": time(12, 0), + "endTime": time(12, 30)}, + {"period": 17, + "startTime": time(12, 30), + "endTime": time(13, 0)}, + {"period": 18, + "startTime": time(13, 0), + "endTime": time(13, 30)}, + {"period": 19, + "startTime": time(13, 30), + "endTime": time(14, 0)}, + {"period": 20, + "startTime": time(14, 0), + "endTime": time(14, 30)}, + {"period": 21, + "startTime": time(14, 30), + "endTime": time(15, 0)}, + {"period": 22, + "startTime": time(15, 0), + "endTime": time(15, 30)}, + {"period": 23, + "startTime": time(15, 30), + "endTime": time(16, 0)}, + {"period": 24, + "startTime": time(16, 0), + "endTime": time(16, 30)}, + {"period": 25, + "startTime": time(16, 30), + "endTime": time(17, 0)}, + {"period": 26, + "startTime": time(17, 0), + "endTime": time(17, 30)}, + {"period": 27, + "startTime": time(17, 30), + "endTime": time(18, 0)}, + {"period": 28, + "startTime": time(18, 0), + "endTime": time(18, 30)}, + {"period": 29, + "startTime": time(18, 30), + "endTime": time(19, 0)}, + {"period": 30, + "startTime": time(19, 0), + "endTime": time(19, 30)}, + {"period": 31, + "startTime": time(19, 30), + "endTime": time(20, 0)}, + {"period": 32, + "startTime": time(20, 0), + "endTime": time(20, 30)}, + {"period": 33, + "startTime": time(20, 30), + "endTime": time(21, 0)}, + {"period": 34, + "startTime": time(21, 0), + "endTime": time(21, 30)}, + {"period": 35, + "startTime": time(21, 30), + "endTime": time(22, 0)}, + {"period": 36, + "startTime": time(22, 0), + "endTime": time(22, 30)}, + {"period": 37, + "startTime": time(22, 30), + "endTime": time(23, 0)}, + {"period": 38, + "startTime": time(23, 0), + "endTime": time(23, 30)}, + {"period": 39, + "startTime": time(23, 30), + "endTime": time.max}, + {"period": 40, + "startTime": time.min, + "endTime": time(3, 0)} + ], + "abm5Tod": [ + {"period": 1, + "startTime": time(3, 0), + "endTime": time(6, 0)}, + {"period": 2, + "startTime": time(6, 0), + "endTime": time(9, 0)}, + {"period": 3, + "startTime": time(9, 0, 0), + "endTime": time(15, 30)}, + {"period": 4, + "startTime": time(15, 30), + "endTime": time(19, 0)}, + {"period": 5, + "startTime": time(19, 0), + "endTime": time.max}, + {"period": 5, + "startTime": time.min, + "endTime": time(3, 0)} + ] + } + + return periods + + @staticmethod + def _map_time_periods(abm_half_hour: pd.Series) -> pd.Series: + """ Map ABM half hour time periods to ABM five time of day periods + + Returns: + A Pandas Series of ABM five time of day periods """ + + conditions = [abm_half_hour.between(1, 3), + abm_half_hour.between(4, 9), + abm_half_hour.between(10, 22), + abm_half_hour.between(23, 29), + abm_half_hour.between(30, 40)] + + choices = [1, 2, 3, 4, 5] + + abm_5_tod = np.select(conditions, choices, default=np.NaN) + + return pd.Series(abm_5_tod).astype("float") + + def _map_vot_categories(self, vot: pd.Series) -> pd.Series: + """ Map Pandas Series of continuous ABM value of time (vot) values to + vot categories ("Low", "Medium", "High") defined in the ABM scenario + properties file. + + Returns: + A Pandas Series of value of time categories. """ + + # get vot thresholds + low = self.properties["valueOfTimeLow"] + med = self.properties["valueOfTimeMedium"] + + # map continuous values of time to categories + conditions = [vot < low, + (low <= vot) & (vot < med), + vot >= med] + + choices = ["Low", "Medium", "High"] + + vot_category = np.select(conditions, choices, default=np.NaN) + + return pd.Series(vot_category).astype("category") + + +class LandUse(ScenarioData): + """ A subclass of the ScenarioData class. Holds all land use + data for a completed ABM scenario model run. As of now, this includes only + the MGRA-based input file. This is held as a class property. + + Properties: + mgra_input: MGRA-based input file + """ + @property + @lru_cache(maxsize=1) + def mgra_input(self) -> pd.DataFrame: + """ Create the MGRA-based input file data-set. """ + # load the MGRA-based input file + fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" + + mgra = pd.read_csv( + os.path.join(self.scenario_path, "input", fn), + usecols=["mgra", + "taz", + "hs", + "hs_sf", + "hs_mf", + "hs_mh", + "hh", + "hh_sf", + "hh_mf", + "hh_mh", + "gq_civ", + "gq_mil", + "i1", + "i2", + "i3", + "i4", + "i5", + "i6", + "i7", + "i8", + "i9", + "i10", + "hhs", + "pop", + "hhp", + "emp_ag", + "emp_const_non_bldg_prod", + "emp_const_non_bldg_office", + "emp_utilities_prod", + "emp_utilities_office", + "emp_const_bldg_prod", + "emp_const_bldg_office", + "emp_mfg_prod", + "emp_mfg_office", + "emp_whsle_whs", + "emp_trans", + "emp_retail", + "emp_prof_bus_svcs", + "emp_prof_bus_svcs_bldg_maint", + "emp_pvt_ed_k12", + "emp_pvt_ed_post_k12_oth", + "emp_health", + "emp_personal_svcs_office", + "emp_amusement", + "emp_hotel", + "emp_restaurant_bar", + "emp_personal_svcs_retail", + "emp_religious", + "emp_pvt_hh", + "emp_state_local_gov_ent", + "emp_fed_non_mil", + "emp_fed_mil", + "emp_state_local_gov_blue", + "emp_state_local_gov_white", + "emp_public_ed", + "emp_own_occ_dwell_mgmt", + "emp_fed_gov_accts", + "emp_st_lcl_gov_accts", + "emp_cap_accts", + "emp_total", + "enrollgradekto8", + "enrollgrade9to12", + "collegeenroll", + "othercollegeenroll", + "adultschenrl", + "ech_dist", + "hch_dist", + "pseudomsa", + "parkarea", + "hstallsoth", + "hstallssam", + "hparkcost", + "numfreehrs", + "dstallsoth", + "dstallssam", + "dparkcost", + "mstallsoth", + "mstallssam", + "mparkcost", + "zip09", + "parkactive", + "openspaceparkpreserve", + "beachactive", + "hotelroomtotal", + "truckregiontype", + "district27", + "milestocoast", + "acres", + "effective_acres", + "land_acres", + "MicroAccessTime", + "remoteAVParking", + "refueling_stations", + "totint", + "duden", + "empden", + "popden", + "retempden", + "totintbin", + "empdenbin", + "dudenbin", + "PopEmpDenPerMi"]) + + return mgra + + +class SyntheticPopulation(ScenarioData): + """ A subclass of the ScenarioData class. Holds all synthetic population + data for a completed ABM scenario model run. This includes the input + synthetic persons and households sampled in the ABM model run as well as + model results pertaining to person and household attributes (e.g. work + location, parking reimbursement, etc...). The synthetic population persons + and households are held as class properties and include: + Synthetic Households + Synthetic Persons + + Properties: + households: Synthetic households sampled + persons: Synthetic persons sampled + """ + @property + @lru_cache(maxsize=1) + def households(self) -> pd.DataFrame: + """ Create the synthetic households data-set. + + Read in the input synthetic household list and the sampled synthetic + household list, combine the lists taking only sampled households, + map field values, and genericize field names. + + Returns: + A Pandas DataFrame of the synthetic households """ + # load input synthetic household list into Pandas DataFrame + input_households = pd.read_csv( + os.path.join(self.scenario_path, "input", "households.csv"), + usecols=["hhid", + "taz", + "mgra", + "hinccat1", + "hinc", + "hworkers", + "persons", + "bldgsz", + "unittype", + "poverty"], + dtype={"hhid": "int32", + "taz": "int16", + "mgra": "int16", + "hinccat1": "int8", + "hinc": "int32", + "hworkers": "int8", + "persons": "int8", + "bldgsz": "int8", + "unittype": "int8", + "poverty": "float32"}) + + # load output sampled synthetic household list + fn = "householdData_" + str(self.properties["iterations"]) + ".csv" + output_households = pd.read_csv( + os.path.join(self.scenario_path, "output", fn), + usecols=["hh_id", + "autos", + "HVs", + "AVs", + "transponder"], + dtype={"hh_id": "int32", + "autos": "int8", + "HVs": "int8", + "AVs": "int8", + "transponder": "bool"}) + + # merge output sampled households with input sampled households + # keep only households present in the sampled households + households = output_households.merge( + input_households, + how="inner", + left_on="hh_id", + right_on="hhid" + ) + + # apply exhaustive field mappings where applicable + mappings = { + "hinccat1": {1: "Less than 30k", + 2: "30k-60k", + 3: "60k-100k", + 4: "100k-150k", + 5: "150k+"}, + "bldgsz": {1: "Mobile Home or Trailer", + 2: "Single Family Home - Detached", + 3: "Single Family Home - Attached", + 8: "Multi-Family Home", + 9: "Other (includes Group Quarters)"}, + "unittype": {0: "Non-Group Quarters", + 1: "Group Quarters"} + } + + for field in mappings: + households[field] = households[field].map(mappings[field]).astype("category") + + # rename columns to standard/generic ABM naming conventions + households.rename(columns={"hh_id": "hhId", + "HVs": "autosHumanVehicles", + "AVs": "autosAutonomousVehicles", + "transponder": "transponderAvailable", + "mgra": "homeMGRA", + "taz": "homeTAZ", + "hinccat1": "hhIncomeCategory", + "hinc": "hhIncome", + "hworkers": "hhWorkers", + "persons": "hhPersons", + "bldgsz": "buildingCategory", + "unittype": "unitType"}, + inplace=True) + + return households[["hhId", + "autos", + "autosHumanVehicles", + "autosAutonomousVehicles", + "transponderAvailable", + "homeMGRA", + "homeTAZ", + "hhIncomeCategory", + "hhIncome", + "hhWorkers", + "hhPersons", + "buildingCategory", + "unitType", + "poverty"]] + + @property + @lru_cache(maxsize=1) + def persons(self) -> pd.DataFrame: + """ Create the synthetic persons data-set. + + Read in the input synthetic person list and the sampled synthetic + person list, combine the lists taking only sampled persons, + map field values, and genericize field names. + + Returns: + A Pandas DataFrame of the synthetic persons """ + # load input synthetic person list into Pandas DataFrame + input_persons = pd.read_csv( + os.path.join(self.scenario_path, "input", "persons.csv"), + usecols=["hhid", + "perid", + "pnum", + "age", + "sex", + "miltary", + "pemploy", + "pstudent", + "ptype", + "educ", + "grade", + "weeks", + "hours", + "rac1p", + "hisp"], + dtype={"hhid": "int32", + "perid": "int32", + "pnum": "int8", + "age": "int8", + "sex": "int8", + "miltary": "int8", + "pemploy": "int8", + "pstudent": "int8", + "ptype": "int8", + "educ": "int8", + "grade": "int8", + "weeks": "int8", + "hours": "int8", + "rac1p": "int8", + "hisp": "int8"}) + + # load output sampled synthetic person list + fn_person_data = "personData_" + str(self.properties["iterations"]) + ".csv" + output_persons = pd.read_csv( + os.path.join(self.scenario_path, "output", fn_person_data), + usecols=["person_id", + "activity_pattern", + "fp_choice", + "reimb_pct", + "tele_choice"], + dtype={"person_id": "int32", + "activity_pattern": "string", + "fp_choice": "int8", + "reimb_pct": "float32", + "tele_choice": "int8"}) + + # load work-school location model results + fn_ws_loc_results = "wsLocResults_" + str(self.properties["iterations"]) + ".csv" + ws_loc_results = pd.read_csv( + os.path.join(self.scenario_path, "output", fn_ws_loc_results), + usecols=["PersonID", + "HomeMGRA", + "WorkSegment", + "SchoolSegment", + "WorkLocation", + "SchoolLocation"], + dtype={"PersonID": "int32", + "HomeMGRA": "int16", + "WorkSegment": "int32", + "SchoolSegment": "int32", + "WorkLocation": "int32", + "SchoolLocation": "int32"}) + + # merge output sampled persons with input sampled persons + # keep only persons present in the sampled persons + persons = output_persons.merge( + input_persons, + how="inner", + left_on="person_id", + right_on="perid" + ) + + # merge in work-school location model results + persons = persons.merge( + ws_loc_results, + how="inner", + left_on="person_id", + right_on="PersonID" + ) + + # if person works at home set work location to home MGRA + # if person is home-school set school location to home MGRA + persons["WorkLocation"] = np.where( + persons["WorkSegment"] == 99999, + persons["HomeMGRA"], + persons["WorkLocation"]) + + persons["SchoolLocation"] = np.where( + persons["SchoolSegment"] == 88888, + persons["HomeMGRA"], + persons["SchoolLocation"]) + + # apply exhaustive field mappings where applicable + mappings = { + "sex": {1: "Male", + 2: "Female"}, + "miltary": {0: "Not Active Military", + 1: "Active Military"}, + "pemploy": {1: "Employed Full-Time", + 2: "Employed Part-Time", + 3: "Unemployed or Not in Labor Force", + 4: "Less than 16 Years Old"}, + "pstudent": {1: "Pre K-12", + 2: "College Undergrad+Grad and Prof. School", + 3: "Not Attending School"}, + "ptype": {1: "Full-Time Worker", + 2: "Part-Time Worker", + 3: "College Student", + 4: "Non-Working Adult", + 5: "Non-Working Senior", + 6: "Driving Age Student", + 7: "Non-Driving Age Student", + 8: "Pre K or Child too Young for School"}, + "educ": {1: "Not a High School Graduate", + 9: "High School Graduate or Associates Degree", + 13: "Bachelors Degree or Higher"}, + "grade": {0: "Preschool or Not Attending School", + 2: "Kindergarten - Grade 8", + 5: "Grade 9 to Grade 12", + 6: "College Undergraduate or Higher"}, + "weeks": {1: "27 or More Weeks Worked per Year", + 5: "Less than 27 Weeks Worked per Year"}, + "hours": {0: "Less than 35 Hours Worked or Not Working", + 35: "35 or More Hours Worked"}, + "rac1p": {1: "White Alone", + 2: "Black or African American Alone", + 3: "American Indian Alone", + 4: "Alaska Native Alone", + 5: "American Indian and Alaska Native Tribes specified; or American Indian or Alaska Native not specified and no other races", + 6: "Asian Alone", + 7: "Native Hawaiian and Other Pacific Islander Alone", + 8: "Some Other Race Alone", + 9: "Two or More Major Race Groups"}, + "hisp": {1: "Non-Hispanic", + 2: "Hispanic"}, + "activity_pattern": {"H": "Home", + "M": "Mandatory", + "N": "Non-Mandatory"}, + "fp_choice": {1: "Has Free Parking", + 2: "Employer Pays for Parking", + 3: "Employer Reimburses for Parking"}, + "tele_choice": {0: "No telecommute", + 1: "One Day a Week", + 2: "Two-Three Days a Week", + 3: "Four or More Days a Week", + 9: "Telecommuter Only"}, + "WorkSegment": {0: "Management Business Science and Labor", + 1: "Services Labor", + 2: "Sales and Office Labor", + 3: "Natural Resources Construction and Maintenance Labor", + 4: "Production Transportation and Material Moving Labor", + 5: "Military Labor", + 99999: "Work from Home"}, + "SchoolSegment": {**{88888: "Home Schooled"}, + **{key: value for (key, value) in zip(list(range(0, 57)), + ["Unknown"] * 57)} + }, + "WorkLocation": {key: value for (key, value) in zip(list(range(1, 23003)), + list(range(1, 23003)))}, + "SchoolLocation": {key: value for (key, value) in zip(list(range(1, 23003)), + list(range(1, 23003)))} + } + + for field in mappings: + if field in ["WorkLocation", "SchoolLocation"]: + persons[field] = persons[field].map(mappings[field]).astype("float32") + else: + persons[field] = persons[field].map(mappings[field]).astype("category") + + # if employer does not reimburse for parking + # set parking reimbursement percentage to missing + persons.loc[persons["fp_choice"] != "Employer Reimburses for Parking", "reimb_pct"] = np.nan + persons.loc[persons["fp_choice"].isna(), "reimb_pct"] = np.nan + + # rename columns to standard/generic ABM naming conventions + persons.rename(columns={"perid": "personId", + "hhid": "hhId", + "pnum": "personNumber", + "miltary": "militaryStatus", + "pemploy": "employmentStatus", + "pstudent": "studentStatus", + "ptype": "abmPersonType", + "educ": "education", + "rac1p": "race", + "hisp": "hispanic", + "activity_pattern": "abmActivityPattern", + "fp_choice": "freeParkingChoice", + "reimb_pct": "parkingReimbursementPercentage", + "tele_choice": "telecommuteChoice", + "WorkSegment": "workSegment", + "SchoolSegment": "schoolSegment", + "WorkLocation": "workLocation", + "SchoolLocation": "schoolLocation"}, + inplace=True) + + return persons[["personId", + "hhId", + "personNumber", + "age", + "sex", + "militaryStatus", + "employmentStatus", + "studentStatus", + "abmPersonType", + "education", + "grade", + "weeks", + "hours", + "race", + "hispanic", + "abmActivityPattern", + "freeParkingChoice", + "parkingReimbursementPercentage", + "telecommuteChoice", + "workSegment", + "schoolSegment", + "workLocation", + "schoolLocation"]] + + +class TourLists(ScenarioData): + """ A subclass of the ScenarioData class. Holds all tour list data for a + completed ABM scenario model run. This includes all data from the ABM + sub-models with tours. These are held as class properties and include: + Cross Border Model + Commercial Vehicle Model + Internal-External Model + Individual Model + Joint Model + Visitor Model + + The tour list data is loaded from raw ABM output files in the scenario + output folder and transformed where applicable. + + Properties: + cross_border: Mexican Resident Cross Border model tour list + cvm: Commercial Vehicle model tour list + ie: San Diego Resident Internal-External model tour list + individual: San Diego Resident Individual travel model tour list + joint: San Diego Resident Joint travel model tour list + Visitor: Visitor model tour list + """ + @property + @lru_cache(maxsize=1) + def cross_border(self) -> pd.DataFrame: + """ Create the Cross-border Model tour list. + + Read in the Cross-border tour list, map field values, and genericize + field names. + + Returns: + A Pandas DataFrame of the Cross-border tour list """ + + # load tour list into Pandas DataFrame + tours = pd.read_csv( + os.path.join(self.scenario_path, "output", "crossBorderTours.csv"), + usecols=["id", + "purpose", + "sentri", + "poe", + "departTime", + "arriveTime", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"], + dtype={"id": "int32", + "purpose": "int8", + "sentri": "boolean", + "poe": "int8", + "departTime": "int8", + "arriveTime": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tourMode": "int8"}) + + # apply exhaustive field mappings where applicable + mappings = { + "purpose": {0: "Work", + 1: "School", + 2: "Cargo", + 3: "Shop", + 4: "Visit", + 5: "Other"}, + "poe": {0: "San Ysidro", + 1: "Otay Mesa", + 2: "Tecate", + 3: "Otay Mesa East", + 4: "Jacumba"}, + "tourMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk"} + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTime) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTime) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"id": "tourID", + "purpose": "tourPurpose", + "poe": "pointOfEntry", + "departTime": "departTimeAbmHalfHour", + "arriveTime": "arriveTimeAbmHalfHour"}, + inplace=True) + + return tours[["tourID", + "tourPurpose", + "sentri", + "pointOfEntry", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"]] + + @property + @lru_cache(maxsize=1) + def cvm(self) -> pd.DataFrame: + """ Create the Commercial Vehicle Model tour list. + + Read in the Commercial Vehicle trip lists, apply share allocation, map + field values, genericize field names, and create the tour list from the + trip list. + + Returns: + A Pandas DataFrame of the Commercial Vehicle tour list """ + # create list of all Commercial Vehicle model trip list files + # files are of the form Trip_<>_<> + files = ["Trip" + "_" + i + "_" + j + ".csv" for i, j in + itertools.product(["FA", "GO", "IN", "RE", "SV", "TH", "WH"], + ["OE", "AM", "MD", "PM", "OL"])] + + # read all trip list files into a Pandas DataFrame + trips = pd.concat(( + pd.read_csv(os.path.join(self.scenario_path, "output", file), + usecols=["SerialNo", + "Trip", + "ActorType", + "HomeZone", + "Mode", + "StartTime", + "EndTime", + "TourType", + "OriginalTimePeriod"], + dtype={"SerialNo": "int32", + "Trip": "int8", + "ActorType": "string", + "HomeZone": "int16", + "Mode": "string", + "StartTime": "float32", + "EndTime": "float32", + "TourType": "string", + "OriginalTimePeriod": "string"}) + for file in files)) + + # apply re-allocation originally implemented in + # Java by Nagendra Dhakar + Joel Freedman at RSG + + # create lookup table of mode-tod-share using scenario properties + lookup = pd.DataFrame( + {"Mode": ["L"] * 5 + ["I"] * 5 + ["M"] * 5 + ["H"] * 5, + "OriginalTimePeriod": ["OE", "AM", "MD", "PM", "OL"] * 4, + "cvmShare": [self.properties["cvmShareLight"]] * 5 + + [0] * 5 + + [self.properties["cvmShareMedium"]] * 5 + + [self.properties["cvmShareHeavy"]] * 5}) + + # merge trip list and lookup table + trips = trips.merge(lookup) + + # within each mode, the properties file designates a percentage of the + # trip weight to be removed from the original trip and given to a new + # identical trip with the "I" (light-heavy duty truck) mode + new_trips = trips.loc[trips["cvmShare"] > 0].copy() + new_trips.reset_index(drop=True, inplace=True) + new_trips["Mode"] = "I" + trips = pd.concat([trips, new_trips], ignore_index=True) + + # create tour surrogate key + # unique tour is defined by (SerialNo, Mode) + trips["tourID"] = trips.groupby(["SerialNo", "Mode"]).ngroup().astype("int32") + 1 + + # create tour list using the first and last trip within each tour + # all tour data constant across trips excepting start/end times + # first trip provides start time, last trip provides end time + tours = trips.sort_values(by=["tourID", "Trip"]).groupby(["tourID"]) + tours = tours.head(1).merge(tours.tail(1)[["tourID", "EndTime"]], + on="tourID", + suffixes=("_start", "")) + + # apply exhaustive field mappings where applicable + mappings = { + "ActorType": {"FA": "Fleet Allocator", + "GO": "Government\\Office", + "IN": "Industry", + "RE": "Retail", + "SV": "Service", + "TH": "Transport", + "WH": "Wholesale"}, + "Mode": {"L": "Drive Alone", + "I": "Light Heavy Duty Truck", + "M": "Medium Heavy Duty Truck", + "H": "Heavy Heavy Duty Truck"}, + "TourType": {"G": "Goods", + "S": "Service", + "O": "Other"} + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # map continuous start and end times to ABM half hour time periods + # times are in continuous hours of the day (0-24) and can wrap into + # the following day or even multiple following days (>24) with no + # upper limit + + # create times from continuous hour start and end times + # taking into account their wrapping into subsequent days + tours["StartTime"] = tours["StartTime"].apply( + lambda x: (datetime.combine(date.today(), time.min) + + timedelta(hours=(x % 24))).time()) + tours["EndTime"] = tours["EndTime"].apply( + lambda x: (datetime.combine(date.today(), time.min) + + timedelta(hours=(x % 24))).time()) + + # map continuous times to abm half hour periods + depart_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in tours["StartTime"]] + depart_half_hour = [val for sublist in depart_half_hour for val in sublist] + tours = tours.assign(departTimeAbmHalfHour=depart_half_hour) + tours["departTimeAbmHalfHour"] = tours["departTimeAbmHalfHour"].astype("int8") + + arrive_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in tours["EndTime"]] + arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] + tours = tours.assign(arriveTimeAbmHalfHour=arrive_half_hour) + tours["arriveTimeAbmHalfHour"] = tours["arriveTimeAbmHalfHour"].astype("int8") + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTimeAbmHalfHour) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTimeAbmHalfHour) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"ActorType": "actorType", + "TourType": "tourPurpose", + "HomeZone": "originTAZ", + "Mode": "tourMode"}, + inplace=True) + + return tours[["tourID", + "actorType", + "tourPurpose", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originTAZ", + "tourMode"]] + + @property + @lru_cache(maxsize=1) + def ie(self) -> pd.DataFrame: + """ Create the Internal-External Model tour list. + + Read in the Internal-External trip list, map field values, genericize + field names, and create the tour list from the trip list. + + Returns: + A Pandas DataFrame of the Internal-External tour list """ + + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "internalExternalTrips.csv"), + usecols=["personID", + "tourID", + "inbound", + "period", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tripMode"], + dtype={"personID": "int32", + "tourID": "int32", + "inbound": "boolean", + "period": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tripMode": "int8"}) + + # create tour list using the first and last trip within each tour + # all tour data constant across trips excepting start/end times + # first trip provides start time, last trip provides end time + # first trip also provides the tour destination + tours = trips.sort_values(by=["tourID", "inbound"]).groupby(["tourID"]) + tours = tours.head(1).merge(tours.tail(1)[["tourID", "period"]], + on="tourID", + suffixes=("Start", "End")) + + # apply exhaustive field mappings where applicable + mappings = { + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.periodStart) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.periodEnd) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"periodStart": "departTimeAbmHalfHour", + "periodEnd": "arriveTimeAbmHalfHour", + "tripMode": "tourMode"}, + inplace=True) + + return tours[["tourID", + "personID", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"]] + + @property + @lru_cache(maxsize=1) + def individual(self) -> pd.DataFrame: + """ Create the Individual Model tour list. + + Read in the Individual tour list, map field values and genericize + field names. + + Returns: + A Pandas DataFrame of the Individual tour list """ + + # load tour list into Pandas DataFrame + fn = "indivTourData_" + str(self.properties["iterations"]) + ".csv" + tours = pd.read_csv( + os.path.join(self.scenario_path, "output", fn), + usecols=["person_id", + "tour_id", + "tour_category", + "tour_purpose", + "orig_mgra", + "dest_mgra", + "start_period", + "end_period", + "tour_mode"], + dtype={"person_id": "int32", + "tour_id": "int8", + "tour_category": "string", + "tour_purpose": "string", + "orig_mgra": "int16", + "dest_mgra": "int16", + "start_period": "int8", + "end_period": "int8", + "tour_mode": "int8"}) + + # apply exhaustive field mappings where applicable + mappings = { + "tour_category": {"AT_WORK": "At-Work", + "INDIVIDUAL_NON_MANDATORY": "Individual Non-Mandatory", + "MANDATORY": "Mandatory"}, + "tour_mode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC", + 13: "School Bus"} + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # create tour surrogate key (person_id, tour_id, tour_purpose) + tour_key = ["person_id", "tour_id", "tour_purpose"] + tours["tourID"] = tours.groupby(tour_key).ngroup().astype("int32") + 1 + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + tours = tours.merge(taz_info, left_on="orig_mgra", right_on="MGRA") + tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + tours = tours.merge(taz_info, left_on="dest_mgra", right_on="MGRA") + tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.start_period) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.end_period) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"person_id": "personID", + "tour_category": "tourCategory", + "tour_purpose": "tourPurpose", + "start_period": "departTimeAbmHalfHour", + "end_period": "arriveTimeAbmHalfHour", + "orig_mgra": "originMGRA", + "dest_mgra": "destinationMGRA", + "tour_mode": "tourMode"}, + inplace=True) + + return tours[["tourID", + "personID", + "tourCategory", + "tourPurpose", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"]] + + @property + @lru_cache(maxsize=1) + def joint(self) -> pd.DataFrame: + """ Create the Joint Model tour list. + + Read in the Joint tour list, map field values and genericize field + names. + + Returns: + A Pandas DataFrame of the Joint tour list """ + + # load tour list into Pandas DataFrame + fn = "jointTourData_" + str(self.properties["iterations"]) + ".csv" + tours = pd.read_csv( + os.path.join(self.scenario_path, "output", fn), + usecols=["hh_id", + "tour_id", + "tour_category", + "tour_purpose", + "tour_participants", + "orig_mgra", + "dest_mgra", + "start_period", + "end_period", + "tour_mode"], + dtype={"hh_id": "int32", + "tour_id": "int8", + "tour_category": "string", + "tour_purpose": "string", + "tour_participants": "string", + "orig_mgra": "int16", + "dest_mgra": "int16", + "start_period": "int8", + "end_period": "int8", + "tour_mode": "int8"}) + + # apply exhaustive field mappings where applicable + mappings = { + "tour_category": {"JOINT_NON_MANDATORY": "Joint Non-Mandatory"}, + "tour_mode": {2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"} + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # create tour surrogate key (hh_id, tour_id) + tour_key = ["hh_id", "tour_id"] + tours["tourID"] = tours.groupby(tour_key).ngroup().astype("int32") + 1 + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + tours = tours.merge(taz_info, left_on="orig_mgra", right_on="MGRA") + tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + tours = tours.merge(taz_info, left_on="dest_mgra", right_on="MGRA") + tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.start_period) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.end_period) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"hh_id": "hhID", + "tour_category": "tourCategory", + "tour_purpose": "tourPurpose", + "tour_participants": "tourParticipants", + "start_period": "departTimeAbmHalfHour", + "end_period": "arriveTimeAbmHalfHour", + "orig_mgra": "originMGRA", + "dest_mgra": "destinationMGRA", + "tour_mode": "tourMode"}, + inplace=True) + + return tours[["tourID", + "hhID", + "tourParticipants", + "tourCategory", + "tourPurpose", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"]] + + @property + @lru_cache(maxsize=1) + def visitor(self) -> pd.DataFrame: + """ Create the Visitor Model tour list. + + Read in the Visitor tour list, map field values and genericize field + names. + + Returns: + A Pandas DataFrame of the Visitor tour list """ + + # load tour list into Pandas DataFrame + tours = pd.read_csv( + os.path.join(self.scenario_path, "output", "visitorTours.csv"), + usecols=["id", + "segment", + "purpose", + "partySize", + "income", + "departTime", + "arriveTime", + "originMGRA", + "destinationMGRA", + "tourMode"], + dtype={"id": "int32", + "segment": "int8", + "purpose": "int8", + "partySize": "int8", + "income": "int8", + "departTime": "int8", + "arriveTime": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "tourMode": "int8"}) + + # apply exhaustive field mappings where applicable + mappings = { + "segment": {0: "Business", + 1: "Personal"}, + "purpose": {0: "Work", + 1: "Recreation", + 2: "Dining"}, + "income": {0: "Less than 30k", + 1: "30k-60k", + 2: "60k-100k", + 3: "100k-150k", + 4: "150k+"}, + "tourMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"} + } + + for field in mappings: + tours[field] = tours[field].map(mappings[field]).astype("category") + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + tours = tours.merge(taz_info, left_on="originMGRA", right_on="MGRA") + tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + tours = tours.merge(taz_info, left_on="destinationMGRA", right_on="MGRA") + tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTime) + tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTime) + + # rename columns to standard/generic ABM naming conventions + tours.rename(columns={"id": "tourID", + "purpose": "tourPurpose", + "departTime": "departTimeAbmHalfHour", + "arriveTime": "arriveTimeAbmHalfHour"}, + inplace=True) + + return tours[["tourID", + "segment", + "tourPurpose", + "partySize", + "income", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tourMode"]] + + +class TripLists(ScenarioData): + """ A subclass of the ScenarioData class. Holds all trip list data for a + completed ABM scenario model run. This includes all data from the ten ABM + sub-models. These are held as class properties and include: + Airport (CBX) Model + Airport (SAN) Model + Cross Border Model + Commercial Vehicle Model + External-External Model + External-Internal Model + Internal-External Model + Individual Model + Joint Model + Truck Model + Visitor Model + Zombie AV Trips + Zombie TNC Trips + + The trip list data is loaded from raw ABM output files in the scenario + output folder and given consistent fields and field values. + + Methods: + _combine_mode_set: Combines ABM mode field with transit skim set + field + _combine_mode_walk: Recodes ABM mode field using the ABM walk mode + field for walk mode trips + + Properties: + airport_cbx: Cross Border Express (CBX) model trip list + airport_san: San Diego Airport (SAN) model trip list + cross_border: Mexican Resident Cross Border model trip list + cvm: Commercial Vehicle model trip list + ee: External-External model trip list, placed here until it can be + properly incorporated into the EMME data exporter process + ei: External-Internal model trip list, placed here until it can be + properly incorporated into the EMME data exporter process + ie: San Diego Resident Internal-External model trip list + individual: San Diego Resident Individual travel model trip list + joint: San Diego Resident Joint travel model trip list + truck: Truck model trip list, placed here until it can be + properly incorporated into the EMME data exporter process + visitor: Visitor model trip list + zombie_av: 0-passenger Autonomous Vehicle trip list + zombie_tnc: 0-passenger TNC Vehicle trip list + """ + + @staticmethod + def _combine_mode_set(mode: pd.Series, transit_set: pd.Series) -> pd.Series: + """ Combine Pandas Series of ABM mode field values with Pandas Series + of ABM transit skim set field values. + + Returns: + A Pandas Series of the combined mode and transit skim set field + values + """ + + # ensure series are string data type + mode = mode.astype("string") + transit_set = transit_set.astype("string") + + # if ABM mode field value contains the string Transit + # append the transit skim set field value + mode = np.where(mode.str.contains("Transit"), + mode + " - " + transit_set, + mode) + + return pd.Series(mode).astype("category") + + @staticmethod + def _combine_mode_walk(mode: pd.Series, walk_mode: pd.Series) -> pd.Series: + """ Combine Pandas Series of ABM mode field values with Pandas Series + of ABM walk mode (micro_walkMode) field values. Update the ABM mode + field value to the indicated ABM walk mode where appropriate. + + Returns: + A Pandas Series of the recoded ABM mode field values. + """ + + # ensure series are string data type + mode = mode.astype("string") + walk_mode = walk_mode.astype("string") + + # if ABM mode field value is Walk then use the ABM walk mode field + # value as the ABM mode, otherwise use the ABM mode field value + mode = np.where(mode == "Walk", + walk_mode, + mode) + + return pd.Series(mode).astype("category") + + @property + @lru_cache(maxsize=1) + def airport_cbx(self) -> pd.DataFrame: + """ Create the Cross Border Express (CBX) Airport Model trip list. + + Read in the CBX trip list, map field values, and genericize field + names. + + Returns: + A Pandas DataFrame of the CBX trip list """ + + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "airport_out.CBX.csv"), + usecols=["id", + "direction", + "purpose", + "size", + "income", + "nights", + "departTime", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tripMode", + "arrivalMode", + "boardingTAP", + "alightingTAP", + "set", + "valueOfTime"], + dtype={"id": "int32", + "direction": "bool", + "purpose": "int8", + "size": "int8", + "income": "int8", + "nights": "int8", + "departTime": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tripMode": "int8", + "arrivalMode": "int8", + "boardingTAP": "int16", + "alightingTAP": "int16", + "set": "int8", + "valueOfTime": "float32"}) + + # apply exhaustive field mappings where applicable + mappings = { + "purpose": {0: "Resident Business", + 1: "Resident Personal", + 2: "Visitor Business", + 3: "Visitor Personal", + 4: "External"}, + "income": {0: "Less than 25k", + 1: "25k-50k", + 2: "50k-75k", + 3: "75k-100k", + 4: "100k-125k", + 5: "125k-150k", + 6: "150k-200k", + 7: "200k+"}, + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "arrivalMode": {1: "Parking lot terminal", + 2: "Parking lot off-site San Diego airport area", + 3: "Parking lot off-site private", + 4: "Pickup/Drop-off escort", + 5: "Pickup/Drop-off curbside", + 6: "Rental car", + 7: "Taxi", + 8: "Non-Pooled TNC", + 9: "Pooled TNC", + 10: "Shuttle/van/courtesy vehicle", + 11: "Transit"}, + "boardingTAP": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "alightingTAP": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {-1: "", + 0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"} + } + + for field in mappings: + if field in ["boardingTAP", "alightingTAP"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.departTime) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, transit_set=trips.set) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories(vot=trips.valueOfTime) + + # no airport trips use transponders or AVs + # no airport trips are allowed to park in another MGRA + # there is no destination purpose + trips["transponderAvailable"] = False + trips["avUsed"] = False + trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + trips["weightTrip"] = 1 / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + trips["weightPersonTrip"] = pd.Series(trips["size"] / self.properties["sampleRate"], dtype="float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"id": "tripID", + "direction": "inbound", + "purpose": "tripPurpose", + "income": "incomeCategory", + "nights": "nightsStayed", + "departTime": "departTimeAbmHalfHour"}, + inplace=True) + + return trips[["tripID", + "inbound", + "tripPurpose", + "incomeCategory", + "nightsStayed", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "arrivalMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"]] + + @property + @lru_cache(maxsize=1) + def airport_san(self) -> pd.DataFrame: + """ Create the San Diego (SAN) Airport Model trip list. + + Read in the SAN trip list, map field values, and genericize field + names. + + Returns: + A Pandas DataFrame of the SAN trip list """ + + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "airport_out.SAN.csv"), + usecols=["id", + "direction", + "purpose", + "size", + "income", + "nights", + "departTime", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tripMode", + "arrivalMode", + "boardingTAP", + "alightingTAP", + "set", + "valueOfTime"], + dtype={"id": "int32", + "direction": "bool", + "purpose": "int8", + "size": "int8", + "income": "int8", + "nights": "int8", + "departTime": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tripMode": "int8", + "arrivalMode": "int8", + "boardingTAP": "int16", + "alightingTAP": "int16", + "set": "int8", + "valueOfTime": "float32"}) + + # apply exhaustive field mappings where applicable + mappings = { + "purpose": {0: "Resident Business", + 1: "Resident Personal", + 2: "Visitor Business", + 3: "Visitor Personal", + 4: "External"}, + "income": {0: "Less than 25k", + 1: "25k-50k", + 2: "50k-75k", + 3: "75k-100k", + 4: "100k-125k", + 5: "125k-150k", + 6: "150k-200k", + 7: "200k+"}, + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "arrivalMode": {1: "Parking lot terminal", + 2: "Parking lot off-site San Diego airport area", + 3: "Parking lot off-site private", + 4: "Pickup/Drop-off escort", + 5: "Pickup/Drop-off curbside", + 6: "Rental car", + 7: "Taxi", + 8: "Non-Pooled TNC", + 9: "Pooled TNC", + 10: "Shuttle/van/courtesy vehicle", + 11: "Transit"}, + "boardingTAP": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "alightingTAP": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {-1: "", + 0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"} + } + + for field in mappings: + if field in ["boardingTAP", "alightingTAP"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.departTime) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, transit_set=trips.set) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories(vot=trips.valueOfTime) + + # no airport trips use transponders or AVs + # no airport trips are allowed to park in another MGRA + # there is no destination purpose + trips["transponderAvailable"] = False + trips["avUsed"] = False + trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + trips["weightTrip"] = 1 / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + trips["weightPersonTrip"] = pd.Series(trips["size"] / self.properties["sampleRate"], dtype="float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"id": "tripID", + "direction": "inbound", + "purpose": "tripPurpose", + "income": "incomeCategory", + "nights": "nightsStayed", + "departTime": "departTimeAbmHalfHour"}, + inplace=True) + + return trips[["tripID", + "inbound", + "tripPurpose", + "incomeCategory", + "nightsStayed", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "arrivalMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"]] + + @property + @lru_cache(maxsize=1) + def cross_border(self) -> pd.DataFrame: + """ Create the Cross-border Model trip list. + + Read in the Cross-border trip list, map field values, and genericize + field names. + + Returns: + A Pandas DataFrame of the Cross-border trip list """ + + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "crossBorderTrips.csv"), + usecols=["tourID", + "tripID", + "inbound", + "period", + "originPurp", + "destPurp", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tripMode", + "boardingTap", + "alightingTap", + "set", + "valueOfTime", + "parkingCost"], + dtype={"tourID": "int32", + "tripID": "int8", + "inbound": "boolean", + "period": "int8", + "originPurp": "int8", + "destPurp": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tripMode": "int8", + "boardingTap": "int16", + "alightingTap": "int16", + "set": "int8", + "valueOfTime": "float32", + "parkingCost": "float32"}) + + # use the tripID column from the data-set as stopID within the tour + # and create actual tripID field + trips.rename(columns={"tripID": "stopID"}, inplace=True) + trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # apply exhaustive field mappings where applicable + mappings = { + "originPurp": {-1: "Unknown", + 0: "Work", + 1: "School", + 2: "Cargo", + 3: "Shop", + 4: "Visit", + 5: "Other"}, + "destPurp": {-1: "Unknown", + 0: "Work", + 1: "School", + 2: "Cargo", + 3: "Shop", + 4: "Visit", + 5: "Other"}, + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "boardingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "alightingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {-1: "", + 0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"} + } + + for field in mappings: + if field in ["boardingTap", "alightingTap"]: + trips[field] = trips[field].map(mappings[field]).astype("float32") + else: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.period) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, + transit_set=trips.set) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories( + vot=trips.valueOfTime) + + # transform parking cost from cents to dollars + trips["parkingCost"] = round(trips.parkingCost / 100, 2) + + # no cross-border trips use transponders or AVs + # no cross-border trips are allowed to park in another MGRA + trips["transponderAvailable"] = False + trips["avUsed"] = False + trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + conditions = [(trips["tripMode"] == "Shared Ride 2"), + (trips["tripMode"] == "Shared Ride 3+"), + (trips["tripMode"] == "Taxi"), + (trips["tripMode"] == "Non-Pooled TNC"), + (trips["tripMode"] == "Pooled TNC")] + choices = [1 / self.properties["sr2Passengers"], + 1 / self.properties["sr3Passengers"], + 1 / self.properties["taxiPassengers"], + 1 / self.properties["nonPooledTNCPassengers"], + 1 / self.properties["pooledTNCPassengers"]] + + trips["weightTrip"] = pd.Series(np.select(conditions, choices, default=1) / self.properties["sampleRate"], dtype="float32") + trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] + trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"id": "tripID", + "period": "departTimeAbmHalfHour", + "originPurp": "tripPurposeOrigin", + "destPurp": "tripPurposeDestination", + "parkingCost": "costParking", + "boardingTap": "boardingTAP", + "alightingTap": "alightingTAP"}, + inplace=True) + + return trips[["tripID", + "tourID", + "stopID", + "inbound", + "tripPurposeOrigin", + "tripPurposeDestination", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "costParking"]] + + @property + @lru_cache(maxsize=1) + def cvm(self) -> pd.DataFrame: + """ Create the Commercial Vehicle Model trip list. + + Read in the Commercial Vehicle trip list, apply trip scaling and share + allocation, map field values, and genericize field names. + + Returns: + A Pandas DataFrame of the Commercial Vehicle trip list """ + + # create list of all Commercial Vehicle model trip list files + # files are of the form Trip_<>_<> + files = ["Trip" + "_" + i + "_" + j + ".csv" for i, j in + itertools.product(["FA", "GO", "IN", "RE", "SV", "TH", "WH"], + ["OE", "AM", "MD", "PM", "OL"])] + + # read all trip list files into a Pandas DataFrame + trips = pd.concat(( + pd.read_csv(os.path.join(self.scenario_path, "output", file), + usecols=["SerialNo", + "Trip", + "HomeZone", + "ActorType", + "OPurp", + "DPurp", + "I", + "J", + "Mode", + "StartTime", + "EndTime", + "StopDuration", + "TourType", + "OriginalTimePeriod"], + dtype={"SerialNo": "int32", + "Trip": "int8", + "ActorType": "string", + "HomeZone": "int16", + "OPurp": "string", + "DPurp": "string", + "I": "int16", + "J": "int16", + "Mode": "string", + "StartTime": "float32", + "EndTime": "float32", + "StopDuration": "float32", + "TourType": "string", + "OriginalTimePeriod": "string"} + ) + for file in files)) + + # apply weighting and share re-allocation originally implemented in + # Java by Nagendra Dhakar + Joel Freedman at RSG + + # create lookup table of mode-tod-scale-share using scenario properties + lookup = pd.DataFrame( + {"Mode": ["L"] * 5 + ["I"] * 5 + ["M"] * 5 + ["H"] * 5, + "OriginalTimePeriod": ["OE", "AM", "MD", "PM", "OL"] * 4, + "cvmScale": self.properties["cvmScaleLight"] + + self.properties["cvmScaleMedium"] + + self.properties["cvmScaleMedium"] + + self.properties["cvmScaleHeavy"], + "cvmShare": [self.properties["cvmShareLight"]] * 5 + + [0] * 5 + + [self.properties["cvmShareMedium"]] * 5 + + [self.properties["cvmShareHeavy"]] * 5}) + + # merge trip list and lookup table + trips = trips.merge(lookup) + + # within each mode, the properties file designates a percentage of the + # trip weight to be removed from the original trip and given to a new + # identical trip with the "I" (light-heavy duty truck) mode + new_trips = trips.loc[trips["cvmShare"] > 0].copy() + new_trips.reset_index(drop=True, inplace=True) + new_trips["Mode"] = "I" + + # within each mode and tour start abm five time of day period, the + # properties file designates a scaling factor to apply to the trip + # weight taking into account the share factor + new_trips["weightTrip"] = new_trips["cvmScale"] * new_trips["cvmShare"] + trips["weightTrip"] = trips["cvmScale"] * (1 - trips["cvmShare"]) + trips = pd.concat([trips, new_trips], ignore_index=True) + + # apply exhaustive field mappings where applicable + mappings = { + "OPurp": {"Est": "Return to Establishment", + "Gds": "Goods", + "Srv": "Service", + "Oth": "Other"}, + "DPurp": {"Est": "Return to Establishment", + "Gds": "Goods", + "Srv": "Service", + "Oth": "Other"}, + "Mode": {"L": "Drive Alone", + "I": "Light Heavy Duty Truck", + "M": "Medium Heavy Duty Truck", + "H": "Heavy Heavy Duty Truck"}, + } + + for field in mappings: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # create tour and trip surrogate keys + # unique tour is defined by (SerialNo, Mode) + # unique trip is defined by (SerialNo, Mode, Trip) + trips["tourID"] = trips.groupby(["SerialNo", "Mode"]).ngroup().astype("int32") + 1 + trips = trips.sort_values(by=["SerialNo", "Mode", "Trip"]).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # map continuous start and end times to ABM half hour time periods + # times are in continuous hours of the day (0-24) and can wrap into + # the following day or even multiple following days (>24) with no + # upper limit + + # create times from continuous hour start and end times + # taking into account their wrapping into subsequent days + trips["StartTime"] = trips["StartTime"].apply( + lambda x: (datetime.combine(date.today(), time.min) + + timedelta(hours=(x % 24))).time()) + trips["EndTime"] = trips["EndTime"].apply( + lambda x: (datetime.combine(date.today(), time.min) + + timedelta(hours=(x % 24))).time()) + + # map continuous times to abm half hour periods + depart_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in trips["StartTime"]] + depart_half_hour = [val for sublist in depart_half_hour for val in sublist] + trips = trips.assign(departTimeAbmHalfHour=depart_half_hour) + trips["departTimeAbmHalfHour"] = trips["departTimeAbmHalfHour"].astype("int8") + + arrive_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in trips["EndTime"]] + arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] + trips = trips.assign(arriveTimeAbmHalfHour=arrive_half_hour) + trips["arriveTimeAbmHalfHour"] = trips["arriveTimeAbmHalfHour"].astype("int8") + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.departTimeAbmHalfHour) + + trips["arriveTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.arriveTimeAbmHalfHour) + + # all cvm trips are high value of time + # only Drive Alone cvm trips use transponders + # no cvm trips use AVs + # no cvm trips are allowed to park in another MGRA + trips["valueOfTimeCategory"] = "High" + trips["valueOfTimeCategory"] = trips["valueOfTimeCategory"].astype("category") + trips["transponderAvailable"] = np.where(trips["Mode"] == "Drive Alone", True, False) + trips["avUsed"] = False + trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") + + # add person-based weight and adjust weights + # by the ABM scenario final iteration sample rate + trips["weightTrip"] = pd.Series(trips["weightTrip"] / self.properties["sampleRate"], dtype="float32") + trips["weightPersonTrip"] = trips["weightTrip"] + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"Trip": "stopID", + "OPurp": "tripPurposeOrigin", + "DPurp": "tripPurposeDestination", + "I": "originTAZ", + "J": "destinationTAZ", + "Mode": "tripMode", + "StopDuration": "stopDuration"}, + inplace=True) + + return trips[["tripID", + "tourID", + "stopID", + "tripPurposeOrigin", + "tripPurposeDestination", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "stopDuration", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"]] + + @property + @lru_cache(maxsize=1) + def ee(self) -> pd.DataFrame: + """ Create the External-External Model trip list. + + Read in the External-External trip last, map field values, and + genericize field names. + + Returns: + A Pandas DataFrame of the External-External trips list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "report", "eetrip.csv"), + usecols=["OTAZ", + "DTAZ", + "TOD", + "MODE", + "TRIPS", + "TIME", + "DIST", + "AOC", + "TOLLCOST"], + dtype={"OTAZ": "int16", + "DTAZ": "int16", + "TOD": "string", + "MODE": "string", + "TRIPS": "float32", + "TIME": "float32", + "DIST": "float32", + "AOC": "float32", + "TOLLCOST": "float32"}) + + # expand trip list by 3x + # divide [TRIPS] field by 3 + # assign each copy of trip list to each value of time category + trips_low = trips.copy() + trips_low["valueOfTimeCategory"] = "Low" + + trips_med = trips.copy() + trips_med["valueOfTimeCategory"] = "Medium" + + trips_high = trips.copy() + trips_high["valueOfTimeCategory"] = "High" + + trips = pd.concat([trips_low, trips_med, trips_high], ignore_index=True) + + trips["TRIPS"] = trips["TRIPS"] / 3 + + # create trip surrogate key + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # apply exhaustive field mappings where applicable + mappings = { + "TOD": {"EA": 1, + "AM": 2, + "MD": 3, + "PM": 4, + "EV": 5}, + "MODE": {"DA": "Drive Alone", + "S2": "Shared Ride 2", + "S3": "Shared Ride 3+"} + } + + for field in mappings: + if field == "TOD": + trips[field] = trips[field].map(mappings[field]).astype("int8") + else: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # convert cents-based cost fields to dollars + trips["AOC"] = trips["AOC"] / 100 + trips["TOLLCOST"] = trips["TOLLCOST"] / 100 + + # no trips use transponders or autonomous vehicles + trips["transponderAvailable"] = False + trips["avUsed"] = False + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + conditions = [(trips["MODE"] == "Shared Ride 2"), + (trips["MODE"] == "Shared Ride 3+")] + choices = [self.properties["sr2Passengers"], + self.properties["sr3Passengers"]] + + trips["weightPersonTrip"] = pd.Series( + trips["TRIPS"] * np.select(conditions, choices, default=1) / self.properties["sampleRate"], + dtype="float32") + trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"OTAZ": "originTAZ", + "DTAZ": "destinationTAZ", + "TOD": "departTimeFiveTod", + "MODE": "tripMode", + "TIME": "timeDrive", + "DIST": "distanceDrive", + "AOC": "costOperatingDrive", + "TOLLCOST": "costTollDrive"}, + inplace=True) + + # create total time/distance/cost columns + trips["timeTotal"] = trips["timeDrive"] + trips["distanceTotal"] = trips["distanceDrive"] + trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] + + return trips[["tripID", + "departTimeFiveTod", + "originTAZ", + "destinationTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "timeDrive", + "distanceDrive", + "costTollDrive", + "costOperatingDrive", + "timeTotal", + "distanceTotal", + "costTotal"]] + + @property + @lru_cache(maxsize=1) + def ei(self) -> pd.DataFrame: + """ Create the External-Internal Model trip list. + + Read in the External-Internal trip last, map field values, and + genericize field names. + + Returns: + A Pandas DataFrame of the External-Internal trips list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "report", "eitrip.csv"), + usecols=["OTAZ", + "DTAZ", + "TOD", + "MODE", + "PURPOSE", + "TRIPS", + "TIME", + "DIST", + "AOC", + "TOLLCOST"], + dtype={"OTAZ": "int16", + "DTAZ": "int16", + "TOD": "string", + "MODE": "string", + "PURPOSE": "string", + "TRIPS": "float32", + "TIME": "float32", + "DIST": "float32", + "AOC": "float32", + "TOLLCOST": "float32"}) + + # expand trip list by 3x + # divide [TRIPS] field by 3 + # assign each copy of trip list to each value of time category + trips_low = trips.copy() + trips_low["valueOfTimeCategory"] = "Low" + + trips_med = trips.copy() + trips_med["valueOfTimeCategory"] = "Medium" + + trips_high = trips.copy() + trips_high["valueOfTimeCategory"] = "High" + + trips = pd.concat([trips_low, trips_med, trips_high], ignore_index=True) + + trips["TRIPS"] = trips["TRIPS"] / 3 + + # create trip surrogate key + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # apply exhaustive field mappings where applicable + mappings = { + "TOD": {"EA": 1, + "AM": 2, + "MD": 3, + "PM": 4, + "EV": 5}, + "MODE": {"DAN": "Drive Alone", + "DAT": "Drive Alone", + "S2N": "Shared Ride 2", + "S2T": "Shared Ride 2", + "S3N": "Shared Ride 3+", + "S3T": "Shared Ride 3+"}, + "PURPOSE": {"NONWORK": "Non-Work", + "WORK": "Work"} + } + + for field in mappings: + if field == "TOD": + trips[field] = trips[field].map(mappings[field]).astype("int8") + else: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # convert cents-based cost fields to dollars + trips["AOC"] = trips["AOC"] / 100 + trips["TOLLCOST"] = trips["TOLLCOST"] / 100 + + # no trips use transponders or autonomous vehicles + trips["transponderAvailable"] = False + trips["avUsed"] = False + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + conditions = [(trips["MODE"] == "Shared Ride 2"), + (trips["MODE"] == "Shared Ride 3+")] + choices = [self.properties["sr2Passengers"], + self.properties["sr3Passengers"]] + + trips["weightPersonTrip"] = pd.Series( + trips["TRIPS"] * np.select(conditions, choices, default=1) / self.properties["sampleRate"], + dtype="float32") + trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"OTAZ": "originTAZ", + "DTAZ": "destinationTAZ", + "TOD": "departTimeFiveTod", + "MODE": "tripMode", + "PURPOSE": "tripPurpose", + "TIME": "timeDrive", + "DIST": "distanceDrive", + "AOC": "costOperatingDrive", + "TOLLCOST": "costTollDrive"}, + inplace=True) + + # create total time/distance/cost columns + trips["timeTotal"] = trips["timeDrive"] + trips["distanceTotal"] = trips["distanceDrive"] + trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] + + return trips[["tripID", + "departTimeFiveTod", + "originTAZ", + "destinationTAZ", + "tripMode", + "tripPurpose", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "timeDrive", + "distanceDrive", + "costTollDrive", + "costOperatingDrive", + "timeTotal", + "distanceTotal", + "costTotal"]] + + @property + @lru_cache(maxsize=1) + def ie(self) -> pd.DataFrame: + """ Create the Internal-External Model trip list. + + Read in the Internal-External trip list, map field values, + and genericize field names. + + Returns: + A Pandas DataFrame of the Internal-External trip list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "internalExternalTrips.csv"), + usecols=["hhID", + "personID", + "tourID", + "inbound", + "period", + "originMGRA", + "destinationMGRA", + "originTAZ", + "destinationTAZ", + "tripMode", + "av_avail", + "boardingTap", + "alightingTap", + "set", + "valueOfTime"], + dtype={"hhID": "int32", + "personID": "int32", + "tourID": "int32", + "inbound": "boolean", + "period": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "originTAZ": "int16", + "destinationTAZ": "int16", + "tripMode": "int8", + "av_avail": "bool", + "boardingTap": "int16", + "alightingTap": "int16", + "set": "int8", + "valueOfTime": "float32"}) + + # load output household transponder ownership data + hh_fn = "householdData_" + str(self.properties["iterations"]) + ".csv" + hh = pd.read_csv( + os.path.join(self.scenario_path, "output", hh_fn), + usecols=["hh_id", + "transponder"], + dtype={"hh_id": "int32", + "transponder": "bool"}) + + # if household has a transponder then all trips can use it + trips = trips.merge(hh, left_on="hhID", right_on="hh_id") + + # apply exhaustive field mappings where applicable + mappings = { + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "boardingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "alightingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {-1: "", + 0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"} + } + + for field in mappings: + if field in ["boardingTap", "alightingTap"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # create trip surrogate key + # create stop surrogate key + # every tourID contains only two trips (outbound and inbound) + trips["stopID"] = trips.sort_values(by=["tourID", "inbound"]).groupby(["tourID"]).cumcount().astype("int8") + 1 + trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.period + ) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set( + mode=trips.tripMode, + transit_set=trips.set + ) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories( + vot=trips.valueOfTime + ) + + # no internal-external trips are allowed to park in another MGRA + trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + conditions = [(trips["tripMode"] == "Shared Ride 2"), + (trips["tripMode"] == "Shared Ride 3+"), + (trips["tripMode"] == "Taxi"), + (trips["tripMode"] == "Non-Pooled TNC"), + (trips["tripMode"] == "Pooled TNC")] + choices = [1 / self.properties["sr2Passengers"], + 1 / self.properties["sr3Passengers"], + 1 / self.properties["taxiPassengers"], + 1 / self.properties["nonPooledTNCPassengers"], + 1 / self.properties["pooledTNCPassengers"]] + + trips["weightTrip"] = pd.Series( + np.select(conditions, choices, default=1) / self.properties["sampleRate"], + dtype="float32") + trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] + trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"period": "departTimeAbmHalfHour", + "av_avail": "avUsed", + "boardingTap": "boardingTAP", + "alightingTap": "alightingTAP", + "transponder": "transponderAvailable"}, + inplace=True) + + return trips[["tripID", + "personID", + "tourID", + "stopID", + "inbound", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"]] + + @property + @lru_cache(maxsize=1) + def individual(self) -> pd.DataFrame: + """ Create the Individual Model trip list. + + Read in the Individual trip list, map field values, and genericize + field names. + + Returns: + A Pandas DataFrame of the Individual trip list """ + # load trip list into Pandas DataFrame + fn = "indivTripData_" + str(self.properties["iterations"]) + ".csv" + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", fn), + usecols=["person_id", + "tour_id", + "stop_id", + "inbound", + "tour_purpose", + "orig_purpose", + "dest_purpose", + "orig_mgra", + "dest_mgra", + "parking_mgra", + "stop_period", + "trip_mode", + "av_avail", + "trip_board_tap", + "trip_alight_tap", + "set", + "valueOfTime", + "transponder_avail", + "micro_walkMode", + "micro_trnAcc", + "micro_trnEgr", + "parkingCost"], + dtype={"person_id": "int32", + "tour_id": "int8", + "stop_id": "int8", + "inbound": "bool", + "tour_purpose": "string", + "orig_purpose": "string", + "dest_purpose": "string", + "orig_mgra": "int16", + "dest_mgra": "int16", + "parking_mgra": "int16", + "stop_period": "int8", + "trip_mode": "int8", + "av_avail": "bool", + "trip_board_tap": "int16", + "trip_alight_tap": "int16", + "set": "int8", + "valueOfTime": "float32", + "transponder_avail": "bool", + "micro_walkMode": "int8", + "micro_trnAcc": "int8", + "micro_trnEgr": "int8", + "parkingCost": "float32"}) + + # apply exhaustive field mappings where applicable + mappings = { + "parking_mgra": {key: value for (key, value) in + zip(list(range(1, 23003)), + list(range(1, 23003)))}, + "trip_mode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC", + 13: "School Bus"}, + "trip_board_tap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "trip_alight_tap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"}, + "micro_walkMode": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"}, + "micro_trnAcc": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"}, + "micro_trnEgr": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"} + } + + for field in mappings: + if field in ["parking_mgra", "trip_board_tap", "trip_alight_tap"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # create tour surrogate key (person_id, tour_id, tour_purpose) + tour_key = ["person_id", "tour_id", "tour_purpose"] + trips["tourID"] = pd.Series(trips.groupby(tour_key).ngroup() + 1, dtype="int32") + + # create tour stop surrogate key (inbound, stop_id) + stop_key = ["inbound", "stop_id"] + trips["stopID"] = pd.Series(trips.sort_values(by=stop_key).groupby(tour_key).cumcount() + 1, dtype="int8") + + # create unique trip surrogate key + trips = trips.sort_values(by=tour_key + stop_key).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + trips = trips.merge(taz_info, left_on="dest_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + trips = trips.merge(taz_info, how="left", left_on="parking_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "parkingTAZ"}, inplace=True) + trips["parkingTAZ"] = trips["parkingTAZ"].astype("float32") + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.stop_period + ) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set( + mode=trips.trip_mode, + transit_set=trips.set + ) + + # set appropriate walk mode for walk trips + trips["tripMode"] = self._combine_mode_walk( + mode=trips.tripMode, + walk_mode=trips.micro_walkMode + ) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories( + vot=trips.valueOfTime + ) + + # transform parking cost from cents to dollars + trips["parkingCost"] = round(trips.parkingCost / 100, 2) + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + conditions = [(trips["tripMode"] == "Shared Ride 2"), + (trips["tripMode"] == "Shared Ride 3+"), + (trips["tripMode"] == "Taxi"), + (trips["tripMode"] == "Non-Pooled TNC"), + (trips["tripMode"] == "Pooled TNC")] + choices = [1 / self.properties["sr2Passengers"], + 1 / self.properties["sr3Passengers"], + 1 / self.properties["taxiPassengers"], + 1 / self.properties["nonPooledTNCPassengers"], + 1 / self.properties["pooledTNCPassengers"]] + + trips["weightTrip"] = pd.Series(np.select(conditions, choices, default=1) / self.properties["sampleRate"], dtype="float32") + trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] + trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"person_id": "personID", + "orig_purpose": "tripPurposeOrigin", + "dest_purpose": "tripPurposeDestination", + "stop_period": "departTimeAbmHalfHour", + "orig_mgra": "originMGRA", + "dest_mgra": "destinationMGRA", + "parking_mgra": "parkingMGRA", + "parkingCost": "costParking", + "av_avail": "avUsed", + "trip_board_tap": "boardingTAP", + "trip_alight_tap": "alightingTAP", + "transponder_avail": "transponderAvailable", + "micro_trnAcc": "microMobilityTransitAccess", + "micro_trnEgr": "microMobilityTransitEgress"}, + inplace=True) + + return trips[["tripID", + "personID", + "tourID", + "stopID", + "inbound", + "tripPurposeOrigin", + "tripPurposeDestination", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "microMobilityTransitAccess", + "microMobilityTransitEgress", + "weightTrip", + "weightPersonTrip", + "costParking"]] + + @property + @lru_cache(maxsize=1) + def joint(self) -> pd.DataFrame: + """ Create the Joint Model trip list. + + Read in the Joint trip list, map field values, genericize field + names, append skim values, replicate data-set records for each trip + participant creating data-set format of one record per participant, + and assign trip weights accounting for replicated records. + + Returns: + A Pandas DataFrame of the Joint trip list """ + # load trip list into Pandas DataFrame + fn_trips = "jointTripData_" + str(self.properties["iterations"]) + ".csv" + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", fn_trips), + usecols=["hh_id", + "tour_id", + "stop_id", + "inbound", + "orig_purpose", + "dest_purpose", + "orig_mgra", + "dest_mgra", + "parking_mgra", + "stop_period", + "trip_mode", + "av_avail", + "num_participants", + "trip_board_tap", + "trip_alight_tap", + "set", + "valueOfTime", + "transponder_avail", + "parkingCost"], + dtype={"hh_id": "int32", + "tour_id": "int8", + "stop_id": "int8", + "inbound": "bool", + "orig_purpose": "string", + "dest_purpose": "string", + "orig_mgra": "int16", + "dest_mgra": "int16", + "parking_mgra": "int16", + "stop_period": "int8", + "trip_mode": "int8", + "av_avail": "bool", + "num_participants": "int8", + "trip_board_tap": "int16", + "trip_alight_tap": "int16", + "set": "int8", + "valueOfTime": "float32", + "transponder_avail": "bool", + "parkingCost": "float32"}) + + # apply exhaustive field mappings where applicable + mappings = { + "parking_mgra": {key: value for (key, value) in + zip(list(range(1, 23003)), + list(range(1, 23003)))}, + "trip_mode": {2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "trip_board_tap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "trip_alight_tap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"}, + } + + for field in mappings: + if field in ["parking_mgra", "trip_board_tap", "trip_alight_tap"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # create tour surrogate key (hh_id, tour_id) + tour_key = ["hh_id", "tour_id"] + trips["tourID"] = pd.Series( + trips.groupby(tour_key).ngroup() + 1, + dtype="int32") + + # create tour stop surrogate key (inbound, stop_id) + stop_key = ["inbound", "stop_id"] + trips["stopID"] = pd.Series( + trips.sort_values(by=stop_key).groupby(tour_key).cumcount() + 1, + dtype="int8") + + # create unique trip surrogate key + trips = trips.sort_values(by=tour_key + stop_key).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + trips = trips.merge(taz_info, left_on="dest_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + trips = trips.merge(taz_info, how="left", left_on="parking_mgra", + right_on="MGRA") + trips.rename(columns={"TAZ": "parkingTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.stop_period + ) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set( + mode=trips.trip_mode, + transit_set=trips.set + ) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories( + vot=trips.valueOfTime + ) + + # transform parking cost from cents to dollars + trips["parkingCost"] = round(trips.parkingCost / 100, 2) + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"orig_purpose": "tripPurposeOrigin", + "dest_purpose": "tripPurposeDestination", + "stop_period": "departTimeAbmHalfHour", + "orig_mgra": "originMGRA", + "dest_mgra": "destinationMGRA", + "parking_mgra": "parkingMGRA", + "parkingCost": "costParking", + "av_avail": "avUsed", + "trip_board_tap": "boardingTAP", + "trip_alight_tap": "alightingTAP", + "transponder_avail": "transponderAvailable"}, + inplace=True) + + # load tour list into Pandas DataFrame + fn_tours = "jointTourData_" + str( + self.properties["iterations"]) + ".csv" + tours = pd.read_csv( + os.path.join(self.scenario_path, "output", fn_tours), + usecols=["hh_id", + "tour_id", + "tour_participants"], + dtype={"hh_id": "int32", + "tour_id": "int8", + "tour_participants": "string"}) + + # split the tour participants column by " " and append in wide-format + # to each record + tours = pd.concat( + [tours[["hh_id", "tour_id"]], + tours["tour_participants"].str.split(" ", expand=True)], + axis=1 + ) + + # melt the wide-format tour participants to long-format + tours = pd.melt(tours, id_vars=["hh_id", "tour_id"], + value_name="person_num") + tours = tours[tours["person_num"].notnull()] + tours["person_num"] = tours["person_num"].astype("int8") + + # load output person data into Pandas DataFrame + fn_persons = "personData_" + str(self.properties["iterations"]) + ".csv" + persons = pd.read_csv( + os.path.join(self.scenario_path, "output", fn_persons), + usecols=["hh_id", + "person_num", + "person_id"], + dtype={"hh_id": "int32", + "person_num": "int8", + "person_id": "int32"}) + persons.rename(columns={"person_id": "personID"}, inplace=True) + + # merge persons with the long-format tour participants to get the person id + tours = tours.merge(persons, on=["hh_id", "person_num"]) + + # merge long-format tour participants with the trip list + # this many-to-one merge replicates trip records for each participant + # as well as appending the person id to each replicated record + trips = trips.merge(tours, on=["hh_id", "tour_id"]) + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + # each record is per-person on trip + # data-set has single person record with multiple trip records + trips["weightTrip"] = pd.Series( + 1 / (trips["num_participants"] * self.properties["sampleRate"]), + dtype="float32") + trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] + trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") + + return trips[["tripID", + "personID", + "tourID", + "stopID", + "inbound", + "tripPurposeOrigin", + "tripPurposeDestination", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "costParking"]] + + @property + @lru_cache(maxsize=1) + def truck(self) -> pd.DataFrame: + """ Create the Truck Model trip list. + + Read in the External-External trip last, map field values, and + genericize field names. + + Returns: + A Pandas DataFrame of the External-External trips list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "report", "trucktrip.csv"), + usecols=["OTAZ", + "DTAZ", + "TOD", + "MODE", + "TRIPS", + "TIME", + "DIST", + "AOC", + "TOLLCOST"], + dtype={"OTAZ": "int16", + "DTAZ": "int16", + "TOD": "string", + "MODE": "string", + "TRIPS": "float32", + "TIME": "float32", + "DIST": "float32", + "AOC": "float32", + "TOLLCOST": "float32"}) + + # create trip surrogate key + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # apply exhaustive field mappings where applicable + mappings = { + "TOD": {"EA": 1, + "AM": 2, + "MD": 3, + "PM": 4, + "EV": 5}, + "MODE": {"lhdn": "Light Heavy Duty Truck", + "lhdt": "Light Heavy Duty Truck", + "mhdn": "Medium Heavy Duty Truck", + "mhdt": "Medium Heavy Duty Truck", + "hhdn": "Heavy Heavy Duty Truck", + "hhdt": "Heavy Heavy Duty Truck"} + } + + for field in mappings: + if field == "TOD": + trips[field] = trips[field].map(mappings[field]).astype("int8") + else: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # convert cents-based cost fields to dollars + trips["AOC"] = trips["AOC"] / 100 + trips["TOLLCOST"] = trips["TOLLCOST"] / 100 + + # all trips are High value of time + # no trips use transponders or autonomous vehicles + trips["valueOfTimeCategory"] = "High" + trips["transponderAvailable"] = False + trips["avUsed"] = False + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] + trips["weightPersonTrip"] = trips["TRIPS"] / self.properties["sampleRate"] + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"OTAZ": "originTAZ", + "DTAZ": "destinationTAZ", + "TOD": "departTimeFiveTod", + "MODE": "tripMode", + "TIME": "timeDrive", + "DIST": "distanceDrive", + "AOC": "costOperatingDrive", + "TOLLCOST": "costTollDrive"}, + inplace=True) + + # create total time/distance/cost columns + trips["timeTotal"] = trips["timeDrive"] + trips["distanceTotal"] = trips["distanceDrive"] + trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] + + return trips[["tripID", + "departTimeFiveTod", + "originTAZ", + "destinationTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "timeDrive", + "distanceDrive", + "costTollDrive", + "costOperatingDrive", + "timeTotal", + "distanceTotal", + "costTotal"]] + + @property + @lru_cache(maxsize=1) + def visitor(self) -> pd.DataFrame: + """ Create the Visitor Model trip list. + + Read in the Visitor trip list, map field values, and genericize + field names. + + Returns: + A Pandas DataFrame of the Visitor trip list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "visitorTrips.csv"), + usecols=["tourID", + "tripID", + "originPurp", + "destPurp", + "originMGRA", + "destinationMGRA", + "inbound", + "period", + "tripMode", + "avAvailable", + "boardingTap", + "alightingTap", + "set", + "valueOfTime", + "partySize", + "micro_walkMode", + "micro_trnAcc", + "micro_trnEgr", + "parkingCost"], + dtype={"tourID": "int32", + "tripID": "int8", + "originPurp": "int8", + "destPurp": "int8", + "originMGRA": "int16", + "destinationMGRA": "int16", + "inbound": "boolean", + "period": "int8", + "tripMode": "int8", + "avAvailable": "bool", + "boardingTap": "int16", + "alightingTap": "int16", + "set": "int8", + "valueOfTime": "float32", + "partySize": "int8", + "micro_walkMode": "int8", + "micro_trnAcc": "int8", + "micro_trnEgr": "int8", + "parkingCost": "float32"}) + + # apply exhaustive field mappings where applicable + mappings = { + "originPurp": {-1: "Unknown", + 0: "Work", + 1: "Recreation", + 2: "Dining"}, + "destPurp": {-1: "Unknown", + 0: "Work", + 1: "Recreation", + 2: "Dining"}, + "tripMode": {1: "Drive Alone", + 2: "Shared Ride 2", + 3: "Shared Ride 3+", + 4: "Walk", + 5: "Bike", + 6: "Walk to Transit", + 7: "Park and Ride to Transit", + 8: "Kiss and Ride to Transit", + 9: "TNC to Transit", + 10: "Taxi", + 11: "Non-Pooled TNC", + 12: "Pooled TNC"}, + "boardingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "alightingTap": {key: value for (key, value) in + zip(list(range(1, 99999)), + list(range(1, 99999)))}, + "set": {0: "Local Bus", + 1: "Premium Transit", + 2: "Local Bus and Premium Transit"}, + "micro_walkMode": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"}, + "micro_trnAcc": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"}, + "micro_trnEgr": {1: "Walk", + 2: "Micro-Mobility", + 3: "Micro-Transit"} + } + + for field in mappings: + if field in ["boardingTap", "alightingTap"]: + data_type = "float32" + else: + data_type = "category" + trips[field] = trips[field].map(mappings[field]).astype(data_type) + + # create unique trip surrogate key + # the tripID field included in the data-set is a stopID + trips.rename(columns={"tripID": "stopID"}, inplace=True) + trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + trips = trips.merge(taz_info, left_on="originMGRA", right_on="MGRA") + trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + trips = trips.merge(taz_info, left_on="destinationMGRA", right_on="MGRA") + trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.period + ) + + # concatenate mode and transit skim set for transit trips + trips["tripMode"] = self._combine_mode_set( + mode=trips.tripMode, + transit_set=trips.set + ) + + # set appropriate walk mode for walk trips + trips["tripMode"] = self._combine_mode_walk( + mode=trips.tripMode, + walk_mode=trips.micro_walkMode + ) + + # calculate value of time category auto skim set used + trips["valueOfTimeCategory"] = self._map_vot_categories( + vot=trips.valueOfTime + ) + + # transform parking cost from cents to dollars + trips["parkingCost"] = round(trips.parkingCost / 100, 2) + + # no visitor trips use transponders + # no visitor trips are allowed to park in another MGRA + trips["transponderAvailable"] = False + trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") + + # add vehicle/trip-based weight and person-based weight + # adjust by the ABM scenario final iteration sample rate + trips["weightTrip"] = 1 / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + trips["weightPersonTrip"] = pd.Series( + trips["partySize"] / self.properties["sampleRate"], + dtype="float32") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"originPurp": "tripPurposeOrigin", + "destPurp": "tripPurposeDestination", + "period": "departTimeAbmHalfHour", + "parkingCost": "costParking", + "avAvailable": "avUsed", + "boardingTap": "boardingTAP", + "alightingTap": "alightingTAP", + "micro_trnAcc": "microMobilityTransitAccess", + "micro_trnEgr": "microMobilityTransitEgress"}, + inplace=True) + + return trips[["tripID", + "tourID", + "stopID", + "inbound", + "tripPurposeOrigin", + "tripPurposeDestination", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "boardingTAP", + "alightingTAP", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "microMobilityTransitAccess", + "microMobilityTransitEgress", + "weightTrip", + "weightPersonTrip", + "costParking"]] + + @property + @lru_cache(maxsize=1) + def zombie_av(self) -> pd.DataFrame: + """ Create the 0-Passenger Autonomous Vehicle trip list. + + Read in the Autonomous Vehicle trip list, select only 0-passenger + trips, map field values, and genericize field names. + + Returns: + A Pandas DataFrame of the 0-Passenger Autonomous Vehicle trip + list """ + fn = os.path.join(self.scenario_path, "output", "householdAVTrips.csv") + # file does not exist if AV-component of model is turned off + if os.path.isfile(fn): + # load trip list into Pandas DataFrame + trips = pd.read_csv( + fn, + usecols=["hh_id", + "veh_id", + "vehicleTrip_id", + "orig_mgra", + "dest_gra", + "period", + "occupants", + "originIsHome", + "destinationIsHome", + "originIsRemoteParking", + "destinationIsRemoteParking", + "remoteParkingCostAtDest"], + dtype={"hh_id": "int32", + "veh_id": "int32", + "vehicleTrip_id": "int32", + "orig_mgra": "int32", + "dest_gra": "int32", + "period": "int32", + "occupants": "int32", + "originIsHome": "bool", + "destinationIsHome": "bool", + "originIsRemoteParking": "bool", + "destinationIsRemoteParking": "bool", + "remoteParkingCostAtDest": "float32"} + ) + + # filter trip list to empty/zombie av trips + trips = trips.loc[trips["occupants"] == 0].copy() + + # create unique trip surrogate key + trip_key = ["hh_id", "veh_id", "vehicleTrip_id"] + trips = trips.sort_values(by=trip_key).reset_index(drop=True) + trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") + + # load output household transponder ownership data + hh_fn = "householdData_" + str(self.properties["iterations"]) + ".csv" + hh = pd.read_csv( + os.path.join(self.scenario_path, "output", hh_fn), + usecols=["hh_id", "transponder"], + dtype={"hh_id": "int32", + "transponder": "bool"}) + + # if household has a transponder then all trips can use it + trips = trips.merge(hh, on="hh_id") + + # add TAZ information in addition to MGRA information + taz_info = self.mgra_xref[["MGRA", "TAZ"]] + + trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") + trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) + + trips = trips.merge(taz_info, left_on="dest_gra", right_on="MGRA") + trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.period) + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.period) + + # all zombie AV trips are Drive Alone and High vot + trips["tripMode"] = "Drive Alone" + trips["valueOfTimeCategory"] = "High" + trips["avUsed"] = True + trips["parkingMGRA"] = np.nan + trips["parkingTAZ"] = np.nan + + # add person-based weight and adjust weights + # by the ABM scenario final iteration sample rate + # no people are in zombie AV trips + trips["weightTrip"] = 1 / self.properties["sampleRate"] + trips["weightPersonTrip"] = 0 + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"hh_id": "hhID", + "veh_id": "vehID", + "vehicleTrip_id": "vehicleTripID", + "orig_mgra": "originMGRA", + "dest_gra": "destinationMGRA", + "period": "departTimeAbmHalfHour", + "transponder": "transponderAvailable", + "remoteParkingCostAtDest": "costParking"}, + inplace=True) + + return trips[["tripID", + "hhID", + "vehID", + "vehicleTripID", + "originIsHome", + "destinationIsHome", + "originIsRemoteParking", + "destinationIsRemoteParking", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip", + "costParking"]] + else: # return empty DataFrame if file does not exist + return(pd.DataFrame( + columns=["tripID", + "hhID", + "vehID", + "vehicleTripID", + "originIsHome", + "destinationIsHome", + "originIsRemoteParking", + "destinationIsRemoteParking", + "parkingChoiceAtDestination", + "departTimeAbmHalfHour", + "departTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"])) + + @property + @lru_cache(maxsize=1) + def zombie_tnc(self) -> pd.DataFrame: + """ Create the 0-Passenger TNC trip list. + + Read in the TNC Vehicle trip list, select only 0-passenger + trips, map field values, and genericize field names. + + Returns: + A Pandas DataFrame of the 0-Passenger TNC Vehicle trip list """ + # load trip list into Pandas DataFrame + trips = pd.read_csv( + os.path.join(self.scenario_path, "output", "TNCTrips.csv"), + usecols=["trip_ID", + "originMgra", + "destinationMgra", + "originTaz", + "destinationTaz", + "totalPassengers", + "startPeriod", + "endPeriod", + " originPurpose", + " destinationPurpose"], + dtype={"trip_ID": "int32", + "originMgra": "int16", + "destinationMgra": "int16", + "originTaz": "int16", + "destinationTaz": "int16", + "totalPassengers": "int8", + "startPeriod": "int16", + "endPeriod": "int16", + " originPurpose": "int8", + " destinationPurpose": "int8"}) + + # filter trip list to empty/zombie tnc trips + trips = trips.loc[trips["totalPassengers"] == 0].copy() + trips.reset_index(drop=True, inplace=True) + + # apply exhaustive field mappings where applicable + mappings = { + " originPurpose": {0: "Home", + 1: "Pickup Only", + 2: "Drop-off Only", + 3: "Pickup and Drop-off", + 4: "Refuel"}, + " destinationPurpose": {0: "Home", + 1: "Pickup Only", + 2: "Drop-off Only", + 3: "Pickup and Drop-off", + 4: "Refuel"} + } + + for field in mappings: + trips[field] = trips[field].map(mappings[field]).astype("category") + + # only map TNC time periods to ABM time periods if they nest + period_width = self.properties["timePeriodWidthTNC"] + if 30 % period_width == 0: + # map TNC time periods to actual period start times + # take the defined width of the time periods multiplied + # by the time period number as the minutes after 3am allowing + # the time periods to wrap around 12am and set the time value + trips["StartTime"] = trips["startPeriod"].apply( + lambda x: (datetime.combine(date.today(), time(3, 0)) + + timedelta(minutes=(x-1) * period_width)).time()) + trips["EndTime"] = trips["endPeriod"].apply( + lambda x: (datetime.combine(date.today(), time(3, 0)) + + timedelta(minutes=(x-1) * period_width)).time()) + + # map continuous times to abm half hour periods + depart_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in trips["StartTime"]] + depart_half_hour = [val for sublist in depart_half_hour for val in sublist] + trips = trips.assign(departTimeAbmHalfHour=depart_half_hour) + trips["departTimeAbmHalfHour"] = trips["departTimeAbmHalfHour"].astype("int8") + + arrive_half_hour = [ + [p["period"] for p in self.time_periods["abmHalfHour"] + if p["startTime"] <= x < p["endTime"]] + for x in trips["EndTime"]] + arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] + trips = trips.assign(arriveTimeAbmHalfHour=arrive_half_hour) + trips["arriveTimeAbmHalfHour"] = trips["arriveTimeAbmHalfHour"].astype("int8") + + # map abm half hours to abm five time of day + trips["departTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.departTimeAbmHalfHour) + + trips["arriveTimeFiveTod"] = self._map_time_periods( + abm_half_hour=trips.arriveTimeAbmHalfHour) + else: + # if time periods are not able to nest within ABM model time periods + # set the ABM model time period fields to NaN + # RSG has been made aware of this issue + trips["departTimeAbmHalfHour"] = pd.Series(np.nan, dtype="float32") + trips["departTimeFiveTod"] = pd.Series(np.nan, dtype="float32") + trips["arriveTimeAbmHalfHour"] = pd.Series(np.nan, dtype="float32") + trips["arriveTimeFiveTod"] = pd.Series(np.nan, dtype="float32") + + # all zombie TNC trips are Drive Alone and High vot + # assumed Transponder ownership for Drive Alone TNC + trips["tripMode"] = "Drive Alone" + trips["tripMode"] = trips["tripMode"].astype("category") + trips["valueOfTimeCategory"] = "High" + trips["valueOfTimeCategory"] = trips["valueOfTimeCategory"].astype("category") + trips["transponderAvailable"] = True + trips["avUsed"] = False + trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") + trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") + + # add person-based weight and adjust weights + # by the ABM scenario final iteration sample rate + # no people are in zombie AV trips + trips["weightTrip"] = 1 / self.properties["sampleRate"] + trips["weightTrip"] = trips["weightTrip"].astype("float32") + trips["weightPersonTrip"] = 0 + trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("int8") + + # rename columns to standard/generic ABM naming conventions + trips.rename(columns={"trip_ID": "tripID", + "originMgra": "originMGRA", + "destinationMgra": "destinationMGRA", + "originTaz": "originTAZ", + "destinationTaz": "destinationTAZ", + " originPurpose": "originPurpose", + " destinationPurpose": "destinationPurpose"}, + inplace=True) + + return trips[["tripID", + "originPurpose", + "destinationPurpose", + "departTimeAbmHalfHour", + "arriveTimeAbmHalfHour", + "departTimeFiveTod", + "arriveTimeFiveTod", + "originMGRA", + "destinationMGRA", + "parkingMGRA", + "originTAZ", + "destinationTAZ", + "parkingTAZ", + "tripMode", + "valueOfTimeCategory", + "transponderAvailable", + "avUsed", + "weightTrip", + "weightPersonTrip"]] diff --git a/sandag_abm/src/main/python/dataExporter/environment.yml b/sandag_abm/src/main/python/dataExporter/environment.yml new file mode 100644 index 0000000..e00766c --- /dev/null +++ b/sandag_abm/src/main/python/dataExporter/environment.yml @@ -0,0 +1,92 @@ +name: abmDataExporter +channels: + - defaults +dependencies: + - attrs=19.3.0=py_0 + - blas=1.0=mkl + - bzip2=1.0.8=he774522_0 + - ca-certificates=2020.6.24=0 + - certifi=2020.6.20=py38_0 + - cfitsio=3.470=he774522_5 + - click=7.1.2=py_0 + - click-plugins=1.1.1=py_0 + - cligj=0.5.0=py38_0 + - curl=7.67.0=h2a8f88b_0 + - expat=2.2.9=h33f27b4_2 + - fiona=1.8.13.post1=py38hd760492_0 + - freexl=1.0.5=hfa6e2cd_0 + - gdal=3.0.2=py38hdf43c64_0 + - geopandas=0.8.1=py_0 + - geos=3.8.0=h33f27b4_0 + - geotiff=1.5.1=h5770a2b_1 + - hdf4=4.2.13=h712560f_2 + - hdf5=1.10.4=h7ebc959_0 + - icc_rt=2019.0.0=h0cc432a_1 + - icu=58.2=ha925a31_3 + - intel-openmp=2020.0=166 + - jpeg=9b=hb83a4c4_2 + - kealib=1.4.7=h07cbb95_6 + - krb5=1.16.4=hc04afaa_0 + - libboost=1.67.0=hd9e427e_4 + - libcurl=7.67.0=h2a8f88b_0 + - libgdal=3.0.2=h1155b67_0 + - libiconv=1.15=h1df5818_7 + - libkml=1.3.0=he5f2a48_4 + - libnetcdf=4.6.1=h411e497_2 + - libpng=1.6.37=h2a8f88b_0 + - libpq=11.2=h3235a2c_0 + - libspatialindex=1.9.3=h33f27b4_0 + - libspatialite=4.3.0a=h7ffb84d_0 + - libssh2=1.9.0=h7a1dbc1_1 + - libtiff=4.1.0=h56a325e_0 + - libxml2=2.9.10=h464c3ec_1 + - lz4-c=1.8.1.2=h2fa13f4_0 + - m2w64-expat=2.1.1=2 + - m2w64-gcc-libgfortran=5.3.0=6 + - m2w64-gcc-libs=5.3.0=7 + - m2w64-gcc-libs-core=5.3.0=7 + - m2w64-gettext=0.19.7=2 + - m2w64-gmp=6.1.0=2 + - m2w64-libiconv=1.14=6 + - m2w64-libwinpthread-git=5.0.0.4634.697f757=2 + - m2w64-xz=5.2.2=2 + - mkl=2020.0=166 + - mkl-service=2.3.0=py38hb782905_0 + - mkl_fft=1.0.15=py38h14836fe_0 + - mkl_random=1.1.0=py38hf9181ef_0 + - msys2-conda-epoch=20160418=1 + - munch=2.5.0=py_0 + - numpy=1.18.1=py38h93ca92e_0 + - numpy-base=1.18.1=py38hc3f5095_1 + - openjpeg=2.3.0=h5ec785f_1 + - openssl=1.1.1g=he774522_1 + - pandas=1.0.1=py38h47e9c7a_0 + - pcre=8.44=ha925a31_0 + - pip=20.0.2=py38_1 + - postgresql=11.2=h3235a2c_0 + - proj=6.2.1=h9f7ef89_0 + - pyproj=2.6.1.post1=py38hcfa1391_1 + - python=3.8.1=h5fd99cc_1 + - python-dateutil=2.8.1=py_0 + - pytz=2019.3=py_0 + - rtree=0.9.4=py38h21ff451_1 + - setuptools=45.2.0=py38_0 + - shapely=1.7.0=py38h210f175_0 + - six=1.14.0=py38_0 + - sqlite=3.31.1=he774522_0 + - tbb=2018.0.5=he980bc4_0 + - tiledb=1.6.3=h7b000aa_0 + - tk=8.6.10=he774522_0 + - vc=14.1=h0510ff6_4 + - vs2015_runtime=14.16.27012=hf0eaf9b_1 + - wheel=0.34.2=py38_0 + - wincertstore=0.2=py38_0 + - xerces-c=3.2.2=ha925a31_0 + - xz=5.2.5=h62dcd97_0 + - zlib=1.2.11=h62dcd97_4 + - zstd=1.3.7=h508b16e_0 + - pip: + - numexpr==2.7.1 + - openmatrix==0.3.5.0 + - tables==3.6.1 + diff --git a/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py b/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py new file mode 100644 index 0000000..1fe6271 --- /dev/null +++ b/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py @@ -0,0 +1,523 @@ +import geopandas +import numpy as np +import os +import pandas as pd +from shapely import wkt + + +def export_highway_shape(scenario_path: str) -> geopandas.GeoDataFrame: + """ Takes an input path to a completed ABM scenario model run, reads the + input and loaded highway networks from the report folder, and outputs a + geography shape file to the report folder of the loaded highway network. + + Args: + scenario_path: String location of the completed ABM scenario folder + + Returns: + A GeoPandas GeoDataFrame of the loaded highway network """ + # read in input highway network + hwy_tcad = pd.read_csv(os.path.join(scenario_path, "report", "hwyTcad.csv"), + usecols=["ID", # highway coverage id + "NM", # link name + "Length", # link length in miles + "COJUR", # count jurisdiction code + "COSTAT", # count station number + "COLOC", # count location code + "IFC", # initial functional class + "IHOV", # link operation type + "ITRUCK", # truck restriction code + "ISPD", # posted speed limit + "IWAY", # one or two way operations + "IMED", # median type + "AN", # A node number + "FXNM", # cross street name at from end of link + "BN", # B node number + "TXNM", # cross street name at to end of link + "ABLN_EA", # lanes - from-to - Early AM + "ABLN_AM", # lanes - from-to - AM Peak + "ABLN_MD", # lanes - from-to - Midday + "ABLN_PM", # lanes - from-to - PM Peak + "ABLN_EV", # lanes - from-to - Evening + "BALN_EA", # lanes - to-from - Early AM + "BALN_AM", # lanes - to-from - AM Peak + "BALN_MD", # lanes - to-from - Midday + "BALN_PM", # lanes - to-from - PM Peak + "BALN_EV", # lanes - to-from - Evening + "ABPRELOAD_EA", # preloaded bus flow - to-from - Early AM + "BAPRELOAD_EA", # preloaded bus flow - from-to - Early AM + "ABPRELOAD_AM", # preloaded bus flow - to-from - AM Peak + "BAPRELOAD_AM", # preloaded bus flow - from-to - AM Peak + "ABPRELOAD_MD", # preloaded bus flow - to-from - Midday + "BAPRELOAD_MD", # preloaded bus flow - from-to - Midday + "ABPRELOAD_PM", # preloaded bus flow - to-from - PM Peak + "BAPRELOAD_PM", # preloaded bus flow - from-to - PM Peak + "ABPRELOAD_EV", # preloaded bus flow - to-from - Evening + "BAPRELOAD_EV", # preloaded bus flow - from-to - Evening + "geometry"]) # WKT geometry + + # read in loaded highway network for each time period + for tod in ["EA", "AM", "MD", "PM", "EV"]: + fn = "hwyload_" + tod + ".csv" + + file = pd.read_csv(os.path.join(scenario_path, "report", fn), + usecols=["ID1", # highway coverage id + "AB_Time", # a-b loaded travel time + "BA_Time", # b-a loaded travel time + "AB_Speed", # a-b loaded speed + "BA_Speed", # b-a loaded speed + "AB_VOC", # a-b volume to capacity + "BA_VOC", # b-a volume to capacity + "AB_Flow_SOV_NTPL", + "BA_Flow_SOV_NTPL", + "AB_Flow_SOV_TPL", + "BA_Flow_SOV_TPL", + "AB_Flow_SR2L", + "BA_Flow_SR2L", + "AB_Flow_SR3L", + "BA_Flow_SR3L", + "AB_Flow_SOV_NTPM", + "BA_Flow_SOV_NTPM", + "AB_Flow_SOV_TPM", + "BA_Flow_SOV_TPM", + "AB_Flow_SR2M", + "BA_Flow_SR2M", + "AB_Flow_SR3M", + "BA_Flow_SR3M", + "AB_Flow_SOV_NTPH", + "BA_Flow_SOV_NTPH", + "AB_Flow_SOV_TPH", + "BA_Flow_SOV_TPH", + "AB_Flow_SR2H", + "BA_Flow_SR2H", + "AB_Flow_SR3H", + "BA_Flow_SR3H", + "AB_Flow_lhd", + "BA_Flow_lhd", + "AB_Flow_mhd", + "BA_Flow_mhd", + "AB_Flow_hhd", + "BA_Flow_hhd"]) + + # match input highway network to loaded highway network + # to get preload bus flows + file = file.merge(right=hwy_tcad, + how="inner", + left_on="ID1", + right_on="ID") + + # calculate aggregated flows + file["AB_Flow_SOV"] = file[["AB_Flow_SOV_NTPL", + "AB_Flow_SOV_TPL", + "AB_Flow_SOV_NTPM", + "AB_Flow_SOV_TPM", + "AB_Flow_SOV_NTPH", + "AB_Flow_SOV_TPH"]].sum(axis=1) + + file["BA_Flow_SOV"] = file[["BA_Flow_SOV_NTPL", + "BA_Flow_SOV_TPL", + "BA_Flow_SOV_NTPM", + "BA_Flow_SOV_TPM", + "BA_Flow_SOV_NTPH", + "BA_Flow_SOV_TPH"]].sum(axis=1) + + file["AB_Flow_SR2"] = file[["AB_Flow_SR2L", + "AB_Flow_SR2M", + "AB_Flow_SR2H"]].sum(axis=1) + + file["BA_Flow_SR2"] = file[["BA_Flow_SR2L", + "BA_Flow_SR2M", + "BA_Flow_SR2H"]].sum(axis=1) + + file["AB_Flow_SR3"] = file[["AB_Flow_SR3L", + "AB_Flow_SR3M", + "AB_Flow_SR3H"]].sum(axis=1) + + file["BA_Flow_SR3"] = file[["BA_Flow_SR3L", + "BA_Flow_SR3M", + "BA_Flow_SR3H"]].sum(axis=1) + + file["AB_Flow_Truck"] = file[["AB_Flow_lhd", + "AB_Flow_mhd", + "AB_Flow_hhd"]].sum(axis=1) + + file["BA_Flow_Truck"] = file[["BA_Flow_lhd", + "BA_Flow_mhd", + "BA_Flow_hhd"]].sum(axis=1) + + file["AB_Flow_Bus"] = file["ABPRELOAD_" + tod] + + file["BA_Flow_Bus"] = file["BAPRELOAD_" + tod] + + file["AB_Flow"] = file[["AB_Flow_SOV", + "AB_Flow_SR2", + "AB_Flow_SR3", + "AB_Flow_Truck", + "AB_Flow_Bus"]].sum(axis=1) + + file["BA_Flow"] = file[["BA_Flow_SOV", + "BA_Flow_SR2", + "BA_Flow_SR3", + "BA_Flow_Truck", + "BA_Flow_Bus"]].sum(axis=1) + + # fill NAs with 0s + na_vars = ["AB_Time", + "BA_Time", + "AB_Speed", + "BA_Speed", + "AB_VOC", + "BA_VOC"] + + file[na_vars] = file[na_vars].fillna(0) + + # select columns of interest + file = file[["ID1", + "AB_Time", + "BA_Time", + "AB_Speed", + "BA_Speed", + "AB_VOC", + "BA_VOC", + "AB_Flow_SOV", + "BA_Flow_SOV", + "AB_Flow_SR2", + "BA_Flow_SR2", + "AB_Flow_SR3", + "BA_Flow_SR3", + "AB_Flow_Truck", + "BA_Flow_Truck", + "AB_Flow_Bus", + "BA_Flow_Bus", + "AB_Flow", + "BA_Flow"]] + + # add time of day suffix to column names + file = file.add_suffix("_" + tod) + + # merge loaded highway network into input highway network + hwy_tcad = hwy_tcad.merge(right=file, + how="inner", + left_on="ID", + right_on="ID1_" + tod) + + # create string description of [IFC] field + conditions = [hwy_tcad["IFC"] == 1, + hwy_tcad["IFC"] == 2, + hwy_tcad["IFC"] == 3, + hwy_tcad["IFC"] == 4, + hwy_tcad["IFC"] == 5, + hwy_tcad["IFC"] == 6, + hwy_tcad["IFC"] == 7, + hwy_tcad["IFC"] == 8, + hwy_tcad["IFC"] == 9, + hwy_tcad["IFC"] == 10] + + choices = ["Freeway", + "Prime Arterial", + "Major Arterial", + "Collector", + "Local Collector", + "Rural Collector", + "Local (non-circulation element) Road", + "Freeway Connector Ramp", + "Local Ramp", + "Zone Connector"] + + hwy_tcad["IFC_Desc"] = np.select(conditions, choices, default="") + + # calculate aggregate flows + hwy_tcad["AB_Flow_SOV"] = hwy_tcad[["AB_Flow_SOV_EA", + "AB_Flow_SOV_AM", + "AB_Flow_SOV_MD", + "AB_Flow_SOV_PM", + "AB_Flow_SOV_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow_SOV"] = hwy_tcad[["BA_Flow_SOV_EA", + "BA_Flow_SOV_AM", + "BA_Flow_SOV_MD", + "BA_Flow_SOV_PM", + "BA_Flow_SOV_EV"]].sum(axis=1) + + hwy_tcad["AB_Flow_SR2"] = hwy_tcad[["AB_Flow_SR2_EA", + "AB_Flow_SR2_AM", + "AB_Flow_SR2_MD", + "AB_Flow_SR2_PM", + "AB_Flow_SR2_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow_SR2"] = hwy_tcad[["BA_Flow_SR2_EA", + "BA_Flow_SR2_AM", + "BA_Flow_SR2_MD", + "BA_Flow_SR2_PM", + "BA_Flow_SR2_EV"]].sum(axis=1) + + hwy_tcad["AB_Flow_SR3"] = hwy_tcad[["AB_Flow_SR3_EA", + "AB_Flow_SR3_AM", + "AB_Flow_SR3_MD", + "AB_Flow_SR3_PM", + "AB_Flow_SR3_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow_SR3"] = hwy_tcad[["BA_Flow_SR3_EA", + "BA_Flow_SR3_AM", + "BA_Flow_SR3_MD", + "BA_Flow_SR3_PM", + "BA_Flow_SR3_EV"]].sum(axis=1) + + hwy_tcad["AB_Flow_Truck"] = hwy_tcad[["AB_Flow_Truck_EA", + "AB_Flow_Truck_AM", + "AB_Flow_Truck_MD", + "AB_Flow_Truck_PM", + "AB_Flow_Truck_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow_Truck"] = hwy_tcad[["BA_Flow_Truck_EA", + "BA_Flow_Truck_AM", + "BA_Flow_Truck_MD", + "BA_Flow_Truck_PM", + "BA_Flow_Truck_EV"]].sum(axis=1) + + hwy_tcad["AB_Flow_Bus"] = hwy_tcad[["AB_Flow_Bus_EA", + "AB_Flow_Bus_AM", + "AB_Flow_Bus_MD", + "AB_Flow_Bus_PM", + "AB_Flow_Bus_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow_Bus"] = hwy_tcad[["BA_Flow_Bus_EA", + "BA_Flow_Bus_AM", + "BA_Flow_Bus_MD", + "BA_Flow_Bus_PM", + "BA_Flow_Bus_EV"]].sum(axis=1) + + hwy_tcad["AB_Flow_Auto"] = hwy_tcad[["AB_Flow_SOV", + "AB_Flow_SR2", + "AB_Flow_SR3"]].sum(axis=1) + + hwy_tcad["BA_Flow_Auto"] = hwy_tcad[["BA_Flow_SOV", + "BA_Flow_SR2", + "BA_Flow_SR3"]].sum(axis=1) + + hwy_tcad["AB_Flow"] = hwy_tcad[["AB_Flow_EA", + "AB_Flow_AM", + "AB_Flow_MD", + "AB_Flow_PM", + "AB_Flow_EV"]].sum(axis=1) + + hwy_tcad["BA_Flow"] = hwy_tcad[["BA_Flow_EA", + "BA_Flow_AM", + "BA_Flow_MD", + "BA_Flow_PM", + "BA_Flow_EV"]].sum(axis=1) + + hwy_tcad["Flow"] = hwy_tcad[["AB_Flow", + "BA_Flow"]].sum(axis=1) + + # calculate vehicle miles travelled (vmt) + hwy_tcad["AB_VMT"] = hwy_tcad["AB_Flow"] * hwy_tcad["Length"] + hwy_tcad["BA_VMT"] = hwy_tcad["BA_Flow"] * hwy_tcad["Length"] + hwy_tcad["VMT"] = hwy_tcad["Flow"] * hwy_tcad["Length"] + + # calculate vehicle hours travelled (vht) + hwy_tcad["AB_VHT"] = hwy_tcad["AB_Time_EA"] * hwy_tcad["AB_Flow_EA"] + \ + hwy_tcad["AB_Time_AM"] * hwy_tcad["AB_Flow_AM"] + \ + hwy_tcad["AB_Time_MD"] * hwy_tcad["AB_Flow_MD"] + \ + hwy_tcad["AB_Time_PM"] * hwy_tcad["AB_Flow_PM"] + \ + hwy_tcad["AB_Time_EV"] * hwy_tcad["AB_Flow_EV"] + + hwy_tcad["BA_VHT"] = hwy_tcad["BA_Time_EA"] * hwy_tcad["BA_Flow_EA"] + \ + hwy_tcad["BA_Time_AM"] * hwy_tcad["BA_Flow_AM"] + \ + hwy_tcad["BA_Time_MD"] * hwy_tcad["BA_Flow_MD"] + \ + hwy_tcad["BA_Time_PM"] * hwy_tcad["BA_Flow_PM"] + \ + hwy_tcad["BA_Time_EV"] * hwy_tcad["BA_Flow_EV"] + + hwy_tcad["VHT"] = hwy_tcad["AB_VHT"] + hwy_tcad["BA_VHT"] + + # select columns of interest + hwy_tcad = hwy_tcad[["ID", + "NM", + "Length", + "COJUR", + "COSTAT", + "COLOC", + "IFC", + "IFC_Desc", + "IHOV", + "ITRUCK", + "ISPD", + "IWAY", + "IMED", + "AN", + "FXNM", + "BN", + "TXNM", + "Flow", + "AB_Flow", + "BA_Flow", + "AB_VMT", + "BA_VMT", + "VMT", + "AB_VHT", + "BA_VHT", + "VHT", + "AB_Flow_EA", + "BA_Flow_EA", + "AB_Flow_AM", + "BA_Flow_AM", + "AB_Flow_MD", + "BA_Flow_MD", + "AB_Flow_PM", + "BA_Flow_PM", + "AB_Flow_EV", + "BA_Flow_EV", + "AB_Flow_Auto", + "BA_Flow_Auto", + "AB_Flow_SOV", + "BA_Flow_SOV", + "AB_Flow_SR2", + "BA_Flow_SR2", + "AB_Flow_SR3", + "BA_Flow_SR3", + "AB_Flow_Truck", + "BA_Flow_Truck", + "AB_Flow_Bus", + "BA_Flow_Bus", + "AB_Speed_EA", + "BA_Speed_EA", + "AB_Speed_AM", + "BA_Speed_AM", + "AB_Speed_MD", + "BA_Speed_MD", + "AB_Speed_PM", + "BA_Speed_PM", + "AB_Speed_EV", + "BA_Speed_EV", + "AB_Time_EA", + "BA_Time_EA", + "AB_Time_AM", + "BA_Time_AM", + "AB_Time_MD", + "BA_Time_MD", + "AB_Time_PM", + "BA_Time_PM", + "AB_Time_EV", + "BA_Time_EV", + "ABLN_EA", + "BALN_EA", + "ABLN_AM", + "BALN_AM", + "ABLN_MD", + "BALN_MD", + "ABLN_PM", + "BALN_PM", + "ABLN_EV", + "BALN_EV", + "AB_VOC_EA", + "BA_VOC_EA", + "AB_VOC_AM", + "BA_VOC_AM", + "AB_VOC_MD", + "BA_VOC_MD", + "AB_VOC_PM", + "BA_VOC_PM", + "AB_VOC_EV", + "BA_VOC_EV", + "geometry"]] + + # rename fields to match old process field names + hwy_tcad.rename(columns={"ID": "hwycovid", + "NM": "link_name", + "Length": "len_mile", + "COJUR": "count_jur", + "COSTAT": "count_stat", + "COLOC": "count_loc", + "IFC": "ifc", + "IFC_Desc": "ifc_desc", + "IHOV": "ihov", + "ITRUCK": "itruck", + "ISPD": "post_speed", + "IWAY": "iway", + "IMED": "imed", + "AN": "from_node", + "FXNM": "from_nm", + "BN": "to_node", + "TXNM": "to_nm", + "Flow": "total_flow", + "AB_Flow": "abTotFlow", + "BA_Flow": "baTotFlow", + "AB_VMT": "ab_vmt", + "BA_VMT": "ba_vmt", + "VMT": "vmt", + "AB_VHT": "ab_vht", + "BA_VHT": "ba_vht", + "VHT": "vht", + "AB_Flow_EA": "ab_ea_flow", + "BA_Flow_EA": "ba_ea_flow", + "AB_Flow_AM": "ab_am_flow", + "BA_Flow_AM": "ba_am_flow", + "AB_Flow_MD": "ab_md_flow", + "BA_Flow_MD": "ba_md_flow", + "AB_Flow_PM": "ab_pm_flow", + "BA_Flow_PM": "ba_pm_flow", + "AB_Flow_EV": "ab_ev_flow", + "BA_Flow_EV": "ba_ev_flow", + "AB_Flow_Auto": "abAutoFlow", + "BA_Flow_Auto": "baAutoFlow", + "AB_Flow_SOV": "abSovFlow", + "BA_Flow_SOV": "baSovFlow", + "AB_Flow_SR2": "abHov2Flow", + "BA_Flow_SR2": "baHov2Flow", + "AB_Flow_SR3": "abHov3Flow", + "BA_Flow_SR3": "baHov3Flow", + "AB_Flow_Truck": "abTrucFlow", + "BA_Flow_Truck": "baTrucFlow", + "AB_Flow_Bus": "abBusFlow", + "BA_Flow_Bus": "baBusFlow", + "AB_Speed_EA": "ab_ea_mph", + "BA_Speed_EA": "ba_ea_mph", + "AB_Speed_AM": "ab_am_mph", + "BA_Speed_AM": "ba_am_mph", + "AB_Speed_MD": "ab_md_mph", + "BA_Speed_MD": "ba_md_mph", + "AB_Speed_PM": "ab_pm_mph", + "BA_Speed_PM": "ba_pm_mph", + "AB_Speed_EV": "ab_ev_mph", + "BA_Speed_EV": "ba_ev_mph", + "AB_Time_EA": "ab_ea_min", + "BA_Time_EA": "ba_ea_min", + "AB_Time_AM": "ab_am_min", + "BA_Time_AM": "ba_am_min", + "AB_Time_MD": "ab_md_min", + "BA_Time_MD": "ba_md_min", + "AB_Time_PM": "ab_pm_min", + "BA_Time_PM": "ba_pm_min", + "AB_Time_EV": "ab_ev_min", + "BA_Time_EV": "ba_ev_min", + "ABLN_EA": "ab_ea_lane", + "BALN_EA": "ba_ea_lane", + "ABLN_AM": "ab_am_lane", + "BALN_AM": "ba_am_lane", + "ABLN_MD": "ab_md_lane", + "BALN_MD": "ba_md_lane", + "ABLN_PM": "ab_pm_lane", + "BALN_PM": "ba_pm_lane", + "ABLN_EV": "ab_ev_lane", + "BALN_EV": "ba_ev_lane", + "AB_VOC_EA": "ab_ea_voc", + "BA_VOC_EA": "ba_ea_voc", + "AB_VOC_AM": "ab_am_voc", + "BA_VOC_AM": "ba_am_voc", + "AB_VOC_MD": "ab_md_voc", + "BA_VOC_MD": "ba_md_voc", + "AB_VOC_PM": "ab_pm_voc", + "BA_VOC_PM": "ba_pm_voc", + "AB_VOC_EV": "ab_ev_voc", + "BA_VOC_EV": "ba_ev_voc"}, + inplace=True) + + # create geometry from WKT geometry field + hwy_tcad["geometry"] = hwy_tcad["geometry"].apply(wkt.loads) + + # create GeoPandas DataFrame + hwy_tcad = geopandas.GeoDataFrame( + hwy_tcad, + geometry="geometry", + crs=2230) + + return hwy_tcad diff --git a/sandag_abm/src/main/python/dataExporter/serialRun.py b/sandag_abm/src/main/python/dataExporter/serialRun.py new file mode 100644 index 0000000..9b6b7b0 --- /dev/null +++ b/sandag_abm/src/main/python/dataExporter/serialRun.py @@ -0,0 +1,168 @@ +from hwyShapeExport import export_highway_shape +from skimAppender import SkimAppender +from abmScenario import ScenarioData, LandUse, SyntheticPopulation, TourLists, TripLists +import os +import sys + + +def export_data(fp): + # set file path to completed ABM run scenario folder + # set report folder path + scenarioPath = fp + reportPath = os.path.join(scenarioPath, "report") + + + # initialize base ABM scenario data class + print("Initializing Scenario Data") + scenario_data = ScenarioData(scenarioPath) + + # write out transit TAP park and ride file + print("Writing: Transit PNR Input File") + scenario_data.pnr_taps.to_csv(os.path.join(reportPath, "transitPNR.csv"), index=False) + + + # initialize land use class + # write out MGRA-based input file + print("Initializing Land Use Output") + land_use = LandUse(scenarioPath) + print("Writing: MGRA-Based Input File") + land_use.mgra_input.to_csv(os.path.join(reportPath, "mgraBasedInput.csv"), index=False) + + + # initialize synthetic population class + # write out households and persons files + print("Initializing Synthetic Population Output") + population = SyntheticPopulation(scenarioPath) + + print("Writing: Households File") + population.households.to_csv(os.path.join(reportPath, "households.csv"), index=False) + + print("Writing: Persons File") + population.persons.to_csv(os.path.join(reportPath, "persons.csv"), index=False) + + + # initialize tour list class + # write out tour list files + print("Initializing Tour List Output") + tours = TourLists(scenarioPath) + + print("Writing: Commercial Vehicle Tours") + tours.cvm.to_csv(os.path.join(reportPath, "commercialVehicleTours.csv"), index=False) + + print("Writing: Cross Border Tours") + tours.cross_border.to_csv(os.path.join(reportPath, "crossBorderTours.csv"), index=False) + + print("Writing: Individual Tours") + tours.individual.to_csv(os.path.join(reportPath, "individualTours.csv"), index=False) + + print("Writing: Internal-External Tours") + tours.ie.to_csv(os.path.join(reportPath, "internalExternalTours.csv"), index=False) + + print("Writing: Joint Tours") + tours.joint.to_csv(os.path.join(reportPath, "jointTours.csv"), index=False) + + print("Writing: Visitor Tours") + tours.visitor.to_csv(os.path.join(reportPath, "visitorTours.csv"), index=False) + + + print("Initializing Trip List Output") + + # initialize trip list class + trips = TripLists(scenarioPath) + + # initialize skim appender class + skims = SkimAppender(scenarioPath) + + # write out trip list files + print("Writing: Airport-SAN Trips") + skims.append_skims(trips.airport_san, + auto_only=False, + terminal_skims=False).to_csv( + os.path.join(reportPath, "airportSANTrips.csv"), + index=False) + + print("Writing: Airport-CBX Trips") + skims.append_skims(trips.airport_cbx, + auto_only=False, + terminal_skims=False).to_csv( + os.path.join(reportPath, "airportCBXTrips.csv"), + index=False) + + print("Writing: Commercial Vehicle Trips") + skims.append_skims(trips.cvm, + auto_only=True, + terminal_skims=False).to_csv( + os.path.join(reportPath, "commercialVehicleTrips.csv"), + index=False) + + print("Writing: Cross-Border Trips") + skims.append_skims(trips.cross_border, + auto_only=False, + terminal_skims=False).to_csv( + os.path.join(reportPath, "crossBorderTrips.csv"), + index=False) + + print("Writing: External-External Trips") + trips.ee.to_csv( + os.path.join(reportPath, "externalExternalTrips.csv"), + index=False) + + print("Writing: External-Internal Trips") + trips.ei.to_csv( + os.path.join(reportPath, "externalInternalTrips.csv"), + index=False) + + print("Writing: Individual Trips") + skims.append_skims(trips.individual, + auto_only=False, + terminal_skims=True).to_csv( + os.path.join(reportPath, "individualTrips.csv"), + index=False) + + print("Writing: Internal-External Trips") + skims.append_skims(trips.ie, + auto_only=False, + terminal_skims=False).to_csv( + os.path.join(reportPath, "internalExternalTrips.csv"), + index=False) + + print("Writing: Joint Trips") + skims.append_skims(trips.joint, + auto_only=False, + terminal_skims=True).to_csv( + os.path.join(reportPath, "jointTrips.csv"), + index=False) + + print("Writing: Truck Trips") + trips.truck.to_csv( + os.path.join(reportPath, "truckTrips.csv"), + index=False) + + print("Writing: Visitor Trips") + skims.append_skims(trips.visitor, + auto_only=False, + terminal_skims=False).to_csv( + os.path.join(reportPath, "visitorTrips.csv"), + index=False) + + print("Writing: Zombie AV Trips") + skims.append_skims(trips.zombie_av, + auto_only=True, + terminal_skims=False).to_csv( + os.path.join(reportPath, "zombieAVTrips.csv"), + index=False) + + print("Writing: Zombie TNC Trips") + skims.append_skims(trips.zombie_tnc, + auto_only=True, + terminal_skims=False).to_csv( + os.path.join(reportPath, "zombieTNCTrips.csv"), + index=False) + + print("Writing: Highway Load Shape File") + export_highway_shape(scenarioPath).to_file( + os.path.join(reportPath, "hwyLoad.shp")) + +if __name__ == '__main__': + targets = sys.argv[1:] + export_data(targets[0]) diff --git a/sandag_abm/src/main/python/dataExporter/skimAppender.py b/sandag_abm/src/main/python/dataExporter/skimAppender.py new file mode 100644 index 0000000..cb6c5f1 --- /dev/null +++ b/sandag_abm/src/main/python/dataExporter/skimAppender.py @@ -0,0 +1,1895 @@ +# -*- coding: utf-8 -*- +""" ABM Scenario Skim Appender Module. + +This module contains classes holding all utilities relating to appending +transportation skims to a completed SANDAG Activity-Based Model (ABM) +scenario. This module is used to append time, distance, cost, and other +related transportation skims to ABM trip lists. + +Notes: + docstring style guide - http://google.github.io/styleguide/pyguide.html +""" + +import itertools +import os +import re +from functools import lru_cache # caching decorator for modules +import numpy as np +import openmatrix as omx # https://github.com/osPlanning/omx-python +import pandas as pd + + +class SkimAppender(object): + """ This class holds all utilities relating to appending transportation + skims to a completed SANDAG Activity-Based Model (ABM) scenario + + Args: + scenario_path: String location of the completed ABM scenario folder + + Methods: + _get_omx_auto_skim_dataset: Maps ABM trip list records to OMX files + and OMX skim matrices + append_skims: Master method to append all skims to ABM trip lists + auto_fare_cost: Appends auto fare cost to ABM trip list records + auto_operating_cost: Appends auto operating cost to ABM trip list records + auto_terminal_skims: Appends auto-mode terminal walk time and distance + from the zone.term file to ABM trip list records + auto_wait_time: Appends auto wait times to ABM trip list records + bicycle_skims: Appends bicycle mode skims (time, distance) to ABM trip + list records + drive_transit_skims: Appends input file accessam.csv drive to + transit auto mode skims (time, distance, no cost) to ABM trip + list records + omx_auto_skim_appender: Appends OMX auto mode skims (time, distance, + cost) to ABM trip list records + omx_transit_skims: Appends OMX transit mode skims (time, distance, + cost) to transit mode trip list records + tnc_fare_cost: Appends TNC fare costs to ABM trip list records + tnc_wait_time: Appends TNC wait time to ABM trip list records + walk_skims: Appends walk/micro-mobility/micro-transit mode skims + (time, distance, cost) to ABM trip list records for + walk/micro-mobility/micro-transit mode trips + _walk_skims_at: Appends walk/micro-mobility/micro-transit mode skims + (time, distance, cost) to ABM trip list records for + walk/micro-mobility/micro-transit mode trips that do not use auto + mode skim sets for trip time + _walk_skims_auto: Appends walk/micro-mobility/micro-transit mode skims + (time, distance, cost) to ABM trip list records for + walk/micro-mobility/micro-transit mode trips that use auto mode + skim sets for trip time + walk_transit_skims: Appends walk/micro-mobility/micro-transit + access/egress to/from transit to ABM transit mode trip list + records + + Properties: + mgra_xref: Pandas DataFrame geography cross-reference of MGRAs to + TAZs and LUZs + properties: Dictionary of ABM properties file token values + (conf/sandag_abm.properties) """ + + def __init__(self, scenario_path: str) -> None: + self.scenario_path = scenario_path + + @property + @lru_cache(maxsize=1) + def mgra_xref(self) -> pd.DataFrame: + """ Cross reference of Master Geographic Reference Area (MGRA) model + geography to Transportation Analysis Zone (TAZ) and Land Use Zone + (LUZ) model geographies. Cross reference is stored in each ABM + scenario input MGRA file (input/mgra13_based_input<>.csv). + """ + + # load the mgra based input file + fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" + + mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), + usecols=["mgra", # MGRA geography + "taz", # TAZ geography + "luz_id", # LUZ geography + "MicroAccessTime"], # Micro-Mobility AccessTime + dtype={"mgra": "int16", + "taz": "int16", + "luz_id": "int16"}) + + # genericize column names + mgra.rename(columns={"mgra": "MGRA", + "taz": "TAZ", + "luz_id": "LUZ"}, + inplace=True) + + return mgra + + @property + @lru_cache(maxsize=1) + def properties(self) -> dict: + """ Get the ABM scenario properties from the ABM scenario + properties file (conf/sandag_abm.properties). + + The return dictionary contains the following ABM scenario properties: + accessTimeMicroTransit - Micro-Transit access time in minutes + aocFuel - auto operating fuel cost in $/mile + aocMaintenance - auto operating maintenance cost in $/mile + baseFareMicroMobility - initial Micro-Mobility fare in $ + baseFareMicroTransit - initial Micro-Mobility fare in $ + baseFareNonPooledTNC - initial Non-Pooled TNC fare in $ + baseFarePooledTNC - initial Pooled TNC fare in $ + baseFareTaxi - initial taxi fare in $ + bicycleSpeed - bicycle mode speed (miles/hour) + costMinimumNonPooledTNC - minimum Non-Pooled TNC fare cost in $ + costMinimumPooledTNC - minimum Pooled TNC fare cost in $ + costPerMileFactorAV - auto operating cost per mile factor to + apply to AV trips + costPerMileNonPooledTNC - Non-Pooled TNC fare cost per mile in $ + costPerMilePooledTNC - Pooled TNC fare cost per mile in $ + costPerMileTaxi - Taxi fare cost per mile in $ + costPerMinuteMicroMobility - Micro-Mobility fare cost per minute in $ + costPerMinuteMicroTransit - Micro-Transit fare cost per minute in $ + costPerMinuteNonPooledTNC - Non-Pooled TNC fare cost per minute in $ + costPerMinutePooledTNC - Pooled TNC fare cost per minute in $ + costPerMinuteTaxi - Taxi fare cost per minute in $ + microMobilitySpeed - Micro-Mobility mode speed (miles/hour) + microTransitSpeed - Micro-Transit mode speed (miles/hour) + terminalTimeFactorAV - terminal time factor to apply to AV trips + waitTimeMicroTransit - Micro-Mobility wait time in minutes + waitTimeNonPooledTNC - list of mean wait times in minutes for + Non-Pooled TNC by PopEmpDenPerMi categories + (see waitTimePopEmpDenPerMi) + waitTimePooledTNC - list of mean wait times in minutes for Pooled + TNC by PopEmpDenPerMi categories (see waitTimePopEmpDenPerMi) + waitTimePopEmpDenPerMi - list of MGRA-Based input file + PopEmpDenPerMi values defining the wait time categories for + Taxi/TNC modes + waitTimeTaxi - list of mean wait times in minutes for Taxi by + PopEmpDenPerMi categories (see waitTimePopEmpDenPerMi) + walkSpeed - walk mode speed (miles/hour) + year - analysis year of the ABM scenario + + Returns: + A dictionary defining the ABM scenario properties. """ + + # create dictionary holding ABM properties file information + # each property contains a dictionary {line, value} where the line + # is the string to match in the properties file to + # return the value of the property + lookup = { + "accessTimeMicroTransit": { + "line": "active.microtransit.accessTime=", + "type": "float", + "value": None + }, + "aocFuel": { + "line": "aoc.fuel=", + "type": "float", + "value": None + }, + "aocMaintenance": { + "line": "aoc.maintenance=", + "type": "float", + "value": None + }, + "baseFareMicroMobility": { + "line": "active.micromobility.fixedCost=", + "type": "float", + "value": None + }, + "baseFareMicroTransit": { + "line": "active.microtransit.fixedCost=", + "type": "float", + "value": None + }, + "baseFareNonPooledTNC": { + "line": "TNC.single.baseFare=", + "type": "float", + "value": None + }, + "baseFarePooledTNC": { + "line": "TNC.shared.baseFare=", + "type": "float", + "value": None + }, + "baseFareTaxi": { + "line": "taxi.baseFare=", + "type": "float", + "value": None + }, + "bicycleSpeed": { + "line": "active.bike.minutes.per.mile=", + "type": "float", + "value": None}, + "costMinimumNonPooledTNC": { + "line": "TNC.single.costMinimum=", + "type": "float", + "value": None + }, + "costMinimumPooledTNC": { + "line": "TNC.shared.costMinimum=", + "type": "float", + "value": None + }, + "costPerMileFactorAV": { + "line": "Mobility.AV.CostPerMileFactor=", + "type": "float", + "value": None + }, + "costPerMileNonPooledTNC": { + "line": "TNC.single.costPerMile=", + "type": "float", + "value": None + }, + "costPerMilePooledTNC": { + "line": "TNC.shared.costPerMile=", + "type": "float", + "value": None + }, + "costPerMileTaxi": { + "line": "taxi.costPerMile=", + "type": "float", + "value": None + }, + "costPerMinuteMicroMobility": { + "line": "active.micromobility.variableCost=", + "type": "float", + "value": None + }, + "costPerMinuteMicroTransit": { + "line": "active.microtransit.variableCost=", + "type": "float", + "value": None + }, + "costPerMinuteNonPooledTNC": { + "line": "TNC.single.costPerMinute=", + "type": "float", + "value": None + }, + "costPerMinutePooledTNC": { + "line": "TNC.shared.costPerMinute=", + "type": "float", + "value": None + }, + "costPerMinuteTaxi": { + "line": "taxi.costPerMinute=", + "type": "float", + "value": None + }, + "microMobilitySpeed": { + "line": "active.micromobility.speed=", + "type": "float", + "value": None}, + "microTransitSpeed": { + "line": "active.microtransit.speed=", + "type": "float", + "value": None}, + "terminalTimeFactorAV": { + "line": "Mobility.AV.TerminalTimeFactor=", + "type": "float", + "value": None + }, + "waitTimeMicroTransit": { + "line": "active.microtransit.waitTime=", + "type": "float", + "value": None + }, + "waitTimeNonPooledTNC": { + "line": "TNC.single.waitTime.mean=", + "type": "list", + "value": None + }, + "waitTimePooledTNC": { + "line": "TNC.shared.waitTime.mean=", + "type": "list", + "value": None + }, + "waitTimePopEmpDenPerMi": { + "line": "WaitTimeDistribution.EndPopEmpPerSqMi=", + "type": "list", + "value": None + }, + "waitTimeTaxi": { + "line": "Taxi.waitTime.mean=", + "type": "list", + "value": None + }, + "walkSpeed": { + "line": "active.walk.minutes.per.mile=", + "type": "float", + "value": None}, + "year": { + "line": "scenarioYear=", + "type": "int", + "value": None} + } + + # open the ABM scenario properties file + file = open(os.path.join(self.scenario_path, "conf", "sandag_abm.properties"), "r") + + # loop through each line of the properties file + for line in file: + # strip all white space from the line + line = line.replace(" ", "") + + # for each element of the properties dictionary + for name in lookup: + item = lookup[name] + + if item["line"] is not None: + match = re.compile(item["line"]).match(line) + # if the properties file contains the matching line + if match: + # for waitTime properties create list from matching line + # + if "waitTime" in name and item["type"] == "list": + value = line[match.end():].split(",") + value = list(map(float, value)) + # otherwise take the final element of the line + else: + value = line[match.end():] + + # update the dictionary value using the appropriate data type + if item["type"] == "float": + value = float(value) + elif item["type"] == "int": + value = int(value) + else: + pass + + item["value"] = value + break + + file.close() + + # convert the property name and value to a non-nested dictionary + results = {} + for name in lookup: + results[name] = lookup[name]["value"] + + # convert auto operating costs from cents per mile to dollars per mile + results["aocFuel"] = results["aocFuel"] / 100 + results["aocMaintenance"] = results["aocMaintenance"] / 100 + + # convert AT speeds from minutes per mile to miles per hour + results["bicycleSpeed"] = (results["bicycleSpeed"] * 1/60)**-1 + results["microMobilitySpeed"] = (results["microMobilitySpeed"] * 1 / 60) ** -1 + results["microTransitSpeed"] = (results["microTransitSpeed"] * 1 / 60) ** -1 + results["walkSpeed"] = (results["walkSpeed"] * 1/60)**-1 + + return results + + @staticmethod + def _get_omx_auto_skim_dataset(df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame and returns a subset of trips + with two columns indicating the proper auto-mode omx file name and + skim matrix to use to append omx auto-mode transportation skims. + + If trips in the input DataFrame are not mapped to omx files + and matrices (non-auto mode trips) they are not present in the return + DataFrame. The return DataFrame is an interim DataFrame ready for + input to the omx_auto_skim_appender + + The process uses the trip departure ABM 5 TOD period, trip mode, + transponder availability, and the trip value of time (vot) category + to select the correct auto skim-set file and matrix to use to get + the auto-mode trip times, distances, and toll costs. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [transponderAvailable] - indicator if transponder available + on the trip (False, True) + [valueOfTimeCategory] - trip value of time categories + (Low, Medium, High) + [originTAZ] - trip origin TAZ (1-4996) + [destinationTAZ] - trip destination TAZ (1-4996) + [parkingTAZ] - trip parking TAZ ("", 1-4996) + + Returns: + A Pandas DataFrame containing mapping of auto-mode trips to omx + files and matrices containing auto-mode skims: + [tripID] - unique identifier of a trip + [omxFileName] - omx skim file name (e.g. traffic_skims_EA) + [matrixName] - omx skim matrix name (e.g. EA_HOV2_M) + [originTAZ] - trip origin TAZ (1-4996) + [destinationTAZ] - trip destination TAZ (1-4996) """ + + # set destinationTAZ equal to parkingTAZ is parkingTAZ is not missing + # auto portion of trip ends at the parkingTAZ + df.destinationTAZ = np.where(pd.notna(df.parkingTAZ), + df.parkingTAZ, + df.destinationTAZ).astype("int16") + + # create mappings from trip list to skim matrices + # segmented by SOV and Non-SOV as process differs between the two + # due to Transponder/Non-Transponder skimming in SOV mode + # SOV - departTimeFiveTod + tripMode + transponderAvailable + + # valueOfTimeCategory + # nonSOV - departTimeFiveTod + tripMode + valueOfTimeCategory + skim_map = {"SOV": {"values": [[1, 2, 3, 4, 5], + ["Drive Alone"], + [False, True], + ["Low", "Medium", "High"]], + "labels": [["EA", "AM", "MD", "PM", "EV"], + ["SOV"], + ["NT", "TR"], + ["L", "M", "H"]], + "cols": ["departTimeFiveTod", + "tripMode", + "transponderAvailable", + "valueOfTimeCategory", + "matrixName"]}, + "Non-SOV": {"values": [[1, 2, 3, 4, 5], + ["Shared Ride 2", + "Shared Ride 3+", + "Light Heavy Duty Truck", + "Medium Heavy Duty Truck", + "Heavy Heavy Duty Truck", + "Taxi", + "Non-Pooled TNC", + "Pooled TNC", + "School Bus"], + ["Low", "Medium", "High"]], + "labels": [["EA", "AM", "MD", "PM", "EV"], + ["HOV2", "HOV3", "TRK", "TRK", + "TRK", "HOV3", "HOV3", "HOV3", + "HOV3"], + ["L", "M", "H"]], + "cols": ["departTimeFiveTod", + "tripMode", + "valueOfTimeCategory", + "matrixName"]}} + + # initialize empty auto trips DataFrame + trips = pd.DataFrame() + + # map possible values of trip list columns to skim matrix names + # filter input DataFrame to auto trips mapped to skim matrices + for key in skim_map: + values = skim_map[key]["values"] + labels = skim_map[key]["labels"] + cols = skim_map[key]["cols"] + + mapping = [list(i) + ["_".join(j)] for i, j in + zip(itertools.product(*values), + itertools.product(*labels))] + + # create Pandas DataFrame lookup table of column values + # to skim matrix names + lookup = pd.DataFrame(mapping, columns=cols) + lookup["omxFileName"] = "traffic_skims_" + lookup["matrixName"].str[0:2] + + # set lookup DataFrame data types + lookup = lookup.astype({ + "departTimeFiveTod": "int8", + "tripMode": "category", + "valueOfTimeCategory": "category", + "omxFileName": "category", + "matrixName": "category"}) + + # merge lookup table to trip list and append to auto trips DataFrame + trips = trips.append(df.merge(lookup, how="inner"), ignore_index=True) + + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + trips.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + return trips[["tripID", + "omxFileName", + "matrixName", + "originTAZ", + "destinationTAZ"]] + + def append_skims(self, df: pd.DataFrame, auto_only: bool, terminal_skims: bool) -> pd.DataFrame: + """ Takes an input Pandas DataFrame, runs all skimming class + methods and appends all skims to the input Pandas DataFrame. + See documentation of included class methods. Additionally creates and + appends three fields to the input Pandas DataFrame: + [timeTotal] - total trip time in minutes + [distanceTotal] - total trip distance in miles + [costTotal] - total trip cost in dollars + + Args: + df: Input Pandas DataFrame containing fields required by the + included class methods. + terminal_skims: Boolean of whether to include auto-mode terminal + skims, only apply to Resident model trip lists. + auto_only: Boolean of whether to include only basic auto-mode + skims, applies to Commercial Vehicle, Zombie AV, and Zombie + TNC trip lists. + + Returns: + A Pandas DataFrame containing all skims appended by the included + class methods. The DataFrame contains all original fields + of the input DataFrame along with fields appended by the + aforementioned class methods. """ + + # append omx auto skims + df = self.omx_auto_skim_appender(df) + + # append auto operating cost + df = self.auto_operating_cost(df) + + if auto_only: + pass + else: + # append auto terminal skims if applicable + if terminal_skims: + df = self.auto_terminal_skims(df) + + # append TNC fare costs + df = self.tnc_fare_cost(df) + + # append TNC wait times + df = self.tnc_wait_time(df) + + # append omx transit skims + df = self.omx_transit_skims(df) + + # append drive to transit skims + # auto-operating cost not included + # TNC fare costs and wait times not included + df = self.drive_transit_skims(df) + + # append (micro-mobility/micro-transit/walk) to transit skims + df = self.walk_transit_skims(df) + + # append micro-mobility/micro-transit/walk skims + df = self.walk_skims(df) + + # append bicycle skims + df = self.bicycle_skims(df) + + # append total trip time, distance, and cost skims + # transit in vehicle times by line haul mode and initial wait time + # are not included due to double-counting + transit_line_haul_cols = ["timeTier1TransitInVehicle", + "timeFreewayRapidTransitInVehicle", + "timeArterialRapidTransitInVehicle", + "timeExpressBusTransitInVehicle", + "timeLocalBusTransitInVehicle", + "timeLightRailTransitInVehicle", + "timeCommuterRailTransitInVehicle", + "timeTransitInitialWait"] + + df["timeTotal"] = df[df.columns.difference(transit_line_haul_cols)].filter(regex="^time").sum(axis=1) + + df["distanceTotal"] = df.filter(regex="^distance").sum(axis=1) + + df["costTotal"] = df.filter(regex="^cost").sum(axis=1) + + # sort return DataFrame by tripID for user-experience + # and faster database loading via ORDER hints + df.sort_values(by="tripID", inplace=True) + + return df + + def auto_operating_cost(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [avUsed]) and the fields appended by the + SkimAppender class method omx_auto_skim_appender + ([distanceDrive]) and returns the associated + auto-mode operating cost. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [avUsed] - indicator if AV used on trip (True, False) + [distanceDrive] - distance in miles for auto mode + + Returns: + A Pandas DataFrame containing all associated auto-mode + operating costs. The DataFrame contains all original fields + of the input DataFrame along with the field: + [costOperatingDrive] - auto operating cost in $ """ + + # calculate auto operating cost as fuel+maintenance cents per mile + # note this is scaled for AV trips by a factor + aoc = self.properties["aocFuel"] + self.properties["aocMaintenance"] + aoc_av = aoc * self.properties["costPerMileFactorAV"] + + # calculate auto operating cost for auto-mode trips + # exclude Taxi/TNC as operating cost is passed to drivers + # who are not considered part of the model universe + auto_modes = ["Drive Alone", + "Shared Ride 2", + "Shared Ride 3+", + "Light Heavy Duty Truck", + "Medium Heavy Duty Truck", + "Heavy Heavy Duty Truck"] + + conditions = [ + np.array(df["tripMode"].isin(auto_modes) & ~df["avUsed"], + dtype="bool"), + np.array(df["tripMode"].isin(auto_modes) & df["avUsed"], + dtype="bool") + ] + + choices = [df["distanceDrive"] * aoc, + df["distanceDrive"] * aoc_av] + + df["costOperatingDrive"] = pd.Series( + np.select(conditions, choices, default=np.NaN), + dtype="float32") + + # return input DataFrame with appended auto operating cost column + return df + + def auto_terminal_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [destinationTAZ]) and appends the auto-mode + terminal walk time from the input/zone.term file. Distance is created + from the walk speed specified in the properties file + conf/sandag_abm.properties + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [destinationTAZ] - indicator if AV used on trip (True, False) + + Returns: + A Pandas DataFrame containing auto-mode terminal walk time and + distance. The DataFrame contains all original fields of the input + DataFrame along with the fields: + [timeAutoTerminalWalk] - auto terminal walk time in minutes + [distanceAutoTerminalWalk] - auto terminal walk distance in + miles + """ + # read in auto terminal time fixed width file + skims = pd.read_fwf( + os.path.join(self.scenario_path, "input", "zone.term"), + widths=[5, 7], + names=["destinationTAZ", + "timeAutoTerminalWalk"], + dtype={"destinationTAZ": "int16", + "timeAutoTerminalWalk": "float32"} + ) + + # merge auto terminal times into input DataFrame + # add 0s for TAZs with no terminal times + df = df.merge(skims, on="destinationTAZ", how="left") + df["timeAutoTerminalWalk"] = df["timeAutoTerminalWalk"].fillna(0) + + # set auto terminal times to 0 for non-auto modes + # reduce auto terminal time if AV is used + auto_modes = ["Drive Alone", "Shared Ride 2", "Shared Ride 3+"] + av_factor = self.properties["terminalTimeFactorAV"] + + conditions = [ + np.array(df["tripMode"].isin(auto_modes) & ~df["avUsed"], + dtype="bool"), + np.array(df["tripMode"].isin(auto_modes) & df["avUsed"], + dtype="bool") + ] + + choices = [df["timeAutoTerminalWalk"], + df["timeAutoTerminalWalk"] * av_factor] + + df["timeAutoTerminalWalk"] = pd.Series( + np.select(conditions, choices, default=np.NaN), + dtype="float32") + + # create auto terminal distance from walk speed property + df["distanceAutoTerminalWalk"] = pd.Series( + df["timeAutoTerminalWalk"] * self.properties["walkSpeed"] / 60, + dtype="float32") + + return df + + def bicycle_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originMGRA], [destinationMGRA], [originTAZ], + [destinationTAZ]) and returns the associated bicycle mode skims for + time and distance. + + Bicycle mode skims are given at the MGRA-MGRA level by the + output/bikeMgraLogsum.csv file and TAZ-TAZ level by the + output/bikeTazLogsum.csv file. If a MGRA-MGRA o-d pair is not present + in the MGRA-MGRA level then the TAZ-TAZ level skim is used. Note the + skims only provide time in minutes so distance is created using the + bicycle speed property set in the conf/sandag_abm.properties file. + Non-bicycle modes have all skims set to NaN. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA (1-23002) + [destinationMGRA] - trip destination MGRA (1-23002) + [originTAZ] - trip origin TAZ (1-4996) + [destinationTAZ] - trip destination TAZ (1-4996) + + Returns: + A Pandas DataFrame containing all associated bicycle mode + skims for time and distance. The DataFrame contains all the + original fields of the input DataFrame along with the fields: + [timeBike] - time in minutes for bicycle mode + [distanceBike] - distance in miles for bicycle mode """ + + # load the MGRA-MGRA bicycle skims + mgra_skims = pd.read_csv( + os.path.join(self.scenario_path, "output", "bikeMgraLogsum.csv"), + usecols=["i", # origin MGRA geography + "j", # destination MGRA geography + "time"], # time in minutes + dtype={"i": "int16", + "j": "int16", + "time": "float32"} + ) + + # load the TAZ-TAZ bicycle skims + taz_skims = pd.read_csv( + os.path.join(self.scenario_path, "output", "bikeTazLogsum.csv"), + usecols=["i", # origin TAZ geography + "j", # destination TAZ geography + "time"], # time in minutes + dtype={"i": "int16", + "j": "int16", + "time": "float32"} + ) + + # merge the skims with the input DataFrame bicycle mode records + # use left outer joins to keep all bicycle mode records + # if MGRA-MGRA skims do not exist + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + records = df.loc[(df["tripMode"] == "Bike")].copy() + records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + records = records.merge( + right=mgra_skims, + how="left", + left_on=["originMGRA", "destinationMGRA"], + right_on=["i", "j"] + ) + records = records.merge( + right=taz_skims, + how="left", + left_on=["originTAZ", "destinationTAZ"], + right_on=["i", "j"], + suffixes=["MGRA", "TAZ"] + ) + + # if MGRA-MGRA skims do not exist use TAZ-TAZ skims + records["timeBike"] = pd.Series( + np.where(records["timeMGRA"].isna(), + records["timeTAZ"], + records["timeMGRA"]), + dtype="float32") + + # calculate distance using bicycle speed + records["distanceBike"] = pd.Series( + records["timeBike"] * self.properties["bicycleSpeed"] / 60, + dtype="float32") + + # merge result set DataFrame back into initial trip list + # keep missing skim records as missing skim means no bike trip + records = records[["tripID", "timeBike", "distanceBike"]] + df = df.merge(records, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df + + def drive_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originTAZ], [destinationTAZ], + [boardingTAP], [alightingTAP], and [inbound]) and returns the + associated drive to transit auto-mode skims for time and distance. + + The model uses an on-the-fly input file (input\accessam.csv) for + drive to transit auto-mode skims and assumes no fare or toll costs. + Non-drive to transit modes have all skims set to NaN. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originTAZ] - trip origin TAZ (1-4996) + [destinationTAZ] - trip destination TAZ (1-4996) + [boardingTAP] - trip boarding transit access point (TAP) + [alightingTAP] - trip alighting transit access point (TAP) + [inbound] - direction of trip on tour (False, True) + + Returns: + A Pandas DataFrame containing all associated drive to transit + auto-mode skims for time and distance (assumed no toll cost). + The DataFrame contains all the original fields of the input + DataFrame along with the fields: + [timeDriveTransit] - time in minutes for auto mode + [distanceDriveTransit] - distance in miles for auto mode """ + + # read in the input accessam csv file + # this file is used in place of auto skim matrices for drive to transit + skims = pd.read_csv(self.scenario_path + "/input/accessam.csv", + names=["TAZ", # TAZ geography + "TAP", # transit access point (TAP) + "timeDriveTransit", # time in minutes + "distanceDriveTransit", # distance in miles + "mode"], + usecols=["TAZ", + "TAP", + "timeDriveTransit", + "distanceDriveTransit"], + dtype={"TAZ": "int16", + "TAP": "int16", + "timeDriveTransit": "float32", + "distanceDriveTransit": "float32"}) + + # select drive to transit records + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + modes = ["Park and Ride to Transit - Local Bus", + "Park and Ride to Transit - Premium Transit", + "Park and Ride to Transit - Local Bus and Premium Transit", + "Kiss and Ride to Transit - Local Bus", + "Kiss and Ride to Transit - Premium Transit", + "Kiss and Ride to Transit - Local Bus and Premium Transit", + "TNC to Transit - Local Bus", + "TNC to Transit - Premium Transit", + "TNC to Transit - Local Bus and Premium Transit"] + records = df.loc[(df["tripMode"].isin(modes))].copy() + records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + # create MGRA-TAP origin-destinations based on inbound direction + records["MGRA"] = np.where(records["inbound"], + records["destinationMGRA"], + records["originMGRA"]) + + records["TAP"] = np.where(records["inbound"], + records["alightingTAP"], + records["boardingTAP"]) + + # use the origin/destination MGRA to derive the origin/destination + # TAZ, this accounts for issues where external TAZs (1-12) do not have + # TAP-based skims, the skims are derived from the internal TAZ of the + # internal MGRA of the trip origin/destination + records = records.merge(self.mgra_xref, on="MGRA") + + # merge with the drive to transit access skims + records = records[["tripID", "TAZ", "TAP"]] + records = records.merge(skims, how="inner", on=["TAZ", "TAP"]) + records = records[["tripID", "timeDriveTransit", "distanceDriveTransit"]] + + # merge result set DataFrame back into initial trip list + # keep missing skim records as missing skim means no transit trip + df = df.merge(records, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df + + def omx_auto_skim_appender(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame and returns the DataFrame with + associated auto-mode skims for time, distance, and toll cost appended. + + The process uses the trip departure ABM 5 TOD period, trip mode, + transponder availability, and the trip value of time (vot) category + to select the correct auto skim-set to get the trip time, distance, + and toll costs. Non-auto modes have all skims set to NaN. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [transponderAvailable] - indicator if transponder available + on the trip (False, True) + [valueOfTimeCategory] - trip value of time categories + (Low, Medium, High) + [originTAZ] - trip origin TAZ (1-4996) + [destinationTAZ] - trip destination TAZ (1-4996) + [parkingTAZ] - trip parking TAZ ("", 1-4996) + + Returns: + The input Pandas DataFrame with auto-mode skims for time, + distance, and toll cost appended. The DataFrame contains all the + original fields of the input DataFrame along with the fields: + [timeDrive] - time in minutes for auto mode + [distanceDrive] - distance in miles for auto mode + [costTollDrive] - toll cost in $ for auto mode """ + + # prepare the input DataFrame for skim appending + # selecting records that use the input omx file + df_map = self._get_omx_auto_skim_dataset(df) + + # initialize output skim list + output = [] + + for omx_fn in df_map.omxFileName.unique(): + + # open the input omx file and TAZ:element mapping + fn = os.path.join(self.scenario_path, "output", omx_fn + ".omx") + omx_file = omx.open_file(fn) + omx_map = omx_file.mapping("zone_number") + + # filter records mapped to omx file + records_omx = df_map.loc[df_map.omxFileName == omx_fn] + + # for each skim matrix in the data-set + for matrix in records_omx.matrixName.unique(): + # filter records mapped to skim matrix + records = records_omx.loc[records_omx.matrixName == matrix] + + # create set of unique origin-destination pairs + # get time, distance, cost associated with the o-d pairs + od = set(zip(records.originTAZ, records.destinationTAZ)) + o, d = zip(*od) + + # map o-ds to omx matrix indices + o_idx = [omx_map[number] for number in o] + d_idx = [omx_map[number] for number in d] + + skims = list(zip( + [omx_fn] * len(o), + [matrix] * len(o), + o, d, + omx_file[matrix + "_TIME"][o_idx, d_idx], + omx_file[matrix + "_DIST"][o_idx, d_idx], + omx_file[matrix + "_TOLLCOST"][o_idx, d_idx] / 100)) + + output.extend(skims) + + omx_file.close() + + # create DataFrame from output skim list + output = pd.DataFrame(data=output, + columns=["omxFileName", + "matrixName", + "originTAZ", + "destinationTAZ", + "timeDrive", + "distanceDrive", + "costTollDrive"]) + + # set data types of output skims list + output = output.astype({ + "omxFileName": "category", + "matrixName": "category", + "originTAZ": "int16", + "destinationTAZ": "int16", + "timeDrive": "float32", + "distanceDrive": "float32", + "costTollDrive": "float32" + }) + + # merge the output skim list back to the mapped input DataFrame + # appending auto skims to tripIDs + output = output.merge(right=df_map, how="inner") + + output = output[["tripID", + "timeDrive", + "distanceDrive", + "costTollDrive"]] + + # merge the output skim list back to the original input DataFrame + # keep missing skim records as missing skim means no auto trip + df = df.merge(right=output, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df + + def omx_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the fields + ([tripID], [departTimeFiveTod], [tripMode], [boardingTAP], + [alightingTAP]) and returns the associated transit mode skims for time, + distance, and fare cost. + + The process uses the trip departure ABM 5 TOD period and trip mode to + select the correct transit skim-set to use to get the trip time, + distance, and fare costs. Non-transit modes have all skims set to NaN. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [boardingTAP] - trip boarding transit access point (TAP) + [alightingTAP] - trip alighting transit access point (TAP) + + Returns: + A Pandas DataFrame containing all associated transit-mode skims for + time, distance, and fare cost. The DataFrame contains all the + original fields of the input DataFrame along with the fields: + [timeTransitInVehicle] - transit in-vehicle time in minutes + [timeTier1TransitInVehicle] - tier 1 line haul mode transit + in-vehicle time in minutes + [timeFreewayRapidTransitInVehicle] - freeway rapid line haul + mode transit in-vehicle time in minutes + [timeArterialRapidTransitInVehicle] - arterial rapid line haul + mode transit in-vehicle time in minutes + [timeExpressBusTransitInVehicle] - express bus line haul + mode transit in-vehicle time in minutes + [timeLocalBusTransitInVehicle] - local bus line haul + mode transit in-vehicle time in minutes + [timeLightRailTransitInVehicle] - light rail line haul + mode transit in-vehicle time in minutes + [timeCommuterRailTransitInVehicle] - commuter rail line haul + mode transit in-vehicle time in minutes + [timeTransitInitialWait] - initial transit wait time in minutes + [timeTransitWait] - total transit wait time in minutes + [timeTransitWalk] - total transit walk time in minutes + [distanceTransitInVehicle] - transit in-vehicle distance in miles + [distanceTransitWalk] - total transit walk distance in miles + [costFareTransit] - fare cost in $ for transit mode + [transfersTransit] - number of transfers """ + + # create mappings from trip list to skim matrices + # departTimeFiveTod + tripMode + skim_map = {"values": [[1, 2, 3, 4, 5], + ["Walk to Transit - Local Bus", + "Walk to Transit - Premium Transit", + "Walk to Transit - Local Bus and Premium Transit", + "Park and Ride to Transit - Local Bus", + "Park and Ride to Transit - Premium Transit", + "Park and Ride to Transit - Local Bus and Premium Transit", + "Kiss and Ride to Transit - Local Bus", + "Kiss and Ride to Transit - Premium Transit", + "Kiss and Ride to Transit - Local Bus and Premium Transit", + "TNC to Transit - Local Bus", + "TNC to Transit - Premium Transit", + "TNC to Transit - Local Bus and Premium Transit"]], + "labels": [["EA", "AM", "MD", "PM", "EV"], + ["BUS", "PREM", "ALLPEN", + "BUS", "PREM", "ALLPEN", + "BUS", "PREM", "ALLPEN", + "BUS", "PREM", "ALLPEN"]], + "cols": ["departTimeFiveTod", + "tripMode", + "matrixName"]} + + # initialize empty result set DataFrame + result = pd.DataFrame() + + # map possible values of trip list columns to skim matrix names + mapping = [list(i) + ["_".join(j)] for i, j in + zip(itertools.product(*skim_map["values"]), + itertools.product(*skim_map["labels"]))] + + # create Pandas DataFrame lookup table of column values + # to skim matrix names + lookup = pd.DataFrame(mapping, columns=skim_map["cols"]) + + # set data types of lookup table + lookup = lookup.astype({ + "departTimeFiveTod": "int8", + "tripMode": "category", + "matrixName": "category" + }) + + # merge lookup table to trip list make DataFrame unique by tripID + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + trips = df.merge(lookup, how="inner") + trips.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + # open omx transit skim file and get TAP:element matrix mapping + omx_file = omx.open_file(self.scenario_path + "/output/transit_skims.omx") + omx_map = omx_file.mapping("zone_number") + + # for each skim matrix in the data-set + for matrix in trips.matrixName.unique(): + # select records that use the skim matrix + records = trips.loc[(trips["matrixName"] == matrix)].copy() + + # get lists of o-ds + o = records.boardingTAP.astype("int16").tolist() + d = records.alightingTAP.astype("int16").tolist() + + # map o-ds to omx matrix indices + o_idx = [omx_map[number] for number in o] + d_idx = [omx_map[number] for number in d] + + # append skims + records["timeTransitInVehicle"] = omx_file[matrix + "_TOTALIVTT"][o_idx, d_idx] + records["timeTier1TransitInVehicle"] = omx_file[matrix + "_TIER1IVTT"][o_idx, d_idx] + records["timeFreewayRapidTransitInVehicle"] = omx_file[matrix + "_BRTYELIVTT"][o_idx, d_idx] + records["timeArterialRapidTransitInVehicle"] = omx_file[matrix + "_BRTREDIVTT"][o_idx, d_idx] + records["timeExpressBusTransitInVehicle"] = omx_file[matrix + "_EXPIVTT"][o_idx, d_idx] + records["timeLocalBusTransitInVehicle"] = omx_file[matrix + "_BUSIVTT"][o_idx, d_idx] + records["timeLightRailTransitInVehicle"] = omx_file[matrix + "_LRTIVTT"][o_idx, d_idx] + records["timeCommuterRailTransitInVehicle"] = omx_file[matrix + "_CMRIVTT"][o_idx, d_idx] + records["timeTransitInitialWait"] = omx_file[matrix + "_FIRSTWAIT"][o_idx, d_idx] + records["timeTransitWait"] = omx_file[matrix + "_TOTALWAIT"][o_idx, d_idx] + records["timeTransitWalk"] = omx_file[matrix + "_TOTALWALK"][o_idx, d_idx] + records["distanceTransitInVehicle"] = omx_file[matrix + "_TOTDIST"][o_idx, d_idx] + records["costFareTransit"] = omx_file[matrix + "_FARE"][o_idx, d_idx] + records["transfersTransit"] = omx_file[matrix + "_XFERS"][o_idx, d_idx] + records["distanceTransitWalk"] = records.timeTransitWalk * self.properties["walkSpeed"] / 60 + + # set skim data types + records = records.astype({ + "timeTransitInVehicle": "float32", + "timeTransitInitialWait": "float32", + "timeTransitWait": "float32", + "timeTransitWalk": "float32", + "distanceTransitInVehicle": "float32", + "costFareTransit": "float32", + "transfersTransit": "float32", + "distanceTransitWalk": "float32"}) + + records = records[["tripID", + "timeTransitInVehicle", + "timeTier1TransitInVehicle", + "timeFreewayRapidTransitInVehicle", + "timeArterialRapidTransitInVehicle", + "timeExpressBusTransitInVehicle", + "timeLocalBusTransitInVehicle", + "timeLightRailTransitInVehicle", + "timeCommuterRailTransitInVehicle", + "timeTransitInitialWait", + "timeTransitWait", + "timeTransitWalk", + "distanceTransitInVehicle", + "distanceTransitWalk", + "costFareTransit", + "transfersTransit"]] + + result = result.append(records, ignore_index=True) + + omx_file.close() + + skim_cols = ["timeTransitInVehicle", + "timeTier1TransitInVehicle", + "timeFreewayRapidTransitInVehicle", + "timeArterialRapidTransitInVehicle", + "timeExpressBusTransitInVehicle", + "timeLocalBusTransitInVehicle", + "timeLightRailTransitInVehicle", + "timeCommuterRailTransitInVehicle", + "timeTransitInitialWait", + "timeTransitWait", + "timeTransitWalk", + "distanceTransitInVehicle", + "distanceTransitWalk", + "costFareTransit", + "transfersTransit"] + + # if there are no transit trip skim records + if result.empty: + # append skim columns to input DataFrame + # set to missing as a missing skim means no transit trip + for col in skim_cols: + df[col] = np.NaN + else: + # merge result set DataFrame back into initial trip list + # keep missing skim records as missing skim means no transit trip + df = df.merge(result, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df + + def tnc_fare_cost(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [avUsed]) and the fields appended by the + SkimAppender class method omx_auto_skim_appender + ([timeDrive], [distanceDrive]) and returns the associated + fare costs. + + Note that drive to transit modes assume no fare costs and are not + included here. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [timeDrive] - time in minutes for auto mode + [distanceDrive] - distance in miles for auto mode + + Returns: + A Pandas DataFrame containing all associated auto-mode fare + costs. The DataFrame contains all original fields of the input + DataFrame along with the field: + [costFareDrive] - auto mode fare cost in $ """ + + # calculate fare costs for non-pooled TNC, pooled TNC, and taxi modes + conditions = [df["tripMode"] == "Non-Pooled TNC", + df["tripMode"] == "Pooled TNC", + df["tripMode"] == "Taxi"] + + # calculate non-pooled TNC fare incorporating minimum fare + non_pooled_tnc_fare = pd.Series( + self.properties["baseFareNonPooledTNC"] + + self.properties["costPerMileNonPooledTNC"] * df["distanceDrive"] + + self.properties["costPerMinuteNonPooledTNC"] * df["timeDrive"], + dtype="float32") + + non_pooled_tnc_fare = np.where( + non_pooled_tnc_fare < self.properties["costMinimumNonPooledTNC"], + self.properties["costMinimumNonPooledTNC"], + non_pooled_tnc_fare) + + # calculate pooled TNC fare incorporating minimum fare + pooled_tnc_fare = pd.Series( + self.properties["baseFarePooledTNC"] + + self.properties["costPerMilePooledTNC"] * df["distanceDrive"] + + self.properties["costPerMinutePooledTNC"] * df["timeDrive"], + dtype="float32") + + pooled_tnc_fare = np.where( + pooled_tnc_fare < self.properties["costMinimumPooledTNC"], + self.properties["costMinimumPooledTNC"], + pooled_tnc_fare) + + choices = [ + non_pooled_tnc_fare, + pooled_tnc_fare, + self.properties["baseFareTaxi"] + + self.properties["costPerMileTaxi"] * df["distanceDrive"] + + self.properties["costPerMinuteTaxi"] * df["timeDrive"] + ] + + df["costFareDrive"] = pd.Series( + np.select(conditions, choices, default=np.NaN), + dtype="float32") + + # return input DataFrame with appended auto fare cost column + return df + + def tnc_wait_time(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originMGRA]) and returns the associated wait + times for Taxi and TNC modes. + + Note that drive to transit modes assume no auto-mode wait time and + are not included here. + + Note that the actual wait times experienced by trips are drawn from a + distribution but not written out to the trip output files making it + impossible to write out the actual wait time experienced. The mean is + used here for all trips as an approximation. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA geography + + Returns: + A Pandas DataFrame containing all associated auto-mode wait times. + The DataFrame contains all original fields of the input DataFrame + along with the field: + [timeWaitDrive] - auto mode wait time in minutes """ + + # load the mgra based input file + fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" + + mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), + usecols=["mgra", # MGRA geography + "PopEmpDenPerMi"], # density per mi + dtype={"mgra": "int16", + "PopEmpDenPerMi": "float32"}) + + # add PopEmpDenPerMi field to the input DataFrame + df = df.merge(mgra, left_on="originMGRA", right_on="mgra") + + # select mean wait time for Taxi/TNC mode trips based on + # the category the PopEmpDenPerMi value falls in + # note the first true condition encountered is chosen + conditions = [ + ((df["tripMode"] == "Non-Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), + ((df["tripMode"] == "Non-Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), + ((df["tripMode"] == "Non-Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), + ((df["tripMode"] == "Non-Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), + ((df["tripMode"] == "Non-Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])), + ((df["tripMode"] == "Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), + ((df["tripMode"] == "Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), + ((df["tripMode"] == "Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), + ((df["tripMode"] == "Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), + ((df["tripMode"] == "Pooled TNC") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])), + ((df["tripMode"] == "Taxi") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), + ((df["tripMode"] == "Taxi") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), + ((df["tripMode"] == "Taxi") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), + ((df["tripMode"] == "Taxi") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), + ((df["tripMode"] == "Taxi") & ( + df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])) + ] + + choices = [ + *self.properties["waitTimeNonPooledTNC"], + *self.properties["waitTimePooledTNC"], + *self.properties["waitTimeTaxi"], + ] + + df["timeWaitDrive"] = pd.Series( + np.select(conditions, choices, default=np.NaN), + dtype="float32") + + # remove mgra, PopEmpDenPerMi from the input DataFrame + df.drop(columns=["mgra", "PopEmpDenPerMi"], inplace=True) + + # return input DataFrame with appended auto wait time column + return df + + def walk_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originMGRA], [destinationMGRA]) and returns + the associated micro-mobility, micro-transit, and walk mode skims for + time, distance, and fare cost. Non-micro-mobility/micro-transit/walk + modes have all skims set to NaN. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA (1-23002) + [destinationMGRA] - trip destination MGRA (1-23002) + + Returns: + A Pandas DataFrame containing all associated micro-mobility, + micro-transit, and walk mode skims for time, distance, and + fare cost. The DataFrame contains all original fields of the + input DataFrame along with the field: + [timeWalk] - time in minutes for walk mode + [distanceWalk] - distance in miles for walk mode + [timeMM] - time in minutes for micro-mobility mode + [distanceMM] - distance in miles for micro-mobility mode + [costFareMM] - fare cost in dollars for micro-mobility mode + [timeMT] - time in minutes for micro-transit mode + [distanceMT] - distance in miles for micro-transit mode + [costFareMT] - fare cost in dollars for micro-transit mode + """ + # get skims for walk/micro-mobility/micro-transit trips + # some use AT skim sets while others use auto skim set + records_at = self._walk_skims_at(df) + records_auto = self._walk_skims_auto(df) + + # if there are no mm/mt/walk trip skim records + if records_at.empty and records_auto.empty: + # append skim columns to input DataFrame + # set to missing as a missing skim means no walk trip + skim_cols = ["timeWalk", + "distanceWalk", + "timeMM", + "distanceMM", + "costFareMM", + "timeMT", + "distanceMT", + "costFareMT"] + + for col in skim_cols: + df[col] = np.NaN + else: + # merge result set DataFrame back into initial trip list + # keep missing skim records as missing skim means no walk trip + records = records_at.append(records_auto, ignore_index=True) + df = df.merge(records, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df + + def _walk_skims_at(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originMGRA], [destinationMGRA]) and returns + the associated micro-mobility, micro-transit, and walk mode skims for + time, distance, and fare cost. Non-micro-mobility/micro-transit/walk + modes have all skims set to NaN. Note that some micro-mobility, + micro-transit and walk mode trips use an auto mode skim to define + time, distance, and fare cost. These are removed from consideration + in this method and handled elsewhere. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA (1-23002) + [destinationMGRA] - trip destination MGRA (1-23002) + + Returns: + A Pandas DataFrame containing all associated micro-mobility, + micro-transit, and walk mode skims for time, distance, and + fare cost. The DataFrame contains only micro-mobility, + micro-transit, and walk mode trip records that do not use the + auto mode skim set for time: + [tripID] - unique identifier of a trip + [timeWalk] - time in minutes for walk mode + [distanceWalk] - distance in miles for walk mode + [timeMM] - time in minutes for micro-mobility mode + [distanceMM] - distance in miles for micro-mobility mode + [costFareMM] - fare cost in dollars for micro-mobility mode + [timeMT] - time in minutes for micro-transit mode + [distanceMT] - distance in miles for micro-transit mode + [costFareMT] - fare cost in dollars for micro-transit mode + """ + # load the mgra-mgra walk/micro-mobility/micro-transit skim file + skims = pd.read_csv(os.path.join(self.scenario_path, + "output", + "microMgraEquivMinutes.csv"), + usecols=["i", # origin MGRA geography + "j", # destination MGRA geography + "walkTime", # walk time in minutes + "dist", # distance in miles + "mmTime", # micro-mobility time in minutes + "mmCost", # micro-mobility cost in dollars + "mtTime", # micro-transit time in minutes + "mtCost"], # micro-transit cost in dollars + dtype={"i": "int16", + "j": "int16", + "walkTime": "float32", + "dist": "float32", + "mmTime": "float32", + "mmCost": "float32", + "mtTime": "float32", + "mtCost": "float32"}) + + # merge the skims with the input DataFrame walk/mm/mt mode records + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + records = df.loc[(df["tripMode"].isin(["Micro-Mobility", + "Micro-Transit", + "Walk"]))].copy() + + records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + records = records.merge( + right=skims, + how="inner", # some of these trips can use auto skims, use inner join to remove them + left_on=["originMGRA", "destinationMGRA"], + right_on=["i", "j"] + ) + + # set skims based on mode + records["timeWalk"] = np.where(records["tripMode"] == "Walk", + records["walkTime"], + 0) + + records["distanceWalk"] = np.where(records["tripMode"] == "Walk", + records["dist"], + 0) + + records["timeMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["mmTime"], + 0) + + records["distanceMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["dist"], + 0) + + records["costFareMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["mmCost"], + 0) + + records["timeMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["mtTime"], + 0) + + records["distanceMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["dist"], + 0) + + records["costFareMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["mtCost"], + 0) + + # return result set of walk/micro-mobility/micro-transit trips + # that use AT skim sets + return records[["tripID", + "timeWalk", + "distanceWalk", + "timeMM", + "distanceMM", + "costFareMM", + "timeMT", + "distanceMT", + "costFareMT"]] + + def _walk_skims_auto(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [tripMode], [originMGRA], [originTAZ] [destinationMGRA], + [destinationTAZ]) and returns the associated micro-mobility, + micro-transit, and walk mode skims for time, distance, and fare cost + for micro-mobility, micro-transit, and walk mode trips that use + auto mode skim set for time. Note that micro-mobility, micro-transit, + and walk mode trips that do not use auto mode skim set for time are + removed from consideration and handled elsewhere. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA (1-23002) + [originTAZ] - trip origin TAZ (1-4996) + [destinationMGRA] - trip destination MGRA (1-23002) + [destinationTAZ] - trip destination TAZ (1-4996) + + Returns: + A Pandas DataFrame containing all associated micro-mobility, + micro-transit, and walk mode skims for time, distance, and + fare cost. The DataFrame contains only micro-mobility, + micro-transit, and walk mode trip records that use the auto + mode skim set for time: + [tripID] - unique identifier of a trip + [timeWalk] - time in minutes for walk mode + [distanceWalk] - distance in miles for walk mode + [timeMM] - time in minutes for micro-mobility mode + [distanceMM] - distance in miles for micro-mobility mode + [costFareMM] - fare cost in dollars for micro-mobility mode + [timeMT] - time in minutes for micro-transit mode + [distanceMT] - distance in miles for micro-transit mode + [costFareMT] - fare cost in dollars for micro-transit mode + """ + # load the mgra-mgra walk/micro-mobility/micro-transit skim file + skims = pd.read_csv(os.path.join(self.scenario_path, + "output", + "microMgraEquivMinutes.csv"), + usecols=["i", # origin MGRA geography + "j", # destination MGRA geography + "walkTime", # walk time in minutes + "dist", # distance in miles + "mmTime", # micro-mobility time in minutes + "mmCost", # micro-mobility cost in dollars + "mtTime", # micro-transit time in minutes + "mtCost"], # micro-transit cost in dollars + dtype={"i": "int16", + "j": "int16", + "walkTime": "float32", + "dist": "float32", + "mmTime": "float32", + "mmCost": "float32", + "mtTime": "float32", + "mtCost": "float32"}) + + # merge the skims with the input DataFrame walk/mm/mt mode records + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + records = df.loc[(df["tripMode"].isin(["Micro-Mobility", + "Micro-Transit", + "Walk"]))].copy() + + records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + # select records that are NOT in the mgra-mgra + # walk/micro-mobility/micro-transit skim file + records = records.merge( + right=skims, + how="left", # use outer join to keep all trip records + left_on=["originMGRA", "destinationMGRA"], + right_on=["i", "j"], + indicator=True + ) + + records = records[records["_merge"] == "left_only"] + + # if there are no eligible records return an empty DataFrame + if records.empty: + return pd.DataFrame( + columns=["tripID", + "timeWalk", + "distanceWalk", + "timeMM", + "distanceMM", + "costFareMM", + "timeMT", + "distanceMT", + "costFareMT"] + ) + else: + # append trip time from auto skim set + # midday drive alone non-transponder low value of time + + # open omx transit skim file and get TAZ matrix mapping + omx_file = omx.open_file(self.scenario_path + "/output/traffic_skims_MD.omx") + omx_map = omx_file.mapping("zone_number") + + # get lists of o-ds + o = records.originTAZ.astype("int16").tolist() + d = records.destinationTAZ.astype("int16").tolist() + + # map o-ds to omx matrix indices + o_idx = [omx_map[number] for number in o] + d_idx = [omx_map[number] for number in d] + + # append travel time skim from auto skim set + records["sovTime"] = omx_file["MD_SOV_NT_M_TIME"][o_idx, d_idx] + + omx_file.close() + + # load the MGRA-MGRA based input file + # merge with trips to get micro-mobility access time for origin MGRAs + records = records.merge( + right=self.mgra_xref, + how="inner", + left_on="originMGRA", + right_on="MGRA" + ) + + # set skims based on mode + records["timeWalk"] = np.where(records["tripMode"] == "Walk", + records["sovTime"], + 0) + + records["distanceWalk"] = np.where(records["tripMode"] == "Walk", + records["sovTime"] * self.properties["walkSpeed"] / 60, + 0) + + records["timeMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["sovTime"] + records["MicroAccessTime"], + 0) + + records["distanceMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["sovTime"] * self.properties["microMobilitySpeed"] / 60, + 0) + + records["costFareMM"] = np.where(records["tripMode"] == "Micro-Mobility", + records["sovTime"] * self.properties["costPerMinuteMicroMobility"] + + self.properties["baseFareMicroMobility"], + 0) + + records["timeMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["sovTime"] + self.properties["accessTimeMicroTransit"] + + self.properties["waitTimeMicroTransit"], + 0) + + records["distanceMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["sovTime"] * self.properties["microTransitSpeed"] / 60, + 0) + + records["costFareMT"] = np.where(records["tripMode"] == "Micro-Transit", + records["sovTime"] * self.properties["costPerMinuteMicroTransit"] + + self.properties["baseFareMicroTransit"], + 0) + + # return result set of walk/micro-mobility/micro-transit trips + # that use auto mode skim set + return records[["tripID", + "timeWalk", + "distanceWalk", + "timeMM", + "distanceMM", + "costFareMM", + "timeMT", + "distanceMT", + "costFareMT"]] + + def walk_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: + """ Takes an input Pandas DataFrame containing the ABM fields + ([tripID], [inbound], [tripMode], [originMGRA], [destinationMGRA], + [boardingTAP], [alightingTAP]) and the optional fields + ([microMobilityTransitAccess], [microMobilityTransitEgress]) and + returns the associated micro-mobility, micro-transit, and walk mode + skims for transit access/egress for time, distance, and fare cost. + Trips without micro-mobility, micro-transit, or walk mode + access/egress to transit have all skims set to 0. + + Args: + df: Input Pandas DataFrame containing the fields + [tripID] - unique identifier of a trip + [inbound] - boolean indicator of inbound/outbound direction + [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) + [originMGRA] - trip origin MGRA (1-23002) + [destinationMGRA] - trip destination MGRA (1-23002) + [boardingTAP] - transit boarding TAP + [alightingTAP] - transit alighting TAP + [microMobilityTransitAccess] - optional field indicating if + micro-mobility, micro-transit, or walk mode was used to + access transit + [microMobilityTransitEgress] - optional field indicating if + micro-mobility, micro-transit, or walk mode was used to + egress transit + + Returns: + A Pandas DataFrame containing all associated micro-mobility, + micro-transit, and walk to transit mode access/egress skims for + time, distance, and fare cost. The DataFrame contains all the + original fields of the input DataFrame along with the fields: + [timeTransitWalkAccessEgress] - time in minutes for walk + portion of transit access/egress + [distanceTransitWalkAccessEgress] - distance in miles for + walk portion of transit access/egress + [timeTransitMMAccessEgress] - time in minutes for + micro-mobility portion of transit access/egress + [distanceTransitMMAccessEgress] - distance in miles for + micro-mobility portion of transit access/egress + [costFareTransitMMAccessEgress] - fare cost in dollars for + micro-mobility portion of transit access/egress + [timeTransitMTAccessEgress] - time in minutes for + micro-transit portion of transit access/egress + [distanceTransitMTAccessEgress] - distance in miles for + micro-transit portion of transit access/egress + [costFareTransitMTAccessEgress] - fare cost in dollars for + micro-transit portion of transit access/egress """ + + # load the micro-mobility, micro-transit, and walk from MGRA-TAP + # skim file containing time, distance, and cost + skims = pd.read_csv(os.path.join(self.scenario_path, + "output", + "microMgraTapEquivMinutes.csv"), + usecols=["mgra", # origin MGRA geography + "tap", # destination TAP + "walkTime", # walk time in minutes + "dist", # distance in miles + "mmTime", # micro-mobility time in minutes + "mmCost", # micro-mobility cost in dollars + "mtTime", # micro-transit time in minutes + "mtCost"], # micro-transit cost in dollars + dtype={"mgra": "int16", + "tap": "int16", + "walkTime": "float32", + "dist": "float32", + "mmTime": "float32", + "mmCost": "float32", + "mtTime": "float32", + "mtCost": "float32"}) + + # filter input DataFrame to records that have access/egress + # micro-mobility, micro-transit, or walk segments + records = df.loc[(df["tripMode"].isin( + ["Walk to Transit - Local Bus", + "Walk to Transit - Premium Transit", + "Walk to Transit - Local Bus and Premium Transit", + "Park and Ride to Transit - Local Bus", + "Park and Ride to Transit - Premium Transit", + "Park and Ride to Transit - Local Bus and Premium Transit", + "Kiss and Ride to Transit - Local Bus", + "Kiss and Ride to Transit - Premium Transit", + "Kiss and Ride to Transit - Local Bus and Premium Transit", + "TNC to Transit - Local Bus", + "TNC to Transit - Premium Transit", + "TNC to Transit - Local Bus and Premium Transit"]))].copy() + + # ABM Joint sub-model has multiple records per tripID + # per person on trip records are identical otherwise + records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) + + # merge micro-mobility, micro-transit, and walk access/egress skims + # for both the access and egress portions + records = records.merge( + right=skims, + how="left", + left_on=["originMGRA", "boardingTAP"], + right_on=["mgra", "tap"] + ) + + records = records.merge( + right=skims, + how="left", + left_on=["destinationMGRA", "alightingTAP"], + right_on=["mgra", "tap"], + suffixes=["Access", "Egress"] + ) + + # conditionally set access/egress skim fields to 0 based on trip mode + # and inbound direction of trip, note that walk to transit uses both + records.loc[(records["tripMode"].isin( + ["Park and Ride to Transit - Local Bus", + "Park and Ride to Transit - Premium Transit", + "Park and Ride to Transit - Local Bus and Premium Transit", + "Kiss and Ride to Transit - Local Bus", + "Kiss and Ride to Transit - Premium Transit", + "Kiss and Ride to Transit - Local Bus and Premium Transit", + "TNC to Transit - Local Bus", + "TNC to Transit - Premium Transit", + "TNC to Transit - Local Bus and Premium Transit"])) & + (records["inbound"] == False), + ["walkTimeAccess", + "distAccess", + "mmTimeAccess", + "mmCostAccess", + "mtTimeAccess", + "mtCostAccess"]] = 0 + + records.loc[(records["tripMode"].isin([ + "Park and Ride to Transit - Local Bus", + "Park and Ride to Transit - Premium Transit", + "Park and Ride to Transit - Local Bus and Premium Transit", + "Kiss and Ride to Transit - Local Bus", + "Kiss and Ride to Transit - Premium Transit", + "Kiss and Ride to Transit - Local Bus and Premium Transit", + "TNC to Transit - Local Bus", + "TNC to Transit - Premium Transit", + "TNC to Transit - Local Bus and Premium Transit"])) & + (records["inbound"] == True), + ["walkTimeEgress", + "distEgress", + "mmTimeEgress", + "mmCostEgress", + "mtTimeEgress", + "mtCostEgress"]] = 0 + + # if optional micro-mobility access/egress fields are not present + # assume Walk modes only for access/egress skims + if not {"microMobilityTransitAccess", "microMobilityTransitEgress"}.issubset(df.columns): + records["timeTransitWalkAccessEgress"] = records["walkTimeAccess"] + records["walkTimeEgress"] + records["distanceTransitWalkAccessEgress"] = records["distAccess"] + records["distEgress"] + records["timeTransitMMAccessEgress"] = 0 + records["distanceTransitMMAccessEgress"] = 0 + records["costFareTransitMMAccessEgress"] = 0 + records["timeTransitMTAccessEgress"] = 0 + records["distanceTransitMTAccessEgress"] = 0 + records["costFareTransitMTAccessEgress"] = 0 + else: # if optional micro-mobility access/egress fields are present + # set skim fields based on micro-mobility access/egress fields + records["timeTransitWalkAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Walk", + records["walkTimeAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Walk", + records["walkTimeEgress"], 0) + + records["distanceTransitWalkAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Walk", + records["distAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Walk", + records["distEgress"], 0) + + records["timeTransitMMAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", + records["mmTimeAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", + records["mmTimeEgress"], 0) + + records["distanceTransitMMAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", + records["distAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", + records["distEgress"], 0) + + records["costFareTransitMMAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", + records["mmCostAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", + records["mmCostEgress"], 0) + + records["timeTransitMTAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Transit", + records["mtTimeAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Transit", + records["mtTimeEgress"], 0) + + records["distanceTransitMTAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Transit", + records["distAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Transit", + records["distEgress"], 0) + + records["costFareTransitMTAccessEgress"] = \ + np.where(records["microMobilityTransitAccess"] == "Micro-Transit", + records["mtCostAccess"], 0) + \ + np.where(records["microMobilityTransitEgress"] == "Micro-Transit", + records["mtCostEgress"], 0) + + # merge result set DataFrame back into initial trip list + records = records[["tripID", + "timeTransitWalkAccessEgress", + "distanceTransitWalkAccessEgress", + "timeTransitMMAccessEgress", + "distanceTransitMMAccessEgress", + "costFareTransitMMAccessEgress", + "timeTransitMTAccessEgress", + "distanceTransitMTAccessEgress", + "costFareTransitMTAccessEgress"]] + + # access/egress transit trip + skim_cols = ["timeTransitWalkAccessEgress", + "distanceTransitWalkAccessEgress", + "timeTransitMMAccessEgress", + "distanceTransitMMAccessEgress", + "costFareTransitMMAccessEgress", + "timeTransitMTAccessEgress", + "distanceTransitMTAccessEgress", + "costFareTransitMTAccessEgress"] + + # if there are no mm/mt/walk access/egress skim records + if records.empty: + # append skim columns to input DataFrame + # set to missing as a missing skim means no transit trip + for col in skim_cols: + df[col] = np.NaN + else: + # merge result set DataFrame back into initial trip list + # keep missing skim records as missing skim means no transit trip + df = df.merge(records, on="tripID", how="left") + + # return input DataFrame with appended skim columns + return df diff --git a/sandag_abm/src/main/python/database_summary.py b/sandag_abm/src/main/python/database_summary.py new file mode 100644 index 0000000..63f441f --- /dev/null +++ b/sandag_abm/src/main/python/database_summary.py @@ -0,0 +1,190 @@ +# __author__ = 'yma' +# This file is to get data summary from the database, 1/23/2019 + +import openpyxl +from openpyxl import load_workbook +from datetime import datetime +from pandas import DataFrame +import pandas as pd +import pyodbc +import sys +import os + +# usage = ("Correct Usage: database_summary.py ") + +# check if too few/many arguments passed raise error +if len(sys.argv) != 4: + sys.exit(-1) + +output_path = str(sys.argv[1]) +sceYear = int(sys.argv[2]) +scenario = int(sys.argv[3]) + +# create settings data frame +settings = {"Parameter": ["Date", "Scenario_id","Year"], + "Value": [datetime.now().strftime("%x %X"),scenario,sceYear]} +settings = pd.DataFrame(data=settings) + + +# set sql server connection to ws +sql_con = pyodbc.connect(driver='{SQL Server}', + server='sql2014a8', + database='ws', + trusted_connection='yes') + + +### data summary for sensitivty analysis + +# 0_scenarioInfor +scenarioInfor = pd.read_sql_query( + sql="exec [ws].[sst2].[m0_scenario] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 1_modeshare +modeShare = pd.read_sql_query( + sql="exec [ws].[sst2].[m1_mode_share] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 4_PtripLengthByPurpose +ptripLengthByPurpose = pd.read_sql_query( + sql="exec [ws].[sst2].[m4_ptrip_distance_purpose] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 5_PtripLengthByMode +ptripDistanceMode = pd.read_sql_query( + sql="exec [ws].[sst2].[m5_ptrip_distance_mode] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 23_VMT +VMT = pd.read_sql_query( + sql="exec [ws].[sst2].[m23_vmt_capita] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 6_VHT +VHT = pd.read_sql_query( + sql="exec [ws].[sst2].[m6_vht_capita] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 7_VHD +VHD = pd.read_sql_query( + sql="exec [ws].[sst2].[m7_vhd_capita] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 8_TransitBoardingByMode +transitBoardingLinehaulMode = pd.read_sql_query( + sql="exec [ws].[sst2].[m8_transit_boarding_linehaulmode] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + +# 9_TransitTripsbyMode +transitTripsbyMode = pd.read_sql_query( + sql="exec [ws].[sst2].[m9_transit_trips_by_mode] @scenario_id = ? ", + con=sql_con, + params=[scenario] +) + + +# output to excel book 'source_sensitivity' +output_file_sens = output_path + "\\analysis\\summary\\source_summary.xlsx" +book = load_workbook(output_file_sens) + +with pd.ExcelWriter(output_file_sens, engine='openpyxl') as writer: + + writer.book = book + writer.sheets = dict((ws.title, ws) for ws in book.worksheets) + + settings.to_excel(writer,'Settings',index=False) + scenarioInfor.to_excel(writer,'0_scenarioInfor', index=False) + modeShare.to_excel(writer,'1_modeshare', index=False) + ptripLengthByPurpose.to_excel(writer, '4_PtripLengthByPurpose', index=False) + ptripDistanceMode.to_excel(writer, '5_PtripLengthByMode', index=False) + VMT.to_excel(writer, '23_VMT', index=False) + VHT.to_excel(writer, '6_VHT', index=False) + VHD.to_excel(writer, '7_VHD', index=False) + transitBoardingLinehaulMode.to_excel(writer, '8_TransitBoardingByMode', index=False) + transitTripsbyMode.to_excel(writer, '9_TransitTripsbyMode', index=False) + + writer.save() + + +### data summary for validation analysis, if scenario_year = 2016 or 2018 + +if sceYear == 2016 or sceYear == 2018: + + # get hwy flow given scenario_id + hwyFlow = pd.read_sql_query( + sql="select * from [ws].[validate2].[FlowDay2016_nonmsa] (?)", + con=sql_con, + params=[scenario] + ) + + # get freeway flow by TOD given scenario_id + fwyFlow = pd.read_sql_query( + sql="select * from [ws].[validate2].[FlowFreewayTod2016] (?)", + con=sql_con, + params=[scenario] + ) + + # get truck flow given scenario_id + truckFlow = pd.read_sql_query( + sql="select * from [ws].[validate2].[TruckFlow2016] (?)", + con=sql_con, + params=[scenario] + ) + + # get hwy speed given scenario_id + hwySpeed = pd.read_sql_query( + sql="select * from [ws].[validate2].[SpeedFreewayTod2016] (?)", + con=sql_con, + params=[scenario] + ) + + # get transit general summary given scenario_id + transitGeneral = pd.read_sql_query( + sql="select * from [ws].[validate2].[transit2016] (?)", + con=sql_con, + params=[scenario] + ) + + # get transit hub summary given scenario_id + transitHub = pd.read_sql_query( + sql="select * from [ws].[validate2].[transit2016_Hub] (?)", + con=sql_con, + params=[scenario] + ) + + + # output to excel book 'source' + output_file_vald = output_path + "\\analysis\\validation\\source.xlsx" + book = load_workbook(output_file_vald) + + with pd.ExcelWriter(output_file_vald, engine='openpyxl') as writer: + + writer.book = book + writer.sheets = dict((ws.title, ws) for ws in book.worksheets) + + settings.to_excel(writer,'Settings',index=False) + hwyFlow.to_excel(writer,'flow_raw', index=False) + fwyFlow.to_excel(writer,'flow_freeway', index=False) + truckFlow.to_excel(writer,'truck_flow', index=False) + hwySpeed.to_excel(writer,'hwy_speed', index=False) + transitGeneral.to_excel(writer,'transit_general', index=False) + transitHub.to_excel(writer,'transit_hub', index=False) + + writer.save() + diff --git a/sandag_abm/src/main/python/excel_update.py b/sandag_abm/src/main/python/excel_update.py new file mode 100644 index 0000000..ad17762 --- /dev/null +++ b/sandag_abm/src/main/python/excel_update.py @@ -0,0 +1,85 @@ +# __author__ = 'yma' +# This file is to force to update links in excel, 1/23/2019 + +import os +import sys +import win32com.client +from win32com.client import Dispatch + +# usage = ("Correct Usage: excel_update.py ") + +# check if too few/many arguments passed raise error +if len(sys.argv) != 4: + sys.exit(-1) + +output_path = str(sys.argv[1]) +sceYear = int(sys.argv[2]) +scenario = str(sys.argv[3]) + + +path_name_s = output_path + "\\analysis\\summary\\" +file_names_s = ["ModelResultSummary"] + +path_name_v = output_path + "\\analysis\\validation\\" +file_names_v = ["HighwayAssignmentValidation_2016_AllClass_EMME", + "HighwayAssignmentValidation_2016_Truck_EMME", + "HighwayAssignmentValidation_2016_Speed_FreewayCorridor_AMPM_EMME", + "HighwayAssignmentValidation_2016_FreewayCorridor_Daily_EMME", + "HighwayAssignmentValidation_2016_FreewayCorridor_AM_EMME", + "HighwayAssignmentValidation_2016_FreewayCorridor_PM_EMME", + "TransitAssignmentValidation_2016_General_EMME", + #"TransitAssignmentValidation_2016_Hub", + ] +ext = ".xlsm" + +# file list of sensitivity +list_s = [] +list_s_new = [] +for file in file_names_s: + file_s = path_name_s + file + ext + #file_s_new = path_name_s + file + "_" + scenario + ext + file_s_new = path_name_s + file + ext + list_s.append(file_s) + list_s_new.append(file_s_new) + +# file list of validation +list_v = [] +list_v_new = [] +for file in file_names_v: + file_v = path_name_v + file + ext + #file_v_new = path_name_v + file + "_" + scenario + ext # since the data was read from EMME, scenario is not needed - CL 05222020 + file_v_new = path_name_v + file + ext + list_v.append(file_v) + list_v_new.append(file_v_new) + + +file_list = list_s +file_list_new = list_s_new + +if sceYear == 2016 or sceYear == 2018: + file_list = file_list + list_v + file_list_new = file_list_new + list_v_new + + +xl = Dispatch("Excel.Application") +xl.Visible = False +xl.DisplayAlerts = False +xl.AskToUpdateLinks = False + +for file in file_list: + wb = xl.workbooks.open(file) + xl.Application.Run("UpdateLinks") + wb.Close(True) + +EMME = os.path.join(path_name_v, 'source_EMME.xlsx') #with source_EMME.xlsx opened, links in validation Excel files will work well. This is a temporary solution - CL 05222020 +wb_emme = xl.Workbooks.Open(EMME) +for file in file_list: + wb = xl.workbooks.open(file) + xl.Application.Run("UpdateLinks") + wb.Close(True) +wb_emme.Close(True) + +xl.Quit() + +#for i in range(len(file_list)): # since the data was read from EMME, scenario is not needed and the file names are not renamed with scenrio ID - CL 05222020 +# os.rename(file_list[i], file_list_new[i]) diff --git a/sandag_abm/src/main/python/parameterUpdate.py b/sandag_abm/src/main/python/parameterUpdate.py new file mode 100644 index 0000000..cfa06ca --- /dev/null +++ b/sandag_abm/src/main/python/parameterUpdate.py @@ -0,0 +1,60 @@ +# Author:Yun.Ma@sandag.org +# Oct 5,2016 + +#import +import os +import csv +import string +import sys + +# check if property file exists +if not os.path.isfile('..\\conf\\sandag_abm.properties'): + print "Property File Not Found" + raise sys.exit() + +# search scenarioYear +TheYear='' +propFile = open('..\\conf\\sandag_abm.properties','r') +for line in propFile: + if line.find('scenarioYear') > -1: + TheYear = line.strip('\n').split('=')[1] + break +else: + print "scenarioYear Not Found" +propFile.close() + +# read csv file +ParaDict={} +csvFileIn = open('..\\input\\parametersByYears.csv','rU') +reader = csv.DictReader(csvFileIn) +for row in reader: + if row['year'] == TheYear: + ParaDict=row.copy() + break +csvFileIn.close() +ParaDict.pop('year',ParaDict['year']) + +# read and update str in property file +OldVal='' +NewVal='' +Paralines=[] + +propInFile = open('..\\conf\\sandag_abm.properties','r') +for line in propInFile: + for key in ParaDict: + if line.find(key) > -1: + NewVal = ParaDict[key] + print NewVal + OldVal = line.strip('\n').split('=')[1] + line = string.replace(line,OldVal,NewVal) + print line + break + Paralines.append(line) +propInFile.close() + +# write into property file +propOutFile = open('..\\conf\\sandag_abm.properties','w') +for line in Paralines: + propOutFile.write(line) +propOutFile.close() + diff --git a/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py b/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py new file mode 100644 index 0000000..a6a9eb9 --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py @@ -0,0 +1,237 @@ +__author__ = 'wsu' +#Wu.Sun@sandag.org 10-27-2016 +import Tkinter +import Tkconstants +import tkFileDialog +import os +from Tkinter import * +from PIL import Image,ImageTk +import popupMsg + +class CreateScenarioGUI(Tkinter.Frame): + def __init__(self, root): + Tkinter.Frame.__init__(self, root, border=5) + body = Tkinter.Frame(self) + body.pack(fill=Tkconstants.X, expand=1) + sticky = Tkconstants.E + Tkconstants.W + body.grid_columnconfigure(1, weight=2) + + #divider line + divider=u"_"*120 + self.releaseDir='T:\\ABM\\release\\ABM' + self.defaultScenarioDir="T:\\projects\\sr14" + self.defaultNetworkDir="T:\\projects\\sr14\\version14_2_0\\network_build" + + self.buttonVar= IntVar(root) + self.yButton=Radiobutton(body, text="Yes", variable=self.buttonVar, value=1, command=self.initStudy) + self.nButton=Radiobutton(body, text="No", variable=self.buttonVar, value=0,command=self.initStudy) + Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=0,columnspan=5) + Tkinter.Label(body, text=u"Create an ABM Work Space", font=("Helvetica", 10, 'bold')).grid(row=1,columnspan=3) + self.yButton.grid(row=2,column=0, columnspan=2) + self.nButton.grid(row=2,column=1, columnspan=2) + + Tkinter.Label(body, text=u"Study Folder", font=("Helvetica", 8, 'bold')).grid(row=3) + self.studypath = Tkinter.Entry(body, width=40) + self.studypath.grid(row=3, column=1, sticky=sticky) + self.studypath.delete(0, Tkconstants.END) + self.studypath.insert(0, self.defaultScenarioDir) + self.studybutton = Tkinter.Button(body, text=u"...",width=4,command=lambda:self.get_path("study")) + self.studybutton.grid(row=3, column=2) + + Tkinter.Label(body, text=u"Network Folder",font=("Helvetica", 8, 'bold')).grid(row=4) + self.studynetworkpath = Tkinter.Entry(body, width=40) + self.studynetworkpath.grid(row=4, column=1, sticky=sticky) + self.studynetworkpath.delete(0, Tkconstants.END) + self.studynetworkpath.insert(0, self.defaultNetworkDir) + self.studynetworkbutton = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("studynetwork")) + self.studynetworkbutton.grid(row=4, column=2) + + self.copyButton = Tkinter.Button(body, text=u"Create", font=("Helvetica", 8, 'bold'),width=10, command=lambda: self.checkPath("study")) + self.copyButton.grid(row=5,column=0,columnspan=4) + + Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=6,columnspan=5) + Tkinter.Label(body, text=u"Create an ABM scenario", font=("Helvetica", 10, 'bold')).grid(row=7,columnspan=3) + + Tkinter.Label(body, text=u"Version", font=("Helvetica", 8, 'bold')).grid(row=8) + var = StringVar(root) + self.version="version_14_2_2" + optionList=["version_14_2_2"] + option=Tkinter.OptionMenu(body,var,*optionList,command=self.setversion) + option.config(width=50) + option.grid(row=8, column=1) + + Tkinter.Label(body, text=u"Emme Version", font=("Helvetica", 8, 'bold')).grid(row=9) + var = StringVar(root) + self.emme_version = "4.4.4.1" + optionList = ["4.3.7", "4.4.4.1"] + option = Tkinter.OptionMenu(body, var, *optionList, command=self.setEmmeVersion) + option.config(width=50) + option.grid(row=9, column=1) + + Tkinter.Label(body, text=u"Year", font=("Helvetica", 8, 'bold')).grid(row=10) + + var = StringVar(root) + self.year="2016" + yearOptionList = ["2016", "2020", "2023", "2025", "2025nb", "2026", "2029", "2030", "2030nb", "2032", "2035", "2035nb", "2040", "2040nb", "2050","2050nb"] + option=Tkinter.OptionMenu(body,var,*yearOptionList,command=self.setyear) + option.config(width=50) + option.grid(row=10, column=1) + + Tkinter.Label(body, text=u"Scenario Folder", font=("Helvetica", 8, 'bold')).grid(row=11) + self.scenariopath = Tkinter.Entry(body, width=40) + self.scenariopath.grid(row=11, column=1, sticky=sticky) + button = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("scenario")) + button.grid(row=11, column=2) + + Tkinter.Label(body, text=u"Network Folder",font=("Helvetica", 8, 'bold')).grid(row=12) + self.networkpath = Tkinter.Entry(body, width=40) + self.networkpath.grid(row=12, column=1, sticky=sticky) + button = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("network")) + button.grid(row=12, column=2) + + buttons = Tkinter.Frame(self) + buttons.pack() + botton = Tkinter.Button(buttons, text=u"Create", font=("Helvetica", 8, 'bold'),width=10, command=lambda: self.checkPath("scenario")) + botton.pack(side=Tkconstants.LEFT) + Tkinter.Frame(buttons, width=10).pack(side=Tkconstants.LEFT) + button = Tkinter.Button(buttons, text=u"Quit", font=("Helvetica", 8, 'bold'), width=10, command=self.quit) + button.pack(side=Tkconstants.RIGHT) + + self.initStudy() + + def initStudy(self): + #disable study setting buttons + if self.buttonVar.get()==1: + self.studypath.config(state=NORMAL) + self.studybutton.config(state=NORMAL) + self.studynetworkpath.config(state=NORMAL) + self.studynetworkbutton.config(state=NORMAL) + self.copyButton.configure(state=NORMAL) + #enable study setting buttons + elif self.buttonVar.get()==0: + self.studypath.config(state=DISABLED) + self.studybutton.config(state=DISABLED) + self.studynetworkpath.config(state=DISABLED) + self.studynetworkbutton.config(state=DISABLED) + self.copyButton.configure(state=DISABLED) + + #set default input and network paths based on selected year + def setversion(self,value): + self.version=value + return + + # set Emme version + def setEmmeVersion(self, value): + self.emme_version = value + return + + #set default input and network paths based on selected year + def setyear(self,value): + self.defaultpath=self.releaseDir+"\\"+self.version+'\\input\\'+value + self.scenariopath.delete(0, Tkconstants.END) + self.scenariopath.insert(0, self.defaultScenarioDir) + self.networkpath.delete(0, Tkconstants.END) + self.networkpath.insert(0, self.defaultpath) + self.year=value + return + + #set cluster + def setcluster(self,value): + self.cluster=value + return + + #set default options for folded browsers + def setPathOptions(self): + self.dir_opt = options = {} + options['initialdir'] = self.defaultScenarioDir + options['mustexist'] = False + options['parent'] = root + options['title'] = 'This is a title' + + #get a path after the browse button is clicked on + def get_path(self,type): + self.setPathOptions() + path = tkFileDialog.askdirectory(**self.dir_opt) + if type=="scenario": + if path: + spath = os.path.normpath(path) + self.scenariopath.delete(0, Tkconstants.END) + self.scenariopath.insert(0, spath) + elif type=="network": + if path: + npath = os.path.normpath(path) + self.networkpath.delete(0, Tkconstants.END) + self.networkpath.insert(0, npath) + elif type=="study": + if path: + studypath = os.path.normpath(path) + self.studypath.delete(0, Tkconstants.END) + self.studypath.insert(0, studypath) + elif type=="studynetwork": + if path: + studynetworkpath = os.path.normpath(path) + self.studynetworkpath.delete(0, Tkconstants.END) + self.studynetworkpath.insert(0, studynetworkpath) + return + + #check if a path already exisits or is empty + def checkPath(self,type): + self.popup=Tkinter.Tk() + if type=="scenario": + if os.path.exists(self.scenariopath.get()): + if not self.networkpath.get(): + popupMsg.popupmsg(self,"Network folder is empty!",1,type) + else: + popupMsg.popupmsg(self,"Selected scenario folder already exists! Proceeding will overwrite existing files!",2,type) + else: + if not self.scenariopath.get(): + popupMsg.popupmsg(self,"Scenario folder is empty!",1,type) + elif not self.networkpath.get(): + popupMsg.popupmsg(self,"Network folder is empty!",1,type) + else: + self.executeBatch(type) + elif type=="study": + if os.path.exists(self.studypath.get()): + if not self.studynetworkpath.get(): + popupMsg.popupmsg(self,"Network folder is empty!",1,type) + else: + popupMsg.popupmsg(self,"Selected study folder already exists! Proceeding will overwrite existing files!",2,type) + else: + if not self.studypath.get(): + popupMsg.popupmsg(self,"Study folder is empty!",1,type) + elif not self.studynetworkpath.get(): + popupMsg.popupmsg(self,"Network folder is empty!",1,type) + else: + self.executeBatch(type) + return + + #execute DOS commands + def executeBatch(self, type): + self.popup.destroy() + if type=="scenario": + commandstr = u"create_scenario.cmd %s %s %s %s" % ( + self.scenariopath.get(), + self.year, + self.networkpath.get(), + self.emme_version + ) + elif type=="study": + commandstr=u"copy_networkfiles_to_study.cmd "+self.studypath.get()+" "+self.studynetworkpath.get() + print (commandstr) + os.chdir(self.releaseDir+"\\"+self.version+'\\') + os.system(commandstr) + self.popup=Tkinter.Tk() + msg="You have successfully created the "+ type+"!" + popupMsg.popupmsg(self,msg,1,type) + return + +root = Tkinter.Tk() +root.resizable(True, False) +root.minsize(370, 0) +logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") +w=Label(root, image=logo, width=200) +w.pack(side='top', fill='both', expand='yes') +CreateScenarioGUI(root).pack(fill=Tkconstants.X, expand=1) +root.mainloop() + + diff --git a/sandag_abm/src/main/python/pythonGUI/parameterEditor.py b/sandag_abm/src/main/python/pythonGUI/parameterEditor.py new file mode 100644 index 0000000..bdc12cb --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/parameterEditor.py @@ -0,0 +1,253 @@ +__author__ = 'wsu' +#Wu.Sun@sandag.org 7-20-2016 + +import Tkinter +import Tkconstants +import tkFileDialog +import os +from Tkinter import * +from PIL import Image,ImageTk +import stringFinder +import popupMsg + + +class ParametersGUI(Tkinter.Frame): + def __init__(self, root): + Tkinter.Frame.__init__(self, root, border=5) + self.status = Tkinter.Label(self, text=u"ABM Parameter Editor", font=("Helvetica", 12, 'bold')) + self.status.pack(fill=Tkconstants.X, expand=1) + body = Tkinter.Frame(self) + body.pack(fill=Tkconstants.X, expand=1) + sticky = Tkconstants.E + Tkconstants.W + body.grid_columnconfigure(1, weight=2) + + #section labels + sectionLabels=(u"Model Initial Start Options",u"Network Building Options",u"Final Assignment Options:",u"Data Loading Options:") + #radio button lables + rbLabels=(u"Copy warm start trip tables:",u"Copy bike AT access files:",u"Create bike AT access files:",u"Build highway network:",u"Build transit network:",u"Run highway assignment:", + u"Run highway skimming:",u"Run transit assignment:",u"Run transit skimming:",u"Export results to CSVs:",u"Load results to database:") + #properties + self.properties=("RunModel.skipCopyWarmupTripTables","RunModel.skipCopyBikeLogsum","RunModel.skipBikeLogsums","RunModel.skipBuildHwyNetwork","RunModel.skipBuildTransitNetwork","RunModel.skipFinalHighwayAssignment", + "RunModel.skipFinalHighwaySkimming","RunModel.skipFinalTransitAssignment","RunModel.skipFinalTransitSkimming", + "RunModel.skipDataExport","RunModel.skipDataLoadRequest") + + #divider line + divider=u"_"*120 + + #number of properties in GUI + self.pNum=self.properties.__len__() + + #initialize yes and no buttons + self.yButton = [0 for x in range(self.pNum)] + self.nButton = [0 for x in range(self.pNum)] + self.buttonVar= [0 for x in range(self.pNum)] + for i in range(self.pNum): + self.buttonVar[i] = IntVar(root) + self.yButton[i]=Radiobutton(body, text="Yes", variable=self.buttonVar[i], value=1) + self.nButton[i]=Radiobutton(body, text="No", variable=self.buttonVar[i], value=0) + + #set standard property values + self.setDefaultProperties() + + #set AT states-activate and deactivate by selections + #self.setATButtons() + + #scenario folder browser + Tkinter.Label(body, text=u"Scenario Folder", font=("Helvetica", 8, 'bold'),width=15).grid(row=0) + self.scenariopath = Tkinter.Entry(body, width=25) + self.scenariopath.grid(row=0, column=1, sticky=sticky, columnspan=3) + self.scenariopath.insert(0,sys.argv[1]) + button = Tkinter.Button(body, text=u"...",width=4,command=self.get_scenariopath) + button.grid(row=0, column=4) + + #initial start section + for i in range(1,25): + if i==1: #intial start section header + Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) + Tkinter.Label(body, text=sectionLabels[0], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) + elif i>2 and i<6: + Tkinter.Label(body, text=rbLabels[i-3], font=("Helvetica", 8, 'bold')).grid(row=i) + self.yButton[i-3].grid(row=i,column=1) + self.nButton[i-3].grid(row=i,column=3) + elif i==6: #network building section header + Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) + Tkinter.Label(body, text=sectionLabels[1], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) + elif i>7 and i<10: + Tkinter.Label(body, text=rbLabels[i-5], font=("Helvetica", 8, 'bold')).grid(row=i) + self.yButton[i-5].grid(row=i,column=1) + self.nButton[i-5].grid(row=i,column=3) + elif i==10: #final assignment section header + Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) + Tkinter.Label(body, text=sectionLabels[2], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) + elif i>11 and i<16: + Tkinter.Label(body, text=rbLabels[i-7], font=("Helvetica", 8, 'bold')).grid(row=i) + self.yButton[i-7].grid(row=i,column=1) + self.nButton[i-7].grid(row=i,column=3) + elif i==16: #data load section header + Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) + Tkinter.Label(body, text=sectionLabels[3], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) + elif i>17 and i<20: + Tkinter.Label(body, text=rbLabels[i-9], font=("Helvetica", 8, 'bold')).grid(row=i) + self.yButton[i-9].grid(row=i,column=1) + self.nButton[i-9].grid(row=i,column=3) + elif i==20: #iteration section + Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) + Tkinter.Label(body, text=u"Iteration Options", font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) + Tkinter.Label(body, text=u"Start from iteration:", font=("Helvetica", 8, 'bold')).grid(row=i+2) + self.var = IntVar(root) + self.button1=Radiobutton(body, text="1", variable=self.var, value=1) + self.button1.grid(row=i+2,column=1) + self.button1.select() + self.button2=Radiobutton(body, text="2", variable=self.var, value=2).grid(row=i+2,column=2) + self.button3=Radiobutton(body, text="3", variable=self.var, value=3).grid(row=i+2,column=3) + self.button4=Radiobutton(body, text="Skip", variable=self.var, value=4).grid(row=i+2,column=4) + elif i==23: + Tkinter.Label(body, text=u"Sample rates:", font=("Helvetica", 8, 'bold')).grid(row=i) + sv = StringVar(root) + sv.set("0.2,0.5,1.0") + self.samplerates="0.2,0.5,1.0" + sv.trace("w", lambda name, index, mode, sv=sv: self.setsamplerates(sv)) + e = Entry(body, textvariable=sv) + e.config(width=15) + e.grid(row=i,column=1,columnspan=3) + elif i==24:#action buttons + Tkinter.Label(body, text=u"", width=30).grid(row=i,columnspan=2) + buttons = Tkinter.Frame(self) + buttons.pack() + botton = Tkinter.Button(buttons, text=u"Update", font=("Helvetica", 9, 'bold'),width=10, command=self.update_parameters) + botton.pack(side=Tkconstants.LEFT) + Tkinter.Frame(buttons, width=10).pack(side=Tkconstants.LEFT) + button = Tkinter.Button(buttons, text=u"Quit", font=("Helvetica", 9, 'bold'), width=10, command=self.quit) + button.pack(side=Tkconstants.RIGHT) + + def setsamplerates(self,value): + self.samplerates=value.get() + + def setDefaultProperties(self): + self.runtimeFile=sys.argv[1]+"\\conf\\sandag_abm.properties" + self.standardFile=sys.argv[1]+"\\conf\\sandag_abm_standard.properties" + self.populateProperties() + """ + for i in range(self.pNum): + if i<3 or i>6: + self.yButton[i].select() + self.nButton[i].deselect() + else: + self.yButton[i].deselect() + self.nButton[i].select() + """ + + def setATButtons(self): + #disable create bike and walk logsums if 'copy' is chosen + self.yButton[1].config(command=lambda: self.yButton[3].config(state=DISABLED)) + #self.yButton[2].config(command=lambda: self.yButton[4].config(state=DISABLED)) + #disable copy bike and walk logsums if 'create' is chosen + self.yButton[3].config(command=lambda: self.yButton[1].config(state=DISABLED)) + #self.yButton[4].config(command=lambda: self.yButton[2].config(state=DISABLED)) + #enable create bike and walk logsums if NOT 'copy' is chosen + self.nButton[1].config(command=lambda: self.yButton[3].config(state=ACTIVE)) + #self.nButton[2].config(command=lambda: self.yButton[4].config(state=ACTIVE)) + #enable copy bike and walk logsums if NOT 'create' is chosen + self.nButton[3].config(command=lambda: self.yButton[1].config(state=ACTIVE)) + #self.nButton[4].config(command=lambda: self.yButton[2].config(state=ACTIVE)) + + #set scenario path + def get_scenariopath(self): + # defining options for opening a directory; initialize default path from command line + self.dir_opt = options = {} + options['initialdir'] = sys.argv[1] + options['mustexist'] = False + options['parent'] = root + options['title'] = 'This is a title' + scenariopath = tkFileDialog.askdirectory(**self.dir_opt) + if scenariopath: + scenariopath = os.path.normpath(scenariopath) + self.scenariopath.delete(0, Tkconstants.END) + self.scenariopath.insert(0, scenariopath) + else: + self.scenariopath.delete(0, Tkconstants.END) + self.scenariopath.insert(0, sys.argv[1]) + + #property file settings + self.runtimeFile=self.scenariopath.get()+"\\conf\\sandag_abm.properties" + self.standardFile=self.scenariopath.get()+"\\conf\\sandag_abm_standard.properties" + + #populate properties + self.populateProperties() + + return + + #populate properties with exisiting settings in scenario folder + def populateProperties(self): + if self.checkFile(): + for i in range(self.pNum): + if stringFinder.find(self.runtimeFile, self.properties[i]+" = true"): + self.yButton[i].deselect() + self.nButton[i].select() + elif stringFinder.find(self.runtimeFile, self.properties[i]+" = false"): + self.yButton[i].select() + self.nButton[i].deselect() + else: + print "Invalid property "+self.properties[i]+" value!, Property either has to be set to true or false." + return + + # update parameters with user inputs + def update_parameters(self): + #property file settings + self.runtimeFile=self.scenariopath.get()+"\\conf\\sandag_abm.properties" + self.standardFile=self.scenariopath.get()+"\\conf\\sandag_abm_standard.properties" + + self.deleteProperty() + self.old_text = [0 for x in range(self.pNum)] + self.new_text = [0 for x in range(self.pNum)] + for i in range(self.pNum): + if self.buttonVar[i].get()==1: + self.old_text[i]=self.properties[i]+" = true" + self.new_text[i]=self.properties[i]+" = false" + elif self.buttonVar[i].get()==0: + self.old_text[i]=self.properties[i]+" = false" + self.new_text[i]=self.properties[i]+" = true" + + #create a property update dictionary + dic=[] + for i in range(self.pNum): + pair=(self.old_text[i],self.new_text[i]) + dic.append(pair) + print dic[i][0],dic[i][1] + #add iteration update to dictionary + dic.append(("RunModel.startFromIteration = 1","RunModel.startFromIteration = "+str(self.var.get()))) + #add sample rates update to dictionary + dic.append(("sample_rates=0.2,0.5,1.0","sample_rates="+self.samplerates)) + stringFinder.replace(self.standardFile,self.runtimeFile,dic) + self.quit() + + #check if property file exists + def checkFile(self): + result=True + if not os.path.exists(self.runtimeFile): + self.popup=Tkinter.Tk() + popupMsg.popupmsg(self,self.runtimeFile+" doesn't exist!",1) + result=False + return result + + #close popup window and run batch + def letsgo(self): + self.popup.destroy() + return + + #run batch + def deleteProperty(self): + commandstr=u"del "+self.scenariopath.get()+"\\conf\\sandag_abm.properties" + print commandstr + os.system(commandstr) + return + +root = Tkinter.Tk() +root.resizable(True, False) +root.minsize(370, 0) +logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") +w=Label(root, image=logo, width=200) +w.pack(side='top', fill='both', expand='yes') +ParametersGUI(root).pack(fill=Tkconstants.X, expand=1, anchor=W) + +root.mainloop() diff --git a/sandag_abm/src/main/python/pythonGUI/popupMsg.py b/sandag_abm/src/main/python/pythonGUI/popupMsg.py new file mode 100644 index 0000000..e9ce061 --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/popupMsg.py @@ -0,0 +1,17 @@ +__author__ = 'wsu' +import Tkinter +import Tkconstants +#popup window for path validity checking +def popupmsg(self,msg,numButtons,type): + self.popup.wm_title("!!!WARNING!!!") + label = Tkinter.Label(self.popup, text=msg) + label.pack(side="top", fill="x", pady=10) + popbuttons = Tkinter.Frame(self.popup) + popbuttons.pack() + #can't pass arguments to a callback, otherwise callback is called before widget is constructed; use lambda function instead + B1 = Tkinter.Button(popbuttons, text="Proceed", command =lambda: self.executeBatch(type)) + B2 = Tkinter.Button(popbuttons, text="Quit", command = self.popup.destroy) + if numButtons>1: + B1.pack(side=Tkconstants.LEFT) + B2.pack(side=Tkconstants.RIGHT) + Tkinter.Frame(popbuttons, width=10).pack(side=Tkconstants.LEFT) diff --git a/sandag_abm/src/main/python/pythonGUI/setup.py b/sandag_abm/src/main/python/pythonGUI/setup.py new file mode 100644 index 0000000..08db6ca --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/setup.py @@ -0,0 +1,9 @@ +__author__ = 'wsu' +from distutils.core import setup +import py2exe + +setup(windows=['./src/main/python/pythonGUI/parameterEditor.py']) +setup(windows=['./src/main/python/pythonGUI/createStudyAndScenario.py']) +setup(windows=['./src/main/python/pythonGUI/validatorGUI.py']) + + diff --git a/sandag_abm/src/main/python/pythonGUI/stringFinder.py b/sandag_abm/src/main/python/pythonGUI/stringFinder.py new file mode 100644 index 0000000..3679eef --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/stringFinder.py @@ -0,0 +1,21 @@ +__author__ = 'wsu' +def check(fname, txt): + with open(fname) as dataf: + return any(txt in line for line in dataf) + +def replace(old_fname, new_fname, replaceDic): + f1 = open(old_fname, 'r') + f2 = open(new_fname, 'w') + for line in f1: + for pair in replaceDic: + line=line.replace(pair[0], pair[1]) + f2.write(line) + f1.close() + f2.close() + return + +def find(fname, txt): + if check(fname, txt): + return True + else: + return False \ No newline at end of file diff --git a/sandag_abm/src/main/python/pythonGUI/validatorGUI.py b/sandag_abm/src/main/python/pythonGUI/validatorGUI.py new file mode 100644 index 0000000..b3770ce --- /dev/null +++ b/sandag_abm/src/main/python/pythonGUI/validatorGUI.py @@ -0,0 +1,122 @@ +__author__ = 'wsu' +#Wu.Sun@sandag.org 2-10-2017 +#wsu updated 5/2/2017 for release 13.3.2 +import Tkinter +import Tkconstants +import tkFileDialog +import os +from Tkinter import * +from PIL import Image,ImageTk +import popupMsg +import stringFinder + +class CreateScenarioGUI(Tkinter.Frame): + def __init__(self, root): + Tkinter.Frame.__init__(self, root, border=5) + body = Tkinter.Frame(self) + body.pack(fill=Tkconstants.X, expand=1) + sticky = Tkconstants.E + Tkconstants.W + body.grid_columnconfigure(1, weight=2) + + self.version="version_13_3_2" + + #divider line + divider=u"_"*120 + self.releaseDir='T:\\ABM\\release\\ABM' + self.defaultScenarioDir="T:\\projects\\sr13" + + #validation section + Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=13,columnspan=5) + Tkinter.Label(body, text=u"Validate a Base Year Scenario", font=("Helvetica", 10, 'bold')).grid(row=14,columnspan=3) + Tkinter.Label(body, text=u"Base Year", font=("Helvetica", 8, 'bold')).grid(row=15) + var = StringVar(root) + self.year="2012" + optionList=["2012", "2014"] + option=Tkinter.OptionMenu(body,var,*optionList,command=self.setyear) + option.config(width=50) + option.grid(row=15, column=1) + + Tkinter.Label(body, text=u"Scenario Number", font=("Helvetica", 8, 'bold')).grid(row=16) + vv = StringVar(root) + vv.set("540") + self.validationScenario="540" + vv.trace("w", lambda name, index, mode, vv=vv: self.setValidationScenario(vv)) + self.ve = Entry(body, textvariable=vv) + self.ve.config(width=15) + self.ve.grid(row=16,column=1,sticky=sticky) + + Tkinter.Label(body, text=u"Output Folder", font=("Helvetica", 8, 'bold')).grid(row=17) + self.validationpath = Tkinter.Entry(body, width=40) + self.validationpath.grid(row=17, column=1, sticky=sticky) + self.voutbutton = Tkinter.Button(body, text=u"...",width=4,command=lambda:self.get_path("validate")) + self.voutbutton.grid(row=17, column=2) + + self.validateButton = Tkinter.Button(body, text=u"Validate", font=("Helvetica", 8, 'bold'),width=10, command=lambda:self.checkPath("validate")) + self.validateButton.grid(row=18,column=0,columnspan=4) + + #set default input and network paths based on selected year + def setyear(self,value): + dic=[] + pair=("xxxx",value) + dic.append(pair) + dir=self.releaseDir+"\\"+self.version+'\\validation\\' + stringFinder.replace(dir+"sandag_validate_generic.properties",dir+"sandag_validate.properties",dic) + return + + #set default options for folded browsers + def setPathOptions(self): + self.dir_opt = options = {} + options['initialdir'] = self.defaultScenarioDir + options['mustexist'] = False + options['parent'] = root + options['title'] = 'This is a title' + + #get a path after the browse button is clicked on + def get_path(self,type): + self.setPathOptions() + path = tkFileDialog.askdirectory(**self.dir_opt) + if path: + vpath = os.path.normpath(path) + self.validationpath.delete(0, Tkconstants.END) + self.validationpath.insert(0, vpath) + return + + #check if a path already exisits or is empty + def checkPath(self,type): + self.popup=Tkinter.Tk() + if not self.validationScenario: + popupMsg.popupmsg(self,"No validation scenario was selected!",1,type) + else: + if not self.validationpath.get(): + popupMsg.popupmsg(self,"Validation output directory is empty!",1,type) + else: + self.executeBatch(type) + return + + #execute DOS commands + def executeBatch(self, type): + self.popup.destroy() + msg="You have successfully created the "+ type+"!" + commandstr=u"validate.cmd "+self.validationScenario+" "+self.validationpath.get()+"\\" + msg="Validation results are in "+self.validationpath.get() + dir=self.releaseDir+"\\"+self.version+'\\validation\\' + print commandstr + os.chdir(dir) + os.system(commandstr) + self.popup=Tkinter.Tk() + popupMsg.popupmsg(self,msg,1,type) + return + + def setValidationScenario(self,value): + self.validationScenario=value.get() + +root = Tkinter.Tk() +root.resizable(True, False) +root.minsize(370, 0) +logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") +w=Label(root, image=logo, width=200) +w.pack(side='top', fill='both', expand='yes') +CreateScenarioGUI(root).pack(fill=Tkconstants.X, expand=1) +root.mainloop() + + diff --git a/sandag_abm/src/main/python/remote_run_traffic.py b/sandag_abm/src/main/python/remote_run_traffic.py new file mode 100644 index 0000000..57634e0 --- /dev/null +++ b/sandag_abm/src/main/python/remote_run_traffic.py @@ -0,0 +1,123 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// Copyright INRO, 2016-2017. /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// remote_run_traffic.py /// +#//// /// +#//// Runs the traffic assignment(s) for the specified periods. /// +#//// For running assignments on a remote server using PsExec, /// +#//// via batch file which configures for Emme python, starts /// +#//// or restarts the ISM and and maps T drive. /// +#//// /// +#//// The input arguments for the traffic assignment is read from /// +#//// start_*.args file in the database directory (database_dir). /// +#//// The "*" is one of the five time period abbreviations. /// +#//// /// +#//// Usage: remote_run_traffic.py database_dir /// +#//// /// +#//// database_dir: The path to the directory with the period /// +#//// specific traffic assignment data (scenarios and /// +#//// matrices). /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// + + +import inro.emme.desktop.app as _app +import inro.modeller as _m +import inro.emme.database.emmebank as _emmebank +import json as _json +import traceback as _traceback +import glob as _glob +import time as _time +import sys +import os + +_join = os.path.join +_dir = os.path.dirname + + +class LogFile(object): + def __init__(self, log_path): + self._log_path = log_path + def write(self, text): + with open(self._log_path, 'a') as f: + f.write(text) + def write_timestamp(self, text): + text = "%s - %s\n" % (_time.strftime("%Y-%m-%d %H:%M:%S"), text) + self.write(text) + def write_dict(self, value): + with open(self._log_path, 'a') as f: + _json.dump(value, f, indent=4) + f.write("\n") + + +def run_assignment(modeller, database_path, period, msa_iteration, + relative_gap, max_assign_iterations, num_processors, + period_scenario, select_link, logger): + logger.write_timestamp("start for period %s" % period) + traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") + export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") + with _emmebank.Emmebank(_join(database_path, 'emmebank')) as eb: + period_scenario = eb.scenario(period_scenario) + logger.write_timestamp("start traffic assignment") + traffic_assign( + period, msa_iteration, relative_gap, max_assign_iterations, + num_processors, period_scenario, select_link) + logger.write_timestamp("traffic assignment finished, start export to OMX") + output_dir = _join(_dir(_dir(database_path)), "output") + omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) + logger.write_timestamp("start export to OMX %s" % omx_file) + if msa_iteration < 4: + export_traffic_skims(period, omx_file, period_scenario) + logger.write_timestamp("export to OMX finished") + logger.write_timestamp("period %s completed successfully" % period) + + +if __name__ == "__main__": + python_file, database_dir = sys.argv + file_ref = os.path.split(database_dir)[1].lower() + log_path = _join(_dir(_dir(database_dir)), "logFiles", "traffic_assign_%s.log" % file_ref) + logger = LogFile(log_path) + try: + logger.write_timestamp("remote process started") + # Test out licence by using the API + eb = _emmebank.Emmebank(_join(database_dir, 'emmebank')) + eb.close() + logger.write_timestamp("starting Emme Desktop application") + project_path = _join(_dir(database_dir), "emme_project.emp") + desktop = _app.start_dedicated(True, "abc", project_path) + try: + logger.write_timestamp("Emme Desktop open") + proc_logbook = _join("%<$ProjectPath>%", "Logbook", "project_%s_temp.mlbk" % file_ref) + desktop.project.par("ModellerLogbook").set(proc_logbook) + modeller = _m.Modeller(desktop) + + from_file = _join(database_dir, "start_*.args") + all_files = _glob.glob(from_file) + for path in all_files: + input_args_file = _join(database_dir, path) # communication file + logger.write_timestamp("input args read from %s" % input_args_file) + with open(input_args_file, 'r') as f: + assign_args = _json.load(f) + logger.write_dict(assign_args) + assign_args["logger"] = logger + assign_args["modeller"] = modeller + run_assignment(**assign_args) + finally: + desktop.close() + except Exception as error: + with open(_join(database_dir, "finish"), 'w') as f: + f.write("FATAL ERROR\n") + logger.write_timestamp("FATAL error execution stopped:") + logger.write(unicode(error) + "\n") + logger.write(_traceback.format_exc(error)) + finally: + _time.sleep(1) + with open(_join(database_dir, "finish"), 'a') as f: + f.write("finish\n") + sys.exit(0) diff --git a/sandag_abm/src/main/python/sdcvm.py b/sandag_abm/src/main/python/sdcvm.py new file mode 100644 index 0000000..aee0e26 --- /dev/null +++ b/sandag_abm/src/main/python/sdcvm.py @@ -0,0 +1,840 @@ +''' +Created on 2010-04-20 + +@author: Kevin +''' + + +# Python libraries +import copy +import csv +import math +import random +import time +from array import array +from optparse import OptionParser + + +# CSTDM libraries +import sdcvm_settings as settings + + +class excelOne(csv.excel): + # define CSV dialect for Excel to avoid blank lines from default \r\n + lineterminator = "\n" + + + + +def logitNestToLogsums(nestDict): + + #print "Evaluating logit tree structure" + # Logit tree evaluation bit + # + # What this does is start with a dictionary where each key corresponds to a node on a + # nested logit tree, including the special code "top" which is the top of the tree + # and the values for utility for each of the alternatives at the bottom of the tree. + # When the dictionary starts out, the nest nodes should have a list of connections; + # each connection is itself a list of [lower node, nest coefficient]. + # + # For instance, a classic mode choice nest would look like: + # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], ['walk', 1.0] ] + # 'auto': [ ['sov', 1.0], ['hov', 0.75] ] + # 'transit': [ ['bus', 0.2], ['lrt', 0.2] ] + # 'hov': [ ['hov 2', 0.6], ['hov 3', 0.6] ] + # 'walk': -4.2911 + # 'sov': 1.0109 + # 'hov 2': -1.201 + # 'hov 3': -1.415 + # 'bus': -3.504 + # 'lrt': -2.879 } + # + # The loop then iterates through the keys of the dictionary (nodes). + # - If the key contains only a float (e.g. 'walk' above), this is assumed to be the + # utility or logsum, and nothing more needs to be done. + # - If the key contains any list elements (connections), the nodes they refer to will be checked; + # if the node it refers to contains only a float (e.g. the ['bus', 0.2] connection + # for the transit nest above, where 'bus' contains the float -3.504), then the nested utility + # will be multiplied by the coefficient and replace the list element. (e.g. -3.504 * 0.2 = -.7008) + # if the node it refers to contains a list (e.g. the ['hov', 0.75] connection above), nothing is done. + # - If the key contains all values as floats (not the case right now with the example above, but after + # the first run through, the 'hov' and 'transit' nodes will be in this state), then the logsum is taken + # of all of the floats, and this replaces the node value. + # + # After the first iteration, the mode choice nest would have: + # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], -4.2911 ] + # 'auto': [ 1.0109, ['hov', 0.75] ] + # 'transit': [ -0.7008, -0.5758 ] + # 'hov': [ -0.7206, -0.849 ] + # 'walk': -4.2911 + # 'sov': 1.0109 + # 'hov 2': -1.201 + # 'hov 3': -1.415 + # 'bus': -3.504 + # 'lrt': -2.879 } + # + # After the second iteration, the mode choice nest would have: + # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], -4.2911 ] + # 'auto': [ 1.0109, ['hov', 0.75] ] + # 'transit': 0.0568 + # 'hov': -0.0896 + # 'walk': -4.2911 + # 'sov': 1.0109 + # 'hov 2': -1.201 + # 'hov 3': -1.415 + # 'bus': -3.504 + # 'lrt': -2.879 } + # + # + # In essence, what happens is that the utilities are passed up; when all of the utilities for a node + # have been calculated, then the logsum is taken at that node. The nest coefficients are always multiplied + # so a value of 1 needs to be specified if no other value is to be used. + # + # Note that this, in the way it works, destroys the connection information, replacing it by logsums + # at the node location instead. So a copy (not a reference, which nestDict = keepDict would do) is needed + # if the connection information needs to be used again. + + x = 0 + nodeList = nestDict.keys() + + # Do this until the top node has been resolved into a float + while isinstance(nestDict["top"], list) is True: + #print 40 * "." + for node in nodeList: + #print node, nestDict[node] + if isinstance(nestDict[node], float): # if it's already only a float, it's a utility or logsum; do nothing + pass + else: + countFloat = 0 + countSub = 0 + + for sn in range(len(nestDict[node])): + if isinstance(nestDict[node][sn], float): + countFloat = countFloat + 1 + elif isinstance(nestDict[node][sn], list): + subName = nestDict[node][sn][0] + if isinstance(nestDict[subName], float): # if the value this refers to is a float + #print "... ", node, subName, nestDict[subName], nestDict[node][sn][1] + nestDict[node][sn] = nestDict[subName] * nestDict[node][sn][1] + + countSub = countSub + 1 + + #print node, countFloat, countSub + if countFloat == len(nestDict[node]): # all the values are floats; take their logsum + expsum = 0 + for element in nestDict[node]: + expsum = expsum + math.exp(element) + if expsum > 0: + nestDict[node] = math.log(expsum) + else: + nestDict[node] = -99999.9 + + if x > 250: # Prevent nesting dictionary errors + print 250 * "!" + print nestDict + raise RuntimeError("Can't resolve nesting structure!") + x = x + 1 + return nestDict + + + +#def zonalProperties(tazList=None, fileName = cvmZonalProperties): +def zonalProperties(fileName, tazList=None): + #=========================================================================== + # Returns a dictionary of zonal properties, which are anything that applies + # to a zone under all circumstances + # + # The dictionary uses the property names as keys (e.g. "PSE" or "Ag Employment") + # and each key leads to a list of properties, in the same order as the tazList + # + # Reads the zonal property file by default, but can also read in another file + # in the same format if specified. + # + # If the tazList argument is blank, reads in all TAZ and returns both the + # props dictionary and the tazList containing all zones. + # + #=========================================================================== + + print " Reading zonal property file", fileName + fin = open(fileName, "rU") + inFile = csv.reader(fin) + header = inFile.next() + + + tempTazDict = {} + tempTazList = [] + for row in inFile: + try: + taz = int(row[header.index("TAZ")]) + except ValueError: + print 120 * "#" + print "Couldn't process a zone in zonal properties file." + raise + + tempTazList.append(taz) + list = [] + for thing in row: + try: + list.append(float(thing)) + except ValueError: + list.append(thing) + tempTazDict[taz] = list + + if tazList is None: + tazList = tempTazList + returnBoth = True + else: + returnBoth = False + + propsDict = {} + for thing in header: + propsDict[thing] = [] + for zone in tazList: + for n in range(len(header)): + propsDict[header[n]].append(tempTazDict[zone][n]) + + if returnBoth == True: + return tazList, propsDict + else: + return propsDict + + + +def hdf5Skim(fromList, toList, fromZoneDict, toZoneDict, table, skimDict, skimList): + # Reads a skim matrix; the from are the rows and the to are the columns + # so skim references are skim[from][to]. + # In the application of the CVM, the from and to are the same, so the matrix is + # symmetric (in that the 15th cell in the 4th row and 4th cell in 15th row refer + # to the same two zones; one-way roads and congestion means that they won't have + # symmetrical costs) + + + # Create blank skims to hold all of the data + + for s in skimList: + skimDict[s] = [] + for c in range(len(fromList)): + for s in skimList: + row = len(toList) * [99999.9] + row = array('f', row) + skimDict[s].append(row) + + # Read in the table row by row + x = 0 + for row in table.iterrows(): + if fromZoneDict.has_key(row['origin']): + iTaz = fromZoneDict[row['origin']] + if toZoneDict.has_key(row['destination']): + jTaz = toZoneDict[row['destination']] + for s in skimList: + try: + skimDict[s][iTaz][jTaz] = row[s] + except: + print "Skim reading error!", iTaz, jTaz + + x = x + 1 + if x % 500000 == 0: + pass + print " read", x/1000000.0, "million rows." + + return skimDict + + +def csvSkim(fromList, toList, fromZoneDict, toZoneDict, skimFile, skimDict, skimName): + # Reads a skim matrix; the from are the rows and the to are the columns + # so skim references are skim[from][to]. + # In the application of the CVM, the from and to are the same, so the matrix is + # symmetric (in that the 15th cell in the 4th row and 4th cell in 15th row refer + # to the same two zones; one-way roads and congestion means that they won't have + # symmetrical costs) + # Uses TransCad GISDK output which is in "row format", i.e. each row is all destinations for a given origin. + + + # Create blank skims to hold all of the data + print skimFile + skimDict[skimName] = [] + for c in range(len(fromList)): + row = len(toList) * [99999.9] + row = array('f', row) + skimDict[skimName].append(row) + + + + # Open skim file and read header + fin = open(skimFile, "r") + inFile = csv.reader(fin) +# header = inFile.next() +# +# for i in range(len(skimList)): #Replace column names in skim list with index of same-named header +# try: +# skimList[i] = header.index(skimList[i]) +# except ValueError: +# print "Skim name", skimList[i], "nots found in skim file", skimFile +# print "Header:", header +# raise ValueError + + # Read in the table row by row + x = 0 + err = 0 + for row in inFile: + + if fromZoneDict.has_key(int(row[0])): #Orig + iTaz = fromZoneDict[int(row[0])] +# if skimName == "Time_Mid": + + while row.count("") > 0: + row[row.index("")] = "0" + err = err + 1 + floatrow = map(float, row[1:]) + skimDict[skimName][iTaz] = array("f", floatrow) + +# else: +# for jTaz in toList: +# if skimName == "Light_Mid": +# skimDict[skimName][iTaz][jTaz-1] = float(row[jTaz]) +# else: +# skimDict[skimName][iTaz][jTaz-1] = float(row[jTaz]) * -0.3 + + x = x + 1 + if x % 500 == 0: + pass + print " read", x, "rows." + print "Replaced", err, "null values." + return skimDict + + + + +def bigrun(): + ts = time.clock() + + # =============================================================================== + # Set Parser Options + # =============================================================================== + parser = OptionParser() + parser.add_option("-s", "--scale", + action="store", dest="scale", default=1.0, + help="scale factor for multiple runs") + parser.add_option("-p", "--path", + action="store", dest="path", + help="project scenario path") + (options, args) = parser.parse_args() + # =============================================================================== + # Input File Names + # =============================================================================== + cvmInputPath = options.path + "/input/" + + cvmZonalProperties = cvmInputPath + "Zonal Properties CVM.csv" + + skimPath = options.path + "/output/" + + skimFileDict = {"Light_Mid": [skimPath + "impldt_MD_DU.TXT"], + "Medium_Mid": [skimPath + "impmhdt_MD_DU.TXT"], + "Heavy_Mid": [skimPath + "imphhdt_MD_DU.TXT"], + "Time_Mid": [skimPath + "impldt_MD_Time.TXT"]} + # =============================================================================== + # Read in scale factor + # =============================================================================== +# fin = open(settings.scaleFactorSource, "r") +# for row in fin: +# if len(row) > 15: # key word has 15 chars, so don't need to look at shorter lines +# if row[0:15] == "cvm.scaleFactor": +# s = row.index("=") +# scale = float(row[s+1:]) +# scaleName = row + + scale = float(options.scale) + print 40*"-" + print "Scaling tour gen with scale factor", scale + print 40*"-" + print + + + testZones = [1578, 88, 971, 2178, 3798, 2711, 4286] + fout = open("AccessVals.csv", "w") + outFileTest = csv.writer(fout, excelOne) + outFileTest.writerow(["I", "J", "AccType", "Cost", "Attr", "AccVal"]) + + + # =============================================================================== + # Produce accessibilities and other CVM-specific derived attributes + # =============================================================================== + # Read in zonal properties file + tazList, zonals = zonalProperties(fileName=cvmZonalProperties) + tazDict = {} + #tazList = tazList[:25] + for t in range(len(tazList)): + tazDict[tazList[t]] = t + + # Calculate accessibilities + print "Calculating accessibilities", round(time.clock(), 2) + + cvmZonals = {} # This is a zonal properties style dictionary, indexed by thing and containing a list in order of properties + accDict = settings.cvmAccDict # Dictionary for creating accessibilities; [skim, property, lambda] + accList = accDict.keys() + + + skimList = [] + for accType in accList: + cvmZonals[accType] = [] + if skimList.count(accDict[accType][0]) == 0: + skimList.append(accDict[accType][0]) + cvmZonals["LnJobs30"] = [] + cvmZonals["REZone"] = [] + + # Calculate percentage employment by industry; binary over 3000 flag + sectors = ["SV", "IN", "RE", "TH", "WH", "GO"] + for sect in sectors: + cvmZonals["Pct" + sect] = [] + cvmZonals["Over3K_" + sect] = [] + for taz in tazList: + idx = tazDict[taz] + for sect in sectors: + cvmZonals["Pct" + sect].append(zonals["CVM_" + sect][idx] / (zonals["TotEmp"][idx] + 0.0001)) + if zonals["CVM_" + sect][idx] > 3000: + cvmZonals["Over3K_" + sect].append(1) + else: + cvmZonals["Over3K_" + sect].append(0) + if cvmZonals["PctRE"][idx] > 0.5: + cvmZonals["REZone"].append(1) + else: + cvmZonals["REZone"].append(0) + + # Calculate employment and population density and cap if necessary + cvmZonals["PopDensCap"] = [] + cvmZonals["EmpDensCap"] = [] + for taz in tazList: + idx = tazDict[taz] + pop = zonals["Pop"][idx] + emp = zonals["TotEmp"][idx] + area = zonals["Area_SqMi"][idx] + if area > 0: + cvmZonals["PopDensCap"].append(min(pop/area, 50000)) + cvmZonals["EmpDensCap"].append(min(emp/area, 100000)) + else: + cvmZonals["PopDensCap"].append(0) + cvmZonals["EmpDensCap"].append(0) + + + # Read in skims + + print "Reading in CVM skims. Time:", round(time.clock()-ts, 2) + skimDict = {} + + for skimName in skimFileDict.keys(): + print "...", skimName, round(time.clock()-ts, 2) + skimList.append(skimName) + skimDict = csvSkim(tazList, tazList, tazDict, tazDict, + skimFileDict[skimName][0], skimDict, skimName) + + print skimDict.keys() + print len(tazDict.keys()) + print len(tazList) + #print tazDict + + print "Skims read in. Time:", round(time.clock()-ts, 2) + idx = 0 + for iTaz in tazList: + currAcc = len(accList) * [0] + jobs30 = 0 +# maxCost = [999999, -1, -1] + for jTaz in tazList: + iIdx = tazDict[iTaz] + jIdx = tazDict[jTaz] + for accType in accList: + skimType = accDict[accType][0] + try: + cost = skimDict[skimType][iIdx][jIdx] + except: + print skimType, iTaz, jTaz, iIdx, jIdx + print len(skimDict[skimType]) + print len(skimDict[skimType][0]) + crash +# if accType == "Acc_LE" and cost < maxCost[0]: +# maxCost[0] = cost +# maxCost[1] = jTaz +# maxCost[2] = jIdx + attr = zonals[accDict[accType][1]][jIdx] + lam = accDict[accType][2] + accVal = attr * math.exp(cost * lam) + #print accType, accList + currAcc[accList.index(accType)] = currAcc[accList.index(accType)] + accVal + if testZones.count(iTaz) > 0: + outFileTest.writerow([iTaz, jTaz, accType, cost, attr, accVal]) + + + cost = skimDict["Time_Mid"][iIdx][jIdx] + if cost < 30: + jobs30 = jobs30 + zonals["TotEmp"][jIdx] + + if idx % 250000 == 0: + print "Processed", idx, "OD pairs, most recently", iTaz, jTaz, round(time.clock()-ts,2) + + idx = idx + 1 +# if idx < 2000000: +# print iTaz, iIdx, maxCost + for c in range(len(accList)): + cvmZonals[accList[c]].append(currAcc[c]) + if jobs30 > 0: + cvmZonals["LnJobs30"].append(math.log(jobs30)) + else: + cvmZonals["LnJobs30"].append(0) + + + # First set is ship/no ship, second set is tours/emp + rangeDict = {'FA': [[0.01, 0.05], [1.22, 2.09]], + 'IN': [[0.37, 0.59], [0.13, 0.52]], + 'WH': [[0.35, 0.50], [0.17, 0.47]], + 'RE': [[0.10, 0.48], [0.33, 0.65]], + 'SV': [[0.10, 0.33], [0.08, 0.33]], + 'GO': [[0.10, 0.33], [0.08, 0.33]], + 'TH': [[0.12, 0.55], [0.25, 0.55]]} + + + adjDict = {'FA': [[-3.4677, -0.9635, -0.6686, -0.3527, 0.0091],[5.0449, 1.6675, 2.3762, 2.0193, 2.0415]], + 'IN': [[-0.6544, -1.8124, -1.3965, -1.2595, 1.094], [-0.2966, 0.9763, 0.9086, 1.0257, 1.1646]], + 'RE': [[-0.456, -1.4629, -1.6433, -0.666, 1.2942], [-0.1883, 0.4861, 1.1003, 0.8668, 0.8273]], + 'SV': [[-0.6344, -0.5504, -0.2762, -0.0793, 0.0627], [0.6254, 1.8833, 2.5968, 2.3156, 2.2625]], + 'GO': [[-0.6344, -0.5504, -0.2762, -0.0793, 0.0627], [0.6254, 1.8833, 2.5968, 2.3156, 2.2625]], + 'TH': [[-1.9161, -1.623, -1.5503, -1.1891, 0.0705], [0.5195, 0.3715, 0.8332, 0.6878, 0.3449]], + 'WH': [[0.1322, -1.0171, -1.1068, -0.8744, 1.2775], [-1.3517, 0.5526, 1.1136, 0.7629, 0.794]]} + + # Scale to match proportions for SANDAG + adjSandagDict = settings.genCalibDict + + + +# cout = open("e:/sjvitm_sdcvm_calibration.csv", "w") +# calibOut = csv.writer(cout, excelOne) +# calibOut.writerow(['Sector', 'Model', 'Iteration', 'Below', 'Within', 'Above', 'Score', 'Average', 'Param', 'Tours']) + + for iter in range(1): + print 15 * "-", "Iteration", iter, 15 * "-" + + # =========================================================================== + # Big loop: iterate through each industry and create generation + # =========================================================================== + for sector in settings.cvmSectors: + print "Calculating tour generation for sector", sector, round(time.clock()-ts, 2) + # Read in control file for this sector + fin = open(cvmInputPath + sector + ".csv", "r") + inFile = csv.reader(fin) + + paramDict = {} + paramDict["ShipNoShip"] = {} + paramDict["GenPerEmployee"] = {} + paramDict["TourTOD"] ={} + paramDict["VehicleTourType"] = {} + + paramDict["TourTOD_nest"] = {} + paramDict["VehicleTourType_nest"] = {} + + for row in inFile: + model = row[0] + if paramDict.has_key(model): + alt = row[1] + type = row[2] + nest = row[3] + param = float(row[5]) + if nest == "nest": + # put into nesting dictionary + if paramDict[model+"_nest"].has_key(type): + pass + else: + paramDict[model+"_nest"][type] = [] + + if param == 0: + paramDict[model+"_nest"][type].append([alt, 1]) + else: + paramDict[model+"_nest"][type].append([alt, param]) + + else: + if paramDict[model].has_key(alt): + pass + else: + paramDict[model][alt] = [] + parSet = (type, param) + paramDict[model][alt].append(parSet) + + #print paramDict + + # add to CVM zonals file / clear out values + for timePer in settings.cvmTimes: + cvmZonals[sector + "_" + timePer] = [] + cvmZonals[sector + "_Ship"] = [] + cvmZonals[sector + "_ToursEmp"] = [] + + + # Medium loop: iterate through each TAZ + for tazNum in tazList: + taz = tazDict[tazNum] + lu = int(round(float(zonals["CVM_LU_Type"][taz])))-1 + + # =================================================================== + # Phase One: Pass logsums up nested logit structure + # =================================================================== + + # --------------------------------- Tour vehicle type / purpose nest + + model = "VehicleTourType" + #print "Processing:", model, round(time.clock(), 2) + + altList = paramDict[model].keys() + utilDict = {} + + # Begin by calculating the utilities for each alternative + for alt in altList: + util = 0 + for name, par in paramDict[model][alt]: + if name == '': + util = util + par + val = "const" + else: + if cvmZonals.has_key(name): + util = util + cvmZonals[name][taz] * par + val = cvmZonals[name][taz] + elif zonals.has_key(name): + util = util + zonals[name][taz] * par + val = zonals[name][taz] + else: + raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) + utilDict[alt] = util +# if tazNum < 20: +# print alt, util, name, par, val +# if tazNum < 20: +# print 20 * "-" + + # Add the parameters to the nested model and pass it to the logsumerizer + modNest = "VehicleTourType_nest" + + for alt in altList: + paramDict[modNest][alt] = utilDict[alt] + + nestDict = copy.deepcopy(paramDict[modNest]) + #print taz, nestDict + vehAndPurpNest = logitNestToLogsums(nestDict) + CUPurpVeh = vehAndPurpNest["top"] + #print CUPurpVeh + + + # ------------------------------------------------------ Time Of Day + model = "TourTOD" + #print "Processing:", model, round(time.clock(), 2) + + altList = paramDict[model].keys() + utilDict = {} + + # Begin by calculating the utilities for each alternative + for alt in altList: + util = 0 + for name, par in paramDict[model][alt]: + if name == '': + util = util + par + val = "const" + else: + if cvmZonals.has_key(name): + util = util + cvmZonals[name][taz] * par + val = cvmZonals[name][taz] + elif zonals.has_key(name): + util = util + zonals[name][taz] * par + val = zonals[name][taz] + elif name == "CUPurpVeh": + util = util + CUPurpVeh * par + val = CUPurpVeh + else: + raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) + utilDict[alt] = util + #print alt, util, name, par, val + #print 20 * "-" + + # Add the parameters to the nested model and pass it to the logsumerizer + modNest = "TourTOD_nest" + + for alt in altList: + paramDict[modNest][alt] = utilDict[alt] + + nestDict = copy.deepcopy(paramDict[modNest]) + #print nestDict + tourTODNest = logitNestToLogsums(nestDict) + CUTimeOD = tourTODNest["top"] + #print tourTODNest + #print CUTimeOD + + + # ----------------------------------------------- Trips per employee + model = "GenPerEmployee" + #print "Processing:", model, round(time.clock(), 2) + + altList = paramDict[model].keys() + utilDict = {} + # Begin by calculating the utilities for each alternative + for alt in altList: + util = 0 + for name, par in paramDict[model][alt]: + if name == '': + util = util + par + val = "const" + else: + if cvmZonals.has_key(name): + util = util + cvmZonals[name][taz] * par + val = cvmZonals[name][taz] + elif zonals.has_key(name): + util = util + zonals[name][taz] * par + val = zonals[name][taz] + elif name == "CUTimeOD": + util = util + CUTimeOD * par + val = CUTimeOD + else: + raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) + utilDict[alt] = util + #print alt, util, name, par, val + #print 20 * "-" + + # Calibration Adjustments; form: xn, xn-1, fn, fn-1 + utilDict["Gen"] = utilDict["Gen"] + adjDict[sector][1][lu] + + genUtil = utilDict["Gen"] + CUGen = math.log(math.exp(genUtil) + math.exp(0)) # 0 is utility for no tours + + # ---------------------------------------------------- Ship / No Ship + model = "ShipNoShip" + #print "Processing:", model, round(time.clock(), 2) + + altList = paramDict[model].keys() + utilDict = {} + + # Begin by calculating the utilities for each alternative + for alt in altList: + util = 0 + for name, par in paramDict[model][alt]: + if name == '': + util = util + par + val = "const" + else: + if cvmZonals.has_key(name): + util = util + cvmZonals[name][taz] * par + val = cvmZonals[name][taz] + elif zonals.has_key(name): + util = util + zonals[name][taz] * par + val = zonals[name][taz] + elif name == "CUGen": + util = util + CUGen * par + val = CUGen + else: + raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) + utilDict[alt] = util + #print alt, util, name, par, val + # Calibration Adjustments; form: xn, xn-1, fn, fn-1 + utilDict["Ship"] = utilDict["Ship"] + adjDict[sector][0][lu] + + + # =================================================================== + # Phase The Second: Calculate tours by time period + # =================================================================== + + # Get base employment (total employment for fleet allocators) + if sector == "FA": + baseEmp = zonals["TotEmp"][taz] + else: + baseEmp = zonals["CVM_"+sector][taz] + + #print "Base Employment:", baseEmp + + # Ship/No Ship proportions + + uShip = utilDict["Ship"] + uNoShip = utilDict["NoShip"] + + propShip = math.exp(uShip) / (math.exp(uShip) + math.exp(uNoShip)) + + shipEmp = baseEmp * propShip + +# tempOut.writerow([sector, tazList[taz], 'ShipNoShip', 'PropShip', propShip]) + cvmZonals[sector + "_Ship"].append(propShip) + #print "Shipping employees:", shipEmp, propShip + + # Generation: tours per employee + toursPerEmp = (10 * math.exp(genUtil)) / (1 + math.exp(genUtil)) + toursAllDay = shipEmp * toursPerEmp + + # Scale for SANDAG behaviours + toursAllDay = toursAllDay * adjSandagDict[sector][lu] + + + #print "Tours per employee:", toursPerEmp, toursAllDay +# tempOut.writerow([sector, tazList[taz], 'Generation', 'ToursPerEmp', toursPerEmp]) + cvmZonals[sector + "_ToursEmp"].append(toursPerEmp) + + # Time period: calculate probabilities of time period + linkDict = paramDict["TourTOD_nest"] + utilDict = tourTODNest + + #print linkDict + #print utilDict + + #print 20 * "-" + probDict = {} + + # First calculate probabilities for each node with subnodes + for node in linkDict.keys(): + if isinstance(linkDict[node], float): + pass + else: + #print node + expSum = 0 + for subNode in linkDict[node]: + expSum = expSum + math.exp(utilDict[subNode[0]] * subNode[1]) + #print subNode[0], utilDict[subNode[0]], subNode[1], expSum + + for subNode in linkDict[node]: + probDict[subNode[0]] = math.exp(utilDict[subNode[0]] * subNode[1]) / expSum + #print probDict + + # Now go down from top node and scale probabilities of any subnests by the nest prob + # (note: this only goes down one level; this works for the existing time of day nests, but needs to be changed for reuse + for node, coeff in linkDict["top"]: + if isinstance(linkDict[node], float): + pass + else: + for subNode, subcoeff in linkDict[node]: + probDict[subNode] = probDict[subNode] * probDict[node] + #print probDict + + + # Divide into time periods and write out + for timePer in settings.cvmTimes: + + try: + tours = probDict[timePer] * toursAllDay * scale + except: + print "Error in scaling tours:", probDict[timePer], toursAllDay, scale + tours = 0 + cvmZonals[sector + "_" + timePer].append(round(tours, 2)) + + +# ========================================================================== +# Output CVM Gen and Accessibility file +# ========================================================================== + print "Writing the data out...", round(time.clock(),2) + fout = open(cvmInputPath + "CVMToursAccess.csv", "w") + outFile = csv.writer(fout, excelOne) + + header = ["Taz"] + keyList = cvmZonals.keys() + keyList.sort() + header.extend(keyList) + outFile.writerow(header) + + for c in range(len(tazList)): + rowOut = [tazList[c]] + for keyType in keyList: + rowOut.append(cvmZonals[keyType][c]) + outFile.writerow(rowOut) + fout.close() + +# tout.close() +# cout.close() + print "DonE!" + +if __name__ == '__main__': + bigrun() diff --git a/sandag_abm/src/main/python/sdcvm_settings.py b/sandag_abm/src/main/python/sdcvm_settings.py new file mode 100644 index 0000000..e5c1192 --- /dev/null +++ b/sandag_abm/src/main/python/sdcvm_settings.py @@ -0,0 +1,43 @@ +# Created on 2012-10-16 +# +# @author: Kevin + + +# =============================================================================== +# SDCVM properties +# =============================================================================== +cvmModes = ["Light", "Medium", "Heavy"] +cvmTimes = ["OE", "AM", "MD", "PM", "OL"] +# cvmTimes = ["OL"] +cvmSectors = ["GO", "SV", "IN", "RE", "TH", "WH", "FA"] # sectors to run; actual industries are hardcoded into sdcvm.py +# cvmSectors = ["WH"] # sectors to run; actual industries are hardcoded into sdcvm.py + +opCostScale = 1.0 # Scale factor for operating cost + +maxTaz = 4996 # highest taz number; assumes skims are 1-maxTaz without gaps + + +# Costs defined as [time, distance, money]. Not used right now. +cvmCostDict = {"Light": [-0.313, -0.138 * opCostScale, -1], + "Medium": [-0.313, -0.492 * opCostScale, -1], + "Heavy": [-0.302, -0.580 * opCostScale, -1]} + +# Dictionary for creating accessibilities; [skim, property, lambda] +cvmAccDict = {"Acc_LE": ["Light_Mid", "TotEmp", 3.0], + "Acc_LP": ["Light_Mid", "Pop", 3.0], + "Acc_ME": ["Medium_Mid", "TotEmp", 2.0], + "Acc_MP": ["Medium_Mid", "Pop", 2.0], + "Acc_HE": ["Heavy_Mid", "TotEmp", 1.0], + "Acc_HP": ["Heavy_Mid", "Pop", 1.0] + } + +# Calibration adjustment scale factors for tour generation +# Factors by land use type: +# [Low dens, Residential, Retail/Comm, Industrial, Emp Node] +genCalibDict = {'FA': [0.4931, 1.8562, 2.3996, 1.7171, 2.4541], + 'GO': [11.8418, 4.5588, 4.1018, 5.1797, 14.1177], + 'IN': [0.6712, 0.7201, 0.7671, 0.7934, 0.2481], + 'RE': [0.6764, 1.2354, 1.7748, 1.2966, 0.1446], + 'SV': [4.1374, 2.2546, 2.6429, 3.2391, 4.2139], + 'TH': [0.7609, 0.4813, 0.4271, 1.1211, 0.5308], + 'WH': [1.2189, 0.5536, 0.5035, 0.4371, 0.2212]} diff --git a/sandag_abm/src/main/python/sdcvm_settings.pyc b/sandag_abm/src/main/python/sdcvm_settings.pyc new file mode 100644 index 0000000000000000000000000000000000000000..83196f823130412577bf7fa69ec67a3c4feb5385 GIT binary patch literal 1332 zcmYk4drVtZ9LIlc>5GTJfFd%dn**vM8Tl`ul7N>w+oO( zm$^8T;T99k7GH^p%hb%E`-iE~hN)4Ek3ZDJiIHt_*)Zxb%q)iK{M}O?ySF{R^Zoo@ z=k!<8LRqoBas5C7mL4lviN5M#0|3&;0B8hmf(Y$RfJK02KntK1xEUhkwE@}zivddj zO99IOZvbwEh!LxI8u_%{?Z`3w3s>S6)K3b+xl0k9FU z3DBuulWye3+X3$Yyc4h)@Ge1ZL|Xti0d59t1#}5oO!RKRHoz@_Zoqp4Eg`xU@Ls@e zfcF8m3u-6Y0oV!103|?=pru5+0KI_SfIh$;LCffTbhus!IsG&s%}h(5!jqb2ptj9X(z`fA26GTuM&g8Xb^j`1_ER380k>RZOI{#1rR&oL#`l%! zQCAtCKF{8m{rEEDnYKNX(%w^yzjt=-#S`Py)cNH2`%l=75`ShoJoV;Av&1h=f712N zoL%A#Ipa&V8H(}!IDh&6N2%#V!>>=j_Vq7}Kk$XM5D82|i(lh0Y^|DuDmB}wvJP7G=&#rgr;Vne=}Tf)I*>(K zT^3;_O+p_Qc|vkR%MPnr3CZk`H>+x)L_8&nL8ap(;rN(vQKatcRubBBCQVDoQp)6V z#v4y5gF0}MhSf!FA?e)~^>hVd!?}zab!~3l5{nO}UD2UYEt$gCD}HO6|A#b_z2HB=jFO;v`9rMJpbXK<{%{{wY;X`BE6 literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/python/sdcvm_summarize.py b/sandag_abm/src/main/python/sdcvm_summarize.py new file mode 100644 index 0000000..16f802a --- /dev/null +++ b/sandag_abm/src/main/python/sdcvm_summarize.py @@ -0,0 +1,99 @@ +import csv, time +import sdcvm +# CSTDM libraries +import sdcvm_settings as settings +from optparse import OptionParser + +# External libraries +ts = time.clock() + + +class excelOne(csv.excel): + # define CSV dialect for Excel to avoid blank lines from default \r\n + lineterminator = "\n" + +# =============================================================================== +# Set Parser Options +# =============================================================================== +parser = OptionParser() +parser.add_option("-p", "--path", + action="store", dest="path", + help="project scenario path") +(options, args) = parser.parse_args() +# =============================================================================== +# Input File Names +# =============================================================================== +cvmInputPath = options.path + "/input/" +cvmZonalProperties = cvmInputPath + "Zonal Properties CVM.csv" +skimPath = options.path + "/output/" +cvmPath = options.path + "/output/" + +tazList = range(1, settings.maxTaz+1) +tazList, zonals = sdcvm.zonalProperties(fileName=cvmZonalProperties) +tazDict = {} +# tazList = tazList[:25] +for t in range(len(tazList)): + tazDict[tazList[t]] = t + +# Read in skims + +print "Reading in CVM skims. Time:", round(time.clock()-ts, 2) +skimDict = {} +skimList = [] + + +print "... Midday distance", round(time.clock()-ts, 2) +skimList.append("Dist_Mid") +skimDict = sdcvm.csvSkim(tazList, tazList, tazDict, tazDict, + skimPath + "impldt_MD_Dist.TXT", skimDict, "Dist_Mid") + + +bigDict = {} +for ind in settings.cvmSectors: + for tim in settings.cvmTimes: + print ind, tim, round(time.clock()-ts, 2) + fin = open(cvmPath + "Trip_" + ind + "_" + tim + ".csv", "r") + inFile = csv.reader(fin) + header = inFile.next() + for row in inFile: + mode = row[header.index("Mode")] + trip = int(row[header.index("Trip")]) + purp = row[header.index("TourType")] + toll = row[header.index("TripMode")] + tollAv = row[header.index("TollAvailable")] + home = int(row[header.index("HomeZone")]) + iTaz = int(row[header.index("I")]) + jTaz = int(row[header.index("J")]) + tim = row[header.index("TripTime")] + + iIdx = tazList.index(iTaz) + jIdx = tazList.index(jTaz) + newjIdx = jIdx # new index created to add 12 to include external zones 1-12 in the skim matrix + + key = (ind, mode, purp, toll, tollAv, tim, home) + if bigDict.has_key(key): + pass + else: + bigDict[key] = [0, 0, 0, 0] + + bigDict[key][1] = bigDict[key][1] + 1 + bigDict[key][2] = bigDict[key][2] + skimDict["Dist_Mid"][iIdx][newjIdx] + if trip == 1: + bigDict[key][0] = bigDict[key][0] + 1 + if iTaz == jTaz: + bigDict[key][3] = bigDict[key][3] + 1 + +fout = open(cvmPath + "Gen and trip sum.csv", "w") +outFile = csv.writer(fout, excelOne) + +keyList = bigDict.keys() +keyList.sort() +header = ["Industry", "Mode", "Purpose", "Toll", "TollAv", "Time", "TAZ", "Tours", "Trips", "Dist", "Intra"] +outFile.writerow(header) + + +for key in keyList: + outRow = list(key) + outRow.extend(bigDict[key]) + outFile.writerow(outRow) +fout.close() diff --git a/sandag_abm/src/main/python/serverswap.py b/sandag_abm/src/main/python/serverswap.py new file mode 100644 index 0000000..5da8c6d --- /dev/null +++ b/sandag_abm/src/main/python/serverswap.py @@ -0,0 +1,92 @@ +# Rick.Curry@sandag.org +# July 12, 2016 + +import os +import socket +import csv +from optparse import OptionParser + + +def swap_servers(infile, outfile, searchitem, sep_str, swap_value, skip_lines): + found = 0 + print infile, outfile + print searchitem, sep_str, swap_value + lines = [] + with open(infile) as propInFile: + for line in propInFile: + if line.find(searchitem) > -1: + if found == skip_lines: + line = line.split(sep_str, 1)[0] + sep_str + swap_value + "\n" + else: + found += 1 + lines.append(line) + with open(outfile, 'w') as propOutFile: + for line in lines: + propOutFile.write(line) + + +# Set Parser Options +parser = OptionParser() +parser.add_option("-p", "--path", + action="store", dest="path", + help="project scenario path") +(options, args) = parser.parse_args() + +# Set Paths +dst_dir_bin = options.path + "/bin/" +dst_dir_conf = options.path + "/conf/" + +# Get IP Address +ip = socket.gethostbyname(socket.gethostname()) +print str(ip) + +# Read Server Info +fileServer = options.path + "/conf/server-config-local.csv" +if not os.path.exists(fileServer): + fileServer = "T:/ABM/release/ABM/config/server-config.csv" +print fileServer +logFile = options.path + "/logFiles/serverswap.log" +if os.path.exists(logFile): + os.remove(logFile) +dictServer = csv.DictReader(open(fileServer)) + +# Check for Matching IP Address and Stop on Row +match = 'false' +for row in dictServer: + print row + if row['ActualIP'] == str(ip): + match = 'true' + print match + serverName = row['ServerName'] + print serverName + modelIP = row['ModelIP'] + break + +# Write error log if IP address not found +logWriteFile = open(logFile, "w") +if match == 'false': + logWriteFile.write('Using server-config file in: ' + fileServer + "\n") + logWriteFile.write('FATAL, Head Node not found - check for ' + str(ip)) + print 'Head Node not found' +else: + # Update Files in serverswap_files.csv + logWriteFile.write('Using server-config file in: ' + fileServer + "\n") + logWriteFile.write('MATCH, Head Node found - ' + str(ip)) + skip = 0 + fileUpdate = options.path + "/conf/serverswap_files.csv" + print fileUpdate + filesToUpdate = csv.DictReader(open(fileUpdate)) + for update in filesToUpdate: + print update + # Special section for StopABM.cmd which does not have a property=value format + if update['property'] == 'pskill': + refValue = row[update['refValue']] + ' java.exe' + print row[update['refValue']] + print refValue + swap_servers(options.path + update['fileName'], options.path + update['fileName'], + update['property'], update['separator'], refValue, skip) + skip += 1 + else: + # General section for file updates + swap_servers(options.path + update['fileName'], options.path + update['fileName'], + update['property'], update['separator'], row[update['refValue']], 0) diff --git a/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R b/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R new file mode 100644 index 0000000..6b1b3a7 --- /dev/null +++ b/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R @@ -0,0 +1,532 @@ +### +# OBJECTIVE: +# outlier analysis for SANDAG INRIX travel time data +# travel time and speed plots +# std dev and mean + +# INPUTS: +# inrix data - "2012_10.csv" +# inrix to tcoved correspondence (created by Fehr and Peers) - "inrix_2012hwy.txt" (also an input to regression analysis script) + +# OUTPUTS: +# travel time SD/ mean travel time in 15 mins - "inrix2012_nooutliers_traveltime.dbf" (input to regression analysis script) + +# INRIX DATA: +# october 2012 inrix data is used. +# the inrix data are in 1-mins increment and available for freeways, major/minor arterials, collectors and ramps. + +# INRIX TO MODEL NETWORK CORRESPONDENCE: +# Doubtful if ramps are in the data, perhaps correspondence to ramps isn't correct. +# fehr and peers established an initial correspondence between the model network segments and INRIX TMC segments. +# the correspondence included one record for each model link-to-inrix segments. Some model links may corresponds to multiple TMCs, +# a TMC proportion which is proportion of the TMC feature covered by the corresponding model network feature was also provided. +# The records that may not represent true correspondence, due to high frequency of link join, were flagged in the correspondence. + +# DATA PROCESSING: +# While preparing the data, the records that were flagged in the correspondence file were removed. +# Then, a TMC segment was assigned with one model link by finding the record with the highest TMC proportion. The characteristics of that model link were attached to the TMC segment. +# The analysis was restricted to weekdays. weekend’s data points were eliminated from the dataset +# plots of speeds and travel times are created +# outlier analysis using adjusted boxplot +# removed data points in period 4:15 am - 4:30 am (unexpected variation). also, removed based on reference_speed (if "reference_speed-speed>5") +# comparison of before and after outlier analysis plots +# travel time variability (standard deviation) is calculated for every 15 minutes time interval +# travel time unit = travel time per sec per mile (=travel_time_sec/seg.length) +# seg.length:=speed*travel_time_minutes/60 +# for a tmc segment (travel time sd/travel time mean) is calculated over all weekdays in 15-mins bins + +# OTHER SCRIPTS USED: +# utilfunc.R + +# by: nagendra.dhakar@rsginc.com +# for: SANDAG SHRP C04 - Tolling and Reliability +### + +library(foreign) +library (stringr) +library(xtable) +library(reshape) +library(reshape2) +library(XLConnect) +library(descr) +library(Hmisc) +library(data.table) +library(plyr) +library(gtools) +library(vioplot) +library(lattice) +library(grid) +library(timeDate) +library(ggplot2) +library(robustbase) +library(readr) + +# -------------------- 0. SOURCE FILES AND CONFIG SETTINGS ----------------------- + +# workspace and source files +setwd("E:/Projects/Clients/SANDAG/Data_2015") +source("utilfunc.R") + +# input files +inrix_2012_file = "./INRIX/2012_10/2012_10.csv" +inrix_2014_file = "./INRIX/2014_10/october_2014.csv" +inrixhwycorresp_file = "./TMC/inrix_2012hwy.txt" + +# config settings +year = "2012" # data year +interval<-15 # time bin interval in minutes +bptype<-"adjusted" # box plot - original or adjusted +field="travel.time.sec.per.mile" +ByDOW<-FALSE # by day of week - false, analysis is performed for all weekdays +OUTLIER<-TRUE # outlier analysis or confidence score (CS) removal + +# -------------------- 1. LOAD DATA ---------------------- + +# read and load inrix data +if (year=="2012"){ + # read and load 2012 data + outputsDir="./INRIX/2012_10/" + readSaveRdata(inrix_2012_file,"inrix_2012") + inrix_2012 <- assignLoad(paste0(inrix_2012_file,".Rdata")) +} else{ + # read and load 2014 data + outputsDir="./INRIX/2014_10/" + readSaveRdata(inrix_2014_file,"inrix_2014") + inrix_2014 <- assignLoad(paste0(inrix_2014_file,".Rdata")) +} + +# read and load inrix to hwy correspondence data +readSaveRdata(inrixhwycorresp_file,"inrixhwycorresp") +inrixhwycorresp <- assignLoad(paste0(inrixhwycorresp_file,".Rdata")) + +# -------------------- 2. CLEAN/PROCESS DATA -------------- + +# add a new field tmc_code without first character ('-' or '+') +inrixhwycorresp[,tmc_code:=substr(TMC,2,nchar(TMC))] + +# remove the segments that are flagged +inrixhwycorresp<-subset(inrixhwycorresp,FLAG==0) + +# get unique tmc_code with variables corresponding to maximum of TMCProp +df.orig <-inrixhwycorresp +df.agg<-aggregate(TMCProp~tmc_code,inrixhwycorresp,max) +df.max <- merge(df.agg, df.orig) + +# facility type - keep only two columns +#tmc_ifc <-df.max[,c("tmc_code","IFC","HWYCOV_ID"),] +tmc_ifc <-df.max[,c("tmc_code","TMC_Len","IFC","HWYCOV_ID","NM","LENGTH","ISPD","ABLNO","ABLNA","ABLNP","BALNO","BALNA","BALNP","ABCNT","BACNT","IHOV","ABAU","BAAU"),] +tmc_ifc$LENGTH<-as.numeric(gsub(",","",tmc_ifc$LENGTH)) +tmc_ifc$HWYCOV_ID<-as.numeric(gsub(",","",tmc_ifc$HWYCOV_ID)) + +# for plots +tmc_ifc_temp <-df.max[,c("tmc_code","IFC"),] + +# merge INRIX travel time data with hwyinrix correspondence file +inrix_2012_ifc<-merge(inrix_2012,tmc_ifc,by="tmc_code") + +# set data +alldata<-inrix_2012_ifc + +# determine Day Of Week (DOW): weekday (1) or weekend (0) +alldata[,DOW:=ifelse(isWeekday(as.Date(measurement_tstamp)),1,0)] + +# keep only weekdays +alldata<-alldata[DOW==1] + +# calculate time stamp +alldata[,date:=substr(measurement_tstamp,0,10)] +alldata[,hour:=as.numeric(substr(measurement_tstamp, 12, 13))] +alldata[,min:=as.numeric(substr(measurement_tstamp, 15, 16))] +alldata[,totalmin:=hour*60+min] +alldata[,totalhour:=hour+(min/60)] + +#travel time +alldata[,travel_time_sec:=travel_time_minutes*60] + +# determine time of day (TOD) +alldata[totalmin>=0 & totalmin<=209,tod:='EV1'] +alldata[totalmin>=210 & totalmin<=359,tod:='EA'] #1-early AM +alldata[totalmin>=360 & totalmin<=539,tod:='AM'] #2-AM peak +alldata[totalmin>=540 & totalmin<=929,tod:='MD'] #3-Midday +alldata[totalmin>=930 & totalmin<=1139,tod:='PM'] #4-PM Peak +alldata[totalmin>=1140 & totalmin<=1440,tod:='EV2'] + +# create time intervals +alldata[,todcat:=findInterval(totalmin,seq(0,1440,by=interval))] + +# --------------- 3. ANALYSIS (OUTLIER DETECTION) ------------------- + +if (OUTLIER) { + # ---------------- outlier analysis -------------------- + # create a columns "outlier" with default set to 0 + alldata[,outlier:=0] + + # find tmc segments + segmentslist<-unique(alldata$tmc_code) + + if (length(segmentslist)==0) { + stop("no tmc segments in the dataset") + } + + # create dataframes by tmc_code + alldata_bytmc<-by(alldata,alldata$tmc_code, function(x) x) + + if (length(segmentslist)>length(alldata_bytmc)) { + warning("not all tmc segments have dataframe") + } + + # identify outliers for each dataframe + lapply(alldata_bytmc, detectoutliers) + + # combine dataframes into one dataset + alldata.outliers<-do.call(rbind,alldata_bytmc) + + # calculate length (mile) + alldata.outliers[,seg.length:=speed*travel_time_minutes/60] + alldata.outliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] + + #free-up memory + #rm(alldata) + rm(inrix_2012) + gc() + + if (sum(alldata.outliers$outlier)==0) { + stop("no outliers are detected") + } + + outliers_byIFC<-c(0,0,0,0,0,0,0,0,0) + outliers_byIFC[1]<-sum(alldata.outliers[IFC==1]$outlier) + outliers_byIFC[2]<-sum(alldata.outliers[IFC==2]$outlier) + outliers_byIFC[3]<-sum(alldata.outliers[IFC==3]$outlier) + outliers_byIFC[4]<-sum(alldata.outliers[IFC==4]$outlier) + outliers_byIFC[5]<-sum(alldata.outliers[IFC==5]$outlier) + outliers_byIFC[6]<-sum(alldata.outliers[IFC==6]$outlier) + outliers_byIFC[7]<-sum(alldata.outliers[IFC==7]$outlier) + outliers_byIFC[8]<-sum(alldata.outliers[IFC==8]$outlier) + outliers_byIFC[9]<-sum(alldata.outliers[IFC==9]$outlier) + + # also remove data points in period 4:15 am - 4:30 am. also, remove based on reference_speed + alldata.nooutliers<-alldata.outliers[todcat==18 & reference_speed-speed>5,outlier:=1] + + # remove outliers + alldata.nooutliers<-alldata.outliers[outlier==0] + +} else { + # filter out data points based on confidence score + # lose about 19% of the data points + alldata.nooutliers <- alldata[alldata$confidence_score>10] + alldata.nooutliers[,seg.length:=speed*travel_time_minutes/60] + alldata.nooutliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] +} + +# free-up memory +rm(alldata_bytmc) +rm(alldata) +rm(alldata.outliers) +rm(inrix_2012_ifc) +gc() + +alldata.nooutliers$dayofweek<-dayOfWeek(as.timeDate(alldata.nooutliers$date)) + +# ---------------- 4. OUTPUT ---------------------- +if (OUTLIER){ + # write out the new dataset + write.table(alldata.nooutliers,"inrix_2012_nooutliers_15mins.txt",sep=",",row.names=F,quote=F) +} else { + # write out the new dataset - no confidence score 10 + write.table(alldata.nooutliers,"inrix_2012_noCS10_15mins.txt",sep=",",row.names=F,quote=F) +} + + +if (FALSE){ + # for test + withnooutliers_file="./inrix_2012_nooutliers.txt" + readSaveRdata(withnooutliers_file,"inrix_2012_nooutliers") + inrix_2012_nooutliers <- assignLoad(paste0(withnooutliers_file,".Rdata")) + alldata.nooutliers <- inrix_2012_nooutliers + + alldata.nooutliers[,seg.length:=speed*travel_time_minutes/60] + alldata.nooutliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] + + # for test + withoutliers_file="./inrix_2012_outliers.txt" + readSaveRdata(withoutliers_file,"inrix_2012_outliers") + inrix_2012_outliers <- assignLoad(paste0(withoutliers_file,".Rdata")) + + inrix_2012_outliers[,seg.length:=speed*travel_time_minutes/60] + inrix_2012_outliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] + + alldata.outliers=inrix_2012_outliers + alldata.nooutliers=inrix_2012_outliers[outlier==0] + +} + +# ----------------- 5. Statistics ----------------- + +# for raw data set - outliers are included (though identified) +if (FALSE){ + # calculate SD - all data points + #temp<-cast(alldata.outliers,tmc_code~todcat,sd,value=field) + temp<-dcast(alldata.outliers,tmc_code+date+todcat~field,fun.aggregate=sd) + + alldata.outliers.sd<-temp[complete.cases(temp),] # keeps only complete values - no missing values + temp=alldata.outliers.sd + temp<-merge(temp,tmc_ifc,by="tmc_code") + + #output SD file + #outfile=paste("inrix2012_outliers_",field,"_sd.txt") + #write.table(temp,outfile,sep="\t",row.names=F,quote=F) + + outfile=paste("inrix2012_outliers_",field,"_sd.dbf") + write.dbf(temp,outfile) +} + +# save data +if (OUTLIER) { + # outlier method + save(alldata.nooutliers,file = "alldatanooutliers.Rdata") +} else { + # confidence score 10 method + save(alldata.nooutliers,file = "alldatanoCS10.Rdata") +} + +# load data +if (OUTLIER) { + # outlier method + alldata.nooutliers <- assignLoad("alldatanooutliers.Rdata") +} else { + # confidence score 10 method + alldata.nooutliers <- assignLoad("alldatanoCS10.Rdata") +} + +# std. dev of all data points +if (!ByDOW){ + # std dev for all weekdays + alldata.nooutliers.sd<-cast(alldata.nooutliers,tmc_code~todcat,sd,value=field) + + # reshape the dataset + alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id="tmc_code") + setnames(alldata.nooutliers.sd.melt,"value","sd") + + # field name got changed to "value" due to cast in the previous step - set back to field name + setnames(alldata.nooutliers,"value",field) +} + +if (FALSE) { + # avg speed for Wu - 05/09/2016 + alldata.nooutliers.mean<-aggregate(alldata.nooutliers$speed,by=list(alldata.nooutliers$tmc_code, alldata.nooutliers$todcat),FUN = mean, na.rm=TRUE, na.action = na.pass) + tmc_length<-aggregate(alldata.nooutliers$seg.length,by=list(alldata.nooutliers$tmc_code),FUN = mean, na.rm=TRUE, na.action = na.pass) + + setnames(alldata.nooutliers.mean,c("Group.1","Group.2","x"),c("tmc_code","todcat","avg_speed")) + setnames(tmc_length,c("Group.1","x"),c("tmc_code","avg_tmc_length")) + + alldata.nooutliers.mean.hwycov<-merge(alldata.nooutliers.mean,tmc_ifc,by="tmc_code") + alldata.nooutliers.mean.hwycov<-merge(alldata.nooutliers.mean.hwycov,tmc_length,by="tmc_code") + + alldata.nooutliers.mean.hwycov$TMC_Len<-as.numeric(gsub(",","",alldata.nooutliers.mean.hwycov$TMC_Len)) + write.table(alldata.nooutliers.mean.hwycov,"inrix_2012_avgspeed_wu.csv",sep=",",row.names=F,quote=F) + +} + +# std. dev of all datapoints of a weekday (ex. all monday, all tuesday,.., all friday) - not used +if (ByDOW) { + alldata.nooutliers.sd<-dcast(alldata.nooutliers,tmc_code+todcat~dayofweek,value.var=field,fun.aggregate=sd,na.rm=TRUE) + + if (FALSE){ + + if (field=="speed"){ + alldata.nooutliers.sd<-aggregate(alldata.nooutliers$speed,by=list(alldata.nooutliers$tmc_code,alldata.nooutliers$dayofweek,alldata.nooutliers$todcat),FUN = sd) + } else { + alldata.nooutliers.sd<-aggregate(alldata.nooutliers$travel.time.sec.per.mile,by=list(alldata.nooutliers$tmc_code,alldata.nooutliers$dayofweek,alldata.nooutliers$todcat),FUN = sd) + } + } + # reshape the dataset - not needed in aggregate method + alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id=c("tmc_code","todcat")) + setnames(alldata.nooutliers.sd.melt,c("variable","value"),c("dayofweek","sd")) + +} + +#alldata.nooutliers.sd<-alldata.nooutliers.sd[complete.cases(alldata.nooutliers.sd),] + +# For Shift Variables - now they are calculated in regression analysis script +# calculate mean + +# all data points +if (!ByDOW){ + # calculate mean + alldata.nooutliers.mean<-dcast(alldata.nooutliers,tmc_code~todcat,value.var=field,fun.aggregate=mean,na.rm=TRUE) + alldata.nooutliers.mean.melt<-melt(alldata.nooutliers.mean,id="tmc_code") + setnames(alldata.nooutliers.mean.melt,c("variable","value"),c("todcat","mean")) +} + +# data points by weekday +if (ByDOW){ + alldata.nooutliers.mean<-dcast(alldata.nooutliers,tmc_code+todcat~dayofweek,value.var=field,fun.aggregate=mean,na.rm=TRUE) + alldata.nooutliers.mean.melt<-melt(alldata.nooutliers.mean,id=c("tmc_code","todcat")) + + # set column names + setnames(alldata.nooutliers.mean.melt,c("variable","value"),c("dayofweek","mean")) + +} + +# five model time periods +alldata.nooutliers.mean.melt$todcat.int <- as.integer(alldata.nooutliers.mean.melt$todcat) + +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>0 & alldata.nooutliers.mean.melt$todcat.int<=14,'EV','') +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>14 & alldata.nooutliers.mean.melt$todcat.int<=24,'EA',alldata.nooutliers.mean.melt$tod) +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>24 & alldata.nooutliers.mean.melt$todcat.int<=36,'AM',alldata.nooutliers.mean.melt$tod) +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>36 & alldata.nooutliers.mean.melt$todcat.int<=62,'MD',alldata.nooutliers.mean.melt$tod) +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>62 & alldata.nooutliers.mean.melt$todcat.int<=76,'PM',alldata.nooutliers.mean.melt$tod) +alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>76 & alldata.nooutliers.mean.melt$todcat.int<=96,'EV',alldata.nooutliers.mean.melt$tod) + +if (FALSE) +{ + # max time by tmc_code and dayofweek + #MaxTime.AM<-do.call(rbind,lapply(split(MeanTime.AM,list(MeanTime.AM$tmc_code,MeanTime.AM$dayofweek)), function(x) x[which.max(x$mean),])) + #MaxTime.AM<-MaxTime.AM[,c("tmc_code","dayofweek","todcat")] + #setnames(MaxTime.AM,"todcat","MaxAMtod") + + MaxTime.AM<-do.call(rbind,lapply(split(MeanTime.AM,list(MeanTime.AM$tmc_code)), function(x) x[which.max(x$mean),])) + MaxTime.AM<-MaxTime.AM[,c("tmc_code","todcat.int")] + setnames(MaxTime.AM,"todcat.int","MaxAMtod") + + #MaxTime.PM<-do.call(rbind,lapply(split(MeanTime.PM,list(MeanTime.PM$tmc_code,MeanTime.PM$dayofweek)), function(x) x[which.max(x$mean),])) + #MaxTime.PM<-MaxTime.PM[,c("tmc_code","dayofweek","todcat")] + #setnames(MaxTime.PM,"todcat","MaxPMtod") + + MaxTime.PM<-do.call(rbind,lapply(split(MeanTime.PM,list(MeanTime.PM$tmc_code)), function(x) x[which.max(x$mean),])) + MaxTime.PM<-MaxTime.PM[,c("tmc_code","todcat.int")] + setnames(MaxTime.PM,"todcat.int","MaxPMtod") + + # merge + #temp<-merge(alldata.nooutliers.sd.melt,MaxTime.AM,by=c("tmc_code","dayofweek"), all.x=TRUE) + #temp1<-merge(temp,MaxTime.PM,by=c("tmc_code","dayofweek"), all.x=TRUE) + + temp<-merge(alldata.nooutliers.sd.melt,MaxTime.AM,by="tmc_code", all.x=TRUE) + temp1<-merge(temp,MaxTime.PM,by="tmc_code", all.x=TRUE) +} + +# merge sd and mean values +temp<-merge(alldata.nooutliers.sd.melt,alldata.nooutliers.mean.melt,by=c("tmc_code","todcat")) +temp$sdpermean<-temp$sd/temp$mean + +# estimation dataset +data.est <- temp +data.est <- data.est[,c("tmc_code","todcat","sd","mean","todcat.int","tod","sdpermean")] + +# save data +if (OUTLIER) { + save(data.est,file = "data.est.nooutliers.Rdata") +} else { + save(data.est,file = "data.est.noCS10.Rdata") +} + +# load data +if (OUTLIER) { + data.est <- assignLoad("data.est.nooutliers.Rdata") +} else { + data.est <- assignLoad("data.est.noCS10.Rdata") +} + +data.est<-na.omit(data.est) + +# don't merge attributes here +#data.est<-merge(data.est,tmc_ifc,by="tmc_code") + +if (OUTLIER) { + # output without outliers SD file - in DBF format + outfile="inrix2012_nooutliers_traveltime.dbf" +} else { + outfile="inrix2012_noCS10_traveltime.dbf" +} + +# write to dbf file +write.dbf(data.est,outfile) + +# output without outliers SD file - in text format +if (FALSE){ + outfile=paste("inrix2012_nooutliers_",field,"_sd.txt") + write.table(temp,outfile,sep="\t",row.names=F,quote=F) +} + +rm(alldata.nooutliers.mean) +gc() + +# ----------------- 6. Plots ----------------- + +# arrange data +alldata.outliers.sd.melt<-melt(alldata.outliers.sd,id="tmc_code") +alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id="tmc_code") + +# determine time of day (TOD) + +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=0 & alldata.outliers.sd.melt$todcat<=7,'EV1','') +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=7 & alldata.outliers.sd.melt$todcat<=12,'EA',alldata.outliers.sd.melt$tod) +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=12 & alldata.outliers.sd.melt$todcat<=18,'AM',alldata.outliers.sd.melt$tod) +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=18 & alldata.outliers.sd.melt$todcat<=31,'MD',alldata.outliers.sd.melt$tod) +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=31 & alldata.outliers.sd.melt$todcat<=38,'PM',alldata.outliers.sd.melt$tod) +alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=38 & alldata.outliers.sd.melt$todcat<=48,'EV2',alldata.outliers.sd.melt$tod) + +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=0 & alldata.nooutliers.sd.melt$todcat<7,'EV1','') +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=7 & alldata.nooutliers.sd.melt$todcat<12,'EA',alldata.nooutliers.sd.melt$tod) +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=12 & alldata.nooutliers.sd.melt$todcat<18,'AM',alldata.nooutliers.sd.melt$tod) +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=18 & alldata.nooutliers.sd.melt$todcat<31,'MD',alldata.nooutliers.sd.melt$tod) +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=31 & alldata.nooutliers.sd.melt$todcat<38,'PM',alldata.nooutliers.sd.melt$tod) +alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=38 & alldata.nooutliers.sd.melt$todcat<=48,'EV2',alldata.nooutliers.sd.melt$tod) + +# arrange data +alldata.outliers.sd.melt<-merge(alldata.outliers.sd.melt,tmc_ifc_temp,by="tmc_code") +alldata.nooutliers.sd.melt<-merge(alldata.nooutliers.sd.melt,tmc_ifc_temp,by="tmc_code") + +# segment by facility type (IFC) +alldata.outliers.sd.melt.byifc<-by(alldata.outliers.sd.melt,alldata.outliers.sd.melt$IFC, function(x) x) +alldata.nooutliers.sd.melt.byifc<-by(alldata.nooutliers.sd.melt,alldata.nooutliers.sd.melt$IFC, function(x) x) + +# make plots +for (p in 1:length(alldata.outliers.sd.melt.byifc)) { + + plot1<-myplot_sd(alldata.outliers.sd.melt.byifc[[p]],"raw",outliers_byIFC[p],field) + plot2<-myplot_sd(alldata.nooutliers.sd.melt.byifc[[p]],"nooutliers",outliers_byIFC[p],field) + + # save as JPEG + print(multiplot(plot1,plot2),cols=1) + dev.copy(jpeg,filename=paste(outputsDir,field, "_SD_","IFC_",p,".jpeg", sep = ""),width=1280, height=1280) + dev.off() + +} + +#debug +#myplot_sd(alldata.nooutliers.sd.melt.byifc[["9"]]) + +myplot_sd(alldata.outliers.sd.melt,"SD_per_mile_outliers.jpeg") +myplot_sd(alldata.nooutliers.sd.melt,"SD_per_mile_nooutliers.jpeg") + +# ---------------- 5. PLTOS ---------------------- +if(FALSE) { + alldata.outliers.day<-alldata.outliers[date=="2012-10-01"] + + alldata.outliers.day[outlier==0, color.codes:="#000000"] #black + alldata.outliers.day[outlier==1, color.codes:="#FF0000"] #red + + alldata.outliers.day[outlier==0, color.names:="valid"] + alldata.outliers.day[outlier==1, color.names:="outlier"] + + # all freeways of a day data points + alldata.facility.type<-by(alldata.outliers.day,alldata.outliers.day$IFC, function(x) x) + + lapply(alldata.facility.type, myplot1) + + alldata.nooutliers.day<-alldata.nooutliers[date=="2012-10-01"] + + alldata.nooutliers.day[outlier==0, color.codes:="#000000"] #black + alldata.nooutliers.day[outlier==1, color.codes:="#FF0000"] #red + + alldata.nooutliers.day[outlier==0, color.names:="valid"] + alldata.nooutliers.day[outlier==1, color.names:="outlier"] + + alldata.facility.type<-by(alldata.nooutliers.day,alldata.nooutliers.day$IFC, function(x) x) + + lapply(alldata.facility.type, myplot1) +} + + diff --git a/sandag_abm/src/main/r/RegressionAnalysis_Final.R b/sandag_abm/src/main/r/RegressionAnalysis_Final.R new file mode 100644 index 0000000..830bb36 --- /dev/null +++ b/sandag_abm/src/main/r/RegressionAnalysis_Final.R @@ -0,0 +1,953 @@ +### +# OBJECTIVE: +# Regression analysis for SANDAG INRIX travel time data +# +# INPUTS: +# travel time SD/ mean travel time in 15 mins (*.dbf) - "inrix2012_nooutliers_traveltime.dbf" (output of inrix outlier analysis) +# inrix to tcoved correspondence (created by Fehr and Peers) - "inrix_2012hwy.txt" +# ABM assignment results by 5 time periods - "hwyload_EA.csv", "hwyload_AM.csv", "hwyload_MD.csv", "hwyload_PM.csv", "hwyload_EV.csv" +# Major interchange distance from SANDAG ABM output folder - "MajorInterchangeDistance.csv" +# other inputs (not used in final analysis) - "InterchangeDistance.csv", "LaneChangeDistance.csv", "FreewayRampMeters.csv" + +# DATA: +# IFC in the roadway network database is re-categorized into four facility classes +# Facility Class IFC Description +# Freeway 1 Freeways +# Arterial 2,3 Major arterials, prime arterials +# Ramp 8,9 Local ramps, freeway ramps +# Other 4,5,6,7 Collectors and local streets +# 80% for estimation (regression analysis) and 20% kept for validation +# low sample size for ramps and others - only freeway and arterial estimations are used + +# REGRESSION VARIABLES (final): +# +# Dependent Variable: +# Travel time per mile Std. Dev. per mean travel time in 15 mins time slices +# +# Independent Variables: +# Number of lanes categories (one, two, three, four, five and more) +# Level of service (LOS) +# LOSC+ = (V/C-0.69)* if (V/C>=0.70) +# LOSD+ = (V/C-0.79)* if (V/C>=0.80) +# LOSE+ = (V/C-0.89)* if (V/C>=0.90) +# LOSF_LOW = (V/C-0.99)* if (V/C>=1.00) +# LOSF_MED = (V/C-1.09)* if (V/C>=1.10) +# LOSF_HIGH = (V/C-1.19)* if (V/C>=1.20) +# Speed +# ISPD70 (1 if posted speed is 70mph, 0 otherwise) for Freeways +# Posted speed categories (ISPD<35, ISPD=35, ISPD=40, ISPD=45, ISPD=50, ISPD>50) for Arterials +# Shift variables (BeforeAM.Shift, AfterAM.Shift, BeforePM.Shift, AfterPM.Shift) +# Control type (none, signal, stop, railroad, ramp meter) +# Upstream and downstream major interchange distance - from midpoint of a freeway segment. inverse distances are used in estimation + +# IMPLEMENTATION (in SANDAG ABM): +# +# By facility type and by 5 time periods +# Estimations for freeways and arterials +# Ramp and other facility types are applied with arterial estimation +# Two reliability components +# LOS +# Static (speed, distance to/from interchanges, intersection type etc.) +# Sum of the two reliability components is multiplied by mean travel time and link length +# Variable calculations are automated including distance to/from interchanges +# Reliability fields are added to highway network +# LOS: include only coefficients +# Static: sum of remaining (un)reliability including the intercept +# Skimming: +# Standard deviation is not additive but variance is +# A link (un)reliability = (MSA Cost – MSA Time) +# Skimmed variance (square of link (un)reliability) +# Final skims are square root of the skimmed value + +# OTHER SCRIPTS USED: +# utilfunc.R + +# by: nagendra.dhakar@rsginc.com +# for: SANDAG SHRP C04 - Tolling and Reliability +#---------------------------------------------------- + +library(foreign) +library(stringr) +library(xtable) +library(reshape) +library(XLConnect) +library(descr) +library(Hmisc) +library(data.table) +library(plyr) +library(gtools) +library(vioplot) +library(lattice) +library(grid) +library(timeDate) +library(ggplot2) +library(robustbase) +library(readr) + +# -------------------- 0. SOURCE FILES AND CONFIG SETTINGS ----------------------- + +# workspace and source files +setwd("E:/Projects/Clients/SANDAG") +source("./Data_2015/utilfunc.R") + +# method of outlier removal - outlier analysis (this is used) or confidence interval 10 +OUTLIER=TRUE + +# input files +inrixhwycorresp.file = "./Data_2015/TMC/inrix_2012hwy.txt" +AssignResultsEA.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_EA.csv" +AssignResultsAM.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_AM.csv" +AssignResultsMD.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_MD.csv" +AssignResultsPM.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_PM.csv" +AssignResultsEV.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_EV.csv" +InterchangeDistance.file = "./Data_2015/InterchangeDistance.csv" +InterchangeDistanceHOV.file = "./Data_2015/InterchangeDistance_HOV.csv" +MajorInterchangeDistance.file = "./Data_2015/MajorInterchangeDistance.csv" +LaneChangeDistance.file = "./Data_2015/LaneChangeDistance.csv" +FreewayRampMeters.file = "./Data_2015/FreewayRampMeters.csv" + +# travel time SD input file +if (OUTLIER) { + StdDev.file = "./Data_2015/inrix2012_nooutliers_traveltime.dbf" +} else { + StdDev.file = "./Data_2015/inrix2012_noCS10_traveltime.dbf" +} + +# outputs directory +outputsDir="./Data_2015/Results" + +# -------------------- 1. LOAD DATA ---------------------- +# load SD file, assignment results, interchange distance etc. +StdDev<- read.dbf(StdDev.file) +AssignResultsEA <- read.csv(AssignResultsEA.file) +AssignResultsAM <- read.csv(AssignResultsAM.file) +AssignResultsMD <- read.csv(AssignResultsMD.file) +AssignResultsPM <- read.csv(AssignResultsPM.file) +AssignResultsEV <- read.csv(AssignResultsEV.file) +InterchangeDistance <- read.csv(InterchangeDistance.file) +InterchangeDistanceHOV <- read.csv(InterchangeDistanceHOV.file) +MajorInterchangeDistance <- read.csv(MajorInterchangeDistance.file) +LaneChangeDistance <- read.csv(LaneChangeDistance.file) # LinkID,Length,LaneIncrease,LaneDrop,DeadEnd,RegionEnd,DownstreamDistance,ihov,BaseThruLanes,DownThruLanes,DownLinks,QueryDown +FreewayRampMeters <- read.csv(FreewayRampMeters.file) # LinkID,Length,IsRampMeter,QueryNode + +# -------------------- 2. SET UP DATA -------------- + +# read and load inrix to hwy correspondence data +readSaveRdata(inrixhwycorresp.file,"inrixhwycorresp") +inrixhwycorresp <- assignLoad(paste0(inrixhwycorresp.file,".Rdata")) + +# add a new field tmc_code without first character ('-' or '+') +inrixhwycorresp[,tmc_code:=substr(TMC,2,nchar(TMC))] + +# remove the segments that are flagged +inrixhwycorresp<-subset(inrixhwycorresp,FLAG==0) + +# get unique tmc_code with variables corresponding to maximum of TMCProp +df.orig <-inrixhwycorresp +df.agg<-aggregate(TMCProp~tmc_code,inrixhwycorresp,max) +df.max <- merge(df.agg, df.orig) + +# Note: there could be cases where one hwcov_id is associated with multiple tmc_code + +# facility type - different capacity fields for mid-link capacity (period fields) +tmc_ifc <-df.max[,c("tmc_code","IFC","HWYCOV_ID","NM","LENGTH","ISPD","ABLNO","ABLNA","ABLNP", + "BALNO","BALNA","BALNP","ABCPO","ABCPA","ABCPP","BACPO","BACPA","BACPP", + "ABCXO","ABCXA","ABCXP","BACXO","BACXA","BACXP","ABCNT","BACNT", + "ABTL","ABRL","ABLL","BATL","BARL","BALL","ABGC","BAGC","IHOV","ABAU","BAAU"),] + +tmc_ifc$LENGTH<-as.numeric(gsub(",","",tmc_ifc$LENGTH)) +tmc_ifc$HWYCOV_ID<-as.numeric(gsub(",","",tmc_ifc$HWYCOV_ID)) + +# merge attributes to INRIX data +StdDev<-merge(StdDev,tmc_ifc,by="tmc_code") + +# ---------------------------------------------------- + +# write to file +if (FALSE) { + write.table(tmc_ifc,"./Data_2015/tmc_hwycov_atrributes.csv",sep=",",row.names=F,quote=F) +} + +# ID1 in the results is the same as HWYCOV_ID in the link file +AssignResultsEA <- AssignResultsEA[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] +AssignResultsAM <- AssignResultsAM[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] +AssignResultsMD <- AssignResultsMD[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] +AssignResultsPM <- AssignResultsPM[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] +AssignResultsEV <- AssignResultsEV[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] + +# use 80% for estimation and 20% for validation - sample segments by facility type + +# create new facility type - freeways, arterials, ramps, and others +StdDev$IFC_Est <- ifelse(StdDev$IFC==1,1,0) # freeways +StdDev$IFC_Est <- ifelse(StdDev$IFC==2 | StdDev$IFC==3,2,StdDev$IFC_Est) # arterials - major and prime +StdDev$IFC_Est <- ifelse(StdDev$IFC==8 | StdDev$IFC==9,3,StdDev$IFC_Est) # ramps - local ramps and freeways ramps +StdDev$IFC_Est <- ifelse(StdDev$IFC>=4 & StdDev$IFC<=7,4,StdDev$IFC_Est) # others - collectors and local streets + +Sample.Rate<-0.8 + +# Freeways +StdDev.Freeways <- subset(StdDev, IFC_Est==1) +StdDev.Freeways.Seg <- unique(StdDev.Freeways$tmc_code) +StdDev.Freeways.Seg<-as.data.frame(StdDev.Freeways.Seg) + +sample_size <- floor(Sample.Rate *nrow(StdDev.Freeways.Seg)) +set.seed(123) +Est.Ind <- sample(seq_len(nrow(StdDev.Freeways.Seg)), size = sample_size) +StdDev.Freeways.Seg.Est <- StdDev.Freeways.Seg[Est.Ind,] +StdDev.Freeways.Seg.Val <- StdDev.Freeways.Seg[-Est.Ind,] + +# Arterials +StdDev.Arterials <- subset(StdDev, IFC_Est==2) +StdDev.Arterials.Seg <- unique(StdDev.Arterials$tmc_code) +StdDev.Arterials.Seg<-as.data.frame(StdDev.Arterials.Seg) + +sample_size <- floor(Sample.Rate *nrow(StdDev.Arterials.Seg)) +set.seed(123) +Est.Ind <- sample(seq_len(nrow(StdDev.Arterials.Seg)), size = sample_size) +StdDev.Arterials.Seg.Est <- StdDev.Arterials.Seg[Est.Ind,] +StdDev.Arterials.Seg.Val <- StdDev.Arterials.Seg[-Est.Ind,] + +# Ramps +StdDev.Ramps <- subset(StdDev, IFC_Est==3) +StdDev.Ramps.Seg <- unique(StdDev.Ramps$tmc_code) +StdDev.Ramps.Seg<-as.data.frame(StdDev.Ramps.Seg) + +sample_size <- floor(Sample.Rate *nrow(StdDev.Ramps.Seg)) +set.seed(123) +Est.Ind <- sample(seq_len(nrow(StdDev.Ramps.Seg)), size = sample_size) +StdDev.Ramps.Seg.Est <- StdDev.Ramps.Seg[Est.Ind,] +StdDev.Ramps.Seg.Val <- StdDev.Ramps.Seg[-Est.Ind,] + +# Others +StdDev.Others <- subset(StdDev, IFC_Est==4) +StdDev.Others.Seg <- unique(StdDev.Others$tmc_code) +StdDev.Others.Seg<-as.data.frame(StdDev.Others.Seg) + +sample_size <- floor(Sample.Rate *nrow(StdDev.Others.Seg)) +set.seed(123) +Est.Ind <- sample(seq_len(nrow(StdDev.Others.Seg)), size = sample_size) +StdDev.Others.Seg.Est <- StdDev.Others.Seg[Est.Ind,] +StdDev.Others.Seg.Val <- StdDev.Others.Seg[-Est.Ind,] + +# ------------------ Estimation Dataset -------------------------------------- + +StdDev.Freeways.Seg.Est<-as.data.frame(StdDev.Freeways.Seg.Est) +StdDev.Arterials.Seg.Est<-as.data.frame(StdDev.Arterials.Seg.Est) +StdDev.Ramps.Seg.Est<-as.data.frame(StdDev.Ramps.Seg.Est) +StdDev.Others.Seg.Est<-as.data.frame(StdDev.Others.Seg.Est) + +setnames(StdDev.Freeways.Seg.Est,"StdDev.Freeways.Seg.Est","tmc_code") +setnames(StdDev.Arterials.Seg.Est,"StdDev.Arterials.Seg.Est","tmc_code") +setnames(StdDev.Ramps.Seg.Est,"StdDev.Ramps.Seg.Est","tmc_code") +setnames(StdDev.Others.Seg.Est,"StdDev.Others.Seg.Est","tmc_code") + +# combine dataframes into one +StdDev.Seg.Est <- do.call(rbind,list(StdDev.Freeways.Seg.Est,StdDev.Arterials.Seg.Est,StdDev.Ramps.Seg.Est,StdDev.Others.Seg.Est)) + +# merge data +StdDev.Est<-merge(StdDev,StdDev.Seg.Est,by="tmc_code") + +# ---------------------------- Validation Dataset ------------------------------ + +StdDev.Freeways.Seg.Val<-as.data.frame(StdDev.Freeways.Seg.Val) +StdDev.Arterials.Seg.Val<-as.data.frame(StdDev.Arterials.Seg.Val) +StdDev.Ramps.Seg.Val<-as.data.frame(StdDev.Ramps.Seg.Val) +StdDev.Others.Seg.Val<-as.data.frame(StdDev.Others.Seg.Val) + +setnames(StdDev.Freeways.Seg.Val,"StdDev.Freeways.Seg.Val","tmc_code") +setnames(StdDev.Arterials.Seg.Val,"StdDev.Arterials.Seg.Val","tmc_code") +setnames(StdDev.Ramps.Seg.Val,"StdDev.Ramps.Seg.Val","tmc_code") +setnames(StdDev.Others.Seg.Val,"StdDev.Others.Seg.Val","tmc_code") + +# combine dataframes into one +StdDev.Seg.Val <- do.call(rbind,list(StdDev.Freeways.Seg.Val,StdDev.Arterials.Seg.Val,StdDev.Ramps.Seg.Val,StdDev.Others.Seg.Val)) + +# merge data +StdDev.Val<-merge(StdDev,StdDev.Seg.Val,by="tmc_code") + +# ------------------- Add Interchange Distance ----------------------- +if (FALSE) { + # Interchange distance + InterchangeDistance <-InterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] + InterchangeDistanceHOV <-InterchangeDistanceHOV[,c("LinkID","Length","upstream.distance","downstream.distance")] + + temp<-merge(InterchangeDistance,InterchangeDistanceHOV,by="LinkID",all.x = TRUE) + temp$upstream.distance <- ifelse(!is.na(temp$Length),temp$upstream.distance.y,temp$upstream.distance.x) + temp$downstream.distance<-ifelse(!is.na(temp$Length),temp$downstream.distance.y,temp$downstream.distance.x) + + InterchangeDistance<-temp + InterchangeDistance <-InterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] + + temp<-merge(StdDev.Est,InterchangeDistance,by.x ="HWYCOV_ID",by.y = "LinkID",all.x=TRUE) + + StdDev.Est<-temp +} + +# Major Interchange distance +MajorInterchangeDistance <-MajorInterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] + +temp<-merge(MajorInterchangeDistance,InterchangeDistanceHOV,by="LinkID",all.x = TRUE) +temp$majorupstream.distance <- ifelse(!is.na(temp$Length),temp$upstream.distance.y,temp$upstream.distance.x) +temp$majordownstream.distance<-ifelse(!is.na(temp$Length),temp$downstream.distance.y,temp$downstream.distance.x) + +MajorInterchangeDistance<-temp +MajorInterchangeDistance <-MajorInterchangeDistance[,c("LinkID","majorupstream.distance","majordownstream.distance")] + +temp<-merge(StdDev.Est,MajorInterchangeDistance,by.x ="HWYCOV_ID",by.y = "LinkID",all.x=TRUE) + +StdDev.Est<-temp + +# FF Time (seconds) +StdDev.Est$FF_Time <- as.numeric(StdDev.Est$LENGTH)*(1/5280)*(1/StdDev.Est$ISPD)*60 +#test <- subset(StdDev.Est, is.na(StdDev.Est$FF_Time)) # just to see if there are any NA values + +# model tod +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>0 & StdDev.Est$todcat<=14,'EV','') #3.5 hours +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>14 & StdDev.Est$todcat<=24,'EA',StdDev.Est$tod.model) #2.5 hours +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>24 & StdDev.Est$todcat<=36,'AM',StdDev.Est$tod.model) #3 hours +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>36 & StdDev.Est$todcat<=62,'MD',StdDev.Est$tod.model) #6.5 hours +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>62 & StdDev.Est$todcat<=76,'PM',StdDev.Est$tod.model) #3.5 hours +StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>76 & StdDev.Est$todcat<=96,'EV',StdDev.Est$tod.model) #5 hours + +# convert to integer - mid link (*CPO, *CPA, *CPP) and intersection (*CXO, *CXA, *CXP) capacities +StdDev.Est$ABCPO<-as.integer(gsub(",","",StdDev.Est$ABCPO)) +StdDev.Est$ABCPA<-as.integer(gsub(",","",StdDev.Est$ABCPA)) +StdDev.Est$ABCPP<-as.integer(gsub(",","",StdDev.Est$ABCPP)) +StdDev.Est$BACPO<-as.integer(gsub(",","",StdDev.Est$BACPO)) +StdDev.Est$BACPA<-as.integer(gsub(",","",StdDev.Est$BACPA)) +StdDev.Est$BACPP<-as.integer(gsub(",","",StdDev.Est$BACPP)) +StdDev.Est$ABCXO<-as.integer(gsub(",","",StdDev.Est$ABCXO)) +StdDev.Est$ABCXA<-as.integer(gsub(",","",StdDev.Est$ABCXA)) +StdDev.Est$ABCXP<-as.integer(gsub(",","",StdDev.Est$ABCXP)) +StdDev.Est$BACXO<-as.integer(gsub(",","",StdDev.Est$BACXO)) +StdDev.Est$BACXA<-as.integer(gsub(",","",StdDev.Est$BACXA)) +StdDev.Est$BACXP<-as.integer(gsub(",","",StdDev.Est$BACXP)) + +# set initial values to 0 +StdDev.Est$NumLanes <-0 +StdDev.Est$ICNT <-0 # intersection control +StdDev.Est$Flow <-0 +StdDev.Est$CAP.MidLink <-0 # mid-link capacity for freeways and ramps +StdDev.Est$CAP.IntAppr <-0 # intersection-approach capacity for arterials and others +StdDev.Est$AuxLanes <-0 +StdDev.Est$ThruLanes <-0 +StdDev.Est$LeftLanes <-0 +StdDev.Est$RightLanes <-0 +StdDev.Est$GCRatio <-0 +Default.Cap<-9999999 # set a very high + +# TMC Codes +# External (between interchanges): '+' (NB or WB - positive direction), '-' (SB or EB - negative direction) +# Internal (within interchanges): 'P' (NB or WB), 'N' (SB or EB) + +# -------------------------------- ESTIMATION --------------------------------- +# This section estimates regression equations using estimation dataset (StdDev.Est) + +# the below is how SANDAG ABM (gisdk) converts capacities from three periods to 5 model periods: +# set capacity fields +# tod_fld ={{"ABCP_EA"},{"ABCP_AM"},{"ABCP_MD"},{"ABCP_PM"},{"ABCP_EV"}, //BA link capacity +# {"BACP_EA"},{"BACP_AM"},{"BACP_MD"},{"BACP_PM"},{"BACP_EV"}, //AB link capacity +# {"ABCX_EA"},{"ABCX_AM"},{"ABCX_MD"},{"ABCX_PM"},{"ABCX_EV"}, //BA intersection capacity +# {"BACX_EA"},{"BACX_AM"},{"BACX_MD"},{"BACX_PM"},{"BACX_EV"}} //AB intersection capacity + +# org_fld ={"ABCPO","ABCPA","ABCPO","ABCPP","ABCPO", +# "BACPO","BACPA","BACPO","BACPP","BACPO", +# "ABCXO","ABCXA","ABCXO","ABCXP","ABCXO", +# "BACXO","BACXA","BACXO","BACXP","BACXO"} + +# factor ={"3/12","1","6.5/12","3.5/3","8/12", +# "3/12","1","6.5/12","3.5/3","8/12", +# "3/12","1","6.5/12","3.5/3","8/12", +# "3/12","1","6.5/12","3.5/3","8/12"} +# +# the capacity calculations below are consistent with the ABM gisdk + +# MERGE assignment results to the estimation data + +setnames(StdDev.Est,"HWYCOV_ID","ID1") +nrow(StdDev.Est) + +# EA Period +temp2 <- merge(StdDev.Est,AssignResultsEA,by="ID1", all.x=TRUE) +temp2$Flow <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) +temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*3/12,temp2$ABCPO*3/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*3/12,temp2$BACPO*3/12)),temp2$CAP.MidLink)) +temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*3/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*3/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*3/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*3/12,Default.Cap))),temp2$CAP.IntAppr)) +temp2$NumLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) +temp2$ICNT <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) +temp2$AuxLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) +temp2$ThruLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) +temp2$LeftLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) +temp2$RightLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) +temp2$GCRatio <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) + +temp2<-temp2[,-c(61:66)] +nrow(temp2) + +# AM Period +temp2 <- merge(temp2,AssignResultsAM,by="ID1", all.x=TRUE) +temp2$Flow <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) +temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPA,temp2$ABCPA),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPA,temp2$BACPA)),temp2$CAP.MidLink)) # AM capacity is for 3 hours +temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXA,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXA,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXA,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXA,Default.Cap))),temp2$CAP.IntAppr)) +temp2$NumLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNA,temp2$ABLNA),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNA,temp2$BALNA)),temp2$NumLanes) +temp2$ICNT <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) +temp2$AuxLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) +temp2$ThruLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) +temp2$LeftLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) +temp2$RightLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) +temp2$GCRatio <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) + +temp2<-temp2[,-c(61:66)] +nrow(temp2) + +# MD Period +temp2 <- merge(temp2,AssignResultsMD,by="ID1", all.x=TRUE) +temp2$Flow <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # the flow is for 9 am - 3:30 pm +temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*6.5/12,temp2$ABCPO*6.5/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*6.5/12,temp2$BACPO*6.5/12)),temp2$CAP.MidLink)) +temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*6.5/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*6.5/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*6.5/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*6.5/12,Default.Cap))),temp2$CAP.IntAppr)) +temp2$NumLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) +temp2$ICNT <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) +temp2$AuxLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) +temp2$ThruLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) +temp2$LeftLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) +temp2$RightLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) +temp2$GCRatio <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) + +temp2<-temp2[,-c(61:66)] +nrow(temp2) + +# PM Period +temp2 <- merge(temp2,AssignResultsPM,by="ID1", all.x=TRUE) +temp2$Flow <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # flow is from 3:30 pm to 7 pm +temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPP*3.5/3,temp2$ABCPP*3.5/3),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPP*3.5/3,temp2$BACPP*3.5/3)),temp2$CAP.MidLink)) # PM capacity is for 3 hours +temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXP*3.5/3,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXP*3.5/3,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXP*3.5/3,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXP*3.5/3,Default.Cap))),temp2$CAP.IntAppr)) +temp2$NumLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNP,temp2$ABLNP),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNP,temp2$BALNP)),temp2$NumLanes) +temp2$ICNT <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) +temp2$AuxLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) +temp2$ThruLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) +temp2$LeftLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) +temp2$RightLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) +temp2$GCRatio <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) + +temp2<-temp2[,-c(61:66)] +nrow(temp2) + +# EV Period +temp2 <- merge(temp2,AssignResultsPM,by="ID1", all.x=TRUE) +temp2$Flow <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # flow is from 7 pm to 3:30 am +temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*8/12,temp2$ABCPO*8/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*8/12,temp2$BACPO*8/12)),temp2$CAP.MidLink)) # OP capacity is for 18 hours +temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*8/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*8/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*8/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*8/12,Default.Cap))),temp2$CAP.IntAppr)) +temp2$NumLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) +temp2$ICNT <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) +temp2$AuxLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) +temp2$ThruLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) +temp2$LeftLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) +temp2$RightLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) +temp2$GCRatio <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) + +temp2<-temp2[,-c(61:66)] + +nrow(temp2) + +# intersection approach capacity, when not available set to 0 +#temp2$CAP.IntAppr<-ifelse(temp2$CAP.IntAppr==9999999,0,temp2$CAP.IntAppr) +temp2$CAP.IntAppr<-ifelse(temp2$CAP.IntAppr==9999999,temp2$CAP.MidLink,temp2$CAP.IntAppr) + +# values of greater and equal to 7 are for non-availability for some reasons, set them to 0 +temp2$ThruLanes <-ifelse(temp2$ThruLanes>=7,0,temp2$ThruLanes) +temp2$LeftLanes <-ifelse(temp2$LeftLanes>=7,0,temp2$LeftLanes) +temp2$RightLanes <-ifelse(temp2$RightLanes>=7,0,temp2$RightLanes) + +# recalculate GC ratio as per createhwynet.rsc (line 1045) +temp2$GCRatio<-ifelse(temp2$GCRatio>10,temp2$GCRatio/100,temp2$GCRatio) +temp2$GCRatio<-ifelse(temp2$GCRatio>1,1,temp2$GCRatio) + +# ----------------------- Create Variables for Regression Models ----------------------------- +EstDataSet<-temp2 + +# NOTE: Link ID (HWYCOV_ID)=29412 29413 31194 31202,40479 are not in the model network. So remove those for now +EstDataSet<-subset(EstDataSet, !is.na(EstDataSet$Flow)) + +# Capacity by facility type (12/07/2015) +# freeways and ramps - mid link capacity +# arterials and others - intersection approach +#EstDataSet$CAP<-ifelse(EstDataSet$IFC_Est==1 | EstDataSet$IFC_Est==3, EstDataSet$CAP.MidLink, EstDataSet$CAP.IntAppr) + +EstDataSet$CAP<-EstDataSet$CAP.MidLink # mid link capacity for all +EstDataSet$VOC <- EstDataSet$Flow/EstDataSet$CAP + +# control type - 0-none, 1-signal, 2-stop, 3-railroad +EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT==1,"Signal","None") # signal +EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT==2 | EstDataSet$ICNT==3,"Stop",EstDataSet$ICNT.Est) # all-way and two way stop +EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT>3,"RailRoad",EstDataSet$ICNT.Est) # other-primarily rail-road crossing +#set the order of factors +EstDataSet$ICNT.Est<-factor(EstDataSet$ICNT.Est, levels = c("None","Signal", "Stop", "RailRoad")) + +# for descriptives +EstDataSet$ICNT.Signal <-ifelse(EstDataSet$ICNT==1,1,0) # signal +EstDataSet$ICNT.Stop <-ifelse(EstDataSet$ICNT==2 | EstDataSet$ICNT==3,1,0) # all-way and two way stop +EstDataSet$ICNT.RailRoad <-ifelse(EstDataSet$ICNT>3,1,0) # other-primarily rail-road crossing +EstDataSet$ICNT.RampMeter <-ifelse(EstDataSet$ICNT==4 | EstDataSet$ICNT==5,1,0) # Ramp meter, ramp meter with HOV bypass (12/07/2015) + +# Major and Minor Arterial - this would be used only for arterials +EstDataSet$MajorArterial <-ifelse(EstDataSet$IFC==2,1,0) + +# I-15 (managed lanes) and SR-125 +EstDataSet$I15<-ifelse(EstDataSet$IFC==1 & EstDataSet$IHOV==2 & str_sub(EstDataSet$NM,1,4)=="I-15",1,0) # I15 +#EstDataSet$SR125<-ifelse(EstDataSet$IFC==1 & EstDataSet$ihov==4,1,0) # No SR125 facility in the dataset + +# LOS variables - additive and multiplicative (12/07/2015) +EstDataSet$LOSC.Up <- ifelse(EstDataSet$VOC>=0.70,EstDataSet$VOC-0.69,0) # LOS C+ +EstDataSet$LOSD.Up <- ifelse(EstDataSet$VOC>=0.80,EstDataSet$VOC-0.79,0) # LOS D+ +EstDataSet$LOSE.Up <- ifelse(EstDataSet$VOC>=0.90,EstDataSet$VOC-0.89,0) # LOS E+ +EstDataSet$LOSF.Low.Up <- ifelse(EstDataSet$VOC>=1.00,EstDataSet$VOC-0.99,0) # LOS F Low+ +EstDataSet$LOSF.Med.Up <- ifelse(EstDataSet$VOC>=1.10,EstDataSet$VOC-1.09,0) # LOS F Med+ +EstDataSet$LOSF.High.Up <- ifelse(EstDataSet$VOC>=1.20,EstDataSet$VOC-1.19,0) # LOS F High+ + +# LOS variables - additive and multiplicative - capping VOC +#EstDataSet$LOSC.Up <- ifelse(EstDataSet$VOC>=0.70 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.69,0) # LOS C+ +#EstDataSet$LOSD.Up <- ifelse(EstDataSet$VOC>=0.80 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.79,0) # LOS D+ +#EstDataSet$LOSE.Up <- ifelse(EstDataSet$VOC>=0.90 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.89,0) # LOS E+ +#EstDataSet$LOSF.Low.Up <- ifelse(EstDataSet$VOC>=1.00,EstDataSet$VOC-0.99,0) # LOS F Low+ +#EstDataSet$LOSF.Med.Up <- ifelse(EstDataSet$VOC>=1.10,EstDataSet$VOC-1.09,0) # LOS F Med+ +#EstDataSet$LOSF.High.Up <- ifelse(EstDataSet$VOC>=1.20,EstDataSet$VOC-1.19,0) # LOS F High+ + +# LOS categories - for descriptives only +EstDataSet$LOS.Cat <-"LOSB" +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.70 & EstDataSet$VOC<0.80,"LOSC",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.80 & EstDataSet$VOC<0.90,"LOSD",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.90 & EstDataSet$VOC<1.00,"LOSE",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.00 & EstDataSet$VOC<1.10,"LOSF.Low",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.10 & EstDataSet$VOC<1.20,"LOSF.Med",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.20,"LOSF.High",EstDataSet$LOS.Cat) +EstDataSet$LOS.Cat<-factor(EstDataSet$LOS.Cat, levels = c("LOSB","LOSC","LOSD", "LOSE", "LOSF.Low","LOSF.Med","LOSF.High")) + +# NumLanes +EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==1,"OneLane","NoLane") +EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==2,"TwoLane",EstDataSet$NumLanesCat) +EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==3,"ThreeLane",EstDataSet$NumLanesCat) +EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==4,"FourLanes",EstDataSet$NumLanesCat) +EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes>=5,"FiveLanes+",EstDataSet$NumLanesCat) +#set order of factors +EstDataSet$NumLanesCat<-factor(EstDataSet$NumLanesCat, levels = c("NoLane","OneLane", "TwoLane", "ThreeLane","FourLanes", "FiveLanes+")) + +# for descriptives +EstDataSet$OneLane <- ifelse(EstDataSet$NumLanes==1,1,0) +EstDataSet$TwoLane <- ifelse(EstDataSet$NumLanes==2,1,0) +EstDataSet$ThreeLane <- ifelse(EstDataSet$NumLanes==3,1,0) +EstDataSet$FourLane <- ifelse(EstDataSet$NumLanes==4,1,0) +EstDataSet$FiveMoreLane <- ifelse(EstDataSet$NumLanes>=5,1,0) + +# calculate reference bins (shift variables) - two peaks (AM and PM) and one low (MD) - generic, for all facility types +#Pre-AM: AM Peak to start of day +#Post-AM: AM Peak to MD low +#Pre-PM: PM Peak to MD low (backward) +#Post-PM: PM Peak to end of day +if (TRUE) { + EstDataSet.mean.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") + EstDataSet.mean.mean<-aggregate(EstDataSet$mean,by=list(EstDataSet$todcat),FUN=mean) + + temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=25 & EstDataSet.mean.mean$Group.1<=36) + Peak_AM <- temp[which.max(temp$x),1] #32 + + temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=37 & EstDataSet.mean.mean$Group.1<=62) + Low_MD <- temp[which.min(temp$x),1] #41 + + temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=63 & EstDataSet.mean.mean$Group.1<=76) + Peak_PM <- temp[which.max(temp$x),1] #70 + + # calculate before and after variables for AM and PM periods + EstDataSet$BeforeAM <-ifelse(EstDataSet$todcat<=Peak_AM,Peak_AM-EstDataSet$todcat,99) + EstDataSet$BeforePM <-ifelse(EstDataSet$todcat>=Low_MD & EstDataSet$todcat <= Peak_PM,Peak_PM-EstDataSet$todcat,99) + + EstDataSet$AfterAM <-ifelse(EstDataSet$todcat>=Peak_AM & EstDataSet$todcat=Peak_PM,EstDataSet$todcat-Peak_PM,99) + +} + +EstDataSet$IsBeforeAM<- ifelse(EstDataSet$BeforeAM<99,1,0) +EstDataSet$IsAfterAM<- ifelse(EstDataSet$AfterAM<99,1,0) +EstDataSet$IsBeforePM<- ifelse(EstDataSet$BeforePM<99,1,0) +EstDataSet$IsAfterPM<- ifelse(EstDataSet$AfterPM<99,1,0) + +# apply shift +EstDataSet$IsBeforeAM.Shift<- EstDataSet$IsBeforeAM*EstDataSet$BeforeAM +EstDataSet$IsAfterAM.Shift<- EstDataSet$IsAfterAM*EstDataSet$AfterAM +EstDataSet$IsBeforePM.Shift<- EstDataSet$IsBeforePM*EstDataSet$BeforePM +EstDataSet$IsAfterPM.Shift<- EstDataSet$IsAfterPM*EstDataSet$AfterPM + +# piecewise functions for shift variables +# Before AM: 32 (Peak_AM) to 29, 29 to 26, 26 to 20, 20 to 1 +# After AM: 32 to 36, 36 to 39, 39 to 41 (Low_MD) +# Before PM: 70 (Peak_PM) to 66, 66 to 62, 62 to 58, 58 to 41 (Low_MD) +# After PM: 70 (Peak_PM) to 71, 71 to 79, 79 to 96 + +EstDataSet$BeforeAM.Step1<-ifelse(EstDataSet$IsBeforeAM==1, EstDataSet$BeforeAM,0) +EstDataSet$BeforeAM.Step2<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<29,EstDataSet$BeforeAM-(Peak_AM-29),0) +EstDataSet$BeforeAM.Step3<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<26,EstDataSet$BeforeAM-(Peak_AM-26),0) +EstDataSet$BeforeAM.Step4<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<20,EstDataSet$BeforeAM-(Peak_AM-20),0) + +EstDataSet$AfterAM.Step1<-ifelse(EstDataSet$IsAfterAM==1, EstDataSet$AfterAM,0) +EstDataSet$AfterAM.Step2<-ifelse(EstDataSet$IsAfterAM==1 & EstDataSet$todcat>36,EstDataSet$AfterAM-(36-Peak_AM),0) +EstDataSet$AfterAM.Step3<-ifelse(EstDataSet$IsAfterAM==1 & EstDataSet$todcat>39,EstDataSet$AfterAM-(39-Peak_AM),0) + +EstDataSet$BeforePM.Step1<-ifelse(EstDataSet$IsBeforePM==1, EstDataSet$BeforePM,0) +EstDataSet$BeforePM.Step2<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<66,EstDataSet$BeforePM-(Peak_PM-66),0) +EstDataSet$BeforePM.Step3<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<62,EstDataSet$BeforePM-(Peak_PM-62),0) +EstDataSet$BeforePM.Step4<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<58,EstDataSet$BeforePM-(Peak_PM-58),0) + +EstDataSet$AfterPM.Step1<-ifelse(EstDataSet$IsAfterPM==1, EstDataSet$AfterPM,0) +EstDataSet$AfterPM.Step2<-ifelse(EstDataSet$IsAfterPM==1 & EstDataSet$todcat>71,EstDataSet$AfterPM-(71-Peak_PM),0) +EstDataSet$AfterPM.Step3<-ifelse(EstDataSet$IsAfterPM==1 & EstDataSet$todcat>79,EstDataSet$AfterPM-(79-Peak_PM),0) + +if (TRUE){ + # calculate mean SD in time slices for the four variables + BeforeAM.sd.mean<-cast(EstDataSet,BeforeAM~IFC_Est,mean,value="sd") + BeforePM.sd.mean<-cast(EstDataSet,BeforePM~IFC_Est,mean,value="sd") + AfterAM.sd.mean<-cast(EstDataSet,AfterAM~IFC_Est,mean,value="sd") + AfterPM.sd.mean<-cast(EstDataSet,AfterPM~IFC_Est,mean,value="sd") + EstDataSet.sd.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="sd") + + setnames(BeforeAM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) + setnames(BeforePM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) + setnames(AfterAM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) + setnames(AfterPM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) + setnames(EstDataSet.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) + + if (OUTLIER) { + # outlier method + write.table(BeforeAM.sd.mean,"BeforeAM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(BeforePM.sd.mean,"BeforePM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(AfterAM.sd.mean,"AfterAM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(AfterPM.sd.mean,"AfterPM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(EstDataSet.sd.mean,"EstDataSet_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) + + } else { + # confidence score 10 method + write.table(BeforeAM.sd.mean,"BeforeAM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(BeforePM.sd.mean,"BeforePM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(AfterAM.sd.mean,"AfterAM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(AfterPM.sd.mean,"AfterPM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(EstDataSet.sd.mean,"EstDataSet_SDMean_noCS10.csv",sep = ",",row.names = FALSE) + + } +} + +if (TRUE){ + # calculate mean SD in time slices for the four variables + BeforeAM.mean.mean<-cast(EstDataSet,BeforeAM~IFC_Est,mean,value="mean") + BeforePM.mean.mean<-cast(EstDataSet,BeforePM~IFC_Est,mean,value="mean") + AfterAM.mean.mean<-cast(EstDataSet,AfterAM~IFC_Est,mean,value="mean") + AfterPM.mean.mean<-cast(EstDataSet,AfterPM~IFC_Est,mean,value="mean") + EstDataSet.mean.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") + + setnames(BeforeAM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) + setnames(BeforePM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) + setnames(AfterAM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) + setnames(AfterPM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) + setnames(EstDataSet.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) + + if (OUTLIER) { + # outlier method + write.table(BeforeAM.mean.mean,"BeforeAM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(BeforePM.mean.mean,"BeforePM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(AfterAM.mean.mean,"AfterAM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(AfterPM.mean.mean,"AfterPM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) + write.table(EstDataSet.mean.mean,"EstDataSet_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) + + } else { + # confidence score 10 method + write.table(BeforeAM.mean.mean,"BeforeAM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(BeforePM.mean.mean,"BeforePM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(AfterAM.mean.mean,"AfterAM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(AfterPM.mean.mean,"AfterPM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) + write.table(EstDataSet.mean.mean,"EstDataSet_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) + + } +} + +# InterchangeDistance +#EstDataSet$Upstream <- EstDataSet$upstream.distance +#EstDataSet$Downstream <- EstDataSet$downstream.distance +EstDataSet$MajorUpstream <- EstDataSet$majorupstream.distance +EstDataSet$MajorDownstream <- EstDataSet$majordownstream.distance + +if (FALSE){ + EstDataSet$AllInt.UpDist<-ifelse(EstDataSet$upstream.distance<0.5,"Short","Long") + EstDataSet$AllInt.UpDist<-ifelse(EstDataSet$upstream.distance>=0.5 & EstDataSet$upstream.distance<=2,"Med",EstDataSet$AllInt.UpDist) + EstDataSet$AllInt.DownDist<-ifelse(EstDataSet$downstream.distance<0.5,"Short","Long") + EstDataSet$AllInt.DownDist<-ifelse(EstDataSet$downstream.distance>=0.5 & EstDataSet$downstream.distance<=2,"Med",EstDataSet$AllInt.DownDist) + + EstDataSet$AllInt.UpDist<-factor(EstDataSet$AllInt.UpDist, levels = c("Short","Med", "Long")) + EstDataSet$AllInt.DownDist<-factor(EstDataSet$AllInt.DownDist, levels = c("Short","Med", "Long")) + + EstDataSet$MajInt.UpDist<-ifelse(EstDataSet$majorupstream.distance<0.5,"Short","Long") + EstDataSet$MajInt.UpDist<-ifelse(EstDataSet$majorupstream.distance>=0.5 & EstDataSet$majorupstream.distance<=2,"Med",EstDataSet$MajInt.UpDist) + EstDataSet$MajInt.DownDist<-ifelse(EstDataSet$majordownstream.distance<0.5,"Short","Long") + EstDataSet$MajInt.DownDist<-ifelse(EstDataSet$majordownstream.distance>=0.5 & EstDataSet$majordownstream.distance<=2,"Med",EstDataSet$MajInt.DownDist) + + EstDataSet$MajInt.UpDist<-factor(EstDataSet$MajInt.UpDist, levels = c("Short","Med", "Long")) + EstDataSet$MajInt.DownDist<-factor(EstDataSet$MajInt.DownDist, levels = c("Short","Med", "Long")) +} + +EstDataSet$MajorUpstream.Inverse <- (1/EstDataSet$MajorUpstream) +EstDataSet$MajorDownstream.Inverse <- (1/EstDataSet$MajorDownstream) + +EstDataSet$AuxLanesBinary <- ifelse(EstDataSet$AuxLanes>0,1,0) +EstDataSet$ISPD70 <- ifelse(EstDataSet$ISPD==70,1,0) + +# subset by facility type - four for now (freeways, arterials, ramps, and others) +EstDataSet.freeways <- subset(EstDataSet,IFC_Est==1) +EstDataSet.arterials <- subset(EstDataSet,IFC_Est==2) +EstDataSet.ramps <- subset(EstDataSet,IFC_Est==3) +EstDataSet.others <- subset(EstDataSet,IFC_Est==4) + +# ----------------------------- 1.FREEWAYS ----------------------------- +# remove 1-lane freeways +EstDataSet.freeways <- subset(EstDataSet.freeways,NumLanes>1) + +# add lane change distance (12/7/2015) +LaneChangeDistance <-LaneChangeDistance[,c("LinkID","LaneIncrease","LaneDrop","DeadEnd","RegionEnd","DownstreamDistance")] +temp<-merge(EstDataSet.freeways,LaneChangeDistance,by.x ="ID1",by.y = "LinkID",all.x=TRUE) + +EstDataSet.freeways<-temp +EstDataSet.freeways$Down.Lane.Increase<-0 +EstDataSet.freeways$Down.Lane.Drop<-0 + +EstDataSet.freeways$Down.Lane.Increase<-ifelse(!is.na(EstDataSet.freeways$LaneIncrease) & EstDataSet.freeways$LaneIncrease>0,EstDataSet.freeways$DownstreamDistance,0) +EstDataSet.freeways$Down.Lane.Drop<-ifelse(!is.na(EstDataSet.freeways$LaneDrop) & EstDataSet.freeways$LaneDrop>0,EstDataSet.freeways$DownstreamDistance,0) + +EstDataSet.freeways$Down.Lane.Increase.Inv<-ifelse(!is.na(EstDataSet.freeways$LaneIncrease) & EstDataSet.freeways$LaneIncrease>0,1/EstDataSet.freeways$DownstreamDistance,0) +EstDataSet.freeways$Down.Lane.Drop.Inv<-ifelse(!is.na(EstDataSet.freeways$LaneDrop) & EstDataSet.freeways$LaneDrop>0,1/EstDataSet.freeways$DownstreamDistance,0) + +# add upstream/downstream node ramp meter (12/16/2015) - only within the periods of 7am-9am and 4pm-6pm +FreewayRampMeters <-FreewayRampMeters[,c("LinkID","IsRampMeterUp","IsRampMeterDown")] +temp<-merge(EstDataSet.freeways,FreewayRampMeters,by.x ="ID1",by.y = "LinkID",all.x=TRUE) +EstDataSet.freeways<-temp + +EstDataSet.freeways$DownNode.RampMeter<-0 +EstDataSet.freeways$DownNode.RampMeter<-ifelse(!is.na(EstDataSet.freeways$IsRampMeterDown) & EstDataSet.freeways$IsRampMeterDown==1 + & ((EstDataSet.freeways$todcat>=28 & EstDataSet.freeways$todcat<=36) + | (EstDataSet.freeways$todcat>=64 & EstDataSet.freeways$todcat<=72)),1,0) + +# Regression model +if (FALSE) { + model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up+ISPD70+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3 + +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) +} + +# significant - remove ISPD70 +model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.High.Up+ISPD70+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3 + +AfterPM.Step1+AfterPM.Step3 + +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) + +# no shift vars +model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.High.Up+ISPD70 + +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) + + +summary(model.freeways) + +# ---------------------------- 2.ARTERIALS ----------------------------- +# Arterials - segment speed (12/7/2015) +EstDataSet.arterials$ISPD.Cat<-"" +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD<35,"ISPD35Less",EstDataSet.arterials$ISPD.Cat) +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==35,"ISPD35",EstDataSet.arterials$ISPD.Cat) +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==40,"ISPD40",EstDataSet.arterials$ISPD.Cat) +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==45,"ISPD45",EstDataSet.arterials$ISPD.Cat) +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==50,"ISPD50",EstDataSet.arterials$ISPD.Cat) +EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD>50,"ISPD50More",EstDataSet.arterials$ISPD.Cat) +#set order of factors +EstDataSet.arterials$ISPD.Cat<-factor(EstDataSet.arterials$ISPD.Cat, levels = c("ISPD35Less","ISPD35", "ISPD40", "ISPD45","ISPD50", "ISPD50More")) + +EstDataSet.arterials<-subset(EstDataSet.arterials,!is.na(EstDataSet.arterials$VOC)) +# Regression model +if (FALSE) { + model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up + +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) +} + +model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSF.Low.Up + +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) +# no shift vars +model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSF.Low.Up + +ISPD.Cat + +ICNT.Est, data=EstDataSet.arterials) + +# different measures of capacity +model.arterials = lm(sdpermen~GCRatio+RightLanes+LeftLanes + +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) + +summary(model.arterials) + +# significant - aggregate from lower LOS + +# ---------------------------- 3.RAMPS --------------------------------- +model.ramps = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up + +BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.RampMeter, data=EstDataSet.ramps) + +# only significant - don't include after LOSE.Up +model.ramps = lm(sdpermean~LOSC.Up+LOSE.Up + +BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step4 + +AfterPM.Step1+AfterPM.Step3+ICNT.RampMeter, data=EstDataSet.ramps) + +# no shift vars +model.ramps = lm(sdpermean~LOSC.Up+LOSE.Up + +ICNT.RampMeter, data=EstDataSet.ramps) + + +summary(model.ramps) + +# ---------------------------- 4.OTHERS -------------------------------- +# combine twolane and threelanes for others +EstDataSet.others$NumLanesCat <- ifelse(EstDataSet.others$NumLanes==1,"OneLane","NoLane") +EstDataSet.others$NumLanesCat <- ifelse(EstDataSet.others$NumLanes>=2,"TwoLane+",EstDataSet.others$NumLanesCat) +EstDataSet.others$NumLanesCat<-factor(EstDataSet.others$NumLanesCat, levels = c("NoLane","OneLane", "TwoLane+")) + +EstDataSet.others$OneLane <- ifelse(EstDataSet.others$NumLanes==1,1,0) +EstDataSet.others$TwoLane <- ifelse(EstDataSet.others$NumLanes>=2,1,0) +EstDataSet.others$ThreeLane <- 0 + +# Regression model +model.others = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up + +ISPD+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3, data=EstDataSet.others) + +model.others = lm(sdpermean~ISPD+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 + +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 + +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3, data=EstDataSet.others) + +# no shift vars +model.others = lm(sdpermean~ISPD, data=EstDataSet.others) + +# view results +summary(model.others) + +# ---------------------------------- Correlation Matrix ------------------------------------- +library(Hmisc) +library(corrplot) + +# Freeways +# keep only a few variables +temp1<-EstDataSet.freeways[,c("sd","sdpermean", + "ISPD70","VOC","LOSC.Up","LOSD.Up","LOSE.Up","LOSF.Low.Up","LOSF.Med.Up","LOSF.High.Up", + "BeforeAM.Step1","BeforeAM.Step2","BeforeAM.Step3","BeforeAM.Step4","AfterAM.Step1","AfterAM.Step2","AfterAM.Step3", + "BeforePM.Step1","BeforePM.Step2","BeforePM.Step3","BeforePM.Step4","AfterPM.Step1","AfterPM.Step2","AfterPM.Step3", + "MajorUpstream.Inverse","MajorDownstream.Inverse")] +mcor<-rcorr(as.matrix(temp1)) +# make plot and write the correlations to a csv file +corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) +write.table(mcor$r,"correlation_freeways.csv",row.names = FALSE, sep = ",") + +# Arterials +temp2<-EstDataSet.arterials[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ThreeLane","FourLane", + "AuxLanesBinary","ISPD","VOC","LOSC","LOSD","LOSE", + "LOSF_Low","LOSF_Med","LOSF_High","ICNT.Signal","ICNT.Stop","ICNT.RailRoad", + "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] +mcor<-rcorr(as.matrix(temp2)) +# make plot and write the correlations to a csv file +corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) +write.table(mcor$r,"correlation_arterials.csv",row.names = TRUE, sep = ",") + +# Ramps +temp3<-EstDataSet.ramps[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ThreeLane","ISPD","VOC", + "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] +mcor<-rcorr(as.matrix(temp3)) +# make plot and write the correlations to a csv file +corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) +write.table(mcor$r,"correlation_ramps.csv",row.names = TRUE, sep = ",") + +# Others +temp4<-EstDataSet.others[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ISPD","VOC","LOSC","LOSD","LOSE", + "LOSF_Low","LOSF_Med", + "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] +mcor<-rcorr(as.matrix(temp4)) +# make plot and write the correlations to a csv file +corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) +write.table(mcor$r,"correlation_others.csv",row.names = TRUE, sep = ",") + +# ----------------------- PLOTS ------------ +p1<-plot(EstDataSet.freeways$VOC,EstDataSet.freeways$sdpermean,type="p") +p2<-plot(EstDataSet.arterials$VOC,EstDataSet.arterials$sdpermean,type="p") +p3<-plot(EstDataSet.ramps$VOC,EstDataSet.ramps$sdpermean,type="p") +p4<-plot(EstDataSet.others$VOC,EstDataSet.others$sdpermean,type="p") + +# Shift variable plot +EstDataSet.sdmean<-cast(EstDataSet,todcat~IFC_Est,mean,value="sd") +EstDataSet.meanmean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") +setnames(EstDataSet.sdmean,c("1","2","3","4"),c("Freeways","Arterials","Ramps","Others")) +setnames(EstDataSet.meanmean,c("1","2","3","4"),c("Freeways","Arterials","Ramps","Others")) + +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>0 & EstDataSet.sdmean$todcat<=14,'EV1','') +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>14 & EstDataSet.sdmean$todcat<=24,'EA',EstDataSet.sdmean$tod.model) +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>24 & EstDataSet.sdmean$todcat<=36,'AM',EstDataSet.sdmean$tod.model) +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>36 & EstDataSet.sdmean$todcat<=62,'MD',EstDataSet.sdmean$tod.model) +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>62 & EstDataSet.sdmean$todcat<=76,'PM',EstDataSet.sdmean$tod.model) +EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>76 & EstDataSet.sdmean$todcat<=96,'EV2',EstDataSet.sdmean$tod.model) + +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>0 & EstDataSet.meanmean$todcat<=14,'EV1','') +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>14 & EstDataSet.meanmean$todcat<=24,'EA',EstDataSet.meanmean$tod.model) +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>24 & EstDataSet.meanmean$todcat<=36,'AM',EstDataSet.meanmean$tod.model) +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>36 & EstDataSet.meanmean$todcat<=62,'MD',EstDataSet.meanmean$tod.model) +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>62 & EstDataSet.meanmean$todcat<=76,'PM',EstDataSet.meanmean$tod.model) +EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>76 & EstDataSet.meanmean$todcat<=96,'EV2',EstDataSet.meanmean$tod.model) + +EstDataSet.sdmean$tod.model<-factor(EstDataSet.sdmean$tod.model, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) +EstDataSet.meanmean$tod.model<-factor(EstDataSet.meanmean$tod.model, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) + + +Peak_AM <- 32 +Low_MD <- 60 +Peak_PM <-76 + +EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat<=Peak_AM,1,0) #blue +EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Peak_AM & EstDataSet.sdmean$todcat<=Low_MD,2,EstDataSet.sdmean$shift.period) # darkgreen +EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Low_MD & EstDataSet.sdmean$todcat<=Peak_PM,3,EstDataSet.sdmean$shift.period) # darkorange4 +EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Peak_PM,4,EstDataSet.sdmean$shift.period) #chocolate4 + +EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat<=Peak_AM,1,0) #blue +EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Peak_AM & EstDataSet.meanmean$todcat<=Low_MD,2,EstDataSet.meanmean$shift.period) # darkgreen +EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Low_MD & EstDataSet.meanmean$todcat<=Peak_PM,3,EstDataSet.meanmean$shift.period) # darkorange4 +EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Peak_PM,4,EstDataSet.meanmean$shift.period) #chocolate4 + +#colors +EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat<=Peak_AM,"#0000FF",0) +EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Peak_AM & EstDataSet.sdmean$todcat<=Low_MD,"#006400",EstDataSet.sdmean$shift.period.color) +EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Low_MD & EstDataSet.sdmean$todcat<=Peak_PM,"#BB4500",EstDataSet.sdmean$shift.period.color) +EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Peak_PM,"#BB4513",EstDataSet.sdmean$shift.period.color) + +EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat<=Peak_AM,"#0000FF",0) +EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Peak_AM & EstDataSet.meanmean$todcat<=Low_MD,"#006400",EstDataSet.meanmean$shift.period.color) +EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Low_MD & EstDataSet.meanmean$todcat<=Peak_PM,"#BB4500",EstDataSet.meanmean$shift.period.color) +EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Peak_PM,"#BB4513",EstDataSet.meanmean$shift.period.color) + +#plot +p1<-ggplot(data=EstDataSet.sdmean,aes(x=todcat,y=Freeways)) + geom_point() + geom_line(aes(colour=EstDataSet.sdmean$shift.period.color)) + theme_bw() +p2<-ggplot(data=EstDataSet.meanmean,aes(x=todcat,y=Freeways)) + geom_point() + geom_line(aes(colour=EstDataSet.meanmean$shift.period.color)) + theme_bw() %+replace% theme(panel.background = element_rect(fill = NA)) + +p<-p + facet_grid( . ~ tod.model, scales="free_x", space="free") + +#p<-p + geom_line(colour=EstDataSet.sdmean$shift.period) +#p<-p + scale_colour_manual(breaks=EstDataSet.sdmean$shift.period, values = unique(as.character(EstDataSet.sdmean$shift.period.color))) + +p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), + plot.title = element_text(lineheight=.8, face="bold", vjust=2)) +p<-p+expand_limits(y = 0) + +p<-p+scale_x_continuous(breaks = seq(0,96, by = 1), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) + +p<-p + labs(x="Time of Day Bins",y="Travel Time SD", title="Travel Time Reliability") +p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + diff --git a/sandag_abm/src/main/r/summarize_SR125data.R b/sandag_abm/src/main/r/summarize_SR125data.R new file mode 100644 index 0000000..7629531 --- /dev/null +++ b/sandag_abm/src/main/r/summarize_SR125data.R @@ -0,0 +1,141 @@ +library(stringr) +library(xtable) +library(foreign) +library(data.table) + +setwd("C:/Projects/SANDAG_PricingAndReliability/data/toll facility data") + +tripdata <- read.csv("RequestedNumbers.csv",header=TRUE) +nrow(tripdata) +tripdata<-data.table(tripdata) + +#start time +tripdata[, starthour := as.numeric(substr(descr,1,2))] +tripdata[, startmin := as.numeric(substr(descr,4,5))] +tripdata[,starttime:=as.numeric(starthour*60+startmin)] + +# end time +tripdata[, endhour := as.numeric(substr(descr,9,10))] +tripdata[, endmin := as.numeric(substr(descr,12,13))] +tripdata[,endtime:=as.numeric(endhour*60+endmin)] + +tripdata[,midtime:=(starttime+endtime)/2] + +# time periods (mins from midnight) +#Early AM (EA) 210 - 359 +#AM Peak (AM) 360 - 539 +#Midday (MD) 540 - 929 +#PM Peak (PM) 930 - 1139 +#Evening (EV) 1140 - 1440 and 0 - 209 + +tripdata$tod=5 # 5-evening +tripdata[midtime>=210 & midtime<=359,tod:=1] #1-early AM +tripdata[midtime>=360 & midtime<=539,tod:=2] #2-AM peak +tripdata[midtime>=540 & midtime<=929,tod:=3] #3-Midday +tripdata[midtime>=930 & midtime<=1139,tod:=4] #4-PM Peak + +tripdata$ea<-ifelse(tripdata$tod==1,1,0) +tripdata$am<-ifelse(tripdata$tod==2,1,0) +tripdata$md<-ifelse(tripdata$tod==3,1,0) +tripdata$pm<-ifelse(tripdata$tod==4,1,0) +tripdata$ev<-ifelse(tripdata$tod==5,1,0) + +#ea trips +tripdata[,ea_FT2Axle:=FT2Axle*ea] +tripdata[,ea_CashCC2Axle:=CashCC2Axle*ea] +tripdata[,ea_FT3PlusAxle:=FT3PlusAxle*ea] +tripdata[,ea_CashCC3PlusAxle:=CashCC3PlusAxle*ea] + +#am trips +tripdata[,am_FT2Axle:=FT2Axle*am] +tripdata[,am_CashCC2Axle:=CashCC2Axle*am] +tripdata[,am_FT3PlusAxle:=FT3PlusAxle*am] +tripdata[,am_CashCC3PlusAxle:=CashCC3PlusAxle*am] + +#md trips +tripdata[,md_FT2Axle:=FT2Axle*md] +tripdata[,md_CashCC2Axle:=CashCC2Axle*md] +tripdata[,md_FT3PlusAxle:=FT3PlusAxle*md] +tripdata[,md_CashCC3PlusAxle:=CashCC3PlusAxle*md] + +#pm trips +tripdata[,pm_FT2Axle:=FT2Axle*pm] +tripdata[,pm_CashCC2Axle:=CashCC2Axle*pm] +tripdata[,pm_FT3PlusAxle:=FT3PlusAxle*pm] +tripdata[,pm_CashCC3PlusAxle:=CashCC3PlusAxle*pm] + +#ev trips +tripdata[,ev_FT2Axle:=FT2Axle*ev] +tripdata[,ev_CashCC2Axle:=CashCC2Axle*ev] +tripdata[,ev_FT3PlusAxle:=FT3PlusAxle*ev] +tripdata[,ev_CashCC3PlusAxle:=CashCC3PlusAxle*ev] + +#ea +trips_FT2Axle<-aggregate(x=tripdata$ea_FT2Axle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$ea_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$ea_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$ea_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) + +trips_ea<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) +trips_ea$CashCC2Axle<-trips_CashCC2Axle$x +trips_ea$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_ea$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +#am +trips_FT2Axle<-aggregate(x=tripdata$am_FT2Axle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$am_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$am_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$am_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) + +trips_am<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) +trips_am$CashCC2Axle<-trips_CashCC2Axle$x +trips_am$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_am$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +#md +trips_FT2Axle<-aggregate(x=tripdata$md_FT2Axle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$md_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$md_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$md_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) + +trips_md<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) +trips_md$CashCC2Axle<-trips_CashCC2Axle$x +trips_md$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_md$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +#pm +trips_FT2Axle<-aggregate(x=tripdata$pm_FT2Axle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$pm_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$pm_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$pm_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) + +trips_pm<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) +trips_pm$CashCC2Axle<-trips_CashCC2Axle$x +trips_pm$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_pm$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +#ev +trips_FT2Axle<-aggregate(x=tripdata$ev_FT2Axle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$ev_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$ev_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$ev_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) + +trips_ev<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) +trips_ev$CashCC2Axle<-trips_CashCC2Axle$x +trips_ev$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_ev$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +#all +trips_FT2Axle<-aggregate(x=tripdata$FT2Axle,by=list(tod=tripdata$tod),FUN=sum) +trips_CashCC2Axle<-aggregate(x=tripdata$CashCC2Axle,by=list(tod=tripdata$tod),FUN=sum) +trips_FT3PlusAxle<-aggregate(x=tripdata$FT3PlusAxle,by=list(tod=tripdata$tod),FUN=sum) +trips_CashCC3PlusAxle<-aggregate(x=tripdata$CashCC3PlusAxle,by=list(tod=tripdata$tod),FUN=sum) + +trips_all<-data.frame(tod=trips_FT2Axle$tod,FT2Axle=trips_FT2Axle$x) +trips_all$CashCC2Axle<-trips_CashCC2Axle$x +trips_all$FT3PlusAxle<-trips_FT3PlusAxle$x +trips_all$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x + +write.table(trips_all,"sr125_summary.csv",row.names=F,quote=F,sep = ",") + +rm(tripdata) \ No newline at end of file diff --git a/sandag_abm/src/main/r/utilfunc.R b/sandag_abm/src/main/r/utilfunc.R new file mode 100644 index 0000000..1aead08 --- /dev/null +++ b/sandag_abm/src/main/r/utilfunc.R @@ -0,0 +1,256 @@ + +#read a file, save to Rdata, and remove from workspace +readSaveRdata <- function(filename,objname) { + assign(objname,fread(filename)) + save(list=objname,file=paste0(filename,".Rdata")) + rm(list=objname) +} + +#load an RData object to an object name +assignLoad <- function(filename) { + load(filename) + get(ls()[ls() != "filename"]) +} + +regressionmodel<-function(myDF) { + +model = lm(SpeedStdDev~NumLanes+VOC+CongSpeed+Ratio.FFTime.CongTime+IFC, data=myDF) +summary(model) + +} + +detectoutliers<-function(myDF) { + # Detects outliers in a dataset + # + # Args: + # myDF: dataframe in list of dataframes + # type: box plot to use - original or adjusted + # + # Returns: + # mydata with outlier=1 for outliers + type<-"adjusted" + + if (type=="original") { + values<-boxplot(travel_time_sec ~ todcat, myDF, main = "Original Boxplot") + } else { + values<-adjbox(travel_time_sec ~ todcat, myDF, main = "Adjusted Boxplot") + } + + if (length(table(myDF$todcat))<48) { + warning(paste("tmc segment ", myDF$tmc_code[1], " is missing some data")) + } + + for (cat in 1:length(unique(myDF$todcat))) { + limit_lower<-values[["stats"]][1,cat] + limit_upper<-values[["stats"]][5,cat] + myDF[todcat==cat & (travel_time_seclimit_upper), outlier:=1] + } + +} + +myplot_sd<-function(myDF,datatype,numoutliers,field) { + # Plots data points by facility type + # + # Args: + # myDF: dataframe in list of dataframes + # outputsDir: directory to save the plot + # + # Returns: + # Nothing. saves a plot in JPEG format in OutputsDir + + # to change the order of variables in plot + outputsDir="./INRIX/2012_10/" + + # include only segments that have data for the entire period + if (length(table(myDF$todcat))==48) { + + myDF$tod<-factor(myDF$tod, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) + IFC<-myDF$IFC[1] + + #p<-ggplot(data=myDF,aes(todcat,value, colour=tmc_code),na.rm=TRUE) + geom_point() + #p<-ggplot(data=myDF,aes(todcat,value),na.rm=TRUE) + geom_point() + + p<-ggplot(data=myDF,aes(todcat,value, group=tmc_code),na.rm=TRUE) + geom_line()+ geom_point() + + # add grids for TOD + p<-p + facet_grid(. ~ tod, scales="free_x", space="free") + + # set space between facets + p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), + plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + # set consistent axis and force axis to start at 0 + p<-p+expand_limits(y = 0) + p<-p+scale_x_continuous(breaks = seq(0,48, by = 1), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) + + # add title + if (datatype=="raw") { + outfile = paste(field, "_SD_rawdata_",IFC,".jpeg") + maintitle<-paste(field," SD by TOD (raw data)") + } else { + outfile = paste("SD_per_mile_nooutliers_",IFC,".jpeg") + maintitle<-paste(field, " SD by TOD (", numoutliers, " outliers removed)") + } + + p<-p + labs(x="Time of Day Category (30 mins interval)",y=paste(field, " SD"), title=maintitle) + p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + #print(p) + + return(p) + + # save as JPEG + #dev.copy(jpeg,filename=paste(outputsDir,outfile, sep = ""),width=1280, height=1280) + #dev.off() + + } + +} + +myplot1<-function(myDF) { + # Plots data points by facility type + # + # Args: + # myDF: dataframe in list of dataframes + # outputsDir: directory to save the plot + # + # Returns: + # Nothing. saves a plot in JPEG format in OutputsDir + + # to change the order of variables in plot + outputsDir="./INRIX/2012_10/" + myDF$tod<-factor(myDF$tod, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) + IFC<-myDF$IFC[1] + + p<-ggplot(data=myDF,aes(totalhour,speed, colour=color.names)) + geom_point() + p<-p + scale_colour_manual(breaks=myDF$color.names, values = unique(as.character(myDF$color.codes))) + + # add grids for TOD + p<-p + facet_grid(. ~ tod, scales="free_x", space="free") + + # set space between facets + p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), + plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + # set consistent axis and force axis to start at 0 + p<-p+expand_limits(y = 0) + p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) + + numoutliers<-sum(myDF$outlier) + # add title + maintitle<-paste("Speed by TOD (outliers=",numoutliers,")") + p<-p + labs(x="Time of Day (hour)",y="Speed (mph)", title=maintitle) + p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + print(p) + # save as JPEG + dev.copy(jpeg,filename=paste(outputsDir,"Speed_IFC_", IFC, "_outliers30mins.jpeg", sep = ""),width=1280, height=1280) + dev.off() + +} + +myplot<- function(indata,datatype,analysistype,xlbl,ylbl,plottitle,ymax) { + # Makes a scatter plot + # + # Args: + # indata: dataset with datapoints + # datatype: all weekdays or one + # analysistype: travel time or speed analysis + # xlbl: x-axis label + # ylbl: y-axis label + # plottitle: plot title + # ymax: y-axis limit + # + # Returns: + # a scatter plot of datapoints in indata + + # setup a plot + if (analysistype == "Time") { + p<-ggplot(data=indata,aes(x=totalhour,y=travel_time_sec)) + geom_point(colour='black',size=2) + #ymax<-max(indata$travel_time_sec)+100 # add 100sec to have some space on top + ybreak <-200 # 200 seconds + } else { + p<-ggplot(data=indata,aes(x=totalhour,y=speed)) + geom_point(colour='black',size=2) + #ymax<-max(indata$speed)+10 # add 10 mph to have some space on top + ybreak <-20 # 20 mph + } + + # set different colors for outliers + p<-p + scale_colour_hue(breaks=indata$outlier) + + # add grids for TOD + if (datatype=="All") { + p<-p + facet_grid(wday ~ tod, scales="free_x", space="free") + } else { + p<-p + facet_grid(day ~ tod, scales="free_x", space="free") + } + + # set space between facets + p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), + plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + # set consistent axis and force axis to start at 0 + p<-p+expand_limits(y = 0) + + # modify x and y axis + if (analysistype == "Time") { + p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) #breaks=seq(0,1000, by = ybreak), + } else { + p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(breaks=seq(0,100, by = 20), expand = c(0, 0)) #breaks=seq(0,100, by = ybreak), + } + + # add title + p<-p + labs(x=xlbl,y=ylbl, title=plottitle) + p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) + + return(p) +} + +# function to add multiple plots in a print area +multiplot <- function(..., plotlist=NULL, file, cols=1, layout=NULL) { + # Combines multiple plots in one print area + # + # Args: + # ...: plots seperated by commas. Provide as many plots as you want. + # plotlist: + # file: + # cols: number of columns in the plot area + # layout: if NULL then 'cols' is used to determine layout + # + # Returns: + # None. Displays the multiplot + + library(grid) + + # Make a list from the ... arguments and plotlist + plots <- c(list(...), plotlist) + + numPlots = length(plots) + + # If layout is NULL, then use 'cols' to determine layout + if (is.null(layout)) { + # Make the panel + # ncol: Number of columns of plots + # nrow: Number of rows needed, calculated from # of cols + layout <- matrix(seq(1, cols * ceiling(numPlots/cols)), + ncol = cols, nrow = ceiling(numPlots/cols)) + } + + if (numPlots==1) { + print(plots[[1]]) + + } else { + # Set up the page + grid.newpage() + pushViewport(viewport(layout = grid.layout(nrow(layout), ncol(layout)))) + + # Make each plot, in the correct location + for (i in 1:numPlots) { + # Get the i,j matrix positions of the regions that contain this subplot + matchidx <- as.data.frame(which(layout == i, arr.ind = TRUE)) + + print(plots[[i]], vp = viewport(layout.pos.row = matchidx$row, + layout.pos.col = matchidx$col)) + } + } +} \ No newline at end of file diff --git a/sandag_abm/src/main/r/visualizer/Master.R b/sandag_abm/src/main/r/visualizer/Master.R new file mode 100644 index 0000000..bfc4337 --- /dev/null +++ b/sandag_abm/src/main/r/visualizer/Master.R @@ -0,0 +1,94 @@ +############################################################################################################################# +# Master script to render final HTML file from R Markdown file +# Loads all required packages from the dependencies folder +# +# Make sure the 'plyr' is not loaded after 'dplyr' library in the same R session +# Under such case, the group_by features of dplyr library does not work. Restart RStudio and make sure +# plyr library is not loaded while generating dashboard +# For more info on this issue: +# https://stackoverflow.com/questions/26923862/why-are-my-dplyr-group-by-summarize-not-working-properly-name-collision-with +# +############################################################################################################################# + +##### LIST OF ALL INPUT FILES ##### +## 0. Path input data : parameters.csv +## 1. Base scenario summary file names : summaryFilesNames_survey.csv +## 2. Build scenario summary file names : summaryFilesNames.csv +## 3. Model area shapefile : summaryFilesNames.csv +## 4. All REF and ABM summary output files + +### Read Command Line Arguments +args <- commandArgs(trailingOnly = TRUE) +Parameters_File <- args[1] +showWarnings=FALSE + +### Read parameters from Parameters_File +parameters <- read.csv(Parameters_File, header = TRUE) +WORKING_DIR <- trimws(paste(parameters$Value[parameters$Key=="WORKING_DIR"])) +BASE_SUMMARY_DIR <- trimws(paste(parameters$Value[parameters$Key=="BASE_SUMMARY_DIR"])) +BUILD_SUMMARY_DIR <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SUMMARY_DIR"])) +BASE_SCENARIO_NAME <- trimws(paste(parameters$Value[parameters$Key=="BASE_SCENARIO_NAME"])) +BUILD_SCENARIO_NAME <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SCENARIO_NAME"])) +BASE_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BASE_SAMPLE_RATE"]))) +BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BUILD_SAMPLE_RATE"]))) +R_LIBRARY <- trimws(paste(parameters$Value[parameters$Key=="R_LIBRARY"])) +OUTPUT_HTML_NAME <- trimws(paste(parameters$Value[parameters$Key=="OUTPUT_HTML_NAME"])) +SHP_FILE_NAME <- trimws(paste(parameters$Value[parameters$Key=="SHP_FILE_NAME"])) +IS_BASE_SURVEY <- trimws(paste(parameters$Value[parameters$Key=="IS_BASE_SURVEY"])) + +### Initialization +# Load global variables +.libPaths(R_LIBRARY) +source(paste(WORKING_DIR, "scripts/_SYSTEM_VARIABLES.R", sep = "/")) + +###create directories +dir.create(BASE_DATA_PATH) +dir.create(BUILD_DATA_PATH) + +### Copy summary CSVs +base_CSV_list <- ifelse(IS_BASE_SURVEY=="Yes", "summaryFilesNames_survey.csv", "summaryFilesNames.csv") +summaryFileList_base <- read.csv(paste(SYSTEM_TEMPLATES_PATH, base_CSV_list, sep = '/'), as.is = T) +summaryFileList_base <- as.list(summaryFileList_base$summaryFile) +retVal <- copyFile(summaryFileList_base, sourceDir = BASE_SUMMARY_DIR, targetDir = BASE_DATA_PATH) +if(retVal) q(save = "no", status = 11) +summaryFileList_build <- read.csv(paste(SYSTEM_TEMPLATES_PATH, "summaryFilesNames.csv", sep = '/'), as.is = T) +summaryFileList_build <- as.list(summaryFileList_build$summaryFile) +retVal <- copyFile(summaryFileList_build, sourceDir = BUILD_SUMMARY_DIR, targetDir = BUILD_DATA_PATH) +if(retVal) q(save = "no", status = 11) + +### Load required libraries +SYSTEM_REPORT_PKGS <- c("DT", "flexdashboard", "leaflet", "geojsonio", "htmltools", "htmlwidgets", "kableExtra", + "knitr", "mapview", "plotly", "RColorBrewer", "rgdal", "rgeos", "crosstalk","treemap", "htmlTable", + "rmarkdown", "scales", "stringr", "jsonlite", "pander", "ggplot2", "reshape", "raster", "dplyr") + +lapply(SYSTEM_REPORT_PKGS, library, character.only = TRUE) + +### Read Target and Output SUmmary files +currDir <- getwd() +setwd(BASE_DATA_PATH) +base_csv = list.files(pattern="*.csv") +base_data <- lapply(base_csv, read.csv) +base_csv_names <- unlist(lapply(base_csv, function (x) {gsub(".csv", "", x)})) + +setwd(BUILD_DATA_PATH) +build_csv = list.files(pattern="*.csv") +build_data <- lapply(build_csv, read.csv) +build_csv_names <- unlist(lapply(build_csv, function (x) {gsub(".csv", "", x)})) + +## Read SHP file +setwd(SYSTEM_SHP_PATH) +zone_shp <- shapefile(SHP_FILE_NAME) +zone_shp <- spTransform(zone_shp, CRS("+proj=longlat +ellps=GRS80")) + +setwd(currDir) + +### Generate dashboard +rmarkdown::render(file.path(SYSTEM_TEMPLATES_PATH, "template.Rmd"), + output_dir = RUNTIME_PATH, + intermediates_dir = RUNTIME_PATH, quiet = TRUE) +template.html <- readLines(file.path(RUNTIME_PATH, "template.html")) +idx <- which(template.html == "window.FlexDashboardComponents = [];")[1] +template.html <- append(template.html, "L_PREFER_CANVAS = true;", after = idx) +writeLines(template.html, file.path(OUTPUT_PATH, paste(OUTPUT_HTML_NAME, ".html", sep = ""))) + +# finish \ No newline at end of file diff --git a/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R b/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R new file mode 100644 index 0000000..fb6eb50 --- /dev/null +++ b/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R @@ -0,0 +1,2448 @@ +####################################################### +### Script for summarizing SANDAG ABM Output +### Author: Binny M Paul, binny.paul@rsginc.com, Oct 2017 +### Edited: Khademul Haque, khademul.haque@rsginc.com, Mar 2019 +####################################################### + +##### LIST OF ALL INPUT FILES ##### +## 0. Path input data : summ_inputs_abm.csv +## 1. household data : householdData_3.csv +## 2. person data : personData_3.csv +## 3. Individual tour data : indivTourData_3.csv +## 4. Individual trip data : indivTripData_3.csv +## 5. Joint tour data : jointTripData_3.csv +## 6. Joint trip data : jointTourData_3.csv +## 7. Work school location data : wsLocResults_3.csv +## 8. Auto ownership data : aoResults.csv +## 9. Auto ownership data : aoResults_Pre.csv +## 10. Geographic crosswalk data : geographicXwalk_PMSA.csv +## 11. Distance skim : traffic_skims_MD.omx -> MD_SOVTOLLH_DIST + +start_time <- Sys.time() + +library(plyr) +library(weights) +library(reshape) +library(data.table) +library(omxr) +showWarnings=FALSE + +# Read Command Line Arguments +args <- commandArgs(trailingOnly = TRUE) +inputs_File <- args[1] + +inputs <- read.csv(inputs_File, header = TRUE) +WD <- trimws(paste(inputs$Value[inputs$Key=="WD"])) +ABMOutputDir <- trimws(paste(inputs$Value[inputs$Key=="ABMOutputDir"])) +geogXWalkDir <- trimws(paste(inputs$Value[inputs$Key=="geogXWalkDir"])) +SkimDir <- trimws(paste(inputs$Value[inputs$Key=="SkimDir"])) +MAX_ITER <- trimws(paste(inputs$Value[inputs$Key=="MAX_ITER"])) + +setwd(ABMOutputDir) +#full model run +hh <- fread(paste("householdData_",MAX_ITER,".csv", sep = "")) +per <- fread(paste("personData_",MAX_ITER,".csv", sep = "")) +tours <- fread(paste("indivTourData_",MAX_ITER,".csv", sep = "")) +trips <- fread(paste("indivTripData_",MAX_ITER,".csv", sep = "")) +jtrips <- fread(paste("jointTripData_",MAX_ITER,".csv", sep = "")) +unique_joint_tours <- fread(paste("jointTourData_",MAX_ITER,".csv", sep = "")) +wsLoc <- fread(paste("wsLocResults_",MAX_ITER,".csv", sep = "")) +aoResults <- fread("aoResults.csv") +aoResults_Pre <- fread("aoResults_Pre.csv") + +visitor_trips <- fread(paste("visitorTrips.csv", sep = "")) + +mazCorrespondence <- fread(paste(geogXWalkDir, "geographicXwalk_PMSA.csv", sep = "/"), stringsAsFactors = F) +districtList <- sort(unique(mazCorrespondence$pmsa)) + +SkimFile <- paste(SkimDir, "traffic_skims_MD.omx", sep = "/") +DST_SKM <- read_omx(SkimFile, "MD_SOV_TR_H_DIST") +skimLookUp <- read_lookup(SkimFile, "zone_number") + +pertypeCodes <- data.frame(code = c(1,2,3,4,5,6,7,8,"All"), + name = c("FT Worker", "PT Worker", "Univ Stud", "Non Worker", "Retiree", "Driv Stud", "NonDriv Stud", "Pre-School", "All")) + +#------------------------------------------- +# Prepare files for computing summary statistics +dir.create(WD, showWarnings = FALSE) +setwd(WD) + +aoResults$HHVEH[aoResults$AO == 0] <- 0 +aoResults$HHVEH[aoResults$AO == 1] <- 1 +aoResults$HHVEH[aoResults$AO == 2] <- 2 +aoResults$HHVEH[aoResults$AO == 3] <- 3 +aoResults$HHVEH[aoResults$AO >= 4] <- 4 + +aoResults_Pre$HHVEH[aoResults_Pre$AO == 0] <- 0 +aoResults_Pre$HHVEH[aoResults_Pre$AO == 1] <- 1 +aoResults_Pre$HHVEH[aoResults_Pre$AO == 2] <- 2 +aoResults_Pre$HHVEH[aoResults_Pre$AO == 3] <- 3 +aoResults_Pre$HHVEH[aoResults_Pre$AO >= 4] <- 4 + +hh$HHVEH[hh$autos == 0] <- 0 +hh$HHVEH[hh$autos == 1] <- 1 +hh$HHVEH[hh$autos == 2] <- 2 +hh$HHVEH[hh$autos == 3] <- 3 +hh$HHVEH[hh$autos >= 4] <- 4 + +hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 0] <- 1 +hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 0] <- 2 +hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 1] <- 3 +hh$VEH_NEWCAT[(hh$HVs == 2) & (hh$AVs) == 0] <- 4 +hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 2] <- 5 +hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 1] <- 6 +hh$VEH_NEWCAT[(hh$HVs == 3) & (hh$AVs) == 0] <- 7 +hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 3] <- 8 +hh$VEH_NEWCAT[(hh$HVs == 2) & (hh$AVs) == 1] <- 9 +hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 2] <- 10 +hh$VEH_NEWCAT[(hh$HVs == 4) & (hh$AVs) == 0] <- 11 + +#HH Size +hhsize <- count(per, c("hh_id"), "hh_id>0") +hh$HHSIZ <- hhsize$freq[match(hh$hh_id, hhsize$hh_id)] +hh$HHSIZE[hh$HHSIZ == 1] <- 1 +hh$HHSIZE[hh$HHSIZ == 2] <- 2 +hh$HHSIZE[hh$HHSIZ == 3] <- 3 +hh$HHSIZE[hh$HHSIZ == 4] <- 4 +hh$HHSIZE[hh$HHSIZ >= 5] <- 5 + +#Adults in the HH +adults <- count(per, c("hh_id"), "age>=18 & age<99") +hh$ADULTS <- adults$freq[match(hh$hh_id, adults$hh_id)] + +per$PERTYPE[per$type=="Full-time worker"] <- 1 +per$PERTYPE[per$type=="Part-time worker"] <- 2 +per$PERTYPE[per$type=="University student"] <- 3 +per$PERTYPE[per$type=="Non-worker"] <- 4 +per$PERTYPE[per$type=="Retired"] <- 5 +per$PERTYPE[per$type=="Student of driving age"] <- 6 +per$PERTYPE[per$type=="Student of non-driving age"] <- 7 +per$PERTYPE[per$type=="Child too young for school"] <- 8 + +# Districts are Pseudo MSA +wsLoc$HDISTRICT <- mazCorrespondence$pmsa[match(wsLoc$HomeMGRA, mazCorrespondence$mgra)] +wsLoc$WDISTRICT <- mazCorrespondence$pmsa[match(wsLoc$WorkLocation, mazCorrespondence$mgra)] + +# Get home, work and school location TAZs +wsLoc$HHTAZ <- mazCorrespondence$taz[match(wsLoc$HomeMGRA, mazCorrespondence$mgra)] +wsLoc$WTAZ <- mazCorrespondence$taz[match(wsLoc$WorkLocation, mazCorrespondence$mgra)] +wsLoc$STAZ <- mazCorrespondence$taz[match(wsLoc$SchoolLocation, mazCorrespondence$mgra)] + +wsLoc$oindex<-match(wsLoc$HHTAZ, skimLookUp$Lookup) +wsLoc$dindex<-match(wsLoc$WTAZ, skimLookUp$Lookup) +wsLoc$dindex2<-match(wsLoc$STAZ, skimLookUp$Lookup) +wsLoc$WorkLocationDistance<-DST_SKM[cbind(wsLoc$oindex, wsLoc$dindex)] +wsLoc$WorkLocationDistance[is.na(wsLoc$WorkLocationDistance)] <- 0 + +wsLoc$SchoolLocationDistance<-DST_SKM[cbind(wsLoc$oindex, wsLoc$dindex2)] +wsLoc$SchoolLocationDistance[is.na(wsLoc$SchoolLocationDistance)] <- 0 + +#--------Compute Summary Statistics------- +#***************************************** + +# Auto ownership +autoOwnership_Pre <- count(aoResults_Pre, c("HHVEH")) +write.csv(autoOwnership_Pre, "autoOwnership_Pre.csv", row.names = TRUE) + +autoOwnership <- count(aoResults, c("HHVEH")) +write.csv(autoOwnership, "autoOwnership.csv", row.names = TRUE) + +autoOwnership_AV <- count(hh, c("AVs")) +write.csv(autoOwnership_AV, "autoOwnership_AV.csv", row.names = TRUE) + +autoOwnership_new <- count(hh, c("VEH_NEWCAT")) +write.csv(autoOwnership_new, "autoOwnership_new.csv", row.names = TRUE) + +# Zero auto HHs by TAZ +hh$HHTAZ <- mazCorrespondence$taz[match(hh$home_mgra, mazCorrespondence$mgra)] +hh$ZeroAutoWgt[hh$HHVEH==0] <- 1 +hh$ZeroAutoWgt[is.na(hh$ZeroAutoWgt)] <- 0 +zeroAutoByTaz <- aggregate(hh$ZeroAutoWgt, list(TAZ = hh$HHTAZ), sum) +write.csv(zeroAutoByTaz, "zeroAutoByTaz.csv", row.names = TRUE) + +# Persons by person type +pertypeDistbn <- count(per, c("PERTYPE")) +write.csv(pertypeDistbn, "pertypeDistbn.csv", row.names = TRUE) + +# Telecommute Freuency +teleCommute <- count(per, c("tele_choice")) +write.csv(teleCommute, "teleCommute_frequency.csv", row.names = TRUE) + +# HH Transponder Ownership +transponder <- count(hh, c("transponder")) +write.csv(transponder, "transponder_ownership.csv", row.names = TRUE) + +# Micro-mobility +micro_r1 <- count(trips, c('micro_walkMode')) +micro_r2 <- count(trips, c('micro_trnAcc')) +micro_r3 <- count(trips, c('micro_trnEgr')) +colnames(micro_r1) <- c("micro_mode","trips") +colnames(micro_r2) <- c("micro_mode","trips") +colnames(micro_r3) <- c("micro_mode","trips") + +micro_v1 <- count(visitor_trips, c('micro_walkMode')) +micro_v2 <- count(visitor_trips, c('micro_trnAcc')) +micro_v3 <- count(visitor_trips, c('micro_trnEgr')) +colnames(micro_v1) <- c("micro_mode","trips") +colnames(micro_v2) <- c("micro_mode","trips") +colnames(micro_v3) <- c("micro_mode","trips") + +micromobility <- rbind(micro_r1,micro_r2,micro_r3,micro_v1,micro_v2,micro_v3) +micromobility_summary <- aggregate(trips ~ micro_mode, data=micromobility, FUN = sum) +write.csv(micromobility_summary, "micormobility.csv", row.names = TRUE) + +# Mandatory DC +workers <- wsLoc[wsLoc$WorkLocation > 0 & wsLoc$WorkLocation != 99999,] +students <- wsLoc[wsLoc$SchoolLocation > 0 & wsLoc$SchoolLocation != 88888,] + +# code distance bins +workers$distbin <- cut(workers$WorkLocationDistance, breaks = c(seq(0,50, by=1), 9999), labels = F, right = F) +students$distbin <- cut(students$SchoolLocationDistance, breaks = c(seq(0,50, by=1), 9999), labels = F, right = F) + +distBinCat <- data.frame(distbin = seq(1,51, by=1)) +districtList_df <- data.frame(id = districtList) + +# compute TLFDs by district and total +tlfd_work <- ddply(workers[,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, work = sum(HDISTRICT>0)) +tlfd_work <- cast(tlfd_work, distbin~HDISTRICT, value = "work", sum) +work_ditbins <- tlfd_work$distbin +tlfd_work <- transpose(tlfd_work[,!colnames(tlfd_work) %in% c("distbin")]) +tlfd_work$id <- row.names(tlfd_work) +tlfd_work <- merge(x = districtList_df, y = tlfd_work, by = "id", all.x = TRUE) +tlfd_work[is.na(tlfd_work)] <- 0 +tlfd_work <- transpose(tlfd_work[,!colnames(tlfd_work) %in% c("id")]) +tlfd_work <- cbind(data.frame(distbin = work_ditbins), tlfd_work) +tlfd_work$Total <- rowSums(tlfd_work[,!colnames(tlfd_work) %in% c("distbin")]) +names(tlfd_work) <- sub("V", "District_", names(tlfd_work)) +tlfd_work_df <- merge(x = distBinCat, y = tlfd_work, by = "distbin", all.x = TRUE) +tlfd_work_df[is.na(tlfd_work_df)] <- 0 + +tlfd_univ <- ddply(students[students$PersonType==3,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, univ = sum(HDISTRICT>0)) +tlfd_univ <- cast(tlfd_univ, distbin~HDISTRICT, value = "univ", sum) +univ_ditbins <- tlfd_univ$distbin +tlfd_univ <- transpose(tlfd_univ[,!colnames(tlfd_univ) %in% c("distbin")]) +tlfd_univ$id <- row.names(tlfd_univ) +tlfd_univ <- merge(x = districtList_df, y = tlfd_univ, by = "id", all.x = TRUE) +tlfd_univ[is.na(tlfd_univ)] <- 0 +tlfd_univ <- transpose(tlfd_univ[,!colnames(tlfd_univ) %in% c("id")]) +tlfd_univ <- cbind(data.frame(distbin = univ_ditbins), tlfd_univ) +tlfd_univ$Total <- rowSums(tlfd_univ[,!colnames(tlfd_univ) %in% c("distbin")]) +names(tlfd_univ) <- sub("V", "District_", names(tlfd_univ)) +tlfd_univ_df <- merge(x = distBinCat, y = tlfd_univ, by = "distbin", all.x = TRUE) +tlfd_univ_df[is.na(tlfd_univ_df)] <- 0 + +tlfd_schl <- ddply(students[students$PersonType>=6,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, schl = sum(HDISTRICT>0)) +tlfd_schl <- cast(tlfd_schl, distbin~HDISTRICT, value = "schl", sum) +schl_ditbins <- tlfd_schl$distbin +tlfd_schl <- transpose(tlfd_schl[,!colnames(tlfd_schl) %in% c("distbin")]) +tlfd_schl$id <- row.names(tlfd_schl) +tlfd_schl <- merge(x = districtList_df, y = tlfd_schl, by = "id", all.x = TRUE) +tlfd_schl[is.na(tlfd_schl)] <- 0 +tlfd_schl <- transpose(tlfd_schl[,!colnames(tlfd_schl) %in% c("id")]) +tlfd_schl <- cbind(data.frame(distbin = schl_ditbins), tlfd_schl) +tlfd_schl$Total <- rowSums(tlfd_schl[,!colnames(tlfd_schl) %in% c("distbin")]) +names(tlfd_schl) <- sub("V", "District_", names(tlfd_schl)) +tlfd_schl_df <- merge(x = distBinCat, y = tlfd_schl, by = "distbin", all.x = TRUE) +tlfd_schl_df[is.na(tlfd_schl_df)] <- 0 + +write.csv(tlfd_work_df, "workTLFD.csv", row.names = F) +write.csv(tlfd_univ_df, "univTLFD.csv", row.names = F) +write.csv(tlfd_schl_df, "schlTLFD.csv", row.names = F) + +cat("\n Average distance to workplace (Total): ", mean(workers$WorkLocationDistance, na.rm = TRUE)) +cat("\n Average distance to university (Total): ", mean(students$SchoolLocationDistance[students$PersonType==3], na.rm = TRUE)) +cat("\n Average distance to school (Total): ", mean(students$SchoolLocationDistance[students$PersonType>=6], na.rm = TRUE)) + +## Output avg trip lengths for visualizer +workTripLengths <- ddply(workers[,c("HDISTRICT", "WorkLocationDistance")], c("HDISTRICT"), summarise, work = mean(WorkLocationDistance)) +totalLength <- data.frame("Total", mean(workers$WorkLocationDistance)) +colnames(totalLength) <- colnames(workTripLengths) +workTripLengths <- rbind(workTripLengths, totalLength) + +univTripLengths <- ddply(students[students$PersonType==3,c("HDISTRICT", "SchoolLocationDistance")], c("HDISTRICT"), summarise, univ = mean(SchoolLocationDistance)) +totalLength <- data.frame("Total", mean(students$SchoolLocationDistance[students$PersonType==3])) +colnames(totalLength) <- colnames(univTripLengths) +univTripLengths <- rbind(univTripLengths, totalLength) + +schlTripLengths <- ddply(students[students$PersonType>=6,c("HDISTRICT", "SchoolLocationDistance")], c("HDISTRICT"), summarise, schl = mean(SchoolLocationDistance)) +totalLength <- data.frame("Total", mean(students$SchoolLocationDistance[students$PersonType>=6])) +colnames(totalLength) <- colnames(schlTripLengths) +schlTripLengths <- rbind(schlTripLengths, totalLength) + +mandTripLengths <- cbind(workTripLengths, univTripLengths$univ, schlTripLengths$schl) +colnames(mandTripLengths) <- c("District", "Work", "Univ", "Schl") +write.csv(mandTripLengths, "mandTripLengths.csv", row.names = F) + +# Work from home [for each district and total] +districtWorkers <- ddply(wsLoc[wsLoc$WorkLocation > 0,c("HDISTRICT")], c("HDISTRICT"), summarise, workers = sum(HDISTRICT>0)) +districtWfh <- ddply(wsLoc[wsLoc$WorkLocation==99999,c("HDISTRICT", "WorkLocation")], c("HDISTRICT"), summarise, wfh = sum(HDISTRICT>0)) +wfh_summary <- cbind(districtWorkers, districtWfh$wfh) +colnames(wfh_summary) <- c("District", "Workers", "WFH") +totalwfh <- data.frame("Total", sum(wsLoc$WorkLocation>0), sum(wsLoc$WorkLocation==99999)) +colnames(totalwfh) <- colnames(wfh_summary) +wfh_summary <- rbind(wfh_summary, totalwfh) +write.csv(wfh_summary, "wfh_summary.csv", row.names = F) +write.csv(totalwfh, "wfh_summary_region.csv", row.names = F) + +# County-County Flows +countyFlows <- xtabs(~HDISTRICT+WDISTRICT, data = workers) +countyFlows[is.na(countyFlows)] <- 0 +countyFlows <- addmargins(as.table(countyFlows)) +countyFlows <- as.data.frame.matrix(countyFlows) +colnames(countyFlows)[colnames(countyFlows)=="Sum"] <- "Total" +colnames(countyFlows) <- paste("District", colnames(countyFlows), sep = "_") +rownames(countyFlows)[rownames(countyFlows)=="Sum"] <- "Total" +rownames(countyFlows) <- paste("District", rownames(countyFlows), sep = "_") +write.csv(countyFlows, "countyFlows.csv", row.names = T) + +# Process Tour file +#------------------ +tours$PERTYPE <- tours$person_type +tours$DISTMILE <- tours$tour_distance +tours$HHVEH <- hh$HHVEH[match(tours$hh_id, hh$hh_id)] +tours$ADULTS <- hh$ADULTS[match(tours$hh_id, hh$hh_id)] +tours$AUTOSUFF[tours$HHVEH == 0] <- 0 +tours$AUTOSUFF[tours$HHVEH < tours$ADULTS & tours$HHVEH > 0] <- 1 +tours$AUTOSUFF[tours$HHVEH >= tours$ADULTS & tours$HHVEH > 0] <- 2 + +tours$num_tot_stops <- tours$num_ob_stops + tours$num_ib_stops + +tours$OTAZ <- mazCorrespondence$taz[match(tours$orig_mgra, mazCorrespondence$mgra)] +tours$DTAZ <- mazCorrespondence$taz[match(tours$dest_mgra, mazCorrespondence$mgra)] + +tours$oindex<-match(tours$OTAZ, skimLookUp$Lookup) +tours$dindex<-match(tours$DTAZ, skimLookUp$Lookup) +tours$SKIMDIST<-DST_SKM[cbind(tours$oindex, tours$dindex)] + + +unique_joint_tours$HHVEH <- hh$HHVEH[match(unique_joint_tours$hh_id, hh$hh_id)] +unique_joint_tours$ADULTS <- hh$ADULTS[match(unique_joint_tours$hh_id, hh$hh_id)] +unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH == 0] <- 0 +unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH < unique_joint_tours$ADULTS & unique_joint_tours$HHVEH > 0] <- 1 +unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH >= unique_joint_tours$ADULTS] <- 2 + +#Code tour purposes +tours$TOURPURP[tours$tour_purpose=="Work"] <- 1 +tours$TOURPURP[tours$tour_purpose=="University"] <- 2 +tours$TOURPURP[tours$tour_purpose=="School"] <- 3 +tours$TOURPURP[tours$tour_purpose=="Escort"] <- 4 +tours$TOURPURP[tours$tour_purpose=="Shop"] <- 5 +tours$TOURPURP[tours$tour_purpose=="Maintenance"] <- 6 +tours$TOURPURP[tours$tour_purpose=="Eating Out"] <- 7 +tours$TOURPURP[tours$tour_purpose=="Visiting"] <- 8 +tours$TOURPURP[tours$tour_purpose=="Discretionary"] <- 9 +tours$TOURPURP[tours$tour_purpose=="Work-Based"] <- 10 + +#[0:Mandatory, 1: Indi Non Mand, 2: At Work] +tours$TOURCAT[tours$tour_purpose=="Work"] <- 0 +tours$TOURCAT[tours$tour_purpose=="University"] <- 0 +tours$TOURCAT[tours$tour_purpose=="School"] <- 0 +tours$TOURCAT[tours$tour_purpose=="Escort"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Shop"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Maintenance"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Eating Out"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Visiting"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Discretionary"] <- 1 +tours$TOURCAT[tours$tour_purpose=="Work-Based"] <- 2 + +#compute duration +tours$tourdur <- tours$end_period - tours$start_period + 1 #[to match survey] + +tours$TOURMODE <- tours$tour_mode +#tours$TOURMODE[tours$tour_mode==1] <- 1 +#tours$TOURMODE[tours$tour_mode==2] <- 2 +#tours$TOURMODE[tours$tour_mode==3] <- 3 +#tours$TOURMODE[tours$tour_mode>=7 & tours$tour_mode<=13] <- tours$tour_mode[tours$tour_mode>=7 & tours$tour_mode<=13]-3 +#tours$TOURMODE[tours$tour_mode>=14 & tours$tour_mode<=15] <- 11 +#tours$TOURMODE[tours$tour_mode==16] <- 12 + +# exclude school escorting stop from ride sharing mandatory tours + +unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Shop'] <- 5 +unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Maintenance'] <- 6 +unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Eating Out'] <- 7 +unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Visiting'] <- 8 +unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Discretionary'] <- 9 + +unique_joint_tours$NUMBER_HH <- as.integer((nchar(as.character(unique_joint_tours$tour_participants))+1)/2) + +# get participant IDs +unique_joint_tours$PER1[unique_joint_tours$NUMBER_HH>=1] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=1]), 1, 1) +unique_joint_tours$PER2[unique_joint_tours$NUMBER_HH>=2] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=2]), 3, 3) +unique_joint_tours$PER3[unique_joint_tours$NUMBER_HH>=3] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=3]), 5, 5) +unique_joint_tours$PER4[unique_joint_tours$NUMBER_HH>=4] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=4]), 7, 7) +unique_joint_tours$PER5[unique_joint_tours$NUMBER_HH>=5] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=5]), 9, 9) +unique_joint_tours$PER6[unique_joint_tours$NUMBER_HH>=6] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=6]), 11, 11) +unique_joint_tours$PER7[unique_joint_tours$NUMBER_HH>=7] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=7]), 13, 13) +unique_joint_tours$PER8[unique_joint_tours$NUMBER_HH>=8] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=8]), 15, 15) + +unique_joint_tours[is.na(unique_joint_tours)] <- 0 + +# get person type for each participant +unique_joint_tours$PTYPE1 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER1, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE2 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER2, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE3 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER3, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE4 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER4, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE5 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER5, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE6 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER6, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE7 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER7, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] +unique_joint_tours$PTYPE8 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER8, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] + +unique_joint_tours[is.na(unique_joint_tours)] <- 0 + +unique_joint_tours$num_tot_stops <- unique_joint_tours$num_ob_stops + unique_joint_tours$num_ib_stops + +unique_joint_tours$OTAZ <- mazCorrespondence$taz[match(unique_joint_tours$orig_mgra, mazCorrespondence$mgra)] +unique_joint_tours$DTAZ <- mazCorrespondence$taz[match(unique_joint_tours$dest_mgra, mazCorrespondence$mgra)] + +#compute duration +unique_joint_tours$tourdur <- unique_joint_tours$end_period - unique_joint_tours$start_period + 1 #[to match survye] + +unique_joint_tours$TOURMODE <- unique_joint_tours$tour_mode +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode<=2] <- 1 +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=3 & unique_joint_tours$tour_mode<=4] <- 2 +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=5 & unique_joint_tours$tour_mode<=6] <- 3 +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=7 & unique_joint_tours$tour_mode<=13] <- unique_joint_tours$tour_mode[unique_joint_tours$tour_mode>=7 & unique_joint_tours$tour_mode<=13]-3 +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=14 & unique_joint_tours$tour_mode<=15] <- 11 +#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode==16] <- 12 + +# ---- +# this part is added by nagendra.dhakar@rsginc.com from binny.paul@rsginc.com soabm summaries + +# create a combined temp tour file for creating stop freq model summary +temp_tour1 <- tours[,c("TOURPURP","num_ob_stops","num_ib_stops")] +temp_tour2 <- unique_joint_tours[,c("JOINT_PURP","num_ob_stops","num_ib_stops")] +colnames(temp_tour2) <- colnames(temp_tour1) +temp_tour <- rbind(temp_tour1,temp_tour2) + +# code stop frequency model alternatives +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==0] <- 1 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==1] <- 2 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==2] <- 3 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops>=3] <- 4 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==0] <- 5 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==1] <- 6 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==2] <- 7 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops>=3] <- 8 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==0] <- 9 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==1] <- 10 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==2] <- 11 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops>=3] <- 12 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==0] <- 13 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==1] <- 14 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==2] <- 15 +temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops>=3] <- 16 +temp_tour$STOP_FREQ_ALT[is.na(temp_tour$STOP_FREQ_ALT)] <- 0 + +stopFreqModel_summary <- xtabs(~STOP_FREQ_ALT+TOURPURP, data = temp_tour[temp_tour$TOURPURP<=10,]) +write.csv(stopFreqModel_summary, "stopFreqModel_summary.csv", row.names = T) + +# ------ + +# Process Trip file +#------------------ +trips$TOURMODE <- trips$tour_mode +trips$TRIPMODE <- trips$trip_mode + +#trips$TOURMODE[trips$tour_mode<=2] <- 1 +#trips$TOURMODE[trips$tour_mode>=3 & trips$tour_mode<=4] <- 2 +#trips$TOURMODE[trips$tour_mode>=5 & trips$tour_mode<=6] <- 3 +#trips$TOURMODE[trips$tour_mode>=7 & trips$tour_mode<=13] <- trips$tour_mode[trips$tour_mode>=7 & trips$tour_mode<=13]-3 +#trips$TOURMODE[trips$tour_mode>=14 & trips$tour_mode<=15] <- 11 +#trips$TOURMODE[trips$tour_mode==16] <- 12 +# +#trips$TRIPMODE[trips$trip_mode<=2] <- 1 +#trips$TRIPMODE[trips$trip_mode>=3 & trips$trip_mode<=4] <- 2 +#trips$TRIPMODE[trips$trip_mode>=5 & trips$trip_mode<=6] <- 3 +#trips$TRIPMODE[trips$trip_mode>=7 & trips$trip_mode<=13] <- trips$trip_mode[trips$trip_mode>=7 & trips$trip_mode<=13]-3 +#trips$TRIPMODE[trips$trip_mode>=14 & trips$trip_mode<=15] <- 11 +#trips$TRIPMODE[trips$trip_mode==16] <- 12 + +#Code tour purposes +trips$TOURPURP[trips$tour_purpose=="Home"] <- 0 +trips$TOURPURP[trips$tour_purpose=="Work"] <- 1 +trips$TOURPURP[trips$tour_purpose=="University"] <- 2 +trips$TOURPURP[trips$tour_purpose=="School"] <- 3 +trips$TOURPURP[trips$tour_purpose=="Escort"] <- 4 +trips$TOURPURP[trips$tour_purpose=="Shop"] <- 5 +trips$TOURPURP[trips$tour_purpose=="Maintenance"] <- 6 +trips$TOURPURP[trips$tour_purpose=="Eating Out"] <- 7 +trips$TOURPURP[trips$tour_purpose=="Visiting"] <- 8 +trips$TOURPURP[trips$tour_purpose=="Discretionary"] <- 9 +trips$TOURPURP[trips$tour_purpose=="Work-Based" | trips$tour_purpose=="work related"] <- 10 + +trips$OPURP[trips$orig_purpose=="Home"] <- 0 +trips$OPURP[trips$orig_purpose=="Work"] <- 1 +trips$OPURP[trips$orig_purpose=="University"] <- 2 +trips$OPURP[trips$orig_purpose=="School"] <- 3 +trips$OPURP[trips$orig_purpose=="Escort"] <- 4 +trips$OPURP[trips$orig_purpose=="Shop"] <- 5 +trips$OPURP[trips$orig_purpose=="Maintenance"] <- 6 +trips$OPURP[trips$orig_purpose=="Eating Out"] <- 7 +trips$OPURP[trips$orig_purpose=="Visiting"] <- 8 +trips$OPURP[trips$orig_purpose=="Discretionary"] <- 9 +trips$OPURP[trips$orig_purpose=="Work-Based" | trips$orig_purpose=="work related"] <- 10 + +trips$DPURP[trips$dest_purpose=="Home"] <- 0 +trips$DPURP[trips$dest_purpose=="Work"] <- 1 +trips$DPURP[trips$dest_purpose=="University"] <- 2 +trips$DPURP[trips$dest_purpose=="School"] <- 3 +trips$DPURP[trips$dest_purpose=="Escort"] <- 4 +trips$DPURP[trips$dest_purpose=="Shop"] <- 5 +trips$DPURP[trips$dest_purpose=="Maintenance"] <- 6 +trips$DPURP[trips$dest_purpose=="Eating Out"] <- 7 +trips$DPURP[trips$dest_purpose=="Visiting"] <- 8 +trips$DPURP[trips$dest_purpose=="Discretionary"] <- 9 +trips$DPURP[trips$dest_purpose=="Work-Based" | trips$dest_purpose=="work related"] <- 10 + +#[0:Mandatory, 1: Indi Non Mand, 3: At Work] +trips$TOURCAT[trips$tour_purpose=="Work"] <- 0 +trips$TOURCAT[trips$tour_purpose=="University"] <- 0 +trips$TOURCAT[trips$tour_purpose=="School"] <- 0 +trips$TOURCAT[trips$tour_purpose=="Escort"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Shop"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Maintenance"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Eating Out"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Visiting"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Discretionary"] <- 1 +trips$TOURCAT[trips$tour_purpose=="Work-Based"] <- 2 + +#Mark stops and get other attributes +nr <- nrow(trips) +trips$inb_next <- 0 +trips$inb_next[1:nr-1] <- trips$inbound[2:nr] +trips$stops[trips$DPURP>0 & ((trips$inbound==0 & trips$inb_next==0) | (trips$inbound==1 & trips$inb_next==1))] <- 1 +trips$stops[is.na(trips$stops)] <- 0 + +trips$OTAZ <- mazCorrespondence$taz[match(trips$orig_mgra, mazCorrespondence$mgra)] +trips$DTAZ <- mazCorrespondence$taz[match(trips$dest_mgra, mazCorrespondence$mgra)] + +trips$TOUROTAZ <- tours$OTAZ[match(trips$hh_id*1000+trips$person_num*100+trips$TOURCAT*10+trips$tour_id, + tours$hh_id*1000+tours$person_num*100+tours$TOURCAT*10+tours$tour_id)] +trips$TOURDTAZ <- tours$DTAZ[match(trips$hh_id*1000+trips$person_num*100+trips$TOURCAT*10+trips$tour_id, + tours$hh_id*1000+tours$person_num*100+tours$TOURCAT*10+tours$tour_id)] + +# trips$od_dist <- DST_SKM$dist[match(paste(trips$OTAZ, trips$DTAZ, sep = "-"), paste(DST_SKM$o, DST_SKM$d, sep = "-"))] +trips$oindex<-match(trips$OTAZ, skimLookUp$Lookup) +trips$dindex<-match(trips$DTAZ, skimLookUp$Lookup) +trips$od_dist<-DST_SKM[cbind(trips$oindex, trips$dindex)] + +#create stops table +stops <- trips[trips$stops==1,] + +stops$finaldestTAZ[stops$inbound==0] <- stops$TOURDTAZ[stops$inbound==0] +stops$finaldestTAZ[stops$inbound==1] <- stops$TOUROTAZ[stops$inbound==1] + +stops$oindex<-match(stops$OTAZ, skimLookUp$Lookup) +stops$dindex<-match(stops$finaldestTAZ, skimLookUp$Lookup) +stops$od_dist <- DST_SKM[cbind(stops$oindex, stops$dindex)] + +stops$oindex2<-match(stops$OTAZ, skimLookUp$Lookup) +stops$dindex2<-match(stops$DTAZ, skimLookUp$Lookup) +stops$os_dist <- DST_SKM[cbind(stops$oindex2, stops$dindex2)] + +stops$oindex3<-match(stops$DTAZ, skimLookUp$Lookup) +stops$dindex3<-match(stops$finaldestTAZ, skimLookUp$Lookup) +stops$sd_dist <- DST_SKM[cbind(stops$oindex3, stops$dindex3)] + +stops$out_dir_dist <- stops$os_dist + stops$sd_dist - stops$od_dist + +#joint trip + +jtrips$TOURMODE <- jtrips$tour_mode +#jtrips$TOURMODE[jtrips$tour_mode<=2] <- 1 +#jtrips$TOURMODE[jtrips$tour_mode>=3 & jtrips$tour_mode<=4] <- 2 +#jtrips$TOURMODE[jtrips$tour_mode>=5 & jtrips$tour_mode<=6] <- 3 +#jtrips$TOURMODE[jtrips$tour_mode>=7 & jtrips$tour_mode<=13] <- jtrips$tour_mode[jtrips$tour_mode>=7 & jtrips$tour_mode<=13]-3 +#jtrips$TOURMODE[jtrips$tour_mode>=14 & jtrips$tour_mode<=15] <- 11 +#jtrips$TOURMODE[jtrips$tour_mode==16] <- 12 + +jtrips$TRIPMODE <- jtrips$trip_mode +#jtrips$TRIPMODE[jtrips$trip_mode<=2] <- 1 +#jtrips$TRIPMODE[jtrips$trip_mode>=3 & jtrips$trip_mode<=4] <- 2 +#jtrips$TRIPMODE[jtrips$trip_mode>=5 & jtrips$trip_mode<=6] <- 3 +#jtrips$TRIPMODE[jtrips$trip_mode>=7 & jtrips$trip_mode<=13] <- jtrips$trip_mode[jtrips$trip_mode>=7 & jtrips$trip_mode<=13]-3 +#jtrips$TRIPMODE[jtrips$trip_mode>=14 & jtrips$trip_mode<=15] <- 11 +#jtrips$TRIPMODE[jtrips$trip_mode==16] <- 12 + +#Code joint tour purposes +jtrips$TOURPURP[jtrips$tour_purpose=="Work"] <- 1 +jtrips$TOURPURP[jtrips$tour_purpose=="University"] <- 2 +jtrips$TOURPURP[jtrips$tour_purpose=="School"] <- 3 +jtrips$TOURPURP[jtrips$tour_purpose=="Escort"] <- 4 +jtrips$TOURPURP[jtrips$tour_purpose=="Shop"] <- 5 +jtrips$TOURPURP[jtrips$tour_purpose=="Maintenance"] <- 6 +jtrips$TOURPURP[jtrips$tour_purpose=="Eating Out"] <- 7 +jtrips$TOURPURP[jtrips$tour_purpose=="Visiting"] <- 8 +jtrips$TOURPURP[jtrips$tour_purpose=="Discretionary"] <- 9 +jtrips$TOURPURP[jtrips$tour_purpose=="Work-Based" | jtrips$tour_purpose=="work related"] <- 10 + +jtrips$OPURP[jtrips$orig_purpose=="Home"] <- 0 +jtrips$OPURP[jtrips$orig_purpose=="Work"] <- 1 +jtrips$OPURP[jtrips$orig_purpose=="University"] <- 2 +jtrips$OPURP[jtrips$orig_purpose=="School"] <- 3 +jtrips$OPURP[jtrips$orig_purpose=="Escort"] <- 4 +jtrips$OPURP[jtrips$orig_purpose=="Shop"] <- 5 +jtrips$OPURP[jtrips$orig_purpose=="Maintenance"] <- 6 +jtrips$OPURP[jtrips$orig_purpose=="Eating Out"] <- 7 +jtrips$OPURP[jtrips$orig_purpose=="Visiting"] <- 8 +jtrips$OPURP[jtrips$orig_purpose=="Discretionary"] <- 9 +jtrips$OPURP[jtrips$orig_purpose=="Work-Based" | jtrips$orig_purpose=="work related"] <- 10 + +jtrips$DPURP[jtrips$dest_purpose=="Home"] <- 0 +jtrips$DPURP[jtrips$dest_purpose=="Work"] <- 1 +jtrips$DPURP[jtrips$dest_purpose=="University"] <- 2 +jtrips$DPURP[jtrips$dest_purpose=="School"] <- 3 +jtrips$DPURP[jtrips$dest_purpose=="Escort"] <- 4 +jtrips$DPURP[jtrips$dest_purpose=="Shop"] <- 5 +jtrips$DPURP[jtrips$dest_purpose=="Maintenance"] <- 6 +jtrips$DPURP[jtrips$dest_purpose=="Eating Out"] <- 7 +jtrips$DPURP[jtrips$dest_purpose=="Visiting"] <- 8 +jtrips$DPURP[jtrips$dest_purpose=="Discretionary"] <- 9 +jtrips$DPURP[jtrips$dest_purpose=="Work-Based" | jtrips$dest_purpose=="work related"] <- 10 + +#[0:Mandatory, 1: Indi Non Mand, 3: At Work] +jtrips$TOURCAT[jtrips$tour_purpose=="Work"] <- 0 +jtrips$TOURCAT[jtrips$tour_purpose=="University"] <- 0 +jtrips$TOURCAT[jtrips$tour_purpose=="School"] <- 0 +jtrips$TOURCAT[jtrips$tour_purpose=="Escort"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Shop"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Maintenance"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Eating Out"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Visiting"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Discretionary"] <- 1 +jtrips$TOURCAT[jtrips$tour_purpose=="Work-Based"] <- 2 + +#Mark stops and get other attributes +nr <- nrow(jtrips) +jtrips$inb_next <- 0 +jtrips$inb_next[1:nr-1] <- jtrips$inbound[2:nr] +jtrips$stops[jtrips$DPURP>0 & ((jtrips$inbound==0 & jtrips$inb_next==0) | (jtrips$inbound==1 & jtrips$inb_next==1))] <- 1 +jtrips$stops[is.na(jtrips$stops)] <- 0 + +jtrips$OTAZ <- mazCorrespondence$taz[match(jtrips$orig_mgra, mazCorrespondence$mgra)] +jtrips$DTAZ <- mazCorrespondence$taz[match(jtrips$dest_mgra, mazCorrespondence$mgra)] + +jtrips$TOUROTAZ <- unique_joint_tours$OTAZ[match(jtrips$hh_id*1000+jtrips$tour_id, + unique_joint_tours$hh_id*1000+unique_joint_tours$tour_id)] +jtrips$TOURDTAZ <- unique_joint_tours$DTAZ[match(jtrips$hh_id*1000+jtrips$tour_id, + unique_joint_tours$hh_id*1000+unique_joint_tours$tour_id)] + +#create stops table +jstops <- jtrips[jtrips$stops==1,] + +jstops$finaldestTAZ[jstops$inbound==0] <- jstops$TOURDTAZ[jstops$inbound==0] +jstops$finaldestTAZ[jstops$inbound==1] <- jstops$TOUROTAZ[jstops$inbound==1] + +jstops$oindex<-match(jstops$OTAZ, skimLookUp$Lookup) +jstops$dindex<-match(jstops$finaldestTAZ, skimLookUp$Lookup) +jstops$od_dist <- DST_SKM[cbind(jstops$oindex, jstops$dindex)] + +jstops$oindex2<-match(jstops$OTAZ, skimLookUp$Lookup) +jstops$dindex2<-match(jstops$DTAZ, skimLookUp$Lookup) +jstops$os_dist <- DST_SKM[cbind(jstops$oindex2, jstops$dindex2)] + +jstops$oindex3<-match(jstops$DTAZ, skimLookUp$Lookup) +jstops$dindex3<-match(jstops$finaldestTAZ, skimLookUp$Lookup) +jstops$sd_dist <- DST_SKM[cbind(jstops$oindex3, jstops$dindex3)] + +jstops$out_dir_dist <- jstops$os_dist + jstops$sd_dist - jstops$od_dist + +#--------------------------------------------------------------------------- + +# Recode workrelated tours which are not at work subtour as work tour +#tours$TOURPURP[tours$TOURPURP == 10 & tours$IS_SUBTOUR == 0] <- 1 + +workCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 1") #[excluding at work subtours] +schlCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 2 | TOURPURP == 3") +inmCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP>=4 & TOURPURP<=9") + +# ----------------------- +# added for calibration by nagendra.dhakar@rsginc.com +# for indivudal NM tour generation +workCounts_temp <- workCounts +schlCounts_temp <- schlCounts +inmCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP>4 & TOURPURP<=9") #excluding school escort tours +atWorkCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP == 10") +escortCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP==4") + + +colnames(workCounts_temp)[3] <- "freq_work" +colnames(schlCounts_temp)[3] <- "freq_schl" +colnames(inmCounts_temp)[3] <- "freq_inm" +colnames(atWorkCounts_temp)[3] <- "freq_atwork" +colnames(escortCounts_temp)[3] <- "freq_escort" + +temp <- merge(workCounts_temp, schlCounts_temp, by = c("hh_id", "person_num")) +temp1 <- merge(temp, inmCounts_temp, by = c("hh_id", "person_num")) +temp1$freq_m <- temp1$freq_work + temp1$freq_schl +temp1$freq_itours <- temp1$freq_m+temp1$freq_inm + +#joint tours +#identify persons that made joint tour +#temp_joint <- melt(unique_joint_tours[,c("hh_id","tour_id" ,"PER1", "PER2", "PER3", "PER4", "PER5", "PER6", "PER7", "PER8")], id = c("hh_id","tour_id")) +temp_joint <- melt(unique_joint_tours[,c("hh_id","tour_id" ,"PER1", "PER2", "PER3", "PER4", "PER5", "PER6", "PER7")], id = c("hh_id","tour_id")) +colnames(temp_joint) <- c("hh_id", "tour_id", "var", "person_num") +temp_joint <- as.data.frame(temp_joint) +temp_joint$person_num <- as.integer(temp_joint$person_num) +temp_joint$joint<- 0 +temp_joint$joint[temp_joint$person_num>0] <- 1 + +temp_joint <- temp_joint[temp_joint$joint==1,] +person_unique_joint <- aggregate(joint~hh_id+person_num, temp_joint, sum) + +temp2 <- merge(temp1, person_unique_joint, by = c("hh_id", "person_num"), all = T) +temp2 <- merge(temp2, atWorkCounts_temp, by = c("hh_id", "person_num"), all = T) +temp2 <- merge(temp2, escortCounts_temp, by = c("hh_id", "person_num"), all = T) +temp2[is.na(temp2)] <- 0 + +#add number of joint tours to non-mandatory +temp2$freq_nm <- temp2$freq_inm + temp2$joint + +#get person type +temp2$PERTYPE <- per$PERTYPE[match(temp2$hh_id*10+temp2$person_num,per$hh_id*10+per$person_num)] + +#total tours +temp2$total_tours <- temp2$freq_nm+temp2$freq_m+temp2$freq_atwork+temp2$freq_escort + +persons_mand <- temp2[temp2$freq_m>0,] #persons with atleast 1 mandatory tours +persons_nomand <- temp2[temp2$freq_m==0,] #active persons with 0 mandatory tours + +freq_nmtours_mand <- count(persons_mand, c("PERTYPE","freq_nm")) +freq_nmtours_nomand <- count(persons_nomand, c("PERTYPE","freq_nm")) +test <- count(temp2, c("PERTYPE","freq_inm","freq_m","freq_nm","freq_atwork","freq_escort")) +write.csv(test, "tour_rate_debug.csv", row.names = F) +write.csv(temp2,"temp2.csv", row.names = F) + +write.table("Non-Mandatory Tours for Persons with at-least 1 Mandatory Tour", "indivNMTourFreq.csv", sep = ",", row.names = F, append = F) +write.table(freq_nmtours_mand, "indivNMTourFreq.csv", sep = ",", row.names = F, append = T) +write.table("Non-Mandatory Tours for Active Persons with 0 Mandatory Tour", "indivNMTourFreq.csv", sep = ",", row.names = F, append = T) +write.table(freq_nmtours_nomand, "indivNMTourFreq.csv", sep = ",", row.names = F, append = TRUE) + +# end of addition for calibration +# ----------------------- + + +# ---------------------- +# added for calibration by nagendra.dhakar@rsginc.com + +i4tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 4") +i5tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 5") +i6tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 6") +i7tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 7") +i8tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 8") +i9tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 9") + +# end of addition for calibration +# ----------------------- + +tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP <= 9") #number of tours per person [excluding at work subtours] +joint5 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==5") +joint6 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==6") +joint7 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==7") +joint8 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==8") +joint9 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==9") + +hh$joint5 <- joint5$freq[match(hh$hh_id, joint5$hh_id)] +hh$joint6 <- joint6$freq[match(hh$hh_id, joint6$hh_id)] +hh$joint7 <- joint7$freq[match(hh$hh_id, joint7$hh_id)] +hh$joint8 <- joint8$freq[match(hh$hh_id, joint8$hh_id)] +hh$joint9 <- joint9$freq[match(hh$hh_id, joint9$hh_id)] +hh$jtours <- hh$joint5+hh$joint6+hh$joint7+hh$joint8+hh$joint9 + +hh$joint5[is.na(hh$joint5)] <- 0 +hh$joint6[is.na(hh$joint6)] <- 0 +hh$joint7[is.na(hh$joint7)] <- 0 +hh$joint8[is.na(hh$joint8)] <- 0 +hh$joint9[is.na(hh$joint9)] <- 0 +hh$jtours[is.na(hh$jtours)] <- 0 + +#joint tour indicator +hh$JOINT <- 0 +hh$JOINT[substr(hh$cdap_pattern, nchar(as.character(hh$cdap_pattern)), nchar(as.character(hh$cdap_pattern)))=="j"] <- 1 + +# code JTF category +hh$jtf[hh$jtours==0] <- 1 +hh$jtf[hh$joint5==1] <- 2 +hh$jtf[hh$joint6==1] <- 3 +hh$jtf[hh$joint7==1] <- 4 +hh$jtf[hh$joint8==1] <- 5 +hh$jtf[hh$joint9==1] <- 6 + +hh$jtf[hh$joint5>=2] <- 7 +hh$jtf[hh$joint6>=2] <- 8 +hh$jtf[hh$joint7>=2] <- 9 +hh$jtf[hh$joint8>=2] <- 10 +hh$jtf[hh$joint9>=2] <- 11 + +hh$jtf[hh$joint5>=1 & hh$joint6>=1] <- 12 +hh$jtf[hh$joint5>=1 & hh$joint7>=1] <- 13 +hh$jtf[hh$joint5>=1 & hh$joint8>=1] <- 14 +hh$jtf[hh$joint5>=1 & hh$joint9>=1] <- 15 + +hh$jtf[hh$joint6>=1 & hh$joint7>=1] <- 16 +hh$jtf[hh$joint6>=1 & hh$joint8>=1] <- 17 +hh$jtf[hh$joint6>=1 & hh$joint9>=1] <- 18 + +hh$jtf[hh$joint7>=1 & hh$joint8>=1] <- 19 +hh$jtf[hh$joint7>=1 & hh$joint9>=1] <- 20 + +hh$jtf[hh$joint8>=1 & hh$joint9>=1] <- 21 + +per$workTours <- workCounts$freq[match(per$hh_id*10+per$person_num, workCounts$hh_id*10+workCounts$person_num)] +per$schlTours <- schlCounts$freq[match(per$hh_id*10+per$person_num, schlCounts$hh_id*10+schlCounts$person_num)] +per$inmTours <- inmCounts$freq[match(per$hh_id*10+per$person_num, inmCounts$hh_id*10+inmCounts$person_num)] +per$inmTours[is.na(per$inmTours)] <- 0 +per$numTours <- tourCounts$freq[match(per$hh_id*10+per$person_num, tourCounts$hh_id*10+tourCounts$person_num)] +per$numTours[is.na(per$numTours)] <- 0 + +# --------------------------------------------------- +# added for calibration by nagendra.dhakar@rsginc.com + +per$i4numTours <- i4tourCounts$freq[match(per$hh_id*10+per$person_num, i4tourCounts$hh_id*10+i4tourCounts$person_num)] +per$i4numTours[is.na(per$i4numTours)] <- 0 +per$i5numTours <- i5tourCounts$freq[match(per$hh_id*10+per$person_num, i5tourCounts$hh_id*10+i5tourCounts$person_num)] +per$i5numTours[is.na(per$i5numTours)] <- 0 +per$i6numTours <- i6tourCounts$freq[match(per$hh_id*10+per$person_num, i6tourCounts$hh_id*10+i6tourCounts$person_num)] +per$i6numTours[is.na(per$i6numTours)] <- 0 +per$i7numTours <- i7tourCounts$freq[match(per$hh_id*10+per$person_num, i7tourCounts$hh_id*10+i7tourCounts$person_num)] +per$i7numTours[is.na(per$i7numTours)] <- 0 +per$i8numTours <- i8tourCounts$freq[match(per$hh_id*10+per$person_num, i8tourCounts$hh_id*10+i8tourCounts$person_num)] +per$i8numTours[is.na(per$i8numTours)] <- 0 +per$i9numTours <- i9tourCounts$freq[match(per$hh_id*10+per$person_num, i9tourCounts$hh_id*10+i9tourCounts$person_num)] +per$i9numTours[is.na(per$i9numTours)] <- 0 + +# end of addition for calibration +# --------------------------------------------------- + +# Total tours by person type +per$numTours[is.na(per$numTours)] <- 0 +toursPertypeDistbn <- count(tours[tours$PERTYPE>0 & tours$TOURPURP!=10,], c("PERTYPE")) +write.csv(toursPertypeDistbn, "toursPertypeDistbn.csv", row.names = TRUE) + +# count joint tour fr each person type +temp_joint <- melt(unique_joint_tours[, c("hh_id","tour_id","PTYPE1","PTYPE2","PTYPE3","PTYPE4","PTYPE5","PTYPE6","PTYPE7","PTYPE8")], id = c("hh_id", "tour_id")) +names(temp_joint)[names(temp_joint)=="value"] <- "PERTYPE" +jtoursPertypeDistbn <- count(temp_joint[temp_joint$PERTYPE>0,], c("PERTYPE")) + +# Total tours by person type for visualizer +totaltoursPertypeDistbn <- toursPertypeDistbn +totaltoursPertypeDistbn$freq <- totaltoursPertypeDistbn$freq + jtoursPertypeDistbn$freq +write.csv(totaltoursPertypeDistbn, "total_tours_by_pertype_vis.csv", row.names = F) + +# Total indi NM tours by person type and purpose +tours_pertype_purpose <- count(tours[tours$TOURPURP>=4 & tours$TOURPURP<=9,], c("PERTYPE", "TOURPURP")) +write.csv(tours_pertype_purpose, "tours_pertype_purpose.csv", row.names = TRUE) + +# --------------------------------------------------- +# added for calibration by nagendra.dhakar@rsginc.com + +# code indi NM tour category +per$i4numTours[per$i4numTours>=2] <- 2 +per$i5numTours[per$i5numTours>=2] <- 2 +per$i6numTours[per$i6numTours>=2] <- 2 +per$i7numTours[per$i7numTours>=1] <- 1 +per$i8numTours[per$i8numTours>=1] <- 1 +per$i9numTours[per$i9numTours>=2] <- 2 + +tours_pertype_esco <- count(per, c("PERTYPE", "i4numTours")) +tours_pertype_shop <- count(per, c("PERTYPE", "i5numTours")) +tours_pertype_main <- count(per, c("PERTYPE", "i6numTours")) +tours_pertype_eati <- count(per, c("PERTYPE", "i7numTours")) +tours_pertype_visi <- count(per, c("PERTYPE", "i8numTours")) +tours_pertype_disc <- count(per, c("PERTYPE", "i9numTours")) + + +colnames(tours_pertype_esco) <- c("PERTYPE","inumTours","freq") +colnames(tours_pertype_shop) <- c("PERTYPE","inumTours","freq") +colnames(tours_pertype_main) <- c("PERTYPE","inumTours","freq") +colnames(tours_pertype_eati) <- c("PERTYPE","inumTours","freq") +colnames(tours_pertype_visi) <- c("PERTYPE","inumTours","freq") +colnames(tours_pertype_disc) <- c("PERTYPE","inumTours","freq") + +tours_pertype_esco$purpose <- 4 +tours_pertype_shop$purpose <- 5 +tours_pertype_main$purpose <- 6 +tours_pertype_eati$purpose <- 7 +tours_pertype_visi$purpose <- 8 +tours_pertype_disc$purpose <- 9 + +indi_nm_tours_pertype <- rbind(tours_pertype_esco,tours_pertype_shop,tours_pertype_main,tours_pertype_eati,tours_pertype_visi,tours_pertype_disc) +write.csv(indi_nm_tours_pertype, "inmtours_pertype_purpose.csv", row.names = F) + +# end of addition for calibration +# --------------------------------------------------- + +tours_pertype_purpose <- xtabs(freq~PERTYPE+TOURPURP, tours_pertype_purpose) +tours_pertype_purpose[is.na(tours_pertype_purpose)] <- 0 +tours_pertype_purpose <- addmargins(as.table(tours_pertype_purpose)) +tours_pertype_purpose <- as.data.frame.matrix(tours_pertype_purpose) + +totalPersons <- sum(pertypeDistbn$freq) +totalPersons_DF <- data.frame("Total", totalPersons) +colnames(totalPersons_DF) <- colnames(pertypeDistbn) +pertypeDF <- rbind(pertypeDistbn, totalPersons_DF) +nm_tour_rates <- tours_pertype_purpose/pertypeDF$freq +nm_tour_rates$pertype <- row.names(nm_tour_rates) +nm_tour_rates <- melt(nm_tour_rates, id = c("pertype")) +colnames(nm_tour_rates) <- c("pertype", "tour_purp", "tour_rate") +nm_tour_rates$pertype <- as.character(nm_tour_rates$pertype) +nm_tour_rates$tour_purp <- as.character(nm_tour_rates$tour_purp) +nm_tour_rates$pertype[nm_tour_rates$pertype=="Sum"] <- "All" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp=="Sum"] <- "All" +nm_tour_rates$pertype <- pertypeCodes$name[match(nm_tour_rates$pertype, pertypeCodes$code)] + +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==4] <- "Escorting" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==5] <- "Shopping" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==6] <- "Maintenance" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==7] <- "EatingOut" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==8] <- "Visiting" +nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==9] <- "Discretionary" + +write.csv(nm_tour_rates, "nm_tour_rates.csv", row.names = F) + +# Total tours by purpose X tourtype +t1 <- hist(tours$TOURPURP[tours$TOURPURP<10], breaks = seq(1,10, by=1), freq = NULL, right=FALSE) +t3 <- hist(unique_joint_tours$JOINT_PURP, breaks = seq(1,10, by=1), freq = NULL, right=FALSE) +tours_purpose_type <- data.frame(t1$counts, t3$counts) +colnames(tours_purpose_type) <- c("indi", "joint") +write.csv(tours_purpose_type, "tours_purpose_type.csv", row.names = TRUE) + +# DAP by pertype +# recode pattern type for at-work and home schooling persons. +# these person have DAP as M. They should be recoded to N or H. +per[per$activity_pattern == 'M' & per$imf_choice==0 & per$inmf_choice>0]$activity_pattern = 'N' +per[per$activity_pattern == 'M' & per$imf_choice==0 & per$inmf_choice==0]$activity_pattern = 'H' + +dapSummary <- count(per, c("PERTYPE", "activity_pattern")) +write.csv(dapSummary, "dapSummary.csv", row.names = TRUE) + +# Prepare DAP summary for visualizer +dapSummary_vis <- xtabs(freq~PERTYPE+activity_pattern, dapSummary) +dapSummary_vis <- addmargins(as.table(dapSummary_vis)) +dapSummary_vis <- as.data.frame.matrix(dapSummary_vis) + +dapSummary_vis$id <- row.names(dapSummary_vis) +dapSummary_vis <- melt(dapSummary_vis, id = c("id")) +colnames(dapSummary_vis) <- c("PERTYPE", "DAP", "freq") +dapSummary_vis$PERTYPE <- as.character(dapSummary_vis$PERTYPE) +dapSummary_vis$DAP <- as.character(dapSummary_vis$DAP) +dapSummary_vis <- dapSummary_vis[dapSummary_vis$DAP!="Sum",] +dapSummary_vis$PERTYPE[dapSummary_vis$PERTYPE=="Sum"] <- "Total" +write.csv(dapSummary_vis, "dapSummary_vis.csv", row.names = TRUE) + +# HHSize X Joint +hhsizeJoint <- count(hh[hh$HHSIZE>=2,], c("HHSIZE", "JOINT")) +write.csv(hhsizeJoint, "hhsizeJoint.csv", row.names = TRUE) + +#mandatory tour frequency +mtfSummary <- count(per[per$imf_choice > 0,], c("PERTYPE", "imf_choice")) +write.csv(mtfSummary, "mtfSummary.csv") +#write.csv(tours, "tours_test.csv") + +# Prepare MTF summary for visualizer +mtfSummary_vis <- xtabs(freq~PERTYPE+imf_choice, mtfSummary) +mtfSummary_vis <- addmargins(as.table(mtfSummary_vis)) +mtfSummary_vis <- as.data.frame.matrix(mtfSummary_vis) + +mtfSummary_vis$id <- row.names(mtfSummary_vis) +mtfSummary_vis <- melt(mtfSummary_vis, id = c("id")) +colnames(mtfSummary_vis) <- c("PERTYPE", "MTF", "freq") +mtfSummary_vis$PERTYPE <- as.character(mtfSummary_vis$PERTYPE) +mtfSummary_vis$MTF <- as.character(mtfSummary_vis$MTF) +mtfSummary_vis <- mtfSummary_vis[mtfSummary_vis$MTF!="Sum",] +mtfSummary_vis$PERTYPE[mtfSummary_vis$PERTYPE=="Sum"] <- "Total" +write.csv(mtfSummary_vis, "mtfSummary_vis.csv") + +# indi NM summary +inm0Summary <- count(per[per$inmTours==0,], c("PERTYPE")) +inm1Summary <- count(per[per$inmTours==1,], c("PERTYPE")) +inm2Summary <- count(per[per$inmTours==2,], c("PERTYPE")) +inm3Summary <- count(per[per$inmTours>=3,], c("PERTYPE")) + +inmSummary <- data.frame(PERTYPE = c(1,2,3,4,5,6,7,8)) +inmSummary$tour0 <- inm0Summary$freq[match(inmSummary$PERTYPE, inm0Summary$PERTYPE)] +inmSummary$tour1 <- inm1Summary$freq[match(inmSummary$PERTYPE, inm1Summary$PERTYPE)] +inmSummary$tour2 <- inm2Summary$freq[match(inmSummary$PERTYPE, inm2Summary$PERTYPE)] +inmSummary$tour3pl <- inm3Summary$freq[match(inmSummary$PERTYPE, inm3Summary$PERTYPE)] + +write.table(inmSummary, "innmSummary.csv", col.names=TRUE, sep=",") + +# prepare INM summary for visualizer +inmSummary_vis <- melt(inmSummary, id=c("PERTYPE")) +inmSummary_vis$variable <- as.character(inmSummary_vis$variable) +inmSummary_vis$variable[inmSummary_vis$variable=="tour0"] <- "0" +inmSummary_vis$variable[inmSummary_vis$variable=="tour1"] <- "1" +inmSummary_vis$variable[inmSummary_vis$variable=="tour2"] <- "2" +inmSummary_vis$variable[inmSummary_vis$variable=="tour3pl"] <- "3pl" +inmSummary_vis <- xtabs(value~PERTYPE+variable, inmSummary_vis) +inmSummary_vis <- addmargins(as.table(inmSummary_vis)) +inmSummary_vis <- as.data.frame.matrix(inmSummary_vis) + +inmSummary_vis$id <- row.names(inmSummary_vis) +inmSummary_vis <- melt(inmSummary_vis, id = c("id")) +colnames(inmSummary_vis) <- c("PERTYPE", "nmtours", "freq") +inmSummary_vis$PERTYPE <- as.character(inmSummary_vis$PERTYPE) +inmSummary_vis$nmtours <- as.character(inmSummary_vis$nmtours) +inmSummary_vis <- inmSummary_vis[inmSummary_vis$nmtours!="Sum",] +inmSummary_vis$PERTYPE[inmSummary_vis$PERTYPE=="Sum"] <- "Total" +write.csv(inmSummary_vis, "inmSummary_vis.csv") + +# Joint Tour Frequency and composition +jtfSummary <- count(hh[!is.na(hh$jtf),], c("jtf")) +jointComp <- count(unique_joint_tours, c("tour_composition")) +jointPartySize <- count(unique_joint_tours, c("NUMBER_HH")) +jointCompPartySize <- count(unique_joint_tours, c("tour_composition","NUMBER_HH")) + +hh$jointCat[hh$jtours==0] <- 0 +hh$jointCat[hh$jtours==1] <- 1 +hh$jointCat[hh$jtours>=2] <- 2 + +jointToursHHSize <- count(hh[!is.na(hh$HHSIZE) & !is.na(hh$jointCat),], c("HHSIZE", "jointCat")) + +write.table(jtfSummary, "jtfSummary.csv", col.names=TRUE, sep=",") +write.table(jointComp, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) +write.table(jointPartySize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) +write.table(jointCompPartySize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) +write.table(jointToursHHSize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) + +#cap joint party size to 5+ +jointPartySize$freq[jointPartySize$NUMBER_HH==5] <- sum(jointPartySize$freq[jointPartySize$NUMBER_HH>=5]) +jointPartySize <- jointPartySize[jointPartySize$NUMBER_HH<=5, ] + +jtf <- data.frame(jtf_code = seq(from = 1, to = 21), + alt_name = c("No Joint Tours", "1 Shopping", "1 Maintenance", "1 Eating Out", "1 Visiting", "1 Other Discretionary", + "2 Shopping", "1 Shopping / 1 Maintenance", "1 Shopping / 1 Eating Out", "1 Shopping / 1 Visiting", + "1 Shopping / 1 Other Discretionary", "2 Maintenance", "1 Maintenance / 1 Eating Out", + "1 Maintenance / 1 Visiting", "1 Maintenance / 1 Other Discretionary", "2 Eating Out", "1 Eating Out / 1 Visiting", + "1 Eating Out / 1 Other Discretionary", "2 Visiting", "1 Visiting / 1 Other Discretionary", "2 Other Discretionary")) +jtf$freq <- jtfSummary$freq[match(jtf$jtf_code, jtfSummary$jtf)] +jtf[is.na(jtf)] <- 0 + +jointComp$tour_composition[jointComp$tour_composition==1] <- "All Adult" +jointComp$tour_composition[jointComp$tour_composition==2] <- "All Children" +jointComp$tour_composition[jointComp$tour_composition==3] <- "Mixed" + +jointToursHHSizeProp <- xtabs(freq~jointCat+HHSIZE, jointToursHHSize[jointToursHHSize$HHSIZE>1,]) +jointToursHHSizeProp <- addmargins(as.table(jointToursHHSizeProp)) +jointToursHHSizeProp <- jointToursHHSizeProp[-4,] #remove last row +jointToursHHSizeProp <- prop.table(jointToursHHSizeProp, margin = 2) +jointToursHHSizeProp <- as.data.frame.matrix(jointToursHHSizeProp) +jointToursHHSizeProp <- jointToursHHSizeProp*100 +jointToursHHSizeProp$jointTours <- row.names(jointToursHHSizeProp) +jointToursHHSizeProp <- melt(jointToursHHSizeProp, id = c("jointTours")) +colnames(jointToursHHSizeProp) <- c("jointTours", "hhsize", "freq") +jointToursHHSizeProp$hhsize <- as.character(jointToursHHSizeProp$hhsize) +jointToursHHSizeProp$hhsize[jointToursHHSizeProp$hhsize=="Sum"] <- "Total" + +jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==1] <- "All Adult" +jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==2] <- "All Children" +jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==3] <- "Mixed" + +jointCompPartySizeProp <- xtabs(freq~tour_composition+NUMBER_HH, jointCompPartySize) +jointCompPartySizeProp <- addmargins(as.table(jointCompPartySizeProp)) +jointCompPartySizeProp <- jointCompPartySizeProp[,-6] #remove last row +jointCompPartySizeProp <- prop.table(jointCompPartySizeProp, margin = 1) +jointCompPartySizeProp <- as.data.frame.matrix(jointCompPartySizeProp) +jointCompPartySizeProp <- jointCompPartySizeProp*100 +jointCompPartySizeProp$comp <- row.names(jointCompPartySizeProp) +jointCompPartySizeProp <- melt(jointCompPartySizeProp, id = c("comp")) +colnames(jointCompPartySizeProp) <- c("comp", "partysize", "freq") +jointCompPartySizeProp$comp <- as.character(jointCompPartySizeProp$comp) +jointCompPartySizeProp$comp[jointCompPartySizeProp$comp=="Sum"] <- "Total" + +# Cap joint comp party size at 5 +jointCompPartySizeProp <- jointCompPartySizeProp[jointCompPartySizeProp$partysize!="Sum",] +jointCompPartySizeProp$partysize <- as.numeric(as.character(jointCompPartySizeProp$partysize)) +jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Adult" & jointCompPartySizeProp$partysize==5] <- + sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Adult" & jointCompPartySizeProp$partysize>=5]) +jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Children" & jointCompPartySizeProp$partysize==5] <- + sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Children" & jointCompPartySizeProp$partysize>=5]) +jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Mixed" & jointCompPartySizeProp$partysize==5] <- + sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Mixed" & jointCompPartySizeProp$partysize>=5]) +jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Total" & jointCompPartySizeProp$partysize==5] <- + sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Total" & jointCompPartySizeProp$partysize>=5]) + +jointCompPartySizeProp <- jointCompPartySizeProp[jointCompPartySizeProp$partysize<=5,] + + +write.csv(jtf, "jtf.csv", row.names = F) +write.csv(jointComp, "jointComp.csv", row.names = F) +write.csv(jointPartySize, "jointPartySize.csv", row.names = F) +write.csv(jointCompPartySizeProp, "jointCompPartySize.csv", row.names = F) +write.csv(jointToursHHSizeProp, "jointToursHHSize.csv", row.names = F) + +# TOD Profile +#work.dep <- table(cut(tours$ANCHOR_DEPART_BIN[!is.na(tours$ANCHOR_DEPART_BIN)], seq(1,48, by=1), right=FALSE)) + +tod1 <- hist(tours$start_period[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod1_2 <- hist(tours$start_period[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod2 <- hist(tours$start_period[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod3 <- hist(tours$start_period[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod4 <- hist(tours$start_period[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi56 <- hist(tours$start_period[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi789 <- hist(tours$start_period[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod6 <- hist(tours$start_period[tours$TOURPURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod7 <- hist(tours$start_period[tours$TOURPURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod8 <- hist(tours$start_period[tours$TOURPURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod9 <- hist(tours$start_period[tours$TOURPURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todj56 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todj789 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod11 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod12 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod13 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod14 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod15 <- hist(tours$start_period[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) + +todDepProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts + , todj56$counts, todj789$counts, tod15$counts) +colnames(todDepProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", + "jmain", "jdisc", "atwork") +write.csv(todDepProfile, "todDepProfile.csv") + +# prepare input for visualizer +todDepProfile_vis <- todDepProfile +todDepProfile_vis$id <- row.names(todDepProfile_vis) +todDepProfile_vis <- melt(todDepProfile_vis, id = c("id")) +colnames(todDepProfile_vis) <- c("id", "purpose", "freq_dep") + +todDepProfile_vis$purpose <- as.character(todDepProfile_vis$purpose) +todDepProfile_vis <- xtabs(freq_dep~id+purpose, todDepProfile_vis) +todDepProfile_vis <- addmargins(as.table(todDepProfile_vis)) +todDepProfile_vis <- as.data.frame.matrix(todDepProfile_vis) +todDepProfile_vis$id <- row.names(todDepProfile_vis) +todDepProfile_vis <- melt(todDepProfile_vis, id = c("id")) +colnames(todDepProfile_vis) <- c("timebin", "PURPOSE", "freq") +todDepProfile_vis$PURPOSE <- as.character(todDepProfile_vis$PURPOSE) +todDepProfile_vis$timebin <- as.character(todDepProfile_vis$timebin) +todDepProfile_vis <- todDepProfile_vis[todDepProfile_vis$timebin!="Sum",] +todDepProfile_vis$PURPOSE[todDepProfile_vis$PURPOSE=="Sum"] <- "Total" +todDepProfile_vis$timebin <- as.numeric(todDepProfile_vis$timebin) + +tod1 <- hist(tours$end_period[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod1_2 <- hist(tours$end_period[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod2 <- hist(tours$end_period[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod3 <- hist(tours$end_period[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod4 <- hist(tours$end_period[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi56 <- hist(tours$end_period[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi789 <- hist(tours$end_period[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod8 <- hist(tours$end_period[tours$TOURPURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod9 <- hist(tours$end_period[tours$TOURPURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todj56 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todj789 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod11 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod12 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod13 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod14 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod15 <- hist(tours$end_period[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) + +todArrProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts + , todj56$counts, todj789$counts, tod15$counts) +colnames(todArrProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", + "jmain", "jdisc", "atwork") +write.csv(todArrProfile, "todArrProfile.csv") + + +##stops by direction, purpose and model tod + +tours$start_tod <- 5 # EA: 3 am - 6 am +tours$start_tod <- ifelse(tours$start_period>=4 & tours$start_period<=9, 1, tours$start_tod) # AM: 6 am - 9 am +tours$start_tod <- ifelse(tours$start_period>=10 & tours$start_period<=22, 2, tours$start_tod) # MD: 9 am - 3:30 pm +tours$start_tod <- ifelse(tours$start_period>=23 & tours$start_period<=29, 3, tours$start_tod) # PM: 3:30 pm - 7 pm +tours$start_tod <- ifelse(tours$start_period>=30 & tours$start_period<=40, 4, tours$start_tod) # EV: 7 pm - 3 am + +tours$end_tod <- 5 # EA: 3 am - 6 am +tours$end_tod <- ifelse(tours$end_period>=4 & tours$end_period<=9, 1, tours$end_tod) # AM: 6 am - 9 am +tours$end_tod <- ifelse(tours$end_period>=10 & tours$end_period<=22, 2, tours$end_tod) # MD: 9 am - 3:30 pm +tours$end_tod <- ifelse(tours$end_period>=23 & tours$end_period<=29, 3, tours$end_tod) # PM: 3:30 pm - 7 pm +tours$end_tod <- ifelse(tours$end_period>=30 & tours$end_period<=40, 4, tours$end_tod) # EV: 7 pm - 3 am + +stops_ib_tod <- aggregate(num_ib_stops~tour_purpose+start_tod+end_tod, data=tours, FUN = sum) +stops_ob_tod <- aggregate(num_ob_stops~tour_purpose+start_tod+end_tod, data=tours, FUN = sum) +write.csv(stops_ib_tod, "todStopsIB.csv", row.names = F) +write.csv(stops_ob_tod, "todStopsOB.csv", row.names = F) + +#joint tours +unique_joint_tours$start_tod <- 5 # EA: 3 am - 6 am +unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=4 & unique_joint_tours$start_period<=9, 1, unique_joint_tours$start_tod) # AM: 6 am - 9 am +unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=10 & unique_joint_tours$start_period<=22, 2, unique_joint_tours$start_tod) # MD: 9 am - 3:30 pm +unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=23 & unique_joint_tours$start_period<=29, 3, unique_joint_tours$start_tod) # PM: 3:30 pm - 7 pm +unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=30 & unique_joint_tours$start_period<=40, 4, unique_joint_tours$start_tod) # EV: 7 pm - 3 am + +unique_joint_tours$end_tod <- 5 # EA: 3 am - 6 am +unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=4 & unique_joint_tours$end_period<=9, 1, unique_joint_tours$end_tod) # AM: 6 am - 9 am +unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=10 & unique_joint_tours$end_period<=22, 2, unique_joint_tours$end_tod) # MD: 9 am - 3:30 pm +unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=23 & unique_joint_tours$end_period<=29, 3, unique_joint_tours$end_tod) # PM: 3:30 pm - 7 pm +unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=30 & unique_joint_tours$end_period<=40, 4, unique_joint_tours$end_tod) # EV: 7 pm - 3 am + +jstops_ib_tod <- aggregate(num_ib_stops~tour_purpose+start_tod+end_tod, data=unique_joint_tours, FUN = sum) +jstops_ob_tod <- aggregate(num_ob_stops~tour_purpose+start_tod+end_tod, data=unique_joint_tours, FUN = sum) +write.csv(jstops_ib_tod, "todStopsIB_joint.csv", row.names = F) +write.csv(jstops_ob_tod, "todStopsOB_joint.csv", row.names = F) + +# prepare input for visualizer +todArrProfile_vis <- todArrProfile +todArrProfile_vis$id <- row.names(todArrProfile_vis) +todArrProfile_vis <- melt(todArrProfile_vis, id = c("id")) +colnames(todArrProfile_vis) <- c("id", "purpose", "freq_arr") + +todArrProfile_vis$purpose <- as.character(todArrProfile_vis$purpose) +todArrProfile_vis <- xtabs(freq_arr~id+purpose, todArrProfile_vis) +todArrProfile_vis <- addmargins(as.table(todArrProfile_vis)) +todArrProfile_vis <- as.data.frame.matrix(todArrProfile_vis) +todArrProfile_vis$id <- row.names(todArrProfile_vis) +todArrProfile_vis <- melt(todArrProfile_vis, id = c("id")) +colnames(todArrProfile_vis) <- c("timebin", "PURPOSE", "freq") +todArrProfile_vis$PURPOSE <- as.character(todArrProfile_vis$PURPOSE) +todArrProfile_vis$timebin <- as.character(todArrProfile_vis$timebin) +todArrProfile_vis <- todArrProfile_vis[todArrProfile_vis$timebin!="Sum",] +todArrProfile_vis$PURPOSE[todArrProfile_vis$PURPOSE=="Sum"] <- "Total" +todArrProfile_vis$timebin <- as.numeric(todArrProfile_vis$timebin) + + +tod1 <- hist(tours$tourdur[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod1_2 <- hist(tours$tourdur[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +tod2 <- hist(tours$tourdur[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod3 <- hist(tours$tourdur[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +tod4 <- hist(tours$tourdur[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi56 <- hist(tours$tourdur[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todi789 <- hist(tours$tourdur[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod6 <- hist(tours$tourdur[tours$TOURPURP==6], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod7 <- hist(tours$tourdur[tours$TOURPURP==7], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod8 <- hist(tours$tourdur[tours$TOURPURP==8], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod9 <- hist(tours$tourdur[tours$TOURPURP==9], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +todj56 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +todj789 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) +#tod11 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==6], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod12 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==7], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod13 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==8], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +#tod14 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==9], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) +tod15 <- hist(tours$tourdur[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) + +todDurProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts + , todj56$counts, todj789$counts, tod15$counts) +colnames(todDurProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", + "jmain", "jdisc", "atwork") +write.csv(todDurProfile, "todDurProfile.csv") + +# prepare input for visualizer +todDurProfile_vis <- todDurProfile +todDurProfile_vis$id <- row.names(todDurProfile_vis) +todDurProfile_vis <- melt(todDurProfile_vis, id = c("id")) +colnames(todDurProfile_vis) <- c("id", "purpose", "freq_dur") + +todDurProfile_vis$purpose <- as.character(todDurProfile_vis$purpose) +todDurProfile_vis <- xtabs(freq_dur~id+purpose, todDurProfile_vis) +todDurProfile_vis <- addmargins(as.table(todDurProfile_vis)) +todDurProfile_vis <- as.data.frame.matrix(todDurProfile_vis) +todDurProfile_vis$id <- row.names(todDurProfile_vis) +todDurProfile_vis <- melt(todDurProfile_vis, id = c("id")) +colnames(todDurProfile_vis) <- c("timebin", "PURPOSE", "freq") +todDurProfile_vis$PURPOSE <- as.character(todDurProfile_vis$PURPOSE) +todDurProfile_vis$timebin <- as.character(todDurProfile_vis$timebin) +todDurProfile_vis <- todDurProfile_vis[todDurProfile_vis$timebin!="Sum",] +todDurProfile_vis$PURPOSE[todDurProfile_vis$PURPOSE=="Sum"] <- "Total" +todDurProfile_vis$timebin <- as.numeric(todDurProfile_vis$timebin) + +todDepProfile_vis <- todDepProfile_vis[order(todDepProfile_vis$timebin, todDepProfile_vis$PURPOSE), ] +todArrProfile_vis <- todArrProfile_vis[order(todArrProfile_vis$timebin, todArrProfile_vis$PURPOSE), ] +todDurProfile_vis <- todDurProfile_vis[order(todDurProfile_vis$timebin, todDurProfile_vis$PURPOSE), ] +todProfile_vis <- data.frame(todDepProfile_vis, todArrProfile_vis$freq, todDurProfile_vis$freq) +colnames(todProfile_vis) <- c("id", "purpose", "freq_dep", "freq_arr", "freq_dur") +write.csv(todProfile_vis, "todProfile_vis.csv", row.names = F) + +# Tour Mode X Auto Suff (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) +tmode1_as0 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode2_as0 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode3_as0 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode4_as0 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode5_as0 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode6_as0 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode7_as0 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode8_as0 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tmodeAS0Profile <- data.frame(tmode1_as0$counts, tmode2_as0$counts, tmode3_as0$counts, tmode4_as0$counts, + tmode5_as0$counts, tmode6_as0$counts, tmode7_as0$counts, tmode8_as0$counts) +colnames(tmodeAS0Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(tmodeAS0Profile, "tmodeAS0Profile.csv") + +# Prepeare data for visualizer (changed from 9 to 12) +tmodeAS0Profile_vis <- tmodeAS0Profile[1:13,] +tmodeAS0Profile_vis$id <- row.names(tmodeAS0Profile_vis) +tmodeAS0Profile_vis <- melt(tmodeAS0Profile_vis, id = c("id")) +colnames(tmodeAS0Profile_vis) <- c("id", "purpose", "freq_as0") + +tmodeAS0Profile_vis <- xtabs(freq_as0~id+purpose, tmodeAS0Profile_vis) +tmodeAS0Profile_vis[is.na(tmodeAS0Profile_vis)] <- 0 +tmodeAS0Profile_vis <- addmargins(as.table(tmodeAS0Profile_vis)) +tmodeAS0Profile_vis <- as.data.frame.matrix(tmodeAS0Profile_vis) + +tmodeAS0Profile_vis$id <- row.names(tmodeAS0Profile_vis) +tmodeAS0Profile_vis <- melt(tmodeAS0Profile_vis, id = c("id")) +colnames(tmodeAS0Profile_vis) <- c("id", "purpose", "freq_as0") +tmodeAS0Profile_vis$id <- as.character(tmodeAS0Profile_vis$id) +tmodeAS0Profile_vis$purpose <- as.character(tmodeAS0Profile_vis$purpose) +tmodeAS0Profile_vis <- tmodeAS0Profile_vis[tmodeAS0Profile_vis$id!="Sum",] +tmodeAS0Profile_vis$purpose[tmodeAS0Profile_vis$purpose=="Sum"] <- "Total" + +# (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) +tmode1_as1 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode2_as1 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode3_as1 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode4_as1 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode5_as1 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode6_as1 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode7_as1 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode8_as1 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tmodeAS1Profile <- data.frame(tmode1_as1$counts, tmode2_as1$counts, tmode3_as1$counts, tmode4_as1$counts, + tmode5_as1$counts, tmode6_as1$counts, tmode7_as1$counts, tmode8_as1$counts) +colnames(tmodeAS1Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(tmodeAS1Profile, "tmodeAS1Profile.csv") + +# Prepeare data for visualizer (changed from 9 to 12) +tmodeAS1Profile_vis <- tmodeAS1Profile[1:13,] +tmodeAS1Profile_vis$id <- row.names(tmodeAS1Profile_vis) +tmodeAS1Profile_vis <- melt(tmodeAS1Profile_vis, id = c("id")) +colnames(tmodeAS1Profile_vis) <- c("id", "purpose", "freq_as1") + +tmodeAS1Profile_vis <- xtabs(freq_as1~id+purpose, tmodeAS1Profile_vis) +tmodeAS1Profile_vis[is.na(tmodeAS1Profile_vis)] <- 0 +tmodeAS1Profile_vis <- addmargins(as.table(tmodeAS1Profile_vis)) +tmodeAS1Profile_vis <- as.data.frame.matrix(tmodeAS1Profile_vis) + +tmodeAS1Profile_vis$id <- row.names(tmodeAS1Profile_vis) +tmodeAS1Profile_vis <- melt(tmodeAS1Profile_vis, id = c("id")) +colnames(tmodeAS1Profile_vis) <- c("id", "purpose", "freq_as1") +tmodeAS1Profile_vis$id <- as.character(tmodeAS1Profile_vis$id) +tmodeAS1Profile_vis$purpose <- as.character(tmodeAS1Profile_vis$purpose) +tmodeAS1Profile_vis <- tmodeAS1Profile_vis[tmodeAS1Profile_vis$id!="Sum",] +tmodeAS1Profile_vis$purpose[tmodeAS1Profile_vis$purpose=="Sum"] <- "Total" + +# (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) +tmode1_as2 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode2_as2 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode3_as2 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode4_as2 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode5_as2 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode6_as2 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode7_as2 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tmode8_as2 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tmodeAS2Profile <- data.frame(tmode1_as2$counts, tmode2_as2$counts, tmode3_as2$counts, tmode4_as2$counts, + tmode5_as2$counts, tmode6_as2$counts, tmode7_as2$counts, tmode8_as2$counts) +colnames(tmodeAS2Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(tmodeAS2Profile, "tmodeAS2Profile.csv") + +# Prepeare data for visualizer (changed from 9 to 12) +tmodeAS2Profile_vis <- tmodeAS2Profile[1:13,] +tmodeAS2Profile_vis$id <- row.names(tmodeAS2Profile_vis) +tmodeAS2Profile_vis <- melt(tmodeAS2Profile_vis, id = c("id")) +colnames(tmodeAS2Profile_vis) <- c("id", "purpose", "freq_as2") + +tmodeAS2Profile_vis <- xtabs(freq_as2~id+purpose, tmodeAS2Profile_vis) +tmodeAS2Profile_vis[is.na(tmodeAS2Profile_vis)] <- 0 +tmodeAS2Profile_vis <- addmargins(as.table(tmodeAS2Profile_vis)) +tmodeAS2Profile_vis <- as.data.frame.matrix(tmodeAS2Profile_vis) + +tmodeAS2Profile_vis$id <- row.names(tmodeAS2Profile_vis) +tmodeAS2Profile_vis <- melt(tmodeAS2Profile_vis, id = c("id")) +colnames(tmodeAS2Profile_vis) <- c("id", "purpose", "freq_as2") +tmodeAS2Profile_vis$id <- as.character(tmodeAS2Profile_vis$id) +tmodeAS2Profile_vis$purpose <- as.character(tmodeAS2Profile_vis$purpose) +tmodeAS2Profile_vis <- tmodeAS2Profile_vis[tmodeAS2Profile_vis$id!="Sum",] +tmodeAS2Profile_vis$purpose[tmodeAS2Profile_vis$purpose=="Sum"] <- "Total" + + +# Combine three AS groups +tmodeProfile_vis <- data.frame(tmodeAS0Profile_vis, tmodeAS1Profile_vis$freq_as1, tmodeAS2Profile_vis$freq_as2) +colnames(tmodeProfile_vis) <- c("id", "purpose", "freq_as0", "freq_as1", "freq_as2") +tmodeProfile_vis$freq_all <- tmodeProfile_vis$freq_as0 + tmodeProfile_vis$freq_as1 + tmodeProfile_vis$freq_as2 +write.csv(tmodeProfile_vis, "tmodeProfile_vis.csv", row.names = F) + + +# Non Mand Tour lengths +tourdist4 <- hist(tours$tour_distance[tours$TOURPURP==4], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +tourdisti56 <- hist(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +tourdisti789 <- hist(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +tourdistj56 <- hist(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +tourdistj789 <- hist(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +tourdist10 <- hist(tours$tour_distance[tours$TOURPURP==10], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) + +tourDistProfile <- data.frame(tourdist4$counts, tourdisti56$counts, tourdisti789$counts, tourdistj56$counts, tourdistj789$counts, tourdist10$counts) + +colnames(tourDistProfile) <- c("esco", "imain", "idisc", "jmain", "jdisc", "atwork") + +write.csv(tourDistProfile, "nonMandTourDistProfile.csv") + +#prepare input for visualizer +tourDistProfile_vis <- tourDistProfile +tourDistProfile_vis$id <- row.names(tourDistProfile_vis) +tourDistProfile_vis <- melt(tourDistProfile_vis, id = c("id")) +colnames(tourDistProfile_vis) <- c("id", "purpose", "freq") + +tourDistProfile_vis <- xtabs(freq~id+purpose, tourDistProfile_vis) +tourDistProfile_vis <- addmargins(as.table(tourDistProfile_vis)) +tourDistProfile_vis <- as.data.frame.matrix(tourDistProfile_vis) +tourDistProfile_vis$id <- row.names(tourDistProfile_vis) +tourDistProfile_vis <- melt(tourDistProfile_vis, id = c("id")) +colnames(tourDistProfile_vis) <- c("distbin", "PURPOSE", "freq") +tourDistProfile_vis$PURPOSE <- as.character(tourDistProfile_vis$PURPOSE) +tourDistProfile_vis$distbin <- as.character(tourDistProfile_vis$distbin) +tourDistProfile_vis <- tourDistProfile_vis[tourDistProfile_vis$distbin!="Sum",] +tourDistProfile_vis$PURPOSE[tourDistProfile_vis$PURPOSE=="Sum"] <- "Total" +tourDistProfile_vis$distbin <- as.numeric(tourDistProfile_vis$distbin) + +write.csv(tourDistProfile_vis, "tourDistProfile_vis.csv", row.names = F) + +cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4], na.rm = TRUE)) +cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], na.rm = TRUE)) +cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], na.rm = TRUE)) +cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], na.rm = TRUE)) +cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], na.rm = TRUE)) +cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10], na.rm = TRUE)) + +## Retirees +#cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4 & tours$PERTYPE==5], na.rm = TRUE)) +#cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6 & tours$PERTYPE==5], na.rm = TRUE)) +#cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$PERTYPE==5], na.rm = TRUE)) +#cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$PERTYPE==5], na.rm = TRUE)) +#cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$PERTYPE==5], na.rm = TRUE)) +#cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10 & tours$PERTYPE==5], na.rm = TRUE)) +# +## Non-reitrees +#cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4 & tours$PERTYPE!=5], na.rm = TRUE)) +#cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6 & tours$PERTYPE!=5], na.rm = TRUE)) +#cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$PERTYPE!=5], na.rm = TRUE)) +#cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$PERTYPE!=5], na.rm = TRUE)) +#cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$PERTYPE!=5], na.rm = TRUE)) +#cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10 & tours$PERTYPE!=5], na.rm = TRUE)) +# + +## Output average trips lengths for visualizer + +avgTripLengths <- c(mean(tours$tour_distance[tours$TOURPURP==4], na.rm = TRUE), + mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], na.rm = TRUE), + mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], na.rm = TRUE), + mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], na.rm = TRUE), + mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], na.rm = TRUE), + mean(tours$tour_distance[tours$TOURPURP==10], na.rm = TRUE)) + +totAvgNonMand <- mean(c(tours$tour_distance[tours$TOURPURP %in% c(4,5,6,7,8,9,10)], + unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP %in% c(5,6,7,8,9)]), + na.rm = T) +avgTripLengths <- c(avgTripLengths, totAvgNonMand) + +nonMandTourPurpose <- c("esco", "imain", "idisc", "jmain", "jdisc", "atwork", "Total") + +nonMandTripLengths <- data.frame(purpose = nonMandTourPurpose, avgTripLength = avgTripLengths) + +write.csv(nonMandTripLengths, "nonMandTripLengths.csv", row.names = F) + +# STop Frequency +#Outbound +stopfreq1 <- hist(tours$num_ob_stops[tours$TOURPURP==1], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(tours$num_ob_stops[tours$TOURPURP==2], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(tours$num_ob_stops[tours$TOURPURP==3], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(tours$num_ob_stops[tours$TOURPURP==4], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(tours$num_ob_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(tours$num_ob_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(unique_joint_tours$num_ob_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(unique_joint_tours$num_ob_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(tours$num_ob_stops[tours$TOURPURP==10], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopFreqOutProfile.csv") + +# prepare stop frequency input for visualizer +stopFreqout_vis <- stopFreq +stopFreqout_vis$id <- row.names(stopFreqout_vis) +stopFreqout_vis <- melt(stopFreqout_vis, id = c("id")) +colnames(stopFreqout_vis) <- c("id", "purpose", "freq") + +stopFreqout_vis <- xtabs(freq~purpose+id, stopFreqout_vis) +stopFreqout_vis <- addmargins(as.table(stopFreqout_vis)) +stopFreqout_vis <- as.data.frame.matrix(stopFreqout_vis) +stopFreqout_vis$id <- row.names(stopFreqout_vis) +stopFreqout_vis <- melt(stopFreqout_vis, id = c("id")) +colnames(stopFreqout_vis) <- c("purpose", "nstops", "freq") +stopFreqout_vis$purpose <- as.character(stopFreqout_vis$purpose) +stopFreqout_vis$nstops <- as.character(stopFreqout_vis$nstops) +stopFreqout_vis <- stopFreqout_vis[stopFreqout_vis$nstops!="Sum",] +stopFreqout_vis$purpose[stopFreqout_vis$purpose=="Sum"] <- "Total" + +#Inbound +stopfreq1 <- hist(tours$num_ib_stops[tours$TOURPURP==1], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(tours$num_ib_stops[tours$TOURPURP==2], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(tours$num_ib_stops[tours$TOURPURP==3], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(tours$num_ib_stops[tours$TOURPURP==4], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(tours$num_ib_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(tours$num_ib_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(unique_joint_tours$num_ib_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(unique_joint_tours$num_ib_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(tours$num_ib_stops[tours$TOURPURP==10], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopFreqInbProfile.csv") + +# prepare stop frequency input for visualizer +stopFreqinb_vis <- stopFreq +stopFreqinb_vis$id <- row.names(stopFreqinb_vis) +stopFreqinb_vis <- melt(stopFreqinb_vis, id = c("id")) +colnames(stopFreqinb_vis) <- c("id", "purpose", "freq") + +stopFreqinb_vis <- xtabs(freq~purpose+id, stopFreqinb_vis) +stopFreqinb_vis <- addmargins(as.table(stopFreqinb_vis)) +stopFreqinb_vis <- as.data.frame.matrix(stopFreqinb_vis) +stopFreqinb_vis$id <- row.names(stopFreqinb_vis) +stopFreqinb_vis <- melt(stopFreqinb_vis, id = c("id")) +colnames(stopFreqinb_vis) <- c("purpose", "nstops", "freq") +stopFreqinb_vis$purpose <- as.character(stopFreqinb_vis$purpose) +stopFreqinb_vis$nstops <- as.character(stopFreqinb_vis$nstops) +stopFreqinb_vis <- stopFreqinb_vis[stopFreqinb_vis$nstops!="Sum",] +stopFreqinb_vis$purpose[stopFreqinb_vis$purpose=="Sum"] <- "Total" + + +stopfreqDir_vis <- data.frame(stopFreqout_vis, stopFreqinb_vis$freq) +colnames(stopfreqDir_vis) <- c("purpose", "nstops", "freq_out", "freq_inb") +write.csv(stopfreqDir_vis, "stopfreqDir_vis.csv", row.names = F) + + +#Total +stopfreq1 <- hist(tours$num_tot_stops[tours$TOURPURP==1], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(tours$num_tot_stops[tours$TOURPURP==2], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(tours$num_tot_stops[tours$TOURPURP==3], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(tours$num_tot_stops[tours$TOURPURP==4], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(tours$num_tot_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(tours$num_tot_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(unique_joint_tours$num_tot_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(unique_joint_tours$num_tot_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(tours$num_tot_stops[tours$TOURPURP==10], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopFreqTotProfile.csv") + +# prepare stop frequency input for visualizer +stopFreq_vis <- stopFreq +stopFreq_vis$id <- row.names(stopFreq_vis) +stopFreq_vis <- melt(stopFreq_vis, id = c("id")) +colnames(stopFreq_vis) <- c("id", "purpose", "freq") + +stopFreq_vis <- xtabs(freq~purpose+id, stopFreq_vis) +stopFreq_vis <- addmargins(as.table(stopFreq_vis)) +stopFreq_vis <- as.data.frame.matrix(stopFreq_vis) +stopFreq_vis$id <- row.names(stopFreq_vis) +stopFreq_vis <- melt(stopFreq_vis, id = c("id")) +colnames(stopFreq_vis) <- c("purpose", "nstops", "freq") +stopFreq_vis$purpose <- as.character(stopFreq_vis$purpose) +stopFreq_vis$nstops <- as.character(stopFreq_vis$nstops) +stopFreq_vis <- stopFreq_vis[stopFreq_vis$nstops!="Sum",] +stopFreq_vis$purpose[stopFreq_vis$purpose=="Sum"] <- "Total" + +write.csv(stopFreq_vis, "stopfreq_total_vis.csv", row.names = F) + +#STop purpose X TourPurpose +stopfreq1 <- hist(stops$DPURP[stops$TOURPURP==1], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(stops$DPURP[stops$TOURPURP==2], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(stops$DPURP[stops$TOURPURP==3], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(stops$DPURP[stops$TOURPURP==4], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(stops$DPURP[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(stops$DPURP[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(jstops$DPURP[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(jstops$DPURP[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(stops$DPURP[stops$TOURPURP==10], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopPurposeByTourPurpose.csv") + +# prepare stop frequency input for visualizer +stopFreq_vis <- stopFreq +stopFreq_vis$id <- row.names(stopFreq_vis) +stopFreq_vis <- melt(stopFreq_vis, id = c("id")) +colnames(stopFreq_vis) <- c("stop_purp", "purpose", "freq") + +stopFreq_vis <- xtabs(freq~purpose+stop_purp, stopFreq_vis) +stopFreq_vis <- addmargins(as.table(stopFreq_vis)) +stopFreq_vis <- as.data.frame.matrix(stopFreq_vis) +stopFreq_vis$purpose <- row.names(stopFreq_vis) +stopFreq_vis <- melt(stopFreq_vis, id = c("purpose")) +colnames(stopFreq_vis) <- c("purpose", "stop_purp", "freq") +stopFreq_vis$purpose <- as.character(stopFreq_vis$purpose) +stopFreq_vis$stop_purp <- as.character(stopFreq_vis$stop_purp) +stopFreq_vis <- stopFreq_vis[stopFreq_vis$stop_purp!="Sum",] +stopFreq_vis$purpose[stopFreq_vis$purpose=="Sum"] <- "Total" + +write.csv(stopFreq_vis, "stoppurpose_tourpurpose_vis.csv", row.names = F) + +#Out of direction - Stop Location +stopfreq1 <- hist(stops$out_dir_dist[stops$TOURPURP==1], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(stops$out_dir_dist[stops$TOURPURP==2], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(stops$out_dir_dist[stops$TOURPURP==3], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(stops$out_dir_dist[stops$TOURPURP==4], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(stops$out_dir_dist[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(stops$out_dir_dist[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(jstops$out_dir_dist[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(jstops$out_dir_dist[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(stops$out_dir_dist[stops$TOURPURP==10], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopOutOfDirectionDC.csv") + +# prepare stop location input for visualizer +stopDC_vis <- stopFreq +stopDC_vis$id <- row.names(stopDC_vis) +stopDC_vis <- melt(stopDC_vis, id = c("id")) +colnames(stopDC_vis) <- c("id", "purpose", "freq") + +stopDC_vis <- xtabs(freq~id+purpose, stopDC_vis) +stopDC_vis <- addmargins(as.table(stopDC_vis)) +stopDC_vis <- as.data.frame.matrix(stopDC_vis) +stopDC_vis$id <- row.names(stopDC_vis) +stopDC_vis <- melt(stopDC_vis, id = c("id")) +colnames(stopDC_vis) <- c("distbin", "PURPOSE", "freq") +stopDC_vis$PURPOSE <- as.character(stopDC_vis$PURPOSE) +stopDC_vis$distbin <- as.character(stopDC_vis$distbin) +stopDC_vis <- stopDC_vis[stopDC_vis$distbin!="Sum",] +stopDC_vis$PURPOSE[stopDC_vis$PURPOSE=="Sum"] <- "Total" +stopDC_vis$distbin <- as.numeric(stopDC_vis$distbin) + +write.csv(stopDC_vis, "stopDC_vis.csv", row.names = F) + +# compute average out of dir distance for visualizer +avgDistances <- c(mean(stops$out_dir_dist[stops$TOURPURP==1], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP==2], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP==3], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP==4], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP>=5 & stops$TOURPURP<=6], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP>=7 & stops$TOURPURP<=9], na.rm = TRUE), + mean(jstops$out_dir_dist[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], na.rm = TRUE), + mean(jstops$out_dir_dist[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], na.rm = TRUE), + mean(stops$out_dir_dist[stops$TOURPURP==10], na.rm = TRUE), + mean(stops$out_dir_dist, na.rm = TRUE)) + +purp <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork", "total") + +avgStopOutofDirectionDist <- data.frame(purpose = purp, avgDist = avgDistances) + +write.csv(avgStopOutofDirectionDist, "avgStopOutofDirectionDist_vis.csv", row.names = F) + +#Stop Departure Time +stopfreq1 <- hist(stops$stop_period[stops$TOURPURP==1], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(stops$stop_period[stops$TOURPURP==2], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(stops$stop_period[stops$TOURPURP==3], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(stops$stop_period[stops$TOURPURP==4], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(stops$stop_period[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(stops$stop_period[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(jstops$stop_period[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(jstops$stop_period[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(stops$stop_period[stops$TOURPURP==10], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "stopDeparture.csv") + +# prepare stop departure input for visualizer +stopDep_vis <- stopFreq +stopDep_vis$id <- row.names(stopDep_vis) +stopDep_vis <- melt(stopDep_vis, id = c("id")) +colnames(stopDep_vis) <- c("id", "purpose", "freq_stop") + +stopDep_vis$purpose <- as.character(stopDep_vis$purpose) +stopDep_vis <- xtabs(freq_stop~id+purpose, stopDep_vis) +stopDep_vis <- addmargins(as.table(stopDep_vis)) +stopDep_vis <- as.data.frame.matrix(stopDep_vis) +stopDep_vis$id <- row.names(stopDep_vis) +stopDep_vis <- melt(stopDep_vis, id = c("id")) +colnames(stopDep_vis) <- c("timebin", "PURPOSE", "freq") +stopDep_vis$PURPOSE <- as.character(stopDep_vis$PURPOSE) +stopDep_vis$timebin <- as.character(stopDep_vis$timebin) +stopDep_vis <- stopDep_vis[stopDep_vis$timebin!="Sum",] +stopDep_vis$PURPOSE[stopDep_vis$PURPOSE=="Sum"] <- "Total" +stopDep_vis$timebin <- as.numeric(stopDep_vis$timebin) + +#Trip Departure Time +stopfreq1 <- hist(trips$stop_period[trips$TOURPURP==1], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq2 <- hist(trips$stop_period[trips$TOURPURP==2], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq3 <- hist(trips$stop_period[trips$TOURPURP==3], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq4 <- hist(trips$stop_period[trips$TOURPURP==4], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi56 <- hist(trips$stop_period[trips$TOURPURP>=5 & trips$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqi789 <- hist(trips$stop_period[trips$TOURPURP>=7 & trips$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj56 <- hist(jtrips$stop_period[jtrips$TOURPURP>=5 & jtrips$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreqj789 <- hist(jtrips$stop_period[jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) +stopfreq10 <- hist(trips$stop_period[trips$TOURPURP==10], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) + +stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts + , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) +colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") +write.csv(stopFreq, "tripDeparture.csv") + +# prepare stop departure input for visualizer +tripDep_vis <- stopFreq +tripDep_vis$id <- row.names(tripDep_vis) +tripDep_vis <- melt(tripDep_vis, id = c("id")) +colnames(tripDep_vis) <- c("id", "purpose", "freq_trip") + +tripDep_vis$purpose <- as.character(tripDep_vis$purpose) +tripDep_vis <- xtabs(freq_trip~id+purpose, tripDep_vis) +tripDep_vis <- addmargins(as.table(tripDep_vis)) +tripDep_vis <- as.data.frame.matrix(tripDep_vis) +tripDep_vis$id <- row.names(tripDep_vis) +tripDep_vis <- melt(tripDep_vis, id = c("id")) +colnames(tripDep_vis) <- c("timebin", "PURPOSE", "freq") +tripDep_vis$PURPOSE <- as.character(tripDep_vis$PURPOSE) +tripDep_vis$timebin <- as.character(tripDep_vis$timebin) +tripDep_vis <- tripDep_vis[tripDep_vis$timebin!="Sum",] +tripDep_vis$PURPOSE[tripDep_vis$PURPOSE=="Sum"] <- "Total" +tripDep_vis$timebin <- as.numeric(tripDep_vis$timebin) + +stopTripDep_vis <- data.frame(stopDep_vis, tripDep_vis$freq) +colnames(stopTripDep_vis) <- c("id", "purpose", "freq_stop", "freq_trip") +write.csv(stopTripDep_vis, "stopTripDep_vis.csv", row.names = F) + +#Trip Mode Summary (added 3 lines due to change in mode codes, changed seq 9 to 13, Khademul Haque) +#Work +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_Work.csv") + +# Prepare data for visualizer (changed from 9 to 12) +tripModeProfile1_vis <- tripModeProfile[1:13,] +tripModeProfile1_vis$id <- row.names(tripModeProfile1_vis) +tripModeProfile1_vis <- melt(tripModeProfile1_vis, id = c("id")) +colnames(tripModeProfile1_vis) <- c("id", "purpose", "freq1") + +tripModeProfile1_vis <- xtabs(freq1~id+purpose, tripModeProfile1_vis) +tripModeProfile1_vis[is.na(tripModeProfile1_vis)] <- 0 +tripModeProfile1_vis <- addmargins(as.table(tripModeProfile1_vis)) +tripModeProfile1_vis <- as.data.frame.matrix(tripModeProfile1_vis) + +tripModeProfile1_vis$id <- row.names(tripModeProfile1_vis) +tripModeProfile1_vis <- melt(tripModeProfile1_vis, id = c("id")) +colnames(tripModeProfile1_vis) <- c("id", "purpose", "freq1") +tripModeProfile1_vis$id <- as.character(tripModeProfile1_vis$id) +tripModeProfile1_vis$purpose <- as.character(tripModeProfile1_vis$purpose) +tripModeProfile1_vis <- tripModeProfile1_vis[tripModeProfile1_vis$id!="Sum",] +tripModeProfile1_vis$purpose[tripModeProfile1_vis$purpose=="Sum"] <- "Total" + + +#University (added 3 lines due to change in mode codes, changed seq 9 to 13, Khademul Haque) +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_Univ.csv") + +tripModeProfile2_vis <- tripModeProfile[1:13,] +tripModeProfile2_vis$id <- row.names(tripModeProfile2_vis) +tripModeProfile2_vis <- melt(tripModeProfile2_vis, id = c("id")) +colnames(tripModeProfile2_vis) <- c("id", "purpose", "freq2") + +tripModeProfile2_vis <- xtabs(freq2~id+purpose, tripModeProfile2_vis) +tripModeProfile2_vis[is.na(tripModeProfile2_vis)] <- 0 +tripModeProfile2_vis <- addmargins(as.table(tripModeProfile2_vis)) +tripModeProfile2_vis <- as.data.frame.matrix(tripModeProfile2_vis) + +tripModeProfile2_vis$id <- row.names(tripModeProfile2_vis) +tripModeProfile2_vis <- melt(tripModeProfile2_vis, id = c("id")) +colnames(tripModeProfile2_vis) <- c("id", "purpose", "freq2") +tripModeProfile2_vis$id <- as.character(tripModeProfile2_vis$id) +tripModeProfile2_vis$purpose <- as.character(tripModeProfile2_vis$purpose) +tripModeProfile2_vis <- tripModeProfile2_vis[tripModeProfile2_vis$id!="Sum",] +tripModeProfile2_vis$purpose[tripModeProfile2_vis$purpose=="Sum"] <- "Total" + +#School +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_Schl.csv") + +tripModeProfile3_vis <- tripModeProfile[1:13,] +tripModeProfile3_vis$id <- row.names(tripModeProfile3_vis) +tripModeProfile3_vis <- melt(tripModeProfile3_vis, id = c("id")) +colnames(tripModeProfile3_vis) <- c("id", "purpose", "freq3") + +tripModeProfile3_vis <- xtabs(freq3~id+purpose, tripModeProfile3_vis) +tripModeProfile3_vis[is.na(tripModeProfile3_vis)] <- 0 +tripModeProfile3_vis <- addmargins(as.table(tripModeProfile3_vis)) +tripModeProfile3_vis <- as.data.frame.matrix(tripModeProfile3_vis) + +tripModeProfile3_vis$id <- row.names(tripModeProfile3_vis) +tripModeProfile3_vis <- melt(tripModeProfile3_vis, id = c("id")) +colnames(tripModeProfile3_vis) <- c("id", "purpose", "freq3") +tripModeProfile3_vis$id <- as.character(tripModeProfile3_vis$id) +tripModeProfile3_vis$purpose <- as.character(tripModeProfile3_vis$purpose) +tripModeProfile3_vis <- tripModeProfile3_vis[tripModeProfile3_vis$id!="Sum",] +tripModeProfile3_vis$purpose[tripModeProfile3_vis$purpose=="Sum"] <- "Total" + +#iMain +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_iMain.csv") + +tripModeProfile4_vis <- tripModeProfile[1:13,] +tripModeProfile4_vis$id <- row.names(tripModeProfile4_vis) +tripModeProfile4_vis <- melt(tripModeProfile4_vis, id = c("id")) +colnames(tripModeProfile4_vis) <- c("id", "purpose", "freq4") + +tripModeProfile4_vis <- xtabs(freq4~id+purpose, tripModeProfile4_vis) +tripModeProfile4_vis[is.na(tripModeProfile4_vis)] <- 0 +tripModeProfile4_vis <- addmargins(as.table(tripModeProfile4_vis)) +tripModeProfile4_vis <- as.data.frame.matrix(tripModeProfile4_vis) + +tripModeProfile4_vis$id <- row.names(tripModeProfile4_vis) +tripModeProfile4_vis <- melt(tripModeProfile4_vis, id = c("id")) +colnames(tripModeProfile4_vis) <- c("id", "purpose", "freq4") +tripModeProfile4_vis$id <- as.character(tripModeProfile4_vis$id) +tripModeProfile4_vis$purpose <- as.character(tripModeProfile4_vis$purpose) +tripModeProfile4_vis <- tripModeProfile4_vis[tripModeProfile4_vis$id!="Sum",] +tripModeProfile4_vis$purpose[tripModeProfile4_vis$purpose=="Sum"] <- "Total" + +#iDisc +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_iDisc.csv") + +tripModeProfile5_vis <- tripModeProfile[1:13,] +tripModeProfile5_vis$id <- row.names(tripModeProfile5_vis) +tripModeProfile5_vis <- melt(tripModeProfile5_vis, id = c("id")) +colnames(tripModeProfile5_vis) <- c("id", "purpose", "freq5") + +tripModeProfile5_vis <- xtabs(freq5~id+purpose, tripModeProfile5_vis) +tripModeProfile5_vis[is.na(tripModeProfile5_vis)] <- 0 +tripModeProfile5_vis <- addmargins(as.table(tripModeProfile5_vis)) +tripModeProfile5_vis <- as.data.frame.matrix(tripModeProfile5_vis) + +tripModeProfile5_vis$id <- row.names(tripModeProfile5_vis) +tripModeProfile5_vis <- melt(tripModeProfile5_vis, id = c("id")) +colnames(tripModeProfile5_vis) <- c("id", "purpose", "freq5") +tripModeProfile5_vis$id <- as.character(tripModeProfile5_vis$id) +tripModeProfile5_vis$purpose <- as.character(tripModeProfile5_vis$purpose) +tripModeProfile5_vis <- tripModeProfile5_vis[tripModeProfile5_vis$id!="Sum",] +tripModeProfile5_vis$purpose[tripModeProfile5_vis$purpose=="Sum"] <- "Total" + +#jMain +tripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_jMain.csv") + +tripModeProfile6_vis <- tripModeProfile[1:13,] +tripModeProfile6_vis$id <- row.names(tripModeProfile6_vis) +tripModeProfile6_vis <- melt(tripModeProfile6_vis, id = c("id")) +colnames(tripModeProfile6_vis) <- c("id", "purpose", "freq6") + +tripModeProfile6_vis <- xtabs(freq6~id+purpose, tripModeProfile6_vis) +tripModeProfile6_vis[is.na(tripModeProfile6_vis)] <- 0 +tripModeProfile6_vis <- addmargins(as.table(tripModeProfile6_vis)) +tripModeProfile6_vis <- as.data.frame.matrix(tripModeProfile6_vis) + +tripModeProfile6_vis$id <- row.names(tripModeProfile6_vis) +tripModeProfile6_vis <- melt(tripModeProfile6_vis, id = c("id")) +colnames(tripModeProfile6_vis) <- c("id", "purpose", "freq6") +tripModeProfile6_vis$id <- as.character(tripModeProfile6_vis$id) +tripModeProfile6_vis$purpose <- as.character(tripModeProfile6_vis$purpose) +tripModeProfile6_vis <- tripModeProfile6_vis[tripModeProfile6_vis$id!="Sum",] +tripModeProfile6_vis$purpose[tripModeProfile6_vis$purpose=="Sum"] <- "Total" + +#jDisc +tripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_jDisc.csv") + +tripModeProfile7_vis <- tripModeProfile[1:13,] +tripModeProfile7_vis$id <- row.names(tripModeProfile7_vis) +tripModeProfile7_vis <- melt(tripModeProfile7_vis, id = c("id")) +colnames(tripModeProfile7_vis) <- c("id", "purpose", "freq7") + +tripModeProfile7_vis <- xtabs(freq7~id+purpose, tripModeProfile7_vis) +tripModeProfile7_vis[is.na(tripModeProfile7_vis)] <- 0 +tripModeProfile7_vis <- addmargins(as.table(tripModeProfile7_vis)) +tripModeProfile7_vis <- as.data.frame.matrix(tripModeProfile7_vis) + +tripModeProfile7_vis$id <- row.names(tripModeProfile7_vis) +tripModeProfile7_vis <- melt(tripModeProfile7_vis, id = c("id")) +colnames(tripModeProfile7_vis) <- c("id", "purpose", "freq7") +tripModeProfile7_vis$id <- as.character(tripModeProfile7_vis$id) +tripModeProfile7_vis$purpose <- as.character(tripModeProfile7_vis$purpose) +tripModeProfile7_vis <- tripModeProfile7_vis[tripModeProfile7_vis$id!="Sum",] +tripModeProfile7_vis$purpose[tripModeProfile7_vis$purpose=="Sum"] <- "Total" + +#At work +tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, + tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, + tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_AtWork.csv") + +tripModeProfile8_vis <- tripModeProfile[1:13,] +tripModeProfile8_vis$id <- row.names(tripModeProfile8_vis) +tripModeProfile8_vis <- melt(tripModeProfile8_vis, id = c("id")) +colnames(tripModeProfile8_vis) <- c("id", "purpose", "freq8") + +tripModeProfile8_vis <- xtabs(freq8~id+purpose, tripModeProfile8_vis) +tripModeProfile8_vis[is.na(tripModeProfile8_vis)] <- 0 +tripModeProfile8_vis <- addmargins(as.table(tripModeProfile8_vis)) +tripModeProfile8_vis <- as.data.frame.matrix(tripModeProfile8_vis) + +tripModeProfile8_vis$id <- row.names(tripModeProfile8_vis) +tripModeProfile8_vis <- melt(tripModeProfile8_vis, id = c("id")) +colnames(tripModeProfile8_vis) <- c("id", "purpose", "freq8") +tripModeProfile8_vis$id <- as.character(tripModeProfile8_vis$id) +tripModeProfile8_vis$purpose <- as.character(tripModeProfile8_vis$purpose) +tripModeProfile8_vis <- tripModeProfile8_vis[tripModeProfile8_vis$id!="Sum",] +tripModeProfile8_vis$purpose[tripModeProfile8_vis$purpose=="Sum"] <- "Total" + +#iTotal +itripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +itripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +#jTotal +jtripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) +jtripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) + +tripModeProfile <- data.frame(itripmode1$counts+jtripmode1$counts, itripmode2$counts+jtripmode2$counts, itripmode3$counts+jtripmode3$counts, itripmode4$counts+jtripmode4$counts, + itripmode5$counts+jtripmode5$counts, itripmode6$counts+jtripmode6$counts, itripmode7$counts+jtripmode7$counts, itripmode8$counts+jtripmode8$counts, + itripmode9$counts+jtripmode9$counts, itripmode10$counts+jtripmode10$counts, itripmode11$counts+jtripmode11$counts, itripmode12$counts+jtripmode12$counts, itripmode13$counts+jtripmode13$counts) +colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") +write.csv(tripModeProfile, "tripModeProfile_Total.csv") + +tripModeProfile9_vis <- tripModeProfile[1:13,] +tripModeProfile9_vis$id <- row.names(tripModeProfile9_vis) +tripModeProfile9_vis <- melt(tripModeProfile9_vis, id = c("id")) +colnames(tripModeProfile9_vis) <- c("id", "purpose", "freq9") + +tripModeProfile9_vis <- xtabs(freq9~id+purpose, tripModeProfile9_vis) +tripModeProfile9_vis[is.na(tripModeProfile9_vis)] <- 0 +tripModeProfile9_vis <- addmargins(as.table(tripModeProfile9_vis)) +tripModeProfile9_vis <- as.data.frame.matrix(tripModeProfile9_vis) + +tripModeProfile9_vis$id <- row.names(tripModeProfile9_vis) +tripModeProfile9_vis <- melt(tripModeProfile9_vis, id = c("id")) +colnames(tripModeProfile9_vis) <- c("id", "purpose", "freq9") +tripModeProfile9_vis$id <- as.character(tripModeProfile9_vis$id) +tripModeProfile9_vis$purpose <- as.character(tripModeProfile9_vis$purpose) +tripModeProfile9_vis <- tripModeProfile9_vis[tripModeProfile9_vis$id!="Sum",] +tripModeProfile9_vis$purpose[tripModeProfile9_vis$purpose=="Sum"] <- "Total" + + +# combine all tripmode profile for visualizer +tripModeProfile_vis <- data.frame(tripModeProfile1_vis, tripModeProfile2_vis$freq2, tripModeProfile3_vis$freq3 + , tripModeProfile4_vis$freq4, tripModeProfile5_vis$freq5, tripModeProfile6_vis$freq6 + , tripModeProfile7_vis$freq7, tripModeProfile8_vis$freq8, tripModeProfile9_vis$freq9) +colnames(tripModeProfile_vis) <- c("tripmode", "tourmode", "work", "univ", "schl", "imain", "idisc", "jmain", "jdisc", "atwork", "total") + +temp <- melt(tripModeProfile_vis, id = c("tripmode", "tourmode")) +#tripModeProfile_vis <- cast(temp, tripmode+variable~tourmode) +#write.csv(tripModeProfile_vis, "tripModeProfile_vis.csv", row.names = F) +temp$grp_var <- paste(temp$variable, temp$tourmode, sep = "") + +# rename tour mode to standard names +temp$tourmode[temp$tourmode=="tourmode1"] <- 'Auto SOV' +temp$tourmode[temp$tourmode=="tourmode2"] <- 'Auto 2 Person' +temp$tourmode[temp$tourmode=="tourmode3"] <- 'Auto 3+ Person' +temp$tourmode[temp$tourmode=="tourmode4"] <- 'Walk' +temp$tourmode[temp$tourmode=="tourmode5"] <- 'Bike/Moped' +temp$tourmode[temp$tourmode=="tourmode6"] <- 'Walk-Transit' +temp$tourmode[temp$tourmode=="tourmode7"] <- 'PNR-Transit' +temp$tourmode[temp$tourmode=="tourmode8"] <- 'KNR-Transit' +temp$tourmode[temp$tourmode=="tourmode9"] <- 'TNC-Transit' +temp$tourmode[temp$tourmode=="tourmode10"] <- 'Taxi' +temp$tourmode[temp$tourmode=="tourmode11"] <- 'TNC-Single' +temp$tourmode[temp$tourmode=="tourmode12"] <- 'TNC-Shared' +temp$tourmode[temp$tourmode=="tourmode13"] <- 'School Bus' + +colnames(temp) <- c("tripmode","tourmode","purpose","value","grp_var") + +write.csv(temp, "tripModeProfile_vis.csv", row.names = F) + + +### +#trip mode by time period +#calculate time of day +trips$tod <- 5 # EA: 3 am - 6 am +trips$tod <- ifelse(trips$stop_period>=4 & trips$stop_period<=9, 1, trips$tod) # AM: 6 am - 9 am +trips$tod <- ifelse(trips$stop_period>=10 & trips$stop_period<=22, 2, trips$tod) # MD: 9 am - 3:30 pm +trips$tod <- ifelse(trips$stop_period>=23 & trips$stop_period<=29, 3, trips$tod) # PM: 3:30 pm - 7 pm +trips$tod <- ifelse(trips$stop_period>=30 & trips$stop_period<=40, 4, trips$tod) # EV: 7 pm - 3 am +trips$num_trips <- 1 + +jtrips$tod <- 5 # EA: 3 am - 6 am +jtrips$tod <- ifelse(jtrips$stop_period>=4 & jtrips$stop_period<=9, 1, jtrips$tod) # AM: 6 am - 9 am +jtrips$tod <- ifelse(jtrips$stop_period>=10 & jtrips$stop_period<=22, 2, jtrips$tod) # MD: 9 am - 3:30 pm +jtrips$tod <- ifelse(jtrips$stop_period>=23 & jtrips$stop_period<=29, 3, jtrips$tod) # PM: 3:30 pm - 7 pm +jtrips$tod <- ifelse(jtrips$stop_period>=30 & jtrips$stop_period<=40, 4, jtrips$tod) # EV: 7 pm - 3 am +jtrips$num_trips <- 1 +#jtrips$num_trips <- jtrips$num_participants + +itrips_summary <- aggregate(num_trips~tod+TOURPURP+TOURMODE+TRIPMODE, data=trips, FUN=sum) +jtrips_summary <- aggregate(num_trips~tod+TOURPURP+TOURMODE+TRIPMODE, data=jtrips, FUN=sum) + +write.csv(itrips_summary, "itrips_tripmode_summary.csv", row.names = F) +write.csv(jtrips_summary, "jtrips_tripmode_summary.csv", row.names = F) + +### + + + + +# Total number of stops, trips & tours +cat("Total number of stops : ", nrow(stops) + nrow(jstops)) +cat("Total number of trips : ", nrow(trips) + nrow(jtrips)) +cat("Total number of tours : ", nrow(tours) + sum(unique_joint_tours$NUMBER_HH)) + + +# output total numbers in a file +total_population <- sum(pertypeDistbn$freq) +total_households <- nrow(hh) +total_tours <- nrow(tours) + sum(unique_joint_tours$NUMBER_HH) +total_trips <- nrow(trips) + nrow(jtrips) +total_stops <- nrow(stops) + nrow(jstops) + +trips$num_travel[trips$TRIPMODE==1] <- 1 #sov +trips$num_travel[trips$TRIPMODE==2] <- 2 #hov2 +trips$num_travel[trips$TRIPMODE==3] <- 3.5 #hov3 +trips$num_travel[trips$TRIPMODE==10] <- 1.1 #taxi +trips$num_travel[trips$TRIPMODE==11] <- 1.2 #tnc single +trips$num_travel[trips$TRIPMODE==12] <- 2.0 #tnc shared +trips$num_travel[is.na(trips$num_travel)] <- 0 + +total_vmt <- sum((trips$od_dist[trips$TRIPMODE<=3])/trips$num_travel[trips$TRIPMODE<=3]) + sum((trips$od_dist[trips$TRIPMODE>=10 & trips$TRIPMODE<=12])/trips$num_travel[trips$TRIPMODE>=10 & trips$TRIPMODE<=12]) + +totals_var <- c("total_population", "total_households", "total_tours", "total_trips", "total_stops", "total_vmt") +totals_val <- c(total_population,total_households, total_tours, total_trips, total_stops, total_vmt) + +totals_df <- data.frame(name = totals_var, value = totals_val) + +write.csv(totals_df, "totals.csv", row.names = F) + +# HH Size distribution +hhSizeDist <- count(hh, c("HHSIZE")) +write.csv(hhSizeDist, "hhSizeDist.csv", row.names = F) + +# Persons by person type +actpertypeDistbn <- count(per[per$activity_pattern!="H"], c("PERTYPE")) +write.csv(actpertypeDistbn, "activePertypeDistbn.csv", row.names = TRUE) + + +### Generate school escorting summaries + +# detach plyr and load dplyr +#detach("package:plyr", unload=TRUE) +if (!"dplyr" %in% installed.packages()) install.packages("dplyr", repos='http://cran.us.r-project.org') +library(dplyr) + +# get driver person type +tours$out_chauffuer_ptype <- per$PERTYPE[match(tours$hh_id*100+tours$driver_num_out, + per$hh_id*100+per$person_num)] +tours$inb_chauffuer_ptype <- per$PERTYPE[match(tours$hh_id*100+tours$driver_num_in, + per$hh_id*100+per$person_num)] + +#tours$out_chauffuer_dap <- per$activity_pattern[match(tours$hh_id*100+tours$driver_num_out, per$hh_id*100+per$person_num)] +#tours$inb_chauffuer_dap <- per$activity_pattern[match(tours$hh_id*100+tours$driver_num_in, per$hh_id*100+per$person_num)] + + +tours[is.na(tours)] <- 0 + +tours_sample <- select(tours, hh_id, person_id, person_num, tour_id, tour_purpose, escort_type_out, escort_type_in, + driver_num_out, driver_num_in, person_type) + +tours_sample <- tours[tours$tour_purpose=="School" & tours$person_type>=6, ] + +# Code no escort as "3" to be same as OHAS data +tours_sample$escort_type_out[tours_sample$escort_type_out==0] <- 3 +tours_sample$escort_type_in[tours_sample$escort_type_in==0] <- 3 + + +# School tours by Escort Type X Child Type +out_table1 <- table(tours_sample$escort_type_out, tours_sample$person_type) +inb_table1 <- table(tours_sample$escort_type_in, tours_sample$person_type) + +# School tours by Escort Type X Chauffuer Type +out_sample2 <- filter(tours_sample, out_chauffuer_ptype>0) +inb_sample2 <- filter(tours_sample, inb_chauffuer_ptype>0) +out_table2 <- table(out_sample2$escort_type_out, out_sample2$out_chauffuer_ptype) +inb_table2 <- table(inb_sample2$escort_type_in, inb_sample2$inb_chauffuer_ptype) + +## Workers summary +# summary of worker with a child which went to school +# by escort type, can be separated by outbound and inbound direction + +#get list of active workers with at least one work tour +active_workers <- tours %>% + filter(tour_purpose %in% c("Work","Work-Based")) %>% #work and work-related + filter(person_type %in% c(1,2)) %>% #full and part-time worker + group_by(hh_id, person_num) %>% + summarise(person_type=max(person_type)) %>% + ungroup() + +workers <- per[per$PERTYPE %in% c(1,2), ] + +#get list of students with at least one school tour +active_students <- tours %>% + filter(tour_purpose %in% c("School")) %>% #school tour + filter(person_type %in% c(6,7,8)) %>% #all school students + group_by(hh_id, person_num) %>% + summarise(person_type=max(person_type)) %>% + ungroup() + +students <- per[per$PERTYPE %in% c(6,7,8), ] + +hh_active_student <- active_students %>% + group_by(hh_id) %>% + mutate(active_student=1) %>% + summarise(active_student = max(active_student)) %>% + ungroup() + +#tag active workers with active students in household +active_workers <- active_workers %>% + left_join(hh_active_student, by = c("hh_id")) %>% + mutate(active_student=ifelse(is.na(active_student), 0, active_student)) + + +#list of workers who did ride share or pure escort for school student +out_rs_workers <- tours %>% + select(hh_id, person_num, tour_id, tour_purpose, + escort_type_out, driver_num_out, out_chauffuer_ptype) %>% + filter(tour_purpose=="School" & escort_type_out==1) %>% + group_by(hh_id, driver_num_out) %>% + mutate(num_escort = 1) %>% + summarise(out_rs_escort = sum(num_escort)) + +out_pe_workers <- tours %>% + select(hh_id, person_num, tour_id, tour_purpose, + escort_type_out, driver_num_out, out_chauffuer_ptype) %>% + filter(tour_purpose=="School" & escort_type_out==2) %>% + group_by(hh_id, driver_num_out) %>% + mutate(num_escort = 1) %>% + summarise(out_pe_escort = sum(num_escort)) + +inb_rs_workers <- tours %>% + select(hh_id, person_num, tour_id, tour_purpose, + escort_type_in, driver_num_in, inb_chauffuer_ptype) %>% + filter(tour_purpose=="School" & escort_type_in==1) %>% + group_by(hh_id, driver_num_in) %>% + mutate(num_escort = 1) %>% + summarise(inb_rs_escort = sum(num_escort)) + +inb_pe_workers <- tours %>% + select(hh_id, person_num, tour_id, tour_purpose, + escort_type_in, driver_num_in, inb_chauffuer_ptype) %>% + filter(tour_purpose=="School" & escort_type_in==2) %>% + group_by(hh_id, driver_num_in) %>% + mutate(num_escort = 1) %>% + summarise(inb_pe_escort = sum(num_escort)) + +active_workers <- active_workers %>% + left_join(out_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% + left_join(out_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% + left_join(inb_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) %>% + left_join(inb_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) + +active_workers[is.na(active_workers)] <- 0 + +#workers <- workers %>% +# left_join(out_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% +# left_join(out_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% +# left_join(inb_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) %>% +# left_join(inb_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) +# +#workers[is.na(workers)] <- 0 + +active_workers <- active_workers %>% + mutate(out_escort_type = 3) %>% + mutate(out_escort_type = ifelse(out_rs_escort>0, 1, out_escort_type)) %>% + mutate(out_escort_type = ifelse(out_pe_escort>0, 2, out_escort_type)) %>% + mutate(inb_escort_type = 3) %>% + mutate(inb_escort_type = ifelse(inb_rs_escort>0, 1, inb_escort_type)) %>% + mutate(inb_escort_type = ifelse(inb_pe_escort>0, 2, inb_escort_type)) + +temp <- filter(active_workers, active_student==1) +worker_table <- table(temp$out_escort_type, temp$inb_escort_type) + +## add marginal totals to all final tables +out_table1 <- addmargins(as.table(out_table1)) +inb_table1 <- addmargins(as.table(inb_table1)) +out_table2 <- addmargins(as.table(out_table2)) +inb_table2 <- addmargins(as.table(inb_table2)) +worker_table <- addmargins(as.table(worker_table)) + +## reshape data in required form for visualizer +out_table1 <- as.data.frame.matrix(out_table1) +out_table1$id <- row.names(out_table1) +out_table1 <- melt(out_table1, id = c("id")) +colnames(out_table1) <- c("esc_type", "child_type", "freq_out") +out_table1$esc_type <- as.character(out_table1$esc_type) +out_table1$child_type <- as.character(out_table1$child_type) +out_table1 <- out_table1[out_table1$esc_type!="Sum",] +out_table1$child_type[out_table1$child_type=="Sum"] <- "Total" + +inb_table1 <- as.data.frame.matrix(inb_table1) +inb_table1$id <- row.names(inb_table1) +inb_table1 <- melt(inb_table1, id = c("id")) +colnames(inb_table1) <- c("esc_type", "child_type", "freq_inb") +inb_table1$esc_type <- as.character(inb_table1$esc_type) +inb_table1$child_type <- as.character(inb_table1$child_type) +inb_table1 <- inb_table1[inb_table1$esc_type!="Sum",] +inb_table1$child_type[inb_table1$child_type=="Sum"] <- "Total" + +table1 <- out_table1 +table1$freq_inb <- inb_table1$freq_inb +table1$esc_type[table1$esc_type=='1'] <- "Ride Share" +table1$esc_type[table1$esc_type=='2'] <- "Pure Escort" +table1$esc_type[table1$esc_type=='3'] <- "No Escort" +table1$child_type[table1$child_type=='6'] <- 'Driv Student' +table1$child_type[table1$child_type=='7'] <- 'Non-DrivStudent' +table1$child_type[table1$child_type=='8'] <- 'Pre-Schooler' + + +out_table2 <- as.data.frame.matrix(out_table2) +out_table2$id <- row.names(out_table2) +out_table2 <- melt(out_table2, id = c("id")) +colnames(out_table2) <- c("esc_type", "chauffeur", "freq_out") +out_table2$esc_type <- as.character(out_table2$esc_type) +out_table2$chauffeur <- as.character(out_table2$chauffeur) +out_table2 <- out_table2[out_table2$esc_type!="Sum",] +out_table2$chauffeur[out_table2$chauffeur=="Sum"] <- "Total" + +inb_table2 <- as.data.frame.matrix(inb_table2) +inb_table2$id <- row.names(inb_table2) +inb_table2 <- melt(inb_table2, id = c("id")) +colnames(inb_table2) <- c("esc_type", "chauffeur", "freq_inb") +inb_table2$esc_type <- as.character(inb_table2$esc_type) +inb_table2$chauffeur <- as.character(inb_table2$chauffeur) +inb_table2 <- inb_table2[inb_table2$esc_type!="Sum",] +inb_table2$chauffeur[inb_table2$chauffeur=="Sum"] <- "Total" + +table2 <- out_table2 +table2$freq_inb <- inb_table2$freq_inb +table2$esc_type[table2$esc_type=="1"] <- "Ride Share" +table2$esc_type[table2$esc_type=="2"] <- "Pure Escort" +table2$esc_type[table2$esc_type=="3"] <- "No Escort" +table2$chauffeur[table2$chauffeur=='1'] <- "FT Worker" +table2$chauffeur[table2$chauffeur=='2'] <- "PT Worker" +table2$chauffeur[table2$chauffeur=='3'] <- "Univ Stud" +table2$chauffeur[table2$chauffeur=='4'] <- "Non-Worker" +table2$chauffeur[table2$chauffeur=='5'] <- "Retiree" +table2$chauffeur[table2$chauffeur=='6'] <- "Driv Student" + +worker_table <- as.data.frame.matrix(worker_table) +colnames(worker_table) <- c("Ride Share", "Pure Escort", "No Escort", "Total") +worker_table$DropOff <- row.names(worker_table) +worker_table$DropOff[worker_table$DropOff=="1"] <- "Ride Share" +worker_table$DropOff[worker_table$DropOff=="2"] <- "Pure Escort" +worker_table$DropOff[worker_table$DropOff=="3"] <- "No Escort" +worker_table$DropOff[worker_table$DropOff=="Sum"] <- "Total" + +worker_table <- worker_table[, c("DropOff", "Ride Share","Pure Escort","No Escort","Total")] + +## write outputs +write.csv(table1, "esctype_by_childtype.csv", row.names = F) +write.csv(table2, "esctype_by_chauffeurtype.csv", row.names = F) +write.csv(worker_table, "worker_school_escorting.csv", row.names = F) + +detach("package:dplyr", unload=TRUE) + + +#District level summary of transit tours and trips +#segment by Walk, PNR, and KNR +# tour mode/trip mode +# 9-Walk to Transit +# 10-PNR +# 11-KNR + +#tours +tours$ODISTRICT <- mazCorrespondence$pmsa[match(tours$orig_mgra, mazCorrespondence$mgra)] +tours$DDISTRICT <- mazCorrespondence$pmsa[match(tours$dest_mgra, mazCorrespondence$mgra)] +tours_transit <- tours[tours$tour_mode>=9 & tours$tour_mode<=12,] +tours_transit <- tours_transit[,c("ODISTRICT","DDISTRICT","tour_mode")] +tours_transit$NUMBER_HH <- 1 + +unique_joint_tours$ODISTRICT <- mazCorrespondence$pmsa[match(unique_joint_tours$orig_mgra, mazCorrespondence$mgra)] +unique_joint_tours$DDISTRICT <- mazCorrespondence$pmsa[match(unique_joint_tours$dest_mgra, mazCorrespondence$mgra)] +unique_joint_tours_transit <- unique_joint_tours[unique_joint_tours$tour_mode>=9 & unique_joint_tours$tour_mode<=12,] +unique_joint_tours_transit <- unique_joint_tours_transit[,c("ODISTRICT","DDISTRICT","tour_mode", "NUMBER_HH")] + +tours_transit_all <- rbind(tours_transit, unique_joint_tours_transit) + +district_flow_tours <- xtabs(NUMBER_HH~tour_mode+ODISTRICT+DDISTRICT, data=tours_transit_all) +write.csv(district_flow_tours, "district_flow_transit_tours.csv") + +#trips +trips$ODISTRICT <- mazCorrespondence$pmsa[match(trips$orig_mgra, mazCorrespondence$mgra)] +trips$DDISTRICT <- mazCorrespondence$pmsa[match(trips$dest_mgra, mazCorrespondence$mgra)] +trips_transit <- trips[trips$trip_mode>=9 & trips$trip_mode<=12,] +trips_transit <- trips_transit[,c("ODISTRICT","DDISTRICT","trip_mode")] +trips_transit$num_participants <- 1 + +jtrips$ODISTRICT <- mazCorrespondence$pmsa[match(jtrips$orig_mgra, mazCorrespondence$mgra)] +jtrips$DDISTRICT <- mazCorrespondence$pmsa[match(jtrips$dest_mgra, mazCorrespondence$mgra)] +jtrips_transit <- jtrips[jtrips$trip_mode>=9 & jtrips$trip_mode<=12,] +jtrips_transit <- jtrips_transit[,c("ODISTRICT","DDISTRICT","trip_mode","num_participants")] + +trips_transit_all <- rbind(trips_transit, jtrips_transit) + +district_flow_trips <- xtabs(num_participants~trip_mode+ODISTRICT+DDISTRICT, data=trips_transit_all) +write.csv(district_flow_trips, "district_flow_transit_trips.csv") + +# finish + +end_time <- Sys.time() +end_time - start_time diff --git a/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R b/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R new file mode 100644 index 0000000..bcbbaad --- /dev/null +++ b/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R @@ -0,0 +1,100 @@ +### Paths +SYSTEM_APP_PATH <- WORKING_DIR +SYSTEM_DATA_PATH <- file.path(SYSTEM_APP_PATH, "data") +SYSTEM_SHP_PATH <- file.path(SYSTEM_DATA_PATH, "SHP") +SYSTEM_TEMPLATES_PATH <- file.path(SYSTEM_APP_PATH, "templates") +SYSTEM_SCRIPTS_PATH <- file.path(SYSTEM_APP_PATH, "scripts") +OUTPUT_PATH <- file.path(SYSTEM_APP_PATH, "outputs") +RUNTIME_PATH <- file.path(SYSTEM_APP_PATH, "runtime") +BASE_DATA_PATH <- file.path(SYSTEM_DATA_PATH, "base") +BUILD_DATA_PATH <- file.path(SYSTEM_DATA_PATH, "build") + +### Names +if(IS_BASE_SURVEY=="Yes"){ + # Surey Base + BASE_SCENARIO_ALT <- "HTS" + DISTRICT_FLOW_CENSUS <- "HTS" + AO_CENSUS_SHORT <- "HTS" + AO_CENSUS_LONG <- "HTS" +}else{ + # Non-Survey Base + BASE_SCENARIO_ALT <- BASE_SCENARIO_NAME + DISTRICT_FLOW_CENSUS <- BASE_SCENARIO_NAME + AO_CENSUS_SHORT <- BASE_SCENARIO_NAME + AO_CENSUS_LONG <- BASE_SCENARIO_NAME +} + +### Other Codes +person_type_codes <- c(1, 2, 3, 4, 5, 6, 7, 8, "Total") +person_type_names <- c("1.FT Worker", "2.PT Worker", "3.Univ Stud", "4.Non-Worker", "5.Retiree", "6.Driv Student", "7.Non-DrivStudent", "8.Pre-Schooler", "Total") +person_type_char <- c("FT Worker", "PT Worker", "Univ Stud", "Non-Worker", "Retiree", "Driv Student", "Non-DrivStudent", "Pre-Schooler", "Total") +person_type_df <- data.frame(code = person_type_codes, name = person_type_names, name_char = person_type_char) + +purpose_type_codes <- c("atwork", "esc", "esco", "idisc", "imain", "jdisc", "jmain", "sch", "schl", "univ", "work", "total", "Total") +purpose_type_names <- c("At-Work", "Escorting", "Escorting", "Indi-Discretionary", "Indi-Maintenance", "Joint-Discretionary", "Joint-Maintenance", "School", "School", "University", "Work", "Total", "Total") +purpose_type_df <- data.frame(code = purpose_type_codes, name = purpose_type_names) + +mtf_codes <- c(1, 2, 3, 4, 5) +mtf_names <- c("1 Work", "2 Work", "1 School", "2 School", "1 Work & 1 School") +mtf_df <- data.frame(code = mtf_codes, name = mtf_names) +dap_types <- c("M", "N", "H") +jtf_alternatives <- c("No Joint Tours", "1 Shopping", "1 Maintenance", "1 Eating Out", "1 Visiting", "1 Other Discretionary", + "2 Shopping", "1 Shopping / 1 Maintenance", "1 Shopping / 1 Eating Out", "1 Shopping / 1 Visiting", + "1 Shopping / 1 Other Discretionary", "2 Maintenance", "1 Maintenance / 1 Eating Out", + "1 Maintenance / 1 Visiting", "1 Maintenance / 1 Other Discretionary", "2 Eating Out", "1 Eating Out / 1 Visiting", + "1 Eating Out / 1 Other Discretionary", "2 Visiting", "1 Visiting / 1 Other Discretionary", "2 Other Discretionary") +todBins <- c("03:00 AM to 05:00 AM","05:00 AM to 05:30 AM","05:30 AM to 06:00 AM","06:00 AM to 06:30 AM","06:30 AM to 07:00 AM", + "07:00 AM to 07:30 AM","07:30 AM to 08:00 AM","08:00 AM to 08:30 AM", "08:30 AM to 09:00 AM","09:00 AM to 09:30 AM", + "09:30 AM to 10:00 AM","10:00 AM to 10:30 AM", "10:30 AM to 11:00 AM","11:00 AM to 11:30 AM", "11:30 AM to 12:00 PM", + "12:00 PM to 12:30 PM", "12:30 PM to 01:00 PM","01:00 PM to 01:30 PM", "01:30 PM to 02:00 PM","02:00 PM to 02:30 PM", + "02:30 PM to 03:00 PM","03:00 PM to 03:30 PM", "03:30 PM to 04:00 PM","04:00 PM to 04:30 PM", "04:30 PM to 05:00 PM", + "05:00 PM to 05:30 PM", "05:30 PM to 06:00 PM","06:00 PM to 06:30 PM", "06:30 PM to 07:00 PM","07:00 PM to 07:30 PM", + "07:30 PM to 08:00 PM","08:00 PM to 08:30 PM", "08:30 PM to 09:00 PM","09:00 PM to 09:30 PM", "09:30 PM to 10:00 PM", + "10:00 PM to 10:30 PM", "10:30 PM to 11:00 PM","11:00 PM to 11:30 PM", "11:30 PM to 12:00 AM","12:00 PM to 03:00 AM") +tod_df <- data.frame(id = seq(from=1, to=40), bin = todBins) +durBins <- c("(0.5 hours)","(1 hours)","(1.5 hours)","(2 hours)","(2.5 hours)","(3 hours)","(3.5 hours)","(4 hours)","(4.5 hours)", + "(5 hours)","(5.5 hours)","(6 hours)","(6.5 hours)","(7 hours)","(7.5 hours)","(8 hours)","(8.5 hours)","(9 hours)", + "(9.5 hours)","(10 hours)","(10.5 hours)","(11 hours)","(11.5 hours)","(12 hours)","(12.5 hours)","(13 hours)", + "(13.5 hours)","(14 hours)","(14.5 hours)","(15 hours)","(15.5 hours)","(16 hours)","(16.5 hours)","(17 hours)", + "(17.5 hours)","(18 hours)","(18.5 hours)","(19 hours)","(19.5 hours)","(20 hours)") +dur_df <- data.frame(id = seq(from=1, to=40), bin = durBins) +stopPurposes <- c("Work","Univ","Schl","Esco","Shop","Main","Eati","Visi","Disc","Work-related") +outDirDist <- c("< 0", "0-1", "1-2", "2-3", "3-4", "4-5", "5-6", "6-7", "7-8", "8-9", "9-10", "10-11", "11-12", "12-13", "13-14", "14-15", "15-16", "16-17", "17-18", "18-19", "19-20", "20-21", + "21-22", "22-23", "23-24", "24-25", "25-26", "26-27", "27-28", "28-29", "29-30", "30-31", "31-32", "32-33", "33-34", "34-35", "35-36", "36-37", "37-38", "38-39", "39-40", "40p") +tourMode <- c('Auto SOV','Auto 2 Person','Auto 3+ Person','Walk','Bike/Moped','Walk-Transit','PNR-Transit','KNR-Transit','TNC-Transit','Taxi','TNC-Single','TNC-Shared','School Bus') +tripMode <- c('Auto SOV','Auto 2 Person','Auto 3+ Person','Walk','Bike/Moped','Walk-Transit','PNR-Transit','KNR-Transit','TNC-Transit','Taxi','TNC-Single','TNC-Shared','School Bus') +sch_esc_types <- c('Ride Share', 'Pure Escort', 'No Escort') +sch_esc_codes <- c(1, 2, 3) +sch_esc_df <- data.frame(code = sch_esc_codes, type = sch_esc_types) +facility_types <- c('Interstate', 'Principal Arterial', 'Minor Arterial', 'Major Collector', 'Minor Collector', 'Local Road', 'Ramp') +facility_codes <- c(1, 3, 4, 5, 6, 7, 30) +facility_df <- data.frame(code = facility_codes, type = facility_types) +timePeriods <- c("EV1","EA","AM","MD","PM","EV") +timePeriodBreaks <- c(0,1,4,10,23,30, 41) +occp_type_codes <- c("occ1", "occ2", "occ3", "occ4", "occ5", "occ6", "Total") +occp_type_names <- c("Management", "Service", "Sales & Office", "Natural Resources", "Production", "Military", "Total") +occp_type_df <- data.frame(code = occp_type_codes, name = occp_type_names) + +### Functions +copyFile <- function(fileList, sourceDir, targetDir){ + error <- F + setwd(sourceDir) + for(file in fileList){ + full_file <- paste(sourceDir, file, sep = "/") + ## check if file exists - copy if exists else error out + if(file.exists(file)){ + file.copy(full_file, targetDir, overwrite = T, copy.date = T) + }else{ + #winDialog("ok", paste(file, "does not exist in", sourceDir)) + write.table(paste(file, "does not exist in", sourceDir), paste(OUTPUT_PATH, "error.txt", sep = "/")) + error <- T + } + if(error) break + } + return(error) +} + + + + + diff --git a/sandag_abm/src/main/r/visualizer/workersByMAZ.R b/sandag_abm/src/main/r/visualizer/workersByMAZ.R new file mode 100644 index 0000000..29f9615 --- /dev/null +++ b/sandag_abm/src/main/r/visualizer/workersByMAZ.R @@ -0,0 +1,114 @@ +########################################################## +### Script to summarize workers by MAZ and Occupation Type + +##### LIST OF ALL INPUT FILES ##### +## 0. Path input data : parameters.csv +## 1. person data : personData_3.csv +## 2. Work school location data : wsLocResults_3.csv +## 3. MAZ data : mgra13_based_input2016.csv +## 4. Occupation factors data : occFactors.csv +## 5. Geographic crosswalk data : geographicXwalk_PMSA.csv + +### Read Command Line Arguments +args <- commandArgs(trailingOnly = TRUE) +Parameters_File <- args[1] +REF <- args[2] + +SYSTEM_REPORT_PKGS <- c("reshape", "dplyr", "data.table") +lib_sink <- suppressWarnings(suppressMessages(lapply(SYSTEM_REPORT_PKGS, library, character.only = TRUE))) + +### Read parameters file +parameters <- read.csv(Parameters_File, header = TRUE) + +### Read parameters from Parameters_File (REF) +PROJECT_DIR <- trimws(paste(parameters$Value[parameters$Key=="PROJECT_DIR"])) +if(REF){ + WD <- trimws(paste(parameters$Value[parameters$Key=="BASE_SUMMARY_DIR"])) + ABMOutputDir <- trimws(paste(parameters$Value[parameters$Key=="REF_DIR"])) + ABMInputDir <- trimws(paste(parameters$Value[parameters$Key=="REF_DIR_INP"])) + BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BASE_SAMPLE_RATE"]))) +} else { + WD <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SUMMARY_DIR"])) + + ABMOutputDir <- file.path(PROJECT_DIR, "output") + ABMInputDir <- file.path(PROJECT_DIR, "input") + BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BUILD_SAMPLE_RATE"]))) +} + +MAX_ITER <- trimws(paste(parameters$Value[parameters$Key=="MAX_ITER"])) +WORKING_DIR <- trimws(paste(parameters$Value[parameters$Key=="WORKING_DIR"])) +geogXWalkDir <- trimws(paste(parameters$Value[parameters$Key=="geogXWalkDir"])) +mazFile <- trimws(paste(parameters$Value[parameters$Key=="mgraInputFile"])) +factorDir <- file.path(WORKING_DIR, "data") + +# read data +per <- read.csv(paste(ABMOutputDir, paste("personData_",MAX_ITER, ".csv", sep = ""), sep = "/"), as.is = T) +wsLoc <- read.csv(paste(ABMOutputDir, paste("wsLocResults_",MAX_ITER, ".csv", sep = ""), sep = "/"), as.is = T) +mazData <- read.csv(paste(ABMInputDir, basename(mazFile), sep = "/"), as.is = T) +occFac <- read.csv(paste(factorDir, "occFactors.csv", sep = "/"), as.is = T) +mazCorrespondence <- fread(paste(geogXWalkDir, "geographicXwalk_PMSA.csv", sep = "/"), stringsAsFactors = F) + +# workers by occupation type +workersbyMAZ <- wsLoc[wsLoc$PersonType<=3 & wsLoc$WorkLocation>0 & wsLoc$WorkSegment %in% c(0,1,2,3,4,5),] %>% + mutate(weight = 1/BUILD_SAMPLE_RATE) %>% + group_by(WorkLocation, WorkSegment) %>% + mutate(num_workers = sum(weight)) %>% + select(WorkLocation, WorkSegment, num_workers) + +ABM_Summary <- cast(workersbyMAZ, WorkLocation~WorkSegment, value = "num_workers", fun.aggregate = max) +ABM_Summary$`0`[is.infinite(ABM_Summary$`0`)] <- 0 +ABM_Summary$`1`[is.infinite(ABM_Summary$`1`)] <- 0 +ABM_Summary$`2`[is.infinite(ABM_Summary$`2`)] <- 0 +ABM_Summary$`3`[is.infinite(ABM_Summary$`3`)] <- 0 +ABM_Summary$`4`[is.infinite(ABM_Summary$`4`)] <- 0 +ABM_Summary$`5`[is.infinite(ABM_Summary$`5`)] <- 0 + +colnames(ABM_Summary) <- c("mgra", "occ1", "occ2", "occ3", "occ4", "occ5", "occ6") + + +# compute jobs by occupation type +empCat <- colnames(occFac)[colnames(occFac)!="emp_code"] + +mazData$occ1 <- 0 +mazData$occ2 <- 0 +mazData$occ3 <- 0 +mazData$occ4 <- 0 +mazData$occ5 <- 0 +mazData$occ6 <- 0 + +for(cat in empCat){ + mazData$occ1 <- mazData$occ1 + mazData[,c(cat)]*occFac[1,c(cat)] + mazData$occ2 <- mazData$occ2 + mazData[,c(cat)]*occFac[2,c(cat)] + mazData$occ3 <- mazData$occ3 + mazData[,c(cat)]*occFac[3,c(cat)] + mazData$occ4 <- mazData$occ4 + mazData[,c(cat)]*occFac[4,c(cat)] + mazData$occ5 <- mazData$occ5 + mazData[,c(cat)]*occFac[5,c(cat)] + mazData$occ6 <- mazData$occ6 + mazData[,c(cat)]*occFac[6,c(cat)] +} + +### get df in right format before outputting +df1 <- mazData[,c("mgra", "hhs")] %>% + left_join(ABM_Summary, by = c("mgra"="mgra")) %>% + select(-hhs) + +df1[is.na(df1)] <- 0 +df1$Total <- rowSums(df1[,!colnames(df1) %in% c("mgra")]) +df1[is.na(df1)] <- 0 +df1 <- melt(df1, id = c("mgra")) +colnames(df1) <- c("mgra", "occp", "value") + +df2 <- mazData[,c("mgra","occ1", "occ2", "occ3", "occ4", "occ5", "occ6")] +df2[is.na(df2)] <- 0 +df2$Total <- rowSums(df2[,!colnames(df2) %in% c("mgra")]) +df2[is.na(df2)] <- 0 +df2 <- melt(df2, id = c("mgra")) +colnames(df2) <- c("mgra", "occp", "value") + +df <- cbind(df1, df2$value) +colnames(df) <- c("mgra", "occp", "workers", "jobs") + +df$DISTRICT <- mazCorrespondence$pmsa[match(df$mgra, mazCorrespondence$mgra)] + +### Write outputs +write.csv(df, paste(WD, "job_worker_summary.csv", sep = "/"), row.names = F) + +# finish \ No newline at end of file diff --git a/sandag_abm/src/main/resources/BatchSubstitute.bat b/sandag_abm/src/main/resources/BatchSubstitute.bat new file mode 100644 index 0000000..99bffdc --- /dev/null +++ b/sandag_abm/src/main/resources/BatchSubstitute.bat @@ -0,0 +1,20 @@ +@echo off +REM -- Prepare the Command Processor -- +SETLOCAL ENABLEEXTENSIONS +SETLOCAL DISABLEDELAYEDEXPANSION + +::BatchSubstitude - parses a File line by line and replaces a substring" +::syntax: BatchSubstitude.bat OldStr NewStr File +:: OldStr [in] - string to be replaced +:: NewStr [in] - string to replace with +:: File [in] - file to be parsed +:$changed 20100115 +:$source http://www.dostips.com +if "%~1"=="" findstr "^::" "%~f0"&GOTO:EOF +for /f "tokens=1,* delims=]" %%A in ('"type %3|find /n /v """') do ( + set "line=%%B" + if defined line ( + call set "line=echo.%%line:%~1=%~2%%" + for /f "delims=" %%X in ('"echo."%%line%%""') do %%~X + ) ELSE echo. +) diff --git a/sandag_abm/src/main/resources/CTRampEnv.bat b/sandag_abm/src/main/resources/CTRampEnv.bat new file mode 100644 index 0000000..5f3e826 --- /dev/null +++ b/sandag_abm/src/main/resources/CTRampEnv.bat @@ -0,0 +1,61 @@ +rem this file has environment variables for CT-RAMP batch files + +rem set ports +set MATRIX_MANAGER_PORT=${matrix.server.port} +set HH_MANAGER_PORT=${household.server.port} + +rem set single node index +set SNODE=${snode} + +rem set machine names +set MAIN=${master.node.name} +set NODE1=${node.1.name} +set NODE2=${node.2.name} +set NODE3=${node.3.name} + + +rem set IP addresses +set MAIN_IP=${master.node.ip} +set HHMGR_IP=${household.server.host} + +rem JVM memory allocations +set MEMORY_MTXMGR_MIN=${mtxmgr.memory.min} +set MEMORY_MTXMGR_MAX=${mtxmgr.memory.max} +set MEMORY_HHMGR_MIN=${hhmgr.memory.min} +set MEMORY_HHMGR_MAX=${hhmgr.memory.max} +set MEMORY_CLIENT_MIN=${client.memory.min} +set MEMORY_CLIENT_MAX=${client.memory.max} +set MEMORY_SPMARKET_MIN=${spmarket.memory.min} +set MEMORY_SPMARKET_MAX=${spmarket.memory.max} +set MEMORY_BIKELOGSUM_MIN=${bikelogsum.memory.min} +set MEMORY_BIKELOGSUM_MAX=${bikelogsum.memory.max} +set MEMORY_WALKLOGSUM_MIN=${walklogsum.memory.min} +set MEMORY_WALKLOGSUM_MAX=${walklogsum.memory.max} +set MEMORY_BIKEROUTE_MIN=${bikeroute.memory.min} +set MEMORY_BIKEROUTE_MAX=${bikeroute.memory.max} +rem set MEMORY_DATAEXPORT_MIN=${dataexport.memory.min} +rem set MEMORY_DATAEXPORT_MAX=${dataexport.memory.max} +set MEMORY_EMFAC_MIN=${emfac.memory.min} +set MEMORY_EMFAC_MAX=${emfac.memory.max} +set MEMORY_VALIDATE_MIN=${validate.memory.min} +set MEMORY_VALIDATE_MAX=${validate.memory.max} + +rem set main property file name +set PROPERTIES_NAME=sandag_abm + +rem all nodes need to map the scenario drive, currently mapped as x: +set MAPDRIVE=${MAPDRIVE} +rem set MAPDRIVEFOLDER=\\${master.node.name}\${map.folder} +rem uncomment next line if use T drive as data folder. +rem !!!Note: much slower than a local data folder!!! +set MAPDRIVEFOLDER=${MAPDRIVEFOLDER} + +rem account settings for remote access using psexec +set USERNAME=${USERNAME} +set PASSWORD=${PASSWORD} + +rem location of mapAndRun.bat on remote machines +set MAPANDRUN=${MAPANDRUN} + +rem set location of java +set JAVA_64_PATH=${JAVA_64_PATH} \ No newline at end of file diff --git a/sandag_abm/src/main/resources/CheckOutput.bat b/sandag_abm/src/main/resources/CheckOutput.bat new file mode 100644 index 0000000..18b2c00 --- /dev/null +++ b/sandag_abm/src/main/resources/CheckOutput.bat @@ -0,0 +1,8 @@ +rem ### Declaring required environment variables +set PROJECT_DIRECTORY=%1 +set CHECK=%2 +set ITERATION=%3 + +rem ### Checking that files were generated +python %PROJECT_DIRECTORY%\python\check_output.py %PROJECT_DIRECTORY% %CHECK% %ITERATION% +if ERRORLEVEL 1 exit 2 diff --git a/sandag_abm/src/main/resources/CreateD2TAccessFile.bat b/sandag_abm/src/main/resources/CreateD2TAccessFile.bat new file mode 100644 index 0000000..fffc318 --- /dev/null +++ b/sandag_abm/src/main/resources/CreateD2TAccessFile.bat @@ -0,0 +1,9 @@ +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call bin\CTRampEnv.bat +set JAR_LOCATION=%PROJECT_DIRECTORY%/application + +%JAVA_64_PATH%\bin\java -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -Djxl.nowarnings=true -Dlog4j.configuration=log4j_d2t.xml -cp application/*;conf/ -Dproject.folder=%PROJECT_DIRECTORY% -Djava.library.path=%JAR_LOCATION% org.sandag.abm.application.SandagMGRAtoPNR %PROPERTIES_NAME% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataExporter.bat b/sandag_abm/src/main/resources/DataExporter.bat new file mode 100644 index 0000000..7bf4bc1 --- /dev/null +++ b/sandag_abm/src/main/resources/DataExporter.bat @@ -0,0 +1,33 @@ +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call bin\CTRampEnv.bat +set JAR_LOCATION=%PROJECT_DIRECTORY%/application + +rem ### Connecting to Anaconda3 Environment +set ENV=C:\ProgramData\Anaconda3 +call %ENV%\Scripts\activate.bat %ENV% + +rem ### Checking if Data Exporter environment exists +rem ### Otherwise creates environment +set EXPORT_ENV=%PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\dataExporter\environment.yml +call conda env list | find /i "abmDataExporter" +if not errorlevel 1 ( + call conda env update --name abmDataExporter --file %EXPORT_ENV% + call activate abmDataExporter +) else ( + call conda env create -f %EXPORT_ENV% + call activate abmDataExporter +) + +rem ### Running Data Exporter on scenario +python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\dataExporter\serialRun.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem ### Check for the Data Exporter output files +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% Exporter %ITERATION% + +rem ### Exiting all Anaconda3 environments +call conda deactivate +call conda deactivate \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataLoadRequest.bat b/sandag_abm/src/main/resources/DataLoadRequest.bat new file mode 100644 index 0000000..74be0d6 --- /dev/null +++ b/sandag_abm/src/main/resources/DataLoadRequest.bat @@ -0,0 +1,7 @@ +set PROJECT_DIRECTORY="%1" +set ITERATION=%2 +set YEAR=%3 +set SAMPLE_RATE=%4 +set ABM_VERSION=${version} + +sqlcmd -d ${database_name} -E -S ${database_server} -Q "EXEC [data_load].[SP_REQUEST] $(year),'$(path)',$(iteration),$(sample_rate),'$(abm_version)'" -v year=%YEAR% path=%PROJECT_DIRECTORY% iteration=%ITERATION% sample_rate=%SAMPLE_RATE% abm_version=%ABM_VERSION% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataSummary.bat b/sandag_abm/src/main/resources/DataSummary.bat new file mode 100644 index 0000000..46f0427 --- /dev/null +++ b/sandag_abm/src/main/resources/DataSummary.bat @@ -0,0 +1,14 @@ +rem forced to update excel links for auto reporting, YMA, 1/23/2019 + +@echo on + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SCENARIOYEAR=%3 +set SCENARIOID=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\database_summary.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% %SCENARIOYEAR% %SCENARIOID% + + diff --git a/sandag_abm/src/main/resources/ExcelUpdate.bat b/sandag_abm/src/main/resources/ExcelUpdate.bat new file mode 100644 index 0000000..4c893e9 --- /dev/null +++ b/sandag_abm/src/main/resources/ExcelUpdate.bat @@ -0,0 +1,16 @@ +rem forced to update excel links for auto reporting, YMA, 1/23/2019 + +@echo on + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SCENARIOYEAR=%3 +set SCENARIOID=%4 + +@echo path + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\excel_update.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% %SCENARIOYEAR% %SCENARIOID% + + diff --git a/sandag_abm/src/main/resources/FHWADataExporter.bat b/sandag_abm/src/main/resources/FHWADataExporter.bat new file mode 100644 index 0000000..7b38dc0 --- /dev/null +++ b/sandag_abm/src/main/resources/FHWADataExporter.bat @@ -0,0 +1,142 @@ +set PROJECT_DRIVE_BASE=%1 +set PROJECT_DIRECTORY_BASE=%2 +set PROJECT_DRIVE_BUILD=%3 +set PROJECT_DIRECTORY_BUILD=%4 + +# ********************************************************************************************************************************** +# STEP 1: +# Create base trips with base skims +# ********************************************************************************************************************************** + +%PROJECT_DRIVE_BASE% +cd %PROJECT_DIRECTORY_BASE% +call bin\CTRampEnv.bat + +set PATH=%TRANSCAD_PATH%;C:\Windows\System32;application + +%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter + +# +# copy reports directory +# +echo d|xcopy report report_basetripbaseskim /S /Y + +# ********************************************************************************************************************************** +# STEP 2: +# Create build trips with build skims +# ********************************************************************************************************************************** + +%PROJECT_DRIVE_BUILD% +cd %PROJECT_DIRECTORY_BUILD% + +%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter + +# +# copy reports directory +# +echo d|xcopy report report_buildtripbuildskim /S /Y + +# ********************************************************************************************************************************** +# STEP 3: +# Create base trips with build skims +# ********************************************************************************************************************************** + +# currently in build directory: rename the disaggregate data +# +rename output\airport_out.csv airport_out.csv.build +rename output\crossBorderTours.csv crossBorderTours.csv.build +rename output\crossBorderTrips.csv crossBorderTrips.csv.build +rename output\householdData_3.csv householdData_3.csv.build +rename output\indivTourData_3.csv indivTourData_3.csv.build +rename output\indivTripData_3.csv indivTripData_3.csv.build +rename output\internalExternalTrips.csv internalExternalTrips.csv.build +rename output\jointTourData_3.csv jointTourData_3.csv.build +rename output\jointTripData_3.csv jointTripData_3.csv.build +rename output\luLogsums_logit.csv luLogsums_logit.csv.build +rename output\luLogsums_simple.csv luLogsums_simple.csv.build +rename output\personData_3.csv personData_3.csv.build +rename output\visitorTours.csv visitorTours.csv.build +rename output\visitorTrips.csv visitorTrips.csv.build +# +# copy base data to build directory +# +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\airport_out.csv output\airport_out.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\crossBorderTours.csv output\crossBorderTours.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\crossBorderTrips.csv output\crossBorderTrips.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\householdData_3.csv output\householdData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\indivTourData_3.csv output\indivTourData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\indivTripData_3.csv output\indivTripData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\internalExternalTrips.csv output\internalExternalTrips.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\jointTourData_3.csv output\jointTourData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\jointTripData_3.csv output\jointTripData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\luLogsums_logit.csv output\luLogsums_logit.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\luLogsums_simple.csv output\luLogsums_simple.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\personData_3.csv output\personData_3.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\visitorTours.csv output\visitorTours.csv +copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\visitorTrips.csv output\visitorTrips.csv +# +# run +# + +cd %PROJECT_DIRECTORY_BUILD% + +%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter +# +# rename report directory +# +echo d|xcopy report report_basetripbuildskim /S /Y +# +# ********************************************************************************************************************************** +# STEP 4: +# Create build trips with base skims +# ********************************************************************************************************************************** + +%PROJECT_DRIVE_BASE% +cd %PROJECT_DIRECTORY_BASE% + +# currently in base directory: rename the disaggregate data +# +rename output\airport_out.csv airport_out.csv.base +rename output\crossBorderTours.csv crossBorderTours.csv.base +rename output\crossBorderTrips.csv crossBorderTrips.csv.base +rename output\householdData_3.csv householdData_3.csv.base +rename output\indivTourData_3.csv indivTourData_3.csv.base +rename output\indivTripData_3.csv indivTripData_3.csv.base +rename output\internalExternalTrips.csv internalExternalTrips.csv.base +rename output\jointTourData_3.csv jointTourData_3.csv.base +rename output\jointTripData_3.csv jointTripData_3.csv.base +rename output\luLogsums_logit.csv luLogsums_logit.csv.base +rename output\luLogsums_simple.csv luLogsums_simple.csv.base +rename output\personData_3.csv personData_3.csv.base +rename output\visitorTours.csv visitorTours.csv.base +rename output\visitorTrips.csv visitorTrips.csv.base + +# +# copy build data to base directory +# + +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\airport_out.csv.build output\airport_out.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\crossBorderTours.csv.build output\crossBorderTours.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\crossBorderTrips.csv.build output\crossBorderTrips.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\householdData_3.csv.build output\householdData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\indivTourData_3.csv.build output\indivTourData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\indivTripData_3.csv.build output\indivTripData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\internalExternalTrips.csv.build output\internalExternalTrips.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\jointTourData_3.csv.build output\jointTourData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\jointTripData_3.csv.build output\jointTripData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\luLogsums_logit.csv.build output\luLogsums_logit.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\luLogsums_simple.csv.build output\luLogsums_simple.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\personData_3.csv.build output\personData_3.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\visitorTours.csv.build output\visitorTours.csv +copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\visitorTrips.csv.build output\visitorTrips.csv + +# +# run +# + +%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter + +# +# copy report directory +# +echo d|xcopy report report_buildtripbaseskim /S /Y diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll b/sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll new file mode 100644 index 0000000000000000000000000000000000000000..747073f1b192ee74ba8612cb7ea58d908e6e7734 GIT binary patch literal 898048 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%0DI0m1qIKnk`}jJOuz?W;@^j zoP><=13im?>jiWIF2@B3?H<6-^Z{1D4FmzLzy?HZ!?p+z2l}d(9Rffg41|Cfz%+C9 zdI1bX0R-p>SnLcS3c3wto{^L0kQ64Oe>4Ys0!NT6!9+BfEZCD+gMh>q)H031L2ZS{m&8M9$zVvz8xJu;oDe(23UP_%vn1z*L?J;)7%B!OA#tclC|G7FD3OKa zL>a0GsX~1=U(_{^EA2u}NFQ>WVmw_)8}b?qAs*ipGKR!lz68*l1XYDO#8fhLF_R@^ z4cS6MbJV9-T1|jBY19~PI+fiKB0ktde5RLgQ^rlMkXmBUge3xnJJd5bSP_XYB?jX2_-@q4K0j^`VyfC4PhanA{`o+WJ8%yF;oiWLS#-p zR0#QukjX44huBJ)+^sS2E1^cn!mftup;{;-Gir3qW~db!(w6KRNyaR*ki(r&Khz7g ziEAoQ-wh3fvocE9Vrv?xVS1Pm9t@lLMwUo46dn$jWUvZw%T+QJpf-rh&b(mM);IS} zBjJizp&bp6JLHyO?O3?W9S<|ZAuGjNw?r&d2|JvTx-8tVfM{-8+&GUD=7ojfmZD=b zG2>#XIx4NmC1JZx6dqQK!)1BYDh?%VQ1JKc7y|ATi6ozgiYbBrfJnnt75UZU{P`GTC?87Dl56lAwe)K5p_A?aFqjv z!Ei-s;f#pQ9EUg(P6{l7oGujBGo#^H*v20bNVygs9BzqA7J&o_Gfjmshn*mr;$&ED z&xT`$YB(L{GO@5!lnLYERJasY$a7v)*tMbIs&;@gDvlEsGQ`P;%i+AP&t*qU)<(F; z?T3{XQe+@PkF>%=nu2aP(hW0+=2)Z{iExxd5gFI%sE270fY}U-t-UZ=VbE5H%e1y) zJW}G2gse!LSjaUP>4ZlkV-ZEft|#P8L>2+GR$JX9jF{E(h(g7SsEK~LpjSq4y*46< z82REzR2DZ#BNje4!jEtwOhqcvxV5< zVT!mSBUmIG zNkys}p(zqcM3i!ewHe7o$ie}3jN^%@h~njBujdD38UDrUfLak56I8Y{IA@Dvru{e!iS5i%O&NsGTj)D546kDyodC zqnfBT8k6vpByrIMo3)M_ah+GT>!SK-Tw;hCqvoh73Rpl{R;ra)qPFOO!bY%WrSKBbRFnv26(WfsG*G9t zvJ|30kS-eoM?nU$kpkrmfhNP4y(+IM`5+gxnOKB47PFKhwnN5&O^}Vkktlee7}Us_ zHUY>6g&=6J=+t^X@c||WRiMV6uo*!$r~tio9cY#rKr2|%nTg(AsId^myB@TGU8aU8 z-=&I}jA|YeOF$DS*721X$0gRvWuRJS6-Yrhn3IWx5EurlJcrItH1gw2Hj6mufFo)L z=mJ$D7>t5GFai!+Lm&t`g*v%N;|0fU0niDSbsjJXx*T$8--3W+#sVl2#$`!x+^!at z!924_^!7NI1s%3DIHtqE5|{xE))YvV_CcJ3f;lh`X4N5e9qbZC{-ACk)&QGe9BlE# zCWW*i;y9`xIfk)Ps+K{;ZQDs!x3&ZJK(>k$YY_rc!^H&Sv7uN~iEtnpCpHor(GJFf zT5F6JqsGQ!j2NBh^hruej2UCa)LcFxx&<+DOrzrx^*$nKso61!K@^k6q%lQI9h1ax z5yKz?t2PGFLrfFX61BfRW;fYm7Gjr>E#}obVl_=pO;tK$ZbAUm zKu;{K_r{#;s*cJ@@Ia|sOn0OdIPo#&BW!?J6CsR1I2MX2bwOs$4#pU)cq~S!0w6YE zfeAa1VI^X0GZN!j4RVf3FYnm#SWB5A6nd70#UKN*2qk8eyNubGOWIZzV(A!Nkc&BO z#Tc#Xkd zRdI~3RP{ufxR&q?IjK=E;_Krh8eN>{fDNX&C2ov+t>(BjuC+Vkj(A-&YM=`3gn?k$ z*fv)@&#H@Hi97Cz`{Ei|AnuO?aT_N@sE9~B9G40ib}OI8#`zEzwS$DOfZ{H8G%k{; zv@t?Zu}blx zA|H>~t8q%T9IwUG@k(4Ot`j<=Z)?U|abh1qT+5|OiR3Wei%VD}NMa&G6z%{-fvAM8 zU_dm;V;X`631Kk|jY4A(6B>tFA~wW=bX<=@s^dUHrC8EpaUqutG4mj;j$_k^WD1Lh z>mJjhY(69)e1;ejLFy2VBZNZCA*0#IvdbU|6w%O>!^TmQ0vfT~1WH15_>C$;afF$5 zxq+}8IlTs=@yw78(nBW52vJRHl}e@HSs*K9gB*|@^0NXEYIG60!w1O~e#i?crB28V zsZAz%kZ>R&h$bBuH*{ghqK!f%dj!H%Fq9xvh*yI^aR?-A2n3-}ir7Gf5n?0@<)A!- z6C3#ogd5S>t5BH`B`~*dHJj^%AZb8Ns0+2AHq?Q7P#+=@L(Ktr-00L%;Ie^hqQW$I zNJr;rMRb?}55j|zVZw=wz@so`9D~PUtips@gc%u=aA3b7X0fvZEH2E4d2mrDQi`Gi zSY;K$dTzidg2k`|j#zzGxfWN+U|43BD`5pJhkHf?cTlK@HLya#wyI$*tRn=90XD+p zT7;X_5I<|M4NfbxdOPf~SYRvcf}OAf_P}n~3;SUoY%>#AEA{{^&;xJ~juLJq0*45} z5+nQy1PiP%T(lyvU4t7ke2FFji}_Av5|)Wma2lowePRqIu3N-*9>roY1auKm2DZrA z#KDj*95CkK0-O=%;kvd23piOAH!@jSvD;CE%do{9R94{{T!HPJ4%~+6(k|S9dvFs5 zjCI1&lMxaES^98>NkIk>tDK5xRCI)fFc1{R;UQ!YVMsB%TRMyg1WF#Ahe)_K7D9~8 zRHMi^Vv#8oW5@_HWU$GlrhvGqV!3-*e0)aG2#4e982YDK#snsF|<*0SQq(k(G zPh+qf5d%_?nvo*YhFB0U+b*yodQn-FGTRZa#fi9-4wjfb!gL@m#EmrM9>l@%A-EvL z^CLBW0O=}i`na+t1`t@2QH78Q5=6oXSrA2Nf;7)Gg$(Q32*?R35~ePKsvAfL=^-64Eiq=I8C$v#pPCp*@Qi%~)O2)hU0p0BT+vWs zG%=PKPK+c56Z8Zr;V}EyyaYQTAmmZSK(|UH%!G*0NSwrYf|W3FWtyInY?LP`Mo1}1 zIE;#f#~jv(6Vik%p-gC{dc7_|96S?zhncWSmV_Z;OxP0YgwJhCXcE?hJ7MNJ63&Dx z;Z1lFb)zrQwdc&CL?Gc$xRu1(wFHoeB_fGvLZ1K=@dT96o8bg(wetrBNP=rdiA#n= zf@7~5^eV2wCQ~voLPjMMB(6)2C(uMHkx%3j*+hoWQpJQ=SxQtA<%C-(RMiq)U6l}2 zJablM5H%7;dn?gSbQ0Y}QqoH}1UPSqM@m+#Mp2&_)u*gd1IL8O29h4BNtBXcJieKn zG^o^;hBz&WYU#5LRYU&L(AY zk{KQ|$x8NgO2{%9CM!)|gW{?FCsrX-xK&+N3FINw&21q&ev|Teuc0$h0MUc5AZ0 z%IY12?s6srNl&t4FiU+&H{rZoNv(`(vC6#3crucV5rFv7S|jV-<0 zF=))BJn|t^Hif6`T(1z7EX3g-Bc&lOVKoP14}zVL^=JGPmwfH z)0DJRA{%}%FGSqsC&N^{ek zwALc{ud7pu(h@@1Nz>xA!XQh_(`>dbtt8ByI!#h*(;C9v5g#0BjRIqv(gFLhbX0~* z0vdDLl4h~2Xv_0erc0I znoSRK*vw2ipSByc3O}1}i3%e8TpDK&a7(Om8n;K)T2ntAV0Y8)bSYgRq#z>A@*5hL z-o>h>E9q9cmTsn<)wlt!q* zqRJ;wu^7a7vZNBD5;VvjV++wSl#j|#0m?(`X%0G$I=L)VO6Wp4%0#;c%qBt|r~&OL z^t>ovjfUA?xeg79T{<(GH>=PP!ORCu7F2;6Q8%hVwWwHcB8(!#VnyRjC1Dh8XvW}0 zV`!fPqfXR^=4~MAvklpsnhd9GN}wb?iy0*pV_D@v!)P2GF$^&Q6hfP(2pT|r@^RUS z6tEghX|#xj&<3X|Pog+lQKirv8q?%azYRf!91P8(Bh~_nq8W71LUL5lHd;nCQWt7B zldUbZNodF#%4apu66({|(LR%`)|fje2@4wgs9$c94q$RV1#6jls2}a34hLCZF%4rZ zjEzyTx?vonV}lq6V_+j#nU&O!5(1Kj(NseiU7}Mk1yyUyT9yeg9%grNv6_vK2?;AH z#l+ZvUV_<-V_1bq2h>HWx=;fKQsH?^k z+betvX2tRv8`jo~nF!mBc`zclkT@{4!HK!BFy_MmLT38000u~dm`dXps5uc#!GQ>y z8OOv%5Q||(;;ItHdg?5OVi|S{8?vOa1ePS^W{z;37=~bgriu+&%UFdFo<*#L6|j_2 zKe2i?F-c0N%se7)RT{Py-#8*l0KvAHx|q6HnPV_z*sfkK=5d zg%9E*xQtMoTw-6K7?jjQkwrpJlVBCaE}XI5Y$tS3|E z712R6?vjZ(7TnDW$gQ|fWWyn*%w)%Lg%5Y*1&s%H;vnCF`*FSiR7Y?C=cxmD2zTLO zJc<{LK^(;6II$ELS21IFSX{S5geI-2$aVzh*%EjXPnc7_*>hMr+$5TH~+tGHPlU!#~T5Vwfi#hSdrNv5LGs)p3Ilo4g! z;x2JVnN-{+mWXBIag%z)?CknnxtLp{5^Ke3ab~4L%+Jz@bz+lP51g(6^hQ&2tzwI~ zt1@1~o7gKJ4)kqFW?_dhKT#r^G(s{`l0TW&FiMgnNy{5886z1h87E1Wj4d87Ns&yD zj4Vr)Bv++L(j^&^Oi4pobAFaYoIRc~Q8GzVP?;?$l;lWqC6gt2lCr9NNr8kaDUu|E zA2QvQZ8@B5dSz=Gt)f&?mr+kGkyJ{I#lq@xNrhxQcrYd`Z!J<*PGe~q2y~GIW=@!tZ zHcLzrYe~Fxgmk#HSCSx21Ri;$luJ#Lj**U%CQH-v#!1IZCrHzzsZww{6EMoU>MUTD zCjzsaBh8jp6ik*%vvQ^R(gG+9GY0O0q{xFfLCnD8Y!DtQ;^D;B1nsu4F?)KQ8u|Op;(xbBpWSD(Bu`3k*QOw z3nxr!s2M98S86Ea~vNB}SOj(93OE#%S zP&U45qKuadP9g-ZJHE6hg_oTJe%;ELI9Zk}(^Rt3vug^nSfvFrs;p2pytG(WB&#bc zk%5CpxsBz@VpXoBpj1{SYc434Rmdu3@|uZxRkA6vhHP~i_DN- zRnR1BmhnsKt1MZ3nE;4%p{%v6IK5LQ0=c&~8C~8k6VnU(Ju)e<>I#`s)(yP6 zCbdhJTp^M1^Bc1(szh03dAViDg&JVn)w1I1NhMm~+s73eWd@mEHlfO#XOgu}VWpN; zs)`Mn7Fl!lxXEUjv_Mu6FHex0^M}h5<#k!5m2J5R^j?`Er+_|EK0-c9Zk2VH)t7Qg z#>tC|ljNi2$rH!QQzng(3-XQWDe{6GW8svN$?4yj)%>ug|EEo2pszDe@Y5 zmAqO`lQZPCa;CgqUMFV*HP4Y}mvmP1> zB{j6MWfu8JTI!_68e>)lEoov~QGDk}#kh2N0XscOFyJtxrfcv7fe!& zpef77rV6Q9xf7>QQ>8R<&16MssW~@Sk*BCG$XCc{*$S$nNKvQ&KiU)}3R7WQTB%}E zb(x}EF@~PnSfQv?Oi|DjRf=jwje@Qyq3WwksSJg>0xU$KuC8V(>J%&mC!MX}6>t=Z z6S;~eMT4SIp-$&1niVaIij-CbUm?mDD9STMpgho~P>P$Tv@1Fkor*3+x1vYUkRet` z6ba=r#pE=lLZL`ZEzByPl2bA~ho9Y<*HoZZs1!vtT^Y=ZrZP(bjn+6(r_d^@s>YQu zQuPXrqPDW9#-K1N%!;0(#`LCilcGB#ZBkE0K`EcsG|8f{DtZ<1$^_+brM@yzIRZ45 zM=3`ulaynWnu_X<(FJLlW0m8SnhAClT$R4Q>uFNla;O2Gbk)Wreb#SX@@Aw3b&FaZ_bk@+@^RO{vI|r`0H{Kw)8ulB=XE8OmBEQ^``+DS3JI zN{%u;ldUutrcdH2Yl<3_$z_d7zOqHxtZY>_fi8nUDN+_^2|=l$L)ir?(cMb1vPWsj zmMEpl+;o{zu2dE%luD&asaCdSYm{20PN`QKl&KXa<;Y62(g^Cq7G*C$FT+&{szlWY z)kxJSP;MBl8lxHu>J7=N@v6+`2`YmqBQridEhkl#q8d@1uF3#ahfHv2FH2?496_I` z%2t6x8>+EWCVgr>ixpK3w02d8s#Dda>Q+rI=}}FrkgLQhi7K_Kq?%EtP$^aUWipji zH73odGN`nvI+a$XR+-aPsItSv?-KD8{HYU{NGNNrpO8ovuz(Pg19bsp$cRMID_ z#YNJb0yR}#s4h|$t1~8*s7uuoa?8}^>I!wGx>`L&olCD$%V{<0MjB1sTTWNks+nqr zTAE!`Ev(YB`OW64v1u$dBeP!3R&&(aY_6KcZ>$y;@zjm#26dCVS>2*;RrA#Xb(>nK z7OC6S9qLX{#z-t}p46kRDHN+E>i9~jTBe>TY>-G5a?s3Bsnu$YT0v8)wQ3zGXy`#d z!>l%{vuGBzRm~Fis>j4@;uZ1gcy0WY_^Nnad{ca4{D}C1_>_dygtUb8gp7pDgsg;# z36m1C6Gp`+#S7yF@uK+A@%s3}grbDvgp!2PgtCP4go=d9geeJC3DpTT3A6-y0yyzM z0i3*?z)Aq8Lnd$%xCy+3h6GwXHJ%=y5}y`7K3im#g4+M)%d{{wR0mea zguvLq8krCn8(1k5`rnxMA7Y_@kO|QH4{2aK8QkW-4FbxAK76PK(*#BW#sVS!ef>Er zxULlvq`M#? zY%P=$_i-`54Pdp%Mc2C73}AK9wbEjBw|1$4C5a16VhJZ;1lLw{-4mH%n?>~sZ+Zr$~Fmu7If%^>G;HmO2;XjIx z!9|H*!(~g(!Q&oXh421&8%9Atz&PqXxV`mPIP2CcSZ{<7l_(VAyCAWQ}G5LCnhgwMGYA)u^6#F0KjXyP{_;^Q_W5JB4! zP|8lk0qQsOYb25F+V{>P@W?a``#e# zhe62pFM^T3Bf^l{VS|t#;u4XHoUusiqzTC5BeRgg&I06(nhNCXxN79~GA8m=XajQL zfi@(gq60a}BS99d)F6EcMr5R~7a5%IMy4^RA~y}6gyaI+wj%G1 z+=C2p?MLEf9!35->I`yu{S{>4u^Y&*{P&R*=mC=IeuNYVe@9kg|3KpV-Xbpq5UBhQ zp{VodC{*l+;V9&WQK&-Gcofx~i7GA6MImhYCoQP2@H3Ri4HDf>OBft0BzEHn)Tac80y5$B_zKbN6w zt!q$)&^pw1Xd{X=cMB>S+J>^V?Ltuzdr%jG_oE=k0TdfPib63?pkVCRDAJHKC=&V{ z>Zt27iUeIjji6pfF-W&ip@sKQftp9CVCt_Z!nzlzn9tv!s%JsyBk?G-Js692pCF=< z5DCr1M4%yCBpMYGi^f8O&_hsh=wN6#y0T&vx~V@IygdQkADMw}$LFBqhUcNfU4`iM z#BwweUxl7uUWZ2E*ys^Ojp!t(8SU#AprJ|;8cXj%ucpY+n=TvBDc5_^-h3|_<(Y~W z44Z>q)4LSyDO`o_KfWHl3%wP+ma`Wf9sd=2XyQ?H#h9br}^A$~+7<~>5kJD#BVq~FoSh(FK}>Lt2_@)x-Of1|-twrCXoE!culd_ZGj zQJAAp5M~M@1Owp-7$}y6p+aF86orB*rAA{$LxV884vxeaaN{sz(P@~CU*%#}9xlaL zs8txFvkvn%yb&|xkr4B`N{Z<|sl~kh-GP~c@?aw3r(zHv=3<0Tmt)3KKf|P7*n}xU zY{5XJtr#-21A`d!1?DvG5N0KG1Op+CVvy?xFoom3#n>t?U|!6;i9sR0$3PGm?i~!8 zbRPrFeu$}ne#Xp=eu9DHe#4*`&oQ=)R~RVmFHCjUd(7iXOwggO(4gvQN>I`G*r2!# zLxWDmCkB1V7#pr?x0(@ zwL!6mdV~J)`-1MSo)NUVc5V=&eo@frw3R_+`5S_!&fXSuc*y>snZ?I~ke#Q3qSMX= zC8b;nI+A`pC=7cmC}HFMAn3@0p!S?6K|juX9fUyr6+|2NA!vjLjm&I~C5y-XfM`tA1o)*V=(iS}(Y|%Ovjd#%)$;7EXKmn5^M#u4149(dMvEpisd}nhb=mB5Ie+w z0_zl?#U@o=#0D2$#TH$1Hh2|jLa?nWBlrY9J2?0l!xBtI_XT5Nd+@jtPjE`lv|x1X ztl)KV^Mfhq#lh$MR|m628-wGa&x4`Vt-;YtcLzT^b1*n=)zRSl!c)Q3q;G@wz?Xxk z)!htsF1{DMbKft)_O#!EO$%QKyKrxUe@;i?5cnY6*(;&A=zcP8)v6erqJJoEXkY{m zb$2xG%JEcOEh-bYxHuPgEWQv|Qc;Q{)lI?8M$mB7tU6pFf{i=)M;k8q^KRU;M@rlg zs}Xl;tsPf%%!8}CGy_+#W&v*7g_XFKv1@UwGCs$ZJ9gl%BlqH%Cl2BUwV%Yzy>u3r zGjJI<7`lcVpx(q`h<9+5^oKaQ;3;m<&Q~}e^e67Zk+--pE8&oZ8-hYkt|EqrBO^l? z;|GWAb|-{PJu*53g&7x;{&!XgYUtz;8M-K>c6CJvgsKdgq^k`Hdcq6Yr)~?ulyroo zMTkhv4jK*9U->So)FZgsUeAj`$LwVUKm28tq9?dS{u?IyfFlFZVoAf zwuPYVJ43?wUxe66UxpA^heNQz$3lqE$q?MxGa(TA+YktUJ|qQoDWqiOwUDXM%@9}T zy$~Psr;wPylaLnA%aG`)Z$fzE5qN$f22YxX!yj5o!3(-#@q_(wcxYh)9t$PnA^S-D zV(eIaG?a|*FHFS;Q8V$Y9^~TDXeyo{D#H^~tMN6&Iy|O=gU5Xk;L{>I@f*fS@#EbZ zdvSJJj_AZox~JmPpc!~5au%M;UWo5UF2%>yuEZzBuEj&Pb@;@vP567! z+wmidzrZt_58*cvPvTdGp1~(=y@WTv`2i1=-os->5Ajt0FL>L)Q+&yh=XiAgD}3RO zfAEQ}5BL*_C_+Xoj*!DA5upBX!rY_LgfhnvLJky9AViNOV2+L@42O~lq(x~2#@&g8 zu!vj&hgw8%KqZ7&R0RQts3OoI1|h{?Pat1!B1|a}5DRsvLJBcKot0`$y7zy<{fkbgP>THQ|oYZ4O>!{-nn#9RU?av`AzT1=pl zmJ%T53IZRon(&gji2%iYP9V9r5mFqx2vFD_0-|Ui0b6{4u*Y_ca8GfH5V!aoff{## z0Anu^pv-H8_P2KlM`L~>pgw;>@F1QMC~dC@Ug&Q^3G@%a-478<5J+MZ6-#vE2*jW? zGSQYEO@vS}#3TEL5#!E{AYM2!hM00>JkbVCAY##J#1Vu^#C}{ZkqYGzAw)hAk1Zsk z+sla%wSpKgsv#l@8N@CGiwHq1Vj{#QT3pRU1k^%|`=Ol};}H`psd6F~s|43q5nYgm z2*>G&kjp?MkxayIi|oX2U@vi!Vmfim)p^9Om?gxSL)H)(hc*+#^LG(njXy}dVLw4U zaqcWJJn;fC34571Uj03BbK-qsM%6DwJmLv)$d#8w#F)Q`;O8810rZ|2hJ`~IUsIbsdYi#I^%Hg4vm{Fk+H7RtKcS30Oh|JIv*K3AG{A=h+e($J!CvhW{dz zjM^V+n{_CZ6n7-lK|LOdL3|xb9eg?zVtpIxL|zL0<-x7c5%G6JZLS|f(;xf__V*<8 zND26c{nMYJNZdc6Bkd6BW_S<@#s-t%ZX&6FK?Dh=Mv**!4I|auA4Rf5qe(>Ecv9`U zOj0bIO?qOdlAy0kNQJHn(!MRVq;X&HNb)spq(W~Oi8WnLLR4u;f-obg3t=VkzjBi} zaZ^bv88b=gsJWyg8y1mv@BNGf9=MXi;ff*ec4k}XA{kRyi3CRFM}XG06)J zJTew)AfqE&$-`4cWNJ|-*@o>V)844bt1$*L_&Xwp&9;#daSk$SiI2=2Ka*UHnM;Nk z^U0+CMdU(g33gWtky9{t$Tr#o@(KJeL?enTnZX$pun~!%84m$l(@Scls2fFG6RxQif~FweTtqk zt*4hl40cl1-VIPNqi0f16wRejDi%_9*_KoCZEGkgyFRBN;4PHF>OGXXl>HO}|1c#E ze~dy6K1qp9IZdG+`Ib_<<}zjOk?WMx)9+FS&<`oc=0Bm}*F2}d&Lb4EYKaJxiXs*;#1R+At0RhIj1kx(O9bMQJ%YN-8$n$-HDW>BtcYON{D>ON z(g+CpugB(E5m8jXHXjGg{uZ%)%j*ap^akwz|2-FyK13{mP?5t?L6Hyw8;ON+k=PPKBx+$;Bz1FSByRfP z$fW+bNXVNQ3ExeMgixa+Q=xH@bEDED;~f(tW1yT!7AZe+=eW|ySVUzc0xSiMq}E3A z^SF^HUt=U5+Y$*Gg^@{-U6Hx_m624BHWGWr9H~(`BW?Yj$e~am68g8fJ|oh1Z*HV_ z>C(sy>dMG))Mt^Rt)EBE8nrW0{OGI5e}4KVawhfL$d0F$z%<76oC$qax#@qE^PmMxmfVQ7FWaC@5}d6dH<)f_g_r zvEVUL<04a{Hp7`wt}!`LZc=^}Wm<7m3c5V%_&?03;x8MbPD3qG{QkBmh%bstqjp6> z;XP6PHfdB!h$_lnt&d_srYHo}5;Yp?i|QHT1^f3!*${y!$Tk&x%xO`ik^NElhCsO|? zMl=gKC%S~WFq$)VRdjgdx@gsgEzxk;_UOvdUqnYk`=Tp?4@6^0hoT|=k!XqU>uA}m zbI~)=SEIvIZbbKI--)LF^)Nc6@^Q4y_cVGC^c>vBe;xPB=pyKKG#C0i8d~@^8ub_v zLs109Bt_z5h6u+}U z7sg=xWii8#Oo_oGYGR;ZdW@aQj7g5;#H^gw6q7-01N->zCSDW+hjqq;*?MAPk1ArS zpBQ3hV=XaM$QlE=Z84j_@yB4Ir^n>qofi|1UK9h_7RQjFB{5j)vY2zW&thQd#uyUf z^O!=<_82yHPs~twf6OT8Pz+<@$(TIsnV7V@7h}-0?_!R?H)C9=+c9?NP7D>k2kygv zJS$^=is369Wyri($fWEK(H|OYXSqIU*J{YD_EyfvHXg+y8muK1+*5#AL;8U6~ilA6XO&-KmTvJ)^}& zm^rcW_Qu!?Ic>4F!uHrCs52Hx>WPIaB(cz0MJ&apjy+E?$DV+CW1+ad*tir&Ec_oH zBX!1_?#zt+@L>g*ZD0;5N8F|M?_AACG_jXKU+RM|^|W2i-v2Mr?y_A#NgC zKDdx>4+*lO z$W7quG7ZcOZFq4)}-v^54ZzP#F9S8U?4q>F^}D2&Tf7a25O) zR0rRII^p;Kxv*2tv51tGc!{u-(d=a_~ zU4pJbSD=e91FnXzLM)gIv*8A~3FgBsun2C0Z$dYq@1a|e1ilO_;2u~7%iu2fJNOoS z1HKAt;M^X=D>|`Gc15x;dWREzk&XS-a`LCQuqq2gvGELmc!lf zb@+SuCVUOn!tbCDkRCR{y|4xTU;h_I&`>lq4NnWF4WlK{5@|`av9$5D3A7YiCXGrf zq?ORhX|=QlS`$r3>!NkjKv9Zjr1jCJ)8^9_(3a7*(Y~Y|r(K}kq}`(3rQM_5r#+(m zPJ2%KgZ7H{j`p7Rfd15254f;q(Z4Bt42ALmy91rSs|O^i29hdNzGBJ)cgc z7tt%|mGmllHJwRk(b;q^y_GJc_s}JD8C^-&(+zYZ-9)$2d+Bz%gYKfc>3({EK8-$u z-cO%RpF^KZUqD|(Urb+4UqRnV-%J0J{uTWI{UH4i{RsUi{UrTs`T+eD{S5sq{agBZ z`X&1J^t<%?^k?)J^tbeP^bg>`4$MF?f*3dkojmP-dVgrl6fc{!1sH;gC3eZ%W%&;?@ z3>U-02ry_KQ_vGYn`>O+NrfOY8Tcnt6f$5S?!kEeYN{*57Zv2JzRUNcA)lr?Zw(lwU=wJ z)?Tmuq4s|5!`k0!U)26l`?B`WT9}DoqM6}LGBbi1$&6+WXC{GuSQ;}EbiqoQmCSl3 zhsg!?ttL>-;xh%z4p7F@g9et7X=6H=Zf1ZvjX8%ok2#;YfVq&ln7Nd>hPjrxj=7$> zjk%rq1#>TRA9FwRE9ODw5#|Zz*USOtS>}1>MdlUeHRdhm_so0Dhs;OJUzm@XPnge` z&zUcoubF=`p*mz8t}dhwUl&$Ku8Xc4R5!S8Xx*^75ubWmqd`S$TwO9KW~G5XRxapb z<%2p_H7H&2>Kf|=pjg!o+Eh|dq%zj^);a6Eb-ub8b#vig>L^^ST^y{~>o{mlAV^|R~e)X%M7Q2$x|n))sEJL?bEAFV%L|8@PT z`ZM)s>(A9+tiM!$x&FKQ>-D$lf2hA-|4aSr`VaMJHkOTJ6WBy{D4WC%V~4Y2*yBER zgEH9@K`Ur7I~SCJ3P2C25Hx~HKpm)(UB#{e{T~)6{P5WVb{kvB?f`wBk3AkGTLtPn zF1DNPV^3vIW6xtRVlQSdV=rf~V6S7ZXK!F{WPi@y%-+J@!QRE*%RanEPuZ{8f3jf?f)m8SaqyflP6Q{G zGl(;U69<|+37;xGAG-UhF(-rbv7%GJDdm)bB2Epc-?V*f)^NH&g+|Jeff9`Z z^l8+fQlkY08Y3vrm^l_sFUQVta;9b2e}`ayE1J zaSn36;S6xja?W$Ea&B{e;5^{`#QB-?3+E~48Rr$}HRmtRJI;H~2M&^p;$peM+Te& zKR3Xg#+}Zc$?fOP<<93Wm!oYgp|ac<-M#tn@d8#gy@Y24Phqj6{B-p2in z2O5tw9&P-#@qFX8#_t-hH{NKx*?70{_r~XqFB{)A;+vwHhBgfYO^cDBOEI=-9OzP{ zG^I6Vg3bgF^dXu+5kd$m5nZ4NA#GALX+Yh<)MRe5fSQB1$=4KUn%OkFX<5^Xrj<>r zn?7q=*R-i=XVdPcJx%+YPBoory3};L>AR-inw~ekXnNTMH)EO!&EcT*kO0aLBb!Gx zr-H^qc5_~H!KZpd8E7=rgBAl1lp0!_`Jk@Q)+_?G1ZlIP*$8R}z0G~i_GV|ZuQ|{> zyLnFY+~#@Bi<*}-uV`M`yuSJK=Izb9oA)(;*?heDo8~jk7n;vCUvIwAe7E^Q^DoVR zHUHfVwFI>gTEbf>EfFozEyG&kTE>I=L0U^@OMXigs0UPnz5u(03)%xspe)b@N&#Zf z3i#LrP_)=urnbyzS>3XxWo^rbmQ5|2TXwYUYB|_)sO50W*DYsT&a|9sx!7{4u?h4_JRH z(EO!9_1CmAT3Nu~H@5Oyg{>kW>g7Pue>C)lR#U4L2zh_&^wycJ^IMm;E^A%gy1w=E z)-|o0Ter1tZ{5+luk}RhH?0G$r&`anUTD48da3nt>y_3YTJN>qZ++DIOY7s-C#_Fg ze{21{^+oF+t*=|(w7zW(;^X)PKABJ9NAaWivHW2`-zW3a`I*4Qmjm@)3Cw#fp9Mra zhu;j`dM95AlzJcE!T0ce`~ZIje-?iZe;$7!e;I!{e>Hy%e=UCle>;B%e=q+l{vrM` z{t5m7{}lf;{|x^e|2+Q+|0@3m{}%r?{|Ekk{zLu?{ww}p{I~pfe1srKfD_;a;7JET zxPUC62qFbhf*8SI!BD|)!5E;_Q-Dp+6nu2&*@D71J&8lhHb5L$rt?GrkLE}>T#5cUh_3g-zI2^R~O z3)cwO3AYHh3bzRl3Xcj;3I~MegcpUEg;#`Eh2IPB3Lgp|3!eyI2>%ei68$mWY;#R)|)KJ`=4IZ4hl1?Ghan9TFWDoe-T8ofe%D zofVxET@YOqT@`&NdLViz`bG3u^i=dr^h)%n=pPX{?W-Ntj%^Qa$F+yHliMlnk?k?< zgW89-4{IL|%jKBszRw)^WhLrgWqNznTryDixU2vW_Vo)g84R zvwY*W_K*=SlqFs zV`;~Vj+Gs2I<|D|=-An@r{k-R!yQLDj&&UG80a|LajxTh$CZv-9d|k&b^O}#q~o`a zHy!BCpiW{ZxihA7aAzD4qv^nQmVUCF)t#))dY~>lIwinSYC3hD=1zO3t8+%@%+6Vz zb2{gDF6>;?xukP@=Z?;uoqIaJ>O9bSu=DTEBb~=Ok9VHz9OyjNd8zZe&KsTIci!v# zsqCQ@W~wV&rsjfmRfE z$-3mgB`Se6)OHzxIWz-1=;(5FdAfXE^Sc&yE$&*{wW{m0uC-koyEb)g?b_M(RoBU` zfv&S%=esU+UF^Etb*<}0*Y{nwyB>Bu>U!MuYuA&mr(Mswo_D?KdfoL;*W0dlU6}6R zZd^C1JFJ`B9oZcPOkz54hLgG{15a4mT?s@W2S`BSC)3vfj9+)RTl_p0u--RrtHbbsExrF&cV-tK+f`@0WxpX?s!zSw=O`@8P@-4DBe>;Ap_ zdG{aPue#rNLp^X0vM01BtS6!;wr5CB9PoP|rC%CQeG`G{%Lj(9w5P16vS&&U4S2j- zAoJ>h$ZPBo^>p_1041mH>Fu%iID0%j-X4F?)Sg*AOL~^|Ebm#-v$p5+o^3tbdv^5f z?m5_VsONCc*FC3t271o+Tx#-t{2FNHInn zBn}pbib>)yag=zFcqou}Nx;~R6HoZ0=q8H`#AU$0)c~=^270X-NVHBM(qv)}O`Azad@=}7521#*JqBK+*CM8Q_ zf!-P=O_Fv>KB}!$AhdFU%;J11^p->3rz|=|bry>1OF} z>0arV(u2~&(i74F>1pXX>3Qij=`HDP=^g1^>3!)#=`YgXrO%`>JsD?6mA#**V#H*{`yvvfpHX%l?tQlcD5TIZlq36XcQd zXnCxBkbERiJ7a*)N%$#VK9TT=&A4Nu?&Q~rbe zk^C3=Gx;m|-}1NecXCL9R$vrC3W6d^5vNE1?j-|6=d%^LpOnhKsFW%oPwGD@6PiM# zXa|Z!24sj1cn`b6t?(%3D&{HXD;6jgDwZhLC_YzgSL{&iR_s-rR$NqER$NhBRa{eC zSA4H{pm?J|Dp5+b5~B=KhA8pMP~}i%oN}x(Svf)ZQGQGUvZGX4rYr|mqY6k3Ch!>z zz+toilhFk%g-z*JdX>|Zvz3dLOOz{>tCic7+m$<&yOjHthm}W^N0ld(XOw4^mzCF) zHtRkzTRI#cds$sxkj8#nl#$pn16@{uI;3rCfkD#jEiq$nhC@_IX;QqTl_-GEg)gP^aOf6R{fhaJlEoz(Eu6C;3YL9w` zx?epYWyND!I z>htPL>dWeD>Kp3Y>U-)R)jz3!R{x@YseYyYQ~kI4jrt$;JM{-Oq=7XkO^^ns3Dd-C z25AyCBQ;5y(V8)uv6^w3WX*WZ1Wk%2RgDSEC%+@T_EYd92EY+;gtkkU1e5P5a*`V2| z*`)bgvsJTAvqQ5}vq$r#<}1x%&4A{V=B(y|=CbCx=BDPB<_FCk%>&IN&1224n%^}q zHE%TkXx?f*XfWDvZKO6z8>fxeCTd4%M{CDuCup;@+1fm9zP3P1)fQ{ZwG~>LmZ4>8 zIocL&t5%?uYn57~)}%FSEn2I#S8Lb0wf)-J+Bw>J+C|!>+SS@M+TGgy+QZr-+N0VN z+OM@|wdb_wwclyK*WS_I)!x%S(EhA_qWw+#yY{*EZ!M%l=#V;;E=Y&f;dFRis4iSb z)MC_r zx_TW?*Qyig+I3Q$OefbVbsC*r=hjWrP1nuSEzvF2Ez_;gt<-`+4?+vzP?CbsxQ+|(bwo{db*yeXX`ooCVh*( zRnOPA=|%cZeUDzESLjuGwO*q)>8*O3-lduN|C9c){)zsX{<;38{xAJMdb9y!2sRK5L_?Ti zh#}rE+>l^MHjFo<7%~kL4aJ5ML%E^CP;Hn;}YXi<0|6@<3{7>#%;zO#@)t!#;=UWj3OlM8!OczbJOpi>zn0_@qF+DfE zH2q_GXL@f6GE>aM%n9aE=F#Rc=CS5v^LX)Jj*=ayu`fRywbeZyxF|XyxqLhyvO{7dB6E9^8xcg z^D*;D^MLuZ`HK0P`8)Fs^G)+D^CRO8mQYKKWsoJ#GQu*_l4KbNa{B3(3`>?J+md4`vrMtnTWT##OPz&hX|Oa}nk>zh z7K^|lv~*g!EmDijqOhneMvKK_wRkLE%S_8$%W}&~%R0+;%NLftmi?BmEC(!yEhjAB zSOzSoEZoMzb>j~>g>(|z^ z)^DxntlwF0SfSqg)`!-gtdFg~TAx~K)gc+&iH+y*I13ptq#Aw70yM*30N+_Huh$dj-91z3si?URkfaSKVvq?d`SqdV0OR zf!^u8vwG+CF6>?0yR3J4Z*Je@zP!HtzJk8uzLLK3K6)Rsudc7Luc@!OPtYgq6ZN(C zb@WO5lzrB|-oCy*Ti?{a>3uW%`uk?}&F@>#x3q6n-|D_~eVh8W^zH6D(091+NZ;|k zfxdHn=lgE;-Rb+W?_uAgz9)UZ_5I%WqVG-LyT12*ANqo9SR2kpun}#cwlEvnMzKZO zhS-MMhS}n5iMElpaklZc6kCQZ(>BqTZOgIc+VX8wTZxTt6WKa!Qk&AIwb^Yx+YH-M z+j84F+j`sQwtcoQZ3DJ%ZRc&5Y*%boZQt3h+wR()+J3V=v%RqWVS8=++xEux-iEY? z*dy&x_GtTH`%rtFJ<&eOKF*$E&$Q>*bM2+}a{Cl}wVi3_*!gyWz0=-h@3u?ra=YGc zxA)s!cDLPQpK70BpKo7iUt(WwUv1xN-)Y}%|H^*Ge%yZ2e%gM{e%^l7{+<1n{kHuF z`#t*u`>*yV_Gk9z_80b7_Sg2m?0?(e*+1AJ2htJhAUi0I7)PvQh$G%H+>zu+b7VMj z9L0_jN2#OCQRSdH=#E+k+tK9UJA{r-ht#2Qm>pI}pTp*`JDd)e!|U)n0*)DuS&q4m z`HqE-#f~M8m5x=8&m3zU>m3^$8y%Y*TOHdSI~}_ndmZ~72ONhShaF!#E;z0_ZaVHd z?mHeh9yuO6o;rSaymtKQc;|TUfSj;1$cc60oFPt@bC7e0bEtEeGu}DeInJ5v9PdnV zraLD(bDX)(JZF)!*jeH%ch)#*&RS=^lkMa<6;6ZG>g;pcoDQeU>2XeX&UVgq&T}qw zE^;n$E_E(*u5+$;Zgg&T9&#Rb9&vu-Jmoy^yx_d(yyCp=taCncK6bu#{^k7J`PK=$ zFfN=c)D`B6aK*T`I48KWU8SyaSA}betHwol)w-B2wu|fHxf)zeF1}0TYIk+IdRz*Z z+NE*nTn3lLWp&wIZr5zr9M?S8a@RW72G>T{Cf7FC4%cqie%C?QA=eStG1oV)0oPgA zdDmsv_pYB^k6pjJp1Gd8Uby~rrMcg@{&Br?A>2rJkQ?tNx=HTA?xF7C?gV$DdxU$G zdyIRmdxAUNo#D=M7r4vZbT`9Y>t?#^+-x_`-Q;d|x42u~e7C^e<`%idZn@j&Ho47i ztGmx_ce~vlx8FV0JxQoJ$Vq26KMcyEF? z$vfUV!JFz$^JaTyz{IHDVsELp#>@3Kdj(#Rx7{oED!nSN+3WPWyl$_@>-GA) ze(zN8G;hCmj(47Sfp?*Ik$1UwrS~)MdhcfM7VlQ?4(}fCA@5=DQSX5Fl=rmvocF5t zn)inHruTdA9q)bbkKPB~$KGGPPrc8*e|Z1$zVUwW!ak%A;|uoTd?7xPFUA+{Bl{@6 zXkV;vkZ-VWsBf4r$(QUK?@RHe`7(VuzFc3Suh>`WtMFC&rueFSG+(W+-pBTFeGR@Q zU#pMr6ZyJ*N}thZ_St+3eT#idd`o>Rd@Fq$e4BjxeFuGqd`EnzeP?`Uec$@d`7Zh{ z`7Zlz`F`;I?EBsK%=g^)%J-)a_GA16KhYoRkMKwOqx~`dSU*rH{*nGs{v>~jKh>Y+ z&-7>ebNu;!s=vrz>M!$G`sseApW|=vH~O3W0)Lxd=okBCeuZD{*ZNIl|2+Q^|5E=7|4RQ_|91aw{~rGr{=NPK{ww~g{%ij4{MY>t{lEBs^*{AL_y6I4 z?f={V&i}!W3ZMf)fzUu$AUr?`L2TB9w zfyzKlfEK6=umhX`H_#Ml4hRD6fzE(7pbO{&#(*hc4)g~40=9rN;0a6(%nbAgW(Vd6 z76cXrRs_}uHU_o^wgq+tz6k6Md=)qpI3D;qFc3HuI32hcxE#0=xElB_a69n-PvY>h2z^oMl(Js-c`uiy6T z+uphDkGK8lwm;wY_uD?Wt#!R^y<`3K`i<*1t$%bKw?4E^UFWO|*JbPSb$DH|u3T5I zTi0#t_I1y?VLiAWTQ987uFtP8tS_&xuAg7OxPHs}`ugVj*82AP&idi{(faW^cZ0Vf z+7NFjHdGs$4c&%m!?I!9uy43F+#9})z=mfdwh`Y*ZlpId8-s@vp`Ai@zx%woqH`TbQj6Z*^^5vvuv(xviVFy0`FKJzITS zq^u-L>`5*27z0*!s%WV_T1JJ-zkZ);G4kwe`Z*cecK}_0rZGTR++Q>DJG;e!KPV z)}OZiy!E%Oe{P{l=n|&XSvps`q10a*D2v(E6Q%n~50pMvda(3x>C2^W zmcCMYqV#0xsnRp0XG<@YUM~Hh^h)W6rPoTYm)7S(!N{BML z+*)ocW6Gz>UFEaoYs%M_&y}w)-&nq>++FS~6UsehQhA^}TplUY%FHsm%qjEA{Iakt zD$C2Nvbt<4+sdx8r|d2J%8_!soGKT})8&QoVtJ{2zI=1}mh!FT+segqxm+!8mv_p0 z<)iZL<-5vvm+vY6SNY!ZedYVh50pPwez^Q-`Ag+5m%m#6TKS3cQ|0fLzgK>_{7U)N z@{h_tEx%R%MfsQIUzdMV{%!ep<#)>OmftJ?wfwj8Kgx&-vVyL(R@y4<6>J4p>8x~B z&Qz|dTvIt$xv}z*N>8P?(pMo?`YXegkqWiKsIV&R3b(?m@GHWKxB^vV6?sKjQB|}R zL&aDzRV)>E#Z&QD0+nzjQb|-&m2@Rj8Lf;}#w#U#L7%d9?DS%9kr&tvptFyz;fm z6P2ebPgkC)JX`rj<(rjnRlZ$$q4M3zOO=-^Kd8J?`C;X?%IlRkDsNVPQhBTLv&zpa zzpnhY@=oQimA_R!s35ASDyDj>db)bHdUf^2>PMz3XH>+FKN_D5YSFKl%s<&7FvwCOsuIk;@&r~0*K2&|U`bhOt)yJz(R-dXq zU45qd_3CrgZ&bfo{Z{q6)mN%Ntp2F_diBTEx2r#^{+aKEg@b=Z);9nB9Ke|1;p+m3C|wtqXc z9odd;r?+$4h3(PpvF(ZN$?c2VH*e4HEbOfAT->>|v$3Dci!0f>CW3b zAKATWm$2Kr+qX;H9o(huGIlw;++E%-e^b57i#7eWCWn+9S0u)xJ{uYVGmbleMR6&(ywNd#?7)+P7-ouDwuuvG%>%%e5cW zUa7rS`%&$U+JD#Hto^k1yV|?8Ki1x>{iXJPt$h!-cirBpy$|i3-MeP*#=V>Ny7zka zhK=WMvB%o$+Y|4>d#XL{o@LLvXWMh`x%NDJ-o5BvY%j5w+{^Ce_dc`t zUwilN-M9DP-skteu=mK`qkCW7``X@9d(ZBDbMMW)pX|N0_x9e;_I|ndhrK`Vq4wMN zG5e?Yuin3A|JwcQ_ixz0asQ+Hgni;ZX`ixB+o$g{_L=*vecnEQU$8IShxVoW%6-+o zdf%{b+_&$$_dWZ*egA%NKeQj-ukSbZkN5A|zi2DK1N1@1LFYl&!Rdo* z4sJT=J|Gx&!@z@xXFmKX4ud4nBQw&%wP1 zpFMcs;K7534jw-E;=!W_Upe^d!Q%%{96WvS?7_DWUO4#9!Al3EN#i?;rf*;DZB19a%@!Tk7a~YaLU^);sE*^;7i^)w}9v z>etk-tKU$+vEE%L)+zO&I-}04v+A5Wuggu|#uCF`l&bq7as|V|mdaRzPC+q2Y zww|l!>xKGgeX2fFpR3Q;7wR9YuhlQsZ>ir}|Bw2|>$la{>znnhda1r$Kd3kANA>YuNFq5f$7OZBhRzgmC1{!IPZ`nT&Z)L*Xup#EC@_4-@&pVfa^ z|5g2W^*_|#tN*3`*ZTYQf7CyyBMwoA9fu!2Jbif0;km=>4zEAF@$jZY!eP%L@sM=b ze>iYRKBOJ84%vsCL(!r1P<{v>Dh@S=+C$@^>Ck*=JG37<58a2p!@yziFmxC@%p7J9 z{fD{3^x^2?_~GQ?%;DVO(&5{OzdwBM@UMp*4OHWFa*grEY-6sm)L3q;Hg0ZQY~0$ot+ClC zHY$y3W2dp(s5SN)w>LiBxTkS%7z48S07z-bluU7M;|#_ zJX$(hK3Y4vaCGz0#iLt}{^O{4v~{$5w0BfLIy$=Z=yN*vEpE*8z ze9iH-$JZTSe|*F7M~?}|#AEU?<#^aqu{OoH;HWj~nZ+?k%GM_Iaj1%O90wGTrB`^szLXH3v2xL6DmrN%2kO#>4fnhS8 zOd~VN3^GKPkY!{k*-W;QEo3{{O}3E-!B|L?oFK=^lVs|EaezRWr#dPp^~W-!hmdmL*`J}WHyCM=28HSPT`aJ6aiU45s^g{Az4foQ+fyB z0To$AQIHiBC0R*<$%+BufO0@hR#P-&4MjDeC2J{qvYw(N>nH;Qnt}cS^?-qFpqR)e ziji!jXa^_*x|5+viW4LxhXxD-AOuJ8f$$X=!=w1gehSD1Q9xdW5+nyHCj&ts4MG_i zFg1bF3^_wN!NhX_i3Mmg@D0~hoD}b^o zx&J$;e31XHwL7XH` z5vPeu#ChTjagjJjTp%tJWh6NXCMiftl8U4zX-HanNiz0^J`kxHWWQwL6-cS5fQ41YNlGKR;rC^ zr#h%ks*CETdZ=D1h*41k)F3rXjZmZ17&T5!P?OXYHBHS>v(y|lPc2YKsbkb}YKS^P z1rQo_nmR+BrOr|3L4|dZxt)DhPBhx6fLD~>)m^MPA z(r7d~jX`75K+TE9p>b(E8lNVh327plm?oh?G$~C+lha_Df(Al)G&M~_)6#S_Jv2! zokAa^4*}78gbva}bUK{@f~zb#o6ezgL2{8#7tn=t5nW7|&>^~%E~CroFkL~n(3Nx* zT}{`}wR9a_PdCtwbQ9f7x6*BNJKaHd(p_{n-9z`%eRMxPKo8PG^e{a_kJ4lGI6Xm6 z(o^&_JwwmZbM!pDKp&-#(Z}f%^hx>@eVRT)pQX>y=jjXdMf%c7Uy;5_$1@0w9!4*t zk3nRR82yX^2AM%&3^Ilo!;BFIl|f_B84L!K!D6r(90r%cWAGUQhL9m*h#3+F#E>#% z3^@a4C>TnHilJs`7+Qvop=THvMuv$2((4Q>!^W^P91JJJ#c(q`3@^jS@G}C8AS1*G zGa`&ABgTj`5{x8hC#D%0MwXFd5}71sKXZUdW>T1g%pvA5bA(A{(wKB6gUMuq+#-|9!n87NOgqy7^iLPl&Gaz6 zOdr$F3^0St5Hrk-Fr&;EGtNvflgtz|&CD>f%p5b%EP&GW7;~IC!JK4HF{ha`%vt6f zbDp`tTx5cJAQM#Jn0OX})x+v#^|6ScRMO8HIC(*24YGzkb&l|^IGSqv7F#bU8p z92S?wWARx6mXIZ4iCGdB#FDaPEIA8iDOgIDilt_0SX!2jrDqvfMwW?XW?5KPmW^d+ zIap4Xi{)l{SYDQotSl?X%Ci7z%Nk>ivnE)R ztSQzsYlbz;nq$qg7Fdg{CDt-)g|*7UvkB}Tb}ze+O=OeU{p^907j^a^dx$;E9${12 zG&Y^hU^Ce)Hk-|1bJ;vLpDkbu*&?=>En!1!DO<*tvthP^tz@g%YPN>0W$V~_wt;PA zo7iTyg>7Zq*mkyq?PR;yZnlT*W&7BEc7PpZhuC3ugdJta*l~7(on)ujX?BL4W#`y= zc7Z+4o?wpxFLRPT#hzx*uxHtG?0NPAdy&1wUSY4Ym)Up@fz!k3<@9lg91^FW(|l9n z4047z!<-Qgl|$puISdYy!{V?x91fSm3RBge!sb1WPy$HuX992_Ud#R+rV91q9K@p1f|04K-^aUz^3C&r0$5}YI_ z#YuBAoGd5D$#V*vRn919j5E%e;7oF+IMbXN&Maq+GtXJzEOM4O%bXPso=f2NaC^CZ zTq2jm?dJ}hypeJTxkKDx?g*F4rE%$82A9cYfto#s%jNR8e6D~iYL)xuQ3^&Woar4{)ca%HE9p_GPC%IGHY3>YnmOICt=Pqy;xl7z-?h1F6 zi{}w|J-l9CACJf*@%niKJTecIl6gbCVcrOj$^&sW9)ri^v3P7AhsWjdczm9KC*+BE zVx9zqkEA>qPtJpR3Z9au;;DHWo|dQM>3N{R0ZJHVo`q-S*?4xIgXiSAcy6AD=jHi$ zeqMkVPcWiMPyK;jQxUd;-6R-^=gg6Zs^5KY!rlw1q#&AL0-5NBC4ejnCjS`7A!0&*5|V zbUqK1Ck1>VU&I&lC47i4<;(bTKFn9}m3$Rn&DZd?d>voUH}H*o6W`3Y@U46s-_CdN zoqQMH&G+!Vd>`M>5AcKh5I@Y1@T2?~Kh96^ll&Av&Cl?&{2V{eFYrhCWBhUc1b>o0 z#h>QS@MrmR{CWNYf04h$U*@mySNV7WLC_=U74!*+0+OI#FmQ6ZB^VS835EqD0;+%} zpbHoRrhp}23pfI1VKSa5EeuPQ9(=)7bFBpK}wJoWCU43PLK!c z{it9}FfNz?2K#h!N-!;$5zGqa1oMJL!Gd5(uq;>+tP1c#g0M%}E9?^zg(P9Wa6m{F zQiOxTVd0Q)L`W6VgiIk_$PluGKr0qV$fsL1+}3gl3^tXc5|kcA-P)6uN|<*dX)@eL}x5APfpa!muzRj0$7IxG*71 z3RA+gFeA(gbHco^ARHBr3CD#K!b#zja9TJcoE6Rq=Y+R3f!VBLbOMkzQmF8AT?MS!5AeMK+OL^^Dwc`mVpyyYE5$0YTC5Rk#X7NGY!Dm8Cb3y;5nIJJv0dyCJH;-sTkH{g z#XhlL91sV^A#qq75l6)_aa^1bC&ejoTAUGQ#W`_aTo8|n$He2}3Gt+ON(?%a;#u*W zcwW39UKB5hSH;WX6)|2ykn~7;C4Ca2ge2*g3`oclieykSBpH^BNT?E;gf3x7m=czR zE#XMG5}t%F5lDm*kwh$!NFa$+B9q7^utXtIN>mcHL?h8kbP~P9ATdfz60^i2u}W+b zyTl=JN?a1R#3S)ad=fthhz2DgNmvq*L?tmvT#}F^B`HZ-l96O3IZ0k};-8O6#w8Py zNy(ICS~4S8$&zGQvLacP;2{Fk1NB0E5D_9l{m=kJhA7Y=Gz1MpBM=p$ zL3D@#O13PB4RIhY#Dn;d01`qXNDN6J2$DiFNDje}0#ZUMNDXNqEu@3=kO2ZQ2*?as zAS+~p?2rR;LN3S+c_1(3gZxkc3PK?$3`L+Q6ocYW0!l(DC=F$xER=)tPyrf+#-MR% z0-A)TplN6ZnuX?|d1wJzgqEOXXa!n@@KS=bN7^gxlMXrJWerZ4&l!l~XX+#>8#-wp+LYkDOq-kkJnw93H zd1*m9Djk!KODCk0(kbb*bVfQWos-T>7o>~QCF!zsMY<}*%LuX_P!Q;ofj_-u{jvcW zSw@i!%7$daGOBDuMw8KH3>j0#l5u2g8CS-W@nr&;P$rUzWspoNlgVJ2LZ*_bWg3}Q zrkBZON|{b(kQrqrnOSC$S!FhvUFMKEWiFXp=8<`2KAB$@kOgHSSy&d4MP)HrT$Yd} zWhq%&mXT#;Iayv-kV#~tvN74XY(h3Eo03h-W@NLnIoZ5yLAEGck}b)5TAq<-Pk3HhXaNDw_ zybQ0vt1wHRil8E-2rD9rs3N9_D-w#NBBe+xGK#Dsr^qV`ic!UwVq7tym{d$DrWG@a zS;d@UUakZu~MRhlv1TkDObWug;J?hDb-4iQmfP{^-6=%s5B|fN{iB} zv?=XMhtjEZDcwqs(yR0-{mOtcs0=B?;G{pIj4ETwxH6$kDpSg|5=4iTIb~j1P>w3c zl;g??<)m^-Ijx*g&MN1W^U4L~qH;;OtXxs9D)B0Usz=qU>QfO_BvrqvxnZOlR1K+y zRU;~@il(Bg7%HZUrQ)c#DxQk35~zeKkxHzRs34V8B~!^&uu7p)s#GeqN~6-MbSk~d zpfajVDznO>vZ`zxpbDx&s<0}eimGC&xGJGas#2=7Dx=D( za;m(lpc++;sm4_ks!7$9YFag;npMrI=2Z)-Mb(mOS+$~CRpHeHb&tAN-KQq1N$P%e zb6ZP2s2)-et4GvSHBC)dGt^8qOU+hu)Lb=B%~uQ5LbXUOR!h{7TB??Tem&4e)k?KW ztyXJ5DOIP|t8Hq7+Nd_E&1#FZCfQPOCHOtU9L#~Ff}X)<`sv zMy8Q#V2whf)TlISjYgx@=rnqbL1WaIG-i!OW7XI+c8x>h)VMTmjYs3v_%wb^Koiu2 zG+|9d6V=2taZN&#)TA_NO-7T|=9qTDR7t z^=f@uzc!!^YD3zvHlmGcW7@bjp-pO2+O#&K&1!Snytben)sAV$wG-M&?UZ&}JENV| z&S~eh3))5Pl6G0UqFn{3K0()`>(%w?h&qz4UpH{F@u?fs4e5q;BRZ;%rlac^AkfIt zv2`3BSI5)wbpoAGC(?;^5*?(I>SQ{(4%R7jN}Wol)@gKFold9M8FWUSNoUqsbXJ{B zXV*D&PMu5V)_HVZoloc21$04ONEg;cbWvSQ7uO|pNnJ{p)@5{AT~3$R6?CJzG2OUs zLN}?K(oO4TbhElS-Mnr=x2RjvE$dcvt2(@%pzqQ5>ihIWJxSlMA2`{#)eq{2^uziQ zJylQB)AbBJQ_lj`WsaV!=jr)+fnKN=>BV}99@0zoGQC_6>lJ#XUZq#-HF~XHr`PKZ zdZXT?H|s5WtKO!!g9ffs@6x;V9=%uZ)BE)SeNZ3LhxHMCR3FpF^$C4apVFuG8GTlt z)93XC{iuFSKdzt9PwJ=i)A||xtbR^E51Mg{`X&9cenr2k#~TQS9z(C8&pFoLSI(QWh?y+)tWZwwfN#*i^=j2NT9 zbcq`i#-uUTv|X~soH1`K7)Om`Mo^P5P8z40-pj0U&NvU$mqp`}aoMN~IoPyeB2678Zca3Xn6x_h0;`A zhJous1+okMitWO-a4cL4&%(C|EJBOOBDP2@kVR^dS>zVjq5x4$l|^mQShN~tk@mc(qfF)=NS;CfxC2EOT;+BLZX-QeqmW(B9$yxH2 zf@Rb)W*N6kSSBq~mTAk3W!5rhnYS!h7A;GbWy=aMW$;#lwa40P?XwcCBx}EQz)H4K ztb^7e>#%jiO108}5yP-Dtt>0s%CT~-JS*QSunMgrtJo^BLRP6&W|doEV8|$~Dyteu zGFq$7s<#@fMyttcwpy%KtIcY+I;>9Xi7?}_daXX7%ml1K>%XL#s5NGdTN6N>0spdU z%~-S6oHcJP0Dor8I&Ph?PFkm|)7IIhMl%CEnnml9b=kUNUA5wE1Y3`-7iclWD_#uQ z_AfDJ*fs*x7#eV67&a!bW7sy1jceoC_%^{6M@C|UY*L`f$ZfDqVN=>vHnmM-(*j#Y zZ!_48Hj~Y4v)HUQo6T->*qk<(%?;EUFK}o4wty{Y3)#Z9h%IW1+2Xc@Eon>H(zc8( zYs=a4wt{WcHf9^QP1q)FQ?_Z_jBVC7XPdVz*cNR|wq@H2&}#5@g1yJyYwxoYo8HU- z5N9a%LHm$>7|1hJV9(H-`V8C7v2*P_JKrv_3+*Di*v_y^?2uh*mo;S?gBjE}ID(FlD9Se>{$C6{&vEo>D;GG0#kF(d==Oj8w&VJ{B z6I|+a4myXN!_E;W)k$;GoeU?_$#SxRcf$qZ4gZRHBX<5vy^%HD8--KlR0I1)3-lW$ z$U+;OMquEWoffCnX>;10PN&Q1b~>CMr`PFo`keu1&>8x-j}vz$fRU4Orh$`_b>^IT zpyiALFK4_d=1e)KftoYxoO8}Q7o3aECFinp#kuOl|68^px=5~mSMyRW5N?K=#tqd) zbJ3g94a>!Laa>#%&&78MT*52z4djx#WG=Z2ZW=f$m)fOqXe_4Z9U?rCa4z11m@C)&VWY*z|HNZmZkow!0l}r`zRr zyFG5N+voPX1MVP@bi%;WiMnIJ(@D6KO;ac1&bo8%yu08Yb&t8n-4pIf_mq3uJ>#Br z&jE2~!M*5S0`AU=d)1Bi5Sl_xpNHrn0VfAs6!#2zhCIVfF^A@%12u=)baOZ!u7~I0 zdjuY#XTT%!h=HU7d88g0@N{60!lU%4JZg`|qxI-KdXK?l^q4$mkEJQ?*gX!9)8q2E zJsyu2s5|~E?oQYf@kBi_Pu!F6Bt6rflqc=Uc(R_HC+{hEMm=MmanFQj(lh0m0V>a& zXWp~mS!~)oE1p#k-b-j&I(=TE_r%i~@PbSE-a+q>ci21PrFv;zx|iW)dRbn!H_7FA zxn7=^?-h83UXfSqm3Seq)GPDKn*xurY4B*gTCWb+J9@9dYxJ7DX0OF-_1e64ufyx~ zy1Z_$$Lsa_ynb)M8}x>}VQ<761!7Mem_13L_N2WTZ`Pah=DnldG4Hr{!aM1m@=kka zytBabnfERz-{`RG0dFnd@& zwvPj}9-fcy6ZnKakx%TC_#mM7$b51i>{IxZK9x`H)A+Pa&&S|1`b<8v&jMs0o6qiZ z0N=;ubNf6#ug~Z6`vO4w3Hidlh%f4k`QpBWFX>D9(!Pu@>&yA_zJhPmH}0G8P5P#M zW4>wMjBnOA=bQH}_!fOjSA-zEp8#B+UVk63eMtU(;QN393LyLp0pn)`C_gmd{4o5? zruD<|bNxI&-!JeB{UX2EFY!ZusbA)o`(eMruk@?@YQM&>_3Qk4zrkMzyNj-2x$j60WL6v z_yIvc2qYmfu!Nw1wCM@K0YyL=Pytm)14JQRKp!v!i~&=?42&Ubz!tCrZO9pL1>8U! z@&%83AQ?yn(t!+6h;o5^pb!`hj0MI66M@OVRA4$V6POLm1?B@M zI?+;KIj|B~4d8=>U{A0&*cT)QNx}Z$KoAV81P6md!QtRYkQ$@~=|M)28Ds_7K~9hx z0-s19Gz3jSbI=mB25mum&;hI> z7to44L2vLxED8jJ!B8+9j0B^>STG(;1e3uOP>eGFn`1NKuiiUtzGy=RLdWaTc0JVq(+#*hh8{&od zz%LR4!AKlBF^r@kSxDZLjFiAKQin7lZAcf=1J%eFG6C7h60(MDz&CP)oFNxbjyxeR zkdFMJKxix!4244BP$U#>x<`pnGL#CXLzz%ElndoUh0thdJTw6;q^YKcG#i=&BGLjd zk(NTsO&19tCWL!{UDOvQhDqW6@W9D%Pi5p_fp(MEI;eZ&wkMobZN#1gSa>=8%A8L>rN5qHEB@kV?R zeA$OQ0}rXtgknaFHp4oFK2 zkwu^_twdHM_$VRT6YY&IM~0$(QDT%7?T-#b$x%vlFgg;YMrl!cbU4b0GNY_0JIaZ2 zqr50TDu@cBqNq43i9%6nQ&Ca?MG20oqUxw7s*UP^sAPy5qo(M;TqRr79(6>WQCHL* z^+df#G#rgYqtO@;mlDxrG!;!pGtq1`7tI5KX%rYty7orh%r*Ezo{)zVuL_j8jg*`s4-fM9%ICqO@E0K z;{t<;9}@tFNfZ;uBrz!VKOU0;h)k-OI;Ls5O!}As*i5FFIcB*cG&y3fo|qR{ zP5xLQ7L0{r;aDUVjm2W|*ooehilt+jST>exI!>dpvDi4!oF-#av1uSW&Bo?p^Rb24 zVr;2tJFUj>Kw|1?T1><^Dc;`{nJDqW_)vVfDKpXH^f)8V1VR%#&WUs5yf{BDhzo(& zBnDy=6qm+jad{k$E8@zyDz1)e;#%N1>EnjD5ok{4e|t`jxHImGyW^gJ%TB>7wo^17 zi^t=Mcru=fr{kG;HlB;;f%h~TAB&I2C*qS$`DrFT8=s5M#~0#@K!I8Y4%BKK4~(Xs zL~o)GNKK?fKhT=U2})uxF$C16kpvaUP4omK!A!6cf&@FkNpKUqE0&WuAxS_9X+oBe zC*Xu4p-iX}>VzhtP3RK(gdt%{m=nf?1z1nEggxO%I1{dfJK;%q6TXB$5l94^0#qas zO~exML?V$)q!Xz`CXr3#68S_SF`5`lj3*`%lZmOsbYdnko0v(g6|507NKL(wwv;tw~$bo^&Lg$$tq^-sFi96-Wk?p{5iS zO~#V(WFnbNrjqGo=879tNRB4QlHK(TWl336wx%ZKOu16-lqcm)`BHvhO9fM*R5%q$MN_d< zJe5c#Q`uB1l}=?+xl}$?NR6h(Qsb$K)MRQZHJzGG&86m33#pmZVrnV1oLWisr&m+> zG$GxS?oIcliNKQ@I2rM5s#3%0f4Ne0AWJdRtTa2#NpsV@G#@xq!n7zY2HF&qmZoKC zc~hNIrd4TmS_AYc9q^|NX=B=yHm5CVYuc8!1CPqt6sbIEFHouc=|DP|4yD8CNIII1 zrQ_*DI+;$T)9Fk)3(Tr~y70fbRa5C{U{}qi=hE~47OYm%t7&|Okm7F;bl^}ifkeg5aDYa|%kY6nCCrF2;*2B%WuzHdMxKE)ii|R& z%BVA%j5eds7&1EGRhcs8j3r~u*fREvBjW^ql{@3fc!6T&&jd2TOeho1L^9FL|Cm;( zOgfXvWHY%;K2ykyX2vq(nTgC~W-2qCnaRv%<}&k{h0KX|wVYYWtY+|8LbfN{o9)XI zv!rZ)cHm^JHanOd$_{5oveYasOV2WZVa3X_vz#n9%gge!f~+tr%ASZ;P*$3iW#w78 zXZWg{%j&ZRpj?@<=In`cWy{*LC)SlK>&|+z-mEX{&jzx=Y$zMfMzYat3>a96 zY!WzF>1+mASh;LITgZ-P$Fk$V#hL^%)^v6zJDZ)$&Sw|0i`k{@a&`r1S@;|w*8@bW zz8o<}%Jt_4PDX=)Z8ek|2EG+FN6XQ3j2yEmU2$^U952Vu339@mC?^K)6_k_aWI1^b z&MAO_rOK&unw&YO&FOObriW$BnR1q#EoaX;a?YF!=vbbdH|NXwb3x!_g>vCsBp1!a za`9Xum&~Pd>0Bn4&E;~1+-Pnrm(PvoCUTRxsoZpKHaC--%gyH&a*Mg8+;Z;3)xzfq z`JQ}lzVC{K)f{mKBGwQvu}1RLJS|TLHWss~V{!7_Jg;eF3G<@77)V)AUYeKX<#{-- z$Sd=zygILG>RI}{5%^h#yg6?Hik1yHT8_Lk@5;M@r{&H2^8S1vAIyjH;d~??&ByZb zd?KIBr}F81CZ7e`Rvvg;qd?pm&rjqh^HV_HngRCKTvOj#%rE7a^DFt)Jib6E^b~pv zeOLUffs^6t!eC*jaN=lDfuuz*Fbd29tH3UB3S8i7@e2ZAYl#YC;A=qzX+c(y7vO@T zpe(2g>Vl@AE$9mRf}vn6mIePc$VGhY0~I*B5YGby@V)0J!RHqcA4p2z zcp0qw5FdQ(tKiQ6KkJRJAwKx%^WgU95#5Lv!14dzT0^{uK#Y7J{M`TD8h`n91oEV} z`}eoO@n;D1@^8S;{;&1-*WU+U|G(GYUvEbufL9M8Fv$ON`}gEo#Uhzqr;zBi4KqA)2NCb+4#2^Nd$j%XD`*|jk{01L+ z3RFOTic%mEkP`W6tpVA3gAHjv??z(n0px{L9LaX(k&UNkz~h-gUR+#8#vZ*DiSD|D zY(?CLY`t#_S+(yXKfZ8;ya?Tn9Kzg%Oxf>67B4)A{8sOy$QHzxk=`etL1GyU(9SnGsi^ z5YSoFIrTXd0)IUU*ZmO`;w1tKeZCJhETy1&p<&b^f{r30Sg1Ubk3u7bC}J0c>PE>> zmv_`CIiyFewU|(SXYHt;U+qJ^vld1@K9oXP$_145+$4%{?L6wch!qs}#tSIiD<4N) z7L-tZO%27 z6cqm(3W4|=3c39O3VNZXrDv+6r5n-Ng6RBEOPqVQ(Ru|fMUAWlje=XCHDycpTl$uElo=#(EiIpV#M|QgbgZQdO0<~n zD6|j}<1N(dW?H)6UTS$vaG~Wk=+>6*uWYs;AE>nae&C?xYlue6t!M9OAq;-9MfCmq zTh6`la7&o_NDCVAXv>|yc%tRe;I~_FT`#uW_vcqzE?o6`3j+UU%Ub!ZmhSM+TfRg1 zZA;|MKee1={H+D?>_1x`??$1~7h2JX2e4@TS|_@K>_Q{TSD|nI{EcW!cQ;yv>O;46 z^`p;5hR}{)2D%N!1`S~jI($xmzOh?^zWzLn&a7$BYJ?8m`AaJrcee{2zdisO!(sHD zYjJexjSTwZXGhW3_f4bGZ_K0j9$P{0BF>{vkKc+0K}qyp-xeA)V9@uS-a+G`J@nn( zhv+r^Wi$eFJ6iLdyV19F-Gd(9d=PCu{{=MS>?3GA{1tS#`*Cy&@=5d>`B^lf8}x{g z-$J9#y?}7ejV*Y{1lD&+%M3#p8YMl*kF~#tMy~x8eWv^G zXhhdP&})w&TGigR)&v3vI=*0CbEs!}uxv3R>8Q*F^6I+>Raw`Hi*m|*h zq!j^CTdzSdT4mjA@T75Ce}ECTp1DKXs&7@c5-w<4ukAIrBA<4&;#xhe!rQ{F=P|L? zt6S2om(=-IWXo9VY1CvZqGh`EeD_={^ZarvQu?vhUg!e2|BJ11&w49jrQC`{R9cbh z9q^cTTA5eZTXAn(ZiU=;wzAEiZtce018%>k^-1adtqA&qtv7kT)M~l=YpwTpKht^| z`;FH9wHI2MiI-aucf8sPQD1A_-up>w3-ayOQpc}aKMK9u+V9wu@)gZLeFcZRk!%+fPuwwkW`w5m2}d*%fU=@2A>s=r6Qw+&|rh?^|d?T(#WR z{@D4p?ssl&yX%2s8|bmO9Uv-g{arh4o49%#g3xI5ciqwEI{T@%uFLndVR}E?hIr$F zHYDP6Z4mOIwr8Jww5<#Ja$6V;8pr6z+s;y-YCAXhT$={@?KTAQ#kTO7m)e5**V=Bx z{vn|aopwa$@7pg!e`?2^`Ad5%;{A3E^!N75 zrx6%xcMAqGw_(f}EauE(r!d+MkPt+G6~ zPh)x?yB9--9>8?0J&YmrK7u)o{4!?e*(Wh!<<~JO%=4HM`a2jL^b+Pd+<#+GKYs($ zed#BdtKRqp2D<7umxFv5pf6h`^dYoh_j!@mhSj0w(F?}vC!RL!1i7E zGS<@dIQBH+2`mD75`5p&Sor#HV5#dbVvojO!FD2lh{g23jzx68fyJPHj6ENI8;iU2 zOYHhve*nJ&tn{C;2=ZUCU95j%p;IUvbXPl0bFLGISUZI)Ca=b|oxTnSdS|#xJUlLN zu^;#5<0H687Xx>>m4idTJRG7$fWy-v+?(AB+!{iO!>(y?r85THIg=en4ZCqvvmY0R zLbxa*ihBo}#&uz`xGG{4hd__v7EjOOx)5{V^LgA~_SSF+=sXU2_bs><?JdPWF>=|6f{%sug>)*p&*nbt*75xzogLwmoi2MY126`KJ;mj{_;q$-6bt8U< zJE#96?!o@|ah>uHaNDQR9mGLg2d2BTL-FQS9gn|yL&t?PAMMCNeI1{Aai}9P$n3cM z0KX%A0CngIijL<{+K&6~wRTj|t`1bEx5NBFyrTt`>UgDpwBy$9$&M#83n1XW)bW!G z7dqN&w{_^A-|isZcF=*is?mY{)}0+)o1g0FI=r_70v+fl5f5}!Z~bBi^y9B|d|diO z$1RUN+kskpt^>9DVg~{7-HxF%uXLP6{C7v!#UFLpUU;ixt@QJb>I=W?K%w64XzaY- zkzZ`-OcOAjgV%nzQ}Cf{JG(Dj-${Mrqn&3E_)Y?~ue19LFb$+CGrt>s)p>zD=YUjl# zZ|)>Ow{~JsmpX}GD|TAWRy(i4?sbNvjZReeap#A^cXr;6_+;mv<-a=7=KDHtP<*}< z0e`U*aq-d4t~Vd+JVSe`^TT(4vvbY%VrTx+%bhm#4?DlP@smzz*V~=Q@BOOtcEUTI z^t=DuIll4t&NJ5`Pp#q5r$A5e6yjXRsm1HAI)z$4cj^rG##7t35Kfunq*Hx^L#I%u zsi!usVV@$X`KK=2EjdMNSDd=knVuDi+_CEuv)gS>yad~s z2x+%FaVYt4BK@y3iI9%x63da76HDW7CKf_(Cl1s-PF%oxnb>adyTrr5$3#To=fsJS zzY|--zY;ymV_~~B&v2Dv?F)gcMM-l5`)3UdK zA>-|^4y}7&5ZgXj-sM&p^vMYr@bnz4wCySkS9A-urML}-gggT8>k$l>@eGy)c>y-4 zU&A0pZ()ZMKENPxpJ0%sf5W!N{D8$Uf5YZZgTfOM;^BvJo#0*s9G;ny3}2Fx4mU%( z!cEIF-~c2OE}fAJpExrg4lNuAPe~|%=bak?Um_R_PwG+#=P#cL?*Pn!cgexQ<70^M z6et<4Iz@+1LX^Yp8C*D`r2^hGts0I`RKlC4191N}8@v!=hi5{a@YI!y;D?z(xG8Zl z9P(l*JYKR2ZXdcH9%gQZ|A6d{B=dcm{{WK8G(^`4&Dc{R?~x^gEnV06}CyViEa;;L8qcCqxnl zfvEGQArcT>5P-Qm!kV9ja2DkvVv719_B#h7Xut?WAutMo{WKYoZkvgKY{Vc~Arc~C zDFv}IgO0fLk%utPsYF2P#E6NRQp6080#U%!BHGV3B8;(hh{9nD5Yr*`h;x1Xh~5hq zBif)#5s=K~h)Jm}h$ERB5PMBK5d9&05UWxSA=W^SA|OLgAU2;mhv<6b3Zi4{O+>un zKB8~NBk(p)5QBjii1>uph;QA$Al6L&iLgszkXeVpmz~o)BAEdMvPXUzvSVF$`}7#z1BvA&ardn09z1ghN8c)KZbK8B8Pq zu#n3kJmjRdN+dJA8rd2nN9IneMRJWsq`8L`yp9vuckCi0B+HL1XlO!aLKY)GB9?=l z=;g={F)hf9e(RC5mTg6D26iHc7w$un8jc{BFi#-Yu+Ae-6N9GrJ`9nA9^V zA)|LvAt5_ya7(`=^rb;b;G6#c`(h7CTC!kdQmB7nQc0g_NrD-(lMW|fk`lWSlMoP6 z5+t6S#9Tm20yoH#V!fQCm93RY9b?5w=7ehSYyUr2240>70Tf9onVO_NhxJJl8_Y>7 zx-Ce0IMti9H8zm684*fK&uC7PCag%pLslnUgltGU(`S29+>5(+*LJ6-WnOV1!@;cv3Y62c7B_WGz8`J+|Wuzv{Al;Ig6MH33&d*LB0Le@4I&nZUYsTQ@6y)$^NQaThkX_@G z8-Pj4T@X`~A$il2k0E9!BV#ei85BbD(oU4*Tu50mxI>q`8#ym|#WrbjhE1M~JFH2r z&IFP#lWfTwj(C!<3~xxb8N$gn*wW;M87q?!urXF4M*nj?+0^e&GFb6Vo?GxbS=IU}xq|#N`TP@TO6-shDa(%{Qy%W?o&w=#q?nD_ zDO>FQQ>GjklJZ_XA!W_B87b>A$7qBL~1MqnR+cIEj6~VODf`4&(vPAS*ep&jHgw5f|K&8f+NEp=f#S1Q2qrWWA?sTHRdrw%Awmg<47OuZ1l zK2?~q6?}aE%M$EJ^}6<_W+M-Smmf}j1wD~E19B?W9CJ4HQTS@=v4oqc<9oHG9xZ>K zin#MGH3sq?yx#vONchEA(FGcs+~#Bpg)yG%*jk~AwVe-$omg0?JeYBzS;#5{i5 z?i_I%q-%9r+YCk8y!k*{i^-Dqaib><$?>NhPHIfE-d>hwoVY5@uykG866fYL+v;6u z?QHwfCXtV%oojzO?NP>sw9zMSq*1=Nr6pjWrL9bO4L*+ld=*08q?u#ir$G}wrWvch zra^#jX+4KV)7C?w>0N;K>G`n4^wyH(^b$_j^fxiR(_@K!(z6!krQ723(?1*=lD>4- zsPwLdGjCZ_WT&jcIJ|HojG={M{3=~JBM^uSJ6`VYvW^cUQwbjaf+>HV`;fj#Qg={E~ErVoT{1@GfOS%v@m z|L%G`5OfSW7X1@_I`(923nUKR4xO4okEUhhWOT{sn$aVpXGRJ-6`g?YfbNLygzk(^ zM8nW?)jw-s zR(aN(tQlG3vU+AQLC2*ki<8C8>XkJtYiQP_oWh*Vkii*4GU76!(G`%k=))*FV>Ki# zGd`zX&RNt&)CJT@)Cts5)DqM-)K=7(jD?w=OmF6*%w+Wc+7tcHKDK4svlnDLvYpwk z?D}kXwkO-0y)b)GwlCYC9mo!5H)J>jmAZvM%zW>qwS*!(GJm$(N59M(Znb$3Xg(MebJ<7 zax^8H8cmC)N4rG3M!QA3M|(tjMtenjM>C?C(X42nXm%78&57nl^P+vD{i6M&`OyK< zfzd%xbhIEkI65RcG&(FgJUSveGCC?cIyxpgHaadkJ~|;<7@ZiM6rCKM5}g{I7M&iQ z5uF*G6`dWO6P+8yM6uDLC@zYR5~9Q?DOwyQM@yoVXlaxhrA5o4kE0X+e+Vj!|AL`X z{eKWt^k@KWKpW8}v>9E8wxF$O8`_RufOeprXcxL3?M8dhUi3orqW{fr$5Qk%^m6nH z^h)$9^lEeqdJTFldL4Q_dINeRdJ}pxdJB3hdK-E>dIx$ZdKY>(dJlRpdLMc}`T+VM z`Vjgsx)pr{eH48ReH?uPeG+{NeHwiReHMKVeI9)QeGz>LeHncPeHDETeI0!ReG`2P zeH;B3`VRUo`X2f|`T@EP{Sf^K{TTfO{S^HS{T%%Q{Sy5O{TlrS{TBTW{T}@R{So~M z{TclQ{Wtob|L`4n;}*s(iu1+!;{tKPxQ4jKxKLbETsUrVTqLeJZb{tIxaDyx;#S73 zid!Am61V0*G@7@>ZH?O&w>@r0+|IaNal7O8#O;mS7q>s|K-|H&Lve@WTH}t!9g902 zcOvd&+^M+J|Kaxk+27cI^)dt$2~C0~LsOus&=_bMG#!l5cZGI?c8B(W_JsC=_J(Fa zGoe|~KG19^3Yr7Wh2}x~Li<7cL-U~npaY?U{x3ynG_(LZ7&-(x6gmuwgARv|fR2QY zf{uocfsTcagN}zzfEGe0LMK5dL#IHeLZ?BeLuWu|LT5o|L+3!}LNQS6|KaP$#5{yt zh-rf)gH_`0F)twf!2wG3OW>YEa+6wxgfCsRsb(R6d((d3X%&_3Q`Nw3epR@6m%`p@U4s901S3W{onibu6aC7?Q>I-)wEI-?R%FcchxKp{~{sAN<`A!;IO5^6GP3Ti598frRf25Kg1 z7HT%i{?9DZguqd99{U2Z1L-i)tC*u0TlY?>GrCj-(uRO5m?yMTe_%VXgS_3eAHWR3 zMj)1VLGc||%w1poQE#?wE#gwnly9rrV4TCov)&SNiSgP>+zR=ZvTLlfE=6zwt1D1O z9vNimM@r`_F7kDXX&ea_<+{sC*Lb;aW%(Q#E8l2n(DP?V#jJCzq1eYYvj`FV8Qm(+ zK)OW`t^f>|byoz(RZa`e?<&f6-Z$O{+7XiMT)esNd*B{6v!uFlF?9=TC2KoR?R16X zrB!^dYAX{*NGI6b=YXEBrM|?XQ_}X7*UYZf>uf~Idt*L7PusWbt8Re&Zy~S#n|rKh zO!KwMuHF=FEn~InhIw4w^~&2|1uBMarGJ;d^_OuwGLAEv-K`}@40;bWupzj|ou)dla?FI7~4F-YjM%Yo_AKyQfF}FxXrj@{H~bO z>e|L_lH+m(eV?q&w1M-2wZr#FSV}mGhjWhjeA+?S?)2ftG{XWzw)l)t79z1}gdaq% zt&N1 zV}bE1X0`BQ-FeRK+TRu5c{5FZLa}x=h#SuqO`?hFudq)O@=e1{Px;$b<2VnfKH9so z=Z+JsFMN#?!r9B7X>~NFJ2Uj3e5)~tz>0>>yfoQpa!=Ywy56rZS^^jebnaAn0V%Vn z&Shprd_DPti^Wxs%7zgaN_g%n)(XyYww7V5`r>(P*zE2?6YxGV?7|rVgG*@haz3ap zkSX&JtQo`-Oef7trn%}bho05T;|M)b#}U8CW7u5@OyL7}cZMW12GBR#u%l@l*Kz$y z?g7bE+*VwrrG42F>H(I}TO;pWpP`8wI1rf7-$u9 z0;Ls&s+OXgN(ASb9ZGr3zE`H#jH}4Njxsio24gnLgBlin0sXzdJAM>#fkeU1_8ZA4 z-h>LPw>{~zXp=6>HiY#-w?m$1YE_*JMBFpVzt)_0a4JXRi(HwN^MGpe$KrUx1x$C$ zWn8e<;cZ`g6sxa)s10lQoRqTlMe&9h#h%(XqBcE_UqbyxJ*(QtN#?9&_h)zY58xbh z6SaL61XeC(8c|)jyl5yZhi<8Sj2Yo_*yo_7{xeiAuux8rMPkvTL8hc*)!^sikfIj^)D(KKSr~kN$}iM>|z8NW`(f6N}^TF z)UH*2#x3$>mte48#IM|QxZSnKq$@-Z^bbpSu>5p1*=Ww!_N4b_BzaQwOB#&qli~)| z9*GJ^4Xz^(lTXHuln3e0m`uuH(VKEa9;4Cf%Z)bNTF7m}W&$&iPn^g+yTX&+f zjSz!*uXsnW`xFhZ<^lc?b(tdz(*v^}$2K1+Z^3sG1hJLOeYjpC8ZN*Ja?1nLoavrh z^4}h|@=E0b=^N=!$FLH$a*m6}FZ73iRrE~pLdtv1Tjd!4j>-*Om_EJ)hwB~qPF+)y zN0ip{xcY`C#fyD^6casG_6cG)aMGZOytRH9|A%XsaU%W$GqvQ8SObl%u zKi2wz*2#aNVUzr~B+Ii2*HeGMF_ZEHN5$q4`%{$M6##g)o^^!faZja#T{d)rSvZHY5dFlsAe&saP=-fp$5+^_+`G4 zAWFJb7o-gmGx@i*4#lmI+u+k*ukFPt@=wsrB0eR~sf^(yu*O9`1C4Siz9L99j}<(z z!~}eSH@;2SbH(%600Bchd_pFpf153u`ktH8jJMrnR6tATr+WcdUubc z=9ql|d8!J}Xm0-D7r4hZJ#?rPlLUA9uUx}i5>H5okjG>9VN z_cLBEsV|=Ad|i3ZGoO+Ud?i$f3AsVE!VuVEWA~x8a!*_c7uviP|zD(of{3c3~{j*-4wQLuF%$Awt(8 z7&xVd!Te^F*8EdaX?_UQST9M>)E0|#85m1GX$5N`X031pYm4zF5knX+yDT0fPr|1W z?l~@1EMP#G??hN~R^=ucL2%2RtY@*8GqXvrWcFs+Nv7%_%U9M{Z@CwzLZQCZgYHc z+|qGOa9(NYq$Zr`t7xKgjHoNNQnJgG#=6W*4CcC2>Yuprz-SCB5;O)Vjh%(=fMc4OIPc%*=@LvhL2?r%m53( zv8gbuNIeO^n~4z}qaL9yb!>J#;+|tn4J@KO*R9ocDBi^%MUi25QkURMO$!>vy9Ub- z`lc6+)qKV51Cw+4_HRTQW`*K}C$hQCFnR6Pa{uLIh* z*m?MixFXY?ikgLgy9!*b_JRHl+J7+IBD4{QuWCey>%^-?Kj?oK-KE6SrimCri`-!r z2#P6N9T|jb@7%!5x}=I2bx!p^zzWP6&vcPF)V_3oaXZcj-)rScY&XM0=@q-Caj5>j zPLHQ>pB{CL)CKlC7+Gj}w%4#P=(? zg~yN+Rh^W(=*sf8S}H5ZT88mU;()KLClY2*Nxoxz&;MI+T@`G&to~fGjrg80R6H9* z%a&39I5q*3Nvp^*CXPISsL>QLiPY8XC9KB;HtQsH6aOu-S@X=O^e&=5v5ll`B*WM* zT<-xgS*cxvea$#P9bcJADAFF0P7_4ImfTnF4xAfvO>u~}QX@BGC=;82B3ol_xk=0+ z$MdGKKNPzx)Ekl?~qXyNK{Q*pZ*U&Sx!8TtiJI)9#s&fbP+Nn-#Z_9*~&^ z2F=gJEhHb|H~?+!E8$0QaAy~;fzTo_NH0V_YQIVMsgtYXYcTwC-WB1oYCG5-f)mpP z<1lu@7W{aXh0qsoD{^2jVIRrQY8KY66z&W(;Y~F!i7&B4V5{lATjD=W@4;*qb)i$a zBMpCivzhxPM|B%(PvZ=NEwxp~Y>dTy(W~Qa#I$p+;miX5DhkRkR6nI2aPz7x7$b-6 zzd>*0rsI#;PDz&IVIb@o~8gvA|dwpUcD+>I1@7ksv2qUbYUgllY`4KxzRY7c2F0W_8ncAWl`mz9-8tCHm{r1^y1kB` z=Dk+3evGG&e~or&=$mbvxx2^`x~sEM_C=axF)E`)7&3;s7pD`?k|mf`kz&FrcLxI) zL}+|xc_>b<`^96G_pj>fTxjfX`lLt}-ew#y^_R6Xb@Sh;yrVqft0Kgj7UIUyoz9W2 z8JKml5Wh~K@OB7ts+S5oC>NGa^k1y3*XQysQTI?nr7!$n{ckXx&5h>u^BySB{_e6> z0tR)EGUP1}-mU%W?%_l+WVN@{$L;5YJ*+3Fljx@f8OpzPesEs@G`??XJ3T9S(|D0> zR$i*Rq8=><8F&h^P9#AJ53vp@u-r||-+mac+1e-a)|Uv*Q|CMXHJgF7WX=m4UZ^TK z#HKy0{`_^OmXceMb;69Q`_wssPELmO68|KtT)GBxf^xocQOQOgiT(gjD$nD;0|yQG z(rKC~>lHZny~MZN)1!2oE=@QYqo5#w85A;Mg#2LOOmPMqX{cp>_umk)h=#^G+{@D6 zz(z`PuwT(r;aJ)O785&xpf$vM_KN%0onqC{52|)>wCukbT}t)d-jNA3EPE0#pDHHJ z^3V0xiCUO#{QJ^-I0N-L>%NdG6|pw@4(O5%1W_0VWzjw!8ABsq|Fl@S5|ih3q-f2F7IZcoBqKQguslN%c!X zlxO4b5D1tEy_MS!xB)b=EW|wgMv2RI+4WGUBK`$@Cx5XKUGoBSy%pFd;V#k%1zOd@ z`Yx%j5;5MG{}R-yzG_A&TRAJ0zgXUe{;pPH34q7UBY!t0nq~C2xPNMnIk2>H z@-LyaVOr@>)s&{!Zhvht9>@COJAtv7%FQoLTg}(VL-}!JF^?xIaNgq~CB6BhE7Jts zSoqW^5n2GgrTVUCU=HM++LTYTGmbMdq)wr8qns1sQ- zrz`_JJHM7sB@zuzePhXpy3veftm*Sc`K4SllPg%QBw|{y&&wKu>p8ddZ)q2?E=_wp z)Rcl5#58GE`F3EpVxFmo);z)QF<#*uP~W2(g!rJ=oh2M!q$ji!U*Od6(8A+PWy4p8 zg+3NceV$c!!EEuw)l3rpRXp2!T+h$g3MB z^P9YBP7f{yeh^+INr(76or0KMoZE{(C)jP#| zs8@7B+-deTz8x54O0^~7G*lyHi(7^%6LN7VdzE`&U{)Qy2I2qUUnP2KEMP3;B#H9O zV#Coey@cxOfN3T4!m_zI!BtW&CWR9URkDthGHRs8HKKQd=SB6VSXCi^6uwcoMYper zOQKg(wRqkc*=yrWel;xvKfk)t_R3ER9Osm~wi(PMl5jd}pnhRVHg*~b>zc#si0MZA zY+EM4uoE@UB`)I+{69e??FDg%=e&+ZndLfZ>1<3i@?29mP#Uh>E*h^s=oq0IBG(c} zV%ODw6lIhi*U{BB;Zmxiw43}OfGJNGe<4eX^808&B1^zD3hnzq!puCeF z;OfOU9V2PgmWhPV%0H}sxQ8(h8URa#dcyIE@JOl0VrYucSqhvnTK&2t4HM-%FmG9l z7WDKS!Vluz1&0@6J=ZZV>>0155iX3SPiC&5p0Hnc4Cl^f1sZMCDAmRuNV-YLQOw8X z*Dd336D+7)$XevSMo2PV7P;j-48|p)Hdk+|^YU)}#Pv!|Ll5H@2T7H} zZdkdoo+q-v~W1+DZ%XgZh|sr+73d-bfoim+AN;fmo)oA;#mN?sxp{=)oy5 z-sk)#T{LpNH?bM>dwZH|KNsOjHi-{WPdiM2${RKZweu~GfIIM1cQsV6GSJAHNuJdK zhMGcUusdlln>PsJNqzN*nymRT{zBHAk{cKkRZSt5h-AIkv+LpY7l3QV^{fws4??=P zA0>t<5zZyg;`DR%SG*Kc$-ul1a#7U)XM&SQeqXuJwoSE+{hhEu!V#8~^y9>qvL4l+e8eNv}0?3XWCut`9~XVoM9 zm)H#rQ|fC;1Z>Kf8vyT77Bs zIzm42sH9N%l};<#fZ0GF&7N-T<|z`0z1x(XiFbV*##4<#FqMthKG5BfD~tTBp%l31 z8}6vup!rk0kdfjsikP(CWCW3jn=5`)+`?Yepf{vi``IC$Uc@TeVZ#k@p0vO)EK(Q{ zhiXFuDppdivc{9r*e8j%En`J-rO#@@wt1CllsV2p44rt5Y<|N?%vXt%8dQyAm+}IN zv&<&y%+mH4l3||EAzO;E^Kp8B@q^o)e~V$J#FU;d+sLRUJPY1)PZfMA9z=S;9F3V< z^+k8m2s2-!^ag#T@wD~2e*A8AG9p%f6flyoO>?mW#L3PMG)Z|Sp|_&1c!9;r{psJM z8$%qT={WzL{~@VW|E{JdJKefev{-{=E#>kF57{Q+>oOX##Wx32E_~ydO>5(ZjjILY zLBr)NIDO>9Ok&R<^fD%5H)84Jj;ucc4zEHQ#V^G-Gv7N>c#YK_zTS>k{Z>kNV~R%- zGyNJVL{r7=Ot&)^8&WBEnda*ElC|O@As{u++h7|&XJWRMhDnco>7t*tjR838gsxtx zv-BbK3I^O(-b30F>|1Fk%qjw$l1v2HdHxx=+n8{8g>E-7+eNgEFn;pai*gv}O2_jm z0-I$6WrqkFLtW!i{aUIrP!u?%`=(e%Q1}}gw`ix>OT^`vYkIEwnd39_JaHXH-SD-n zeBQ+(B(dC?Yc5kkgR}7Oh|7y1ew%0q^@NDPgVV~WnSubIA|bUixWtMk%=Ai()e*!J zdV?JET2GkU3Lu<~xXl6;;k)%z`7z;7VU`m|TU)ou*xBEwvdO)f`;&2$@en)J)|l3ioQ#a_V~KS*}^u z!NXScRsN-`2{zU7s5P2bfkN9^yt{fW;|=$b<0rK$^svkhA`lxKn;acI$K4LXepXIo zK%|h^A9KJ+k%ld4Wfa>c(ec1#VhFR-(V_YPu|wd2N|nrrF1cRqBy>}?24YA&F#52a+Rf<;0sw**roCX z`c9%Dfn4DS%oVv7H$pJJ3h+n_0g9q#IwzOL^>{UXoPSZ)6CuVblfe!F&=r{C;lK## z1@BHA10!z;u)AUYA$;b|HuRuu6+tlHOU4jxus`}1V>@EmHc!I{3fc2h{R;B|@XC$C zF!cqcjM1*bVCh!$sf>botbB^m7`?1yp4X$M9m1#?Ck!J!W?-Fs6$U(LSAM+m1Y;)V zjAEJiv16m7qw}&Omhnq}PJcl+%v5P=E8bQ4IXE=%lhT{`kMFhUI91#@ik8ao1SEuy z#)5e@rILt`o~PdfZjhtQeA9I61vkreN>R-SGze80ya&XmEH?J2;gozg{d}Nf{dbF- z_*v)_jTGvcC-Hgm`SdQ>CCulHY+8YQqvx=1k93c4qO1*I;> zW2+~IINh_4^oq8Wkx)Lk@}qPsU|&!px=OunGdEN^MWG>)j^Y)d=e3+{ur(9Du>J10 z#U;de@&uVj+(m!aec3pnJdeo=eBiCpEpQ!mt@Lf9t*A}luVS_fbOF4W=Nw3>?(?}RYsxlZ(c8jQ(pMXKob6htuhk>DraNrJW08hu-FO<9Z)fM__I4kgm zAoet{J`q1y8r|)Hbbor?Z8t@9Ue?U*t%S;RRX+{ATsGe-te?L}e#X&0IEX#nKP<4@ z+=o3;X<&Si+A%528t!`*yK(|Imwr-y+0*J+Y+TGA#4R9pu|8EU17Zymg|jgyb#s6( zio}XQC7(66$c!x{w`a2|54v6vUT}BoZZf*DcdJ7ewo|(E>O;VJ?L>lpO2(wAgzn9pQr8J>g}_WdZnID#oOdNVrLYQ=|F^ru#Uw-Y@y-*tne5vv z0%49It0u-Y$2)~osH~OD5~*=5#3`;Q<*fLah(_%{{~Bg~?NxCR^_3w}Os2*)T^IDL z4^n^W#xQs2_IM;NEag1=fqI(@Pq@e}Rv&ka2^`by^BHwRi|t+rf1bW0P43C#Ju551 z-shJV(Xd72)%5?l@B2II!gR(I;t-f9eJUXp(}3xP8QdhKRut!9_NpgfQVG*B11Mvp zVgf?=p!f+8Y`j{jQ>Hf$$24OQ%`{8}W)a3_q*8?>c=0iei?EomoUoFxk-!a>xw;Yl z1~s>Vgz*HLiwEjzvG@c$4Akg`{9Buw1FCW5pdPo$JRfhw8}J_dBD@ht)bGO_#hk_* z!Mwmc#JtAb$8^Vj#$&@LPdnuiL|BJ^VISQFNb z)nT>RwV*_}9_zzC!@j`2!XC%|g?)^@i@l6}fc=K;3rd2YvH3-C&?6jB^atCcs52-P z=7IL#WKa~G0UCosPy<|9w7O_b(cYpPMe(?|MQ@6p6}>BJkL!fHhZ!d<0ae5c*d@)= zgf+Ed-0!l}k%mSCE`{q;TvmSBo;+c09=3fm4$is6b4P!FZX*>>yK z;kMuo;P&HA;sUs3xRtmij#ZA;jUrjQ z1QtSXdhYzY4f@9e1-qZU!18ASSp6L3o$Q_FO~g;4EW@wIZ@?eKM~Rp4H}D_u4+!t7 z&nv&s|Hl2o4Qkp+h+&k<-q_>Q951E@e1*K5{Du6U{EhsC ze2?6zB(dZ-`7Iez(yOE^Xpm)qj+n5-P+~6$l>j9zja{3g*gHRKJZEeL+vnTC68c(Wv+8J6k5ozjXiZ`X0QcZ1O7rh5qGN+noU;%w0*f<|*DgdkHbkiUc z)->HT*~Bmvn$k^7u!KGu?4a{ZT$2E7q*s`Prg=#UZ<&Brkq?St<%=6sB5XSf=zOD-Nw2wSSjBFmdH2PU9D@ay9;*4b1m&G zeJyF0v6esd1jbmgCDKv%7V0w-jFCRaWvZ_NRvnr7BM)*04ctedQNtSt6FVCrNx zdop_i`ybGL(Xi{;+u0)adiDZPhS|v8&;G>z#%^Or*~i%T*%v^z@&x-J2#HVRbmDa9 zWN!teECEnk*~Qt-Y2@tX9Ohi&+~hprJmPfVqCjD#6XOb_s~pK-5Iz!q z6J8Qt5PlMph-t)lP;mV3YGW~IJ7$7zBLy@f2NR7%A80UEgIZ&V*hDmgA|sEun0Se} zfp~6~C{#g|F1NpDEkNY6<3NXJO;Nl!_i zNS{gPNasl}NPm-_kbLf5#R3QiD z=?m$5=}YM+=@;o&X^iZ%G*0$I`cs-NOO|EII?8&q#hrWY;kiLyha9yEarr)8zpueK;H5_A{X53(0WW+JwF`&$c zj3f(XF#lq1XJ(Y=l&6By))ywE99G^RRI=pd>&t!RtIKzl zUn~E+{2pUl;HCey|D*r2|8M_4{$Kt-;ITw#AU==~=osh}=p0B4zynEv)IeGwJm zM6?7nE*6P=pmxC&)q=`JH}Of)anXGdOngYxCPIpziSCJRi_*mFM4iPOMg7HMF-BY| z-YRB@En|_EiKmDkh_{Fv#BJgMRgc6g#jC{!#QVg%#5cv;#TUhH@ey$|NM}u} znqHM%m0I;(Jg91T)$FPoFpt=ws&AFO%2nm8QdLQ-YO6F=m#a=!t*+WrwX^C-)x)ZH zRrjmfs@_+1lk|~vmyD7OlVBxu2|zVd{nSO&mDJtTLGnmwDCY_F9Tjc=K%HLS-m&xF z)$pfaFZ_x7rTdxtv-^d+9oP%+XE<039|TsyaUQ&<$RqHSfTeJ% zXAiH&Gv8wbE8!-v6J7+Ga>bx3H-@&3SOj{6)gZOSqwUxC10A~oj2h!u#+W8C%|J0x zT$DAGwUp(QAmt2Y9VJ3pLAgTNO4&rYNjXSqR~lb>kCIUOm~xBqozl6q11MAuEk%Ng zWiqHy@a+BH^x1lpK1ZLc@2Bst&({yo57MLc1^U7IA^Kr@4P7t4Q#M}i_teY0 zvW>E3vS!&u*%8@g*)G{O*EL&utzvgfk1m}RZ6dhEjD)=k;R?v@<@f-N7K_jY#AK`D{J3uq4mj8j@L2!%TQSg$#ncrD( zo4n&%HPd@#D@u5`M>%9lA``4I$=Q5X$ojMnFKvS_lXA@QG8H}8V|}) zGeKW!hrkVbPD;Td(0Mu{5Lf&X^sdOL_zoVA{~+j85d%s~j|B+OP8tOYOOrrr$qbrG zKt%)SCk?GQS7EKNRa~gBSMWf2NnO!YvASYk#kz_O71t`ZRJ2x{sd!M~uV|}qRYWTW z&3jvsKCcTX9DS;o0;)+%KoyA!dP?=6Wi$>Hmr6j*$UDz7Z_T{mymj+N$nY{lrLodl zX|7yYxv0`r8LDimTw1xdveGSbSG%QdnOp8wxK(bYTkY1kYu!4x!EJP#-B!25?Q++< z-R=f=$bF%{6Dyt7oz)xk`!YefFB|mwo*UlRe5r}nz~xAJvV17_9rr!=1GgiuD^JRg zig zI;mc2mO7*^sav{GngA+%o53*CEe;fTRr$H{cje!e-z(b*+Y94`iNaK27hzXnH({o* zFX$!p16`zHp!-(Ioh!vli=}8O%aVp~2ExE1U_a22e*oA5tOGUx+ksucbKnK=5_kpN z0geF2fZM=%;3k0J-veF)AAv8xTOiQ@GkgMm0jc~XL#cDPp_d`w(8Vy=FvBp%Krqx8 zD27r)wSj4~A| z$v(+p$u7w{$ri~A$r%ZvI=MQndQf#iwXB*}EvTMaEv=qat*X{kH&kz{-cj9By`;LW z`dan#>hIM(rFl}Jq)RogdQbIuP+gq-Z)0(cx==k$JyT6p)6@iYshXuGtC{L5HD9e! z*Qpn%L+aJ)ZR*R6OJGc`43sYU%!#0Q$!4x&#+N68ZseNs#pP$qFPF!&06-5c683K$VHg?~3-y1Z6j6ma?mIfHFrpOgU0nq{JzwD<>;Ul>#M8$yMr=F6DeBpmZv| z%4X#f?{GZ@q7mZ?kW!Z@X`&Z?O@2l^-@0ahlFA5Gy#QNj??fo769sP-Zm>=#> z@+bRK{Hgvle;0pue@}lee{X-5zn{Oqzpp>vU*I3?AL<|OAK@SAAL}3QpWvV5ALgIp zpX#6HpXs0D$M~^+yr1AF`iuQ!e~F*sr~2uBhQG{T?q~Tqey*SA=lccz3jaKRrC;c; z@>lz%ewlx?|GDyw@^s+3`mOr0`n5Vyr|7ImS9DaM z6kQalimr<8im{3;MTVkKF+hP=$~m+8np({ zSTr_`U$aWHM6+JgqS>I?rs3EsY*O2No8D%$Ewi1l?XvB+owhx)U9dg2J+*zbb+iAn zy|DeUeXwQPyV{fO-R&LiL+$^v^ zKwlsaC;&PE!+^=aG+-`(2P%MSz+_i701c)FdxNvV)!=UMHG~?P8X^tN4NDuAHLPq{ z)iBpWvdbSuw1Zgw(PZRv23!O zwp_Gqx4g7`w0yF>wS2Z*vLsvYTfSHxSZ-M^TjH$kt=}vO){fRhYX@r|>lkYS2DT43#K9buhdHCnT*^Q}6o*2=U>taGeltHVmLvaPjNj+J8NS{GS2SzD}E ztS78CYn^q2HE1C%@2qLIWZNHWM_YGW zrfrmMtZle{l75PQsvcqdq4}*z(8g(@+Ei^vZGSCXJ6$_c+fUn7J5XDo&CsH>+1er6 z@!DS65^bq=s#dO@qb<`iwPLMS%hF1<)mo2sm3Fmuw|2dDv9?89r(LTJYnN%)X}4=H zYg@JFwD+`sX`gDJXa$S?|wI$NF1X1is3V58d?>znnf z^_%pY^;`9|wKKWi+Qqfu+DPrP+O4(wYY)`6)}E|AS9`wpRPFKFo3$@$#l{xiA^ipY zb^TNQL;W-TOZ_|jXZ^QY<@{CiH-h1+Cf)AH_K3`7cKKXh*A^GwC2~!7&UDW57I`sV zte4|udCR>LZ-uwYJKwAGO1(8+wb$vb^Lo54uh+ZGyWG3hyTQBByUDx5yWP9jOLFQW zyIrqc?dmD@74OJ-0`bfRXX>=}d);SxTK4%m6hW)<%FZ*5lQ~PWCd;9nWISxYbW#Dt*eIPy< z6HExggRo$~;D8`HI6eq8o0{vIEzQj ze&NQ2rxuQ1G;h(FMear8eU(0a*ch%0Tf)|`E!;=r2-k-_VQ+Y0*cbMPL*b@yB-|Wc z5?&f!7G54+8D14$9c~G)39k!p2yYB;4sQu>4Q~r?5AO)?4DSl>4etx@4<8603m*@k z2%ij}37-w03!e{P2wx0e3SSLh3ttc44Brah4&M#m3qJ@y3fBhfg5Kc5pf9*7xGuOn zxHY&vxFxhXv@28+*%{g!+8=5S9S$829SNNcoeNzKO>LUoG`(qN(~PFMO}M5P4T+5d z8%d4C#s{H?p=Y7@q1U0Gp|_#$p@b%QQ*u-9ranyvA_pU5bjFA^Vv9H;3nMocEsiut z?nLfJ?nT-n4UUV9@jj+c}lY&Q<#|x&VYjH$jt1# zth~IuoV?L_`FZ{F(0K#%CS?xI8=N;PFEbC7mz&o&Z$Mr_UZ1@Fd4uwX=1j_&oHHe7YR|9iCPHtXq-`sw={d4nk2jmXS9h8gCEy(?EwA~4`BX@l#diCA6zDwH> z+cQ8*C0$8(EtR%Xm84oFRh72VQrh=@-odtw|ySVFK#AjDw}kOh+1afl%# z1QQ65%!DNbLNZ}VAR)mC{r+lCPR^XiobzVRdwtHYb#F_$OCMd`QvH74-v?3a)dsau zZBm=nsM?~&)VSKJwyEuELQSeEbwC|dgYcDFQb*J=bzGfLC)Fu+TAfj6)j4%uT~HU* zC3RU{QCHP9bzR+1H`Oh5TisE2)jf4z9aZni59CMkV>v9>$qzwFc9xnw1-aR_yssE2 zhKiA5te7aKiW9}D;!JU_xKNzSFXU(POSvF-$(?eKJSNY`{c@iiu!VA7&dC8gBhSgR zaz^f!N97T@B#+C(a2T8&n#-P7)C544BcBki#k*6Or~Rqh+-YtyAmLa#~(1Xx&&<3?3ZCER5Big7orj2V8+N3t6 zO=~mStTw03YYW< zKw>#Bm&+A$NUo7<ly)VdB$bqsRx(Of=}uc?8B&Iok}{%d6VL=TAx&5#X(F1aCZ>sN5}KqYrAcctnyepQrtb7>+bU%Ggk^i3VNj7%B8z=Q?9B zY}hvuZXmyyY?MEs;tyjJ)E-C`#YoId~LF z_|(!E+cif>i!g{ii*1Wz*%u~^;jZ^2m;pv zR*W+?v3Z?6eKXbv9(0%s8Dew<5uec6tBY-s>RP7LS?*W+&AJruTdTyMMI&o$A>9^6_X2H#(L%{=zyI8ucpYFLLGB~<+##CcGY3aY>ElCdM%1E3 zyA7n%^e`P~<0c&uw-|?Z&NINoAQP|I_Vtc<)9wULPL{QU-V7N(l8lj}i;Kst0vO%nusG?@agpzKBK|q z>tT7HEkI^R20>pz@vMV=(SgAoM3&C#!b}!(0UV1Jfc%n*@IZPo`COO}s4y8vlOJIh zOw{fKGEB^|D|s=G-4-IzQpAZIIU=0( z_|sV-kjbiHvon+}V_8$udu5Tr*=)4elI$0FSVy!HdECohxn_(C3XPuq_P-p5upXma5#)i+_83<1b8lO71r{4v= zr>kt-djSL*kzk`9{v<1hFM&p*0QL-LmOD*6LXNX`2OB%hPFN^6lWA0o#4haz z?oAkYHc=qh#B!4etZzkecOrN0i(`3k#_up#0J=^5j%$+v!p+oq5Nn1jIbJB|DD*|& zrkEqJs;7}V1u0{iBW*!;rrnHOthnAsAgol4T;sRA<#a~6S>HFY$Amqij4&~dRC4oC%E0FsthY$wy? zAXzXCfxU%V_QS1YfarmaOvTttmWd8fwN8_A3k3IRm%a;3t)O$}YfXW$b(I{1Fys;( zs+|L8>mr$RoZ=AZ9Zw`FwIO={q6X;Vs_#|RVI0BzEw&fLRh%6;XgJm=hA zs^i&Dg_6f$0zhv+7Yi~Fo(3MUh~nFnUaNRAMAP9HtxUBn4y4Pu2| z1FAEitiKxH^%uPDc+O-A;uZ^Vb%ur(@OcMjB-RYzm|8G<*HW~eAQ8xT|+ z&c1hMsMK*g&{WK%GDK8H{7g}qh&G00XwZdlRdoJQQ>Zv98LO5zT0?%W!UI!fCaSnA zlF)C?+7V&kaZm%m~byL^N)xL^P(J*B#u|8HJjD z$a{%%F({bIG@kMwSfb%wtQxrR2d(CEnrjmiUB>OOvevU80M`f{rI{m45ze@d*%lJ8 z;AM_FmaJuaxoQZTXg4^NGRB31xy0|=?7m)9F2?zDQ>;?3+VvtIuPDF|1sC#W6hU9c zYxidkgo|KS-zeu(oNGz$m=Sbuk16m%9XF`>&}b;JjC)&pjR<@I#23QiOtXS zDj|Q?4FS2zC{{DWO72TnHHLHq2_Slgyo> zBxVf+Iky+?`it?lEvOR(E24Jqn|B4sf07fV+1vo-IQQtN(_ArfjIfcU@x&mH9Q(WQ z1z2+h>1Q^Vj!83${HfWGLwTH4=4VEi&cLHKXMGx-SSS#?p{OxfXc*hfC3=L09DZ}S5Grt{)2PZD1EST1U?eOdsSqt3Q5=~Bw$&c# zqh8dlxMV1x@M>&KMitv5yvbJZ%(>@aGf~}9)+Xq(hnCBr3T^J z8kI}lF?VbvT$3`~s+8*%ozY6a=p1>2jl54h@|FVv($o!fZ3iR4p_v4h3|jPV-Klpf zW%4Hbr6HS7VYxhl^K?Gn5^df>o-w){MWBS8hRXTFKqas6Rr9p5mX~l=%oSp~QZQfO ztsbsexB?EyakfyfM=rd(3{wn7W-^y61W*y84Wuqo-(foiMx0>vLdLAe4?J#nsZhBi zcTHzjS9n-Bbeu`FgG9Q(?>eTeZn^QaptJT1VbKyE7b2EPA>;y^gQN2HNkP@VDg*>0 zCE^jHh{@f%z#mbxBe=Rf5|!-?HZ{qMh^g;UTNHk|D@gQBGXv}rC}0w+C{iJHk##Bk z+M*fL6cacEM6f*|f(?AmkQHabXRM}uA*zaF=CF9ILyENZphy_&F1Wa7?|O%L&2tQ- zFtz`vsP(1srm<`f0$|*t+b;%?17p@wGd2C@q6c?zf`}G7dTsj*sJ(k_OR*3%7Be9c z)fY|0tl3ahkXR87a|XQVz^8nfx3T-}OFD@k5rofLG!b3hG;A*(wuzz^J80XAWYL5j znNX2|XR-=KQboOkE+Q$o-9TGDb)zbt3K7oi=k-Bf$(AV&a81J|rJXay=Kwf9N5Juk z*^V1_nlelcYP4FD_5GQ(3B1tG+hKQC7WHfv? zpmbDWlitA<;EDOnF*F4wQE#;#>ihNp90?W$Rn+7)nJh$7Qln8vFcT*vnsg~~jjrK1 zcc?vTUBswiLm4bk*AJMfc@%++J;cD)*+Vj%slluxM@g2HJ?8bnes3fr+U-#(BT8ww zz(RD-Y7ALOxnmdUhczT4L^Br>Y&qn#?s&#V7@X=TEHG?DAGDy6i-rNN5F-GG5)mKm z^C>(>Vu(eIvCLR9TSK@KMjS0eU%G50GP}TQulQ0Kop+bCMUxpPY`{YNkgeHGh==UC z`&Jw`8^_?6O)-Gfri~5Y;=TUhY=0~@hUsQSgkHwmKt5*j(p{u*;jb@%w-NIy36obXhlNKlIaZ3i8?+M)^KjrUbFIe ztwV-HaIj*q);P&wveiz66gu`8$WDL&BAOLD)TL~;22*q1(a?Bz8Q!;AEL3gk6>1Y- zTZ&L@&BUBrb|YLZCse{@%>n#0i>qknYiS|~tTd^Xvihw_;G`vLXSx`yXFX`aY3K5s zA9!k!THg|@jicvms#Zq=wYHTeytUKF3DrlkHA;xrlu|>7OFm$*!KM)K)=+W^P7$V@ zW3pa@#6hhQna%2)2He;w+DF6TE)dsRwVoyDsnm9T0|JHnwO);kmTFor#G`o9*0^J} zm1{<0wdQa&fzbxrj-n^PW`n(x8b;M>W~K+vW_ut@koPe;Uxge39@Z%4g<-8_J*{P! zvU6$~!LjIpwd-I#yY)j}Q7@1TaaD_2&cO_jF9_9H4nwBMUDY*?cq~xnQ^WL$6bZ4zIORfR zcigf#e5PTrV~pdv79cQvC`&MQlO@LK{JMbd$|36Z8J(nTQW)k@-LctrfzGxMe71w8 zUogh{jEXsN9ySH`MX60PpP6;df=l02V&i}gMu6J3XXNOb%h)_}XNzPg8JBwxBSTj% zUc@o1P~7+DiZ0%Q83OrYgFChy+G6^okMzup&5`%eqqoka&4hUFOqjDSAsF$MfV(;1 zDSfqAbe4gW^c$B2}zX2AjJX} zapKzJ3`m0M0Hd>8)R?Eh?!?Rr$E28cXpjqMj#WhS~NARu~9IfGwQIx3ZEgK1z>lr@aan@0e1` zlHPS-v)NQ7SG0iaml*4DsfoiSjYC%|N)Uko0~9aWN)e|MbCYt^P?Cos?8;^>5#Cew z++X1dkWSOzaYE}JpC?QjOE6_Bi9QMs8Hb$O=8_J01u6g`lnr3PLoAFjrMA%d;YXL0Swb}Da92_Rkl#dmoD_RQr_r^R!RqWFzWZ5 zG)tw4#T$v2Ql&FGQ*zTRDbiyd;_(@~rDJE_IWB4G7MTuoN=IxPIHsf06`cmQ>6O4r z{nA;W3B1!mY2W8|=v)}xC<&Cwqtl-Q_w)=nrxzuk9@9_3c8!x#%V8sHVMUpRl{5L& zsw&INSEZ3oQ`X)wQ1{Al=RWXH4}gDqRL)soV4(K!e5=PejX}Ilb;7LO8I6QPh`mKM zj%kx28ZhV)4hIJU?g#DZP=DjMmQ2^6SB@X&vZh%HVF4bXc25@W_s$$H54EYShOL-09+hvjRIEFM5n*0CEf(2y0}0<`)~HGr`ys>gW4JFCR3Tbwgko zUtpuW*%Gi{uxZR`8w1mrG*NzyFv*`t7on5<#U0_eLC$<*7dexTRfRAG6*Pq_L{acE zG@=I9u@Xqf`vqQXM>NbqA?iOWbg^9haHb#Q`Y;Uq<9vPQAL|Qp&&)r*5(T;+8hd9F zDrIlbC7_{Jyhf%_HRBO~o~i>8byS6r2Jlf^Kt{#*l2op0%~jx|4y*M@8~CXGY7QP( z2QCNMCVNy5Sg8}Yq9(6V(K8^Xo`WgaE8wMGR3$3zx^ys(whIDYstSmyQ|tsRckXd} zytY=hTm>{hP(7?2)zs9?Pdxy3>V8cJ{M6ZWtQjb(e#=4cum?LEFu`TLUf)dmB%HiK z->6Eb=@-RM0fn<;zUMjP&v_;Un(^pSkJh2(u7?jrtqs0v6xq7*sB?&Cs>dCz9_i>h zRU6zf`}~d(2LorO(oCjt@r>Q^3jh$50)H$*h`Za+MCVI(Us^_HIsL!71=^W`^ zKalHr0OJK1@!qka22K=}dyq5VEA$vcvDcB3o}eAEg5_JiQV(pej-Vm4{eWxqrU6eR ziPd|!$>7~}%Bg0r(`)y7y;jeTcY8+9peO0Wx?ykBoAmOAes9`4>z(w*z02Nd@1l3r zllLi~vJdqq)Ojy0^bLKT&a>NB_Ye9$N!!nYd+N1WMU6HcPusvx* z5)>G>MP}L_5iR>kd+JFB9MOq%(LQgVw$Iv^?Opq^Ti&sm6djViYG-UNgfXc)SPbgy zcIwPPukI*27Qd!5rL~>C&VEMkbh^e7-iWyOJNw>)4lEsZ*oi9%84j3A9~DBuy~v;& zvop3~*JIj;kA<1=dFneh4+*s0q+#CDo;dY{wSD3c1KQ}!{VW7#<|hlX4^H5F)+8R7 zpZ0d$;dIvdVE9D~)X_dA<%XJNW`7DG5pm)(xn~~e*dJ^UC?w==o4KyF=ZLhsMk9R) z*WE|DaJ%U(7$qQuM%!n`nG%|6dvp^=$C_wcJq;w;Zn6%;7|ym6WX<4pq}wc!X{*>A z@I#C3DCNRYTcKTQSK8%vwQUy~tv*@ho2`zs)dF8CFi=^BwmHn5cFC zq@{C`)^Q6rPVi}K8b7z;^hxWq1)(nrsqN5FtVyO_9;_(Z+?Y3jUg)~6NW@7rnQ=G9 z6r$5^NFQ^x16<^!JLz8HPQ7Gv26MRFf6+bfo_0O@Fb})0x*3>{Tz1=fX~OvwJ$cV= zOjs}QGD@>huj$_HX##}3<1+QleTmQ-&_3=ByACBwAM5*F%Ah}j_X9TAV6&$T0biD| z_0I@vpXl5BWS{C^Q3~ozuuGWTVy7hqaqKnrvo^P>A7O^<(2F{UuAu-shWxPab`N(& zLB|dK!+f2Uu0A)^x(3$fkg$}7hsYIPbhn4~Van1OmWP#LW0Z&K=%mJqj1|Wf7@dv&fW3E!VzJ4ZxC!?eU znm#lZeNhH)Sj!Ae&U|pKl>|b#sVbUYRKhVlV|3Cvoug_DJF5iB0W%yNI8;xN|5*)J zhsN%Im&18ps{Xqj&f5}I`!I^?IATZ+sbR));;qGWScW!tAhV@|b|e_@M5Jz%iP$ER$PUpN1gtKS$)i=uIm(KzQPPib+=w3uBlqYmeuehSy-;oh_N3?y{Z24Hs`(2e zk1K2}8A_w#s648Ssv`s=;M(ZW+z##s{iKzEnpuf!4ErZ0$n9mTVB-E9t@l~X5e~wQ zezV`|xBH!bx8Lvg`jTVNKQoQ`M?qIcaHsTOkjgS*$Nfp4rOqihG6i$`dlc$S+E4p_ z!&x5z9&*l|2~zHuRN%6t+)6c-Vj8Ly6|Dp115rMrOfWwc5=yi`oMF2}koX zG&8AvW*QjP6FwU8N{HxhqCT;yG3s@}rpg0GJAhH0b@-Y}L291EXP7xw?EkxA9chx4 zc(X=u!6;CxTWG9_*&esG+`1AR-4D@AzEwVuA%ou>zWe>z`O1@Pq~Xg3kcX2H1036U7%klJe+p` z&+Kai$)VF`OXu(?S(X134D}2gXH5s6EnGEyUU@62Glx>aOJHML^>8B`-5-tgRD_Py zCBd#9fdQhC79@<^_G~D}=XrWq5e6Pp!Qig>l95A@H##06BO9h46=8U!8!2oOVj7u8 z6%&m3{f5zj!8k%kmXX?vjqs6m6m;80_7O44jI316y3c0< zP7Om59!r^0cuK)oQ;w7^mG!A2{**TrOjT_AG@5d!d?|Y>Xm-G^lpXAL)8JOKC*@2L zsicbtkSQT0rVf2HX!&p{J{3j+DamaPCsWzfuBVc!GN}|r#8YRm$x=>*Qfw-c8gV%S z=WRMgD3Ov5Ebcas%d2n5K~3Y zQiD`Cb&=|%dchhJ<%-;Ss-2pq#wo}?N~MuuYGSRWSbRu^t@TtR)lW@QrzwLm&Q7lKt2^t`H4{u*(}A?jBH7RE0w$y#X(H`L7?5yPA`-~GicT>d zP8TSr*_ZaGS&)MDrUM>6O{Y1=>OAs!(p;KKJJTsz8`=wn(j)%~S4q3lxpaxnr=#hU zNF3y6duSx>cOIGw=~FBpuBJspB7II2(^-(FjT@8cR63JBww2Q5^f=8k1@|DGj|H5g z^hg?}dmwk)P1n*LF~)U3>b8*{xZ7#3&P3wYetMdofIM!;aSlfJpv+0S%be4B$>_UG zdypEN73{9F^hNp#jA8D=YR3F|X^nWIdcl*5_JJRv0@%PhsE6@^ zX+R8Y1IxfQU1HI0z30sC(cW zoYHB@GXNtcgU}#8u!b6@zECnp2eCl{BdAU~)L z3WF+0e$@xrL21w!9E8e)F)|qJi{k-@pALG1$>3ygHW&{2gLCZM*d0s}o|2gJa{VDV`E}%1qC(%@XOI8|NBO>|@9R(E`5+W# zMi!18dHeo;kr4agexx7h2l%|BNMBe&eW@R^B>NHU5{~y{{a`=QkM>)f;7#{!@t|>|T!6A5!XUtQ(GIEeOOpwlM)Ff!7th*~6C9)C0 z04I(U&7cmXY4wS)U`Xr>#)K*1*LNf41dJaU(S#+@u^b0#K`gS+XZESI=6 z25>&17lnj7Ve^6JPr{d&B7uZIQ45F(i3uh`i7?3Y;-V^e3P%$Au4rOvlQXeIJaNgG zJ&8mzVJD-`R3e=)xH5@sg0r-7i|J%^Ix2cC<6Lv-CvXx|h2#0rcuzM##^dQYfKA2- zb2>g58*~i`Lv(hGb2DcCKIaUo(h!yc0}ufr@Ck-SB2KC>o3>hshyw4@s620arn z$FOu`6wWjvAhw)sWGIx*hJ|RZk#Ce5g+{TF;q@qMDmNN}m-iatg@U8fFp_Ar+Bk?_ zaG{ttmI~iGjsB_KwZ;iRaYn z8~exL7p5sVmd3Vmc#MZydZi~aj*j#8*f{G+jyVL3M~)NY)VMItjkDwQI6uyeOXKpm z=po9eu{f@cAx~vo9oxsOju<y6NW~(Gd|UK$9=F`crYH0+x(v8 z+*LI3p52B~e-&<7PQ`Y8FWRX`=r&ivy7eB&>Ei1GQ@Q{vZAZ% zYC4;~#!=<0j;Fc?jVl&9b)tbw%a}e4xx*JWx@UfmOEOWgzCDHDN&A<<7Lw52&T%J za=MJTpkT6G01y0JxgX7!Q@Zm&pk7#2?xLTC?D zB>S$5#CamgUxL6vCVG`fP^#o8q)sZ6nxxgnlTdOu2}QKY!{kx&AW36NVn4Z;bcG4z zIC*M;lem7@Xf^4QV-u3(b%vxqX$1LrQxZ*Dl33CoIvb5$Otj><7+sFeM^1ZJ44J** zJ&$56A8W9yQN*nr2g!zh&#M|kV}ngI){eb4lmP=K?!EEuSUv8s2V<+D9gQ0H$4BGC z@$ony1{f^|k9A{YtREZ3#<6*98lz(#35UEwmyAW=AVKwPd+3SPIdM(62|p2l+v5fg z7VpG2VLOhFv(uE6Ca&Y^@ExI}a=AO64s4zny&XT;vfxC;PM{O)Buz*9P{hd^b>Q?9 z*tkH1CD*wBi_81A$!*w2cU!BC76gSKU1cYPy#ZZP(-6>+W|A)Is;mf8svs z9(K?4Czj(b+%1T@F4C=-2~poQbdB9Um-Cpq<}TW`bg^zu!n4&4ZmSz3!TwUJ)9rS9-GX1K2PZ&_;6;2<(g{AjVZgbHq93#atb9=@IvIr&VVj^; znfdFV-g^J-~Bp?;f5uA(^_sM86R*V-D#bnWp77`&- zDFGINCOTX>L7GQ_ad0R_ZJ;R%_CzS)khh+&p@P*$D`{HaNbJ!tYqVwj%>>P-b*%(y z!orvOc7g{_$3;gc(M|Lc4NCH){ne11?kCFbL89Wm3XafW;utph{ZiXKO0Y#0MSyfF zH3_gNR}3UYN8s4%wT!w>Frq+D&h^ZsZ*)u=A@u}AX(pOUk&2QET^qDm_a_to-lQiT zOf1X@KLQEY3>%~N+*bcF$hzt#COQnCI*>`uwF`m~h`nrdnF%7uR2WaJi=Nt40TG6$ zy6FM9K&JYsVJf)%b~D(bpV7^-*OsZ$=jHY>v}UPcHQGK3`ol3We4;WnTo?V0U+->; z-D=hAw!}@B{SxePU-gyYL%Kz=PMuvokgSRUG*Awhpwa2_ss_4{gD7}*2kLb1D0mtz`c*wR)uwmDVXrQb)L*)c2tns6xR9?DDqujTqKr!3awTmoRVtNg zr9<0owMvCnWA#d-64ciNDxz8GI$M=?rBgX{b}MSWSLs&<6$?16HL4Woab;4eMW&UL zipzpev8jD(oysv{N>1r1HDy7%-7$57lzYvcMZ;5R8kxpG-aR@^PUF*PG&N06Gt=xe zH+3iXV$*Wla!E9Be(G{^Q(@|vx~JYL=JQR}Y%%1Y=xK2hm;@&wkZz7l_I1%oY!aU& zCP|QWPJ_&|;LlESll-JGDNah09$B7LK+~Z%sZSap3Ei5sCmoQ7?oIlW!DKiYO*HoL zWHOmfP9|sE>Evv3KDn4&POc_$kkW^y%Bc$E^wm?%w4>8b_on;P1CZQLQb*I{X`emr z?ONg9DT6xjfUbx1^gRREA!7}+0_L8nrw#|bXb%H>W$>Q0*FKXXG3`XV29eIk+C*6MD9(yXWb7dp@v>CW?u@Krh$}^};==7wN@%s!+5S?_pyL zT>`rTz$SaJHNZeOrp@V)?v1*mI-Lhw1e|y&J`if8vX{UPT?cGy+Maf%y=iyapN^)( z>0mmZPNwyCqup#tObwhQbS=nJ(-?P&!=ZomktDTr&rUa)6h!m^%KLS zL>VWhiFtxfEE8-3aroqn(pYNkh9w)9Lkb9js&W;y3#lOuq=ojNedqu>gpQzN2!?bJ z0_hnl%y z_y6~5MfN1P@7uxs{^z>A^dqSHau=i@N_qvboZ4$e#2+Bm3^n-l-Og6w&4-T$|Bd;ZU5-@W|}@c;EsvfEp8*JQFEU$<}H zxOV%t0u%y@zIJ=d2X6cSta+JuZ9^8k_RJ@u;QIfloGcFB|BqUfJ^E|ln~^>G^z85C zwd32hYcI&U;5IMTk8WRq&+}hE{XD4mg8H3nTe4@b{pj{*z}sI1_4lBz&E3AeGxy4s zd*{A#TM6Da04~=)CzJi3*X{Wq_Y=m=J^aeh-1#eWb04zR=cMn9=HB(f)m-{}KRL(D zJvz5ezHW|u;Z1Y)N8dj8gY);z9X#^kx!av*=YIILPt9R3`_s8^z3WSJbKm%@xmSPs zTXX;T$Un_V#vjdbUzwl(mD}6%zxnt*^YJgY{%@LWGy+1yG_s{0$3)A|%EZmxZ=re=)n|F`rZ*EWLpZUbayj*^3UM9OW z|E{|poL|`Z$$8nW*Upm*kIX;Qd)>Td<@NK=to-c!g6xg+bFw#q>)t$nzWY2LN)2k(2?!j?g~5EAzm-u@uGu<(R_Vd*rp%`0(=&f~91SEatZ#Tl|O7n-;$ke`4`%KlLk%H$M2Qi??JSTzsnW(M1?} zW-;^rPb{9j^-mVxo%`bA{Aa(sDEou2Exr-{=3?Zv-&vIP{$cU4yMC~E_bYz1NZ&KJ zlzV1rX+^fYv~p{8>DKo8l5AmP>9N1NYw0I6_b%zbp;(G7YnEh>?JceR09jh~p-Zp% zKDqSjf901#?+Yz`;-1*j#XZ@jjg`XEmmBS+yRY?@Ha;<4GW^oz(#{k2FKw$IT>5>@ zBTEYUv8B5fUcV%lya9yR?|EWLMn1W;Klje1jpyDCp3vXBB-?)9l0x_XC%&-shClqu(myQz^^!{Vw@WeTyGvhE z{marH(YKe-Pp>R52ph}f#@6!NjQ1{Uc3!qDd+p8TAKs;1UVGiqvUUzxj%=HksU>{* z=7Mecfdz8;yAL>*WisdT!Un%ATM(8HDF5=(jnHyvS@eEPLtUjF{}`t+O-O6^(uyX4^qbo8kz9M@w zv9c{AR~BT{icG<*$YjjQEg8FV^QLn}R^wN`@rR+6pU)*$IbDB=?P=U2WZ|I$jt^o^B=@B7xuufO7-SKe{g?UlLm z!s-+7%Ie9lZm)hEx@YyLTQ^q~f1_S~4)&pn-4 z<)5yt-h;GP!|xug^1pep`qFz}wVK;_a8>c)M^}IMg`ZuGFTHh@`Pw^HZ$9v@)jwN# z@2c$9`&MPr2Uit0e{J=FYrnbrr0lm=W!&$qZr^-n_10YM}?X2JY#7owDfB3TXn?J2sFEq64o_mhge@2F^-@3=R{_vY^ z>%aPxWBnS!ub+J}xc)JjwEp$)<<@_1p}g+>TzegT`*8iqCr{T0;w#tJ-uS@!zdrED zdhmB2U;hmI#`V{4|J?esH=kI)Mg7A1HvLZUIYD7ht&?-_UVrPKe0crZ@kiG`@UCaq z$>08ibwv5c>)y|PVcq@wm)4Af4O$6vAWpPo={Jo)A0jZfY*Y|Jg0Hy#-h8?TJIH*lG_aeF> z_}G)p4Xv!RvA8$fP%ch4R>`vs=%u$d6t&lEEZ%x(W9R0>8#`}*Y-2(8x{b$5Z`k0u zw`{!Xt|vDB6nfVNeD8ZUZfQTT@%YMzH|D~>xv}%mZ*N??`SixZ6VGnQ7JhHz*8R_I z%)RE58;aZ?Z#*V@e&gEu=Qb3w&u`4#__K{2^1p4oTK46QYxjL^LnixQ*V|-&v+;oQ z+Z&&_@x2Y%-1j&Bb>W2#*?m9S_|nnJbvIcR*MH;I-Pd>Db>H=WT~l12dt~>z zMzMGOD^DC>KmRH0`sY8+UN_2x>mOeVT>rp7C$8VMoVzX+s@FR|-MfD8YbV!#*t!4u z<_iyBhhO^G^?%{sc>VZOZ@>QEFW-Cp@z{s1Kj8bw^(Q<}Uq3qg#Pz@Z)@QH3Jn~DS7q1Zi}uf6Z(n>R<$=4X|A zoA^`u&AS)PoA>1K&Dc*;n=+2sls)6vti4Luy!JZZ<}W@F+Wh=Ma`VB5a+@FdSZ#AU z-`m`t8*H+#nrx!7)6Jdt-r5w%2R5J4JhUkbJiN)~erogXm%M&+d-&!}+4fsDUnP6% zCO7xQ=HlEhY%Y=S+?;>rsZE*e#hTlA&*qBky_+}Z-oGh(>{mBs7&3;jeB!|IxqQyruZorflguoAu>?+9YrN^ClyE zVe@AA_9i$~+j^c_+S*X9Z*4!YxwZZ38(Y_QUb3Z`yJu^Kyl)GZ{pYPG$(viUIr$bT zQ*6m@LR+$z>~4MSSKzH%I>Q$G6t?x;v(%RIIoH-*@Ahpi+!D8B?+kiTf3Quw;un!$HB+?>8;nu-mrD;{cqlqk#E_0On738 ze&`ps7C!mQTaRwMZ)@?Fer;=>{>avoil?_0WY28X9(-==+QuJjm89plPPae5H7EPR z))Qa)GWa-O+4^1bueKiM{&wq{_^qv5^WWW)E&TnKO!oI%3*z^;($BoG_57{dTiZ{} zZLj?-c#8Yz_V)X{_ifAnOuoJS^4;y+f_D4vS08P24}-G|L)=TlMkP5kB?rxZTZZ@+pF+TZHwYhZ?8P`hHY8>&D)>jpWGJa z-m$$+{?hhKZ~e-4>FXcd{=!fG7I;d1di$lJ=e8f;|J1hZ!RNOfzwm|aue{_hwjY*# zb$fo}>)V}szqKv<%fH{&^?tCubnA!P?|A989qqf8cVstLc5dCXzVq(0yLY&+-nX-i z+}!#6hxc~AiRgA-m&JCT6p5X$-*oKAE`*(1+n$|UgTT&f-x}X}N|xSv1zp-vY}R*l zbL}13PIu?^3&S0SY`n8VPItCtCp+7z%N=I^RXd;f*N1oB@#Nz>w-$e9NB_WEc4P}b zw{uJO)}58kFYLJYp4!>H|9v}OeCpSCu4zBItW>`JGqE z{$xi6>NEe>pYJ5)U)}k}2mW>^ulw%K%B_Fck*$1hhkW)2JM}02ZAbRbYd2z#uG~O` z%^UsnPuy6OUUuV&H$yj~hX*$n$>SUE6ihedH!U|l024Q!BlsIPu6b@eyDi>W+m&t< z*u;%jiJ2R>-dMOn-YVUAW}$jRc7OfG{LR)4ne4^72kqaG%?)n+-@Kg%bko?j_hlzQ zsAJL_5PFGAAoOY=bg*UFmSx$pEf?96>`)RQaf|7_x6ndofT8!^hhB%?d+)uIZ)e~= znEU45``%k`eXR9c`<$brE{8`Rv~B zk%fh4B6k$tj(qm)Wn|$OK~&)R+>hK{PDVWQIdaZ$E5J)%m-_m1*44o1C+8WMG<#F(gui}RzpHlGz0 zU2I+yQhY&_F!XEGg~Sz69k*?WYV>q>)a2KPqs%={N9_z=j#|CrUX;1Pi>Qpd@1xA^ z1kuRYu;})!BckOM%0@RU9vwZ$niw7RwOVwsLA9fWO{CEsi`9#cDAOQX8QnY@L7GPo zt=~F2o@gK4H%$>8b6y)=Zks9kQx`UR?C{KJWKhp&WLI`{g}Z_1=qH1s2d^0w-K%hF z^mNKPOLYfemLjX+G7*#lx&(a;!h1dBmLBV&dS z92X;upBNKrH9aPx#H<*3o4GL}!TgwL#gdqnq2)11{K^>7@9Sdv2{*qldvGX97OC(gzQzg~?g(fm$Kt+WR*`BPrR zWQD(rIe83;jXE3=E3%e}t<*L;c2v#e*rnQPvC=kmV#gmyiw&t8#10iUj%^}u5v$6{ zh!xlF96L1BEf(2?#n!5Z$F?p*#+HjP#a7O+#v*O(u@T`+tSf?xEsnTj!y~)L)=kTf z9V_e`+tSc4cF^@9u?=gFj?EB_i;b=|DYm~TKbC1z5L~a3A{L5k81Tia&yRk74ZZ2_a8l*%R2`Q;0v1i=IRO2ZhfHW52vB8<$s@ z9QU;_HLmbe)40OI&T)lb^l^ndT+q&mD?~zZyCVn1rGyWOD_&+qTtta6an)ic#)$-z z;l6tfi#QD#0yB~+g z$ImPszepAppE5Ze1dh1RQ!v9VP^6wEL-4hUb510$cdqM;Ry&(P*T z2130d{zQ161Ub?d#{LF*4vc>%A{W}Z3D*k;CT#mUE}^h+4zMaAFKYUqC>4bdhoDzqfMTgxZCTGl72Ff5dmBIpmx{|7uDob;)1QqsGbbCQmKS)Q~* zus&%|%(kSO@q3a=_1mA+X3>!(q|~vbh<+!NB0{HtGfCm`=aaV2yO|V;+=hMqA29ZA z(p~mhQlrZsk_!3a$%QSVlHY%>nEc^Yo#b$2RCWy{H5n-)Bu?(Wqe=49d+n0HUQ;K} zr3}gETRD8^{5mjcz`4c?jxy!C&&rGW2Bql74nDRCBh3nBBuo(kh6mG zf{TL7g3ri9WK~$JiW!2Af)9cYg+H90{{OeOt2ZiIDl4M<(fD!s?#u2yZYpzO=A+C7 z-5vCNQ~czPgnHl`e+lzMYrszS%GS=;%JG4CCF)?WkyEegzvxZmmf-E+hv3v{$9uN3 zcDDL+r)8yP56xbZeLmaiT*n>iIgbyG89&B6s^RoSz2*(U`ZeZHjbEdSPo;s`btQx&*D3*?>-ESOv z6y1cK9;VaopE!A9{>0-GPfa{M@$SU*Ny(E^Ce@tOVp7{lnUk_7g=P$#F?_~^8IxyB znUOzZ>Wm#TcFx!}RNvI#)c#WkOdUN{ zG(CR$Tt}!NdEB^y$XTzj$?gnK5@aOw@BX;^a?1hJ$1!=j!PmVw2LrA_6j1nMlc zM!%e#IBkrhpogk=Vb2O|d2WZR1T&h+Vye5bJ|p@*>@z7pG(<9Vh!6EI8{K=-ZpT>l z#@rZXWo0wvT;+UpJX%VVXWyfS|vZp%TuJ^95?9E^{AFi(Ex+@f>~5#GK1HVY#thllOY>%icbH#^=BErTHoU z%o$7lIsH!#csOupUd6o1qt&A|Q_9TxG_mLj-C1u}z1{Wp)Z1I{_j>#4?XP#B-obi@ z>K(3kJ^e=d&GcL8x6|*W-%Y=len0&|`or``>5tQ&q(4o6mi|2bMf%J1SLv_Q-=x1y zf0zC~{X_c4^iS!Z)4!yDO()Xz>14Vg-Iz|Lo65gMVcyHEKWN{;=C)##DDQD_NH1y6`2Ml|-MwG+ zu0B09Y@vUUqK9_hxWD`S|DtH+Y@?{tTdcoPfDKMo4^fR#Jx5C`t182kw9>2Om5-Ef zl=aan=v!qkbiHD^N~FKYL;W+hR+*1&!M0*Yv5VL{Y@d3+x`)oKTcw+&E6{D%9nqB| z$`EgKC5h%lTcRDP8nz(R#86@&@t#L7&;U12V+U*u>F zIjmbvhfR+s6;4VVUSe|8rXRSz=DGa&1;XLe4u zHG6ROtnAs@`Pr+p?@wvxoCURb2b@oxU0kzWk*-QE&b8I`$<^Jp%C*b&$Q8w&bd_gM zxgNVx+54{TEX&H-?og9$g?ey3yOsT&y~ti>OLKQ1VPG=1mb=pHg)tg(R44IH^ViU) z?6BTO2kWTO{k`Lpqa@P;vJOm)n;FH7U{s@Yqim2JyUIpeE!F|n+txkSWY98MVf$*U zVvn>xu}3-MokYKr{f_6K&Oetwna%HUxyQ#IWqKy}JlwNVuPZ%fI@(*~Ec-2|EGsP+ zEYWl;x6?h)Je$nBhq=Jv|X&z+q+KlgLr zPauJtlG8ee%;}x8KIcYGdail+r#?xL9Wc1>Z+(~bwfFAcdjK2AyVB=wpT{8a`ZYJh zo7bnHPt5{pL0Uomf^G#^fx19bfEVZs=mKkjqo8|1cEQ}<;e8_dROoZkJIFiQJI}k( zyWM-WZ(3+X{^+Dnb@_Z9~&wM5Q&wVleFn@}_ zY5&1PmkixG^uW*>!}<&xmp7pQ*8actj~ei`KQe$C&~8A+fC>W^4wyRN>wv_8p9j<) z*ks_MfyW2V8n}DlyMa9htr;|T(D}(52VEa@F>iTrZE$PwY_L>bTLU+$9p)ODInpyS zc65!=HAkbP^T*B}yKU@&v3tf|99uXxaa@^kDdXCVTQ}zR*uf)T3|lo~%ZU9W9*!tE zvf@bLloPJvS^Z}9pS5rNxQT!F`K^^&BeiL&IJHh{TBtc$F&w~x50*Dq`Ats<6s#onsYYEp@}y0ofT=B?{Z^VX2I^w#sHdmDPIi5qz9 zgX=)9xR$i0w7S?UPW9II*72rFn|SL;n|oV$4Z&7kpIGi~?QP?2>s3oKytSpZydcgj zZs%?9wRmZ-)ob?##5Qk0)>v$kG!ZuyHxoO&jMw0GdR<=D+tCZ@{1S;&CY4INcsqML zdAoYMdDEn*mzAl!O0U9;d2z2xruJ&QT5nyk&Rb7fSDG%>dqI>{N_uPfntH2yja~}= zZ($^@#0`CNaZ71yaVu#PUn8Gf+89#XoB5oQ*3#y_7NAnE54Hle`g%TEV)izW*7r4$ zW_r7OdwP3#dwH8mv%J~f-d?G{5BL!D_4bf?y*7zmlIv|Q_Idr@fVZ+Q2!hg8eGZ9L z!g+bG+w1Yl#C3cINvf~DxS_O>w6U~-uZ6Upzqz!Tw7SpXCnYXF<9GU5Kj$ZKk8RWF=<0F=5Oe4iaeQwf$YBwfvcqI{s9@*pG@yu~u5wkBOUk)zT~pC*%EYe|NvfZ;)yvHN~0!9{yhb z+Tx!6TH@aRzJ6Yo>(BQ0@oObn{v5y8AN2eDl>-64-*1#w3E+}yfjVNHq4?_I)T~&F^FAB0%CDve`-J)NDIgUgv2bhNG0OBfqDVGq(Pv5AU)79ppzQ> zoW$t2`UxrJH~Gzey);{<5?lQ3#cm1hujQ-lYb4I_xASMobmA6biNCAV8wtA#hrXzz_FmSubU6`wUuT_6+Vqm>r?vFKFrr%s`7Oa>wLJcofL$3eIO|a9tQ^S zH|PpNk|v+o*G)|O6yo~+27X@m!N^ZGF5;NaMB!^JKzaq2C^ZIjtm$Ant(RY zM}h}B2h8Fwfv$mWf!-2DKpE&AkOx`?Y|=J?)`7MGkVOcz3pk|h1B|poz$xt*prwrh zjRQ>rR%z2fvp^?t^FWJ0%Rm(=5pYTM0an^0&@<30kQK-dbPw>->cJX8R$MdamDCE> z4yFb(rIKKspg1TEc9;4jJ)}NaFKN2iE!G6J!8T$%s0+3g_mX9ZiC|AzUum_VI?yoK zN6LxSL0;S^P&JtD&k<{+da>4z`}@dx%VfbUX`^71VB=u3VDn(ppghmPgZi^(74?4JAFLjU7XCXG8xR z;Ye?|t9_vVoru1O1<8dzAos$4eX!gQ{U3xQ0q6%H+b|E7g`gcmLMiC0%ShS`+*2qfFRMo!g()(@xKy67Q@(LL?HYP;y)l-iU>l>;aFB6DSOwz@->K1 zz7DR{dT0X?8{k@RL?W=Q(Em=_cEl~%gG5x@2V?u-^SU1q2@b-%L(u;l1c%|dJp#`; zVd6X9i`{{s@2M>O=K z1>r~x#P5V+1tCE^^nYS(BJ}^B5WJg&pTqwec`}?(1vu7%>97~sSN!eKth%v z1*r;S|8p$iHDJ9OknLOxVl4qAdkI8H9cX_e0^Xs*J7E|TL;Q|N0@qXu*GC3p|AbJQ zK#0_Z2mN|*?a~DUkp|HIE1{G|F!oPKX$;H$38E$fMA#IT1EFRDB&E3^T+kBw zEn#1+AbzK0Ye7iZ7W&^w$$&N>ZzmWi>ImoFNr1Sz!q~qiLhwDGH>8B?j>3Et?vqLY zA8u#^{WTE3BhW${kmC@)5gsC-{hf&KxxN#PF!lohLJ5RpEpQEKL21M;KoXqrEV3|% z6_iSG!@fMQ?@V|`y2Ji@3c~wm!?J8Rhu&~+`al~%b6`%c0EzZN^usX(V7(x0GY^*k z6*&@uv2Vzce$ehGNJ9od{1*rY!t#OeYz%_ zIt=kA!jC{7XnhRgPo$lI{t1CF{0~^?6mVJ~ke`LI{{|y3z`XB-EZ;^ z->Y!HuEF_S2X4SJKyVWxAiO0&Qf|X}-GTlcc((4s@!x|s5Pl!xeV*!mkgXioy zEPoE`ybwfWycV>s_FfQ@e}rrENgx{f6@CUF!e~S&6d_^4a3owP6ciWA;ff>uN(lww zWniqV5PnA%B7$-dfrwv(5r_!dB4L~GD2U&Hnhc?6Y#j7|5RSw{{|E2^f^`#xB0;hc zNvQzKeh`jSgnmV#Jd^?vkXMHI6Dd`M^62WY{~AIhye7n2LPS&>=KNRW5vec^)RI8_ zj$8_DAU;hfN9sbmuCN%A4l!LQMCuEX;tiqQP?&-=g4hVIM`MVMVQv$MfT$_VYXUs{2;0nq zex5K~&=2B2M~L)?(wNC2Z>@?CTZ8e~uh^4a@)c z$p2D5_6D}~?`b1=5A(he3VjsDhkt=<{S}`5LKp|&KkR;pFf1Y^JPZmq!jSOdVFGzX zSQ=6)4Bl`3J)+W3PgN!?gp>`7C@F$DKOrv<{eOWV5|)1>9EpPVUlAeEFcuvqC>96t z8^YF!(EbTQ67>HYbgU3oO_&1bRT<_3p(ihhdziUEp&ib0AYis6c}iph#8im8ejikXT6#XQ9l#R|nr z#VW;G#d^gC#YV*@#a6{O#dgI3#Sz7E#RBl6N-0;iR)R5{vYoPn5=_s`G@k9^0e}d@~rZL@{;np@}}~(@{aNzI-#A> zE@(H<*;1hxszx;^j_ObX)uRTKLQSX{wV*U=MQx}ZWl$&TLRpkUdDMeuqTSIRXiu~k znuTVgz0n*r7xkikG=TO)`=bNUf#_g#2s#WMj*dV_qNC9<=vZ_dWE4$6C!&+k$>;QTYJ%k=ck3c5YG4up_68!@`g`P&wpy$x@=mqo= zdKtZfUPZ5=*U=m3P4pIe8@+?xM<1XM(MRYr^f~$heTlwC-=go(_vi=oBl;Qrf__B{ zQFxi55~{*f;i_V)2vtc{DODL&SyegJFDj9$yedi+t%^~_s^V1fssvS{DoK^Bs-UW< zs-#L$RZ&${RZ~@0)lk(`)l$_~)lsFY#43qOs*RJ&DsRKKeZs1B))s*b5nsQyr$Qk_)x_#xVoZkBg`B1OSOcs%))LFW+F|Xn4p>L56V?S& zU`hmVb*yca8s@~t81!jscWlK)nc_oEmNnd>#FOi)7ACW4b%O=_&r#1;|EgZ1{!P79y$lMISE^U3SF6{k*Q(d4H>fwN zH>o$Px2m_Px2t!m_o(-(e^(dfqa9NpSD#S-p+2QPqrRxVtiGbYs=lGVslKOvt^TO~ zsxDN+pV*pWni87QnrKbDCP9;^N!C=+RMS)k|JF23x&}-!HBB`wG_5qPHElE*ns%BF znogRonr<3}MyXM0NDZYiY0MgnhSu0L4vka8YB&wAacex9Oig!94^1{Gvw1Z^O`fKo zW`JgpW{75}W|(HUW|U@}W`bslCSNl{Q=plpnXQ?lnX6f#S*}^BS*=;8S+Cik*`(R5 z*{a#D*`xVgb4YVob5?U+b4hbub3=1eb4znu^GNep^Gx$x^Fs4l^G5Sl^G@?#^Fi}T z^Hqarh1xJ}xVD%!LR(5(T3be2R$ERRsg2UcXk)c;+IVe(Hc^|bt)Q)_t)#82t*))1 zt);E4P1Q=Y^|bZ14InG8v9^h}skWK6g;uU@qiv_{pzWybqU{Q~dJ3<I}dxwN|6m zX$h@fOKJ^TN^8=ZwHB>aYty>4td@toz8>11+Fsf$ZMHU7>(vIddD=nR!P=49QQFbk zvD)$43ED~8$=WH}Y1--90_|+=9PM1~JjkyReN1~ zQ+rSQK>I}dRQp`}Li<|#R{LK2LHkkrN&8t_s1@R2csO1RFOHYMOW~#QvbYE@k4NHB zcr+e^$Kr8#0-lH`;}!4}yfR(|ua4KiYvHx=I(RBB#btOJo{l%b8{v)drg(F_1>O>u z1y|xIuEH@~4a&_pPT~ff!cDjtx8O8x!yPz-J8=)5 ziFe0);JxrHJR9$g_r-JYT-=KX@F1Rthwy%Qe|!MsAP>QZ;=}Oa_y~L?J_;X;kHg2~ z6Y)v-WPA#qk59#?;nN|XawcAY&%$TpbMU$Ne0%}E5dRflgfGUI;J@KZ@n!gOd?mgL zUyZN9*W&B&_4r176TTVWf^WmO<2&%3_%3`mz6aln|Bmm&_u~ifgZN?m2!0elh9AdI z;3x4v@Kg91{49PRzkpxHZ{c_FyZAl)KK=lIh(E@k;?MAx_$&M^Wq^qE-s7ujR)>Y9}(^c2i z(ACtX>gww1>FVnm=o;!8=^E>r>so+EZ);s!U52imuD!0Ku9L2d4%4Z18l6^$>j<4* zXVuws4xJP7QZscubUk&wblJMzx*T1u&Z`UPg1S6iNY_s{L^o77OgCINN;g_JMmJVB zPB#(qkSFV==<;<_At!l;ZYJck&eAQ=Ez&K~E!C~it%N++HM(`W^|}qZO}fpxExN6` zZIJc4L$_16OSf0|yKbLuzwUtUpzaW4!XDL~(4Ews(w)_v(_PYC)?L-z(B0JC(%sSB z)!ozG*FDfZgskVsx~ID5x)-{ax>ve4y0^M_x=*^#y05xIodB|Y!---<1W}SG1zEM_ zh+ha15lKW5(L^i}N5m6}L=urqR3Iu5RfwuYb)p7Qlc+`1CQ=D8At9uMjHpZ0Bhrcb zL_?wx(U@pLG$oorW^N0jCDDqI6Rn9h-?MZ(5FLpkm*Os9=&K}90)wnwErAm{LJwKl zMXt^iVJ0jDO<2LR*$$4)PQpd7;9Kq?GKua)56Il^MPw1#;6&bs=u6}fx!@D-Bm6{w z$Rqkg{_kL72;>0|Cq@t>iBZI8Vhk~s7)MNmeDBG`6e6FPPRt-?60?ap#9U$?F`rmK zEF^v}*hFk5wh-Hh?ZggZC$XE@L+mB? z5&MY)#6jW^ahNzl93_qs$B7ffAH*r*G;x+VN1P`v5EqF{#AV_Nah146TqkZ2H;G%s z9pWx=kGM}fARZD=h-btL;wAB#ctgB{%;k^7C*m{lh4@Mo5+E6)57QUZ7uQGVOX^GM zOY6(%%jre>^7=@9ls--$uTRof&{x!_=qu~1=&R~$=xc#1dK!3_r|TQ&8-gQw3w=v{ zD{v%lt8b@oukWnys_&*p^%&$v>-7e3TBr0Ty;)D|ZF;*NyukG?J*(&RyxtAD)0z72 z;3nQv-%HPPFx=*Q~E=_l$Z>GSo|^wae-^fUFd z^t1JI^mFwK^}p(u>X+$P>R0L4=-2Ak>DTKw=r`&&>$mE+>38UN>UZgP>-XsQ>VMbo z*B{Uy)F06w)gRLz*Pqd!)nCwG(qGkI(_h!$(BIVG*5B3N(?8HZ)IZWc(LdEc(?8e0 z(7)8b(!bWf(ZAKd)4$h$)PIJGiLZKiNllg_OOs{Ea^x?hh>Rqo$rv)0j3X1sL^6r2 zKvpCxlU2xSWRV(eEl{3KC5u#MrJyOBMy8Vu$VOz5er!{+8QFqtNwy+elWoW%P1ts1 z2eKn5zkbtwRgx&FBEKoV;-ro&(s(tHMar&4x~>+I1~orBR1+|yi)2Zjbdw(Pn=W5Z zvKM5E=a9LeeCa0xWRMJz{mB000H{0|L=Gm0kVDB~@>g;ZxtRQoTnbs}%gGhwN^%vsmRwJ6 zAUBbl$t~noavQmW+)3_%3WwjxedK=f0C|W!OdchVk;lmsB`ihT?`2h6qC`14uO(${NZUelds)Gpf zv^8WH+8Np#Iv6?{IvF|}x){0|x)~G(CAc%=29v=8e$GyCbLI^mL#Cm-p_d`s&g zY&L8$Y&GmO>@w^z955U-95Ng>95oy>95>oHd*?oHtxBTr^xVTsB-c+%()W z+%?=YJTN>oJTg2sJTW{oyfC~pyfVBtyfM5ryfb_;d^I3Op)t%@%vi!0VJvMdV=QYd zXA~JDjZwxpV}dcsSkYL?m}0DKtP0N1wZI#?jxp6JF-naxW16v^5zJdbN4^p0z-NFu zdgN;Ls!;Hg?BaEYsV~pdC6O5CLGogB7mT|Umj&Z(mfpMYn zSE#30Vq9)qVO(ikZCqmC}nX2-q3VkiANKal&$v}I)F6hdqQw>0iz7gooH=&Bu<(q?+yd2cw z+fZ#mIldjJ#CN1RfquLKl;lxRoyRCOXv=FU9CYIKP&-nj0&fQ8c$%_OHmXPso}rwe z{LWDxDwFC#^`v@HS)dT#hw4k^P`Q+s@=<;&0P5~}REX+F4WI@>{mEcz2sM-%Mh&M% zP$Q{P)M#o9HI^C&RVfpwiPR)&GBt(Dr>0TUsTouOHH(@}&7tN}^QigM0%{@kE47GP zOf8{)qn1+3sO8iOY9+ObT1~B?)>7-J_0&dc6SbMzLT#nCQQN5<)J|#_RJ-h<_EP(( z{nP=ddpSfMrjAg@sN>WL>Lm3Cb&5Jooq>9+bJTh20(FtPOkJa{Q@5zw)I;hq^@Ms( zy`kPx@2L;eN9q&xnJT0ZQ@E*wsiditskEuAshla&6lIDtRWwyHr9cf%70|h_2Kx22 zK*7F_Db*zYQ>(t7Dcw}x)WFox)Wp=()Y8-nH0fKLz!Kcl!PF5P#ygw3n7Wx1CZ!2A zsZ5wjZNg1D6Jdg4Ig=55P|YR__^8@V4if{Os4f!=l|H=54Q{BJrtYR5;ECGX)E690 zgQk9_{-yz@L8hUmVWyF$QKr$R@uqy!bkj`J9MfFWJkxyBBGY2i64P&{rKaVk6{eM@ zRi@RZwWiIcEvBueZKfTjou*x;-KIUJy{6wy`%L>y2cSCWi0P>5nCZCbgz2>DjOm=| zyy=4JqUo~fis`E9n(4afhUup1mg%a8oxVeP6q`AzWb!sAWq&eCgYfdmHnv=~H%oWX*%~ilhwVJt>In^vS zOU-HKdggR9cpgGSxz&zMI z$~@XU#yrkE!93AC**wiW-8|D=V4h{3ZJukMXP$3fU|wkc)x6mJn|Y~ug?XiUm3g&! zjd`tky?LW~lX5LSY*}Ji4i!;r zEbA=mEgLMGESoLcEZZ$REITc`EW0gxEPJ75>VV~-<%s2|<(TET<%H!AsG&M-Icqrw zbySxuS1mUzH!Zg;w=H)p4=pb&FD-8@?=0^vpDdp(Uo2lOg%%+lMu*eI=n`}(x-?ye z7SZMDNIHs+reo+lTv5V;K_4eSq-U%GmyMdQF3jXS9a9023sII4rJk^b~ zi8j*~nx<{w_U)h|03D?B=n&nH?oSV( z2hxM+!SoP%C_RiGPLHHV(PQYb^f-DvJ%OG?Po}5P`SetJ8a2>sadLzAw-b`+_&{yee^mY0MeUrXL-=^=-_vrid z1NtHTh<;2zp`X&v=;!nc`X&8}eoeon-_h^s5A;X+6aAU~LVu+ZtH3I>hFQa{#jGW) zC9P4`XlsHs*;>I`(VAkd3SNLUt+oDi1C&^0)--F8>t6$FLu+Ge6KhjzGi!5e8*7HO zowYsG11qhlRSVAVI;$Ss-i=nuYOoG=_KQtqi?l`AqHQs@I9t3e!B$jNSkqS9CbmgzQd=50 zpr_mF+Zxy!+8Ws!+nU*$+gjRMfkSy~TZXNJts^LI|EHd|?T3oC(MH)UHoMJXV{9&4 zkteGgoKt()dfBpU*|y%c99yoNZ zcF1Pji1aH>H_9o!Z+T7mKF1NR_x3zb& zcLAr>Zs2mN1g}$-T@Bu+xLs%0+ey0t{7NZ%k;^GuvD-r!o=*X{!^(ttf^&$EZ@{pS@zlXIrh2sdG-bNh4x?Vi|kA6zu8yVSJ_wF*VxzE*V{MQx7fGY zx7&BvciH#Y_uBW__uCKH584me58IE}kK0e$PuWk~&)Cn}FW4{IFWE2Kuh_5IuiJ0f zZ`tqI@7eF$AJ`w-AK4#6Rq9jwGy8M<3;Rp^EBkBv8~a=PJNtY42m43+C;Mmn7yDOx zpH>`j$)4DjuMUtM=3{XM;S+1M>)qY4w0k0Bgzr&h;hU^;v5N%L`RY%*-^n! z(NW2f0$!_Cids}e_%$7(%Cx}%Y!siV20g+uOW?P%j@>&S4l2WN)P z;J(n!p#bLv>`zw(9XKeE4x_{DusCRNN^m%w4i{A6at@Cp)6w10(~;%q?dapkapXF@ z4k)K{1ROy}o+ISw=jiVk;27u_KNu2?il45?HJ=2@0jSA0^8Z%9prc5)aKW@dynKn!Y(~fD+bYMC%otVx{H%7srjEW%`J!4>u z33Gnbji%x4xb3z@~t66QB%DYJ}O z!K`FfF{_z1%sOU0vw_*jY-YAFTbXUl4rV8_i`mWWVfHe=Gy9nR%mL;gbBH<29ASduaG<~qGliym->oc)~xoP(T$okN|& zogp9MW~ujaaD0ubyWlZ+8W?sTMHa&#eaI%7CF||1;^S3;8WYw)y&ljJn6f*y1A4t z3>@cmF2bb;H+qB12!8S=ml>+>-QWVB3HA3qTs>XATv^}+pX17Pd0jqNk$Zd4mFEh% z`nv|YhPZ~hhPj5jMz}`0M!812#=6G2#=9oECb=fN@?BG1(_GVCGh8!W1+H08*+0iM z&o$q*z_r-5#PyqNscSjZ_pf%Xb**=8aBXyLa&2~Pfx7=~uI;WJQ1`#bwb%8#YoBYs z>wxQ!>xk=^>$vNL>!j$K~v>zwO?>yqoT>x%2D>zeC^>!$0r>yGQL>z?br>w)Vb zs0KWCJ#jsAy>z{Dy>`8Iy#sxVPp&VnudYHD!V1|iHk>WamS7{;l58oq4EqaPo{eIo z*%&sKjbr241U8XPVw2elY(=&*TZOI4R%dIlHQ8E4p7m@hD+Z5x8C#dF$EJe{K|}DR zZwy}b&DiE_OHe0h&9-GT*mmGQ-+}GOb^;grBKP@j;6<+l&v~$gV%4mc#aROM3k={h zZ)RywGqAG`aE5n*z5&O&Sr2F&^k9ozc(TEVrw?1?yc1x9Y#!T>9l#D^hqEKtk?d%8 z3_F$`$Bt(wu#?#->{NCdJCiM7XR)){IqY0^9y_01z%FDLvrE{e>~eMmyOLeau3^`) z>)DO$CU!IECv0Q4vpd+G>~3}syBE|H_Obif1METeFng3e&YoodU{A4U*t6_8_B?yx zPpyS3>{a##dy~Dz-e&KxciDUFef9zSkbT5HW}mXp*yrpE_7(e@eZ#(G-?8u659~+w z6Z@I{$|9VA6LMi(I9H4-&XwRIxRP9Ht_)X}E64r9iMaAyBp1a+b1_^jm%t@*NnA2l zfvd=+{OK@MjjPVp;A(QUxl~TfNjNE&#?|A}x%yl~t`XOmYr-|+Ho3QozXIE+(s8cxgMoQ@+oBS&dgcD0hrI&Yj>+ za({5AxYOJj?ksnXJI`I$&fVZ{a<{nK+#T*NcaOWzJ>VX4kGRL& z6YeSZjC;<#;9hdCxYyhp?k)F@d(VB~K60P9&)gU8D_6*Y6cI1v!}xH%7+;((!AI~V z`BHpoz6@WMFUSAFi}><+q?( zn3wQUUdE^Kb@_C@KHq?E$T#Mj@y+=bd@Ekgx8{qSI@|Fb`636-ZoGn5@+kOoV!WEy z@LC?{bv(h7;J9hzDc;1Jc?%wmk;nk zK93LagZQERNPaXwh9Aq1<0tZy_$hooKb4=s&*TgES^R8%E1<+t(M`JMb8elNd|-_IZ55AuijBm6P`1b>o0 z&tK#(^H=z*{5AeMf0Mt(-{$Y|clmq#ef|Oen18}Q<)8B}_?P@^{w@EGf6ss9Kk=XW zuY4hoxCL&ZJIo#Kj&PTBmvWbOmvNVKN4n$O$?gj7itZG5Rd@A2JzXU3y6$v$Lw6&0 z6L(X0Gk0@$OLr@GTXzO{-FI|%ad&lh1MhjITjj>UnH~qHdA*wi*Ls`V;buU!gm=3^ zwImb#nR~jk+`Zj>-8t@W9?W^}5NMhVaSwNoaF242c8_z9cTaLpc29BVgVMao+_TIo@$;Nps`ZhBlbuqHBt_e_Jp*&_#V=3~iE^@4PfftF}^UWK)mnX~98yvxNJzkH`6YvB*d7hA` zpJ#w)pl7gWh-bKGglCjzjAyK8yl1j!iYMPQ)icdA1C(7Bc$RvWc~*K>dDeiw%X-fS z&nC}i&lb;C&o<9?&o0ky&mPYK&q2>2&r#1Y&vDNQ&mW#sp3|N)o^zgyo=cv~o@<_) zo?D(fo_n7Ao(Gl3Nynp!!wI#mdK3A zESXt4^OwxX%&7mz)?Gy_u5An3E_Zi#%4L?jySuyV0)ht!5QrxvBmtt7ySux)ySuyV ztUpz2?{m)I+IKrHMjQQp^yf7fHJ31#G?zA)F_$x!H&-&+TB-zx6w7&sBjH88eF5y zqmA~~pYqlOqq{Z9JlXuerL8}mt^ZcG>_%tHX?FcBY=LI)f9hHxGi(l<5ktEn%#@in zGv=t_-tcC@ESlryggI%J%!*kxYvz=BwxQz8GcPbNG%T^D=4Iv;=9T7E=GEr4<_(5F zwbi`MywkkfyvMxPP;(BN519{}j~ar`aq~&@Df4ObS@U`GW%E_@HS-PgP4g}D1M?HZ z)p=olX?|sXZGK~ZYkp^bZ~kEZWd3aaV*YCWZvJ8ZY5rwSH~%&pi)oh3mMoU6mh6@s zmRy$HmOPfcmVA~1mVy=o=Cl;Il(3Yvl(Lkzl(Cexl(Uq#RJ2sGRJNG@cEYM#YFp}B z(u@+=p9a`}>t8J`t&HwhTT44jd!zH!(bCD%+0xC@-O|I-%hJcv_dh+aftEp*!A8kz z=zkhsqyM(M#vARfNk+MA>i_9>Su9oyV6huckjvt>fEJ$xvIHzaOUMFS!j^~ywO|(9 zLRn}FXNg&Oi(nBgaZAFIv`7}&qF7YJ?@C!_Tjp5iTIO38SQc6qSr%KCT9#RsTUJ_CTGmIbk_z zIb}I*Ib%6%xnQ|yxn#L)xnj9$xn{X;xna3!xn;Ryxo>%3d1!fLd2D%Nd1`rPd2V@O zcxbOIZ!PaE?=2rJA1$9OpDkZ3UoGD(-z`5ZKP|s3>6YJ?4AzX+OxDcSEY_^nY}V}7 z9M+uHT-Mx%i_V5-de$EyjQYTv6`$^ zt<|j6tu?GQt+lMRtp=1~t!GWM{D-)`%6cVpiNrSZOO`jaoTt%*tCut7Mg}s#UYjw$8E6wa&B7w=S?Q zv@WtPwl1+QwJx(Rx30CWw{EmOc*kCQu8g4b%bZ0u6wMKqH_r&=hC}GzVG$ErC`*YoHC#7H9{w z2RZ;9flfddpexV~=nnJ*dINoden5Yt<2ukNy$&{7uS1Qh>u{s$IuaOdv|RsGTqgh% zfyuyh;J-C-3t$BRzyUaoZnzr&0UzK80zeQ50Wc5-A^-|t01glU3D5uoL;)6v0X!f8 zB9H)*fCQ+32IxQvm<`MW<^v0Wg}@?UF|Y(!1}q0w0IPu2z#3pJunt%cYy>s~TY#;= zc3=mv6W9gp2KE4ZfqlS!;2>}aI1C&CjsnMk#$o8jf+{UOF zw>KQPjz-nEv(Yu~Zj_Dx^ok9-&eq>(6%VovwhjGTDIR6ih{xF`8V%yfMw57k(HEX& zbcU@)b=Yom7!6{#Q6cu({5HrIwnc2H4YT1k!baK{8*7W%c$;X8+Y+{0#irUc zn{G?lX4~f4=Go@k7T6Zr7TFftme`ismf4ouR@heC*4ftEHrO`Vw%c~tcG`B?cH8#Y z_Sz2F4%?2{j@gdePS{S`&e+b`F4!*GF4->IuG+5KZrF}zI+5vGrot79Rxnj)Tp?@4 zoE4K5rHXPzrDD-aeNCsnpNZ-!#B9&@{+2*fhj6)HKXA+%&>8(lp96 z+BC*A)-=vE-Za59(KN|4*)+v8)iljC-892A(=^NUkI8JZn5-tiWMq|1uqkYcm=F_c z!c4e{Fp(z8M4K2>)Wn)NlW2;Y5~iJ|U8eh{2d0OnN2Z)r6O{_3l}#&`_W$-fE2dRS ztDIIP&6HL(ty)_3v>Is(8n6wyhI+$P!`ThzH9Xkxbi;EE&o{i(FfIN5?}vu4Sg~UL z2ETq6sPON7Khv}{J?&RodRn%$RNCyc?&&?!d#3kF@15Q!y>EKI^#17s(g&pvP9Ks! zG<|sbi1d-^qtZvGk4Yb!J}!NH`h@g}>66kYr%y>YfXMV2=`+*+->>|0n*IMd!A;vO z+a23|+XLG}+aud!+Y{S!+Y8$#+h^N1+b>(X?YAw1J(E3)J*z#NJ-a=JJ*PdlJ&!%F zJ-@wxy`a62y@yeTaRSeYkyueWZPqeYAayeXM=F zeS&?WeUjZs3fiaIr`f05XV_=j&33CDu-oi*yTk6ZyXyj!7ke4_Jm!s%Z7=l+I4%%KHIPn=h+w77yk7Tm)Muum)lp` zR~cU7M*AlFX8TtAHv3NdF8gl#9{WD~0sBGwA^Tzb5&H@IN&9L0dHV(XMf)YgRlH)q zYQJv3VZUj=Wxr#;XTNVKix2IO?N1HQ^M(DTVS2tX?8SHX5B87tPxjCDFZQqYZ-&44 z!~W9{8Pn~*?Z(2rBas!%@po$50&WI~q8eI+{6}J6bqeI$As0INCbe8M0$X!*}fL=;G+= z=;r9|=;7$;=;i3`=;P??=;!F~7~mLaNTWj>!;Dn?ILCOyfSl-<CC;2gX|aEOk$ zBjHFoB*T+b9I8Wer2guX^BnUX3mgj_iyVs$Pj-c4rDK(2wPTHAtz(^Iy<>x8qhpg} zvtx^6t7DsEhhwK>mt(hMk7KW6pJTt{fa9R!kmIo9h~uc^nB%zPgyW>+l;gDHjN`21 zoa2JyqT`a|vg3-O)LwI3cieE?a@=v;b3Af9aXfWAb3AvvaJ+Q9a=dlCbG&zaaC~xn zb$oMtcl>btbo_FpJ2E&kIx{&lJF_~oIkP)+ICDC4IdePnIP*CRI14$8IEy-qIg2|> zJIgxDIV%_)-%3Wgx60pEZ%w1n+sJ6~Hg~o%iqUPI?TzwuM`tHz7o#)X-Pyz0+u6_A z-#NfJ&^g%orz1VgC`OMly3yl|ZuA7_M57!%)hI?!H_Fhnoc}n@h7)OZ0!9_u>2x{W zPOsDF^gAJEz!`GF&ae}4qE6gNI7uhvWSmha>x?-?XTm8vHK*>J?VRsi>|Ekp>Rjer z?p)>E?A+qq?%d(rpbVY;Jo6z=DhB_>AdB<u57OCt{kqMuH3G?u6(Ziu7a*Y zuEMS&u41lIt}?E&u5zx5Mwz{uQEjd5s_Uxfs_$yx`qPJOZqy-Lx>_66*AA{uuFkG5 z|I;Y$Y1E1Py88X6;W^kf^nd!Dqg`YFQ|_GPn*4vNo&PpF|F_ub{M+mN)9UoQg07GY zc12vMi*!*g#uasOu9!=3#a#)PcinW|a@}^_b=`B_cRg@DG?dRLuBWbNuIH{7hWGi(_15*l^~v?w^~Lqw^~3ec z_1l%foyncqoz+XjCNw$0rw-K~wnYFl>)cPDpecNcdzcXxLWcTaaOqm$al-Pb+9Jzc;)ME5N0k_R$)Kk(^+EdO@S}S@gd8&A-dun=Wc?>wv zQ{U6T)6mn{)5O!-)7CItJ9s*JI(fQyx_P>LdU$$zdVBhM`g;a?26=`VzVZmqNY5zG z7|&SGIL~;`1kXgzB+q2e6wg%8G|zO8#bfmV9-GJRaTpS;%j5BY9 z6l?}I2U~+}!46<(jWsyK^BaGJSc!77zYzz z5|j+TR{=Fp2UFl|aE@X5&I9Lz3k=bB5x5v!3N8begDb$5;3{x6xCUGct^?PD8^KNB zW^fC*72FB#0r!Ik!9(B?@F;j3JPDo#&wyva3*bfYGI#~N23`kmfH%Qg;BD|Ocn`b} zJ_4VD&%u}AYw#WT9{d1)1V4eF!7t!9@H_YuOb3618NFG&S-si4IlMW&xx9J2`Mm|a z1-*s5g}p_+#k?iFCB3D+WxQp*<-HZXmAq99kF~0|nzy?5uN7+uvvs_6z4g3l-um7K z-bUWW-X`8=-WJ|A-v7F>oxGh51-6@EzxMF<{OiB=G33{NhWz@k`Z~n0UWa=}7-sCh z&gEd3e>B=^hH?6@L)pOB$k*7{ z#E@^B8|rOqUt8b*s*+uN-F$zP(>}g|hHpC5H{3VsuV6afH_A;I*cbLie25S8;XcAg`Y0dmV|=WS^YK2xFrJe>*{Aqa zpXQtGo8z17Ti{#hTjX2qTjN{nTkqT8+vMBq+v3~i+wR-p+v(fo+wI%q+w0r!JK#I$ zJLEg+JLWs?JK;O!JMBB;JL@~|yXd>*yX?E-yXw2)yXm{_yW_j-yYGA8d+2-Od+K}U zd*OTOdu7PdZ+-85AABEupM0NvUwz+v-+e!PzkI)a8T=XjnGA6{i$AMBn?Hv?r$3iJ zuRou^fWM%>kiW3Mh`*@6xW9zIq`#EEw7-nMtiPP0Q&;p?@>eztT$8`L;osIW+}k?- zy8b_AY<+(Ne?xyGLwjvv=)8YS-j;^Q+uAUBJNP^L|0uIv4B5AvVf*&*_cUbRK8ERQ zh`#;-{(*+lJJ=9Zhx&*4hx`P2Qs{TZN)P$nodlm*HPWrMOqIiQ?S zZYU3w7s?0ahYCOip+Zn$s0dUPDh3sYN2bG5^8a8kh$Yl7y)eRfC zrlAAZ`KtogH!R?Wh6LQ$kbs;1HF8@RLT;D`0Cj{qL7ky4P*>N{jpTXLF50|QT-o7btd#j^|e6% ztL{1>7vzCJ$P4)(KLkMmCLCc{P&`M|(v<6xWt%KG>8=#HQ zCTKIX1={C@We8*pWC~;sWC>&qWD8^u zrjlyR^#ct84Fin=jRQ>rO#{sW%>ykAi@0^5 zO`xse5qJ1&5O+2d;%1^l`u&xM2L=Wi_VFM2cvxU~U_@Y~p$LyQ z9O1EnafTW^Au!Ppf+rg~@KnPGo*tNC_`tIa8Q5%C!PWo}u>Bu>*b@K)-hj_=bRk0x z4hBL2I1mXS0W^RG@Bk4Y15|(xFo9@*4RC>2fDZ@(F%Si zgMEU1g9CyCgF}MDgCm2Zf}?|Dg5!b{gOh@jgHwW2gEND(g8u|9L2J++bOv2PchD2` z27N()5DEr^phz0Q=5hR0jFdF27u^=B5f?_ZcOa`T(98`i@P!G-x&JQjK zE($IVE)6aVE)T8~-VELf-VWXmJ_tSxJ_$YzJ`cVKz6`!H>MZYr zAA_HQpMzh5UxVL*--Ew`>A~N@Orh){qpuvw70MUNA1W9s94Zwm9V#0t7b+jB7^)nq z5~>xd9jX(m8>$yd3pEHe3N;V42(=2e3AGEg4|NK44s{K63w00m3H1&23-u2T3=Iwq z2@MSm4~-0s3XKkp35^Yn3ylv=2u%!43QZ182~7=63(XA83jGr@hb$p$$QH7P93f}O z6>^6>Au!|(`9l5>6bgjkP&kB!h!7QuhS(4niiP-)5Q>KqAt|JUW``Dp7KfIEmWEb_ z)`m8OHib5awuZKac7zUvj)abej)jhgPK3^c&W6r~&WA39E`~0Ju7>V}?uPD%9)uo; zUWYz~zJ$JoeuUCPze5IL2xo?~z**t!a1J;(oChun7l%v2rQtGgS-3o00j>yFf-A#S zU=v&wt_D|!Yrr+(T5uh>E}RB8gd4$);ihnNxCPu2ZUwi7+rsVO_HYNdBisq@40naQ z!@c1?a9?-;JOmyJ4}(X-qu|l-7jl@LG5sydK^NZ-O_&Ti~tmHh3qz3*HUyfe*rm;3M!+_!xWwJ_VnK&%kGm?#6le z0(=p^1Yd@)z*pgG@OAhGd=tI}--hqNcj0^RefS~#2!0Gdg`dGM;MedQ_$~Yneh+_w zKf_<(ukbhcJNyg&4QCE#59bKy4Ce~x4(AK!4;Ks<4i^m<3l|TU2$u|(3YQI62v-T4 z!d1i7!ZpHm!wteM!mYxs!)?Ru!tKKy!X3k%!(GDN!ac&h!o9O!c)W3!!yD&!?VKwgn_U#>k9K z{ow=QL*c{WW8t&m%i(L`>)~7B`{5_yr{QPe=i!&(SK-&;x8e8U58+SY&*87(@8O@} zU*YudZ==AGE0Q~sH&P%{C{j35EK(v;(x`Eii8>40=ZIw75ru1GheJJJK` ziS$BxBYlv*NIzr%G7uSr3`T|^!;sO)7-X!`ftr9!LM9_qkg3QtWCmhJEQl4cAr8ci zfQS$ABM=fmf=CE~kqCkyD1spr5=A&9hVY1hh)5hsAW1|Kvp8Fkk!aqWF4{r*@SFHwjeu@oycxvFR~vwfE+>&BS(;<$O+^m za@uIJoJTGqmyj#SRpc6S9l3$rL~bMZko(9(9r7OefP6$g zA)k$k%U9$Z@*Vks{6f-^3}{9)6Pg*#f@Vdtq1n+KXihX2nj6iF=0o$N1<*oh5ws{; z3@w3{L`$P((6VSbv;tZQt&CPdO=wlL8d@E#f!0LppmouDXc}4{ZGbjJ8=;NSCTJV9 zE!qxkk9I&ip`Fn#Xjil++6(QC_Cfoi{m}mC0CXTa2pxC>!7!9Kl6hjG=L@AU; z88nKrXbk010Tt0Wnn07NgvzLbs%Q$Gjm|;mqVv%C=mK;Rx)@!8E<;zKE74WxI&?j{ z0o{mhLN}vZ(e3CCbSJtC-Hq-+_oDmJ1L#5Y5PBFrf*wVWqbJal=qdCxdJa90UPLdW zSJ12IHS{`q1HFabM(-GXoO|ef^a1(^eT+UqpQ6vv7wAj$75W-|gT6)Iq3_WT=tuMu z`WgL#enr2b-_f7wFEky^fMvuoVVSWkST-y>mIKR)<-&4fd9b`#J}f_004szQ#)@D? zvEo<>tRz+nD~*-G%3|fP@>m6|B322jj8(x*SXHb#Ruij*)yC>zb+LL_8de``fHlM# zVU4jSSW~Py)&gsZwZ__DZLxM(d#nT25$l9?#=2l#v2Iv*tOwQ;>xB)#24aJ;A=prC z7&aUmiH*WWV`H&#*m!IrHVK=IO~IyO)3E8-4D27wf>|*eX2%?u6LVp1%!5TR1Vb?l z!!ZIQF&5*n7{+4)7RQp9gvpqKshEb%#^zviv3b~hY!S8?TY;^@R%7e24cHcJE4B^W zf$hY0VY{(C*gkAOb^tqs9mP&!r?B(b1?&=b8M}gA$8KV`u)Ekj>^}AYdxSm4o?uV0 z=hzGECH4mUgnh=oVBfJH*iY;i_8ZHHXTh`L+3?(W9y}kOA1{Cx#Eam?@RE3GyewWG zuYgyUd4O7G4{#gV)2;@CJB8yfNMcZ-zI=Ti~tm)_6O-J>CKDhv#&EjnBpB z;q&nY_(FUUz8GJEFTsV{uBR&|HciBnaD(BA+i$Li0niTA}5iX$V22M z@)7xo0z^Tg5K)9EPLv=@5~YYTL|LL7QJ$zkR3@qrCZZ})ji^r4AZik|h}uLQqApR7 zNF(YK4T(lXW1@jLv2Z)2j5u>?voH#+8Bu)`$h_l2w;sSA*xI$bf zZV)$#Tf}YRE^&`|Ogtf;5zmPi#7p88@tSx;yeB>ppNTKTcj5=}i}+1sATyGg$t+}6 zG8>tl%t7WPbCY?9k;TapWErw7S&l4ERv;^pmB}h(Rk9jcovcCD zChL%O$$DgcvH{tUY)m#Kn~}}Q7Gz7Z71^3>L$)Q`ksZj6WM{Gq*_G@@b|-s~J;`2V zZ?X^Bm+VLOCkK!N$wA~`atJw;97YZ&N01}QQRHZH966qxKu#nlk(0@(auKOXL;uDtV2(LEa+o zkax*@P+XjjB&Ipc+z*sK!(iswvfsYEHGF zT2ig3)>K=n9o3%dKy{=#QJtwSR9C7y)r0Cu^`d%HeW<=vKdL`9fEq{*q6Sk#sG-y_ zYB)858cB_!MpI*`vD7$fJT-xuNKK+9Q&Xs^)HG^3HG`T-&7%IH%#?+)QUGP6?39CY zQZC9(c_=UCqx=*^1*jkuqF^daMJR+qDU8A?f}$ygic&1aQ89|A1S(D?s3av(GNn)| zrBOPSqGnTbsJYZUYCg4qT1YLT7E?>8rPMNNIkkdXNv)z*Q){TT)H-TCwSn45ZK5_) zTd1wnHflSygW5^$qIOezsJ+xaYCm;=I!GO&4pT>{qtr3#ICX+LNu8ojQ)j5N)H&)r zb%DA_U7{{iSE#GhHR?KbgStuGqHa@nsJqlX>OS>=dPqH@9#c=Kr_?j*IrV~iNxh<8 zQ*WrZ)H~`u^?~|GeWE^7U#PFtH|jg}gZfGRqSC3~R0cXDor%s&XQ8vw+2|Z}PC6Hz zo6bY$rSsAG=>l{?x)5EME3Vb;U7v11H>4ZUjp-(IQ@R=5oNht4q+8Lg={9s*x*gq~?m%~> zJJFr#E_7GA8{M7mLHDG4(Y@(DbYHq3-Jc#n52OdtgXtmkP-!leYBs3=l~t0 zLo`f>=?IO`D2>rLP0%Dw(KOA_QJSSWI!5!fK#O#oPS8nOqGej4Ra&EUIz`W>=g=$Z zfBS!D|5E^L#x!SIFfEx@OlzhM)0Sz+bYMC%otVx{H>L;Elj+U$Vfr%tnEuQFW*{?& z8Nv)@hB3pLk<2J&G&6=7%Zy_tGLx9e%oJuSGmV+f%wqmAYJ?WX$^eXwaWGEC#kd(C z<7Xfyzyz5P12Yi@VK4?~2!>{&49jp#jNuu95t#&&WF$smR7PWTCdJHV<}h=adCYuf z0ke=<#4Ki(FiV+b%yMQ0vyxfGtY+3QYngS-dS(N&k=ev-X0|X}nQhE=W(TvA*~RQ; z_Aq;yeawF50CSKz#2jXhFh`kV%yH%fbCNm5oMz52XPI-%dFBFhk-5ZNX09+-nQP2- z<_2?`Y zX1*|AnQzQ@<_GhW`NgC&znKitjL}Td%+V~-tkG=I?9m+2oY7p-+|fMIywQBo{Luo@ zg3&_J!qFnpqS0c};?WY(lF?Gp($O-}3eo=20nwq+vC+xVY0>G?8PQo$bJP+AqPD0f z3P!zAU(_FkqJd~I8j6Oakti0$qePU9GSO(1jdIagREUbvL^K(dqHe&E{qEu?5(IY$3KVTa+!v7H3PaCD~GJX|@bo zfvw0^VNGmRqi9*3t;N=1(^%tU7u(2a7dK^_u`StFY#X*M+mY?ec42$5z1co&U!#3F zkR8MhW{0pt*<)G(yNlh;?qT<```G>L0rnt! zh&{|6VUM!M*yHR8_9T0XJ;R=5&#~v(3+zSq5__4w!d_*svDeuf>`nF-dz-z(-evEx z_t^*RL-rB-n0>-NWuLLn*%$0f_7(e@eZ#(G-?8u659~+w6Z@I{!hU7HvESJr>`(R= zo6i1bGjJKXOk8Fz3zwD4#%1Sna5=eLTy8E8mzT@O<>v};1-U|8VXg>Qlq<#+=SpxT zxl&wdt_)X}E60`RDsUCKN?c{G3TNV~a@Dx%Tn(-!SBtC7)#2)L^|&;yKG%S2$Ti{` zb4|FWTr;jY*Me)wwc=WHZMe2vJFY#~f$PY1;yQC(xUO6`t~=L*>&f-vdUJiazFa@9 zKR19I$PMBKb3?eH+%Rr9H-a0r@mz&4U z=N58{xW(KOZaKG-Tg9#B)^O{&4cta<6Ssxi%5CR%a67p@++OYgcbGfI9p_GPC%IGH z8SXrHnY+SW=k9X%xCh)5?j`qzd&j-!K60P9&)j$J2ltap=YDh9V>x1ZV)n zVkKi`VwGc6Vy0NNSoK(qSj||iSnXI^tbVLPtYNHCtZA%ytVOJKtZl4atYfTmtV^t0 ztb43StY@rOtWRuEY;bHyYApAHpUr!I3X5~C1Y|-jp?!3u{lN! zc42H$Y)NcsY*}nYY*lQ1Y)fpXQHb3W+Z)?&RALVst=MC+6S0%AQ?c{0i?K_w%dxAm zYeqfxcI-~OYB?hd+bN-r_q_sz-Qz$ z^I7?9e0DwupOeqU=Qhf-dHH;Ne!c)-kT1j+=8Nz}`C@!=z64*2FU^W8^%s1hi^3C`bd`rF+-XM?Tw~w zC%!Y^mG8#)=LhnG_`&=Tei%QTAIXp6$MWO&Dg0D^8b6c&hqv=y-pBj-5D)WVKEfkB z#uGfnvpmO(e4J14NnYY*Ug32<#n0yF@N@Zj{6c;aznEXbFXfl=~PyQF5&j02!2pNS;LS`Y0kX6VgWEXM>IfYz8ZXu75SI8&i7YYakg+fAM zp@>jaC?*saN(d!|QbK8=j8Il6CzKZ|2o;4&LS>5$X!{ zgfyYP&_HM?G!hyMO@yXGGoiWALTD+p5?TvwgtkIEp}o*S=qPj&ItyKdu0l7VyU;`E zDfAM03w?yXLO-FuFhCe63=#$lLxiEiFk!ebLKrEG5=IMSgt5XnVZ1Owm?%sVCJR%9 zslqg2x-dhSDa;c75zK-`unK@+6YPRha0za~BY=We@Ckka5&}X{2nnzd79s*7paLe~ z0wItBCC~yRL~8yRbvpDeMw<3wwmU!aiZYa6mXH z91;!-M}%X-ap8n;QaB}?7S0G~g>%Ar;ev2cxFlQ_t_W9!Yr=KmhHz83CEOP72zP~h z!hPX^@KAUpJQkh^Plac~bK!;XQg|i27TySNg?GYx;e+r|_#}K5z6f81Z^C!shwxMQ zC8P_#g$!awF_V~C%pztLvx(Wo9AZu}mzZ13Bjy$JiTT9>VnMNxSXe9~78Q$$#l;e0 zNwJhzS}Y@$70Zd`#R_6Yv65I>tRkAks$w;!T3Db^Bei*>}hVm&cUtS>eY8;Xs@ z#$pq(sn|?xF18R`imk-fVq3AD*k0@)b`(2_oy9IvSRm^fSy#GT?Uakscf+$-)A_lpO_gW@6auy{l~DjpM0 zh$qEU;%V`Wcvd_oo)<5O7sX5B74fQgO}sAN5O0dN#M|N>@veAJye~cwABvB}$Kn(5 zsrXEMF1`?7im$}i;v4a;_)dH;eh@#3pTy7N7xAn3P5dtY5Pyol#B}kum?54qo++L= zo+X|&o-Lj|o+F+!o-3X^o+q9+o-dw1ULamDUMOBTUL;;LUMyZbULsyHUMgNXUM5~P zUM^lfULjsFUMXHVUL|gdSB+PTSC7|-*NoSS*N)eT*NxYUr^V~X8^jyN8^s&Po5Y*O zo5h>QTf|$&Tg6+)+s50)+s8Y^JH|W3JIA}kyT-f4yT^OPd&Yakd&m34`^Njl`^N{w z2gV1*2girRhsKA+hsQ_6N5)6RN5{v+$HvFS$HyncC&nkmC&#D6r^ctnr^jc+XT|@C zo8y+aH4emWaeLemcg9_Dcia;P539*HAyKAw!r@gwmw@eA>b z@$2!&@gMOri3*8oiH3>RiH?bGiJpnUi4lo$33I}ca3JHVn#9_~y2Sd#hQ!9iro`sNmc-V?w#4?t zj>OKyuEg%dp2Xh7zQq2-fyBYYp~T_Dk;Ku&vBdGjiNwjosl@5TnZ((|xy1Rzg~Y|g zrNrgLmBiJ=wZ!$rjl|8wt;Frboy6V5y~O>*gT%wcqr~IHlf=u!??m=w;be(q>14TN zg=FPqm82fOksOsAog9-KmzP5P4lB$Ny!gUL`5 zPKJ|_B$C9FWRglUNjAwRg`}8FB$G)ssU=g%*~vM{xygCS`N;*zg~>(9#mS|~Wy$5q z70H#!Rms)KHOY0!^~sINP020Et;y}l9mze({mBE#qsimR6Unp5bIA+Ii^si;(3Dj}7WN=c=qvQl}ef>cqeBvqEGNG7SO zR86Wb)sSjQwWYdJJ*mFbP--MKmYPUSr4~|4sg=}PYAdyq+Djdzj#4M7v(!cEDs_{( zOFg8XQZK2u)JN(o^^^KbgQUUIP-&PnLK-EFmc~jGrAg8hX{t0$njy`UW=a1@X2~L1 zB|x%CPRT8KBvA57J_(WnQcwy>uoRXe5+Y#|E)fzbQ4%Au5+}tZQA$WjNs<&vl{870 zQqpW`jx!c0RCTX*@McOKDleSAcq@B_(X}7dT z+AHmo_Dct(gVG`CuyjN^Djk!KODCk0(kbb*bVfQWos-T>7o>~QCF!zsMY<|oldelQ zq?^($>9%x7x+~q2?n@7(htebIsq|WUBfXX0N$;hP(iiEg^iBFM{g8f2zoc~Ow`64g za$&iMTvRS57ne)OCFN3bX}OGC zRxT%(mn+B>T(Uard&&|E!UCj%Jt+lxxU;$ZYVdB8_P}PrgAg6 zx!gi-DYuea%WdSgayz-b+(GUrcal5HUF5EEH@UmqL+&Z}l6%X29w-lz z2g^g`q4F?!xI980DUXsz%VXrR@;G_CJVBl)Pm(9gQ{<`gGJ}4iO56eg7qw+ENxO_rBDW8(h$YM3bTeWiiYP-&zzR+=bHm1atF zrG?T`X{EGQ+9++6c1nAtgVIsyq;yugC|#9qN_VA)(o^ZB^j7*PeU*Mne`SC&P#K~O zRfZ|Ul@ZEFWt1{n8KaC<#wp{K3Cbj8iegth3aI#$fD%?B3aKy(tEh^m=t@eNt;|v8 zD)W^2$^vDfvRGN7EK`;%E0mSWDrL2@Mp>(@Q`RdRl#R+JWwWwH*{W<)wktc7oysm{ zx3WjstL#(uD+iQ=$|2>jazr_*98-=fCzO-QDdn_sMmejTQ_d?Fl#9wG<+5@`xvE@K zt}8c`o60TawsJ?gtK3uWD-V>1$|L2m@)2+-e>*ubNNIuNF`X zs)f|TY7w=lT1+jjmQYKorPR`D8MUlhPA#uiP%EmH)XHiV)udKctEtu18fs0omReh_ zqt;dHscC9`wSn4DZKO6Y_h1ybWrM6bvsBP7DYJ0VV+EMMKc2>KnUDa-C zceRJwQ|+bpR{N-Z)qZM!b$~ih9i$Fchp0oUed6I#HdZ zPFAO=Q`KqebajS0Q?;oPl~kiDtH#xYs;GPB^wx=r1o z?oxNFd(;E!LG`eDL_Mk=Q;(~s)YIx2^{jecy{KMNFRNG7YwC6NhI&)IrQTNWs`u3g z>O=LB`b2%IK2x8oFV)xTTlJm#Uj3kcR6nVo)i3H-^}G60{iUX>zts#{MlF+;Ma!yX z)3R$hwOm?mEsvI0%cm943TlP4B3e#6n9`e^;M z0op)qur@>+stwbIYa_Ig+9++bHbxt(jnl?!6SRriByF-bMVqQk)23@Pw3*s0?H|po zSv0E#Xg1BRIW(u{(%hOy12wPa)BGBw1+<_R(qJvDMKnZ1HB7@bLL)Uwqcuj0YOKa- zF^$&*P1NFALQ86rCTohOYMQ2NDQ&hkN1Ln7)8=anw1wItZLzjQTdFP7mTN1tmD(z8 zwYElEtF6=4Ya6tU+9qwYwnf{jZPT`EJG7nJE^W8AN879I)Anlzw1e6q?XY%4JE|Si zj%z2hliDfmw01^2tDV!%YZtVO+9mC>c163YUDK{>H?*7DE$y~;N4u-t)9z~zw1?Uw z?XmVmd#XLto@+0(m)a}swf07PtG(0SYag_a+9&O^_C@=uebc^cKeV6PFD+gBt!2Y4P+dKNvao=wlL=g@QNx%Aw69zCy~PtUIx&Xr1$dKKNISJkWO)%6;BO}&<0Td$+n)$8eLdVRft-cWC(H`bfzP4#Aa zbG?P$Qg5ZV*4yZ9^>%uDy@TFS@1%FuyXal@ZhCjUhu%}~rT5nR=zaBmdVhU@K2RT| z57vk1L-k?$aD9Y6QXi#{*2m~$^>O-meS$twpQKOLr|47lY5H`1hCWlDrT?Rwb&GD* z0o|tCb%*ZMUAkNM=%DV^eLAEE^pGCW5uMa2ozXd6)RVfT%etbc^yT_WeT}|N->vV_ z_v-uf{rXY;n0`(_uV2@1=y&yd`UCx;{z!kUKhdA+&-CZ|3;m`3N`I}t(ckLt^!NG) z{iFU#|Ezz}zv|!g@A?n@r~XS%*MI97QbrIml{u9ql{J+ul|7Xsl{1wql{=Lul{b|y zl|NM=RWMa3RX9~7RWwyBRXkN9RWem7RXSBBRW?;FRX$ZARWVg5RXJ59WlB{|RZCS* z)kxJ$)k@V))k)P&)k~$N>Zcl{8m1bh8mF41nx>kinx|T%TBcg1TBq8i+NRp2+NV0C zI;J|MI;Xm%x~96Nx~F=idZv1%dZ+rN`lkA&`lkk@2Brq32B(IkhNgz4hNniPMy5uk zMyJN4#-_%l#-}EvCZ;B(Ca0#PrlzK)rl)44W~OGP{z;irmXtLGq--gB%8_!WTq$?T zlLAxTlrQB^L8(A0mG%Za>)6)zzBh5rJ(=0SA%|^4+95g4*MRU_UG%w9Z^V0&fAT2}-(;~De zEk=ve60{^OMN88%v@9)0%hL+9Z_+-XeT(*O+IMK*rF}^Ii1t0&$F%R$KB4`9_Cwl_ zXrI!4O#2D#r?j8ZKBIk3drkW}?H9CP(!QYmJKCQB{v`0HfIkiV8Q{+Xe-8Nbz+V9V zBJh`h{|flaz+VCWD)85U{~Gw~z~2D=Ch)g_zYY8y;J*R>F7V$1e-HTkz&`-~A@GlY ze+K+>;9mj%8u&NB{{#G6paf_HT7Wj79q0hMfL>q_7zRdwQD6+11eSnhU=7#+Hi0c* z8`uGMfjwX!H~=R0#pD1 zAmAO~UEn?7ec)@rH-IPL6+j0V025#VY=8rB0Up2y1b_$-0}?<6$N?pw0knV)&;tg* z1egH}U1PmKMnpF@XvyO4*c`rUjY9i_?N)H4E{CnuY-RB z{9EAP2LBHDcfr30{(bNtfd3HuN8mpO{|Wd{!G8w+_u&5k{&VnOfd3NwAHn|#{9nL- z1^#RBe+B%e-j0c-@D zz-F)wYzI5QPOuB?27AC>un+792f#sa2pk4Sz)^4v90w=BNpK3B24}!oa1NXY7r;eu z30wwOz*TS!Tn9J6O>hg`26w<+a1Y!E55Pn42tUGY5(cq>G0|3>G%op zgnU9hojjdBp`Xs4&Yv!xE}t+@S5Mfd>nGe3{)zB3`ReY~?Gy2d^i=Yy^i|ob@>dnF zDqmH-s(w}TYV_6EtEpGhuSQ;tznXZ}_p0_)-K+Xn4X+wsHN9$n)$*$KRokofR~@h3 zyvSAXMy`QBavh=`(eS*ITM(`AOm0VXAUdCaaxbFqjiWsBvXm#`r#uZ$<=N+}yogwW zzw!!VwRn~{Ue@w1Vh^#8I6xdCju6N2VMe~On9t$IeEBk&aqwj(d}%Y2i=X)sK}FEs zSk3PhU-Q@D+5Cifg`guCh%fJE0YZonA%69n

4hSI1e4&>{2)L-9JBpSQCe{?5+- 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

Nw&OK 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 z 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 P40!s_t4cr`a#qgovK_^ox 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%0D^wd+N0L+pJ%NBCx6r0yVgFm_N}!at^KOKr1rAf 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 z�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 z�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 literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll b/sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll new file mode 100644 index 0000000000000000000000000000000000000000..4f309be7f6ad138ec936f463bfad8c1fe4fc65d0 GIT binary patch literal 101888 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 YWq3vyJ7TX0)SmMR$^;*8|Ed=F9}hIWod5s; literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/tee.exe b/sandag_abm/src/main/resources/GnuWin32/bin/tee.exe new file mode 100644 index 0000000000000000000000000000000000000000..aacad3799eb4909ac000c4f0d4fa0ec18773c471 GIT binary patch literal 24576 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 Km2mvue*JIUyc?MS literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/resources/HPPowerOff.bat b/sandag_abm/src/main/resources/HPPowerOff.bat new file mode 100644 index 0000000..94bbe66 --- /dev/null +++ b/sandag_abm/src/main/resources/HPPowerOff.bat @@ -0,0 +1 @@ +powercfg -SETACTIVE SCHEME_BALANCED \ No newline at end of file diff --git a/sandag_abm/src/main/resources/HPPowerOn.bat b/sandag_abm/src/main/resources/HPPowerOn.bat new file mode 100644 index 0000000..fd86012 --- /dev/null +++ b/sandag_abm/src/main/resources/HPPowerOn.bat @@ -0,0 +1 @@ +powercfg -SETACTIVE SCHEME_MIN \ No newline at end of file diff --git a/sandag_abm/src/main/resources/RunEMFAC2011.cmd b/sandag_abm/src/main/resources/RunEMFAC2011.cmd new file mode 100644 index 0000000..37cec7d --- /dev/null +++ b/sandag_abm/src/main/resources/RunEMFAC2011.cmd @@ -0,0 +1,12 @@ +rem Run EMFAC2011 after scenario is loaded into database +rem Takes 3 arguments: 1-prject drive 2-porject directory 3-scenario_id +rem EMFAC2011 results are written to a default location-%PROJECT_DRIVE%\%PROJECT_DIRECTORY%\output + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SCENARIO=%3 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat +python.exe %PROJECT_DIRECTORY%\bin\emfac2011_abm.py %SCENARO% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output \ No newline at end of file diff --git a/sandag_abm/src/main/resources/RunEMFAC2014.cmd b/sandag_abm/src/main/resources/RunEMFAC2014.cmd new file mode 100644 index 0000000..7f242d4 --- /dev/null +++ b/sandag_abm/src/main/resources/RunEMFAC2014.cmd @@ -0,0 +1,16 @@ +rem Run EMFAC2014 after scenario is loaded into database +rem Takes 3 arguments: 1-prject drive 2-porject directory 3-scenario_id 4-SB375 switch +rem EMFAC2014 results are written to a default location-%PROJECT_DRIVE%\%PROJECT_DIRECTORY%\output + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SCENARIO=%3 +set SB375=%4 + +call %PROJECT_DRIVE%%PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python + +python.exe emfac2014_abm.py %SCENARIO% Annual %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output +python.exe emfac2014_abm.py %SCENARIO% Summer %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output +python.exe emfac2014_abm.py %SCENARIO% Winter %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output diff --git a/sandag_abm/src/main/resources/RunViz.bat b/sandag_abm/src/main/resources/RunViz.bat new file mode 100644 index 0000000..b13f18f --- /dev/null +++ b/sandag_abm/src/main/resources/RunViz.bat @@ -0,0 +1,126 @@ +:: ############################################################################ +:: # Batch file to summarize CT-RAMP outputs and generate HTML Visualizer +:: # khademul.haque@rsginc.com, March 2019 +:: # +:: # This script summarizes the CTRAMP outputs of both ABM and Reference scenarios and generates a visualizer comparing the ABM to the reference scenario +:: # ------------------------------------------------------------------------------ +:: # To-Do +:: # 1. create log files from scripts (for later) +:: # 2. identify the input files names used by the summary R scripts and visualizer. maybe we can put them in the batch file +:: # script arguments (for testing): +:: SET PROJECT_DRIVE=C: +:: SET PROJECT_DIRECTORY=\ABM_runs\maint_2019_RSG\Model\ABM2_14_2_0 +:: SET REFER_DIR=T:\projects\sr14\abm2_test\abm_runs\14_1_0\2016_local_mask_2\ +:: SET OUTPUT_HTML_NAME=SANDAG_Dashboard_2016_calib_3_19_19_final_test +:: SET IS_BASE_SURVEY=No +:: SET BASE_SCENARIO_NAME=REFERENCE +:: SET BUILD_SCENARIO_NAME=SDABM16 +:: SET MGRA_INPUT_FILE=input/mgra13_based_input2016.csv + +:: ############################################################################ + +@ECHO off +:: Inputs from arguments +SET PROJECT_DRIVE=%1 +SET PROJECT_DIRECTORY=%2 +SET REFER_DIR=%3 +SET OUTPUT_HTML_NAME=%4 +SET IS_BASE_SURVEY=%5 +SET BASE_SCENARIO_NAME=%6 +SET BUILD_SCENARIO_NAME=%7 +SET MGRA_INPUT_FILE=%8 + +:: Default inputs +SET MAX_ITER=3 +SET BASE_SAMPLE_RATE=1.0 +SET BUILD_SAMPLE_RATE=1.0 +SET SHP_FILE_NAME=pseudomsa.shp + +:: Set Directories + +SET PROJECT_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%\ +SET REFER_DIR=%REFER_DIR%\ +SET CURRENT_DIR=%PROJECT_DIR%visualizer\ +SET WORKING_DIR=%CURRENT_DIR% +SET SUMM_DIR=%WORKING_DIR%outputs\summaries\ +SET REF_DIR=%REFER_DIR%output +SET REF_DIR_INP=%REFER_DIR%input +SET BASE_SUMMARY_DIR=%SUMM_DIR%REF +SET BUILD_SUMMARY_DIR=%SUMM_DIR%BUILD +SET R_SCRIPT=%WORKING_DIR%dependencies\R-3.4.1\bin\Rscript +SET R_LIBRARY=%WORKING_DIR%dependencies\R-3.4.1\library +SET RSTUDIO_PANDOC=%WORKING_DIR%dependencies\Pandoc + +:: Extract Dependencies.zip +IF NOT EXIST %WORKING_DIR%dependencies unzip %WORKING_DIR%dependencies.zip -d %WORKING_DIR% + +:: Summarize BUILD +SET WD=%BUILD_SUMMARY_DIR% +SET ABMOutputDir=%PROJECT_DIR%output +SET INPUT_FILE_ABM=%SUMM_DIR%summ_inputs_abm.csv + +ECHO Key,Value > %INPUT_FILE_ABM% +ECHO WD,%WD% >> %INPUT_FILE_ABM% +ECHO ABMOutputDir,%ABMOutputDir% >> %INPUT_FILE_ABM% +ECHO geogXWalkDir,%WORKING_DIR%data >> %INPUT_FILE_ABM% +ECHO SkimDir,%ABMOutputDir% >> %INPUT_FILE_ABM% +ECHO MAX_ITER,%MAX_ITER% >> %INPUT_FILE_ABM% +:: Call R script to summarize BUILD outputs +ECHO %startTime%%Time%: Running R script to summarize BUILD outputs... +%R_SCRIPT% %WORKING_DIR%scripts\SummarizeABM2016.R %INPUT_FILE_ABM% + +:: Summarize REF +SET WD=%BASE_SUMMARY_DIR% +SET INPUT_FILE_REF=%SUMM_DIR%summ_inputs_ref.csv + +ECHO Key,Value > %INPUT_FILE_REF% +ECHO WD,%WD% >> %INPUT_FILE_REF% +ECHO ABMOutputDir,%REF_DIR% >> %INPUT_FILE_REF% +ECHO geogXWalkDir,%WORKING_DIR%data >> %INPUT_FILE_REF% +ECHO SkimDir,%REF_DIR% >> %INPUT_FILE_REF% +ECHO MAX_ITER,%MAX_ITER% >> %INPUT_FILE_REF% + +:: Call R script to summarize REF outputs +ECHO %startTime%%Time%: Running R script to summarize REF outputs... +%R_SCRIPT% %WORKING_DIR%scripts\SummarizeABM2016.R %INPUT_FILE_REF% + +:: Create Visualizer +:: Parameters file +SET PARAMETERS_FILE=%WORKING_DIR%runtime\parameters.csv + +ECHO Key,Value > %PARAMETERS_FILE% +ECHO PROJECT_DIR,%PROJECT_DIR% >> %PARAMETERS_FILE% +ECHO WORKING_DIR,%WORKING_DIR% >> %PARAMETERS_FILE% +ECHO REF_DIR,%REF_DIR% >> %PARAMETERS_FILE% +ECHO REF_DIR_INP,%REF_DIR_INP% >> %PARAMETERS_FILE% +ECHO BASE_SUMMARY_DIR,%BASE_SUMMARY_DIR% >> %PARAMETERS_FILE% +ECHO BUILD_SUMMARY_DIR,%BUILD_SUMMARY_DIR% >> %PARAMETERS_FILE% +ECHO BASE_SCENARIO_NAME,%BASE_SCENARIO_NAME% >> %PARAMETERS_FILE% +ECHO BUILD_SCENARIO_NAME,%BUILD_SCENARIO_NAME% >> %PARAMETERS_FILE% +ECHO BASE_SAMPLE_RATE,%BASE_SAMPLE_RATE% >> %PARAMETERS_FILE% +ECHO BUILD_SAMPLE_RATE,%BUILD_SAMPLE_RATE% >> %PARAMETERS_FILE% +ECHO R_LIBRARY,%R_LIBRARY% >> %PARAMETERS_FILE% +ECHO OUTPUT_HTML_NAME,%OUTPUT_HTML_NAME% >> %PARAMETERS_FILE% +ECHO SHP_FILE_NAME,%SHP_FILE_NAME% >> %PARAMETERS_FILE% +ECHO IS_BASE_SURVEY,%IS_BASE_SURVEY% >> %PARAMETERS_FILE% +ECHO MAX_ITER,%MAX_ITER% >> %PARAMETERS_FILE% +ECHO geogXWalkDir,%WORKING_DIR%data >> %PARAMETERS_FILE% +ECHO mgraInputFile,%MGRA_INPUT_FILE% >> %PARAMETERS_FILE% + +:: Call the R Script to process REF and BUILD output +:: ####################################### +ECHO %startTime%%Time%: Running R script to process REF output... +%R_SCRIPT% %WORKING_DIR%scripts\workersByMAZ.R %PARAMETERS_FILE% TRUE + +ECHO %startTime%%Time%: Running R script to process BUILD output... +%R_SCRIPT% %WORKING_DIR%scripts\workersByMAZ.R %PARAMETERS_FILE% FALSE + +:: Call the master R script +:: ######################## +ECHO %startTime%%Time%: Running R script to generate visualizer... +%R_SCRIPT% %WORKING_DIR%scripts\Master.R %PARAMETERS_FILE% +IF %ERRORLEVEL% EQU 11 ( + ECHO File missing error. Check error file in outputs. + EXIT /b %errorlevel% +) +ECHO %startTime%%Time%: Dashboard creation complete... \ No newline at end of file diff --git a/sandag_abm/src/main/resources/StartHHAndNodes.cmd b/sandag_abm/src/main/resources/StartHHAndNodes.cmd new file mode 100644 index 0000000..ba4ffc8 --- /dev/null +++ b/sandag_abm/src/main/resources/StartHHAndNodes.cmd @@ -0,0 +1,44 @@ +set PROJECT_DRIVE=%1 +set PATH_NO_DRIVE=%2 + +%PROJECT_DRIVE% +cd %PATH_NO_DRIVE% + +rem remove active connections so that limit is not exceeded +net session /delete /Y + +call %PATH_NO_DRIVE%\bin\CTRampEnv.bat + +If %SNODE%==yes goto :snode + +%PATH_NO_DRIVE%\bin\pskill \\%NODE1% java +%PATH_NO_DRIVE%\bin\pskill \\%NODE2% java +%PATH_NO_DRIVE%\bin\pskill \\%NODE3% java + +rem Start HH Manager on master node +call %PATH_NO_DRIVE%\bin\runHhMgr.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% + +rem Start remote worker nodes: SANDAG02 +set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag02.cmd %MAPDRIVE% %PATH_NO_DRIVE% +start %PATH_NO_DRIVE%\bin\psExec \\%NODE1% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% + +rem start remote worker nodes: SANDAG03 +set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag03.cmd %MAPDRIVE% %PATH_NO_DRIVE% +start %PATH_NO_DRIVE%\bin\psExec \\%NODE2% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% + +rem start remote worker nodes: SANDAG04 +set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag04.cmd %MAPDRIVE% %PATH_NO_DRIVE% +start %PATH_NO_DRIVE%\bin\psExec \\%NODE3% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% +goto :end + +:snode +rem Start HH Manager on master node +call %PATH_NO_DRIVE%\bin\runHhMgr.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% +call %PATH_NO_DRIVE%\bin\runSandag01.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% + +:end + + + + + diff --git a/sandag_abm/src/main/resources/assignScenarioID.cmd b/sandag_abm/src/main/resources/assignScenarioID.cmd new file mode 100644 index 0000000..9c8e0f7 --- /dev/null +++ b/sandag_abm/src/main/resources/assignScenarioID.cmd @@ -0,0 +1,6 @@ +set root=C:\ProgramData\Anaconda3 +call %root%\Scripts\activate.bat %root% +cd.. +cd python +python assignScenarioID.py +pause \ No newline at end of file diff --git a/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd b/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd new file mode 100644 index 0000000..c4707c8 --- /dev/null +++ b/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd @@ -0,0 +1,44 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem AT and Transit Networks Consistency Checking Java +%JAVA_64_PATH%\bin\java -showversion -server -Xms12000m -Xmx15000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_AtTransitCheck.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.utilities.TapAtConsistencyCheck %PROPERTIES_NAME% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\input + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat b/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat new file mode 100644 index 0000000..487231c --- /dev/null +++ b/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat @@ -0,0 +1,2 @@ +set minSpace=%1 +python.exe ..\python\checkFreeSpace.py "c:\\" %minSpace% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd b/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd new file mode 100644 index 0000000..909a941 --- /dev/null +++ b/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd @@ -0,0 +1,18 @@ +@echo off + +if "%1"=="" goto usage +if "%2"=="" goto usage + +set STUDY_FOLDER=%1 +set NETWORKDIR=%2 + +@echo creating study folder +md %STUDY_FOLDER%\network_build +cacls %STUDY_FOLDER% /t /e /g Everyone:f + +@echo /Y/E %NETWORKDIR%\"*.*" %STUDY_FOLDER%\network_build +xcopy /Y/E %NETWORKDIR%\"*.*" %STUDY_FOLDER%\network_build + +:usage + +@echo Usage: %0 ^ ^ diff --git a/sandag_abm/src/main/resources/copy_networks.cmd b/sandag_abm/src/main/resources/copy_networks.cmd new file mode 100644 index 0000000..6219ef8 --- /dev/null +++ b/sandag_abm/src/main/resources/copy_networks.cmd @@ -0,0 +1,17 @@ +@echo off + +if "%1"=="" goto usage +if "%2"=="" goto usage + +set FILE_LIST=hwycov.e00 trcov.e00 turns.csv trlink.csv trrt.csv trstop.csv tap.elev tap.ptype timexfer_EA.csv timexfer_AM.csv timexfer_MD.csv timexfer_PM.csv timexfer_EV.csv special_fares.txt linktypeturns.dbf tapcov.dbf tapcov.shp tapcov.shx tapcov.shp.xml mobilityhubtaps.csv mobilityHubMGRAs.csv SANDAG_Bike_Net.sbn SANDAG_Bike_Net.sbx SANDAG_Bike_Net.dbf SANDAG_Bike_Net.shp SANDAG_Bike_Net.shx SANDAG_Bike_Net.prj SANDAG_Bike_Node.sbn SANDAG_Bike_Node.sbx SANDAG_Bike_Node.dbf SANDAG_Bike_Node.shp SANDAG_Bike_Node.shx SANDAG_Bike_Node.prj rtcov.shp rtcov.shx rtcov.dbf rtcov.shp.xml + +@echo %FILE_LIST% + +for %%i in (%FILE_LIST%) do ( +@echo Copying and overwriting %1\%%i +xcopy /Y %1\%%i %2) + +goto :eof + +:usage +@echo Usage: %0 ^ ^ \ No newline at end of file diff --git a/sandag_abm/src/main/resources/create_scenario.cmd b/sandag_abm/src/main/resources/create_scenario.cmd new file mode 100644 index 0000000..843d69b --- /dev/null +++ b/sandag_abm/src/main/resources/create_scenario.cmd @@ -0,0 +1,107 @@ +rem create_scenario.cmd T:\projects\sr14\version_14_1_x\abm_runs\2016 2016 T:\projects\sr14\version_14_1_x\network_build\2016 4.4.0 + +@echo off + +if "%1"=="" goto usage +if "%2"=="" goto usage +if "%3"=="" goto usage +if "%4"=="" goto usage + +set SCENARIO_FOLDER=%1 +set YEAR=%2 +set NETWORKDIR=%3 +set EMME_VERSION=%4 + +@echo creating scenario folders +set FOLDERS=input application bin conf input_truck logFiles output python report sql uec analysis visualizer visualizer\outputs\summaries input_checker +for %%i in (%FOLDERS%) do ( +md %SCENARIO_FOLDER%\%%i) + +rem grant full permissions to scenario folder +cacls %SCENARIO_FOLDER% /t /e /g Everyone:f + +rem copy master server-config.csv to a scenario folder +rem to make local copy of server configuration file effective, user needs to rename it to server-config-local.csv +xcopy /Y T:\ABM\release\ABM\config\server-config.csv %SCENARIO_FOLDER%\conf + +rem setup model folders +xcopy /Y .\common\application\"*.*" %SCENARIO_FOLDER%\application +xcopy /E/Y/i .\common\application\GnuWin32\"*.*" %SCENARIO_FOLDER%\application\GnuWin32 +xcopy /Y/E .\common\python\"*.*" %SCENARIO_FOLDER%\python +xcopy /Y/E .\common\sql\"*.*" %SCENARIO_FOLDER%\sql +xcopy /Y .\common\uec\"*.*" %SCENARIO_FOLDER%\uec +xcopy /Y .\common\bin\"*.*" %SCENARIO_FOLDER%\bin +rem xcopy /Y .\conf\%YEAR%\"*.*" %SCENARIO_FOLDER%\conf +xcopy /Y .\common\conf\"*.*" %SCENARIO_FOLDER%\conf +xcopy /Y .\common\output\"*.*" %SCENARIO_FOLDER%\output +xcopy /s/Y .\common\visualizer %SCENARIO_FOLDER%\visualizer +xcopy /s/Y .\dependencies.* %SCENARIO_FOLDER%\visualizer +xcopy /Y/s/E .\common\input\input_checker\"*.*" %SCENARIO_FOLDER%\input_checker + +@echo assemble inputs +del %SCENARIO_FOLDER%\input /q +rem copy pop, hh, landuse, and other input files +xcopy /Y .\input\%YEAR%\"*.*" %SCENARIO_FOLDER%\input +rem copy common geography files to input folder +xcopy /Y .\common\input\geography\"*.*" %SCENARIO_FOLDER%\input +rem copy ctm paramter tables to input folder +xcopy /Y .\common\input\ctm\"*.*" %SCENARIO_FOLDER%\input +rem copy common model files to input folder +xcopy /Y .\common\input\model\"*.*" %SCENARIO_FOLDER%\input +rem copy common truck files to input_truck folder +xcopy /Y .\common\input\truck\"*.*" %SCENARIO_FOLDER%\input_truck +rem copy airport input files +xcopy /Y .\common\input\airports\"*.*" %SCENARIO_FOLDER%\input +rem copy ei input files +xcopy /Y .\common\input\ei\"*.*" %SCENARIO_FOLDER%\input +rem copy ie input files +xcopy /Y .\common\input\ie\"*.*" %SCENARIO_FOLDER%\input +rem copy ee input files +xcopy /Y .\common\input\ee\"*.*" %SCENARIO_FOLDER%\input +rem copy emfact input files +xcopy /Y .\common\input\emfact\"*.*" %SCENARIO_FOLDER%\input +rem copy special event input files +xcopy /Y .\common\input\specialevent\"*.*" %SCENARIO_FOLDER%\input +rem copy xborder input files +xcopy /Y .\common\input\xborder\"*.*" %SCENARIO_FOLDER%\input +rem copy visitor input files +xcopy /Y .\common\input\visitor\"*.*" %SCENARIO_FOLDER%\input +rem copy input checker config files +xcopy /Y .\common\input\input_checker\"*.*" %SCENARIO_FOLDER% +rem copy network inputs +call copy_networks.cmd %NETWORKDIR% %SCENARIO_FOLDER%\input + + +rem copy analysis templates +@echo copy analysis templates +if %YEAR%==2016 (xcopy /Y/S .\common\input\template\validation\2016\"*.*" %SCENARIO_FOLDER%\analysis\validation\) +if %YEAR%==2018 (xcopy /Y/S .\common\input\template\validation\2018\"*.*" %SCENARIO_FOLDER%\\analysis\validation\) +xcopy /Y/S .\common\input\template\summary\"*.*" %SCENARIO_FOLDER%\analysis\summary\ + +rem populate scenario year into sandag_abm.properties +set PROP_FILE=%SCENARIO_FOLDER%\conf\sandag_abm.properties +set TEMP_FILE=%SCENARIO_FOLDER%\conf\temp.properties +set RAW_YEAR=%YEAR:nb=% +type nul>%TEMP_FILE% +for /f "USEBACKQ delims=" %%A in (`type "%PROP_FILE%" ^| find /V /N ""`) do ( + set ln=%%A + setlocal enableDelayedExpansion + set ln=!ln:${year}=%RAW_YEAR%! + set ln=!ln:${year_build}=%YEAR%! + set ln=!ln:*]=! + echo(!ln!>>%TEMP_FILE% + endlocal +) +del %PROP_FILE% +move %TEMP_FILE% %PROP_FILE% + +@echo init emme folder +call init_emme.cmd %SCENARIO_FOLDER% %EMME_VERSION% + +:usage + +@echo Usage: %0 ^ ^ ^ ^ +@echo If 3rd parameter is empty, default network inputs in standard release are used + + + diff --git a/sandag_abm/src/main/resources/cvm.bat b/sandag_abm/src/main/resources/cvm.bat new file mode 100644 index 0000000..7ded2f6 --- /dev/null +++ b/sandag_abm/src/main/resources/cvm.bat @@ -0,0 +1,30 @@ +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set PROJECT_DIRECTORY_FWD=%3 +set CVM_ScaleFactor=%4 +set MGRA_DATA=%5 +set TAZ_CENTROIDS=%6 + +set "SCEN_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%" +set "SCEN_DIR_FWD=%PROJECT_DRIVE%%PROJECT_DIRECTORY_FWD%" +%PROJECT_DRIVE% +cd %SCEN_DIR% + +call %SCEN_DIR%\bin\CTRampEnv.bat + +set CLASSPATH=%SCEN_DIR%/application/* + +REM create the land-use data +python %SCEN_DIR%\python\cvm_input_create.py %SCEN_DIR% %MGRA_DATA% %TAZ_CENTROIDS% "Zonal Properties CVM.csv" + +REM create the commercial vehicle tours +python %SCEN_DIR%\python\sdcvm.py -s %CVM_ScaleFactor% -p %SCEN_DIR% + +REM run the java code +%JAVA_64_PATH%\bin\java.exe -Xmx24000m -Xmn16000M -Dlog4j.configuration=file:./conf/log4j.xml -Djava.library.path=%SCEN_DIR%/application -DSCENDIR=%SCEN_DIR_FWD% -cp %CLASSPATH% org.sandag.cvm.activityTravel.cvm.GenerateCommercialTours "conf/cvm.properties" + +REM summarize model outputs +python %SCEN_DIR%\python\sdcvm_summarize.py -p %SCEN_DIR% + +REM checking for CVM outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CVM diff --git a/sandag_abm/src/main/resources/cvm.properties b/sandag_abm/src/main/resources/cvm.properties new file mode 100644 index 0000000..6eade7a --- /dev/null +++ b/sandag_abm/src/main/resources/cvm.properties @@ -0,0 +1,39 @@ +# location of input files +CSVFileLocation=%SCENDIR%/input/ +# input files names +ZonalPropertiesFileName=Zonal Properties CVM +ZonalPropertiesFileName2=CVMToursAccess + +UseTripModes = true + +# output file location +TripLogPath=%SCENDIR%/output/ + +# Note with TRANSCAD SKIMS you need +# you need to put the Transcad path's into your system path (see the end of the line below) +#C:\Program Files\Microsoft Visual Studio\Common\Tools;C:\Program Files\Microsoft Visual Studio\Common\Msdev98\BIN;C:\Program Files\Microsoft Visual Studio\DF98\BIN;C:\Program Files\Microsoft Visual Studio\VC98\BIN;C:\WINDOWS\system32;C:\WINDOWS;C:\WINDOWS\System32\Wbem;C:\Program Files (x86)\Dell\SysMgt\RAC5;C:\Program Files (x86)\Dell\SysMgt\oma\bin;C:\WINDOWS\system32\WindowsPowerShell\v1.0;C:\Program Files (x86)\Common Files\Acronis\SnapAPI\;C:\WINDOWS\system32\WindowsPowerShell\v1.0;C:\mpj-v0_38\bin;C:\Program Files\Java\jre6\bin;C:\arcgis\arcexe10x\bin;C:\Program Files\SlikSvn\bin;C:\Program Files\TortoiseSVN\bin;C:\Program Files (x86)\TransCAD;;C:\Program Files\TransCAD\GISDK\Matrices + + +OMXSkimLocation=%SCENDIR%/output/ +#These next two are to write new Transcad Matrix files +#CSVOutputFileLocation=%SCENDIR%/output/ +#ReadOutputMatrices=FALSE + +# To write CSV files instead of TRANSCAD matrices +# First need to tell program not to read the output matrices, to instead create them +ReadOutputMatrices=False +# Then need to explain where to write them when they are populated with trips. +CSVOutputFileLocation=%SCENDIR%/output/TripMatrices.csv + +StartZone=13 +EndZone=4996 +#RunZones=101, 102, 103 + +nThreads=22 + +# Tour counts are FirstPart_SecondPart in zonal properties files +# Coefficient files are FirstPart.csv and SecondPart.csv +FirstPart=FA,GO,IN,SV,TH,RE,WH +SecondPart=MD,AM,PM,OE,OL +#FirstPart=GO +#SecondPart=OE,OL diff --git a/sandag_abm/src/main/resources/emme_python.bat b/sandag_abm/src/main/resources/emme_python.bat new file mode 100644 index 0000000..4059839 --- /dev/null +++ b/sandag_abm/src/main/resources/emme_python.bat @@ -0,0 +1,31 @@ +rem ////////////////////////////////////////////////////////////////////////////// +rem //// +rem //// emme_python.bat +rem //// +rem //// Configure environment and start Python script to run Emme-related task. +rem //// Passes the input script name and one argument for the python script. +rem //// 1 : drive, e.g. "T:" +rem //// 2 : full path for working directory, including drive +rem //// 3 : full path to Emme python script +rem //// 4 : single argument for python script +rem //// +rem //// +rem ////////////////////////////////////////////////////////////////////////////// +rem +rem if necessary can set the EMMEPATH to point to a specific version of Emme +rem set EMMEPATH=C:\Program Files\INRO\Emme\Emme 4\Emme-4.3.5 +rem +rem +set MODELLER_PYTHON=%EMMEPATH%\Python27\ +set path=%EMMEPATH%\programs;%MODELLER_PYTHON%;%PATH% +rem map T drive for file access +net use t: \\sandag.org\transdata /persistent:yes +%1 rem set the drive +cd %2 rem change to the correct directory +rem restart the ISM as script user, must be configured to already be connected to mustang +taskkill /F /IM INROSoftwareManager.exe /T +PING localhost -n 5 >NUL +start /d "C:\Program Files (x86)\INRO\INRO Software Manager\INRO Software Manager 1.1.0" INROSoftwareManager.exe +PING localhost -n 5 >NUL +rem start the python script with one input +python %3 %4 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/init_emme.cmd b/sandag_abm/src/main/resources/init_emme.cmd new file mode 100644 index 0000000..97ecddc --- /dev/null +++ b/sandag_abm/src/main/resources/init_emme.cmd @@ -0,0 +1,37 @@ +rem @echo off + +if "%1"=="" goto usage +if "%2"=="" goto usage +set SCENARIO_FOLDER=%1 +set EMME_VERSION=%2 + +rem add EMME to PATH +set E_PATH=C:\\Program Files\\INRO\\Emme\\Emme 4\\Emme-%EMME_VERSION% +set PATH=%E_PATH%\\programs;%E_PATH%\\python27;%PATH% +set EMMEPATH=%E_PATH% + +rem delete existing emme_project folder +:removedir +if exist %SCENARIO_FOLDER%\emme_project ( + rd /s /q %SCENARIO_FOLDER%\\emme_project + goto removedir +) + +rem create EMME project folder +python .\\common\\python\\emme\\init_emme_project.py -r %SCENARIO_FOLDER% -t emmebank -v %EMME_VERSION% + +rem create toolbox +python .\\common\\python\\emme\\toolbox\\build_toolbox.py -s .\\common\\python\\emme\\toolbox -p %SCENARIO_FOLDER%\emme_project\Scripts\sandag_toolbox.mtbx +copy .\\common\\python\\emme\\solutions.mtbx %SCENARIO_FOLDER%\emme_project\Scripts\solutions.mtbx + +rem create a batch script at startup +( +echo set python_virtualenv=C:\python_virtualenv\abm14_2_0 +echo start "TITLE" "%E_PATH%\\programs\\EmmeDesktop.exe" ./emme_project.emp +)>%SCENARIO_FOLDER%\emme_project\start_emme_with_virtualenv.bat + +rem mkdir %SCENARIO_FOLDER%\emme_project\Scripts\yaml +rem copy .\\common\\python\\emme\\yaml\\*.* %SCENARIO_FOLDER%\emme_project\Scripts\yaml + +:usage +@echo Usage: %0 ^ ^ \ No newline at end of file diff --git a/sandag_abm/src/main/resources/jhdf.dll b/sandag_abm/src/main/resources/jhdf.dll new file mode 100644 index 0000000000000000000000000000000000000000..fd03758eae7789c3a9582d1773b6b758329fbc99 GIT binary patch literal 843264 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 z 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%0AUghYR?2?Y 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 &FYdX;dY>GC#T1w{l$c5hH5Bu6o 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

o~a@RWsJD_N| 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%0A|W4d-jl&hHy%+Jy1}ilyjp~}Q*UNqUC=yD? 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≪NlR{6ZgUG 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%0Dx=Yt?1;`TfIDgPqRf4abvbBzWfrQ(1Ut;!`dG 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�elR)Fh<71>3iiHr6q0F9@)c=v@ZP4C-qcNMWqDyk*J{QqV zM``R~isXRo&Cpo1qrX|oxa{pQP#U=kb zyElymeV~%{arr9GyZA$OzOP^OFf_+C5@;2pwSDOjF zTZ{s0(6JZ=)_`L%3ak;=ViZ`!Fek2fMGI#s-T^+|ncz*7hq(e+C6T|i;#MSOzxuKn z!AO~TTINc>OlK)W;CH!w;(CQFTGK~!4fk^mHMM{-E}f|wy8>@c2(wZf@q+3MS9luR zcZ|~^m>QCv;7PM=K=wjhHFQW)s8C%hvtKw;Fo0bZDDZZfNGT$X@Kfy9wklanq@>T? zucyk-0yh}9$PbOWyOll&8bYz_WUGVDDcl;7L4G{}$DddLD}5O6#01a-&%HqhVmME- zb(twHNyX1wYIcXC&w4+}()TGYq{K)lDeXN{Qrj9*c8^z5`Y0*dwpW8D6e}UWRG^i* z=MG<&KP6q0dV|0fvZOlOLW;Do2nPpf^gzxJ@W#LBOK3RTY%yu<`lkRo5~6~)OPZIV zGFj9oh)9Pef&F0vuhm%6L3_ZLqT?FM8A!E`z--X~iw)6_Oz;}uqe^rax*LJTnb}E1 zcOzWyC* z&-#SfzVmBR2l1eqF@=ado`Coe9e)!aVmcM<(E1)|jaa{)j$QbkFOh=rF;TATYEnm# zP!NrtJt8131;mnHIlpoTHFb!d!+}y^h@VGqWq>^jFoTC@+Trayn=T{T`UHU5%dTgI z_H$rE<4;D--eXE%LOwUWsOEr+T&=?FNLABgD$ma^?>s@QL%#NfclorRhcpwp>fBqa z!V7n$av(~*G!yWm(Vzg;sXuD>Hp?$`&|`Q2G#To4-L93IKr!>1=DIQej9)ORHl-y!Jx<)V!5cKkx9M!u^D40jGZM+Kb-_QD{+6 zMYs&gjYx|iGQNbzNWprk=E@E=sY~1{zJ89@4+lfflW+XN`^f-H%tOF2G|3fk`ag|; zuN40yG8qB)&mU|A{0W^NH^oU|L&9cANL2oLgUHePdsgbVkQ{Ns-!#1k# zNL}UWpZ63gSoAg=bD5F#d)WH9l0e+Qd7*;s{|}Xl1mc?gaw694A<4>LD5kS=hRb&h zJy5pmHu&u!BPtrX96bZ$3wOeGX}v7_p?XKeNK{Iw&4%Jo>@<)#lA7qj*Z>URqVpCh zxNBv&dzc?bt5%NiLv4OU*KN{A>r-hO66cHst1v_L3Tw+lkHT^||0V)#t!WnMrYV zIxwpNPomc{TBuL&AByQpc2nbjKuTPm%|ZjI>E$*@_0hi7#p*_WQ{9 zdwKeM$iJAPx>{}!$N6z|^%R5Ln801g-SmD*7O;@pzXygc*A?J}vJ+Sqog65&cGDU| z)xGN#sn;I`6Wh$*Fo^AFyV9_gYh9*s9T!UBx-oq(pIc8t3VB zbt3F*hO1=a!^HuizveA@=9@gJJUP;CrqW*GNxK;UU1=|?a>ShJ0LwDK?xZ*qs6J&Y zs1{RM>r4@9g49 z9*%u3%B$*r-r6P6un0t=pm|lTP+}5DUY@Nor8BQAYa$W4wRXxQ+V*sR zD4u61cG#Edf?M1QY-Una;POeP_EDQzH>pip%MUiG(NI;bM1zpA2jg`BE;rXq7un$+ zd2P|@uWxM6d}T#@J^7|E*W4Phy?W3w_LFxBO@VuxiDFrDdL8E2$f9|mO(F08S{{K} za;7wWR=vPXZ6TD+t4gbo80gNAQpQ1Zy1K6nKrw zj`qliw-)jbHV({eu6XL^k_?9trS!~!S<-7-Kjl?*p2Xl~KcLX*(sQzGygq2E8Y*&Q zIeugrXv+9jbnXEUU$p34bf{-`{;19qxMfLtkXTSdgL2N+Na`~blN-7mW3a|!B~d{y zxj}-2#HBS`uAl7=_%zLZr(}CsvVny0N|9BQ7w$Z%BLOie*?RxQ(b6Vq@{KFf^&KYf z;s&Kn_7vq8cAl_HzL+py{>pHm$I5V2;%Ggyqh(#3scZL;LNqPO?RMQIGajuQuf)?H z@-l8i$X)+v21LDhhp5>f%*27?aGz*<4$5mV!Z@8AhOtc(VN8yf;2gxcj)O9Qgo4=j zkn5ln$~XRy(!aiQ9F#B`lz&7+m9x+*bK!v_;OR(CBZ{sEbL-=HFwShne*}=@mLG>pHO|dS!9z`@oqfk z)|Zew;TT4`V@ZOU{mMl9or*2^)~nbppk?bWISfm`-;Cg-1CA8jDQY5~=M+~rT0iJk z?Dg`EKlon%jZ?AXu5>+5?vH#->P? z5fMG%i|9fl7_7#VP?YGTAr^`HFuyul@;>V4?Ljpd>Lo8J#My}y;{0lqin9wv9C3~= z_r!VDRVq&8mwX=so7RfjXAQ)2lN$i_?1CpvSWee`=GLm6D5~y_w-dL^FV9S=9)zi1 zW3PiG*`_b&sG8}f0@lSr#Dy?Nod!@4M*;3vsh`2n(9r$$c;BlSPnx$u`=dd|)P63O z#VXE`jcDU(0Q41o((>l(nq4{Ur#Ppt^=qj$9Kt!pl}FC>i*nxM@mk0lpJ5fI@u^j` zXBNvKr&Af`;%#PM=)IW~%vN-T4H(0jQ+0LPE=Q}R2pS!tw;JJ6B^$=MVqR=g!T1Rf zDZaoJpX*U401DcdkS4$s4CgQBIXfKay^u%9@0nw7jQ>#^WplK3Ntt~f%TS^{FCfTt z97m+FS(@uFIA>kYSdaqtH)N(%q04kzIiWyz#g=5SNx89IFT(D zv>NL1QPMzLE@=CUw^NF`2s-@<|L&($g;@Nb^o~3%a|WU20{A_GpQ(kU%A2i zz({5E4j=C(U~^uthj;H7AMaPB)jFK(;QcsN@UF3+%|?pcsvwZpA~ZPaW*}$#kmo5! zS9y^CNP>((GLW}jYB*v)ghD>>w6tZNcj{{Zf6n49?B_axZ$wqDfJL88-8`QJIL8Hi zy94-}h60%LMts0apb?5^8Nic#z_luiYdpYxk^pbL!J)W=1Na^V%v#>Hc7vP-T-#PJ zFzXF%z-#S|bFrq3tG3z8N#$<~%v5j0S|hp{w%_1oDaTJ&Y4u&!zOYF4?apj38_9qt zY-We6WtvH)|NMCkT=m%wv5>xUgv#W7;lRkGDWryClReTqL$0entz6Q_^f$uZAkCMi zS15jDI6*skQfcAblfcY7fKkGTG$)+*?oRj#%u=t$S{}Bh9}xa#EPRRbwPhKMO=exD zxtYBYeIG9TL>A`QcNEHHbE@#Q&9$4Vrn^s$iZKY(_Bu0F?0;c*Z!Mkc6gzi{?@XjqI*b=#i?a z44!0jsnnA+FNP0z-{JAh*cZ_iGf=6Zh5bEK&PNGEZzm^hVG(#?_6$QsE1!xMO2t3C z>Zu;(GkY6pWp;Hx!>qXsc&4`Q62dY$%$Z6}2uIvlIJ3QRGoNg4^HTRi3q`}_^Qf@( zH?5@V()Qy+@0O9lezk8V`zSa@vBZkVw^HJprO}7lFCgu_(jigw9^GGom^(vf#pxun zB@u+NXY!kZ-t5WXvHk_-aT_2Kr|@Fo>2<#U?QH+MDE4gfVR8dAr%Qr~)I9(ejSOrysIJq}#66id}JySFtA}wovRXPo(Ex?~Ak(X%*?@ON>b6 z{Jl~P<^si!!A!!hctv>GP~zeb$pIXbWm5G+P(r{ULqOOkpco{f*aDA$*hrs%f00%K z4)j$5L&VMf+F9b_;tAQ_>EwqFul;ziejmU;_@UEy<8Lz8x)cG?aRoR zB8M?O)Ls{NRi|&qRN8Mo!=eKkkN>rtB)$Fn^7RqEmeOfS=f&8$eWBbUo6Up7EOPYa zo9(;n@7(s~`uk-2GWn(bU148;pNjeeY=uyUunvVmr2X?pY`icMVhFj}qdI@Y9uQ3BK@*~dc_Eb>MWsp zU2Xg-+2LmO%Bh(yctq0pnf80dpRvaXh=m zdzg#q-+c_GU)0%{?>HOtJ!eCzB*ftL^1D1|BVsCK?oef|+DEus5w`FU9=X;>_#Up1(~Uw@kyzy$13)AMycZWwZymC<*fVs~uL_IFR+i30%U9*o@*hZVw?9WYpCF zKG6sK4+Sg>!eN0Hl>nHl=eM|kmxYatKBOIqJigUuujct?dddBTSzg&l>5o$ol;x8SZgf{Ip_80h*Ndt-d4QU}m zT05UK(SuNIsz=&`!+oA3q*X6A_BK2(A{&F^I-}c1Zr0IlYLT@r(WEp#!EY$=(87>T zEr;ZUWE@8E_W!8z8vd8aoEmuJ3Ki5tLX;8IH(=z7RF9g}Bx-&e=1^1QP$LUVq~p?t zZ(W|lA+0;%UX2^zJXb_oqqrjC5|tG}%W%Q=SFT9AVp;t;)b&JgJAz@{js+CgdI6T#6F-^%wymLg4rI&`oK^&*k25$gR2E);vn z6YA?j{VJYAnp{Eqr(#pZt|uaErFG~gy5@L7bvzLlzwe3sN?8u(xCGo}2>72pk$C_G#T0|$N*_gYtJZr60aZ7Z{Bt8!v)#xYc49wJ#{5%KitB>H##Zq1kiF$)GGfb(yh_uo#reJjkb6BXe8d4zCKVG6U>TMecr^6B1de{a?-Z79z_iUC1Y_(79dtUM8 zQe4ImvUp%n3+D1b= zp5MKiaGm|aN%WyI_I*n}R~Ipi1~h|*b$zWRAzh`)TyD@V(1UsdQIjoo(&fVZzom?! zBo+Lmq;9Uz6oKoTNujD6FViNzW1ttfem#UPnt6aqS{a8(kFYXZLMJtI89Ncg`!N?n zaQ@P%g6SC_QyVJ8V44j4#6wg?Im*rsKtE3dFV8^o#K-Eq^itV7UiQD)J1#Ez|D`>w zzt6OXCH!i8_>Ks@(;gmLBlWF^9T{Rczib_&oQe(T}4hmz@a+QTo+gv+me zE;Buuj}r*V?O~bb{npPL_ptSpyhpc(&u_2O9tQ0bV&i`Cm%ZW*rMTN3dY|YkXFe7w z6#Kz1GsP?OP&s9eZVzKXnbaP>)+LKS`^CF?#UGI3ZhQDR(?>Z+weUB;Oe3#MTPgFG z?cpb}qyM!%Jly<$*&cLo;I)SiD0e8P2L+qjJvhkK(s3q5d$^hu?O|nqZ4aw1_1eRn z3%&Mm9rbV8!?DmwdyxGR$?ai&P%tGeG73;3rag=XzP5+3Qd8^8WiQe&d5X#Ne{K&1 z*_qmRx(0&TGDK&GbgA_v%Yk!i?E{?N?hGwnrP0Idolru&^FH}8g2{n^Q0x*)8iL9$ z_xs*L(!34Y_Z1lbVjS6+CC-sk1&2$zn)spw!F9%Djos~W?Hy&_w7KfSV98_fKL6&UNXV=0sEL?O?Hf2qoqes zI(`os#73^NKij3>WdZBS`hBDi{fB|n%QyvDRaLJtPwsdXOKWW@t`60wv``$ho7|+* z6A&*VJxoy!sEKhN*4eFY#hlQ+$-<50jYWK?0>KeT!V^d_*b-MWm97~dJw-AmE%03F zWo#uGIgloG0HHaZN?&dtFejW20mV`T7>b?lqbMi8VQAU_P0`@TlcEvaOiM@Rcl!<&|aAblN=TG62}9u5JB4Ta%i;cFO<1d zTkqQFBmyLa`Z`AeyeCy|f1Q~Q8waE+CCoGsq>#|TkTA?A!3K?4#cW7OOCsSwe}{zL zLINkc_5(bT7V)PVo|LY7BEWY772qqVOX{b-54?{8mt~y>_$vT($M#RSz;|~yz<){e zj`lGqZf++LmOyknAHZ^j!E&v^@}zJAR05)-QIAk;vxlWP2}@TO%S;E$JpzpBnI#{I z^tMuU*)pmJ5)PaNW@aitrS)&;0riM)*5jv7S<@oUb#QndZ~< zfpRWR2F37A=9T%JKSNr9?dfJ}cA3(&++M#@8hT)s_#4$_^@Pm6AIoh93s@%GKz3SO zh{HkAa?AlUv#Hm!&;r4i8T3(+3||q++pujM4}<>PB=qfE^fwFoRjfeWAD>4!&nzn< zSh#&t9h@sUa~mjEIXL%ztvEk5y>F(EvxnmR-edX0em=|pCJhYY5$Ab0<#>DTf7a@R z{o>(Aw1b{iPAEnO!Qy_?_M}ZtCu#R+ic3+Q-ja?_{2dNi3A`p@1rAW@7L|GMB3e7@ zF1s!!Em9)886wAveE5uDVh|FFNekhbdMQpKvaw5KIYdeju1s|P5k^%Ka(ZxuR8GYh zNO+8HPSCD*(CS3Epsih~Xcv@eLpUhd4cZ=xRu0KBXxCoi%jR#Sc^kBMbxlIs2DGKj zxQ*v%E2V{{6r2cTPo-Ulc|VFVuqb8`>8qFzDksEJsMustNDf&wW-02e1CNH+Yqu@U z_3x>wv?m)X2l!MLp&6mru^yF&`}$OVPg<$Whf0ylG>IM(qma5mX?`8=`7CYwDE#;O zsTB4K8Y6|-aLSdp9{eYh;NRoI-_gauKSkl2dA3q(sT@p`lkSCFGl7ccUv6;R=i|sy z94B}K@7hVzdGAtnh3Ex*yn~n5>NC8Ex*vvuj`T4Fy5_ZU@5}!WW{*H!L$&ra#Yh`i!u( z*M2qNbsJ(}P7$uEQ-A8K#Ao}M=P73Cw+-eEeSFLdNvp&%4CYd6U%X6^p{6v~<+cqH zXUHN*o-yp*ZICP$E)9E470H<%k{gqdT;(Ep3K_RKy_9bZGyuWv ziZaN^3CY#~k%X86#~%E8joJgDC0b8uX<}&U>C@6hN03ud`o4iRn7z^w!5rZ}|-N0m*(M_zcr8D$^Hxh$2adCc20g<@ih|8kAal zv3!J^3tk3XmmigsfDdPnhqHlyk@dA{t01d0U-14Ai`!b!PG+6cb0? zX^&Td1@k6W)df1(1A12y&>a`MKsSdKsL0O%UBHV_>{1_SBLh?p|7DG7dO3C$vu@|i zq}!@4SOKKI91hX91aVz}Mj|dasGo%FCJ))02H8`FB{&$M$S(JhJ$|-wASTUGYi4$k z&6nve??(#}iNyfuB%6OP5e%*IorX%70IUGJ*YR#A~ zZPU86P`1jl#VmK9y{9t}Sw(2q;sQg*@ugNoS+)9<gS$iMOM%8)_@Ah3H(CU zHAau!O>>Y(2E^W~jjx7z9b^^9lb|*&N>oqJMM93Ao~qhXf-N)Zw4;~bAXbs)ZO}gX zEaZ>B-c3v%QXXCsIb3hjiJUz^H#4AnUsg2KUR>R@zCDi$fw*kDyI~_hmU3PQ4<{%O zvTPJ7>-~Iv3gxE50|_=@3@Aj2^6BX83#ZHG47~hW_3~?=7{Z5#mGH?zKP2JyEQ7dE zY_LaoyClMocL~4xOhdRV7=jc8KA^j_;2CF#xK*AN)&Sh(KG}8T$$6Yd3-ghoDohzw zpYewZb9HZ}=^G)<(6mL>YnVsVy~VyTr;%1+uIUU-Yw^?xURY&0vL$awk=T}X_PI@D zx;_wnLG*)j1Az?hBMhF~d^{V06^f1Y@N`JRbE1pqI)f*{KFv{lT;t~VA zkpUj`fmbT<8$94EihL$wq?L&@kBJPaP!3)jkOc>|>l0!PCSYa7n`}p2%B4T*)5@?? zj$V-!S+WY!J-(Rx(^!BUj7$zwW;<(CWz!`y9PGvB3N3)2D8SK_nx5T}i@7Z{C|l!2 zfUh!T#b~|j~moX%d$kl&t>_dJv{*kmF!ChjVyQA3J z#m+d3vqs~2tv{GdNVq%ub@@d-Q|ylk@L?xUqjF}vb7D$Sb)PCnuTC^#$J!V6BqY*C z?U(&Ebs)_|3KZ^=0{w1t3bZy#K!I^m;77n?zNYh|(;RkV20+f4j-m_gSg}!5eaY=k z>8+qc@0Jv~-9aaS(8nAqcDsXlJsh0~+DA@jAGa(BG+PhW*sqQj*s}CgcORVcnFNey zkRdTgv&fkbQS#8aNuJ}u|NEzsCqkb1WU9nUJ;-pRl$9~oX0%PVZmA`;kN-*6T}Th# z23kU~`(ZI0XRGdpIA>mqMtVUdjZqz4fli}u8D8>gHl$r)~M#?j*# z?J=Xm(cIGLm^40R$&DU`lv@*+c|ZQLj+R*fKJijDK{CZOT-m=5MR4lu45|pGsqq#e zYJ5CHjc7!$Vozq4)0myJir3g{mcVYbC)#Ze8vTuG4#kr03mOjq18Qv{piTpn4r*m* zYXkeP5Y$i=_84*q*LD*kM@y6*j0@ar`iV5K8G2v&oC zkl>$Wpat!>Vq#7zkCV>Len}q!Gk;ZHdGV;e2pcbhn;HVIskpG8)pVY+WsODn_Pi=3 z7u!b+N99zV`olaSKsJ8Hf8#&h#wlwKO>N$zwJBA$h#eeN;r%~{qtaWVp){+z13$gRp!ON%+4ldLE$g!k@ttJHMq zGzgsz_nW!yP?Tl(i`~jXpPb5i1M4Xc;2XF2mGx{WCSx(ZX3xH2t|)L_mQ{iAha4r2 zs{5vLKz08wT$f;poLV`GW0M@tBe!y;+N=-R+!u<;IyxG_ z*h0_Wxv;=hz4aGp3Y+@~DQxaPE>xSl{36fhZa!5kz$#V|+D+Ja_NRV#Df1@D{chKP zNq)cI{S6DJ^ubLT*EgI2QQGag`@zOc(76WP5Ox)5sw>65JzqOju~nU4_huTFLS!g* z4mnKAiRaUDqU4Vh+*aYtG~7%=L9A}6AedC_-aZrYesVk4lV>#0fj!932Fo*o7 zK~KL-4kM;T=Od2orSv}9jm zd%m%XgTje?oxY>Tw_j_cP~>|k()( zR8BW3>HkR~zW0qTe$0>C|F`wT*X%Si%kpIV=|Xa3$#|V;VM$0@m^fNE9L2d!{^G01DXOL!`OlDt{UG# zihAf-plV#ok5dl|scrl)QR6!?4=hDvAqMf`D>Im0WQy^tVdhQlf;#UU#dh^O@_lxs zV8wXkQ&FxiolqLIKV~XfZvU!ik-HHIR9{@rzPcZ;)WHi<)7gtzR&imY^c3Rv(5P{KtFnXU&!Hz#a2r04@ENL{c|XCpkC$8R?jT3%TOXH zLkYGB{#?;gdQwi{0h9U^6-~lzN&GK7&gmn8`7Npa$VuDUu^S z@l2gT&g+>SxI+SR;)|rjB{ma^*xbZ3wS8G&=6$^5v0lYkd08<;DAMJy-_i=Gi**-h zT`Z8ML3QCr?4puFvAGy|Q+-{#`QyHY=V?_2?F?!@vZgjJ%lqueWU|fIuG=;W!PFK`al4qg2@R+=r`g)FcbS%^WA?Q?-BBe zWAWa`tg}GBm<9U3*gubzE?1{8`_XuQ4{2RQ^0h9yNb@jt@l98$i{3cTrY`#PtW5@^ z)|E7GgZ9i;&WUI7jjYPg-BDB69S@*ZcQ>i)8yywoE)@Q)t1Zm1PqU8E;KH-AgrD)h z!QIHpt$O`*@)4V+zxKNs!-7 zevD?@Bd+iU+2VdYr9IXNpUv^6B4Kl7E?q!#K&iK8zjC)w%Z-`UndKCU&Ip2+Q;Vs| zwVV}`W8d(sj)MhiK%tKIq3#0~VgzcTKo!?bi&ke!|7ReaV`uvi9_Do@HpYkWrU&83 zxj>leL%7s~P-K7mj7P%>3Sq1dVS)!?zCe(BVPw=lU~nCQ%tzljm%uMJ%&Yg96l1Hl zOZsGK8jSZM3x?ZceZ=P~gbE+RLCP8CdkO@CkS>HTey0OQd@~kgBTPHn2eK4h;^+e( z$eSLJBj*VJK9FlXARGsnVt+Nq6Txr=Qsn~~=K+Zc5ay*_N`7NSsiB0MNmA^meISQu zi=o&QA4n$;NQD6Tg+(S#T`rJjtL$*0P>Ma+2XdzZnd$@ihN4C$XA6*5eIWfjAY}ri zxeuhR0-5Fmnd1T3odb|jK9JqgB9)SI0kZBNo=oQQg(CocAXj=oUJxLs`ao{>fD{Ul zyL}+t6v!PukP|&1!v)Ca?EQ6QayZYW;qZEmr!h`4`c&HjZ9h!kiYvt?(%?) zqB+|sK9If&rWe0K>9^SBIZl{?a@iT`6Nn=Ff4P+s-26T+ImOoJtyw&bS{95eBSPVXMAeivIAz=-soO zmvopAJ=~69Dje*4x(fDJeV`EmI>^EP`|Z-TDr@^yOXm3*uaJyqIoRzzhlS2SbOa+{ zXuVm%zUAlcEV)xOH_=agVR1UDyerBjYOX zf}pj?Kz+f7da=^_ya%;Y64ZaXP=7l zAD?tmhiwpSQ#I(%ZM;BfE$dnfRUC1tuUu?r4%CTm8J)hY{0#KLo(2br7n$fZ zd6Tt&_3GMCHYbGnQ!uq9b)Si=mdM0+TNzuOH}Za3Jm>lq&E`b+`XM6A3vEv1TR6ey zWJ@w+I^UP2a5lbtmv7(i^}(;kgamU}6b0T6=O|q6=w4BA!AS2sH{iSA4fGNq?B2Ur zvSrHsWb)mB=R4niOor~w9}!=Wv0Rp1?<(<{qUxdbN}@SM)>Hb5`mw>VhQ zjF?1sv1KLEKHCy4qMYgI44j2D;uVS1FfR*?7+W_uLsT;#mnKPhG_GQXqpD`dX6qZmJ6>b75USL$Sc@c^=1CwJ@#~G+b2){IzzO6#m z4w26V*UAI;dwT`bD80o!}3is4No7o@t87ok{3A6N?)m?%MX6vW!+T`#UM&(Rt3EUo9O0WQ&F zlkmKHx+@gR;3@U++@N^Q@bS!R;t9o?EjGtqN9Z+jph<;zJy*W3>rSo$AWD!57{}}B zpjwav9HS=ffa^AzV@UHhXb=1o>;ZkW#}&{orD_|gXF$J5%W@= zo{9ew9o32{sh~Z&OhQ;agNa78h6c5AU}c{)`&3hiZo6lieWH0_4Oq_%yRkm7o2O@% z{h6nY7eR@ujdyi)wd6WzqIx?RwLu$;coB+S4g*dbx6+K3RHqLS*A7SuX` z(JM_Vh{@a8LDi!zwaEzM@kYup?95mqSceTEhCeW9-QAuh5{cKV=r8|uqhG5y7i4?s zPB_%eM>f<$mgXV*v5l$v!~+}vGzXxQ+9kthd4kN#%Tf-08Y~aY`i$S2;tRxGh`c9= z8_1R!IZ7$a!2r|w4b0lg*Sy=Z19!^wc%aQp^75G#*wgt%+-4w;1ZZ>Y%aO9xnRB@_ zG*n%J$1>l%q623&h^+m`^B6Bh`7Ia35@jCFLm}vizhfIbpw*cp44nG%9mpy6wROK2 zneWHQcXcj$@?D0ON(U3iBg57gOeb-%Gd`j3&sz69IbFcD?=9w{Dm`4gIGq3a1Ur7J zxQ^ASH_{u0Vncj!w=|-qZa1I8aP}rBAIabURvu4s+bHrty#IT7s7p2RqPqN)8h$G7 zbvd}E*HIkYY&dwCZX^^N?Q<{~x~Tv-I6*j=3;UeLSSWp8mVKDG72SzUQ4ZFE6DgRO zBr%R%XR5}_r&ILPE%J#Kc*QsdQ>MN@j1ZPIAjmiFGbt4~9xmvPh=6%h4J zR)5dAB;k$+LqR;r$h)HY^#ik>LnflmI$0=t#mac0iYmW$%>^n6IbrxAC3dGfI<->2 zFeD<~8LeDmhnbYL{g2xz!KW#~_R|d@RVZQ6!v#M?1>)i z)<8@=OTWyzL8`gdE+U#ny)kUQgiI7ZRSW;*7B<7?Z(BlADZW+uYLg1;E#yfv>)6LI z^NFR^>0dIQW~WbK^~DSANoPw(UGFy8j`AK$WMJegHsm7ttE2*q)$0pQoJ)d$jCM^1 zh1*dN5p>c~&#mur95&OAa_LUepS#}S*8WUVI_KY#6#HP`ZzkboW}x6c;>-r^7CE8Y6++n@Ff(}EsRDQmw5zE zYk};0(wST76X}fX3&jW{M5(eVy0l7`rMUWHbOIs1D8-seF+xDg?b+jr)G>qwc8)jaNTk)meBr2Z$$zGwKi$ipBl(B=`8Vy9`~{N#5A2ge z%v-!Q@OA#=EIz!}Upw*UDD=NaQAN*^#B9r;|^88i7$oAMHgKZFx!&Rlid_DSfK)MxHS%*TjE zdU~eh{Fmkw7stuD=R|VKmN!8vv!ZEu`+I3Bj>;FtMKyFgl4)R10NSYhX;Sn{Y=a|@ z0&yj+wS6u27yG<5Qj9~vpbMIGQUeq@>%_4y!vA92+);QN1#c@)GrTRjO=Tn9`=h)H z#jd4%Vua9W`b8{h*bF0EkBuz5|UrZ9BvNnz+dIGJ44nLoDj4BfSODi}Hm?3J5UFMrxQ zE=KQYd$)sAd=BQdZr#TmS#lh|(2-(yY3?m&+BMtjTc&`WIV^G*x}I*1_{H(fYu$}~ zgq7`TQqSO5AuM7rK?Me)ty}R5&L#sV@ircQ))7dpZBuco>08#>*JCD}%XmcGL9MQ8 zpd?PTmPK~9;jBSfezZrcA!!}-sh)r;Aq0{TPydh4}M!*Fpcmm!cRfK?7x>a=bepkTn^9uQ70Y?S=j_5+N zfR8!9j(|Tx$3n3!bp*@_o~ifg`)#~;G;om-D_v4^2r&YF_IMTWm@F0WjVPisQr&gX z7w`*zsDKk$di>~;x*|Qmy^%)^UjJ(7s!~2$Ux|-z&+B>x6nN++1tb@s(Znp62z2qzeaI}q(UmRAU4Y=WQldQ}y2>FO3Fs^@kiuHMz`HjHC%wN=2bYb?= z9*r4;_v~3as>C_F;d`-`rvk;P#J-SX_`e8jiSDc+yyDNgPsw-b3_~%_@^MRkq$b=dy@Qj`VQUzKT{Kg!bKAfd`3uE3Va+Ach8)A(@g{`{NHM7(X5dUi<}&1u z;kgJB&9#w{B~@(ZyK2aliy3PzUqa2{l}R|koHO;jkKjQMK`h|P;2$OxOgzO-q$)^x zv;7Ax)603X(!Gu1#_{4|;?h;}TW{xAV{;7yl;k zTZil)zjG<5oFJH|9ff(MU?!?m2lHef^AX^OV#_?t|7_-pl< zBJ=n#8NPA^0H=Sv4(odJgFWo?rnI9s+b@tex{D|*N;0CT8h0kAPzx0^+L~;>qlelpLugclad>ypoA@m-G!L6^#E)OE%P!t+eE4e#r)2 z$$4JMN8OU;TC!e6VaO_OZVty4^snJjP`LA%t2b^_2i_rdUw<~*>)Wxui%VCUm=!t|BLt9JeB^3M4w0Oi(fuoy#d901*M z_HV~3(LuXtud+T$Xkl;~EoU*~EkRmEsdZMgjlHz5+`!I`rQT_FeEzXw4g#}Qpq9)g z24;$W>LGaSZyus@4M;nvPW`pDBDu~WIU>v(Bk&V|I6J{Z(lQB2Z4-y%z%6gCxZu`MZzgJVnyhL-IC&T}~c%k;H#iPB={h^ZJ z!u`6jBbr8czt}B`8E{=73 zR5E*|?e@wbU{raV#M5X~}mKON^4!HqV^ z>-)&}f`GB9hrCS^@&+#QOAT_l<|%roU`ad%1(P0hC>Z8Ykf#&`X2P??cfZG<1Ltqe zl;L|twnnJ8E6xi%oUf$$4BtnZTtWNupOoSH;LKN?ftf)A_aOuKDFgRBA8tp5D_3Oz z0mCpL32rwR?wtni%V;lh3fIWp=-9K5F2>wMduM2C)guS1Qy(;1@u@+yMr3CgxC3CJ z*df-L7(}}p`3$TjtqimV5mHztnqZ3EMKJ{Ob-Fuc^B?l!vP}dsIdC@~;05cq;UIU# z{R(C4k0J(f(1!XgqQ_28tKNaeP^=#%93(FrMq&RlLkVEdcW46n^&kgj3MjFQlNEZR z0ReA0=u5j(T&?jE-JgS4WSgV7Au_=6{Tye^a@6_(=j6851?q?$dZQ#lYlFjMx)OuvoEl$rk_wSV!?=uU+MpFTu;SX~OT_zQg5x42=&Nz%v5w+Z?! zBHv_Sig!EB>ROLJf%8o@aBUUr#~@gXBRh{_AlV?Ys-cxSi||)u)p0E6D3Uc&YphLL zaYKKTE@zwS(-Tl>r8k95ko<;1gkp2t8lzXuvC>~kMQgOfz1O6I@mHi!W^ci_*1ms& zb7mvIqE+`Rkh@(V%z*)1cUM@MSyE)3J;W{2*A$sWL*aB;zeu(ektqt(X|S~~gvI)n zUEiX9bGWQuza zb4WnF38<;`ZtNHFp&l@>fch-35*=A4mbjFRtu{)Rf2*JIb`SjqgA~;sMhet(nkeei z6}8I$>BM1YpY{kEqRa%;o#Xt|@cZa^Mx@}<^Hr4@P{{qz=<$WTKrd44tC?JY9SNc~ ztMc^tMQ91dc0#Zb(y9gscsm6m1=n8S=3zWR8ek~L1zQABqoOj2WIsk!Iu7^R>9@;CRXYm!deU~i+MmXCH89$ zZClPEqfU6_YAbyT1cTMuAcFqI6|~XF6CvJcFbfXqn_$uXG7u(jfF%oasG)R(~M(3llEYAP~vKT zzLQezsbfXAvWY6i`rXXZ)bK$4rr5hvJhgkY(o?$}2;m?tTz=HXv zPY@<`cxrEBF%ultD`?{@t_td@{2s;71@(Qk(VhVPSk*&Y}I)S zl$7tCR6bYBpYNBy+$;aOl;^MtQ=a8aDDNt}NmAj@wd2h33%BqJPn5!P)|MbY=Hw%_ ziF^vL>EKg7LkoBC3x5iWM*8PT;XNW(3b$ZhSHb~eC{EKdSNLUK^2*egGV`^JW*F=Sr~^(j02tju);;h->5 zTdy+9KEAC^x>f<>B!w{+7>O1*9QM`?sxL$Mv|0`K1Rwku$Tm#JkF=B&_IMpBnrn8d zHrKx&OLepmUu) zv!%|^s+O0lc>d#0pnnBAgQ!3dZ3P5Ho@ws|Nu}y{YeyRMOTFrqY9yte5yT87b(a4I zSqXI=Gac*fD^AzGSmyd`RQU6K5W-jDYK(AYo|RdT=o+;pX0XG0Wl8mc{x}r-$S?CA zWel-%{-E@kpm-7eL5NbnMT(-v&pyn{UQBjVzds$;`kmcC>o?ZWtKYM}_Ud=9$W3?4 zI`anq-}U?5xl+Hnw|qNEt(?OI{+#;#*Lbgfdq4_jDnO;-WF!Q|Z`b-gB1$52(!aPW z5;cx8o$MF#O~hQCdI7++es_Fx*qf6+`&6kh-C-#;_PUC0k1vZuMgmx~fr79Z2lCiQKeR7JKkPk@Suvfp`=ry^}& zlk=dx;aN|JO58;I7bO$@*yyU*=+C|OFW-zPTLCHYCUyD#SH^>#IG#buF0ZvLrQy_RErGYwBD?BEk3G88xa&9G zKO0g_TX~hF3SvD0TBohtHr5lwi4ekN9gW?#az5n5Uxdp@!NzkOSu7)=AclPA%Hj?A zW@M28Fl{TZQGZ4ZtLz3rBL)$BZw8W~*m6)f-To(rdBo5eY>5l3Oo>nGiXYmCbcz)p zyQhbyo};B#`=!TI#%S#$2V9++N(!ACTMJ*+nZ1yuBgFT=pnjZbxZ9OzWg2|}Da6-{ll%~0Bc$kvZ|dis4u+`RIyx8@J4Eyz9rT)H zvi!Q8nJm8!JgNLj_IdK#cY`Ot`=EgP(K%|=RP7u4kl#v`UmwwBk>5qGCU5z~m0xH1 zX5<$In5tp+PG5dg8Bw_MTfo4N%hEyNsNn!l5-)%)G0Z)my(96DP;9K9v6Gi^s$^u- zzX+NEQ%SkqgcvmHAfm2O1ULKT?l1yI)R$4th?a4m?fsFj2-N-)| zGUM}9{;Q-OMgAYTRe1S!SN?P5n~{HGfT{dnLT6kva2UVHm4Bi3myi15*<^%*cuoUb zqL+%5Wf6tk10%96l~~DUsjQ&gfEfj-mi4}K6wZ@AoR%J(AAWVkJcbk+z50-fx$9w1 z%rEco#5@k$z^xaVVUAmx4iiY3^S_RfjjYioqT0jjoat1?PBz!RUfBu0jbQtdRKboX z=I;o$^(arUZ$k)Yr+9+>WG{l1@o1!ALT5*?BS|QTKbV361ir`s`NkjozJ9$U*nVh; zE7<2w_67T$b~8(TflNW1rW#j+Es<;V6=nT_Tl0V~X9P{ru6l)EXtR+Wwecc_jIy3a z3MoGFyGrpVejH^T@rftJ)2VLPs2Wp{WKq=~$nv%f8MLCB?M{a;4m9Cn0OP7RfRd{6v>_?}ep<%*VzUUqa<`$ML1jQE<%HzU5A0j7GH@hOCI zUOQ7A);e*Dv+dR`REmLF51G85(fbl{@P(3%P!b*V%o_XkQ+2Dy2!uC8CH{w>FU=F) z6%c5IXaB6in=E_st26(@k0ZQ}+oexhS6kh$zMX)mu<>)yHkJWmOHnlGoeE{S50`C* zWzK&A8})wqcX&T%QcjkrPeto+{wlfOEYO49RcmWAcYl@1wSJP5)XKq%>$)GZHovtC6z-2s;eTWOA9CP_S{BSZTk}|rL z!LxG2knZrFmQ0yHEpW~EpdH$Vis<6nq@|{A`)S#coTmMxlT-!BAuj7Ez-J>otdkV$ z?;h62!Q`}`bCnGgpo7cC2Oql%aDqb$42=hvDnJ0Gb=%LiCndEX`nymp>EfDmO&!t{ z?g3lUI?)27U=n$Jxsj+!11NXIvz@F)FsGUnV)^DLRe_P*GK4H&+wZAB`>kI4i9?m! zel)(AHIP~=)V+0iXAa0_;RrR=n_rcZ9C(40q^2IdRu1Ou+m9An1T_vF#b5_nJ0eYn zoKX5rRhf2L9+a)_ju+BFh6N zF0o7F684nMDx$fYY5pVG!g*-^ddXRqpL16|HZ_?hJ{iocs%U!#iIu*T@?UQ=F z-Hh+21BZ>M?}q89tNXmf{?nIuT;JDXTWZhOaXjrK?T%r>`?kWAifLdkr$+1CH@4T{ zL`|y=PBd;JDR?L~Jhb*>EC2ZzL&K_%Lc6=#u|TxBOiov)?o#oj`-2bM#Oh2s26a!zdg-+Yy+&C6OM zIi~I`+m5x+S5o(d_Iw^Bg$6BU4yfXU600xUC&Iysa|o^!3O;VIDMp{!0yJBuG4pgI z5Le7rr|w6nEJqPMy@U$x^kAt>{pa-zTlIVeW!Ndwo8!#A1MtFJI9#oWFcxBS2eU zB+}j$Q^10){Jj&NnzdkB!`d20P!B0Va^GA0IjK>&vj(@y8=N&^FdkL#U75&iu!GBd zdu+phUHD}~d-qv#L!`_>X#{dH(V9@~Te3Ng@*C^EDU}>}*Ar-T-fn94L z`J%`QWreN&!SKSbXu+F`&Zr4jU+lEr4m+gv4x#n-%&IL$_99@pLB0J6e)`6f(@LUa z3(X7hubZ~8&s5W)hhNlJ!sKQxhp?qW&I+Ej3azMY^R!o=BusnUJVgj96qC-wDKs24 zG)@0MyD2W+esb}rD9!*BSx9|%)6$3i((Sy`6_l2}H?YMyesYUKVPC5;>wu+A#o9dusid$H*8`0dR-&(g+rhxy@5dbO0Wvz4s4sg^ z%*llRjHm$%eH1LwA1KrXp{cRIgv;A}ATAqdbSqft{bkl6_cQ`HWgX>Ts*S{ z&$Vm>rTR<`As`&&_U#R6DjHqAPY&tvV4u{r+RmAEb;w}e;tPl9hCe9@xLNbSk7n`HWu^BuE?H>yCz!xFmOLHs2CNs`| zNGLH}z5tp~>|T$A?cew$yhU08HFQYGQxfDXp}OAD;yT{Zvdv;&hXq+#pbk>MuziBD z2tvvIM&2cck|sVS5vAlYkCNU=l$_yGGG(=qw{*U;Q}E<~ni@G=?<50w4})tQ&Upgg zu-HifKkfnk@oS&Ojii;uwgzxwFb1pyt66B*E>_o80?ALB3VhgrzT5}>6LfJ1j|cj} zB+$3GpqH&unu}SSaj`BF=xV_AGT<8d;9gg7+*so1Ls1gAY!}=u2AuJpes4izVI9l@ zA~TIJS!-#%4FNnZlMspY1~F8+xW7JDU2{ia{UA_STD zHKQ!n&(RM#cL4~9X}<`FZ+#FepgR621-KcpkJWlD-IA4BPSi|gU1TxBB?Pop^<|z} zpe=U}eM<#~Vn6wC;&5yDj(kbSd?{;U1k@^fYCp3G;a`-h`cXbzyvz|W+o}@N;4A=`N)rlT($y@Gn+kH&~#1kZuppQSzM&`dnyS9aUuuI znEK6-Z~VdFXn<+`rlHQJe%INb9PuK)^q+s+7;y_GD_xnz2NN<|5C zSsxHsGQGRC$xbmlZp2d>OT%I-fnT=Vhz#NVvJ)t0OZ>u&$t1YJK3r9%6a4};UV#>H zW~BWhMUnRF5`w7C?C%FqzOC`3{R+&u($>gHQez`2O^ag^7^ldYJsO1KHjwW09QY*h zY4vOCbG%dTzB58$**9v1f_dY!CJmCPRZiIYT+VY6Pqe57W7|=Mm7UZnot=)<57eV@ zBjc=cLXe(jELlriq}g}rFsC?JyDS`8EF;tj?+W)oq>q(RYM8NUP$M#R<70)9f@?%6 zl)7KodQw?gliJr+jKUwaRCWsq z0TC)`0i#42QJR75uZ%URtkEW5B+YcF+TRa-tZO{1eZZBNZJMbAQ&=q=~Pa?oWxd_k~LtZT~jmIXbG#1-Qso zz1>plM)87)A~*wv5=BAiVUK<9Sn-ttv+tDJ)-ik7+9Kn%-)@u-RUP|AJGYlmL06Pt zWG^hLokZbPxgr;5n{X1NRnPgHyYG4O#1JlszS5X@KtwpLZ ziKndeU&up69n!nj>$&jEa)Vw{zLbtGy|YP~AXbvj6Pe^Z^o1-8N~!3tN0UkA1nnhU zp@+2ZM28F<6sYB~bV?f{`q0Ns><6GxmftZ(Jj;_nj4@ zh2iS-#&l2WApK(RJDv$oGwPRB^e^^RB`i1A@$bHM-2I6tp&Z|7BsYm?*ZV)8w5nfp zA*-nu#NU!uQk}kven~>1W%dJtgegyk7~=f)Fi7^AFnJdr$t*?kq=)3VBqW=+J4o7t zBz_VoN~`;%;pE8OnXB#3OLZ56fPWw2sm6y4_|3u!)X|VeD)>1b_{Rb1@-WE-zjOiM zMXp3vGxYqR2CQPEjrIm~p#hy|Ku_^OzoDRC@IYUb1iGaQ`UXHJt}V41xS-AbaCW=S zf*=sRQLu8q7|@~rV`B(1eI!Q|$%`J6Pqz8=zfD@}AQdEGB*QJHpHxXxnJl4=K~F2p z=?3OPq2I{l2*ibAZ+bBAOoDlh3v=E_2B!VkEDz#0fP%`i48(Ch#5XAwioNYY?2-gA z!-Y5uh}?scVJh!9!VM0gHyN;7X?(Q7EFbJ=3RVs>f^eLp&$jx^$4D#lX+o&QVe=dH z%D@u3bo;VnjPg~dAJ3y)_tAqlxULv{j`Uw#Q%`J^1i$H%uaXEjhxxekNlAywALVao zuqORGx=o_ZuLlH&9B!{;=&LUli=+3}8Y#~;1VZ+OCWS-XkQ4%JZ8#MZe3@ADw9-ad7kghPubh%^f@zgX6B52%xfAsyHnOAx-Y0W&S330ClcFNV=C1J9(LmZC z)m6b5XJ-05rpF_qKDZAg^3`5uzGSc3k=Nw`Hnf%9P#v)N6&2Y>uo?(6VN-tKqe7*6 z53Mq0vR8;VRNoF|;rO#2)z576slJ=*SeJajDeXJoMa~t~6l=uoEf2yy4us@YQbj5= zPNiRApIyhP9i%zDn4rD*p-0sZWqd_5Db=M3pICvmK;-n{Z;^P)>BId!=x1@o+2}^_ zAuOujUe(Q|6rdTsVX@RqIqCXe5_jC}m{V0C!cN->BHZ_UcHT5-5XiS%U4bk&SE1xR zt~<38fRH-aRUDw5ELZv548VK^xYP$Y*aLW<2XLGNFmZtlLIgsC5eh+cwEbowsoICo z%7f6?gV521@VEBz>*P(S8=Uqy*U?;d^gXB#0o&(w>GTC-Gvq;KU>t%9$JrY0+VF)Q z=sjCpZnvAOP~r(Klp*~MQO@{U=uhOTzDp;p-SGsYkU*p|oKD|B`ra}sk^7K)Z`HEW zQaa@XTC4NxN=0+N=Y4)%{{!=$*Y11jtH};41Mxmom;)7Cg41vUY0*Dwm667pOs{sm zh_=hzW&F8E>uk|(wTmv?Md&f*S{a|f8!ZNt5T-9xTX@yJ)UM5X6~chp$Gwibbnk|7 z1dY`ri)7(h){z<%uMs;k!wp_CpF&21ZoH&_pYkG7v&-p|Nq@{7RCzJ&s@*rH>h;>_5r>gFqeq?0<8(4ndTv$1|VXa8zwKb8> zx8#l=NgW6JNA^ml3f*Smby#NmSv^nD_X&IIT!UOuXNrll7}xRbCiGjG*&Jax%|g3H z>aG%H(z}Xr1Pv#Q!5dgE$ttXDS~s|`R)iRN&61g2YuG)tCsJ#42hqU<**HAqLREuA zl1c#?j`#Jc7!2-288(kVBh;lFBhU#oee{IJ&1s58MS#TDPU+vRSAC)KeY6U40 z@Qh)@?};tim!?st8Pg5`SV$rfCHmI>>?Rd#uwn$CS$&%pRnE%J;!+M)Z>)3z$gVa% zQgAj^zEjrM4@bm_Tfte7_mUPtYn2&Y=x5kz^`%yA62-|JYFxu#lOQ+IRjXf7_NXFG z(z9wsVzn8k6Nc=XF7(+qGbKn9CHVSjq1$~PFAzCrP(J#uQkoX5cpeUorEr9hH7?zn zl=aT(JR2EnT8l%4n{w0;Iy5KRiOh@bvrPtbL9q5m5#l{Qg1a|@;HAIWi5Coa{=2Lp zc%%=JR#v+csSUavFH=e=Oa#o@XTUQZjb?Iz6{M6Hya~=TLX)g2Qr6zqks}FWkhWip zExeKjU5>`=&I~Z4R?NUYsjljku6ITbi?=Z2SW>I_Ao5!{Rs5F?QpMxYYN~ik?oAc{ zd%ai1FMdm_IN2Dg~Wmq=~O@Q`s(ldc; zsgJ!0*u(J?4Kk3=bCH2~IM*tQ*WX}RSCjZz9FNs3QW;4{?)I3=Rnh{!@FoS{)qo%F zgMU|fY2$&e_zzemAemQKqX68JB%EgZQC{D%+@pjmx{}{X;X9p;8<|R_cALF^v@WM(dU8DDT07Ik$_4EwcPZM@2JI~24uBZY zK$Y4V9@^e1XxqDJCmXaf6P7aMlT8d~fE><-nr_%rbQl>spd*Stpy0ubt!ci`JGeGmucPhwt1e@XL7pM-$ z&+|B%0zkJ)40eIP@R|zpd#sw?akZMapPT6rYmC*ArF4$Ci&>2$&Xm+e^nZ>nQp+iX zpcywRLFtB|wmv~8C_zOYK|ihb30lLoYW5f41aIS9TnPW!}T0$gM)!=FnR?3dr`@LQeu71y5P zAfzH5GC#$kgiF{nX6C2xr^KlfWuvMC&+|}yMUKgZDG}P=&cj#i498h)=TiNQP}ZW{ z+jX{Pm8XAm6+*Y$zE);Ghi6Rf_gBSNEKAHmXPv5`D4*T#$RHyB(9TMOz|}BonmO*5 z2Lrgz;eI|v*xX9kEn=;wWC71Z{0S1y>Z3j@GhQ@cc6V#$n{OKZ;#= zlFTc>ui}|zlJN&)?f2Cv<2}HZoP)5x$6&PAONZ?hx&ZtR1-5yDxq z{fs7A(~Vqgz-kI&FuaDbh2yeIOfeYMp9gHC`eV&iC>e1hxpHQ8E&G#QCh|G42QYdn zjHxaRD=-AuMrUoLytaj#R_K-3dEiUc+dkBX=z}k-&MHUO)*02eu-!=I%^WYduS|fi zLc5gNEIBP{|3u{g#FJkE=y3cFuo_e|K;Zhl;SMtU^_NKTb%Uh%vNHD(FOv@SZDwfL z3*dj=p}kww3;Y(mFRoW5Ot}7^>=KF zGcfqdG1P^WGFZP}N_klbKy@tvz&l5X^eBGnqn^vjQFE#17a#2ve_Dz^OYzbh3#^RO z0iYu^4@&EYY1|yCt;Uwa@rFK_ejb>i0%jOsv^15fQtTdmopp+8R)$SiY4OGsPqZ@h zRuLo1kXO5p5 zB=Oh_6mPZ2 z#;U_-a9){N>pxy&)$BRy;=bF8aY>yqL5xs&O)IYLx3ga3ntfDqZ@tE4 z+#%mIdm`a_r<%1n6Dpt!pZo!Ih2uRT-bmrA4^T^h0?{tf{f;BKnF|HsugAGs-th&qwV$$Xn&lXW%jmbT^F<=adBZ~J@S zKF_yFlCH>+MjP$5d(_CpEbIYNW}si@IIm2ZlsOjzeUpB_HoMY?oRY0|;d08ptLc(2 z_scG(gpowzeOGtYT+yKA?o{3FMTKUFXxDqjG=$^UWAu}nMZJ58#tO3^>GYPyBlFv&6Vy}u}Z|_uKps{slI5I zgD!D9;5C`cm0aYW5S%tpAN;7Hn|j-%NCia_y?9I#mlCVX5?r|SeTQiPL(DE;ZYRi9 z#9l_KmIA9qQ$kXXb0cD6dcp;CeM9zDzbZwNhjrse0$zu%l*M?@_mp+sd(b8|#LCPB z9|m#?u2C#-8U=?Al!ABotQ-(Y$s9bU@sN%zB*exn-qq&J9ZA{$ICT`Q_t~LZ|ML@1 z><@yAQy4w5SH0_ZW9D)VZ6Q1P8YHIrp1$p z)oEviLBMDhhmPyOVBp|06;sCpoQ5Xm8orG4PK?2Jx5!PpCt`hX9Z-ASx7XO~_xuXW zb_Np=j=jEfx+l24P)UMLPjELcMhm^T&1;b*7LUC)bS>WcQJWj=_3vNQUi$z??RD={ zj=iGf6+d|9`ei>6=@MT~nPL*(8S4yATggT4C`sTG_dXElbSf7cGqu;i<} zx!E5>yBczrEdp#cz!DGhj?7`ZcqN&$V*FZNe#FE4cbnc#CBmlji4ckn6Y`U&th*RD zVol551Qaof;_lY^?y_>n**!YK82cbhSACTtDg*n* zPpkPfNhCFJsot)T+YD~oKqR>Ud_WzsBm*dFNTE&~#TA^qfKP&x&iD?eP836=S0|ca zO^2%!leA6@m0BUAWOqN{k!+-Ldgy=u9C>Lr3ZQr@X`1cm`jd+0Tf`#to&{}KOd_I*sz#_}Iu3@7^X1RFU~ z*4p=cq5Twka)ET*!a+-O*7A`d?(ett3aRbn3aQP)`b7U=aHMwZJg=?$&lB3#Mdip! zoyFnTizJu}i5Ew^gxTV!y8_SR0((uf-z3L1vXWhaQ%_Y(JHudlMAQT^c=dLW6^^%I zY+;Bih9o5d^3LPBI`FM%fj{1d?5U4I*rEG3Q~j3)jg9Myc6mjL>XKzQBy=ll;CQIy zWg7CXgS;o^-4}*dz)~`h=b4DYPBrB(73t zKz~VG%>WiE*l+22u8FdYwvVoC-QyQtLXqSp94o^%KKY=PI-sBZ50=Arvc15PPSkZa zfeZ=iwlXfKwtl|P|9rKP2vVB&mfBP~P8H73;&>ilUC$yY$GYB~D^4BAc30SzsNyOD zC00fbK%k{dFX8kb7r2ReM1W#q9&OB3C~=z`6%_q5-q!V4lbxd=2?#FuQMFI&wM!PP z?~om<5AjTupY@$*o8RP!@~6nx{GoiNOC9CU7F^m=FM88e{xjxEl%L$J%3lYsDF2&B z)HV}$OC8mo>stG$Z&iUZ^?uqV!q)db@GCu;E%tyHI>3{UGflG!atdfbs`wRo<+Jvj znNK}^i$^vm(-2;gh*7ZO6&_d_T~*eB;SdCQMM56rI}yWxz}AT2S8u3Fnu?Ct3?v!X zXp zgec=85t}v3Ne((Idp`HTDVE7a87XUP?aVT=(8kO(PSd>;-xSXXXB_L0rVNiT=f$S#p*s2h;IaicQ6z@^>_OoZNW{C7wcXjY)?b>G$X z=2}6`e@KZZgW~KmsT9OJcdApDr4pUA+WqaD&^}WPU^(WB`c}^YrS(HbDQ5eJ8xv+o zIhwBEB)WoMndXjFq09R^V-)FCM`9(^rU_ZH)kFMdQLH2sj;_}#Tij_&_sZ_QyIXs^ zclz4y_&`&puUkjHr$2X1b>KXHQEVFdhqQ7@UW}QoRT^JxP-#3NLV-^le<#RfyM)YJ zsxn9WIAOWV;yDnGMxw9a`tr&lT-!5>ltQOab7^OlGV7vYcPmFgID&$Uvi(g#? zw0+7&jmJfU6fW+2)e+;@_Zu#*RrqF}gV`KAi;Mtx=VBq2_|&v)5?D)${b3Rel|hC) z%avzivahD=JWBSfj**wrg|a`}%PbH@CrDRFWc?B~Uie?9vcANpavqd~;~C@<<8vxS z6YgDE?|#Lh@_dI%=~2qGd0M1;S~klh)pX^$oNM{B@8$R@L0sY@u4524HHf##53e!i zZxwM95Ankwa}kes5wEz+rV_pU#BuYKEc{uhdE|+^vKkxO;W2a#!Lb$xtNC zJtSwPAUVoKGR#5p4dRsMy;g9#40KS#`!qN9)Nf=)O@pa|>s*7Yo55Ar$8{<2!ts!Y zYxT=MLr-%J+adecdksSq!DZI3Tbt~f&jM*Z35GP5JRqEMErX5{=|e0;nqtPFy$3|F zxNeQISSi?z9LFn*IUd?RDQHh~(cbEytqoc+<0|-5QR21jl6pG25i9|NWu(D!vX5ms zkizly9+n?o@|j!3wKCV-!7>-<>|tmk2`3w~?nAJpmDTGc+mUN9SRL>cD}G@@T8ZO# z*3?Ko0&a5{@TIQ#_#&9nQp`0zW5sF}!*rq3h@qDf*wrJjM+$)_y9AEEN2MRZL+Pp< zAE<4f`}I=sBfXEl!t*f!Bx^IbFq#Vt$(0 zdmFwsWUrLIi&lN7NX7!#gQlJsUO4_MP?RPk`j4MiB_R5R<|>rk#NkS|>z#K4hd;%K7+A>@`DUu>d#C?RsfT%` z!ABM9#}!yd7_7G`R%i;@_uT1D7YvljZuZ0t`_~3e$Nv3%R((FlAj9a|-KTp#57cO{ zBr1p#mH}m><2}2776QU?>ERoptbG=#%1(`Fm(jZbC3b%W7YgEYFV9+$yc5=lpC0Gf zeg0jp-EXj)?(posV_0PMHUJ#EzZzL3Bn!9>HcsI}_vy-P7Blw!trl0%0x^xYi(rpq z&~&>#vKtfHb{a79fcJrLu#a0I(3iW=jRiM@39hib!a}`Lw#GomZDpV+PzClQSg&KC)3wgkY?NBdc1$5y%0_O4$pHDBxehkh+N zHXK5h^L2uQ2#6N~Si5V?a14;Ud@FFVgd4355o?I;%$IlxyYO)NT2GFHt?FP#I7feJ&6EwP1Rp_y-@ zTQx=&A)5i7S-LF>IK2F62yqBMTM6Xf418h0eyLzjM4O4G!n+Fb@@-zVy`_r?F=Z0= zWk^p*Q;{mic7B0EEg*C7jvOaT0lYEwoI?Sg;6e`tcu1cRW|%5ZE=Vh#ES21m^N(fq zwaOG->1XJN2=6*Ku(U<*+x&BM4|bc(P$P+Ww5u<&2JTZOOlg<84K@d7t%zC}8A#CRaGn?xuY_ z4RJdTVBAdPY!V#g;tOP8?$5&d9_cXkfuxV^KSC`x<7 zgnM8fVB0uZCYCP|q+|W>HhSN+fpY^zb{hux88C!*See_cYq&1JSDVSKe?{ zIn+dY(ZhUsXY{c5Xh#ooFd@?(Y_zZ5lG^w8f5T&Uqk ztp=-&4iHqxzJTyC6pQcq0U?FskNQ9^fKEO_{u3Vq+?npO0l(R$)Y(;1>S4cB*yw;# z_fX1cqZL7FCR-Whre+H?mB!BrHM{W->Q9$O^o`ph%k% z8fy`pW{dFaN;jjc@st~;AA5x$AMi7vf zg)20(>I2oxRFvsxW@XgVj0LlqXIe)rSlkdS59b zI1>zx2^)&quF=Q9|->1?yaXXfzC&gr9WNp!kGVVv#5I8(@!xpu-U@^P9b zHk=(7RJs0V9X(VQsFEYF0L_TkLfzN7{)dgd53iCB&x4@H2`o7TNk8$rG>sAsgPM(6 z-Ol$*zXCNzd>=mSD)n`)P-+tlfa>-VKaNuOPW6T|zsGW|83x*{J8sRYb|QKIXdoJ^|hp2R!{^rXwyS9WOSL-0rlhL&gF^VFUV#gLT= zS%&cAAeSXD@MLWAmxr7HSIEwX;Ba(ZO`-&|h{Wb*i#^Gqt?BaX1&dlD+~BnHcm{47 zd3s>{G~hq`gf(gnGNrW)BM--AC8}c$_akpZ!W>W<ygU?iJrxJoEMYLk&b+ktC8<44a&JI z{KD39GaEB5ry5a9cSPo7r2b2;8fQ+XoEL0|EOh2$WE&JpDQ(rMsxVfoFcv05Wtc>+ zVG_B9L;Rn^{~h>0mx- z`6^_;e;ZY1RZSu(G>DZ+#7BZ#A1P7yD!nLhU)f)}e^AITZ7r)k7dRiR;+GFbjQvYO zn4Pw8ePA33AWm@Sh9Z@-bn?{xc`#!od7pQ=jO19CCUIHWjLrl@F+ zDPkr<(OCB?1d2W;pfw%4dEtQmZV2wPW^V=L0wg8nFui^aen3f?1Ow-KWYo1Q**r zU0fYJTy5mjo>pKDQ6$A<`obO593(SJj;7bLkN&$MHfA#P55-QyaoJktkp6{-IeD+^ ztX6Y{X^+MA;8BHDL_#8n{894DB&-UhdBtCkNE6qtI5r!jdVCvRR* zWUbYa@kh+4?A1J5!Pv3j0N%OUr@2gSAPJAz-pC5;pCQC80|d(z%gC%Pai;b5=4D=2 z_@`6FelPdSwh~jAK?y7H4$&$tf_non{am^p8@(~%;_JQo7g;D!WIeA#!HeKE;t6n- zEMQD1Yw{*&<9jvV>6*OWdfP{CGkDuap%nX29G%0*?z>#=-^vx*KmSeD{>^WB+JAR~ z*J+xHT>nQ;*z}VQu?MN&(o@n|o@?04MAb^@EYBYW))!(Iy?6+1Y4%uRn~q(y40v|2 z2*Rjdd|?>>bPj?X$8EGrxfq$)MX76AExmB}mGX{1_ZYOUl?ZHQ3Cc-vplqMLW7> zXRVkfzHFlNHKsJ}qNCj+O%jcnq5QaD}NH_qv)&5kAau6nBmH zOyx2}{clqtxzGfOSUXEu)LVa(G^Rv4EDWjJBKC# zN*Qt-QlFeiWQ*l0B*OH5JggqiR8DP~F>or<_FA5tZiK#Ye3;+gtHjLKA_T_&D;-gt z%7ud1Zl-*3uvzlXi0WE^neqS4R3vqJ%C9;u`<$W+Q(zaT4DdO}sWs zHF4qs)x?YZIGWf)@6gf2Td?qN19gKghsxiL|1L!qW_cox>wfN3SaHI$j=V>vdGfA- z3eL=??KO6J688x*+9B_1(Vm#*VpnA&KXeXAS3U!_L%p){@qQz zMHv00bq56v)VoFm2&0)Gs{jnc={|&{q~Db910S)m1o#9b(p5?0LSgxobJ5eFswWhCn zY8lAbz){NyxDZz@Mj40Lzu%q=i*{+g1X>B4yFJtDu3KzW`g9)_Z&Tdiai+=B7&uKTx7QwvX!wZYR zujQW5mc&gm)Um$he-M%nB65!wis_#{=|ihJNuZ@nxc&ll2@bw86EP$_G!rqTNYvy! ztopS6p|k(kAd%@mGRyx_5#2CJru%#Nupb{l7v>o@o7 zOUV1T1w3q{ZD(Z;r4^yMx_RF}HQ~<7HR}D9R?sRR{)NC#K4t~JB$A{3RjJ^`ipQ)N zV;vrg@OTiiKYv(_Poh(!Ulyjld0;EB0h;&-mE>@h=nH4`Z%COG*z|&Ynlgd9(=fuP z?M)$?rNuLqrQ;1tBcRf$^?%%m3b8L~!QaONu+*uNnTLZHe0xhQ4{nd>%D(=-R(gWn zn$^GT?ATfJd1NmmF6dZS$zNh!gZx-y9gmgvP-XJ;2exCAEZ8YnDaN&047BPd*SI#? z_r}$@q;Y$#m9Tq-5AJIZ-1&e@-r@$+I?R&~uk=4$=6#qcA2I=PsR=iovNzNCU8DYL zX{)4Xvr@H^zCv+d3GVk4_XryG7b*YbKMh(Rv%~zy%`AC2F znE0&%eUg=YGj4=Y{V-HifLZSJ_ zHMgY-Id$j#F$CtUb)wNIg7P#vM$*j6e9^zUbpNV9V;BFc{tV-h9i8SWufqXR-mgB5 z%M4{G@5gC~UG|}pvjUZS^(Q9!FUQ=B?l#)}NSuc5r2gD=f(S%5_&6T$DJ%d+0rwB! zl0Dd;Nbrl<(GPH4v_%bna%u-dF|80MFZU|MOA737AJ}jY*p&b?^1Fa5Rq(-l4_+=2K!d*M&+k_+9uP4F0#> z#^5Cle_Iv-vQ^wKwOq97rr}?#3OmVv`-;bT8ZeE*X5X#~yYod=ST1(&82tKcJ%hg+ zNq8FjuLggJzMP)ot?c(<5VK#_aO}{$7PK4fNAc9Q2vOp5J1~vl+W)U2+7h4c5<{)n ze>tL+jr7=m{2=@1&Te;|FWa=&efvLFWLxILD@LM;w_N*|c?E3`EK!-6J)$b~!~f%M z@&AN>QvH9C#uU{5k0um=w1csy1XyLDvtlauG5zdOzGn(ryo7aexFG7p7eAOfF`rVI zED&VJxnF7lrIJ~^c~LcZv;XF?gWwqr-Uehe`0R74!5pfEQy<rsj{Tpq?05v8?mSg-2lmsyNW7{LkAKJd{)E)r}IoQn7&Js z7mUr6GFD7h?OIEJ|G4v#@jVgQJe9yF%uG2VJKOLL$jP@V-+w=&^P8}85)beSA7WaL z1g^cB@!rOo#4s>a);)@WOgJ`GFxb0C=(Ynk6HTIRJT8$FB(=$YT(rBMaHZQVyVkSWZFii!xahoGuHxql+;N zc#R4glS6@X$2b+K^NQe%e|RQSQMoeT)&Uk!aJ!~ijIKHDYNN>8naVh4fH;m|Y$*jG zBMmb6H1V|B!WuCNjD!Ju5OTutBfXtE7fkl&;F6QnltT8o!x@sDDSC*;4fCC4Ptz~D z00zdmISu{ri6Tu$@cK0QoLmr}FkN3fTCmc=yY(eZrq1SR4 zl|^Q(+n}pjnmvKFvuut|&8T3`rwCsbmNV#fNx4A#H_(bYt?sg!tAJf~))!O`%1%tL zi3KuvUnc3dixC-m?f8`H^=DD9(Q9XD3CC~s^g81fU$0HL<}31NT&aomAj!$@c|}-v zwaZF`2C{E7gwE}Ep9`?pl3gEV1FTiylQvFUT_)MK?rW~77e1+|Z}w4FAzf0Vd#Hby z=%ap+YeoItFb8$wJKBXpt6{brG?&i1G0f45S+DP7l$$RcF&SOpq{-uGO<9Pbn^<@_ zK_;e|e`oqxHsOhfA>coN|A$DKkW5(*lO1BCcvQmwrA4t>+2x4YUN+pz%_%PBYa9l6 zzj;C#dPYPF0P2Xyj&%zjL&v2sR6W6As2L1JtltyMq?yxLSH)_RTEg9#R%q2Fcb{3a zBG07CvkahwJ!JE_lsF1ctIX5V0>m#9z&u*ck50ey=v}$Cc1!tHaw$me*Jr;6TTbIb zl3Wt}5Q)OfqTX3)7h823Av8xsCc(tTRzah50U~yKiPey$v(DZ+k(vGcRY-=KsV8Ek z!W#uV`5btN5{b*#FKRWO&@EE4g41rYBAv%%=Lc_>M8*-OyXj?Ty!IGvX7<~4(pc|M z1xD2-{!chVKP7-Eb*3o;XJJWDKJ;-_LrXC)UZaM(h@FM^o*M4G*|)2Fu9cQMhZ;4U zhi6BpRcH}lM;TySM70Lk8|aOImj~>k6kxxMcYt+ufXOLc2(=^+!zw+$Aa}8-uI%_o z9cvdC=x_+3uhNE+ino42Z1Gb(J(8n6K=Ie2R}!itDoR2@YuH?t@Bg&S>GnF2!>$H|%$jYG4RiJRfAGH(@h*iv{m^ zoYC*8=g&l}=pLTuO|F+-+Nr=Qs>j?a#v|@;yWR4>M#d7JAz+X&`-ol<4evkou-Tsws6KU1A`~l)+ea zNlr~V|B@`aA&%9{2C28MG&-{~MgteTL+(}%o_@?hh+(tO_9p>c#H>}uT&`fPCj{{z zB-`jaLW}+BV4=M_u|ZlvYh_|3zbV5JJ@4EK9Q!h*tp?q#&E~KYS_M{k<1oXr zw^t;R-~?BscvsIXx@+IPM6GM&kO_f{?@|KW34xWFD?~mFy}(4Yg*rq za|Cl1WXhhR=%IO-rN5`^k0cEcv#g!=h4+=eh9!hFF#d6Y^?i}G*ZQY)r2Y5@jnc}s z_OR-O^+dr62@6^Gh}{qhq~+a6lU}jqwv!#BEUL_GA&tD1UetMW=}|?USC=*n*ATg@ z$}Q-;bnqtO<1Jj2RDCXl)SOA^Xe=NTM;kijjbL z|I4sAcBCN#M<`1S@Zb0+zwmYAPrKAVozn4^L1qOkGa&noGkn0yBgubcgX-k==ujWfo_ID z*Vr{KE3g-;SUu$_tpeIyrFC@Zw95yYj_#v^!qjsbr<{-+hE|GWp7{VK^2qb#JeJ)a z>IXPfg7=4fEA{FxL&@KdDC5%!H9C@d0Br2d_bB=ECY2KMdfr@x?EM3j4u*qx%X|%E zM`E=cEV0s@blK`TvB28U(^`&f5R84_-Qr#*UJd3gaE%P226ICn^IpYVKi&0-Q^DuT zpuLNGvfy6D%&H@4oY71X2hm^Y?#PK~R&{b0emD|~YNHy9O3Q11HPCcVUzBBTm4UiY zK^6;;IebqsQpQ3WcpSf7Vk{)FgF>bjd*dO}B#>{$k0<08T8qR|I_;F{BPz)kg?SI| z;=%Ziss7o-Ma%&`yYi*`IpH);n2T2DLUHHWQef;(>4NJV?ikrK&rx#{7nRLfa&QjB zziLES@~|pzwMf#`{YH=;j&Joucs>lcBK-bFr|#$VR}m&35sn;w+7Pv}Z)@KP_O>qd)FD`4xnfIJsmaJ9wolGO7vzMR&(|hOW zLQ01VVF23ch>~vzP`vw?P?j@G?Ys-e$54A>tW{S39U(N?uE2Svfrc=n3wjny?Ap;U>TI^Rei_}>xXz|< zbXVQ3mHlcTQE1&R@EB}{%imEjJ!9bz|L5?32ma5EW5fg#J}wLvtcujEiUeO>6&b!X zQfF0RtdKs<-FtZ5Gw=3Y`~;`VJow09%L4I>wXJmgVZIButa-$`xWgiW7I{#L*DB1> z`IK;Q;qAhSyxwd_ch!zo)%Cte-V{C1xU7o@gavbl0Le;T^1mQtB~SMX*H`Sx#_sn% zw!&I+Mt#=~{Dzcr-Buadx~s53xWm4#jcb?Ic{BBK{zY<;OJt3}Feo zS%K@HlBmvpvmv$u*Vlh zyA8?!a}}aBXnnt$@jv{~06)-2turBKeRbLrUNm3Tw6=&?$pm{*aZ~!PJ<;1%d-fxe zaN=_6CvETGbo(?iy{9y{&GC!?Uv-}Xe_U{wI`Tf4!||WJI?{}L*H13J&Z#3u2^GYZ zo@EedvCO-%HC^H*v2IPv-&d0z-!WoM*LA(IZm4m3m5>?hh9;-4GMBiP>6^@D2i;Wt zP9wRKPRk_Pj!7Z@?_ftx)i8v}tqblBiklgVcgv5V_`ESfaqvj+YpuQ-DqV^jyA)s1 z+fe))J*JAP+e%eTm7FgU7qrCV>GKM_ko{X5jdKMnWQ8zzpGIHcee7P9)XhHLdx1^3 z#l!okMr$9f~g&uRD z%K__fRpN6cs8YFVX}Buu4NeQk2uJJgQI4J#)JBp^fy-2uhx6DJoGV8;9A!EJxZsnp-I&O4I zORqGzt#VJ2HJj+C$-<8(VbX6(#S1w4X%V(nlt&LeJ>5DT`}~a9)#Z? zlsy~9VV*bdI3&EscbL}q;xFkE#sVSU*&k`Te=ieHF18bn5A_NA%8&zLw?Ym|XvSa< zN}(^C2dHBk4^-wM`?{V;nmrZmu%z-u>X?Gle-J@emVe9-U+)&rhxP7W|BpLY@)VE4 z@sU2xR}JIfY`~Q+_A083iq!yMqyyj;13;!Zj#B`md;m9i09K74w0===VN{w|S;9kD zFLNukKASb>8)kBEk1&8l`w#OX9KX>A)Y=0y8bF5a_FN%{Msw5{KjO#XeoT?tnJRvn z=~h2vG9XLb`u^JmcBv1$Qh1ZI<3zPiDA(y&8^wD4VRLAr6}a)y6lF5A@d>N=MVqm} zrM$5lvaC}Q&C^300K)4r-f;XrA7Pfq%_se*E@kMemDzvTmQyTaPSoBpO1cyO+;^|7(himINb zN$=ii3sSZ%NDPeUi;;7BSamn$H7-4>ux4pt^p84YcN=JfmodK0F6k-rOM59IGbdDgQLRQ&`k1Q_?f$%rC55Kxl&17Y=$7kx z-l;!yz{X{Ob%2TRa9&oh;&1WKSlBR~4V8MGFSirqDq{C~kYM*k9GupRj!!RhXd06n zybIaCkZ<0J7G(V`^M+S6^c=%Rh~*>d!rxc97WksM3L&45E0w$6FbUD$7n=H4Dx@G3 zNRe%ZzuZkg0uvhrX68+KBnLbWuqVmcokU|RWH*7;Ur$&Qw^&e_PijJn+Y?y{31vF( zqPiX9Qx-s2iTk8lS-odP&6EG^A7IC%>E79aQ!SWI>nB8m@l4qLA*dv{eGk(t!mb>(7pQ~knkiyBpCtFKf;e|?r~ z=p{VB&|4ukr`9fvATu{zcHRT33C_GZ2?Btf`={vR4i~mDm~j9x2Gh!1g%USOJ!bXY zTumxx7T~4sB@q*H?ftoj0B!AXhP4c4Bhkeob_~6eS2Da>Ywx~T+awuYeQ3gK889LK zPk!Kzz;!QZ49CyTa64i)j8xbn;?dPDg|6LKICP!w(e+=BUvuLD zV7OE_;5_)kqb?Ant=r5*pQtl*|E8DJSC+cqBLc;$dBRuPIzT?oyYq{Kvi%w zn7<5lc<(BJ*Gh6Wc7oFnriXB&S(7g{ZOmjLnjKLt%;CUHT*xTPolknY`hQw)>ogHoAyo9e4IJ)T?fmwa^dDR*xcjeN)#&z3s#M4tR;L zMFoT`edl@ni;xBSmTqLxZ$eZ`_WPCqG@yD@+)WjK1D_7m zHHhzlO7|L_qPMb-TlX6MsTDZxb5{@U44?sq4xGWagVg_g5LGtjO!vG_Ybb=gFk<_@ z`@a*#k0WF5qz*l;;kmL)TgL&;D4>WoDa723&Y75y$uKK$E>_LViB9@UyktLfCOFJ$ zoIpV=aJ<<3J$Hx<%9^pc4m<0-KN9S^pD=L!Nc6k?$%d&i*9kUZ9fp4!XjFjit}{$OQlY(;GsA8RHB9)CxVf{~KC`VjR(!=? zGIOIP+3bHI?H`d{AJ?r01GTc+U;tOl!IivQru3x+!>f$J7R4zHD8eBY+n?PfoY0F&Ocu28J@>$7Z1!Oq8!R{Li z$ZvtQ%g*HzA!W)b1C965C9GT4pkSuXR3+=LjEXR8OOXyUO_AWjkBefDl#7frctjji z?q_lzTuAAeI*xHBii6XyqmBvCQ0YnbV^1-m8jEh?Q-8Xl%G^m{2t$p%Vcx!BLPBqt zw{OT>=qs&XZpkRAE@LgJfmE)pF%kcIdu+9eUxwoI&<~qwUOFTD21xCj zi!h9(pUb`5@cq%>sNOET05LJE4+wWWcmGU-qmhrJiNR4|aFqEt?p7R+c{sKM*Tr$a zi{qQ~!I8!{(Z#|>dRc?$Vi`Nua~W)Gww83>KfZzCxK1)ZGL`0)pr!MqcJefNLaFhs z3(^;hSpK*C!tXWa8+Qgjz()JzDJW0X6_zcQZU~xy>lbyxC#~4}r5l1SsC|TFSAUT5 zk>U0`tQ<&-)Y*+u6e0PU?Hl3}cSU47l8Ap5(n9=g>GC4}L9UrHmk9fubK;5S?aO@8 zOyU})gzPn45Y0E@mnt(B(lIHtJ}QbWmgxX~q z7>zFiEbAsblm#g$(_NJP1*PP4DvE`emUB8B*GgY2Hi}@Wc_L?#lr*;2%yB0S<-4NT z079A02azn9T9oeoH>;GMHin-5K0P@~&l-=OH!oGTaR;}UtB}3=JgAWt*U{AYdeyah zMGDyHl`)>?>m3QrBZ^(3gX*Zwk36w!8>#G($BtC4FjUOHO{KC~^kJm(1h`1v>QQkT z_pXip+|N-&Sm+o69U#tZC+m?!6av2>Ho?6_Bq6b(1w8J4ZgpvPaN#{-iPjox>5}iw zWCoTL-uhiZo&C`->vpdk)BK2VIa5XTm62r7DX{tb=$EylOY1Izs5JS2ojL`l+Ky}s zhp&&+LA*Mn;ZPBYMRJ_^^fG740AOm({)|E7)cxM)6FNATu@ZFVB&8>)*AQPc2dwXl z9IboR%CXX||i4YEuFrSE^hz_+Zp7q*9)4z2lLaw%?b95kO3~2>pflGL5m*vP1C}_Vt zmAJYUv_7`q;HvY-Wmm}4w>#<>Mi#(-S|T!dnMTWz!Ka)QURYYhGQ(IC46zfo41bO zjo!Y+T*0+k%qzJX@PRf|Fvd(14<)3uN(K|f_5~)AR|uZ34W8Bp&vwzL!E+}JvRTf< z(<24X@5K(D3k6S2vQEVM(}@(C@t;Y+QD|k9si*@ZZ%}SNxWxf)dLsJ;jP^vH;R=oP z7Zo`@k*mOLe|wJbA@B(+Lb*sgl_Ou|2sy&~pIGX6KNdKGdwZ|xZp#q*ejKdSL!qNG z(-x);w@cd4{4t&}W@K6Hvr5zo@8^m|ecC zK#G|4-5OleAodDu!TL`uFp!T`MGdKxDf52UV*uoB!tpo+XBkyfF<^nZ!$|KdjIZZEO83hOmZg#aq{2tcB@h3nF(croe z(5_g1zQi$tus|pIQ{aPoe2CAP0YTPmZS^Vxr{;)vY=Fscqo=+C%(5mj7PyhGWbhuc z?^0hmaj+G*NX?){?ieNK`3a&Pj{h>a*BE@%-w9knXItnr3$h$G7oEjI{lHJcz7z|` zpNF2rb|;!;_L7E34t2N_+x)F8>Xb}SMgubT)U93^Mr!7l$Q5CY7V0W9pAwlF8>#?t zIG)dH3q$ZRkm(9*MNdaqO`%b|#=$`kt(o{9d&Os-7m$-NFQY<$tLA3KmFwfW6xiW- zfrsm5;Jdg6y11S@Q<6FPVVRFVR*h3FKUWC`HeRRT=L>k@!BopDrMykg zjDXVa1z=Mx@hRdm4`Juu^&w?Fj}<7bI4Jbbw0FuxPJdq7jmQY@M83p6Z&Xbm;RY_L z=g@D2{AQ6_e+4Y?0OJ_U188mt&5l_--bt-rlH2VoO}ENlqCtw)leLQ>Nx;(wXC)n> zAT8K-v{PM&-m^4fX))55sr0nIJZ#cZNbuuInLa~lL-ASo-1d~fB& zcRYV#8i_&EAl_>Uk&3lRLaJ6Utf`)3fjmC2i)E=56idT23u|O1#&4K9!#~}18L=^M z1A{fiIZm8!m@OmQiT=0O@U0Q!^*vOK8wkZ1F*fA^Vq8IB#1Z45GmQ5;yuzi%F`sY< zKX*hA;^z~WpN%BMhy}**maPb*&_|lRcE6~6kMJd)`X%M72LlMFcAf6gy5wS|8eQLL zu0m9i0w{;xOr_U;rK53p-vAluN%UwGX_`o7!AuVfhV04fh(Hq8*P#13cvoPg85o<( zyeeG}DUck(TL66_2;FM-bD@$EvcrN*_WEzI$D7mE#YT=)2KxF8j0INWTx_Ib=L9(= zY*bq5MMc&{bdZA_R`RYXNwEG2a8N!<)u)|z%bTjXQgCY3EHzi5ge-~w-=S1ByTj|} z`>F8f&ZyEpkt=q)1fc1t!~*MiX;0g5P)+IQ7doOl9x-5#74pL;#1uq#;rL3N2KUJ8Sj6Mn z(lZx1w)ER+YDIW&Js{B~_xB4OaxZkqtpa{g z^8EJaxJ_Ut~Zl;UwqxPnz zF_V^W4}x8%{$8<5W&WIjdXx|K6ouNQo0RvQr5oJQXT)jyR2Xvr9yh zJe(ks=6-?vk}MLpk~yN(9QpF{Z*@2kmL>aQV1rqZ35D!6W4ShUyD^Jfto>pioSOhv z}~Sr%J|NuQa~ANImQSl z8XCToA7k_5V3dKhM??P<8tS+-^l@mo3L2OX{u&aP5B`uJ;vX;5O4%ioKxpQJMb2%9 ztDKK8=*Icz3INHWj2=3>o6qvSTtigI{^}IN@-HAD79k2{5?PMen`RE6W$mcUER3?B zhfl`|O}|P}KESXG)u#C5?&&hE@Whp3IxhP8V zD>Em*l*t0d4NPgs;t3(P-d!oJY6IE|?CWSl#3cDKVqOj+wo`gU?CI+BJ(p_*m29hg z%Yt-cFV^iZp(e1@j4_|QUU_clqie0`j`Gk=O+nY*MK{@? zOY8%-v>1o9mnr>M^3qSx{p~wA%1^Q;rlfZ`jW>30rU1}`%)0&amh`w@W}&0Q`dtPe z?tgKZ?!ZP-YT#2I4U0(`LmZ1H%35#qMh}hi#pq^9V5$PRKpR2h9FD)^6PV)> zcnSn2C5L9-XTrprf-!^JBlPxlzsPT1k&7q-4fM>mvG6$n&TyI z*a46K?z#Nm`pb?!^iXZx!~Vv2bi!5|b1*6Az=us5Am&2PdXRVaVxskqRz|%k49S*C zM=O6qEp(%m)?6jdre>2109~q9G!amhX!dQ4MDxnXx@gTKkl>Yy|=`DTZ0>ViP}Z-uNxE{u&PE%Uf6*#%sV0Ia~tH&X+=Arlrr zf0i**VP%g+bZ29M`n)f!?8fr;Zj1Qc#c!28dz?-he?)Xj`$!Czn2%?$BvZ&C`}8tx zuO!h@zmG*wJ^V4#ksiYhVAC-CT^GFy;q4}0qfA#aLS#JIrZrBCT{V>^)x^QMtVqpT zz<5<7WZGR=Pk-ra)_Y^7>p}h6C)!k=Vjy+~qGPTLK+J(6&}59Z9Q1BeIm98welc6- z?WLpC5oV>zkCTx?Mso@Syxm8_*QG9BrvGt{6xQKIrnw3w+EVB+{!oG>WcuhbPHWwM z8940q@~7nR)b7EofK%W0FA!Wwu~)Hfa=95Isc&zcM}2!<6*;s_>Kh> z)c@PU*-xd*JS1LeUsEA1m<}60r5a;CBTOe@q2Gn=4~5})9bA%8_8VY!m3<@EJSG44 ziE3v=*#GPLBi)aak;$e-yXrRS5Cs}=EhC%29`wbJ?3IfJcFIkZ9XDp&Oa=Crb|)q#Fo zD2R_3iUHzuGmN!fv^~(;?WQLy1 zB%>eEkpU`+>!*=0L3{49xZ@+b&EfcbpQN!6LTN}U&BKhl2{AWF<6j(!ktlr`**(dJ zvQ4ybH>I>4@2a<&bef{n%J!a8rU1mXu_L+0F#j^$sup18>yTX(((YMhkJ|Q8B5O_7 z#}N+t~pxt5&=P~)Za>YCrTraGQc!d>SwK_ zWoO~ig%sD*v}AKQ3zo2tY`yvm138i_IU7{R-t3|ki+ba7(Zxp)k!a3q#)qqbO?t`Z zR+suTi82MYK(Nu=cj~k{d4`qwCJu%E>*}ZeAv0&K($9eM_g2x(r@lrXgtvrrD%hT8 z1vZIsfrLqBa1Hnz&GbQM1sATT%kFxPl6 zj|V2u=_HG||E{u~$%!Z(=Di{7JfSNb_+V?63~wugG*%IR@3s zHY-2h`OQ2+Z)x_!#C{zcyNK?zV`ICZE*x*>wY7D*Ph?Oa+GT;5tk~F7u9==}cClb% zcgZ^ue`Vm4=N%iH*iwB`*S-`^^+_GcAIruWP$Xo6SA9}UF6j1`iP=cE|9A~KpU6ka zTdhDFjS*)4`Iy)3*PcH5?VwmivJ!is%P9Ss8>3?TUZ;Nn-SKdoJv}ayonXqS`mD21 zwRHLmbiH6m@QEff_4Wk6NSe_f9;XjQj3#%TshT_>Jr%|zDAdtp%i}#wel0@yud@bj zYxA%1mzN(Y+GX{1h&HvSFP`Ex9;y~UB-rn0aq4rP7LSES;<%7+*gPLf6a7?+--}6z z7C(0l;YuAMiKkdA@AyLiO#!B!;ti})#{e=Q*D-*Ew!+cB@D%@Z5HN^NLO4Dc8vZ>9 z_#FRj1&Vdx_J0}#Jj#L|(P)*S?sNLs;rP|Qdai^)!|L!(Xp}*~1%hI&-EY2eIx+~D zxls&nm|yB7(Y_1J$zc&Ib1ybTe~42@PAswlN$v_OBXxr<%LP>4F{J|| zl2c(xex&o#v3snH_kjn-Nm?;qbn)P6$!F0>;A%&DT}Qo=p+N&9icc68(L*`%7@Y_t zQayMXJHB6Ki|mqJ^$P1a+cffZ8Zz3Yhgdt~o3hc>f%9A=|KK-m%VcBBfCKItMSsCB( z(`k!%kz)EorL7CBi|Q6u7S^3pkXNg;QKV+65y+;oyORwfoi_z>1)X5=yKc;u<5=?Ah9j#|aDYTQ;%0?^{)-A*uHe|s*+`N*;WwP_{bm$Dn?|0{@Sc_n2Ktacw zE86!pry05uqoj^#pVaD`W7hCStO%2a7l-6AGb^&&w;R2c5{T%Q6Dz43v3jHvY7b6( zQw7Ak%FNqWW~LQJs~X$SKPn43gE3tdMI0Mz!FGNj+R9g@B0D3rwvZBzFYttRJp{Qm zp}j+#JvkHmXH)V532n%sTumfpK%2{T=v*;^Q84^|hbF@Dr+j|@@c8|!y>bb^pO~vq zVuJL0ln;>_Y>g835P4N;y@*wu&SCdk-dSb+`(QAdO5b>Yo0)mNEX>mhnI1<`01(K}6)sj&5% zIr}|tLgV1H?PA(YOqC;2JAHCpTu?KKii|P~JnT!~Q%@?`f4)fBeB(-G^GqN6(_je4 zw|m$dq+tK?REN#;eC*eQ)9Tw;`r)ITxjSdhPN*P}nbdaCv2Yi$uNo>9%q%~*GDL0= z^IM@3(?*C98JK3}0l}utN*PxuV>Y4^1Cf#O=MZSWbu=xi@h4)XSglwpXEZrb$m4a-SAd4J?EIFYFD`(onTHn)(i)*!`okyi|mORq>Jz3x3 zRJUK>4y#C=2At0s_PSHGg%*`f{fQ38@i-y5pzUOEZjV@_`j@4+U+&hH^bLmlkawm} zUQEgR)g!NC3VA=ab@dNBiQQBYcJC|x3w7h|^qPrzC@-pN~z_gIc^ z5w~!y7O^@AOKi~YD)ntv_Ab!UeonDF_Cm^LC5gmuuB(z>Zf9UQz`J;IdjL?0twJt! zH6N{iE?T!%*_S-vZRoy7#eRWb{1by8&1Fk*X-V_g!nLR|7r~46Ly2i<&Ga1Ey7m&V z7u2I7mYKhk$}A5`xEEg_=EyCL8~%WxWuf(9AqnIdq(|g4?6p*OxsvY3e(M8m0@Pkn zEaM5Euu7wQ2)prQ6%i)YlPe_r(-5_VD-g6ZB%9R;;mM+v_4aGm$=r2HjJkuXdv8-t zcu4BrG*|y*+9=HqX-*Tu@l|lrX{S**Y0t@Zl-FW5OXJH%RtIk9S3I5_V23{TEPcJD z4{LoJZb(~B7{$@g_0SxSmwU>;tBtSxTe*(4I0tYd7<*-gF@dACaoK2ZA1n5?U%Hdb zwGTl*AZmr6kij2=9SkCVE2D=%QeWDZD|Fu#3V5Kt^bU~Q*9!2pHHl}STvRhmU4|^g z*aa&>EY*_LA2JUwYLq9%&JG>B9Roc%C^B7)^CeB}Cq`CG6o&-stQA7L&_6_@+EN#` zV*OCrPpGWSd`qMPJnF*3Dvop2PwM(^(jmdc1IYfD_z}KftwZTGal3@f#h~zrnpcy9O+oE(S=MB zs|G4+B&XCy`}1pDldeKBL?tolac-!{#NXjH-S4Ek^SfqNsnbr@+Z9^2`mFICHq?n8Eu2961muED;6)3jU%N-`;Ha0nNA@G|4>6;lw_X&X>U5My?qpdC zBLGA80B5d$fEm5SsAS>zQ9i$Q4VUmcv=uGckp+bc8i~ABN+=_P-GS7_GCD6HcKOO46t5VoPncFO}It|OcQTHc?HVMg+4PgU?F+3 z1gVKs?quDw> z03sO!Ez+0E+U@^G+MB>xIsFg(Q_ZbuneM1YgVM#IB*heE8s^qicQmCEqA5h7mdIMRJmZptl+^s+pU-)gJBjb_|NqZR?s=Z)oX>tf z=bX>k%qU+)W#kq8aaf^MDOXDuySGYb^i?0X=3Hz?)fQL}ks$ckU_|gM+liC73hKxJ z1@n1!Q2)?Sq8Vl(Ks6F0AguA3VT5Py$}Zy>%PHcBJ!?psJ;2PA$!XVT!7vtkj$p0G z90Mu``B8Ak##Vz1+*i7|z%so8{~2#*ez@X%LQKTl z%@rH0DlNUDI#9|>^)kD^)<#A)-M;c)iQ5gA?EHn}cl!Fe*pLRnvyN0EA$L3}c$`Kc z5*}y4sw1>l0_sT<*Exct`}H40V{T`L!7jWn1A{rOq`g#~eEL9Ad-A0Xh?tr)aEW%a zr29CsKlui~^Ty`X4zS+C36x$8OeU?0TAOOjV-egi@Gur)$+_v=kzQyuk-fr(Q>RBS z(VhNj1y=7=t0fQ|-WAq{+mzj6h&s^_)y)ufl224iC2EyNRMRA)zHj3Yb*d6ITPHi+ zBc8CISPw%M14Kd~M4cFW<=gXEYUT7X?xDsZinoWTC|(ofMike8g(*G{>nmVVRbvF# zyI3n5p!5x9serBqw6A3|ToFSX5sphoxSG(8jBE@oo@PwK#FwQ_sgyHXP?0P(I0?Eq z&kpnsJ?W@CaXH>jKlke;pLhA6w?l5CpL9QE&Zdi&mgUL`T-jiHCN#{1l?r4$+QRM- z^ZN2%QcGD0=mVN>c*E!)IYLeSjlr&yx``JU{hi3lG5X5-UQ;x|T5f7BOcGf3nG_38 zYERw%j+{iyk{suPN;WC=$ouph)7e?iuU6B+Tf9Dd^V_wvb0)+iPqJUFlYa!G!Y+I|kS!*ACg>-gw>pGpCMKF%&uF)xmY~U#Z)7hCT zfV8tS-lReauKlD2OJ9bX-d$@tJr&@6ARCRw5I!q0bEF~rrWZYNKZnEMfHxO9?Vl&^ zt^!$IaSM|QVV7qQQhr0QO`m3PJ!5V(HRy4zef>rBX;^9E9eYJPftu@HqAKe>oVqWA z`AzT#;D!cpwfq=CE>=Mnd4hcOFkg^mq*a6)Ql#!P^hjt zA!CBmN<>@}6#L7N{2_i|C@Xb-DR|aY5mur#{rCW2tkmW z1}XYdi3duf3m)E(SUMj@qg{=$)prp!n4ivbCmjvkuE4w}Th zgl{>BSD4kw_OV%3?^gIUtYCoJz0|7qbfQD_q-yOaOH|GhC|{4|gyWWvC+6Y#rnPHu z3r#9ib0Vb$l@lsG9=~P(^{p2w6@UX^lvdIQ*u~g}$j1XX$OR~8{SaM->4uRWQ?p82 zNVjdHjG22(4RQa1q;Scj?QNIKWe?}-(;z2AddecL{(^PGDT4JuAM33K16Y4M)MevS zlM2;bCRopDWxA*0@x{dR<#DtOndgWYc@wdV3nc=&6}K5Im%gEuz8<*X`?;&ulHEm4 z^)PPW4nd%9;5t&${txg+?vF#9{oFp)KgVSUXMBvh@CA1Nd|@LPdyi*8pNPSP<8S(` z)Ecs3rGrUfU_p}-7f>QJaLziYFtC9JLCp++)s3&d+y(k=nl<7N5yQniln;bc>T*$= z>2mQTDL^mdJ8Xp{mb*x}s9~M61aa3Pk{**XO0JOE5Nl(RRoK4B3Wp+AVOH&VA>rw1 z5oJXn!ysDIUrD{e#b*VshL3Q3F+hyoaJ$5{lKW92Bh$3NCk;RV>D_&hB=Gu7Eoe>fZqvw}%43afR0y%AofH z0|1RoaX{5vNqN(a7;=^z2g9 z$~xCSpSPIVIbvA^0f=SEDKpj<>9>i~9DKSE`h!pCZjaDiEtOLU{m`VubOxWL`uLh3 zOsD1Ez1po8G8q{c#JbmsP>0b(M^lCg^74z(xLmLOQef6J0rx9mCtcSGNb2Q4v z=?cz00#b|T!iIg_d9vZ5NLD7jY!*Ak`}wETu}aizAJrHUZ;vRC z(}Y#S<0uRz91k>cO@2fRpU3W`70@kvs9{9-aYd2rB?jvRgSD}b^=OrB3lHnILwu~uNGsMu9IO%=Bt&l4D>l&{@C~pA zqz$n8@EU1x*-zsE>_U|xzEzL8LK&yfz!)K1eWgy&4Uu@!79d!GZm?BU(RRoQhy-hP zwoQ*TXfeq!DP*;bc`oeX9EhjK2B}Uj@}}J?j0C3}pT{s^V7AUWwisZou%}H`fh(?Y ziYox4;5rp|n1t((4A=ZOkpkCw2G{regyTp1xSBh-XgN37M=Ag2Y6`jGfExO)-3F_W zi*X{RL!bVG>CkKCrm~CK^@-)m;t?~vp}(t#OK?@eMbhUupug*`A7TVXu*flMFMfnn z^JSL#dokvnr;1v%KPnd1k%d2u(4pP2to=ryxTRqpyW4`w?u`ui5$$e%9HAcj2STOl z+>5SdKbFu>j`3nL%rk78s`Cu8rn4ij7@hlgZZOcn^9Ilp_w;x}OY1y?J6|#W2??^S zW!k~M-++=K>u6Q&cpud99;oX9RTH6qf!Hdcg;Pb!Eq*xyRr=BN0jOy^i2h-9gBUAI z)G+*-bg6kUiWjR%GCv_k$-gmDIU`I|VQ<)_1V(rpl3r}M12G>*{R}LIipg+!A1Ny^=Z?u zTAvbEV2Wix!dh=ei++3u5fZhQex@EpY3x!>k~(gWe^do0c4cW__AsB0QxTX^OBb-$ z$ZiQHlMfixd2yYC#B=JI1q3e=6hQ62$`9~R`(`*~1`<>T{>G-h+82}NZOGoc)AS1# z!PWoYcuwtPWwRJIN0=C7TSL^vK2g1usC&)!~FrWj?y zUOoQJ>pzr=E+{zdcJ_Qt?U zuc{6lLPBluF7~g}ca?)X22a)C#R-P6b9}=70(m%og-2K@iLhp*s{_k=8N*c)2MdYL z9Z$rJX%1+7yn(LcG!7W=e@DgS7*tR=!dPKMjMXn|>{|T-lL{fd`S~inXZfMexMjyd zU4oA2C2)z6jwQ5KC%#9Yp*I6=9sINCNGU)6`exu~DG~+paj<%P`{_#BZ9ZuoJkpN$ zNDDZm)fDU46pxWp=>?jEcO(D#VMqRVkWl%zTIb3Cy~!&78+s~hADrnK#$OCA^W|befd*EjqDHbPkwx>IGT6w8I&%Ronf1q(KA~M%6O*) zK0<^aNr+1r1c8ovYCs!~hsj|`O6R#o4l|AB#q)(n77RZ_LP2!lED1ocVE8`y#vj1G zndB@Oz8Q02AK;-ZZJ97D|t!P(Fej|1lf+ z=*^-;oI05eQETm{eu4H{pub;WxD=2rLplQf8iK-QJ+o673=3@xzh=BmqHGEa7G}Ic ziit8AjWbEt*~)BzAEZ~-vBWGC*7Hu(R}S_O{3+siSb$Oc2Wf8v78~h+EWtI2bRNn- zD-8A_mRd=ihG0^Wwa#=KWb#USZz5MWT*4#!h0H2m=?mS`QDk(nkMBJ28AZI@kiJm2 zoJ!r%a;D}xn|!)1IYS=@_{D~M#h#%U0O$qhlA?mOVi1Ry9+2lXEjRsyS*@*=9W7^~ zGPB*F_@AaI|IOZx@A-Fo%itY^rPw*lP&oFsb)si)6QL>`pF|GB=KO$XZ-rvLVsCk_ z$yTm$?d=5l#vd{p2QXcr-h%3pWN+s%9N<84AlI?CcTe@~t>!_ox2P|+HjNHqZ}nPk&V@ zioLb&Ao%9`__iT_qlkjkBzxP?Uh>7rr}nl$A8+xCCA?yJ6f^d=wt?E);?vaLzQF99 zu1u4JXKx#}|G%|2XaD7!J8+6N>Q#Cg@OrB%u|V`+^6rk1woN17#0;Lj(JGr6m0PxBrSdwLT9 zj{kZWl)BYj_f1HIPxnncCg9{)!wPwmi!Yi%g1Orht;`BM80YCcew{H2nTu(0k_dmf z54KcPGsy}ZqG2}3U>C2--02$1;7~#gAv^e`r|Jie6DZPjImFf0cU64>DbKaMth%it zpQ`$9S}V?|_KT$$1=Cs$F#vEaTkEN+*K|`=@9gfWx?PQ@>Ti*m5LS1x!0T3eLymv* zFV&}^iv!xT=XnQpDfR~hKk(*isE0+tjwoM&y%_CS!+o^2P}q&<_R2c|uHmsFkhtQX!4|1$g<-@jCAL$S+8 zaDpI^o%=<4p@fVdaG^uxDc=e_Pb|svCo+R9?N_1oju;zFSjhf)vkGhwV(OVK;t16D zn0+Y4wXZ$3ho>Yir5n&lSAr!Tmr)*_Hyun4iu6b;xeEZxj}ZVV07!(KX0E;3>~D~( zw%9>Y?DVNGL77Iat3V-CS%IOXXyyjws4ZT8s%MLLZu4w$7@GVa&D??ZsP~2F(63iF zOoP4lvh`d2&Plw0CeFXukAXPnU){@=L+aA~ho{thLH=mo6=Kg~fx}%R z{&JaXff4z}ACkEPU}}L~Y}fB>uv5FJor!}uYOdJX2w(I=5q)BuED1Re_fj&rfc-4Q z$h4Ot`)%19{+(HZAdSG+xgz-g_{9%1!la6y39X5)IqqnXYqIHevX7fmzs(b*7yG5Y z_e%Aklu<_YK2^rnlT{g?5FT*afQH|B%2+E*F>^TD^$vA?!U6uk*?;|Jjw-=S-~P$A zDDnTiQ$4I9+V1-Q>peXLp^B3(Jw4<=n;tiX{}+{t9zJkYd0v&Phf4Wo^w1b!#{XkV zZg+I@NnU-QcaMl9=8NQAQFRNkSGgza7oe#Bw@K&*+l`Xg&;DA8jybdTRFjET+f_79 z5h~mQJ}}gO6>X{Z#E}fM0<9T)dC@6}M-^+=lLeMC?k5~&7D5q?8fs6vOGl^@jF@w@ zX!=2)z+9s#G~MklG+kn49s*(lQO7aOK&K=)<3i$7GxY22 zeAQiZU$HVU9IueabiTgWROv5##!}|*RZCgbSuN!``rVGDeEzj(DGL$1 zzZsU``u}0Swb)BB>v2v-ize*Jw1%;l@w`)enMcImv6sHrdG@jeip=c{hSml=*GL~b zn)jSopV-SIuE`zuwrekQinDy-fJl*vlX!TXdq>i}v=eyi4SibbrDFh@)n! z6c&5Yp~{)`4wmo+>(G(PXQek?`!}v5AvqG?))1 zt!a{Rie>a9oTexbR15?9QZo&EUH=qzH=re>kY+qPD!AaQy|synR_5=x6*y>eg<~5N zjcuH+X1~_g?h{i>XmGwpwyt8HqfF~`|{+qurYbT?5w{}J`?urk`1jg}BP2e8l@{S4gzQ!|w z&mk)u|Ha6JN|M5J&3rY1#bQum0t;L#I>zj%VKA)5W=!CtF^&n8V#cn^DAbD#)dVhQ z`BymphcCN%$fTwVDAWXkB-CZ-nGLchUmAXO8Pf1$6cUz}SoWRr*!FgOu$5kq2$why ziDBbBZu->B9}Sat;%Q{6Nz4E_&d9DbFN)>GXyZjBuw?>^Ne)GPiR1~{C(Q92%9%%s za+@=a=a|(tBt#};_3a+RI`o*y}Cj!5eyUzS66gQAaXL~@LS8qpCX#8$+SSZzn* zg*mT^FdsH@DC@elfe{+T3MSMs0G4)4P-6HTvOBP9j(`EIPK3JN^t2H448l7I`gi#u z7Zv?WWqiFCh=2cg$D^Q~XGtrpeV>a85?!s#d+|sZ%3_u+YRjVmH_kV|LzwI4Gj}6k z6IaVTfy{W^Z$xylA)=b`EJTd;i8xn@=;29!KoSuhTp}hwgsz!w)LJa7i;t`Y#YH5X zRc10n`_A9gvW6R6so-)f>tw(n7hNY~|F*?-SH;@9F&(GB&(G05F`w|$32IcV7qvaR z`iOUEEXCeK?BB7g$47g1btVL{8QtS-BxGr4BbpZ!6BWB^;F{*l*Ic{W!PYlbCFcnN zOzmp(7rtFBKGw6VHJqTwN$9>b{sN`0T~_BLgI4USWEx$T{#M(T_;o8#PUOkyXRdV` zBr%yK+Z4YH6kejq3CCGC?%K(VMzYBG`Cn1(5G!*wP7qUA$XW@Bj66xHO)S6*$1n9u zjir>=$ef)iFh}@bFZ5t81|}LrZ>N*OP+vM;ZKR0skfXPWpL!nQ44i}85A{#yO74;V zt$oq+fI4+h{nhD@9?Cn_Uk@e|9Q|#(+|%DQ$RYX!&BnkU*@boAGogo0SL z#MNIH`NkhInFuh|UlxYvHZDIN`>amr74Dkz)X;`m0Yu{ep}uwLkimB&0vO zh6Ig^G$7(IX3AsRQ}D7@`lCdoq}^D+I~>PPT-D~bC)&WsG>*0)>F=?9}m-CqEpgCiDp>M6)E&3@tz49a>7UcS&s13@slS<(W`Vs0+vQ*=}u2s2Cb+)=?mu z*GvpjTtJFzh0&K?6Z)yWXF{g}OigIhC$0%?u%A52Goe*6F`>R-aGd%cP}UTKLJti( zlY|}`Bqs-|2^}eky_k^nSiCcVg5{NPVdecy+h?U;N(f0zNX(1ws&*^>xpNZ1r3&(5 znWR^P!tCzB>N!wy!2r?zQvN8{50A7#0y48Vb)3eJE&NM!;`y%ie&c!w&iQ*U{HiY z?WK?u2-)@u&%1wtcZJNphRp9p;!>1-ztA6!_hc6@&(c5q$rtqk(hBJ352;JcFiu|~ zG%nAnbW>h9=3R$=%tyMiF7cGEODSDkxo@Jq8jRUaWN7W_(|WR@^(jWhh&0!yRcK>g z)T6a~60N`e=+N5xf72>d%i<8d3MG{Dll6O#PIplsOU7~j;W^X=8J~5JROH=qEXQ1N z6*=q1Sh%-QLSt;0N{jTkvU^GW+Er}cN{d8)s8_m;;M|E~-d(IOB>xfI$H;XscIO5u zwaX*1&fNEwmfLm@Ryz563AJ{S)!5x*o!Ws^;yz;-4Mtall5d>OXYHS!(?Lgskl*k4 z2Bx7ojf^8eC0Ne!$#T;i6Ix%(+31_(`=0dRE!WfiMy{+LD!-pxRv-%2=qpiz%iL86K-X=&ZI(E+M}i$R-xT6``P!60Yz6hM`^Ms%Gy&%hqkt`d<88Ta5a`X zMH0DNgERD&T_ieI*6y&S^^NQ@h%Ktf`}G*S)W|;ytu}wiSyb0@VLPi(YBQ}bTH(FC zWm0V}*2&$rhKz=O&+`&!WY!-%#^4h__y+$RX*Kx$??Eq1U0(%fVJ!6zBXZ;#Wm=&8 zKHi>H(ds*XsGh5fVk0vOVppdX#QHbmUl!>$MX_-lPtw0jZRW3^LszVKcUeCqcSl4o zQo?4?QeoZK$T$d;)WlUF569(rJww=#B*G4L3A@Y?mN+W9uSrSs=)MLejjMaNr%Xzs z5ecEFwgIPWl^!Z##zR;-+ugpjlPq21yjA^0nmRu(CyAGlot*;7T7kkUouQL9#G!0~ zQ7fw6RpY99hDn8LGFa|nrB5LXMaOH=Hf~YrfmrEPl2zA7pKVg1n$I`_)At=(A6YM9 z1)9^a`uF2T_i#1tUW{{U=iKzA^uwkc9~oJN!Tx-dv{cQiq&B8Ud!&qKhd~x>c!y0w zip(=&heJ)bGQg9=&HF0nuRJ*)4N>Z|W&C412C#2~Yp~8jihPdDku?QY;3^2ResHMl zBZs>#Q$mOEK6smqAam23Lu|0(ag~EaXDhn(6p|s{>bgps=x%(Ao{nN7} z_4ypdQ`FsUpgUXWDV1RBCvoSy)R-mVQKz9PjwU?&o+bkYX;8dsXcO^GW)R1q)C$f|Jfb6E2KG$}?&viVq3dd_a zHh=!kXY*dt2qR=~d)LSgsJa0a(C*43U~dH)u#YmZ|LeoPL}AO_h(<5ZCBe>hVc!aD z7no_EGejV)?q*6WJk+#$tSYQKRgSQc(j!m$l zIcX!?ineh!!y(SST7#aJ`iC>0`7nth>zppK$v3^w>XlX7KNWx7MHY0V4+MyPavM1l zOs`XyO()CFmrkP$@Erq5Hyq#|mwu~q!jW8NQlXl0fVX>3B~q_{U^0Bc2!1Ub+0VZ7&ZCs%AEJZoS`4K>i;n=gGE!E;MH?uifrvl-B=jw zl_jr=Vy!mT_Rg9WtmWb9Y-_Ad88_4t zXjgZSem|}1`>fjaZrRj=*zh(5R{u7IRuz4M$T6#?GpkXHq>5IiH2qi{K}N7$N{>TI z*s@fZ#dGPe;pyz;u1&c?ZVYZog$c*+^Tn~xJov^lsw- z3Eq4IuoADpuVYlQYM1(1ES4)1TlwJ;V!8KWR^fPWA5p1?sJDmcr>~8rVus6(q6$}J zHl-NV<-~W?x)S#TsXMR8u+U`qLHfj|s=o?E25z7*cG?KIZdJe2foy-hMj_R35>8SZ z#=JYV*01ZaKrf!l*Gz3(wI>VRTUE<3Zg=;Y>mQCJPk~kC-!vEtH04#;s;t_b72P_G zn@NMWt;K|L150UPgB51RUuO^UYS@CtDScMYYK?KM5lF0De0VDjLfzr=zquw$ieR6Td(xZUSkj~F-@g?TS z2|m|sbtN{=q(T_h?sjTS_oGJ9=Ghbw@pY}x5BuYY7eL4w; z({U(mX7^=JivzlLHFtVA0%!+?bpBa(@6h?Dp#=e~-olWY{z2fIkmTv&=ef!#o;;Q0 zfgjxBKvFRNoMC({MU=acU9l?J`OElB`&#>1sps<(3vB1vbOC>CEzVrKfTiMwCIbt| zQw-!od2&cS8&a9|gf1h})*dlcUm9OwXCxtKi>fgFF6aB7r{_lz2WEdic#Et*9a5OB z;t(mf-*+M3g14O@m6UkBR8k4H$i)DWicWomOzHNGSxIPyg&}etWmFbA&U{!?S)^pu z!GPG+o3-#Pdb ze#|%{-*nJQKTYnO^~t#5^_)uSnnmQ+pfYfX!yT)=Gc$EZKvgaYmzz0zXCyw%sZ4y0 zXpcgsi1tDzl`x;B@?)&^Hv|}tr>D81ed7z?T1Sy4SIB;8g<7jl4SMjc^di&%>}P-m z?ETC|D^N3HxLpeCh@mYhkoF}TFG8{xu?hI>iwsl`;Ja3p=P{>VlN+d@V>ab7Ag{?FJ*q9ltKe z*OeU2wkurJSy_cvr2AP`t2L3>Ad`-zu30XdJ(m6Idv;SN#UUl_Z+F(m&1bm>L&nWI znIJ3DQc6#ab(+On`{~cTP~xyb(!-bupN^mtpQ|vbN|~f|_-+NhrXDlZQm~RL6xhUr zy=Wz#MwbY!w!$+{eVlHW(6imk6We^8O=(iERwkT#{_IbP?KVRdpbMI zC@4n>jn;y<`*d9A(ed7AN)c6h5-H?d)JCmiK0i*Cetj7+`*jimv!_wPh<=H=6^o^f z7Wc~7An8yFM_$f+UVb$1hC>h)a*B-n_995v)r zp$5I;WDi_xyJj>o2>Hv&u#UZ|zrBrJvqjdY$u4&iuj~goe#_|s7eUd-KWhwQtU>nV z;hsmBiu_q>>2;u@Ui~+F=GmOu5NYNksKV0pn_9hdvooAbq z4fSlE#!{M+?Ij>7KYuU(7Wqx=MsLm`{q{=Y(#@d))8%KZMC}NbNy2Wd(j_;h$J?H8Kr+&4c}lXgBb&Y%YzyL$%;rEFk-v z%Rigiw7~ixT=i#GIQsRpaP-gGsm~YWW|q(@PkExel1+M&b-n7if)R%*)TQ?U z+MWaH)%K_(!yD*58~B4*nQjGI00>pK1VPdk+tr&~sog+|h9DQxA^0wY#~G^7;mGdk zH|ogF^zV)YyFdOZu-@mUq*x%0w{?d=#-v`OzhO)Y_nSx;lX9bg=SUQAV3=K(zsl*B z2t(}Yd|fxtg9!si+h-t6YVM?#V1RuzKMp8s*mMK>Qn zJrBT!4*?(@-yd|6`xHOV^gM);sW*R@ zkImb<@{twmxq-QKmVy=2Nh6=NY#Q|C(_iG%SLE{n+ge=toTlvE?UR0_(I)cQ@d4}& z7s>cy$(Q=1a%rwS-{BXmF`QEHRVgTPc^_W6+Q!dwfd~9K@))_~kwPx#G1ZIFTtX2? zE|XucE0<*ddw%02|9fgIFT%5ZHB$^uw;-#ZbPF!>hayiKd{DRPeOmKdkq~FP29uoW z-8|T%F;VggA`Ki;vz*F17=g&ZS@uA~seGc8mQA`eFpWaWa zDu}QA(lb3yK7S8xSy;#dbvvbj*Y_NtB|eqDXGl(D$d|K5Iltg>-De&FW=qZSr4O@- zkqOEg156{GLrEc>Mn*c_`EjIk?rX+Ov}gIZ{-o1hAJ2+=uf5KR*cxdCO7S*c%R8C}I~M~j|7@bzM;GV`$GiLGKNevegMDnnGh>v4 z$@H0K>gi`%<{_B@62s?7q~NnUs8%rO5RcDmUUjV?2ZwPGEAaGB|Kv2*f+X|Xo@cv1 z@nT33es}j4e)s#NScU-jePsibU&eYNyobl{1Rm`E*rCI3K6G&fm|wHM7bzWUeGJFh zwCE1UG3B_CpY2@_&3{2-IPOUbj*m4QU(A%dquEJIT#my?`fnU3^V=^_m*4I@+rtU- zJN%|8zZd$1+;7ms@5=Rt-@DoLO+~%s`Yhl~r*Lua}cd*oW9V{hv(UbW-pM@I9`i<~xU(F1J!*5nE zA>=8)N+6W-S(M_b_ zXt?3%0e&2g;xD@#O+%IkaHRamB>dlk{2bJ5&ghI-`wL~!&8HmJ$vE(j_lhbmWl6=F zT0Ok;tcs$e>eXgG@U~2PY>dSE$govsSPL;v2)YmAX;Y5i$??0Rpfnu617`>22xG^) ze)s!j(!34X4WOEScj6k(L{6Ho3NEQ9w~N$P&FWP1z!z30_w|S{ph^=uXEm)@T52Y6-m?wcbBtP3bSruz!pBhZAXg zqDD*S|7jQCpD!4Ids!u6WzNwS=nA5Pa(rx|_1Dg438Hn8>7_s3JF6n~D2vPU%;Bi^ z^BnwFWKR7)7rr_?q#`@qkYUM>5!v(bVJ?m|BAc8C%_v zpbz}Q348T`A3<_sGZk#aAc@P5;b@>D+3g{@Cke?A7svD7-NQ9$e2)>(|!&&q8 zMTih@Cde-26n@j zlN#m{^Tcy?IbXoEgPg)6E5G5bJ-~(e;?q3le@B7f`?9Grf3}bBTVRIca%7Xi*FFiK zUF|S`+5z~wfX~Xj8z&CmqwKU(_v4~;$&3hkI{Re7b%Vk6jr9JUO49>ViDvT@i_PdBR5{ev4D3S{jtiTq+M@(^prG3u( zGVN=hdz^dFmD<8vXdeF^pb(5>N0+oHSD2ajia>pX`y9b-owe#e^N ztSrbn2U|cCD;ldPy7^SSqeL9xNLHEz0*q%l@3gH{hlL&GZw;n&(6Ua^Z4a$YPP(b?Yx}9 zZ}2VF*DclZX1k<;5iI|Xk5=F)LTFm23#|0(X#fgDW%y1gZ7+6X&e;grPcCjsKDk&Z zST3iiF*mvkeH7BZ)GuVpwd1ln{i7z@J*r+sK$-kiKm4s%9L32B3 zD$ebUz-u$tzSJtE;@lCWIY6&A^9>#gG3MOr2&a}s@CF?>1l#3Hw}gjT1wrZU?IfP^ z63VE}T=^p20S0CqjVh$+%?`-w8zyvAhZms3hr6i`J2q4uJ|I8FUcOr!uzVQ#5e3xwW^%;DIyG;Yx@QnP|F z**%H0vUyGI+Lx-*x;EfRtAR>uq9?6>Rlc+ukygsiTXdkb`pBpjGR8oL6?lt`;J#Ot zFw{MQFvS$dLBCb0W0z?L`wo7>@zJ1EwS??Ho;VQvHkgXw9hE6LqOTG*JQ*Kv!a?U- zfj$QDksyYxDqQLxKl^?e-l!o|cl;ja$_NzG70O7}np%iB^Cd0nVr-TvWE(hpf}cdIAy5XK#R=S4^+u^&X&_T0w~6vi!}e!Vlo2Ul%o z=($!YuaDq~0}a(#1$p6mDsFVXRPJDk$O6%PYSIa+3vF)Cgf0kLRL||0G&d9kg#tCa zm2ZkKdc?Q5W`Cn)FCSgYhJaO#QtW;^Tr;@L06YVnAUYF7N%p+};0Ddbq@@1woq&Xo z?ZHP6S+0*ijwa4g*HD=;_SzFw*oenH8>lOWV_Yp*f$k6qbqiBmF*Gvd zq#3zL{eMHm@T^Z*;BSPM0&Po33>=k>P(yUIDq49LUg6l6H(mN|(fs6~Z<#~bon-#OM!<|cAJKRVL${LG296OYWJUhch%@8xuNc*XbUga`7 zQ3@jkIn*=*(_}f@V0-5`Y6KFyJs(JLoG%C~GCjOjpqI*3POC8-KMZD<L$Ysn5T&f7<@EwDoQxbxj*Bu1i9R%EW zC5N}jS+1E8TBJ%;T9muh9PHYAe|<;4JxI|NuusG@5^+g)Cg24Dh;v_o7jKc(M!l~3hszZIO~)t5foWEa-M>eqg@hD%Tf58F3(OR zE(|Kn-CPvxwcJjB2+|1cX(4Co%Ab$$<9q&ne5|L+`H9UdY-O>&khq@pOOds&!1}I7 z+p?bzPmvQtO85n3ihZxdV9oZrf=-T6(^f!qIDRblXmnqX=b9WId$g6k%%w+H2UhW` z5PpA7jWrnS^eV65=<^fVS+@jPPr^8KCB7WU-u=knj1BziLP}X_(Y+muXyt`WXoNXW z>O>Cwg%p`XDI6l<)y^PvSscHAD4K>$U?3<(OgoAd>DNAH_3A{d2|zL0^<(HJISMLB7JwY zJa+UUd@Pbx2ptS8s==#E^j(O`^5~!HYQEzKBiS{dK;;`qMmE+l^^2q+uL3-I%m;cLsg{-%U<{kCx;e??$A2gGMps@>{F-RWMW~Ctgfj$J8jm(5W1I5bh1y)ER1<80j=hgdcCUd7r7<;ab7BGYly5< zh2)gr@g3AnUSVjRAU}{wQ&|so%o#OCN5?{e+i)C0S{W*SKwOnM8Zmo;OfA$fjAnQ3 z`@e`dOR4`P6zA_xE1&-X=hy8O=dZg}KB+#=3g9N=eBosu=X}!Dfg1%be~SI<`+e+> zKcv`aAv~<5{e$XK^sxH^oG#{6Zm``WKZe8emBTwJVK{sw30r5gZQN0_Gyc&3hHcQ3l5;K8^;8<6aL(FbT)nmmD02C-Lzmlq1DTWQLa) z*dq+=m4B$TK9V0p{}U>$c^>SSfahxHIv4g^_xUoq*}xtJ?84|@^=}mD`wjFf4D<;; z^d1WReh+$i67)k{=*3CUvki3F*4(RX@^Q5(O8s~Pzmb8T<-`93NrdAIJ^05a!GGaJ zM=~i%@K;cIkj&#~Ujl)}IaG$V)b+mgbkx0D{2{M(E2DVk+|@S>wlXgV6l}gtCBz8I zc6rjfK=Cj3@IOXIBRz(PA=>6KXD`h5$-trh-Uzs1iC_F;9l@`&$gbGk03t5NCq+3yT+BxTH-F2?X z6_?AjQ6K60?d((0r(W3ln5)vbYl2xMojkG|on|gZ0ZhKMGBRs3a*OhKm+_m={N=IQ z5YZ+dS$p3LE~{3cFKIROyXUHJs;QR=QxbK3LwmaRfH2Vna!hd1d~VQmGHCkxXf^;U zc|T`P5}GZ~J7`W4G+eAu(o)c{YLH0S1+~l?Y9zxj@{X~YwZ6=)&7(=n=r<_5v>ia`6FV*wJ54-o`9D*fv@M56u9WhNEEaRnU& z=YoI;pg96yR{8T-gg(ANVph5QzmRrYrgtK-a`}-*akM<@kJa2>{$*-PL2$|DqS*X0 z$;hh(yn@Dsw7j~Q)A>Bh0q~k+i^BXId|I5?o z{1(NYNfU0%oreq_>}$6ptysCt52!(LR|4Ea)9**A+>C!iU`E3qKdW{_qdK1yweln6 z%4G-?SH*|yFYY$#KIRZ0R+kGB{)58V*<1OIRDD&Se%C^oO4uQ0q*Dn@vZT*22ro09 zX{zW8bud7v22(hGBT$T@&Jh|^QHPn7DC%<1%D_w*7JDwwkoIEUH(+F$lS$PO{`?p3EAW&L~j*R zV&%?{z^>-|b`=R`%Z^>dYu)6^O#kjF1JiBge$_&GfZ&$g!=e!Bf{JYn#-A}#E=s;*Qc*4?Uv^7n%?P$sOV zKSjG{5BNbH$_{x&wPD3nBhd5p(Y8u%`L9mJHnN^+=)s{BkwRYOa8)PpH2aQ!6-dqf zOx@;GvSOovKKCFQ<@HeQ2WL*xko0b`kJ4lNSEL>v?Ku+^&1ROgP8y#@qE)GdwJTT# z%I0osz7mdI=l8GZdVJ*SUEL!MUXXWI-&V1LcCCU-`ew{3Xh$9D+XVw}*CoP@NwL&+ zQr$!z(Fv_m2t8u|!+J$~zLrvwg+)AB);eVsg*g4Aoz__?QDk*(R~Vf-0ox7G>%pOj z6WJKX->2Hf{C}>Z`>{; zae@+7^B<;aoGc|7+575bX%yws*_4u@Wp6%Dsd}aDBbF$!s9lJOc?;Q?HWz)^G3F#V z0;N$Tx^;Kr$bZgvspJzQa_vHPrHSXO?#`(JYR{2eHHlzUj!&rL`D(J+?v8PN`-|4A zPCL}y^xY}dOGk`bo)9X zo*X>SEvk!WEwEiYH@bLMRs2sp@24rAyI=&jc9Fp|-pA8a`LR8IMke72x_B=B-+1H> zXZT4qcz)lm{51FRTmmc>=zDnfKIZfD1ZkuhvUkn?pZwGZ5AqAk)$Lz|E2eNd&fQxv zo$*Bxt3q4~6l#0lkCBdRLGTY$F|r#MD6qbg8_{J_sKF<1 zWf+Pmuqx3}*c7_7hs#2H7U+ko?E0|vRfA(zeeiXIjp69l)Ls<@^_#d2YGm%~%b(>&5LKZ<6_q3hk@J7;BootMppvD@E8BUV0JQJ@g~BqSimDcDsl%nOZUN zQPFNbF2dA`0MD+qu6e}KZkF|T?Mkn*GgQ%(9YsSfCw|=Vr3^J3Qp71LEk!*>XO&*W zos>W(^WsC5k_v)J+MzXQZ!@T9_lA<9LRDag;rIaPQffl>UAG>bhNop7H#Q9ar`W-?8ewx?bRCiJG7(9hBD>qo0oT?$A3QDhZokz;&XxH>ia>`1{7 zw4@NiQtqF0hQ{8rlXX^7zsYN~vKZ|iR&9L2c@rz?TzA|N8M&&c!PZER`CIuIoGFKq zI~ z_reB38of%_wce~J@(~JbP0Hm#>(a< zRJSkupjP&ds#tw*%ZJ6vDm*KDXrXUqBS~WfA^XKyf7h^E1Dr19TlD8G^%|&7c#hO+ zdmvK4w_siaCbUi|?D%CQ){IjkG)w=AW?icbGPHenUQwkp5Ve62E6P9`PI^QJvePup zO-5Ftfk37AGRW=Yg5p-`nZ^31+2PGH@3U3XBQ}w*mnsj z?E4u+6^=jZ+xN@2_}nLPU9*-f($H-bZe_=DQ~Sw{a#u$e89s`=r|XqC9aCFF`R4&2P3UFCF?E7qK12 z!tob9<^B1Pue`@etMY!Id2sTZ35S)`1?Hs($f?*bDjz$-(W!Jp=_fK|id3Yw`xkSk zJy2Cwv}%`R?QF$v9*MqI2Dp@xUXkccp_Gzbx=>qrE^;~_g|Tbf544J!qw67@vQ@$v zg9Nvvqb1U;om0mDr91Q{zw|X^Q4ckOzD8kgb@1j3bfJnFLt_SgkYv^X)_sObnud{O zs9yPE%DK8K;>J*77)WsEjSSeVkEOrTUIO1Cd+y8zf+zMm`H}RM8&%S`ey@^#L@hxO51=oJk2syDIl*_O;Ez!jPF4-l< zQG$80a-sPgnTcZToCxR-GuD`>1{BxZU$xhmyc~x1@%n@8zW5AlR4fw&@4wVXx#Rq;r^E9f7~y@Snw-m z&E>jsnX4pX%SojfhdCdG~DPCdO$G=P=GXB-~?CR>^|G+TJ1QXpN|I8UbPa2~2bi;3{-PhE|2~s9Bd^!6woIMyBg|Ts2vh!>#->e#) zAk6tJtYP)SSnt*R;~b~noB6kue>VSi6wFvNz6pkr&%Y4~8S0LZZY`CnAiS%pK8hO_ z>{mrH6+z0P7#1taV;YZA^@_}629FYq8)P0sJPzVOs$yy2WCHL4 ztd{!@e7XL0pR2(?=ON+VMHNt1AQJnC`XrW!1C;v=p7>ss69LFyy7_6152Ti&V8IL) zniMtaY*N&@xnFB_pH30S3~noPYL*;MjIddIka-h9Fvgtr*AWbsxz({cw~8hYcD;n| z&(ElHUfA>>#gMJMo9V_jqKKYA08Pxa4 zpEK9ERap4$D`nxWZ*-b?!!)&Gz0gdSOyxuVIdF$BwC`-BD~V45Nl0b1*GJOMlhKt1 zG&1VwAW6)N0EtFauCb0S&Rzh?qBew0yhBoh?DN-2kKu?yt3d%inY!OBTRsdcv^q+S zXSI2?J(BS=Zcp$JgIdl{k$4fg@Up`VE_Af-US$pZPn%RIadAYz&d#m`u`m@;^{>Tm zqW$(rF~o-yyZ4ixD`>CqTlnyEJotBb@cTRP5#oZaD#VGH0}k=lsm2v>$P+BvP2nwv z3S>B!3)DrOTz|*OE8{|i)+Bnu-F_9En)@kWYdDPL29EG+|4OE zTv8>0+%9yctepkcGFdyXF#mdt8Z4 zBn3BLGed>mpT;sAtA1^YF$+13jdQBf=q5c%63=_v1M&)?;@41|r~{j!BPXedJ#DS% zm9R^{G$0JeBxfqo#-+h$0BN^*rT+f-zoPmn|U$jTx1c@cB$la?|1M` z){eXjPDTlO3?*8D-*Mg#^Kq%#& zFswHmpXZC?Ov5?`^3|Ofh}?z3(Hp_?cS%TBayN8EWG2ZlESG{vKc+6je|Y;HmGiSI zQcv?Nwhr6ZKw6PA9Ae6`X`!5(j2kG58QOW*plWNHyq@d6lZU3OH5;FM*Shbkwq|I) zfB2Xcnd^Ru(CyC;&RD0cpqU6-k&&0uNNqyQVwxd)f37Uy(oWNiCs3tw^N%`Qz%5OG zSD2?&qq+XN%{Dko&VN;#E~XEksuJwPHX=Bl#OA~E9zX3_aKDd(H}p~}wq+qWv4 zH3nX}S~q^Z*53Rf3oJG8+pI|J&&GIbR;cjT+DrM!d3tw-%eg^cGl-&vz5NhTI7ReY_=1rL|^hW15!A?#Lv^;%TqIlJaSUSMuo$K zquVaN^j{ZFz~si`+(~Y{O{-7U+pPS|^YZWCs2;~1UP*1A15fmxr>_vlsNMMymn3)Q zX|E36pAX9^!=?%Ale0^&RNN7^zGRssk;gvP_8x6VtTJmP2l0#Lq}bzb_tr>;QiWF; z^@`QfzP|*`Mjl_>P91)Us*a1CK9Naex(8zks@X%UymgJ&caY-C{o;?pd`*e^hoRZU zjCzjuGZ$#)m44BlX2k!5OkZtNTUF0;!{*o;@!`#j6?T&l_z; zAa@w&bFqfNYP*`4p;Sd}B#X6m+f<+(&m&g})f2i#DrRdjd)=3^-J=0CU_F~rPLlQf za+{8n!B_KwcqveoImq3XcZRdfKpNlr26=x6dA<;7b|^CTmkPIVrHV|N)n|2e;kN?K zNPLPX@kwB<2`fI+blVqXXmcz>naewxaqq`!OcOrg$CSSeJ#h*z0FB-*1ByHDJ%luG zLw0c~s}gmnUh^wC@w)9Xg|Hrz#}i~at82q00Q3|V8;<|%(>UM2p_cqw0gbl` zh|I;fdz#CYf9|Ud=dRMzK0eWPu7bfK?k# z#)yFltd;Z^L-xhG`DmgX)cC<>AxF+9cA$O)ECY3dNm1F3BSmF9jqo-dpkWkox>T2y z?Bx*r?Zhq=h5g+!)rFj@^uuu~(*~wYHf0>ZxG(i&y%bC} zN1AM{Uyy8T$fj#lSF1>>{Zctzsd7UfB56qqkz{XDk@UfG9FYtehyQV}H_F5rNnd#@ zz;wJsPx)$k%Hs$PPUJBEs=ffE!WHv|vhRJfnFRvVlx*HW^5QHz{_9G#&YH^IEQa;}2jX0H!ya>_y<5 z-=CgSnRM0|zWt>|wcIY$q$G#*6Pj*ZB2;{!qSZNgLrX)?ZiJlbaF5A(7QpEtf~)j|ia~$aO}^d-$HAD)Bv7)q;+r842H#0w zNnB)qzO1dqWPNPnGifvm!J5A-wp1EP5y z*{|j3wk=6jOn%JM?N9qW-OdDAINr#sCQrqXl}u?w^QO|+sBUj?Tb%(7UERtaXZb@G zWdKv%w!_i%-broUBMsi13y%2-3qzAy3$Z^6xtL?Vj5Pn~4S=YyLS95{Hxz$7TB5yu_PJA0)guHGb6;ztHtadnYz z{K42@fT_54pv~8t@Dh(wz=j18m)KT*NH~8<+7M%cEm2WoDib#)mzHsX>q&mv4YF*Fbxa2_R2jVUr}}XG(T#5+BlF8lJwt~f;{HXCj9Y`Om>Bf2~WIYwfx4t-Lh{_KKuy%Rmx-Px9ONKxc{oxTB;kDvk zF11f$RljJpExNesNYO>Y2a_jYJ`h^uaIDq#dn=41`qPki6(SACfAaHBH?kM|7hWGB zn>__&(cO)iuv%+ZE%r_5q8p{~PQUO*3MZ~~Fn#Y}lKv``ZAOP25#ncV?O|F|PO+Zk z`AhT2Xwb3AX#$X_Xy7dUvd#Z;jaT|@N*ik#KniPl_IU&^$AV<*k`cy{sxn8R_7@^|cH?F?imN5@9<9-`4x_jae- z`dwSkwUaXWX2i2_w__*yST=HQ{IG8)`xdJF)lNQvt8n}%FgYR`A!<6!wUa|hsGX$i zn_ZRvEa;agg=KNOoohHKEe7{A2cI~hrAG+9Q~biWQ8+QiM|QJ=%rT;0 zQEfPWs-ORT1Z}G343Mc2y{dWS_D4rTrwdFmqJVxm#{bgZBlRXq8zV|1g%M3;zK!1D zh3|Pr)bA?Kh?SIG3m|YK`V~l~zXXe``;IHz{_5TJ_TOk}~BR zf1o@SlnINm+NP`BreUq!fk>-35$O#bd$>rs*q768BOO#a1WIbIkd7w4BdYo z$LE2fM%oH##ie0bh|z@O_xqAMQ8c-g929(jg2o+OVsCxi7=k#CHd#_gu4b%xT3{Im z@NA{hC6;o8LS5)X?PX+*{C}GUd>Q`bR3`Nw*)L|=lEI)Ybx<)O6FB60LG?J061|Bq zIjCMrD5^KeP^PM0LuK$SQ;9DiCqRs@!alKH}r;!Ka6dE_zXa6BnY;k9?w z6eh0oD%0hcdlkH+2=~BN!qvVPU-A2QN56uC+zT9A@Srx(jY$W`2EN|u*}x?5hvV-; ztD*j(sR&BeS4Z>4NwXt1aE064^xEs%z)<-nN~j2w0!(e76$Z?>`iuLG4KySDo~9d@ zX3^bS>AcS&9XIcFwDSc53di5~MVaM^@&+Mh1!w>@lOoo7YqLFtMDAuSdXpCY&@cLh zkt+&6O^U`$(fjGy7|&2L4CJ%*Q4TRFk$sMxfM@1=qyy8%PG2TWmS(;tQZvTldsWwb z{5SSbIE=)41krVzF!GU4&2W#J(kZUKPa}oCH@~C$j;;0dJ${s@?_31=FMY`5`hFTK zv{UHD?~xJh@jR%i&thi4QS~1?JXOyDbvT}iqcf_0c`_oBuvRp0mNW~Z>d7P&#Q6;O z0fDc+Qoe~ODgw6xOjX?h<3w~j=NVN?zk6yc;l44@9cv$q1`ML7L6o?JDz;4XH`b8L zv)4%erhfjTy!?|Se`o6hFobH93>h5dX`KxE1x6X6qU=K{V5-yRNh;ph8WryeNZDzd zT95R^`?(PR&(-Pg^**|tBnd!#+#Hf%IHrDThAhLNHaH| zA4eItUg|01?92Yu)(`Bz^m`r|$+%Sw{#2e+4;K*Ubqv1Yc25rvfuATZL>hjUPec#P zkZ?5b-X)G6ED{Rh@?EYTCdfCVhj~9adgzY%7(HyTXD{&eFzXPJ-fCYC!;xO%d}+i_ zmvL)1y>ntCmVDfg3V%Ez?|7XQTJIM+!H5ju6+xl~>c6j+JTed9RHr7MT1Jq^2>6Ey zD&Xa-RKPtc;t04yu_xf|P+r&Co$-Zx9;q`D{pne_meVtLS~cQKKIK^jvyV8NBipBb z^kmx`BEs={0avzHKuXPp6o}^KO7kPmq_f-fH2K|?Z5#P!WSa*tweiof2Hf&vcjMq? zHd@;H(Q03E-;(wlSlTp`&(6mj{g@^nWh@!p+oU8A-P^ddi3HP*W^AZqlX4AMHzk)N z!Aep8GZdipkWNTX&5-VAct|p6e2XPI*D-J@xrz{j^^vs>6s0HIk8D>qq}I&RvaS5G zdvA1WO^j6H94?fZX7~C<3;arf&RXCwzrZ_QfnicWCTCjSFFFtUc@8z2#x6#i+=PF2 zjaQv-S)n?Qtn_r=?P5>oe~Lui5do*3N@G&&h0K!Mk3SZp53fFL<4JYic)X|cB|msN zKN})g{{(%eK8=Hvno7|?UO#DvMCZ9~8+7QeuFl)bH~!$T5MZiv`x5GthRrtEJ?VtG zel2p8h#>5Xpec5hI9KXZcd1V~PQYi;8wy=&)45tx`uRnQJjw1abtF3vX>p;NpXUuP zPji?swdnzJwKo)HmqV!hlyZ1k-y7>TB}%l6X=E)3CG9z*-s>sQS_1$S4I;^A*1Qb zR7UUc|*k8j>onIRq{pdF9!ypyzI%o{feI z$p3R3^7;;rqj~G4K`6bxI(J(|$yc|$DpBd(! z0hqyK%;0XG4O6kU*8|l4{6g8HUeA6MQF{l#Fks8R$hG0M_JsS44U2F1aFhsgo=r!xbrnAjB`;lo z4fx}KwjHY_w>ZD69bGX9ru{K5RGpfuxuEXa+dcc7yv(veYJ`A8Axa$h>TjZCnE zAyDFv|EubZ|H(w*=aEx5&K>kd^UnDHdRQ^zf0L>+{(nx*N5}tKc&%HQ@jv+fHvXUO zkN@8mWE=ea%Z$vRv_ue@@xMuhu*;F;m+?P8GX5W?((}jvxHxmL`2-l9`t;Mco^EGC1E(=VsL}0HW09kH5avIgakO>=2?a6WC)bf*BHxVG z$^fSRrybUZAF(TRAgw+785&0W-4{bYPYeq|q<-f$6bxPBcivPf@A0$0OeRCu874)m zmqH4O6~3hs8^Vucb7!9K`JGe@=wN=wS+6Q5idRB?&Sz{ct?19f)Zw%ES0?MU_%~Ua z#eDf+cllqJf{Na?+ znSTc-o5HyFMq#B1V6)uKfpww>>j{B%>A%C8tvmby*Xbr9O-!yXxGy%koJ!QErp5mP zw^bliWIjIe;NWiaz{y(08aY=@MDP!SE6N@Ntoy`74Bejm8Vxy($GGk-2NDuD;rc%v zis#Dw7O%`e=b}$c&p--)awg2j6Mo^*Ug6WFa2_i#ORhrPyWv{rDOTI*pQ>Qwn8u?; zc5w~78r<^?(+J|5t6V`mMhe%u?G1IUXZo)7+rge|eezs!t?V{sVAQah>wVycE_uUh zf?c9v970Z|tY)t-duOCJu`T?7VTact(NlTOF~?}#xBw}dKJ%3%Rd})ClRze* zYUTr=u%dTuVfk`xyV`61BmDe#dilQtVPdS6nMYGZi?`%sUrnc+Ay`6wsVa}mJEhd= zvb2k~E_3>u2$|kr591=G`FoI8;dr)Rb`K>?^LM66(frjTg^q?SRUO@rs+{I8e~_o6 zfEdxiy4_O6-240*J%qf?cn^Es^`Im2!-zOK_PXkG5qTeQGq;&w#h$fE1no-FnVfyp60tnw9HCy!`*+>`UNmtp5MUG8tP#T)Rw6LiSOX zNzG;C4kJX?WJs2h#EhjR8ryh`R0>%`NRl?WSsINPCfO3vqA1UGDNFWc{_pqaJkL3I zY<+)!uh%%|dG7gq&gZ;8pYu7(a~5bVnFXjx8)gGGeu&s`+$5Z)Z4XvnnBPp^h9Ng852XF zKG?^FdM;vsy7rd>bweQT)T^JdejBUqxB}}vmd>46!Xg(1EKWwED1~nUx!@g->}0E0 zTx!$T-lfxb)-nvi0|di)n3$+^ux_@WSo6RJN8Y_kN?@JyG_d|tVZ8`8V0k`*uZ2IZ zx|>+b=U+Ph`Hcuwz_6&TiyQ`KfrVfJw0pRyZDFy2_71Rt6deA22W#{C|28bTgD}|J zrFdH?xJjtLJs?}_XqZr)0#&dk4j4%a8n4)Ff88Hg&hbM!58Q@YPtA6fO3lc ztG1H%|0awQ%T2%_SR1SJ`={k&jbTr?3=+$7uZD(l9 zXxd{*(7<5VRM!vJ9^M)sa@KDLq)yA(xB)SF)evWC;&(~>2rc?$ z7ow+8FoOD=Y6MyT`@`J4{&9xEd5}5vU&UDe5d%Cwd?N771xk*EHhWaVQve2%3m){z z`=k|)$+)rK!$UEh?)Qc1G*4L$eGE~-rYL8W)s*#DyNS&bBV8LU-SDRzCklAVv!aZO z?h-MKfW9I{`p^(1YD^Y_h+XJQaHKHKp;+PKBKc2!0`ak-8m+TxM=ITgo;pB*covE$ zCHBO&3;pNScoNKh!VH~bPZM?#Gub)@5itBLtA%L}5rQcI>=^WBdH#j>0}9WhHE>Y~ z)MH6@85Bln;#MSX1Xq{d-*I(Wc4YvFY-va>1p{mra^3SvRdSf|g(qDMzCsKbto>Lp z=m;nr3@$d;7;He5%VkIaB6ru>X+O+(SZzN}ZMytTfaA3SUR-eHwC>aKN)Y1r0mrKP z>?A^Gji~39fDrW_!~%;!0Fb_IaXDw~wHjq`R14kSgHk{qli(=%tS(?{Qt`w}%d-If z449VPrrjQC->gemamKw%dyl3)(;c++LCZ=S2Y74u130Lpo52DCSEPNR#$i{5S=jM7 zGdu%78#^jGZU}Q2jH>PRbLWpEU1y!cN6cX-5;F%Ma%e#KA!HwuLqQ=gWXMmF=naR6 zz3!tc6;T~UM2Gyjh*iW8tp!oAr<=`oMUj@g4o7&b>u-N6>sq8A*-a=*f--n7lQV+} zP6qQ-GEN_qkYuGy22UXaRj(Fzb8V_GVz8-+*kcMVVc_D})cam-Q_n(`m_EoGlM%AH z`BvN8Z=7#&?W@%^*cUe4VksGN`o+0$Iry|`cs9^zP6p3lFIRCT2_-pqtc!?==`As zhSG{47T_VHP}(XORXk^U(7Yvi)0^js-6-=sJ^^=_AwQ@~UW?>hY3v<)c~S|-n;a6P zMws+Vb^5EFk$w*pJochaK2nmuWRj22$-igv#Yk>>s=!&dz;G;n2fi*Pp@zSSHY%j| zRk$WCV4UIdt}f$1a8VY32F-J9IaQlw-=)@~(xi64X?Xn;L8~l)STEJ(B!2(K>Jl)dW1;e!YNWu&HTw! zyby(df_X$o;ZJR&!WS7y9r>W}@{ljUZwFBL*Dy-UvlR{(N+q97iNcO7!%xUQwmwIHIzo{KUxG zUSSDYXFuVh+5<75`s*@5^*jOx)j1DoRC~o4RR5Q$3O!$eZN|ybQT&8e7qhQED}4S$ zeda?H-dLAn4%(l+jIX9@T8B<7Ufj1FVywKu*IMebkTt*!0_e}aB2==Cj8u0DRmhxi znx;D35memu!0q6JiAK6-4=Nr_b%rdBm*#SX)WvhqpVJmrVQxGqjC4tI;-05Y1=8)X zdm6g@c;)rWW_u-XkogT1IyHc9C7JE8YZxN6zwnaDY&D4*I+=~(dz=kGVg3A4Wj|69 zCv>9s&UQUhYIVr|%t55(Wl-1wE~MC!KM7Qr*nS~3kSDog>O~w2!0#XUfyb2j8qbMt z2j0nbd70&u=s3jfhdwTZ9lHWWJ;}ktl&VNYn8Pf2R+KoD3KtTS4$IhaI>el4oB<gZPnB^}rp6tk z+hJs2Wk*&TgG7+ILLNEzGRpeK7mF;077)VVUn~Waz^-0KCai^#Ux{y{3!IL=1@4hS zzQMMpyq-VHs+52Fb;`#jCAXm{XBh{EcRCC=sATAfg%_4&jZHyDKkhoqP3_%kJpnOP z>wT)ZT!XgRsn*{%l4?ySIVGBPkOzoNYghuQ=R~uX;sBQ@Y)9qcE&5v$D>S~>k3G%~ zA*%8IR9%fT@WwtloK;w6vA7;%tVsBzk?@L^u#=YXNIQ|Rg_5w8k+2FS%*v7ZW2!g_ zFnn2tXl2<)Rnx+b$rQZu`g@s7hAP(ZZtKB1ILqznWJ|{c`*jp8y~Rwt!a-GrwxOmS z3)(<88a1~`D}!ubUGFD9CzSmRCGA=lERjB+UB{D3mdE^R2M3#Hr^g!C9;rE1Dw z+Jf>goTh}D0<92IYUR=ca;yt6$9~8}bRLW2pk=RBI3^=@*Qjoz?~u>VnO@76zlGI0 zzc+U4kQT#(sh9Gmcw1QQig~Tp#o}Xk4XHPLpEqZlwKMttgmhoYHuvvW=U>KKd0JRS z3b(-Ft&zjO5X%OO-9wI^$prdmaJon6_k}2*}x!vZps#ZbSuwj@-9Y;;hey_EpU|7#Pw`x zQ*hf@+c+or(72SDOjE(vcxYVmLPh~YO9;~;&OXz)1e{h%O{*miQG-;GQwBf*k*|-y z|5n>Xt3zV!w#X?@;7PEGYRonybvcL^(6`*hpe*7QW9RVdS_b>M=649wJ~CpPQNG#i^35 zB6yVU^x?~WR#!mMte7u$|B&&+XD3<-WpKGyN{PhA&B{cMAB^JQ5og%IGUU~`6!#S; z-ehilHzU?dW!hN#6Vp3KB;0=yXN~h;WTkQViS;~BE2~H(L3@q6-Xka0#s3KqgbmG| zZ5ro|950*Z;TY@jX&g>R;E+GLN`iHl@BW>Ut0S5eDV9>imy>}TU{3*qV0ihA)csUo zN3Z3D0k(>uN-|*Ck+1K@|KN_pp<1C_$n*ogp178;b21ZBzllmn-58Y^doc2~UdSjn zq@ulCkl}Z8_8wJ8d&N|}S}g+~)!yIY7o2JphDo<(L){n+wR~F3LVwQdA7aG$Qj7CRYxs(R00`~xOk{%A6dUuL z2M~k>3p)?6GcCe!M}*;@Ns-x&h?hKun>-q#c!3D&7=3gUPCj(*u;UHM)}<6{lp*<- z!t8|xmE#he;kdl@c;}ot<3KMLa;yui&d~p z1w2p`zp(%aK34&Ch^KWb$WQ@040xG^AR#Sv*;QC?TIwnltWg2BcJU&2ZRNzxVI~8Q{wECE9k?!_uMyydx%w$IArORt!<^n&yQ;=F5?iPfjZeRK~@+Z&Ov zrTI=|fUa+Tmqo>CjBD1=OnLlo6)>?YPA_k0BBd(F{TFm!ddm+Zp>>%M&n^R>>fb?I6T zTQ{1T+r;Pkw>~XVhwFdxUF{+>zb(gzD5@;drfbh$?_O78407Fk3_0h4<} zoqIYN)u9jL8lY63(E)ZKjx?MiXbV?6g|gRRQAAaX$wd` z9GZKq4ksLv;KDJ1vkNYpjW~(hlw+}r8{(fe@k$cQ7m7H@o&nj}#UA&$G!flFI}5M8 zevhForfIu@Rv}OUF(B~Kd_mx`_cQ|CYib0d>Iwp5H?h3kGcn$ZgCVO2r~Bp3pUV0M z%oiYGYSYM<0SxX>byu`pdC5Vi625*orv2R-qN7Haf=*8*8qi6LhgFq;0s&C1+MvjJ zm3H`tSOTXd9XxyKVtq-9-=ToioT$-EJ5|z#%uPO^(^^dX1=6CwC7(E(N!kR%x2R6r zlWEh)w<)*WVu87R{DPP!o8qp5b9+NLhsk+rkDQ?d<;;g+d)M%QYWtz;^e2hY9b*5X1!dnXht`3N2$D^b7E4m-k-}6}|a4B8nIJS5k!R}0XB{A~j18rgt3~%Sn z+2~CxhMSHe5^%voeC&=PyDU$MRx*1v?h`Sl$nkH15W-$5O`~|m2umDPE$-1AVlPt3 z!lg_-O|mOG+2cqSd=;!dfw6e~PntCQb(-2tLmPZX(sVOvKG$gqF-=2`)8b&b_=MZe zK7Up~<*afA7$jCY8N$v;uH>BC%r)Mgh{1Tn=8EyYF;5%syz1I`pGHkKp8;r?7$V~{ zz)*B$Lx#wq<0aRA+m6U>zYpP=O0s)mp3<@3BCBb?(~O++m2|M*tzO`{5b0sR&)?4q z{wkawrWa0rY%2J`QY`1&hW@gq-wyiVGj6x{HUlT4mm256SqQMc@PJG_xm2tFpa`-v#H0E zaL&vy(L(^|p6XKuR^grKVh`qO9bJt1oVxhQU|B*H4Z3I=2VK-wzT|fp2PSDF$6Wyf zIVt1}BDifPBB?;&elUH+{mxpPUd1KFsu~S96l<$LvnT(TE_ZmXO}6j2RfNLZWLtYO`8i!d(7h-zbP^u_KO z@+arPd>pHyG46tzv5ni^HvSQAFMEv_HSAKIHv;$uq}*mYqH*pWoA#Id{i zFtR2^y601xY1+ViWqdibMbxV42-`?_4FtgYhIS7ql(H+T7=ZchApvI93<0M2 zEDg-v(HfY;l#a_AlBDOr4jN)^f5ENq>1+N5^W3jrT&sarNo;JeO z(86_uNUqbp0QFcNwlny!`SNlD%=uyjroW+|py|7S-Zl)ODUVk;Z^h7l z1quaBe$rO=;p1z&A#}PnEEIboP0}xQHboT?Fq|DL7>;{KFx(C(9pgGsSz|cADlrW9 z=9kgR{JvDtCvE`-hY2w_1ymlqDHf6ASyu}iuB_=YhE9~4f~#zCiCH$T%nWa4CFUuH z3vKZ`N7cOLwg~H9D=QJzD6LRT2J7TtuA?8p^H#?&5%h(b;!N-QS zw5AO-1}!(Zr2h3MT^uKz#&+|Bjs1QsUc&U?D0u^m&;AfR9TU7DCWw?hlMOk2O&k;XGF7lLd3nhuk^#D*T@0 zM?3aOG8_r(=C>BPsQlWf80>gYBeCP}-V!?we_Pw}T~XSO_f-))&PzLUHURPb^SfoD zoTk~HjZ;IZO;-<6P7|N?mmvx@h1uB+wV{nzK|?!YS?Ncq*@9pw?{Cq3qj7Ze>5#t6|^p1|0zS>8$=4^6%C=(=XgFZv>C^5Tv6Z^ z=1p&(w?(=)k6pKg*61t3(J9*BDZT>ht?fN}qSYikwQ?0S?{kuVl#I1d)RH=Lc@3&v4OaQgf}(?ZV|A za54JqC7EU01LkUdE)k2ZVlpeMGy4#kAzSFP9%9gEt7)Ror{B=}99%)`^8sQgi>5;L zS*QF>^cmV7Oz5XIN~NbZEr}|N_JB*K*xe7c&X^>oryCZ$I} zm$@FNt!i?=3!9q{gs8hA`d7>M7>R6@6e4|!&y69@es%<<>S?HIXsRnQpmJAn(bvxC zC}sm69|s>FZ*N(s&8OsavKeI9bU>EMeCC1;d^|q9V?fWj7{|w7fseneeEi*BZ9$(2 zqrqULEQtA5l6-gA_s!N8)JN*b5R>0NEjJ%_il(6~D z^kUc04%F-n%$pBdD~sflK%V0nUxoYn_2u5e z>=QH;I8N6vo35GtRTs=&0JB8Pz3gf)k*Qs(1df&WVagA!8$Q{}nRh$Tz*R#o6eP;d z%w*Pjk>v67EbkhVNp}t46U?NLNStL)Sxy^?9DkCfwYKBJt^@KWxjB2kECeR}_rasv zcRk0!rk)>fzK9uQm<*LFRW;LNP&Xe)hP(}TWz8Mn~eZpXc&jqfdJJY07=V& zw3I{o_p^D!f|znqD~DYBivw017st&I`|DtNYJ&l?SZd+>Aoc-FMumW=n3Fz>AH4oO z$W#@zF`lK8RKs%0D7cIUAhBDPy6l_|Q4|&4v=gkbtmOqfQp+>&Xl}A``*U)C5HDbm zD{c0Z#0qcIOV@f*VUca3^+ybjSjQJc=izRtqn{w=YWW_<*4vxGtO@G6+?SN|b>S7YDF| z{3zN=s2Q#{xQKP89Byc{-F;2&j| zE~&7sgo@%yWIyg^=auaC6Ia6W8nW{GD;ikJf@Poq>NDiFD~Qm($QFNVMg@h@^ymc? z)yk$eel1=iEM_-BsIV#*enndnzY7Y#E07PBXXF{;cdy28D{D4}U~awMf|AAYCQ8(2 ztU2-Wem>mJCiin;cGC98OFgFScWxxB`;6?fkVRkwYijJTTygQ8-Thr^1qX+V**pM2 z`4c9$vhAoBQ1N5`=EHWo1Rv5XE=FU8YZ6AZ@uC9%)2!ipEIJZ7Y;pLacqwHhY*f0XpW5)$A>k?e`}Cq zjNuqdj&i@^7HTAeN`MQ<&w`EJ#Om$dvnW?uQD``*ko~Vo(!A@BL%-o!TKBv)bPw_4 zd4#o{fwdXLQua0w@&tZHE@-Ta!OXrwyUpHMpZ2+$$GzfApD-NiqPU8~WO8_yV&el4 z+w~Wx&;UD|v_*8Jme*Yb0V&_pxzO2R|IIUDujoP=VSb`wAd%KT`fr zYkV=^=1at)umo8LhY*MM0Z?M2caF z4DM{Hvwl(987sU@cW^VaPgu*w_96}$JchGE?Xjx4q@$B_4HTV`)sG`nvMvg%B zyHvN%+=Ull(Hb&3m8Of5Bv{Lb9(Eimue}-G2tI@xVJtKuIB(am{uQi)i+}}c!vdhf z+HkI@?)JI^VW&t^AKdT%Jx~?YmksRi`Vmi35iVlZ51G+4m&|54Hkr&C6EFGp{`7sw zb`WqYRGeMb3TPYPQqtcx7FrPZAb9=RhIpnXewM`boRfxUslxR@ekkq?P zQm56N3SykdSAEtfj}}dS}FOlp`;n zFpnkVYcdCxa@%PfeuJRn)H3dLPgNW-%6@$(6x)&YGET-s^J}vF0~sk?^&`wSu+X9}0B zM1_IQs{(|_AUzDUSy6Npc)kKpp0Yfx;W`1*x;IRPZiI^xaNvnyv_&(T1xDt(&Lcr8 zDa(H0P0j=y9GG0gBv(Y#uc*ZZJ#+n8fK zQWKz-gGL|Ap>06l|q}HwoaBh+Dp?ki;}?F>mO@a^wTUp zA`92@JED2Q+R~78T2epKDa&t(81&cf717^F1dioTh|v0LT|`?x3P;ONC~zB*NC311g8B*h1frAEY$wt`Dp)BL+t-787(!8wrEDP zz$nE2ahj*t|Ley7<$%u*4ee!6IQFk%w132aU$s%n{t-C%Jz7BHR~UBX*uN}WEgX{( z+VWoszYUmZ_tg1-*r+)6_(^&iE4$x_&LYANhGwG?(;^cwY>EQMVsU3H|~q+VvJ;5!xURKX7_U@smi+0909 zNCih!V5{J`3QnrvUlm+X!4(y-1R&3^f(R89Q2|fc!fz#2z}-!Fs-S`>6-29`8UmEP z5yn&q;#A;MK|2+6Q9+Ul`m12D3OE)6-53>2QURO*U!ph90JC%qmZ@Ns3f8C~Lj_qX z*r|g3DmbJ9TLmXoa6tuO9KNQljZi@`6_imylnSb;psorUsUS`TJ{7c6K^GMysi40K z2CHBM0$ve};b}W;eLzXo`$;O8rh=I&SfGMsDp;k0H7dwZL6!=3s$jng4ynLa!ATWd zP(c`HA5c;i6jMPN6-23^nhNTwppgpVRDk&n8a%g`LDA53YJd45HDo<7Jna|Xp_r(y zJ{A;%bD3-#g)J=T2$29kj^gdzxNE*mx-}CO1s#pWBjGb;`5k|#mav6$sPd1`QII+b zPu{ebvC{bEX$WWHZvmD7FJmH{n7N95EBv+wf57%-o~F5z?-26zgD|Au4+)Z#=xZk- zU;EwB;&j?LH-=)~c(WV~d@M!0%HiP%P!deOZK`O_3d|_)I`P+3EXv-M7fuH|s`!bS z0%Hb0ydrSm37cbp*G=(A_cS{UFS(`*iSYHG1((v=d6ir^#B(El3p|Y?Dy~R!J*>QC zHwg$nCuW!k#wC0r_{+vjd$TOO5qONq$8I<2WMlw{qr!~v1WVs2} z#a+NtqUjXbOf3vko;3pEpl3nviJk@9C-J5-nT@+w616v7fSJLVgWUZDz@%4zrI%r# zhP@p^9#k6ovEtX}$eE>hqF1(cv#-kk;TVF^9*_N*oq!0UQCa!h8Ij#Mp6|cqbNk%Wr~Q=h^lNs zTNe+c9j|DoM0b%F_@$lm%hN8`!FT~qjS!Grc6=fp78b|XKIgSa#6!W!wGwFYTM{0D zDx)KuWTYO0cvx65%8Zr?4b#SA`)VF>Jb96&hd$ zG=@P4397+RVXaScQ^B0*Wk`ki2*%ft0%4YVmxT9y@W;b;lkqpoUip$+4-O$oU_Hus zTM<}ri16wmR#6D9MUd?6o|+wvGN8T!-ljxj6uS#N_46j|ifMK>*{N^Rw_!(3LAtTG zV7GCkTUb_-z)m6v6mE(40>!S`J;eLQ!Vb~wPLkc&|6-S+*lmfrIlC&F9XlfdwSw(7 z($S_+*iDsg&hCd1ZeiJ}2^@xz-?p%m6uZ5*U^iZ~W9KL^{lD1JvLW5yci&vP2Q@p^ zmw;Lnf6KzE6O;#uXh8WoqdPX-Eh&3yfs3q{w^l11d%CC8y_D-lUh_0BcK814WalYT z*}!f_^v#ReR-~I~iyHf| z|8=tS6m<`Fy5}q|3Ah=%XEZzZg#*d>@3u-vLP^s&q`!$0%4%NhPzTt7zAdjl;1&MT z%@w*m#VuwOd2vK=TV6cC`aK@4a1*)4YF_Nx|M$tx$>7!I_nVjT0mTa^J68v<|2Wxs z6c`OGe{-RZCrbfwvU4rsgnjz!rMEUv_13nJ(7m-%B~5SbuDjgc+7HDg7G+ntc!`tp z&ZH6>0`yt_K7SW#8yXNkV>SdquGtWo-%dRqXFr2pu`@5hCD-#lK(YO@l|*$FUo7E{ zRdXre1$(?xc^L+$AZRU5!wqvx+#N$9XZm6qNN~s= z%dK)bs(52yUJtS0OgU0=v>YkNT7MF*m*1_*C>ESRc|cfnH!K!w7JnDR*#2?!47nV& zzlCl2q&@zptWY8C@9YWsdm7q-n)XxDmI7@Ize%7Jdd8U6C+7>dvo2cO>{<;qZXLY- z*OUtDVjZAUqQ**5DA8MBiBbZ$dk_O|pAHq=g2Oa!N#_-An5a%dw(WWfcyuaNrQ})m z-LGo!I*9b+jP%P@2FU8)NPzbiz#9=3IrU9c8Fw1!W4K^~uupZx3Btd~9`{@uHcx9d zAAwCE0U%%+q{V|MCqB%e$2|>c1smcT2VAjaH_M)my>`xgOPa8J)37Y5S#~4KbLd(F zS5rz2E+QJ=IGpja;{l>8OC&l#)`+E2HPVI%aG$bF> zcRCd$?>UXGvs=$K%`CKXEJKD20Y)5X4E#se<*&55m*m$S_CodRIxd0kkJ0 zRHuppJob?=yZAOt#mDNAM^1zq1}w0y(ok&$0t$=rpqI~$mM0k=1lkY0tmhy(iP-xO z=6KQ2p3*sHB1dyj)5X#XrUA(WC1~;s6vRLTC5!`+Wi5w!*M4G|B7ehp2$1lS%9-t$(ZY_reH4i6Z*Y3Z*$;{ z&4BuR>%W^hs29gfq=Wi~F~oqYog9m5ufziCFBbuROBL#sU_zFs7JMN-#x6wfHspM! z@@y&m_Zb$;HH%STVa{jzv=leVV`!OWw}mDk0g+^Rz5bdYH|H}wDJk)l;CuUCl~T@m z#apkxj>+f~uvfaiT-XJ3Jz@ZJ!V3cCMAR4u%qyogm>XFd=NsAo^zD`aJNSm~pJ->D z#vm8B@^px>zE7}hyGwoLEVx%_os_FPqz&*mUCZ+ioUGM0!j3@~5-Tbr$DeT=6#d-8 zXeIe+=0P=bpo4l2vU~kWrnujzLP9k~5DWYWt8)F@j&h3pyy}aSyrhu#HsqkFe)`tG!rI2>Tks$28$|5bq7Sks=bEM{^+Y0BF9daC{$>q$DaGd{S5Bq6 zNSuWPBtM}#>O|rj5o)jzYNRI5EC}*+kQ*d6VFxozlMdo{Axt)e@tSZl2!ow93@;oK zB6PyS$DtZ~{UZ!fq$YZTM3s@|f}{yKjkd6c=^3O^ZsAOXSn>}8QJ{3sdKk20$@Nc) zC3DkAf1V^WSlP1Ea=H%q27hoGYa+Pb%l)ML8=j$#Fgq{4Q8@14<58^3i;VEilsK@t zfe^=a2gOjA+2m6oXwiT*ZlPm4_T{&OB<;;{n%yRqIn9 z{0HEb*MHdXAEWtiQ(V#dOmX-Jn{z1^6w<+d?`}AFX=<#ZF9KAsdQ?P37gC9;)|B{3 zDyvCas#p}Dj(=95F8Vy7wkMnb%x;g^HBAW>eyL5nV+)Y8AFmCfVv&Qx#Ug1M6T~nDATi;!CNgHyq>0({}YoJxtYzkr!d+q=c zyAx`gQ_ZrEYYUl%8t1mYXubrSz-4@J2q}?k4-sgTls#~Yyd9JlX2%Ut%92f$vL4){ zl+{uVf*yddeULLQFEj<+qeMhOD{5hX%kLP2OjA%{j-EVo;X!B@dtX#pwQ_Cm2>l1_ z;%68PYR)!f)gHR>`X4n}_19T_sxp9GBs*CJv3O`8FbADiL{taTK^MvhK4&^Hf@Ug4 zBk0**(2PK}qke*y*ze6fp3wToRw`Hjh${W(Gx{Gv9S%0Ku2OPQ|M{T*=g1#MrrZ(q zU&FYSU9yO0sG(h~%Xy8oaz$S}I_303+?;v{w(IqWoG;o|XEqB2%3!)9h9XYyCq=v$ zB#yzndkn>tEBcb)T)2BUw8W+>j5~nf-jt2l_YPM2l0yh!w4QbQ_z5-JzVfUd@oWT) zSRrl7u}%pC4DzxZHC%8}gt=0LZ%kT)S0P=5nR%rMc~lX4!}OgZ>OLQ92FeyM$nP`+sd>H~LSM5dY#3-@)uU}{|sd~^0&8dB7Y((S(FT_ zoKxK^V=1~bo+^o`RGfxkx+y=B%!>1qk$Jev7)JA67@G1&L4dl`0!G_6vhxXiTELl6 zeIx7GkrNBzC&a>cK?{)+FH5^q53UnRT>`8^mT^>)dkqeoOjhbl#xRrLK}$!J1k=d3 zRjm!E(qC|8&nzp8G9ESLhe4$5qB_YJI^-OUPC;6l)scQ$EnS5?WZU3kO@1t=Y*?@_ zvmOlEY7Xr(Lwj}$vksc9`)k^jqz$c8X)p?}|2LDik1pkEq*dCCM+|P^i9TYDcS5O- zHdFu7?T8OxyKdnoEnAHz!T+Z#8vh9|68{!Pva(vT=8#7g?*ad#(7ok3i^`9_ z>$A@hf0@)_&*rdPA<)XO7@}FkgGCU1`)vEcGkP@g7EB5Y$xQlRb^5wY-v}6})7UA-#g}n=7vUjzbLg`$aE-@v)~hjG2cujFXOv{hIZpF&Q^s zT#9~CD4xTyWys*!7f*s`+GUOBgM*1@iV<~mp zBwKFQhj(_8{v|_yuP)+w(oYs+R?E)6dWJxbHfi71X^$dpuvcoVI!kM$BphoJ4%7)Z zFd-2ie}{v3mi>}yv59!3P);_Kcj)4@1Eum9cOeGEyC(_aKLdEz|No&8FGT}#rYg{z z=$Uy-#6xDD@0c)Et)pfA0cuBCP;cXj4tX3Q0dMJ^A^~6gk7SG43eS|y%j*I9ed;bcXb>L%Hm6b)1q&Fp2aw1MCqzU zNreD%UteGIV=ySBEE1Bgmpyoz~M2b<*s ziA}g!>)TYz*AG&OecTBX#oBkY7_g5w`WkGCZYOaeLp(%-n5Ty z(w1^BkuxW|6+{Y#5-J85_MQ_MMn5Sq%m*3{3?+ZnF#JRT|5y7k;}f)hHMN2G5fyNL zKC8h=dzR%3*@06*3-~C+5(7B_ZCIYmaCoT)=4M-YeU>xm#!j-wY}`Qs_Z4RySA6Q@D8|C-q5D%jJKX~k+Be?y>u01o&Z1& zGTRPlWL~H0u5&22v6qR}L+xc4o&=Ivm}qqD<t;& z9sxKGDxd$PQ5i{{{r}iYziQfEx*{szjKVHG2b}7C2~G*KWPF_x3*h{9%CVPpn62fR z4aW+5DTX%0wU;RvMtJ>Y3~_l~x)UVU_EPw=vX{5IYI})-v%uAbhPH#wcot~kY2aVF zBSw4aEXdpofE;Ar-LH}9N!9)D_R{&jPk;MmJYJ+-~mhqLhdn;G)PI`34F zD^yw_22>_=5>z4qj)Tg$eHxVn>g<2Fm+SVg)~^bXLQcG@glB=N8K!>$6Ndr+Jk63W zGorqx1O!Yg{=qO{g&Ib*qYXO;<}OBs_xe{El8L%>KY=7T%JMt_cMM3Eqaot#uk+EO zitKd#+z@`M^BPCO<}n$m*Ya=*qJQM{3^?7qKo?;3x9{mJ*zzOBHB>CGpO+svuJLz} zsut}&6@$eE{uXRoJt5eBhQPtL+K(FB6DX34?f>*mQ`)!C@5Q^Lfv^5vjDGV5ybyqY z$NZ{ehf{kK;JrrPd$hc*A*HzT)6lr(34@!)_uG^wjUAr)hQtR9@fc0~IEkfda;|5; zoTHFsS5r+tfx1_K`qj{W4hjX-wG*xpW*`P5Y|PVE(mj(AI6&q6pn=+iT6z1Y*+OeYKYuWWhI~vNi2yFz{!PxRxRI>2j1KvG!kK*r(+9uM?n0 z-t13rlVcm|8}gAl@64bJ!+691!-J0t3~doOFl^ebVMwEr{!jkP*>61j9>K@iTh|v) zf=?Qz10f?F*MmJ-#E?^Iw<^hi&zWOzTs$3u^B=HdXY+hVcv`RjT{F?YLzn6v&<6Wi zo>$<>U{0sI7$^sQ&3xtrmzPnQGnN5w!y0*jNQTOUw8uSk2Dl7=wfUH1@Wo~_YJ(>>}2q|6N@;#{!~LX zSLgErsa(7Uh2WJ9nElw&eEN*3OJacE8v0i>{hq&|>tJlf4%`mpNYGkyc!M0c^(-bs_6RWz*4vK$POZ1)5^wGi!t3kS+hX74GA6dd zx0TVzvlR9pMHDtk$a%En9ArXO_ES3N*2r0b{?ngg3^kOmz8@rxF_hUZpi?lx2ZH^X z4Z+54V%D;#l$gJPX%}L?RW+5wywsz_{2!C=6Dl`gJ|DRSC#VBb1?wRj^0_NM&r2oY zLM}(#Uy&F>f~-GcfkX|IE6)=l^4#eELi=OQupQ0x$8hpyZM=|r-5!&(PQAX?g_Wln zQCE1qp#@lb1OhCDXlOTj!Wu2lGw^umlKt^G>Z>|fn?>*LCySPb#WBrd9axz4d%Zo* zx6v&7^$uby{PMF-R^=sv+^pX_S<;#J{N8>>rDPZGSxFhPw)zyE$AVbSgA zc2H5eC!`&IC|g(bJgD<-MK`@N(@TSC%3S%4_`2tRlm)cz=be3c5Lho^*3f}Ss>EkM`TGgo*T%?jjR82wmCI6JVnBGy4y>?)5+n4c9f*(0>dy zIdyN&R;{02fW_61C;|#G-;5lsgtqsc;So?Ho$qar;z<-T0P~lQLP~WaICD$^KTr~( zqV4$|z)nyK84e?nt?V1{yai1#;U0um1aJr2nr5osYO?#lusf&OWr3aQRl^^7ek`uA zp4U#pbOsQ5{T~_H8k#m4v;Yid{2*e0>AY&+CzE9Dw)f)qOQ-9A_ zu<-Smh2nKzY1*&Toh*$9fvX87H66H)c4YYu7#Uwuk^!#8hX7Zmg3Gs!Zh>tWny75} z|Bz4n@n4X|Z-xbNR@U%7S&RV-*CJOHQz(r9#EwO77h^hX$n%3d7%vui{x=~kw>A1V zKs*%0V}6e+lr@Q4Um-|dpT@c&e)IZI7%tx+Yb8X4iUI3wzX{eKs=h%5(>Ex%MPvOv zOB&kuydK_M^SxUNID!rw#8GHl>`_(x zBk%^W#9nQTCI0dm8ATdK12vjq-v&FVF$oCF;8Y_7l1;cbL>~w5?3Gw zOMJgI$gw^LAUPE__^n1~73yv-bR0_zPyIcQI%k9*&2#%ybDrn|V1s7;3|93z9V#|< ziPqw1x=Fbvp896XTN-ov3043LZ|p>X05ExI)U1bDjy9$o`<3*t*2=mZM}CoVoNXoL zc+ixi3d@0W(T4f3cRDW7+8&R^qWcol`b0*k0-O&RDEY48!Ujo;i4aGdpa_T$Ns1&Odt_90E)53*GWyF7hZ1E0rrQ+5lLue+Cnb zM*_=J7mWaqL0y4{!W|V=`-F1LHgbHaQLJNXL(5zENtCr3bFhxGo^DH! z4jTE-K~`1HD8vGNK>%fK+GLc)yDq)wH)(B+`G+k201G)9-%->#QPg6TRbBKIC;&3c zGZ79JW}81Dmm7cOW3u2)NKOZ_h(`O@1CZhnisshNk$N)rFWy z^7laQ)Vm!xm=zj_aYxyawyt+`63BtKxj6#-13Cr6Sj5cfRF`oC80%fWW|BMWY#FE( zr%oB#Oif#mw63WnNgg2!sWY{d9z_9DS@e^b$`SPR;fc;-kmZ=l9~s(ImO>dc6;q|( z|N0hH8tYfb9$&!X5?40FO*AM)NGvmgbhgVe-^^joA0S4?Tu@b~F#)Y= zXb0(x*X$P&0Mtu}p~c8q0#|q0NvdH(FDS{7`uxKFm?t>N12fnCEBR3MVSK*rh z*%i2N*I%jo3F;d90=n2gf!^5Qa2)hS$Uc5lAY<41N<%`{z|h7b?_gCidi&_#V&|MD zYXF177aEhUxz3{p@_=>$Z6(A2ZAQF6`yvo_pnYnChPDb7pUVdCS$`8y=~rbw!?D3+ zF31gaS9!5hKz9;PV^zAK>cJeQgX%Y}h-x<@@e@i;p!({MK-FnYs-SvUp6YPEKy^uc zF7Hz5L$c^;Sgh16#*oGPP>kzc$~%+#9aOtYY!^I8%tEe}`3*$D0lw6A(PKb~dtue7 zxMPs@#UXu6o8G z2K6*XT5#Eoz){cF>$G~t5sn)U>>s@rD#Cu_3`mXXh6M<&oBt5|U&WtsUN08YI=W)Y z)TW)_Kw$s)rVz)rKOT>wt$v*ERWrmT6A1OYrT|No*wEIl9{@GCcB2lY(2#IaPeHjv zB~oyjZ-|!bQePnvCp<9k32zDAZiC|u82~i@fEg|_w7YeNi%H9#1$E2)43A*qo%LpT zb6syXH6Y+2Yo_yoJUIL-mdl)?pJ;81$L)GhM3j917YU-Ubo}?Bg5a!AllNE1QE4j? zG3Dc!@n+2Jv(flIR%`mOm$V_5zf9G%KkRWe{SIQ#bU&m87aM`2>AhcRO}`5$jiwh+ zZcHdm_e*7@CUbeXQ2%CQ!ELU(I1H83@o)VR72R5#YgM{!)RIu|GR4hSfP3RNnGpdB#e zPe6iTihBi0ix?b?-98_8v_yeXm|gpj)>URRlKyH)@6^ICK+dK;Dks@y+m)MWT{Vm& z?H`8r4^SvQS5q-K&11V%d-RCt>LHLgx~lHiy84^Y+)yj+FK^;Fo%sjoEA@C3yDN1u z9O2DO;g&alT3*}~o?Hc|jW=!|HLQF>Y_XxGd{*;DxWUazoOZ>VmiWdMUu=h{A%Ak& zasm8=)2v{D$*2Zc_WUfn^^JJpc^7upI^ojwGGLwZkX6{uhIykwO2%dB?TYc`ePd&3 z39kTu&q`RJtd-u8<9Rl-MA+u|hjYS$r8XA`3*z>6tf9vH$KChjk4$7^KgEaa1&~3x z<9WhjIQ11p!$j$DytTJ3u1V)r@wh2{Xb~^QNxUdLbyGwpUJSd_D;I^kuW*GrPTn4f z!HG4XkuTOBTAT}|%Y;(l?XVK+ZEjMuM#XRqdmP6R6>!OVTrzNzch_^kch$R@AO+dz z1i6{Z5ro`9KVros3|_lsi1tqq2eX} z;~MwT(05dV)jR^%g$@~w|AxOP#skiGv}xQtD&<)+l4x>gMjKLh^%JFi2(UodAIpwl-O)*?H!m;1F>cH7 zw+ctUtw9lZ^jikB%cI|Lwe-t7@weap2Iot`P2ro&6#N8d^*i>2!FlJ!*aYI-{}2NB z41A$EzN4)24RD*92JF)TSR{ z={<5>7fsKVXsw9m2cJZYiOV3wzLho@6Cbbw?8Ll^q|~Lo5RCN%`eEH@?k6~8U4zvfj?E-2~gs-lq`cAdg#oZj^?U5YE8Qzn@QMIW*Y;)_xJBXnQ zoyXt?_GvfLZR*}nbesAvw7@pi$`?Aqo4!i7sotl&J@h=6wjcAM@AoUYtj7m)QD(E) zF`E^R*<3IO%AY;P8NyR`t;SL{I6e4M4TANFDUe?k14_=P3-s+)DbRzeK&4HA^0Ppg z9!!xD8;+E3ehBTbS;nR;8t+7#vx*&+*}_l^ZU1$A8d$f$oatpbve^1R<9_$gMozE%`wO9 zRIif{u{wmTaY#i8lS0#vW|pR#)8q$E1ie+N;=4qBc^;XUzrF!MonEV;}r zay%1cse)+r7PoecKn%4b&jV6BwjpqOoL{cgwPO@ObbFk+x0GE!v#2yI{mnT0 z?1^jcAdBZsm3o87c7|T0Ah-iim$thWC$L$>Sq@}SSuCTZE|eTjq|PJOIsxfWoY(DpDA#%kK}pbd0| z+SnBxzk75e?&p(jYdz=6wl&fx*J9pJ8~PkjDZpEhzO1Szn2F6boHE~I+F@zN*T;#P zd@99hQXTCGH_P z?1$B-V2#FP#`4TVts4+;Oh1NiENY(2ZFa4 z8^EiHINnC#K4cdxA9wMk;>H)(_cK)aWrgO~UgdWQGbzA5oA9grN=knG`l#}|@{P%F z9paK-;r}JS2z&ZzUA~RT0p(k$^6PH$TZ}Sz{n0wVDIxhKy7|ri;P&Nn&yU0RE^Z@( zjoJirW?4vaeG4!E_F6ga*SwfF^A;ERjtb zEoL_a$MF`1|N>>Xy;%FzCzv-_CYaX!*@wA~00 z4E;435sbhs&paC~{!~G{-CZ`Q{bA+4iMs ztU!@ze&h-I*)TiYzG>)JYx*l0ZdKiY7*u$^zL?BRXwq?dSC;}H3Me-mZcT1y&ps)N zZwc1Lz#3E=6;*HQUm4nYu=Dz#M4n2}9e8#PyMxL!%3cW8g%8Tze0E>x!Tr`I--}~1 z=!LHTcuaezJ(&KEtjviyh#roQS6cJe{*5a>a;|JxO5ZKvF$l#nI^h!pEQ5#QpOQ*uf?S)z(eLhj``nTZZ8O8!+->)mk)~4wwo5|cjo8cK!%4TNc*|nKUh~sUPJ?MRyBajc~$8=C3Wji&1 zzA~1Oue4j>D6#Ds_c!SCFG6=JatoLRi*#4yLt`m^uNaG*qaa!Hi2Goo_6L#-=Ql*_WM2br%=R+by{_rarV1cX zrb@;Q*Igvpej`jOL{M7Uy3Va47GhB3zwG*@d!B(lohq_z5gfX@?d){`gla9Fv>3Ks z*LBuReKz;ay>?m#Ef-jy<D}2`9hn2~0dr-O*VW z2oZb+#xF$8glG$~oneT+!&|REpCPJ3B7Jz~EA%*^ct%lh{bMVkC}1egLOi97SQSHe z;p$hSjbZFFrhE25DULQCTIilGlozgxrwfT+4PE~zoo^Y7B*VU(dc(d2|B1hK*FR#R zyIKE;2T{u#MSa|LU5o4fq4lSLiF z;w{bM8L$X8JAcqM@Qbjy@)@az@&b$`J6~l{3>P?M;Rk$sH9RswsSqouVn8h0FNk$y z*D>AmK7ex&EA*bW-yIaxK}>Lg5ntCVsqtBA(`+nir$@mCsP%{nGKba(GHk`uYq405 z84NUb1o{b%SZ4E zV>m$C%o&`vyk^ELhv|08?mBgy(6^F=(HqFpJ4+@3k zXJ5KVrXU6+@5Y1SYJy7u()2@T%!hzfYqR%`H@-p_(8HNxyv zBESJ&S`F|xTZ9&B-U6D`8hADUaSz%Hx0=nhNWa3+*VKi%2KwN0tYmC#*2yR|%l<(% zHiUivUSaE_q3xsdUQXIN7|wS|y^trlCniL@gJ6;nd~OK3>&!j^0Wd^;j71E{UBN^P zvh@H!2f3H#UQ(ACqt7q*!Pt#p63pXf4a!aWhe@eTYaIug3D!>um<&L>5@0ngg?9pL zV@z1Wfw5?RRVOSR%rwtPB_Cj2^aYB>GfrBePFS8kaNC$!-d@FEnlPEfT@3MlO}w1M zdWFp@8Lnp8V^ovN`K6|Kg&wn^Euv{VfmT6O1~EX?SV43efdf(JIoh}?P~(|6qYh11 zAvv8Z4)=iE!A40v`tVo)evBO``Mn)pV5*iY`p|(I!If%bU_Agg2^}cdK=YG?3=0a<)xbvWU^@sZUwa!B?Sk_xuJe5Oryay7z}W{Ox*6%5$6 z{2V6BixvQH5bEa1Vtu`GsMsdF<^3h2Dtf}ezD`EuM1GUTIb;7sPUJQP#4bcmoU3Ag zM^1c8#ZE>}Ohe4_tc72I@kgV1aE*V#Bues@k))26Bo8DBrqKA~V1)(f5Us4!@sY*x z0yyEL75>LpTYGwWaSSw?t}=T&$FS=NCdx6q`x)4|%guI4y4OR}>1>xEozigw#GvEn zt4cMv^FFQPq3>!PH-ay*G9#-wny3a%0C?b!4>MF%=#1FS`oC`7o=Go-HMLpgq%4o& z=28lW^LVRECp}CFk;VYhVaDPmNGCQ2rEU1sm3fScMcF%MLB{JR7As3G*>-Q03#Q37 zBNsDya~hek%hOE#j;~r-_&tz<;R*%iL2O?;Nh;|i+n9uh)yG`d28H(W8`|23R!39} zG7IqqGE8L zS5}GL%#8+^*nfb)@eQ#vwcUilO?-XaH%GG=Lyah1 zwJ50&K=$L-fT3ERc(e)V-F?I}VV#+b>f=ewn~EH3TBC_4lUVP^Ee6l#?8oh)+F$nS z$_Wgw81ev!6bvyc1{ltMEHJFDA~5Vg;K0yehKAuH1-yN)PW64I|7&gXrFFp#2DtwZ z_CQzkUMaa#s^}Z6oCgUYz6d|bJGpR;E8Iieex_O^sfzm!@io>*;AV39M$80ud^$e% zr+Z$%7h>XqSwe&dIB{7(tYm@D2CxuFSyZHE2a?Ant- zn%y5{*Xj1`5@i~^$t`$Q*1Y~AuR^!yHATK_LbQCq%>BTwu7gq~-BZjh=`pffE4_`| zlyoOjfnBt)E5irpb}7YfoM!ho*$w!0P6tYQpPAC9KlC?jWz3x92q_dJGBx(Z<4y+^jP6 z((DS6UBZ8{<5~x>d$#h;OL~`PC)e#z6Spbp6uCMS?3xR^VoaUOfHRA_!0|#M8BN`m zU4|-b{#&q1R_xGv*95y@PdMaL*zC`{4cX;@Ag@)TGvmTe@h$1$z3@9<#3AjU*tqRr zh&RgLpvmOb@r4OSDycF#T}iHrYlS*r15gM6LR?jiACA1-Vjeu z4?jm14xm>=*R2-T@_UKbDu~ZMo$R{1udy${Z!~ZJEtx%r(xWIZu=${zR`?iX;Pw9i zdOV|z&q6lRq@b~x|i_G7b>18&yi(lc^Aa!UX;QoLhvcJOXHJ*bwkLyugbbDVN!A?fs@zI6#@#MW#H%H z(;soXjk3Rd{g(J3ziRwmwkhbWLadhS^A^V+SZ|pi_}P)ium&S7pGX^}dbx@|!X9@E z`q2L1eZT7dU|B>VbC~@QrYN2MV4ys85b|+$l#(APEm>;%gS0Pa)|dUk&j5+nA98!o zXpQYoP_aMw^0 z(uk6vMHvbK7Ky=&8b;e zr)adkK_ULT+eqgp_g&$8<(CHqKMapKCSHOk!EY;O6&?JBl_h>5yJZV%;bI_C;7^z+ z@Y@HSSsoAk9)=a&N&psfCHkI`WTDzc@mReziw0z|5G**3N1+9JkXUV)x$#dkD)3{Dw)M*KEoxSZ2N^GS`GgK zTCe|%p{N52rG}Knt{NI51~tTCtPd^$1dbYpSy~OT@FllVgW%6~zqn);7j2|At$zSY z;2L1rCysH=-FP8-@M1<2D9|YXE=>^clD>+7I9ADos{i34l#KH}kEc@!)6gCOk-g}= zy5pMma@PQF>v8JxAx-=#iKQ;{D_>GT!9VFPfu#>Ju7y`#e|bY&Qqw*KS^xy=j6w{s zcq<4jyAe3B^qfprv?C|IZC<+yY{bVS^w~KIo!-aoTU;jG@@C)h4kp+4#eR=Z8|ph= zc8(|dZGOkQHl*C<_?R3h{|1IJ$VF#?pJiS=WV{q=|NH509rIs`2Z4PXX73#HA6|;3 zk2j?+q{K6uF#qo8Nr#w!5g3WrpI}&=*DTtT#n}+^AK~C1 zp++9uqPreo=GsF%okXia#E;%+1*_VHwn;0<)Gz!+c|$e>_hxGL)h$m;xG@;UI{0a~ zmaH1YN!36vx^N{2>h15;!wTr;V32Rp=Dg?DD4NcCo2HX(S1hP&)G?vp zM|Q{TegTE5QB_n7rt{|lF`c{1iRsKh;FwPIBweEdg!g|P54-cz2Y;k)#j$NiR5W`M zGoy}X=ayvAha2JQYT>#;q<~!Syw$#zM$JA1UyibkMF*JB@muQEWYUi^^iwqbQ>1tI zrErS+y`!pH)l*F*>+>MI^7_>{M(Xo=5Gia5AQnvF^f4AmOr0gVv*0q}7Q-9?$l>3C zQ3;mYSRSI+vvqCm5-l=xj%UNY&=Y5kqZWR7O;iA0&=S z-k6|O(g_vYZJABoVl3qQN|(J_1+5Sj6+DlrK}QAUO0Wd{pzhemM@j@#u6^?W;pZqi zD%Z;}1oM5RKEJQj@{YaCZQ$~urMa9m5OOoqJYA-fV8i~+D7bRyZzH_D>l2#}W#%hP z=HKbe=aEYHZ&v)Nlox-87&iMiT?NHehBm*ZZ3S9|ViClEVoGU2aRUMe#n$6Bip6MM z*L`TYu6^Kr44*wbO z9(e?PaXRKK`(f3@9R)nH0dVD}o* z+8zk0mQx&reA0g8atYc9eP%vv3=4!2ppRVUNRA^g{WKy2f0&LMu?l~fwi>Yj8#^Kp z&g8z1EpbsYR!tkRKW#*@EpcUNX!!9E{%rfs9H~oPz5az@j`GyP7B`e<9!B)YWS>VD zs}aDFVz8}1*_BnCn=P>@yG1HM=T4S6YIr(!6Q%8$36|T=s4WsnVE8>QAtX*1vvNNQ zzSH7IzKBD-7vj0btfRO@ICLXM&ubzsX4iNsf*jifI{#@kF6K@OY#CG`p2#H|zH7 z)Si}^K*_sCO)-S>O47VOC9gKlrjhF)*QO&8wO>f)Q&)kAy@{XJaB~m#?99a9G!$k8 z{zf4_34de!oX6=4aGsUY^u>Cf{`_xO!%V1t5f%D^Q=}DA)V35S$qwv=Ue*?*2 zFxX{b%g|1IHpoGIKjMcZzNNX%(0=`jVBI-By)D-Lhp&nk%ZPN ztk?^_di;T}o(-`wADo;HSXDfUXG_2r0jG^PSA;(2O=d{*v%=F<8BnP{`+v;6d3;pW z`S_oZM25`?2pSc2)U=I8D;m_qNX>+a+zAs5qKF$talxG>L2v~pXvWJ}Y_V#qtyWu? z+7Gm~O4Ql}kX=Ls!4=%ui9W<#+pL^i*BX--+1^2wE*LYHV?);UH z&p-B&4m}%msKN(r0swjE`}CZp^Wku_$c;THf-b{!oK+AE2H#yN6)g@bn&VeAo{G%) ziIN(0-a>=bp=SIP{G81zqSu15ZIm$h>1|TrXZ5cPeg-IhZUap&e)`<(<7X{kcEb-g zG1{ij51`2x^6uj+O!(K;AWhC+W|);}@b7q5v@N>ZN87c%1!D2ryKLCQhxjPq>^P4C z44rEsUFA1=f$u>DVrgX^yqE9e=Nu_G|Gxl=0qW~{nt~--Nio*0EL4`5+TUl1t{Z%o_y;8LSb|CB zbR#!z^Z?JWMwalv6?^c*?^nVP{&~?b#GSm5jyQk$rOy!KLxP1v0(d_&P$J?7lF~DV zV2FR-ByyB#>LojYnwL|FJ?Pb6D^-T}2!v_DM5*nFpte)|-uI`rK$yZDUM4E)BMPHz z1ap1$JTtv(9(wWI?ocCy)P30F9@1? zzz6VJDVr@cS1=(FB30-5LUWnUQL;7*t)Y+zO}+|E0>(gSLMk+J=`Bu!b+ho(7=6i8dx zSCQ5W+_~~{=XE~P#-Mb50BN4Q)Mv@d>HoF7OiB(rY$^C;4A;SE01t}GYTt{c>_39CzwukWnzEfUJUu^rfx-E3GxrcPYWiv^o3+>JWhpW2l}@irm@>)t za|zTJTM9mpIqQwrLVS>~>senbf~MzD-0=AucX}kdpA;lJsE?9tEI%&E-fZ$o_Mo7I z<8ld5h}-8D#_g#^ar^xX%p1;<$bRP;49f4~s)aPSFM%|^eKn0IV9Igs!}03U=F}e2 zQ%=P2F#Ng!5ZdIC^3L;o6gJW!L3GYCmhNR>coubpm31JaqQ~g>=(Pl#59C$%W54HJ z>1;+{^FIdZYs`?^0M_zW2`HZjWzX?Jd4#ft(Jv(h)_V3} zCSyO&>;q(r0?1koJmBK}JLuK*An*=Y&mPoq{M@RS7s!#$E=4WB9|uU`fSE*42#0g z0~8HeQ}7vn+lz1a&@h`6Xt-uCMZ?eeanZ2k8XpZQp_sZE{tJHg)>m%$hhY{3{c4hm zn7J$vxQH3{jUc8x=-XNY5{TJ#8+|(&PI(B{vriW9j|9Kfz4m;lQDRS_N-xr|F$UC8M{UyTys#=ESZD<5f7|zkt zEP(pfx*u+M(MO68Y51dKL@%U+hJ{CH4sPjc_8NZHq#_+`^;h?z-+fiI?tf5*xKh1P zJUg^zHBt8N*!}`ZW!HD{!MnQc&Mslsg#iGEnqC6HNSf`C*giOR8<>YFuEN6`KIZ5Y zF-Pi-=~8!0mLMPKgMYP@UP8V0pf%_^7`^gypV4I@@u_jr^{f$bxZm<1DJzS&o=&_s zVTSsJor*cxSb3&4EeDkZ8!Ia`8~2w6+4kZUaqE)J-&pzaNU3d=bA`z!hQl+wVj+oa zE(08Ca1I(KB|SPEM+%-y<|$7;$&ag2CST#p-3e$HPwtR7H6RRu-1*}Vo)`U-R*0nM z!Duvu9=BA;JEc|`&QO2h@loU@1 zT5QlqwW_$|oHIT2Go_iFrt``iX&y`o>dr9u2*#Ogq2=ttG1=$}m*`@MPCcEqy8m^W16>J!D z;f-0IOc$FJ6YYP^R9-qQS9z&{AD5SYG|}g!kFOM7>g)j@nZS;#Fc9$VHOMh=-Ojoe z4%=hCVHf>(_Ey>ZN+6kUj4Rc_hbe8r-SonR|5B(+Z&&fgsHXD)Ucsd!2a)D&#JTD+ zqnve~p(5~a=Nsiu`#R}Jl3*&wsl$Kia`~<=gv<3rA%noQX#*~Q<`x)DmfO%t=Uz^O z?D5XM?s|LBuTVUQ^v11$l`)2qG}UVzj?v3vuJ5N2 zhYtm15Ae%2NZBJj*(UF+5HH%Q9Av;9;M02|d3gt#zPefIeU_p3%z)mf44Ad<4*q^bo zJ$vkkF@mWqHNMama{)diVm%?ue&H&W)#r&*gD>)cy?DeZu~@j0M!;_d4%zyjqM6gANS{p9`ui z`_(wNDG-7eac=&b-gzWJ6b}ng(*KNQ3CujCo;pT7X64zulO3jCYe(wL~ja%b(gJBqBZ&Mz|cK#Nabk zlas0mITxQOq9gY8jDvvbOd8+WMP0;wcw5n-s-D%r?C6mcBjb8o1xG6FP-~wk#_cz zliDoHwx|kP=*IseO_bWxKB_PK zDykcVWA=M+eZ}#+<`WUb`qZcV!W%hGf83kkRLq+fwn9Gegqz^3>qXQn$zkYq!wir! zo5zad$1&;f+4?39F)paO(JByrQ15)YJJiqs)$r#ACU&NsNq+?}T@%sG>u* zZYYW6DviH9p~58WwPt{1$-2KQS4bk3hT0h)0B%+JiQ(oqNw94`e7mSkjHbh654}`y ziMC-$0n?4=>4q_+&5S0wGt;ymxto4Q?w*>vRv;4p#IyPQ=luS6qG)v8Q=HJx;Y5tK z%Ib<+<2BSspj|<4CZO7fn7xX&I9Qg@g_q2Q4>v19!YQ4kr3-HPR_dq^>ew~i6N4Bj zt^rIJlrE7xr;-OYAl~XX@+d2zZnGKiZzwU^6h`@M}_yH zfYGK6vJZNqm_U=n1d=t*dhfaqUC%5$bFAzKd=M>H6|g1Thn7H)YN zXAv8w1K9Tlp&gf?i#5Awezxg~{sHq<&SQ0=j$U@QvlP;aV%bYx69+Pbbh;1&ODGwk6-GgFb zYfZCAOtK?4+$^5i>*RBqcetGshV_SNo8h|4$r(Z!3Xtn)AE{M_@=`64w8Ya1R>-7bps^S*+DKD6NC`bVIk^Lq5ib>}nj zZIYSgQu=JJ|LWwhcx28V^ssm$?NyeX|GkXhkcdK1XGR<@xhkmbk}~@S^*#XTrV}4c zp#?$zninKbzaYzb^C`a}OvOZSN{ z!4LQ?2Q(ayNj{Ro@ipDL;m6hTPvyg4`!qwuj^8&TT(cb8!SLFdTYv z*icaJ1;1Py<){G`xRVqtusl5O9 zkjGp9*j*m4Z@@{;$?=Opj# zLEg!J-o=vlL21(62-gFwaUM#Lx5m$V4|zMU6S^bV>%<=@vee7@C#So6v+2^GO7V11 ze3YRMqFN-y%e|B1*GAJ;p7T^nXRWeJ;V;^uyMmJCe#st`G!%aF8l~`#&y>QWKKCiy zd8RMq3qaTp3b{vIPu|5|Vujx|mJoKXtP^=ZEd%>vEY(Z+)?gu;Yo1Dbtn&vT9t4OT zW62?Mpz1+Fg4AAmUL(AnvKpfq=ae0VdxT7z%U3Te}bYM_8ma-ERYy>-g1@Fy#IC- zlZyGHyue_t=Y=cZTgMaT{b!A{H1{B>6nH-kG2(9Dnmyfwsob{4H zDIE4|DX@P~X*qD49^85*)&JPNrSOtxB;P(kKFg=VugPc7UqT8kd*7!jgDd%Qsc_lp zz6>5UHcJNmX^S3DtweFhTW(X)8J8=EuPf6;-K!&=W#4?q9-w)Kofx3&Yi6$tH;cGM zle=;oWrOt7p8~|+l9eex=58&%eDgw~ovUmgQ3{s$-r$<8q_@zV553J|Mpph@i zGlgK7F5aj%kqd^rL0&Vt@+^~`A0FrNS}+UB9_Ifq4)VSB=_bmHG$Qn(028_Ny5lq4 zA^%~ZzPrkZ^4m`$%7t?9ykG!hO*%BLY#jq;^*y-rx{!q9?0$PwB}l*aNkRHFuPL%( zNSt&z=vG4_WKxmN2~we~uE_~A^<1GT99<$WMN!xr#EKH9&bB4qZdDBulK6A z3+|B$b!6Kx(WB|ZCM7g2`9x_NA^SCt)WMzZ$+fkoX=z{)>?-^oZBMddU zvE>$Bs#@k;mY)#Vi}`W2OztQ{;Pl8I&WqA#f2{ujJ>!i4mVIjjIJ8pHB9ap&X%m)K z&M zlAo>A2XSLm_>gAEMDLOx`9iL+_$#r3uTj_pVnf5yi?w9TOVpD{hb#A|`DkL!2b>&w3yLp*G z_nZa4+xU0#$kTX(UBlH1ZGw~V{H5{qz-7**wcv$-`2x%Srn4XpV}&wH6ZZRD4O98z zlzVE#wZ1PjizdJC8E)3aMSHd(Vu+)h-DZ9wWuP$?&((_MdIMAOz^sbjn&4JkO2t$M78-vXK7`7{mVH6Ui9HH# z)BR>^dmEigXbc>Q(7tUi?X0v?<(-FEc3N?ftjKAEt$M}LWc-?!sS2>y)GLwtv+#ds z&qS^!pgW#$OyBNDD^^7>aIeA1W)|gd*EJEb>{$l$tTvPjxJL``4DkT|CyO&rzAw8?V3U7V>NZn(V{v&P44!<6+6gAWfgevJCTkRY)fUhUH{~a+HmIks=@Zo=FPYjW^2XqfTGh z%!@-73=*5gVoKkH1_fR*$^S~@(biLfavuV3$4FhG>34hwkfEza*6+S3o;2r*dn)ON zJzducmCP^ToXaxe+EH7f1hGh2U}^=vYB7Qv3?!E8Xui zJTJ~uQrwmOpOZO}niz^QL3AbmfiW`Dso0OM!68&`+54dhME0G@r;5;j)zsH=WjR&U zNf_{WSoaqGz7U0QZ`6KKdfJfX`=+OQ;SjTiepdqna_H3kTxRtLje>sB)Hbd==#YLCSEnCOSqwD1Isu@*@bi5&6+Er)wugfbORXMQdbw}0BUN`+G znMLtrdnlf43st^4<=r~_+z?eYEfEA3%924*G(9|bJX3b>xvXvcVjWY)?LUzE18}tWxnkhxb2Knso%#wi?Kbwz>U!Ocg4ig+$js&XzOnGr47`hqqR+|r7i+0@?>YeX{r8>)J2;5o|No9YN-YXs+=lZ&91^t`v#8oKB9}z8y|YM`QAg1C3*ce!wwpZyJ|kn#;Cup^kXjp zMyDUE&y{S0E+HE)BI1PoIq*Au1m%Rn$f6iT^z6f&%1Oq5~HS=Fe>1E zTMi50TmlblyPy=1qpkP)xIG2HJ!`2aX~pf>YK1z1l`%;)cthRn4;n)fV0P7SQP{Si zRPfHs(8IhdmcdXj3NR8q!izkuP+n9-^)JM(AClDCH2i&ic9T58{$+<{h= z(7Y0;7i}#JYMEf@gjW3rtwJ%h>PS)$JpVl<_}l!rTJ?@9Xld@xI4vg5ozA`SH|g~g z8yq=fK`@6Vyb6z@hUr036Wie73)9W>_n7I0AkChPCi)6Hj_4g|bQD);qhISvms4}r zd?vmuKbP^6HRB;%8_O8yRBFpO#;j8VISqFLf1#Q}gOlg;Ca|lZ$AlJrt{aX9{EJxy&@) zY(3xTp%^C zoaIVQ+Aus1)9r}Me^xNnh_7n)vTRra2Gcsx9R1zI5#&+VNE&#zS! z&)8(<3H2hLtoYzRqc8OAAk(uK7;fE<}|a?8TH#skaN)!8d+?fe7cGRIpKvW|%L?~1d+ z-t@6VACa+XWB0)Y6q@}DKGLMdcyf9uN1ceLnNBrqR&sU7kLlE@kcaD4S-m#oI){#U zojT0x)Z}4GE@NWgEFcCd$BvgfI}eg8v+`7VV0;)ldjD6QIaa6Fvp%utG*?yEGss!O z``8Sh*OX<<1V)Cp4v@_Cya3@swZOED*seU&6We(v6@fOb?`Ss;qZ_D^S5G$3%~~Q2 z$@EVB?fuTh<>GubM+{Okm#vE~6tx&FecQR_qybDW6pBQ; z_-(`8IA_RMXCrw7oKrQ&*edn9;?(=?U_ot-S1QgJZ04fVJ^schAB`)?87Mt;=O_Q7 z=0s|G$R6d+PBN!@vy*W1Y?Y~S%s#aomqcIBW3#$KCMjE4-8KMPcBjQWg(Z*ku}siF z3?2*{bK~}%!o5)&-@B#&pyf?rq)|}T;6pAN797b!h2Tss92Wc#573=9_u%@HSR(hL92K*LUA;CNFMB~LGjk`Y`KK8 zgV<8>zcgJU7~1HZIZH5f4BY@j^>1~<(4SBDFmxTMXzMF3hB}9Uo)4f+wDqL`J)I^u zY~JLTJH@0#s4f;-rU$*m4?=Zwg$h;2VL+3mp97;sHFa`WU5nZS+;qF$yGQy&UV-c! z=Nw#@u0OcvQW>xZaomtgt6%%TEE&z@Kc51$I9%^QM1OhoUH^@;#3<|2PpVf&l&xr<1PXYZ{GwzwX&x(AR<5KI>QB5OQnnR8u7IE1*&|_NU~Rnd zYAb!?(s6OSERnuyX_u9bVeyu3Cx2}^e+^@5k9cxxA?ru^RdKt9`OdaNEAKRnh7pCB z3oA#3=by2<-pcFXVUa#qiM%awV%%=}r)A#|kvm%+RG6MDUdS)g!|8=_J8zU7ThlNq zFMoL*<*-DKGBfMOy02J&B42J}pSsG}#;)+Kv*6AID}Y;g8@G8?XD8Nm#nXLr6P4{v zeJEQIPsP@CbskZd%Exve9I zVDA0JUn1)qf-$0@a|PYAm#AU;yb@YaYfx-ZE;oQrl*}}=TmUTn-b(eCr?otR=TyaR zr1VCAv2XnK2+4>uJ{1+cl#c{hvL3OUy@07~CG2P+=CM5*Fykf8+57gEP-yXSPNNiT zH{U2J(St7Aoj*$0?vG&6u-$!i0oz?jnzs>W!BE3?N$zkJ?vSfo)!t;Y;tU|%!+u*>Z++68JfaG3rB$46WkpUE0={7 zC~oCOS?TEBL2KEp29PHI^b`4qSSfwyJ&Pj+r%Dg;oOnJJS>wo z6bU(Q|Esj)cHdl?C`jj-Rto}}Hd{7D1eqbN*tOdIZhQ8qc&^WuP8fSnmb7=iro_Ag z#m0z}lpmu*7AAz47y47~)guF9&LgeFeE&F)nAn__JyO<`QnuihVRJ84NH~u%=et{^ z-u~X-uzSbdFqtMPBykqh`Xkl(|CEs`>!Pylrd@{314wnqVlN}r3viU@@*6NkMk-ky zQn$G6NDbnXaeMU+y)w*MP7R$i#1Du|T=5Z3JwS!v@Bv)GU)#=D(y6rEfPW_6umqoBvXYH~&3eM;Arn z?kIkEqtI&)6vyp)Ld~opvb4&4YmpVMUaQNjRtkSe6{?azK^2Y^Jk46SwM>yOX?(zi z@gpdoTVGN&%3_mqOcB_mm=xP9;3Ahj zuA~xoR=cO%-3j}4X9|GsrCjW)dQTsyFFDW!H<1Udo*-Gdw@*uG$DY*okMeav+lZiJ zEu^($8-JP&Q@Y>Rf|zCMFa~elLXxtvFaPXOa{c{jl1Bbd^&MB|KPS>#PqQuuZ8vR+ z{FCau0$R31OHcR*U4!tul7AP`krT}kLWlcFfuJ85*8V9!ErkN zTE<)FG6Rf^u!8}Hc7_=aTN-fK;Q(bqo;2piY}l9Kk| zZ0Ac8vn5U5C%l&~ILtITCTMaFO+;HmAjUMggl4=Z=Xg!NdDMSwQt2HbLV?NH>v)bE zbT0PJkpps;4Hp|nm?OC);tvu(3BC$hQB1ZYwTtSb3nLVD?xmK5J)VG{Wfl=o z_Dj*sm=Hu2fAYnKq0cw;Qog1W7|7QGc(nBfS~i{977IG{3~Am*oW0ymi3{o9c^VB~ zRSRYsiaacX-*IBxJxBuydvv|9FRHTHgdqwD@?gRoMf>v5LRQI!&TTH9={`iFnbWa@ zmyGLfE=asay_IPw@!K=cL%e5CzAD*$uufJbm&s2@CYQpP|_~+!Uj|H!P=g#o^S4UF&H;GjMw|@s|{~7}r7Z}KZ z@r`QwH;n$xg2-O~HeBvcb44Y5sHW>Ef&WxLsU+gs&3P z;LuYEYH&#JpkKu6pi9r-8Tn-CCl|xBH-D$+vAPm%e=ij4vz*sGVtv-~R9MsJmBu5Z z*no|L_5ZykPO<5*%%B%v9*8qOGw2{p^Ul39gZ{9B-@P(}9#}U}YlQlAGl<#{4J6|M(!7m0pOggRE`4z2HMnnL5s^Afp*HEc;<7IB zKfF-#ynl~yZhsq<#Y^PVt@EV|$hZyad|wZ=JWqE9IC3)JKY zFJ@g_(IwP9UQn$ugZe6p?R~fuD-70sSNQE8XnG0@KVGHuy#k{KM18@J zOJC2|;zMSwot<0wN$80uhCBLv;_GgEg*0Ye(@LA)0C!U7f@w$aoC|*H`#_;qxsXx3T;)>rM2X-eURTl$l45kj55f{_roK?NJFWwTViGmr zE0Q&D#GJ4`Idnpryqn)z`>7A4xmx@A52OY03|=cf?gb+yQ`6ZjA#*_sotl2IzurW% zUpKCvWU)+RrRJ?}0rVMo&jFZ1vITP{Z?yY;dq(Yz(sCp-ulO;(#`C_5m+r7PH9f3w z$*OUEJhMcBOH^(N-#S73)2zucz~gVCgjFghgb4k@vtPxA)9}3NS5&`_T&pB}UVaP- z&xTCV){IZW%TEqSIE=Ki{hY&_I>>O^c|BYl$C~ zKyQAO9QNj9fBb!kBuLG1wx8rH_y^{Tf{%Ia+XX{R2a$?%Xj|=zT)5R6f6u(g=nWZs zC-0K7@u2KnzwAW%i%(q`Az;V3<{(viShK90CG7SZ`|FzYWx2w+vv9fE*O!a@Oe0yjY)VgAoPVRU zxBa!Qw`~$DR_8=kh1xXX+2@ndvMthw6@azrpAc5BSQk~)K z6^F!JQoKPhLwqTG=5p5NlN2o=_fS%x<@=YsEi4~YXq?CK;K@bMk%#WeM3bvC9q(y9 z^NJhbCv`Tja^AT~8{wA2jeP99FFhg?OP`vFPgsjvn9VipkvwC46->BV3Rx9B!n5xq zV>Gi|S9D8P&v-vRp{+*byP5a+cM}yCFR2R8mMe4PrAuaP@3=}*49~HOA>rl=c#uI% zA5P~1EKsK$hLN!E6Q5^IMOnC6u4=&NYOwNWRW_8 z9*NWe_wx9J6*U9^xBOXhrlYx1dpvLP+(bnLZ6z<0P9zQjKJB=JC*ctL69$yWAh8)> zqE#}$Ea175NNGI1O8i$uk{=`~uiE&{+&z;HL1|e^*%!V>j3uYhAsyv7zxtzUh=!uJ ziD9qy;adA7o=xqN5|l~x4Vr#cT9`@FAHJl8W*CDzErS+rmlhI2U6iH*hPJPV63U4l zv^DmZ)`W~T;VW00=TX3oUu)L(WQk@6kPOPsQWzWl(YHIB{`C6^ryk3Bcm#&PL5?It8o-LQ{BvNzM zPonBksd@Rjkeyi>PdcHx(zj+TBywl^p>r*%&8nEArNgt$4Nay{ymbAHEj9KwJ#AZH zxdJD6qI69n?`2?-ppTqO=O%0fV3d8oLKByD^u<=GDPLRKY#Oax9-e)$V$d{GYd0&< z3hZJ_K4%1>9&oE?JTAZuHvDNiW>v=4BR3nhkLzbR9yL4yWa=Yw98S394XKHh3nj8k z?9AyA`6P!e$WjhVsX3LK!n5Uf&w7SP)>`*dw_57;(}1i}Le)KZ!H_i71FnRj_E{|q zbz;C!)hBq$srh(SP8TvKF-m*{FBp67<%O%9(u0k1+7rxjL_#+0*sFAS9ZVUDo-aQ} zG+v`J4v3;*o>7JS^6Xhi?+jHKBF@on{Tk{2uJ9aO-o4W_6izLbPsBs5l+N5AbY?P@ zcHFf~d-}M3oEvmqi+dHOLOd>knsjn^sMd=Qyx?2kGInan*kb27+$uw-6l z5W0vivcos{m@6NjFdt{@#|l1nq~(pf+G7To#yr}P6JccRirP!qhCGBrVTfj%$}-#g zKNx~fyofAO%K07=JruRKC+uBHc9qz9#q}^YE0pey-BsX;c;oB7$=D7AlBLNhhCsh$ znhAlPmmfo*3W&hX5Ws2()KnG_Xc%eUMw~eZr~!h}q0L{wxiu0;B755aN37Bo{e7x< zF2C&8n0+q%nXn%fQ<~*{waJ$$K65!e%5C}uQ$1XfV^Y}UD~~fxc1ZXyrq80w{QcwN z%RZMn7)e(z)o$#NAJdJ0QA4zKFDf_P_#f)=_-vY6r}N_eyD}Z8QE@!^ZEn+rc(l>P zjO6SL&(u#64%5xO)DHn^<`UD)ok25UZRSwFnGspdguQ0QOEZoN=yK;!-L(}?>Go^Z z!A;G5OwB`snqQ}MwDl)`&0CKRI`DVWN{4)@nK1I4${pd`WNTKO||JcL#0Xy z-gt4}0X!#$f-OG6a`JKkFFS`zfc`sv^Gh|fGClM~sgO*xU7MaaRO)kPY9%}<%L0Z} zawRV%tjE+$bHtx+zR@SSZ2d3cbT?@6Iw$PFxjkO=N&W>`iMC$iW9+#h0mhP~6=Uo6 z+m+#7Xeezqb`qv9#GWq1epq^)F1W|IUZ$G{9+Dr!Q-^5-ll%tG$!g#*uYt)lAij{% zvXSGdv(-|oq1JXiJnJzEyS1%0CE)e}CIz>@b&LU%{_R|>{X2~2VfzEA6^maF0!`jh zlG`*gdCRz*#+u3<(>U32WJ8_3v}~PIHb`8pFIwrd8X(X6#*16T%g#Z>Phw(y)F7F4 zKAFpF*SQ-{9a2a5@X?h!rXFnB%Kun92res7Ou9(K^ER-S-PuPKzDjlpXFxFf5i!@p z-sL5;9})N6w1PM(B)(0zkS#2XL;9EctAQT1tIEr^qNeJIi?%-}I+Cf#Qcx9beIr2av#6W4I#(6}ctX=Z=$U(9<~i zZsX1vY3!v;M+DZ^S^R#IKyOyjR8~ADQ*K0HPH|x`@T&5_l(0(Q908Eovv?$(vi?MV zPO-(K&gF@$4(9UH8$_Th?$^)LhD-Ek&>nPsOPVTMClS0~6M*{~AtCyf=Hlr@L5?%- zOLPWtu$88YDXXl~=e6nZ>^cTWomGF*uE;r`A7t}V`7y}JgCfz^PkrR z^Mrwl9N{NHhs^Im0t9#G&q5A{FE;Par8tB?lLn%#d2FxHX3YF~iBzZa=Q~X*(!pt0 z8N_1Nl5Q0Cy=USdy~>L!PwLh2F!OsXWrUmMz&?9ciBjpt;wscc^Lo^TJtI`PESy|H zGV?O?DkAY%3}u^|HDS9oVKei~7Fj*;;v*p314HP1l0A9D?TvFYFEeGyx00Q?(+hal zv<>FUFVizgY6`6hFPCm4B6)dTx-Xu-#Ez28BIkOrPpQl*BGU<3oBihWQWW@(5?m@V zt+(kw%kJUUODqeayHY*DFK>lrYyqCsbg^P^g5rwMn09)@stzH`mbe#*RMriu>CtRy zUacBpbD@Oj{jb>;YBs7dYeX6sul{B>!YAxkVil>Qz+PirtRfx?-zEWOiS&Ik7!ihn z6v4Oi+!$@$>bV%_U@b+?2~GL1rlK}9?Z|lg7m_c0$V>nrdk?ogOOEH0&~C4Z53~V4 zGQ%*5%M_N~-dG)DXA5$Yj+QXhY#d_EYOj*cFh@;x-W;D+#grmuO??QuwZvh1U2 zk4WF5m9Iz-Ky+1z!#YuPpIhm<2K*jYa(YQl)83rk2sF{ktb${|M><)qBPKw10!WhZ!gatV~4K%>0^aDaxg!p@=$(PO@ zX%EaxRKDDFi1f~`mF{!MWH$`NQ#qY^@eH40 z=Dp-H43t?TA#^8F@lD6E{gCPJnt18!W`0_&949IO4lf*;BDUYIjbQYdTd7{E`KI#mDK?p(&2M7a5M&~CRHb@*g)Imlt{r=W61Xi6R`5&u;BTF%Bw z4IvvJF0pWHS1ATpB+}=W*k9M##Eezph6z}$d7x58S#97RtTqr*TV$<#cfzX9*n3Gdt=u4DZT@(1pb1z6!%0yMuorfyW)A#EdCM| z#tG!dl;G8>o6rubP%zWSBy4tU^pzbOaeE+6aNAJ`GJ1;Ec8S10ffmC7*vhpirrdf>CtIEkN9yG zhs0%&gC&n)b@o5ic32mf-52~wY===;me>;5bnh1jw!@!D^ETq_>LbUJb(UB}^ji{n zImF5=lUo;%My!Kz5p<_~EELM2~`y1H2Bv8XKK8(m5B#e@TsdEI?{~AOk(f(Rk`Pl^%ysU&$pL zp$DO7;r|e`%5r{p4nzjR2BGwsQ2J(LLhom~=>#Tp6J3e64leL0z4Nf3o9(2vn}u|< zY^x;`O=>som2Gw7PmM3^ zM9XM$TnTeb25YEppQgOAx3g4yKOpO1UPFvqC@;Il0B`n{it?ozF$soF1}D+hlR$*a z02i3v!~Gu~s(4^VGvA~lo!77)&vz?*(DTOr9`w9$j=wqNuYQpmNnw~bsog9RRQ@;c3#!3I#Nb3~4NimArxd zTa1Iq+Opw#;)vqf^>F9lub=HwM4B2-dk6rdCDIrZK>Ujvf(0)!Adn zcUr``$JG;i!xCl64#-?5JDgneJYwWn8hf*8fLP)YHtc)y$ZJ2N;-JKcyPiYiLE=b)zQREUz$UOcvAOWsx z4P0wE<moty}P8k#tBpCP}tix0@sxZ|x$L;cDNi|# zRjjBVX4yYCIq#B}ncyCbtd43W&RpehlRGubROq3^2n$Ft^zqkE;dglQY<6SF<|H;N z3-6v2_y-bp>^q2zo*VdmJa0Loano2av~Y5b47o3A5y>aO=V0Vg(qZU7pr_}$^&kn zXw21tfF#igAXd64T7%BL<7V?-)zLED6f(htgjuklxum4yT!xA)m1M?muA#@qsM5}= zAp((#UHxY4kKWEY!9ZS@*oA~!8g#!){Dr4BK(Rb2u7d_t-`RI5vQNrsER#r31ym-f z7hCB&ibP1nc5epd5F$LE>27;OW=))@q#Duwg>*ues6q76n4%1g>wNfXRSorajsrgS z_sCSI%`;cum33gL!w1NVACzrgZhT}vlJG)~GlaDVyg(8k`SzU>gD_d<|3-mLI#87X z!{o0IAbvs48;Rxz@e6lH{jfCI8Nc^JDL#$j-p=?QjS@O}4GJjQIz6a1KG>`FVXCDm zM&Dyep{&+DtxELpXZ^*oNx6)M?$&^z=%gTYQZ!}<_i|*Ezun!(342AdJg4z!LHV=N zK|%SMR@^pf#yH2I1*3gqe&R&eh10Ly*^%QNEZwNRUodF0}3%AK3?c+ z`CImv(dQ%34u&D&wxU4GFD9*8en3vPmUmAVm`G1V)uSj0X*-kcY)g#;z~fhfGQSt~ zg=>Yo8!~x36YkN+OLt|ZcY0~XHOG#L;cE8@KiyMvYjQ+xR=P)4I?qcBD6x3DUO*wf zLfik}8rQ0;cWM>VIL}Q>!%vSeAe)AJc@6NscUIbTMU4ndkKPJ($62r%ll!$|cA|sE ziC&uM4FSHaNy6N(QWQ^(ky``2Xd1jslaPA}zJW{K{Z=TfLV1Jy)-kUMKV6?2ZGln7 z=v&=yH-@)XkCxu@GEUauWuCmGe5Nf<*`+zT-JZ}eS>OJzJ&i2n5F0ojeVyV^TIo@; zJwKir$vqx%+kCCyYxg%N?eV74etGt59?nge^i`qYCQenPu zf=={#w18&eWKHQ9N|k^UE$3HCIj;zi5Ow<+t3?U>rEL9TSEFARMQT&A{2`2eTg1U9 zZq2&9B?(~M9JRl%oA8EjAC0PLd}~Pfw#kyi?u?~R*3I4V_74lpe7Gj>BN?AdRpRFr zN9EUkJ7K@*_8)mbN2)Noo#Rbc7N9||`LOffec1JbA9$(a-Wg2|8(M;Er}kV1&} z7AphmrXM+%GYEcUC1dM{WbnAoUT4{h!m)2OqdIiRjILf^K4T*cQAvbP$mIx6U%~=7 zkJ{Aetu2k^%SO%WEmmIjCUHaxbCd%>(;f4hwToSb4jobT4Oz;p&Na-fQeKqB3&>&_mE0*>Ij2W{h)9V&5ph1`R!8um3mP1>j zxum1*pXgCmQMR03n1^yD4pVY@2Pd+Uh<^Ry@pUpvgy9Bbn*%LPfW#T%pwK~bLn!i)ew z%o}?jj8_VBK3&|-7YDb++)47<`&gAPPvMZWQ!x$nRXl?zOk)mCXviVJ2I9=T%^RlN z5j|ay%jvL~3yc1&=E8Fed~>0`znBYUy*zWFtuQbb=8;wkzW?=3jg)=pKM#P6e|4#m z5eBrPF>YMVOmWbfQ*}5PF;-C$ZXU)ft{`KrXzG@D>V{2qsR?WAQd5bjm|PxBkI%sM zdnM8hA&%8dKPb~miR3;}mecGFb@prB@X2lgw2oOBl9h*;96N-Y7;m>FqJ^@A-*iCP zYEAei`cGcUj`?u&Eu<0^SAAO*zRgh%iMby{2C?(Pvt^B~Ha&C)KMX@eXiy@qhcm_P zcbw3U{&b9cYNUhiSDD=)QfMQ2>wNfSe}$c;zzub*KC}K46cMm9%h3+(;Ewg~ikEI~ z+TsB#r@k+wh6M3i3*ow*r>)cX*Czl1WN5EvUGKp`iWJ)6bdOH>gS-H*M z3dE+5w;u|3D01IywCI21aLJoKP1T;{7fnV5%sr3bU>4DOtE*J&tzZ!#8jfxADwd3^ ztKy|E&)6t*DT@>Tg5KK@x0lY9H5uKoE@NP60$MQdL*LbIikB{{P0cYhd0oadPTIhM z^+@qVIwC>cll1qg;_3SPr0z(NNQefW<+#dOOJ+)n5GmOjp=P^fvy>+xe?sXj%F2Zh ziHa*Djo08cQvJsZR(Y8l!^|LQIPh{vHi}U`h|d|#B^Z_OL*zk_FApX1AQX^?h(iR8 znBXmP&hzdx>byYa4*jx7(pjg1?NyEZy4)hGXD*MZKP$xy-;mZyn%p6e@>yher=j85 z`w1y`M@hQikpEE$-}1Oh`0G#l5`Im;?h^h^-$26uN?N;?`{hmorC-{k;%@e>=t@A( z_IGeT+dryx6BI6wO>#clXR;6KkGuo9InD|e6x?a%w3#wbJh->PCVsMB045mcPDjAd z^V#0mv6I<(Mh{LItD!!J2l@N&O)mF*wlz+;tlKQK<#DO~N8}D>p|UZ@0TBd%1+^^p zTRq5ho>}Oh`sgfl$zzJU2ma*GLO=c-#b-`Gng7*J0_-~v6raw(gy;bd_EUGIlMi3~ z(4DF}oyx+X;E%DC`JrB??9LCxU&PTTU7Rnb?3^cD36kqY*-nxfm}gJ0a<$9K+b7(j zG2)d2zn{vvgaet9;daa8Q{@@oqr>RC}nm(hvMyX;KkJmFij61H*s|_DG8H=jpJn}ayaih%x;Z?*-ihbcvrIX<3vJVzOEXXJ zs{Cfkvy2EhouTG=#+k}lvFU!j=Bc#a@khm0sR%8L0_^NBffb{z3&0Z3>|;5(7f4Rv zD(SbENk#C+jMDK?!0(K@1}#*iAsPE*=q2MZ*;w*gNw2MLW@n1OegEn zTK*@|bXBHu`P99v zT>P2kvC1}<25r_y37TP2PH0dQCuWRh!+0CBqoy@MEfCH-7RG0A*YFg_V5uOL!+8Oj zt+W>549K^YtjN9YL=#yQ<+$B#TAgoSkoQyi>2{gJ>a@#EpO{?46}hx4Yhzo@-EExj zBkNtp*!64_p&uy2TVBdCx}XYXrAmQ={s!x5556lx>aM40XIPpt<8841BSQs3_PLBf zrwG3GvMPp$rZ7qDX;tK1ca*F3!_6t$LG!aTJ2vF}-KhMC1XfM;R6bsG(FygUVZAjT z_IIO?P~w04y|VZ`;dH~|H|-^A=^%F0m=XIA`GFB}HEG^PoaZ}L+wW1Y-;PHvh87`G z7QSTniT#RD>LmCfRuL_qcC4I9`Fi*9FE4K2A?|gd{}Hj?Itc_En{NdXawcV_@uWET2NN03&wIpN(eIETeuQfJBdSCNH1>+8M~9quA(iLx-Q?Em z(NxKK)b8XwQ0NiRbM59m>4`ty)wLdzaAc-*I^6s;a1y_Q+_#B%R~UOWZAG&Ck0dxc z^P7PhI6=90h+x8<0@YNuPkAeDpI#NUYs)3vwaQt!vKJII6Kc`3=b~q~%MPh@!DSE^ zQkOiWq<&l|Zb`dVSr=oY}Rl39ZXkc<#{}( zTxccH)+y||@4QgF5Vdl>-lsy>?j=4<`^Og+PL%% zA<9z$GLAMKrTZfQ%3!j;Nku@?3lAugGV}!#upTfmCOV!I=N#_s@w|e$rSRgRRAit0`S#;wiB?#9gx#JHWwR zQ{5%@_*qq|tn8PZ1DEzn1EuOjs8aXp>)%K}_U$`@49UZ#QVwXbN;U_kiST=ma*HhCHU4 zdz)TlXOvtT0>M}L?Ou(1m43zfD!+Svl`hIx`Q80%EZ2#rV!i2d_ZI27yx+d(`!0Ep z8PYv}zW7XgYnN};t$qfD-id39&qQj0NKA*EhY+(bD<=j4J_#rLfW&TW9zD`8Bd^$fQj#GlJuH+!F2gG!>Jxmid`w&sf6jBo#xhnUn9!&7#>@tG`_Gu{r_JvQ^W!2h7} zez?nyU;@?CI)YFvP}hrnQX%IrRO^|R?)m0J7_cI4^d!Ik|F-w!O27XjdvD8zAGG)M zcA#jz%yF3Em9zc6U3kADIxgJ&3@LR#sJZ)Nyx0wxN@~!Fhfr*#F4dP`Z<3h$Smp5Y zDQpTHL#|cM0szLb#wmdX$20oF7_Z%!{jui;Su_ z`th#k5=T9}oY?FJ?yZn3g^J%1Pd6ES>m4gnLrn-#%Qn}Hz2fTW--n;V;x?jb+EeVXmm^OyZ0J+>*Q z3fI{<^>%+gbiT#`UxDhdGa6c4)>I zg=1K2nlAp~x%x1$Lw6N2G3@Iw*7hcFtDc!R4=Do9?Y`UiEfQyfCCfxjj6mDvGB~V5 zy+(y-W5T&3nV$4FWku@pj@p)4%!+8`UQ=e#d=F{OY|-?EJipMH<6E?q7t!|sJqpfG`OE&o7SIQPoh?-0fI-Dn0sV(@* zS0n5XY|`Uxj=7gXC1~WHQ_GAKLb<2V&hF+8&&hFg531A$bm&ZYN0-7 zM<53aNUI!V-r9*dzEwj`#PW?;71wbfbIo`1un%$0h2Z1!kQjp-o#ToHB{xSbdpv6d z{r;w}Uvm!B*IcsoX48k_#%A&4#g5O7afqa_K32opn8b&SIg_cN#c(W=#M&iMda^3D zO@;!3O<5arxWFTI!xmk!a(%{Zp_Bm}r8`tCUQQx$9tY<_7jb3AUi!5~A8Z$K^z zoQ600=Xme!J_!~-5}Y9+9z|jp}!EZZmv#BVME?6_A&hxBzZ-0>{5w&xitAg3j&GPyhl^Uq54 zJrX(|Sx9WyktmECj|_DbQLWLce&@0ks_GZ@&(a0Cw#;)IFrzTG2*!0JM({P<;QSUh zFIInlbu|hOFe&(vNJX{O66taHgIdJev?eSD>WgGeFOgPN@am{^qNpoT%9UGZgdA_Z z+^%8(`WW=$^G=EHp&Ei!Z9#qP6X79(_XOiH-o-Y-64Drqy+{rrx`8spG``@++umPX zPrKUJWok0}&}DU4qvmvJWK8$?DGRZX_u#oYmYIJnM)D5Y^n%VD9T2nGCq$_SZ}3<;&1(c9iceA zg>_j9?7KEmy4?34aOin2{Lie@ccA^)G50BM+Ojt}9*Ob2Y%3RO_6X0BBYGrqfG~na zA`6fQ;#ap=9CKZ|9X8R{OSdITKf=wOJ}vZ$lWDWJ6Z?bs9L3({-TXab0N{XZXGkKH zHatJJ@AgFAazb4CZr81myy-`3la*vI*UOeYlsT8%MHY0pNXww8rtAc%Ue5tTfDM5G_BAr-gsl&?!hKKz%<1K%V&(Y0;m(Jy`K=xXK_57C=<`Ng?rhanT4I9TwSIu~n z`w}OtOt>2k>5IkQF|g8dvZ#~^i=vh5rw&WVj!DNGE^pN>8Fw}aZ*<7U(cO4S z`*RZSYbuvajln%udf>|*k48(|Vs@KHPmiBuQlSo*Js65vN7L&bL&?<^ZAPgj>HGh8 zrRKH&|AL>4KJ#?h|ISQ9i-m#U9kJhzQAYCgn#V{Ur*&*#x!z*=8~^P*MXQ+9avKN} zcco70ayg+xb=46TG#~~<@vcc3!>m*Y10}d&NUu&Ek7<%Mn32@+I~dfuk7;uLS~`#;a(Qx@?Ql~5v$QK4}dY;xS`}0{J{(qQz z^Z2TY_ez>cyZ$)H_^h}5XOyD4w*Q#$z_3n7{5z33tyz9w>X`Bk6^v9voPZu{X7s9ZoJXJgd4WqMZ5CA{5rH@z8jUN%y!zd{5pza8bfn~tjj*-p+}>jj|DMxU42U-yi&my` z!#iX??2uaAT%5{m?2wozp-OsJORNasz|dTf?(mUa#c< zO!1PoPeQN$n6~f!-#oRHLjbT4L~ZM#>k_bkfi z6ec?W9o|lT#Y-PD5%Bf92qbYcnV2UYIIDNHc#9n1AWx?3S+B3hp%+A%99$PEbIw$y zgYIJ^0E%Xq-tn{(6qOb5r`G|kRNLTT1U;gdPzwZ1?(bjBrucKL_JBH<# zS?cy#CdJvwtG$Oc5VI7viVfXY>bwuQ({O2>9pc2G=vjG=4Zx4N0r)W+fIHiSR2PSz zJ28b#j_Wd=wD&?m#VzgRWgLmzS7U9tAm>U}r290PI7a;V`FKlRHPZ0aj16*9&0S*2 zRPHY_4{A7Y!gMq)ACIpnjX|s9@fA3SNQW_t0!SM1$sJu07LUp^+4`07852-fS!-^~ z*vY*1{KImCgF&-~Q4=aEH zH#;$EvV4&(>1}Hf-gOlz74?P?<_2C}*IPqOIx%Rq?ohVq)Md5bf0L@jB6AQx{7r>1J@agb;9sz zD7g0FAT`}Z0l|Y!2GN=pBEP?F@uHtq6R4;%tRSoqPv~I$q{^T=&Eo2cpTlvjT|;xa z9V1!uE-JpR{$PUKh__<3f{CPwyhGHV8XVm()8d#jpBR6|UQGpq06QPaJ>E(Jq*^i~ zeCl$;u>4l0`mFE&8vYl#OM2)n1wo&I+$%}c_Q5@#H4Fy`H+k6sD}UaGpVBA4`!oEp z)DY@%K{%q2R*^hu08|_3sP31+vRkUMy&RMpmrJHK%xNJ=!=rNQkHJb(6j1na3xG67 zIz_z4)t*IycT;s3F9mL6(S9f@?h&-xngU$OmDs9y%6Us@PHrC&mDHYWgV3gKAF zuT*PUy50V%cH6X+Z%TwPQ&>n7h>2F{zTT4z2!8AKi{HBa5=oz51OS|uq$^i*!CNKw z8S-vw1>=!Kkog&Alu2o=ws=ZQq{8ke z(AsU;=%d>lFO>Xv7fq4?c(39jhD#A{Fp0)BkT~hxCDn#~x2>iTJ?nA{rN!!U^ZQYW z<_M{lZcHW8UahGzdy^a2D(|bBRTbVcjYB$G+^2D%3F`Nc3>=^eS%n>%u5>T!ITZTD zz-nH%)6uv~+--^Uyw$vtAOBkWeWJH-zxTPz{N7Dv@d?_t`AuF%<>h*yIaUL~`yUx7 zbx_-f6^2Kwa+nmN5qy+lSdhm>cWijDg1+|qg)#8D`x*w+W6wXY3#~U7*WKAeX zLF*E#F$%Hu?u^~F4mza>)ye=icf8Qi7nTrsoS-MEoO4-)* zo>bdsNji5vwn^^Telo>I*7f-zH{>S?+el43YP?m&i+Eo>B6q-cZS0dl1?znTK0}2j z)|yUby*yX|?81vzB-stGXcPHCBPi|gfLHn4)E+41-l^eU&G!{t2Dz3CBx}B%D9Lq{ zcbqWIn|R<~wZ)62&Ltkp81WSeCeyWcHS{;%1tSWC8HgatSPJrjr;%AZ;RB?e$OH5vgl+2FaK zo+X-ksL;cz;Gx`~-b~wr%Rhru2W#EC z+I3RUf7R%fp2~CR({wOt-j@5>XWYgndzx%)xZo>n(nP^m>ew#XIg-Z$|H8>&5N=q1 zs;R=Jmk**fs?AqEL-arItHhj+)4#cXYXCyzneDE z!mk}#Cxe#Bn^(Ht5IL_)*uqIXhb=sWw6)M%RLFhL;`~F`Lq6|S2sH^fdmozlZq`kZd!)GGFD8}|3QPM|`N{dk6)!CnC;L(&~E zcz+ehu>R$Bz1^kCTh72*EXxgV%??l;Fn^9#Bjwl#Fm6kDhw?2%Agt3#Pjny$F#IF` zP8L2NP3TsE&y~)gu7DID2(ER3sV)Vd2D(%^w0pV1q+<^X;MX?eY50X)VyrClEA8Y$ zQ^3FAP~${#&iRud0gyuTb?8xE=o)$)QS^wv>epy*m4$|P8vv+CUB{Ji3a`%lBvn~F zWuH{#;Iio~JTd>6icdN7uJO!{p2M%j@{cD!FT5cv#S~Aq!#)#PM{1Xt`%l>|-h~)9 zKh&QP+dpAUqGCFyyscT$COlQ;4-Nm;HBGov3~blanAlKF{U13r4Ce|RgT@D?rwh2L7~gbWHd z)*FSNLr9_UR!`}u*UPQVo0&owT#?EiP9uv_eToSu(5v}NYa1&`aM%#$zJZJMt`!l; za(Hge$BD)lc~Pt7dKSCbZL54gaA)z3Ew^^GTi__*g4I;M-|+9|t3tRULI(v}UEyCt zHsC^p!OTpjR@B1LvX;pf@ZwOGX}P8uQcMsjT8_6#$s&o{M-Mu2)15>KT9wS-9us4V zYd9LW#HAp{i*Gt&G=HH@Yu?l6(ABIo?^3yC=y(KOffMV}6biXG|uUDOffJkcPh&l3@LII`_j<4AvrK3#(| zBu{{0GXGDS;krtS=TKjC39$E1cmjQ~thITWk#eQ?Qs7;OXFG3-{CBmookgCgO>y17 zf$`LK-xSAxiGizaxYhPaRNFgm#I@Z(nq1}nn#ZlS7VMa=cYlgf*4wPj6v#5p{1vMNJ*5J7s6hnLitd0xKfEKW_9fa@&{%f%>n5*LCpHuA?&@alKf{?1?Dx9QWm+ zHruZj_t9)WRL(`u&f<-3fa9(1Qp)b8l&ot52|bVqaZ9a2uu`Qb2n|M{{}UQ6_9ort@e8aHM??Fv#iN~fN?Va zTwL>It`-{FDXMw(3Ttw?1>?niuNy-!!`6Mh?=mFnQSsOc6ZOBz-WSX@Z{j@<)r|Rx z#&J5GD!-%|(-ed5z1KX(zILwVPMpjZS|+JHwk$EPgHk}RV}qJowtfNpKTu)@{&E2r zIJLatptLtks71sDCuw|pu`*r{>l1Jc0Y zAS{zd*QkBXohUK?ds~s!YdR3iyHUj^7b1pWQL!k|DMa-zR*FSkfBH8PBPPL{mN)Je zAg=N+a@&kGbz)F{q<>IQp4m#<`WK0zl%HHs|HWbXK0*00{`X(2{$Z%LW3f|SoH{P; zji^X_obqX1>|b*iAHjSEz!QyjiAo#xZA9JCx@yAi40NL62C5HD4RR=_snG-oG8Z+^ zVAj!=`;W)PQ{LGXA&7*=x*PS)!AyBSHdj*EBx4z{Ofu#!R_eb>#My|J1bs!ld z{+nc+9+U_qV^nwWK2%~wEB~jG(bERi#C)M_s|^8XqKbb>GS)JcH&owd)ybCo{c&Hl zm5iwTFG8OZ7f)yJ8uj*VNbzdg!Au9J z@V?%{HJz*Fbd-gVA^Ka?nV$SUAXy1WXmL_>X#o*e%J{v%>yX7NKG6>OTX89Y3xncw zexg`1f16l6S}gltuzD}VMLTYJt=Vc}sh|@wISB>_>Mm3Qr6JzOS}7@kDPOi_wUmgJ z|0PyG$6#!tvP)2Yq`!Yq-dMd~L7OKtLn%2UlY4@Y0?7^6T2+Z+wqA-jbFIU@)RXEDepzTb}P#)U2!jlKC$M{d;Op=1)@L&nK!`akxtS)slZv@={B3t9DVdrrXK<(@M^;>69p`ZU*cq`-_ey8CwPJ_6s;~oC0(R9b z^EaR%y)Qyx3f6ufZ5f%&k5pZoCG+FdZ03KbL^Hmxo(*Z6aVh3|wWOGPTzs<4Ju1l@ ztk2HB*XAD@)b0|>g)<Y?=Y?zdp{K% z?+7!a6{*@x#PeyOM%`*E_SQEi$&>cBuH&3(W+MGAT{)~QF>e6EWcyhxK59NcPtA;K z*`#`{nJW%NLoaql{IRa~hO?7-qJ$SUpHCY!4bivi5VPMKW0xYPj*p4-E0$O1>%}5BTqGL*WyAj_G?`@TH>3$-YKGa7 z_LeQUC!t1B^VS6= z)D$6Ep0r&n;sY=9JG=Ha%s?fIpnV)yZEhEAMV0%XFs2z%4m2zS%64PA71=fag|Zmw z5}cbuApZ^wos_@%C!(x+cQC)vga@K7xVDg}_mCfobXAYoUX3%W&RhSK5$NuP5edTrp|e-rj1{Z3NQxs=w?8 zN85uVI=C(WO{gSMyQ!rOd54R+DYR7h`d{eK<+Xi zvf3Iy5@~)eiGBa-sH#~+qEVx?925Tl?In$u(s+|b6GaqWi;0O)`M-$C6_T+)){x zvE+eD>Qph;Xe)W2C689}W=o!-eSJe z&Bg(F-eaPyb0*4@*Fg(LE;CX7{wuVYOXV^n{>(1}a&l~BM%?p6Xht03QkW6#Nnu8u zaDtf;-|`cf5y@M%&D+>p%^+SDOGOh?tm|%c;Xnuk*!9!#wm@C(isto$<|g9Wa9}3$ zH&Z{)L|l97aq6|D5V7fZ zX3wk1g}gvpukfE^`J62uHS`&}B}lw}tTnVpjKqP~&>;~LyA>d@=Fy;`hXhD$X-H7J zzwv1a0ZaWh90%giyt!e}h7hm*Gs0mdwSvP4hr?z1al}@g$^5Pn4u7HEkPxSbI4r!u z5#my-BO*j~%7s2DB*Z*3Y6=MPge|NZA@XzUr#+-U^ZDnj8ZstpM+Aw=Q+ z>l^_Z&3cqHo)w|c&h%3A9ILqv{ZsEm6sV)k(Es_Pi~=3v>aU{;_SPd^)1{#L)sF-e z*f*%Y)T)o@k5T-y9;pRl2oU$k{_gA+m| z?qP>Qt|qgrp|TApJ>dk=&C~)xZ23dTjY47g8=*JPnV(Ia*1~fCxxZSwY-&Y^0%j(H z`ISlsFn=5s!aTvjJRpL(m%*G9!#pR3sZI*aeIuCH7r?wbg4w~rTS!+0-SbH*YAQ#oA>vHMmApa#7*@-&NI;C<)Rg{PJx#GDtbhj89?HjoJ`5EVz z+fuA(RCYDCf;TjmB{Pd!8(yu+{#{V)=8G}ZLp3$tr)KO8>GPq@y3!{v;+~G)tu@(O zg*$09~HH{h+OIH-yejXEvKZsoJ?f@y_}HNEPg3oTKeV>eO;#UwMv6ke`|#I zR=6H)7X?`Zbg1N_`Q(xxASZP7CGCwVZ}=xdqY6<%JJw3=uxl5(Ih7~JPiS*Mfo@QN z9j)5bXG{H=RBzK6cPW`ft(!T4qr~%D1O8!;W5}(NK>1 zPO<^rU9FQ6V%{L16KN&%u0d;+VGNzCfv|&(c6Wem`)VD+8flFnG>gik*~SI zwH9G_JXf${&cID2Pk!hfKF|BYDdq^5N&K zlniTsDRY%@aud%NzkYSoDUxPH&JsUH6vyR9k+6oAi*X76nMZ)aWi{P6pVd;5E*``v z9zj8*PGupn=4thie+Eo>W`O&7^;} z#Ak^)7CV_%RO|eiP;WlMb%1}5uz4>dL?9KjU|#c4Og1t|8WdAWB_aE!(Cmmo#*uf-V}_S}IVri=wA(Q@gt=ivOB^xM+d@%cHUN^tY>7tg5L%*(5*ynk=De=NPo;@@ z=8Bx_dPT?Y36=7^*Lcw*A!Rdv7*I5=3OAl=?^s&6%;{q=e^C1{*-N@ew)LwM{hV_Q z4$?`_HFmig8aKQq0_wOHu`;YDNK2@+sEh7y?5(1T3c>Mx>5CnIG(!m&`_d(J=Vbm+ z?!5>kVah!r6EC?NCJr?vwZflyqT~HUJ?BaD>zBhjko3%Ll zH5|*;_JJp;z5Z|U~sYu zq&}AR8haPMYbiD=+S%);hBz>b4Vrv8F}e?N*Xna{-)T^?1GRYgaTjkvmM zOS!eBelPY4^(faj<8FI{)OA6Jkh{x6?y?g$l8RldTs7<9{;KWdxVCoC3moY++o?A5 zlgM=~W)a4p_P*HS$b{Xl;!9F0ep+07IE7sC)pt@{p?)M$sV1A3@r34KH^@dll!g?S zh$Av-gI#1x?dg<>^r2I5too<=JYy#!t}{K8>hoE;&ughZUn6v`EH_KE*^ezjuiEsF z2>!L%#xe`e<&|^~-s)91X#orkmrrkc*_)ER3SO1lt6(DO>PmSf-55MEYzX#}tts<~ zNU`Pms7y`ST3&Q7d_lTqtN2ApS*@2=2BQh1L7Io}hKf#E3c#aWsnf%5JpkB5!7m($Ur|>sSAGhAU*YC`A?%xjImlf%`@SF1Mc~g78 zOyf~ESu*bY4fo%?i?ViC{&EuM?S_M!O7G;AgvfvKwhfxYNd~8HqYG-Dh-I*oYZ@Ya z!0pH&cRAA{f7PWMT6Jcuh@FeR%YVdrv)L!d@JnwWBnXRR2oEpJwBNCN_-(z4T1JrPA(80{Pj@Gx^C4Yws_E_{h<(d3jflJ|?NSjh8WMaG~NBi3L;W zzwazw&7Ny>=e4}ra`EbJmgv5Cb+68A2^*lRR7+pq+w97|>HxW(RYC!zSUEo!M3eb9 zP$xnALDJ9y;jKvXw%q^i#m*$W=vh^q9EWc>RrQopvyK+a@nIpqLL+-y??oiJWb~vS z>CUUkwiSV3dV!UD+8VS-F%4tj@9`Z>4Q^x4;XGkFgjj_{23h=B@i(b^Q@oX<0s9m@ z^XmWjG!QdZ3D%H`FKY>J$Yy}G?EiwQO z)t8u64xq{2O7Ew5Kkk1I@5^=#FZO>SoZq0byO_N%r25o!;l{#6s#7+A<_-=2Vt8ku z*W6&v8aL+gXucI7C4$&~K;kcKjl87q<$f==AUGA7XnhVFv~qc3_V+(=_%!8pJ^L1| zla!{$cbA$<9vbl2EhbF68D83+QZ*@LU|=#nuCNC7`?KZ;cE4k${9(jg9q|=M(QOg) zBbCJYP7WCu+eVG#n47m{=Ck$$Xs0>YQT3xhr=uCUr*AGXF^O9Qnoibs6i(JP?z_oJ z8hGCi9H-?0`)uqZUW@|LF8DF@chW@un zY!KMl-N%v{z_j@7hMvKeh8+@#uT=^514EN-SN~fak3oxzXi7(+3zGSxx&0#G&-QRd zhXw6q6Ngvqe0J96_iNZbrx?b4>3Q%cnLjQr_FmMQ&a?qW_NubQHuT8LH<-jV!P!9L z^_UyTT3bY2yt0DvO$REO(@$0ywngxmfV)XuZNExa(0I)?1NYXM=7wZb>G9!!clXnF z-`hUmc5<0$x0TIxgB(@6 zZqRal$o<<%LAlM{2+3*z}~OgYc0PF zEAwCdwtW%2Is!Uenlz2@D9(BGEE|6u4qJdjGXJqd8FK7N{h{*T#$i?NPonP5SOIcR zA4bHlH~e*L4Yb6d4>*GHSO>aJh&#{=D4xuJ5`ljCx)}6Rq{&t8cMY1A8`jVNs#f$e z**f$w!?%M2x?VpHXjcRJSp@W<0-&ddKo@3=R)q-vHXXr!CU3CsMf*XE>tfjZ8Em;c zP#3f~x&U_95cXIHn>DVoaIa#Sf9F<`IPP-Im3|9Y$WiB@9~wh{9SF(1=28y&_t(bc zc!0E#U=bL{gI%(}dm(;)=2?x$eZ&l2SlI zN!(2e(jD(Clwyhmy28Kcmvx0&Ondt^wNvM7N_G>;81=Q(I-n_O$nsbY38L{TXn$xw z>wqkWw&k3VshZ9SE=ciJQWsnWX7ohf88tGj+#{Vbqx)^%J#768;EX>p^*XMu`77$A znF5)rmsSF;VfUue#Uu6jFozmP@#-;AfZ6!8MY7sgYtKk9J~bE@$1ny7 zh8Be}w=Jlsci;rHH{;tZGI0uRj;wBPhlBO<3G*72&(dRhS+%ELmfp(3lRSm1YrL1$ z_B!v`l=mj&tk8~xe}JtAB(jfE1Pn@Tu((wZt`(O-C`Ym{t_GI{g52{))Ux-ozp(T_ za_>tto~0%5T^;B@`zb_vIPgkZ9c9673tdq=mDr{fhqDyuIaL z=KtdniRvFDJkRW_D$b6pD3y%=M3+nIj#|L4#EkN9-NJ-h{iB<~YlutS8X?tANG+Eh zkm&eN?O`fHg}$j&HRr~K?vD!HbtQ$a3JQ(!YdM}0^u)OkI+>pt7dkg8G*yL8QXwR2 zE<{5?vr@Ip-*!QS=!(5n=S6YxPEql46@QE7rM+dzg;1qPF(O}<&sjMSMgG*W?Oh|> z`mKetCqvp>WWAu=A4KE`&`i*_M-hu;{u1hGnPOekTF)fDq9!-=r<&X#A~LJfJaDdc zdleaLMdZ8La+qla_D%Y+NbcXeyMg7*^`G^T*fzr$ceqGZANTpzNHd=0d^L)B?jh08 zxg#xdl5zfDLb3e~DO7IL!6vqsRYoee;e070vFLH?Q)-O=!N={2n)?f`H@II@v3TD! zIMBHY)i8KK(_m9;I=(ESYpT{-lCoE%!u6QIHu-IfwN#A1sP=8 zEhsaTvlQscI(qxJergc7qO)|s!Gds848e;a{0Rsx6J(ArgUs#xA2&DVtJK4^HkrR8 zF7+dM9f3|zsXhBIYM!L8+*4;+Pmi;9iYgfE3ZVi%yWD6wbD+_(29gHD*Pb)w7dWiK zOO_#n9i`dk*gCT;Hl2= zyA;u4e#6DiFOdXpk-zp&HYAIg*KZaL$-l9hWeGbH9Zzj7r)R3FIF)VV@qQ$%{H(v5 zFfbnPsBioMsXvff9$`Fw4Ys-QbeX?5abDjVf%BV@!$jc7~?_4!Y;9V7MN{aWwt50q{Ex6Z z9ldw_`tHV)lR$6-T#y~*>V8~5PIzX4UNWy3c#bmlAQ=izU(zN#GiOLzH}Bu-pLi&Z z*6gqC6pfFl8rG(hxsk>GecYZNgo)5xvW~iw`5KKPuFkuF+j5-UD(Vziq!hODSMO{% zOWi$bda8X@0&^=3b!YYeUX>mS87 zMldl5*}8~l*Q#vQ=>YYYP7}+FQ4<|XDGxtiw(GS}FovZPTGG+q??bW5Z8qNtCdl{~ zFdK(k?ccDo6!0+(Vt+Z>)Wv2(E7x;+j9m|i&V5b}8IbE2A_HRnrGX3`1Va5=o9}>E zguSS67h$9bhxCH~`3I)9RmVjbRy6&V1FVkjaUIK{Kr;VkRL64QhH}~%*716S)iFr& z5R43d;@=VQ5(j*U1HM2%P9UE)YQ7x-Kf3_<9wG2)4mjX!_sxwRi_45}Lj9@T4E0ah z5`sKqV(57wB=he@(6=an{z5JwPcH|39=Y0#jjZ~LXOcgOSIEm;*A)=m--R`<0uJ>@ zKA_$y4uv=M;~Kb*YLfYnA{1ux98&M_5QV?g8}&*Ig{`@mz-)^V>@Q-nsD*v^w-%N< z@Y7=OHyZe_BJlebfdBN8poPD2;BS|b)DfVdH~+E4db8l4f1km>eSf3iRx$j;4E}c! z{1w0q(VrK>Updv#A7bza_v=rjNQ@4oeDDA~m)PN5)~4IaAW_#Px|9FIX2wN@CR)F% z;jpvA;m`VUB>aYYlKIvMhhYUc6o)t*2@cImG-$MPhj(Dt?C>#tjDdm`cgElMyBa-@ zpsu6#v%3DVpS5@QxULqfs|+!wZaUQaxPOK1y^1t%Ij=QkLuunoOa>w^=`U{7I_P;T<*WN$m%?X zI`Y^OtMZ{h--#{UsC}&k#Fn8)2daWj(!I;eTE8Ka#qA!=`Y2}Hi%C~ zpul&m#?t>y^4>)cgXi&z5 zP>68`bl22rf?xw`KbI<};cM6>qq%cwKZDiTXS=3~@m$VL0~d5BTOUV}AwU*- zF;jI0r-kW=Up=m=R>Rb$(qnjGP;b7=#DAjXGni$GD%8)mUqU!m?M-G!j{ack z$g#6ai5&YNUy$Qbejvxh-$M?fSK8RbC^acGHV$WqPnFAUtDDvz!6tZaSex>OGx6z? zS-)BR9+~xJ^}A%&cdXwbv%XV(ugv<=`tFupV_gi7yoos$7?8Qf_WM{@AA>5;7RqHmJ(c$DpDsi2TrxJ-~3!Tx3iu9yq0ndT3hUGt)S&&aNQYP zq2XsR9U4yMDIon#;77D9@NuO7d`_(LZzIjya{rsNjMy#L7-dvhWD&HL2Sh>sZYB
2dzK*)zj;p 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 z%0A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Ê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%0Dc1*lYae0b(GV18*s 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%0D%`=P9i@mh+Ymme4+c) z@${KjbqrV8@Gk5=nGVd?avHPom4_GXZi1rSFM5X?png~LSlF$1-z)KHSA>O`1~ z3?BBwF8V3f9gDQMj~sXi)2e2Od@?$=#-W`~T)ZIzuCQW&CM!cD!6PtOn&)FGy)ulg z7)iH)6HzB>qxwk*_NLPU#a)=e$bm`)T8-=O z#wgdYX9R+^{SZtS2t0?dS_|FhU>_4F#FcK<*p8ya$QgZZHNOVQKW-19qr)ggXe_%f zf{kT!{GQ?yVNY>$#Myl^RV#+YqGnc$Yq#8`a57CO+GH8m1P?(KZjr)!b4a0{iU`s<3f?^4$&Fp!=Wbnsmyi}`EiM=pFuSxQJP2LF_BDI*o z)+1K)LqHAAX5!j1st>$pta<<*ZtE__smZ(!;p$e}J2%GUX}0z=o0<$S4LG%j_qR05 z<+U)^hfDZ!ESYc&dU$E++vqWeaYqY-bnQ46i(KKdf!`PK%Ba z=-jxRp0_wn^3he^mEbAw5W9uL>a{arZ(l-ANo6Ia?AI-PFtR@lr(%N78`vv&RSmgI zyCRDdZkzm$ zmhc;|qf#*EU;>9LaFGskO&n}AaX(bnaSq07=1w-_*{aretU&kywSk_-BcXYC zUxJW(Cc~Y~Wb={Wk>f|cKGKwQP-*M9ptb@k4G?oDr*?d0JhFPI1#s)kt0fCy*CsB^ zLd3XxCa1x`3`hG9*ezqBJB* zMWSRRG9!_m3(}6KwCR*~1f?BHX+a6Wz;759OO^x5owz-%9>^(9>!>4hlZqwBp0*=X znXbaUt_54G=W0VUEJxtiZ(%abM(j?-9ll8i_%+VM9iMi5HtuN0XO=?>wCt?mdSAVUUNYzK`%itOTPZcy1W?GHz9tm9iBNqyFA) z#)=JfD38O;CyxTRJvmWrt?Q1fI~i|G-brr8VZ;4UyKV9&G)*^TR)=K)_(?pc;-xy_R-X1E7 zL6GF_7z)DyTd55tUk4_8tqvsL*Z2s2idav8+5=7G#>2ly^j*{GnRAf=?i_S1tc_asOKGUB3&cet&{Z7O!8r0^!+1 zjet5{kB`1}T-R(0JQYW6otGzhKS4g>okXbXhLYgT=m&^by>ZhKufo2b-}eUMq`(U} z)%WlMnL4&oP|!=~{5sJ?^C@XAYD(^J~v{Yx9VtyfV$ zE4xBbCGRjaVt*1TCn<0N3rDS$gQxNCAJgEC%SAbEP&wiLr{ePQ75or!s*lq2CY)3I z2BO@T=F5xRmn!7B?n{qinz^)2^1X;~(VPY_&Y+*bWR1J2Y-MlQBSGC z2P5y9!BV`#`{+z%ynAEJbY*m*p41&lngfvh91d^`2Y8j5CA0w#H z?6?ZE<71c|AI9wXkYGRo-+m-EG%`c7i>M`m$4 zmnw1ZP^D`#fDblF4E?)na5)9C#gg}r*ha%vnU`cjFi2T^9SvC-F6s@2tkj7O@c|s^59Jf_H8np9XW>$#9h?n>+DH@B4*{VnFV1Tus|F^JFDRFMbOTxqv;dbHqdFnTH~aZl z&+K zomO>A#Z(BxlVbfDdNhxxpp^oFy|C|r%iVj9 zA@uvoRai#%hg?jMztHS`B>@N(&u3X;- zGnkbjcoKB=!5qhp5UWp#r4DuS;0k25BUAE{JY|O!j!5~UR~glpU4MY(8LizS@i=+<1O^Y#$u0m_~UQlJba%qM$WaRmRZfcNt?gxFNY zC@Er}3|8PqBY5~;3f8#jAWh6C=%2R;D`Sa*LEIQh-w`8gA6cQK9kKE-5tAvH1~sP2 z_Z3-u8IiS>SjmoLWptPeAp(3_!$l_c7iy*b$>E&@pzh(TSsob(5G!KuAYSw@bGbZD zyqT2YWhJg`frPCR7(9G|42r1{<(1ePKdADi6jPM+c6>NMvxP$({{(y+@KNX}WKV~* z&aGUR#{=ce@J@;p_yfglY&vDJhf*8Lm7SpsPoNg>VF-VCU%;A-zF7*u$~{~TVA!D3 zDif`l8pL0*d`D?LT(Dq}M}&8JDdD#tV~hSOy~08z-yFbAH(%nD8RC~P`bfU;E5t)6 z&YuaA?=eo)2?WL{<=Vp12?wxyXa~Mwp87n^_5mN?`4x4|mnY1Z$IX}1=gULq%SrQP zY5si4cRg_UQ$734p*VV&3O%){mYsya&y#QbYuML>FVRC{VP`!Y{2kIn5&4MaNyG=t zD}Sf+muH4QU^C3cpNzjF_k;D?euI^>i-X-x8;}723Hcj zZEjf)Ceb5h(?x7Bxvi4B0NlDC!O~MB z5Lk1_9n5rJT)Zx&k629QBdV83%mtI9rey-Tr2+CbYyn*fWSDjTXu zli&G$&mjxFMnB^3qK;enJJoSeMnX*vR77&1>Da_hh8QS~FO*6Vo5<(A;GKBYq$9!G zBkwW69K4Gu&5ia6PL4#y2Gj8l!H6=IEZ_3RS2uO}_Yftp9T>QX#hQk(V=Dy)Pa{MK zOnZwhc+^J({3-*KAy|ud!h20$yc9`zw{g7mg)KeDNbkKd(s%Gj9LqpK$b$&i=4iYH zO9;#YLC72}kOU#T@q)RC5(P*NPK~^01~c(qW8{L6ACn+tj-C`Xq$T|h%Tq20`Ed*c zApqcWJQsur+$RWNh2$mMdHJ+WqvsD{WRaeV>m~17^u#v(bv!+<63fNo06hX+ zaK!~B?4}bK1SBG%zF00_tp;d726hr3V^3juTw|OgC?n~}QxK^2fp7sLcn5OO#UWo~ zGLyV?vC>20kPB#eIFiLLi9=G+eai7?CMm(wSeZf=(#ZQFO@U%OXuCAA1~UY<-FxoS8Q$6ot&fV$`)K2_gs(+Wgw3Jnpc3BfX+v)@-uG4rsw zehM$zXbkTpp3FDbl!xGo*kTiw08de(@2-D88TF6d^&aBZw7Wiyq8a3?v@wGtaxuV5 zw4L=j0Z%(?8s*QRPce z_9(jdZ8Q~9i&#U)f(TPalZ1HPqs8J_{Z)S)fSeGo$NEe%1b zd|M(S0|7$H9zeWE9A!(xXQ?p)gJ;hHz=LcmN#89ERW{$!&`V@X^9(*{OY{D}i2`s~ z+R{)UZE5JK(Sa?^2@)Ukt%Sa%p`3efX*eZ20fsFNy${^daEQkU5p8LRUGOapy|An6 zX-mVy%2hti!y^ozlHuqjMdi~_&Ea4ZN{T%TKfiz-c)oW^kVmqkXS=zUj#vqo20M^T zlm)QLD?PI5mU5<#P70?3Y8nUYd#D)I zLj|d4>Zv@M^5&HAjJEp<45Xej;}l*Tf&>^guZ1jx-7je1nmD9@8;4Gb?xp3rG`I=h zuH*_UDb!7k!T2gB=S?54y~)+QnAC2dZV8L(%0%`I6aX6yIASvhHMxb*J;N>vDq8ylpxzp1idA@3e`wCKsirbCAb+^Kq z_4l9)N!oEl-2U(A-F~Xk-UW^xX7>}LSb7Tjki!sN2H&_1{OlbrvmYphg7l2c2MVBU z`tnJ1!6pPQ$eZ;dG+O1HQZ@mqn^?8NPYw{F+u;ZpZURWdi;57m;}x0&ef*%nZRFbjNOfF#?VRfK9 zNVA((Xvz08e2`0Ustf7K8hjzcN(@FSatV8m7oL?rBg;>_lL zO6x)HELH%?6{4*|t2ia40|>zER+<;bJH_-;mMAik;gzjWPn+kPU++6=sUget8)kf$dYFrT`iM+^cD7VCUq3yHA$yHX>Rw&K}@`c{21PxkGK6>m7 z&#c8I1%?H`A zMv5{%>q9wpzHFX92o;{>c*sdYBiV=^D#qc}pU#QsRSl&3o6f>e`UHgxD$E+ufYVi0 zuhuFlidw3%j)+qxv%N?Nb<$Vx=&jTGAly=*?SKrbQU(+r9Mera;Sz*g@RXu9Qjt5f zl1}FZZ?vDnXoo6N-ZHZ?(=~+s2wk=uo{s5zY{kpa(+LzNJJ=l+=!59}aO$!2RJhQr z53+0|hO2XVJV)+1J+ba|yrvf&n46mnMe0|mrQkjNT@X>9a~yOam#fk3G)-Xi4qex) zM*}x7H-^&gKw1UIgagi!K8;c7yZb9@e}f`?`F z)WvKF#&}(Ti6}TBmB-1EsDtbTuB@TAdgXYaSorKzGqB(QpffoBfo_$R@!g5_+xD z1y&F)K83zJ5xGOl#bYspU}F;fn9d@!l5+M9*txKLR23z;qf-&U>DD2FRO; zPz3JhT?nI)awVPda=rwPEYV1Qg5sWH(={yHI2JxTpry!lC$#&aIX^OQX{wx>JAvOe zf?j)E&OtbB$*Co;pZtDEE4zO;t)@w5U!gJpC}E|vYbd~kD!x#wCqFBP0`^u&C*vLL znMRO=K_yoqrCR+u$nji^**GPyICVp2&QVOU()0VUsW3ZBU5Q+;UR)Fcw1V44sTQ7O zb!P4|uoKly>A8+R@0;RcVz3L&)bAsTH+In`p*VFqefOyArj z9|48l<<&xKag0}fz3Lch6g$hvE+-q)X}dkujpacruKH9j?QXOmruBn==`lIczqF~r z9!`b8$lgSqsd{NR0N;J=>ls)-pbk>_f`cs>%R387x0^&-z5~iINJ^{;1|=@f-YlEk z#bGzR(lr~i>WOXF(rLY0o>wH_1n;rz-IIKR9U5j^@z(2-1^)RclEw8E*%H?pk7@E*%Z#5!G zXe3@pq!UWU3m+qx3R|%je;0@illHx1fY8oT_W_NcO%^#D{bDMXz!PVY!i?m74oPPj zfJyQ}eF=QEg(I7W7o;XZ$RpT$W}TAP|<9TlPslNhF|5ksrLryYc30gfewX zGj7pz9GO&{?mf6Z0Wk!FgWKVoi`(#K+FXp{#uz-sttO#;VN9%Q+mWfqn?qPFw~%H< z=yy9(lHm6Z-Vb~MF`>^v#tX#!Q#73+#3liEdl#s^sZK1 zqN2X@2}VDmc?`E_cO#qHNe7_TpM|~?BTw1<5V8qxqU1=0v{4dHS7#1(*Q#5eKlc3x z9D%)nU)2zlQJZvOP0TX&%H01If!bQiHW3P*%z*;ohR++U5J8OwD83p(+i6aUM}M@2 z5@}MRu8F&_r{L3T93&ZPyn2{sm`ibL+Viy1D92<|UW4v?Fs6C|W}N=^pkyU%0Tk`%HJp(lqE3b0J;<8#9vMI$n3#)3sonhh}!}k2UJG?p9 zg9LIhVu#Jf;ROy5;U|o4uY_k-FYRJ;!Sj#=)5loI6virPxS0v@9l-Cnk;kBO)|Zsv z`{FeH`!DVGFbK#70Uyvve8*r#N2^N3%>;Y%q(&t!u{bqkOoVvEo;s<~QYX)yMcqrj zewM+C$JO|X6<>JJmkBp+5N9$d;~VVVF>vvX(N0PZydNYYEH-2W`{=Nz()g6TJJEqb zgw?;4NxmKU00(;q<*c1l2XqFj7h?w7Su7#xcrS1hlLT5xKxx}<$_BiIyb=T8Wj zv=2e($CCFaRLjNyF2*A2B$kh;VjiIoi|mY*A3+C!qbCr!xr&{A?*oHUP9+bXKnP?o zL?_+C1gNy-L3Dr6Dm_VCel35ip1S{F>M zJ(=MCeAAFa)tE=WQ?9}YFk`L*T5%Di&*s47HfO3;f~BY*^Ukz^GnKWGL9DSAmyx*2 z=F`M!#2r;)S;Uj3>-^)1|hO zs=5~bcOH<4K%#&~MIAL!ssY1OLO?Slfkcxqy*A7eA-V2G&{o(G_EQng&5eg)l!h|C_>_&;y z4U;L_!wJePc5H!V*Yu;nQ5ZfDnk0?8Ad;a;%o66!)0&W2I->XBHO6C^28ALAKmEIh?{gD;|5Iv6nk zF}`rNMowbstPTb@9^66}tWM-Z&Qpx956RU!T&`+zHJ~s>NIhe)Dw_{2XFd9C)*~++ zEls^JosT1jywaG7S&w8LElnY0+3~C`XOf3hZIQPR{$4t!K6BgIb6yFv;5{C_ap!NP zte4x)e9kPZ4V7xu@wTGw?7(P#&+Zx7mexH!I*aFX0tNi$^jt~E`O)00N9&JoADw#C zd0}>F3}y6RXx0-=%X;)VDI*d-PH>h9;6e++6QOc}kjhU)Z{Bt;ZJ6^)=&d*uq+Zxx zb)v1PTmw+90Vvl1lxqOWH2~!%04GiZzzII?OF1uwFKJ^fa0azzZIREfCw2vEf#oa) z_+aNiYPm!lwt6)B?6aoh6GI=Hby@n3YsH@%abUl_D7N% z%HhE7H>DRWLH|oU3+3?r@7IzkXGOL}IdUY}6#X^jL_1}HgIL4&zlUU*JY1GSye#DOJ`r09&$<(E@oXb{aG_$dGf-?{5p17<~B8F7oSyrHkxhr)sD6^_gWM~ zW)^5!kL+7;v~&(dI;2QB_<8QObIksITS#o@KxW&SPcI-T=b^*dZ5%AVLwCvw#zVB< z?t>Gd+brBf`%dbEO?n`+cHzDlK5kksU*vB8{O6o{skonghh=hmz`wD!smx73=1dne z3>lj$G>yZ!<0IuKrtdXgpo8io?b#96q?(8;x5gMxg1+6E%Se*RiQeq~RJp`l?UHX2 zm*!^IyE@)$zne;02U+yk_G7!5M03+y@0AczO^PhZvHosLs}Hd&0jrv z`;66*(Ld;~%pKidKDuw+4-lWuZBGHFNa+tUjS8Kchvhs(9~H1$`$bL#pd8x5MGvhJ z{k~~!y1>8+a`EU$&FI3#nc)$-X}f9{KOSPKA3f80ppeo1H)Qsgr1!5HUBB41Wa+Y{ zH}Rq8!F{$iV$ z7wy>eSK*C4E*4`&uW)pMQjYFS&Dt`^LV!c*p=-7k=cex1cYJHvdz|Vm??_)7nphql z8+rQJKCD^K>wELP{(9tDWa}dbHvy6v5x&G&V%Bx7u1grRhTAy74bV(o(HW7aUo!ex zn(qvx^_OMC@Qoi9Fa=Rg(XQ}C#z>t{E7iy9M}$SuN*`smYKSrd;>MtET zDZ-8$_bb$m&5vZIy?E4_ULFc3a*_xQMO z*0##Ad6KY15?*U3>|Os}inV4Xao#A7NlVJhCFOaNa=e|AR6@L@x_nfvTvbA2T#<~y z86yWL?9exQWNrQkDI!%y(eqiib@NzMm7Vpml@gYbQgwKH>B#h>ij{**fz+FtwfP*@ z5DeEbk*XuTg`>hFdNb3bX}ykjBg>AUc}UyUzl_Ty4r5%{w@;9c445+ae9mI=GIva9 z^wzWzC&)c`yF`{9E-%^}Ja@Y@rFZtI@F=Im>`5ENJvC+7(ayoL><%^-4ZWo}m<)G# zlquw~3M%7%6&1QJl5rGZvHt5_KR7D%!#!zZA{i{zwpNZD5zQ-Fc7%1FMBLqx(lOY9 zWp$?UoyVoavvqJf=}Um8uq-?4qYH%zO^9TetuK}~rbb6>O9A%CuB%w3(*d+GYm2l{ z4*BfGzhT>%EY(Mr9K4kmltfvNJR8ko5wLL_l?TRh+1Q!Iy+7Q_;8Y=F+qkSpp38c& zUtY0s7-M{IpAf#NccQmASxKj-?T;462qw3LWyRLTBi5arfVp}WlXu3Amko4H?p=wgiDnjKGnXgi-(mQu%X zX&JE{qqLqv$Y?W}`&d$2gpw+^5+xCM0-@Zb38mg9lu2CRbbM6k`yFGJwDqS6u^bm- zk%xD9$ctp`DIIzFsOXhKJQ-%f8Gd$BQq$EZ{=&A;zIgzaGp|R;l zM+ymzO^s%@MHE$)l+Y`1wS%8z1~bXHW5ms2K`TlJbWlZX_33Y*S3;Bd|FOL-^yy;2slyq~-CB|Z1fpJYN zow*A5ONPk?pG~@d=D4*3B=^r0rgQ&H=qcJixDj@7!Lv>nc$!(-!JkS4gsDa!bQyL3 zCVUg!FjvpaX2ay1xb(c?9jON9e1KiPALv|q-gunH{(@8OP}I;$ecQ~JrBrGBR%!fE zlvk{4mWkcKZ)9cjO!z{hIR=JlE#NR41ty}kwne~fc7;7c^?ED3|0Q{nt#HnW1W)w+ z@eI~W(X>c7rfSZkQL-_dT^rlHUdM`4J_Bt$0uZ~{(C6Fg;RBR=0Ve7&yMuspjOub4X(oZ`K1bb_eYQTA-~Smg z*6II}7$EgOyib0$ZF1XfpE=G;+im(ZVB2k95{sj@+w{5Pw%fFgZQE_%euBWeKB3Ln zBVlgyO7GW4M4#DGKtu|9AJdAC({&v#(^e2eBdq8;xGd~XU*m2T{U*J)WfD2Knt%C$O!g%W6l27>Hgj~4IcAb*_3kNUg^`B;M{{=15aYNuef(Vu(m(`C(<+=CF z=laJ;NK7lu9IS8hd-86f^TU0QiJClR!#Sdbk6n@W=kIHRV#~i**fHxO%vH+s()7f2 z@;!gH{R-N%gv-Zta`XDJOk)GUFSRcLWHZ5nVsh&#z-NRI$ z{`XX${yx<|&WAHjB7t7|TR{HfgQp|`i9Q_+Lr6f9+fWQc0yC4LF#AAXqPWTeVQ3#@ zo(_PaB#>!WKTIDy@}(psnAJm`nZ}{-o|*&@a)uuQL#t;=by^a*Cb@cZ7}^&kc32P$ z8HN&lIv9rZfh4z~7^V-tcUKY;Oeh-3CwMsUol{bRME@O7Nb>iA;;K_Yajpf$&8LP! zqW=ykB>DS5@wG1|;ld;chalpY-##@#I8Fz^P=avS)hE&8ruCUJR1z-A6d7b-qs6vO zWv5ZdDo)!}WS)ForU2-dcm&Z?M(aJw4NM}#l_2-Jx{U8j3Mr%aq#HS7dOzwKoY2dB zoj9v#i_Z2hAWjBI#QS1s{=>q+_F`$Sp+jV5cP=o&x9{S)fzri<|GvEYUH2fz%tyEQ2q~2^4NGJ;S;;@*LwMe3ChG0yXl_pUIGWBm17ojGpf@ z9{xNs271^to>9O06`>rYKaMr}APFB}TTkwm(iH){qoJWNoD)uOXo!w9anms<_dHIt z3=j@C8cV2TDL3Aoos7EymX+H*heY)v76`^FNYm2A>M2yC{4YR4Va%L~nO(+b2}aj( zD%{62KSpNyeKbprZS~_f*QH#DJ=ywf6*@cVhTqFrLUK71jgdRYPE{nbyf6#eY~0CQ zi!g7|h85AFj7`bmX|W7gO(kY9Bgo)N4rg6r)o{(JBFJ!Ya(Ie;g{(k@4QXN}u{g3} zrIQ>ba*kEi3&X^bjXq1vmey{uDyi)=!^Dy8mZ4FN@!D=4CXQ?+$x*2tGDk^OSDq?@ z3}+{Yr^m4O4kJ9Ff6NFn{5Cl}GnPT*-h>Qh1Q{Mo3XcnQEs!27U#R8EO>5HcLr@Q2 z(h=|OwPrR=Mc92?7e-q|W!r_=TMp6>#$EJXOn78G>w7C%ne5#Wv+dJ+0=NEEb`i#R zoU(QpVd)$|&Zf4H!AhL6x+64iBkY zbJ(27EP6MD?eus8iu;bIgo=#6J(Xmi@dQR*v;>w=FG{FA#;=m&?~29G+CF281N#Ox z;0geycxitI3O%hRJ|FAV)gwACT@uY6oMkd3<362nG%RJ=uEzvJyB?F~?s`mM-}RW3 zWGXX_UxRc6|9%80y|yr385*!R9?+5Be!narAI9ug?8WzD}P}h^JdVT{&|!5ZEu|nct0K|#<>{D?Plj<-Rz4(Ywjr^EdX4?&2a-0%*iop za|J`vjzM|z;ZFj3gR$>8_;ne_(MK#@das!=YqRXmVAb^lB9|6(XiLJ4dAzZvL#wkn zop@6z+Iv*D?%-QYICkk{T*f#lz5hbV*WSxCH#lK4^PrfKKTSezo9rJW4Z26-R_&O} zD!;Ap!acm8A)4^21im~s?P)Xkd=iGvzh@F2jXJBp>W!;L=p>SFe{`hTA!2?~+i_i_ zY+$gUv3)Z|pwW6EGEeN`27kf0%_>~BFFcC)h}yd3x+vQu@|2U1uRFbCuzkOkQ|HN+ zJj}KO%(apU+bR8+W8Ks&d#bw``rMLid_nvBje9jKz*gu4JvwMK)84{mubQ7c=tyhm zD2}WjVAt{;l!!1O0JE&3W@&r+@qK&Jdas=v#?E#xNLC9P3{j6S_3p?Xt4->-h3qPB z)GSyxVQ%5Zvlc}Egk^1bH?+9-Nl?D4_u4t(*<_k1nZRrc`jZi)%veCig7trq_#EO- zpo?BQG_r8n-cZ`HWe2jJ9LcUT$i7SiJlKyxH*|JEA*2IwG<#eQOR{4P(13 z=?l2V+UU_w#!i$R_AbI`v6SVa%oN!t;RCpbBhO@Qd4~q(!Ag7AjwtWjdjiGY5zO(Q z>1e(4rep7NUgqNMAM;^ib?4)^g9-^jtGs9DKx*W{Zpc4+ru8ycMnu`@?H{{}WOr^~ zKKPVQKiJW_d_hNs+ycGp&SQg-`r}l5)xOB<+ZPVr+4p8z+S5Car$&C^eYultX?xOk zZQq=l2V0iP;bW_IWL))hT3Jp}ea^;_+e0JYiKeCGBbSV&kj`#O3nx z=okl{i`QJszf}CT!kIbv;R!@cd0*0E0f~g&x^@O;mjF==_@-%8bufqiz?R&I@9>E*!Yc*c$*c<2CMM^8F!KXydMd> zqBui}TTCc{MERxjLi`Zsy z7FK2UoMCK})R#v0p)_S{r0<k;6yu$%HOzZ!8 z#_AEhPdBnj_jJM5v*hb5fVXRPCJNWW`V)cI;Rr=#nSu1J7e&9a^}LF0=d#gtR>jtg zg+=cK-jXld-1O|zqs=pxut#Rt%a$L-paF?oAW z9A*9C9J&5n%(TY4YK*tOB20F4Yva9bM!Q)QBJiz;@+{EjxPBx4gVM zlZ}wT9s68$Z5|=Rn5`SZQ0w#Zat@ZYbNty4rl224vwIB zyoq8#T+XO)R$(%Fjf!oHuTNerCku|O&GYp`Zq_FXc|hHBW$(KM zC^O2&3vZ3i*p|Uuq+#AAoR6<(VD|IfW#iNS z6;hO@N1mkijrXUAFStJ?JP+fe&>5T1)*?3boIxuw{X4Omk3D??Vl~-BI+fd`jrdwbL=z)L;Ch6`L=~-enjN_0XNSLb+Vvr02jP$|=ckDoqePs4bFNT-?T9{L!n* zGG97wpBfkYK-7D`wCp@Yk%Zd+552nS3tc)+cy+TyBr3S-Cx^w3Zy8)SYGZoQ+ky1RE=~vTq3iTn=I*mZDiIUr(5yS9 z_BA6nzOvjMKn~>5=~6$&u)f zi{9P%VXg*zJGhEvnAN++vM(fBRAWx8%`Emat1hK=TPa#>N003%_ec?*r5Mn5HFZT?px;ZEYQ zEj&pihu^Str(0!DJlklaRQAL>&RB9#Y|c@6#y`Z8XOt7mhghcIb)0c?e}TDW{0CHQ zTAiVhk+c~u2V6PV8i{?3CG>dV`s|GpVb|+N2PYWEzM~8L6<)_VdIpGo%VjJSJisOG z@316nhJR9%yWR{RV|-n*2^?%d4;6B%X_ejs|AG`|4p$~MG5-fC94D4O7bIK5LJYea zjE0|(SWum9R8V-#o-0}y%vjDZY=x zxnQ;lvfIlA`|h1F>|bCQ-`3bSeu^Kf`=A&*CUwLm##1DqBfb_(XZA!3CcE@RG}$8I zd3h^Zz$ zY#&{)&521`W7Z>U(=kaiZ@qcx_@;MNF)Z7K--|yVpFQ@_<>GUk;v(^}1f17fk=`2_ zwY_p`?zXdf-`v#u9CBFnisaCb_CA*yHG19Y8>3_5F};`cZsAn)O`JBADwgdM63A0K z6eo_T+9C#q!#oYiIvNf7Vzm=@fl%tzwUeRbuUQ44X<|;xQrRW7AS`(IQO<1lmVJX1lXbHzo1T&W! zU!ZIYvR|6QUxQgX#CY)2N+f?h#K#!jI;AWIqQ4XQHuj02F-Cnn>Tz=%TZhwa65^8% z=0}x6uegl6pB55AO2nQ_K9|H4qUEi>F-C>s9*AZcX|WPdMK|9>Q={qTFH)3=ufBQv zI_HbI_RPa9wzR6VrIAYmE-0LNoPbcRF%ppl&VRj`V#`sqtb8jZG)D&ZC>p(#&6+D* z#vms%`O4Mm^CBaA=gyuNK1&2Um+>9+C{VX7`1|&7Fo}|0F~3R~;YiZ#l;{!NZSvs~ zy*RnRxb9zsJHv6Dc_oHprt$kBQiEOnrmjb?D=b=CEc<8Ne{q5Ft#4|W^swg? zYDWyF9)YPy%d4~Dv7w(+ym2ohnhrTZaJ|d;J&7ZS7jX2^s4sTc%y+rt(o)& z9wRFHA?CaWmr)sq&15v6#==NgK4(9*WGp~xE32(Sw%eZ&LnAqlAtx=_?Bio!_uJ&D z*YCy;?Z!o~eQI!{HUrNz(t&0vT_!VM=mj)+hoK%>N=wiFMyTo(t62|~?IP{V^pZXl zd%u7%ickW-D+P$k`h*;j%uz(b$f3#BVfaKwN3)8QuUYqFM+Lo3-jSLit*)n2qHpjS?xEO8mT$QL=ju!FPk zSDPvUNJPx;rc=H}B9nNGAcY->Udh!XjV|MIU73_`DYiO)V^N7&=j>BV)8AkrE3F)_ zTd7B0-=W#qjma8puW|Gv)WI5Ty&O%JFOALS;KyX++h32FQ0uLsRVEf^bMzK%_C7i9 z3;n%(@VZo4_1cTDy?PP$G&4OL7$Ti+%UIa=q+W$>EgRf`0(>tEu`;fHa~VI~&d3e# zg;7M$lCa3f^=gc3eOQV?LA$Yqoy-ARYb7QYvsYndnz4{D3gb{M^@j5ZPFQ%&JbB@D zX@AxD!m9CdKyk1jJU+JO8nuwJD9AjcduL7Lc%vBO_Se5*eC!TcYafVwSs?1K8bG{8 zvc`ay(+R8-#wS**iM#XhGTp8vqj~9clFbPnm1T!U zGGy3u7Xw#p$mI(W7*HQ(Gl1Q1WasJ6rtKN`lEl^@;ZVjlHOQAU%LWgkN*{a@xAAdu z^s=zKf`wh3EbPVZ!)J?n_$ajpBGO$IXrm zM$gd;x-|U130!eo{8!8QJI5Vb(YbgpEl^uDSC1B`No_f znUBArGe@sOB`e3hSZ$7{&-GO1HI;?YSsde+FkEJP3QM@b?$KfcXpGbe9E_vv#M+rs zv(Q1ALFU$ROoN`Vem4H51=N@x5k=zkqm7YyA|AjIBN=cYkNM?InQS`vZWNX#@+HeX zp*p$lYpL8Or02sup%O70Iqt&F{TfLHjm$z(&zu)TNh`)Q>(4^15TactjD=yiE;Vq2 zp;QvS$JF`Br}3gkCBfyAAbMD{G5vXIQTnss&{(`5j~EEb5MhX#u@*1k6ery z8fCn}11d38V0+@a1r%GWv)H8V<=V82VkaU);K;J1Pw?rZH+*{U%)N4YC42_&>BoM{ zH}GnY{nq*nUu*98H1mUezfF7PCZbHClq=xO+tEkLErWkaz5uT8P=4He2i=ma>j%;8 z{aH2`%MKLIn=88R*&Qj5?L7~jKPcrRs1xpBRYYS(KQlt2;X7OykYLhd2E&k1i)C++ z(}CRE*!S1Lbw?s&r1ZXb zFbH~wI*NYA7gq35{BaiGHSWAATtAVU9mYl~IM_FlszRAXzu>Y~skc6x!#k-<4u&bL zXkXSRI%Rq&%ifUQkrq9u4>XI}j(n_^c5D+cKZGHCYG9yu{U14A@yF0Fwz*RHI`_V9 zXU;(lJm46wDTYee($cFg|_0O{EKgXCQ_4CE_(2Yq(RNW_mU>Z@8aH>r3s9+Do z>J6PML9Zf6n4`rcCw8?Pjq@m;iwOpdY^kSg|I&jG10r^0zbsszNmq}I6mv8<)9iO% z#1>2h#uK;7cw5>hI*}j2f{Qqs>ynmjFB1jv8;tvjhT~%-+Ak+!f5`}Bm~ zntQE-SHyM)=>u)

Z+a!RQlcIx0*r1Xl^K>TQ9g=YW5*?ie^!L9&Us+YOom6PUz zXJiyClLNP_uuM&y3qJIVVdet4v_n=b_FV89bHS7$nGd01N6L)eAX8>}Hj~f*_<170pPp7>&Ncir-Hjd7WHx*8${ zjnu&!q_%@qh<`oRaxGo|`a`Tci;Iqh=1$)+dwGh!%;oq!(xjphNB9koGG@Yr?$sf? z!Yibga|pRl^DteGZ#*l8F;)?wU-##j`NM4H z8sE`TQXG=|uKpZzx!xa^)zU*($vu)MjAptha=hr>rR!=6FA7y}pYbG?P6M5kNhRDg zRdxc^CJWjey?lu*NSD{c3%AdBtnebPGywd;>w+B}>6|l>W*Fyzb9$8u>!Z~g&Nrdo zqIcJ)9{bJqg{ks+!CB;)C;77aWO?rDNVoF-DsI)oIdaAj$D|+YKdXOfx)@5GryD6a zS`$GJ%-Z=Y?9LTq6l9DdqhK6#H(@LEF)O^$n81^!F?+Sw^s??YgwV({SM5nFJG3>Y zSiaRNXE_!#g|5(3PdG~?S!3^=f_ORT)@c0HXQCv;{eUNor+5P^OTj79b~`wf&~L^< znJ;T4Fi&*UgL$GXAIy_W=mzud*S`)*Zl_XYjCPyfEE zfA{O(H}vm-{ym_74gGsa{~p%AGMx_Q9oN4n^si#B8_Y}BznS`XwEoT3zvK09j{cpf zf8`*|V4j?f8O)ocfAjUP%$b9Ea`)_D-faCV*QyQX73<$p{adboEA_9aP%JXxWmP%YCmo-?m^t` zxDajyt{hi@yA+p+dy{-G<96V_gZnb>6p$lHu_;jYC^!HvgVio2b1M&kdGvVV`mR=i?^f=HRZ!HQ+jMy}0{vPvc(4 z$=^lbBONDyd*%?1lfNtJ^NVn^rSEaq;~H=sxHY(JocwJg{oS}VR=yy@d?Aia#@-n? zohaewTQ$YXy{%Z?^POV#^WPP#Z~vh<7JKm5#j2?`{vM04AJ+cvdU=$Z7vdhA%y>9A zRdt10eZDC;!m2}y!=Av}+O@%ur?aA~%^&D=hrIqSqExM?)YBYZ=?ZvPdwrgj9tm5I zYwC6fTAO_Cu9Z^@rcZBbE@)~CPAia5;+h?03l@~sIj*m-sVJ>jTwgYY1P+Iz(&Ok3 zG+p5-40^gkjuwAcDBun`S|y|<;3dQp2)VsoK}T!A?+pbV9pQkt)%)SFCpg8?;135K z-TrVO;Pu?%aIf}z1CG`!+~H8b6H9LvzFHEx0}dY%KEIb(k7Ko;bcC*@q+!75@;EyE zt==}Tq$6j%SOA~^N(JIJfSnx>@K0U5Bj}|WJ|ZW#cpa4Kqc~5X(-R6wCQ1euCYmf9 z2AijXxS*R~M;RF%!7JR|-9BlyLol<_?elb!H{a38tGkl~UI%^U?FxAUU2dNStIO>T z2Hl|0AEvph1MU2d{?r0KpZq^#AH6L$N&A$b9P;VwtW<8!q7J3HMgJ)ox6n}8}u zm*CeQ@CGfET;W;M;tK~M4XMZHcKE!V-jJphK{7qj?e7Y@o4s}o(qASf%^HFZZ&#}~ z;Asg3^BwM%7EdrpKZ0ouwu2`7q}g5WP#Boyv4!SB3;7PO&mmxV!XX`=T)9A#09=cA zaw|pCV{||qodLJ*S=wG;wNlWq5<;G$d&H#aHt*!=v!*xsyw^y##5<=cv_?X%XuhT? zba#cko_uL`VxE%`7z_oxEg@1; zVwZ;2=UwUP0xMD!{pXp?8;Fh!!eP10RrX&Hs@GrhvFlV5Sks86;%~=&bo?4 z)sE7#T1V-ksw(Hgvf9ZN)pbo*v=r9xWwowF)wRy~6_ph_ zp~JPPzQ$3RP=v)PZEm+3UgvKHKf>)i$-|!Z6BMMSZ49$67yum_bO=o_$~>Kfm;^|7 zx4_XzYz=gt%|S?F;WNTsgx3@}+J%fXt{pS;3kn>$j`~`Q_fjmwSpYKGEI_qFcsiRq z+;VcYvB9q?lsYXQknc$B>3Fy!*ewLp;)O^U?>3{*WRk!tXf_O0$bbelOAb9S+JvMM z$8kcB(F=>Jb;sC!=2+;gaaPw=SThq7lTD+H3_W1iyKZW zY#VMF@Q2{l!tYy|gk?qvN+$}5z`!m3Y%@0@5Mj^OEE<^NsB}x`>Ddr2F4G?~A+w1+ zKPT1+kCPeIZWuE~CzD`{9`CgN!|`dW(;aAOcSGTD#Vg!edLULqsqxFC77j9GPfNa|#NFj?b>~ABTFlWD3{rjWu%JZqZh*7VsX;PF zHnG-}CQ=BU)&^4)$eHyMIJn8bn;kBn$IV>XMd!_NIAMhC@YJBcE!5*?z=}X*!5H$i zGR^?elLQ}q%>2?uKA^FnBbhG7y)xi2`Gu4PfvonnQcAPO=kJlS79bn7v?q>3pLhDK zq>0o~5pp25_#Ilzp=}<-5{I`fnebP2d3qfFfFs!M5BpkSOshSPRbBp`E{D5W<{AO6 z+aC;iwY0I;hPRc;c%{gTjye*Mlxzu73I)bxm`?XvL6U}Jts~@jtc6;^g!G3z2k471 zWU|%~OX-tlO4adFC^E5vHn7kFPP#&F<~tcN9;k)so?%5&KbaYh7L?{>WA@vya)&K( z&=uZJVfFz>bHL+nX_q#e)Y5{06c3}DWzt-m?^qeuqK1d=l*YhPWtikUOtAvEO<9eU zU7nSeWPwy-gNf`1qWxBbO#xO$Fq0=;zSiu(7I!m@%;)ZjQR4fwCs~eicddo6x>kz7 z0Ap#j>$B+6Bxl15@^*!er;Sn(Bgt_P-3(tBLs^aa1pZ$ zWV*N4G~VemuSpX8SlqO@yMqOgln5trnb#)ZkYAUQ&~A~OhLAn-qlCyZ$uYHHiU^Us zOf7U&#tWn!faEBPi`i|$a6GMEhKajgqwA$kC0&OkOcmS5*?{w5ItwjVWtEqNY z*6Q@KiVB8hIiUrmF5g;f>bFVI5$+a@*H2-Z&z(7?pn#EM6)i)dLuo2D3>`54&ARHpxNy7r(mF@Ian29Z5 zavyUpW@H5&J?&n`J46f_#5geBFMTAFpw0a|dNr(Vz&Z<=5P zAf+nkR@Qg$3_7?A*+X+ASz4L1EqLq>2Vl+M8j9`mcTM(owFSVYo;MOYIzC<)aEzBB z!_FKpGV)A#%#gl3_cc3G1yLWbltkv+6qOy-=_ z5rPig10W!AN=V46=Q7=`R?Z%8sGX4`lpkZCf}OSIAnj>qq*4|OU=~XOVR=+y(=7!L zZG&GEoZZnT7#|2onN){s0OGiw{Ip0Gg3yHx5Ek@{f${NPS{=1$2}H8eYM=^@%2PKV)&SYq%*L93^OIT;YDZ~g&&7PL_R}WArVYWSDnP}O<+u}Vzw{F#1d8vGHUW+XKilQuClWCpc|2;gc%_%6rmCk z!|pzl9}LkyoO%{*X`5zKsV2W`5<1Z-)14`mRJwaC&XB-%lu6w>->iguWe91Y!fEqY zq1Ive3W^NC7bK#Cvt%+o&1x1W&F*d$g3phMB9ogD$y;NxjLt7PM8dKK914W8g)<<4 zQU$KwIYH4POdWwo*CR7UB1 zOAOX@Y4awTQmQl&M{|u^3JdD? zw&*;Tx>dkn{y0sEW-n^KG*p%~2O$%qhQt)0wAwV*KnLT4IV!_8=@M3zh# zJkk!$?wOwaw5WTnmCc<0?GG*DOAy%ZiAfcJD#u8uh$cP)-CPv_z7Ed=?0ubw$1r!*Pe4aIWEvk98o~mODVc`yWxLSN1orZ5~u01uO`zxwz z>ztL9rUu@1BcAWOD%-4gNyRZ7h}Rgn_c0~W?|YQq@I#>!1y&= zX7|$6cPT_jHC3|E!=$bkvoXIcgUzsB3iAl;t~?E~P2-4NviVMB+_AmwZy_=wb}Zo#C;3l%bJ|*{uhDTl=gv|I4yn34nb$PIaA28 z8y4Crf(@7mLUAExBUqi8)Lbr^i=`MN87e!PQ-+GgS6KUPLxfmU3N{nRY3h`Fr#8z< z6q0qu_sN&6E3~C*v&xhA&qZL@Dz&W)8bE_23mL zzm>Xtez#U=ntD7;F>LuHC_-Dlr2(>JMspP@dO>zrx~w?iSLo|JEaRAZWy`!xrUerO zyZ>#yy9P!`Cb9JvUO&=wWfPms9-0UJRR!B%Y%rIZQ)Wz^GL8Q;3uYCqZ(8m3^sGXu zH=`-sGS%!x8Jx}UKSW_e=Hh!svJT1~Uu>_IlvUfKO614c8nTZxwfKDq^36U+Q4@NT zf+kt2b-@OFVb8RgGfxgIC=y$hk2D2B2*b!+Uj5V3y>jZbnbVqDJ#E0jue~q)k)}4b zg(#l~p`;%6RGZqm10SB&)Q0K{p(2D5*p2?iU9^6^N<{8m4|=ww^aF6SUE`LCZB^} zp9&J@AT&rC2W7Qu|7zk|_~%jGgn5Yf5h6MLs*`_`zg10+)#f074>gBK(XPvpQWMLF zmtCgnRV9$|wKM*zR4plMb-g~ybI?W~P|IH>B?=5(7Ig8}bm~xPc@6qaPwVt)mPwK- zV}a&G^6!TBq>Nf?t=p7{^rk#arxfrfLTI;AsZ+O}vf9~_w5lqR{mib`N+iWPzca9M zbtC1;l7or?L0z5j+|)Eh*T`z4Gq}=*RH-qnMgK4U=`YLlG%Gc~q8f_<<^^hPs&JL$ z(pLTKEUl`j&egCGqmeMGN@R2PZAGX>iy)I4lZJ5-B3X-gEb1m?J?cqZ{AkYmes%sGWoQ`2l*qPV*A=wRTp#8I z=Ll@^a9avtG+N6rNnpmCGHD2%EEuq(O?;3jFApMP4z2O~LwR{7e#fA}M8D0NZfJ%n zQ)C2K`{p_yuWC%4x6>p0lqPs)Mj$Vdop+QO6ol9Y`pKOfmROQmHv7g}&u?axd;w3V z>}8`b^)8zSQR$q z70g&7PB7Zk;10wh>SR4^Dr9tB3_Mx0`52U~Q%33f!Fqs=R1{%BMV@nVuFuRZv7t}VC$a8JT-`#CuL({4&D9v}& zn){eK@1%k?d22o-X+0q;90Ci4H9uPfGHXk!uohT}jEDYI2J2*cSlWhY5%O4BhQtAr zMf8F;rRw!K7a~X)3syP04ilgur6^VE@p(>DhY%kkhHjb~MWq&m)DxuT@o;(Twh|`d zL~x|7N?D)Fc#<4?cv;OV2?c!Y_;tHO%n3FEAYi?G6#ZgI^Ez4Pi_(R~eOIfmaCLij-XSe z;Kat4`mfJqF1${uzrs)2?4>rtO8R|Dnz+pa!ep$01WcdNSd z1~qU4|AOj^RcfG$e@RGbrI&p*C6rm zw_&#CB5Grynk4VN1xiMY+B8`$R~IYw(Ru1BdGE_qnTksu#&YRMmbyb_eON74XDOB4 zqZ-w@N}c6F;u@jU*#RW5vy~bbLiQT3)cCbXQs*f(A&ex$*HX^$B9&3bxqh{Xf>cg7 zvKhyU&$~%|gqEoDmtpyQA=mSTaS|Fd8zGNkq z*W5jEX*;rCCST_5P}fVZ234RQ?*ZMCBl^v>;W|?^9N*2KnO8)y;}e*(+7hrK;6frKWZwogq|A%U5%y znCVm0wc^jvG@+FKex3wei8FWnQ<=PJmuThAfd_#D)Qrqz> zu!`R-k(1?zPkYT!h>-kQAXm-e`T=>@bfDCtsazM%t81E4pTu80T`Bmvswq(FQ~0$o zkx%0f^Ea?%;9~>*1CfDS2euAu8@P=%UfCLN$QAI5p|YYuK`l9(lK2#wCXy~V!$^qP z-kLs{6Y1Q#%0;!cW{A8htwhi>hJIC;j5;f-6B3Z7W>&%U8DmB{;W*|NmR4BHblm9{ z13<#Dyy-+23B!=}qGl~6!lQKkK5uhC7OE@~{Y>61A$CSbSxgc36>`gVH2fu@^easB z5ZN6GVOy>_n^uk`>?ceyAvWQgK4#Qphcga_D95U5D=OiSfT$O*rvA%Rv0`q54L_V# z_A}UC)TzAe38OCe-lCV!qcmF=>yQ@NGIO{c!iCXfX?0KntW(5r4Dq#7>`Q#!RUQ`U z(i}trk#a?~1Ur|?z?8r=S=1tW@|<*ai!XpVak$I0X4G)H<3ORX3WQr^wab4US?cpu ziI5|0mwuTh?G`D_)8!AZY)8rM##T@CGg@AgEnQwrL-0g=O&x--SldnIa|!!z8sA!T zE=I^xJ~%#X*b1Z$DM{dqSI}z3Pn#|A;RIp8Lchn+?s0dU(9*_f*PNz}JzkL_68kNl zZ`QE+f}z&ay*#i}<8k&B1yjYpo zBX$GA7_-Y0NL|a|BJZ*l2rxqBHP#?!=!+^(HV8%2Cj$j?4|j!wfE+5Lqej8hC7CRP zI48EeuA-_;{AxBPYbr`~SkKB*=MCvjRct-Ko#Irpt>;vpQ&Q8?M~uwKJY&?EqsNTR z%0BDtapNbPb8gOg=U*`K!iz3;Typ7Uxp|jgG3i5BUX?$2%GCu^r%j(RbJjJp3+EKg zopjauQ;L zXsdjd_%o?1?DJWY5Hi(B5xBBd+DJ7rH9NhmtZan*%`8A-<8P8-HCT3UjtzHY%}BoOUSomHv(y}r7lWKpSp z6j^cya-04`9^8Sfr~f{i@QnDfLsv@7#}l4Ek??$b!ZYGZEIp!1>=|(-_FS^48nqAo zvZ#7deO<2T9}x!@Sd&`8bfh8WENyVQDo}2f3T1NU&tuCk=Bim%nX>LByqg{~D)aOQR-UqtyEU_Ma^`$NfpH zV9bBrd|6 z_Wy1QNZAzom+}Gs>>d>Vqbc!!sVN`qpPm2Y*8jaB?>7RZrt$ompuYWV}+c$rA;NO1#@5aIZh#vaG zAOCdt&u_hbG>GuCm*Z*I}|N9Aa!^dlZ?ti-d z<>hlaJLibzM5$K(@*P$DIpkq2f2ykL>ekk)dEn1gh3tYue#A6Oi%%@>#`Ubtq%0@UwvVs+A+@twcz9+)5x%b?qF3w89!at(=80U+P%4$x{`2X-QX$ zftL2&A;~NG6zLV|B#p!?_to9X-FbC)C*_B=&HUt`*k=p!tamosSF-hj$_tGO5VSfj zM-kNQUg;-IkiDk%$*Zuh7v{;ri&*(JNYVlyFO?8v)AoJHeW0B;xJ7~$#g zmRmf!FYLnX6s*!Tn4H*DvilJR7QOE-aAPKjsp8}|Z6kJ!`cR0(M~;EE-GJWVIc2hQTe z@_-k#UT#hIb~)4o-GaA-ayUC2bo*AjTm6ApUQbIn>75Rzvnd@lmp0vGrDT`cf+EZ< zO-z7SEF?Ocy;4m)T{}v!g!g7Yrv4B5 z>^8c)CU>pnIY>7pgf;tD(YB;?yg&nzeX`x3Ud~pvPZo;6C!XCc_TFRB_PZ~mfvT)9j%hLy0IyA-IgOWNy(g7pg@H)SY~pz^_>xAMbSlsZ9a<3b zoDfcf?eN(66C)aH{Mm7O943q}J6y(&{oDbCNvGNGY-+}yp*C+bhtce@BQObV*=YgT zdkdXuzk`JMvpJS@f3=3$JewSAo|17Vor>wPc?gv5vcu%0o12ao8i|L=@oTZ-jfcs2 zw}!i&PJCRuiXoOV>{@jPX%b+M#DXoM`(t4at1)Dag~{{o56xF))%EH@%fHL=Z`Zz4 zZN;yyoUhjD_Y$=R|Hkri71Dl%^5K`1*QzB}c&YVXV8u_=?{#W4zLax(l3Cv&?K{;Q z+OJgmv|pm0(Efb&p!O@&-P*ri-J$&()jI9hsukL=Q+3*3sLHipujX2QKK_E5`D&u} zOH`)zE7kEUB)nD`_@!k|^{Vzu)DG=est2`StM0}xtFBPD<5$$xs?GS-i)vK2gKBg{IXi?bF@ESWoy4uow!`W7b!#gCF%w3m#QbUzd+ru{W5ij_7|#N?U$== z?N_N4+OJlX+OJS^weM7uw11<@(SD6e*M7Y^k|*^oRRh{@Q2X#}8y2X&_*|o*c1w8m zLbXHt_38=Be+a+6)~O!AUpT);-G^UNUajsD|3=`m!Z%y~I{fO&Qq_%Le#1i5D(^Q| zsuh-BhcEfdEq^Y)tENKDz^`q%LFMA#=v<(3@NWRGnb!M>T=U()ud7?6;T3rH^Dh2g ze1TKo7k~8V^G9c7&q^PiIXOKoeROtq=7pK;M5#q3F6G2k$9QUWjHe3SkL>NewRcl5 z3a+$UdvE<{@AwfT&~RGa|cjsNxg#Gh4DEZK(5KfnI;@mDmS zK0bPy_#ZDRRxjb?&u-69_`kjE^zqU6iT}{er_aA~Nc;;!@cZZGW`00aKjwKjI7?QX zzMi*+#LG8^rA)iLzcnZ49~%F{%H;TcL-O}}lj9TH;rA7*KAim7^$o@I1>L8QpLlC> zd}8@h?xbzW@k7ghZhLb4{vq(s`$QbK6$#Hn%Q==1m&Trk-JYTFIKE`o2WaZocg3Io zWIfyYhL->ESK?*vPIw*)Z_hpPbbSfWiRF#{PO+MVlRvw@+Y;LSmuHIAJGjL7#QYNe zrDu!9i{&5M?`+ED_`fx9 z`uI7&O^zR0{@eq}@rm&Lp1A)wP5i8XD^_!G@@K<86u!6KCw{q+oPS3`|4O+N{v$a) zvHbi$6sr=P{Mq#nrQ`ROIFsUsw*Q*m=&A)9({7^hB`&4rNq4Co`og6>3e&1J4AD?-5a{NO>`tjxm zlj9TVOmK7d!>5nGen|Y#_D6=q4~4(`$>j2f#{cT6)5lME`tULllH-Tg zf9@Wqs+*H4sL+2u;`#nJQ_2!85!RXzcAV$O2pgXeHiP?1+X*Y49WU=K?rFVa zAVuXTr27TBY1Cs2eqxXEz6Lw9O)Amzr^wHOR+OC@0}?PNNE@V?dHn^7qqg zeOy(lJ}ouYd0SR-BBYC@>{iQ0UeNDU%_&luZ6uU8q7lusN1{}#FG z;Js>u5B)Cp8Sp$=AcJnj0$)xCn0*#dH*nQiS*1<;J*lSMVDV9!e$&(A~8TI{CeMCV&f7g62^|v_H z9Rg96yPGZo@6YSn^ttE~Zo^UwOU1xP3O!r7*b5R!oLB`_l~yjaaZ!+3<5{T(>kXc@ zYQ4Nsq~=eWk~<|N&a-sbH9C)M;CRTy7lX{vE_{c>jrKU{O+JGHwkDITmHw40J*|tn#M(*q)9)+&`kDGJR?gNAHWBLltAs^aH=GC6 zXG!H>74D7&S#157hS=g|I+Ie5NKe(WYXe_dYacA~2TI*7KGg&L*`~H9CsFE`v@W2b z_5?K5*P3kQwI>L&>t z%M7bvhv8w1rtDM?YffVg_=+wVp;OIE-~r~ictDqBas@LalIHSwm$}>uxX{WimTcBzY{-nE4M)EwJve+c5S7XaEVlPt0S`Cz zhSXJQmGFn|T78ZXieH%)o6+k0_B5v^C#ItJ{8EqEmzGTMwm>QqSKV4BvSz`9X1w8U zQ!@L4ZizL&fSsK5nvk!)tB0vuHKr`!&ii)BZHi>@wnK@yrk%7H4E~N#d%O?nZ)aC) zEk^=Pa8_uUq%3YfZMHAyRQm;vlOvRSm>iAj^;72n3*Si7T-oIbXyZ-I`ap@ub>Je91Wu!5{tLO%9>9&q#>mGleM9?2%eZ;lQzwFG?EW z!=>iI+=J+N75S#+YI9$0Q^{qC4M| z_N+1QR1fNQo4w2(g>V%a3AstB8r7?M zn*ha*ilnqyh;ULlC#FxP20Sr(ZcPrM{06g>m}x!7WVxhQ3*V<&M*lM`0RuTB=c|Dc z?D^pir_EOn;nv|?xB}dcRGxABQ|7Blxbi{v>fUpzfxls^kDG`)bi%3b#x22hzw1OJgK#kj41BJCfY>h?cCsyNpn(*6g&;ZzP> z@xSAKPrd`B{hd?!aP2txOTktDzWDL)o%a;uA>prUzWIWmB8lw3Z2KYZi`FGENj%1U zl6Z0d{QD;cJ}3rI&kp`|WpRwIguc$f*|d~qJ*VSy4pq)Y$T?Fr6301OIrA##TG^~N zeF>9yIS(muXX5NQDOd7}pM{e;gv=%EEZo_alkiJ$5pV49c&+8b zUUm6P@uhB`72b*Oz`>&R`Pu-!z$)z!czSUH^CrvxD89t^;Us=DPT-Lvrys(7*?PYR zU*NkBC-r>;C+Qx=Nt<58N&GKx5}yN71V1*;5{G#T;qqtu=3D&z{lbTfAFiC?(%a#- zZ|DD?^aY3apF3aE(mH(U8yCLx*<(%q@%R_v9~nPi&Yr8=@g4X}@Grq%F>b!P6o1Fr^VJ&s2kKI1{Wqq6&(}t%kqyJp=ZlEo$U7I;Afw?vhkF?JD(-FE zSv05sSB>k$Nq)&Q2e%a0gZnJ*A>1ptw{Y2{yBb%8^Wpya_m9N@Lms+IRR$zD2`9o% zF}}$8OYrgZpP5dnql7`Ky;iW2a4A8$YU?oJcb_6$c+F#y)%vr`k?_5CAbw)_e%>=T z@eSg~)71epLS@g*V5eEd-dnHBP&WdM9Y2x_l`2#EtRg<7s-lDV?# z#@n$qFzXRYw_z$K?EU@*NHF%|AB|NHo$=F49!shG=_UGlo^I<_4UT!p?HtYAy2Hh2 z+&xmds8&As=b18Q%&1(e@YIywytZJ)_qNz>QO?XQZr&H=M)4s|Hf~O5WMy3gpyA(Yd_}cd<5q}wqifH^)gzR4NZvbiDpK!x4axWGwtc3pM?)B ze=z0dUCXEKxY;TpQkt9I;ObBvDT5PAfje?|%h7Pg&4YR6ymX{A3{I%zp@Ei+8Jyta z(TXx(rpL;$;n{R(BE)}>96g!7yMJ?T4sW&fkq2@o@z5A~B6l|a^k25ufw2B{xjFq! zxentQv_BxG|Kg%(_`2D5<<1~YxH7UiS7v{gv4nY_=%QU&TOJ_bnGAl^@?|$gZqvn2 zf5Awd4(@hK&AY7Zp)vKw=UDbgv8A~a8;n|{Z=S|e%H}c>Zoe;A=m*q|-`bQr?ims_ z@?LmzY1U@eMQZx9(|5ELRnHENi!{vLHhTWn%Jkf_}$lCk^N@z5`q60Jl{Wq5i{9lwROLKDu)t=JaaVe4G2e(VePqw>K zwo9DamRUUK`S3eeJ?+|FmpiGx#&Bs;7)=MX>1444z3b9*r-a5Q2hMIv&kfIuEGTsu zmky~soH3Z8D+NQ+388Z$)uqz>>3`_IO}8VO4h`i*Mi%V~WuzTG_-6`;_;MYyK~HF6 zQ8biijQv{-iCdl~`Y~ba(esEqquzKo-z4vhWBV`IactjZVDx6XD3`kEEPAPlrdYjX zlgkSS*U@#cw-;X6bm(@9?6{U8QrAJ>OEfs`P;`(SXA+{RGe;VuqjE7_8X&5*l3E5Q zbPqUX&}D4eEv4+Lt#4>g-AHjn^}eac=4QM|U2c11=53_y$gB~X7TpYNO@kBG zP?HqT+jv~g^d0;b;PN9sx%1e2?JNEdao+ z(SiISDuGEtW{8l4`X`-E@m_>E{HX*JPlo2?FqXcnul813x#+$2*0#1+Az0LeU=oxn z2m&g#u}$la6E!M^0b_Mf%aUVH7e)?Rzr z^Sei$0#Sc*o!<^IY8XzS~YkP=7JO0Hy2@I!~cvphyvLTTHm& z?jnO3_ad9|Ub+FBcIAM!z5;@WkjmPHTA2JMf0|c6p)tY|uo{sEShjM1f-fIRkMcl@ z3T;qNZ$_T;tA<-+^M`7v0lSOZq4xeyYHvS6ZN@0?{7!`*veXjZ5r0~{=G=| z6U5yF zV56SC7Tgbxy%zIKKsC`FGP+aFXUb8)r~r&?Pyn)Q9SSQ4?WFHaQg9!v!%C0WM@EL< zrKd0AYTS1abwWF)=;@2`4qc{}OmrN1X%XtwtDdxab%mvk7S2ew;zjIEJ|^}wI^M2V zrz_owRDJ^P-UE*V3_FEU4lQ1GK;#u9p0S-WUZxCT8G8ozm$0P1?$FjLZM14K8awbXFJ*vdU?f&+fG4c1zo*{TC)A1FK73s6bZ-ar#C8BLV(Mibmd58E1fF{6`bsIkwX+|U=t zk$P2(wett0y#xe}ei`kcU(^o3TV5i=Jl;8^M4K4XFZ0C%jl8sEzUV!0m>Oz{HblLO zHFPi14j2TyyhEU*CE6)>8XzC|iqkOGooMf9)W-Gc1!^km8~StsGkbKbqSP|R>cFmO z$IuL#O?!z1iub@%0A+NpC}WSC^#B04A=;al@TMHju5ujlxiQDfJ@u+(8FdSS{0U58 z53E3vSbDk(j0mVsaf{>B0#j91>=bm zN}P`E$HUp$)H|GCyi8qem;!yE73~*PVZS!18#F$Io#c2}6RINr|OAab`R*j-R>I zIXDKt<-pIfUz=J2v^jndwpDIsH}I>CBj{+ai{S@r4zkDsWXQ@OnPBQ~#)Kq%`wxL4 z^bq~mAH-aVz2M&eU*qk)#_e{Ssf}b~AgPT;5*pG98j0L0A1r|`wpi9R(JY`w%TH7&s&yRYMcAZh z-WdLZ9=r`fEz~9!v>+fi&$tbS6*v1D@({P1z^yb&Cqwv&rny#u#7U0 zS*c~wcvpEq&gB?gvXTupBu=66jV_vN)&Nkam#cMdC%Gsug4tR@mVg@;4-F{>(WQXx z69P{CMDAei5dY0P|V8W176Vj`p2F5s=P;4n!St=Uf2a$q(I z55aeaaAXf$V+fA~PBF2Z?LmZ)wu{$;pqMe6G(LfhHqtaew#6uXGA;l(4>wvOEyBGK zKbZi>lM9bsMfOQ1c?S8v0F%lB=4;smlfZNmCV5dzKDwF=4hS+@&C|Mlo`=>n__O10 zF8(Ny);aU=mw~@LFREu?pr>O@7rC)ip9lRom#kqd-SgO4>|TP&Cv+uI{}UWXg6-0y z9jzTeh48I|Tw{wyk>ONq8nB`MX#cEuL;tdIqop9-VF@%IxtmS-6v6r!NQ~0MhBx$nE^eGIxC7Qa+=x7j zai`T-DelA$F5EFt7Emw`kHi+sjU*dJDuY#QsP~{nHb~-M>t-*& z==6XX#4xmko5u-K(FJ$V3wG*4MmXC;jAZAuakHNx6;s(>KM0yDJZ+6u29-kRTs*^RE2`-F)Q9DgQVO@f20N2tooa0aecuWuj)xrbx9 z>D-L8z&IZCunTZkVW5kug^#K`AE9P zw}@bFh{5at9iam`&GPL;`0d@Xgj(D z(}b4%BiR1Ik{=0US*DWA6N|L3WUjWtu6eTET1jTPwj!fK^VslgEtmGKu;4|+i-4Cj zn3n9v7$r~Z8Fp<3HJw*0?dx1~NZ@8{)4)-T!6^Ur;0@f=N4H&eMz=@p1Go|U_2N$K zN6Ex~^iDiS?_k4hViNFOCQZ#VAYHk-9)bkU5lqo};DKC zMU#P(&7^hZo}zq9BDutdJie!i<2KN65~3ZD^q`Fuk-vbde635(djP>dkEMX@t-Cl) z{iYT&XEI${Q$x}-+kUr1%)HN<7Wg=;wTAGbonSe+7R^1?Uxhy7it5scX zQ{DEK(z)z6*kbwaC*&tl4rf+Dusw~bU62T<@;`Zey9mAGL_R|ADnk+ZUcguxrp1lz z{WDP%28mlG6V!dPzW- zD}qVN*FW3Qw4A7QxxK!%gvFgPES|WU+(^y#UW}iw6j^SzG8@&6ewKqK1IVd>B4-Z| z(X`Bil-*{SNZ>FlI}0-f#-=q<3Q}k+Sd}9Vk^<*U!r9K(SFk{uV~7)2LcGG`X1n-g z2jWumFd5Ce>?U-aN*s$doWdKn!J7F2pOa_M{pg(ZL*q3arWw--X+$lX0$0KOvQsDP ziNZwc{W+YJ*$|ZmD73<-PNk=uT>_DyPaqkdCWTm~4JZSmG5=mui-%!MX$_X{;s`*S zg5V+YX=ydq;QLbZFX*uwTPrcA>>;IHWOu!~nm)>({2G&*3e=X zO1A$qmRXukQ>ID?bLfl>?Ry zz)C~J6tKeON5zXHK#i}FxKnL zq}z=B7P8e$wD|$ixB#4o%JZnxXW-_s81%%bB|6>*81fns9a=m2vWeSKu}jkc>G4td zO+Hj`K7p^Sa=h>1o)O<#^pONKb|8zI=X=W0}&G;e@=|_$#zzg$T;11MFlEn*k&>!yc~HUgO&Id1!&A!$hkwg9{>llRZzW zjx5V6(s4vkudfBD7_?+>L6&I|npL_y*W(ND{?k08sef=bkuW-dnEzj3)8P?NiY#dx z@2{Ix!-Iq}7~TgoBCORvv7D`efoROsDd;MaDp;wJgoz8GYBdGsLF8~3d0+~lS=|PG z_Rvy^51O@HKVeWO5&ktmF&eC3jFS)z%0Ke50SpyrL;d5-9;MH2=)iFcktl;zwyw2{ zw8*uQ;bHS~$BloUhy7lTl%aapA zf#d}FSB6)h^C&lO8gF&U(=b=8G>6+K#Kr70M*pd3o zuo`PS38R!ag6ROmeQ>0?4F3oqdB9CJ;5BY}NAlP|;>*4P?p4=4_(`5U7(A1E%r4ga z;)nx+x(|ZK9V#BYcP|xC?_-)QYQ;8(uD;w0T;02U>tg^52+RGZYU7t zuuW|k4h7FpxlrgVZspN@4_gj5q#7g;;gmLvk*QO=k98N5uF^H0PP@@_f>cW@8@yMg zLqVo$E*<+I9ebu=c;nLXDEwPIoWj;-5@5}&vRf+ZAqXsBAc*+q2s12bJP<@Sc6<`5 z7TE=9zdI2W%aHb)$}8L+){my3v1-Ayo!_`Wq&0JP;$acQpb*J?y#-4G+dXJ5Q%`>Z z6Q5X;1Pazx4^UeXFZ(o&vmwlII1z!0y_~s`aiD)Q{7WZ&B1e+IwE)YIn zEg+h>*tBRWyL%odlAiuNh**H(ZrnK2>pM%Bd_hc3f~PQx%2@mfY$mD)KMCZ~yr`)1 zvQ|dJN3%c6*F$w)iPe#K9T9atCUprL41*W7bQ-NB55{dpaEU++!ncUMP=yD955(An zIwQQsV=VaeQw7tZ(d3ERN{FQNHWboN)snLc|C$EYp^;I2wH9xfN{FCo!?;)a6O}i0 z$o56-OK#NyZK;_SMOwj0z^R0Xrcyt>_b6NF@N=a8ebXv2BLY~6YE`lT=E&1_ z*=dAfQ2TJB!jx+ito$AA(dGZq^5~?c_M(l0w_yyBlLnahOfVke?%jN) zJuM%4d3#>eEImV=#1?;7k#z(W$l{|l7<9|CXxqa!LLH%_;R%$}mCW@^l-jg=Ixh5& z%ZDg!Iim9SN!kFs=s2Q85~asF_TIZ5hHUzJx~BSY1sWDLbt|4|(`Y&-OJK^LHiWtC z4Is_6mmb3tyOZdR{%PqhJb8TUaD^+(!p8_6V9ED_1++@z@?QX2F`70F2#_(O<8Q*7 zk1imVz+KR!*}#}L`yee~G<*+W{sUm##t>Lj@%}X}b`IMxcvKT(cNJt^(_w<#8QWZl z@9$uDL27;)xK_f4jRBAVfsf8|do}`?uxPXLy$}OFu^iS{tEeMctE1fsPE46VqzSNG01-mXOY|2N8~8xRjrrTqhEE8Vn3$d?|8i)s+Wp-Xoh^5 z7<^*Z`J+|%Fvl-dmv3*e;3rb^O(3PQd&oMB$mojx)937@i8Xxq9ZnZU?V`#S-hO66GN~MqU z(y`5&Z!BHPOV6XydyUfHSZp>g#s_4(Q4EV_1Fz?(?sp8luua^4vkP(We7rSz5w7}nr|S8m+NGJ zN$gId$p`bo)Sa{ePxOx^8BkDg3W*PrF7R8vMb4hXqEbLWOo6FdI!RN%sRAa;3GRWi z91_Rw#*@8kiaXV_mk&+_YY{2Jibt^C@~uZQ{dAiuWp>w13m@hgRv z=s6S&uIG?))pIC@+OYZb9E!5mb4Z%&ITV_u=Pcq^3O~_v^7xfDQS_W+z*Ntf%b#iI zPtPHxuIJDep`JrKh6@H}vPCciQ zUn$5>&!Lb(V8yQ#@TcccY>=MQ!s|BkYa_pI=hv>mq(N@#_M9&Ewa3{5qFk?fja}ubKRs!LK%cweqWlUqyZu zaP?b!t{cEnce@}+Wo(lMWTl|zeiD|2FkH{W_gt;qHw-0z1x$o$EQv9DQ{ZBC(ra~e z&B~zZOiDwaNxoT^#qdPbd;<kOz^BDL0#oH#NDM zT@3^}fz3L?=HzVvcV8c2pIfFgB1UBzXb_`{FIx4VY$j0jGJ=9&;6!HvsM1f&~i!^=|>!*NZO{WKlC=s(f z$-}B)U{#~k6|g4~N|Y{1`5R6;j4KbR&s*lTMtY?m>_KCdUaf$DjiNKdKxauDoqco0 zp_7KefRPe5JFxT%XGYt97ZQfrpZ1OIpC;F$e1qRoo1gUfeTW-Q$9S+pw1heV3Ek*_ z5FKFWAf41a_6fh$X`O4rHV@0BV(eX{b@MNbK)OVG0>XH-&$~)2G=ldqb1iC@h+64- zHBBq^sVf96&6(P^7VS;ZQcvQLa=G!mtF{Ijr*WGFLBHA(QCYX)>rSo1pfL16CETCh{2Hs& z{0^wB%eGM2AX|n$=GTbw4Qvv6Q%V22`G;88k$Xt0+7{Ie16px}pdpr1f2jW_#6b-f z=cT|KuTVn1Y~4JU9mT%S*y^o+UmYW~PAeYFUu~1KbQxw82)^sv)QW{#DS{JS`Symd z!%h%U+GhfK6u z(S2NBS!3g@G>E3abY0HWeKZOr>Kta^jbr{p$!2pR)rtLNZh~ zz_t_2D){!Dr{r(3#aQNy?d?SawMN=ez<8b;9Z&KARBAOA=Le{kbXm2|e^~A3s79l? zL6q&G;sG}z4E}{%P!?)KYJQ0F%g)h=_HV*7LTM5aRV(eADD6KY?fZP`P*HN0pjL^2 z+QwT{fXuN#5=H2vcW>2*)89pp|X?cnE(jI!{J%901BQqafrpk z2TW+8UiwW3a?8H1H+O(#r2T`yFPtguKOV9q4lT*D22zo&W}&FG|MRd}+W!UKCvkL+ z7Y-FAW(k428l8m8k?{TFa0%V1m&dVruKsdtt^_u68OQAJFp>860FA3;(GvyyPPANMQ22M_-WwKbb(_sD~8E5uSfX^nnZGr)Yi#jZCy?O zBVdm6-Da?YoQ_7(v~oe*c-@!J==R4>mby&4weFWFSLqivT9jV1xktAL*)o{;_D~fzRK4>Z7-8h~vFLi6yq0Dp_|SA@Ui_*;d)@8A!# zhJW#gnYhE(_`?j`5jyrm8}3%zu^aP<1$PVXBJLvY0`9{2u`y`W2zV{{dmMky5$xF5 zl$b$*F=N5s(c3ZOfWD!Okv4AZqM9O`DrsoD zTHG7LwoN63N2s{h-Caz-`t?wE@c?9ki4}06Y}L#9w85=loZgm zWgIHDxkq*BPd8F?)~K4KJQ7ebP5X6#BNNZfK+ULvB^~;~-(^$fF$M0HseVShekSUV z>SzMtx~#Uyn zwH8$4%H1a4BJDTVE0HA|Q*ga&iy83*(5%w_36~`n7A{*R?QbU~wx}K^SxY}wruHA>Vosl0PyJv0GVUb{*K&3VF}~FhwdJ323oKy zG-_9K`)iO5e~BR~79H<9JLS3T;LPz;?E9-Zt;d?vunPmwedqFPMfn!Mce4@jc6D`l ziL~$dP|=MrN3L+ST5mZKxRu)|Z+CnsPI^7R%py-J4QIiu$*-~mQe7}RVCx*IcfKKa z<4*Z$)XemlJq{NK3I^Ks-sn!fLJN2tf1o*k^Z`dLm zYvn(UGu3>N5Q!U==+y)0)gbk126|P_81L1hnn=JKGgVf2TPI8V4oUkWLq#{EpR<>= zPDVe|rF|D+NKER^FSf|(p(Q#ZfWoQyHN#;urI^*S;U%}6kUQ%LSj>vmiiiLAU9wV_ zzyN$VbF27Ec;lPBH zNaK{~Z{o8{#K6(ghFEc5$57EC&>;&9)M~n=lbfJcj5o@(*H7x<#g5=58T={jJ61AO zbPI}Sv`zy#Xn~OivLXBJ`bmdSqNLv0A!iUrJU*V`_*^n)3XUOF?sK$53OPT(IVNY9oXt)l`rR;6`C5ZLTE!^{MbiHVbP@-;WD74F zIH}f%mF!w90jgDbc&O0glDAZ{^AM)hI4lx?1wDdQ@3D@HNm`Hc;jE4ilg*aLo`Rn< zoOGZU>*b*(_J`zaspt$UYJ;hHAxY_-)zO=57D7hBHeS$WZwzFz#g}1rag~8tS)ZOS z_sMfAS;{8JxClvKn%tG_{&iSmBap$%G~_6UGcR|U{0gp;P8!BjFifjs!h5V@lebXC zv5IX}t!=w3&&rh^7b>-vWsL=S_horwsx6CUwH?E{uyc=T2exwjHk|2_C$!OGE1c}? zYM7vP?WG*Q;AT^{l6={F4n{o+&GZW%bs^=k3jj5mk>tN(b#15P?E?fioV16^tCmEi zBU?EnG+l64;(KHt`$uX~q8BB++M$j}k|qRc6ts)VN3%LUO4fu4G+3guD?kBCiRye* zU)prRRgTJUl0UYS4PeWzu|OoJQy__b8_4MksV%m90Ir+pG9I27-aqHy#;ck}^3e8{ zE3`Lr`S5pG%h?W0$EFn)QzLp1o~U$XV`JfIxQ}8I8Z%0X*czrO?N&D{M`5kl(o|}c zhY&BeP=K6KB`VWVgZJFov2j)bZ!n{2okf30*K<42Q4VMt*bD03sQWB;eeo#-L8)A@ z0cyk4XdQ)+{0OHUzNt9Emt_(0V9Fc1^SqqNc4k0b)4!;XqNsv%?WE4|(xDQI61dsJ z-h&6vgNTeZSml-X` zoH|fJZw1~19*(!q%a;6N>y|&ju#tY#U_D=Ap?zOy2esjx_9vVREy+-eBjE`YH;OF| z+Ch93$Y?7pO&`*Zb>LfYwZYXgSK9R(O26bhvoR6zH@ZBm-^6_Q#@e+zgoIn66LbHR z&@dcc>8y!tT#4_kz^!?gh@k{WH&Aecb(lE-CD!B@U_re(kmu7h@XeaQq85{wR-U|aK97C2iqSa^nt7i zv97@-psiMHtT-HHhcI`sdEWHqG}p9_aEy6rT&FT>Xb{}+5(*TOv)u#+IR z#f0|kX|WMDg!Z3S@)35lvo}x;63)h&u>VAIdYoZhz_o()vJnO=guAFtJJlin>Lm=E zgu(l24QULRYL}?-PH_0s>9NKQH`m><#>3l)6`^8z$-pT``=Xh~;~UJ|M4}n%5;>LK z!vTOvP2tyJo-CRLHMBAg)IWkCG`Ei&Wo+%kdb= zB+gNg+Vm|{ICc$IV?RMD+J4YCtk&JD-0a^t!whJmmZFyRVe>GjnYfI?=XAM`&6IF# zmrE=U+fa=9_u=AZ#khjEuj7-pf+y`XlJI#Tz71itjo2$-peBc}guv6vSE=q*TKQ_# zy;>_5%2l_(?1WCiV6A?oET2$CGbe&)bSS9t!13B zpxt&4J5+>C3EE1>rsvWJK*&M3QNIk_!G2rBvAX?GN z))6k1_AeEwQc6g8?GehGL$&B_7rmuZHF|5Mw+VP?!9%Oqd?b+6c%fHLZrqvxk2Nhi zqA2g7J|K$RgTOXSh&0%@>OxBE{}Fnq6y)K5&)Fc zd<=O}3iP0_6y)(Tl3>9Hk4VkG!{g>6f3M^1)#pEly|Z!A_m0>?OT+?`fTOEZSrKgx z90t#2b|XLlvU9SVHNl-j6EdaSL_7%Y)XreD;eCYM*N2MFL25=SqqImr%$p1>h33L} zXQ#6vvgM|rzWK(`CZS&0F`OW0DYu#e8I3z8;$)kL)uK594HY&>Dr}|Fg&<#4-EzE= zU4){5bD^3xHbVFH!7MROGGuqkg|WPc0d;^I}9;%4^qDMU*WY%sigd02!nMgzqp z_~bc|;3&RhGKv!n7s1$Pz*rN5!L|XfZ>W6&<*2qRR(l=Q-fGm&ir1#E{C-32$D*|p zW3?Np_AN&3KT#s#&n|p(+umsH!!T5$s)t*JT8TyLY5dUac)#&NqU`zww_EbD1L)Om z%|%euj(J`LBk%hzYPs2uu#SR`f&nL?UUU04N$LGsEwO|rkZAy$unrZKdI2K47JHb{ zxk$z*Hb+xDY$3*|I{YmU+YUaaflDHLZx#$IlM8|cTb^ZT2#2~FQSSVi>MPr6@Swo_ zHs4d!+vT&N?sc`HPi;60!Q4-sfl)|n#$G31V@M&)XcJ#cgoNkw7X@lH(~t``tq0wx zIEQAd9d9pHtA=?IO?X}_wG4GAfp6tki88qL4#JEk)l3~0$Bg_2QML_rBnBj$Zi!~e z=9J>IXqCZ4GnIs!5neC%a&Y>B=RIr-2&an zd8mhawh^ghN{K|LRB9=F4e??v+#-gFB(h(VSosFdc#t4c;isuE!9GeQi9Z>^o<^|S zIdn30x?p}cV@))6MZFYS4IUO6jyg6K&I^C`8gnOP7n?~i2?I6^gB5A)=6iuSEom*Z zq!n3TH2q)7`O21^G~T`JKT!}a(ikOTljtaXX`SrXC`IcUG;u9&!Vq3#+?4YgIk!c* zXiP@lWf>H^Nr+@zmL7L_W?q&Nt00U^*Z9UKcE*$=39w{S&mxEj=l##Wcjeo2j;rbP zTGPoTtI~sG*ZE0Cfevhb`!jrE9w_}B+;Ldf!J~~~;x0OSevKtCi`gc{=xx}eZx?ZT zDuyxqbtCv{uA!?K7? zC7>%T5Lw2%ncj;L`XJOx!D*<1-NH~6Q!9tHH5S^ZC>u~NjRfkjR}&J}Xv=rnLPa}a z=MQRFH9$SWTXdSa;Nn1|3BmU5u7{Q(qCo>aM&vge8+22^t|gC0VRx^Af`_AU3u~nJ z;&Y^{aGC^&18-CK2&~_x{ChUt-wW4N0pr$gwvXX@Ue(~tQ5!VpwTfm@p%Y5G<7pG+$cK#vJ4^pL0xoqvnRj_G>1Fk z@z58pf~CfuUKEWB(vOWUjC|{+Q1B8#L;ZB08?Ug|RvYh0)YP#kOaj*fgp9S!F_bv! z7uk#8Ziq&ay##p_)r1~xFg!_12EatL)g?G(apPcY-9+jOYXS+Dn^jB0FemEAc^S*b zfmj&~5Wb&)-SsSX-GuQw1}j&|tf4BuRmdoRp@6(zE2A}LuRj`s`m zv#dZ^YJLqItY%s8+*)k?H}b+M4Hg`DcoOOQ!ZQT^_cx)M7Ev3{!06kmFFZ+!|IQN! zA^S+(%e~5{0-kYp)XN@5R(^8e6TIA^-^8}#73Y=KL}kj-j^m?kh>IGML{UWS#>e;8CCp9{A9xJ?mwduSOH zQyoq|)e#fGg8DyGO#eznXOLL@@15;Z@P4o^QRy~d0rl!-OFt%FhH2tud)E*-;2P^Q zC>eoA5~GU)?Kx`+=eU+>Y2}RLods{V8#LttbCMt#B1>q9Wccm4|=b!l!t6fd|Sg9^h7xkLflpz%3Uk z6yO$(ZF`$y+iDGrqC29o9X$F-oDt)eXT-Rb05Q%HQU9@EAZF}_UddzJ5Zd-Tuxo5X zle-TOTjFFY$b=wBgggFeke>$i4GfGC1nNRaJMD3|mEDD4tNbLB$z_sn+Ort9rZ#$^ zkIBn92PQ#7*XOrnQ1*1CEr5I4<}}!4Eewx1wgS7%(N0STAF=mjG%yb2qRT;EGY;g8 z7)a|FknnMZ1qy_P)p|oB_#A>%^k=ebh~(@xNEx1BWz%@Vj;$auEy_;9_X@YU=)aPs zB2>IuPd^EDwD%|py00-`_u4=Xwg^wWl*m4c5WuIJq6sPN6eXb5B%%%&51yP(O4>6B zvvlt{T;O70xakY~aAp2uT) z^QW$ch!g&BFY;qbtqz?=Rw@nyck=tCZlSS--348&yDgb)OS?S8oJy$C{JCO};QZJNcvrftK2sxfuSv&Ymh@ia={MoU z2$nt#cSxa=SKNN4R_8TFGwasGVye%gnv%n#vOD+C4VYr@Wa!nocvcpFbr$20AkPd5 zU!B2|y2$qwmacl9(w)Ipz`7uFB4ELXnC#o%2J-muonq%Dp&-JaYjn1<4=s3Jq1tdO zK*>*4u<;RqxH{ow6OV&auZGI#!l!>CRoJ&X%S+aAgDizg$~aYcVl@A6wQHhcx>(c~ zyhdR0od`5h)2RpWEl|8Mycy!;bA+_;rEm*Lfl2U_ICLyXusX}nqP`*yJsX}^#G#iW zb<)v1Qv7h1qOl0TDtY~g?MXRyy9lwhmCfh@7xpzULk+2L~kM21<86;RD^{!AI) zlmU^v?0VQ&+}@fF?;Yl3HA{)tqUF&f_15qNFH54Vn1vBrDDMS-bONjuj1*H$2>cEi zU4|!2o{BHbP$ri;li6>{^`PGr-Bb88Yq;&ic{+m>0Vmo7wiE@!r}rKM@kb{NWC7X1i!IN$t3T zW+qp3x@)OaY7YT`pVHieEz2q~*KI-IDavh3QQB?TeGSLO1GG1q(3V&sCoEoVBzK!d z=^F0%Ow!)^?h!gS7}}Lk1O(NxK`rfeYy=K<;F$0UWK@PS=WA2ds&krWSgnd^Rp-w!5@D1?Qcjy!7|Lu~E60y=jK_`iXF8d7^Os#O$?gMI$Nl z93U`zMmRHcM{hjmi8ivfkeN)G)dCi|TBWG15#Z1?mz`7Z5@4@rxK@a2r3kdN!!RAEm19Mr#sD-THn)W&;>xnS%NtwZ)IdzNLic&Jd*YZMEkabL{Y60&rQO8bv z9BSrxr&zIJswS&NdBZTObWsX^iTpjpbmR*^LkOlG2198JN!t2MF8pF2wrNT&zb8Rl zS&rklJfByI!cK9E{8e*VjZEd~78+6mE>cHI3ylJ$*2ZmxCOJh}CyekEWvi&m_u-2+ z2vXNi@g#kNb##W1I*774;RrU@(uv=5dqT@650r;WZ8+U6KEuj9bsY6)z9-2|mHC*>#`+iVffVhP3wIg(dYZGrwg$x1>xP@us zajRm?L!%a-z26(MnR$8N;U%guiP4na>M#PIz!MZ_mgpm?}JQeF-Yz{wQ&w;!k_3jqAM)}q#Q%LhjCM_xPJyE;Tj&KSGUmJ$nZcfNhGOm zm*3E;z-W;q+JX-~w(p3An3CT+G19iVg_4(aa}yQwPV$xRoNHsStK61??eG z-3Eu2HTO-#Hqhpq^vE+Ptu+iooOI~I%Fy!p!-~#@Y@JqfE|hsmt+7HtQvDjMS}m$~ z3M5hP6mfM{i?X+M`2j^mp`zVI+FRjlo@WiMyi9mB zbEJ_Ro99*%#AU;wq8o=*QC*&Zh(e@PTVGgqVT7(R) zNKo$_4y|Bp`eC;_^?(w^5pP{!p^l}f@ZfeaLwT5AefB8#kR~uT*uZc*_nFOvX(sYQ# zm-3Qj5Y}brcUIh_pcQ$_EqItfh0?SOuzmtlZpYaf0o?qv|g>^63(F$2nfUT7cUXXTuie(5a z9A~0j2X}yTsUUlmLpE%BVx!W@^(wec+VvQDG>!9){*mxa{^9VA{)^!o{E_f={tMx& z{g=X5`Ok-E`o9Wa;XfCi=KnH0#Xl5I^M8&*$6th#{e$5||7WYiuW-%G9ZAOe0YdOG zU>QB@YnI`Y5DOx3lWo3&eFT*`cIoiUwKQpL;Wze@nGX>VuA%TK=SW~Foz+{~q*KK$ zlf0~rb`qhguV8yGgYoSsjBK^QSS!#H8?`vNR2CpD%}=CKXQRspWx%$F8yICGOdfL* zUvYO#S3VrX0sw4K2|md7cpD}m$mGPQN?Ak?r$VhMuVV#u;$vC{q|*9Cdl&1DN3GA{4k zjf1F>&RqBkriM8#;s7%JI!_ANppw0Z)N1WctD2w{T9rk5;IC>HN3zhWSz*#B7xBeL zZ5LN;+Va$Lii)HX8zu?SiZZ+q0f0! zo)QHNp#JPS*kwnS1&_ckD?6vHx*%v1bFr@aJgVOy(P)a*B4o5Vfu`6Ug7XehUJG5# zHFU$(+FN8m;7r&eXgf%`Icr;rRQ!!dvO_Byz`|nor0_gwXQS9IZgLx?XU`bH=Hvi= z->wFB#lY-4h3G;Av6gp;+V=%@n}7zj?~~ynHPhiXAQ{zscZezqlv4q zgs84TiYkVd()Y2CV7Jxy+MbT3IXYN`&myh2TSNHE?M1K<-|C)UJnVSQtx`B&1hO~k zqJZHX{|aB{*bm|vJD24#Lmd0DICbn8qL__gXXe-B-U+0@{Q_%k`o>Rjk(4bMTm;3m zV(YY{Wt^(-}T~7 z!dK^T7#`MzJQr|Wym^s_9iW1rI?Y{v^MOZk{@Z5)klY@uww7$><>3^Y>=-i8*lvJ8 zG|P~WzBY&G|NoGw)EHD(8)ZOb__T&WC=`15@2P6&MYC$Z?R}geSkii?S;aosazq`rbe%HorPiyt*@5Lq)kL=xE9o zN&(3w0Md`qQxX&fc~fgTuYcN%=py{M8GWmYr3Nge0rH5o2?G3UZBzJFKrl~d+hlqs z=chL4BcbEaLnWDBH(QL>9ql+mqjg}}sqPRFoZj)Fby7!YhxM1g{N(}q4R{PD3g^E0 zDphQ?CA3&wE9LW{opccV6%$Vrk+woUA6uy6S1yyF0{O38Qmh_g4}%Q;eYd_@(9_Rh zfL^6~WaeH1Z5{0`%}Ml0N*98=&$R|WMbWMxC1f|Z2W|*G`56+VeK_W0cJ^$UmfPXN z*RE43-=AH3X5IM{J}(BeI@KMsMsCs1Du}&W5g18jT2i zmR!U%0_?Y7CoQsve>!8?hCr6`(0Y7`LEGX(Ae-pVX3PFWYP5@)us_o2Pi1wqcgR)b z79Y+pIT!NpYutoWidW#E$S{Tyb46+zAx6P(aIo~3P<{^;QkUUZB>98fhoTV_jroDW z9H#Ii8um0(nWdlDpHC**W5n>#F|*Hp4?{ld2}~koWJI3XokWiy1%#XMXZ#2NVvCER zD-` z8$Jd2x9}8_KWj~EMi}cMyx?R0m@G&7{~ga`z*GQ?R@MhF-#e0ru!GQV2WgPxzVLN8 z?2LIfSK4B+v0OziHZT^yaHE)T_b%*?=_9Z)wBK`X@EgP4c zvjms3WAhMT@6ZuFu>b@Q70x^`0#J>j=(=6LpcM*i3Tgl(yue|ziZCccWEqm`L6TJA zXY&^lk4VAQfD4Bf&4F5)1W5;J62uq`iUaJm`v9#G&cO$X56RGJS@1{zAA;Zvb)<9L zL10Haf@*EP9-2bW&=)ll*KT%%x%62l)PZG*@$eX}nCX20G5p;HCXGl<%Rg6Lrd{fNOpx8&nX~@-;mn6JR`HE<=AS<=r=NGE;c~k zWsrPoX*p9(d4EHM3Wht5_Eath;)q{-kr2QC@ zhT*K%cD^vKz|7GC`^kW)*7p%SmOPq%=>>EI72A!9MS$$mb{@vt^~CIM_8tmoI}u!N zdbJUk+-^7P=1Iq*ZwK(E?`y{!`!yb6Lxf&Ai8pORfe&H)K=yIMN=8WU=dhjho7oRg zthze11K%Ftzb?@9P~;3aBrsWi=*|RfmVuUjuuoN z#6g6KnEA2yPL2maQ-_lXBpMn;htQy7Y6>z4HBwCf-_SykV>gfg!>6pED6d2xrXx6R zs@H!Lc33g(voYHAK_t+#6Pw02qkzMs=6Z1u>Zg=N6+LVgMjZR0)udv3Y2=_{ z9V9KZZk#Yh5C|fnNibR9A7JlH0WS<1y1irr?7|(;?V0QqO5U(Os^IHSQ|&G`3pG%k z?ZC|9G9py6UMrwaYq9U5(J|53e1>+Q=(iC^ip}FbV=Cc9*u4a~bOWp%zv(XgwBS{x!`?yaNfKMwJ4ii44=vAPpw-lal-oiqp{8bI9&0T$N7dBT{JDoeufx>P z8lS`S-nB?ZgrrZLk6yluxMLo`Z!Ci$fi_Y9xVYq}8}Z?bn1KTUEX3v5jlV^)nD|-) zxO8-5G3eu<`p6*RgEh%Le4Hrwa;mZj9{2)d0NrdOlvy0W)hOJSI{2%Xus2sCc-YD3 zk;$5X9O8#XU={uo68N)){mFQyl}AlKg<)X)^QbcxlxOc)F&q(Cx+3#Axs*Z-U?0X^ zV)y@X%#QvV_8L%QeP4k9%huz&Y0G?Q8Xee(C|ynfU_T9dEso^3fFw|! z)IlTS1H7yoZbx9-2XOPFn3I{8!=L+?181oVh$V^&%gS-G66^k+g z*ZU=R=m(8R^W!Y@Cf)`P{l>l+_PmX^(em>jqC~jE=bC|4HTEmmk2TT-{ZAp8LOer@ z^IB)g2m*9T%A-hYymW)qj3OqthYmf8VJAu@$kR|@)0W0dbpae$MfiZ6Me21$*V=E?YR8OpaRtkKQR))N+lR(zd4-9QXQrDh7qYKAE>3wOQP655hr z;K{ydB`I574Fd={Fn~sdBPl)lfC4`(zmZeYf@4bYj_*g92&Q7F#04TvbNh}99GrDUZY#%bhpU^~k6m7-iU zzu-Q?o46!SssTtOV5@J!xy5rjep7Bf%>$Hx)i4PA9Il>-Qao(w=oLbld9lXfnlReW zi?@9j+D5}o4Z|;>)$J`x!=OpNT`O^GS;nF#fI}Gg=C7QhZ+5d;zrw`ON>-tv(d1Q3 zL z*{9a7Qfrs1wQjYxv@W)aX&QI|qcpmZL5w^NKF&Nf*wi3j%oJy+FRulMsKIr(vOPGl zkn{v9g-YkDp;brg?V$&!fLcc(Dr*PBg?)GnpVrC7jt`;;H|NROJ~G zJf=`^J#JW29V7`%b%K9Iu957nbhA(45ak>tHIp4C?JpEGP82mETF4u1buYvtN@76- zGbM`C#z|U`+IdAvg*oiTHwz71QtlzBG^OKyvm%IUP=Neq`7hFN1Sb2FD|P(z&M#=oIoTg*kZXZg#35|~)9XE@wh`#)Fd_~kWa2Nz0 zWf+_!ewT%Ck^piHC!vBq%x!ll=%$;G1|SsN3QXLUP2I;(k?l0533!&o?UOJSAj*u% zrEe;Y7G|CsVnehQC1R;=!ni%jtp)=C&o(TeLTYdWUO=knwZc-k?$qE~yoc}!Fr_2; zdq27r61OmH!-Zq^S43UwU{5u)W%1o)Nd9yi72X)W@jA z_SeE!X}k{b%?JtI*wmO#6}&<-S2*#ML8 zXg{E16zga2DyEX~jy|;U6CbTc^o7S@Z5jaF?ACuHvREj>&V^};CD?ZaM5hSq1gsd; zon(vANkBQp=sbO~wHP}~7LUHW5zUhPov-nM)L$gZf}m{RKWIQ`v0iNu8XGr!lwgYV zKM%HQdYML^FITPyIB*K_)#5z;=D1mwjwR01fTM3(#h$=jxo0_82wW2z>7JqW zRE7DRp$x~^=f4~?10DzZqcZ?aReG@b_=Q*H@&)2I)A>dKUzd7=ETaAsAQ%wQithTC{pw1cx!YzrDleQZZ5b`(ew%_!EiDl3XQnN2od5zP_m~UOp1vKqU>}MolG49D=bR?KB;ip3b6X^#; zrJxB3*sn7mC)e*Y zDh&i!j?u6z&gMi2M-eE*N>;g)w4>Hx1tQU%4_;* zc;N6_P3wvm`Y=@Zumpnb#Uw=0<{XdtVT|LRe}Sq1BS7}g19!4YY%|coqds;I5LH%e zpinD+f6+KTmBH#-b`QH_xUGfITh%uSOI9x@6I zc^VmXel*KR(gAiV`wBjJ-f#Ri2Ys`cW(r{cTNE~wjww{T9^TtW0NcxsfPB^DHj{6R zy3FKjHD?=N{H{j{e+FN7D(x103I*WUWB95H`E8BQ*L2a~NDulD+I_e=!zHO8thegc) zs_^IwkJ3_iq2@|6S9x_Wh=&+q+^-H~)7{<7aNr zL41Xqvm&8obS<6t`xt$noA(>^isFSIT|o{0o84FH3wvp>iV1lfA&b?uQ33n}OB5#m zYRSSrDtRT9#IZ{J>O-8I)9APfWsZ<)!l}trBDB7TNKE8T1KlzJQAy=Q@Biv8z6#$> zvt=h70GE*+{+h$D0TBEtu*17c*7~l1+ZBQOJLnfd?Z!ypr&h?5dqUkwL+2wB;lLNY z6Ft=(8@hIqFQc$)=bd;j>@M+(D=%WzL1E5vx#J( z=*KwuSH!+5GjLr@LI!TZAaL>-^!_DDGR~#=#ajF*I#C!$hUi;3{A)AGVTj8UKSwJ6 zmHIrS*BK+{VGp4p-j3+EvOR}UhI0I(uZ{aZdXYBl-u%YfV&iSC@#Zq#Rv2$Z#+%D{D>dG3G~U)0&`?q%vnc_i;UFq;|IrQ~+kl@|HazdfJIrQec&_903#nhW1^y>l44nyQD}h!I-nqy*boVl zpY;%{_l1D|JMccJkPoBbD#TspL3u4+_$Kui|>jsRIWiSzaMcY6|w{x%WYgD zC&z5^bKy~x0teinI7mB_bZ zOgFf$zPy7E*DFmQRxmUyC(Z0WRoOz*tp7N=D5IrgpnD8~KT`O2LI=T}mC01I{>gFF z0nv6Y?;0=S>=q(XlH4vw=G7V04@zD|`-kwhCM=8z^m-k1p+>LDAV#oT>^0&M8~KzbRFo5=-OX#8w?2bEe`OYbl->d*E%p z*SV);<+FB_FDJMe2q@V)Dr_2Xa`b-n7$!Nf@{p(Z?^pRE3YzCw>Q@g?-Sq2Mq*@lv zVi8VSU}7ZXYwGxb)Fzf}{m@yOd9Z>Hmvp6^VK&o3s}u|gQq_Z~EmFqWlhq{;fwB7wWG@v4 zb2*E-Z3+&Qd-crIX)OUoUa*DRvUFK?m{N}-Ltj9o9%pR=#M25yf%9~#|}N9A1f zLg=RN&G_e?d*WeRq4uhX)3xF9)e%^p!IC-5n9uxnP1ww@*N23hnc0lDM!Yr6Y;i@B zP52YZM`yOkLvT>(_pb@*{6uY7LRQB4BnE!!TEUuDd7E0NnEGyElwBvWT7vh(1)l0t zcte+D`8#QcOyTb8DHw)5)k*k8{kf~B;>BIvLcg9<^!xZp`rYv%{q8cjl8tDZmp_9h-rZLMy@L(PIS0&_D@$!IWQ&(09)_Oughez)6m z8ZXV9sZnM_eKTj4#z1%VX)p@^$LOU_?2q6j8E+oWI7XLvd|)bzg8mdSb0frH*=;`=346Ru!Hgp;)HI{(&J}MAuInR*NU=c4%u8yN2n1G-yf0lW-qngFi zLBu{oanQ_7%ls`!y@FC-=3NUcU9SqGIZ8H0)YUTYL41m8%e6o>T2P4XWH=gz?b7_dzs&XLRANn$30UB-}KY?E#5N1dM`bq&HOR0g$IV7;*ZHE`Qy3|@px)$ zs0Ao$mZ7d2pJ-%t27G(aC!#34nyg$khvooAjcz9a>$*|z&7p1h9YQTFv>k8M()hmL z&70uPdIy-{ZBVqlIy*?(H1c`G0ZqRR= z&Rf51ai}*!)mX2=bM>ET(*-nhQ&+*7*?XTkNo9+@TyrmyAKewyc;k zqureqvy<@26-d=fC=I^1!js0w>UZ!05q;{ic=VJoa*>l2ug*w9;-qY!0RP(DA|}}c zFuz4b343SpIQzv7QmJYy0?OwIb84NSzvgDo+}QzRYMuX!4snqH#0B9EmHOUkL(xUPFYox5spfbtN7YKZP|B-qGdbT8sIRu2x ztSAF^rt&qZ%i@HX*p?e7ZT%mBgbSL!F}>RmY zhnYzD%z`#*0gFU(u))#JLMYv=t|D&{uNqv$Y$N|Tymn%+MC(^oG(okZ&(hHgX;hGS z_DaOFWyMQXpCO0&(6frkL-kR;v53FW3`yrKsDlsH8B=+IqO<`O1qxnpcOyKx!wY$n zp*&sbYFVEGDW$NGHT+9J9o&?v(cZY-_hm++t_&pj53REDRHt^zQMC>heYpUBJ4*?^nlIA zH1Y~_Liwr!*LQ8l{x-Q7|}eqhQ!OM=Y!9NaFF^o$xzb&S^yvEx`_sqg!Y@Rt5p)c3aai4Z)I6`Z#)!x+)f@WLA}|PiR{jvME~u5@H5bT4R&z^PxtH)(JlIpIWRQ znPD)*8yq{Hx-;>FJRxrf^ueAZDDM3%YQ-G}I;b)rXc6LcE$Pha#-~{x%9es*Ox>hG zOD7^YHe=fnRQD_tgun?eO$$oX4V`c})uD`AQB}S$&!0d6*?BMw{|iw`4;IAZXE{|8 z)!S_dswhNI#R>#HSZv47uO0YVw-P_)r2yQEu!;u}RsH*iNNK|Jqhh;QGZaSTd;ZDqmK(bD8Z&d3fTF1vt~X zy0?;Y8FcN}> zs({@Xi2VK*;YJWsTFxp*Lhw#^t(SGbNG7gbQD6Mwhk z_HQ)C3HIy@7%#D~gv@3Ey~2DWsWcX9*SB%1G0T%ady+Kej&gz9{KHJ;s7{ShRA>*|=~U*60rt)j~$YhNIzQ;{|(o zj5OGkS$yQKe8kM*_;XRK$!Dr0=mx0I+fnUN4ZE;ca-XoBUbAyJY)yMLZwuFFA5k2 zYCe&wmf@{-xZA_`Z62|cqkUM-<7e?(^*+=^z|S0IKOx|pigs+$mPmxv9-11Tws`$8 z>A)fBKqW0Rl@UQ~IpM4nrD@qU>4y|`Yg{x5`!{21`&y@Swf_q#^U(E@QoEo#e~3XtI6 z>b)-T9`C&p@3p41n_S~6e`B2P8Z}3ciUQK2_Jl(TDw!kP$Pso?A^R&(F}x*s|3e6? z;MbC#zmlr{Mb8Ie;f|gz*h!}62B~T{Jr_$=f5X$gn_7dXnmU&Qu4o*hlG&dyQDO|O zt)`Y&Bu`X5)bTvKspT;z1W>r9A(E}a>iv1FmHJ`i0K=OL`qWUmR!LhK24Jb`A@p&j zD_`1r4%IG(V})hNRy3Z)__yTbNY#&GU4e@c&DGQt5z8Q2zbG_&Dz8q@j&=aKa}W0p zl0LYrpT)1_$w30mTs#p*4Qm%ZJNW>Sgz}ZDpG07vdUE5l7wn4Y1*yD0V9a-glH{fW ztwtbPK_}2G-TNtRsp=otl;8k1a_5H_93b>Spvr%n`S3%T@KyjvVJP)a!4vZOP-277 z0;4=XO~A{Okdvgn^ST|vv3xaU89E#^EI8PfI>g{dxf7cCNJokIhS$#PuHlHy9FGD$ z{9AD4oq>c@xGU7U7F2X>(8`3Pr|BYl)O@9bP#B^-$T8a!51r`hlZiG+B(zcB8@%p3 z9>}kQ%|DBV;(ZsvN>{10^%By?whq5a*FDmfH+lHI(r>3@u2Q;wC9hVxel1T?x*n9O zp9QRg*HWcm8%!AFyR_OsalwAv%>ka-7@X|VmhpN>OGUJJIjUH#rlwg8Tl~-(-~(t_ zZ^3x~Oa;t`4ZBDJW2Qmn0iFEDwN&*YSXIN-LbK9r@%|ehu3?oI?{(cHniIYaJ{-u? zFi;Nd6?$&kPrr45l+RXqyWG^Y42T8neyGIM@<<8bw$$UrOkwadHXtqN2+E)XVR6#i0@RL0U{m2Lnj9>s-UMIzcAU z`>BoOd+9=*$O=`P+p`a$z66D>m8$k&SU~fxa#Nkt+^`1`oB6bTHE{dYIfh8ImtO(1 zmLr(+nyNnpsid0ff8){2Cv~(1gEtE}s5C^7s&6IA&E}4Vcp3U|v?}Omz+y={IIr7h zrp~h+OQ3l$gJhRY+p$%6zZYq&?W2lO8_l++wZl43D(@RQKkdz&))}^;MSQAu!Epa) z3a@KJSyZ3qFh=1ca3>#!`Zo==IkM3eb|q)pgS#@kz4-Z{0*@wP1eULZhzksFzuke| z^L@{vfi@&57gF6jcmwHDKP?Y4@zSgry3n`ZlXX6^DO+O<<)BG=ck_;V7ur|2`fg=v zl3d{DeOKkbO~xdx0R??eZadpcN$jWC(C*(2uiJd}=0~&RH2`9i9kvZk4F`ydr z#%2MxT_u0!vu*Iy>+&>0O$uGj;Zhfx;+ zVQV}YYjBYp&&A-2>oaNAK1cTQ1rzl+IszwF&1)x!X^uM`ZWqJ-IX^Bs9JcBnf&u1- zLM2OIu^ZT<{gMO`!HNL71^~5)w=Rc0-l738Jt7CD^#ke!90TdyR>-~j7I5yjdI589 z}>mI zjgdw#Qr86_oJ*>~fBm&Tmui5L_al(!I^2{WIuf#iUDp+m=e`aI*w@aItgHq^Ll;SO(tQDZZtup-><~s=;W5 z0bHs)!iRGh)Ycd_LW^}hi^Lm73b(~K2}Lwf6j3Z|hvrnXp|^-$!BRsMf8XkdItU6X z&;r=!qO1e8-miJ^&d`M=!ey1-x7N`0OE~7c!7$OvF{K=VA(Z0OTo}S2C2`2zasJ)P z)>cqXTGhN5Tu%AGg)MZ)?ct6O;J6#n1! zZo)?xM#Dhh*x8fmU(%=_>8(nD5R-h6=c^}itPS7fmd-<1Bf~XTZG>{jIP-1!J$M$% z6M-&Y{8iAlw2ns{chrT~7UmCsJI*g5ggk4k&5HUNU(LppaP9&uL~ZWISvdJe zxFf-g^F}S7ka&m8G`gnYg#D_ScNtxiBTmcXh>MI6*JuO|n|YTB7mXUs$Wz2=*ZV5n zMcOvD{zQvXZ*1&5MC%MV1r4FHGP=G+@`!Y*^AMIz2Ko776#%0DE<&4#{#&5GUug*O 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
  • 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%0AHCD!}Ym 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 literal 0 HcmV?d00001 diff --git a/sandag_abm/src/main/resources/runDriver.cmd b/sandag_abm/src/main/resources/runDriver.cmd new file mode 100644 index 0000000..7ace356 --- /dev/null +++ b/sandag_abm/src/main/resources/runDriver.cmd @@ -0,0 +1,17 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ############ PARAMETERS ############ + +rem ############ JPPF DRIVER ############ +set JPPF_LIB=%PROJECT_DIRECTORY%\application\* +set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% + +start %JAVA_64_PATH%\bin\java -server -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-driver.properties -Djppf.config=jppf-driver.properties org.jppf.server.DriverLauncher diff --git a/sandag_abm/src/main/resources/runHhMgr.cmd b/sandag_abm/src/main/resources/runHhMgr.cmd new file mode 100644 index 0000000..2fd44c6 --- /dev/null +++ b/sandag_abm/src/main/resources/runHhMgr.cmd @@ -0,0 +1,54 @@ +rem @echo off + +rem %1 is the project directory +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem JVM allocations +set MEMORY_HHMGR_MIN=%MEMORY_HHMGR_MIN% +set MEMORY_HHMGR_MAX=%MEMORY_HHMGR_MAX% + +rem Running on SAG02 +set HOST_IP_ADDRESS=%HHMGR_IP% + +rem 1129 used to calibrate the 20% sample +set HOST_PORT=%HH_MANAGER_PORT% + +rem (X:) is mapped to \\w-ampdx-d-sag01\C +rem set DRIVE=%MAPDRIVE% +set DRIVE=%PROJECT_DRIVE% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +rem set RUNTIME=%DRIVE%%PROJECT_DIRECTORY% +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + +set JAR_LOCATION=%RUNTIME%/application + +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%;%JAR_LOCATION%\* + + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_64_PATH%\bin;%OLDPATH% + +rem ### Change current directory to RUNTIME, and issue the java command to run the model. +rem Note: Running java script in separate window to properly redirect console output +start %PROJECT_DIRECTORY%\bin\runHhMgr_log.bat %JAVA_64_PATH% %MEMORY_HHMGR_MIN% %MEMORY_HHMGR_MAX% %CLASSPATH% %HOST_IP_ADDRESS% %HOST_PORT% %RUNTIME% +rem start %JAVA_64_PATH%/bin/java -server -Xms%MEMORY_HHMGR_MIN% -Xmx%MEMORY_HHMGR_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% -port %HOST_PORT% +rem java -Xdebug -Xrunjdwp:transport=dt_socket,address=1044,server=y,suspend=y -server -Xmx12000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% + +rem ### restore saved environment variable values, and change back to original current directory +set PATH=%OLDPATH% + diff --git a/sandag_abm/src/main/resources/runHhMgr_log.bat b/sandag_abm/src/main/resources/runHhMgr_log.bat new file mode 100644 index 0000000..ae594fa --- /dev/null +++ b/sandag_abm/src/main/resources/runHhMgr_log.bat @@ -0,0 +1,18 @@ +rem @echo off + +rem ### Declaring required environment variables +set JAVA_64_PATH = %1 +set MEMORY_HHMGR_MIN = %2 +set MEMORY_HHMGR_MAX = %3 +set CLASSPATH = %4;%5;%6;%7 +set HOST_IP_ADDRESS = %8 +set HOST_PORT = %9 +shift +set RUNTIME = %9 + + +rem ### Running household manager and redirecting output to {PROJECT_DIRECTORY}\logFiles\hhMgrConsole.log +2>&1 (%JAVA_64_PATH%/bin/java -server -Xms%MEMORY_HHMGR_MIN% -Xmx%MEMORY_HHMGR_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% -port %HOST_PORT%) | %RUNTIME%\application\GnuWin32\bin\tee.exe %RUNTIME%\logFiles\hhMgrConsole.log + +rem ### Exit window +exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runMtxMgr.cmd b/sandag_abm/src/main/resources/runMtxMgr.cmd new file mode 100644 index 0000000..6503f3a --- /dev/null +++ b/sandag_abm/src/main/resources/runMtxMgr.cmd @@ -0,0 +1,43 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem call ctramp properties +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem JVM allocations +set MEMORY_MTXMGR_MIN=%MEMORY_MTXMGR_MIN% +set MEMORY_MTXMGR_MAX=%MEMORY_MTXMGR_MAX% + +rem Run the matrix manager on SAG1 +set HOST_IP_ADDRESS=%MAIN_IP% + +rem kill java tasks +taskkill /F /IM java.exe + +rem run ping to add a pause so that taskkill has time to fully kill java processes +ping -n 10 %MAIN% > nul + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set CONFIG=%PROJECT_DIRECTORY%/conf + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set OLDPATH=%PATH% +set CLASSPATH=%CONFIG%;%JAR_LOCATION%\* + +rem java -Dname=p%2 -Xdebug -Xrunjdwp:transport=dt_socket,address=1049,server=y,suspend=y -server -Xms8000m -Xmx8000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_mtx.xml org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %HOST_MATRIX_PORT% -label "SANDAG Matrix Sever" +rem Note: Running Java script in separate window to properly redirect console output +start %PROJECT_DIRECTORY%\bin\runMtxMgr_log.bat %JAVA_64_PATH% %MEMORY_MTXMGR_MIN% %MEMORY_MTXMGR_MAX% %JAR_LOCATION% %HOST_IP_ADDRESS% %MATRIX_MANAGER_PORT% %PROJECT_DIRECTORY% +rem start %JAVA_64_PATH%\bin\java -server -Xms%MEMORY_MTXMGR_MIN% -Xmx%MEMORY_MTXMGR_MAX% -Dlog4j.configuration=log4j_mtx.xml -Djava.library.path=%JAR_LOCATION% org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %MATRIX_MANAGER_PORT% -ram 1500 -label "SANDAG Matrix Server" + +set CLASSPATH=%OLDCLASSPATH% +set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runMtxMgr_log.bat b/sandag_abm/src/main/resources/runMtxMgr_log.bat new file mode 100644 index 0000000..7b5f6b0 --- /dev/null +++ b/sandag_abm/src/main/resources/runMtxMgr_log.bat @@ -0,0 +1,17 @@ +rem @echo off + +rem ### Declaring required environment variables +set JAVA_64_PATH = %1 +set MEMORY_MTXMGR_MIN = %2 +set MEMORY_MTXMGR_MAX = %3 +set JAR_LOCATION = %4 +set HOST_IP_ADDRESS = %5 +set MATRIX_MANAGER_PORT = %6 +set PROJECT_DIRECTORY = %7 + + +rem ### Running matrix manager and redirecting output to {PROJECT_DIRECTORY}\logFiles\mtxMgrConsole.log +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_MTXMGR_MIN% -Xmx%MEMORY_MTXMGR_MAX% -Dlog4j.configuration=log4j_mtx.xml -Djava.library.path=%PROJECT_DIRECTORY%\application org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %MATRIX_MANAGER_PORT% -ram 1500 -label "SANDAG Matrix Server" 2>&1 | %PROJECT_DIRECTORY%\application\GnuWin32\bin\tee.exe %PROJECT_DIRECTORY%\logFiles\mtxMgrConsole.log + +rem ### Exit window +exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandag01.cmd b/sandag_abm/src/main/resources/runSandag01.cmd new file mode 100644 index 0000000..9bcf0be --- /dev/null +++ b/sandag_abm/src/main/resources/runSandag01.cmd @@ -0,0 +1,16 @@ +@echo off + +rem ############ PARAMETERS ############ +set DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +rem ############ JPPF DRIVER ############ +set JPPF_LIB=%PROJECT_DIRECTORY%\application\* +set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +%DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +start %PROJECT_DIRECTORY%\bin\runSandag01_log.bat %JAVA_64_PATH% %CLASSPATH% %PROJECT_DIRECTORY% +rem start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag01.properties -Djppf.config=jppf-sandag01.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag01_log.bat b/sandag_abm/src/main/resources/runSandag01_log.bat new file mode 100644 index 0000000..93ae263 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandag01_log.bat @@ -0,0 +1,13 @@ +rem @echo off + +rem ### Declaring required environment variables +set JAVA_64_PATH = %1 +set CLASSPATH = %2;%3 +set PROJECT_DIRECTORY = %4 + + +rem ### Running master node and redirecting output to {PROJECT_DIRECTORY}\logFiles\sandag01Console.log +2>&1 (%JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag01.properties -Djppf.config=jppf-sandag01.properties org.jppf.node.NodeLauncher) | %PROJECT_DIRECTORY%\application\GnuWin32\bin\tee.exe %PROJECT_DIRECTORY%\logFiles\sandag01Console.log + +rem ### Exit window +exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandag02.cmd b/sandag_abm/src/main/resources/runSandag02.cmd new file mode 100644 index 0000000..4171880 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandag02.cmd @@ -0,0 +1,15 @@ +@echo off + +rem ############ PARAMETERS ############ +set DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +rem ############ JPPF DRIVER ############ +set JPPF_LIB=%PROJECT_DIRECTORY%\application\* +set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +%DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag02.properties -Djppf.config=jppf-sandag02.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag03.cmd b/sandag_abm/src/main/resources/runSandag03.cmd new file mode 100644 index 0000000..97591ae --- /dev/null +++ b/sandag_abm/src/main/resources/runSandag03.cmd @@ -0,0 +1,15 @@ +@echo off + +rem ############ PARAMETERS ############ +set DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +rem ############ JPPF DRIVER ############ +set JPPF_LIB=%PROJECT_DIRECTORY%\application\* +set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +%DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag03.properties -Djppf.config=jppf-sandag03.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag04.cmd b/sandag_abm/src/main/resources/runSandag04.cmd new file mode 100644 index 0000000..42aa4fb --- /dev/null +++ b/sandag_abm/src/main/resources/runSandag04.cmd @@ -0,0 +1,15 @@ +@echo off + +rem ############ PARAMETERS ############ +set DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +rem ############ JPPF DRIVER ############ +set JPPF_LIB=%PROJECT_DIRECTORY%\application\* +set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% + +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +%DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag04.properties -Djppf.config=jppf-sandag04.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd b/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd new file mode 100644 index 0000000..05e804b --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd @@ -0,0 +1,59 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start +ping -n 10 %MAIN% > nul + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem ### TNC Fleet Model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.maas.TNCFleetModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for TNC outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% TNC %ITERATION% + +rem ### Household AV Allocation Model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.maas.HouseholdAVAllocationModelRunner %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for AV outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% AV %ITERATION% + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd b/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd new file mode 100644 index 0000000..dd526a2 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd @@ -0,0 +1,55 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 +set SEED=2354345 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat +set PROPERTIES_NAME=sandag_abm_mcd + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +rem run ping to add a pause so that hhMgr and mtxMgr have time to fully start +ping -n 10 %MAIN% > nul + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem ## works for both single node and distributed settings; modified jppf-clientDistrubuted.properties to handle both single and distributed settings## +%JAVA_64_PATH%\bin\java -showversion -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.utilities.RunModeChoice %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -sampleSeed %SEED% + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% + +rem kill java tasks +taskkill /F /IM java.exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd new file mode 100644 index 0000000..95810a6 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd @@ -0,0 +1,68 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%JAR_LOCATION%\GnuWin32\bin;%OLDPATH% + +rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start +ping -n 10 %MAIN% > nul + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem ## works for both single node and distributed settings; modified jppf-clientDistrubuted.properties to handle both single and distributed settings## +%JAVA_64_PATH%\bin\java -showversion -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.application.SandagTourBasedModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -sampleSeed %SEED% -luAcc false 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\sdrmConsole_%ITERATION%.log + +rem ### Build trip tables +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.application.SandagTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for Resident Model outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% SDRM %ITERATION% + +rem ### Internal-external model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.internalexternal.InternalExternalModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Build internal-external model trip tables +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.internalexternal.InternalExternalTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for IE outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% IE %ITERATION% + +rem kill java tasks +rem taskkill /F /IM java.exe + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd new file mode 100644 index 0000000..c3c1283 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd @@ -0,0 +1,57 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +rem run ping to add a pause so that hhMgr and mtxMgr have time to fully start +ping -n 10 %MAIN% > nul + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem Special Event model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.specialevent.SpecialEventModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem Build Special Event model trip tables +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -Djava.library.path=%TRANSCAD_PATH% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.specialevent.SpecialEventTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + + +rem kill java tasks +taskkill /F /IM java.exe + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd new file mode 100644 index 0000000..d243f44 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd @@ -0,0 +1,86 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%JAR_LOCATION%\GnuWin32\bin;%OLDPATH% + +rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start +ping -n 10 %MAIN% > nul + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem ### Airport model - SAN +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport SAN 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\airportSANModelConsole_%ITERATION%.log + +rem ### Build airport model trip tables - SAN +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport SAN + +rem ### Checking for SAN outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% SAN %ITERATION% + +rem ### Airport model - CBX +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport CBX 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\airportCBXModelConsole_%ITERATION%.log + +rem ### Build airport model trip tables - CBX +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport CBX + +rem ### Checking for CBX outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CBX %ITERATION% + +rem ### Cross-border model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.crossborder.CrossBorderModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\crossBorderModelConsole_%ITERATION%.log + +rem ### Build cross-border model trip tables +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.crossborder.CrossBorderTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for CBM outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CBM %ITERATION% + +rem ### Visitor model +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.visitor.VisitorModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\visitorModelConsole_%ITERATION%.log + +rem ### Build visitor model trip tables +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.visitor.VisitorTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% + +rem ### Checking for Visitor outputs +call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% Visitor %ITERATION% + +rem kill java tasks +rem taskkill /F /IM java.exe + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd b/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd new file mode 100644 index 0000000..79a20c9 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd @@ -0,0 +1,45 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem rem build bike logsums +%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_BIKELOGSUM_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagBikePathChoiceLogsumMatrixApplication %PROPERTIES_NAME% +if ERRORLEVEL 1 goto DONE + +:done +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% + + diff --git a/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd b/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd new file mode 100644 index 0000000..aa243b7 --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd @@ -0,0 +1,38 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set SAMPLERATE=%3 +set ITERATION=%4 +set PROPERTIES_NAME=sandag_abm + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + +rem ### Set the name of the properties file the application uses by giving just the base part of the name (with ".xxx" extension) +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem run bike assignment +%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_BIKEROUTE_MAX% -XX:-UseGCOverheadLimit -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagBikePathChoiceEdgeAssignmentApplication %PROPERTIES_NAME% %SAMPLERATE% %ITERATION% +if ERRORLEVEL 1 goto DONE + +:done +rem kill java tasks +rem taskkill /F /IM java.exe + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd b/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd new file mode 100644 index 0000000..248541c --- /dev/null +++ b/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd @@ -0,0 +1,45 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem rem build walk skims +%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_WALKLOGSUM_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication %PROPERTIES_NAME% +if ERRORLEVEL 1 goto DONE + +python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\calculate_micromobility.py --properties_file %PROJECT_DRIVE%%PROJECT_DIRECTORY%\conf\sandag_abm.properties --outputs_directory %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output --inputs_parent_directory %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +:done +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runTransitReporter.cmd b/sandag_abm/src/main/resources/runTransitReporter.cmd new file mode 100644 index 0000000..3162c79 --- /dev/null +++ b/sandag_abm/src/main/resources/runTransitReporter.cmd @@ -0,0 +1,48 @@ +rem @echo off + +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set THRESHOLD=%3 +set TOD=%4 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% +call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat + +rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. +set OLDJAVAPATH=%JAVA_PATH% + +rem ### Set the directory of the jdk version desired for this model run +rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates +rem ### and compiles code during the model run, and uses javac in the jdk to do this. +set JAVA_PATH=%JAVA_64_PATH% + +rem ### Name the project directory. This directory will hava data and runtime subdirectories +set RUNTIME=%PROJECT_DIRECTORY% +set CONFIG=%RUNTIME%/conf + + +set JAR_LOCATION=%PROJECT_DIRECTORY%/application +set LIB_JAR_PATH=%JAR_LOCATION%\* + +rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. +set OLDCLASSPATH=%CLASSPATH% +set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; + +rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. +set OLDPATH=%PATH% + +rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. +rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. +set PATH=%JAVA_PATH%\bin;%OLDPATH% + +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% + +rem TransitTimeReporter +rem %JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.reporting.TransitTimeReporter %PROPERTIES_NAME% -threshold %THRESHOLD% -period %TOD% -outWalkFileName walkMgrasWithin%THRESHOLD%Min.csv -outDriveFileName driveMgrasWithin%THRESHOLD%Min.csv +%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.reporting.TransitTimeReporter %PROPERTIES_NAME% -threshold %THRESHOLD% -period %TOD% -outWalkFileName walkMgrasWithin%THRESHOLD%Min.csv + +rem ### restore saved environment variable values, and change back to original current directory +set JAVA_PATH=%OLDJAVAPATH% +set PATH=%OLDPATH% +set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/sandag_abm.properties b/sandag_abm/src/main/resources/sandag_abm.properties new file mode 100644 index 0000000..9325e4a --- /dev/null +++ b/sandag_abm/src/main/resources/sandag_abm.properties @@ -0,0 +1,1357 @@ +#SANDAG ABM Properties +#Software Version +version=${version} +############################################################################################################################################################################# +# +# CLUSTER PROPERTIES: MODIFY WHEN CHANGING CLUSTER CONFIGURATION OR MOVING TO NEW CLUSTER. +# +############################################################################################################################################################################# +RunModel.MatrixServerAddress=${matrix.server.host} +RunModel.MatrixServerPort=${matrix.server.port} +RunModel.HouseholdServerAddress=${household.server.host} +RunModel.HouseholdServerPort=${household.server.port} + +############################################################################################################################################################################# +# +# RUN PROPERTIES: MODEL COMPONENT SWITCHES +# +############################################################################################################################################################################# +#set sample rates +sample_rates=0.2,0.5,1.0 + +#highway assignment convergence criteria +convergence=0.0005 + +#set warm up inputs +RunModel.useLocalDrive = true +RunModel.skipInitialization = false +RunModel.deleteAllMatrices = true +RunModel.skip4Ds = false +RunModel.skipInputChecker = false +RunModel.skipCopyWarmupTripTables = false +RunModel.skipCopyBikeLogsum = false +RunModel.skipBikeLogsums = true +#always create walk logsum (walk to TAP file); modification from GUI disallowed +RunModel.skipCopyWalkImpedance = true +RunModel.skipWalkLogsums = false + +#build networks +RunModel.skipBuildNetwork = false + +# start looping +# set startFromIteration to 4 if only want +# to run final hwy and transit steps +RunModel.startFromIteration = 1 +RunModel.skipHighwayAssignment = false,false,false +RunModel.skipTransitSkimming = false,false,false +RunModel.skipTransponderExport = false,false,false +RunModel.skipCoreABM = false,false,false +RunModel.skipOtherSimulateModel = false,false,false +RunModel.skipMAASModel = false,false,false +RunModel.skipSpecialEventModel = true,true,true +RunModel.skipCTM = false,false,false +RunModel.skipEI = false,false,false +RunModel.skipTruck = false,false,false +RunModel.skipTripTableCreation = false,false,false +# end looping + +RunModel.skipFinalHighwayAssignment = false +RunModel.skipFinalHighwayAssignmentStochastic = true +RunModel.skipFinalTransitAssignment = false +RunModel.collapseOnOffByRoute=false +RunModel.skipLUZSkimCreation = true +RunModel.skipVisualizer = true +RunModel.skipDataExport = false +RunModel.skipDataLoadRequest = false +RunModel.skipDeleteIntermediateFiles = false +RunModel.MatrixPrecision = 0.0005 +# minimual space (MB) on C drive +RunModel.minSpaceOnC =250 + +TNC.totalThreads=10 + +############################################################################################################################################################################# +# +# LOGGING PROPERTIES: USE FOR TRACING HOUSEHOLDS OR AGENTS THROUGH SIMULATION. +# +# Note that the way that logging works right now, the trace zones also have to be valid transit stops or the code will crash. Check the skims to make sure they exist. +# Turn off trace debugging in routine model runs to speed things up (comment out Debug.Trace.HouseholdIdList) +# +############################################################################################################################################################################# +# Resident models +Trace = false +#Trace.otaz = 1638 +#Trace.dtaz = 2447 +Trace.otaz = +Trace.dtaz = +Seek = false +Process.Debug.HHs.Only = false +Debug.Trace.HouseholdIdList= + +# Internal-External models +internalExternal.seek = false +internalExternal.trace = 50 + +# Cross-Border models +crossBorder.seek = false +# trace by tourId +crossBorder.trace = 12 + +# Visitor models +visitor.seek = false +#trace by tourId +#visitor.trace = 742 +visitor.trace = 742 + +# Special event models +specialEvent.seek = false +specialEvent.trace = 5855 + +# Trace TransCAD trip table creation by TAZ (to/from); only applies to SD resident model +tripTable.trace=4384 + +RunModel.LogResults = true + +############################################################################################################################################################################# +# PATH PROPERTIES: MODIFY AS NEEDED WHEN COPY RELEASE TO A LOCAL RUN FOLDER +############################################################################################################################################################################# +Project.Directory = %project.folder%/ +generic.path = %project.folder%/input/ +scenario.path = %project.folder%/ +skims.path = %project.folder%/output/ +uec.path = %project.folder%/uec/ +report.path = %project.folder%/report/ + +# Visitor model is run using Java 7 Fork\Join Framework. Parallelism controls number of simultaneous threads. Can increase if more processors. +# 5 threads provided optimum runtimes on a 6 core, 24 thread machine with 128GB of RAM. +visitor.run.concurrent = true +visitor.concurrent.parallelism = 5 + +############################################################################################################################################################################# +# +# SCENARIO PROPERTIES: MODIFY WHEN RUNNING NEW SCENARIO, IF NECESSARY +# +############################################################################################################################################################################# +# MGRA data file: this token is referred to in many UECs +mgra.socec.file = input/mgra13_based_input${year}.csv + +# scenario year +scenarioYear=${year} + +# scenario build +scenarioBuild=${year_build} + +# Auto operating costs: these tokens are referred to in many UECs +aoc.fuel =${aoc.fuel} +aoc.maintenance =${aoc.maintenance} + +# Cross border model is run using Java 7 Fork\Join Framework. Parallelism controls number of simultaneous threads. Can increase if more processors. +crossBorder.run.concurrent = true +crossBorder.concurrent.parallelism = 8 + +# Cross border model settings: Number of tours, share of tours that are SENTRI. +crossBorder.tours =${crossBorder.tours} +crossBorder.sentriShare = ${crossBorder.sentriShare} + +# Visitor model settings: occupancy rates for hotels, households and share of each that are business visitors +visitor.hotel.occupancyRate = 0.7 +visitor.household.occupancyRate = 0.018 +visitor.hotel.businessPercent = 0.3 +visitor.household.businessPercent = 0.04 + +# Airport model settings: enplanements, connecting passengers, average party size, MGRA that the airport is in +airport.SAN.enplanements =${airport.SAN.enplanements} +airport.SAN.connecting =${airport.SAN.connecting} +airport.SAN.annualizationFactor = 365 +airport.SAN.averageSize = 1.7 +airport.SAN.airportMgra =${airport.SAN.airportMgra} + +airport.CBX.enplanements =${airport.CBX.enplanements} +airport.CBX.connecting =${airport.CBX.connecting} +airport.CBX.annualizationFactor = 365 +airport.CBX.averageSize = 2.2 +airport.CBX.airportMgra =${airport.CBX.airportMgra} + +# Truck model settings: +truck.FFyear =${year} + +# Destination zones for the transponder accessibility calculator +transponder.destinations = 4027,2563,2258 + +#taz crosswalk file +taz.to.cluster.crosswalk.file = taz_crosswalk.csv +cluster.zone.centroid.file = cluster_zones.csv + +############################################################################################ +# EMERGING MOBILITY SECTION: MODIFY WHEN CHANGE AV, TNC, and MICROMOBILITY ASSUMPTIONS +#------------------------------------------------------------------------------------------- +# AV Mobility Scenario Parameters +#------------------------------------------------------------------------------------------- +# AV.Share: the share of vehicles assumed to be AVs in the vehicle fleet; Auto ownership ASCs will be calibrated for different levels of AV penetration +# AV.ProbabilityBoost: the increased probability (multiplicative) for using AVs for tours, based on autos to drivers. The highest this should go is 1.2 +# AV.IVTFactor: the auto in-vehicle time factor to apply to AVs +# AV.ParkingCostFactor: The auto parking cost factor to apply to AVs, assuming some AVs are sent to remote locations or home +# AV.CostPerMileFactor: The auto cost per mile factor to apply to AVs, assuming AVs are more efficient in terms of fuel consumption than human-driven vehicles +# AV.TerminalTimeFactor: The factor to apply to terminal time for AVs, assuming AVs offer curbside passenger pickup/dropoff +# TNC.shared.IVTFactor: The factor to apply to in-vehicle time for shared TNC mode, reflecting out-direction travel for pickup/dropoff of other passengers + +Mobility.AV.Share = ${Mobility.AV.Share} +Mobility.AV.ProbabilityBoost.AutosLTDrivers = 1.2 +Mobility.AV.ProbabilityBoost.AutosGEDrivers = 1.1 +Mobility.AV.IVTFactor = 0.75 +Mobility.AV.ParkingCostFactor = 0.5 +Mobility.AV.CostPerMileFactor = 0.7 +Mobility.AV.TerminalTimeFactor = 0.65 +Mobility.AV.MinimumAgeDriveAlone = 13 +Mobility.TNC.shared.IVTFactor = 1.25 +crossBorder.avShare = 0.0 + +#------------------------------------------------------------------------------------------- +# Taxi and TNC cost and wait time parameters +#------------------------------------------------------------------------------------------- +# 3 modes: taxi, TNC - single, and TNC - shared +# baseFare: Initial fare +# costPerMile: The cost per mile +# costPerMinute: The cost per minute +# costMinimum: The minimum cost (for TNC modes only) +# +# Wait times are drawn from a distribution by area type (emp+hh)/sq. miles +# The mean and standard deviation is given for each area type range +# The ranges are configurable, set by WaitTimeDistribution.EndPopEmpPerSqMi + +taxi.baseFare = ${taxi.baseFare} +taxi.costPerMile = ${taxi.costPerMile} +taxi.costPerMinute = ${taxi.costPerMinute} + +TNC.single.baseFare = ${TNC.single.baseFare} +TNC.single.costPerMile = ${TNC.single.costPerMile} +TNC.single.costPerMinute = ${TNC.single.costPerMinute} +TNC.single.costMinimum = ${TNC.single.costMinimum} + +# use lower costs - these are synthesized, need real prices +TNC.shared.baseFare = ${TNC.shared.baseFare} +TNC.shared.costPerMile = ${TNC.shared.costPerMile} +TNC.shared.costPerMinute = ${TNC.shared.costPerMinute} +TNC.shared.costMinimum = ${TNC.shared.costMinimum} + +#Note: the following comma-separated value properties cannot have spaces between them, or else the RuntimeConfiguration.py code won't work +TNC.single.waitTime.mean = 10.3,8.5,8.4,6.3,4.7 +TNC.single.waitTime.sd = 4.1,4.1,4.1,4.1,4.1 + +TNC.shared.waitTime.mean = 15.0,15.0,11.0,8.0,7.0 +TNC.shared.waitTime.sd = 4.1,4.1,4.1,4.1,4.1 + +Taxi.waitTime.mean = 26.5,17.3,13.3,9.5,5.5 +Taxi.waitTime.sd = 6.4,6.4,6.4,6.4,6.4 + +WaitTimeDistribution.EndPopEmpPerSqMi = 500,2000,5000,15000,9999999999 + +#------------------------------------------------------------------------------------------- +# Taxi and TNC vehcicle trip conversion factors +#------------------------------------------------------------------------------------------- +# The following properties are used to split out the taxi, TNC-single, and TNC-shared trips into vehicle trips to be added to the rest of the vehicle trips by occupancy prior to assignment. + +Taxi.da.share = 0.0 +Taxi.s2.share = 0.9 +Taxi.s3.share = 0.1 +Taxi.passengersPerVehicle = 1.1 + +TNC.single.da.share = 0.0 +TNC.single.s2.share = 0.8 +TNC.single.s3.share = 0.2 +TNC.single.passengersPerVehicle = 1.2 + +TNC.shared.da.share = 0.0 +TNC.shared.s2.share = 0.3 +TNC.shared.s3.share = 0.7 +TNC.shared.passengersPerVehicle = 2.0 + +#------------------------------------------------------------------------------------------- +# Maas Routing Model Properties +#------------------------------------------------------------------------------------------- +Maas.RoutingModel.maxDistanceForPickup = 5 +Maas.RoutingModel.maxDiversionTimeForPickup = 5 +Maas.RoutingModel.minutesPerSimulationPeriod = 5 +Maas.RoutingModel.maxPassengers=6 +Maas.RoutingModel.maxWalkDistance = 0.15 +Maas.RoutingModel.vehicletrip.output.file=output/TNCTrips.csv +Maas.RoutingModel.vehicletrip.output.matrix=output/TNCVehicleTrips + +Maas.RoutingModel.routeIntrazonal=false +#NULL,DRIVEALONE,SHARED2,SHARED3,WALK,BIKE,WALK_SET,PNR_SET,KNR_SET,TNC_SET,TAXI,TNC_SINGLE,TNC_SHARED,SCHBUS +Maas.RoutingModel.Modes =0,0,0,0,0,0,0,0,0,0,1,1,1,0 +Maas.RoutingModel.SharedEligible=0,0,0,0,0,0,0,0,0,0,0,0,1,0 +Maas.RoutingModel.maxDistanceBeforeRefuel = 300 +Maas.RoutingModel.timeRequiredForRefuel = 15 + +Maas.AVAllocationModel.vehicletrip.output.file = output/householdAVTrips.csv +Maas.AVAllocationModel.vehicletrip.output.matrix = output/emptyAVTrips + +Maas.AVAllocation.uec.file = AutonomousVehicleAllocationChoice.xls +Maas.AVAllocation.data.page = 0 +Maas.AVAllocation.vehiclechoice.model.page = 1 +Maas.AVAllocation.parkingchoice.model.page = 2 +Maas.AVAllocation.triputility.model.page = 3 +Mobility.AV.RemoteParkingCostPerHour = ${Mobility.AV.RemoteParkingCostPerHour} + +# END--EMERGING MOBILITY SECTION +############################################################################################ + +############################################################################################################################################################################# +# +# CORE MODEL RUN PROPERTIES: CONTROL STEPS RUN IN CORE MODEL +# +############################################################################################################################################################################# +Model.Random.Seed = 1 + +RunModel.Clear.MatrixMgr.At.Start=false + +# Set to true if read the accessibilities from an input file instead of calculating them prior to running CTRAMP +acc.read.input.file = false + +# Setting shadow price files to null will reset prices to 0. If running new land-use scenario, set files to null and set maximum iterations to 20. +# Then copy shadow price output files to input directory, set maximum iterations to 1 for any subsequent runs with the same land-use file. +UsualWorkLocationChoice.ShadowPrice.Input.File = input/${workShadowPricing.iteration} +UsualSchoolLocationChoice.ShadowPrice.Input.File = input/${schoolShadowPricing.iteration} +uwsl.ShadowPricing.Work.MaximumIterations = 1 +uwsl.ShadowPricing.School.MaximumIterations = 1 +uwsl.ShadowPricing.OutputFile = output/ShadowPricingOutput.csv + +uwsl.run.workLocChoice = true +uwsl.run.schoolLocChoice = true +uwsl.write.results = true + +uwsl.use.new.soa = false +nmdc.use.new.soa = false +slc.use.new.soa = false + +# properties for distributed time coefficient +distributedTimeCoefficients = true + +timeDistribution.mean.work = 1.0 +timeDistribution.standardDeviation.work = 0.7 +timeDistribution.mean.nonWork = 1.0 +timeDistribution.standardDeviation.nonWork = 0.6 + +timeDistribution.randomSeed = 2301832 + +# value of time thresholds for skimming, assignment, mode choice UECs and trip tables ($/hr). +valueOfTime.threshold.low = 8.81 +valueOfTime.threshold.med = 18.00 + + +# save tour mode choice utilities and probabilities (for debugging purpose) +TourModeChoice.Save.UtilsAndProbs = true + +# packet size for distributing households, DO NOT change +distributed.task.packet.size = 200 + +#RunModel.RestartWithHhServer = uwsl +RunModel.RestartWithHhServer = none +#RunModel.RestartWithHhServer = ao +#RunModel.RestartWithHhServer = stf + +# Model Component run flags; Wu's note: not functional yet +RunModel.PreAutoOwnership = true +RunModel.UsualWorkAndSchoolLocationChoice = true +RunModel.AutoOwnership = true +RunModel.TransponderChoice = true +RunModel.FreeParking = true +RunModel.CoordinatedDailyActivityPattern = true +RunModel.IndividualMandatoryTourFrequency = true +RunModel.MandatoryTourModeChoice = true +RunModel.MandatoryTourDepartureTimeAndDuration = true +RunModel.SchoolEscortModel = true +RunModel.JointTourFrequency = true +RunModel.JointTourLocationChoice = true +RunModel.JointTourDepartureTimeAndDuration = true +RunModel.JointTourModeChoice = true +RunModel.IndividualNonMandatoryTourFrequency = true +RunModel.IndividualNonMandatoryTourLocationChoice = true +RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration = true +RunModel.IndividualNonMandatoryTourModeChoice = true +RunModel.AtWorkSubTourFrequency = true +RunModel.AtWorkSubTourLocationChoice = true +RunModel.AtWorkSubTourDepartureTimeAndDuration = true +RunModel.AtWorkSubTourModeChoice = true +RunModel.StopFrequency =true +RunModel.StopLocation = true + +############################################################################################################################################################################# +# +# INPUT PROPERTIES +# +############################################################################################################################################################################# +#PopSyn Inputs +PopulationSynthesizer.InputToCTRAMP.HouseholdFile = input/households.csv +PopulationSynthesizer.InputToCTRAMP.PersonFile = input/persons.csv +PopulationSynthesizer.OccupCodes = input/pecas_occ_occsoc_acs.csv +PopulationSynthesizer.IndustryCodes = input/activity_code_indcen_acs.csv +# +# The military industry ranges are used to recode military occupation. This is +# necessary because military workers identify themselves as non-military occupations. +# The models need to be consistent with PECAS, where all military workers are in +# the military occupation category 56. +PopulationSynthesizer.MilitaryIndustryRange=9670,9870 + +# auxiliary inputs, these are scenario-specific +taz.driveaccess.taps.file = input/accessam.csv +tap.ptype.file = input/tap.ptype +taz.parkingtype.file = input/zone.park +taz.terminal.time.file = input/zone.term +maz.tap.tapLines = output/tapLines.csv + +# transit stop attribute file +transit.stop.file = input/trstop.csv + +############################################################################################################################################################################# +# +# OUTPUT PROPERTIES +# +############################################################################################################################################################################# +Results.WriteDataToFiles= true +Results.HouseholdDataFile = output/householdData.csv +Results.PersonDataFile = output/personData.csv +Results.IndivTourDataFile = output/indivTourData.csv +Results.JointTourDataFile = output/jointTourData.csv +Results.IndivTripDataFile = output/indivTripData.csv +Results.JointTripDataFile = output/jointTripData.csv +Results.WriteDataToDatabase = false +Results.HouseholdTable = household_data +Results.PersonTable = person_data +Results.IndivTourTable = indiv_tour_data +Results.JointTourTable = joint_tour_data +Results.IndivTripTable = indiv_trip_data +Results.JointTripTable = joint_trip_data +Results.AutoTripMatrix = output/autoTrips +Results.TranTripMatrix = output/tranTrips +Results.NMotTripMatrix = output/nmotTrips +Results.OthrTripMatrix = output/othrTrips +Results.PNRFile = output/PNRByTAP_Vehicles.csv +Results.CBDFile = output/CBDByMGRA_Vehicles.csv +Results.MatrixType = OMX +Results.segmentByTransponderOwnership = true + + +Results.AutoOwnership=output/aoResults.csv +read.pre.ao.results=false +read.pre.ao.filename=output/aoResults_pre.csv + +Results.UsualWorkAndSchoolLocationChoice=output/wsLocResults.csv +read.uwsl.results=false +read.uwsl.filename=output/wsLocResults_1.csv + +############################################################################################################################################################################# +# +# CORE MODEL UECS +# +############################################################################################################################################################################# +# UECs for calculating accessibilities +acc.uec.file = %project.folder%/uec/Accessibilities.xls +acc.data.page = 0 +acc.sov.offpeak.page = 1 +acc.sov.peak.page = 2 +acc.hov.offpeak.page = 3 +acc.hov.peak.page = 4 +acc.maas.offpeak.page = 5 +acc.maas.peak.page = 6 +acc.nonmotorized.page = 7 +acc.constants.page = 8 +acc.sizeTerm.page = 9 +acc.schoolSizeTerm.page = 10 +acc.workerSizeTerm.page = 11 +acc.dcUtility.uec.file = %project.folder%/uec/Accessibilities_DC.xls +acc.dcUtility.data.page = 0 +acc.dcUtility.page = 1 + +# accessibility file location +acc.output.file = input/accessibilities.csv + +#UECs for calculating destination choice based land use accessibilities +lu.acc.dcUtility.uec.file = %project.folder%/uec/Accessibilities_LU_DC.xls +lu.acc.dcUtility.data.page = 0 +lu.acc.dcUtility.page = 1 +lu.accessibility.alts.file = Acc_LU_alts.csv + +# land use accessibililty file locations +lu.acc.output.file = output/luAccessibilities.csv +lu.acc.mc.logsums.output.file = output/luLogsums.csv + +# set either or both averaging methods to be used to write LU accessibilities files +# also requires command line parameter "-luAcc true" and acc.read.input.file = false +lu.acc.simple.averaging.method = true +lu.acc.logit.averaging.method = true + +accessibility.alts.file = Acc_alts.csv + +#UEC for Mandatory accessibilities +acc.mandatory.uec.file = %project.folder%/uec/MandatoryAccess.xls +acc.mandatory.data.page = 0 +acc.mandatory.auto.page = 1 +acc.mandatory.autoLogsum.page = 2 +acc.mandatory.bestWalkTransit.page = 3 +acc.mandatory.bestDriveTransit.page = 4 +acc.mandatory.transitLogsum.page = 5 + +# UECs for auto ownership model +ao.uec.file = AutoOwnership.xls +ao.data.page = 0 +ao.model.page = 1 + +# UECs for Mandatory tour destination choice model +uwsl.dc.uec.file = ${uwsl.dc.uec.file} +uwsl.dc2.uec.file = TourDestinationChoice2.xls +uwsl.soa.uec.file = DestinationChoiceAlternativeSample.xls +uwsl.soa.alts.file = DestinationChoiceAlternatives.csv +uwsl.work.soa.SampleSize = 30 +uwsl.school.soa.SampleSize = 30 + +# The UEC file for work purposes includes TAZ Size in the expressions +work.soa.uec.file = TourDcSoaDistance.xls +work.soa.uec.data = 0 +work.soa.uec.model = 1 + +# The UEC file for school purposes does not include TAZ Size in the expressions +# so that the utilities can be stored as exponentiated distance utility matrices +# for univ, hs, gs, and ps, and then multiplied by the various school segment +# size terms for each of these 4 groups of school segments. +univ.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls +univ.soa.uec.data = 0 +univ.soa.uec.model = 1 + +hs.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls +hs.soa.uec.data = 0 +hs.soa.uec.model = 2 + +gs.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls +gs.soa.uec.data = 0 +gs.soa.uec.model = 3 + +ps.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls +ps.soa.uec.data = 0 +ps.soa.uec.model = 4 + +#UECs for transponder ownership model +tc.choice.avgtts.file = output/transponderModelAccessibilities.csv +tc.uec.file = TransponderOwnership.xls +tc.data.page = 0 +tc.model.page = 1 +tc.everyone.owns = ${tc.everyone.owns} + + +#UECs for parking provision model +fp.uec.file = ParkingProvision.xls +fp.data.page = 0 +fp.model.page = 1 + +#UEC for telecommute model +te.uec.file = Telecommute.xls +te.data.page = 0 +te.model.page = 1 + + +#UECs for CDAP model +cdap.uec.file = CoordinatedDailyActivityPattern.xls +cdap.data.page = 0 +cdap.one.person.page = 1 +cdap.two.person.page = 2 +cdap.three.person.page = 3 +cdap.all.person.page = 4 +cdap.joint.page = 5 + +#UECs for individual mandatory tour frequency model +imtf.uec.file = MandatoryTourFrequency.xls +imtf.data.page = 0 +imtf.model.page = 1 + +#UECs for Non-Mandatory tour destination sampling +nonSchool.soa.uec.file = TourDcSoaDistance.xls +escort.soa.uec.data = 0 +escort.soa.uec.model = 2 +other.nonman.soa.uec.data = 0 +other.nonman.soa.uec.model = 3 +atwork.soa.uec.data = 0 +atwork.soa.uec.model = 4 + +soa.taz.dist.alts.file = SoaTazDistAlts.csv + +nmdc.dist.alts.file = NonMandatoryTlcAlternatives.csv +nmdc.soa.alts.file = DestinationChoiceAlternatives.csv +nmdc.soa.SampleSize = 30 + +#UECs for Non-Mandatory tour destination choice model +nmdc.uec.file2 = TourDestinationChoice2.xls +nmdc.uec.file = ${nmdc.uec.file} +nmdc.data.page = 0 +nmdc.escort.model.page = 7 +nmdc.shop.model.page = 8 +nmdc.maint.model.page = 9 +nmdc.eat.model.page = 10 +nmdc.visit.model.page = 11 +nmdc.discr.model.page = 12 +nmdc.atwork.model.page = 13 + +# following properties use tod sampling instead of logsums +nmdc.SampleTODPeriod = true +nmdc.SampleTODPeriod.file = input/Non_Mand_Tours_ArrDep_Distbn.csv + +#UECs for Non-Mandatory tour destination sampling +nmdc.soa.uec.file = DestinationChoiceAlternativeSample.xls +nmdc.soa.data.page = 0 +nmdc.soa.escort.model.page = 6 +nmdc.soa.shop.model.page = 7 +nmdc.soa.maint.model.page = 7 +nmdc.soa.eat.model.page = 7 +nmdc.soa.visit.model.page = 7 +nmdc.soa.discr.model.page = 7 +nmdc.soa.atwork.model.page = 8 + +#UECs for School Escorting Model +school.escort.uec.filename = SchoolEscorting.xls +school.escort.alts.file = SchoolEscortingAlts.csv +school.escort.data.sheet = 0 +school.escort.outbound.model.sheet = 1 +school.escort.inbound.conditonal.model.sheet = 2 +school.escort.outbound.conditonal.model.sheet = 3 +school.escort.RNG.offset = 384571483 + +#UECs for tour mode choice model +tourModeChoice.uec.file =TourModeChoice.xls +tourModeChoice.maint.model.page = 4 +tourModeChoice.discr.model.page = 5 +tourModeChoice.atwork.model.page = 6 + +# utility coefficients by tour purpose (work, univ, school, maintenance, discretionary, work-based). These are at tour level. +tour.utility.ivt.coeffs = -0.016,-0.016,-0.01,-0.017,-0.015,-0.032 +tour.utility.income.coeffs = -0.625,-0.262,-0.262,-0.262,-0.262,-0.262 +tour.utility.income.exponents = 0.6,0.5,0.5,0.5,0.5,0.5 + +#UECs for tour TOD choice model +departTime.uec.file = TourDepartureAndDuration.xls +departTime.data.page = 0 +departTime.work.page = 1 +departTime.univ.page = 2 +departTime.school.page = 3 +departTime.escort.page = 4 +departTime.shop.page = 5 +departTime.maint.page = 6 +departTime.eat.page = 7 +departTime.visit.page = 8 +departTime.discr.page = 9 +departTime.atwork.page = 10 +departTime.alts.file = DepartureTimeAndDurationAlternatives.csv + +#UECs for joint tour frequency choice model +jtfcp.uec.file = JointTourFrequency.xls +jtfcp.alternatives.file = JointAlternatives.csv +jtfcp.data.page = 0 +jtfcp.freq.comp.page = 1 +jtfcp.participate.page = 2 + +#UECs for individual non-mandatory tour frequency model +inmtf.uec.file = NonMandatoryIndividualTourFrequency.xls +inmtf.FrequencyExtension.ProbabilityFile = IndividualNonMandatoryTourFrequencyExtensionProbabilities_p1.csv +IndividualNonMandatoryTourFrequency.AlternativesList.InputFile = IndividualNonMandatoryTourFrequencyAlternatives.csv +inmtf.data.page = 0 +inmtf.perstype1.page = 1 +inmtf.perstype2.page = 2 +inmtf.perstype3.page = 3 +inmtf.perstype4.page = 4 +inmtf.perstype5.page = 5 +inmtf.perstype6.page = 6 +inmtf.perstype7.page = 7 +inmtf.perstype8.page = 8 + +#UECs for at work subtour frequency model +awtf.uec.file = AtWorkSubtourFrequency.xls +awtf.data.page = 0 +awtf.model.page = 1 + +#UECs for stop frequency model +stf.uec.file = StopFrequency.xls +stf.purposeLookup.proportions = StopPurposeLookupProportions.csv +stf.data.page = 0 +stf.work.page = 1 +stf.univ.page = 2 +stf.school.page = 3 +stf.escort.page = 4 +stf.shop.page = 5 +stf.maint.page = 6 +stf.eat.page = 7 +stf.visit.page = 8 +stf.discr.page = 9 +stf.subtour.page = 10 + +#UECs for stop location choice model +slc.uec.file = StopLocationChoice.xls +slc.uec.data.page = 0 +slc.mandatory.uec.model.page = 1 +slc.maintenance.uec.model.page = 2 +slc.discretionary.uec.model.page = 3 +slc.alts.file = SlcAlternatives.csv + +slc.soa.uec.file = SlcSoaSize.xls +slc.soa.alts.file = DestinationChoiceAlternatives.csv + +auto.slc.soa.distance.uec.file = SlcSoaDistanceUtility.xls +auto.slc.soa.distance.data.page = 0 +auto.slc.soa.distance.model.page = 1 + +slc.soa.size.uec.file = SlcSoaSize.xls +slc.soa.size.uec.data.page = 0 +slc.soa.size.uec.model.page = 1 + +stop.depart.arrive.proportions = StopDepartArriveProportions.csv + +#UECs for trip mode choice model +tripModeChoice.uec.file =TripModeChoice.xls + +# utility coefficients by tour purpose (work, univ, school, maintenance, discretionary, work-based). These are at trip level. +trip.utility.ivt.coeffs = -0.032,-0.032,-0.02,-0.034,-0.03,-0.064 +trip.utility.income.coeffs = -1.25,-0.524,-0.524,-0.524,-0.524,-0.524 +trip.utility.income.exponents = 0.6,0.5,0.5,0.5,0.5,0.5 + + +#UECs for parking location choice model +plc.uec.file = ParkLocationChoice.xls +plc.uec.data.page = 0 +plc.uec.model.page = 1 + +plc.alts.corresp.file = ParkLocationAlts.csv +plc.alts.file = ParkLocationSampleAlts.csv + +mgra.avg.cost.output.file = output/mgraParkingCost.csv + +mgra.avg.cost.dist.coeff.work = -8.6 +mgra.avg.cost.dist.coeff.other = -4.9 + +park.cost.reimb.mean = -0.05 +park.cost.reimb.std.dev = 0.54 + +#UECs for best transit path finding +utility.bestTransitPath.uec.file =BestTransitPathUtility.xls +utility.bestTransitPath.data.page = 0 +utility.bestTransitPath.tapToTap.page = 1 +utility.bestTransitPath.walkAccess.page = 2 +utility.bestTransitPath.driveAccess.page = 3 +utility.bestTransitPath.walkEgress.page = 4 +utility.bestTransitPath.driveEgress.page = 5 +utility.bestTransitPath.driveAccDisutility.page = 6 +utility.bestTransitPath.driveEgrDisutility.page = 7 +utility.bestTransitPath.skim.sets = 3 +utility.bestTransitPath.alts = 4 +utility.bestTransitPath.maxPathsPerSkimSetForLogsum = 1,1,1 +utility.bestTransitPath.nesting.coeff = 0.24 + +#UECs for auto skimming +skims.auto.uec.file = AutoSkims.xls +skims.auto.data.page = 0 +skims.auto.ea.page = 1 +skims.auto.am.page = 2 +skims.auto.md.page = 3 +skims.auto.pm.page = 4 +skims.auto.ev.page = 5 + +#UECs for TAZ distances +taz.distance.uec.file = tazDistance.xls +taz.distance.data.page = 0 +taz.od.distance.ea.page = 1 +taz.od.distance.am.page = 2 +taz.od.distance.md.page = 3 +taz.od.distance.pm.page = 4 +taz.od.distance.ev.page = 5 + +#UECs for TAZ times +taz.od.time.ea.page = 6 +taz.od.time.am.page = 7 +taz.od.time.md.page = 8 +taz.od.time.pm.page = 9 +taz.od.time.ev.page = 10 + + +#UECs for walk-transit-walk skimming +skim.walk.transit.walk.uec.file = WalkTransitWalkSkims.xls +skim.walk.transit.walk.data.page = 0 +skim.walk.transit.walk.skim.page = 1 +skim.walk.transit.walk.skims = 13 + +#UECs for walk-transit-drive skimming +skim.walk.transit.drive.uec.file = WalkTransitDriveSkims.xls +skim.walk.transit.drive.data.page = 0 +skim.walk.transit.drive.skim.page = 1 +skim.walk.transit.drive.skims = 13 + +#UECs for drive-transit-walk skimming +skim.drive.transit.walk.uec.file = DriveTransitWalkSkims.xls +skim.drive.transit.walk.data.page = 0 +skim.drive.transit.walk.skim.page = 1 +skim.drive.transit.walk.skims = 13 + + +##################################################################################### +# IE Model Settings (run as part of CT-RAMP) +##################################################################################### + +RunModel.InternalExternal = true + +ie.uec.file = InternalExternalTripChoice.xls +ie.data.page = 0 +ie.model.page = 1 +ie.logsum.distance.coeff = -0.05 +external.tazs = 1,2,3,4,5,6,7,8,9,10,11,12 + + +internalExternal.dc.uec.file = InternalExternalDestinationChoice.xls +internalExternal.dc.uec.data.page = 0 +internalExternal.dc.uec.model.page = 1 +internalExternal.dc.uec.alts.file = InternalExternalDestinationChoiceAlternatives.csv + +internalExternal.tour.tod.file = input/internalExternal_tourTOD.csv + +internalExternal.trip.mc.uec.file =InternalExternalTripModeChoice.xls +internalExternal.trip.mc.data.page = 0 +internalExternal.trip.mc.model.page = 1 + +internalExternal.trip.output.file = output/internalExternalTrips.csv + +internalExternal.results.autoTripMatrix = output/autoInternalExternalTrips +internalExternal.results.nMotTripMatrix = output/nmotInternalExternalTrips +internalExternal.results.tranTripMatrix = output/tranInternalExternalTrips +internalExternal.results.othrTripMatrix = output/othrInternalExternalTrips + +##################################################################################### +# Cross-Border Model Settings +##################################################################################### +crossBorder.purpose.nonsentri.file = input/crossBorder_tourPurpose_nonSENTRI.csv +crossBorder.purpose.sentri.file = input/crossBorder_tourPurpose_SENTRI.csv + +crossBorder.tour.tod.file = input/crossBorder_tourEntryAndReturn.csv + +crossBorder.dc.soa.uec.file = CrossBorderDestinationChoiceSample.xls +crossBorder.dc.soa.data.page = 0 +crossBorder.dc.soa.model.page = 1 +crossBorder.dc.soa.size.page = 2 +crossborder.dc.soa.alts.file =${crossborder.dc.soa.alts.file} + +crossBorder.dc.uec.file =${crossBorder.dc.uec.file} +crossBorder.dc.data.page = 0 +crossBorder.dc.model.page = 1 +crossborder.dc.alts.file = CrossBorderDestinationChoiceAlternatives.csv + +crossBorder.dc.colonia.file = input/crossBorder_supercolonia.csv +crossBorder.dc.colonia.distance.parameter = -0.19 +crossBorder.dc.soa.sampleRate = 30 + +#crossBorder.tour.mc.uec.file = CrossBorderTourModeChoice.xls +crossBorder.tour.mc.uec.file =${crossBorder.tour.mc.uec.file} +crossBorder.tour.mc.data.page = 0 +crossBorder.tour.mc.mandatory.model.page = 1 +crossBorder.tour.mc.nonmandatory.model.page = 2 +crossBorder.poe.waittime.file = input/crossBorder_pointOfEntryWaitTime.csv + +crossBorder.trip.mc.uec.file =CrossBorderTripModeChoice.xls +crossBorder.trip.mc.data.page = 0 +crossBorder.trip.mc.model.page = 1 + +crossBorder.stop.frequency.file = input/crossBorder_stopFrequency.csv +crossBorder.stop.purpose.file = input/crossBorder_stopPurpose.csv + +crossBorder.slc.soa.uec.file = CrossBorderStopLocationChoiceSample.xls +crossBorder.slc.soa.data.page = 0 +crossBorder.slc.soa.model.page = 1 +crossBorder.slc.soa.alts.file = SoaTazDistAlts.csv + +crossBorder.slc.uec.file = CrossBorderStopLocationChoice.xls +crossBorder.slc.data.page = 0 +crossBorder.slc.model.page = 1 + +crossBorder.stop.outbound.duration.file = input/crossBorder_outboundStopDuration.csv +crossBorder.stop.inbound.duration.file = input/crossBorder_inboundStopDuration.csv + +crossBorder.tour.output.file = output/crossBorderTours.csv +crossBorder.trip.output.file = output/crossBorderTrips.csv + +crossBorder.results.autoTripMatrix = output/autoCrossBorderTrips +crossBorder.results.nMotTripMatrix = output/nmotCrossBorderTrips +crossBorder.results.tranTripMatrix = output/tranCrossBorderTrips +crossBorder.results.othrTripMatrix = output/othrCrossBorderTrips + +##################################################################################### +# Visitor Model Settings +##################################################################################### +visitor.business.tour.file = input/visitor_businessFrequency.csv +visitor.personal.tour.file = input/visitor_personalFrequency.csv + +visitor.partySize.file = input/visitor_partySize.csv +visitor.autoAvailable.file = input/visitor_autoAvailable.csv +visitor.income.file = input/visitor_income.csv + +visitor.dc.soa.uec.file = VisitorDestinationChoiceSample.xls +visitor.dc.soa.data.page = 0 +visitor.dc.soa.work.page = 1 +visitor.dc.soa.recreate.page = 2 +visitor.dc.soa.dining.page = 3 +visitor.dc.soa.size.page = 4 +visitor.dc.soa.alts.file = SoaTazDistAlts.csv + +visitor.dc.uec.file = VisitorDestinationChoice.xls +visitor.dc.data.page = 0 +visitor.dc.work.page = 1 +visitor.dc.recreate.page = 2 +visitor.dc.dining.page = 3 + +visitor.tour.tod.file = input/visitor_tourTOD.csv + +visitor.mc.uec.file =VisitorTourModeChoice.xls +visitor.mc.data.page = 0 +visitor.mc.model.page = 1 + +visitor.stop.frequency.file = input/visitor_stopFrequency.csv +visitor.stop.purpose.file = input/visitor_stopPurpose.csv +visitor.stop.outbound.duration.file = input/visitor_outboundStopDuration.csv +visitor.stop.inbound.duration.file = input/visitor_inboundStopDuration.csv + +visitor.slc.soa.uec.file = VisitorStopLocationChoiceSample.xls +visitor.slc.soa.data.page = 0 +visitor.slc.soa.model.page = 1 + +visitor.slc.uec.file = VisitorStopLocationChoice.xls +visitor.slc.data.page = 0 +visitor.slc.model.page = 1 + +visitor.trip.mc.uec.file =VisitorTripModeChoice.xls +visitor.trip.mc.data.page = 0 +visitor.trip.mc.model.page = 1 + +visitor.micromobility.uec.file = VisitorMicromobilityChoice.xls +visitor.micromobility.data.page = 0 +visitor.micromobility.model.page = 1 + + + + +visitor.tour.output.file = output/visitorTours.csv +visitor.trip.output.file = output/visitorTrips.csv + +visitor.results.autoTripMatrix = output/autoVisitorTrips +visitor.results.nMotTripMatrix = output/nmotVisitorTrips +visitor.results.tranTripMatrix = output/tranVisitorTrips +visitor.results.othrTripMatrix = output/othrVisitorTrips + + +# These settings are for building an estimation file, not used for main visitor model code +visitor.uec.file = VisitorSize.xls +visitor.uec.data.page = 0 +visitor.uec.sizeTerms.page = 1 + +##################################################################################### +# SAN Airport Model Settings +##################################################################################### +airport.SAN.purpose.file = input/airport_purpose.SAN.csv +airport.SAN.size.file = input/airport_party.SAN.csv +airport.SAN.duration.file = input/airport_nights.SAN.csv +airport.SAN.income.file = input/airport_income.SAN.csv +airport.SAN.departureTime.file = input/airport_departure.SAN.csv +airport.SAN.arrivalTime.file = input/airport_arrival.SAN.csv +airport.SAN.output.file = output/airport_out.SAN.csv + +airport.SAN.dc.uec.file = AirportDestinationChoice.SAN.xls +airport.SAN.dc.data.page = 0 +airport.SAN.dc.size.page = 5 +airport.SAN.dc.segment1.page = 1 +airport.SAN.dc.segment2.page = 2 +airport.SAN.dc.segment3.page = 3 +airport.SAN.dc.segment4.page = 4 + +airport.SAN.mc.uec.file =AirportModeChoice.SAN.xls +airport.SAN.mc.data.page = 0 +airport.SAN.mc.da.page = 1 +airport.SAN.mc.s2.page = 2 +airport.SAN.mc.s3.page = 3 +airport.SAN.mc.transit.page = 4 +airport.SAN.mc.accessMode.page = 5 + +airport.SAN.externalStationFile = uec/InternalExternalDestinationChoiceAlternatives.csv + +airport.SAN.results.autoTripMatrix = output/autoAirportTrips.SAN +airport.SAN.results.nMotTripMatrix = output/nmotAirportTrips.SAN +airport.SAN.results.tranTripMatrix = output/tranAirportTrips.SAN +airport.SAN.results.othrTripMatrix = output/othrAirportTrips.SAN + +##################################################################################### +# CBX Airport Model Settings +##################################################################################### +airport.CBX.purpose.file = input/airport_purpose.CBX.csv +airport.CBX.size.file = input/airport_party.CBX.csv +airport.CBX.duration.file = input/airport_nights.CBX.csv +airport.CBX.income.file = input/airport_income.CBX.csv +airport.CBX.departureTime.file = input/airport_departure.CBX.csv +airport.CBX.arrivalTime.file = input/airport_arrival.CBX.csv +airport.CBX.output.file = output/airport_out.CBX.csv + +airport.CBX.dc.uec.file = AirportDestinationChoice.CBX.xls +airport.CBX.dc.data.page = 0 +airport.CBX.dc.size.page = 5 +airport.CBX.dc.segment1.page = 1 +airport.CBX.dc.segment2.page = 2 +airport.CBX.dc.segment3.page = 3 +airport.CBX.dc.segment4.page = 4 + +airport.CBX.mc.uec.file =AirportModeChoice.CBX.xls +airport.CBX.mc.data.page = 0 +airport.CBX.mc.da.page = 1 +airport.CBX.mc.s2.page = 2 +airport.CBX.mc.s3.page = 3 +airport.CBX.mc.transit.page = 4 +airport.CBX.mc.accessMode.page = 5 + +airport.CBX.externalStationFile = uec/InternalExternalDestinationChoiceAlternatives.csv + +airport.CBX.results.autoTripMatrix = output/autoAirportTrips.CBX +airport.CBX.results.nMotTripMatrix = output/nmotAirportTrips.CBX +airport.CBX.results.tranTripMatrix = output/tranAirportTrips.CBX +airport.CBX.results.othrTripMatrix = output/othrAirportTrips.CBX + +##################################################################################### +# Truck Model Settings +##################################################################################### +truck.DFyear = ${model_years} +truck.luOverRide = "False" + +##################################################################################### +# Commercial Vehicle Model Settings +##################################################################################### +#scale factor to use in cvm trip generation. Also, used during demand import to factor-in demand accordingly +cvm.scale_factor = 1 +#scale factors by vehicle (light, medium, and heavy) and time of day (ea,am,md,pm,ev) - used to boost cvm demand +#light vehicles +cvm.scale_light = 1,2,3.5,2,1 +#medium vehicles +cvm.scale_medium = 1,1,1,1,1 +#heavy vehicles +cvm.scale_heavy = 1,1,1,1,1 +#cvm vehicle shares representing portions of the cvm vehicle trips that go to light-heavy trucks. +#share value should be between 0 and 1. 0 representing none will go to light-heavy truck and 1 means all will go. +cvm.share.light = 0.04 +cvm.share.medium = 0.64 +cvm.share.heavy = 0 + +################################################################# +# Report Section +################################################################# +Report.exportData=True +Report.iteration=3 +Report.tables = taztotap,indivtrips,jointtrips,airporttripsSAN,airporttripsCBX,cbtrips,visitortours,visitortrips,ietrip,commtrip +#aggregate trips eetrip, eitrip, and trucktrip are exported in Python, always +#Report.writeTransitIVT = True +##################################################################################### +# Trip Table Settings +##################################################################################### +# occupancies needed for trip table creation +occ3plus.purpose.Work = 3.34 +occ3plus.purpose.University = 3.34 +occ3plus.purpose.School = 3.34 +occ3plus.purpose.Escort = 3.34 +occ3plus.purpose.Shop = 3.34 +occ3plus.purpose.Maintenance = 3.34 +occ3plus.purpose.EatingOut = 3.34 +occ3plus.purpose.Visiting = 3.34 +occ3plus.purpose.Discretionary = 3.34 +occ3plus.purpose.WorkBased = 3.34 + +################################################################# +# Active Transportation Model Settings +# updated 4/2/2014 wsu +################################################################# +active.node.file = %project.folder%/input/SANDAG_Bike_NODE.dbf +active.node.id = NodeLev_ID +active.node.fieldnames = mgra,taz,x,y,tap,signalized +active.node.columns = MGRA,TAZ,XCOORD,YCOORD,TAP,Signal +active.edge.file = %project.folder%/input/SANDAG_Bike_NET.dbf +active.edge.anode = A +active.edge.bnode = B +active.edge.directional = false +active.edge.fieldnames = functionalClass,distance,gain,bikeClass,lanes,cycleTrack,bikeBlvd,roadsegid +active.edge.columns.ab = Func_Class,Distance,AB_Gain,ABBikeClas,AB_Lanes,Bike2Sep,Bike3Blvd,ROADSEGID +active.edge.columns.ba = Func_Class,Distance,BA_Gain,BABikeClas,BA_Lanes,Bike2Sep,Bike3Blvd,ROADSEGID +active.edge.centroid.field = functionalClass +active.edge.centroid.value = 10 +active.edge.autospermitted.field = functionalClass +active.edge.autospermitted.values = 1, 2, 3, 4, 5, 6, 7 +# distance bins for control of path sampling +active.sample.distance.breaks = 99 +# minimum path sizes of alternative lists for each distance bin +active.sample.pathsizes = 2 +# minimum count of samples for each distance bin +active.sample.count.min = 10 +# maximum count of samples for each distance bin +active.sample.count.max = 100 +# scale of random cost for each sampling iteration where random cost = cost + scale * unif(0,1) * distance +active.sample.random.scale.coef = 0.5 +active.sample.random.scale.link = 0.7 +active.sample.random.seeded = true +active.sample.maxcost = 998 +active.maxdist.walk.mgra = 3.0 +active.maxdist.walk.tap = 1.0 +active.maxdist.bike.taz = ${active.maxdist.bike.taz} +active.maxdist.bike.mgra = ${active.maxdist.bike.mgra} +active.maxdist.micromobility.mgra = 3.0 +active.maxdist.micromobility.tap = 1.0 +active.maxdist.microtransit.mgra = 3.0 +active.maxdist.microtransit.tap = 3.0 +active.output.bike = %project.folder%/output/ +active.output.walk = %project.folder%/output/ +active.coef.distcla0 = ${active.coef.distcla0} +active.coef.distcla1 = ${active.coef.distcla1} +active.coef.distcla2 = ${active.coef.distcla2} +active.coef.distcla3 = ${active.coef.distcla3} +active.coef.dartne2 = ${active.coef.dartne2} +active.coef.dwrongwy = ${active.coef.dwrongwy} +active.coef.dcyctrac = ${active.coef.dcyctrac} +active.coef.dbikblvd = ${active.coef.dbikblvd} +active.coef.nonscenic = 0.300 +active.coef.gain = 0.015 +active.coef.turn = 0.083 +active.coef.signals = 0.040 +active.coef.unlfrma = 0.360 +active.coef.unlfrmi = 0.150 +active.coef.untoma = 0.480 +active.coef.untomi = 0.100 +active.coef.gain.walk = 0.034 + +active.walk.minutes.per.mile = 20 +active.bike.minutes.per.mile = ${active.bike.minutes.per.mile} +active.ebike.ownership = ${active.ebike.ownership} +active.ebike.max.benefit = 10 +active.micromobility.speed = 15 +active.micromobility.variableCost = ${active.micromobility.variableCost} +active.micromobility.fixedCost = ${active.micromobility.fixedCost} +active.micromobility.rentalTime = 1 +active.micromobility.constant = 60 +# 2020 VOT $15 converted to 2010 $ at $12.17 +active.micromobility.vot = 12.17 + +micromobility.uec.file = MicromobilityChoice.xls +micromobility.data.page = 0 +micromobility.model.page = 1 + +active.microtransit.speed = 17 +active.microtransit.variableCost = 0.0 +active.microtransit.fixedCost = ${active.microtransit.fixedCost} +active.microtransit.waitTime = 4.0 +active.microtransit.accessTime = 0.0 +active.microtransit.constant = 120 +active.microtransit.notAvailable = 999 + +active.microtransit.tap.file = input/mobilityHubTaps.csv +active.microtransit.mgra.file = input/mobilityHubMGRAs.csv + +#active.trace.origins.taz = 500, 1000, 1500, 2000, 2500, 3000, 3500, 4000, 4500 +#active.trace.origins.mgra = 1000, 2000, 3000, 4000, 5000, 6000, 7000, 8000, 9000, 10000, 11000, 12000, 13000, 14000, 15000, 16000, 17000, 18000, 19000, 20000 +#active.trace.origins.tap = 1, 3, 5, 7, 8, 9, 15 +#active.trace.exclusive = false +#active.debug.origin = 200003500 +#active.debug.destination = 200003601 + +active.trace.outputassignmentpaths = false + +path.choice.uec.spreadsheet = %project.folder%/uec/BikeTripPathChoice.xls +path.choice.uec.model.sheet = 1 +path.choice.uec.data.sheet = 0 +path.choice.max.path.count = 200 +btpc.alts.file = bike_path_alts.csv +active.logsum.matrix.file.bike.taz = bikeTazLogsum.csv +active.logsum.matrix.file.bike.mgra = bikeMgraLogsum.csv +active.logsum.matrix.file.walk.mgra = walkMgraEquivMinutes.csv +active.logsum.matrix.file.walk.mgratap = walkMgraTapEquivMinutes.csv + +active.bike.write.derived.network = true +active.bike.derived.network.edges = derivedBikeEdges.csv +active.bike.derived.network.nodes = derivedBikeNodes.csv +active.bike.derived.network.traversals = derivedBikeTraversals.csv + +active.assignment.file.bike = bikeAssignmentResults.csv +active.micromobility.file.walk.mgra = microMgraEquivMinutes.csv +active.micromobility.file.walk.mgratap = microMgraTapEquivMinutes.csv + +AtTransitConsistency.xThreshold=1.0 +AtTransitConsistency.yThreshold=1.0 + +##################################################################################### +# SUMMIT Settings +##################################################################################### +summit.output.directory = output/ +# Purposes (which correspond to SUMMIT files) are as follows: +summit.purpose.Work = 1 +summit.purpose.University = 2 +summit.purpose.School = 3 +summit.purpose.Escort = 4 +summit.purpose.Shop = 4 +summit.purpose.Maintenance = 4 +summit.purpose.EatingOut = 5 +summit.purpose.Visiting = 5 +summit.purpose.Discretionary = 5 +summit.purpose.WorkBased = 6 + +summit.filename.1 = Work +summit.filename.2 = University +summit.filename.3 = School +summit.filename.4 = Maintenance +summit.filename.5 = Discretionary +summit.filename.6 = Workbased + +summit.ivt.file.1 = -0.016 +summit.ivt.file.2 = -0.016 +summit.ivt.file.3 = -0.010 +summit.ivt.file.4 = -0.017 +summit.ivt.file.5 = -0.015 +summit.ivt.file.6 = -0.032 + +summit.modes = 26 +# 1=wt,2=dt 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 +summit.mode.array = 0,0,0,0,0,0,0,0,0, 0, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0 + +summit.upperEA = 3 +summit.upperAM = 9 +summit.upperMD = 22 +summit.upperPM = 29 + +################################################################# +# TAPS Creation Settings +# updated 4/28/2015 ymm +################################################################# +taps.formal.premium.maxDist = 10.0 +taps.formal.express.maxDist = 4.0 +taps.formal.local.maxDist = 4.0 +taps.informal.premium.maxDist = 4.0 +taps.informal.express.maxDist = 2.0 +taps.informal.local.maxDist = 2.0 + +taps.premium.modes = 4,5,6,7 +taps.express.modes = 8,9 +taps.local.modes = 10 + +taps.skim = traffic_skims_AM.omx +taps.skim.dist = AM_SOV_NT_M_DIST +taps.skim.time = AM_SOV_NT_M_TIME + +################################################################# +# Military Adjustment Section +# updated 4/26/2016 Wu Sun +################################################################# +RunModel.militaryCtmAdjustment=true + +##################################################################################### +# Special Event Model Settings +# Wu Sun 5/15/2017 +##################################################################################### +specialEvent.event.file = input/specialEvent_eventData.csv + +specialEvent.partySize.file = input/specialEvent_partySize.csv +specialEvent.income.file = input/specialEvent_income.csv + +specialEvent.dc.uec.file = SpecialEventOriginChoice.xls +specialEvent.dc.data.page = 0 +specialEvent.dc.model.page = 1 +specialEvent.dc.size.page = 2 + +specialEvent.saveUtilsAndProbs= false + +specialEvent.trip.mc.uec.file =SpecialEventTripModeChoice.xls +specialEvent.trip.mc.data.page = 0 +specialEvent.trip.mc.model.page = 1 + +specialEvent.tour.output.file = output/specialEventTours.csv +specialEvent.trip.output.file = output/specialEventTrips.csv + +specialEvent.results.autoTripMatrix = output/autoSpecialEventTrips +specialEvent.results.nMotTripMatrix = output/nmotSpecialEventTrips +specialEvent.results.tranTripMatrix = output/tranSpecialEventTrips +specialEvent.results.othrTripMatrix = output/othrSpecialEventTrips + +##################################################################################### +# Transit Shed Properties wsu 8/7/18 +##################################################################################### +RunModel.skipTransitShed= true +#transit access threshold (in minutes, must be integer) +transitShed.threshold=30 +#TOD to use in Transit Shed analysis-EA, AM, MD, PM, and EV +transitShed.TOD=AM +#Transit Shed time components (walk to transit). Options: walkAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime +transitShed.walkTransitTimeComponents=walkAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime +#Transit Shed time components (drive to transit). Options: driveAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime +transitShed.driveTransitTimeComponents=drvAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime + +##################################################################################### +# Smart Signal Properties wsu 8/22/18 +##################################################################################### +smartSignal.factor.LC=${smartSignal.factor.LC} +smartSignal.factor.MA=${smartSignal.factor.MA} +smartSignal.factor.PA=${smartSignal.factor.PA} + +##################################################################################### +##################################################################################### +# Transit Tier 1 EMME Link Name zou 5/7/20 +##################################################################################### +transit.newMode = TIER 1 RAIL +transit.newMode.route = 581,582,583 + +##################################################################################### +# ATDM Properties wsu 8/22/18 +##################################################################################### +atdm.factor = ${atdm.factor} +##################################################################################### +# Transit PCE VEH Conversion cliu 8/19/20 +##################################################################################### +transit.bus.pceveh = 3.0 +##################################################################################### +# Local Drive Run Settings wsu 1/19/19 +##################################################################################### +RunModel.FileMask.Download = output,report,sql,logFiles +RunModel.FileMask.Upload = application,bin,input_truck,uec,output\iter*,output\*_1.csv,output\*_2.csv +##################################################################################### +# Visualizer Settings (run once after feedback loops) +##################################################################################### +visualizer.reference.path = ${visualizer.reference.path} +visualizer.output = SANDAG_Dashboard +visualizer.reference.label = REFERENCE +visualizer.build.label = SDABM + +##################################################################################### +# add year specific vehicle class toll factor wsu 6/18/20 +##################################################################################### +vehicle.class.toll.factor=vehicle_class_toll_factors.csv +vehicle.class.toll.factor.path = input/vehicle_class_toll_factors.csv + +##################################################################################### +# Stochastic traffic assignment settings ag 10/07/20 +##################################################################################### +stochasticHighwayAssignment.distributionType = GUMBEL +stochasticHighwayAssignment.replications = 10 +stochasticHighwayAssignment.aParameter = 1.0 +stochasticHighwayAssignment.bParameter = 0.05 +stochasticHighwayAssignment.seed = 1 + +############################################################################### +# DTA post-processing properties +############################################################################### + +dta.postprocessing.RandomSeed = 1004831 + +dta.postprocessing.outputs.path = %project.folder%/output/ + +dta.postprocessing.disaggregateTOD.path = %project.folder%//input/ +dta.postprocessing.disaggregateZone.path = %project.folder%//input/ +dta.postprocessing.disaggregateNode.path = %project.folder%//input/ + +dta.postprocessing.DetailedTODFile = DetailedTODFactors.csv +dta.postprocessing.NodeFile = NodeFactors.csv +dta.postprocessing.ZoneFile = MGRAFactors.csv +dta.postprocessing.BroadTODFile = BroadTODFactors.csv + +skims.path = %project.folder%/output/ +skims.extension = .omx +da.no.toll.skims.prefix = traffic_skims_ +skims.mat.name.suffix = SOVGPM_TIME + +dta.postprocessing.outputs.TripFile = dtaTripsOut.csv \ No newline at end of file diff --git a/sandag_abm/src/main/resources/serverswap.bat b/sandag_abm/src/main/resources/serverswap.bat new file mode 100644 index 0000000..303c7da --- /dev/null +++ b/sandag_abm/src/main/resources/serverswap.bat @@ -0,0 +1,9 @@ +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 +set PROJECT_DIRECTORY_FWD=%3 + +%PROJECT_DRIVE% +set "SCEN_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%" +set "SCEN_DIR_FWD=%PROJECT_DRIVE%%PROJECT_DIRECTORY_FWD%" + +python.exe %SCEN_DIR%\python\serverswap.py -p %SCEN_DIR_FWD% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/serverswap_files.csv b/sandag_abm/src/main/resources/serverswap_files.csv new file mode 100644 index 0000000..4f155bd --- /dev/null +++ b/sandag_abm/src/main/resources/serverswap_files.csv @@ -0,0 +1,25 @@ +fileName,property,separator,refValue +/bin/CTRampEnv.bat,MAIN,=,MAIN +/bin/CTRampEnv.bat,NODE1,=,NODE1 +/bin/CTRampEnv.bat,NODE2,=,NODE2 +/bin/CTRampEnv.bat,NODE3,=,NODE3 +/bin/CTRampEnv.bat,MAIN_IP,=,ModelIP +/bin/CTRampEnv.bat,HHMGR_IP,=,ModelIP +/bin/CTRampEnv.bat,SNODE,=,SNODE +/bin/CTRampEnv.bat,TRANSCAD_PATH,=,TRANSCAD_PATH +/bin/stopABM.cmd,pskill,\\,MAIN +/bin/stopABM.cmd,pskill,\\,NODE1 +/bin/stopABM.cmd,pskill,\\,NODE2 +/conf/jppf-client.properties,driver1.jppf.server.host, = ,MAIN +/conf/jppf-client.properties,jppf.processing.threads, = ,THREADN1 +/conf/jppf-clientDistributed.properties,driver1.jppf.server.host, = ,MAIN +/conf/jppf-sandag01.properties,jppf.server.host, = ,MAIN +/conf/jppf-sandag01.properties,jppf.processing.threads, = ,THREADM +/conf/jppf-sandag02.properties,jppf.server.host, = ,MAIN +/conf/jppf-sandag02.properties,jppf.processing.threads, = ,THREADN1 +/conf/jppf-sandag03.properties,jppf.server.host, = ,MAIN +/conf/jppf-sandag03.properties,jppf.processing.threads, = ,THREADN2 +/conf/jppf-sandag04.properties,jppf.server.host, = ,MAIN +/conf/jppf-sandag04.properties,jppf.processing.threads, = ,THREADN3 +/conf/sandag_abm.properties,RunModel.MatrixServerAddress,=,ModelIP +/conf/sandag_abm.properties,RunModel.HouseholdServerAddress,=,ModelIP diff --git a/sandag_abm/src/main/resources/setup.bat b/sandag_abm/src/main/resources/setup.bat new file mode 100644 index 0000000..88bf3fe --- /dev/null +++ b/sandag_abm/src/main/resources/setup.bat @@ -0,0 +1 @@ +python ./src/main/python/pythonGUI/setup.py py2exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/stopABM.cmd b/sandag_abm/src/main/resources/stopABM.cmd new file mode 100644 index 0000000..509cfd6 --- /dev/null +++ b/sandag_abm/src/main/resources/stopABM.cmd @@ -0,0 +1,5 @@ +rem stopping all java processes on cluster + +%CD%\pskill \\${master.node.name} java.exe +%CD%\pskill \\${node.1.name} java.exe +%CD%\pskill \\${node.2.name} java.exe diff --git a/sandag_abm/src/main/resources/taskkill.bat b/sandag_abm/src/main/resources/taskkill.bat new file mode 100644 index 0000000..29cd2a6 --- /dev/null +++ b/sandag_abm/src/main/resources/taskkill.bat @@ -0,0 +1,2 @@ +rem kill java tasks +taskkill /F /IM java.exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/updateYearSpecificProps.bat b/sandag_abm/src/main/resources/updateYearSpecificProps.bat new file mode 100644 index 0000000..fe51ecd --- /dev/null +++ b/sandag_abm/src/main/resources/updateYearSpecificProps.bat @@ -0,0 +1,6 @@ +set PROJECT_DRIVE=%1 +set PROJECT_DIRECTORY=%2 + +%PROJECT_DRIVE% +cd %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python +python.exe parameterUpdate.py diff --git a/sandag_abm/src/main/resources/w9xpopen.exe b/sandag_abm/src/main/resources/w9xpopen.exe new file mode 100644 index 0000000000000000000000000000000000000000..f6033fdeaaaf14dc80579adeb1a8827f08d1de4f GIT binary patch literal 49664 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 Tr^i`Tghv5p&5Ap|&Gr8P1XY4u literal 0 HcmV?d00001 From 0f4598ea03382b65b733707fa7eae9cd437a45a9 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Mon, 11 Jul 2022 18:20:09 -0500 Subject: [PATCH 21/30] customizable start values for cluster_id --- notebooks/ZoneAggDemo.ipynb | 253 +++++++++++++++++++++++++++++------- sandag_rsm/zone_agg.py | 10 +- 2 files changed, 211 insertions(+), 52 deletions(-) diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index 4d78d67..a104c0a 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -4,7 +4,11 @@ "cell_type": "code", "execution_count": null, "id": "4fc00298", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "import requests\n", @@ -22,7 +26,11 @@ { "cell_type": "markdown", "id": "de5861aa", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%% md\n" + } + }, "source": [ "## Remote I/O" ] @@ -31,10 +39,16 @@ "cell_type": "code", "execution_count": null, "id": "1d4896bb", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ - "data_dir = \"../test/data/\"\n", + "data_dir = \"./data-dl/\"\n", + "os.makedirs(data_dir, exist_ok=True)\n", + "\n", "resource_url = \"https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/\"" ] }, @@ -42,20 +56,30 @@ "cell_type": "code", "execution_count": null, "id": "4f6ca008", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "mgra_filename = \"mgra13_based_input2016.csv.gz\"\n", "skim_filename = \"traffic_skims_AM_mini.omx\"\n", "trips_filename = \"trips_sample.pq\"\n", - "download_files_vector = [mgra_filename, skim_filename, trips_filename]" + "download_files_vector = [\n", + " mgra_filename, \n", + " trips_filename, skim_filename, ]" ] }, { "cell_type": "code", "execution_count": null, "id": "691bbb9b", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "for download_file in download_files_vector:\n", @@ -66,7 +90,11 @@ { "cell_type": "markdown", "id": "fe41ea9a", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%% md\n" + } + }, "source": [ "## Demo" ] @@ -75,7 +103,11 @@ "cell_type": "code", "execution_count": null, "id": "c88b96b5", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "mgra = load_mgra_data(data_dir=data_dir, simplify_tolerance=10, topo=True)" @@ -85,7 +117,11 @@ "cell_type": "code", "execution_count": null, "id": "ae5d7f1c", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "mgra['taz20'] = mgra.taz % 20" @@ -95,7 +131,11 @@ "cell_type": "code", "execution_count": null, "id": "e6d1c5ef", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "trips = load_trip_list(\"trips_sample.pq\", data_dir=data_dir)" @@ -105,7 +145,11 @@ "cell_type": "code", "execution_count": null, "id": "4bb2cb35", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "tazs = merge_zone_data(mgra, cluster_id=\"taz\")" @@ -115,7 +159,11 @@ "cell_type": "code", "execution_count": null, "id": "f3593b1c", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "trip_mode_shares = trip_mode_shares_by_taz(trips, tazs=tazs.index, mgra_gdf=mgra)" @@ -125,7 +173,11 @@ "cell_type": "code", "execution_count": null, "id": "e64f7a0b", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "tazs = tazs.join(trip_mode_shares.add_prefix(\"modeshare_\"), on='taz')" @@ -135,7 +187,11 @@ "cell_type": "code", "execution_count": null, "id": "65c404fa", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "poi = poi_taz_mgra(mgra)" @@ -145,7 +201,11 @@ "cell_type": "code", "execution_count": null, "id": "569cfadc", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "poi" @@ -155,7 +215,11 @@ "cell_type": "code", "execution_count": null, "id": "2c974741", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}" @@ -165,7 +229,11 @@ "cell_type": "code", "execution_count": null, "id": "6cd4cee2", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "tazs, cluster_factors = attach_poi_taz_skims(\n", @@ -182,12 +250,16 @@ "cell_type": "code", "execution_count": null, "id": "772c319e", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "explicit_agg=[\n", - " 571, 588, 606, \n", - " [143, 270, 15],\n", + "# 571, 588, 606, \n", + "# [143, 270, 15],\n", "]\n" ] }, @@ -195,7 +267,11 @@ "cell_type": "code", "execution_count": null, "id": "2f7c2125", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "d1 = tazs.query(\"district27 == 1\")" @@ -205,7 +281,11 @@ "cell_type": "code", "execution_count": null, "id": "a93bd34a", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer(d1, color='popden', marker_line_width=0)" @@ -215,7 +295,11 @@ "cell_type": "code", "execution_count": null, "id": "073d0c19", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer(d1, color='outside_pendleton_gate_AM_SOV_TR_M_TIME', marker_line_width=0)" @@ -225,7 +309,11 @@ "cell_type": "code", "execution_count": null, "id": "f66915c5", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer(d1, color='modeshare_WT', marker_line_width=0)" @@ -235,12 +323,13 @@ "cell_type": "code", "execution_count": null, "id": "61b2c6c7", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ - "# import itertools\n", - "# cluster_factors={'popden':1, 'empden':1, 'modeshare_NM':100, 'modeshare_WT':100}\n", - "# cluster_factors |= {f\"{i}_{j}\":1 for i,j in itertools.product(poi.keys(), ['AM_SOV_TR_M_TIME'])}\n", "cluster_factors" ] }, @@ -248,7 +337,11 @@ "cell_type": "code", "execution_count": null, "id": "3234d883", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "kmeans1 = aggregate_zones(\n", @@ -265,7 +358,11 @@ "cell_type": "code", "execution_count": null, "id": "b50a4bc3", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer2(edges=kmeans1, colors=d1, color_col='empden')" @@ -275,7 +372,11 @@ "cell_type": "code", "execution_count": null, "id": "872ac31b", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "from sandag_rsm.zone_agg import aggregate_zones_within_districts\n", @@ -294,7 +395,11 @@ "cell_type": "code", "execution_count": null, "id": "f5d6cd56", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "kmeans = kmeans.reset_index(drop=True)" @@ -304,7 +409,11 @@ "cell_type": "code", "execution_count": null, "id": "7e7b56ea", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer2(edges=kmeans, colors=kmeans, color_col='empden')" @@ -314,10 +423,14 @@ "cell_type": "code", "execution_count": null, "id": "aa31c88d", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ - "agglom3full = aggregate_zones_within_districts(\n", + "agglom3full = aggregate_zones(\n", " tazs, \n", " cluster_factors=cluster_factors, \n", " n_zones=2000,\n", @@ -331,8 +444,12 @@ { "cell_type": "code", "execution_count": null, - "id": "b1ac6e48", - "metadata": {}, + "id": "7e8cec6a", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "taz_crosswalk = make_crosswalk(agglom3full, tazs, old_index='taz').sort_values('taz')" @@ -341,8 +458,12 @@ { "cell_type": "code", "execution_count": null, - "id": "08c914b6", - "metadata": {}, + "id": "736eac0a", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "mgra_crosswalk = make_crosswalk(agglom3full, mgra, old_index='MGRA').sort_values('MGRA')" @@ -351,8 +472,12 @@ { "cell_type": "code", "execution_count": null, - "id": "219babb2", - "metadata": {}, + "id": "71956608", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "agglom3full = mark_centroids(agglom3full)" @@ -361,8 +486,12 @@ { "cell_type": "code", "execution_count": null, - "id": "4a3b908f", - "metadata": {}, + "id": "a11469a8", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "mgra_crosswalk.to_csv(\"mgra_crosswalk.csv\", index=False)" @@ -371,18 +500,40 @@ { "cell_type": "code", "execution_count": null, - "id": "a5effc15", - "metadata": {}, + "id": "00c3e83e", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "taz_crosswalk.to_csv(\"taz_crosswalk.csv\", index=False)" ] }, + { + "cell_type": "code", + "execution_count": null, + "id": "ecdfc521", + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, + "outputs": [], + "source": [ + "agglom3full.to_csv(\"cluster_zones.csv\", index=False)" + ] + }, { "cell_type": "code", "execution_count": null, "id": "18d6902a", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer2(edges=agglom3full, colors=agglom3full, color_col='empden')" @@ -392,7 +543,11 @@ "cell_type": "code", "execution_count": null, "id": "391a94af", - "metadata": {}, + "metadata": { + "pycharm": { + "name": "#%%\n" + } + }, "outputs": [], "source": [ "viewer2(edges=agglom3full, colors=agglom3full, color_col='popden')" @@ -415,7 +570,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.9.12" + "version": "3.9.10" }, "toc": { "base_numbering": 1, diff --git a/sandag_rsm/zone_agg.py b/sandag_rsm/zone_agg.py index 44e0afe..1e8b160 100644 --- a/sandag_rsm/zone_agg.py +++ b/sandag_rsm/zone_agg.py @@ -21,7 +21,6 @@ def merge_zone_data( cluster_id="cluster_id", ): if agg_instruction is None: - # TODO: fill out the aggregation data system # make a sliver of area series as a backstop for weighted avg on other things # we add this small value to each weighting, so that if the desired weighting values # are all zero for any weighted average, the geographic area becomes the backup @@ -201,6 +200,7 @@ def aggregate_zones( explicit_agg=(), explicit_col="mgra", agg_instruction=None, + start_cluster_ids=13, ): """ Aggregate zones. @@ -225,6 +225,10 @@ def aggregate_zones( 'mgra' or 'taz' agg_instruction : dict Dictionary passed to pandas `agg` that says how to aggregate data columns. + start_cluster_ids : int, default 13 + Cluster id's start at this value. Can be 1, but typically SANDAG has the + smallest id's reserved for external zones, so starting at a greater value + is typical. Returns ------- @@ -234,7 +238,7 @@ def aggregate_zones( if cluster_factors is None: cluster_factors = {} - n = 1 + n = start_cluster_ids if explicit_agg: explicit_agg_ids = {} for i in explicit_agg: @@ -434,7 +438,7 @@ def make_crosswalk(new_zones, old_zones, new_index="cluster_id", old_index=None) def mark_centroids(gdf, crs="NAD 1983 StatePlane California VI FIPS 0406 Feet"): - c = gdf.to_crs("NAD 1983 StatePlane California VI FIPS 0406 Feet").centroid + c = gdf.to_crs(crs).centroid gdf["centroid_x"] = c.x.astype(int) gdf["centroid_y"] = c.y.astype(int) return gdf From 57d84bb54acb4483e13b8be37def793ab05c51be Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Fri, 15 Jul 2022 13:25:00 -0500 Subject: [PATCH 22/30] Network Centroid Connectors Removing the existing centroid connectors and creating new ones based on aggregate zone structure --- .../toolbox/import/adjust_network_links.py | 156 ++++++++++++++++++ .../emme/toolbox/import/import_network.py | 101 +----------- .../src/main/resources/sandag_abm.properties | 4 +- 3 files changed, 159 insertions(+), 102 deletions(-) create mode 100644 sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py diff --git a/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py b/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py new file mode 100644 index 0000000..7630a5d --- /dev/null +++ b/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py @@ -0,0 +1,156 @@ +#////////////////////////////////////////////////////////////////////////////// +#//// /// +#//// /// +#//// Rights to use and modify are granted to the /// +#//// San Diego Association of Governments and partner agencies. /// +#//// This copyright notice must be preserved. /// +#//// /// +#//// import/adjust_network_links.py /// +#//// /// +#//// /// +#//// /// +#//// /// +#////////////////////////////////////////////////////////////////////////////// +# +# +# deletes the existing centroid connectors and create new centroid connectors +# connecting the centroid of aggregated zones to the original end points (on network) +# of old centroid connectors +# +# Inputs: +# source: path to the location of the input network files +# base_scenario: scenario that has highway network only +# emmebank: the Emme database in which to the new network is published +# external_zone: string "1-12" that refernces to range of external zones +# taz_cwk_file: input csv file created after zone aggregation. It has the crosswalk between existing TAZ to new zone structure +# cluster_zone_file: input csv file created after zone aggregation. It has the centroid coordinates of the new zone structure +# +# + +import inro.modeller as _m +import os + +import pandas as pd +from scipy.spatial import distance + + +def adjust_network_links(source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file) + + taz_cwk = pd.read_csv(os.path.join(source, taz_cwk_file), index_col = 0) + taz_cwk = taz_cwk['cluster_id'].to_dict() + + emmebank = _m.Modeller().emmebank + scenario = emmebank.scenario(base_scenario) + hwy_network = scenario.get_network() + + centroid_nodes = [] + exclude_nodes = [] + + + ext_zones = [int(s) for s in external_zone.split() if s.isdigit()] + + for node in range(ext_zones[0],ext_zones[1],1): + exclude_nodes.append(hwy_network.node(node)) + + for node in hwy_network.centroids(): + if not node in exclude_nodes: + centroid_nodes.append(node) + + i_nodes = [] + j_nodes = [] + data1 = [] + length = [] + links = [] + + for link in hwy_network.links(): + if link.i_node in centroid_nodes: + links.append(link) + i_nodes.append(int(link.i_node)) + j_nodes.append(int(link.j_node)) + data1.append(link.data1) + length.append(link.length) + + df = pd.DataFrame({'links' : links, 'i_nodes' : i_nodes, 'j_nodes': j_nodes, 'ul1_org': data1, 'length_org':length}) + df['i_nodes_new'] = df['i_nodes'].map(taz_cwk) + + #get XY of existing centroids + j_nodes_list = df['j_nodes'].unique() + j_nodes_list = [hwy_network.node(x) for x in j_nodes_list] + + j_nodes = [] + j_x = [] + j_y = [] + for nodes in hwy_network.nodes(): + if nodes in j_nodes_list: + j_nodes.append(nodes) + j_x.append(nodes.x) + j_y.append(nodes.y) + + j_nodes_XY = pd.DataFrame({'j_nodes' : j_nodes, 'j_x' : j_x, 'j_y': j_y}) + j_nodes_XY['j_nodes'] = [int(x) for x in j_nodes_XY['j_nodes']] + df = pd.merge(df, j_nodes_XY, on = 'j_nodes', how = 'left') + + agg_node_coords = pd.read_csv(os.path.join(source, cluster_zone_file)) + df = pd.merge(df, agg_node_coords, left_on = 'i_nodes_new', right_on = 'cluster_id', how = 'left') + df = df.drop(columns = 'cluster_id') + df = df.rename(columns = {'centroid_x' : 'i_new_x', 'centroid_y' : 'i_new_y'}) + + i_coords = zip(df['j_x'], df['j_y']) + j_coords = zip(df['i_new_x'], df['i_new_y']) + + df['length'] = [distance.euclidean(i, j)/5280.0 for i,j in zip(i_coords, j_coords)] + + #delete all the existing centroid nodes + for index,row in df.iterrows(): + if hwy_network.node(row['i_nodes']): + hwy_network.delete_node(row['i_nodes'], True) + + # create new nodes (centroids of clusters) + for index,row in agg_node_coords.iterrows(): + new_node = hwy_network.create_node(row['cluster_id'], is_centroid = True) + new_node.x = int(row['centroid_x']) + new_node.y = int(row['centroid_y']) + + df['type'] = 10 + df['num_lanes'] = 1 + df['vdf'] = 11 + df['ul3'] = 999999 + + final_df = df[["i_nodes_new", "j_nodes", "length", "type", "num_lanes", "vdf", "ul3"]] + final_df = final_df.drop_duplicates() + final_df = final_df.reset_index(drop=True) + final_df['type'] = final_df['type'].astype("int") + + # create new links + for index,row in final_df.iterrows(): + + link_ij = hwy_network.create_link(row['i_nodes_new'], row['j_nodes'], + modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) + link_ij.length = row['length'] + link_ij.type = row['type'].astype("int") + link_ij.num_lanes = row['num_lanes'].astype("int") + link_ij.volume_delay_func = row['vdf'].astype("int") + link_ij.data3 = row['ul3'].astype("int") + link_ij['@lane_ea'] = 1 # had to do this as they are being replaced in highway assignment by the values in these columns + link_ij['@lane_am'] = 1 + link_ij['@lane_md'] = 1 + link_ij['@lane_pm'] = 1 + link_ij['@lane_ev'] = 1 + + + link_ji = hwy_network.create_link(row['j_nodes'], row['i_nodes_new'], + modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) + link_ji.length = row['length'] + link_ji.type = row['type'].astype("int") + link_ji.num_lanes = row['num_lanes'].astype("int") + link_ji.volume_delay_func = row['vdf'].astype("int") + link_ji.data3 = row['ul3'].astype("int") + link_ji['@lane_ea'] = 1 # had to do this as they are being replaced in highway assignment by the values in these columns + link_ji['@lane_am'] = 1 + link_ji['@lane_md'] = 1 + link_ji['@lane_pm'] = 1 + link_ji['@lane_ev'] = 1 + + return(hwy_network) + #publish the highway network to the scenario + #scenario.publish_network(hwy_network) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/import/import_network.py b/sandag_abm/src/main/emme/toolbox/import/import_network.py index 4da01bc..e068a73 100644 --- a/sandag_abm/src/main/emme/toolbox/import/import_network.py +++ b/sandag_abm/src/main/emme/toolbox/import/import_network.py @@ -494,7 +494,6 @@ def create_attributes(scenario, attr_map): transit_scenario.publish_network(transit_network, resolve_attributes=True) if self.merged_scenario_id: self.add_transit_to_traffic(traffic_network, transit_network) - self.adjust_network(traffic_network) finally: if self.merged_scenario_id: scenario.publish_network(traffic_network, resolve_attributes=True) @@ -1823,105 +1822,7 @@ def _split_link(self, network, i_node, j_node, new_node_id): for attr in cost_attrs: link[attr] = 0.5 * link[attr] - def adjust_network(self, hwy_network): - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - taz_cwk = pd.read_csv(_join(self.source, props["taz.to.cluster.crosswalk.file"]), index_col = 0) - taz_cwk = taz_cwk['cluster_id'].to_dict() - - centroid_nodes = [] - exclude_nodes = [] - - for node in range(1,13,1): - exclude_nodes.append(hwy_network.node(node)) - - for node in hwy_network.centroids(): - if not node in exclude_nodes: - centroid_nodes.append(node) - - i_nodes = [] - j_nodes = [] - data1 = [] - length = [] - links = [] - - for link in hwy_network.links(): - if link.i_node in centroid_nodes: - links.append(link) - i_nodes.append(int(link.i_node)) - j_nodes.append(int(link.j_node)) - data1.append(link.data1) - length.append(link.length) - - df = pd.DataFrame({'links' : links, 'i_nodes' : i_nodes, 'j_nodes': j_nodes, 'ul1_org': data1, 'length_org':length}) - df['i_nodes_new'] = df['i_nodes'].map(taz_cwk) - - j_nodes_list = df['j_nodes'].unique() - j_nodes_list = [hwy_network.node(x) for x in j_nodes_list] - - j_nodes = [] - j_x = [] - j_y = [] - for nodes in hwy_network.nodes(): - if nodes in j_nodes_list: - j_nodes.append(nodes) - j_x.append(nodes.x) - j_y.append(nodes.y) - - j_nodes_XY = pd.DataFrame({'j_nodes' : j_nodes, 'j_x' : j_x, 'j_y': j_y}) - j_nodes_XY['j_nodes'] = [int(x) for x in j_nodes_XY['j_nodes']] - df = pd.merge(df, j_nodes_XY, on = 'j_nodes', how = 'left') - - agg_node_coords = pd.read_csv(_join(self.source, props["cluster.zone.centroid.file"])) - df = pd.merge(df, agg_node_coords, left_on = 'i_nodes_new', right_on = 'cluster_id', how = 'left') - df = df.drop(columns = 'cluster_id') - df = df.rename(columns = {'centroid_x' : 'i_new_x', 'centroid_y' : 'i_new_y'}) - - i_coords = zip(df['j_x'], df['j_y']) - j_coords = zip(df['i_new_x'], df['i_new_y']) - - df['length'] = [distance.euclidean(i, j)/5280.0 for i,j in zip(i_coords, j_coords)] - - #delete all the existing centroid nodes - for index,row in df.iterrows(): - if hwy_network.node(row['i_nodes']): - hwy_network.delete_node(row['i_nodes'], True) - - # create new nodes (centroids of clusters) - for index,row in agg_node_coords.iterrows(): - new_node = hwy_network.create_node(row['cluster_id'], is_centroid = True) - new_node.x = int(row['centroid_x']) - new_node.y = int(row['centroid_y']) - - df['type'] = 10 - df['num_lanes'] = 1.0 - df['vdf'] = 11 - df['ul3'] = 999999 - - # create new links - for index,row in df.iterrows(): - try: - link_ij = hwy_network.create_link(row['i_nodes_new'], row['j_nodes'], - modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) - link_ij.length = row['length'] - link_ij.type = row['type'] - link_ij.num_lanes = row['num_lanes'] - link_ij.volume_delay_func = row['vdf'] - link_ij.data3 = row['ul3'] - - link_ji = hwy_network.create_link(row['j_nodes'], row['i_nodes_new'], - modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) - link_ji.length = row['length'] - link_ji.type = row['type'] - link_ji.num_lanes = row['num_lanes'] - link_ji.volume_delay_func = row['vdf'] - link_ji.data3 = row['ul3'] - - except: - continue - - + @_m.logbook_trace("Set database functions (VDF, TPF and TTF)") def set_functions(self, scenario): create_function = _m.Modeller().tool( diff --git a/sandag_abm/src/main/resources/sandag_abm.properties b/sandag_abm/src/main/resources/sandag_abm.properties index 9325e4a..a93d575 100644 --- a/sandag_abm/src/main/resources/sandag_abm.properties +++ b/sandag_abm/src/main/resources/sandag_abm.properties @@ -179,8 +179,8 @@ truck.FFyear =${year} transponder.destinations = 4027,2563,2258 #taz crosswalk file -taz.to.cluster.crosswalk.file = taz_crosswalk.csv -cluster.zone.centroid.file = cluster_zones.csv +taz.to.cluster.crosswalk.file = input/taz_crosswalk.csv +cluster.zone.centroid.file = input/cluster_zones.csv ############################################################################################ # EMERGING MOBILITY SECTION: MODIFY WHEN CHANGE AV, TNC, and MICROMOBILITY ASSUMPTIONS From e2d8f61218a7697d02442fdfb42d3ea3c167939f Mon Sep 17 00:00:00 2001 From: Vivek Yadav Date: Fri, 15 Jul 2022 13:48:12 -0500 Subject: [PATCH 23/30] Implement as method --- sandag_abm/src/main/emme/toolbox/master_run.py | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/sandag_abm/src/main/emme/toolbox/master_run.py b/sandag_abm/src/main/emme/toolbox/master_run.py index b24daf9..de0a8ad 100644 --- a/sandag_abm/src/main/emme/toolbox/master_run.py +++ b/sandag_abm/src/main/emme/toolbox/master_run.py @@ -294,6 +294,10 @@ def __call__(self, main_directory, scenario_id, scenario_title, emmebank_title, visualizer_build_label = props["visualizer.build.label"] mgraInputFile = props["mgra.socec.file"] + #for zone restructing in network files + taz_cwk_file = props["taz.to.cluster.crosswalk.file"] + cluster_zone_file = props["cluster.zone.centroid.file"] + period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) useLocalDrive = props["RunModel.useLocalDrive"] @@ -418,6 +422,16 @@ def __call__(self, main_directory, scenario_id, scenario_title, emmebank_title, except ImportError as e: pass + hwy_network = self.update_centroid_connectors( + input_dir, + base_scenario, + main_emmebank, + external_zones, + taz_cwk_file, + cluster_zone_file) + + base_scenario.publish_network(hwy_network) + if not skipInputChecker: input_checker(path=self._path) @@ -1134,6 +1148,10 @@ def get_link_attributes(self): export_utils = _m.Modeller().module("inro.emme.utility.export_utilities") return export_utils.get_link_attributes(_m.Modeller().scenario) + def update_centroid_connectors(self, source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file): + adjust_network = _m.Modeller().module("inro.import.adjust_network_links") + return adjust_network.adjust_network_links(source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file) + def sql_select_scenario(self, year, iteration, sample, path, dbtime): # YMA, 1/24/2019 """Return scenario_id from [dimension].[scenario] given path""" From 23202e5ff6e2e8818701c5786da4100948f3d01a Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 19 Oct 2022 07:55:30 -0500 Subject: [PATCH 24/30] add load data function --- .pre-commit-config.yaml | 8 ++++---- sandag_rsm/data_load/__init__.py | 34 ++++++++++++++++++++++++++++++++ 2 files changed, 38 insertions(+), 4 deletions(-) create mode 100644 sandag_rsm/data_load/__init__.py diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 35cc926..333916f 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,7 +1,7 @@ repos: - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.1.0 + rev: v4.3.0 hooks: - id: check-yaml - id: end-of-file-fixer @@ -9,7 +9,7 @@ repos: - id: trailing-whitespace - repo: https://github.com/kynan/nbstripout - rev: 0.5.0 + rev: 0.6.1 hooks: - id: nbstripout @@ -20,11 +20,11 @@ repos: args: ["--profile", "black", "--filter-files"] - repo: https://github.com/psf/black - rev: 21.12b0 + rev: 22.10.0 hooks: - id: black - repo: https://github.com/PyCQA/flake8 - rev: 4.0.1 + rev: 5.0.4 hooks: - id: flake8 diff --git a/sandag_rsm/data_load/__init__.py b/sandag_rsm/data_load/__init__.py new file mode 100644 index 0000000..0e02160 --- /dev/null +++ b/sandag_rsm/data_load/__init__.py @@ -0,0 +1,34 @@ +import os +from pathlib import Path + +import requests + + +def get_test_file(download_file, destination_dir="."): + """ + Download one or more test files from GitHib resources. + + Parameters + ---------- + download_file : str or list[str] + One or more test file names to download from GitHib resources. + destination_dir : path-like, optional + Location to save downloaded files. + + """ + resource_urls = [ + "https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/", + "https://raw.githubusercontent.com/wsp-sag/client_sandag_rsm_resources/main/", + "https://media.githubusercontent.com/media/camsys/client_sandag_rsm_resources/main/", + "https://raw.githubusercontent.com/camsys/client_sandag_rsm_resources/main/", + ] + if isinstance(download_file, (str, Path)): + while True: + for resource_url in resource_urls: + r = requests.get((resource_url + download_file), allow_redirects=True) + open(os.path.join(destination_dir + download_file), "wb").write( + r.content + ) + else: + for f in download_file: + get_test_file(f, destination_dir) From 5fc482235e2de1126f914f546f9086aab73fd861 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 19 Oct 2022 08:07:15 -0500 Subject: [PATCH 25/30] test data loader --- sandag_rsm/data_load/__init__.py | 18 ++++++++++++++---- 1 file changed, 14 insertions(+), 4 deletions(-) diff --git a/sandag_rsm/data_load/__init__.py b/sandag_rsm/data_load/__init__.py index 0e02160..96370c6 100644 --- a/sandag_rsm/data_load/__init__.py +++ b/sandag_rsm/data_load/__init__.py @@ -1,8 +1,11 @@ +import logging import os from pathlib import Path import requests +logger = logging.getLogger(__name__) + def get_test_file(download_file, destination_dir="."): """ @@ -16,6 +19,7 @@ def get_test_file(download_file, destination_dir="."): Location to save downloaded files. """ + os.makedirs(destination_dir, exist_ok=True) resource_urls = [ "https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/", "https://raw.githubusercontent.com/wsp-sag/client_sandag_rsm_resources/main/", @@ -23,12 +27,18 @@ def get_test_file(download_file, destination_dir="."): "https://raw.githubusercontent.com/camsys/client_sandag_rsm_resources/main/", ] if isinstance(download_file, (str, Path)): - while True: - for resource_url in resource_urls: - r = requests.get((resource_url + download_file), allow_redirects=True) - open(os.path.join(destination_dir + download_file), "wb").write( + if os.path.exists(os.path.join(destination_dir, download_file)): + logger.warning(f"file {download_file!r} already exists") + return + for resource_url in resource_urls: + r = requests.get((resource_url + download_file), allow_redirects=True) + if r.ok: + open(os.path.join(destination_dir, download_file), "wb").write( r.content ) + break + else: + raise FileNotFoundError(download_file) else: for f in download_file: get_test_file(f, destination_dir) From b99c07b04a9016533182edf5d9ce9b6ba87c130b Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 19 Oct 2022 08:16:17 -0500 Subject: [PATCH 26/30] fix error when no chdir --- sandag_rsm/data_load/skims.py | 3 ++- sandag_rsm/data_load/zones.py | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/sandag_rsm/data_load/skims.py b/sandag_rsm/data_load/skims.py index 824de00..51dea70 100644 --- a/sandag_rsm/data_load/skims.py +++ b/sandag_rsm/data_load/skims.py @@ -24,4 +24,5 @@ def open_skims( finally: # change back to original cwd - os.chdir(cwd) + if cwd is not None: + os.chdir(cwd) diff --git a/sandag_rsm/data_load/zones.py b/sandag_rsm/data_load/zones.py index c6d0a70..06e6fea 100644 --- a/sandag_rsm/data_load/zones.py +++ b/sandag_rsm/data_load/zones.py @@ -92,4 +92,5 @@ def load_mgra_data( finally: # change back to original cwd - os.chdir(cwd) + if cwd is not None: + os.chdir(cwd) From 13ce926e3218a7092200a7e564417ae3841f0fa8 Mon Sep 17 00:00:00 2001 From: Jeff Newman Date: Wed, 19 Oct 2022 08:18:51 -0500 Subject: [PATCH 27/30] update notebook --- notebooks/ZoneAggDemo.ipynb | 64 ++++++++----------------------------- 1 file changed, 14 insertions(+), 50 deletions(-) diff --git a/notebooks/ZoneAggDemo.ipynb b/notebooks/ZoneAggDemo.ipynb index a104c0a..310db9d 100644 --- a/notebooks/ZoneAggDemo.ipynb +++ b/notebooks/ZoneAggDemo.ipynb @@ -3,7 +3,7 @@ { "cell_type": "code", "execution_count": null, - "id": "4fc00298", + "id": "a845c147", "metadata": { "pycharm": { "name": "#%%\n" @@ -11,7 +11,6 @@ }, "outputs": [], "source": [ - "import requests\n", "import os\n", "\n", "from sandag_rsm.data_load.zones import load_mgra_data\n", @@ -25,7 +24,7 @@ }, { "cell_type": "markdown", - "id": "de5861aa", + "id": "73a4f97c", "metadata": { "pycharm": { "name": "#%% md\n" @@ -38,7 +37,7 @@ { "cell_type": "code", "execution_count": null, - "id": "1d4896bb", + "id": "542ccfa1", "metadata": { "pycharm": { "name": "#%%\n" @@ -46,55 +45,24 @@ }, "outputs": [], "source": [ + "from sandag_rsm.data_load import get_test_file\n", "data_dir = \"./data-dl/\"\n", - "os.makedirs(data_dir, exist_ok=True)\n", "\n", - "resource_url = \"https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/\"" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "4f6ca008", - "metadata": { - "pycharm": { - "name": "#%%\n" - } - }, - "outputs": [], - "source": [ "mgra_filename = \"mgra13_based_input2016.csv.gz\"\n", "skim_filename = \"traffic_skims_AM_mini.omx\"\n", "trips_filename = \"trips_sample.pq\"\n", - "download_files_vector = [\n", + "\n", + "get_test_file([\n", " mgra_filename, \n", - " trips_filename, skim_filename, ]" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "691bbb9b", - "metadata": { - "pycharm": { - "name": "#%%\n" - } - }, - "outputs": [], - "source": [ - "for download_file in download_files_vector:\n", - " r = requests.get((resource_url+download_file), allow_redirects=True)\n", - " open((data_dir+download_file), 'wb').write(r.content)" + " trips_filename, \n", + " skim_filename, \n", + "], data_dir)" ] }, { "cell_type": "markdown", - "id": "fe41ea9a", - "metadata": { - "pycharm": { - "name": "#%% md\n" - } - }, + "id": "4b5e3239", + "metadata": {}, "source": [ "## Demo" ] @@ -102,12 +70,8 @@ { "cell_type": "code", "execution_count": null, - "id": "c88b96b5", - "metadata": { - "pycharm": { - "name": "#%%\n" - } - }, + "id": "b8dc27e3", + "metadata": {}, "outputs": [], "source": [ "mgra = load_mgra_data(data_dir=data_dir, simplify_tolerance=10, topo=True)" @@ -570,7 +534,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.9.10" + "version": "3.10.6" }, "toc": { "base_numbering": 1, From cc5609ec80618cdcc06591357dd6edb64e974575 Mon Sep 17 00:00:00 2001 From: Ashish Kulshrestha Date: Thu, 27 Oct 2022 14:31:48 -0700 Subject: [PATCH 28/30] translate demand code for aggregating omx matrices --- notebooks/TranslateDemand.ipynb | 293 ++++++++++++++++++++------------ sandag_rsm/emme.py | 37 ---- sandag_rsm/translate.py | 129 ++++++++++---- 3 files changed, 279 insertions(+), 180 deletions(-) delete mode 100644 sandag_rsm/emme.py diff --git a/notebooks/TranslateDemand.ipynb b/notebooks/TranslateDemand.ipynb index 1165ef3..822302d 100644 --- a/notebooks/TranslateDemand.ipynb +++ b/notebooks/TranslateDemand.ipynb @@ -2,23 +2,16 @@ "cells": [ { "cell_type": "code", - "execution_count": null, + "execution_count": 18, "id": "4fc00298", "metadata": {}, "outputs": [], "source": [ + "import os\n", "import requests\n", + "import openmatrix as omx\n", "\n", - "from sandag_rsm.data_load.zones import load_mgra_data\n", - "from sandag_rsm.data_load.triplist import load_trip_list, trip_mode_shares_by_mgra, trip_mode_shares_by_taz\n", - "from sandag_rsm.poi import poi_taz_mgra, attach_poi_taz_skims\n", - "from sandag_rsm.zone_agg import aggregate_zones, viewer, viewer2, aggregate_zones_within_districts, merge_zone_data\n", - "\n", - "from sandag_rsm.translate import translate_demand\n", - "import os\n", - "import numpy as np\n", - "import pandas as pd\n", - "import geopandas as gpd" + "from sandag_rsm.translate import translate_demand" ] }, { @@ -31,57 +24,51 @@ }, { "cell_type": "code", - "execution_count": null, + "execution_count": 12, "id": "1d4896bb", "metadata": {}, "outputs": [], "source": [ - "data_dir = \"../test/data/\"\n", - "export_dir = \"../test/export/\"\n", - "resource_url = \"https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/\"" + "data_dir = './data-dl/'\n", + "\n", + "os.makedirs(data_dir, exist_ok=True)\n", + "\n", + "resource_url = 'https://media.githubusercontent.com/media/wsp-sag/client_sandag_rsm_resources/main/original_omx/'\n", + "\n", + "download_files_vector = [\n", + " 'trip_EA.omx',\n", + " 'trip_AM.omx',\n", + " 'trip_MD.omx',\n", + " 'trip_PM.omx',\n", + " 'trip_EV.omx',\n", + "]\n", + "\n", + "# for download_file in download_files_vector:\n", + "# r = requests.get((resource_url+download_file), allow_redirects=True)\n", + "# open((data_dir+download_file), 'w').write(r.content)" ] }, { "cell_type": "code", - "execution_count": null, - "id": "4f6ca008", + "execution_count": 13, + "id": "e09fa2ee", "metadata": {}, "outputs": [], "source": [ - "skim_filename = \"traffic_skims_AM_mini.omx\"\n", - "matrix_name = 'AM_SOV_TR_M_TIME'" - ] - }, - { - "cell_type": "raw", - "id": "9323795c-fa59-4669-bc08-56b5b4732066", - "metadata": {}, - "source": [ - "for download_file in download_files_vector:\n", - " r = requests.get((resource_url+download_file), allow_redirects=True)\n", - " open((data_dir+download_file), 'wb').write(r.content)" + "input_dir = './data-dl/'\n", + "output_dir = './data-dl/export/'\n", + "matrix_names = ['trip_EA.omx', 'trip_AM.omx', 'trip_MD.omx', 'trip_PM.omx', 'trip_EV.omx']\n", + "agg_zone_mapping = './../test/data/taz_crosswalk.csv'" ] }, { "cell_type": "code", - "execution_count": null, - "id": "2c158043-e9cf-40c5-a8c3-c74ef9eccc1d", + "execution_count": 14, + "id": "b23b92f9", "metadata": {}, "outputs": [], "source": [ - "## Create Dummy 'zone_to_cluster' Dictionary\n", - "zone_to_cluster = {i:1+i%100 for i in range(4996)}" - ] - }, - { - "cell_type": "raw", - "id": "ad52614e-4d7d-42f9-8c4a-6239457dd0ff", - "metadata": {}, - "source": [ - "## Use Prototype Crosswalks for Testing\n", - "zone_to_cluster = pd.read_csv(os.path.join(data_dir, 'taz_crosswalk.csv')).sort_values('taz')\n", - "zone_to_cluster['taz'] = zone_to_cluster['taz'] - 1\n", - "zone_to_cluster = dict(zip(zone_to_cluster['taz'], zone_to_cluster['cluster_id']))" + "os.makedirs(output_dir, exist_ok=True)" ] }, { @@ -89,126 +76,215 @@ "id": "fe41ea9a", "metadata": {}, "source": [ - "## Demo" + "## Aggregate Matrices" ] }, { "cell_type": "code", - "execution_count": null, + "execution_count": 15, "id": "458ad160-33cf-4149-94ca-b436c2a5aafb", "metadata": {}, - "outputs": [], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "[04:47.70] INFO: Agregating Matrix: trip_EA.omx ...\n", + "[06:23.38] INFO: Agregating Matrix: trip_AM.omx ...\n", + "[08:00.64] INFO: Agregating Matrix: trip_MD.omx ...\n", + "[09:39.43] INFO: Agregating Matrix: trip_PM.omx ...\n", + "[11:16.33] INFO: Agregating Matrix: trip_EV.omx ...\n" + ] + } + ], "source": [ - "matrix_load, matrix = translate_demand(skim_filename, matrix_name, zone_to_cluster, data_dir, export_dir)" + "translate_demand(\n", + " matrix_names,\n", + " agg_zone_mapping,\n", + " input_dir,\n", + " output_dir\n", + ")" ] }, { - "cell_type": "code", - "execution_count": null, - "id": "eb4bcad1-fe01-405c-8da7-96ac040a10d6", + "cell_type": "markdown", + "id": "62a0c438", "metadata": {}, - "outputs": [], "source": [ - "print('Loaded Matrix')\n", - "print(f'total sum: {matrix_load.sum().sum():,.0f}\\n')\n", - "print(matrix_load.sum().describe().apply(\"{0:,.2f}\".format))" + "## Compare Original and Aggregated Matrices" ] }, { "cell_type": "code", - "execution_count": null, - "id": "692ecce3-946b-4ca1-ac01-b8610991b8bc", + "execution_count": 16, + "id": "cd5d0436", "metadata": {}, "outputs": [], "source": [ - "print('Aggregated Matrix')\n", - "print(f'total sum: {matrix.sum().sum():,.0f}\\n')\n", - "print(matrix.sum().describe().apply(\"{0:,.2f}\".format))" + "matrix_name = 'trip_AM.omx'" ] }, { "cell_type": "code", - "execution_count": null, - "id": "c3030557-03f4-402c-8da2-306c0b4f268c", + "execution_count": 20, + "id": "eb4bcad1-fe01-405c-8da7-96ac040a10d6", "metadata": {}, "outputs": [], "source": [ - "matrix_load" + "input_matrix = omx.open_file(os.path.join(input_dir, matrix_name), mode=\"r\") \n", + "output_matrix = omx.open_file(os.path.join(output_dir, matrix_name), mode=\"r\")" ] }, { "cell_type": "code", - "execution_count": null, - "id": "650baf9b-569f-424b-a703-5cb7c75110d4", + "execution_count": 21, + "id": "692ecce3-946b-4ca1-ac01-b8610991b8bc", "metadata": {}, "outputs": [], "source": [ - "matrix" + "matrix_cores = input_matrix.list_matrices()" ] }, { "cell_type": "code", - "execution_count": null, - "id": "a9552989-8251-445f-a3b1-4070d66c06fa", - "metadata": {}, - "outputs": [], - "source": [] - }, - { - "cell_type": "markdown", - "id": "a639d2b1-c5fd-461d-8a58-6fc26b6987a7", + "execution_count": 22, + "id": "c3030557-03f4-402c-8da2-306c0b4f268c", "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "(4996, 4996)" + ] + }, + "execution_count": 22, + "metadata": {}, + "output_type": "execute_result" + } + ], "source": [ - "## Visualize" + "input_matrix.shape()" ] }, { "cell_type": "code", - "execution_count": null, - "id": "b9f93146-e68b-4d62-a38f-4b655335fcdb", + "execution_count": 23, + "id": "650baf9b-569f-424b-a703-5cb7c75110d4", "metadata": {}, - "outputs": [], + "outputs": [ + { + "data": { + "text/plain": [ + "(2012, 2012)" + ] + }, + "execution_count": 23, + "metadata": {}, + "output_type": "execute_result" + } + ], "source": [ - "use_cluster = 731\n", - "use_col = f'{matrix_name}_{use_cluster}'\n", - "\n", - "plot_col = matrix[use_cluster].reset_index()\n", - "plot_col.columns = ['cluster_id', use_col]" + "output_matrix.shape()" ] }, { "cell_type": "code", - "execution_count": null, - "id": "5ac6939e-2dac-4ffa-9007-1243ec608e32", + "execution_count": 24, + "id": "a9552989-8251-445f-a3b1-4070d66c06fa", "metadata": {}, - "outputs": [], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Core: AM_HOV2_H\n", + "Input Sum: 90,480\n", + "Output Sum: 90,480\n", + "\n", + "Core: AM_HOV2_L\n", + "Input Sum: 101,330\n", + "Output Sum: 101,330\n", + "\n", + "Core: AM_HOV2_M\n", + "Input Sum: 229,308\n", + "Output Sum: 229,308\n", + "\n", + "Core: AM_HOV3_H\n", + "Input Sum: 81,707\n", + "Output Sum: 81,707\n", + "\n", + "Core: AM_HOV3_L\n", + "Input Sum: 38,662\n", + "Output Sum: 38,662\n", + "\n", + "Core: AM_HOV3_M\n", + "Input Sum: 98,403\n", + "Output Sum: 98,403\n", + "\n", + "Core: AM_SOV_NT_H\n", + "Input Sum: 222,616\n", + "Output Sum: 222,616\n", + "\n", + "Core: AM_SOV_NT_L\n", + "Input Sum: 535,342\n", + "Output Sum: 535,342\n", + "\n", + "Core: AM_SOV_NT_M\n", + "Input Sum: 426,238\n", + "Output Sum: 426,238\n", + "\n", + "Core: AM_SOV_TR_H\n", + "Input Sum: 185,345\n", + "Output Sum: 185,345\n", + "\n", + "Core: AM_SOV_TR_L\n", + "Input Sum: 21,590\n", + "Output Sum: 21,590\n", + "\n", + "Core: AM_SOV_TR_M\n", + "Input Sum: 16,563\n", + "Output Sum: 16,563\n", + "\n", + "Core: AM_TRK_H\n", + "Input Sum: 58,381\n", + "Output Sum: 58,381\n", + "\n", + "Core: AM_TRK_L\n", + "Input Sum: 32,998\n", + "Output Sum: 32,998\n", + "\n", + "Core: AM_TRK_M\n", + "Input Sum: 14,039\n", + "Output Sum: 14,039\n", + "\n" + ] + } + ], "source": [ - "mgra = gpd.read_file(os.path.join(data_dir, 'MGRASHAPE.zip'))\n", - "mgra_xref = pd.read_csv(os.path.join(data_dir, 'mgra_crosswalk.csv'))\n", - "\n", - "clusters = pd.merge(mgra, mgra_xref, on='MGRA', how='left')\n", - "clusters = clusters[['cluster_id', 'geometry']].dissolve(by='cluster_id').reset_index()\n", - "\n", - "clusters = pd.merge(clusters, plot_col, on='cluster_id', how='left')" + "for core in matrix_cores:\n", + " input_core = input_matrix[core].read()\n", + " output_core = output_matrix[core].read()\n", + " \n", + " input_mtx_sum = input_core.sum().sum()\n", + " output_mtx_sum = input_core.sum().sum()\n", + " \n", + " print(f'Core: {core}')\n", + " print(f'Input Sum: {input_mtx_sum:,.0f}')\n", + " print(f'Output Sum: {output_mtx_sum:,.0f}\\n')\n", + " \n", + " assert output_mtx_sum == input_mtx_sum" ] }, { "cell_type": "code", - "execution_count": null, - "id": "a93bd34a", + "execution_count": 25, + "id": "da9761de-2fec-4ba5-a5c1-f0f276398195", "metadata": {}, "outputs": [], "source": [ - "viewer(clusters, color=use_col, marker_line_width=0)" + "input_matrix.close()\n", + "output_matrix.close()" ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "da9761de-2fec-4ba5-a5c1-f0f276398195", - "metadata": {}, - "outputs": [], - "source": [] } ], "metadata": { @@ -227,7 +303,7 @@ "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", - "version": "3.9.10" + "version": "3.9.12" }, "toc": { "base_numbering": 1, @@ -241,6 +317,11 @@ "toc_position": {}, "toc_section_display": true, "toc_window_display": false + }, + "vscode": { + "interpreter": { + "hash": "6969d5340a2324284ea9e82958789f0af31a889f2a17dbf954a94cd3bfb1e1ef" + } } }, "nbformat": 4, diff --git a/sandag_rsm/emme.py b/sandag_rsm/emme.py deleted file mode 100644 index d9124d0..0000000 --- a/sandag_rsm/emme.py +++ /dev/null @@ -1,37 +0,0 @@ -import os -import inro.emme.database.emmebank as _eb -import inro.modeller as _m - - -def load_matrix_emme( - skims_file, emme_path -): -""" -SAMPLE CODE -_m = inro.modeller -NAMESPACE = "inro.emme.data.matrix.export_to_omx" -export_to_omx = _m.Modeller().tool(NAMESPACE) -emmebank_dir = os.path.dirname(_m.Modeller().emmebank.path) -omx_file = os.path.join(emmebank_dir, "exported_demand_matrices.omx") -export_to_omx(matrices=["mf1", "mf2"], - export_file=omx_file, - append_to_file=False) -""" - - - -def save_matrix_emme( - emme_path, matrix_name -): -""" -SAMPLE CODE -_m = inro.modeller -NAMESPACE = "inro.emme.data.matrix.create_matrix" -create_matrix = _m.Modeller().tool(NAMESPACE) -new_mat = create_matrix(matrix_id="mf13", - matrix_name="diftrt", - matrix_description="transit time difference", - default_value=0) - - - diff --git a/sandag_rsm/translate.py b/sandag_rsm/translate.py index 3a5f298..c1fde7a 100644 --- a/sandag_rsm/translate.py +++ b/sandag_rsm/translate.py @@ -2,45 +2,100 @@ import logging import pandas as pd from pathlib import Path - -from .data_load.skims import open_skims +import openmatrix as omx logger = logging.getLogger(__name__) + +def _aggregate_matrix(input_mtx, aggregate_mapping_dict): + matrix_array = input_mtx.read() + matrix_df = pd.DataFrame(matrix_array, columns = list(aggregate_mapping_dict.keys())) + + matrix_agg_df = matrix_df.rename(columns=(aggregate_mapping_dict)) + matrix_agg_df.index = list(aggregate_mapping_dict.values()) + + matrix_agg_df = matrix_agg_df.stack().groupby(level=[0,1]).sum().unstack() + matrix_agg_df = matrix_agg_df[sorted(matrix_agg_df.columns)] + matrix_agg_df = matrix_agg_df.sort_index() + + output_mtx = matrix_agg_df.to_numpy() + + return output_mtx + def translate_demand( - skims_file, matrix_name, agg_map, data_dir=None, export_dir=None -): - matrix_load = load_matrix(skims_file, matrix_name, data_dir) - matrix = aggregate_matrix(matrix_load, agg_map) - export_matrix(matrix, matrix_name, export_dir) - - return matrix_load, matrix - - -def load_matrix( - skims_file, matrix_name, data_dir -): - logger.info('Loading Matrix: matrix_name') - # if isinstance(skims_file, emme_path): - # matrix_load = load_matrix_emme(skims_file, emme_path=emme_path) - if isinstance(skims_file, (str, Path)): - matrix_load = open_skims(skims_file, data_dir=data_dir) - - return pd.DataFrame(matrix_load[matrix_name]) - -def aggregate_matrix( - matrix, agg_map -): - logger.info('\tAggregating Matrix') - matrix = matrix.rename(columns=(agg_map)) - matrix = matrix.rename(index=(agg_map)) - - return matrix.stack().groupby(level=[0,1]).sum().unstack() - -def export_matrix( - matrix, matrix_name, export_dir -): - logger.info('\tExporting Matrix') - matrix.to_csv(os.path.join(export_dir, f'{matrix_name}.csv')) - # save_matrix_emme(emme_path, matrix_name) + matrix_names, + agg_zone_mapping, + input_dir=".", + output_dir="." +): + """ + aggregates the omx demand matrix to aggregated zone system + + Parameters + ---------- + matrix_names : list + omx matrix filenames to aggregate + agg_zone_mapping: Path-like or pandas.DataFrame + zone number mapping between original and aggregated zones. + columns: original zones as 'taz' and aggregated zones as 'cluster_id' + input_dir : Path-like, default "." + output_dir : Path-like, default "." + + Returns + ------- + + """ + + input_dir = Path(input_dir or ".") + output_dir = Path(output_dir or ".") + + def _resolve_df(x): + if isinstance(x, (str, Path)): + # read in the file to a pandas DataFrame + x = Path(x).expanduser() + if not x.is_absolute(): + x = x.absolute() + try: + result = pd.read_csv(x) + except FileNotFoundError: + raise + elif isinstance(x, pd.DataFrame): + result = x + else: + raise TypeError(x + " must be path-like or DataFrame") + + return result + + agg_zone_mapping_df = _resolve_df(agg_zone_mapping) + agg_zone_mapping_df = agg_zone_mapping_df.sort_values('taz') + + zone_mapping = dict(zip(agg_zone_mapping_df['taz'], agg_zone_mapping_df['cluster_id'])) + agg_zones = sorted(agg_zone_mapping_df['cluster_id'].unique()) + + for mat_name in matrix_names: + if '.omx' not in mat_name: + mat_name = mat_name + ".omx" + + logger.info("Aggregating Matrix: " + mat_name + " ...") + + input_skim_file = Path(input_dir).expanduser().joinpath(mat_name) + output_skim_file = Path(output_dir).expanduser().joinpath(mat_name) + + assert os.path.isfile(input_skim_file) + + input_matrix = omx.open_file(input_skim_file, mode="r") + input_mapping_name = input_matrix.list_mappings()[0] + input_cores = input_matrix.list_matrices() + + output_matrix = omx.open_file(output_skim_file, mode="w") + + for core in input_cores: + matrix = input_matrix[core] + matrix_agg = _aggregate_matrix(matrix, zone_mapping) + output_matrix[core] = matrix_agg + + output_matrix.create_mapping(title=input_mapping_name, entries=agg_zones) + + input_matrix.close() + output_matrix.close() \ No newline at end of file From da991df5b09775aa150c8d613d5bff5855814f28 Mon Sep 17 00:00:00 2001 From: Ashish Kulshrestha Date: Thu, 27 Oct 2022 14:32:52 -0700 Subject: [PATCH 29/30] add gitignore to ignore test files --- notebooks/data-dl/.gitignore | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 notebooks/data-dl/.gitignore diff --git a/notebooks/data-dl/.gitignore b/notebooks/data-dl/.gitignore new file mode 100644 index 0000000..928939a --- /dev/null +++ b/notebooks/data-dl/.gitignore @@ -0,0 +1,5 @@ +*.csv +*.csv.gz +*.gpkg +*.omx +*.pq From 41e8daff6b114e093fe82ad3efe01fe57e314f5d Mon Sep 17 00:00:00 2001 From: David Ory Date: Wed, 9 Nov 2022 15:06:41 -0500 Subject: [PATCH 30/30] Remove `sandag_abm` files --- sandag_abm/src/main/emme/init_emme_project.py | 97 - .../src/main/emme/python_virtualenv.pth | 3 - sandag_abm/src/main/emme/solutions.mtbx | Bin 118784 -> 0 bytes .../main/emme/solutions_unconsolidated.mtbx | Bin 24576 -> 0 bytes .../assignment/build_transit_scenario.py | 679 --- .../toolbox/assignment/traffic_assignment.py | 1087 ---- .../toolbox/assignment/transit_assignment.py | 785 --- .../assignment/transit_select_analysis.py | 217 - .../src/main/emme/toolbox/build_toolbox.py | 411 -- .../diagnostic/mode_choice_diagnostic.py | 669 --- .../export/export_data_loader_matrices.py | 302 - .../export/export_data_loader_network.py | 1203 ---- .../export/export_for_commercial_vehicle.py | 158 - .../toolbox/export/export_for_transponder.py | 374 -- .../export/export_tap_adjacent_lines.py | 122 - .../toolbox/export/export_traffic_skims.py | 95 - .../toolbox/export/export_transit_skims.py | 108 - .../toolbox/import/adjust_network_links.py | 156 - .../emme/toolbox/import/import_auto_demand.py | 516 -- .../emme/toolbox/import/import_network.py | 2151 ------- .../emme/toolbox/import/import_seed_demand.py | 193 - .../toolbox/import/import_transit_demand.py | 230 - .../main/emme/toolbox/import/input_checker.py | 767 --- .../src/main/emme/toolbox/import/run4Ds.py | 412 -- .../toolbox/initialize/initialize_matrices.py | 426 -- .../initialize/initialize_transit_database.py | 168 - .../src/main/emme/toolbox/master_run.py | 1220 ---- .../model/commercial_vehicle/distribution.py | 197 - .../model/commercial_vehicle/generation.py | 187 - .../run_commercial_vehicle_model.py | 122 - .../model/commercial_vehicle/time_of_day.py | 99 - .../commercial_vehicle/toll_diversion.py | 119 - .../emme/toolbox/model/external_external.py | 190 - .../emme/toolbox/model/external_internal.py | 331 -- .../emme/toolbox/model/truck/distribution.py | 288 - .../emme/toolbox/model/truck/generation.py | 443 -- .../toolbox/model/truck/run_truck_model.py | 130 - .../src/main/emme/toolbox/utilities/demand.py | 297 - .../emme/toolbox/utilities/file_manager.py | 370 -- .../main/emme/toolbox/utilities/general.py | 386 -- .../main/emme/toolbox/utilities/omxwrapper.py | 91 - .../main/emme/toolbox/utilities/properties.py | 599 -- .../emme/toolbox/utilities/run_summary.py | 326 -- .../emme/toolbox/validation/validation.py | 290 - sandag_abm/src/main/gisdk/SandagCommon.rsc | 405 -- sandag_abm/src/main/gisdk/TC2OMX.rsc | 69 - sandag_abm/src/main/gisdk/TruckModel.rsc | 1504 ----- sandag_abm/src/main/gisdk/Utilities.rsc | 568 -- sandag_abm/src/main/gisdk/commVehDist.rsc | 106 - .../src/main/gisdk/commVehDiversion.rsc | 71 - sandag_abm/src/main/gisdk/commVehGen.rsc | 184 - sandag_abm/src/main/gisdk/commVehTOD.rsc | 122 - .../src/main/gisdk/create_LUZ_Skims.rsc | 254 - sandag_abm/src/main/gisdk/createhwynet.rsc | 2509 -------- sandag_abm/src/main/gisdk/createtodtables.rsc | 916 --- sandag_abm/src/main/gisdk/createtrnroutes.rsc | 655 --- sandag_abm/src/main/gisdk/dbox.rsc | 256 - sandag_abm/src/main/gisdk/exportTCData.rsc | 40 - .../src/main/gisdk/externalInternal.rsc | 356 -- sandag_abm/src/main/gisdk/gui_generic.rsc | 45 - sandag_abm/src/main/gisdk/hwyassign.rsc | 896 --- sandag_abm/src/main/gisdk/hwyskim.rsc | 1257 ---- .../main/gisdk/matrixPrecisionReduction.rsc | 51 - sandag_abm/src/main/gisdk/parameter.rsc | 8 - sandag_abm/src/main/gisdk/sandag_abm.lst | 22 - .../src/main/gisdk/sandag_abm_generic.lst | 23 - .../src/main/gisdk/sandag_abm_master.rsc | 411 -- .../src/main/gisdk/sandag_abm_outputs.rsc | 293 - sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc | 223 - sandag_abm/src/main/gisdk/sellink_volume.rsc | 85 - sandag_abm/src/main/gisdk/trnassign.rsc | 177 - sandag_abm/src/main/gisdk/trnskim.rsc | 1038 ---- .../accessibilities/AccessibilitiesDMU.java | 236 - .../accessibilities/AccessibilitiesTable.java | 407 -- .../AutoAndNonMotorizedSkimsCalculator.java | 1024 ---- .../abm/accessibilities/AutoSkimsDMU.java | 82 - .../AutoTazSkimsCalculator.java | 206 - .../BestTransitPathCalculator.java | 1548 ----- .../accessibilities/BuildAccessibilities.java | 1534 ----- .../accessibilities/DcUtilitiesTaskJppf.java | 880 --- .../DriveTransitWalkSkimsCalculator.java | 313 - .../MandatoryAccessibilitiesCalculator.java | 593 -- .../MandatoryAccessibilitiesDMU.java | 206 - .../accessibilities/McLogsumsAppender.java | 980 ---- ...andatoryDcEstimationMcLogsumsAppender.java | 368 -- ...ndatoryTodEstimationMcLogsumsAppender.java | 277 - .../accessibilities/NonTransitUtilities.java | 641 --- .../ParkLocationEstimationAppender.java | 297 - .../abm/accessibilities/SkimsAppender.java | 788 --- ...opLocationEstimationMcLogsumsAppender.java | 331 -- .../StopLocationSampleCalculator.java | 502 -- .../StoredTransitSkimData.java | 51 - .../accessibilities/StoredUtilityData.java | 133 - ...SubtourTodEstimationMcLogsumsAppender.java | 266 - .../abm/accessibilities/TransitPath.java | 45 - .../accessibilities/TripSkimsAppender.java | 569 -- .../VisitorTourLocationChoiceAppender.java | 367 -- .../WalkTransitDriveSkimsCalculator.java | 295 - .../WalkTransitWalkSkimsCalculator.java | 299 - .../accessibilities/XBorderSkimsAppender.java | 518 -- .../abm/active/AbstractNetworkFactory.java | 51 - ...ctPathChoiceEdgeAssignmentApplication.java | 217 - ...ractPathChoiceLogsumMatrixApplication.java | 422 -- .../active/AbstractShortestPathResultSet.java | 20 - .../active/BasicShortestPathResultSet.java | 59 - .../org/sandag/abm/active/BinarySearch.java | 35 - .../CompositeShortestPathResultSet.java | 57 - .../active/DestinationNotFoundException.java | 10 - .../main/java/org/sandag/abm/active/Edge.java | 9 - .../org/sandag/abm/active/EdgeEvaluator.java | 6 - .../abm/active/IntrazonalCalculation.java | 35 - .../abm/active/IntrazonalCalculations.java | 365 -- .../ModifiableShortestPathResultSet.java | 11 - .../java/org/sandag/abm/active/Network.java | 163 - .../org/sandag/abm/active/NetworkFactory.java | 35 - .../main/java/org/sandag/abm/active/Node.java | 7 - .../java/org/sandag/abm/active/NodePair.java | 50 - .../active/ParallelSingleSourceDijkstra.java | 230 - .../main/java/org/sandag/abm/active/Path.java | 82 - .../abm/active/PathAlternativeList.java | 177 - ...lternativeListGenerationConfiguration.java | 52 - .../abm/active/PathAlternativeListWriter.java | 59 - .../active/RepeatedSingleSourceDijkstra.java | 156 - .../sandag/abm/active/ShortestPathResult.java | 31 - .../abm/active/ShortestPathResultSet.java | 13 - .../abm/active/ShortestPathStrategy.java | 11 - .../org/sandag/abm/active/SimpleEdge.java | 50 - .../org/sandag/abm/active/SimpleNetwork.java | 141 - .../org/sandag/abm/active/SimpleNode.java | 45 - .../sandag/abm/active/SimpleTraversal.java | 46 - .../java/org/sandag/abm/active/Traversal.java | Bin 131 -> 0 bytes .../sandag/abm/active/TraversalEvaluator.java | 6 - .../sandag/BikeAssignmentTripReader.java | 216 - .../abm/active/sandag/PropertyParser.java | 99 - .../abm/active/sandag/SandagBikeEdge.java | 19 - ...lternativeListGenerationConfiguration.java | 36 - .../sandag/SandagBikeNetworkFactory.java | 581 -- .../abm/active/sandag/SandagBikeNode.java | 16 - ...lternativeListGenerationConfiguration.java | 430 -- .../sandag/SandagBikePathAlternatives.java | 207 - .../sandag/SandagBikePathChoiceDmu.java | 300 - ...kePathChoiceEdgeAssignmentApplication.java | 281 - ...BikePathChoiceLogsumMatrixApplication.java | 317 - .../sandag/SandagBikePathChoiceModel.java | 134 - ...lternativeListGenerationConfiguration.java | 35 - .../active/sandag/SandagBikeTraversal.java | 22 - ...lternativeListGenerationConfiguration.java | 62 - ...lternativeListGenerationConfiguration.java | 62 - ...lternativeListGenerationConfiguration.java | 310 - ...WalkPathChoiceLogsumMatrixApplication.java | 232 - ...lternativeListGenerationConfiguration.java | 62 - .../sandag/abm/active/sandag/TurnType.java | 35 - .../abm/airport/AirportDestChoiceModel.java | 581 -- .../sandag/abm/airport/AirportDmuFactory.java | 35 - .../abm/airport/AirportDmuFactoryIf.java | 11 - .../abm/airport/AirportModeChoiceModel.java | 378 -- .../org/sandag/abm/airport/AirportModel.java | 233 - .../sandag/abm/airport/AirportModelDMU.java | 478 -- .../abm/airport/AirportModelStructure.java | 239 - .../org/sandag/abm/airport/AirportParty.java | 321 -- .../abm/airport/AirportPartyManager.java | 410 -- .../sandag/abm/airport/AirportTripTables.java | 707 --- .../application/SandagAppendMcLogsumDMU.java | 615 -- .../SandagAtWorkSubtourFrequencyDMU.java | 64 - .../SandagAutoOwnershipChoiceDMU.java | 104 - ...dagCoordinatedDailyActivityPatternDMU.java | 176 - .../SandagCreateTripGenerationFiles.java | 1055 ---- .../application/SandagCtrampApplication.java | 23 - .../application/SandagCtrampDmuFactory.java | 170 - .../abm/application/SandagDcSoaDMU.java | 89 - .../abm/application/SandagDestChoiceDMU.java | 161 - .../SandagDestChoiceSoaTwoStageModelDMU.java | 158 - ...estChoiceSoaTwoStageTazDistUtilityDMU.java | 66 - .../SandagHouseholdDataManager.java | 624 -- .../SandagHouseholdDataManager2.java | 788 --- ...agIndividualMandatoryTourFrequencyDMU.java | 101 - ...ndividualNonMandatoryTourFrequencyDMU.java | 272 - .../SandagInternalExternalTripChoiceDMU.java | 50 - .../application/SandagJointTourModelsDMU.java | 137 - .../abm/application/SandagMGRAtoPNR.java | 287 - .../SandagMicromobilityChoiceDMU.java | 52 - .../abm/application/SandagModelStructure.java | 1299 ----- .../application/SandagParkingChoiceDMU.java | 94 - .../SandagParkingProvisionChoiceDMU.java | 83 - .../SandagSamplePopulationGenerator.java | 101 - .../application/SandagStopFrequencyDMU.java | 255 - .../application/SandagStopLocationDMU.java | 132 - .../abm/application/SandagSummitFile.java | 820 --- .../abm/application/SandagTelecommuteDMU.java | 67 - .../sandag/abm/application/SandagTestSOA.java | 216 - .../abm/application/SandagTourBasedModel.java | 358 -- ...SandagTourDepartureTimeAndDurationDMU.java | 345 -- .../application/SandagTourModeChoiceDMU.java | 502 -- .../SandagTransponderChoiceDMU.java | 56 - .../application/SandagTripModeChoiceDMU.java | 572 -- .../abm/application/SandagTripTables.java | 904 --- .../crossborder/CrossBorderDmuFactory.java | 51 - .../crossborder/CrossBorderDmuFactoryIf.java | 17 - .../abm/crossborder/CrossBorderModel.java | 525 -- .../CrossBorderModelStructure.java | 106 - .../CrossBorderStationDestChoiceDMU.java | 405 -- .../CrossBorderStationDestChoiceModel.java | 818 --- .../abm/crossborder/CrossBorderStop.java | 187 - .../CrossBorderStopFrequencyModel.java | 316 - .../CrossBorderStopLocationChoiceDMU.java | 406 -- .../CrossBorderStopLocationChoiceModel.java | 536 -- .../CrossBorderStopPurposeModel.java | 233 - .../CrossBorderStopTimeOfDayChoiceModel.java | 365 -- .../abm/crossborder/CrossBorderTour.java | 367 -- .../crossborder/CrossBorderTourManager.java | 358 -- .../CrossBorderTourModeChoiceDMU.java | 406 -- .../CrossBorderTourModeChoiceModel.java | 590 -- .../CrossBorderTourTimeOfDayChoiceModel.java | 188 - .../abm/crossborder/CrossBorderTrip.java | 502 -- .../CrossBorderTripModeChoiceDMU.java | 762 --- .../CrossBorderTripModeChoiceModel.java | 369 -- .../crossborder/CrossBorderTripTables.java | 704 --- .../abm/ctramp/AtWorkSubtourFrequencyDMU.java | 204 - .../abm/ctramp/AutoOwnershipChoiceDMU.java | 267 - .../org/sandag/abm/ctramp/BikeLogsum.java | 259 - .../sandag/abm/ctramp/BikeLogsumSegment.java | 117 - .../sandag/abm/ctramp/ConnectionHelper.java | 55 - .../CoordinatedDailyActivityPatternDMU.java | 442 -- .../sandag/abm/ctramp/CtrampApplication.java | 2481 -------- .../sandag/abm/ctramp/CtrampDmuFactoryIf.java | 52 - .../org/sandag/abm/ctramp/DAOException.java | 23 - .../java/org/sandag/abm/ctramp/DcSoaDMU.java | 215 - .../org/sandag/abm/ctramp/Definitions.java | 22 - .../org/sandag/abm/ctramp/DestChoiceDMU.java | 394 -- .../abm/ctramp/DestChoiceModelManager.java | 1082 ---- .../org/sandag/abm/ctramp/DestChoiceSize.java | 965 ---- .../abm/ctramp/DestChoiceTwoStageModel.java | 625 -- .../ctramp/DestChoiceTwoStageModelDMU.java | 408 -- ...iceTwoStageSoaProbabilitiesCalculator.java | 159 - ...hoiceTwoStageSoaTazDistanceUtilityDMU.java | 156 - .../DestinationSampleOfAlternativesModel.java | 612 -- .../java/org/sandag/abm/ctramp/Household.java | 1847 ------ .../HouseholdAtWorkSubtourFrequencyModel.java | 334 -- .../ctramp/HouseholdAutoOwnershipModel.java | 343 -- .../ctramp/HouseholdChoiceModelRunner.java | 254 - .../abm/ctramp/HouseholdChoiceModels.java | 980 ---- .../ctramp/HouseholdChoiceModelsManager.java | 682 --- .../ctramp/HouseholdChoiceModelsTaskJppf.java | 207 - ...dCoordinatedDailyActivityPatternModel.java | 1423 ----- .../abm/ctramp/HouseholdDataManager.java | 2039 ------- .../abm/ctramp/HouseholdDataManagerIf.java | 137 - .../abm/ctramp/HouseholdDataManagerRmi.java | 373 -- .../abm/ctramp/HouseholdDataWriter.java | 1871 ------ ...MandatoryTourDepartureAndDurationTime.java | 1672 ------ ...IndividualMandatoryTourFrequencyModel.java | 414 -- ...ividualNonMandatoryTourFrequencyModel.java | 823 --- .../sandag/abm/ctramp/HouseholdValidator.java | 82 - .../IndividualMandatoryTourFrequencyDMU.java | 301 - ...ndividualNonMandatoryTourFrequencyDMU.java | 773 --- .../ctramp/IntermediateStopChoiceModels.java | 5075 ----------------- .../ctramp/InternalExternalTripChoiceDMU.java | 113 - .../InternalExternalTripChoiceModel.java | 120 - .../sandag/abm/ctramp/JointTourModels.java | 916 --- .../sandag/abm/ctramp/JointTourModelsDMU.java | 310 - .../abm/ctramp/MandatoryDestChoiceModel.java | 885 --- .../sandag/abm/ctramp/MatrixDataServer.java | 209 - .../abm/ctramp/MatrixDataServerRmi.java | 98 - .../abm/ctramp/McLogsumsCalculator.java | 754 --- .../abm/ctramp/MicromobilityChoiceDMU.java | 119 - .../abm/ctramp/MicromobilityChoiceModel.java | 449 -- .../org/sandag/abm/ctramp/ModelStructure.java | 600 -- .../java/org/sandag/abm/ctramp/MyLogit.java | 115 - .../ctramp/NonMandatoryDestChoiceModel.java | 1527 ----- ...MandatoryTourDepartureAndDurationTime.java | 1448 ----- .../sandag/abm/ctramp/ParkingChoiceDMU.java | 332 -- .../abm/ctramp/ParkingProvisionChoiceDMU.java | 256 - .../abm/ctramp/ParkingProvisionModel.java | 216 - .../java/org/sandag/abm/ctramp/Person.java | 2032 ------- .../ctramp/SchoolEscortChauffeurResult.java | 67 - .../abm/ctramp/SchoolEscortChildResult.java | 68 - .../abm/ctramp/SchoolEscortingBundle.java | 499 -- .../sandag/abm/ctramp/SchoolEscortingDmu.java | 1716 ------ .../abm/ctramp/SchoolEscortingModel.java | 2154 ------- .../abm/ctramp/SchoolLocationChoiceModel.java | 603 -- .../ctramp/SchoolLocationChoiceTaskJppf.java | 238 - .../SchoolLocationChoiceTaskJppfNew.java | 252 - .../abm/ctramp/SegmentedSparseMatrix.java | 26 - .../java/org/sandag/abm/ctramp/SoaDMU.java | 10 - .../org/sandag/abm/ctramp/SqliteService.java | 118 - .../main/java/org/sandag/abm/ctramp/Stop.java | 304 - .../org/sandag/abm/ctramp/StopDCSoaDMU.java | 167 - .../ctramp/StopDepartArrivePeriodModel.java | 144 - .../sandag/abm/ctramp/StopDestChoiceSize.java | 214 - .../sandag/abm/ctramp/StopFrequencyDMU.java | 428 -- .../sandag/abm/ctramp/StopFrequencyModel.java | 811 --- .../sandag/abm/ctramp/StopLocationDMU.java | 411 -- .../SubtourDepartureAndDurationTime.java | 956 ---- .../abm/ctramp/SubtourDestChoiceModel.java | 1185 ---- .../ctramp/TNCAndTaxiWaitTimeCalculator.java | 264 - .../org/sandag/abm/ctramp/TazDataHandler.java | 870 --- .../sandag/abm/ctramp/TazDataHandlerRmi.java | 297 - .../java/org/sandag/abm/ctramp/TazDataIf.java | 152 - .../org/sandag/abm/ctramp/TelecommuteDMU.java | 172 - .../sandag/abm/ctramp/TelecommuteModel.java | 144 - .../ctramp/TimeCoefficientDistributions.java | 284 - .../java/org/sandag/abm/ctramp/TimeDMU.java | 105 - .../main/java/org/sandag/abm/ctramp/Tour.java | 875 --- .../TourDepartureTimeAndDurationDMU.java | 864 --- .../sandag/abm/ctramp/TourModeChoiceDMU.java | 523 -- .../abm/ctramp/TourModeChoiceModel.java | 762 --- .../ctramp/TourVehicleTypeChoiceModel.java | 189 - .../abm/ctramp/TransponderChoiceDMU.java | 158 - .../abm/ctramp/TransponderChoiceModel.java | 131 - .../sandag/abm/ctramp/TripModeChoiceDMU.java | 846 --- .../UsualWorkSchoolLocationChoiceModel.java | 1003 ---- .../main/java/org/sandag/abm/ctramp/Util.java | 379 -- .../java/org/sandag/abm/ctramp/UtilRmi.java | 137 - .../abm/ctramp/WorkLocationChoiceModel.java | 692 --- .../ctramp/WorkLocationChoiceTaskJppf.java | 246 - .../ctramp/WorkLocationChoiceTaskJppfNew.java | 253 - .../dta/postprocessing/PostprocessModel.java | 247 - .../postprocessing/broadTODProcessing.java | 224 - .../postprocessing/detailedTODProcessing.java | 563 -- .../abm/dta/postprocessing/dtaTrip.java | 386 -- .../spatialDisaggregationModel.java | 144 - .../todDisaggregationModel.java | 118 - .../InternalExternalDmuFactory.java | 40 - .../InternalExternalDmuFactoryIf.java | 13 - .../InternalExternalModel.java | 309 - .../InternalExternalModelStructure.java | 95 - .../InternalExternalTour.java | 307 - .../InternalExternalTourDestChoiceDMU.java | 114 - .../InternalExternalTourDestChoiceModel.java | 172 - .../InternalExternalTourManager.java | 377 -- ...ernalExternalTourTimeOfDayChoiceModel.java | 182 - .../InternalExternalTrip.java | 313 - .../InternalExternalTripModeChoiceDMU.java | 551 -- .../InternalExternalTripModeChoiceModel.java | 277 - .../InternalExternalTripTables.java | 698 --- .../maas/HouseholdAVAllocationManager.java | 1496 ----- .../abm/maas/HouseholdAVAllocationModel.java | 671 --- ...holdAVAllocationModelParkingChoiceDMU.java | 239 - .../HouseholdAVAllocationModelRunner.java | 260 - ...seholdAVAllocationModelTripUtilityDMU.java | 101 - ...holdAVAllocationModelVehicleChoiceDMU.java | 212 - .../java/org/sandag/abm/maas/PersonTrip.java | 277 - .../sandag/abm/maas/PersonTripManager.java | 1087 ---- .../org/sandag/abm/maas/TNCFleetModel.java | 448 -- .../java/org/sandag/abm/maas/TNCVehicle.java | 156 - .../sandag/abm/maas/TNCVehicleManager.java | 878 --- .../org/sandag/abm/maas/TNCVehicleTrip.java | 428 -- .../sandag/abm/maas/TransportCostManager.java | 438 -- .../org/sandag/abm/modechoice/AutoDMU.java | 136 - .../org/sandag/abm/modechoice/AutoUEC.java | 138 - .../org/sandag/abm/modechoice/Constants.java | 56 - .../org/sandag/abm/modechoice/MaasDMU.java | 128 - .../org/sandag/abm/modechoice/MaasUEC.java | 141 - .../abm/modechoice/MgraDataManager.java | 1551 ----- .../java/org/sandag/abm/modechoice/Modes.java | 115 - .../sandag/abm/modechoice/NonMotorDMU.java | 146 - .../sandag/abm/modechoice/NonMotorUEC.java | 189 - .../sandag/abm/modechoice/TapDataManager.java | 321 -- .../sandag/abm/modechoice/TazDataManager.java | 924 --- .../abm/modechoice/TransitDriveAccessDMU.java | 465 -- .../abm/modechoice/TransitWalkAccessDMU.java | 332 -- .../abm/reporting/AbstractCsvExporter.java | 32 - .../org/sandag/abm/reporting/CVMExporter.java | 304 - .../org/sandag/abm/reporting/CVMScaler.java | 179 - .../java/org/sandag/abm/reporting/CsvRow.java | 22 - .../sandag/abm/reporting/CsvWriterThread.java | 86 - .../sandag/abm/reporting/DataExporter.java | 2688 --------- .../abm/reporting/DoubleFormatUtil.java | 484 -- .../org/sandag/abm/reporting/IExporter.java | 11 - .../org/sandag/abm/reporting/IMatrixDao.java | 8 - .../sandag/abm/reporting/OMXMatrixDao.java | 36 - .../org/sandag/abm/reporting/SkimBuilder.java | 721 --- .../abm/reporting/TranscadMatrixDao.java | 28 - .../abm/reporting/TransitTimeReporter.java | 449 -- .../abm/reporting/TruckCsvExporter.java | 58 - .../reporting/TruckCsvPublisherThread.java | 67 - .../abm/reporting/TruckOmxExporter.java | 56 - .../emfac2011/AquavisDataBuilder.java | 47 - .../emfac2011/Emfac2011AquavisIntrazonal.java | 45 - .../reporting/emfac2011/Emfac2011Data.java | 302 - .../emfac2011/Emfac2011Definitions.java | 86 - .../emfac2011/Emfac2011InputFileCreator.java | 577 -- .../emfac2011/Emfac2011Properties.java | 101 - .../reporting/emfac2011/Emfac2011Runner.java | 216 - .../emfac2011/Emfac2011SpeedCategory.java | 73 - .../reporting/emfac2011/Emfac2011SqlUtil.java | 145 - .../emfac2011/Emfac2011VehicleType.java | 146 - .../reporting/emfac2011/RunEmfacDialog.java | 174 - .../reporting/emfac2011/SandagAutoModes.java | 11 - .../specialevent/SpecialEventDmuFactory.java | 42 - .../SpecialEventDmuFactoryIf.java | 13 - .../abm/specialevent/SpecialEventModel.java | 347 -- .../SpecialEventOriginChoiceDMU.java | 205 - .../SpecialEventOriginChoiceModel.java | 384 -- .../abm/specialevent/SpecialEventTour.java | 291 - .../specialevent/SpecialEventTourManager.java | 342 -- .../abm/specialevent/SpecialEventTrip.java | 292 - .../SpecialEventTripModeChoiceDMU.java | 451 -- .../SpecialEventTripModeChoiceModel.java | 284 - .../specialevent/SpecialEventTripTables.java | 595 -- .../org/sandag/abm/survey/OutputTapPairs.java | 363 -- .../sandag/abm/utilities/CreateLogsums.java | 410 -- .../sandag/abm/utilities/ErrorLogging.java | 41 - .../abm/utilities/ModelOutputReader.java | 836 --- .../sandag/abm/utilities/RunModeChoice.java | 402 -- .../abm/utilities/TapAtConsistencyCheck.java | 183 - .../abm/validation/MainApplication.java | 437 -- .../sandag/abm/visitor/VisitorDmuFactory.java | 55 - .../abm/visitor/VisitorDmuFactoryIf.java | 18 - .../VisitorMicromobilityChoiceDMU.java | 139 - .../VisitorMicromobilityChoiceModel.java | 325 -- .../org/sandag/abm/visitor/VisitorModel.java | 499 -- .../abm/visitor/VisitorModelStructure.java | 95 - .../org/sandag/abm/visitor/VisitorStop.java | 176 - .../visitor/VisitorStopFrequencyModel.java | 325 -- .../visitor/VisitorStopLocationChoiceDMU.java | 406 -- .../VisitorStopLocationChoiceModel.java | 538 -- .../abm/visitor/VisitorStopPurposeModel.java | 233 - .../VisitorStopTimeOfDayChoiceModel.java | 365 -- .../org/sandag/abm/visitor/VisitorTour.java | 350 -- .../abm/visitor/VisitorTourDestChoiceDMU.java | 317 - .../visitor/VisitorTourDestChoiceModel.java | 586 -- .../visitor/VisitorTourEstimationFile.java | 382 -- .../abm/visitor/VisitorTourManager.java | 531 -- .../abm/visitor/VisitorTourModeChoiceDMU.java | 570 -- .../visitor/VisitorTourModeChoiceModel.java | 399 -- .../VisitorTourTimeOfDayChoiceModel.java | 192 - .../org/sandag/abm/visitor/VisitorTrip.java | 518 -- .../abm/visitor/VisitorTripModeChoiceDMU.java | 883 --- .../visitor/VisitorTripModeChoiceModel.java | 327 -- .../sandag/abm/visitor/VisitorTripTables.java | 682 --- .../AlternativeUsesMatrices.java | 42 - .../ChangingTravelAttributeGetter.java | 52 - .../cvm/activityTravel/CodedAlternative.java | 35 - .../CoefficientFormatError.java | 61 - .../cvm/activityTravel/DurationModel.java | 91 - .../activityTravel/HouseholdInterface.java | 46 - .../LoggingStopAlternative.java | 79 - .../cvm/activityTravel/ModelUsesMatrices.java | 39 - .../activityTravel/ModelWithCoefficients.java | 54 - .../cvm/activityTravel/PersonInterface.java | 54 - .../RealNumberDistribution.java | 37 - .../org/sandag/cvm/activityTravel/Stop.java | 42 - .../cvm/activityTravel/StopAlternative.java | 106 - .../sandag/cvm/activityTravel/StopChoice.java | 230 - .../org/sandag/cvm/activityTravel/Tour.java | 290 - .../cvm/activityTravel/TourInterface.java | 61 - .../TourNextStopPurposeChoice.java | 45 - .../sandag/cvm/activityTravel/TourType.java | 115 - .../cvm/activityTravel/TravelTimeTracker.java | 117 - .../sandag/cvm/activityTravel/TripMode.java | 74 - .../cvm/activityTravel/TripModeChoice.java | 96 - .../activityTravel/VehicleTourTypeChoice.java | 51 - .../activityTravel/ZonePairDisutility.java | 7 - .../cvm/AlogitLogitModelNest.java | 21 - .../cvm/CommercialNextStopChoice.java | 64 - .../cvm/CommercialNextStopPurposeChoice.java | 249 - .../activityTravel/cvm/CommercialTour.java | 328 -- .../cvm/CommercialTravelTimeTracker.java | 77 - .../cvm/CommercialTripMode.java | 256 - .../cvm/CommercialVehicleTourType.java | 112 - .../cvm/CommercialVehicleTourTypeChoice.java | 238 - .../cvm/CommercialVehicleTripModeChoice.java | 216 - .../activityTravel/cvm/DurationModel2.java | 49 - .../cvm/GenerateCommercialTours.java | 855 --- .../cvm/TourStartTimeModel.java | 115 - .../cvm/VehicleTourTypeNest.java | 16 - .../calgary/weekend/GenerateWeekendTours.java | 292 - .../weekend/NextWeekendTourStartTime.java | 101 - .../cvm/calgary/weekend/TourInTimeBand.java | 185 - .../cvm/calgary/weekend/WeekendHousehold.java | 366 -- .../cvm/calgary/weekend/WeekendPerson.java | 105 - .../calgary/weekend/WeekendStopChoice.java | 54 - .../weekend/WeekendStopPurposeChoice.java | 217 - .../cvm/calgary/weekend/WeekendTour.java | 345 -- .../cvm/calgary/weekend/WeekendTourType.java | 73 - .../weekend/WeekendTourTypeChoice.java | 130 - .../weekend/WeekendTravelTimeTracker.java | 72 - .../cvm/common/datafile/BaseDataFile.java | 446 -- .../cvm/common/datafile/BinaryFileReader.java | 114 - .../cvm/common/datafile/BinaryFileWriter.java | 106 - .../cvm/common/datafile/CSVFileReader.java | 810 --- .../cvm/common/datafile/CSVFileWriter.java | 248 - .../cvm/common/datafile/D211FileReader.java | 304 - .../cvm/common/datafile/D231FileReader.java | 129 - .../cvm/common/datafile/DBFFileReader.java | 225 - .../cvm/common/datafile/DBFFileWriter.java | 200 - .../sandag/cvm/common/datafile/DataFile.java | 232 - .../cvm/common/datafile/DataHeader.java | 114 - .../cvm/common/datafile/DataReader.java | 94 - .../sandag/cvm/common/datafile/DataTypes.java | 38 - .../cvm/common/datafile/DataWriter.java | 105 - .../datafile/DbByteArrayOutputStream.java | 47 - .../cvm/common/datafile/DiskObjectArray.java | 227 - .../cvm/common/datafile/ExcelFileReader.java | 692 --- .../sandag/cvm/common/datafile/FileType.java | 55 - .../datafile/FixedFormatTextFileReader.java | 315 - .../common/datafile/GeneralDecimalFormat.java | 53 - .../cvm/common/datafile/JDBCTableReader.java | 221 - .../cvm/common/datafile/JDBCTableWriter.java | 136 - .../datafile/MissingValueException.java | 37 - .../datafile/MultipleValueException.java | 39 - .../common/datafile/NEW_CSVFileReader.java | 647 --- .../common/datafile/OLD_CSVFileReader.java | 702 --- .../cvm/common/datafile/ReportGenerator.java | 28 - .../common/datafile/TableDataFileReader.java | 106 - .../common/datafile/TableDataFileWriter.java | 105 - .../cvm/common/datafile/TableDataReader.java | 67 - .../cvm/common/datafile/TableDataSet.java | 1927 ------- .../datafile/TableDataSetCacheCollection.java | 218 - .../datafile/TableDataSetCollection.java | 200 - .../datafile/TableDataSetCrosstabber.java | 407 -- .../common/datafile/TableDataSetIndex.java | 225 - .../datafile/TableDataSetIndexedValue.java | 601 -- .../common/datafile/TableDataSetLoader.java | 59 - .../datafile/TableDataSetSoftReference.java | 40 - .../cvm/common/datafile/TableDataWriter.java | 67 - .../sandag/cvm/common/datafile/TextFile.java | 240 - .../common/datafile/tests/BinaryFileTest.java | 105 - .../datafile/tests/CSVFileReaderTest.java | 83 - .../datafile/tests/CSVFileWriterTest.java | 64 - .../common/datafile/tests/DataFileTest.java | 85 - .../datafile/tests/DiskObjectArrayTest.java | 193 - .../datafile/tests/JDBCTableReaderTest.java | 87 - .../datafile/tests/TableDataSetTest.java | 322 -- .../common/discreteEvent/EventDispatcher.java | 33 - .../cvm/common/discreteEvent/EventQueue.java | 34 - .../discreteEvent/TestEventDispatcher.java | 65 - .../cvm/common/discreteEvent/TimedEvent.java | 57 - .../emme2/Emme2MatrixHashtableReader.java | 65 - .../common/emme2/IndexConditionFunction.java | 124 - .../cvm/common/emme2/IndexLinearFunction.java | 147 - .../common/emme2/MatrixAndTAZTableCache.java | 83 - .../cvm/common/emme2/MatrixCacheReader.java | 45 - .../emme2/TestEmme2311MatrixReader.java | 29 - .../common/model/AggregateAlternative.java | 26 - .../sandag/cvm/common/model/Alternative.java | 26 - .../model/ChoiceModelOverflowException.java | 20 - .../cvm/common/model/DiscreteChoiceModel.java | 63 - .../model/DiscreteChoiceModelInterface.java | 47 - .../common/model/FixedUtilityAlternative.java | 37 - .../cvm/common/model/GumbelErrorTerm.java | 77 - .../model/LinearInParametersFunction.java | 72 - .../sandag/cvm/common/model/LogitModel.java | 261 - .../common/model/NoAlternativeAvailable.java | 28 - ...icalDerivativeSingleParameterFunction.java | 34 - .../cvm/common/model/RandomVariable.java | 35 - .../common/model/SingleParameterFunction.java | 28 - .../cvm/common/model/TestLogitModel.java | 63 - .../model/UtilityMaximizingChoiceModel.java | 64 - ...imizingChoiceModelWithErrorTermVector.java | 84 - .../cvm/common/skims/HDF5MatrixReader.java | 395 -- .../cvm/common/skims/HDF5MatrixWriter.java | 281 - .../skims/LinearFunctionOfSomeSkims.java | 109 - .../org/sandag/cvm/common/skims/Location.java | 8 - .../skims/OMXMatrixCollectionReader.java | 67 - .../sandag/cvm/common/skims/SomeSkims.java | 437 -- .../skims/TranscadMatrixCollectionReader.java | 67 - .../skims/TravelAttributesInterface.java | 23 - .../TravelUtilityCalculatorInterface.java | 28 - .../patternDetail/DestinationRandomTerms.java | 121 - .../TestDestinationRandomTerms.java | 273 - .../htm/applications/SandagCountyModel.java | 352 -- .../org/sandag/htm/applications/ohio.java | 118 - .../htm/applications/sandagZonalModel.java | 338 -- .../sandag/htm/applications/sandag_tm.java | 45 - .../sandag/htm/applications/utilities.java | 217 - .../org/sandag/htm/applications/yumaMPO.java | 180 - .../org/sandag/htm/processFAF/ModesFAF.java | 19 - .../org/sandag/htm/processFAF/ReadFAF4.java | 522 -- .../htm/processFAF/commodityClassType.java | 13 - .../htm/processFAF/convertTonsToTrucks.java | 113 - .../htm/processFAF/countyTruckModel.java | 597 -- .../disaggregateAndAggregateFlows.java | 67 - .../htm/processFAF/disaggregateFlows.java | 1357 ----- .../org/sandag/htm/processFAF/fafUtils.java | 331 -- .../org/sandag/htm/processFAF/readFAF2.java | 547 -- .../org/sandag/htm/processFAF/readFAF3.java | 520 -- .../sandag/htm/processFAF/reportFormat.java | 23 - .../src/main/python/assignScenarioID.py | 27 - .../main/python/calculate_micromobility.py | 273 - sandag_abm/src/main/python/checkFreeSpace.py | 33 - sandag_abm/src/main/python/check_output.py | 139 - sandag_abm/src/main/python/cvm_analysis.zip | Bin 29456 -> 0 bytes .../src/main/python/cvm_input_create.py | 462 -- .../main/python/dataExporter/abmScenario.py | 3914 ------------- .../main/python/dataExporter/environment.yml | 92 - .../python/dataExporter/hwyShapeExport.py | 523 -- .../src/main/python/dataExporter/serialRun.py | 168 - .../main/python/dataExporter/skimAppender.py | 1895 ------ .../src/main/python/database_summary.py | 190 - sandag_abm/src/main/python/excel_update.py | 85 - sandag_abm/src/main/python/parameterUpdate.py | 60 - .../pythonGUI/createStudyAndScenario.py | 237 - .../main/python/pythonGUI/parameterEditor.py | 253 - .../src/main/python/pythonGUI/popupMsg.py | 17 - sandag_abm/src/main/python/pythonGUI/setup.py | 9 - .../src/main/python/pythonGUI/stringFinder.py | 21 - .../src/main/python/pythonGUI/validatorGUI.py | 122 - .../src/main/python/remote_run_traffic.py | 123 - sandag_abm/src/main/python/sdcvm.py | 840 --- sandag_abm/src/main/python/sdcvm_settings.py | 43 - sandag_abm/src/main/python/sdcvm_settings.pyc | Bin 1332 -> 0 bytes sandag_abm/src/main/python/sdcvm_summarize.py | 99 - sandag_abm/src/main/python/serverswap.py | 92 - .../src/main/r/INRIX_OutlierAnalysis_Final.R | 532 -- .../src/main/r/RegressionAnalysis_Final.R | 953 ---- sandag_abm/src/main/r/summarize_SR125data.R | 141 - sandag_abm/src/main/r/utilfunc.R | 256 - sandag_abm/src/main/r/visualizer/Master.R | 94 - .../src/main/r/visualizer/SummarizeABM2016.R | 2448 -------- .../src/main/r/visualizer/_SYSTEM_VARIABLES.R | 100 - .../src/main/r/visualizer/workersByMAZ.R | 114 - .../src/main/resources/BatchSubstitute.bat | 20 - sandag_abm/src/main/resources/CTRampEnv.bat | 61 - sandag_abm/src/main/resources/CheckOutput.bat | 8 - .../main/resources/CreateD2TAccessFile.bat | 9 - .../src/main/resources/DataExporter.bat | 33 - .../src/main/resources/DataLoadRequest.bat | 7 - sandag_abm/src/main/resources/DataSummary.bat | 14 - sandag_abm/src/main/resources/ExcelUpdate.bat | 16 - .../src/main/resources/FHWADataExporter.bat | 142 - .../main/resources/GnuWin32/bin/libiconv2.dll | Bin 898048 -> 0 bytes .../main/resources/GnuWin32/bin/libintl3.dll | Bin 101888 -> 0 bytes .../src/main/resources/GnuWin32/bin/tee.exe | Bin 24576 -> 0 bytes sandag_abm/src/main/resources/HPPowerOff.bat | 1 - sandag_abm/src/main/resources/HPPowerOn.bat | 1 - .../src/main/resources/RunEMFAC2011.cmd | 12 - .../src/main/resources/RunEMFAC2014.cmd | 16 - sandag_abm/src/main/resources/RunViz.bat | 126 - .../src/main/resources/StartHHAndNodes.cmd | 44 - .../src/main/resources/assignScenarioID.cmd | 6 - .../checkAtTransitNetworkConsistency.cmd | 44 - .../src/main/resources/checkFreeSpaceOnC.bat | 2 - .../resources/copy_networkfiles_to_study.cmd | 18 - .../src/main/resources/copy_networks.cmd | 17 - .../src/main/resources/create_scenario.cmd | 107 - sandag_abm/src/main/resources/cvm.bat | 30 - sandag_abm/src/main/resources/cvm.properties | 39 - sandag_abm/src/main/resources/emme_python.bat | 31 - sandag_abm/src/main/resources/init_emme.cmd | 37 - sandag_abm/src/main/resources/jhdf.dll | Bin 843264 -> 0 bytes sandag_abm/src/main/resources/jhdf5.dll | Bin 2284544 -> 0 bytes .../src/main/resources/jppf-client.properties | 141 - .../jppf-clientDistributed.properties | 37 - .../src/main/resources/jppf-driver.properties | 297 - .../main/resources/jppf-sandag01.properties | 298 - .../main/resources/jppf-sandag02.properties | 298 - .../main/resources/jppf-sandag03.properties | 298 - .../main/resources/jppf-sandag04.properties | 298 - .../main/resources/log4j-client.properties | 39 - .../main/resources/log4j-driver.properties | 42 - .../main/resources/log4j-sandag01.properties | 46 - .../src/main/resources/log4j-sandag01.xml | 439 -- .../main/resources/log4j-sandag02.properties | 46 - .../src/main/resources/log4j-sandag02.xml | 439 -- .../main/resources/log4j-sandag03.properties | 46 - .../src/main/resources/log4j-sandag03.xml | 439 -- .../main/resources/log4j-sandag04.properties | 46 - .../src/main/resources/log4j-sandag04.xml | 439 -- sandag_abm/src/main/resources/log4j.xml | 443 -- .../main/resources/log4j_AtTransitCheck.xml | 36 - sandag_abm/src/main/resources/log4j_d2t.xml | 36 - sandag_abm/src/main/resources/log4j_hh.xml | 36 - sandag_abm/src/main/resources/log4j_mtx.xml | 36 - sandag_abm/src/main/resources/log4j_test.xml | 455 -- sandag_abm/src/main/resources/mapAndRun.bat | 33 - sandag_abm/src/main/resources/pskill.exe | Bin 621944 -> 0 bytes sandag_abm/src/main/resources/runDriver.cmd | 17 - sandag_abm/src/main/resources/runHhMgr.cmd | 54 - .../src/main/resources/runHhMgr_log.bat | 18 - sandag_abm/src/main/resources/runMtxMgr.cmd | 43 - .../src/main/resources/runMtxMgr_log.bat | 17 - sandag_abm/src/main/resources/runSandag01.cmd | 16 - .../src/main/resources/runSandag01_log.bat | 13 - sandag_abm/src/main/resources/runSandag02.cmd | 15 - sandag_abm/src/main/resources/runSandag03.cmd | 15 - sandag_abm/src/main/resources/runSandag04.cmd | 15 - .../src/main/resources/runSandagAbm_MAAS.cmd | 59 - .../resources/runSandagAbm_MCDiagnostic.cmd | 55 - .../src/main/resources/runSandagAbm_SDRM.cmd | 68 - .../src/main/resources/runSandagAbm_SEM.cmd | 57 - .../src/main/resources/runSandagAbm_SMM.cmd | 86 - .../main/resources/runSandagBikeLogsums.cmd | 45 - .../resources/runSandagBikeRouteChoice.cmd | 38 - .../main/resources/runSandagWalkLogsums.cmd | 45 - .../src/main/resources/runTransitReporter.cmd | 48 - .../src/main/resources/sandag_abm.properties | 1357 ----- sandag_abm/src/main/resources/serverswap.bat | 9 - .../src/main/resources/serverswap_files.csv | 25 - sandag_abm/src/main/resources/setup.bat | 1 - sandag_abm/src/main/resources/stopABM.cmd | 5 - sandag_abm/src/main/resources/taskkill.bat | 2 - .../resources/updateYearSpecificProps.bat | 6 - sandag_abm/src/main/resources/w9xpopen.exe | Bin 49664 -> 0 bytes 694 files changed, 217477 deletions(-) delete mode 100644 sandag_abm/src/main/emme/init_emme_project.py delete mode 100644 sandag_abm/src/main/emme/python_virtualenv.pth delete mode 100644 sandag_abm/src/main/emme/solutions.mtbx delete mode 100644 sandag_abm/src/main/emme/solutions_unconsolidated.mtbx delete mode 100644 sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py delete mode 100644 sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py delete mode 100644 sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py delete mode 100644 sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py delete mode 100644 sandag_abm/src/main/emme/toolbox/build_toolbox.py delete mode 100644 sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py delete mode 100644 sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/import_network.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/input_checker.py delete mode 100644 sandag_abm/src/main/emme/toolbox/import/run4Ds.py delete mode 100644 sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py delete mode 100644 sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py delete mode 100644 sandag_abm/src/main/emme/toolbox/master_run.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/external_external.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/external_internal.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/distribution.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/generation.py delete mode 100644 sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/demand.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/file_manager.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/general.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/properties.py delete mode 100644 sandag_abm/src/main/emme/toolbox/utilities/run_summary.py delete mode 100644 sandag_abm/src/main/emme/toolbox/validation/validation.py delete mode 100644 sandag_abm/src/main/gisdk/SandagCommon.rsc delete mode 100644 sandag_abm/src/main/gisdk/TC2OMX.rsc delete mode 100644 sandag_abm/src/main/gisdk/TruckModel.rsc delete mode 100644 sandag_abm/src/main/gisdk/Utilities.rsc delete mode 100644 sandag_abm/src/main/gisdk/commVehDist.rsc delete mode 100644 sandag_abm/src/main/gisdk/commVehDiversion.rsc delete mode 100644 sandag_abm/src/main/gisdk/commVehGen.rsc delete mode 100644 sandag_abm/src/main/gisdk/commVehTOD.rsc delete mode 100644 sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc delete mode 100644 sandag_abm/src/main/gisdk/createhwynet.rsc delete mode 100644 sandag_abm/src/main/gisdk/createtodtables.rsc delete mode 100644 sandag_abm/src/main/gisdk/createtrnroutes.rsc delete mode 100644 sandag_abm/src/main/gisdk/dbox.rsc delete mode 100644 sandag_abm/src/main/gisdk/exportTCData.rsc delete mode 100644 sandag_abm/src/main/gisdk/externalInternal.rsc delete mode 100644 sandag_abm/src/main/gisdk/gui_generic.rsc delete mode 100644 sandag_abm/src/main/gisdk/hwyassign.rsc delete mode 100644 sandag_abm/src/main/gisdk/hwyskim.rsc delete mode 100644 sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc delete mode 100644 sandag_abm/src/main/gisdk/parameter.rsc delete mode 100644 sandag_abm/src/main/gisdk/sandag_abm.lst delete mode 100644 sandag_abm/src/main/gisdk/sandag_abm_generic.lst delete mode 100644 sandag_abm/src/main/gisdk/sandag_abm_master.rsc delete mode 100644 sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc delete mode 100644 sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc delete mode 100644 sandag_abm/src/main/gisdk/sellink_volume.rsc delete mode 100644 sandag_abm/src/main/gisdk/trnassign.rsc delete mode 100644 sandag_abm/src/main/gisdk/trnskim.rsc delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/SubtourTodEstimationMcLogsumsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Edge.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Network.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Node.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Path.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceLogsumMatrixApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppfNew.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SegmentedSparseMatrix.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandler.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/Util.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppfNew.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/PostprocessModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/detailedTODProcessing.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/todDisaggregationModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ModelUsesMatrices.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ModelWithCoefficients.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/PersonInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/RealNumberDistribution.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Stop.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTripModeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTourType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTourTypeChoice.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTravelTimeTracker.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BaseDataFile.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataFile.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/FileType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/FixedFormatTextFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/MissingValueException.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/MultipleValueException.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/NEW_CSVFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/OLD_CSVFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ReportGenerator.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataFileReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataFileWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSet.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetIndex.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetIndexedValue.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetLoader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetSoftReference.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TextFile.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TimedEvent.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexLinearFunction.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/MatrixAndTAZTableCache.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/MatrixCacheReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/emme2/TestEmme2311MatrixReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/LogitModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixWriter.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/LinearFunctionOfSomeSkims.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/Location.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/OMXMatrixCollectionReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java delete mode 100644 sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/TestDestinationRandomTerms.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/SandagCountyModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java delete mode 100644 sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java delete mode 100644 sandag_abm/src/main/python/assignScenarioID.py delete mode 100644 sandag_abm/src/main/python/calculate_micromobility.py delete mode 100644 sandag_abm/src/main/python/checkFreeSpace.py delete mode 100644 sandag_abm/src/main/python/check_output.py delete mode 100644 sandag_abm/src/main/python/cvm_analysis.zip delete mode 100644 sandag_abm/src/main/python/cvm_input_create.py delete mode 100644 sandag_abm/src/main/python/dataExporter/abmScenario.py delete mode 100644 sandag_abm/src/main/python/dataExporter/environment.yml delete mode 100644 sandag_abm/src/main/python/dataExporter/hwyShapeExport.py delete mode 100644 sandag_abm/src/main/python/dataExporter/serialRun.py delete mode 100644 sandag_abm/src/main/python/dataExporter/skimAppender.py delete mode 100644 sandag_abm/src/main/python/database_summary.py delete mode 100644 sandag_abm/src/main/python/excel_update.py delete mode 100644 sandag_abm/src/main/python/parameterUpdate.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/parameterEditor.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/popupMsg.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/setup.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/stringFinder.py delete mode 100644 sandag_abm/src/main/python/pythonGUI/validatorGUI.py delete mode 100644 sandag_abm/src/main/python/remote_run_traffic.py delete mode 100644 sandag_abm/src/main/python/sdcvm.py delete mode 100644 sandag_abm/src/main/python/sdcvm_settings.py delete mode 100644 sandag_abm/src/main/python/sdcvm_settings.pyc delete mode 100644 sandag_abm/src/main/python/sdcvm_summarize.py delete mode 100644 sandag_abm/src/main/python/serverswap.py delete mode 100644 sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R delete mode 100644 sandag_abm/src/main/r/RegressionAnalysis_Final.R delete mode 100644 sandag_abm/src/main/r/summarize_SR125data.R delete mode 100644 sandag_abm/src/main/r/utilfunc.R delete mode 100644 sandag_abm/src/main/r/visualizer/Master.R delete mode 100644 sandag_abm/src/main/r/visualizer/SummarizeABM2016.R delete mode 100644 sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R delete mode 100644 sandag_abm/src/main/r/visualizer/workersByMAZ.R delete mode 100644 sandag_abm/src/main/resources/BatchSubstitute.bat delete mode 100644 sandag_abm/src/main/resources/CTRampEnv.bat delete mode 100644 sandag_abm/src/main/resources/CheckOutput.bat delete mode 100644 sandag_abm/src/main/resources/CreateD2TAccessFile.bat delete mode 100644 sandag_abm/src/main/resources/DataExporter.bat delete mode 100644 sandag_abm/src/main/resources/DataLoadRequest.bat delete mode 100644 sandag_abm/src/main/resources/DataSummary.bat delete mode 100644 sandag_abm/src/main/resources/ExcelUpdate.bat delete mode 100644 sandag_abm/src/main/resources/FHWADataExporter.bat delete mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll delete mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll delete mode 100644 sandag_abm/src/main/resources/GnuWin32/bin/tee.exe delete mode 100644 sandag_abm/src/main/resources/HPPowerOff.bat delete mode 100644 sandag_abm/src/main/resources/HPPowerOn.bat delete mode 100644 sandag_abm/src/main/resources/RunEMFAC2011.cmd delete mode 100644 sandag_abm/src/main/resources/RunEMFAC2014.cmd delete mode 100644 sandag_abm/src/main/resources/RunViz.bat delete mode 100644 sandag_abm/src/main/resources/StartHHAndNodes.cmd delete mode 100644 sandag_abm/src/main/resources/assignScenarioID.cmd delete mode 100644 sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd delete mode 100644 sandag_abm/src/main/resources/checkFreeSpaceOnC.bat delete mode 100644 sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd delete mode 100644 sandag_abm/src/main/resources/copy_networks.cmd delete mode 100644 sandag_abm/src/main/resources/create_scenario.cmd delete mode 100644 sandag_abm/src/main/resources/cvm.bat delete mode 100644 sandag_abm/src/main/resources/cvm.properties delete mode 100644 sandag_abm/src/main/resources/emme_python.bat delete mode 100644 sandag_abm/src/main/resources/init_emme.cmd delete mode 100644 sandag_abm/src/main/resources/jhdf.dll delete mode 100644 sandag_abm/src/main/resources/jhdf5.dll delete mode 100644 sandag_abm/src/main/resources/jppf-client.properties delete mode 100644 sandag_abm/src/main/resources/jppf-clientDistributed.properties delete mode 100644 sandag_abm/src/main/resources/jppf-driver.properties delete mode 100644 sandag_abm/src/main/resources/jppf-sandag01.properties delete mode 100644 sandag_abm/src/main/resources/jppf-sandag02.properties delete mode 100644 sandag_abm/src/main/resources/jppf-sandag03.properties delete mode 100644 sandag_abm/src/main/resources/jppf-sandag04.properties delete mode 100644 sandag_abm/src/main/resources/log4j-client.properties delete mode 100644 sandag_abm/src/main/resources/log4j-driver.properties delete mode 100644 sandag_abm/src/main/resources/log4j-sandag01.properties delete mode 100644 sandag_abm/src/main/resources/log4j-sandag01.xml delete mode 100644 sandag_abm/src/main/resources/log4j-sandag02.properties delete mode 100644 sandag_abm/src/main/resources/log4j-sandag02.xml delete mode 100644 sandag_abm/src/main/resources/log4j-sandag03.properties delete mode 100644 sandag_abm/src/main/resources/log4j-sandag03.xml delete mode 100644 sandag_abm/src/main/resources/log4j-sandag04.properties delete mode 100644 sandag_abm/src/main/resources/log4j-sandag04.xml delete mode 100644 sandag_abm/src/main/resources/log4j.xml delete mode 100644 sandag_abm/src/main/resources/log4j_AtTransitCheck.xml delete mode 100644 sandag_abm/src/main/resources/log4j_d2t.xml delete mode 100644 sandag_abm/src/main/resources/log4j_hh.xml delete mode 100644 sandag_abm/src/main/resources/log4j_mtx.xml delete mode 100644 sandag_abm/src/main/resources/log4j_test.xml delete mode 100644 sandag_abm/src/main/resources/mapAndRun.bat delete mode 100644 sandag_abm/src/main/resources/pskill.exe delete mode 100644 sandag_abm/src/main/resources/runDriver.cmd delete mode 100644 sandag_abm/src/main/resources/runHhMgr.cmd delete mode 100644 sandag_abm/src/main/resources/runHhMgr_log.bat delete mode 100644 sandag_abm/src/main/resources/runMtxMgr.cmd delete mode 100644 sandag_abm/src/main/resources/runMtxMgr_log.bat delete mode 100644 sandag_abm/src/main/resources/runSandag01.cmd delete mode 100644 sandag_abm/src/main/resources/runSandag01_log.bat delete mode 100644 sandag_abm/src/main/resources/runSandag02.cmd delete mode 100644 sandag_abm/src/main/resources/runSandag03.cmd delete mode 100644 sandag_abm/src/main/resources/runSandag04.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagAbm_SEM.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagAbm_SMM.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagBikeLogsums.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd delete mode 100644 sandag_abm/src/main/resources/runSandagWalkLogsums.cmd delete mode 100644 sandag_abm/src/main/resources/runTransitReporter.cmd delete mode 100644 sandag_abm/src/main/resources/sandag_abm.properties delete mode 100644 sandag_abm/src/main/resources/serverswap.bat delete mode 100644 sandag_abm/src/main/resources/serverswap_files.csv delete mode 100644 sandag_abm/src/main/resources/setup.bat delete mode 100644 sandag_abm/src/main/resources/stopABM.cmd delete mode 100644 sandag_abm/src/main/resources/taskkill.bat delete mode 100644 sandag_abm/src/main/resources/updateYearSpecificProps.bat delete mode 100644 sandag_abm/src/main/resources/w9xpopen.exe diff --git a/sandag_abm/src/main/emme/init_emme_project.py b/sandag_abm/src/main/emme/init_emme_project.py deleted file mode 100644 index 2809405..0000000 --- a/sandag_abm/src/main/emme/init_emme_project.py +++ /dev/null @@ -1,97 +0,0 @@ -#/////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2019. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// init_emme_project.py /// -#//// /// -#//// Usage: init_emme_project.py [-r root] [-t title] /// -#//// /// -#//// [-r root]: Specifies the root directory in which to create /// -#//// the Emme project. /// -#//// If omitted, defaults to the current working directory /// -#//// [-t title]: The title of the Emme project and Emme database. /// -#//// If omitted, defaults to SANDAG empty database. /// -#//// [-v emmeversion]: Emme version to use to create the project. /// -#//// If omitted, defaults to 4.3.7. /// -#//// /// -#//// /// -#//// /// -#//// /// -#/////////////////////////////////////////////////////////////////////////////// - -import inro.emme.desktop.app as _app -import inro.emme.desktop.types as _ws_types -import inro.emme.database.emmebank as _eb -import argparse -import os - -WKT_PROJECTION = 'PROJCS["NAD_1983_NSRS2007_StatePlane_California_VI_FIPS_0406_Ft_US",GEOGCS["GCS_NAD_1983_NSRS2007",DATUM["D_NAD_1983_NSRS2007",SPHEROID["GRS_1980",6378137.0,298.257222101]],PRIMEM["Greenwich",0.0],UNIT["Degree",0.0174532925199433]],PROJECTION["Lambert_Conformal_Conic"],PARAMETER["False_Easting",6561666.666666666],PARAMETER["False_Northing",1640416.666666667],PARAMETER["Central_Meridian",-116.25],PARAMETER["Standard_Parallel_1",32.78333333333333],PARAMETER["Standard_Parallel_2",33.88333333333333],PARAMETER["Latitude_Of_Origin",32.16666666666666],UNIT["Foot_US",0.3048006096012192]];-118608900 -91259500 3048.00609601219;-100000 10000;-100000 10000;3.28083333333333E-03;0.001;0.001;IsHighPrecision' - -def init_emme_project(root, title, emmeversion): - project_path = _app.create_project(root, "emme_project") - desktop = _app.start_dedicated( - project=project_path, user_initials="WS", visible=False) - project = desktop.project - project.name = "SANDAG Emme project" - prj_file_path = os.path.join(os.path.dirname(project_path), 'NAD 1983 NSRS2007 StatePlane California VI FIPS 0406 (US Feet).prj') - with open(prj_file_path, 'w') as f: - f.write(WKT_PROJECTION) - project.spatial_reference_file = prj_file_path - project.initial_view = _ws_types.Box(6.18187e+06, 1.75917e+06, 6.42519e+06, 1.89371e+06) - project_root = os.path.dirname(project_path) - dimensions = { - 'scalar_matrices': 9999, - 'destination_matrices': 999, - 'origin_matrices': 999, - 'full_matrices': 1600, - - 'scenarios': 10, - 'centroids': 5000, - 'regular_nodes': 29999, - 'links': 90000, - 'turn_entries': 13000, - 'transit_vehicles': 200, - 'transit_lines': 450, - 'transit_segments': 40000, - 'extra_attribute_values': 28000000, - - 'functions': 99, - 'operators': 5000 - } - - # for Emme version > 4.3.7, add the sola_analyses dimension - if emmeversion != '4.3.7': - dimensions['sola_analyses'] = 240 - - os.mkdir(os.path.join(project_root, "Database")) - emmebank = _eb.create(os.path.join(project_root, "Database", "emmebank"), dimensions) - emmebank.title = title - emmebank.coord_unit_length = 0.000189394 # feet to miles - emmebank.unit_of_length = "mi" - emmebank.unit_of_cost = "$" - emmebank.unit_of_energy = "MJ" - emmebank.node_number_digits = 6 - emmebank.use_engineering_notation = True - scenario = emmebank.create_scenario(100) - scenario.title = "Empty scenario" - emmebank.dispose() - - desktop.data_explorer().add_database(emmebank.path) - desktop.add_modeller_toolbox("%<$ProjectPath>%/scripts/sandag_toolbox.mtbx") - desktop.add_modeller_toolbox("%<$ProjectPath>%/scripts/solutions.mtbx") - project.save() - - -if __name__ == "__main__": - parser = argparse.ArgumentParser(description="Create a new empty Emme project and database with Sandag defaults.") - parser.add_argument('-r', '--root', help="path to the root ABM folder, default is the working folder", - default=os.path.abspath(os.getcwd())) - parser.add_argument('-t', '--title', help="the Emmebank title", - default="SANDAG empty database") - parser.add_argument('-v', '--emmeversion', help='the Emme version', default='4.3.7') - args = parser.parse_args() - - init_emme_project(args.root, args.title, args.emmeversion) diff --git a/sandag_abm/src/main/emme/python_virtualenv.pth b/sandag_abm/src/main/emme/python_virtualenv.pth deleted file mode 100644 index c169252..0000000 --- a/sandag_abm/src/main/emme/python_virtualenv.pth +++ /dev/null @@ -1,3 +0,0 @@ -# Inserts defined python_virtualenv site-packages into the python module search path if defined -# -import sys, os; r=os.environ.get("PYTHON_VIRTUALENV"); t = 1 if r is None else sys.path.insert(0, os.path.join(r, "Lib\site-packages")); \ No newline at end of file diff --git a/sandag_abm/src/main/emme/solutions.mtbx b/sandag_abm/src/main/emme/solutions.mtbx deleted file mode 100644 index 10d5345c4324f915c320462cba2c8740f5fe522e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 118784 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 diff --git a/sandag_abm/src/main/emme/solutions_unconsolidated.mtbx b/sandag_abm/src/main/emme/solutions_unconsolidated.mtbx deleted file mode 100644 index 0674c2b5978a6c4e5ea11a14d6fb308a1417fd8d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 24576 zcmeI4&u`mg7{~p(PE%)z->g%Js;P=g5o^^hiSxs4h9=>pPFAf=Mw&4OLY|v=t+yoZ zVrO&`mzIBn3;zP*#08{XCM39XLPA4A95_sZ%MS3^iIcQqV>?ZQ=T#EF@8kE!^L$=k zo^NdLytidKhIrfRb#zC}GEs&Q=3P-_7$yMUY54lZZF7DDj%1?#tZINsMxRE6IVO1K zYla@vw6G9aiaZWm(8PiSkN^@u0!RP}AOR$R1dzZB6FABQgXdS~XO1s8J#%l*=oxLJ zV{{!`YeCI1{AQ`HDh*X^)Hf>?wJ!SRVXsh8E!XO*xS?*T(97zMI)uP3s!HjKSg+j> zi)v-F8WN!ldzS09ZBch%?cKg(*y4>VDoit+T{BzaWwExkksKS|38~eMBY8S_{$gV0 zcxC8x%WC!~y&knsxjnK!)9+D_7jfLDFm&4=JNA3RGH|@W1>*|~x?53K%fPH!Qg?iA_R_f9@~Stkt9K0XeWhNyqSObL`+B>7 z%5<&1xvkW1idWT}6Em!e?%?I+Y+yQGSs;wrZ5fC5y*4C#n%;LTudR&((4=wQP5fru zYwy6rG6~=F?)y5#S1HA`Vu)7N{#b{!) z{%N3p+FtT&u)+|pqwk%h>lla5DT{8learGSb~?VYLT-96hAeI4Uf<|8jZ@936KGDI zTynQ&6-F+9N`mo3f*e)6DSZ#^QH4D*&B#1J>^CQo{R?BqblRiKXE_2}Oz0B^K3I?d z5fCP{L z57&;+Ky(}lAOR$R z1dsp{Kmter2_OL^fCP}h|AfF>Y>?skN`%>QtmeLMJ7!aKdiw3#aMW$v-0Qke*+fZ@ zVfX~}n*9HNHu5b)|DccQSM+oG3B5&YbdCzbQ{gw^C*gbHmarQ6EAlIJ!h!^l01`j~ zNB{{S0VIF~kN^@u0rRap0F^55J!U9Jv>4l{)wVQADW&CNUe;tKpVM+#Nzn>Q9)@jNErXun`1y0q zeY4$ymkfNA`S@&z<6{YroSzNzJjchGj@h+(xfzZ(Vs5O(+6P)mUQZXY*^E}&kR>gb zFJ`quT25;@N!nP?{VutbwJZtVxswry`S7h!>`Q$LfXP%AFM+GSs5q&@K84 z{gHk_|D-MYA^nbiDLkPHO$iT$pM@Xjo3HF5$Aysq5aM^L zj)1^Ge1YK%NC`3GstX~|8REnf6?h3m1@bbY!b}u$cQPR;1j8UGkfES3?M>~;2mt}a e0&hJpOb*nJIWo`)ro1^O@_*KqIrxs7|Njm5dXxJA diff --git a/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py b/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py deleted file mode 100644 index ec8d4a9..0000000 --- a/sandag_abm/src/main/emme/toolbox/assignment/build_transit_scenario.py +++ /dev/null @@ -1,679 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// build_transit_scenario.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# The build transit scenario tool generates a new scenario in the Transit -# database (under the Database_transit directory) as a copy of a scenario in -# the base (traffic assignment) database. The base traffic scenario should have -# valid results from a traffic assignment for the travel times on links to be -# available for transit lines in mixed traffic operation. -# -# -# Inputs: -# period: the corresponding period for the scenario -# base_scenario_id: the base traffic assignment scenario in the main Emme database -# scenario_id: the ID to use for the new scenario in the Transit Emme database -# scenario_title: the title for the new scenario -# data_table_name: the root name for the source data table for the timed transfer -# line pairs and the day and regional pass costs. -# Usually the ScenarioYear -# overwrite: overwrite the scenario if it already exists. -# -# -# Script example: -""" -import inro.modeller as _m -import os -modeller = _m.Modeller() -desktop = modeller.desktop - -build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") -transit_assign = modeller.tool("sandag.assignment.transit_assignment") -load_properties = modeller.tool('sandag.utilities.properties') - -project_dir = os.path.dirname(desktop.project_path()) -main_directory = os.path.dirname(project_dir) -props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) -main_emmebank = os.path.join(project_dir, "Database", "emmebank") -scenario_id = 100 -base_scenario = main_emmebank.scenario(scenario_id) - -transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") - -periods = ["EA", "AM", "MD", "PM", "EV"] -period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) -num_processors = "MAX-1" -scenarioYear = str(props["scenarioYear"]) - -for number, period in period_ids: - src_period_scenario = main_emmebank.scenario(number) - transit_assign_scen = build_transit_scen( - period=period, base_scenario=src_period_scenario, - transit_emmebank=transit_emmebank, - scenario_id=src_period_scenario.id, - scenario_title="%s %s transit assign" % (base_scenario.title, period), - data_table_name=scenarioYear, overwrite=True) - transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, - skims_only=True, num_processors=num_processors) -""" - - - -TOOLBOX_ORDER = 21 - - -import inro.modeller as _m -import inro.emme.core.exception as _except -import inro.emme.database.emmebank as _eb -import traceback as _traceback -from copy import deepcopy as _copy -from collections import defaultdict as _defaultdict -import contextlib as _context - -import os -import sys -import math - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class BuildTransitNetwork(_m.Tool(), gen_utils.Snapshot): - - period = _m.Attribute(unicode) - scenario_id = _m.Attribute(int) - base_scenario_id = _m.Attribute(str) - - data_table_name = _m.Attribute(unicode) - scenario_title = _m.Attribute(unicode) - overwrite = _m.Attribute(bool) - - tool_run_msg = "" - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.data_table_name = None - self.base_scenario = _m.Modeller().scenario - self.scenario_id = 100 - self.scenario_title = "" - self.overwrite = False - self.attributes = [ - "period", "scenario_id", "base_scenario_id", - "data_table_name", "scenario_title", "overwrite"] - - def page(self): - if not self.data_table_name: - load_properties = _m.Modeller().tool('sandag.utilities.properties') - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_directory = os.path.dirname(project_dir) - props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - self.data_table_name = props["scenarioYear"] - - pb = _m.ToolPageBuilder(self) - pb.title = "Build transit network" - pb.description = """ - Builds the transit network for the specified period based - on existing base (traffic + transit) scenario.""" - pb.branding_text = "- SANDAG - " - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - options = [("EA", "Early AM"), - ("AM", "AM peak"), - ("MD", "Mid-day"), - ("PM", "PM peak"), - ("EV", "Evening")] - pb.add_select("period", options, title="Period:") - - root_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database", "emmebank")) - options = [(scen.id, "%s - %s" % (scen.id, scen.title)) for scen in main_emmebank.scenarios()] - pb.add_select("base_scenario_id", options, - title="Base scenario (with traffic and transit data):", - note="With period traffic results from main (traffic assignment) database at:
    %s" % main_emmebank.path) - - pb.add_text_box("scenario_id", title="ID for transit assignment scenario:") - pb.add_text_box("scenario_title", title="Scenario title:", size=80) - pb.add_text_box("data_table_name", title="Data table prefix name:", note="Default is the ScenarioYear") - pb.add_checkbox("overwrite", title=" ", label="Overwrite existing scenario") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - root_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database", "emmebank")) - base_scenario = main_emmebank.scenario(self.base_scenario_id) - transit_emmebank = _eb.Emmebank(os.path.join(root_dir, "Database_transit", "emmebank")) - results = self( - self.period, base_scenario, transit_emmebank, - self.scenario_id, self.scenario_title, - self.data_table_name, self.overwrite) - run_msg = "Transit scenario created" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, period, base_scenario, transit_emmebank, scenario_id, scenario_title, - data_table_name, overwrite=False): - modeller = _m.Modeller() - attrs = { - "period": period, - "base_scenario_id": base_scenario.id, - "transit_emmebank": transit_emmebank.path, - "scenario_id": scenario_id, - "scenario_title": scenario_title, - "data_table_name": data_table_name, - "overwrite": overwrite, - "self": str(self) - } - with _m.logbook_trace("Build transit network for period %s" % period, attributes=attrs): - gen_utils.log_snapshot("Build transit network", str(self), attrs) - copy_scenario = modeller.tool( - "inro.emme.data.scenario.copy_scenario") - periods = ["EA", "AM", "MD", "PM", "EV"] - if not period in periods: - raise Exception( - 'period: unknown value - specify one of %s' % periods) - - transit_assignment = modeller.tool( - "sandag.assignment.transit_assignment") - if transit_emmebank.scenario(scenario_id): - if overwrite: - transit_emmebank.delete_scenario(scenario_id) - else: - raise Exception("scenario_id: scenario %s already exists" % scenario_id) - - scenario = transit_emmebank.create_scenario(scenario_id) - scenario.title = scenario_title[:80] - scenario.has_traffic_results = base_scenario.has_traffic_results - scenario.has_transit_results = base_scenario.has_transit_results - for attr in sorted(base_scenario.extra_attributes(), key=lambda x: x._id): - dst_attr = scenario.create_extra_attribute(attr.type, attr.name, attr.default_value) - dst_attr.description = attr.description - for field in base_scenario.network_fields(): - scenario.create_network_field(field.type, field.name, field.atype, field.description) - network = base_scenario.get_network() - new_attrs = [ - ("TRANSIT_LINE", "@xfer_from_day", "Fare for xfer from daypass/trolley"), - ("TRANSIT_LINE", "@xfer_from_premium", "Fare for first xfer from premium"), - ("TRANSIT_LINE", "@xfer_from_coaster", "Fare for first xfer from coaster"), - ("TRANSIT_LINE", "@xfer_regional_pass", "0-fare for regional pass"), - ("TRANSIT_SEGMENT", "@xfer_from_bus", "Fare for first xfer from bus"), - ("TRANSIT_SEGMENT", "@headway_seg", "Headway adj for special xfers"), - ("TRANSIT_SEGMENT", "@transfer_penalty_s", "Xfer pen adj for special xfers"), - ("TRANSIT_SEGMENT", "@layover_board", "Boarding cost adj for special xfers"), - ("NODE", "@network_adj", "Model: 1=TAP adj, 2=circle, 3=timedxfer"), - ("NODE", "@network_adj_src", "Orig src node for timedxfer splits"), - ] - for elem, name, desc in new_attrs: - attr = scenario.create_extra_attribute(elem, name) - attr.description = desc - network.create_attribute(elem, name) - network.create_attribute("TRANSIT_LINE", "xfer_from_bus") - self._init_node_id(network) - - transit_passes = gen_utils.DataTableProc("%s_transit_passes" % data_table_name) - transit_passes = {row["pass_type"]: row["cost"] for row in transit_passes} - day_pass = float(transit_passes["day_pass"]) / 2.0 - regional_pass = float(transit_passes["regional_pass"]) / 2.0 - params = transit_assignment.get_perception_parameters(period) - mode_groups = transit_assignment.group_modes_by_fare(network, day_pass) - - bus_fares = {} - for mode_id, fares in mode_groups["bus"]: - for fare, count in fares.items(): - bus_fares[fare] = bus_fares.get(fare, 0) + count - # set nominal bus fare as unweighted average of two most frequent fares - bus_fares = sorted(bus_fares.items(), key=lambda x: x[1], reverse=True) - - if len(bus_fares) >= 2: - bus_fare = (bus_fares[0][0] + bus_fares[1][0]) / 2 - elif len(bus_fares) == 1: # unless there is only one fare value, in which case use that one - bus_fare = bus_fares[0][0] - else: - bus_fare = 0 - # find max premium mode fare - premium_fare = 0 - for mode_id, fares in mode_groups["premium"]: - for fare in fares.keys(): - premium_fare = max(premium_fare, fare) - # find max coaster_fare by checking the cumulative fare along each line - coaster_fare = 0 - for line in network.transit_lines(): - if line.mode.id != "c": - continue - segments = line.segments() - first = segments.next() - fare = first["@coaster_fare_board"] - for seg in segments: - fare += seg["@coaster_fare_inveh"] - coaster_fare = max(coaster_fare, fare) - - bus_fare_modes = [x[0] for x in mode_groups["bus"]] # have a bus fare, less than the day pass - day_pass_modes = [x[0] for x in mode_groups["day_pass"]] # boarding fare is the same as the day pass - premium_fare_modes = ["c"] + [x[0] for x in mode_groups["premium"]] # special premium services not covered by day pass - - for line in list(network.transit_lines()): - # remove the "unavailable" lines in this period - if line[params["xfer_headway"]] == 0: - network.delete_transit_line(line) - continue - # Adjust fare perception by VOT - line[params["fare"]] = line[params["fare"]] / params["vot"] - # set the fare increments for transfer combinations with day pass / regional pass - if line.mode.id in bus_fare_modes: - line["xfer_from_bus"] = max(min(day_pass - line["@fare"], line["@fare"]), 0) - line["@xfer_from_day"] = 0.0 - line["@xfer_from_premium"] = max(min(regional_pass - premium_fare, line["@fare"]), 0) - line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) - elif line.mode.id in day_pass_modes: - line["xfer_from_bus"] = max(day_pass - bus_fare, 0.0) - line["@xfer_from_day"] = 0.0 - line["@xfer_from_premium"] = max(min(regional_pass - premium_fare, line["@fare"]), 0) - line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) - elif line.mode.id in premium_fare_modes: - if line["@fare"] > day_pass or line.mode.id == "c": - # increment from bus to regional - line["xfer_from_bus"] = max(regional_pass - bus_fare, 0) - line["@xfer_from_day"] = max(regional_pass - day_pass, 0) - else: - # some "premium" modes lines are really regular fare - # increment from bus to day pass - line["xfer_from_bus"] = max(day_pass - bus_fare, 0) - line["@xfer_from_day"] = 0.0 - line["@xfer_from_premium"] = max(regional_pass - premium_fare, 0) - line["@xfer_from_coaster"] = max(min(regional_pass - coaster_fare, line["@fare"]), 0) - - for segment in network.transit_segments(): - line = segment.line - segment["@headway_seg"] = line[params["xfer_headway"]] - segment["@transfer_penalty_s"] = line["@transfer_penalty"] - segment["@xfer_from_bus"] = line["xfer_from_bus"] - network.delete_attribute("TRANSIT_LINE", "xfer_from_bus") - - self.taps_to_centroids(network) - # changed to allow timed xfers for different periods - timed_transfers_with_walk = list(gen_utils.DataTableProc("%s_timed_xfer_%s" % (data_table_name,period))) - self.timed_transfers(network, timed_transfers_with_walk, period) - #self.connect_circle_lines(network) - self.duplicate_tap_adajcent_stops(network) - # The fixed guideway travel times are stored in "@trtime_link_xx" - # and copied to data2 (ul2) for the ttf - # The congested auto times for mixed traffic are in "@auto_time" - # (output from traffic assignment) which needs to be copied to auto_time (a.k.a. timau) - # (The auto_time attribute is generated from the VDF values which include reliability factor) - src_attrs = [params["fixed_link_time"]] - dst_attrs = ["data2"] - if scenario.has_traffic_results and "@auto_time" in scenario.attributes("LINK"): - src_attrs.append("@auto_time") - dst_attrs.append("auto_time") - values = network.get_attribute_values("LINK", src_attrs) - network.set_attribute_values("LINK", dst_attrs, values) - scenario.publish_network(network) - - return scenario - - @_m.logbook_trace("Convert TAP nodes to centroids") - def taps_to_centroids(self, network): - # delete existing traffic centroids - for centroid in list(network.centroids()): - network.delete_node(centroid, cascade=True) - - node_attrs = network.attributes("NODE") - link_attrs = network.attributes("LINK") - for node in list(network.nodes()): - if node["@tap_id"] > 0: - centroid = network.create_node(node["@tap_id"], is_centroid=True) - for attr in node_attrs: - centroid[attr] = node[attr] - for link in node.outgoing_links(): - connector = network.create_link(centroid, link.j_node, link.modes) - connector.vertices = link.vertices - for attr in link_attrs: - connector[attr] = link[attr] - for link in node.incoming_links(): - connector = network.create_link(link.i_node, centroid, link.modes) - connector.vertices = link.vertices - for attr in link_attrs: - connector[attr] = link[attr] - network.delete_node(node, cascade=True) - - @_m.logbook_trace("Duplicate TAP access and transfer access stops") - def duplicate_tap_adajcent_stops(self, network): - # Expand network by duplicating TAP adjacent stops - network.create_attribute("NODE", "tap_stop", False) - all_transit_modes = set([mode for mode in network.modes() if mode.type == "TRANSIT"]) - access_mode = set([network.mode("a")]) - transfer_mode = network.mode("x") - walk_mode = network.mode("w") - - # Mark TAP adjacent stops and split TAP connectors - for centroid in network.centroids(): - out_links = list(centroid.outgoing_links()) - in_links = list(centroid.incoming_links()) - for link in out_links + in_links: - link.length = 0.0005 # setting length so that connector access time = 0.01 - for link in out_links: - real_stop = link.j_node - has_adjacent_transfer_links = False - has_adjacent_walk_links = False - for stop_link in real_stop.outgoing_links(): - if stop_link == link.reverse_link: - continue - if transfer_mode in stop_link.modes : - has_adjacent_transfer_links = True - if walk_mode in stop_link.modes : - has_adjacent_walk_links = True - - if has_adjacent_transfer_links or has_adjacent_walk_links: - length = link.length - tap_stop = network.split_link(centroid, real_stop, self._get_node_id(), include_reverse=True) - tap_stop["@network_adj"] = 1 - real_stop.tap_stop = tap_stop - transit_access_link = network.link(real_stop, tap_stop) - for link in transit_access_link, transit_access_link.reverse_link: - link.modes = all_transit_modes - link.length = 0 - for p in ["ea", "am", "md", "pm", "ev"]: - link["@time_link_" + p] = 0 - access_link = network.link(tap_stop, centroid) - access_link.modes = access_mode - access_link.reverse_link.modes = access_mode - access_link.length = length - access_link.reverse_link.length = length - - line_attributes = network.attributes("TRANSIT_LINE") - seg_attributes = network.attributes("TRANSIT_SEGMENT") - - # re-route the transit lines through the new TAP-stops - for line in network.transit_lines(): - # store line and segment data for re-routing - line_data = dict((k, line[k]) for k in line_attributes) - line_data["id"] = line.id - line_data["vehicle"] = line.vehicle - - seg_data = {} - itinerary = [] - tap_adjacent_stops = [] - - for seg in line.segments(include_hidden=True): - seg_data[(seg.i_node, seg.j_node, seg.loop_index)] = \ - dict((k, seg[k]) for k in seg_attributes) - itinerary.append(seg.i_node.number) - if seg.i_node.tap_stop and seg.allow_boardings: - # insert tap_stop, real_stop loop after tap_stop - real_stop = seg.i_node - tap_stop = real_stop.tap_stop - itinerary.extend([tap_stop.number, real_stop.number]) - tap_adjacent_stops.append(len(itinerary) - 1) # index of "real" stop in itinerary - - if tap_adjacent_stops: - network.delete_transit_line(line) - new_line = network.create_transit_line( - line_data.pop("id"), - line_data.pop("vehicle"), - itinerary) - for k, v in line_data.iteritems(): - new_line[k] = v - - for seg in new_line.segments(include_hidden=True): - data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) - for k, v in data.iteritems(): - seg[k] = v - for index in tap_adjacent_stops: - access_seg = new_line.segment(index - 2) - egress_seg = new_line.segment(index - 1) - real_seg = new_line.segment(index) - for k in seg_attributes: - access_seg[k] = egress_seg[k] = real_seg[k] - access_seg.allow_boardings = False - access_seg.allow_alightings = True - access_seg.transit_time_func = 3 - access_seg.dwell_time = real_seg.dwell_time - egress_seg.allow_boardings = True - egress_seg.allow_alightings = True - egress_seg.transit_time_func = 3 - egress_seg.dwell_time = 0 - real_seg.allow_boardings = True - real_seg.allow_alightings = False - real_seg.dwell_time = 0 - - network.delete_attribute("NODE", "tap_stop") - - @_m.logbook_trace("Add timed-transfer links", save_arguments=True) - def timed_transfers(self, network, timed_transfers_with_walk, period): - no_walk_link_error = "no walk link from line %s to %s" - node_not_found_error = "node %s not found in itinerary for line %s; "\ - "the to_line may end at the transfer stop" - - def find_walk_link(from_line, to_line): - to_nodes = set([s.i_node for s in to_line.segments(True) - if s.allow_boardings]) - link_candidates = [] - for seg in from_line.segments(True): - if not s.allow_alightings: - continue - for link in seg.i_node.outgoing_links(): - if link.j_node in to_nodes: - link_candidates.append(link) - if not link_candidates: - raise Exception(no_walk_link_error % (from_line, to_line)) - # if there are multiple connecting links take the shortest one - return sorted(link_candidates, key=lambda x: x.length)[0] - - def link_on_line(line, node, near_side_stop): - node = network.node(node) - if near_side_stop: - for seg in line.segments(): - if seg.j_node == node: - return seg.link - else: - for seg in line.segments(): - if seg.i_node == node: - return seg.link - raise Exception(node_not_found_error % (node, line)) - - # Group parallel transfers together (same pair of alighting-boarding nodes from the same line) - walk_transfers = _defaultdict(lambda: []) - for i, transfer in enumerate(timed_transfers_with_walk, start=1): - try: - from_line = network.transit_line(transfer["from_line"]) - if not from_line: - raise Exception("from_line %s does not exist" % transfer["from_line"]) - to_line = network.transit_line(transfer["to_line"]) - if not to_line: - raise Exception("to_line %s does not exist" % transfer["to_line"]) - walk_link = find_walk_link(from_line, to_line) - from_link = link_on_line(from_line, walk_link.i_node, near_side_stop=True) - to_link = link_on_line(to_line, walk_link.j_node, near_side_stop=False) - walk_transfers[(from_link, to_link)].append({ - "to_line": to_line, - "from_line": from_line, - "walk_link": walk_link, - "wait": transfer["wait_time"], - }) - except Exception as error: - new_message = "Timed transfer[%s]: %s" % (i, error.message) - raise type(error), type(error)(new_message), sys.exc_info()[2] - - # If there is only one transfer at the location (redundant case) - # OR all transfers are from the same line (can have different waits) - # OR all transfers are to the same line and have the same wait - # Merge all transfers onto the same transfer node - network_transfers = [] - for (from_link, to_link), transfers in walk_transfers.iteritems(): - walk_links = set([t["walk_link"] for t in transfers]) - from_lines = set([t["from_line"] for t in transfers]) - to_lines = set([t["to_line"] for t in transfers]) - waits = set(t["wait"] for t in transfers) - if len(transfers) == 1 or len(from_lines) == 1 or (len(to_lines) == 1 and len(waits) == 1): - network_transfers.append({ - "from_link": from_link, - "to_link": to_link, - "to_lines": list(to_lines), - "from_lines": list(from_lines), - "walk_link": walk_links.pop(), - "wait": dict((t["to_line"], t["wait"]) for t in transfers)}) - else: - for transfer in transfers: - network_transfers.append({ - "from_link": from_link, - "to_link": to_link, - "to_lines": [transfer["to_line"]], - "from_lines": [transfer["from_line"]], - "walk_link": transfer["walk_link"], - "wait": {transfer["to_line"]: transfer["wait"]}}) - - def split_link(link, node_id, lines, split_links, stop_attr, waits=None): - near_side_stop = (stop_attr == "allow_alightings") - orig_link = link - if link in split_links: - link = split_links[link] - i_node, j_node = link.i_node, link.j_node - length = link.length - proportion = min(0.006 / length, 0.2) - if near_side_stop: - proportion = 1 - proportion - new_node = network.split_link(i_node, j_node, node_id, False, proportion) - new_node["@network_adj"] = 3 - new_node["@network_adj_src"] = orig_link.j_node.number if near_side_stop else orig_link.i_node.number - in_link = network.link(i_node, new_node) - out_link = network.link(new_node, j_node) - split_links[orig_link] = in_link if near_side_stop else out_link - if near_side_stop: - in_link.length = length - out_link.length = 0 - for p in ["ea", "am", "md", "pm", "ev"]: - out_link["@trtime_link_" + p] = 0 - else: - out_link.length = length - in_link.length = 0 - for p in ["ea", "am", "md", "pm", "ev"]: - in_link["@trtime_link_" + p] = 0 - - for seg in in_link.segments(): - if not near_side_stop: - seg.transit_time_func = 3 - seg["@coaster_fare_inveh"] = 0 - for seg in out_link.segments(): - if near_side_stop: - seg.transit_time_func = 3 - seg.allow_alightings = seg.allow_boardings = False - seg.dwell_time = 0 - if seg.line in lines: - seg[stop_attr] = True - if stop_attr == "allow_boardings": - seg["@headway_seg"] = float(waits[seg.line]) * 2 - return new_node - - # process the transfer points, split links and set attributes - split_links = {} - for transfer in network_transfers: - new_alight_node = split_link( - transfer["from_link"], self._get_node_id(), transfer["from_lines"], - split_links, "allow_alightings") - new_board_node = split_link( - transfer["to_link"], self._get_node_id(), transfer["to_lines"], - split_links, "allow_boardings", waits=transfer["wait"]) - walk_link = transfer["walk_link"] - transfer_link = network.create_link( - new_alight_node, new_board_node, [network.mode("x")]) - for attr in network.attributes("LINK"): - transfer_link[attr] = walk_link[attr] - - @_m.logbook_trace("Add circle line free layover transfers") - def connect_circle_lines(self, network): - network.create_attribute("NODE", "circle_lines") - line_attributes = network.attributes("TRANSIT_LINE") - seg_attributes = network.attributes("TRANSIT_SEGMENT") - - def offset_coords(node): - rho = math.sqrt(5000) - phi = 3 * math.pi / 4 + node.circle_lines * math.pi / 12 - x = node.x + rho * math.cos(phi) - y = node.y + rho * math.sin(phi) - node.circle_lines += 1 - return(x, y) - - transit_lines = list(network.transit_lines()) - for line in transit_lines: - first_seg = line.segment(0) - last_seg = line.segment(-1) - if first_seg.i_node == last_seg.i_node: - # Add new node, offset from existing node - start_node = line.segment(0).i_node - xfer_node = network.create_node(self._get_node_id(), False) - xfer_node["@network_adj"] = 2 - xfer_node.x, xfer_node.y = offset_coords(start_node) - network.create_link(start_node, xfer_node, [line.vehicle.mode]) - network.create_link(xfer_node, start_node, [line.vehicle.mode]) - - # copy transit line data, re-route itinerary to and from new node - line_data = dict((k, line[k]) for k in line_attributes) - line_data["id"] = line.id - line_data["vehicle"] = line.vehicle - first_seg.allow_boardings = True - first_seg.allow_alightings = False - first_seg_data = dict((k, first_seg[k]) for k in seg_attributes) - first_seg_data.update({ - "@headway_seg": 0.01, "dwell_time": 0, "transit_time_func": 3, - "@transfer_penalty_s": 0, "@xfer_from_bus": 0, "@layover_board": 1 - }) - last_seg.allow_boardings = False - last_seg.allow_alightings = True - last_seg_data = dict((k, last_seg[k]) for k in seg_attributes) - last_seg_data.update({ - "@headway_seg": 0.01, "dwell_time": 5.0, "transit_time_func": 3 - # incremental dwell time for layover of 5 min - # Note: some lines seem to have a layover of 0, most of 5 mins - }) - seg_data = { - (xfer_node, start_node, 1): first_seg_data, - (xfer_node, None, 1): last_seg_data} - itinerary = [xfer_node.number] - for seg in line.segments(): - seg_data[(seg.i_node, seg.j_node, seg.loop_index)] = dict((k, seg[k]) for k in seg_attributes) - itinerary.append(seg.i_node.number) - last_seg = line.segment(-1) - seg_data[(last_seg.i_node, xfer_node, 1)] = dict((k, last_seg[k]) for k in seg_attributes) - seg_data[(last_seg.i_node, xfer_node, 1)]["transit_time_func"] = 3 - itinerary.extend([last_seg.i_node.number, xfer_node.number]) - - network.delete_transit_line(line) - new_line = network.create_transit_line( - line_data.pop("id"), line_data.pop("vehicle"), itinerary) - for k, v in line_data.iteritems(): - new_line[k] = v - for seg in new_line.segments(include_hidden=True): - data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) - for k, v in data.iteritems(): - seg[k] = v - - network.delete_attribute("NODE", "circle_lines") - - def _init_node_id(self, network): - new_node_id = max(n.number for n in network.nodes()) - self._new_node_id = math.ceil(new_node_id / 10000.0) * 10000 - - def _get_node_id(self): - self._new_node_id += 1 - return self._new_node_id diff --git a/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py b/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py deleted file mode 100644 index cdf652e..0000000 --- a/sandag_abm/src/main/emme/toolbox/assignment/traffic_assignment.py +++ /dev/null @@ -1,1087 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// traffic_assignment.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# The Traffic assignment tool runs the traffic assignment and skims per -# period on the current primary scenario. -# -# The traffic assignment is a 15-class assignment with generalized cost on -# links and BPR-type volume-delay functions which include capacities on links -# and at intersection approaches. The assignment is run using the -# fast-converging Second-Order Linear Approximation (SOLA) method in Emme to -# a relative gap of 5x10-4. The per-link fixed costs include toll values and -# operating costs which vary by class of demand. -# Assignment matrices and resulting network flows are always in PCE. -# -# Inputs: -# period: the time-of-day period, one of EA, AM, MD, PM, EV. -# msa_iteration: global iteration number. If greater than 1, existing flow -# values must be present and the resulting flows on links and turns will -# be the weighted average of this assignment and the existing values. -# relative_gap: minimum relative stopping criteria. -# max_iterations: maximum iterations stopping criteria. -# num_processors: number of processors to use for the traffic assignments. -# select_link: specify one or more select link analysis setups as a list of -# specifications with three keys: -# "expression": selection expression to identify the link(s) of interest. -# "suffix": the suffix to use in the naming of per-class result -# attributes and matrices, up to 6 characters. -# "threshold": the minimum number of links which must be encountered -# for the path selection. -# Example: -# select_link = [ -# {"expression": "@tov_id=4578 or @tcov_id=9203", "suffix": "fwy", "threshold": "1"} -# ] -# raise_zero_dist: if checked, the distance skim for the SOVGP is checked for -# any zero values, which would indicate a disconnected zone, in which case -# an error is raised and the model run is halted. -# -# Matrices: -# All traffic demand and skim matrices. -# See list of classes under __call__ method, or matrix list under report method. -# -# Script example: -""" -import inro.modeller as _m -import os -import inro.emme.database.emmebank as _eb - -modeller = _m.Modeller() -desktop = modeller.desktop -traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") -export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") -load_properties = modeller.tool('sandag.utilities.properties') -project_dir = os.path.dirname(_m.Modeller().desktop.project.path) -main_directory = os.path.dirname(project_dir) -output_dir = os.path.join(main_directory, "output") -props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - - -main_emmebank = os.path.join(project_dir, "Database", "emmebank") -scenario_id = 100 -base_scenario = main_emmebank.scenario(scenario_id) - -periods = ["EA", "AM", "MD", "PM", "EV"] -period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) - -msa_iteration = 1 -relative_gap = 0.0005 -max_assign_iterations = 100 -num_processors = "MAX-1" -select_link = None # Optional select link specification - -for number, period in period_ids: - period_scenario = main_emmebank.scenario(number) - traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, - num_processors, period_scenario, select_link) - omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) - if msa_iteration < 4: - export_traffic_skims(period, omx_file, base_scenario) -""" - - -TOOLBOX_ORDER = 20 - - -import inro.modeller as _m -import inro.emme.core.exception as _except -import traceback as _traceback -from contextlib import contextmanager as _context -import numpy -import array -import os -import json as _json - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class TrafficAssignment(_m.Tool(), gen_utils.Snapshot): - - period = _m.Attribute(unicode) - msa_iteration = _m.Attribute(int) - relative_gap = _m.Attribute(float) - max_iterations = _m.Attribute(int) - num_processors = _m.Attribute(str) - select_link = _m.Attribute(unicode) - raise_zero_dist = _m.Attribute(bool) - stochastic = _m.Attribute(bool) - input_directory = _m.Attribute(str) - - tool_run_msg = "" - - def __init__(self): - self.msa_iteration = 1 - self.relative_gap = 0.0005 - self.max_iterations = 100 - self.num_processors = "MAX-1" - self.raise_zero_dist = True - self.select_link = '[]' - self.stochastic = False - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.attributes = ["period", "msa_iteration", "relative_gap", "max_iterations", - "num_processors", "select_link", "raise_zero_dist", "stochastic", "input_directory"] - version = os.environ.get("EMMEPATH", "") - self._version = version[-5:] if version else "" - self._skim_classes_separately = True # Used for debugging only - self._stats = {} - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Traffic assignment" - pb.description = """ -The Traffic assignment tool runs the traffic assignment and skims per -period on the current primary scenario. -
    -The traffic assignment is a 15-class assignment with generalized cost on -links and BPR-type volume-delay functions which include capacities on links -and at intersection approaches. The assignment is run using the -fast-converging Second-Order Linear Approximation (SOLA) method in Emme to -a relative gap of 5x10-4. The per-link fixed costs include toll values and -operating costs which vary by class of demand. -Assignment matrices and resulting network flows are always in PCE. -""" - pb.branding_text = "- SANDAG - Assignment" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - options = [("EA","Early AM"), - ("AM","AM peak"), - ("MD","Mid-day"), - ("PM","PM peak"), - ("EV","Evening")] - pb.add_select("period", options, title="Period:") - pb.add_text_box("msa_iteration", title="MSA iteration:", note="If >1 will apply MSA to flows.") - pb.add_text_box("relative_gap", title="Relative gap:") - pb.add_text_box("max_iterations", title="Max iterations:") - dem_utils.add_select_processors("num_processors", pb, self) - pb.add_checkbox("raise_zero_dist", title=" ", label="Raise on zero distance value", - note="Check for and raise an exception if a zero value is found in the SOVGP_DIST matrix.") - pb.add_checkbox( - 'stochastic', - title=" ", - label="Run as a stochastic assignment", - note="If the current MSA iteration is the last (4th) one, the SOLA traffic assignment is replaced with a stochastic traffic assignment." - ) - pb.add_select_file('input_directory', 'directory', title='Select input directory') - self._add_select_link_interface(pb) - return pb.render() - - - def _add_select_link_interface(self, pb): - pb.add_html(""" -""") - pb.add_text_box("select_link", multi_line=True) - pb.wrap_html(title="Select link(s):", - body=""" - - - -

    - -
    -
    -
    - Click for available attributes -
    -
    - -
    -
    """) - pb.add_html(""" -""" % {"tool_proxy_tag": pb.tool_proxy_tag, }) - - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - results = self(self.period, self.msa_iteration, self.relative_gap, self.max_iterations, - self.num_processors, scenario, self.select_link, self.raise_zero_dist, - self.stochastic, self.input_directory) - run_msg = "Traffic assignment completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, period, msa_iteration, relative_gap, max_iterations, num_processors, scenario, - select_link=[], raise_zero_dist=True, stochastic=False, input_directory=None): - select_link = _json.loads(select_link) if isinstance(select_link, basestring) else select_link - attrs = { - "period": period, - "msa_iteration": msa_iteration, - "relative_gap": relative_gap, - "max_iterations": max_iterations, - "num_processors": num_processors, - "scenario": scenario.id, - "select_link": _json.dumps(select_link), - "raise_zero_dist": raise_zero_dist, - "stochastic": stochastic, - "input_directory": input_directory, - "self": str(self) - } - self._stats = {} - with _m.logbook_trace("Traffic assignment for period %s" % period, attributes=attrs): - gen_utils.log_snapshot("Traffic assignment", str(self), attrs) - periods = ["EA", "AM", "MD", "PM", "EV"] - if not period in periods: - raise _except.ArgumentError( - 'period: unknown value - specify one of %s' % periods) - num_processors = dem_utils.parse_num_processors(num_processors) - # Main list of assignment classes - classes = [ - { # 0 - "name": 'SOV_NT_L', "mode": 's', "PCE": 1, "VOT": 8.81, "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 1 - "name": 'SOV_TR_L', "mode": 'S', "PCE": 1, "VOT": 8.81, "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 2 - "name": 'HOV2_L', "mode": 'H', "PCE": 1, "VOT": 8.81, "cost": '@cost_hov2', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] - }, - { # 3 - "name": 'HOV3_L', "mode": 'I', "PCE": 1, "VOT": 8.81, "cost": '@cost_hov3', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] - }, - { # 4 - "name": 'SOV_NT_M', "mode": 's', "PCE": 1, "VOT": 18.0, "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 5 - "name": 'SOV_TR_M', "mode": 'S', "PCE": 1, "VOT": 18.0, "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 6 - "name": 'HOV2_M', "mode": 'H', "PCE": 1, "VOT": 18.0, "cost": '@cost_hov2', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] - }, - { # 7 - "name": 'HOV3_M', "mode": 'I', "PCE": 1, "VOT": 18.0, "cost": '@cost_hov3', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] - }, - { # 8 - "name": 'SOV_NT_H', "mode": 's', "PCE": 1, "VOT": 85., "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 9 - "name": 'SOV_TR_H', "mode": 'S', "PCE": 1, "VOT": 85., "cost": '@cost_auto', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.SOV", "TOLLDIST"] - }, - { # 10 - "name": 'HOV2_H', "mode": 'H', "PCE": 1, "VOT": 85., "cost": '@cost_hov2', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV2", "TOLLDIST.HOV2", "HOVDIST"] - }, - { # 11 - "name": 'HOV3_H', "mode": 'I', "PCE": 1, "VOT": 85., "cost": '@cost_hov3', - "skims": ["TIME", "DIST", "REL", "TOLLCOST.HOV3", "TOLLDIST.HOV3", "HOVDIST"] - }, - { # 12 - "name": 'TRK_L', "mode": 'T', "PCE": 1.3, "VOT": 67., "cost": '@cost_lgt_truck', - "skims": ["TIME", "DIST", "TOLLCOST.TRK_L"] - }, - { # 13 - "name": 'TRK_M', "mode": 'M', "PCE": 1.5, "VOT": 68., "cost": '@cost_med_truck', - "skims": ["TIME", "DIST", "TOLLCOST.TRK_M"] - }, - { # 14 - "name": 'TRK_H', "mode": 'V', "PCE": 2.5, "VOT": 89., "cost": '@cost_hvy_truck', - "skims": ["TIME", "DIST", "TOLLCOST.TRK_H"] - }, - ] - - # change mode to allow sovntp on SR125 - # TODO: incorporate this into import_network instead - # also, consider updating mode definitions - self.change_mode_sovntp(scenario) - - if period == "MD" and (msa_iteration == 1 or not scenario.mode('D')): - self.prepare_midday_generic_truck(scenario) - - if 1 < msa_iteration < 4: - # Link and turn flows - link_attrs = ["auto_volume"] - turn_attrs = ["auto_volume"] - for traffic_class in classes: - link_attrs.append("@%s" % (traffic_class["name"].lower())) - turn_attrs.append("@p%s" % (traffic_class["name"].lower())) - msa_link_flows = scenario.get_attribute_values("LINK", link_attrs)[1:] - msa_turn_flows = scenario.get_attribute_values("TURN", turn_attrs)[1:] - - if stochastic: - self.run_stochastic_assignment( - period, - relative_gap, - max_iterations, - num_processors, - scenario, - classes, - input_directory - ) - else: - self.run_assignment(period, relative_gap, max_iterations, num_processors, scenario, classes, select_link) - - - if 1 < msa_iteration < 4: - link_flows = scenario.get_attribute_values("LINK", link_attrs) - values = [link_flows.pop(0)] - for msa_array, flow_array in zip(msa_link_flows, link_flows): - msa_vals = numpy.frombuffer(msa_array, dtype='float32') - flow_vals = numpy.frombuffer(flow_array, dtype='float32') - result = msa_vals + (1.0 / msa_iteration) * (flow_vals - msa_vals) - result_array = array.array('f') - result_array.fromstring(result.tostring()) - values.append(result_array) - scenario.set_attribute_values("LINK", link_attrs, values) - - turn_flows = scenario.get_attribute_values("TURN", turn_attrs) - values = [turn_flows.pop(0)] - for msa_array, flow_array in zip(msa_turn_flows, turn_flows): - msa_vals = numpy.frombuffer(msa_array, dtype='float32') - flow_vals = numpy.frombuffer(flow_array, dtype='float32') - result = msa_vals + (1.0 / msa_iteration) * (flow_vals - msa_vals) - result_array = array.array('f') - result_array.fromstring(result.tostring()) - values.append(result_array) - scenario.set_attribute_values("TURN", turn_attrs, values) - - self.calc_network_results(period, num_processors, scenario) - - if msa_iteration <= 4: - self.run_skims(period, num_processors, scenario, classes) - self.report(period, scenario, classes) - # Check that the distance matrix is valid (no disconnected zones) - # Using SOVGPL class as representative - if raise_zero_dist: - name = "%s_SOV_TR_H_DIST" % period - dist_stats = self._stats[name] - if dist_stats[1] == 0: - zones = scenario.zone_numbers - matrix = scenario.emmebank.matrix(name) - data = matrix.get_numpy_data(scenario) - row, col = numpy.unravel_index(data.argmin(), data.shape) - row, col = zones[row], zones[col] - raise Exception("Disconnected zone error: 0 value found in matrix %s from zone %s to %s" % (name, row, col)) - - def run_assignment(self, period, relative_gap, max_iterations, num_processors, scenario, classes, select_link): - emmebank = scenario.emmebank - - modeller = _m.Modeller() - set_extra_function_para = modeller.tool( - "inro.emme.traffic_assignment.set_extra_function_parameters") - create_attribute = modeller.tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - traffic_assign = modeller.tool( - "inro.emme.traffic_assignment.sola_traffic_assignment") - net_calc = gen_utils.NetworkCalculator(scenario) - - if period in ["AM", "PM"]: - # For freeway links in AM and PM periods, convert VDF to type 25 - net_calc("vdf", "25", "vdf=10") - - p = period.lower() - assign_spec = self.base_assignment_spec( - relative_gap, max_iterations, num_processors) - with _m.logbook_trace("Prepare traffic data for period %s" % period): - with _m.logbook_trace("Input link attributes"): - # set extra attributes for the period for VDF - # ul1 = @time_link (period) - # ul2 = transit flow -> volad (for assignment only) - # ul3 = @capacity_link (period) - el1 = "@green_to_cycle" - el2 = "@sta_reliability" - el3 = "@capacity_inter" - set_extra_function_para(el1, el2, el3, emmebank=emmebank) - - # set green to cycle to el1=@green_to_cycle for VDF - att_name = "@green_to_cycle_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@green_to_cycle" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set static reliability to el2=@sta_reliability for VDF - att_name = "@sta_reliability_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@sta_reliability" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set capacity_inter to el3=@capacity_inter for VDF - att_name = "@capacity_inter_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@capacity_inter" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set link time - net_calc("ul1", "@time_link_%s" % p, "modes=d") - net_calc("ul3", "@capacity_link_%s" % p, "modes=d") - # set number of lanes (not used in VDF, just for reference) - net_calc("lanes", "@lane_%s" % p, "modes=d") - if period in ["EA", "MD", "EV"]: - # For links with signals inactive in the off-peak periods, convert VDF to type 11 - net_calc("vdf", "11", "modes=d and @green_to_cycle=0 and @traffic_control=4,5 and vdf=24") - # # Set HOV2 cost attribute - # create_attribute("LINK", "@cost_hov2_%s" % p, "toll (non-mngd) + cost for HOV2", - # 0, overwrite=True, scenario=scenario) - # net_calc("@cost_hov2_%s" % p, "@cost_hov_%s" % p, "modes=d") - # net_calc("@cost_hov2_%s" % p, "@cost_auto_%s" % p, "@lane_restriction=3") - - with _m.logbook_trace("Transit line headway and background traffic"): - # set headway for the period - hdw = {"ea": "@headway_op", - "am": "@headway_am", - "md": "@headway_op", - "pm": "@headway_pm", - "ev": "@headway_op"} - net_calc("hdw", hdw[p], {"transit_line": "all"}) - - # transit vehicle as background flow with periods - period_hours = {'ea': 3, 'am': 3, 'md': 6.5, 'pm': 3.5, 'ev': 5} - expression = "(60 / hdw) * vauteq * %s" % (period_hours[p]) - net_calc("ul2", "0", "modes=d") - net_calc("ul2", expression, - selections={"link": "modes=d", "transit_line": "hdw=0.02,9999"}, - aggregation="+") - - with _m.logbook_trace("Per-class flow attributes"): - for traffic_class in classes: - demand = 'mf"%s_%s"' % (period, traffic_class["name"]) - link_cost = "%s_%s" % (traffic_class["cost"], p) if traffic_class["cost"] else "@cost_operating" - - att_name = "@%s" % (traffic_class["name"].lower()) - att_des = "%s %s link volume" % (period, traffic_class["name"]) - link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) - att_name = "@p%s" % (traffic_class["name"].lower()) - att_des = "%s %s turn volume" % (period, traffic_class["name"]) - turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) - - class_spec = { - "mode": traffic_class["mode"], - "demand": demand, - "generalized_cost": { - "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] - }, - "results": { - "link_volumes": link_flow.id, "turn_volumes": turn_flow.id, - "od_travel_times": None - } - } - assign_spec["classes"].append(class_spec) - if select_link: - for class_spec in assign_spec["classes"]: - class_spec["path_analyses"] = [] - for sub_spec in select_link: - expr = sub_spec["expression"] - suffix = sub_spec["suffix"] - threshold = sub_spec["threshold"] - if not expr and not suffix: - continue - with _m.logbook_trace("Prepare for select link analysis '%s' - %s" % (expr, suffix)): - slink = create_attribute("LINK", "@slink_%s" % suffix, "selected link for %s" % suffix, 0, - overwrite=True, scenario=scenario) - net_calc(slink.id, "1", expr) - with _m.logbook_trace("Initialize result matrices and extra attributes"): - for traffic_class, class_spec in zip(classes, assign_spec["classes"]): - att_name = "@sl_%s_%s" % (traffic_class["name"].lower(), suffix) - att_des = "%s %s '%s' sel link flow"% (period, traffic_class["name"], suffix) - link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) - att_name = "@psl_%s_%s" % (traffic_class["name"].lower(), suffix) - att_des = "%s %s '%s' sel turn flow" % (period, traffic_class["name"], suffix) - turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) - - name = "SELDEM_%s_%s_%s" % (period, traffic_class["name"], suffix) - desc = "Selected demand for %s %s %s" % (period, traffic_class["name"], suffix) - seldem = dem_utils.create_full_matrix(name, desc, scenario=scenario) - - # add select link analysis - class_spec["path_analyses"].append({ - "link_component": slink.id, - "turn_component": None, - "operator": "+", - "selection_threshold": { "lower": threshold, "upper": 999999}, - "path_to_od_composition": { - "considered_paths": "SELECTED", - "multiply_path_proportions_by": {"analyzed_demand": True, "path_value": False} - }, - "analyzed_demand": None, - "results": { - "selected_link_volumes": link_flow.id, - "selected_turn_volumes": turn_flow.id, - "od_values": seldem.named_id - } - }) - # Run assignment - traffic_assign(assign_spec, scenario, chart_log_interval=2) - return - - def run_stochastic_assignment( - self, period, relative_gap, max_iterations, num_processors, scenario, - classes, input_directory): - load_properties = _m.Modeller().tool('sandag.utilities.properties') - main_directory = os.path.dirname(input_directory) - props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - distribution_type = props['stochasticHighwayAssignment.distributionType'] - replications = props['stochasticHighwayAssignment.replications'] - a_parameter = props['stochasticHighwayAssignment.aParameter'] - b_parameter = props['stochasticHighwayAssignment.bParameter'] - seed = props['stochasticHighwayAssignment.seed'] - - emmebank = scenario.emmebank - - modeller = _m.Modeller() - set_extra_function_para = modeller.tool( - "inro.emme.traffic_assignment.set_extra_function_parameters") - create_attribute = modeller.tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - traffic_assign = modeller.tool( - "solutions.stochastic_traffic_assignment") - net_calc = gen_utils.NetworkCalculator(scenario) - - if period in ["AM", "PM"]: - # For freeway links in AM and PM periods, convert VDF to type 25 - net_calc("vdf", "25", "vdf=10") - - p = period.lower() - assign_spec = self.base_assignment_spec( - relative_gap, max_iterations, num_processors) - assign_spec['background_traffic'] = { - "link_component": None, - "turn_component": None, - "add_transit_vehicles": True - } - with _m.logbook_trace("Prepare traffic data for period %s" % period): - with _m.logbook_trace("Input link attributes"): - # set extra attributes for the period for VDF - # ul1 = @time_link (period) - # ul2 = transit flow -> volad (for assignment only) - # ul3 = @capacity_link (period) - el1 = "@green_to_cycle" - el2 = "@sta_reliability" - el3 = "@capacity_inter" - set_extra_function_para(el1, el2, el3, emmebank=emmebank) - - # set green to cycle to el1=@green_to_cycle for VDF - att_name = "@green_to_cycle_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@green_to_cycle" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set static reliability to el2=@sta_reliability for VDF - att_name = "@sta_reliability_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@sta_reliability" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set capacity_inter to el3=@capacity_inter for VDF - att_name = "@capacity_inter_%s" % p - att = scenario.extra_attribute(att_name) - new_att_name = "@capacity_inter" - create_attribute("LINK", new_att_name, att.description, - 0, overwrite=True, scenario=scenario) - net_calc(new_att_name, att_name, "modes=d") - # set link time - net_calc("ul1", "@time_link_%s" % p, "modes=d") - net_calc("ul3", "@capacity_link_%s" % p, "modes=d") - # set number of lanes (not used in VDF, just for reference) - net_calc("lanes", "@lane_%s" % p, "modes=d") - if period in ["EA", "MD", "EV"]: - # For links with signals inactive in the off-peak periods, convert VDF to type 11 - net_calc("vdf", "11", "modes=d and @green_to_cycle=0 and @traffic_control=4,5 and vdf=24") - # # Set HOV2 cost attribute - # create_attribute("LINK", "@cost_hov2_%s" % p, "toll (non-mngd) + cost for HOV2", - # 0, overwrite=True, scenario=scenario) - # net_calc("@cost_hov2_%s" % p, "@cost_hov_%s" % p, "modes=d") - # net_calc("@cost_hov2_%s" % p, "@cost_auto_%s" % p, "@lane_restriction=3") - - with _m.logbook_trace("Transit line headway and background traffic"): - # set headway for the period: format is (attribute_name, period duration in hours) - hdw = {"ea": ("@headway_op", 3), - "am": ("@headway_am", 3), - "md": ("@headway_op", 6.5), - "pm": ("@headway_pm", 3.5), - "ev": ("@headway_op", 5)} - net_calc('ul2', '0', {'link': 'all'}) - net_calc('hdw', '9999.99', {'transit_line': 'all'}) - net_calc( - 'hdw', "{hdw} / {p} ".format(hdw=hdw[p][0], p=hdw[p][1]), - {"transit_line": "%s=0.02,9999" % hdw[p][0]} - ) - - with _m.logbook_trace("Per-class flow attributes"): - for traffic_class in classes: - demand = 'mf"%s_%s"' % (period, traffic_class["name"]) - link_cost = "%s_%s" % (traffic_class["cost"], p) if traffic_class["cost"] else "@cost_operating" - - att_name = "@%s" % (traffic_class["name"].lower()) - att_des = "%s %s link volume" % (period, traffic_class["name"]) - link_flow = create_attribute("LINK", att_name, att_des, 0, overwrite=True, scenario=scenario) - att_name = "@p%s" % (traffic_class["name"].lower()) - att_des = "%s %s turn volume" % (period, traffic_class["name"]) - turn_flow = create_attribute("TURN", att_name, att_des, 0, overwrite=True, scenario=scenario) - - class_spec = { - "mode": traffic_class["mode"], - "demand": demand, - "generalized_cost": { - "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] - }, - "results": { - "link_volumes": link_flow.id, "turn_volumes": turn_flow.id, - "od_travel_times": None - } - } - assign_spec["classes"].append(class_spec) - - # Run assignment - traffic_assign( - assign_spec, - dist_par={'type': distribution_type, 'A': a_parameter, 'B': b_parameter}, - replications=replications, - seed=seed, - orig_func=False, - random_term='ul2', - compute_travel_times=False, - scenario=scenario - ) - - with _m.logbook_trace("Reset transit line headways"): - # set headway for the period - hdw = {"ea": "@headway_op", - "am": "@headway_am", - "md": "@headway_op", - "pm": "@headway_pm", - "ev": "@headway_op"} - net_calc("hdw", hdw[p], {"transit_line": "all"}) - return - - def calc_network_results(self, period, num_processors, scenario): - modeller = _m.Modeller() - create_attribute = modeller.tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - net_calc = gen_utils.NetworkCalculator(scenario) - emmebank = scenario.emmebank - p = period.lower() - # ul2 is the total flow (volau + volad) in the skim assignment - with _m.logbook_trace("Calculation of attributes for skims"): - link_attributes = [ - ("@hovdist", "distance for HOV"), - ("@tollcost", "Toll cost for SOV autos"), - ("@h2tollcost", "Toll cost for hov2"), - ("@h3tollcost", "Toll cost for hov3"), - ("@trk_ltollcost", "Toll cost for light trucks"), - ("@trk_mtollcost", "Toll cost for medium trucks"), - ("@trk_htollcost", "Toll cost for heavy trucks"), - ("@mlcost", "Manage lane cost in cents"), - ("@tolldist", "Toll distance"), - ("@h2tolldist", "Toll distance for hov2"), - ("@h3tolldist", "Toll distance for hov3"), - ("@reliability", "Reliability factor"), - ("@reliability_sq", "Reliability factor squared"), - ("@auto_volume", "traffic link volume (volau)"), - ("@auto_time", "traffic link time (timau)"), - ] - for name, description in link_attributes: - create_attribute("LINK", name, description, - 0, overwrite=True, scenario=scenario) - create_attribute("TURN", "@auto_time_turn", "traffic turn time (ptimau)", - overwrite=True, scenario=scenario) - - net_calc("@hovdist", "length", {"link": "@lane_restriction=2,3"}) - net_calc("@tollcost", "@cost_auto_%s - @cost_operating" % p) - net_calc("@h2tollcost", "@cost_hov2_%s - @cost_operating" % p, {"link": "@lane_restriction=3,4"}) - net_calc("@h3tollcost", "@cost_hov3_%s - @cost_operating" % p, {"link": "@lane_restriction=4"}) - net_calc("@trk_ltollcost", "@cost_lgt_truck_%s - @cost_operating" % p) - net_calc("@trk_mtollcost", "@cost_med_truck_%s - @cost_operating" % p) - net_calc("@trk_htollcost", "@cost_hvy_truck_%s - @cost_operating" % p) - net_calc("@mlcost", "@toll_%s" % p, {"link": "not @lane_restriction=4"}) - net_calc("@tolldist", "length", {"link": "@lane_restriction=2,4"}) - net_calc("@h2tolldist", "length", {"link": "@lane_restriction=3,4"}) - net_calc("@h3tolldist", "length", {"link": "@lane_restriction=4"}) - net_calc("@auto_volume", "volau", {"link": "modes=d"}) - net_calc("ul2", "volau+volad", {"link": "modes=d"}) - vdfs = [f for f in emmebank.functions() if f.type == "VOLUME_DELAY"] - exf_pars = emmebank.extra_function_parameters - for function in vdfs: - expression = function.expression - for exf_par in ["el1", "el2", "el3"]: - expression = expression.replace(exf_par, getattr(exf_pars, exf_par)) - # split function into time component and reliability component - time_expr, reliability_expr = expression.split("*(1+@sta_reliability+") - net_calc("@auto_time", time_expr, {"link": "vdf=%s" % function.id[2:]}) - net_calc("@reliability", "(@sta_reliability+" + reliability_expr, - {"link": "vdf=%s" % function.id[2:]}) - - net_calc("@reliability_sq", "@reliability**2", {"link": "modes=d"}) - net_calc("@auto_time_turn", "ptimau*(ptimau.gt.0)", - {"incoming_link": "all", "outgoing_link": "all"}) - - def run_skims(self, period, num_processors, scenario, classes): - modeller = _m.Modeller() - traffic_assign = modeller.tool( - "inro.emme.traffic_assignment.sola_traffic_assignment") - emmebank = scenario.emmebank - p = period.lower() - analysis_link = { - "TIME": "@auto_time", - "DIST": "length", - "HOVDIST": "@hovdist", - "TOLLCOST.SOV": "@tollcost", - "TOLLCOST.HOV2": "@h2tollcost", - "TOLLCOST.HOV3": "@h3tollcost", - "TOLLCOST.TRK_L": "@trk_ltollcost", - "TOLLCOST.TRK_M": "@trk_mtollcost", - "TOLLCOST.TRK_H": "@trk_htollcost", - "MLCOST": "@mlcost", - "TOLLDIST": "@tolldist", - "TOLLDIST.HOV2": "@h2tolldist", - "TOLLDIST.HOV3": "@h3tolldist", - "REL": "@reliability_sq" - } - analysis_turn = {"TIME": "@auto_time_turn"} - with self.setup_skims(period, scenario): - if period == "MD": - gen_truck_mode = 'D' - classes.append({ - "name": 'TRK', "mode": gen_truck_mode, "PCE": 1, "VOT": 67., "cost": '', - "skims": ["TIME"] - }) - skim_spec = self.base_assignment_spec(0, 0, num_processors, background_traffic=False) - for traffic_class in classes: - if not traffic_class["skims"]: - continue - class_analysis = [] - if "GENCOST" in traffic_class["skims"]: - od_travel_times = 'mf"%s_%s_%s"' % (period, traffic_class["name"], "GENCOST") - traffic_class["skims"].remove("GENCOST") - else: - od_travel_times = None - for skim_type in traffic_class["skims"]: - skim_name = skim_type.split(".")[0] - class_analysis.append({ - "link_component": analysis_link.get(skim_type), - "turn_component": analysis_turn.get(skim_type), - "operator": "+", - "selection_threshold": {"lower": None, "upper": None}, - "path_to_od_composition": { - "considered_paths": "ALL", - "multiply_path_proportions_by": - {"analyzed_demand": False, "path_value": True} - }, - "results": { - "od_values": 'mf"%s_%s_%s"' % (period, traffic_class["name"], skim_name), - "selected_link_volumes": None, - "selected_turn_volumes": None - } - }) - if traffic_class["cost"]: - link_cost = "%s_%s" % (traffic_class["cost"], p) - else: - link_cost = "@cost_operating" - skim_spec["classes"].append({ - "mode": traffic_class["mode"], - "demand": 'ms"zero"', # 0 demand for skim with 0 iteration and fix flow in ul2 in vdf - "generalized_cost": { - "link_costs": link_cost, "perception_factor": 1.0 / traffic_class["VOT"] - }, - "results": { - "link_volumes": None, "turn_volumes": None, - "od_travel_times": {"shortest_paths": od_travel_times} - }, - "path_analyses": class_analysis, - }) - - # skim assignment - if self._skim_classes_separately: - # Debugging check - skim_classes = skim_spec["classes"][:] - for kls in skim_classes: - skim_spec["classes"] = [kls] - traffic_assign(skim_spec, scenario) - else: - traffic_assign(skim_spec, scenario) - - # compute diagonal value for TIME and DIST - with _m.logbook_trace("Compute diagonal values for period %s" % period): - num_cells = len(scenario.zone_numbers) ** 2 - for traffic_class in classes: - class_name = traffic_class["name"] - skims = traffic_class["skims"] - with _m.logbook_trace("Class %s" % class_name): - for skim_type in skims: - skim_name = skim_type.split(".")[0] - name = '%s_%s_%s' % (period, class_name, skim_name) - matrix = emmebank.matrix(name) - data = matrix.get_numpy_data(scenario) - if skim_name == "TIME" or skim_name == "DIST": - numpy.fill_diagonal(data, 999999999.0) - data[numpy.diag_indices_from(data)] = 0.5 * numpy.nanmin(data[::,12::], 1) - internal_data = data[12::, 12::] # Exclude the first 12 zones, external zones - self._stats[name] = (name, internal_data.min(), internal_data.max(), internal_data.mean(), internal_data.sum(), 0) - elif skim_name == "REL": - data = numpy.sqrt(data) - else: - self._stats[name] = (name, data.min(), data.max(), data.mean(), data.sum(), 0) - numpy.fill_diagonal(data, 0.0) - matrix.set_numpy_data(data, scenario) - return - - def base_assignment_spec(self, relative_gap, max_iterations, num_processors, background_traffic=True): - base_spec = { - "type": "SOLA_TRAFFIC_ASSIGNMENT", - "background_traffic": None, - "classes": [], - "stopping_criteria": { - "max_iterations": int(max_iterations), "best_relative_gap": 0.0, - "relative_gap": float(relative_gap), "normalized_gap": 0.0 - }, - "performance_settings": {"number_of_processors": num_processors}, - } - if background_traffic: - base_spec["background_traffic"] = { - "link_component": "ul2", # ul2 = transit flow of the period - "turn_component": None, - "add_transit_vehicles": False - } - return base_spec - - @_context - def setup_skims(self, period, scenario): - emmebank = scenario.emmebank - with _m.logbook_trace("Extract skims for period %s" % period): - # temp_functions converts to skim-type VDFs - with temp_functions(emmebank): - backup_attributes = {"LINK": ["data2", "auto_volume", "auto_time", "additional_volume"]} - with gen_utils.backup_and_restore(scenario, backup_attributes): - yield - - def prepare_midday_generic_truck(self, scenario): - modeller = _m.Modeller() - create_mode = modeller.tool( - "inro.emme.data.network.mode.create_mode") - delete_mode = modeller.tool( - "inro.emme.data.network.mode.delete_mode") - change_link_modes = modeller.tool( - "inro.emme.data.network.base.change_link_modes") - with _m.logbook_trace("Preparation for generic truck skim"): - gen_truck_mode = 'D' - truck_mode = scenario.mode(gen_truck_mode) - if not truck_mode: - truck_mode = create_mode( - mode_type="AUX_AUTO", mode_id=gen_truck_mode, - mode_description="all trucks", scenario=scenario) - change_link_modes(modes=[truck_mode], action="ADD", - selection="modes=vVmMtT", scenario=scenario) - - #added by RSG (nagendra.dhakar@rsginc.com) for collapsed assignment classes testing - #this adds non-transponder SOV mode to SR-125 links - # TODO: move this to the network_import step for consistency and foward-compatibility - def change_mode_sovntp(self, scenario): - modeller = _m.Modeller() - change_link_modes = modeller.tool( - "inro.emme.data.network.base.change_link_modes") - with _m.logbook_trace("Preparation for sov ntp assignment"): - gen_sov_mode = 's' - sov_mode = scenario.mode(gen_sov_mode) - change_link_modes(modes=[sov_mode], action="ADD", - selection="@lane_restriction=4", scenario=scenario) - - def report(self, period, scenario, classes): - emmebank = scenario.emmebank - text = ['
    '] - matrices = [] - for traffic_class in classes: - matrices.extend(["%s_%s" % (traffic_class["name"], s.split(".")[0]) for s in traffic_class["skims"]]) - num_zones = len(scenario.zone_numbers) - num_cells = num_zones ** 2 - text.append(""" - Number of zones: %s. Number of O-D pairs: %s. - Values outside -9999999, 9999999 are masked in summaries.
    """ % (num_zones, num_cells)) - text.append("%-25s %9s %9s %9s %13s %9s" % ("name", "min", "max", "mean", "sum", "mask num")) - for name in matrices: - name = period + "_" + name - matrix = emmebank.matrix(name) - stats = self._stats.get(name) - if stats is None: - data = matrix.get_numpy_data(scenario) - data = numpy.ma.masked_outside(data, -9999999, 9999999, copy=False) - stats = (name, data.min(), data.max(), data.mean(), data.sum(), num_cells-data.count()) - text.append("%-25s %9.4g %9.4g %9.4g %13.7g %9d" % stats) - text.append("
    ") - title = 'Traffic impedance summary for period %s' % period - report = _m.PageBuilder(title) - report.wrap_html('Matrix details', "
    ".join(text)) - _m.logbook_write(title, report.render()) - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg - - @_m.method(return_type=unicode) - def get_link_attributes(self): - export_utils = _m.Modeller().module("inro.emme.utility.export_utilities") - return export_utils.get_link_attributes(_m.Modeller().scenario) - - -@_context -def temp_functions(emmebank): - change_function = _m.Modeller().tool( - "inro.emme.data.function.change_function") - orig_expression = {} - with _m.logbook_trace("Set functions to skim parameter"): - for func in emmebank.functions(): - if func.prefix=="fd": - exp = func.expression - orig_expression[func] = exp - if "volau+volad" in exp: - exp = exp.replace("volau+volad", "ul2") - change_function(func, exp, emmebank) - try: - yield - finally: - with _m.logbook_trace("Reset functions to assignment parameters"): - for func, expression in orig_expression.iteritems(): - change_function(func, expression, emmebank) - diff --git a/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py b/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py deleted file mode 100644 index cd23ba6..0000000 --- a/sandag_abm/src/main/emme/toolbox/assignment/transit_assignment.py +++ /dev/null @@ -1,785 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// transit_assignment.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# The Transit assignment tool runs the transit assignment and skims for each -# period on the current primary scenario. -# -# The Build transit network tool must be run first to prepare the scenario for -# assignment. Note that this tool must be run with the Transit database -# (under the Database_transit directory) open (as the active database in the -# Emme desktop). -# -# -# Inputs: -# period: the time-of-day period, one of EA, AM, MD, PM, EV. -# scenario: Transit assignment scenario -# skims_only: Only run assignment for skim matrices, if True only two assignments -# are run to generate the skim matrices for the BUS and ALL skim classes. -# Otherwise, all 15 assignments are run to generate the total network flows. -# num_processors: number of processors to use for the traffic assignments. -# -# Matrices: -# All transit demand and skim matrices. -# See list of matrices under report method. -# -# Script example: -""" -import inro.modeller as _m -import os -modeller = _m.Modeller() -desktop = modeller.desktop - -build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") -transit_assign = modeller.tool("sandag.assignment.transit_assignment") -load_properties = modeller.tool('sandag.utilities.properties') - -project_dir = os.path.dirname(desktop.project_path()) -main_directory = os.path.dirname(project_dir) -props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) -main_emmebank = os.path.join(project_dir, "Database", "emmebank") -scenario_id = 100 -base_scenario = main_emmebank.scenario(scenario_id) - -transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") - -periods = ["EA", "AM", "MD", "PM", "EV"] -period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) -num_processors = "MAX-1" -scenarioYear = str(props["scenarioYear"]) - -for number, period in period_ids: - src_period_scenario = main_emmebank.scenario(number) - transit_assign_scen = build_transit_scen( - period=period, base_scenario=src_period_scenario, - transit_emmebank=transit_emmebank, - scenario_id=src_period_scenario.id, - scenario_title="%s %s transit assign" % (base_scenario.title, period), - data_table_name=scenarioYear, overwrite=True) - transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, - skims_only=True, num_processors=num_processors) - -omx_file = os.path.join(output_dir, "transit_skims.omx") -export_transit_skims(omx_file, periods, transit_scenario) -""" - - -TOOLBOX_ORDER = 21 - - -import inro.modeller as _m -import inro.emme.core.exception as _except -import traceback as _traceback -from copy import deepcopy as _copy -from collections import defaultdict as _defaultdict, OrderedDict -import contextlib as _context -import numpy - -import os -import sys -import math - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class TransitAssignment(_m.Tool(), gen_utils.Snapshot): - - period = _m.Attribute(unicode) - scenario = _m.Attribute(_m.InstanceType) - data_table_name = _m.Attribute(unicode) - assignment_only = _m.Attribute(bool) - skims_only = _m.Attribute(bool) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.assignment_only = False - self.skims_only = False - self.scenario = _m.Modeller().scenario - self.num_processors = "MAX-1" - self.attributes = [ - "period", "scenario", "data_table_name", "assignment_only", "skims_only", "num_processors"] - self._dt_db = _m.Modeller().desktop.project.data_tables() - self._matrix_cache = {} # used to hold data for reporting and post-processing of skims - - def from_snapshot(self, snapshot): - super(TransitAssignment, self).from_snapshot(snapshot) - # custom from_snapshot to load scenario and database objects - self.scenario = _m.Modeller().emmebank.scenario(self.scenario) - return self - - def page(self): - if not self.data_table_name: - load_properties = _m.Modeller().tool('sandag.utilities.properties') - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_directory = os.path.dirname(project_dir) - props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - self.data_table_name = props["scenarioYear"] - - pb = _m.ToolPageBuilder(self) - pb.title = "Transit assignment" - pb.description = """Assign transit demand for the selected time period.""" - pb.branding_text = "- SANDAG - " - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - options = [("EA", "Early AM"), - ("AM", "AM peak"), - ("MD", "Mid-day"), - ("PM", "PM peak"), - ("EV", "Evening")] - pb.add_select("period", options, title="Period:") - pb.add_select_scenario("scenario", - title="Transit assignment scenario:") - pb.add_text_box("data_table_name", title="Data table prefix name:", note="Default is the ScenarioYear") - pb.add_checkbox("assignment_only", title=" ", label="Only assign trips (no skims)") - pb.add_checkbox("skims_only", title=" ", label="Only run assignments relevant for skims") - dem_utils.add_select_processors("num_processors", pb, self) - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - results = self( - self.period, self.scenario, self.data_table_name, - self.assignment_only, self.skims_only, self.num_processors) - run_msg = "Transit assignment completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, period, scenario, data_table_name, assignment_only=False, skims_only=False, - num_processors="MAX-1"): - attrs = { - "period": period, - "scenario": scenario.id, - "data_table_name": data_table_name, - "assignment_only": assignment_only, - "skims_only": skims_only, - "num_processors": num_processors, - "self": str(self) - } - self.scenario = scenario - if not scenario.has_traffic_results: - raise Exception("missing traffic assignment results for period %s scenario %s" % (period, scenario)) - emmebank = scenario.emmebank - with self.setup(attrs): - gen_utils.log_snapshot("Transit assignment", str(self), attrs) - periods = ["EA", "AM", "MD", "PM", "EV"] - if not period in periods: - raise Exception('period: unknown value - specify one of %s' % periods) - num_processors = dem_utils.parse_num_processors(num_processors) - params = self.get_perception_parameters(period) - network = scenario.get_partial_network( - element_types=["TRANSIT_LINE"], include_attributes=True) - coaster_mode = network.mode("c") - params["coaster_fare_percep"] = 0 - for line in list(network.transit_lines()): - # get the coaster fare perception for use in journey levels - if line.mode == coaster_mode: - params["coaster_fare_percep"] = line[params["fare"]] - break - - transit_passes = gen_utils.DataTableProc("%s_transit_passes" % data_table_name) - transit_passes = {row["pass_type"]: row["cost"] for row in transit_passes} - day_pass = float(transit_passes["day_pass"]) / 2.0 - regional_pass = float(transit_passes["regional_pass"]) / 2.0 - - self.run_assignment(period, params, network, day_pass, skims_only, num_processors) - - if not assignment_only: - # max_fare = day_pass for local bus and regional_pass for premium modes - self.run_skims("BUS", period, params, day_pass, num_processors, network) - self.run_skims("PREM", period, params, regional_pass, num_processors, network) - self.run_skims("ALLPEN", period, params, regional_pass, num_processors, network) - self.mask_allpen(period) - self.report(period) - - @_context.contextmanager - def setup(self, attrs): - self._matrix_cache = {} # initialize cache at beginning of run - emmebank = self.scenario.emmebank - period = attrs["period"] - with _m.logbook_trace("Transit assignment for period %s" % period, attributes=attrs): - with gen_utils.temp_matrices(emmebank, "FULL", 3) as matrices: - matrices[0].name = "TEMP_IN_VEHICLE_COST" - matrices[1].name = "TEMP_LAYOVER_BOARD" - matrices[2].name = "TEMP_PERCEIVED_FARE" - try: - yield - finally: - self._matrix_cache = {} # clear cache at end of run - - def get_perception_parameters(self, period): - perception_parameters = { - "EA": { - "vot": 0.27, - "init_wait": 1.5, - "xfer_wait": 3.0, - "walk": 2.0, - "init_headway": "@headway_rev_op", - "xfer_headway": "@headway_op", - "fare": "@fare_per_op", - "in_vehicle": "@vehicle_per_op", - "fixed_link_time": "@trtime_link_ea" - }, - "AM": { - "vot": 0.27, - "init_wait": 1.5, - "xfer_wait": 3.0, - "walk": 2.0, - "init_headway": "@headway_rev_am", - "xfer_headway": "@headway_am", - "fare": "@fare_per_pk", - "in_vehicle": "@vehicle_per_pk", - "fixed_link_time": "@trtime_link_am" - }, - "MD": { - "vot": 0.27, - "init_wait": 1.5, - "xfer_wait": 3.0, - "walk": 2.0, - "init_headway": "@headway_rev_op", - "xfer_headway": "@headway_op", - "fare": "@fare_per_op", - "in_vehicle": "@vehicle_per_op", - "fixed_link_time": "@trtime_link_md" - }, - "PM": { - "vot": 0.27, - "init_wait": 1.5, - "xfer_wait": 3.0, - "walk": 2.0, - "init_headway": "@headway_rev_pm", - "xfer_headway": "@headway_pm", - "fare": "@fare_per_pk", - "in_vehicle": "@vehicle_per_pk", - "fixed_link_time": "@trtime_link_pm" - }, - "EV": { - "vot": 0.27, - "init_wait": 1.5, - "xfer_wait": 3.0, - "walk": 2.0, - "init_headway": "@headway_rev_op", - "xfer_headway": "@headway_op", - "fare": "@fare_per_op", - "in_vehicle": "@vehicle_per_op", - "fixed_link_time": "@trtime_link_ev" - } - } - return perception_parameters[period] - - def group_modes_by_fare(self, network, day_pass_cost): - # Identify all the unique boarding fare values - fare_set = {mode.id: _defaultdict(lambda:0) - for mode in network.modes() - if mode.type == "TRANSIT"} - for line in network.transit_lines(): - fare_set[line.mode.id][line["@fare"]] += 1 - del fare_set['c'] # remove coaster mode, this fare is handled separately - # group the modes relative to day_pass - mode_groups = { - "bus": [], # have a bus fare, less than 1/2 day pass - "day_pass": [], # boarding fare is the same as 1/2 day pass - "premium": [] # special premium services not covered by day pass - } - for mode_id, fares in fare_set.items(): - try: - max_fare = max(fares.keys()) - except ValueError: - continue # an empty set means this mode is unused in this period - if numpy.isclose(max_fare, day_pass_cost, rtol=0.0001): - mode_groups["day_pass"].append((mode_id, fares)) - elif max_fare < day_pass_cost: - mode_groups["bus"].append((mode_id, fares)) - else: - mode_groups["premium"].append((mode_id, fares)) - return mode_groups - - def all_modes_journey_levels(self, params, network, day_pass_cost): - transfer_penalty = {"on_segments": {"penalty": "@transfer_penalty_s", "perception_factor": 5.0}} - transfer_wait = { - "effective_headways": "@headway_seg", - "headway_fraction": 0.5, - "perception_factor": params["xfer_wait"], - "spread_factor": 1.0 - } - mode_groups = self.group_modes_by_fare(network, day_pass_cost) - - def get_transition_rules(next_level): - rules = [] - for name, group in mode_groups.items(): - for mode_id, fares in group: - rules.append({"mode": mode_id, "next_journey_level": next_level[name]}) - rules.append({"mode": "c", "next_journey_level": next_level["coaster"]}) - return rules - - journey_levels = [ - { - "description": "base", - "destinations_reachable": False, - "transition_rules": get_transition_rules({"bus": 1, "day_pass": 2, "premium": 3, "coaster": 4}), - "boarding_time": {"global": {"penalty": 0, "perception_factor": 1}}, - "waiting_time": { - "effective_headways": params["init_headway"], "headway_fraction": 0.5, - "perception_factor": params["init_wait"], "spread_factor": 1.0 - }, - "boarding_cost": { - "on_lines": {"penalty": "@fare", "perception_factor": params["fare"]}, - "on_segments": {"penalty": "@coaster_fare_board", "perception_factor": params["coaster_fare_percep"]}, - }, - }, - { - "description": "boarded_bus", - "destinations_reachable": True, - "transition_rules": get_transition_rules({"bus": 2, "day_pass": 2, "premium": 5, "coaster": 5}), - "boarding_time": transfer_penalty, - "waiting_time": transfer_wait, - "boarding_cost": { - # xfer from bus fare is on segments so circle lines get free transfer - "on_segments": {"penalty": "@xfer_from_bus", "perception_factor": params["fare"]}, - }, - }, - { - "description": "day_pass", - "destinations_reachable": True, - "transition_rules": get_transition_rules({"bus": 2, "day_pass": 2, "premium": 5, "coaster": 5}), - "boarding_time": transfer_penalty, - "waiting_time": transfer_wait, - "boarding_cost": { - "on_lines": {"penalty": "@xfer_from_day", "perception_factor": params["fare"]}, - }, - }, - { - "description": "boarded_premium", - "destinations_reachable": True, - "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), - "boarding_time": transfer_penalty, - "waiting_time": transfer_wait, - "boarding_cost": { - "on_lines": {"penalty": "@xfer_from_premium", "perception_factor": params["fare"]}, - }, - }, - { - "description": "boarded_coaster", - "destinations_reachable": True, - "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), - "boarding_time": transfer_penalty, - "waiting_time": transfer_wait, - "boarding_cost": { - "on_lines": {"penalty": "@xfer_from_coaster", "perception_factor": params["fare"]}, - }, - }, - { - "description": "regional_pass", - "destinations_reachable": True, - "transition_rules": get_transition_rules({"bus": 5, "day_pass": 5, "premium": 5, "coaster": 5}), - "boarding_time": transfer_penalty, - "waiting_time": transfer_wait, - "boarding_cost": { - "on_lines": {"penalty": "@xfer_regional_pass", "perception_factor": params["fare"]}, - }, - } - ] - return journey_levels - - def filter_journey_levels_by_mode(self, modes, journey_levels): - # remove rules for unused modes from provided journey_levels - # (restrict to provided modes) - journey_levels = _copy(journey_levels) - for level in journey_levels: - rules = level["transition_rules"] - rules = [r for r in rules if r["mode"] in modes] - level["transition_rules"] = rules - # count level transition rules references to find unused levels - num_levels = len(journey_levels) - level_count = [0] * len(journey_levels) - - def follow_rule(next_level): - level_count[next_level] += 1 - if level_count[next_level] > 1: - return - for rule in journey_levels[next_level]["transition_rules"]: - follow_rule(rule["next_journey_level"]) - - follow_rule(0) - # remove unreachable levels - # and find new index for transition rules for remaining levels - level_map = {i:i for i in range(num_levels)} - for level_id, count in reversed(list(enumerate(level_count))): - if count == 0: - for index in range(level_id, num_levels): - level_map[index] -= 1 - del journey_levels[level_id] - # re-index remaining journey_levels - for level in journey_levels: - for rule in level["transition_rules"]: - next_level = rule["next_journey_level"] - rule["next_journey_level"] = level_map[next_level] - return journey_levels - - @_m.logbook_trace("Transit assignment by demand set", save_arguments=True) - def run_assignment(self, period, params, network, day_pass_cost, skims_only, num_processors): - modeller = _m.Modeller() - scenario = self.scenario - emmebank = scenario.emmebank - assign_transit = modeller.tool( - "inro.emme.transit_assignment.extended_transit_assignment") - - walk_modes = ["a", "w", "x"] - local_bus_mode = ["b"] - premium_modes = ["c", "l", "e", "p", "r", "y", "o"] - - # get the generic all-modes journey levels table - journey_levels = self.all_modes_journey_levels(params, network, day_pass_cost) - local_bus_journey_levels = self.filter_journey_levels_by_mode(local_bus_mode, journey_levels) - premium_modes_journey_levels = self.filter_journey_levels_by_mode(premium_modes, journey_levels) - # All modes transfer penalty assignment uses penalty of 15 minutes - for level in journey_levels[1:]: - level["boarding_time"] = {"global": {"penalty": 15, "perception_factor": 1}} - - base_spec = { - "type": "EXTENDED_TRANSIT_ASSIGNMENT", - "modes": [], - "demand": "", - "waiting_time": { - "effective_headways": params["init_headway"], "headway_fraction": 0.5, - "perception_factor": params["init_wait"], "spread_factor": 1.0 - }, - # Fare attributes - "boarding_cost": {"global": {"penalty": 0, "perception_factor": 1}}, - "boarding_time": {"global": {"penalty": 0, "perception_factor": 1}}, - "in_vehicle_cost": {"penalty": "@coaster_fare_inveh", - "perception_factor": params["coaster_fare_percep"]}, - "in_vehicle_time": {"perception_factor": params["in_vehicle"]}, - "aux_transit_time": {"perception_factor": params["walk"]}, - "aux_transit_cost": None, - "journey_levels": [], - "flow_distribution_between_lines": {"consider_total_impedance": False}, - "flow_distribution_at_origins": { - "fixed_proportions_on_connectors": None, - "choices_at_origins": "OPTIMAL_STRATEGY" - }, - "flow_distribution_at_regular_nodes_with_aux_transit_choices": { - "choices_at_regular_nodes": "OPTIMAL_STRATEGY" - }, - #"circular_lines": { - # "stay": True - #}, - "connector_to_connector_path_prohibition": None, - "od_results": {"total_impedance": None}, - "performance_settings": {"number_of_processors": num_processors} - } - - skim_parameters = OrderedDict([ - ("BUS", { - "modes": walk_modes + local_bus_mode, - "journey_levels": local_bus_journey_levels - }), - ("PREM", { - "modes": walk_modes + premium_modes, - "journey_levels": premium_modes_journey_levels - }), - ("ALLPEN", { - "modes": walk_modes + local_bus_mode + premium_modes, - "journey_levels": journey_levels - }), - ]) - - if skims_only: - access_modes = ["WLK"] - else: - access_modes = ["WLK", "PNR", "KNR"] - add_volumes = False - for a_name in access_modes: - for mode_name, parameters in skim_parameters.iteritems(): - spec = _copy(base_spec) - name = "%s_%s%s" % (period, a_name, mode_name) - spec["modes"] = parameters["modes"] - spec["demand"] = 'mf"%s"' % name - spec["journey_levels"] = parameters["journey_levels"] - assign_transit(spec, class_name=name, add_volumes=add_volumes, scenario=self.scenario) - add_volumes = True - - @_m.logbook_trace("Extract skims", save_arguments=True) - def run_skims(self, name, period, params, max_fare, num_processors, network): - modeller = _m.Modeller() - scenario = self.scenario - emmebank = scenario.emmebank - matrix_calc = modeller.tool( - "inro.emme.matrix_calculation.matrix_calculator") - network_calc = modeller.tool( - "inro.emme.network_calculation.network_calculator") - matrix_results = modeller.tool( - "inro.emme.transit_assignment.extended.matrix_results") - path_analysis = modeller.tool( - "inro.emme.transit_assignment.extended.path_based_analysis") - strategy_analysis = modeller.tool( - "inro.emme.transit_assignment.extended.strategy_based_analysis") - - class_name = "%s_WLK%s" % (period, name) - skim_name = "%s_%s" % (period, name) - self.run_skims.logbook_cursor.write(name="Extract skims for %s, using assignment class %s" % (name, class_name)) - - with _m.logbook_trace("First and total wait time, number of boardings, fares, total walk time, in-vehicle time"): - # First and total wait time, number of boardings, fares, total walk time, in-vehicle time - spec = { - "type": "EXTENDED_TRANSIT_MATRIX_RESULTS", - "actual_first_waiting_times": 'mf"%s_FIRSTWAIT"' % skim_name, - "actual_total_waiting_times": 'mf"%s_TOTALWAIT"' % skim_name, - "total_impedance": 'mf"%s_GENCOST"' % skim_name, - "by_mode_subset": { - "modes": [mode.id for mode in network.modes() if mode.type == "TRANSIT" or mode.type == "AUX_TRANSIT"], - "avg_boardings": 'mf"%s_XFERS"' % skim_name, - "actual_in_vehicle_times": 'mf"%s_TOTALIVTT"' % skim_name, - "actual_in_vehicle_costs": 'mf"TEMP_IN_VEHICLE_COST"', - "actual_total_boarding_costs": 'mf"%s_FARE"' % skim_name, - "perceived_total_boarding_costs": 'mf"TEMP_PERCEIVED_FARE"', - "actual_aux_transit_times": 'mf"%s_TOTALWALK"' % skim_name, - }, - } - matrix_results(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - with _m.logbook_trace("Distance and in-vehicle time by mode"): - mode_combinations = [ - ("BUS", ["b"], ["IVTT", "DIST"]), - ("LRT", ["l"], ["IVTT", "DIST"]), - ("CMR", ["c"], ["IVTT", "DIST"]), - ("EXP", ["e", "p"], ["IVTT", "DIST"]), - ("BRT", ["r", "y"], ["DIST"]), - ("BRTRED", ["r"], ["IVTT"]), - ("BRTYEL", ["y"], ["IVTT"]), - ("TIER1", ["o"], ["IVTT", "DIST"]), - ] - for mode_name, modes, skim_types in mode_combinations: - dist = 'mf"%s_%sDIST"' % (skim_name, mode_name) if "DIST" in skim_types else None - ivtt = 'mf"%s_%sIVTT"' % (skim_name, mode_name) if "IVTT" in skim_types else None - spec = { - "type": "EXTENDED_TRANSIT_MATRIX_RESULTS", - "by_mode_subset": { - "modes": modes, - "distance": dist, - "actual_in_vehicle_times": ivtt, - }, - } - matrix_results(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - # Sum total distance - spec = { - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_TOTDIST"' % skim_name, - "expression": ('mf"{0}_BUSDIST" + mf"{0}_LRTDIST" + mf"{0}_CMRDIST"' - ' + mf"{0}_EXPDIST" + mf"{0}_BRTDIST" + mf"{0}_TIER1DIST"').format(skim_name), - } - matrix_calc(spec, scenario=scenario, num_processors=num_processors) - - # convert number of boardings to number of transfers - # and subtract transfers to the same line at layover points - with _m.logbook_trace("Number of transfers and total fare"): - spec = { - "trip_components": {"boarding": "@layover_board"}, - "sub_path_combination_operator": "+", - "sub_strategy_combination_operator": "average", - "selected_demand_and_transit_volumes": { - "sub_strategies_to_retain": "ALL", - "selection_threshold": {"lower": -999999, "upper": 999999} - }, - "results": { - "strategy_values": 'TEMP_LAYOVER_BOARD', - }, - "type": "EXTENDED_TRANSIT_STRATEGY_ANALYSIS" - } - strategy_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - spec = { - "type": "MATRIX_CALCULATION", - "constraint":{ - "by_value": { - "od_values": 'mf"%s_XFERS"' % skim_name, - "interval_min": 1, "interval_max": 9999999, - "condition": "INCLUDE"}, - }, - "result": 'mf"%s_XFERS"' % skim_name, - "expression": '(%s_XFERS - 1 - TEMP_LAYOVER_BOARD).max.0' % skim_name, - } - matrix_calc(spec, scenario=scenario, num_processors=num_processors) - - # sum in-vehicle cost and boarding cost to get the fare paid - spec = { - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_FARE"' % skim_name, - "expression": '(%s_FARE + TEMP_IN_VEHICLE_COST).min.%s' % (skim_name, max_fare), - } - matrix_calc(spec, scenario=scenario, num_processors=num_processors) - - # walk access time - get distance and convert to time with 3 miles / hr - with _m.logbook_trace("Walk time access, egress and xfer"): - path_spec = { - "portion_of_path": "ORIGIN_TO_INITIAL_BOARDING", - "trip_components": {"aux_transit": "length",}, - "path_operator": "+", - "path_selection_threshold": {"lower": 0, "upper": 999999 }, - "path_to_od_aggregation": { - "operator": "average", - "aggregated_path_values": 'mf"%s_ACCWALK"' % skim_name, - }, - "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" - } - path_analysis(path_spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - - # walk egress time - get distance and convert to time with 3 miles/ hr - path_spec = { - "portion_of_path": "FINAL_ALIGHTING_TO_DESTINATION", - "trip_components": {"aux_transit": "length",}, - "path_operator": "+", - "path_selection_threshold": {"lower": 0, "upper": 999999 }, - "path_to_od_aggregation": { - "operator": "average", - "aggregated_path_values": 'mf"%s_EGRWALK"' % skim_name - }, - "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" - } - path_analysis(path_spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - - spec_list = [ - { # walk access time - convert to time with 3 miles/ hr - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_ACCWALK"' % skim_name, - "expression": '60.0 * %s_ACCWALK / 3.0' % skim_name, - }, - { # walk egress time - convert to time with 3 miles/ hr - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_EGRWALK"' % skim_name, - "expression": '60.0 * %s_EGRWALK / 3.0' % skim_name, - }, - { # transfer walk time = total - access - egress - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_XFERWALK"' % skim_name, - "expression": '({name}_TOTALWALK - {name}_ACCWALK - {name}_EGRWALK).max.0'.format(name=skim_name), - }] - matrix_calc(spec_list, scenario=scenario, num_processors=num_processors) - - # transfer wait time - with _m.logbook_trace("Wait time - xfer"): - spec = { - "type": "MATRIX_CALCULATION", - "constraint":{ - "by_value": { - "od_values": 'mf"%s_TOTALWAIT"' % skim_name, - "interval_min": 0, "interval_max": 9999999, - "condition": "INCLUDE"}, - }, - "result": 'mf"%s_XFERWAIT"' % skim_name, - "expression": '({name}_TOTALWAIT - {name}_FIRSTWAIT).max.0'.format(name=skim_name), - } - matrix_calc(spec, scenario=scenario, num_processors=num_processors) - - with _m.logbook_trace("Calculate dwell time"): - with gen_utils.temp_attrs(scenario, "TRANSIT_SEGMENT", ["@dwt_for_analysis"]): - values = scenario.get_attribute_values("TRANSIT_SEGMENT", ["dwell_time"]) - scenario.set_attribute_values("TRANSIT_SEGMENT", ["@dwt_for_analysis"], values) - - spec = { - "trip_components": {"in_vehicle": "@dwt_for_analysis"}, - "sub_path_combination_operator": "+", - "sub_strategy_combination_operator": "average", - "selected_demand_and_transit_volumes": { - "sub_strategies_to_retain": "ALL", - "selection_threshold": {"lower": -999999, "upper": 999999} - }, - "results": { - "strategy_values": 'mf"%s_DWELLTIME"' % skim_name, - }, - "type": "EXTENDED_TRANSIT_STRATEGY_ANALYSIS" - } - strategy_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - - expr_params = _copy(params) - expr_params["xfers"] = 15.0 - expr_params["name"] = skim_name - spec = { - "type": "MATRIX_CALCULATION", - "constraint": None, - "result": 'mf"%s_GENCOST"' % skim_name, - "expression": ("{xfer_wait} * {name}_TOTALWAIT " - "- ({xfer_wait} - {init_wait}) * {name}_FIRSTWAIT " - "+ 1.0 * {name}_TOTALIVTT + 0.5 * {name}_BUSIVTT" - "+ (1 / {vot}) * (TEMP_PERCEIVED_FARE + {coaster_fare_percep} * TEMP_IN_VEHICLE_COST)" - "+ {xfers} *({name}_XFERS.max.0) " - "+ {walk} * {name}_TOTALWALK").format(**expr_params) - } - matrix_calc(spec, scenario=scenario, num_processors=num_processors) - return - - def mask_allpen(self, period): - # Reset skims to 0 if not both local and premium - skims = [ - "FIRSTWAIT", "TOTALWAIT", "DWELLTIME", "BUSIVTT", "XFERS", "TOTALWALK", - "LRTIVTT", "CMRIVTT", "EXPIVTT", "LTDEXPIVTT", "BRTREDIVTT", "BRTYELIVTT", "TIER1IVTT", - "GENCOST", "XFERWAIT", "FARE", - "ACCWALK", "XFERWALK", "EGRWALK", "TOTALIVTT", - "BUSDIST", "LRTDIST", "CMRDIST", "EXPDIST", "BRTDIST" , "TIER1DIST"] - localivt_skim = self.get_matrix_data(period + "_ALLPEN_BUSIVTT") - totalivt_skim = self.get_matrix_data(period + "_ALLPEN_TOTALIVTT") - has_premium = numpy.greater((totalivt_skim - localivt_skim), 0) - has_both = numpy.greater(localivt_skim, 0) * has_premium - for skim in skims: - mat_name = period + "_ALLPEN_" + skim - data = self.get_matrix_data(mat_name) - self.set_matrix_data(mat_name, data * has_both) - - def get_matrix_data(self, name): - data = self._matrix_cache.get(name) - if data is None: - matrix = self.scenario.emmebank.matrix(name) - data = matrix.get_numpy_data(self.scenario) - self._matrix_cache[name] = data - return data - - def set_matrix_data(self, name, data): - matrix = self.scenario.emmebank.matrix(name) - self._matrix_cache[name] = data - matrix.set_numpy_data(data, self.scenario) - - def report(self, period): - text = ['
    '] - init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") - matrices = init_matrices.get_matrix_names("transit_skims", [period], self.scenario) - num_zones = len(self.scenario.zone_numbers) - num_cells = num_zones ** 2 - text.append( - "Number of zones: %s. Number of O-D pairs: %s. " - "Values outside -9999999, 9999999 are masked in summaries.
    " % (num_zones, num_cells)) - text.append("%-25s %9s %9s %9s %13s %9s" % ("name", "min", "max", "mean", "sum", "mask num")) - for name in matrices: - data = self.get_matrix_data(name) - data = numpy.ma.masked_outside(data, -9999999, 9999999, copy=False) - stats = (name, data.min(), data.max(), data.mean(), data.sum(), num_cells-data.count()) - text.append("%-25s %9.4g %9.4g %9.4g %13.7g %9d" % stats) - text.append("
    ") - title = 'Transit impedance summary for period %s' % period - report = _m.PageBuilder(title) - report.wrap_html('Matrix details', "
    ".join(text)) - _m.logbook_write(title, report.render()) diff --git a/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py b/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py deleted file mode 100644 index 8f4b387..0000000 --- a/sandag_abm/src/main/emme/toolbox/assignment/transit_select_analysis.py +++ /dev/null @@ -1,217 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// transit_select_analysis.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# This tool runs select type network analysis on the results of one or more -# transit assignments. It is run as a post-process tool after the assignment -# tools are complete, using the saved transit strategies. Any number of -# analyses can be run without needing to rerun the assignments. -# -# -# Inputs: -# Trip components for selection: pick one or more extra attributes which -# identify the network elements of interest by trip component: -# in_vehicle -# aux_transit -# initial_boarding -# transfer_boarding -# transfer_alighting -# final_alighting -# Result suffix: the suffix to use in the naming of per-class result -# attributes and matrices, up to 6 characters. -# Threshold: the minimum number of elements which must be encountered -# for the path selection. -# Scenario: the scenario to analyse. -# -# -# Script example: -""" -import inro.modeller as _m -import os -modeller = _m.Modeller() -desktop = modeller.desktop - -select_link = modeller.tool("sandag.assignment.transit_select_link") - -project_dir = os.path.dirname(desktop.project_path()) -main_directory = os.path.dirname(project_dir) - -transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") - -periods = ["EA", "AM", "MD", "PM", "EV"] -period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) - -suffix = "LRT" -threshold = 1 -num_processors = "MAX-1" -selection = { - "in_vehicle": None, - "aux_transit": None, - "initial_boarding": "@selected_line", - "transfer_boarding": None, - "transfer_alighting": None, - "final_alighting": None, -} - -for number, period in period_ids: - scenario = transit_emmebank.scenario(number) - select_link(selection, suffix, threshold, scenario, num_processors) -""" - -TOOLBOX_ORDER = 25 - - -import inro.modeller as _m -import inro.emme.core.exception as _except -import traceback as _traceback - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class TransitSelectAnalysis(_m.Tool(), gen_utils.Snapshot): - - in_vehicle = _m.Attribute(_m.InstanceType) - aux_transit = _m.Attribute(_m.InstanceType) - initial_boarding = _m.Attribute(_m.InstanceType) - transfer_boarding = _m.Attribute(_m.InstanceType) - transfer_alighting = _m.Attribute(_m.InstanceType) - final_alighting = _m.Attribute(_m.InstanceType) - - suffix = _m.Attribute(str) - threshold = _m.Attribute(int) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - def __init__(self): - self.threshold = 1 - self.num_processors = "MAX-1" - self.attributes = [ - "in_vehicle", "aux_transit", "initial_boarding", "transfer_boarding", - "transfer_alighting", "final_alighting", "suffix", "threshold", - "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Transit select analysis" - pb.description = """ - Run select type of analysis (select link, select node, select line ...) on - the results of the transit assignment(s) using a path-based analysis. - Can be used after a transit assignment has been completed.""" - pb.branding_text = "- SANDAG - Assignment" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - with pb.section("Trip components for selection:"): - domains = ["LINK", "NODE", "TRANSIT_SEGMENT", "TRANSIT_LINE"] - pb.add_select_extra_attribute("in_vehicle", title="In-vehicle", filter=domains, allow_none=True) - pb.add_select_extra_attribute("aux_transit", title="Auxilary transit", filter=domains, allow_none=True) - pb.add_select_extra_attribute("initial_boarding", title="Initial boarding", filter=domains, allow_none=True) - pb.add_select_extra_attribute("transfer_boarding", title="Transfer boarding", filter=domains, allow_none=True) - pb.add_select_extra_attribute("transfer_alighting", title="Transfer alighting", filter=domains, allow_none=True) - pb.add_select_extra_attribute("final_alighting", title="Final alighting", filter=domains, allow_none=True) - - pb.add_text_box("suffix", title="Suffix for results (matrices and attributes):", size=6, - note="The suffix to use in the naming of per-class result attributes and matrices, up to 6 characters. " - "Should be unique (existing attributes / matrices will be overwritten).") - pb.add_text_box("threshold", title="Threshold for selection:", - note="The minimum number of links which must be encountered for the path selection. " - "The default value of 1 indicates an 'any' link selection.") - dem_utils.add_select_processors("num_processors", pb, self) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - selection = { - "in_vehicle": self.in_vehicle, - "aux_transit": self.aux_transit, - "initial_boarding": self.initial_boarding, - "transfer_boarding": self.transfer_boarding, - "transfer_alighting": self.transfer_alighting, - "final_alighting": self.final_alighting, - } - scenario = _m.Modeller().scenario - results = self(selection, self.suffix, self.threshold, scenario, self.num_processors) - run_msg = "Traffic assignment completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, selection, suffix, threshold, scenario, num_processors): - attrs = { - "selection": selection, - "suffix": suffix, - "threshold": threshold, - "scenario": scenario.id, - "num_processors": num_processors - } - with _m.logbook_trace("Transit select analysis %s" % suffix, attributes=attrs): - attrs.update(dict((k,v) for k,v in attrs["selection"].iteritems())) - gen_utils.log_snapshot("Transit select analysis", str(self), attrs) - - path_analysis = _m.Modeller().tool( - "inro.emme.transit_assignment.extended.path_based_analysis") - create_attribute = _m.Modeller().tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - - spec = { - "portion_of_path": "COMPLETE", - "trip_components": selection, - "path_operator": "+", - "path_selection_threshold": {"lower": threshold, "upper": 999999}, - "path_to_od_aggregation": None, - "constraint": None, - "analyzed_demand": None, - "results_from_retained_paths": None, - "path_to_od_statistics": None, - "path_details": None, - "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" - } - strategies = scenario.transit_strategies - classes = [x.name for x in strategies.strat_files()] - if not classes: - raise Exception("Results for multi-class transit assignment not available") - - for class_name in classes: - with _m.logbook_trace("Analysis for class %s" % class_name): - seldem_name = "SELDEM_%s_%s" % (class_name, suffix) - desc = "Selected demand for %s %s" % (class_name, suffix) - seldem = dem_utils.create_full_matrix(seldem_name, desc, scenario=scenario) - results_from_retained_paths = { - "paths_to_retain": "SELECTED", - "demand": seldem.named_id, - } - attributes = [ - ("transit_volumes", "TRANSIT_SEGMENT", "@seltr_%s_%s", "%s '%s' sel segment flow"), - ("aux_transit_volumes", "LINK", "@selax_%s_%s", "%s '%s' sel aux transit flow"), - ("total_boardings", "TRANSIT_SEGMENT", "@selbr_%s_%s", "%s '%s' sel boardings"), - ("total_alightings", "TRANSIT_SEGMENT", "@selal_%s_%s", "%s '%s' sel alightings"), - ] - mode_name = class_name.lower()[3:] - for key, domain, name, desc in attributes: - attr = create_attribute(domain, name % (mode_name, suffix), desc % (class_name, suffix), - 0, overwrite=True, scenario=scenario) - results_from_retained_paths[key] = attr.id - spec["results_from_retained_paths"] = results_from_retained_paths - path_analysis(spec, class_name=class_name, scenario=scenario, num_processors=num_processors) - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/build_toolbox.py b/sandag_abm/src/main/emme/toolbox/build_toolbox.py deleted file mode 100644 index 1a9d1b7..0000000 --- a/sandag_abm/src/main/emme/toolbox/build_toolbox.py +++ /dev/null @@ -1,411 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// build_toolbox.py /// -#//// /// -#//// Generates an mtbx (Emme Modeller Toolbox), based on the structure /// -#//// of the Python source tree. /// -#//// /// -#//// Usage: build_toolbox.py [-s source_folder] [-p toolbox_path] /// -#//// /// -#//// [-p toolbox_path]: Specifies the name of the MTBX file. /// -#//// If omitted,defaults to "sandag_toolbox.mtbx" /// -#//// [-s source_folder]: The location of the source code folder. /// -#//// If omitted, defaults to the working directory. /// -#//// [-l] [--link] Build the toolbox with references to the files /// -#//// Use with developing or debugging scripts, changes to the /// -#//// scripts can be used with a "Refresh" of the toolbox /// -#//// [-c] [--consolidate] Build the toolbox with copies of the /// -#//// scripts included inside the toolbox. /// -#//// Use to have a "frozen" version of the scripts with node /// -#//// changes available. /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Example: -# python "T:\projects\sr13\develop\emme_conversion\git\sandag_abm\ABM_EMME\src\main\emme\toolbox\build_toolbox.py" --link -# -p "T:\projects\sr14\abm2_test\abm_runs\14_2_0\2035D_Hyperloop\emme_project\Scripts\sandag_toolbox.mtbx" -# -s T:\projects\sr13\develop\emme_conversion\git\sandag_abm\ABM_EMME\src\main\emme\toolbox - - -import os -import re -from datetime import datetime -import subprocess -import sqlite3.dbapi2 as sqllib -import base64 -import pickle - - -def check_namespace(ns): - if not re.match("^[a-zA-Z][a-zA-Z0-9_]*$", ns): - raise Exception("Namespace '%s' is invalid" % ns) - - -def get_emme_version(): - emme_process = subprocess.Popen(['Emme', '-V'], stdout=subprocess.PIPE, stderr=subprocess.PIPE) - output = emme_process.communicate()[0] - return output.split(',')[0] - - -def usc_transform(value): - try: - return unicode(value) - except Exception: - return unicode(str(value), encoding="raw-unicode-escape") - - -class BaseNode(object): - def __init__(self, namespace, title): - check_namespace(namespace) - self.namespace = namespace - self.title = title - self.element_id = None - self.parent = None - self.root = None - self.children = [] - - def add_folder(self, namespace): - node = FolderNode(namespace, parent=self) - self.children.append(node) - return node - - def add_tool(self, script_path, namespace): - try: - node = ToolNode(namespace, script_path, parent=self) - self.children.append(node) - with open(script_path, 'r') as f: - for line in f: - if line.startswith("TOOLBOX_ORDER"): - node.order = int(line.split("=")[1]) - if line.startswith("TOOLBOX_TITLE"): - title = line.split("=")[1].strip() - node.title = title[1:-1] # exclude first and last quotes - except Exception, e: - print script_path, namespace - print type(e), str(e) - return None - return node - - def consolidate(self): - for child in self.children: - child.consolidate() - - def set_toolbox_order(self): - self.element_id = self.root.next_id() - self.children.sort(key=lambda x: x.order) - for child in self.children: - child.set_toolbox_order() - - -class ElementTree(BaseNode): - - def __init__(self, namespace, title): - super(ElementTree, self).__init__(namespace, title) - self.next_element_id = 0 - self.begin = str(datetime.now()) - self.version = "Emme %s" % get_emme_version() - self.root = self - - def next_id(self): - self.next_element_id += 1 - return self.next_element_id - - -class FolderNode(BaseNode): - - def __init__(self, namespace, parent): - title = namespace.replace("_", " ").capitalize() - super(FolderNode, self).__init__(namespace, title) - self.parent = parent - self.root = parent.root - self.element_id = None - - @property - def order(self): - child_order = [child.order for child in self.children if child.order is not None] - if child_order: - return min(child_order) - return None - - -class ToolNode(): - - def __init__(self, namespace, script_path, parent): - check_namespace(namespace) - self.namespace = namespace - self.title = namespace.replace("_", " ").capitalize() - - self.root = parent.root - self.parent = parent - self.element_id = None - self.order = None - - self.script = script_path - self.extension = '.py' - self.code = '' - - def consolidate(self): - with open(self.script, 'r') as f: - code = f.read() - self.code = usc_transform(base64.b64encode(pickle.dumps(code))) - self.script = '' - - def set_toolbox_order(self): - self.element_id = self.root.next_id() - -class MTBXDatabase(): - FORMAT_MAGIC_NUMBER = 'B8C224F6_7C94_4E6F_8C2C_5CC06F145271' - TOOLBOX_MAGIC_NUMBER = 'TOOLBOX_C6809332_CD61_45B3_9060_411D825669F8' - CATEGORY_MAGIC_NUMBER = 'CATEGORY_984876A0_3350_4374_B47C_6D9C5A47BBC8' - TOOL_MAGIC_NUMBER = 'TOOL_1AC06B56_6A54_431A_9515_0BF77013646F' - - def __init__(self, filepath, title): - if os.path.exists(filepath): - os.remove(filepath) - - self.db = sqllib.connect(filepath) - - self._create_attribute_table() - self._create_element_table() - self._create_document_table() - self._create_triggers() - - self._initialize_documents_table(title) - - def _create_attribute_table(self): - sql = """CREATE TABLE attributes( - element_id INTEGER REFERENCES elements(element_id), - name VARCHAR, - value VARCHAR, - PRIMARY KEY(element_id, name));""" - - self.db.execute(sql) - - def _create_element_table(self): - sql = """CREATE TABLE elements( - element_id INTEGER PRIMARY KEY AUTOINCREMENT, - parent_id INTEGER REFERENCES elements(element_id), - document_id INTEGER REFERENCES documents(document_id), - tag VARCHAR, - text VARCHAR, - tail VARCHAR);""" - - self.db.execute(sql) - - def _create_document_table(self): - sql = """CREATE TABLE documents( - document_id INTEGER PRIMARY KEY AUTOINCREMENT, - title VARCHAR);""" - - self.db.execute(sql) - - def _create_triggers(self): - sql = """CREATE TRIGGER documents_delete - BEFORE DELETE on documents - FOR EACH ROW BEGIN - DELETE FROM elements WHERE document_id = OLD.document_id; - END""" - - self.db.execute(sql) - - sql = """CREATE TRIGGER elements_delete - BEFORE DELETE on elements - FOR EACH ROW BEGIN - DELETE FROM attributes WHERE element_id = OLD.element_id; - END""" - - self.db.execute(sql) - - def _initialize_documents_table(self, title): - sql = """INSERT INTO documents (document_id, title) - VALUES (1, '%s');""" % title - - self.db.execute(sql) - self.db.commit() - - def populate_tables_from_tree(self, tree): - - #Insert into the elements table - column_string = "element_id, document_id, tag, text, tail" - value_string = "{id}, 1, '{title}', '', ''".format( - id=tree.element_id, title=tree.title) - sql = """INSERT INTO elements (%s) - VALUES (%s);""" % (column_string, value_string) - self.db.execute(sql) - - #Insert into the attributes table - column_string = "element_id, name, value" - atts = {'major': '', - 'format': MTBXDatabase.FORMAT_MAGIC_NUMBER, - 'begin': tree.begin, - 'version': tree.version, - 'maintenance': '', - 'minor': '', - 'name': tree.title, - 'description': '', - 'namespace': tree.namespace, - MTBXDatabase.TOOLBOX_MAGIC_NUMBER: 'True'} - for key, val in atts.iteritems(): - value_string = "{id}, '{name}', '{value}'".format( - id=tree.element_id, name=key, value=val) - sql = """INSERT INTO attributes (%s) - VALUES (%s);""" % (column_string, value_string) - self.db.execute(sql) - - self.db.commit() - - #Handle children nodes - for child in tree.children: - if isinstance(child, ToolNode): - self._insert_tool(child) - else: - self._insert_folder(child) - - def _insert_folder(self, node): - #Insert into the elements table - column_string = "element_id, parent_id, document_id, tag, text, tail" - value_string = "{id}, {parent}, 1, '{title}', '', ''".format( - id=node.element_id, parent=node.parent.element_id, title=node.title) - sql = """INSERT INTO elements (%s) - VALUES (%s);""" % (column_string, value_string) - self.db.execute(sql) - - #Insert into the attributes table - column_string = "element_id, name, value" - atts = {'namespace': node.namespace, - 'description': '', - 'name': node.title, - 'children': [c.element_id for c in node.children], - MTBXDatabase.CATEGORY_MAGIC_NUMBER: 'True'} - for key, val in atts.iteritems(): - value_string = "{id}, '{name}', '{value}'".format( - id=node.element_id, name=key, value=val) - sql = """INSERT INTO attributes (%s) - VALUES (%s);""" % (column_string, value_string) - self.db.execute(sql) - - self.db.commit() - - #Handle children nodes - for child in node.children: - if isinstance(child, ToolNode): - self._insert_tool(child) - else: - self._insert_folder(child) - - def _insert_tool(self, node): - #Insert into the elements table - column_string = "element_id, parent_id, document_id, tag, text, tail" - value_string = "{id}, {parent}, 1, '{title}', '', ''".format( - id=node.element_id, parent=node.parent.element_id, title=node.title) - - sql = """INSERT INTO elements (%s) - VALUES (%s);""" % (column_string, value_string) - self.db.execute(sql) - - #Insert into the attributes table - column_string = "element_id, name, value" - atts = {'code': node.code, - 'description': '', - 'script': node.script, - 'namespace': node.namespace, - 'python_suffix': node.extension, - 'name': node.title, - MTBXDatabase.TOOL_MAGIC_NUMBER: 'True'} - for key, val in atts.iteritems(): - value_string = "{id}, '{name}', '{value!s}'".format( - id=node.element_id, name=key, value=val) - sql = """INSERT INTO attributes (%s) - VALUES (?, ?, ?);""" % column_string - self.db.execute(sql, (node.element_id, key, val)) - - self.db.commit() - - -def build_toolbox(toolbox_file, source_folder, title, namespace, consolidate): - print "------------------------" - print " Build Toolbox Utility" - print "------------------------" - print "" - print "toolbox: %s" % toolbox_file - print "source folder: %s" % source_folder - print "title: %s" % title - print "namespace: %s" % namespace - print "" - - print "Loading toolbox structure" - tree = ElementTree(namespace, title) - explore_source_folder(source_folder, tree) - tree.set_toolbox_order() - print "Done. Found %s elements." % (tree.next_element_id) - if consolidate: - print "Consolidating code..." - tree.consolidate() - print "Consolidate done" - - print "" - print "Building MTBX file..." - mtbx = MTBXDatabase(toolbox_file, title) - mtbx.populate_tables_from_tree(tree) - print "Build MTBX file done." - - -def explore_source_folder(root_folder_path, parent_node): - folders = [] - files = [] - for item in os.listdir(root_folder_path): - itempath = os.path.join(root_folder_path, item) - if os.path.isfile(itempath): - name, extension = os.path.splitext(item) - if extension != '.py': - continue # skip non-Python files - if os.path.normpath(itempath) == os.path.normpath(os.path.abspath(__file__)): - continue # skip this file - files.append((name, extension)) - else: - folders.append(item) - - for foldername in folders: - folderpath = os.path.join(root_folder_path, foldername) - folder_node = parent_node.add_folder(namespace=foldername) - explore_source_folder(folderpath, folder_node) - - for filename, ext in files: - script_path = os.path.join(root_folder_path, filename + ext) - parent_node.add_tool(script_path, namespace=filename) - - -if __name__ == "__main__": - ''' - Usage: build_toolbox.py [-p toolbox_path] [-s source_folder] [-l] [-c] - ''' - - import argparse - parser = argparse.ArgumentParser() - parser.add_argument('-s', '--src', help= "Path to the source code folder. Default is the working folder.") - parser.add_argument('-p', '--path', help= "Output file path. Default is 'sandag_toolbox.mtbx' in the source code folder.") - parser.add_argument('-l', '--link', help= "Link the python source files from their current location (instead of consolidate (compile) the toolbox).", action= 'store_true') - parser.add_argument('-c', '--consolidate', help= "Consolidate (compile) the toolbox (default option).", action= 'store_true') - - args = parser.parse_args() - - source_folder = args.src or os.path.dirname(os.path.abspath(__file__)) - folder_name = os.path.split(source_folder)[1] - toolbox_file = args.path or "sandag_toolbox.mtbx" - title = "SANDAG toolbox" - namespace = "sandag" - consolidate = args.consolidate - link = args.link - if consolidate and link: - raise Exception("-l and -c (--link and --consolidate) are mutually exclusive options") - if not consolidate and not link: - consolidate = True # default if neither is specified - - build_toolbox(toolbox_file, source_folder, title, namespace, consolidate) diff --git a/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py b/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py deleted file mode 100644 index f9f1ab1..0000000 --- a/sandag_abm/src/main/emme/toolbox/diagnostic/mode_choice_diagnostic.py +++ /dev/null @@ -1,669 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright RSG, 2019-2020. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/mode_choice_diagnostic.py /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Diagnostic tool for the SANDAG activity-based travel model mode choice results. -# This script first generates synthetic population files for target markets. -# Users may input target market parameters via the "syn_pop_attributes.yaml" file. -# Users must additionally input origin and destination MAZs (i.e. MGRAs) via the -# "origin_mgra.csv" and "destination_mgra.csv" files. -# -# Once all synthetic population files have been created, the script creates a copy of -# the "sandag_abm.properties" file and modifies specific property parameters so that -# it is compatible with a the mode choice diagnostic tool. The modified properties -# file is renamed as "sandag_abm_mcd.properties" -# -# Finally, the mode choice diagnostic tool is run via "runSandagAbm_MCDiagnostic.cmd" -# The mode choice diagnostic tool uses the synthetic population files as inputs and -# outputs a tour file with utilities and probabilities for each tour mode. -# -# Files referenced: -# input\mcd\destination_mgra.csv -# input\mcd\origin_mgra.csv -# input\mcd\syn_pop_attributes.yaml -# output\mcd\mcd_households.csv -# output\mcd\mcd_persons.csv -# output\mcd\mcd_output_households.csv -# output\mcd\mcd_output_persons.csv -# output\mcd\mcd_work_location.csv -# output\mcd\mcd_tour_file.csv -# conf\sandag_abm.properties -# bin\runSandagAbm_MCDiagnostic.cmd - -import inro.modeller as _m - -import pandas as pd -import collections, os -import shutil as _shutil -import yaml -import warnings -import traceback as _traceback -import tempfile as _tempfile -import subprocess as _subprocess - -warnings.filterwarnings("ignore") - -_join = os.path.join -_dir = os.path.dirname - -class mode_choice_diagnostic(_m.Tool()): - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = _dir(_m.Modeller().desktop.project.path) - self.main_directory = _dir(project_dir) - self.properties_path = _join(_dir(project_dir), "conf") - self.mcd_out_path = _join(_dir(project_dir), "output", "mcd") - self.syn_pop_attributes_path = _join(_dir(project_dir), "input", "mcd", "syn_pop_attributes.yaml") - self.origin_mgra_path = _join(_dir(project_dir), "input", "mcd", "origin_mgra.csv") - self.destination_mgra_path = _join(_dir(project_dir), "input", "mcd", "destination_mgra.csv") - self.household_df = pd.DataFrame() - self.household_out_df = pd.DataFrame() - self.person_df = pd.DataFrame() - self.person_out_df = pd.DataFrame() - self.work_location_df = pd.DataFrame() - self.tour_df = pd.DataFrame() - self.household_attributes = {} - self.person_attributes = {} - self.tour_attributes = {} - self._log_level = "ENABLED" - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Mode Choice Diagnostic Tool" - pb.description = """ - Diagnostic tool for the activity-based travel model mode choice results.
    -
    -
    - This tool first generates synthetic population files for specified target markets. - Users may edit target market attributes via a configuration file. - Users may additionally select origin and destination MAZs (i.e. MGRAs) of interest via - input CSV files.

    - The configuration file and MAZ selection CSV files are read from the following locations:
    -
      -
    • input\mcd\syn_pop_attributes.yaml
    • -
    • input\mcd\origin_mgra.csv
    • -
    • input\mcd\destination_mgra.csv
    • -
    - The synthetic population generator outputs the following files:
    -
      -
    • output\mcd\mcd_households.csv
    • -
    • output\mcd\mcd_persons.csv
    • -
    • output\mcd\mcd_output_households.csv
    • -
    • output\mcd\mcd_output_persons.csv
    • -
    • output\mcd\mcd_work_location.csv
    • -
    • output\mcd\mcd_tour_file.csv
    • -
    - Once all synthetic population files have been created, the script creates a copy of - the "sandag_abm.properties" file and modifies specific property parameters so that - it is compatible with the mode choice diagnostic tool. The modified properties - file is renamed and output as "conf\sandag_abm_mcd.properties"
    -
    - Finally, the mode choice diagnostic tool is run via runSandagAbm_MCDiagnostic.cmd - The mode choice diagnostic tool uses the synthetic population files as inputs and - outputs a tour file with utilities and probabilities for each tour mode. The tour file - is output as "output\mcd\indivTourData_5.csv" -
    - """ - pb.branding_text = "SANDAG - Mode Choice Diagnostic Tool" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self() - run_msg = "Mode Choice Diagnostic Complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self): - _m.logbook_write("Started running mode choice diagnostic...") - - # check if transit shapefiles are present in mcd input directory - # if present, will move to mcd output directory - _m.logbook_write("Checking for transit shapefiles...") - self.check_shp() - - # run synthetic population generator - _m.logbook_write("Creating synthetic population...") - self.syn_pop() - - # copy and edit properties file - _m.logbook_write("Copying and editing properties file...") - mcd_props = self.copy_edit_props() - - self.set_global_logbook_level(mcd_props) - - drive, path_no_drive = os.path.splitdrive(self.main_directory) - - # run matrix manager - _m.logbook_write("Running matrix manager...") - self.run_proc("runMtxMgr.cmd", [drive, drive + path_no_drive], "Start matrix manager") - - # run driver - _m.logbook_write("Running JPPF driver...") - self.run_proc("runDriver.cmd", [drive, drive + path_no_drive], "Start JPPF driver") - - # run household manager - _m.logbook_write("Running household manager, JPPF driver, and nodes...") - self.run_proc("StartHHAndNodes.cmd", [drive, path_no_drive], "Start household manager, JPPF driver, and nodes") - - # run diagnostic tool - _m.logbook_write("Running mode choice diagnostic tool...") - path_forward_slash = path_no_drive.replace("\\", "/") - self.run_proc( - "runSandagAbm_MCDiagnostic.cmd", - [drive, drive + path_forward_slash, 1.0, 5], - "Java-Run Mode Choice Diagnostic Tool", capture_output=True) - - # move final output mcd files to the mcd output directory - self.move_mcd_files() - - def syn_pop(self): - # Creates sample synthetic population files for desired target market. Files will in turn - # be used as inputs to the mode choice diagnostic tool - - load_properties = _m.Modeller().tool("sandag.utilities.properties") - props = load_properties(self.properties_path) - - mgra_data_path = _join(self.main_directory, props["mgra.socec.file"]) - - file_paths = [self.syn_pop_attributes_path, self.origin_mgra_path, self.destination_mgra_path, mgra_data_path] - - for path in file_paths: - if not os.path.exists(path): - raise Exception("missing file '%s'" % (path)) - - # create output directory if it donesn't already exist - if not os.path.exists(self.mcd_out_path): - os.makedirs(self.mcd_out_path) - - # read inputs - mgra_data = pd.read_csv(mgra_data_path)[['mgra', 'taz']] - origin_mgra_data = list(set(pd.read_csv(self.origin_mgra_path)['MGRA'])) - destination_mgra_data = list(set(pd.read_csv(self.destination_mgra_path)['MGRA'])) - - # read synthetic population attributes - with open (self.syn_pop_attributes_path) as file: - syn_pop_attributes = yaml.load(file, Loader = yaml.Loader) - - self.household_attributes = syn_pop_attributes["household"] - self.person_attributes = syn_pop_attributes["person"] - self.tour_attributes = syn_pop_attributes["tour"] - - # create households input file - self.household_in(origin_mgra_data, mgra_data) - - # create households output file - self.household_out() - - # create persons input file - self.person_in() - - # create persons output file - self.person_out() - - # create output work location file - self.work_location(destination_mgra_data) - - # create individual tour file - self.tour() - - def household_in(self, origin_mgra_data, mgra_data): - # Creates the input household file - - # fixed household attributes - household = collections.OrderedDict([ - ('hworkers', [1]), # number of hh workers: one worker per household - ('persons', [2]), # number of hh persons: two persons per household - ('version', [0]), # synthetic population version - ]) - - household_df = pd.DataFrame.from_dict(household) - household_df = self.replicate_df_for_variable(household_df, 'mgra', origin_mgra_data) - for key, values in self.household_attributes.items(): - household_df = self.replicate_df_for_variable(household_df, key, self.maybe_list(values)) - household_df['hinccat1'] = household_df.apply(lambda hh_row: self.hinccat(hh_row), axis = 1) - household_df = self.replicate_df_for_variable(household_df, 'poverty', [1]) - household_df = pd.merge(left = household_df, right = mgra_data, on = 'mgra') - household_df = household_df.reset_index(drop = True) - household_df['hhid'] = household_df.index + 1 - household_df['household_serial_no'] = 0 - - # reorder columns - household_df = household_df[['hhid', 'household_serial_no', 'taz', 'mgra', 'hinccat1', 'hinc', 'hworkers', - 'veh','persons', 'hht', 'bldgsz', 'unittype', 'version', 'poverty']] - - self.household_df = household_df - - # print - household_df.to_csv(_join(self.mcd_out_path, 'mcd_households.csv'), index = False) - - def household_out(self): - # Creates the output household file - - household_out_df = self.household_df.copy() - household_out_df = household_out_df[['hhid', 'mgra', 'hinc', 'veh']] - household_out_df['transponder'] = 1 - household_out_df['cdap_pattern'] = 'MNj' - household_out_df['out_escort_choice'] = 0 - household_out_df['inb_escort_choice'] = 0 - household_out_df['jtf_choice'] = 0 - if self.tour_attributes['av_avail']: - household_out_df['AVs'] = household_out_df['veh'] - household_out_df['veh'] = 0 - else: - household_out_df['AVs'] = 0 - - # rename columns - household_out_df.rename(columns = {'hhid':'hh_id', 'mgra':'home_mgra', 'hinc':'income', 'veh':'HVs'}, - inplace = True) - - self.household_out_df = household_out_df - - # print - household_out_df.to_csv(_join(self.mcd_out_path, 'mcd_output_households.csv'), index = False) - - def person_in(self): - # Creates the input person file - - # fixed person attributes - persons = collections.OrderedDict([ - ('pnum', [1, 2]), # person number: two per household - ('pemploy', [1, 3]), # employment status: full-time employee and unemployed - ('ptype', [1, 4]), # person type: full-time worker and non-working adult - ('occen5', [0, 0]), # occupation - ('occsoc5', ['11-1021', '00-0000']), # occupation code# - ('indcen', [0, 0]), # industry code - ('weeks', [1, 0]), # weeks worked - ('hours', [35, 0]), # hours worked - ('race1p', [9, 9]), # race - ('hisp', [1, 1]), # hispanic flag - ('version', [9, 9]), # synthetic population run version: 9 is new for disaggregate population - ('timeFactorWork', [1, 1]), # work travel time factor: 1 is the mean - ('timeFactorNonWork', [1, 1]), # non work travel time factor: 2 is the mean - ('DAP', ['M', 'N']) # daily activity pattern: M (Mandatory), N (Non-Mandatory) - ]) - - persons.update(self.person_attributes) - person_df = pd.DataFrame.from_dict(persons) - person_df['join_key'] = 1 - self.household_df['join_key'] = 1 - person_df = pd.merge(left = person_df, right = self.household_df[['hhid','household_serial_no', 'join_key']]).\ - drop(columns = ['join_key']) - person_df['pstudent'] = person_df.apply(lambda person_row: self.pstudent(person_row), axis = 1) - person_df = person_df.sort_values(by = 'hhid') - person_df = person_df.reset_index(drop = True) - person_df['perid'] = person_df.index + 1 - - # reorder columns - person_df = person_df[['hhid', 'perid', 'household_serial_no', 'pnum', 'age', 'sex', 'miltary', 'pemploy', - 'pstudent', 'ptype', 'educ', 'grade', 'occen5', 'occsoc5', 'indcen', 'weeks', 'hours', - 'race1p', 'hisp', 'version', 'timeFactorWork', 'timeFactorNonWork', 'DAP']] - - self.person_df = person_df - - # print - person_df.to_csv(_join(self.mcd_out_path, 'mcd_persons.csv'), index = False) - - def person_out(self): - # Creates the output person file - - person_out_df = self.person_df.copy() - person_out_df = person_out_df[['hhid', 'perid', 'pnum', 'age', 'sex', 'ptype', 'DAP', - 'timeFactorWork', 'timeFactorNonWork']] - person_out_df['gender'] = person_out_df['sex'].apply(lambda x: 'male' if x == 1 else 'female') - person_out_df['type'] = person_out_df.apply(lambda person_row: self.p_type(person_row), axis = 1) - person_out_df['value_of_time'] = 0 - person_out_df['imf_choice'] = person_out_df['pnum'].apply(lambda x: 1 if x == 1 else 0) - person_out_df['inmf_choice'] = person_out_df['pnum'].apply(lambda x: 0 if x == 1 else 36) - person_out_df['fp_choice'] = person_out_df['pnum'].apply(lambda x: 2 if x == 1 else -1) - person_out_df['reimb_pct'] = 0 - person_out_df['tele_choice'] = person_out_df['pnum'].apply(lambda x: 1 if x == 1 else -1) - person_out_df['ie_choice'] = 1 - - # drop columns not required - person_out_df.drop(columns = ['sex', 'ptype'], inplace = True) - - # rename columns - person_out_df.rename(columns = {'hhid':'hh_id', 'perid':'person_id', - 'pnum':'person_num', 'DAP':'activity_pattern'}, - inplace = True) - - # reorder columns - person_out_df = person_out_df[['hh_id', 'person_id', 'person_num', 'age', 'gender', 'type', 'value_of_time', - 'activity_pattern', 'imf_choice', 'inmf_choice', 'fp_choice', 'reimb_pct', - 'tele_choice', 'ie_choice', 'timeFactorWork', 'timeFactorNonWork']] - - self.person_out_df = person_out_df - - # print - person_out_df.to_csv(_join(self.mcd_out_path, 'mcd_output_persons.csv'), index = False) - - def work_location(self, destination_mgra_data): - # Creates the output work location file - - # create copies and subset household and person dataframes - household_subset_df = self.household_df.copy() - person_subset_df = self.person_df.copy() - household_subset_df = household_subset_df[['hhid', 'mgra', 'hinc']] - person_subset_df = person_subset_df[['hhid', 'perid', 'pnum', 'ptype', 'age', 'pemploy', 'pstudent']] - - # merge to create work location dataframe - work_location_df = pd.merge(left = household_subset_df, right = person_subset_df, on = 'hhid') - work_location_df['WorkSegment'] = work_location_df['pnum'].apply(lambda x: 0 if x == 1 else -1) - work_location_df['SchoolSegment'] = -1 - work_location_df = self.replicate_df_for_variable(work_location_df, 'WorkLocation', self.maybe_list(destination_mgra_data)) - work_location_df['WorkLocationDistance'] = 0 - work_location_df['WorkLocationLogsum'] = 0 - work_location_df['SchoolLocation'] = -1 - work_location_df['SchoolLocationDistance'] = 0 - work_location_df['SchoolLocationLogsum'] = 0 - - # rename columns - work_location_df.rename(columns = {'hhid':'HHID', 'mgra':'homeMGRA', 'hinc':'income', 'perid':'personID', - 'pnum':'personNum', 'ptype':'personType', 'age':'personAge', - 'pemploy':'Employment Category', 'pstudent':'StudentCategory'}, - inplace = True) - - # reorder columns - work_location_df = work_location_df[['HHID', 'homeMGRA', 'income', 'personID', 'personNum', 'personType', - 'personAge', 'Employment Category', 'StudentCategory', 'WorkSegment', - 'SchoolSegment', 'WorkLocation', 'WorkLocationDistance', 'WorkLocationLogsum', - 'SchoolLocation', 'SchoolLocationDistance', 'SchoolLocationLogsum']] - - self.work_location_df = work_location_df - - # print - work_location_df.to_csv(_join(self.mcd_out_path, 'mcd_work_location.csv'), index = False) - - def tour(self): - # Creates the individual tour file - - tour_df = self.work_location_df.copy() - tour_df = tour_df[['HHID', 'personID', 'personNum', 'personType', 'homeMGRA', 'WorkLocation']] - tour_df = tour_df.sort_values(by = list(tour_df.columns), ascending = True) - tour_df['tour_id'] = tour_df.groupby(['HHID', 'personID']).cumcount() - tour_df['tour_category'] = tour_df['personNum'].\ - apply(lambda x: 'INDIVIDUAL_MANDATORY' if x == 1 else 'INDIVIDUAL_NON_MANDATORY') - tour_df['tour_purpose'] = tour_df['personNum'].apply(lambda x: 'Work' if x == 1 else 'Shop') - tour_df['start_period'] = tour_df['personNum'].apply(lambda x: self.tour_attributes['start_period'][0] if x == 1 else self.tour_attributes['start_period'][1]) - tour_df['end_period'] = tour_df['personNum'].apply(lambda x: self.tour_attributes['end_period'][0] if x == 1 else self.tour_attributes['end_period'][1]) - tour_df['tour_mode'] = 0 - if self.tour_attributes['av_avail']: - tour_df['av_avail'] = 1 - else: - tour_df['av_avail'] = 0 - tour_df['tour_distance'] = 0 - tour_df['atwork_freq'] = tour_df['personNum'].apply(lambda x: 1 if x == 1 else 0) - tour_df['num_ob_stops'] = 0 - tour_df['num_ib_stops'] = 0 - tour_df['valueOfTime'] = 0 - tour_df['escort_type_out'] = 0 - tour_df['escort_type_in'] = 0 - tour_df['driver_num_out'] = 0 - tour_df['driver_num_in'] = 0 - - # utilities 1 through 13 - util_cols = [] - for x in range(1, 14, 1): - col_name = 'util_{}'.format(x) - tour_df[col_name] = 0 - util_cols.append(col_name) - - # probabilities 1 through 13 - prob_cols = [] - for x in range(1, 14, 1): - col_name = 'prob_{}'.format(x) - tour_df[col_name] = 0 - prob_cols.append(col_name) - - # rename columns - tour_df.rename(columns = {'HHID':'hh_id', 'personID':'person_id', 'personNum':'person_num', - 'personType':'person_type', 'homeMGRA':'orig_mgra', 'WorkLocation':'dest_mgra'}, - inplace = True) - - # reorder columns - tour_df = tour_df[['hh_id', 'person_id', 'person_num', 'person_type', 'tour_id', 'tour_category', - 'tour_purpose', 'orig_mgra', 'dest_mgra', 'start_period', 'end_period', - 'tour_mode', 'av_avail', 'tour_distance', 'atwork_freq', 'num_ob_stops', - 'num_ib_stops', 'valueOfTime', 'escort_type_out', 'escort_type_in', - 'driver_num_out', 'driver_num_in'] + util_cols + prob_cols] - - self.tour_df = tour_df - - # print - tour_df.to_csv(_join(self.mcd_out_path, 'mcd_tour_file.csv'), index = False) - - def replicate_df_for_variable(self, df, var_name, var_values): - new_var_df = pd.DataFrame({var_name: var_values}) - new_var_df['join_key'] = 1 - df['join_key'] = 1 - - ret_df = pd.merge(left = df, right = new_var_df, how = 'outer').drop(columns=['join_key']) - return ret_df - - def maybe_list(self, values): - if (type(values) is not list) and (type(values) is not int): - raise Exception('Attribute values may only be of type list or int.') - if type(values) is not list: - return [values] - else: - return values - - def hinccat(self, hh_row): - if hh_row['hinc'] < 30000: - return 1 - if hh_row['hinc'] >= 30000 and hh_row['hinc'] < 60000: - return 2 - if hh_row['hinc'] >= 60000 and hh_row['hinc'] < 100000: - return 3 - if hh_row['hinc'] >= 100000 and hh_row['hinc'] < 150000: - return 4 - if hh_row['hinc'] >= 150000: - return 5 - - def pstudent(self, person_row): - if person_row['grade'] == 0: - return 3 - if person_row['grade'] == 1: - return 1 - if person_row['grade'] == 2: - return 1 - if person_row['grade'] == 3: - return 1 - if person_row['grade'] == 4: - return 1 - if person_row['grade'] == 5: - return 1 - if person_row['grade'] == 6: - return 2 - if person_row['grade'] == 7: - return 2 - - def p_type(self, person_row): - if person_row['ptype'] == 1: - return 'Full-time worker' - if person_row['ptype'] == 2: - return 'Part-time worker' - if person_row['ptype'] == 3: - return 'University student' - if person_row['ptype'] == 4: - return 'Non-worker' - if person_row['ptype'] == 5: - return 'Retired' - if person_row['ptype'] == 6: - return 'Student of driving age' - if person_row['ptype'] == 7: - return 'Student of non-driving age' - if person_row['ptype'] == 8: - return 'Child too young for school' - - def copy_edit_props(self): - # Copy and edit properties file tokens to be compatible with the mode choice diagnostic tool - - load_properties = _m.Modeller().tool("sandag.utilities.properties") - mcd_props = load_properties(_join(self.properties_path, "sandag_abm.properties")) - - # update properties - - # PopSyn inputs - mcd_props["RunModel.MandatoryTourModeChoice"] = "true" - mcd_props["RunModel.IndividualNonMandatoryTourModeChoice"] = "true" - - # data file paths - mcd_props["PopulationSynthesizer.InputToCTRAMP.HouseholdFile"] = "output/mcd/mcd_households.csv" - mcd_props["PopulationSynthesizer.InputToCTRAMP.PersonFile"] = "output/mcd/mcd_persons.csv" - mcd_props["Accessibilities.HouseholdDataFile"] = "output/mcd/mcd_output_households.csv" - mcd_props["Accessibilities.PersonDataFile"] = "output/mcd/mcd_output_persons.csv" - mcd_props["Accessibilities.IndivTourDataFile"] = "output/mcd/mcd_tour_file.csv" - mcd_props["Accessibilities.JointTourDataFile"] = "" - mcd_props["Accessibilities.IndivTripDataFile"] = "" - mcd_props["Accessibilities.JointTripDataFile"] = "" - - # model component run flags - mcd_props["RunModel.PreAutoOwnership"] = "false" - mcd_props["RunModel.UsualWorkAndSchoolLocationChoice"] = "false" - mcd_props["RunModel.AutoOwnership"] = "false" - mcd_props["RunModel.TransponderChoice"] = "false" - mcd_props["RunModel.FreeParking"] = "false" - mcd_props["RunModel.CoordinatedDailyActivityPattern"] = "false" - mcd_props["RunModel.IndividualMandatoryTourFrequency"] = "false" - mcd_props["RunModel.MandatoryTourModeChoice"] = "true" - mcd_props["RunModel.MandatoryTourDepartureTimeAndDuration"] = "false" - mcd_props["RunModel.SchoolEscortModel"] = "false" - mcd_props["RunModel.JointTourFrequency"] = "false" - mcd_props["RunModel.JointTourLocationChoice"] = "false" - mcd_props["RunModel.JointTourDepartureTimeAndDuration"] = "false" - mcd_props["RunModel.JointTourModeChoice"] = "true" - mcd_props["RunModel.IndividualNonMandatoryTourFrequency"] = "false" - mcd_props["RunModel.IndividualNonMandatoryTourLocationChoice"] = "false" - mcd_props["RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration"] = "false" - mcd_props["RunModel.IndividualNonMandatoryTourModeChoice"] = "true" - mcd_props["RunModel.AtWorkSubTourFrequency"] = "false" - mcd_props["RunModel.AtWorkSubTourLocationChoice"] = "false" - mcd_props["RunModel.AtWorkSubTourDepartureTimeAndDuration"] = "false" - mcd_props["RunModel.AtWorkSubTourModeChoice"] = "true" - mcd_props["RunModel.StopFrequency"] = "false" - mcd_props["RunModel.StopLocation"] = "false" - - mcd_props.save(_join(self.properties_path, "sandag_abm_mcd.properties")) - - return(mcd_props) - - def run_proc(self, name, arguments, log_message, capture_output=False): - path = _join(self.main_directory, "bin", name) - if not os.path.exists(path): - raise Exception("No command / batch file '%s'" % path) - command = path + " " + " ".join([str(x) for x in arguments]) - attrs = {"command": command, "name": name, "arguments": arguments} - with _m.logbook_trace(log_message, attributes=attrs): - if capture_output and self._log_level != "NO_EXTERNAL_REPORTS": - report = _m.PageBuilder(title="Process run %s" % name) - report.add_html('Command:

    %s

    ' % command) - # temporary file to capture output error messages generated by Java - err_file_ref, err_file_path = _tempfile.mkstemp(suffix='.log') - err_file = os.fdopen(err_file_ref, "w") - try: - output = _subprocess.check_output(command, stderr=err_file, cwd=self.main_directory, shell=True) - report.add_html('Output:

    %s
    ' % output) - except _subprocess.CalledProcessError as error: - report.add_html('Output:

    %s
    ' % error.output) - raise - finally: - err_file.close() - with open(err_file_path, 'r') as f: - error_msg = f.read() - os.remove(err_file_path) - if error_msg: - report.add_html('Error message(s):

    %s
    ' % error_msg) - try: - # No raise on writing report error - # due to observed issue with runs generating reports which cause - # errors when logged - _m.logbook_write("Process run %s report" % name, report.render()) - except Exception as error: - print _time.strftime("%Y-%M-%d %H:%m:%S") - print "Error writing report '%s' to logbook" % name - print error - print _traceback.format_exc(error) - if self._log_level == "DISABLE_ON_ERROR": - _m.logbook_level(_m.LogbookLevel.NONE) - else: - _subprocess.check_call(command, cwd=self.main_directory, shell=True) - - def set_global_logbook_level(self, props): - self._log_level = props.get("RunModel.LogbookLevel", "ENABLED") - log_all = _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.VALUE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG - log_states = { - "ENABLED": log_all, - "DISABLE_ON_ERROR": log_all, - "NO_EXTERNAL_REPORTS": log_all, - "NO_REPORTS": _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, - "TITLES_ONLY": _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, - "DISABLED": _m.LogbookLevel.NONE, - } - _m.logbook_write("Setting logbook level to %s" % self._log_level) - try: - _m.logbook_level(log_states[self._log_level]) - except KeyError: - raise Exception("properties.RunModel.LogLevel: value must be one of %s" % ",".join(log_states.keys())) - - def move_mcd_files(self): - - out_directory = _join(self.main_directory, "output") - - hh_data = "householdData_5.csv" - ind_tour = "indivTourData_5.csv" - ind_trip = "indivTripData_5.csv" - joint_tour = "jointTourData_5.csv" - joint_trip = "jointTripData_5.csv" - per_data = "personData_5.csv" - mgra_park = "mgraParkingCost.csv" - - files = [hh_data, ind_tour, ind_trip, joint_tour, joint_trip, per_data, mgra_park] - - for file in files: - src = _join(out_directory, file) - if not os.path.exists(src): - raise Exception("missing output file '%s'" % (src)) - dst = _join(self.mcd_out_path, file) - _shutil.move(src, dst) - - def check_shp(self): - - in_directory = _join(self.main_directory, "input", "mcd") - out_directory = self.mcd_out_path - - shp_names = ["tapcov", "rtcov"] - - for shp in shp_names: - - files_to_move = [f for f in os.listdir(in_directory) if shp in f] - - for file in files_to_move: - - src = _join(in_directory, file) - dst = _join(out_directory, file) - if not os.path.exists(src): - raise Exception("missing shapefile '%s'" % (src)) - _shutil.move(src, dst) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py deleted file mode 100644 index c0df738..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_matrices.py +++ /dev/null @@ -1,302 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export/export_data_loader_matrices.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports the matrix results to OMX and csv files for use by the Java Data -# export process and the Data loader to the reporting database. -# -# -# Inputs: -# output_dir: the output directory for the created files -# base_scenario_id: scenario ID for the base scenario (same used in the Import network tool) -# transit_scenario_id: scenario ID for the base transit scenario -# -# Files created: -# CSV format files -# ../report/trucktrip.csv -# ../report/eetrip.csv -# ../report/eitrip.csv -# OMX format files -# trip_pp.omx -# -# -# Script example: -""" - import os - import inro.emme.database.emmebank as _eb - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - output_dir = os.path.join(main_directory, "output") - num_processors = "MAX-1" - export_data_loader_matrices = modeller.tool( - "sandag.export.export_data_loader_matrices") - export_data_loader_matrices(output_dir, 100, main_emmebank, transit_emmebank, num_processors) -""" -TOOLBOX_ORDER = 74 - - -import inro.modeller as _m -import inro.emme.database.emmebank as _eb -import traceback as _traceback -from collections import OrderedDict -import os -import numpy -import warnings -import tables - - -warnings.filterwarnings('ignore', category=tables.NaturalNameWarning) -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - -_join = os.path.join -_dir = os.path.dirname - -class ExportDataLoaderMatrices(_m.Tool(), gen_utils.Snapshot): - - output_dir = _m.Attribute(str) - base_scenario_id = _m.Attribute(int) - transit_scenario_id = _m.Attribute(int) - - tool_run_msg = "" - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.output_dir = os.path.join(os.path.dirname(project_dir), "output") - self.base_scenario_id = 100 - self.transit_scenario_id = 100 - self.periods = ["EA", "AM", "MD", "PM", "EV"] - self.attributes = ["main_directory", "base_scenario_id", "transit_scenario_id"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export matrices for Data Loader" - pb.description = """ - Export model results to OMX files for export by Data Exporter - to CSV format for load in SQL Data loader.""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('output_dir', 'directory', - title='Select output directory') - - pb.add_text_box('base_scenario_id', title="Base scenario ID:", size=10) - pb.add_text_box('transit_scenario_id', title="Transit scenario ID:", size=10) - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - base_emmebank = _eb.Emmebank(os.path.join(project_dir, "Database", "emmebank")) - transit_emmebank = _eb.Emmebank(os.path.join(project_dir, "Database_transit", "emmebank")) - base_scenario = base_emmebank.scenario(self.base_scenario_id) - transit_scenario = transit_emmebank.scenario(self.transit_scenario_id) - - results = self(self.output_dir, base_scenario, transit_scenario) - run_msg = "Export completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export matrices for Data Loader", save_arguments=True) - def __call__(self, output_dir, base_scenario, transit_scenario): - attrs = { - "output_dir": output_dir, - "base_scenario_id": base_scenario.id, - "transit_scenario_id": transit_scenario.id, - "self": str(self) - } - gen_utils.log_snapshot("Export Matrices for Data Loader", str(self), attrs) - self.output_dir = output_dir - self.base_scenario = base_scenario - self.transit_scenario = transit_scenario - - self.truck_demand() - self.external_demand() - self.total_demand() - - @_m.logbook_trace("Export truck demand") - def truck_demand(self): - name_mapping = [ - # ("lhdn", "TRKLGP", 1.3), - # ("mhdn", "TRKMGP", 1.5), - # ("hhdn", "TRKHGP", 2.5), - ("lhdt", "TRK_L", 1.3), - ("mhdt", "TRK_M", 1.5), - ("hhdt", "TRK_H", 2.5), - ] - scenario = self.base_scenario - emmebank = scenario.emmebank - zones = scenario.zone_numbers - formater = lambda x: ("%.5f" % x).rstrip('0').rstrip(".") - truck_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "trucktrip.csv") - - # get auto operating cost - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.output_dir), "conf", "sandag_abm.properties")) - try: - aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) - except ValueError: - raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") - - with open(truck_trip_path, 'w') as f: - f.write("OTAZ,DTAZ,TOD,MODE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") - for period in self.periods: - for key, name, pce in name_mapping: - matrix_data = emmebank.matrix(period + "_" + name + "_VEH").get_data(scenario) - matrix_data_time = emmebank.matrix(period + "_" + name + "_TIME").get_data(scenario) - matrix_data_dist = emmebank.matrix(period + "_" + name + "_DIST").get_data(scenario) - matrix_data_tollcost = emmebank.matrix(period + "_" + name + "_TOLLCOST").get_data(scenario) - rounded_demand = 0 - for orig in zones: - for dest in zones: - value = matrix_data.get(orig, dest) - # skip trips less than 0.00001 to avoid 0 trips records in database - if value < 0.00001: - rounded_demand += value - continue - time = matrix_data_time.get(orig, dest) - distance = matrix_data_dist.get(orig, dest) - tollcost = matrix_data_tollcost.get(orig, dest) - od_aoc = distance * aoc - f.write(",".join([str(orig), str(dest), period, key, formater(value), formater(time), formater(distance), formater(od_aoc), formater(tollcost)])) - f.write("\n") - if rounded_demand > 0: - print period + "_" + name + "_VEH", "rounded_demand", rounded_demand - - def external_demand(self): - #get auto operating cost - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.output_dir), "conf", "sandag_abm.properties")) - try: - aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) - except ValueError: - raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") - - # EXTERNAL-EXTERNAL TRIP TABLE (toll-eligible) - name_mapping = [ - ("DA", "SOV"), - ("S2", "HOV2"), - ("S3", "HOV3"), - ] - scenario = self.base_scenario - emmebank = scenario.emmebank - zones = scenario.zone_numbers - formater = lambda x: ("%.5f" % x).rstrip('0').rstrip(".") - ee_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "eetrip.csv") - with _m.logbook_trace("Export external-external demand"): - with open(ee_trip_path, 'w') as f: - f.write("OTAZ,DTAZ,TOD,MODE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") - for period in self.periods: - matrix_data_time = emmebank.matrix(period + "_SOV_NT_M_TIME").get_data(scenario) - matrix_data_dist = emmebank.matrix(period + "_SOV_NT_M_DIST").get_data(scenario) - matrix_data_tollcost = emmebank.matrix(period + "_SOV_NT_M_TOLLCOST").get_data(scenario) - for key, name in name_mapping: - matrix_data = emmebank.matrix(period + "_" + name + "_EETRIPS").get_data(scenario) - rounded_demand = 0 - for orig in zones: - for dest in zones: - value = matrix_data.get(orig, dest) - # skip trips less than 0.00001 to avoid 0 trips records in database - if value < 0.00001: - rounded_demand += value - continue - time = matrix_data_time.get(orig, dest) - distance = matrix_data_dist.get(orig, dest) - tollcost = 0 - tollcost = matrix_data_tollcost.get(orig, dest) - od_aoc = distance * aoc - f.write(",".join( - [str(orig), str(dest), period, key, formater(value), formater(time), - formater(distance), formater(od_aoc), formater(tollcost)])) - f.write("\n") - if rounded_demand > 0: - print period + "_" + name + "_EETRIPS", "rounded_demand", rounded_demand - - # EXTERNAL-INTERNAL TRIP TABLE - name_mapping = [ - ("DAN", "SOVGP"), - ("DAT", "SOVTOLL"), - ("S2N", "HOV2HOV"), - ("S2T", "HOV2TOLL"), - ("S3N", "HOV3HOV"), - ("S3T", "HOV3TOLL"), - ] - ei_trip_path = os.path.join(os.path.dirname(self.output_dir), "report", "eitrip.csv") - - with _m.logbook_trace("Export external-internal demand"): - with open(ei_trip_path, 'w') as f: - f.write("OTAZ,DTAZ,TOD,MODE,PURPOSE,TRIPS,TIME,DIST,AOC,TOLLCOST\n") - for period in self.periods: - matrix_data_time = emmebank.matrix(period + "_SOV_TR_M_TIME").get_data(scenario) - matrix_data_dist = emmebank.matrix(period + "_SOV_TR_M_DIST").get_data(scenario) - if "TOLL" in name: - matrix_data_tollcost = emmebank.matrix(period + "_SOV_NT_M_TOLLCOST").get_data(scenario) - for purpose in ["WORK", "NONWORK"]: - for key, name in name_mapping: - matrix_data = emmebank.matrix(period + "_" + name + "_EI" + purpose).get_data(scenario) - rounded_demand = 0 - for orig in zones: - for dest in zones: - value = matrix_data.get(orig, dest) - # skip trips less than 0.00001 to avoid 0 trips records in database - if value < 0.00001: - rounded_demand += value - continue - time = matrix_data_time.get(orig, dest) - distance = matrix_data_dist.get(orig, dest) - tollcost = 0 - if "TOLL" in name: - tollcost = matrix_data_tollcost.get(orig, dest) - od_aoc = distance * aoc - f.write(",".join( - [str(orig), str(dest), period, key, purpose, formater(value), formater(time), - formater(distance), formater(od_aoc), formater(tollcost)])) - f.write("\n") - if rounded_demand > 0: - print period + "_" + name + "_EI" + purpose, "rounded_demand", rounded_demand - - @_m.logbook_trace("Export total auto and truck demand to OMX") - def total_demand(self): - for period in self.periods: - matrices = { - "%s_SOV_NT_L": 'mf"%s_SOV_NT_L"', - "%s_SOV_TR_L": 'mf"%s_SOV_TR_L"', - "%s_HOV2_L": 'mf"%s_HOV2_L"', - "%s_HOV3_L": 'mf"%s_HOV3_L"', - "%s_SOV_NT_M": 'mf"%s_SOV_NT_M"', - "%s_SOV_TR_M": 'mf"%s_SOV_TR_M"', - "%s_HOV2_M": 'mf"%s_HOV2_M"', - "%s_HOV3_M": 'mf"%s_HOV3_M"', - "%s_SOV_NT_H": 'mf"%s_SOV_NT_H"', - "%s_SOV_TR_H": 'mf"%s_SOV_TR_H"', - "%s_HOV2_H": 'mf"%s_HOV2_H"', - "%s_HOV3_H": 'mf"%s_HOV3_H"', - "%s_TRK_H": 'mf"%s_TRK_H"', - "%s_TRK_L": 'mf"%s_TRK_L"', - "%s_TRK_M": 'mf"%s_TRK_M"', - } - matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) - omx_file = os.path.join(self.output_dir, "trip_%s.omx" % period) - with gen_utils.ExportOMX(omx_file, self.base_scenario) as exporter: - exporter.write_matrices(matrices) - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py b/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py deleted file mode 100644 index ec38ed3..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_data_loader_network.py +++ /dev/null @@ -1,1203 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export_data_loader_network.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports the network results to csv file for use by the Java Data export process -# and the Data loader to the reporting database. -# -# -# Inputs: -# main_directory: main ABM directory -# base_scenario_id: scenario ID for the base scenario (same used in the Import network tool) -# traffic_emmebank: the base, traffic, Emme database -# transit_emmebank: the transit database -# num_processors: number of processors to use in the transit analysis calculations -# -# Files created: -# report/hwyload_pp.csv -# report/hwy_tcad.csv rename to hwyTcad.csv -# report/transit_aggflow.csv -# report/transit_flow.csv -# report/transit_onoff.csv -# report/trrt.csv rename to transitRoute.csv -# report/trstop.csv renmae to transitStop.csv -# report/transitTap.csv -# report/transitLink.csv -# -# Script example: -""" - import os - import inro.emme.database.emmebank as _eb - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database_transit", "emmebank")) - num_processors = "MAX-1" - export_data_loader_network = modeller.tool( - "sandag.export.export_data_loader_network") - export_data_loader_network(main_directory, 100, main_emmebank, transit_emmebank, num_processors) -""" - -TOOLBOX_ORDER = 73 - - -import inro.modeller as _m -import traceback as _traceback -import inro.emme.database.emmebank as _eb -import inro.emme.desktop.worksheet as _ws -import inro.emme.datatable as _dt -import inro.emme.core.exception as _except -from contextlib import contextmanager as _context -from collections import OrderedDict -from itertools import chain as _chain -import math -import os -import pandas as pd -import numpy as _np - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - -format = lambda x: ("%.6f" % x).rstrip('0').rstrip(".") -id_format = lambda x: str(int(x)) - -class ExportDataLoaderNetwork(_m.Tool(), gen_utils.Snapshot): - - main_directory = _m.Attribute(str) - base_scenario_id = _m.Attribute(int) - traffic_emmebank = _m.Attribute(str) - transit_emmebank = _m.Attribute(str) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.main_directory = os.path.dirname(project_dir) - self.base_scenario_id = 100 - self.traffic_emmebank = os.path.join(project_dir, "Database", "emmebank") - self.transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") - self.num_processors = "MAX-1" - self.attributes = ["main_directory", "base_scenario_id", "traffic_emmebank", "transit_emmebank", "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export network for Data Loader" - pb.description = """ -Export network results to csv files for SQL data loader.""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('main_directory', 'directory', - title='Select main directory') - - pb.add_text_box('base_scenario_id', title="Base scenario ID:", size=10) - pb.add_select_file('traffic_emmebank', 'file', - title='Select traffic emmebank') - pb.add_select_file('transit_emmebank', 'file', - title='Select transit emmebank') - - dem_utils.add_select_processors("num_processors", pb, self) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - results = self(self.main_directory, self.base_scenario_id, - self.traffic_emmebank, self.transit_emmebank, - self.num_processors) - run_msg = "Export completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export network results for Data Loader", save_arguments=True) - def __call__(self, main_directory, base_scenario_id, traffic_emmebank, transit_emmebank, num_processors): - attrs = { - "traffic_emmebank": str(traffic_emmebank), - "transit_emmebank": str(transit_emmebank), - "main_directory": main_directory, - "base_scenario_id": base_scenario_id, - "self": str(self) - } - gen_utils.log_snapshot("Export network results", str(self), attrs) - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - - traffic_emmebank = _eb.Emmebank(traffic_emmebank) - transit_emmebank = _eb.Emmebank(transit_emmebank) - export_path = os.path.join(main_directory, "report") - input_path = os.path.join(main_directory,"input") - num_processors = dem_utils.parse_num_processors(num_processors) - - periods = ["EA", "AM", "MD", "PM", "EV"] - period_scenario_ids = OrderedDict((v, i) for i, v in enumerate(periods, start=base_scenario_id + 1)) - - base_scenario = traffic_emmebank.scenario(base_scenario_id) - - self.export_traffic_attribute(base_scenario, export_path, traffic_emmebank, period_scenario_ids, props) - self.export_traffic_load_by_period(export_path, traffic_emmebank, period_scenario_ids) - self.export_transit_results(export_path, input_path, transit_emmebank, period_scenario_ids, num_processors) - self.export_geometry(export_path, traffic_emmebank) - - @_m.logbook_trace("Export traffic attribute data") - def export_traffic_attribute(self, base_scenario, export_path, traffic_emmebank, period_scenario_ids, props): - # Several column names are legacy from the original network files - # and data loader process, and are populated with zeros. - # items are ("column name", "attribute name") or ("column name", ("attribute name", default)) - hwylink_attrs = [ - ("ID", "@tcov_id"), - ("Length", "length"), - ("Dir", "is_one_way"), - ("hwycov-id:1", "@tcov_id"), - ("ID:1", "@tcov_id"), - ("Length:1", "length_feet"), - ("QID", "zero"), - ("CCSTYLE", "zero"), - ("UVOL", "zero"), - ("AVOL", "zero"), - ("TMP1", "zero"), - ("TMP2", "zero"), - ("PLOT", "zero"), - ("SPHERE", "@sphere"), - ("RTNO", "zero"), - ("LKNO", "zero"), - ("NM", "#name"), - ("FXNM", "#name_from"), - ("TXNM", "#name_to"), - ("AN", "i"), - ("BN", "j"), - ("COJUR", "zero"), - ("COSTAT", "zero"), - ("COLOC", "zero"), - ("RLOOP", "zero"), - ("ADTLK", "zero"), - ("ADTVL", "zero"), - ("PKPCT", "zero"), - ("TRPCT", "zero"), - ("SECNO", "zero"), - ("DIR:1", "zero"), - ("FFC", "type"), - ("CLASS", "zero"), - ("ASPD", "@speed_adjusted"), - ("IYR", "@year_open_traffic"), - ("IPROJ", "@project_code"), - ("IJUR", "@jurisdiction_type"), - ("IFC", "type"), - ("IHOV", "@lane_restriction"), - ("ITRUCK", "@truck_restriction"), - ("ISPD", "@speed_posted"), - ("ITSPD", "zero"), - ("IWAY", "iway"), - ("IMED", "@median"), - ("COST", "@cost_operating"), - ("ITOLLO", "@toll_md"), - ("ITOLLA", "@toll_am"), - ("ITOLLP", "@toll_pm"), - ] - directional_attrs = [ - ("ABLNO", "@lane_md", "0"), - ("ABLNA", "@lane_am", "0"), - ("ABLNP", "@lane_pm", "0"), - ("ABAU", "@lane_auxiliary", "0"), - ("ABPCT", "zero", "0"), - ("ABPHF", "zero", "0"), - ("ABCNT", "@traffic_control", "0"), - ("ABTL", "@turn_thru", "0"), - ("ABRL", "@turn_right", "0"), - ("ABLL", "@turn_left", "0"), - ("ABTLB", "zero", "0"), - ("ABRLB", "zero", "0"), - ("ABLLB", "zero", "0"), - ("ABGC", "@green_to_cycle_init", "0"), - ("ABPLC", "per_lane_capacity", "1900"), - ("ABCPO", "@capacity_link_md", "999999"), - ("ABCPA", "@capacity_link_am", "999999"), - ("ABCPP", "@capacity_link_pm", "999999"), - ("ABCXO", "@capacity_inter_md", "999999"), - ("ABCXA", "@capacity_inter_am", "999999"), - ("ABCXP", "@capacity_inter_pm", "999999"), - ("ABCHO", "@capacity_hourly_op", "0"), - ("ABCHA", "@capacity_hourly_am", "0"), - ("ABCHP", "@capacity_hourly_pm", "0"), - ("ABTMO", "@time_link_md", "999"), - ("ABTMA", "@time_link_am", "999"), - ("ABTMP", "@time_link_pm", "999"), - ("ABTXO", "@time_inter_md", "0"), - ("ABTXA", "@time_inter_am", "0"), - ("ABTXP", "@time_inter_pm", "0"), - ("ABCST", "zero", "999.999"), - ("ABVLA", "zero", "0"), - ("ABVLP", "zero", "0"), - ("ABLOS", "zero", "0"), - ] - for key, name, default in directional_attrs: - hwylink_attrs.append((key, name)) - for key, name, default in directional_attrs: - hwylink_attrs.append(("BA" + key[2:], (name, default))) - hwylink_attrs.append(("relifac", "relifac")) - - time_period_atts = [ - ("ITOLL2", "@toll"), - ("ITOLL3", "@cost_auto"), - ("ITOLL4", "@cost_med_truck"), - ("ITOLL5", "@cost_hvy_truck"), - ("ITOLL", "toll_hov"), - ("ABCP", "@capacity_link", "999999"), - ("ABCX", "@capacity_inter", "999999"), - ("ABTM", "@time_link", "999"), - ("ABTX", "@time_inter", "0"), - ("ABLN", "@lane", "0"), - ("ABSCST", "sov_total_gencost", ""), - ("ABH2CST", "hov2_total_gencost", ""), - ("ABH3CST", "hov3_total_gencost", ""), - ("ABSTM", "auto_time", ""), - ("ABHTM", "auto_time", ""), - ] - periods = ["_ea", "_am", "_md", "_pm", "_ev"] - for column in time_period_atts: - for period in periods: - key = column[0] + period.upper() - name = column[1] + period - hwylink_attrs.append((key, name)) - if key.startswith("AB"): - for period in periods: - key = column[0] + period.upper() - name = column[1] + period - default = column[2] - hwylink_attrs.append(("BA" + key[2:], (name, default))) - for period in periods: - key = "ABPRELOAD" + period.upper() - name = "additional_volume" + period - default = "0" - hwylink_attrs.append((key, name)) - hwylink_attrs.append(("BA" + key[2:], (name, default))) - - vdf_attrs = [ - ("AB_GCRatio", "@green_to_cycle", ""), - ("AB_Cycle", "@cycle", ""), - ("AB_PF", "progression_factor", ""), - ("ALPHA1", "alpha1", "0.8"), - ("BETA1", "beta1", "4"), - ("ALPHA2", "alpha2", "4.5"), - ("BETA2", "beta2", "2"), - ] - for key, name, default in vdf_attrs: - name = name + "_am" if name.startswith("@") else name - hwylink_attrs.append((key, name)) - if key.startswith("AB"): - hwylink_attrs.append(("BA" + key[2:], (name, default))) - for period in periods: - for key, name, default in vdf_attrs: - name = name + period if name.startswith("@") else name - default = default or "0" - hwylink_attrs.append((key + period.upper(), name)) - if key.startswith("AB"): - hwylink_attrs.append(("BA" + key[2:] + period.upper(), (name, default))) - - network = base_scenario.get_partial_network(["LINK"], include_attributes=True) - - #copy assignment from period scenarios - for period, scenario_id in period_scenario_ids.iteritems(): - from_scenario = traffic_emmebank.scenario(scenario_id) - src_attrs = ["@auto_time", "additional_volume"] - dst_attrs = ["auto_time_" + period.lower(), - "additional_volume_" + period.lower()] - for dst_attr in dst_attrs: - network.create_attribute("LINK", dst_attr) - values = from_scenario.get_attribute_values("LINK", src_attrs) - network.set_attribute_values("LINK", dst_attrs, values) - # add in and calculate additional columns - new_attrs = [ - ("zero", 0), ("is_one_way", 0), ("iway", 2), ("length_feet", 0), - ("toll_hov", 0), ("per_lane_capacity", 1900), - ("progression_factor", 1.0), ("alpha1", 0.8), ("beta1", 4.0), - ("alpha2", 4.5), ("beta2", 2.0), ("relifac", 1.0), - ] - for name, default in new_attrs: - network.create_attribute("LINK", name, default) - for period in periods: - network.create_attribute("LINK", "toll_hov" + period, 0) - network.create_attribute("LINK", "sov_total_gencost" + period, 0) - network.create_attribute("LINK", "hov2_total_gencost" + period, 0) - network.create_attribute("LINK", "hov3_total_gencost" + period, 0) - for link in network.links(): - link.is_one_way = 1 if link.reverse_link else 0 - link.iway = 2 if link.reverse_link else 1 - link.length_feet = link.length * 5280 - for period in periods: - link["toll_hov" + period] = link["@cost_hov2" + period] - link["@cost_operating"] - link["sov_total_gencost" + period] = link["auto_time" + period] + link["@cost_auto" + period] - link["hov2_total_gencost" + period] = link["auto_time" + period] + link["@cost_hov2" + period] - link["hov3_total_gencost" + period] = link["auto_time" + period] + link["@cost_hov3" + period] - if link.volume_delay_func == 24: - link.alpha2 = 6.0 - link.per_lane_capacity = max([(link["@capacity_link" + p] / link["@lane" + p]) - for p in periods if link["@lane" + p] > 0] + [0]) - - hwylink_atts_file = os.path.join(export_path, "hwy_tcad.csv") - busPCE = props["transit.bus.pceveh"] - self.export_traffic_to_csv(hwylink_atts_file, hwylink_attrs, network, busPCE) - - @_m.logbook_trace("Export traffic load data by period") - def export_traffic_load_by_period(self, export_path, traffic_emmebank, period_scenario_ids): - create_attribute = _m.Modeller().tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - net_calculator = _m.Modeller().tool( - "inro.emme.network_calculation.network_calculator") - hwyload_attrs = [("ID1", "@tcov_id")] - - dir_atts = [ - ("AB_Flow_PCE", "@pce_flow"), # sum of pce flow - ("AB_Time", "@auto_time"), # computed vdf based on pce flow - ("AB_VOC", "@voc"), - ("AB_V_Dist_T", "length"), - ("AB_VHT", "@vht"), - ("AB_Speed", "@speed"), - ("AB_VDF", "@msa_time"), - ("AB_MSA_Flow", "@msa_flow"), - ("AB_MSA_Time", "@msa_time"), - ("AB_Flow_SOV_NTPL", "@sov_nt_l"), - ("AB_Flow_SOV_TPL", "@sov_tr_l"), - ("AB_Flow_SR2L", "@hov2_l"), - ("AB_Flow_SR3L", "@hov3_l"), - ("AB_Flow_SOV_NTPM", "@sov_nt_m"), - ("AB_Flow_SOV_TPM", "@sov_tr_m"), - ("AB_Flow_SR2M", "@hov2_m"), - ("AB_Flow_SR3M", "@hov3_m"), - ("AB_Flow_SOV_NTPH", "@sov_nt_h"), - ("AB_Flow_SOV_TPH", "@sov_tr_h"), - ("AB_Flow_SR2H", "@hov2_h"), - ("AB_Flow_SR3H", "@hov3_h"), - ("AB_Flow_lhd", "@trk_l_non_pce"), - ("AB_Flow_mhd", "@trk_m_non_pce"), - ("AB_Flow_hhd", "@trk_h_non_pce"), - ("AB_Flow", "@non_pce_flow"), - ] - - for key, attr in dir_atts: - hwyload_attrs.append((key, attr)) - hwyload_attrs.append((key.replace("AB_", "BA_"), (attr, ""))) # default for BA on one-way links is blank - for p, scen_id in period_scenario_ids.iteritems(): - scenario = traffic_emmebank.scenario(scen_id) - new_atts = [ - ("@speed", "link travel speed", "length*60/@auto_time"), - ("@sov_nt_all", "total number of SOV GP vehicles", - "@sov_nt_l+@sov_nt_m+@sov_nt_h" ), - ("@sov_tr_all", "total number of SOV TOLL vehicles", - "@sov_tr_l+@sov_tr_m+@sov_tr_h" ), - ("@hov2_all", "total number of HOV2 HOV vehicles", - "@hov2_l+@hov2_m+@hov2_h" ), - ("@hov3_all", "total number of HOV3 HOV vehicles", - "@hov3_l+@hov3_m+@hov3_h" ), - ("@trk_l_non_pce", "total number of light trucks in non-Pce", - "(@trk_l)/1.3" ), - ("@trk_m_non_pce", "total medium trucks in non-Pce", - "(@trk_m)/1.5" ), - ("@trk_h_non_pce", "total heavy trucks in non-Pce", - "(@trk_h)/2.5" ), - ("@pce_flow", "total number of vehicles in Pce", - "@sov_nt_all+@sov_tr_all+ \ - @hov2_all+ \ - @hov3_all+ \ - (@trk_l) + (@trk_m) + \ - (@trk_h) + volad" ), - ("@non_pce_flow", "total number of vehicles in non-Pce", - "@sov_nt_all+@sov_tr_all+ \ - @hov2_all+ \ - @hov3_all+ \ - (@trk_l)/1.3 + (@trk_m)/1.5 + \ - (@trk_h)/2.5 + volad/3" ), #volad includes bus flow - pce factor is 3 - ("@msa_flow", "MSA flow", "@non_pce_flow"), #flow from final assignment - ("@msa_time", "MSA time", "timau"), #skim assignment time on msa flow - ("@voc", "volume over capacity", "@pce_flow/ul3"), #pce flow over road capacity - ("@vht", "vehicle hours travelled", "@non_pce_flow*@auto_time/60") #vehicle flow (non-pce)*time - ] - - for name, des, formula in new_atts: - att = scenario.extra_attribute(name) - if not att: - att = create_attribute("LINK", name, des, 0, overwrite=True, scenario=scenario) - cal_spec = {"result": att.id, - "expression": formula, - "aggregation": None, - "selections": {"link": "mode=d"}, - "type": "NETWORK_CALCULATION" - } - net_calculator(cal_spec, scenario=scenario) - file_path = os.path.join(export_path, "hwyload_%s.csv" % p) - network = self.get_partial_network(scenario, {"LINK": ["@tcov_id"] + [a[1] for a in dir_atts]}) - self.export_traffic_to_csv(file_path, hwyload_attrs, network) - - def export_traffic_to_csv(self, filename, att_list, network, busPCE = None): - auto_mode = network.mode("d") - # only the original forward direction links and auto links only - links = [l for l in network.links() - if l["@tcov_id"] > 0 and - (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes)) - ] - links.sort(key=lambda l: l["@tcov_id"]) - with open(filename, 'w') as fout: - fout.write(",".join(['"%s"' % x[0] for x in att_list])) - fout.write("\n") - for link in links: - key, att = att_list[0] # expected to be the link id - values = [id_format(link[att])] - reverse_link = link.reverse_link - for key, att in att_list[1:]: - if key == "AN": - values.append(link.i_node.id) - elif key == "BN": - values.append(link.j_node.id) - elif key.startswith("BA"): - name, default = att - if reverse_link and (abs(link["@tcov_id"]) == abs(reverse_link["@tcov_id"])): - if "additional_volume" in name: - values.append(format(float(reverse_link[name]) / busPCE)) - else: - values.append(format(reverse_link[name])) - else: - values.append(default) - - #values.append(format(reverse_link[name]) if reverse_link else default) - elif att.startswith("#"): - values.append('"%s"' % link[att]) - else: - if "additional_volume" in att: - values.append(format(float(link[att]) / busPCE)) - else: - values.append(format(link[att])) - fout.write(",".join(values)) - fout.write("\n") - - @_m.logbook_trace("Export transit results") - def export_transit_results(self, export_path, input_path, transit_emmebank, period_scenario_ids, num_processors): - # Note: Node analysis for transfers is VERY time consuming - # this implementation will be replaced when new Emme version is available - - trrt_atts = ["Route_ID","Route_Name","Mode","AM_Headway","PM_Headway","OP_Headway","Night_Headway","Night_Hours","Config","Fare"] - trstop_atts = ["Stop_ID","Route_ID","Link_ID","Pass_Count","Milepost","Longitude","Latitude","NearNode","FareZone","StopName"] - - #transit route file - trrt_infile = os.path.join(input_path, "trrt.csv") - trrt = pd.read_csv(trrt_infile) - trrt = trrt.rename(columns=lambda x:x.strip()) - trrt_out = trrt[trrt_atts] - trrt_outfile = os.path.join(export_path, "trrt.csv") - trrt_out.to_csv(trrt_outfile, index=False) - - #transit stop file - trstop_infile = os.path.join(input_path, "trstop.csv") - trstop = pd.read_csv(trstop_infile) - trstop = trstop.rename(columns={"HwyNode":"NearNode"}) - trstop = trstop.rename(columns=lambda x:x.strip()) - trstop_out = trstop[trstop_atts] - trstop_outfile = os.path.join(export_path, "trstop.csv") - trstop_out.to_csv(trstop_outfile, index=False) - - use_node_analysis_to_get_transit_transfers = False - - copy_scenario = _m.Modeller().tool( - "inro.emme.data.scenario.copy_scenario") - create_attribute = _m.Modeller().tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - net_calculator = _m.Modeller().tool( - "inro.emme.network_calculation.network_calculator") - copy_attribute= _m.Modeller().tool( - "inro.emme.data.network.copy_attribute") - delete_scenario = _m.Modeller().tool( - "inro.emme.data.scenario.delete_scenario") - transit_flow_atts = [ - "MODE", - "ACCESSMODE", - "TOD", - "ROUTE", - "FROM_STOP", - "TO_STOP", - "CENTROID", - "FROMMP", - "TOMP", - "TRANSITFLOW", - "BASEIVTT", - "COST", - "VOC", - ] - transit_aggregate_flow_atts = [ - "MODE", - "ACCESSMODE", - "TOD", - "LINK_ID", - "AB_TransitFlow", - "BA_TransitFlow", - "AB_NonTransit", - "BA_NonTransit", - "AB_TotalFlow", - "BA_TotalFlow", - "AB_Access_Walk_Flow", - "BA_Access_Walk_Flow", - "AB_Xfer_Walk_Flow", - "BA_Xfer_Walk_Flow", - "AB_Egress_Walk_Flow", - "BA_Egress_Walk_Flow" - ] - transit_onoff_atts = [ - "MODE", - "ACCESSMODE", - "TOD", - "ROUTE", - "STOP", - "BOARDINGS", - "ALIGHTINGS", - "WALKACCESSON", - "DIRECTTRANSFERON", - "WALKTRANSFERON", - "DIRECTTRANSFEROFF", - "WALKTRANSFEROFF", - "EGRESSOFF" - ] - - transit_flow_file = os.path.join(export_path, "transit_flow.csv") - fout_seg = open(transit_flow_file, 'w') - fout_seg.write(",".join(['"%s"' % x for x in transit_flow_atts])) - fout_seg.write("\n") - - transit_aggregate_flow_file = os.path.join(export_path, "transit_aggflow.csv") - fout_link = open(transit_aggregate_flow_file, 'w') - fout_link.write(",".join(['"%s"' % x for x in transit_aggregate_flow_atts])) - fout_link.write("\n") - - transit_onoff_file = os.path.join(export_path, "transit_onoff.csv") - fout_stop = open(transit_onoff_file, 'w') - fout_stop.write(",".join(['"%s"' % x for x in transit_onoff_atts])) - fout_stop.write("\n") - try: - access_modes = ["WLK", "PNR", "KNR"] - main_modes = ["BUS", "PREM","ALLPEN"] - all_modes = ["b", "c", "e", "l", "r", "p", "y", "o", "a", "w", "x"] - local_bus_modes = ["b", "a", "w", "x"] - premium_modes = ["c", "l", "e", "p", "r", "y", "o", "a", "w", "x"] - for tod, scen_id in period_scenario_ids.iteritems(): - with _m.logbook_trace("Processing period %s" % tod): - scenario = transit_emmebank.scenario(scen_id) - # attributes - total_walk_flow = create_attribute("LINK", "@volax", "total walk flow on links", - 0, overwrite=True, scenario=scenario) - segment_flow = create_attribute("TRANSIT_SEGMENT", "@voltr", "transit segment flow", - 0, overwrite=True, scenario=scenario) - link_transit_flow = create_attribute("LINK", "@link_voltr", "total transit flow on link", - 0, overwrite=True, scenario=scenario) - initial_boardings = create_attribute("TRANSIT_SEGMENT", - "@init_boardings", "transit initial boardings", - 0, overwrite=True, scenario=scenario) - xfer_boardings = create_attribute("TRANSIT_SEGMENT", - "@xfer_boardings", "transit transfer boardings", - 0, overwrite=True, scenario=scenario) - total_boardings = create_attribute("TRANSIT_SEGMENT", - "@total_boardings", "transit total boardings", - 0, overwrite=True, scenario=scenario) - final_alightings = create_attribute("TRANSIT_SEGMENT", - "@final_alightings", "transit final alightings", - 0, overwrite=True, scenario=scenario) - xfer_alightings = create_attribute("TRANSIT_SEGMENT", - "@xfer_alightings", "transit transfer alightings", - 0, overwrite=True, scenario=scenario) - total_alightings = create_attribute("TRANSIT_SEGMENT", - "@total_alightings", "transit total alightings", - 0, overwrite=True, scenario=scenario) - - access_walk_flow = create_attribute("LINK", - "@access_walk_flow", "access walks (orig to init board)", - 0, overwrite=True, scenario=scenario) - xfer_walk_flow = create_attribute("LINK", - "@xfer_walk_flow", "xfer walks (init board to final alight)", - 0, overwrite=True, scenario=scenario) - egress_walk_flow = create_attribute("LINK", - "@egress_walk_flow", "egress walks (final alight to dest)", - 0, overwrite=True, scenario=scenario) - - for main_mode in main_modes: - mode = main_mode - if main_mode == "BUS": - mode_list = local_bus_modes - elif main_mode == "PREM": - mode_list = premium_modes - else: - mode_list = all_modes - - for access_type in access_modes: - with _m.logbook_trace("Main mode %s access mode %s" % (main_mode, access_type)): - class_name = "%s_%s%s" % (tod, access_type, main_mode) - segment_results = { - "transit_volumes": segment_flow.id, - "initial_boardings": initial_boardings.id, - "total_boardings": total_boardings.id, - "final_alightings": final_alightings.id, - "total_alightings": total_alightings.id, - "transfer_boardings": xfer_boardings.id, - "transfer_alightings": xfer_alightings.id - } - link_results = { - "total_walk_flow": total_walk_flow.id, - "link_transit_flow": link_transit_flow.id, - "access_walk_flow": access_walk_flow.id, - "xfer_walk_flow": xfer_walk_flow.id, - "egress_walk_flow": egress_walk_flow.id - } - - self.calc_additional_results( - scenario, class_name, num_processors, - total_walk_flow, segment_results, link_transit_flow, - access_walk_flow, xfer_walk_flow, egress_walk_flow) - attributes = { - "NODE": ["@network_adj", "@network_adj_src"],#, "initial_boardings", "final_alightings"], - "LINK": link_results.values() + ["@tcov_id", "length"], - "TRANSIT_LINE": ["@route_id"], - "TRANSIT_SEGMENT": segment_results.values() + [ - "transit_time", "dwell_time", "@stop_id", "allow_boardings", "allow_alightings"], - } - network = self.get_partial_network(scenario, attributes) - self.collapse_network_adjustments(network, segment_results, link_results) - # =============================================== - # analysis for nodes with/without walk option - if use_node_analysis_to_get_transit_transfers: - stop_on, stop_off = self.transfer_analysis(scenario, class_name, num_processors) - else: - stop_on, stop_off = {}, {} - # =============================================== - transit_modes = [m for m in network.modes() if m.type in ("TRANSIT", "AUX_TRANSIT")] - links = [link for link in network.links() - if link["@tcov_id"] > 0 and (link.modes.union(transit_modes))] - links.sort(key=lambda l: l["@tcov_id"]) - lines = [line for line in network.transit_lines() if line.mode.id in mode_list] - lines.sort(key=lambda l: l["@route_id"]) - - label = ",".join([mode, access_type, tod]) - self.output_transit_flow(label, lines, segment_flow.id, fout_seg) - self.output_transit_aggregate_flow( - label, links, link_transit_flow.id, total_walk_flow.id, access_walk_flow.id, - xfer_walk_flow.id, egress_walk_flow.id, fout_link) - self.output_transit_onoff( - label, lines, total_boardings.id, total_alightings.id, initial_boardings.id, - xfer_boardings.id, xfer_alightings.id, final_alightings.id, - stop_on, stop_off, fout_stop) - finally: - fout_stop.close() - fout_link.close() - fout_seg.close() - return - - @_m.logbook_trace("Export geometries") - def export_geometry(self, export_path, traffic_emmebank): - # --------------------------Export Transit Nework Geometory----------------------------- - # domain: NODE, LINK, TURN, TRANSIT_LINE, TRANSIT_VEHICLE, TRANSIT_SEGMENT - def export_as_csv(domain, attributes, scenario = None): - if scenario is None: - scenario = _m.Modeller().scenario - initial_scenario = _m.Modeller().scenario - #if initial_scenario.number != scenario.number: - #data_explorer.replace_primary_scenario(scenario) - # Create the network table - network_table = project.new_network_table(domain) - for k, a in enumerate(attributes): - column = _ws.Column() - column.name = column.expression = a - network_table.add_column(k, column) - # Extract data - data = network_table.get_data() - f = _np.vectorize(lambda x: x.text) # required to get the WKT representation of the geometry column - data_dict = {} - for a in data.attributes(): - if isinstance(a, _dt.GeometryAttribute): - data_dict[a.name] = f(a.values) - else: - data_dict[a.name] = a.values - df = pd.DataFrame(data_dict) - - network_table.close() - #if initial_scenario.number != scenario.number: - # data_explorer.replace_primary_scenario(initial_scenario) - return df - - desktop = _m.Modeller().desktop - desktop.refresh_data() - data_explorer = desktop.data_explorer() - previous_active_database = data_explorer.active_database() - try: - desktop_traffic_database = data_explorer.add_database(traffic_emmebank.path) - desktop_traffic_database.open() - except Exception as error: - import traceback - print (traceback.format_exc()) - project = desktop.project - scenario = _m.Modeller().emmebank.scenario(101) - data_explorer.replace_primary_scenario(scenario) - node_attributes = ['i','@tap_id'] - link_attributes = ['i', 'j', '@tcov_id', 'modes'] - transit_line_attributes = ['line', 'routeID'] - transit_segment_attributes = ['line', 'i', 'j', 'loop_index','@tcov_id','@stop_id'] - mode_talbe = ['mode', 'type'] - network_table = project.new_network_table('MODE') - for k, a in enumerate(mode_talbe): - column = _ws.Column() - column.name = column.expression = a - network_table.add_column(k, column) - data = network_table.get_data() - data_dict = {} - for a in data.attributes(): - data_dict[a.name] = a.values - df = pd.DataFrame(data_dict) - mode_list = df[df['type'].isin([2.0, 3.0])]['mode'].tolist() - - df = export_as_csv('NODE', node_attributes, scenario) - df = df[['@tap_id', 'geometry']] - is_tap = df['@tap_id'] > 0 - df = df[is_tap] - df.columns = ['tapID', 'geometry'] - df.to_csv(os.path.join(export_path, 'transitTap.csv'), index=False) - - df = export_as_csv('TRANSIT_LINE', transit_line_attributes) - df = df[['line', 'geometry']] - df.columns = ['Route_Name', 'geometry'] - df['Route_Name'] = df['Route_Name'].astype(int) - df_routeFull = pd.read_csv(os.path.join(export_path, 'trrt.csv')) - result = pd.merge(df_routeFull, df, how='left', on=['Route_Name']) - result.to_csv(os.path.join(export_path, 'transitRoute.csv'), index=False) - os.remove(os.path.join(export_path, 'trrt.csv')) - - df = export_as_csv('TRANSIT_SEGMENT', transit_segment_attributes, None) - df_seg = df[['@tcov_id', 'geometry']] - df_seg.columns = ['trcovID', 'geometry'] - df_seg = df_seg.drop_duplicates() - #df_seg.to_csv(os.path.join(export_path, 'transitLink.csv'), index=False) - #df_stop = df[(df['@stop_id'] > 0) & (df['@tcov_id'] > 0)] - df_stop = df[(df['@stop_id'] > 0)] - df_stop = df_stop[['@stop_id', 'geometry']] - df_stop = df_stop.drop_duplicates() - df_stop.columns = ['Stop_ID', 'geometry'] - temp=[] - for value in df_stop['geometry']: - value=value.split(',') - value[0]=value[0]+')' - value[0]=value[0].replace("LINESTRING", "POINT") - temp.append(value[0]) - df_stop['geometry'] = temp - df_stopFull = pd.read_csv(os.path.join(export_path, 'trstop.csv')) - result = pd.merge(df_stopFull, df_stop, how='left', on=['Stop_ID']) - result.to_csv(os.path.join(export_path, 'transitStop.csv'), index=False) - os.remove(os.path.join(export_path, 'trstop.csv')) - - df = export_as_csv('LINK', link_attributes, None) - df_link = df[['@tcov_id', 'geometry']] - df_link.columns = ['hwycov-id:1', 'geometry'] - df_linkFull = pd.read_csv(os.path.join(export_path, 'hwy_tcad.csv')) - result = pd.merge(df_linkFull, df_link, how='left', on=['hwycov-id:1']) - result.to_csv(os.path.join(export_path, 'hwyTcad.csv'), index=False) - os.remove(os.path.join(export_path, 'hwy_tcad.csv')) - ##mode_list = ['Y','b','c','e','l','p','r','y','a','x','w']## - df_transit_link = df[df.modes.str.contains('|'.join(mode_list))] - df_transit_link = df_transit_link[['@tcov_id', 'geometry']] - df_transit_link.columns = ['trcovID', 'geometry'] - df_transit_link = df_transit_link[df_transit_link['trcovID'] != 0] - df_transit_link['AB'] = df_transit_link['trcovID'].apply(lambda x: 1 if x > 0 else 0) - df_transit_link['trcovID'] = abs(df_transit_link['trcovID']) - df_transit_link = df_transit_link[['trcovID', 'AB', 'geometry']] - df_transit_link.to_csv(os.path.join(export_path, 'transitLink.csv'), index=False) - network_table.close() - try: - previous_active_database.open() - data_explorer.remove_database(desktop_traffic_database) - except: - pass - - def get_partial_network(self, scenario, attributes): - domains = attributes.keys() - network = scenario.get_partial_network(domains, include_attributes=False) - for domain, attrs in attributes.iteritems(): - if attrs: - values = scenario.get_attribute_values(domain, attrs) - network.set_attribute_values(domain, attrs, values) - return network - - def output_transit_flow(self, label, lines, segment_flow, fout_seg): - # output segment data (transit_flow) - centroid = "0" # always 0 - voc = "" # volume/capacity, not actually used, - for line in lines: - line_id = id_format(line["@route_id"]) - ivtt = from_mp = to_mp = 0 - segments = iter(line.segments(include_hidden=True)) - seg = segments.next() - from_stop = id_format(seg["@stop_id"]) - for next_seg in segments: - to_mp += seg.link.length - ivtt += seg.transit_time - next_seg.dwell_time - transit_flow = seg[segment_flow] - seg = next_seg - if not next_seg.allow_alightings: - continue - to_stop = id_format(next_seg["@stop_id"]) - formatted_ivtt = format(ivtt) - fout_seg.write(",".join([ - label, line_id, from_stop, to_stop, centroid, format(from_mp), format(to_mp), - format(transit_flow), formatted_ivtt, formatted_ivtt, voc])) - fout_seg.write("\n") - from_stop = to_stop - from_mp = to_mp - ivtt = 0 - - def output_transit_aggregate_flow(self, label, links, - link_transit_flow, total_walk_flow, access_walk_flow, - xfer_walk_flow, egress_walk_flow, fout_link): - # output link data (transit_aggregate_flow) - for link in links: - link_id = id_format(link["@tcov_id"]) - ab_transit_flow = link[link_transit_flow] - ab_non_transit_flow = link[total_walk_flow] - ab_total_flow = ab_transit_flow + ab_non_transit_flow - ab_access_walk_flow = link[access_walk_flow] - ab_xfer_walk_flow = link[xfer_walk_flow] - ab_egress_walk_flow = link[egress_walk_flow] - if link.reverse_link: - ba_transit_flow = link.reverse_link[link_transit_flow] - ba_non_transit_flow = link.reverse_link[total_walk_flow] - ba_total_flow = ba_transit_flow + ba_non_transit_flow - ba_access_walk_flow = link.reverse_link[access_walk_flow] - ba_xfer_walk_flow = link.reverse_link[xfer_walk_flow] - ba_egress_walk_flow = link.reverse_link[egress_walk_flow] - else: - ba_transit_flow = 0.0 - ba_non_transit_flow = 0.0 - ba_total_flow = 0.0 - ba_access_walk_flow = 0.0 - ba_xfer_walk_flow = 0.0 - ba_egress_walk_flow = 0.0 - - fout_link.write(",".join( - [label, link_id, - format(ab_transit_flow), format(ba_transit_flow), - format(ab_non_transit_flow), format(ba_non_transit_flow), - format(ab_total_flow), format(ba_total_flow), - format(ab_access_walk_flow), format(ba_access_walk_flow), - format(ab_xfer_walk_flow), format(ba_xfer_walk_flow), - format(ab_egress_walk_flow), format(ba_egress_walk_flow)])) - fout_link.write("\n") - - def output_transit_onoff(self, label, lines, - total_boardings, total_alightings, initial_boardings, - xfer_boardings, xfer_alightings, final_alightings, - stop_on, stop_off, fout_stop): - # output stop data (transit_onoff) - for line in lines: - line_id = id_format(line["@route_id"]) - for seg in line.segments(True): - if not (seg.allow_alightings or seg.allow_boardings): - continue - i_node = seg.i_node.id - boardings = seg[total_boardings] - alightings = seg[total_alightings] - walk_access_on = seg[initial_boardings] - direct_xfer_on = seg[xfer_boardings] - walk_xfer_on = 0.0 - direct_xfer_off = seg[xfer_alightings] - walk_xfer_off = 0.0 - if stop_on.has_key(i_node): - if stop_on[i_node].has_key(line.id): - if direct_xfer_on > 0: - walk_xfer_on = direct_xfer_on - stop_on[i_node][line.id] - direct_xfer_on = stop_on[i_node][line.id] - if stop_off.has_key(i_node): - if stop_off[i_node].has_key(line.id): - if direct_xfer_off > 0: - walk_xfer_off = direct_xfer_off - stop_off[i_node][line.id] - direct_xfer_off = stop_off[i_node][line.id] - - egress_off = seg[final_alightings] - fout_stop.write(",".join([ - label, line_id, id_format(seg["@stop_id"]), - format(boardings), format(alightings), format(walk_access_on), - format(direct_xfer_on), format(walk_xfer_on), format(direct_xfer_off), - format(walk_xfer_off), format(egress_off)])) - fout_stop.write("\n") - - def collapse_network_adjustments(self, network, segment_results, link_results): - segment_alights = [v for k, v in segment_results.items() if "alightings" in k] - segment_boards = [v for k, v in segment_results.items() if "boardings" in k] + ["transit_boardings"] - segment_result_attrs = segment_alights + segment_boards - link_result_attrs = link_results.values() + ["aux_transit_volume"] - link_attrs = network.attributes("LINK") - link_modified_attrs = [ - "length", "@trtime_link_ea", "@trtime_link_am", "@trtime_link_md", - "@trtime_link_pm", "@trtime_link_ev", link_results["link_transit_flow"]] - seg_attrs = network.attributes("TRANSIT_SEGMENT") - line_attrs = network.attributes("TRANSIT_LINE") - - transit_modes = set([network.mode(m) for m in "blryepc"]) - aux_modes = set([network.mode(m) for m in "wxa"]) - xfer_mode = network.mode('x') - - def copy_seg_attrs(src_seg, dst_seg): - for attr in segment_result_attrs: - dst_seg[attr] += src_seg[attr] - dst_seg["allow_alightings"] |= src_seg["allow_alightings"] - dst_seg["allow_boardings"] |= src_seg["allow_boardings"] - - def get_xfer_link(node, timed_xfer_link, is_outgoing=True): - links = node.outgoing_links() if is_outgoing else node.incoming_links() - for link in links: - if xfer_mode in link.modes and link.length == timed_xfer_link.length: - return link - return None - - lines_to_update = set([]) - nodes_to_merge = [] - nodes_to_delete = [] - - for node in network.regular_nodes(): - if node["@network_adj"] == 1: - nodes_to_merge.append(node) - # copy boarding / alighting attributes for the segments to the original segment / stop - for seg in node.incoming_segments(): - lines_to_update.add(seg.line) - copy_seg_attrs(seg, seg.line.segment(seg.number+2)) - for seg in node.outgoing_segments(): - lines_to_update.add(seg.line) - copy_seg_attrs(seg, seg.line.segment(seg.number+1)) - elif node["@network_adj"] == 2: - nodes_to_delete.append(node) - # copy boarding / alighting attributes for the segments to the original segment / stop - for seg in node.outgoing_segments(True): - lines_to_update.add(seg.line) - if seg.j_node: - copy_seg_attrs(seg, seg.line.segment(seg.number+1)) - else: - copy_seg_attrs(seg, seg.line.segment(seg.number-1)) - elif node["@network_adj"] == 3: - orig_node = network.node(node["@network_adj_src"]) - # Remove transfer walk links and copy data to source walk link - for link in node.outgoing_links(): - if xfer_mode in link.modes and link.j_node["@network_adj"] == 3: - orig_xfer_link = get_xfer_link(orig_node, link) - for attr in link_result_attrs: - orig_xfer_link[attr] += link[attr] - network.delete_link(link.i_node, link.j_node) - # Sum link and segment results and merge links - mapping = network.merge_links_mapping(node) - for (link1, link2), attr_map in mapping['links'].iteritems(): - for attr in link_modified_attrs: - attr_map[attr] = max(link1[attr], link2[attr]) - - for (seg1, seg2), attr_map in mapping['transit_segments'].iteritems(): - if seg2.allow_alightings: - for attr in seg_attrs: - attr_map[attr] = seg1[attr] - else: # if it is a boarding stop or non-stop - for attr in seg_attrs: - attr_map[attr] = max(seg1[attr], seg2[attr]) - attr_map["transit_time_func"] = min(seg1["transit_time_func"], seg2["transit_time_func"]) - for attr in segment_boards: - attr_map[attr] = seg1[attr] + seg2[attr] - next_seg = seg2.line.segment(seg2.number+1) - for attr in segment_alights: - next_seg[attr] += seg2[attr] - network.merge_links(node, mapping) - - # Backup transit lines with altered routes and remove from network - lines = [] - for line in lines_to_update: - seg_data = {} - itinerary = [] - for seg in line.segments(include_hidden=True): - if seg.i_node["@network_adj"] in [1,2] or (seg.j_node and seg.j_node["@network_adj"] == 1): - continue - # for circle line transfers, j_node is now None for new "hidden" segment - j_node = seg.j_node - if (seg.j_node and seg.j_node["@network_adj"] == 2): - j_node = None - seg_data[(seg.i_node, j_node, seg.loop_index)] = dict((k, seg[k]) for k in seg_attrs) - itinerary.append(seg.i_node.number) - - lines.append({ - "id": line.id, - "vehicle": line.vehicle, - "itinerary": itinerary, - "attributes": dict((k, line[k]) for k in line_attrs), - "seg_attributes": seg_data}) - network.delete_transit_line(line) - # Remove duplicate network elements (undo network adjustments) - for node in nodes_to_delete: - for link in _chain(node.incoming_links(), node.outgoing_links()): - network.delete_link(link.i_node, link.j_node) - network.delete_node(node) - for node in nodes_to_merge: - mapping = network.merge_links_mapping(node) - for (link1, link2), attr_map in mapping["links"].iteritems(): - if link2.j_node.is_centroid: - link1, link2 = link2, link1 - for attr in link_attrs: - attr_map[attr] = link1[attr] - network.merge_links(node, mapping) - # Re-create transit lines on new itineraries - for line_data in lines: - new_line = network.create_transit_line( - line_data["id"], line_data["vehicle"], line_data["itinerary"]) - for k, v in line_data["attributes"].iteritems(): - new_line[k] = v - seg_data = line_data["seg_attributes"] - for seg in new_line.segments(include_hidden=True): - data = seg_data.get((seg.i_node, seg.j_node, seg.loop_index), {}) - for k, v in data.iteritems(): - seg[k] = v - - def calc_additional_results(self, scenario, class_name, num_processors, - total_walk_flow, segment_results, link_transit_flow, - access_walk_flow, xfer_walk_flow, egress_walk_flow): - network_results = _m.Modeller().tool( - "inro.emme.transit_assignment.extended.network_results") - path_based_analysis = _m.Modeller().tool( - "inro.emme.transit_assignment.extended.path_based_analysis") - net_calculator = _m.Modeller().tool( - "inro.emme.network_calculation.network_calculator") - - spec = { - "on_links": { - "aux_transit_volumes": total_walk_flow.id - }, - "on_segments": segment_results, - "aggregated_from_segments": None, - "analyzed_demand": None, - "constraint": None, - "type": "EXTENDED_TRANSIT_NETWORK_RESULTS" - } - network_results(specification=spec, scenario=scenario, - class_name=class_name, num_processors=num_processors) - cal_spec = { - "result": "%s" % link_transit_flow.id, - "expression": "%s" % segment_results["transit_volumes"], - "aggregation": "+", - "selections": { - "link": "all", - "transit_line": "all" - }, - "type": "NETWORK_CALCULATION" - } - net_calculator(cal_spec, scenario=scenario) - - walk_flows = [("INITIAL_BOARDING_TO_FINAL_ALIGHTING", access_walk_flow.id), - ("INITIAL_BOARDING_TO_FINAL_ALIGHTING", xfer_walk_flow.id), - ("FINAL_ALIGHTING_TO_DESTINATION", egress_walk_flow.id)] - for portion_of_path, aux_transit_volumes in walk_flows: - spec = { - "portion_of_path": portion_of_path, - "trip_components": { - "in_vehicle": None, - "aux_transit": "length", - "initial_boarding": None, - "transfer_boarding": None, - "transfer_alighting": None, - "final_alighting": None - }, - "path_operator": ".max.", - "path_selection_threshold": { - "lower": -1.0, - "upper": 999999.0 - }, - "path_to_od_aggregation": None, - "constraint": None, - "analyzed_demand": None, - "results_from_retained_paths": { - "paths_to_retain": "SELECTED", - "aux_transit_volumes": aux_transit_volumes - }, - "path_to_od_statistics": None, - "path_details": None, - "type": "EXTENDED_TRANSIT_PATH_ANALYSIS" - } - path_based_analysis( - specification=spec, scenario=scenario, - class_name=class_name, num_processors=num_processors) - - def transfer_analysis(self, scenario, net, class_name, num_processors): - create_attribute = _m.Modeller().tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - transfers_at_stops = _m.Modeller().tool( - "inro.emme.transit_assignment.extended.apps.transfers_at_stops") - - # find stop with/without walk transfer option - stop_walk_list = [] # stop (id) with walk option - stop_flag = "@stop_flag" - create_attribute("NODE", att, "1=stop without walk option, 2=otherwise", - 0, overwrite=True, scenario=scenario) - stop_nline = "@stop_nline" - create_attribute("NODE", stop_nline, "number of lines on the stop", - 0, overwrite=True, scenario=scenario) - - for line in net.transit_lines(): - for seg in line.segments(True): - node = seg.i_node - if seg.allow_alightings or seg.allow_boardings: - node[stop_nline] += 1 - if node[stop_flag] > 0 : #node checked - continue - if seg.allow_alightings or seg.allow_boardings: - node[stop_flag] = 1 - for ilink in node.incoming_links(): - # skip connector - if ilink.i_node.is_centroid: - continue - for m in ilink.modes: - if m.type=="AUX_TRANSIT": - node[stop_flag] = 2 - stop_walk_list.append(node.id) - break - if node[stop_flag]>=2: - break - if node[stop_flag]>=2: - continue - for olink in node.outgoing_links(): - # skip connector - if olink.j_node.is_centroid: - continue - for m in olink.modes: - if m.type=="AUX_TRANSIT": - node[stop_flag] = 2 - stop_walk_list.append(node.id) - break - if node[stop_flag]>=2: - break - #scenario.publish_network(net) - stop_off = {} - stop_on = {} - for stop in stop_walk_list: - stop_off[stop] = {} - stop_on[stop] = {} - selection = "i=%s" % stop - results = transfers_at_stops( - selection, scenario=scenario, - class_name=class_name, num_processors=num_processors) - for off_line in results: - stop_off[stop][off_line] = 0.0 - for on_line in results[off_line]: - trip = float(results[off_line][on_line]) - stop_off[stop][off_line] += trip - if not stop_on[stop].has_key(on_line): - stop_on[stop][on_line] = 0.0 - stop_on[stop][on_line] += trip - return stop_off, stop_on - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py b/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py deleted file mode 100644 index fee626d..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_for_commercial_vehicle.py +++ /dev/null @@ -1,158 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export/export_for_commercial_vehicle.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports the required skims in CSV format for the commercial vehicle model. -# -# -# Inputs: -# source: -# -# Files referenced: -# -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - source_dir = os.path.join(main_directory, "input") - title = "Base 2012 scenario" - tool = modeller.tool("sandag.export.export_for_commercial_vehicle") -""" - - -TOOLBOX_ORDER = 51 - - -import inro.modeller as _m -import numpy as _np -import subprocess as _subprocess -import tempfile as _tempfile -import traceback as _traceback -import os - -_join = os.path.join -_dir = os.path.dirname - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ExportForCommercialVehicleModel(_m.Tool(), gen_utils.Snapshot): - - output_directory = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = _dir(_m.Modeller().desktop.project.path) - self.output_directory = _join(_dir(project_dir), "output") - self.attributes = ["output_directory"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export for commercial vehicle model" - pb.description = """ - Exports the required skims in CSV format for the commercial vehicle model. - """ - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('output_directory', 'directory', - title='Select output directory') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.output_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Export skims for commercial vehicle model', save_arguments=True) - def __call__(self, output_directory, scenario): - emmebank = scenario.emmebank - modes = ['ldn', 'ldt', 'lhdn', 'lhdt', 'mhdn', 'mhdt', 'hhdn', 'hhdt'] - classes = ['SOV_NT_H', 'SOV_TR_H', 'TRK_L', 'TRK_L', 'TRK_M', 'TRK_M', 'TRK_H', 'TRK_H'] - # Mappings between COMMVEH modes and Emme classes - mode_class = dict(zip(modes, classes)) - class_mode = dict(zip(classes, modes)) - - is_toll_mode = lambda m: m.endswith('t') - #periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - period = "MD" - skims = ['TIME', 'DIST', 'TOLLCOST'] - DUCoef = [ - [-0.313, -0.138, -0.01], - [-0.313, -0.492, -0.01], - [-0.302, -0.580, -0.02] - ] - # Mappings for DUCoef utility index - modes_util = { - 'ldn': 0, - 'ldt': 0, - 'lhdn': 1, - 'lhdt': 1, - 'mhdn': 1, - 'mhdt': 1, - 'hhdn': 2, - 'hhdt': 2 - } - - # Lookup relevant skims as numpy arrays - skim_mat = {} - for cls in classes: - for skim in skims: - name = '%s_%s_%s' % (period, cls, skim) - if name not in skim_mat: - skim_mat[name] = emmebank.matrix(name).get_numpy_data(scenario) - - output_matrices = { - 'impldt_MD_Time.txt': skim_mat['MD_SOV_TR_H_TIME'], - 'impldt_MD_Dist.txt': skim_mat['MD_SOV_TR_H_DIST'], - } - - # Calculate DU matrices in numpy - for mode in modes: - time = skim_mat['%s_%s_TIME' % (period, mode_class[mode])] - distance = skim_mat['%s_%s_DIST' % (period, mode_class[mode])] - # All classes now have a tollcost skim available - toll_cost = skim_mat['%s_%s_TOLLCOST' % (period, mode_class[mode])] - _np.fill_diagonal(toll_cost, 0) - - coeffs = DUCoef[modes_util[mode]] - disutil_mat = coeffs[0] * time + coeffs[1] * distance + coeffs[2] * toll_cost - output_matrices['imp%s_%s_DU.txt' % (mode, period)] = disutil_mat - - # Insert row number into first column of the array - # Note: assumes zone IDs are continuous - for key, array in output_matrices.iteritems(): - output_matrices[key] = _np.insert(array, 0, range(1, array.shape[0]+1), axis=1) - - # Output DU matrices to CSV - # Print first column as integer, subsequent columns as floats rounded to 6 decimals - fmt_spec = ['%i'] + ['%.6f'] * (disutil_mat.shape[0]) - # Save to separate files - for name, array in output_matrices.iteritems(): - _np.savetxt(_join(output_directory, name), array, fmt=fmt_spec, delimiter=',') diff --git a/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py b/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py deleted file mode 100644 index f5736a3..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_for_transponder.py +++ /dev/null @@ -1,374 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2019-2020. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export_for_transponder.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# - - -TOOLBOX_ORDER = 57 - - -import inro.modeller as _m - -import numpy as _np -import pandas as _pd -import string as _string -import traceback as _traceback -import math -import os -_dir, _join = os.path.dirname, os.path.join - -from shapely.geometry import MultiLineString, Point, LineString -from contextlib import contextmanager as _context -from itertools import izip as _izip - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module('sandag.utilities.demand') - - -class ExportForTransponder(_m.Tool(), gen_utils.Snapshot): - - scenario = _m.Attribute(_m.InstanceType) - output_directory = _m.Attribute(unicode) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - def __init__(self): - project_dir = _dir(_m.Modeller().desktop.project.path) - modeller = _m.Modeller() - if modeller.emmebank.path == _join(project_dir, "Database", "emmebank"): - self.scenario = modeller.emmebank.scenario(102) - self.num_processors = "max-1" - self.output_directory = _join(_dir(project_dir), "output") - self.attributes = ["scenario", "output_directory", "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export for transponder ownership model" - pb.description = """ -

    Calculates and exports the following results for each origin zone:

    -
      -
    • "MLDIST" - Managed lane distance - straight-line distance to the - nearest managed lane facility -
    • "AVGTTS" - Average travel time savings - average travel time savings - across all possible destinations. -
    • "PCTDETOUR" - Percent detour - The percent difference between the AM - transponder travel time and the AM non-transponder travel time - to sample zones when the general purpose lanes parallel to all toll - lanes using transponders are not available. -
    - .""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_scenario("scenario", - title="Representative scenario") - pb.add_select_file('output_directory', 'directory', - title='Select output directory') - - dem_utils.add_select_processors("num_processors", pb, self) - return pb.render() - - - def run(self): - self.tool_run_msg = "" - try: - self(self.output_directory, self.num_processors, self.scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export results for transponder ownership model", save_arguments=True) - def __call__(self, output_directory, num_processors, scenario): - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties( - _join(_dir(output_directory), "conf", "sandag_abm.properties")) - input_directory = _join(_dir(output_directory), "input") - num_processors = dem_utils.parse_num_processors(num_processors) - network = scenario.get_network() - distances = self.ml_facility_dist(network) - savings = self.avg_travel_time_savings(scenario, input_directory, props, num_processors) - detour = self.percent_detour(scenario, network, props, num_processors) - self.export_results(output_directory, scenario, distances, savings, detour) - - @_m.logbook_trace("Calculate distance to nearest managed lane facility") - def ml_facility_dist(self, network): - # DIST: Straight line distance to the nearest ML facility (nearest link with a ML Cost) - # managed lane is : - # HOV2+ only (carpool lane): "IFC" = 1 and "IHOV" = 2 and "ITOLLO" = 0 and "ITOLLA" = 0 and "ITOLLP" = 0 - # HOV3+ only (carpool lane): "IFC" = 1 and "IHOV" = 3 and "ITOLLO" = 0 and "ITOLLA" = 0 and "ITOLLP" = 0 - # HOV2+ & HOT (managed lane. HOV 2+ free. SOV pay toll): ): "IFC" = 1 and "IHOV" = 2 and "ITOLLO" > 0 and "ITOLLA" > 0 and "ITOLLP" > 0 - # HOV2+ & HOT (managed lane. HOV 3+ free. HOV2 & SOV pay toll): ): "IFC" = 1 and "IHOV" = 3 and "ITOLLO" > 0 and "ITOLLA" > 0 and "ITOLLP" > 0 - # Tollway (all vehicles tolled): "IFC" = 1 and "IHOV" = 4 - #$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ - # NOTE: NOT ALL MANAGED LANE LINKS HAVE A TOLL COST, - # SOME COSTS ARE JUST SPECIFIED ON THE ENTRANCE / EXIT LINKS - #$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ - - ml_link_coords = [] - ml_links = [] - for link in network.links(): - if link["type"] == 1 and link["@lane_restriction"] in (2,3) and ( - link["@toll_am"] + link["@toll_md"] + link["@toll_pm"]) > 0: - ml_link_coords.append(LineString(link.shape)) - ml_links.append(link) - ml_link_collection = MultiLineString(ml_link_coords) - distances = [] - for zone in network.centroids(): - zone_point = Point((zone.x, zone.y)) - distances.append(zone_point.distance(ml_link_collection) / 5280) - - # distances is a Python list of the distance from each zone to nearest ML link - # in same order as centroids - return distances - - @_m.logbook_trace("Calculate average travel time savings") - def avg_travel_time_savings(self, scenario, input_directory, props, num_processors): - # AVGTTS: The average travel savings of all households in each zone over all possible - # work destinations d. - # This average was calculated using an expected value with probabilities taken - # from a simplified destination - # choice model. The expected travel time savings of households in a zone z is - # SUM[d](NTTime[zd] - TRTime[zd]) * Employment[d] * exp(-0.01*NTTime[zd]) / - # SUM[d]Employmentd * exp(-0.01*NTTime[zd]) - # - # NTTime[zd] = AM_NTTime[zd] + PM_NTTime[dz] - # TRTime[zd] = AM_TRTime[zd] + PM_TRTime[dz] - - emmebank = scenario.emmebank - year = int(props['scenarioYear']) - mgra = _pd.read_csv( - _join(input_directory, 'mgra13_based_input%s.csv' % year)) - taz = mgra[['taz', 'emp_total']].groupby('taz').sum() - taz.reset_index(inplace=True) - taz = dem_utils.add_missing_zones(taz, scenario) - taz.reset_index(inplace=True) - - with setup_for_tt_savings_calc(emmebank): - employment_matrix = emmebank.matrix("mdemployment") - employment_matrix.set_numpy_data(taz["emp_total"].values, scenario.id) - matrix_calc = dem_utils.MatrixCalculator(scenario, num_processors) - matrix_calc.add("NTTime", "AM_SOV_NT_M_TIME + PM_SOV_NT_M_TIME'") - matrix_calc.add("TRTime", "AM_SOV_TR_M_TIME + PM_SOV_TR_M_TIME'") - matrix_calc.add("numerator", "((NTTime - TRTime).max.0) * employment * exp(-0.01 * NTTime)", - aggregation={"destinations": "+"}) - matrix_calc.add("denominator", "employment * exp(-0.01 * NTTime)", - aggregation={"destinations": "+"}) - matrix_calc.add("AVGTTS", "numerator / denominator") - matrix_calc.run() - avg_tts = emmebank.matrix("AVGTTS").get_numpy_data(scenario.id) - return avg_tts - - @_m.logbook_trace("Calculate percent detour without managed lane facilities") - def percent_detour(self, scenario, network, props, num_processors): - # PCTDETOUR: The percent difference between the AM non-toll travel time - # to a sample downtown zone and the AM non-toll travel time to downtown - # when the general purpose lanes parallel to all toll lanes requiring - # transponders are not available. This variable - # is calculated as - # 100*(TimeWithoutFacility - NonTransponderTime) / NonTransponderTime - - destinations = props["transponder.destinations"] - - network.create_attribute("NODE", "@root") - network.create_attribute("NODE", "@leaf") - - mode_id = get_available_mode_id(network) - new_mode = network.create_mode("AUX_AUTO", mode_id) - sov_non_toll_mode = network.mode("s") - - # Find special managed links and potential parallel GP facilities - ml_link_coords = [] - freeway_links = [] - for link in network.links(): - if link["@lane_restriction"] in [2, 3] and link["type"] == 1 and ( - link["@toll_am"] + link["@toll_md"] + link["@toll_pm"]) > 0: - ml_link_coords.append(LineString(link.shape)) - if sov_non_toll_mode in link.modes: - link.modes |= set([new_mode]) - if link["type"] == 1: - freeway_links.append(link) - - # remove mode from nearby GP links to special managed lanes - ml_link_collection = MultiLineString(ml_link_coords) - for link in freeway_links: - link_shape = LineString(link.shape) - distance = link_shape.distance(ml_link_collection) - if distance < 100: - for ml_shape in ml_link_collection: - if ml_shape.distance(link_shape) and close_bearing(link_shape, ml_shape): - link.modes -= set([new_mode]) - break - - for node in network.centroids(): - node["@root"] = 1 - for dst in destinations: - network.node(dst)["@leaf"] = 1 - - reverse_auto_network(network, "@auto_time") - detour_impedances = shortest_paths_impedances( - network, new_mode, "@auto_time", destinations) - direct_impedances = shortest_paths_impedances( - network, sov_non_toll_mode, "@auto_time", destinations) - - percent_detour = (detour_impedances - direct_impedances) / direct_impedances - avg_percent_detour = _np.sum(percent_detour, axis=1) / len(destinations) - avg_percent_detour = _np.nan_to_num(avg_percent_detour) - return avg_percent_detour - - @_m.logbook_trace("Export results to transponderModelAccessibilities.csv file") - def export_results(self, output_directory, scenario, distances, savings, detour): - zones = scenario.zone_numbers - output_file = _join(output_directory, "transponderModelAccessibilities.csv") - with open(output_file, 'w') as f: - f.write("TAZ,DIST,AVGTTS,PCTDETOUR\n") - for row in _izip(zones, distances, savings, detour): - f.write("%d, %.4f, %.5f, %.5f\n" % row) - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg - - -def reverse_auto_network(network, link_cost): - # swap directionality of modes and specified link costs, as well as turn prohibitions - # delete all transit lines - for line in network.transit_lines(): - network.delete_transit_line(line) - - # backup modes so that turns can be swapped (auto mode remains avialable) - network.create_attribute("LINK", "backup_modes") - for link in network.links(): - link.backup_modes = link.modes - # add new reverse links (where needed) and get the one-way links to be deleted - auto_mode = network.mode("d") - links_to_delete = [] - for link in network.links(): - reverse_link = network.link(link.j_node.id, link.i_node.id) - if reverse_link is None: - reverse_link = network.create_link(link.j_node.id, link.i_node.id, link.modes) - reverse_link.backup_modes = reverse_link.modes - links_to_delete.append(link) - reverse_link.modes |= link.modes - - # reverse the turn data - visited = set([]) - for turn in network.turns(): - if turn in visited: - continue - reverse_turn = network.turn(turn.k_node, turn.j_node, turn.i_node) - time, reverse_time = turn["data1"], turn["data1"] - turn["data1"], turn["data1"] = time, reverse_time - tpf, reverse_tpf = turn.penalty_func, reverse_turn.penalty_func - reverse_turn.penalty_func, turn.penalty_func = tpf, reverse_tpf - visited.add(turn) - visited.add(reverse_turn) - - # reverse the link data - visited = set([]) - for link in network.links(): - if link in visited: - continue - reverse_link = network.link(link.j_node.id, link.i_node.id) - time, reverse_time = link[link_cost], reverse_link[link_cost] - reverse_link[link_cost], link[link_cost] = time, reverse_time - reverse_link.modes, link.modes = link.backup_modes, reverse_link.backup_modes - visited.add(link) - visited.add(reverse_link) - - # delete the one-way links - for link in links_to_delete: - network.delete_link(link.i_node, link.j_node) - - -def shortest_paths_impedances(network, mode, link_cost, destinations): - excluded_links = [] - for link in network.links(): - if mode not in link.modes: - excluded_links.append(link) - - impedances = [] - for dest_id in destinations: - tree = network.shortest_path_tree( - dest_id, link_cost, excluded_links=excluded_links, consider_turns=True) - costs = [] - for node in network.centroids(): - if node.number == dest_id: - costs.append(0) - else: - try: - path_cost = tree.cost_to_node(node.id) - except KeyError: - path_cost = 600 - costs.append(path_cost) - impedances.append(costs) - return _np.array(impedances).T - - -@_context -def setup_for_tt_savings_calc(emmebank): - with gen_utils.temp_matrices(emmebank, "FULL", 2) as mf: - mf[0].name = "NTTime" - mf[0].description = "Temp AM + PM' Auto non-transponder time" - mf[1].name = "TRTime" - mf[1].description = "Temp AM + PM' Auto transponder time" - with gen_utils.temp_matrices(emmebank, "ORIGIN", 3) as mo: - mo[0].name = "numerator" - mo[1].name = "denominator" - mo[2].name = "AVGTTS" - mo[2].description = "Temp average travel time savings" - with gen_utils.temp_matrices(emmebank, "DESTINATION", 1) as md: - md[0].name = "employment" - md[0].description = "Temp employment per zone" - yield - -@_context -def get_temp_scenario(src_scenario): - delete_scenario = _m.Modeller().tool( - "inro.emme.data.scenario.delete_scenario") - emmebank = src_scenario.emmebank - scenario_id = get_available_scenario_id(emmebank) - temp_scenario = emmebank.copy_scenario(src_scenario, scenario_id) - try: - yield temp_scenario - finally: - delete_scenario(temp_scenario) - -def get_available_mode_id(network): - for mode_id in _string.letters: - if network.mode(mode_id) is None: - return mode_id - -def get_available_scenario_id(emmebank): - for i in range(1,10000): - if not emmebank.scenario(i): - return i - -def bearing(shape): - pt1 = shape.coords[0] - pt2 = shape.coords[-1] - x_diff = pt2[0] - pt1[0] - y_diff = pt2[1] - pt1[1] - return math.degrees(math.atan2(y_diff, x_diff)) - -def close_bearing(shape1, shape2, tol=25): - b1 = bearing(shape1) - b2 = bearing(shape2) - diff = (b1 - b2) % 360 - if diff >= 180: - diff -= 360 - return abs(diff) < tol diff --git a/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py b/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py deleted file mode 100644 index 068481a..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_tap_adjacent_lines.py +++ /dev/null @@ -1,122 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export/export_tap_adjacent_lines.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports a list of transit lines adjacent to each TAP. -# -# -# Inputs: -# file_path: export path for the tap adjacency file -# scenario: scenario ID for the base scenario (same used in the Import network tool) -# -# Files created: -# output/tapLines.csv (or as specified) -# -# -# Script example: -""" -import inro.modeller as _m -import os -modeller = _m.Modeller() -desktop = modeller.desktop - -export_tap_adjacent_lines = modeller.tool("sandag.export.export_tap_adjacent_lines") - -project_dir = os.path.dirname(desktop.project_path()) -main_directory = os.path.dirname(project_dir) -output_dir = os.path.join(main_directory, "output") - -main_emmebank = os.path.join(project_dir, "Database", "emmebank") -scenario_id = 100 -base_scenario = main_emmebank.scenario(scenario_id) - -export_tap_adjacent_lines(os.path.join(output_dir, "tapLines.csv"), base_scenario) - -""" - - -TOOLBOX_ORDER = 75 - - -import inro.modeller as _m -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ExportLines(_m.Tool(), gen_utils.Snapshot): - - file_path = _m.Attribute(unicode) - - tool_run_msg = "" - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_dir = os.path.dirname(project_dir) - self.file_path = os.path.join(main_dir, "output", "tapLines.csv") - self.attributes = ["file_path"] - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export TAP adjacent lines" - pb.description = """Exports a list of the transit lines adjacent to each tap.""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('file_path', 'save_file',title='Select file path') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.file_path, scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export list of TAP adjacent lines", save_arguments=True) - def __call__(self, file_path, scenario): - attributes = {"file_path": file_path} - gen_utils.log_snapshot("Export list of TAP adjacent lines", str(self), attributes) - - network = scenario.get_partial_network( - ["NODE", "TRANSIT_LINE"], include_attributes=False) - values = scenario.get_attribute_values("NODE", ["@tap_id"]) - network.set_attribute_values("NODE", ["@tap_id"], values) - with open(file_path, 'w') as f: - f.write("TAP,LINES\n") - for node in network.nodes(): - if node["@tap_id"] == 0: - continue - lines = set([]) - for link in node.outgoing_links(): - for seg in link.j_node.outgoing_segments(include_hidden=True): - if seg.allow_alightings: - lines.add(seg.line) - if not lines: - continue - f.write("%d," % node["@tap_id"]) - f.write(" ".join([l.id for l in lines])) - f.write("\n") diff --git a/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py b/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py deleted file mode 100644 index 247eb60..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_traffic_skims.py +++ /dev/null @@ -1,95 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export/export_traffic_skims.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports the traffic skims for use in the disaggregate demand models (CT-RAMP) -# and the data loader process. -# -# Note the matrix name mapping from the OMX file names to the Emme database names. -# -# Inputs: -# omx_file: output directory to read the OMX files from -# period: the period for which to export the skim matrices, "EA", "AM", "MD", "PM", "EV" -# scenario: base traffic scenario to use for reference zone system -# -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - output_dir = os.path.join(main_directory, "output") - scenario = modeller.scenario - periods = ["EA", "AM", "MD", "PM", "EV"] - export_traffic_skims = modeller.tool("sandag.import.export_traffic_skims") - for period in periods: - omx_file_path = os.path.join(output_dir, "traffic_skims_%s.omx" % period - export_traffic_skims(output_dir, period, scenario) -""" - -TOOLBOX_ORDER = 71 - - -import inro.modeller as _m -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ExportSkims(_m.Tool(), gen_utils.Snapshot): - - omx_file = _m.Attribute(unicode) - period = _m.Attribute(str) - tool_run_msg = "" - - def __init__(self): - self.attributes = ["omx_file", "period"] - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export traffic skims" - pb.description = """Export the skim matrices to OMX format for the selected period.""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_select_file('omx_file', 'save_file', title='Select OMX file') - options = [(x, x) for x in ["EA", "AM", "MD", "PM", "EV"]] - pb.add_select("period", keyvalues=options, title="Select corresponding period") - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.period, self.omx_file, scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export traffic skims to OMX", save_arguments=True) - def __call__(self, period, omx_file, scenario): - attributes = {"omx_file": omx_file, "period": period} - gen_utils.log_snapshot("Export traffic skims to OMX", str(self), attributes) - init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") - matrices = init_matrices.get_matrix_names("traffic_skims", [period], scenario) - with gen_utils.ExportOMX(omx_file, scenario, omx_key="NAME") as exporter: - exporter.write_matrices(matrices) diff --git a/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py b/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py deleted file mode 100644 index 7c22691..0000000 --- a/sandag_abm/src/main/emme/toolbox/export/export_transit_skims.py +++ /dev/null @@ -1,108 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// export/export_transit_skims.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Exports the transit skims for use in the disaggregate demand models (CT-RAMP) -# and the data loader process. -# -# Note the matrix name mapping from the OMX file names to the Emme database names. -# -# Inputs: -# omx_file: output directory to read the OMX files from -# periods: list of periods, using the standard two-character abbreviation -# big_to_zero: replace big values (>10E6) with zero -# This is used in the final iteration skim (after the demand models are -# complete) to filter large values from the OMX files which are not -# compatible with the data loader process -# scenario: transit scenario to use for reference zone system -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - output_dir = os.path.join(main_directory, "output") - scenario = modeller.scenario - export_transit_skims = modeller.tool("sandag.import.export_transit_skims") - omx_file_path = os.path.join(output_dir, "transit_skims.omx" - export_transit_skims(output_dir, period, scenario) -""" - - -TOOLBOX_ORDER = 72 - - -import inro.modeller as _m -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ExportSkims(_m.Tool(), gen_utils.Snapshot): - omx_file = _m.Attribute(unicode) - periods = _m.Attribute(unicode) - big_to_zero = _m.Attribute(bool) - - tool_run_msg = "" - - def __init__(self): - self.attributes = ["omx_file", "periods", "big_to_zero"] - self.periods = "EA, AM, MD, PM, EV" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Export transit skim matrices" - pb.description = """Export the skim matrices to OMX format for all periods.""" - pb.branding_text = "- SANDAG - Export" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_select_file('omx_file', 'save_file', title='Select OMX file') - pb.add_text_box('periods', title="Selected periods:") - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - periods = [x.strip() for x in self.periods.split(",")] - self(self.omx_file, periods, scenario, self.big_to_zero) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Export transit skims to OMX", save_arguments=True) - def __call__(self, omx_file, periods, scenario, big_to_zero=False): - attributes = {"omx_file": omx_file, "periods": periods, "big_to_zero": big_to_zero} - gen_utils.log_snapshot("Export transit skims to OMX", str(self), attributes) - init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") - matrices = init_matrices.get_matrix_names( - "transit_skims", periods, scenario) - with gen_utils.ExportOMX(omx_file, scenario, omx_key="NAME") as exporter: - if big_to_zero: - emmebank = scenario.emmebank - for name in matrices: - matrix = emmebank.matrix(name) - array = matrix.get_numpy_data(scenario) - array[array>10E6] = 0 - exporter.write_array(array, exporter.generate_key(matrix)) - else: - exporter.write_matrices(matrices) diff --git a/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py b/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py deleted file mode 100644 index 7630a5d..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/adjust_network_links.py +++ /dev/null @@ -1,156 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/adjust_network_links.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# deletes the existing centroid connectors and create new centroid connectors -# connecting the centroid of aggregated zones to the original end points (on network) -# of old centroid connectors -# -# Inputs: -# source: path to the location of the input network files -# base_scenario: scenario that has highway network only -# emmebank: the Emme database in which to the new network is published -# external_zone: string "1-12" that refernces to range of external zones -# taz_cwk_file: input csv file created after zone aggregation. It has the crosswalk between existing TAZ to new zone structure -# cluster_zone_file: input csv file created after zone aggregation. It has the centroid coordinates of the new zone structure -# -# - -import inro.modeller as _m -import os - -import pandas as pd -from scipy.spatial import distance - - -def adjust_network_links(source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file) - - taz_cwk = pd.read_csv(os.path.join(source, taz_cwk_file), index_col = 0) - taz_cwk = taz_cwk['cluster_id'].to_dict() - - emmebank = _m.Modeller().emmebank - scenario = emmebank.scenario(base_scenario) - hwy_network = scenario.get_network() - - centroid_nodes = [] - exclude_nodes = [] - - - ext_zones = [int(s) for s in external_zone.split() if s.isdigit()] - - for node in range(ext_zones[0],ext_zones[1],1): - exclude_nodes.append(hwy_network.node(node)) - - for node in hwy_network.centroids(): - if not node in exclude_nodes: - centroid_nodes.append(node) - - i_nodes = [] - j_nodes = [] - data1 = [] - length = [] - links = [] - - for link in hwy_network.links(): - if link.i_node in centroid_nodes: - links.append(link) - i_nodes.append(int(link.i_node)) - j_nodes.append(int(link.j_node)) - data1.append(link.data1) - length.append(link.length) - - df = pd.DataFrame({'links' : links, 'i_nodes' : i_nodes, 'j_nodes': j_nodes, 'ul1_org': data1, 'length_org':length}) - df['i_nodes_new'] = df['i_nodes'].map(taz_cwk) - - #get XY of existing centroids - j_nodes_list = df['j_nodes'].unique() - j_nodes_list = [hwy_network.node(x) for x in j_nodes_list] - - j_nodes = [] - j_x = [] - j_y = [] - for nodes in hwy_network.nodes(): - if nodes in j_nodes_list: - j_nodes.append(nodes) - j_x.append(nodes.x) - j_y.append(nodes.y) - - j_nodes_XY = pd.DataFrame({'j_nodes' : j_nodes, 'j_x' : j_x, 'j_y': j_y}) - j_nodes_XY['j_nodes'] = [int(x) for x in j_nodes_XY['j_nodes']] - df = pd.merge(df, j_nodes_XY, on = 'j_nodes', how = 'left') - - agg_node_coords = pd.read_csv(os.path.join(source, cluster_zone_file)) - df = pd.merge(df, agg_node_coords, left_on = 'i_nodes_new', right_on = 'cluster_id', how = 'left') - df = df.drop(columns = 'cluster_id') - df = df.rename(columns = {'centroid_x' : 'i_new_x', 'centroid_y' : 'i_new_y'}) - - i_coords = zip(df['j_x'], df['j_y']) - j_coords = zip(df['i_new_x'], df['i_new_y']) - - df['length'] = [distance.euclidean(i, j)/5280.0 for i,j in zip(i_coords, j_coords)] - - #delete all the existing centroid nodes - for index,row in df.iterrows(): - if hwy_network.node(row['i_nodes']): - hwy_network.delete_node(row['i_nodes'], True) - - # create new nodes (centroids of clusters) - for index,row in agg_node_coords.iterrows(): - new_node = hwy_network.create_node(row['cluster_id'], is_centroid = True) - new_node.x = int(row['centroid_x']) - new_node.y = int(row['centroid_y']) - - df['type'] = 10 - df['num_lanes'] = 1 - df['vdf'] = 11 - df['ul3'] = 999999 - - final_df = df[["i_nodes_new", "j_nodes", "length", "type", "num_lanes", "vdf", "ul3"]] - final_df = final_df.drop_duplicates() - final_df = final_df.reset_index(drop=True) - final_df['type'] = final_df['type'].astype("int") - - # create new links - for index,row in final_df.iterrows(): - - link_ij = hwy_network.create_link(row['i_nodes_new'], row['j_nodes'], - modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) - link_ij.length = row['length'] - link_ij.type = row['type'].astype("int") - link_ij.num_lanes = row['num_lanes'].astype("int") - link_ij.volume_delay_func = row['vdf'].astype("int") - link_ij.data3 = row['ul3'].astype("int") - link_ij['@lane_ea'] = 1 # had to do this as they are being replaced in highway assignment by the values in these columns - link_ij['@lane_am'] = 1 - link_ij['@lane_md'] = 1 - link_ij['@lane_pm'] = 1 - link_ij['@lane_ev'] = 1 - - - link_ji = hwy_network.create_link(row['j_nodes'], row['i_nodes_new'], - modes = ["d", "h", "H", "i","I","s", "S", "v", "V", "m", "M", "t", "T"]) - link_ji.length = row['length'] - link_ji.type = row['type'].astype("int") - link_ji.num_lanes = row['num_lanes'].astype("int") - link_ji.volume_delay_func = row['vdf'].astype("int") - link_ji.data3 = row['ul3'].astype("int") - link_ji['@lane_ea'] = 1 # had to do this as they are being replaced in highway assignment by the values in these columns - link_ji['@lane_am'] = 1 - link_ji['@lane_md'] = 1 - link_ji['@lane_pm'] = 1 - link_ji['@lane_ev'] = 1 - - return(hwy_network) - #publish the highway network to the scenario - #scenario.publish_network(hwy_network) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py deleted file mode 100644 index 375a27c..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/import_auto_demand.py +++ /dev/null @@ -1,516 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/import_auto_demand.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Imports the auto demand matrices generated from an iteration of the disaggregate -# demand models (CT-RAMP) and adds the saved disaggregated demand matrices to -# generate the total auto demand in preparation for the auto assignment. -# -# Note the matrix name mapping from the OMX file names to the Emme database names. -# -# Inputs: -# external_zones: set of external zone IDs as a range "1-12" -# output_dir: output directory to read the OMX files from -# num_processors: number of processors to use in the matrix calculations -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: pp is time period, one of EA, AM, MD, PM, EV, vot is one of low, med, high -# output/autoInternalExternalTrips_pp_vot.omx -# output/autoVisitorTrips_pp_vot.omx -# output/autoCrossBorderTrips_pp_vot.omx -# output/autoAirportTrips.SAN_pp_vot.omx -# output/autoAirportTrips.CDX_pp_vot.omx (if they exist) -# output/autoTrips_pp_vot.omx -# output/othrTrips_pp.omx (added to high vot) -# output/TripMatrices.csv -# output/EmptyAVTrips.omx (added to high vot) -# output/TNCVehicleTrips_pp.omx (added to high vot) -# -# Matrix inputs: -# pp_SOVGP_EIWORK, pp_SOVGP_EINONWORK, pp_SOVTOLL_EIWORK, pp_SOVTOLL_EINONWORK, -# pp_HOV2HOV_EIWORK, pp_HOV2HOV_EINONWORK, pp_HOV2TOLL_EIWORK, pp_HOV2TOLL_EINONWORK, -# pp_HOV3HOV_EIWORK, pp_HOV3HOV_EINONWORK, pp_HOV3TOLL_EIWORK, pp_HOV3TOLL_EINONWORK -# pp_SOV_EETRIPS, pp_HOV2_EETRIPS, pp_HOV3_EETRIPS -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV, v is one of L, M, H -# pp_SOV_TR_v, pp_SOV_NT_v, pp_HOV2_v, pp_HOV3_v, pp_HOV3_v -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - output_dir = os.path.join(main_directory, "output") - external_zones = "1-12" - num_processors = "MAX-1" - base_scenario = modeller.scenario - import_auto_demand = modeller.tool("sandag.import.import_auto_demand") - import_auto_demand(external_zones, output_dir, num_processors, base_scenario) -""" - -TOOLBOX_ORDER = 13 - - -import inro.modeller as _m -import traceback as _traceback -import pandas as _pandas -import os -import numpy -from contextlib import contextmanager as _context - -_join = os.path.join - -dem_utils = _m.Modeller().module('sandag.utilities.demand') -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ImportMatrices(_m.Tool(), gen_utils.Snapshot): - - external_zones = _m.Attribute(str) - output_dir = _m.Attribute(unicode) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.external_zones = "1-12" - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_dir = os.path.dirname(project_dir) - self.main_dir = main_dir - self.output_dir = os.path.join(main_dir, "output") - self.num_processors = "MAX-1" - self.attributes = ["external_zones", "output_dir", "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Import auto demand and sum matrices" - pb.description = """ -
    - Imports the trip matrices generated by CT-RAMP in OMX format, - the commercial vehicle demand in CSV format, - and adds the demand from the aggregate models for the final - trip assignments.
    - A total of 90 OMX files are expected, for 5 time periods - EA, AM, MD, PM and EV, and value-of-time level low, med or high, - with internal matrices by SOV, HOV2, HOV3+ and toll access type: -
      -
    • autoInternalExternalTrips_pp_vot.omx
    • -
    • autoVisitorTrips_pp_vot.omx
    • -
    • autoCrossBorderTrips_pp_vot.omx
    • -
    • autoAirportTrips.SAN_pp_vot.omx
    • -
    • autoAirportTrips.CDX_pp_vot.omx (optional)
    • -
    • autoTrips_pp_vot.omx
    • -
    • othrTrips_pp.omx (added to high vot)
    • -
    • EmptyAVTrips.omx (added to high vot)
    • -
    • TNCVehicleTrips_pp.omx (added to high vot)
    • -
    - As well as one CSV file "TripMatrices.csv" for the commercial vehicle trips. - Adds the aggregate demand from the - external-external and external-internal demand matrices: -
      -
    • pp_SOVGP_EETRIPS, pp_HOV2HOV_EETRIPS, pp_HOV3HOV_EETRIPS
    • -
    • pp_SOVGP_EIWORK, pp_SOVGP_EINONWORK, pp_SOVTOLL_EIWORK, pp_SOVTOLL_EINONWORK
    • -
    • pp_HOV2HOV_EIWORK, pp_HOV2HOV_EINONWORK, pp_HOV2TOLL_EIWORK, pp_HOV2TOLL_EINONWORK
    • -
    • pp_HOV3HOV_EIWORK, pp_HOV3HOV_EINONWORK, pp_HOV3TOLL_EIWORK, pp_HOV3TOLL_EINONWORK
    • -
    - to the time-of-day total demand matrices. -
    -
    - """ - pb.branding_text = "- SANDAG - Model" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_select_file('output_dir', 'directory', - title='Select output directory') - pb.add_text_box("external_zones", title="External zones:") - dem_utils.add_select_processors("num_processors", pb, self) - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.output_dir, self.external_zones, self.num_processors, scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Create TOD auto trip tables", save_arguments=True) - def __call__(self, output_dir, external_zones, num_processors, scenario): - attributes = { - "output_dir": output_dir, - "external_zones": external_zones, - "num_processors": num_processors} - gen_utils.log_snapshot("Create TOD auto trip tables", str(self), attributes) - - #get parameters from sandag_abm.properties - modeller = _m.Modeller() - load_properties = modeller.tool('sandag.utilities.properties') - props = load_properties(_join(self.main_dir, "conf", "sandag_abm.properties")) - - self.scenario = scenario - self.output_dir = output_dir - self.external_zones = external_zones - self.num_processors = num_processors - self.import_traffic_trips(props) - self.import_commercial_vehicle_demand(props) - #self.convert_light_trucks_to_pce() - self.add_aggregate_demand() - - @_context - def setup(self): - emmebank = self.scenario.emmebank - self._matrix_cache = {} - with gen_utils.OMXManager(self.output_dir, "%sTrips%s%s.omx") as omx_manager: - try: - yield omx_manager - finally: - for name, value in self._matrix_cache.iteritems(): - matrix = emmebank.matrix(name) - matrix.set_numpy_data(value, self.scenario.id) - - def set_data(self, name, value): - if name in self._matrix_cache: - value = value + self._matrix_cache[name] - self._matrix_cache[name] = value - - @_m.logbook_trace("Import CT-RAMP traffic trips from OMX") - def import_traffic_trips(self, props): - title = "Import CT-RAMP traffic trips from OMX report" - report = _m.PageBuilder(title) - - taxi_da_share = props["Taxi.da.share"] - taxi_s2_share = props["Taxi.s2.share"] - taxi_s3_share = props["Taxi.s3.share"] - taxi_pce = props["Taxi.passengersPerVehicle"] - tnc_single_da_share = props["TNC.single.da.share"] - tnc_single_s2_share = props["TNC.single.s2.share"] - tnc_single_s3_share = props["TNC.single.s3.share"] - tnc_single_pce = props["TNC.single.passengersPerVehicle"] - tnc_shared_da_share = props["TNC.shared.da.share"] - tnc_shared_s2_share = props["TNC.shared.s2.share"] - tnc_shared_s3_share = props["TNC.shared.s3.share"] - tnc_shared_pce = props["TNC.shared.passengersPerVehicle"] - av_share = props["Mobility.AV.Share"] - - periods = ["_EA", "_AM", "_MD", "_PM", "_EV"] - vot_bins = ["_low", "_med", "_high"] - mode_shares = [ - ("mf%s_SOV_TR_H", { - "TAXI": taxi_da_share / taxi_pce - }), - ("mf%s_HOV2_H", { - "TAXI": taxi_s2_share / taxi_pce - }), - ("mf%s_HOV3_H", { - "TAXI": taxi_s3_share / taxi_pce - }), - ] - - with self.setup() as omx_manager: - # SOV transponder "TRPDR" = "TR" and non-transponder "NOTRPDR" = "NT" - for period in periods: - for vot in vot_bins: - # SOV non-transponder demand - matrix_name = "mf%s_SOV_NT_%s" % (period[1:], vot[1].upper()) - logbook_label = "Import auto from OMX SOVNOTRPDR to matrix %s" % (matrix_name) - resident_demand = omx_manager.lookup(("auto", period, vot), "SOVNOTRPDR%s" % period) - visitor_demand = omx_manager.lookup(("autoVisitor", period, vot), "SOV%s" % period) - cross_border_demand = omx_manager.lookup(("autoCrossBorder", period, vot), "SOV%s" % period) - # NOTE: No non-transponder airport or internal-external demand - total_ct_ramp_trips = ( - resident_demand + visitor_demand + cross_border_demand) - dem_utils.demand_report([ - ("resident", resident_demand), - ("cross_border", cross_border_demand), - ("visitor", visitor_demand), - ("total", total_ct_ramp_trips) - ], - logbook_label, self.scenario, report) - self.set_data(matrix_name, total_ct_ramp_trips) - - # SOV transponder demand - matrix_name = "mf%s_SOV_TR_%s" % (period[1:], vot[1].upper()) - logbook_label = "Import auto from OMX SOVTRPDR to matrix %s" % (matrix_name) - resident_demand = omx_manager.lookup(("auto", period, vot), "SOVTRPDR%s" % period) - # NOTE: No transponder visitor or cross-border demand - airport_demand = omx_manager.lookup(("autoAirport", ".SAN" + period, vot), "SOV%s" % period) - if omx_manager.file_exists(("autoAirport", ".CBX" + period, vot)): - airport_demand += omx_manager.lookup(("autoAirport", ".CBX" + period, vot), "SOV%s" % period) - internal_external_demand = omx_manager.lookup(("autoInternalExternal", period, vot), "SOV%s" % period) - - total_ct_ramp_trips = ( - resident_demand + airport_demand + internal_external_demand) - dem_utils.demand_report([ - ("resident", resident_demand), - ("airport", airport_demand), - ("internal_external", internal_external_demand), - ("total", total_ct_ramp_trips) - ], - logbook_label, self.scenario, report) - self.set_data(matrix_name, total_ct_ramp_trips) - - # HOV2 and HOV3 demand - matrix_name_map = [ - ("mf%s_HOV2_%s", "SR2%s"), - ("mf%s_HOV3_%s", "SR3%s") - ] - for matrix_name_tmplt, omx_name in matrix_name_map: - matrix_name = matrix_name_tmplt % (period[1:], vot[1].upper()) - logbook_label = "Import auto from OMX %s to matrix %s" % (omx_name[:3], matrix_name) - resident_demand = ( - omx_manager.lookup(("auto", period, vot), omx_name % ("TRPDR" + period)) - + omx_manager.lookup(("auto", period, vot), omx_name % ("NOTRPDR" + period))) - visitor_demand = omx_manager.lookup(("autoVisitor", period, vot), omx_name % period) - cross_border_demand = omx_manager.lookup(("autoCrossBorder", period, vot), omx_name % period) - airport_demand = omx_manager.lookup(("autoAirport", ".SAN" + period, vot), omx_name % period) - if omx_manager.file_exists(("autoAirport", ".CBX" + period, vot)): - airport_demand += omx_manager.lookup(("autoAirport", ".CBX" + period, vot), omx_name % period) - internal_external_demand = omx_manager.lookup(("autoInternalExternal", period, vot), omx_name % period) - - total_ct_ramp_trips = ( - resident_demand + visitor_demand + cross_border_demand + airport_demand + internal_external_demand) - dem_utils.demand_report([ - ("resident", resident_demand), - ("cross_border", cross_border_demand), - ("visitor", visitor_demand), - ("airport", airport_demand), - ("internal_external", internal_external_demand), - ("total", total_ct_ramp_trips) - ], - logbook_label, self.scenario, report) - self.set_data(matrix_name, total_ct_ramp_trips) - - # add TNC and TAXI demand to vot="high" - for matrix_name_tmplt, share in mode_shares: - matrix_name = matrix_name_tmplt % period[1:] - logbook_label = "Import othr from TAXI, empty AV, and TNC to matrix %s" % (matrix_name) - resident_taxi_demand = ( - omx_manager.lookup(("othr", period, ""), "TAXI" + period) * share["TAXI"]) - visitor_taxi_demand = ( - omx_manager.lookup(("othrVisitor", period, ""), "TAXI" + period) * share["TAXI"]) - cross_border_taxi_demand = ( - omx_manager.lookup(("othrCrossBorder", period, ""), "TAXI" + period) * share["TAXI"]) - # airport SAN - airport_taxi_demand = ( - omx_manager.lookup(("othrAirport", ".SAN", period), "TAXI" + period) * share["TAXI"]) - # airport CBX (optional) - if omx_manager.file_exists(("othrAirport", ".CBX", period)): - airport_taxi_demand += ( - omx_manager.lookup(("othrAirport",".CBX", period), "TAXI" + period) * share["TAXI"]) - internal_external_taxi_demand = ( - omx_manager.lookup(("othrInternalExternal", period, ""), "TAXI" + period) * share["TAXI"]) - - #AV routing models and TNC fleet model demand - empty_av_demand = omx_manager.lookup(("EmptyAV","",""), "EmptyAV%s" % period) - tnc_demand_0 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_0" % period) - tnc_demand_1 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_1" % period) - tnc_demand_2 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_2" % period) - tnc_demand_3 = omx_manager.lookup(("TNCVehicle","",period), "TNC%s_3" % period) - - #AVs: no driver. No AVs: driver - #AVs: 0 and 1 passenger would be SOV. there will be empty vehicles as well. No AVs: 0 passanger would be SOV - #AVs: 2 passenger would be HOV2. No AVs: 1 passenger would be HOV2 - #AVs: 3 passenger would be HOV3. No AVs: 2 and 3 passengers would be HOV3 - if (av_share>0): - if (matrix_name_tmplt[5:-2] == "SOV_TR"): - av_demand = empty_av_demand + tnc_demand_0 + tnc_demand_1 - elif (matrix_name_tmplt[5:-2] == "HOV2"): - av_demand = tnc_demand_2 - else: - av_demand = tnc_demand_3 - else: - if (matrix_name_tmplt[5:-2] == "SOV_TR"): - av_demand = tnc_demand_0 - elif (matrix_name_tmplt[5:-2] == "HOV2"): - av_demand = tnc_demand_1 - else: - av_demand = tnc_demand_2 + tnc_demand_3 - - total_ct_ramp_trips = ( - resident_taxi_demand + visitor_taxi_demand + cross_border_taxi_demand - + airport_taxi_demand + internal_external_taxi_demand + av_demand) - dem_utils.demand_report([ - ("resident_taxi", resident_taxi_demand), - ("visitor_taxi", visitor_taxi_demand), - ("cross_border_taxi", cross_border_taxi_demand), - ("airport_taxi", airport_taxi_demand), - ("internal_external_taxi", internal_external_taxi_demand), - ("av_fleet", av_demand), - ("total", total_ct_ramp_trips) - ], - logbook_label, self.scenario, report) - self.set_data(matrix_name, total_ct_ramp_trips) - _m.logbook_write(title, report.render()) - - @_m.logbook_trace('Import commercial vehicle demand') - def import_commercial_vehicle_demand(self, props): - scale_factor = props["cvm.scale_factor"] - scale_light = props["cvm.scale_light"] - scale_medium = props["cvm.scale_medium"] - scale_heavy = props["cvm.scale_heavy"] - share_light = props["cvm.share.light"] - share_medium = props["cvm.share.medium"] - share_heavy = props["cvm.share.heavy"] - - scenario = self.scenario - emmebank = scenario.emmebank - - mapping = {} - periods = ["EA", "AM", "MD", "PM", "EV"] - # The SOV demand is modified in-place, which was imported - # prior from the CT-RAMP demand - # The truck demand in vehicles is copied from separate matrices - for index, period in enumerate(periods): - mapping["CVM_%s:LNT" % period] = { - "orig": "%s_SOV_TR_H" % period, - "dest": "%s_SOV_TR_H" % period, - "pce": 1.0, - "scale": scale_light[index], - "share": share_light, - "period": period - } - mapping["CVM_%s:INT" % period] = { - "orig": "%s_TRK_L_VEH" % period, - "dest": "%s_TRK_L" % period, - "pce": 1.3, - "scale": scale_medium[index], - "share": share_medium, - "period": period - } - mapping["CVM_%s:MNT" % period] = { - "orig": "%s_TRK_M_VEH" % period, - "dest": "%s_TRK_M" % period, - "pce": 1.5, - "scale": scale_medium[index], - "share": share_medium, - "period": period - } - mapping["CVM_%s:HNT" % period] = { - "orig": "%s_TRK_H_VEH" % period, - "dest": "%s_TRK_H" % period, - "pce": 2.5, - "scale": scale_heavy[index], - "share": share_heavy, - "period": period - } - with _m.logbook_trace('Load starting SOV and truck matrices'): - for key, value in mapping.iteritems(): - value["array"] = emmebank.matrix(value["orig"]).get_numpy_data(scenario) - - with _m.logbook_trace('Processing CVM from TripMatrices.csv'): - path = os.path.join(self.output_dir, "TripMatrices.csv") - table = _pandas.read_csv(path) - for key, value in mapping.iteritems(): - cvm_array = table[key].values.reshape((4996, 4996)) # reshape method deprecated since v 0.19.0, yma, 2/12/2019 - #factor in cvm demand by the scale factor used in trip generation - cvm_array = cvm_array/scale_factor - #scale trips to take care of underestimation - cvm_array = cvm_array * value["scale"] - - #add remaining share to the correspnding truck matrix - value["array"] = value["array"] + (cvm_array * (1-value["share"])) - - #add cvm truck vehicles to light-heavy trucks - for key, value in mapping.iteritems(): - period = value["period"] - cvm_vehs = ['L','M','H'] - if key == "CVM_%s:INT" % period: - for veh in cvm_vehs: - key_new = "CVM_%s:%sNT" % (period, veh) - value_new = mapping[key_new] - if value_new["share"] != 0.0: - cvm_array = table[key_new].values.reshape((4996, 4996)) - cvm_array = cvm_array/scale_factor - cvm_array = cvm_array * value_new["scale"] - value["array"] = value["array"] + (cvm_array * value_new["share"]) - matrix_unique = {} - with _m.logbook_trace('Save SOV matrix and convert CV and truck vehicle demand to PCEs for assignment'): - for key, value in mapping.iteritems(): - matrix = emmebank.matrix(value["dest"]) - array = value["array"] * value["pce"] - if (matrix in matrix_unique.keys()): - array = array + emmebank.matrix(value["dest"]).get_numpy_data(scenario) - matrix.set_numpy_data(array, scenario) - matrix_unique[matrix] = 1 - - @_m.logbook_trace('Convert light truck vehicle demand to PCEs for assignment') - def convert_light_trucks_to_pce(self): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - # Calculate PCEs for trucks - periods = ["EA", "AM", "MD", "PM", "EV"] - mat_trucks = ['TRK_L'] - pce_values = [1.3] - for period in periods: - with matrix_calc.trace_run("Period %s" % period): - for name, pce in zip(mat_trucks, pce_values): - demand_name = 'mf%s_%s' % (period, name) - matrix_calc.add(demand_name, '(%s_VEH * %s).max.0' % (demand_name, pce)) - - @_m.logbook_trace('Add aggregate demand') - def add_aggregate_demand(self): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - periods = ["EA", "AM", "MD", "PM", "EV"] - vots = ["L", "M", "H"] - # External-internal trips DO have transponder - # all SOV trips go to SOVTP - with matrix_calc.trace_run("Add external-internal trips to auto demand"): - modes = ["SOVGP", "SOVTOLL", "HOV2HOV", "HOV2TOLL", "HOV3HOV", "HOV3TOLL"] - modes_assign = {"SOVGP": "SOV_TR", - "SOVTOLL": "SOV_TR", - "HOV2HOV": "HOV2", - "HOV2TOLL": "HOV2", - "HOV3HOV": "HOV3", - "HOV3TOLL": "HOV3"} - for period in periods: - for mode in modes: - for vot in vots: - # Segment imported demand into 3 equal parts for VOT Low/Med/High - assign_mode = modes_assign[mode] - params = {'p': period, 'm': mode, 'v': vot, 'am': assign_mode} - matrix_calc.add("mf%s_%s_%s" % (period, assign_mode, vot), - "mf%(p)s_%(am)s_%(v)s " - "+ (1.0/3.0)*mf%(p)s_%(m)s_EIWORK " - "+ (1.0/3.0)*mf%(p)s_%(m)s_EINONWORK" % params) - - # External - external faster with single-processor as number of O-D pairs is so small (12 X 12) - # External-external trips do not have transpnder - # all SOV trips go to SOVNTP - matrix_calc.num_processors = 0 - with matrix_calc.trace_run("Add external-external trips to auto demand"): - modes = ["SOV", "HOV2", "HOV3"] - for period in periods: - for mode in modes: - for vot in vots: - # Segment imported demand into 3 equal parts for VOT Low/Med/High - params = {'p': period, 'm': mode, 'v': vot} - if (mode == "SOV"): - matrix_calc.add( - "mf%(p)s_%(m)s_NT_%(v)s" % params, - "mf%(p)s_%(m)s_NT_%(v)s + (1.0/3.0)*mf%(p)s_%(m)s_EETRIPS" % params, - {"origins": self.external_zones, "destinations": self.external_zones}) - else: - matrix_calc.add( - "mf%(p)s_%(m)s_%(v)s" % params, - "mf%(p)s_%(m)s_%(v)s + (1.0/3.0)*mf%(p)s_%(m)s_EETRIPS" % params, - {"origins": self.external_zones, "destinations": self.external_zones}) diff --git a/sandag_abm/src/main/emme/toolbox/import/import_network.py b/sandag_abm/src/main/emme/toolbox/import/import_network.py deleted file mode 100644 index e068a73..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/import_network.py +++ /dev/null @@ -1,2151 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/import_network.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Imports the network from the input network files. -# -# -# Inputs: -# source: path to the location of the input network files -# traffic_scenario_id: optional scenario to store the imported network from the traffic files only -# transit_scenario_id: optional scenario to store the imported network from the transit files only -# merged_scenario_id: scenario to store the combined traffic and transit data from all network files -# title: the title to use for the imported scenario -# save_data_tables: if checked, create a data table for each reference file for viewing in the Emme Desktop -# data_table_name: prefix to use to identify all data tables -# overwrite: check to overwrite any existing data tables or scenarios with the same ID or name -# emmebank: the Emme database in which to create the scenario. Default is the current open database -# -# Files referenced: -# hwycov.e00: base nodes and links for traffic network with traffic attributes in ESRI input exchange format -# linktypeturns.dbf: fixed turn travel times by to/from link type (field IFC) pairs -# turns.csv: turn bans and fixed costs by link from/to ID (field HWYCOV-ID) -# trcov.e00: base nodes and links for transit network in ESRI input exchange format -# trrt.csv: transit routes and their attributes -# trlink.csv: itineraries for each route as sequence of link IDs (TRCOV-ID field) -# trstop.csv: transit stop attributes -# timexfer_period.csv: table of timed transfer pairs of lines, by period -# mode5tod.csv: global (per-mode) transit cost and perception attributes -# special_fares.txt: table listing special fares in terms of boarding and incremental in-vehicle costs. -# off_peak_toll_factors.csv (optional): factors to calculate the toll for EA, MD, and EV periods from the OP toll input for specified facilities -# vehicle_class_toll_factors.csv (optional): factors to adjust the toll cost by facility name and class (DA, S2, S3, TRK_L, TRK_M, TRK_H) -# -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - source_dir = os.path.join(main_directory, "input") - title = "Base 2012 scenario" - import_network = modeller.tool("sandag.import.import_network") - import_network(output_dir, merged_scenario_id=100, title=title, - data_table_name="2012_base", overwrite=True) -""" - - -TOOLBOX_ORDER = 11 - - -import inro.modeller as _m -import inro.emme.datatable as _dt -import inro.emme.network as _network -from inro.emme.core.exception import Error as _NetworkError - -from itertools import izip as _izip -from collections import defaultdict as _defaultdict, OrderedDict -from contextlib import contextmanager as _context -import fiona as _fiona - -from math import ceil as _ceiling -from copy import deepcopy as _copy -import numpy as _np -import heapq as _heapq - -import traceback as _traceback -import os - -import pandas as pd -from scipy.spatial import distance - -_join = os.path.join -_dir = os.path.dirname - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - -FILE_NAMES = { - "FARES": "special_fares.txt", - "OFF_PEAK": "off_peak_toll_factors.csv", - "VEHICLE_CLASS": "vehicle_class_toll_factors.csv" -} - - -class ImportNetwork(_m.Tool(), gen_utils.Snapshot): - - source = _m.Attribute(unicode) - traffic_scenario_id = _m.Attribute(int) - transit_scenario_id = _m.Attribute(int) - merged_scenario_id = _m.Attribute(int) - overwrite = _m.Attribute(bool) - title = _m.Attribute(unicode) - save_data_tables = _m.Attribute(bool) - data_table_name = _m.Attribute(unicode) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self._log = [] - self._error = [] - project_dir = _dir(_m.Modeller().desktop.project.path) - self.source = _join(_dir(project_dir), "input") - self.overwrite = False - self.title = "" - self.data_table_name = "" - self.attributes = [ - "source", "traffic_scenario_id", "transit_scenario_id", "merged_scenario_id", - "overwrite", "title", "save_data_tables", "data_table_name"] - - def page(self): - if not self.data_table_name: - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - self.data_table_name = props["scenarioYear"] - - pb = _m.ToolPageBuilder(self) - pb.title = "Import network" - pb.description = """ - Create an Emme network from the E00 and associated files - generated from TCOVED. - The timed transfer is stored in data tables with the suffix "_timed_xfers_period". -
    -
    -
    - The following files are used: -
      -
    • hwycov.e00
    • -
    • LINKTYPETURNS.DBF
    • -
    • turns.csv
    • -
    • trcov.e00
    • -
    • trrt.csv
    • -
    • trlink.csv
    • -
    • trstop.csv
    • -
    • timexfer_period.csv, where period = EA,AM,MD,PM,EV
    • -
    • MODE5TOD.csv
    • -
    • special_fares.txt
    • -
    • off_peak_toll_factors.csv (optional)
    • -
    • vehicle_class_toll_factors.csv (optional)
    • -
    -
    - """ - pb.branding_text = "- SANDAG - Import" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file("source", window_type="directory", file_filter="", - title="Source directory:",) - - pb.add_text_box("traffic_scenario_id", size=6, title="Scenario ID for traffic (optional):") - pb.add_text_box("transit_scenario_id", size=6, title="Scenario ID for transit (optional):") - pb.add_text_box("merged_scenario_id", size=6, title="Scenario ID for merged network:") - pb.add_text_box("title", size=80, title="Scenario title:") - pb.add_checkbox("save_data_tables", title=" ", label="Save reference data tables of file data") - pb.add_text_box("data_table_name", size=80, title="Name for data tables:", - note="Prefix name to use for all saved data tables") - pb.add_checkbox("overwrite", title=" ", label="Overwrite existing scenarios and data tables") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self.emmebank = _m.Modeller().emmebank - with self.setup(): - self.execute() - run_msg = "Network import complete" - if self._error: - run_msg += " with %s non-fatal errors. See logbook for details" % len(self._error) - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc()) - raise - - def __call__(self, source, - traffic_scenario_id=None, transit_scenario_id=None, merged_scenario_id=None, - title="", save_data_tables=False, data_table_name="", overwrite=False, - emmebank=None): - - self.source = source - self.traffic_scenario_id = traffic_scenario_id - self.transit_scenario_id = transit_scenario_id - self.merged_scenario_id = merged_scenario_id - self.title = title - self.save_data_tables = save_data_tables - self.data_table_name = data_table_name - self.overwrite = overwrite - if not emmebank: - self.emmebank = _m.Modeller().emmebank - else: - self.emmebank = emmebank - - with self.setup(): - self.execute() - - return self.emmebank.scenario(merged_scenario_id) - - @_context - def setup(self): - self._log = [] - self._error = [] - fatal_error = False - attributes = OrderedDict([ - ("self", str(self)), - ("source", self.source), - ("traffic_scenario_id", self.traffic_scenario_id), - ("transit_scenario_id", self.transit_scenario_id), - ("merged_scenario_id", self.merged_scenario_id), - ("title", self.title), - ("save_data_tables", self.save_data_tables), - ("data_table_name", self.data_table_name), - ("overwrite", self.overwrite), - ]) - self._log = [{ - "content": attributes.items(), - "type": "table", "header": ["name", "value"], - "title": "Tool input values" - }] - with _m.logbook_trace("Import network", attributes=attributes) as trace: - gen_utils.log_snapshot("Import network", str(self), attributes) - try: - yield - except Exception as error: - self._log.append({"type": "text", "content": error}) - trace_text = _traceback.format_exc().replace("\n", "
    ") - self._log.append({"type": "text", "content": trace_text}) - self._error.append(error) - fatal_error = True - raise - finally: - self.log_report() - if self._error: - if fatal_error: - trace.write("Import network failed (%s errors)" % len(self._error), attributes=attributes) - else: - trace.write("Import network completed (%s non-fatal errors)" % len(self._error), attributes=attributes) - - def execute(self): - traffic_attr_map = { - "NODE": { - "interchange": ("@interchange", "DERIVED", "EXTRA", "is interchange node") - }, - "LINK": OrderedDict([ - ("HWYCOV-ID", ("@tcov_id", "TWO_WAY", "EXTRA", "SANDAG-assigned link ID")), - ("SPHERE", ("@sphere", "TWO_WAY", "EXTRA", "Jurisdiction sphere of influence")), - ("NM", ("#name", "TWO_WAY", "STRING", "Street name")), - ("FXNM", ("#name_from", "TWO_WAY", "STRING", "Cross street at the FROM end")), - ("TXNM", ("#name_to", "TWO_WAY", "STRING", "Cross street name at the TO end")), - ("DIR", ("@direction_cardinal", "TWO_WAY", "EXTRA", "Link direction")), - ("ASPD", ("@speed_adjusted", "TWO_WAY", "EXTRA", "Adjusted link speed (miles/hr)")), - ("IYR", ("@year_open_traffic", "TWO_WAY", "EXTRA", "The year the link opened to traffic")), - ("IPROJ", ("@project_code", "TWO_WAY", "EXTRA", "Project number for use with hwyproj.xls")), - ("IJUR", ("@jurisdiction_type", "TWO_WAY", "EXTRA", "Link jurisdiction type")), - ("IFC", ("type", "TWO_WAY", "STANDARD", "")), - ("IHOV", ("@lane_restriction", "TWO_WAY", "EXTRA", "Link operation type")), - ("ITRUCK", ("@truck_restriction", "TWO_WAY", "EXTRA", "Truck restriction code (ITRUCK)")), - ("ISPD", ("@speed_posted", "TWO_WAY", "EXTRA", "Posted speed limit (mph)")), - ("IMED", ("@median", "TWO_WAY", "EXTRA", "Median type")), - ("AU", ("@lane_auxiliary", "ONE_WAY", "EXTRA", "Number of auxiliary lanes")), - ("CNT", ("@traffic_control", "ONE_WAY", "EXTRA", "Intersection control type")), - ("TL", ("@turn_thru", "ONE_WAY", "EXTRA", "Intersection approach through lanes")), - ("RL", ("@turn_right", "ONE_WAY", "EXTRA", "Intersection approach right-turn lanes")), - ("LL", ("@turn_left", "ONE_WAY", "EXTRA", "Intersection approach left-turn lanes")), - ("GC", ("@green_to_cycle_init", "ONE_WAY", "EXTRA", "Initial green-to-cycle ratio")), - ("CHO", ("@capacity_hourly_op", "ONE_WAY", "EXTRA", "Off-Peak hourly mid-link capacity")), - ("CHA", ("@capacity_hourly_am", "ONE_WAY", "EXTRA", "AM Peak hourly mid-link capacity")), - ("CHP", ("@capacity_hourly_pm", "ONE_WAY", "EXTRA", "PM Peak hourly mid-link capacity")), - # These attributes are expanded from 3 time periods to 5 - ("ITOLLO", ("toll_op", "TWO_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("ITOLLA", ("toll_am", "TWO_WAY", "INTERNAL", "")), - ("ITOLLP", ("toll_pm", "TWO_WAY", "INTERNAL", "")), - ("LNO", ("lane_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("LNA", ("lane_am", "ONE_WAY", "INTERNAL", "")), - ("LNP", ("lane_pm", "ONE_WAY", "INTERNAL", "")), - ("CPO", ("capacity_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("CPA", ("capacity_link_am", "ONE_WAY", "INTERNAL", "")), - ("CPP", ("capacity_link_pm", "ONE_WAY", "INTERNAL", "")), - ("CXO", ("capacity_inter_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("CXA", ("capacity_inter_am", "ONE_WAY", "INTERNAL", "")), - ("CXP", ("capacity_inter_pm", "ONE_WAY", "INTERNAL", "")), - ("TMO", ("time_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("TMA", ("time_link_am", "ONE_WAY", "INTERNAL", "")), - ("TMP", ("time_link_pm", "ONE_WAY", "INTERNAL", "")), - ("TXO", ("time_inter_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("TXA", ("time_inter_am", "ONE_WAY", "INTERNAL", "")), - ("TXP", ("time_inter_pm", "ONE_WAY", "INTERNAL", "")), - # These three attributes are used to cross-reference the turn directions - ("TLB", ("through_link", "ONE_WAY", "INTERNAL", "")), - ("RLB", ("right_link", "ONE_WAY", "INTERNAL", "")), - ("LLB", ("left_link", "ONE_WAY", "INTERNAL", "")), - ("@cost_operating", ("@cost_operating","DERIVED", "EXTRA", "Fuel and maintenance cost")), - ("INTDIST_UP", ("@intdist_up", "DERIVED", "EXTRA", "Upstream major intersection distance")), - ("INTDIST_DOWN", ("@intdist_down", "DERIVED", "EXTRA", "Downstream major intersection distance")), - ]) - } - time_period_attrs = OrderedDict([ - ("@cost_auto", "toll + cost autos"), - ("@cost_hov2", "toll (non-mngd) + cost HOV2"), - ("@cost_hov3", "toll (non-mngd) + cost HOV3+"), - ("@cost_lgt_truck", "toll + cost light trucks"), - ("@cost_med_truck", "toll + cost medium trucks"), - ("@cost_hvy_truck", "toll + cost heavy trucks"), - ("@cycle", "cycle length (minutes)"), - ("@green_to_cycle", "green to cycle ratio"), - ("@capacity_link", "mid-link capacity"), - ("@capacity_inter", "approach capacity"), - ("@toll", "toll cost (cent)"), - ("@lane", "number of lanes"), - ("@time_link", "link time in minutes"), - ("@time_inter", "intersection delay time"), - ("@sta_reliability", "static reliability") - ]) - time_name = { - "_ea": "Early AM ", "_am": "AM Peak ", "_md": "Mid-day ", "_pm": "PM Peak ", "_ev": "Evening " - } - time_periods = ["_ea", "_am", "_md", "_pm", "_ev"] - for attr, desc_tmplt in time_period_attrs.iteritems(): - for time in time_periods: - traffic_attr_map["LINK"][attr + time] = \ - (attr + time, "DERIVED", "EXTRA", time_name[time] + desc_tmplt) - - transit_attr_map = { - "NODE": OrderedDict([ - ("@tap_id", ("@tap_id", "DERIVED", "EXTRA", "Transit-access point ID")), - ]), - "LINK": OrderedDict([ - ("TRCOV-ID", ("@tcov_id", "TWO_WAY", "EXTRA", "SANDAG-assigned link ID")), - ("NM", ("#name", "TWO_WAY", "STRING", "Street name")), - ("FXNM", ("#name_from", "TWO_WAY", "STRING", "Cross street at the FROM end")), - ("TXNM", ("#name_to", "TWO_WAY", "STRING", "Cross street name at the TO end")), - ("DIR", ("@direction_cardinal", "TWO_WAY", "EXTRA", "Link direction")), - ("OSPD", ("@speed_observed", "TWO_WAY", "EXTRA", "Observed speed")), - ("IYR", ("@year_open_traffic", "TWO_WAY", "EXTRA", "The year the link opened to traffic ")), - ("IFC", ("type", "TWO_WAY", "STANDARD", "")), - ("IHOV", ("@lane_restriction_tr", "TWO_WAY", "EXTRA", "Link operation type")), - ("ISPD", ("@speed_posted_tr_l", "TWO_WAY", "EXTRA", "Posted speed limit (mph)")), - ("IMED", ("@median", "TWO_WAY", "EXTRA", "Median type")), - ("TMO", ("trtime_link_op", "ONE_WAY", "INTERNAL", "Expanded to EA, MD and EV")), - ("TMEA", ("@trtime_link_ea", "DERIVED", "EXTRA", "Early AM transit link time in minutes")), - ("TMA", ("@trtime_link_am", "ONE_WAY", "EXTRA", "AM Peak transit link time in minutes")), - ("TMMD", ("@trtime_link_md", "DERIVED", "EXTRA", "Mid-day transit link time in minutes")), - ("TMP", ("@trtime_link_pm", "ONE_WAY", "EXTRA", "PM Peak transit link time in minutes")), - ("TMEV", ("@trtime_link_ev", "DERIVED", "EXTRA", "Evening transit link time in minutes")), - ("MINMODE", ("@mode_hierarchy", "TWO_WAY", "EXTRA", "Transit mode type")), - ]), - "TRANSIT_LINE": OrderedDict([ - ("AM_Headway", ("@headway_am", "TRRT", "EXTRA", "AM Peak actual headway")), - ("PM_Headway", ("@headway_pm", "TRRT", "EXTRA", "PM Peak actual headway")), - ("OP_Headway", ("@headway_op", "TRRT", "EXTRA", "Off-Peak actual headway")), - ("Night_Headway", ("@headway_night", "TRRT", "EXTRA", "Night actual headway")), - ("AM_Headway_rev", ("@headway_rev_am", "DERIVED", "EXTRA", "AM Peak revised headway")), - ("PM_Headway_rev", ("@headway_rev_pm", "DERIVED", "EXTRA", "PM Peak revised headway")), - ("OP_Headway_rev", ("@headway_rev_op", "DERIVED", "EXTRA", "Off-Peak revised headway")), - ("WT_IVTPK", ("@vehicle_per_pk", "MODE5TOD", "EXTRA", "Peak in-vehicle perception factor")), - ("WT_IVTOP", ("@vehicle_per_op", "MODE5TOD", "EXTRA", "Off-Peak in-vehicle perception factor")), - ("WT_FAREPK", ("@fare_per_pk", "MODE5TOD", "EXTRA", "Peak fare perception factor")), - ("WT_FAREOP", ("@fare_per_op", "MODE5TOD", "EXTRA", "Off-Peak fare perception factor")), - ("DWELLTIME", ("default_dwell_time", "MODE5TOD", "INTERNAL", "")), - ("Fare", ("@fare", "TRRT", "EXTRA", "Boarding fare ($)")), - ("@transfer_penalty",("@transfer_penalty","DERIVED", "EXTRA", "Transfer penalty (min)")), - ("Route_ID", ("@route_id", "TRRT", "EXTRA", "Transit line internal ID")), - ("Night_Hours", ("@night_hours", "TRRT", "EXTRA", "Night hours")), - ("Config", ("@config", "TRRT", "EXTRA", "Config ID (same as route name)")), - ]), - "TRANSIT_SEGMENT": OrderedDict([ - ("Stop_ID", ("@stop_id", "TRSTOP", "EXTRA", "Stop ID from trcov")), - ("Pass_Count", ("@pass_count", "TRSTOP", "EXTRA", "Number of times this stop is passed")), - ("Milepost", ("@milepost", "TRSTOP", "EXTRA", "Distance from start of line")), - ("FareZone", ("@fare_zone", "TRSTOP", "EXTRA", "Fare zone ID")), - ("StopName", ("#stop_name", "TRSTOP", "STRING", "Name of stop")), - ("@coaster_fare_board", ("@coaster_fare_board", "DERIVED", "EXTRA", "Boarding fare for coaster")), - ("@coaster_fare_inveh", ("@coaster_fare_inveh", "DERIVED", "EXTRA", "Incremental fare for Coaster")), - ]) - } - - create_scenario = _m.Modeller().tool( - "inro.emme.data.scenario.create_scenario") - - file_names = [ - "hwycov.e00", "LINKTYPETURNS.DBF", "turns.csv", - "trcov.e00", "trrt.csv", "trlink.csv", "trstop.csv", - "timexfer_EA.csv", "timexfer_AM.csv","timexfer_MD.csv", - "timexfer_PM.csv","timexfer_EV.csv","MODE5TOD.csv", - ] - for name in file_names: - file_path = _join(self.source, name) - if not os.path.exists(file_path): - raise Exception("missing file '%s' in directory %s" % (name, self.source)) - - title = self.title - if not title: - existing_scenario = self.emmebank.scenario(self.merged_scenario_id) - if existing_scenario: - title = existing_scenario.title - - def create_attributes(scenario, attr_map): - for elem_type, mapping in attr_map.iteritems(): - for name, _tcoved_type, emme_type, desc in mapping.values(): - if emme_type == "EXTRA": - if not scenario.extra_attribute(name): - xatt = scenario.create_extra_attribute(elem_type, name) - xatt.description = desc - elif emme_type == "STRING": - if not scenario.network_field(elem_type, name): - scenario.create_network_field(elem_type, name, 'STRING', description=desc) - - if self.traffic_scenario_id: - traffic_scenario = create_scenario( - self.traffic_scenario_id, title + " Traffic", - overwrite=self.overwrite, emmebank=self.emmebank) - create_attributes(traffic_scenario, traffic_attr_map) - else: - traffic_scenario = None - if self.transit_scenario_id: - transit_scenario = create_scenario( - self.transit_scenario_id, title + " Transit", - overwrite=self.overwrite, emmebank=self.emmebank) - create_attributes(transit_scenario, transit_attr_map) - else: - transit_scenario = None - if self.merged_scenario_id: - scenario = create_scenario( - self.merged_scenario_id, title, - overwrite=self.overwrite, emmebank=self.emmebank) - create_attributes(scenario, traffic_attr_map) - create_attributes(scenario, transit_attr_map) - else: - scenario = traffic_scenario or transit_scenario - - traffic_network = _network.Network() - transit_network = _network.Network() - try: - if self.traffic_scenario_id or self.merged_scenario_id: - for elem_type, attrs in traffic_attr_map.iteritems(): - log_content = [] - for k, v in attrs.iteritems(): - if v[3] == "DERIVED": - k = "--" - log_content.append([k] + list(v)) - self._log.append({ - "content": log_content, - "type": "table", - "header": ["TCOVED", "Emme", "Source", "Type", "Description"], - "title": "Traffic %s attributes" % elem_type.lower().replace("_", " "), - "disclosure": True - }) - try: - self.create_traffic_base(traffic_network, traffic_attr_map) - self.create_turns(traffic_network) - self.calc_traffic_attributes(traffic_network) - self.check_zone_access(traffic_network, traffic_network.mode("d")) - finally: - if traffic_scenario: - traffic_scenario.publish_network(traffic_network, resolve_attributes=True) - - if self.transit_scenario_id or self.merged_scenario_id: - for elem_type, attrs in transit_attr_map.iteritems(): - log_content = [] - for k, v in attrs.iteritems(): - if v[3] == "DERIVED": - k = "--" - log_content.append([k] + list(v)) - self._log.append({ - "content": log_content, - "type": "table", - "header": ["TCOVED", "Emme", "Source", "Type", "Description"], - "title": "Transit %s attributes" % elem_type.lower().replace("_", " "), - "disclosure": True - }) - try: - self.create_transit_base(transit_network, transit_attr_map) - self.create_transit_lines(transit_network, transit_attr_map) - self.calc_transit_attributes(transit_network) - finally: - if transit_scenario: - for link in transit_network.links(): - if link.type <= 0: - link.type = 99 - transit_scenario.publish_network(transit_network, resolve_attributes=True) - if self.merged_scenario_id: - self.add_transit_to_traffic(traffic_network, transit_network) - finally: - if self.merged_scenario_id: - scenario.publish_network(traffic_network, resolve_attributes=True) - - self.set_functions(scenario) - self.check_connectivity(scenario) - - def create_traffic_base(self, network, attr_map): - self._log.append({"type": "header", "content": "Import traffic base network from hwycov.e00"}) - hwy_data = gen_utils.DataTableProc("ARC", _join(self.source, "hwycov.e00")) - - if self.save_data_tables: - hwy_data.save("%s_hwycov" % self.data_table_name, self.overwrite) - - for elem_type in "NODE", "TURN": - mapping = attr_map.get(elem_type) - if not mapping: - continue - for field, (attr, tcoved_type, emme_type, desc) in mapping.iteritems(): - default = "" if emme_type == "STRING" else 0 - network.create_attribute(elem_type, attr, default) - - # Create Modes - dummy_auto = network.create_mode("AUTO", "d") - hov2 = network.create_mode("AUX_AUTO", "h") - hov2_toll = network.create_mode("AUX_AUTO", "H") - hov3 = network.create_mode("AUX_AUTO", "i") - hov3_toll = network.create_mode("AUX_AUTO", "I") - sov = network.create_mode("AUX_AUTO", "s") - sov_toll = network.create_mode("AUX_AUTO", "S") - heavy_trk = network.create_mode("AUX_AUTO", "v") - heavy_trk_toll = network.create_mode("AUX_AUTO", "V") - medium_trk = network.create_mode("AUX_AUTO", "m") - medium_trk_toll = network.create_mode("AUX_AUTO", "M") - light_trk = network.create_mode("AUX_AUTO", "t") - light_trk_toll = network.create_mode("AUX_AUTO", "T") - - dummy_auto.description = "dummy auto" - sov.description = "SOV" - hov2.description = "HOV2" - hov3.description = "HOV3+" - light_trk.description = "TRKL" - medium_trk.description = "TRKM" - heavy_trk.description = "TRKH" - - sov_toll.description = "SOV TOLL" - hov2_toll.description = "HOV2 TOLL" - hov3_toll.description = "HOV3+ TOLL" - light_trk_toll.description = "TRKL TOLL" - medium_trk_toll.description = "TRKM TOLL" - heavy_trk_toll.description = "TRKH TOLL" - - is_centroid = lambda arc, node : (arc["IFC"] == 10) and (node == "AN") - - # Note: only truck types 1, 3, 4, and 7 found in 2012 base network - modes_gp_lanes= { - 1: set([dummy_auto, sov, hov2, hov3, light_trk, medium_trk, heavy_trk, - sov_toll, hov2_toll, hov3_toll, light_trk_toll, medium_trk_toll, - heavy_trk_toll]), - 2: set([dummy_auto, sov, hov2, hov3, light_trk, medium_trk, - sov_toll, hov2_toll, hov3_toll, light_trk_toll, medium_trk_toll]), - 3: set([dummy_auto, sov, hov2, hov3, light_trk, sov_toll, hov2_toll, - hov3_toll, light_trk_toll]), - 4: set([dummy_auto, sov, hov2, hov3, sov_toll, hov2_toll, hov3_toll]), - 5: set([dummy_auto, heavy_trk, heavy_trk_toll]), - 6: set([dummy_auto, medium_trk, heavy_trk, medium_trk_toll, heavy_trk_toll]), - 7: set([dummy_auto, light_trk, medium_trk, heavy_trk, light_trk_toll, - medium_trk_toll, heavy_trk_toll]), - } - modes_toll_lanes = { - 1: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll, - medium_trk_toll, heavy_trk_toll]), - 2: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll, - medium_trk_toll]), - 3: set([dummy_auto, sov_toll, hov2_toll, hov3_toll, light_trk_toll]), - 4: set([dummy_auto, sov_toll, hov2_toll, hov3_toll]), - 5: set([dummy_auto, heavy_trk_toll]), - 6: set([dummy_auto, medium_trk_toll, heavy_trk_toll]), - 7: set([dummy_auto, light_trk_toll, medium_trk_toll, heavy_trk_toll]), - } - modes_HOV2 = set([dummy_auto, hov2, hov3, hov2_toll, hov3_toll]) - modes_HOV3 = set([dummy_auto, hov3, hov3_toll]) - - - def define_modes(arc): - if arc["IFC"] == 10: # connector - return modes_gp_lanes[1] - elif arc["IHOV"] == 1: - return modes_gp_lanes[arc["ITRUCK"]] - elif arc["IHOV"] == 2: - # managed lanes, free for HOV2 and HOV3+, tolls for SOV - if arc["ITOLLO"] + arc["ITOLLA"] + arc["ITOLLP"] > 0: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 - # special case of I-15 managed lanes base year and 2020, no build - elif arc["IFC"] == 1 and arc["IPROJ"] in [41, 42, 486, 373, 711]: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 - elif arc["IFC"] == 8 or arc["IFC"] == 9: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV2 - else: - return modes_HOV2 - elif arc["IHOV"] == 3: - # managed lanes, free for HOV3+, tolls for SOV and HOV2 - if arc["ITOLLO"] + arc["ITOLLA"] + arc["ITOLLP"] > 0: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 - # special case of I-15 managed lanes for base year and 2020, no build - elif arc["IFC"] == 1 and arc["IPROJ"] in [41, 42, 486, 373, 711]: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 - elif arc["IFC"] == 8 or arc["IFC"] == 9: - return modes_toll_lanes[arc["ITRUCK"]] | modes_HOV3 - else: - return modes_HOV3 - elif arc["IHOV"] == 4: - return modes_toll_lanes[arc["ITRUCK"]] - else: - return modes_gp_lanes[arc["ITRUCK"]] - - self._create_base_net( - hwy_data, network, mode_callback=define_modes, centroid_callback=is_centroid, - arc_id_name="HWYCOV-ID", link_attr_map=attr_map["LINK"]) - self._log.append({"type": "text", "content": "Import traffic base network complete"}) - - def create_transit_base(self, network, attr_map): - self._log.append({"type": "header", "content": "Import transit base network from trcov.e00"}) - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - transit_data = gen_utils.DataTableProc("ARC", _join(self.source, "trcov.e00")) - - if self.save_data_tables: - transit_data.save("%s_trcov" % self.data_table_name, self.overwrite) - - # aux mode speed is always 3 (miles/hr) - access = network.create_mode("AUX_TRANSIT", "a") - transfer = network.create_mode("AUX_TRANSIT", "x") - walk = network.create_mode("AUX_TRANSIT", "w") - - bus = network.create_mode("TRANSIT", "b") - express_bus = network.create_mode("TRANSIT", "e") - ltdexp_bus = network.create_mode("TRANSIT", "p") - brt_red = network.create_mode("TRANSIT", "r") - brt_yellow = network.create_mode("TRANSIT", "y") - lrt = network.create_mode("TRANSIT", "l") - coaster_rail = network.create_mode("TRANSIT", "c") - tier1 = network.create_mode("TRANSIT", "o") - - access.description = "ACCESS" - transfer.description = "TRANSFER" - walk.description = "WALK" - bus.description = "BUS" # (vehicle type 100, PCE=3.0) - express_bus.description = "EXP BUS" # (vehicle type 90 , PCE=3.0) - ltdexp_bus.description = "LTDEXP BUS" # (vehicle type 80 , PCE=3.0) - lrt.description = "LRT" # (vehicle type 50) - brt_yellow.description = "BRT YEL" # (vehicle type 60 , PCE=3.0) - brt_red.description = "BRT RED" # (vehicle type 70 , PCE=3.0) - coaster_rail.description = "CMR" # (vehicle type 40) - tier1.description = "TIER1" # (vehicle type 45) - - access.speed = 3 - transfer.speed = 3 - walk.speed = 3 - - # define TAP connectors as centroids - is_centroid = lambda arc, node: (int(arc["MINMODE"]) == 3) and (node == "BN") - - mode_setting = { - 1: set([transfer]), # 1 = special transfer walk links between certain nearby stops - 2: set([walk]), # 2 = walk links in the downtown area - 3: set([access]), # 3 = the special TAP connectors - 4: set([coaster_rail]), # 4 = Coaster Rail Line - 5: set([lrt]), # 5 = Light Rail Transit (LRT) Line - 6: set([brt_yellow, ltdexp_bus, express_bus, bus]), # 6 = Yellow Car Bus Rapid Transit (BRT) - 7: set([brt_red, ltdexp_bus, express_bus, bus]), # 7 = Red Car Bus Rapid Transit (BRT) - 8: set([ltdexp_bus, express_bus, bus]), # 8 = Limited Express Bus - 9: set([ltdexp_bus, express_bus, bus]), # 9 = Express Bus - 10: set([ltdexp_bus, express_bus, bus]), # 10 = Local Bus - } - tier1_rail_link_name = props["transit.newMode"] - - def define_modes(arc): - if arc["NM"] == tier1_rail_link_name: - return set([tier1]) - return mode_setting[arc["MINMODE"]] - - arc_filter = lambda arc: (arc["MINMODE"] > 2) - - # first pass to create the main base network for vehicles, xfer links and TAPs - self._create_base_net( - transit_data, network, mode_callback=define_modes, centroid_callback=is_centroid, - arc_id_name="TRCOV-ID", link_attr_map=attr_map["LINK"], arc_filter=arc_filter) - - # second pass to add special walk links / modify modes on existing links - reverse_dir_map = {1:3, 3:1, 2:4, 4:2, 0:0} - - def set_reverse_link(link, modes): - reverse_link = link.reverse_link - if reverse_link: - reverse_link.modes |= modes - else: - reverse_link = network.create_link(link.j_node, link.i_node, modes) - for attr in network.attributes("LINK"): - reverse_link[attr] = link[attr] - reverse_link["@direction_cardinal"] = reverse_dir_map[link["@direction_cardinal"]] - reverse_link["@tcov_id"] = -1*link["@tcov_id"] - reverse_link.vertices = list(reversed(link.vertices)) - - def epsilon_compare(a, b, epsilon): - return abs((a - b) / (a if abs(a) > 1 else 1)) <= epsilon - - for arc in transit_data: - # possible improvement: snap walk nodes to nearby node if not matched and within distance - if arc_filter(arc): - continue - if float(arc["AN"]) == 0 or float(arc["BN"]) == 0: - self._log.append({"type": "text", - "content": "Node ID 0 in AN (%s) or BN (%s) for link ID %s." % - (arc["AN"], arc["BN"], arc["TRCOV-ID"])}) - continue - coordinates = arc["geo_coordinates"] - arc_length = arc["LENGTH"] / 5280.0 # convert feet to miles - i_node = get_node(network, arc['AN'], coordinates[0], False) - j_node = get_node(network, arc['BN'], coordinates[-1], False) - modes = define_modes(arc) - link = network.link(i_node, j_node) - split_link_case = False - if link: - link.modes |= modes - else: - # Note: additional cases of "tunnel" walk links could be - # considered to optimize network matching - # check if this a special "split" link case where - # we do not need to add a "tunnel" walk link - for link1 in i_node.outgoing_links(): - if split_link_case: - break - for link2 in link1.j_node.outgoing_links(): - if link2.j_node == j_node: - if epsilon_compare(link1.length + link2.length, arc_length, 10**-5): - self._log.append({"type": "text", - "content": "Walk link AN %s BN %s matched to two links TCOV-ID %s, %s" % - (arc['AN'], arc['BN'], link1["@tcov_id"], link2["@tcov_id"])}) - link1.modes |= modes - link2.modes |= modes - set_reverse_link(link1, modes) - set_reverse_link(link2, modes) - split_link_case = True - break - if not split_link_case: - link = network.create_link(i_node, j_node, modes) - link.length = arc_length - if len(coordinates) > 2: - link.vertices = coordinates[1:-1] - if not split_link_case: - set_reverse_link(link, modes) - self._log.append({"type": "text", "content": "Import transit base network complete"}) - - def _create_base_net(self, data, network, mode_callback, centroid_callback, arc_id_name, link_attr_map, arc_filter=None): - forward_attr_map = {} - reverse_attr_map = {} - for field, (name, tcoved_type, emme_type, desc) in link_attr_map.iteritems(): - if emme_type != "STANDARD": - default = "" if emme_type == "STRING" else 0 - network.create_attribute("LINK", name, default) - - if field in [arc_id_name, "DIR"]: - # these attributes are special cases for reverse link - forward_attr_map[field] = name - elif tcoved_type == "TWO_WAY": - forward_attr_map[field] = name - reverse_attr_map[field] = name - elif tcoved_type == "ONE_WAY": - forward_attr_map["AB" + field] = name - reverse_attr_map["BA" + field] = name - - emme_id_name = forward_attr_map[arc_id_name] - dir_name = forward_attr_map["DIR"] - reverse_dir_map = {1:3, 3:1, 2:4, 4:2, 0:0} - new_node_id = max(data.values("AN").max(), data.values("BN").max()) + 1 - if arc_filter is None: - arc_filter = lambda arc : True - - # Create nodes and links - for arc in data: - if not arc_filter(arc): - continue - if float(arc["AN"]) == 0 or float(arc["BN"]) == 0: - self._log.append({"type": "text", - "content": "Node ID 0 in AN (%s) or BN (%s) for link ID %s." % - (arc["AN"], arc["BN"], arc[arc_id_name])}) - continue - coordinates = arc["geo_coordinates"] - i_node = get_node(network, arc['AN'], coordinates[0], centroid_callback(arc, "AN")) - j_node = get_node(network, arc['BN'], coordinates[-1], centroid_callback(arc, "BN")) - existing_link = network.link(i_node, j_node) - if existing_link: - msg = "Duplicate link between AN %s and BN %s. Link IDs %s and %s." % \ - (arc["AN"], arc["BN"], existing_link[emme_id_name], arc[arc_id_name]) - self._log.append({"type": "text", "content": msg}) - self._error.append(msg) - self._split_link(network, i_node, j_node, new_node_id) - new_node_id += 1 - - modes = mode_callback(arc) - link = network.create_link(i_node, j_node, modes) - link.length = arc["LENGTH"] / 5280.0 # convert feet to miles - if len(coordinates) > 2: - link.vertices = coordinates[1:-1] - for field, attr in forward_attr_map.iteritems(): - link[attr] = arc[field] - if arc["IWAY"] == 2 or arc["IWAY"] == 0: - reverse_link = network.create_link(j_node, i_node, modes) - reverse_link.length = link.length - reverse_link.vertices = list(reversed(link.vertices)) - for field, attr in reverse_attr_map.iteritems(): - reverse_link[attr] = arc[field] - reverse_link[emme_id_name] = -1*arc[arc_id_name] - reverse_link[dir_name] = reverse_dir_map[arc["DIR"]] - - def create_transit_lines(self, network, attr_map): - self._log.append({"type": "header", "content": "Import transit lines"}) - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - fatal_errors = 0 - # Route_ID,Route_Name,Mode,AM_Headway,PM_Headway,OP_Headway,Night_Headway,Night_Hours,Config,Fare - transit_line_data = gen_utils.DataTableProc("trrt", _join(self.source, "trrt.csv")) - # Route_ID,Link_ID,Direction - transit_link_data = gen_utils.DataTableProc("trlink", _join(self.source, "trlink.csv")) - # Stop_ID,Route_ID,Link_ID,Pass_Count,Milepost,Longitude, Latitude,HwyNode,TrnNode,FareZone,StopName - transit_stop_data = gen_utils.DataTableProc("trstop", _join(self.source, "trstop.csv")) - # From_line,To_line,Board_stop,Wait_time - # Note: Board_stop is not used - # Timed xfer data - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - timed_xfer_data = {} - for period in periods: - timed_xfer_data[period] = gen_utils.DataTableProc("timexfer_"+period, _join(self.source, "timexfer_"+period+".csv")) - - mode_properties = gen_utils.DataTableProc("MODE5TOD", _join(self.source, "MODE5TOD.csv"), convert_numeric=True) - mode_details = {} - for record in mode_properties: - mode_details[int(record["MODE_ID"])] = record - - if self.save_data_tables: - transit_link_data.save("%s_trlink" % self.data_table_name, self.overwrite) - transit_line_data.save("%s_trrt" % self.data_table_name, self.overwrite) - transit_stop_data.save("%s_trstop" % self.data_table_name, self.overwrite) - mode_properties.save("%s_MODE5TOD" % self.data_table_name, self.overwrite) - - coaster = network.create_transit_vehicle(40, 'c') # 4 coaster - trolley = network.create_transit_vehicle(50, 'l') # 5 sprinter/trolley - brt_yellow = network.create_transit_vehicle(60, 'y') # 6 BRT yellow line (future line) - brt_red = network.create_transit_vehicle(70, 'r') # 7 BRT red line (future line) - premium_bus = network.create_transit_vehicle(80, 'p') # 8 prem express - express_bus = network.create_transit_vehicle(90, 'e') # 9 regular express - local_bus = network.create_transit_vehicle(100, 'b') # 10 local bus - tier1 = network.create_transit_vehicle(45, 'o') # 11 Tier 1 - - brt_yellow.auto_equivalent = 3.0 - brt_red.auto_equivalent = 3.0 - premium_bus.auto_equivalent = 3.0 - express_bus.auto_equivalent = 3.0 - local_bus.auto_equivalent = 3.0 - - # Capacities - for reference / post-assignment analysis - tier1.seated_capacity, tier1.total_capacity = 7 * 142, 7 * 276 - trolley.seated_capacity, trolley.total_capacity = 4 * 64, 4 * 200 - brt_yellow.seated_capacity, brt_yellow.total_capacity = 32, 70 - brt_red.seated_capacity, brt_red.total_capacity = 32, 70 - premium_bus.seated_capacity, premium_bus.total_capacity = 32, 70 - express_bus.seated_capacity, express_bus.total_capacity = 32, 70 - local_bus.seated_capacity, local_bus.total_capacity = 32, 70 - - trrt_attrs = [] - mode5tod_attrs = [] - for elem_type in "TRANSIT_LINE", "TRANSIT_SEGMENT", "NODE": - mapping = attr_map[elem_type] - for field, (attr, tcoved_type, emme_type, desc) in mapping.iteritems(): - default = "" if emme_type == "STRING" else 0 - network.create_attribute(elem_type, attr, default) - if tcoved_type == "TRRT": - trrt_attrs.append((field, attr)) - elif tcoved_type == "MODE5TOD": - mode5tod_attrs.append((field, attr)) - - # Pre-process transit line (trrt.csv) to know the route names for errors / warnings - transit_line_records = list(transit_line_data) - line_names = {} - for record in transit_line_records: - line_names[int(record["Route_ID"])] = record["Route_Name"].strip() - - links = dict((link["@tcov_id"], link) for link in network.links()) - transit_routes = _defaultdict(lambda: []) - for record in transit_link_data: - line_ref = line_names.get(int(record["Route_ID"]), record["Route_ID"]) - link_id = int(record["Link_ID"]) - if "+" in record["Direction"]: - link = links.get(link_id) - else: - link = links.get(-1*link_id) - if not link: - link = links.get(link_id) - if link and not link.reverse_link: - reverse_link = network.create_link(link.j_node, link.i_node, link.modes) - reverse_link.vertices = list(reversed(link.vertices)) - for attr in network.attributes("LINK"): - if attr not in set(["vertices"]): - reverse_link[attr] = link[attr] - reverse_link["@tcov_id"] = -1 * link["@tcov_id"] - msg = "Transit line %s : Missing reverse link with ID %s (%s) (reverse link created)" % ( - line_ref, record["Link_ID"], link) - self._log.append({"type": "text", "content": msg}) - self._error.append("Transit route import: " + msg) - link = reverse_link - if not link: - msg = "Transit line %s : No link with ID %s, line not created" % ( - line_ref, record["Link_ID"]) - self._log.append({"type": "text", "content": msg}) - self._error.append("Transit route import: " + msg) - fatal_errors += 1 - continue - transit_routes[int(record["Route_ID"])].append(link) - - # lookup list of special tier 1 mode route names - tier1_rail_route_names = [str(n) for n in props["transit.newMode.route"]] - dummy_links = set([]) - transit_lines = {} - for record in transit_line_records: - try: - route = transit_routes[int(record["Route_ID"])] - # Find if name matches one of the names listed in transit.newMode.route and convert to tier 1 rail - is_tier1_rail = False - for name in tier1_rail_route_names: - if str(record["Route_Name"]).startswith(name): - print 'record["Route_Name"]2', record["Route_Name"] - is_tier1_rail = True - break - if is_tier1_rail: - vehicle_type = 45 - mode = network.transit_vehicle(vehicle_type).mode - else: - vehicle_type = int(record["Mode"]) * 10 - mode = network.transit_vehicle(vehicle_type).mode - prev_link = route[0] - itinerary = [prev_link] - for link in route[1:]: - if prev_link.j_node != link.i_node: # filling in the missing gap - msg = "line %s : Links not adjacent, shortest path interpolation used (%s and %s)" % ( - record["Route_Name"], prev_link["@tcov_id"], link["@tcov_id"]) - log_record = {"type": "text", "content": msg} - self._log.append(log_record) - sub_path = find_path(prev_link, link, mode) - itinerary.extend(sub_path) - log_record["content"] = log_record["content"] + " through %s links" % (len(sub_path)) - itinerary.append(link) - prev_link = link - - node_itinerary = [itinerary[0].i_node] + [l.j_node for l in itinerary] - try: - tline = network.create_transit_line( - record["Route_Name"].strip(), vehicle_type, node_itinerary) - except: - msg = "Transit line %s : missing mode added to at least one link" % ( - record["Route_Name"]) - self._log.append({"type": "text", "content": msg}) - for link in itinerary: - link.modes |= set([mode]) - tline = network.create_transit_line( - record["Route_Name"].strip(), vehicle_type, node_itinerary) - - for field, attr in trrt_attrs: - tline[attr] = float(record[field]) - if is_tier1_rail: - line_details = mode_details[11] - else: - line_details = mode_details[int(record["Mode"])] - for field, attr in mode5tod_attrs: - tline[attr] = float(line_details[field]) - #"XFERPENTM": "Transfer penalty time: " - #"WTXFERTM": "Transfer perception:" - # NOTE: an additional transfer penalty perception factor of 5.0 is included - # in assignment - tline["@transfer_penalty"] = float(line_details["XFERPENTM"]) * float(line_details["WTXFERTM"]) - tline.headway = tline["@headway_am"] if tline["@headway_am"] > 0 else 999 - tline.layover_time = 5 - - transit_lines[int(record["Route_ID"])] = tline - for segment in tline.segments(): - segment.allow_boardings = False - segment.allow_alightings = False - segment.transit_time_func = 2 - # ft2 = ul2 -> copied @trtime_link_XX - # segments on links matched to auto network (with auto mode) are changed to ft1 = timau - except Exception as error: - msg = "Transit line %s: %s %s" % (record["Route_Name"], type(error), error) - self._log.append({"type": "text", "content": msg}) - trace_text = _traceback.format_exc().replace("\n", "
    ") - self._log.append({"type": "text", "content": trace_text}) - self._error.append("Transit route import: line %s not created" % record["Route_Name"]) - fatal_errors += 1 - for link in dummy_links: - network.delete_link(link.i_node, link.j_node) - - line_stops = _defaultdict(lambda: []) - for record in transit_stop_data: - try: - line_name = line_names[int(record["Route_ID"])] - line_stops[line_name].append(record) - except KeyError: - self._log.append( - {"type": "text", - "content": "Stop %s: could not find transit line by ID %s (link ID %s)" % ( - record["Stop_ID"], record["Route_ID"], record["Link_ID"])}) - - seg_float_attr_map = [] - seg_string_attr_map = [] - for field, (attr, t_type, e_type, desc) in attr_map["TRANSIT_SEGMENT"].iteritems(): - if t_type == "TRSTOP": - if e_type == "STRING": - seg_string_attr_map.append([field, attr]) - else: - seg_float_attr_map.append([field, attr]) - - for line_name, stops in line_stops.iteritems(): - tline = network.transit_line(line_name) - if not tline: - continue - itinerary = tline.segments(include_hidden=True) - segment = itinerary.next() - tcov_id = abs(segment.link["@tcov_id"]) - for stop in stops: - if "DUMMY" in stop["StopName"]: - continue - link_id = int(stop['Link_ID']) - node_id = int(stop['TrnNode']) - while tcov_id != link_id: - segment = itinerary.next() - if segment.link is None: - break - tcov_id = abs(segment.link["@tcov_id"]) - - if node_id == segment.i_node.number: - pass - elif node_id == segment.j_node.number: - segment = itinerary.next() # its the next segment - else: - msg = "Transit line %s: could not find stop on link ID %s at node ID %s" % (line_name, link_id, node_id) - self._log.append({"type": "text", "content": msg}) - self._error.append(msg) - fatal_errors += 1 - continue - segment.allow_boardings = True - segment.allow_alightings = True - segment.dwell_time = tline.default_dwell_time - for field, attr in seg_string_attr_map: - segment[attr] = stop[field] - for field, attr in seg_float_attr_map: - segment[attr] = float(stop[field]) - - def lookup_line(ident): - line = network.transit_line(ident) - if line: - return line.id - line = transit_lines.get(int(ident)) - if line: - return line.id - raise Exception("'%s' is not a route name or route ID" % ident) - - # Normalizing the case of the headers as different examples have been seen - for period in periods: - norm_data = [] - for record in timed_xfer_data[period]: - norm_record = {} - for key, val in record.iteritems(): - norm_record[key.lower()] = val - norm_data.append(norm_record) - - from_line, to_line, wait_time = [], [], [] - for i, record in enumerate(norm_data, start=2): - try: - from_line.append(lookup_line(record["from_line"])) - to_line.append(lookup_line(record["to_line"])) - wait_time.append(float(record["wait_time"])) - except Exception as error: - msg = "Error processing timexfer_%s.csv on file line %s: %s" % (period, i, error) - self._log.append({"type": "text", "content": msg}) - self._error.append(msg) - fatal_errors += 1 - - timed_xfer = _dt.Data() - timed_xfer.add_attribute(_dt.Attribute("from_line", _np.array(from_line).astype("O"))) - timed_xfer.add_attribute(_dt.Attribute("to_line", _np.array(to_line).astype("O"))) - timed_xfer.add_attribute(_dt.Attribute("wait_time", _np.array(wait_time))) - # Creates and saves the new table - gen_utils.DataTableProc("%s_timed_xfer_%s" % (self.data_table_name, period), data=timed_xfer) - - if fatal_errors > 0: - raise Exception("Cannot create transit network, %s fatal errors found" % fatal_errors) - self._log.append({"type": "text", "content": "Import transit lines complete"}) - - def calc_transit_attributes(self, network): - self._log.append({"type": "header", "content": "Calculate derived transit attributes"}) - # - TM by 5 TOD periods copied from TM for 3 time periods - # NOTE: the values of @trtime_link_## are only used for - # separate guideway. - # Links shared with the traffic network use the - # assignment results in timau - for link in network.links(): - for time in ["_ea", "_md", "_ev"]: - link["@trtime_link" + time] = link["trtime_link_op"] - if link.type == 0: # walk only links have IFC ==0 - link.type = 99 - - # ON TRANSIT LINES - # Set 3-period headway based on revised headway calculation - for line in network.transit_lines(): - for period in ["am", "pm", "op"]: - line["@headway_rev_" + period] = revised_headway(line["@headway_" + period]) - - for c in network.centroids(): - c["@tap_id"] = c.number - - # Special incremental boarding and in-vehicle fares - # to recreate the coaster zone fares - fares_file_name = FILE_NAMES["FARES"] - special_fare_path = _join(self.source, fares_file_name) - if os.path.isfile(special_fare_path): - with open(special_fare_path) as fare_file: - self._log.append({"type": "text", "content": "Using fare details (for coaster) from %s" % fares_file_name}) - special_fares = None - yaml_installed = True - try: - import yaml - special_fares = yaml.load(fare_file) - self._log.append({"type": "text", "content": yaml.dump(special_fares).replace("\n", "
    ")}) - except ImportError: - yaml_installed = False - except: - pass - if special_fares is None: - try: - import json - special_fares = json.load(fare_file) - self._log.append({"type": "text", "content": json.dumps(special_fares, indent=4).replace("\n", "
    ")}) - except: - pass - if special_fares is None: - msg = "YAML or JSON" if yaml_installed else "JSON (YAML parser not installed)" - raise Exception(fares_file_name + ": file could not be parsed as " + msg) - else: - # Default coaster fare for 2012 base year - special_fares = { - "boarding_cost": { - "base": [ - {"line": "398104", "cost" : 4.0}, - {"line": "398204", "cost" : 4.0} - ], - "stop_increment": [ - {"line": "398104", "stop": "SORRENTO VALLEY", "cost": 0.5}, - {"line": "398204", "stop": "SORRENTO VALLEY", "cost": 0.5} - ] - }, - "in_vehicle_cost": [ - {"line": "398104", "from": "SOLANA BEACH", "cost": 1.0}, - {"line": "398104", "from": "SORRENTO VALLEY", "cost": 0.5}, - {"line": "398204", "from": "OLD TOWN", "cost": 1.0}, - {"line": "398204", "from": "SORRENTO VALLEY", "cost": 0.5} - ], - "day_pass": 5.0, - "regional_pass": 12.0 - } - self._log.append({"type": "text", "content": "Using default coaster fare based on 2012 base year setup."}) - - def get_line(line_id): - line = network.transit_line(line_id) - if line is None: - raise Exception("%s: line does not exist: %s" % (fares_file_name, line_id)) - return line - - for record in special_fares["boarding_cost"]["base"]: - line = get_line(record["line"]) - line["@fare"] = 0 - for seg in line.segments(): - seg["@coaster_fare_board"] = record["cost"] - for record in special_fares["boarding_cost"].get("stop_increment", []): - line = get_line(record["line"]) - for seg in line.segments(True): - if record["stop"] in seg["#stop_name"]: - seg["@coaster_fare_board"] += record["cost"] - break - for record in special_fares["in_vehicle_cost"]: - line = get_line(record["line"]) - for seg in line.segments(True): - if record["from"] in seg["#stop_name"]: - seg["@coaster_fare_inveh"] = record["cost"] - break - pass_cost_keys = ['day_pass', 'regional_pass'] - pass_costs = [] - for key in pass_cost_keys: - cost = special_fares.get(key) - if cost is None: - raise Exception("key '%s' missing from %s" % (key, fares_file_name)) - pass_costs.append(cost) - pass_values = _dt.Data() - pass_values.add_attribute(_dt.Attribute("pass_type", _np.array(pass_cost_keys).astype("O"))) - pass_values.add_attribute(_dt.Attribute("cost", _np.array(pass_costs).astype("f8"))) - gen_utils.DataTableProc("%s_transit_passes" % self.data_table_name, data=pass_values) - self._log.append({"type": "text", "content": "Calculate derived transit attributes complete"}) - return - - def create_turns(self, network): - self._log.append({"type": "header", "content": "Import turns and turn restrictions"}) - self._log.append({"type": "text", "content": "Process LINKTYPETURNS.DBF for turn prohibited by type"}) - # Process LINKTYPETURNS.DBF for turn prohibited by type - with _fiona.open(_join(self.source, "LINKTYPETURNS.DBF"), 'r') as f: - link_type_turns = _defaultdict(lambda: {}) - for record in f: - record = record['properties'] - link_type_turns[record["FROM"]][record["TO"]] = { - "LEFT": record["LEFT"], - "RIGHT": record["RIGHT"], - "STRAIGHT": record["STRAIGHT"], - "UTURN": record["UTURN"] - } - for from_link in network.links(): - if from_link.type in link_type_turns: - to_link_turns = link_type_turns[from_link.type] - for to_link in from_link.j_node.outgoing_links(): - if to_link.type in to_link_turns: - record = to_link_turns[to_link.type] - if not from_link.j_node.is_intersection: - network.create_intersection(from_link.j_node) - turn = network.turn(from_link.i_node, from_link.j_node, to_link.j_node) - turn.penalty_func = 1 - if to_link["@tcov_id"] == from_link["left_link"]: - turn.data1 = record["LEFT"] - elif to_link["@tcov_id"] == from_link["through_link"]: - turn.data1 = record["STRAIGHT"] - elif to_link["@tcov_id"] == from_link["right_link"]: - turn.data1 = record["RIGHT"] - else: - turn.data1 = record["UTURN"] - - self._log.append({"type": "text", "content": "Process turns.csv for turn prohibited by ID"}) - turn_data = gen_utils.DataTableProc("turns", _join(self.source, "turns.csv")) - if self.save_data_tables: - turn_data.save("%s_turns" % self.data_table_name, self.overwrite) - links = dict((link["@tcov_id"], link) for link in network.links()) - - # Process turns.csv for prohibited turns from_id, to_id, penalty - for i, record in enumerate(turn_data): - from_link_id, to_link_id = int(record["from_id"]), int(record["to_id"]) - from_link, to_link = links[from_link_id], links[to_link_id] - if from_link.j_node == to_link.i_node: - pass - elif from_link.j_node == to_link.j_node: - to_link = to_link.reverse_link - elif from_link.i_node == to_link.i_node: - from_link = from_link.reverse_link - elif from_link.i_node == to_link.j_node: - from_link = from_link.reverse_link - to_link = to_link.reverse_link - else: - msg = "Record %s: links are not adjacent %s - %s." % (i, from_link_id, to_link_id) - self._log.append({"type": "text", "content": msg}) - self._error.append("Turn import: " + msg) - continue - if not from_link or not to_link: - msg = "Record %s: links adjacent but in reverse direction %s - %s." % (i, from_link_id, to_link_id) - self._log.append({"type": "text", "content": msg}) - self._error.append("Turn import: " + msg) - continue - - node = from_link.j_node - if not node.is_intersection: - network.create_intersection(node) - turn = network.turn(from_link.i_node, node, to_link.j_node) - if not record["penalty"]: - turn.penalty_func = 0 # prohibit turn - else: - turn.penalty_func = 1 - turn.data1 = float(record["penalty"]) - self._log.append({"type": "text", "content": "Import turns and turn restrictions complete"}) - - def calc_traffic_attributes(self, network): - self._log.append({"type": "header", "content": "Calculate derived traffic attributes"}) - # "COST": "@cost_operating" - # "ITOLL": "@toll_flag" # ITOLL - Toll + 100 *[0,1] if managed lane (I-15 tolls) - # Note: toll_flag is no longer used - # "ITOLL2": "@toll" # ITOLL2 - Toll - # "ITOLL3": "@cost_auto" # ITOLL3 - Toll + AOC - # "@cost_hov" - # "ITOLL4": "@cost_med_truck" # ITOLL4 - Toll * 1.03 + AOC - # "ITOLL5": "@cost_hvy_truck" # ITOLL5 - Toll * 2.33 + AOC - fatal_errors = 0 - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - try: - aoc = float(props["aoc.fuel"]) + float(props["aoc.maintenance"]) - except ValueError: - raise Exception("Error during float conversion for aoc.fuel or aoc.maintenance from sandag_abm.properties file") - scenario_year = int(props["scenarioYear"]) - periods = ["EA", "AM", "MD", "PM", "EV"] - time_periods = ["_ea", "_am", "_md", "_pm", "_ev"] - src_time_periods = ["_op", "_am", "_op", "_pm", "_op"] - mode_d = network.mode('d') - - # Calculate upstream and downstream interchange distance - # First, label the intersection nodes as nodes with type 1 links (freeway) and - # type 8 links (freeway-to-freeway ramp) - network.create_attribute("NODE", "is_interchange") - interchange_points = [] - for node in network.nodes(): - adj_links = list(node.incoming_links()) + list(node.outgoing_links()) - has_freeway_links = bool( - [l for l in adj_links - if l.type == 1 and mode_d in l.modes]) - has_ramp_links = bool( - [l for l in adj_links - if l.type == 8 and mode_d in l.modes and not "HOV" in l["#name"]]) - if has_freeway_links and has_ramp_links: - node.is_interchange = True - interchange_points.append(node) - else: - node.is_interchange = False - for node in network.nodes(): - node["@interchange"] = node.is_interchange - - for link in network.links(): - if link.type == 1 and mode_d in link.modes: - link["@intdist_down"] = interchange_distance(link, "DOWNSTREAM") - link["@intdist_up"] = interchange_distance(link, "UPSTREAM") - self._log.append({"type": "text", "content": "Calculate of nearest interchange distance complete"}) - - # Static reliability parameters - # freeway coefficients - freeway_rel = { - "intercept": 0.1078, - "speed>70": 0.01393, - "upstream": 0.011, - "downstream": 0.0005445, - } - # arterial/ramp/other coefficients - road_rel = { - "intercept": 0.0546552, - "lanes": { - 1: 0.0, - 2: 0.0103589, - 3: 0.0361211, - 4: 0.0446958, - 5: 0.0 - }, - "speed": { - "<35": 0, - 35: 0.0075674, - 40: 0.0091012, - 45: 0.0080996, - 50: -0.0022938, - ">50": -0.0046211 - }, - "control": { - 0: 0, # Uncontrolled - 1: 0.0030973, # Signal - 2: -0.0063281, # Stop - 3: -0.0063281, # Stop - 4: 0.0127692, # Other, Railway, etc. - } - } - for link in network.links(): - # Change SR125 toll speed to 70MPH - if link["@lane_restriction"] == 4 and link.type == 1: - link["@speed_posted"] = 70 - - link["@cost_operating"] = link.length * aoc - - # Expand off-peak TOD attributes, copy peak period attributes - for time, src_time in zip(time_periods, src_time_periods): - link["@lane" + time] = link["lane" + src_time] - link["@time_link" + time] = link["time_link" + src_time] - - # add link delay (30 sec=0.5mins) to HOV connectors to discourage travel - if link.type == 8 and (link["@lane_restriction"] == 2 or link["@lane_restriction"] == 3): - link["@time_link" + time] = link["@time_link" + time] + 0.375 - - # make speed on HOV lanes (70mph) the same as parallel GP lanes (65mph) - # - set speed back to posted speed - increase travel time by (speed_adj/speed_posted) - if link.type == 1 and (link["@lane_restriction"] == 2 or link["@lane_restriction"] == 3): - speed_adj = link["@speed_adjusted"] - speed_posted = link["@speed_posted"] - if speed_adj>0: - link["@time_link" + time] = (speed_adj/(speed_posted*1.0)) * link["@time_link" + time] - - link["@time_inter" + time] = link["time_inter" + src_time] - link["@toll" + time] = link["toll" + src_time] - - off_peak_factor_file = FILE_NAMES["OFF_PEAK"] - if os.path.exists(_join(self.source, off_peak_factor_file)): - msg = "Adjusting off-peak tolls based on factors from %s" % off_peak_factor_file - self._log.append({"type": "text", "content": msg}) - tolled_links = list(link for link in network.links() if link["toll_op"] > 0) - # NOTE: CSV Reader sets the field names to UPPERCASE for consistency - with gen_utils.CSVReader(_join(self.source, off_peak_factor_file)) as r: - for row in r: - name = row["FACILITY_NAME"] - ea_factor = float(row["OP_EA_FACTOR"]) - md_factor = float(row["OP_MD_FACTOR"]) - ev_factor = float(row["OP_EV_FACTOR"]) - count = 0 - for link in tolled_links: - if name in link["#name"]: - count += 1 - link["@toll_ea"] = link["@toll_ea"] * ea_factor - link["@toll_md"] = link["@toll_md"] * md_factor - link["@toll_ev"] = link["@toll_ev"] * ev_factor - - msg = "Facility name '%s' matched to %s links." % (name, count) - msg += " Adjusted off-peak period tolls EA: %s, MD: %s, EV: %s" % (ea_factor, md_factor, ev_factor) - self._log.append({"type": "text2", "content": msg}) - - for link in network.links(): - factors = [(3.0/12.0), 1.0, (6.5/12.0), (3.5/3.0), (8.0/12.0)] - for f, time, src_time in zip(factors, time_periods, src_time_periods): - if link["capacity_link" + src_time] != 999999: - link["@capacity_link" + time] = f * link["capacity_link" + src_time] - else: - link["@capacity_link" + time] = 999999 - if link["capacity_inter" + src_time] != 999999: - link["@capacity_inter" + time] = f * link["capacity_inter" + src_time] - else: - link["@capacity_inter" + time] = 999999 - - # Required file - vehicle_class_factor_file = FILE_NAMES["VEHICLE_CLASS"] - facility_factors = _defaultdict(lambda: {}) - facility_factors["DEFAULT_FACTORS"] = { - "ALL": { - "auto": 1.0, - "hov2": 1.0, - "hov3": 1.0, - "lgt_truck": 1.0, - "med_truck": 1.03, - "hvy_truck": 2.03 - }, - "count": 0 - } - if os.path.exists(_join(self.source, vehicle_class_factor_file)): - msg = "Adjusting tolls based on factors from %s" % vehicle_class_factor_file - self._log.append({"type": "text", "content": msg}) - # NOTE: CSV Reader sets the field names to UPPERCASE for consistency - with gen_utils.CSVReader(_join(self.source, vehicle_class_factor_file)) as r: - for row in r: - if "YEAR" in r.fields and int(row["YEAR"]) != scenario_year: # optional year column - continue - name = row["FACILITY_NAME"] - # optional time-of-day entry, default to ALL if no column or blank - fac_time = row.get("TIME_OF_DAY") - if fac_time is None: - fac_time = "ALL" - facility_factors[name][fac_time] = { - "auto": float(row["DA_FACTOR"]), - "hov2": float(row["S2_FACTOR"]), - "hov3": float(row["S3_FACTOR"]), - "lgt_truck": float(row["TRK_L_FACTOR"]), - "med_truck": float(row["TRK_M_FACTOR"]), - "hvy_truck": float(row["TRK_H_FACTOR"]) - } - facility_factors[name]["count"] = 0 - - # validate ToD entry, either list EA, AM, MD, PM and EV, or ALL, but not both - for name, factors in facility_factors.iteritems(): - # default keys should be "ALL" and "count" - if "ALL" in factors: - if len(factors) > 2: - fatal_errors += 1 - msg = ("Individual time periods and 'ALL' (or blank) listed under " - "TIME_OF_DAY column in {} for facility {}").format(vehicle_class_factor_file, name) - self._log.append({"type": "text", "content": msg}) - self._error.append(msg) - elif set(periods + ["count"]) != set(factors.keys()): - fatal_errors += 1 - msg = ("Missing time periods {} under TIME_OF_DAY column in {} for facility {}").format( - (set(periods) - set(factors.keys())), vehicle_class_factor_file, name) - self._log.append({"type": "text", "content": msg}) - self._error.append(msg) - - def lookup_link_name(link): - for attr_name in ["#name", "#name_from", "#name_to"]: - for name, _factors in facility_factors.iteritems(): - if name in link[attr_name]: - return _factors - return facility_factors["DEFAULT_FACTORS"] - - def match_facility_factors(link): - factors = lookup_link_name(link) - factors["count"] += 1 - factors = _copy(factors) - del factors["count"] - # @lane_restriction = 2 or 3 overrides hov2 and hov3 costs - if link["@lane_restriction"] == 2: - for _, time_factors in factors.iteritems(): - time_factors["hov2"] = 0.0 - time_factors["hov3"] = 0.0 - elif link["@lane_restriction"] == 3: - for _, time_factors in factors.iteritems(): - time_factors["hov3"] = 0.0 - return factors - - vehicle_classes = ["auto", "hov2", "hov3", "lgt_truck", "med_truck", "hvy_truck"] - for link in network.links(): - if sum(link["@toll" + time] for time in time_periods) > 0: - factors = match_facility_factors(link) - for time, period in zip(time_periods, periods): - time_factors = factors.get(period, factors.get("ALL")) - for name in vehicle_classes: - link["@cost_" + name + time] = time_factors[name] * link["@toll" + time] + link["@cost_operating"] - else: - for time in time_periods: - for name in vehicle_classes: - link["@cost_" + name + time] = link["@cost_operating"] - for name, class_factors in facility_factors.iteritems(): - msg = "Facility name '%s' matched to %s links." % (name, class_factors["count"]) - self._log.append({"type": "text2", "content": msg}) - - self._log.append({"type": "text", "content": "Calculation and time period expansion of costs, tolls, capacities and times complete"}) - - # calculate static reliability - for link in network.links(): - for time in time_periods: - sta_reliability = "@sta_reliability" + time - # if freeway apply freeway parameters to this link - if link["type"] == 1 and link["@lane" + time] > 0: - high_speed_factor = freeway_rel["speed>70"] if link["@speed_posted"] >= 70 else 0.0 - upstream_factor = freeway_rel["upstream"] * 1 / link["@intdist_up"] - downstream_factor = freeway_rel["downstream"] * 1 / link["@intdist_down"] - link[sta_reliability] = ( - freeway_rel["intercept"] + high_speed_factor + upstream_factor + downstream_factor) - # arterial/ramp/other apply road parameters - elif link["type"] <= 9 and link["@lane" + time] > 0: - lane_factor = road_rel["lanes"].get(link["@lane" + time], 0.0) - speed_bin = link["@speed_posted"] - if speed_bin < 35: - speed_bin = "<35" - elif speed_bin > 50: - speed_bin = ">50" - speed_factor = road_rel["speed"][speed_bin] - control_bin = min(max(link["@traffic_control"], 0), 4) - control_factor = road_rel["control"][control_bin] - link[sta_reliability] = road_rel["intercept"] + lane_factor + speed_factor + control_factor - else: - link[sta_reliability] = 0.0 - self._log.append({"type": "text", "content": "Calculate of link static reliability factors complete"}) - - # Cycle length matrix - # Intersecting Link - # Approach Link 2 3 4 5 6 7 8 9 - # IFC Description - # 2 Prime Arterial 2.5 2 2 2 2 2 2 2 - # 3 Major Arterial 2 2 2 2 2 2 2 2 - # 4 Collector 2 2 1.5 1.5 1.5 1.5 1.5 1.5 - # 5 Local Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - # 6 Rural Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - # 7 Local Road 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - # 8 Freeway connector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - # 9 Local Ramp 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - - # Volume-delay functions - # fd10: freeway node approach - # fd11: non-intersection node approach - # fd20: cycle length 1.25 - # fd21: cycle length 1.5 - # fd22: cycle length 2.0 - # fd23: cycle length 2.5 - # fd24: cycle length 2.5 and metered ramp - # fd25: freeway node approach AM and PM only - network.create_attribute("LINK", "green_to_cycle") - network.create_attribute("LINK", "cycle") - vdf_cycle_map = {1.25: 20, 1.5: 21, 2.0: 22, 2.5: 23} - for node in network.nodes(): - incoming = list(node.incoming_links()) - outgoing = list(node.outgoing_links()) - is_signal = False - for link in incoming: - if link["@green_to_cycle_init"] > 0: - is_signal = True - break - if is_signal: - lcs = [link.type for link in incoming + outgoing] - min_lc = max(lcs) # Note: minimum class is actually the HIGHEST value, - max_lc = min(lcs) # and maximum is the LOWEST - - for link in incoming: - # Metered ramps - if link["@traffic_control"] in [4, 5]: - link["cycle"] = 2.5 - link["green_to_cycle"] = 0.42 - link.volume_delay_func = 24 - # Stops - elif link["@traffic_control"] in [2, 3]: - link["cycle"] = 1.25 - link["green_to_cycle"] = 0.42 - link.volume_delay_func = 20 - elif link["@green_to_cycle_init"] > 0 and is_signal: - if link.type == 2: - c_len = 2.5 if min_lc == 2 else 2.0 - elif link.type == 3: - c_len = 2.0 # Major arterial & anything - elif link.type == 4: - c_len = 1.5 if max_lc > 2 else 2.0 - elif link.type > 4: - if max_lc > 4: - c_len = 1.25 - elif max_lc == 4: - c_len = 1.5 - else: - c_len = 2.0 - if link["@green_to_cycle_init"] > 10: - link["green_to_cycle"] = link["@green_to_cycle_init"] / 100.0 - if link["green_to_cycle"] > 1.0: - link["green_to_cycle"] = 1.0 - link["cycle"] = c_len - link.volume_delay_func = vdf_cycle_map[c_len] - elif link.type == 1: - link.volume_delay_func = 10 # freeway - else: - link.volume_delay_func = 11 # non-controlled approach - self._log.append({"type": "text", "content": "Derive cycle, green_to_cycle, and VDF by approach node complete"}) - - for link in network.links(): - if link.volume_delay_func in [10, 11]: - continue - if link["@traffic_control"] in [4, 5]: - # Ramp meter controlled links are only enabled during the peak periods - for time in ["_am", "_pm"]: - link["@cycle" + time] = link["cycle"] - link["@green_to_cycle" + time] = link["green_to_cycle"] - else: - for time in time_periods: - link["@cycle" + time] = link["cycle"] - link["@green_to_cycle" + time] = link["green_to_cycle"] - self._log.append({"type": "text", "content": "Setting of time period @cycle and @green_to_cycle complete"}) - - network.delete_attribute("LINK", "green_to_cycle") - network.delete_attribute("LINK", "cycle") - network.delete_attribute("NODE", "is_interchange") - self._log.append({"type": "text", "content": "Calculate derived traffic attributes complete"}) - if fatal_errors > 0: - raise Exception("%s fatal errors during calculation of traffic attributes" % fatal_errors) - return - - def check_zone_access(self, network, mode): - # Verify that every centroid has at least one available - # access and egress connector - for centroid in network.centroids(): - access = egress = False - for link in centroid.outgoing_links(): - if mode in link.modes: - if link.j_node.is_intersection: - for turn in link.outgoing_turns(): - if turn.i_node != turn.k_node and turn.penalty_func != 0: - egress = True - else: - egress = True - if not egress: - raise Exception("No egress permitted from zone %s" % centroid.id) - for link in centroid.incoming_links(): - if mode in link.modes: - if link.j_node.is_intersection: - for turn in link.incoming_turns(): - if turn.i_node != turn.k_node and turn.penalty_func != 0: - access = True - else: - access = True - if not access: - raise Exception("No access permitted to zone %s" % centroid.id) - - def add_transit_to_traffic(self, hwy_network, tr_network): - if not self.merged_scenario_id or not hwy_network or not tr_network: - return - self._log.append({"type": "header", "content": "Merge transit network to traffic network"}) - fatal_errors = 0 - for tr_mode in tr_network.modes(): - hwy_mode = hwy_network.create_mode(tr_mode.type, tr_mode.id) - hwy_mode.description = tr_mode.description - hwy_mode.speed = tr_mode.speed - for tr_veh in tr_network.transit_vehicles(): - hwy_veh = hwy_network.create_transit_vehicle(tr_veh.id, tr_veh.mode.id) - hwy_veh.description = tr_veh.description - hwy_veh.auto_equivalent = tr_veh.auto_equivalent - hwy_veh.seated_capacity = tr_veh.seated_capacity - hwy_veh.total_capacity = tr_veh.total_capacity - - for elem_type in ["NODE", "LINK", "TRANSIT_LINE", "TRANSIT_SEGMENT"]: - for attr in tr_network.attributes(elem_type): - if not attr in hwy_network.attributes(elem_type): - default = "" if attr.startswith("#") else 0 - new_attr = hwy_network.create_attribute(elem_type, attr, default) - - hwy_link_index = dict((l["@tcov_id"], l) for l in hwy_network.links()) - hwy_node_position_index = dict(((n.x, n.y), n) for n in hwy_network.nodes()) - hwy_node_index = dict() - not_matched_links = [] - for tr_link in tr_network.links(): - tcov_id = tr_link["@tcov_id"] - if tcov_id == 0: - i_node = hwy_node_position_index.get((tr_link.i_node.x, tr_link.i_node.y)) - j_node = hwy_node_position_index.get((tr_link.j_node.x, tr_link.j_node.y)) - if i_node and j_node: - hwy_link = hwy_network.link(i_node, j_node) - else: - hwy_link = None - else: - hwy_link = hwy_link_index.get(tcov_id) - if not hwy_link: - not_matched_links.append(tr_link) - else: - hwy_node_index[tr_link.i_node] = hwy_link.i_node - hwy_node_index[tr_link.j_node] = hwy_link.j_node - hwy_link.modes |= tr_link.modes - - new_node_id = max(n.number for n in hwy_network.nodes()) - new_node_id = int(_ceiling(new_node_id / 10000.0) * 10000) - bus_mode = tr_network.mode("b") - - def lookup_node(src_node, new_node_id): - node = hwy_node_index.get(src_node) - if not node: - node = hwy_node_position_index.get((src_node.x, src_node.y)) - if not node: - node = hwy_network.create_regular_node(new_node_id) - new_node_id += 1 - for attr in tr_network.attributes("NODE"): - node[attr] = src_node[attr] - hwy_node_index[src_node] = node - return node, new_node_id - - for tr_link in not_matched_links: - i_node, new_node_id = lookup_node(tr_link.i_node, new_node_id) - j_node, new_node_id = lookup_node(tr_link.j_node, new_node_id) - # check for duplicate but different links - # All cases to be logged and then an error raised at end - ex_link = hwy_network.link(i_node, j_node) - if ex_link: - self._log.append({ - "type": "text", - "content": "Duplicate links between the same nodes with different IDs in traffic/transit merge. " - "Traffic link ID %s, transit link ID %s." % (ex_link["@tcov_id"], tr_link["@tcov_id"]) - }) - self._error.append("Duplicate links with different IDs between traffic (%s) and transit (%s) networks" % - (ex_link["@tcov_id"], tr_link["@tcov_id"])) - self._split_link(hwy_network, i_node, j_node, new_node_id) - new_node_id += 1 - fatal_errors += 1 - try: - link = hwy_network.create_link(i_node, j_node, tr_link.modes) - except Exception as error: - self._log.append({ - "type": "text", - "content": "Error creating link '%s', I-node '%s', J-node '%s'. Error message %s" % - (tr_link["@tcov_id"], i_node, j_node, error) - }) - self._error.append("Cannot create transit link '%s' in traffic network" % tr_link["@tcov_id"]) - fatal_errors += 1 - continue - hwy_link_index[tr_link["@tcov_id"]] = link - for attr in tr_network.attributes("LINK"): - link[attr] = tr_link[attr] - link.vertices = tr_link.vertices - - # Create transit lines and copy segment data - for tr_line in tr_network.transit_lines(): - itinerary = [] - for seg in tr_line.segments(True): - itinerary.append(hwy_node_index[seg.i_node]) - try: - hwy_line = hwy_network.create_transit_line(tr_line.id, tr_line.vehicle.id, itinerary) - except Exception as error: - msg = "Transit line %s, error message %s" % (tr_line.id, error) - self._log.append({"type": "text", "content": msg}) - self._error.append("Cannot create transit line '%s' in traffic network" % tr_line.id) - fatal_errors += 1 - continue - for attr in hwy_network.attributes("TRANSIT_LINE"): - hwy_line[attr] = tr_line[attr] - for tr_seg, hwy_seg in _izip(tr_line.segments(True), hwy_line.segments(True)): - for attr in hwy_network.attributes("TRANSIT_SEGMENT"): - hwy_seg[attr] = tr_seg[attr] - - # Change ttf from ft2 (fixed speed) to ft1 (congested auto time) - auto_mode = hwy_network.mode("d") - for hwy_link in hwy_network.links(): - if auto_mode in hwy_link.modes: - for seg in hwy_link.segments(): - seg.transit_time_func = 1 - if fatal_errors > 0: - raise Exception("Cannot merge traffic and transit network, %s fatal errors found" % fatal_errors) - - self._log.append({"type": "text", "content": "Merge transit network to traffic network complete"}) - - def _split_link(self, network, i_node, j_node, new_node_id): - # Attribute types to maintain consistency for correspondence with incoming / outgoing link data - periods = ["ea", "am", "md", "pm", "ev"] - approach_attrs = ["@traffic_control", "@turn_thru", "@turn_right", "@turn_left", - "@lane_auxiliary", "@green_to_cycle_init"] - for p_attr in ["@green_to_cycle_", "@time_inter_", "@cycle_"]: - approach_attrs.extend([p_attr + p for p in periods]) - capacity_inter = ["@capacity_inter_" + p for p in periods] - cost_attrs = ["@cost_operating"] - for p_attr in ["@cost_lgt_truck_", "@cost_med_truck_", "@cost_hvy_truck_", "@cost_hov2_", - "@cost_hov3_", "@cost_auto_", "@time_link_", "@trtime_link_", "@toll_"]: - cost_attrs.extend([p_attr + p for p in periods]) - approach_attrs = [a for a in approach_attrs if a in network.attributes("LINK")] - capacity_inter = [a for a in capacity_inter if a in network.attributes("LINK")] - cost_attrs = [a for a in cost_attrs if a in network.attributes("LINK")] - - new_node = network.split_link(i_node, j_node, new_node_id) - - # Correct attributes on the split links - for link in new_node.incoming_links(): - link["#name_to"] = "" - for attr in approach_attrs: - link[attr] = 0 - for attr in capacity_inter: - link[attr] = 999999 - for attr in cost_attrs: - link[attr] = 0.5 * link[attr] - link.volume_delay_func = 10 - for link in new_node.outgoing_links(): - link["#name_from"] = "" - for attr in cost_attrs: - link[attr] = 0.5 * link[attr] - - - @_m.logbook_trace("Set database functions (VDF, TPF and TTF)") - def set_functions(self, scenario): - create_function = _m.Modeller().tool( - "inro.emme.data.function.create_function") - set_extra_function_params = _m.Modeller().tool( - "inro.emme.traffic_assignment.set_extra_function_parameters") - emmebank = self.emmebank - for f_id in ["fd10", "fd11", "fd20", "fd21", "fd22", "fd23", "fd24", "fp1", "ft1", "ft2", "ft3", "ft4"]: - function = emmebank.function(f_id) - if function: - emmebank.delete_function(function) - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(_dir(self.source), "conf", "sandag_abm.properties")) - smartSignalf_CL = props["smartSignal.factor.LC"] - smartSignalf_MA = props["smartSignal.factor.MA"] - smartSignalf_PA = props["smartSignal.factor.PA"] - atdmf = props["atdm.factor"] - - reliability_tmplt = ( - "* (1 + el2 + {0}*(".format(atdmf)+ - "( {factor[LOS_C]} * ( put(get(1).min.1.5) - {threshold[LOS_C]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_C]})" - "+ ( {factor[LOS_D]} * ( get(2) - {threshold[LOS_D]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_D]})" - "+ ( {factor[LOS_E]} * ( get(2) - {threshold[LOS_E]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_E]})" - "+ ( {factor[LOS_FL]} * ( get(2) - {threshold[LOS_FL]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_FL]})" - "+ ( {factor[LOS_FH]} * ( get(2) - {threshold[LOS_FH]} + 0.01 ) ) * (get(1) .gt. {threshold[LOS_FH]})" - "))") - parameters = { - "freeway": { - "factor": { - "LOS_C": 0.2429, "LOS_D": 0.1705, "LOS_E": -0.2278, "LOS_FL": -0.1983, "LOS_FH": 1.022 - }, - "threshold": { - "LOS_C": 0.7, "LOS_D": 0.8, "LOS_E": 0.9, "LOS_FL": 1.0, "LOS_FH": 1.2 - }, - }, - "road": { # for arterials, ramps, collectors, local roads, etc. - "factor": { - "LOS_C": 0.1561, "LOS_D": 0.0, "LOS_E": 0.0, "LOS_FL": -0.449, "LOS_FH": 0.0 - }, - "threshold": { - "LOS_C": 0.7, "LOS_D": 0.8, "LOS_E": 0.9, "LOS_FL": 1.0, "LOS_FH": 1.2 - }, - } - } - # freeway fd10 - create_function( - "fd10", - "(ul1 * (1.0 + 0.24 * put((volau + volad) / ul3) ** 5.5))" - + reliability_tmplt.format(**parameters["freeway"]), - emmebank=emmebank) - # non-freeway link which is not an intersection approach fd11 - create_function( - "fd11", - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0))" - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - create_function( - "fd20", # Local collector and lower intersection and stop controlled approaches - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" - "1.25 / 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))" - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - create_function( - "fd21", # Collector intersection approaches - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" - "{0} * 1.5/ 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_CL) - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - create_function( - "fd22", # Major arterial and major or prime arterial intersection approaches - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" - "{0} * 2.0 / 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_MA) - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - create_function( - "fd23", # Primary arterial intersection approaches - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" - "{0} * 2.5/ 2 * (1-el1) ** 2 * (1.0 + 4.5 * ( (volau + volad) / el3 ) ** 2.0))".format(smartSignalf_PA) - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - create_function( - "fd24", # Metered ramps - "(ul1 * (1.0 + 0.8 * put((volau + volad) / ul3) ** 4.0) +" - "2.5/ 2 * (1-el1) ** 2 * (1.0 + 6.0 * ( (volau + volad) / el3 ) ** 2.0))" - + reliability_tmplt.format(**parameters["road"]), - emmebank=emmebank) - # freeway fd25 (AM and PM only) - create_function( - "fd25", - "(ul1 * (1.0 + 0.6 * put((volau + volad) / ul3) ** 4))" - + reliability_tmplt.format(**parameters["freeway"]), - emmebank=emmebank) - - set_extra_function_params( - el1="@green_to_cycle", el2="@sta_reliability", el3="@capacity_inter_am", - emmebank=emmebank) - - create_function("fp1", "up1", emmebank=emmebank) # fixed cost turns stored in turn data 1 (up1) - - # buses in mixed traffic, use auto time - create_function("ft1", "timau", emmebank=emmebank) - # fixed speed for separate guideway operations - create_function("ft2", "ul2", emmebank=emmebank) - # special 0-cost segments for prohibition of walk to different stop from centroid - create_function("ft3", "0", emmebank=emmebank) - # fixed guideway systems according to vehicle speed (not used at the moment) - create_function("ft4", "60 * length / speed", emmebank=emmebank) - - @_m.logbook_trace("Traffic zone connectivity check") - def check_connectivity(self, scenario): - modeller = _m.Modeller() - sola_assign = modeller.tool( - "inro.emme.traffic_assignment.sola_traffic_assignment") - set_extra_function_para = modeller.tool( - "inro.emme.traffic_assignment.set_extra_function_parameters") - create_matrix = _m.Modeller().tool( - "inro.emme.data.matrix.create_matrix") - net_calc = gen_utils.NetworkCalculator(scenario) - - emmebank = scenario.emmebank - zone_index = dict(enumerate(scenario.zone_numbers)) - num_processors = dem_utils.parse_num_processors("MAX-1") - - # Note matrix is also created in initialize_matrices - create_matrix("ms1", "zero", "zero", scenario=scenario, overwrite=True) - with gen_utils.temp_matrices(emmebank, "FULL", 1) as (result_matrix,): - result_matrix.name = "TEMP_SOV_TRAVEL_TIME" - set_extra_function_para( - el1="@green_to_cycle_am", - el2="@sta_reliability_am", - el3="@capacity_inter_am", emmebank=emmebank) - net_calc("ul1", "@time_link_am", "modes=d") - net_calc("ul3", "@capacity_link_am", "modes=d") - net_calc("lanes", "@lane_am", "modes=d") - spec = { - "type": "SOLA_TRAFFIC_ASSIGNMENT", - "background_traffic": None, - "classes": [ - { - "mode": "S", # SOV toll mode - "demand": 'ms"zero"', - "generalized_cost": None, - "results": { - "od_travel_times": {"shortest_paths": result_matrix.named_id} - } - } - ], - "stopping_criteria": { - "max_iterations": 0, "best_relative_gap": 0.0, - "relative_gap": 0.0, "normalized_gap": 0.0 - }, - "performance_settings": {"number_of_processors": num_processors}, - } - sola_assign(spec, scenario=scenario) - travel_time = result_matrix.get_numpy_data(scenario) - - is_disconnected = (travel_time == 1e20) - disconnected_pairs = is_disconnected.sum() - if disconnected_pairs > 0: - error_msg = "Connectivity error(s) between %s O-D pairs" % disconnected_pairs - self._log.append({"type": "header", "content": error_msg}) - count_disconnects = [] - for axis, term in [(0, "from"), (1, "to")]: - axis_totals = is_disconnected.sum(axis=axis) - for i, v in enumerate(axis_totals): - if v > 0: - count_disconnects.append((zone_index[i], term, v)) - count_disconnects.sort(key=lambda x: x[2], reverse=True) - for z, direction, count in count_disconnects[:50]: - msg ="Zone %s disconnected %s %d other zones" % (z, direction, count) - self._log.append({"type": "text", "content": msg}) - if disconnected_pairs > 50: - self._log.append({"type": "text", "content": "[List truncated]"}) - raise Exception(error_msg) - self._log.append({"type": "header", "content": - "Zone connectivity verified for AM period on SOV toll ('S') mode"}) - scenario.has_traffic_results = False - - def log_report(self): - report = _m.PageBuilder(title="Import network from TCOVED files report") - try: - if self._error: - report.add_html("
    Errors detected during import: %s
    " % len(self._error)) - error_msg = ["
      "] - for error in self._error: - error_msg.append("
    • %s
    • " % error) - error_msg.append("
    ") - report.add_html("".join(error_msg)) - else: - report.add_html("No errors detected during import") - - for item in self._log: - if item["type"] == "text": - report.add_html("
    %s
    " % item["content"]) - if item["type"] == "text2": - report.add_html("
    %s
    " % item["content"]) - elif item["type"] == "header": - report.add_html("

    %s

    " % item["content"]) - elif item["type"] == "table": - table_msg = ["
    ", "

    %s

    " % item["title"]] - if "header" in item: - table_msg.append("") - for label in item["header"]: - table_msg.append("" % label) - table_msg.append("") - for row in item["content"]: - table_msg.append("") - for cell in row: - table_msg.append("" % cell) - table_msg.append("") - table_msg.append("
    %s
    %s
    ") - report.add_html("".join(table_msg)) - - except Exception as error: - # no raise during report to avoid masking real error - report.add_html("Error generating report") - report.add_html(unicode(error)) - report.add_html(_traceback.format_exc()) - - _m.logbook_write("Import network report", report.render()) - - -def get_node(network, number, coordinates, is_centroid): - node = network.node(number) - if not node: - node = network.create_node(number, is_centroid) - node.x, node.y = coordinates - return node - - -# shortest path interpolation -def find_path(orig_link, dest_link, mode): - visited = set([]) - visited_add = visited.add - back_links = {} - heap = [] - - for link in orig_link.j_node.outgoing_links(): - if mode in link.modes: - back_links[link] = None - _heapq.heappush(heap, (link["length"], link)) - - link_found = False - try: - while not link_found: - link_cost, link = _heapq.heappop(heap) - if link in visited: - continue - visited_add(link) - for outgoing in link.j_node.outgoing_links(): - if mode not in outgoing.modes: - continue - if outgoing in visited: - continue - back_links[outgoing] = link - if outgoing == dest_link: - link_found = True - break - outgoing_cost = link_cost + link["length"] - _heapq.heappush(heap, (outgoing_cost, outgoing)) - except IndexError: - pass # IndexError if heap is empty - if not link_found: - raise NoPathException( - "no path found between links with trcov_id %s and %s (Emme IDs %s and %s)" % ( - orig_link["@tcov_id"], dest_link["@tcov_id"], orig_link, dest_link)) - - prev_link = back_links[dest_link] - route = [] - while prev_link: - route.append(prev_link) - prev_link = back_links[prev_link] - return list(reversed(route)) - - -class NoPathException(Exception): - pass - - -def revised_headway(headway): - # CALCULATE REVISED HEADWAY - # new headway calculation is less aggressive; also only being used for initial wait - # It uses a negative exponential formula to calculate headway - # - if headway <= 10: - rev_headway = headway - else: - rev_headway = headway * (0.275 + 0.788 * _np.exp(-0.011*headway)) - return rev_headway - - -def interchange_distance(orig_link, direction): - visited = set([]) - visited_add = visited.add - back_links = {} - heap = [] - if direction == "DOWNSTREAM": - get_links = lambda l: l.j_node.outgoing_links() - check_far_node = lambda l: l.j_node.is_interchange - elif direction == "UPSTREAM": - get_links = lambda l: l.i_node.incoming_links() - check_far_node = lambda l: l.i_node.is_interchange - # Shortest path search for nearest interchange node along freeway - for link in get_links(orig_link): - _heapq.heappush(heap, (link["length"], link)) - interchange_found = False - try: - while not interchange_found: - link_cost, link = _heapq.heappop(heap) - if link in visited: - continue - visited_add(link) - if check_far_node(link): - interchange_found = True - break - for next_link in get_links(link): - if next_link in visited: - continue - next_cost = link_cost + link["length"] - _heapq.heappush(heap, (next_cost, next_link)) - except IndexError: - # IndexError if heap is empty - # case where start / end of highway, dist = 99 - return 99 - return orig_link["length"] / 2.0 + link_cost diff --git a/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py deleted file mode 100644 index 2b87f35..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/import_seed_demand.py +++ /dev/null @@ -1,193 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/import_seed_demand.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Imports the warm start demand matrices from specified OMX files for auto, truck and transit. -# -# Note the matrix name mapping from the OMX file names to the Emme database names. -# -# Inputs: -# omx_file: source -# demand_type: The type of demand in the provided OMX file, one of "AUTO", "TRUCK", "TRANSIT". -# Used to determine the matrix mapping for the import. -# period: The period for which to import the matrices, one of "EA", "AM", "MD", "PM", "EV" -# scenario: traffic scenario to use for reference zone system -# convert_truck_to_pce: boolean, if True the result matrices are adjusted to PCEs instead of -# vehicles (default, and required for traffic assignment). Only used if the demand_type is TRUCK. -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# For AUTO: -# pp_SOVNTP, pp_SOVTB, pp_HOV2, pp_HOV3 -# For TRUCK: -# pp_TRKH, pp_TRKL, pp_TRKM -# For TRANSIT: -# pp_WLKBUS, pp_WLKLRT, pp_WLKCMR, pp_WLKEXP, pp_WLKBRT, -# pp_PNRBUS, pp_PNRLRT, pp_PNRCMR, pp_PNREXP, pp_PNRBRT, -# pp_KNRBUS, pp_KNRLRT, pp_KNRCMR, pp_KNREXP, pp_KNRBRT -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - period = "AM" - input_omx_file = os.path.join(main_directory, "input", "trip_%s.omx" % period) - demand_type = "TRUCK" - demand_as_pce = True - base_scenario = modeller.scenario - import_seed_demand = modeller.tool("sandag.import.import_seed_demand") - import_seed_demand(input_omx_file, demand_type, period, demand_as_pce, base_scenario) -""" - - -TOOLBOX_ORDER = 12 - - -import inro.modeller as _m -import inro.emme.matrix as _matrix -import traceback as _traceback - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -_omx = _m.Modeller().module("sandag.utilities.omxwrapper") - - -class ImportMatrices(_m.Tool(), gen_utils.Snapshot): - - omx_file = _m.Attribute(unicode) - demand_type = _m.Attribute(str) - period = _m.Attribute(str) - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.attributes = ["omx_file", "demand_type", "period"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Import demand matrices" - pb.description = """Imports the seed demand matrices.""" - pb.branding_text = "- SANDAG - Import" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('omx_file', 'file', - title='Select input OMX file') - options = [(x, x) for x in ["AUTO", "TRUCK", "TRANSIT"]] - pb.add_select("demand_type", keyvalues=options, title="Select corresponding demand type") - options = [(x, x) for x in ["EA", "AM", "MD", "PM", "EV"]] - pb.add_select("period", keyvalues=options, title="Select corresponding period") - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.omx_file, self.demand_type, self.period, scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, omx_file, demand_type, period, scenario): - attributes = { - "omx_file": omx_file, - "demand_type": demand_type, - "period": period, - "scenario": scenario.id, - "self": str(self) - } - with _m.logbook_trace("Import %s matrices for period %s" % (demand_type, period), attributes=attributes): - gen_utils.log_snapshot("Import matrices", str(self), attributes) - demand_types = ["AUTO", "TRUCK", "TRANSIT"] - if demand_type not in demand_types: - raise Exception("Invalid demand_type, must be one of %s" % demand_types) - periods = ["EA", "AM", "MD", "PM", "EV"] - if period not in periods: - raise Exception("Invalid period, must be one of %s" % periods) - - if demand_type == "AUTO": - # TODO: update for new seed matrices - matrices = { - '%s_SOV_NT_L': 'mf"%s_SOV_NT_L"', - '%s_SOV_TR_L': 'mf"%s_SOV_TR_L"', - '%s_HOV2_L': 'mf"%s_HOV2_L"', - '%s_HOV3_L': 'mf"%s_HOV3_L"', - '%s_SOV_NT_M': 'mf"%s_SOV_NT_M"', - '%s_SOV_TR_M': 'mf"%s_SOV_TR_M"', - '%s_HOV2_M': 'mf"%s_HOV2_M"', - '%s_HOV3_M': 'mf"%s_HOV3_M"', - '%s_SOV_NT_H': 'mf"%s_SOV_NT_H"', - '%s_SOV_TR_H': 'mf"%s_SOV_TR_H"', - '%s_HOV2_H': 'mf"%s_HOV2_H"', - '%s_HOV3_H': 'mf"%s_HOV3_H"'} - matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) - self._import_from_omx(omx_file, matrices, scenario) - - if demand_type == "TRUCK": - # TODO: update for new seed matrices - matrices = { - '%s_TRK_H': 'mf"%s_TRK_H"', - '%s_TRK_L': 'mf"%s_TRK_L"', - '%s_TRK_M': 'mf"%s_TRK_M"'} - matrices = dict((k % period, v % period) for k, v in matrices.iteritems()) - self._import_from_omx(omx_file, matrices, scenario) - - if demand_type == "TRANSIT": - matrices = { - 'SET1': 'mf"%s_WLKBUS"', - 'SET2': 'mf"%s_WLKPREM"', - 'SET3': 'mf"%s_WLKALLPEN"',} - matrices = dict((k, v % period) for k, v in matrices.iteritems()) - # special custom mapping from subset of TAPs to all TAPs - self._import_from_omx(omx_file, matrices, scenario) - - def _import_from_omx(self, file_path, matrices, scenario): - matrices_to_write = {} - emme_zones = scenario.zone_numbers - emmebank = scenario.emmebank - omx_file_obj = _omx.open_file(file_path, 'r') - try: - zone_mapping = omx_file_obj.mapping(omx_file_obj.list_mappings()[0]).items() - zone_mapping.sort(key=lambda x: x[1]) - omx_zones = [x[0] for x in zone_mapping] - for omx_name, emme_name in matrices.iteritems(): - omx_data = omx_file_obj[omx_name].read() - if emme_name not in matrices_to_write: - matrices_to_write[emme_name] = omx_data - else: - # Allow multiple src matrices from OMX to sum to same matrix in Emme - matrices_to_write[emme_name] = omx_data + matrices_to_write[emme_name] - except Exception as error: - import traceback - print (traceback.format_exc()) - omx_file_obj.close() - - if omx_zones != emme_zones: - # special custom mapping from subset of TAPs to all TAPs - for emme_name, omx_data in matrices_to_write.iteritems(): - matrix_data = _matrix.MatrixData(type='f', indices=[omx_zones, omx_zones]) - matrix_data.from_numpy(omx_data) - expanded_matrix_data = matrix_data.expand([emme_zones, emme_zones]) - matrix = emmebank.matrix(emme_name) - matrix.set_data(expanded_matrix_data, scenario) - else: - for emme_name, omx_data in matrices_to_write.iteritems(): - matrix = emmebank.matrix(emme_name) - matrix.set_numpy_data(omx_data, scenario) diff --git a/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py b/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py deleted file mode 100644 index 827baff..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/import_transit_demand.py +++ /dev/null @@ -1,230 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/import_transit_demand.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Imports the transit demand generated from an iteration of the disaggregate -# demand models (CT-RAMP) in preparation for the transit assignment -# -# Note the matrix name mapping from the OMX file names to the Emme database names. -# -# Inputs: -# output_dir: output directory to read the OMX files from -# scenario: transit scenario to use for reference zone system -# -# Files referenced: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# output/tranTrips_pp.omx -# output/tranCrossBorderTrips_pp.omx -# output/tranAirportTrips.SAN_pp.omx -# output/tranAirportTrips.CBX_pp.omx (optional) -# output/tranVisitorTrips_pp.omx -# output/tranInternalExternalTrips_pp.omx -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# pp_WLKBUS, pp_WLKLRT, pp_WLKCMR, pp_WLKEXP, pp_WLKBRT, -# pp_PNRBUS, pp_PNRLRT, pp_PNRCMR, pp_PNREXP, pp_PNRBRT, -# pp_KNRBUS, pp_KNRLRT, pp_KNRCMR, pp_KNREXP, pp_KNRBRT -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - output_dir = os.path.join(main_directory, "output") - scenario = modeller.scenario - import_transit_demand = modeller.tool("sandag.import.import_transit_demand") - import_transit_demand(output_dir, scenario) -""" - - -TOOLBOX_ORDER = 14 - - -import inro.modeller as _m -import inro.emme.matrix as _matrix -import traceback as _traceback -import os - - -dem_utils = _m.Modeller().module('sandag.utilities.demand') -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ImportMatrices(_m.Tool(), gen_utils.Snapshot): - - output_dir = _m.Attribute(unicode) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - main_dir = os.path.dirname(project_dir) - self.output_dir = os.path.join(main_dir, "output") - self.attributes = ["output_dir"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Import transit demand" - pb.description = """ -
    - Imports the trip matrices generated by CT-RAMP in OMX format.
    - A total of 30 OMX files are expected, for 5 time periods - EA, AM, MD, PM and EV, with internal matrices by 3 model segments - (assignment access sets) and 3 access modes (walk, PNR, KNR): -
      -
    • tranTrips_pp.omx
    • -
    • tranCrossBorderTrips_pp.omx
    • -
    • tranAirportTrips.SAN_pp.omx
    • -
    • tranAirportTrips.CBX_pp.omx (optional)
    • -
    • tranVisitorTrips_pp.omx
    • -
    • tranInternalExternalTrips_pp.omx
    • -
    -
    - """ - pb.branding_text = "- SANDAG - Model" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_select_file('output_dir', 'directory', - title='Select output directory') - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.output_dir, scenario) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Create TOD transit trip tables", save_arguments=True) - def __call__(self, output_dir, scenario): - attributes = {"output_dir": output_dir} - gen_utils.log_snapshot("Sum demand", str(self), attributes) - - self.scenario = scenario - self.output_dir = output_dir - self.import_transit_trips() - - @_m.logbook_trace("Import CT-RAMP transit trips from OMX") - def import_transit_trips(self): - emmebank = self.scenario.emmebank - emme_zones = self.scenario.zone_numbers - matrix_name_tmplts = [ - ("mf%s_%sBUS", "%s_%s_set1_%s"), - ("mf%s_%sPREM", "%s_%s_set2_%s"), - ("mf%s_%sALLPEN", "%s_%s_set3_%s") - ] - periods = ["EA", "AM", "MD", "PM", "EV"] - access_modes = ["WLK", "PNR", "KNR"] - matrix_names = [] - for period in periods: - for acc_mode in access_modes: - #for trip_set in trip_sets: - for emme_name, omx_name in matrix_name_tmplts: - matrix_names.append( - ("_" + period, emme_name % (period, acc_mode), omx_name % (acc_mode, "%s", period))) - - with gen_utils.OMXManager(self.output_dir, "tran%sTrips%s.omx") as omx_manager: - for period, matrix_name, omx_key in matrix_names: - logbook_label = "Report on import from OMX key %s to matrix %s" % (omx_key % "SET", matrix_name) - - #add both KNR_SET and TNC_SET into KNR - if ("KNR" in matrix_name): - #resident - person_knr_demand = omx_manager.lookup(("", period), omx_key % "SET") - person_tnc_Demand = omx_manager.lookup(("", period), omx_key.replace("KNR","TNC") % "SET") - person_demand = person_knr_demand + person_tnc_Demand - #visitor - visitor_knr_demand = omx_manager.lookup(("Visitor", period), omx_key % "SET") - visitor_tnc_Demand = omx_manager.lookup(("Visitor", period), omx_key.replace("KNR","TNC") % "SET") - visitor_demand = visitor_knr_demand + visitor_tnc_Demand - #cross border - cross_border_knr_demand = omx_manager.lookup(("CrossBorder", period), omx_key % "SET") - cross_border_tnc_Demand = omx_manager.lookup(("CrossBorder", period), omx_key.replace("KNR","TNC") % "SET") - cross_border_demand = cross_border_knr_demand + cross_border_tnc_Demand - #airport SAN - airport_knr_demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key % "SET") - airport_tnc_Demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key.replace("KNR","TNC") % "SET") - #airport CBX - if omx_manager.file_exists(("Airport", ".CBX" + period)): - airport_knr_demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key % "SET") - airport_tnc_Demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key.replace("KNR","TNC") % "SET") - airport_demand = airport_knr_demand + airport_tnc_Demand - #internal external - internal_external_knr_demand = omx_manager.lookup(("InternalExternal", period), omx_key % "SET") - internal_external_tnc_Demand = omx_manager.lookup(("InternalExternal", period), omx_key.replace("KNR","TNC") % "SET") - internal_external_demand = internal_external_knr_demand + internal_external_tnc_Demand - else: - person_demand = omx_manager.lookup(("", period), omx_key % "SET") - visitor_demand = omx_manager.lookup(("Visitor", period), omx_key % "SET") - cross_border_demand = omx_manager.lookup(("CrossBorder", period), omx_key % "SET" ) - airport_demand = omx_manager.lookup(("Airport", ".SAN" + period), omx_key % "SET") - if omx_manager.file_exists(("Airport", ".CBX" + period)): - airport_demand += omx_manager.lookup(("Airport", ".CBX" + period), omx_key % "SET") - - internal_external_demand = omx_manager.lookup(("InternalExternal", period), omx_key % "SET") - - total_ct_ramp_trips = person_demand #for testing only - total_ct_ramp_trips = ( - visitor_demand + cross_border_demand + airport_demand - + person_demand + internal_external_demand) - - # Check the OMX zones are the same Emme database, assume all files have the same zones - omx_zones = omx_manager.zone_list("tranTrips%s.omx" % period) - matrix = emmebank.matrix(matrix_name) - if omx_zones != emme_zones: - matrix_data = _matrix.MatrixData(type='f', indices=[omx_zones, omx_zones]) - matrix_data.from_numpy(total_ct_ramp_trips) - expanded_matrix_data = matrix_data.expand([emme_zones, emme_zones]) - matrix.set_data(expanded_matrix_data, self.scenario) - else: - matrix.set_numpy_data(total_ct_ramp_trips, self.scenario) - - if ("KNR" in matrix_name): - dem_utils.demand_report([ - ("person_demand", person_demand), - (" person_knr_demand", person_knr_demand), - (" person_tnc_Demand", person_tnc_Demand), - ("internal_external_demand", internal_external_demand), - (" internal_external_knr_demand", internal_external_knr_demand), - (" internal_external_tnc_Demand", internal_external_tnc_Demand), - ("cross_border_demand", cross_border_demand), - (" cross_border_knr_demand", cross_border_knr_demand), - (" cross_border_tnc_Demand", cross_border_tnc_Demand), - ("airport_demand", airport_demand), - (" airport_knr_demand", airport_knr_demand), - (" airport_tnc_Demand", airport_tnc_Demand), - ("visitor_demand", visitor_demand), - (" visitor_knr_demand", visitor_knr_demand), - (" visitor_tnc_Demand", visitor_tnc_Demand), - ("total_ct_ramp_trips", total_ct_ramp_trips) - ], logbook_label, self.scenario) - else: - dem_utils.demand_report([ - ("person_demand", person_demand), - ("internal_external_demand", internal_external_demand), - ("cross_border_demand", cross_border_demand), - ("airport_demand", airport_demand), - ("visitor_demand", visitor_demand), - ("total_ct_ramp_trips", total_ct_ramp_trips) - ], logbook_label, self.scenario) diff --git a/sandag_abm/src/main/emme/toolbox/import/input_checker.py b/sandag_abm/src/main/emme/toolbox/import/input_checker.py deleted file mode 100644 index eedffcb..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/input_checker.py +++ /dev/null @@ -1,767 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright RSG, 2019-2020. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/input_checker.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Reviews all inputs to SANDAG ABM for possible issues that will result in model errors -# -# Files referenced: -# input_checker\config\inputs_checks.csv -# input_checker\config\inputs_list.csv - -import os, shutil, sys, time, csv, logging -import win32com.client as com -import numpy as np -import pandas as pd -import traceback as _traceback -import datetime -import warnings -from simpledbf import Dbf5 -import inro.modeller as _m -import inro.emme.database.emmebank as _eb -import inro.director.util.qtdialog as dialog -import textwrap - -warnings.filterwarnings("ignore") - -_join = os.path.join -_dir = os.path.dirname - -class input_checker(_m.Tool()): - - path = _m.Attribute(unicode) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = _dir(_m.Modeller().desktop.project.path) - self.path = _dir(project_dir) - self.input_checker_path = '' - self.inputs_list_path = '' - self.inputs_checks_path = '' - self.log_path = '' - self.logical_log_path = '' - self.prop_input_paths = {} - self.inputs_list = pd.DataFrame() - self.inputs_checks = pd.DataFrame() - self.inputs = {} - self.results = {} - self.result_list = {} - self.problem_ids = {} - self.report_stat = {} - self.num_fatal = int() - self.num_warning = int() - self.num_logical = int() - self.logical_fails = pd.DataFrame() - self.scenario_df = pd.DataFrame() - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Input Checker" - pb.description = """ - Reviews all inputs to SANDAG ABM for possible issues that could result - in model errors. List of inputs and checks are read from two CSV files: -
    -
    -
      -
    • input_checker\config\inputs_checks.csv
    • -
    • input_checker\config\inputs_list.csv
    • -
    -
    - The input checker goes through the list of checks and evaluates each - one as True or False. A summary file is produced at the end with results - for each check. The input checker additionally outputs a report for - failed checks of severity type Logical with more than 25 failed records. - The additional summary report lists every failed record. - The following reports are output: -
    -
    -
      -
    • input_checker\inputCheckerSummary_[YEAR-MM-DD].txt
    • -
    • completeLogicalFails_[YEAR-MM-DD].txt
    • -
    -
    - """ - pb.branding_text = "SANDAG - Input Checker" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self(path = self.path) - run_msg = "Input Checker Complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, path = ""): - _m.logbook_write("Started running input checker...") - - self.path = path - - self.input_checker_path = _join(self.path, 'input_checker') - self.inputs_list_path = _join(self.input_checker_path, 'config', 'inputs_list.csv') - self.inputs_checks_path = _join(self.input_checker_path, 'config', 'inputs_checks.csv') - - file_paths = [self.inputs_list_path, self.inputs_checks_path] - for path in file_paths: - if not os.path.exists(path): - raise Exception("missing file '%s'" % (path)) - - _m.logbook_write("Reading inputs...") - self.read_inputs() - - _m.logbook_write("Conducting checks...") - self.checks() - - _m.logbook_write("Writing logical fail logs...") - self.write_logical_log() - - _m.logbook_write("Writing logs...") - self.write_log() - - _m.logbook_write("Checking for logical errors...") - self.check_logical() - - _m.logbook_write("Checking for fatal errors...") - self.check_num_fatal() - - _m.logbook_write("Finisehd running input checker") - - def read_inputs(self): - # read list of inputs from CSV file - self.inputs_list = pd.read_csv(self.inputs_list_path) - - # remove all commented inputs from the inputs list - self.inputs_list = self.inputs_list.loc[[not i for i in (self.inputs_list['Input_Table'].str.startswith('#'))]] - - # obtain file paths from the sandag_abm.properties - self.prop_file_paths() - - # load emmebank - eb_path = _join(self.path, "emme_project", "Database", "emmebank") - eb = _eb.Emmebank(eb_path) - - # load emme network - network = eb.scenario(100).get_network() - - # create extra network attributes (maybe temporary) - - # link modes_str attribute - network.create_attribute("LINK", "mode_str") - for link in network.links(): - link.mode_str = "".join([m.id for m in link.modes]) - - # link isTransit flag attribute - network.create_attribute("LINK", "isTransit") - transit_modes = set([m for m in network.modes() if m.type == "TRANSIT"]) - for link in network.links(): - link.isTransit = bool(link.modes.intersection(transit_modes)) - - # transit segment isFirst and isLast flags attributes - network.create_attribute("TRANSIT_SEGMENT", "isFirst", False) - network.create_attribute("TRANSIT_SEGMENT", "isLast", False) - for line in network.transit_lines(): - first_seg = line.segment(0) - last_seg = line.segment(-2) - first_seg.isFirst = True - last_seg.isLast = True - - # node isCentroid flag attribute - network.create_attribute("NODE", "isCentroid", False) - centroids = [c for c in network.nodes() if c.is_centroid] - for node in network.nodes(): - node.isCentroid = bool(node in centroids) - - # node numInLinks and numOutLinks attributes - network.create_attribute("NODE", "numInLinks") - network.create_attribute("NODE", "numOutLinks") - for node in network.nodes(): - node.numInLinks = len(list(node.incoming_links())) - node.numOutLinks = len(list(node.outgoing_links())) - - # node hasLocalConnection flag attribute - class BreakLoop (Exception): - pass - - network.create_attribute("NODE", "hasLocalConnection", False) - for node in network.centroids(): - try: - for zone_connector in node.incoming_links(): - for local_link in zone_connector.i_node.incoming_links(): - if local_link["@lane_restriction"] == 1.0: - node.hasLocalConnection = True - raise BreakLoop("") - except: - pass - - # transit line hasTAP flag attribute - network.create_attribute("TRANSIT_LINE", "hasTAP", False) - for line in network.transit_lines(): - has_first_tap = False - has_last_tap = False - for link in line.segment(0).i_node.outgoing_links(): - if link.j_node["@tap_id"] > 0: - has_first_tap = True - break - for link in line.segment(-2).j_node.outgoing_links(): - if link.j_node["@tap_id"] > 0: - has_last_tap = True - break - line.hasTAP = has_first_tap and has_last_tap - - # link names attribute - network.create_attribute("LINK", "linkNames") - for link in network.links(): - link.linkNames = str(link['#name'] + "," + link['#name_from'] + "," + link['#name_to']) - - def get_emme_object(emme_network, emme_network_object, fields_to_export): - # Emme network attribute and object names - net_attr = { - 'NODE':'nodes', - 'LINK':'links', - 'TRANSIT_SEGMENT':'transit_segments', - 'TRANSIT_LINE':'transit_lines', - 'CENTROID':'centroids' - } - - # read-in entire emme network object as a list - get_objs = 'list(emme_network.' + net_attr[emme_network_object] + '())' - uda = eval(get_objs) - - # get list of network object attributes - obj_attr = [] - if fields_to_export[0] in ['all','All','ALL']: - if emme_network_object == 'CENTROID': - obj_attr = emme_network.attributes('NODE') - else: - obj_attr = emme_network.attributes(emme_network_object) - else: - obj_attr = fields_to_export - - # instantiate list of network objects - net_objs = [] - for i in range(len(uda)): - obj_fields = [] - get_id = 'uda[i].id' - obj_fields.append(eval(get_id)) - for attr in obj_attr: - get_field = 'uda[i]["' + attr + '"]' - obj_fields.append(eval(get_field)) - net_objs.append(obj_fields) - net_obj_df = pd.DataFrame(net_objs, columns = ['id'] + obj_attr) - - return(net_obj_df) - - for item, row in self.inputs_list.iterrows(): - - table_name = row['Input_Table'] - emme_network_object = row['Emme_Object'] - column_map = row['Column_Map'] - fields_to_export = row['Fields'].split(',') - - # obtain emme network object, csv or dbf input - if not (pd.isnull(emme_network_object)): - df = get_emme_object(network, emme_network_object, fields_to_export) - self.inputs[table_name] = df - else: - input_path = self.prop_input_paths[table_name] - input_ext = os.path.splitext(input_path)[1] - if input_ext == '.csv': - df = pd.read_csv(_join(self.path, input_path)) - self.inputs[table_name] = df - else: - dbf_path = input_path - if '%project.folder%' in dbf_path: - dbf_path = dbf_path.replace('%project.folder%/', '') - dbf = Dbf5(_join(self.path, dbf_path)) - df = dbf.to_dataframe() - self.inputs[table_name] = df - - # add scenario year - self.inputs['scenario'] = self.scenario_df - - def checks(self): - # read all input DFs into memory - for key, df in self.inputs.items(): - expr = key + ' = df' - exec(expr) - - # copy of locals(), a dictionary of all local variables - calc_dict = locals() - - # read list of checks from CSV file - self.inputs_checks = pd.read_csv(self.inputs_checks_path) - - # remove all commented checks from the checks list - self.inputs_checks = self.inputs_checks.loc[[not i for i in (self.inputs_checks['Test'].str.startswith('#'))]] - - # perform calculations and add user-defined data frame subsets - for item, row in self.inputs_checks.iterrows(): - - test = row['Test'] - table = row['Input_Table'] - id_col = row['Input_ID_Column'] - expr = row['Expression'] - test_vals = row['Test_Vals'] - if not (pd.isnull(row['Test_Vals'])): - test_vals = test_vals.split(',') - test_vals = [txt.strip() for txt in test_vals] - test_type = row['Type'] - Severity = row['Severity'] - stat_expr = row['Report_Statistic'] - - if test_type == 'Calculation': - - try: - calc_expr = test + ' = ' + expr - exec(calc_expr, {}, calc_dict) - calc_out = eval(expr, calc_dict) - except Exception as error: - print('An error occurred with the calculation: {}'.format(test)) - raise - - if str(type(calc_out)) == "": - print('added '+ row['Test'] + ' as new DataFrame input') - self.inputs[row['Test']] = calc_out - self.inputs_list = self.inputs_list.append({'Input_Table': row['Test'],'Property_Token':'NA','Emme_Object':'NA', \ - 'Fields':'NA','Column_Map':'NA','Input_Description':'NA'}, ignore_index = True) - self.inputs_checks = self.inputs_checks.append({'Test':test, 'Input_Table': table, 'Input_ID_Column':id_col, 'Severity':Severity, \ - 'Type':test_type, 'Expression': expr, 'Test_Vals':test_vals, 'Report_Statistic':stat_expr, 'Test_Description': row['Test_Description']}, \ - ignore_index = True) - - # loop through list of checks and conduct all checks - # checks must evaluate to True if inputs are correct - for item, row in self.inputs_checks.iterrows(): - - test = row['Test'] - table = row['Input_Table'] - id_col = row['Input_ID_Column'] - expr = row['Expression'] - test_vals = row['Test_Vals'] - if not (pd.isnull(row['Test_Vals'])): - test_vals = test_vals.split(',') - test_vals = [txt.strip() for txt in test_vals] - test_type = row['Type'] - Severity = row['Severity'] - stat_expr = row['Report_Statistic'] - - if test_type == 'Test': - - if (pd.isnull(row['Test_Vals'])): - - # perform test - try: - out = eval(expr, calc_dict) - except Exception as error: - print('An error occurred with the check: {}'.format(test)) - raise - - # check if test result is a series - if str(type(out)) == "": - # for series, the test must be evaluated across all items - # result is False if a single False is found - self.results[test] = not (False in out.values) - - # reverse results list since we need all False IDs - reverse_results = [not i for i in out.values] - error_expr = table + "['" + id_col + "']" + "[reverse_results]" - error_id_list = eval(error_expr) - - # report first 25 problem IDs in the log - self.problem_ids[test] = error_id_list if error_id_list.size > 0 else [] - - # compute report statistics - if (pd.isnull(stat_expr)): - self.report_stat[test] = '' - else: - stat_list = eval(stat_expr) - self.report_stat[test] = stat_list[reverse_results] - else: - self.results[test] = out - self.problem_ids[test] = [] - if (pd.isnull(stat_expr)): - self.report_stat[test] = '' - else: - self.report_stat[test] = eval(stat_expr) - else: - # loop through test_vals and perform test for each item - self.result_list[test] = [] - for test_val in test_vals: - # perform test (test result must not be of type Series) - try: - out = eval(expr) - except Exception as error: - print('An error occurred with the check: {}'.format(test)) - raise - - # compute report statistic - if (pd.isnull(stat_expr)): - self.report_stat[test] = '' - else: - self.report_stat[test] = eval(stat_expr) - - # append to list - self.result_list[test].append(out) - self.results[test] = not (False in self.result_list[test]) - self.problem_ids[test] = [] - else: - # perform calculation - try: - calc_expr = test + ' = ' + expr - exec(calc_expr, {}, calc_dict) - except Exception as error: - print('An error occurred with the calculation: {}'.format(test)) - raise - - def prop_file_paths(self): - prop_files = self.inputs_list[['Input_Table','Property_Token']].dropna() - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(self.path, 'conf', 'sandag_abm.properties')) - - for item, row in prop_files.iterrows(): - input_table = row['Input_Table'] - input_path = props[row['Property_Token']] - self.prop_input_paths[input_table] = input_path - - # obtain scenario year - self.scenario_df['Year'] = [props['scenarioYear']] - - def write_log(self): - # function to write out the input checker log file - # there are four blocks - # - Introduction - # - Summary of checks - # - Action Required: FATAL, LOGICAL, WARNINGS - # - List of passed checks - - # create log file - now = datetime.datetime.now() - - self.log_path = _join(self.input_checker_path, ('inputCheckerSummary_' + now.strftime("[%Y-%m-%d]") + '.txt')) - f = open(self.log_path, 'wb') - - # define re-usable elements - seperator1 = '###########################################################' - seperator2 = '***********************************************************' - - # write out Header - f.write(seperator1 + seperator1 + "\r\n") - f.write(seperator1 + seperator1 + "\r\n\r\n") - f.write("\t SANDAG ABM Input Checker Summary File \r\n") - f.write("\t _____________________________________ \r\n\r\n\r\n") - f.write("\t Created on: " + now.strftime("%Y-%m-%d %H:%M") + "\r\n\r\n") - f.write("\t Notes:-\r\n") - f.write("\t The SANDAG ABM Input Checker performs various QA/QC checks on SANDAG ABM inputs as specified by the user.\r\n") - f.write("\t The Input Checker allows the user to specify three severity levels for each QA/QC check:\r\n\r\n") - f.write("\t 1) FATAL 2) LOGICAL 3) WARNING\r\n\r\n") - f.write("\t FATAL Checks: The failure of these checks would result in a FATAL errors in the SANDAG ABM run.\r\n") - f.write("\t In case of FATAL failure, the Input Checker returns a return code of 1 to the\r\n") - f.write("\t main SANDAG ABM model, cauing the model run to halt.\r\n") - f.write("\t LOGICAL Checks: The failure of these checks indicate logical inconsistencies in the inputs.\r\n") - f.write("\t With logical errors in inputs, the SANDAG ABM outputs may not be meaningful.\r\n") - f.write("\t WARNING Checks: The failure of Warning checks would indicate problems in data that would not.\r\n") - f.write("\t halt the run or affect model outputs but might indicate an issue with inputs.\r\n\r\n\r\n") - f.write("\t The contents of this summary file are organized as follows: \r\n\r\n") - f.write("\t TALLY OF FAILED CHECKS:\r\n") - f.write("\t -----------------------\r\n") - f.write("\t A tally of all failed checks per severity level\r\n\r\n") - f.write("\t IMMEDIATE ACTION REQUIRED:\r\n") - f.write("\t -------------------------\r\n") - f.write("\t A log under this heading will be generated in case of failure of a FATAL check\r\n\r\n") - f.write("\t ACTION REQUIRED:\r\n") - f.write("\t ---------------\r\n") - f.write("\t A log under this heading will be generated in case of failure of a LOGICAL check\r\n\r\n") - f.write("\t WARNINGS:\r\n") - f.write("\t ---------\r\n") - f.write("\t A log under this heading will be generated in case of failure of a WARNING check\r\n\r\n") - f.write("\t SUMMARY OF ALL PASSED CHECKS:\r\n") - f.write("\t ----------------------------\r\n") - f.write("\t A complete listing of results of all passed checks\r\n\r\n") - f.write(seperator1 + seperator1 + "\r\n") - f.write(seperator1 + seperator1 + "\r\n\r\n\r\n\r\n") - - # combine results, inputs_checks and inputs_list - self.inputs_checks['result'] = self.inputs_checks['Test'].map(self.results) - checks_df = pd.merge(self.inputs_checks, self.inputs_list, on='Input_Table') - checks_df = checks_df[checks_df.Type=='Test'] - checks_df['reverse_result'] = [not i for i in checks_df.result] - - # get count of all FATAL failures - self.num_fatal = checks_df.result[(checks_df.Severity=='Fatal') & (checks_df.reverse_result)].count() - - # get count of all LOGICAL failures - self.num_logical = checks_df.result[(checks_df.Severity=='Logical') & (checks_df.reverse_result)].count() - self.logical_fails = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] - - # get count of all WARNING failures - self.num_warning = checks_df.result[(checks_df.Severity=='Warning') & (checks_df.reverse_result)].count() - - # write summary of failed checks - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "TALLY OF FAILED CHECKS \r\n") - f.write('\t' + "---------------------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n\t") - f.write(' Number of Fatal Errors: ' + str(self.num_fatal)) - f.write('\r\n\t Number of Logical Errors: ' + str(self.num_logical)) - f.write('\r\n\t Number of Warnings: ' + str(self.num_warning)) - - def write_check_log(self, fh, row): - # define constants - seperator2 = '-----------------------------------------------------------' - - # integerize problem ID list - problem_ids = self.problem_ids[row['Test']] - #problem_ids = [int(x) for x in problem_ids] - - # write check summary - fh.write('\r\n\r\n' + seperator2 + seperator2) - fh.write("\r\n\t Input File Name: " + ('NA' if not pd.isnull(row['Emme_Object']) else - (self.prop_input_paths[row['Input_Table']].rsplit('/', 1)[-1]))) - fh.write("\r\n\t Input File Location: " + ('NA' if not pd.isnull(row['Emme_Object']) else - (_join(self.input_checker_path, self.prop_input_paths[row['Input_Table']].replace('/','\\'))))) - fh.write("\r\n\t Emme Object: " + (row['Emme_Object'] if not pd.isnull(row['Emme_Object']) else 'NA')) - fh.write("\r\n\t Input Description: " + (row['Input_Description'] if not pd.isnull(row['Input_Description']) else "")) - fh.write("\r\n\t Test Name: " + row['Test']) - fh.write("\r\n\t Test_Description: " + (row['Test_Description'] if not pd.isnull(row['Test_Description']) else "")) - fh.write("\r\n\t Test Severity: " + row['Severity']) - fh.write("\r\n\r\n\t TEST RESULT: " + ('PASSED' if row['result'] else 'FAILED')) - - # display problem IDs for failed column checks - wrapper = textwrap.TextWrapper(width = 70) - if (not row['result']) & (len(problem_ids)>0) : - fh.write("\r\n\t TEST failed for following values of ID Column: " + row['Input_ID_Column'] + " (only up to 25 IDs displayed)") - fh.write("\r\n\t " + row['Input_ID_Column'] + ": " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, problem_ids[0:25]))))) - if not (pd.isnull(row['Report_Statistic'])): - this_report_stat = self.report_stat[row['Test']] - fh.write("\r\n\t Test Statistics: " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, this_report_stat[0:25]))))) - fh.write("\r\n\t Total number of failures: " + str(len(self.problem_ids[row['Test']]))) - if ((len(self.problem_ids[row['Test']])) > 25) and (row['Severity'] == 'Logical'): - fh.write("\r\n\t Open {} for complete list of failed Logical failures.".format(self.logical_log_path)) - else: - if not (pd.isnull(row['Report_Statistic'])): - fh.write("\r\n\t Test Statistic: " + str(self.report_stat[row['Test']])) - - # display result for each test val if it was specified - if not (pd.isnull(row['Test_Vals'])): - fh.write("\r\n\t TEST results for each test val") - result_tuples = zip(row['Test_Vals'].split(","), self.result_list[row['Test']]) - fh.write("\r\n\t ") - fh.write(','.join('[{} - {}]'.format(x[0],x[1]) for x in result_tuples)) - - fh.write("\r\n" + seperator2 + seperator2 + "\r\n\r\n") - - # write out IMMEDIATE ACTION REQUIRED section if needed - if self.num_fatal > 0: - fatal_checks = checks_df[(checks_df.Severity=='Fatal') & (checks_df.reverse_result)] - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "IMMEDIATE ACTION REQUIRED \r\n") - f.write('\t' + "------------------------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n") - - # write out log for each check - for item, row in fatal_checks.iterrows(): - #self.write_check_log(f, row, self.problem_ids[row['Test']]) - #write_check_log(self, f, row, self.problem_ids[row['Test']]) - write_check_log(self, f, row) - - # write out ACTION REQUIRED section if needed - if self.num_logical > 0: - logical_checks = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "ACTION REQUIRED \r\n") - f.write('\t' + "--------------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n") - - #write out log for each check - for item, row in logical_checks.iterrows(): - write_check_log(self, f, row) - - # write out WARNINGS section if needed - if self.num_warning > 0: - warning_checks = checks_df[(checks_df.Severity=='Warning') & (checks_df.reverse_result)] - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "WARNINGS \r\n") - f.write('\t' + "-------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n") - - # write out log for each check - for item, row in warning_checks.iterrows(): - write_check_log(self, f, row) - - # write out the complete listing of all checks that passed - passed_checks = checks_df[(checks_df.result)] - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "LOG OF ALL PASSED CHECKS \r\n") - f.write('\t' + "------------------------ \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n") - - # write out log for each check - for item, row in passed_checks.iterrows(): - write_check_log(self, f, row) - - f.close() - - def write_logical_log(self): - # function to write out the complete list of Logical failures - - # combine results, inputs_checks and inputs_list - self.inputs_checks['result'] = self.inputs_checks['Test'].map(self.results) - checks_df = pd.merge(self.inputs_checks, self.inputs_list, on='Input_Table') - checks_df = checks_df[checks_df.Type=='Test'] - checks_df['reverse_result'] = [not i for i in checks_df.result] - - # get count of all LOGICAL failures - self.num_logical = checks_df.result[(checks_df.Severity=='Logical') & (checks_df.reverse_result)].count() - self.logical_fails = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] - - log_fail_id_tally = 0 - if self.num_logical > 0: - for item, row in self.logical_fails.iterrows(): - problem_ids = self.problem_ids[row['Test']] - if len(problem_ids) > 0: - log_fail_id_tally += 1 - - if log_fail_id_tally > 0: - - # create log file - now = datetime.datetime.now() - - self.logical_log_path = _join(self.input_checker_path, ('completeLogicalFails_' + now.strftime("[%Y-%m-%d]") + '.txt')) - f = open(self.logical_log_path, 'wb') - - # define re-usable elements - seperator1 = '###########################################################' - seperator2 = '***********************************************************' - - # write out Header - f.write(seperator1 + seperator1 + "\r\n") - f.write(seperator1 + seperator1 + "\r\n\r\n") - f.write("\t SANDAG ABM Input Checker Logical Failures Complete List \r\n") - f.write("\t _______________________________________________________ \r\n\r\n\r\n") - f.write("\t Created on: " + now.strftime("%Y-%m-%d %H:%M") + "\r\n\r\n") - f.write("\t Notes:-\r\n") - f.write("\t The SANDAG ABM Input Checker performs various QA/QC checks on SANDAG ABM inputs as specified by the user.\r\n") - f.write("\t The Input Checker allows the user to specify three severity levels for each QA/QC check:\r\n\r\n") - f.write("\t 1) FATAL 2) LOGICAL 3) WARNING\r\n\r\n") - f.write("\t This file provides the complete list of failed checks for checks of severity type Logical. \r\n") - f.write(seperator1 + seperator1 + "\r\n") - f.write(seperator1 + seperator1 + "\r\n\r\n\r\n\r\n") - - # write total number of failed logical checks - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "TALLY OF FAILED CHECKS \r\n") - f.write('\t' + "---------------------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n\t") - f.write('\r\n\t Number of Logical Errors: ' + str(self.num_logical)) - - def write_logical_check_log(self, fh, row): - # define constants - seperator2 = '-----------------------------------------------------------' - - # integerize problem ID list - problem_ids = self.problem_ids[row['Test']] - #problem_ids = [int(x) for x in problem_ids] - - # write check summary - fh.write('\r\n\r\n' + seperator2 + seperator2) - fh.write("\r\n\t Input File Name: " + ('NA' if not pd.isnull(row['Emme_Object']) else - (self.prop_input_paths[row['Input_Table']].rsplit('/', 1)[-1]))) - fh.write("\r\n\t Input File Location: " + ('NA' if not pd.isnull(row['Emme_Object']) else - (_join(self.input_checker_path, self.prop_input_paths[row['Input_Table']].replace('/','\\'))))) - fh.write("\r\n\t Emme Object: " + (row['Emme_Object'] if not pd.isnull(row['Emme_Object']) else 'NA')) - fh.write("\r\n\t Input Description: " + (row['Input_Description'] if not pd.isnull(row['Input_Description']) else "")) - fh.write("\r\n\t Test Name: " + row['Test']) - fh.write("\r\n\t Test_Description: " + (row['Test_Description'] if not pd.isnull(row['Test_Description']) else "")) - fh.write("\r\n\t Test Severity: " + row['Severity']) - fh.write("\r\n\r\n\t TEST RESULT: " + ('PASSED' if row['result'] else 'FAILED')) - - # display problem IDs for failed column checks - wrapper = textwrap.TextWrapper(width = 70) - if (not row['result']) & (len(problem_ids)>0) : - fh.write("\r\n\t TEST failed for following values of ID Column: " + row['Input_ID_Column']) - fh.write("\r\n\t " + row['Input_ID_Column'] + ": " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, problem_ids))))) - if not (pd.isnull(row['Report_Statistic'])): - this_report_stat = self.report_stat[row['Test']] - fh.write("\r\n\t Test Statistics: " + "\r\n\t " + "\r\n\t ".join(wrapper.wrap(text = ", ".join(map(str, this_report_stat))))) - fh.write("\r\n\t Total number of failures: " + str(len(self.problem_ids[row['Test']]))) - else: - if not (pd.isnull(row['Report_Statistic'])): - fh.write("\r\n\t Test Statistic: " + str(self.report_stat[row['Test']])) - - # display result for each test val if it was specified - if not (pd.isnull(row['Test_Vals'])): - fh.write("\r\n\t TEST results for each test val") - result_tuples = zip(row['Test_Vals'].split(","), self.result_list[row['Test']]) - fh.write("\r\n\t ") - fh.write(','.join('[{} - {}]'.format(x[0],x[1]) for x in result_tuples)) - - fh.write("\r\n" + seperator2 + seperator2 + "\r\n\r\n") - - # write out ACTION REQUIRED section if needed - if self.num_logical > 0: - logical_checks = checks_df[(checks_df.Severity=='Logical') & (checks_df.reverse_result)] - f.write('\r\n\r\n' + seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n\r\n") - f.write('\t' + "LOG OF ALL FAILED LOGICAL CHECKS \r\n") - f.write('\t' + "-------------------------------- \r\n\r\n") - f.write(seperator2 + seperator2 + "\r\n") - f.write(seperator2 + seperator2 + "\r\n") - - #write out log for each check - for item, row in logical_checks.iterrows(): - if len(self.problem_ids[row['Test']]) > 25: - write_logical_check_log(self, f, row) - - f.close() - - def check_logical(self): - if self.num_logical > 0: - # raise exception for each logical check fail - for item, row in self.logical_fails.iterrows(): - answer = dialog.alert_question( - message = "The following Logical check resulted in at least 1 error: {} \n Open {} for details. \ - \n\n Click OK to continue or Cancel to stop run.".format(row['Test'], self.log_path), - title = "Logical Check Error", - answers = [("OK", dialog.YES_ROLE), ("Cancel", dialog.REJECT_ROLE)] - ) - - if answer == 1: - raise Exception("Input checker was cancelled") - - def check_num_fatal(self): - # return code to the main model based on input checks and results - if self.num_fatal > 0: - raise Exception("Input checker failed, {} fatal errors found. Open {} for details.".format(self.num_fatal, self.log_path)) \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/import/run4Ds.py b/sandag_abm/src/main/emme/toolbox/import/run4Ds.py deleted file mode 100644 index 89ff0d0..0000000 --- a/sandag_abm/src/main/emme/toolbox/import/run4Ds.py +++ /dev/null @@ -1,412 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright RSG, 2019-2020. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// import/run4Ds.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Generates density variables and adds in mgra socio economic variables -# -# -# Inputs: -# path: path to the current scenario -# ref_path: path to the comparison model scenario -# int_radius: buffer radius for intersection counts -# maps: default unchecked - means not generating spatial heat maps for -# intersection counts. This functionality requires -# following packages; geopandas, folium, and branca -# -# File referenced: -# input\mgra13_based_input2016.csv -# input\SANDAG_Bike_Net.dbf -# input\SANDAG_Bike_Node.dbf -# output\walkMgraEquivMinutes.csv -# -# Script example -# python C:\ABM_runs\maint_2019_RSG\Tasks\4ds\emme_toolbox\emme\toolbox\import\run4Ds.py -# 0.65 r'C:\ABM_runs\maint_2019_RSG\Model\ABM2_14_2_0' r'C:\ABM_runs\maint_2019_RSG\Model\abm_test_fortran_4d' - - -TOOLBOX_ORDER = 10 - -#import modules -import inro.modeller as _m -from simpledbf import Dbf5 -import os -import pandas as pd, numpy as np -#import datetime -import matplotlib.pyplot as plt -import seaborn as sns -import warnings -import traceback as _traceback - -warnings.filterwarnings("ignore") - -_join = os.path.join -_dir = os.path.dirname - -gen_utils = _m.Modeller().module("sandag.utilities.general") - -class FourDs(_m.Tool()): - - path = _m.Attribute(unicode) - ref_path = _m.Attribute(unicode) - int_radius = _m.Attribute(float) - maps = _m.Attribute(bool) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self._log = [] - self._error = [] - project_dir = _dir(_m.Modeller().desktop.project.path) - self.path = _dir(project_dir) - self.mgradata_file = '' - self.equivmins_file = '' - self.inNet = '' - self.inNode = '' - self.ref_path = '' - self.maps = False - self.int_radius = 0.65 #mile - self.oth_radius = self.int_radius #same as intersection radius - self.new_cols = ['totint','duden','empden','popden','retempden','totintbin','empdenbin','dudenbin','PopEmpDenPerMi'] - self.continuous_fields = ['totint', 'popden', 'empden', 'retempden'] - self.discrete_fields = ['totintbin', 'empdenbin', 'dudenbin'] - self.mgra_shape_file = '' - self.base = pd.DataFrame() - self.build = pd.DataFrame() - self.mgra_data = pd.DataFrame() - self.base_cols = [] - self.attributes = ["path", "int_radius", "ref_path"] - - def page(self): - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(self.path, "conf", "sandag_abm.properties")) - self.ref_path = props["visualizer.reference.path"] - - pb = _m.ToolPageBuilder(self) - pb.title = "Run 4Ds" - pb.description = """ - Generate Density Variables. - Generated from MGRA socio economic file and active transportation (AT) network. -
    -
    - The following files are used: -
    -
      -
    • input\mgra13_based_input2016.csv
    • -
    • input\SANDAG_Bike_Net.dbf
    • -
    • input\SANDAG_Bike_Node.dbf
    • -
    • output\walkMgraEquivMinutes.csv
    • -
    -
    - """ - pb.branding_text = "SANDAG - Run 4Ds" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file("path", window_type="directory", file_filter="", - title="Source directory:",) - - pb.add_text_box("int_radius", size=6, title="Buffer size (miles):") - #pb.add_checkbox("maps", title=" ", label="Generate 4D maps") - pb.add_select_file("ref_path", window_type="directory", file_filter="", title="Reference directory for comparison") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self(path=self.path, int_radius=self.int_radius, ref_path=self.ref_path) - run_msg = "Run 4Ds complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, path= "", - int_radius = 0.65, - ref_path = ""): - _m.logbook_write("Started running 4Ds ...") - - self.path = path - self.ref_path = ref_path - self.int_radius = int_radius - #self.maps = maps - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(self.path, "conf", "sandag_abm.properties")) - - - self.mgradata_file = props["mgra.socec.file"] #input/filename - self.syn_households_file = props["PopulationSynthesizer.InputToCTRAMP.HouseholdFile"] #input/filename - self.equivmins_file = props["active.logsum.matrix.file.walk.mgra"] #filename - self.inNet = os.path.basename(props["active.edge.file"]) #filename - self.inNode = os.path.basename(props["active.node.file"]) #filename - - attributes = { - "path": self.path, - "ref_path": self.ref_path, - "int_radius": self.int_radius, - "maps": self.maps, - } - gen_utils.log_snapshot("Run 4Ds", str(self), attributes) - - file_paths = [_join(self.path, self.mgradata_file),_join(self.path, self.syn_households_file),_join(self.path, "output", self.equivmins_file), _join(self.path, "input", self.inNet), _join(self.path, "input", self.inNode)] - for path in file_paths: - if not os.path.exists(path): - raise Exception("missing file '%s'" % (path)) - - self.mgra_data = pd.read_csv(os.path.join(self.path,self.mgradata_file)) - self.base_cols = self.mgra_data.columns.tolist() - - _m.logbook_write("Tagging intersections to mgra") - self.get_intersection_count() - - _m.logbook_write("Generating density variables") - self.get_density() - - _m.logbook_write("Creating comparison plots") - self.make_plots() - - _m.logbook_write("Finished running 4Ds") - - def get_intersection_count(self): - links = Dbf5(_join(self.path, "input", self.inNet)) - links = links.to_dataframe() - - nodes = Dbf5(_join(self.path, "input", self.inNode)) - nodes = nodes.to_dataframe() - - nodes_int = nodes.loc[(nodes.NodeLev_ID < 100000000)] - - #links - #remove taz, mgra, and tap connectors - links = links.loc[(links.A <100000000) & (links.B <100000000)] - - #remove freeways (Func_Class=1), ramps (Func_Class=2), and others (Func_Class =0 or -1) - links = links.loc[(links.Func_Class > 2)] - links['link_count'] = 1 - - #aggregate by Node A and Node B - links_nodeA = links[['A', 'link_count']].groupby('A').sum().reset_index() - links_nodeB = links[['B', 'link_count']].groupby('B').sum().reset_index() - - #merge the two and keep all records from both dataframes (how='outer') - nodes_linkcount = pd.merge(links_nodeA, links_nodeB, left_on='A', right_on='B', how = 'outer') - nodes_linkcount = nodes_linkcount.fillna(0) - nodes_linkcount['link_count'] = nodes_linkcount['link_count_x'] + nodes_linkcount['link_count_y'] - - #get node id from both dataframes - nodes_linkcount['N']=0 - nodes_linkcount['N'][nodes_linkcount.A>0] = nodes_linkcount['A'] - nodes_linkcount['N'][nodes_linkcount.B>0] = nodes_linkcount['B'] - nodes_linkcount['N']=nodes_linkcount['N'].astype(float) - nodes_linkcount = nodes_linkcount[['N','link_count']] - - #keep nodes with 3+ link count - intersections_temp = nodes_linkcount.loc[nodes_linkcount.link_count>=3] - - #get node X and Y - intersections = pd.merge(intersections_temp,nodes_int[['NodeLev_ID','XCOORD','YCOORD']], left_on = 'N', right_on = 'NodeLev_ID', how = 'left') - intersections = intersections[['N','XCOORD','YCOORD']] - intersections = intersections.rename(columns = {'XCOORD': 'X', 'YCOORD': 'Y'}) - - mgra_nodes = nodes[nodes.MGRA > 0][['MGRA','XCOORD','YCOORD']] - mgra_nodes.columns = ['mgra','x','y'] - int_dict = {} - for int in intersections.iterrows(): - mgra_nodes['dist'] = np.sqrt((int[1][1] - mgra_nodes['x'])**2+(int[1][2] - mgra_nodes['y'])**2) - int_dict[int[1][0]] = mgra_nodes.loc[mgra_nodes['dist'] == mgra_nodes['dist'].min()]['mgra'].values[0] - - intersections['near_mgra'] = intersections['N'].map(int_dict) - intersections = intersections.groupby('near_mgra', as_index = False).count()[['near_mgra','N']].rename(columns = {'near_mgra':'mgra','N':'icnt'}) - try: - self.mgra_data = self.mgra_data.drop('icnt',axis = 1).merge(intersections, how = 'outer', on = "mgra") - except: - self.mgra_data = self.mgra_data.merge(intersections, how = 'outer', on = "mgra") - - def get_density(self): - if len(self.mgra_data) == 0: - mgra_landuse = pd.read_csv(os.path.join(self.path, self.mgradata_file)) - else: - mgra_landuse = self.mgra_data - - # get population from synthetic population instead of mgra data file - syn_pop = pd.read_csv(os.path.join(self.path, self.syn_households_file)) - syn_pop = syn_pop.rename(columns = {'MGRA':'mgra'})[['persons','mgra']].groupby('mgra',as_index = False).sum() - #remove if 4D columns exist - for col in self.new_cols: - if col in self.base_cols: - self.base_cols.remove(col) - mgra_landuse = mgra_landuse.drop(col,axis=1) - - #merge syntetic population to landuse - mgra_landuse = mgra_landuse.merge(syn_pop, how = 'left', on = 'mgra') - #all street distance - equiv_min = pd.read_csv(_join(self.path, "output", self.equivmins_file)) - equiv_min['dist'] = equiv_min['actual']/60*3 - print("MGRA input landuse: " + self.mgradata_file) - - def density_function(mgra_in): - eqmn = equiv_min[equiv_min['i'] == mgra_in] - mgra_circa_int = eqmn[eqmn['dist'] < self.int_radius]['j'].unique() - mgra_circa_oth = eqmn[eqmn['dist'] < self.oth_radius]['j'].unique() - totEmp = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_total'].sum() - totRet = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_retail'].sum() + mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_personal_svcs_retail'].sum() + mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['emp_restaurant_bar'].sum() - totHH = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['hh'].sum() - totPop = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['persons'].sum() - totAcres = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_oth)]['land_acres'].sum() - totInt = mgra_landuse[mgra_landuse.mgra.isin(mgra_circa_int)]['icnt'].sum() - if(totAcres>0): - empDen = totEmp/totAcres - retDen = totRet/totAcres - duDen = totHH/totAcres - popDen = totPop/totAcres - popEmpDenPerMi = (totEmp+totPop)/(totAcres/640) #Acres to miles - tot_icnt = totInt - else: - empDen = 0 - retDen = 0 - duDen = 0 - popDen = 0 - popEmpDenPerMi = 0 - tot_icnt = 0 - return tot_icnt,duDen,empDen,popDen,retDen,popEmpDenPerMi - - #new_cols = [0-'totint',1-'duden',2-'empden',3-'popden',4-'retempden',5-'totintbin',6-'empdenbin',7-'dudenbin',8-'PopEmpDenPerMi'] - mgra_landuse[self.new_cols[0]],mgra_landuse[self.new_cols[1]],mgra_landuse[self.new_cols[2]],mgra_landuse[self.new_cols[3]],mgra_landuse[self.new_cols[4]],mgra_landuse[self.new_cols[8]] = zip(*mgra_landuse['mgra'].map(density_function)) - - mgra_landuse = mgra_landuse.fillna(0) - mgra_landuse[self.new_cols[5]] = np.where(mgra_landuse[self.new_cols[0]] < 80, 1, np.where(mgra_landuse[self.new_cols[0]] < 130, 2, 3)) - mgra_landuse[self.new_cols[6]] = np.where(mgra_landuse[self.new_cols[2]] < 10, 1, np.where(mgra_landuse[self.new_cols[2]] < 30, 2,3)) - mgra_landuse[self.new_cols[7]] = np.where(mgra_landuse[self.new_cols[1]] < 5, 1, np.where(mgra_landuse[self.new_cols[1]] < 10, 2,3)) - - mgra_landuse[self.base_cols+self.new_cols].to_csv(os.path.join(self.path, self.mgradata_file), index = False, float_format='%.4f' ) - - self.mgra_data = mgra_landuse - print( "*** Finished ***") - - #plot comparisons of build and old density values and create heat maps - def make_plots(self): - if len(self.mgra_data) == 0: - self.build = pd.read_csv(os.path.join(self.path, self.mgradata_file)) - else: - self.build = self.mgra_data - - def plot_continuous(field): - #colors - rsg_orange = '#f68b1f' - rsg_marine = '#006fa1' - #rsg_leaf = '#63af5e' - #rsg_grey = '#48484a' - #rsg_mist = '#dcddde' - - max = self.base[field].max() + self.base[field].max()%5 - div = max/5 if max/5 >= 10 else max/2 - bins = np.linspace(0,max,div) - plt.hist(self.base[field], bins, normed = True, alpha = 0.5, label = 'Base', color = rsg_marine) - plt.hist(self.build[field], bins, normed = True, alpha = 0.5, label = 'Build', color = rsg_orange) - mean_base = self.base[field].mean() - mean = self.build[field].mean() - median_base = self.base[field].median() - median = self.build[field].median() - plt.axvline(mean_base, color = 'b', linestyle = '-', label = 'Base Mean') - plt.axvline(median_base, color = 'b', linestyle = '--', label = 'Base Median') - plt.axvline(mean, color = 'r', linestyle = '-', label = 'Build Mean') - plt.axvline(median, color = 'r', linestyle = '--',label = 'Build Median') - plt.legend(loc = 'upper right') - ylims = plt.ylim()[1] - plt.text(mean_base + div/4, ylims-ylims/32, "mean: {:0.2f}".format(mean_base), color = 'b') - plt.text(mean_base + div/4, ylims - 5*ylims/32, "median: {:0.0f}".format(median_base), color = 'b') - plt.text(mean_base + div/4, ylims-2*ylims/32, "mean: {:0.2f}".format(mean), size = 'medium',color = 'r') - plt.text(mean_base + div/4, ylims-6*ylims/32, "median: {:0.0f}".format(median), color = 'r') - plt.text(self.base[field].min() , ylims/32, "min: {:0.0f}".format(self.base[field].min()), color = 'b') - plt.text(self.base[field].max()-div , ylims/32, "max: {:0.0f}".format(self.base[field].max()), color = 'b') - plt.text(self.build[field].min() , 2*ylims/32, "min: {:0.0f}".format(self.build[field].min()), color = 'r') - plt.text(self.base[field].max()-div , 2*ylims/32, "max: {:0.0f}".format(self.build[field].max()), color = 'r') - - plt.xlabel(field) - plt.ylabel("MGRA's") - plt.title(field.replace('den','') + ' Density') - outfile = _join(self.path, "output", '4Ds_{}_plot.png'.format(field)) - if os.path.isfile(outfile): - os.remove(outfile) - plt.savefig(outfile) - plt.clf() - - def plot_discrete(field): - fig, ax = plt.subplots() - df1 = discretedf_base.groupby(field, as_index = False).agg({'mgra':'count','type':'first'}) - df2 = discretedf_build.groupby(field, as_index = False).agg({'mgra':'count','type':'first'}) - df = df1.append(df2) - ax = sns.barplot(x=field, y = 'mgra', hue = 'type', data = df) - ax.set_title(field) - outfile = _join(self.path, "output", '4Ds_{}_plot.png'.format(field)) - if os.path.isfile(outfile): - os.remove(outfile) - ax.get_figure().savefig(outfile) - - self.base = pd.read_csv(self.ref_path) - self.base['type'] = 'base' - self.build['type'] = 'build' - - discretedf_base = self.base[['mgra','type']+self.discrete_fields] - discretedf_build = self.build[['mgra','type']+self.discrete_fields] - - for f in self.continuous_fields: - plot_continuous(f) - for f in self.discrete_fields: - plot_discrete(f) - - if self.maps: - import geopandas as gpd - import folium - from branca.colormap import linear - compare_int = self.base.merge(self.build, how = 'outer', on = 'mgra', suffixes = ['_base','_build']) - compare_int['diff'] = compare_int['TotInt'] - compare_int['totint'] - - compare_int = gpd.read_file(self.mgra_shape_file).rename(columns = {'MGRA':'mgra'}).merge(compare_int, how = 'left', on = 'mgra') - compare_int = compare_int.to_crs({'init': 'epsg:4326'}) - - colormap = linear.OrRd_09.scale( - compare_int.TotInt.min(), - compare_int.TotInt.max()) - colormapA = linear.RdBu_04.scale( - compare_int['diff'].min(), - compare_int['diff'].min()*-1) - - compare_int['colordiff'] = compare_int['diff'].map(lambda n: colormapA(n)) - compare_int['colororig'] = compare_int['TotInt'].map(lambda n: colormap(n)) - compare_int['colornew'] = compare_int['totint'].map(lambda n: colormap(n)) - - def makeheatmap(self,df, colormp,color_field,caption): - mapname = folium.Map(location=[32.76, -117.15], zoom_start = 13.459) - folium.GeoJson(compare_int, - style_function=lambda feature: { - 'fillColor': feature['properties'][color_field], - 'color' : rsg_marine, - 'weight' : 0, - 'fillOpacity' : 0.75, - }).add_to(mapname) - - colormp.caption = caption - colormp.add_to(mapname) - return mapname - - makeheatmap(compare_int,colormapA,'colordiff','Intersection Diff (base - build)').save('diff_intersections.html') - makeheatmap(compare_int,colormap,'colororig','Intersections').save('base_intersections.html') - makeheatmap(compare_int,colormap,'colororig','Intersections').save('build_intersections.html') diff --git a/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py b/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py deleted file mode 100644 index f4e880a..0000000 --- a/sandag_abm/src/main/emme/toolbox/initialize/initialize_matrices.py +++ /dev/null @@ -1,426 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// initialize_matrices.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Coordinates the initialization of all matrices. -# The matrix names are listed for each of the model components / steps, -# and the matrix IDs are assigned consistently from the set of matrices. -# In each of the model steps the matrices are only referenced by name, -# never by ID. -# -# -# Inputs: -# components: A list of the model components / steps for which to initialize matrices -# One or more of "traffic_demand", "transit_demand", -# "traffic_skims", "transit_skims", "external_internal_model", -# "external_external_model", "truck_model", "commercial_vehicle_model" -# periods: A list of periods for which to initialize matrices, "EA", "AM", "MD", "PM", "EV" -# scenario: scenario to use for reference zone system and the emmebank in which -# the matrices will be created -# -# Script example: -""" - import os - import inro.emme.database.emmebank as _eb - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - transit_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - periods = ["EA", "AM", "MD", "PM", "EV"] - traffic_components = [ - "traffic_demand", "traffic_skims", "external_internal_model", - "external_external_model", "truck_model", "commercial_vehicle_model"] - transit_components = ["transit_demand", "transit_skims"] - base_scenario = main_emmebank.scenario(100) - transit_scenario = transit_emmebank.scenario(100) - initialize_matrices = modeller.tool("sandag.initialize.initialize_matrices") - # Create / initialize matrices in the base, traffic emmebank - initialize_matrices(traffic_components, periods, base_scenario) - # Create / initialize matrices in the transit emmebank - initialize_matrices(transit_components, periods, transit_scenario) -""" - - -TOOLBOX_ORDER = 9 - - -import inro.modeller as _m -import traceback as _traceback - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class Initialize(_m.Tool(), gen_utils.Snapshot): - - components = _m.Attribute(_m.ListType) - periods = _m.Attribute(_m.ListType) - delete_all_existing = _m.Attribute(bool) - - tool_run_msg = "" - - def __init__(self): - self._all_components = [ - "traffic_demand", - "transit_demand", - "traffic_skims", - "transit_skims", - "external_internal_model", - "external_external_model", - "truck_model", - "commercial_vehicle_model", - ] - self._all_periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - self.components = self._all_components[:] - self.periods = self._all_periods[:] - self.attributes = ["components", "periods", "delete_all_existing"] - self._matrices = {} - self._count = {} - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Initialize matrices" - pb.description = """Creates and initializes the required matrices - for the selected components / sub-models. - Includes all components by default.""" - pb.branding_text = "- SANDAG" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select("components", keyvalues=[(k,k) for k in self._all_components], - title="Select components:") - pb.add_select("periods", keyvalues=[(k,k) for k in self._all_periods], - title="Select periods:") - pb.add_checkbox("delete_all_existing", label="Delete all existing matrices") - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.components, self.periods, scenario, self.delete_all_existing) - run_msg = "Tool completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace("Create and initialize matrices", save_arguments=True) - def __call__(self, components, periods, scenario, delete_all_existing=False): - attributes = { - "components": components, - "periods": periods, - "delete_all_existing": delete_all_existing - } - gen_utils.log_snapshot("Initialize matrices", str(self), attributes) - - self.scenario = scenario - emmebank = scenario.emmebank - self._create_matrix_tool = _m.Modeller().tool( - "inro.emme.data.matrix.create_matrix") - if components == "all": - components = self._all_components[:] - if periods == "all": - periods = self._all_periods[:] - if delete_all_existing: - with _m.logbook_trace("Delete all existing matrices"): - for matrix in emmebank.matrices(): - emmebank.delete_matrix(matrix) - self.generate_matrix_list(self.scenario) - matrices = [] - for component in components: - matrices.extend(self.create_matrices(component, periods)) - # Note: matrix is also created in import_network - self._create_matrix_tool("ms1", "zero", "zero", scenario=self.scenario, overwrite=True) - return matrices - - def generate_matrix_list(self, scenario): - self._matrices = dict( - (name, dict((k, []) for k in self._all_periods + ["ALL"])) - for name in self._all_components) - self._count = {"ms": 2, "md": 100, "mo": 100, "mf": 100} - - for component in self._all_components: - fcn = getattr(self, component) - fcn() - # check dimensions can fit full set of matrices - type_names = [ - ('mf', 'full_matrices'), - ('mo', 'origin_matrices'), - ('md', 'destination_matrices'), - ('ms', 'scalar_matrices')] - dims = scenario.emmebank.dimensions - for prefix, name in type_names: - if self._count[prefix] > dims[name]: - raise Exception("emmebank capacity error, increase %s to at least %s" % (name, self._count[prefix])) - - def traffic_demand(self): - tmplt_matrices = [ - ("SOV_NT_L", "SOV non-transponder demand low VOT"), - ("SOV_TR_L", "SOV transponder demand low VOT"), - ("HOV2_L", "HOV2 demand low VOT"), - ("HOV3_L", "HOV3+ demand low VOT"), - ("SOV_NT_M", "SOV non-transponder demand medium VOT"), - ("SOV_TR_M", "SOV transponder demand medium VOT"), - ("HOV2_M", "HOV2 demand medium VOT"), - ("HOV3_M", "HOV3+ demand medium VOT"), - ("SOV_NT_H", "SOV non-transponder demand high VOT"), - ("SOV_TR_H", "SOV transponder demand high VOT"), - ("HOV2_H", "HOV2 demand high VOT"), - ("HOV3_H", "HOV3+ demand high VOT"), - ("TRK_H", "Truck Heavy PCE demand"), - ("TRK_L", "Truck Light PCE demand"), - ("TRK_M", "Truck Medium PCE demand"), - ] - for period in self._all_periods: - self.add_matrices("traffic_demand", period, - [("mf", period + "_" + name, period + " " + desc) - for name, desc in tmplt_matrices]) - - def transit_demand(self): - tmplt_matrices = [ - ("BUS", "local bus demand"), - ("PREM", "Premium modes demand"), - ("ALLPEN", "all modes xfer pen demand"), - ] - for period in self._all_periods: - for a_name in ["WLK", "PNR", "KNR"]: - self.add_matrices("transit_demand", period, - [("mf", "%s_%s%s" % (period, a_name, name), "%s %s access %s" % (period, a_name, desc)) - for name, desc in tmplt_matrices]) - - def traffic_skims(self): - tp_desc = {"TR": "transponder", "NT": "non-transponder"} - vot_desc = {"L": "low", "M": "medium", "H": "high"} - truck_desc = {"L": "light", "M": "medium", "H": "heavy"} - - sov_tmplt_matrices = [ - ("TIME", "SOV %s travel time"), - ("DIST", "SOV %s distance"), - ("REL", "SOV %s reliability skim"), - ("TOLLCOST", "SOV %s toll cost $0.01"), - ("TOLLDIST", "SOV %s distance on toll facility"), - ] - hov_tmplt_matrices = [ - ("TIME", "HOV%s travel time"), - ("DIST", "HOV%s distance"), - ("REL", "HOV%s reliability skim"), - ("TOLLCOST", "HOV%s toll cost $0.01"), - ("TOLLDIST", "HOV%s distance on toll facility"), - ("HOVDIST", "HOV%s HOV distance on HOV facility") - ] - truck_tmplt_matrices = [ - ("TIME", "Truck %s travel time"), - ("DIST", "Truck %s distance"), - ("TOLLCOST", "Truck %s toll cost $0.01") - ] - for period in self._all_periods: - for vot_type in "L", "M", "H": - for tp_type in "NT", "TR": - cls_name = "SOV_" + tp_type + "_" + vot_type - cls_desc = tp_desc[tp_type] + " " + vot_desc[vot_type] + " VOT" - self.add_matrices("traffic_skims", period, - [("mf", period + "_" + cls_name + "_" + name, period + " " + desc % cls_desc) for name, desc in sov_tmplt_matrices]) - for hov_type in "2", "3": - cls_name = "HOV" + hov_type + "_" + vot_type - cls_desc = hov_type + " " + vot_desc[vot_type] + " VOT" - self.add_matrices("traffic_skims", period, - [("mf", period + "_" + cls_name + "_" + name, - period + " " + desc % cls_desc) - for name, desc in hov_tmplt_matrices]) - for truck_type in "L", "M", "H": - cls_name = "TRK" + "_" + truck_type - cls_desc = truck_desc[truck_type] - self.add_matrices("traffic_skims", period, - [("mf", period + "_" + cls_name + "_" + name, - period + " " + desc % cls_desc) - for name, desc in truck_tmplt_matrices]) - - self.add_matrices("traffic_skims", "MD", - [("mf", "MD_TRK_TIME", "MD Truck generic travel time")]) - - def transit_skims(self): - tmplt_matrices = [ - ("GENCOST", "total impedance"), - ("FIRSTWAIT", "first wait time"), - ("XFERWAIT", "transfer wait time"), - ("TOTALWAIT", "total wait time"), - ("FARE", "fare"), - ("XFERS", "num transfers"), - ("ACCWALK", "access walk time"), - ("XFERWALK", "transfer walk time"), - ("EGRWALK", "egress walk time"), - ("TOTALWALK", "total walk time"), - ("TOTALIVTT", "in-vehicle time"), - ("DWELLTIME", "dwell time"), - ("BUSIVTT", "local bus in-vehicle time"), - ("LRTIVTT", "LRT in-vehicle time"), - ("CMRIVTT", "Rail in-vehicle time"), - ("EXPIVTT", "Express in-vehicle time"), - ("LTDEXPIVTT", "Ltd exp bus in-vehicle time"), - ("BRTREDIVTT", "BRT red in-vehicle time"), - ("BRTYELIVTT", "BRT yellow in-vehicle time"), - ("TIER1IVTT", "Tier1 in-vehicle time"), - ("BUSDIST", "Bus IV distance"), - ("LRTDIST", "LRT IV distance"), - ("CMRDIST", "Rail IV distance"), - ("EXPDIST", "Express and Ltd IV distance"), - ("BRTDIST", "BRT red and yel IV distance"), - ("TIER1DIST", "Tier1 distance"), - ("TOTDIST", "Total transit distance") - ] - skim_sets = [ - ("BUS", "Local bus only"), - ("PREM", "Premium modes only"), - ("ALLPEN", "All w/ xfer pen") - ] - for period in self._all_periods: - for set_name, set_desc in skim_sets: - self.add_matrices("transit_skims", period, - [("mf", period + "_" + set_name + "_" + name, - period + " " + set_desc + ": " + desc) - for name, desc in tmplt_matrices]) - - def truck_model(self): - tmplt_matrices = [ - ("TRKL", "Truck Light"), - ("TRKM", "Truck Medium"), - ("TRKH", "Truck Heavy"), - ("TRKEI", "Truck external-internal"), - ("TRKIE", "Truck internal-external"), - ] - self.add_matrices("truck_model", "ALL", - [("mo", name + '_PROD', desc + ' production') - for name, desc in tmplt_matrices]) - self.add_matrices("truck_model", "ALL", - [("md", name + '_ATTR', desc + ' attraction') - for name, desc in tmplt_matrices]) - - tmplt_matrices = [ - ("TRKEE_DEMAND", "Truck total external-external demand"), - ("TRKL_FRICTION", "Truck Light friction factors"), - ("TRKM_FRICTION", "Truck Medium friction factors"), - ("TRKH_FRICTION", "Truck Heavy friction factors"), - ("TRKIE_FRICTION", "Truck internal-external friction factors"), - ("TRKEI_FRICTION", "Truck external-internal friction factors"), - ("TRKL_DEMAND", "Truck Light total demand"), - ("TRKM_DEMAND", "Truck Medium total demand"), - ("TRKH_DEMAND", "Truck Heavy total demand"), - ("TRKIE_DEMAND", "Truck internal-external total demand"), - ("TRKEI_DEMAND", "Truck external-internal total demand"), - ] - self.add_matrices("truck_model", "ALL", - [("mf", name, desc) for name, desc in tmplt_matrices]) - - # TODO: remove GP and TOLL matrices, no longer used - tmplt_matrices = [ - ("TRK_L_VEH", "Truck Light demand"), - ("TRKLGP_VEH", "Truck Light GP-only vehicle demand"), - ("TRKLTOLL_VEH", "Truck Light toll vehicle demand"), - ("TRK_M_VEH", "Truck Medium demand"), - ("TRKMGP_VEH", "Truck Medium GP-only vehicle demand"), - ("TRKMTOLL_VEH", "Truck Medium toll vehicle demand"), - ("TRK_H_VEH", "Truck Heavy demand"), - ("TRKHGP_VEH", "Truck Heavy GP-only vehicle demand"), - ("TRKHTOLL_VEH", "Truck Heavy toll vehicle demand"), - ] - for period in self._all_periods: - self.add_matrices("truck_model", period, - [("mf", period + "_" + name, period + " " + desc) - for name, desc in tmplt_matrices]) - - def commercial_vehicle_model(self): - # TODO : remove commercial vehicle matrices, no longer used - tmplt_matrices = [ - ('mo', 'COMVEH_PROD', 'Commercial vehicle production'), - ('md', 'COMVEH_ATTR', 'Commercial vehicle attraction'), - ('mf', 'COMVEH_BLENDED_SKIM', 'Commercial vehicle blended skim'), - ('mf', 'COMVEH_FRICTION', 'Commercial vehicle friction factors'), - ('mf', 'COMVEH_TOTAL_DEMAND', 'Commercial vehicle total demand all periods'), - ] - self.add_matrices("commercial_vehicle_model", "ALL", - [(ident, name, desc) for ident, name, desc in tmplt_matrices]) - - tmplt_matrices = [ - ('COMVEH', 'Commerical vehicle total demand'), - ('COMVEHGP', 'Commerical vehicle GP demand'), - ('COMVEHTOLL', 'Commerical vehicle Toll demand'), - ] - for period in self._all_periods: - self.add_matrices("commercial_vehicle_model", period, - [("mf", period + "_" + name, period + " " + desc) - for name, desc in tmplt_matrices]) - - def external_internal_model(self): - tmplt_matrices = [ - ('SOVTOLL_EIWORK', 'US to SD SOV Work TOLL demand'), - ('HOV2TOLL_EIWORK', 'US to SD HOV2 Work TOLL demand'), - ('HOV3TOLL_EIWORK', 'US to SD HOV3 Work TOLL demand'), - ('SOVGP_EIWORK', 'US to SD SOV Work GP demand'), - ('HOV2HOV_EIWORK', 'US to SD HOV2 Work HOV demand'), - ('HOV3HOV_EIWORK', 'US to SD HOV3 Work HOV demand'), - ('SOVTOLL_EINONWORK', 'US to SD SOV Non-Work TOLL demand'), - ('HOV2TOLL_EINONWORK', 'US to SD HOV2 Non-Work TOLL demand'), - ('HOV3TOLL_EINONWORK', 'US to SD HOV3 Non-Work TOLL demand'), - ('SOVGP_EINONWORK', 'US to SD SOV Non-Work GP demand'), - ('HOV2HOV_EINONWORK', 'US to SD HOV2 Non-Work HOV demand'), - ('HOV3HOV_EINONWORK', 'US to SD HOV3 Non-Work HOV demand'), - ] - for period in self._all_periods: - self.add_matrices("external_internal_model", period, - [("mf", period + "_" + name, period + " " + desc) - for name, desc in tmplt_matrices]) - - def external_external_model(self): - self.add_matrices("external_external_model", "ALL", - [("mf", "ALL_TOTAL_EETRIPS", "All periods Total for all modes external-external trips")]) - tmplt_matrices = [ - ('SOV_EETRIPS', 'SOV external-external demand'), - ('HOV2_EETRIPS', 'HOV2 external-external demand'), - ('HOV3_EETRIPS', 'HOV3 external-external demand'), - ] - for period in self._all_periods: - self.add_matrices("external_external_model", period, - [("mf", period + "_" + name, period + " " + desc) - for name, desc in tmplt_matrices]) - - def add_matrices(self, component, period, matrices): - for ident, name, desc in matrices: - self._matrices[component][period].append([ident+str(self._count[ident]), name, desc]) - self._count[ident] += 1 - - def create_matrices(self, component, periods): - with _m.logbook_trace("Create matrices for component %s" % (component.replace("_", " "))): - emmebank = self.scenario.emmebank - matrices = [] - for period in periods + ["ALL"]: - with _m.logbook_trace("For period %s" % (period)): - for ident, name, desc in self._matrices[component][period]: - existing_matrix = emmebank.matrix(name) - if existing_matrix and (existing_matrix.id != ident): - raise Exception("Matrix name conflict '%s', with id %s instead of %s. Delete all matrices first." - % (name, existing_matrix.id, ident)) - matrices.append(self._create_matrix_tool(ident, name, desc, scenario=self.scenario, overwrite=True)) - return matrices - - def get_matrix_names(self, component, periods, scenario): - self.generate_matrix_list(scenario) - matrices = [] - for period in periods: - matrices.extend([m[1] for m in self._matrices[component][period]]) - return matrices - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg diff --git a/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py b/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py deleted file mode 100644 index 513dba3..0000000 --- a/sandag_abm/src/main/emme/toolbox/initialize/initialize_transit_database.py +++ /dev/null @@ -1,168 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// initialize_transit_databse.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Coordinates the initialization of all matrices. -# The matrix names are listed for each of the model components / steps, -# and the matrix IDs are assigned consistently from the set of matrices. -# In each of the model steps the matrices are only referenced by name, -# never by ID. -# -# -# Inputs: -# components: A list of the model components / steps for which to initialize matrices -# One or more of "traffic_demand", "transit_demand", -# "traffic_skims", "transit_skims", "external_internal_model", -# "external_external_model", "truck_model", "commercial_vehicle_model" -# periods: A list of periods for which to initialize matrices, "EA", "AM", "MD", "PM", "EV" -# scenario: scenario to use for reference zone system and the emmebank in which -# the matrices will be created. Defaults to the current primary scenario. -# -# Script example: -""" - import os - import inro.emme.database.emmebank as _eb - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - main_emmebank = _eb.Emmebank(os.path.join(main_directory, "emme_project", "Database", "emmebank")) - base_scenario = main_emmebank.scenario(100) - initialize_transit_db = modeller.tool("sandag.initialize.initialize_transit_database") - initialize_transit_db(base_scenario) -""" -TOOLBOX_ORDER = 8 - - -import inro.modeller as _m -import inro.emme.network as _network -import inro.emme.database.emmebank as _eb -from inro.emme.desktop.exception import AddDatabaseError -import traceback as _traceback -import shutil as _shutil -import time -import os - -join = os.path.join - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class InitializeTransitDatabase(_m.Tool(), gen_utils.Snapshot): - - base_scenario = _m.Attribute(_m.InstanceType) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.base_scenario = _m.Modeller().scenario - self.attributes = ["base_scenario"] - - def from_snapshot(self, snapshot): - super(InitializeTransitDatabase, self).from_snapshot(snapshot) - # custom from_snapshot to load scenario object - self.base_scenario = _m.Modeller().emmebank.scenario(self.base_scenario) - return self - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Initialize transit database" - pb.description = """Create and setup database for transit assignments under 'Database_transit' directory. - Will overwrite an existing database. The TAZs will be removed and TAP nodes converted to zones.""" - pb.branding_text = "- SANDAG" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_scenario("base_scenario", - title="Base scenario:", note="Base traffic and transit scenario with TAZs.") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self(self.base_scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Initialize transit database', save_arguments=True) - def __call__(self, base_scenario, add_database=True): - attributes = {"base_scenario": base_scenario.id} - gen_utils.log_snapshot("Initialize transit database", str(self), attributes) - create_function = _m.Modeller().tool("inro.emme.data.function.create_function") - build_transit_scen = _m.Modeller().tool("sandag.assignment.build_transit_scenario") - load_properties = _m.Modeller().tool('sandag.utilities.properties') - - base_eb = base_scenario.emmebank - project_dir = os.path.dirname(os.path.dirname(base_eb.path)) - main_directory = os.path.dirname(project_dir) - props = load_properties(os.path.join(main_directory, "conf", "sandag_abm.properties")) - scenarioYear = props["scenarioYear"] - - transit_db_dir = join(project_dir, "Database_transit") - transit_db_path = join(transit_db_dir, "emmebank") - network = base_scenario.get_partial_network(["NODE"], include_attributes=True) - num_zones = sum([1 for n in network.nodes() if n["@tap_id"] > 0]) - dimensions = base_eb.dimensions - dimensions["centroids"] = num_zones - dimensions["scenarios"] = 10 - if not os.path.exists(transit_db_dir): - os.mkdir(transit_db_dir) - if os.path.exists(transit_db_path): - transit_eb = _eb.Emmebank(transit_db_path) - for scenario in transit_eb.scenarios(): - transit_eb.delete_scenario(scenario.id) - for function in transit_eb.functions(): - transit_eb.delete_function(function.id) - if transit_eb.dimensions != dimensions: - _eb.change_dimensions(transit_db_path, dimensions, keep_backup=False) - else: - transit_eb = _eb.create(transit_db_path, dimensions) - - transit_eb.title = base_eb.title[:65] + "-transit" - transit_eb.coord_unit_length = base_eb.coord_unit_length - transit_eb.unit_of_length = base_eb.unit_of_length - transit_eb.unit_of_cost = base_eb.unit_of_cost - transit_eb.unit_of_energy = base_eb.unit_of_energy - transit_eb.use_engineering_notation = base_eb.use_engineering_notation - transit_eb.node_number_digits = base_eb.node_number_digits - - zone_scenario = build_transit_scen( - period="AM", base_scenario=base_scenario, transit_emmebank=transit_eb, - scenario_id=base_scenario.id, scenario_title="%s transit zones" % (base_scenario.title), - data_table_name=scenarioYear, overwrite=True) - for function in base_scenario.emmebank.functions(): - create_function(function.id, function.expression, transit_eb) - if add_database: - self.add_database(transit_eb) - return zone_scenario - - def add_database(self, emmebank): - modeller = _m.Modeller() - desktop = modeller.desktop - data_explorer = desktop.data_explorer() - for db in data_explorer.databases(): - if os.path.normpath(db.path) == os.path.normpath(emmebank.path): - return - try: - data_explorer.add_database(emmebank.path) - except AddDatabaseError: - pass # database has already been added to the project diff --git a/sandag_abm/src/main/emme/toolbox/master_run.py b/sandag_abm/src/main/emme/toolbox/master_run.py deleted file mode 100644 index de0a8ad..0000000 --- a/sandag_abm/src/main/emme/toolbox/master_run.py +++ /dev/null @@ -1,1220 +0,0 @@ -# ////////////////////////////////////////////////////////////////////////////// -# //// /// -# //// Copyright INRO, 2016-2017. /// -# //// Rights to use and modify are granted to the /// -# //// San Diego Association of Governments and partner agencies. /// -# //// This copyright notice must be preserved. /// -# //// /// -# //// model/master_run.py /// -# //// /// -# //// /// -# //// /// -# //// /// -# ////////////////////////////////////////////////////////////////////////////// -# -# The Master run tool is the primary method to operate the SANDAG -# travel demand model. It operates all the model components. -# -# main_directory: Main ABM directory: directory which contains all of the -# ABM scenario data, including this project. The default is the parent -# directory of the current Emme project. -# scenario_id: Scenario ID for the base imported network data. The result -# scenarios are indexed in the next five scenarios by time period. -# scenario_title: title to use for the scenario. -# emmebank_title: title to use for the Emmebank (Emme database) -# num_processors: the number of processors to use for traffic and transit -# assignments and skims, aggregate demand models (where required) and -# other parallelized procedures in Emme. Default is Max available - 1. -# Properties loaded from conf/sandag_abm.properties: -# When using the tool UI, the sandag_abm.properties file is read -# and the values cached and the inputs below are pre-set. When the tool -# is started button is clicked this file is written out with the -# values specified. -# Sample rate by iteration: three values for the sample rates for each iteration -# Start from iteration: iteration from which to start the model run -# Skip steps: optional checkboxes to skip model steps. -# Note that most steps are dependent upon the results of the previous steps. -# Select link: add select link analyses for traffic. -# See the Select link analysis section under the Traffic assignment tool. -# -# Also reads and processes the per-scenario -# vehicle_class_availability.csv (optional): 0 or 1 indicators by vehicle class and specified facilities to indicate availability -# -# Script example: -""" -import inro.modeller as _m -import os -modeller = _m.Modeller() -desktop = modeller.desktop - -master_run = modeller.tool("sandag.master_run") -main_directory = os.path.dirname(os.path.dirname(desktop.project_path())) -scenario_id = 100 -scenario_title = "Base 2015 scenario" -emmebank_title = "Base 2015 with updated landuse" -num_processors = "MAX-1" -master_run(main_directory, scenario_id, scenario_title, emmebank_title, num_processors) -""" - -TOOLBOX_ORDER = 1 -VIRUTALENV_PATH = "C:\\python_virtualenv\\abm14_2_0" - -import inro.modeller as _m -import inro.emme.database.emmebank as _eb - -import traceback as _traceback -import glob as _glob -import subprocess as _subprocess -import ctypes as _ctypes -import json as _json -import shutil as _shutil -import tempfile as _tempfile -from copy import deepcopy as _copy -from collections import defaultdict as _defaultdict -import time as _time -import socket as _socket -import sys -import os - -import pandas as pd -import numpy as np -import csv -import datetime -import pyodbc -import win32com.client as win32 - -_join = os.path.join -_dir = os.path.dirname -_norm = os.path.normpath - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") -props_utils = _m.Modeller().module("sandag.utilities.properties") - - -class MasterRun(props_utils.PropertiesSetter, _m.Tool(), gen_utils.Snapshot): - main_directory = _m.Attribute(unicode) - scenario_id = _m.Attribute(int) - scenario_title = _m.Attribute(unicode) - emmebank_title = _m.Attribute(unicode) - num_processors = _m.Attribute(str) - select_link = _m.Attribute(unicode) - username = _m.Attribute(unicode) - password = _m.Attribute(unicode) - - properties_path = _m.Attribute(unicode) - - tool_run_msg = "" - - def __init__(self): - super(MasterRun, self).__init__() - project_dir = _dir(_m.Modeller().desktop.project.path) - self.main_directory = _dir(project_dir) - self.properties_path = _join(_dir(project_dir), "conf", "sandag_abm.properties") - self.scenario_id = 100 - self.scenario_title = "" - self.emmebank_title = "" - self.num_processors = "MAX-1" - self.select_link = '[]' - self.username = os.environ.get("USERNAME") - self.attributes = [ - "main_directory", "scenario_id", "scenario_title", "emmebank_title", - "num_processors", "select_link" - ] - self._log_level = "ENABLED" - self.LOCAL_ROOT = "C:\\abm_runs" - - def page(self): - self.load_properties() - pb = _m.ToolPageBuilder(self) - pb.title = "Master run ABM" - pb.description = """Runs the SANDAG ABM, assignments, and other demand model tools.""" - pb.branding_text = "- SANDAG - Model" - tool_proxy_tag = pb.tool_proxy_tag - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('main_directory', 'directory', - title='Select main ABM directory', note='') - pb.add_text_box('scenario_id', title="Scenario ID:") - pb.add_text_box('scenario_title', title="Scenario title:", size=80) - pb.add_text_box('emmebank_title', title="Emmebank title:", size=60) - dem_utils.add_select_processors("num_processors", pb, self) - - # username and password input for distributed assignment - # username also used in the folder name for the local drive operation - pb.add_html(''' -
    -
    Credentials for remote run
    -
    - Username: - - Password: - -
    -
    - Note: required for running distributed traffic assignments using PsExec. -
    - Distributed / single node modes are configured in "config/server-config.csv". -
    The username is also used for the folder name when running on the local drive. -
    -
    ''' % {"tool_proxy_tag": tool_proxy_tag}) - - # defined in properties utilities - self.add_properties_interface(pb, disclosure=True) - # redirect properties file after browse of main_directory - pb.add_html(""" -""" % {"tool_proxy_tag": tool_proxy_tag}) - - traffic_assign = _m.Modeller().tool("sandag.assignment.traffic_assignment") - traffic_assign._add_select_link_interface(pb) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self.save_properties() - self(self.main_directory, self.scenario_id, self.scenario_title, self.emmebank_title, - self.num_processors, self.select_link, username=self.username, password=self.password) - run_msg = "Model run complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception(error, _traceback.format_exc()) - - raise - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - @_m.logbook_trace("Master run model", save_arguments=True) - def __call__(self, main_directory, scenario_id, scenario_title, emmebank_title, num_processors, - select_link=None, periods=["EA", "AM", "MD", "PM", "EV"], username=None, password=None): - attributes = { - "main_directory": main_directory, - "scenario_id": scenario_id, - "scenario_title": scenario_title, - "emmebank_title": emmebank_title, - "num_processors": num_processors, - "select_link": select_link, - "periods": periods, - "username": username, - } - gen_utils.log_snapshot("Master run model", str(self), attributes) - - modeller = _m.Modeller() - # Checking that the virtualenv path is set and the folder is installed - if not os.path.exists(VIRUTALENV_PATH): - raise Exception("Python virtual environment not installed at expected location %s" % VIRUTALENV_PATH) - venv_path = os.environ.get("PYTHON_VIRTUALENV") - if not venv_path: - raise Exception("Environment variable PYTHON_VIRTUALENV not set, start Emme from 'start_emme_with_virtualenv.bat'") - if not venv_path == VIRUTALENV_PATH: - raise Exception("PYTHON_VIRTUALENV is not the expected value (%s instead of %s)" % (venv_path, VIRUTALENV_PATH)) - venv_path_found = False - for path in sys.path: - if VIRUTALENV_PATH in path: - venv_path_found = True - break - if not venv_path_found: - raise Exception("Python virtual environment not found in system path %s" % VIRUTALENV_PATH) - copy_scenario = modeller.tool("inro.emme.data.scenario.copy_scenario") - run4Ds = modeller.tool("sandag.import.run4Ds") - import_network = modeller.tool("sandag.import.import_network") - input_checker = modeller.tool("sandag.import.input_checker") - init_transit_db = modeller.tool("sandag.initialize.initialize_transit_database") - init_matrices = modeller.tool("sandag.initialize.initialize_matrices") - import_demand = modeller.tool("sandag.import.import_seed_demand") - build_transit_scen = modeller.tool("sandag.assignment.build_transit_scenario") - transit_assign = modeller.tool("sandag.assignment.transit_assignment") - run_truck = modeller.tool("sandag.model.truck.run_truck_model") - external_internal = modeller.tool("sandag.model.external_internal") - external_external = modeller.tool("sandag.model.external_external") - import_auto_demand = modeller.tool("sandag.import.import_auto_demand") - import_transit_demand = modeller.tool("sandag.import.import_transit_demand") - export_transit_skims = modeller.tool("sandag.export.export_transit_skims") - export_for_transponder = modeller.tool("sandag.export.export_for_transponder") - export_network_data = modeller.tool("sandag.export.export_data_loader_network") - export_matrix_data = modeller.tool("sandag.export.export_data_loader_matrices") - export_tap_adjacent_lines = modeller.tool("sandag.export.export_tap_adjacent_lines") - export_for_commercial_vehicle = modeller.tool("sandag.export.export_for_commercial_vehicle") - validation = modeller.tool("sandag.validation.validation") - file_manager = modeller.tool("sandag.utilities.file_manager") - utils = modeller.module('sandag.utilities.demand') - load_properties = modeller.tool('sandag.utilities.properties') - run_summary = modeller.tool("sandag.utilities.run_summary") - - self.username = username - self.password = password - - props = load_properties(_join(main_directory, "conf", "sandag_abm.properties")) - props.set_year_specific_properties(_join(main_directory, "input", "parametersByYears.csv")) - props.set_year_specific_properties(_join(main_directory, "input", "filesByYears.csv")) - props.save() - # Log current state of props file for debugging of UI / file sync issues - attributes = dict((name, props["RunModel." + name]) for name in self._run_model_names) - _m.logbook_write("SANDAG properties file", attributes=attributes) - if self._properties: # Tool has been called via the UI - # Compare UI values and file values to make sure they are the same - error_text = ("Different value found in sandag_abm.properties than specified in UI for '%s'. " - "Close sandag_abm.properties if open in any text editor, check UI and re-run.") - for name in self._run_model_names: - if getattr(self, name) != props["RunModel." + name]: - raise Exception(error_text % name) - - scenarioYear = str(props["scenarioYear"]) - startFromIteration = props["RunModel.startFromIteration"] - precision = props["RunModel.MatrixPrecision"] - minSpaceOnC = props["RunModel.minSpaceOnC"] - sample_rate = props["sample_rates"] - end_iteration = len(sample_rate) - scale_factor = props["cvm.scale_factor"] - visualizer_reference_path = props["visualizer.reference.path"] - visualizer_output_file = props["visualizer.output"] - visualizer_reference_label = props["visualizer.reference.label"] - visualizer_build_label = props["visualizer.build.label"] - mgraInputFile = props["mgra.socec.file"] - - #for zone restructing in network files - taz_cwk_file = props["taz.to.cluster.crosswalk.file"] - cluster_zone_file = props["cluster.zone.centroid.file"] - - period_ids = list(enumerate(periods, start=int(scenario_id) + 1)) - - useLocalDrive = props["RunModel.useLocalDrive"] - - skip4Ds = props["RunModel.skip4Ds"] - skipInputChecker = props["RunModel.skipInputChecker"] - skipInitialization = props["RunModel.skipInitialization"] - deleteAllMatrices = props["RunModel.deleteAllMatrices"] - skipCopyWarmupTripTables = props["RunModel.skipCopyWarmupTripTables"] - skipCopyBikeLogsum = props["RunModel.skipCopyBikeLogsum"] - skipCopyWalkImpedance = props["RunModel.skipCopyWalkImpedance"] - skipWalkLogsums = props["RunModel.skipWalkLogsums"] - skipBikeLogsums = props["RunModel.skipBikeLogsums"] - skipBuildNetwork = props["RunModel.skipBuildNetwork"] - skipHighwayAssignment = props["RunModel.skipHighwayAssignment"] - skipTransitSkimming = props["RunModel.skipTransitSkimming"] - skipTransponderExport = props["RunModel.skipTransponderExport"] - skipCoreABM = props["RunModel.skipCoreABM"] - skipOtherSimulateModel = props["RunModel.skipOtherSimulateModel"] - skipMAASModel = props["RunModel.skipMAASModel"] - skipCTM = props["RunModel.skipCTM"] - skipEI = props["RunModel.skipEI"] - skipExternal = props["RunModel.skipExternalExternal"] - skipTruck = props["RunModel.skipTruck"] - skipTripTableCreation = props["RunModel.skipTripTableCreation"] - skipFinalHighwayAssignment = props["RunModel.skipFinalHighwayAssignment"] - skipFinalHighwayAssignmentStochastic = props["RunModel.skipFinalHighwayAssignmentStochastic"] - if skipFinalHighwayAssignmentStochastic == True: - makeFinalHighwayAssignmentStochastic = False - else: - makeFinalHighwayAssignmentStochastic = True - skipFinalTransitAssignment = props["RunModel.skipFinalTransitAssignment"] - skipVisualizer = props["RunModel.skipVisualizer"] - skipDataExport = props["RunModel.skipDataExport"] - skipDataLoadRequest = props["RunModel.skipDataLoadRequest"] - skipDeleteIntermediateFiles = props["RunModel.skipDeleteIntermediateFiles"] - skipTransitShed = props["RunModel.skipTransitShed"] - transitShedThreshold = props["transitShed.threshold"] - transitShedTOD = props["transitShed.TOD"] - - #check if visualizer.reference.path is valid in filesbyyears.csv - if not os.path.exists(visualizer_reference_path): - raise Exception("Visualizer reference %s does not exist. Check filesbyyears.csv." %(visualizer_reference_path)) - - if useLocalDrive: - folder_name = os.path.basename(main_directory) - if not os.path.exists(_join(self.LOCAL_ROOT, username, folder_name, "report")): # check free space only if it is a new run - self.check_free_space(minSpaceOnC) - # if initialization copy ALL files from remote - # else check file meta data and copy those that have changed - initialize = (skipInitialization == False and startFromIteration == 1) - local_directory = file_manager( - "DOWNLOAD", main_directory, username, scenario_id, initialize=initialize) - self._path = local_directory - else: - self._path = main_directory - - drive, path_no_drive = os.path.splitdrive(self._path) - path_forward_slash = path_no_drive.replace("\\", "/") - input_dir = _join(self._path, "input") - input_truck_dir = _join(self._path, "input_truck") - output_dir = _join(self._path, "output") - validation_dir = _join(self._path, "analysis/validation") - main_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database", "emmebank")) - if emmebank_title: - main_emmebank.title = emmebank_title - external_zones = "1-12" - - travel_modes = ["auto", "tran", "nmot", "othr"] - core_abm_files = ["Trips*.omx", "InternalExternalTrips*.omx"] - core_abm_files = [mode + name for name in core_abm_files for mode in travel_modes] - smm_abm_files = ["AirportTrips*.omx", "CrossBorderTrips*.omx", "VisitorTrips*.omx"] - smm_abm_files = [mode + name for name in smm_abm_files for mode in travel_modes] - maas_abm_files = ["EmptyAVTrips.omx", "TNCVehicleTrips*.omx"] - - relative_gap = props["convergence"] - max_assign_iterations = 1000 - mgra_lu_input_file = props["mgra.socec.file"] - - with _m.logbook_trace("Setup and initialization"): - self.set_global_logbook_level(props) - - # Swap Server Configurations - self.run_proc("serverswap.bat", [drive, path_no_drive, path_forward_slash], "Run ServerSwap") - self.check_for_fatal(_join(self._path, "logFiles", "serverswap.log"), - "ServerSwap failed! Open logFiles/serverswap.log for details.") - self.run_proc("checkAtTransitNetworkConsistency.cmd", [drive, path_forward_slash], - "Checking if AT and Transit Networks are consistent") - self.check_for_fatal(_join(self._path, "logFiles", "AtTransitCheck_event.log"), - "AT and Transit network consistency checking failed! Open AtTransitCheck_event.log for details.") - - if startFromIteration == 1: # only run the setup / init steps if starting from iteration 1 - if not skipWalkLogsums: - self.run_proc("runSandagWalkLogsums.cmd", [drive, path_forward_slash], - "Walk - create AT logsums and impedances") - if not skipCopyWalkImpedance: - self.copy_files(["walkMgraEquivMinutes.csv", "walkMgraTapEquivMinutes.csv", "microMgraEquivMinutes.csv", "microMgraTapEquivMinutes.csv"], - input_dir, output_dir) - - if not skip4Ds: - run4Ds(path=self._path, int_radius=0.65, ref_path=visualizer_reference_path) - - - mgraFile = 'mgra13_based_input' + str(scenarioYear) + '.csv' - self.complete_work(scenarioYear, input_dir, output_dir, mgraFile, "walkMgraEquivMinutes.csv") - - if not skipBuildNetwork: - base_scenario = import_network( - source=input_dir, - merged_scenario_id=scenario_id, - title=scenario_title, - data_table_name=scenarioYear, - overwrite=True, - emmebank=main_emmebank) - - if "modify_network.py" in os.listdir(os.getcwd()): - try: - with _m.logbook_trace("Modify network script"): - import modify_network - reload(modify_network) - modify_network.run(base_scenario) - except ImportError as e: - pass - - hwy_network = self.update_centroid_connectors( - input_dir, - base_scenario, - main_emmebank, - external_zones, - taz_cwk_file, - cluster_zone_file) - - base_scenario.publish_network(hwy_network) - - if not skipInputChecker: - input_checker(path=self._path) - - export_tap_adjacent_lines(_join(output_dir, "tapLines.csv"), base_scenario) - # parse vehicle availablility file by time-of-day - availability_file = "vehicle_class_availability.csv" - availabilities = self.parse_availability_file(_join(input_dir, availability_file), periods) - # initialize per time-period scenarios - for number, period in period_ids: - title = "%s - %s assign" % (base_scenario.title, period) - # copy_scenario(base_scenario, number, title, overwrite=True) - _m.logbook_write( - name="Copy scenario %s to %s" % (base_scenario.number, number), - attributes={ - 'from_scenario': base_scenario.number, - 'scenario_id': number, - 'overwrite': True, - 'scenario_title': title - } - ) - if main_emmebank.scenario(number): - main_emmebank.delete_scenario(number) - scenario = main_emmebank.copy_scenario(base_scenario.number, number) - scenario.title = title - # Apply availabilities by facility and vehicle class to this time period - self.apply_availabilities(period, scenario, availabilities) - else: - base_scenario = main_emmebank.scenario(scenario_id) - - if not skipInitialization: - # initialize traffic demand, skims, truck, CV, EI, EE matrices - traffic_components = [ - "traffic_skims", - "truck_model", - "external_internal_model", "external_external_model"] - if not skipCopyWarmupTripTables: - traffic_components.append("traffic_demand") - init_matrices(traffic_components, periods, base_scenario, deleteAllMatrices) - - transit_scenario = init_transit_db(base_scenario, add_database=not useLocalDrive) - transit_emmebank = transit_scenario.emmebank - transit_components = ["transit_skims"] - if not skipCopyWarmupTripTables: - transit_components.append("transit_demand") - init_matrices(transit_components, periods, transit_scenario, deleteAllMatrices) - else: - transit_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database_transit", "emmebank")) - transit_scenario = transit_emmebank.scenario(base_scenario.number) - - if not skipCopyWarmupTripTables: - # import seed auto demand and seed truck demand - for period in periods: - omx_file = _join(input_dir, "trip_%s.omx" % period) - import_demand(omx_file, "AUTO", period, base_scenario) - import_demand(omx_file, "TRUCK", period, base_scenario) - - if not skipBikeLogsums: - self.run_proc("runSandagBikeLogsums.cmd", [drive, path_forward_slash], - "Bike - create AT logsums and impedances") - if not skipCopyBikeLogsum: - self.copy_files(["bikeMgraLogsum.csv", "bikeTazLogsum.csv"], input_dir, output_dir) - - else: - base_scenario = main_emmebank.scenario(scenario_id) - transit_emmebank = _eb.Emmebank(_join(self._path, "emme_project", "Database_transit", "emmebank")) - transit_scenario = transit_emmebank.scenario(base_scenario.number) - - # Check that setup files were generated - self.run_proc("CheckOutput.bat", [drive + path_no_drive, 'Setup'], "Check for outputs") - - # Note: iteration indexes from 0, msa_iteration indexes from 1 - for iteration in range(startFromIteration - 1, end_iteration): - msa_iteration = iteration + 1 - with _m.logbook_trace("Iteration %s" % msa_iteration): - if not skipCoreABM[iteration] or not skipOtherSimulateModel[iteration] or not skipMAASModel[iteration]: - self.run_proc("runMtxMgr.cmd", [drive, drive + path_no_drive], "Start matrix manager") - self.run_proc("runHhMgr.cmd", [drive, drive + path_no_drive], "Start Hh manager") - - if not skipHighwayAssignment[iteration]: - # run traffic assignment - # export traffic skims - with _m.logbook_trace("Traffic assignment and skims"): - self.run_traffic_assignments( - base_scenario, period_ids, msa_iteration, relative_gap, - max_assign_iterations, num_processors) - self.run_proc("CreateD2TAccessFile.bat", [drive, path_forward_slash], - "Create drive to transit access file", capture_output=True) - - if not skipTransitSkimming[iteration]: - # run transit assignment - # export transit skims - with _m.logbook_trace("Transit assignments and skims"): - for number, period in period_ids: - src_period_scenario = main_emmebank.scenario(number) - transit_assign_scen = build_transit_scen( - period=period, base_scenario=src_period_scenario, - transit_emmebank=transit_emmebank, - scenario_id=src_period_scenario.id, - scenario_title="%s %s transit assign" % (base_scenario.title, period), - data_table_name=scenarioYear, overwrite=True) - transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, - skims_only=True, num_processors=num_processors) - - omx_file = _join(output_dir, "transit_skims.omx") - export_transit_skims(omx_file, periods, transit_scenario) - - if not skipTransponderExport[iteration]: - am_scenario = main_emmebank.scenario(base_scenario.number + 2) - export_for_transponder(output_dir, num_processors, am_scenario) - - # For each step move trip matrices so run will stop if ctramp model - # doesn't produced csv/omx files for assignment - # also needed as CT-RAMP does not overwrite existing files - if not skipCoreABM[iteration]: - self.remove_prev_iter_files(core_abm_files, output_dir, iteration) - self.run_proc( - "runSandagAbm_SDRM.cmd", - [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], - "Java-Run CT-RAMP", capture_output=True) - if not skipOtherSimulateModel[iteration]: - self.remove_prev_iter_files(smm_abm_files, output_dir, iteration) - self.run_proc( - "runSandagAbm_SMM.cmd", - [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], - "Java-Run airport model, visitor model, cross-border model", capture_output=True) - - if not skipMAASModel[iteration]: - self.remove_prev_iter_files(maas_abm_files, output_dir, iteration) - self.run_proc( - "runSandagAbm_MAAS.cmd", - [drive, drive + path_forward_slash, sample_rate[iteration], msa_iteration], - "Java-Run AV allocation model and TNC routing model", capture_output=True) - - if not skipCTM[iteration]: - export_for_commercial_vehicle(output_dir, base_scenario) - self.run_proc( - "cvm.bat", - [drive, path_no_drive, path_forward_slash, scale_factor, mgra_lu_input_file, - "tazcentroids_cvm.csv"], - "Commercial vehicle model", capture_output=True) - if msa_iteration == startFromIteration: - external_zones = "1-12" - if not skipTruck[iteration]: - # run truck model (generate truck trips) - run_truck(True, input_dir, input_truck_dir, num_processors, base_scenario) - # run EI model "US to SD External Trip Model" - if not skipEI[iteration]: - external_internal(input_dir, base_scenario) - # run EE model - if not skipExternal[iteration]: - external_external(input_dir, external_zones, base_scenario) - - # import demand from all sub-market models from CT-RAMP and - # add CV trips to auto demand - # add EE and EI trips to auto demand - if not skipTripTableCreation[iteration]: - import_auto_demand(output_dir, external_zones, num_processors, base_scenario) - - if not skipFinalHighwayAssignment: - with _m.logbook_trace("Final traffic assignments"): - # Final iteration is assignment only, no skims - final_iteration = 4 - self.run_traffic_assignments( - base_scenario, period_ids, final_iteration, relative_gap, max_assign_iterations, - num_processors, select_link, makeFinalHighwayAssignmentStochastic, input_dir) - - if not skipFinalTransitAssignment: - import_transit_demand(output_dir, transit_scenario) - with _m.logbook_trace("Final transit assignments"): - # Final iteration includes the transit skims per ABM-1072 - for number, period in period_ids: - src_period_scenario = main_emmebank.scenario(number) - transit_assign_scen = build_transit_scen( - period=period, base_scenario=src_period_scenario, - transit_emmebank=transit_emmebank, scenario_id=src_period_scenario.id, - scenario_title="%s - %s transit assign" % (base_scenario.title, period), - data_table_name=scenarioYear, overwrite=True) - transit_assign(period, transit_assign_scen, data_table_name=scenarioYear, - num_processors=num_processors) - omx_file = _join(output_dir, "transit_skims.omx") - export_transit_skims(omx_file, periods, transit_scenario, big_to_zero=True) - - if not skipTransitShed: - # write walk and drive transit sheds - self.run_proc("runtransitreporter.cmd", [drive, path_forward_slash, transitShedThreshold, transitShedTOD], - "Create walk and drive transit sheds", - capture_output=True) - - if not skipVisualizer: - self.run_proc("RunViz.bat", - [drive, path_no_drive, visualizer_reference_path, visualizer_output_file, "NO", visualizer_reference_label, visualizer_build_label, mgraInputFile], - "HTML Visualizer", capture_output=True) - - if not skipDataExport: - # export network and matrix results from Emme directly to T if using local drive - output_directory = _join(self._path, "output") - export_network_data(self._path, scenario_id, main_emmebank, transit_emmebank, num_processors) - export_matrix_data(output_directory, base_scenario, transit_scenario) - # export core ABM data - # Note: uses relative project structure, so cannot redirect to T drive - self.run_proc("DataExporter.bat", [drive, path_no_drive], "Export core ABM data",capture_output=True) - - #Validation for 2016 scenario - if scenarioYear == "2016": - validation(self._path, main_emmebank, base_scenario) # to create source_EMME.xlsx - - # #Create Worksheet for ABM Validation using PowerBI Visualization #JY: can be uncommented if deciding to incorporate PowerBI vis in ABM workflow - # self.run_proc("VisPowerBI.bat", # forced to update excel links - # [drive, path_no_drive, scenarioYear, 0], - # "VisPowerBI", - # capture_output=True) - - ### CL: Below step is temporarily used to update validation output files. When Gregor complete Upload procedure, below step should be removed. 05/31/20 - # self.run_proc("ExcelUpdate.bat", # forced to update excel links - # [drive, path_no_drive, scenarioYear, 0], - # "ExcelUpdate", - # capture_output=True) - - ### ES: Commented out until this segment is updated to reference new database. 9/10/20 ### - # add segments below for auto-reporting, YMA, 1/23/2019 - # add this loop to find the sceanro_id in the [dimension].[scenario] table - - #database_scenario_id = 0 - #int_hour = 0 - #while int_hour <= 96: - - # database_scenario_id = self.sql_select_scenario(scenarioYear, end_iteration, - # sample_rate[end_iteration - 1], path_no_drive, - # start_db_time) - # if database_scenario_id > 0: - # break - - # int_hour = int_hour + 1 - # _time.sleep(900) # wait for 15 mins - - # if load failed, then send notification email - #if database_scenario_id == 0 and int_hour > 96: - # str_request_check_result = self.sql_check_load_request(scenarioYear, path_no_drive, username, - # start_db_time) - # print(str_request_check_result) - # sys.exit(0) - # self.send_notification(str_request_check_result,username) #not working in server - #else: - # print(database_scenario_id) - # self.run_proc("DataSummary.bat", # get summary from database, added for auto-reporting - # [drive, path_no_drive, scenarioYear, database_scenario_id], - # "Data Summary") - - # self.run_proc("ExcelUpdate.bat", # forced to update excel links - # [drive, path_no_drive, scenarioYear, database_scenario_id], - # "Excel Update", - # capture_output=True) - - # terminate all java processes - _subprocess.call("taskkill /F /IM java.exe") - - # close all DOS windows - _subprocess.call("taskkill /F /IM cmd.exe") - - # UPLOAD DATA AND SWITCH PATHS - if useLocalDrive: - file_manager("UPLOAD", main_directory, username, scenario_id, - delete_local_files=not skipDeleteIntermediateFiles) - self._path = main_directory - drive, path_no_drive = os.path.splitdrive(self._path) - # self._path = main_directory - # drive, path_no_drive = os.path.splitdrive(self._path) - init_transit_db.add_database( - _eb.Emmebank(_join(main_directory, "emme_project", "Database_transit", "emmebank"))) - - if not skipDataLoadRequest: - start_db_time = datetime.datetime.now() # record the time to search for request id in the load request table, YMA, 1/23/2019 - # start_db_time = start_db_time + datetime.timedelta(minutes=0) - self.run_proc("DataLoadRequest.bat", - [drive + path_no_drive, end_iteration, scenarioYear, sample_rate[end_iteration - 1]], - "Data load request") - - # delete trip table files in iteration sub folder if model finishes without errors - if not useLocalDrive and not skipDeleteIntermediateFiles: - for msa_iteration in range(startFromIteration, end_iteration + 1): - self.delete_files( - ["auto*Trips*.omx", "tran*Trips*.omx", "nmot*.omx", "othr*.omx", "trip*.omx"], - _join(output_dir, "iter%s" % (msa_iteration))) - - # record run time - run_summary(path=self._path) - - def set_global_logbook_level(self, props): - self._log_level = props.get("RunModel.LogbookLevel", "ENABLED") - log_all = _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.VALUE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG - log_states = { - "ENABLED": log_all, - "DISABLE_ON_ERROR": log_all, - "NO_EXTERNAL_REPORTS": log_all, - "NO_REPORTS": _m.LogbookLevel.ATTRIBUTE | _m.LogbookLevel.COOKIE | _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, - "TITLES_ONLY": _m.LogbookLevel.TRACE | _m.LogbookLevel.LOG, - "DISABLED": _m.LogbookLevel.NONE, - } - _m.logbook_write("Setting logbook level to %s" % self._log_level) - try: - _m.logbook_level(log_states[self._log_level]) - except KeyError: - raise Exception("properties.RunModel.LogLevel: value must be one of %s" % ",".join(log_states.keys())) - - def run_traffic_assignments(self, base_scenario, period_ids, msa_iteration, relative_gap, - max_assign_iterations, num_processors, select_link=None, - makeFinalHighwayAssignmentStochastic=False, input_dir=None): - modeller = _m.Modeller() - traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") - export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") - output_dir = _join(self._path, "output") - main_emmebank = base_scenario.emmebank - - machine_name = _socket.gethostname().lower() - with open(_join(self._path, "conf", "server-config.csv")) as f: - columns = f.next().split(",") - for line in f: - values = dict(zip(columns, line.split(","))) - name = values["ServerName"].lower() - if name == machine_name: - server_config = values - break - else: - _m.logbook_write("Warning: current machine name not found in " - "conf\\server-config.csv ServerName column") - server_config = {"SNODE": "yes"} - distributed = server_config["SNODE"] == "no" - if distributed and not makeFinalHighwayAssignmentStochastic: - scen_map = dict((p, main_emmebank.scenario(n)) for n, p in period_ids) - input_args = { - "msa_iteration": msa_iteration, - "relative_gap": relative_gap, - "max_assign_iterations": max_assign_iterations, - "select_link": select_link - } - - periods_node1 = ["PM", "MD"] - input_args["num_processors"] = server_config["THREADN1"], - database_path1, skim_names1 = self.setup_remote_database( - [scen_map[p] for p in periods_node1], periods_node1, 1, msa_iteration) - self.start_assignments( - server_config["NODE1"], database_path1, periods_node1, scen_map, input_args) - - periods_node2 = ["AM"] - input_args["num_processors"] = server_config["THREADN2"] - database_path2, skim_names2 = self.setup_remote_database( - [scen_map[p] for p in periods_node2], periods_node2, 2, msa_iteration) - self.start_assignments( - server_config["NODE2"], database_path2, periods_node2, scen_map, input_args) - - try: - # run assignments locally - periods_local = ["EA", "EV"] - for period in periods_local: - local_scenario = scen_map[period] - traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, - num_processors, local_scenario, select_link) - omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) - if msa_iteration <= 4: - export_traffic_skims(period, omx_file, base_scenario) - scenarios = { - database_path1: [scen_map[p] for p in periods_node1], - database_path2: [scen_map[p] for p in periods_node2] - } - skim_names = { - database_path1: skim_names1, database_path2: skim_names2 - } - self.wait_and_copy([database_path1, database_path2], scenarios, skim_names) - except: - # Note: this will kill ALL python processes - not suitable if servers are being - # used for other tasks - _subprocess.call("taskkill /F /T /S \\\\%s /IM python.exe" % server_config["NODE1"]) - _subprocess.call("taskkill /F /T /S \\\\%s /IM python.exe" % server_config["NODE2"]) - raise - else: - for number, period in period_ids: - period_scenario = main_emmebank.scenario(number) - traffic_assign(period, msa_iteration, relative_gap, max_assign_iterations, - num_processors, period_scenario, select_link, stochastic=makeFinalHighwayAssignmentStochastic, input_directory=input_dir) - omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) - if msa_iteration <= 4: - export_traffic_skims(period, omx_file, base_scenario) - - def run_proc(self, name, arguments, log_message, capture_output=False): - path = _join(self._path, "bin", name) - if not os.path.exists(path): - raise Exception("No command / batch file '%s'" % path) - command = path + " " + " ".join([str(x) for x in arguments]) - attrs = {"command": command, "name": name, "arguments": arguments} - with _m.logbook_trace(log_message, attributes=attrs): - if capture_output and self._log_level != "NO_EXTERNAL_REPORTS": - report = _m.PageBuilder(title="Process run %s" % name) - report.add_html('Command:

    %s

    ' % command) - # temporary file to capture output error messages generated by Java - err_file_ref, err_file_path = _tempfile.mkstemp(suffix='.log') - err_file = os.fdopen(err_file_ref, "w") - try: - output = _subprocess.check_output(command, stderr=err_file, cwd=self._path, shell=True) - report.add_html('Output:

    %s
    ' % output) - except _subprocess.CalledProcessError as error: - report.add_html('Output:

    %s
    ' % error.output) - raise - finally: - err_file.close() - with open(err_file_path, 'r') as f: - error_msg = f.read() - os.remove(err_file_path) - if error_msg: - report.add_html('Error message(s):

    %s
    ' % error_msg) - try: - # No raise on writing report error - # due to observed issue with runs generating reports which cause - # errors when logged - _m.logbook_write("Process run %s report" % name, report.render()) - except Exception as error: - print _time.strftime("%Y-%M-%d %H:%m:%S") - print "Error writing report '%s' to logbook" % name - print error - print _traceback.format_exc(error) - if self._log_level == "DISABLE_ON_ERROR": - _m.logbook_level(_m.LogbookLevel.NONE) - else: - _subprocess.check_call(command, cwd=self._path, shell=True) - - @_m.logbook_trace("Check free space on C") - def check_free_space(self, min_space): - path = "c:\\" - temp, total, free = _ctypes.c_ulonglong(), _ctypes.c_ulonglong(), _ctypes.c_ulonglong() - if sys.version_info >= (3,) or isinstance(path, unicode): - fun = _ctypes.windll.kernel32.GetDiskFreeSpaceExW - else: - fun = _ctypes.windll.kernel32.GetDiskFreeSpaceExA - ret = fun(path, _ctypes.byref(temp), _ctypes.byref(total), _ctypes.byref(free)) - if ret == 0: - raise _ctypes.WinError() - total = total.value / (1024.0 ** 3) - free = free.value / (1024.0 ** 3) - if free < min_space: - raise Exception("Free space on C drive %s is less than %s" % (free, min_space)) - - def remove_prev_iter_files(self, file_names, output_dir, iteration): - if iteration == 0: - self.delete_files(file_names, output_dir) - else: - self.move_files(file_names, output_dir, _join(output_dir, "iter%s" % (iteration))) - - def copy_files(self, file_names, from_dir, to_dir): - with _m.logbook_trace("Copy files %s" % ", ".join(file_names)): - for file_name in file_names: - from_file = _join(from_dir, file_name) - _shutil.copy(from_file, to_dir) - - def complete_work(self, scenarioYear, input_dir, output_dir, input_file, output_file): - - fullList = np.array(pd.read_csv(_join(input_dir, input_file))['mgra']) - workList = np.array(pd.read_csv(_join(output_dir, output_file))['i']) - - list_set = set(workList) - unique_list = (list(list_set)) - notMatch = [x for x in fullList if x not in unique_list] - - if notMatch: - out_file = _join(output_dir, output_file) - with open(out_file, 'ab') as csvfile: - spamwriter = csv.writer(csvfile) - # spamwriter.writerow([]) - for item in notMatch: - # pdb.set_trace() - spamwriter.writerow([item, item, '30', '30', '30']) - - def move_files(self, file_names, from_dir, to_dir): - with _m.logbook_trace("Move files %s" % ", ".join(file_names)): - if not os.path.exists(to_dir): - os.mkdir(to_dir) - for file_name in file_names: - all_files = _glob.glob(_join(from_dir, file_name)) - for path in all_files: - try: - dst_file = _join(to_dir, os.path.basename(path)) - if os.path.exists(dst_file): - os.remove(dst_file) - _shutil.move(path, to_dir) - except Exception as error: - _m.logbook_write( - "Error moving file %s" % path, {"error": _traceback.format_exc(error)}) - - def delete_files(self, file_names, directory): - with _m.logbook_trace("Delete files %s" % ", ".join(file_names)): - for file_name in file_names: - all_files = _glob.glob(_join(directory, file_name)) - for path in all_files: - os.remove(path) - - def check_for_fatal(self, file_name, error_msg): - with open(file_name, 'a+') as f: - for line in f: - if "FATAL" in line: - raise Exception(error_msg) - - def set_active(self, emmebank): - modeller = _m.Modeller() - desktop = modeller.desktop - data_explorer = desktop.data_explorer() - for db in data_explorer.databases(): - if _norm(db.path) == _norm(unicode(emmebank)): - db.open() - return db - return None - - def parse_availability_file(self, file_path, periods): - if os.path.exists(file_path): - availabilities = _defaultdict(lambda: _defaultdict(lambda: dict())) - # NOTE: CSV Reader sets the field names to UPPERCASE for consistency - with gen_utils.CSVReader(file_path) as r: - for row in r: - name = row.pop("FACILITY_NAME") - class_name = row.pop("VEHICLE_CLASS") - for period in periods: - is_avail = int(row[period + "_AVAIL"]) - if is_avail not in [1, 0]: - msg = "Error processing file '%s': value for period %s class %s facility %s is not 1 or 0" - raise Exception(msg % (file_path, period, class_name, name)) - availabilities[period][name][class_name] = is_avail - else: - availabilities = None - return availabilities - - def apply_availabilities(self, period, scenario, availabilities): - if availabilities is None: - return - - network = scenario.get_network() - hov2 = network.mode("h") - hov2_trnpdr = network.mode("H") - hov3 = network.mode("i") - hov3_trnpdr = network.mode("I") - sov = network.mode("s") - sov_trnpdr = network.mode("S") - heavy_trk = network.mode("v") - heavy_trk_trnpdr = network.mode("V") - medium_trk = network.mode("m") - medium_trk_trnpdr = network.mode("M") - light_trk = network.mode("t") - light_trk_trnpdr = network.mode("T") - - class_mode_map = { - "DA": set([sov_trnpdr, sov]), - "S2": set([hov2_trnpdr, hov2]), - "S3": set([hov3_trnpdr, hov3]), - "TRK_L": set([light_trk_trnpdr, light_trk]), - "TRK_M": set([medium_trk_trnpdr, medium_trk]), - "TRK_H": set([heavy_trk_trnpdr, heavy_trk]), - } - report = ["
    Link mode changes
    "] - for name, class_availabilities in availabilities[period].iteritems(): - report.append("
    %s
    " % name) - changes = _defaultdict(lambda: 0) - for link in network.links(): - if name in link["#name"]: - for class_name, is_avail in class_availabilities.iteritems(): - modes = class_mode_map[class_name] - if is_avail == 1 and not modes.issubset(link.modes): - link.modes |= modes - changes["added %s to" % class_name] += 1 - elif is_avail == 0 and modes.issubset(link.modes): - link.modes -= modes - changes["removed %s from" % class_name] += 1 - report.append("
      ") - for x in changes.iteritems(): - report.append("
    • %s %s links
    • " % x) - report.append("
    ") - scenario.publish_network(network) - - title = "Apply global class availabilities by faclity name for period %s" % period - log_report = _m.PageBuilder(title=title) - for item in report: - log_report.add_html(item) - _m.logbook_write(title, log_report.render()) - - def setup_remote_database(self, src_scenarios, periods, remote_num, msa_iteration): - with _m.logbook_trace("Set up remote database #%s for %s" % (remote_num, ", ".join(periods))): - init_matrices = _m.Modeller().tool("sandag.initialize.initialize_matrices") - create_function = _m.Modeller().tool("inro.emme.data.function.create_function") - src_emmebank = src_scenarios[0].emmebank - remote_db_dir = _join(self._path, "emme_project", "Database_remote" + str(remote_num)) - if msa_iteration == 1: - # Create and initialize database at first iteration, overwrite existing - if os.path.exists(remote_db_dir): - _shutil.rmtree(remote_db_dir) - _time.sleep(1) - os.mkdir(remote_db_dir) - dimensions = src_emmebank.dimensions - dimensions["scenarios"] = len(src_scenarios) - remote_emmebank = _eb.create(_join(remote_db_dir, "emmebank"), dimensions) - try: - remote_emmebank.title = src_emmebank.title - remote_emmebank.coord_unit_length = src_emmebank.coord_unit_length - remote_emmebank.unit_of_length = src_emmebank.unit_of_length - remote_emmebank.unit_of_cost = src_emmebank.unit_of_cost - remote_emmebank.unit_of_energy = src_emmebank.unit_of_energy - remote_emmebank.use_engineering_notation = src_emmebank.use_engineering_notation - remote_emmebank.node_number_digits = src_emmebank.node_number_digits - - for src_scen in src_scenarios: - remote_scen = remote_emmebank.create_scenario(src_scen.id) - remote_scen.title = src_scen.title - for attr in sorted(src_scen.extra_attributes(), key=lambda x: x._id): - dst_attr = remote_scen.create_extra_attribute( - attr.type, attr.name, attr.default_value) - dst_attr.description = attr.description - for field in src_scen.network_fields(): - remote_scen.create_network_field( - field.type, field.name, field.atype, field.description) - remote_scen.has_traffic_results = src_scen.has_traffic_results - remote_scen.has_transit_results = src_scen.has_transit_results - remote_scen.publish_network(src_scen.get_network()) - for function in src_emmebank.functions(): - create_function(function.id, function.expression, remote_emmebank) - init_matrices(["traffic_skims", "traffic_demand"], periods, remote_scen) - finally: - remote_emmebank.dispose() - - src_scen = src_scenarios[0] - with _m.logbook_trace("Copy demand matrices to remote database"): - with _eb.Emmebank(_join(remote_db_dir, "emmebank")) as remote_emmebank: - demand_matrices = init_matrices.get_matrix_names("traffic_demand", periods, src_scen) - for matrix_name in demand_matrices: - matrix = remote_emmebank.matrix(matrix_name) - src_matrix = src_emmebank.matrix(matrix_name) - if matrix.type == "SCALAR": - matrix.data = src_matrix.data - else: - matrix.set_data(src_matrix.get_data(src_scen.id), src_scen.id) - skim_matrices = init_matrices.get_matrix_names("traffic_skims", periods, src_scen) - return remote_db_dir, skim_matrices - - def start_assignments(self, machine, database_path, periods, scenarios, assign_args): - with _m.logbook_trace("Start remote process for traffic assignments %s" % (", ".join(periods))): - assign_args["database_path"] = database_path - end_path = _join(database_path, "finish") - if os.path.exists(end_path): - os.remove(end_path) - for period in periods: - assign_args["period_scenario"] = scenarios[period].id - assign_args["period"] = period - with open(_join(database_path, "start_%s.args" % period), 'w') as f: - _json.dump(assign_args, f, indent=4) - script_dir = _join(self._path, "python") - bin_dir = _join(self._path, "bin") - args = [ - 'start %s\\PsExec.exe' % bin_dir, - '-c', - '-f', - '\\\\%s' % machine, - '-u \%s' % self.username, - '-p %s' % self.password, - "-d", - '%s\\emme_python.bat' % bin_dir, - "T:", - self._path, - '%s\\remote_run_traffic.py' % script_dir, - database_path, - ] - command = " ".join(args) - p = _subprocess.Popen(command, shell=True) - - @_m.logbook_trace("Wait for remote assignments to complete and copy results") - def wait_and_copy(self, database_dirs, scenarios, matrices): - database_dirs = database_dirs[:] - wait = True - while wait: - _time.sleep(5) - for path in database_dirs: - end_path = _join(path, "finish") - if os.path.exists(end_path): - database_dirs.remove(path) - _time.sleep(2) - self.check_for_fatal( - end_path, "error during remote run of traffic assignment. " - "Check logFiles/traffic_assign_database_remote*.log") - self.copy_results(path, scenarios[path], matrices[path]) - if not database_dirs: - wait = False - - @_m.logbook_trace("Copy skim results from remote database", save_arguments=True) - def copy_results(self, database_path, scenarios, matrices): - with _eb.Emmebank(_join(database_path, "emmebank")) as remote_emmebank: - for dst_scen in scenarios: - remote_scen = remote_emmebank.scenario(dst_scen.id) - # Create extra attributes and network fields which do not exist - for attr in sorted(remote_scen.extra_attributes(), key=lambda x: x._id): - if not dst_scen.extra_attribute(attr.name): - dst_attr = dst_scen.create_extra_attribute( - attr.type, attr.name, attr.default_value) - dst_attr.description = attr.description - for field in remote_scen.network_fields(): - if not dst_scen.network_field(field.type, field.name): - dst_scen.create_network_field( - field.type, field.name, field.atype, field.description) - dst_scen.has_traffic_results = remote_scen.has_traffic_results - dst_scen.has_transit_results = remote_scen.has_transit_results - - dst_scen.publish_network(remote_scen.get_network()) - - dst_emmebank = dst_scen.emmebank - scen_id = dst_scen.id - for matrix_id in matrices: - src_matrix = remote_emmebank.matrix(matrix_id) - dst_matrix = dst_emmebank.matrix(matrix_id) - dst_matrix.set_data(src_matrix.get_data(scen_id), scen_id) - - @_m.method(return_type=unicode) - def get_link_attributes(self): - export_utils = _m.Modeller().module("inro.emme.utility.export_utilities") - return export_utils.get_link_attributes(_m.Modeller().scenario) - - def update_centroid_connectors(self, source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file): - adjust_network = _m.Modeller().module("inro.import.adjust_network_links") - return adjust_network.adjust_network_links(source, base_scenario, emmebank, external_zone, taz_cwk_file, cluster_zone_file) - - def sql_select_scenario(self, year, iteration, sample, path, dbtime): # YMA, 1/24/2019 - """Return scenario_id from [dimension].[scenario] given path""" - - import datetime - - sql_con = pyodbc.connect(driver='{SQL Server}', - server='sql2014a8', - database='abm_2', - trusted_connection='yes') - - # dbtime = dbtime + datetime.timedelta(days=0) - - df = pd.read_sql_query( - sql=("SELECT [scenario_id] " - "FROM [dimension].[scenario]" - "WHERE [year] = ? AND [iteration] = ? AND [sample_rate]= ? AND [path] Like ('%' + ? + '%') AND [date_loaded] > ? "), - con=sql_con, - params=[year, iteration, sample, path, dbtime] - ) - - if len(df) > 0: - return (df.iloc[len(df) - 1]['scenario_id']) - else: - return 0 - - def sql_check_load_request(self, year, path, user, ldtime): # YMA, 1/24/2019 - """Return information from [data_load].[load_request] given path,username,and requested time""" - - import datetime - - t0 = ldtime + datetime.timedelta(minutes=-1) - t1 = t0 + datetime.timedelta(minutes=30) - - sql_con = pyodbc.connect(driver='{SQL Server}', - server='sql2014a8', - database='abm_2', - trusted_connection='yes') - - df = pd.read_sql_query( - sql=( - "SELECT [load_request_id],[year],[name],[path],[iteration],[sample_rate],[abm_version],[user_name],[date_requested],[loading],[loading_failed],[scenario_id] " - "FROM [data_load].[load_request] " - "WHERE [year] = ? AND [path] LIKE ('%' + ? + '%') AND [user_name] LIKE ('%' + ? + '%') AND [date_requested] >= ? AND [date_requested] <= ? "), - con=sql_con, - params=[year, path, user, t0, t1] - ) - - if len(df) > 0: - return "You have successfully made the loading request, but the loading to the database failed. \r\nThe information is below. \r\n\r\n" + df.to_string() - else: - return "The data load request was not successfully made, please double check the [data_load].[load_request] table to confirm." - - -''' - def send_notification(self,str_message,user): # YMA, 1/24/2019, not working on server - """automate to send email notification if load request or loading failed""" - - import win32com.client as win32 - - outlook = win32.Dispatch('outlook.application') - Msg = outlook.CreateItem(0) - Msg.To = user + '@sandag.org' - Msg.CC = 'yma@sandag.org' - Msg.Subject = 'Loading Scenario to Database Failed' - Msg.body = str_message + '\r\n' + '\r\n' + 'This email alert is auto generated.\r\n' + 'Please do not respond.\r\n' - Msg.send''' diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py deleted file mode 100644 index c271e56..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/distribution.py +++ /dev/null @@ -1,197 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/commercial_vehicle/distribution.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Distributes the total daily trips from production and attraction vectors. -# Friction factors are calculated based on a blended travel time skim of -# 1/3 AM_SOV_NT_M_TIME and 2/3 MD_SOV_NT_M_TIME, and a table of friction factor -# lookup values from commVehFF.csv. -# -# Inputs: -# input_directory: source directory for input files -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# input/commVehFF.csv -# -# Matrix inputs: -# moCOMVEH_PROD, mdCOMVEH_ATTR -# mfAM_SOV_NT_M_TIME, mfMD_SOV_NT_M_TIME -# -# Matrix intermediates (only used internally): -# mfCOMVEH_BLENDED_SKIM, mfCOMVEH_FRICTION -# -# Matrix results: -# mfCOMVEH_TOTAL_DEMAND -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(project_dir, "input") - base_scenario = modeller.scenario - distribution = modeller.tool("sandag.model.commercial_vehicle.distribution") - distribution(input_dir, base_scenario) -""" - - -TOOLBOX_ORDER = 53 - - -import inro.modeller as _m -import traceback as _traceback - -import pandas as pd -import os -import numpy as np - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class CommercialVehicleDistribution(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.attributes = ["input_directory"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Commercial Vehicle Distribution" - pb.description = """ -
    - Calculates total daily trips. - The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. -
    - A simple gravity model is used to distribute the truck trips. - A blended travel time of - 1/3 AM_SOV_NT_M_TIME and 2/3 MD_SOV_NT_M_TIME is used, along with - friction factor lookup table stored in commVehFF.csv. -
    - Input: -
    • - (1) Level-of-service matrices for the AM peak period (6 am to 10 am) 'mfAM_SOVGPM_TIME' - and midday period (10 am to 3 pm) 'mfMD_SOVGPM_TIME' - which contain congested travel time (in minutes). -
    • - (2) Trip generation results 'moCOMVEH_PROD' and 'mdCOMVEH_ATTR' -
    • - (4) A table of friction factors in commVehFF.csv with: -
      • - (a) impedance measure (blended travel time) index; -
      • - (b) friction factors -
      -
    - Output: -
    • - (1) A trip table matrix 'mfCOMVEH_TOTAL_DEMAND' - of daily class-specific truck trips for very small trucks. -
    • - (2) A blended travel time matrix 'mfCOMVEH_BLENDED_SKIM' -
    -
    """ - pb.branding_text = "- SANDAG - Model - Commercial vehicle" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Commercial vehicle distribution') - def __call__(self, input_directory, scenario): - attributes = {"input_directory": input_directory} - gen_utils.log_snapshot("Commercial vehicle distribution", str(self), attributes) - self.calc_blended_skims(scenario) - self.calc_friction_matrix(input_directory, scenario) - self.balance_matrix(scenario) - - @_m.logbook_trace('Calculate blended skims') - def calc_blended_skims(self, scenario): - matrix_calc = _m.Modeller().tool( - 'inro.emme.matrix_calculation.matrix_calculator') - spec = { - "expression": "0.3333 * mfAM_SOV_NT_M_TIME + 0.6666 * mfMD_SOV_NT_M_TIME", - "result": "mfCOMVEH_BLENDED_SKIM", - "type": "MATRIX_CALCULATION" - } - matrix_calc(spec, scenario=scenario) - - # Prevent intrazonal trips - spec = { - "expression": "99999 * (p.eq.q) + mfCOMVEH_BLENDED_SKIM * (p.ne.q)", - "result": "mfCOMVEH_BLENDED_SKIM", - "type": "MATRIX_CALCULATION" - } - matrix_calc(spec, scenario=scenario) - - @_m.logbook_trace('Calculate friction factor matrix') - def calc_friction_matrix(self, input_directory, scenario): - emmebank = scenario.emmebank - - imp_matrix = emmebank.matrix('mfCOMVEH_BLENDED_SKIM') - friction_matrix = emmebank.matrix('mfCOMVEH_FRICTION') - imp_array = imp_matrix.get_numpy_data(scenario_id=scenario.id) - - # create the vector function to bin the impedances and get friction values - friction_table = pd.read_csv( - os.path.join(input_directory, 'commVehFF.csv'), - header=None, names=['index', 'factors']) - - factors_array = friction_table.factors.values - max_index = len(factors_array) - 1 - - # interpolation: floor - values_array = np.clip(np.floor(imp_array).astype(int) - 1, 0, max_index) - friction_array = np.take(factors_array, values_array) - friction_matrix.set_numpy_data(friction_array, scenario_id=scenario.id) - - def balance_matrix(self, scenario): - balance = _m.Modeller().tool( - 'inro.emme.matrix_calculation.matrix_balancing') - spec = { - "type": "MATRIX_BALANCING", - "od_values_to_balance": "mfCOMVEH_FRICTION", - "origin_totals": "moCOMVEH_PROD", - "destination_totals": "mdCOMVEH_ATTR", - "results": { - "od_balanced_values": "mfCOMVEH_TOTAL_DEMAND", - }, - "max_iterations": 100, - "max_relative_error": 0.001 - } - balance(spec, scenario=scenario) diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py deleted file mode 100644 index 9842de1..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/generation.py +++ /dev/null @@ -1,187 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/commercial_vehicle/generation.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the commercial vehicle generation step, calculate commercial vehicle -# productions and attractions. Linear regression models generate trip ends, -# balancing attractions to productions. -# -# Inputs: -# input_directory: source directory for most input files, including demographics and trip rates -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file -# input/mgra13_based_inputYEAR.csv -# -# Matrix results: -# moCOMVEH_PROD, mdCOMVEH_ATTR -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(project_dir, "input") - base_scenario = modeller.scenario - generation = modeller.tool("sandag.model.commercial_vehicle.generation") - generation(input_dir, base_scenario) -""" - - -TOOLBOX_ORDER = 52 - - -import inro.modeller as _m -import traceback as _traceback - -import pandas as pd -import os - - -dem_utils = _m.Modeller().module('sandag.utilities.demand') -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class CommercialVehicleDistribution(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.attributes = ["input_directory"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Commercial Vehicle Generation" - pb.description = """ -
    - Calculate commerical vehicle productions and attractions - based on mgra13_based_inputYYYY.csv. - The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. - - Linear regression models generate trip ends, balancing attractions to productions. -
    - Input: MGRA file in CSV format with the following fields: -
      -
    • - (a) TOTEMP, total employment (same regardless of classification system); -
    • - (b) RETEMPN, retail trade employment per the NAICS classification system; -
    • - (c) FPSEMPN, financial and professional services employment per the NAICS classification system; -
    • - (d) HEREMPN, health, educational, and recreational employment per the NAICS classification system; -
    • - (e) OTHEMPN, other employment per the NAICS classification system; -
    • - (f) AGREMPN, agricultural employmentper the NAICS classificatin system; -
    • - (g) MWTEMPN, manufacturing, warehousing, and transportation emp;loyment per the NAICS classification system; and, -
    • - (h) TOTHH, total households. -
    -
    - Output: Trip productions and attractions in matrices 'moCOMMVEH_PROD' and 'mdCOMMVEH_ATTR' respectively. -
    - """ - - pb.branding_text = "- SANDAG - Model - Commercial vehicle" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Commercial vehicle generation') - def __call__(self, input_directory, scenario): - attributes = {"input_directory": input_directory} - gen_utils.log_snapshot("Commercial vehicle generation", str(self), attributes) - emmebank = scenario.emmebank - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - main_directory = os.path.dirname(input_directory) - props = load_properties( - os.path.join(main_directory, "conf", "sandag_abm.properties")) - year = props['scenarioYear'] - mgra = pd.read_csv( - os.path.join(input_directory, 'mgra13_based_input%s.csv' % year)) - - calibration = 1.4 - - mgra['RETEMPN'] = mgra.emp_retail + mgra.emp_personal_svcs_retail - mgra['FPSEMPN'] = mgra.emp_prof_bus_svcs - mgra['HEREMPN'] = mgra.emp_health + mgra.emp_pvt_ed_k12 \ - + mgra.emp_pvt_ed_post_k12_oth + mgra.emp_amusement - mgra['AGREMPN'] = mgra.emp_ag - mgra['MWTEMPN'] = mgra.emp_const_non_bldg_prod \ - + mgra.emp_const_bldg_prod + mgra.emp_mfg_prod \ - + mgra.emp_trans - mgra['MILITARY'] = mgra.emp_fed_mil - mgra['TOTEMP'] = mgra.emp_total - mgra['OTHEMPN'] = mgra.TOTEMP - (mgra.RETEMPN + mgra.FPSEMPN - + mgra.HEREMPN + mgra.AGREMPN - + mgra.MWTEMPN + mgra.MILITARY) - mgra['TOTHH'] = mgra.hh - - mgra['PROD'] = calibration * ( - 0.95409 * mgra.RETEMPN + 0.54333 * mgra.FPSEMPN - + 0.50769 * mgra.HEREMPN + 0.63558 * mgra.OTHEMPN - + 1.10181 * mgra.AGREMPN + 0.81576 * mgra.MWTEMPN - + 0.15000 * mgra.MILITARY + 0.1 * mgra.TOTHH) - mgra['ATTR'] = mgra.PROD - - # Adjustment to match military CTM trips to match gate counts - military_ctm_adjustment = props["RunModel.militaryCtmAdjustment"] - if military_ctm_adjustment: - mgra_m = pd.read_csv(os.path.join( - input_directory, 'cvm_military_adjustment.csv')) - mgra = pd.merge(mgra, mgra_m, on='mgra', how='outer') - mgra.fillna(1, inplace=True) - mgra['PROD'] = mgra['PROD'] * mgra['scale'] - mgra['ATTR'] = mgra['ATTR'] * mgra['scale'] - - taz = mgra.groupby('taz').sum() - taz.reset_index(inplace=True) - taz = dem_utils.add_missing_zones(taz, scenario) - taz.sort('taz', inplace=True) - - prod = emmebank.matrix('moCOMVEH_PROD') - attr = emmebank.matrix('mdCOMVEH_ATTR') - prod.set_numpy_data(taz.PROD.values, scenario) - attr.set_numpy_data(taz.ATTR.values, scenario) - - return taz[['taz', 'PROD', 'ATTR']] diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py deleted file mode 100644 index 3899456..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/run_commercial_vehicle_model.py +++ /dev/null @@ -1,122 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/commercial_vehicle/run_commercial_vehicle_model.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the commercial vehicle model, the generation, distribution, time-of-day, -# and toll diversion tools, in sequence. -# 1) Generates -# 2) Generates -# 3) Distributes trips based on congested skims -# 4) Time-of-day split -# 5) Applies toll diversion model with toll and non-toll SOV skims -# -# The very small truck generation model is based on the Phoenix -# four-tire truck model documented in the TMIP Quick Response Freight Manual. -# -# Inputs: -# run_generation: boolean, if True run generation tool. -# input_directory: source directory for most input files, including demographics and trip rates -# (see generation and distribtuion tools) -# scenario: traffic scenario to use for reference zone system -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(project_dir, "input") - base_scenario = modeller.scenario - run_truck = modeller.tool("sandag.model.commercial_vehicle.run_commercial_vehicle") - run_truck(True, input_dir, base_scenario) -""" - -TOOLBOX_ORDER = 51 - - -import inro.modeller as _m -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class CommercialVehicleModel(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - run_generation = _m.Attribute(bool) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.run_generation = True - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.attributes = ["input_directory", "run_generation"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Commercial vehicle model" - pb.description = """ -
    - Run the 4 steps of the commercial vehicle model: generation, distribution, - time of day, toll diversion. - - The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. -
    -""" - pb.branding_text = "- SANDAG - Model - Commercial vehicle" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_checkbox("run_generation", title=" ", label="Run generation (first iteration)") - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.run_generation, self.input_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Commercial vehicle model', save_arguments=True) - def __call__(self, run_generation, input_directory, scenario): - attributes = {"run_generation": run_generation, "input_directory": input_directory} - gen_utils.log_snapshot("Commercial vehicle model", str(self), attributes) - generation = _m.Modeller().tool( - 'sandag.model.commercial_vehicle.generation') - distribution = _m.Modeller().tool( - 'sandag.model.commercial_vehicle.distribution') - time_of_day = _m.Modeller().tool( - 'sandag.model.commercial_vehicle.time_of_day') - diversion = _m.Modeller().tool( - 'sandag.model.commercial_vehicle.toll_diversion') - if run_generation: - generation(input_directory, scenario) - distribution(input_directory, scenario) - time_of_day(scenario) - diversion(scenario) diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py deleted file mode 100644 index 163bba7..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/time_of_day.py +++ /dev/null @@ -1,99 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/commercial_vehicle/time_of_day.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Applies time-of-day factoring to the Commercial vehicle total daily demand. -# The diurnal factors are taken from the BAYCAST-90 model with adjustments -# made during calibration to the very small truck values to better match counts. -# -# Inputs: -# scenario: traffic scenario to use for reference zone system -# -# Matrix inputs: -# mfCOMVEH_TOTAL_DEMAND -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# mfpp_COMVEH -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - base_scenario = modeller.scenario - time_of_day = modeller.tool("sandag.model.commercial_vehicle.time_of_day") - time_of_day(base_scenario) -""" - -TOOLBOX_ORDER = 54 - - -import inro.modeller as _m -import traceback as _traceback - - -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class TimeOfDay(_m.Tool()): - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Commercial Vehicle Time of Day split" - pb.description = """ -
    - Commercial vehicle time-of-day factoring. - The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. -
    - The diurnal factors are taken from the BAYCAST-90 model with adjustments - made during calibration to the very small truck values to better match counts. -

    Input: A production/attraction format trip table matrix of daily very small truck trips.

    -

    Output: Five, time-of-day-specific trip table matrices for very small trucks, - of the form 'mfpp_COMVEH'. -

    -
    """ - pb.branding_text = "- SANDAG - Model - Commercial vehicle" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Commercial vehicle Time of Day split') - def __call__(self, scenario): - matrix_calc = dem_utils.MatrixCalculator(scenario, 0) - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - period_factors = [0.0235, 0.1, 0.5080, 0.1980, 0.1705] - for p, f in zip(periods, period_factors): - matrix_calc.add( - "mf%s_COMVEH" % p, - "%s * 0.5 * (mfCOMVEH_TOTAL_DEMAND + mfCOMVEH_TOTAL_DEMAND')" % f) - matrix_calc.run() diff --git a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py b/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py deleted file mode 100644 index 07bc4fe..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/commercial_vehicle/toll_diversion.py +++ /dev/null @@ -1,119 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/commercial_vehicle/toll_diversion.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# Applies toll and non-toll split to Commercial vehicle time period demand. -# Uses the travel TIME for GP and TOLL modes as well as the TOLLCOST -# by time period. -# -# Inputs: -# scenario: traffic scenario to use for reference zone system -# -# Matrix inputs: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# mfpp_COMVEH -# mfpp_SOVGPM_TIME, mfpp_SOVTOLLM_TIME, mfpp_SOVTOLLM_TOLLCOST -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# mfpp_COMVEHGP, mfpp_COMVEHTOLL -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - base_scenario = modeller.scenario - toll_diversion = modeller.tool("sandag.model.commercial_vehicle.toll_diversion") - toll_diversion(base_scenario) -""" - -TOOLBOX_ORDER = 55 - - -import inro.modeller as _m -import traceback as _traceback - - -gen_utils = _m.Modeller().module('sandag.utilities.general') -dem_utils = _m.Modeller().module('sandag.utilities.demand') - - -class TollDiversion(_m.Tool()): - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Commercial vehicle toll diversion" - pb.description = """ -
    - Commercial vehicle toll and non-toll (GP) split. - The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. -
    -

    Input: Time-of-day-specific trip table matrices 'mfpp_COMVEH', - and travel time for GP and TOLL modes 'mfpp_SOVGPM_TIME', 'mfpp_SOVTOLLM_TIME', - and toll cost 'mfpp_SOVTOLLM_TOLLCOST' (medium VOT bin). -

    -

    Output: Corresponding time-of-day 'mfpp_COMVEHGP' and 'mfpp_COMVEHTOLL' - trip demand matrices.

    -
    -""" - pb.branding_text = "- SANDAG - Model - Commercial vehicle" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Commercial vehicle toll diversion') - def __call__(self, scenario): - emmebank = scenario.emmebank - matrix_calc = dem_utils.MatrixCalculator(scenario, "MAX-1") - init_matrix = _m.Modeller().tool( - "inro.emme.data.matrix.init_matrix") - - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - for p in periods: - init_matrix("mf%s_COMVEHTOLL" % p, scenario=scenario) - - nest = 10 - vot = 0.02 - toll_factor = 1 - for p in periods: - with matrix_calc.trace_run("Diversion for %s" % p): - init_matrix("mf%s_COMVEHTOLL" % p, scenario=scenario) - params = {'p': p, 'v': vot, 'tf': toll_factor, 'n': nest} - utility = ('(mf%(p)s_SOVGPM_TIME - mf%(p)s_SOVTOLLM_TIME' - '- %(v)s * mf%(p)s_SOVTOLLM_TOLLCOST * %(tf)s) / %(n)s') % params - matrix_calc.add( - "mf%s_COMVEHTOLL" % p, - "mf%s_COMVEH / (1 + exp(- %s))" % (p, utility), - ["mf%s_SOVTOLLM_TOLLCOST" % p, 0, 0, "EXCLUDE"]) - matrix_calc.add( - "mf%s_COMVEHGP" % p, "mf%(p)s_COMVEH - mf%(p)s_COMVEHTOLL" % {'p': p}) diff --git a/sandag_abm/src/main/emme/toolbox/model/external_external.py b/sandag_abm/src/main/emme/toolbox/model/external_external.py deleted file mode 100644 index c928d36..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/external_external.py +++ /dev/null @@ -1,190 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/external_external.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the external-external, cross-regional demand model. Imports the total -# daily demand from file and splits by time-of-day and SOVGP, HOV2HOV, and -# HOV3HOV classes using fixed factors. -# -# -# Inputs: -# input_directory: source directory for input file -# external_zones: the set of external zones specified as a range, default is "1-12" -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: YEAR is replaced by scenarioYear in the conf/sandag_abm.properties file -# input/mgra13_based_inputYEAR.csv -# input/externalInternalControlTotalsByYear.csv -# input/externalInternalControlTotals.csv -# (if externalInternalControlTotalsByYear.csv is unavailable) -# -# Matrix results: -# pp_SOV_EETRIPS, pp_HOV2_EETRIPS, pp_HOV3_EETRIPS -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(main_directory, "input") - external_zones = "1-12" - base_scenario = modeller.scenario - external_external = modeller.tool("sandag.model.external_external") - external_external(input_dir, external_zones, base_scenario) -""" - - -TOOLBOX_ORDER = 62 - - -import inro.modeller as _m - -import multiprocessing as _multiprocessing -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class ExternalExternal(_m.Tool(), gen_utils.Snapshot): - input_directory = _m.Attribute(unicode) - external_zones = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.external_zones = "1-12" - self.attributes = ["external_zones", "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "External external model" - pb.description = """ - Total trips are read from externalExternalTripsByYear.csv for - the year in sandag_abm.properties. If this file does not exist - externalExternalTrips.csv will be used instead. - The total trips are split by time-of-day and traffic class - SOVGP, HOV2HOV, and HOV3HOV using fixed factors. - """ - pb.branding_text = "- SANDAG - Model" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - pb.add_text_box("external_zones", title="External zones:") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, self.external_zones, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('External-external model', save_arguments=True) - def __call__(self, input_directory, external_zones, scenario): - attributes = { - "external_zones": external_zones, - "input_directory": input_directory, - } - gen_utils.log_snapshot("External-external model", str(self), attributes) - emmebank = scenario.emmebank - matrix_calc = _m.Modeller().tool( - "inro.emme.matrix_calculation.matrix_calculator") - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties( - os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) - year = int(props['scenarioYear']) - - periods = ["EA", "AM", "MD", "PM", "EV"] - time_of_day_factors = [0.074, 0.137, 0.472, 0.183, 0.133] - modes = ["SOV", "HOV2", "HOV3"] - mode_factors = [0.43, 0.42, 0.15] - - ee_matrix = emmebank.matrix("ALL_TOTAL_EETRIPS") - matrix_data = ee_matrix.get_data(scenario) - file_path = os.path.join( - input_directory, "externalExternalTripsByYear.csv") - if os.path.isfile(file_path): - with open(file_path, 'r') as f: - header = f.readline() - for line in f: - tyear, orig, dest, trips = line.split(",") - if int(tyear) == year: - matrix_data.set(int(orig), int(dest), float(trips)) - else: - file_path = os.path.join( - input_directory, "externalExternalTrips.csv") - if not os.path.isfile(file_path): - raise Exception("External-external model: no file 'externalExternalTrips.csv' or 'externalExternalTripsByYear.csv'") - with open(file_path, 'r') as f: - header = f.readline() - for line in f: - orig, dest, trips = line.split(",") - matrix_data.set(int(orig), int(dest), float(trips)) - _m.logbook_write("Control totals read from %s" % file_path) - ee_matrix.set_data(matrix_data, scenario) - - # factor for final demand matrix by time and mode type - # all external-external trips are non-toll - # SOV_GP, SR2_HOV SR3_HOV = "SOV", "HOV2", "HOV3" - for period, tod_fac in zip(periods, time_of_day_factors): - for mode, mode_fac in zip(modes, mode_factors): - spec = { - "expression": "ALL_TOTAL_EETRIPS * %s * %s" % (tod_fac, mode_fac), - "result": "mf%s_%s_EETRIPS" % (period, mode), - "constraint": { - "by_zone": { - "origins": external_zones, - "destinations": external_zones - } - }, - "type": "MATRIX_CALCULATION" - } - matrix_calc(spec, scenario=scenario) - - precision = float(props['RunModel.MatrixPrecision']) - self.matrix_rounding(scenario, precision) - - @_m.logbook_trace('Controlled rounding of demand') - def matrix_rounding(self, scenario, precision): - round_matrix = _m.Modeller().tool( - "inro.emme.matrix_calculation.matrix_controlled_rounding") - emmebank = scenario.emmebank - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - modes = ["SOV", "HOV2", "HOV3"] - for period in periods: - for mode in modes: - matrix = emmebank.matrix("mf%s_%s_EETRIPS" % (period, mode)) - report = round_matrix(demand_to_round=matrix, - rounded_demand=matrix, - min_demand=precision, - values_to_round="SMALLER_THAN_MIN", - scenario=scenario) - diff --git a/sandag_abm/src/main/emme/toolbox/model/external_internal.py b/sandag_abm/src/main/emme/toolbox/model/external_internal.py deleted file mode 100644 index 19203e6..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/external_internal.py +++ /dev/null @@ -1,331 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/external_internal.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the external USA to internal demand model. -# 1) Work and non-work trip gateway total trips are read from control totals -# 2) Generates internal trip ends based on relative attractiveness from employment (by category) and households -# 3) Applies time-of-day and occupancy factors -# 4) Applies toll diversion model with toll and non-toll skims -# Control totals are read from externalInternalControlTotalsByYear.csv for -# the specified year in sandag_abm.properties. If this file does not exist -# externalInternalControlTotals.csv will be used instead. -# -# Inputs: -# input_directory: source directory for most input files, including demographics and trip rates -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: YEAR is replaced by scenarioYear in the conf/sandag_abm.properties file -# input/mgra13_based_inputYEAR.csv -# input/externalInternalControlTotalsByYear.csv -# input/externalInternalControlTotals.csv -# (if externalInternalControlTotalsByYear.csv is unavailable) -# -# Matrix results: -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(main_directory, "input") - base_scenario = modeller.scenario - external_internal = modeller.tool("sandag.model.external_internal") - external_internal(input_dir, input_truck_dir, base_scenario) -""" - -TOOLBOX_ORDER = 61 - - -import inro.modeller as _m -import numpy as np -import pandas as pd -import traceback as _traceback -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -class ExternalInternal(_m.Tool(), gen_utils.Snapshot): - input_directory = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.attributes = ["input_directory"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "External internal model" - pb.description = """ - Runs the external USA to internal demand model. - Control totals are read from externalInternalControlTotalsByYear.csv for - the specified year in sandag_abm.properties. If this file does not exist - externalInternalControlTotals.csv will be used instead.""" - pb.branding_text = "- SANDAG - Model" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('External-internal model', save_arguments=True) - def __call__(self, input_directory, scenario): - attributes = {"input_directory": input_directory} - gen_utils.log_snapshot("External-internal model", str(self), attributes) - np.seterr(divide='ignore', invalid='ignore') - - emmebank = scenario.emmebank - zones = scenario.zone_numbers - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties( - os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) - - year = int(props['scenarioYear']) - mgra = pd.read_csv( - os.path.join(input_directory, 'mgra13_based_input%s.csv' % year)) - - # Load data - file_path = os.path.join( - input_directory, "externalInternalControlTotalsByYear.csv") - if os.path.isfile(file_path): - control_totals = pd.read_csv(file_path) - control_totals = control_totals[control_totals.year==year] - control_totals = control_totals.drop("year", axis=1) - else: - file_path = os.path.join( - input_directory, 'externalInternalControlTotals.csv') - if not os.path.isfile(file_path): - raise Exception( - "External-internal model: no file 'externalInternalControlTotals.csv' " - "or 'externalInternalControlTotalsByYear.csv'") - control_totals = pd.read_csv(file_path) - _m.logbook_write("Control totals read from %s" % file_path) - - # Aggregate purposes - mgra['emp_blu'] = (mgra.emp_const_non_bldg_prod - + mgra.emp_const_non_bldg_office - + mgra.emp_utilities_prod - + mgra.emp_utilities_office - + mgra.emp_const_bldg_prod - + mgra.emp_const_bldg_office - + mgra.emp_mfg_prod - + mgra.emp_mfg_office - + mgra.emp_whsle_whs - + mgra.emp_trans) - - mgra['emp_svc'] = (mgra.emp_prof_bus_svcs - + mgra.emp_prof_bus_svcs_bldg_maint - + mgra.emp_personal_svcs_office - + mgra.emp_personal_svcs_retail) - - mgra['emp_edu'] = (mgra.emp_pvt_ed_k12 - + mgra.emp_pvt_ed_post_k12_oth - + mgra.emp_public_ed) - - mgra['emp_gov'] = (mgra.emp_state_local_gov_ent - + mgra.emp_fed_non_mil - + mgra.emp_fed_non_mil - + mgra.emp_state_local_gov_blue - + mgra.emp_state_local_gov_white) - - mgra['emp_ent'] = (mgra.emp_amusement - + mgra.emp_hotel - + mgra.emp_restaurant_bar) - - mgra['emp_oth'] = (mgra.emp_religious - + mgra.emp_pvt_hh - + mgra.emp_fed_mil) - - mgra['work_size'] = (mgra.emp_blu + - 1.364 * mgra.emp_retail + - 4.264 * mgra.emp_ent + - 0.781 * mgra.emp_svc + - 1.403 * mgra.emp_edu + - 1.779 * mgra.emp_health + - 0.819 * mgra.emp_gov + - 0.708 * mgra.emp_oth) - - mgra['non_work_size'] = (mgra.hh + - 1.069 * mgra.emp_blu + - 4.001 * mgra.emp_retail + - 6.274 * mgra.emp_ent + - 0.901 * mgra.emp_svc + - 1.129 * mgra.emp_edu + - 2.754 * mgra.emp_health + - 1.407 * mgra.emp_gov + - 0.304 * mgra.emp_oth) - - # aggregate to TAZ - taz = mgra[['taz', 'work_size', 'non_work_size']].groupby('taz').sum() - taz.reset_index(inplace=True) - taz = dem_utils.add_missing_zones(taz, scenario) - taz.sort_values('taz', ascending=True, inplace=True) # method sort was deprecated since pandas version 0.20.0, yma, 2/12/2019 - taz.reset_index(inplace=True, drop=True) - control_totals = pd.merge(control_totals, taz[['taz']], how='outer') - control_totals.sort_values('taz', inplace=True) # method sort was deprecated since pandas version 0.20.0, yma, 2/12/2019 - - length_skim = emmebank.matrix('mf"MD_SOV_TR_M_DIST"').get_numpy_data(scenario) - - # Compute probabilities for work purpose - wrk_dist_coef = -0.029 - wrk_prob = taz.work_size.values * np.exp(wrk_dist_coef * length_skim) - wrk_sum = np.sum(wrk_prob, 1) - wrk_prob = wrk_prob / wrk_sum[:, np.newaxis] - wrk_prob = np.nan_to_num(wrk_prob) - # Apply probabilities to control totals - wrk_pa_mtx = wrk_prob * control_totals.work.values[:, np.newaxis] - wrk_pa_mtx = np.nan_to_num(wrk_pa_mtx) - wrk_pa_mtx = wrk_pa_mtx.astype("float32") - - # compute probabilities for non work purpose - non_wrk_dist_coef = -0.006 - nwrk_prob = taz.non_work_size.values * np.exp(non_wrk_dist_coef * length_skim) - non_wrk_sum = np.sum(nwrk_prob, 1) - nwrk_prob = nwrk_prob / non_wrk_sum[:, np.newaxis] - nwrk_prob = np.nan_to_num(nwrk_prob) - # Apply probabilities to control totals - nwrk_pa_mtx = nwrk_prob * control_totals.nonwork.values[:, np.newaxis] - nwrk_pa_mtx = np.nan_to_num(nwrk_pa_mtx) - nwrk_pa_mtx = nwrk_pa_mtx.astype("float32") - - # Convert PA to OD and apply Diurnal Facotrs - wrk_ap_mtx = 0.5 * np.transpose(wrk_pa_mtx) - wrk_pa_mtx = 0.5 * wrk_pa_mtx - nwrk_ap_mtx = 0.5 * np.transpose(nwrk_pa_mtx) - nwrk_pa_mtx = 0.5 * nwrk_pa_mtx - - # Apply occupancy and diurnal factors - work_time_PA_factors = [0.26, 0.26, 0.41, 0.06, 0.02] - work_time_AP_factors = [0.08, 0.07, 0.41, 0.42, 0.02] - - nonwork_time_PA_factors = [0.25, 0.39, 0.30, 0.04, 0.02] - nonwork_time_AP_factors = [0.12, 0.11, 0.37, 0.38, 0.02] - - work_occupancy_factors = [0.58, 0.31, 0.11] - nonwork_occupancy_factors = [0.55, 0.29, 0.15] - - # value of time is in cents per minute (toll cost is in cents) - vot_work = 15.00 # $9.00/hr - vot_non_work = 22.86 # $13.70/hr - ivt_coef = -0.03 - - gp_modes = ["SOVGP", "HOV2HOV", "HOV3HOV"] - toll_modes = ["SOVTOLL", "HOV2TOLL", "HOV3TOLL"] - # TODO: the GP vs. TOLL distinction should be collapsed - # (all demand added to transponder demand in import_auto_demand) - skim_lookup = { - "SOVGP": "SOV_NT_M", - "HOV2HOV": "HOV2_M", - "HOV3HOV": "HOV3_M", - "SOVTOLL": "SOV_TR_M", - "HOV2TOLL": "HOV2_M", - "HOV3TOLL": "HOV3_M" - } - periods = ["EA", "AM", "MD", "PM", "EV"] - for p, w_d_pa, w_d_ap, nw_d_pa, nw_d_ap in zip( - periods, work_time_PA_factors, work_time_AP_factors, - nonwork_time_PA_factors, nonwork_time_AP_factors): - for gp_mode, toll_mode, w_o, nw_o in zip( - gp_modes, toll_modes, work_occupancy_factors, nonwork_occupancy_factors): - wrk_mtx = w_o * (w_d_pa * wrk_pa_mtx + w_d_ap * wrk_ap_mtx) - nwrk_mtx = nw_o * (nw_d_pa * nwrk_pa_mtx + nw_d_ap * nwrk_ap_mtx) - - # Toll choice split - f_tm_imp = emmebank.matrix('mf%s_%s_TIME' % (p, skim_lookup[gp_mode])).get_numpy_data(scenario) - t_tm_imp = emmebank.matrix('mf%s_%s_TIME' % (p, skim_lookup[toll_mode])).get_numpy_data(scenario) - t_cst_imp = emmebank.matrix('mf%s_%s_TOLLCOST' % (p, skim_lookup[toll_mode])).get_numpy_data(scenario) - - # Toll diversion for work purpose - # TODO: .mod no longer needed, to confirm - wrk_toll_prb = np.exp( - ivt_coef * (t_tm_imp - f_tm_imp + np.mod(t_cst_imp, 10000) / vot_work) - 3.39 - ) - wrk_toll_prb[t_cst_imp <= 0] = 0 - wrk_toll_prb = wrk_toll_prb / (1 + wrk_toll_prb) - work_matrix_toll = wrk_mtx * wrk_toll_prb - work_matrix_non_toll = wrk_mtx * (1 - wrk_toll_prb) - - toll_eiwork = emmebank.matrix('%s_%s_EIWORK' % (p, toll_mode)) - gp_ei_work = emmebank.matrix('%s_%s_EIWORK' % (p, gp_mode)) - toll_eiwork.set_numpy_data(work_matrix_toll, scenario) - gp_ei_work.set_numpy_data(work_matrix_non_toll, scenario) - - # Toll diversion for non work purpose - nwrk_toll_prb = np.exp( - ivt_coef * (t_tm_imp - f_tm_imp + np.mod(t_cst_imp, 10000) / vot_non_work) - 3.39 - ) - - nwrk_toll_prb[t_cst_imp <= 0] = 0 - nwrk_toll_prb = nwrk_toll_prb / (1 + nwrk_toll_prb) - - non_work_toll_matrix = nwrk_mtx * nwrk_toll_prb - non_work_gp_matrix = nwrk_mtx * (1 - nwrk_toll_prb) - - toll_einonwork = emmebank.matrix('%s_%s_EINONWORK' % (p, toll_mode)) - gp_einonwork = emmebank.matrix('%s_%s_EINONWORK' % (p, gp_mode)) - toll_einonwork.set_numpy_data(non_work_toll_matrix, scenario) - gp_einonwork.set_numpy_data(non_work_gp_matrix, scenario) - - precision = float(props['RunModel.MatrixPrecision']) - self.matrix_rounding(scenario, precision) - - @_m.logbook_trace('Controlled rounding of demand') - def matrix_rounding(self, scenario, precision): - round_matrix = _m.Modeller().tool( - "inro.emme.matrix_calculation.matrix_controlled_rounding") - emmebank = scenario.emmebank - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - modes = ["SOVGP", "HOV2HOV", "HOV3HOV", "SOVTOLL", "HOV2TOLL", "HOV3TOLL"] - purpose_types = ["EIWORK", "EINONWORK"] - for period in periods: - for mode in modes: - for purpose in purpose_types: - matrix = emmebank.matrix("mf%s_%s_%s" % (period, mode, purpose)) - try: - report = round_matrix(demand_to_round=matrix, - rounded_demand=matrix, - min_demand=precision, - values_to_round="SMALLER_THAN_MIN", - scenario=scenario) - except: - max_val = matrix.get_numpy_data(scenario.id).max() - if max_val == 0: - # if max_val is 0 the error is that the matrix is 0, log a warning - _m.logbook_write('Warning: matrix %s is all 0s' % matrix.named_id) - else: - raise diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py b/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py deleted file mode 100644 index 7a9207a..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/truck/distribution.py +++ /dev/null @@ -1,288 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/truck/distribution.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the truck distribution step. Distributes truck trips with congested -# skims and splits by time of day. -# The distribution is based on the mid-day travel time for the "generic" -# truck skim "mfMD_TRK_TIME". Applies truck toll diversion model with -# toll and non-toll skims. -# -# Inputs: -# input_directory: source directory for input files -# num_processors: Number of processors to use, either as a number or "MAX-#" -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file -# input/TruckTripRates.csv -# input/mgra13_based_inputYEAR.csv -# input/specialGenerators.csv -# -# Matrix inputs: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# moTRKL_PROD, moTRKM_PROD, moTRKH_PROD, moTRKEI_PROD, moTRKIE_PROD -# mdTRKL_ATTR, mdTRKM_ATTR, mdTRKH_ATTR, mdTRKEI_ATTR, mdTRKIE_ATTR -# mfTRKEE_DEMAND -# mfMD_TRK_TIME -# mfpp_TRKLGP_TIME, mfpp_TRKLTOLL_TIME, mfpp_TRKLTOLL_TOLLCOST -# mfpp_TRKMGP_TIME, mfpp_TRKMTOLL_TIME, mfpp_TRKMTOLL_TOLLCOST -# mfpp_TRKHGP_TIME, mfpp_TRKHTOLL_TIME, mfpp_TRKHTOLL_TOLLCOST -# -# Matrix intermediates (only used internally): -# mfTRKEI_FRICTION, mfTRKIE_FRICTION, mfTRKL_FRICTION, mfTRKM_FRICTION, mfTRKH_FRICTION -# -# Matrix results: -# Note: pp is time period, one of EA, AM, MD, PM, EV -# mfpp_TRKLGP_VEH, mfpp_TRKMGP_VEH, mfpp_TRKHGP_VEH -# mfpp_TRKLTOLL_VEH, mfpp_TRKMTOLL_VEH, mfpp_TRKHTOLL_VEH -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - main_directory = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(main_directory, "input") - num_processors = "MAX-1" - base_scenario = modeller.scenario - distribution = modeller.tool("sandag.model.truck.distribution") - distribution(input_dir, num_processors, base_scenario) -""" - - -TOOLBOX_ORDER = 43 - -import traceback as _traceback -import pandas as pd -import numpy as np -import os - -import inro.modeller as _m - - -gen_utils = _m.Modeller().module('sandag.utilities.general') -dem_utils = _m.Modeller().module('sandag.utilities.demand') - - -class TruckModel(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.num_processors = "MAX-1" - self.attributes = ["input_directory", "num_processors"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Truck distribution" - pb.description = """ -
    - Distributes truck trips with congested skims and splits by time of day. - The distribution is based on the mid-day travel time for the "generic" truck - skim "mfMD_TRK_TIME". -
    - Applies truck toll diversion model with toll and non-toll skims, - and generates truck vehicle trips. -
    - Note that the truck vehicle trips must be converted to PCE values by the Import auto - demand tool and stored in matrices without the _VEH ending for the auto assignment. -
    - """ - pb.branding_text = "- SANDAG - Model - Truck" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - dem_utils.add_select_processors("num_processors", pb, self) - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, self.num_processors, scenario) - run_msg = "Truck trip distribution complete." - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Truck distribution') - def __call__(self, input_directory, num_processors, scenario): - attributes = { - "input_directory": input_directory, - "num_processors": num_processors - } - gen_utils.log_snapshot("Truck distribution", str(self), attributes) - self.scenario = scenario - self.num_processors = num_processors - - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties( - os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) - - with _m.logbook_trace('Daily demand matrices'): - coefficents = [0.045, 0.03, 0.03, 0.03, 0.03] - truck_list = ['L', 'M', 'H', 'IE', 'EI'] - # distribution based on the "generic" truck MD time only - time_skim = scenario.emmebank.matrix('mf"MD_TRK_TIME"') - for truck_type, coeff in zip(truck_list, coefficents): - with _m.logbook_trace('Create %s daily demand matrix' % truck_type): - self.calc_friction_factors(truck_type, time_skim, coeff) - self.matrix_balancing(truck_type) - - self.split_external_demand() - self.split_into_time_of_day() - # NOTE: TOLL diversion skipped with new class definitions - #self.toll_diversion() - - with _m.logbook_trace('Reduce matrix precision'): - precision = props['RunModel.MatrixPrecision'] - matrices = [] - for t, pce in [('L', 1.3), ('M', 1.5), ('H', 2.5)]: - for p in ['EA', 'AM', 'MD', 'PM', 'EV']: - matrices.append('mf%s_TRK_%s_VEH' % (p, t)) - dem_utils.reduce_matrix_precision(matrices, precision, num_processors, scenario) - - @_m.logbook_trace('Create friction factors matrix') - def calc_friction_factors(self, truck_type, impedance, coeff): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - matrix_calc.run_single('mfTRK%s_FRICTION' % truck_type, - 'exp(-%s*%s)' % (coeff, impedance.named_id)) - return - - def matrix_balancing(self, truck_type): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - emmebank = self.scenario.emmebank - with _m.logbook_trace('Matrix balancing for %s' % truck_type): - if truck_type == 'IE': - with gen_utils.temp_matrices(emmebank, "DESTINATION") as (temp_md,): - temp_md.name = 'TRKIE_ROWTOTAL' - matrix_calc.add('md"TRKIE_ROWTOTAL"', 'mf"TRKIE_FRICTION"', aggregation={"origins": "+", "destinations": None}) - matrix_calc.add('mf"TRKIE_DEMAND"', 'mf"TRKIE_FRICTION" * md"TRKIE_ATTR" / md"TRKIE_ROWTOTAL"', - constraint=['md"TRKIE_ROWTOTAL"', 0, 0, "EXCLUDE"]) - matrix_calc.run() - - elif truck_type == 'EI': - with gen_utils.temp_matrices(emmebank, "ORIGIN") as (temp_mo,): - temp_mo.name = 'TRKEI_COLTOTAL' - matrix_calc.add('mo"TRKEI_COLTOTAL"', 'mf"TRKEI_FRICTION"', aggregation={"origins": None, "destinations": "+"}) - matrix_calc.add('mf"TRKEI_DEMAND"', 'mf"TRKEI_FRICTION" * mo"TRKEI_PROD" / mo"TRKEI_COLTOTAL"', - constraint=['mo"TRKEI_COLTOTAL"', 0, 0, "EXCLUDE"]) - matrix_calc.run() - else: - matrix_balancing = _m.Modeller().tool( - 'inro.emme.matrix_calculation.matrix_balancing') - spec = { - "type": "MATRIX_BALANCING", - "od_values_to_balance": 'mf"TRK%s_FRICTION"' % truck_type, - "origin_totals": 'mo"TRK%s_PROD"' % truck_type, - "destination_totals": 'md"TRK%s_ATTR"' % truck_type, - "results": { - "od_balanced_values": 'mf"TRK%s_DEMAND"' % truck_type, - }, - "max_iterations": 100, - "max_relative_error": 0.01 - } - matrix_balancing(spec, self.scenario) - - @_m.logbook_trace('Split cross-regional demand by truck type') - def split_external_demand(self): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - - truck_types = ['L', 'M', 'H'] - truck_share = [0.307, 0.155, 0.538] - for t_type, share in zip(truck_types, truck_share): - matrix_calc.add('mf"TRK%s_DEMAND"' % (t_type), - '%s * (mf"TRKEI_DEMAND" + mf"TRKIE_DEMAND" + mf"TRKEE_DEMAND")' % (share)) - # Set intrazonal truck trips to 0 - matrix_calc.add('mf"TRK%s_DEMAND"' % (t_type), 'mf"TRK%s_DEMAND" * (p.ne.q)' % (t_type)) - matrix_calc.run() - - @_m.logbook_trace('Distribute daily demand into time of day') - def split_into_time_of_day(self): - matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - time_share = [0.1018, 0.1698, 0.4284, 0.1543, 0.1457] - border_time_share = [0.0188, 0.1812, 0.4629, 0.2310, 0.1061] - border_correction = [bs/s for bs, s in zip(border_time_share, time_share)] - - truck_types = ['L', 'M', 'H'] - truck_names = {"L": "light trucks", "M": "medium trucks", "H": "heavy trucks"} - - for period, share, border_corr in zip(periods, time_share, border_correction): - for t in truck_types: - with matrix_calc.trace_run('Calculate %s demand matrix for %s' % (period, truck_names[t])): - tod_demand = 'mf"%s_TRK_%s_VEH"' % (period, t) - matrix_calc.add(tod_demand, 'mf"TRK%s_DEMAND"' % (t)) - matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, share)) - matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, border_corr), - {"origins": "1-5", "destinations": "1-9999"}) - matrix_calc.add(tod_demand, 'mf%s_TRK_%s_VEH * %s' % (period, t, border_corr), - {"origins": "1-9999", "destinations": "1-5"}) - - @_m.logbook_trace('Toll diversion') - def toll_diversion(self): - # NOTE: toll diversion skipped - pass - # matrix_calc = dem_utils.MatrixCalculator(self.scenario, self.num_processors) - # nest_factor = 10 - # vot = 0.02 # cent/min - # periods = ['EA', 'AM', 'MD', 'PM', 'EV'] - # truck_types = ['L', 'M', 'H'] - # truck_toll_factors = [1, 1.03, 2.33] - - # for period in periods: - # for truck, toll_factor in zip(truck_types, truck_toll_factors): - # with matrix_calc.trace_run('Toll diversion for period %s, truck type %s' % (period, truck) ): - # # Define the utility expression - # utility = """ - # ( - # (mf"%(p)s_TRK%(t)sGP_TIME" - mf"%(p)s_TRK%(t)sTOLL_TIME") - # - %(vot)s * mf"%(p)s_TRK%(t)sTOLL_TOLLCOST" * %(t_fact)s - # ) - # / %(n_fact)s - # """ % { - # 'p': period, - # 't': truck, - # 'vot': vot, - # 't_fact': toll_factor, - # 'n_fact': nest_factor - # } - # # If there is no toll probability of using toll is 0 - # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), '0') - # # If there is a non-zero toll value compute the share of - # # toll-available passengers using the utility expression defined earlier - # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), - # 'mf"%(p)s_TRK%(t)s" * (1/(1 + exp(- %(u)s)))' % {'p': period, 't': truck, 'u': utility}, - # ['mf"%s_TRK%sTOLL_TOLLCOST"' % (period, truck), 0, 0 , "EXCLUDE"]) - # # if non-toll path is not available (GP time=0), set all demand to toll - # matrix_calc.add('mf"%s_TRK%sTOLL_VEH"' % (period, truck), - # 'mf"%(p)s_TRK%(t)s"' % {'p': period, 't': truck}, - # ['mf"%(p)s_TRK%(t)sGP_TIME"' % {'p': period, 't': truck}, 0, 0 , "INCLUDE"]) - # # Compute the truck demand for non toll - # matrix_calc.add('mf"%s_TRK%sGP_VEH"' % (period, truck), - # '(mf"%(p)s_TRK%(t)s" - mf"%(p)s_TRK%(t)sTOLL_VEH").max.0' % {'p': period, 't': truck}) diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/generation.py b/sandag_abm/src/main/emme/toolbox/model/truck/generation.py deleted file mode 100644 index c0c10a8..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/truck/generation.py +++ /dev/null @@ -1,443 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// model/truck/generation.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the truck generation step. Generates standard truck trip and special (military) truck trips, -# and generates regional truck trips, IE trips, EI trips and EE trips and balances truck trips. -# -# Inputs: -# input_directory: source directory for most input files, including demographics and trip rates -# input_truck_directory: source for special truck files -# scenario: traffic scenario to use for reference zone system -# -# Files referenced: -# Note: YEAR is replaced by truck.FFyear in the conf/sandag_abm.properties file -# input/TruckTripRates.csv -# file referenced by key mgra.socec.file, e.g. input/mgra13_based_inputYEAR.csv -# input/specialGenerators.csv -# input_truck/regionalIEtripsYEAR.csv -# input_truck/regionalEItripsYEAR.csv -# input_truck/regionalEEtripsYEAR.csv -# -# Matrix results: -# moTRKL_PROD, moTRKM_PROD, moTRKH_PROD, moTRKEI_PROD, moTRKIE_PROD -# mdTRKL_ATTR, mdTRKM_ATTR, mdTRKH_ATTR, mdTRKEI_ATTR, mdTRKIE_ATTR -# mfTRKEE_DEMAND -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(project_dir, "input") - input_truck_dir = os.path.join(project_dir, "input_truck") - base_scenario = modeller.scenario - generation = modeller.tool("sandag.model.truck.generation") - generation(input_dir, input_truck_dir, base_scenario) -""" - - - - -TOOLBOX_ORDER = 42 - - -import inro.modeller as _m -import traceback as _traceback -import numpy as np -import pandas as pd -import os - - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class TruckGeneration(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - input_truck_directory = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.input_truck_directory = os.path.join(os.path.dirname(project_dir), "input_truck") - self.attributes = ["input_directory", "input_truck_directory"] - self._properties = None - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Truck generation" - pb.description = """ -
    - Generates standard truck trip and special (military) truck trips as well as - regional truck trips, IE trips, EI trips and EE trips and balances truck trips - productions / attractions. -
    """ - pb.branding_text = "- SANDAG - Model - Truck" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - pb.add_select_file('input_truck_directory', 'directory', - title='Select truck input directory') - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.input_directory, self.input_truck_directory, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Truck generation') - def __call__(self, input_directory, input_truck_directory, scenario): - attributes = {"input_directory": input_directory, "input_truck_directory": input_truck_directory} - gen_utils.log_snapshot("Truck generation", str(self), attributes) - self.input_directory = input_directory - self.input_truck_directory = input_truck_directory - self.scenario = scenario - load_properties = _m.Modeller().tool('sandag.utilities.properties') - - self._properties = load_properties( - os.path.join(os.path.dirname(input_directory), "conf", "sandag_abm.properties")) - base_trucks_PA = self.truck_standard_generation() - special_trucks_PA = self.special_truck_generation(base_trucks_PA) - trucks_PA = self.balance_truck_PA(special_trucks_PA) - self.store_PA_to_matrices(trucks_PA) - self.read_external_external_demand() - return trucks_PA - - def truck_standard_generation(self): - year = self._properties['truck.FFyear'] - is_interim_year, prev_year, next_year = self.interim_year_check(year) - head, mgra_input_file = os.path.split(self._properties['mgra.socec.file']) - if is_interim_year: - taz_prev_year = self.create_demographics_by_taz( - mgra_input_file.replace(str(year), str(prev_year))) - taz_next_year = self.create_demographics_by_taz( - mgra_input_file.replace(str(year), str(next_year))) - taz = self.interpolate_df(prev_year, year, next_year, taz_prev_year, taz_next_year) - else: - taz = self.create_demographics_by_taz(mgra_input_file) - - trip_rates = pd.read_csv( - os.path.join(self.input_directory, 'TruckTripRates.csv')) - taz = pd.merge(taz, trip_rates, - left_on='truckregiontype', right_on='RegionType', how='left') - taz.fillna(0, inplace=True) - - # Compute lhd truck productions "AGREMPN", "CONEMPN", "RETEMPN", "GOVEMPN", - # "MANEMPN", "UTLEMPN", "WHSEMPN", "OTHEMPN" - taz['LHD_Productions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_L_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TG_L_Retail'] \ - + taz['emp_gov'] * taz['TG_L_Government'] \ - + taz['emp_mfg'] * taz['TG_L_Manufacturing'] \ - + taz['emp_twu'] * taz['TG_L_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TG_L_Wholesale'] \ - + taz['emp_other'] * taz['TG_L_Other'] \ - + taz['hh'] * taz['TG_L_Households'] - - taz['LHD_Attractions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_L_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TA_L_Retail'] \ - + taz['emp_gov'] * taz['TA_L_Government'] \ - + taz['emp_mfg'] * taz['TA_L_Manufacturing'] \ - + taz['emp_twu'] * taz['TA_L_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TA_L_Wholesale'] \ - + taz['emp_other'] * taz['TA_L_Other'] \ - + taz['hh'] * taz['TA_L_Households'] - - taz['MHD_Productions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_M_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TG_M_Retail'] \ - + taz['emp_gov'] * taz['TG_M_Government'] \ - + taz['emp_mfg'] * taz['TG_M_Manufacturing'] \ - + taz['emp_twu'] * taz['TG_M_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TG_M_Wholesale'] \ - + taz['emp_other'] * taz['TG_M_Other'] \ - + taz['hh'] * taz['TG_M_Households'] - - taz['MHD_Attractions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_M_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TA_M_Retail'] \ - + taz['emp_gov'] * taz['TA_M_Government'] \ - + taz['emp_mfg'] * taz['TA_M_Manufacturing'] \ - + taz['emp_twu'] * taz['TA_M_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TA_M_Wholesale'] \ - + taz['emp_other'] * taz['TA_M_Other'] \ - + taz['hh'] * taz['TA_M_Households'] - - taz['HHD_Productions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TG_H_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TG_H_Retail'] \ - + taz['emp_gov'] * taz['TG_H_Government'] \ - + taz['emp_mfg'] * taz['TG_H_Manufacturing'] \ - + taz['emp_twu'] * taz['TG_H_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TG_H_Wholesale'] \ - + taz['emp_other'] * taz['TG_H_Other'] \ - + taz['hh'] * taz['TG_H_Households'] - - taz['HHD_Attractions'] = \ - (taz['emp_agmin'] + taz['emp_cons']) * taz['TA_H_Ag/Min/Constr'] \ - + taz['emp_retrade'] * taz['TA_H_Retail'] \ - + taz['emp_gov'] * taz['TA_H_Government'] \ - + taz['emp_mfg'] * taz['TA_H_Manufacturing'] \ - + taz['emp_twu'] * taz['TA_H_Transp/Utilities'] \ - + taz['emp_whtrade'] * taz['TA_H_Wholesale'] \ - + taz['emp_other'] * taz['TA_H_Other'] \ - + taz['hh'] * taz['TA_H_Households'] - - taz.reset_index(inplace=True) - taz = taz[['taz', - 'LHD_Productions', 'LHD_Attractions', - 'MHD_Productions', 'MHD_Attractions', - 'HHD_Productions', 'HHD_Attractions']] - return taz - - # Creates households and employments by TAZ. - # Specific to the truck trip generation model. - # Inputs: - # - sandag.properties - # - input/mgra13_based_input20XX.csv (referenced by mgra.socec.file in properties file) - def create_demographics_by_taz(self, mgra_input_file): - utils = _m.Modeller().module('sandag.utilities.demand') - dt = _m.Modeller().desktop.project.data_tables() - file_path = os.path.join(self.input_directory, mgra_input_file) - if not os.path.exists(file_path): - raise Exception("MGRA input file '%s' does not exist" % file_path) - mgra = pd.read_csv(file_path) - # Combine employment fields that match to the truck trip rate classification - mgra['TOTEMP'] = mgra.emp_total - mgra['emp_agmin'] = mgra.emp_ag - mgra['emp_cons'] = mgra.emp_const_bldg_prod + mgra.emp_const_bldg_office - mgra['emp_retrade'] = mgra.emp_retail + mgra.emp_personal_svcs_retail - mgra['emp_gov']= mgra.emp_state_local_gov_ent \ - + mgra.emp_state_local_gov_blue \ - + mgra.emp_state_local_gov_white \ - + mgra.emp_fed_non_mil \ - + mgra.emp_fed_mil - mgra['emp_mfg'] = mgra.emp_mfg_prod \ - + mgra.emp_mfg_office - mgra['emp_twu'] = mgra.emp_trans \ - + mgra.emp_utilities_office \ - + mgra.emp_utilities_prod - mgra['emp_whtrade'] = mgra.emp_whsle_whs - mgra['emp_other'] = mgra.TOTEMP \ - - mgra.emp_agmin \ - - mgra.emp_cons \ - - mgra.emp_retrade \ - - mgra.emp_gov \ - - mgra.emp_mfg \ - - mgra.emp_twu \ - - mgra.emp_whtrade - - f = { - 'truckregiontype':['mean'], - 'emp_agmin':['sum'], - 'emp_cons': ['sum'], - 'emp_retrade': ['sum'], - 'emp_gov': ['sum'], - 'emp_mfg': ['sum'], - 'emp_twu': ['sum'], - 'emp_whtrade': ['sum'], - 'emp_other': ['sum'], - 'hh': ['sum'] - } - - mgra = mgra[['truckregiontype', 'emp_agmin', 'emp_cons', - 'emp_retrade', 'emp_gov', 'emp_mfg', 'emp_twu', - 'emp_whtrade', 'emp_other', 'taz', 'hh']] - taz = mgra.groupby('taz').agg(f) - taz.reset_index(inplace=True) - taz.columns = taz.columns.droplevel(-1) - # Add external zones - taz = utils.add_missing_zones(taz, self.scenario) - return taz - - # Add trucks generated by special generators, such as military sites, - # mail to//from airport, cruise ships etc - # Inputs: - # - input/specialGenerators.csv - # - dataframe: base_trucks - def special_truck_generation(self, base_trucks): - year = self._properties['truck.FFyear'] - is_interim_year, prev_year, next_year = self.interim_year_check(year) - spec_gen = pd.read_csv(os.path.join(self.input_directory, 'specialGenerators.csv')) - spec_gen = pd.merge(spec_gen, base_trucks, - left_on=['TAZ'], right_on=['taz'], how='outer') - spec_gen.fillna(0, inplace=True) - if is_interim_year: - year_ratio = float(year - prev_year) / (next_year - prev_year) - prev_year, next_year = 'Y%s' % prev_year, 'Y%s' % next_year - spec_gen['Y%s' % year] = spec_gen[prev_year] + year_ratio * (spec_gen[next_year] - spec_gen[prev_year]) - - for t in ['L', 'M', 'H']: - spec_gen['%sHD_Attr' % t] = spec_gen['%sHD_Attractions' % t] + \ - (spec_gen['Y%s' % year] * - spec_gen['trkAttraction'] * - spec_gen['%shdShare' % t.lower()]) - spec_gen['%sHD_Prod' % t] = spec_gen['%sHD_Productions' % t] + \ - (spec_gen['Y%s' % year] * - spec_gen['trkProduction'] * - spec_gen['%shdShare' % t.lower()]) - - special_trucks = spec_gen[ - ['taz', 'LHD_Prod', 'LHD_Attr', 'MHD_Prod', 'MHD_Attr', 'HHD_Prod', 'HHD_Attr']] - return special_trucks - - # Balance truck Productions and Attractions - def balance_truck_PA(self, truck_pa): - truck_pa = self.balance_internal_truck_PA(truck_pa) - regional_truck_pa = self.get_regional_truck_PA() - truck_pa = self.add_balanced_regional_PA(regional_truck_pa, truck_pa) - truck_pa.fillna(0, inplace=True) - - truck_pa['TRKL_Prod'] = truck_pa['LHD_Prod'] - truck_pa['TRKM_Prod'] = truck_pa['MHD_Prod'] - truck_pa['TRKH_Prod'] = truck_pa['HHD_Prod'] - truck_pa['TRKIE_Prod'] = truck_pa['IE_Prod'] - truck_pa['TRKEI_Prod'] = truck_pa['EI_Prod'] - - truck_pa['TRKL_Attr'] = truck_pa['LHD_Attr'] - truck_pa['TRKM_Attr'] = truck_pa['MHD_Attr'] - truck_pa['TRKH_Attr'] = truck_pa['HHD_Attr'] - truck_pa['TRKIE_Attr'] = truck_pa['IE_Attr'] - truck_pa['TRKEI_Attr'] = truck_pa['EI_Attr'] - return truck_pa - - def get_regional_truck_PA(self): - year = self._properties['truck.FFyear'] - trips = {} - regional_trip_types = ['IE', 'EI', 'EE'] - is_interim_year, prev_year, next_year = self.interim_year_check(year) - if is_interim_year: - for t in regional_trip_types: - prev_trips = pd.read_csv(os.path.join( - self.input_truck_directory, - 'regional%strips%s.csv' % (t, prev_year))) - next_trips = pd.read_csv(os.path.join( - self.input_truck_directory, - 'regional%strips%s.csv' % (t, next_year))) - trips_df = self.interpolate_df(prev_year, year, next_year, prev_trips, next_trips) - trips[t] = trips_df - - for t in regional_trip_types: - trips[t] = pd.read_csv(os.path.join( - self.input_truck_directory, - 'regional%strips%s.csv' % (t, year))) - return trips - - def balance_internal_truck_PA(self, truck_pa): - truck_types = ['LHD', 'MHD', 'HHD'] - for t in truck_types: - s1 = truck_pa['%s_Prod' % t].sum() - s2 = truck_pa['%s_Attr' % t].sum() - avg = (s1 + s2)/2.0 - w1 = avg / s1 - w2 = avg / s2 - truck_pa['%s_Prod_unbalanced' % t] = truck_pa['%s_Prod' % t] - truck_pa['%s_Attr_unbalanced' % t] = truck_pa['%s_Attr' % t] - truck_pa['%s_Prod' % t] = truck_pa['%s_Prod' % t] * w1 - truck_pa['%s_Attr' % t] = truck_pa['%s_Attr' % t] * w2 - return truck_pa - - # Balance only EI and IE. EE truck trips are already balanced and can be - # directly imported as a matrix - def add_balanced_regional_PA(self, regional_trips, truck_pa): - ei_trips = regional_trips['EI'] - ei_trips = ei_trips.groupby('fromZone').sum() - ei_trips.reset_index(inplace=True) - - truck_pa = pd.merge(truck_pa, - ei_trips[['fromZone', 'EITrucks']], - left_on='taz', right_on='fromZone', - how='outer') - - sum_ei = ei_trips['EITrucks'].sum() - sum_hhd_attr = truck_pa['HHD_Attr'].sum() - truck_pa['EI_Attr'] = truck_pa['HHD_Attr'] * sum_ei / sum_hhd_attr - truck_pa['EI_Prod'] = truck_pa['EITrucks'] - - ie_trips = regional_trips['IE'] - ie_trips = ie_trips.groupby('toZone').sum() - ie_trips.reset_index(inplace=True) - truck_pa = pd.merge(truck_pa, - ie_trips[['toZone', 'IETrucks']], - left_on='taz', right_on='toZone', - how='outer') - - sum_ie = ie_trips['IETrucks'].sum() - sum_hhd_prod = truck_pa['HHD_Prod'].sum() - truck_pa['IE_Prod'] = truck_pa['HHD_Prod'] * sum_ie / sum_hhd_prod - truck_pa['IE_Attr'] = truck_pa['IETrucks'] - truck_pa.fillna(0, inplace=True) - - return truck_pa - - def store_PA_to_matrices(self, truck_pa): - emmebank = self.scenario.emmebank - truck_pa.sort_values('taz', inplace=True) #sort method was deprecated since version 0.20.0, yma, 2/12/2019 - control_to_store = ['L', 'M', 'H', 'EI', 'IE'] - for t in control_to_store: - prod = emmebank.matrix('moTRK%s_PROD' % t) - prod.set_numpy_data(truck_pa['TRK%s_Prod' % t].values, self.scenario) - attr = emmebank.matrix('mdTRK%s_ATTR' % t) - attr.set_numpy_data(truck_pa['TRK%s_Attr' % t].values, self.scenario) - - @_m.logbook_trace('External - external truck matrix') - def read_external_external_demand(self): - utils = _m.Modeller().module('sandag.utilities.demand') - emmebank = self.scenario.emmebank - regional_trips = self.get_regional_truck_PA() - ee = regional_trips['EE'] - m_ee = emmebank.matrix('mfTRKEE_DEMAND') - m_ee_data = m_ee.get_data(self.scenario) - for i, row in ee.iterrows(): - m_ee_data.set(row['fromZone'], row['toZone'], row['EETrucks']) - m_ee.set_data(m_ee_data, self.scenario) - - def interpolate_df(self, prev_year, new_year, next_year, prev_year_df, next_year_df): - current_year_df = pd.DataFrame() - year_ratio = float(new_year - prev_year) / (next_year - prev_year) - for key in prev_year_df.columns: - current_year_df[key] = ( - prev_year_df[key] + year_ratio * (next_year_df[key] - prev_year_df[key])) - return current_year_df - - def interim_year_check(self, year): - years_with_data = self._properties['truck.DFyear'] - if year in years_with_data: - return (False, year, year) - else: - next_year_idx = np.searchsorted(years_with_data, year) - if next_year_idx == 0 or next_year_idx > len(years_with_data): - raise Exception('Cannot interpolate data for year %s' % year) - prev_year = years_with_data[next_year_idx - 1] - next_year = years_with_data[next_year_idx] - return (True, prev_year, next_year) diff --git a/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py b/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py deleted file mode 100644 index 8185ca0..0000000 --- a/sandag_abm/src/main/emme/toolbox/model/truck/run_truck_model.py +++ /dev/null @@ -1,130 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// truck/run_truck_model.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -# -# Runs the truck model, the generation and distribution tools, in sequence. -# 1) Generates standard truck trip and special (military) truck trips -# 2) Generates regional truck trips, IE trips, EI trips and EE trips and -# balances truck trips -# 3) Distributes trips based on congested skims and splits by time of day -# 4) Applies truck toll diversion model with toll and non-toll skims -# -# Inputs: -# run_generation: boolean, if True run generation tool. -# input_directory: source directory for most input files, including demographics and trip rates -# (see generation and distribtuion tools) -# input_truck_directory: source for special truck files (see generation tool) -# num_processors: Number of processors to use, either as a number or "MAX-#" -# scenario: traffic scenario to use for reference zone system -# -# Script example: -""" - import os - modeller = inro.modeller.Modeller() - project_dir = os.path.dirname(os.path.dirname(modeller.desktop.project.path)) - input_dir = os.path.join(project_dir, "input") - input_truck_dir = os.path.join(project_dir, "input_truck") - base_scenario = modeller.scenario - num_processors = "MAX-1" - run_truck = modeller.tool("sandag.model.truck.run_truck_model") - run_truck(True, input_dir, input_truck_dir, num_processors, base_scenario) -""" - - - -TOOLBOX_ORDER = 41 - - -import inro.modeller as _m -import traceback as _traceback -import os - - -dem_utils = _m.Modeller().module("sandag.utilities.demand") -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class TruckModel(_m.Tool(), gen_utils.Snapshot): - - input_directory = _m.Attribute(str) - input_truck_directory = _m.Attribute(str) - run_generation = _m.Attribute(bool) - num_processors = _m.Attribute(str) - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def __init__(self): - self.run_generation = True - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.input_directory = os.path.join(os.path.dirname(project_dir), "input") - self.input_truck_directory = os.path.join(os.path.dirname(project_dir), "input_truck") - self.num_processors = "MAX-1" - self.attributes = ["input_directory", "input_truck_directory", "run_generation", "num_processors"] - - def page(self): - # Equivalent to TruckModel.rsc - pb = _m.ToolPageBuilder(self) - pb.title = "Truck model" - pb.description = """ -
    - 1) Generates standard truck trip and special (military) truck trips
    - 2) Gets regional truck trips, IE trips, EI trips and EE trips and balances truck trips
    - 3) Distributes truck trips with congested skims and splits by time of day
    - 4) Applies truck toll diversion model with free-flow toll and non-toll skims
    -
    -""" - pb.branding_text = "- SANDAG - Model - Truck" - - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - pb.add_checkbox("run_generation", title=" ", label="Run generation (first iteration)") - - pb.add_select_file('input_directory', 'directory', - title='Select input directory') - pb.add_select_file('input_truck_directory', 'directory', - title='Select truck input directory') - dem_utils.add_select_processors("num_processors", pb, self) - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - scenario = _m.Modeller().scenario - self(self.run_generation, self.input_directory, self.input_truck_directory, self.num_processors, scenario) - run_msg = "Tool complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - @_m.logbook_trace('Truck model', save_arguments=True) - def __call__(self, run_generation, input_directory, input_truck_directory, num_processors, scenario): - attributes = { - "input_directory": input_directory, "input_truck_directory": input_truck_directory, - "run_generation": run_generation, "num_processors": num_processors - } - gen_utils.log_snapshot("Truck model", str(self), attributes) - - generation = _m.Modeller().tool('sandag.model.truck.generation') - distribution = _m.Modeller().tool('sandag.model.truck.distribution') - - if run_generation: - generation(input_directory, input_truck_directory, scenario) - distribution(input_directory, num_processors, scenario) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/demand.py b/sandag_abm/src/main/emme/toolbox/utilities/demand.py deleted file mode 100644 index bcd0f5f..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/demand.py +++ /dev/null @@ -1,297 +0,0 @@ -##////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// utilities/demand.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// - -TOOLBOX_ORDER = 101 - - -import inro.emme.datatable as _dt -import inro.modeller as _m -from collections import OrderedDict -from contextlib import contextmanager as _context -from copy import deepcopy as _copy -import multiprocessing as _multiprocessing -import re as _re -import pandas as _pandas -import numpy as _numpy -import os - - -class Utils(_m.Tool()): - - def page(self): - pb = _m.ToolPageBuilder(self, runnable=False) - pb.title = 'Demand utility' - pb.description = """Utility tool / module for common code. Not runnable.""" - pb.branding_text = ' - SANDAG - Utilities' - return pb.render() - - -# Read a CSV file, store it as a DataTable and return a representative DataFrame -def csv_to_data_table(path, overwrite=False): - layer_name = os.path.splitext(os.path.basename(path))[0] - data_source = _dt.DataSource(path) - data = data_source.layer(layer_name).get_data() - desktop = _m.Modeller().desktop - dt_db = desktop.project.data_tables() - table = dt_db.create_table(layer_name, data, overwrite=overwrite) - return table_to_dataframe(table) - - -# Convert a DataTable into a DataFrame -def table_to_dataframe(table): - if type(table) == str: - desktop = _m.Modeller().desktop - dt_db = desktop.project.data_tables() - table_name = table - table = dt_db.table(table) - if not table: - raise Exception('%s is not a valid table name.' %table_name) - - df = _pandas.DataFrame() - for attribute in table.get_data().attributes(): - try: - df[attribute.name] = attribute.values.astype(float) - except Exception, e: - df[attribute.name] = attribute.values - - return df - - -# Convert a dataframe to a datatable -def dataframe_to_table(df, name): - desktop = _m.Modeller().desktop - dt_db = desktop.project.data_tables() - data = _dt.Data() - for key in df.columns: - found_dtype = False - dtypes = [ - (bool, True, 'BOOLEAN'), - (int, 0, 'INTEGER32'), - (int, 0, 'INTEGER'), - (float, 0, 'REAL') - ] - for caster, default, name in dtypes: - try: - df[[key]] = df[[key]].fillna(default) - values = df[key].astype(caster) - attribute = _dt.Attribute(key, values, name) - found_dtype = True - break - except ValueError: - pass - - if not found_dtype: - df[[key]] = df[[key]].fillna(0) - values = df[key].astype(str) - attribute = _dt.Attribute(key, values, 'STRING') - - data.add_attribute(attribute) - - table = dt_db.create_table(name, data, overwrite=True) - return table - -# Add missing (usually external zones 1 to 12) zones to the DataFrame -# and populate with zeros -def add_missing_zones(df, scenario): - all_zones = scenario.zone_numbers - existing_zones = df['taz'].values - missing_zones = set(all_zones) - set(existing_zones) - num_missing = len(missing_zones) - if num_missing == 0: - return df - - ext_df = _pandas.DataFrame() - for c in df.columns: - ext_df[c] = _numpy.zeros(num_missing) - ext_df['taz'] = _numpy.array(list(missing_zones)) - df = _pandas.concat([df, ext_df]) - df = df.sort_values('taz', ascending=True) # sort method was deprecated in version 0.20.0,yma,2/12/2019 - return df - - -def add_select_processors(tool_attr_name, pb, tool): - max_processors = _multiprocessing.cpu_count() - tool._max_processors = max_processors - options = [("MAX-1", "Maximum available - 1"), ("MAX", "Maximum available")] - options.extend([(n, "%s processors" % n) for n in range(1, max_processors + 1) ]) - pb.add_select(tool_attr_name, options, title="Number of processors:") - - -def parse_num_processors(value): - max_processors = _multiprocessing.cpu_count() - if isinstance(value, int): - return value - if isinstance(value, basestring): - if value == "MAX": - return max_processors - if _re.match("^[0-9]+$", value): - return int(value) - result = _re.split("^MAX[\s]*-[\s]*", value) - if len(result) == 2: - return max(max_processors - int(result[1]), 1) - if value: - return int(value) - return value - -class MatrixCalculator(object): - def __init__(self, scenario, num_processors=0): - self._scenario = scenario - self._matrix_calc = _m.Modeller().tool( - "inro.emme.matrix_calculation.matrix_calculator") - self._specs = [] - self._last_report = None - self.num_processors = num_processors - - @property - def num_processors(self): - return self._num_processors - - @num_processors.setter - def num_processors(self, value): - self._num_processors = parse_num_processors(value) - - @property - def last_report(self): - return _copy(self._last_report) - - @_context - def trace_run(self, name): - with _m.logbook_trace(name): - yield - self.run() - - def add(self, result, expression, constraint=None, aggregation=None): - spec = self._format_spec(result, expression, constraint, aggregation) - self._specs.append(spec) - - def _format_spec(self, result, expression, constraint, aggregation): - spec = { - "result": result, - "expression": expression, - "type": "MATRIX_CALCULATION" - } - if constraint is not None: - if isinstance(constraint, (list, tuple)): - # specified as list of by_value inputs - constraint = { - "by_value": { - "od_values": constraint[0], - "interval_min": constraint[1], - "interval_max": constraint[2], - "condition": constraint[3] - } - } - elif "od_values" in constraint: - # specified as the by_value sub-dictionary only - constraint = {"by_value": constraint} - # By zone constraints - elif ("destinations" in constraint or "origins" in constraint): - # specified as the by_zone sub-dictionary only - constraint = {"by_zone": constraint} - # otherwise, specified as a regular full constraint dictionary - if "by_value" in constraint: - # cast the inputs to the correct values - constraint["by_value"]["od_values"] = \ - str(constraint["by_value"]["od_values"]) - constraint["by_value"]["condition"] = \ - constraint["by_value"]["condition"].upper() - spec["constraint"] = constraint - - #Add None for missing key values if needed - if "by_value" not in constraint: - constraint["by_value"] = None - if "by_zone" not in constraint: - constraint["by_zone"] = None - - else: - spec["constraint"] = None - - if aggregation is not None: - if isinstance(aggregation, basestring): - aggregation = {"origins": aggregation} - spec["aggregation"] = aggregation - else: - spec["aggregation"] = None - return spec - - def add_spec(self, spec): - self._specs.append(spec) - - def run(self): - specs, self._specs = self._specs, [] - report = self._matrix_calc(specs, scenario=self._scenario, - num_processors=self._num_processors) - self._last_report = report - return report - - def run_single(self, result, expression, constraint=None, aggregation=None): - spec = self._format_spec(result, expression, constraint, aggregation) - return self._matrix_calc(spec, scenario=self._scenario, - num_processors=self._num_processors) - - -def reduce_matrix_precision(matrices, precision, num_processors, scenario): - emmebank = scenario.emmebank - calc = MatrixCalculator(scenario, num_processors) - gen_utils = _m.Modeller().module('sandag.utilities.general') - with gen_utils.temp_matrices(emmebank, "SCALAR", 2) as (sum1, sum2): - sum1.name = "ORIGINAL_SUM" - sum2.name = "ROUNDED_SUM" - for mat in matrices: - mat = emmebank.matrix(mat).named_id - with calc.trace_run('Reduce precision for matrix %s' % mat): - calc.add(sum1.named_id, mat, aggregation={"destinations": "+", "origins": "+"}) - calc.add(mat, "{mat} * ({mat} >= {precision})".format( - mat=mat, precision=precision)) - calc.add(sum2.named_id, mat, aggregation={"destinations": "+", "origins": "+"}) - calc.add(sum2.named_id, "({sum2} + ({sum2} == 0))".format(sum2=sum2.named_id)) - calc.add(mat, "{mat} * ({sum1} / {sum2})".format( - mat=mat, sum2=sum2.named_id, sum1=sum1.named_id)) - - -def create_full_matrix(name, desc, scenario): - create_matrix = _m.Modeller().tool( - "inro.emme.data.matrix.create_matrix") - emmebank = scenario.emmebank - matrix = emmebank.matrix(name) - if matrix: - ident = matrix.id - else: - used_ids = set([]) - for m in emmebank.matrices(): - if m.prefix == "mf": - used_ids.add(int(m.id[2:])) - for i in range(900, emmebank.dimensions["full_matrices"]): - if i not in used_ids: - ident = "mf" + str(i) - break - else: - raise Exception("Not enough available matrix IDs for selected demand. Change database dimensions to increase full matrices.") - return create_matrix(ident, name, desc, scenario=scenario, overwrite=True) - - -def demand_report(matrices, label, scenario, report=None): - text = ['
    '] - text.append("%-28s %13s" % ("name", "sum")) - for name, data in matrices: - stats = (name, data.sum()) - text.append("%-28s %13.7g" % stats) - text.append("
    ") - title = "Demand summary" - if report is None: - report = _m.PageBuilder(title) - report.wrap_html('Matrix details', "
    ".join(text)) - _m.logbook_write(label, report.render()) - else: - report.wrap_html(label, "
    ".join(text)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py b/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py deleted file mode 100644 index 6e56df5..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/file_manager.py +++ /dev/null @@ -1,370 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2018. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// utilities/file_manager.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// -# -TOOLBOX_ORDER = 104 - - -import inro.modeller as _m -import inro.emme.database.emmebank as _eb -import inro.director.logging as _log - -import traceback as _traceback -import shutil as _shutil -import time as _time -import os -from fnmatch import fnmatch as _fnmatch -from math import log10 - -_join = os.path.join -_dir = os.path.dirname -_norm = os.path.normpath - -gen_utils = _m.Modeller().module("sandag.utilities.general") - - -class FileManagerTool(_m.Tool(), gen_utils.Snapshot): - - operation = _m.Attribute(unicode) - remote_dir = _m.Attribute(unicode) - local_dir = _m.Attribute(unicode) - user_folder = _m.Attribute(unicode) - scenario_id = _m.Attribute(unicode) - initialize = _m.Attribute(_m.BooleanType) - delete_local_files = _m.Attribute(_m.BooleanType) - - tool_run_msg = "" - LOCAL_ROOT = "C:\\abm_runs" - - def __init__(self): - self.operation = "UPLOAD" - project_dir = _dir(_m.Modeller().desktop.project.path) - self.remote_dir = _dir(project_dir) - folder_name = os.path.basename(self.remote_dir) - self.user_folder = os.environ.get("USERNAME") - self.scenario_id = 100 - self.initialize = True - self.delete_local_files = True - self.attributes = [ - "operation", "remote_dir", "local_dir", "user_folder", - "scenario_id", "initialize", "delete_local_files" - ] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "File run manager utility" - pb.description = """ -

    - Utility tool to manually manage the use of the local drive for subsequent model run. - The remote data can be downloaded (copied) to the local drive; - or the local data can be uploaded to the remote drive. - In normal operation this tool does not need to run manually, but in case of an - error it may be necessary to upload the project data in order to run on - a different machine, or operate directly on the server. -

    -

    - Note that file masks are used from config/sandag_abm.properties to identify which - files to copy. See RunModel.FileMask.Upload and RunModel.FileMask.Download for - upload and download respectively. -

    """ - pb.branding_text = "- SANDAG" - if self.tool_run_msg: - pb.add_html(self.tool_run_msg) - - pb.add_radio_group('operation', title="File copy operation", - keyvalues=[("UPLOAD", "Upload from local directory to remote directory"), - ("DOWNLOAD", "Download from remote directory to local directory")], ) - pb.add_select_file('remote_dir','directory', - title='Select remote ABM directory (e.g. on T drive)', note='') - pb.add_text_box('user_folder', title="User folder (for local drive):") - pb.add_text_box('scenario_id', title="Base scenario ID:") - pb.add_checkbox_group( - [{"attribute": "delete_local_files", "label": "Delete all local files on completion (upload only)"}, - {"attribute": "initialize", "label": "Initialize all local files; if false only download files which are different (download only)"}]) - pb.add_html(""" -""") - - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self(self.operation, self.remote_dir, self.user_folder, self.scenario_id, - self.initialize, self.delete_local_files) - run_msg = "File copying complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - def __call__(self, operation, remote_dir, user_folder, scenario_id, initialize=True, delete_local_files=True): - load_properties = _m.Modeller().tool('sandag.utilities.properties') - props = load_properties(_join(remote_dir, "conf", "sandag_abm.properties")) - if operation == "DOWNLOAD": - file_masks = props.get("RunModel.FileMask.Download") - return self.download(remote_dir, user_folder, scenario_id, initialize, file_masks) - elif operation == "UPLOAD": - file_masks = props.get("RunModel.FileMask.Upload") - self.upload(remote_dir, user_folder, scenario_id, delete_local_files, file_masks) - else: - raise Exception("operation must be one of UPLOAD or DOWNLOAD") - - @_m.logbook_trace("Copy project data to local drive", save_arguments=True) - def download(self, remote_dir, user_folder, scenario_id, initialize, file_masks): - folder_name = os.path.basename(remote_dir) - user_folder = user_folder or os.environ.get("USERNAME") - if not user_folder: - raise Exception("Username must be specified for local drive operation " - "(or define USERNAME environment variable)") - if not os.path.exists(self.LOCAL_ROOT): - os.mkdir(self.LOCAL_ROOT) - user_directory = _join(self.LOCAL_ROOT, user_folder) - if not os.path.exists(user_directory): - os.mkdir(user_directory) - local_dir = _join(user_directory, folder_name) - if not os.path.exists(local_dir): - os.mkdir(local_dir) - - self._report = ["Copy"] - self._stats = {"size": 0, "count": 0} - if not file_masks: - # suggested default: "output", "report", "sql", "logFiles" - file_masks = [] - file_masks = [_join(remote_dir, p) for p in file_masks] - file_masks.append(_join(remote_dir, "emme_project")) - if initialize: - # make sure that all of the root directories are created - root_dirs = [ - "application", "bin", "conf", "emme_project", "input", "input_truck", - "logFiles", "output", "python", "report", "sql", "uec" - ] - for name in root_dirs: - if not os.path.exists(_join(local_dir, name)): - os.mkdir(_join(local_dir, name)) - # create new Emmebanks with scenario and matrix data - title_fcn = lambda t: "(local) " + t[:50] - emmebank_paths = self._copy_emme_data( - src=remote_dir, dst=local_dir, initialize=True, - title_fcn=title_fcn, scenario_id=scenario_id) - # add new emmebanks to the open project - # db_paths = set([db.core_emmebank.path for db in data_explorer.databases()]) - # for path in emmebank_paths: - # if path not in db_paths: - # _m.Modeller().desktop.data_explorer().add_database(path) - - # copy all files (except Emme project, and other file_masks) - self._copy_dir(src=remote_dir, dst=local_dir, - file_masks=file_masks, check_metadata=not initialize) - self.log_report() - return local_dir - - @_m.logbook_trace("Copy project data to remote drive", save_arguments=True) - def upload(self, remote_dir, user_folder, scenario_id, delete_local_files, file_masks): - folder_name = os.path.basename(remote_dir) - user_folder = user_folder or os.environ.get("USERNAME") - user_directory = _join(self.LOCAL_ROOT, user_folder) - local_dir = _join(user_directory, folder_name) - - self._report = [] - self._stats = {"size": 0, "count": 0} - if not file_masks: - # suggested defaults: "application", "bin", "input", "input_truck", "uec", - # "output\\iter*", "output\\*_1.csv", "output\\*_2.csv" - file_masks = [] - # prepend the src dir to the project masks - file_masks = [_join(local_dir, p) for p in file_masks] - # add to mask the emme_project folder - file_masks.append(_join(local_dir, "emme_project")) - - title_fcn = lambda t: t[8:] if t.startswith("(local)") else t - emmebank_paths = self._copy_emme_data( - src=local_dir, dst=remote_dir, title_fcn=title_fcn, scenario_id=scenario_id) - # copy all files (except Emme project, and other file_masks) - self._copy_dir(src=local_dir, dst=remote_dir, file_masks=file_masks) - self.log_report() - - # data_explorer = _m.Modeller().desktop.data_explorer() - # for path in emmebank_paths: - # for db in data_explorer.databases(): - # if db.core_emmebank.path == path: - # db.close() - # data_explorer.remove_database(db) - # data_explorer.databases()[0].open() - - if delete_local_files: - # small pause for file handles to close - _time.sleep(2) - for name in os.listdir(local_dir): - path = os.path.join(local_dir, name) - if os.path.isfile(path): - try: # no raise, local files can be left behind - os.remove(path) - except: - pass - elif os.path.isdir(path): - try: - _shutil.rmtree(path, ignore_errors=False) - except: - pass - - _shutil.rmtree(local_dir, ignore_errors=False) - - def _copy_emme_data(self, src, dst, title_fcn, scenario_id, initialize=False): - # copy data from Database and Database_transit using API and import tool - # create new emmebanks and copy emmebank data to local drive - import_from_db = _m.Modeller().tool("inro.emme.data.database.import_from_database") - emmebank_paths = [] - for db_dir in ["Database", "Database_transit"]: - src_db_path = _join(src, "emme_project", db_dir, "emmebank") - if not os.path.exists(src_db_path): - # skip if the database does not exist (will be created later) - continue - src_db = _eb.Emmebank(src_db_path) - dst_db_dir = _join(dst, "emme_project", db_dir) - dst_db_path = _join(dst_db_dir, "emmebank") - emmebank_paths.append(dst_db_path) - self._report.append("Copying Emme data
    from %s
    to %s" % (src_db_path, dst_db_path)) - self._report.append("Start: %s" % _time.strftime("%c")) - if initialize: - # remove any existing database (overwrite) - if os.path.exists(dst_db_path): - self._report.append("Warning: overwritting existing Emme database %s" % dst_db_path) - dst_db = _eb.Emmebank(dst_db_path) - dst_db.dispose() - if os.path.exists(dst_db_dir): - gen_utils.retry(lambda: _shutil.rmtree(dst_db_dir)) - gen_utils.retry(lambda: os.mkdir(dst_db_dir)) - dst_db = _eb.create(dst_db_path, src_db.dimensions) - else: - if not os.path.exists(dst_db_dir): - os.mkdir(dst_db_dir) - if os.path.exists(dst_db_path): - dst_db = _eb.Emmebank(dst_db_path) - else: - dst_db = _eb.create(dst_db_path, src_db.dimensions) - - dst_db.title = title_fcn(src_db.title) - for prop in ["coord_unit_length", "unit_of_length", "unit_of_cost", - "unit_of_energy", "use_engineering_notation", "node_number_digits"]: - setattr(dst_db, prop, getattr(src_db, prop)) - - if initialize: - src_db.dispose() - continue - - exfpars = [p for p in dir(src_db.extra_function_parameters) if p.startswith("e")] - for exfpar in exfpars: - value = getattr(src_db.extra_function_parameters, exfpar) - setattr(dst_db.extra_function_parameters, exfpar, value) - - for s in src_db.scenarios(): - if dst_db.scenario(s.id): - dst_db.delete_scenario(s) - for f in src_db.functions(): - if dst_db.function(f.id): - dst_db.delete_function(f) - for m in src_db.matrices(): - if dst_db.matrix(m.id): - dst_db.delete_matrix(m) - for p in dst_db.partitions(): - p.description = "" - p.initialize(0) - ref_scen = dst_db.scenario(999) - if not ref_scen: - ref_scen = dst_db.create_scenario(999) - import_from_db( - src_database=src_db, - src_scenario_ids=[s.id for s in src_db.scenarios()], - src_function_ids=[f.id for f in src_db.functions()], - copy_path_strat_files=True, - dst_database=dst_db, - dst_zone_system_scenario=ref_scen) - dst_db.delete_scenario(999) - src_matrices = [m.id for m in src_db.matrices()] - src_partitions = [p.id for p in src_db.partitions() - if not(p.description == '' and not (sum(p.raw_data)))] - if src_matrices or src_partitions: - import_from_db( - src_database=src_db, - src_zone_system_scenario=src_db.scenario(scenario_id), - src_matrix_ids=src_matrices, - src_partition_ids=src_partitions, - dst_database=dst_db, - dst_zone_system_scenario=dst_db.scenario(scenario_id)) - src_db.dispose() - self._report.append("End: %s" % _time.strftime("%c")) - return emmebank_paths - - def _copy_dir(self, src, dst, file_masks, check_metadata=False): - for name in os.listdir(src): - src_path = _join(src, name) - skip_file = bool([1 for mask in file_masks if _fnmatch(src_path, mask)]) - if skip_file: - continue - dst_path = _join(dst, name) - if os.path.isfile(src_path): - size = os.path.getsize(src_path) - if check_metadata and os.path.exists(dst_path): - same_size = os.path.getsize(dst_path) == size - same_time = os.path.getmtime(dst_path) == os.path.getmtime(src_path) - if same_size and same_time: - continue - self._report.append(_time.strftime("%c")) - self._report.append(dst_path + file_size(size)) - self._stats["size"] += size - self._stats["count"] += 1 - # shutil.copy2 performs 5-10 times faster on download, and ~20% faster on upload - # than os.system copy calls - src_time = os.path.getmtime(src_path) - if name == 'persons.csv' or "mgra13_based" in name: - src_time = os.path.getmtime(src_path) - if os.path.exists(dst_path): - dest_time = os.path.getmtime(dst_path) - if dest_time <= src_time: - _shutil.copy2(src_path, dst_path) - else: - pass - else: - _shutil.copy2(src_path, dst_path) - else: - _shutil.copy2(src_path, dst_path) - self._report.append(_time.strftime("%c")) - elif os.path.isdir(src_path): - if not os.path.exists(dst_path): - os.mkdir(dst_path) - self._report.append(dst_path) - self._copy_dir(src_path, dst_path, file_masks, check_metadata) - - def log_report(self): - size, count = file_size(self._stats["size"]), self._stats["count"] - name = "File copy report: copied {count} files {size}".format(count=count, size=size) - report = _m.PageBuilder(title=name) - report.add_html("
    ".join(self._report)) - _m.logbook_write(name, report.render()) - - -_suffixes = ['bytes', 'KiB', 'MiB', 'GiB', 'TiB'] - -def file_size(size): - order = int(log10(size) / 3) if size else 0 - return ' {} {}'.format(round(float(size) / (10**(order*3)), 1), _suffixes[order]) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/general.py b/sandag_abm/src/main/emme/toolbox/utilities/general.py deleted file mode 100644 index 6879412..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/general.py +++ /dev/null @@ -1,386 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// transit_assignment.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// - -TOOLBOX_ORDER = 102 - - -import inro.modeller as _m -import inro.emme.datatable as _dt -import inro.emme.core.exception as _except -from osgeo import ogr as _ogr -from contextlib import contextmanager as _context -from itertools import izip as _izip -import traceback as _traceback -import re as _re -import json as _json -import time as _time -import os -import numpy as _numpy - -_omx = _m.Modeller().module("sandag.utilities.omxwrapper") - - -class UtilityTool(_m.Tool()): - - tool_run_msg = "" - - def page(self): - pb = _m.ToolPageBuilder(self, runnable=False) - pb.title = "General utility" - pb.description = """Utility tool / module for common code. Not runnable.""" - pb.branding_text = "- SANDAG" - if self.tool_run_msg: - pb.add_html(self.tool_run_msg) - - return pb.render() - - def run(self): - pass - - -class NetworkCalculator(object): - def __init__(self, scenario): - self._scenario = scenario - self._network_calc = _m.Modeller().tool( - "inro.emme.network_calculation.network_calculator") - - def __call__(self, result, expression, selections=None, aggregation=None): - spec = { - "result": result, - "expression": expression, - "aggregation": aggregation, - "type": "NETWORK_CALCULATION" - } - if selections is not None: - if isinstance(selections, basestring): - selections = {"link": selections} - spec["selections"] = selections - else: - spec["selections"] = {"link": "all"} - return self._network_calc(spec, self._scenario) - - -@_context -def temp_matrices(emmebank, mat_type, total=1, default_value=0.0): - matrices = [] - try: - while len(matrices) != int(total): - try: - ident = emmebank.available_matrix_identifier(mat_type) - except _except.CapacityError: - raise _except.CapacityError( - "Insufficient room for %s required temp matrices." % total) - matrices.append(emmebank.create_matrix(ident, default_value)) - yield matrices[:] - finally: - for matrix in matrices: - # In case of transient file conflicts and lag in windows file handles over the network - # attempt to delete file 10 times with increasing delays 0.05, 0.2, 0.45, 0.8 ... 5 - remove_matrix = lambda: emmebank.delete_matrix(matrix) - retry(remove_matrix) - - -def retry(fcn, attempts=10, init_wait=0.05, error_types=(RuntimeError, WindowsError)): - for attempt in range(1, attempts + 1): - try: - fcn() - return - except error_types: - if attempt > attempts: - raise - _time.sleep(init_wait * (attempt**2)) - - -@_context -def temp_attrs(scenario, attr_type, idents, default_value=0.0): - attrs = [] - try: - for ident in idents: - attrs.append(scenario.create_extra_attribute(attr_type, ident, default_value)) - yield attrs[:] - finally: - for attr in attrs: - scenario.delete_extra_attribute(attr) - - -@_context -def backup_and_restore(scenario, backup_attributes): - backup = {} - for elem_type, attributes in backup_attributes.iteritems(): - backup[elem_type] = scenario.get_attribute_values(elem_type, attributes) - try: - yield - finally: - for elem_type, attributes in backup_attributes.iteritems(): - scenario.set_attribute_values(elem_type, attributes, backup[elem_type]) - - -class DataTableProc(object): - - def __init__(self, table_name, path=None, data=None, convert_numeric=False): - modeller = _m.Modeller() - desktop = modeller.desktop - project = desktop.project - self._dt_db = dt_db = project.data_tables() - self._convert_numeric = convert_numeric - if path: - #try: - source = _dt.DataSource(path) - #except: - # raise Exception("Cannot open file at %s" % path) - layer = source.layer(table_name) - self._data = layer.get_data() - elif data: - table = dt_db.create_table(table_name, data, overwrite=True) - self._data = data - else: - table = dt_db.table(table_name) - self._data = table.get_data() - self._load_data() - - def _load_data(self): - data = self._data - if self._convert_numeric: - values = [] - for a in data.attributes(): - attr_values = _numpy.copy(a.values) - attr_values[attr_values == ''] = 0 - try: - values.append(attr_values.astype("int")) - except ValueError: - try: - values.append(attr_values.astype("float")) - except ValueError: - values.append(a.values) - self._values = values - else: - self._values = [a.values for a in data.attributes()] - self._attr_names = [a.name for a in data.attributes()] - self._index = dict((k, i) for i,k in enumerate(self._attr_names)) - if "geometry" in self._attr_names: - geo_coords = [] - attr = data.attribute("geometry") - for record in attr.values: - geo_obj = _ogr.CreateGeometryFromWkt(record.text) - geo_coords.append(geo_obj.GetPoints()) - self._values.append(geo_coords) - self._attr_names.append("geo_coordinates") - - def __iter__(self): - values, attr_names = self._values, self._attr_names - return (dict(_izip(attr_names, record)) - for record in _izip(*values)) - - def save(self, name, overwrite=False): - self._dt_db.create_table(name, self._data, overwrite=overwrite) - - def values(self, name): - index = self._index[name] - return self._values[index] - - -class Snapshot(object): - def __getitem__(self, key): - return getattr(self, key) - - def __setitem__(self, key, value): - setattr(self, key, value) - - def to_snapshot(self): - try: - attributes = getattr(self, "attributes", []) - snapshot = {} - for name in attributes: - snapshot[name] = unicode(self[name]) - return _json.dumps(snapshot) - except Exception: - return "{}" - - def from_snapshot(self, snapshot): - try: - snapshot = _json.loads(snapshot) - attributes = getattr(self, "attributes", []) - for name in attributes: - self[name] = snapshot[name] - except Exception, error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error), False) - return self - - def get_state(self): - attributes = getattr(self, "attributes", []) - state = {} - for name in attributes: - try: - state[name] = self[name] - except _m.AttributeError, error: - state[name] = unicode(error) - return state - - -def log_snapshot(name, namespace, snapshot): - try: - _m.logbook_snapshot(name=name, comment="", namespace=namespace, - value=_json.dumps(snapshot)) - except Exception as error: - print error - - -class ExportOMX(object): - def __init__(self, file_path, scenario, omx_key="NAME"): - self.file_path = file_path - self.scenario = scenario - self.emmebank = scenario.emmebank - self.omx_key = omx_key - - @property - def omx_key(self): - return self._omx_key - - @omx_key.setter - def omx_key(self, omx_key): - self._omx_key = omx_key - text_encoding = self.emmebank.text_encoding - if omx_key == "ID_NAME": - self.generate_key = lambda m: "%s_%s" % ( - m.id.encode(text_encoding), m.name.encode(text_encoding)) - elif omx_key == "NAME": - self.generate_key = lambda m: m.name.encode(text_encoding) - elif omx_key == "ID": - self.generate_key = lambda m: m.id.encode(text_encoding) - - def __enter__(self): - self.trace = _m.logbook_trace(name="Export matrices to OMX", - attributes={ - "file_path": self.file_path, "omx_key": self.omx_key, - "scenario": self.scenario, "emmebank": self.emmebank.path}) - self.trace.__enter__() - self.omx_file = _omx.open_file(self.file_path, 'w') - try: - self.omx_file.create_mapping('zone_number', self.scenario.zone_numbers) - except LookupError: - pass - return self - - def __exit__(self, exc_type, exc_val, exc_tb): - self.omx_file.close() - self.trace.__exit__(exc_type, exc_val, exc_tb) - - def write_matrices(self, matrices): - if isinstance(matrices, dict): - for key, matrix in matrices.iteritems(): - self.write_matrix(matrix, key) - else: - for matrix in matrices: - self.write_matrix(matrix) - - def write_matrix(self, matrix, key=None): - text_encoding = self.emmebank.text_encoding - matrix = self.emmebank.matrix(matrix) - if key is None: - key = self.generate_key(matrix) - numpy_array = matrix.get_numpy_data(self.scenario.id) - if matrix.type == "DESTINATION": - n_zones = len(numpy_array) - numpy_array = _numpy.resize(numpy_array, (1, n_zones)) - elif matrix.type == "ORIGIN": - n_zones = len(numpy_array) - numpy_array = _numpy.resize(numpy_array, (n_zones, 1)) - attrs = {"description": matrix.description.encode(text_encoding)} - self.write_array(numpy_array, key, attrs) - - def write_clipped_array(self, numpy_array, key, a_min, a_max=None, attrs={}): - if a_max is not None: - numpy_array = numpy_array.clip(a_min, a_max) - else: - numpy_array = numpy_array.clip(a_min) - self.write_array(numpy_array, key, attrs) - - def write_array(self, numpy_array, key, attrs={}): - shape = numpy_array.shape - if len(shape) == 2: - chunkshape = (1, shape[0]) - else: - chunkshape = None - attrs["source"] = "Emme" - numpy_array = numpy_array.astype(dtype="float64", copy=False) - omx_matrix = self.omx_file.create_matrix( - key, obj=numpy_array, chunkshape=chunkshape, attrs=attrs) - - -class OMXManager(object): - def __init__(self, directory, name_tmplt): - self._directory = directory - self._name_tmplt = name_tmplt - self._omx_files = {} - - def lookup(self, name_args, key): - file_name = self._name_tmplt % name_args - omx_file = self._omx_files.get(file_name) - if omx_file is None: - file_path = os.path.join(self._directory, file_name) - omx_file = _omx.open_file(file_path, 'r') - self._omx_files[file_name] = omx_file - return omx_file[key].read() - - def file_exists(self, name_args): - file_name = self._name_tmplt % name_args - file_path = os.path.join(self._directory, file_name) - return os.path.isfile(file_path) - - def zone_list(self, file_name): - omx_file = self._omx_files[file_name] - mapping_name = omx_file.list_mappings()[0] - zone_mapping = omx_file.mapping(mapping_name).items() - zone_mapping.sort(key=lambda x: x[1]) - omx_zones = [x[0] for x in zone_mapping] - return omx_zones - - def __enter__(self): - return self - - def __exit__(self, exc_type, exc_val, exc_tb): - for omx_file in self._omx_files.values(): - omx_file.close() - self._omx_files = {} - - -class CSVReader(object): - def __init__(self, path): - self._path = path - self._f = None - self._fields = None - - def __enter__(self): - self._f = open(self._path) - header = self._f.next() - self._fields = [h.strip().upper() for h in header.split(",")] - return self - - def __exit__(self, exception_type, exception_value, traceback): - self._f.close() - self._f = None - self._fields = None - - def __iter__(self): - return self - - @property - def fields(self): - return list(self._fields) - - def next(self): - line = self._f.next() - tokens = [t.strip() for t in line.split(",")] - return dict(zip(self._fields, tokens)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py b/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py deleted file mode 100644 index 67564f8..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/omxwrapper.py +++ /dev/null @@ -1,91 +0,0 @@ -##////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2019. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// utilities/omxwrapper.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#/////////////////////////////////////////////////////////////////////////////// -import inro.modeller as _m - - -try: - import openmatrix as _omx - - - def open_file(file_path, mode): - return OmxMatrix(_omx.open_file(file_path, mode)) -except Exception, e: - import omx as _omx - - - def open_file(file_path, mode): - return OmxMatrix(_omx.openFile(file_path, mode)) - -class OmxMatrix(object): - - def __init__(self, matrix): - self.matrix = matrix - - def mapping(self, name): - return self.matrix.mapping(name) - - def list_mappings(self): - return self.matrix.listMappings() - - def __getitem__(self, key): - return self.matrix[key] - - def __setitem__(self, key, value): - self.matrix[key] = value - - def create_mapping(self, name, ids): - exception_raised = False - try: - self.matrix.create_mapping(name, ids) # Emme 44 and above - except Exception, e: - exception_raised = True - - if exception_raised: - self.matrix.createMapping(name, ids) # Emme 437 - - - def create_matrix(self, key, obj, chunkshape, attrs): - exception_raised = False - try: # Emme 44 and above - self.matrix.create_matrix( - key, - obj=obj, - chunkshape=chunkshape, - attrs=attrs - ) - except Exception, e: - exception_raised = True - - if exception_raised: # Emme 437 - self.matrix.createMatrix( - key, - obj=obj, - chunkshape=chunkshape, - attrs=attrs - ) - - def close(self): - self.matrix.close() - - - -class OmxWrapper(_m.Tool()): - def page(self): - pb = _m.ToolPageBuilder( - self, - runnable=False, - title="OMX wrapper", - description="OMX utility for handling of OMX related libraries" - ) - return pb.render() \ No newline at end of file diff --git a/sandag_abm/src/main/emme/toolbox/utilities/properties.py b/sandag_abm/src/main/emme/toolbox/utilities/properties.py deleted file mode 100644 index 228bc07..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/properties.py +++ /dev/null @@ -1,599 +0,0 @@ -##////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// utilities/properties.py /// -#//// /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// - -TOOLBOX_ORDER = 103 - - -import inro.modeller as _m -import traceback as _traceback -from collections import OrderedDict -import csv -import os -import time - - -class PropertiesSetter(object): - - startFromIteration = _m.Attribute(int) - sample_rates = _m.Attribute(str) - - useLocalDrive = _m.Attribute(bool) - skip4Ds = _m.Attribute(bool) - skipBuildNetwork = _m.Attribute(bool) - skipInputChecker = _m.Attribute(bool) - skipInitialization = _m.Attribute(bool) - deleteAllMatrices = _m.Attribute(bool) - skipCopyWarmupTripTables = _m.Attribute(bool) - skipWalkLogsums = _m.Attribute(bool) - skipCopyWalkImpedance = _m.Attribute(bool) - skipBikeLogsums = _m.Attribute(bool) - skipCopyBikeLogsum = _m.Attribute(bool) - - skipHighwayAssignment_1 = _m.Attribute(bool) - skipHighwayAssignment_2 = _m.Attribute(bool) - skipHighwayAssignment_3 = _m.Attribute(bool) - skipTransitSkimming_1 = _m.Attribute(bool) - skipTransitSkimming_2 = _m.Attribute(bool) - skipTransitSkimming_3 = _m.Attribute(bool) - skipTransponderExport_1 = _m.Attribute(bool) - skipTransponderExport_2 = _m.Attribute(bool) - skipTransponderExport_3 = _m.Attribute(bool) - skipCoreABM_1 = _m.Attribute(bool) - skipCoreABM_2 = _m.Attribute(bool) - skipCoreABM_3 = _m.Attribute(bool) - skipOtherSimulateModel_1 = _m.Attribute(bool) - skipOtherSimulateModel_2 = _m.Attribute(bool) - skipOtherSimulateModel_3 = _m.Attribute(bool) - skipMAASModel_1 = _m.Attribute(bool) - skipMAASModel_2 = _m.Attribute(bool) - skipMAASModel_3 = _m.Attribute(bool) - skipCTM_1 = _m.Attribute(bool) - skipCTM_2 = _m.Attribute(bool) - skipCTM_3 = _m.Attribute(bool) - skipEI_1 = _m.Attribute(bool) - skipEI_2 = _m.Attribute(bool) - skipEI_3 = _m.Attribute(bool) - skipExternalExternal_1 = _m.Attribute(bool) - skipExternalExternal_2 = _m.Attribute(bool) - skipExternalExternal_3 = _m.Attribute(bool) - skipTruck_1 = _m.Attribute(bool) - skipTruck_2 = _m.Attribute(bool) - skipTruck_3 = _m.Attribute(bool) - skipTripTableCreation_1 = _m.Attribute(bool) - skipTripTableCreation_2 = _m.Attribute(bool) - skipTripTableCreation_3 = _m.Attribute(bool) - - skipFinalHighwayAssignment = _m.Attribute(bool) - skipFinalHighwayAssignmentStochastic = _m.Attribute(bool) - skipFinalTransitAssignment = _m.Attribute(bool) - skipVisualizer = _m.Attribute(bool) - skipDataExport = _m.Attribute(bool) - skipDataLoadRequest = _m.Attribute(bool) - skipDeleteIntermediateFiles = _m.Attribute(bool) - - def _get_list_prop(self, name): - return [getattr(self, name + suffix) for suffix in ["_1", "_2", "_3"]] - - def _set_list_prop(self, name, value): - try: - for v_sub, suffix in zip(value, ["_1", "_2", "_3"]): - setattr(self, name + suffix, v_sub) - except: - for suffix in ["_1", "_2", "_3"]: - setattr(self, name + suffix, False) - - skipHighwayAssignment = property( - fget=lambda self: self._get_list_prop("skipHighwayAssignment"), - fset=lambda self, value: self._set_list_prop("skipHighwayAssignment", value)) - skipTransitSkimming = property( - fget=lambda self: self._get_list_prop("skipTransitSkimming"), - fset=lambda self, value: self._set_list_prop("skipTransitSkimming", value)) - skipTransponderExport = property( - fget=lambda self: self._get_list_prop("skipTransponderExport"), - fset=lambda self, value: self._set_list_prop("skipTransponderExport", value)) - skipCoreABM = property( - fget=lambda self: self._get_list_prop("skipCoreABM"), - fset=lambda self, value: self._set_list_prop("skipCoreABM", value)) - skipOtherSimulateModel = property( - fget=lambda self: self._get_list_prop("skipOtherSimulateModel"), - fset=lambda self, value: self._set_list_prop("skipOtherSimulateModel", value)) - skipMAASModel = property( - fget=lambda self: self._get_list_prop("skipMAASModel"), - fset=lambda self, value: self._set_list_prop("skipMAASModel", value)) - skipCTM = property( - fget=lambda self: self._get_list_prop("skipCTM"), - fset=lambda self, value: self._set_list_prop("skipCTM", value)) - skipEI = property( - fget=lambda self: self._get_list_prop("skipEI"), - fset=lambda self, value: self._set_list_prop("skipEI", value)) - skipExternalExternal = property( - fget=lambda self: self._get_list_prop("skipExternalExternal"), - fset=lambda self, value: self._set_list_prop("skipExternalExternal", value)) - skipTruck = property( - fget=lambda self: self._get_list_prop("skipTruck"), - fset=lambda self, value: self._set_list_prop("skipTruck", value)) - skipTripTableCreation = property( - fget=lambda self: self._get_list_prop("skipTripTableCreation"), - fset=lambda self, value: self._set_list_prop("skipTripTableCreation", value)) - - def __init__(self): - self._run_model_names = ( - "useLocalDrive", "skip4Ds", "skipInputChecker", - "startFromIteration", "skipInitialization", "deleteAllMatrices", "skipCopyWarmupTripTables", - "skipCopyBikeLogsum", "skipCopyWalkImpedance", "skipWalkLogsums", "skipBikeLogsums", "skipBuildNetwork", - "skipHighwayAssignment", "skipTransitSkimming", "skipTransponderExport", "skipCoreABM", "skipOtherSimulateModel", "skipMAASModel","skipCTM", - "skipEI", "skipExternalExternal", "skipTruck", "skipTripTableCreation", "skipFinalHighwayAssignment", 'skipFinalHighwayAssignmentStochastic', - "skipFinalTransitAssignment", "skipVisualizer", "skipDataExport", "skipDataLoadRequest", - "skipDeleteIntermediateFiles") - self._properties = None - - def add_properties_interface(self, pb, disclosure=False): - tool_proxy_tag = pb.tool_proxy_tag - title = "Run model - skip steps" - - pb.add_text_box('sample_rates', title="Sample rate by iteration:", size=20) - - contents = [""" -
    -
    - -            - -
    - - - - - - - - """ % {"tool_proxy_tag": tool_proxy_tag}] - - skip_startup_items = [ - ("useLocalDrive", "Use the local drive during the model run"), - ("skip4Ds", "Skip running 4Ds"), - ("skipBuildNetwork", "Skip build of highway and transit network"), - ("skipInputChecker", "Skip running input checker"), - ("skipInitialization", "Skip matrix and transit database initialization"), - ("deleteAllMatrices", "    Delete all matrices"), - ("skipCopyWarmupTripTables","Skip import of warmup trip tables"), - ("skipWalkLogsums", "Skip walk logsums"), - ("skipCopyWalkImpedance", "Skip copy of walk impedance"), - ("skipBikeLogsums", "Skip bike logsums"), - ("skipCopyBikeLogsum", "Skip copy of bike logsum"), - ] - skip_per_iteration_items = [ - ("skipHighwayAssignment", "Skip highway assignments and skims"), - ("skipTransitSkimming", "Skip transit skims"), - ("skipTransponderExport", "Skip transponder accessibilities"), - ("skipCoreABM", "Skip core ABM"), - ("skipOtherSimulateModel", "Skip other simulation model"), - ("skipMAASModel", "Skip MAAS model"), - ("skipCTM", "Skip commercial vehicle sub-model"), - ("skipTruck", "Skip truck sub-model"), - ("skipEI", "Skip external-internal sub-model"), - ("skipExternalExternal", "Skip external-external sub-model"), - ("skipTripTableCreation", "Skip trip table creation"), - ] - skip_final_items = [ - ("skipFinalHighwayAssignment", "Skip final highway assignments"), - ("skipFinalHighwayAssignmentStochastic", "    Skip stochastic assignment"), - ("skipFinalTransitAssignment", "Skip final transit assignments"), - ("skipVisualizer", "Skip running visualizer"), - ("skipDataExport", "Skip data export"), - ("skipDataLoadRequest", "Skip data load request"), - ("skipDeleteIntermediateFiles", "Skip delete intermediate files"), - ] - - if disclosure: - contents.insert(0, """ -
    -
    %s
    """ % title) - title = "" - - checkbox = '
    ' - checkbox_no_data = '' - - for name, label in skip_startup_items: - contents.append("" % label) - contents.append(checkbox % {"name": name, "tag": tool_proxy_tag}) - contents.append("") - contents.append("") - for i in range(1,4): - contents.append(checkbox_no_data % {"name": "all" + "_" + str(i)}) - for name, label in skip_per_iteration_items: - contents.append("" % label) - for i in range(1,4): - contents.append(checkbox % {"name": name + "_" + str(i), "tag": tool_proxy_tag}) - for name, label in skip_final_items: - contents.append("" % label) - contents.append("") - contents.append(checkbox % {"name": name, "tag": tool_proxy_tag}) - - contents.append("
    Iteration 1Iteration 2Iteration 3
    %s
    Set / reset all
        %s
    %s
    ") - if disclosure: - contents.append("") - - pb.wrap_html(title, "".join(contents)) - - pb.add_html(""" -""" % {"tool_proxy_tag": tool_proxy_tag, - "iter_items": str([x[0] for x in skip_per_iteration_items]), - "startup_items": str([x[0] for x in skip_startup_items]), - }) - return - - @_m.method(return_type=bool, argument_types=(str,)) - def get_value(self, name): - return bool(getattr(self, name)) - - @_m.method() - def load_properties(self): - if not os.path.exists(self.properties_path): - return - self._properties = props = Properties(self.properties_path) - _m.logbook_write("SANDAG properties interface load") - - self.startFromIteration = props.get("RunModel.startFromIteration", 1) - self.sample_rates = ",".join(str(x) for x in props.get("sample_rates")) - - self.useLocalDrive = props.get("RunModel.useLocalDrive", True) - self.skip4Ds = props.get("RunModel.skip4Ds", False) - self.skipBuildNetwork = props.get("RunModel.skipBuildNetwork", False) - self.skipInputChecker = props.get("RunModel.skipInputChecker", False) - self.skipInitialization = props.get("RunModel.skipInitialization", False) - self.deleteAllMatrices = props.get("RunModel.deleteAllMatrices", False) - self.skipCopyWarmupTripTables = props.get("RunModel.skipCopyWarmupTripTables", False) - self.skipWalkLogsums = props.get("RunModel.skipWalkLogsums", False) - self.skipCopyWalkImpedance = props.get("RunModel.skipCopyWalkImpedance", False) - self.skipBikeLogsums = props.get("RunModel.skipBikeLogsums", False) - self.skipCopyBikeLogsum = props.get("RunModel.skipCopyBikeLogsum", False) - - self.skipHighwayAssignment = props.get("RunModel.skipHighwayAssignment", [False, False, False]) - self.skipTransitSkimming = props.get("RunModel.skipTransitSkimming", [False, False, False]) - self.skipTransponderExport = props.get("RunModel.skipTransponderExport", [False, False, False]) - self.skipCoreABM = props.get("RunModel.skipCoreABM", [False, False, False]) - self.skipOtherSimulateModel = props.get("RunModel.skipOtherSimulateModel", [False, False, False]) - self.skipMAASModel = props.get("RunModel.skipMAASModel", [False, False, False]) - self.skipCTM = props.get("RunModel.skipCTM", [False, False, False]) - self.skipEI = props.get("RunModel.skipEI", [False, False, False]) - self.skipExternalExternal = props.get("RunModel.skipExternalExternal", [False, False, False]) - self.skipTruck = props.get("RunModel.skipTruck", [False, False, False]) - self.skipTripTableCreation = props.get("RunModel.skipTripTableCreation", [False, False, False]) - - self.skipFinalHighwayAssignment = props.get("RunModel.skipFinalHighwayAssignment", False) - self.skipFinalHighwayAssignmentStochastic = props.get("RunModel.skipFinalHighwayAssignmentStochastic", True) - self.skipFinalTransitAssignment = props.get("RunModel.skipFinalTransitAssignment", False) - self.skipVisualizer = props.get("RunModel.skipVisualizer", False) - self.skipDataExport = props.get("RunModel.skipDataExport", False) - self.skipDataLoadRequest = props.get("RunModel.skipDataLoadRequest", False) - self.skipDeleteIntermediateFiles = props.get("RunModel.skipDeleteIntermediateFiles", False) - - def save_properties(self): - props = self._properties - props["RunModel.startFromIteration"] = self.startFromIteration - props["sample_rates"] = [float(x) for x in self.sample_rates.split(",")] - - props["RunModel.useLocalDrive"] = self.useLocalDrive - props["RunModel.skip4Ds"] = self.skip4Ds - props["RunModel.skipBuildNetwork"] = self.skipBuildNetwork - props["RunModel.skipInputChecker"] = self.skipInputChecker - props["RunModel.skipInitialization"] = self.skipInitialization - props["RunModel.deleteAllMatrices"] = self.deleteAllMatrices - props["RunModel.skipCopyWarmupTripTables"] = self.skipCopyWarmupTripTables - props["RunModel.skipWalkLogsums"] = self.skipWalkLogsums - props["RunModel.skipCopyWalkImpedance"] = self.skipCopyWalkImpedance - props["RunModel.skipBikeLogsums"] = self.skipBikeLogsums - props["RunModel.skipCopyBikeLogsum"] = self.skipCopyBikeLogsum - - props["RunModel.skipHighwayAssignment"] = self.skipHighwayAssignment - props["RunModel.skipTransitSkimming"] = self.skipTransitSkimming - props["RunModel.skipTransponderExport"] = self.skipTransponderExport - props["RunModel.skipCoreABM"] = self.skipCoreABM - props["RunModel.skipOtherSimulateModel"] = self.skipOtherSimulateModel - props["RunModel.skipMAASModel"] = self.skipMAASModel - props["RunModel.skipCTM"] = self.skipCTM - props["RunModel.skipEI"] = self.skipEI - props["RunModel.skipExternalExternal"] = self.skipExternalExternal - props["RunModel.skipTruck"] = self.skipTruck - props["RunModel.skipTripTableCreation"] = self.skipTripTableCreation - - props["RunModel.skipFinalHighwayAssignment"] = self.skipFinalHighwayAssignment - props["RunModel.skipFinalHighwayAssignmentStochastic"] = self.skipFinalHighwayAssignmentStochastic - props["RunModel.skipFinalTransitAssignment"] = self.skipFinalTransitAssignment - props["RunModel.skipVisualizer"] = self.skipVisualizer - props["RunModel.skipDataExport"] = self.skipDataExport - props["RunModel.skipDataLoadRequest"] = self.skipDataLoadRequest - props["RunModel.skipDeleteIntermediateFiles"] = self.skipDeleteIntermediateFiles - - props.save() - - # Log current state of props interface for debugging of UI / file sync issues - tool_attributes = dict((name, getattr(self, name)) for name in self._run_model_names) - _m.logbook_write("SANDAG properties interface save", attributes=tool_attributes) - - -class PropertiesTool(PropertiesSetter, _m.Tool()): - - properties_path = _m.Attribute(unicode) - - def __init__(self): - super(PropertiesTool, self).__init__() - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.properties_path = os.path.join( - os.path.dirname(project_dir), "conf", "sandag_abm.properties") - - tool_run_msg = "" - - @_m.method(return_type=_m.UnicodeType) - def tool_run_msg_status(self): - return self.tool_run_msg - - def page(self): - if os.path.exists(self.properties_path): - self.load_properties() - pb = _m.ToolPageBuilder(self) - pb.title = 'Set properties' - pb.description = """Properties setting tool.""" - pb.branding_text = ' - SANDAG - Utilities' - tool_proxy_tag = pb.tool_proxy_tag - - pb.add_select_file('properties_path', 'file', title='Path to properties file:') - - pb.wrap_html("", """ -
    """) - - pb.add_html(""" -""" % {"tool_proxy_tag": tool_proxy_tag}) - self.add_properties_interface(pb) - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - self.save_properties() - message = "Properties file saved" - self.tool_run_msg = _m.PageBuilder.format_info(message, escape=False) - except Exception, e: - self.tool_run_msg = _m.PageBuilder.format_exception( - e, _traceback.format_exc(e)) - raise - - def __call__(self, file_path): - return Properties(file_path) - - -class Properties(object): - - def __init__(self, path): - if os.path.isdir(path): - path = os.path.join(path, "sandag_abm.properties") - if not os.path.isfile(path): - raise Exception("properties files does not exist '%s'" % path) - self._path = os.path.normpath(os.path.abspath(path)) - self.load_properties() - - def load_properties(self): - self._prop = prop = OrderedDict() - self._comments = comments = {} - with open(self._path, 'r') as properties: - comment = [] - for line in properties: - line = line.strip() - if not line or line.startswith('#'): - comment.append(line) - continue - key, value = line.split('=') - key = key.strip() - tokens = value.split(',') - if len(tokens) > 1: - value = self._parse_list(tokens) - else: - value = self._parse(value) - prop[key] = value - comments[key], comment = comment, [] - self._timestamp = os.path.getmtime(self._path) - - def _parse_list(self, values): - converted_values = [] - for v in values: - converted_values.append(self._parse(v)) - return converted_values - - def _parse(self, value): - value = str(value).strip() - if value == 'true': - return True - elif value == 'false': - return False - for caster in int, float: - try: - return caster(value) - except ValueError: - pass - return value - - def _format(self, value): - if isinstance(value, bool): - return "true" if value else "false" - return str(value) - - def save(self, path=None): - if not path: - path = self._path - # check for possible interference if user edits the - # properties files directly while it is already open in Modeller - timestamp = os.path.getmtime(path) - if timestamp != self._timestamp: - raise Exception("%s file conflict - edited externally after loading" % path) - self["SavedFrom"] = "Emme Modeller properties writer Process ID %s" % os.getpid() - self["SavedLast"] = time.strftime("%b-%d-%Y %H:%M:%S") - with open(path, 'w') as f: - for key, value in self.iteritems(): - if isinstance(value, list): - value = ",".join([self._format(v) for v in value]) - else: - value = self._format(value) - comment = self._comments.get(key) - if comment: - for line in comment: - f.write(line) - f.write("\n") - f.write("%s = %s\n" % (key, value)) - self._timestamp = os.path.getmtime(path) - - def set_year_specific_properties(self, file_path): - with open(file_path, 'r') as f: - reader = csv.DictReader(f) - properties_by_year = {} - for row in reader: - year = str(row.pop("year")) - properties_by_year[year] = row - year_properties = properties_by_year.get(str(self["scenarioBuild"])) - if year_properties is None: - raise Exception("Row with year %s not found in %s" % (self["scenarioBuild"], file_path)) - self.update(year_properties) - - def __setitem__(self, key, item): - self._prop[key] = item - - def __getitem__(self, key): - return self._prop[key] - - def __repr__(self): - return "Properties(%s)" % self._path - - def __len__(self): - return len(self._prop) - - def __delitem__(self, key): - del self._prop[key] - - def clear(self): - return self._prop.clear() - - def has_key(self, k): - return self._prop.has_key(k) - - def pop(self, k, d=None): - return self._prop.pop(k, d) - - def update(self, *args, **kwargs): - return self._prop.update(*args, **kwargs) - - def keys(self): - return self._prop.keys() - - def values(self): - return self._prop.values() - - def items(self): - return self._prop.items() - - def iteritems(self): - return self._prop.iteritems() - - def pop(self, *args): - return self._prop.pop(*args) - - def get(self, k, default=None): - try: - return self[k] - except KeyError: - return default - - def __cmp__(self, dict): - return cmp(self._prop, dict) - - def __contains__(self, item): - return item in self._prop - - def __iter__(self): - return iter(self._prop) - - def __unicode__(self): - return unicode(repr(self._prop)) diff --git a/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py b/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py deleted file mode 100644 index ebaf2e1..0000000 --- a/sandag_abm/src/main/emme/toolbox/utilities/run_summary.py +++ /dev/null @@ -1,326 +0,0 @@ -""" ABM Run Time Summary Tool - -Generates a CSV file containing a run time summary for -a completed ABM run. Utilizes the Emme Modeler API to -query the Emme logbook for the run time information. - -""" - -# Importing libraries -import os -import pandas as pd -import traceback as _traceback -import inro.emme.desktop.app as _app -import inro.modeller as _m -from functools import reduce - -_dir = os.path.dirname -_join = os.path.join - -ATTR_SUFFIX = "_304A7365_C276_493A_AB3B_9B2D195E203F" - -# Define unneeded entries -exclude = ('Copy project data to local drive', - 'Export results for transponder ownership model', - 'Check free space on C', - 'Data load request', - 'Delete', - 'Move', - 'Create drive', - 'Start matrix', - 'Start JPPF', - 'Start Hh', - 'Start HH') - - -class RunTime(_m.Tool()): - - def __init__(self): - project_dir = _dir(_m.Modeller().desktop.project.path) - self.path = _dir(project_dir) - self.output_path = '' - self.output_summary_path = '' - self.begin = '' - self.end = '' - - def run(self): - """ - Executes Run Time Summary tool - """ - self.tool_run_msg = "" - try: - self(path=self.path) - run_msg = "Run Time Tool Complete" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg, - escape=False) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - - return - - def __call__(self, path=""): - """ - Calculates ABM run times and saves to CSV file - - :param path: Scenario file path - """ - - # Get element IDs for model runs - run_ids = self.get_runs() - - # Define needed attributes (begin and end times) - self.begin = "begin" + ATTR_SUFFIX - self.end = "end" + ATTR_SUFFIX - attrs = [self.begin, self.end] - - runtime_dfs = [] - for run_id in run_ids: - name = 'Run ID: {}'.format(run_id) - - # Creating dummy total time - total_entry = (run_id, 'Total Run Time', '0:0') - - # Get second level child entry run times if they exist - child_runtimes = self.get_child_runtimes(run_id, attrs) - - # Get third (final) level child entry run times - final_runtimes = [total_entry] - for index, info in enumerate(child_runtimes): - if info[1] == 'Final traffic assignments': - final_runtimes.append([0, 'Iteration 4', 0]) - final_runtimes.append(info) - if 'Iteration' in info[1]: - iter_str = '_{}'.format(info[1]) - - # Manually inserting matrix, hh, node, and jppf runtimes - start_proc = "Start Matrix manager, JPPF Driver, " + \ - "HH manager, and Nodes manager" + iter_str - final_runtimes += [[0, start_proc, '0:01']] - - # Add iteration to children - iteration_children = self.get_child_runtimes( - info[0], attrs) - for index, child in enumerate(iteration_children): - step = child[1] - iteration_children[index][1] = step + iter_str - - final_runtimes += iteration_children - - # Create run time summary table - index = [x[1] for x in final_runtimes] - values = [x[2] for x in final_runtimes] - runtime_series = pd.Series(index=index, data=values) - runtime_series.name = name - runtime_df = runtime_series.to_frame() - - # Create intial time - zero_time = pd.to_datetime('0:0', format='%H:%M') - - # Calculate iteration 4 run time if it exists - iter_str = 'Iteration 4' - if iter_str in runtime_df.index: - iter_4_index = runtime_df.index.get_loc('Iteration 4') - iter_4_df = runtime_df.iloc[iter_4_index+1:, :].copy() - iter_4_df[name] = (pd.to_datetime( - iter_4_df[name], format='%H:%M') - - zero_time) - iter_4_time = iter_4_df[name].sum() - runtime_df.loc[iter_str, :] = self.format_runtime(iter_4_time) - - # Calculate total runtime - is_iter_row = pd.Series(runtime_df.index).str.startswith('Iter') - total_df = runtime_df[~is_iter_row.values].copy() - total_df[name] = (pd.to_datetime(total_df[name], format='%H:%M') - - zero_time) - total_time = total_df[name].sum() - run_str = 'Total Run Time' - runtime_df.loc[run_str, :] = self.format_runtime(total_time) - - # Remove unneeded entries - is_excluded = pd.Series(runtime_df.index).str.startswith(exclude) - runtime_df = runtime_df[~(is_excluded.values)] - runtime_dfs.append(runtime_df) - - # Merge all run time data frames if more than one exists and save - file_name = 'runtime_summary.csv' - self.output_path = _join(path, 'output', file_name) - result = self.combine_dfs(runtime_dfs) - if result[1]: - result[0].to_csv(self.output_path, header=True, index=False) - else: - result[0].to_csv(self.output_path, header=False) - - return - - def get_runs(self): - """ - Queries the Emme logbook to retrieve the IDs of all - model runs. - - :returns: List of IDs of model runs - """ - - # Emme logbook query - query = """ - SELECT elements.element_id, elements.tag - FROM elements - JOIN attributes KEYVAL1 ON (elements.element_id=KEYVAL1.element_id) - WHERE (KEYVAL1.name=="self" - AND KEYVAL1.value LIKE "sandag.master_run") - ORDER BY elements.element_id ASC - """ - all_entries = _m.logbook_query(query) - - # Retrieves model run IDs - run_ids = [] - for entry in all_entries: - parent_id = entry[0] - run_ids.append(parent_id) - - if len(run_ids) == 0: - raise ValueError('A model run does not exist.') - - return run_ids - - def get_attributes(self, element_id): - """ - Queries all the attributes of an Emme logbook element - - :param element_id: Integer ID of element - :returns: List of tuples containing information for - different attributes of an element. - """ - - # Emme logbook query - query = """ - SELECT name, value FROM attributes - WHERE attributes.element_id == %i - """ % element_id - - return _m.logbook_query(query) - - def format_runtime(self, time): - """ - Transforms a datetime object to a reformatted - date string. Formatted as '{hours}:{'minutes'}' - - :param time Datetime object - """ - - hours = str(int(time.total_seconds() // 3600)) - minutes = str(int((time.total_seconds() % 3600) // 60)).zfill(2) - formatted_runtime = hours + ":" + minutes - - return formatted_runtime - - def calc_runtime(self, begin, end): - """ - Helper function for get_child_runtimes - - Converts beginning and end datetime strings into - a formatted time delta. Formatted as '{hours}:{minutes}' - - :param begin: String representing beginning date - :param end: String representating ending date - :returns: String representing element runtime - """ - - # Calculate total run time - total_runtime = pd.to_datetime(end) - pd.to_datetime(begin) - - # Format run time: '{hours}:{minutes}' - formatted_runtime = self.format_runtime(total_runtime) - - # Defaulting zero second times to 1 second - if formatted_runtime == '0:00': - formatted_runtime = '0:01' - - return formatted_runtime - - def get_children(self, parent_id): - """ - Retrieves all child elements for a parent element - - :param parent_id: Integer ID of parent element - :returns: List of tuples containing child IDs and names - """ - - # Emme logbook query - query = """ - SELECT elements.element_id, elements.tag - FROM elements WHERE parent_id==%i - ORDER BY elements.element_id ASC - """ % parent_id - child_entries = _m.logbook_query(query) - - return child_entries - - def get_child_runtimes(self, parent_id, attrs): - """ - Calculates the run times for the child elements of - a parent element - - :param parent_id: Integer ID of parent element - :param attrs: List of strings representing attributes to query - :returns: List of tuples containing information for child elements - """ - - # Get child elements - all_child_entries = self.get_children(parent_id) - - # Calculates run times for each child element - runtime_child_entries = [] - for element_id, name in all_child_entries: - attributes = dict(self.get_attributes(element_id)) - - # Gets element information if desired attribute is - # available and it is not included in the excluded list - if attrs[0] in attributes: - begin = attributes[attrs[0]] - - # Handles cases where model fails mid iteration - try: - end = attributes[attrs[1]] - runtime = self.calc_runtime(begin, end) - except KeyError: - end = None - runtime = None - runtime_child_entries.append([element_id, name, runtime]) - - return runtime_child_entries - - def combine_dfs(self, df_list): - """ - Combines a list of Pandas DataFrames into a single - summary DataFrame - - :param df_list: List of Pandas DataFrames - :returns: Tuple contianing single run time summary DataFrame - and boolean whether it contains multiple runs - """ - if len(df_list) > 1: - # Drop tables with less than 2 entries - final_dfs = [] - for df in df_list: - if len(df.dropna()) > 1: - final_dfs.append(df.reset_index(drop=False)) - - # Merge all data frames - final_df = reduce(lambda left, right: - pd.merge(left, right, on=['index'], how='outer'), - final_dfs) - - # Remove appended iteration markers - final_df['index'] = (final_df['index'].apply( - lambda x: x.split('_')[0])) - - final_df = final_df.rename(columns={'index': 'Step'}) - result = (final_df, True) - - else: - final_df = df_list[0] - result = (final_df, False) - - return result diff --git a/sandag_abm/src/main/emme/toolbox/validation/validation.py b/sandag_abm/src/main/emme/toolbox/validation/validation.py deleted file mode 100644 index 13ae759..0000000 --- a/sandag_abm/src/main/emme/toolbox/validation/validation.py +++ /dev/null @@ -1,290 +0,0 @@ -""" -Created on March 2020 - -@author: cliu -""" - -TOOLBOX_ORDER = 105 - -import inro.modeller as _m -import traceback as _traceback -import inro.emme.database.emmebank as _eb -import inro.emme.desktop.app as _app -import inro.emme.core.exception as _except -from collections import OrderedDict -import os -import pandas as pd -import openpyxl -from functools import reduce - - -gen_utils = _m.Modeller().module("sandag.utilities.general") -dem_utils = _m.Modeller().module("sandag.utilities.demand") - - -format = lambda x: ("%.6f" % x).rstrip('0').rstrip(".") -id_format = lambda x: str(int(x)) - -class validation(_m.Tool(), gen_utils.Snapshot): - - main_directory = _m.Attribute(str) - base_scenario_id = _m.Attribute(int) - traffic_emmebank = _m.Attribute(str) - #transit_emmebank = _m.Attribute(str) - attributes = _m.Attribute(str) - - tool_run_msg = "" - - def __init__(self): - project_dir = os.path.dirname(_m.Modeller().desktop.project.path) - self.main_directory = os.path.dirname(project_dir) - self.base_scenario_id = 100 - self.traffic_emmebank = os.path.join(project_dir, "Database", "emmebank") - #self.transit_emmebank = os.path.join(project_dir, "Database_transit", "emmebank") - self.attributes = ["main_directory", "traffic_emmebank", "transit_emmebank","base_scenario_id"] - - def page(self): - pb = _m.ToolPageBuilder(self) - pb.title = "Validation Procedure" - pb.description = """ -Export traffic flow to Excel files for base year validation.""" - pb.branding_text = "- SANDAG - Validation" - if self.tool_run_msg != "": - pb.tool_run_status(self.tool_run_msg_status) - - pb.add_select_file('main_directory', 'directory', - title='Select main directory') - - pb.add_select_file('traffic_emmebank', 'file', - title='Select traffic emmebank') - #pb.add_select_file('transit_emmebank', 'file', - # title='Select transit emmebank') - return pb.render() - - def run(self): - self.tool_run_msg = "" - try: - results = self(self.main_directory, self.traffic_emmebank, self.base_scenario_id) - #results = self(self.main_directory, self.traffic_emmebank, self.transit_emmebank, self.base_scenario_id) - run_msg = "Export completed" - self.tool_run_msg = _m.PageBuilder.format_info(run_msg) - except Exception as error: - self.tool_run_msg = _m.PageBuilder.format_exception( - error, _traceback.format_exc(error)) - raise - @_m.logbook_trace("Export network data for base year Validation", save_arguments=True) - - def __call__(self, main_directory, traffic_emmebank, base_scenario_id): - #def __call__(self, main_directory, traffic_emmebank, transit_emmebank, base_scenario_id): - print "in validation module" - attrs = { - "main_directory": main_directory, - "traffic_emmebank": str(traffic_emmebank), - #"transit_emmebank": str(transit_emmebank), - "base_scenario_id": base_scenario_id, - "self": str(self) - } - - gen_utils.log_snapshot("Validation procedure", str(self), attrs) - - traffic_emmebank = _eb.Emmebank(traffic_emmebank) - #transit_emmebank = _eb.Emmebank(transit_emmebank) - export_path = os.path.join(main_directory, "analysis/validation") - transitbank_path = os.path.join(main_directory, "emme_project/Database_transit/emmebank") - source_file = os.path.join(export_path, "source_EMME.xlsx") - df = pd.read_excel(source_file, header=None, sheet_name='raw') - writer = pd.ExcelWriter(source_file, engine='openpyxl') - book = openpyxl.load_workbook(source_file) - writer.book = book - writer.sheets = dict((ws.title, ws) for ws in book.worksheets) - - #periods = ["EA"] - periods = ["EA", "AM", "MD", "PM", "EV"] - - period_scenario_ids = OrderedDict((v, i) for i, v in enumerate(periods, start=int(base_scenario_id) + 1)) - - #-------export tranffic data-------- - dfHwycov = pd.read_excel(source_file, sheetname='raw', usecols ="A") - for p, scen_id in period_scenario_ids.iteritems(): - base_scenario = traffic_emmebank.scenario(scen_id) - - #create and calculate @trk_non_pce - create_attribute = _m.Modeller().tool( - "inro.emme.data.extra_attribute.create_extra_attribute") - net_calculator = _m.Modeller().tool( - "inro.emme.network_calculation.network_calculator") - try: - att = create_attribute("LINK", "@trk_non_pce", "total trucs in non-Pce", 0, overwrite=True, scenario = base_scenario) - except: #if "@trk_non_pce" has been created - pass - cal_spec = {"result": att.id, - "expression": "@trk_l_non_pce+@trk_m_non_pce+@trk_h_non_pce", - "aggregation": None, - "selections": {"link": "mode=d"}, - "type": "NETWORK_CALCULATION" - } - net_calculator(cal_spec, scenario = base_scenario) - - network = base_scenario.get_partial_network(["LINK"], include_attributes=True) - - df, dftr, dfsp = self.export_traffic(export_path, traffic_emmebank, scen_id, network, source_file, p, dfHwycov) - dfsp[p + "_Speed"] = dfsp[p + "_Speed"].astype(int) - if p == "EA": - df_total = df - dftr_total = dftr - dfsp_total = dfsp - else: - df_total = df_total.join(df[p + "_Flow"]) - dftr_total = dftr_total.join(dftr[p + "_TruckFlow"]) - dfsp_total = dfsp_total.join(dfsp[p + "_Speed"]) - - df_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=0, startrow=0) - dftr_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=7, startrow=0) - dfsp_total.to_excel(writer, sheet_name='raw', header=True, index=False, startcol=14, startrow=0) - writer.save() - - #----------------------------------export transit data---------------------------------- - - desktop = _m.Modeller().desktop - data_explorer = desktop.data_explorer() - - try: - data_explorer.add_database(transitbank_path) - except: - pass #if transit database already included in the project - all_databases = data_explorer.databases() - for database in all_databases: - if "transit" in database.name(): - database.open() - break - for p, scen_id in period_scenario_ids.iteritems(): - scen = database.scenario_by_number(scen_id) - data_explorer.replace_primary_scenario(scen) - self.export_transit(export_path, desktop, p) - - # -----------------close or remove transit databack from the project----------------- - database.close() - if "T:" not in main_directory: - data_explorer.remove_database(database) - all_databases = data_explorer.databases() - for database in all_databases: - if "transit" not in database.name(): - database.open() - break - - #------Combine into one datafram and write out - routeDict = {'c':23, 'l':24, 'y':25, 'r':26, 'p':27, 'e':28, 'b':29} - filenames = [] - for p in periods: - file = os.path.join(export_path, "transit_" + p + ".csv") - filenames.append(file) - - df_detail = [] - df_board = [] - df_passMile = [] - - for f in filenames: - df_detail.append(pd.read_csv(f)) - - pnum = 0 - for datafm in df_detail: - datafm['route'] = datafm.apply(lambda row: str(int(row.Line/1000))+row.Mode, axis = 1) - df_board.append(datafm.groupby(['route'])['Pass.'].agg('sum').reset_index()) - df_board[pnum].rename(columns = {'Pass.':periods[pnum]+'_Board'}, inplace=True) - - df_passMile.append(datafm.groupby(['route'])['Pass. dist.'].agg('sum').reset_index()) - df_passMile[pnum].rename(columns = {'Pass. dist.':periods[pnum]+'_PsgrMile'}, inplace=True) - - pnum += 1 - - frame1 = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), df_board ) - frame2 = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), df_passMile ) - frame = reduce(lambda x,y: pd.merge(x,y, on='route', how='outer'), [frame1,frame2]) - - idx = 1 - frame.insert(loc=idx, column='mode_transit_route_id', value="") - frame['mode_transit_route_id'] = frame['route'].apply(lambda x: routeDict[x[-1]]) - frame['route'] = frame['route'].apply(lambda x: int(x[:-1])) - - writer = pd.ExcelWriter(source_file, engine='openpyxl') - book = openpyxl.load_workbook(source_file) - writer.book = book - writer.sheets = dict((ws.title, ws) for ws in book.worksheets) - frame.to_excel(writer, sheet_name='transit_general', header=True, index=False, startcol=1, startrow=0) - writer.save() - - for p in periods: - file = os.path.join(export_path, "transit_" + p + ".csv") - os.remove(file) - - @_m.logbook_trace("Export traffic load data by period - validaton") - def export_traffic(self, export_path, traffic_emmebank, scen_id, network, filename, period, dfHwycov): - def get_network_value(attrs, emmeAttrName, headerStr, df): - reverse_link = link.reverse_link - key, att = attrs[0] # expected to be the link id - values = [id_format(link[att])] - reverse_link = link.reverse_link - for key, att in attrs[1:]: - if key == "AN": - values.append(link.i_node.id) - elif key == "BN": - values.append(link.j_node.id) - elif key.startswith("BA"): - print "line 148 key, att", key, att - name, default = att - if reverse_link and (abs(link["@tcov_id"]) == abs(reverse_link["@tcov_id"])): - values.append(format(reverse_link[name])) - else: - values.append(default) - elif att.startswith("#"): - values.append('"%s"' % link[att]) - else: - values.append(format(link[emmeAttrName])) - df = df.append({'TCOVID': values[0], period + headerStr: values[1]}, ignore_index=True) - df['TCOVID'] = df['TCOVID'].astype(float) - df[period + headerStr] = df[period + headerStr].astype(float) - return df - - # only the original forward direction links and auto links only - hwyload_attrs = [("TCOVID", "@tcov_id"), (period + "_Flow", "@non_pce_flow")] - trkload_attrs = [("TCOVID", "@tcov_id"), (period + "_TruckFlow", "@trk_non_pce")] - speedload_attrs = [("TCOVID", "@tcov_id"), (period + "_Speed", "@speed")] - df = pd.DataFrame(columns=['TCOVID', period + "_Flow"]) - dftr = pd.DataFrame(columns=['TCOVID', period + "_TruckFlow"]) - dfsp = pd.DataFrame(columns=['TCOVID', period + "_Speed"]) - print "scen_id", scen_id - auto_mode = network.mode("d") - scenario = traffic_emmebank.scenario(scen_id) - links = [l for l in network.links() if (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes))] - #links = [l for l in network.links() if l["@tcov_id"] > 0 and (auto_mode in l.modes or (l.reverse_link and auto_mode in l.reverse_link.modes))] - links.sort(key=lambda l: l["@tcov_id"]) - - for link in links: - if link["@tcov_id"] in dfHwycov['TCOVID'].tolist(): - reverse_link = link.reverse_link - df = get_network_value(hwyload_attrs, '@non_pce_flow', "_Flow", df) - dftr = get_network_value(trkload_attrs, '@trk_non_pce', "_TruckFlow", dftr) - dfsp = get_network_value(speedload_attrs, '@speed', "_Speed", dfsp) - return df, dftr, dfsp - - @_m.logbook_trace("Export transit load data by period - validaton") - def export_transit(self, export_path, desktop, p): - project_table_db = desktop.project.data_tables() - ws_path = ["General", "Results Analysis", "Transit", "Summaries", "Summary by line"] - root_ws_f = desktop.root_worksheet_folder() - table_item = root_ws_f.find_item(ws_path) - transit_table = table_item.open() - - #for i in delete_table_list: - # transit_table.delete_column(i) - transit_dt = transit_table.save_as_data_table("Transit_Summary", overwrite=True) - transit_table.close() - - transit_data = transit_dt.get_data() - project_path = r'T:\ABM\ABM_FY19\model_runs\ABM2Plus\SenTests4TAC' - transit_filepath = os.path.join(export_path, "transit_"+ p +".csv") - transit_data.export_to_csv(transit_filepath, separator=",") - - @_m.method(return_type=unicode) - def tool_run_msg_status(self): - return self.tool_run_msg \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/SandagCommon.rsc b/sandag_abm/src/main/gisdk/SandagCommon.rsc deleted file mode 100644 index ac345f1..0000000 --- a/sandag_abm/src/main/gisdk/SandagCommon.rsc +++ /dev/null @@ -1,405 +0,0 @@ -//**************************************************************** -//**************************************************************** -//Common Macros -// -//Find a string in a text file -//read properties -//Export Matrix to CSV -//Export Matrix -//Matrix Size -//Create Matrix -//Aggregate Matrices -//Go GetMatrixCoreNames -//Get SL Query # -//close all -//CloseViews -//date and time -//SDdeletefile -//SDcopyfile -//SDrenamefile -//HwycadLog -//ForecastYearStr -//ForecastYearInt -//DeleteInterimFiles -//FileCheckDelete -//**************************************************************** -//**************************************************************** -Macro "find String"(file,key) -//file a string in a text file -//file--text file name; key--string to be found in text file -//return 0 is string found, otherwise return 1 -//wsu 6-20-2014 - shared path, path_study - result=1 - -// Get file name - dif2=GetDirectoryInfo(path+"\\"+file, "file") - if dif2.length>0 then do //use scenario file - fptr=openfile(path+"\\"+file,"r") - end - else do //use study file from data directory - fptr=openfile(path_study+"\\data\\"+file,"r") - end - - while not FileAtEOF(fptr) do - pos = PositionFrom(,ReadLine(fptr),key) - if pos>0 then do - result=0 - end - end - CloseFile(fptr) - Return(result) -EndMacro - -Macro "read properties"(file,key,ctype) - //ctype as string - Character Type - Valid "I" or anything else - //macro only reads integers and strings - //reads property as string and returns either an integer or a string - - shared path, path_study - -// Get file name - dif2=GetDirectoryInfo(path+"\\"+file, "file") - if dif2.length>0 then do //use scenario file - fptr=openfile(path+"\\"+file,"r") - end - else do //use study file from data directory - fptr=openfile(path_study+"\\data\\"+file,"r") - end - -// Search key in properties file - a = ReadArray(fptr) - for k=1 to a.length do - // search for the key (line number is stored as k value) - pos1 = position(a[k],key) - if pos1 =1 then do - // gets the integer on the rightside of "=" - keyword=ParseString(a[k], "=") - keyvaltrim=trim(keyword[2]) - if ctype = "I" then do // integer - keyval=S2I(keyvaltrim) - end - else do // if not I then it's a string - keyval = keyvaltrim // gets the string on the rightside of "=" - end - end - end - CloseFile(fptr) - Return(keyval) -EndMacro - -Macro "read properties array"(file,key,ctype) - //this Macro is to read an array property,index is 1-based. - shared path, path_study - - pStr=RunMacro("read properties",file,key,ctype) - pStr=trim(pStr) - pArray=ParseString(pStr, ",") - Return(pArray) -EndMacro - -Macro "Export Matrix to CSV" (path,filename,corename,filenameout) - m = OpenMatrix(path+"\\"+filename, "True") - mc = CreateMatrixCurrency(m,corename,,,) - rows = GetMatrixRowLabels(mc) - ExportMatrix(mc, rows, "Rows", "CSV", path+"\\"+filenameout, ) - return(1) -EndMacro - -Macro "Export Matrix" (path,filename,corename,filenameout,outputtype) - //path as string - path="T:\\transnet2\\devel\\sr12\\sr12_byear\\byear" - //filename as string - must be a matrix - filename="SLAgg.mtx" - //corename as string - corename="DAN" - //filenameout as string - filenameout="SLAgg.csv" - //outputtype as string - ("dBASE", "FFA", "FFB" or "CSV") - - m = OpenMatrix(path+"\\"+filename, "True") - mc = CreateMatrixCurrency(m,corename,,,) - rows = GetMatrixRowLabels(mc) - ExportMatrix(mc, rows, "Rows", outputtype, path+"\\"+filenameout, ) - return(1) -EndMacro - -Macro "Matrix Size" (path, filename, corename) - //gets the size (number of zones) in the matrix - useful for sr11 vs sr12 and for split zones - m = OpenMatrix(path+"\\"+filename, "True") - base_indicies = GetMatrixBaseIndex(m) - mc = CreateMatrixCurrency(m, corename, base_indicies[1], base_indicies[2], ) - v = GetMatrixVector(mc, {{"Marginal", "Row Count"}}) - vcount = VectorStatistic(v, "Count", ) - return(vcount) -EndMacro - -Macro "Create Matrix" (path, filename, label, corenames, zone) - Opts = null - Opts.[File Name] = (path+"\\"+filename) - Opts.Label = label - Opts.Type = "Float" - Opts.Tables = corenames - Opts.[Column Major] = "No" - Opts.[File Based] = "Yes" - Opts.Compression = 0 - m = CreateMatrixFromScratch(label, zone, zone, Opts) - return(1) -EndMacro - -Macro "Aggregate Matrices" (path, xref, xrefcol1, xrefcol2, mtx, corenm, aggmtx) - // Aggregate Matrix Options - m = OpenMatrix(path+"\\"+mtx, "True") - base_indicies = GetMatrixBaseIndex(m) - Opts = null - Opts.Input.[Matrix Currency] = {path+"\\"+mtx, corenm, base_indicies[1], base_indicies[2]} - Opts.Input.[Aggregation View] = {xref, "xref"} - Opts.Global.[Row Names] = {"xref."+xrefcol1, "xref."+xrefcol2} - Opts.Global.[Column Names] = {"xref."+xrefcol1, "xref."+xrefcol2} - Opts.Output.[Aggregated Matrix].Label = "AggMtx"+"_"+corenm - Opts.Output.[Aggregated Matrix].Compression = 1 - Opts.Output.[Aggregated Matrix].[File Name] = path+"\\"+aggmtx - - ok = RunMacro("TCB Run Operation", 1, "Aggregate Matrix", Opts, ) - return(ok) -EndMacro - -Macro "Go GetMatrixCoreNames" (path, matrix) - m = OpenMatrix(path+"\\"+matrix, ) - core_names=GetMatrixCoreNames(m) - return(core_names) -EndMacro - -Macro "Get SL Query # from QRY" (path) - - selinkqry_file="\\selectlink_query.qry" - fptr_from = OpenFile(path + selinkqry_file, "r") - qry_array = readarray(fptr_from) - // query = 0 - query_list = null - - for j = 1 to qry_array.length do - line = qry_array[j] - p1 = Position(line, "\") - p2 = Position(line, "\<\/name\>") - if p1 > 0 & p2 > 0 then do - start = p1 + 6 - name_len = p2 - start - name = Substring(line, start, name_len) - query_list = query_list + {name} - end - // if Position(qry_array[j], "\") >0 then query = query + 1 - end - return(query_list) -EndMacro - -Macro "close all" - maps = GetMaps() - if maps <> null then do - for k = 1 to maps[1].length do - SetMapSaveFlag(maps[1][k],"False") - end - end - RunMacro("G30 File Close All") - mtxs = GetMatrices() - if mtxs <> null then do - handles = mtxs[1] - for k = 1 to handles.length do - handles[k] = null - end - end - views = GetViews() - if views <> null then do - handles = views[1] - for k = 1 to handles.length do - handles[k] = null - end - end -EndMacro - -Macro "CloseViews" - vws = GetViewNames() - for i = 1 to vws.length do - CloseView(vws[i]) - end -EndMacro - -// returns a nicely formatted day and time -Macro "date and time" - date_arr = ParseString(GetDateAndTime(), " ") - day = date_arr[1] - mth = date_arr[2] - num = date_arr[3] - time = Left(date_arr[4], StringLength(date_arr[4])-3) - year = SubString(date_arr[5], 1, 4) - today = mth + "/" + num + "/" + year + " " + time - //showmessage(today) - Return(today) -EndMacro - -Macro "SDdeletefile"(arr) - file=arr[1] - dif2=GetDirectoryInfo(file,"file") - if dif2.length>0 then deletefile(file) - ok=1 - quit: - return(ok) -EndMacro - -Macro "SDcopyfile"(arr) - file1=arr[1] - file2=arr[2] - dif2=GetDirectoryInfo(file2,"file") - if dif2.length>0 then deletefile(file2) - dif2=GetDirectoryInfo(file1,"file") - if dif2.length>0 then copyfile(file1,file2) - ok=1 - quit: - return(ok) -EndMacro - -Macro "SDrenamefile"(arr) - file1=arr[1] - file2=arr[2] - dif1=GetDirectoryInfo(file2,"file") - if dif1.length>0 then deletefile(file2) - dif2=GetDirectoryInfo(file1,"file") - if dif2.length>0 then RenameFile(file1, file2) - ok=1 - quit: - return(ok) -EndMacro - -Macro "HwycadLog"(arr) - shared path - fprlog=null - log1=arr[1] - log2=arr[2] - dif2=GetDirectoryInfo(path+"\\hwycadx.log","file") - if dif2.length>0 then fprlog=OpenFile(path+"\\hwycadx.log","a") - else fprlog=OpenFile(path+"\\hwycadx.log","w") - mytime=GetDateAndTime() - writeline(fprlog,mytime+", "+log1+", "+log2) - CloseFile(fprlog) - fprlog = null - return() -EndMacro - -Macro "ForecastYearStr" - shared path_study,path - fptr = OpenFile(path+"\\year", "r") - strYear = ReadLine(fptr) - closefile(fptr) - return(strYear) -EndMacro - -Macro "ForecastYearInt" - //usage: myyear=RunMacro("ForecastYearInt") - shared path_study,path - fptr = OpenFile(path+"\\year", "r") - strFyear = ReadLine(fptr) - closefile(fptr) - intFyear=S2I(strFyear) - return(intFyear) -EndMacro - -Macro "DeleteInterimFiles" (path, FileNameArray,RscName,MacroName,FileDescription) - RunMacro("HwycadLog",{RscName+": "+MacroName,"SDdeletefile, "+FileDescription}) - for i = 1 to FileNameArray.length do //delete existing files - ok=RunMacro("SDdeletefile",{path+"\\"+FileNameArray[i]}) if !ok then goto quit - end - quit: - return(ok) -EndMacro - -Macro "FileCheckDelete" (path,filename) - //usage: RunMacro("FileCheckDelete",path,filename) where path and filename are strings - di = GetDirectoryInfo(path+"\\"+filename, "File") - if di.length > 0 then do - ok=RunMacro("SDdeletefile",{path+"\\"+filename}) - return(ok) - end -EndMacro - -// Macro "getpathdirectory" doesn't allow the selected path with different path_study. -Macro "GetPathDirectory" - shared path,path_study,scr - opts={{"Initial Directory", path_study}} - tmp_path=choosedirectory("Choose an alternative directory in the same study area", opts) - strlen=len(tmp_path) - for i = 1 to strlen do - tmp=right(tmp_path,i) - tmpx=left(tmp,1) - if tmpx="\\" then goto endfor - end - endfor: - strlenx=strlen-i - tmppath_study=left(tmp_path,strlenx) - if path_study=tmppath_study then do - path=tmp_path - tmp_flag=0 - for i=1 to scr.length do - if scr[i]=path then do - tmp_flag=1 - i=scr.length+1 - end - else i=i+1 - end - if tmp_flag=0 then do - tmp = CopyArray(scr) - tmp = tmp + {tmp_path} - scr = CopyArray(tmp) - end - //showmessage("write description of the alternative in the head file") - //x=RunProgram("notepad "+path+"\\head",) - mytime=GetDateAndTime() - fptr=openfile(path+"\\tplog","a") - WriteLine(fptr, mytime) - closefile(fptr) - //showmessage("type in the reason why you are doing the model run in tplog") - //x=RunProgram("notepad "+path+"\\tplog",) - end - else do - path=null - msg1="The alternative directory selected is invalid because it has different study area! " - msg2="Please select again within the same study area " - msg3=" or use the Browse button to select a different study area." - showMessage(msg1+msg2+path_study+msg3) - end -EndMacro -/*********************************************************************************************************************************** -* -* Run Program -* Runs the program for a set of control files -* -***********************************************************************************************************************************/ - -Macro "Run Program" (scenarioDirectory, executableString, controlString) - - - //drive letter - path = SplitPath(scenarioDirectory) - - //open the batch file to run - fileString = scenarioDirectory+"\\programs\\source.bat" - ptr = OpenFile(fileString, "w") - WriteLine(ptr,path[1]) - WriteLine(ptr,"cd "+scenarioDirectory ) - - runString = "call "+executableString + " " + controlString - WriteLine(ptr,runString) - - //write the return code check - failString = "IF NOT ERRORLEVEL = 0 ECHO "+controlString+" > failed.txt" - WriteLine(ptr,failString) - - CloseFile(ptr) - status = RunProgram(fileString, {{"Minimize", "True"}}) - - info = GetFileInfo(scenarioDirectory+"\\failed.txt") - if(info != null) then do - ret_value=0 - goto quit - end - - Return(1) - quit: - Return( RunMacro("TCB Closing", ret_value, True ) ) -EndMacro - diff --git a/sandag_abm/src/main/gisdk/TC2OMX.rsc b/sandag_abm/src/main/gisdk/TC2OMX.rsc deleted file mode 100644 index 09fa7cb..0000000 --- a/sandag_abm/src/main/gisdk/TC2OMX.rsc +++ /dev/null @@ -1,69 +0,0 @@ -Macro "TC to OMX" - - p="T:\\ABM\\ActivitySim\\SANDAG_ActivitySim\\data\\skims" - - files = GetDirectoryInfo(RunMacro("FormPath",{p,"*"}),"All") - for i = 1 to files.length do - f = RunMacro("FormPath",{p,files[i][1]}) - subs = ParseString(f,".", {{"Include Empty",True}}) - if files[i][2] = "file" then do - if subs[2]="mtx" then do - RunMacro("ExportMatrix",subs[1]+".mtx") - end - end - end -EndMacro - -Macro "ExportMatrix" (matrix) - subs = ParseString(matrix,".", {{"Include Empty",True}}) - m = OpenMatrix(matrix, "True") - mc = CreateMatrixCurrency(m,,,,) - CopyMatrix(mc, { - {"File Name", subs[1]+".omx"}, - {"OMX", "True"} - } - ) -EndMacro - -Macro "FormPath" (path_elements) - if TypeOf(path_elements) <> "array" then do - ShowMessage("Must form a path out of a list of elements, not: " + TypeOf(path_elements)) - ShowMessage(2) - end - //path_elements is an array of elements - path = "" - for i = 1 to path_elements.length do - //change / to \ - p = RunMacro("NormalizePath",path_elements[i]) - if Right(p,1) = "\\" then do - if Len(p) > 1 then do - p = Substring(p,1,Len(p)-1) - end - else do - p = "" - end - end - if Left(p,1) = "\\" then do - if Len(p) > 1 then do - p = Substring(p,2,Len(p)) - end - else do - p = "" - end - end - if path = "" then do - path = p - end - else do - path = path + "\\" + p - end - end - return(path) -EndMacro - -Macro "NormalizePath" (path) - if Len(path) > 1 and path[2] = ":" then do - path = Lower(path[1]) + Right(path,Len(path)-1) - end - return(Substitute(path,"/","\\",)) -EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/TruckModel.rsc b/sandag_abm/src/main/gisdk/TruckModel.rsc deleted file mode 100644 index 05cda11..0000000 --- a/sandag_abm/src/main/gisdk/TruckModel.rsc +++ /dev/null @@ -1,1504 +0,0 @@ -/********************************************************************************************************** -Runs truck model -About: - Script to run the SANDAG Truck Model in TransCAD - Study Area: County of San Diego, California - TransCAD version 4.8 Build 545 - Author: Parsons Brinckerhoff (R. Moeckel, S. Gupta, C. Frazier) - Developed: June to December 2008 - -Modifications: - 1) Added truck toll diversion model to go from three truck types - to six truck types ({lhd/mhd/hhd} * {toll/non-toll}) - Ben Stabler, stabler@pbworld.com, 12/02/10 - - 2) Modified to integrate with SANDAG ABM - Amar Sarvepalli, sarvepalli@pbworld.com, 09/06/12 - - 3) Modified to remove truck skimming; added truck skims to hwyskim_vot.rsc - JEF joel.freedman@rsginc.com, 5/10/2015 - -Steps: - 1) Generates standard truck trip and special (military) truck trips - 2) Gets regional truck trips, IE trips, EI trips and EE trips and balances truck trips - 3) Distributes truck trips with congested skims and splits by time of day - 4) Applies truck toll diversion model with free-flow toll and non-toll skims - - Note: truck trip generation and free-flow skims are run only for the first iteration - - - -**********************************************************************************************************/ -Macro "truck model"(properties, iteration) - shared path, inputDir, outputDir, inputTruckDir - - // read properties from sandag_abm.properties in /conf folder - properties = "\\conf\\sandag_abm.properties" - startFromIteration = s2i(RunMacro("read properties",properties,"RunModel.startFromIteration", "S")) - - // Generate trips and free-flow truck skims for the first iteration - if (iteration = startFromIteration) then do - // Generate daily truck trips - RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","truck-tripgen"}) - ok = RunMacro("truck-tripgen",properties) - if !ok then goto quit - end - - // Distribute daily truck trips and split them by time period - RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","trkDistribution,(properties)"}) - ok = RunMacro("trkDistribution",properties) - if !ok then goto quit - - // Apply toll-diversion model - RunMacro("HwycadLog",{"TruckModel.rsc: truckmodel","trk toll diversion model"}) - ok = RunMacro("trk toll diversion model") - if !ok then goto quit - - run_ok = 1 - RunMacro("close all") - Return(run_ok) - - quit: - RunMacro("close all") - Return(ok) -EndMacro - - - -/********************************************************************************************************** -Generates daily truck trips (standard and special generations) - -Inputs: - sandag.properties - -Outputs: - output\gmTruckDataBalanced.bin - output\regionalEEtrips.csv - -**********************************************************************************************************/ -Macro "truck-tripgen"(properties) - shared path, inputDir, outputDir, mxzone - dim arrInterimYear[3] - - strFyear = RunMacro("read properties",properties,"truck.FFyear","S") - intFyear = StringToInt(strFyear) - arrInterimYear=RunMacro("InterimYearCheck",properties,intFyear) - - RunMacro("trkStdTripGen",strFyear,intFyear,arrInterimYear,properties) - RunMacro("trkSpecialGen",strFyear,intFyear,arrInterimYear) - RunMacro("trkBalance",strFyear,intFyear,arrInterimYear) - run_ok=1 - - exit: - RunMacro("close all") - Return(run_ok) -EndMacro - - - - -/********************************************************************************************************** -Produces standard truck trips - -Inputs: - input\mgra13_based_input2010.csv - input\TruckTripRates.csv - output\hhByTaz.csv.csv - output\empByTaz.csv.csv - - (optional, used for interpolation) - input\hhByTaz.csv - input\hhByTaz.csv - input\empByTaz.csv - input\empByTaz.csv - - (optional, used only for landuse override) - input\lu.csv - output\EmpDistLUovrAgg.csv - output\hhluagg.csv - -Outputs: - output\gmTruckDataII.csv - -**********************************************************************************************************/ -Macro "trkStdTripGen" (strFyear,intFyear,arrInterimYear,properties) - shared path, inputDir, outputDir, mxzone - booInterimYear = arrInterimYear[1] - - // Creates household and employment data by taz from mgra data - RunMacro("Create hh and emp by taz") - - //-------------------------------------------------- - //This section checks available data and interpolates if doesn't exist - //-------------------------------------------------- - // Do interpolate = True - if booInterimYear = 2 then do - prevYear = arrInterimYear[2] - nextYear = arrInterimYear[3] - end - - // Override landuse data if option is "True" - check_luOverride = RunMacro("read properties",properties,"truck.luOverRide", "S") - if check_luOverride = "True" then do - RunMacro("EmploymentLUOverride") - RunMacro("EmploymentDistLUOverride") - RunMacro("HouseholdLUOverride") - end - - - // If data is not available then interpolate from closest available years - if booInterimYear = 2 then do - // Copy prev and next year data files to output directory - ok=RunMacro("SDcopyfile",{inputDir+"\\hhByTaz"+I2S(prevYear)+".csv",outputDir+"\\hhByTaz_prev.csv"}) - ok=RunMacro("SDcopyfile",{inputDir+"\\hhByTaz"+I2S(nextYear)+".csv",outputDir+"\\hhByTaz_next.csv"}) - ok=RunMacro("SDcopyfile",{inputDir+"\\empByTaz"+I2S(prevYear)+".csv",outputDir+"\\empByTaz_prev.csv"}) - ok=RunMacro("SDcopyfile",{inputDir+"\\empByTaz"+I2S(nextYear)+".csv",outputDir+"\\empByTaz_next.csv"}) - - // Interpolate data from prev and next years - ok=RunMacro("Interpolate",{"hhByTaz_prev.csv","hhByTaz_next.csv","hhByTaz.csv",intFyear,prevYear,nextYear}) - ok=RunMacro("Interpolate",{"empByTaz_prev.csv","empByTaz_next.csv","empByTaz.csv",intFyear,prevYear,nextYear}) - - // Delete prev and next year data files - ok=RunMacro("SDdeletefile",{outputDir+"\\hhByTaz_prev.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\hhByTaz_next.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\empByTaz_prev.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\empByTaz_next.csv"}) - end - - // join data and parameter tables - empView = Opentable("Employment", "CSV", {outputDir+"\\empByTaz.csv"}) - hhView = Opentable("HouseHolds", "CSV", {outputDir+"\\hhByTaz.csv"}) - - - //-------------------------------------------------- - // This section overrides LU for employment and households - //-------------------------------------------------- - if check_luOverride = "True" then do - - //LU override file for Employment - empAggDistLUovr_filecsv = outputDir+"\\EmpDistLUovrAgg.csv" - di = GetDirectoryInfo(empAggDistLUovr_filecsv, "File") - - // Export employment by TAZ to binary file and get zones - ExportView("Employment|", "FFB", outputDir+"\\empByTaz.BIN", null, ) - vwEmpBin = Opentable("EmploymentByZoneBin", "FFB", {outputDir+"\\empByTaz.BIN"}) - vwEmpDistLUovrAgg = Opentable("EmploymentDistributionOverride", "CSV", {empAggDistLUovr_filecsv}) - vectZonerecords = GetRecordCount(vwEmpDistLUovrAgg, null) - vectZone = GetDataVectors(vwEmpDistLUovrAgg+"|",{"zone"},) - - // Convert real to string zones - for i = 1 to vectZonerecords do - strZone = RealToString(vectZone[1][i]) - rh = LocateRecord("EmploymentByZoneBin|","TAZ",{strZone},{{"Exact", "True"}}) - x = GetView() - DeleteRecord("EmploymentByZoneBin", rh) - end - - // Get data from all fields in EmpDistLUovrAgg file - fldsEmpDistLUovrAgg = GetFields(vwEmpDistLUovrAgg, "All") - vectEMPovr = GetDataVectors(vwEmpDistLUovrAgg+"|",fldsEmpDistLUovrAgg[1],) - - // Write fields to employment by TAZ - for i = 1 to vectZonerecords do - rh = AddRecord(vwEmpBin, { - {"TAZ", R2I(vectEMPovr[1][i])}, - {"First TAZ", R2I(vectEMPovr[2][i])}, - {fldsEmpDistLUovrAgg[1][3], vectEMPovr[3][i]}, - {fldsEmpDistLUovrAgg[1][4], vectEMPovr[4][i]}, - {fldsEmpDistLUovrAgg[1][5], vectEMPovr[5][i]}, - {fldsEmpDistLUovrAgg[1][6], vectEMPovr[6][i]}, - {fldsEmpDistLUovrAgg[1][7], vectEMPovr[7][i]}, - {fldsEmpDistLUovrAgg[1][8], vectEMPovr[8][i]}, - {fldsEmpDistLUovrAgg[1][9], vectEMPovr[9][i]}, - {fldsEmpDistLUovrAgg[1][10], vectEMPovr[10][i]}, - {fldsEmpDistLUovrAgg[1][11], vectEMPovr[11][i]} - }) - end - - // LU override file for Households - HHLUagg_filecsv = outputDir+"\\hhluagg.csv" - di = GetDirectoryInfo(HHLUagg_filecsv, "File") - - // Export households by TAZ to binary file and get zones - ExportView(hhView+"|", "FFB", outputDir+"\\hhByTaz.BIN", null, ) - vwHHBin = Opentable("HouseholdsByZoneBin", "FFB", {outputDir+"\\HHByTaz.BIN"}) - vwHHLUagg = Opentable("HouseholdsOverride", "CSV", {HHLUagg_filecsv}) - vectHHZoneRecords = GetRecordCount(vwHHLUagg, null) - vectHHZone = GetDataVectors(vwHHLUagg+"|",{"TAZ"},) - - // Convert real to string zones - for i = 1 to vectHHZoneRecords do - strZone = RealToString(vectHHZone[1][i]) - rh = LocateRecord("HouseholdsByZoneBin|","TAZ",{strZone},{{"Exact", "True"}}) - DeleteRecord("HouseholdsByZoneBin", rh) - end - - // Get data from all fields in hhluagg file - fldsHHLUAgg = GetFields(vwHHLUagg, "All") - vectHHovr = GetDataVectors(vwHHLUagg+"|",fldsHHLUAgg[1],) - - // Write fields to households by TAZ - for i = 1 to vectHHZoneRecords do - rh = AddRecord(vwHHBin, { - {"TAZ", R2I(vectHHovr[1][i])}, - {"First TAZ", R2I(vectHHovr[2][i])}, - {fldsHHLUAgg[1][3], vectHHovr[3][i]} - }) - end - end - - - //-------------------------------------------------- - // This section applies truck trip production and attraction rates by truck type - //-------------------------------------------------- - // Open truck data file - viewtripRates = Opentable("TruckTripRates", "CSV", {inputDir+"\\TruckTripRates.csv"}) - - // Join all data - jv1 = Joinviews("JV1", hhView+".ZONE", empView+".ZONE",) - jv9 = Joinviews("JV9", jv1+"."+hhView+".TruckRegionType", viewtripRates+".RegionType",) - Setview(jv9) - - // Compute lhd truck productions "AGREMPN","CONEMPN","RETEMPN","GOVEMPN","MANEMPN","UTLEMPN","WHSEMPN","OTHEMPN"} - lhd_p = CreateExpression(jv9, "LHD_ProductionsTemp", "Nz(emp_agmin + emp_cons) * [TG_L_Ag/Min/Constr] + Nz(emp_retrade) * TG_L_Retail + Nz(emp_gov) * TG_L_Government + Nz(emp_mfg) * TG_L_Manufacturing + Nz(emp_twu) * [TG_L_Transp/Utilities]", ) - lhd_p = CreateExpression(jv9, "LHD_Productions", "Nz(LHD_ProductionsTemp) + Nz(emp_whtrade) * TG_L_Wholesale + Nz(emp_other) * TG_L_Other + Nz(HH) * TG_L_Households", ) - - // Compute lhd truck attractions - lhd_a = CreateExpression(jv9, "LHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_L_Ag/Min/Constr] + emp_retrade * TA_L_Retail + emp_gov * TA_L_Government + emp_mfg * TA_L_Manufacturing + emp_twu * [TA_L_Transp/Utilities] + emp_whtrade * TA_L_Wholesale", ) - lhd_a = CreateExpression(jv9, "LHD_Attractions", "Nz(LHD_AttractionsTemp) + Nz(emp_other) * TA_L_Other + Nz(HH) * TA_L_Households", ) - - // Compute mhd truck productions - mhd_p = CreateExpression(jv9, "MHD_ProductionsTemp", "(emp_agmin + emp_cons) * [TG_M_Ag/Min/Constr] + emp_retrade * TG_M_Retail + emp_gov * TG_M_Government + emp_mfg * TG_M_Manufacturing + emp_twu * [TG_M_Transp/Utilities] + emp_whtrade * TG_M_Wholesale ", ) - mhd_p = CreateExpression(jv9, "MHD_Productions", "Nz(MHD_ProductionsTemp) + Nz(emp_other) * TG_M_Other + Nz(HH) * TG_M_Households", ) - - // Compute mhd truck attractions - mhd_a = CreateExpression(jv9, "MHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_M_Ag/Min/Constr] + emp_retrade * TA_M_Retail + emp_gov * TA_M_Government + emp_mfg * TA_M_Manufacturing + emp_twu * [TA_M_Transp/Utilities] + emp_whtrade * TA_M_Wholesale", ) - mhd_a = CreateExpression(jv9, "MHD_Attractions", "Nz(MHD_AttractionsTemp) + Nz(emp_other) * TA_M_Other + Nz(HH) * TA_M_Households", ) - - // Compute hhd truck productions - hhd_p = CreateExpression(jv9, "HHD_ProductionsTemp", "(emp_agmin + emp_cons) * [TG_H_Ag/Min/Constr] + emp_retrade * TG_H_Retail + emp_gov * TG_H_Government + emp_mfg * TG_H_Manufacturing + emp_twu * [TG_H_Transp/Utilities] + emp_whtrade * TG_H_Wholesale ", ) - hhd_p = CreateExpression(jv9, "HHD_Productions", "Nz(HHD_ProductionsTemp) + Nz(emp_other) * TG_H_Other + Nz(HH) * TG_H_Households", ) - - // Compute hhd truck attractions - hhd_a = CreateExpression(jv9, "HHD_AttractionsTemp", "(emp_agmin + emp_cons) * [TA_H_Ag/Min/Constr] + emp_retrade * TA_H_Retail + emp_gov * TA_H_Government + emp_mfg * TA_H_Manufacturing + emp_twu * [TA_H_Transp/Utilities] + emp_whtrade * TA_H_Wholesale", ) - hhd_a = CreateExpression(jv9, "HHD_Attractions", "Nz(HHD_AttractionsTemp) + Nz(emp_other) * TA_H_Other + Nz(HH) * TA_H_Households", ) - - // Export the productions and attractions to csv file - RunMacro("HwycadLog",{"TruckModel.rsc: trkStdGen","ExportView P&A"}) - ExportView(jv9+"|", "CSV", outputDir+"\\gmTruckDataII.csv", {hhView+".ZONE", "LHD_Productions", "LHD_Attractions", "MHD_Productions", "MHD_Attractions", "HHD_Productions", "HHD_Attractions"}, {{"CSV Header"}}) - - - // Close all and delete temp files - RunMacro("close all") - ok=RunMacro("SDdeletefile",{outputDir+"\\hhdata.csv"}) - if!ok then goto exit - ok=RunMacro("SDdeletefile",{outputDir+"\\emp.csv"}) - if!ok then goto exit - - run_ok=1 - exit: - RunMacro("close all") - Return(run_ok) -EndMacro - - - -/********************************************************************************************************** -Creates households by taz and employment by taz files to use in the truck trip generation model - -Inputs: - sandag.properties - input\mgra13_based_input2012.csv - -Outputs: - output\empByTaz.csv - output\hhByTaz.csv - -**********************************************************************************************************/ -Macro "Create hh and emp by taz" - shared path, inputDir, outputDir, mxzone , scenarioYear - mgraDataFile = "mgra13_based_input"+scenarioYear+".csv" - empbytaz = "empByTaz.csv" - hhbytaz = "hhByTaz.csv" - - RunMacro("SDcopyfile",{inputDir+"\\"+mgraDataFile,outputDir+"\\"+mgraDataFile}) - mgraView = OpenTable("MGRA View", "CSV", {outputDir+"\\"+mgraDataFile}, {{"Shared", "True"}}) - - // Get data fields into vectors - mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_total = GetDataVector(mgraView+"|", "emp_total", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - - - // Combine employment fields that match to the truck trip rate classification - TOTEMP = emp_total // Total employment - emp_agmin = emp_ag // Agriculture + Mining - emp_cons = emp_const_bldg_prod + emp_const_bldg_office // Construction - emp_retrade = emp_retail + emp_personal_svcs_retail // Retail - emp_gov = emp_state_local_gov_ent + emp_state_local_gov_blue + emp_state_local_gov_white + emp_fed_non_mil + emp_fed_mil // Government - emp_mfg = emp_mfg_prod + emp_mfg_office // Manufacturing - emp_twu = emp_trans + emp_utilities_office + emp_utilities_prod // Transportation + Utilities - emp_whtrade = emp_whsle_whs // Wholesale - emp_other = TOTEMP - (emp_agmin +emp_cons+ emp_retrade + emp_gov + emp_mfg + emp_twu + emp_whtrade) // Other - - - // Add fields employment by taz - strct = GetTableStructure(mgraView) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"emp_agmin" , "Float", 8, 4, "True", , , , , , , null}} - strct = strct + {{"emp_cons" , "Float", 8, 4, "True", , , , , , , null}} - strct = strct + {{"emp_retrade" , "Float", 8, 0, "True", , , , , , , null}} - strct = strct + {{"emp_gov" , "Float", 8, 0, "True", , , , , , , null}} - strct = strct + {{"emp_mfg" , "Float", 8, 0, "True", , , , , , , null}} - strct = strct + {{"emp_twu" , "Float", 8, 0, "True", , , , , , , null}} - strct = strct + {{"emp_whtrade" , "Float", 8, 0, "True", , , , , , , null}} - strct = strct + {{"emp_other" , "Float", 8, 4, "True", , , , , , , null}} - ModifyTable(mgraView, strct) - - ExportView(mgraView+"|","FFB",outputDir+"\\temp.bin",{"MGRA","TAZ","HH","TruckRegionType","emp_agmin","emp_cons","emp_retrade","emp_gov","emp_mfg","emp_twu","emp_whtrade","emp_other"},) - MgraTazView = OpenTable("MGRA View", "FFB", {outputDir+"\\temp.bin"},) - - // Set data into new fields - SetDataVectors(MgraTazView+"|",{{"emp_agmin" ,emp_agmin }, - {"emp_cons" ,emp_cons }, - {"emp_retrade",emp_retrade}, - {"emp_gov" ,emp_gov }, - {"emp_mfg" ,emp_mfg }, - {"emp_twu" ,emp_twu }, - {"emp_whtrade",emp_whtrade}, - {"emp_other" ,emp_other }},) - - // Aggregate employment fields by taz - emp = AggregateTable("employment", MgraTazView+"|","FFB", outputDir+"\\emp_temp.bin","TAZ", { - {"TruckRegionType","avg","TruckRegionType"}, - {"emp_agmin" ,"sum","emp_agmin" }, - {"emp_cons" ,"sum","emp_cons" }, - {"emp_retrade" ,"sum","emp_retrade" }, - {"emp_gov" ,"sum","emp_gov" }, - {"emp_mfg" ,"sum","emp_mfg" }, - {"emp_twu" ,"sum","emp_twu" }, - {"emp_whtrade" ,"sum","emp_whtrade" }, - {"emp_other" ,"sum","emp_other" } - },null) - - emp_view = OpenTable("emp_view", "FFB", {outputDir+"\\emp_temp.bin"},) - RenameField(emp_view+".Avg TruckRegionType", "TruckRegionType") - - // Create a temp file with all zones (internal + external zones) from the highway network file - db_file = outputDir+"\\hwy.dbd" - {node_lyr,} = RunMacro("TCB Add DB Layers", db_file,,) - SetLayer(node_lyr) - n= SelectByQuery("Zones", "Several", "Select * where ID <= "+ String(mxzone),) - zones = GetDataVector(node_lyr+"|Zones", "ID",{{"Sort Order", {{"ID", "Ascending"}}}}) - - // Create a temp file with all zones - all_vw = CreateTable("allzones", outputDir+"\\temp_zones.bin", "FFB",{ - {"ZONE", "Integer", 8, null, "Yes"}}) - SetView(all_vw) - for i = 1 to zones.length do - rh = AddRecord(all_vw, {{"ZONE", zones[i]}}) - end - - join_vw = JoinViews("joined_view1", all_vw+".ZONE",emp_view+".TAZ",) - ExportView(join_vw+"|","CSV",outputDir+"\\"+empbytaz,,) - CloseView(join_vw) - - - // Aggregate households by taz - hh = AggregateTable("households", MgraTazView+"|","FFB", outputDir+"\\hh_temp.bin","TAZ", { - {"TruckRegionType","avg","TruckRegionType"}, - {"HH" ,"sum","HH"} - },null) - - hh_view = OpenTable("hh_temp", "FFB", {outputDir+"\\hh_temp.bin"},) - RenameField(hh_view+".Avg TruckRegionType", "TruckRegionType") - join_vw = JoinViews("joined_view1", all_vw+".ZONE",hh_view+".TAZ",) - ExportView(join_vw+"|","CSV",outputDir+"\\"+hhbytaz,,) - - RunMacro("close all") - - DeleteFile(outputDir+"\\temp_zones.bin") - DeleteFile(outputDir+"\\emp_temp.bin") - DeleteFile(outputDir+"\\hh_temp.bin") -EndMacro - - - -/********************************************************************************************************** -Creates household override file from a land use file LU.CSV - -Inputs: - input\lu.csv - -Outputs: - output\hhlu.csv - output\hhluagg.csv - -**********************************************************************************************************/ -Macro "HouseholdLUOverride" - // Creates Household Override file from a Land Use based LU.csv - shared path, inputDir, outputDir - RunMacro("TCB Init") - - // Copy LU.csv, Open Copy, and Rename Fields - no header line in lu.csv - ok=RunMacro("SDcopyfile",{inputDir+"\\lu.csv",outputDir+"\\hhlu.csv"}) if!ok then goto quit - vwHHLUovr = Opentable("HHLUOverride", "CSV",{outputDir+"\\hhlu.csv"}) - RenameField(vwHHLUovr+".FIELD_1", "TAZ") - RenameField(vwHHLUovr+".FIELD_2", "RateType") - RenameField(vwHHLUovr+".FIELD_3", "LUCode") - RenameField(vwHHLUovr+".FIELD_4", "HH") - Setview(vwHHLUovr) - - // Select HH's From LU.csv - // Ratetype=DU=1 - qry1 = "Select * where RateType = 1" - DUqry = SelectByQuery("HHSelection", "Several", qry1,) - - // Aggregate HH's to TAZ Level (might have SF & MF as separate codes) - RunMacro("HwycadLog",{"TruckModel.rsc: HouseholdLUOverride","AggregateTable"}) - aggtable = AggregateTable("AggHHLUOvr","HHLUOverride|HHSelection", "CSV", outputDir+"\\hhluagg.csv", "TAZ", { - {"TAZ","dominant"}, - {"HH","sum", } - }, {"Missing as zero"}) - - done: - RunMacro("close all") - Return( RunMacro("TCB Closing", 1, "FALSE" ) ) - - quit: - RunMacro("close all") - Return( RunMacro("TCB Closing", 0, "FALSE" ) ) -EndMacro - - -/********************************************************************************************************** -Creates Employment Override File from a land use file LU.CSV - -Inputs: - input\lu.csv - input\emplbylu.csv - input\Zone_sphere.csv - input\emp_lu_ksf.csv - input\emp_lu_rm.csv - input\emp_lu_site.csv - -Outputs: - output\EmpLUovr.bin" - output\EmpLUovr.csv" -**********************************************************************************************************/ -Macro "EmploymentLUOverride" - shared path,inputDir, outputDir - - // Remove dcc & dcb files if already exist - ok=RunMacro("SDdeletefile",{inputDir+"\\emplbylu.dcc"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{inputDir+"\\Zone_sphere.dcc"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{inputDir+"\\emp_lu_ksf.dcc"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{inputDir+"\\emp_lu_site.dcc"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{inputDir+"\\lu.dcc"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{outputDir+"\\EmpLUovr.dcb"}) - if!ok then goto quit - ok=RunMacro("SDdeletefile",{outputDir+"\\EmpLUovr.dcc"}) - if!ok then goto quit - - // Open data files and LU.CSV - vwEmpbyLU = Opentable("EmploymentByLU", "CSV", {inputDir+"\\emplbylu.csv"}) - vwZoneSphere = Opentable("XREF_ZoneSphere","CSV", {inputDir+"\\Zone_sphere.csv"}) - vwAcreToKSF = Opentable("Conv_Acre_KSF", "CSV", {inputDir+"\\emp_lu_ksf.csv"}) - vwAcreToRM = Opentable("Conv_Acre_RM", "CSV", {inputDir+"\\emp_lu_rm.csv"}) - vwEmpSite = Opentable("SiteEmploymentbySphere", "CSV", {inputDir+"\\emp_lu_site.csv"}) - vwLUovr = Opentable("LUOverride", "CSV", {inputDir+"\\lu.csv"}) - - // Rename fields for LU.CSV - no header line in file - RenameField(vwLUovr+".FIELD_1", "TAZ") - RenameField(vwLUovr+".FIELD_2", "RateType") - RenameField(vwLUovr+".FIELD_3", "LUCode") - RenameField(vwLUovr+".FIELD_4", "Amt") - - // Join data files to LU.CSV - Setview(vwLUovr) - jvwLUovrSphere = Joinviews("jvwLUovrSphere", vwLUovr+".TAZ", vwZoneSphere+".Zone",) - jvwLUovrSphereEmp = Joinviews("jvwLUovrSphereEmp", jvwLUovrSphere+".LUCode", vwEmpbyLU+".LU",) - jvwLUovrSphereEmpKSF = Joinviews("jvwLUovrSphereEmpKSF", jvwLUovrSphereEmp+".LUCode", vwAcreToKSF+".LU",) - jvwLUovrSphereEmpKSFSite = Joinviews("jvwLUovrSphereEmpKSFSite", jvwLUovrSphereEmpKSF+".LUCode", vwEmpSite+".LU",) - jvwLUovrSphereEmpKSFSiteRM = Joinviews("jvwLUovrSphereEmpKSFSiteRM", jvwLUovrSphereEmpKSFSite+".LUCode", vwAcreToRM+".LU",) - - // Set Output Files - empLUovr_file = outputDir+"\\EmpLUovr.BIN" - empLUovr_filecsv = outputDir+"\\EmpLUovr.csv" - - // Export Joined View and Add Employment Fields - Setview(jvwLUovrSphereEmpKSFSiteRM) - ExportView(jvwLUovrSphereEmpKSFSiteRM+"|", "FFB", empLUovr_file, null, { - {"Additional Fields",{ {"empbyacres", "Real", 16, 4, },{"empdirect", "Real", 16, 4, },{"empbysite", "Real", 16, 4, },{"empbyksf", "Real", 16, 4, },{"empbyrm", "Real", 16, 4, },{"emp", "Real", 16, 4, }} }, - } ) - - // Apply Acre Based Employment Rates - // RateType=2=Acre - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbyacres"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere=1404 and RateType=2) then nz(sph1404acre*Amt) else if (Sphere=1441 and RateType=2) then nz(sph1441acre*Amt) else if (Sphere>=1900 and RateType=2) then nz(sph1900acre*Amt) else if RateType=2 then nz(sphOtheracre*Amt)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Add Directly Entered Employment - // RateType=3=Employee - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empdirect"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if RateType=3 then nz(Amt)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Apply Site Based Employment Rates - // RateType=4=Site - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbysite"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere=1404 and RateType=4) then nz(sph1404site*Amt) else if (Sphere=1441 and RateType=4) then nz(sph1441site*Amt) else if (Sphere>=1900 and RateType=4) then nz(sph1900site*Amt) else if RateType=4 then nz(sphOthersite*Amt)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Convert KSF to Acres and Apply Acre Based Employment Rates - // RateType=6=KSF - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbyksf"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere=1404 and RateType=6) then nz(sph1404acre*Amt*ksf2acre) else if (Sphere=1441 and RateType=6) then nz(sph1441acre*Amt*ksf2acre)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbyksf"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere>=1900 and RateType=6) then nz(sph1900acre*Amt*ksf2acre) else if RateType=6 then nz(sphOtheracre*Amt*ksf2acre)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Convert KSF to Acres and Apply Acre Based Employment Rates - // RateType=7=Hotel Room - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbyRM"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere=1404 and RateType=7) then nz(sph1404acre*Amt*rm2acre) else if (Sphere=1441 and RateType=7) then nz(sph1441acre*Amt*rm2acre)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"empbyRM"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if (Sphere>=1900 and RateType=7) then nz(sph1900acre*Amt*rm2acre) else if RateType=7 then nz(sphOtheracre*Amt*rm2acre)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Combine Acre, KSF, and Site Employment into EMP Field - Opts = null - Opts.Input.[Dataview Set] = {empLUovr_file, "jvwLUovrSphereEmpKSFSiteRM"} - Opts.Global.Fields = {"emp"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"nz(empbyacres)+nz(empdirect)+nz(empbysite)+nz(empbyksf)+nz(empbyrm)"} - ret_value = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ret_value then goto quit - - // Export Employment Override File - RunMacro("HwycadLog",{"TruckModel.rsc: EmploymentLUOverride","ExportView"}) - CloseView(jvwLUovrSphereEmpKSFSiteRM) - jvwLUovrSphereEmpKSFSiteRM=opentable("jvwLUovrSphereEmpKSFSiteRM", "FFB", {empLUovr_file,}) - Setview(jvwLUovrSphereEmpKSFSiteRM) - ExportView(jvwLUovrSphereEmpKSFSiteRM+"|", "CSV", empLUovr_filecsv, - {"TAZ","RateType","LUCode","Amt","emp"}, - { - {"CSV Header", "False"}, - {"CSV Drop Quotes", "True"} - } ) - - done: - RunMacro("close all") - Return( RunMacro("TCB Closing", 1, "FALSE" ) ) - - quit: - RunMacro("close all") - Return( RunMacro("TCB Closing", 0, "FALSE" ) ) -EndMacro - - -/********************************************************************************************************** -Takes employment override file and creates employment distribution to employment categories - -Inputs: - output\EmpLUovr.csv" - input\empldistbylu.csv - -Outputs: - output\EmpDistLUovrAgg.csv - output\EmpDistLUovr.csv" - -**********************************************************************************************************/ -Macro "EmploymentDistLUOverride" - shared path,inputDir, outputDir - RunMacro("TCB Init") - - // Open Employment Override and Employment Distribution Files - empLUovr_filecsv = outputDir+"\\EmpLUovr.csv" - vwEmpLUovr=opentable("EmploymentLUoverride", "CSV", {empLUovr_filecsv,}) - vwEmpDistbyLU = Opentable("EmploymentDistribution", "CSV", {inputDir+"\\empldistbylu.csv"}) - - // Join Files - Setview(vwEmpLUovr) - jvwEmpLUovrDist = Joinviews("EmploymentOverrideWithDistribution", vwEmpLUovr+".LUCode", vwEmpDistbyLU+".plu",) - - // Set Output Files - empAggDistLUovr_filecsv = outputDir+"\\EmpDistLUovrAgg.csv" - empDistLUovr_filecsv = outputDir+"\\EmpDistLUovr.csv" - - // Calculate Employment Distribution - expZone = CreateExpression(jvwEmpLUovrDist, "zone", "TAZ", ) - expEmp_mil = CreateExpression(jvwEmpLUovrDist, "emp_mil", "Nz(emp*mil)", ) - expEmp_agmin = CreateExpression(jvwEmpLUovrDist, "emp_agmin", "Nz(emp*ag)", ) - expEmp_cons = CreateExpression(jvwEmpLUovrDist, "emp_cons", "Nz(emp*con)", ) - expEmp_mfg = CreateExpression(jvwEmpLUovrDist, "emp_mfg", "Nz(emp*mfg)", ) - expEmp_whtrade = CreateExpression(jvwEmpLUovrDist, "emp_whtrade", "Nz(emp*whtrade)", ) - expEmp_retrade = CreateExpression(jvwEmpLUovrDist, "emp_retrade", "Nz(emp*retrade)", ) - expEmp_twu = CreateExpression(jvwEmpLUovrDist, "emp_twu", "Nz(emp*twu)", ) - expEmp_fre = CreateExpression(jvwEmpLUovrDist, "emp_fre", "Nz(emp*fre)", ) - expEmp_info = CreateExpression(jvwEmpLUovrDist, "emp_info", "Nz(emp*info)", ) - expEmp_pbs = CreateExpression(jvwEmpLUovrDist, "emp_pbs", "Nz(emp*pbs)", ) - expEmp_lh = CreateExpression(jvwEmpLUovrDist, "emp_lh", "Nz(emp*lh)", ) - expEmp_os = CreateExpression(jvwEmpLUovrDist, "emp_os", "Nz(emp*os)", ) - expEmp_edhs = CreateExpression(jvwEmpLUovrDist, "emp_edhs", "Nz(emp*edhs)", ) - expEmp_gov = CreateExpression(jvwEmpLUovrDist, "emp_gov", "Nz(emp*gov)", ) - expEmp_sedw = CreateExpression(jvwEmpLUovrDist, "emp_sedw", "Nz(emp*sedw)", ) - expEmp_civ = CreateExpression(jvwEmpLUovrDist, "emp_civ", "emp_agmin + emp_cons + emp_mfg + emp_whtrade + emp_retrade + emp_twu + emp_fre + emp_info + emp_pbs + emp_lh + emp_os + emp_edhs + emp_gov + emp_sedw", ) - - // Export employment distribution before aggregation - ExportView(jvwEmpLUovrDist+"|", "CSV", empDistLUovr_filecsv, {"zone", "emp", "emp_civ", "emp_mil", "emp_agmin", "emp_cons", "emp_mfg", "emp_whtrade", "emp_retrade", "emp_twu", "emp_fre", "emp_info", "emp_pbs", "emp_lh", "emp_os", "emp_edhs", "emp_gov", "emp_sedw"}, {{"CSV Header", "True"},{"CSV Drop Quotes", "True"}}) - - // Aggregate employment distribution by LU Code by TAZ to employment distribution by TAZ - RunMacro("HwycadLog",{"TruckModel.rsc: EmploymentDistLUOverride","AggregateTable"}) - vwEmpDistLUovr=opentable("DistributedEmploymentOverride", "CSV", {empDistLUovr_filecsv,}) - Setview(vwEmpDistLUovr) - AggAll = CreateSet("AggregatedZones") - SelectAll(AggAll) - sets_list = GetSets(vwEmpDistLUovr) - aggtable = AggregateTable("ZoneAggEmpDistLUOvr","DistributedEmploymentOverride|AggregatedZones", "CSV", empAggDistLUovr_filecsv, "zone", { - {"zone","dominant"}, - {"emp_agmin","sum", }, - {"emp_cons","sum", }, - {"emp_retrade","sum", }, - {"emp_gov","sum", }, - {"emp_mfg","sum", }, - {"emp_twu","sum", }, - {"emp_whtrade","sum", }, - {"emp_os","sum", }, - {"emp_sedw","sum", } - }, {"Missing as zero"}) - - done: - RunMacro("close all") - Return( RunMacro("TCB Closing", 1, "FALSE" ) ) - - quit: - RunMacro("close all") - Return( RunMacro("TCB Closing", 0, "FALSE" ) ) -EndMacro - - -/********************************************************************************************************** -Checks whether data is available for forecast year and if not then interpolates from closet available years - -Inputs: - forecast year - -Outputs: - returns an array {availability of data, previous data year, next data year} -**********************************************************************************************************/ -Macro "InterimYearCheck" (properties,intFyear) - dim arrInterimYear[3] - - // Reads all years for which data is available - DFyear =RunMacro("read properties",properties,"truck.DFyear","S") - - // Lists data year with delimiter "," - arrAllDFyears = ParseString(DFyear, ",") - for i = 1 to arrAllDFyears.length do - intTargetYear = s2i(trim(arrAllDFyears[i])) - if i < arrAllDFyears.length then do - intNextTargetYear = s2i(trim(arrAllDFyears[i+1])) - end - - // Forecast year has data available - if intFyear = intTargetYear then do - // Do interpolate = False - arrInterimYear[1]=1 - end - - // Forecast year has no data - else do - // Check for previous and next closest years - if (intFyear > intTargetYear & intFyear < intNextTargetYear) then do - // Do interpolate = True - arrInterimYear[1] = 2 - // Gets previous closest year - arrInterimYear[2] = intTargetYear - // Gets next closest year - arrInterimYear[3] = intNextTargetYear - end - end - end - - Return(arrInterimYear) -EndMacro - - - -/********************************************************************************************************** -Adds trucks generated by special generators, such as military sites, mail to/from airport, cruise ships, etc - -Inputs: - input\specialGenerators.csv - output\gmTruckDataII.csv - -Outputs: - output\gmTruckDataIISP.csv - -**********************************************************************************************************/ -Macro "trkSpecialGen" (strFyear,intFyear,arrInterimYear) - shared path, inputDir, outputDir - - booInterimYear = arrInterimYear[1] - // 1 = Forecast year, 2 = Interim year and needs interpolation - if booInterimYear = 2 then do - prevYear = arrInterimYear[2] - nextYear = arrInterimYear[3] - end - - // Open truck trips and truck special generators - baseTrucks = Opentable("TruckGeneration", "CSV", {outputDir+"\\gmTruckDataII.csv"}) - specGenerators = Opentable("SpecGenMilit", "CSV",{inputDir+"\\specialGenerators.csv"}) - jv1 = Joinviews("JV1", baseTrucks+".ZONE", specGenerators+".TAZ", ) - Setview(jv1) - - // Forecast year has data available - if booInterimYear = 1 then do - col_name="Y"+strFyear - lhd_a = CreateExpression(jv1, "LHD_Attr", "LHD_Attractions + Nz("+col_name+" * trkAttraction * lhdShare)", ) - lhd_p = CreateExpression(jv1, "LHD_Prod", "LHD_Productions + Nz("+col_name+" * trkProduction * lhdShare)", ) - mhd_a = CreateExpression(jv1, "MHD_Attr", "MHD_Attractions + Nz("+col_name+" * trkAttraction * mhdShare)", ) - mhd_p = CreateExpression(jv1, "MHD_Prod", "MHD_Productions + Nz("+col_name+" * trkProduction * mhdShare)", ) - hhd_a = CreateExpression(jv1, "HHD_Attr", "HHD_Attractions + Nz("+col_name+" * trkAttraction * hhdShare)", ) - hhd_p = CreateExpression(jv1, "HHD_Prod", "HHD_Productions + Nz("+col_name+" * trkProduction * hhdShare)", ) - RunMacro("HwycadLog",{"TruckModel.rsc: trkSpecialGen","ExportView P&A + Special Generators"}) - ExportView(jv1+"|", "CSV", outputDir+"\\gmTruckDataIISP.csv", {baseTrucks+".ZONE", "LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr"}, {{"CSV Header"}}) - end - - // If data is not available then interpolate from closest available years - else do - // Get previous and next closest year - dim IY_Year[2] - IY_Year[1] = prevYear - IY_Year[2] = nextYear - - for j = 1 to 2 do - view="jv"+I2S(j) - Setview(view) - col_name="Y"+I2S(IY_Year[j]) - lhd_a = CreateExpression(view, "LHD_Attr", "LHD_Attractions + Nz("+col_name+" * trkAttraction * lhdShare)", ) - lhd_p = CreateExpression(view, "LHD_Prod", "LHD_Productions + Nz("+col_name+" * trkProduction * lhdShare)", ) - mhd_a = CreateExpression(view, "MHD_Attr", "MHD_Attractions + Nz("+col_name+" * trkAttraction * mhdShare)", ) - mhd_p = CreateExpression(view, "MHD_Prod", "MHD_Productions + Nz("+col_name+" * trkProduction * mhdShare)", ) - hhd_a = CreateExpression(view, "HHD_Attr", "HHD_Attractions + Nz("+col_name+" * trkAttraction * hhdShare)", ) - hhd_p = CreateExpression(view, "HHD_Prod", "HHD_Productions + Nz("+col_name+" * trkProduction * hhdShare)", ) - RunMacro("HwycadLog",{"TruckModel.rsc: trkSpecialGen","ExportView P&A + Special Generators"}) - ExportView(view+"|", "CSV", outputDir+"\\gmTruckDataIISP"+I2S(IY_Year[j])+".csv", {"ZONE", "LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr"}, {{"CSV Header"}}) - Closeview(view) - jv2 = Joinviews("JV2", baseTrucks+".ZONE", specGenerators+".TAZ", ) - end - - ok=RunMacro("Interpolate",{"gmTruckDataIISP"+I2S(prevYear)+".csv","gmTruckDataIISP"+I2S(nextYear)+".csv","gmTruckDataIISP.csv",intFyear,prevYear,nextYear}) - end - -EndMacro - - -/********************************************************************************************************** -Balances total production and attraction for lhd, mhd, hhd, ei and ie trips - -Inputs: - input\specialGenerators.csv - output\gmTruckDataII.csv - - inputTruckDir\regionalEItrips.csv - inputTruckDir\regionalEItrips.csv - inputTruckDir\regionalIEtrips.csv - inputTruckDir\regionalIEtrips.csv - inputTruckDir\regionalEEtrips.csv - inputTruckDir\regionalEEtrips.csv - -Outputs: - output\gmTruckDataBalanced.bin - -**********************************************************************************************************/ -Macro "trkBalance" (strFyear,intFyear,arrInterimYear) - shared path, inputDir, outputDir, inputTruckDir - - // Check if interim year and interpolate trip files if it is - booInterimYear = arrInterimYear[1] - - // 1 = Forecast year, 2 = Interim year and needs interpolation - if booInterimYear = 2 then do - prevYear = arrInterimYear[2] - nextYear = arrInterimYear[3] - - // copy files to scenario directory - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEItrips"+I2S(prevYear)+".csv",outputDir+"\\regionalEItrips_prev.csv"}) - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEItrips"+I2S(nextYear)+".csv",outputDir+"\\regionalEItrips_next.csv"}) - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalIEtrips"+I2S(prevYear)+".csv",outputDir+"\\regionalIEtrips_prev.csv"}) - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalIEtrips"+I2S(nextYear)+".csv",outputDir+"\\regionalIEtrips_next.csv"}) - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+I2S(prevYear)+".csv",outputDir+"\\regionalEEtrips_prev.csv"}) - ok=RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+I2S(nextYear)+".csv",outputDir+"\\regionalEEtrips_next.csv"}) - - // Call Macro Interpolate //arr = {"previous year data file","next year data file","new year data file name(macro will create this)} - ok=RunMacro("Interpolate",{"regionalEItrips_prev.csv","regionalEItrips_next.csv","regionalEItrips.csv",intFyear,prevYear,nextYear}) - ok=RunMacro("Interpolate",{"regionalIEtrips_prev.csv","regionalIEtrips_next.csv","regionalIEtrips.csv",intFyear,prevYear,nextYear}) - ok=RunMacro("Interpolate",{"regionalEEtrips_prev.csv","regionalEEtrips_next.csv","regionalEEtrips.csv",intFyear,prevYear,nextYear}) - - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_prev.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_next.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_prev.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_next.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_prev.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_next.csv"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_prev.dcc"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEItrips_next.dcc"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_prev.dcc"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalIEtrips_next.dcc"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_prev.dcc"}) - ok=RunMacro("SDdeletefile",{outputDir+"\\regionalEEtrips_next.dcc"}) - end - - // If not interim year then copy EE to output folder - if booInterimYear = 1 then do - RunMacro("SDcopyfile",{inputTruckDir+"\\regionalEEtrips"+strFyear+".csv",outputDir+"\\regionalEEtrips.csv"}) - end - // Balance each truck type (except EE [external-external], which is already balanced) - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance LHD"}) - RunMacro("trkBalanceOneType", "LHD") - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance MHD"}) - RunMacro("trkBalanceOneType", "MHD") - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance HHD"}) - RunMacro("trkBalanceOneType", "HHD") - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance EI"}) - RunMacro("trkBalanceRegionalTrucks", "EI", strFyear,booInterimYear) - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","Balance IE"}) - RunMacro("trkBalanceRegionalTrucks", "IE", strFyear,booInterimYear) - - // Combine single balanced files - vw1 = Opentable("BALANCE_LHD", "FFB", {outputDir+"\\gmTruckDataBal_LHD.bin",}) - vw2 = Opentable("BALANCE_MHD", "FFB", {outputDir+"\\gmTruckDataBal_MHD.bin",}) - vw3 = Opentable("BALANCE_HHD", "FFB", {outputDir+"\\gmTruckDataBal_HHD.bin",}) - vw4 = Opentable("BALANCE_EI", "FFB", {outputDir+"\\regionalEItripsBalanced.bin",}) - RenameField("BALANCE_EI.HHD_Attr" ,"EI_Attr") - RenameField("BALANCE_EI.EItrucks" ,"EI_Prod") - - Opts = null - Opts.Input.[Dataview Set] = {outputDir+"\\regionalEItripsBalanced.bin" , vw4} - Opts.Global.Fields = {"EI_Prod"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if EI_Prod=null then 0.00 else EI_Prod"} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - - vw5 = Opentable("BALANCE_IE", "FFB", {outputDir+ "\\regionalIEtripsBalanced.bin",}) - RenameField("BALANCE_IE.IEtrucks" ,"IE_Attr") - RenameField("BALANCE_IE.HHD_Prod" ,"IE_Prod") - Opts = null - Opts.Input.[Dataview Set] = {outputDir+ "\\regionalIEtripsBalanced.bin" , vw5} - Opts.Global.Fields = {"IE_Attr"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"if IE_Attr=null then 0.00 else IE_Attr"} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - - jv1 = Joinviews("JV1", vw1 + ".ID1", vw2 + ".ID1",) - jv2 = Joinviews("JV2", jv1 + ".BALANCE_LHD.ID1", vw3 + ".ID1",) - jv3 = Joinviews("JV3", jv2 + ".BALANCE_LHD.ID1", vw4 + ".ID1",) - jv4 = Joinviews("JV4", jv3 + ".BALANCE_LHD.ID1", vw5 + ".ID1",) - RenameField("BALANCE_LHD.ID1", "ID") - - Setview(jv4) - n = SelectByQuery("notzeros", "several", "Select * where ID <> 0",) - - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalance","ExportView - Combined Balanced Flow"}) - ExportView("JV4|notzeros", "FFB", outputDir+"\\gmTruckDataBalanced.bin", {"ID","LHD_Prod", "LHD_Attr", "MHD_Prod", "MHD_Attr", "HHD_Prod", "HHD_Attr", "IE_Prod", "IE_Attr", "EI_Prod","EI_Attr"},) - - Closeview(jv1) - Closeview(jv2) - Closeview(jv3) - Closeview(jv4) - Closeview(vw1) - Closeview(vw2) - Closeview(vw3) - Closeview(vw4) - Closeview(vw5) -EndMacro - - -/********************************************************************************************************** -Balances total production and attraction for truck type (type = lhd, mhd or hhd) - -Inputs: - output\gmTruckDataIISP.csv - -Outputs: - output\gmTruckDataBal_lhd.bin - output\gmTruckDataBal_mhd.bin - output\gmTruckDataBal_hhd.bin -**********************************************************************************************************/ -Macro "trkBalanceOneType" (type) - shared path, inputDir, outputDir - - RunMacro("TCB Init") - - Opts = null - Opts.Input.[Data View Set] = {outputDir+"\\gmTruckDataIISP.csv", "gmTruckDataIISP"} - Opts.Field.[Vector 1] = {"[gmTruckDataIISP]." + type + "_Prod"} - Opts.Field.[Vector 2] = {"[gmTruckDataIISP]." + type + "_Attr"} - Opts.Global.[Holding Method] = {"Weighted Sum"} - Opts.Global.[Percent Weight] = {50} - Opts.Global.[Sum Weight] = {100} - Opts.Global.[Store Type] = 1 - Opts.Output.[Output Table] = outputDir+"\\gmTruckDataBal_" + type + ".bin" - - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalanceOneType", "Balance "+type}) - ok=RunMacro("TCB Run Procedure", "Balance", Opts, &Ret) -EndMacro - - -/********************************************************************************************************** -Balances EI trips where sum of E is fixed and I of truck attraction is adjusted or -IE trips where sum of E is fixed and I of truck production is adjusted - -Inputs: - output\regionalIEtrips.csv - output\regionalEItrips.csv - inputTruckDir\regionalIEtrips.csv - inputTruckDir\regionalIEtrips.csv - inputTruckDir\regionalEItrips.csv - inputTruckDir\regionalEItrips.csv - output\gmTruckDataBal_HHD.bin - -Outputs: - output\regionaltripsBalanced.bin - -**********************************************************************************************************/ -Macro "trkBalanceRegionalTrucks" (direction,strFyear,booInterimYear) - shared path, inputDir, outputDir, inputTruckDir - - // Not an interim year, read regional trips from inputTruckDir - if booInterimYear=1 then do - strPathRegTrips=inputTruckDir+"\\regional"+direction+"trips"+strFyear+".csv" - end - - //Interim year, read regional trips from output directory - else do - strPathRegTrips=outputDir+"\\regional"+direction+"trips.csv" - end - - Opts = null - if direction = "EI" then do - Opts.Input.[Data Set] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "fromZone"}, "BALANCE_HHD+regionalEItrips"+strFyear} - Opts.Input.[Data View] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "fromZone"}, "BALANCE_HHD+regionalEItrips"+strFyear} - end - else do - Opts.Input.[Data Set] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "toZone"}, "BALANCE_HHD+regionalIEtrips"+strFyear} - Opts.Input.[Data View] = {{outputDir+"\\gmTruckDataBal_HHD.bin", strPathRegTrips, "ID1", "toZone"}, "BALANCE_HHD+regionalIEtrips"+strFyear} - end - - Opts.Input.[V1 Holding Sets] = {} - Opts.Input.[V2 Holding Sets] = {} - if direction = "EI" then do - Opts.Field.[Vector 1] = {"[BALANCE_HHD+regionalEItrips"+strFyear+"].HHD_Attr"} - Opts.Field.[Vector 2] = {"[BALANCE_HHD+regionalEItrips"+strFyear+"].EITrucks"} - end - else do - Opts.Field.[Vector 1] = {"[BALANCE_HHD+regionalIEtrips"+strFyear+"].HHD_Prod"} - Opts.Field.[Vector 2] = {"[BALANCE_HHD+regionalIEtrips"+strFyear+"].IETrucks"} - end - - Opts.Global.Pairs = 1 - Opts.Global.[Holding Method] = {2} - Opts.Global.[Percent Weight] = {50} - Opts.Global.[Sum Weight] = {100} - Opts.Global.[V1 Options] = {1} - Opts.Global.[V2 Options] = {1} - Opts.Global.[Store Type] = 1 - Opts.Output.[Output Table] = outputDir+"\\regional"+direction+"tripsBalanced.bin" - - RunMacro("HwycadLog",{"TruckModel.rsc: trkBalanceRegionalTrucks","Balance "+direction}) - RunMacro("TCB Run Procedure", 1, "Balance", Opts) -EndMacro - - - -/********************************************************************************************************** -Distributes truck lhd, mhd, hhd, ie and ei truck trips - -Inputs: - output\gmTruckDataBalanced.bin - output\regionalEEtrips.csv - output\imptrk_EA.mtx - output\imptrk_AM.mtx - output\imptrk_MD.mtx - output\imptrk_PM.mtx - output\imptrk_EV.mtx - -Outputs: - output\distributionMatricesTruck.mtx - outputDir\regionalEEtrips.mtx - outputDir\dailyDistributionMatricesTruckAll.mtx - outputDir\dailyDistributionMatricesTruckEA.mtx - outputDir\dailyDistributionMatricesTruckAM.mtx - outputDir\dailyDistributionMatricesTruckMD.mtx - outputDir\dailyDistributionMatricesTruckPM.mtx - outputDir\dailyDistributionMatricesTruckEV.mtx - -**********************************************************************************************************/ - -Macro "trkDistribution" (properties) - shared path, inputDir, outputDir - - // Get forecast year - strFyear=RunMacro("read properties",properties,"truck.FFyear","S") - - //-------------------------------------------------- - // This section does the truck trip distribution - //-------------------------------------------------- - // Use mid-day (md) truck skim for distribution - periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} - p = 3 - md_truck_skim = outputDir +"\\imptrk"+periods[p]+".mtx" - md_truck_FF = "*SCST"+periods[p] - md_truck_IM = "*STM"+periods[p]+" (Skim)" - md_truck_KF = "*STM"+periods[p]+" (Skim)" - - // Open md truck skims - Opts.Input.[FF Matrix Currencies] = {{md_truck_skim, md_truck_FF,,},{md_truck_skim, md_truck_FF,,}, {md_truck_skim, md_truck_FF,,}, {md_truck_skim, md_truck_FF,,},{md_truck_skim, md_truck_FF,,}} - Opts.Input.[Imp Matrix Currencies] = {{md_truck_skim, md_truck_IM,,},{md_truck_skim, md_truck_IM,,}, {md_truck_skim, md_truck_IM,,}, {md_truck_skim, md_truck_IM,,},{md_truck_skim, md_truck_IM,,}} - Opts.Input.[KF Matrix Currencies] = {{md_truck_skim, md_truck_KF,,},{md_truck_skim, md_truck_KF,,}, {md_truck_skim, md_truck_KF,,}, {md_truck_skim, md_truck_KF,,},{md_truck_skim, md_truck_KF,,}} - - // Open truck trips - Opts.Input.[PA View Set] = {outputDir+"\\gmTruckDataBalanced.bin", "gmTruckDataBalanced"} - - // Set gravity model settings - Opts.Input.[FF Tables] = {{outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}, {outputDir+"\\gmTruckDataBalanced.bin"}} - Opts.Field.[Prod Fields] = {"gmTruckDataBalanced.LHD_Prod", "gmTruckDataBalanced.MHD_Prod", "gmTruckDataBalanced.HHD_Prod", "gmTruckDataBalanced.IE_Prod", "gmTruckDataBalanced.EI_Prod"} - Opts.Field.[Attr Fields] = {"gmTruckDataBalanced.LHD_Attr", "gmTruckDataBalanced.MHD_Attr", "gmTruckDataBalanced.HHD_Attr", "gmTruckDataBalanced.IE_Attr", "gmTruckDataBalanced.EI_Attr"} - Opts.Field.[FF Table Fields] = {"gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID"} - Opts.Field.[FF Table Times] = {"gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID", "gmTruckDataBalanced.ID"} - Opts.Global.[Purpose Names] = {"lhd", "mhd", "hhd", "IE", "EI"} - Opts.Global.Iterations = {100, 100, 100, 100, 100} - Opts.Global.Convergence = {0.01, 0.01, 0.01, 0.01, 0.01} - Opts.Global.[Constraint Type] = {"Double", "Double", "Double", "Columns", "Rows"} - Opts.Global.[Fric Factor Type] = {"Exponential", "Exponential", "Exponential", "Exponential", "Exponential"} - Opts.Global.[A List] = {1, 1, 1, 1, 1} - Opts.Global.[B List] = {0.3, 0.3, 0.3, 0.3, 0.3} - Opts.Global.[C List] = {0.045, 0.03, 0.03, 0.03, 0.03} - Opts.Flag.[Use K Factors] = {0, 0, 0, 0, 0} - Opts.Output.[Output Matrix].Label = "Distribution Matrix" - Opts.Output.[Output Matrix].Compression = 1 - Opts.Output.[Output Matrix].[File Name] = outputDir + "\\distributionMatricesTruck.mtx" - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Gravity"}) - RunMacro("TCB Run Procedure", 1, "Gravity", Opts) - - - //-------------------------------------------------- - // Adds IE, EI and EE trips to the truck trips by type - //-------------------------------------------------- - // Create EE trip matrix - viewEE = Opentable("EEtrips", "CSV", {outputDir + "\\regionalEEtrips.csv"}) - matrixEE = CreateMatrixFromView("EEmatrix", "EEtrips|", "fromZone", "toZone", {"EEtrucks"}, {{"File Name", outputDir + "\\regionalEEtrips.mtx"}, {"Type", "Float"}, {"Sparse", "No"}, {"Column Major", "No"}, {"File Based", "Yes"}}) - Closeview(viewEE) - - // Split truck trips by type - trkShare = {0.307, 0.155, 0.538} - trkTypes = {"lhd", "mhd", "hhd"} - - Opts = null - Opts.Input.[Matrix Currencies] = {{outputDir + "\\distributionMatricesTruck.mtx", "lhd", "Row ID's", "Col ID's"}, - {outputDir + "\\distributionMatricesTruck.mtx", "mhd", "Row ID's", "Col ID's"}, - {outputDir + "\\distributionMatricesTruck.mtx", "hhd", "Row ID's", "Col ID's"}, - {outputDir + "\\distributionMatricesTruck.mtx", "IE", "Row ID's", "Col ID's"}, - {outputDir + "\\distributionMatricesTruck.mtx", "EI", "Row ID's", "Col ID's"}, - {outputDir + "\\regionalEEtrips.mtx", "EEtrucks", "fromZone", "toZone"}} - Opts.Global.Operation = "Union" - Opts.Output.[Combined Matrix].Label = "Union Combine" - Opts.Output.[Combined Matrix].Compression = 1 - Opts.Output.[Combined Matrix].[File Name] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Combine Matrix Files"}) - RunMacro("TCB Run Operation", 1, "Combine Matrix Files", Opts) - - - // Aportion and add IE, EI and EE trips to LHD, MHD and HHD - for i = 1 to trkTypes.length do - Opts = null - Opts.Input.[Matrix Currency] = {outputDir + "\\dailyDistributionMatricesTruckAll.mtx", trkTypes[i], "Rows", "Columns"} - Opts.Input.[Core Currencies] = {{outputDir + "\\dailyDistributionMatricesTruckAll.mtx",trkTypes[i], "Rows", "Columns"}, - {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","EI", "Rows", "Columns"}, - {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","IE", "Rows", "Columns"}, - {outputDir + "\\dailyDistributionMatricesTruckAll.mtx","EEtrucks", "Rows", "Columns"}} - Opts.Global.Method = 7 - Opts.Global.[Cell Range] = 2 - Opts.Global.[Matrix K] = {1, trkShare[i], trkShare[i], trkShare[i]} - Opts.Global.[Force Missing] = "No" - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","Fill Matrices"}) - ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) - end - - // Drop IE matrix core - Opts = null - Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" - Opts.global.[Drop Core] = {"IE"} - RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) - - // Drop EI matrix core - Opts = null - Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" - Opts.global.[Drop Core] = {"EI"} - RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) - - // Drop EE matrix core - Opts = null - Opts.input.[Input Matrix] = outputDir + "\\dailyDistributionMatricesTruckAll.mtx" - Opts.global.[Drop Core] = {"EEtrucks"} - RunMacro("TCB Run Operation", 1, "Drop Matrix Core", Opts) - - - //-------------------------------------------------- - // Set intrazonal truck trips to 0. Note: intrazonal truck trips are not necessarily 0 in reality, but they are - // not simulated in this model. Having an undefined value for intrazonal trips provides problems in the emission - // estimation of the EMFAC2007 model. Therefore, intrazonals are artificially set to 0. - //-------------------------------------------------- - Opts = null - Opts.Input.[Matrix Currency] = {outputDir + "\\dailyDistributionMatricesTruckAll.mtx", "lhd", "Rows", "Columns"} - Opts.Global.Method = 1 - Opts.Global.Value = 0 - Opts.Global.[Cell Range] = 3 - Opts.Global.[Matrix Range] = 3 - Opts.Global.[Matrix List] = {"lhd", "mhd", "hhd"} - ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) - if !ok then goto quit - - // Split into time of day - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, EA"}) - RunMacro("splitIntoTimeOfDay", 1) // EA Off-peak - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, AM"}) - RunMacro("splitIntoTimeOfDay", 2) // AM Peak - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, MD"}) - RunMacro("splitIntoTimeOfDay", 3) // MD Off-peak - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, PM"}) - RunMacro("splitIntoTimeOfDay", 4) // PM Peak - RunMacro("HwycadLog",{"TruckModel.rsc: trkDistribution","splitIntoTimeOfDay, EV"}) - RunMacro("splitIntoTimeOfDay", 5) // EV Off-peak - - run_ok=1 - Return(run_ok) - - quit: - Return(ok) -EndMacro - - -/********************************************************************************************************** -Splits daily truck trips to five periods EA, AM, MD, PM and EV - -Inputs: - outputDir\dailyDistributionMatricesTruckAll.mtx - timeShare = {0.1018, 0.1698, 0.4284, 0.1543, 0.1457} - borderTimeShare = {0.0188, 0.1812, 0.4629, 0.2310, 0.1061} - -Outputs: - outputDir\dailyDistributionMatricesTruckEA.mtx - outputDir\dailyDistributionMatricesTruckAM.mtx - outputDir\dailyDistributionMatricesTruckMD.mtx - outputDir\dailyDistributionMatricesTruckPM.mtx - outputDir\dailyDistributionMatricesTruckEV.mtx - -**********************************************************************************************************/ -Macro "splitIntoTimeOfDay" (period) - shared path, inputDir, outputDir - - // Split lhd, mhd and hhd truck trips into time-of-day periods - periodName = {"EA","AM","MD","PM","EV"} - mode = {"lhd", "mhd", "hhd"} - timeShare = {0.1018, 0.1698, 0.4284, 0.1543, 0.1457} // share of truck trips per time period for all zones except border crossings - borderTimeShare = {0.0188, 0.1812, 0.4629, 0.2310, 0.1061} // share of truck trips per time period for border crossings - dim borderCorrection[timeShare.length] // correct values at border after multiplication with timeShare - for i = 1 to borderCorrection.length do - borderCorrection[i] = borderTimeShare[i] / timeShare[i] - end - - // Copy original matrix into time-of-day matrix - mat = OpenMatrix(outputDir + "\\dailyDistributionMatricesTruckAll.mtx", ) - mc = CreateMatrixCurrency(mat, "lhd", "Rows", "Columns", ) - newFile = outputDir + "\\dailyDistributionMatricesTruck" + periodName[period] + ".mtx" - label = "Truck Table " + periodName[period] + " Peak" - new_mat = CopyMatrix(mc, {{"File Name", newFile}, {"Label", label}, {"File Based", "Yes"}}) - - // Multiply entire matrix with time-of-day share - Opts = null - Opts.Input.[Matrix Currency] = {newFile, "lhd", "Rows", "Columns"} - Opts.Global.Method = 5 - Opts.Global.Value = timeShare[period] - Opts.Global.[Cell Range] = 2 - Opts.Global.[Matrix Range] = 3 - Opts.Global.[Matrix List] = {"lhd", "mhd", "hhd"} - RunMacro("HwycadLog",{"TruckModel.rsc: splitIntoTimeOfDay","Fill Matrices"}) - ok = RunMacro("TCB Run Operation", 1, "Fill Matrices", Opts) - if !ok then goto quit - - // Correct time-of-day share for destination zones 1 through 5 (border zones) - for i = 1 to mode.length do - mat = OpenMatrix(newFile,) - mc = CreateMatrixCurrency(mat, mode[i], "Rows", "Columns", ) - cols = {"1", "2", "3", "4", "5"} - operation = {"Multiply", borderCorrection[period]} - FillMatrix(mc, null, cols, operation, ) - - mat = OpenMatrix(newFile,) - mc = CreateMatrixCurrency(mat, mode[i], "Rows", "Columns", ) - rows = {"1", "2", "3", "4", "5"} - operation = {"Multiply", borderCorrection[period]} - FillMatrix(mc, rows, null, operation, ) - end - - run_ok=1 - Return(run_ok) - - quit: - Return(ok) -EndMacro - - - - -/********************************************************************************************************** - Truck Toll Diversion Model - Splits Truck Demand to Non-Toll and Toll - called after truck model but before combine trktrips - Ben Stabler, stabler@pbworld.com, 12/02/10 - -Inputs: - Each skim matrix is suffixed with xx, yy where: - xx is mode indicating following truck types: - lhdn, mhdn, hhdn, lhdt, mhdt, and hhdt - - yy is period, as follows: - EA: Early AM - AM: AM peak - MD: Midday - PM: PM peak - EV: Evening - - output\impXXYY.mtx - outputDir\dailyDistributionMatricesTruckYY.mtx - -Outputs: - Adds toll and non-toll cores to - outputDir\dailyDistributionMatricesTruckYY.mtx - -**********************************************************************************************************/ -Macro "trk toll diversion model" - shared path, inputDir, outputDir - RunMacro("TCB Init") - - // Toll diversion curve settings - nest_param = 10 - vot = 0.02 //(minutes/cent) - - periodName = {"EA","AM","MD","PM","EV"} // must be consistent with filename arrays below - trkTypes = {"lhd", "mhd", "hhd"} // truck types - trkTollFactor = {1, 1.03, 2.33} // truck toll factor - - // Loop by time period - for period = 1 to periodName.length do - - // Open truck trips - fileNameTruck = outputDir + "\\dailyDistributionMatricesTruck" + periodName[period] + ".mtx" - m = OpenMatrix(fileNameTruck,) - - // Loop by truck class - for trkType = 1 to trkTypes.length do - nontollmtx=outputDir+"\\imp"+trkTypes[trkType]+"n_"+periodName[period]+".mtx" // non-toll skims - tollmtx=outputDir+"\\imp"+trkTypes[trkType]+"t_"+periodName[period]+".mtx" // toll skims - - // Check and if exist, drop toll and non-toll matrix cores by truck type - coreNames = GetMatrixCoreNames(m) - for c = 1 to coreNames.length do - if (coreNames[c] = trkTypes[trkType] + "t") then DropMatrixCore(m, trkTypes[trkType] + "t") - if (coreNames[c] = trkTypes[trkType] + "n") then DropMatrixCore(m, trkTypes[trkType] + "n") - end - - // Add toll and non-toll matrix - AddMatrixCore(m, trkTypes[trkType] + "t") - AddMatrixCore(m, trkTypes[trkType] + "n") - - // Diversion curve (time is in minutes, cost is in cents) - utility = "(([impedance truck].[*STM_"+periodName[period]+" (Skim)] - [impedance truck toll].[*STM_"+periodName[period]+" (Skim)]) - " + - String(vot) + " * " + "[impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"] * " + String(trkTollFactor[trkType]) + " ) / " + String(nest_param) - - expression = "if([impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"]!=0) then ( 1 / ( 1 + exp(-" + utility + ") ) ) else [impedance truck toll].["+trkTypes[trkType]+"t - "+"ITOLL2_"+periodName[period]+"]" - - // Calculate toll matrix - Opts = null - Opts.Input.[Matrix Currency] = {fileNameTruck, trkTypes[trkType] + "t", "Rows", "Columns"} - Opts.Input.[Formula Currencies] = {{nontollmtx, "*STM_"+periodName[period]+" (Skim)", "Origin", "Destination"}, {tollmtx, "*STM_"+periodName[period]+" (Skim)", "Origin", "Destination"}} - Opts.Global.Method = 11 - Opts.Global.[Cell Range] = 2 - Opts.Global.[Expression Text] = "[" + trkTypes[trkType] + "] * " + expression - Opts.Global.[Formula Labels] = {"impedance truck", "impedance truck toll"} - Opts.Global.[Force Missing] = "Yes" - ok = RunMacro("TCB Run Operation", "Fill Matrices", Opts) - if !ok then goto quit - - // Calculate non-toll matrix - mc_n = CreateMatrixCurrency(m, trkTypes[trkType] + "n", "Rows", "Columns",) - mc_t = CreateMatrixCurrency(m, trkTypes[trkType] + "t", "Rows", "Columns",) - mc = CreateMatrixCurrency(m, trkTypes[trkType], "Rows", "Columns",) - mc_n := mc - mc_t - end - end - - //return 1 if macro completed - run_ok = 1 - Return(run_ok) - - quit: - Return(ok) - -EndMacro - - -/********************************************************************************************************** -Used to gerenarte data for the forecast years from the years where data is avaialble -Interpolates data from the closest previous year and next years -All files are from read and written to output directory - -Inputs: - output\file.csv - output\file.csv - -Outputs: - output\file.csv - -**********************************************************************************************************/ -Macro "Interpolate" (arr) - shared outputDir - prevfile= arr[1] - nextfile= arr[2] - newfile = arr[3] - Fyear = arr[4] - prevYear= arr[5] - nextYear= arr[6] - - newview = ParseString(newfile, ".") // Get file name and use it as view name - pview = ParseString(prevfile, ".") // Get file name - nview = ParseString(nextfile, ".") // Get file name - - // Delete dictionary files if exist - di = GetDirectoryInfo(outputDir+"\\"+pview[1]+".dcc", "File") - if di.length > 0 then do - ok=RunMacro("SDdeletefile",{outputDir+"\\"+pview[1]+".dcc"}) - end - di = GetDirectoryInfo(outputDir+"\\"+nview[1]+".dcc", "File") - if di.length > 0 then do - ok=RunMacro("SDdeletefile",{outputDir+"\\"+nview[1]+".dcc"}) - end - di = GetDirectoryInfo(outputDir+"\\"+newview[1]+".dcc", "File") - if di.length > 0 then do - ok=RunMacro("SDdeletefile",{outputDir+"\\"+newview[1]+".dcc"}) - end - - // Open previous year data - prevview = Opentable("prevview", "CSV", {outputDir+"\\"+prevfile,}) - - // Open next year data - nextview = Opentable("nextview", "CSV", {outputDir+"\\"+nextfile,}) - - // Number of zones (assuming both prev and next year contain same number of zones) - zones = GetRecordCount(prevview, null) - - // Get data table structure - str = GetTableStructure(prevview) - dim pfieldName[str.length],nfieldName[str.length] - for i =1 to str.length do - pfieldName[i] = prevview+"."+str[i][1] // gets field names from previous year data - nfieldName[i] = nextview+"."+str[i][1] // gets field names from previous year data - end - - // Get fields from previous and next years - prevf = GetDataVectors(prevview+"|",pfieldName,) - nextf = GetDataVectors(nextview+"|",nfieldName,) - - // Create a Fyear table and do interpolation - view = CreateTable(newview[1], outputDir+"\\"+ newfile, "CSV", str) - SetView(view) - - // Interpolate and set values for Fyear - for i = 1 to zones do // row loop - dim v[str.length] // array to hold new field computation - v[1] = {str[1][1], prevf[1][i]} // fill new fields, first field is zone and no interpolation - - for j = 1 to str.length do // field loop - // interpolate - v[j] = {str[j][1] , (prevf[j][i]+((Fyear-prevYear)*((nextf[j][i]-prevf[j][i])/(nextYear-prevYear))))} - end - rh = AddRecord(view,v) - end - - // Close all opened views - vws = GetViewNames() - for p = 1 to vws.length do - CloseView(vws[p]) - end -EndMacro - diff --git a/sandag_abm/src/main/gisdk/Utilities.rsc b/sandag_abm/src/main/gisdk/Utilities.rsc deleted file mode 100644 index 7fcefcc..0000000 --- a/sandag_abm/src/main/gisdk/Utilities.rsc +++ /dev/null @@ -1,568 +0,0 @@ -Macro "ModifyOptionsOption" (options_array,option_key,key,value) - spec = options_array.(option_key) - spec.(key) = value -EndMacro - -Macro "CloseAll" - // close all files in workspace - map_arr=GetMaps() - if ArrayLength(map_arr)>0 then do - open_maps=ArrayLength(map_arr[1]) - for mm=1 to open_maps do - CloseMap(map_arr[1][mm]) - end - end - - On NotFound goto no_more_eds - still_more_eds: - CloseEditor() - goto still_more_eds - - no_more_eds: - On NotFound default - - view_arr=GetViews() - if ArrayLength(view_arr)>0 then do - On NotFound goto cont_views - open_views=ArrayLength(view_arr[1]) - for vv=1 to open_views do - CloseView(view_arr[1][vv]) - cont_views: - end - end -endMacro - -Macro "IsMapOpen" (map) - maps = GetMapNames() - for i = 1 to maps.length do - if maps[i] = map then do - return(True) - end - end - return(False) -EndMacro - -Macro "IsViewOpen" (view) - views = GetViewNames() - for i = 1 to views.length do - if views[i] = view then do - return(True) - end - end - return(False) -EndMacro - -Macro "SafeDeleteFile" (file) - //just ignores any errors - if GetFileInfo(file) <> null then do - On Error goto safe_delete_error - DeleteFile(file) - safe_delete_error: - On Error default - end -EndMacro - -Macro "DeleteFiles" (path) - files = GetDirectoryInfo(path,"All") - for i = 1 to files.length do - DeleteFile(RunMacro("FormPath",{path,files[i][1]})) - end -EndMacro - -Macro "SafeDeleteFiles" (path) - files = GetDirectoryInfo(path,"All") - for i = 1 to files.length do - RunMacro("SafeDeleteFile",RunMacro("FormPath",{path,files[i][1]})) - end -EndMacro - -Macro "SafeDeleteDatabase" (database_file) - //just ignores any errors - On Error goto safe_delete_database_error - On NotFound goto safe_delete_database_error - DeleteDatabase(file) - safe_delete_database_error: - On Error default - On NotFound default -EndMacro - -Macro "NormalizePath" (path) - if Len(path) > 1 and path[2] = ":" then do - path = Lower(path[1]) + Right(path,Len(path)-1) - end - return(Substitute(path,"/","\\",)) -EndMacro - -Macro "FormPath" (path_elements) - if TypeOf(path_elements) <> "array" then do - ShowMessage("Must form a path out of a list of elements, not: " + TypeOf(path_elements)) - ShowMessage(2) - end - //path_elements is an array of elements - path = "" - for i = 1 to path_elements.length do - //change / to \ - p = RunMacro("NormalizePath",path_elements[i]) - if Right(p,1) = "\\" then do - if Len(p) > 1 then do - p = Substring(p,1,Len(p)-1) - end - else do - p = "" - end - end - if Left(p,1) = "\\" then do - if Len(p) > 1 then do - p = Substring(p,2,Len(p)) - end - else do - p = "" - end - end - if path = "" then do - path = p - end - else do - path = path + "\\" + p - end - end - return(path) -EndMacro - -Macro "CreateMapForDatabase" (database_file,map_name) - linfo=GetDBInfo(database_file) - scope=linfo[1] - maps = GetMapNames() - map_name_not_ok = true - while map_name_not_ok do - map_name_not_ok = false - for i = 1 to maps.length do - if maps[i] = map_name then do - map_name = map_name + " " - map_name_not_ok = true - end - end - end - map=createMap(map_name,{{"Scope", scope},{"Auto Project","True"}}) - SetMapUnits("Miles") -EndMacro - -Macro "OpenDatabaseInMap" (database_file,map) - info=GetDBInfo(database_file) - map_made = False - if map <> null then do - map_made = RunMacro("IsMapOpen",map) - end - else do - map = info[2] - end - if not map_made then do - RunMacro("CreateMapForDatabase",database_file,map) - end - NewLayer=GetDBLayers(database_file) - layer = AddLayer(map,NewLayer[1],database_file,NewLayer[1]) - if NewLayer.length = 2 then do - //assumes it is a network file, and hides the nodes and adds the lines - SetLayerVisibility(map + "|" + layer,"False") - AddLayer(map,NewLayer[2],database_file,NewLayer[2]) - end - return(map) -EndMacro - -Macro "OpenDatabase" (database_file) - return(RunMacro("OpenDatabaseInMap",database_file,)) -EndMacro - -Macro "OpenRouteSystemInMap" (route_system_file,map) - info = GetRouteSystemInfo(route_system_file) - map_made = False - if map <> null then do - map_made = RunMacro("IsMapOpen",map) - end - else do - map = info[3].Label - end - if not map_made then do - RunMacro("CreateMapForDatabase",info[1],map) - end - RunMacro("Set Default RS Style",AddRouteSystemLayer(map,info[3].Label,route_system_file,),"TRUE","FALSE") - return(map) -EndMacro - -Macro "OpenRouteSystem" (route_system_file) - return(RunMacro("OpenRouteSystemInMap",route_system_file,)) -EndMacro - -Macro "CleanRecordValuesOptionsArray" (options_array,view_name) - for i = 1 to options_array.length do - options_array[i][1] = Substitute(options_array[i][1],view_name + ".","",) - end -EndMacro - -Macro "FormFieldSpec" (view,field) - //don't think the following is necessary - //issue_chars = {":"} - //fix = False - //for i = 1 to issue_chars.length do - // if Position(field,issue_chars[i]) > 0 then do - // fix = True - // end - //end - //if fix then do - // field = "[" + field + "]" - //end - return(view + "." + field) -EndMacro - -Macro "ToString" (value) - type = TypeOf(value) - if type = "string" then do - return(value) - end - else if type = "int" then do - return(i2s(value)) - end - else if type = "double" then do - return(r2s(value)) - end - else if type = "null" then do - return("") - end - ShowMessage("Type " + type + " not supported by ToString method") -EndMacro - -Macro "GetArrayIndex" (array,value) - //returns the index of value in array, or 0 if it is not found - type = TypeOf(value) - for i = 1 to array.length do - if TypeOf(array[i]) = type and array[i] = value then do - return(i) - end - end - return(0) -EndMacro - -Macro "ArraysEqual" (array1,array2) - if array1.length <> array2.length then do - return(False) - end - for i = 1 to array1.length do - if TypeOf(array1[i]) = "array" then do - if TypeOf(array2[i]) = "array" then do - if not RunMacro("ArraysEqual",array1[i],array2[i]) then do - return(False) - end - end - else do - return(False) - end - end - else if TypeOf(array2[i]) = "array" then do - return(False) - end - else do - if array1[i] <> array2[i] then do - return(False) - end - end - end - return(True) -EndMacro - -Macro "GetDatabaseColumns" (database_file,layer_name) - columns = null - if database_file <> null and GetFileInfo(database_file) <> null then do - current_layer = GetLayer() - current_view = GetView() - lyr = AddLayerToWorkspace("__temp__",database_file,layer_name,{{"Shared","True"}}) - layer_in_use = lyr <> "__temp__" - SetLayer(lyr) - v = GetView() - info = GetTableStructure(v) - for i = 1 to info.length do - columns = columns + {info[i][1]} - end - if not layer_in_use then do - DropLayerFromWorkspace(lyr) - end - if current_layer <> null then do - SetLayer(current_layer) - end - if current_view <> null then do - SetView(current_view) - end - end - return(columns) -EndMacro - -//same as built in TC function, but with error checking for escape and for if a file is in use -Macro "ChooseFileName" (file_types,title,options) - on escape do - fname = null - goto cfn_done - end - openfile: - fname = ChooseFileName(file_types,title,options) - if FileCheckUsage({fname},) then do - ShowMessage("File already in use. Please choose again.") - goto openfile - end - cfn_done: - on escape default - return(fname) -EndMacro - -Macro "RunProgram" (program_with_arguments,working_directory) //can't get output file to work right now..boo hoo - wd = "" - if working_directory <> null then do - wd = " /D" + working_directory - end - RunProgram("cmd /s /c \"start \"cmd\" " + wd + " /WAIT " + program_with_arguments + "\"",) -EndMacro - -Macro "AddElementToSortedArraySet" (array,element) - index = array.length + 1 - not_done = True - for i = 1 to array.length do - if not_done then do - if element = array[i] then do - index = -1 - not_done = False - end - else if element < array[i] then do - index = i - not_done = False - end - end - end - if index > 0 then do - array = InsertArrayElements(array,index,{element}) - end - return(array) -EndMacro - -Macro "ClearAndDeleteDirectory" (path) - //this doesn't do any error handling - info = GetDirectoryInfo(RunMacro("FormPath",{path,"*"}),"All") - for i = 1 to info.length do - f = RunMacro("FormPath",{path,info[i][1]}) - if info[i][2] = "file" then do - DeleteFile(f) - end - else if info[i][2] = "directory" then do - RunMacro("ClearAndDeleteDirectory",f) - end - end - RemoveDirectory(path) -EndMacro - -Macro "ReadPropertiesFile" (properties_file) - props = null - f = OpenFile(properties_file,"r") - while not FileAtEOF(f) do - line = Trim(ReadLine(f)) - if Len(line) > 0 then do - subs = ParseString(line,"=", {{"Include Empty",True}}) - key = subs[1] - value = JoinStrings(Subarray(subs,2,subs.length-1),"=") - props.(Trim(key)) = Trim(value) - end - end - CloseFile(f) - return(props) -EndMacro - -Macro "DetokenizePropertyValues" (properties,token_map) - for i = 1 to properties.length do - value = token_map[i][2] - for j = 1 to token_map.length do - value = Substitute(value,token_map[i][1],token_map[i][2],) - end - token_map[i][2] = value - end -EndMacro - -Macro "ComputeAreaBufferOverlayPercentages" (area_layer_file,centroid_layer_file,centroid_query,area_taz_field,node_taz_field,buffer_size) - //assumes node layer holds centroids from area layer, and bases its buffer around this - //returns array of percentage arrays, each holding {centroid_taz,overlay_taz,percentage} - omap_name = GetMap() - olayer_name = GetLayer() - oview_name = GetView() - - map = RunMacro("OpenDatabase",area_layer_file) - RunMacro("OpenDatabaseInMap",centroid_layer_file,map) - node_layer = GetMapLayers(map,"Point") - node_layer = node_layer[1][1] - area_layer = GetMapLayers(map,"Area") - area_layer = area_layer[1][1] - - SetLayer(node_layer) - centroid_selection = "centroids" - SelectByQuery(centroid_selection,"Several",centroid_query) - node_ids = GetSetIDs(node_layer + "|" + centroid_selection) - node_to_taz = null - for i = 1 to node_ids.length do - node_id = node_ids[i] - value = GetRecordValues(node_layer,IDToRecordHandle(node_id),{node_taz_field}) - node_to_taz.(i2s(node_id)) = value[1][2] - end - - percentages = null - temp_dir = GetFileInfo(area_layer_file) - temp_dir = Substring(area_layer_file,1,Len(area_layer_file) - Len(temp_dir[1])) - intersection_file = "temp_buffers.dbd" - percentages_file = "tempintersect" - temp_intersection_file = RunMacro("FormPath",{temp_dir,intersection_file}) - temp_percentages_file = RunMacro("FormPath",{temp_dir,percentages_file}) - EnableProgressBar("Calculating Area Buffer Percentages (buffer = " + r2s(buffer_size) + ")", 1) // Allow only a single progress bar - CreateProgressBar("", "True") - - nlen = node_ids.length - //for i = 1 to nlen do - for i = 1 to 20 do - node_id = node_ids[i] - node_taz = node_to_taz.(i2s(node_id)) - stat = UpdateProgressBar("Zone: " + i2s(node_taz), r2i(i/nlen*100)) - if stat = "True" then do - percentages = null - goto quit_loop - end - SetLayer(node_layer) - SelectByQuery("centroid","Several","SELECT * WHERE id=" + i2s(node_id)) - CreateBuffers(temp_intersection_file,"buffers",{"centroid"},"Value",{buffer_size},{{"Interior","Separate"},{"Exterior","Separate"}}) - - NewLayer = GetDBLayers(temp_intersection_file) - intersection_layer = AddLayer(map,"inter",temp_intersection_file,NewLayer[1]) - SetLayer(area_layer) - n = SelectByVicinity("subtaz","several",node_layer+"|centroid",buffer_size,{{"Inclusion","Intersecting"}}) - if n > 0 then do - ComputeIntersectionPercentages({intersection_layer, area_layer + "|subtaz"}, temp_percentages_file + ".bin",) - t = OpenTable("int_table", "FFB", {temp_percentages_file + ".bin"},) - tbar = t + "|" - rh = GetFirstRecord(tbar,) - while rh <> null do - vals = GetRecordValues(t,rh,{"Area_1", "Area_2","Percent_2"}) - if vals[1][2] = 1 and vals[2][2] <> 0 then do - value = GetRecordValues(area_layer,IDToRecordHandle(vals[2][2]),{area_taz_field}) - area_taz = node_to_taz.(i2s(value[1][2])) - percentages = percentages + {{node_taz,area_taz,vals[3][2]}} - end - rh = GetNextRecord(t+"|",,) - end - CloseView(t) - end - DropLayer(map,intersection_layer) - end - - quit_loop: - DestroyProgressBar() - CloseMap(map) - if omap_name <> null then do - SetMap(omap_name) - if olayer_name <> null then do - SetLayer(olayer_name) - end - end - if oview_name <> null then do - SetView(oview_name) - end - DeleteDatabase(temp_intersection_file) - DeleteFile(temp_percentages_file + ".bin") - DeleteFile(temp_percentages_file + ".BX") - DeleteFile(temp_percentages_file + ".dcb") - - return(percentages) -EndMacro - -Macro "ExportBintoCSV"(input_file_base, output_file_base) - - view = OpenTable("Binary Table","FFB",{input_file_base+".bin",}, {{"Shared", "True"}}) - SetView(view) - ExportView(view+"|", "CSV", output_file_base+".csv",,{{"CSV Header", "True"}, {"Force Numeric Type", "double"}}) - CloseView(view) - ok=1 - quit: - return(ok) -EndMacro - - -Macro "ComputeAreaOverlayPercentages" (area_layer_file,overlay_layer_file,area_id_field,overlay_id_field) - //returns percentage array, each element holding {area_id,overlay_id,% of overlay in area} - omap_name = GetMap() - olayer_name = GetLayer() - oview_name = GetView() - - map = RunMacro("OpenDatabase",area_layer_file) - area_layer = GetMapLayers(map,"Area") - area_layer = area_layer[1][1] - RunMacro("OpenDatabaseInMap",overlay_layer_file,map) - overlay_layer = GetMapLayers(map,"Area") - if overlay_layer[1][1] = area_layer then do - overlay_layer = overlay_layer[1][2] - end - else do - overlay_layer = overlay_layer[1][1] - end - - area_ids = GetSetIDs(area_layer + "|") - - percentages = null - temp_dir = GetFileInfo(area_layer_file) - temp_dir = Substring(area_layer_file,1,Len(area_layer_file) - Len(temp_dir[1])) - percentages_file = "tempintersect" - temp_percentages_file = RunMacro("FormPath",{temp_dir,percentages_file}) - EnableProgressBar("Calculating Area Intersections", 1) // Allow only a single progress bar - CreateProgressBar("", "True") - - nlen = area_ids.length - for i = 1 to nlen do - area_id = area_ids[i] - stat = UpdateProgressBar("Area id: " + i2s(area_id), r2i(i/nlen*100)) - if stat = "True" then do - percentages = null - goto quit_loop - end - SetLayer(area_layer) - SelectByQuery("select","Several","SELECT * WHERE id=" + i2s(area_id)) - area_sid = GetRecordValues(area_layer,IDToRecordHandle(area_id),{area_id_field}) - area_sid = area_sid[1][2] - SetLayer(overlay_layer) - n = SelectByVicinity("subtaz","several",area_layer+"|select",0,{{"Inclusion","Intersecting"}}) - if n > 0 then do - ComputeIntersectionPercentages({area_layer+"|select",overlay_layer + "|subtaz"}, temp_percentages_file + ".bin",) - t = OpenTable("int_table", "FFB", {temp_percentages_file + ".bin"},) - tbar = t + "|" - rh = GetFirstRecord(tbar,) - while rh <> null do - vals = GetRecordValues(t,rh,{"Area_1", "Area_2","Percent_2"}) - if vals[1][2] > 0 and vals[2][2] <> 0 and vals[3][2] > 0.0 then do - value = GetRecordValues(overlay_layer,IDToRecordHandle(vals[2][2]),{overlay_id_field}) - percentages = percentages + {{area_sid,value[1][2],vals[3][2]}} - end - rh = GetNextRecord(t+"|",,) - end - CloseView(t) - end - end - - quit_loop: - DestroyProgressBar() - CloseMap(map) - if omap_name <> null then do - SetMap(omap_name) - if olayer_name <> null then do - SetLayer(olayer_name) - end - end - if oview_name <> null then do - SetView(oview_name) - end - DeleteFile(temp_percentages_file + ".bin") - DeleteFile(temp_percentages_file + ".BX") - DeleteFile(temp_percentages_file + ".dcb") - - return(percentages) -EndMacro - - - diff --git a/sandag_abm/src/main/gisdk/commVehDist.rsc b/sandag_abm/src/main/gisdk/commVehDist.rsc deleted file mode 100644 index 5afba39..0000000 --- a/sandag_abm/src/main/gisdk/commVehDist.rsc +++ /dev/null @@ -1,106 +0,0 @@ -/************************************************************** - CommVehTOD.rsc - - TransCAD Macro used to run truck commercial vehicle distribution model. The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. - - A simple gravity model is used to distribute the truck trips. A blended travel time is used as the impedance measure, - specifically the weighted average of the AM travel time (one-third weight) and the midday travel time (two-thirds weight). - - Input: (1) Level-of-service matrices for the AM peak period (6 am to 10 am) and midday period (10 am to 3 pm) - which contain truck-class specific estimates of congested travel time (in minutes) - (2) Trip generation results in ASCII format with the following fields - (a) TAZ: zone number; - (b) PROD: very small truck trip productions; - (c) ATTR: very small truck trip attractions; - (4) A table of friction factors in ASCII format with the following fields (each 12 columns wide): (a) - impedance measure (blended travel time); (b) friction factors for very small trucks; - - Output: (1) A production/attraction trip table matrix of daily class-specific truck trips for very small trucks. - (2) A blended travel time matrix - - See also: (1) CommVehGen.rsc, which applies the generation model. - (2) CommVehTOD.rsc, which applies diurnal factors to the daily trips generated here. - - authors: jef (2012 03 11) dto (2011 09 08); dto (2010 08 31); cp (date unknown) - - -**************************************************************/ -Macro "Commercial Vehicle Distribution" - - shared path, inputDir, outputDir - - /* testing - RunMacro("TCB Init") - scenarioDirectory = "d:\\projects\\SANDAG\\AB_Model\\commercial_vehicles" - */ - - tazCommTripFile = "tazCommVeh.csv" - amMatrixName = "impcvn_AM.mtx" - amTableName = "*STM_AM (Skim)" - mdMatrixName = "impcvn_MD.mtx" - mdTableName = "*STM_MD (Skim)" - - frictionTable = "commVehFF.csv" - pa_tb = outputDir+"\\"+tazCommTripFile - ff_tb = inputDir+"\\"+frictionTable - - //outputs - blendMatrixName = "blendMatrix.mtx" - commVehTripTable = "commVehTrips.mtx" - - //create blended skim - amMatrix = OpenMatrix(outputDir + "\\"+amMatrixName, ) - amMC = CreateMatrixCurrency(amMatrix, amTableName, "Origin", "Destination", ) - mdMatrix = OpenMatrix(outputDir + "\\"+mdMatrixName, ) - mdMC = CreateMatrixCurrency(mdMatrix, mdTableName, "Origin", "Destination", ) - - blendMatrix = CopyMatrix(amMC, {{"File Name", outputDir+"\\"+blendMatrixName}, - {"Label", "AMMDBlend"}, - {"Table", amTableName}, - {"File Based", "Yes"}}) - - blendMC = CreateMatrixCurrency(blendMatrix, amTableName, "Origin", "Destination", ) - - blendMC := 0.3333*amMC + 0.6666*mdMC - - //prevent intrazonal - EvaluateMatrixExpression(blendMC, "99999", null, null,{{"Diagonal","true"}} ) - - - ff_vw = RunMacro("TCB OpenTable",,, {ff_tb}) - ok = (ff_vw <> null) if !ok then goto quit - - pa_vw = RunMacro("TCB OpenTable",,, {pa_tb}) - ok = (pa_vw <> null) if !ok then goto quit - - //Gravity Application - Opts = null - Opts.Input.[PA View Set] = {pa_tb} - Opts.Field.[Prod Fields] = {pa_vw + ".PROD"} - Opts.Field.[Attr Fields] = {pa_vw + ".ATTR"} - Opts.Input.[FF Matrix Currencies] = {{outputDir + "\\" + blendMatrixName, amTableName, "Origin", "Destination"}} - Opts.Input.[Imp Matrix Currencies] = {{outputDir + "\\" + blendMatrixName, amTableName, "Origin", "Destination"}} - Opts.Input.[FF Tables] = {{ff_tb}} - Opts.Input.[KF Matrix Currencies] = {{outputDir + "\\" +blendMatrixName, amTableName, "Origin", "Destination"}} - Opts.Field.[FF Table Fields] = {ff_vw +".FF"} - Opts.Field.[FF Table Times] = {ff_vw +".TIME"} - Opts.Global.[Purpose Names] = {"CommVeh"} - Opts.Global.Iterations = {50} - Opts.Global.Convergence = {0.1} - Opts.Global.[Constraint Type] = {"Double"} - Opts.Global.[Fric Factor Type] = {"Table"} - Opts.Global.[A List] = {1} - Opts.Global.[B List] = {0.3} - Opts.Global.[C List] = {0.005} - Opts.Flag.[Use K Factors] = {0} - Opts.Output.[Output Matrix].Label = "Output Matrix" - Opts.Output.[Output Matrix].[File Name] = outputDir + "\\" +commVehTripTable - ok = RunMacro("TCB Run Procedure", "Gravity", Opts) - - - RunMacro("close all") - quit: - Return(ok) - -EndMacro diff --git a/sandag_abm/src/main/gisdk/commVehDiversion.rsc b/sandag_abm/src/main/gisdk/commVehDiversion.rsc deleted file mode 100644 index ba1c928..0000000 --- a/sandag_abm/src/main/gisdk/commVehDiversion.rsc +++ /dev/null @@ -1,71 +0,0 @@ -Macro "cv toll diversion model" - shared path, inputDir, outputDir -/* RunMacro("TCB Init") - //inputs - path = "D:\\projects\\sandag\\series13\\2012_test" - inputDir = path+"\\input" - outputDir = path+"\\output" - scenarioDirectory = "D:\\projects\\SANDAG\\series13\\2012_test" -*/ - // Toll diversion curve settings - nest_param = 10 - vot = 0.02 //(minutes/cent), currently $0.50 a minute - - periodName = {"EA","AM","MD","PM","EV"} // must be consistent with filename arrays below - //periodName = {"AM"} // must be consistent with filename arrays below - cvTollFactor = 1 // cv toll factor - - // Loop by time period - for period = 1 to periodName.length do - - // Open cv trips - fileNameCV = outputDir + "\\commVehTODTrips.mtx" - m = OpenMatrix(fileNameCV,) - - nontollmtx=outputDir+"\\impcv"+"n_"+periodName[period]+".mtx" // non-toll commercial vehicle skims - tollmtx=outputDir+"\\impcv"+"t_"+periodName[period]+".mtx" // toll commercial vehicle skims - OpenMatrix(nontollmtx,) - OpenMatrix(tollmtx,) - - // Add toll and non-toll matrix - AddMatrixCore(m, periodName[period]+ " Toll") - AddMatrixCore(m, periodName[period]+ " NonToll") - - // Diversion curve (time is in minutes, cost is in cents) - // First scale the toll cost since the cost is scaled for SR125 - tollCost = "[Output Matrix:1].[cvt - ITOLL2_"+ - periodName[period]+"]" - utility = "(([Output Matrix].[*STM_"+periodName[period]+" (Skim)] - [Output Matrix:1].[*STM_"+periodName[period]+" (Skim)]) - " + - String(vot) + " * " + tollCost + " * " + String(cvTollFactor) + " ) / " + String(nest_param) - - expression = "if(" + tollCost + "!=0) then ( 1 / ( 1 + exp(-" + utility + ") ) ) else " + tollCost - - // Calculate toll matrix - Opts = null - Opts.Input.[Matrix Currency] = {fileNameCV, periodName[period]+ " Toll", "Row ID's", "Col ID's"} - Opts.Input.[Formula Currencies] = {{nontollmtx, "*CVCST_"+periodName[period], "Origin", "Destination"}, {tollmtx, "*CVCST_"+periodName[period], "Origin", "Destination"}} - Opts.Global.Method = 11 - Opts.Global.[Cell Range] = 2 - Opts.Global.[Expression Text] = "[" + periodName[period] + " Trips] * " + expression - Opts.Global.[Formula Labels] = {"Output Matrix", "Output Matrix:1"} - Opts.Global.[Force Missing] = "Yes" - ok = RunMacro("TCB Run Operation", "Fill Matrices", Opts) - if !ok then goto quit - - // Calculate non-toll matrix - mc_n = CreateMatrixCurrency(m, periodName[period]+ " NonToll", "Row ID's", "Col ID's",) - mc_t = CreateMatrixCurrency(m, periodName[period]+ " Toll", "Row ID's", "Col ID's",) - mc = CreateMatrixCurrency(m, periodName[period]+ " Trips" , "Row ID's", "Col ID's",) - mc_n := mc - mc_t - - end - - //return 1 if macro completed - run_ok = 1 - Return(run_ok) - - quit: - Return(ok) - -EndMacro - \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/commVehGen.rsc b/sandag_abm/src/main/gisdk/commVehGen.rsc deleted file mode 100644 index ccb8239..0000000 --- a/sandag_abm/src/main/gisdk/commVehGen.rsc +++ /dev/null @@ -1,184 +0,0 @@ -/************************************************************** - CommVehGen.rsc - - TransCAD Macro used to run truck trip generation model. The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. - - Linear regression models generate trip ends, balancing attractions to productions. - - Input: (1) MGRA file in CSV format with the following fields: (a) TOTEMP, total employment (same regardless - of classification system); (b) RETEMPN, retail trade employment per the NAICS classification system; - (c) FPSEMPN, financial and professional services employment per the NAICS classification system; (d) - HEREMPN, health, educational, and recreational employment per the NAICS classification system; (e) - OTHEMPN, other employment per the NAICS classification system; (f) AGREMPN, agricultural employment - per the NAICS classificatin system; (g) MWTEMPN, manufacturing, warehousing, and transportation - emp;loyment per the NAICS classification system; and, (h) TOTHH, total households. - - Output: (1) An ASCII file containing the following fields: (a) zone number; (b) very small truck trip productions; - (c) very small truck trip attractions - - See also: (1) TruckTripDistribution.job, which applies the distribution model. - (2) TruckTimeOfDay.job, which applies diurnal factors to the daily trips generated here. - (3) TruckTollChoice.job, which applies a toll/no toll choice model for trucks. - - version: 0.1 - authors: dto (2010 08 31); jef (2012 03 07) - - 5-2013 wsu fixed indexing bug - 7-2013 jef reduced truck rate for military employment to 0.3 - 8-2014 wsu reduced truck rate for military employment to 0.15 - -**************************************************************/ -Macro "Commercial Vehicle Generation" - - shared path, inputDir, outputDir ,scenarioYear - - mgraCommTripFile = "mgraCommVeh.csv" - tazCommTripFile = "tazCommVeh.csv" - - writeMgraData = true - calibrationFactor = 1.4 - - // read in the mgra data in CSV format - mgraView = OpenTable("MGRA View", "CSV", {inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) - - mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_total = GetDataVector(mgraView+"|", "emp_total", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - - RETEMPN = emp_retail + emp_personal_svcs_retail - FPSEMPN = emp_prof_bus_svcs - HEREMPN = emp_health + emp_pvt_ed_k12 + emp_pvt_ed_post_k12_oth + emp_amusement - AGREMPN = emp_ag - MWTEMPN = emp_const_non_bldg_prod + emp_const_bldg_prod + emp_mfg_prod + emp_trans - MILITARY = emp_fed_mil - TOTEMP = emp_total - OTHEMPN = TOTEMP - (RETEMPN + FPSEMPN + HEREMPN + AGREMPN + MWTEMPN + MILITARY) - TOTHH = hh - - verySmallP = calibrationFactor * (0.95409 * RETEMPN + 0.54333 * FPSEMPN + 0.50769 * HEREMPN + - 0.63558 * OTHEMPN + 1.10181 * AGREMPN + 0.81576 * MWTEMPN + - 0.15000 * MILITARY + - 0.1 * TOTHH) - - - // Wu added this section for military CTM trips adjustment to match military gate counts - properties = "\\conf\\sandag_abm.properties" - militaryCtmAdjustment = RunMacro("read properties",properties,"RunModel.militaryCtmAdjustment", "S") - if militaryCtmAdjustment = "true" then do - mgraView_m = OpenTable("Military MGRA View", "CSV", {inputDir+"\\cvm_military_adjustment.csv"}, {{"Shared", "True"}}) - //base_id = GetDataVector(mgraView_m+"|", "ID", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - //base_name = GetDataVector(mgraView_m+"|", "base", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - //taz_m = GetDataVector(mgraView_m+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - //mgra_m = GetDataVector(mgraView_m+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - base_id = GetDataVector(mgraView_m+"|", "ID",) - base_name = GetDataVector(mgraView_m+"|", "base",) - taz_m = GetDataVector(mgraView_m+"|", "TAZ",) - mgra_m = GetDataVector(mgraView_m+"|", "mgra",) - scale = GetDataVector(mgraView_m+"|", "scale", {{"Type", {{"scale", "Float"}}}} ) - - //scale verySmallP and verySmallA: why is his called verySmallP and verySmallA??? - for i = 1 to mgra.length do - for j = 1 to mgra_m.length do - if mgra[i] = mgra_m[j] then do - verySmallP[i]=verySmallP[i]*scale[j] - end - end - end - end - - verySmallA = verySmallP - - if writeMgraData = true then do - //create a table with the mgra trips - truckTripsMgra = CreateTable("truckTripsMgra",outputDir+"\\"+mgraCommTripFile, "CSV", { - {"MGRA", "Integer", 8, null, }, - {"PROD", "Real", 12, 4, }, - {"ATTR", "Real", 12, 4, } - }) - end - - //create a table with the taz trips - truckTripsTaz= CreateTable("truckTripsTaz",outputDir+"\\"+tazCommTripFile, "CSV", { - {"TAZ", "Integer", 8, null, }, - {"PROD", "Real", 12, 4, }, - {"ATTR", "Real", 12, 4, } - }) - - //now aggregate by TAZ - maxTaz = 0 - for i=1 to taz.length do - if taz[i] > maxTaz then maxTaz = taz[i] - end - - //arrays for holding productions and attractions by TAZ - dim tazProd[maxTaz] - dim tazAttr[maxTaz] - - //initialize arrays to 0 - for i = 1 to tazProd.length do - tazProd[i]=0 - tazAttr[i]=0 - end - - //aggregate mgra to taz arrays - for i = 1 to mgra.length do - - tazNumber = taz[i] - tazProd[tazNumber] = tazProd[tazNumber] + verySmallP[i] - tazAttr[tazNumber] = tazAttr[tazNumber] + verySmallA[i] - - if writeMgraData = true then do - AddRecord("truckTripsMgra", { - {"MGRA", mgra[i]}, - {"PROD", verySmallP[i]}, - {"ATTR", verySmallA[i]} - }) - end - end - - if writeMgraData = true then CloseView(truckTripsMgra) - - //add taz data to table - for i = 1 to maxTaz do - - AddRecord("truckTripsTaz", { - {"TAZ", i}, - {"PROD", tazProd[i]}, - {"ATTR", tazAttr[i]} - }) - end - - RunMacro("close all") - Return(1) -EndMacro diff --git a/sandag_abm/src/main/gisdk/commVehTOD.rsc b/sandag_abm/src/main/gisdk/commVehTOD.rsc deleted file mode 100644 index 9b1a459..0000000 --- a/sandag_abm/src/main/gisdk/commVehTOD.rsc +++ /dev/null @@ -1,122 +0,0 @@ -/************************************************************** - CommVehTOD.rsc - - TransCAD Macro used to run truck commercial vehicle time-of-day factoring. The very small truck generation model is based on the Phoenix - four-tire truck model documented in the TMIP Quick Response Freight Manual. - - The diurnal factors are taken from the BAYCAST-90 model with adjustments made during calibration to the very - small truck values to better match counts. - - Input: A production/attraction format trip table matrix of daily very small truck trips. - - Output: Five, time-of-day-specific trip table matrices containing very small trucks. - - See also: (1) CommVehGen.rsc, which applies the generation model. - (2) CommVehDist.rsc, which applies the distribution model. - - authors: jef (2012 03 11) dto (2011 09 08); dto (2010 08 31); cp (date unknown) - - -**************************************************************/ -Macro "Commercial Vehicle Time Of Day" (scenarioDirectory) - shared path, inputDir, outputDir - -/* - RunMacro("TCB Init") - - //inputs - scenarioDirectory = "d:\\projects\\SANDAG\\AB_Model\\commercial_vehicles" -*/ - - commVehTripTable = "commVehTrips.mtx" - - //outputs - commVehTODTable = "commVehTODTrips.mtx" - - //open input table - dailyMatrix = OpenMatrix(outputDir + "\\"+commVehTripTable, ) - dailyMC = CreateMatrixCurrency(dailyMatrix, "CommVeh", ,, ) - - //create transposed daily trip table - tmat = TransposeMatrix(dailyMatrix, {{"File Name", outputDir + "\\"+commVehTripTable+"t"}, - {"Label", "commVehT"}, - {"Type", "Double"}, - {"Sparse", "No"}, - {"Column Major", "No"}, - {"File Based", "No"}}) - - transMC = CreateMatrixCurrency(tmat, "commVeh", ,, ) - - //create output matrix - todMatrix = CopyMatrix(dailyMC, {{"File Name", outputDir+"\\"+commVehTODTable}, - {"Label", "CommVehTOD"}, - {"File Based", "Yes"}}) - - Opts = null - Opts.Input.[Input Matrix] =outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "OD Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "EA Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "AM Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "MD Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "PM Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[Input Matrix] = outputDir+"\\"+commVehTODTable - Opts.Input.[New Core] = "EV Trips" - ok = RunMacro("TCB Run Operation", "Add Matrix Core", Opts) - if !ok then goto quit - - odMC = CreateMatrixCurrency(todMatrix, "OD Trips", ,, ) - eaMC = CreateMatrixCurrency(todMatrix, "EA Trips", ,, ) - amMC = CreateMatrixCurrency(todMatrix, "AM Trips", ,, ) - mdMC = CreateMatrixCurrency(todMatrix, "MD Trips", ,, ) - pmMC = CreateMatrixCurrency(todMatrix, "PM Trips", ,, ) - evMC = CreateMatrixCurrency(todMatrix, "EV Trips", ,, ) - - odMC := (0.5 * dailyMC) + (0.5 * transMC) - - // - early AM - eaMC := 0.0235 * odMC - - - // - AM peak - amMC := 0.1000 * odMC - - - // - midday - mdMC := 0.5080 * odMC - - // - PM peak - pmMC := 0.1980 * odMC - - - // - evening - evMC := 0.1705 * odMC - - RunMacro("close all") - quit: - Return(ok) - -EndMacro diff --git a/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc b/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc deleted file mode 100644 index 9c74141..0000000 --- a/sandag_abm/src/main/gisdk/create_LUZ_Skims.rsc +++ /dev/null @@ -1,254 +0,0 @@ -/* - Create landuse skims(LUZ skims): - -About: - Creates LUZ skims from the following TAZ skims including length, time, and cost: - impdat_AM.mtx (Length (Skim), *STM_AM (Skim), dat_AM – itoll_AM) - impdat_MD.mtx (Length (Skim), *STM_MD (Skim), dat_MD – itoll_MD) - impmhdt_AM.mtx (Length (Skim), *STM_AM (Skim), mhdt – ITOLL2_AM) - impmhdt_MD.mtx (Length (Skim), *STM_MD (Skim), mhdt – ITOLL2_MD) - -Inputs: - 1) luzToTazSeries13.xls (Luz to TAZ reference) - 2) ExternalZones.xls (Luz internal to external reference) - 3) impdat_AM.mtx - 4) impdat_MD.mtx - 5) impmhdt_AM.mtx - 6) impmhdt_MD.mtx - -Outputs: - 1) impdat_AM.mtx (csv) - 2) impdat_MD.mtx (csv) - 3) impmhdt_AM.mtx (csv) - 4) impmhdt_MD.mtx (csv) - - -*/ - -Macro "Create LUZ Skims" - - shared path, inputDir, outputDir - - // Input Files - ext_luz_excel = inputDir+"\\ExternalZones.xls" - luz_taz_excel = inputDir+"\\luzToTazSeries13.xls" - - // Temp files - ext_luz_file = outputDir+"\\ExternalZones.bin" - luz_taz_file = outputDir+"\\luzToTazSeries13.bin" - tempfile = outputDir+"\\temp_luz.bin" - luzskims_bin = outputDir+"\\temp_luz_export.bin" - - // Convert excel to bin file - ExportExcel(ext_luz_excel, "FFB", ext_luz_file, ) - ExportExcel(luz_taz_excel, "FFB", luz_taz_file, ) - - // Open tables - luztaz_view = Opentable("luztaz", "FFB", {luz_taz_file}) - ext_luz_view = Opentable("luzIE", "FFB", {ext_luz_file}) - - /* ------------------------------------------------------------------------------------------------ - // Step 0: Prepares input and output files - --------------------------------------------------------------------------------------------------*/ - // Create list of LUZ I+E zones - LUZ_I = V2A(GetDataVector(luztaz_view+"|","luz_id",)) - LUZ_E = V2A(GetDataVector(ext_luz_view+"|","External LUZ",{{"Sort Order", {{"External LUZ", "Ascending"}}}})) - LUZ_IE = LUZ_I + LUZ_E - - // Get the maximum internal zone - max_LUZ_I = ArrayMax(LUZ_I) - min_LUZ_E = ArrayMin(LUZ_E) - max_LUZ_E = ArrayMax(LUZ_E) - - // Write the list to a file (temp luz zonal file) - luz_vw = CreateTable("luz", tempfile, "FFB",{ - {"luz_id", "Integer", 8, null, "Yes"}}) - SetView(view) - for r = 1 to LUZ_IE.length do - rh = AddRecord(luz_vw, { - {"luz_id", LUZ_IE[r]} - }) - end - - - period = {"AM","MD"} - vehicle = {"dat","mhdt"} - - for v =1 to vehicle.length do - for p= 1 to period.length do - - /* ------------------------------------------------------------------------------------------------ - //Step 1: Create matrix with LUZ internal and external Zones - --------------------------------------------------------------------------------------------------*/ - // Input taz skim - tazskims = outputDir+"\\imp"+vehicle[v]+"_"+period[p]+".mtx" - - // Output luz skim (mtx and csv files) - luzskims = outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".mtx" - luzskims_csv = outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".csv" - - // List of cores (same core names for inputs and outputs) - m = OpenMatrix(tazskims, ) - coreNames = GetMatrixCoreNames(m) - coreNames = Subarray(coreNames,2,3) // Only the distance, time, & toll skims - - // Open the temp luz zonal file - luz_view = Opentable("luz", "FFB", {tempfile}) - - // Create output matrix file with both LUZ internal & external zones - luz_mat =CreateMatrix({luz_view+"|", "luz_id", "All"}, - {luz_view+"|", "luz_id", "All"}, - {{"File Name", luzskims}, {"Type", "Float"}, {"Tables",coreNames}}) - - // Create LUZ II, IE/EI and EE indices - SetView(luz_view) - set_i = SelectByQuery("Internal", "Several", "Select * where luz_id <= "+ String(max_LUZ_I),) - Internal = CreateMatrixIndex("Internal", luz_mat, "Both", luz_view +"|Internal", "luz_id", "luz_id" ) - - set_e = SelectByQuery("External", "Several", "Select * where (luz_id >= "+ String(min_LUZ_E) +" & luz_id <= "+ String(max_LUZ_E)+")",) - External = CreateMatrixIndex("External", luz_mat, "Both", luz_view +"|External", "luz_id", "luz_id" ) - CloseView(luz_view) - - // Create luz currencies for II, EI and IE (each array has 3 currencies; one of each core) - mc_luz_II = CreateMatrixCurrencies(luz_mat,"Internal","Internal",) - mc_luz_EI = CreateMatrixCurrencies(luz_mat,"External","Internal",) - mc_luz_IE = CreateMatrixCurrencies(luz_mat,"Internal","External",) - mc_luz_EE = CreateMatrixCurrencies(luz_mat,"External","External",) - - /* ------------------------------------------------------------------------------------------------ - // Step 2: Create LUZ internal skims from TAZ skims - --------------------------------------------------------------------------------------------------*/ - // Create aggregate tables for each selected cores - for c = 1 to coreNames.length do - - // Create currency in the input file - mc = CreateMatrixCurrency(m, coreNames[c], , , ) - row_names = {"luztaz.taz", "luztaz.luz_id"} - col_names = {"luztaz.taz", "luztaz.luz_id"} - - // Create LUZ internal skims - tempLuzMtx = outputDir+"\\temp"+vehicle[v]+period[p]+"_"+String(c)+".mtx" - AggregateMatrix(mc, row_names, col_names, - {{"File Name", tempLuzMtx}, - {"Label", "LUZ"+coreNames[c]}, - {"File Based", "Yes"}}) - - // Add the aggregate table to the internal zones in the new core - mat = OpenMatrix(tempLuzMtx, ) - mc_temp = CreateMatrixCurrency(mat, coreNames[c], , , ) - mc_luz_II.(coreNames[c]):= mc_temp - - // Get the internal zones to the corresponding - intZones = GetDataVector(ext_luz_view+"|","Internal Cordon LUZ",) - extZones = GetDataVector(ext_luz_view+"|","External LUZ",) - distance = GetDataVector(ext_luz_view+"|","Miles to be Added to Cordon Point",) - time = GetDataVector(ext_luz_view+"|","Minutes to be Added to Cordon Point",) - - // Add the EI and IE mat values based on cordon data - for e = 1 to extZones.length do - // Get mat values for the corresponding internal zones - vec_EI = GetMatrixVector(mc_temp, {{"Row",intZones[e]}}) - vec_IE = GetMatrixVector(mc_temp, {{"Column",intZones[e]}}) - - // Add cordon time, distance (depending on the core) - if (c = 1) then do // length - vec_EI = vec_EI + distance[e] - vec_IE = vec_IE + distance[e] - end - if (c = 2) then do // time - vec_EI = vec_EI + time[e] - vec_IE = vec_IE + time[e] - end - - // Set vectors to the matrix - SetMatrixVector(mc_luz_EI.(coreNames[c]), vec_EI, {{"Row",extZones[e]}}) - SetMatrixVector(mc_luz_IE.(coreNames[c]), vec_IE, {{"Column",extZones[e]}}) - end // ext zones - - // EE values are filled based on II, and IE values - for i = 1 to extZones.length do - for j = 1 to extZones.length do - // Get II value if Intrazonal - if i=j then do - II_val = GetMatrixValue(mc_luz_II.(coreNames[c]), String(intZones[i]),String(intZones[j])) - SetMatrixValue(mc_luz_EE.(coreNames[c]),String(extZones[i]),String(extZones[j]),II_val) - end - else do - IE_val = GetMatrixValue(mc_luz_IE.(coreNames[c]), String(intZones[i]),String(extZones[j])) - if (c=1) then IE_val = IE_val + distance[i] - if (c=2) then IE_val = IE_val + time[i] - SetMatrixValue(mc_luz_EE.(coreNames[c]),String(extZones[i]),String(extZones[j]),IE_val) - end // if - end // j loop - end // i loop - end // cores - - /* ------------------------------------------------------------------------------------------------ - // Step 3: Export to CSV file - --------------------------------------------------------------------------------------------------*/ - // Export matrix values to a temp bin file - SetMatrixIndex(luz_mat, "All", "All") - CreateTableFromMatrix(luz_mat, luzskims_bin, "FFB", {{"Complete", "Yes"}}) - - // Add header and then export to csv file - export_vw = OpenTable("luztaz", "FFB", {luzskims_bin}) - strct = GetTableStructure(export_vw) - for s = 1 to strct.length do - strct[s] = strct[s] + {strct[s][1]} - end - - // Rename to first and second columns to Origin and Destination - strct[1][1] = "origin LUZ" - strct[2][1] = "destination LUZ" - ModifyTable(export_vw, strct) - - // Export to CSV file - ExportView(export_vw+"|", "CSV", luzskims_csv, null,{{"CSV Header", "True"}}) - CloseView(export_vw) - - end // time period - end // vehicle - - - /* ------------------------------------------------------------------------------------------------ - // Step 4: Close all views and delete temp files - --------------------------------------------------------------------------------------------------*/ - vws = GetViewNames() - if vws<> null then do - for w = 1 to vws.length do - CloseView(vws[w]) - end - end - - // Close matrices - mtxs = GetMatrices() - if mtxs <> null then do - handles = mtxs[1] - for m = 1 to handles.length do - handles[m] = null - end - end - - // Close rest - RunMacro("G30 File Close All") - - // Delete temp matrices - for v =1 to vehicle.length do - for p= 1 to period.length do - for c = 1 to coreNames.length do - DeleteFile(outputDir+"\\temp"+vehicle[v]+period[p]+"_"+String(c)+".mtx") - end - // Also delete tcad headers for the CSV file (as the csv files have headers) - DeleteFile(outputDir+"\\luz_imp"+vehicle[v]+"_"+period[p]+".DCC") - end - end - - // Delete temp luz files - delFiles = {"temp_luz.bin","temp_luz.BX","temp_luz.DCB","luzToTazSeries13.bin","luzToTazSeries13.DCB", - "ExternalZones.bin","ExternalZones.DCB","temp_luz_export.bin","temp_luz_export.DCB"} - for d =1 to delFiles.length do - info = GetFileInfo(outputDir+"\\"+delFiles[d]) - if info[1] <> null then DeleteFile(outputDir+"\\"+delFiles[d]) - end - -EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/createhwynet.rsc b/sandag_abm/src/main/gisdk/createhwynet.rsc deleted file mode 100644 index b6c5cf5..0000000 --- a/sandag_abm/src/main/gisdk/createhwynet.rsc +++ /dev/null @@ -1,2509 +0,0 @@ -//******************************************************************** -//procedure to import e00 file to geo dbd file -//create highway network -//written on 4/19/01 -//macro "import highway layer", macro"fill oneway streets", -//macro "createhwynet1" -// -//input files: hwycov.e00 - hwy line layer ESRI exchange file -//output files: hwy.dbd - hwy line geographic file -// hwycad.log- a log file -// hwycad.err - error file with error info -//Oct 08, 2010: Added Lines 164-186, Create a copy of Toll fields -//Oct 08, 2010: Added Lines 284-287 Build Highway Network with ITOLL fields -//April 22, 2014: Wu checked all SR125 related changes are included -//Feb 02, 2016: Added reliability fields -//******************************************************************** - -macro "run create hwy" - shared path,inputDir,outputDir,mxzone - -/* exported highway layer is copied manually to the output folder (I15 SB toll entry/exit links are modified by RSG) - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","import highway layer"}) - ok=RunMacro("import highway layer") - if !ok then goto quit -*/ - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","copy highway database"}) - ok=RunMacro("copy database") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","fill oneway streets"}) - ok=RunMacro("fill oneway streets") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add TOD attributes"}) - ok=RunMacro("add TOD attributes") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","calculate distance to/from major interchange"}) - ok=RunMacro("DistanceToInterchange") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add reliability fields"}) - ok=RunMacro("add reliability fields") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","add preload attributes"}) - ok=RunMacro("add preload attributes") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","Code VDF fields"}) - ok=RunMacro("Code VDF fields") - if !ok then goto quit - - RunMacro("HwycadLog",{"createhwynet.rsc: run create hwy","create hwynet"}) - ok=RunMacro("create hwynet") - if !ok then goto quit - - quit: - return(ok) -EndMacro - -/********************************************************************************************************** - import e00 file - if numofzone=tdz then e00file=hwycovtdz.e00, hwytdz.dbd - else e00file=hwy.e00, hwy.dbd - - Inputs: - input\turns.csv - input\turns.DCC - input\hwycov.e00 - - Outputs: - output\turns.dbf - output\hwytmp.dbd - output\hwy.dbd - -**********************************************************************************************************/ -macro "import highway layer" - shared path, inputDir,outputDir - ok=0 - - RunMacro("close all") - - di=GetDirectoryInfo(path + "\\tchc1.err","file") - if di.length>0 then do - ok=0 - RunMacro("TCB Error","chech tchc1.err file!") - goto exit - end - - di=GetDirectoryInfo(path + "\\tchc.err","file") - if di.length>0 then do - ok=0 - RunMacro("TCB Error","chech tchc.err file!") - goto exit - end - - di=GetDirectoryInfo(inputDir + "\\turns.csv","file") - if di.length=0 then do - ok=0 - RunMacro("TCB Error","turns.csv does not exist!") - goto exit - end - - // assume that the dictionary file exists in the input directory for the model run - //ok=RunMacro("SDcopyfile",{path_study+"\\data\\turns.DCC",path+"\\turns.DCC"}) - //if !ok then goto exit - - //export turns.csv to turns.dbf - vw = OpenTable("turns", "CSV", {inputDir+"\\turns.csv",}) - ExportView("turns|", "dbase", outputDir+"\\turns.dbf",,) - - // writeline(fpr,mytime+", exporting turns.csv to turns.dbf") - - // import e00 file - e00file="hwycov.e00" - - //check e00 file exists - di = GetDirectoryInfo(inputDir +"\\"+e00file, "File") - if di.length = 0 then do - ok=0 - RunMacro("TCB Error",e00file+" does not exist!") - goto exit - end - - ImportE00(inputDir +"\\"+e00file, outputDir + "\\hwytmp.dbd","line",outputDir + "\\hwytmp.bin",{ - {"Label","street line file"}, - {"Layer Name","hwyline"}, - {"optimize","True"}, - {"Median Split", "True"}, - {"Node Layer Name", "hwynode"}, - {"Node Table", outputDir + "\\hwytmp_.bin"}, - {"Projection","NAD83:406",{"_cdist=1000","_limit=1000","units=us-ft"}}, - }) - - //writeline(fpr,mytime+", importing e00 file") - - //export geo file by specify the line id field and the node id field - db_file=outputDir + "\\hwytmp.dbd" - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto exit - - allflds=Getfields(link_lyr,"All") - fullflds=allflds[2] - allnodeflds = GetFields(node_lyr, "All") - - // need to specify full field specifications - lineidfield = link_lyr+".hwycov-id" - nodeidfield = "hwynode.hnode"//for centroids purposes - - opts = {{"Layer Name", "hwyline"}, - {"File Type", "FFB"}, - {"ID Field", lineidfield}, - {"Field Spec", fullflds}, - {"Indexed Fields", {fullflds[1]}}, - {"Label", "street line file"}, - {"Node layer name","hwynode"}, - {"Node ID Field", nodeidfield}, - {"Node Field Spec", allnodeflds[2]}} - - if node_idx > 1 then - opts = opts + {{"Node ID Field", node_aflds[2][node_idx - 1]}} - hwy_db=outputDir + "\\hwy.dbd" - - exportgeography(link_lyr,hwy_db,opts) - - // writeline(fpr,mytime+", exporting e00 file") - - RunMacro("close all") //before delete db_file, close it - deleteDatabase(db_file) - - ok=1 - exit: - //if fpr<>null then closefile(fpr) - return(ok) -endMacro - -/* -copies database (hwy.dbd) and turns file (TURNS.DBF) - -this is required after edits made to the transcad highway database for I15 SB managed lane links - -*/ - -macro "copy database" - shared path, inputDir, outputDir - - hwy_db_in = inputDir + "\\hwy.dbd" - hwy_db_out = outputDir + "\\hwy.dbd" - - /// copye highway database - CopyDatabase(hwy_db_in, hwy_db_out) - - // copy turns file - CopyFile(inputDir+"\\TURNS.DBF", outputDir+"\\TURNS.DBF") - - ok=1 - return(ok) - -endMacro - -/********************************************************************************************************** - fill oneway street with dir field, and calculate toll fields and change AOC and add reliability factor - - Inputs - output\hwy.dbd - - Outputs: - output\hwy.dbd (modified) - - Adds fields to link layer (by period: _EA, _AM, _MD, _PM, _EV) - ITOLL - Toll + 10000 *[0,1] if SR125 toll lane - ITOLL2 - Toll - ITOLL3 - Toll + AOC - ITOLL4 - Toll * 1.03 + AOC - ITOLL5 - Toll * 2.33 + AOC - - Note: Link operation type (IHOV) where: - 1 = General purpose - 2 = 2+ HOV (Managed lanes if lanes > 1) - 3 = 3+ HOV (Managed lanes if lanes > 1) - 4 = Toll lanes - - - -**********************************************************************************************************/ -macro "fill oneway streets" - shared path, inputDir, outputDir - ok=0 - - RunMacro("close all") - - properties = "\\conf\\sandag_abm.properties" - aoc_f = RunMacro("read properties",properties,"aoc.fuel", "S") - aoc_m = RunMacro("read properties",properties,"aoc.maintenance", "S") - aoc=S2R(aoc_f)+S2R(aoc_m) - - db_file=outputDir+"\\hwy.dbd" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr = db_file + "|" + link_lyr - -// writeline(fpr,mytime+", fill one way streets") -// closefile(fpr) - -/* - //oneway streets, dir = 1 - Opts = null - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select * where iway = 1"} - Opts.Global.Fields = {"Dir"} - Opts.Global.Method = "Value" - Opts.Global.Parameter = {1} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit -*/ - //CHANGE SR125 TOLL SPEED TO 70MPH (ISPD=70) DELETE THIS SECTION AFTER TESTING - Opts = null - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ISPD"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = "if ihov=4 and IFC=1 then 70 else ISPD" - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - - // Create RELIABILITY OF FACILITY (TOLL) field - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"relifac", "Real", 10, 2, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - //change reliability field for SR125 to 0.65, and all other facilities are 0 - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"relifac"} - Opts.Global.Method = "Formula" -// Opts.Global.Parameter = "if ihov=4 & ifc=1 then 0.65 else 1" - //since we now have reliability fields, setting all reliability factors to 1 - Opts.Global.Parameter = "1" - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - //change AOC to appropriate value for year and cents per mile in COST field and add reliability factor to COST calc. - Opts = null - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"COST"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = "Length * "+R2S(aoc)+" * relifac" - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // Create copy of ITOLL fields to preserved original settings - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // changed field types to real (for I15 tolls) - by nagendra.dhakar@rsginc.com - strct = strct + {{"ITOLL2_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL2_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL2_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL2_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL2_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - tollfld={{"ITOLL2_EA"},{"ITOLL2_AM"},{"ITOLL2_MD"},{"ITOLL2_PM"},{"ITOLL2_EV"}} - tollfld_flg={{"ITOLLO"},{"ITOLLA"},{"ITOLLO"},{"ITOLLP"},{"ITOLLO"}} //note - change this once e00 file contains fields for each of 5 periods - - // set SR125 tolls - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = tollfld_flg[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // clear I15 tolls from previous step - // set other link tolls to 0 - creates a problem in skimming if left to null - // added by nagendra.dhakar@rsginc.com - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select *where ihov=2"} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Value" - Opts.Global.Parameter = {0} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - - // set I15 tolls - added by nagendra.dhakar@rsginc.com - RunMacro("set I15 tolls", link_lyr, tollfld) - - // Create ITOLL3 fields with ITOLL2A and COST - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"ITOLL3_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL3_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL3_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL3_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL3_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - tollfld={{"ITOLL3_EA"},{"ITOLL3_AM"},{"ITOLL3_MD"},{"ITOLL3_PM"},{"ITOLL3_EV"}} - tollfld_flg={{"ITOLL2_EA+COST"},{"ITOLL2_AM+COST"},{"ITOLL2_MD+COST"},{"ITOLL2_PM+COST"},{"ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = tollfld_flg[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // Create ITOLL4 fields with 1.03*(ITOLL2) and COST - // ITOLL4 = is applied to LHD and MHD only - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"ITOLL4_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL4_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL4_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL4_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL4_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - tollfld={{"ITOLL4_EA"},{"ITOLL4_AM"},{"ITOLL4_MD"},{"ITOLL4_PM"},{"ITOLL4_EV"}} - tollfld_flg={{"1.03*ITOLL2_EA+COST"},{"1.03*ITOLL2_AM+COST"},{"1.03*ITOLL2_MD+COST"},{"1.03*ITOLL2_PM+COST"},{"1.03*ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = tollfld_flg[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // Create ITOLL5 fields with 2.33*(ITOLL2) and COST - // ITOLL5 = is applied to HHD only - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"ITOLL5_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL5_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL5_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL5_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL5_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - tollfld={{"ITOLL5_EA"},{"ITOLL5_AM"},{"ITOLL5_MD"},{"ITOLL5_PM"},{"ITOLL5_EV"}} - tollfld_flg={{"2.33*ITOLL2_EA+COST"},{"2.33*ITOLL2_AM+COST"},{"2.33*ITOLL2_MD+COST"},{"2.33*ITOLL2_PM+COST"},{"2.33*ITOLL2_EV+COST"}} //note - change this once e00 file contains fields for each of 5 periods - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = tollfld_flg[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // Create ITOLL fields - vw = GetView() - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - strct = strct + {{"ITOLL_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ITOLL_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - //adding $100 to toll fields to flag toll values from manage lane toll values in skim matrix - tollfld={{"ITOLL_EA"},{"ITOLL_AM"},{"ITOLL_MD"},{"ITOLL_PM"},{"ITOLL_EV"}} - tollfld_flg={{"if ihov=4 then ITOLL2_EA+10000 else ITOLL2_EA"}, - {"if ihov=4 then ITOLL2_AM+10000 else ITOLL2_AM"}, - {"if ihov=4 then ITOLL2_MD+10000 else ITOLL2_MD"}, - {"if ihov=4 then ITOLL2_PM+10000 else ITOLL2_PM"}, - {"if ihov=4 then ITOLL2_EV+10000 else ITOLL2_EV"}} //note - change this once e00 file contains fields for each of 5 periods - - // modified by nagendra.dhakar@rsginc.com to calculate every toll field from itoll2, which are set to the tolls fields in tcoved - - for i=1 to tollfld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tollfld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = tollfld_flg[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - RunMacro("close all") - - ok=1 - quit: - return(ok) -EndMacro - -/********************************************************************************************************* -add i-15 tolls by direction and period - - link ids and corresponding tolls are inputs - toll values are coded by link ids - tolls are determined by gate-to-gate toll optimization, solved using excel solver - tolls from two methods are used - -traversed links (NB PM and SB AM) - -entry and exit links (remaining) - -by: nagendra.dhakar@rsginc.com -**********************************************************************************************************/ - -Macro "set I15 tolls" (lyr, toll_fields) - shared path, inputDir, outputDir - - direction = {"NB","SB"} - periods={"EA","AM","MD","PM","EV"} - - toll_links = {} - tolls = {} - - // NB toll links and corresponding tolls - toll_links.NB = {} - - toll_links.NB.traverse = {29716,460,526,23044,459,463,512,464,469,470,510,29368,9808} - toll_links.NB.entryexit = {31143,29472,52505,52507,52508,475,34231,52511,52512,34229,34228,38793,29765,29766,52513,29764,26766} - - tolls.NB = {} - - tolls.NB.traverse = {} - tolls.NB.entryexit = {} - - // tolls are in cents - tolls.NB.entryexit.EA = {35.00,35.00,35.00,35.00,35.00,15.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00} - tolls.NB.entryexit.AM = {45.05,42.43,31.54,30.00,30.00,20.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,17.41,32.59,32.59,32.59} - tolls.NB.entryexit.MD = {69.91,73.91,70.42,66.12,51.61,0.00,26.88,25.00,25.00,25.00,12.07,37.93,47.51,3.35,46.65,60.66,65.74} - tolls.NB.traverse.PM = {21.83,31.11,50.00,55.34,113.23,50.00,50.00,50.00,50.00,50.00,50.00,50.00,0.00} - tolls.NB.entryexit.EV = {41.73,36.26,32.01,30.00,30.00,20.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,17.77,32.23,32.23,32.23} - - // SB toll links and corresponding tolls - toll_links.SB = {} -/* - // old network - toll_links.SB.traverse = {12193,25749,29442,23128,515,31204,520,22275,524,525,553,528,29415} - toll_links.SB.entryexit = {52514,38796,29768,38794,29763,52510,52509,52506,52510,34227,34233,29407,26398,29767,34226,34232,29471,52515} -*/ - // new network - toll_links.SB.traverse = {12193,25749,52567,23128,515,31204,52569,52550,524,525,52555,52559,52561,52565} - toll_links.SB.entryexit = {52568,52570,29768,38794,29763,52560,52562,52566,52556,34227,34233,29407,26398,52571,52572,29767,52575,52576,52574,34226,34232,29471,52573} - - tolls.SB = {} - - tolls.SB.traverse = {} - tolls.SB.entryexit = {} -/* - // old network - tolls.SB.entryexit.EA = {26.69,25.54,26.96,25.54,39.23,25.54,24.46,25.54,25.54,25.54,24.46,24.46,36.30,23.31,24.46,24.46,28.04,0.00} - tolls.SB.traverse.AM = {0.00,50.00,50.00,89.82,50.00,50.00,63.74,0.00,0.11,76.40,38.80,63.58,83.28} - tolls.SB.entryexit.MD = {26.39,25.00,25.00,25.00,35.00,25.47,22.74,27.26,25.47,25.00,24.53,25.00,35.61,23.61,25.00,25.00,32.65,0.00} - tolls.SB.entryexit.PM = {25.00,25.00,25.00,25.00,35.00,25.00,24.34,25.66,25.00,25.00,25.00,25.00,35.00,25.00,25.00,25.00,26.69,0.00} - tolls.SB.entryexit.EV = {25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,25.00,35.00,25.00,25.00,25.00,25.00,0.00} -*/ - // new network - tolls.SB.traverse.AM = {0.00,59.74,50.00,80.42,50.00,50.00,69.24,0.00,3.18,50.00,50.17,19.06,50.00,84.26} - tolls.SB.entryexit.EA = {25.00,25.00,25.00,25.00,35.00,7.29,7.29,7.29,17.30,25.00,17.30,17.30,35.00,15.00,25.00,25.00,32.70,42.71,32.70,25.00,25.00,42.71,25.00} - tolls.SB.entryexit.MD = {32.80,25.00,25.00,25.00,32.80,11.43,11.43,10.65,18.85,25.00,18.85,18.85,32.80,17.20,25.00,17.20,31.15,38.57,31.15,25.00,25.00,39.35,25.00} - tolls.SB.entryexit.PM = {27.78,25.00,25.00,25.00,35.00,12.67,12.67,12.67,19.36,25.00,19.36,19.36,35.00,15.00,25.00,22.22,30.64,37.33,30.64,25.00,25.00,37.33,25.00} - tolls.SB.entryexit.EV = {29.12,25.00,25.00,25.00,35.00,13.14,13.14,13.14,21.56,25.00,21.56,21.56,35.00,15.00,25.00,20.88,28.44,36.86,28.44,25.00,25.00,36.86,25.00} - - for dir=1 to 2 do - for per=1 to periods.length do - // locate record - - if (direction[dir]="NB" and periods[per] = "PM") or (direction[dir]="SB" and periods[per] = "AM") then method = "traverse" - else method = "entryexit" - - links_array = toll_links.(direction[dir]).(method) - tolls_array = tolls.(direction[dir]).(method).(periods[per]) - - // set toll values - for i=1 to links_array.length do - record_handle = LocateRecord (lyr+"|", "ID", {links_array[i]},{{"Exact", "True"}}) - SetRecordValues(lyr, record_handle, {{toll_fields[per][1],tolls_array[i]}}) - end - end - end - -EndMacro - -/********************************************************************************************************** - add link attributes for tod periods - - Inputs - output\hwy.dbd - - Outputs: - output\hwy.dbd (modified) - -**********************************************************************************************************/ -Macro "add TOD attributes" - - shared path, inputDir, outputDir - ok=0 - - db_file=outputDir+"\\hwy.dbd" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr = db_file + "|" + link_lyr - - vw = SetView(link_lyr) - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // AB Link capacity - strct = strct + {{"ABCP_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCP_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCP_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCP_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCP_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Link capacity - strct = strct + {{"BACP_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACP_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACP_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACP_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACP_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Intersection capacity - strct = strct + {{"ABCX_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCX_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCX_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCX_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCX_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Intersection capacity - strct = strct + {{"BACX_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACX_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACX_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACX_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACX_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Link time - strct = strct + {{"ABTM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTM_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Link time - strct = strct + {{"BATM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATM_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Intersection time - strct = strct + {{"ABTX_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTX_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTX_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTX_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABTX_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Intersection time - strct = strct + {{"BATX_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATX_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATX_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATX_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BATX_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Lanes - strct = strct + {{"ABLN_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLN_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLN_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLN_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLN_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Lanes - strct = strct + {{"BALN_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALN_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALN_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALN_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALN_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Drive-alone cost - strct = strct + {{"ABSCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Drive-alone cost - strct = strct + {{"BASCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Shared 2 cost - strct = strct + {{"ABH2CST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH2CST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH2CST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH2CST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH2CST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Shared 2 cost - strct = strct + {{"BAH2CST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH2CST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH2CST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH2CST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH2CST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Shared-3 cost - strct = strct + {{"ABH3CST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH3CST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH3CST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH3CST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABH3CST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Shared-3 cost - strct = strct + {{"BAH3CST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH3CST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH3CST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH3CST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAH3CST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Light-Heavy truck cost - strct = strct + {{"ABLHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Light-Heavy truck cost - strct = strct + {{"BALHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Medium-Heavy truck cost - strct = strct + {{"ABMHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABMHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABMHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABMHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABMHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Medium-Heavy truck cost - strct = strct + {{"BAMHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAMHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAMHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAMHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAMHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Heavy-Heavy truck cost - strct = strct + {{"ABHHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Heavy-Heavy truck cost - strct = strct + {{"BAHHCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHHCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHHCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHHCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHHCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB Commercial vehicle cost - strct = strct + {{"ABCVCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCVCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCVCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCVCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABCVCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Commercial vehicle cost - strct = strct + {{"BACVCST_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACVCST_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACVCST_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACVCST_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BACVCST_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB SOV Time - strct = strct + {{"ABSTM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTM_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA SOV Time - strct = strct + {{"BASTM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTM_EV", "Real", 14, 6, "True", , , , , , , null}} - - // AB HOV Time - strct = strct + {{"ABHTM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHTM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHTM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHTM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABHTM_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA HOV Time - strct = strct + {{"BAHTM_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHTM_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHTM_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHTM_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BAHTM_EV", "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - // initialize time and cost fields to 999999 - tod_fld = {{"ABSCST_EA"},{"ABSCST_AM"},{"ABSCST_MD"},{"ABSCST_PM"},{"ABSCST_EV"}, - {"BASCST_EA"},{"BASCST_AM"},{"BASCST_MD"},{"BASCST_PM"},{"BASCST_EV"}, - {"ABH2CST_EA"},{"ABH2CST_AM"},{"ABH2CST_MD"},{"ABH2CST_PM"},{"ABH2CST_EV"}, - {"BAH2CST_EA"},{"BAH2CST_AM"},{"BAH2CST_MD"},{"BAH2CST_PM"},{"BAH2CST_EV"}, - {"ABH3CST_EA"},{"ABH3CST_AM"},{"ABH3CST_MD"},{"ABH3CST_PM"},{"ABH3CST_EV"}, - {"BAH3CST_EA"},{"BAH3CST_AM"},{"BAH3CST_MD"},{"BAH3CST_PM"},{"BAH3CST_EV"}, - {"ABSTM_EA"},{"ABSTM_AM"},{"ABSTM_MD"},{"ABSTM_PM"},{"ABSTM_EV"}, - {"BASTM_EA"},{"BASTM_AM"},{"BASTM_MD"},{"BASTM_PM"},{"BASTM_EV"}, - {"ABHTM_EA"},{"ABHTM_AM"},{"ABHTM_MD"},{"ABHTM_PM"},{"ABHTM_EV"}, - {"BAHTM_EA"},{"BAHTM_AM"},{"BAHTM_MD"},{"BAHTM_PM"},{"BAHTM_EV"}, - {"ABLHCST_EA"},{"ABLHCST_AM"},{"ABLHCST_MD"},{"ABLHCST_PM"},{"ABLHCST_EV"}, - {"BALHCST_EA"},{"BALHCST_AM"},{"BALHCST_MD"},{"BALHCST_PM"},{"BALHCST_EV"}, - {"ABMHCST_EA"},{"ABMHCST_AM"},{"ABMHCST_MD"},{"ABMHCST_PM"},{"ABMHCST_EV"}, - {"BAMHCST_EA"},{"BAMHCST_AM"},{"BAMHCST_MD"},{"BAMHCST_PM"},{"BAMHCST_EV"}, - {"ABHHCST_EA"},{"ABHHCST_AM"},{"ABHHCST_MD"},{"ABHHCST_PM"},{"ABHHCST_EV"}, - {"BAHHCST_EA"},{"BAHHCST_AM"},{"BAHHCST_MD"},{"BAHHCST_PM"},{"BAHHCST_EV"}, - {"ABCVCST_EA"},{"ABCVCST_AM"},{"ABCVCST_MD"},{"ABCVCST_PM"},{"ABCVCST_EV"}, - {"BACVCST_EA"},{"BACVCST_AM"},{"BACVCST_MD"},{"BACVCST_PM"},{"BACVCST_EV"} - } - - // now calculate fields - for i=1 to tod_fld.length do - - calcString = {"999999"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tod_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - - // set capacity fields - tod_fld ={{"ABCP_EA"},{"ABCP_AM"},{"ABCP_MD"},{"ABCP_PM"},{"ABCP_EV"}, //BA link capacity - {"BACP_EA"},{"BACP_AM"},{"BACP_MD"},{"BACP_PM"},{"BACP_EV"}, //AB link capacity - {"ABCX_EA"},{"ABCX_AM"},{"ABCX_MD"},{"ABCX_PM"},{"ABCX_EV"}, //BA intersection capacity - {"BACX_EA"},{"BACX_AM"},{"BACX_MD"},{"BACX_PM"},{"BACX_EV"}} //AB intersection capacity - - org_fld ={"ABCPO","ABCPA","ABCPO","ABCPP","ABCPO", - "BACPO","BACPA","BACPO","BACPP","BACPO", - "ABCXO","ABCXA","ABCXO","ABCXP","ABCXO", - "BACXO","BACXA","BACXO","BACXP","BACXO"} - - factor ={"3/12","1","6.5/12","3.5/3","8/12", - "3/12","1","6.5/12","3.5/3","8/12", - "3/12","1","6.5/12","3.5/3","8/12", - "3/12","1","6.5/12","3.5/3","8/12"} - - // now calculate capacity - for i=1 to tod_fld.length do - - calcString = {"if "+org_fld[i]+ " != 999999 then " + factor[i] + " * " + org_fld[i] + " else 999999"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tod_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - - // set time fields - tod_fld ={{"ABTM_EA"},{"ABTM_AM"},{"ABTM_MD"},{"ABTM_PM"},{"ABTM_EV"}, //BA link time - {"BATM_EA"},{"BATM_AM"},{"BATM_MD"},{"BATM_PM"},{"BATM_EV"}, //AB link time - {"ABTX_EA"},{"ABTX_AM"},{"ABTX_MD"},{"ABTX_PM"},{"ABTX_EV"}, //BA intersection time - {"BATX_EA"},{"BATX_AM"},{"BATX_MD"},{"BATX_PM"},{"BATX_EV"}} //AB intersection time - - org_fld ={"ABTMO","ABTMA","ABTMO","ABTMP","ABTMO", - "BATMO","BATMA","BATMO","BATMP","BATMO", - "ABTXO","ABTXA","ABTXO","ABTXP","ABTXO", - "BATXO","BATXA","BATXO","BATXP","BATXO"} - - factor ={"1","1","1","1","1", - "1","1","1","1","1", - "1","1","1","1","1", - "1","1","1","1","1"} - - // now calculate time - for i=1 to tod_fld.length do - - calcString = { factor[i] + " * " + org_fld[i]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tod_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // set lane fields - tod_fld ={{"ABLN_EA"},{"ABLN_AM"},{"ABLN_MD"},{"ABLN_PM"},{"ABLN_EV"}, //AB lanes - {"BALN_EA"},{"BALN_AM"},{"BALN_MD"},{"BALN_PM"},{"BALN_EV"}} //BA lanes - - org_fld ={"ABLNO","ABLNA","ABLNO","ABLNP","ABLNO", - "BALNO","BALNA","BALNO","BALNP","BALNO"} - - factor ={"1","1","1","1","1", - "1","1","1","1","1"} - - // now calculate time - for i=1 to tod_fld.length do - - calcString = { factor[i] + " * " + org_fld[i]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = tod_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - RunMacro("close all") - - ok=1 - quit: - return(ok) -EndMacro - -/********************************************************************************************************** - Add link fields for reliability - - v/c factor fields: - {"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, - {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}, - - static reliability fields: - {"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, - {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"} - - interchange fields - used in static reliability calculations - {"INTDIST_UP"},{"INTDIST_DOWN"} - - Regression equations for static reliability: - - static reliability(freeway) = intercept + coeff1*ISPD70 + coeff2*1/MajorUpstream + coeff3*1/MajorDownstream - static reliability(arterial) = intercept + coeff1*NumLanesOneLane + coeff2*NumLanesCatTwoLane + coeff3*NumLanesCatThreeLane + coeff4*NumLanesCatFourLanes + coeff5*NumLanesFiveMoreLane + - coeff6*ISPD.CatISPD35Less + coeff7*ISPD.CatISPD35 + coeff8*ISPD.CatISPD40 + coeff9*ISPD.CatISPD45 + coeff1*ISPD.CatISPD50 + coeff10*ISPD.CatISPD50More + - coeff11*ICNT.EstSignal + coeff12*ICNT.EstStop + coeff13*ICNT.EstRailRoad - - Where; - ISPD70: 1 if ISPD=70 else 0 (ISPD is posted speed) - MajorUpstream: distance to major interchange upstream (miles) - MajorDownstream: distance to major interchange downstream (miles) - NumLanesOneLane: 1 if lane=1 else 0 - NumLanesCatTwoLane: 1 if lane=2 else 0 - NumLanesCatThreeLane: 1 if lane=3 else 0 - NumLanesCatFourLanes: 1 if lane=4 else 0 - NumLanesFiveMoreLane: 1 if lane>=5 else 0 - ISPD.CatISPD35Less: 1 if ISPD <35 else 0 - ISPD.CatISPD35: 1 if ISPD =35 else 0 - ISPD.CatISPD40: 1 if ISPD =40 else 0 - ISPD.CatISPD45: 1 if ISPD =45 else 0 - ISPD.CatISPD50: 1 if ISPD =50 else 0 - ISPD.CatISPD50More: 1 if ISPD >=50 else 0 - ICNT.EstSignal: 1 if ICNT=1 else 0 (ICNT is intersection control type); signal-controlled - ICNT.EstStop: 1 if ICNT=2 or ICNT=3 else 0; stop-controlled - ICNT.EstRailRoad: 1 if ICNT>3 else 0; other - railroad etc. - - Steps: - 1. add new fields - 2. populate with default values - 3. calculate v/c factor fields by setting them to estimated coefficients by facility type - freeway and arterial. Ramp and other use arterial coefficients. - 4. pupulate interchange fields by joining highway database with major interchange distance file (output from distance to interchange macro). - 5. calculate static reliability fields for freeway - 6. calculate static reliability fields for arterial, ramp, and other - - Inputs - output\hwy.dbd - output\MajorInterchangeDistance.csv - - Outputs: - output\hwy.dbd (modified) - -by: nagendra.dhakar@rsginc.com -**********************************************************************************************************/ -Macro "add reliability fields" - - shared path, inputDir, outputDir - ok=0 - - db_file=outputDir+"\\hwy.dbd" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr = db_file + "|" + link_lyr - - vw = SetView(link_lyr) - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // **** step 1. add new fields - - // AB v/c factors - strct = strct + {{"ABLOSC_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLOSD_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLOSE_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLOSFL_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABLOSFH_FACT", "Real", 14, 6, "True", , , , , , , null}} - - // BA v/c factors - strct = strct + {{"BALOSC_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALOSD_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALOSE_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALOSFL_FACT", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BALOSFH_FACT", "Real", 14, 6, "True", , , , , , , null}} - - // AB Static Reliability - strct = strct + {{"ABSTATREL_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTATREL_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTATREL_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTATREL_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ABSTATREL_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA Static Reliability - strct = strct + {{"BASTATREL_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTATREL_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTATREL_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTATREL_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BASTATREL_EV", "Real", 14, 6, "True", , , , , , , null}} - - // interchange distance - strct = strct + {{"INTDIST_UP", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"INTDIST_DOWN", "Real", 14, 6, "True", , , , , , , null}} - - // AB total reliability - strct = strct + {{"AB_TOTREL_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_TOTREL_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_TOTREL_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_TOTREL_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_TOTREL_EV", "Real", 14, 6, "True", , , , , , , null}} - - // BA total reliability - strct = strct + {{"BA_TOTREL_EA", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_TOTREL_AM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_TOTREL_MD", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_TOTREL_PM", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_TOTREL_EV", "Real", 14, 6, "True", , , , , , , null}} - - // reliability fields - reliability_fld = {{"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, - {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}, - {"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, - {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"}, - {"AB_TOTREL_EA"},{"AB_TOTREL_AM"},{"AB_TOTREL_MD"},{"AB_TOTREL_PM"},{"AB_TOTREL_EV"}, - {"BA_TOTREL_EA"},{"BA_TOTREL_AM"},{"BA_TOTREL_MD"},{"BA_TOTREL_PM"},{"BA_TOTREL_EV"}} - - ModifyTable(view1, strct) - - // for debug - //RunMacro("TCB Init") - - // **** step 2. populate with default value of 0 - for i=1 to reliability_fld.length do - - calcString = {"0"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = reliability_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // **** step 3. calculate v/c factor fields - - los_fld ={{"ABLOSC_FACT"},{"ABLOSD_FACT"},{"ABLOSE_FACT"},{"ABLOSFL_FACT"},{"ABLOSFH_FACT"}, //BA link time - {"BALOSC_FACT"},{"BALOSD_FACT"},{"BALOSE_FACT"},{"BALOSFL_FACT"},{"BALOSFH_FACT"}} //AB intersection time - - factor_freeway ={"0.2429","0.1705","-0.2278","-0.1983","1.022", - "0.2429","0.1705","-0.2278","-0.1983","1.022"} - - factor_arterial ={"0.1561","0.0","0.0","-0.1449","0", - "0.1561","0.0","0.0","-0.1449","0"} - - facility_type = {"freeway","arterial","ramp","other"} // freeway (IFC=1), arterial (IFC=2,3), ramp (IFC=8,9), other (IFC=4,5,6,7) - - // lower and upper bounds of IFC for respective facility type = {freeway, arterial, ramp, other} - lwr_bound = {"1","2","8","4"} - upr_bound = {"1","3","9","7"} - - // now calculate v/c factor fields - for fac_type=1 to facility_type.length do - - // set factors (coefficients) for facility type - if fac_type=1 then factor=factor_freeway - else factor=factor_arterial - - for i=1 to los_fld.length do - - query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = los_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = factor[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - end - - // **** step 4. populate interchange fields (upstream/downstream distance to major interchange) - - distance_file = outputDir+"\\MajorInterchangeDistance.csv" - distance_fld = {"updistance","downdistance"} - - // interchange distance fields - interchange_fld = {{"INTDIST_UP"},{"INTDIST_DOWN"}} - - // set initial value to 9999 - for i=1 to interchange_fld.length do - - calcString = {"9999"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = interchange_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // now set to distances - in miles - for i=1 to interchange_fld.length do - Opts = null - Opts.Input.[Dataview Set] = {{db_link_lyr, distance_file,{"ID"},{"LinkID"}},"JoinedView"} - Opts.Global.Fields = interchange_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = distance_fld[i] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - // **** step 5. calculate static reliability fields for freeway - - // static reliability(freeway) = intercept + coeff1*ISPD70 + coeff2*1/MajorUpstream + coeff3*1/MajorDownstream - - static_fld = {{"ABSTATREL_EA"},{"ABSTATREL_AM"},{"ABSTATREL_MD"},{"ABSTATREL_PM"},{"ABSTATREL_EV"}, - {"BASTATREL_EA"},{"BASTATREL_AM"},{"BASTATREL_MD"},{"BASTATREL_PM"},{"BASTATREL_EV"}} - - // Freeway coefficients - intercept = {"0.1078"} - speed_factor = {"0.01393"} // ISPD70 - interchange_factor = {"0.011","0.0005445"} //MajorUpstream.Inverse, MajorDownstream.Inverse - - fac_type=1 - factor=factor_freeway - - for i=1 to static_fld.length do - query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] - - // intercept - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = intercept[1] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // ISPD - add to intercept - calcString = {"if ISPD=70 then " + static_fld[i][1] + "+" +speed_factor[1] + " else " + static_fld[i][1]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // Upstream interchange distance - apply inverse and add to intercept and ISPD - for j=1 to interchange_factor.length do - - calcString = {"if " + interchange_fld[j][1] + "<>null then " + static_fld[i][1] + "+" +interchange_factor[j]+"*1/"+interchange_fld[j][1] + " else " + static_fld[i][1]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - - end - - // **** step 6. calculate static reliability fields for arterial, ramp, and other - - // Factors (coefficients) - intercept = {"0.0546552"} - lane_factor = {"0.0","0.0103589","0.0361211","0.0446958","0.0"} //{NumLanesOneLane, NumLanesCatTwoLane, NumLanesCatThreeLane, NumLanesCatFourLanes, NumLanesFiveMoreLane} - speed_factor = {"0.0","0.0075674","0.0091012","0.0080996","-0.0022938","-0.0046211"} //{ISPD.CatISPD35Less (base), ISPD.CatISPD35, ISPD.CatISPD40, ISPD.CatISPD45, ISPD.CatISPD50, ISPD.CatISPD50More} - intersection_factor = {"0.0030973","-0.0063281","0.0127692"} //{ICNT.EstSignal, ICNT.EstStop, ICNT.EstRailRoad} - - // lane fields in network - lane_fld ={{"ABLN_EA"},{"ABLN_AM"},{"ABLN_MD"},{"ABLN_PM"},{"ABLN_EV"}, //AB lanes - {"BALN_EA"},{"BALN_AM"},{"BALN_MD"},{"BALN_PM"},{"BALN_EV"}} //BA lanes - - // intersection fields in network - intersection_fld = {{"ABCNT"},{"ABCNT"},{"ABCNT"},{"ABCNT"},{"ABCNT"}, - {"BACNT"},{"BACNT"},{"BACNT"},{"BACNT"},{"BACNT"}} - - // static reliability(arterial) = intercept + coeff1*NumLanesOneLane + coeff2*NumLanesCatTwoLane + coeff3*umLanesCatThreeLane + coeff4*NumLanesCatFourLanes + coeff5*NumLanesFiveMoreLane+ - // coeff6*ISPD.CatISPD35Less + coeff7*ISPD.CatISPD35 + coeff8*ISPD.CatISPD40 + coeff9*ISPD.CatISPD45 + coeff1*ISPD.CatISPD50 + coeff10*ISPD.CatISPD50More+ - // coeff11*ICNT.EstSignal + coeff12*ICNT.EstStop + coeff13*ICNT.EstRailRoad - - for fac_type=2 to facility_type.length do - - // selection query - to identify links with a facility type - query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] - - for i=1 to static_fld.length do - - // intercept - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = intercept[1] - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // NumLanes Factors - calcString = {"if " + lane_fld[i][1] + "=1 then "+static_fld[i][1] + "+" +lane_factor[1]+ - " else if " + lane_fld[i][1] + "=2 then "+static_fld[i][1] + "+" +lane_factor[2]+ - " else if " + lane_fld[i][1] + "=3 then "+static_fld[i][1] + "+" +lane_factor[3]+ - " else if " + lane_fld[i][1] + "=4 then "+static_fld[i][1] + "+" +lane_factor[4]+ - " else "+static_fld[i][1] + "+" +lane_factor[5]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // Speed Factors - - calcString = {"if ISPD<35 then "+static_fld[i][1] + "+" +speed_factor[1]+ - " else if ISPD=35 then "+static_fld[i][1] + "+" +speed_factor[2]+ - " else if ISPD=40 then "+static_fld[i][1] + "+" +speed_factor[3]+ - " else if ISPD=45 then "+static_fld[i][1] + "+" +speed_factor[4]+ - " else if ISPD=50 then "+static_fld[i][1] + "+" +speed_factor[5]+ - " else "+static_fld[i][1] + "+" +speed_factor[6]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // Intersection Factors - calcString = {"if " + intersection_fld[i][1]+ "=1 then " + static_fld[i][1] + "+" +intersection_factor[1]+ - " else if " + intersection_fld[i][1]+ "=2 or " + intersection_fld[i][1]+ "=3 then " + static_fld[i][1] + "+" + intersection_factor[2]+ - " else if " + intersection_fld[i][1]+ ">3 then " + static_fld[i][1] + "+" + intersection_factor[3] + - " else " + static_fld[i][1]} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection" , query} - Opts.Global.Fields = static_fld[i] - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - end - end - - RunMacro("close all") - - ok=1 - quit: - return(ok) -EndMacro -/************************************************************************************************ -add preload attributes - -Adds fields to highway line layer for storing preload volumes (currently bus volumes) - -************************************************************************************************/ -Macro "add preload attributes" - - shared path, inputDir, outputDir - - db_file=outputDir+"\\hwy.dbd" - - periods={"_EA","_AM","_MD","_PM","_EV"} - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr = db_file + "|" + link_lyr - - vw = SetView(link_lyr) - strct = GetTableStructure(vw) - - // Copy the current name to the end of strct - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // Add fields to the output table - new_struct = strct + { - {"ABPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}} - - // Modify table structure - ModifyTable(vw, new_struct) - - // initialize to 0 - for i = 1 to periods.length do - - ABField = "ABPRELOAD"+periods[i] - BAField = "BAPRELOAD"+periods[i] - - //initialize to 0 - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {ABField} - Opts.Global.Method = "Value" - Opts.Global.Parameter = {0} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {BAField} - Opts.Global.Method = "Value" - Opts.Global.Parameter = {0} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - end - - - RunMacro("close all") - - ok=1 - quit: - return(ok) - -EndMacro -/************************************************************************************************** - -Macro Code VDF fields - -This macro codes fields for the Tucson volume-delay function on the input highway line layer. It -should be run prior to constructing highway networks for skim-building & assignment. Eventually -the logic in this macro will be replaced by GIS code. - -Functional class (IFC) - - 1 = Freeway - 2 = Prime arterial - 3 = Major arterial - 4 = Collector - 5 = Local collector - 6 = Rural collector - 7 = Local (non Circulation Element) road - 8 = Freeway connector ramps - 9 = Local ramps - 10 = Zone connectors - -Cycle length matrix - - Intersecting Link -Approach Link 2 3 4 5 6 7 8 9 -IFC Description Prime Arterial Major Arterial Collector Local Collector Rural Collector Local Road Freeway connector Local Ramp -2 Prime Arterial 2.5 2 2 2 2 2 2 2 -3 Major Arterial 2 2 2 2 2 2 2 2 -4 Collector 2 2 1.5 1.5 1.5 1.5 1.5 1.5 -5 Local Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 -6 Rural Collector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 -7 Local Road 2 2 1.5 1.25 1.25 1.25 1.25 1.25 -8 Freeway connector 2 2 1.5 1.25 1.25 1.25 1.25 1.25 -9 Local Ramp 2 2 1.5 1.25 1.25 1.25 1.25 1.25 - -Ramp with meter (abcnt = 4 or 5) - Cycle length = 2.5 - GC ratio 0.42 (adjusted down from 0.5 for yellow) - -Stop controlled intersection - Cycle length =1.25 - GC ratio 0.42 (adjusted down from 0.5 for yellow) - -*************************************************************************************************/ -Macro "Code VDF fields" - shared path, inputDir, outputDir - - db_file=outputDir+"\\hwy.dbd" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr = db_file + "|" + link_lyr - - - // Add AB_Cycle, AB_PF, BA_Cycle, and BA_PF - vw = SetView(link_lyr) - strct = GetTableStructure(vw) - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - strct = strct + {{"AB_GCRatio", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_GCRatio", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_Cycle", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_Cycle", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_PF", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_PF", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ALPHA1", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BETA1", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ALPHA2", "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BETA2", "Real", 14, 6, "True", , , , , , , null}} - ModifyTable(view1, strct) - - // Now set the view to the node layer - SetLayer(node_lyr) - - nodes = GetDataVector(node_lyr+"|", "ID",) - - //create from/end node field in the line layer - start_fld = CreateNodeField(link_lyr, "start_node", node_lyr+".ID", "From", ) - end_fld = CreateNodeField(link_lyr, "end_node", node_lyr+".ID", "To", ) - - //get the count of records for both line and node layer - tot_line_count = GetRecordCount(link_lyr, ) //get number of records in linelayer - tot_node_count = GetRecordCount(node_lyr, ) //get number of records in nodelayer - - //initilize several vectors -// linkclass = {Vector(tot_line_count, "Float", {{"Constant", null}}),Vector(tot_line_count, "Float", {{"Constant", null}})} //line link-classes - - //pass the attibutes to the vectors -// linkclass = GetDataVector(link_lyr+"|", "IFC",) - -// lineId = GetDataVector(link_lyr+"|", "ID",) - - - //go node by node - for i = 1 to tot_node_count do - - //find the IDs of the links that connect at the node, put the effective links to vector 'link_list' - link_list = null - rec_handles = null - - all_link_list = GetNodeLinks(nodes[i]) //get all links connecting at this node - link_list = Vector(all_link_list.length, "Short", {{"Constant", 0}}) //to contain non-connector/ramp links coming to the node - - all_rec_handles = Vector(all_link_list.length, "String", {{"Constant", null}}) //to contain the record handle of all links coming to the node - rec_handles = Vector(all_link_list.length, "String", {{"Constant", null}}) //to contain the record handle of the non-connector/ramp links coming to the node - - //count how many links entering the node - link_count = 0 - two_oneway = 0 - signal = 0 - - for j = 1 to all_link_list.length do - record_handle = LocateRecord(link_lyr+"|","ID",{all_link_list[j]}, {{"Exact", "True"}}) - all_rec_handles[j] = record_handle - - ends_at_node_AB_direction = 0 - ends_at_node_BA_direction = 0 - - signal_at_end = 0 - - if(link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_AB_direction = 1 - if(link_lyr.start_node = nodes[i] and link_lyr.Dir = -1) then ends_at_node_AB_direction = 1 - if(link_lyr.start_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_BA_direction = 1 - - if ( ends_at_node_AB_direction = 1 and (link_lyr.[ABGC] <> null and link_lyr.[ABGC] > 0)) then signal_at_end = 1 - if ( ends_at_node_BA_direction = 1 and (link_lyr.[BAGC] <> null and link_lyr.[BAGC] > 0)) then signal_at_end = 1 - - - //only count the links that have approach toward the node - if( ends_at_node_AB_direction = 1 and signal_at_end = 1) then do - signal = signal + 1 - end - else if( ends_at_node_BA_direction and signal_at_end = 1) then do - signal = signal + 1 - end - - link_count = link_count + 1 - link_list[link_count] = all_link_list[j] - rec_handles[link_count] = record_handle - if link_lyr.Dir <> 0 then two_oneway = two_oneway+1 - - - end - - // if at least one incoming link has a gc ratio - if (signal>0) then do - min_lc = 999 - max_lc = 0 - //process the links and find the lowest and highest linkclasses - for j = 1 to link_count do - //find the line record that owns the line ID - SetRecord(link_lyr, rec_handles[j]) //set the current record with the record handle stored in vector 'rec_handles' - - if ((link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) or - (link_lyr.start_node = nodes[i] and link_lyr.Dir = -1))then do - if link_lyr.[IFC] <> null and link_lyr.[IFC] > 1 and link_lyr.[IFC] < 10 then do //don't count freeways or centroid connectors - if link_lyr.[IFC] > max_lc then max_lc = link_lyr.[IFC] - if link_lyr.[IFC] < min_lc then min_lc = link_lyr.[IFC] - end - end - - end - end - - //iterate through all links at this node and set cycle length - for j = 1 to all_link_list.length do - - SetRecord(link_lyr, all_rec_handles[j]) //set the current record with the record handle stored in vector 'all_rec_handles' - - if(link_lyr.end_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_AB_direction = 1 - if(link_lyr.start_node = nodes[i] and link_lyr.Dir = -1) then ends_at_node_AB_direction = 1 - if(link_lyr.start_node = nodes[i] and (link_lyr.Dir = 1 or link_lyr.Dir = 0)) then ends_at_node_BA_direction = 1 - - // Set AB fields for links whose end node is this node and are coded in the A->B direction - if (ends_at_node_AB_direction = 1) then do - - //defaults are 1.25 minute cycle length and 1.0 progression factor - c_len = 1.25 - p_factor = 1.0 - - //set up the cycle length for AB direction if there is a gc ratio and more than 2 links - if (link_lyr.[ABGC]<>0 and signal > 0) then do - - if (link_lyr.[IFC] = 2) then do - if (max_lc = 2) then c_len = 2.5 //Prime arterial & Prime arterial - else c_len = 2.0 //Prime arterial & anything lower - end - else if (link_lyr.[IFC] = 3) then do - if (max_lc > 3) then c_len = 2.0 //Major arterial & anything lower than a Major arterial - else c_len = 2.0 //Major arterial & Prime arterial or Major arterial - end - else if (link_lyr.[IFC] = 4) then do - if (min_lc < 4) then c_len = 2.0 //Anything lower than a Major arterial & Prime arterial - else c_len = 1.5 //Anything lower than a Major arterial & anything lower than a Prime arterial - end - else if (link_lyr.[IFC] > 4) then do - if (min_lc < 4) then c_len = 2.0 - if (min_lc = 4) then c_len = 1.5 - if (min_lc > 4) then c_len = 1.25 - end - - //update attributes - if( link_lyr.[ABGC] > 10) then link_lyr.[AB_GCRatio] = link_lyr.[ABGC]/100 - if( link_lyr.[AB_GCRatio] > 1.0) then link_lyr.[AB_GCRatio] = 1.0 - - link_lyr.[AB_Cycle] = c_len - link_lyr.[AB_PF] = p_factor - - end - - end // end for AB links - - // Set BA fields for links whose start node is this node and are coded in the A->B direction - if (ends_at_node_BA_direction = 1 ) then do - - // Only code links with an existing GC ratio (indicating a signalized intersection) - if (link_lyr.[BAGC]<>0 and signal > 0) then do - - //defaults are 0.4 gc ratio, 1.25 minute cycle length and 1.0 progression factor - gc_ratio = 0.4 - c_len = 1.25 - p_factor = 1.0 - - if (link_lyr.[IFC] = 2) then do - if (max_lc = 2) then c_len = 2.5 //Prime arterial & Prime arterial - else c_len = 2.0 //Prime arterial & anything lower - end - else if (link_lyr.[IFC] = 3) then do - if (max_lc > 3) then c_len = 2.0 //Major arterial & anything lower than a Major arterial - else c_len = 2.0 //Major arterial & Prime arterial or Major arterial - end - else if (link_lyr.[IFC] = 4) then do - if (min_lc < 4) then c_len = 2.0 //Anything lower than a Major arterial & Prime arterial - else c_len = 1.5 //Anything lower than a Major arterial & anything lower than a Prime arterial - end - else if (link_lyr.[IFC] > 4) then do - if (min_lc < 4) then c_len = 2.0 - if (min_lc = 4) then c_len = 1.5 - if (min_lc > 4) then c_len = 1.25 - end - - //update attributes - if( link_lyr.[BAGC] > 10) then link_lyr.[BA_GCRatio] = link_lyr.[BAGC]/100 - if( link_lyr.[BA_GCRatio] > 1.0) then link_lyr.[BA_GCRatio] = 1.0 - link_lyr.[BA_Cycle] = c_len - link_lyr.[BA_PF] = p_factor - - end - - end // end for BA links - - //code metered ramps AB Direction - if(ends_at_node_AB_direction = 1 and (link_lyr.[ABCNT]= 4 or link_lyr.[ABCNT] = 5)) then do - link_lyr.[AB_Cycle] = 2.5 - link_lyr.[AB_GCRatio] = 0.42 - link_lyr.[AB_PF] = 1.0 - end - - //code metered ramps BA Direction - if(ends_at_node_BA_direction = 1 and (link_lyr.[BACNT]= 4 or link_lyr.[BACNT] = 5)) then do - link_lyr.[BA_Cycle] = 2.5 - link_lyr.[BA_GCRatio] = 0.42 - link_lyr.[BA_PF] = 1.0 - end - - //code stops AB Direction - if(ends_at_node_AB_direction = 1 and (link_lyr.[ABCNT]= 2 or link_lyr.[ABCNT] = 3)) then do - link_lyr.[AB_Cycle] = 1.25 - link_lyr.[AB_GCRatio] = 0.42 - link_lyr.[AB_PF] = 1.0 - end - - //code stops BA Direction - if(ends_at_node_BA_direction = 1 and (link_lyr.[BACNT]= 2 or link_lyr.[BACNT] = 3)) then do - link_lyr.[BA_Cycle] = 1.25 - link_lyr.[BA_GCRatio] = 0.42 - link_lyr.[BA_PF] = 1.0 - end - - end // end for links - - end // end for nodes - - - // set alpha1 and beta1 fields, which are based upon free-flow speed to match POSTLOAD loaded time factors - lwr_bound = { " 0", "25", "30", "35", "40", "45", "50", "55", "60", "65", "70", "75"} - upr_bound = { "24", "29", "34", "39", "44", "49", "54", "59", "64", "69", "74", "99"} - alpha1 = {"0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8","0.8"} - beta1 = { "4", "4", "4", "4", "4", "4", "4", "4", "4", "4" , "4", "4"} - - for j = 1 to lwr_bound.length do - - //alpha1 - calcString = { "if ISPD >= " + lwr_bound[j] + " and ISPD <= "+upr_bound[j] + " then "+alpha1[j]+" else ALPHA1"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ALPHA1"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - //beta1 - calcString = { "if ISPD >= " + lwr_bound[j] + " and ISPD <= "+upr_bound[j] + " then "+beta1[j]+" else BETA1"} - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"BETA1"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = calcString - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - end - - //set alpha2 and beta2 fields (note that signalized intersections and stop-controlled intersections have same parameters, only meters vary) - alpha2_default = "4.5" - beta2_default = "2.0" - alpha2_meter = "6.0" - beta2_meter = "2.0" - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ALPHA2","BETA2"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {alpha2_default, beta2_default} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr,"Selection", "Select * where (abcnt=4 or abcnt=5)"} - Opts.Global.Fields = {"ALPHA2","BETA2"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {alpha2_meter,beta2_meter} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - // replicate across all time periods and add fields for vdf parameters - periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} - meters = { 0, 1, 0, 1, 0} - - for i = 1 to periods.length do - - // Add AB_Cycle, AB_PF, BA_Cycle, and BA_PF - vw = SetView(link_lyr) - strct = GetTableStructure(vw) - for j = 1 to strct.length do - strct[j] = strct[j] + {strct[j][1]} - end - - strct = strct + {{"AB_GCRatio"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_GCRatio"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_Cycle"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_Cycle"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"AB_PF"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BA_PF"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ALPHA1"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BETA1"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"ALPHA2"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - strct = strct + {{"BETA2"+periods[i], "Real", 14, 6, "True", , , , , , , null}} - - ModifyTable(view1, strct) - - in_fld ={"AB_GCRatio", "BA_GCRatio", "AB_Cycle","BA_Cycle","AB_PF","BA_PF"} - - out_fld ={"AB_GCRatio"+periods[i],"BA_GCRatio"+periods[i],"AB_Cycle"+periods[i],"BA_Cycle"+periods[i],"AB_PF"+periods[i],"BA_PF"+periods[i]} - - - values = {0.0,0.0,0.0,0.0,1.0,1.0} - // set GCRatio, Cycle length, PF - for j=1 to out_fld.length do - - //initialize to 0 - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {out_fld[j]} - Opts.Global.Method = "Value" - Opts.Global.Parameter = {values[j]} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr, "Selection", "Select * where "+in_fld[j]+"> 0 and "+in_fld[j]+"<>null"} - Opts.Global.Fields = {out_fld[j]} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {in_fld[j]} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - end - - // reset GCRatio, cycle length and PF fields to 0 if metered ramp and off-peak period - if(meters[i] = 0) then do - - for j=1 to out_fld.length do - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr,"Selection", "Select * where (abcnt=4 or abcnt=5)"} - Opts.Global.Fields = {out_fld[j]} - Opts.Global.Method = "Value" - Opts.Global.Parameter = {values[j]} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - end - end - - - //set alpha1 and beta1 fields, which currently do not vary by time period - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ALPHA1"+periods[i], "BETA1"+periods[i]} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"ALPHA1","BETA1"} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - //set alpha2 and beta2 fields, which currently do not vary by time period - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ALPHA2"+periods[i],"BETA2"+periods[i]} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"ALPHA2","BETA2"} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - end // end for periods - - quit: - return(ok) -EndMacro -//***************************************************************************************************************************************************************** - -/************************************************************************8 -build highway network in TransCAD format -input file: - hwy.dbd - hwy line geographic file - turns.csv - turn prohibitor csv file, fields: from (link), to (link), penalty (null) - inktypeturns.dbf - dbf file between freeways and ramp added 0.5 min penalty - linktypelog.dbf - link type look up binary file, format for MMA assignment - -output file: hwy.net - hwy network file - link field included in network file: - Length (in miles) - IFC: functional classification, also used as link type look up field - *TM_EA (ABTM_EA/BATM_EA): Early AM period link travel time - *TM_AM (ABTM_AM/BATM_AM): AM Peak period link travel time - *TM_MD (ABTM_MD/BATM_MD): Midday period link travel time - *TM_PM (ABTM_PM/BATM_PM): PM Peak period link travel time - *TM_EV (ABTM_EV/BATM_EV): Evening period link travel time - *CP_EA (ABCP_EA/BACP_EA): Early AM period link capacity - *CP_AM (ABCP_AM/BACP_AM): AM Peak period link capacity - *CP_MD (ABCP_MD/BACP_MD): Midday period link capacity - *CP_PM (ABCP_PM/BACP_PM): PM Peak period link capacity - *CP_EV (ABCP_EV/BACP_EV): Evening period link capacity - *TX_EA (ABTX_EA/BATX_EA): Early AM period intersection travel time - *TX_AM (ABTX_AM/BATX_AM): AM Peak period intersection travel time - *TX_MD (ABTX_MD/BATX_MD): Midday period intersection travel time - *TX_PM (ABTX_PM/BATX_PM): PM Peak period intersection travel time - *TX_EV (ABTX_EV/BATX_EV): Evening period intersection travel time - *CX_EA (ABCX_EA/BACX_EA): Early AM period intersection capacity - *CX_AM (ABCX_AM/BACX_AM): AM Peak period intersection capacity - *CX_MD (ABCX_MD/BACX_MD): Midday period intersection capacity - *CX_PM (ABCX_PM/BACX_PM): PM Peak period intersection capacity - *CX_EV (ABCX_EV/BACX_EV): Evening period intersection capacity - COST: cost of distance (in cents) 19cents/mile - *CST (ABCST/BACST): generalized cost (in cents) of 19cents/mile + 35cents/minute - *SCST - *H2CST - *H3CST - ID: link ID of hwycad-id - - specify zone centroids, and create network - change network settings with linktype lookup table - turn prohibitors and link type turn penalty file - -************************************************************************************/ - -macro "create hwynet" - shared path, mxzone, inputDir, outputDir - ok = 0 - - RunMacro("close all") - - - // fpr=openfile(path+"\\hwycad.log","a") - // mytime=GetDateAndTime() - - //input file - db_file = outputDir + "\\hwy.dbd" - d_tp_tb = inputDir + "\\linktypeturns.dbf" //turn penalty in cents - s_tp_tb = outputDir + "\\turns.dbf" - - //output files - net_file = outputDir+"\\hwy.net" - - di2= GetDirectoryInfo(d_tp_tb, "file") - di3= GetDirectoryInfo(s_tp_tb, "file") - //check for files - if di2.length=0 and di3.length=0 then do - RunMacro("TCB Error",d_tp_tb+" "+s_tp_tb+" does not exist!") - goto quit - end - - // RunMacro("TCB Init") - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - db_link_lyr=db_file+"|"+link_lyr - db_node_lyr=db_file+"|"+node_lyr - - // STEP 1: Build Highway Network - Opts = null - Opts.Input.[Link Set] = {db_link_lyr,link_lyr} - Opts.Global.[Network Options].[Node ID] = node_lyr+ ".ID" - Opts.Global.[Network Options].[Link ID] = link_lyr + ".ID" - Opts.Global.[Network Options].[Turn Penalties] = "Yes" - Opts.Global.[Network Options].[Keep Duplicate Links] = "FALSE" - Opts.Global.[Network Options].[Ignore Link Direction] = "FALSE" - Opts.Global.[Link Options] = {{"Length", link_lyr+".Length", link_lyr+".Length"}, - {"ID", link_lyr+".ID", link_lyr+".ID"}, - {"IFC", link_lyr+".IFC", link_lyr+".IFC"}, - {"IHOV", link_lyr+".IHOV", link_lyr+".IHOV"}, - {"COST", link_lyr+".COST", link_lyr+".COST"}, - {"*LN_EA", link_lyr + ".ABLN_EA", link_lyr + ".BALN_EA"}, - {"*LN_AM", link_lyr + ".ABLN_AM", link_lyr + ".BALN_AM"}, - {"*LN_MD", link_lyr + ".ABLN_MD", link_lyr + ".BALN_MD"}, - {"*LN_PM", link_lyr + ".ABLN_PM", link_lyr + ".BALN_PM"}, - {"*LN_EV", link_lyr + ".ABLN_EV", link_lyr + ".BALN_EV"}, - {"ITOLL_EA", link_lyr + ".ITOLL_EA", link_lyr + ".ITOLL_EA"}, // Oct-08-2010, added to include toll+cost - {"ITOLL_AM", link_lyr + ".ITOLL_AM", link_lyr + ".ITOLL_AM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL_MD", link_lyr + ".ITOLL_MD", link_lyr + ".ITOLL_MD"}, // Oct-08-2010, added to include toll+cost - {"ITOLL_PM", link_lyr + ".ITOLL_PM", link_lyr + ".ITOLL_PM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL_EV", link_lyr + ".ITOLL_EV", link_lyr + ".ITOLL_EV"}, // Oct-08-2010, added to include toll+cost - {"ITOLL2_EA", link_lyr + ".ITOLL2_EA", link_lyr + ".ITOLL2_EA"}, // Oct-08-2010, added to include toll+cost - {"ITOLL2_AM", link_lyr + ".ITOLL2_AM", link_lyr + ".ITOLL2_AM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL2_MD", link_lyr + ".ITOLL2_MD", link_lyr + ".ITOLL2_MD"}, // Oct-08-2010, added to include toll+cost - {"ITOLL2_PM", link_lyr + ".ITOLL2_PM", link_lyr + ".ITOLL2_PM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL2_EV", link_lyr + ".ITOLL2_EV", link_lyr + ".ITOLL2_EV"}, // Oct-08-2010, added to include toll+cost - {"ITOLL3_EA", link_lyr + ".ITOLL3_EA", link_lyr + ".ITOLL3_EA"}, // Oct-08-2010, added to include toll+cost - {"ITOLL3_AM", link_lyr + ".ITOLL3_AM", link_lyr + ".ITOLL3_AM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL3_MD", link_lyr + ".ITOLL3_MD", link_lyr + ".ITOLL3_MD"}, // Oct-08-2010, added to include toll+cost - {"ITOLL3_PM", link_lyr + ".ITOLL3_PM", link_lyr + ".ITOLL3_PM"}, // Oct-08-2010, added to include toll+cost - {"ITOLL3_EV", link_lyr + ".ITOLL3_EV", link_lyr + ".ITOLL3_EV"}, // Oct-08-2010, added to include toll+cost - {"ITOLL4_EA", link_lyr + ".ITOLL4_EA", link_lyr + ".ITOLL4_EA"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost - {"ITOLL4_AM", link_lyr + ".ITOLL4_AM", link_lyr + ".ITOLL4_AM"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost - {"ITOLL4_MD", link_lyr + ".ITOLL4_MD", link_lyr + ".ITOLL4_MD"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost - {"ITOLL4_PM", link_lyr + ".ITOLL4_PM", link_lyr + ".ITOLL4_PM"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost - {"ITOLL4_EV", link_lyr + ".ITOLL4_EV", link_lyr + ".ITOLL4_EV"}, // Nov-3-2010, added lhd & mhd toll=2*toll+cost - {"ITOLL5_EA", link_lyr + ".ITOLL5_EA", link_lyr + ".ITOLL5_EA"}, // Nov-3-2010, added hhd toll = 3*toll+cost - {"ITOLL5_AM", link_lyr + ".ITOLL5_AM", link_lyr + ".ITOLL5_AM"}, // Nov-3-2010, added hhd toll = 3*toll+cost - {"ITOLL5_MD", link_lyr + ".ITOLL5_MD", link_lyr + ".ITOLL5_MD"}, // Nov-3-2010, added hhd toll = 3*toll+cost - {"ITOLL5_PM", link_lyr + ".ITOLL5_PM", link_lyr + ".ITOLL5_PM"}, // Nov-3-2010, added hhd toll = 3*toll+cost - {"ITOLL5_EV", link_lyr + ".ITOLL5_EV", link_lyr + ".ITOLL5_EV"}, // Nov-3-2010, added hhd toll = 3*toll+cost - {"ITRUCK", link_lyr + ".ITRUCK", link_lyr + ".ITRUCK"}, // Sep-30-2011, added ITRUCK to the network - {"*CP_EA", link_lyr+".ABCP_EA", link_lyr+".BACP_EA"}, - {"*CP_AM", link_lyr+".ABCP_AM", link_lyr+".BACP_AM"}, - {"*CP_MD", link_lyr+".ABCP_MD", link_lyr+".BACP_MD"}, - {"*CP_PM", link_lyr+".ABCP_PM", link_lyr+".BACP_PM"}, - {"*CP_EV", link_lyr+".ABCP_EV", link_lyr+".BACP_EV"}, - {"*CX_EA", link_lyr+".ABCX_EA", link_lyr+".BACX_EA"}, - {"*CX_AM", link_lyr+".ABCX_AM", link_lyr+".BACX_AM"}, - {"*CX_MD", link_lyr+".ABCX_MD", link_lyr+".BACX_MD"}, - {"*CX_PM", link_lyr+".ABCX_PM", link_lyr+".BACX_PM"}, - {"*CX_EV", link_lyr+".ABCX_EV", link_lyr+".BACX_EV"}, - {"*TM_EA", link_lyr+".ABTM_EA", link_lyr+".BATM_EA"}, - {"*TM_AM", link_lyr+".ABTM_AM", link_lyr+".BATM_AM"}, - {"*TM_MD", link_lyr+".ABTM_MD", link_lyr+".BATM_MD"}, - {"*TM_PM", link_lyr+".ABTM_PM", link_lyr+".BATM_PM"}, - {"*TM_EV", link_lyr+".ABTM_EV", link_lyr+".BATM_EV"}, - {"*TX_EA", link_lyr+".ABTX_EA", link_lyr+".BATX_EA"}, - {"*TX_AM", link_lyr+".ABTX_AM", link_lyr+".BATX_AM"}, - {"*TX_MD", link_lyr+".ABTX_MD", link_lyr+".BATX_MD"}, - {"*TX_PM", link_lyr+".ABTX_PM", link_lyr+".BATX_PM"}, - {"*TX_EV", link_lyr+".ABTX_EV", link_lyr+".BATX_EV"}, - {"*CST", link_lyr+".ABCST", link_lyr+".BACST"}, - {"*SCST_EA", link_lyr+".ABSCST_EA", link_lyr+".BASCST_EA"}, - {"*SCST_AM", link_lyr+".ABSCST_AM", link_lyr+".BASCST_AM"}, - {"*SCST_MD", link_lyr+".ABSCST_MD", link_lyr+".BASCST_MD"}, - {"*SCST_PM", link_lyr+".ABSCST_PM", link_lyr+".BASCST_PM"}, - {"*SCST_EV", link_lyr+".ABSCST_EV", link_lyr+".BASCST_EV"}, - {"*H2CST_EA", link_lyr+".ABH2CST_EA", link_lyr+".BAH2CST_EA"}, - {"*H2CST_AM", link_lyr+".ABH2CST_AM", link_lyr+".BAH2CST_AM"}, - {"*H2CST_MD", link_lyr+".ABH2CST_MD", link_lyr+".BAH2CST_MD"}, - {"*H2CST_PM", link_lyr+".ABH2CST_PM", link_lyr+".BAH2CST_PM"}, - {"*H2CST_EV", link_lyr+".ABH2CST_EV", link_lyr+".BAH2CST_EV"}, - {"*H3CST_EA", link_lyr+".ABH3CST_EA", link_lyr+".BAH3CST_EA"}, - {"*H3CST_AM", link_lyr+".ABH3CST_AM", link_lyr+".BAH3CST_AM"}, - {"*H3CST_MD", link_lyr+".ABH3CST_MD", link_lyr+".BAH3CST_MD"}, - {"*H3CST_PM", link_lyr+".ABH3CST_PM", link_lyr+".BAH3CST_PM"}, - {"*H3CST_EV", link_lyr+".ABH3CST_EV", link_lyr+".BAH3CST_EV"}, - {"*LHCST_EA", link_lyr+".ABLHCST_EA", link_lyr+".BALHCST_EA"}, - {"*LHCST_AM", link_lyr+".ABLHCST_AM", link_lyr+".BALHCST_AM"}, - {"*LHCST_MD", link_lyr+".ABLHCST_MD", link_lyr+".BALHCST_MD"}, - {"*LHCST_PM", link_lyr+".ABLHCST_PM", link_lyr+".BALHCST_PM"}, - {"*LHCST_EV", link_lyr+".ABLHCST_EV", link_lyr+".BALHCST_EV"}, - {"*MHCST_EA", link_lyr+".ABMHCST_EA", link_lyr+".BAMHCST_EA"}, - {"*MHCST_AM", link_lyr+".ABMHCST_AM", link_lyr+".BAMHCST_AM"}, - {"*MHCST_MD", link_lyr+".ABMHCST_MD", link_lyr+".BAMHCST_MD"}, - {"*MHCST_PM", link_lyr+".ABMHCST_PM", link_lyr+".BAMHCST_PM"}, - {"*MHCST_EV", link_lyr+".ABMHCST_EV", link_lyr+".BAMHCST_EV"}, - {"*HHCST_EA", link_lyr+".ABHHCST_EA", link_lyr+".BAHHCST_EA"}, - {"*HHCST_AM", link_lyr+".ABHHCST_AM", link_lyr+".BAHHCST_AM"}, - {"*HHCST_MD", link_lyr+".ABHHCST_MD", link_lyr+".BAHHCST_MD"}, - {"*HHCST_PM", link_lyr+".ABHHCST_PM", link_lyr+".BAHHCST_PM"}, - {"*HHCST_EV", link_lyr+".ABHHCST_EV", link_lyr+".BAHHCST_EV"}, - {"*CVCST_EA", link_lyr+".ABCVCST_EA", link_lyr+".BACVCST_EA"}, - {"*CVCST_AM", link_lyr+".ABCVCST_AM", link_lyr+".BACVCST_AM"}, - {"*CVCST_MD", link_lyr+".ABCVCST_MD", link_lyr+".BACVCST_MD"}, - {"*CVCST_PM", link_lyr+".ABCVCST_PM", link_lyr+".BACVCST_PM"}, - {"*CVCST_EV", link_lyr+".ABCVCST_EV", link_lyr+".BACVCST_EV"}, - {"*STM_EA", link_lyr+".ABSTM_EA", link_lyr+".BASTM_EA"}, - {"*STM_AM", link_lyr+".ABSTM_AM", link_lyr+".BASTM_AM"}, - {"*STM_MD", link_lyr+".ABSTM_MD", link_lyr+".BASTM_MD"}, - {"*STM_PM", link_lyr+".ABSTM_PM", link_lyr+".BASTM_PM"}, - {"*STM_EV", link_lyr+".ABSTM_EV", link_lyr+".BASTM_EV"}, - {"*HTM_EA", link_lyr+".ABHTM_EA", link_lyr+".BAHTM_EA"}, - {"*HTM_AM", link_lyr+".ABHTM_AM", link_lyr+".BAHTM_AM"}, - {"*HTM_MD", link_lyr+".ABHTM_MD", link_lyr+".BAHTM_MD"}, - {"*HTM_PM", link_lyr+".ABHTM_PM", link_lyr+".BAHTM_PM"}, - {"*HTM_EV", link_lyr+".ABHTM_EV", link_lyr+".BAHTM_EV"}, - {"*GCRATIO_EA", link_lyr+".AB_GCRatio_EA", link_lyr+".BA_GCRatio_EA"}, - {"*GCRATIO_AM", link_lyr+".AB_GCRatio_AM", link_lyr+".BA_GCRatio_AM"}, - {"*GCRATIO_MD", link_lyr+".AB_GCRatio_MD", link_lyr+".BA_GCRatio_MD"}, - {"*GCRATIO_PM", link_lyr+".AB_GCRatio_PM", link_lyr+".BA_GCRatio_PM"}, - {"*GCRATIO_EV", link_lyr+".AB_GCRatio_EV", link_lyr+".BA_GCRatio_EV"}, - {"*CYCLE_EA", link_lyr+".AB_Cycle_EA", link_lyr+".BA_Cycle_EA"}, - {"*CYCLE_AM", link_lyr+".AB_Cycle_AM", link_lyr+".BA_Cycle_AM"}, - {"*CYCLE_MD", link_lyr+".AB_Cycle_MD", link_lyr+".BA_Cycle_MD"}, - {"*CYCLE_PM", link_lyr+".AB_Cycle_PM", link_lyr+".BA_Cycle_PM"}, - {"*CYCLE_EV", link_lyr+".AB_Cycle_EV", link_lyr+".BA_Cycle_EV"}, - {"*PF_EA", link_lyr+".AB_PF_EA", link_lyr+".BA_PF_EA"}, - {"*PF_AM", link_lyr+".AB_PF_AM", link_lyr+".BA_PF_AM"}, - {"*PF_MD", link_lyr+".AB_PF_MD", link_lyr+".BA_PF_MD"}, - {"*PF_PM", link_lyr+".AB_PF_PM", link_lyr+".BA_PF_PM"}, - {"*PF_EV", link_lyr+".AB_PF_EV", link_lyr+".BA_PF_EV"}, - {"*ALPHA1_EA", link_lyr+".ALPHA1_EA", link_lyr+".ALPHA1_EA"}, - {"*ALPHA1_AM", link_lyr+".ALPHA1_AM", link_lyr+".ALPHA1_AM"}, - {"*ALPHA1_MD", link_lyr+".ALPHA1_MD", link_lyr+".ALPHA1_MD"}, - {"*ALPHA1_PM", link_lyr+".ALPHA1_PM", link_lyr+".ALPHA1_PM"}, - {"*ALPHA1_EV", link_lyr+".ALPHA1_EV", link_lyr+".ALPHA1_EV"}, - {"*BETA1_EA", link_lyr+".BETA1_EA", link_lyr+".BETA1_EA"}, - {"*BETA1_AM", link_lyr+".BETA1_AM", link_lyr+".BETA1_AM"}, - {"*BETA1_MD", link_lyr+".BETA1_MD", link_lyr+".BETA1_MD"}, - {"*BETA1_PM", link_lyr+".BETA1_PM", link_lyr+".BETA1_PM"}, - {"*BETA1_EV", link_lyr+".BETA1_EV", link_lyr+".BETA1_EV"}, - {"*ALPHA2_EA", link_lyr+".ALPHA2_EA", link_lyr+".ALPHA2_EA"}, - {"*ALPHA2_AM", link_lyr+".ALPHA2_AM", link_lyr+".ALPHA2_AM"}, - {"*ALPHA2_MD", link_lyr+".ALPHA2_MD", link_lyr+".ALPHA2_MD"}, - {"*ALPHA2_PM", link_lyr+".ALPHA2_PM", link_lyr+".ALPHA2_PM"}, - {"*ALPHA2_EV", link_lyr+".ALPHA2_EV", link_lyr+".ALPHA2_EV"}, - {"*BETA2_EA", link_lyr+".BETA2_EA", link_lyr+".BETA2_EA"}, - {"*BETA2_AM", link_lyr+".BETA2_AM", link_lyr+".BETA2_AM"}, - {"*BETA2_MD", link_lyr+".BETA2_MD", link_lyr+".BETA2_MD"}, - {"*BETA2_PM", link_lyr+".BETA2_PM", link_lyr+".BETA2_PM"}, - {"*BETA2_EV", link_lyr+".BETA2_EV", link_lyr+".BETA2_EV"}, - {"*PRELOAD_EA", link_lyr+".ABPRELOAD_EA", link_lyr+".BAPRELOAD_EA"}, - {"*PRELOAD_AM", link_lyr+".ABPRELOAD_AM", link_lyr+".BAPRELOAD_AM"}, - {"*PRELOAD_MD", link_lyr+".ABPRELOAD_MD", link_lyr+".BAPRELOAD_MD"}, - {"*PRELOAD_PM", link_lyr+".ABPRELOAD_PM", link_lyr+".BAPRELOAD_PM"}, - {"*PRELOAD_EV", link_lyr+".ABPRELOAD_EV", link_lyr+".BAPRELOAD_EV"}, - {"*LOSC_FACT", link_lyr+".ABLOSC_FACT", link_lyr+".BALOSC_FACT"}, // added for reliability - 02/02/2016 - {"*LOSD_FACT", link_lyr+".ABLOSD_FACT", link_lyr+".BALOSD_FACT"}, // added for reliability - 02/02/2016 - {"*LOSE_FACT", link_lyr+".ABLOSE_FACT", link_lyr+".BALOSE_FACT"}, // added for reliability - 02/02/2016 - {"*LOSFL_FACT", link_lyr+".ABLOSFL_FACT", link_lyr+".BALOSFL_FACT"}, // added for reliability - 02/02/2016 - {"*LOSFH_FACT", link_lyr+".ABLOSFH_FACT", link_lyr+".BALOSFH_FACT"}, // added for reliability - 02/02/2016 - {"*STATREL_EA", link_lyr+".ABSTATREL_EA", link_lyr+".BASTATREL_EA"}, // added for reliability - 02/02/2016 - {"*STATREL_AM", link_lyr+".ABSTATREL_AM", link_lyr+".BASTATREL_AM"}, // added for reliability - 02/02/2016 - {"*STATREL_MD", link_lyr+".ABSTATREL_MD", link_lyr+".BASTATREL_MD"}, // added for reliability - 02/02/2016 - {"*STATREL_PM", link_lyr+".ABSTATREL_PM", link_lyr+".BASTATREL_PM"}, // added for reliability - 02/02/2016 - {"*STATREL_EV", link_lyr+".ABSTATREL_EV", link_lyr+".BASTATREL_EV"}, - {"*_TOTREL_EA", link_lyr+".AB_TOTREL_EA", link_lyr+".BA_TOTREL_EA"}, - {"*_TOTREL_AM", link_lyr+".AB_TOTREL_AM", link_lyr+".BA_TOTREL_AM"}, - {"*_TOTREL_MD", link_lyr+".AB_TOTREL_MD", link_lyr+".BA_TOTREL_MD"}, - {"*_TOTREL_PM", link_lyr+".AB_TOTREL_PM", link_lyr+".BA_TOTREL_PM"}, - {"*_TOTREL_EV", link_lyr+".AB_TOTREL_EV", link_lyr+".BA_TOTREL_EV"}} // added for reliability - 02/02/2016 - - // add two node fields into the network for turning movement purposes, by JXu - Opts.Global.[Node Options].ID = node_lyr + ".ID" - Opts.Global.[Node Options].DATA = node_lyr + ".temp" - Opts.Output.[Network File] = net_file - RunMacro("HwycadLog",{"createhwynet_turn.rsc: create hwynet1","Build Highway Network"}) - ok = RunMacro("TCB Run Operation", 1, "Build Highway Network", Opts) - if !ok then goto quit - - // STEP 2: Highway Network Setting - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Centroids Set] = {db_node_lyr, node_lyr, "Selection", "select * where ID <="+i2s(mxzone)} - Opts.Global.[Spc Turn Pen Method] = 3 - Opts.Input.[Def Turn Pen Table] = {d_tp_tb} - Opts.Input.[Spc Turn Pen Table] = {s_tp_tb} - Opts.Field.[Link type] = "IFC" - Opts.Global.[Global Turn Penalties] = {0, 0, 0, 0} - Opts.Flag.[Use Link Types] = "True" - RunMacro("HwycadLog",{"createhwynet_turn.rsc: create hwynet1","Highway Network Setting"}) - ok = RunMacro("TCB Run Operation", 2, "Highway Network Setting", Opts) - if !ok then goto quit - - mytime=GetDateAndTime() - - RunMacro("close all") //before delete db_file, close it - - - //writeline(fpr,mytime+", network setting") - ok=1 - quit: - //if fpr<>null then closefile(fpr) - return(ok) -endMacro -/************************************************************************************* -Calculate upstream and downsstream distances to major interchanges for freeway segments - -steps: -1. identify major interchange nodes -2. for each freeway segment, calculate upstream and downstream segment - -input: - output\\hwy.dbd - -output: - output\\MajorInterchangeDistance.csv - -by: nagendra.dhakar@rsginc.com -*************************************************************************************/ -Macro "DistanceToInterchange" - shared path, inputDir, outputDir - shared interchanges, freeways, linklayer, nodelayer - - ok=0 - - // input highway database - db_file=outputDir+"\\hwy.dbd" - hwy_dbd=db_file - - // output settings - out_file = outputDir+"\\MajorInterchangeDistance.csv" - - // add layers - layers = GetDBLayers(hwy_dbd) - linklayer=layers[2] - nodelayer=layers[1] - - db_linklayer=hwy_dbd+"|"+linklayer - db_nodelayer=hwy_dbd+"|"+nodelayer - - info = GetDBInfo(hwy_dbd) - temp_map = CreateMap("temp",{{"scope",info[1]}}) - - temp_layer = AddLayer(temp_map,linklayer,hwy_dbd,linklayer) - temp_layer = AddLayer(temp_map,nodelayer,hwy_dbd,nodelayer) - - // Identify Interchanges - SetLayer(linklayer) - - // Major interchange - query_ramps = 'Select * where (IFC=8) AND position(NM,"HOV")=0' // Major interchange - HOV access connectors are removed - MaxLinks = 50 - - on Error do ShowMessage("The SQL query: (" + query_ramps + ") is not correct.") end - VerifyQuery(query_ramps) - - nramps = SelectByQuery("Ramp Set", "Several", query_ramps,) - - query_freeways = "Select * where IFC=1" - on Error do ShowMessage("The SQL query: (" + query_freeways + ") is not correct.") end - VerifyQuery(query_freeways) - - nfreeways = SelectByQuery("Freeway Set", "Several",query_freeways,) - - // get freeway segments ids, IHOV, ANode BNode - freeways = GetDataVectors("Freeway Set", {"ID","IHOV"},) - - // intersect with nodes to select connected nodes - SetLayer(nodelayer) - nnodes = SelectByLinks("Interchange Set", "Several", "Ramp Set") - ninterchanges = SelectByLinks("Interchange Set", "Subset", "Freeway Set") - - // get interchange ids - interchanges = GetDataVector("Interchange Set","ID",) - - outfile = OpenFile(out_file, "w") - WriteLine(outfile, JoinStrings({"LinkID","Length","updistance","downdistance","ihov","UpLinks","DownLinks"},",")) - - on Error goto quit - - // loop through the freeway segments - dim upstreamlinkset[freeways[1].Length,MaxLinks-1], downstreamlinkset[freeways[1].Length,MaxLinks-1] - - CreateProgressBar("Calculating Interchange Distances", "True") - - for i=1 to freeways[1].Length do - perc=RealToInt(100*i/freeways[1].Length) - - UpdateProgressBar("Calculating Interchange Distances for LinkID: " +string(freeways[1][i]), perc) - - SetLayer(linklayer) - linkid = freeways[1][i] - ihov = freeways[2][i] - - query = "Select * where ID="+String(linkid) - count = SelectByQuery("Selection", "Several", query,) - linklength = GetDataVector("Selection", "Length",) - - nodes = GetEndPoints(linkid) - FromNode=nodes[1] - ToNode=nodes[2] - - // upstream - from node - isInterchange = RunMacro("NodeIsInterchange", FromNode) - - BaseLink = linkid - j=1 - - query1="n/a" - query2="n/a" - upstreamdistance=linklength[1]*0.5 - downstreamdistance=linklength[1]*0.5 - numupstreamlinks=0 - numdownstreamlinks=0 - - if (ihov<>2) then do - while (isInterchange=0) do - SetLayer(nodelayer) - links = GetNodeLinks(FromNode) // links set that meet at the node - - coordinates_base=GetCoordsFromLinks(linklayer, , {{BaseLink,1}}) - - // find the freeway link that is not the current link - upstreamlink = RunMacro("FindNextLink",links,BaseLink) - - if (upstreamlink <> null) then do - - coordinates_upstream=GetCoordsFromLinks(linklayer, , {{upstreamlink,1}}) - opposite = RunMacro("IsOppositeDirection",coordinates_base[1],coordinates_upstream[1]) // 1- true, 0- false - - if (opposite=0 and j1 and upstreamlink= linkid) then isInterchange=1 - else do - upstreamlinkset[i][j] = upstreamlink - j=j+1 - end - end - else do - isInterchange=1 - if (j=MaxLinks) then upstreamdistance=99 - end - end - else do - isInterchange=1 - if (j=MaxLinks) then upstreamdistance=99 - end - - end - - numupstreamlinks =j-1 - - // calculate upstream distance - SetLayer(linklayer) - - if j>=2 then do - query1 = "Select * where ID="+String(upstreamlinkset[i][1]) - - if j>2 then do - for iter=2 to numupstreamlinks do - query1 = JoinStrings({query1," or ID=",r2s(upstreamlinkset[i][iter])},"") - end - end - - count = SelectByQuery("Selection", "Several", query1,) - lengths = GetDataVector("Selection", "Length",) - upstreamdistance=VectorStatistic(lengths,"Sum",) - - // add half of the current link length to make the distance from midpoint - upstreamdistance = upstreamdistance + (linklength[1]*0.5) - end - - - // downstream - to node - isInterchange = RunMacro("NodeIsInterchange", ToNode) - - BaseLink = linkid - j=1 - while (isInterchange=0) do - SetLayer(nodelayer) - links = GetNodeLinks(ToNode) - - coordinates_base=GetCoordsFromLinks(linklayer, , {{BaseLink,-1}}) - - // find the freeway link that is not the current link - downstreamlink = RunMacro("FindNextLink",links,BaseLink) - - if (downstreamlink <> null) then do - coordinates_downstream=GetCoordsFromLinks(linklayer, , {{downstreamlink,-1}}) - opposite = RunMacro("IsOppositeDirection",coordinates_base[1],coordinates_downstream[1]) // 1- true, 0- false - - if (opposite=0 and j1 and downstreamlink=linkid) then isInterchange=1 - else do - downstreamlinkset[i][j] = downstreamlink - j=j+1 - end - - end - else do - isInterchange=1 - if (j=MaxLinks) then downstreamdistance=99 - end - end - else do - isInterchange=1 - if (j=MaxLinks) then downstreamdistance=99 - end - - end - numdownstreamlinks =j-1 - - // calculate downstream distance - SetLayer(linklayer) - - if j>=2 then do - query2 = "Select * where ID="+String(downstreamlinkset[i][1]) - - if j>2 then do - for iter=2 to numdownstreamlinks do - query2 = JoinStrings({query2," or ID=",r2s(downstreamlinkset[i][iter])},"") - end - end - - count = SelectByQuery("Selection", "Several", query2,) - lengths = GetDataVector("Selection", "Length",) - downstreamdistance=VectorStatistic(lengths,"Sum",) - - // add half of the current link length to make the distance from midpoint - downstreamdistance = downstreamdistance + (linklength[1]*0.5) - - end - - end - - else do - // HOV segments - set default value of 9999 miles. Distances are not calculated as HOV segments are pretty reliable. - upstreamdistance = 9999 - downstreamdistance = 9999 - numupstreamlinks = 9999 - numdownstreamlinks = 9999 - - end - - WriteLine(outfile, JoinStrings({i2s(linkid),r2s(linklength[1]),r2s(upstreamdistance),r2s(downstreamdistance),i2s(ihov),i2s(numupstreamlinks),i2s(numdownstreamlinks)},",")) - - end - - CloseFile(outfile) - - DestroyProgressBar() - ok=1 - return(ok) - - quit: - showmessage("Error, i: " + string(i) + ", j: " + string(j) + ", linkid: " + string(linkid)) - -EndMacro -/************************************************************************************* -Check if the nodes is an interchange: 0-No, 1- Yes -**************************************************************************************/ -Macro "NodeIsInterchange" (nodeid) - shared interchanges - - isInter = 0 - - for i=1 to interchanges.Length do - if nodeid=interchanges[i] then isInter = 1 - end - - return (isInter) - -EndMacro -/************************************************************************************* -Identify forward links: - Identify links that are not the previous link (linkid) - Selects the link that is also a freeway and assign that as the next link - Assumption: there is only one next freeway link -**************************************************************************************/ -Macro "FindNextLink" (linkset,linkid) - shared linklayer - - nextlink = null - - for i=1 to linkset.Length do - if linkid <> linkset[i] then do - IsFreeway = RunMacro("LinkIsFreeway",linkset[i]) - if (IsFreeway=1) then do - IsWrongDirection = RunMacro("WrongDirection",linkset[i],linkid) - IsHov = RunMacro("LinkIsHov",linkset[i]) - if (IsHov=0 & IsWrongDirection=0) then nextlink = linkset[i] - end - end - end - - return (nextlink) - -EndMacro -/************************************************************************************* -Check if the link is in the wrong direction -**************************************************************************************/ -Macro "WrongDirection" (link,linkid) - shared linklayer - - SetLayer(linklayer) - nodes1 = GetEndPoints(linkid) - nodes2 = GetEndPoints(link) - - tonode1 = nodes1[2] // base link - tonode2 = nodes2[2] // next link - - // compare ToNode - if they are same then wrong direction - if tonode1 = tonode2 then return(1) - else return(0) - -EndMacro - -/************************************************************************************* -Check if the links is a freeway link: 0-No, 1- Yes -**************************************************************************************/ -Macro "LinkIsFreeway" (link) - shared freeways - - freeway=0 - for j=1 to freeways[1].Length do - if link = freeways[1][j] then freeway=1 - end - - return (freeway) - -EndMacro -/************************************************************************************* -Check if link is a HOV segment -**************************************************************************************/ -Macro "LinkIsHov" (link) - shared freeways - - hov=0 - for j=1 to freeways[1].Length do - if link = freeways[1][j] then do - if freeways[2][j]=2 then hov=1 - end - end - - return(hov) - -EndMacro -/************************************************************************************* -Find link direction (coordinates as input) -**************************************************************************************/ -Macro "GetLinkDirection" (coordinates) - - maxnum = coordinates.length - - nodeA = coordinates[1] - nodeB = coordinates[maxnum] - - deltaX = nodeB.lon-nodeA.lon - deltaY = nodeB.lat-nodeA.lat - - if deltaY>0 then slope1 = deltaX/deltaY - else slope1 = deltaX - - if deltaX>0 then slope2 = deltaY/deltaX - else slope2 = deltaY - - direction = "" - if abs(slope1) > abs(slope2) then do - //pre_dir = "EW" - if deltaX<0 then direction = "WB" - else direction = "EB" - end - else do - //pre_dir = "NS" - if deltaY<0 then direction = "SB" - else direction = "NB" - end - - return(direction) - -EndMacro -/************************************************************************************* -Check if the two segments (coordinates as input) have opposite direction -**************************************************************************************/ -Macro "IsOppositeDirection" (coordinates1, coordinates2) - - direction1 = RunMacro("GetLinkDirection",coordinates1) - direction2 = RunMacro("GetLinkDirection",coordinates2) - - if direction1="NB" and direction2="SB" then return(1) - else if direction2="NB" and direction1="SB" then return(1) - else if direction1="EB" and direction2="WB" then return(1) - else if direction2="EB" and direction1="WB" then return(1) - else return(0) - -EndMacro diff --git a/sandag_abm/src/main/gisdk/createtodtables.rsc b/sandag_abm/src/main/gisdk/createtodtables.rsc deleted file mode 100644 index 677d87d..0000000 --- a/sandag_abm/src/main/gisdk/createtodtables.rsc +++ /dev/null @@ -1,916 +0,0 @@ -/************************************************************** - CreateAutoTables //modified "externalExternalTripsByYear.csv" input file and code, on 09/16/16, YMA - - Inputs - input\airportAutoTrips_XX.mtx - input\autoTrips_XX.mtx - input\extTrip_XX.mtx - - where XX is period = {_EA,_AM,_MD,_PM,_EV} - - input\commVehTODTrips.mtx - input\dailyDistributionMatricesTruckam.mtx - input\dailyDistributionMatricesTruckpm.mtx - input\dailyDistributionMatricesTruckop.mtx - - Outputs - output\Trip_XX.mtx - -*******************************************************************/ -Macro "Create Auto Tables" - - shared path, inputDir, outputDir - - periods={"_EA","_AM","_MD","_PM","_EV"} - - // read properties from sandag_abm.properties in /conf folder - properties = "\\conf\\sandag_abm.properties" - skipSpecialEventModel = RunMacro("read properties",properties,"RunModel.skipSpecialEventModel", "S") - //VOT bins for non resident models, 1->3 - votBinEE = 3 - votBinExternalInternal = 3 - votBinCommercialVehicles=3 - - /* - truckTables = { - inputDir+"\\dailyDistributionMatricesTruckam.mtx", - inputDir+"\\dailyDistributionMatricesTruckop.mtx", - inputDir+"\\dailyDistributionMatricesTruckpm.mtx" } - - truckPeriods = {2, 1, 2, 3, 2} - truckFactors = {0.1, 1.0, 0.65, 1.0, 0.25} - truckMatrices ={"lhdn","lhdt","mhdn","mhdt","hhdn","hhdt"} - */ - - truckTables = { - outputDir+"\\dailyDistributionMatricesTruckEA.mtx", - outputDir+"\\dailyDistributionMatricesTruckAM.mtx", - outputDir+"\\dailyDistributionMatricesTruckMD.mtx", - outputDir+"\\dailyDistributionMatricesTruckPM.mtx", - outputDir+"\\dailyDistributionMatricesTruckEV.mtx" } - - dim truckMatrices[truckTables.length] - dim truckCurrencies[truckTables.length] - - for i = 1 to truckTables.length do - // create truck matrix currencies - truckMatrices[i] = OpenMatrix(truckTables[i], ) - truckCurrencies[i] = CreateMatrixCurrencies(truckMatrices[i], , , ) - end - - externalInternalTables = { - outputDir+"\\usSdWrk", - outputDir+"\\usSdNon" - } - - //create external-external matrix from csv input file - ok = RunMacro("TCB Run Macro", 1,"Create External-External Trip Matrix",{}) - if !ok then goto quit - - //create external-external currencies - externalExternalMatrixName = outputDir + "\\externalExternalTrips.mtx" - externalExternalMatrix = OpenMatrix(externalExternalMatrixName, ) - - externalExternalCurrency = CreateMatrixCurrency(externalExternalMatrix,'Trips',,,) - externalExternalDiurnalFactors = { 0.074, 0.137, 0.472, 0.183, 0.133} - externalExternalOccupancyFactors = {0.43, 0.42, 0.15 } - - - internalExternalTables = { - { - outputDir+"\\autoInternalExternalTrips_EA_low.mtx", - outputDir+"\\autoInternalExternalTrips_AM_low.mtx", - outputDir+"\\autoInternalExternalTrips_MD_low.mtx", - outputDir+"\\autoInternalExternalTrips_PM_low.mtx", - outputDir+"\\autoInternalExternalTrips_EV_low.mtx"}, - { - outputDir+"\\autoInternalExternalTrips_EA_med.mtx", - outputDir+"\\autoInternalExternalTrips_AM_med.mtx", - outputDir+"\\autoInternalExternalTrips_MD_med.mtx", - outputDir+"\\autoInternalExternalTrips_PM_med.mtx", - outputDir+"\\autoInternalExternalTrips_EV_med.mtx" - }, - { - outputDir+"\\autoInternalExternalTrips_EA_high.mtx", - outputDir+"\\autoInternalExternalTrips_AM_high.mtx", - outputDir+"\\autoInternalExternalTrips_MD_high.mtx", - outputDir+"\\autoInternalExternalTrips_PM_high.mtx", - outputDir+"\\autoInternalExternalTrips_EV_high.mtx" - } - } - visitorTables = { - { - outputDir+"\\autoVisitorTrips_EA_low.mtx", - outputDir+"\\autoVisitorTrips_AM_low.mtx", - outputDir+"\\autoVisitorTrips_MD_low.mtx", - outputDir+"\\autoVisitorTrips_PM_low.mtx", - outputDir+"\\autoVisitorTrips_EV_low.mtx"}, - { - outputDir+"\\autoVisitorTrips_EA_med.mtx", - outputDir+"\\autoVisitorTrips_AM_med.mtx", - outputDir+"\\autoVisitorTrips_MD_med.mtx", - outputDir+"\\autoVisitorTrips_PM_med.mtx", - outputDir+"\\autoVisitorTrips_EV_med.mtx"}, - { - outputDir+"\\autoVisitorTrips_EA_high.mtx", - outputDir+"\\autoVisitorTrips_AM_high.mtx", - outputDir+"\\autoVisitorTrips_MD_high.mtx", - outputDir+"\\autoVisitorTrips_PM_high.mtx", - outputDir+"\\autoVisitorTrips_EV_high.mtx"} - } - crossBorderTables = { - { - outputDir+"\\autoCrossBorderTrips_EA_low.mtx", - outputDir+"\\autoCrossBorderTrips_AM_low.mtx", - outputDir+"\\autoCrossBorderTrips_MD_low.mtx", - outputDir+"\\autoCrossBorderTrips_PM_low.mtx", - outputDir+"\\autoCrossBorderTrips_EV_low.mtx"}, - { - outputDir+"\\autoCrossBorderTrips_EA_med.mtx", - outputDir+"\\autoCrossBorderTrips_AM_med.mtx", - outputDir+"\\autoCrossBorderTrips_MD_med.mtx", - outputDir+"\\autoCrossBorderTrips_PM_med.mtx", - outputDir+"\\autoCrossBorderTrips_EV_med.mtx"}, - { - outputDir+"\\autoCrossBorderTrips_EA_high.mtx", - outputDir+"\\autoCrossBorderTrips_AM_high.mtx", - outputDir+"\\autoCrossBorderTrips_MD_high.mtx", - outputDir+"\\autoCrossBorderTrips_PM_high.mtx", - outputDir+"\\autoCrossBorderTrips_EV_high.mtx"} - } - - airportTables = { - { - outputDir+"\\autoAirportTrips_EA_low.mtx", - outputDir+"\\autoAirportTrips_AM_low.mtx", - outputDir+"\\autoAirportTrips_MD_low.mtx", - outputDir+"\\autoAirportTrips_PM_low.mtx", - outputDir+"\\autoAirportTrips_EV_low.mtx"}, - { - outputDir+"\\autoAirportTrips_EA_med.mtx", - outputDir+"\\autoAirportTrips_AM_med.mtx", - outputDir+"\\autoAirportTrips_MD_med.mtx", - outputDir+"\\autoAirportTrips_PM_med.mtx", - outputDir+"\\autoAirportTrips_EV_med.mtx"}, - { - outputDir+"\\autoAirportTrips_EA_high.mtx", - outputDir+"\\autoAirportTrips_AM_high.mtx", - outputDir+"\\autoAirportTrips_MD_high.mtx", - outputDir+"\\autoAirportTrips_PM_high.mtx", - outputDir+"\\autoAirportTrips_EV_high.mtx"} - } - personTables = { - { - outputDir+"\\autoTrips_EA_low.mtx", - outputDir+"\\autoTrips_AM_low.mtx", - outputDir+"\\autoTrips_MD_low.mtx", - outputDir+"\\autoTrips_PM_low.mtx", - outputDir+"\\autoTrips_EV_low.mtx"}, - { - outputDir+"\\autoTrips_EA_med.mtx", - outputDir+"\\autoTrips_AM_med.mtx", - outputDir+"\\autoTrips_MD_med.mtx", - outputDir+"\\autoTrips_PM_med.mtx", - outputDir+"\\autoTrips_EV_med.mtx"}, - { - outputDir+"\\autoTrips_EA_high.mtx", - outputDir+"\\autoTrips_AM_high.mtx", - outputDir+"\\autoTrips_MD_high.mtx", - outputDir+"\\autoTrips_PM_high.mtx", - outputDir+"\\autoTrips_EV_high.mtx"} - } - //the following table names have the period appended - CTRampMatrices = {"SOV_GP","SOV_PAY","SR2_GP","SR2_HOV","SR2_PAY","SR3_GP","SR3_HOV","SR3_PAY"} - - //output files - outMatrixNames = {"Trip"+periods[1]+"_VOT.mtx", "Trip"+periods[2]+"_VOT.mtx", "Trip"+periods[3]+"_VOT.mtx", "Trip"+periods[4]+"_VOT.mtx", "Trip"+periods[5]+"_VOT.mtx"} - outTableNames = {"SOV_GP_LOW", "SOV_PAY_LOW", "SR2_GP_LOW","SR2_HOV_LOW", "SR2_PAY_LOW", "SR3_GP_LOW","SR3_HOV_LOW","SR3_PAY_LOW", - "SOV_GP_MED", "SOV_PAY_MED", "SR2_GP_MED","SR2_HOV_MED", "SR2_PAY_MED", "SR3_GP_MED","SR3_HOV_MED","SR3_PAY_MED", - "SOV_GP_HIGH", "SOV_PAY_HIGH", "SR2_GP_HIGH","SR2_HOV_HIGH", "SR2_PAY_HIGH", "SR3_GP_HIGH","SR3_HOV_HIGH","SR3_PAY_HIGH", - "lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} - - commVehTable = outputDir+"\\commVehTODTrips.mtx" - commVehMatrices = {"EA NonToll","AM NonToll","MD NonToll","PM NonToll","EV NonToll","EA Toll","AM Toll","MD Toll","PM Toll","EV Toll"} - - // create comm vehicle matrix currencies - commVehMatrix = OpenMatrix(commVehTable, ) - commVehCurrencies = CreateMatrixCurrencies(commVehMatrix, , , ) - - - for i = 1 to periods.length do - - //open person trip matrix currencies - personMatrixLow = OpenMatrix(personTables[1][i], ) - personCurrenciesLow = CreateMatrixCurrencies(personMatrixLow, , , ) - - personMatrixMed = OpenMatrix(personTables[2][i], ) - personCurrenciesMed = CreateMatrixCurrencies(personMatrixMed, , , ) - - personMatrixHigh = OpenMatrix(personTables[3][i], ) - personCurrenciesHigh = CreateMatrixCurrencies(personMatrixHigh, , , ) - - totalOutMatrices = CTRampMatrices.length * 3 - - counter = 0 - dim curr_array[totalOutMatrices] - for j = 1 to totalOutMatrices do - counter = counter + 1 - curr_array[j] = CreateMatrixCurrency(personMatrixLow, CTRampMatrices[counter]+periods[i], ,, ) - if counter = CTRampMatrices.length then counter=0 - - end - - //open internal-external matrix currencies - internalExternalMatrixLow = OpenMatrix(internalExternalTables[1][i], ) - internalExternalCurrenciesLow = CreateMatrixCurrencies(internalExternalMatrixLow, , , ) - - internalExternalMatrixMed = OpenMatrix(internalExternalTables[2][i], ) - internalExternalCurrenciesMed = CreateMatrixCurrencies(internalExternalMatrixMed, , , ) - - internalExternalMatrixHigh = OpenMatrix(internalExternalTables[3][i], ) - internalExternalCurrenciesHigh = CreateMatrixCurrencies(internalExternalMatrixHigh, , , ) - - //open airport matrix currencies - airportMatrixLow = OpenMatrix(airportTables[1][i], ) - airportCurrenciesLow = CreateMatrixCurrencies(airportMatrixLow, , , ) - - airportMatrixMed = OpenMatrix(airportTables[2][i], ) - airportCurrenciesMed = CreateMatrixCurrencies(airportMatrixMed, , , ) - - airportMatrixHigh = OpenMatrix(airportTables[3][i], ) - airportCurrenciesHigh = CreateMatrixCurrencies(airportMatrixHigh, , , ) - - //open visitor matrix currencies - visitorMatrixLow = OpenMatrix(visitorTables[1][i], ) - visitorCurrenciesLow = CreateMatrixCurrencies(visitorMatrixLow, , , ) - - visitorMatrixMed = OpenMatrix(visitorTables[2][i], ) - visitorCurrenciesMed = CreateMatrixCurrencies(visitorMatrixMed, , , ) - - visitorMatrixHigh = OpenMatrix(visitorTables[3][i], ) - visitorCurrenciesHigh = CreateMatrixCurrencies(visitorMatrixHigh, , , ) - - //open cross border matrix currencies - crossBorderMatrixLow = OpenMatrix(crossBorderTables[1][i], ) - crossBorderCurrenciesLow = CreateMatrixCurrencies(crossBorderMatrixLow, , , ) - - crossBorderMatrixMed = OpenMatrix(crossBorderTables[2][i], ) - crossBorderCurrenciesMed = CreateMatrixCurrencies(crossBorderMatrixMed, , , ) - - crossBorderMatrixHigh = OpenMatrix(crossBorderTables[3][i], ) - crossBorderCurrenciesHigh = CreateMatrixCurrencies(crossBorderMatrixHigh, , , ) - - - //open external-internal work matrix currencies - externalInternalWrkMatrix = OpenMatrix(externalInternalTables[1]+periods[i]+".mtx", ) - externalInternalWrkCurrencies = CreateMatrixCurrencies(externalInternalWrkMatrix, , , ) - - //open external-internal non-work matrix currencies - externalInternalNonMatrix = OpenMatrix(externalInternalTables[2]+periods[i]+".mtx", ) - externalInternalNonCurrencies = CreateMatrixCurrencies(externalInternalNonMatrix, , , ) - - //open special event matrix currencies; Wu added 5/16/2017 - specialEventMatrix = OpenMatrix(specialEventables[i], ) - specialEventCurrencies = CreateMatrixCurrencies(specialEventMatrix, , , ) - - - //create output trip table and matrix currencies for this time period - outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outputDir+"\\"+outMatrixNames[i]}, - {"Label", outMatrixNames[i]}, - {"Tables",outTableNames}, - {"File Based", "Yes"}}) - SetMatrixCoreNames(outMatrix, outTableNames) - - outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) - -//LOW VOT - // calculate output matrices - outCurrencies.SOV_GP_LOW :=Nz(outCurrencies.SOV_GP_LOW) - outCurrencies.SOV_GP_LOW := Nz(personCurrenciesLow.("SOV_GP"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SOV_GP"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SOV_GP"+periods[i])) - + Nz(airportCurrenciesLow.("SOV_GP"+periods[i])) - + Nz(visitorCurrenciesLow.("SOV_GP"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) - if votBinExternalExternal=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) - if votBinCommercialVehicles=1 then outCurrencies.SOV_GP_LOW := outCurrencies.SOV_GP_LOW + Nz(commVehCurrencies.(commVehMatrices[i])) - - outCurrencies.SOV_PAY_LOW :=Nz(outCurrencies.SOV_PAY_LOW) - outCurrencies.SOV_PAY_LOW := Nz(personCurrenciesLow.("SOV_PAY"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SOV_PAY"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SOV_PAY"+periods[i])) - + Nz(airportCurrenciesLow.("SOV_PAY"+periods[i])) - + Nz(visitorCurrenciesLow.("SOV_PAY"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SOV_PAY_LOW := outCurrencies.SOV_PAY_LOW + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) - if votBinCommercialVehicles=1 then outCurrencies.SOV_PAY_LOW := outCurrencies.SOV_PAY_LOW + Nz(commVehCurrencies.(commVehMatrices[i+5])) - - outCurrencies.SR2_GP_LOW :=Nz(outCurrencies.SR2_GP_LOW) - outCurrencies.SR2_GP_LOW := Nz(personCurrenciesLow.("SR2_GP"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR2_GP"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR2_GP"+periods[i])) - + Nz(airportCurrenciesLow.("SR2_GP"+periods[i])) - + Nz(visitorCurrenciesLow.("SR2_GP"+periods[i])) - - outCurrencies.SR2_HOV_LOW :=Nz(outCurrencies.SR2_HOV_LOW) - outCurrencies.SR2_HOV_LOW := Nz(personCurrenciesLow.("SR2_HOV"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR2_HOV"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR2_HOV"+periods[i])) - + Nz(airportCurrenciesLow.("SR2_HOV"+periods[i])) - + Nz(visitorCurrenciesLow.("SR2_HOV"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SR2_HOV_LOW := outCurrencies.SR2_HOV_LOW + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) - if votBinExternalExternal=1 then outCurrencies.SR2_HOV_LOW := outCurrencies.SR2_HOV_LOW + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] - - outCurrencies.SR2_PAY_LOW :=Nz(outCurrencies.SR2_PAY_LOW) - outCurrencies.SR2_PAY_LOW := Nz(personCurrenciesLow.("SR2_PAY"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR2_PAY"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR2_PAY"+periods[i])) - + Nz(airportCurrenciesLow.("SR2_PAY"+periods[i])) - + Nz(visitorCurrenciesLow.("SR2_PAY"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SR2_PAY_LOW := outCurrencies.SR2_PAY_LOW + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) - - outCurrencies.SR3_GP_LOW :=Nz(outCurrencies.SR3_GP_LOW) - outCurrencies.SR3_GP_LOW := Nz(personCurrenciesLow.("SR3_GP"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR3_GP"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR3_GP"+periods[i])) - + Nz(airportCurrenciesLow.("SR3_GP"+periods[i])) - + Nz(visitorCurrenciesLow.("SR3_GP"+periods[i])) - - outCurrencies.SR3_HOV_LOW :=Nz(outCurrencies.SR3_HOV_LOW) - outCurrencies.SR3_HOV_LOW := Nz(personCurrenciesLow.("SR3_HOV"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR3_HOV"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR3_HOV"+periods[i])) - + Nz(airportCurrenciesLow.("SR3_HOV"+periods[i])) - + Nz(visitorCurrenciesLow.("SR3_HOV"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SR3_HOV_LOW := outCurrencies.SR3_HOV_LOW + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) - if votBinExternalExternal=1 then outCurrencies.SR3_HOV_LOW := outCurrencies.SR3_HOV_LOW + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] - - outCurrencies.SR3_PAY_LOW :=Nz(outCurrencies.SR3_PAY_LOW) - outCurrencies.SR3_PAY_LOW := Nz(personCurrenciesLow.("SR3_PAY"+periods[i])) - + Nz(internalExternalCurrenciesLow.("SR3_PAY"+periods[i])) - + Nz(crossBorderCurrenciesLow.("SR3_PAY"+periods[i])) - + Nz(airportCurrenciesLow.("SR3_PAY"+periods[i])) - + Nz(visitorCurrenciesLow.("SR3_PAY"+periods[i])) - - if votBinExternalInternal=1 then outCurrencies.SR3_PAY_LOW := outCurrencies.SR3_PAY_LOW + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) - -//MED VOT - // calculate output matrices - outCurrencies.SOV_GP_MED :=Nz(outCurrencies.SOV_GP_MED) - outCurrencies.SOV_GP_MED := Nz(personCurrenciesMed.("SOV_GP"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SOV_GP"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SOV_GP"+periods[i])) - + Nz(airportCurrenciesMed.("SOV_GP"+periods[i])) - + Nz(visitorCurrenciesMed.("SOV_GP"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) - if votBinExternalExternal=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) - if votBinCommercialVehicles=2 then outCurrencies.SOV_GP_MED := outCurrencies.SOV_GP_MED + Nz(commVehCurrencies.(commVehMatrices[i])) - - outCurrencies.SOV_PAY_MED :=Nz(outCurrencies.SOV_PAY_MED) - outCurrencies.SOV_PAY_MED := Nz(personCurrenciesMed.("SOV_PAY"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SOV_PAY"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SOV_PAY"+periods[i])) - + Nz(airportCurrenciesMed.("SOV_PAY"+periods[i])) - + Nz(visitorCurrenciesMed.("SOV_PAY"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SOV_PAY_MED := outCurrencies.SOV_PAY_MED + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) - if votBinCommercialVehicles=2 then outCurrencies.SOV_PAY_MED := outCurrencies.SOV_PAY_MED + Nz(commVehCurrencies.(commVehMatrices[i+5])) - - outCurrencies.SR2_GP_MED :=Nz(outCurrencies.SR2_GP_MED) - outCurrencies.SR2_GP_MED := Nz(personCurrenciesMed.("SR2_GP"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR2_GP"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR2_GP"+periods[i])) - + Nz(airportCurrenciesMed.("SR2_GP"+periods[i])) - + Nz(visitorCurrenciesMed.("SR2_GP"+periods[i])) - - outCurrencies.SR2_HOV_MED :=Nz(outCurrencies.SR2_HOV_MED) - outCurrencies.SR2_HOV_MED := Nz(personCurrenciesMed.("SR2_HOV"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR2_HOV"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR2_HOV"+periods[i])) - + Nz(airportCurrenciesMed.("SR2_HOV"+periods[i])) - + Nz(visitorCurrenciesMed.("SR2_HOV"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SR2_HOV_MED := outCurrencies.SR2_HOV_MED + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) - if votBinExternalExternal=2 then outCurrencies.SR2_HOV_MED := outCurrencies.SR2_HOV_MED + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] - - outCurrencies.SR2_PAY_MED :=Nz(outCurrencies.SR2_PAY_MED) - outCurrencies.SR2_PAY_MED := Nz(personCurrenciesMed.("SR2_PAY"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR2_PAY"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR2_PAY"+periods[i])) - + Nz(airportCurrenciesMed.("SR2_PAY"+periods[i])) - + Nz(visitorCurrenciesMed.("SR2_PAY"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SR2_PAY_MED := outCurrencies.SR2_PAY_MED + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) - - outCurrencies.SR3_GP_MED :=Nz(outCurrencies.SR3_GP_MED) - outCurrencies.SR3_GP_MED := Nz(personCurrenciesMed.("SR3_GP"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR3_GP"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR3_GP"+periods[i])) - + Nz(airportCurrenciesMed.("SR3_GP"+periods[i])) - + Nz(visitorCurrenciesMed.("SR3_GP"+periods[i])) - - outCurrencies.SR3_HOV_MED :=Nz(outCurrencies.SR3_HOV_MED) - outCurrencies.SR3_HOV_MED := Nz(personCurrenciesMed.("SR3_HOV"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR3_HOV"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR3_HOV"+periods[i])) - + Nz(airportCurrenciesMed.("SR3_HOV"+periods[i])) - + Nz(visitorCurrenciesMed.("SR3_HOV"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SR3_HOV_MED := outCurrencies.SR3_HOV_MED + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) - if votBinExternalExternal=2 then outCurrencies.SR3_HOV_MED := outCurrencies.SR3_HOV_MED + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] - - outCurrencies.SR3_PAY_MED :=Nz(outCurrencies.SR3_PAY_MED) - outCurrencies.SR3_PAY_MED := Nz(personCurrenciesMed.("SR3_PAY"+periods[i])) - + Nz(internalExternalCurrenciesMed.("SR3_PAY"+periods[i])) - + Nz(crossBorderCurrenciesMed.("SR3_PAY"+periods[i])) - + Nz(airportCurrenciesMed.("SR3_PAY"+periods[i])) - + Nz(visitorCurrenciesMed.("SR3_PAY"+periods[i])) - - if votBinExternalInternal=2 then outCurrencies.SR3_PAY_MED := outCurrencies.SR3_PAY_MED + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) - -//HIGH VOT - // calculate output matrices - outCurrencies.SOV_GP_HIGH :=Nz(outCurrencies.SOV_GP_HIGH) - outCurrencies.SOV_GP_HIGH := Nz(personCurrenciesHigh.("SOV_GP"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SOV_GP"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SOV_GP"+periods[i])) - + Nz(airportCurrenciesHigh.("SOV_GP"+periods[i])) - + Nz(visitorCurrenciesHigh.("SOV_GP"+periods[i])) - - if votBinExternalInternal=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + Nz(externalInternalWrkCurrencies.("DAN")) + Nz(externalInternalNonCurrencies.("DAN")) - if votBinExternalExternal=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + (Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[1]) - if votBinCommercialVehicles=3 then outCurrencies.SOV_GP_HIGH := outCurrencies.SOV_GP_HIGH + Nz(commVehCurrencies.(commVehMatrices[i])) - - outCurrencies.SOV_PAY_HIGH :=Nz(outCurrencies.SOV_PAY_HIGH) - outCurrencies.SOV_PAY_HIGH := Nz(personCurrenciesHigh.("SOV_PAY"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SOV_PAY"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SOV_PAY"+periods[i])) - + Nz(airportCurrenciesHigh.("SOV_PAY"+periods[i])) - + Nz(visitorCurrenciesHigh.("SOV_PAY"+periods[i])) - - if votBinExternalInternal=3 then outCurrencies.SOV_PAY_HIGH := outCurrencies.SOV_PAY_HIGH + Nz(externalInternalWrkCurrencies.("DAT")) + Nz(externalInternalNonCurrencies.("DAT")) - if votBinCommercialVehicles=3 then outCurrencies.SOV_PAY_HIGH := outCurrencies.SOV_PAY_HIGH + Nz(commVehCurrencies.(commVehMatrices[i+5])) - - outCurrencies.SR2_GP_HIGH :=Nz(outCurrencies.SR2_GP_HIGH) - outCurrencies.SR2_GP_HIGH := Nz(personCurrenciesHigh.("SR2_GP"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR2_GP"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR2_GP"+periods[i])) - + Nz(airportCurrenciesHigh.("SR2_GP"+periods[i])) - + Nz(visitorCurrenciesHigh.("SR2_GP"+periods[i])) - - outCurrencies.SR2_HOV_HIGH :=Nz(outCurrencies.SR2_HOV_HIGH) - outCurrencies.SR2_HOV_HIGH := Nz(personCurrenciesHigh.("SR2_HOV"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR2_HOV"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR2_HOV"+periods[i])) - + Nz(airportCurrenciesHigh.("SR2_HOV"+periods[i])) - + Nz(visitorCurrenciesHigh.("SR2_HOV"+periods[i])) - - if votBinExternalInternal=3 then outCurrencies.SR2_HOV_HIGH := outCurrencies.SR2_HOV_HIGH + Nz(externalInternalWrkCurrencies.("S2N")) + Nz(externalInternalNonCurrencies.("S2N")) - if votBinExternalExternal=3 then outCurrencies.SR2_HOV_HIGH := outCurrencies.SR2_HOV_HIGH + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[2] - - outCurrencies.SR2_PAY_HIGH :=Nz(outCurrencies.SR2_PAY_HIGH) - outCurrencies.SR2_PAY_HIGH := Nz(personCurrenciesHigh.("SR2_PAY"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR2_PAY"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR2_PAY"+periods[i])) - + Nz(airportCurrenciesHigh.("SR2_PAY"+periods[i])) - + Nz(visitorCurrenciesHigh.("SR2_PAY"+periods[i])) - - if votBinExternalInternal=3 then outCurrencies.SR2_PAY_HIGH := outCurrencies.SR2_PAY_HIGH + Nz(externalInternalWrkCurrencies.("S2T")) + Nz(externalInternalNonCurrencies.("S2T")) - - outCurrencies.SR3_GP_HIGH :=Nz(outCurrencies.SR3_GP_HIGH) - outCurrencies.SR3_GP_HIGH := Nz(personCurrenciesHigh.("SR3_GP"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR3_GP"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR3_GP"+periods[i])) - + Nz(airportCurrencies.("SR3_GP"+periods[i])) - + Nz(visitorCurrencies.("SR3_GP"+periods[i])) - - outCurrencies.SR3_HOV_HIGH :=Nz(outCurrencies.SR3_HOV_HIGH) - outCurrencies.SR3_HOV_HIGH := Nz(personCurrenciesHigh.("SR3_HOV"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR3_HOV"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR3_HOV"+periods[i])) - + Nz(airportCurrenciesHigh.("SR3_HOV"+periods[i])) - + Nz(visitorCurrenciesHigh.("SR3_HOV"+periods[i])) - - - if votBinExternalInternal=3 then outCurrencies.SR3_HOV_HIGH := outCurrencies.SR3_HOV_HIGH + Nz(externalInternalWrkCurrencies.("S3N")) + Nz(externalInternalNonCurrencies.("S3N")) - if votBinExternalExternal=3 then outCurrencies.SR3_HOV_HIGH := outCurrencies.SR3_HOV_HIGH + Nz(externalExternalCurrency) * externalExternalDiurnalFactors[i] * externalExternalOccupancyFactors[3] - - outCurrencies.SR3_PAY_HIGH :=Nz(outCurrencies.SR3_PAY_HIGH) - outCurrencies.SR3_PAY_HIGH := Nz(personCurrenciesHigh.("SR3_PAY"+periods[i])) - + Nz(internalExternalCurrenciesHigh.("SR3_PAY"+periods[i])) - + Nz(crossBorderCurrenciesHigh.("SR3_PAY"+periods[i])) - + Nz(airportCurrenciesHigh.("SR3_PAY"+periods[i])) - + Nz(visitorCurrenciesHigh.("SR3_PAY"+periods[i])) - - if votBinExternalInternal=3 then outCurrencies.SR3_PAY_HIGH := outCurrencies.SR3_PAY_HIGH + Nz(externalInternalWrkCurrencies.("S3T")) + Nz(externalInternalNonCurrencies.("S3T")) - - outCurrencies.lhdn := truckCurrencies[i].lhdn - outCurrencies.mhdn := truckCurrencies[i].mhdn - outCurrencies.hhdn := truckCurrencies[i].hhdn - outCurrencies.lhdt := truckCurrencies[i].lhdt - outCurrencies.mhdt := truckCurrencies[i].mhdt - outCurrencies.hhdt := truckCurrencies[i].hhdt - end - RunMacro("close all" ) - quit: - Return(1 ) - -EndMacro - - -/************************************************************** - CreateTransitTables - - Inputs - input\tranTrips_XX.mtx - input\tranAirportTrips_XX.mtx - - where XX is period = {_EA,_AM,_MD,_PM,_EV} - - - Outputs - output\tranTotalTrips_XX.mtx - - TODO: Add Mexican resident and visitor trips - -*******************************************************************/ -Macro "Create Transit Tables" - - shared path, inputDir, outputDir - - periods={"_EA","_AM","_MD","_PM","_EV"} - - dim personFiles[periods.length] - dim airportFiles[periods.length] - dim crossBorderFiles[periods.length] - dim visitorFiles[periods.length] - dim internalExternalFiles[periods.length] - dim outFiles[periods.length] - - - for i = 1 to periods.length do - personFiles[i] = outputDir+"\\tranTrips"+periods[i]+".mtx" - airportFiles[i] = outputDir+"\\tranAirportTrips"+periods[i]+".mtx" - crossBorderFiles[i] = outputDir+"\\tranCrossBorderTrips"+periods[i]+".mtx" - visitorFiles[i] = outputDir+"\\tranVisitorTrips"+periods[i]+".mtx" - internalExternalFiles[i] = outputDir+"\\tranInternalExternalTrips"+periods[i]+".mtx" - outFiles[i] = outputDir+"\\tranTotalTrips"+periods[i]+".mtx" - end - - - //the following table names have the period appended - tableNames = {"WLK_LOC","WLK_EXP","WLK_BRT","WLK_LRT","WLK_CMR","PNR_LOC","PNR_EXP","PNR_BRT","PNR_LRT","PNR_CMR","KNR_LOC","KNR_EXP","KNR_BRT","KNR_LRT","KNR_CMR"} - - - segments = {"CTRAMP","Airport","Visitor","CrossBorder","Int-Ext","Total"} - numberSegments = segments.length - dim statistics[numberSegments] - dim totalTrips[numberSegments,tableNames.length] - - for i = 1 to periods.length do - - //open person trip matrix currencies - personMatrix = OpenMatrix(personFiles[i], ) - personCurrencies = CreateMatrixCurrencies(personMatrix, , , ) - - //open airport matrix currencies - airportMatrix = OpenMatrix(airportFiles[i], ) - airportCurrencies = CreateMatrixCurrencies(airportMatrix, , , ) - - //open visitor matrix currencies - visitorMatrix = OpenMatrix(visitorFiles[i], ) - visitorCurrencies = CreateMatrixCurrencies(visitorMatrix, , , ) - - //open crossBorder matrix currencies - crossBorderMatrix = OpenMatrix(crossBorderFiles[i], ) - crossBorderCurrencies = CreateMatrixCurrencies(crossBorderMatrix, , , ) - - //open internalExternal matrix currencies - internalExternalMatrix = OpenMatrix(internalExternalFiles[i], ) - internalExternalCurrencies = CreateMatrixCurrencies(internalExternalMatrix, , , ) - - - dim curr_array[tableNames.length] - for j = 1 to curr_array.length do - curr_array[j] = CreateMatrixCurrency(personMatrix, tableNames[j]+periods[i], ,, ) - end - - //create output trip table and matrix currencies for this time period - outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outFiles[i]}, - {"Label", outFiles[i]}, - {"Tables",tableNames}, - {"Compression",0}, - {"File Based", "Yes"}}) - SetMatrixCoreNames(outMatrix, tableNames) - - outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) - - - // calculate output matrices - for j = 1 to tableNames.length do - - outCurrencies.(tableNames[j]) := personCurrencies.(tableNames[j]+periods[i]) - + airportCurrencies.(tableNames[j]+periods[i]) - + visitorCurrencies.(tableNames[j]+periods[i]) - + crossBorderCurrencies.(tableNames[j]+periods[i]) - + internalExternalCurrencies.(tableNames[j]+periods[i]) - - end - - //for reporting - statistics[1] = MatrixStatistics(personMatrix,) - statistics[2] = MatrixStatistics(airportMatrix,) - statistics[3] = MatrixStatistics(visitorMatrix,) - statistics[4] = MatrixStatistics(crossBorderMatrix,) - statistics[5] = MatrixStatistics(internalExternalMatrix,) - statistics[6] = MatrixStatistics(outMatrix,) - - // calculate totals and save in arrays - for j = 1 to tableNames.length do - - for k = 1 to numberSegments do - - // Sum the tables in the person trip file - - if k<6 then totalTrips[k][j] = statistics[k].(tableNames[j]+periods[i]).Sum else totalTrips[k][j] = statistics[k].(tableNames[j]).Sum - end - end - - //write the table for inputs to the report file - AppendToReportFile(0, "Transit Factoring for Period "+periods[i], {{"Section", "True"}}) - fileColumn = { {"Name", "File"}, {"Percentage Width", 20}, {"Alignment", "Left"}} - modeColumns = null - for j = 1 to tableNames.length do - modeColumns = modeColumns + { { {"Name", tableNames[j]}, {"Percentage Width", (100-20)/tableNames.length}, {"Alignment", "Left"}, {"Decimals", 0} } } - end - columns = {fileColumn} + modeColumns - AppendTableToReportFile( columns, {{"Title", "Transit Factor Input File Totals"}}) - - for j = 1 to numberSegments do - outRow = null - for k = 1 to tableNames.length do - outRow = outRow + {totalTrips[j][k] } - end - outRow = { segments[j] } + outRow - AppendRowToReportFile(outRow,) - end - - CloseReportFileSection() - - - - end - RunMacro("close all" ) - quit: - Return(1 ) - -EndMacro -/************************************************************** -Create TOD Tables From 4Step Model - - TransCAD Macro used to create trip tables from 4-step model - for assignment to 5 tod networks for first iteration of AB - model. Only needs to be run once for any given scenario year. - - Inputs - input\trptollam2.mtx - input\trptollop2.mtx - input\trptollpm2.mtx - - Outputs - output\Trip_EA.mtx - output\Trip_AM.mtx - output\Trip_MD.mtx - output\Trip_PM.mtx - output\Trip_EV.mtx - -Each matrix has the following cores: - SOV_GP SOV General purpose lanes - SOV_PAY SOV Toll eligible - SR2_GP SR2 General purpose lanes - SR2_HOV SR2 HOV lanes - SR2_PAY SR2 Toll eligible - SR3_GP SR3 General purpose lanes - SR3_HOV SR3 HOV lanes - SR3_PAY SR3 Toll eligible - lhdn Light heavy duty general purpose lanes - mhdn Medium heavy duty general purpose lanes - hhdn Heavy heavy duty general purpose lanes - lhdt Light heavy duty toll eligibl - mhdt Medium heavy duty general purpose lanes - hhdt Heavy heavy duty general purpose lanes - -**************************************************************/ -Macro "Create TOD Tables From 4Step Model" - - shared path, inputDir, outputDir - - /* - inputDir = "c:\\projects\\sandag\\series12\\base2008\\input" - outputDir = "c:\\projects\\sandag\\series12\\base2008\\output" - */ - - //inputs - amMatrixName = "trptollam2.mtx" - opMatrixName = "trptollop2.mtx" - pmMatrixName = "trptollpm2.mtx" - - inTableNames = {"dan", "dat", "s2nn", "s2nh", "s2th", "M1", "M2", "M3", "lhdn","mhdn","hhdn", "lhdt","mhdt","hhdt"} - - - //output files - outMatrixNames = {"Trip_EA.mtx", "Trip_AM.mtx", "Trip_MD.mtx", "Trip_PM.mtx", "Trip_EV.mtx"} - outTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} - - // open input matrices - amMatrix = OpenMatrix(inputDir+"\\"+amMatrixName, ) - opMatrix = OpenMatrix(inputDir+"\\"+opMatrixName, ) - pmMatrix = OpenMatrix(inputDir+"\\"+pmMatrixName, ) - - // create input matrix currencies - dim inCurrencies[outMatrixNames.length,inTableNames.length] - for i = 1 to inTableNames.length do - inCurrencies[1][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - inCurrencies[2][i] = CreateMatrixCurrency(amMatrix, inTableNames[i], ,, ) - inCurrencies[3][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - inCurrencies[4][i] = CreateMatrixCurrency(pmMatrix, inTableNames[i], ,, ) - inCurrencies[5][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - end - - - // create the output matrices, copying the matrix structure of the input matrices. Then rename the matrices. - dim outMatrices[outMatrixNames.length] - dim outCurrencies[outMatrixNames.length,outTableNames.length] - for i = 1 to outMatrixNames.length do - - outMatrices[i] = CopyMatrixStructure(inCurrencies[i], {{"File Name", outputDir+"\\"+outMatrixNames[i]}, - {"Label", outMatrixNames[i]}, - {"Tables",outTableNames}, - {"File Based", "Yes"}}) - SetMatrixCoreNames(outMatrices[i], outTableNames) - - for j = 1 to outTableNames.length do - outCurrencies[i][j] = CreateMatrixCurrency(outMatrices[i], outTableNames[j], ,, ) - end - end - - // factor the off-peak input table to 3 time periods (1=EA,3=MD,5=PM, and the factors are generic across input tables) - for i = 1 to outTableNames.length do - outCurrencies[1][i] := inCurrencies[1][i] * 0.10 - outCurrencies[2][i] := inCurrencies[2][i] - outCurrencies[3][i] := inCurrencies[3][i] * 0.65 - outCurrencies[4][i] := inCurrencies[4][i] - outCurrencies[5][i] := inCurrencies[5][i] * 0.25 - end - - Return(1) - - - -EndMacro - - -/*************************************************************************************************************************** - - - -****************************************************************************************************************************/ -Macro "Create EE & EI Trips" - - - shared path, inputDir, outputDir, inputTruckDir, mxzone, mxtap, mxext,mxlink,mxrte - - //inputs - amMatrixName = "trptollam2.mtx" - opMatrixName = "trptollop2.mtx" - pmMatrixName = "trptollpm2.mtx" - - inTableNames = {"dan", "dat", "s2nn", "s2nh", "s2th", "M1", "M2", "M3", "lhdn","mhdn","hhdn", "lhdt","mhdt","hhdt"} - - //output files - outMatrixNames = {"ExtTrip_EA.mtx", "ExtTrip_AM.mtx", "ExtTrip_MD.mtx", "ExtTrip_PM.mtx", "ExtTrip_EV.mtx"} - outTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} - - // open input matrices - amMatrix = OpenMatrix(inputDir+"\\"+amMatrixName, ) - opMatrix = OpenMatrix(inputDir+"\\"+opMatrixName, ) - pmMatrix = OpenMatrix(inputDir+"\\"+pmMatrixName, ) - - // create input matrix currencies - dim inCurrencies[outMatrixNames.length,inTableNames.length] - for i = 1 to inTableNames.length do - inCurrencies[1][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - inCurrencies[2][i] = CreateMatrixCurrency(amMatrix, inTableNames[i], ,, ) - inCurrencies[3][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - inCurrencies[4][i] = CreateMatrixCurrency(pmMatrix, inTableNames[i], ,, ) - inCurrencies[5][i] = CreateMatrixCurrency(opMatrix, inTableNames[i], ,, ) - end - - //create an array of internal zone ids - dim zones[mxzone] - for i = 1 to zones.length do - zones[i]=i2s(i+mxext) - end - // create the output matrices, copying the matrix structure of the input matrices. Then rename the matrices. - dim outMatrices[outMatrixNames.length] - dim outCurrencies[outMatrixNames.length,outTableNames.length] - for i = 1 to outMatrixNames.length do - - outMatrices[i] = CopyMatrixStructure(inCurrencies[i], {{"File Name", outputDir+"\\"+outMatrixNames[i]}, - {"Label", outMatrixNames[i]}, - {"Tables",outTableNames}, - {"File Based", "Yes"}}) - SetMatrixCoreNames(outMatrices[i], outTableNames) - - for j = 1 to outTableNames.length do - outCurrencies[i][j] = CreateMatrixCurrency(outMatrices[i], outTableNames[j], ,, ) - - // set the output matrix to the input matrix - outCurrencies[i][j] := inCurrencies[i][j] - - //set the internal-internal values to 0 - FillMatrix(outCurrencies[i][j], zones, zones, {"Copy", 0.0},) - - end - end - - // factor the off-peak input table to 3 time periods (1=EA,3=MD,5=PM, and the factors are generic across input tables) - for i = 1 to outTableNames.length do - outCurrencies[1][i] := outCurrencies[1][i] * 0.10 - outCurrencies[2][i] := outCurrencies[2][i] - outCurrencies[3][i] := outCurrencies[3][i] * 0.65 - outCurrencies[4][i] := outCurrencies[4][i] - outCurrencies[5][i] := outCurrencies[5][i] * 0.25 - end - RunMacro("close all" ) - quit: - Return(1 ) - - -EndMacro -/*************************************************************************************************************************** - -Create external-external trip table -- modified 09/16/16 YMA - -****************************************************************************************************************************/ - -Macro "Create External-External Trip Matrix" // modified "externalExternalTripsByYear.csv" input and code, 09/14/16 YMA - - shared path, inputDir, outputDir, mxzone, mxext, scenarioYear - - //TODO: open external-external matrix here - //externalExternalFileName = inputDir+"\\externalExternalTrips.csv" // old input file - externalExternalFileName = inputDir+"\\externalExternalTripsByYear.csv" - externalExternalMatrixName = outputDir + "\\externalExternalTrips.mtx" - - extExtView = OpenTable("extExt", "CSV", {externalExternalFileName}, {{"Shared", "True"}}) - - opts = {} - opts.Label = "Trips" - opts.Type = "Float" - opts.Tables = {"2012","2014","2016","2017","2020","2025","2030","2035","2040","2045","2050"} - opts.[File Name] = externalExternalMatrixName - - extMatrix = CreateMatrixFromScratch(externalExternalMatrixName,mxzone,mxzone,opts) - coreNames = GetMatrixCoreNames(extMatrix ) - - for c = 1 to coreNames.length do - - if s2i(coreNames[c]) = s2i(scenarioYear) then do - - extCurren = CreateMatrixCurrency(extMatrix, coreNames[c], , , ) - extCurren := 0 - - rec = LocateRecord(extExtView+"|","year", {scenarioYear},{{"Exact", "True"}}) - - while rec <> null do - rec_vals = GetRecordValues(extExtView, rec, {"year","originTaz","destinationTaz","Trips"}) - year = rec_vals[1][2] - if year = s2i(scenarioYear) then do - originTaz = rec_vals[2][2] - destinationTaz = rec_vals[3][2] - Trips = rec_vals[4][2] - SetMatrixValue(extCurren, i2s(originTaz), i2s(destinationTaz), Trips) - rec= GetNextrecord(extExtView+"|",rec ,{{"year", "Ascending"},{"originTaz", "Ascending"}}) - end - else do - rec=null - end - end - SetMatrixCoreName(extMatrix, coreNames[c], "Trips") - end - else DropMatrixCore(extMatrix,coreNames[c]) - end - - - RunMacro("close all" ) - quit: - Return(1 ) -EndMacro - - diff --git a/sandag_abm/src/main/gisdk/createtrnroutes.rsc b/sandag_abm/src/main/gisdk/createtrnroutes.rsc deleted file mode 100644 index ae1f5e9..0000000 --- a/sandag_abm/src/main/gisdk/createtrnroutes.rsc +++ /dev/null @@ -1,655 +0,0 @@ -/****************************************************************************** - -Create all transit - -update travel time with congested time from adjuststr.bin - Macro "update travel time" -then create transit network from selection set of routes - Macro"create transit networks" -(input: mode.dbf include fields: mode_id, mode_name, fare, fare_type, fare_field ) - -c********************************************************************************/ - -Macro "Create all transit" - shared path, inputDir, outputDir, mxtap - - ok=RunMacro("Import transit layer",{}) - if !ok then goto quit - - ok=RunMacro("Add transit time fields") - if !ok then goto quit - - ok=RunMacro("Update stop xy") - if !ok then goto quit - - ok=RunMacro("Create transit routes") - if !ok then goto quit - - ok=RunMacro("calc preload") - if !ok then goto quit - - ok=RunMacro("update preload fields") - if !ok then goto quit - - ok = RunMacro("TCB Run Macro", 1, "update headways",{}) - if !ok then goto quit - - RunMacro("close all") - - ok=1 - quit: - return(ok) - -EndMacro - -/************************************************************************************ - -Import transit layer - -import e00 and export to geo file - -Inputs - input\trcov.e00 - -Outputs - output\transit.dbd - - -************************************************************************************/ - -Macro "Import transit layer" - shared path, inputDir, outputDir, mxtap - - //check e00 file exists - di = GetDirectoryInfo(inputDir+"\\trcov.e00", "File") - if di.length = 0 then do - RunMacro("TCB Error", "trcov.e00 doesn't exist") - return(0) - end - - - ImportE00(inputDir + "\\trcov.e00", outputDir + "\\trtmp.dbd","line",outputDir+ "\\trtmp.bin",{ - {"Label","transit line file"}, - {"Layer Name","transit"}, - {"optimize","True"}, - {"Median Split", "True"}, - {"Node Layer Name", "Endpoints"}, - {"Node Table", outputDir + "\\trtmp_.bin"}, - {"Projection","NAD83:406",{"_cdist=1000","_limit=1000","units=us-ft"}}, - }) - - //open the intermediate transit line layer geo file - map = RunMacro("G30 new map", outputDir + "\\trtmp.dbd","False") - SetLayer("transit") - allflds=GetFields("transit","All") - fullflds=allflds[2] - allnodeflds = GetFields("endpoints", "All") - - // need to specify full field specifications - lineidfield = "transit.trcov-id"//arcinfo id field - nodeidfield = "endpoints.tnode"//for centroids purposes - - opts = {{"Layer Name", "transit"}, - {"File Type", "FFB"}, - {"ID Field", lineidfield}, - {"Field Spec", fullflds}, - {"Indexed Fields", {fullflds[1]}}, - {"Label", "transit line file"}, - {"Node layer name","trnode"}, - {"Node ID Field", nodeidfield}, - {"Node Field Spec", allnodeflds[2]}} - - if node_idx > 1 then - opts = opts + {{"Node ID Field", node_aflds[2][node_idx - 1]}} - - ExportGeography("transit",outputDir + "\\transit.dbd",opts) - - RunMacro("close all") - DeleteDatabase(outputDir+"\\trtmp.dbd") - - return(1) - - quit: - return(0) -EndMacro - -/******************************************************************************************* - -Add transit time fields - -Adds fields to the transit line layer. Local and premium time fields are added for -each period and direction. The field names are - -xxField_yy where - -Field is LOCTIME (local transit time) or PRETIME (premium transit time) -Field is TM - -xx is AB or BA -yy is period - EA: Early AM - AM: AM peak - MD: Midday - PM: PM peak - EV: Evening - -Note: this should be replaced by a revised transit network - -*******************************************************************************************/ -Macro "Add transit time fields" - - shared path, inputDir, outputDir - db_file = outputDir+"\\transit.dbd" - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - db_link_lyr = db_file + "|" + link_lyr - SetLayer(link_lyr) - vw = GetView() - strct = GetTableStructure(vw) - - // Copy the current name to the end of strct - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // Add fields to the output table - new_struct = strct + { - {"ABTM_EA", "real", 14, 6, "False",,,,,,, null}, - {"BATM_EA", "real", 14, 6, "False",,,,,,, null}, - {"ABTM_AM", "real", 14, 6, "False",,,,,,, null}, - {"BATM_AM", "real", 14, 6, "False",,,,,,, null}, - {"ABTM_MD", "real", 14, 6, "False",,,,,,, null}, - {"BATM_MD", "real", 14, 6, "False",,,,,,, null}, - {"ABTM_PM", "real", 14, 6, "False",,,,,,, null}, - {"BATM_PM", "real", 14, 6, "False",,,,,,, null}, - {"ABTM_EV", "real", 14, 6, "False",,,,,,, null}, - {"BATM_EV", "real", 14, 6, "False",,,,,,, null}} - - // Modify table structure - ModifyTable(vw, new_struct) - vw = GetView() - - // Set time fields to their respective input fields (especially needed for transit-only links) - periods = {"_EA","_AM","_MD","_PM","_EV"} - orig_periods = {"O", "A", "O", "P", "O"} - for i = 1 to periods.length do - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"ABTM"+periods[i]} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"ABTM"+orig_periods[i]} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - Opts = null - Opts.Input.[View Set] = {db_link_lyr, link_lyr} - Opts.Input.[Dataview Set] = {db_link_lyr, link_lyr} - Opts.Global.Fields = {"BATM"+periods[i]} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = {"BATM"+orig_periods[i]} - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - if !ok then goto quit - - - end - - - RunMacro("close all") - - ok=1 - quit: - return(ok) - -EndMacro - -/*************************************************************************************************************************** -Update stop xy - -Update transit node file with longitude and latitude of nearest node(?) - -Input files: - input\trstop.bin - -Output files: - output\transit.dbd - -***************************************************************************************************************************/ -Macro "Update stop xy" - shared path, inputDir, outputDir - - Opts = null - Opts.Input.[Dataview Set] = {{inputDir+"\\trstop.bin", outputDir+"\\transit.dbd|trnode", "NearNode", "ID"}, "trstop+trnode"} - Opts.Global.Fields = {"trstop.Longitude"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = "trnode.Longitude" - - ok = RunMacro("TCB Run Operation", 1, "Fill Dataview", Opts) - - if !ok then goto quit - - // STEP 2: Fill Dataview - Opts = null - Opts.Input.[Dataview Set] = {{inputDir+"\\trstop.bin", outputDir+"\\transit.dbd|trnode", "NearNode", "ID"}, "trstop+trnode"} - Opts.Global.Fields = {"trstop.Latitude"} - Opts.Global.Method = "Formula" - Opts.Global.Parameter = "trnode.Latitude" - - ok = RunMacro("TCB Run Operation", 2, "Fill Dataview", Opts) - if !ok then goto quit - ok=1 - quit: - Return(ok) - -EndMacro - -/************************************************************************** -Create transit routes - -Create transit routes from table, and add the following fields to route table - -mode -headway -real route -fare -configdir - - -input files: - input\trlink.bin binary file for routes with link-id numbers - rte_number: sequential route number - link_id: node id - direction: +/- - input\trstop.bin stop table - input\trrt.bin route table - output\transit.dbd transit line layer with the updated congested travel time - -output files: - output\transitrt.rts transit route file - -***************************************************************************/ -Macro "Create transit routes" - shared path, inputDir, outputDir - - //check input bin files exist - lk_tb=inputDir+"\\trlink.bin" - stp_tb=inputDir+"\\trstop.bin" - rte_tb=inputDir+"\\trrt.bin" - - fnm=lk_tb - di = GetDirectoryInfo(fnm, "File") - if di.length = 0 then do - ok=0 - RunMacro("TCB Error",fnm +"does not exist!") - goto quit - end - - fnm=stp_tb - di = GetDirectoryInfo(fnm, "File") - if di.length = 0 then do - ok=0 - RunMacro("TCB Error",fnm +"does not exist!") - goto quit - end - - fnm=rte_tb - di = GetDirectoryInfo(fnm, "File") - if di.length = 0 then do - ok=0 - RunMacro("TCB Error",fnm +"does not exist!") - goto quit - end - - // delete any old index file left from last time - fnm=outputDir+"\\trlink.bx" - ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit - - fnm=outputDir+"\\trstop.bx" - ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit - - fnm=outputDir+"\\trrt.bx" - ok=RunMacro("SDdeletefile",{fnm}) if !ok then goto quit - - Opts = null - Opts = {{"Routes Table" , rte_tb}, - {"Stops Table", stp_tb}, - {"Stops", "Route Stops",}} - Opts.Label = "Transit Routes" - Opts.Name = "Transit Routes" - geo_path = outputDir+"\\transit.dbd" - geo_layer = "transit" - rte_file=outputDir+"\\transitrt.rts" - info = CreateRouteSystemFromTables(lk_tb, geo_path, geo_layer,rte_file , Opts) - map = RunMacro("G30 new rt map", rte_file, "False", "False",) - - RunMacro("close all") - - ok=1 - quit: - return(ok) -EndMacro -/******************************************************************** - -Create pre-load volumes on highway network from bus routes in -transit line layer. - - - -input files: hwycov.e00 - hwy line layer ESRI exchange file -output files: hwy.dbd - hwy line geographic file - hwycad.log- a log file - hwycad.err - error file with error info - -v0.1 5/28/2012 jef - -********************************************************************/ - -macro "calc preload" - shared path,inputDir,outputDir,mxzone - - db_file=outputDir+"\\transit.dbd" - rte_file=outputDir+"\\transitrt.rts" - - - periods = {"_EA","_AM","_MD","_PM","_EV"} - hours = { 3, 3, 6.5, 3.5, 5} - headway_flds = {"OP_Headway","AM_Headway","OP_Headway","PM_Headway","OP_Headway"} - - bus_pce = 3.0 - - //load transit line layer and route file - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) // line database - LinkIDField = link_lyr+".ID" - {rs_lyr, stop_lyr, ph_lyr} = RunMacro("TCB Add RS Layers", rte_file, "ALL",) // route system - if rs_lyr = null then goto quit - - //add fields for transit pce - SetLayer(link_lyr) - vw = GetView() - strct = GetTableStructure(vw) - - // Copy the current name to the end of strct - for i = 1 to strct.length do - strct[i] = strct[i] + {strct[i][1]} - end - - // Add fields to the output table - new_struct = strct + { - {"ABPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_EA", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_AM", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_MD", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_PM", "real", 14, 6, "False",,,,,,, null}, - {"ABPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}, - {"BAPRELOAD_EV", "real", 14, 6, "False",,,,,,, null}} - - // Modify table structure - ModifyTable(vw, new_struct) - - // query to determine valid routes - SetLayer(rs_lyr) - qry = "Select * where Mode > 0" - sel="All" - - n = SelectByQuery(sel, "Several", qry,) - CreateProgressBar("Loading...", "True") - RT_ViewSet = rs_lyr+"|" - - rh = GetFirstRecord(RT_ViewSet, null) - nRecords = GetRecordCount(rs_lyr, sel ) - count = 1 - - // loop through each route - while rh <> null do - - - // loop through periods - for i = 1 to periods.length do // for time period - - - hdwyvals = GetRecordValues(rs_lyr, rh, {"Route_ID", headway_flds[i]}) // get route headway - rtnm = GetRouteNam(rs_lyr, hdwyvals[1][2]) - - // get the links for each route - rt_links = GetRouteLinks(rs_lyr, rtnm) - msg = "Loading Route " + rtnm + " ..." - if UpdateProgressBar(msg, r2i(count/nRecords*100)) = "True" then do - ShowMessage("Execution stopped by user.") - DestroyProgressBar() - Return() - end - - // calculate bus frequency based on headway - veh_per_hour=0.0 - if (hdwyvals[2][2] <> null and hdwyvals[2][2]>0) then veh_per_hour = 60.0 / hdwyvals[2][2] // 60 / HDWY - - if veh_per_hour > 0 then do - View_Set = link_lyr + "|" - - // loop for every link along the route - for link = 1 to rt_links.length do - - // set record for the link - rh2 = LocateRecord(View_Set, LinkIDField, {rt_links[link][1], rt_links[link][2]},) - if rh2 <> null then do - - - ABFillField = "ABPRELOAD"+periods[i] - BAFillField = "BAPRELOAD"+periods[i] - - // get bus flow - fldvals = GetRecordValues(link_lyr, rh2, {ABFillField, BAFillField}) - ab_val = fldvals[1][2] - ba_val = fldvals[2][2] - - transit_pce = veh_per_hour * hours[i] * bus_pce - - if rt_links[link][2] = 1 then do // FORWARD - - if fldvals[1][2] = null then do - ab_val = transit_pce - end - else do - ab_val = fldvals[1][2] + transit_pce - end - end - else do // REVERSE - if fldvals[2][2] = null then do - ba_val = transit_pce - end - else do - ba_val = fldvals[2][2] + transit_pce - end - end - - // set the proper link id with the preload value - SetRecordValues(link_lyr, rh2, {{ABFillField, ab_val}, {BAFillField, ba_val}}) - - end - - end - end - end - - count = count + 1 - next_rcd: - rh = GetNextRecord(RT_ViewSet, null, null) - end - DestroyProgressBar() - - RunMacro("close all") - - ok=1 - quit: - return(ok) -EndMacro -/********************************************************************************************************** -Update preload fields - -Updates preload fields on the highway line layer from the transit line layer -**********************************************************************************************************/ -Macro "update preload fields" - - shared path, inputDir, outputDir - - periods = {"_EA","_AM","_MD","_PM","_EV"} - db_file=outputDir+"\\hwy.dbd" - net_file = outputDir + "\\hwy.net" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - vw = SetView(link_lyr) - - for i = 1 to periods.length do - - transitTable = outputDir+"\\transit.bin" - ABField = "ABPRELOAD"+periods[i] - BAField = "BAPRELOAD"+periods[i] - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, transitTable, {"ID"}, {"[TRCOV-ID]"}}, ABField } - Opts.Global.Fields = {"hwyline."+ABField} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if transit."+ABField+" <>null then transit."+ABField+" else 0.0" } - ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ok then goto quit - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, transitTable, {"ID"}, {"[TRCOV-ID]"}}, BAField} - Opts.Global.Fields = {"hwyline."+BAField} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if transit."+BAField+" <>null then transit."+BAField+" else 0.0" } - ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ok then goto quit - - end - - - //update the highway network with the new fields - for i = 1 to periods.length do - - field = "*PRELOAD"+periods[i] - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*PRELOAD"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABPRELOAD"+periods[i],link_lyr+".BAPRELOAD"+periods[i] } } - Opts.Global.Options.Constants = {1} - ok = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ok then goto quit - end - - RunMacro("close all") - - ok=1 - quit: - return(ok) - -EndMacro - -/********************************************************************************************************** -Update headway fields based upon Vovsha headway function - -**********************************************************************************************************/ -Macro "update headways" - - shared path, inputDir, outputDir - - db_file=outputDir+"\\transit.dbd" - rte_file=outputDir+"\\transitrt.rts" - - headway_flds = {"AM_Headway","OP_Headway","PM_Headway"} - - dim rev_headway[headway_flds.length] - - //load transit line layer and route file - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) // line database - LinkIDField = link_lyr+".ID" - {rs_lyr, stop_lyr, ph_lyr} = RunMacro("TCB Add RS Layers", rte_file, "ALL",) // route system - if rs_lyr = null then goto quit - - SetLayer(rs_lyr) - vw = GetView() - - RT_ViewSet = rs_lyr+"|" - - rh = GetFirstRecord(RT_ViewSet, null) - - // loop through each route - while rh <> null do - - hdwyvals = GetRecordValues(rs_lyr, rh, headway_flds) // get route headway - - //for each headway - for i = 1 to hdwyvals.length do - - //calculate revised headway - rev_headway[i] = RunMacro("calculate revised headway",hdwyvals[i][2]) - - // set the headways back in the route record - SetRecordValues(rs_lyr, rh, {{headway_flds[i], rev_headway[i]}}) - end - - rh = GetNextRecord(RT_ViewSet, null, null) - end - - - RunMacro("close all") - - ok=1 - quit: - Return(ok) -EndMacro - - -/***************************************************************************** - - Revised headways - -*****************************************************************************/ -Macro "calculate revised headway" (headway) - - // CALCULATE REVISED HEADWAY - - slope_1=1.0 //slope for 1st segment (high frequency) transit - slope_2=0.8 //slope for 2nd segemnt (med frequency) transit - slope_3=0.7 //slope for 3rd segment (low frequency) transit - slope_4=0.2 //slope for 4th segment (very low freq) transit - - break_1=10 //breakpoint of 1st segment, min - break_2=20 //breakpoint of 2nd segment, min - break_3=30 //breakpoint of 3rd segment, min - - - if headway < break_1 then do - rev_headway = headway * slope_1 - end - else if headway < break_2 then do - part_1_headway = break_1 * slope_1 - part_2_headway = (headway - break_1) * slope_2 - rev_headway = part_1_headway + part_2_headway - end - else if headway < break_3 then do - part_1_headway = break_1 * slope_1 - part_2_headway = (break_2 - break_1) * slope_2 - part_3_headway = (headway - break_2) * slope_3 - rev_headway = part_1_headway + part_2_headway + part_3_headway - end - else do - part_1_headway = break_1 * slope_1 - part_2_headway = (break_2 - break_1) * slope_2 - part_3_headway = (break_3 - break_2) * slope_3 - part_4_headway = (headway - break_3) * slope_4 - rev_headway = part_1_headway + part_2_headway + part_3_headway + part_4_headway - end - - Return(rev_headway) - -EndMacro - - \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/dbox.rsc b/sandag_abm/src/main/gisdk/dbox.rsc deleted file mode 100644 index c111630..0000000 --- a/sandag_abm/src/main/gisdk/dbox.rsc +++ /dev/null @@ -1,256 +0,0 @@ - -dBox "Run ABM" title: "SANDAG ABM" - init do - shared path, path_study - shared scr - - path_study="${workpath}" - scen_namefile = "\\scen_name.txt" - scen_info = GetFileInfo(path_study + scen_namefile) - if scen_info<>null then do - fptr_from = OpenFile(path_study + scen_namefile, "r") - scr=readarray(fptr_from) - end - else do - scr=null - while scr=Null do - RunMacro("getpathdirectory") - end - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - end -//end of editing - scen_num={1} - - path=scr[scen_num[1]] - -enditem - -text "Study Area:" 1,0,12 -text 12,0,34,1 Framed variable: path_study - -Button "Browse..." 48.5, 3, 16 do - opts={{"Initial Directory", path_study}} - path_study = ChooseDirectory("Choose a Study Area Directory",opts) - path_fortran=path_study+"\\fortran" - path_vb=path_study+"\\vb" - scen_info = GetFileInfo(path_study + scen_namefile) - if scen_info<>null then do - fptr_from = OpenFile(path_study + scen_namefile, "r") - scr=readarray(fptr_from) - end - else do - scr=null - while scr=Null do - RunMacro("getpathdirectory") - end - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - end - scen_num={1} - path=scr[scen_num[1]] - -enditem -//end of editing - -text "Add Scenario pathes in sequence order" 1,1.5, 38 - scroll list "scens" 1, 2.5, 46, 9 list:scr variable: scen_num multiple - help: "Select one or more scenarios to run" do - if scr = null then do // if any scenario chosen - while scr=null do - RunMacro("getpathdirectory") - end - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - scen_num ={1} // first scenario chosen - if tab_indx=1 then RunMacro("enable all1") else RunMacro("enable allfdlp") //enable all the "run" buttons - end - path=scr[scen_num[1]] - -enditem - -button "Quit" 48.5, 0, 16 do - batch_run_mode= false -//added by JXu on Nov 1, 2006 - maps = GetMaps() - if maps <> null then do - for i = 1 to maps[1].length do - SetMapSaveFlag(maps[1][i],"False") - end - end - RunMacro("G30 File Close All") - mtxs = GetMatrices() - if mtxs <> null then do - handles = mtxs[1] - for i = 1 to handles.length do - handles[i] = null - end - end -//end of editing - Return() -enditem - -button "Add" same , 1.5, 7 do - RunMacro("getpathdirectory") -//added by JXu - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) -//end of editing - scen_num={1} - path=scr[scen_num[1]] - enableitem("Delete") -enditem - -// delete a scenario -button "Delete" 57.5,1.5,7 do - scr = ExcludeArrayElements(scr,scen_num[1],scen_num.length) - while scr= null do - RunMacro("getpathdirectory") - end - - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - scen_num={1} - path=scr[scen_num[1]] - -enditem - - -button "Change Study" 48.5, 4.5, 16 do - scr = null - RunMacro("getstudydirectory") - scen_info = GetFileInfo(path_study + scen_namefile) - if scen_info<>null then do - fptr_from = OpenFile(path_study + scen_namefile, "r") - scr=readarray(fptr_from) - tmp_flag=0 - for i=1 to scr.length do - if scr[i]=path then do - i=scr.length+1 - tmp_flag=1 - end - else i=i+1 - end - if tmp_flag=0 then do - scr=scr+{path} - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - end - end - else do - scr={path} - fptr_from = OpenFile(path_study + scen_namefile, "w") - WriteArray(fptr_from,scr) - closefile(fptr_from) - end - - scen_num={1} - - path=scr[scen_num[1]] -enditem - - - button "Run ABM" same, 6, 16 do - hideDbox() - RunMacro("TCB Init") - for sc = 1 to scen_num.length do - path = scr[ scen_num[sc] ] - ok = RunMacro("Run SANDAG ABM") - if !ok then goto exit - end - exit: - showdbox() - RunMacro("TCB Closing", run_ok, "False") - enditem - - - button "Export Data" same, 7.5, 16 do - hideDbox() - RunMacro("TCB Init") - for sc = 1 to scen_num.length do - path = scr[ scen_num[sc] ] - ok = RunMacro("ExportSandagData") - if !ok then goto exit - end - exit: - showdbox() - RunMacro("TCB Closing", run_ok, "False") - enditem - - button "Sum Transit Sellink" same, 9, 16 do - hideDbox() - RunMacro("TCB Init") - for sc = 1 to scen_num.length do - path = scr[ scen_num[sc] ] - ok = RunMacro("Sum Up Select Link Transit Trips") - if !ok then goto exit - end - exit: - showdbox() - RunMacro("TCB Closing", run_ok, "False") - enditem - - button "Sum Hwy Sellink" same, 10.5, 16 do - hideDbox() - RunMacro("TCB Init") - for sc = 1 to scen_num.length do - path = scr[ scen_num[sc] ] - ok = RunMacro("Sum Up Select Link Highway Trips") - if !ok then goto exit - end - exit: - showdbox() - RunMacro("TCB Closing", run_ok, "False") - enditem - - -EndDbox - -// Macro "getpathdirectory" doesn't allow the selected path with different path_study. -Macro "getpathdirectory" - shared path,path_study,scr - opts={{"Initial Directory", path_study}} - tmp_path=choosedirectory("Choose an alternative directory in the same study area", opts) - strlen=len(tmp_path) - for i = 1 to strlen do - tmp=right(tmp_path,i) - tmpx=left(tmp,1) - if tmpx="\\" then goto endfor - end - endfor: - strlenx=strlen-i - tmppath_study=left(tmp_path,strlenx) - if path_study=tmppath_study then do - path=tmp_path - tmp_flag=0 - for i=1 to scr.length do - if scr[i]=path then do - tmp_flag=1 - i=scr.length+1 - end - else i=i+1 - end - if tmp_flag=0 then do - tmp = CopyArray(scr) - tmp = tmp + {tmp_path} - scr = CopyArray(tmp) - end - //showmessage("write description of the alternative in the head file") - //x=RunProgram("notepad "+path+"\\head",) - mytime=GetDateAndTime() - - end - else do - path=null - msg1="The alternative directory selected is invalid because it has different study area! " - msg2="Please select again within the same study area " - msg3=" or use the Browse button to select a different study area." - showMessage(msg1+msg2+path_study+msg3) - end -EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/exportTCData.rsc b/sandag_abm/src/main/gisdk/exportTCData.rsc deleted file mode 100644 index 2b1ce4b..0000000 --- a/sandag_abm/src/main/gisdk/exportTCData.rsc +++ /dev/null @@ -1,40 +0,0 @@ -Macro "ExportSandagData" - - shared path_study, path, inputDir, outputDir - - RunMacro("close all") - RunMacro("TCB Init") - - - inputDir = path+"\\input" - outputDir = path+"\\output" - reportDir = path+"\\report" - - network_file = outputDir+"\\hwy.dbd" - - output_network_file = reportDir + "\\hwy_tcad" - output_transit_onoff_file = reportDir + "\\transit_onoff" - output_transit_flow_file = reportDir + "\\transit_flow" - output_transit_aggflow_file = reportDir + "\\transit_aggflow" - input_route_file = inputDir + "\\trrt" - output_route_file = reportDir + "\\trrt" - input_stop_file = inputDir + "\\trstop" - output_stop_file = reportDir + "\\trstop" - - input_hwyload_file = RunMacro("FormPath",{outputDir,"hwyload_"}) - output_hwyload_file = RunMacro("FormPath",{reportDir,"hwyload_"}) - - external_zones = {"1","2","3","4","5","6","7","8","9","10","11","12"} - - RunMacro("ExportNetworkToCsv",network_file,output_network_file) - RunMacro("ExportHwyloadtoCSV",input_hwyload_file,output_hwyload_file) - RunMacro("ExportBintoCSV",input_route_file, output_route_file) - RunMacro("ExportBintoCSV",input_stop_file, output_stop_file) - RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildOnOffOptions"),output_transit_onoff_file) - RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildTransitFlowOptions"),output_transit_flow_file) - RunMacro("ExportTransitTablesToCsv",outputDir,RunMacro("BuildAggFlowOptions"),output_transit_aggflow_file) - - RunMacro("G30 File Close All") - return(1) - -EndMacro diff --git a/sandag_abm/src/main/gisdk/externalInternal.rsc b/sandag_abm/src/main/gisdk/externalInternal.rsc deleted file mode 100644 index d59e81d..0000000 --- a/sandag_abm/src/main/gisdk/externalInternal.rsc +++ /dev/null @@ -1,356 +0,0 @@ -Macro "US to SD External Trip Model" - - shared path, inputDir, outputDir, mxext, scenarioYear - - controlTotals = "externalInternalControlTotals.csv" - - controlTotalsView = OpenTable("Control Totals", "CSV", {inputDir+"\\"+controlTotals}, {{"Shared", "True"}}) - mgraView = OpenTable("MGRA View", "CSV",{inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) - - eaDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_EA_high.mtx", ) - eaDanMC = CreateMatrixCurrencies(eaDanMatrix,,,) - eaDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_EA_high.mtx", ) - eaDatMC = CreateMatrixCurrencies(eaDatMatrix,,,) - eaS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_EA_high.mtx", ) - eaS2nhMC = CreateMatrixCurrencies(eaS2nhMatrix,,,) - eaS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_EA_high.mtx", ) - eaS2thMC = CreateMatrixCurrencies(eaS2thMatrix,,,) - eaS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_EA_high.mtx", ) - eaS3nhMC = CreateMatrixCurrencies(eaS3nhMatrix,,,) - eaS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_EA_high.mtx", ) - eaS3thMC = CreateMatrixCurrencies(eaS3thMatrix,,,) - - amDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_AM_high.mtx", ) - amDanMC = CreateMatrixCurrencies(amDanMatrix,,,) - amDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_AM_high.mtx", ) - amDatMC = CreateMatrixCurrencies(amDatMatrix,,,) - amS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_AM_high.mtx", ) - amS2nhMC = CreateMatrixCurrencies(amS2nhMatrix,,,) - amS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_AM_high.mtx", ) - amS2thMC = CreateMatrixCurrencies(amS2thMatrix,,,) - amS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_AM_high.mtx", ) - amS3nhMC = CreateMatrixCurrencies(amS3nhMatrix,,,) - amS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_AM_high.mtx", ) - amS3thMC = CreateMatrixCurrencies(amS3thMatrix,,,) - - mdDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_MD_high.mtx", ) - mdDanMC = CreateMatrixCurrencies(mdDanMatrix,,,) - mdDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_MD_high.mtx", ) - mdDatMC = CreateMatrixCurrencies(mdDatMatrix,,,) - mdS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_MD_high.mtx", ) - mdS2nhMC = CreateMatrixCurrencies(mdS2nhMatrix,,,) - mdS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_MD_high.mtx", ) - mdS2thMC = CreateMatrixCurrencies(mdS2thMatrix,,,) - mdS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_MD_high.mtx", ) - mdS3nhMC = CreateMatrixCurrencies(mdS3nhMatrix,,,) - mdS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_MD_high.mtx", ) - mdS3thMC = CreateMatrixCurrencies(mdS3thMatrix,,,) - - pmDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_PM_high.mtx", ) - pmDanMC = CreateMatrixCurrencies(pmDanMatrix,,,) - pmDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_PM_high.mtx", ) - pmDatMC = CreateMatrixCurrencies(pmDatMatrix,,,) - pmS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_PM_high.mtx", ) - pmS2nhMC = CreateMatrixCurrencies(pmS2nhMatrix,,,) - pmS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_PM_high.mtx", ) - pmS2thMC = CreateMatrixCurrencies(pmS2thMatrix,,,) - pmS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_PM_high.mtx", ) - pmS3nhMC = CreateMatrixCurrencies(pmS3nhMatrix,,,) - pmS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_PM_high.mtx", ) - pmS3thMC = CreateMatrixCurrencies(pmS3thMatrix,,,) - - evDanMatrix = OpenMatrix(outputDir+"\\"+"impdan_EV_high.mtx", ) - evDanMC = CreateMatrixCurrencies(evDanMatrix,,,) - evDatMatrix = OpenMatrix(outputDir+"\\"+"impdat_EV_high.mtx", ) - evDatMC = CreateMatrixCurrencies(evDatMatrix,,,) - evS2nhMatrix = OpenMatrix(outputDir+"\\"+"imps2nh_EV_high.mtx", ) - evS2nhMC = CreateMatrixCurrencies(evS2nhMatrix,,,) - evS2thMatrix = OpenMatrix(outputDir+"\\"+"imps2th_EV_high.mtx", ) - evS2thMC = CreateMatrixCurrencies(evS2thMatrix,,,) - evS3nhMatrix = OpenMatrix(outputDir+"\\"+"imps3nh_EV_high.mtx", ) - evS3nhMC = CreateMatrixCurrencies(evS3nhMatrix,,,) - evS3thMatrix = OpenMatrix(outputDir+"\\"+"imps3th_EV_high.mtx", ) - evS3thMC = CreateMatrixCurrencies(evS3thMatrix,,,) - - freeTimeCurrencies = {eaDanMC.("*STM_EA (Skim)"),eaS2nhMC.("*HTM_EA (Skim)"),eaS3nhMC.("*HTM_EA (Skim)"), - amDanMC.("*STM_AM (Skim)"),amS2nhMC.("*HTM_AM (Skim)"),amS3nhMC.("*HTM_AM (Skim)"), - mdDanMC.("*STM_MD (Skim)"),mdS2nhMC.("*HTM_MD (Skim)"),mdS3nhMC.("*HTM_MD (Skim)"), - pmDanMC.("*STM_PM (Skim)"),pmS2nhMC.("*HTM_PM (Skim)"),pmS3nhMC.("*HTM_PM (Skim)"), - evDanMC.("*STM_EV (Skim)"),evS2nhMC.("*HTM_EV (Skim)"),evS3nhMC.("*HTM_EV (Skim)")} - - tollTimeCurrencies = {eaDatMC.("*STM_EA (Skim)"),eaS2thMC.("*HTM_EA (Skim)"),eaS3thMC.("*HTM_EA (Skim)"), - amDatMC.("*STM_AM (Skim)"),amS2thMC.("*HTM_AM (Skim)"),amS3thMC.("*HTM_AM (Skim)"), - mdDatMC.("*STM_MD (Skim)"),mdS2thMC.("*HTM_MD (Skim)"),mdS3thMC.("*HTM_MD (Skim)"), - pmDatMC.("*STM_PM (Skim)"),pmS2thMC.("*HTM_PM (Skim)"),pmS3thMC.("*HTM_PM (Skim)"), - evDatMC.("*STM_EV (Skim)"),evS2thMC.("*HTM_EV (Skim)"),evS3thMC.("*HTM_EV (Skim)")} - - //mod(eaS3thMC.("s3th_EA - itoll_EA"),10000), - - tollCostCurrencies = {eaDatMC.("dat_EA - itoll_EA"),eaS2thMC.("s2t_EA - itoll_EA"),eaS3thMC.("s3t_EA - itoll_EA"), - amDatMC.("dat_AM - itoll_AM"),amS2thMC.("s2t_AM - itoll_AM"),amS3thMC.("s3t_AM - itoll_AM"), - mdDatMC.("dat_MD - itoll_MD"),mdS2thMC.("s2t_MD - itoll_MD"),mdS3thMC.("s3t_MD - itoll_MD"), - pmDatMC.("dat_PM - itoll_PM"),pmS2thMC.("s2t_PM - itoll_PM"),pmS3thMC.("s3t_PM - itoll_PM"), - evDatMC.("dat_EV - itoll_EV"),evS2thMC.("s2t_EV - itoll_EV"),evS3thMC.("s3t_EV - itoll_EV")} - - controlTaz = GetDataVector(controlTotalsView+"|", "taz", {{"Sort Order", {{"taz", "Ascending"}}}} ) - controlWrk = GetDataVector(controlTotalsView+"|", "work", {{"Sort Order", {{"taz", "Ascending"}}}} ) - controlNon = GetDataVector(controlTotalsView+"|", "nonwork", {{"Sort Order", {{"taz", "Ascending"}}}} ) - - // create mgra size vectrors - - mgraView = OpenTable("MGRA View", "CSV", {inputDir+"\\mgra13_based_input"+scenarioYear+".csv"}, {{"Shared", "True"}}) - - mgra = GetDataVector(mgraView+"|", "mgra", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - taz = GetDataVector(mgraView+"|", "TAZ", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - hh = GetDataVector(mgraView+"|", "hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_ag = GetDataVector(mgraView+"|", "emp_ag", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_prod = GetDataVector(mgraView+"|", "emp_const_non_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_non_bldg_office = GetDataVector(mgraView+"|", "emp_const_non_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_prod = GetDataVector(mgraView+"|", "emp_utilities_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_utilities_office = GetDataVector(mgraView+"|", "emp_utilities_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_prod = GetDataVector(mgraView+"|", "emp_const_bldg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_const_bldg_office = GetDataVector(mgraView+"|", "emp_const_bldg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_prod = GetDataVector(mgraView+"|", "emp_mfg_prod", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_mfg_office = GetDataVector(mgraView+"|", "emp_mfg_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_whsle_whs = GetDataVector(mgraView+"|", "emp_whsle_whs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_trans = GetDataVector(mgraView+"|", "emp_trans", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_retail = GetDataVector(mgraView+"|", "emp_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs = GetDataVector(mgraView+"|", "emp_prof_bus_svcs", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_prof_bus_svcs_bldg_maint = GetDataVector(mgraView+"|", "emp_prof_bus_svcs_bldg_maint", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_k12 = GetDataVector(mgraView+"|", "emp_pvt_ed_k12", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_ed_post_k12_oth = GetDataVector(mgraView+"|", "emp_pvt_ed_post_k12_oth", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_health = GetDataVector(mgraView+"|", "emp_health", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_office = GetDataVector(mgraView+"|", "emp_personal_svcs_office", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_amusement = GetDataVector(mgraView+"|", "emp_amusement", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_hotel = GetDataVector(mgraView+"|", "emp_hotel", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_restaurant_bar = GetDataVector(mgraView+"|", "emp_restaurant_bar", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_personal_svcs_retail = GetDataVector(mgraView+"|", "emp_personal_svcs_retail", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_religious = GetDataVector(mgraView+"|", "emp_religious", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_pvt_hh = GetDataVector(mgraView+"|", "emp_pvt_hh", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_ent = GetDataVector(mgraView+"|", "emp_state_local_gov_ent", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_non_mil = GetDataVector(mgraView+"|", "emp_fed_non_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_fed_mil = GetDataVector(mgraView+"|", "emp_fed_mil", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_blue = GetDataVector(mgraView+"|", "emp_state_local_gov_blue", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_state_local_gov_white = GetDataVector(mgraView+"|", "emp_state_local_gov_white", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - emp_public_ed = GetDataVector(mgraView+"|", "emp_public_ed", {{"Sort Order", {{"mgra", "Ascending"}}}} ) - - emp_blu = emp_const_non_bldg_prod + emp_const_non_bldg_office + emp_utilities_prod + emp_utilities_office + emp_const_bldg_prod + emp_const_bldg_office + emp_mfg_prod + emp_mfg_office + emp_whsle_whs + emp_trans - emp_svc = emp_prof_bus_svcs + emp_prof_bus_svcs_bldg_maint + emp_personal_svcs_office + emp_personal_svcs_retail - emp_edu = emp_pvt_ed_k12 + emp_pvt_ed_post_k12_oth + emp_public_ed - emp_gov = emp_state_local_gov_ent + emp_fed_non_mil + emp_fed_non_mil + emp_state_local_gov_blue + emp_state_local_gov_white - emp_ent = emp_amusement + emp_hotel + emp_restaurant_bar - emp_oth = emp_religious + emp_pvt_hh + emp_fed_mil - - wrk_size_mgra = ( emp_blu + - 1.364 * emp_retail + - 4.264 * emp_ent + - 0.781 * emp_svc + - 1.403 * emp_edu + - 1.779 * emp_health + - 0.819 * emp_gov + - 0.708 * emp_oth ) - - non_size_mgra = ( hh + - 1.069 * emp_blu + - 4.001 * emp_retail + - 6.274 * emp_ent + - 0.901 * emp_svc + - 1.129 * emp_edu + - 2.754 * emp_health + - 1.407 * emp_gov + - 0.304 * emp_oth ) - - // aggregate to TAZ - - rowLabels = GetMatrixRowLabels(mdDatMC.("Length (Skim)")) - numZones = rowLabels.length - dim wrkSizeTaz[numZones] - dim nonSizeTaz[numZones] - - for i=1 to numZones do - wrkSizeTaz[i] = 0 - nonSizeTaz[i] = 0 - end - - for i=1 to mgra.length do - tazNumber = taz[i] - wrkSizeTaz[tazNumber] = wrkSizeTaz[tazNumber] + wrk_size_mgra[i] - nonSizeTaz[tazNumber] = nonSizeTaz[tazNumber] + non_size_mgra[i] - end - - wrkSizeVector = ArrayToVector(wrkSizeTaz,{{"Row Based","False"},{"Type","Float"}} ) - nonSizeVector = ArrayToVector(nonSizeTaz,{{"Row Based","False"},{"Type","Float"}}) - - - // Initialize matrices - - opts = {} - opts.Label = "Trips" - opts.Type = "Double" - opts.Tables = {'Trips'} - - opts.[File Name] = outputDir+"\\"+"usSdWrkPA.mtx" - wrkMatrixPA = CreateMatrixFromScratch("wrkMatrixPA",numZones,numZones,opts) - wrkCurrenPA = CreateMatrixCurrency(wrkMatrixPA,'Trips',,,) - - opts.[File Name] = outputDir+"\\"+"usSdNonPA.mtx" - nonMatrixPA = CreateMatrixFromScratch("nonMatrixPA",numZones,numZones,opts) - nonCurrenPA = CreateMatrixCurrency(nonMatrixPA,'Trips',,,) - - //create exponentiated distance impedance matrices - opts.Label = "Prob" - opts.Type = "Double" - opts.Tables = {'Prob'} - - opts.[File Name] = outputDir+"\\"+"wrkProb.mtx" - wrkProbMatrix = CreateMatrixFromScratch("wrkProb",numZones,numZones,opts) - wrkProb = CreateMatrixCurrency(wrkProbMatrix,'Prob',,,) - - opts.[File Name] = outputDir+"\\"+"nonProb.mtx" - nonProbMatrix = CreateMatrixFromScratch("nonProb",numZones,numZones,opts) - nonProb = CreateMatrixCurrency(nonProbMatrix,'Prob',,,) - - wrkDistCoef = -0.029 - nonDistCoef = -0.006 - - wrkProb := wrkSizeVector * exp( wrkDistCoef * mdDatMC.("Length (Skim)")) - nonProb := nonSizeVector * exp( nonDistCoef * mdDatMC.("Length (Skim)")) - - wrkSumVector = GetMatrixVector(wrkProb, {{"Marginal", "Row Sum"}}) - wrkProb := wrkProb/wrkSumVector - - nonSumVector = GetMatrixVector(nonProb, {{"Marginal", "Row Sum"}}) - nonProb := nonProb/nonSumVector - - wrkCurrenPA := 0 - nonCurrenPA := 0 - - // Loop over external zones and set values in output matrix - for i=1 to controlTaz.length do - - wrkTotal = controlWrk[i] - nonTotal = controlNon[i] - - wrkTripVector = wrkTotal * GetMatrixVector(wrkProb,{{"Row", controlTaz[i]}}) - nonTripVector = nonTotal * GetMatrixVector(wrkProb,{{"Row", controlTaz[i]}}) - - SetMatrixVector(wrkCurrenPA, wrkTripVector, {{"Row",controlTaz[i]}}) - SetMatrixVector(nonCurrenPA, nonTripVector, {{"Row",controlTaz[i]}}) - - end - - // Convert PA to OD and Apply Diurnal Factors - opts.Label = "Trips" - opts.Type = "Float" - opts.Tables = {'Trips'} - - opts.[File Name] = outputDir+"\\"+"usSdWrkDaily.mtx" - wrkMatrixAP = TransposeMatrix(wrkMatrixPA,opts) - wrkCurrenAP = CreateMatrixCurrency(wrkMatrixAP,'Trips',,,) - - opts.[File Name] = outputDir+"\\"+"usSdNonDaily.mtx" - nonMatrixAP = TransposeMatrix(nonMatrixPA,opts) - nonCurrenAP = CreateMatrixCurrency(nonMatrixAP,'Trips',,,) - - wrkCurrenPA := 0.5 * wrkCurrenPA - nonCurrenPA := 0.5 * nonCurrenPA - wrkCurrenAP := 0.5 * wrkCurrenAP - nonCurrenAP := 0.5 * nonCurrenAP - - // Apply Occupancy and Diurnal Factors - - opts.Tables = {"DAN","S2N","S3N","DAT","S2T","S3T"} - - opts.[File Name] = outputDir+"\\"+"usSdWrk_EA.mtx" - wrkMatrixEA = CreateMatrixFromScratch("wrkMatrixEA",numZones,numZones,opts) - wrkCurrenEA = CreateMatrixCurrencies(wrkMatrixEA,,,) - opts.[File Name] = outputDir+"\\"+"usSdWrk_AM.mtx" - wrkMatrixAM = CreateMatrixFromScratch("wrkMatrixAM",numZones,numZones,opts) - wrkCurrenAM = CreateMatrixCurrencies(wrkMatrixAM,,,) - opts.[File Name] = outputDir+"\\"+"usSdWrk_MD.mtx" - wrkMatrixMD = CreateMatrixFromScratch("wrkMatrixMD",numZones,numZones,opts) - wrkCurrenMD = CreateMatrixCurrencies(wrkMatrixMD,,,) - opts.[File Name] = outputDir+"\\"+"usSdWrk_PM.mtx" - wrkMatrixPM = CreateMatrixFromScratch("wrkMatrixPM",numZones,numZones,opts) - wrkCurrenPM = CreateMatrixCurrencies(wrkMatrixPM,,,) - opts.[File Name] = outputDir+"\\"+"usSdWrk_EV.mtx" - wrkMatrixEV = CreateMatrixFromScratch("wrkMatrixEV",numZones,numZones,opts) - wrkCurrenEV = CreateMatrixCurrencies(wrkMatrixEV,,,) - - opts.[File Name] = outputDir+"\\"+"usSdNon_EA.mtx" - nonMatrixEA = CreateMatrixFromScratch("nonMatrixEA",numZones,numZones,opts) - nonCurrenEA = CreateMatrixCurrencies(nonMatrixEA,,,) - opts.[File Name] = outputDir+"\\"+"usSdNon_AM.mtx" - nonMatrixAM = CreateMatrixFromScratch("nonMatrixAM",numZones,numZones,opts) - nonCurrenAM = CreateMatrixCurrencies(nonMatrixAM,,,) - opts.[File Name] = outputDir+"\\"+"usSdNon_MD.mtx" - nonMatrixMD = CreateMatrixFromScratch("nonMatrixMD",numZones,numZones,opts) - nonCurrenMD = CreateMatrixCurrencies(nonMatrixMD,,,) - opts.[File Name] = outputDir+"\\"+"usSdNon_PM.mtx" - nonMatrixPM = CreateMatrixFromScratch("nonMatrixPM",numZones,numZones,opts) - nonCurrenPM = CreateMatrixCurrencies(nonMatrixPM,,,) - opts.[File Name] = outputDir+"\\"+"usSdNon_EV.mtx" - nonMatrixEV = CreateMatrixFromScratch("nonMatrixEV",numZones,numZones,opts) - nonCurrenEV = CreateMatrixCurrencies(nonMatrixEV,,,) - - wrkCurrenAll = {wrkCurrenEA,wrkCurrenAM,wrkCurrenMD,wrkCurrenPM,wrkCurrenEV} - nonCurrenAll = {nonCurrenEA,nonCurrenAM,nonCurrenMD,nonCurrenPM,nonCurrenEV} - - wrkDiurnalPA = {0.26,0.26,0.41,0.06,0.02} - wrkDiurnalAP = {0.08,0.07,0.41,0.42,0.02} - - nonDiurnalPA = {0.25,0.39,0.30,0.04,0.02} - nonDiurnalAP = {0.12,0.11,0.37,0.38,0.02} - - wrkOccupancy = {0.58,0.31,0.11} - nonOccupancy = {0.55,0.29,0.15} - - matrixNames = {"DAN","S2N","S3N","DAT","S2T","S3T"} - - for periodIdx=1 to 5 do - for occupIdx = 1 to 3 do - - wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) := wrkOccupancy[occupIdx] * ( wrkDiurnalPA[periodIdx] * wrkCurrenPA + wrkDiurnalAP[periodIdx] * wrkCurrenAP ) - nonCurrenAll[periodIdx].(matrixNames[occupIdx]) := nonOccupancy[occupIdx] * ( nonDiurnalPA[periodIdx] * nonCurrenPA + nonDiurnalAP[periodIdx] * nonCurrenAP ) - - end - end - - // Toll choice split - - // values of time is cents per minute (toll cost is in cents) - votWork = 15.00 // $9.00/hr - votNonwork = 22.86 // $13.70/hr - ivtCoef = -0.03 - - for periodIdx=1 to 5 do - for occupIdx = 1 to 3 do - - currIndex = (periodIdx - 1) * 3 + occupIdx - - //wrkProb is work toll probability - wrkProb := if tollCostCurrencies[currIndex]>1000 then exp(ivtCoef * ( tollTimeCurrencies[ currIndex] - freeTimeCurrencies[ currIndex ] + mod(tollCostCurrencies[ currIndex ],10000) / votWork ) - 3.39) else - exp(ivtCoef * ( tollTimeCurrencies[ currIndex] - freeTimeCurrencies[ currIndex ] + tollCostCurrencies[ currIndex ] / votWork ) - 3.39) - - wrkProb := if tollCostCurrencies[ currIndex ] > 0 then wrkProb else 0 - wrkProb := wrkProb / ( 1 + wrkProb ) - - - wrkCurrenAll[periodIdx].(matrixNames[occupIdx+3]) := wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) * wrkProb - wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) := wrkCurrenAll[periodIdx].(matrixNames[occupIdx]) * (1.0 - wrkProb) - - //nonProb is non-work toll probability - nonProb := exp(ivtCoef * ( tollTimeCurrencies[ currIndex ] - freeTimeCurrencies[ currIndex] + mod(tollCostCurrencies[ currIndex],10000) / votNonwork )- 3.39) - nonProb := if tollCostCurrencies[ currIndex ] > 0 then nonProb else 0 - nonProb := nonProb / ( 1 + nonProb ) - nonCurrenAll[periodIdx].(matrixNames[occupIdx+3]) := nonCurrenAll[periodIdx].(matrixNames[occupIdx]) * nonProb - nonCurrenAll[periodIdx].(matrixNames[occupIdx]) := nonCurrenAll[periodIdx].(matrixNames[occupIdx]) * (1.0 - nonProb) - - end - end - - RunMacro("close all") - Return(1) - quit: - Return(0) -EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/gui_generic.rsc b/sandag_abm/src/main/gisdk/gui_generic.rsc deleted file mode 100644 index 5692403..0000000 --- a/sandag_abm/src/main/gisdk/gui_generic.rsc +++ /dev/null @@ -1,45 +0,0 @@ - -dBox "Setup Scenario" title: "SANDAG ABM" - init do - shared path, path_study - path = "${workpath}" -enditem - -// set model run parameters -button "Set Model Parameters" 0,0,30, 2 do - RunMacro("TCB Init") - runString = "T:\\ABM\\release\\ABM\\${version}\\dist\\parameterEditor.exe "+path - RunMacro("HwycadLog",{"gui.rsc:","Create a scenario"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Create a scenario", runString) -enditem - -// run model -button "Run Model" 0, 3, 30, 2 do - //hideDbox() - RunMacro("TCB Init") - RunMacro("getpathdirectory") - pFile_info = GetFileInfo(path+'\\conf\\sandag_abm.properties') - if pFile_info=null then do - CopyFile(path+"\\conf\\sandag_abm_standard.properties", path+"\\conf\\sandag_abm.properties") - end - ok = RunMacro("Run SANDAG ABM") - if !ok then goto exit - exit: - showdbox() - RunMacro("TCB Closing", run_ok, "False") -enditem - -//exit -button "Quit" 0, 6, 30, 2 do - RunMacro("G30 File Close All") - Return() -enditem - -EndDbox - -// Macro "getpathdirectory" doesn't allow the selected path with different path_study. -Macro "getpathdirectory" - shared path,path_study,scr - opts={{"Initial Directory", path}} - path=choosedirectory("Choose a scenario folder", opts) -EndMacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/hwyassign.rsc b/sandag_abm/src/main/gisdk/hwyassign.rsc deleted file mode 100644 index d6809c5..0000000 --- a/sandag_abm/src/main/gisdk/hwyassign.rsc +++ /dev/null @@ -1,896 +0,0 @@ -/********************************************************************************* -Multi-model Multi-class Assignement -macro "hwy assignment" - -input files: hwy.dbd - hwy.net - Trip_EA.mtx: Early AM auto trip matrix file - Trip_AM.mtx: AM Peak auto trip matrix file - Trip_MD.mtx: Midday auto trip matrix file - Trip_PM.mtx: PM Peak auto trip matrix file - Trip_EV.mtx: Evening auto trip matrix file - -each file has 14 cores: - - Name Description - ------- --------------------------------------- - SOV_GP Drive Alone Non-Toll - SOV_PAY Drive Alone Toll - SR2_GP Shared-ride 2 Person Non-HOV Non-Toll - SR2_HOV Shared-ride 2 Person HOV Non-Toll - SR2_PAY Shared-ride 2 Person HOV Toll Eligible - SR3_GP Shared-ride 3+ Person Non-HOV Non-Toll - SR2_HOV Shared-ride 3+ Person HOV Non-Toll - SR2_PAY Shared-ride 3+ Person HOV Toll Eligible - lhdn Light heavy-duty Truck Non-Toll - mhdn Medium heavy-duty Truck Non-Toll - hhdnv Heavy heavy-duty Truck Non-Toll - lhdt Light heavy-duty Truck Toll - mhdt Medium heavy-duty Truck Toll - hhdt Heavy heavy-duty Truck Toll - - Functions are added by J Xu between Dec 2006 and March 2007 - (1) Select Link Analysis and split the resulting flow table by each select link inquiries; - (2) Enhanced highway assignment for four different cases, involving toll. - (3) Turning movement - -output files: - - hwyload_EA.bin: Early AM loaded network binary file - hwyload_AM.bin: Am Peak loaded network binary file - hwyload_MD.bin: Midday loaded network binary file - hwyload_PM.bin: PM Peak loaded network binary file - hwyload_EV.bin: Evening loaded network binary file - -Optionally (for select link and turning movements): - - turns_EA.bin: Early AM turning movement file - turns_AM.bin: Am Peak turning movement file - turns_MD.bin: Midday turning movement file - turns_PM.bin: PM Peak turning movement file - turns_EV.bin: Evening turning movement file - - select_EA.mtx: Early AM select link trip matrix file - select_AM.mtx: Am Peak select link trip matrix file - select_MD.mtx: Midday select link trip matrix file - select_PM.mtx: PM Peak select link trip matrix file - select_EV.mtx: Evening select link trip matrix file - - -SANDAG ABM Version 1.0 - JEF 2012-03-20 - changed linktypeturnscst.dbf to linktypeturns.dbf as Joel suggested. -*************************************************************************************/ - -Macro "hwy assignment" (args) - - Shared path, inputDir, outputDir, mxzone - - properties = "\\conf\\sandag_abm.properties" - convergence = RunMacro("read properties",properties,"convergence", "S") - assign_reliability="true" - - turn_file="\\nodes.txt" - turn_flag=0 - NumofCPU = 8 - iteration = args[1] - assignByVOT= args[2] - - // for debug - - periods = {"_EA","_AM","_MD","_PM","_EV"} - - RunMacro("close all") - dim excl_qry[periods.length],excl_toll[periods.length],excl_dat[periods.length],excl_s2nh[periods.length],excl_s2th[periods.length],excl_s3nh[periods.length] - dim excl_s3th[periods.length],excl_lhdn[periods.length],excl_mhdn[periods.length],excl_hhdn[periods.length],excl_lhdt[periods.length],excl_mhdt[periods.length] - dim excl_hhdt[periods.length],toll_fld[periods.length],toll_fld2[periods.length] - - linkt= {"*TM_EA","*TM_AM","*TM_MD","*TM_PM","*TM_EV"} - linkcap={"*CP_EA","*CP_AM","*CP_MD","*CP_PM","*CP_EV"} -// xt= {"*TX_EA","*TX_AM","*TX_MD","*TX_PM","*TX_EV"} - xcap= {"*CX_EA","*CX_AM","*CX_MD","*CX_PM","*CX_EV"} - - cycle={"*CYCLE_EA","*CYCLE_AM","*CYCLE_MD","*CYCLE_PM","*CYCLE_EV"} - pfact={"*PF_EA","*PF_AM","*PF_MD","*PF_PM","*PF_EV"} - gcrat={"*GCRATIO_EA","*GCRATIO_AM","*GCRATIO_MD","*GCRATIO_PM","*GCRATIO_EV"} - alpha1={"*ALPHA1_EA","*ALPHA1_AM","*ALPHA1_MD","*ALPHA1_PM","*ALPHA1_EV"} - beta1={"*BETA1_EA","*BETA1_AM","*BETA1_MD","*BETA1_PM","*BETA1_EV"} - alpha2={"*ALPHA2_EA","*ALPHA2_AM","*ALPHA2_MD","*ALPHA2_PM","*ALPHA2_EV"} - beta2={"*BETA2_EA","*BETA2_AM","*BETA2_MD","*BETA2_PM","*BETA2_EV"} - preload={"*PRELOAD_EA","*PRELOAD_AM","*PRELOAD_MD","*PRELOAD_PM","*PRELOAD_EV"} - statrel={"*STATREL_EA","*STATREL_AM","*STATREL_MD","*STATREL_PM","*STATREL_EV"} - - db_file = outputDir + "\\hwy.dbd" - net_file= outputDir+"\\hwy.net" - - turn={"turns_EA.bin","turns_AM.bin","turns_MD.bin","turns_PM.bin","turns_EV.bin"} - selectlink_mtx={"select_EA.mtx","select_AM.mtx","select_MD.mtx","select_PM.mtx","select_EV.mtx"} //added for select link analysis by JXu - selinkqry_file="selectlink_query.txt" - if GetFileInfo(inputDir + "\\"+ selinkqry_file) <> null then do //select link analysis is only available in stage II - selink_flag =1 - fptr_from = OpenFile(inputDir + "\\"+selinkqry_file, "r") - tmp_qry=readarray(fptr_from) - index =1 - selinkqry_name=null - selink_qry=null - subs=null - while index <=ArrayLength(tmp_qry) do - if left(trim(tmp_qry[index]),1)!="*" then do - subs=ParseString(trim(tmp_qry[index]),",") - if subs!=null then do - query=subs[3] - if ArrayLength(subs)>3 then do - for i=4 to ArrayLength(subs) do - query=query+" "+subs[2]+" "+subs[i] - end - end - selinkqry_name=selinkqry_name+{subs[1]} - selink_qry=selink_qry+{query} - end - end - index = index + 1 - end - end - - asign = {"hwyload_EA.bin","hwyload_AM.bin","hwyload_MD.bin","hwyload_PM.bin","hwyload_EV.bin"} - oue_path = {"oue_path_EA.obt", "oue_path_AM.obt","oue_path_MD.obt","oue_path_PM.obt","oue_path_EV.obt"} - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - db_link_lyr=db_file+"|"+link_lyr - db_node_lyr=db_file+"|"+node_lyr - - // drive-alone non-toll exclusion set - excl_dan={db_link_lyr, link_lyr, "dan", "Select * where !((ihov=1|ifc>7)&ITRUCK<5)"} - - // shared-2 non-toll non-HOV exclusion set - excl_s2nn=excl_dan - - // shared 3+ non-toll non-HOV exclusion set - excl_s3nn=excl_dan - - - for i = 1 to periods.length do - // drive-alone toll exclusion set - //excl_dat[i]={db_link_lyr, link_lyr, "dat", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)"} - excl_dat[i]={db_link_lyr, link_lyr, "dat", "Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)"} - - // shared-2 non-toll HOV exclusion set - excl_s2nh[i]={db_link_lyr, link_lyr, "s2nh", "Select * where !((ihov=1|(ihov=2&abln"+periods[i]+" <9)|ifc>7)&ITRUCK<5)"} - - // shared-2 toll HOV exclusion set - excl_s2th[i]={db_link_lyr, link_lyr, "s2th", "Select * where !(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)"} - - // shared=3+ non-toll non-HOV exclusion set - excl_s3nh[i]={db_link_lyr, link_lyr, "s3nh", "Select * where !((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)"} - - // shared=3+ toll HOV exclusion set - excl_s3th[i]={db_link_lyr, link_lyr, "s3th", "Select * where abln"+periods[i]+"=9|ITRUCK>4"} - - // light-heavy truck non-toll exclusion set - excl_lhdn[i]={db_link_lyr, link_lyr, "lhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK<4|ITRUCK=7))"} - - // medium-heavy truck non-toll exclusion set - excl_mhdn[i]={db_link_lyr, link_lyr, "mhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK<3|ITRUCK>5))"} - - // heavy-heavy truck non-toll exclusion set - excl_hhdn[i]={db_link_lyr, link_lyr, "hhdn", "Select * where !((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))"} - - // light-heavy truck toll exclusion set - //excl_lhdt[i]={db_link_lyr, link_lyr, "lhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))"} - excl_lhdt[i]={db_link_lyr, link_lyr, "lhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))"} - - // medium-heavy truck toll exclusion set - //excl_mhdt[i]={db_link_lyr, link_lyr, "mhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))"} - excl_mhdt[i]={db_link_lyr, link_lyr, "mhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))"} - - // heavy-heavy truck toll exclusion set - //excl_hhdt[i]={db_link_lyr, link_lyr, "hhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))"} - excl_hhdt[i]={db_link_lyr, link_lyr, "hhd", "Select * where !(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))"} - - end - - //reset exclusion array value based on the selection set results - - set = "dat" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - for i = 1 to periods.length do - //n = SelectByQuery(set, "Several","Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)",) - n = SelectByQuery(set, "Several","Select * where !((ihov=1|ihov=4|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)",) - if n = 0 then excl_dat[i]=null - end - - set = "s2nh" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - for i = 1 to periods.length do - n = SelectByQuery(set, "Several","Select * where !((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)",) - if n = 0 then excl_s2nh[i]=null - end - - set = "s2th" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - for i = 1 to periods.length do - n = SelectByQuery(set, "Several", "Select * where !(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)",) - if n = 0 then excl_s2th[i]=null - end - - set = "s3nh" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - for i = 1 to periods.length do - n = SelectByQuery(set, "Several","Select * where !((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)&ITRUCK<5)",) - if n = 0 then excl_s3nh[i]=null - end - - set = "s3th" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - for i = 1 to periods.length do - n = SelectByQuery(set, "Several", "Select * where abln"+periods[i]+"=9|ITRUCK>4",) - if n = 0 then excl_s3th[i]=null - end - - if assignByVOT = "false" then do - trip={"Trip_EA.mtx","Trip_AM.mtx","Trip_MD.mtx","Trip_PM.mtx","Trip_EV.mtx"} - num_class=14 - vehclass={1,2,3,4,5,6,7,8,9,10,11,12,13,14} - class_PCE={1,1,1,1,1,1,1,1,1.3,1.5,2.5,1.3,1.5,2.5} - VOT={67,67,67,67,67,67,67,67,67,68,89,67,68,89} // vot is in cents per minute: 67 cents/min = $40.20/hour - - for i = 1 to periods.length do - excl_qry[i]={excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i],excl_lhdn[i],excl_mhdn[i],excl_hhdn[i],excl_lhdt[i],excl_mhdt[i],excl_hhdt[i]} - toll_fld2[i]= {"COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","COST","ITOLL3"+periods[i],"ITOLL4"+periods[i],"ITOLL5"+periods[i]} - end - end - else do - - // for VOT assignment, explode the passenger modes 1 through 8 to 1 through 24. First 8 low vot, next 8 med vot, final 8 high vot. Then 6 truck classes for total 30 classes. - trip={"Trip"+periods[1]+"_VOT.mtx", "Trip"+periods[2]+"_VOT.mtx", "Trip"+periods[3]+"_VOT.mtx", "Trip"+periods[4]+"_VOT.mtx", "Trip"+periods[5]+"_VOT.mtx"} - num_class=30 - vehclass={1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30} - class_PCE={1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1.3,1.5,2.5,1.3,1.5,2.5} - VOT={16.6,16.6,16.6,16.6,16.6,16.6,16.6,16.6,33.3,33.3,33.3,33.3,33.3,33.3,33.3,33.3,100,100,100,100,100,100,100,100,67,68,89,67,68,89} //assuming $10/hr, $20/hr, and $60/hr as max of each range - - for i = 1 to periods.length do - excl_qry[i]={ - excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], - excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], - excl_dan,excl_dat[i],excl_s2nn,excl_s2nh[i],excl_s2th[i],excl_s3nn,excl_s3nh[i],excl_s3th[i], - excl_lhdn[i],excl_mhdn[i],excl_hhdn[i],excl_lhdt[i],excl_mhdt[i],excl_hhdt[i]} - toll_fld2[i]= { - "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], - "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], - "COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i],"COST","COST","ITOLL3"+periods[i], - "COST","COST","COST","ITOLL3"+periods[i],"ITOLL4"+periods[i],"ITOLL5"+periods[i]} - end - end - - - - //Prepare selection set for turning movement report, by JXu - if (turn_flag=1 & iteration=4) then do - if GetFileInfo(inputDir+turn_file)!=null then do - fptr_turn = OpenFile(inputDir + turn_file,"r") - tmp_qry=readarray(fptr_turn) - turn_qry=null - index=1 - while index <=ArrayLength(tmp_qry) do - if index=1 then - turn_qry = "select * where " + "ID=" + tmp_qry[1] - else - turn_qry = turn_qry + " OR " + "ID=" + tmp_qry[index] - if tmp_qry[index]="all" or tmp_qry[index]="All" or tmp_qry[index]="ALL" then do - turn_qry = "select * where ID>"+i2s(mxzone) //select all nodes except centroids - index=ArrayLength(tmp_qry) - end - index=index+1 - end - closefile(fptr_turn) - end - else turn_qry = "select * where temp=1" - if GetFileInfo(path+"\\turn.err")!=null then do - ok=RunMacro("SDdeletefile",{path+"\\turn.err"}) - if !ok then goto quit - end - - tmpset = "turn" - vw_set = node_lyr + "|" + tmpset - SetLayer(node_lyr) - n = SelectByQuery(tmpset , "Several", turn_qry,) - if n = 0 then do - showmessage("Warning!!! No intersections selected for turning movement.") - fp_tmp = OpenFile(path + "\\turn.err","w") - WriteArray(fp_tmp,{"No intersections have been selected for turning movement."}) - closefile(fp_tmp) - return(1) - end - end - - - //set hwy.net with turn penalty of time in minutes - d_tp_tb = inputDir + "\\linktypeturns.dbf" - s_tp_tb = outputDir + "\\turns.dbf" - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Toll Set] = {db_link_lyr, link_lyr} - Opts.Input.[Centroids Set] = {db_node_lyr, node_lyr, "Selection", "select * where ID <="+i2s(mxzone)} - Opts.Global.[Spc Turn Pen Method] = 3 - Opts.Input.[Def Turn Pen Table] = {d_tp_tb} - Opts.Input.[Spc Turn Pen Table] = {s_tp_tb} - Opts.Field.[Link type] = "IFC" - Opts.Global.[Global Turn Penalties] = {0, 0, 0, 0} - Opts.Flag.[Use Link Types] = "True" - RunMacro("HwycadLog",{"hwyassign.rsc: hwy assignment","Highway Network Setting"}) - ok = RunMacro("TCB Run Operation", 1, "Highway Network Setting", Opts) - if !ok then goto quit - - // STEP 1: MMA - for i = 1 to periods.length do - - //set the vdf DLL and vdf field names based on whether reliability is being used or not -// if assign_reliability="true" then do - vdf_fields = {linkt[i], linkcap[i], xcap[i], cycle[i],pfact[i], gcrat[i], alpha1[i], beta1[i], alpha2[i], beta2[i], - "*LOSC_FACT","*LOSD_FACT","*LOSE_FACT", "*LOSFL_FACT", "*LOSFH_FACT", statrel[i], "Length", preload[i]} - vdf_file = "shrp.vdf" - vdf_defaults={ , , ,1.5 ,1 , 0.4 , 0.15, 4, 0.15, 4, 0, 0, 0, 0, 0, 0, 0, 0 } -// end -/* else do - vdf_fields = {linkt[i], linkcap[i], xcap[i], cycle[i],pfact[i], gcrat[i], alpha1[i], beta1[i], alpha2[i], beta2[i], preload[i]} - vdf_file = "tucson_vdf_rev.vdf" - vdf_defaults={ , , ,1.5 ,1 , 0.4 , 0.15, 4, 0.15, 4, 0 } - end - */ - net = ReadNetwork(net_file) - NetOpts = null - NetOpts.[Link ID] = link_lyr+".ID" - NetOpts.[Type] = "Enable" - NetOpts.[Write to file] = "Yes" - ChangeLinkStatus(net,, NetOpts) - - // Open the trip table to assign, and get the first table name - ODMatrix = outputDir + "\\"+trip[i] - m = OpenMatrix(ODMatrix,) - matrixCores = GetMatrixCoreNames(GetMatrix()) - coreName = matrixCores[1] - - - //settings for highway assignment - Opts = null - Opts.Global.[Force Threads] = 2 - Opts.Input.Database = db_file - Opts.Input.Network = net_file - - Opts.Input.[OD Matrix Currency] = {ODMatrix, coreName, , } - Opts.Input.[Exclusion Link Sets] = excl_qry[i] - Opts.Field.[Vehicle Classes] = vehclass - Opts.Field.[Fixed Toll Fields] = toll_fld2[i] - Opts.Field.[VDF Fld Names] = vdf_fields - Opts.Global.[Number of Classes] = num_class - Opts.Global.[Class PCEs] = class_PCE - Opts.Global.[Class VOIs] = VOT - Opts.Global.[Load Method] = "NCFW" - Opts.Global.[N Conjugate] = 2 - Opts.Global.[Loading Multiplier] = 1 - Opts.Global.Convergence = 0.0005 - Opts.Global.[Cost Function File] = vdf_file - Opts.Global.[VDF Defaults] = vdf_defaults - Opts.Global.[Iterations]=1000 - Opts.Flag.[Do Share Report] = 1 - Opts.Output.[Flow Table] = outputDir+"\\"+asign[i] - if (turn_flag=1 & iteration=4) then Opts.Input.[Turning Movement Node Set] = {db_node_lyr, node_lyr, "Selection", turn_qry} - if (turn_flag=1 & iteration=4) then Opts.Flag.[Do Turn Movement] = 1 - if (turn_flag=1 & iteration=4) then Opts.Output.[Movement Table] = outputDir+"\\"+turn[i] - Opts.Field.[MSA Flow] = "_MSAFlow" + periods[i] - Opts.Field.[MSA Cost] = "_MSACost" + periods[i] - Opts.Field.[MSA Time] = "_MSATime" + periods[i] - Opts.Global.[MSA Iteration] = iteration - if (selink_flag = 1 & iteration = 4) then do - Opts.Global.[Critical Queries] = selink_qry - Opts.Global.[Critical Set names] = selinkqry_name - Opts.Output.[Critical Matrix].Label = "Select Link Matrix" - Opts.Output.[Critical Matrix].Compression = 1 - Opts.Output.[Critical Matrix].[File Name] = outputDir +"\\"+selectlink_mtx[i] - end - RunMacro("HwycadLog",{"hwyassign.rsc: hwy assignment","MMA: "+asign[i]}) - ok = RunMacro("TCB Run Procedure", i, "MMA", Opts) - if !ok then goto quit - end - if!ok then goto quit - - ok=1 - quit: - RunMacro("close all") - return(ok) -EndMacro - -//added by JXu to split the flow table by queries. - -Macro "Selink Flow Split" (arr_selink) - shared path, inputDir, outputDir - asign=arr_selink[1] - selinkqry_name=arr_selink[2] - m=ArrayLength(selinkqry_name)+1 - dim new_flowtb[ArrayLength(asign),ArrayLength(selinkqry_name)+1] //All new flow table names after splitting for OP, AM and PM period assignments (3x5=15 names) - for i=1 to ArrayLength(asign) do - for j=1 to ArrayLength(selinkqry_name) do - new_flowtb[i][j] = outputDir+"\\"+left(asign[i],len(asign[i])-4)+"sl"+ i2s(j) +".bin" - end - new_flowtb[i][ArrayLength(selinkqry_name)+1]=outputDir+"\\"+asign[i] - end -//rename the original flow table file name to avoid the file name conflit with the splitted flow table. - for i=1 to ArrayLength(asign) do - new_file=left(asign[i],len(asign[i])-4)+"_orig.bin" - dict_nm = left(asign[i],len(asign[i])-4)+".dcb" - newdict_nm = left(asign[i],len(asign[i])-4)+"_orig.dcb" - ok=RunMacro("SDrenamefile",{outputDir+"\\"+asign[i],outputDir+"\\"+new_file}) if!ok then goto quit - ok=RunMacro("SDrenamefile",{outputDir+"\\"+dict_nm,outputDir+"\\"+newdict_nm}) if!ok then goto quit - asign[i]=new_file - end - -// This loop closes all views: - tmp = GetViews() - if tmp<>null then - for i = 1 to ArrayLength(tmp[1]) do - CloseView(tmp[1][i]) - end - - flowtb_vw = OpenTable("Flow Table", "FFB", {outputDir+"\\"+asign[1],}) - flowtb_fldinfo = GetViewStructure(flowtb_vw) //Get all the fields info from the flow table - dim flds_flag[ArrayLength(flowtb_fldinfo),ArrayLength(selinkqry_name)+1] -//flds_flag[i][j]=1 if flowtb_fldinfo[i][1] will be exported to .bin file for query j; -//if j is more than number of queries, then flds_flag[i][j] decides if flowtb_fldinfo[i][1] will be exported to .bin flow table without query info. - for i=1 to ArrayLength(flowtb_fldinfo) do - flag_tot = 0 - for j=1 to ArrayLength(selinkqry_name) do - if Position(flowtb_fldinfo[i][1],selinkqry_name[j])=0 then do - flds_flag[i][j]=0 - flag_tot=flag_tot+1 - end - else flds_flag[i][j]=1 - end - if flag_tot = ArrayLength(selinkqry_name) then do - for j=1 to ArrayLength(selinkqry_name) do - flds_flag[i][j]=1 - end - flds_flag[i][ArrayLength(selinkqry_name)+1]=1 //This will be exported to the flow table of original format without any query info - end - else flds_flag[i][ArrayLength(selinkqry_name)+1]=0 - end - - dim newflds[ArrayLength(selinkqry_name)+1] - for j=1 to ArrayLength(selinkqry_name)+1 do - for i=1 to ArrayLength(flowtb_fldinfo) do - if flds_flag[i][j]=1 then - newflds[j]=newflds[j]+{flowtb_fldinfo[i][1]} - end - end - - for i=1 to arraylength(asign) do - flow_vw = OpenTable("All Flow", "FFB", {outputDir+"\\"+asign[i],}) - for j=1 to arraylength(selinkqry_name) do - ExportView(flow_vw+"|", "FFB", new_flowtb[i][j],newflds[j], - {{"Additional Fields",{{"AB_Flow_"+selinkqry_name[j],"Real",15,4,"No"}, - {"BA_Flow_"+selinkqry_name[j],"Real",15,4,"No"}, - {"Tot_Flow_"+selinkqry_name[j],"Real",15,4,"No"}}} - }) - end - ExportView(flow_vw+"|", "FFB", new_flowtb[i][m],newflds[m],) - end - - //Fill in the newly added fields in the splitted flow table for each query - for i=1 to arraylength(asign) do - newflow_vw=null - newflow_fldinfo=null - For j=1 to arraylength(selinkqry_name) do - newflow_vw = OpenTable("Splitted Flow", "FFB", {new_flowtb[i][j],}) - newflow_fldinfo = GetViewStructure(newflow_vw) - AB_qry_flds=null - BA_qry_flds=null - for k=1 to ArrayLength(newflow_fldinfo) do - if Position(newflow_fldinfo[k][1],selinkqry_name[j])<>0 then do - if Position(newflow_fldinfo[k][1],"AB")<>0 then - AB_qry_flds=AB_qry_flds+{newflow_fldinfo[k][1]} - if Position(newflow_fldinfo[k][1],"BA")<>0 then - BA_qry_flds=BA_qry_flds+{newflow_fldinfo[k][1]} - end - end - order = {{"ID1", "Ascending"}} - rh = GetFirstRecord(newflow_vw+ "|", order) - while rh <> null do - AB_vals=0 - BA_vals=0 - for k=1 to ArrayLength(AB_qry_flds) do - vals=GetRecordValues(newflow_vw, rh, {AB_qry_flds[k],BA_qry_flds[k]}) - AB_vals=AB_vals+NZ(vals[1][2]) - BA_vals=BA_vals+NZ(vals[2][2]) - Tot_vals=AB_vals+BA_vals - end - SetRecordValues(newflow_vw, rh, {{"AB_Flow_"+selinkqry_name[j], AB_vals}, - {"BA_Flow_"+selinkqry_name[j], BA_vals}, - {"Tot_Flow_"+selinkqry_name[j],Tot_vals}}) - setRecord(newflow_vw, rh) - rh = GetNextRecord(newflow_vw + "|", rh, order) - end - end - end - - ok=1 - quit: - RunMacro("close all") - return(ok) - -endMacro - - -/********************************************************************************************************** - - combine truck tt_nt assign - - -**********************************************************************************************************/ -Macro "combine truck tt_nt assign"(arr) - shared path, inputDir, outputDir - stage = arr[1] - - asignbin={"lodtollop2.bin","lodtollam2.bin","lodtollpm2.bin"} - asigndcb={"lodtollop2.DCB","lodtollam2.DCB","lodtollpm2.DCB"} - copybin={"lodtollclassop2.bin","lodtollclassam2.bin","lodtollclasspm2.bin"} - copydcb={"lodtollclassop2.DCB","lodtollclassam2.DCB","lodtollclasspm2.DCB"} - viewNames ={"lodtollop2","lodtollam2","lodtollpm2"} - - // Copy files - for k=1 to 3 do - // check if copy files already exist, if exist delete - file=outputDir+"\\"+copybin[k] - dif2=GetDirectoryInfo(file,"file") - if dif2.length>0 then deletefile(file) - ok=1 - - CopyTableFiles(null,"FFB", outputDir+"\\"+asignbin[k], outputDir+"\\"+asigndcb[k],outputDir+"\\"+copybin[k], outputDir+"\\"+copydcb[k]) - - // delete the original highway files once copied - // check if copy files already exist, if exist delete - file=outputDir+"\\"+asignbin[k] - dif2=GetDirectoryInfo(file,"file") - if dif2.length>0 then deletefile(file) - ok=1 - - // Get copied files - view = OpenTable("assignment", "FFB", {outputDir+"\\"+copybin[k],} ) - ok1 = (view1 != null) - - // number of records - records = GetRecordCount(view, null) - - hov3_info = GetFileInfo(inputDir+"\\hov3") - hov3out_info = GetFileInfo(inputDir+"\\hov3out") - if (hov3_info=null & hov3out_info=null) then do - //get fields - fvector = GetDataVectors(view+"|",{"ID1", "AB_Flow_PCE", "BA_Flow_PCE", "Tot_Flow_PCE", "AB_Time", - "BA_Time", "Max_Time","AB_VOC","BA_VOC","Max_VOC","AB_V_Dist_T", - "BA_V_Dist_T","Tot_V_Dist_T","AB_VHT","BA_VHT","Tot_VHT", - "AB_Speed","BA_Speed","AB_VDF","BA_VDF","Max_VDF", - "AB_Flow_dan","BA_Flow_dan","AB_Flow_dat","BA_Flow_dat", - "AB_Flow_s2nn","BA_Flow_s2nn","AB_Flow_s2nh","BA_Flow_s2nh","AB_Flow_s2th","BA_Flow_s2th", - "AB_Flow_M1","BA_Flow_M1","AB_Flow_M2","BA_Flow_M2","AB_Flow_M3","BA_Flow_M3", - "AB_Flow_lhdn","BA_Flow_lhdn","AB_Flow_lhdt","BA_Flow_lhdt", - "AB_Flow_mhdn","BA_Flow_mhdn","AB_Flow_mhdt","BA_Flow_mhdt", - "AB_Flow_hhdn","BA_Flow_hhdn","AB_Flow_hhdt","BA_Flow_hhdt", - "AB_Flow","BA_Flow","Tot_Flow"},) - //create output file - view = CreateTable(viewNames[k], outputDir+"\\"+asignbin[k], "FFB", { - {"ID1" , "Integer (4 bytes)" , 10, null,"No", }, - {"AB_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Flow "}, - {"BA_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Flow "}, - {"Tot_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Flow "}, - {"AB_Time" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Travel Time "}, - {"BA_Time" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Travel Time "}, - {"Max_Time" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Loaded Time "}, - {"AB_VOC" , "Real (8 bytes)" , 15, 4 ,"No","AB Volume to Capacity Ratio "}, - {"BA_VOC" , "Real (8 bytes)" , 15, 4 ,"No","BA Volume to Capacity Ratio "}, - {"Max_VOC" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Volume to Capacity Ratio "}, - {"AB_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle miles or km of travel "}, - {"BA_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle miles or km of travel "}, - {"Tot_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle miles or km of travel "}, - {"AB_VHT" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle hours of travel "}, - {"BA_VHT" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle hours of travel "}, - {"Tot_VHT" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle hours of travel "}, - {"AB_Speed" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Speed "}, - {"BA_Speed" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Speed "}, - {"AB_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Volume Delay Function "}, - {"BA_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Volume Delay Function "}, - {"Max_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Link Volume Delay Function Value "}, - {"AB_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dan "}, - {"BA_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dan "}, - {"AB_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dat "}, - {"BA_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dat "}, - {"AB_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nn "}, - {"BA_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nn "}, - {"AB_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nh "}, - {"BA_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nh "}, - {"AB_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2th "}, - {"BA_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2th "}, - {"AB_Flow_M1" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M1 "}, - {"BA_Flow_M1" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M1 "}, - {"AB_Flow_M2" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M2 "}, - {"BA_Flow_M2" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M2 "}, - {"AB_Flow_M3" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for M3 "}, - {"BA_Flow_M3" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for M3 "}, - {"AB_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for lhd "}, - {"BA_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for lhd "}, - {"AB_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for mhd "}, - {"BA_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for mhd "}, - {"AB_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for hhd "}, - {"BA_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for hhd "}, - {"AB_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Veh Flow "}, - {"BA_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Veh Flow "}, - {"Tot_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Veh Flow "} - }) - SetView(view) - - //calculate and set values - for i = 1 to records do - rh = AddRecord(view, { - {"ID1" , fvector[1 ][i]}, - {"AB_Flow_PCE" , fvector[2 ][i]}, - {"BA_Flow_PCE" , fvector[3 ][i]}, - {"Tot_Flow_PCE" , fvector[4 ][i]}, - {"AB_Time" , fvector[5 ][i]}, - {"BA_Time" , fvector[6 ][i]}, - {"Max_Time" , fvector[7 ][i]}, - {"AB_VOC" , fvector[8 ][i]}, - {"BA_VOC" , fvector[9 ][i]}, - {"Max_VOC" , fvector[10][i]}, - {"AB_V_Dist_T" , fvector[11][i]}, - {"BA_V_Dist_T" , fvector[12][i]}, - {"Tot_V_Dist_T" , fvector[13][i]}, - {"AB_VHT" , fvector[14][i]}, - {"BA_VHT" , fvector[15][i]}, - {"Tot_VHT" , fvector[16][i]}, - {"AB_Speed" , fvector[17][i]}, - {"BA_Speed" , fvector[18][i]}, - {"AB_VDF" , fvector[19][i]}, - {"BA_VDF" , fvector[20][i]}, - {"Max_VDF" , fvector[21][i]}, - {"AB_Flow_dan" , fvector[22][i]}, - {"BA_Flow_dan" , fvector[23][i]}, - {"AB_Flow_dat" , fvector[24][i]}, - {"BA_Flow_dat" , fvector[25][i]}, - {"AB_Flow_s2nn" , fvector[26][i]}, - {"BA_Flow_s2nn" , fvector[27][i]}, - {"AB_Flow_s2nh" , fvector[28][i]}, - {"BA_Flow_s2nh" , fvector[29][i]}, - {"AB_Flow_s2th" , fvector[30][i]}, - {"BA_Flow_s2th" , fvector[31][i]}, - {"AB_Flow_M1" , fvector[32][i]}, - {"BA_Flow_M1" , fvector[33][i]}, - {"AB_Flow_M2" , fvector[34][i]}, - {"BA_Flow_M2" , fvector[35][i]}, - {"AB_Flow_M3" , fvector[36][i]}, - {"BA_Flow_M3" , fvector[37][i]}, - {"AB_Flow_lhd" , fvector[38][i] + fvector[40][i]}, - {"BA_Flow_lhd" , fvector[39][i] + fvector[41][i]}, - {"AB_Flow_mhd" , fvector[42][i] + fvector[44][i]}, - {"BA_Flow_mhd" , fvector[43][i] + fvector[45][i]}, - {"AB_Flow_hhd" , fvector[46][i] + fvector[48][i]}, - {"BA_Flow_hhd" , fvector[47][i] + fvector[49][i]}, - {"AB_Flow" , fvector[50][i]}, - {"BA_Flow" , fvector[51][i]}, - {"Tot_Flow" , fvector[52][i]} - }) - end - end - else do - fvector = GetDataVectors(view+"|",{"ID1", "AB_Flow_PCE", "BA_Flow_PCE", "Tot_Flow_PCE", "AB_Time", - "BA_Time", "Max_Time","AB_VOC","BA_VOC","Max_VOC","AB_V_Dist_T", - "BA_V_Dist_T","Tot_V_Dist_T","AB_VHT","BA_VHT","Tot_VHT", - "AB_Speed","BA_Speed","AB_VDF","BA_VDF","Max_VDF", - "AB_Flow_dan","BA_Flow_dan","AB_Flow_dat","BA_Flow_dat", - "AB_Flow_s2nn","BA_Flow_s2nn","AB_Flow_s2nh","BA_Flow_s2nh","AB_Flow_s2th","BA_Flow_s2th", - "AB_Flow_s3nn","BA_Flow_s3nn","AB_Flow_s3nh","BA_Flow_s3nh","AB_Flow_s3th","BA_Flow_s3th", - "AB_Flow_lhdn","BA_Flow_lhdn","AB_Flow_lhdt","BA_Flow_lhdt", - "AB_Flow_mhdn","BA_Flow_mhdn","AB_Flow_mhdt","BA_Flow_mhdt", - "AB_Flow_hhdn","BA_Flow_hhdn","AB_Flow_hhdt","BA_Flow_hhdt", - "AB_Flow","BA_Flow","Tot_Flow"},) - //create output file - view = CreateTable(viewNames[k], path+"\\"+asignbin[k], "FFB", { - {"ID1" , "Integer (4 bytes)" , 10, null,"No", }, - {"AB_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Flow "}, - {"BA_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Flow "}, - {"Tot_Flow_PCE" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Flow "}, - {"AB_Time" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Travel Time "}, - {"BA_Time" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Travel Time "}, - {"Max_Time" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Loaded Time "}, - {"AB_VOC" , "Real (8 bytes)" , 15, 4 ,"No","AB Volume to Capacity Ratio "}, - {"BA_VOC" , "Real (8 bytes)" , 15, 4 ,"No","BA Volume to Capacity Ratio "}, - {"Max_VOC" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Volume to Capacity Ratio "}, - {"AB_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle miles or km of travel "}, - {"BA_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle miles or km of travel "}, - {"Tot_V_Dist_T" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle miles or km of travel "}, - {"AB_VHT" , "Real (8 bytes)" , 15, 4 ,"No","AB vehicle hours of travel "}, - {"BA_VHT" , "Real (8 bytes)" , 15, 4 ,"No","BA vehicle hours of travel "}, - {"Tot_VHT" , "Real (8 bytes)" , 15, 4 ,"No","Total vehicle hours of travel "}, - {"AB_Speed" , "Real (8 bytes)" , 15, 4 ,"No","AB Loaded Speed "}, - {"BA_Speed" , "Real (8 bytes)" , 15, 4 ,"No","BA Loaded Speed "}, - {"AB_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Volume Delay Function "}, - {"BA_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Volume Delay Function "}, - {"Max_VDF" , "Real (8 bytes)" , 15, 4 ,"No","Maximum Link Volume Delay Function Value "}, - {"AB_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dan "}, - {"BA_Flow_dan" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dan "}, - {"AB_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for dat "}, - {"BA_Flow_dat" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for dat "}, - {"AB_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nn "}, - {"BA_Flow_s2nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nn "}, - {"AB_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2nh "}, - {"BA_Flow_s2nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2nh "}, - {"AB_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s2th "}, - {"BA_Flow_s2th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s2th "}, - {"AB_Flow_s3nn" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3nn "}, - {"BA_Flow_s3nn" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3nn "}, - {"AB_Flow_s3nh" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3nh "}, - {"BA_Flow_s3nh" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3nh "}, - {"AB_Flow_s3th" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for s3th "}, - {"BA_Flow_s3th" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for s3th "}, - {"AB_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for lhd "}, - {"BA_Flow_lhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for lhd "}, - {"AB_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for mhd "}, - {"BA_Flow_mhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for mhd "}, - {"AB_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","AB Flow for hhd "}, - {"BA_Flow_hhd" , "Real (8 bytes)" , 15, 4 ,"No","BA Flow for hhd "}, - {"AB_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link AB Veh Flow "}, - {"BA_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link BA Veh Flow "}, - {"Tot_Flow" , "Real (8 bytes)" , 15, 4 ,"No","Link Total Veh Flow "} - }) - SetView(view) - - //calculate and set values - for i = 1 to records do - rh = AddRecord(view, { - {"ID1" , fvector[1 ][i]}, - {"AB_Flow_PCE" , fvector[2 ][i]}, - {"BA_Flow_PCE" , fvector[3 ][i]}, - {"Tot_Flow_PCE" , fvector[4 ][i]}, - {"AB_Time" , fvector[5 ][i]}, - {"BA_Time" , fvector[6 ][i]}, - {"Max_Time" , fvector[7 ][i]}, - {"AB_VOC" , fvector[8 ][i]}, - {"BA_VOC" , fvector[9 ][i]}, - {"Max_VOC" , fvector[10][i]}, - {"AB_V_Dist_T" , fvector[11][i]}, - {"BA_V_Dist_T" , fvector[12][i]}, - {"Tot_V_Dist_T" , fvector[13][i]}, - {"AB_VHT" , fvector[14][i]}, - {"BA_VHT" , fvector[15][i]}, - {"Tot_VHT" , fvector[16][i]}, - {"AB_Speed" , fvector[17][i]}, - {"BA_Speed" , fvector[18][i]}, - {"AB_VDF" , fvector[19][i]}, - {"BA_VDF" , fvector[20][i]}, - {"Max_VDF" , fvector[21][i]}, - {"AB_Flow_dan" , fvector[22][i]}, - {"BA_Flow_dan" , fvector[23][i]}, - {"AB_Flow_dat" , fvector[24][i]}, - {"BA_Flow_dat" , fvector[25][i]}, - {"AB_Flow_s2nn" , fvector[26][i]}, - {"BA_Flow_s2nn" , fvector[27][i]}, - {"AB_Flow_s2nh" , fvector[28][i]}, - {"BA_Flow_s2nh" , fvector[29][i]}, - {"AB_Flow_s2th" , fvector[30][i]}, - {"BA_Flow_s2th" , fvector[31][i]}, - {"AB_Flow_s3nn" , fvector[32][i]}, - {"BA_Flow_s3nn" , fvector[33][i]}, - {"AB_Flow_s3nh" , fvector[34][i]}, - {"BA_Flow_s3nh" , fvector[35][i]}, - {"AB_Flow_s3th" , fvector[36][i]}, - {"BA_Flow_s3th" , fvector[37][i]}, - {"AB_Flow_lhd" , fvector[38][i] + fvector[40][i]}, - {"BA_Flow_lhd" , fvector[39][i] + fvector[41][i]}, - {"AB_Flow_mhd" , fvector[42][i] + fvector[44][i]}, - {"BA_Flow_mhd" , fvector[43][i] + fvector[45][i]}, - {"AB_Flow_hhd" , fvector[46][i] + fvector[48][i]}, - {"BA_Flow_hhd" , fvector[47][i] + fvector[49][i]}, - {"AB_Flow" , fvector[50][i]}, - {"BA_Flow" , fvector[51][i]}, - {"Tot_Flow" , fvector[52][i]} - }) - end - end - end // end for loop - - vws = GetViewNames() - for p = 1 to vws.length do - CloseView(vws[p]) - end - return(1) - - quit: - return(0) -EndMacro - - - - -Macro "create trip tables by VOT"(args) - - shared path, inputDir, outputDir - - inFiles={"Trip_EA.mtx","Trip_AM.mtx","Trip_MD.mtx","Trip_PM.mtx","Trip_EV.mtx"} - outFiles={"Trip_EA_VOT.mtx","Trip_AM_VOT.mtx","Trip_MD_VOT.mtx","Trip_PM_VOT.mtx","Trip_EV_VOT.mtx"} - inTableNames = {"SOV_GP", "SOV_PAY", "SR2_GP","SR2_HOV", "SR2_PAY", "SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} - outTableNames = { - "SOV_GP_LOW", "SOV_PAY_LOW", "SR2_GP_LOW","SR2_HOV_LOW", "SR2_PAY_LOW", "SR3_GP_LOW","SR3_HOV_LOW","SR3_PAY_LOW", - "SOV_GP_MED", "SOV_PAY_MED", "SR2_GP_MED","SR2_HOV_MED", "SR2_PAY_MED", "SR3_GP_MED","SR3_HOV_MED","SR3_PAY_MED", - "SOV_GP_HI", "SOV_PAY_HI", "SR2_GP_HI","SR2_HOV_HI", "SR2_PAY_HI", "SR3_GP_HI","SR3_HOV_HI","SR3_PAY_HI", - "lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"} - - for i = 1 to inFiles.length do - - //open person trip matrix currencies - inMatrix = OpenMatrix(outputDir+"\\"+inFiles[i], ) - inCurrencies = CreateMatrixCurrencies(inMatrix, , , ) - - dim curr_array[inTableNames.length] - for j = 1 to inTableNames.length do - curr_array[j] = CreateMatrixCurrency(inMatrix, inTableNames[j], ,, ) - end - - //create output trip table and matrix currencies for this time period - outMatrix = CopyMatrixStructure(curr_array, {{"File Name", outputDir+"\\"+outFiles[i]}, - {"Label", outFiles[i]}, - {"Tables",outTableNames}, - {"File Based", "Yes"}}) - SetMatrixCoreNames(outMatrix, outTableNames) - - outCurrencies= CreateMatrixCurrencies(outMatrix, , , ) - - // calculate output matrices - outCurrencies.SOV_GP_LOW := inCurrencies.SOV_GP * 0.3333333 - outCurrencies.SOV_GP_MED := inCurrencies.SOV_GP * 0.3333333 - outCurrencies.SOV_GP_HI := inCurrencies.SOV_GP * 0.3333333 - - outCurrencies.SOV_PAY_LOW := inCurrencies.PAY_GP * 0.3333333 - outCurrencies.SOV_PAY_MED := inCurrencies.PAY_GP * 0.3333333 - outCurrencies.SOV_PAY_HI := inCurrencies.PAY_GP * 0.3333333 - - outCurrencies.SR2_GP_LOW := inCurrencies.SR2_GP * 0.3333333 - outCurrencies.SR2_GP_MED := inCurrencies.SR2_GP * 0.3333333 - outCurrencies.SR2_GP_HI := inCurrencies.SR2_GP * 0.3333333 - - outCurrencies.SR2_HOV_LOW := inCurrencies.SR2_HOV * 0.3333333 - outCurrencies.SR2_HOV_MED := inCurrencies.SR2_HOV * 0.3333333 - outCurrencies.SR2_HOV_HI := inCurrencies.SR2_HOV * 0.3333333 - - outCurrencies.SR2_PAY_LOW := inCurrencies.SR2_PAY * 0.3333333 - outCurrencies.SR2_PAY_MED := inCurrencies.SR2_PAY * 0.3333333 - outCurrencies.SR2_PAY_HI := inCurrencies.SR2_PAY * 0.3333333 - - outCurrencies.SR3_GP_LOW := inCurrencies.SR3_GP * 0.3333333 - outCurrencies.SR3_GP_MED := inCurrencies.SR3_GP * 0.3333333 - outCurrencies.SR3_GP_HI := inCurrencies.SR3_GP * 0.3333333 - - outCurrencies.SR3_HOV_LOW := inCurrencies.SR3_HOV * 0.3333333 - outCurrencies.SR3_HOV_MED := inCurrencies.SR3_HOV * 0.3333333 - outCurrencies.SR3_HOV_HI := inCurrencies.SR3_HOV * 0.3333333 - - outCurrencies.SR3_PAY_LOW := inCurrencies.SR3_PAY * 0.3333333 - outCurrencies.SR3_PAY_MED := inCurrencies.SR3_PAY * 0.3333333 - outCurrencies.SR3_PAY_HI := inCurrencies.SR3_PAY * 0.3333333 - - outCurrencies.lhdn := inCurrencies.lhdn - outCurrencies.mhdn := inCurrencies.mhdn - outCurrencies.hhdn := inCurrencies.hhdn - outCurrencies.lhdt := inCurrencies.lhdt - outCurrencies.mhdt := inCurrencies.mhdt - outCurrencies.hhdt := inCurrencies.hhdt - - end - RunMacro("close all" ) - - Return(1) - quit: - Return(0) -EndMacro diff --git a/sandag_abm/src/main/gisdk/hwyskim.rsc b/sandag_abm/src/main/gisdk/hwyskim.rsc deleted file mode 100644 index 01576e0..0000000 --- a/sandag_abm/src/main/gisdk/hwyskim.rsc +++ /dev/null @@ -1,1257 +0,0 @@ -/*********************************************** -Hwy skim all - -This macro calls macro "Update highway network", which updates highway network with times -from last highway assignment, and then skims highway network by calling macro "hwy skim" for -the following modes: - -dant Drive-alone non-toll -dat Drive-alone toll -s2nh Shared-2 non-toll HOV -s2th Shared-2 toll HOV -s3nh Shared-3 non-toll HOV -s3th Shared-3 toll HOV -truck Truck - -***********************************************/ -Macro "Hwy skim all" (args) - - skimByVOT= args[1] - - if skimByVOT="false" then do - - da_vot=67.00 // $0.67 cents per minute VOT ($40.2 per hour) - s2_vot=67.00 - s3_vot=67.00 - lh_vot=67.00 - mh_vot=68.00 - hh_vot=89.00 - cv_vot=67.00 - - vot_array = {da_vot, s2_vot, s3_vot, lh_vot, mh_vot, hh_vot, cv_vot} - - ok=RunMacro("Update highway network", vot_array) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"dant",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"dat",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s2nh",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s2th",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s3nh",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s3th",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"cvn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"cvt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"lhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"lhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"mhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"mhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"hhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"hhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"truck",}) - if !ok then goto quit - - end - else do - - vot_bins = {"low", "med", "high"} - // da, s2, s3, lh, mh, hh, cv} - vot_by_bin = {{16.6, 16.6, 16.6, 67.0, 68.0, 89.0, 67.0}, - {33.3, 33.3, 33.3, 67.0, 68.0, 89.0, 67.0}, - { 100, 100, 100, 67.0, 68.0, 89.0, 67.0} - } - - for i = 1 to vot_bins.length do - - ok=RunMacro("Update highway network", vot_by_bin[i]) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"dant",vot_bins[i]}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"dat",vot_bins[i]}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s2nh",vot_bins[i]}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s2th",vot_bins[i]}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s3nh",vot_bins[i]}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"s3th",vot_bins[i]}) - if !ok then goto quit - -/* - // reliability for time periods (15-mins) - for tod=1 to 96 do - - if (tod=25 or tod=26 or tod=35 or tod=36) then do //AM Period - 30 min shoulders - ok=RunMacro("hwy skim time bins",{"dant",vot_bins[i],tod}) - if !ok then goto quit - - ok=RunMacro("hwy skim time bins",{"dat",vot_bins[i],tod}) - if !ok then goto quit - - ok=RunMacro("hwy skim time bins",{"s2nh",vot_bins[i],tod}) - if !ok then goto quit - - ok=RunMacro("hwy skim time bins",{"s2th",vot_bins[i],tod}) - if !ok then goto quit - - ok=RunMacro("hwy skim time bins",{"s3nh",vot_bins[i],tod}) - if !ok then goto quit - - ok=RunMacro("hwy skim time bins",{"s3th",vot_bins[i],tod}) - if !ok then goto quit - end - - end - */ - end - - // don't skim commercial vehicles or trucks by vot - ok=RunMacro("hwy skim",{"cvn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"cvt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"lhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"lhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"mhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"mhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"hhdn",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"hhdt",}) - if !ok then goto quit - - ok=RunMacro("hwy skim",{"truck",}) - if !ok then goto quit - - end - - return(1) - - quit: - return(0) - -EndMacro - -/******************************************************************************** - -Update highway network - -Updates highway line layer and network with fields from latest assignment flow tables. - - Arguments - 1 drive-alone value-of-time (cents/min) - 2 shared 2 value-of-time (cents/min) - 3 shared 3+ value-of-time (cents/min) - 4 light-heavy truck value-of-time (cents/min) - 5 medium-heavy truck value-of-time (cents/min) - 6 heavy-heavy truck value-of-time (cents/min) - 7 commercial vehicle value-of-time (cents/min) - - -The following fields are updated on the line layer: - -Field Description -------- --------------- -STM SOV time -HTM HOV time -SCST SOV generalized cost -H2CST Shared-2 generalized cost -H3CST Shared-3 generalized cost -LHCST Light-heavy truck generalized cost -MHCST Medium-heavy truck generalized cost -HHCST Heavy-heavy truck generalized cost -CVCST Heavy-heavy truck generalized cost - - -Each field is xxField_yy where: - - xx is AB or BA indicating direction - yy is period, as follows: - EA: Early AM - AM: AM peak - MD: Midday - PM: PM peak - EV: Evening - -Inputs: - input\hwy.dbd Highway line layer - input\hwy.net Highway network - output\hwyload_yy.bin Loaded flow table from assignment, one per period (yy) - -Outputs: - output\hwy.dbd Updated highway line layer - output\hwy.net Updated highway network - -********************************************************************************/ -Macro "Update highway network" (args) - - shared path, inputDir, outputDir - - da_vot= args[1] - s2_vot= args[2] - s3_vot= args[3] - lh_vot= args[4] - mh_vot= args[5] - hh_vot= args[6] - cv_vot= args[7] - - // input files - db_file = outputDir + "\\hwy.dbd" - net_file = outputDir + "\\hwy.net" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - db_node_lyr = db_file + "|" + node_lyr - - periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} - - da_vot=da_vot*60/100 //Convert to dollars per hour VOT so don't have to change gen cost function below - s2_vot=s2_vot*60/100 - s3_vot=s3_vot*60/100 - lh_vot=lh_vot*60/100 - mh_vot=mh_vot*60/100 - hh_vot=hh_vot*60/100 - cv_vot=cv_vot*60/100 - - //Recompute generalized cost using MSA cost in flow table, - for i = 1 to periods.length do - - flowTable = outputDir+"\\hwyload"+periods[i]+".bin" - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"sovtime"+periods[i] } - Opts.Global.Fields = {"ABSTM"+periods[i],"BASTM"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost" , - "BA_MSA_Cost" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"hovtime"+periods[i] } - Opts.Global.Fields = {"ABHTM"+periods[i],"BAHTM"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost" , - "BA_MSA_Cost" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } - Opts.Global.Fields = {"ABSCST"+periods[i],"BASCST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(da_vot)+"*60)" , - "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(da_vot)+"*60)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // Light-Heavy truck cost - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } - Opts.Global.Fields = {"ABLHCST"+periods[i],"BALHCST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(lh_vot)+"*60)" , - "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(lh_vot)+"*60)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // Medium-Heavy truck cost - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } - Opts.Global.Fields = {"ABMHCST"+periods[i],"BAMHCST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(mh_vot)+"*60)" , - "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(mh_vot)+"*60)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // Heavy-Heavy truck cost - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } - Opts.Global.Fields = {"ABHHCST"+periods[i],"BAHHCST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(hh_vot)+"*60)" , - "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(hh_vot)+"*60)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // Commercial vehicle cost - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"dajoin"+periods[i] } - Opts.Global.Fields = {"ABCVCST"+periods[i],"BACVCST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(cv_vot)+"*60)" , - "BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(cv_vot)+"*60)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"s2join"+periods[i] } - Opts.Global.Fields = {"ABH2CST"+periods[i],"BAH2CST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if (IHOV=3 or IHOV=4) then (AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s2_vot)+"*60)) else AB_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" , - "if (IHOV=3 or IHOV=4) then (BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s2_vot)+"*60)) else BA_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"s3join"+periods[i] } - Opts.Global.Fields = {"ABH3CST"+periods[i],"BAH3CST"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if IHOV=4 then (AB_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s3_vot)+"*60)) else AB_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" , - "if IHOV=4 then (BA_MSA_Cost + ((ITOLL3"+periods[i]+"/100)/"+String(s3_vot)+"*60)) else BA_MSA_Cost + (COST/100)/"+String(s2_vot)+"*60" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - // Total Reliability - The Dataview Set is a joined view of the link layer and the flow table, based on link ID - // calculate as square of link reliability - after skimming take square root of the total reliability - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"reliabilityjoin"+periods[i] } - Opts.Global.Fields = {"AB_TOTREL"+periods[i],"BA_TOTREL"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"pow(AB_MSA_Cost - AB_MSA_Time,2)", - "pow(BA_MSA_Cost - BA_MSA_Time,2)" } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - - //Now update the network with the calculated cost fields - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*STM"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABSTM"+periods[i],link_lyr+".BASTM"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*HTM"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABHTM"+periods[i],link_lyr+".BAHTM"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*SCST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABSCST"+periods[i],link_lyr+".BASCST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*H2CST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABH2CST"+periods[i],link_lyr+".BAH2CST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*H3CST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABH3CST"+periods[i],link_lyr+".BAH3CST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*LHCST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABLHCST"+periods[i],link_lyr+".BALHCST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*MHCST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABMHCST"+periods[i],link_lyr+".BAMHCST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*HHCST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABHHCST"+periods[i],link_lyr+".BAHHCST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*CVCST"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".ABCVCST"+periods[i],link_lyr+".BACVCST"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - // update total reliability fields - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*_TOTREL"+periods[i] - Opts.Global.Options.[Link Fields] = { {link_lyr+".AB_TOTREL"+periods[i],link_lyr+".BA_TOTREL"+periods[i] } } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - end - - - ok=1 - runmacro("close all") - quit: - - return(ok) - -EndMacro - -/******************************************************************************************** - -hwy skim - -Skim highway network for the following modes (passed as argument) - - -Mode Description Cost attribute ----- --------------------- -------------- -truck Truck SCST -lhdn Light-heavy-duty non-toll LHCST -mhdn Medium-heavy-duty non-toll MHCST -hhdn Heavy-heavy-duty non-toll HHCST -lhdt Light-heavy-duty toll LHCST -mhdt Medium-heavy-duty toll MHCST -hhdt Heavy-heavy-duty toll HHCST -cvn Commercial vehicle non-toll CVCST -cvt Commercial vehicle toll CVCST -dant Drive-alone non-toll SCST -dat Drive-alone toll SCST -s2nh Shared-2 non-toll HOV H2CST -s2th Shared-2 toll non-HOV H2CST -s3nh Shared-3 non-toll HOV H3CST -s3th Shared-3 toll HOV H3CST - -Note: dant skims also apply to shared-2 non-toll, non-HOV and shared 3+ non-toll, non-HOV - -v1.0 jef 3/30/2012 -v2.0 jef 5/10/2015 added value-of-time bins and commercial vehicle modes - -*/ -Macro "hwy skim" (args) - - shared path, inputDir, outputDir, mxzone - - mode=args[1] - - //vot_bin is the value-of-time bin that will be appended to each skim name; prepend "_" - if args[2]=null then vot_bin="" else vot_bin="_"+args[2] - - dim skimbyset1[3],skimbyset2[3] - - - // input files - db_file = outputDir + "\\hwy.dbd" - net_file = outputDir + "\\hwy.net" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - db_node_lyr = db_file + "|" + node_lyr - net = ReadNetwork(net_file) - - periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} - - for i = 1 to periods.length do - - skimbyset1 = null // second skim varaible (in addition to LENGTH) - skimbyset2 = null // third skim variable - skimbyset3 = null // fourth skim variable - - // The truck skim is used for heavy trip distribution - if mode = "truck" then do - - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!((ihov=1|ihov=4|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for exclusion link set - - set = "TrkToll"+periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (ihov=4 and (ITRUCK=1|ITRUCK>4))",) - if n > 0 then skimbyset1={vw_set, {"itoll"+periods[i]}} - - // skimbyset2 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imptrk"+periods[i]+vot_bin+".mtx" // output skim matrices - - end - - // The skims by weight class are used for truck toll diversion - else if mode = "lhdn" then do // light duty truck non-toll - - CostFld = "*LHCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!((ihov=1|ifc>7)&(ITRUCK<4|ITRUCK=7))" // query for lhd non-toll exclusion link set - - // skimbyset1 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - else if mode = "mhdn" then do // medium duty truck non-toll - - CostFld = "*MHCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!((ihov=1|ifc>7)&(ITRUCK<3|ITRUCK>5))" // query for mhd non-toll exclusion link set - - // skimbyset1 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - else if mode = "hhdn" then do // heavy duty truck non-toll - - CostFld = "*HHCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd non-toll exclusion link set - - // skimbyset1 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - - else if mode = "lhdt" then do - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7) & (ITRUCK<4|ITRUCK=7))" // query for lhd toll exclusion link set - - tollfield = "ITOLL2" // toll value - - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) - if n = 0 then excl_qry=null // reset value if no selection records - - // skimbyset1 = toll - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} - - // skimbyset2 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - else if mode = "mhdt" then do - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK<3|ITRUCK>5))" // query for mhd toll exclusion link set - - tollfield = "ITOLL2" // toll value - - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) - if n = 0 then excl_qry=null // reset value if no selection records - - // skimbyset1 = toll - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} - - // skimbyset2 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} - - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - else if mode = "hhdt" then do - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd toll exclusion link set - - tollfield = "ITOLL2" // toll value - - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) - if n = 0 then excl_qry=null // reset value if no selection records - - // skimbyset1 = toll - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} - - // skimbyset2 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imp"+mode+periods[i]+vot_bin+".mtx" // output skim matrices - end - - else if mode = "cvn" then do // commercial vehicles non-toll - CostFld = "*CVCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM" +periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!((ihov=1|ifc>7)&(ITRUCK=1|ITRUCK>4))" // query for hhd non-toll exclusion link set - - // skimbyset1 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "impcvn"+periods[i]+vot_bin+".mtx" // output skim matrices - end - - else if mode = "cvt" then do // commercial vehicle toll skims; uses same selection set as drive-alone toll - CostFld = "*CVCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several","Select * where "+excl_qry,) - if n = 0 then excl_qry=null //reset value if no selection records - - tollfield = "ITOLL2" // toll value - - // skimbyset1 = toll - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {tollfield + periods[i] }} - - // skimbyset2 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "impcvt"+periods[i]+vot_bin+".mtx" - end - else if mode = "dant" then do - - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) - - excl_qry = "!(ihov=1&ITRUCK<5)" // query for exclusion link set - - // skimbyset1 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset1={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "impdan"+periods[i]+vot_bin+".mtx" // output skim matrices - end - else if mode = "dat" then do - - CostFld = "*SCST"+periods[i] // minimizing cost field - SkimVar1 = "*STM"+periods[i] // first skim varaible (in addition to LENGTH) - - //excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))|ifc>7)&ITRUCK<5)" - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several","Select * where "+excl_qry,) - if n = 0 then excl_qry=null //reset value if no selection records - - // skimbyset1 = length on toll lanes - set = "datdst"+periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - //n = SelectByQuery(set, "Several", "Select * where ((ihov=4|((ihov=2|ihov=3)&(itoll"+periods[i]+">0&abln"+periods[i]+"<9)))&ITRUCK<5)",) - n = SelectByQuery(set, "Several", "Select * where ((ihov=4|((ihov=2|ihov=3)&(abln"+periods[i]+"<9)))&ITRUCK<5)",) - if n > 0 then skimbyset1={vw_set, {"Length"}} - - // skimbyset2 = cost - set = mode + periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} - - // skimbyset3 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "impdat"+periods[i]+vot_bin+".mtx" - end - else if mode = "s2nh" then do - - CostFld = "*H2CST"+periods[i] // minimizing cost field - SkimVar1 = "*HTM"+periods[i] - - excl_qry ="!((ihov=1|(ihov=2&abln"+periods[i]+" <9)|ifc>7)&ITRUCK<5)"//initialize the value - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several","Select * where "+excl_qry,) - if n = 0 then excl_qry=null //reset value if no selection records - - set = "s2hdst" + periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (abln"+periods[i]+"<9 and ihov=2 and ITRUCK<5)",) - if n > 0 then skimbyset2={vw_set, {"Length"}} - - // skimbyset3 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imps2nh"+periods[i]+vot_bin+".mtx" - end - else if mode = "s2th" then do - - CostFld = "*H2CST"+periods[i] // minimizing cost field - SkimVar1 ="*HTM"+periods[i] - - excl_qry = "!(((ihov=1|(ihov=2&abln"+periods[i]+"<9)|ihov=4|(ihov=3&itoll"+periods[i]+">0&abln"+periods[i]+"<9))|ifc>7)&ITRUCK<5)" - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several","Select * where " + excl_qry,) - if n = 0 then excl_qry=null //reset value if no selection records - - set = "s2tdst"+periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (((abln"+periods[i]+"<9 & ihov=2) | (ihov=4 | (ihov=3 & itoll"+periods[i]+" >0 & abln"+periods[i]+"< 9)))& ITRUCK<5)",) - if n > 0 then skimbyset1={vw_set, {"Length"}} - - set = "s2t"+periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where ((ihov=4|(ihov=3 & itoll"+periods[i]+" >0 & abln"+periods[i]+" < 9)) & ITRUCK<5)",) - if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} - - // skimbyset3 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imps2th"+periods[i]+vot_bin+".mtx" - - end - else if mode = "s3nh" then do - - CostFld = "*H3CST"+periods[i] // minimizing cost field - SkimVar1 = "*HTM" +periods[i] - - excl_qry = "!((ihov=1|((ihov=2|ihov=3)&abln"+periods[i]+"<9)|ifc>7)& ITRUCK<5)" - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) - if n = 0 then excl_qry=null - - set = "s3hdst"+periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (abln"+periods[i]+"<9 & (ihov=2 | ihov=3) & ITRUCK<5)",) - if n > 0 then skimbyset2={vw_set, {"Length"}} - - // skimbyset3 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imps3nh"+periods[i]+vot_bin+".mtx" - - end - else if mode = "s3th" then do - - CostFld = "*H3CST" + periods[i] // minimizing cost field - SkimVar1 = "*HTM" + periods[i] - - excl_qry = "(abln"+periods[i]+"=9 | ITRUCK >4)" - set = mode + periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where " + excl_qry,) - if n = 0 then excl_qry=null - - set = "s3tdst" + periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (((abln"+periods[i]+"<9 and (ihov=2 or ihov=3)) or ihov=4)and ITRUCK <5)",) - if n > 0 then skimbyset1={vw_set, {"Length"}} - - set = "s3t" + periods[i] - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where (ihov=4 and ITRUCK <5)",) - if n > 0 then skimbyset2={vw_set, {"itoll"+periods[i]}} - - // skimbyset3 = reliability - set = mode - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where 1=1",) // for all links - if n > 0 then skimbyset3={vw_set, {"*_TOTREL" + periods[i]}} - - skimmat = "imps3th"+periods[i]+vot_bin+".mtx" - end - - - //delete existing skim file - ok=RunMacro("SDdeletefile",{outputDir+"\\"+skimmat}) - if !ok then goto quit - - //skim network - set = "AllLinks" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectAll(set) - NetOpts = null - NetOpts.[Link ID] = link_lyr+".ID" - NetOpts.[Type] = "Enable" - ChangeLinkStatus(net,vw_set, NetOpts) // first enable all links - - if excl_qry<>null then do - set = "toll" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where "+excl_qry,) - NetOpts = null - NetOpts.[Link ID] = link_lyr+".ID" - NetOpts.[Type] = "Disable" - ChangeLinkStatus(net,vw_set, NetOpts) // disable exclusion query - end - - Opts = null - Opts.Input.Network = net_file - Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Centroids", "select * where ID <= " + i2s(mxzone)} - Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Centroids"} - Opts.Input.[Via Set] = {db_node_lyr, node_lyr} - Opts.Field.Minimize = CostFld - Opts.Field.Nodes = node_lyr + ".ID" - Opts.Field.[Skim Fields]={{"Length","All"},{SkimVar1,"All"}} - - // provide skimset - if skimbyset1 <> null then do - if skimbyset2 <> null then do - if skimbyset3 <> null then - Opts.Field.[Skim by Set]={skimbyset1,skimbyset2,skimbyset3} - else - Opts.Field.[Skim by Set]={skimbyset1,skimbyset2} - end - else if skimbyset3 <> null then - Opts.Field.[Skim by Set]={skimbyset1,skimbyset3} - else - Opts.Field.[Skim by Set]={skimbyset1} - end - else if skimbyset2 <> null then do - if skimbyset3 <> null then - Opts.Field.[Skim by Set]={skimbyset2,skimbyset3} - else - Opts.Field.[Skim by Set]={skimbyset2} - end - else if skimbyset3 <> null then - Opts.Field.[Skim by Set]={skimbyset3} - - //end of previous if string - if (mode = "lhdn" | mode = "mhdn" | mode = "hhdn" | mode = "lhdt" | mode = "mhdt" | mode = "hhdt") then - if (mode = "lhdn" | mode = "mhdn" | mode = "hhdn") then - Opts.Output.[Output Matrix].Label = "impedance truck" - else if (mode = "lhdt" | mode = "mhdt" | mode = "hhdt") then - Opts.Output.[Output Matrix].Label = "impedance truck toll" - else - Opts.Output.[Output Matrix].Label = "congested " + mode + " impedance" - Opts.Output.[Output Matrix].Compression = 0 //uncompressed, for version 4.8 plus - Opts.Output.[Output Matrix].[File Name] = outputDir + "\\"+skimmat - - RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim","TCSPMAT: "+skimmat+"; "+CostFld}) - ok = RunMacro("TCB Run Procedure", 1, "TCSPMAT", Opts) - if !ok then goto quit - - // STEP 2: Intrazonal added by Ziying Ouyang, June 3, 2009 - // mtxcore={"Length (Skim)"}+{SkimVar1[i]+" (Skim)"}+{SkimVar2[i]+" (Skim)"}+{SkimVar3[i]+" (Skim)"} - mtxcore={"Length (Skim)"}+{SkimVar1+" (Skim)"} - for j = 1 to mtxcore.length do - Opts = null - Opts.Global.Factor = 0.5 - Opts.Global.Neighbors = 3 - Opts.Global.Operation = 1 - Opts.Global.[Treat Missing] = 2 - Opts.Input.[Matrix Currency] = {outputDir + "\\"+skimmat,mtxcore[j], , } - RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim","Intrazonal: "+skimmat+"; "+mtxcore[j]}) - ok = RunMacro("TCB Run Procedure", j, "Intrazonal", Opts) - if !ok then goto quit - end - - // take square root of the reliability which is sum of square of link reliability - write code after generating skims - todo - mtxcore = mode+" - *_TOTREL"+periods[i] - m=OpenMatrix(outputDir + "\\"+skimmat,) - mc=CreateMatrixCurrency(m,mtxcore,,,) - mc:=Nz(mc) // zero out intra-zonal values - mc:=sqrt(mc) // take square root - - end - - ok=1 - runmacro("close all") - quit: - - return(ok) - -EndMacro - -/* - -hwy skim 15 mins time slices - -Skim reliability with following shift variables: - -Variable Time Bin Estimate-Freeway Estimate-Arterial --------- -------- ---------------- ----------------- -BeforeAM.Step1 32 -0.0183 -0.0054 -BeforeAM.Step2 29 0.0092 -0.0032 -BeforeAM.Step3 26 0.0107 0.0030 -BeforeAM.Step4 20 -0.0019 0.0055 -AfterAM.Step1 32 -0.0082 -0.0009 -AfterAM.Step2 36 0.0000 0.0000 -AfterAM.Step3 39 0.0000 0.0000 -BeforePM.Step1 70 -0.0067 0.0011 -BeforePM.Step2 66 -0.0028 0.0000 -BeforePM.Step3 62 0.0094 -0.0018 -BeforePM.Step4 58 0.0000 0.0000 -AfterPM.Step1 70 -0.0077 -0.0079 -AfterPM.Step2 71 0.0000 0.0025 -AfterPM.Step3 79 0.0075 0.0037 - -{"EA","AM","MD","PM","EV1","EV2"} = {{15,24},{25,36},{37,62},{63,76},{77,96},{0,14}} - -*/ - -macro "hwy skim time bins" (args) - - shared path, inputDir, outputDir, mxzone - - mode = args[1] - - //vot_bin is the value-of-time bin that will be appended to each skim name; prepend "_" - if args[2]=null then vot_bin="" else vot_bin="_"+args[2] - timebin=args[3] - - // period thresholds - Peak_AM = 32 - Low_MD = 41 - Peak_PM = 70 - - // shift variable settings - // {beforeAM_step1, beforeAM_step2,beforeAM_step3,beforeAM_step4,afterAM_step1,afterAM_step2,afterAM_step3,beforePM_step1,beforePM_step2,beforePM_step3,beforePM_step4,afterPM_step1,afterPM_step2,afterPM_step3} - time_bins = {Peak_AM,29,26,20,Peak_AM,36,39,Peak_PM,66,62,58,Peak_PM,71,79} - factor_freeway = {-0.0183,0.0092,0.0107,-0.0019,-0.0082,0.0000,0.0000,-0.0067,-0.0028,0.0094,0.0000,-0.0077,0.0000,0.0075} - factor_arterial = {-0.0054,-0.0032,0.0030,0.0055,-0.0009,0.0000,0.0000,0.0011,0.0000,-0.0018,0.0000,-0.0079,0.0025,0.0037} - - facility_type = {"freeway","arterial","ramp","other"} // freeway (IFC=1), arterial (IFC=2,3), ramp (IFC=8,9), other (IFC=4,5,6,7) - periods = {"_EA", "_AM", "_MD", "_PM", "_EV"} - - // lower and upper bounds of IFC for respective facility type = {freeway, arterial, ramp, other} - lwr_bound = {"1","2","8","4"} - upr_bound = {"1","3","9","7"} - - // input files - db_file = outputDir + "\\hwy.dbd" - net_file = outputDir + "\\hwy.net" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - db_node_lyr = db_file + "|" + node_lyr - net = ReadNetwork(net_file) - - //Recompute total reliability for 15 mins time slices - for fac_type=1 to facility_type.length do - - if fac_type=1 then factor=factor_freeway - else factor=factor_arterial - - // corresponding model time period - if timebin>=15 and timebin<=24 then period = periods[1] // EA - if timebin>=25 and timebin<=36 then period = periods[2] // AM - if timebin>=37 and timebin<=62 then period = periods[3] // MD - if timebin>=63 and timebin<=76 then period = periods[4] // PM - if (timebin>=1 and timebin<=14) or (timebin>=77 and timebin<=96) then period = periods[5] // EV - - // initialize shift variables - beforeAM_step1=0 - beforeAM_step2=0 - beforeAM_step3=0 - beforeAM_step4=0 - afterAM_step1=0 - afterAM_step2=0 - afterAM_step3=0 - beforePM_step1=0 - beforePM_step2=0 - beforePM_step3=0 - beforePM_step4=0 - afterPM_step1=0 - afterPM_step2=0 - afterPM_step3=0 - - // calculate shift reliability - - // before AM - if timebintime_bins[5] and timebintime_bins[6] and timebintime_bins[7] and timebin=Low_MD and timebin=Low_MD and timebin=Low_MD and timebin=Low_MD and timebintime_bins[12] then afterPM_step1=factor[12]*(timebin-time_bins[12]) - if timebin>time_bins[13] then afterPM_step2=factor[13]*(timebin-time_bins[13]) - if timebin>time_bins[14] then afterPM_step3=factor[14]*(timebin-time_bins[14]) - - // calculate total shift reliability factors - shift_factor = beforeAM_step1 + beforeAM_step2 + beforeAM_step3 + beforeAM_step4 + - afterAM_step1 + afterAM_step2 + afterAM_step3 + - beforePM_step1 + beforePM_step2 + beforePM_step3 + beforePM_step4 + - afterPM_step1 + afterPM_step2 + afterPM_step3 - - // expressions to calculate AB/BA shift reliability - squareof link reliability - expression_AB = "AB_TOTREL" + period + "+ pow(" + String(shift_factor) + "*Length*AB_Time,2)" - expression_BA = "BA_TOTREL" + period + "+ pow(" + String(shift_factor) + "*Length*BA_Time,2)" - - flowTable = outputDir+"\\hwyload"+period+".bin" - - query = "Select * where IFC >= " + lwr_bound[fac_type] + " and IFC <= "+upr_bound[fac_type] - - // Total Reliability - The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"reliabilityjoin"+period, "selection", query} - Opts.Global.Fields = {"AB_TOTREL"+period,"BA_TOTREL"+period} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {expression_AB, - expression_BA } - ret_value = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ret_value then goto quit - end - - // update total reliability fields - Opts = null - Opts.Input.Database = db_file - Opts.Input.Network = net_file - Opts.Input.[Link Set] = {db_file+"|"+link_lyr, link_lyr} - Opts.Global.[Fields Indices] = "*_TOTREL"+period - Opts.Global.Options.[Link Fields] = { {link_lyr+".AB_TOTREL"+period,link_lyr+".BA_TOTREL"+period} } - Opts.Global.Options.Constants = {1} - ret_value = RunMacro("TCB Run Operation", "Update Network Field", Opts) - if !ret_value then goto quit - - // settings for skim - if mode = "dant" then do - CostFld = "*SCST"+period // minimizing cost field - excl_qry = "!(ihov=1&ITRUCK<5)" // query for exclusion link set - skimmat = "impdan_"+String(timebin)+vot_bin+".mtx" // output skim matrices - end - else if mode = "dat" then do - CostFld = "*SCST"+period // minimizing cost field - excl_qry = "!(((ihov=1|ihov=4|((ihov=2|ihov=3)&(itoll"+period+">0&abln"+period+"<9)))|ifc>7)&ITRUCK<5)" - skimmat = "impdat_"+String(timebin)+vot_bin+".mtx" - end - else if mode = "s2nh" then do - CostFld = "*H2CST"+period // minimizing cost field - excl_qry ="!((ihov=1|(ihov=2&abln"+period+" <9)|ifc>7)&ITRUCK<5)"//initialize the value - skimmat = "imps2nh"+String(timebin)+vot_bin+".mtx" - end - else if mode = "s2th" then do - CostFld = "*H2CST"+period // minimizing cost field - excl_qry = "!(((ihov=1|(ihov=2&abln"+period+"<9)|ihov=4|(ihov=3&itoll"+period+">0&abln"+period+"<9))|ifc>7)&ITRUCK<5)" - skimmat = "imps2th"+String(timebin)+vot_bin+".mtx" - end - else if mode = "s3nh" then do - CostFld = "*H3CST"+period // minimizing cost field - excl_qry = "!((ihov=1|((ihov=2|ihov=3)&abln"+period+"<9)|ifc>7)& ITRUCK<5)" - skimmat = "imps3nh"+String(timebin)+vot_bin+".mtx" - end - else if mode = "s3th" then do - CostFld = "*H3CST"+period // minimizing cost field - excl_qry = "(abln"+period+"=9 | ITRUCK >4)" - skimmat = "imps3th"+String(timebin)+vot_bin+".mtx" - end - - //skim network - set = "AllLinks" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectAll(set) - NetOpts = null - NetOpts.[Link ID] = link_lyr+".ID" - NetOpts.[Type] = "Enable" - ChangeLinkStatus(net,vw_set, NetOpts) // first enable all links - - if excl_qry<>null then do - set = "toll" - vw_set = link_lyr + "|" + set - SetLayer(link_lyr) - n = SelectByQuery(set, "Several", "Select * where "+excl_qry,) - NetOpts = null - NetOpts.[Link ID] = link_lyr+".ID" - NetOpts.[Type] = "Disable" - ChangeLinkStatus(net,vw_set, NetOpts) // disable exclusion query - end - - Opts = null - Opts.Input.Network = net_file - Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Centroids", "select * where ID <= " + i2s(mxzone)} - Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Centroids"} - Opts.Input.[Via Set] = {db_node_lyr, node_lyr} - Opts.Field.Minimize = CostFld - Opts.Field.Nodes = node_lyr + ".ID" - //Opts.Field.[Skim Fields].["*_TOTREL"+period]="All" - Opts.Field.[Skim Fields]={{"*_TOTREL"+period,"All"}} - Opts.Output.[Output Matrix].Label = "reliability " + mode - - Opts.Output.[Output Matrix].Compression = 0 //uncompressed, for version 4.8 plus - Opts.Output.[Output Matrix].[File Name] = outputDir + "\\"+skimmat - - RunMacro("HwycadLog",{"hwyskim.rsc: hwy skim time bins","TCSPMAT: "+skimmat+"; "+CostFld}) - ok = RunMacro("TCB Run Procedure", 1, "TCSPMAT", Opts, &Ret) - if !ok then goto quit - - // update skims by taking square root of reliability - mtxcore = "*_TOTREL"+period + " (Skim)" - m=OpenMatrix(outputDir + "\\"+skimmat,) - mc=CreateMatrixCurrency(m,mtxcore,,,) - mc:=Nz(mc) // zero out intra-zonal values - mc:=sqrt(mc) // take square root - - ok=1 - runmacro("close all") - quit: - - return(ok) - -EndMacro - diff --git a/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc b/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc deleted file mode 100644 index e605f73..0000000 --- a/sandag_abm/src/main/gisdk/matrixPrecisionReduction.rsc +++ /dev/null @@ -1,51 +0,0 @@ -/********************************************************************************************************** -Reduce matrix precision -About: - Script to reduce matrix precision. Precision defined in property file. - Author: Wu Sun, SANDAG - Developed: May 2015 - -Note: -Aggregate models such as truck, commercial vehicle, and EI models tend to have fake precisions. -The fake precisions make these matrices very large. -When matrixes are loaded into ABM database, it takes a large amount of DB space. -This scripts is to reduce fake precisions to make space managable. -**********************************************************************************************************/ -Macro "reduce matrix precision"(dir, mat, precision) - - //get matrix info - m = OpenMatrix(dir+"//"+mat, ) - coreNames = GetMatrixCoreNames(m) - numCores=coreNames.length - - //initialize arrays - dim sum[numCores] - dim rsum[numCores] - - //get matrix currency - currency=RunMacro("set input matrix currencies",dir, mat) - - for i = 1 to numCores do - //zero out null cells - currency[i]:=Nz(currency[i]) - - //sums of matrix by core - marginal_sums = GetMatrixMarginals(currency[i], "Sum", "row" ) - sum[i]=Sum(marginal_sums) - RunMacro("HwycadLog",{"matrixPrecisionReduction.rsc:","sum of "+coreNames[i]+" before reduction:"+r2s(sum[i])}) - - //reduce matrix precision using matrix expression. - expr="if(["+coreNames[i]+"]<"+precision+") then 0.0 else ["+coreNames[i]+"]" - EvaluateMatrixExpression(currency[i], expr,,, ) - rmarginal_sums = GetMatrixMarginals(currency[i], "Sum", "row" ) - rsum[i]=Sum(rmarginal_sums) - RunMacro("HwycadLog",{"matrixPrecisionReduction.rsc:","sum of "+coreNames[i]+" after reduction:"+r2s(rsum[i])}) - end - - //scale up reduced matrix to orirginal sum - for i = 1 to numCores do - currency[i]:=currency[i]*sum[i]/rsum[i] - end - -EndMacro - diff --git a/sandag_abm/src/main/gisdk/parameter.rsc b/sandag_abm/src/main/gisdk/parameter.rsc deleted file mode 100644 index d569002..0000000 --- a/sandag_abm/src/main/gisdk/parameter.rsc +++ /dev/null @@ -1,8 +0,0 @@ -macro "parameters" -shared mxzone,mxtap,mxext,mxlink,mxrte -mxzone=4996 -mxtap=2500 -mxext=12 -mxrte=380 -mxlink=41000 -endmacro \ No newline at end of file diff --git a/sandag_abm/src/main/gisdk/sandag_abm.lst b/sandag_abm/src/main/gisdk/sandag_abm.lst deleted file mode 100644 index 07cd1e8..0000000 --- a/sandag_abm/src/main/gisdk/sandag_abm.lst +++ /dev/null @@ -1,22 +0,0 @@ -${workpath}\\${year}\\gisdk\\dbox.rsc -${workpath}\\${year}\\gisdk\\sandag_abm_master.rsc -${workpath}\\${year}\\gisdk\\parameter.rsc -${workpath}\\${year}\\gisdk\\SandagCommon.rsc -${workpath}\\${year}\\gisdk\\createhwynet.rsc -${workpath}\\${year}\\gisdk\\hwyassign.rsc -${workpath}\\${year}\\gisdk\\hwyskim.rsc -${workpath}\\${year}\\gisdk\\createtrnroutes.rsc -${workpath}\\${year}\\gisdk\\trnskim.rsc -${workpath}\\${year}\\gisdk\\trnassign.rsc -${workpath}\\${year}\\gisdk\\commVehGen.rsc -${workpath}\\${year}\\gisdk\\commVehDist.rsc -${workpath}\\${year}\\gisdk\\commVehTOD.rsc -${workpath}\\${year}\\gisdk\\commVehDiversion.rsc -${workpath}\\${year}\\gisdk\\createtodtables.rsc -${workpath}\\${year}\\gisdk\\externalInternal.rsc -${workpath}\\${year}\\gisdk\\TruckModel.rsc -${workpath}\\${year}\\gisdk\\create_LUZ_Skims.rsc -${workpath}\\${year}\\gisdk\\sandag_abm_outputs.rsc -${workpath}\\${year}\\gisdk\\exportTCData.rsc -${workpath}\\${year}\\gisdk\\Utilities.rsc -${workpath}\\${year}\\gisdk\\matrixPrecisionReduction.rsc diff --git a/sandag_abm/src/main/gisdk/sandag_abm_generic.lst b/sandag_abm/src/main/gisdk/sandag_abm_generic.lst deleted file mode 100644 index 2f8b0f4..0000000 --- a/sandag_abm/src/main/gisdk/sandag_abm_generic.lst +++ /dev/null @@ -1,23 +0,0 @@ -${workpath}\\gisdk\\dbox.rsc -${workpath}\\gisdk\\gui.rsc -${workpath}\\gisdk\\sandag_abm_master.rsc -${workpath}\\gisdk\\parameter.rsc -${workpath}\\gisdk\\SandagCommon.rsc -${workpath}\\gisdk\\createhwynet.rsc -${workpath}\\gisdk\\hwyassign.rsc -${workpath}\\gisdk\\hwyskim.rsc -${workpath}\\gisdk\\createtrnroutes.rsc -${workpath}\\gisdk\\trnskim.rsc -${workpath}\\gisdk\\trnassign.rsc -${workpath}\\gisdk\\commVehGen.rsc -${workpath}\\gisdk\\commVehDist.rsc -${workpath}\\gisdk\\commVehTOD.rsc -${workpath}\\gisdk\\commVehDiversion.rsc -${workpath}\\gisdk\\createtodtables.rsc -${workpath}\\gisdk\\externalInternal.rsc -${workpath}\\gisdk\\TruckModel.rsc -${workpath}\\gisdk\\create_LUZ_Skims.rsc -${workpath}\\gisdk\\sandag_abm_outputs.rsc -${workpath}\\gisdk\\exportTCData.rsc -${workpath}\\gisdk\\Utilities.rsc -${workpath}\\gisdk\\matrixPrecisionReduction.rsc diff --git a/sandag_abm/src/main/gisdk/sandag_abm_master.rsc b/sandag_abm/src/main/gisdk/sandag_abm_master.rsc deleted file mode 100644 index ccd497b..0000000 --- a/sandag_abm/src/main/gisdk/sandag_abm_master.rsc +++ /dev/null @@ -1,411 +0,0 @@ -Macro "Run SANDAG ABM" - - RunMacro("TCB Init") - - shared path, inputDir, outputDir, inputTruckDir, mxzone, mxtap, mxext,mxlink,mxrte,scenarioYear,version - - - sample_rate = { 0.2, 0.5, 1.0 } - max_iterations=sample_rate.length //number of feedback loops - skimByVOT = "true" - assignByVOT = "true" - - path = "${workpath}" - - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","*********Model Run Starting************"}) - - path_parts = SplitPath(path) - path_no_drive = path_parts[2]+path_parts[3] - drive=path_parts[1] - path_forward_slash = Substitute(path_no_drive, "\\", "/", ) - - inputDir = path+"\\input" - outputDir = path+"\\output" - inputTruckDir = path+"\\input_truck" - - SetLogFileName(path+"\\logFiles\\tclog.xml") - SetReportFileName(path+"\\logFiles\\tcreport.xml") - - RunMacro("parameters") - - // read properties from sandag_abm.properties in /conf folder - properties = "\\conf\\sandag_abm.properties" - sample_rate = RunMacro("read properties array",properties,"sample_rates", "S") - max_iterations=sample_rate.length //number of feedback loops - scenarioYear = RunMacro("read properties",properties,"scenarioYear", "S") - skipCopyWarmupTripTables = RunMacro("read properties",properties,"RunModel.skipCopyWarmupTripTables", "S") - skipCopyBikeLogsum = RunMacro("read properties",properties,"RunModel.skipCopyBikeLogsum", "S") - skipCopyWalkImpedance= RunMacro("read properties",properties,"RunModel.skipCopyWalkImpedance", "S") - skipWalkLogsums= RunMacro("read properties",properties,"RunModel.skipWalkLogsums", "S") - skipBikeLogsums= RunMacro("read properties",properties,"RunModel.skipBikeLogsums", "S") - skipBuildHwyNetwork = RunMacro("read properties",properties,"RunModel.skipBuildHwyNetwork", "S") - skipBuildTransitNetwork= RunMacro("read properties",properties,"RunModel.skipBuildTransitNetwork", "S") - startFromIteration = s2i(RunMacro("read properties",properties,"RunModel.startFromIteration", "S")) - skipHighwayAssignment = RunMacro("read properties array",properties,"RunModel.skipHighwayAssignment", "S") - skipHighwaySkimming = RunMacro("read properties array",properties,"RunModel.skipHighwaySkimming", "S") - skipTransitSkimming = RunMacro("read properties array",properties,"RunModel.skipTransitSkimming", "S") - skipCoreABM = RunMacro("read properties array",properties,"RunModel.skipCoreABM", "S") - skipOtherSimulateModel = RunMacro("read properties array",properties,"RunModel.skipOtherSimulateModel", "S") - skipSpecialEventModel = RunMacro("read properties array",properties,"RunModel.skipSpecialEventModel", "S") - skipCTM = RunMacro("read properties array",properties,"RunModel.skipCTM", "S") - skipEI = RunMacro("read properties array",properties,"RunModel.skipEI", "S") - skipTruck = RunMacro("read properties array",properties,"RunModel.skipTruck", "S") - skipTripTableCreation = RunMacro("read properties array",properties,"RunModel.skipTripTableCreation", "S") - skipFinalHighwayAssignment = RunMacro("read properties",properties,"RunModel.skipFinalHighwayAssignment", "S") - skipFinalTransitAssignment = RunMacro("read properties",properties,"RunModel.skipFinalTransitAssignment", "S") - skipFinalHighwaySkimming = RunMacro("read properties",properties,"RunModel.skipFinalHighwaySkimming", "S") - skipFinalTransitSkimming = RunMacro("read properties",properties,"RunModel.skipFinalTransitSkimming", "S") - skipLUZSkimCreation = RunMacro("read properties",properties,"RunModel.skipLUZSkimCreation", "S") - skipDataExport = RunMacro("read properties",properties,"RunModel.skipDataExport", "S") - skipDataLoadRequest = RunMacro("read properties",properties,"RunModel.skipDataLoadRequest", "S") - skipDeleteIntermediateFiles = RunMacro("read properties",properties,"RunModel.skipDeleteIntermediateFiles", "S") - precision = RunMacro("read properties",properties,"RunModel.MatrixPrecision", "S") - minSpaceOnC=RunMacro("read properties",properties,"RunModel.minSpaceOnC", "S") - - // Swap Server Configurations - runString = path+"\\bin\\serverswap.bat "+drive+" "+path_no_drive+" "+path_forward_slash - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Server Config Swap: "+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run ServerSwap", runString) - if !ok then goto quit - ok=RunMacro("find String","\\logFiles\\serverswap.log","FATAL") - if !ok then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","ServerSwap failed! Open logFiles/serverswap.log for details."}) - goto quit - end - - //Update year specific properties - runString = path+"\\bin\\updateYearSpecificProps.bat "+drive+" "+path_no_drive+" "+path_forward_slash - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Update year specific properties: "+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Update Year Specific Properties", runString) - if !ok then goto quit - - - //check free space on C drive - runString = path+"\\bin\\checkFreeSpaceOnC.bat "+minSpaceOnC - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Checking if there is enough space on C drive: "+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Check space on C drive", runString) - - //check AT and Transit networks consistency - runString = path+"\\bin\\checkAtTransitNetworkConsistency.cmd "+drive+" "+path_forward_slash - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Checking if AT and Transit Networks are consistent: "+" "+runString}) - RunProgram(runString, ) - ok=RunMacro("find String","\\logFiles\\AtTransitCheck_event.log","FATAL") - if !ok then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","AT and Transit network consistency chekcing failed! Open AtTransitCheck_event.log for details."}) - goto quit - end - - // copy bike logsums from input to output folder - if skipCopyBikeLogsum = "false" then do - CopyFile(inputDir+"\\bikeMgraLogsum.csv", outputDir+"\\bikeMgraLogsum.csv") - CopyFile(inputDir+"\\bikeTazLogsum.csv", outputDir+"\\bikeTazLogsum.csv") - end - if skipBikeLogsums = "false" then do - runString = path+"\\bin\\runSandagBikeLogsums.cmd "+drive+" "+path_forward_slash - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run create AT logsums and walk impedances"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run AT-Logsums", runString) - if !ok then goto quit - end - - // copy walk impedance from input to output folder - if skipCopyWalkImpedance = "false" then do - CopyFile(inputDir+"\\walkMgraEquivMinutes.csv", outputDir+"\\walkMgraEquivMinutes.csv") - CopyFile(inputDir+"\\walkMgraTapEquivMinutes.csv", outputDir+"\\walkMgraTapEquivMinutes.csv") - end - if skipWalkLogsums = "false" then do - runString = path+"\\bin\\runSandagWalkLogsums.cmd "+drive+" "+path_forward_slash - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run create AT logsums and walk impedances"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run AT-Logsums", runString) - if !ok then goto quit - end - - // copy initial trip tables from input to output folder - if skipCopyWarmupTripTables = "false" then do - CopyFile(inputDir+"\\Trip_EA_VOT.mtx", outputDir+"\\Trip_EA_VOT.mtx") - CopyFile(inputDir+"\\Trip_AM_VOT.mtx", outputDir+"\\Trip_AM_VOT.mtx") - CopyFile(inputDir+"\\Trip_MD_VOT.mtx", outputDir+"\\Trip_MD_VOT.mtx") - CopyFile(inputDir+"\\Trip_PM_VOT.mtx", outputDir+"\\Trip_PM_VOT.mtx") - CopyFile(inputDir+"\\Trip_EV_VOT.mtx", outputDir+"\\Trip_EV_VOT.mtx") - end - - // Build highway network - if skipBuildHwyNetwork = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run create hwy"}) - ok = RunMacro("TCB Run Macro", 1, "run create hwy",{}) - if !ok then goto quit - end - - // Create transit routes - if skipBuildTransitNetwork = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run create all transit"}) - ok = RunMacro("TCB Run Macro", 1, "Create all transit",{}) - if !ok then goto quit - end - - //Looping - for iteration = startFromIteration to max_iterations do - - if skipCoreABM[iteration] = "false" or skipOtherSimulateModel[iteration] = "false" or skipSpecialEventModel[iteration] = "false" then do //Wu modified to add special event model 5/19/2017 - // Start matrix manager locally - runString = path+"\\bin\\runMtxMgr.cmd "+drive+" "+path_no_drive - ok = RunMacro("TCB Run Command", 1, "Start matrix manager", runString) - if !ok then goto quit - - // Start JPPF driver - runString = path+"\\bin\\runDriver.cmd "+drive+" "+path_no_drive - ok = RunMacro("TCB Run Command", 1, "Start JPPF Driver", runString) - if !ok then goto quit - - // Start HH Manager, and worker nodes - runString = path+"\\bin\\StartHHAndNodes.cmd "+drive+" "+path_no_drive - ok = RunMacro("TCB Run Command", 1, "Start HH Manager, JPPF Driver, and nodes", runString) - if !ok then goto quit - end - - // Run highway assignment - if skipHighwayAssignment[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - hwy assignment"}) - ok = RunMacro("TCB Run Macro", 1, "hwy assignment",{iteration, assignByVOT}) - if !ok then goto quit - end - - // Skim highway network - if skipHighwaySkimming[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Hwy skim all"}) - ok = RunMacro("TCB Run Macro", 1, "Hwy skim all",{skimByVOT}) - if !ok then goto quit - - // Create drive to transit access file - runString = path+"\\bin\\CreateD2TAccessFile.bat "+drive+" "+path_forward_slash - - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java - Creating drive to transit access file"}) - ok = RunMacro("TCB Run Command", 1, "Creating drive to transit access file", runString) - if !ok then goto quit - end - - // Skim transit network - if skipTransitSkimming[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Build transit skims"}) - ok = RunMacro("TCB Run Macro", 1, "Build transit skims",{}) - if !ok then goto quit - end - - // First move some trip matrices so model will crash if ctramp model doesn't produced csv/mtx files for assignment - if (iteration > startFromIteration) then do - fromDir = outputDir - toDir = outputDir+"\\iter"+String(iteration-1) - //check for directory of output - if GetDirectoryInfo(toDir, "Directory")=null then do - CreateDirectory( toDir) - end - status = RunProgram("cmd.exe /c copy "+fromDir+"\\auto*.mtx "+toDir+"\\auto*.mtx",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\tran*.mtx "+toDir+"\\tran*.mtx",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\nmot*.mtx "+toDir+"\\nmot*.mtx",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\othr*.mtx "+toDir+"\\othr*.mtx",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\trip*.mtx "+toDir+"\\trip*.mtx",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\internalExternalTrips.csv "+toDir+"\\internalExternalTrips.csv",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\airport_out.csv "+toDir+"\\airport_out.csv",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\crossBorderTrips.csv "+toDir+"\\crossBorderTrips.csv",) - status = RunProgram("cmd.exe /c copy "+fromDir+"\\visitorTrips.csv "+toDir+"\\visitorTrips.csv",) - -// status = RunProgram("cmd.exe /c copy "+fromDir+"\\daily*.mtx "+toDir+"\\daily*.mtx",) -// status = RunProgram("cmd.exe /c copy "+fromDir+"\\comm*.mtx "+toDir+"\\comm*.mtx",) - - status = RunProgram("cmd.exe /c del "+fromDir+"\\auto*.mtx",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\tran*.mtx",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\nmot*.mtx",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\othr*.mtx",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\trip*.mtx",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\internalExternalTrips.csv",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\airport_out.csv",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\crossBorderTrips.csv",) - status = RunProgram("cmd.exe /c del "+fromDir+"\\visitorTrips.csv",) - -// status = RunProgram("cmd.exe /c del "+fromDir+"\\daily*.mtx",) -// status = RunProgram("cmd.exe /c del "+fromDir+"\\comm*.mtx",) - - end - - - // Run CT-RAMP model - if skipCoreABM[iteration] = "false" then do - runString = path+"\\bin\\runSandagAbm_SDRM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run CT-RAMP"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) - if !ok then goto quit - end - - - // Run airport model, visitor model, cross-border model, internal-external model - if skipOtherSimulateModel[iteration] = "false" then do - runString = path+"\\bin\\runSandagAbm_SMM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run airport model, visitor model, cross-border model"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) - if !ok then goto quit - end - - // Run special event model - if skipSpecialEventModel[iteration] = "false" then do - runString = path+"\\bin\\runSandagAbm_SEM.cmd "+drive+" "+path_forward_slash +" "+sample_rate[iteration]+" "+i2s(iteration) - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Java-Run special event model"+" "+runString}) - ok = RunMacro("TCB Run Command", 1, "Run CT-RAMP", runString) - if !ok then goto quit - end - - - if skipCTM[iteration] = "false" then do - //Commercial vehicle trip generation - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle generation"}) - ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Generation",{}) - if !ok then goto quit - - //Commercial vehicle trip distribution - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle distribution"}) - ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Distribution",{}) - if !ok then goto quit - - //Commercial vehicle time-of-day - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle Time Of Day"}) - ok = RunMacro("TCB Run Macro", 1, "Commercial Vehicle Time Of Day",{}) - if !ok then goto quit - - //Commercial vehicle toll diversion model - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - run commercial vehicle Toll Diversion"}) - ok = RunMacro("TCB Run Macro", 1, "cv toll diversion model",{}) - if !ok then goto quit - - // reduce commerical travel matrix precisions - RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce matrix precision for commVehTODTrips.mtx"}) - RunMacro("reduce matrix precision",outputDir,"commVehTODTrips.mtx", precision) - end - - /* - @WSU 2-23-2017 - Run EI and truck models only in the starting iteration (not necessarily the 1st iteration) - Purpose: - Reduce number of iterations to cut model run time by 2-2.5 hrs - Notes: - 1) Combined EI and truck trips are less than 2.5% of total trips; - 2) Trip generations are not sensitive to skims, total EI and truck demands are not affected; - 3) Skims only affect EI and truck destination choices - */ - if iteration = startFromIteration then do - //Run External(U.S.)-Internal Model - if skipEI[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - US to SD External Trip Model"}) - ok = RunMacro("TCB Run Macro", 1, "US to SD External Trip Model",{}) - if !ok then goto quit - - // reduce EI matrix precisions - - m={"usSdWrk_EA.mtx","usSdWrk_AM.mtx","usSdWrk_MD.mtx","usSdWrk_PM.mtx","usSdWrk_EV.mtx","usSdNon_EA.mtx","usSdNon_AM.mtx","usSdNon_MD.mtx","usSdNon_PM.mtx","usSdNon_EV.mtx"} - for i = 1 to m.length do - RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce precision for:"+m[i]}) - RunMacro("reduce matrix precision",outputDir,m[i], precision) - end - end - - //Run Truck Model - if skipTruck[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - truck model"}) - ok = RunMacro("truck model",properties, iteration) - if !ok then goto quit - - // reduce truck matrix precisions - m={"dailyDistributionMatricesTruckEA.mtx","dailyDistributionMatricesTruckAM.mtx","dailyDistributionMatricesTruckMD.mtx","dailyDistributionMatricesTruckPM.mtx","dailyDistributionMatricesTruckEV.mtx"} - for i = 1 to m.length do - RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce precision for:"+m[i]}) - RunMacro("reduce matrix precision",outputDir,m[i], precision) - end - end - end - - //Construct trip tables - if skipTripTableCreation[iteration] = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create Auto Tables"}) - ok = RunMacro("TCB Run Macro", 1, "Create Auto Tables",{}) - if !ok then goto quit - - // reduce EE matrix precisions - RunMacro("HwycadLog",{"sandag_abm_master.rsc","reduce matrix precision for externalExternalTrips.mtx"}) - RunMacro("reduce matrix precision",outputDir,"externalExternalTrips.mtx", precision) - end - - end - - // Run final highway assignment - if skipFinalHighwayAssignment = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - hwy assignment"}) - ok = RunMacro("TCB Run Macro", 1, "hwy assignment",{4,assignByVOT}) - if !ok then goto quit - end - - if skipFinalTransitAssignment = "false" then do - //Construct transit trip tables - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create Transit Tables"}) - ok = RunMacro("TCB Run Macro", 1, "Create Transit Tables",{}) - if !ok then goto quit - - //Run final and only transit assignment - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Assign Transit"}) - ok = RunMacro("TCB Run Macro", 1, "Assign Transit",{4}) - if !ok then goto quit - end - - // Skim highway network based on final highway assignment - if skipFinalHighwaySkimming = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Hwy skim all"}) - ok = RunMacro("TCB Run Macro", 1, "Hwy skim all",{}) - if !ok then goto quit - end - - // Skim transit network based on final transit assignemnt - if skipFinalTransitSkimming = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Build transit skims"}) - ok = RunMacro("TCB Run Macro", 1, "Build transit skims",{}) - if !ok then goto quit - end - - //Create LUZ skims - if skipLUZSkimCreation = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Create LUZ Skims"}) - ok = RunMacro("TCB Run Macro", 1, "Create LUZ Skims",{}) - if !ok then goto quit - end - - //export TransCAD data (networks and trip tables) - if skipDataExport = "false" then do - RunMacro("HwycadLog",{"sandag_abm_master.rsc:","Macro - Export TransCAD Data"}) - ok = RunMacro("TCB Run Macro", 1, "ExportSandagData",{}) - if !ok then goto quit - - // export core ABM data - runString = path+"\\bin\\DataExporter.bat "+drive+" "+path_no_drive - ok = RunMacro("TCB Run Command", 1, "Export core ABM data", runString) - if !ok then goto quit - end - - //request data load after model run finish successfully - if skipDataLoadRequest = "false" then do - runString = path+"\\bin\\DataLoadRequest.bat "+drive+path_no_drive+" "+String(max_iterations)+" "+scenarioYear+" "+sample_rate[max_iterations] - ok = RunMacro("TCB Run Command", 1, "Data load request", runString) - if !ok then goto quit - end - - // delete trip table files in iteration sub folder if model finishes without crashing - if skipDeleteIntermediateFiles = "false" then do - for iteration = startFromIteration to max_iterations-1 do - toDir = outputDir+"\\iter"+String(iteration-1) - status = RunProgram("cmd.exe /c del "+toDir+"\\auto*.mtx",) - status = RunProgram("cmd.exe /c del "+toDir+"\\tran*.mtx",) - status = RunProgram("cmd.exe /c del "+toDir+"\\nmot*.mtx",) - status = RunProgram("cmd.exe /c del "+toDir+"\\othr*.mtx",) - status = RunProgram("cmd.exe /c del "+toDir+"\\trip*.mtx",) - end - end - - RunMacro("TCB Closing", ok, "False") - return(1) - quit: - return(0) -EndMacro diff --git a/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc b/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc deleted file mode 100644 index 5213abf..0000000 --- a/sandag_abm/src/main/gisdk/sandag_abm_outputs.rsc +++ /dev/null @@ -1,293 +0,0 @@ -Macro "_OpenTable" (file,view_name) - ext = Lower(Right(file,3)) - if ext = "csv" then do - type = "CSV" - end - else if ext = "bin" then do - type = "FFB" - end - else do - ShowMessage("Cannot open table of type " + ext) - ShowMessage(2) - end - return(OpenTable(view_name,type,{file,})) -EndMacro - -Macro "GetTodToken" - return("%%TOD%%") -EndMacro - -Macro "GetSkimToken" - return("%%SKIM%%") -EndMacro - -Macro "GetTodPeriods" - return({"EA","AM","MD","PM","EV"}) -EndMacro - -Macro "GetSkimPeriods" - return({"AM","PM","OP"}) -EndMacro - -Macro "GetTodSkimMapping" - mapping = null - mapping.EA = "OP" - mapping.AM = "AM" - mapping.MD = "OP" - mapping.PM = "PM" - mapping.EV = "OP" - return(mapping) -EndMacro - -Macro "GetHighwayModes" - return({"SOV_GP","SOV_PAY","SR2_GP","SR2_HOV","SR2_PAY","SR3_GP","SR3_HOV","SR3_PAY","lhdn","mhdn","hhdn","lhdt","mhdt","hhdt"}) -EndMacro - -Macro "GetTransitModes" - return({"LOC","LRT","EXP","CMR","BRT"}) -EndMacro - -Macro "GetTransitAccessModes" - return({"WLK","PNR","KNR"}) -EndMacro - -Macro "ExportNetworkToCsv" (network_file,output_file_base) - {node_layer,line_layer} = GetDBLayers(network_file) - network_layer = AddLayerToWorkspace("network_line_layer",network_file,line_layer) - SetLayer(network_layer) - view = GetView() - ExportView(view+"|","CSV",output_file_base+".csv",,{{"CSV Header", "True"},{"Force Numeric Type", "double"}}) - CloseView(view) - -EndMacro - - -Macro "ExportHwyloadtoCSV"(input_file_base, output_file_base) - - tod_periods = RunMacro("GetTodPeriods") - - for t = 1 to tod_periods.length do - tod = tod_periods[t] - input_file= input_file_base+tod+".bin" - view = OpenTable("Binary Table","FFB",{input_file,}, {{"Shared", "True"}}) - SetView(view) - ExportView(view+"|", "CSV", output_file_base+tod+".csv", - {"ID1", - "AB_Flow_PCE", - "BA_Flow_PCE", - "AB_Time", - "BA_Time", - "AB_VOC", - "BA_VOC", - "AB_V_Dist_T", - "BA_V_Dist_T", - "AB_VHT", - "BA_VHT", - "AB_Speed", - "BA_Speed", - "AB_VDF", - "BA_VDF", - "AB_MSA_Flow", - "BA_MSA_Flow", - "AB_MSA_Time", - "BA_MSA_Time", - "AB_Flow_SOV_GP_LOW", - "BA_Flow_SOV_GP_LOW", - "AB_Flow_SOV_PAY_LOW", - "BA_Flow_SOV_PAY_LOW", - "AB_Flow_SR2_GP_LOW", - "BA_Flow_SR2_GP_LOW", - "AB_Flow_SR2_HOV_LOW", - "BA_Flow_SR2_HOV_LOW", - "AB_Flow_SR2_PAY_LOW", - "BA_Flow_SR2_PAY_LOW", - "AB_Flow_SR3_GP_LOW", - "BA_Flow_SR3_GP_LOW", - "AB_Flow_SR3_HOV_LOW", - "BA_Flow_SR3_HOV_LOW", - "AB_Flow_SR3_PAY_LOW", - "BA_Flow_SR3_PAY_LOW", - "AB_Flow_SOV_GP_MED", - "BA_Flow_SOV_GP_MED", - "AB_Flow_SOV_PAY_MED", - "BA_Flow_SOV_PAY_MED", - "AB_Flow_SR2_GP_MED", - "BA_Flow_SR2_GP_MED", - "AB_Flow_SR2_HOV_MED", - "BA_Flow_SR2_HOV_MED", - "AB_Flow_SR2_PAY_MED", - "BA_Flow_SR2_PAY_MED", - "AB_Flow_SR3_GP_MED", - "BA_Flow_SR3_GP_MED", - "AB_Flow_SR3_HOV_MED", - "BA_Flow_SR3_HOV_MED", - "AB_Flow_SR3_PAY_MED", - "BA_Flow_SR3_PAY_MED", - "AB_Flow_SOV_GP_HIGH", - "BA_Flow_SOV_GP_HIGH", - "AB_Flow_SOV_PAY_HIGH", - "BA_Flow_SOV_PAY_HIGH", - "AB_Flow_SR2_GP_HIGH", - "BA_Flow_SR2_GP_HIGH", - "AB_Flow_SR2_HOV_HIGH", - "BA_Flow_SR2_HOV_HIGH", - "AB_Flow_SR2_PAY_HIGH", - "BA_Flow_SR2_PAY_HIGH", - "AB_Flow_SR3_GP_HIGH", - "BA_Flow_SR3_GP_HIGH", - "AB_Flow_SR3_HOV_HIGH", - "BA_Flow_SR3_HOV_HIGH", - "AB_Flow_SR3_PAY_HIGH", - "BA_Flow_SR3_PAY_HIGH", - "AB_Flow_lhdn", - "BA_Flow_lhdn", - "AB_Flow_mhdn", - "BA_Flow_mhdn", - "AB_Flow_hhdn", - "BA_Flow_hhdn", - "AB_Flow_lhdt", - "BA_Flow_lhdt", - "AB_Flow_mhdt", - "BA_Flow_mhdt", - "AB_Flow_hhdt", - "BA_Flow_hhdt", - "AB_Flow", - "BA_Flow"}, - {{"CSV Header", "True"},{"Force Numeric Type", "double"}}) - CloseView(view) - end - ok=1 - quit: - return(ok) -EndMacro - - - - - -Macro "BuildTransitFlowOptions" - //name,source_name,primary key column - skim_token = RunMacro("GetSkimToken") - topts = {{"ROUTE", "Route" ,True }, - {"FROM_STOP", "From_Stop" ,True }, - {"TO_STOP", "To_Stop" ,True }, - {"CENTROID", "Centroid" ,False }, - {"FROMMP", "From_MP" ,False }, - {"TOMP", "To_MP" ,False }, - {"TRANSITFLOW", "TransitFlow" ,False }, - {"BASEIVTT", "BaseIVTT" ,False }, - {"COST", "Cost" ,False }, - {"VOC", "VOC" ,False }} - fopts = {"flow",""} - return({topts,fopts}) -EndMacro - -Macro "BuildOnOffOptions" - //name,source_name,primary key column - skim_token = RunMacro("GetSkimToken") - topts = {{"ROUTE", "ROUTE" ,True }, - {"STOP", "STOP" ,True }, - {"BOARDINGS", "On" ,False}, - {"ALIGHTINGS", "Off" ,False}, - {"WALKACCESSON", "WalkAccessOn" ,False}, - {"DIRECTTRANSFERON", "DirectTransferOn" ,False}, - {"WALKTRANSFERON", "WalkTransferOn" ,False}, - {"DIRECTTRANSFEROFF", "DirectTransferOff" ,False}, - {"WALKTRANSFEROFF", "WalkTransferOff" ,False}, - {"EGRESSOFF", "EgressOff" ,False}} - fopts = {"ono",""} - return({topts,fopts}) -EndMacro - - -Macro "BuildAggFlowOptions" - //name,source_name,primary key column - skim_token = RunMacro("GetSkimToken") - topts = {{"LINK_ID", "ID1" ,True }, - {"AB_TransitFlow", "AB_TransitFlow" ,false}, - {"BA_TransitFlow", "BA_TransitFlow" ,False}, - {"AB_NonTransit", "AB_NonTransit" ,False}, - {"BA_NonTransit", "BA_NonTransit" ,False}, - {"AB_TotalFlow", "AB_TotalFlow" ,False}, - {"BA_TotalFlow", "BA_TotalFlow" ,False}, - {"AB_Access_Walk_Flow", "AB_Access_Walk_Flow" ,False}, - {"BA_Access_Walk_Flow", "BA_Access_Walk_Flow" ,False}, - {"AB_Xfer_Walk_Flow", "AB_Xfer_Walk_Flow" ,False}, - {"BA_Xfer_Walk_Flow", "BA_Xfer_Walk_Flow" ,False}, - {"AB_Egress_Walk_Flow", "AB_Egress_Walk_Flow" ,False}, - {"BA_Egress_Walk_Flow", "BA_Egress_Walk_Flow" ,False}} - fopts = {"agg",""} - return({topts,fopts}) -EndMacro - - - -Macro "ExportTransitTablesToCsv" (results_dir,transit_options,output_file_base) - fopts = transit_options[2] - transit_options = transit_options[1] - header = "MODE,ACCESSMODE,TOD" - for i = 1 to transit_options.length do - header = header + "," + transit_options[i][1] - end - - f = OpenFile(output_file_base + ".csv","w") - WriteLine(f,header) - - table_name = Upper(Right(output_file_base,Len(output_file_base) - PositionTo(,output_file_base,"\\"))) - - transit_modes = RunMacro("GetTransitModes") - transit_access_modes = RunMacro("GetTransitAccessModes") - tod_periods = RunMacro("GetTodPeriods") - tod_skim_mapping = RunMacro("GetTodSkimMapping") - skim_token = RunMacro("GetSkimToken") - - for i = 1 to tod_periods.length do - tod = tod_periods[i] - for t = 1 to transit_modes.length do - transit_mode = transit_modes[t] - for ta = 1 to transit_access_modes.length do - transit_access_mode = transit_access_modes[ta] - tf = RunMacro("FormPath",{results_dir,fopts[1] + transit_access_modes[ta] + "_" + transit_mode + "_" + tod + fopts[2] + ".bin"}) - if GetFileInfo(tf) <> null then do - view = RunMacro("_OpenTable",tf,"tview") - - rh = GetFirstRecord(view + "|",) - while rh <> null do - line = transit_mode + "," + transit_access_mode + "," + tod - for to = 1 to transit_options.length do - line = line + "," + RunMacro("ToString",view.(Substitute(transit_options[to][2],skim_token,tod_skim_mapping.(tod),))) - end - WriteLine(f,line) - structure = GetViewStructure(view) - for to = 1 to transit_options.length do - field_id = -1 - field = Substitute(transit_options[to][2],skim_token,tod_skim_mapping.(tod),) - for s = 1 to structure.length do - if structure[s][1] = field then do - field_id = s - end - end - if field_id < 1 then do - ShowMessage("couldn't find field: " + field) - ShowMessage(2) - end - - end - rh = GetNextRecord(view + "|",rh,) - end - CloseView(view) - end - end - end - end - - CloseFile(f) - -EndMacro - -Macro "SaveMatrix" (matrix_in,file_out) - m = OpenMatrix(matrix_in,) - CreateTableFromMatrix(m,file_out,"CSV",{{"Complete","Yes"}}) -EndMacro - - diff --git a/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc b/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc deleted file mode 100644 index d0be578..0000000 --- a/sandag_abm/src/main/gisdk/sellinkMtxAgg.rsc +++ /dev/null @@ -1,223 +0,0 @@ -//Aggregate transit select link trip matrix -//Input: trn_sellinkxxxxx_xx.mtx select link trip table by access mode, line haul mode, and time of day -// a text file with a list of TAPs to sum up trips -//Ouput: trn_sellink_total.csv -//Author: Ziying Ouyang -// @ziying.ouyang@sandag.org -//Date: Feb 5, 2014 - - -Macro "Sum Up Select Link Transit Trips" - shared path_study, path, outputDir - RunMacro("TCB Init") - - sum_all = 0 - sum_sg = 0 - pct_sg = 0 - - outputDir = path+"\\output" - sellink_file = outputDir+"\\trn_sellinkWLK_LRT_EA.mtx" - if GetFileInfo(sellink_file) <> null then do - periodName = {"_EA", "_AM", "_MD", "_PM", "_EV"} - - matrixCore = { - "WLK_LOC", - "WLK_EXP", - "WLK_BRT", - "WLK_LRT", - "WLK_CMR", - "PNR_LOC", - "PNR_EXP", - "PNR_BRT", - "PNR_LRT", - "PNR_CMR", - "KNR_LOC", - "KNR_EXP", - "KNR_BRT", - "KNR_LRT", - "KNR_CMR" } - - //open smart growth tap CSV file - in_file = path_study + "\\sg_tap.csv" - if GetFileInfo(in_file) <> null then do - - view = OpenTable("SmartGrowthTap","CSV",{in_file,}, {{"Shared", "True"}}) - SetView(view) - vw_TAP = GetView() - view_set = vw_TAP +"|" - - for per = 1 to periodName.length do - for mat = 1 to matrixCore.length do - - - sellkmtx = outputDir+"\\trn_sellink"+matrixCore[mat]+periodName[per]+".mtx" - m = OpenMatrix(sellkmtx,) - - sg_orig_index = CreateMatrixIndex("Smart Growth Orig", m, "Row",view_set, "sg_tap", "sg_tap" ) - sg_dest_index = CreateMatrixIndex("Smart Growth Dest", m, "Column",view_set, "sg_tap", "sg_tap" ) - - mc = CreateMatrixCurrency(m, , , , ) - mc_sg_orig = CreateMatrixCurrency(m, ,"Smart Growth Orig","Columns", ) - mc_sg_dest = CreateMatrixCurrency(m, ,"Rows","Smart Growth Dest", ) - mc_sg_to_sg = CreateMatrixCurrency(m, ,"Smart Growth Orig","Smart Growth Dest", ) - - sum_row = GetMatrixMarginals(mc, "Sum", "row" ) - - sum_row_sg_orig = GetMatrixMarginals(mc_sg_orig, "Sum", "row" ) - sum_row_sg_dest = GetMatrixMarginals(mc_sg_dest, "Sum", "row" ) - sum_row_sg_to_sg = GetMatrixMarginals(mc_sg_to_sg, "Sum", "row" ) - - sum_all = sum_all + Sum(sum_row) - sum_sg = sum_sg + Sum(sum_row_sg_orig) + Sum(sum_row_sg_dest) - Sum(sum_row_sg_to_sg) - - DeleteMatrixIndex(m, "Smart Growth Orig") - DeleteMatrixIndex(m, "Smart Growth Dest") - - //close matrix - mc = null - m = null - - end - end - - if sum_all > 0 then pct_sg = sum_sg /sum_all - - out_file = path_study + "\\trn_sellink_sg.csv" - - dif2 = GetDirectoryInfo(out_file,"file") - if dif2.length <= 0 then do - fpr = OpenFile(out_file,"w") - WriteLine(fpr, "Scenario, Total select link trips, smart growth select link trips, % of sg trips") - end - else do - fpr = OpenFile(out_file,"a") - end - - WriteLine(fpr,path + "," + r2s(sum_all) + "," + r2s(sum_sg) + "," + r2s(pct_sg)) - CloseFile(fpr) - end - else ShowMessage("Missing smart growth tap file in " + path_study) - End - RunMacro("close all") - RunMacro("TCB Closing", ok, "False") - return(1) -EndMacro - -Macro "Sum Up Select Link Highway Trips" - - shared path_study, path, outputDir - RunMacro("TCB Init") - - outputDir = path+"\\output" - sellink_file = outputDir+"\\select_EA.mtx" - if GetFileInfo(sellink_file) <> null then do - - m = OpenMatrix(sellink_file,) - coreNames = GetMatrixCoreNames(m) - m=null - - //hard coded the occupancy rate, could be improved based on coreNames - occupancy = {1, 1, 2, 2, 2, 3.34, 3.34, 3.34, 1, 1, 1, 1, 1, 1} - - sum_all = 0 - sum_sg = 0 - pct_sg = 0 - sum_ind = 0 - pct_ind = 0 - - periodName = {"_EA","_AM","_MD","_PM","_EV"} - - //open smart growth TAZ CSV file - in_file = path_study + "\\sg_taz.csv" - in_file_indian = path_study + "\\indian_reservation_taz.csv" - - if GetFileInfo(in_file) <> null then do - view = OpenTable("SmartGrowthTAZ","CSV",{in_file,}, {{"Shared", "True"}}) - SetView(view) - vw_TAZ = GetView() - view_set = vw_TAZ +"|" - - view_ind = OpenTable("IndianReservationTAZ","CSV",{in_file_indian,}, {{"Shared", "True"}}) - SetView(view_ind) - vw_TAZ_ind = GetView() - view_set_ind = vw_TAZ_ind +"|" - - for per = 1 to periodName.length do - - sellkmtx = outputDir + "\\select" + periodName[per] +".mtx" - m = OpenMatrix(sellkmtx,) - - sg_index_orig = CreateMatrixIndex("SG Index Orig", m, "Row",view_set,"SG_TAZ" , "SG_TAZ" ) - sg_index_dest = CreateMatrixIndex("SG Index Dest", m, "Column",view_set,"SG_TAZ" , "SG_TAZ" ) - - ind_index_orig = CreateMatrixIndex("Indian Index Orig", m, "Row",view_set_ind,"Indian_TAZ" , "Indian_TAZ" ) - ind_index_dest = CreateMatrixIndex("Indian Index Dest", m, "Column",view_set_ind,"Indian_TAZ" , "Indian_TAZ" ) - - mc = CreateMatrixCurrencies(m, "Rows", "Columns", ) - mc_sg_orig = CreateMatrixCurrencies(m, "SG Index Orig","Columns",) - mc_sg_dest = CreateMatrixCurrencies(m, "Rows","SG Index Dest",) - mc_sg_to_sg = CreateMatrixCurrencies(m, "SG Index Orig","SG Index Dest",) - - mc_ind_orig = CreateMatrixCurrencies(m, "Indian Index Orig","Columns",) - mc_ind_dest = CreateMatrixCurrencies(m, "Rows","Indian Index Dest",) - mc_ind_to_ind = CreateMatrixCurrencies(m, "Indian Index Orig","Indian Index Dest",) - - // select link assignment trip matrix includes the total trips as the last core - for core = 1 to coreNames.length - 1 do - sum_row = GetMatrixMarginals(mc.(coreNames[core]), "Sum", "row" ) - - sum_row_sg_orig = GetMatrixMarginals(mc_sg_orig.(coreNames[core]), "Sum", "row" ) - sum_row_sg_dest = GetMatrixMarginals(mc_sg_dest.(coreNames[core]), "Sum", "row" ) - sum_row_sg_to_sg = GetMatrixMarginals(mc_sg_to_sg.(coreNames[core]), "Sum", "row" ) - - - sum_row_ind_orig = GetMatrixMarginals(mc_ind_orig.(coreNames[core]), "Sum", "row" ) - sum_row_ind_dest = GetMatrixMarginals(mc_ind_dest.(coreNames[core]), "Sum", "row" ) - sum_row_ind_to_ind = GetMatrixMarginals(mc_ind_to_ind.(coreNames[core]), "Sum", "row" ) - - sum_all = sum_all + Sum(sum_row) * occupancy[core] - sum_sg = sum_sg + (Sum(sum_row_sg_orig) + Sum(sum_row_sg_dest)) * occupancy[core] - Sum(sum_row_sg_to_sg) * occupancy[core] - sum_ind = sum_ind + (Sum(sum_row_ind_orig) + Sum(sum_row_ind_dest)) * occupancy[core] - Sum(sum_row_ind_to_ind) * occupancy[core] - - end - - //close matrix - DeleteMatrixIndex(m, "SG Index Orig") - DeleteMatrixIndex(m, "SG Index Dest") - DeleteMatrixIndex(m, "Indian Index Orig") - DeleteMatrixIndex(m, "Indian Index Dest") - mc = null - mc_sg = null - mc_ind = null - m = null - end - - if sum_all > 0 then do - pct_sg = sum_sg /sum_all - pct_ind = sum_ind/sum_all - end - - out_file = path_study + "\\hwy_sellink.csv" - - dif2 = GetDirectoryInfo(out_file,"file") - if dif2.length <= 0 then do - fpr = OpenFile(out_file,"w") - WriteLine(fpr, "Scenario, Total select link trips, smart growth select link trips, % of sg trips, indian reservation trips, % of indian reservation trips") - end - else do - fpr = OpenFile(out_file,"a") - end - - WriteLine(fpr,path + "," + r2s(sum_all) + "," + r2s(sum_sg) + "," + r2s(pct_sg) + "," + r2s(sum_ind) + "," + r2s(pct_ind)) - CloseFile(fpr) - - end - else ShowMessage("Missing smart growth TAZ file in " + path_study) - - end - RunMacro("close all") - RunMacro("TCB Closing", ok, "False") - return(1) - -EndMacro - diff --git a/sandag_abm/src/main/gisdk/sellink_volume.rsc b/sandag_abm/src/main/gisdk/sellink_volume.rsc deleted file mode 100644 index dd462c3..0000000 --- a/sandag_abm/src/main/gisdk/sellink_volume.rsc +++ /dev/null @@ -1,85 +0,0 @@ -/* - Export daily select link volumes by direction - Need to figure out vary by # of query - author: Ziying Ouyang zou@sandag.org - date: 12/16/2015 -*/ -Macro "ExportHwyloadtoCSV Select Link" - - shared path, input_path, output_path - path="T:\\projects\\sr13\\version13_3_0\\abm_runs\\2012" - input_path = path+"\\input" - output_path = path+"\\output" - - query_list = RunMacro("Get SL Query # from QRY",input_path) - if query_list.length > 0 then do - - Dim v_ab_flow_slk[query_list.length], v_ba_flow_slk[query_list.length] - Dim v_ab_flow_slk_pk[2,query_list.length], v_ba_flow_slk_pk[2,query_list.length] - input_file = {"hwyload_EA.bin","hwyload_AM.bin","hwyload_MD.bin","hwyload_PM.bin","hwyload_EV.bin"} - - fields = {"ID1"} - for j = 1 to query_list.length do - fields = fields + {"AB_Flow_" + query_list[j]} - fields = fields + {"BA_Flow_" + query_list[j]} - end - - for i = 1 to input_file.length do - view = OpenTable("Binary Table","FFB",{output_path+"\\"+input_file[i],}, {{"Shared", "True"}}) - SetView(view) - v_lodselk = GetDataVectors(view+"|", fields, ) - for j = 1 to query_list.length do - v_ab_flow_slk[j] = Nz(v_ab_flow_slk[j]) + v_lodselk[2*(j-1)+2] - v_ba_flow_slk[j] = Nz(v_ba_flow_slk[j]) + v_lodselk[2*(j-1)+3] - end - if i = 2 or i = 4 then do //save AM/PM select link ab/ba volumes (AM 1, PM 2) - for j = 1 to query_list.length do - v_ab_flow_slk_pk[i/2][j] = v_lodselk[2*(j-1)+2] - v_ba_flow_slk_pk[i/2][j] = v_lodselk[2*(j-1)+3] - end - end - - end - - header = "ID1" - for j = 1 to query_list.length do - header = header + "," + "AB_Flow_" + query_list[j] + "," + "BA_Flow_" + query_list[j] + "," + "Tot_Flow_" + query_list[j] - end - - f = OpenFile(output_path+ "\\"+"loadselk.csv","w") - WriteLine(f,header) - - for i = 1 to v_lodselk[1].length do - line = i2s(v_lodselk[1][i]) - for j = 1 to query_list.length do - line = line + "," + r2s(v_ab_flow_slk[j][i]) + "," + r2s(Nz(v_ba_flow_slk[j][i])) + "," + r2s(v_ab_flow_slk[j][i]+Nz(v_ba_flow_slk[j][i])) - end - WriteLine(f,line) - end - - closeFile(f) - //Peak Period Select Link Volumes - header = "ID1" - for j = 1 to query_list.length do - header = header + "," + "AB_Flow_AM_" + query_list[j] + "," + "BA_Flow_AM_" + query_list[j] + "," + "Tot_Flow_AM_" + query_list[j] + "," + "AB_Flow_PM_" + query_list[j] + "," + "BA_Flow_PM_" + query_list[j] + "," + "Tot_Flow_PM_" + query_list[j] - end - - f = OpenFile(output_path+ "\\"+"loadselkpk.csv","w") - WriteLine(f,header) - - for i = 1 to v_lodselk[1].length do - line = i2s(v_lodselk[1][i]) - for j = 1 to query_list.length do - line = line + "," + r2s(v_ab_flow_slk_pk[1][j][i]) + "," + r2s(Nz(v_ba_flow_slk_pk[1][j][i])) + "," + r2s(v_ab_flow_slk_pk[1][j][i]+Nz(v_ba_flow_slk_pk[1][j][i])) - line = line + "," + r2s(v_ab_flow_slk_pk[2][j][i]) + "," + r2s(Nz(v_ba_flow_slk_pk[2][j][i])) + "," + r2s(v_ab_flow_slk_pk[2][j][i]+Nz(v_ba_flow_slk_pk[2][j][i])) - end - WriteLine(f,line) - end - - CloseFile(f) - end - - else ShowMessage("Number of select links is 0") - RunMacro("close all") - -EndMacro diff --git a/sandag_abm/src/main/gisdk/trnassign.rsc b/sandag_abm/src/main/gisdk/trnassign.rsc deleted file mode 100644 index e61c1e7..0000000 --- a/sandag_abm/src/main/gisdk/trnassign.rsc +++ /dev/null @@ -1,177 +0,0 @@ -/********************************************************************************* -Transit Assignment - - input files: transitrt.rts - CT-RAMP trip tables ("tranTrips_period.mtx", where period = EA, AM, MD, PM, and NT) - each file has 15 cores, named as follows: ACC_LHM_PER - where: - ACC = access mode - WLK,PNR,KNR - LHM = line-haul mode - LOC,EXP,BRT,LRT,CMR - PER = period - EA, AM, MD, PM, NT - Transit networks (localpk.tnw,localop,tnw,prempk.tnw,premop.tnw) - Transit route file (transitrt.rts) - output files: - 75 flow bin file (3 access modes * 5 line-haul modes * 5 time periods) - 75 walk bin file - 75 onoff bin file - 75 collapsed onoff files in both binary and csv format - -*********************************************************************************/ - -Macro "Assign Transit" (args) - - - shared path,inputDir, outputDir, mxtap - -iteration = args[1] -rt_file=outputDir + "\\transitrt.rts" - -periodName = {"_EA", "_AM", "_MD", "_PM", "_EV"} - -matrixCore = { - "WLK_LOC", - "WLK_EXP", - "WLK_BRT", - "WLK_LRT", - "WLK_CMR", - "PNR_LOC", - "PNR_EXP", - "PNR_BRT", - "PNR_LRT", - "PNR_CMR", - "KNR_LOC", - "KNR_EXP", - "KNR_BRT", - "KNR_LRT", - "KNR_CMR" } - - network={"locl","prem","prem","prem","prem","locl","prem","prem","prem","prem","locl","prem","prem","prem","prem"} - dim onOffTables[matrixCore.length * periodName.length] - - selinkqry = inputDir + "\\"+"sellink_transit.qry" - if GetFileInfo(selinkqry) <> null then sellink_flag = 1 //select link analysis is only for last iteration (4) - - i = 0 - k = 1 - for per = 1 to periodName.length do - for mat = 1 to matrixCore.length do - - i = i + 1 - - networkFile = outputDir+"\\"+network[mat]+periodName[per]+".tnw" - matrixFile = outputDir+"\\tranTotalTrips"+periodName[per]+".mtx" - matrixName = matrixCore[mat] - flowFile = outputDir+"\\flow"+matrixCore[mat]+periodName[per]+".bin" - walkFile = outputDir+"\\ntl"+matrixCore[mat]+periodName[per]+".bin" - onOffFile = outputDir+"\\ono"+matrixCore[mat]+periodName[per]+".bin" - aggFile = outputDir+"\\agg"+matrixCore[mat]+periodName[per]+".bin" - - onOffTables[k] = onOffFile - k = k +1 - - if sellink_flag = 1 & iteration = 4 then sellkmtx = outputDir+"\\trn_sellink"+matrixCore[mat]+periodName[per]+".mtx" - - - // STEP 1: Transit Assignment - Opts = null - Opts.Input.[Transit RS] = rt_file - Opts.Input.Network = networkFile - Opts.Input.[OD Matrix Currency] = {matrixFile,matrixName,,} - - Opts.Output.[Flow Table] = flowFile - Opts.Output.[Walk Flow Table] = walkFile - Opts.Output.[OnOff Table] = onOffFile - Opts.Output.[Aggre Table] = aggFile - Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 - if sellink_flag = 1 & iteration = 4 then do - Opts.Flag.critFlag = 1 - Opts.Global.[Critical Query File] = selinkqry - Opts.Output.[Critical Matrix].Label = "Critical Matrix" - Opts.Output.[Critical Matrix].[File Name] = sellkmtx - end - RunMacro("HwycadLog",{"trassigns.rsc: transit assigns","Transit Assignment PF: "+matrixCore[mat]+periodName[per]}) - ok = RunMacro("TCB Run Procedure", (per*100+mat), "Transit Assignment PF", Opts) - - if !ok then goto quit - - end - end - - properties = "\\conf\\sandag_abm.properties" - collapseOnOffByRoute = RunMacro("read properties",properties,"RunModel.collapseOnOffByRoute", "S") - if collapseOnOffByRoute = "true" then do - ok = RunMacro("Collapse OnOffs By Route", onOffTables, rt_file) - if !ok then goto quit - end - - quit: - RunMacro("close all") - Return( ok ) - -EndMacro -/************************************************************* -* -* A macro that will collapse transit on-offs by route and append -* route name. -* -* Arguments -* onOffTables An array of on-off tables -* rtsfile A transit route file -* -*************************************************************/ -Macro "Collapse OnOffs By Route" (onOffTables,rtsfile) - - {rte_lyr,stp_lyr,} = RunMacro("TCB Add RS Layers", rtsfile, "ALL", ) - - fields = { - {"On","Sum",}, - {"Off","Sum",}, - {"DriveAccessOn","Sum",}, - {"WalkAccessOn","Sum",}, - {"DirectTransferOn","Sum",}, - {"WalkTransferOn","Sum",}, - {"DirectTransferOff","Sum",}, - {"WalkTransferOff","Sum",}, - {"EgressOff","Sum",} - } - - // for all on off tables - for i = 1 to onOffTables.length do - - onOffView = OpenTable("OnOffTable", "FFB", {onOffTables[i], null}) - path = SplitPath(onOffTables[i]) - outFile = path[1]+path[2]+path[3]+"_COLL.bin" - - fields = GetFields(onOffView, "All") - - //include all fields in each table except for STOP and ROUTE - collFields = null - for j = 1 to fields[1].length do - - if(fields[1][j] !="STOP" and fields[1][j]!= "ROUTE") then do - - collFields = collFields + {{fields[1][j],"Sum",}} - - end - end - - // Collapse stops out of the table by collapsing on ROUTE - rslt = AggregateTable("CollapsedView", onOffView+"|", "FFB", outFile, "ROUTE", collFields, ) - - CloseView(onOffView) - - // Join the route layer for route name and other potentially useful data - onOffCollView = OpenTable("OnOffTableColl", "FFB", {outFile}) - joinedView = JoinViews("OnOffJoin", onOffCollView+".Route", rte_lyr+".Route_ID",) - - // Write the joined data to a binary file - outJoinFile = path[1]+path[2]+path[3]+"_COLL_JOIN.bin" - ExportView(joinedView+"|","FFB", outJoinFile , , ) - outJoinFile = path[1]+path[2]+path[3]+"_COLL_JOIN.csv" - ExportView(joinedView+"|","CSV", outJoinFile , , ) - end - - Return(1) - quit: - Return( RunMacro("TCB Closing", ret_value, True ) ) -EndMacro diff --git a/sandag_abm/src/main/gisdk/trnskim.rsc b/sandag_abm/src/main/gisdk/trnskim.rsc deleted file mode 100644 index 4313570..0000000 --- a/sandag_abm/src/main/gisdk/trnskim.rsc +++ /dev/null @@ -1,1038 +0,0 @@ -/******************************************************************************** -transit skim matrix to create skim matrix files - skim values includes fare, in vehicle time, initial wait time, - transfer wait time, tranfer walk time, access time, egress time - dwell time, transfer penalty, # of transfers - *TMO, or *TMP (depends on which time period) -for premium service skim *TMO, or *TMP by mode, additional cores were created -input files: transit.dbd - transit line layer - transitrt.rts - transit route layer - localop.tnw - local bus off peak transit network - localpk.tnw - local bus peak transit network - premop.tnw - premium bus service off peak transit network - prempk.tnw - premium bus service peak transit network - modenu.dbf - dbf file for mode table - fare.mtx - fare matrix for zonal fares for lightrail and coaster - -output files: - implbopx2.mtx - skim matrix file for local bus off peak service - implbpkx2.mtx - skim matrix file for local bus peak service - imppropx2.mtx - skim matrix file for premium bus off peak service - impprpkx2.mtx - skim matrix file for premium bus peak service - implr.mtx - skim ltrzone - impcr.mtx - skim crzone -history - 10/8/03 zou to change the fare system to flat fare and skim - rail link fields separately. - 9/3/04 retain the commuter rail from the light rail category - 1/25/05 rewrite the script using moduels - 4/14/09 ZOU changed it to 3tod -********************************************************************************/ -Macro "Create transit network" - - shared path,inputDir, outputDir, mxtap - ok=RunMacro("Create transit networks") - if !ok then goto quit - Return(1) - - quit: - Return(0) -EndMacro - -/*********************************************************************************** -*******************************************************************/ - -Macro "Build transit skims" - - shared path,inputDir, outputDir, mxtap - - ok = RunMacro("Update transit time fields") - if !ok then goto quit - - /* - ok=RunMacro("Create rail net") - if !ok then goto quit - */ - ok=RunMacro("Create transit networks") - if !ok then goto quit - - ok = RunMacro("Skim transit networks") - if !ok then goto quit - - /* - ok= RunMacro("zero null ivt time") - if !ok then goto quit -*/ - ok= RunMacro("Process transit skims") - if !ok then goto quit - - Return(1) - - quit: - Return(0) -EndMacro -/*********************************************************************************** - - -Inputs: - input\mode5tod.dbf Mode table - output\transit.dbd Transit line layer - output\xxxx_yy.tnw Transit networks - - where xxxxx is transit mode (locl or prem) and yy is period (EA, AM, MD, PM, EV) - -Outputs: - impxxxx_yy.mtx Transit skims - - where xxxxx is transit mode (locl or prem) and yy is period (EA, AM, MD, PM, EV) - -***********************************************************************************/ -Macro "Skim transit networks" - - shared path,inputDir, outputDir, mxtap - NumofCPU=2 - - mode_tb = inputDir+"\\mode5tod.dbf" - // xfer_tb = path+"\\modexfer.dbf" - db_file = outputDir + "\\transit.dbd" - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - db_node_lyr = db_file + "|" + node_lyr - - periods = {"_EA","_AM","_MD","_PM","_EV"} - modes = {"prem","locl"} - - //varies by modes - skmodes={{4,5,6,7,8,9,10},} - skvar={{"Fare", "Initial Wait Time", "Transfer Wait Time", "Transfer Walk Time", "Access Walk Time","Egress Walk Time","Dwelling Time", "Number of Transfers"}, - {"Fare", "In-Vehicle Time", "Initial Wait Time", "Transfer Wait Time", "Transfer Walk Time","Access Walk Time","Egress Walk Time", "Dwelling Time", "Number of Transfers"}} - - // not sure what the PRE and LOC time are yet - skvars={skvar[1]+{"Length","*TM"},skvar[2]} - - - for i = 1 to periods.length do - for j = 1 to modes.length do - Opts = null - // Opts.Global.[Force Threads] = NumofCPU - Opts.Input.Database = db_file - Opts.Input.Network = outputDir+"\\"+modes[j]+periods[i]+".tnw" - Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Selection", "Select * where id <"+i2s(mxtap)} - Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Selection"} - Opts.Input.[Mode Table] = {mode_tb} - // Opts.Input.[Mode Xfer Table] = {xfer_tb} - Opts.Global.[Skim Var] = skvars[j] - Opts.Global.[OD Layer Type] = 2 - if skmodes<> null then Opts.Global.[Skim Modes] = skmodes[j] - Opts.Flag.[Do Skimming] = 1 - Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 - Opts.Output.[Skim Matrix].Label = "Skim Matrix ("+modes[j]+periods[i]+")" - Opts.Output.[Skim Matrix].Compression = 0 //uncompressed - Opts.Output.[Skim Matrix].[File Name] = outputDir+"\\imp"+modes[j]+periods[i]+".mtx" - // Opts.Output.[TPS Table] = outputDir+"\\"+modes[j]+periods[i]+".tps" - ok = RunMacro("TCB Run Procedure", i, "Transit Skim PF", Opts) - - if !ok then goto quit - end - end - - ok=1 - quit: - RunMacro("close all") - Return(ok) -EndMacro -/*********************************************************************************** - - -Inputs: - input\mode5tod.dbf - output\transit.dbd - -***********************************************************************************/ -Macro "Special transit skims" (arr) - - shared path,inputDir, outputDir, mxtap - - skimvar=arr[1] - - mode_tb = inputDir+"\\mode5tod.dbf" - // xfer_tb = path+"\\modexfer.dbf" - db_file = outputDir + "\\transit.dbd" - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - db_node_lyr = db_file + "|" + node_lyr - - fpr=openfile(path+"\\hwycad.log","a") - mytime=GetDateAndTime() - writeline(fpr,mytime+", skim trnets") - - if skimvar="rail" then do //skim lrtzone and crzone to get the trolley and coaster fares - skvars={{"lrtzone"},{"Crzone"}} - trskimmtxs={"implr.mtx","impcr.mtx"} - trnets={"lr.tnw","cr.tnw"} - mtxdes={"Skim Matrix (light Rail)","Skim Matrix (coaster)"} - end - else if skimvar="fwylen" then do //skim length and fwylen - skvars={{"length","fwylen"},{"length","fwylen"}} - trskimmtxs={"imppropdst2.mtx","impprpkdst2.mtx"} - trnets={"prem_MD.tnw","prem_AM.tnw"} - mtxdes={"Length (Shortest OP Premium Path)","Length (Shortest PK Premium Path)"} - end - - for i = 1 to trnets.length do - Opts = null - Opts.Global.[Force Threads] = NumofCPU - Opts.Input.Database = db_file - Opts.Input.Network = path+"\\"+trnets[i] - Opts.Input.[Origin Set] = {db_node_lyr, node_lyr, "Selection", "Select * where id <"+i2s(mxtap)} - Opts.Input.[Destination Set] = {db_node_lyr, node_lyr, "Selection"} - Opts.Input.[Mode Table] = {mode_tb} - // Opts.Input.[Mode Xfer Table] = {xfer_tb} - Opts.Global.[Skim Var] = skvars[i] - Opts.Global.[OD Layer Type] = 2 - if skmodes<> null then Opts.Global.[Skim Modes] = skmodes[i] - Opts.Flag.[Do Skimming] = 1 - Opts.Flag.[Do Maximum Fare] = 1 //added for 4.8 build 401 - Opts.Output.[Skim Matrix].Label = mtxdes[i] - Opts.Output.[Skim Matrix].Compression = 0 //uncompressed - Opts.Output.[Skim Matrix].[File Name] = path+"\\"+trskimmtxs[i] - ok = RunMacro("TCB Run Procedure", i, "Transit Skim PF", Opts) - // ok = RunMacro("TCB Run Procedure", i, "Transit Skim Max Fare", Opts)//using maximized fare - - if !ok then goto quit - end - - ok=1 - quit: - RunMacro("close all") - Return(ok) -EndMacro - -/*************************************************************************************************************************** -This macro puts zeros in the null cells for unprocessed transit skims: premium and local modes -3/19/2015 Wu modified to zero out all ivts -****************************************************************************************************************************/ -Macro "zero null ivt time" - - shared path, outputDir - periods = {"_EA","_AM","_MD","_PM","_EV"} - - for i=1 to periods.length do - - //open matrix - fileNameSkimP = outputDir + "\\impprem"+periods[i]+".mtx" - mp = OpenMatrix(fileNameSkimP,) - currsp= CreateMatrixCurrencies(mp, , , ) - currsp.("*TM (Local)"):=Nz(currsp.("*TM (Local)")) - currsp.("*TM (Commuter Rail)"):=Nz(currsp.("*TM (Commuter Rail)")) - currsp.("*TM (Light Rail)"):=Nz(currsp.("*TM (Light Rail)")) - currsp.("*TM (Regional BRT (Yello)"):=Nz(currsp.("*TM (Regional BRT (Yello)")) - currsp.("*TM (Regional BRT (Red))"):=Nz(currsp.("*TM (Regional BRT (Red))")) - currsp.("*TM (Limited Express)"):=Nz(currsp.("*TM (Limited Express)")) - currsp.("*TM (Express)"):=Nz(currsp.("*TM (Express)")) - - fileNameSkimL = outputDir + "\\implocl"+periods[i]+".mtx" - ml = OpenMatrix(fileNameSkimL,) - currsl= CreateMatrixCurrencies(ml, , , ) - currsl.("In-Vehicle Time"):=Nz(currsl.("In-Vehicle Time")) - end - quit: - Return(1) -EndMacro - -/*********************************************************************************************************************************** - Extract Main Mode from Transit Skims - - This macro: ------------- - 1- extracts the main mode from transit skims - 2- writes main mode as a core to output transit matrix - - Inputs: ----------- - - output\impxxxx_yy.mtx - - where: - xxxx is mode (locl or prem) - yy is time period (EA, AM, MD, PM, NT) - - - Outputs: - - - output\impxxxx_yyo.mtx - - where: - xxxx is mode (locl or prem) - yy is time period (EA, AM, MD, PM, NT) - - - Input transit modes: ----------------------- - CR, LR, BRT Yellow, BRT Red, Limited EXP, EXP, LB - - Output transit modes: ------------------------ - 1-CR Mode choice code 4 - 2-LR Mode choice code 5 - 3-BRT (BRT Yellow+BRT Red) Mode choice code 6 - 4-EXP (EXP+Limited EXP) Mode choice code 7 - 5-LB Mode choice code 8 - - Input-output Matrix cores correspondence table: -------------------------------------------------- - 1) Premium input-output - Input Output - Fare(1) Fare(1) - Initial Wait Time(2) Initial Wait Time(2) - Transfer Wait Time(3) Transfer Wait Time(3) - Transfer Walk Time(4)+Access Walk Time(5)+Egress Walk Time(6) Walk Time(4) - Dwelling time(7) ------ - Number of Transfers(8) Number of Transfers(5) - Length:CR(9) Length:CR(6) - Length:LR(10) Length:LR(7) - Length:BRT Yellow(11)+BRT Red(12) Length:BRT(8) - Length:Limited EXP(13)+EXP(14) Length:EXP(9) - Length:LB(15) Length:LB(10) - IVT:CR(16) IVT:CR(11) - IVT:LR(17) IVT:LR(12) - IVT:BRT Yellow(18)+BRT Red(19) IVT:BRT(13) - IVT:Limited EXP(20)+EXP(21) IVT:EXP(14) - IVT:LB(22) IVT:LB(15) - --- IVT:Sum(16) - --- IVT:Main Mode(17) - - 2) Local input-output - Input Output - Fare(1) Fare(1) - In-Vehicle Time(2)+Dwelling Time(8) Total IV Time(2) - Initial Wait Time(3) Initial Wait Time(3) - Transfer Wait Time(4) Transfer Wait Time(4) - Transfer Walk Time(5)+Access Walk Time(6)+Egress Walk Time(7) Walk Time(5) - Number of Transfers(9) Number of Transfers(6) - - Author: Wu Sun - wsu@sandag.org, SANDAG - 05/25/09 ver2, - modified 4/16/2012 jef - for AB model - - -***********************************************************************************************************************************/ -Macro "Process transit skims" - shared path,inputDir, outputDir - - periods = {"_EA","_AM","_MD","_PM","_EV"} - modes = {"prem","locl"} - - //output core names, by mode - outMatrixCores={{"Fare","Initial Wait Time","Transfer Wait Time","Walk Time", "Number of Transfers", "Length:CR", - "Length:LR","Length:BRT","Length:EXP","Length:LB","IVT:CR","IVT:LR","IVT:BRT","IVT:EXP","IVT:LB","IVT:Sum","Main Mode"}, - {"Fare","Total IV Time","Initial Wait Time","Transfer Wait Time","Walk Time", "Number of Transfers"}} - - //input output matrix core lookup table, by mode - //Note: if index is set to -1, it represents an aggregation - inOutCoreLookUp={{1,2,3,-1,8,9,10,-1,-1,15,16,17,-1,-1,22}, - {1,-1,3,4,-1,9}} - - //skim aggretation lookup table, by mode - //Note: items match up with those in inOutCoreLookUp array where index is set to -1 - aggLookUp={ {{4,5,6},{11,12},{13,14},{18,19},{20,21}}, - {{2,8},{5,6,7}}} - - //dwelling time core index, by mode - // Note: set to -1 if no dwelling time allocation - dwlTimeIndex={7, -1} - - //dwelling time allocation line-haul modes (line-haul modes that ivt times need to be adjusted using dwelling time), by mode - // Note 1: rail modes are not included because they already include station dwell time. - // Note 2: Set to -1 if no dwelling time allocation - dwlAlloModes={{13,14,15}, -1} - - //ivt core start index, by mode - ivtStartIndex={11,2} - - //number of line-haul modes in output matrices, by mode - numOutModes={5,1} - - //calculate indices, including ivt sum index, main mode index, and ivt end idnex - dim ivtSumIndex[modes.length] - dim ivtEndIndex[modes.length] - for i=1 to modes.length do - ivtSumIndex[i]=outMatrixCores[i].length-1 - ivtEndIndex[i]=ivtStartIndex[i]+numOutModes[i]-1 - end - - //evaluation expression for coding main mode, by mode - //Note: if no main mode coding is necessary, leave null - expr={{"if(([IVT:LB]/ [IVT:Sum]) >0.5) then 8 else if [IVT:Sum]=null then 0 else if ([IVT:EXP]> [IVT:CR] & [IVT:EXP]> [IVT:LR] & [IVT:EXP]> [IVT:BRT] & [IVT:EXP]> [IVT:LB]) then 7 else if ([IVT:BRT]> [IVT:CR] & [IVT:BRT]> [IVT:LR] ) then 6", - "if ([Main Mode]=null & [IVT:LR]> [IVT:CR]) then 5 else if([Main Mode]=null) then 4 else [Main Mode]"},} - - - //-------------------------------------------------- - //This section aggregates matrices, allocates dwell time, and extracts main mode - //-------------------------------------------------- - - for i=1 to modes.length do - for j=1 to periods.length do - - //set input matrix currencies - inputFile = "imp"+modes[i]+periods[j]+".mtx" - inMatrixCurrency=RunMacro("set input matrix currencies",outputDir,inputFile) - - //set up output matrix currencies, empty at this point - outputFile = "imp"+modes[i]+periods[j]+"o.mtx" - matrixLabel = modes[i]+" "+periods[j] - outMatrixCurrency=RunMacro("set output matrix currencies",outputDir,inMatrixCurrency[1],outputFile,outMatrixCores[i],matrixLabel) - - //populate output matrix currencies except 'ivt sum' and 'main mode' cores - outMatrixCurrency=RunMacro("transit aggregate skims",inMatrixCurrency,outMatrixCurrency,inOutCoreLookUp[i],aggLookUp[i]) - - //populate 'main mode' core in output matrix - if expr[i]<>null then do - outMatrixCurrency=RunMacro("set main mode",inMatrixCurrency,outMatrixCurrency,dwlTimeIndex[i],ivtStartIndex[i],ivtEndIndex[i],ivtSumIndex[i],dwlAlloModes[i],expr[i]) - end - - end - end - - Return(1) -EndMacro - -/******************************************************************************************************************************** -set input matrix currencies - -Create and return an array of matrix currencies for the specified file in the specified directory - -*********************************************************************************************************************************/ -Macro "set input matrix currencies" (dir, trnInSkim) - //inputs, keyed to scenarioDirectory - inskim=dir+"\\"+trnInSkim - - //open input transit matrices - inMatrix = OpenMatrix(inskim, "True") - inMatrixCores = GetMatrixCoreNames(inMatrix) - numCoresIn=inMatrixCores.length - matrixInfo=GetMatrixInfo(inMatrix) - - //create inMatrix currencies - dim inMatrixCurrency[numCoresIn] - for i = 1 to numCoresIn do - inMatrixCurrency[i] = CreateMatrixCurrency(inMatrix, inMatrixCores[i], null, null, ) - end - - Return(inMatrixCurrency) -EndMacro - -/******************************************************************************************************************************** -set output matrix currencies - -Create a matrix file and return an array of matrix currencies for the file - -*********************************************************************************************************************************/ - -Macro "set output matrix currencies" (dir, inMatrixCurrency, trnOutSkim, outMatrixCores, label) - //outputs, keyed to scenarioDirectory - outskim=dir+"\\"+trnOutSkim - - //outMatrix core length - numCoresOut=outMatrixCores.length - - //outMatirx structure - dim outMatrixStructure[numCoresOut] - for i=1 to numCoresOut do - outMatrixStructure[i]=inMatrixCurrency - end - - //Create the output transit matrix (with main mode core) - Opts = null - Opts.[Compression] = 1 - Opts.[Tables] = outMatrixCores - Opts.[Type] = "Float" - Opts.[File Name] =outskim - Opts.[Label] = label - outMatrix = CopyMatrixStructure(outMatrixStructure,Opts) - - //create outMatrix currencies - dim outMatrixCurrency[numCoresOut] - for i=1 to numCoresOut do - outMatrixCurrency[i] = CreateMatrixCurrency(outMatrix, outMatrixCores[i], null, null, ) - outMatrixCurrency[i]:=0.0 - end - - //return outMatrixCurrency - return(outMatrixCurrency) -EndMacro - -/******************************************************************************************************************************** -transit aggregate skims - -Aggregate matrix currrencies in the input file and store the results in the output file - -*********************************************************************************************************************************/ - -Macro "transit aggregate skims"(inMatrixCurrency,outMatrixCurrency,inOutCoreLookUp,aggLookUp) - //populate all outMatrix cores except the last 2, using input-output core lookup table and aggregation lookup table - aggCounter=0 - for i = 1 to inOutCoreLookUp.length do - outMatrixCurrency[i]:=0.0 - lookupIndex=inOutCoreLookUp[i] - if lookupIndex=-1 then do - aggCounter=aggCounter+1 - aggIndices=aggLookUp[aggCounter] - for j=1 to aggIndices.length do - //name = inMatrixCurrency[aggIndices[j]].Core - //expr = "if ["+ name + "] = null then 0.0 else [" + name +"]" - //EvaluateMatrixExpression(inMatrixCurrency[aggIndices[j]], expr, , , ) - outMatrixCurrency[i]:=outMatrixCurrency[i]+Nz(inMatrixCurrency[aggIndices[j]]) - end - end - else do - outMatrixCurrency[i]:=Nz(inMatrixCurrency[lookupIndex]) - end - end - - return(outMatrixCurrency) -EndMacro - -/******************************************************************************************************************************** -set main mode - -Set the main mode in the file based upon the expression - -*********************************************************************************************************************************/ - -Macro "set main mode"(inMatrixCurrency,outMatrixCurrency,dwlTimeIndex,ivtStartIndex,ivtEndIndex,ivtSumIndex,dam,expr) - - //initialize outMatrixCurrency[ivtSumIndex], use as a temporary currency for storing sum values - outMatrixCurrency[ivtSumIndex]:=0.0 - - //sum ivt of dwelling allocation modes - for i=1 to dam.length do - outMatrixCurrency[ivtSumIndex]:=outMatrixCurrency[ivtSumIndex]+outMatrixCurrency[dam[i]] - end - - //allocate dwelling time - for i=1 to dam.length do - outMatrixCurrency[dam[i]]:=outMatrixCurrency[dam[i]]*(1.0+inMatrixCurrency[dwlTimeIndex]/outMatrixCurrency[ivtSumIndex]) - end - - //zero out outMatrixCurrency[ivtSumIndex] - outMatrixCurrency[ivtSumIndex]:=0.0 - - //set total ivt time to outMatrix - for i=ivtStartIndex to ivtEndIndex do - outMatrixCurrency[ivtSumIndex]:=outMatrixCurrency[ivtSumIndex]+Nz(outMatrixCurrency[i]) - end - - //set main mode to outMatrix using an expression - for i=1 to expr.length do - EvaluateMatrixExpression(outMatrixCurrency[outMatrixCurrency.length], expr[i],,, ) - end - - return(outMatrixCurrency) -EndMacro -/********************************************************************************* - -Update transit time fields - -This macro updates transit time fields with MSA times from flow tables - -The following fields are updated: - -xxField_yy where - -Field is TM - -xx is AB or BA -yy is period - EA: Early AM - AM: AM peak - MD: Midday - PM: PM peak - EV: Evening - - TODO: Update times for transit based on postload code pasted below - -*********************************************************************************/ -Macro "Update transit time fields" - - shared path, inputDir, outputDir - - periods = {"_EA","_AM","_MD","_PM","_EV"} - - // input files - db_file = outputDir + "\\transit.dbd" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file) - SetLayer(link_lyr) - vw = GetView() - - /* - if(shoulder) then !adjust times for freeway shoulder operation - xxspd=xlen*60.0/xtime - if(xxspd.lt.35.0) xtime=xlen*(60.0/35.0) - endif - tranlt(ipk,idir)=xtime - brtlt(ipk,idir)=xtime - - adjust time for priority treatment - if(aatfc(1).gt.1.and.aatfc(1).lt.8.and.artxlkid(aatid).and. - * myear.eq.2030) then !adjust times for priority treatment - xtime=xtime*0.90 - - if(aatcnt(idir,1).eq.5) then - hovlt(ipk,idir)=hovlt(ipk,idir)*hovfac(ipk) !hovfac = 0.33 for peak, 1 for off-peak - tranlt(ipk,idir)=tranlt(ipk,idir)*hovfac(ipk) - brtlt(ipk,idir)=brtlt(ipk,idir)*hovfac(ipk) - - */ - - //Recompute generalized cost using MSA cost in flow table, for links with MSA cost (so that transit only links aren't overwritten with null) - for i = 1 to periods.length do - flowTable = outputDir+"\\hwyload"+periods[i]+".bin" - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"AB time"+periods[i] } - Opts.Global.Fields = {"ABTM"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if AB_MSA_Cost<>null then AB_MSA_Cost else ABTM"+periods[i] } - ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ok then goto quit - - // The Dataview Set is a joined view of the link layer and the flow table, based on link ID - Opts.Input.[Dataview Set] = {{db_file+"|"+link_lyr, flowTable, {"ID"}, {"ID1"}},"BA time"+periods[i]} - Opts.Global.Fields = {"BATM"+periods[i]} // the field to fill - Opts.Global.Method = "Formula" // the fill method - Opts.Global.Parameter = {"if BA_MSA_Cost<>null then BA_MSA_Cost else BATM"+periods[i] } - ok = RunMacro("TCB Run Operation", "Fill Dataview", Opts, &Ret) - if !ok then goto quit - - end - - ok=1 - quit: - RunMacro("close all") - return(ok) - -EndMacro - -/*************************************************************************************** -Create rail net - -This script creates two rail networks; one for commuter rail and one for light-rail. The -networks are used to create light rail zonal fare matrix by skimming light rail stops for LR network. - -to create lrzone skim matrix replaced the value of lrzone by light rail fares - -Inputs: - input\modenu061.dbf Mode table - output\transit.dbd Transit line layer - output\transitrt.rts Transit route system - -Outputs: - output\cr.tnw Commuter rail transit network - output\lr.tnw Light rail transit network - -3/16/05 zou c -***************************************************************************************/ -Macro "Create rail net" - - shared path,mxtap, inputDir, outputDir - - db_file=outputDir+"\\transit.dbd" - rte_file=outputDir+"\\transitrt.rts" - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - db_link_lyr=db_file+"|"+link_lyr - db_node_lyr=db_file+"|"+node_lyr - rte_lyr = RunMacro("TCB Add RS Layers", rte_file, , ) - - stp_lyr = GetStopsLayerFromRS(rte_lyr) - db_rte_lyr = rte_file + "|" + rte_lyr - db_stp_lyr = rte_file + "|" + stp_lyr - - //route selection set names - sets = {"crpk","lrpk"} - - //selection query strings - //rcu - changed query to drop routes 399 and 599 12/27/06 - // todo: fix this - query_strs={"select * where am_headway >0 and route_name <> '399103' and route_name <> '399203' and route_name <> '599101' and route_name <> '599201'", - "select * where am_headway >0 and route_name <> '399103' and route_name <> '399203' and route_name <> '599101' and route_name <> '599201'"} - trnets = {"cr.tnw","lr.tnw"} - - // Build 4(2?) Transit Networks - for i = 1 to trnets.length do - Opts = null - Opts.Input.[Transit RS] = rte_file - Opts.Input.[RS Set] = {db_rte_lyr, rte_lyr,sets[i],query_strs[i]} - Opts.Input.[Walk Link Set] = {db_link_lyr, link_lyr, "walklink", "Select * where MINMODE<4"} - Opts.Input.[Stop Set] = {db_stp_lyr, stp_lyr} - Opts.Global.[Network Options].[Route Attributes].mode = {rte_lyr+".mode"} - Opts.Global.[Network Options].[Route Attributes].AM_HEADWAY = {rte_lyr+".AM_HEADWAY"} - Opts.Global.[Network Options].[Route Attributes].OP_HEADWAY = {rte_lyr+".OP_HEADWAY"} - Opts.Global.[Network Options].[Route Attributes].FARE = {rte_lyr+".FARE"} - Opts.Global.[Network Options].[Street Attributes].Length = {link_lyr+".Length",link_lyr+".Length"} - Opts.Global.[Network Options].[Street Attributes].[*TM_MD] = {link_lyr+".ABTM_MD", link_lyr+".BATM_MD"} - Opts.Global.[Network Options].[Street Attributes].[*TM_AM] = {link_lyr+".ABTM_AM", link_lyr+".BATM_AM"} - Opts.Global.[Network Options].[Street Attributes].LRTZONE = {link_lyr+".LRTZONE", link_lyr+".LRTZONE"} - Opts.Global.[Network Options].[Street Attributes].CRZONE = {link_lyr+".CRZONE", link_lyr+".CRZONE"} - Opts.Global.[Network Options].Walk = "Yes" - Opts.Global.[Network Options].Overide = {stp_lyr+".ID", stp_lyr+".nearnode"} - Opts.Global.[Network Options].[Link Attributes] = - {{"Length", {link_lyr+".Length",link_lyr+".Length"}, "SUMFRAC"}, - {"*TM_MD", {link_lyr+".ABTM_MD", link_lyr+".BATM_MD"}, "SUMFRAC"}, - {"*TM_AM", {link_lyr+".ABTM_AM", link_lyr+".BATM_AM"}, "SUMFRAC"}, - {"lrtzone",{link_lyr+".lrtzone", link_lyr+".lrtzone"}, "SUMFRAC"}, - {"crzone", {link_lyr+".crzone", link_lyr+".crzone"}, "SUMFRAC"}} - Opts.Global.[Network Options].[Mode Field] = rte_lyr+".Mode" - Opts.Global.[Network Options].[Walk Mode] = link_lyr+".minmode" - Opts.Output.[Network File] = outputDir+"\\"+trnets[i] - RunMacro("HwycadLog",{"createtrnnet.rsc: Create rail net","Build Transit Network: "+trnets[i]}) - ok = RunMacro("TCB Run Operation", i, "Build Transit Network", Opts) - - if !ok then goto quit - end - - mode_tb = inputDir+"\\modenu061.dbf" - mode_vw = RunMacro("TCB OpenTable",,, {mode_tb}) - - // xfer_tb = path+"\\xferfares.dbf" - // xferf_vw = RunMacro("TCB OpenTable",,, {xfer_tb}) - - //transit network settings - impds = {"*tma", "*tma"} - impwts = {mode_vw+".wt_ivtpk", mode_vw+".wt_ivtpk"} - iwtwts = {mode_vw+".wt_fwtpk", mode_vw+".wt_fwtpk"} - xwaitwts = {mode_vw+".wt_xwtpk", mode_vw+".wt_xwtpk"} - modeused={mode_vw+".crmode",mode_vw+".lrmode"} - - for i = 1 to trnets.length do - Opts = null - Opts.Input.[Transit RS] = rte_file - Opts.Input.[Transit Network] = outputDir+"\\"+trnets[i] - Opts.Input.[Mode Table] = {mode_tb} - // Opts.Input.[Mode Cost Table] = {xfer_tb} - //Opts.Input.[Fare Currency] = {inputDir+"\\fare.mtx", "coaster fare", , } - Opts.Input.[Centroid Set] = {db_node_lyr,node_lyr, "Selection", "Select * where ID<"+i2s(mxtap)} - Opts.Field.[Link Impedance] = "*TM" - Opts.Field.[Route Headway] = headways[i] - Opts.Field.[Route Fare] = "Fare" - Opts.Field.[Stop Zone ID] = "farezone" - Opts.Field.[Mode Fare Type] = mode_vw+".faretype" - Opts.Field.[Mode Fare Core] = mode_vw+".farefield" - Opts.Field.[Mode Fare Weight] = farewts[i] - Opts.Field.[Mode Xfer Time] = mode_vw+".xferpentm" - Opts.Field.[Mode Xfer Weight] = mode_vw+".wtxfertm" - Opts.Field.[Mode Impedance] = trntime[i] //impedance by transit mode - Opts.Field.[Mode Imp Weight] = impwts[i] - Opts.Field.[Mode IWait Weight] = iwtwts[i] - Opts.Field.[Mode XWait Weight] = xwaitwts[i] - Opts.Field.[Mode Dwell Weight] = impwts[i] - Opts.Field.[Mode Dwell On Time] = mode_vw+".dwelltime" - Opts.Field.[Mode Used] = modeused[j] - Opts.Field.[Mode Access] = mode_vw+".mode_acces" - Opts.Field.[Mode Egress] = mode_vw+".mode_egres" - Opts.Field.[Inter-Mode Xfer From] =xferf_vw+".from" - Opts.Field.[Inter-Mode Xfer To] = xferf_vw+".to" - Opts.Field.[Inter-Mode Xfer Stop] = xferf_vw+".stop" - Opts.Field.[Inter-Mode Xfer Proh] = xferf_vw+".prohibitio" - Opts.Field.[Inter-Mode Xfer Time] = xferf_vw+".xfer_penal" - Opts.Field.[Inter-Mode Xfer Fare] = xferf_vw+".fare" - Opts.Field.[Inter-Mode Xfer Wait] = xferf_vw+".wait_time" - Opts.Global.[Class Names] = {"Class 1"} - Opts.Global.[Class Description] = {"Class 1"} - Opts.Global.[current class] = "Class 1" - Opts.Global.[Global Fare Type] = "Flat" - Opts.Global.[Global Fare Value] = 2.25 - Opts.Global.[Global Xfer Fare] = 0 - Opts.Global.[Global Fare Core] = "coaster fare" - Opts.Global.[Global Fare Weight] = 1 - Opts.Global.[Global Imp Weight] = 1 - Opts.Global.[Global Init Weight] = 1 - Opts.Global.[Global Xfer Weight] = 1 - Opts.Global.[Global IWait Weight] = 2 - Opts.Global.[Global XWait Weight] = 2 - Opts.Global.[Global Dwell Weight] = 1 - Opts.Global.[Global Dwell On Time] = 0 - Opts.Global.[Global Dwell Off Time] = 0 - Opts.Global.[Global Headway] = 30 - Opts.Global.[Global Init Time] = 0 - Opts.Global.[Global Xfer Time] = 10 - Opts.Global.[Global Max IWait] = 60 - Opts.Global.[Global Min IWait] = 2 - Opts.Global.[Global Max XWait] = 60 - Opts.Global.[Global Min XWait] = 2 - Opts.Global.[Global Layover Time] = 5 - Opts.Global.[Global Max WACC Path] = 20 - Opts.Global.[Global Max Access] = 30 - Opts.Global.[Global Max Egress] = 30 - Opts.Global.[Global Max Transfer] = 20 - Opts.Global.[Global Max Imp] = 180 - Opts.Global.[Value of Time] = vot[i] - Opts.Global.[Max Xfer Number] = 3 - Opts.Global.[Max Trip Time] = 999 - Opts.Global.[Walk Weight] = 1.8 - Opts.Global.[Zonal Fare Method] = 1 - Opts.Global.[Interarrival Para] = 0.5 - Opts.Global.[Path Threshold] = 0 - Opts.Flag.[Use All Walk Path] = "No" - Opts.Flag.[Use Mode] = "Yes" - Opts.Flag.[Use Mode Cost] = "Yes" - Opts.Flag.[Combine By Mode] = "Yes" - Opts.Flag.[Fare By Mode] = "No" - Opts.Flag.[M2M Fare Method] = 2 - Opts.Flag.[Fare System] = 3 - Opts.Flag.[Use Park and Ride] = "No" - Opts.Flag.[Use Egress Park and Ride] = "No" - Opts.Flag.[Use P&R Walk Access] = "No" - Opts.Flag.[Use P&R Walk Egress] = "No" - Opts.Flag.[Use Parking Capacity] = "No" - RunMacro("HwycadLog",{"createtrnnet.rsc: create rail net","Transit Network Setting PF: "+trnets[i]}) - ok = RunMacro("TCB Run Operation", i, "Transit Network Setting PF", Opts) - if !ok then goto quit - end - - ok=1 - quit: -// if fpr<> null then closefile(fpr) - Return(ok) -EndMacro - - -/******************************************************************************* -Create transit networks - -create 6 transit network from the route system: -local bus op, local bus pk, premium op, premium pk - -Input files: - output\transit.dbd - output\transitrt.rts - input\timexfer.dbf - input\mode3tod.dbf - dbf file for mode table - input\modexfer.dbf - output\fare.mtx - - -output files: - localop.tnw - off peak local bus transit network - localpk.tnw - peak local bus transit network - prepop.tnw - off peak premium service transit network - preppk.tnw - peak premium service transit network - -*****************************************************************************/ -Macro "Create transit networks" - shared path, inputDir, outputDir, mxtap - - db_file=outputDir+"\\transit.dbd" - rte_file=outputDir+"\\transitrt.rts" - timexfer_tb = inputDir+"\\timexfer.bin" - mode_tb = inputDir+"\\mode5tod.dbf" - modexfer_tb = inputDir+"\\modexfer.dbf" - - periods = {"_EA","_AM","_MD","_PM","_EV"} - modes = {"prem","locl"} - - // timexfer_per = {"NO", "YES", "NO", "YES", "NO"} - timexfer_per = {"NO", "YES", "NO", "NO", "NO"} - timexfer_mod = {"YES", "NO"} - - - - ftype = RunMacro("G30 table type", timexfer_tb) - view = OpenTable("test", ftype, {timexfer_tb}) - if view = null then do - RunMacro("TCB Error", "Can't open table " + timexfer_tb) - Return(0) - end - - {node_lyr, link_lyr} = RunMacro("TCB Add DB Layers", db_file,,) - ok = (node_lyr <> null && link_lyr <> null) - if !ok then goto quit - - db_link_lyr=db_file+"|"+link_lyr - db_node_lyr=db_file+"|"+node_lyr - rte_lyr = RunMacro("TCB Add RS Layers", rte_file, , ) - stp_lyr = GetStopsLayerFromRS(rte_lyr) - db_rte_lyr = rte_file + "|" + rte_lyr - db_stp_lyr = rte_file + "|" + stp_lyr - - /* - fpr=OpenFile(path+"\\hwycad.log","a") - mytime=GetDateAndTime() - writeline(fpr,mytime+", create trnets") - */ - - //selection query strings: vary by period - query_strs={"select * where op_headway >0", - "select * where am_headway >0", - "select * where op_headway >0", - "select * where pm_headway >0", - "select * where op_headway >0"} - - // Build Transit Networks - for i = 1 to periods.length do - for j = 1 to modes.length do - Opts = null - Opts.Input.[Transit RS] = rte_file - Opts.Input.[RS Set] = {db_rte_lyr, rte_lyr,modes[j]+periods[i],query_strs[i]} - Opts.Input.[Walk Link Set] = {db_link_lyr, link_lyr, "walklink", "Select * where MINMODE<4"} - Opts.Input.[Stop Set] = {db_stp_lyr, stp_lyr} - Opts.Global.[Network Options].[Route Attributes].mode = {rte_lyr+".mode"} - Opts.Global.[Network Options].[Route Attributes].OP_HEADWAY = {rte_lyr+".OP_HEADWAY"} - Opts.Global.[Network Options].[Route Attributes].AM_HEADWAY = {rte_lyr+".AM_HEADWAY"} - Opts.Global.[Network Options].[Route Attributes].PM_HEADWAY = {rte_lyr+".PM_HEADWAY"} - Opts.Global.[Network Options].[Route Attributes].FARE = {rte_lyr+".FARE"} - Opts.Global.[Network Options].[Stop Attributes].farezone = {stp_lyr+".farezone"} - Opts.Global.[Network Options].[Street Attributes].Length = {link_lyr+".Length",link_lyr+".Length"} - Opts.Global.[Network Options].[Street Attributes].FWYLEN = {link_lyr+".FWYLEN", link_lyr+".FWYLEN"} - Opts.Global.[Network Options].[Street Attributes].[*TM] = {link_lyr+".ABTM"+periods[i], link_lyr+".BATM"+periods[i]} - // Opts.Global.[Network Options].[Street Attributes].[*PRETIME] = {link_lyr+".ABPRETIME"+periods[i], link_lyr+".BAPRETIME"+periods[i]} - // Opts.Global.[Network Options].[Street Attributes].[*LOCTIME] = {link_lyr+".ABLOCTIME"+periods[i], link_lyr+".BALOCTIME"+periods[i]} - Opts.Global.[Network Options].[Street Attributes].LRTZONE = {link_lyr+".LRTZONE", link_lyr+".LRTZONE"} - Opts.Global.[Network Options].[Street Attributes].CRZONE = {link_lyr+".CRZONE", link_lyr+".CRZONE"} - Opts.Global.[Network Options].Walk = "Yes" - Opts.Global.[Network Options].Overide = {stp_lyr+".ID", stp_lyr+".nearnode"} - Opts.Global.[Network Options].[Link Attributes] = - {{"Length", {link_lyr+".Length",link_lyr+".Length"}, "SUMFRAC"}, - {"fwylen",{link_lyr+".fwylen", link_lyr+".fwylen"}, "SUMFRAC"}, - {"*TM", {link_lyr+".ABTM"+periods[i], link_lyr+".BATM"+periods[i]}, "SUMFRAC"}, - // {"*PRETIME", {link_lyr+".ABPRETIME"+periods[i], link_lyr+".BAPRETIME"+periods[i]}, "SUMFRAC"}, - // {"*LOCTIME", {link_lyr+".ABLOCTIME"+periods[i], link_lyr+".BALOCTIME"+periods[i]}, "SUMFRAC"}, - {"lrtzone",{link_lyr+".lrtzone", link_lyr+".lrtzone"}, "SUMFRAC"}, - {"crzone", {link_lyr+".crzone", link_lyr+".crzone"}, "SUMFRAC"}} - Opts.Global.[Network Options].[Mode Field] = rte_lyr+".Mode" - Opts.Global.[Network Options].[Walk Mode] = link_lyr+".minmode" - Opts.Output.[Network File] = outputDir+"\\"+modes[j]+periods[i]+".tnw" - - ok = RunMacro("TCB Run Operation", i, "Build Transit Network", Opts) - if !ok then goto quit - end - end - - dif2=GetDirectoryInfo(timexfer_tb,"file") - if dif2.length>0 then blnxfer=1 else blnxfer=0 - - mode_vw = RunMacro("TCB OpenTable",,, {mode_tb}) - xferf_vw = RunMacro("TCB OpenTable",,, {modexfer_tb}) - - // following vary by period - headways = {"op_headway", "am_headway", "op_headway", "pm_headway", "op_headway"} - trntime= {mode_vw+".TRNTIME_EA",mode_vw+".TRNTIME_AM",mode_vw+".TRNTIME_MD",mode_vw+".TRNTIME_PM",mode_vw+".TRNTIME_EV"}//transit travel time by mode - impwts = {mode_vw+".wt_ivtop", mode_vw+".wt_ivtpk", mode_vw+".wt_ivtop", mode_vw+".wt_ivtpk", mode_vw+".wt_ivtop"} - iwtwts = {mode_vw+".wt_fwtop", mode_vw+".wt_fwtpk", mode_vw+".wt_fwtop", mode_vw+".wt_fwtpk", mode_vw+".wt_fwtop"} - xwaitwts = {mode_vw+".wt_xwtop", mode_vw+".wt_xwtpk", mode_vw+".wt_xwtop", mode_vw+".wt_xwtpk", mode_vw+".wt_xwtop"} - farewts= {mode_vw+".wt_fareop", mode_vw+".wt_farepk", mode_vw+".wt_fareop", mode_vw+".wt_farepk", mode_vw+".wt_fareop"} - vot = {0.05, 0.1, 0.05, 0.1, 0.05} //PB recommended 0.1 - wt_walk = { 1.6, 1.8, 1.6, 1.8, 1.6} - - // following varies by mode - modeused={mode_vw+".premode",mode_vw+".locmode"} - faresys={3, 1} //3, mixed fare, 1, flat fare - - - //transit network settings in TransCAD 6.0 R2 - for i = 1 to periods.length do - for j = 1 to modes.length do - Opts = null - Opts.Input.[Transit RS] = rte_file - Opts.Input.[Transit Network] = outputDir+"\\"+modes[j]+periods[i]+".tnw" - Opts.Input.[Mode Table] = {mode_tb} - Opts.Input.[Mode Cost Table] = {modexfer_tb} - Opts.Input.[Fare Currency] = {inputDir+"\\fare.mtx", "coaster fare", , } - // add timed transfers to premium peak networks (?) - if blnxfer=1 and timexfer_per[i] ="YES" and timexfer_mod[j]="YES" then Opts.Input.[Xfer Wait Table] = {timexfer_tb} - Opts.Input.[Centroid Set] = {db_node_lyr,node_lyr, "Selection", "Select * where ID<"+i2s(mxtap)} - Opts.Field.[Link Impedance] = "*TM" - Opts.Field.[Route Headway] = headways[i] - Opts.Field.[Route Fare] = "Fare" - Opts.Field.[Stop Zone ID] = "farezone" - Opts.Field.[Mode Fare Type] = mode_vw+".faretype" - Opts.Field.[Mode Fare Core] = mode_vw+".farefield" - Opts.Field.[Mode Fare Weight] = farewts[i] - Opts.Field.[Mode Xfer Time] = mode_vw+".xferpentm" - Opts.Field.[Mode Xfer Weight] = mode_vw+".wtxfertm" - Opts.Field.[Mode Impedance] = trntime[i] //impedance by transit mode - Opts.Field.[Mode Imp Weight] = impwts[i] - Opts.Field.[Mode IWait Weight] = iwtwts[i] - Opts.Field.[Mode XWait Weight] = xwaitwts[i] - Opts.Field.[Mode Dwell Weight] = impwts[i] - Opts.Field.[Mode Dwell On Time] = mode_vw+".dwelltime" - Opts.Field.[Mode Used] = modeused[j] - Opts.Field.[Mode Access] = mode_vw+".mode_acces" - Opts.Field.[Mode Egress] = mode_vw+".mode_egres" - Opts.Field.[Inter-Mode Xfer From] =xferf_vw+".from" - Opts.Field.[Inter-Mode Xfer To] = xferf_vw+".to" - Opts.Field.[Inter-Mode Xfer Stop] = xferf_vw+".stop" - Opts.Field.[Inter-Mode Xfer Proh] = xferf_vw+".prohibitio" - Opts.Field.[Inter-Mode Xfer Time] = xferf_vw+".xfer_penal" - Opts.Field.[Inter-Mode Xfer Fare] = xferf_vw+".fare" - Opts.Field.[Inter-Mode Xfer Wait] = xferf_vw+".wait_time" - Opts.Global.[Class Names] = {"Class 1"} - Opts.Global.[Class Description] = {"Class 1"} - Opts.Global.[current class] = "Class 1" - Opts.Global.[Global Fare Type] = "Flat" - Opts.Global.[Global Fare Value] = 2.25 - Opts.Global.[Global Xfer Fare] = 0 - Opts.Global.[Global Fare Core] = "coaster fare" - Opts.Global.[Global Fare Weight] = 1 - Opts.Global.[Global Imp Weight] = 1 - Opts.Global.[Global Init Weight] = 1 - Opts.Global.[Global Xfer Weight] = 1 - Opts.Global.[Global IWait Weight] = 2 - Opts.Global.[Global XWait Weight] = 2 - Opts.Global.[Global Dwell Weight] = 1 - Opts.Global.[Global Dwell On Time] = 0 - Opts.Global.[Global Dwell Off Time] = 0 - Opts.Global.[Global Headway] = 30 - Opts.Global.[Global Init Time] = 0 - Opts.Global.[Global Xfer Time] = 10 - Opts.Global.[Global Max IWait] = 60 - Opts.Global.[Global Min IWait] = 2 - Opts.Global.[Global Max XWait] = 60 - Opts.Global.[Global Min XWait] = 2 - Opts.Global.[Global Layover Time] = 5 - Opts.Global.[Global Max WACC Path] = 20 - Opts.Global.[Global Max Access] = 30 - Opts.Global.[Global Max Egress] = 30 - Opts.Global.[Global Max Transfer] = 20 - Opts.Global.[Global Max Imp] = 180 - Opts.Global.[Value of Time] = vot[i] - Opts.Global.[Max Xfer Number] = 3 - Opts.Global.[Max Trip Time] = 999 - Opts.Global.[Walk Weight] = 1.8 - Opts.Global.[Zonal Fare Method] = 1 - Opts.Global.[Interarrival Para] = 0.5 - Opts.Global.[Path Threshold] = 0 - Opts.Flag.[Use All Walk Path] = "No" - Opts.Flag.[Use Mode] = "Yes" - Opts.Flag.[Use Mode Cost] = "Yes" - Opts.Flag.[Combine By Mode] = "Yes" - Opts.Flag.[Fare By Mode] = "No" - Opts.Flag.[M2M Fare Method] = 2 - Opts.Flag.[Fare System] = 3 - Opts.Flag.[Use Park and Ride] = "No" - Opts.Flag.[Use Egress Park and Ride] = "No" - Opts.Flag.[Use P&R Walk Access] = "No" - Opts.Flag.[Use P&R Walk Egress] = "No" - Opts.Flag.[Use Parking Capacity] = "No" - ok = RunMacro("TCB Run Operation", i, "Transit Network Setting PF", Opts) - if !ok then goto quit - end - end - - ok=1 - quit: - if fpr <> null then closefile(fpr) - RunMacro("close all") - Return(ok) -EndMacro diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java deleted file mode 100644 index 8c13f8d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesDMU.java +++ /dev/null @@ -1,236 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Household; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; - -public class AccessibilitiesDMU - implements Serializable, VariableTable -{ - - - protected transient Logger logger = Logger.getLogger(AccessibilitiesDMU.class); - - protected HashMap methodIndexMap; - - private double[] workSizeTerms; - private double[] schoolSizeTerms; - private double[] sizeTerms; - // size - // terms - // purpose - // (as - // defined - // in - // uec), - // for - // a - // given - // mgra - - private double[] logsums; // logsums/accessibilities, - // for - // a - // given - // mgra-pair - - private Household hhObject; - - // the alternativeData tabledataset has the following fields - // sizeTermIndex: Used to index into the sizeTerms array - // logsumIndex: Used to index into the logsums array - private TableDataSet alternativeData; - private int logsumIndex, sizeIndex; - - private int autoSufficiency; - - public AccessibilitiesDMU() - { - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getAutoSufficiency", 0); - methodIndexMap.put("getSizeTerm", 1); - methodIndexMap.put("getLogsum", 2); - methodIndexMap.put("getNumPreschool", 3); - methodIndexMap.put("getNumGradeSchoolStudents", 4); - methodIndexMap.put("getNumHighSchoolStudents", 5); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getAutoSufficiency(); - case 1: - return getSizeTerm(arrayIndex); - case 2: - return getLogsum(arrayIndex); - case 3: - return getNumPreschool(); - case 4: - return getNumGradeSchoolStudents(); - case 5: - return getNumHighSchoolStudents(); - case 6: - return getWorkSizeTerm(arrayIndex); - case 7: - return getSchoolSizeTerm(arrayIndex); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public void setAlternativeData(TableDataSet alternativeData) - { - this.alternativeData = alternativeData; - logsumIndex = alternativeData.getColumnPosition("logsumIndex"); - sizeIndex = alternativeData.getColumnPosition("sizeTermIndex"); - - } - - public int getNumPreschool() - { - return hhObject.getNumPreschool(); - } - - public int getNumGradeSchoolStudents() - { - return hhObject.getNumGradeSchoolStudents(); - } - - public int getNumHighSchoolStudents() - { - return hhObject.getNumHighSchoolStudents(); - } - - public int getAutoSufficiency() - { - return autoSufficiency; - } - - public void setHouseholdObject(Household hh) - { - hhObject = hh; - } - - public void setAutoSufficiency(int autoSufficiency) - { - this.autoSufficiency = autoSufficiency; - } - - public void setSizeTerms(double[] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - public void setWorkSizeTerms(double[] sizeTerms) - { - workSizeTerms = sizeTerms; - } - - public void setSchoolSizeTerms(double[] sizeTerms) - { - schoolSizeTerms = sizeTerms; - } - - public void setLogsums(double[] logsums) - { - this.logsums = logsums; - } - - /** - * For the given alternative, look up the work size term and return it. - * - * @param alt - * @return - */ - public double getWorkSizeTerm(int alt) - { - - int index = (int) alternativeData.getValueAt(alt, sizeIndex); - - return workSizeTerms[index]; - } - - /** - * For the given alternative, look up the school size term and return it. - * - * @param alt - * @return - */ - public double getSchoolSizeTerm(int alt) - { - - int index = (int) alternativeData.getValueAt(alt, sizeIndex); - - return schoolSizeTerms[index]; - } - - /** - * For the given alternative, look up the size term and return it. - * - * @param alt - * @return - */ - public double getSizeTerm(int alt) - { - - int index = (int) alternativeData.getValueAt(alt, sizeIndex); - - return sizeTerms[index]; - } - - /** - * For the given alternative, look up the size term and return it. - * - * @param alt - * @return - */ - public double getLogsum(int alt) - { - - int index = (int) alternativeData.getValueAt(alt, logsumIndex); - - return logsums[index]; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java deleted file mode 100644 index d44f2fe..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AccessibilitiesTable.java +++ /dev/null @@ -1,407 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import org.apache.log4j.Logger; -import com.pb.common.datafile.CSVFileWriter; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class holds the accessibility table that is built, or reads it from a - * previously written file. - * - * @author Jim Hicks - * @version May, 2011 - */ -public final class AccessibilitiesTable - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(AccessibilitiesTable.class); - - private static final int NONMANDATORY_AUTO_ACCESSIBILITY_FIELD_NUMBER = 1; - private static final int NONMANDATORY_TRANSIT_ACCESSIBILITY_FIELD_NUMBER = 2; - private static final int NONMANDATORY_NONMOTOR_ACCESSIBILITY_FIELD_NUMBER = 3; - private static final int NONMANDATORY_SOV_0_ACCESSIBILITY_FIELD_NUMBER = 4; - private static final int NONMANDATORY_SOV_1_ACCESSIBILITY_FIELD_NUMBER = 5; - private static final int NONMANDATORY_SOV_2_ACCESSIBILITY_FIELD_NUMBER = 6; - private static final int NONMANDATORY_HOV_0_ACCESSIBILITY_FIELD_NUMBER = 7; - private static final int NONMANDATORY_HOV_1_ACCESSIBILITY_FIELD_NUMBER = 8; - private static final int NONMANDATORY_HOV_2_ACCESSIBILITY_FIELD_NUMBER = 9; - private static final int SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 10; - private static final int SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 11; - private static final int SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 12; - private static final int MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 13; - private static final int MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 14; - private static final int MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 15; - private static final int EAT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 16; - private static final int EAT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 17; - private static final int EAT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 18; - private static final int VISIT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 19; - private static final int VISIT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 20; - private static final int VISIT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 21; - private static final int DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 22; - private static final int DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 23; - private static final int DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 24; - private static final int ESCORT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX = 25; - private static final int ESCORT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX = 26; - private static final int ESCORT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX = 27; - private static final int SHOP_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 28; - private static final int SHOP_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 29; - private static final int SHOP_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 30; - private static final int MAINT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 31; - private static final int MAINT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 32; - private static final int MAINT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 33; - private static final int EAT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 34; - private static final int EAT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 35; - private static final int EAT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 36; - private static final int VISIT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 37; - private static final int VISIT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 38; - private static final int VISIT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 39; - private static final int DISCR_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 40; - private static final int DISCR_ACCESSIBILITY_SOV_SUFFICIENT_INDEX = 41; - private static final int DISCR_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 42; - private static final int ATWORK_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX = 43; - private static final int ATWORK_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX = 44; - private static final int TOTAL_EMPLOYMENT_ACCESSIBILITY_INDEX = 45; - private static final int ATWORK_ACCESSIBILITY_NMOT_INDEX = 46; - private static final int ALLHH_ACCESSIBILITY_TRANSIT_INDEX = 47; - private static final int NONMANDATORY_MAAS_ACCESSIBILITY_FIELD_NUMBER = 48; - - // accessibilities by mgra, accessibility alternative - private float[][] accessibilities; - - /** - * array of previously computed accessibilities - * - * @param computedAccessibilities - * array of accessibilities - * - * use this constructor if the accessibilities were calculated as - * opposed to read from a file. - */ - public AccessibilitiesTable(float[][] computedAccessibilities) - { - accessibilities = computedAccessibilities; - } - - /** - * file name for store accessibilities - * - * @param accessibilitiesInputFileName - * path and filename of file to read - * - * use this constructor if the accessibilities are to be read - * from a file. - */ - public AccessibilitiesTable(String accessibilitiesInputFileName) - { - readAccessibilityTableFromFile(accessibilitiesInputFileName); - } - - private void readAccessibilityTableFromFile(String fileName) - { - - File accFile = new File(fileName); - - // read in the csv table - TableDataSet accTable; - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - accTable = reader.readFile(accFile); - - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading accessibility data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(); - } - - // create accessibilities array as a 1-based array - float[][] temp = accTable.getValues(); - accessibilities = new float[temp.length + 1][]; - for (int i = 0; i < temp.length; i++) - { - accessibilities[i + 1] = new float[temp[i].length]; - for (int j = 0; j < temp[i].length; j++) - { - accessibilities[i + 1][j] = temp[i][j]; - } - } - - } - - public void writeAccessibilityTableToFile(String accFileName) - { - - File accFile = new File(accFileName); - - // the accessibilities array is indexed by mgra values which might no be - // consecutive. - // create an arraylist of data table rows, with the last field being the - // mgra value, - // convert to a tabledataset, then write to a csv file. - - ArrayList dataColumnHeadings = new ArrayList(); - dataColumnHeadings.add("NONMAN_AUTO"); - dataColumnHeadings.add("NONMAN_TRANSIT"); - dataColumnHeadings.add("NONMAN_NONMOTOR"); - dataColumnHeadings.add("NONMAN_SOV_0"); - dataColumnHeadings.add("NONMAN_SOV_1"); - dataColumnHeadings.add("NONMAN_SOV_2"); - dataColumnHeadings.add("NONMAN_HOV_0"); - dataColumnHeadings.add("NONMAN_HOV_1"); - dataColumnHeadings.add("NONMAN_HOV_2"); - dataColumnHeadings.add("SHOP_HOV_0"); - dataColumnHeadings.add("SHOP_HOV_1"); - dataColumnHeadings.add("SHOP_HOV_2"); - dataColumnHeadings.add("MAINT_HOV_0"); - dataColumnHeadings.add("MAINT_HOV_1"); - dataColumnHeadings.add("MAINT_HOV_2"); - dataColumnHeadings.add("EAT_HOV_0"); - dataColumnHeadings.add("EAT_HOV_1"); - dataColumnHeadings.add("EAT_HOV_2"); - dataColumnHeadings.add("VISIT_HOV_0"); - dataColumnHeadings.add("VISIT_HOV_1"); - dataColumnHeadings.add("VISIT_HOV_2"); - dataColumnHeadings.add("DISCR_HOV_0"); - dataColumnHeadings.add("DISCR_HOV_1"); - dataColumnHeadings.add("DISCR_HOV_2"); - dataColumnHeadings.add("ESCORT_HOV_0"); - dataColumnHeadings.add("ESCORT_HOV_1"); - dataColumnHeadings.add("ESCORT_HOV_2"); - dataColumnHeadings.add("SHOP_SOV_0"); - dataColumnHeadings.add("SHOP_SOV_1"); - dataColumnHeadings.add("SHOP_SOV_2"); - dataColumnHeadings.add("MAINT_SOV_0"); - dataColumnHeadings.add("MAINT_SOV_1"); - dataColumnHeadings.add("MAINT_SOV_2"); - dataColumnHeadings.add("EAT_SOV_0"); - dataColumnHeadings.add("EAT_SOV_1"); - dataColumnHeadings.add("EAT_SOV_2"); - dataColumnHeadings.add("VISIT_SOV_0"); - dataColumnHeadings.add("VISIT_SOV_1"); - dataColumnHeadings.add("VISIT_SOV_2"); - dataColumnHeadings.add("DISCR_SOV_0"); - dataColumnHeadings.add("DISCR_SOV_1"); - dataColumnHeadings.add("DISCR_SOV_2"); - dataColumnHeadings.add("ATWORK_SOV_0"); - dataColumnHeadings.add("ATWORK_SOV_2"); - dataColumnHeadings.add("TOTAL_EMP"); - dataColumnHeadings.add("ATWORK_NM"); - dataColumnHeadings.add("ALL_HHS_TRANSIT"); - dataColumnHeadings.add("NONMAN_MAAS"); - dataColumnHeadings.add("MGRA"); - - // copy accessibilities array into a 0-based array - float[][] dataTableValues = new float[accessibilities.length - 1][]; - for (int r = 1; r < accessibilities.length; r++) - { - dataTableValues[r - 1] = new float[accessibilities[r].length]; - for (int c = 0; c < accessibilities[r].length; c++) - { - dataTableValues[r - 1][c] = accessibilities[r][c]; - } - } - - TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); - CSVFileWriter csv = new CSVFileWriter(); - try - { - csv.writeFile(accData, accFile); - } catch (IOException e) - { - logger.error("Error trying to write accessiblities data file " + accFileName); - throw new RuntimeException(e); - } - - } - - public void writeLandUseAccessibilityTableToFile(String luAccFileName, float[][] luAccessibility) - { - - File accFile = new File(luAccFileName); - - // the accessibilities array is indexed by mgra values which might no be - // consecutive. - // create an arraylist of data table rows, with the last field being the - // mgra value, - // convert to a tabledataset, then write to a csv file. - - ArrayList dataTableRows = new ArrayList(); - ArrayList dataColumnHeadings = new ArrayList(); - dataColumnHeadings.add("AM_WORK_1"); - dataColumnHeadings.add("AM_WORK_2"); - dataColumnHeadings.add("AM_WORK_3"); - dataColumnHeadings.add("AM_WORK_4"); - dataColumnHeadings.add("AM_WORK_5"); - dataColumnHeadings.add("AM_WORK_6"); - dataColumnHeadings.add("AM_SCHOOL_1"); - dataColumnHeadings.add("AM_SCHOOL_2"); - dataColumnHeadings.add("AM_SCHOOL_3"); - dataColumnHeadings.add("AM_SCHOOL_4"); - dataColumnHeadings.add("AM_SCHOOL_5"); - dataColumnHeadings.add("MD_NONMAN_LS0"); - dataColumnHeadings.add("MD_NONMAN_LS1"); - dataColumnHeadings.add("MD_NONMAN_LS2"); - dataColumnHeadings.add("LUZ"); - - for (int r = 0; r < luAccessibility.length; r++) - { - - if (luAccessibility[r] != null) - { - - float[] values = new float[luAccessibility[r].length]; - for (int c = 0; c < luAccessibility[r].length; c++) - values[c] = luAccessibility[r][c]; - - dataTableRows.add(values); - - } - - } - - float[][] dataTableValues = new float[dataTableRows.size()][]; - for (int r = 0; r < dataTableValues.length; r++) - dataTableValues[r] = dataTableRows.get(r); - - TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); - CSVFileWriter csv = new CSVFileWriter(); - try - { - csv.writeFile(accData, accFile); - } catch (IOException e) - { - logger.error("Error trying to write land use accessiblities data file " + luAccFileName); - throw new RuntimeException(e); - } - - } - - public void writeLandUseLogsumTablesToFile(String luLogsumFileName, double[][][][] luLogsums) - { - - File accFile = new File(luLogsumFileName); - - // the accessibilities array is indexed by mgra values which might no be - // consecutive. - // create an arraylist of data table rows, with the last field being the - // mgra value, - // convert to a tabledataset, then write to a csv file. - - ArrayList dataTableRows = new ArrayList(); - ArrayList dataColumnHeadings = new ArrayList(); - dataColumnHeadings.add("OrigLuz"); - dataColumnHeadings.add("DestLuz"); - dataColumnHeadings.add("AM_LS0"); - dataColumnHeadings.add("AM_LS1"); - dataColumnHeadings.add("AM_LS2"); - dataColumnHeadings.add("MD_LS0"); - dataColumnHeadings.add("MD_LS1"); - dataColumnHeadings.add("MD_LS2"); - - for (int l = 1; l <= BuildAccessibilities.MAX_LUZ; l++) - { - for (int m = 1; m <= BuildAccessibilities.MAX_LUZ; m++) - { - float[] values = new float[8]; - values[0] = l; - values[1] = m; - values[2] = (float) luLogsums[0][0][l][m]; - values[3] = (float) luLogsums[0][1][l][m]; - values[4] = (float) luLogsums[0][2][l][m]; - values[5] = (float) luLogsums[1][0][l][m]; - values[6] = (float) luLogsums[1][1][l][m]; - values[7] = (float) luLogsums[1][2][l][m]; - dataTableRows.add(values); - } - } - - float[][] dataTableValues = new float[dataTableRows.size()][]; - for (int r = 0; r < dataTableValues.length; r++) - dataTableValues[r] = dataTableRows.get(r); - - TableDataSet accData = TableDataSet.create(dataTableValues, dataColumnHeadings); - CSVFileWriter csv = new CSVFileWriter(); - try - { - csv.writeFile(accData, accFile); - } catch (IOException e) - { - logger.error("Error trying to write land use logsums data file " + luLogsumFileName); - throw new RuntimeException(e); - } - - } - - public float getAggregateAccessibility(String type, int homeMgra) - { - float returnValue = 0; - - if (type.equalsIgnoreCase("auto")) returnValue = accessibilities[homeMgra][NONMANDATORY_AUTO_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("transit")) returnValue = accessibilities[homeMgra][NONMANDATORY_TRANSIT_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("maas")) returnValue = accessibilities[homeMgra][NONMANDATORY_MAAS_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("nonmotor")) returnValue = accessibilities[homeMgra][NONMANDATORY_NONMOTOR_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("sov0")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_0_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("sov1")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_1_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("sov2")) returnValue = accessibilities[homeMgra][NONMANDATORY_SOV_2_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("hov0")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_0_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("hov1")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_1_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("hov2")) returnValue = accessibilities[homeMgra][NONMANDATORY_HOV_2_ACCESSIBILITY_FIELD_NUMBER - 1]; - else if (type.equalsIgnoreCase("shop0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shop1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shop2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maint0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maint1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maint2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("eatOut0")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("eatOut1")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("eatOut2")) returnValue = accessibilities[homeMgra][EAT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("visit0")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("visit1")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("visit2")) returnValue = accessibilities[homeMgra][VISIT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discr0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discr1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discr2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("escort0")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("escort1")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("escort2")) returnValue = accessibilities[homeMgra][ESCORT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("totEmp")) returnValue = accessibilities[homeMgra][TOTAL_EMPLOYMENT_ACCESSIBILITY_INDEX - 1]; - else if (type.equalsIgnoreCase("shopSov0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shopSov1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shopSov2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintSov0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintSov1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintSov2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrSov0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrSov1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrSov2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_SOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shopHov0")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shopHov1")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("shopHov2")) returnValue = accessibilities[homeMgra][SHOP_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintHov0")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintHov1")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("maintHov2")) returnValue = accessibilities[homeMgra][MAINT_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrHov0")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_INSUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrHov1")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_SUFFICIENT_INDEX - 1]; - else if (type.equalsIgnoreCase("discrHov2")) returnValue = accessibilities[homeMgra][DISCR_ACCESSIBILITY_HOV_OVERSUFFICIENT_INDEX - 1]; - else - { - logger.error("argument type = " - + type - + " is not valid"); - throw new RuntimeException(); - } - - return returnValue; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java deleted file mode 100644 index 78e29ee..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoAndNonMotorizedSkimsCalculator.java +++ /dev/null @@ -1,1024 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * This class is used to return auto skim values and non-motorized skim values - * for MGRA pairs associated with estimation data file records. - * - * @author Jim Hicks - * @version March, 2010 - */ -public class AutoAndNonMotorizedSkimsCalculator - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(AutoAndNonMotorizedSkimsCalculator.class); - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; - - // set the indices used for the non-motorized names array and the return - // skims - // array - private static final int WALK_INDEX = 0; - private static final int BIKE_INDEX = 1; - - private static final double WALK_SPEED = 3.0; // mph - private static final double BIKE_SPEED = 12.0; // mph - - // declare an array of UEC objects, 1 for each time period - private UtilityExpressionCalculator[] autoSkimUECs; - private IndexValues iv; - - // A simple DMU with no variables - private VariableTable dmu = null; - private AutoSkimsDMU autoSkimsDMU = null; - - private MgraDataManager mgraManager; - - private static final String[] AUTO_SKIM_NAMES = { - "da_nt_time", - "da_nt_fftime", - "da_nt_dist", - "da_nt_toll", - "da_nt_tdist", - "da_nt_std", - "da_tr_time", - "da_tr_fftime", - "da_tr_dist", - "da_tr_toll", - "da_tr_tdist", - "da_tr_std", - "s2_time", - "s2_fftime", - "s2_dist", - "s2_hdist", - "s2_toll", - "s2_tdist", - "s2_std", - "s3_time", - "s3_fftime", - "s3_dist", - "s3_hdist", - "s3_toll", - "s3_tdist", - "s3_std"}; - private static final int NUM_AUTO_SKIMS = AUTO_SKIM_NAMES.length; - - private static final String[] AUTO_SKIM_DESCRIPTIONS = { - "SOV Non-transponder Time", //0 - "SOV Non-transponder Free-flow Time", //1 - "SOV Non-transponder Distance", //2 - "SOV Non-transponder Toll", //3 - "SOV Non-transponder Toll Distance", //4 - "SOV Non-transponder Std. Deviation", //5 - "SOV Transponder Time", //6 - "SOV Transponder Free-flow Time", //7 - "SOV Transponder Distance", //8 - "SOV Transponder Toll", //9 - "SOV Transponder Toll Distance", //10 - "SOV Transponder Std. Deviation", //11 - "Shared 2 Time", //12 - "Shared 2 Free-flow Time", //13 - "Shared 2 Distance", //14 - "Shared 2 HOV Distance", //15 - "Shared 2 Toll", //16 - "Shared 2 Toll Distance", //17 - "Shared 2 Std. Deviation", //18 - "Shared 3+ Time", //19 - "Shared 3+ Free-flow Time", //20 - "Shared 3+ Distance", //21 - "Shared 3+ HOV Distance", //22 - "Shared 3+ Toll", //23 - "Shared 3+ Toll Distance", //24 - "Shared 3+ Std. Deviation" //25 - }; - - private static final String[] NM_SKIM_NAMES = {"walkTime", "bikeTime"}; - private static final int NUM_NM_SKIMS = NM_SKIM_NAMES.length; - - private static final String[] NM_SKIM_DESCRIPTIONS = {"walk time", "bike time"}; - - private double[][][] storedFromTazDistanceSkims; - private double[][][] storedToTazDistanceSkims; - - /** - * Get distance from taz to all zones. - * - * @param taz - * @param period - * @return An array of distances to all other zones. - */ - public double[] getTazDistanceFromTaz(int taz, int period) - { - - return storedFromTazDistanceSkims[period][taz]; - } - - /** - * Get distance from taz to all zones. - * - * @param taz - * @param period - * @return An array of distances to all other zones. - */ - public double[] getTazDistanceToTaz(int taz, int period) - { - - return storedToTazDistanceSkims[period][taz]; - } - - public AutoAndNonMotorizedSkimsCalculator(HashMap rbMap) - { - - // Create the UECs - String uecPath = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = uecPath - + Util.getStringValueFromPropertyMap(rbMap, "skims.auto.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.data.page"); - int autoSkimEaPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.ea.page"); - int autoSkimAmPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.am.page"); - int autoSkimMdPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.md.page"); - int autoSkimPmPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.pm.page"); - int autoSkimEvPage = Util.getIntegerValueFromPropertyMap(rbMap, "skims.auto.ev.page"); - - File uecFile = new File(uecFileName); - autoSkimsDMU = new AutoSkimsDMU(); - autoSkimUECs = new UtilityExpressionCalculator[NUM_PERIODS]; - autoSkimUECs[EA] = new UtilityExpressionCalculator(uecFile, autoSkimEaPage, dataPage, - rbMap, autoSkimsDMU); - autoSkimUECs[AM] = new UtilityExpressionCalculator(uecFile, autoSkimAmPage, dataPage, - rbMap, autoSkimsDMU); - autoSkimUECs[MD] = new UtilityExpressionCalculator(uecFile, autoSkimMdPage, dataPage, - rbMap, autoSkimsDMU); - autoSkimUECs[PM] = new UtilityExpressionCalculator(uecFile, autoSkimPmPage, dataPage, - rbMap, autoSkimsDMU); - autoSkimUECs[EV] = new UtilityExpressionCalculator(uecFile, autoSkimEvPage, dataPage, - rbMap, autoSkimsDMU); - iv = new IndexValues(); - - mgraManager = MgraDataManager.getInstance(); - - // distances = new double[mgraManager.getMaxMgra()+1]; - } - - public void setTazDistanceSkimArrays(double[][][] storedFromTazDistanceSkims, - double[][][] storedToTazDistanceSkims) - { - this.storedFromTazDistanceSkims = storedFromTazDistanceSkims; - this.storedToTazDistanceSkims = storedToTazDistanceSkims; - } - - /** - * Return the array of auto skims for the origin MGRA, destination MGRA, and - * departure time period. Used for appending skims data to estimation files, - * not part of main ABM. - * - * @param origMgra - * Origin MGRA - * @param workMgra - * Destination MGRA - * @param departPeriod - * Departure skim period index (currently 1-5) - * @param vot Value-of-time ($/hr) - * @return Array of 9 skim values for the MGRA pair and departure period - */ - public double[] getAutoSkims(int origMgra, int destMgra, int departPeriod, float vot, boolean debug, - Logger logger) - { - - String separator = ""; - String header = ""; - if (debug) - { - logger.info(""); - logger.info(""); - header = "get auto skims debug info for origMgra=" + origMgra + ", destMgra=" - + destMgra + ", period index=" + departPeriod + ", period label=" - + PERIODS[departPeriod]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - } - - // assign a helper UEC object to the array element for the desired - // departure - // time period - UtilityExpressionCalculator autoSkimUEC = autoSkimUECs[departPeriod]; - - // declare the array to hold the skim values, which will be returned - double[] autoSkims = null; - - if (origMgra > 0 && destMgra > 0) - { - - int oTaz = mgraManager.getTaz(origMgra); - int dTaz = mgraManager.getTaz(destMgra); - - iv.setOriginZone(oTaz); - iv.setDestZone(dTaz); - - autoSkimsDMU.setVOT(vot); - - // use the UEC to return skim values for the orign/destination TAZs - // associated with the MGRAs - autoSkims = autoSkimUEC.solve(iv, autoSkimsDMU, null); - if (debug) - autoSkimUEC.logAnswersArray(logger, String.format( - "autoSkimUEC: oMgra=%d, dMgra=%d, period=%d", origMgra, destMgra, - departPeriod)); - - } - - if (debug) - { - - logger.info(separator); - logger.info(header); - logger.info(separator); - - logger.info("auto skims array values"); - logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); - for (int i = 0; i < autoSkims.length; i++) - { - logger.info(String.format("%5d %40s %15.2f", i, AUTO_SKIM_DESCRIPTIONS[i], - autoSkims[i])); - } - - } - - return autoSkims; - - } - - //Wu modified for xborder trips; 9/27/2019 - public double[] getAutoSkimsByTAZ(int origTAZ, int destTAZ, int departPeriod, float vot, boolean debug, - Logger logger) - { - - String separator = ""; - String header = ""; - if (debug) - { - logger.info(""); - logger.info(""); - header = "get auto skims debug info for origTaz=" + origTAZ + ", destTaz=" - + destTAZ + ", period index=" + departPeriod + ", period label=" - + PERIODS[departPeriod]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - } - - // assign a helper UEC object to the array element for the desired - // departure - // time period - UtilityExpressionCalculator autoSkimUEC = autoSkimUECs[departPeriod]; - - // declare the array to hold the skim values, which will be returned - double[] autoSkims = null; - - if (origTAZ > 0 && destTAZ > 0) - { - - int oTaz = origTAZ; - int dTaz = destTAZ; - - iv.setOriginZone(oTaz); - iv.setDestZone(dTaz); - - autoSkimsDMU.setVOT(vot); - - // use the UEC to return skim values for the orign/destination TAZs - // associated with the MGRAs - autoSkims = autoSkimUEC.solve(iv, autoSkimsDMU, null); - if (debug) - autoSkimUEC.logAnswersArray(logger, String.format( - "autoSkimUEC: oTaz=%d, dTaz=%d, period=%d", origTAZ, destTAZ, - departPeriod)); - - } - - if (debug) - { - - logger.info(separator); - logger.info(header); - logger.info(separator); - - logger.info("auto skims array values"); - logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); - for (int i = 0; i < autoSkims.length; i++) - { - logger.info(String.format("%5d %40s %15.2f", i, AUTO_SKIM_DESCRIPTIONS[i], - autoSkims[i])); - } - - } - - return autoSkims; - - } - - /** - * Get the non-motorized skims. - * - * Get all the mgras within walking distance of the origin mgra. If the set - * of mgras is not null, and the destination mgra is in the set, get the - * walk and bike times from the mgraManager; - * - * If the destination mgra is not within walking distance of the origin - * MGRA, get the drive-alone non-toll off-peak distance skim value for the - * mgra pair and calculate the walk time and bike time. - * - * @param origMgra - * The origin mgra - * @param destMgra - * The destination mgra - * @return An array of distances - */ - public double[] getNonMotorizedSkims(int origMgra, int destMgra, int departPeriod, - boolean debug, Logger logger) - { - - String separator = ""; - String header = ""; - if (debug) - { - logger.info(""); - logger.info(""); - header = "get non-motorized skims debug info for origMgra=" + origMgra + ", destMgra=" - + destMgra; - for (int i = 0; i < header.length(); i++) - separator += "^"; - } - - double[] nmSkims = new double[NUM_NM_SKIMS]; - - // get the array of mgras within walking distance of the origin - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); - - // if one of the walk mgras is the destination, set the skim values and - // return - if (walkMgras != null) - { - - for (int wMgra : walkMgras) - { - - if (wMgra == destMgra) - { - nmSkims[WALK_INDEX] = mgraManager.getMgraToMgraWalkTime(origMgra, destMgra); - nmSkims[BIKE_INDEX] = mgraManager.getMgraToMgraBikeTime(origMgra, destMgra); - - if (debug) - { - - logger.info(separator); - logger.info(header); - logger.info(separator); - - logger.info("non-motorized skims array values"); - logger.info("determined from the mgraManager for an mgra pair within walking distance of each other."); - logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - logger.info(String.format("%5s %40s %15s", "-----", "----------", - "----------")); - for (int i = 0; i < nmSkims.length; i++) - { - logger.info(String.format("%5d %40s %15.2f", i, - NM_SKIM_DESCRIPTIONS[i], nmSkims[i])); - } - - } - - return nmSkims; - } - - } - - } - - // the destination was not within walk distance, so calculate walk and - // bike - // times from the TAZ-TAZ skim distance - int oTaz = mgraManager.getTaz(origMgra); - int dTaz = mgraManager.getTaz(destMgra); - - if (debug) - { - - logger.info(separator); - logger.info(header); - logger.info(separator); - - logger.info("non-motorized skims array values"); - logger.info("calculated for an mgra pair not within walking distance of each other."); - logger.info("origTaz = " + oTaz + ", destTaz = " + dTaz + ", period = " + departPeriod - + ", od distance = " - + (float) storedFromTazDistanceSkims[departPeriod][oTaz][dTaz]); - logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - logger.info(String.format("%5s %40s %15s", "-----", "----------", "----------")); - for (int i = 0; i < nmSkims.length; i++) - { - logger.info(String - .format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], nmSkims[i])); - } - - } - - nmSkims[WALK_INDEX] = (storedFromTazDistanceSkims[departPeriod][oTaz][dTaz] / WALK_SPEED) * 60.0; - nmSkims[BIKE_INDEX] = (storedFromTazDistanceSkims[departPeriod][oTaz][dTaz] / BIKE_SPEED) * 60.0; - - return nmSkims; - - } - - /* - * public double[] getNonMotorizedSkims(int origMgra, int destMgra, int - * departPeriod, boolean debug, Logger logger) { - * - * String separator = ""; String header = ""; if (debug) { logger.info(""); - * logger.info(""); header = - * "get non-motorized skims debug info for origMgra=" + origMgra + - * ", destMgra=" + destMgra; for (int i = 0; i < header.length(); i++) - * separator += "^"; } - * - * double[] nmSkims = new double[NUM_NM_SKIMS]; - * - * // get the array of mgras within walking distance of the origin int[] - * walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); - * - * // if one of the walk mgras is the destination, set the skim values and - * // return if (walkMgras != null) { - * - * for (int wMgra : walkMgras) { - * - * if (wMgra == destMgra) { nmSkims[WALK_INDEX] = - * mgraManager.getMgraToMgraWalkTime(origMgra, destMgra); - * nmSkims[BIKE_INDEX] = mgraManager.getMgraToMgraBikeTime(origMgra, - * destMgra); - * - * if (debug) { - * - * logger.info(separator); logger.info(header); logger.info(separator); - * - * logger.info("non-motorized skims array values"); logger .info( - * "determined from the mgraManager for an mgra pair within walking distance of each other." - * ); logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - * logger.info(String.format("%5s %40s %15s", "-----", "----------", - * "----------")); for (int i = 0; i < nmSkims.length; i++) { - * logger.info(String.format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], - * nmSkims[i])); } - * - * } - * - * return nmSkims; } - * - * } - * - * } - * - * // the destination was not within walk distance, so calculate walk and - * bike // times from the TAZ-TAZ skim distance int oTaz = - * mgraManager.getTaz(origMgra); int dTaz = mgraManager.getTaz(destMgra); - * - * iv.setOriginZone(oTaz); iv.setDestZone(dTaz); - * - * // get the DA NT OP distance value for the mgra pair double[] autoSkims = - * autoSkimUECs[OP].solve(iv, dmu, null); double distance = autoSkims[2]; - * - * nmSkims[WALK_INDEX] = (distance / WALK_SPEED) * 60.0; nmSkims[BIKE_INDEX] - * = (distance / BIKE_SPEED) * 60.0; - * - * if (debug) { - * - * logger.info(separator); logger.info(header); logger.info(separator); - * - * logger.info("non-motorized skims array values"); logger.info( - * "calculated for an mgra pair not within walking distance of each other." - * ); logger.info("origTaz = " + oTaz + ", destTaz = " + dTaz + - * ", od distance = " + (float) distance); - * logger.info(String.format("%5s %40s %15s", "i", "skimName", "value")); - * logger.info(String.format("%5s %40s %15s", "-----", "----------", - * "----------")); for (int i = 0; i < nmSkims.length; i++) { - * logger.info(String .format("%5d %40s %15.2f", i, NM_SKIM_DESCRIPTIONS[i], - * nmSkims[i])); } - * - * } - * - * return nmSkims; - * - * } - */ - /** - * Get all the mgras within walking distance of the origin mgra and set the - * distances to those mgras. - * - * Then loop through all mgras without a distance and get the drive-alone - * non-toll off-peak distance skim value for the taz pair associated with - * each mgra pair. - * - * @param origMgra - * The origin mgra - * @param An - * array in which to put the distances - * @param tourModeIsAuto - * is a boolean set to true if tour mode is not non-motorized, - * transit, or school bus. if auto tour mode, then no need to - * determine walk distance, and drive skims can be used directly. - * public void getDistancesFromMgra( int origMgra, double[] - * distances, boolean tourModeIsAuto ) { - * - * Arrays.fill(distances, 0); - * - * if ( ! tourModeIsAuto ){ - * - * // get the array of mgras within walking distance of the - * destination int[] walkMgras = - * mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); - * - * // set the distance values for the mgras walkable to the - * destination if (walkMgras != null) { - * - * // get distances, in feet, and convert to miles for (int wMgra - * : walkMgras) distances[wMgra] = - * mgraManager.getMgraToMgraWalkDistFrom(origMgra, wMgra) / - * 5280.0; - * - * } - * - * } - * - * - * int oTaz = mgraManager.getTaz(origMgra); - * iv.setOriginZone(oTaz); for (int wMgra=1; wMgra <= - * mgraManager.getMaxMgra(); wMgra++) { - * - * // skip mgras where distance has already been set if ( - * distances[wMgra] > 0 ) continue; - * - * // calculate distances from the TAZ-TAZ skim distance int dTaz - * = mgraManager.getTaz(wMgra); iv.setDestZone(dTaz); double[] - * autoSkims = autoSkimUECs[OP].solve(iv, dmu, null); - * - * distances[wMgra] = autoSkims[2]; } - * - * } - */ - - /** - * Get all the mgras within walking distance of the origin mgra and set the - * distances to those mgras. - * - * Then loop through all mgras without a distance and get the drive-alone - * non-toll off-peak distance skim value for the taz pair associated with - * each mgra pair. - * - * @param origMgra - * The origin mgra - * @param An - * array in which to put the distances - * @param tourModeIsAuto - * is a boolean set to true if tour mode is not non-motorized, - * transit, or school bus. if auto tour mode, then no need to - * determine walk distance, and drive skims can be used directly. - * - */ - - public void getDistancesFromMgra(int origMgra, double[] distances, boolean tourModeIsAuto) - { - - Arrays.fill(distances, 0); - - if (!tourModeIsAuto) - { - - // get the array of mgras within walking distance of the destination - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(origMgra); - - // set the distance values for the mgras walkable to the destination - if (walkMgras != null) - { - - // get distances, in feet, and convert to miles - for (int wMgra : walkMgras) - distances[wMgra] = mgraManager.getMgraToMgraWalkDistFrom(origMgra, wMgra) / 5280.0; - - } - - } - - int oTaz = mgraManager.getTaz(origMgra); - - for (int wMgra = 1; wMgra <= mgraManager.getMaxMgra(); wMgra++) - { - - // skip mgras where distance has already been set - if (distances[wMgra] > 0) continue; - - int dTaz = mgraManager.getTaz(wMgra); - distances[wMgra] = storedFromTazDistanceSkims[MD][oTaz][dTaz]; - } - - } - - /** - * Get all the mgras within walking distance of the destination mgra and set - * the distances from those mgras. - * - * Then loop through all mgras without a distance and get the drive-alone - * non-toll off-peak distance skim value for the taz pair associated with - * each mgra pair. - * - * @param destMgra - * The destination mgra - * @param An - * array in which to put the distances - * @param tourModeIsAuto - * is a boolean set to true if tour mode is not non-motorized, - * transit, or school bus. if auto tour mode, then no need to - * determine walk distance, and drive skims can be used directly. - */ - - public void getDistancesToMgra(int destMgra, double[] distances, boolean tourModeIsAuto) - { - - Arrays.fill(distances, 0); - - if (!tourModeIsAuto) - { - - // get the array of mgras within walking distance of the destination - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); - - // set the distance values for the mgras walkable to the destination - if (walkMgras != null) - { - - // get distances, in feet, and convert to miles - // get distances from destMgra since this is the direction of - // distances read from the data file - for (int wMgra : walkMgras) - distances[wMgra] = mgraManager.getMgraToMgraWalkDistTo(wMgra, destMgra) / 5280.0; - - } - - } - - // if the TAZ distances have not been computed yet for this destination - // TAZ, compute them from the UEC. - int dTaz = mgraManager.getTaz(destMgra); - - for (int wMgra = 1; wMgra <= mgraManager.getMaxMgra(); wMgra++) - { - - // skip mgras where distance has already been set - if (distances[wMgra] > 0) continue; - - int oTaz = mgraManager.getTaz(wMgra); - distances[wMgra] = storedToTazDistanceSkims[MD][dTaz][oTaz]; - - } - - } - - /* - * public void getDistancesToMgra( int destMgra, double[] distances, boolean - * tourModeIsAuto ) { - * - * Arrays.fill(distances, 0); - * - * if ( ! tourModeIsAuto ){ - * - * // get the array of mgras within walking distance of the destination - * int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); - * - * // set the distance values for the mgras walkable to the destination if - * (walkMgras != null) { - * - * // get distances, in feet, and convert to miles // get distances from - * destMgra since this is the direction of distances read from the data file - * for (int wMgra : walkMgras) distances[wMgra] = - * mgraManager.getMgraToMgraWalkDistTo(destMgra, wMgra) / 5280.0; - * - * } - * - * } - * - * - * int dTaz = mgraManager.getTaz(destMgra); iv.setDestZone(dTaz); for (int - * wMgra=1; wMgra <= mgraManager.getMaxMgra(); wMgra++) { - * - * // skip mgras where distance has already been set if ( distances[wMgra] > - * 0 ) continue; - * - * // calculate distances from the TAZ-TAZ skim distance int oTaz = - * mgraManager.getTaz(wMgra); iv.setOriginZone(oTaz); double[] autoSkims = - * autoSkimUECs[OP].solve(iv, dmu, null); - * - * distances[wMgra] = autoSkims[2]; } - * - * } - */ - - /* - * public double[] getDistancesToMgra( int destMgra, boolean tourModeIsAuto - * ) { - * - * - * Arrays.fill(distances, 0); - * - * if ( ! tourModeIsAuto ){ - * - * // get the array of mgras within walking distance of the destination - * int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(destMgra); - * - * // set the distance values for the mgras walkable to the destination if - * (walkMgras != null) { - * - * // get distances, in feet, and convert to miles // get distances from - * destMgra since this is the direction of distances read from the data file - * for (int wMgra : walkMgras) distances[wMgra] = - * mgraManager.getMgraToMgraWalkDistTo(destMgra, wMgra) / 5280.0; - * - * } - * - * } - * - * - * - * // if the TAZ distances have not been computed yet for this destination - * TAZ, compute them from the UEC. int dTaz = mgraManager.getTaz(destMgra); - * - * for (int wMgra=1; wMgra <= mgraManager.getMaxMgra(); wMgra++) { - * - * // skip mgras where distance has already been set if ( distances[wMgra] > - * 0 ) continue; - * - * int oTaz = mgraManager.getTaz(wMgra); distances[wMgra] = - * storedToTazDistanceSkims[OP][dTaz][oTaz]; - * - * } - * - * return Arrays.copyOf( distances, distances.length ); - * - * } - * - * - * /** Calculate utility expressions for auto skims for a given origin to - * get distances to all destination mgras, and return off-peak sov distance. - * - * @param oMgra The origin mgra - * - * @return An array of off-peak sov distances - */ - /* - * public void getOpSkimDistancesFromMgra( int oMgra, double[] distances ) { - * - * int oTaz = mgraManager.getTaz(oMgra); - * - * for (int i=1; i <= mgraManager.getMaxMgra(); i++) { - * - * // calculate distances from the TAZ-TAZ skim distance int dTaz = - * mgraManager.getTaz(i); distances[i] = - * storedFromTazDistanceSkims[OP][oTaz][dTaz]; - * - * } - * - * } - */ - - /* - * public double[] getAmPkSkimDistancesFromMgra( int oMgra ) { - * - * int oTaz = mgraManager.getTaz(oMgra); - * - * for (int i=1; i <= mgraManager.getMaxMgra(); i++) { - * - * // calculate distances from the TAZ-TAZ skim distance int dTaz = - * mgraManager.getTaz(i); distances[i] = - * storedFromTazDistanceSkims[AM][oTaz][dTaz]; - * - * } - * - * return Arrays.copyOf( distances, distances.length ); - * - * } - */ - - /* - * public double[] getOpSkimDistancesFromMgra(int oMgra) { - * - * double[] distances = new double[mgraManager.getMaxMgra() + 1]; - * - * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); - * - * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { - * - * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); - * - * // sov distance double[] autoResults = autoSkimUECs[OP].solve(iv, dmu, - * null); distances[i] = autoResults[2]; - * - * } - * - * return distances; } - */ - - /* - * public void getOpSkimDistancesFromMgra(int oMgra, double[] distances) { - * - * Arrays.fill( distances, 0 ); - * - * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); - * - * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { - * - * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); - * - * // sov distance double[] autoResults = autoSkimUECs[OP].solve(iv, dmu, - * null); distances[i] = autoResults[2]; - * - * } - * - * } - */ - - public void getOpSkimDistancesFromMgra(int oMgra, double[] distances) - { - - int oTaz = mgraManager.getTaz(oMgra); - - for (int i = 1; i <= mgraManager.getMaxMgra(); i++) - { - - // calculate distances from the TAZ-TAZ skim distance - int dTaz = mgraManager.getTaz(i); - distances[i] = storedFromTazDistanceSkims[MD][oTaz][dTaz]; - - } - - } - - /** - * Calculate utility expressions for auto skims for a given origin to get - * distances to all destination mgras, and return am peak sov distance. - * - * @param oMgra - * The origin mgra - * @return An array of am peak sov distances public double[] - * getAmPkSkimDistancesFromMgra(int oMgra) { - * - * double[] distances = new double[mgraManager.getMaxMgra() + 1]; - * - * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); - * - * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { - * - * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); - * - * // sov distance double[] autoResults = autoSkimUECs[AM].solve(iv, - * dmu, null); distances[i] = autoResults[2]; - * - * } - * - * return distances; } - */ - - public void getAmPkSkimDistancesFromMgra(int oMgra, double[] distances) - { - - int oTaz = mgraManager.getTaz(oMgra); - - for (int i = 1; i <= mgraManager.getMaxMgra(); i++) - { - - // calculate distances from the TAZ-TAZ skim distance - int dTaz = mgraManager.getTaz(i); - distances[i] = storedFromTazDistanceSkims[AM][oTaz][dTaz]; - - } - - } - - /* - * public void getAmPkSkimDistancesFromMgra(int oMgra, double[] distances) { - * - * Arrays.fill( distances, 0 ); - * - * int oTaz = mgraManager.getTaz(oMgra); iv.setOriginZone(oTaz); - * - * for (int i = 1; i <= mgraManager.getMaxMgra(); i++) { - * - * int dTaz = mgraManager.getTaz(i); iv.setDestZone(dTaz); - * - * // sov distance double[] autoResults = autoSkimUECs[AM].solve(iv, dmu, - * null); distances[i] = autoResults[2]; - * - * } - * - * } - */ - - /** - * log a report of the final skim values for the MGRA odt - * - * @param orig - * is the origin mgra for the segment - * @param dest - * is the destination mgra for the segment - * @param depart - * is the departure period for the segment - * @param bestTapPairs - * is an int[][] of TAP values with the first dimesion the ride - * mode and second dimension a 2 element array with best orig and - * dest TAP - * @param returnedSkims - * is a double[][] of skim values with the first dimesion the - * ride mode indices and second dimention the skim categories - */ - public void logReturnedSkims(int orig, int dest, int depart, double[] skims, String skimLabel, - Logger logger) - { - - String separator = ""; - String header = ""; - - logger.info(""); - logger.info(""); - header = skimLabel + " skim value tables for origMgra=" + orig + ", destMgra=" + dest - + ", departperiod=" + depart; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - logger.info(separator); - logger.info(header); - logger.info(""); - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - tableRecord = String.format("%-5d %12.5f ", i + 1, skims[i]); - logger.info(tableRecord); - } - - logger.info(""); - logger.info(separator); - } - - public int getNumSkimPeriods() - { - return NUM_PERIODS; - } - - public int getNumAutoSkims() - { - return NUM_AUTO_SKIMS; - } - - public String[] getAutoSkimNames() - { - return AUTO_SKIM_NAMES; - } - - public int getNumNmSkims() - { - return NUM_NM_SKIMS; - } - - public String[] getNmSkimNames() - { - return NM_SKIM_NAMES; - } - - public int getNmWalkTimeSkimIndex() - { - return WALK_INDEX; - } - - public int getNmBikeTimeSkimIndex() - { - return BIKE_INDEX; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java deleted file mode 100644 index 8467036..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoSkimsDMU.java +++ /dev/null @@ -1,82 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -public class AutoSkimsDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(AutoSkimsDMU.class); - - protected HashMap methodIndexMap; - - protected float vot; - - public AutoSkimsDMU() - { - setupMethodIndexMap(); - } - - public float getVOT() - { - return vot; - } - - public void setVOT(float vot) - { - this.vot = vot; - } - - - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getVOT", 0); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getVOT(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java deleted file mode 100644 index 9e0811d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/AutoTazSkimsCalculator.java +++ /dev/null @@ -1,206 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.Serializable; -import java.util.HashMap; - -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * This class is used to return auto skim values and non-motorized skim values - * for MGRA pairs associated with estimation data file records. - * - * @author Jim Hicks - * @version March, 2010 - */ -public class AutoTazSkimsCalculator - implements Serializable -{ - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - - // declare an array of UEC objects, 1 for each time period - private UtilityExpressionCalculator[] autoDistOD_UECs; - - // The simple auto skims UEC does not use any DMU variables - private VariableTable dmu = null; - - private TazDataManager tazManager; - - private double[][][] storedFromTazDistanceSkims; - private double[][][] storedToTazDistanceSkims; - private int maxTaz; - - public AutoTazSkimsCalculator(HashMap rbMap) - { - - // Create the UECs - String uecPath = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = uecPath - + Util.getStringValueFromPropertyMap(rbMap, "taz.distance.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "taz.distance.data.page"); - int autoSkimEaOdPage = Util - .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.ea.page"); - int autoSkimAmOdPage = Util - .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.am.page"); - int autoSkimMdOdPage = Util - .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.md.page"); - int autoSkimPmOdPage = Util - .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.pm.page"); - int autoSkimEvOdPage = Util - .getIntegerValueFromPropertyMap(rbMap, "taz.od.distance.ev.page"); - - File uecFile = new File(uecFileName); - autoDistOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; - autoDistOD_UECs[EA] = new UtilityExpressionCalculator(uecFile, autoSkimEaOdPage, dataPage, - rbMap, dmu); - autoDistOD_UECs[AM] = new UtilityExpressionCalculator(uecFile, autoSkimAmOdPage, dataPage, - rbMap, dmu); - autoDistOD_UECs[MD] = new UtilityExpressionCalculator(uecFile, autoSkimMdOdPage, dataPage, - rbMap, dmu); - autoDistOD_UECs[PM] = new UtilityExpressionCalculator(uecFile, autoSkimPmOdPage, dataPage, - rbMap, dmu); - autoDistOD_UECs[EV] = new UtilityExpressionCalculator(uecFile, autoSkimEvOdPage, dataPage, - rbMap, dmu); - - tazManager = TazDataManager.getInstance(); - maxTaz = tazManager.getMaxTaz(); - - storedFromTazDistanceSkims = new double[NUM_PERIODS + 1][maxTaz + 1][]; - storedToTazDistanceSkims = new double[NUM_PERIODS + 1][maxTaz + 1][]; - - } - - /** - * Get all the mgras within walking distance of the origin mgra and set the - * distances to those mgras. - * - * Then loop through all mgras without a distance and get the drive-alone - * non-toll off-peak distance skim value for the taz pair associated with - * each mgra pair. - * - * @param origMgra - * The origin mgra - * @param An - * array in which to put the distances - * @param tourModeIsAuto - * is a boolean set to true if tour mode is not non-motorized, - * transit, or school bus. if auto tour mode, then no need to - * determine walk distance, and drive skims can be used directly. - */ - public void computeTazDistanceArrays() - { - - IndexValues iv = new IndexValues(); - - for (int oTaz = 1; oTaz <= maxTaz; oTaz++) - { - - storedFromTazDistanceSkims[EA][oTaz] = new double[maxTaz + 1]; - storedToTazDistanceSkims[EA][oTaz] = new double[maxTaz + 1]; - storedFromTazDistanceSkims[AM][oTaz] = new double[maxTaz + 1]; - storedToTazDistanceSkims[AM][oTaz] = new double[maxTaz + 1]; - storedFromTazDistanceSkims[MD][oTaz] = new double[maxTaz + 1]; - storedToTazDistanceSkims[MD][oTaz] = new double[maxTaz + 1]; - storedFromTazDistanceSkims[PM][oTaz] = new double[maxTaz + 1]; - storedToTazDistanceSkims[PM][oTaz] = new double[maxTaz + 1]; - storedFromTazDistanceSkims[EV][oTaz] = new double[maxTaz + 1]; - storedToTazDistanceSkims[EV][oTaz] = new double[maxTaz + 1]; - - } - - for (int oTaz = 1; oTaz <= maxTaz; oTaz++) - { - - iv.setOriginZone(oTaz); - - double[] eaAutoDist = autoDistOD_UECs[EA].solve(iv, dmu, null); - double[] amAutoDist = autoDistOD_UECs[AM].solve(iv, dmu, null); - double[] mdAutoDist = autoDistOD_UECs[MD].solve(iv, dmu, null); - double[] pmAutoDist = autoDistOD_UECs[PM].solve(iv, dmu, null); - double[] evAutoDist = autoDistOD_UECs[EV].solve(iv, dmu, null); - - for (int d = 0; d < maxTaz; d++) - { - - storedFromTazDistanceSkims[EA][oTaz][d + 1] = eaAutoDist[d]; - storedFromTazDistanceSkims[AM][oTaz][d + 1] = amAutoDist[d]; - storedFromTazDistanceSkims[MD][oTaz][d + 1] = mdAutoDist[d]; - storedFromTazDistanceSkims[PM][oTaz][d + 1] = pmAutoDist[d]; - storedFromTazDistanceSkims[EV][oTaz][d + 1] = evAutoDist[d]; - - storedToTazDistanceSkims[EA][d + 1][oTaz] = eaAutoDist[d]; - storedToTazDistanceSkims[AM][d + 1][oTaz] = amAutoDist[d]; - storedToTazDistanceSkims[MD][d + 1][oTaz] = mdAutoDist[d]; - storedToTazDistanceSkims[PM][d + 1][oTaz] = pmAutoDist[d]; - storedToTazDistanceSkims[EV][d + 1][oTaz] = evAutoDist[d]; - - } - - // iv.setDestZone( oTaz ); - // - // amAutoDist = autoDistDO_UECs[AM].solve(iv, dmu, null); - // opAutoDist = autoDistDO_UECs[OP].solve(iv, dmu, null); - // - // for (int d=0; d < maxTaz; d++) - // { - // - // storedToTazDistanceSkims[AM][d+1][oTaz] = amAutoDist[d]; - // storedToTazDistanceSkims[OP][d+1][oTaz] = opAutoDist[d]; - // - // } - - } - - } - - public double getTazToTazDistance(int period, int oTaz, int dTaz){ - - return storedFromTazDistanceSkims[period][oTaz][dTaz]; - } - - public double[][][] getStoredFromTazToAllTazsDistanceSkims() - { - return storedFromTazDistanceSkims; - } - - public double[][][] getStoredToTazFromAllTazsDistanceSkims() - { - return storedToTazDistanceSkims; - } - - public void clearStoredTazsDistanceSkims() - { - - for (int i = 0; i < storedFromTazDistanceSkims.length; i++) - { - for (int j = 0; j < storedFromTazDistanceSkims[i].length; j++) - storedFromTazDistanceSkims[i][j] = null; - storedFromTazDistanceSkims[i] = null; - } - storedFromTazDistanceSkims = null; - - for (int i = 0; i < storedToTazDistanceSkims.length; i++) - { - for (int j = 0; j < storedToTazDistanceSkims[i].length; j++) - storedToTazDistanceSkims[i][j] = null; - storedToTazDistanceSkims[i] = null; - } - storedToTazDistanceSkims = null; - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java deleted file mode 100644 index a31110a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BestTransitPathCalculator.java +++ /dev/null @@ -1,1548 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.PrintWriter; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.concurrent.ConcurrentHashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.util.Tracer; - -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.Modes.AccessMode; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; -import org.sandag.abm.reporting.OMXMatrixDao; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.matrix.Matrix; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.newmodel.Alternative; -import com.pb.common.newmodel.ConcreteAlternative; -import com.pb.common.newmodel.LogitModel; -/** - * WalkPathUEC calculates the best walk-transit utilities for a given MGRA pair. - * - * @author Joel Freedman - * @version 1.0, May 2009 - */ -public class BestTransitPathCalculator implements Serializable -{ - - private transient Logger logger = Logger.getLogger(BestTransitPathCalculator.class); - - //TODO: combine APP_TYPE_xxx constants into a enum structure - public static final int APP_TYPE_GENERIC = 0; - public static final int APP_TYPE_TOURMC = 1; - public static final int APP_TYPE_TRIPMC = 2; - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - - public static final float NA = -999; - public static final int WTW = 0; - public static final int WTD = 1; - public static final int DTW = 2; - public static final int[] ACC_EGR = {WTW,WTD,DTW}; - public static final int NUM_ACC_EGR = ACC_EGR.length; - public static final String[] ACC_EGR_STRING = {"WTW","WTD","DTW"}; - - // seek and trace - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - protected Tracer tracer; - - private TazDataManager tazManager; - private TapDataManager tapManager; - private MgraDataManager mgraManager; - - private int maxMgra; - private int maxTap; - private int maxTaz; - - // piece-wise utilities are being computed - private UtilityExpressionCalculator walkAccessUEC; - private UtilityExpressionCalculator walkEgressUEC; - private UtilityExpressionCalculator driveAccessUEC; - private UtilityExpressionCalculator driveEgressUEC; - private UtilityExpressionCalculator tapToTapUEC; - private UtilityExpressionCalculator driveAccDisutilityUEC; - private UtilityExpressionCalculator driveEgrDisutilityUEC; - - private static final String TAPS_SKIM = "taps.skim"; - private static final String TAPS_SKIM_DIST = "taps.skim.dist"; - - - // utility data cache for each transit path segment - private StoredUtilityData storedDataObject; //Encapsulates data shared by the BestTransitPathCalculator objects created for each hh choice model object - // note that access/egress utilities are independent of transit skim set - private float[][] storedWalkAccessUtils; // references StoredUtilityData.storedWalkAccessUtils - private float[][] storedDriveAccessUtils;// references StoredUtilityData.storedDriveAccessUtils - private float[][] storedWalkEgressUtils; // references StoredUtilityData.storedWalkEgressUtils - private float[][] storedDriveEgressUtils;// references StoredUtilityData.storedDriveEgressUtils - private HashMap>> storedDepartPeriodTapTapUtils; //references StoredUtilityData.storedDepartPeriodTapTapUtils - - private IndexValues index = new IndexValues(); - - // arrays storing information about the n (array length) best paths - private double[] bestUtilities; - private double[] bestAccessUtilities; - private double[] bestEgressUtilities; - private int[] bestPTap; - private int[] bestATap; - private int[] bestSet; //since two of the best paths can be in the same set, need to store set as well - - private int numSkimSets; - private int numTransitAlts; - private int[] maxLogsumUtilitiesBySkimSet; //maximum number of utilities for each skims set in logsum calcs - private int[] utilityCount; //counter for utilities - private double[] expUtilities; //exponentiated utility array for path choice - - private double nestingCoefficient; - private static double WORST_UTILITY = -500; - - /** - * Constructor. - * - * @param rbMap HashMap - * @param UECFileName The path/name of the UEC containing the walk-transit model. - * @param modelSheet The sheet (0-indexed) containing the model specification. - * @param dataSheet The sheet (0-indexed) containing the data specification. - */ - public BestTransitPathCalculator(HashMap rbMap) - { - - // read in resource bundle properties - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if ( trace ) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - - - String uecPath = Util.getStringValueFromPropertyMap(rbMap,CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = Paths.get(uecPath,rbMap.get("utility.bestTransitPath.uec.file")).toString(); - - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, - "utility.bestTransitPath.data.page"); - - int walkAccessPage = Util.getIntegerValueFromPropertyMap(rbMap, - "utility.bestTransitPath.walkAccess.page"); - int driveAccessPage = Util.getIntegerValueFromPropertyMap(rbMap, - "utility.bestTransitPath.driveAccess.page"); - int walkEgressPage = Util.getIntegerValueFromPropertyMap(rbMap, - "utility.bestTransitPath.walkEgress.page"); - int driveEgressPage = Util.getIntegerValueFromPropertyMap(rbMap, - "utility.bestTransitPath.driveEgress.page"); - int tapToTapPage = Util.getIntegerValueFromPropertyMap( rbMap, - "utility.bestTransitPath.tapToTap.page" ); - int driveAccDisutilityPage = Util.getIntegerValueFromPropertyMap( rbMap, - "utility.bestTransitPath.driveAccDisutility.page" ); - int driveEgrDisutilityPage = Util.getIntegerValueFromPropertyMap( rbMap, - "utility.bestTransitPath.driveEgrDisutility.page" ); - - - File uecFile = new File(uecFileName); - walkAccessUEC = createUEC(uecFile, walkAccessPage, dataPage, rbMap, new TransitWalkAccessDMU()); - driveAccessUEC = createUEC(uecFile, driveAccessPage, dataPage, rbMap, new TransitDriveAccessDMU()); - walkEgressUEC = createUEC(uecFile, walkEgressPage, dataPage, rbMap, new TransitWalkAccessDMU()); - driveEgressUEC = createUEC(uecFile, driveEgressPage, dataPage, rbMap, new TransitDriveAccessDMU()); - tapToTapUEC = createUEC(uecFile, tapToTapPage, dataPage, rbMap, new TransitWalkAccessDMU()); - driveAccDisutilityUEC = createUEC(uecFile, driveAccDisutilityPage, dataPage, rbMap, new TransitDriveAccessDMU()); - driveEgrDisutilityUEC = createUEC(uecFile, driveEgrDisutilityPage, dataPage, rbMap, new TransitDriveAccessDMU()); - - mgraManager = MgraDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - maxMgra = mgraManager.getMaxMgra(); - maxTap = mgraManager.getMaxTap(); - maxTaz = tazManager.getMaxTaz(); - - // these arrays are shared by the BestTransitPathCalculator objects created for each hh choice model object - storedDataObject = StoredUtilityData.getInstance( maxMgra, maxTap, maxTaz, ACC_EGR, ModelStructure.PERIODCODES); - storedWalkAccessUtils = storedDataObject.getStoredWalkAccessUtils(); - storedDriveAccessUtils = storedDataObject.getStoredDriveAccessUtils(); - storedWalkEgressUtils = storedDataObject.getStoredWalkEgressUtils(); - storedDriveEgressUtils = storedDataObject.getStoredDriveEgressUtils(); - storedDepartPeriodTapTapUtils = storedDataObject.getStoredDepartPeriodTapTapUtils(); - - //setup arrays - numSkimSets = Util.getIntegerValueFromPropertyMap( rbMap, "utility.bestTransitPath.skim.sets" ); - numTransitAlts = Util.getIntegerValueFromPropertyMap( rbMap, "utility.bestTransitPath.alts" ); - - bestUtilities = new double[numTransitAlts]; - bestPTap = new int[numTransitAlts]; - bestATap = new int[numTransitAlts]; - bestSet = new int[numTransitAlts]; - maxLogsumUtilitiesBySkimSet = Util.getIntegerArrayFromPropertyMap(rbMap, "utility.bestTransitPath.maxPathsPerSkimSetForLogsum"); - utilityCount = new int[numSkimSets]; - expUtilities = new double[numTransitAlts]; - - nestingCoefficient = new Double(Util.getStringValueFromPropertyMap(rbMap, "utility.bestTransitPath.nesting.coeff")).floatValue(); - - } - - - - /** - * This is the main method that finds the best N TAP-pairs. It - * cycles through walk TAPs at the origin end (associated with the origin MGRA) - * and alighting TAPs at the destination end (associated with the destination - * MGRA) and calculates a utility for every available alt for each TAP - * pair. It stores the N origin and destination TAP that had the best utility. - * - * @param pMgra The origin/production MGRA. - * @param aMgra The destination/attraction MGRA. - * - */ - public void findBestWalkTransitWalkTaps(TransitWalkAccessDMU walkDmu, int period, int pMgra, int aMgra, boolean debug, Logger myLogger) - { - - clearBestArrays(Double.NEGATIVE_INFINITY); - - int[] pMgraSet = mgraManager.getMgraWlkTapsDistArray()[pMgra][0]; - int[] aMgraSet = mgraManager.getMgraWlkTapsDistArray()[aMgra][0]; - - if (pMgraSet == null || aMgraSet == null) - { - return; - } - - int pTaz = mgraManager.getTaz(pMgra); - int aTaz = mgraManager.getTaz(aMgra); - - boolean writeCalculations = false; - if ((tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz))|| debug) - { - writeCalculations = true; - } - - //create transit path collection - ArrayList paths = new ArrayList(); - - for (int pTap : pMgraSet) - { - - // Calculate the pMgra to pTap walk access utility values - float accUtil; - if (storedWalkAccessUtils[pMgra][pTap] == StoredUtilityData.default_utility) { - accUtil = calcWalkAccessUtility(walkDmu, pMgra, pTap, writeCalculations, myLogger); - storedWalkAccessUtils[pMgra][pTap] = accUtil; - } else { - accUtil = storedWalkAccessUtils[pMgra][pTap]; - if(writeCalculations){ - myLogger.info("Stored walk access utility from Mgra "+pMgra+" to Tap "+pTap+" is "+accUtil); - } - } - - for (int aTap : aMgraSet) - { - - // Calculate the aTap to aMgra walk egress utility values - float egrUtil; - if (storedWalkEgressUtils[aTap][aMgra] == StoredUtilityData.default_utility) { - egrUtil = calcWalkEgressUtility(walkDmu, aTap, aMgra, writeCalculations, myLogger); - storedWalkEgressUtils[aTap][aMgra] = egrUtil; - } else { - egrUtil = storedWalkEgressUtils[aTap][aMgra]; - if(writeCalculations){ - myLogger.info("Stored walk egress utility from Tap "+aTap+" to Mgra "+aMgra+" is "+egrUtil); - } - } - - // Calculate the pTap to aTap utility values - float tapTapUtil[] = new float[numSkimSets]; - if(!storedDepartPeriodTapTapUtils.get(WTW).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { - - //loop across number of skim sets the pTap to aTap utility values - for (int set=0; set paths = new ArrayList(); - - float[][][] tapParkingInfo = tapManager.getTapParkingInfo(); - - int[] pTapArray = tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode); - for ( int pTap : pTapArray ) - { - // Calculate the pTaz to pTap drive access utility values - float accUtil; - float accDisutil; - if (storedDriveAccessUtils[pTaz][pTap] == StoredUtilityData.default_utility) { - accUtil = calcDriveAccessUtility(driveDmu, pMgra, pTaz, pTap, accMode, writeCalculations, myLogger); - storedDriveAccessUtils[pTaz][pTap] = accUtil; - } else { - accUtil = storedDriveAccessUtils[pTaz][pTap]; - if(writeCalculations) - myLogger.info("Stored drive access utility from TAZ "+pTaz+" to Tap "+pTap+" is "+accUtil); - } - - - int lotID = (int)tapParkingInfo[pTap][0][0]; // lot ID - float lotCapacity = tapParkingInfo[pTap][2][0]; // lot capacity - - if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) - || (accMode == AccessMode.KISS_N_RIDE)) - { - - //always calculate the access disutility since it changes based on od - accDisutil = calcDriveAccessRatioDisutility(driveDmu, pMgra, pTaz, pTap, odDistance, accMode, writeCalculations, myLogger); - - for (int aTap : mgraManager.getMgraWlkTapsDistArray()[aMgra][0]) - { - - // Calculate the aTap to aMgra walk egress utility values - float egrUtil; - if (storedWalkEgressUtils[aTap][aMgra] == StoredUtilityData.default_utility) { - egrUtil = calcWalkEgressUtility(walkDmu, aTap, aMgra, writeCalculations, myLogger); - storedWalkEgressUtils[aTap][aMgra] = egrUtil; - } else { - egrUtil = storedWalkEgressUtils[aTap][aMgra]; - if(writeCalculations){ - myLogger.info("Stored walk egress utility from Tap "+aTap+" to Mgra "+aMgra+" is "+egrUtil); - } - } - // Calculate the pTap to aTap utility values - float tapTapUtil[] = new float[numSkimSets]; - if(!storedDepartPeriodTapTapUtils.get(DTW).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { - - //loop across number of skim sets the pTap to aTap utility values - for (int set=0; set paths = new ArrayList(); - - for (int pTap : mgraManager.getMgraWlkTapsDistArray()[pMgra][0]) - { - // Calculate the pMgra to pTap walk access utility values - float accUtil; - if (storedWalkAccessUtils[pMgra][pTap] == StoredUtilityData.default_utility) { - accUtil = calcWalkAccessUtility(walkDmu, pMgra, pTap, writeCalculations, myLogger); - storedWalkAccessUtils[pMgra][pTap] = accUtil; - } else { - accUtil = storedWalkAccessUtils[pMgra][pTap]; - if(writeCalculations){ - myLogger.info("Stored walk access utility from Mgra "+pMgra+" to Tap "+pTap+" is "+accUtil); - } - } - for (int aTap : tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode)) - { - - int lotID = (int) tapManager.getTapParkingInfo()[aTap][0][0]; // lot - // ID - float lotCapacity = tapManager.getTapParkingInfo()[aTap][2][0]; // lot - // capacity - if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) - || (accMode == AccessMode.KISS_N_RIDE)) - { - - // Calculate the aTap to aMgra drive egress utility values - float egrUtil; - float egrDisutil; - if (storedDriveEgressUtils[aTap][aTaz] == StoredUtilityData.default_utility) { - egrUtil = calcDriveEgressUtility(driveDmu, aTap, aTaz, aMgra, accMode, writeCalculations, myLogger); - storedDriveEgressUtils[aTap][aTaz] = egrUtil; - } else { - egrUtil = storedDriveEgressUtils[aTap][aTaz]; - if(writeCalculations){ - myLogger.info("Stored drive egress utility from Tap "+aTap+" to TAZ "+aTaz+" is "+egrUtil); - } - } - - //always calculate the access disutility since it changes based on od - egrDisutil = calcDriveEgressRatioDisutility(driveDmu, aTap, aMgra, aTaz, odDistance, accMode, writeCalculations, myLogger); - - // Calculate the pTap to aTap utility values - float tapTapUtil[] = new float[numSkimSets]; - if(!storedDepartPeriodTapTapUtils.get(WTD).get(period).containsKey(storedDataObject.paTapKey(pTap, aTap))) { - - //loop across number of skim sets the pTap to aTap utility values - for (int set=0; set paths Collection of paths - */ - public void trimPaths(ArrayList paths) - { - - //sort paths by total utility in reverse order to get highest utility first - Collections.sort(paths, Collections.reverseOrder()); - - //get best N paths - int count = 0; - for(TransitPath path : paths) { - - if (path.getTotalUtility() > NA) { - - //get data - bestUtilities[count] = path.getTotalUtility(); - bestPTap[count] = path.pTap; - bestATap[count] = path.aTap; - bestSet[count] = path.set; - - count = count + 1; - if(count == numTransitAlts) { - break; - } - } - } - } - - public float calcPathUtility(TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accEgr, int period, int origMgra, int pTap, int aTap, int destMgra, int set, boolean myTrace, Logger myLogger, float odDistance) { - - float accUtil =NA; - float egrUtil =NA; - float tapTapUtil =NA; - float accDisutil =0f; - float egrDisutil =0f; - - - if(accEgr==WTW) { - accUtil = calcWalkAccessUtility(walkDmu, origMgra, pTap, myTrace, myLogger); - egrUtil = calcWalkEgressUtility(walkDmu, aTap, destMgra, myTrace, myLogger); - tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, WTW, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); - } else if(accEgr==WTD) { - int aTaz = mgraManager.getTaz(destMgra); - AccessMode accMode = AccessMode.PARK_N_RIDE; - accUtil = calcWalkAccessUtility(walkDmu, origMgra, pTap, myTrace, myLogger); - egrUtil = calcDriveEgressUtility(driveDmu, aTap, aTaz, destMgra, accMode, myTrace, myLogger); - egrDisutil = calcDriveEgressRatioDisutility(driveDmu, aTap, destMgra,aTaz, odDistance, accMode, myTrace, myLogger); - tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, WTD, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); - } else if(accEgr==DTW) { - int pTaz = mgraManager.getTaz(origMgra); - AccessMode accMode = AccessMode.PARK_N_RIDE; - accUtil = calcDriveAccessUtility(driveDmu, origMgra, pTaz, pTap, accMode, myTrace, myLogger); - accDisutil = calcDriveAccessRatioDisutility(driveDmu, origMgra, pTaz, pTap, odDistance, accMode, myTrace, myLogger); - egrUtil = calcWalkEgressUtility(walkDmu, aTap, destMgra, myTrace, myLogger); - tapTapUtil = calcUtilitiesForTapPair(walkDmu, period, DTW, pTap, aTap, set, origMgra, destMgra, myTrace, myLogger); - } - return(accUtil + tapTapUtil + egrUtil + accDisutil + egrDisutil); - } - - /** - * Return the array of transit best tap pairs for the given access/egress mode, origin MGRA, - * destination MGRA, and departure time period. - * - * @param TransitWalkAccessDMU walkDmu - * @param TransitDriveAccessDMU driveDmu - * @param Modes.AccessMode accMode - * @param origMgra Origin MGRA - * @param workMgra Destination MGRA - * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 =OffPeak period - * @param debug boolean flag to indicate if debugging reports should be logged - * @param logger Logger to which debugging reports should be logged if debug is true - * @return double[][] Array of best tap pair values - rows are N-path, columns are orig tap, dest tap, skim set, utility - */ - public double[][] getBestTapPairs(TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accMode, int origMgra, int destMgra, int departPeriod, boolean debug, Logger myLogger, float odDistance) - { - - String separator = ""; - String header = ""; - if (debug) - { - myLogger.info(""); - myLogger.info(""); - header = ACC_EGR[accMode] + " best tap pairs debug info for origMgra=" + origMgra - + ", destMgra=" + destMgra + ", period index=" + departPeriod - + ", period label=" + ModelStructure.SKIM_PERIOD_STRINGS[departPeriod]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - myLogger.info(""); - myLogger.info(separator); - myLogger.info("Calculating " + header); - } - - double[][] bestTaps = null; - - if(accMode==WTW) { - findBestWalkTransitWalkTaps(walkDmu, departPeriod, origMgra, destMgra, debug, myLogger); - } else if(accMode==DTW) { - findBestDriveTransitWalkTaps(walkDmu, driveDmu, departPeriod, origMgra, destMgra, debug, myLogger, odDistance); - } else if(accMode==WTD) { - findBestWalkTransitDriveTaps(walkDmu, driveDmu, departPeriod, origMgra, destMgra, debug, myLogger, odDistance); - } - - // get and log the best tap-tap utilities by alt - double[] bestUtilities = getBestUtilities(); - bestTaps = new double[bestUtilities.length][]; - - for (int i = 0; i < bestUtilities.length; i++) - { - //only initialize tap data if valid; otherwise null array - if (bestUtilities[i] > NA) bestTaps[i] = getBestTaps(i); - } - - // log the best utilities and tap pairs for each alt - if (debug) - { - myLogger.info(""); - myLogger.info(separator); - myLogger.info(header); - myLogger.info("Final Best Utilities:"); - myLogger.info("Alt, Alt, Utility, bestITap, bestJTap, bestSet"); - for (int i = 0; i < bestUtilities.length; i++) - { - myLogger.info(i + "," + i + "," + bestUtilities[i] + "," - + (bestTaps[i] == null ? "NA" : bestTaps[i][0]) + "," - + (bestTaps[i] == null ? "NA" : bestTaps[i][1]) + "," - + (bestTaps[i] == null ? "NA" : bestTaps[i][2])); - } - - myLogger.info(separator); - } - return bestTaps; - } - - /** - * Calculate utilities for the best tap pairs using person specific attributes. - * - * @param double[][] bestTapPairs - * @param TransitWalkAccessDMU walkDmu - * @param TransitDriveAccessDMU driveDmu - * @param Modes.AccessMode accMode - * @param origMgra Origin MGRA - * @param workMgra Destination MGRA - * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 =OffPeak period - * @param debug boolean flag to indicate if debugging reports should be logged - * @param logger Logger to which debugging reports should be logged if debug is true - * @return double[][] Array of best tap pair values - rows are N-path, columns are orig tap, dest tap, skim set, utility - */ - public double[][] calcPersonSpecificUtilities(double[][] bestTapPairs, TransitWalkAccessDMU walkDmu, TransitDriveAccessDMU driveDmu, int accMode, int origMgra, int destMgra, int departPeriod, boolean debug, Logger myLogger, float odDistance) - { - - String separator = ""; - String header = ""; - if (debug) - { - myLogger.info(""); - myLogger.info(""); - header = accMode + " best tap pairs person specific utility info for origMgra=" + origMgra - + ", destMgra=" + destMgra + ", period index=" + departPeriod - + ", period label=" + ModelStructure.SKIM_PERIOD_STRINGS[departPeriod]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - myLogger.info(""); - myLogger.info(separator); - myLogger.info("Calculating " + header); - } - - //re-calculate utilities - for (int i = 0; i < bestTapPairs.length; i++) { - if (bestTapPairs[i] != null) { - int pTap = (int)bestTapPairs[i][0]; - int aTap = (int)bestTapPairs[i][1]; - int set = (int)bestTapPairs[i][2]; - double utility = calcPathUtility(walkDmu, driveDmu, accMode, departPeriod, origMgra, pTap, aTap, destMgra, set, debug, myLogger, odDistance); - bestTapPairs[i][3] = utility; - } - } - - // log the best utilities and tap pairs for each alt - if (debug) - { - myLogger.info(""); - myLogger.info(separator); - myLogger.info(header); - myLogger.info("Final Person Specific Best Utilities:"); - myLogger.info("Alt, Alt, Utility, bestITap, bestJTap, bestSet"); - int availableModeCount = 0; - for (int i = 0; i < bestUtilities.length; i++) - { - if (bestTapPairs[i] != null) availableModeCount++; - - myLogger.info(i + "," + i + "," - + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][3]) + "," - + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][0]) + "," - + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][1]) + "," - + (bestTapPairs[i] == null ? "NA" : bestTapPairs[i][2])); - } - - myLogger.info(separator); - } - return bestTapPairs; - } - - /* - private LogitModel setupTripLogSum(double[][] bestTapPairs, boolean myTrace, Logger myLogger) { - - //must size logit model ahead of time - int alts = 0; - for (int i=0; i0){ - double cumProb=0; - //re-iterate through paths and calculate probability, choose alternative based on rnum - for(int i = 0; i0) - logsum = Math.log(sumExpUtility); - return logsum; - } - - /** - * Get the best path logsum, subject to constraints. The constraints - * are that the logsum only include a certain number of paths for each - * skim set, as defined in the property utility.bestTransitPath.maxPathsPerSkimSetForLogsum. - * This allows the logsum - * to reduce or eliminate path overlap should any exist in the path set, without having - * access to actual route data in the utility calculation. Transit trips - * are still subject to choice across all paths in the best utility set. - * - * @return The constrained transit logsum. - */ - public double getTransitBestPathLogsum(double[][] bestTapPairs, boolean myTrace, Logger myLogger){ - - double logsum = NA; - double sumExpUtility = getSumExpUtilities(bestTapPairs, myTrace, myLogger); - if(sumExpUtility>0.0) - logsum = Math.log(sumExpUtility); - - if(myTrace) - myLogger.info("Best Transit Path Logsum "+logsum); - return logsum; - } - - - /** - * Get the sum of exponentiated utilities, subject to constraints. The constraints - * are that the sum only include a certain number of paths for each - * skim set, as defined in the property utility.bestTransitPath.maxPathsPerSkimSetForLogsum. - * to reduce or eliminate path overlap should any exist in the path set, without having - * access to actual route data in the utility calculation. Transit trips - * are still subject to choice across all paths in the best utility set. - * - * @param bestTapPairs The tap pairs to calculate the sum exponentiated utility over - * @param myTrace Trace calculations - * @param myLogger The logger to write tracing to - * - * @return The constrained sum of exponentiated utilities. - */ - public double getSumExpUtilities(double[][] bestTapPairs, boolean myTrace, Logger myLogger){ - double sumExpUtility=0; - - if(myTrace){ - myLogger.info("Calculating sum of exponentiated utilities for transit best TAP pairs"); - myLogger.info("Best_Path Utility Skim_Set Included? ExpUtility Sum"); - } - - //utilityCount tracks how many utilities included in logsum calc by skimset - Arrays.fill(utilityCount,0); - for(int i = 0; i WORST_UTILITY){ - int skimSet = bestSet[i]; - - //only include the utility in the logsum if the count - //by skimset hasn't been met yet. - if(utilityCount[skimSet] rbMap, VariableTable dmu) - { - return new UtilityExpressionCalculator(uecSpreadsheet, modelSheet, dataSheet, rbMap, dmu); - } - - /** - * Clears the arrays. This method gets called for two different purposes. One is - * to compare alternatives based on utilities and the other based on - * exponentiated utilities. For this reason, the bestUtilities will be - * initialized by the value passed in as an argument set by the calling method. - * - * @param initialization value - */ - public void clearBestArrays(double initialValue) - { - Arrays.fill(bestUtilities, initialValue); - Arrays.fill(bestPTap, 0); - Arrays.fill(bestATap, 0); - Arrays.fill(bestSet, 0); - } - - /** - * Get the best ptap, atap, and skim set in an array. Only to be called after trimPaths() has been called. - * - * @param alt. - * @return element 0 = best ptap, element 1 = best atap, element 2 = set, element 3= utility - */ - public double[] getBestTaps(int alt) - { - - double[] bestTaps = new double[4]; - - bestTaps[0] = bestPTap[alt]; - bestTaps[1] = bestATap[alt]; - bestTaps[2] = bestSet[alt]; - bestTaps[3] = bestUtilities[alt]; - - return bestTaps; - } - - /** - * Get the best transit alt. Returns null if no transit alt has a valid utility. - * Call only after calling findBestWalkTransitWalkTaps(). - * - * @return The best transit alt (highest utility), or null if no alt have a valid utility. - */ - public int getBestTransitAlt() - { - - int best = -1; - double bestUtility = Double.NEGATIVE_INFINITY; - for (int i = 0; i < bestUtilities.length; ++i) - { - if (bestUtilities[i] > bestUtility) { - best = i; - bestUtility = bestUtilities[i]; - } - } - - int returnSet = best; - if (best > -1) { - returnSet = best; - } - return returnSet; - } - - - /** - * This method writes the utilities for all TAP-pairs for each ride mode. - * It cycles through walk TAPs at the origin end (associated with the origin - * MGRA) and alighting TAPs at the destination end (associated with the - * destination MGRA) and calculates a utility for every available ride mode - * for each TAP pair and writes the result to the outwriter. - * - * The results written will be as follows: - * label,WTW,period,pTap,aTap,mode,combinedUtilities[mode] - * - * @param period The time period (AM, PM, Off) - * @param pMgra - * The origin/production MGRA. - * @param aMgra - * The destination/attraction MGRA. - * @param myLogger A logger for logging problems - * @param outwriter A printwriter for writing results - * @param label A label for the record. - */ - public void writeAllWalkTransitWalkTaps(int period, int pMgra, int aMgra, Logger myLogger, PrintWriter outwriter, String label) - { - - //TODO: Fix this - - /* - - clearBestArrays(Double.NEGATIVE_INFINITY); - - int[] pMgraSet = mgraManager.getMgraWlkTapsDistArray()[pMgra][0]; - int[] aMgraSet = mgraManager.getMgraWlkTapsDistArray()[aMgra][0]; - - if (pMgraSet == null || aMgraSet == null) - { - return; - } - - int pPos = -1; - for (int pTap : pMgraSet) - { - // used to know where we are in time/dist arrays for taps - pPos++; - - // Set the pMgra to pTap walk access utility values, if they haven't - // already been computed. - setWalkAccessUtility(pMgra, pPos, pTap, false, myLogger); - - int aPos = -1; - for (int aTap : aMgraSet) - { - // used to know where we are in time/dist arrays for taps - aPos++; - - // set the pTap to aTap utility values, if they haven't already - // been computed. - setUtilitiesForTapPair(WTW, period, pTap, aTap, false, myLogger); - - // Set the aTap to aMgra walk egress utility values, if they - // haven't already been computed. - setWalkEgressUtility(aTap, aMgra, aPos, false, myLogger); - - // write the utilities for each ride mode - try - { - for (int i = 0; i < combinedUtilities.length; i++){ - combinedUtilities[i] = storedWalkAccessUtils[pMgra][pTap][i] - + storedTapToTapUtils[WTW][period][pTap][aTap][i] - + storedWalkEgressUtils[aTap][aMgra][i]; - - if(combinedUtilities[i]>-500){ - outwriter.print(label); - outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",WTW,period,pTap,aTap,i,combinedUtilities[i]); - } - } - } catch (Exception e) - { - logger.error("exception computing combinedUtilities for WTW"); - logger.error("aTap=" + aTap + "pTap=" + pTap + "aMgra=" + aMgra + "pMgra=" - + pMgra + "period=" + period, e); - throw new RuntimeException(); - } - - - - } - } - */ - } - - /** - * This method writes all TAP-pairs for each ride mode. It cycles - * through drive access TAPs at the origin end (associated with the origin - * MGRA) and alighting TAPs at the destination end (associated with the - * destination MGRA) and calculates a utility for every available ride mode - * for each TAP pair. - * The results written will be as follows: - * - * label,DTW,period,pTap,aTap,mode,combinedUtilities[mode] - * - * @param period The time period (AM, PM, Off) - * @param pMgra - * The origin/production MGRA. - * @param aMgra - * The destination/attraction MGRA. - * @param myLogger A logger for logging problems - * @param outwriter A printwriter for writing results - * @param label A label for the record. - * - */ - public void writeAllDriveTransitWalkTaps(int period, int pMgra, int aMgra, - Logger myLogger, PrintWriter outwriter, String label) - { - - // TODO: Fix this - - /* - - clearBestArrays(Double.NEGATIVE_INFINITY); - - Modes.AccessMode accMode = AccessMode.PARK_N_RIDE; - - int pTaz = mgraManager.getTaz(pMgra); - - if (tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode) == null - || mgraManager.getMgraWlkTapsDistArray()[aMgra][0] == null) - { - return; - } - - float[][][] tapParkingInfo = tapManager.getTapParkingInfo(); - - int pPos = -1; - int[] pTapArray = tazManager.getParkRideOrKissRideTapsForZone(pTaz, accMode); - for (int pTap : pTapArray) - { - pPos++; // used to know where we are in time/dist arrays for taps - - // Set the pTaz to pTap drive access utility values, if they haven't - // already been computed. - setDriveAccessUtility(pTaz, pPos, pTap, accMode, false, myLogger); - - int lotID = (int) tapParkingInfo[pTap][0][0]; // lot ID - float lotCapacity = tapParkingInfo[pTap][2][0]; // lot capacity - - if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) - || (accMode == AccessMode.KISS_N_RIDE)) - { - - int aPos = -1; - for (int aTap : mgraManager.getMgraWlkTapsDistArray()[aMgra][0]) - { - aPos++; - - // Set the aTap to aMgra walk egress utility values, if they - // haven't already been computed. - setWalkEgressUtility(aTap, aMgra, aPos, false, myLogger); - - - // set the pTap to aTap utility values, if they haven't - // already been computed. - setUtilitiesForTapPair(DTW, period, pTap, aTap, false, myLogger); - - // compare the utilities for this TAP pair to previously - // calculated utilities for each ride mode and store the TAP numbers if - // this TAP pair is the best. - try - { - for (int i = 0; i < combinedUtilities.length; i++){ - combinedUtilities[i] = storedDriveAccessUtils[pTaz][pTap][i] - + storedTapToTapUtils[DTW][period][pTap][aTap][i] - + storedWalkEgressUtils[aTap][aMgra][i]; - if(combinedUtilities[i]>-500){ - outwriter.print(label); - outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",DTW,period,pTap,aTap,i,combinedUtilities[i]); - } - } - } catch (Exception e) - { - logger.error("exception computing combinedUtilities for DTW"); - logger.error("aTap=" + aTap + ",pTap=" + pTap + ",aMgra=" + aMgra - + ",pMgra=" + pMgra + ",period=" + period, e); - throw new RuntimeException(); - } - - } - } - - } - */ - } - - /** - * This method finds the best TAP-pairs for each ride mode. It cycles - * through drive access TAPs at the origin end (associated with the origin - * MGRA) and alighting TAPs at the destination end (associated with the - * destination MGRA) and calculates a utility for every available ride mode - * for each TAP pair. - * The results written will be as follows: - * - * label,WTD,period,pTap,aTap,mode,combinedUtilities[mode] - * - * @param period The time period (AM, PM, Off) - * @param pMgra - * The origin/production MGRA. - * @param aMgra - * The destination/attraction MGRA. - * @param myLogger A logger for logging problems - * @param outwriter A printwriter for writing results - * @param label A label for the record. - - * - */ - public void writeAllWalkTransitDriveTaps(int period, int pMgra, int aMgra, - Logger myLogger, PrintWriter outwriter, String label) - { - - // TODO: Fix this - - /* - clearBestArrays(Double.NEGATIVE_INFINITY); - - Modes.AccessMode accMode = AccessMode.PARK_N_RIDE; - - int aTaz = mgraManager.getTaz(aMgra); - - if (mgraManager.getMgraWlkTapsDistArray()[pMgra][0] == null - || tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode) == null) - { - return; - } - - int pPos = -1; - for (int pTap : mgraManager.getMgraWlkTapsDistArray()[pMgra][0]) - { - pPos++; // used to know where we are in time/dist arrays for taps - - // Set the pMgra to pTap walk access utility values, if they haven't - // already been computed. - setWalkAccessUtility(pMgra, pPos, pTap, false, myLogger); - - int aPos = -1; - for (int aTap : tazManager.getParkRideOrKissRideTapsForZone(aTaz, accMode)) - { - aPos++; - - int lotID = (int) tapManager.getTapParkingInfo()[aTap][0][0]; // lot - // ID - float lotCapacity = tapManager.getTapParkingInfo()[aTap][2][0]; // lot - // capacity - if ((accMode == AccessMode.PARK_N_RIDE && tapManager.getLotUse(lotID) < lotCapacity) - || (accMode == AccessMode.KISS_N_RIDE)) - { - - // Set the pTaz to pTap drive access utility values, if they - // haven't already been computed. - setDriveEgressUtility(aTap, aTaz, aPos, accMode, false, myLogger); - - // set the pTap to aTap utility values, if they haven't - // already - // been computed. - setUtilitiesForTapPair(WTD, period, pTap, aTap, false, myLogger); - - // compare the utilities for this TAP pair to previously - // calculated utilities for each ride mode and store the TAP numbers if - // this TAP pair is the best. - try - { - for (int i = 0; i < combinedUtilities.length; i++){ - combinedUtilities[i] = storedWalkAccessUtils[pMgra][pTap][i] - + storedTapToTapUtils[WTD][period][pTap][aTap][i] - + storedDriveEgressUtils[aTap][aTaz][i]; - if(combinedUtilities[i]>-500){ - outwriter.print(label); - outwriter.format(",%d,%d,%d,%d,%d,%9.4f\n",WTD,period,pTap,aTap,i,combinedUtilities[i]); - } - } - } catch (Exception e) - { - logger.error("exception computing combinedUtilities for WTD"); - logger.error("aTap=" + aTap + ",pTap=" + pTap + ",aMgra=" + aMgra - + ",pMgra=" + pMgra + ",period=" + period, e); - throw new RuntimeException(); - } - - } - } - - } - */ - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java deleted file mode 100644 index ae7c5da..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/BuildAccessibilities.java +++ /dev/null @@ -1,1534 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Set; -import java.util.TreeSet; -import java.util.concurrent.ExecutionException; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.Future; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -/** - * This class builds accessibility components for all modes. - * - * @author Joel Freedman - * @version May, 2009 - */ -public final class BuildAccessibilities - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(BuildAccessibilities.class); - - /* - * OLD names private static final String[] WORK_OCCUP_SEGMENT_NAME_LIST = { - * "White Collar", "Services", "Health", "Retail and Food", "Blue Collar", - * "Military" }; private static final int[] WORK_OCCUP_SEGMENT_VALUE_LIST = - * { 71, 72,74, 75, 76, 77 }; - */ - private static final String[] WORK_OCCUP_SEGMENT_NAME_LIST = { - "Management Business Science and Arts", "Services", "Sales and Office", - "Natural Resources Construction and Maintenance", - "Production Transportation and Material Moving", "Military" }; - private static final int[] WORK_OCCUP_SEGMENT_VALUE_LIST = {50, 51, 53, - 54, 55, 56 }; - - // these segment group labels and indices are used for creating all the - // school location choice segments - public static final String[] SCHOOL_DC_SIZE_SEGMENT_NAME_LIST = {"preschool", - "k-8", "unified k-8", "9-12", "unified 9-12", "univ typical", "univ non-typical"}; - public static final int PRESCHOOL_SEGMENT_GROUP_INDEX = 0; - public static final int GRADE_SCHOOL_SEGMENT_GROUP_INDEX = 1; - public static final int UNIFIED_GRADE_SCHOOL_SEGMENT_GROUP_INDEX = 2; - public static final int HIGH_SCHOOL_SEGMENT_GROUP_INDEX = 3; - public static final int UNIFIED_HIGH_SCHOOL_SEGMENT_GROUP_INDEX = 4; - public static final int UNIV_TYPICAL_SEGMENT_GROUP_INDEX = 5; - public static final int UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX = 6; - - private static final int UNIFIED_DISTRICT_OFFSET = 1000000; - - // these indices define the alternative numbers and size term calculation - // indices - public static final int PRESCHOOL_ALT_INDEX = 0; - public static final int GRADE_SCHOOL_ALT_INDEX = 1; - public static final int HIGH_SCHOOL_ALT_INDEX = 2; - public static final int UNIV_TYPICAL_ALT_INDEX = 3; - public static final int UNIV_NONTYPICAL_ALT_INDEX = 4; - - // school segments: preschool, grade school, high school, university - // typical, university non-typical - private static final int[] SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX = {3, 4, 5, 6, 6}; - private static final int[] SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX = {5, 4, 3, 2, 2}; - private static final int[] SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX = {3, 3, 3, 2, 2}; - - // MAX_LUZ is the largest LUZ value, including external LUZs. - public static final int MAX_LUZ = 236; - - // using trip mode choice IVT coefficient value - public static final double TIME_COEFFICIENT = -0.032; - - public static final int[] EXTERNAL_LUZS = {230, 231, 232, - 233, 234, 235, 236 }; - - public static final int[] EXTERNAL_LUZ_CORDON_LUZS = {156, 156, 213, - 157, 122, 156, 156 }; - - public static final int[] MINUTES_TO_ADD_TO_CORDON_FOR_EXTERNAL_LUZ = {105, 480, - 1800, 1740, 155, 8160, 8760 }; - - public static final int[] CORDON_LUZS = {122, 156, 157, - 213 }; - - public static final int[][] EXTERNAL_LUZS_FOR_CORDON_LUZ = { {234}, - {230, 231, 235, 236}, {233}, {232} }; - - // in the LU logsums array, 1st dimension is averaging type, 2nd is pk or - // op, 3rd is auto sufficiency segment, 4th is orig LUZ, 5th is dest LUZ. - public static final int SIMPLE = 0; - public static final int LOGIT = 1; - public static final int PK = 0; - public static final int OP = 1; - public static final int LS0 = 0; - public static final int LS1 = 1; - public static final int LS2 = 2; - - private int[][] externalLuzsForCordonLuz; - private int[] cordonLuzForExternalLuz; - private int[] cordonLuzMinutesForExternalLuz; - - public static final int NUM_AVG_METHODS = 2; - public static final int NUM_PERIODS = 2; - public static final int NUM_SUFFICIENCY_SEGMENTS = 3; - - private static BuildAccessibilities objInstance = null; - - private int[] nonUniversitySegments; - private int[] universitySegments; - - private int univTypicalSegment; - private int univNonTypicalSegment; - - private int[] mgraGsDistrict; - private int[] mgraHsDistrict; - private HashMap gsDistrictIndexMap = new HashMap(); - private HashMap hsDistrictIndexMap = new HashMap(); - - private String[] schoolSegmentSizeNames; - private int[] schoolDcSoaUecSheets; - private int[] schoolDcUecSheets; - private int[] schoolStfUecSheets; - - // a set of school segment indices for which shadow pricing is not done - - // currently includes pre-school segment only. - private HashSet noShadowPriceSchoolSegmentIndices; - - private HashMap psSegmentIndexNameMap; - private HashMap psSegmentNameIndexMap; - private HashMap gsSegmentIndexNameMap; - private HashMap gsSegmentNameIndexMap; - private HashMap hsSegmentIndexNameMap; - private HashMap hsSegmentNameIndexMap; - private HashMap univTypSegmentIndexNameMap; - private HashMap univTypSegmentNameIndexMap; - private HashMap univNonTypSegmentIndexNameMap; - private HashMap univNonTypSegmentNameIndexMap; - - private HashMap schoolSegmentIndexNameMap; - private HashMap schoolSegmentNameIndexMap; - - private HashMap workSegmentIndexNameMap; - private HashMap workSegmentNameIndexMap; - - public static final int TOTAL_LOGSUM_FIELD_NUMBER = 13; - - private static int numThreads = 10; - private static final int DISTRIBUTED_PACKET_SIZE = 1000; - - private HashMap workerOccupValueSegmentIndexMap; - - private UtilityExpressionCalculator constantsUEC; - private UtilityExpressionCalculator sizeTermUEC; - private UtilityExpressionCalculator workerSizeTermUEC; - private UtilityExpressionCalculator schoolSizeTermUEC; - - private AccessibilitiesDMU aDmu; - - private IndexValues iv; - - private double[][][] sovExpUtilities; - private double[][][] hovExpUtilities; - private double[][][] nMotorExpUtilities; - - private MgraDataManager mgraManager; - - // purpose (defined in UEC) - private double[][] sizeTerms; // mgra, - - // indicates whether this mgra has a size term - private boolean[] hasSizeTerm; // mgra, - - // purpose (defined in UEC) - private double[][] workerSizeTerms; // mgra, - - // purpose (defined in UEC) - private double[][] schoolSizeTerms; // mgra, - - // Land Use size terms for school are not segmented by school district - private double[][] luSchoolSizeTerms; - - // auto sufficiency (0 autos, autos=adults), - // and mode (SOV,HOV,Walk-Transit,Non-Motorized) - private double[][] expConstants; - - // accessibilities by mgra, accessibility alternative - private AccessibilitiesTable accessibilitiesTableObject; - - // array for storing land use accessibility mode choice logsum values - private double[][][][][] landUseLogsums; - private int[][] landUseLogsumsCount; - private float[][] landUseAccessibilities; - - private boolean calculateLuAccessibilities; - - private static final int MARKET_SEGMENTS = 3; - - public static final int ESCORT_INDEX = 0; - public static final int SHOP_INDEX = 1; - public static final int OTH_MAINT_INDEX = 2; - public static final int EATOUT_INDEX = 3; - public static final int VISIT_INDEX = 4; - public static final int OTH_DISCR_INDEX = 5; - - private HashMap nonMandatorySizeSegmentNameIndexMap; - - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - private Tracer tracer; - private boolean seek; - - private int maxMgra; - - private boolean accessibilitiesBuilt = false; - private boolean logResults=false; - - private BuildAccessibilities() - { - } - - public static synchronized BuildAccessibilities getInstance() - { - if (objInstance == null) - { - objInstance = new BuildAccessibilities(); - objInstance.accessibilitiesBuilt = false; - return objInstance; - } else - { - objInstance.accessibilitiesBuilt = true; - return objInstance; - } - } - - public void setupBuildAccessibilities(HashMap rbMap, - boolean calculateLuAccessibilities) - { - - this.calculateLuAccessibilities = calculateLuAccessibilities; - - logResults = Util.getStringValueFromPropertyMap(rbMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - - Runtime runtime = Runtime.getRuntime(); - if (numThreads < 0) - { - int nrOfProcessors = runtime.availableProcessors(); - numThreads = nrOfProcessors; - } - - gsDistrictIndexMap = new HashMap(); - hsDistrictIndexMap = new HashMap(); - workerOccupValueSegmentIndexMap = new HashMap(); - - for (int i = 0; i < WORK_OCCUP_SEGMENT_VALUE_LIST.length; ++i) - { - workerOccupValueSegmentIndexMap.put(WORK_OCCUP_SEGMENT_VALUE_LIST[i], i); - } - aDmu = new AccessibilitiesDMU(); - - // Create the UECs - String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.data.page"); - int constantsPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.constants.page"); - int sizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sizeTerm.page"); - int workerSizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, - "acc.workerSizeTerm.page"); - int schoolSizeTermPage = Util.getIntegerValueFromPropertyMap(rbMap, - "acc.schoolSizeTerm.page"); - - File uecFile = new File(uecFileName); - constantsUEC = new UtilityExpressionCalculator(uecFile, constantsPage, dataPage, rbMap, - aDmu); - sizeTermUEC = new UtilityExpressionCalculator(uecFile, sizeTermPage, dataPage, rbMap, aDmu); - workerSizeTermUEC = new UtilityExpressionCalculator(uecFile, workerSizeTermPage, dataPage, - rbMap, aDmu); - schoolSizeTermUEC = new UtilityExpressionCalculator(uecFile, schoolSizeTermPage, dataPage, - rbMap, aDmu); - - mgraManager = MgraDataManager.getInstance(rbMap); - maxMgra = mgraManager.getMaxMgra(); - - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - iv = new IndexValues(); - - workSegmentIndexNameMap = new HashMap(); - workSegmentNameIndexMap = new HashMap(); - - noShadowPriceSchoolSegmentIndices = new HashSet(); - - schoolSegmentIndexNameMap = new HashMap(); - schoolSegmentNameIndexMap = new HashMap(); - - psSegmentIndexNameMap = new HashMap(); - psSegmentNameIndexMap = new HashMap(); - gsSegmentIndexNameMap = new HashMap(); - gsSegmentNameIndexMap = new HashMap(); - hsSegmentIndexNameMap = new HashMap(); - hsSegmentNameIndexMap = new HashMap(); - univTypSegmentIndexNameMap = new HashMap(); - univTypSegmentNameIndexMap = new HashMap(); - univNonTypSegmentIndexNameMap = new HashMap(); - univNonTypSegmentNameIndexMap = new HashMap(); - - nonMandatorySizeSegmentNameIndexMap = new HashMap(); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, - ESCORT_INDEX); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, - SHOP_INDEX); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, - OTH_MAINT_INDEX); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, - EATOUT_INDEX); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, - VISIT_INDEX); - nonMandatorySizeSegmentNameIndexMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME, - OTH_DISCR_INDEX); - - } - - - // This method is only called if the main ABM command line argument for - // calculating land use accessibilities is true - // otherwise, the array lanUseLogsums is null and serves as an indicator - // that LU accessibilities are not needed - public void setCalculatedLandUseAccessibilities() - { - - landUseLogsums = new double[NUM_AVG_METHODS][NUM_PERIODS][NUM_SUFFICIENCY_SEGMENTS][MAX_LUZ + 1][MAX_LUZ + 1]; - landUseLogsumsCount = new int[MAX_LUZ + 1][MAX_LUZ + 1]; - - int maxMgra = mgraManager.getMaxMgra(); - landUseAccessibilities = new float[maxMgra + 1][]; - - externalLuzsForCordonLuz = new int[MAX_LUZ + 1][]; - cordonLuzForExternalLuz = new int[MAX_LUZ + 1]; - cordonLuzMinutesForExternalLuz = new int[MAX_LUZ + 1]; - - // associate the array of external LUZs that belong to each cordon LUZ. - for (int i = 0; i < CORDON_LUZS.length; i++) - { - int cordonLuz = CORDON_LUZS[i]; - externalLuzsForCordonLuz[cordonLuz] = EXTERNAL_LUZS_FOR_CORDON_LUZ[i]; - } - - // associate the cordon LUZ that belongs to each external LUZ. - for (int i = 0; i < EXTERNAL_LUZS.length; i++) - { - int externalLuz = EXTERNAL_LUZS[i]; - cordonLuzForExternalLuz[externalLuz] = EXTERNAL_LUZ_CORDON_LUZS[i]; - } - - // associate the minutes to add to cordon LUZ to represent time to each - // external LUZ. - for (int i = 0; i < EXTERNAL_LUZS.length; i++) - { - int externalLuz = EXTERNAL_LUZS[i]; - cordonLuzMinutesForExternalLuz[externalLuz] = MINUTES_TO_ADD_TO_CORDON_FOR_EXTERNAL_LUZ[i]; - } - - } - - /** - * Calculate size terms and store in sizeTerms array. This method - * initializes the sizeTerms array and loops through mgras in the - * mgraManager, calculates the size term for all size term purposes as - * defined in the size term uec, and stores the results in the sizeTerms - * array. - * - */ - public void calculateSizeTerms() - { - - logger.info("Calculating Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int alternatives = sizeTermUEC.getNumberOfAlternatives(); - sizeTerms = new double[maxMgra + 1][alternatives]; - hasSizeTerm = new boolean[maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = sizeTermUEC.solve(iv, aDmu, null); - - // if ( mgra < 100 ) - // sizeTermUEC.logAnswersArray(logger, - // "NonMandatory Size Terms, MGRA = " + mgra ); - - // store the size terms - for (int purp = 0; purp < alternatives; ++purp) - { - sizeTerms[mgra][purp] = utilities[purp]; - if (sizeTerms[mgra][purp] > 0) hasSizeTerm[mgra] = true; - } - - // log - if (tracer.isTraceOn() && tracer.isTraceZone(taz)) - { - - logger.info("Size Term calculations for mgra " + mgra); - sizeTermUEC.logResultsArray(logger, 0, mgra); - - } - } - } - - /** - * Calculate size terms used for worker DC and store in workerSizeTerms - * array. This method initializes the workerSizeTerms array and loops - * through mgras in the mgraManager, calculates the size term for all work - * size term occupation categories as defined in the worker size term uec, - * and stores the results in the workerSizeTerms array. - * - */ - public void calculateWorkerSizeTerms() - { - - logger.info("Calculating Worker DC Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int alternatives = workerSizeTermUEC.getNumberOfAlternatives(); - workerSizeTerms = new double[alternatives][maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = workerSizeTermUEC.solve(iv, aDmu, null); - - // store the size terms - for (int segment = 0; segment < alternatives; segment++) - { - workerSizeTerms[segment][mgra] = utilities[segment]; - } - - // log - if (tracer.isTraceOn() && tracer.isTraceZone(taz)) - { - logger.info("Worker Size Term calculations for mgra " + mgra); - workerSizeTermUEC.logResultsArray(logger, 0, mgra); - } - } - } - - /** - * Calculate size terms used for school DC and store in schoolSizeTerms - * array. This method initializes the schoolSizeTerms array and loops - * through mgras in the mgraManager, calculates the size term for all school - * size term categories, preschool is defined in the preschool size term - * uec, K-8 and 9-12 use their respective enrollments as size terms, and - * university uses a size term uec, segmented by "typical student". Size - * terms for preschool, k-8, 9-12, university typical amd university - * non-typical are stored in the studentSizeTerms array. - * - */ - public void calculateSchoolSizeTerms() - { - - logger.info("Calculating Student DC Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - - String[] schoolSizeNames = getSchoolSegmentNameList(); - schoolSizeTerms = new double[schoolSizeNames.length][maxMgra + 1]; - luSchoolSizeTerms = new double[UNIV_NONTYPICAL_ALT_INDEX + 1][maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - // int dummy=0; - // if ( mgra == 1801 ){ - // dummy = 1; - // } - - int gsDistrict = getMgraGradeSchoolDistrict(mgra); - int hsDistrict = getMgraHighSchoolDistrict(mgra); - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = schoolSizeTermUEC.solve(iv, aDmu, null); - - // store the preschool size terms - schoolSizeTerms[PRESCHOOL_ALT_INDEX][mgra] = utilities[PRESCHOOL_ALT_INDEX]; - luSchoolSizeTerms[PRESCHOOL_ALT_INDEX][mgra] = utilities[PRESCHOOL_ALT_INDEX]; - - // store the grade school size term for the district this mgra is in - int seg = getGsDistrictIndex(gsDistrict); - schoolSizeTerms[seg][mgra] = utilities[GRADE_SCHOOL_ALT_INDEX]; - luSchoolSizeTerms[GRADE_SCHOOL_ALT_INDEX][mgra] = utilities[GRADE_SCHOOL_ALT_INDEX]; - - // store the high school size term for the district this mgra is in - seg = getHsDistrictIndex(hsDistrict); - schoolSizeTerms[seg][mgra] = utilities[HIGH_SCHOOL_ALT_INDEX]; - luSchoolSizeTerms[HIGH_SCHOOL_ALT_INDEX][mgra] = utilities[HIGH_SCHOOL_ALT_INDEX]; - - // store the university typical size terms - schoolSizeTerms[univTypicalSegment][mgra] = utilities[UNIV_TYPICAL_ALT_INDEX]; - luSchoolSizeTerms[UNIV_TYPICAL_ALT_INDEX][mgra] = utilities[UNIV_TYPICAL_ALT_INDEX]; - - // store the university non-typical size terms - schoolSizeTerms[univNonTypicalSegment][mgra] = utilities[UNIV_NONTYPICAL_ALT_INDEX]; - luSchoolSizeTerms[UNIV_NONTYPICAL_ALT_INDEX][mgra] = utilities[UNIV_NONTYPICAL_ALT_INDEX]; - - // log - if (tracer.isTraceOn() && tracer.isTraceZone(taz)) - { - logger.info("School Size Term calculations for mgra " + mgra); - schoolSizeTermUEC.logResultsArray(logger, 0, mgra); - } - } - } - - public double[][] calculateSchoolSegmentFactors() - { - - ArrayList mgras = mgraManager.getMgras(); - int maxMgra = mgraManager.getMaxMgra(); - String[] schoolSizeNames = getSchoolSegmentNameList(); - double[][] schoolFactors = new double[schoolSizeNames.length][maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - // store the size terms - double univEnrollment = getTotalMgraUniversityEnrollment(mgra); - - for (int seg : nonUniversitySegments) - schoolFactors[seg][mgra] = 1.0; - - double totalSize = 0.0; - for (int seg : universitySegments) - totalSize += schoolSizeTerms[seg][mgra]; - - for (int seg : universitySegments) - if (totalSize == 0) schoolFactors[seg][mgra] = 0; - else schoolFactors[seg][mgra] = univEnrollment / totalSize; - - } - - return schoolFactors; - - } - - /** - * Calculate constant terms, exponentiate, and store in constants array. - */ - public void calculateConstants() - { - - logger.info("Calculating constants"); - - int modes = constantsUEC.getNumberOfAlternatives(); - expConstants = new double[MARKET_SEGMENTS + 1][modes]; // last element - // in - // market segments is - // for total - - for (int i = 0; i < MARKET_SEGMENTS + 1; ++i) - { - - aDmu.setAutoSufficiency(i); - - double[] utilities = constantsUEC.solve(iv, aDmu, null); - - // exponentiate the constants - for (int j = 0; j < modes; ++j) - { - expConstants[i][j] = Math.exp(utilities[j]); - logger.info("Exp. Constant, market " + i + " mode " + j + " = " - + expConstants[i][j]); - } - } - } - - public void calculateDCUtilitiesDistributed(HashMap rbMap) - { - - int packetSize = DISTRIBUTED_PACKET_SIZE; - int numPackets = mgraManager.getMgras().size(); - - int startIndex = 0; - int endIndex = 0; - - ArrayList startEndIndexList = new ArrayList(); - - // assign start, end MGRA ranges to be used to assign to tasks - while (endIndex < numPackets - 1) - { - endIndex = startIndex + packetSize - 1; - - if (endIndex + packetSize > numPackets) endIndex = numPackets - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += packetSize; - } - - float[][] accessibilities = submitTasks(startEndIndexList, rbMap); - accessibilitiesTableObject = new AccessibilitiesTable(accessibilities); - - // output data - String projectDirectory = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(rbMap, "acc.output.file"); - accessibilitiesTableObject.writeAccessibilityTableToFile(accFileName); - accessibilitiesBuilt = true; - - if (landUseLogsums != null) - { - - boolean useSimpleMethod = false; - useSimpleMethod = Util.getBooleanValueFromPropertyMap(rbMap, - "lu.acc.simple.averaging.method"); - - boolean useLogitMethod = false; - useLogitMethod = Util.getBooleanValueFromPropertyMap(rbMap, - "lu.acc.logit.averaging.method"); - - // output land use data - String luAccFileName = projectDirectory - + Util.getStringValueFromPropertyMap(rbMap, "lu.acc.output.file"); - float[][][] luzAccessibilities = aggregateMgraAccessibilitiesToLuz(landUseAccessibilities); - if (useSimpleMethod) - accessibilitiesTableObject - .writeLandUseAccessibilityTableToFile( - luAccFileName.substring(0, luAccFileName.indexOf('.')) + "_simple" - + ".csv", luzAccessibilities[SIMPLE]); - if (useLogitMethod) - accessibilitiesTableObject.writeLandUseAccessibilityTableToFile( - luAccFileName.substring(0, luAccFileName.indexOf('.')) + "_logit" + ".csv", - luzAccessibilities[LOGIT]); - - // output land use mode choice logsums - String luMcLogsumsFileName = projectDirectory - + Util.getStringValueFromPropertyMap(rbMap, "lu.acc.mc.logsums.output.file"); - if (useSimpleMethod) - accessibilitiesTableObject.writeLandUseLogsumTablesToFile( - luMcLogsumsFileName.substring(0, luMcLogsumsFileName.indexOf('.')) - + "_simple" + ".csv", landUseLogsums[SIMPLE]); - if (useLogitMethod) - accessibilitiesTableObject.writeLandUseLogsumTablesToFile( - luMcLogsumsFileName.substring(0, luMcLogsumsFileName.indexOf('.')) - + "_logit" + ".csv", landUseLogsums[LOGIT]); - - } - - } - - private float[][][] aggregateMgraAccessibilitiesToLuz(float[][] landUseAccessibilities) - { - - // simple averaging uses accumulated logsum values, logit averaging uses - // accumulated utility values - float[][][] returnValues = new float[NUM_AVG_METHODS][MAX_LUZ + 1][]; - int[] numberOfMgraValues = new int[MAX_LUZ + 1]; - - for (int i = 1; i < landUseAccessibilities.length; i++) - { - - if (landUseAccessibilities[i] == null) continue; - - // the last column of accessibilities holds the mgra value - int numColumns = landUseAccessibilities[i].length; - int mgra = (int) landUseAccessibilities[i][numColumns - 1]; - int luz = mgraManager.getMgraLuz(mgra); - - // allocate the columns for aggregate arrays - for (int k = 0; k < NUM_AVG_METHODS; k++) - { - if (returnValues[k][luz] == null) - { - returnValues[k][luz] = new float[numColumns]; - returnValues[k][luz][numColumns - 1] = luz; - } - } - - numberOfMgraValues[luz]++; - - // aggregate the mgra column values in the corresponding luz row of - // the return array - excluding the last column - for (int j = 0; j < numColumns - 1; j++) - { - returnValues[SIMPLE][luz][j] += landUseAccessibilities[i][j]; - returnValues[LOGIT][luz][j] += Math.exp(landUseAccessibilities[i][j]); - } - - // calculate logsums from external LUZs to all destination LUZs if - // the origin LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[luz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[luz]) - { - - // allocate the columns for aggregate arrays - for (int k = 0; k < NUM_AVG_METHODS; k++) - { - if (returnValues[k][exLuz] == null) - { - returnValues[k][exLuz] = new float[numColumns]; - returnValues[k][exLuz][numColumns - 1] = exLuz; - } - } - - numberOfMgraValues[exLuz]++; - - for (int j = 0; j < numColumns - 1; j++) - { - returnValues[SIMPLE][exLuz][j] += landUseAccessibilities[i][j]; - returnValues[LOGIT][exLuz][j] += Math.exp(landUseAccessibilities[i][j]); - } - - } - - } - - } - - for (int i = 0; i < returnValues[SIMPLE].length; i++) - { - - if (returnValues[SIMPLE][i] == null) continue; - - // the last column of accessibilities holds the mgra value - int numColumns = returnValues[SIMPLE][i].length; - int luz = (int) returnValues[SIMPLE][i][numColumns - 1]; - - // average the the luz values of the return array - excluding the - // last column - for (int j = 0; j < numColumns - 1; j++) - { - returnValues[SIMPLE][luz][j] /= numberOfMgraValues[luz]; - returnValues[LOGIT][luz][j] = (float) Math.log(returnValues[LOGIT][luz][j]); - } - - // calculate logsums from external LUZs to all destination LUZs if - // the origin LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[luz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[luz]) - { - - for (int j = 0; j < numColumns - 1; j++) - { - returnValues[SIMPLE][exLuz][j] /= numberOfMgraValues[exLuz]; - returnValues[LOGIT][exLuz][j] = (float) Math - .log(returnValues[LOGIT][exLuz][j]); - } - - } - - } - - } - - return returnValues; - - } - - public void readAccessibilityTableFromFile(String accFileName) - { - accessibilitiesTableObject = new AccessibilitiesTable(accFileName); - logger.info("accessibilities table read from file."); - } - - public boolean getAccessibilitiesAreBuilt() - { - return objInstance.accessibilitiesBuilt; - } - - public AccessibilitiesTable getAccessibilitiesTableObject() - { - return accessibilitiesTableObject; - } - - /** - * @param client - * is a JPPFClient object which is used to establish a connection - * to a computing node, submit tasks, and receive results. - */ - private float[][] submitTasks(ArrayList startEndIndexList, HashMap rbMap) - { - - // Create a setup task object and submit it to the computing node. - // This setup task creates the HouseholdChoiceModelManager and causes it - // to - // create the necessary numuber - // of HouseholdChoiceModels objects which will operate in parallel on - // the - // computing node. - - ExecutorService exec = Executors.newFixedThreadPool(numThreads); - ArrayList>> results = new ArrayList>>(); - - float[][][][][] accumulatedLandUseLogsums; - int[][] accumulatedLandUseLogsumsCount; - - int startIndex = 0; - int endIndex = 0; - int taskIndex = 1; - for (int[] startEndIndices : startEndIndexList) - { - startIndex = startEndIndices[0]; - endIndex = startEndIndices[1]; - - logger.info(String.format("creating TASK: %d range: %d to %d.", taskIndex, startIndex, - endIndex)); - - DcUtilitiesTaskJppf task = new DcUtilitiesTaskJppf(taskIndex, startIndex, endIndex, - sovExpUtilities, hovExpUtilities, nMotorExpUtilities, hasSizeTerm, - expConstants, sizeTerms, workerSizeTerms, luSchoolSizeTerms, - externalLuzsForCordonLuz, cordonLuzForExternalLuz, - cordonLuzMinutesForExternalLuz, rbMap, calculateLuAccessibilities); - - results.add(exec.submit(task)); - taskIndex++; - } - - float[][] accessibilities = new float[maxMgra + 1][]; - - for (Future> fs : results) - { - - try - { - List resultBundle = fs.get(); - int task = (Integer) resultBundle.get(0); - int start = (Integer) resultBundle.get(1); - int end = (Integer) resultBundle.get(2); - if (logResults) logger.info(String.format("returned TASK: %d, start=%d, end=%d.", task, start, end)); - float[][] taskAccessibilities = (float[][]) resultBundle.get(3); - for (int i = 0; i < taskAccessibilities.length; i++) - { - if (taskAccessibilities[i] == null) continue; - int numColumns = taskAccessibilities[i].length; - int mgra = (int) taskAccessibilities[i][numColumns - 1]; - accessibilities[mgra] = taskAccessibilities[i]; - } - - if (landUseLogsums != null) - { - - // get land use accessibilities result if they were - // calculated - taskAccessibilities = (float[][]) resultBundle.get(4); - if (taskAccessibilities != null) - { - for (int i = 0; i < taskAccessibilities.length; i++) - { - if (taskAccessibilities[i] == null) continue; - int numColumns = taskAccessibilities[i].length; - int mgra = (int) taskAccessibilities[i][numColumns - 1]; - landUseAccessibilities[mgra] = taskAccessibilities[i]; - } - } - - // store the land use logsums array - accumulatedLandUseLogsums = (float[][][][][]) resultBundle.get(5); - accumulatedLandUseLogsumsCount = (int[][]) resultBundle.get(6); - accumulateLandUseModeChoiceLogsums(accumulatedLandUseLogsums, - accumulatedLandUseLogsumsCount); - - } - - } catch (InterruptedException e) - { - e.printStackTrace(); - throw new RuntimeException(); - } catch (ExecutionException e) - { - logger.error("Exception returned in place of result object.", e); - throw new RuntimeException(); - } finally - { - exec.shutdown(); - } - - } // future - - // calculate the averages for the land use mode choice logsums - if (landUseLogsums != null) averageLandUseModeChoiceLogsums(); - - return accessibilities; - - } - - private void averageLandUseModeChoiceLogsums() - { - - for (int i = 0; i < NUM_PERIODS; i++) - { - for (int j = 0; j < NUM_SUFFICIENCY_SEGMENTS; j++) - { - for (int k = 1; k <= MAX_LUZ; k++) - { - for (int m = 1; m <= MAX_LUZ; m++) - { - landUseLogsums[SIMPLE][i][j][k][m] = landUseLogsums[SIMPLE][i][j][k][m] - / landUseLogsumsCount[k][m]; - landUseLogsums[LOGIT][i][j][k][m] = Math - .log(landUseLogsums[LOGIT][i][j][k][m]); - } - } - } - } - } - - private void accumulateLandUseModeChoiceLogsums(float[][][][][] accumulatedLandUseLogsums, - int[][] accumulatedLandUseLogsumsCount) - { - - for (int k = 1; k <= MAX_LUZ; k++) - { - for (int m = 1; m <= MAX_LUZ; m++) - { - landUseLogsumsCount[k][m] += accumulatedLandUseLogsumsCount[k][m]; - for (int i = 0; i < NUM_PERIODS; i++) - { - for (int j = 0; j < NUM_SUFFICIENCY_SEGMENTS; j++) - { - landUseLogsums[SIMPLE][i][j][k][m] += accumulatedLandUseLogsums[SIMPLE][i][j][k][m]; - landUseLogsums[LOGIT][i][j][k][m] += accumulatedLandUseLogsums[LOGIT][i][j][k][m]; - } - } - } - } - - } - - public double[][] getExpConstants() - { - return expConstants; - } - - /** - * @return the array of alternative labels from the UEC used to calculate - * work tour destination choice size terms. - */ - public String[] getWorkSegmentNameList() - { - return WORK_OCCUP_SEGMENT_NAME_LIST; - } - - /** - * @return the table, MGRAs by occupations, of size terms calcuated for - * worker DC. - */ - public double[][] getWorkerSizeTerms() - { - return workerSizeTerms; - } - - /** - * @return the array of alternative labels from the UEC used to calculate - * work tour destination choice size terms. - */ - public String[] getSchoolSegmentNameList() - { - return schoolSegmentSizeNames; - } - - /** - * Specify the mapping between PECAS occupation codes, occupation segment - * labels, and work location choice segment indices. - */ - public void createWorkSegmentNameIndices() - { - - // get the list of segment names for worker tour destination choice size - String[] occupNameList = WORK_OCCUP_SEGMENT_NAME_LIST; - - // get the list of segment values for worker tour destination choice - // size - int[] occupValueList = WORK_OCCUP_SEGMENT_VALUE_LIST; - - for (int value : occupValueList) - { - int index = workerOccupValueSegmentIndexMap.get(value); - String name = occupNameList[index]; - workSegmentIndexNameMap.put(index, name); - workSegmentNameIndexMap.put(name, index); - } - - } - - /** - * Specify/create the mapping between school segment labels, and school - * location choice segment indices. - */ - public void createSchoolSegmentNameIndices() - { - - ArrayList segmentNames = new ArrayList(); - Set gsDistrictSet = getGradeSchoolDistrictIndices(); - Set hsDistrictSet = getHighSchoolDistrictIndices(); - - String segmentName = ""; - - // add preschool segment to list of segments which will not be shadow - // price adjusted - int sizeSegmentIndex = 0; - segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[PRESCHOOL_SEGMENT_GROUP_INDEX]; - noShadowPriceSchoolSegmentIndices.add(sizeSegmentIndex); - - // add preschool segment as the first segment, with index=0. - segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[PRESCHOOL_SEGMENT_GROUP_INDEX]; - segmentNames.add(segmentName); - schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - psSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - psSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - - // increment the segmentIndex so it's new value is the first grade - // school index - // add grade school segments to list - sizeSegmentIndex++; - gsDistrictIndexMap = new HashMap(); - for (int gsDist : gsDistrictSet) - { - if (gsDist > UNIFIED_DISTRICT_OFFSET) segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIFIED_GRADE_SCHOOL_SEGMENT_GROUP_INDEX] - + "_" + (gsDist - UNIFIED_DISTRICT_OFFSET); - else segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[GRADE_SCHOOL_SEGMENT_GROUP_INDEX] - + "_" + gsDist; - segmentNames.add(segmentName); - schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - gsSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - gsSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - gsDistrictIndexMap.put(gsDist, sizeSegmentIndex); - sizeSegmentIndex++; - } - - // add high school segments to list - hsDistrictIndexMap = new HashMap(); - for (int hsDist : hsDistrictSet) - { - if (hsDist > UNIFIED_DISTRICT_OFFSET) segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIFIED_HIGH_SCHOOL_SEGMENT_GROUP_INDEX] - + "_" + (hsDist - UNIFIED_DISTRICT_OFFSET); - else segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[HIGH_SCHOOL_SEGMENT_GROUP_INDEX] - + "_" + hsDist; - segmentNames.add(segmentName); - schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - hsSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - hsSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - hsDistrictIndexMap.put(hsDist, sizeSegmentIndex); - sizeSegmentIndex++; - } - - // add typical university/colleger segments to list - segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIV_TYPICAL_SEGMENT_GROUP_INDEX]; - segmentNames.add(segmentName); - schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - univTypSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - univTypSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - univTypicalSegment = sizeSegmentIndex; - sizeSegmentIndex++; - - // add non-typical university/colleger segments to list - segmentName = SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]; - segmentNames.add(segmentName); - schoolSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - schoolSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - univNonTypSegmentIndexNameMap.put(sizeSegmentIndex, segmentName); - univNonTypSegmentNameIndexMap.put(segmentName, sizeSegmentIndex); - univNonTypicalSegment = sizeSegmentIndex; - - // create arrays dimensioned as the number of school segments created - // and assign their values - - // 2 university segments - universitySegments = new int[2]; - - // num GS Dists + num HS dists + preschool non-university segments - nonUniversitySegments = new int[segmentNames.size() - universitySegments.length]; - - int u = 0; - int nu = 0; - - schoolSegmentSizeNames = new String[segmentNames.size()]; - for (int i = 0; i < schoolSegmentSizeNames.length; i++) - schoolSegmentSizeNames[i] = segmentNames.get(i); - - schoolDcSoaUecSheets = new int[schoolSegmentSizeNames.length]; - schoolDcUecSheets = new int[schoolSegmentSizeNames.length]; - schoolStfUecSheets = new int[schoolSegmentSizeNames.length]; - - sizeSegmentIndex = 0; - schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; - schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; - schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[PRESCHOOL_ALT_INDEX]; - nonUniversitySegments[nu++] = sizeSegmentIndex; - sizeSegmentIndex++; - - for (int segmentIndex : gsDistrictIndexMap.values()) - { - schoolDcSoaUecSheets[segmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; - schoolDcUecSheets[segmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; - schoolStfUecSheets[segmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[GRADE_SCHOOL_ALT_INDEX]; - nonUniversitySegments[nu++] = segmentIndex; - sizeSegmentIndex++; - } - - for (int segmentIndex : hsDistrictIndexMap.values()) - { - schoolDcSoaUecSheets[segmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; - schoolDcUecSheets[segmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; - schoolStfUecSheets[segmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[HIGH_SCHOOL_ALT_INDEX]; - nonUniversitySegments[nu++] = segmentIndex; - sizeSegmentIndex++; - } - - schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; - schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; - schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[UNIV_TYPICAL_ALT_INDEX]; - universitySegments[u++] = sizeSegmentIndex; - sizeSegmentIndex++; - - schoolDcSoaUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; - schoolDcUecSheets[sizeSegmentIndex] = SCHOOL_LOC_SEGMENT_TO_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; - schoolStfUecSheets[sizeSegmentIndex] = SCHOOL_SEGMENT_TO_STF_UEC_SHEET_INDEX[UNIV_NONTYPICAL_ALT_INDEX]; - universitySegments[u++] = sizeSegmentIndex; - - } - - public HashSet getNoShadowPriceSchoolSegmentIndexSet() - { - return noShadowPriceSchoolSegmentIndices; - } - - public HashMap getSchoolSegmentIndexNameMap() - { - return schoolSegmentIndexNameMap; - } - - public HashMap getSchoolSegmentNameIndexMap() - { - return schoolSegmentNameIndexMap; - } - - /** - * @return pre-school segment index to segment name hashmap - */ - public HashMap getPsSegmentIndexNameMap() - { - return psSegmentIndexNameMap; - } - - /** - * @return pre-school segment name to segment index hashmap - */ - public HashMap getPsSegmentNameIndexMap() - { - return psSegmentNameIndexMap; - } - - /** - * @return grade school segment index to segment name hashmap - */ - public HashMap getGsSegmentIndexNameMap() - { - return gsSegmentIndexNameMap; - } - - /** - * @return grade school segment name to segment index hashmap - */ - public HashMap getGsSegmentNameIndexMap() - { - return gsSegmentNameIndexMap; - } - - /** - * @return high school segment index to segment name hashmap - */ - public HashMap getHsSegmentIndexNameMap() - { - return hsSegmentIndexNameMap; - } - - /** - * @return high school segment name to segment index hashmap - */ - public HashMap getHsSegmentNameIndexMap() - { - return hsSegmentNameIndexMap; - } - - /** - * @return university typical segment index to segment name hashmap - */ - public HashMap getUnivTypicalSegmentIndexNameMap() - { - return univTypSegmentIndexNameMap; - } - - /** - * @return university typical segment name to segment index hashmap - */ - public HashMap getUnivTypicalSegmentNameIndexMap() - { - return univTypSegmentNameIndexMap; - } - - /** - * @return university non typical segment index to segment name hashmap - */ - public HashMap getUnivNonTypicalSegmentIndexNameMap() - { - return univNonTypSegmentIndexNameMap; - } - - /** - * @return university non typical segment name to segment index hashmap - */ - public HashMap getUnivNonTypicalSegmentNameIndexMap() - { - return univNonTypSegmentNameIndexMap; - } - - public HashMap getWorkSegmentIndexNameMap() - { - return workSegmentIndexNameMap; - } - - public HashMap getWorkSegmentNameIndexMap() - { - return workSegmentNameIndexMap; - } - - public HashMap getWorkOccupValueIndexMap() - { - return workerOccupValueSegmentIndexMap; - } - - public int getGsDistrictIndex(int district) - { - return gsDistrictIndexMap.get(district); - } - - public int getHsDistrictIndex(int district) - { - return hsDistrictIndexMap.get(district); - } - - public int[] getSchoolDcSoaUecSheets() - { - return schoolDcSoaUecSheets; - } - - public int[] getSchoolDcUecSheets() - { - return schoolDcUecSheets; - } - - /** - * @return the array of stop frequency uec model sheet indices, indexed by - * school segment - */ - public int[] getSchoolStfUecSheets() - { - return schoolStfUecSheets; - } - - /** - * @return the table, MGRAs by school types, of size terms calcuated for - * school DC. - */ - public double[][] getSchoolSizeTerms() - { - return schoolSizeTerms; - } - - /** - * @return the table, MGRAs by non-mandatory types, of size terms calcuated. - */ - public double[][] getSizeTerms() - { - return sizeTerms; - } - - /** - * @param mgra - * for which table data is desired - * @return population for the specified mgra. - */ - public double getMgraPopulation(int mgra) - { - return mgraManager.getMgraPopulation(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return households for the specified mgra. - */ - public double getMgraHouseholds(int mgra) - { - return mgraManager.getMgraHouseholds(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return grade school enrollment for the specified mgra. - */ - public double getMgraGradeSchoolEnrollment(int mgra) - { - return mgraManager.getMgraGradeSchoolEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return high school enrollment for the specified mgra. - */ - public double getMgraHighSchoolEnrollment(int mgra) - { - return mgraManager.getMgraHighSchoolEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return university enrollment for the specified mgra. - */ - public double getTotalMgraUniversityEnrollment(int mgra) - { - return mgraManager.getMgraUniversityEnrollment(mgra) - + mgraManager.getMgraOtherCollegeEnrollment(mgra) - + mgraManager.getMgraAdultSchoolEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return university enrollment for the specified mgra. - */ - public double getMgraUniversityEnrollment(int mgra) - { - return mgraManager.getMgraUniversityEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return other college enrollment for the specified mgra. - */ - public double getMgraOtherCollegeEnrollment(int mgra) - { - return mgraManager.getMgraOtherCollegeEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return adult school enrollment for the specified mgra. - */ - public double getMgraAdultSchoolEnrollment(int mgra) - { - return mgraManager.getMgraAdultSchoolEnrollment(mgra); - } - - /** - * @param mgra - * for which table data is desired - * @return grade school district for the specified mgra. - */ - public int getMgraGradeSchoolDistrict(int mgra) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); - int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); - if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - return gsDist; - } - - /** - * @param mgra - * for which table data is desired - * @return high school district for the specified mgra. - */ - public int getMgraHighSchoolDistrict(int mgra) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); - int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); - if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - return hsDist; - } - - /** - * @param mgra - * for which table data is desired - * @return school location choice model segment for the specified mgra. - */ - public int getMgraGradeSchoolSegmentIndex(int mgra) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); - int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); - if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - - return gsDistrictIndexMap.get(gsDist); - } - - /** - * @param mgra - * for which table data is desired - * @return school location choice model segment for the specified mgra. - */ - public int getMgraHighSchoolSegmentIndex(int mgra) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(mgra); - int hsDist = mgraManager.getMgraHighSchoolDistrict(mgra); - if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - - return hsDistrictIndexMap.get(hsDist); - } - - /** - * @return set of unique grade school district indices - */ - private TreeSet getGradeSchoolDistrictIndices() - { - int maxMgra = mgraManager.getMaxMgra(); - mgraGsDistrict = new int[maxMgra + 1]; - TreeSet set = new TreeSet(); - for (int r = 1; r <= maxMgra; r++) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(r); - int hsDist = mgraManager.getMgraHighSchoolDistrict(r); - if (gsDist == 0) gsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - set.add(gsDist); - mgraGsDistrict[r] = gsDist; - } - return set; - } - - /** - * @return set of unique high school district indices - */ - private TreeSet getHighSchoolDistrictIndices() - { - int maxMgra = mgraManager.getMaxMgra(); - mgraHsDistrict = new int[maxMgra + 1]; - TreeSet set = new TreeSet(); - for (int r = 1; r <= maxMgra; r++) - { - int gsDist = mgraManager.getMgraGradeSchoolDistrict(r); - int hsDist = mgraManager.getMgraHighSchoolDistrict(r); - if (gsDist == 0) hsDist = UNIFIED_DISTRICT_OFFSET + hsDist; - set.add(hsDist); - mgraHsDistrict[r] = hsDist; - } - return set; - } - - public int[] getMgraGsDistrict() - { - return mgraGsDistrict; - } - - public int[] getMgraHsDistrict() - { - return mgraHsDistrict; - } - - public HashMap getGsDistrictIndexMap() - { - return gsDistrictIndexMap; - } - - public HashMap getHsDistrictIndexMap() - { - return hsDistrictIndexMap; - } - - public HashMap getNonMandatoryPurposeNameIndexMap() - { - return nonMandatorySizeSegmentNameIndexMap; - } - - public int getEscortSizeArraySegmentIndex() - { - return nonMandatorySizeSegmentNameIndexMap.get(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java deleted file mode 100644 index 9d05d1a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DcUtilitiesTaskJppf.java +++ /dev/null @@ -1,880 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.PrintWriter; -import java.net.UnknownHostException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.concurrent.Callable; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - - -public class DcUtilitiesTaskJppf - implements Callable> -{ - - private static final int MIN_EXP_FUNCTION_ARGUMENT = -500; - - private static final String[] LOGSUM_SEGMENTS = {"SOV ", - "HOV ", "Transit ", "NMotorized", "SOVLS_0 ", "SOVLS_1 ", "SOVLS_2 ", - "HOVLS_0_OP", "HOVLS_1_OP", "HOVLS_2_OP", "HOVLS_0_PK", "HOVLS_1_PK", "HOVLS_2_PK", - "TOTAL", "MAAS" }; - - public static final String[] LU_LOGSUM_SEGMENTS = {"LS_0_PK", "LS_1_PK", - "LS_2_PK", "LS_0_OP", "LS_1_OP", "LS_2_OP", "All_PK " }; - - // setting to -1 will prevent debug files from being written - private static final int DEBUG_ILUZ = -1; - private static final int DEBUG_JLUZ = -1; - // private static final int DEBUG_ILUZ = 66; - // private static final int DEBUG_JLUZ = 82; - - private static final int MAX_LU_SIZE_TERM_INDEX = 23; - private static final int MAX_LU_NONMAN_SIZE_TERM_INDEX = 12; - private static final int MAX_LU_WORK_SIZE_TERM_INDEX = 18; - private static final int MAX_LU_SCHOOL_SIZE_TERM_INDEX = 23; - - private MgraDataManager mgraManager; - private boolean[] hasSizeTerm; - private double[][] expConstants; - private double[][] sizeTerms; - private double[][] luSizeTerms; - - // store taz-taz exponentiated utilities (period, from taz, to taz) - private double[][][] sovExpUtilities; - private double[][][] hovExpUtilities; - private double[][][] nMotorExpUtilities; - private double[][][] maasExpUtilities; - - - private float[][] accessibilities; - private float[][] luAccessibilities; - - private int startRange; - private int endRange; - private int taskIndex; - - private boolean seek; - private Tracer tracer; - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - - private BestTransitPathCalculator bestPathCalculator; - - private UtilityExpressionCalculator dcUEC; - private AccessibilitiesDMU aDmu; - - private UtilityExpressionCalculator luUEC; - private AccessibilitiesDMU luDmu; - - private HashMap rbMap; - - private int[][] externalLuzsForCordonLuz; - private int[] cordonLuzForExternalLuz; - private int[] cordonLuzMinutesForExternalLuz; - - private boolean calculateLuAccessibilities; - private PrintWriter outStream; - - public DcUtilitiesTaskJppf(int taskIndex, int startRange, int endRange, - double[][][] mySovExpUtilities, double[][][] myHovExpUtilities, - double[][][] myNMotorExpUtilities, boolean[] hasSizeTerm, double[][] expConstants, - double[][] sizeTerms, double[][] workSizeTerms, double[][] schoolSizeTerms, - int[][] myExternalLuzsForCordonLuz, int[] myCordonLuzForExternalLuz, - int[] myCordonLuzMinutesForExternalLuz, HashMap myRbMap, - boolean myCalculateLuAccessibilities) - { - - rbMap = myRbMap; - sovExpUtilities = mySovExpUtilities; - hovExpUtilities = myHovExpUtilities; - nMotorExpUtilities = myNMotorExpUtilities; - externalLuzsForCordonLuz = myExternalLuzsForCordonLuz; - cordonLuzForExternalLuz = myCordonLuzForExternalLuz; - cordonLuzMinutesForExternalLuz = myCordonLuzMinutesForExternalLuz; - calculateLuAccessibilities = myCalculateLuAccessibilities; - - mgraManager = MgraDataManager.getInstance(rbMap); - - aDmu = new AccessibilitiesDMU(); - - String dcUecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.dcUtility.uec.file"); - int dcDataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.dcUtility.data.page"); - int dcUtilityPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.dcUtility.page"); - - File dcUecFile = new File(dcUecFileName); - dcUEC = new UtilityExpressionCalculator(dcUecFile, dcUtilityPage, dcDataPage, rbMap, aDmu); - - accessibilities = new float[mgraManager.getMaxMgra() + 1][]; - - if (calculateLuAccessibilities) - { - - luDmu = new AccessibilitiesDMU(); - - dcUecFileName = Util.getStringValueFromPropertyMap(rbMap, "lu.acc.dcUtility.uec.file"); - dcDataPage = Util.getIntegerValueFromPropertyMap(rbMap, "lu.acc.dcUtility.data.page"); - dcUtilityPage = Util.getIntegerValueFromPropertyMap(rbMap, "lu.acc.dcUtility.page"); - - File luUecFile = new File(dcUecFileName); - luUEC = new UtilityExpressionCalculator(luUecFile, dcUtilityPage, dcDataPage, rbMap, - luDmu); - - TableDataSet luAltData = luUEC.getAlternativeData(); - luDmu.setAlternativeData(luAltData); - int luAlts = luUEC.getNumberOfAlternatives(); - - luAccessibilities = new float[mgraManager.getMaxMgra() + 1][luAlts + 1]; - - // combine non-mandatory, work, and school size terms into one array - // to be indexed into as follows - // 0-12 non-mandatory, 13-18 work, 19-23 school - luSizeTerms = new double[sizeTerms.length][MAX_LU_SIZE_TERM_INDEX + 1]; - for (int c = 0; c <= MAX_LU_NONMAN_SIZE_TERM_INDEX; c++) - for (int r = 0; r < sizeTerms.length; r++) - luSizeTerms[r][c] = sizeTerms[r][c]; - - for (int c = 0; c < workSizeTerms.length; c++) - for (int r = 0; r < workSizeTerms[c].length; r++) - luSizeTerms[r][c + MAX_LU_NONMAN_SIZE_TERM_INDEX + 1] = workSizeTerms[c][r]; - - for (int c = 0; c < schoolSizeTerms.length; c++) - for (int r = 0; r < schoolSizeTerms[c].length; r++) - luSizeTerms[r][c + MAX_LU_WORK_SIZE_TERM_INDEX + 1] = schoolSizeTerms[c][r]; - - // for ( int c=MAX_LU_NONMAN_SIZE_TERM_INDEX+1; c <= - // MAX_LU_WORK_SIZE_TERM_INDEX; c++ ) - // for ( int r=0; r < workSizeTerms.length; r++ ) - // luSizeTerms[r][c] = - // workSizeTerms[r][c-MAX_LU_NONMAN_SIZE_TERM_INDEX-1]; - // - // for ( int c=MAX_LU_WORK_SIZE_TERM_INDEX+1; c <= - // MAX_LU_SCHOOL_SIZE_TERM_INDEX; c++ ) - // for ( int r=0; r < schoolSizeTerms.length; r++ ) - // luSizeTerms[r][c] = - // schoolSizeTerms[r][c-MAX_LU_WORK_SIZE_TERM_INDEX-1]; - - // try { - // outStream = new PrintWriter( new BufferedWriter( new FileWriter( - // "landUseModeChoiceLogsumCheck" + "_" + taskIndex + ".csv" ) ) ); - // } - // catch (IOException e) { - // System.out.println("IO Exception writing file for checking integerizing procedure: " - // ); - // e.printStackTrace(); - // System.exit(-1); - // } - - } - - this.taskIndex = taskIndex; - this.startRange = startRange; - this.endRange = endRange; - this.hasSizeTerm = hasSizeTerm; - this.expConstants = expConstants; - this.sizeTerms = sizeTerms; - - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - } - - public String getId() - { - return Integer.toString(taskIndex); - } - - public List call() - { - - Logger logger = Logger.getLogger(this.getClass()); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + ", task:" - + taskIndex + "] " + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - logger.info(threadName + " - Calculating Accessibilities"); - - NonTransitUtilities ntUtilities = new NonTransitUtilities(rbMap, sovExpUtilities, - hovExpUtilities, nMotorExpUtilities, maasExpUtilities); - // ntUtilities.setAllUtilities(ntUtilitiesArrays); - // ntUtilities.setNonMotorUtilsMap(ntUtilitiesMap); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(rbMap); - bestPathCalculator = logsumHelper.getBestTransitPathCalculator(); - AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - - // get the accessibilities alternatives - int alts = dcUEC.getNumberOfAlternatives(); - TableDataSet altData = dcUEC.getAlternativeData(); - aDmu.setAlternativeData(altData); - - int luAlts = -1; - double[] luUtilities = null; - ArrayList luUtilityList = null; - float[][][][][] accumulatedLandUseLogsums = null; - int[][] accumulatedLandUseLogsumsCount = null; - - if (calculateLuAccessibilities) - { - // get the land use accessibilities alternatives - luAlts = luUEC.getNumberOfAlternatives(); - TableDataSet luAltData = luUEC.getAlternativeData(); - luDmu.setAlternativeData(luAltData); - - // declare logsums array for LU accessibility - luUtilities = new double[LU_LOGSUM_SEGMENTS.length]; - - luUtilityList = new ArrayList(); - - accumulatedLandUseLogsums = new float[BuildAccessibilities.NUM_AVG_METHODS][BuildAccessibilities.NUM_PERIODS][BuildAccessibilities.NUM_SUFFICIENCY_SEGMENTS][BuildAccessibilities.MAX_LUZ + 1][BuildAccessibilities.MAX_LUZ + 1]; - - accumulatedLandUseLogsumsCount = new int[BuildAccessibilities.MAX_LUZ + 1][BuildAccessibilities.MAX_LUZ + 1]; - - } - - // declare logsums array for ABM - double[] logsums = new double[LOGSUM_SEGMENTS.length]; - - float[] luUtilityResult = new float[LU_LOGSUM_SEGMENTS.length + 2]; - - // DMUs for this UEC - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // LOOP OVER RANGE OF ORIGIN MGRA - ArrayList mgraValues = mgraManager.getMgras(); - for (int i = startRange; i <= endRange; i++) - { // Origin MGRA - - int iMgra = mgraValues.get(i); - - accessibilities[iMgra] = new float[alts + 1]; - - // pre-calculate the hov, sov, and non-motorized exponentiated - // utilities for the origin MGRA. - // the method called returns cached values if they were already - // calculated. - ntUtilities.buildUtilitiesForOrigMgraAndPeriod(iMgra, - NonTransitUtilities.PEAK_PERIOD_INDEX); - ntUtilities.buildUtilitiesForOrigMgraAndPeriod(iMgra, - NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - - // if(originMgras<=10 || (originMgras % 500) ==0 ) - // logger.info("...Origin MGRA "+iMgra); - - int iTaz = mgraManager.getTaz(iMgra); - boolean trace = false; - - if (tracer.isTraceOn() && tracer.isTraceZone(iTaz)) - { - - logger.info("origMGRA, destMGRA, OPSOV, OPHOV, WTRAN, NMOT, SOV0OP, SOV1OP, SOV2OP, HOV0OP, HOV1OP, HOV2OP, HOV0PK, HOV1PK, HOV2PK, ALL"); - trace = true; - } - // for tracing accessibility and logsum calculations - String accString = null; - - // LOOP OVER DESTINATION MGRA - for (Integer jMgra : mgraManager.getMgras()) - { // Destination MGRA - - if (!hasSizeTerm[jMgra]) continue; - - int jTaz = mgraManager.getTaz(jMgra); - - if (seek && !trace) continue; - - double opSovExpUtility = 0; - double opHovExpUtility = 0; - double opMaasExpUtility = 0; - try - { - opSovExpUtility = ntUtilities.getSovExpUtility(iTaz, jTaz, - NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - opHovExpUtility = ntUtilities.getHovExpUtility(iTaz, jTaz, - NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - opMaasExpUtility = ntUtilities.getMaasExpUtility(iTaz, jTaz, - NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - // opSovExpUtility = - // ntUtilities.getAllUtilities()[0][0][iTaz][jTaz]; - // opHovExpUtility = - // ntUtilities.getAllUtilities()[1][0][iTaz][jTaz]; - } catch (Exception e) - { - logger.error("exception for op sov/hov utilitiy taskIndex=" + taskIndex - + ", i=" + i + ", startRange=" + startRange + ", endRange=" + endRange, - e); - System.exit(-1); - } - - // calculate walk-transit exponentiated utility - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.MD_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger); - - // sum the exponentiated utilities over modes - double opWTExpUtility = bestPathCalculator.getSumExpUtilities(); - - double pkSovExpUtility = 0; - double pkHovExpUtility = 0; - double pkMaasExpUtility = 0; - try - { - pkSovExpUtility = ntUtilities.getSovExpUtility(iTaz, jTaz, - NonTransitUtilities.PEAK_PERIOD_INDEX); - pkHovExpUtility = ntUtilities.getHovExpUtility(iTaz, jTaz, - NonTransitUtilities.PEAK_PERIOD_INDEX); - - pkMaasExpUtility = ntUtilities.getMaasExpUtility(iTaz, jTaz, - NonTransitUtilities.PEAK_PERIOD_INDEX); - - // pkSovExpUtility = - // ntUtilities.getAllUtilities()[0][1][iTaz][jTaz]; - // pkHovExpUtility = - // ntUtilities.getAllUtilities()[1][1][iTaz][jTaz]; - } catch (Exception e) - { - logger.error("exception for pk sov/hov utility taskIndex=" + taskIndex + ", i=" - + i + ", startRange=" + startRange + ", endRange=" + endRange, e); - System.exit(-1); - } - - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger); - - // sum the exponentiated utilities over modes - double pkWTExpUtility = bestPathCalculator.getSumExpUtilities(); - - double pkDTExpUtility = 0; - double opDTExpUtility = 0; - - if (calculateLuAccessibilities) - { - - float odDistance = (float) anm.getTazDistanceFromTaz(iTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[jTaz]; - - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger, odDistance); - - // sum the exponentiated utilities over modes - double driveTransitWalkUtilities[] = bestPathCalculator.getBestUtilities(); - if(trace){ - logger.info("PK Drive Transit Utilities (TAP pair number,utility, sum)"); - } - for (int k=0; k < driveTransitWalkUtilities.length; k++){ - if ( driveTransitWalkUtilities[k] > MIN_EXP_FUNCTION_ARGUMENT ) - pkDTExpUtility += Math.exp(driveTransitWalkUtilities[k]); - if(trace) - logger.info(k+","+driveTransitWalkUtilities[k]+","+pkDTExpUtility); - } - - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.MD_SKIM_PERIOD_INDEX, iMgra, jMgra, trace, logger, odDistance); - - // sum the exponentiated utilities over modes - driveTransitWalkUtilities = bestPathCalculator.getBestUtilities(); - if(trace){ - logger.info("OP Drive Transit Utilities (TAP pair number,utility, sum)"); - } - for (int k=0; k < driveTransitWalkUtilities.length; k++){ - if ( driveTransitWalkUtilities[k] > MIN_EXP_FUNCTION_ARGUMENT ) - opDTExpUtility += Math.exp(driveTransitWalkUtilities[k]); - if(trace) - logger.info(k+","+driveTransitWalkUtilities[k]+","+opDTExpUtility); - } - - } - - double nmExpUtility = 0; - try - { - nmExpUtility = ntUtilities.getNMotorExpUtility(iMgra, jMgra, - NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - } catch (Exception e) - { - logger.error("exception for non-motorized utilitiy taskIndex=" + taskIndex - + ", i=" + i + ", startRange=" + startRange + ", endRange=" + endRange, - e); - System.exit(-1); - } - - Arrays.fill(logsums, -999f); - - // 0: OP SOV - logsums[0] = Math.log(opSovExpUtility); - - // 1: OP HOV - logsums[1] = Math.log(opHovExpUtility); - - // 2: Walk-Transit - if (opWTExpUtility > 0) logsums[2] = Math.log(opWTExpUtility); - - // 3: Non-Motorized - if (nmExpUtility > 0) logsums[3] = Math.log(nmExpUtility); - - // 4: SOVLS_0 - logsums[4] = Math.log(opSovExpUtility * expConstants[0][0] + opWTExpUtility - * expConstants[0][2] + nmExpUtility * expConstants[0][3]); - // 5: SOVLS_1 - logsums[5] = Math.log(opSovExpUtility * expConstants[1][0] + opWTExpUtility - * expConstants[1][2] + nmExpUtility * expConstants[1][3]); - - // 6: SOVLS_2 - logsums[6] = Math.log(opSovExpUtility * expConstants[2][0] + opWTExpUtility - * expConstants[2][2] + nmExpUtility * expConstants[2][3]); - - // 7: HOVLS_0_OP - logsums[7] = Math.log(opHovExpUtility * expConstants[0][1] + opWTExpUtility - * expConstants[0][2] + nmExpUtility * expConstants[0][3]); - - // 8: HOVLS_1_OP - logsums[8] = Math.log(opHovExpUtility * expConstants[1][1] + opWTExpUtility - * expConstants[1][2] + nmExpUtility * expConstants[1][3]); - - // 9: HOVLS_2_OP - logsums[9] = Math.log(opHovExpUtility * expConstants[2][1] + opWTExpUtility - * expConstants[2][2] + nmExpUtility * expConstants[2][3]); - - // 10: HOVLS_0_PK - logsums[10] = Math.log(pkHovExpUtility * expConstants[0][1] + pkWTExpUtility - * expConstants[0][2] + nmExpUtility * expConstants[0][3]); - - // 11: HOVLS_1_PK - logsums[11] = Math.log(pkHovExpUtility * expConstants[1][1] + pkWTExpUtility - * expConstants[1][2] + nmExpUtility * expConstants[1][3]); - - // 12: HOVLS_2_PK - logsums[12] = Math.log(pkHovExpUtility * expConstants[2][1] + pkWTExpUtility - * expConstants[2][2] + nmExpUtility * expConstants[2][3]); - - // 13: ALL - logsums[13] = Math.log(pkSovExpUtility * expConstants[3][0] + pkHovExpUtility - * expConstants[3][1] + pkWTExpUtility * expConstants[3][2] + nmExpUtility - * expConstants[3][3]); - - // 14: MAAS - logsums[14] = Math.log(opMaasExpUtility); - - aDmu.setLogsums(logsums); - aDmu.setSizeTerms(sizeTerms[jMgra]); - // double[] utilities = dcUEC.solve(iv, aDmu, null); - - if (trace) - { - String printString = new String(); - printString += (iMgra + "," + jMgra); - for (int j = 0; j < 14; ++j) - { - printString += "," + String.format("%9.2f", logsums[j]); - } - logger.info(printString); - - accString = new String(); - accString = "iMgra, jMgra, Alternative, Logsum, SizeTerm, Accessibility\n"; - - } - // add accessibilities for origin mgra - for (int alt = 0; alt < alts; ++alt) - { - - double logsum = aDmu.getLogsum(alt + 1); - double sizeTerm = aDmu.getSizeTerm(alt + 1); - - accessibilities[iMgra][alt] += (Math.exp(logsum) * sizeTerm); - - if (trace) - { - accString += iMgra + "," + alt + "," + logsum + "," + sizeTerm + "," - + accessibilities[iMgra][alt] + "\n"; - } - - } - - // if luModeChoiceLogsums is null, is has not been initialized, - // meaning that LU accessibility calculations are not needed - if (calculateLuAccessibilities) - { - - // 0: AM Mode Choice utility for 0-autos auto sufficiency - luUtilities[0] = pkSovExpUtility * expConstants[0][0] + pkHovExpUtility - * expConstants[0][1] + pkWTExpUtility * expConstants[0][2] - + pkDTExpUtility * expConstants[0][4] + nmExpUtility - * expConstants[0][3]; - - // 1: AM Mode Choice utility for autos=adults auto - // sufficiency - luUtilities[2] = pkSovExpUtility * expConstants[2][0] + pkHovExpUtility - * expConstants[2][1] + pkWTExpUtility * expConstants[2][2] - + pkDTExpUtility * expConstants[2][4] + nmExpUtility - * expConstants[2][3]; - - // 3: MD Mode Choice utility for 0-autos auto sufficiency - luUtilities[3] = opSovExpUtility * expConstants[0][0] + opHovExpUtility - * expConstants[0][1] + opWTExpUtility * expConstants[0][2] - + opDTExpUtility * expConstants[0][4] + nmExpUtility - * expConstants[0][3]; - - // 4: MD Mode Choice utility for autos=adults auto - // sufficiency - luUtilities[5] = opSovExpUtility * expConstants[2][0] + opHovExpUtility - * expConstants[2][1] + opWTExpUtility * expConstants[2][2] - + opDTExpUtility * expConstants[2][4] + nmExpUtility - * expConstants[2][3]; - - // 6: AM Mode Choice utility for all households - luUtilities[6] = pkSovExpUtility * expConstants[3][0] + pkHovExpUtility - * expConstants[3][1] + pkWTExpUtility * expConstants[3][2] - + pkDTExpUtility * expConstants[3][4] + nmExpUtility - * expConstants[3][3]; - - // calculate non-mandatory destination choice logsums - - luDmu.setLogsums(luUtilities); - luDmu.setSizeTerms(luSizeTerms[jMgra]); - - if (trace) - { - String printString = new String(); - printString += (iMgra + "," + jMgra); - for (int j = 0; j < luUtilities.length; ++j) - { - printString += "," + String.format("%9.2f", luUtilities[j]); - } - logger.info(printString); - - accString = new String(); - accString = "Non-mandatory: iMgra, jMgra, Alternative, LU_Logsum, SizeTerm, LU_Accessibility\n"; - - } - // add accessibilities for origin mgra - for (int alt = 0; alt < luAlts; ++alt) - { - - double logsum = luDmu.getLogsum(alt + 1); - double sizeTerm = luDmu.getSizeTerm(alt + 1); - - luAccessibilities[iMgra][alt] += (logsum * sizeTerm); - - if (trace) - { - accString += iMgra + "," + alt + "," + logsum + "," + sizeTerm + "," - + luAccessibilities[iMgra][alt] + "\n"; - } - } - - // save calculated utilities in a table to return tot the - // calling method for accumulating - luUtilityResult[0] = iMgra; - luUtilityResult[1] = jMgra; - for (int k = 0; k < luUtilities.length; k++) - luUtilityResult[2 + k] = (float) luUtilities[k]; - - accumulateLandUseModeChoiceLogsums(luUtilityResult, - accumulatedLandUseLogsumsCount, accumulatedLandUseLogsums); - - } - - } // end for destinations - - if (trace) - { - logger.info(accString); - } - - // calculate the logsum - for (int alt = 0; alt < alts; ++alt) - { - if (accessibilities[iMgra][alt] > 0) - accessibilities[iMgra][alt] = (float) Math.log(accessibilities[iMgra][alt]); - } - accessibilities[iMgra][alts] = iMgra; - - if (calculateLuAccessibilities) - { - // calculate the land use accessibility logsums - for (int alt = 0; alt < luAlts; ++alt) - { - if (luAccessibilities[iMgra][alt] > 0) - luAccessibilities[iMgra][alt] = (float) Math - .log(luAccessibilities[iMgra][alt]); - - } - luAccessibilities[iMgra][luAlts] = iMgra; - } - - } - - List resultBundle = new ArrayList(7); - resultBundle.add(taskIndex); - resultBundle.add(startRange); - resultBundle.add(endRange); - resultBundle.add(accessibilities); - - if (calculateLuAccessibilities) - { - resultBundle.add(luAccessibilities); - resultBundle.add(accumulatedLandUseLogsums); - resultBundle.add(accumulatedLandUseLogsumsCount); - } else - { - resultBundle.add(null); - resultBundle.add(null); - resultBundle.add(null); - } - - // outStream.close(); - - return resultBundle; - - } - - // private void debugLandUseModeChoiceLogsums( int iMgra, int jMgra, int - // iLuz, int jLuz, float[] luUtilities ) { - // - // String record = ( iLuz + "," + jLuz + "," + iMgra + "," + jMgra + "," + - // luUtilities[0] ); - // // don't need to report the last logsum (not used for mode choice - // logsums) - // for( int j=1; j < luUtilities.length - 1; j++ ) - // record += ( "," + luUtilities[j] ); - // outStream.println ( record ); - // - // } - - private void accumulateLandUseModeChoiceLogsums(float[] luUtilitiesValues, - int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) - { - - float[] luUtilities = new float[DcUtilitiesTaskJppf.LU_LOGSUM_SEGMENTS.length]; - - int iMgra = (int) luUtilitiesValues[0]; - int jMgra = (int) luUtilitiesValues[1]; - - int iLuz = mgraManager.getMgraLuz(iMgra); - int jLuz = mgraManager.getMgraLuz(jMgra); - - for (int i = 0; i < luUtilities.length; i++) - luUtilities[i] = luUtilitiesValues[i + 2]; - - accumulatedLandUseLogsumsCount[iLuz][jLuz]++; - - accumulateSimple(iLuz, jLuz, luUtilities, accumulatedLandUseLogsumsCount, - accumulatedLandUseLogsums); - accumulateLogit(iLuz, jLuz, luUtilities, accumulatedLandUseLogsumsCount, - accumulatedLandUseLogsums); - - // if ( iLuz == DEBUG_ILUZ && jLuz == DEBUG_JLUZ ) - // debugLandUseModeChoiceLogsums( iMgra, jMgra, iLuz, jLuz, luUtilities - // ); - - } - - private void accumulateSimple(int iLuz, int jLuz, float[] luUtilities, - int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) - { - - // simple averaging uses accumulated logsum values - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][jLuz] += Math - .log(luUtilities[0]); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][jLuz] += Math - .log(luUtilities[1]); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][jLuz] += Math - .log(luUtilities[2]); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][jLuz] += Math - .log(luUtilities[3]); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][jLuz] += Math - .log(luUtilities[4]); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][jLuz] += Math - .log(luUtilities[5]); - - // calculate logsums from external LUZs to all destination LUZs if the - // origin LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[iLuz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[iLuz]) - { - - double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] - * BuildAccessibilities.TIME_COEFFICIENT); - - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][exLuz][jLuz] += Math - .log(luUtilities[0] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][exLuz][jLuz] += Math - .log(luUtilities[1] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][exLuz][jLuz] += Math - .log(luUtilities[2] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][exLuz][jLuz] += Math - .log(luUtilities[3] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][exLuz][jLuz] += Math - .log(luUtilities[4] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][exLuz][jLuz] += Math - .log(luUtilities[5] + additionalUtility); - - accumulatedLandUseLogsumsCount[exLuz][jLuz]++; - - } - - } - - // calculate logsums to external LUZs from all origin LUZs if the - // destination LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[jLuz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[jLuz]) - { - - double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] - * BuildAccessibilities.TIME_COEFFICIENT); - - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][exLuz] += Math - .log(luUtilities[0] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][exLuz] += Math - .log(luUtilities[1] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][exLuz] += Math - .log(luUtilities[2] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][exLuz] += Math - .log(luUtilities[3] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][exLuz] += Math - .log(luUtilities[4] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.SIMPLE][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][exLuz] += Math - .log(luUtilities[5] + additionalUtility); - - accumulatedLandUseLogsumsCount[iLuz][exLuz]++; - - } - - } - - } - - private void accumulateLogit(int iLuz, int jLuz, float[] luUtilities, - int[][] accumulatedLandUseLogsumsCount, float[][][][][] accumulatedLandUseLogsums) - { - - // logit averaging uses accumulated utility values - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][jLuz] += luUtilities[0]; - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][jLuz] += luUtilities[1]; - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][jLuz] += luUtilities[2]; - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][jLuz] += luUtilities[3]; - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][jLuz] += luUtilities[4]; - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][jLuz] += luUtilities[5]; - - // calculate logsums from external LUZs to all destination LUZs if the - // origin LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[iLuz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[iLuz]) - { - - double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] - * BuildAccessibilities.TIME_COEFFICIENT); - - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][exLuz][jLuz] += (luUtilities[0] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][exLuz][jLuz] += (luUtilities[1] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][exLuz][jLuz] += (luUtilities[2] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][exLuz][jLuz] += (luUtilities[3] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][exLuz][jLuz] += (luUtilities[4] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][exLuz][jLuz] += (luUtilities[5] + additionalUtility); - - accumulatedLandUseLogsumsCount[exLuz][jLuz]++; - - } - - } - - // calculate logsums to external LUZs from all origin LUZs if the - // destination LUZ is a cordon LUZ - if (externalLuzsForCordonLuz[jLuz] != null) - { - - for (int exLuz : externalLuzsForCordonLuz[jLuz]) - { - - double additionalUtility = Math.exp(cordonLuzMinutesForExternalLuz[exLuz] - * BuildAccessibilities.TIME_COEFFICIENT); - - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS0][iLuz][exLuz] += (luUtilities[0] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS1][iLuz][exLuz] += (luUtilities[1] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.PK][BuildAccessibilities.LS2][iLuz][exLuz] += (luUtilities[2] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS0][iLuz][exLuz] += (luUtilities[3] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS1][iLuz][exLuz] += (luUtilities[4] + additionalUtility); - accumulatedLandUseLogsums[BuildAccessibilities.LOGIT][BuildAccessibilities.OP][BuildAccessibilities.LS2][iLuz][exLuz] += (luUtilities[5] + additionalUtility); - - accumulatedLandUseLogsumsCount[iLuz][exLuz]++; - - } - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java deleted file mode 100644 index d28d05c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/DriveTransitWalkSkimsCalculator.java +++ /dev/null @@ -1,313 +0,0 @@ -package org.sandag.abm.accessibilities; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import java.io.File; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; - -/** - * This class is used to return drive-transit-walk skim values for MGRA pairs - * associated with estimation data file records. - * - * @author Jim Hicks - * @version March, 2010 - */ -public class DriveTransitWalkSkimsCalculator - implements Serializable -{ - - private transient Logger primaryLogger; - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; - - private static final int ACCESS_TIME_INDEX = 0; - private static final int EGRESS_TIME_INDEX = 1; - private static final int NA = -999; - - private int maxDTWSkimSets = 5; - private int[] NUM_SKIMS; - private double[] defaultSkims; - - // declare UEC object - private UtilityExpressionCalculator driveWalkSkimUEC; - private IndexValues iv; - - // The simple auto skims UEC does not use any DMU variables - private TransitDriveAccessDMU dmu = new TransitDriveAccessDMU(); // DMU - // for - // this - // UEC - - private MgraDataManager mgraManager; - private int maxTap; - private String[] skimNames; - - - // skim values array for transit service type(local, premium), - // depart skim period(am, pm, op), - // and Tap-Tap pair. - private double[][][][][] storedDepartPeriodTapTapSkims; - - private BestTransitPathCalculator bestPathUEC; - - private MatrixDataServerIf ms; - - public DriveTransitWalkSkimsCalculator(HashMap rbMap) - { - mgraManager = MgraDataManager.getInstance(); - maxTap = mgraManager.getMaxTap(); - } - - public void setup(HashMap rbMap, Logger logger, - BestTransitPathCalculator myBestPathUEC) - { - - primaryLogger = logger; - - // set the best transit path utility UECs - bestPathUEC = myBestPathUEC; - - // Create the skim UECs - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.drive.transit.walk.data.page"); - int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.drive.transit.walk.skim.page"); - int dtwNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.drive.transit.walk.skims"); - String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.drive.transit.walk.uec.file")).toString(); - File uecFile = new File(uecFileName); - driveWalkSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); - - skimNames = driveWalkSkimUEC.getAlternativeNames(); - - //setup index values - iv = new IndexValues(); - - //setup default skim values - defaultSkims = new double[dtwNumSkims]; - for (int j = 0; j < dtwNumSkims; j++) { - defaultSkims[j] = NA; - } - - // point the stored Array of skims: by Prem or Local, DepartPeriod, O tap, D tap, skim values[] to a shared data store - StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxDTWSkimSets, NUM_PERIODS, maxTap ); - storedDepartPeriodTapTapSkims = storedDataObject.getStoredDtwDepartPeriodTapTapSkims(); - - } - - - - /** - * Return the array of drive-transit-walk skims for the ride mode, origin TAP, - * destination TAP, and departure time period. - * - * @param set for set source skims - * @param origTap best Origin TAP for the MGRA pair - * @param workTap best Destination TAP for the MGRA pair - * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 = - * OffPeak period - * @return Array of 55 skim values for the MGRA pair and departure period - */ - public double[] getDriveTransitWalkSkims(int set, double pDriveTime, double aWalkTime, int origTap, int destTap, - int departPeriod, boolean debug) - { - - dmu.setDriveTimeToTap(pDriveTime); - dmu.setMgraTapWalkTime(aWalkTime); - - iv.setOriginZone(origTap); - iv.setDestZone(destTap); - - // allocate space for the origin tap if it hasn't been allocated already - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) - { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; - } - - // if the destTap skims are not already stored, calculate them and store - // them - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) - { - dmu.setTOD(departPeriod); - dmu.setSet(set); - double[] results = driveWalkSkimUEC.solve(iv, dmu, null); - if (debug) - driveWalkSkimUEC.logAnswersArray(primaryLogger, "Drive-Walk Skims"); - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pDriveTime; - } - catch ( Exception e ) { - primaryLogger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pDriveTime=" + pDriveTime); - primaryLogger.error ("exception setting drive-transit-walk drive access time in stored array.", e); - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aWalkTime; - } - catch ( Exception e ) { - primaryLogger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aWalkTime=" + aWalkTime); - primaryLogger.error ("exception setting drive-transit-walk walk egress time in stored array.", e); - } - return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; - - - } - - - public double[] getNullTransitSkims() - { - return defaultSkims; - } - - /** - * Start the matrix server - * - * @param rb is a ResourceBundle for the properties file for this application - */ - private void startMatrixServer(ResourceBundle rb) - { - - primaryLogger.info(""); - primaryLogger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - primaryLogger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - primaryLogger.error(String - .format("exception caught running ctramp model components -- exiting."), e); - throw new RuntimeException(); - - } - - } - - /** - * log a report of the final skim values for the MGRA odt - * - * @param odt is an int[] with the first element the origin mgra and the second - * element the dest mgra and third element the departure period index - * @param bestTapPairs is an int[][] of TAP values with the first dimesion the - * ride mode and second dimension a 2 element array with best orig and - * dest TAP - * @param returnedSkims is a double[][] of skim values with the first dimesion - * the ride mode indices and second dimention the skim categories - */ - public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) - { - - Modes.TransitMode[] mode = Modes.TransitMode.values(); - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String separator = ""; - String header = ""; - - primaryLogger.info(""); - primaryLogger.info(""); - header = "Returned drive-transit-walk skim value tables for origMgra=" + odt[0] - + ", destMgra=" + odt[1] + ", period index=" + odt[2] + ", period label=" - + PERIODS[odt[2]]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - primaryLogger.info(separator); - primaryLogger.info(header); - primaryLogger.info(""); - - String modeHeading = String.format("%-12s %3s ", "RideMode:", mode[0]); - for (int i = 1; i < bestTapPairs.length; i++) - modeHeading += String.format(" %3s ", mode[i]); - primaryLogger.info(modeHeading); - - String[] logValues = {"NA", "NA"}; - if (bestTapPairs[0] != null) - { - logValues[0] = String.valueOf(bestTapPairs[0][0]); - logValues[1] = String.valueOf(bestTapPairs[0][1]); - } - String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", logValues[0], - logValues[1]); - - for (int i = 1; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] != null) - { - logValues[0] = String.valueOf(bestTapPairs[i][0]); - logValues[1] = String.valueOf(bestTapPairs[i][1]); - } else - { - logValues[0] = "NA"; - logValues[1] = "NA"; - } - tapHeading += String.format(" %4s-%4s ", logValues[0], logValues[1]); - } - primaryLogger.info(tapHeading); - - String underLine = String.format("%-12s %9s ", "---------", "---------"); - for (int i = 1; i < bestTapPairs.length; i++) - underLine += String.format(" %9s ", "---------"); - primaryLogger.info(underLine); - - for (int j = 0; j < ncols; j++) - { - String tableRecord = ""; - if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, - skims[0][j]); - else tableRecord = String.format("%-12d %12s ", j + 1, ""); - for (int i = 1; i < bestTapPairs.length; i++) - { - if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); - else tableRecord += String.format(" %12s ", ""); - } - primaryLogger.info(tableRecord); - } - - primaryLogger.info(""); - primaryLogger.info(separator); - } - - public String[] getSkimNames() { - return skimNames; - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java deleted file mode 100644 index c158ad1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesCalculator.java +++ /dev/null @@ -1,593 +0,0 @@ -package org.sandag.abm.accessibilities; - -import com.pb.common.util.Tracer; -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import java.io.File; -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; -import org.sandag.abm.modechoice.Modes; - -/** - * This class builds accessibility components for all modes. - * - * @author Joel Freedman - * @version May, 2009 - */ -public class MandatoryAccessibilitiesCalculator - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(MandatoryAccessibilitiesCalculator.class); - - private static final int MIN_EXP_FUNCTION_ARGUMENT = -500; - - private static final int PEAK_NONTOLL_SOV_TIME_INDEX = 0; - private static final int PEAK_NONTOLL_SOV_DIST_INDEX = 1; - private static final int OFFPEAK_NONTOLL_SOV_TIME_INDEX = 2; - private static final int OFFPEAK_NONTOLL_SOV_DIST_INDEX = 3; - - private UtilityExpressionCalculator autoSkimUEC; - private UtilityExpressionCalculator bestWalkTransitUEC; - private UtilityExpressionCalculator bestDriveTransitUEC; - private UtilityExpressionCalculator autoLogsumUEC; - private UtilityExpressionCalculator transitLogsumUEC; - - private MandatoryAccessibilitiesDMU dmu; - private IndexValues iv; - - private NonTransitUtilities ntUtilities; - - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - - // auto sufficiency (0 autos, autos=adults), - // and mode (SOV,HOV,Walk-Transit,Non-Motorized) - private double[][] expConstants; - - private String[] accNames = { - "SovTime", // 0 - "SovDist", // 1 - "WTTime", // 2 - "DTTime", // 3 - "SovUtility", // 4 - "WTUtility", // 5 - "AutoLogsum", // 6 - "WTLogsum", // 7 - "TransitLogsum", // 8 - "WTRailShare", // 9 - "DTRailShare", // 10 - "DTLogsum", // 11 - "HovUtility" // 12 - }; - - private BestTransitPathCalculator bestPathCalculator; - - - public MandatoryAccessibilitiesCalculator(HashMap rbMap, - NonTransitUtilities aNtUtilities, double[][] aExpConstants, BestTransitPathCalculator myBestPathCalculator) - { - - ntUtilities = aNtUtilities; - expConstants = aExpConstants; - - // Create the UECs - String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.mandatory.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.data.page"); - int autoSkimPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.auto.page"); - int bestWalkTransitPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.bestWalkTransit.page"); - int bestDriveTransitPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.bestDriveTransit.page"); - int autoLogsumPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.autoLogsum.page"); - int transitLogsumPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.mandatory.transitLogsum.page"); - - - dmu = new MandatoryAccessibilitiesDMU(); - - File uecFile = new File(uecFileName); - autoSkimUEC = new UtilityExpressionCalculator(uecFile, autoSkimPage, dataPage, rbMap, dmu); - bestWalkTransitUEC = new UtilityExpressionCalculator(uecFile, bestWalkTransitPage, dataPage, rbMap, dmu); - bestDriveTransitUEC = new UtilityExpressionCalculator(uecFile, bestDriveTransitPage, dataPage, rbMap, dmu); - autoLogsumUEC = new UtilityExpressionCalculator(uecFile, autoLogsumPage, dataPage, rbMap, dmu); - transitLogsumUEC = new UtilityExpressionCalculator(uecFile, transitLogsumPage, dataPage, rbMap, dmu); - - iv = new IndexValues(); - - tazManager = TazDataManager.getInstance(); - tapManager = TapDataManager.getInstance(); - mgraManager = MgraDataManager.getInstance(); - - bestPathCalculator = myBestPathCalculator; - } - - - public double[] calculateWorkerMandatoryAccessibilities(int hhMgra, int workMgra) - { - return calculateAccessibilitiesForMgraPair(hhMgra, workMgra, false, null); - } - - public double[] calculateStudentMandatoryAccessibilities(int hhMgra, int schoolMgra) - { - return calculateAccessibilitiesForMgraPair(hhMgra, schoolMgra, false, null); - } - - /** - * Calculate the work logsum for the household MGRA and sampled work location - * MGRA. - * - * @param hhMgra Household MGRA - * @param workMgra Sampled work MGRA - * @param autoSufficiency Auto sufficiency category - * @return Work mode choice logsum - */ - public double calculateWorkLogsum(int hhMgra, int workMgra, int autoSufficiency, boolean debug, Logger aLogger) - { - - String separator = ""; - String header = ""; - if (debug) - { - aLogger.info(""); - aLogger.info(""); - header = "calculateWorkLogsum() debug info for homeMgra=" + hhMgra + ", workMgra=" + workMgra; - for (int i = 0; i < header.length(); i++) - separator += "^"; - } - - double[] accessibilities = calculateAccessibilitiesForMgraPair(hhMgra, workMgra, debug, aLogger); - - double sovUtility = accessibilities[4]; - double hovUtility = accessibilities[12]; - double transitLogsum = accessibilities[8]; // includes both walk and drive access - double nmExpUtility = ntUtilities.getNMotorExpUtility(hhMgra, workMgra, NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - - // constrain auto sufficiency to 0,1,2 - autoSufficiency = Math.min(autoSufficiency, 2); - - double utilSum = Math.exp(sovUtility) * expConstants[autoSufficiency][0] - + Math.exp(hovUtility) * expConstants[autoSufficiency][1] - + Math.exp(transitLogsum) * expConstants[autoSufficiency][2] - + nmExpUtility * expConstants[autoSufficiency][3]; - - double logsum = Math.log(utilSum); - - if (debug) - { - - aLogger.info(separator); - aLogger.info(header); - aLogger.info(separator); - - aLogger.info("accessibilities array values"); - aLogger.info(String.format("%5s %15s %15s", "i", "accName", "value")); - aLogger.info(String.format("%5s %15s %15s", "-----", "----------", "----------")); - for (int i = 0; i < accessibilities.length; i++) - { - aLogger.info(String.format("%5d %15s %15.5e", i, accNames[i], accessibilities[i])); - } - - aLogger.info(""); - aLogger.info(""); - aLogger.info("logsum component values"); - aLogger.info(String.format("autoSufficiency = %d", autoSufficiency)); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "sovUtility", sovUtility, "exp(sovUtility)", Math.exp(sovUtility), String - .format("expConst suff=%d 0", autoSufficiency), - expConstants[autoSufficiency][0])); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "hovUtility", hovUtility, "exp(hovUtility)", Math.exp(hovUtility), String - .format("expConst suff=%d 1", autoSufficiency), - expConstants[autoSufficiency][1])); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "transitLogsum", transitLogsum, "exp(transitLogsum)", Math.exp(transitLogsum), - String.format("expConst suff=%d 2", autoSufficiency), - expConstants[autoSufficiency][2])); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e", "nmExpUtility", - nmExpUtility, String.format("expConst suff=%d 3", autoSufficiency), - expConstants[autoSufficiency][3])); - aLogger.info(String.format("%-15s = %15.5e", "utilSum", utilSum)); - aLogger.info(String.format("%-15s = %15.5e", "logsum", logsum)); - aLogger.info(separator); - } - - return logsum; - } - - /** - * Calculate the school logsum for the household MGRA and sampled school location - * MGRA. - * - * @param hhMgra Household MGRA - * @param schoolMgra Sampled work MGRA - * @param autoSufficiency Auto sufficiency category - * @param studentType Student type 0=Pre-school (SOV not available) 1=K-8 (SOV - * not available) 2=9-12 (Normal car-sufficiency-based logsum) - * 3=College/university(typical) (Normal car-sufficiency-based logsum) - * 4=College/university(non-typical) (Normal car-sufficiency-based - * logsum) - * @return School mode choice logsum - */ - public double calculateSchoolLogsum(int hhMgra, int schoolMgra, int autoSufficiency, int studentType, boolean debug, Logger aLogger) - { - - String separator = ""; - String header = ""; - if (debug) - { - aLogger.info(""); - aLogger.info(""); - header = "calculateSchoolLogsum() debug info for homeMgra=" + hhMgra + ", schoolMgra=" - + schoolMgra; - for (int i = 0; i < header.length(); i++) - separator += "^"; - } - - double[] accessibilities = calculateAccessibilitiesForMgraPair(hhMgra, schoolMgra, debug, aLogger); - - double sovUtility = accessibilities[4]; - double hovUtility = accessibilities[12]; - double transitLogsum = accessibilities[8]; // includes both walk and drive access - double nmExpUtility = ntUtilities.getNMotorExpUtility(hhMgra, schoolMgra, NonTransitUtilities.OFFPEAK_PERIOD_INDEX); - - // constrain auto sufficiency to 0,1,2 - autoSufficiency = Math.min(autoSufficiency, 2); - - double logsum = Math.exp(hovUtility) * expConstants[autoSufficiency][1] - + Math.exp(transitLogsum) * expConstants[autoSufficiency][2] - + nmExpUtility * expConstants[autoSufficiency][3]; - - // used for debugging - double logsum1 = logsum; - - if (studentType >= 2) - { - logsum = logsum + Math.exp(sovUtility) * expConstants[autoSufficiency][0]; - } - - // used for debugging - double logsum2 = logsum; - - logsum = Math.log(logsum); - - if (debug) - { - - aLogger.info(separator); - aLogger.info(header); - aLogger.info(separator); - - aLogger.info("accessibilities array values"); - aLogger.info(String.format("%5s %15s %15s", "i", "accName", "value")); - aLogger.info(String.format("%5s %15s %15s", "-----", "----------", "----------")); - for (int i = 0; i < accessibilities.length; i++) - { - aLogger.info(String.format("%5d %15s %15.5e", i, accNames[i], accessibilities[i])); - } - - aLogger.info(""); - aLogger.info(""); - aLogger.info("logsum component values"); - aLogger.info(String.format("autoSufficiency = %d", autoSufficiency)); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "hovUtility", hovUtility, "exp(hovUtility)", Math.exp(hovUtility), - String.format("expConst suff=%d 1", autoSufficiency), - expConstants[autoSufficiency][1])); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "transitLogsum", transitLogsum, "exp(transitLogsum)", Math.exp(transitLogsum), - String.format("expConst suff=%d 2", autoSufficiency), - expConstants[autoSufficiency][2])); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e", "nmExpUtility", - nmExpUtility, String.format("expConst suff=%d 3", autoSufficiency), - expConstants[autoSufficiency][3])); - aLogger.info(String.format("%s = %15.5e", "utility sum (before adding sovUtility)", - logsum1)); - if (studentType >= 2) - { - aLogger.info(String.format("studentType = %d", studentType)); - aLogger.info(String.format("%-15s = %15.5e, %-18s = %15.5e, %-18s = %15.5e", - "sovUtility", sovUtility, "exp(sovUtility)", Math.exp(sovUtility), String.format("expConst suff=%d 0", autoSufficiency), - expConstants[autoSufficiency][0])); - aLogger.info(String.format("%s = %15.5e", "utility sum (after adding sovUtility)", logsum2)); - } else - { - aLogger.info(String.format( - "studentType = %d, no additional contribution to utility sum", - studentType)); - } - aLogger.info(String.format("%s = %15.5e, %s = %15.5e", "final utility sum", logsum2, - "final logsum", logsum)); - aLogger.info(separator); - } - - return logsum; - } - - /** - * Calculate the accessibilities for a given origin and destination mgra - * - * @param oMgra The origin mgra - * @param dMgra The destination mgra - * @return An array of accessibilities - */ - public double[] calculateAccessibilitiesForMgraPair(int oMgra, int dMgra, boolean debug, Logger aLogger) - { - - double[] accessibilities = new double[accNames.length]; - - // DMUs for this UEC - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - if (oMgra > 0 && dMgra > 0) - { - - int oTaz = mgraManager.getTaz(oMgra); - int dTaz = mgraManager.getTaz(dMgra); - - iv.setOriginZone(oTaz); - iv.setDestZone(dTaz); - - // sov time and distance - double[] autoResults = autoSkimUEC.solve(iv, dmu, null); - if (debug) - autoSkimUEC.logAnswersArray( aLogger, String.format( "autoSkimUEC: oMgra=%d, dMgra=%d", oMgra, dMgra ) ); - - // autoResults[0] is peak non-toll sov time, autoResults[1] is peak non-toll sov dist - // autoResults[2] is off-peak non-toll sov time, autoResults[3] is off-peak non-toll sov dist - accessibilities[0] = autoResults[PEAK_NONTOLL_SOV_TIME_INDEX]; - accessibilities[1] = autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]; - - // pre-calculate the hov, sov, and non-motorized exponentiated utilities for the origin MGRA. - // the method called returns cached values if they were already calculated. - ntUtilities.buildUtilitiesForOrigMgraAndPeriod( oMgra, NonTransitUtilities.PEAK_PERIOD_INDEX ); - - // auto logsum - double pkSovExpUtility = ntUtilities.getSovExpUtility(oTaz, dTaz, NonTransitUtilities.PEAK_PERIOD_INDEX); - double pkHovExpUtility = ntUtilities.getHovExpUtility(oTaz, dTaz, NonTransitUtilities.PEAK_PERIOD_INDEX); - - dmu.setSovNestLogsum(-999); - if (pkSovExpUtility > 0) - { - dmu.setSovNestLogsum(Math.log(pkSovExpUtility)); - accessibilities[4] = dmu.getSovNestLogsum(); - } - dmu.setHovNestLogsum(-999); - if (pkHovExpUtility > 0) - { - dmu.setHovNestLogsum(Math.log(pkHovExpUtility)); - accessibilities[12] = dmu.getHovNestLogsum(); - } - - double[] autoLogsum = autoLogsumUEC.solve(iv, dmu, null); - if (debug) - autoLogsumUEC.logAnswersArray(aLogger, String.format( - "autoLogsumUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); - accessibilities[6] = autoLogsum[0]; - - - ////////////////////////////////////////////////////////////////////////// - // walk transit - ////////////////////////////////////////////////////////////////////////// - - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestWalkTransitWalkTaps(walkDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, oMgra, dMgra, debug, aLogger); - - // sum the exponentiated utilities over modes - double sumWlkExpUtilities = bestPathCalculator.getSumExpUtilities(); - double[] walkTransitWalkUtilities = bestPathCalculator.getBestUtilities(); - - // calculate ln( sum of exponentiated utilities ) and set in accessibilities array and the dmu object - if (sumWlkExpUtilities > 0) - accessibilities[7] = Math.log(sumWlkExpUtilities); - else - accessibilities[7] = -999; - - dmu.setWlkNestLogsum(accessibilities[7]); - - - int bestAlt = bestPathCalculator.getBestTransitAlt(); - - if (bestAlt >= 0) - { - double[] bestTaps = bestPathCalculator.getBestTaps(bestAlt); - int oTapPosition = mgraManager.getTapPosition(oMgra, (int)bestTaps[0]); - int dTapPosition = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); - int set = (int)bestTaps[2]; - - if (oTapPosition == -1 || dTapPosition == -1) - { - logger.fatal("Error: Best walk transit alt " + bestAlt + " found for origin mgra " - + oMgra + " to destination mgra " + dMgra + " but oTap pos " - + oTapPosition + " and dTap pos " + dTapPosition); - throw new RuntimeException(); - } - - if (walkTransitWalkUtilities[bestAlt] <= MIN_EXP_FUNCTION_ARGUMENT) - { - logger.fatal("Error: Best walk transit alt " + bestAlt + " found for origin mgra " - + oMgra + " to destination mgra " + dMgra + " but Utility = " - + walkTransitWalkUtilities[bestAlt]); - throw new RuntimeException(); - } - accessibilities[5] = Math.log(walkTransitWalkUtilities[bestAlt]); - - //set access and egress times - int oPos = mgraManager.getTapPosition(oMgra, (int)bestTaps[0]); - float mgraTapWalkTime = mgraManager.getMgraToTapWalkTime(oMgra, oPos); - dmu.setMgraTapWalkTime(mgraTapWalkTime); - - int dPos = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); - float tapMgraWalkTime = mgraManager.getMgraToTapWalkTime(dMgra, dPos); - dmu.setTapMgraWalkTime(tapMgraWalkTime); - - dmu.setBestSet(set); - iv.setOriginZone((int)bestTaps[0]); - iv.setDestZone((int)bestTaps[1]); - double[] wlkTransitTimes = bestWalkTransitUEC.solve(iv, dmu, null); - - if (debug){ - bestWalkTransitUEC.logAnswersArray(aLogger, String.format("bestWalkTransitUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); - } - - accessibilities[2] = wlkTransitTimes[0]; - accessibilities[9] = wlkTransitTimes[1]; - - } - - ////////////////////////////////////////////////////////////////////////// - // drive transit - ////////////////////////////////////////////////////////////////////////// - - // determine the best transit path, which also stores the best utilities array and the best mode - bestPathCalculator.findBestDriveTransitWalkTaps(walkDmu, driveDmu, ModelStructure.AM_SKIM_PERIOD_INDEX, oMgra, dMgra, debug, aLogger, (float) autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]); - - // sum the exponentiated utilities over modes - double sumDrvExpUtilities = 0; - double[] driveTransitWalkUtilities = bestPathCalculator.getBestUtilities(); - for (int i=0; i < driveTransitWalkUtilities.length; i++){ - if ( driveTransitWalkUtilities[i] > MIN_EXP_FUNCTION_ARGUMENT ) - sumDrvExpUtilities += Math.exp(driveTransitWalkUtilities[i]); - } - - - // calculate ln( sum of exponentiated utilities ) and set in accessibilities array and the dmu object - if (sumDrvExpUtilities > 0) - accessibilities[11] = Math.log(sumDrvExpUtilities); - else - accessibilities[11] = -999; - - dmu.setDrvNestLogsum(accessibilities[11]); - - - bestAlt = bestPathCalculator.getBestTransitAlt(); - - if (bestAlt >= 0) - { - double[] bestTaps = bestPathCalculator.getBestTaps(bestAlt); - int oTapPosition = tazManager.getTapPosition(oTaz, (int)bestTaps[0], Modes.AccessMode.PARK_N_RIDE); - int dTapPosition = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); - int set = (int)bestTaps[2]; - - if (oTapPosition == -1 || dTapPosition == -1) - { - logger.fatal("Error: Best drive transit alt " + bestAlt + " found for origin mgra " - + oMgra + " to destination mgra " + dMgra + " but oTap pos " - + oTapPosition + " and dTap pos " + dTapPosition); - throw new RuntimeException(); - } - - if (driveTransitWalkUtilities[bestAlt] <= MIN_EXP_FUNCTION_ARGUMENT) - { - logger.fatal("Error: Best drive transit alt " + bestAlt + " found for origin mgra " - + oMgra + " to destination mgra " + dMgra + " but Utility = " - + driveTransitWalkUtilities[bestAlt]); - throw new RuntimeException(); - } - - //set access and egress times - dmu.setDriveTimeToTap(tazManager.getTapTime(oTaz, oTapPosition, Modes.AccessMode.PARK_N_RIDE)); - dmu.setDriveDistToTap(tazManager.getTapDist(oTaz, oTapPosition, Modes.AccessMode.PARK_N_RIDE)); - - int dPos = mgraManager.getTapPosition(dMgra, (int)bestTaps[1]); - float tapMgraWalkTime = mgraManager.getMgraToTapWalkTime(dMgra, dPos); - dmu.setTapMgraWalkTime(tapMgraWalkTime); - - dmu.setBestSet(set); - iv.setOriginZone((int)bestTaps[0]); - iv.setDestZone((int)bestTaps[1]); - double[] drvTransitTimes = bestDriveTransitUEC.solve(iv, dmu, null); - - if (debug){ - bestDriveTransitUEC.logAnswersArray(aLogger, String.format("bestDriveTransitUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); - } - - accessibilities[3] = drvTransitTimes[0]; - accessibilities[10] = drvTransitTimes[1]; - - } - - - double[] transitLogsumResults = transitLogsumUEC.solve(iv, dmu, null); - if (debug){ - transitLogsumUEC.logAnswersArray(aLogger, String.format("transitLogsumUEC: oMgra=%d, dMgra=%d", oMgra, dMgra)); - } - - // transit logsum results array has only 1 alternative, so result is in 0 element. - accessibilities[8] = transitLogsumResults[0]; - - } // end if oMgra and dMgra > 0 - - return accessibilities; - } - - /** - * Calculate auto skims for a given origin to all destination mgras, and return - * auto distance. - * - * @param oMgra The origin mgra - * @return An array of distances - */ - public double[] calculateDistancesForAllMgras(int oMgra) - { - - double[] distances = new double[mgraManager.getMaxMgra() + 1]; - - int oTaz = mgraManager.getTaz(oMgra); - iv.setOriginZone(oTaz); - - for (int i = 0; i < mgraManager.getMgras().size(); i++) - { - - int dTaz = mgraManager.getTaz(mgraManager.getMgras().get(i)); - iv.setDestZone(dTaz); - - // sov distance - double[] autoResults = autoSkimUEC.solve(iv, dmu, null); - distances[dTaz] = autoResults[PEAK_NONTOLL_SOV_DIST_INDEX]; - - } - - return distances; - } - - /** - * Calculate auto skims for a given origin to all destination mgras, and return - * auto distance. - * - * @param oMgra The origin mgra - * @return An array of distances - */ - public double[] calculateOffPeakDistancesForAllMgras(int oMgra) - { - - double[] distances = new double[mgraManager.getMaxMgra() + 1]; - - int oTaz = mgraManager.getTaz(oMgra); - iv.setOriginZone(oTaz); - - for (int i = 0; i < mgraManager.getMgras().size(); i++) - { - - int dTaz = mgraManager.getTaz(mgraManager.getMgras().get(i)); - iv.setDestZone(dTaz); - - // sov distance - double[] autoResults = autoSkimUEC.solve(iv, dmu, null); - distances[dTaz] = autoResults[OFFPEAK_NONTOLL_SOV_DIST_INDEX]; - - } - - return distances; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java deleted file mode 100644 index cc68cf4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/MandatoryAccessibilitiesDMU.java +++ /dev/null @@ -1,206 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -public class MandatoryAccessibilitiesDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(MandatoryAccessibilitiesDMU.class); - - protected HashMap methodIndexMap; - - protected double sovNestLogsum; - protected double hovNestLogsum; - protected double wlkNestLogsum; - protected double drvNestLogsum; - protected int bestSet; - protected float mgraTapWalkTime; - protected float tapMgraWalkTime; - protected float driveDistToTap; - protected float driveTimeToTap; - protected int autoSufficiency; - - public MandatoryAccessibilitiesDMU() - { - setupMethodIndexMap(); - } - - public int getAutoSufficiency() - { - return autoSufficiency; - } - - public void setAutoSufficiency(int autoSufficiency) - { - this.autoSufficiency = autoSufficiency; - } - - public double getSovNestLogsum() - { - return sovNestLogsum; - } - - public void setSovNestLogsum(double sovNestLogsum) - { - this.sovNestLogsum = sovNestLogsum; - } - - public double getHovNestLogsum() - { - return hovNestLogsum; - } - - public void setHovNestLogsum(double hovNestLogsum) - { - this.hovNestLogsum = hovNestLogsum; - } - - public double getWlkNestLogsum() - { - return wlkNestLogsum; - } - - public void setWlkNestLogsum(double wlkNestLogsum) - { - this.wlkNestLogsum = wlkNestLogsum; - } - - public double getDrvNestLogsum() - { - return drvNestLogsum; - } - - public void setDrvNestLogsum(double drvNestLogsum) - { - this.drvNestLogsum = drvNestLogsum; - } - - public int getBestSet() - { - return bestSet; - } - - public void setBestSet(int bestSet) - { - this.bestSet = bestSet; - } - - public float getMgraTapWalkTime() - { - return mgraTapWalkTime; - } - - public void setMgraTapWalkTime(float mgraTapWalkTime) - { - this.mgraTapWalkTime = mgraTapWalkTime; - } - - public float getTapMgraWalkTime() - { - return tapMgraWalkTime; - } - - public void setTapMgraWalkTime(float tapMgraWalkTime) - { - this.tapMgraWalkTime = tapMgraWalkTime; - } - - public float getDriveDistToTap() - { - return driveDistToTap; - } - - public void setDriveDistToTap(float drvDistToTap) - { - this.driveDistToTap = drvDistToTap; - } - - public float getDriveTimeToTap() - { - return driveTimeToTap; - } - - public void setDriveTimeToTap(float drvTimeToTap) - { - this.driveTimeToTap = drvTimeToTap; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getBestSet", 0); - methodIndexMap.put("getDriveDistToTap", 1); - methodIndexMap.put("getDriveTimeToTap", 2); - methodIndexMap.put("getWlkNestLogsum", 3); - methodIndexMap.put("getDrvNestLogsum", 4); - methodIndexMap.put("getSovNestLogsum", 5); - methodIndexMap.put("getHovNestLogsum", 6); - methodIndexMap.put("getMgraTapWalkTime", 7); - methodIndexMap.put("getTapMgraWalkTime", 8); - methodIndexMap.put("getAutoSufficiency", 9); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getBestSet(); - case 1: - return getDriveDistToTap(); - case 2: - return getDriveTimeToTap(); - case 3: - return getWlkNestLogsum(); - case 4: - return getDrvNestLogsum(); - case 5: - return getSovNestLogsum(); - case 6: - return getHovNestLogsum(); - case 7: - return getMgraTapWalkTime(); - case 8: - return getTapMgraWalkTime(); - case 9: - return getAutoSufficiency(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java deleted file mode 100644 index f2af9c9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/McLogsumsAppender.java +++ /dev/null @@ -1,980 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagAppendMcLogsumDMU; -import org.sandag.abm.application.SandagTripModeChoiceDMU; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class McLogsumsAppender - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(McLogsumsAppender.class); - protected transient Logger slcSoaLogger = Logger.getLogger("slcSoa"); - protected transient Logger nonManLogsumsLogger = Logger.getLogger("nonManLogsums"); - - protected int debugEstimationFileRecord1; - protected int debugEstimationFileRecord2; - - protected static final String ESTIMATION_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; - public static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; - public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; - - public static final int WORK_SHEET = 1; - public static final int UNIVERSITY_SHEET = 2; - public static final int SCHOOL_SHEET = 3; - public static final int MAINTENANCE_SHEET = 4; - public static final int DISCRETIONARY_SHEET = 5; - public static final int SUBTOUR_SHEET = 6; - public static final int[] MC_PURPOSE_SHEET_INDICES = {-1, WORK_SHEET, - UNIVERSITY_SHEET, SCHOOL_SHEET, MAINTENANCE_SHEET, MAINTENANCE_SHEET, - MAINTENANCE_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, - SUBTOUR_SHEET }; - - public static final int WORK_CATEGORY = 0; - public static final int UNIVERSITY_CATEGORY = 1; - public static final int SCHOOL_CATEGORY = 2; - public static final int MAINTENANCE_CATEGORY = 3; - public static final int DISCRETIONARY_CATEGORY = 4; - public static final int SUBTOUR_CATEGORY = 5; - public static final String[] PURPOSE_CATEGORY_LABELS = {"work", "university", - "school", "maintenance", "discretionary", "subtour" }; - public static final int[] PURPOSE_CATEGORIES = {-1, WORK_CATEGORY, - UNIVERSITY_CATEGORY, SCHOOL_CATEGORY, MAINTENANCE_CATEGORY, MAINTENANCE_CATEGORY, - MAINTENANCE_CATEGORY, DISCRETIONARY_CATEGORY, DISCRETIONARY_CATEGORY, - DISCRETIONARY_CATEGORY, SUBTOUR_CATEGORY }; - - protected static final int ORIG_MGRA = 1; - protected static final int DEST_MGRA = 2; - protected static final int ADULTS = 3; - protected static final int AUTOS = 4; - protected static final int HHSIZE = 5; - protected static final int FEMALE = 6; - protected static final int AGE = 7; - protected static final int JOINT = 8; - protected static final int PARTYSIZE = 9; - protected static final int TOUR_PURPOSE = 10; - protected static final int INCOME = 11; - protected static final int ESCORT = 12; - protected static final int DEPART_PERIOD = 13; - protected static final int ARRIVE_PERIOD = 14; - protected static final int SAMPNO = 15; - protected static final int WORK_TOUR_MODE = 16; - protected static final int OUT_STOPS = 17; - protected static final int IN_STOPS = 18; - protected static final int FIRST_TRIP = 19; - protected static final int LAST_TRIP = 20; - protected static final int TOUR_MODE = 21; - protected static final int TRIP_PERIOD = 22; - protected static final int CHOSEN_MGRA = 23; - protected static final int DIRECTION = 24; - protected static final int PORTION = 25; - protected static final int PERNO = 26; - protected static final int TOUR_ID = 27; - protected static final int TRIPNO = 28; - protected static final int STOPID = 29; - protected static final int STOPNO = 30; - protected static final int STOP_PURPOSE = 31; - protected static final int NUM_FIELDS = 32; - - private static final int INBOUND_DIRCETION_CODE = 2; - - // estimation file defines time periods as: - // 1 | Early AM: 3:00 AM - 5:59 AM | - // 2 | AM Peak: 6:00 AM - 8:59 AM | - // 3 | Early MD: 9:00 AM - 11:59 PM | - // 4 | Late MD: 12:00 PM - 3:29 PM | - // 5 | PM Peak: 3:30 PM - 6:59 PM | - // 6 | Evening: 7:00 PM - 2:59 AM | - - protected static final int LAST_EA_INDEX = 3; - protected static final int LAST_AM_INDEX = 9; - protected static final int LAST_MD_INDEX = 22; - protected static final int LAST_PM_INDEX = 29; - - protected static final int EA = 1; - protected static final int AM = 2; - protected static final int MD = 3; - protected static final int PM = 4; - protected static final int EV = 5; - - protected static final int EA_D = 1; // 5am - protected static final int AM_D = 5; // 7am - protected static final int MD_D = 15; // 12pm - protected static final int PM_D = 27; // 6pm - protected static final int EV_D = 35; // 10pm - protected static final int[] DEFAULT_DEPART_INDICES = {-1, EA_D, AM_D, MD_D, - PM_D, EV_D }; - - protected static final int EA_A = 2; // 5:30am - protected static final int AM_A = 6; // 7:30am - protected static final int MD_A = 16; // 12:30pm - protected static final int PM_A = 28; // 6:30pm - protected static final int EV_A = 36; // 10:30pm - protected static final int[] DEFAULT_ARRIVE_INDICES = {-1, EA_A, AM_A, MD_A, - PM_A, EV_A }; - - protected String[][] departArriveCombinationLabels = { {"EA", "EA"}, - {"EA", "AM"}, {"EA", "MD"}, {"EA", "PM"}, {"EA", "EV"}, {"AM", "AM"}, {"AM", "MD"}, - {"AM", "PM"}, {"AM", "EV"}, {"MD", "MD"}, {"MD", "PM"}, {"MD", "EV"}, {"PM", "PM"}, - {"PM", "EV"}, {"EV", "EV"} }; - - protected int[][] departArriveCombinations = { {EA, EA}, {EA, AM}, - {EA, MD}, {EA, PM}, {EA, EV}, {AM, AM}, {AM, MD}, {AM, PM}, {AM, EV}, {MD, MD}, - {MD, PM}, {MD, EV}, {PM, PM}, {PM, EV}, {EV, EV} }; - - private BestTransitPathCalculator bestPathUEC; - - // modeChoiceLogsums is an array of logsums for each unique depart/arrive - // skim - // period combination, for each sample destination - protected double[][] modeChoiceLogsums; - protected double[][] tripModeChoiceLogsums; - - protected double[] tripModeChoiceSegmentLogsums = new double[2]; - - protected double[] tripModeChoiceSegmentStoredProbabilities; - - // departArriveLogsums is the array of values for all 15 depart/arrive - // combinations - protected double[][] departArriveLogsums; - - protected int chosenLogsumTodIndex = 0; - - protected int[] chosenDepartArriveCombination = new int[2]; - - protected int numMgraFields; - protected int[] mgraSetForLogsums; - - protected int[][] mgras; - - protected TazDataManager tazs; - protected MgraDataManager mgraManager; - protected TapDataManager tapManager; - protected MatrixDataServerIf ms; - - protected ModelStructure modelStructure; - - protected int[][] bestTapPairs; - protected double[] nmSkimsOut; - protected double[] nmSkimsIn; - protected double[] lbSkimsOut; - protected double[] lbSkimsIn; - protected double[] ebSkimsOut; - protected double[] ebSkimsIn; - protected double[] brSkimsOut; - protected double[] brSkimsIn; - protected double[] lrSkimsOut; - protected double[] lrSkimsIn; - protected double[] crSkimsOut; - protected double[] crSkimsIn; - - protected double[] lsWgtAvgCostM; - protected double[] lsWgtAvgCostD; - protected double[] lsWgtAvgCostH; - - protected int totalTime1 = 0; - protected int totalTime2 = 0; - - private McLogsumsCalculator logsumHelper; - - public McLogsumsAppender(HashMap propertyMap) - { - logsumHelper = new McLogsumsCalculator(); - - logsumHelper.setupSkimCalculators(propertyMap); - - } - - public BestTransitPathCalculator getBestTransitPathCalculator() - { - return bestPathUEC; - } - - protected TableDataSet getEstimationDataTableDataSet(HashMap rbMap) - { - - String estFileName = Util.getStringValueFromPropertyMap(rbMap, - ESTIMATION_DATA_RECORDS_FILE_KEY); - if (estFileName == null) - { - logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); - logger.error("Properties file target: " + ESTIMATION_DATA_RECORDS_FILE_KEY - + " not found."); - logger.error("Please specify a filename value for the " - + ESTIMATION_DATA_RECORDS_FILE_KEY + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFile(new File(estFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", - estFileName)); - throw new RuntimeException(e); - } - - } - - protected void calculateModeChoiceLogsums(HashMap rbMap, - ChoiceModelApplication mcModel, SandagAppendMcLogsumDMU mcDmuObject, int[] odt, - int[] mgraSample, int[] departAvailable, int[] arriveAvailable, boolean chosenTodOnly) - { - - boolean isChosenDepartArriveCombo = false; - - int origMgra = odt[ORIG_MGRA]; - int chosenMgra = odt[DEST_MGRA]; - - int inc = 0; - if (odt[INCOME] == 1) inc = 7500; - else if (odt[INCOME] == 2) inc = 22500; - else if (odt[INCOME] == 3) inc = 37500; - else if (odt[INCOME] == 4) inc = 52500; - else if (odt[INCOME] == 5) inc = 67500; - else if (odt[INCOME] == 6) inc = 87500; - else if (odt[INCOME] == 7) inc = 112500; - else if (odt[INCOME] == 8) inc = 137500; - else if (odt[INCOME] == 9) inc = 175000; - else if (odt[INCOME] == 10) inc = 250000; - else if (odt[INCOME] == 99) inc = 37500; - - mcDmuObject.setIncomeInDollars(inc); - mcDmuObject.setAdults(odt[ADULTS]); - mcDmuObject.setAutos(odt[AUTOS]); - mcDmuObject.setHhSize(odt[HHSIZE]); - mcDmuObject.setPersonIsFemale(odt[FEMALE]); - mcDmuObject.setAge(odt[AGE]); - mcDmuObject.setTourCategoryJoint(odt[JOINT]); - mcDmuObject.setTourCategoryEscort(odt[ESCORT]); - mcDmuObject.setNumberOfParticipantsInJointTour(odt[PARTYSIZE]); - - mcDmuObject.setWorkTourModeIsSOV(odt[WORK_TOUR_MODE] == 1 || odt[WORK_TOUR_MODE] == 2 ? 1 - : 0); - mcDmuObject.setWorkTourModeIsBike(odt[WORK_TOUR_MODE] == 10 ? 1 : 0); - mcDmuObject.setWorkTourModeIsHOV(odt[WORK_TOUR_MODE] >= 3 || odt[WORK_TOUR_MODE] <= 8 ? 1 - : 0); - - mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); - mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); - mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); - - mcDmuObject - .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); - - chosenDepartArriveCombination[0] = odt[DEPART_PERIOD]; - chosenDepartArriveCombination[1] = odt[ARRIVE_PERIOD]; - - // create an array with the chosen dest and the sample dests, for which - // to - // compute the logsums - mgraSetForLogsums[0] = chosenMgra; - for (int m = 0; m < mgraSample.length; m++) - mgraSetForLogsums[m + 1] = mgraSample[m]; - - int m = 0; - for (int destMgra : mgraSetForLogsums) - { - - if (mcModel != null && destMgra > 0) - { - // set the mode choice attributes needed by @variables in the - // UEC - // spreadsheets - mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, destMgra, false); - - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(destMgra)); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(destMgra)); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(destMgra)); - - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(destMgra))); - - modeChoiceLogsums[m] = new double[modelStructure.getSkimPeriodCombinationIndices().length]; - Arrays.fill(modeChoiceLogsums[m], -999); - } - - // compute the logsum for each depart/arrival time combination for - // the - // selected destination mgra - int i = 0; - for (int[] combo : departArriveCombinations) - { - - // mcModel might be null in the case where an estimation file - // record - // contains multiple purposes and the logsums are not desired - // for - // some purposes. - // destMgra might be null in the case of destination choice - // where - // some sample destinations are repeated, so the set of 30 or 40 - // contain 0s to reflect that. - if (mcModel == null || destMgra == 0) - { - continue; - } - - if (combo[0] == chosenDepartArriveCombination[0] - && combo[1] == chosenDepartArriveCombination[1]) - { - isChosenDepartArriveCombo = true; - chosenLogsumTodIndex = i; - } - - if (!chosenTodOnly || isChosenDepartArriveCombo) - { - - int departPeriod = DEFAULT_DEPART_INDICES[combo[0]]; - int arrivePeriod = DEFAULT_ARRIVE_INDICES[combo[1]]; - - // if the depart/arrive combination was flagged as - // unavailable, - // can skip the logsum calculation - if (unavailableCombination(departPeriod, arrivePeriod, departAvailable, - arriveAvailable)) - { - departArriveLogsums[m][i++] = -999; - continue; - } - - int logsumIndex = modelStructure.getSkimPeriodCombinationIndex(departPeriod, - arrivePeriod); - - // if a depart/arrive period combination results in a logsum - // index that's already had logsums computed, skip to next - // combination. - if (modeChoiceLogsums[m][logsumIndex] > -999) - { - departArriveLogsums[m][i++] = modeChoiceLogsums[m][logsumIndex]; - continue; - } - - mcDmuObject.setDepartPeriod(departPeriod); - mcDmuObject.setArrivePeriod(arrivePeriod); - - double logsum = logsumHelper.calculateTourMcLogsum(origMgra, destMgra, - departPeriod, arrivePeriod, mcModel, mcDmuObject); - - modeChoiceLogsums[m][logsumIndex] = logsum; - departArriveLogsums[m][i] = logsum; - - // write UEC calculation results to logsum specific log file - // if - // its the chosen dest and its the chosen time combo - if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2) - && (m == 0) /* && isChosenDepartArriveCombo */) - { - - nonManLogsumsLogger.info("Logsum[" + i - + "] calculation for estimation file record number " + odt[0]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); - nonManLogsumsLogger.info("mc purpose sheet = " - + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); - nonManLogsumsLogger.info("purpose category = " - + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " - + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); - nonManLogsumsLogger.info("origin mgra = " + odt[ORIG_MGRA]); - nonManLogsumsLogger.info("destination mgra = " + odt[DEST_MGRA]); - nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); - nonManLogsumsLogger.info("destination taz = " - + mgraManager.getTaz(destMgra)); - nonManLogsumsLogger.info("depart interval = " - + departArriveCombinationLabels[i][0] + ", @timeOutbound = " - + mcDmuObject.getTimeOutbound() + ", chosen depart = " - + departPeriod); - nonManLogsumsLogger.info("arrive interval = " - + departArriveCombinationLabels[i][1] + ", @timeInbound = " - + mcDmuObject.getTimeInbound() + ", chosen arrive = " - + arrivePeriod); - nonManLogsumsLogger.info("income category = " + odt[INCOME] - + ", @income = " + mcDmuObject.getIncome()); - nonManLogsumsLogger.info("adults = " + odt[ADULTS]); - nonManLogsumsLogger.info("autos = " + odt[AUTOS]); - nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); - nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " - + mcDmuObject.getFemale()); - nonManLogsumsLogger.info("jointTourCategory = " + odt[JOINT] - + ", @tourcategoryJoint = " + mcDmuObject.getTourCategoryJoint()); - nonManLogsumsLogger.info("partySize = " + odt[PARTYSIZE]); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info(""); - - mcModel.logUECResults(nonManLogsumsLogger, "Est Record: " + odt[0]); - nonManLogsumsLogger.info("Logsum Calculation for index: " + logsumIndex - + " , Logsum value: " + modeChoiceLogsums[m][logsumIndex]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger.info(""); - - isChosenDepartArriveCombo = false; - } - - } - i++; - } - - m++; - } - - } - - protected void calculateTripModeChoiceLogsums(HashMap rbMap, - ChoiceModelApplication mcModel, SandagTripModeChoiceDMU mcDmuObject, int[] odt, - int[] mgraSample) - { - - int origMgra = odt[ORIG_MGRA]; - int destMgra = odt[DEST_MGRA]; - int chosenMgra = odt[CHOSEN_MGRA]; - - for (int m = 0; m < tripModeChoiceLogsums.length; m++) - { - tripModeChoiceLogsums[m][0] = -999; - tripModeChoiceLogsums[m][1] = -999; - } - - if (origMgra == 0 || destMgra == 0 || odt[TOUR_MODE] == 0) return; - - int inc = 0; - if (odt[INCOME] == 1) inc = 7500; - else if (odt[INCOME] == 2) inc = 22500; - else if (odt[INCOME] == 3) inc = 37500; - else if (odt[INCOME] == 4) inc = 52500; - else if (odt[INCOME] == 5) inc = 67500; - else if (odt[INCOME] == 6) inc = 87500; - else if (odt[INCOME] == 7) inc = 112500; - else if (odt[INCOME] == 8) inc = 137500; - else if (odt[INCOME] == 9) inc = 175000; - else if (odt[INCOME] == 10) inc = 250000; - else if (odt[INCOME] == 99) inc = 37500; - - mcDmuObject.setOutboundHalfTourDirection(odt[DIRECTION]); - - mcDmuObject.setJointTour(odt[JOINT]); - mcDmuObject - .setEscortTour(odt[TOUR_PURPOSE] == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 - : 0); - - mcDmuObject.setIncomeInDollars(inc); - mcDmuObject.setAdults(odt[ADULTS]); - mcDmuObject.setAutos(odt[AUTOS]); - mcDmuObject.setAge(odt[AGE]); - mcDmuObject.setHhSize(odt[HHSIZE]); - mcDmuObject.setPersonIsFemale(odt[FEMALE] == 2 ? 1 : 0); - - mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(odt[TOUR_MODE]) ? 1 - : 0); - - mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); - mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); - mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); - - mcDmuObject - .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); - - mcDmuObject.setDepartPeriod(odt[DEPART_PERIOD]); - mcDmuObject.setTripPeriod(odt[TRIP_PERIOD]); - - int departPeriod = odt[TRIP_PERIOD]; - - // create an array with the chosen dest and the sample dests, for which - // to - // compute the logsums - mgraSetForLogsums[0] = chosenMgra; - for (int m = 0; m < mgraSample.length; m++) - { - mgraSetForLogsums[m + 1] = mgraSample[m]; - } - - if (mcModel == null || origMgra == 0) return; - - int m = 0; - for (int sampleMgra : mgraSetForLogsums) - { - - tripModeChoiceLogsums[m][0] = -999; - tripModeChoiceLogsums[m][1] = -999; - - if (mcModel != null && sampleMgra > 0) - { - // set the mode choice attributes needed by @variables in the - // UEC - // spreadsheets - mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, sampleMgra, false); - - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(sampleMgra)); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(sampleMgra)); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(sampleMgra)); - - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(sampleMgra))); - - if(odt[DIRECTION]==INBOUND_DIRCETION_CODE) - mcDmuObject.setInbound(true); - else - mcDmuObject.setInbound(false); - } - - // mcModel might be null in the case where an estimation file record - // contains multiple purposes and the logsums are not desired for - // some purposes. - // destMgra might be null in the case of destination choice where - // some sample destinations are repeated, so the set of 30 or 40 - // contain 0s to reflect that. - if (mcModel == null || sampleMgra == 0) - { - continue; - } - - // write UEC calculation results to logsum specific log file if - // its the chosen dest and its the chosen time combo - if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) - { - - nonManLogsumsLogger.info("IK Logsum calculation for estimation file record number " - + odt[0]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); - nonManLogsumsLogger.info("mc purpose sheet = " - + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); - nonManLogsumsLogger.info("purpose category = " - + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " - + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); - nonManLogsumsLogger.info("tour mode = " + odt[TOUR_MODE]); - nonManLogsumsLogger.info("origin mgra = " + origMgra); - nonManLogsumsLogger.info("sample destination mgra = " + sampleMgra); - nonManLogsumsLogger.info("final destination mgra = " + destMgra); - nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); - nonManLogsumsLogger.info("sample destination taz = " - + mgraManager.getTaz(sampleMgra)); - nonManLogsumsLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); - nonManLogsumsLogger.info("depart interval = " + departPeriod); - nonManLogsumsLogger.info("income category = " + odt[INCOME] + ", @income = " - + mcDmuObject.getIncome()); - nonManLogsumsLogger.info("adults = " + odt[ADULTS]); - nonManLogsumsLogger.info("autos = " + odt[AUTOS]); - nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); - nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " - + mcDmuObject.getFemale()); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info(""); - - mcDmuObject.getDmuIndexValues().setDebug(true); - } - - if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) - { - logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - - double logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, - mcModel, mcDmuObject, nonManLogsumsLogger); - tripModeChoiceLogsums[m][0] = logsum; - - if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) - { - nonManLogsumsLogger.info("IK Logsum value: " + tripModeChoiceLogsums[m][0]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger.info(""); - } - - // write UEC calculation results to logsum specific log file if - // its the chosen dest and its the chosen time combo - if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) - { - - nonManLogsumsLogger.info("KJ Logsum calculation for estimation file record number " - + odt[0]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); - nonManLogsumsLogger.info("mc purpose sheet = " - + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); - nonManLogsumsLogger.info("purpose category = " - + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " - + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); - nonManLogsumsLogger.info("origin mgra = " + sampleMgra); - nonManLogsumsLogger.info("sample destination mgra = " + destMgra); - nonManLogsumsLogger.info("final destination mgra = " + destMgra); - nonManLogsumsLogger.info("origin taz = " + mgraManager.getTaz(sampleMgra)); - nonManLogsumsLogger - .info("sample destination taz = " + mgraManager.getTaz(destMgra)); - nonManLogsumsLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); - nonManLogsumsLogger.info("depart interval = " + departPeriod); - nonManLogsumsLogger.info("income category = " + odt[INCOME] + ", @income = " - + mcDmuObject.getIncome()); - nonManLogsumsLogger.info("adults = " + odt[ADULTS]); - nonManLogsumsLogger.info("autos = " + odt[AUTOS]); - nonManLogsumsLogger.info("hhsize = " + odt[HHSIZE]); - nonManLogsumsLogger.info("gender = " + odt[FEMALE] + ", @female = " - + mcDmuObject.getFemale()); - nonManLogsumsLogger - .info("--------------------------------------------------------------------------------------------------------"); - nonManLogsumsLogger.info(""); - - mcDmuObject.getDmuIndexValues().setDebug(true); - } - - if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) - { - logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - - logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, - mcModel, mcDmuObject, nonManLogsumsLogger); - tripModeChoiceLogsums[m][1] = logsum; - - if ((odt[0] == debugEstimationFileRecord1 || odt[0] == debugEstimationFileRecord2)) - { - nonManLogsumsLogger.info("KJ Logsum value: " + tripModeChoiceLogsums[m][1]); - nonManLogsumsLogger.info(""); - nonManLogsumsLogger.info(""); - } - - m++; - } - - } - - protected double[] calculateTripModeChoiceLogsumForEstimationRecord( - HashMap rbMap, ChoiceModelApplication mcModel, - SandagTripModeChoiceDMU mcDmuObject, int[] odt, int sampleMgra) - { - - int origMgra = odt[ORIG_MGRA]; - int destMgra = odt[DEST_MGRA]; - - double[] tripModeChoiceLogsums = new double[2]; - tripModeChoiceLogsums[0] = -999; - tripModeChoiceLogsums[1] = -999; - - // mcModel would be null if the estimation file record has a stop - // purpose for which no ChoiceModelApplication has been defined. - if (origMgra == 0 || destMgra == 0 || sampleMgra == 0 || odt[TOUR_MODE] == 0 - || mcModel == null) return tripModeChoiceLogsums; - - mcDmuObject.setOutboundHalfTourDirection(odt[DIRECTION]); - - mcDmuObject.setJointTour(odt[JOINT]); - mcDmuObject - .setEscortTour(odt[TOUR_PURPOSE] == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 - : 0); - - int inc = 0; - if (odt[INCOME] == 1) inc = 7500; - else if (odt[INCOME] == 2) inc = 22500; - else if (odt[INCOME] == 3) inc = 37500; - else if (odt[INCOME] == 4) inc = 52500; - else if (odt[INCOME] == 5) inc = 67500; - else if (odt[INCOME] == 6) inc = 87500; - else if (odt[INCOME] == 7) inc = 112500; - else if (odt[INCOME] == 8) inc = 137500; - else if (odt[INCOME] == 9) inc = 175000; - else if (odt[INCOME] == 10) inc = 250000; - else if (odt[INCOME] == 99) inc = 37500; - - mcDmuObject.setIncomeInDollars(inc); - mcDmuObject.setAdults(odt[ADULTS]); - mcDmuObject.setAutos(odt[AUTOS]); - mcDmuObject.setAge(odt[AGE]); - mcDmuObject.setHhSize(odt[HHSIZE]); - mcDmuObject.setPersonIsFemale(odt[FEMALE] == 2 ? 1 : 0); - - mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(odt[TOUR_MODE]) ? 1 : 0); - mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(odt[TOUR_MODE]) ? 1 - : 0); - - mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(origMgra)); - mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(origMgra)); - mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(origMgra)); - - mcDmuObject - .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); - - mcDmuObject.setDepartPeriod(odt[DEPART_PERIOD]); - mcDmuObject.setTripPeriod(odt[TRIP_PERIOD]); - - int departPeriod = odt[TRIP_PERIOD]; - - // set the mode choice attributes needed by @variables in the UEC - // spreadsheets - mcDmuObject.setDmuIndexValues(odt[0], origMgra, origMgra, sampleMgra, false); - - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(sampleMgra)); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(sampleMgra)); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(sampleMgra)); - - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(sampleMgra))); - - if (mcDmuObject.getDmuIndexValues().getDebug()) - { - - // write UEC calculation results to logsum specific log file if - // its the chosen dest and its the chosen time combo - slcSoaLogger.info("IK Logsum calculation for estimation file record number " + odt[0]); - slcSoaLogger.info(""); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); - slcSoaLogger.info("mc purpose sheet = " + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); - slcSoaLogger.info("purpose category = " + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " - + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); - slcSoaLogger.info("tour mode = " + odt[TOUR_MODE]); - slcSoaLogger.info("origin mgra = " + origMgra); - slcSoaLogger.info("sample destination mgra = " + sampleMgra); - slcSoaLogger.info("final destination mgra = " + destMgra); - slcSoaLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); - slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(sampleMgra)); - slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); - slcSoaLogger.info("depart interval = " + departPeriod); - slcSoaLogger.info("income category = " + odt[INCOME] + ", @income = " - + mcDmuObject.getIncome()); - slcSoaLogger.info("adults = " + odt[ADULTS]); - slcSoaLogger.info("autos = " + odt[AUTOS]); - slcSoaLogger.info("hhsize = " + odt[HHSIZE]); - slcSoaLogger.info("gender = " + odt[FEMALE] + ", @female = " + mcDmuObject.getFemale()); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info(""); - - } - - if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) - { - logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, origMgra, sampleMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - - double logsum = logsumHelper.calculateTripMcLogsum(origMgra, sampleMgra, departPeriod, - mcModel, mcDmuObject, nonManLogsumsLogger); - tripModeChoiceLogsums[0] = logsum; - - if (mcDmuObject.getDmuIndexValues().getDebug()) - { - - slcSoaLogger.info("IK Mode Choice Logsum value: " + tripModeChoiceLogsums[0]); - slcSoaLogger.info(""); - slcSoaLogger.info(""); - - // write UEC calculation results to logsum specific log file if - // its the chosen dest and its the chosen time combo - slcSoaLogger.info("KJ Logsum calculation for estimation file record number " + odt[0]); - slcSoaLogger.info(""); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info("tour purpose = " + odt[TOUR_PURPOSE]); - slcSoaLogger.info("mc purpose sheet = " + MC_PURPOSE_SHEET_INDICES[odt[TOUR_PURPOSE]]); - slcSoaLogger.info("purpose category = " + PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]] + ": " - + PURPOSE_CATEGORY_LABELS[PURPOSE_CATEGORIES[odt[TOUR_PURPOSE]]]); - slcSoaLogger.info("origin mgra = " + sampleMgra); - slcSoaLogger.info("sample destination mgra = " + destMgra); - slcSoaLogger.info("final destination mgra = " + destMgra); - slcSoaLogger.info("origin taz = " + mgraManager.getTaz(sampleMgra)); - slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(destMgra)); - slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); - slcSoaLogger.info("depart interval = " + departPeriod); - slcSoaLogger.info("income category = " + odt[INCOME] + ", @income = " - + mcDmuObject.getIncome()); - slcSoaLogger.info("adults = " + odt[ADULTS]); - slcSoaLogger.info("autos = " + odt[AUTOS]); - slcSoaLogger.info("hhsize = " + odt[HHSIZE]); - slcSoaLogger.info("gender = " + odt[FEMALE] + ", @female = " + mcDmuObject.getFemale()); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info(""); - - } - - if ((odt[DIRECTION] == INBOUND_DIRCETION_CODE)) - { - logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - } else logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, - departPeriod, mcDmuObject.getDmuIndexValues().getDebug()); - - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, sampleMgra, destMgra, departPeriod, - mcDmuObject.getDmuIndexValues().getDebug()); - - logsum = logsumHelper.calculateTripMcLogsum(sampleMgra, destMgra, departPeriod, mcModel, - mcDmuObject, nonManLogsumsLogger); - tripModeChoiceLogsums[1] = logsum; - - if (mcDmuObject.getDmuIndexValues().getDebug()) - { - - slcSoaLogger.info("KJ Mode Choice Logsum value: " + tripModeChoiceLogsums[1]); - slcSoaLogger.info(""); - slcSoaLogger.info(""); - - } - - return tripModeChoiceLogsums; - - } - - protected int getModelPeriodFromTodIndex(int index) - { - int returnValue = -1; - if (index <= LAST_EA_INDEX) returnValue = EA; - else if (index <= LAST_AM_INDEX) returnValue = AM; - else if (index <= LAST_MD_INDEX) returnValue = MD; - else if (index <= LAST_PM_INDEX) returnValue = PM; - else returnValue = EV; - - return returnValue; - } - - /** - * - * @param departPeriod - * is the departure interval - * @param arrivePeriod - * is the arrival interval - * @param departAvailable - * is the model time period the departure interval belongs to - * (EA, AM, MD, PM, EV) - * @param arriveAvailable - * is the model time period the arrival interval belongs to (EA, - * AM, MD, PM, EV) - * @return true if the depart and/or arrival periods are unavailable, false - * if both are available. - */ - protected boolean unavailableCombination(int departPeriod, int arrivePeriod, - int[] departAvailable, int[] arriveAvailable) - { - - int departModelPeriod = getModelPeriodFromTodIndex(departPeriod); - int arriveModelPeriod = getModelPeriodFromTodIndex(arrivePeriod); - - boolean returnValue = true; - if (departAvailable[departModelPeriod] == 1 && arriveAvailable[arriveModelPeriod] == 1) - returnValue = false; - - return returnValue; - - } - - /** - * return the array of mode choice model cumulative probabilities determined - * while computing the mode choice logsum for the trip segmen during stop - * location choice. These probabilities arrays are stored for each sampled - * stop location so that when the selected sample stop location is known, - * the mode choice can be drawn from the already computed probabilities. - * - * @return mode choice cumulative probabilities array - */ - public double[] getStoredSegmentCumulativeProbabilities() - { - return tripModeChoiceSegmentStoredProbabilities; - } - - /** - * Start the matrix server - * - * @param rb - * is a ResourceBundle for the properties file for this - * application - */ - protected void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error("exception caught running ctramp model components -- exiting.", e); - throw new RuntimeException(); - - } - - } - - public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() - { - return logsumHelper.getAnmSkimCalculator(); - } - - public void setTazDistanceSkimArrays(double[][][] storedFromTazDistanceSkims, - double[][][] storedToTazDistanceSkims) - { - AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); - anm.setTazDistanceSkimArrays(storedFromTazDistanceSkims, storedToTazDistanceSkims); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java deleted file mode 100644 index 8a9f4dd..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryDcEstimationMcLogsumsAppender.java +++ /dev/null @@ -1,368 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.application.SandagAppendMcLogsumDMU; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -public final class NonMandatoryDcEstimationMcLogsumsAppender - extends McLogsumsAppender -{ - - private static final int DEBUG_EST_RECORD1 = 1; - private static final int DEBUG_EST_RECORD2 = -1; - - /* - * for DC estimation file - */ - private static final int SEQ_FIELD = 2; - - // for Atwork subtour DC - // private static final int ORIG_MGRA_FIELD = 79; - // private static final int DEST_MGRA_FIELD = 220; - // private static final int MGRA1_FIELD = 221; - // private static final int PURPOSE_INDEX_OFFSET = 4; - - // for Escort DC - private static final int ORIG_MGRA_FIELD = 76; - private static final int DEST_MGRA_FIELD = 79; - private static final int MGRA1_FIELD = 217; - private static final int PURPOSE_INDEX_OFFSET = 0; - - // for NonMandatory DC - // private static final int ORIG_MGRA_FIELD = 76; - // private static final int DEST_MGRA_FIELD = 79; - // private static final int MGRA1_FIELD = 221; - // private static final int PURPOSE_INDEX_OFFSET = 0; - - private static final int DEPART_PERIOD_FIELD = 189; - private static final int ARRIVE_PERIOD_FIELD = 190; - private static final int INCOME_FIELD = 20; - private static final int ADULTS_FIELD = 32; - private static final int AUTOS_FIELD = 6; - private static final int HHSIZE_FIELD = 5; - private static final int GENDER_FIELD = 38; - private static final int AGE_FIELD = 39; - private static final int PURPOSE_FIELD = 80; - private static final int JOINT_ID_FIELD = 125; - private static final int JOINT_PURPOSE_FIELD = 126; - private static final int JOINT_P1_FIELD = 151; - private static final int JOINT_P2_FIELD = 152; - private static final int JOINT_P3_FIELD = 153; - private static final int JOINT_P4_FIELD = 154; - private static final int JOINT_P5_FIELD = 155; - private static final int NUM_MGRA_FIELDS = 30; - - private static final String OUTPUT_SAMPLE_DEST_LOGSUMS = "output.sample.dest.logsums"; - - public NonMandatoryDcEstimationMcLogsumsAppender(HashMap rbMap) - { - super(rbMap); - - debugEstimationFileRecord1 = DEBUG_EST_RECORD1; - debugEstimationFileRecord2 = DEBUG_EST_RECORD2; - - numMgraFields = NUM_MGRA_FIELDS; - } - - private void runLogsumAppender(ResourceBundle rb) - { - - totalTime1 = 0; - totalTime2 = 0; - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - mgraSetForLogsums = new int[numMgraFields + 1]; - - // allocate the logsums array for the chosen destination alternative - modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; - - departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; - - String outputAllKey = Util.getStringValueFromPropertyMap(rbMap, OUTPUT_SAMPLE_DEST_LOGSUMS); - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - "dc.est.skims.output.file"); - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - int dotIndex = outputFileName.indexOf("."); - String baseName = outputFileName.substring(0, dotIndex); - String extension = outputFileName.substring(dotIndex); - - // output1 is only written if "all" was set in propoerties file - String outputName1 = ""; - if (outputAllKey.equalsIgnoreCase("all")) outputName1 = baseName + "_" + "all" + extension; - - // output1 is written in any case - String outputName2 = baseName + "_" + "chosen" + extension; - - PrintWriter outStream1 = null; - PrintWriter outStream2 = null; - - try - { - if (outputAllKey.equalsIgnoreCase("all")) - outStream1 = new PrintWriter(new BufferedWriter(new FileWriter( - new File(outputName1)))); - outStream2 = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName2)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - writeDcFile(rbMap, outStream1, outStream2); - - logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); - logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); - - } - - private void writeDcFile(HashMap rbMap, PrintWriter outStream1, - PrintWriter outStream2) - { - - // print the chosen destMgra and the depart/arrive logsum field names to - // both - // files - if (outStream1 != null) outStream1.print("seq,sampno,chosenMgra"); - - // attach the OB and IB period labels to the logsum field names for each - // period - if (outStream1 != null) - { - for (String[] labels : departArriveCombinationLabels) - outStream1.print(",logsum" + labels[0] + labels[1]); - } - - outStream2.print("seq,sampno,chosenMgra,chosenTodLogsum"); - - // print each set of sample destMgra and the depart/arrive logsum - // fieldnames - // to file 1. - // print each set of sample destMgra and the chosen depart/arrive logsum - // fieldname to file 2. - for (int m = 1; m < departArriveLogsums.length; m++) - { - if (outStream1 != null) - { - outStream1.print(",sampleMgra" + m); - for (String[] labels : departArriveCombinationLabels) - outStream1.print(",logsum" + m + labels[0] + labels[1]); - } - - outStream2.print(",sampleMgra" + m); - outStream2.print(",sampleLogsum" + m); - } - if (outStream1 != null) outStream1.print("\n"); - outStream2.print("\n"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - int[] mgraSet = mgras[i]; - - odtSet[0] = seq; - - if (outStream1 != null) - { - outStream1.print(seq + "," + odtSet[SAMPNO]); - } - outStream2.print(seq + "," + odtSet[SAMPNO]); - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - - int[] departAvailable = {-1, 1, 1, 1, 1, 1}; - int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; - calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], - mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); - - // write chosen dest and logsums to both files - if (outStream1 != null) - { - outStream1.print("," + odtSet[DEST_MGRA]); - for (double logsum : departArriveLogsums[0]) - outStream1.printf(",%.8f", logsum); - } - - outStream2.print("," + odtSet[DEST_MGRA]); - outStream2.printf(",%.8f", departArriveLogsums[0][chosenLogsumTodIndex]); - - // write logsum sets for each dest in the sample to file 1 - for (int m = 1; m < departArriveLogsums.length; m++) - { - if (outStream1 != null) - { - outStream1.print("," + mgraSet[m - 1]); - for (double logsum : departArriveLogsums[m]) - outStream1.printf(",%.8f", logsum); - } - - outStream2.print("," + mgraSet[m - 1]); - outStream2.printf(",%.8f", departArriveLogsums[m][chosenLogsumTodIndex]); - } - if (outStream1 != null) outStream1.print("\n"); - outStream2.print("\n"); - - if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); - - seq++; - } - - if (outStream1 != null) outStream1.close(); - outStream2.close(); - - } - - private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); - int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); - - int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); - int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); - int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); - int[] income = hisTds.getColumnAsInt(INCOME_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); - int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); - int[] jointPerson1Participates = hisTds.getColumnAsInt(JOINT_P1_FIELD); - int[] jointPerson2Participates = hisTds.getColumnAsInt(JOINT_P2_FIELD); - int[] jointPerson3Participates = hisTds.getColumnAsInt(JOINT_P3_FIELD); - int[] jointPerson4Participates = hisTds.getColumnAsInt(JOINT_P4_FIELD); - int[] jointPerson5Participates = hisTds.getColumnAsInt(JOINT_P5_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][SAMPNO] = hisseq[r - 1]; - - odts[r - 1][DEPART_PERIOD] = departs[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][INCOME] = income[r - 1]; - odts[r - 1][ADULTS] = adults[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][HHSIZE] = hhsize[r - 1]; - odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; - - // the offest constant is used because at-work subtours in - // estimation file are coded as work purpose index (=1), - // but the model index to use is 5. Nonmandatory and escort files - // have correct purpose codes, so offset is 0. - int purposeIndex = purpose[r - 1] + PURPOSE_INDEX_OFFSET; - - odts[r - 1][ESCORT] = purposeIndex == 4 ? 1 : 0; - - odts[r - 1][PARTYSIZE] = jointPerson1Participates[r - 1] - + jointPerson2Participates[r - 1] + jointPerson3Participates[r - 1] - + jointPerson4Participates[r - 1] + jointPerson5Participates[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purposeIndex > 4 ? jtPurpose[r - 1] - : purposeIndex; - - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - NonMandatoryDcEstimationMcLogsumsAppender appender = new NonMandatoryDcEstimationMcLogsumsAppender( - rbMap); - - appender.startMatrixServer(rb); - appender.runLogsumAppender(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java deleted file mode 100644 index f66cd14..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonMandatoryTodEstimationMcLogsumsAppender.java +++ /dev/null @@ -1,277 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.application.SandagAppendMcLogsumDMU; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -public final class NonMandatoryTodEstimationMcLogsumsAppender - extends McLogsumsAppender -{ - - private static final int DEBUG_EST_RECORD1 = 1; - private static final int DEBUG_EST_RECORD2 = -1; - - private static final int SEQ_FIELD = 7; - private static final int ORIG_MGRA_FIELD = 42; - private static final int DEST_MGRA_FIELD = 44; - private static final int DEPART_PERIOD_FIELD = 4; - private static final int ARRIVE_PERIOD_FIELD = 5; - private static final int INCOME_FIELD = 149; - private static final int ADULTS_FT_FIELD = 150; - private static final int ADULTS_PT_FIELD = 151; - private static final int ADULTS_UN_FIELD = 152; - private static final int ADULTS_RT_FIELD = 153; - private static final int ADULTS_NW_FIELD = 154; - private static final int AUTOS_FIELD = 146; - private static final int HHSIZE_FIELD = 147; - private static final int GENDER_FIELD = 24; - private static final int AGE_FIELD = 25; - private static final int PURPOSE_FIELD = 45; - private static final int JOINT_PURPOSE_FIELD = 58; - private static final int JOINT_ID_FIELD = 57; - private static final int JOINT_PARTICIPANTS_FIELD = 104; - private static final int MGRA1_FIELD = 1; - private static final int NUM_MGRA_FIELDS = 0; - - private NonMandatoryTodEstimationMcLogsumsAppender(HashMap rbMap) - { - super(rbMap); - - debugEstimationFileRecord1 = DEBUG_EST_RECORD1; - debugEstimationFileRecord2 = DEBUG_EST_RECORD2; - - numMgraFields = NUM_MGRA_FIELDS; - - } - - private void runLogsumAppender(ResourceBundle rb) - { - - totalTime1 = 0; - totalTime2 = 0; - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - mgraSetForLogsums = new int[numMgraFields + 1]; - - // allocate the logsums array for the chosen destination alternative - modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; - - departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - "tod.est.skims.output.file"); - - PrintWriter outStream = null; - - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - try - { - outStream = new PrintWriter( - new BufferedWriter(new FileWriter(new File(outputFileName)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - writeTodFile(rbMap, outStream); - - logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); - logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); - - } - - private void writeTodFile(HashMap rbMap, PrintWriter outStream2) - { - - // print the chosen destMgra and the depart/arrive logsum field names to - // the - // file - outStream2.print("seq,hisseq,chosenMgra"); - for (String[] labels : departArriveCombinationLabels) - { - outStream2.print(",logsum" + labels[0] + labels[1]); - } - outStream2.print("\n"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getTodEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - int[] mgraSet = mgras[i]; - - odtSet[0] = seq; - - outStream2.print(seq + "," + odtSet[SAMPNO]); - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - - int[] departAvailable = {-1, 1, 1, 1, 1, 1}; - int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; - calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], - mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); - - // write chosen dest and logsums to both files - outStream2.print("," + odtSet[DEST_MGRA]); - for (double logsum : departArriveLogsums[0]) - { - outStream2.printf(",%.8f", logsum); - } - outStream2.print("\n"); - - if (seq % 1000 == 0) logger.info("wrote TOD Estimation file record: " + seq); - - seq++; - } - - outStream2.close(); - - } - - private int[][] getTodEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); - int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); - - int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); - int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); - int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); - int[] income = hisTds.getColumnAsInt(INCOME_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] adultsFt = hisTds.getColumnAsInt(ADULTS_FT_FIELD); - int[] adultsPt = hisTds.getColumnAsInt(ADULTS_PT_FIELD); - int[] adultsUn = hisTds.getColumnAsInt(ADULTS_UN_FIELD); - int[] adultsRt = hisTds.getColumnAsInt(ADULTS_RT_FIELD); - int[] adultsNw = hisTds.getColumnAsInt(ADULTS_NW_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); - int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); - int[] jointParticipants = hisTds.getColumnAsInt(JOINT_PARTICIPANTS_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][SAMPNO] = hisseq[r - 1]; - - odts[r - 1][DEPART_PERIOD] = departs[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][INCOME] = income[r - 1]; - odts[r - 1][ADULTS] = adultsFt[r - 1] + adultsPt[r - 1] + adultsUn[r - 1] - + adultsRt[r - 1] + adultsNw[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][HHSIZE] = hhsize[r - 1]; - odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; - odts[r - 1][ESCORT] = purpose[r - 1] == 4 ? 1 : 0; - odts[r - 1][PARTYSIZE] = jointParticipants[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purpose[r - 1] > 4 ? jtPurpose[r - 1] - : purpose[r - 1]; - - } - - return odts; - } - - public static void main(String[] args) - { - - long startTime = System.currentTimeMillis(); - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - NonMandatoryTodEstimationMcLogsumsAppender appender = new NonMandatoryTodEstimationMcLogsumsAppender( - rbMap); - - appender.startMatrixServer(rb); - appender.runLogsumAppender(rb); - - System.out.println("total runtime = " + ((System.currentTimeMillis() - startTime) / 1000) - + " seconds."); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java deleted file mode 100644 index d7f07be..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/NonTransitUtilities.java +++ /dev/null @@ -1,641 +0,0 @@ -package org.sandag.abm.accessibilities; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; -import java.io.File; -import java.io.Serializable; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.AutoUEC; -import org.sandag.abm.modechoice.MaasUEC; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.NonMotorUEC; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; -import com.pb.common.util.Tracer; - -/** - * This class builds utility components for auto modes (SOV and HOV). - * - * @author Joel Freedman - * @version May, 2009 - */ -public class NonTransitUtilities - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(NonTransitUtilities.class); - - public static final int OFFPEAK_PERIOD_INDEX = 0; - public static final int PEAK_PERIOD_INDEX = 1; - - private static final String[] SOVPERIODS = {"OP", "PK"}; - private static final String[] HOVPERIODS = {"OP", "PK"}; - private static final String[] NMTPERIODS = {"OP"}; - private static final String[] MAASPERIODS = {"OP", "PK"}; - - - // store taz-taz exponentiated utilities (period, from taz, to taz) - private double[][][] sovExpUtilities; - private double[][][] hovExpUtilities; - private double[][][] nMotorExpUtilities; - private double[][][] maasExpUtilities; - - private double[] avgTazHourlyParkingCost; - private float[] avgTazTaxiWaitTime; - private float[] avgTazSingleTNCWaitTime; - private float[] avgTazSharedTNCWaitTime; - - // A HashMap of non-motorized utilities, period,oMgra,dMgra (where dMgra is - // ragged) - private HashMap[][] mgraNMotorExpUtilities; - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private AutoUEC[] sovUEC; - private AutoUEC[] hovUEC; - private NonMotorUEC[] nMotorUEC; - private MaasUEC[] maasUEC; - - private int maxTaz; - - private boolean trace, seek; - private Tracer tracer; - - //added for TNC and Taxi modes - TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; - - - /** - * Constructor. - * - * @param rb - * Resourcebundle with path to acc.uec.file, acc.data.page, - * acc.sov.offpeak.page, and acc.hov.offpeak.page - */ - - public NonTransitUtilities(HashMap rbMap, double[][][] mySovExpUtilities, - double[][][] myHovExpUtilities, double[][][] myNMotorExpUtilities, double[][][] myMaasExpUtilities) - { - - sovExpUtilities = mySovExpUtilities; - hovExpUtilities = myHovExpUtilities; - nMotorExpUtilities = myNMotorExpUtilities; - maasExpUtilities = myMaasExpUtilities; - - mgraManager = MgraDataManager.getInstance(); - tazManager = TazDataManager.getInstance(rbMap); - - maxTaz = tazManager.maxTaz; - - logger.info("max Taz " + maxTaz); - - // Create the peak and off-peak UECs - String uecFileName = Util.getStringValueFromPropertyMap(rbMap, "acc.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.data.page"); - int offpeakSOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sov.offpeak.page"); - int offpeakHOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.hov.offpeak.page"); - int peakSOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.sov.peak.page"); - int peakHOVPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.hov.peak.page"); - int nonMotorPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.nonmotorized.page"); - int peakMaasPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.maas.peak.page"); - int offpeakMaasPage = Util.getIntegerValueFromPropertyMap(rbMap, "acc.maas.offpeak.page"); - - sovUEC = new AutoUEC[SOVPERIODS.length]; - hovUEC = new AutoUEC[HOVPERIODS.length]; - nMotorUEC = new NonMotorUEC[NMTPERIODS.length]; - maasUEC = new MaasUEC[MAASPERIODS.length]; - - sovUEC[OFFPEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, offpeakSOVPage, dataPage); - sovUEC[PEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, peakSOVPage, dataPage); - hovUEC[OFFPEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, offpeakHOVPage, dataPage); - hovUEC[PEAK_PERIOD_INDEX] = new AutoUEC(rbMap, uecFileName, peakHOVPage, dataPage); - nMotorUEC[OFFPEAK_PERIOD_INDEX] = new NonMotorUEC(rbMap, uecFileName, nonMotorPage, - dataPage); - maasUEC[OFFPEAK_PERIOD_INDEX] = new MaasUEC(rbMap, uecFileName, offpeakMaasPage, dataPage); - maasUEC[PEAK_PERIOD_INDEX] = new MaasUEC(rbMap, uecFileName, peakMaasPage, dataPage); - - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - int[] traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - int[] traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - logger.info("Setting trace zone pair in NonTransitUtilities Object for i: "+ traceOtaz[i] + " j: " + traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - int maxMgra = mgraManager.getMaxMgra(); -// logger.info("max Mgra " + maxMgra); - - mgraNMotorExpUtilities = new HashMap[NMTPERIODS.length][maxMgra + 1]; - - sovExpUtilities = new double[SOVPERIODS.length][maxTaz + 1][]; - hovExpUtilities = new double[HOVPERIODS.length][maxTaz + 1][]; - nMotorExpUtilities = new double[NMTPERIODS.length][maxTaz + 1][]; - - maasExpUtilities = new double[MAASPERIODS.length][maxTaz + 1][]; - - calculateAverageTazParkingCosts(); - - tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); - tncTaxiWaitTimeCalculator.createWaitTimeDistributions(rbMap); - - calculateAverageMaasWaitTimes(); - - - - } - - /** - * set the utilities values created by another object by calling - * buildUtilities() - */ - public void setAllUtilities(double[][][][] ntUtilities) - { - this.sovExpUtilities = ntUtilities[0]; - this.hovExpUtilities = ntUtilities[1]; - this.nMotorExpUtilities = ntUtilities[2]; - this.maasExpUtilities = ntUtilities[3]; - } - - /** - * get the set of utilities arrays built by calling buildUtilities(). - * - * @return array of 4 utilities arrays: sovExpUtilities, hovExpUtilities, - * nMotorExpUtilities, maasExpUtilities - */ - public double[][][][] getAllUtilities() - { - double[][][][] allUtilities = new double[3][][][]; - allUtilities[0] = sovExpUtilities; - allUtilities[1] = hovExpUtilities; - allUtilities[2] = nMotorExpUtilities; - allUtilities[3] = maasExpUtilities; - return allUtilities; - } - - /** - * set the HashMap of non-motorized utilities, period,oMgra,dMgra (where - * dMgra is ragged) - * - * @param mgraNMotorExpUtilities - */ - public void setNonMotorUtilsMap(HashMap[][] aMgraNMotorExpUtilities) - { - this.mgraNMotorExpUtilities = aMgraNMotorExpUtilities; - } - - /** - * get the HashMap of non-motorized utilities, period,oMgra,dMgra (where - * dMgra is ragged) that was built by calling buildUtilities(). - * - * @return mgraNMotorExpUtilities - */ - public HashMap[][] getNonMotorUtilsMap() - { - return mgraNMotorExpUtilities; - } - - /** - * Build SOV, HOV, and non-motorized exponentiated utilities for all - * TAZ-pairs. Also builds non-motorized exponentiated utilities for close-in - * mgra pairs. - * - */ - public void buildUtilities() - { - } - - /* - * public void buildUtilities() { - * - * logger.info("Calculating Non-Transit Zonal Utilities"); - * - * // first calculate the Tap-Tap utilities, exponentiate, and store - * logger.info("Calculating Taz-Taz utilities"); - * - * int maxMgra = mgraManager.getMaxMgra(); logger.info("max Mgra " + - * maxMgra); - * - * mgraNMotorExpUtilities = new HashMap[NMTPERIODS.length][maxMgra + 1]; - * - * sovExpUtilities = new double[SOVPERIODS.length][maxTaz + 1][maxTaz + 1]; - * hovExpUtilities = new double[HOVPERIODS.length][maxTaz + 1][maxTaz + 1]; - * nMotorExpUtilities = new double[NMTPERIODS.length][maxTaz + 1][maxTaz + - * 1]; - * - * for (int iTaz = 1; iTaz <= maxTaz; ++iTaz) { - * - * if (iTaz <= 10 || (iTaz % 500) == 0) logger.info("...Origin TAZ " + - * iTaz); - * - * // calculate the utilities for close-in mgras int[] oMgras = - * tazManager.getMgraArray(iTaz); if ( oMgras == null ) continue; - * - * for (int oMgra : oMgras) { - * - * // if there are mgras within walking distance if - * (mgraManager.getMgrasWithinWalkDistanceFrom(oMgra) != null) { int[] - * dMgras = mgraManager.getMgrasWithinWalkDistanceFrom(oMgra); - * - * int mgraNumber = 0; - * - * // cycle through periods, and calculate utilities for (int period = 0; - * period < NMTPERIODS.length; ++period) { - * - * mgraNMotorExpUtilities[period][oMgra] = new HashMap(); - * - * // cycle through the destination mgras for (int dMgra : dMgras) { - * - * double nmtUtility = nMotorUEC[period].calculateUtilitiesForMgraPair( - * oMgra, dMgra ); - * - * // exponentiate the utility if (nmtUtility > -500) - * mgraNMotorExpUtilities[period][oMgra].put( dMgra, Math.exp(nmtUtility) ); - * ++mgraNumber; - * - * } - * - * } - * - * } } - * - * for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) { - * - * if (seek && !tracer.isTraceZonePair(iTaz, jTaz)) continue; - * - * for (int period = 0; period < SOVPERIODS.length; ++period) { - * - * double sovUtility = sovUEC[period].calculateUtilitiesForTazPair(iTaz, - * jTaz); // exponentiate the SOV utility if (sovUtility > -500) - * sovExpUtilities[period][iTaz][jTaz] = Math.exp(sovUtility); } - * - * for (int period = 0; period < HOVPERIODS.length; ++period) { - * - * double hovUtility = hovUEC[period].calculateUtilitiesForTazPair(iTaz, - * jTaz); // exponentiate the SOV utility if (hovUtility > -500) - * hovExpUtilities[period][iTaz][jTaz] = Math.exp(hovUtility); } - * - * for (int period = 0; period < NMTPERIODS.length; ++period) { - * - * double nmtUtility = nMotorUEC[period].calculateUtilitiesForTazPair(iTaz, - * jTaz); // exponentiate the SOV utility if (nmtUtility > -500) - * nMotorExpUtilities[period][iTaz][jTaz] = Math.exp(nmtUtility); } - * - * } } - * - * } - */ - - /** - * calculate an average TAZ parking cost to use in accessibilities - * calculation which are done at TAZ level. - */ - private void calculateAverageTazParkingCosts() - { - - avgTazHourlyParkingCost = new double[maxTaz + 1]; - - for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) - { - - int[] mgras = tazManager.getMgraArray(jTaz); - if (mgras == null || mgras.length == 0) continue; - - double cost = 0; - int count = 0; - for (int mgra : mgras) - { - float mgraCost = mgraManager.getMgraHourlyParkingCost(mgra); - if (mgraCost > 0) - { - cost += mgraCost; - count++; - } - if (count > 0) cost /= count; - } - - avgTazHourlyParkingCost[jTaz] = cost; - - } - - } - - /** - * calculate an average TAZ parking cost to use in accessibilities - * calculation which are done at TAZ level. - */ - private void calculateAverageMaasWaitTimes() - { - - avgTazTaxiWaitTime = new float[maxTaz + 1]; - avgTazSingleTNCWaitTime = new float[maxTaz + 1]; - avgTazSharedTNCWaitTime = new float[maxTaz + 1]; - - for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) - { - - int[] mgras = tazManager.getMgraArray(jTaz); - if (mgras == null || mgras.length == 0) continue; - - float taxiWait = 0; - float singleTNCWait = 0; - float sharedTNCWait = 0; - int singleTNCCount = 0; - int sharedTNCCount = 0; - int taxiCount = 0; - - - for (int mgra : mgras) - { - float popEmpDen = (float) mgraManager.getPopEmpPerSqMi(mgra); - float singleTNCWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDen); - float sharedTNCWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDen); - float taxiWaitTime = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDen); - - if (taxiWaitTime > 0) - { - taxiWait += taxiWaitTime; - taxiCount++; - } - if (singleTNCWaitTime > 0) - { - singleTNCWait += singleTNCWaitTime; - singleTNCCount++; - } - if (sharedTNCWaitTime > 0) - { - sharedTNCWait += sharedTNCWaitTime; - sharedTNCCount++; - } - if (taxiCount > 0) taxiWait /= taxiCount; - if (singleTNCCount > 0) singleTNCWait /= singleTNCCount; - if (sharedTNCCount > 0) sharedTNCWait /= sharedTNCCount; - } - - avgTazTaxiWaitTime[jTaz] = taxiWait; - avgTazSingleTNCWaitTime[jTaz] = singleTNCWait; - avgTazSharedTNCWaitTime[jTaz] = sharedTNCWait; - } - - } - - public void buildUtilitiesForOrigMgraAndPeriod(int iMgra, int period) - { - - int iTaz = mgraManager.getTaz(iMgra); - if (sovExpUtilities[period][iTaz] != null) return; - - sovExpUtilities[period][iTaz] = new double[maxTaz + 1]; - hovExpUtilities[period][iTaz] = new double[maxTaz + 1]; - maasExpUtilities[period][iTaz] = new double[maxTaz + 1]; - - for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) - { - - double sovUtility = sovUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, - avgTazHourlyParkingCost[jTaz]); - // exponentiate the SOV utility - if (sovUtility > -500) sovExpUtilities[period][iTaz][jTaz] = Math.exp(sovUtility); - - double hovUtility = hovUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, - avgTazHourlyParkingCost[jTaz]); - // exponentiate the HOV utility - if (hovUtility > -500) hovExpUtilities[period][iTaz][jTaz] = Math.exp(hovUtility); - - double maasUtility = maasUEC[period].calculateUtilitiesForTazPair(iTaz, jTaz, - avgTazTaxiWaitTime[jTaz], avgTazSingleTNCWaitTime[jTaz], avgTazSharedTNCWaitTime[jTaz]); - // exponentiate the SOV utility - if (maasUtility > -500) maasExpUtilities[period][iTaz][jTaz] = Math.exp(maasUtility); - - - } - - // non-motorized utilities are only needed for off-peak period, so if - // period index == 1 (peak) no nead to calculate off-peak - if (nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz] == null) - { - - nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz] = new double[maxTaz + 1]; - - for (int jTaz = 1; jTaz <= maxTaz; ++jTaz) - { - - double nmtUtility = nMotorUEC[OFFPEAK_PERIOD_INDEX].calculateUtilitiesForTazPair( - iTaz, jTaz); - // exponentiate the SOV utility - if (nmtUtility > -500) - nMotorExpUtilities[OFFPEAK_PERIOD_INDEX][iTaz][jTaz] = Math.exp(nmtUtility); - - } - - } - - } - - /** - * Get the non-motorized exponentiated utility for the mgra-pair and period. - * This method will return the taz-taz exponentiated non-motorized utility - * if the mgra-mgra exp utility doesn't exist. Otherwise the mgra-mgra exp. - * utility will be returned. - * - * @param iMgra - * Origin/production mgra. - * @param jMgra - * Destination/attraction mgra. - * @param period - * Period. - * @return The non-motorized exponentiated utility. - */ - /* - * public double getNMotorExpUtility(int iMgra, int jMgra, int period) { // - * no mgra-mgra utilities for this origin if - * (mgraNMotorExpUtilities[period][iMgra] == null) { int iTaz = - * mgraManager.getTaz(iMgra); int jTaz = mgraManager.getTaz(jMgra); return - * nMotorExpUtilities[period][iTaz][jTaz]; } - * - * // mgra-mgra utilities exist if - * (mgraNMotorExpUtilities[period][iMgra].containsKey(jMgra)) { return - * mgraNMotorExpUtilities[period][iMgra].get(jMgra); } - * - * // no mgra-mgra utilities for this destination int iTaz = - * mgraManager.getTaz(iMgra); int jTaz = mgraManager.getTaz(jMgra); return - * nMotorExpUtilities[period][iTaz][jTaz]; } - */ - public double getNMotorExpUtility(int iMgra, int jMgra, int period) - { - - // if no utilities exist for period and origin mgra, try to compute them - if (mgraNMotorExpUtilities[period][iMgra] == null) - { - - // get the mgras within walking distance of the iMgra - int[] dMgras = mgraManager.getMgrasWithinWalkDistanceFrom(iMgra); - - if (dMgras == null) - { - mgraNMotorExpUtilities[period][iMgra] = new HashMap(0); - } else - { - mgraNMotorExpUtilities[period][iMgra] = new HashMap(dMgras.length); - - // cycle through the destination mgras - for (int dMgra : dMgras) - { - // calculate utility for the specified mgra and period - double nmtUtility = nMotorUEC[period].calculateUtilitiesForMgraPair(iMgra, - dMgra); - - // exponentiate the utility - if (nmtUtility > -500) - mgraNMotorExpUtilities[period][iMgra].put(dMgra, Math.exp(nmtUtility)); - } - - } - - } - - // if jMgra is in the HashMap, return its utility value - if (mgraNMotorExpUtilities[period][iMgra].containsKey(jMgra)) - return mgraNMotorExpUtilities[period][iMgra].get(jMgra); - - // otherwise, get exponentiated utilities based on highway skim values - // for the taz pair associated with iMgra and jMgra. - int iTaz = mgraManager.getTaz(iMgra); - int jTaz = mgraManager.getTaz(jMgra); - return nMotorExpUtilities[period][iTaz][jTaz]; - - } - - /** - * Get the SOV Exponentiated Utility for a given ptaz, ataz, and period - * - * @param pTaz - * Production/Origin TAZ - * @param aTaz - * Attraction/Destination TAZ - * @param period - * Period - * @return SOV Exponentiated Utility. - */ - public double getSovExpUtility(int pTaz, int aTaz, int period) - { - return sovExpUtilities[period][pTaz][aTaz]; - } - - /** - * Get the HOV Exponentiated Utility for a given ptaz, ataz, and period - * - * @param pTaz - * Production/Origin TAZ - * @param aTaz - * Attraction/Destination TAZ - * @param period - * Period - * @return SOV Exponentiated Utility. - */ - public double getHovExpUtility(int pTaz, int aTaz, int period) - { - return hovExpUtilities[period][pTaz][aTaz]; - } - - /** - * Get the Maas Exponentiated Utility for a given ptaz, ataz, and period - * - * @param pTaz - * Production/Origin TAZ - * @param aTaz - * Attraction/Destination TAZ - * @param period - * Period - * @return Maas Exponentiated Utility. - */ - public double getMaasExpUtility(int pTaz, int aTaz, int period) - { - return maasExpUtilities[period][pTaz][aTaz]; - } - /** - * The main method runs this class, for testing purposes. - * - * @param args - * args[0] is the property file for this test run. - */ - public static void main(String[] args) - { - - ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - boolean os64bit = false; - MatrixDataServer matrixServer = null; - - os64bit = Boolean.parseBoolean(Util.getStringValueFromPropertyMap(rbMap, - "operatingsystem.64bit")); - if (os64bit) - { - - String serverAddress = Util.getStringValueFromPropertyMap(rbMap, "server.address"); - - int serverPort = Util.getIntegerValueFromPropertyMap(rbMap, "server.port"); - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - - matrixServer = new MatrixDataServer(); - - // bind this concrete object with the cajo library objects for - // managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - System.out.println(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort)); - e.printStackTrace(); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - System.out.println(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort)); - e.printStackTrace(); - throw new RuntimeException(); - } - } - - double[][][] sovExpUtilities = null; - double[][][] hovExpUtilities = null; - double[][][] nMotorExpUtilities = null; - double[][][] maasExpUtilities = null; - - NonTransitUtilities au = new NonTransitUtilities(rbMap, sovExpUtilities, hovExpUtilities, - nMotorExpUtilities, maasExpUtilities); - au.buildUtilities(); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java deleted file mode 100644 index 3023ff4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/ParkLocationEstimationAppender.java +++ /dev/null @@ -1,297 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -public final class ParkLocationEstimationAppender -{ - - private transient Logger logger = Logger.getLogger(ParkLocationEstimationAppender.class); - - private static final int DEBUG_EST_RECORD = 1; - - private static final String ESTIMATION_DATA_RECORDS_FILE_KEY = "plc.estimation.data.file"; - private static final String OUTPUT_DATA_RECORDS_FILE_KEY = "plc.est.skims.output.file"; - - // define input table field indices - private static final int ID_FIELD = 1; - private static final int DEPART_PERIOD_FIELD = 2; - private static final int TYPE_FIELD = 4; - private static final int ORIG_FIELD = 7; - private static final int DEST_FIELD = 8; - - // define indices for storing input data in an internal table. - // start field indices at 1; reserve 0 for the input file record sequence - // number. - private static final int ID = 1; - private static final int DEPART = 2; - private static final int TYPE = 3; - private static final int ORIG = 4; - private static final int DEST = 5; - private static final int NUM_FIELDS = 5; - - private static final int OD_TYPE_INDEX = 0; - private static final int OP_TYPE_INDEX = 1; - private static final int PD_TYPE_INDEX = 2; - private static final int[] TYPE_INDICES = {OD_TYPE_INDEX, OP_TYPE_INDEX, - PD_TYPE_INDEX }; - private static final String OD_TYPE = "OD"; - private static final String OP_TYPE = "OP"; - private static final String PD_TYPE = "PD"; - private static final String[] TYPE_LABELS = {OD_TYPE, OP_TYPE, PD_TYPE}; - - private static final int AUTO_TIME_SKIM_INDEX = 0; - private static final int AUTO_DIST_SKIM_INDEX = 2; - - private static final double WALK_SPEED = 3.0; // mph; - - private static final float defaultVOT = 15.0f; - private MatrixDataServerIf ms; - - public ParkLocationEstimationAppender() - { - } - - private void runAppender(ResourceBundle rb) - { - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - MgraDataManager mgraManager = MgraDataManager.getInstance(rbMap); - SandagModelStructure modelStructure = new SandagModelStructure(); - - double[] distances = new double[mgraManager.getMaxMgra() + 1]; - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - OUTPUT_DATA_RECORDS_FILE_KEY); - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - int dotIndex = outputFileName.indexOf("."); - String baseName = outputFileName.substring(0, dotIndex); - String extension = outputFileName.substring(dotIndex); - String outputName = baseName + extension; - - PrintWriter outStream = null; - - try - { - outStream = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - outStream - .println("seq,id,origMgra,destMgra,departPeriod,index,autoDist,autoTime,walkDist,walkTime"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getPlcEstimationData(estTds); - - AutoAndNonMotorizedSkimsCalculator anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - odtSet[0] = seq; - - double aDist = 999; - double aTime = 999; - double wDist = 999; - double wTime = 999; - if (odtSet[ORIG] > 0 && odtSet[DEST] > 0) - { - - int skimPeriodIndex = modelStructure.getSkimPeriodIndex(odtSet[DEPART]) + 1; // depart - // skim - // period - double[] autoSkims = anm.getAutoSkims(odtSet[ORIG], odtSet[DEST], skimPeriodIndex, defaultVOT, - (seq == DEBUG_EST_RECORD), logger); - aDist = autoSkims[AUTO_DIST_SKIM_INDEX]; - aTime = autoSkims[AUTO_TIME_SKIM_INDEX]; - - // get the array of mgras within walking distance of the - // destination - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(odtSet[ORIG]); - - // set the distance values for the mgras walkable to the - // destination - if (walkMgras != null) - { - - // get distances, in feet, and convert to miles - for (int wMgra : walkMgras) - { - if (wMgra == odtSet[DEST]) - { - wDist = mgraManager.getMgraToMgraWalkDistFrom(odtSet[ORIG], wMgra) / 5280.0; - wTime = (wDist / WALK_SPEED) * 60.0; - break; - } - } - } - - } - - outStream.println(seq + "," + odtSet[ID] + "," + odtSet[ORIG] + "," + odtSet[DEST] - + "," + odtSet[DEPART] + "," + TYPE_LABELS[odtSet[TYPE]] + "," + aDist + "," - + aTime + "," + wDist + "," + wTime); - - if (seq % 1000 == 0) logger.info("wrote PLC Estimation file record: " + seq); - - seq++; - } - - outStream.close(); - - } - - private int[][] getPlcEstimationData(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS + 1]; - - int[] ids = hisTds.getColumnAsInt(ID_FIELD); - int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); - String[] types = hisTds.getColumnAsString(TYPE_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_FIELD); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - odts[r - 1][ID] = ids[r - 1]; - odts[r - 1][DEPART] = departs[r - 1]; - - for (int i = 0; i < TYPE_INDICES.length; i++) - { - if (types[r - 1].equalsIgnoreCase(TYPE_LABELS[i])) - { - odts[r - 1][TYPE] = TYPE_INDICES[i]; - break; - } - } - - odts[r - 1][ORIG] = origs[r - 1]; - odts[r - 1][DEST] = dests[r - 1]; - - } - - return odts; - } - - protected TableDataSet getEstimationDataTableDataSet(HashMap rbMap) - { - - String estFileName = Util.getStringValueFromPropertyMap(rbMap, - ESTIMATION_DATA_RECORDS_FILE_KEY); - if (estFileName == null) - { - logger.error("Error getting the filename from the properties file for the Sandag estimation data records file."); - logger.error("Properties file target: " + ESTIMATION_DATA_RECORDS_FILE_KEY - + " not found."); - logger.error("Please specify a filename value for the " - + ESTIMATION_DATA_RECORDS_FILE_KEY + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFile(new File(estFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag estimation data records file: %s into TableDataSet object.", - estFileName)); - throw new RuntimeException(e); - } - - } - - /** - * Start the matrix server - * - * @param rb - * is a ResourceBundle for the properties file for this - * application - */ - protected void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error("exception caught running ctramp model components -- exiting.", e); - throw new RuntimeException(); - - } - - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - ParkLocationEstimationAppender appender = new ParkLocationEstimationAppender(); - - appender.startMatrixServer(rb); - appender.runAppender(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java deleted file mode 100644 index 95726b6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/SkimsAppender.java +++ /dev/null @@ -1,788 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -public final class SkimsAppender -{ - - protected transient Logger logger = Logger.getLogger(SkimsAppender.class); - - private static final String OBS_DATA_RECORDS_FILE_KEY = "onBoard.survey.file"; - private static final String HIS_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; - - private static final int OBS_UNIQUE_ID = 1; - private static final int OBS_ORIG_MGRA = 78; - private static final int OBS_DEST_MGRA = 79; - - // used for trip file: - private static final int OBS_OUT_TOUR_PERIOD = 133; - private static final int OBS_IN_TOUR_PERIOD = 134; - - // used for tour file: - // private static final int OBS_DEPART_PERIOD = 132; - // private static final int OBS_ARRIVE_PERIOD = 133; - - /* - * for home based tour mode choice estimation files private static final int - * HIS_ORIG_MGRA = 72; private static final int HIS_DEST_MGRA = 75; private - * static final int HIS_DEPART_PERIOD = 185; private static final int - * HIS_ARRIVE_PERIOD = 186; - */ - - /* - * for work based tour mode choice estimation files - */ - private static final int HIS_ORIG_MGRA = 76; - private static final int HIS_DEST_MGRA = 84; - private static final int HIS_DEPART_PERIOD = 159; - private static final int HIS_ARRIVE_PERIOD = 160; - - // survey periods are: 0=not used, 1=03:00-05:59, 2=06:00-08:59, - // 3=09:00-11:59, - // 4=12:00-15:29, 5=15:30-18:59, 6=19:00-02:59 - // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), - // 6=3(OP) - - // define a conversion array to convert period values in the survey file to - // skim - // period indices used in this propgram: 1=am peak, 2=pm peak, - // 3=off-peak. - private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; - private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; - - private MatrixDataServerIf ms; - private BestTransitPathCalculator bestPathUEC; - - TransitWalkAccessDMU walkDmu; - - private SkimsAppender() - { - } - - private void runSkimsAppender(ResourceBundle rb) - { - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - // instantiate these objects right away - TazDataManager tazs = TazDataManager.getInstance(rbMap); - MgraDataManager mgraManager = MgraDataManager.getInstance(rbMap); - TapDataManager tapManager = TapDataManager.getInstance(rbMap); - - AutoAndNonMotorizedSkimsCalculator anm = null; - WalkTransitWalkSkimsCalculator wtw = null; - WalkTransitDriveSkimsCalculator wtd = null; - DriveTransitWalkSkimsCalculator dtw = null; - - Logger autoLogger = Logger.getLogger("auto"); - Logger wtwLogger = Logger.getLogger("wtw"); - Logger wtdLogger = Logger.getLogger("wtd"); - Logger dtwLogger = Logger.getLogger("dtw"); - - String outputFileNameObs = Util.getStringValueFromPropertyMap(rbMap, - "obs.skims.output.file"); - String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, - "his.skims.output.file"); - - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - - FileWriter writer; - PrintWriter outStreamObs = null; - PrintWriter outStreamHis = null; - - PrintWriter[] outStreamObsTod = new PrintWriter[SKIM_PERIOD_LABELS.length]; - PrintWriter[] outStreamHisTod = new PrintWriter[SKIM_PERIOD_LABELS.length]; - - if (!outputFileNameObs.isEmpty() || !outputFileNameHis.isEmpty()) - { - - anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - bestPathUEC = logsumHelper.getBestTransitPathCalculator(); - - wtw = new WalkTransitWalkSkimsCalculator(rbMap); - wtw.setup(rbMap, wtwLogger, bestPathUEC); - - wtd = new WalkTransitDriveSkimsCalculator(rbMap); - wtd.setup(rbMap, wtdLogger, bestPathUEC); - - dtw = new DriveTransitWalkSkimsCalculator(rbMap); - dtw.setup(rbMap, dtwLogger, bestPathUEC); - - String heading = "Seq,Id"; - - heading += ",obOrigMgra,obDestMgra,obPeriod"; - heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); - heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); - heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); - heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); - heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); - - heading += ",ObsSeq,Id,ibOrigMgra,ibDestMgra,ibPeriod"; - heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); - heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); - heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); - heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); - heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); - - if (!outputFileNameObs.isEmpty()) - { - try - { - // create an output stream for the mode choice estimation - // file - // with - // observed TOD - writer = new FileWriter(new File(outputFileNameObs)); - outStreamObs = new PrintWriter(new BufferedWriter(writer)); - - // create an array of similar files, 1 for each TOD period, - // for - // TOD - // choice estimation - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - int dotIndex = outputFileNameObs.lastIndexOf('.'); - String newName = outputFileNameObs.substring(0, dotIndex) + "_" - + SKIM_PERIOD_LABELS[i] + outputFileNameObs.substring(dotIndex); - writer = new FileWriter(new File(newName)); - outStreamObsTod[i] = new PrintWriter(new BufferedWriter(writer)); - } - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileNameObs)); - throw new RuntimeException(e); - } - - outStreamObs.println("obs" + heading); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamObsTod[i].println("obs" + heading); - } - - if (!outputFileNameHis.isEmpty()) - { - try - { - writer = new FileWriter(new File(outputFileNameHis)); - outStreamHis = new PrintWriter(new BufferedWriter(writer)); - - // create an array of similar files, 1 for each TOD period, - // for - // TOD - // choice estimation - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - int dotIndex = outputFileNameHis.lastIndexOf('.'); - String newName = outputFileNameHis.substring(0, dotIndex) + "_" - + SKIM_PERIOD_LABELS[i] + outputFileNameHis.substring(dotIndex); - writer = new FileWriter(new File(newName)); - outStreamHisTod[i] = new PrintWriter(new BufferedWriter(writer)); - } - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileNameHis)); - throw new RuntimeException(e); - } - - outStreamHis.println("his" + heading); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamHisTod[i].println("his" + heading); - } - - } - - Logger[] loggers = new Logger[4]; - loggers[0] = autoLogger; - loggers[1] = autoLogger; - loggers[2] = wtdLogger; - loggers[3] = dtwLogger; - - float defaultVOT= 15f; //$15 default VOT - - int[] odt = new int[5]; - - if (!outputFileNameObs.isEmpty()) - { - TableDataSet obsTds = getOnBoardSurveyTableDataSet(rbMap); - int[][] obsOdts = getOnBoardSurveyOrigDestTimes(obsTds); - - // write skims data for on-board survey records - int seq = 1; - for (int[] obsOdt : obsOdts) - { - // write outbound direction - odt[0] = obsOdt[0]; // orig - odt[1] = obsOdt[1]; // dest - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[obsOdt[2]]; // depart skim - // period - odt[3] = obsOdt[3]; - odt[4] = obsOdt[4]; - - if (odt[0] == 0 || odt[1] == 0) - { - outStreamObs.println(String.format("%d,%d,%d,%d,%d", seq, odt[4], odt[0], - odt[1], odt[2])); - seq++; - continue; - } - - // index - - // for debugging a specific mgra pair - // odt[0] = 25646; - // odt[1] = 4319; - // odt[2] = 1; - - boolean debugFlag = false; - if (odt[0] == 25646 && odt[1] == 4319) debugFlag = true; - - writeSkimsToFile(seq, outStreamObs, debugFlag, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); - - // set odt[2] to be each skim priod index (1,2,3) and write a - // separate - // output file - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - odt[2] = i + 1; - writeSkimsToFile(seq, outStreamObsTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, - loggers); - } - - // write inbound direction - odt[0] = obsOdt[1]; // dest - odt[1] = obsOdt[0]; // orig - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[obsOdt[3]]; // arrival - // skim - // period - // index - - outStreamObs.print(","); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamObsTod[i].print(","); - - writeSkimsToFile(seq, outStreamObs, debugFlag, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); - - // set odt[2] to be each skim priod index (1,2,3) and write a - // separate - // output file - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - odt[2] = i + 1; - writeSkimsToFile(seq, outStreamObsTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, - loggers); - } - - if (outStreamObs != null) - { - outStreamObs.println(""); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamObsTod[i].println(""); - } - - if (seq % 1000 == 0) logger.info("wrote OBS record: " + seq); - - seq++; - } - if (outStreamObs != null) - { - outStreamObs.close(); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamObsTod[i].close(); - } - } - - if (!outputFileNameHis.isEmpty()) - { - TableDataSet hisTds = getHomeInterviewSurveyTableDataSet(rbMap); - int[][] hisOdts = getHomeInterviewSurveyOrigDestTimes(hisTds); - - // write skims data for home interview survey records - int seq = 1; - for (int[] hisOdt : hisOdts) - { - // write outbound direction - odt[0] = hisOdt[0]; // orig - odt[1] = hisOdt[1]; // dest - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim - // period - // index - writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); - - // set odt[2] to be each skim priod index (1,2,3) and write a - // separate - // output file - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - odt[2] = i + 1; - writeSkimsToFile(seq, outStreamHisTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, - loggers); - } - - // write inbound direction - odt[0] = hisOdt[1]; // dest - odt[1] = hisOdt[0]; // orig - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[3]]; // arrival - // skim - // period - // index - - outStreamHis.print(","); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamHisTod[i].print(","); - - writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, defaultVOT, loggers); - - // set odt[2] to be each skim priod index (1,2,3) and write a - // separate - // output file - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - { - odt[2] = i + 1; - writeSkimsToFile(seq, outStreamHisTod[i], false, odt, anm, wtw, wtd, dtw, defaultVOT, - loggers); - } - - if (outStreamHis != null) - { - outStreamHis.println(""); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamHisTod[i].println(""); - } - - if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); - - seq++; - } - if (outStreamHis != null) - { - outStreamHis.close(); - for (int i = 0; i < SKIM_PERIOD_LABELS.length; i++) - outStreamHisTod[i].close(); - } - } - - } - - private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, - int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, - WalkTransitDriveSkimsCalculator wtd, DriveTransitWalkSkimsCalculator dtw, float vot, - Logger[] loggers) - { - - Logger autoLogger = loggers[0]; - Logger wtwLogger = loggers[1]; - Logger wtdLogger = loggers[2]; - Logger dtwLogger = loggers[3]; - - int[][] bestTapPairs = null; - double[][] returnedSkims = null; - - if (outStream != null) - outStream.print(String.format("%d,%d,%d,%d,%d", sequence, odt[4], odt[0], odt[1], - odt[2])); - - double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], vot,loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); - - if (outStream != null) - { - String autoRecord = getAutoSkimsRecord(skims); - outStream.print(autoRecord); - } - - skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); - - if (outStream != null) - { - String nmRecord = getAutoSkimsRecord(skims); - outStream.print(nmRecord); - } - - /* TODO: - * Fix the following code - - - BestTransitPathCalculator bestTransitPathCalculator = wtw.getBestPathUEC(); - - bestTransitPathCalculator.findBestWalkTransitWalkTaps(walkDmu, odt[2], odt[0], odt[1], loggingEnabled, wtwLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); - else - { - returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, bestPathUEC.getBestAccessTime(i), - bestPathUEC.getBestEgressTime(i), bestTapPairs[i][0], bestTapPairs[i][1], - odt[2], loggingEnabled); - } - } - if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - if (outStream != null) - { - String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.print(wtwRecord); - } - - bestTapPairs = wtd.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtdLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = wtd.getNullTransitSkims(i); - else - { - returnedSkims[i] = wtd.getWalkTransitDriveSkims(i, - bestPathUEC.getBestAccessTime(i), bestPathUEC.getBestEgressTime(i), - bestTapPairs[i][0], bestTapPairs[i][1], odt[2], loggingEnabled); - } - } - if (loggingEnabled) wtd.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - if (outStream != null) - { - String wtdRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.print(wtdRecord); - } - - bestTapPairs = dtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, dtwLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = dtw.getNullTransitSkims(i); - else - { - returnedSkims[i] = dtw.getDriveTransitWalkSkims(i, - bestPathUEC.getBestAccessTime(i), bestPathUEC.getBestEgressTime(i), - bestTapPairs[i][0], bestTapPairs[i][1], odt[2], loggingEnabled); - } - } - if (loggingEnabled) dtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - if (outStream != null) - { - String dtwRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.print(dtwRecord); - } - */ - } - - /** - * Start the matrix server - * - * @param rb - * is a ResourceBundle for the properties file for this - * application - */ - private void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[][] of skim values with the first dimesion the - * ride mode indices and second dimention the skim categories - */ - private String getTransitSkimsRecord(int[] odt, double[][] skims) - { - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - for (int j = 0; j < skims[i].length; j++) - tableRecord += String.format(",%.5f", skims[i][j]); - } - - return tableRecord; - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[] of skim values - */ - private String getAutoSkimsRecord(double[] skims) - { - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - tableRecord += String.format(",%.5f", skims[i]); - } - - return tableRecord; - - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getTransitSkimsHeaderRecord(String transitServiceLabel, String[] skimNames) - { - - Modes.TransitMode[] mode = Modes.TransitMode.values(); - - String heading = ""; - - for (int i = 0; i < mode.length; i++) - { - for (int j = 0; j < skimNames.length; j++) - heading += String.format(",%s_%s_%s", transitServiceLabel, mode[i], - skimNames[j]); - } - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getAutoSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - private TableDataSet getOnBoardSurveyTableDataSet(HashMap rbMap) - { - - String obsFileName = Util.getStringValueFromPropertyMap(rbMap, OBS_DATA_RECORDS_FILE_KEY); - if (obsFileName == null) - { - logger.error("Error getting the filename from the properties file for the Sandag on-board survey data records file."); - logger.error("Properties file target: " + OBS_DATA_RECORDS_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + OBS_DATA_RECORDS_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFile(new File(obsFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag on-board survey data records file: %s into TableDataSet object.", - obsFileName)); - throw new RuntimeException(e); - } - - } - - private TableDataSet getHomeInterviewSurveyTableDataSet(HashMap rbMap) - { - - String hisFileName = Util.getStringValueFromPropertyMap(rbMap, HIS_DATA_RECORDS_FILE_KEY); - if (hisFileName == null) - { - logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); - logger.error("Properties file target: " + HIS_DATA_RECORDS_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + HIS_DATA_RECORDS_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFile(new File(hisFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", - hisFileName)); - throw new RuntimeException(e); - } - - } - - private int[][] getOnBoardSurveyOrigDestTimes(TableDataSet obsTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[obsTds.getRowCount()][5]; - - int[] origs = obsTds.getColumnAsInt(OBS_ORIG_MGRA); - int[] dests = obsTds.getColumnAsInt(OBS_DEST_MGRA); - int[] departs = obsTds.getColumnAsInt(OBS_OUT_TOUR_PERIOD); - int[] arrives = obsTds.getColumnAsInt(OBS_IN_TOUR_PERIOD); - int[] ids = obsTds.getColumnAsInt(OBS_UNIQUE_ID); - - for (int r = 1; r <= obsTds.getRowCount(); r++) - { - odts[r - 1][0] = origs[r - 1]; - odts[r - 1][1] = dests[r - 1]; - odts[r - 1][2] = departs[r - 1]; - odts[r - 1][3] = arrives[r - 1]; - odts[r - 1][4] = ids[r - 1]; - } - - return odts; - } - - private int[][] getHomeInterviewSurveyOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][4]; - - int[] origs = hisTds.getColumnAsInt(HIS_ORIG_MGRA); - int[] dests = hisTds.getColumnAsInt(HIS_DEST_MGRA); - int[] departs = hisTds.getColumnAsInt(HIS_DEPART_PERIOD); - int[] arrives = hisTds.getColumnAsInt(HIS_ARRIVE_PERIOD); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - odts[r - 1][0] = origs[r - 1]; - odts[r - 1][1] = dests[r - 1]; - odts[r - 1][2] = departs[r - 1]; - odts[r - 1][3] = arrives[r - 1]; - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - SkimsAppender appender = new SkimsAppender(); - - appender.startMatrixServer(rb); - appender.runSkimsAppender(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java deleted file mode 100644 index deeb3b8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationEstimationMcLogsumsAppender.java +++ /dev/null @@ -1,331 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTripModeChoiceDMU; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -public final class StopLocationEstimationMcLogsumsAppender - extends McLogsumsAppender -{ - - private static final int DEBUG_EST_RECORD1 = 266; - private static final int DEBUG_EST_RECORD2 = -1; - - /* - * for stop location choice estimation file - */ - private static final int PORTION_FIELD = 1; - private static final int SAMPNO_FIELD = 2; - private static final int PERNO_FIELD = 3; - private static final int TOUR_ID_FIELD = 4; - private static final int TRIPNO_FIELD = 5; - private static final int STOPID_FIELD = 6; - private static final int STOPNO_FIELD = 7; - - private static final int ORIG_MGRA_FIELD = 16; - private static final int DEST_MGRA_FIELD = 23; - private static final int CHOSEN_MGRA_FIELD = 17; - private static final int MGRA1_FIELD = 49; - - private static final int TOUR_DEPART_PERIOD_FIELD = 40; - private static final int TOUR_ARRIVE_PERIOD_FIELD = 41; - private static final int TRIP_START_PERIOD_FIELD = 29; - private static final int TOUR_MODE_FIELD = 30; - private static final int INCOME_FIELD = 24; - private static final int ADULTS_FIELD = 47; - private static final int AUTOS_FIELD = 28; - private static final int HHSIZE_FIELD = 27; - private static final int GENDER_FIELD = 26; - private static final int OUT_STOPS_FIELD = 45; - private static final int IN_STOPS_FIELD = 46; - private static final int FIRST_TRIP_FIELD = 43; - private static final int LAST_TRIP_FIELD = 44; - private static final int PURPOSE_FIELD = 12; - private static final int AGE_FIELD = 25; - private static final int DIR_FIELD = 42; - private static final int J_TOUR_ID_FIELD = 10; - private static final int J_TOUR_PARTICIPANTS_FIELD = 11; - private static final int NUM_MGRA_FIELDS = 30; - - public StopLocationEstimationMcLogsumsAppender(HashMap rbMap) - { - super(rbMap); - - debugEstimationFileRecord1 = DEBUG_EST_RECORD1; - debugEstimationFileRecord2 = DEBUG_EST_RECORD2; - - numMgraFields = NUM_MGRA_FIELDS; - } - - private void runLogsumAppender(ResourceBundle rb) - { - - totalTime1 = 0; - totalTime2 = 0; - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - mgraSetForLogsums = new int[numMgraFields + 1]; - - // allocate the logsums array for the chosen destination alternative - tripModeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][2]; - - departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - "slc.est.skims.output.file"); - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - int dotIndex = outputFileName.indexOf("."); - String baseName = outputFileName.substring(0, dotIndex); - String extension = outputFileName.substring(dotIndex); - - String outputName = baseName + extension; - - PrintWriter outStream = null; - - try - { - outStream = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - writeDcFile(rbMap, outStream); - - logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); - logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); - - } - - private void writeDcFile(HashMap rbMap, PrintWriter outStream) - { - - outStream - .print("seq,portion,sampn,perno,tour_id,tripno,stopid,stopno,chosenMgra,chosenMgraLogsumIK,chosenMgraLogsumKJ"); - - // print each set of sample destMgra and the depart/arrive logsum - // fieldnames - // to file 1. - // print each set of sample destMgra and the chosen depart/arrive logsum - // fieldname to file 2. - for (int m = 1; m < tripModeChoiceLogsums.length; m++) - { - outStream.print(",sampleMgra_" + m); - outStream.print(",sampleLogsumIK_" + m); - outStream.print(",sampleLogsumKJ_" + m); - } - outStream.print("\n"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - SandagTripModeChoiceDMU mcDmuObject = new SandagTripModeChoiceDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - int[] mgraSet = mgras[i]; - - odtSet[0] = seq; - - outStream.print(seq + "," + odtSet[PORTION] + "," + odtSet[SAMPNO] + "," - + odtSet[PERNO] + "," + odtSet[TOUR_ID] + "," + odtSet[TRIPNO] + "," - + odtSet[STOPID] + "," + odtSet[STOPNO]); - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - - try - { - calculateTripModeChoiceLogsums(rbMap, mcModel[category], mcDmuObject, odtSet, - mgraSet); - } catch (Exception e) - { - logger.error("exception caught processing survey record for i = " + i); - throw new RuntimeException(); - } - - outStream.print("," + odtSet[CHOSEN_MGRA]); - outStream - .printf(",%.8f,%.8f", tripModeChoiceLogsums[0][0], tripModeChoiceLogsums[0][1]); - - // write logsum sets for each dest in the sample to file 1 - for (int m = 1; m < tripModeChoiceLogsums.length; m++) - { - outStream.print("," + mgraSet[m - 1]); - outStream.printf(",%.8f,%.8f", tripModeChoiceLogsums[m][0], - tripModeChoiceLogsums[m][1]); - } - outStream.print("\n"); - - if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); - - seq++; - } - - outStream.close(); - - } - - private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] tourDeparts = hisTds.getColumnAsInt(TOUR_DEPART_PERIOD_FIELD); - int[] tourArrives = hisTds.getColumnAsInt(TOUR_ARRIVE_PERIOD_FIELD); - int[] tripStarts = hisTds.getColumnAsInt(TRIP_START_PERIOD_FIELD); - - int[] direction = hisTds.getColumnAsInt(DIR_FIELD); - - int[] portion = hisTds.getColumnAsInt(PORTION_FIELD); - int[] sampno = hisTds.getColumnAsInt(SAMPNO_FIELD); - int[] perno = hisTds.getColumnAsInt(PERNO_FIELD); - int[] tour_id = hisTds.getColumnAsInt(TOUR_ID_FIELD); - int[] tripno = hisTds.getColumnAsInt(TRIPNO_FIELD); - int[] stopid = hisTds.getColumnAsInt(STOPID_FIELD); - int[] stopno = hisTds.getColumnAsInt(STOPNO_FIELD); - int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); - int[] jTourId = hisTds.getColumnAsInt(J_TOUR_ID_FIELD); - int[] jTourParticipants = hisTds.getColumnAsInt(J_TOUR_PARTICIPANTS_FIELD); - int[] income = hisTds.getColumnAsInt(INCOME_FIELD); - int[] mode = hisTds.getColumnAsInt(TOUR_MODE_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] chosen = hisTds.getColumnAsInt(CHOSEN_MGRA_FIELD); - int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); - int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); - int[] outStops = hisTds.getColumnAsInt(OUT_STOPS_FIELD); - int[] inStops = hisTds.getColumnAsInt(IN_STOPS_FIELD); - int[] firstTrip = hisTds.getColumnAsInt(FIRST_TRIP_FIELD); - int[] lastTrip = hisTds.getColumnAsInt(LAST_TRIP_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][PORTION] = portion[r - 1]; - odts[r - 1][SAMPNO] = sampno[r - 1]; - odts[r - 1][PERNO] = perno[r - 1]; - odts[r - 1][TOUR_ID] = tour_id[r - 1]; - odts[r - 1][TRIPNO] = tripno[r - 1]; - odts[r - 1][STOPID] = stopid[r - 1]; - odts[r - 1][STOPNO] = stopno[r - 1]; - - odts[r - 1][DEPART_PERIOD] = tourDeparts[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = tourArrives[r - 1]; - odts[r - 1][TRIP_PERIOD] = tripStarts[r - 1]; - - odts[r - 1][DIRECTION] = direction[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][CHOSEN_MGRA] = chosen[r - 1]; - odts[r - 1][TOUR_MODE] = mode[r - 1]; - odts[r - 1][INCOME] = income[r - 1]; - odts[r - 1][ADULTS] = adults[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][HHSIZE] = hhsize[r - 1]; - odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; - odts[r - 1][FIRST_TRIP] = firstTrip[r - 1]; - odts[r - 1][LAST_TRIP] = lastTrip[r - 1]; - odts[r - 1][OUT_STOPS] = outStops[r - 1]; - odts[r - 1][IN_STOPS] = inStops[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = purpose[r - 1]; - odts[r - 1][JOINT] = jTourId[r - 1] > 0 ? 1 : 0; - odts[r - 1][PARTYSIZE] = jTourParticipants[r - 1]; - - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - StopLocationEstimationMcLogsumsAppender appender = new StopLocationEstimationMcLogsumsAppender( - rbMap); - - appender.startMatrixServer(rb); - appender.runLogsumAppender(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java deleted file mode 100644 index b6ffadb..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StopLocationSampleCalculator.java +++ /dev/null @@ -1,502 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.File; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTripModeChoiceDMU; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.ResourceUtil; - -public final class StopLocationSampleCalculator - extends McLogsumsAppender -{ - - private static final int DEBUG_EST_RECORD = 5; - - /* - * for stop location choice estimation file - */ - private static final int PORTION_FIELD = 1; - private static final int SAMPNO_FIELD = 2; - private static final int PERNO_FIELD = 3; - private static final int TOUR_ID_FIELD = 4; - private static final int TRIPNO_FIELD = 5; - private static final int STOPID_FIELD = 6; - private static final int STOPNO_FIELD = 7; - - private static final int ORIG_MGRA_FIELD = 16; - private static final int DEST_MGRA_FIELD = 23; - private static final int CHOSEN_MGRA_FIELD = 17; - private static final int MGRA1_FIELD = 49; - - private static final int TOUR_DEPART_PERIOD_FIELD = 40; - private static final int TOUR_ARRIVE_PERIOD_FIELD = 41; - private static final int TRIP_START_PERIOD_FIELD = 29; - private static final int TOUR_MODE_FIELD = 30; - private static final int INCOME_FIELD = 24; - private static final int ADULTS_FIELD = 47; - private static final int AUTOS_FIELD = 28; - private static final int HHSIZE_FIELD = 27; - private static final int GENDER_FIELD = 26; - private static final int OUT_STOPS_FIELD = 45; - private static final int IN_STOPS_FIELD = 46; - private static final int FIRST_TRIP_FIELD = 43; - private static final int LAST_TRIP_FIELD = 44; - private static final int TOUR_PURPOSE_FIELD = 12; - private static final int STOP_PURPOSE_FIELD = 12; - private static final int AGE_FIELD = 25; - private static final int DIR_FIELD = 42; - private static final int J_TOUR_ID_FIELD = 10; - private static final int J_TOUR_PARTICIPANTS_FIELD = 11; - private static final int NUM_MGRA_FIELDS = 30; - - private static final String PROPERTIES_UEC_SLC_SOA_CHOICE = "slc.soa.uec.file"; - private static final String PROPERTIES_UEC_STOP_SOA_SIZE = "slc.soa.size.uec.file"; - private static final String PROPERTIES_UEC_STOP_SOA_SIZE_DATA = "slc.soa.size.uec.data.page"; - private static final String PROPERTIES_UEC_STOP_SOA_SIZE_MODEL = "slc.soa.size.uec.model.page"; - - private static final int WORK_STOP_PURPOSE_INDEX = 1; - private static final int UNIV_STOP_PURPOSE_INDEX = 2; - private static final int ESCORT_STOP_PURPOSE_INDEX = 4; - private static final int SHOP_STOP_PURPOSE_INDEX = 5; - private static final int MAINT_STOP_PURPOSE_INDEX = 6; - private static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; - private static final int VISIT_STOP_PURPOSE_INDEX = 8; - private static final int DISCR_STOP_PURPOSE_INDEX = 9; - private static final int MAX_STOP_PURPOSE_INDEX = 9; - - private static final int WORK_STOP_PURPOSE_SOA_SIZE_INDEX = 0; - private static final int UNIV_STOP_PURPOSE_SOA_SIZE_INDEX = 1; - private static final int ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX = 2; - private static final int ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX = 3; - private static final int ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 4; - private static final int ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 5; - private static final int SHOP_STOP_PURPOSE_SOA_SIZE_INDEX = 6; - private static final int MAINT_STOP_PURPOSE_SOA_SIZE_INDEX = 7; - private static final int EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX = 8; - private static final int VISIT_STOP_PURPOSE_SOA_SIZE_INDEX = 9; - private static final int DISCR_STOP_PURPOSE_SOA_SIZE_INDEX = 10; - - private static final int AUTO_DIST_SKIM_INDEX = 2; - - private McLogsumsCalculator logsumHelper; - - private static final float defaultVOT = 15.0f; - - public StopLocationSampleCalculator(HashMap rbMap) - { - super(rbMap); - debugEstimationFileRecord1 = DEBUG_EST_RECORD; - - } - - private void runSampleProbabilitiesCalculator(ResourceBundle rb) - { - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(rbMap); - - SandagTripModeChoiceDMU mcDmuObject = new SandagTripModeChoiceDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - - /* - * SandagStopDCSoaDMU slcSoaDmuObject = new - * SandagStopDCSoaDMU(modelStructure); - * - * String slcSoaUecFile = rbMap.get(PROPERTIES_UEC_SLC_SOA_CHOICE); - * slcSoaUecFile = uecPath + slcSoaUecFile; - * - * ChoiceModelApplication[] slcSoaModel = new - * ChoiceModelApplication[MAX_STOP_PURPOSE_INDEX+1]; - * slcSoaModel[WORK_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, WORK_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[UNIV_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, UNIV_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[ESCORT_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, ESCORT_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[SHOP_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, SHOP_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[EAT_OUT_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, EAT_OUT_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[MAINT_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, MAINT_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[VISIT_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, VISIT_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - * slcSoaModel[DISCR_STOP_PURPOSE_INDEX] = new - * ChoiceModelApplication(slcSoaUecFile, DISCR_STOP_PURPOSE_INDEX, 0, - * rbMap, (VariableTable) slcSoaDmuObject); - */ - - double absoulteDistanceDeviationCoefficient = -0.05; - - int i = DEBUG_EST_RECORD - 1; - - int[] odtSet = estDataOdts[i]; - - int mgra = mgras[i][24]; - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - // int stopPurpose = odtSet[STOP_PURPOSE]; - int stopPurpose = 6; - - double[] logsums = null; - try - { - logsums = calculateTripModeChoiceLogsumForEstimationRecord(rbMap, mcModel[category], - mcDmuObject, odtSet, mgra); - } catch (Exception e) - { - logger.error("exception caught calculating trip mode choice logsum for survey record i = " - + (i + 1)); - throw new RuntimeException(); - } - - double[] slcSoaSizeTerms = null; - try - { - boolean preSchoolInHh = false; - boolean gradeSchoolInHh = false; - boolean highSchoolInHh = false; - double[][] sizeTerms = calculateSlcSoaSizeTerms(rbMap, mgra); - slcSoaSizeTerms = getSlcSoaSizeTermsForStopPurpose(stopPurpose, preSchoolInHh, - gradeSchoolInHh, highSchoolInHh, sizeTerms); - } catch (Exception e) - { - logger.error("exception caught calculating stop location choice size terms for survey record i = " - + (i + 1)); - throw new RuntimeException(); - } - - try - { - int origMgra = odtSet[ORIG_MGRA]; - int destMgra = odtSet[DEST_MGRA]; - int departPeriod = odtSet[TRIP_PERIOD]; // depart period - int skimPeriodIndex = modelStructure.getSkimPeriodIndex(departPeriod) + 1; // depart - // skim - - double odDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(origMgra, destMgra, - skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; - double osDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(origMgra, mgra, - skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; - double sdDist = logsumHelper.getAnmSkimCalculator().getAutoSkims(mgra, destMgra, - skimPeriodIndex, defaultVOT, false, logger)[AUTO_DIST_SKIM_INDEX]; - double distance = osDist + sdDist - odDist; - int availability = 1; - - boolean walkTransitIsAvailable = false; - if (mgraManager.getMgraWlkTapsDistArray()[mgra][0] != null) - walkTransitIsAvailable = true; - - double util = -999; - if (availability == 1) - util = Math.log(slcSoaSizeTerms[mgra]) + absoulteDistanceDeviationCoefficient - * distance; - - logUtilityCalculation(origMgra, mgra, destMgra, departPeriod, skimPeriodIndex, osDist, - sdDist, odDist, distance, slcSoaSizeTerms[mgra], util, logsums); - } catch (Exception e) - { - logger.error("exception caught calculating and logging utility calculation for survey record i = " - + (i + 1)); - throw new RuntimeException(); - } - - } - - private void logUtilityCalculation(int origMgra, int sampleMgra, int destMgra, - int departPeriodIndex, int skimPeriodIndex, double osDist, double sdDist, - double odDist, double distance, double size, double util, double[] logsums) - { - - // write UEC calculation results to logsum specific log file if - // its the chosen dest and its the chosen time combo - slcSoaLogger.info("Stop Location Sample Probabilities Calculation:"); - slcSoaLogger.info(""); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info("origin mgra = " + origMgra); - slcSoaLogger.info("sample destination mgra = " + sampleMgra); - slcSoaLogger.info("final destination mgra = " + destMgra); - slcSoaLogger.info("origin taz = " + mgraManager.getTaz(origMgra)); - slcSoaLogger.info("sample destination taz = " + mgraManager.getTaz(sampleMgra)); - slcSoaLogger.info("final destination taz = " + mgraManager.getTaz(destMgra)); - slcSoaLogger.info("depart period interval = " + departPeriodIndex); - slcSoaLogger.info("skim period index = " + skimPeriodIndex); - slcSoaLogger.info("orig to stop distance = " + osDist); - slcSoaLogger.info("stop to dest distance = " + sdDist); - slcSoaLogger.info("orig to dest distance = " + odDist); - slcSoaLogger.info("distance = " + distance); - slcSoaLogger.info("size = " + size); - slcSoaLogger.info(""); - slcSoaLogger.info("util = size * exp ( -0.05*distance )"); - slcSoaLogger.info("util = " + util); - slcSoaLogger.info("os logsum = " + logsums[0]); - slcSoaLogger.info("sd logsum = " + logsums[1]); - slcSoaLogger - .info("--------------------------------------------------------------------------------------------------------"); - slcSoaLogger.info(""); - - } - - private double[] getSlcSoaSizeTermsForStopPurpose(int stopPurpose, boolean preSchoolInHh, - boolean gradeSchoolInHh, boolean highSchoolInHh, double[][] sizeTerms) - { - - double[] slcSoaSizeTerms = null; - switch (stopPurpose) - { - - case WORK_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case UNIV_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case ESCORT_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; - - // add preschool size term if the hh has a preschool child - if (preSchoolInHh) - { - for (int j = 0; j < sizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) - slcSoaSizeTerms[j] += sizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; - } - - // add grade school size term if the hh has a grade school child - if (gradeSchoolInHh) - { - for (int j = 0; j < sizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) - slcSoaSizeTerms[j] += sizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; - } - - // add high school size term if the hh has a high school child - if (highSchoolInHh) - { - for (int j = 0; j < sizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX].length; j++) - slcSoaSizeTerms[j] += sizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX][j]; - } - break; - case SHOP_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case MAINT_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case EAT_OUT_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case VISIT_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case DISCR_STOP_PURPOSE_INDEX: - slcSoaSizeTerms = sizeTerms[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - } - - return slcSoaSizeTerms; - - } - - private double[][] calculateSlcSoaSizeTerms(HashMap rbMap, int sampleMgra) - { - - logger.info(""); - logger.info(""); - logger.info("Calculating Stop Location SOA Size Terms"); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String slcSoaSizeUecFile = rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE); - slcSoaSizeUecFile = uecPath + slcSoaSizeUecFile; - int slcSoaSizeUecData = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE_DATA)); - int slcSoaSizeUecModel = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_SOA_SIZE_MODEL)); - - IndexValues iv = new IndexValues(); - UtilityExpressionCalculator slcSoaSizeUec = new UtilityExpressionCalculator(new File( - slcSoaSizeUecFile), slcSoaSizeUecModel, slcSoaSizeUecData, rbMap, null); - - ArrayList mgras = mgraManager.getMgras(); - int maxMgra = mgraManager.getMaxMgra(); - int alternatives = slcSoaSizeUec.getNumberOfAlternatives(); - double[][] slcSoaSize = new double[alternatives][maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - iv.setZoneIndex(mgra); - double[] utilities = slcSoaSizeUec.solve(iv, null, null); - - if (mgra == sampleMgra) - slcSoaSizeUec.logAnswersArray(slcSoaLogger, "Stop Location SOA Size Terms, MGRA = " - + mgra); - - // store the size terms - for (int i = 0; i < alternatives; i++) - slcSoaSize[i][mgra] = utilities[i]; - - } - - return slcSoaSize; - - } - - private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] tourDeparts = hisTds.getColumnAsInt(TOUR_DEPART_PERIOD_FIELD); - int[] tourArrives = hisTds.getColumnAsInt(TOUR_ARRIVE_PERIOD_FIELD); - int[] tripStarts = hisTds.getColumnAsInt(TRIP_START_PERIOD_FIELD); - - int[] direction = hisTds.getColumnAsInt(DIR_FIELD); - - int[] portion = hisTds.getColumnAsInt(PORTION_FIELD); - int[] sampno = hisTds.getColumnAsInt(SAMPNO_FIELD); - int[] perno = hisTds.getColumnAsInt(PERNO_FIELD); - int[] tour_id = hisTds.getColumnAsInt(TOUR_ID_FIELD); - int[] tripno = hisTds.getColumnAsInt(TRIPNO_FIELD); - int[] stopid = hisTds.getColumnAsInt(STOPID_FIELD); - int[] stopno = hisTds.getColumnAsInt(STOPNO_FIELD); - int[] tourPurpose = hisTds.getColumnAsInt(TOUR_PURPOSE_FIELD); - int[] stopPurpose = hisTds.getColumnAsInt(STOP_PURPOSE_FIELD); - int[] jTourId = hisTds.getColumnAsInt(J_TOUR_ID_FIELD); - int[] jTourParticipants = hisTds.getColumnAsInt(J_TOUR_PARTICIPANTS_FIELD); - int[] income = hisTds.getColumnAsInt(INCOME_FIELD); - int[] mode = hisTds.getColumnAsInt(TOUR_MODE_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] chosen = hisTds.getColumnAsInt(CHOSEN_MGRA_FIELD); - int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); - int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); - int[] outStops = hisTds.getColumnAsInt(OUT_STOPS_FIELD); - int[] inStops = hisTds.getColumnAsInt(IN_STOPS_FIELD); - int[] firstTrip = hisTds.getColumnAsInt(FIRST_TRIP_FIELD); - int[] lastTrip = hisTds.getColumnAsInt(LAST_TRIP_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][PORTION] = portion[r - 1]; - odts[r - 1][SAMPNO] = sampno[r - 1]; - odts[r - 1][PERNO] = perno[r - 1]; - odts[r - 1][TOUR_ID] = tour_id[r - 1]; - odts[r - 1][TRIPNO] = tripno[r - 1]; - odts[r - 1][STOPID] = stopid[r - 1]; - odts[r - 1][STOPNO] = stopno[r - 1]; - - odts[r - 1][DEPART_PERIOD] = tourDeparts[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = tourArrives[r - 1]; - odts[r - 1][TRIP_PERIOD] = tripStarts[r - 1]; - - odts[r - 1][DIRECTION] = direction[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][CHOSEN_MGRA] = chosen[r - 1]; - odts[r - 1][TOUR_MODE] = mode[r - 1]; - odts[r - 1][INCOME] = income[r - 1]; - odts[r - 1][ADULTS] = adults[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][HHSIZE] = hhsize[r - 1]; - odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; - odts[r - 1][FIRST_TRIP] = firstTrip[r - 1]; - odts[r - 1][LAST_TRIP] = lastTrip[r - 1]; - odts[r - 1][OUT_STOPS] = outStops[r - 1]; - odts[r - 1][IN_STOPS] = inStops[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = tourPurpose[r - 1]; - odts[r - 1][STOP_PURPOSE] = stopPurpose[r - 1]; - odts[r - 1][JOINT] = jTourId[r - 1] > 0 ? 1 : 0; - odts[r - 1][PARTYSIZE] = jTourParticipants[r - 1]; - - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - StopLocationSampleCalculator appender = new StopLocationSampleCalculator(rbMap); - - appender.startMatrixServer(rb); - appender.runSampleProbabilitiesCalculator(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java deleted file mode 100644 index 0f37752..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredTransitSkimData.java +++ /dev/null @@ -1,51 +0,0 @@ -package org.sandag.abm.accessibilities; - - -public class StoredTransitSkimData -{ - - private static StoredTransitSkimData objInstance = null; - private int numSets; - - // these arrays are shared by McLogsumsAppender objects and are used by wtw, wtd, and dtw calculators. - private double[][][][][] storedWtwDepartPeriodTapTapSkims; - private double[][][][][] storedWtdDepartPeriodTapTapSkims; - private double[][][][][] storedDtwDepartPeriodTapTapSkims; - - - private StoredTransitSkimData(){ - this.numSets = numSets; - } - - public static synchronized StoredTransitSkimData getInstance( int numSets, int numPeriods, int maxTap ) - { - if (objInstance == null) { - objInstance = new StoredTransitSkimData(); - objInstance.setupStoredDataArrays(numSets, numPeriods, maxTap ); - return objInstance; - } - else { - return objInstance; - } - } - - private void setupStoredDataArrays(int numSets, int numPeriods, int maxTap ){ - storedWtwDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; - storedWtdDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; - storedDtwDepartPeriodTapTapSkims = new double[numSets][numPeriods + 1][maxTap + 1][][]; - } - - public double[][][][][] getStoredWtwDepartPeriodTapTapSkims() { - return storedWtwDepartPeriodTapTapSkims; - } - - public double[][][][][] getStoredWtdDepartPeriodTapTapSkims() { - return storedWtdDepartPeriodTapTapSkims; - } - - public double[][][][][] getStoredDtwDepartPeriodTapTapSkims() { - return storedDtwDepartPeriodTapTapSkims; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java deleted file mode 100644 index 51644a5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/StoredUtilityData.java +++ /dev/null @@ -1,133 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.util.concurrent.ConcurrentHashMap; -import java.util.HashMap; - - -public class StoredUtilityData -{ - - private static StoredUtilityData objInstance = null; - public static final float default_utility = -999; - - // these arrays are shared by multiple BestTransitPathCalculator objects in a distributed computing environment - private float[][] storedWalkAccessUtils; // dim#1: MGRA id, dim#2: TAP id - private float[][] storedDriveAccessUtils; // dim#1: TAZ id, dim#2: TAP id - private float[][] storedWalkEgressUtils; // dim#1: TAP id, dim#2: MGRA id - private float[][] storedDriveEgressUtils; // dim#1: TAP id, dim#2: TAZ id - - // {0:WTW, 1:WTD, 2:DTW} -> TOD period number -> pTAP*100000+aTAP -> utility - private HashMap>> storedDepartPeriodTapTapUtils; - - - private StoredUtilityData(){ - } - - public static synchronized StoredUtilityData getInstance( int maxMgra, int maxTap, int maxTaz, int[] accEgrSegments, int[] periods) - { - if (objInstance == null) { - objInstance = new StoredUtilityData(); - objInstance.setupStoredDataArrays( maxMgra, maxTap, maxTaz, accEgrSegments, periods); - return objInstance; - } - else { - return objInstance; - } - } - - private void setupStoredDataArrays( int maxMgra, int maxTap, int maxTaz, int[] accEgrSegments, int[] periods){ - // dimension the arrays - storedWalkAccessUtils = new float[maxMgra + 1][maxTap + 1]; - storedDriveAccessUtils = new float[maxTaz + 1][maxTap + 1]; - storedWalkEgressUtils = new float[maxTap + 1][maxMgra + 1]; - storedDriveEgressUtils = new float[maxTap + 1][maxTaz + 1]; - // assign default values to array elements - for (int i=0; i<=maxMgra; i++) - for (int j=0; j<=maxTap; j++) { - storedWalkAccessUtils[i][j] = default_utility; - storedWalkEgressUtils[j][i] = default_utility; - } - // assign default values to array elements - for (int i=0; i<=maxTaz; i++) - for (int j=0; j<=maxTap; j++) { - storedDriveAccessUtils[i][j] = default_utility; - storedDriveEgressUtils[j][i] = default_utility; - } - - //put into concurrent hashmap - storedDepartPeriodTapTapUtils = new HashMap>>(); - for(int i=0; i>()); - for(int j=0; j> hm = storedDepartPeriodTapTapUtils.get(accEgrSegments[i]); - hm.put(periods[j], new ConcurrentHashMap()); //key method paTapKey below - } - } - } - - public float[][] getStoredWalkAccessUtils() { - return storedWalkAccessUtils; - } - - public float[][] getStoredDriveAccessUtils() { - return storedDriveAccessUtils; - } - - public float[][] getStoredWalkEgressUtils() { - return storedWalkEgressUtils; - } - - public float[][]getStoredDriveEgressUtils() { - return storedDriveEgressUtils; - } - - public HashMap>> getStoredDepartPeriodTapTapUtils() { - return storedDepartPeriodTapTapUtils; - } - - //create p to a hash key - up to 99,999 - public long paTapKey(int p, int a) { - return(p * 100000 + a); - } - - //convert double array to float array - public float[] d2f(double[] d) { - float[] f = new float[d.length]; - for(int i=0; i rbMap) - { - super(rbMap); - - debugEstimationFileRecord1 = DEBUG_EST_RECORD1; - debugEstimationFileRecord2 = DEBUG_EST_RECORD2; - - } - - private void runLogsumAppender(ResourceBundle rb) - { - - totalTime1 = 0; - totalTime2 = 0; - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - mgraSetForLogsums = new int[numMgraFields + 1]; - - // allocate the logsums array for the chosen destination alternative - modeChoiceLogsums = new double[1][]; - - departArriveLogsums = new double[1][departArriveCombinations.length]; - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - "tod.est.skims.output.file"); - - PrintWriter outStream = null; - - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - try - { - outStream = new PrintWriter( - new BufferedWriter(new FileWriter(new File(outputFileName)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - writeTodFile(rbMap, outStream); - - logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); - logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); - - } - - private void writeTodFile(HashMap rbMap, PrintWriter outStream2) - { - - // print the chosen destMgra and the depart/arrive logsum field names to - // the - // file - outStream2.print("seq,hisseq,chosenMgra"); - for (String[] labels : departArriveCombinationLabels) - { - outStream2.print(",logsum" + labels[0] + labels[1]); - } - outStream2.print("\n"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getTodEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - int[] mgraSet = mgras[i]; - - odtSet[0] = seq; - - outStream2.print(seq + "," + odtSet[SAMPNO]); - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - - int[] departAvailable = {-1, 1, 1, 1, 1, 1}; - int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; - calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], - mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); - - // write chosen dest and logsums to both files - outStream2.print("," + odtSet[DEST_MGRA]); - for (double logsum : departArriveLogsums[0]) - { - outStream2.printf(",%.8f", logsum); - } - outStream2.print("\n"); - - if (seq % 1000 == 0) logger.info("wrote TOD Estimation file record: " + seq); - - seq++; - } - - outStream2.close(); - - } - - private int[][] getTodEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); - int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); - - int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] workTourMode = hisTds.getColumnAsInt(WORK_TOUR_MODE_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][SAMPNO] = hisseq[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = SUBTOUR_PURPOSE; - - odts[r - 1][DEPART_PERIOD] = departs[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][WORK_TOUR_MODE] = workTourMode[r - 1]; - - } - - return odts; - } - - public static void main(String[] args) - { - - long startTime = System.currentTimeMillis(); - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - SubtourTodEstimationMcLogsumsAppender appender = new SubtourTodEstimationMcLogsumsAppender( - rbMap); - - appender.startMatrixServer(rb); - appender.runLogsumAppender(rb); - - /* - * used this to read/parse the UEC expressions - debugging the UEC - * sheet. - * - * HashMap rbMap = - * ResourceUtil.changeResourceBundleIntoHashMap(rb); - * - * String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - * String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - * mcUecFile = uecPath + mcUecFile; - * - * ModelStructure modelStructure = new SandagModelStructure(); - * SandagAppendMcLogsumDMU mcDmuObject = new - * SandagAppendMcLogsumDMU(modelStructure); ChoiceModelApplication - * mcModel = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, - * rbMap, (VariableTable) mcDmuObject); - */ - - System.out.println("total runtime = " + ((System.currentTimeMillis() - startTime) / 1000) - + " seconds."); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java deleted file mode 100644 index c3985af..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TransitPath.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.sandag.abm.accessibilities; - -public class TransitPath implements Comparable{ - - public int oMaz; - public int dMaz; - public int pTap; - public int aTap; - public int set; - public int accEgr; - public float accUtil; - public float tapTapUtil; - public float egrUtil; - public static final int NA = -999; - - public TransitPath(int oMaz, int dMaz, int pTap, int aTap, int set, int accEgr, float accUtil, float tapTapUtil, float egrUtil) { - this.oMaz = oMaz; - this.dMaz = dMaz; - this.pTap = pTap; - this.aTap = aTap; - this.set = set; - this.accEgr = accEgr; - this.accUtil = accUtil; - this.tapTapUtil = tapTapUtil; - this.egrUtil = egrUtil; - } - - public float getTotalUtility() { - return(accUtil + tapTapUtil + egrUtil); - } - - @Override - public int compareTo(TransitPath o) { - - //return compareTo value - if ( getTotalUtility() < o.getTotalUtility() ) { - return -1; - } else if (getTotalUtility() == o.getTotalUtility()) { - return 0; - } else { - return 1; - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java deleted file mode 100644 index dab1318..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/TripSkimsAppender.java +++ /dev/null @@ -1,569 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -public final class TripSkimsAppender -{ - - protected transient Logger logger = Logger.getLogger(TripSkimsAppender.class); - - /* - * for trip mode choice estimation files - */ - private static final String HIS_DATA_RECORDS_FILE_KEY = "homeInterview.survey.file"; - - // private static final int HIS_SAMPNO = 2; - // private static final int HIS_ORIG_MGRA = 188; - // private static final int HIS_DEST_MGRA = 189; - // private static final int HIS_DEPART_PERIOD = 192; - - private static final int HIS_SAMPNO = 1; - private static final int HIS_ORIG_MGRA = 2; - private static final int HIS_DEST_MGRA = 3; - private static final int HIS_DEPART_PERIOD = 5; - - // private static final int HIS_SAMPNO = 2; - // private static final int HIS_ORIG_MGRA = 247; - // private static final int HIS_DEST_MGRA = 248; - // private static final int HIS_DEPART_PERIOD = 325; - - // survey periods are: - // 0=not used, - // 1=03:00-05:59, - // 2=06:00-08:59, - // 3=09:00-11:59, - // 4=12:00-15:29, - // 5=15:30-18:59, - // 6=19:00-02:59 - // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), - // 6=3(OP) - - // define a conversion array to convert period values in the survey file to - // skim - // period indices used in this propgram: 1=am peak, 2=pm peak, 3=off-peak. - private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; - private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; - - private static int debugOrigMgra = 0; - private static int debugDestMgra = 0; - private static int departModelPeriod = 0; - - private MatrixDataServerIf ms; - private BestTransitPathCalculator bestPathUEC; - - private static final float defaultVOT = 15.0f; - - private TripSkimsAppender() - { - } - - private void runSkimsAppender(ResourceBundle rb) - { - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - Logger autoLogger = Logger.getLogger("auto"); - Logger wtwLogger = Logger.getLogger("wtw"); - Logger wtdLogger = Logger.getLogger("wtd"); - Logger dtwLogger = Logger.getLogger("dtw"); - - String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, - "his.trip.skims.output.file"); - - FileWriter writer; - PrintWriter outStreamHis = null; - - McLogsumsAppender logsumHelper = new McLogsumsAppender(rbMap); - bestPathUEC = logsumHelper.getBestTransitPathCalculator(); - - AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); - WalkTransitWalkSkimsCalculator wtw = new WalkTransitWalkSkimsCalculator(rbMap); - WalkTransitDriveSkimsCalculator wtd = new WalkTransitDriveSkimsCalculator(rbMap); - DriveTransitWalkSkimsCalculator dtw = new DriveTransitWalkSkimsCalculator(rbMap); - - String heading = "seq,sampno"; - - heading += ",origMgra,destMgra,departPeriod"; - heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); - heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); - heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); - heading += getTransitSkimsHeaderRecord("wtd", wtd.getSkimNames()); - heading += getTransitSkimsHeaderRecord("dtw", dtw.getSkimNames()); - - try - { - writer = new FileWriter(new File(outputFileNameHis)); - outStreamHis = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileNameHis)); - throw new RuntimeException(e); - } - outStreamHis.println("his" + heading); - - Logger[] loggers = new Logger[4]; - loggers[0] = autoLogger; - loggers[1] = wtwLogger; - loggers[2] = wtdLogger; - loggers[3] = dtwLogger; - - int[] odt = new int[4]; - - TableDataSet hisTds = getHomeInterviewSurveyTableDataSet(rbMap); - int[][] hisOdts = getHomeInterviewSurveyOrigDestTimes(hisTds); - // 11100, pnrCoaster, mgra 4357 = taz 3641, mgra 26931 = taz 1140 - // int[][] hisOdts = { { 4357, 26931, 5, 0 } }; - // 25040, wlkCoaster, mgra 4989 = taz 3270, mgra 7796 = taz 1986 - // int[][] hisOdts = { { 7796, 4989, 2, 0 } }; - - // if ( debugOrigMgra <= 0 || debugDestMgra <= 0 || departModelPeriod <= - // 0 || departModelPeriod > 6 ) - // { - // logger.error("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); - // System.exit(-1); - // } - // int[][] hisOdts = { { debugOrigMgra, debugDestMgra, - // departModelPeriod, 0 } }; - - // write skims data for home interview survey records - int seq = 1; - for (int[] hisOdt : hisOdts) - { - // write outbound direction - odt[0] = hisOdt[0]; // orig - odt[1] = hisOdt[1]; // dest - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim - // period - odt[3] = hisOdt[3]; - - try - { - - int dummy = 0; - if (seq == 3424) - { - dummy = 1; - } - - writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, wtd, dtw, loggers); - } catch (Exception e) - { - logger.error("Exception caught processing record: " + seq + " of " + hisOdts.length - + "."); - break; - } - - if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); - - seq++; - } - - outStreamHis.close(); - - } - - private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, - int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, - WalkTransitDriveSkimsCalculator wtd, DriveTransitWalkSkimsCalculator dtw, - Logger[] loggers) - { - - Logger autoLogger = loggers[0]; - Logger wtwLogger = loggers[1]; - Logger wtdLogger = loggers[2]; - Logger dtwLogger = loggers[3]; - - int[][] bestTapPairs = null; - double[][] returnedSkims = null; - - outStream.print(String.format("%d,%d,%d,%d,%d", sequence, odt[3], odt[0], odt[1], odt[2])); - - double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], defaultVOT, loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); - - String autoRecord = getAutoSkimsRecord(skims); - outStream.print(autoRecord); - - skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); - - String nmRecord = getAutoSkimsRecord(skims); - outStream.print(nmRecord); - - /* - * TODO: Fix this code - - bestTapPairs = wtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtwLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); - else - { - returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, BestTransitPathCalculator - .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), - BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], - bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], - odt[2], loggingEnabled); - } - } - if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.print(wtwRecord); - - bestTapPairs = wtd.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtdLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = wtd.getNullTransitSkims(i); - else - { - returnedSkims[i] = wtd.getWalkTransitDriveSkims(i, BestTransitPathCalculator - .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), - BestTransitPathCalculator.findDriveTransitEgressTime(odt[1], - bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], - odt[2], loggingEnabled); - } - } - if (loggingEnabled) wtd.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - String wtdRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.print(wtdRecord); - - bestTapPairs = dtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, dtwLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = dtw.getNullTransitSkims(i); - else - { - returnedSkims[i] = dtw.getDriveTransitWalkSkims(i, BestTransitPathCalculator - .findDriveTransitAccessTime(odt[0], bestTapPairs[i][0]), - BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], - bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], - odt[2], loggingEnabled); - } - } - if (loggingEnabled) dtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - String dtwRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.println(dtwRecord); - */ - } - - /** - * Start the matrix server - * - * @param rb - * is a ResourceBundle for the properties file for this - * application - */ - private void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[][] of skim values with the first dimesion the - * ride mode indices and second dimention the skim categories - */ - private String getTransitSkimsRecord(int[] odt, double[][] skims) - { - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - for (int j = 0; j < skims[i].length; j++) - tableRecord += String.format(",%.5f", skims[i][j]); - } - - return tableRecord; - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[] of skim values - */ - private String getAutoSkimsRecord(double[] skims) - { - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - tableRecord += String.format(",%.5f", skims[i]); - } - - return tableRecord; - - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getTransitSkimsHeaderRecord(String transitServiveLabel, String[] skimNames) - { - - Modes.TransitMode[] mode = Modes.TransitMode.values(); - - String heading = ""; - - for (int i = 0; i < mode.length; i++) - { - for (int j = 0; j < skimNames.length; j++) - heading += String.format(",%s_%s_%s", transitServiveLabel, mode[i], - skimNames[j]); - - } - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getAutoSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - private TableDataSet getHomeInterviewSurveyTableDataSet(HashMap rbMap) - { - - String hisFileName = Util.getStringValueFromPropertyMap(rbMap, HIS_DATA_RECORDS_FILE_KEY); - if (hisFileName == null) - { - logger.error("Error getting the filename from the properties file for the Sandag home interview survey data records file."); - logger.error("Properties file target: " + HIS_DATA_RECORDS_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + HIS_DATA_RECORDS_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFile(new File(hisFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag home interview survey data records file: %s into TableDataSet object.", - hisFileName)); - throw new RuntimeException(e); - } - - } - - private int[][] getHomeInterviewSurveyOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure period(1-6), and his sampno. - int[][] odts = new int[hisTds.getRowCount()][4]; - - int[] sampnos = hisTds.getColumnAsInt(HIS_SAMPNO); - int[] origs = hisTds.getColumnAsInt(HIS_ORIG_MGRA); - int[] dests = hisTds.getColumnAsInt(HIS_DEST_MGRA); - int[] departs = hisTds.getColumnAsInt(HIS_DEPART_PERIOD); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - odts[r - 1][0] = origs[r - 1]; - odts[r - 1][1] = dests[r - 1]; - odts[r - 1][2] = departs[r - 1]; - odts[r - 1][3] = sampnos[r - 1]; - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb = null; - if (args.length == 0) - { - System.out - .println(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else if (args.length == 4) - { - rb = ResourceBundle.getBundle(args[0]); - - debugOrigMgra = Integer.parseInt(args[1]); - debugDestMgra = Integer.parseInt(args[2]); - departModelPeriod = Integer.parseInt(args[3]); - } else - { - System.out - .println("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); - System.exit(-1); - } - - try - { - - MatrixDataServerIf ms = null; - String serverAddress = null; - int serverPort = -1; - - HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - System.out.println(""); - System.out.println(""); - serverAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - - String serverPortString = (String) propertyMap.get("RunModel.MatrixServerPort"); - if (serverPortString != null) serverPort = Integer.parseInt(serverPortString); - - if (serverAddress != null && serverPort > 0) - { - try - { - System.out.println("attempting connection to matrix server " + serverAddress - + ":" + serverPort); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - System.out.println("connected to matrix server " + serverAddress + ":" - + serverPort); - - } catch (Exception e) - { - System.out - .println("exception caught running ctramp model components -- exiting."); - e.printStackTrace(); - throw new RuntimeException(); - } - } - - TazDataManager tazs = TazDataManager.getInstance(propertyMap); - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TapDataManager tapManager = TapDataManager.getInstance(propertyMap); - - // create an appender object and run it - TripSkimsAppender appender = new TripSkimsAppender(); - appender.runSkimsAppender(rb); - - } catch (RuntimeException e) - { - e.printStackTrace(); - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java deleted file mode 100644 index dafb12e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/VisitorTourLocationChoiceAppender.java +++ /dev/null @@ -1,367 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.application.SandagAppendMcLogsumDMU; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -public final class VisitorTourLocationChoiceAppender - extends McLogsumsAppender -{ - - private static final int DEBUG_EST_RECORD1 = 1; - private static final int DEBUG_EST_RECORD2 = -1; - - /* - * for DC estimation file - */ - private static final int SEQ_FIELD = 2; - - // for Atwork subtour DC - // private static final int ORIG_MGRA_FIELD = 79; - // private static final int DEST_MGRA_FIELD = 220; - // private static final int MGRA1_FIELD = 221; - // private static final int PURPOSE_INDEX_OFFSET = 4; - - // for Escort DC - private static final int ORIG_MGRA_FIELD = 76; - private static final int DEST_MGRA_FIELD = 79; - private static final int MGRA1_FIELD = 217; - private static final int PURPOSE_INDEX_OFFSET = 0; - - // for NonMandatory DC - // private static final int ORIG_MGRA_FIELD = 76; - // private static final int DEST_MGRA_FIELD = 79; - // private static final int MGRA1_FIELD = 221; - // private static final int PURPOSE_INDEX_OFFSET = 0; - - private static final int DEPART_PERIOD_FIELD = 189; - private static final int ARRIVE_PERIOD_FIELD = 190; - private static final int INCOME_FIELD = 20; - private static final int ADULTS_FIELD = 32; - private static final int AUTOS_FIELD = 6; - private static final int HHSIZE_FIELD = 5; - private static final int GENDER_FIELD = 38; - private static final int AGE_FIELD = 39; - private static final int PURPOSE_FIELD = 80; - private static final int JOINT_ID_FIELD = 125; - private static final int JOINT_PURPOSE_FIELD = 126; - private static final int JOINT_P1_FIELD = 151; - private static final int JOINT_P2_FIELD = 152; - private static final int JOINT_P3_FIELD = 153; - private static final int JOINT_P4_FIELD = 154; - private static final int JOINT_P5_FIELD = 155; - private static final int NUM_MGRA_FIELDS = 30; - - private static final String OUTPUT_SAMPLE_DEST_LOGSUMS = "output.sample.dest.logsums"; - - public VisitorTourLocationChoiceAppender(HashMap rbMap) - { - super(rbMap); - - debugEstimationFileRecord1 = DEBUG_EST_RECORD1; - debugEstimationFileRecord2 = DEBUG_EST_RECORD2; - - numMgraFields = NUM_MGRA_FIELDS; - } - - private void runLogsumAppender(ResourceBundle rb) - { - - totalTime1 = 0; - totalTime2 = 0; - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - tazs = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - mgraSetForLogsums = new int[numMgraFields + 1]; - - // allocate the logsums array for the chosen destination alternative - modeChoiceLogsums = new double[NUM_MGRA_FIELDS + 1][]; - - departArriveLogsums = new double[NUM_MGRA_FIELDS + 1][departArriveCombinations.length]; - - String outputAllKey = Util.getStringValueFromPropertyMap(rbMap, OUTPUT_SAMPLE_DEST_LOGSUMS); - - String outputFileName = Util.getStringValueFromPropertyMap(rbMap, - "dc.est.skims.output.file"); - if (outputFileName == null) - { - logger.info("no output file name was specified in the properties file. Nothing to do."); - return; - } - - int dotIndex = outputFileName.indexOf("."); - String baseName = outputFileName.substring(0, dotIndex); - String extension = outputFileName.substring(dotIndex); - - // output1 is only written if "all" was set in propoerties file - String outputName1 = ""; - if (outputAllKey.equalsIgnoreCase("all")) outputName1 = baseName + "_" + "all" + extension; - - // output1 is written in any case - String outputName2 = baseName + "_" + "chosen" + extension; - - PrintWriter outStream1 = null; - PrintWriter outStream2 = null; - - try - { - if (outputAllKey.equalsIgnoreCase("all")) - outStream1 = new PrintWriter(new BufferedWriter(new FileWriter( - new File(outputName1)))); - outStream2 = new PrintWriter(new BufferedWriter(new FileWriter(new File(outputName2)))); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileName)); - throw new RuntimeException(e); - } - - writeDcFile(rbMap, outStream1, outStream2); - - logger.info("total part 1 runtime = " + (totalTime1 / 1000) + " seconds."); - logger.info("total part 2 runtime = " + (totalTime2 / 1000) + " seconds."); - - } - - private void writeDcFile(HashMap rbMap, PrintWriter outStream1, - PrintWriter outStream2) - { - - // print the chosen destMgra and the depart/arrive logsum field names to - // both - // files - if (outStream1 != null) outStream1.print("seq,sampno,chosenMgra"); - - // attach the OB and IB period labels to the logsum field names for each - // period - if (outStream1 != null) - { - for (String[] labels : departArriveCombinationLabels) - outStream1.print(",logsum" + labels[0] + labels[1]); - } - - outStream2.print("seq,sampno,chosenMgra,chosenTodLogsum"); - - // print each set of sample destMgra and the depart/arrive logsum - // fieldnames - // to file 1. - // print each set of sample destMgra and the chosen depart/arrive logsum - // fieldname to file 2. - for (int m = 1; m < departArriveLogsums.length; m++) - { - if (outStream1 != null) - { - outStream1.print(",sampleMgra" + m); - for (String[] labels : departArriveCombinationLabels) - outStream1.print(",logsum" + m + labels[0] + labels[1]); - } - - outStream2.print(",sampleMgra" + m); - outStream2.print(",sampleLogsum" + m); - } - if (outStream1 != null) outStream1.print("\n"); - outStream2.print("\n"); - - TableDataSet estTds = getEstimationDataTableDataSet(rbMap); - int[][] estDataOdts = getDcEstimationDataOrigDestTimes(estTds); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = rbMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - SandagAppendMcLogsumDMU mcDmuObject = new SandagAppendMcLogsumDMU(modelStructure, null); - - ChoiceModelApplication[] mcModel = new ChoiceModelApplication[5 + 1]; - mcModel[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - mcModel[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, MAINTENANCE_SHEET, 0, - rbMap, (VariableTable) mcDmuObject); - mcModel[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, rbMap, (VariableTable) mcDmuObject); - mcModel[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, rbMap, - (VariableTable) mcDmuObject); - - // write skims data for estimation data file records - int seq = 1; - for (int i = 0; i < estDataOdts.length; i++) - { - - int[] odtSet = estDataOdts[i]; - int[] mgraSet = mgras[i]; - - odtSet[0] = seq; - - if (outStream1 != null) - { - outStream1.print(seq + "," + odtSet[SAMPNO]); - } - outStream2.print(seq + "," + odtSet[SAMPNO]); - - int category = PURPOSE_CATEGORIES[odtSet[TOUR_PURPOSE]]; - - int[] departAvailable = {-1, 1, 1, 1, 1, 1}; - int[] arriveAvailable = {-1, 1, 1, 1, 1, 1}; - calculateModeChoiceLogsums(rbMap, category == -1 ? null : mcModel[category], - mcDmuObject, odtSet, mgraSet, departAvailable, arriveAvailable, false); - - // write chosen dest and logsums to both files - if (outStream1 != null) - { - outStream1.print("," + odtSet[DEST_MGRA]); - for (double logsum : departArriveLogsums[0]) - outStream1.printf(",%.8f", logsum); - } - - outStream2.print("," + odtSet[DEST_MGRA]); - outStream2.printf(",%.8f", departArriveLogsums[0][chosenLogsumTodIndex]); - - // write logsum sets for each dest in the sample to file 1 - for (int m = 1; m < departArriveLogsums.length; m++) - { - if (outStream1 != null) - { - outStream1.print("," + mgraSet[m - 1]); - for (double logsum : departArriveLogsums[m]) - outStream1.printf(",%.8f", logsum); - } - - outStream2.print("," + mgraSet[m - 1]); - outStream2.printf(",%.8f", departArriveLogsums[m][chosenLogsumTodIndex]); - } - if (outStream1 != null) outStream1.print("\n"); - outStream2.print("\n"); - - if (seq % 1000 == 0) logger.info("wrote DC Estimation file record: " + seq); - - seq++; - } - - if (outStream1 != null) outStream1.close(); - outStream2.close(); - - } - - private int[][] getDcEstimationDataOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure - // period(1-6), and arrival period(1-6). - int[][] odts = new int[hisTds.getRowCount()][NUM_FIELDS]; - mgras = new int[hisTds.getRowCount()][NUM_MGRA_FIELDS]; - int[][] mgraData = new int[NUM_MGRA_FIELDS][]; - - int[] departs = hisTds.getColumnAsInt(DEPART_PERIOD_FIELD); - int[] arrives = hisTds.getColumnAsInt(ARRIVE_PERIOD_FIELD); - - int[] hisseq = hisTds.getColumnAsInt(SEQ_FIELD); - int[] purpose = hisTds.getColumnAsInt(PURPOSE_FIELD); - int[] jtPurpose = hisTds.getColumnAsInt(JOINT_PURPOSE_FIELD); - int[] income = hisTds.getColumnAsInt(INCOME_FIELD); - int[] origs = hisTds.getColumnAsInt(ORIG_MGRA_FIELD); - int[] dests = hisTds.getColumnAsInt(DEST_MGRA_FIELD); - int[] adults = hisTds.getColumnAsInt(ADULTS_FIELD); - int[] autos = hisTds.getColumnAsInt(AUTOS_FIELD); - int[] hhsize = hisTds.getColumnAsInt(HHSIZE_FIELD); - int[] gender = hisTds.getColumnAsInt(GENDER_FIELD); - int[] age = hisTds.getColumnAsInt(AGE_FIELD); - int[] jointId = hisTds.getColumnAsInt(JOINT_ID_FIELD); - int[] jointPerson1Participates = hisTds.getColumnAsInt(JOINT_P1_FIELD); - int[] jointPerson2Participates = hisTds.getColumnAsInt(JOINT_P2_FIELD); - int[] jointPerson3Participates = hisTds.getColumnAsInt(JOINT_P3_FIELD); - int[] jointPerson4Participates = hisTds.getColumnAsInt(JOINT_P4_FIELD); - int[] jointPerson5Participates = hisTds.getColumnAsInt(JOINT_P5_FIELD); - - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgraData[i] = hisTds.getColumnAsInt(MGRA1_FIELD + i); - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - for (int i = 0; i < NUM_MGRA_FIELDS; i++) - mgras[r - 1][i] = mgraData[i][r - 1]; - - odts[r - 1][SAMPNO] = hisseq[r - 1]; - - odts[r - 1][DEPART_PERIOD] = departs[r - 1]; - odts[r - 1][ARRIVE_PERIOD] = arrives[r - 1]; - - odts[r - 1][ORIG_MGRA] = origs[r - 1]; - odts[r - 1][DEST_MGRA] = dests[r - 1]; - odts[r - 1][INCOME] = income[r - 1]; - odts[r - 1][ADULTS] = adults[r - 1]; - odts[r - 1][AUTOS] = autos[r - 1]; - odts[r - 1][HHSIZE] = hhsize[r - 1]; - odts[r - 1][FEMALE] = gender[r - 1] == 2 ? 1 : 0; - odts[r - 1][AGE] = age[r - 1]; - odts[r - 1][JOINT] = jointId[r - 1] > 0 ? 1 : 0; - - // the offest constant is used because at-work subtours in - // estimation file are coded as work purpose index (=1), - // but the model index to use is 5. Nonmandatory and escort files - // have correct purpose codes, so offset is 0. - int purposeIndex = purpose[r - 1] + PURPOSE_INDEX_OFFSET; - - odts[r - 1][ESCORT] = purposeIndex == 4 ? 1 : 0; - - odts[r - 1][PARTYSIZE] = jointPerson1Participates[r - 1] - + jointPerson2Participates[r - 1] + jointPerson3Participates[r - 1] - + jointPerson4Participates[r - 1] + jointPerson5Participates[r - 1]; - - odts[r - 1][TOUR_PURPOSE] = odts[r - 1][JOINT] == 1 && purposeIndex > 4 ? jtPurpose[r - 1] - : purposeIndex; - - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb; - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - VisitorTourLocationChoiceAppender appender = new VisitorTourLocationChoiceAppender(rbMap); - - appender.startMatrixServer(rb); - appender.runLogsumAppender(rb); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java deleted file mode 100644 index d97d976..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitDriveSkimsCalculator.java +++ /dev/null @@ -1,295 +0,0 @@ -package org.sandag.abm.accessibilities; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import java.io.File; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; - -/** - * This class is used to return walk-transit-drive skim values for MGRA pairs - * associated with estimation data file records. - * - * @author Jim Hicks - * @version March, 2010 - */ -public class WalkTransitDriveSkimsCalculator - implements Serializable -{ - - private transient Logger logger; - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; - - private static final int ACCESS_TIME_INDEX = 0; - private static final int EGRESS_TIME_INDEX = 1; - private static final int NA = -999; - - private int maxWTDSkimSets = 5; - private int[] NUM_SKIMS; - private double[] defaultSkims; - - // declare UEC object - private UtilityExpressionCalculator walkDriveSkimUEC; - private IndexValues iv; - - // The simple auto skims UEC does not use any DMU variables - private TransitDriveAccessDMU dmu = new TransitDriveAccessDMU(); // DMU - // for - // this - // UEC - - private BestTransitPathCalculator bestPathUEC; - - private MgraDataManager mgraManager; - private int maxTap; - - private String[] skimNames; - - // skim values for transit service type(local, premium), - // transit ride mode(lbs, ebs, brt, lrt, crl), - // depart skim period(am, pm, op), and Tap-Tap pair. - private double[][][][][] storedDepartPeriodTapTapSkims; - - private MatrixDataServerIf ms; - - public WalkTransitDriveSkimsCalculator(HashMap rbMap) - { - mgraManager = MgraDataManager.getInstance(); - maxTap = mgraManager.getMaxTap(); - } - - public void setup(HashMap rbMap, Logger aLogger, - BestTransitPathCalculator myBestPathUEC) - { - - logger = aLogger; - - // set the best transit path utility UECs - bestPathUEC = myBestPathUEC; - - // Create the skim UECs - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.drive.data.page"); - int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.drive.skim.page"); - int wtdNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.walk.transit.drive.skims"); - String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.walk.transit.drive.uec.file")).toString(); - File uecFile = new File(uecFileName); - walkDriveSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); - - skimNames = walkDriveSkimUEC.getAlternativeNames(); - - //setup index values - iv = new IndexValues(); - - //setup default skim values - defaultSkims = new double[wtdNumSkims]; - for (int j = 0; j < wtdNumSkims; j++) { - defaultSkims[j] = NA; - } - - // point the stored Array of skims: by Prem or Local, DepartPeriod, O tap, D tap, skim values[] to a shared data store - StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxWTDSkimSets, NUM_PERIODS, maxTap ); - storedDepartPeriodTapTapSkims = storedDataObject.getStoredWtdDepartPeriodTapTapSkims(); - } - - - - /** - * Return the array of walk-transit-drive skims for the ride mode, origin TAP, - * destination TAP, and departure time period. - * - * @param set for set source skims - * @param origTap best Origin TAP for the MGRA pair - * @param workTap best Destination TAP for the MGRA pair - * @param departPeriod Departure time period - 1 = AM period, 2 = PM period, 3 = - * OffPeak period - * @return Array of 55 skim values for the MGRA pair and departure period - */ - public double[] getWalkTransitDriveSkims(int set, double pWalkTime, double aDriveTime, int origTap, int destTap, int departPeriod, boolean debug) - { - - dmu.setMgraTapWalkTime(pWalkTime); - dmu.setDriveTimeFromTap(aDriveTime); - - iv.setOriginZone(origTap); - iv.setDestZone(destTap); - - // allocate space for the origin tap if it hasn't been allocated already - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) - { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; - } - - // if the destTap skims are not already stored, calculate them and store - // them - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) - { - dmu.setTOD(departPeriod); - dmu.setSet(set); - double[] results = walkDriveSkimUEC.solve(iv, dmu, null); - if (debug) - walkDriveSkimUEC.logAnswersArray(logger, "Walk-Drive Skims"); - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pWalkTime; - } - catch ( Exception e ) { - logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pWalkTime=" + pWalkTime); - logger.error ("exception setting walk-transit-drive walk access time in stored array.", e); - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aDriveTime; - } - catch ( Exception e ) { - logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aDriveTime=" + aDriveTime); - logger.error ("exception setting walk-transit-drive drive egress time in stored array.", e); - } - return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; - - - } - - public double[] getNullTransitSkims() - { - return defaultSkims; - } - - /** - * Start the matrix server - * - * @param rb is a ResourceBundle for the properties file for this application - */ - private void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error(String - .format("exception caught running ctramp model components -- exiting."), e); - throw new RuntimeException(); - - } - - } - - /** - * log a report of the final skim values for the MGRA odt - * - * @param odt is an int[] with the first element the origin mgra and the second - * element the dest mgra and third element the departure period index - * @param bestTapPairs is an int[][] of TAP values with the first dimesion the - * ride mode and second dimension a 2 element array with best orig and - * dest TAP - * @param returnedSkims is a double[][] of skim values with the first dimesion - * the ride mode indices and second dimention the skim categories - */ - public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) - { - - Modes.TransitMode[] mode = Modes.TransitMode.values(); - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String separator = ""; - String header = ""; - - logger.info(""); - logger.info(""); - header = "Returned walk-transit-drive skim value tables for origMgra=" + odt[0] - + ", destMgra=" + odt[1] + ", period index=" + odt[2] + ", period label=" - + PERIODS[odt[2]]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - logger.info(separator); - logger.info(header); - logger.info(""); - - String modeHeading = String.format("%-12s %3s ", "RideMode:", mode[0]); - for (int i = 1; i < bestTapPairs.length; i++) - modeHeading += String.format(" %3s ", mode[i]); - logger.info(modeHeading); - - String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", - bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][0]) : "NA", - bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][1]) : "NA"); - for (int i = 1; i < bestTapPairs.length; i++) - tapHeading += String.format(" %4s-%4s ", bestTapPairs[i] != null ? String - .valueOf(bestTapPairs[i][0]) : "NA", bestTapPairs[i] != null ? String - .valueOf(bestTapPairs[i][1]) : "NA"); - logger.info(tapHeading); - - String underLine = String.format("%-12s %9s ", "---------", "---------"); - for (int i = 1; i < bestTapPairs.length; i++) - underLine += String.format(" %9s ", "---------"); - logger.info(underLine); - - for (int j = 0; j < ncols; j++) - { - String tableRecord = ""; - if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, - skims[0][j]); - else tableRecord = String.format("%-12d %12s ", j + 1, ""); - for (int i = 1; i < bestTapPairs.length; i++) - { - if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); - else tableRecord += String.format(" %12s ", ""); - } - logger.info(tableRecord); - } - - logger.info(""); - logger.info(separator); - } - - public String[] getSkimNames() { - return skimNames; - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java deleted file mode 100644 index 3f90731..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/WalkTransitWalkSkimsCalculator.java +++ /dev/null @@ -1,299 +0,0 @@ -package org.sandag.abm.accessibilities; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import java.io.File; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; - -/** - * This class is used to return walk-transit-walk skim values for MGRA pairs - * associated with estimation data file records. - * - * @author Jim Hicks - * @version March, 2010 - */ -public class WalkTransitWalkSkimsCalculator - implements Serializable -{ - - private transient Logger logger; - - private static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - private static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - private static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - private static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - private static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - private static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; - - private static final int ACCESS_TIME_INDEX = 0; - private static final int EGRESS_TIME_INDEX = 1; - private static final int NA = -999; - - private int maxWTWSkimSets = 5; - private int[] NUM_SKIMS; - private double[] defaultSkims; - - // declare UEC object - private UtilityExpressionCalculator walkWalkSkimUEC; - private IndexValues iv; - - private String[] skimNames; - - // The simple auto skims UEC does not use any DMU variables - private TransitWalkAccessDMU dmu = new TransitWalkAccessDMU(); - // DMU - // for - // this - // UEC - - private MgraDataManager mgraManager; - private int maxTap; - - // skim values for transit skim set - // depart skim period(am, pm, op) - // and Tap-Tap pair. - private double[][][][][] storedDepartPeriodTapTapSkims; - - private BestTransitPathCalculator bestPathUEC; - - private MatrixDataServerIf ms; - - public WalkTransitWalkSkimsCalculator(HashMap rbMap) - { - mgraManager = MgraDataManager.getInstance(); - maxTap = mgraManager.getMaxTap(); - } - - public void setup(HashMap rbMap, Logger aLogger, BestTransitPathCalculator myBestPathUEC) - { - - logger = aLogger; - - // Create the utility UECs - bestPathUEC = myBestPathUEC; - - // Create the skim UECs - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.walk.data.page"); - int skimPage = Util.getIntegerValueFromPropertyMap(rbMap,"skim.walk.transit.walk.skim.page"); - int wtwNumSkims = Util.getIntegerValueFromPropertyMap(rbMap, "skim.walk.transit.walk.skims"); - String uecPath = Util.getStringValueFromPropertyMap(rbMap, CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = Paths.get(uecPath,Util.getStringValueFromPropertyMap(rbMap, "skim.walk.transit.walk.uec.file")).toString(); - File uecFile = new File(uecFileName); - walkWalkSkimUEC = new UtilityExpressionCalculator(uecFile, skimPage, dataPage, rbMap, dmu); - - //setup index values - iv = new IndexValues(); - - //setup default skim values - defaultSkims = new double[wtwNumSkims]; - for (int j = 0; j < wtwNumSkims; j++) { - defaultSkims[j] = NA; - } - - skimNames = walkWalkSkimUEC.getAlternativeNames(); - - // point the stored Array of skims: skim set, period, O tap, D tap, skim values[] to a shared data store - StoredTransitSkimData storedDataObject = StoredTransitSkimData.getInstance( maxWTWSkimSets, NUM_PERIODS, maxTap ); - storedDepartPeriodTapTapSkims = storedDataObject.getStoredWtwDepartPeriodTapTapSkims(); - - } - - - - /** - * Return the array of walk-transit skims for the ride mode, origin TAP, - * destination TAP, and departure time period. - * - * @param set for set source skims - * @param origTap best Origin TAP for the MGRA pair - * @param destTap best Destination TAP for the MGRA pair - * @param departPeriod skim period index for the departure period - 0 = AM - * period, 1 = PM period, 2 = OffPeak period - * @return Array of skim values for the MGRA pair and departure period for the - * skim set - */ - public double[] getWalkTransitWalkSkims(int set, double pWalkTime, double aWalkTime, int origTap, int destTap, - int departPeriod, boolean debug) - { - - dmu.setMgraTapWalkTime(pWalkTime); - dmu.setTapMgraWalkTime(aWalkTime); - - iv.setOriginZone(origTap); - iv.setDestZone(destTap); - - // allocate space for the origin tap if it hasn't been allocated already - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap] == null) - { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap] = new double[maxTap + 1][]; - } - - // if the destTap skims are not already stored, calculate them and store - // them - if (storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] == null) - { - dmu.setTOD(departPeriod); - dmu.setSet(set); - double[] results = walkWalkSkimUEC.solve(iv, dmu, null); - if (debug) - walkWalkSkimUEC.logAnswersArray(logger, "Walk-Walk Tap-Tap Skims"); - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap] = results; - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][ACCESS_TIME_INDEX] = pWalkTime; - } - catch ( Exception e ) { - logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", pWalkTime=" + pWalkTime); - logger.error ("exception setting walk-transit-walk walk access time in stored array.", e); - } - - try { - storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap][EGRESS_TIME_INDEX] = aWalkTime; - } - catch ( Exception e ) { - logger.error ("departPeriod=" + departPeriod + ", origTap=" + origTap + ", destTap=" + destTap + ", aWalkTime=" + aWalkTime); - logger.error ("exception setting walk-transit-walk walk egress time in stored array.", e); - } - return storedDepartPeriodTapTapSkims[set][departPeriod][origTap][destTap]; - - - } - - public double[] getNullTransitSkims() - { - return defaultSkims; - } - - /** - * Start the matrix server - * - * @param rb is a ResourceBundle for the properties file for this application - */ - private void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error(String - .format("exception caught running ctramp model components -- exiting."), e); - throw new RuntimeException(); - - } - - } - - /** - * log a report of the final skim values for the MGRA odt - * - * @param odt is an int[] with the first element the origin mgra and the second - * element the dest mgra and third element the departure period index - * @param bestTapPairs is an int[][] of TAP values with the first dimesion the - * ride mode and second dimension a 2 element array with best orig and - * dest TAP - * @param returnedSkims is a double[][] of skim values with the first dimesion - * the ride mode indices and second dimention the skim categories - */ - public void logReturnedSkims(int[] odt, int[][] bestTapPairs, double[][] skims) - { - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String separator = ""; - String header = ""; - - logger.info(""); - logger.info(""); - header = "Returned walktransit skim value tables for origMgra=" + odt[0] + ", destMgra=" - + odt[1] + ", period index=" + odt[2] + ", period label=" + PERIODS[odt[2]]; - for (int i = 0; i < header.length(); i++) - separator += "^"; - - logger.info(separator); - logger.info(header); - logger.info(""); - - String modeHeading = String.format("%-12s %3s ", "Alt:"); - for (int i = 1; i < bestTapPairs.length; i++) - modeHeading += String.format(" %3s ", i); - logger.info(modeHeading); - - String tapHeading = String.format("%-12s %4s-%4s ", "TAP Pair:", - bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][0]) : "NA", - bestTapPairs[0] != null ? String.valueOf(bestTapPairs[0][1]) : "NA"); - for (int i = 1; i < bestTapPairs.length; i++) - tapHeading += String.format(" %4s-%4s ", bestTapPairs[i] != null ? String - .valueOf(bestTapPairs[i][0]) : "NA", bestTapPairs[i] != null ? String - .valueOf(bestTapPairs[i][1]) : "NA"); - logger.info(tapHeading); - - String underLine = String.format("%-12s %9s ", "---------", "---------"); - for (int i = 1; i < bestTapPairs.length; i++) - underLine += String.format(" %9s ", "---------"); - logger.info(underLine); - - for (int j = 0; j < ncols; j++) - { - String tableRecord = ""; - if (j < skims[0].length) tableRecord = String.format("%-12d %12.5f ", j + 1, - skims[0][j]); - else tableRecord = String.format("%-12d %12s ", j + 1, ""); - for (int i = 1; i < bestTapPairs.length; i++) - { - if (j < skims[i].length) tableRecord += String.format(" %12.5f ", skims[i][j]); - else tableRecord += String.format(" %12s ", ""); - } - logger.info(tableRecord); - } - - logger.info(""); - logger.info(separator); - } - - public String[] getSkimNames() { - return skimNames; - } - - public BestTransitPathCalculator getBestPathUEC() { - return bestPathUEC; - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java b/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java deleted file mode 100644 index 1b29ea5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/accessibilities/XBorderSkimsAppender.java +++ /dev/null @@ -1,518 +0,0 @@ -package org.sandag.abm.accessibilities; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -public final class XBorderSkimsAppender -{ - - protected transient Logger logger = Logger.getLogger(XBorderSkimsAppender.class); - - /* - * for trip mode choice estimation files - */ - private static final String MGRA_O_D_RECORDS_FILE_KEY = "xborder.mgra.list.file"; - private static final String APPENDED_SKIMS_FILE_KEY = "xborder.appended.file"; - - private String[] inputFormats = {"NUMBER", "NUMBER", "NUMBER", - "STRING", "NUMBER", "NUMBER", "STRING" }; - private static final int INPUT_ORIG_MGRA = 5; - private static final int INPUT_DEST_MGRA = 6; - private static final int INPUT_DEPART_PERIOD = 7; - - // survey periods are: - // 0=not used, - // 1=03:00-05:59, - // 2=06:00-08:59, - // 3=09:00-11:59, - // 4=12:00-15:29, - // 5=15:30-18:59, - // 6=19:00-02:59 - // skim periods are: 0=0(N/A), 1=3(OP), 2=1(AM), 3=3(OP), 4=3(OP), 5=2(PM), - // 6=3(OP) - - // define a conversion array to convert period values in the survey file to - // skim - // period indices used in this propgram: 1=am peak, 2=pm peak, 3=off-peak. - private static final String[] SKIM_PERIOD_LABELS = {"am", "pm", "op"}; - private static final int[] SURVEY_PERIOD_TO_SKIM_PERIOD = {0, 3, 1, 3, 3, 2, 3}; - - private static int debugOrigMgra = 0; - private static int debugDestMgra = 0; - private static int departModelPeriod = 0; - - private MatrixDataServerIf ms; - private BestTransitPathCalculator bestPathUEC; - private static final float defaultVOT = 15.0f; - - private XBorderSkimsAppender() - { - } - - private void runSkimsAppender(ResourceBundle rb) - { - - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - Logger autoLogger = Logger.getLogger("auto"); - Logger wtwLogger = Logger.getLogger("wtw"); - - String outputFileNameHis = Util.getStringValueFromPropertyMap(rbMap, - APPENDED_SKIMS_FILE_KEY); - - FileWriter writer; - PrintWriter outStreamHis = null; - - AutoTazSkimsCalculator tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - McLogsumsAppender logsumHelper = new McLogsumsAppender(rbMap); - bestPathUEC = logsumHelper.getBestTransitPathCalculator(); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - AutoAndNonMotorizedSkimsCalculator anm = logsumHelper.getAnmSkimCalculator(); - WalkTransitWalkSkimsCalculator wtw = new WalkTransitWalkSkimsCalculator(rbMap); - - String heading = "seq"; - - heading += ",origMgra,destMgra,departPeriod"; - heading += getAutoSkimsHeaderRecord("auto", anm.getAutoSkimNames()); - heading += getNonMotorizedSkimsHeaderRecord("nm", anm.getNmSkimNames()); - heading += getTransitSkimsHeaderRecord("wtw", wtw.getSkimNames()); - - try - { - writer = new FileWriter(new File(outputFileNameHis)); - outStreamHis = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening output skims file: %s.", - outputFileNameHis)); - throw new RuntimeException(e); - } - outStreamHis.println(heading); - - Logger[] loggers = new Logger[4]; - loggers[0] = autoLogger; - loggers[1] = wtwLogger; - - int[] odt = new int[3]; - - TableDataSet hisTds = getInputTableDataSet(rbMap); - int[][] hisOdts = getInputOrigDestTimes(hisTds); - // 11100, pnrCoaster, mgra 4357 = taz 3641, mgra 26931 = taz 1140 - // int[][] hisOdts = { { 4357, 26931, 5, 0 } }; - // 25040, wlkCoaster, mgra 4989 = taz 3270, mgra 7796 = taz 1986 - // int[][] hisOdts = { { 7796, 4989, 2, 0 } }; - - // if ( debugOrigMgra <= 0 || debugDestMgra <= 0 || departModelPeriod <= - // 0 || departModelPeriod > 6 ) - // { - // logger.error("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); - // System.exit(-1); - // } - // int[][] hisOdts = { { debugOrigMgra, debugDestMgra, - // departModelPeriod, 0 } }; - - // write skims data for home interview survey records - int seq = 1; - for (int[] hisOdt : hisOdts) - { - // write outbound direction - odt[0] = hisOdt[0]; // orig - odt[1] = hisOdt[1]; // dest - odt[2] = SURVEY_PERIOD_TO_SKIM_PERIOD[hisOdt[2]]; // depart skim - // period - - try - { - - writeSkimsToFile(seq, outStreamHis, false, odt, anm, wtw, loggers); - } catch (Exception e) - { - logger.error("Exception caught processing record: " + seq + " of " + hisOdts.length - + "."); - break; - } - - if (seq % 1000 == 0) logger.info("wrote HIS record: " + seq); - - seq++; - } - - outStreamHis.close(); - - } - - private void writeSkimsToFile(int sequence, PrintWriter outStream, boolean loggingEnabled, - int[] odt, AutoAndNonMotorizedSkimsCalculator anm, WalkTransitWalkSkimsCalculator wtw, - Logger[] loggers) - { - - Logger autoLogger = loggers[0]; - Logger wtwLogger = loggers[1]; - - int[][] bestTapPairs = null; - double[][] returnedSkims = null; - - outStream.print(String.format("%d,%d,%d,%s", sequence, odt[0], odt[1], - SKIM_PERIOD_LABELS[odt[2] - 1])); - - double[] skims = anm.getAutoSkims(odt[0], odt[1], odt[2], defaultVOT, loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "auto", autoLogger); - - String autoRecord = getAutoSkimsRecord(skims); - outStream.print(autoRecord); - - skims = anm.getNonMotorizedSkims(odt[0], odt[1], odt[2], loggingEnabled, autoLogger); - if (loggingEnabled) - anm.logReturnedSkims(odt[0], odt[1], odt[2], skims, "non-motorized", autoLogger); - - String nmRecord = getAutoSkimsRecord(skims); - outStream.print(nmRecord); - - /* - * TODO: Fix this code - - bestTapPairs = wtw.getBestTapPairs(odt[0], odt[1], odt[2], loggingEnabled, wtwLogger); - returnedSkims = new double[bestTapPairs.length][]; - for (int i = 0; i < bestTapPairs.length; i++) - { - if (bestTapPairs[i] == null) returnedSkims[i] = wtw.getNullTransitSkims(i); - else - { - returnedSkims[i] = wtw.getWalkTransitWalkSkims(i, BestTransitPathCalculator - .findWalkTransitAccessTime(odt[0], bestTapPairs[i][0]), - BestTransitPathCalculator.findWalkTransitEgressTime(odt[1], - bestTapPairs[i][1]), bestTapPairs[i][0], bestTapPairs[i][1], - odt[2], loggingEnabled); - } - } - if (loggingEnabled) wtw.logReturnedSkims(odt, bestTapPairs, returnedSkims); - - String wtwRecord = getTransitSkimsRecord(odt, returnedSkims); - outStream.println(wtwRecord); - */ - } - - /** - * Start the matrix server - * - * @param rb - * is a ResourceBundle for the properties file for this - * application - */ - private void startMatrixServer(ResourceBundle rb) - { - - logger.info(""); - logger.info(""); - String serverAddress = rb.getString("RunModel.MatrixServerAddress"); - int serverPort = new Integer(rb.getString("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try - { - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[][] of skim values with the first dimesion the - * ride mode indices and second dimention the skim categories - */ - private String getTransitSkimsRecord(int[] odt, double[][] skims) - { - - int nrows = skims.length; - int ncols = 0; - for (int i = 0; i < nrows; i++) - if (skims[i].length > ncols) ncols = skims[i].length; - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - for (int j = 0; j < skims[i].length; j++) - tableRecord += String.format(",%.5f", skims[i][j]); - } - - return tableRecord; - - } - - /** - * create a String which can be written to an output file with all the skim - * values for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - * @param skims - * is a double[] of skim values - */ - private String getAutoSkimsRecord(double[] skims) - { - - String tableRecord = ""; - for (int i = 0; i < skims.length; i++) - { - tableRecord += String.format(",%.5f", skims[i]); - } - - return tableRecord; - - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getTransitSkimsHeaderRecord(String transitServiveLabel, String[] skimNames) - { - - Modes.TransitMode[] mode = Modes.TransitMode.values(); - - String heading = ""; - - for (int i = 0; i < mode.length; i++) - { - for (int j = 0; j < skimNames.length; j++) - heading += String.format(",%s_%s_%s", transitServiveLabel, mode[i], - skimNames[j]); - - } - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getAutoSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - /** - * create a String for the output file header record which can be written to - * an output file with all the skim value namess for the orig/dest/period. - * - * @param odt - * is an int[] with the first element the origin mgra and the - * second element the dest mgra and third element the departure - * period index - */ - private String getNonMotorizedSkimsHeaderRecord(String label, String[] names) - { - - String heading = ""; - - for (int i = 0; i < names.length; i++) - heading += String.format(",%s_%s", label, names[i]); - - return heading; - } - - private TableDataSet getInputTableDataSet(HashMap rbMap) - { - - String hisFileName = Util.getStringValueFromPropertyMap(rbMap, MGRA_O_D_RECORDS_FILE_KEY); - if (hisFileName == null) - { - logger.error("Error getting the filename from the properties file for the XBorder MGRA List data records file."); - logger.error("Properties file target: " + MGRA_O_D_RECORDS_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + MGRA_O_D_RECORDS_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - try - { - TableDataSet inTds = null; - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTds = reader.readFileWithFormats(new File(hisFileName), inputFormats); - // inTds = reader.readFile(new File(hisFileName)); - return inTds; - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading Sandag XBorder MGRA List data records file: %s into TableDataSet object.", - hisFileName)); - throw new RuntimeException(e); - } - - } - - private int[][] getInputOrigDestTimes(TableDataSet hisTds) - { - - // odts are an array with elements: origin mgra, destination mgra, - // departure period(1-6), and his sampno. - int[][] odts = new int[hisTds.getRowCount()][4]; - - int[] origs = hisTds.getColumnAsInt(INPUT_ORIG_MGRA); - int[] dests = hisTds.getColumnAsInt(INPUT_DEST_MGRA); - String[] departStrings = hisTds.getColumnAsString(INPUT_DEPART_PERIOD); - int[] departs = new int[departStrings.length]; - - for (int r = 1; r <= hisTds.getRowCount(); r++) - { - if (departStrings[r - 1].equalsIgnoreCase("am")) departs[r - 1] = 2; - else if (departStrings[r - 1].equalsIgnoreCase("pm")) departs[r - 1] = 5; - else if (departStrings[r - 1].equalsIgnoreCase("op")) departs[r - 1] = 1; - else departs[r - 1] = -1; - - odts[r - 1][0] = origs[r - 1]; - odts[r - 1][1] = dests[r - 1]; - odts[r - 1][2] = departs[r - 1]; - } - - return odts; - } - - public static void main(String[] args) - { - - ResourceBundle rb = null; - if (args.length == 0) - { - System.out - .println(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else if (args.length == 4) - { - rb = ResourceBundle.getBundle(args[0]); - - debugOrigMgra = Integer.parseInt(args[1]); - debugDestMgra = Integer.parseInt(args[2]); - departModelPeriod = Integer.parseInt(args[3]); - } else - { - System.out - .println("please set values for command line arguments: properties file, orig mgra, dest mgra, depart model period."); - System.exit(-1); - } - - try - { - - MatrixDataServerIf ms = null; - String serverAddress = null; - int serverPort = -1; - - HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - System.out.println(""); - System.out.println(""); - serverAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - - String serverPortString = (String) propertyMap.get("RunModel.MatrixServerPort"); - if (serverPortString != null) serverPort = Integer.parseInt(serverPortString); - - if (serverAddress != null && serverPort > 0) - { - try - { - System.out.println("attempting connection to matrix server " + serverAddress - + ":" + serverPort); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - System.out.println("connected to matrix server " + serverAddress + ":" - + serverPort); - - } catch (Exception e) - { - System.out - .println("exception caught running ctramp model components -- exiting."); - e.printStackTrace(); - throw new RuntimeException(); - } - } - - TazDataManager tazs = TazDataManager.getInstance(propertyMap); - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TapDataManager tapManager = TapDataManager.getInstance(propertyMap); - - // create an appender object and run it - XBorderSkimsAppender appender = new XBorderSkimsAppender(); - appender.runSkimsAppender(rb); - - } catch (RuntimeException e) - { - - e.printStackTrace(); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java deleted file mode 100644 index b3fd57d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractNetworkFactory.java +++ /dev/null @@ -1,51 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; -import java.util.Collections; -import java.util.HashMap; -import java.util.LinkedHashSet; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Set; - -public abstract class AbstractNetworkFactory, T extends Traversal> - extends NetworkFactory -{ - - @Override - public Network createNetwork() - { - Network network = new SimpleNetwork<>(getNodes(), getEdges(), getTraversals()); - calculateDerivedNodeAttributes(network); - calculateDerivedEdgeAttributes(network); - calculateDerivedTraversalAttributes(network); - return network; - } - - @Override - protected Collection getTraversals() - { - Collection edges = getEdges(); - Map> predecessors = new HashMap<>(); - for (N node : getNodes()) - predecessors.put(node, new LinkedList()); - for (E edge : edges) - predecessors.get(edge.getToNode()).add(edge); - Set traversals = new LinkedHashSet<>(); - for (E toEdge : getEdges()) - { - for (E fromEdge : predecessors.get(toEdge.getFromNode())) - if (!isReversal(fromEdge, toEdge)) traversals.add(getTraversal(fromEdge, toEdge)); - } - return Collections.unmodifiableCollection(traversals); - } - - private boolean isReversal(E edge1, E edge2) - { - return (edge1.getToNode().equals(edge2.getFromNode())) - && (edge1.getFromNode().equals(edge2.getToNode())); - } - - abstract protected T getTraversal(E fromEdge, E toEdge); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java deleted file mode 100644 index c94f312..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceEdgeAssignmentApplication.java +++ /dev/null @@ -1,217 +0,0 @@ -package org.sandag.abm.active; - -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Objects; -import java.util.Queue; -import java.util.Set; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.ConcurrentLinkedQueue; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.atomic.AtomicInteger; -import org.apache.log4j.Logger; - -public abstract class AbstractPathChoiceEdgeAssignmentApplication, T extends Traversal> -{ - private static final Logger logger = Logger.getLogger(AbstractPathChoiceEdgeAssignmentApplication.class); - - protected PathAlternativeListGenerationConfiguration configuration; - private boolean randomCostSeeded; - private double maxCost; - private TraversalEvaluator traversalCostEvaluator; - private EdgeEvaluator edgeLengthEvaluator; - long startTime; - protected Network network; - String outputDir; - - double[] sampleDistanceBreaks; - double[] samplePathSizes; - double[] sampleMinCounts; - double[] sampleMaxCounts; - - private static final int TRIP_PROGRESS_REPORT_COUNT = 1000; - - public AbstractPathChoiceEdgeAssignmentApplication( - PathAlternativeListGenerationConfiguration configuration) - { - this.configuration = configuration; - configuration.getOriginZonalCentroidIdMap(); - this.randomCostSeeded = configuration.isRandomCostSeeded(); - this.maxCost = configuration.getMaxCost(); - this.traversalCostEvaluator = configuration.getTraversalCostEvaluator(); - this.edgeLengthEvaluator = configuration.getEdgeLengthEvaluator(); - this.sampleDistanceBreaks = configuration.getSampleDistanceBreaks(); - this.samplePathSizes = configuration.getSamplePathSizes(); - this.sampleMinCounts = configuration.getSampleMinCounts(); - this.sampleMaxCounts = configuration.getSampleMaxCounts(); - this.network = configuration.getNetwork(); - this.outputDir = configuration.getOutputDirectory(); - } - - protected abstract Map assignTrip(int tripNum, - PathAlternativeList alternativeList); - - public Map assignTrips(List tripNums) - { - logger.info("Assigning trips..."); - logger.info("Writing to " + outputDir); - ConcurrentHashMap volumes = new ConcurrentHashMap<>(); - int threadCount = Runtime.getRuntime().availableProcessors() -1; - ExecutorService executor = Executors.newFixedThreadPool(threadCount); - final Queue tripQueue = new ConcurrentLinkedQueue<>(tripNums); - final CountDownLatch latch = new CountDownLatch(threadCount); - final AtomicInteger counter = new AtomicInteger(); - startTime = System.currentTimeMillis(); - for (int i = 0; i < threadCount; i++) - executor.execute(new CalculationTask(tripQueue, counter, latch, volumes)); - try - { - latch.await(); - } catch (InterruptedException e) - { - throw new RuntimeException(e); - } - executor.shutdown(); - - return volumes; - } - - private class CalculationTask - implements Runnable - { - private final Queue tripQueue; - private final AtomicInteger counter; - private final CountDownLatch latch; - private final ConcurrentHashMap volumes; - - private CalculationTask(Queue tripQueue, AtomicInteger counter, - CountDownLatch latch, ConcurrentHashMap volumes) - { - this.tripQueue = tripQueue; - this.counter = counter; - this.latch = latch; - this.volumes = volumes; - } - - private PathAlternativeList generateAlternatives(int tripId) - { - Set singleOriginNode = new HashSet<>(); - Set singleDestinationNode = new HashSet<>(); - - EdgeEvaluator randomizedEdgeCost; - ShortestPathStrategy shortestPathStrategy; - ShortestPathResultSet result; - - singleOriginNode.add(getOriginNode(tripId)); - singleDestinationNode.add(getDestinationNode(tripId)); - - NodePair odPair = new NodePair<>(getOriginNode(tripId), getDestinationNode(tripId)); - PathAlternativeList alternativeList = new PathAlternativeList<>(odPair, network, - edgeLengthEvaluator); - - TraversalEvaluator zeroTraversalEvaluator = new ZeroTraversalEvaluator(); - - shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, - edgeLengthEvaluator, zeroTraversalEvaluator); - result = shortestPathStrategy.getShortestPaths(singleOriginNode, singleDestinationNode, - Double.MAX_VALUE); - if (result.getShortestPathResult(odPair) == null) - { - logger.error("no path found for trip with origin " + getOriginNode(tripId) - + " and destination " + getDestinationNode(tripId)); - return alternativeList; - } - double distance = result.getShortestPathResult(odPair).getCost(); - int distanceIndex = findFirstIndexGreaterThan(distance, sampleDistanceBreaks); - - for (int iterCount = 1; iterCount <= sampleMinCounts[distanceIndex]; iterCount++) - { - if (randomCostSeeded) - { - randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, - Objects.hash(tripId, iterCount)); - } else - { - randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, 0); - } - - shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, - randomizedEdgeCost, traversalCostEvaluator); - result = shortestPathStrategy.getShortestPaths(singleOriginNode, - singleDestinationNode, Double.MAX_VALUE); - - alternativeList.add(result.getShortestPathResult(odPair).getPath()); - } - return alternativeList; - } - - public void run() - { - while (tripQueue.size() > 0) - { - int tripId = tripQueue.poll(); - PathAlternativeList alternativeList = generateAlternatives(tripId); - - if (alternativeList.getCount() > 0) - { - Map tripVolumes = assignTrip(tripId, alternativeList); - - for (E edge : tripVolumes.keySet()) - { - if (volumes.containsKey(edge)) - { - double[] values = volumes.get(edge); - for (int i = 0; i < values.length; i++) - values[i] += tripVolumes.get(edge)[i]; - volumes.put(edge, values); - } else - { - volumes.put(edge, tripVolumes.get(edge)); - } - } - } - - int c = counter.addAndGet(1); - if ((c % TRIP_PROGRESS_REPORT_COUNT) == 0) - { - System.out.println(" done with " + c + " trips, run time: " - + (System.currentTimeMillis() - startTime) / 1000 + " sec."); - } - } - - latch.countDown(); - } - } - - protected abstract N getOriginNode(int tripId); - - protected abstract N getDestinationNode(int tripId); - - private class ZeroTraversalEvaluator - implements TraversalEvaluator - { - private ZeroTraversalEvaluator() - { - } - - public double evaluate(T traversal) - { - return 0.0; - } - } - - protected int findFirstIndexGreaterThan(double value, double[] array) - { - for (int i = 0; i < array.length; i++) - { - if (array[i] >= value) - { - return i; - } - } - return array.length; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java deleted file mode 100644 index 04c4abc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractPathChoiceLogsumMatrixApplication.java +++ /dev/null @@ -1,422 +0,0 @@ -package org.sandag.abm.active; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Objects; -import java.util.Queue; -import java.util.Random; -import java.util.Set; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.ConcurrentLinkedQueue; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.atomic.AtomicInteger; -import org.apache.log4j.Logger; - -public abstract class AbstractPathChoiceLogsumMatrixApplication, T extends Traversal> -{ - private static final Logger logger = Logger.getLogger(AbstractPathChoiceLogsumMatrixApplication.class); - - protected PathAlternativeListGenerationConfiguration configuration; - Network network; - Map> nearbyZonalDistanceMap; - Map originZonalCentroidIdMap; - Map destinationZonalCentroidIdMap; - double[] sampleDistanceBreaks; - double[] samplePathSizes; - double[] sampleMinCounts; - double[] sampleMaxCounts; - EdgeEvaluator edgeLengthEvaluator; - EdgeEvaluator edgeCostEvaluator; - TraversalEvaluator traversalCostEvaluator; - double maxCost; - long startTime; - String outputDir; - Set traceOrigins; - protected Map propertyMap; - boolean randomCostSeeded; - boolean intrazonalsNeeded; - - private static final int ORIGIN_PROGRESS_REPORT_COUNT = 50; - private static final double DOUBLE_PRECISION_TOLERANCE = 0.001; - - protected abstract double[] calculateMarketSegmentLogsums( - PathAlternativeList alternativeList); - - protected abstract List> getMarketSegmentIntrazonalCalculations(); - - public AbstractPathChoiceLogsumMatrixApplication( - PathAlternativeListGenerationConfiguration configuration) - { - this.configuration = configuration; - this.network = configuration.getNetwork(); - this.nearbyZonalDistanceMap = Collections.unmodifiableMap(configuration - .getNearbyZonalDistanceMap()); - this.originZonalCentroidIdMap = Collections.unmodifiableMap(configuration - .getOriginZonalCentroidIdMap()); - this.destinationZonalCentroidIdMap = Collections.unmodifiableMap(configuration - .getDestinationZonalCentroidIdMap()); - this.sampleDistanceBreaks = configuration.getSampleDistanceBreaks(); - this.samplePathSizes = configuration.getSamplePathSizes(); - this.sampleMinCounts = configuration.getSampleMinCounts(); - this.sampleMaxCounts = configuration.getSampleMaxCounts(); - this.edgeLengthEvaluator = configuration.getEdgeLengthEvaluator(); - this.edgeCostEvaluator = configuration.getEdgeCostEvaluator(); - this.traversalCostEvaluator = configuration.getTraversalCostEvaluator(); - this.maxCost = configuration.getMaxCost(); - this.outputDir = configuration.getOutputDirectory(); - this.traceOrigins = configuration.getTraceOrigins(); - this.propertyMap = configuration.getPropertyMap(); - this.randomCostSeeded = configuration.isRandomCostSeeded(); - this.intrazonalsNeeded = configuration.isIntrazonalsNeeded(); - } - - public Map, double[]> calculateMarketSegmentLogsums() - { - logger.info("Generating path alternative lists..."); - logger.info("Writing to " + outputDir); - Map> logsums = new ConcurrentHashMap<>(); - startTime = System.currentTimeMillis(); - int threadCount = Runtime.getRuntime().availableProcessors() -1; - ExecutorService executor = Executors.newFixedThreadPool(threadCount); - final Queue originQueue = new ConcurrentLinkedQueue<>( - originZonalCentroidIdMap.keySet()); - - final ConcurrentHashMap> insufficientSamplePairs = new ConcurrentHashMap<>(); - final CountDownLatch latch = new CountDownLatch(threadCount); - final AtomicInteger counter = new AtomicInteger(); - for (int i = 0; i < threadCount; i++) - executor.execute(new CalculationTask(originQueue, counter, latch, logsums, - insufficientSamplePairs)); - try - { - latch.await(); - } catch (InterruptedException e) - { - throw new RuntimeException(e); - } - executor.shutdown(); - - /* - * for (int origin : insufficientSamplePairs.keySet() ) { String message - * = "Sample insufficient for origin zone " + origin + - * " and destination zones "; for (int destination : - * insufficientSamplePairs.get(origin) ) { message = message + - * destination + " "; } System.out.println(message); } - */ - - int totalPairs = 0; - for (int o : nearbyZonalDistanceMap.keySet()) - { - totalPairs += nearbyZonalDistanceMap.get(o).size(); - } - logger.info("Total OD pairs: " + totalPairs); - - int totalInsuffPairs = 0; - for (int o : insufficientSamplePairs.keySet()) - { - totalInsuffPairs += insufficientSamplePairs.get(o).size(); - } - - logger.info("Total insufficient sample pairs: " + totalInsuffPairs); - - if (intrazonalsNeeded) - { - logger.info("Calculating intrazonals"); - List> intrazonalCalculations = getMarketSegmentIntrazonalCalculations(); - int segments = intrazonalCalculations.size(); - for (int segment = 0; segment < segments; segment++) - { - for (N origin : logsums.keySet()) - { - Map originLogsums = logsums.get(origin); - if (segment == 0) originLogsums.put(origin, new double[segments]); - originLogsums.get(origin)[segment] = intrazonalCalculations.get(segment) - .getIntrazonalValue(origin, originLogsums, segment); - } - } - } - - Map, double[]> pairLogsums = new HashMap<>(); - for (N oNode : logsums.keySet()) - { - for (N dNode : logsums.get(oNode).keySet()) - { - pairLogsums.put(new NodePair(oNode, dNode), logsums.get(oNode).get(dNode)); - } - } - - return pairLogsums; - } - - private int findFirstIndexGreaterThan(double value, double[] array) - { - for (int i = 0; i < array.length; i++) - { - if (array[i] >= value) - { - return i; - } - } - return array.length; - } - - private class CalculationTask - implements Runnable - { - private final Queue originQueue; - private final AtomicInteger counter; - private final CountDownLatch latch; - - private final ConcurrentHashMap> insufficientSamplePairs; - private final Map> logsums; - - private CalculationTask(Queue originQueue, AtomicInteger counter, - CountDownLatch latch, Map> logsums, - ConcurrentHashMap> insufficientSamplePairs) - { - this.originQueue = originQueue; - this.counter = counter; - this.latch = latch; - this.insufficientSamplePairs = insufficientSamplePairs; - this.logsums = logsums; - } - - private Map, PathAlternativeList> generateAlternatives(int origin) - { - Set singleOriginNode = new HashSet<>(); - Set destinationNodes = new HashSet<>(); - Map destinationZoneMap = new HashMap<>(); - Map destinationDistanceMap = new HashMap<>(); - Map destinationPathSizeMap = new HashMap<>(); - Map destinationMinCountMap = new HashMap<>(); - Map destinationMaxCountMap = new HashMap<>(); - HashMap, PathAlternativeList> alternativeLists = new HashMap<>(); - EdgeEvaluator randomizedEdgeCost; - ShortestPathStrategy shortestPathStrategy; - ShortestPathResultSet result; - int distanceIndex; - - singleOriginNode.add(network.getNode(originZonalCentroidIdMap.get(origin))); - N destinationNode = null; - PathAlternativeList alternativeList; - - if (nearbyZonalDistanceMap.containsKey(origin)) - { - for (int destination : nearbyZonalDistanceMap.get(origin).keySet()) - { - try - { - destinationNode = network.getNode(destinationZonalCentroidIdMap - .get(destination)); - } catch (NullPointerException e) - { - logger.warn(destinationZonalCentroidIdMap.get(destination)); - } - destinationNodes.add(destinationNode); - destinationDistanceMap.put(destinationNode, nearbyZonalDistanceMap.get(origin) - .get(destination)); - destinationZoneMap.put(destinationNode, destination); - distanceIndex = findFirstIndexGreaterThan( - destinationDistanceMap.get(destinationNode), sampleDistanceBreaks); - destinationPathSizeMap.put(destinationNode, samplePathSizes[distanceIndex]); - destinationMinCountMap.put(destinationNode, sampleMinCounts[distanceIndex]); - destinationMaxCountMap.put(destinationNode, sampleMaxCounts[distanceIndex]); - } - } - - int iterCount = 1; - while (destinationNodes.size() > 0) - { - - if (randomCostSeeded) - { - randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, - Objects.hash(origin, iterCount)); - } else - { - randomizedEdgeCost = configuration.getRandomizedEdgeCostEvaluator(iterCount, 0); - } - - shortestPathStrategy = new RepeatedSingleSourceDijkstra(network, - randomizedEdgeCost, traversalCostEvaluator); - result = shortestPathStrategy.getShortestPaths(singleOriginNode, destinationNodes, - maxCost); - - for (NodePair odPair : result) - { - if (!alternativeLists.containsKey(odPair)) - { - alternativeLists.put(odPair, new PathAlternativeList(odPair, network, - edgeLengthEvaluator)); - } - alternativeList = alternativeLists.get(odPair); - alternativeList.add(result.getShortestPathResult(odPair).getPath()); - destinationNode = odPair.getToNode(); - - if (alternativeList.getSizeMeasureTotal() >= destinationPathSizeMap - .get(destinationNode) - DOUBLE_PRECISION_TOLERANCE - && iterCount >= destinationMinCountMap.get(destinationNode)) - { - destinationNodes.remove(odPair.getToNode()); - alternativeList.clearPathSizeCalculator(); - } else if (iterCount >= destinationMaxCountMap.get(destinationNode)) - { - destinationNodes.remove(odPair.getToNode()); - alternativeList.clearPathSizeCalculator(); - if (!insufficientSamplePairs.containsKey(origin)) - insufficientSamplePairs.put(origin, new ArrayList()); - insufficientSamplePairs.get(origin).add( - destinationZoneMap.get(destinationNode)); - } - } - - iterCount++; - } - - if (traceOrigins.contains(origin)) - { - try - { - PathAlternativeListWriter writer = new PathAlternativeListWriter( - outputDir + "origpaths_" + origin + ".csv", outputDir + "origlinks_" - + origin + ".csv"); - writer.writeHeaders(); - for (PathAlternativeList list : alternativeLists.values()) - { - writer.write(list); - } - writer.close(); - } catch (IOException e) - { - throw new RuntimeException(e.getMessage()); - } - - } - - for (NodePair odPair : alternativeLists.keySet()) - alternativeLists.put( - odPair, - resampleAlternatives(alternativeLists.get(odPair), - destinationPathSizeMap.get(odPair.getToNode()))); - - return alternativeLists; - } - - private PathAlternativeList resampleAlternatives(PathAlternativeList alts, - double targetSize) - { - if (targetSize >= alts.getSizeMeasureTotal()) - { - return alts; - } - Random r; - if (randomCostSeeded) - { - r = new Random(alts.getODPair().hashCode()); - } else - { - r = new Random(); - } - PathAlternativeList newAlts = new PathAlternativeList<>(alts.getODPair(), - network, alts.getLengthEvaluator()); - double[] prob = new double[alts.getCount()]; - double[] cum = new double[alts.getCount()]; - double tot = 0.0; - for (int i = 0; i < prob.length; i++) - { - prob[i] = alts.getSizeMeasures().get(i) / alts.getSizeMeasureTotal(); - tot = tot + prob[i]; - cum[i] = tot; - } - cum[alts.getCount() - 1] = 1.0; - - while (newAlts.getSizeMeasureTotal() < targetSize - && newAlts.getCount() < alts.getCount()) - { - double p = r.nextDouble(); - int idx = BinarySearch.binarySearch(cum, p); - newAlts.add(alts.get(idx)); - double curProb = cum[idx]; - if (idx > 0) - { - curProb = curProb - cum[idx - 1]; - } - for (int i = 0; i < cum.length; i++) - { - if (i < idx) - { - cum[i] = cum[i] / (1 - curProb); - } else - { - cum[i] = (cum[i] - curProb) / (1 - curProb); - } - } - } - return newAlts; - } - - @Override - public void run() - { - while (originQueue.size() > 0) - { - int origin = originQueue.poll(); - - Map, PathAlternativeList> alternativeLists = generateAlternatives(origin); - - if (traceOrigins.contains(origin)) - { - try - { - PathAlternativeListWriter writer = new PathAlternativeListWriter( - outputDir + "resamplepaths_" + origin + ".csv", outputDir - + "resamplelinks_" + origin + ".csv"); - writer.writeHeaders(); - for (PathAlternativeList list : alternativeLists.values()) - { - writer.write(list); - } - writer.close(); - } catch (IOException e) - { - throw new RuntimeException(e.getMessage()); - } - } - - double[] logsumValues; - for (NodePair odPair : alternativeLists.keySet()) - { - - if (!odPair.getFromNode().equals(odPair.getToNode())) - { - - logsumValues = calculateMarketSegmentLogsums(alternativeLists.get(odPair)); - if (!logsums.containsKey(odPair.getFromNode())) - { - logsums.put(odPair.getFromNode(), new ConcurrentHashMap()); - } - logsums.get(odPair.getFromNode()).put(odPair.getToNode(), logsumValues); - } - - } - - int c = counter.addAndGet(1); - if ((c % ORIGIN_PROGRESS_REPORT_COUNT) == 0) - { - logger.info(" done with " + c + " origins, run time: " - + (System.currentTimeMillis() - startTime) / 1000 + " sec."); - } - } - - latch.countDown(); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java deleted file mode 100644 index 8921833..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/AbstractShortestPathResultSet.java +++ /dev/null @@ -1,20 +0,0 @@ -package org.sandag.abm.active; - -public abstract class AbstractShortestPathResultSet - implements ModifiableShortestPathResultSet -{ - - @Override - public void addResult(NodePair od, Path path, double cost) - { - addResult(new ShortestPathResult(od, path, cost)); - } - - @Override - public void addAll(ShortestPathResultSet results) - { - for (ShortestPathResult result : results.getResults()) - addResult(result); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java deleted file mode 100644 index 417a9e1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/BasicShortestPathResultSet.java +++ /dev/null @@ -1,59 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; - -public class BasicShortestPathResultSet - extends AbstractShortestPathResultSet -{ - private final Map, ShortestPathResult> results; - - public BasicShortestPathResultSet() - { - results = new HashMap<>(); // iteration order may not matter, but just - // in case, this is cheap - } - - @Override - public void addResult(ShortestPathResult spResult) - { - ShortestPathResult spr = results.put(spResult.getOriginDestination(), spResult); - if (spr != null) - throw new IllegalArgumentException("Repeated shortest path results for node pair: (" - + spResult.getOriginDestination().getFromNode().getId() + "," - + spResult.getOriginDestination().getToNode().getId() + ")"); - } - - @Override - public void addResult(NodePair od, Path path, double cost) - { - addResult(new ShortestPathResult(od, path, cost)); - } - - @Override - public Iterator> iterator() - { - return results.keySet().iterator(); - } - - @Override - public ShortestPathResult getShortestPathResult(NodePair od) - { - return results.get(od); - } - - @Override - public int size() - { - return results.size(); - } - - @Override - public Collection> getResults() - { - return results.values(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java b/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java deleted file mode 100644 index c76b467..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/BinarySearch.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.abm.active; - -public class BinarySearch -{ - public static int binarySearch(double[] values, double target) - { - return binarySearch(values, target, 0, values.length - 1); - } - - public static int binarySearch(double[] values, double target, int lower, int upper) - { - if (lower <= upper) - { - int mid = (lower + upper) / 2; - - switch (Double.compare(values[mid], target)) - { - case 0: - return mid; - case 1: - return binarySearch(values, target, lower, upper - 1); - case -1: - return binarySearch(values, target, mid + 1, upper); - } - } - - if (values[lower] >= target) - { - return lower; - } else - { - return lower + 1; - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java deleted file mode 100644 index eddfe63..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/CompositeShortestPathResultSet.java +++ /dev/null @@ -1,57 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; -import java.util.HashMap; -import java.util.Iterator; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; - -public class CompositeShortestPathResultSet - implements ShortestPathResultSet -{ - private final Map, ShortestPathResultSet> spResultsLookup; - - public CompositeShortestPathResultSet() - { - spResultsLookup = new HashMap<>(); - } - - public void addShortestPathResults(ShortestPathResultSet spResults) - { - for (NodePair nodePair : spResults) - if (spResultsLookup.put(nodePair, spResults) != null) - throw new IllegalArgumentException( - "Repeated shortest path results for node pair: (" - + nodePair.getFromNode().getId() + "," - + nodePair.getToNode().getId() + ")"); - } - - @Override - public Iterator> iterator() - { - return spResultsLookup.keySet().iterator(); - } - - @Override - public ShortestPathResult getShortestPathResult(NodePair od) - { - return spResultsLookup.get(od).getShortestPathResult(od); - } - - @Override - public int size() - { - return spResultsLookup.size(); - } - - @Override - public Collection> getResults() - { - List> results = new LinkedList<>(); - for (ShortestPathResultSet spr : spResultsLookup.values()) - results.addAll(spr.getResults()); - return results; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java b/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java deleted file mode 100644 index cee8240..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/DestinationNotFoundException.java +++ /dev/null @@ -1,10 +0,0 @@ -package org.sandag.abm.active; - -public class DestinationNotFoundException - extends Exception -{ - public DestinationNotFoundException(String message) - { - super(message); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java b/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java deleted file mode 100644 index 4d25097..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/Edge.java +++ /dev/null @@ -1,9 +0,0 @@ -package org.sandag.abm.active; - -public interface Edge - extends Comparable> -{ - N getFromNode(); - - N getToNode(); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java b/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java deleted file mode 100644 index 1ee55f4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/EdgeEvaluator.java +++ /dev/null @@ -1,6 +0,0 @@ -package org.sandag.abm.active; - -public interface EdgeEvaluator> -{ - double evaluate(E edge); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java deleted file mode 100644 index 16e757b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculation.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Map; - -/** - * The {@code IntrazonalCalculation} class provides a framework for calculation - * intrazonals. - * - * @param - * The type of the zone nodes. - */ -public interface IntrazonalCalculation -{ - /** - * Get the intrazonal value given the origin node and the logsum values with - * that origin node. The logsum values may be stratified across markets, so - * the index of the market of interest is also provided. - * - * @param originNode - * The origin node. - * - * @param logsums - * The logsums with {@code originNode} as their origin. The - * logsums are stored as a map with the destination node as the - * key and an array of logsums as the value. The logsum array has - * a different logsum for each market. - * - * @param logsumIndex - * The index for the logsum of interest in the logsum arrays - * provided in {@code logsums}. - * - * @return the intrazonal logsum for {@code originNode}. - */ - double getIntrazonalValue(N originNode, Map logsums, int logsumIndex); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java b/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java deleted file mode 100644 index be82cc1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/IntrazonalCalculations.java +++ /dev/null @@ -1,365 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Map; -import org.apache.log4j.Logger; - -/** - * The {@code IntrazonalCalculations} class provides convenient default - * implementations of the {IntrazonalCalculation} interface. - * - */ -public class IntrazonalCalculations -{ - - private static final Logger logger = Logger.getLogger(IntrazonalCalculations.class); - - // this class is more of a static factory provider, so constructor is hidden - private IntrazonalCalculations() - { - } - - /** - * The {@code Factorizer} interface provides a framework for transforming an - * input value. It is essentially a function of one variable. - */ - public static interface Factorizer - { - /** - * Factor, or transform, an input value. - * - * @param inputValue - * The input value. - * - * @return the transformation of {@code inputValue}. - */ - double factor(double inputValue); - } - - /** - * Get a simple {@code Factorizer} implementation which applies a linear - * scale and offset. That is, for an input x, the function will - * return the following value: - *

    - * factor*x + offset - * - * @param factor - * The multiplicative factor. - * - * @param offset - * The addititive offset. - * - * @return the factorizer which will linearly scale and offset an input. - */ - public static Factorizer simpleFactorizer(final double factor, final double offset) - { - return new Factorizer() - { - @Override - public double factor(double inputValue) - { - return inputValue * factor + offset; - } - }; - } - - /** - * Get a {@code Factorizer} which applies a linear transformation, with - * different scale and offset for positive and negative input values. That - * is, for an input x, the factorizer function will return the - * following value - *

    - * factor*x + offset - *

    - * where factor and offset may differ according to - * whether x is positive or negative (if x is 0, - * it is considered positive). - * - * @param negativeFactor - * The multiplicative factor for negative input values. - * - * @param negativeOffset - * The additivie offset for negative input values. - * - * @param positiveFactor - * The multiplicative factor for positive input values. - * - * @param positiveOffset - * The additivie offset for positive input values. - * - * @return the factorizer which will linearly scale and offset an input, - * using different transoformations based on the input's sign. - */ - public static Factorizer positiveNegativeFactorizer(final double negativeFactor, - final double negativeOffset, final double positiveFactor, final double positiveOffset) - { - return new Factorizer() - { - @Override - public double factor(double inputValue) - { - if (inputValue < 0) return negativeFactor * inputValue + negativeOffset; - return positiveFactor * inputValue + positiveOffset; - } - }; - } - - /** - * Get an {@code IntrazonalCalculation} which will apply a function to the - * sum of the largest origin-based logsum values. That is, an intrazonal - * value is calculated by a function (defined by a {@code Factorizer}) which - * acts on the sum of the largest maxCount logsum values whose - * origin is the intrazonal's zone, where maxCount is set by - * the call to this function. - * - * @param - * The type of the zone nodes. - * - * @param factorizer - * The factorizer used to calculate the intrazonal value. - * - * @param maxCount - * The number of logsum values to be used in the intrazonal - * calculation. - * - * @return an intrazonal calculation which will apply {@code factorizer} to - * the sum of the largest {@code maxCount} logsum values. - */ - public static IntrazonalCalculation maxFactorIntrazonalCalculation( - final Factorizer factorizer, final int maxCount) - { - return new IntrazonalCalculation() - { - - @Override - public double getIntrazonalValue(N originNode, Map logsums, int logsumIndex) - { - MinHeap maxValues = new MinHeap(maxCount); - int initialCount = maxCount; - double minValue = 0; // will be filled in when needed - for (N node : logsums.keySet()) - { - if (!node.equals(originNode)) - { - double value = logsums.get(node)[logsumIndex]; - if (initialCount > 0) - { - maxValues.insert(value); - if (--initialCount == 0) minValue = maxValues.getMin(); - } else if (value > minValue) - { - maxValues.removeMin(); - maxValues.insert(value); - minValue = maxValues.getMin(); - } - } - } - return factorizer.factor(maxValues.getSum()); - } - }; - } - - /** - * Get an {@code IntrazonalCalculation} which will apply a function to the - * sum of the smallest origin-based logsum values. That is, an intrazonal - * value is calculated by a function (defined by a {@code Factorizer}) which - * acts on the sum of the smallest minCount logsum values whose - * origin is the intrazonal's zone, where minCount is set by - * the call to this function. - * - * @param - * The type of the zone nodes. - * - * @param factorizer - * The factorizer used to calculate the intrazonal value. - * - * @param minCount - * The number of logsum values to be used in the intrazonal - * calculation. - * - * @return an intrazonal calculation which will apply {@code factorizer} to - * the sum of the smallest {@code minCount} logsum values. - */ - public static IntrazonalCalculation minFactorIntrazonalCalculation( - final Factorizer factorizer, final int minCount) - { - return new IntrazonalCalculation() - { - - @Override - public double getIntrazonalValue(N originNode, Map logsums, int logsumIndex) - { - MaxHeap minValues = new MaxHeap(minCount); - int initialCount = minCount; - double maxValue = 0; // will be filled in when needed - for (N node : logsums.keySet()) - { - if (!node.equals(originNode)) - { - double value = logsums.get(node)[logsumIndex]; - if (initialCount > 0) - { - minValues.insert(value); - if (--initialCount == 0) maxValue = minValues.getMax(); - } else if (value < maxValue) - { - minValues.removeMax(); - minValues.insert(value); - maxValue = minValues.getMax(); - } - } - } - return factorizer.factor(minValues.getSum()); - } - }; - } - - private static class Heap - { - protected final double[] heap; - protected int end; - - private Heap(int size) - { - heap = new double[size]; - end = 0; - } - - public double getSum() - { - double sum = 0; - for (int i = 0; i < end; i++) - sum += heap[i]; - return sum; - } - } - - private static class MaxHeap - extends Heap - { - - private MaxHeap(int size) - { - super(size); - } - - public void insert(double value) - { - int point = end++; - if (point == 0) - { - heap[0] = value; - return; - } - while (point > 0) - { - int newPoint = (point - 1) / 2; - if (heap[newPoint] < value) - { - heap[point] = heap[newPoint]; - point = newPoint; - } else - { - heap[point] = value; - break; - } - if (point == 0) heap[0] = value; - } - } - - public double getMax() - { - return heap[0]; - } - - public double removeMax() - { - double max = heap[0]; - heap[0] = heap[--end]; - double value = heap[0]; - int point = 0; - while (true) - { - int left = 2 * point + 1; - int right = left + 1; - int largest = point; - if ((left < end) && (heap[left] > heap[largest])) largest = left; - if ((right < end) && (heap[right] > heap[largest])) largest = right; - if (largest != point) - { - heap[point] = heap[largest]; - point = largest; - } else - { - heap[point] = value; - break; - } - } - return max; - } - } - - private static class MinHeap - extends Heap - { - - private MinHeap(int size) - { - super(size); - } - - public void insert(double value) - { - int point = end++; - if (point == 0) - { - heap[0] = value; - return; - } - while (point > 0) - { - int newPoint = (point - 1) / 2; - if (heap[newPoint] > value) - { - heap[point] = heap[newPoint]; - point = newPoint; - } else - { - heap[point] = value; - break; - } - if (point == 0) heap[0] = value; - } - } - - public double getMin() - { - return heap[0]; - } - - public double removeMin() - { - double min = heap[0]; - heap[0] = heap[--end]; - double value = heap[0]; - int point = 0; - while (true) - { - int left = 2 * point + 1; - int right = left + 1; - int largest = point; - if ((left < end) && (heap[left] < heap[largest])) largest = left; - if ((right < end) && (heap[right] < heap[largest])) largest = right; - if (largest != point) - { - heap[point] = heap[largest]; - point = largest; - } else - { - heap[point] = value; - break; - } - } - return min; - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java deleted file mode 100644 index 5a52993..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/ModifiableShortestPathResultSet.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.sandag.abm.active; - -public interface ModifiableShortestPathResultSet - extends ShortestPathResultSet -{ - void addResult(ShortestPathResult spResult); - - void addResult(NodePair od, Path path, double cost); - - void addAll(ShortestPathResultSet results); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Network.java b/sandag_abm/src/main/java/org/sandag/abm/active/Network.java deleted file mode 100644 index 025035e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/Network.java +++ /dev/null @@ -1,163 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; -import java.util.Iterator; - -public interface Network, T extends Traversal> -{ - N getNode(int nodeId); - - E getEdge(N fromNode, N toNode); - - E getEdge(NodePair nodes); - - T getTraversal(E fromEdge, E toEdge); - - Collection getSuccessors(N node); - - Collection getPredecessors(N node); - - Iterator nodeIterator(); - - Iterator edgeIterator(); - - Iterator traversalIterator(); - - boolean containsNodeId(int id); - - boolean containsNode(N node); - - boolean containsEdge(N fromNode, N toNode); - - boolean containsTraversal(E fromEdge, E toEdge); - - // public void addNode(T node) - // { - // if ( nodeIndex.containsKey(node.getId()) ) { - // throw new RuntimeException("Network already contains Node with id " + - // node.getId()); - // } - // nodeIndex.put(node.getId(), nodes.size()); - // nodes.add(node); - // if (! successorIndex.containsKey(node.getId()) ) { - // successorIndex.put(node.getId(), new ArrayList()); } - // if (! predecessorIndex.containsKey(node.getId()) ) { - // predecessorIndex.put(node.getId(), new ArrayList()); } - // } - // - // public void addEdge(U edge) - // { - // int fromId = edge.getFromNode(); - // int toId = edge.getToNode(); - // EdgeKey edgeIndexKey = new EdgeKey(fromId, toId); - // - // if ( edgeIndex.containsKey(edgeIndexKey) ) { - // throw new RuntimeException("Network already contains Edge with fromId " + - // edge.getFromNode() + " and toId " + edge.getToNode()); - // } - // - // edgeIndex.put(edgeIndexKey, edges.size()); - // edges.add(edge); - // - // if ( ! successorIndex.containsKey(fromId) ) { successorIndex.put(fromId, - // new ArrayList()); } - // if ( ! predecessorIndex.containsKey(toId) ) { predecessorIndex.put(toId, - // new ArrayList()); } - // - // if ( ! successorIndex.get(fromId).contains(toId) ) { - // successorIndex.get(fromId).add(toId); } - // if ( ! predecessorIndex.get(toId).contains(fromId) ) { - // predecessorIndex.get(toId).add(fromId); } - // } - // - // public void addTraversal(V traversal) - // { - // int startId = traversal.getStartId(); - // int thruId = traversal.getThruId(); - // int endId = traversal.getEndId(); - // TraversalKey traversalIndexKey = new TraversalKey(startId, thruId, - // endId); - // - // traversalIndex.put(traversalIndexKey, traversals.size()); - // traversals.add(traversal); - // } - // - // public boolean containsNodeId(int id) { - // return nodeIndex.containsKey(id); - // } - // - // public boolean containsEdgeIds(int[] ids) { - // return edgeIndex.containsKey(new EdgeKey(ids[0],ids[1])); - // } - // - // public boolean containsTraversalIds(int[] ids) { - // return traversalIndex.containsKey(new - // TraversalKey(ids[0],ids[1],ids[2])); - // } - // - // private class EdgeKey { - // private int fromId, toId; - // - // EdgeKey(int fromId, int toId) { - // this.fromId = fromId; - // this.toId = toId; - // } - // - // @Override - // public int hashCode() - // { - // final int prime = 31; - // int result = 1; - // result = prime * result + fromId; - // result = prime * result + toId; - // return result; - // } - // - // @Override - // public boolean equals(Object obj) - // { - // if (this == obj) return true; - // if (obj == null) return false; - // if (getClass() != obj.getClass()) return false; - // EdgeKey other = (EdgeKey) obj; - // if (fromId != other.fromId) return false; - // if (toId != other.toId) return false; - // return true; - // } - // } - // - // private class TraversalKey { - // private int startId, thruId, endId; - // - // public TraversalKey(int startId, int thruId, int endId) - // { - // this.startId = startId; - // this.thruId = thruId; - // this.endId = endId; - // } - // - // @Override - // public int hashCode() - // { - // final int prime = 31; - // int result = 1; - // result = prime * result + endId; - // result = prime * result + startId; - // result = prime * result + thruId; - // return result; - // } - // - // @Override - // public boolean equals(Object obj) - // { - // if (this == obj) return true; - // if (obj == null) return false; - // if (getClass() != obj.getClass()) return false; - // TraversalKey other = (TraversalKey) obj; - // if (endId != other.endId) return false; - // if (startId != other.startId) return false; - // if (thruId != other.thruId) return false; - // return true; - // } - // } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java deleted file mode 100644 index 02d471b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/NetworkFactory.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; - -public abstract class NetworkFactory, T extends Traversal> -{ - - public Network createNetwork() - { - Network network = new SimpleNetwork<>(getNodes(), getEdges(), getTraversals()); - calculateDerivedNodeAttributes(network); - calculateDerivedEdgeAttributes(network); - calculateDerivedTraversalAttributes(network); - return network; - } - - protected abstract Collection getNodes(); - - protected abstract Collection getEdges(); - - protected abstract Collection getTraversals(); - - protected void calculateDerivedNodeAttributes(Network network) - { - } - - protected void calculateDerivedEdgeAttributes(Network network) - { - } - - protected void calculateDerivedTraversalAttributes(Network network) - { - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Node.java b/sandag_abm/src/main/java/org/sandag/abm/active/Node.java deleted file mode 100644 index d0fcd45..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/Node.java +++ /dev/null @@ -1,7 +0,0 @@ -package org.sandag.abm.active; - -public interface Node - extends Comparable -{ - int getId(); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java b/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java deleted file mode 100644 index c375a3d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/NodePair.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Objects; - -public class NodePair - implements Comparable> -{ - private final N fromNode; - private final N toNode; - - public NodePair(N fromNode, N toNode) - { - this.fromNode = fromNode; - this.toNode = toNode; - } - - public int compareTo(NodePair other) - { - int c = fromNode.compareTo(other.fromNode); - if (c == 0) c = toNode.compareTo(other.toNode); - return c; - } - - public N getFromNode() - { - return fromNode; - } - - public N getToNode() - { - return toNode; - } - - public boolean equals(Object other) - { - if ((other == null) || (!(other instanceof NodePair))) return false; - NodePair np = (NodePair) other; - return (fromNode.equals(np.fromNode)) && (toNode.equals(np.toNode)); - } - - public int hashCode() - { - return Objects.hash(fromNode, toNode); - } - - public String toString() - { - return ""; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java b/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java deleted file mode 100644 index ca82302..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/ParallelSingleSourceDijkstra.java +++ /dev/null @@ -1,230 +0,0 @@ -package org.sandag.abm.active; - -import java.util.HashSet; -import java.util.Queue; -import java.util.Set; -import java.util.concurrent.ConcurrentLinkedQueue; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.ForkJoinPool; -import java.util.concurrent.atomic.AtomicInteger; -import com.pb.sawdust.util.concurrent.DnCRecursiveTask; - -public class ParallelSingleSourceDijkstra - implements ShortestPathStrategy -{ - private final ShortestPathStrategy sp; - private final ParallelMethod method; - private final int SEGMENT_SIZE = 50; - - public ParallelSingleSourceDijkstra(ShortestPathStrategy sp, ParallelMethod method) - { - this.sp = sp; - this.method = method; - } - - public static enum ParallelMethod - { - FORK_JOIN, QUEUE - } - - @Override - public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, - double maxCost) - { - switch (method) - { - case FORK_JOIN: - { - ShortestPathRecursiveTask task = new ShortestPathRecursiveTask(sp, originNodes, - destinationNodes, maxCost); - new ForkJoinPool().execute(task); - ModifiableShortestPathResultSet sprc = task.getResult(); - return sprc; - } - case QUEUE: - { - int threadCount = Runtime.getRuntime().availableProcessors(); - ExecutorService executor = Executors.newFixedThreadPool(threadCount); - final Queue> sprcQueue = new ConcurrentLinkedQueue<>(); - final Queue originNodeQueue = new ConcurrentLinkedQueue<>(originNodes); - ThreadLocal> sprcThreadLocal = new ThreadLocal>() - { - @Override - public ModifiableShortestPathResultSet initialValue() - { - ModifiableShortestPathResultSet sprc = new BasicShortestPathResultSet<>(); - sprcQueue.add(sprc); - return sprc; - } - }; - final CountDownLatch latch = new CountDownLatch(threadCount); - final AtomicInteger counter = new AtomicInteger(); - for (int i = 0; i < threadCount; i++) - executor.execute(new QueueMethodTask(sp, originNodeQueue, destinationNodes, - maxCost, counter, sprcThreadLocal, latch)); - try - { - latch.await(); - } catch (InterruptedException e) - { - throw new RuntimeException(e); - } - executor.shutdown(); - - ModifiableShortestPathResultSet finalContainer = null; - for (ModifiableShortestPathResultSet sprc : sprcQueue) - if (finalContainer == null) finalContainer = sprc; - else finalContainer.addAll(sprc); - - return finalContainer; - } - default: - throw new IllegalStateException("Should not be here."); - } - } - - @Override - public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes) - { - return getShortestPaths(originNodes, destinationNodes, Double.POSITIVE_INFINITY); - } - - private class QueueMethodTask - implements Runnable - { - private final ShortestPathStrategy sp; - private final Queue originNodes; - private final Set destinationNodes; - private final double maxCost; - private final AtomicInteger counter; - private final ThreadLocal> spr; - private final CountDownLatch latch; - - private QueueMethodTask(ShortestPathStrategy sp, Queue originNodes, - Set destinationNodes, double maxCost, AtomicInteger counter, - ThreadLocal> spr, CountDownLatch latch) - { - this.sp = sp; - this.destinationNodes = destinationNodes; - this.originNodes = originNodes; - this.maxCost = maxCost; - this.counter = counter; - this.spr = spr; - this.latch = latch; - } - - @Override - public void run() - { - int segmentSize = SEGMENT_SIZE; - final Set origins = new HashSet<>(); - while (originNodes.size() > 0) - { - while ((originNodes.size() > 0) && (origins.size() < segmentSize)) - { - N origin = originNodes.poll(); - if (origin != null) origins.add(origin); - } - if (origins.size() == 0) break; - ShortestPathResultSet result = sp.getShortestPaths(origins, destinationNodes, - maxCost); - ModifiableShortestPathResultSet sprc = spr.get(); - for (ShortestPathResult spResult : result.getResults()) - sprc.addResult(spResult); - int c = counter.addAndGet(origins.size()); - if (c % segmentSize < origins.size()) - System.out.println(" done with " + ((c / segmentSize) * segmentSize) - + " origins"); - origins.clear(); - } - latch.countDown(); - } - } - - private class ShortestPathRecursiveTask - extends DnCRecursiveTask> - { - AtomicInteger counter; - private final ShortestPathStrategy sp; - private final Set destinations; - private final Node[] origins; - private final double maxCost; - - protected ShortestPathRecursiveTask(ShortestPathStrategy sp, Set origins, - Set destinations, double maxCost) - { - super(0, origins.size()); - this.sp = sp; - this.origins = origins.toArray(new Node[origins.size()]); - this.destinations = destinations; - this.maxCost = maxCost; - counter = new AtomicInteger(0); - } - - protected ShortestPathRecursiveTask(long start, long length, - DnCRecursiveTask> next, - ShortestPathStrategy sp, Node[] origins, Set destinations, double maxCost, - AtomicInteger counter) - { - super(start, length, next); - this.sp = sp; - this.origins = origins; - this.destinations = destinations; - this.maxCost = maxCost; - this.counter = counter; - } - - @Override - @SuppressWarnings("unchecked") - // origins only hold N, we just can't declare as such because of - // generics - protected ModifiableShortestPathResultSet computeTask(long start, long length) - { - Set originNodes = new HashSet<>(); - int end = (int) (start + length); - for (int n = (int) start; n < end; n++) - originNodes.add((N) origins[n]); - ShortestPathResultSet result = sp.getShortestPaths(originNodes, destinations); - ModifiableShortestPathResultSet spr = new BasicShortestPathResultSet<>(); - for (ShortestPathResult spResult : result.getResults()) - spr.addResult(spResult); - - int c = counter.addAndGet((int) length); - if (c % 10 < length) - System.out.println(" done with " + ((c / 10) * 10) + " origins"); - return spr; - } - - @Override - protected boolean continueDividing(long newLength) - { - return (newLength > 5) && (getSurplusQueuedTaskCount() < 3); - } - - @Override - protected DnCRecursiveTask> getNextTask(long start, - long length, DnCRecursiveTask> next) - { - return new ShortestPathRecursiveTask(start, length, next, sp, origins, destinations, - maxCost, counter); - } - - @Override - protected ModifiableShortestPathResultSet joinResults( - ModifiableShortestPathResultSet spr1, ModifiableShortestPathResultSet spr2) - { - if (spr1.size() > spr2.size()) - { - spr1.addAll(spr2); - return spr1; - } else - { - spr2.addAll(spr1); - return spr2; - } - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Path.java b/sandag_abm/src/main/java/org/sandag/abm/active/Path.java deleted file mode 100644 index 337db3b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/Path.java +++ /dev/null @@ -1,82 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Iterator; -import java.util.NoSuchElementException; - -public class Path - implements Iterable -{ - private final Path predecessorPath; - private final N next; - private final int length; - - public Path(Path predecessorPath, N next) - { - this.predecessorPath = predecessorPath; - this.next = next; - this.length = predecessorPath == null ? 1 : predecessorPath.length + 1; - } - - public Path(N first) - { - this(null, first); - } - - public int getLength() - { - return length; - } - - public N getNode(int index) - { - if (index < 0 || index >= length) - throw new IllegalArgumentException("Invalid index " + index + " for path of length " - + length); - return getNodeNoChecks(index + 1); - } - - private N getNodeNoChecks(int index) - { // index here is 1-based, not zero based! - return index == length ? next : predecessorPath.getNodeNoChecks(index); - } - - public Path extendPath(N next) - { - return new Path(this, next); - } - - public String getPathString() - { - StringBuilder sb = new StringBuilder(); - for (N n : this) - sb.append(n.getId()).append(n == next ? "" : " "); - return sb.toString(); - } - - public Iterator iterator() - { - return new Iterator() - { - private int point = 0; - - @Override - public boolean hasNext() - { - return point < length; - } - - @Override - public N next() - { - if (point < length) return getNodeNoChecks(++point); - else throw new NoSuchElementException(); - } - - @Override - public void remove() - { - throw new UnsupportedOperationException(); - } - }; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java deleted file mode 100644 index d8c4e25..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeList.java +++ /dev/null @@ -1,177 +0,0 @@ -package org.sandag.abm.active; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -public class PathAlternativeList> -{ - private List> paths; - private NodePair odPair; - private List sizeMeasures; - private PathSizeCalculator sizeCalculator; - private double sizeMeasureTotal; - private boolean sizeMeasuresUpdated; - Network network; - EdgeEvaluator lengthEvaluator; - - public PathAlternativeList(NodePair odPair, Network network, - EdgeEvaluator lengthEvaluator) - { - paths = new ArrayList>(); - sizeMeasures = new ArrayList(); - this.odPair = odPair; - sizeMeasuresUpdated = true; - this.network = network; - this.lengthEvaluator = lengthEvaluator; - this.sizeCalculator = new PathSizeCalculator(this); - this.sizeMeasureTotal = 0.0; - } - - public Network getNetwork() - { - return network; - } - - public void add(Path path) - { - if (!path.getNode(0).equals(odPair.getFromNode()) - || !path.getNode(path.getLength() - 1).equals(odPair.getToNode())) - { - throw new IllegalStateException( - "OD pair of path does not match that of path alternative list"); - } - for (Path otherPath : paths) - { - if (path.equals(otherPath)) - { - return; - } - } - paths.add(path); - sizeMeasures.add(0.0); - if (sizeCalculator == null) - { - sizeMeasuresUpdated = false; - } else - { - sizeCalculator.update(); - } - } - - public List getSizeMeasures() - { - return sizeMeasures; - } - - private void setSizeMeasure(int index, double value) - { - sizeMeasureTotal += value - sizeMeasures.get(index); - sizeMeasures.set(index, value); - } - - public double getSizeMeasureTotal() - { - return sizeMeasureTotal; - } - - public int getCount() - { - return paths.size(); - } - - public Path get(int index) - { - return paths.get(index); - } - - public boolean areSizeMeasuresUpdated() - { - return sizeMeasuresUpdated; - } - - public void clearPathSizeCalculator() - { - sizeCalculator = null; - } - - public void restartPathSizeCalculator() - { - if (sizeCalculator == null) - { - sizeCalculator = new PathSizeCalculator(this); - sizeMeasuresUpdated = true; - } - } - - private class PathSizeCalculator - { - Map> incidenceMap; - List lengths; - PathAlternativeList alternatives; - int nUsingEdge; - double edgeLength; - - private PathSizeCalculator(PathAlternativeList alternatives) - { - incidenceMap = new HashMap>(); - lengths = new ArrayList(); - this.alternatives = alternatives; - if (alternatives.getCount() > 0) - { - for (int i = 0; i < alternatives.getCount(); i++) - { - alternatives.setSizeMeasure(i, 0.0); - update(); - } - } - } - - private void update() - { - lengths.add(0.0); - N previous = null; - E edge; - int index = lengths.size() - 1; - double decrement; - for (N node : alternatives.get(index)) - { - if (previous != null) - { - edge = network.getEdge(previous, node); - if (!incidenceMap.containsKey(edge)) - { - incidenceMap.put(edge, new ArrayList()); - } - incidenceMap.get(edge).add(index); - edgeLength = lengthEvaluator.evaluate(edge); - lengths.set(index, lengths.get(index) + edgeLength); - nUsingEdge = incidenceMap.get(edge).size(); - alternatives.setSizeMeasure(index, alternatives.getSizeMeasures().get(index) - + edgeLength / nUsingEdge); - for (Integer i : incidenceMap.get(edge).subList(0, nUsingEdge - 1)) - { - decrement = edgeLength / lengths.get(i) / (nUsingEdge) / (nUsingEdge - 1); - alternatives.setSizeMeasure(i, alternatives.getSizeMeasures().get(i) - - decrement); - } - } - previous = node; - } - alternatives.setSizeMeasure(index, - alternatives.getSizeMeasures().get(index) / lengths.get(index)); - } - - } - - public NodePair getODPair() - { - return odPair; - } - - public EdgeEvaluator getLengthEvaluator() - { - return lengthEvaluator; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java deleted file mode 100644 index 756f2ad..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Map; -import java.util.Set; - -public interface PathAlternativeListGenerationConfiguration, T extends Traversal> -{ - public Network getNetwork(); - - public EdgeEvaluator getEdgeLengthEvaluator(); - - public EdgeEvaluator getEdgeCostEvaluator(); - - public TraversalEvaluator getTraversalCostEvaluator(); - - public double getMaxCost(); - - public double[] getSampleDistanceBreaks(); - - public double[] getSamplePathSizes(); - - public double[] getSampleMinCounts(); - - public double[] getSampleMaxCounts(); - - public boolean isRandomCostSeeded(); - - public Map> getNearbyZonalDistanceMap(); - - public Map getOriginZonalCentroidIdMap(); - - public Map getDestinationZonalCentroidIdMap(); - - public String getOutputDirectory(); - - public Set getTraceOrigins(); - - public Map getPropertyMap(); - - public Map getInverseOriginZonalCentroidIdMap(); - - public Map getInverseDestinationZonalCentroidIdMap(); - - public boolean isTraceExclusive(); - - public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed); - - public boolean isIntrazonalsNeeded(); - - public double getDefaultMinutesPerMile(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java b/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java deleted file mode 100644 index a446045..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/PathAlternativeListWriter.java +++ /dev/null @@ -1,59 +0,0 @@ -package org.sandag.abm.active; - -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; - -public class PathAlternativeListWriter> - implements AutoCloseable -{ - private FileWriter pathWriter; - private FileWriter linkWriter; - - public PathAlternativeListWriter(String pathFileName, String linkFileName) throws IOException - { - pathWriter = new FileWriter(new File(pathFileName)); - linkWriter = new FileWriter(new File(linkFileName)); - } - - public void writeHeaders() throws IOException - { - pathWriter.write("alt,origNode,destNode,length,size\n"); - linkWriter.write("alt,origNode,destNode,link,fromNode,toNode\n"); - } - - public void write(PathAlternativeList alternativeList) throws IOException - { - Path path; - int index = 1; - for (int i = 0; i < alternativeList.getCount(); i++) - { - path = alternativeList.get(i); - pathWriter.write(index + "," + path.getNode(0).getId() + "," - + path.getNode(path.getLength() - 1).getId() + "," + path.getLength() + "," - + alternativeList.getSizeMeasures().get(i) + "\n"); - N previous = null; - int j = 0; - for (N node : path) - { - if (previous != null) - { - linkWriter.write(index + "," + path.getNode(0).getId() + "," - + path.getNode(path.getLength() - 1).getId() + "," + j + "," - + previous.getId() + "," + node.getId() + "\n"); - } - previous = node; - j++; - } - index++; - } - } - - public void close() throws IOException - { - pathWriter.flush(); - pathWriter.close(); - linkWriter.flush(); - linkWriter.close(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java b/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java deleted file mode 100644 index bf2ba01..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/RepeatedSingleSourceDijkstra.java +++ /dev/null @@ -1,156 +0,0 @@ -package org.sandag.abm.active; - -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.PriorityQueue; -import java.util.Set; - -public class RepeatedSingleSourceDijkstra, T extends Traversal> - implements ShortestPathStrategy -{ - private final Network network; - private final EdgeEvaluator edgeEvaluator; - private final TraversalEvaluator traversalEvaluator; - - public RepeatedSingleSourceDijkstra(Network network, EdgeEvaluator edgeEvaluator, - TraversalEvaluator traversalEvaluator) - { - this.network = network; - this.edgeEvaluator = edgeEvaluator; - this.traversalEvaluator = traversalEvaluator; - } - - private class TraversedEdge - implements Comparable - { - private final E edge; - private final double cost; - private final Path path; - - private TraversedEdge(E edge, double cost, Path path) - { - this.edge = edge; - this.cost = cost; - this.path = path; - } - - public int compareTo(TraversedEdge other) - { - return Double.compare(cost, other.cost); - } - } - - @Override - public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes) - { - return getShortestPaths(originNodes, destinationNodes, Double.POSITIVE_INFINITY); - } - - @Override - public ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, - double maxCost) - { - ModifiableShortestPathResultSet spResults = new BasicShortestPathResultSet<>(); - for (N originNode : originNodes) - spResults.addAll(getShortestPaths(originNode, destinationNodes, maxCost)); - return spResults; - } - - protected ShortestPathResultSet getShortestPaths(N originNode, Set destinationNodes, - double maxCost) - { - - BasicShortestPathResultSet spResults = new BasicShortestPathResultSet<>(); - Map finalCosts = new HashMap<>(); // cost to (and including) - // edge - - PriorityQueue traversalQueue = new PriorityQueue<>(); - - Set targets = new HashSet<>(destinationNodes); - Path basePath = new Path<>(originNode); - - // Don't remove origin node, and then we can force a circle for - // intrazonal trips - // if (targets.contains(originNode)) { - // targets.remove(originNode); - // costs.put(originNode,0.0); - // paths.put(originNode,basePath); - // } - - // initialize traversalQueue and costs - for (N successor : network.getSuccessors(originNode)) - { - E edge = network.getEdge(originNode, successor); - double edgeCost = edgeEvaluator.evaluate(edge); - if (edgeCost < 0) - { - throw new RuntimeException("Negative weight found for edge with fromNode " - + edge.getFromNode().getId() + " and toNode " + edge.getToNode().getId()); - } - - if (edgeCost < maxCost) - { - TraversedEdge traversedEdge = new TraversedEdge(edge, edgeCost, - basePath.extendPath(successor)); - traversalQueue.add(traversedEdge); - } - } - - double traversalCost; - - // dijkstra - while (!traversalQueue.isEmpty() && !targets.isEmpty()) - { - TraversedEdge traversedEdge = traversalQueue.poll(); - E edge = traversedEdge.edge; - - if (finalCosts.containsKey(edge)) // already considered - continue; - Path path = traversedEdge.path; - double cost = traversedEdge.cost; - - finalCosts.put(edge, cost); - N fromNode = edge.getFromNode(); - N toNode = edge.getToNode(); - if (targets.remove(toNode)) - { - spResults.addResult(new NodePair(originNode, toNode), path, cost); - } - - for (N successor : network.getSuccessors(toNode)) - { - if (successor.equals(fromNode)) continue; // no u-turns will be - // allowed, so don't - // pollute heap - T traversal = network.getTraversal(traversedEdge.edge, - network.getEdge(toNode, successor)); - traversalCost = evaluateTraversalCost(traversal); - if (traversalCost < 0) - { - throw new RuntimeException( - "Negative weight found for traversal with start node " - + traversal.getFromEdge().getFromNode().getId() - + ", thru node " + traversal.getFromEdge().getToNode().getId() - + ", and end node " + traversal.getToEdge().getToNode().getId()); - } - traversalCost += cost; - if (traversalCost < maxCost) - traversalQueue.add(new TraversedEdge(traversal.getToEdge(), traversalCost, path - .extendPath(successor))); - } - } - - // Not returning null path references and infinite costs for nodes not - // found for possibility of insufficient memory - - return spResults; - } - - protected double evaluateTraversalCost(T traversal) - { - return edgeEvaluator.evaluate(traversal.getToEdge()) - + traversalEvaluator.evaluate(traversal); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java deleted file mode 100644 index 3804815..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResult.java +++ /dev/null @@ -1,31 +0,0 @@ -package org.sandag.abm.active; - -public class ShortestPathResult -{ - private final NodePair od; - private final Path path; - private final double cost; - - public ShortestPathResult(NodePair od, Path path, double cost) - { - this.od = od; - this.path = path; - this.cost = cost; - } - - public NodePair getOriginDestination() - { - return od; - } - - public Path getPath() - { - return path; - } - - public double getCost() - { - return cost; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java deleted file mode 100644 index cf2334d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathResultSet.java +++ /dev/null @@ -1,13 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; - -public interface ShortestPathResultSet - extends Iterable> -{ - int size(); - - ShortestPathResult getShortestPathResult(NodePair od); - - Collection> getResults(); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java b/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java deleted file mode 100644 index 9d47c64..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/ShortestPathStrategy.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Set; - -public interface ShortestPathStrategy -{ - ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes, - double maxCost); - - ShortestPathResultSet getShortestPaths(Set originNodes, Set destinationNodes); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java deleted file mode 100644 index 0bdbe5b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleEdge.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Objects; - -public class SimpleEdge - implements Edge -{ - private final N fromNode; - private final N toNode; - - public SimpleEdge(N fromNode, N toNode) - { - this.fromNode = fromNode; - this.toNode = toNode; - } - - @Override - public N getFromNode() - { - return fromNode; - } - - @Override - public N getToNode() - { - return toNode; - } - - @Override - public int compareTo(Edge o) - { - int fromResult = this.fromNode.compareTo(o.getFromNode()); - int toResult = this.toNode.compareTo(o.getToNode()); - return fromResult + ((fromResult == 0) ? 1 : 0) * toResult; - } - - @Override - public int hashCode() - { - return Objects.hash(fromNode, toNode); - } - - @Override - public boolean equals(Object o) - { - if ((o == null) || !(o instanceof Edge)) return false; - Edge other = (Edge) o; - return fromNode.equals(other.getFromNode()) && toNode.equals(other.getToNode()); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java deleted file mode 100644 index d88553b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNetwork.java +++ /dev/null @@ -1,141 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Collection; -import java.util.Collections; -import java.util.HashMap; -import java.util.Iterator; -import java.util.LinkedHashMap; -import java.util.LinkedList; -import java.util.Map; - -public class SimpleNetwork, T extends Traversal> - implements Network -{ - private final Map nodes; - private final Map, E> edges; - private final Map traversals; - - private final Map> successors; - private final Map> predecessors; - - public SimpleNetwork(Collection nodes, Collection edges, Collection traversals) - { - this.nodes = new LinkedHashMap<>(); // use LinkedHashMap for fast - // iteration over keys - this.edges = new LinkedHashMap<>(); - this.traversals = new LinkedHashMap<>(); - - for (N node : nodes) - this.nodes.put(node.getId(), node); - for (E edge : edges) - this.edges.put(new NodePair(edge.getFromNode(), edge.getToNode()), edge); - for (T traversal : traversals) - this.traversals.put(new EdgePair(traversal.getFromEdge(), traversal.getToEdge()), - traversal); - - successors = new HashMap<>(); // save memory and insertion time over - // LinkedHashMap - predecessors = new HashMap<>(); - - for (N node : this.nodes.values()) - { - successors.put(node, new LinkedList()); - predecessors.put(node, new LinkedList()); - } - for (NodePair nodePair : this.edges.keySet()) - { - N from = nodePair.getFromNode(); - N to = nodePair.getToNode(); - successors.get(from).add(to); - predecessors.get(to).add(from); - } - } - - @Override - public N getNode(int nodeId) - { - return nodes.get(nodeId); - } - - @Override - public E getEdge(N fromNode, N toNode) - { - return getEdge(new NodePair(fromNode, toNode)); - } - - @Override - public E getEdge(NodePair nodes) - { - return edges.get(nodes); - } - - @Override - public T getTraversal(E fromEdge, E toEdge) - { - return traversals.get(new EdgePair(fromEdge, toEdge)); - } - - @Override - public Collection getSuccessors(Node node) - { - return Collections.unmodifiableCollection(successors.get(node)); - } - - @Override - public Collection getPredecessors(Node node) - { - return Collections.unmodifiableCollection(predecessors.get(node)); - } - - @Override - public Iterator nodeIterator() - { - return nodes.values().iterator(); - } - - @Override - public Iterator edgeIterator() - { - return edges.values().iterator(); - } - - @Override - public Iterator traversalIterator() - { - return traversals.values().iterator(); - } - - @Override - public boolean containsNodeId(int id) - { - return nodes.containsKey(id); - } - - @Override - public boolean containsNode(Node node) - { - return nodes.containsValue(node); - } - - @Override - public boolean containsEdge(N fromNode, N toNode) - { - return edges.containsKey(new NodePair(fromNode, toNode)); - } - - @Override - public boolean containsTraversal(E fromEdge, E toEdge) - { - return traversals.containsKey(new EdgePair(fromEdge, toEdge)); - } - - private class EdgePair - extends SimpleTraversal - { - public EdgePair(E fromEdge, E toEdge) - { - super(fromEdge, toEdge); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java deleted file mode 100644 index f3b6bc9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleNode.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Objects; - -public class SimpleNode - implements Node -{ - private final int id; - - public SimpleNode(int id) - { - this.id = id; - } - - @Override - public int getId() - { - return id; - } - - @Override - public int compareTo(Node node) - { - return Integer.compare(id, node.getId()); - } - - @Override - public int hashCode() - { - return Objects.hash(id); - } - - @Override - public boolean equals(Object o) - { - if ((o == null) || !(o instanceof Node)) return false; - return id == ((Node) o).getId(); - } - - public String toString() - { - return ""; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java deleted file mode 100644 index 02b7f49..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/SimpleTraversal.java +++ /dev/null @@ -1,46 +0,0 @@ -package org.sandag.abm.active; - -import java.util.Objects; - -public class SimpleTraversal> - implements Traversal -{ - private final E fromEdge; - private final E toEdge; - - public SimpleTraversal(E fromEdge, E toEdge) - { - this.fromEdge = fromEdge; - this.toEdge = toEdge; - } - - @Override - public E getFromEdge() - { - return fromEdge; - } - - @Override - public E getToEdge() - { - return toEdge; - } - - @Override - public int hashCode() - { - return Objects.hash(fromEdge, toEdge); - } - - @Override - public boolean equals(Object obj) - { - if ((obj == null) || (!(obj instanceof Traversal))) return false; - Traversal traversal = (Traversal) obj; - if (fromEdge == null) return (fromEdge == traversal.getFromEdge()) - && (toEdge.equals(traversal.getToEdge())); - else return (fromEdge.equals(traversal.getFromEdge())) - && (toEdge.equals(traversal.getToEdge())); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/Traversal.java deleted file mode 100644 index 6e508e144dc957939a3ef1e872486f5df66c528d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 131 LcmZQz7;pdp0D}Ml diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java b/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java deleted file mode 100644 index 0e4c197..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/TraversalEvaluator.java +++ /dev/null @@ -1,6 +0,0 @@ -package org.sandag.abm.active; - -public interface TraversalEvaluator> -{ - double evaluate(T traversal); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java deleted file mode 100644 index 2729f27..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/BikeAssignmentTripReader.java +++ /dev/null @@ -1,216 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.io.BufferedReader; -import java.io.FileReader; -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Stop; -import org.sandag.abm.ctramp.Tour; - -public class BikeAssignmentTripReader -{ - private String indivTripFileName, jointTripFileName, indivTourFileName, - jointTourFileName, personFileName, hhFileName; - private static final int DEFAULT_MANDATORY_PURPOSE_INDEX = 1; - private static final int DEFAULT_NONMANDATORY_PURPOSE_INDEX = 4; - private static final int BIKE_MODE_INDEX = 10; - - private static String PROPERTIES_HOUSEHOLD_FILENAME = "PopulationSynthesizer.InputToCTRAMP.HouseholdFile"; - private static String PROPERTIES_PERSON_FILENAME = "PopulationSynthesizer.InputToCTRAMP.PersonFile"; - private static String PROPERTIES_INDIV_TOUR_FILENAME = "Results.IndivTourDataFile"; - private static String PROPERTIES_JOINT_TOUR_FILENAME = "Results.JointTourDataFile"; - private static String PROPERTIES_INDIV_TRIP_FILENAME = "Results.IndivTripDataFile"; - private static String PROPERTIES_JOINT_TRIP_FILENAME = "Results.JointTripDataFile"; - private static String PROPERTIES_PROJECT_DIR = "Project.Directory"; - - public BikeAssignmentTripReader(Map propertyMap) - { - String dir = propertyMap.get(PROPERTIES_PROJECT_DIR); - this.indivTripFileName = dir + "/" + propertyMap.get(PROPERTIES_INDIV_TRIP_FILENAME); - this.jointTripFileName = dir + "/" + propertyMap.get(PROPERTIES_JOINT_TRIP_FILENAME); - this.indivTourFileName = dir + "/" + propertyMap.get(PROPERTIES_INDIV_TOUR_FILENAME); - this.jointTourFileName = dir + "/" + propertyMap.get(PROPERTIES_JOINT_TOUR_FILENAME); - this.personFileName = dir + "/" + propertyMap.get(PROPERTIES_PERSON_FILENAME); - this.hhFileName = dir + "/" + propertyMap.get(PROPERTIES_HOUSEHOLD_FILENAME); - } - - public BikeAssignmentTripReader(Map propertyMap, int iter) - { - String dir = propertyMap.get(PROPERTIES_PROJECT_DIR); - this.indivTripFileName = dir - + "/" - + propertyMap.get(PROPERTIES_INDIV_TRIP_FILENAME).substring(0, - PROPERTIES_INDIV_TRIP_FILENAME.length() - 5) + "_" + iter + ".csv"; - this.jointTripFileName = dir - + "/" - + propertyMap.get(PROPERTIES_JOINT_TRIP_FILENAME).substring(0, - PROPERTIES_JOINT_TRIP_FILENAME.length() - 5) + "_" + iter + ".csv"; - this.indivTourFileName = dir - + "/" - + propertyMap.get(PROPERTIES_INDIV_TOUR_FILENAME).substring(0, - PROPERTIES_INDIV_TOUR_FILENAME.length() - 5) + "_" + iter + ".csv"; - this.jointTourFileName = dir - + "/" - + propertyMap.get(PROPERTIES_JOINT_TOUR_FILENAME).substring(0, - PROPERTIES_JOINT_TOUR_FILENAME.length() - 5) + "_" + iter + ".csv"; - this.personFileName = dir + "/" + propertyMap.get(PROPERTIES_PERSON_FILENAME); - this.hhFileName = dir + "/" + propertyMap.get(PROPERTIES_HOUSEHOLD_FILENAME); - } - - public List createTripList() - { - - SandagModelStructure modelStructure = new SandagModelStructure(); - - Map indivTourMap = new HashMap<>(); - Map> jointTourMap = new HashMap<>(); - Map hhMap = new HashMap<>(); - List stops = new ArrayList<>(); - - try - { - - String line; - BufferedReader reader = new BufferedReader(new FileReader(hhFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - Household h = new Household(modelStructure); - int hhSize = Integer.parseInt(row[8]); - h.setHhSize(hhSize); - int hhId = Integer.parseInt(row[0]); - hhMap.put(hhId, h); - } - reader.close(); - - reader = new BufferedReader(new FileReader(personFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - int hhId = Integer.parseInt(row[0]); - int perId = Integer.parseInt(row[1]); - int perNo = Integer.parseInt(row[3]); - Household h = hhMap.get(hhId); - Person p = h.getPerson(perNo); - int gender = Integer.parseInt(row[5]); - p.setPersGender(gender); - p.setPersId(perId); - } - reader.close(); - - reader = new BufferedReader(new FileReader(indivTourFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - int hhId = Integer.parseInt(row[0]); - int perId = Integer.parseInt(row[1]); - int perNo = Integer.parseInt(row[2]); - int tourId = Integer.parseInt(row[4]); - String tourCategory = row[5]; - int tourPurpose = DEFAULT_MANDATORY_PURPOSE_INDEX; - if (tourCategory != "MANDATORY") - { - tourPurpose = DEFAULT_NONMANDATORY_PURPOSE_INDEX; - } - Tour t = new Tour(hhMap.get(hhId).getPerson(perNo), tourId, tourPurpose); - if (!indivTourMap.containsKey(perId)) - { - indivTourMap.put(perId, new Tour[20]); - } - indivTourMap.get(perId)[tourId] = t; - } - reader.close(); - - reader = new BufferedReader(new FileReader(jointTourFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - int hhId = Integer.parseInt(row[0]); - int tourId = Integer.parseInt(row[1]); - String[] tourParty = row[5].split(" "); - Household h = hhMap.get(hhId); - int[] participantArray = new int[tourParty.length]; - for (int i = 0; i < participantArray.length; i++) - { - participantArray[i] = Integer.parseInt(tourParty[i]); - } - Tour t = new Tour(h, "", modelStructure.JOINT_NON_MANDATORY_CATEGORY, - DEFAULT_NONMANDATORY_PURPOSE_INDEX); - t.setPersonObject(h.getPerson(participantArray[0])); - t.setPersonNumArray(participantArray); - t.setTourId(tourId); - if (!jointTourMap.containsKey(hhId)) - { - jointTourMap.put(hhId, new ArrayList()); - } - jointTourMap.get(hhId).add(t); - } - reader.close(); - - reader = new BufferedReader(new FileReader(indivTripFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - int tripMode = Integer.parseInt(row[13]); - if (tripMode == BIKE_MODE_INDEX) - { - int perId = Integer.parseInt(row[1]); - int tourId = Integer.parseInt(row[3]); - boolean inbound = Boolean.parseBoolean(row[5]); - Tour t = indivTourMap.get(perId)[tourId]; - Stop s = new Stop(t, "", "", 0, inbound, 0); - int stopPeriod = Integer.parseInt(row[12]); - s.setStopPeriod(stopPeriod); - int oMgra = Integer.parseInt(row[9]); - int dMgra = Integer.parseInt(row[10]); - s.setOrig(oMgra); - s.setDest(dMgra); - stops.add(s); - } - } - reader.close(); - - reader = new BufferedReader(new FileReader(jointTripFileName)); - reader.readLine(); - while ((line = reader.readLine()) != null) - { - String[] row = line.split(","); - int tripMode = Integer.parseInt(row[11]); - if (tripMode == BIKE_MODE_INDEX) - { - int hhId = Integer.parseInt(row[0]); - int tourId = Integer.parseInt(row[1]); - boolean inbound = Boolean.parseBoolean(row[3]); - Tour t = jointTourMap.get(hhId).get(tourId); - Stop s = new Stop(t, "", "", 0, inbound, 0); - int stopPeriod = Integer.parseInt(row[10]); - s.setStopPeriod(stopPeriod); - int oMgra = Integer.parseInt(row[7]); - int dMgra = Integer.parseInt(row[8]); - s.setOrig(oMgra); - s.setDest(dMgra); - stops.add(s); - } - } - reader.close(); - - } catch (IOException e) - { - throw new RuntimeException(e); - } - - return stops; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java deleted file mode 100644 index 1ffdbbb..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/PropertyParser.java +++ /dev/null @@ -1,99 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -public class PropertyParser -{ - public Map propertyMap; - - public PropertyParser(Map propertyMap) - { - this.propertyMap = propertyMap; - } - - public List parseStringPropertyList(String property) - { - return Arrays.asList(propertyMap.get(property.trim()).split("\\s*,\\s*")); - } - - public List parseFloatPropertyList(String property) - { - List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); - List floatList = new ArrayList(); - for (String str : stringList) - { - floatList.add(Float.parseFloat(str)); - } - return floatList; - } - - public double[] parseDoublePropertyArray(String property) - { - List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); - double[] array = new double[stringList.size()]; - for (int i = 0; i < stringList.size(); i++) - { - array[i] = Double.parseDouble(stringList.get(i)); - } - return array; - } - - public List parseIntPropertyList(String property) - { - List stringList = Arrays.asList(propertyMap.get(property).split("\\s*,\\s*")); - List intList = new ArrayList(); - for (String str : stringList) - { - intList.add(Integer.parseInt(str)); - } - return intList; - } - - public Map mapStringPropertyListToStrings(String keyProperty, - String stringValueProperty) - { - Map map = new HashMap(); - List keys = parseStringPropertyList(keyProperty); - List values = parseStringPropertyList(stringValueProperty); - for (int i = 0; i < keys.size(); i += 1) - { - map.put(keys.get(i), values.get(i)); - } - return map; - } - - public Map mapStringPropertyListToFloats(String keyProperty, - String stringValueProperty) - { - Map map = new HashMap(); - List keys = parseStringPropertyList(keyProperty); - List values = parseFloatPropertyList(stringValueProperty); - for (int i = 0; i < keys.size(); i += 1) - { - map.put(keys.get(i), values.get(i)); - } - return map; - } - - private boolean isIntValueIn(int value, List referenceValues) - { - for (int v : referenceValues) - { - if (v == value) - { - return true; - } - } - return false; - } - - public boolean isIntValueInPropertyList(int value, String property) - { - List referenceValues = parseIntPropertyList(property); - return isIntValueIn(value, referenceValues); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java deleted file mode 100644 index 7b5a52f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeEdge.java +++ /dev/null @@ -1,19 +0,0 @@ -package org.sandag.abm.active.sandag; - -import org.sandag.abm.active.SimpleEdge; - -public class SandagBikeEdge - extends SimpleEdge -{ - public volatile byte bikeClass, lanes, functionalClass; - public volatile boolean centroidConnector, autosPermitted, cycleTrack, bikeBlvd; - public volatile float distance, scenicIndex; - public volatile short gain; - public volatile double bikeCost, walkCost; - public long roadsegid; - - public SandagBikeEdge(SandagBikeNode fromNode, SandagBikeNode toNode) - { - super(fromNode, toNode); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index 8280f42..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeMgraPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,36 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import org.sandag.abm.active.Network; - -public class SandagBikeMgraPathAlternativeListGenerationConfiguration - extends SandagBikePathAlternativeListGenerationConfiguration -{ - - public SandagBikeMgraPathAlternativeListGenerationConfiguration( - Map propertyMap, - Network network) - { - super(propertyMap, network); - this.PROPERTIES_MAXDIST_ZONE = Double.parseDouble(propertyMap.get("active.maxdist.bike.mgra")); - this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; - } - - protected void createZonalCentroidIdMap() - { - System.out.println("Creating MGRA Zonal Centroid Id Map..."); - zonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.mgra > 0) - { - zonalCentroidIdMap.put((int) n.mgra, n.getId()); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java deleted file mode 100644 index e5aebca..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNetworkFactory.java +++ /dev/null @@ -1,581 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.lang.reflect.Field; -import java.util.Collection; -import java.util.HashMap; -import java.util.Iterator; -import java.util.LinkedHashSet; -import java.util.Map; -import java.util.Set; -import org.apache.log4j.Logger; -import org.sandag.abm.active.AbstractNetworkFactory; -import org.sandag.abm.active.Network; -import com.linuxense.javadbf.DBFReader; - -public class SandagBikeNetworkFactory - extends AbstractNetworkFactory -{ - protected Logger logger = Logger.getLogger(SandagBikeNetworkFactory.class); - private Map propertyMap; - private PropertyParser propertyParser; - private Collection nodes = null; - private Collection edges = null; - private Collection traversals = null; - - private static final String PROPERTIES_NODE_FILE = "active.node.file"; - private static final String PROPERTIES_NODE_ID = "active.node.id"; - private static final String PROPERTIES_NODE_FIELDNAMES = "active.node.fieldnames"; - private static final String PROPERTIES_NODE_COLUMNS = "active.node.columns"; - private static final String PROPERTIES_EDGE_FILE = "active.edge.file"; - private static final String PROPERTIES_EDGE_ANODE = "active.edge.anode"; - private static final String PROPERTIES_EDGE_BNODE = "active.edge.bnode"; - private static final String PROPERTIES_EDGE_DIRECTIONAL = "active.edge.directional"; - private static final String PROPERTIES_EDGE_FIELDNAMES = "active.edge.fieldnames"; - private static final String PROPERTIES_EDGE_COLUMNS_AB = "active.edge.columns.ab"; - private static final String PROPERTIES_EDGE_COLUMNS_BA = "active.edge.columns.ba"; - private static final String PROPERTIES_EDGE_CENTROID_FIELD = "active.edge.centroid.field"; - private static final String PROPERTIES_EDGE_CENTROID_VALUE = "active.edge.centroid.value"; - private static final String PROPERTIES_EDGE_AUTOSPERMITTED_FIELD = "active.edge.autospermitted.field"; - private static final String PROPERTIES_EDGE_AUTOSPERMITTED_VALUES = "active.edge.autospermitted.values"; - - private static final double TURN_ANGLE_TOLERANCE = Math.PI / 6; - private static final double DISTANCE_CONVERSION_FACTOR = 0.000189; - private static final double INACCESSIBLE_COST_COEF = 999.0; - - private static final String PROPERTIES_COEF_DISTCLA0 = "active.coef.distcla0"; - private static final String PROPERTIES_COEF_DISTCLA1 = "active.coef.distcla1"; - private static final String PROPERTIES_COEF_DISTCLA2 = "active.coef.distcla2"; - private static final String PROPERTIES_COEF_DISTCLA3 = "active.coef.distcla3"; - private static final String PROPERTIES_COEF_DARTNE2 = "active.coef.dartne2"; - private static final String PROPERTIES_COEF_DWRONGWY = "active.coef.dwrongwy"; - private static final String PROPERTIES_COEF_GAIN = "active.coef.gain"; - private static final String PROPERTIES_COEF_TURN = "active.coef.turn"; - private static final String PROPERTIES_COEF_GAIN_WALK = "active.coef.gain.walk"; - private static final String PROPERTIES_COEF_DCYCTRAC = "active.coef.dcyctrac"; - private static final String PROPERTIES_COEF_DBIKBLVD = "active.coef.dbikblvd"; - private static final String PROPERTIES_COEF_SIGNALS = "active.coef.signals"; - private static final String PROPERTIES_COEF_UNLFRMA = "active.coef.unlfrma"; - private static final String PROPERTIES_COEF_UNLFRMI = "active.coef.unlfrmi"; - private static final String PROPERTIES_COEF_UNTOMA = "active.coef.untoma"; - private static final String PROPERTIES_COEF_UNTOMI = "active.coef.untomi"; - private static final String PROPERTIES_COEF_DISTANCE_WALK = "active.walk.minutes.per.mile"; - - - public SandagBikeNetworkFactory(Map propertyMap) - { - this.propertyMap = propertyMap; - propertyParser = new PropertyParser(propertyMap); - } - - protected Collection readNodes() - { - Set nodes = new LinkedHashSet<>(); - try - { - InputStream stream = new FileInputStream(propertyMap.get(PROPERTIES_NODE_FILE)); - DBFReader reader = new DBFReader(stream); - Map fieldMap = propertyParser.mapStringPropertyListToStrings( - PROPERTIES_NODE_FIELDNAMES, PROPERTIES_NODE_COLUMNS); - Field f; - int fieldCount = reader.getFieldCount(); - Map labels = new HashMap(); - for (int i = 0; i < fieldCount; i++) - { - labels.put(reader.getField(i).getName(), i); - } - Object[] rowObjects; - while ((rowObjects = reader.nextRecord()) != null) - { - int id = ((Number) rowObjects[labels.get(propertyMap.get(PROPERTIES_NODE_ID))]) - .intValue(); - SandagBikeNode node = new SandagBikeNode(id); - for (String fieldName : fieldMap.keySet()) - { - try - { - f = node.getClass().getField(fieldName); - setNumericFieldWithCast(node, f, - (Number) rowObjects[labels.get(fieldMap.get(fieldName))]); - } catch (NoSuchFieldException | SecurityException e) - { - logger.error("Exception caught getting class field " + fieldName - + " for object of class " + node.getClass().getName(), e); - throw new RuntimeException(); - } - } - nodes.add(node); - } - } catch (IOException e) - { - logger.error("Exception caught reading nodes from disk.", e); - throw new RuntimeException(); - } - return nodes; - } - - protected Collection readEdges(Collection nodes) - { - Set edges = new LinkedHashSet<>(); - Map idNodeMap = new HashMap<>(); - for (SandagBikeNode node : nodes) - idNodeMap.put(node.getId(), node); - - try - { - InputStream stream = new FileInputStream(propertyMap.get(PROPERTIES_EDGE_FILE)); - DBFReader reader = new DBFReader(stream); - Map abFieldMap = propertyParser.mapStringPropertyListToStrings( - PROPERTIES_EDGE_FIELDNAMES, PROPERTIES_EDGE_COLUMNS_AB); - Map baFieldMap = new HashMap(); - boolean directional = Boolean - .parseBoolean(propertyMap.get(PROPERTIES_EDGE_DIRECTIONAL)); - if (!directional) - { - baFieldMap = propertyParser.mapStringPropertyListToStrings( - PROPERTIES_EDGE_FIELDNAMES, PROPERTIES_EDGE_COLUMNS_BA); - } - int columnCount = reader.getFieldCount(); - Map labels = new HashMap(); - for (int i = 0; i < columnCount; i++) - { - labels.put(reader.getField(i).getName(), i); - } - Object[] rowObjects; - while ((rowObjects = reader.nextRecord()) != null) - { - SandagBikeNode a = idNodeMap.get(((Number) rowObjects[labels.get(propertyMap - .get(PROPERTIES_EDGE_ANODE))]).intValue()); - SandagBikeNode b = idNodeMap.get(((Number) rowObjects[labels.get(propertyMap - .get(PROPERTIES_EDGE_BNODE))]).intValue()); - - SandagBikeEdge edge = new SandagBikeEdge(a, b); - for (String fieldName : abFieldMap.keySet()) - { - try - { - Field f = edge.getClass().getField(fieldName); - setNumericFieldWithCast(edge, f, - (Number) rowObjects[labels.get(abFieldMap.get(fieldName))]); - } catch (NoSuchFieldException | SecurityException e) - { - logger.error("Exception caught getting class field " + fieldName - + " for object of class " + edge.getClass().getName(), e); - throw new RuntimeException(); - } - } - edges.add(edge); - - if (!directional) - { - edge = new SandagBikeEdge(b, a); - for (String fieldName : baFieldMap.keySet()) - { - try - { - Field f = edge.getClass().getField(fieldName); - setNumericFieldWithCast(edge, f, - (Number) rowObjects[labels.get(baFieldMap.get(fieldName))]); - } catch (NoSuchFieldException | SecurityException e) - { - logger.error("Exception caught getting class field " + fieldName - + " for object of class " + edge.getClass().getName(), e); - throw new RuntimeException(); - } - } - edges.add(edge); - } - } - } catch (IOException e) - { - logger.error("Exception caught reading edges from disk.", e); - throw new RuntimeException(); - } - return edges; - } - - @Override - protected void calculateDerivedNodeAttributes( - Network network) - { - Iterator nodeIterator = network.nodeIterator(); - while (nodeIterator.hasNext()) - { - SandagBikeNode n = nodeIterator.next(); - n.centroid = (n.mgra > 0) || (n.taz > 0); - if (n.mgra > 0) - { - n.taz = 0; - } - } - } - - @Override - protected void calculateDerivedEdgeAttributes( - Network network) - { - try - { - Iterator edgeIterator = network.edgeIterator(); - Field apf = SandagBikeEdge.class.getField(propertyMap - .get(PROPERTIES_EDGE_AUTOSPERMITTED_FIELD)); - Field cf = SandagBikeEdge.class.getField(propertyMap - .get(PROPERTIES_EDGE_CENTROID_FIELD)); - while (edgeIterator.hasNext()) - { - SandagBikeEdge edge = edgeIterator.next(); - edge.autosPermitted = propertyParser.isIntValueInPropertyList(apf.getInt(edge), - PROPERTIES_EDGE_AUTOSPERMITTED_VALUES); - edge.centroidConnector = propertyParser.isIntValueInPropertyList(cf.getInt(edge), - PROPERTIES_EDGE_CENTROID_VALUE); - edge.distance = edge.distance * (float) DISTANCE_CONVERSION_FACTOR; - edge.bikeCost = (double) edge.distance - * (Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA0)) - * ((edge.bikeClass < 1 ? 1 : 0) + (edge.bikeClass > 3 ? 1 : 0)) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA1)) - * (edge.bikeClass == 1 ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA2)) - * (edge.bikeClass == 2 ? 1 : 0) * (edge.cycleTrack ? 0 : 1) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA3)) - * (edge.bikeClass == 3 ? 1 : 0) * (edge.bikeBlvd ? 0 : 1) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DARTNE2)) - * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) - * ((edge.functionalClass < 4 && edge.functionalClass > 0) ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DWRONGWY)) - * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DCYCTRAC)) - * (edge.cycleTrack ? 1 : 0) + Double.parseDouble(propertyMap - .get(PROPERTIES_COEF_DBIKBLVD)) * (edge.bikeBlvd ? 1 : 0)) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN)) * edge.gain - + INACCESSIBLE_COST_COEF - * ((edge.functionalClass < 3 && edge.functionalClass > 0) ? 1 : 0); - edge.walkCost = (double) edge.distance - * Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTANCE_WALK)) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN_WALK)) - * edge.gain; - } - - } catch (NoSuchFieldException | IllegalAccessException e) - { - logger.error("Exception caught calculating derived edge attributes.", e); - throw new RuntimeException(); - } - } - - @Override - protected void calculateDerivedTraversalAttributes( - Network network) - { - Iterator traversalIterator = network.traversalIterator(); - while (traversalIterator.hasNext()) - { - SandagBikeTraversal t = traversalIterator.next(); - t.turnType = calculateTurnType(t, network); - t.thruCentroid = t.getFromEdge().centroidConnector && t.getToEdge().centroidConnector; - boolean signalized = t.getFromEdge().getToNode().signalized; - boolean fromMajorArt = t.getFromEdge().functionalClass <= 3 - && t.getFromEdge().functionalClass > 0 && t.getFromEdge().bikeClass != 1; - boolean fromMinorArt = t.getFromEdge().functionalClass == 4 - && t.getFromEdge().bikeClass != 1; - t.signalExclRightAndThruJunction = signalized && t.turnType != TurnType.RIGHT - && !isThruJunction(t, network); - t.unsigLeftFromMajorArt = !signalized && fromMajorArt && t.turnType == TurnType.LEFT; - t.unsigLeftFromMinorArt = !signalized && fromMinorArt && t.turnType == TurnType.LEFT; - t.unsigCrossMajorArt = !signalized && isCrossingOfMajorArterial(t, network); - t.unsigCrossMinorArt = !signalized && isCrossingOfMinorArterial(t, network); - t.cost = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_TURN)) - * ((t.turnType != TurnType.NONE) ? 1 : 0) + INACCESSIBLE_COST_COEF - * (t.thruCentroid ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_SIGNALS)) - * (t.signalExclRightAndThruJunction ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNLFRMA)) - * (t.unsigLeftFromMajorArt ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNLFRMI)) - * (t.unsigLeftFromMinorArt ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNTOMA)) - * (t.unsigCrossMajorArt ? 1 : 0) - + Double.parseDouble(propertyMap.get(PROPERTIES_COEF_UNTOMI)) - * (t.unsigCrossMinorArt ? 1 : 0); - } - } - - private boolean isCrossingOfMajorArterial(SandagBikeTraversal t, - Network network) - { - SandagBikeNode startNode = t.getFromEdge().getFromNode(); - SandagBikeNode thruNode = t.getFromEdge().getToNode(); - SandagBikeNode endNode = t.getToEdge().getToNode(); - - if (startNode.centroid || thruNode.centroid || endNode.centroid) - { - return false; - } - - SandagBikeEdge edge = t.getToEdge(); - if (t.turnType == TurnType.LEFT && edge.functionalClass <= 3 && edge.functionalClass > 0 - && edge.bikeClass != 1) - { - return true; - } - - if (t.turnType != TurnType.NONE) - { - return false; - } - - int majorArtCount = 0; - for (SandagBikeNode successor : network.getSuccessors(thruNode)) - { - edge = network.getEdge(thruNode, successor); - boolean majorArt = edge.functionalClass <= 3 && edge.functionalClass > 0 - && edge.bikeClass != 1; - if (majorArt && (!(successor.equals(startNode))) && (!(successor.equals(endNode)))) - { - majorArtCount += 1; - } - } - - return majorArtCount >= 2; - } - - private boolean isCrossingOfMinorArterial(SandagBikeTraversal t, - Network network) - { - if (isCrossingOfMajorArterial(t, network)) - { - return false; - } - - SandagBikeNode startNode = t.getFromEdge().getFromNode(); - SandagBikeNode thruNode = t.getFromEdge().getToNode(); - SandagBikeNode endNode = t.getToEdge().getToNode(); - - if (startNode.centroid || thruNode.centroid || endNode.centroid) - { - return false; - } - - SandagBikeEdge edge = t.getToEdge(); - if (t.turnType == TurnType.LEFT && edge.functionalClass == 4 && edge.bikeClass != 1) - { - return true; - } - - if (t.turnType != TurnType.NONE) - { - return false; - } - - int artCount = 0; - for (SandagBikeNode successor : network.getSuccessors(thruNode)) - { - edge = network.getEdge(thruNode, successor); - boolean art = edge.functionalClass <= 4 && edge.functionalClass > 0 - && edge.bikeClass != 1; - if (art && (!(successor.equals(startNode))) && (!(successor.equals(endNode)))) - { - artCount += 1; - } - } - - return artCount >= 2; - } - - private boolean isThruJunction(SandagBikeTraversal t, - Network network) - { - SandagBikeNode startNode = t.getFromEdge().getFromNode(); - SandagBikeNode thruNode = t.getFromEdge().getToNode(); - SandagBikeNode endNode = t.getToEdge().getToNode(); - - if (startNode.centroid || thruNode.centroid || endNode.centroid) - { - return false; - } - - if (t.turnType != TurnType.NONE) - { - return false; - } - - boolean rightTurnExists = false; - for (SandagBikeNode successor : network.getSuccessors(thruNode)) - { - SandagBikeTraversal traversal = network.getTraversal(t.getFromEdge(), - network.getEdge(thruNode, successor)); - if ((!(successor.equals(startNode))) && (!(successor.equals(endNode)))) - { - rightTurnExists = traversal.turnType == TurnType.NONE; - } - } - - return !rightTurnExists; - } - - private double calculateTraversalAngle(SandagBikeTraversal t) - { - float xDiff1 = t.getFromEdge().getToNode().x - t.getFromEdge().getFromNode().x; - float xDiff2 = t.getToEdge().getToNode().x - t.getToEdge().getFromNode().x; - float yDiff1 = t.getFromEdge().getToNode().y - t.getFromEdge().getFromNode().y; - float yDiff2 = t.getToEdge().getToNode().y - t.getToEdge().getFromNode().y; - - double angle = Math.atan2(yDiff2, xDiff2) - Math.atan2(yDiff1, xDiff1); - - if (angle > Math.PI) - { - angle = angle - 2 * Math.PI; - } - if (angle < -Math.PI) - { - angle = angle + 2 * Math.PI; - } - - return angle; - } - - private TurnType calculateTurnType(SandagBikeTraversal t, - Network network) - { - SandagBikeNode startNode = t.getFromEdge().getFromNode(); - SandagBikeNode thruNode = t.getFromEdge().getToNode(); - SandagBikeNode endNode = t.getToEdge().getToNode(); - - if (startNode.centroid || thruNode.centroid || endNode.centroid) - { - return TurnType.NONE; - } - - if (startNode.equals(endNode)) - { - return TurnType.REVERSAL; - } - - double thisAngle = calculateTraversalAngle(t); - - if (thisAngle < -Math.PI + TURN_ANGLE_TOLERANCE - || thisAngle > Math.PI - TURN_ANGLE_TOLERANCE) - { - return TurnType.REVERSAL; - } - - double minAngle = Math.PI; - double maxAngle = -Math.PI; - double minAbsAngle = Math.PI; - double currentAngle; - int legCount = 1; - - SandagBikeEdge startEdge = network.getEdge(startNode, thruNode); - for (SandagBikeNode successor : network.getSuccessors(thruNode)) - { - SandagBikeEdge edge = network.getEdge(thruNode, successor); - if (edge.autosPermitted && (!(successor.equals(startNode)))) - { - currentAngle = calculateTraversalAngle(network.getTraversal(startEdge, edge)); - minAngle = Math.min(minAngle, currentAngle); - maxAngle = Math.max(maxAngle, currentAngle); - minAbsAngle = Math.min(minAbsAngle, Math.abs(currentAngle)); - legCount += 1; - } - } - - if (legCount <= 2) - { - return TurnType.NONE; - } else if (legCount == 3) - { - if (thisAngle <= minAngle && Math.abs(thisAngle) > TURN_ANGLE_TOLERANCE) - { - return TurnType.RIGHT; - } else if (thisAngle >= maxAngle && Math.abs(thisAngle) > TURN_ANGLE_TOLERANCE) - { - return TurnType.LEFT; - } else - { - return TurnType.NONE; - } - } else - { - if (Math.abs(thisAngle) <= minAbsAngle - || (Math.abs(thisAngle) < TURN_ANGLE_TOLERANCE && thisAngle > minAngle && thisAngle < maxAngle)) - { - return TurnType.NONE; - } else if (thisAngle < 0) - { - return TurnType.RIGHT; - } else - { - return TurnType.LEFT; - } - } - } - - private void setNumericFieldWithCast(Object o, Field f, Number n) - { - Class c = f.getType(); - try - { - if (c.equals(Integer.class) || c.equals(Integer.TYPE)) - { - f.set(o, n.intValue()); - } else if (c.equals(Float.class) || c.equals(Float.TYPE)) - { - f.set(o, n.floatValue()); - } else if (c.equals(Double.class) || c.equals(Double.TYPE)) - { - f.set(o, n.doubleValue()); - } else if (c.equals(Boolean.class) || c.equals(Boolean.TYPE)) - { - f.set(o, n.intValue() == 1); - } else if (c.equals(Byte.class) || c.equals(Byte.TYPE)) - { - f.set(o, n.byteValue()); - } else if (c.equals(Short.class) || c.equals(Short.TYPE)) - { - f.set(o, n.shortValue()); - } else if (c.equals(Long.class) || c.equals(Long.TYPE)) - { - f.set(o, n.longValue()); - } else - { - throw new RuntimeException("Field " + f.getName() + " in class " - + o.getClass().getName() + " is not numeric"); - } - } catch (IllegalArgumentException | IllegalAccessException e) - { - logger.error("Exception caught setting class field " + f.getName() - + " for object of class " + o.getClass().getName(), e); - throw new RuntimeException(); - } - } - - private void loadNetworkData() - { - if (nodes == null) - { - nodes = readNodes(); - edges = readEdges(nodes); - } - } - - @Override - protected Collection getNodes() - { - loadNetworkData(); - return nodes; - } - - @Override - protected Collection getEdges() - { - loadNetworkData(); - return edges; - } - - @Override - protected SandagBikeTraversal getTraversal(SandagBikeEdge fromEdge, SandagBikeEdge toEdge) - { - return new SandagBikeTraversal(fromEdge, toEdge); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java deleted file mode 100644 index a25d42d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeNode.java +++ /dev/null @@ -1,16 +0,0 @@ -package org.sandag.abm.active.sandag; - -import org.sandag.abm.active.SimpleNode; - -public class SandagBikeNode - extends SimpleNode -{ - public volatile float x, y; - public volatile short mgra, taz, tap; - public volatile boolean signalized, centroid; - - public SandagBikeNode(int id) - { - super(id); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java deleted file mode 100644 index d685c6b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,430 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Objects; -import java.util.Random; -import java.util.Set; -import org.sandag.abm.active.EdgeEvaluator; -import org.sandag.abm.active.Network; -import org.sandag.abm.active.NodePair; -import org.sandag.abm.active.ParallelSingleSourceDijkstra; -import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; -import org.sandag.abm.active.RepeatedSingleSourceDijkstra; -import org.sandag.abm.active.ShortestPathResultSet; -import org.sandag.abm.active.ShortestPathStrategy; -import org.sandag.abm.active.TraversalEvaluator; - -public abstract class SandagBikePathAlternativeListGenerationConfiguration - implements - PathAlternativeListGenerationConfiguration -{ - - public static final String PROPERTIES_COEF_DISTCLA0 = "active.coef.distcla0"; - public static final String PROPERTIES_COEF_DISTCLA1 = "active.coef.distcla1"; - public static final String PROPERTIES_COEF_DISTCLA2 = "active.coef.distcla2"; - public static final String PROPERTIES_COEF_DISTCLA3 = "active.coef.distcla3"; - public static final String PROPERTIES_COEF_DARTNE2 = "active.coef.dartne2"; - public static final String PROPERTIES_COEF_DWRONGWY = "active.coef.dwrongwy"; - public static final String PROPERTIES_COEF_GAIN = "active.coef.gain"; - public static final String PROPERTIES_COEF_TURN = "active.coef.turn"; - public static final String PROPERTIES_COEF_GAIN_WALK = "active.coef.gain.walk"; - public static final String PROPERTIES_COEF_DCYCTRAC = "active.coef.dcyctrac"; - public static final String PROPERTIES_COEF_DBIKBLVD = "active.coef.dbikblvd"; - public static final String PROPERTIES_COEF_SIGNALS = "active.coef.signals"; - public static final String PROPERTIES_COEF_UNLFRMA = "active.coef.unlfrma"; - public static final String PROPERTIES_COEF_UNLFRMI = "active.coef.unlfrmi"; - public static final String PROPERTIES_COEF_UNTOMA = "active.coef.untoma"; - public static final String PROPERTIES_COEF_UNTOMI = "active.coef.untomi"; - public static final String PROPERTIES_COEF_NONSCENIC = "active.coef.nonscenic"; - public static final String PROPERTIES_BIKE_MINUTES_PER_MILE = "active.bike.minutes.per.mile"; - public static final String PROPERTIES_OUTPUT = "active.output.bike"; - private static final double INACCESSIBLE_COST_COEF = 999.0; - - protected Map propertyMap; - protected PropertyParser propertyParser; - protected final String PROPERTIES_SAMPLE_MAXCOST = "active.sample.maxcost"; - protected final String PROPERTIES_SAMPLE_RANDOM_SEEDED = "active.sample.random.seeded"; - protected final String PROPERTIES_SAMPLE_DISTANCE_BREAKS = "active.sample.distance.breaks"; - protected final String PROPERTIES_SAMPLE_PATHSIZES = "active.sample.pathsizes"; - protected final String PROPERTIES_SAMPLE_COUNT_MIN = "active.sample.count.min"; - protected final String PROPERTIES_SAMPLE_COUNT_MAX = "active.sample.count.max"; - protected final String PROPERTIES_TRACE_EXCLUSIVE = "active.trace.exclusive"; - protected final String PROPERTIES_RANDOM_SCALE_COEF = "active.sample.random.scale.coef"; - protected final String PROPERTIES_RANDOM_SCALE_LINK = "active.sample.random.scale.link"; - - protected final String PROPERTIES_TRACE_OUTPUTASSIGNMENTPATHS = "active.trace.outputassignmentpaths"; - - protected double PROPERTIES_MAXDIST_ZONE; - protected String PROPERTIES_TRACE_ORIGINS; - - protected Map> nearbyZonalDistanceMap; - protected Map zonalCentroidIdMap; - protected Network network; - private final double bikeMinutesPerMile; - - public SandagBikePathAlternativeListGenerationConfiguration(Map propertyMap, - Network network) - { - this.propertyMap = propertyMap; - this.propertyParser = new PropertyParser(propertyMap); - this.nearbyZonalDistanceMap = null; - this.zonalCentroidIdMap = null; - this.network = network; - bikeMinutesPerMile = Double.parseDouble(propertyMap.get(PROPERTIES_BIKE_MINUTES_PER_MILE)); - } - - public Set getTraceOrigins() - { - return propertyMap.containsKey(PROPERTIES_TRACE_ORIGINS) ? new HashSet<>( - propertyParser.parseIntPropertyList(PROPERTIES_TRACE_ORIGINS)) - : new HashSet(); - } - - @Override - public Network getNetwork() - { - return network; - } - - public String getOutputDirectory() - { - return propertyMap.get(PROPERTIES_OUTPUT); - } - - static class SandagBikeDistanceEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.distance; - } - } - - static class SandagBikeAccessibleDistanceEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.distance + (edge.bikeCost > 998 ? 999 : 0); - } - } - - static class ZeroTraversalEvaluator - implements TraversalEvaluator - { - public double evaluate(SandagBikeTraversal traversal) - { - return 999 * (traversal.thruCentroid ? 1 : 0); - } - } - - @Override - public EdgeEvaluator getEdgeLengthEvaluator() - { - return new SandagBikeDistanceEvaluator(); - } - - @Override - public EdgeEvaluator getEdgeCostEvaluator() - { - final class SandagBikeEdgeCostEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.bikeCost; - } - } - - return new SandagBikeEdgeCostEvaluator(); - } - - @Override - public TraversalEvaluator getTraversalCostEvaluator() - { - final class SandagBikeTraversalCostEvaluator - implements TraversalEvaluator - { - public double evaluate(SandagBikeTraversal traversal) - { - return traversal.cost; - } - } - - return new SandagBikeTraversalCostEvaluator(); - } - - @Override - public double getMaxCost() - { - return Double.parseDouble(propertyMap.get(PROPERTIES_SAMPLE_MAXCOST)); - } - - @Override - public double getDefaultMinutesPerMile() - { - return bikeMinutesPerMile; - } - - @Override - public double[] getSampleDistanceBreaks() - { - return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_DISTANCE_BREAKS); - } - - @Override - public double[] getSamplePathSizes() - { - return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_PATHSIZES); - } - - @Override - public double[] getSampleMinCounts() - { - return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_COUNT_MIN); - } - - @Override - public double[] getSampleMaxCounts() - { - return propertyParser.parseDoublePropertyArray(PROPERTIES_SAMPLE_COUNT_MAX); - } - - @Override - public boolean isRandomCostSeeded() - { - return Boolean.parseBoolean(propertyMap.get(PROPERTIES_SAMPLE_RANDOM_SEEDED)); - } - - @Override - public Map> getNearbyZonalDistanceMap() - { - if (nearbyZonalDistanceMap == null) - { - nearbyZonalDistanceMap = new HashMap<>(); - ShortestPathStrategy sps = new ParallelSingleSourceDijkstra( - new RepeatedSingleSourceDijkstra( - network, new SandagBikeAccessibleDistanceEvaluator(), - new ZeroTraversalEvaluator()), - ParallelSingleSourceDijkstra.ParallelMethod.QUEUE); - if (zonalCentroidIdMap == null) - { - createZonalCentroidIdMap(); - } - Set originNodes = new HashSet<>(); - Set destinationNodes = new HashSet<>(); - Map inverseOriginZonalCentroidMap = new HashMap<>(); - Map inverseDestinationZonalCentroidMap = new HashMap<>(); - SandagBikeNode n; - Map relevantOriginZonalCentroidIdMap = getOriginZonalCentroidIdMap(); - Map destinationZonalCentroidIdMap = getDestinationZonalCentroidIdMap(); - for (int zone : relevantOriginZonalCentroidIdMap.keySet()) - { - n = network.getNode(zonalCentroidIdMap.get(zone)); - originNodes.add(n); - inverseOriginZonalCentroidMap.put(n, zone); - } - for (int zone : destinationZonalCentroidIdMap.keySet()) - { - n = network.getNode(zonalCentroidIdMap.get(zone)); - destinationNodes.add(n); - inverseDestinationZonalCentroidMap.put(n, zone); - } - System.out.println("Calculating nearby Zonal Distance Map"); - ShortestPathResultSet resultSet = sps.getShortestPaths(originNodes, - destinationNodes, PROPERTIES_MAXDIST_ZONE); - int originZone, destinationZone; - for (NodePair odPair : resultSet) - { - originZone = inverseOriginZonalCentroidMap.get(odPair.getFromNode()); - destinationZone = inverseDestinationZonalCentroidMap.get(odPair.getToNode()); - if (!nearbyZonalDistanceMap.containsKey(originZone)) - { - nearbyZonalDistanceMap.put(originZone, new HashMap()); - } - nearbyZonalDistanceMap.get(originZone).put(destinationZone, - resultSet.getShortestPathResult(odPair).getCost()); - } - } - return nearbyZonalDistanceMap; - } - - @Override - public Map getOriginZonalCentroidIdMap() - { - if (zonalCentroidIdMap == null) - { - createZonalCentroidIdMap(); - } - - if (isTraceExclusive()) - { - Map m = new HashMap<>(); - for (int o : getTraceOrigins()) - { - m.put(o, zonalCentroidIdMap.get(o)); - } - return m; - } else return zonalCentroidIdMap; - } - - public Map getOriginZonalCentroidIdMapNonExclusiveOfTrace() - { - if (zonalCentroidIdMap == null) - { - createZonalCentroidIdMap(); - } - - return zonalCentroidIdMap; - } - - @Override - public Map getDestinationZonalCentroidIdMap() - { - return getOriginZonalCentroidIdMapNonExclusiveOfTrace(); - } - - @Override - public Map getPropertyMap() - { - return propertyMap; - } - - protected abstract void createZonalCentroidIdMap(); - - public Map getInverseOriginZonalCentroidIdMap() - { - HashMap newMap = new HashMap<>(); - Map origMap = getOriginZonalCentroidIdMap(); - for (Integer o : origMap.keySet()) - { - newMap.put(origMap.get(o), o); - } - return newMap; - } - - public Map getInverseDestinationZonalCentroidIdMap() - { - HashMap newMap = new HashMap<>(); - Map origMap = getDestinationZonalCentroidIdMap(); - for (Integer d : origMap.keySet()) - { - newMap.put(origMap.get(d), d); - } - return newMap; - } - - @Override - public boolean isTraceExclusive() - { - return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_EXCLUSIVE)); - } - - private class RandomizedEdgeCostEvaluator - implements EdgeEvaluator - { - long seed; - Random random; - double cDistCla0, cDistCla1, cDistCla2, cDistCla3, cArtNe2, cWrongWay, cCycTrac, cBikeBlvd, - cGain, cNonScenic; - - public RandomizedEdgeCostEvaluator(long seed) - { - this.seed = seed; - - if (isRandomCostSeeded()) - { - random = new Random(seed); - } else - { - random = new Random(); - } - - cDistCla0 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA0)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cDistCla1 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA1)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cDistCla2 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA2)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cDistCla3 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DISTCLA3)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cArtNe2 = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DARTNE2)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cWrongWay = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DWRONGWY)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cCycTrac = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DCYCTRAC)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cBikeBlvd = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_DBIKBLVD)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cGain = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_GAIN)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - cNonScenic = Double.parseDouble(propertyMap.get(PROPERTIES_COEF_NONSCENIC)) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_COEF)) - * (2 * random.nextDouble() - 1)); - - } - - public double evaluate(SandagBikeEdge edge) - { - - if (isRandomCostSeeded()) - { - random = new Random(Objects.hash(seed, edge)); - } else - { - random = new Random(); - } - - return (edge.distance - * ((cDistCla0 * ((edge.bikeClass < 1 ? 1 : 0) + (edge.bikeClass > 3 ? 1 : 0)) - + cDistCla1 * (edge.bikeClass == 1 ? 1 : 0) + cDistCla2 - * (edge.bikeClass == 2 ? 1 : 0) * (edge.cycleTrack ? 0 : 1) + cDistCla3 - * (edge.bikeClass == 3 ? 1 : 0) * (edge.bikeBlvd ? 0 : 1) + cArtNe2 - * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) - * ((edge.functionalClass < 5 && edge.functionalClass > 0) ? 1 : 0) - + cWrongWay * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0) - + cCycTrac * (edge.cycleTrack ? 1 : 0) + cBikeBlvd - * (edge.bikeBlvd ? 1 : 0)) + cNonScenic * (1-edge.scenicIndex) - ) - + cGain * edge.gain - ) - * (1 + Double.parseDouble(propertyMap.get(PROPERTIES_RANDOM_SCALE_LINK)) - * (random.nextBoolean() ? 1 : -1)) + INACCESSIBLE_COST_COEF - * ((edge.functionalClass < 3 && edge.functionalClass > 0) ? 1 : 0); - } - } - - public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed) - { - - if (iter == 1) - { - return getEdgeCostEvaluator(); - } else - { - return new RandomizedEdgeCostEvaluator(seed); - } - - } - - public boolean isIntrazonalsNeeded() - { - return true; - } - - public boolean isAssignmentPathOutputNeeded() - { - return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_OUTPUTASSIGNMENTPATHS)); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java deleted file mode 100644 index d2d0923..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathAlternatives.java +++ /dev/null @@ -1,207 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.ArrayList; -import java.util.List; -import org.sandag.abm.active.Network; -import org.sandag.abm.active.PathAlternativeList; - -public class SandagBikePathAlternatives -{ - private final PathAlternativeList pathAlternativeList; - private List distance, distClass1, - distClass2, distClass3, distArtNoLane, distWrongWay, distCycTrack, distBikeBlvd, distScenic, gain, - turns, signals, unlfrma, unlfrmi, untoma, untomi, netCost; - private Network network; - - @SuppressWarnings("unchecked") - public SandagBikePathAlternatives( - PathAlternativeList pathAlternativeList) - { - this.pathAlternativeList = pathAlternativeList; - this.network = (Network) pathAlternativeList - .getNetwork(); - calcAndStoreAttributes(); - } - - private void calcAndStoreAttributes() - { - distance = new ArrayList<>(); - distClass1 = new ArrayList<>(); - distClass2 = new ArrayList<>(); - distClass3 = new ArrayList<>(); - distArtNoLane = new ArrayList<>(); - distWrongWay = new ArrayList<>(); - distCycTrack = new ArrayList<>(); - distBikeBlvd = new ArrayList<>(); - distScenic = new ArrayList<>(); - gain = new ArrayList<>(); - turns = new ArrayList<>(); - signals = new ArrayList<>(); - unlfrma = new ArrayList<>(); - unlfrmi = new ArrayList<>(); - untoma = new ArrayList<>(); - untomi = new ArrayList<>(); - netCost = new ArrayList<>(); - - for (int i = 0; i < getPathCount(); i++) - { - distance.add(0.0); - distClass1.add(0.0); - distClass2.add(0.0); - distClass3.add(0.0); - distArtNoLane.add(0.0); - distWrongWay.add(0.0); - distCycTrack.add(0.0); - distBikeBlvd.add(0.0); - distScenic.add(0.0); - gain.add(0.0); - turns.add(0.0); - signals.add(0.0); - unlfrma.add(0.0); - unlfrmi.add(0.0); - untoma.add(0.0); - untomi.add(0.0); - netCost.add(0.0); - SandagBikeNode parent = null, grandparent = null; - for (SandagBikeNode current : pathAlternativeList.get(i)) - { - if (parent != null) - { - SandagBikeEdge edge = network.getEdge(parent, current); - distance.set(i, distance.get(i) + edge.distance); - distClass1.set(i, distClass1.get(i) + edge.distance - * (edge.bikeClass == 1 ? 1 : 0)); - distClass2.set(i, distClass2.get(i) + edge.distance - * (edge.bikeClass == 2 ? 1 : 0)); - distClass3.set(i, distClass3.get(i) + edge.distance - * (edge.bikeClass == 3 ? 1 : 0)); - distArtNoLane.set(i, distArtNoLane.get(i) + edge.distance - * (edge.bikeClass != 2 && edge.bikeClass != 1 ? 1 : 0) - * ((edge.functionalClass < 4 && edge.functionalClass > 0) ? 1 : 0)); - distWrongWay.set(i, distWrongWay.get(i) + edge.distance - * (edge.bikeClass != 1 ? 1 : 0) * (edge.lanes == 0 ? 1 : 0)); - distCycTrack.set(i, distCycTrack.get(i) + edge.distance - * (edge.cycleTrack ? 1 : 0)); - distBikeBlvd.set(i, distBikeBlvd.get(i) + edge.distance - * (edge.bikeBlvd ? 1 : 0)); - distScenic.set(i, distScenic.get(i) + edge.distance - * edge.scenicIndex); - gain.set(i, gain.get(i) + edge.gain); - netCost.set(i, netCost.get(i) + edge.bikeCost); - if (grandparent != null) - { - SandagBikeEdge fromEdge = network.getEdge(grandparent, parent); - SandagBikeTraversal traversal = network.getTraversal(fromEdge, edge); - turns.set(i, turns.get(i) + (traversal.turnType != TurnType.NONE ? 1 : 0)); - signals.set(i, signals.get(i) - + (traversal.signalExclRightAndThruJunction ? 1 : 0)); - unlfrma.set(i, unlfrma.get(i) + (traversal.unsigLeftFromMajorArt ? 1 : 0)); - unlfrmi.set(i, unlfrmi.get(i) + (traversal.unsigLeftFromMinorArt ? 1 : 0)); - untoma.set(i, untoma.get(i) + (traversal.unsigCrossMajorArt ? 1 : 0)); - untomi.set(i, untomi.get(i) + (traversal.unsigCrossMinorArt ? 1 : 0)); - netCost.set(i, netCost.get(i) + traversal.cost); - } - } - grandparent = parent; - parent = current; - } - } - } - - public double getSizeAlt(int path) - { - return pathAlternativeList.getSizeMeasures().get(path) - / pathAlternativeList.getSizeMeasureTotal(); - } - - public double getDistanceAlt(int path) - { - return distance.get(path); - } - - public double getDistanceClass1Alt(int path) - { - return distClass1.get(path); - } - - public double getDistanceClass2Alt(int path) - { - return distClass2.get(path); - } - - public double getDistanceClass3Alt(int path) - { - return distClass3.get(path); - } - - public double getDistanceArtNoLaneAlt(int path) - { - return distArtNoLane.get(path); - } - - public double getDistanceCycleTrackAlt(int path) - { - return distCycTrack.get(path); - } - - public double getDistanceBikeBlvdAlt(int path) - { - return distBikeBlvd.get(path); - } - - public double getDistanceWrongWayAlt(int path) - { - return distWrongWay.get(path); - } - - public double getDistanceScenicAlt(int path) - { - return distScenic.get(path); - } - - public double getGainAlt(int path) - { - return gain.get(path); - } - - public double getTurnsAlt(int path) - { - return turns.get(path); - } - - public double getSignalsAlt(int path) - { - return signals.get(path); - } - - public double getUnsigLeftFromMajorArtAlt(int path) - { - return unlfrma.get(path); - } - - public double getUnsigLeftFromMinorArtAlt(int path) - { - return unlfrmi.get(path); - } - - public double getUnsigCrossMajorArtAlt(int path) - { - return untoma.get(path); - } - - public double getUnsigCrossMinorArtAlt(int path) - { - return untomi.get(path); - } - - public int getPathCount() - { - return pathAlternativeList.getCount(); - } - - public double getNetworkCostAlt(int path) - { - return netCost.get(path); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java deleted file mode 100644 index ce01a7d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceDmu.java +++ /dev/null @@ -1,300 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Map; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class SandagBikePathChoiceDmu - implements VariableTable, Serializable -{ - - private final transient Logger logger = Logger.getLogger(SandagBikePathChoiceDmu.class); - - protected final Map methodIndexMap; - private final IndexValues dmuIndex; - - private int personIsFemale; - private int isInboundTrip; - private int tourPurpose; - private SandagBikePathAlternatives paths; - private int pathCount = 0; - - public SandagBikePathChoiceDmu() - { - methodIndexMap = new HashMap<>(); - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - } - - // not needed right now - // public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, - // int destIndex, boolean debug) { - // dmuIndex.setHHIndex(hhIndex); - // dmuIndex.setZoneIndex(zoneIndex); - // dmuIndex.setOriginZone(origIndex); - // dmuIndex.setDestZone(destIndex); - // - // dmuIndex.setDebug(false); - // dmuIndex.setDebugLabel(""); - // if (debug) { - // dmuIndex.setDebug(true); - // dmuIndex.setDebugLabel("Debug Path Choice UEC"); - // } - // - // } - - public void setPersonIsFemale(boolean isFemale) - { - personIsFemale = isFemale ? 1 : 0; - } - - public void setIsInboundTrip(boolean isInboundTrip) - { - this.isInboundTrip = isInboundTrip ? 1 : 0; - } - - public void setTourPurpose(int tourPurpose) - { - this.tourPurpose = tourPurpose; - } - - public void setPathAlternatives(SandagBikePathAlternatives paths) - { - this.paths = paths; - pathCount = paths.getPathCount(); - } - - public SandagBikePathAlternatives getPathAlternatives() - { - return paths; - } - - public int getFemale() - { - return personIsFemale; - } - - public int getInbound() - { - return isInboundTrip; - } - - public int getTourPurpose() - { - return tourPurpose; - } - - public double getSizeAlt(int path) - { - return paths.getSizeAlt(path - 1); - } - - public double getDistanceAlt(int path) - { - return paths.getDistanceAlt(path - 1); - } - - public double getDistanceClass1Alt(int path) - { - return paths.getDistanceClass1Alt(path - 1); - } - - public double getDistanceClass2Alt(int path) - { - return paths.getDistanceClass2Alt(path - 1); - } - - public double getDistanceClass3Alt(int path) - { - return paths.getDistanceClass3Alt(path - 1); - } - - public double getDistanceArtNoLaneAlt(int path) - { - return paths.getDistanceArtNoLaneAlt(path - 1); - } - - public double getDistanceCycleTrackAlt(int path) - { - return paths.getDistanceCycleTrackAlt(path - 1); - } - - public double getDistanceBikeBlvdAlt(int path) - { - return paths.getDistanceBikeBlvdAlt(path - 1); - } - - public double getDistanceWrongWayAlt(int path) - { - return paths.getDistanceWrongWayAlt(path - 1); - } - - public double getDistanceScenicAlt(int path) - { - return paths.getDistanceScenicAlt(path - 1); - } - - public double getGainAlt(int path) - { - return paths.getGainAlt(path - 1); - } - - public double getTurnsAlt(int path) - { - return paths.getTurnsAlt(path - 1); - } - - public double getSignalsAlt(int path) - { - return paths.getSignalsAlt(path - 1); - } - - public double getUnsigLeftFromMajorArtAlt(int path) - { - return paths.getUnsigLeftFromMajorArtAlt(path - 1); - } - - public double getUnsigLeftFromMinorArtAlt(int path) - { - return paths.getUnsigLeftFromMinorArtAlt(path - 1); - } - - public double getUnsigCrossMajorArtAlt(int path) - { - return paths.getUnsigCrossMajorArtAlt(path - 1); - } - - public double getUnsigCrossMinorArtAlt(int path) - { - return paths.getUnsigCrossMinorArtAlt(path - 1); - } - - public double getNetworkCostAlt(int path) - { - return paths.getNetworkCostAlt(path - 1); - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - @Override - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - @Override - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - @Override - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - @Override - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - @Override - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - private void setupMethodIndexMap() - { - methodIndexMap.clear(); - - methodIndexMap.put("getSizeAlt", 0); - methodIndexMap.put("getDistanceAlt", 1); - methodIndexMap.put("getDistanceClass1Alt", 2); - methodIndexMap.put("getDistanceClass2Alt", 3); - methodIndexMap.put("getDistanceClass3Alt", 4); - methodIndexMap.put("getDistanceArtNoLaneAlt", 5); - methodIndexMap.put("getDistanceCycleTrackAlt", 6); - methodIndexMap.put("getDistanceBikeBlvdAlt", 7); - methodIndexMap.put("getDistanceWrongWayAlt", 8); - methodIndexMap.put("getGainAlt", 9); - methodIndexMap.put("getTurnsAlt", 10); - - methodIndexMap.put("getFemale", 11); - methodIndexMap.put("getInbound", 12); - methodIndexMap.put("getTourPurpose", 13); - - methodIndexMap.put("getSignalsAlt", 14); - methodIndexMap.put("getUnsigLeftFromMajorArtAlt", 15); - methodIndexMap.put("getUnsigLeftFromMinorArtAlt", 16); - methodIndexMap.put("getUnsigCrossMajorArtAlt", 17); - methodIndexMap.put("getUnsigCrossMinorArtAlt", 18); - methodIndexMap.put("getNetworkCostAlt", 19); - methodIndexMap.put("getDistanceScenicAlt", 20); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - switch (variableIndex) - { - case 0: - return getSizeAlt(arrayIndex); - case 1: - return getDistanceAlt(arrayIndex); - case 2: - return getDistanceClass1Alt(arrayIndex); - case 3: - return getDistanceClass2Alt(arrayIndex); - case 4: - return getDistanceClass3Alt(arrayIndex); - case 5: - return getDistanceArtNoLaneAlt(arrayIndex); - case 6: - return getDistanceCycleTrackAlt(arrayIndex); - case 7: - return getDistanceBikeBlvdAlt(arrayIndex); - case 8: - return getDistanceWrongWayAlt(arrayIndex); - case 9: - return getGainAlt(arrayIndex); - case 10: - return getTurnsAlt(arrayIndex); - case 11: - return getFemale(); - case 12: - return getInbound(); - case 13: - return getTourPurpose(); - case 14: - return getSignalsAlt(arrayIndex); - case 15: - return getUnsigLeftFromMajorArtAlt(arrayIndex); - case 16: - return getUnsigLeftFromMinorArtAlt(arrayIndex); - case 17: - return getUnsigCrossMajorArtAlt(arrayIndex); - case 18: - return getUnsigCrossMinorArtAlt(arrayIndex); - case 19: - return getNetworkCostAlt(arrayIndex); - case 20: - return getDistanceScenicAlt(arrayIndex); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java deleted file mode 100644 index 0e41082..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceEdgeAssignmentApplication.java +++ /dev/null @@ -1,281 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.io.IOException; -import java.io.PrintWriter; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.text.DecimalFormat; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Iterator; -import java.util.List; -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.active.AbstractPathChoiceEdgeAssignmentApplication; -import org.sandag.abm.active.Network; -import org.sandag.abm.active.PathAlternativeList; -import org.sandag.abm.ctramp.Stop; -import org.sandag.abm.ctramp.Tour; -import com.pb.common.util.ResourceUtil; - -public class SandagBikePathChoiceEdgeAssignmentApplication - extends - AbstractPathChoiceEdgeAssignmentApplication -{ - private static final Logger logger = Logger.getLogger(SandagBikePathChoiceEdgeAssignmentApplication.class); - private static final String BIKE_ASSIGNMENT_FILE_PROPERTY = "active.assignment.file.bike"; - - List stops; - final static String[] TIME_PERIOD_LABELS = {"EA", "AM", - "MD", "PM", "EV" }; - final static double[] TIME_PERIOD_BREAKS = {3, 9, 22, 29, - 99 }; - - private ThreadLocal model; - private Map storedPathData; - - private class PathData { - PathAlternativeList pal; - int tripNum; - int householdId; - int tourId; - int stopId; - int inbound; - double[] probs; - - public PathData(PathAlternativeList pal, int tripNum, - int householdId, int tourId, int stopId, int inbound, double[] probs) - { - this.pal = pal; - this.tripNum = tripNum; - this.householdId = householdId; - this.tourId = tourId; - this.stopId = stopId; - this.inbound = inbound; - this.probs = probs; - } - } - - SandagBikePathAlternativeListGenerationConfiguration configuration; - - public SandagBikePathChoiceEdgeAssignmentApplication( - SandagBikePathAlternativeListGenerationConfiguration configuration, - List stops, final Map propertyMap) - { - super(configuration); - this.stops = stops; - model = new ThreadLocal() - { - @Override - protected SandagBikePathChoiceModel initialValue() - { - return new SandagBikePathChoiceModel((HashMap) propertyMap); - } - }; - storedPathData = new ConcurrentHashMap<>(); - this.configuration = configuration; - } - - @Override - protected Map assignTrip(int tripNum, - PathAlternativeList alternativeList) - { - Stop stop = stops.get(tripNum); - Tour tour = stop.getTour(); - SandagBikePathAlternatives paths = new SandagBikePathAlternatives(alternativeList); - double[] probs = model.get().getPathProbabilities(tour.getPersonObject(), paths, - stop.isInboundStop(), tour, false); - double numPersons = 1; - if (tour.getPersonNumArray() != null && tour.getPersonNumArray().length > 1) - { - numPersons = tour.getPersonNumArray().length; - } - int periodIdx = findFirstIndexGreaterThan((double) stop.getStopPeriod(), TIME_PERIOD_BREAKS); - - Map volumes = new HashMap<>(); - for (int pathIdx = 0; pathIdx < probs.length; pathIdx++) - { - SandagBikeNode parent = null; - for (SandagBikeNode node : alternativeList.get(pathIdx)) - { - if (parent != null) - { - SandagBikeEdge edge = network.getEdge(parent, node); - double[] values; - if (volumes.containsKey(edge)) - { - values = volumes.get(edge); - } else - { - values = new double[TIME_PERIOD_BREAKS.length]; - Arrays.fill(values, 0.0); - } - values[periodIdx] += probs[pathIdx] * numPersons; - volumes.put(edge, values); - } - parent = node; - } - } - - if ( this.configuration.isAssignmentPathOutputNeeded() ) { - PathData pd = new PathData(alternativeList,tripNum,tour.getHhId(),tour.getTourId(),stop.isInboundStop() ? 1 : 0,stop.getStopId(),probs); - storedPathData.put(tripNum, pd); - } - - return volumes; - } - - @Override - protected SandagBikeNode getOriginNode(int tripId) - { - return network.getNode(configuration.getOriginZonalCentroidIdMap().get( - stops.get(tripId).getOrig())); - } - - @Override - protected SandagBikeNode getDestinationNode(int tripId) - { - return network.getNode(configuration.getDestinationZonalCentroidIdMap().get( - stops.get(tripId).getDest())); - } - - public static void main(String... args) - { - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } - logger.info("loading property file: " - + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() - .toString()); - - if (args.length < 2) - { - logger.error(String.format("no sample rate was specified as an argument.")); - return; - } - double sampleRate = Double.parseDouble(args[1]); - - // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; - @SuppressWarnings("unchecked") - // this is ok - the map will be String->String - Map propertyMap = (Map) ResourceUtil - .getResourceBundleAsHashMap(args[0]); - DecimalFormat formatter = new DecimalFormat("#.###"); - - SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); - Network network = factory - .createNetwork(); - - SandagBikePathAlternativeListGenerationConfiguration configuration = new SandagBikeMgraPathAlternativeListGenerationConfiguration( - propertyMap, network); - - BikeAssignmentTripReader reader; - if (args.length >= 3) - { - reader = new BikeAssignmentTripReader(propertyMap, Integer.parseInt(args[2])); - } else - { - reader = new BikeAssignmentTripReader(propertyMap); - } - - List stops = reader.createTripList(); - - Path outputDirectory = Paths.get(configuration.getOutputDirectory()); - Path outputFile = outputDirectory.resolve(propertyMap.get(BIKE_ASSIGNMENT_FILE_PROPERTY)); - SandagBikePathChoiceEdgeAssignmentApplication application = new SandagBikePathChoiceEdgeAssignmentApplication( - configuration, stops, propertyMap); - - List relevantTripNums = new ArrayList<>(); - for (int i = 0; i < stops.size(); i++) - relevantTripNums.add(i); - - Map volumes = application.assignTrips(relevantTripNums); - - try - { - Files.createDirectories(outputDirectory); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - StringBuilder sb = new StringBuilder("roadsegid,from,to"); - for (String segment : TIME_PERIOD_LABELS) - sb.append(",").append(segment); - writer.println(sb.toString()); - Iterator edgeIterator = network.edgeIterator(); - while (edgeIterator.hasNext()) - { - SandagBikeEdge edge = edgeIterator.next(); - sb = new StringBuilder(); - sb.append(edge.roadsegid).append(",").append(edge.getFromNode().getId()).append(",").append(edge.getToNode().getId()); - if (volumes.containsKey(edge)) - { - for (double value : volumes.get(edge)) - sb.append(",").append(formatter.format(value / sampleRate)); - } else - { - for (int i = 0; i < TIME_PERIOD_BREAKS.length; i++) - sb.append(",").append(formatter.format(0.0)); - } - writer.println(sb.toString()); - } - } catch (IOException e) - { - logger.fatal(e); - throw new RuntimeException(e); - } - - if ( configuration.isAssignmentPathOutputNeeded() ) { - - Path probFile = outputDirectory.resolve("bikeAssignmentDisaggregatePathProbabilities.csv"); - - try (PrintWriter writer = new PrintWriter(probFile.toFile())) - { - writer.println("tripid,hhid,tourid,stopid,inbound,pathid,prob"); - for (PathData pd : application.storedPathData.values()) { - for (int i=0;i -{ - private static final Logger logger = Logger.getLogger(SandagBikePathChoiceLogsumMatrixApplication.class); - - public static final String PROPERTIES_DEBUG_ORIGIN = "active.debug.origin"; - public static final String PROPERTIES_DEBUG_DESTINATION = "active.debug.destination"; - public static final String PROPERTIES_WRITE_DERIVED_BIKE_NETWORK = "active.bike.write.derived.network"; - public static final String PROPERTIES_DERIVED_NETWORK_EDGES_FILE = "active.bike.derived.network.edges"; - public static final String PROPERTIES_DERIVED_NETWORK_NODES_FILE = "active.bike.derived.network.nodes"; - public static final String PROPERTIES_DERIVED_NETWORK_TRAVERSALS_FILE = "active.bike.derived.network.traversals"; - - private static final String[] MARKET_SEGMENT_NAMES = {"logsum"}; - private static final int[] MARKET_SEGMENT_GENDER_VALUES = {1}; - private static final int[] MARKET_SEGMENT_TOUR_PURPOSE_INDICES = {1}; - private static final boolean[] MARKET_SEGMENT_INBOUND_TRIP_VALUES = {false}; - - private ThreadLocal model; - private Person[] persons; - private Tour[] tours; - private final boolean runDebug; - private final int debugOrigin; - private final int debugDestination; - - public SandagBikePathChoiceLogsumMatrixApplication( - PathAlternativeListGenerationConfiguration configuration, - final Map propertyMap) - { - super(configuration); - model = new ThreadLocal() - { - @Override - protected SandagBikePathChoiceModel initialValue() - { - return new SandagBikePathChoiceModel((HashMap) propertyMap); - } - }; - persons = new Person[MARKET_SEGMENT_NAMES.length]; - tours = new Tour[MARKET_SEGMENT_NAMES.length]; - - // for dummy person - SandagModelStructure modelStructure = new SandagModelStructure(); - for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) - { - persons[i] = new Person(null, 1, modelStructure); - persons[i].setPersGender(MARKET_SEGMENT_GENDER_VALUES[i]); - tours[i] = new Tour(persons[i], 1, MARKET_SEGMENT_TOUR_PURPOSE_INDICES[i]); - } - - debugOrigin = propertyMap.containsKey(PROPERTIES_DEBUG_ORIGIN) ? Integer - .parseInt(this.propertyMap.get(PROPERTIES_DEBUG_ORIGIN)) : -1; - debugDestination = propertyMap.containsKey(PROPERTIES_DEBUG_DESTINATION) ? Integer - .parseInt(this.propertyMap.get(PROPERTIES_DEBUG_DESTINATION)) : -1; - runDebug = (debugOrigin > 0) && (debugDestination > 0); - } - - @Override - protected double[] calculateMarketSegmentLogsums( - PathAlternativeList alternativeList) - { - SandagBikePathAlternatives alts = new SandagBikePathAlternatives(alternativeList); - double[] logsums = new double[MARKET_SEGMENT_NAMES.length + 1]; - - boolean debug = runDebug - && (alternativeList.getODPair().getFromNode().getId() == debugOrigin) - && (alternativeList.getODPair().getToNode().getId() == debugDestination); - - for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) - logsums[i] = model.get().getPathLogsums(persons[i], alts, - MARKET_SEGMENT_INBOUND_TRIP_VALUES[i], tours[i], debug); - - double[] probs = model.get().getPathProbabilities(persons[0], alts, false, tours[0], debug); - double avgDist = 0; - for (int i = 0; i < alts.getPathCount(); i++) - avgDist += probs[i] * alts.getDistanceAlt(i); - logsums[logsums.length - 1] = avgDist * configuration.getDefaultMinutesPerMile(); - - return logsums; - } - - @Override - protected List> getMarketSegmentIntrazonalCalculations() - { - List> intrazonalCalculations = new ArrayList<>(); - IntrazonalCalculation logsumIntrazonalCalculation = IntrazonalCalculations - .maxFactorIntrazonalCalculation( - IntrazonalCalculations.positiveNegativeFactorizer(0.5, 0, 2, 0), 1); - // IntrazonalCalculations.maxFactorIntrazonalCalculation(IntrazonalCalculations.simpleFactorizer(1,Math.log(2)),1); - for (int i = 0; i < MARKET_SEGMENT_NAMES.length; i++) - intrazonalCalculations.add(logsumIntrazonalCalculation); - // do distance - intrazonalCalculations.add(IntrazonalCalculations - .minFactorIntrazonalCalculation( - IntrazonalCalculations.simpleFactorizer(0.5, 0), 1)); - return intrazonalCalculations; - } - - // these methods might be moved to the network classes... - public static void writeDerivedNetworkEdges( - Network network, Path outputFile) - { - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - logger.info("Writing edges with derived attributes to " + outputFile.toString()); - StringBuilder sb = new StringBuilder(); - sb.append("fromNode").append(",").append("toNode").append(",").append("bikeClass") - .append(",").append("lanes").append(",").append("functionalClass").append(",") - .append("centroidConnector").append(",").append("autosPermitted").append(",") - .append("cycleTrack").append(",").append("bikeBlvd").append(",") - .append("distance").append(",").append("gain").append(",").append("bikeCost") - .append(",").append("walkCost"); - writer.println(sb.toString()); - Iterator it = network.edgeIterator(); - while (it.hasNext()) - { - SandagBikeEdge edge = it.next(); - sb = new StringBuilder(); - sb.append(edge.getFromNode().getId()).append(",").append(edge.getToNode().getId()) - .append(",").append(edge.bikeClass).append(",").append(edge.lanes) - .append(",").append(edge.functionalClass).append(",") - .append(edge.centroidConnector).append(",").append(edge.autosPermitted) - .append(",").append(edge.cycleTrack).append(",").append(edge.bikeBlvd) - .append(",").append(edge.distance).append(",").append(edge.gain) - .append(",").append(edge.bikeCost).append(",").append(edge.walkCost); - writer.println(sb.toString()); - } - } catch (IOException e) - { - logger.fatal(e); - throw new RuntimeException(e); - } - } - - public static void writeDerivedNetworkNodes( - Network network, Path outputFile) - { - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - logger.info("Writing nodes with derived attributes to " + outputFile.toString()); - StringBuilder sb = new StringBuilder(); - sb.append("id").append(",").append("x").append(",").append("y").append(",") - .append("mgra").append(",").append("taz").append(",").append("tap").append(",") - .append("signalized").append(",").append("centroid"); - writer.println(sb.toString()); - Iterator it = network.nodeIterator(); - while (it.hasNext()) - { - SandagBikeNode node = it.next(); - sb = new StringBuilder(); - sb.append(node.getId()).append(",").append(node.x).append(",").append(node.y) - .append(",").append(node.mgra).append(",").append(node.taz).append(",") - .append(node.tap).append(",").append(node.signalized).append(",") - .append(node.centroid); - writer.println(sb.toString()); - } - } catch (IOException e) - { - logger.fatal(e); - throw new RuntimeException(e); - } - } - - public static void writeDerivedNetworkTraversals( - Network network, Path outputFile) - { - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - logger.info("Writing traversals with derived attributes to " + outputFile.toString()); - StringBuilder sb = new StringBuilder(); - sb.append("start").append(",").append("thru").append(",").append("end").append(",") - .append("turnType").append(",").append("bikecost").append(",") - .append("thruCentroid").append(",").append("signalExclRight").append(",") - .append("unlfrma").append(",").append("unlfrmi").append(",").append("unxma") - .append(",").append("unxmi"); - writer.println(sb.toString()); - Iterator it = network.traversalIterator(); - while (it.hasNext()) - { - SandagBikeTraversal traversal = it.next(); - sb = new StringBuilder(); - sb.append(traversal.getFromEdge().getFromNode().getId()).append(",") - .append(traversal.getFromEdge().getToNode().getId()).append(",") - .append(traversal.getToEdge().getToNode().getId()).append(",") - .append(traversal.turnType.getKey()).append(",").append(traversal.cost) - .append(",").append(traversal.thruCentroid).append(",") - .append(traversal.signalExclRightAndThruJunction).append(",") - .append(traversal.unsigLeftFromMajorArt).append(",") - .append(traversal.unsigLeftFromMinorArt).append(",") - .append(traversal.unsigCrossMajorArt).append(",") - .append(traversal.unsigCrossMinorArt); - writer.println(sb.toString()); - } - } catch (IOException e) - { - logger.fatal(e); - throw new RuntimeException(e); - } - } - - public static void main(String... args) - { - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } - logger.info("loading property file: " - + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() - .toString()); - - // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; - @SuppressWarnings("unchecked") - // this is ok - the map will be String->String - Map propertyMap = (Map) ResourceUtil - .getResourceBundleAsHashMap(args[0]); - DecimalFormat formatter = new DecimalFormat("#.###"); - - SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); - Network network = factory - .createNetwork(); - - // order matters, taz first, then mgra, so use linked hash map - Map, String> configurationOutputMap = new LinkedHashMap<>(); - configurationOutputMap.put(new SandagBikeTazPathAlternativeListGenerationConfiguration( - propertyMap, network), propertyMap.get(BikeLogsum.BIKE_LOGSUM_TAZ_FILE_PROPERTY)); - configurationOutputMap.put(new SandagBikeMgraPathAlternativeListGenerationConfiguration( - propertyMap, network), propertyMap.get(BikeLogsum.BIKE_LOGSUM_MGRA_FILE_PROPERTY)); - - for (PathAlternativeListGenerationConfiguration configuration : configurationOutputMap - .keySet()) - { - Path outputDirectory = Paths.get(configuration.getOutputDirectory()); - Path outputFile = outputDirectory.resolve(configurationOutputMap.get(configuration)); - SandagBikePathChoiceLogsumMatrixApplication application = new SandagBikePathChoiceLogsumMatrixApplication( - configuration, propertyMap); - - Map origins = configuration.getInverseOriginZonalCentroidIdMap(); - Map dests = configuration.getInverseDestinationZonalCentroidIdMap(); - - try - { - Files.createDirectories(outputDirectory); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - Map, double[]> logsums = application - .calculateMarketSegmentLogsums(); - - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - StringBuilder sb = new StringBuilder("i,j"); - for (String segment : MARKET_SEGMENT_NAMES) - sb.append(",").append(segment); - sb.append(",time"); - writer.println(sb.toString()); - for (NodePair od : logsums.keySet()) - { - sb = new StringBuilder(); - sb.append(origins.get(od.getFromNode().getId())).append(",") - .append(dests.get(od.getToNode().getId())); - for (double value : logsums.get(od)) - sb.append(",").append(formatter.format(value)); - writer.println(sb.toString()); - } - } catch (IOException e) - { - logger.fatal(e); - throw new RuntimeException(e); - } - } - - if (Boolean.parseBoolean(propertyMap.get(PROPERTIES_WRITE_DERIVED_BIKE_NETWORK))) - { - Path outputDirectory = Paths.get(propertyMap - .get(SandagBikePathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT)); - writeDerivedNetworkEdges(network, - outputDirectory.resolve(propertyMap.get(PROPERTIES_DERIVED_NETWORK_EDGES_FILE))); - writeDerivedNetworkNodes(network, - outputDirectory.resolve(propertyMap.get(PROPERTIES_DERIVED_NETWORK_NODES_FILE))); - writeDerivedNetworkTraversals(network, outputDirectory.resolve(propertyMap - .get(PROPERTIES_DERIVED_NETWORK_TRAVERSALS_FILE))); - } - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java deleted file mode 100644 index 77932da..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikePathChoiceModel.java +++ /dev/null @@ -1,134 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.Arrays; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Tour; -import com.pb.common.newmodel.ChoiceModelApplication; - -// note this class is not thread safe! - use a ThreadLocal or something to -// isolate it amongst threads -public class SandagBikePathChoiceModel -{ - public static final String PATH_CHOICE_MODEL_UEC_SPREADSHEET_PROPERTY = "path.choice.uec.spreadsheet"; - public static final String PATH_CHOICE_MODEL_UEC_MODEL_SHEET_PROPERTY = "path.choice.uec.model.sheet"; - public static final String PATH_CHOICE_MODEL_UEC_DATA_SHEET_PROPERTY = "path.choice.uec.data.sheet"; - public static final String PATH_CHOICE_MODEL_MAX_PATH_COUNT_PROPERTY = "path.choice.max.path.count"; - - public static final String PATH_CHOICE_ALTS_ID_TOKEN_PROPERTY = "path.alts.id.token"; - public static final String PATH_CHOICE_ALTS_FILE_PROPERTY = "path.alts.file"; - public static final String PATH_CHOICE_ALTS_LINK_FILE_PROPERTY = "path.alts.link.file"; - - private static final Logger logger = Logger.getLogger(SandagBikePathChoiceModel.class); - - private final ThreadLocal model; - private final ThreadLocal dmu; - private final int maxPathCount; - private final ThreadLocal pathAltsAvailable; - private final ThreadLocal pathAltsSample; - - public SandagBikePathChoiceModel(final HashMap propertyMap) - { - final String uecSpreadsheet = propertyMap.get(PATH_CHOICE_MODEL_UEC_SPREADSHEET_PROPERTY); - final int modelSheet = Integer.parseInt(propertyMap - .get(PATH_CHOICE_MODEL_UEC_MODEL_SHEET_PROPERTY)); - final int dataSheet = Integer.parseInt(propertyMap - .get(PATH_CHOICE_MODEL_UEC_DATA_SHEET_PROPERTY)); - - dmu = new ThreadLocal() - { - @Override - protected SandagBikePathChoiceDmu initialValue() - { - return new SandagBikePathChoiceDmu(); - } - }; - model = new ThreadLocal() - { - @Override - protected ChoiceModelApplication initialValue() - { - return new ChoiceModelApplication(uecSpreadsheet, modelSheet, dataSheet, - propertyMap, dmu.get()); - } - }; - - maxPathCount = Integer.parseInt(propertyMap.get(PATH_CHOICE_MODEL_MAX_PATH_COUNT_PROPERTY)); - pathAltsAvailable = new ThreadLocal() - { - @Override - protected boolean[] initialValue() - { - return new boolean[maxPathCount + 1]; - } - }; - pathAltsSample = new ThreadLocal() - { - @Override - protected int[] initialValue() - { - return new int[maxPathCount + 1]; - } - }; - } - - public double[] getPathProbabilities(Person person, SandagBikePathAlternatives paths, - boolean inboundTrip, Tour tour, boolean debug) - { - applyPathChoiceModel(person, paths, inboundTrip, tour); - double[] probabilities = Arrays - .copyOf(model.get().getProbabilities(), paths.getPathCount()); - if (debug) debugPathChoiceModel(person, paths, inboundTrip, tour, probabilities); - return probabilities; - - } - - public double getPathLogsums(Person person, SandagBikePathAlternatives paths, - boolean inboundTrip, Tour tour, boolean debug) - { - applyPathChoiceModel(person, paths, inboundTrip, tour); - if (debug) - { - model.get().logUECResults(logger); - logger.info("logsum: " + model.get().getLogsum()); - } - return model.get().getLogsum(); - } - - private void applyPathChoiceModel(Person person, SandagBikePathAlternatives paths, - boolean inboundTrip, Tour tour) - { - SandagBikePathChoiceDmu dmu = this.dmu.get(); - dmu.setPersonIsFemale(person.getPersonIsFemale() == 1); - dmu.setTourPurpose(tour.getTourPrimaryPurposeIndex()); - dmu.setIsInboundTrip(inboundTrip); - dmu.setPathAlternatives(paths); - - boolean[] pathAltsAvailable = this.pathAltsAvailable.get(); - int[] pathAltsSample = this.pathAltsSample.get(); - Arrays.fill(pathAltsAvailable, false); - Arrays.fill(pathAltsSample, 0); - - for (int i = 1; i <= paths.getPathCount(); i++) - { - pathAltsAvailable[i] = true; - pathAltsSample[i] = 1; - } - - model.get().computeUtilities(dmu, dmu.getDmuIndexValues(), pathAltsAvailable, - pathAltsSample); - } - - private void debugPathChoiceModel(Person person, SandagBikePathAlternatives paths, - boolean inboundTrip, Tour tour, double[] probabilities) - { - // want: person id, tour id, inbound trip, path count, path - // probabilities - logger.info(String - .format("debug of path choice model for person id %s, tour id %s, inbound trip: %s, path alternative count: %s", - person.getPersonId(), tour.getTourId(), inboundTrip, paths.getPathCount())); - logger.info(" path probabilities: " + Arrays.toString(probabilities)); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index 0eaef70..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTazPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import org.sandag.abm.active.Network; - -public class SandagBikeTazPathAlternativeListGenerationConfiguration - extends SandagBikePathAlternativeListGenerationConfiguration -{ - - public SandagBikeTazPathAlternativeListGenerationConfiguration(Map propertyMap, - Network network) - { - super(propertyMap, network); - this.PROPERTIES_MAXDIST_ZONE = Double.parseDouble(propertyMap.get("active.maxdist.bike.taz")); - this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.taz"; - } - - protected void createZonalCentroidIdMap() - { - zonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.taz > 0) - { - zonalCentroidIdMap.put((int) n.taz, n.getId()); - } - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java deleted file mode 100644 index d29c38b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagBikeTraversal.java +++ /dev/null @@ -1,22 +0,0 @@ -package org.sandag.abm.active.sandag; - -import org.sandag.abm.active.SimpleTraversal; - -public class SandagBikeTraversal - extends SimpleTraversal -{ - public volatile TurnType turnType; - public volatile double cost; - public volatile boolean thruCentroid, signalExclRightAndThruJunction, unsigLeftFromMajorArt, - unsigLeftFromMinorArt, unsigCrossMajorArt, unsigCrossMinorArt; - - public SandagBikeTraversal(SandagBikeEdge fromEdge, SandagBikeEdge toEdge) - { - super(fromEdge, toEdge); - } - - public SandagBikeTraversal(SandagBikeEdge edge) - { - super(null, edge); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index a9c7992..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraMgraPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,62 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import org.sandag.abm.active.Network; - -public class SandagWalkMgraMgraPathAlternativeListGenerationConfiguration - extends SandagWalkPathAlternativeListGenerationConfiguration -{ - - public SandagWalkMgraMgraPathAlternativeListGenerationConfiguration( - Map propertyMap, - Network network) - { - super(propertyMap, network); - this.PROPERTIES_MAXDIST_ZONE = Math.max( - Math.max( - Double.parseDouble(propertyMap.get("active.maxdist.walk.mgra")), - Double.parseDouble(propertyMap.get("active.maxdist.micromobility.mgra"))), - Double.parseDouble(propertyMap.get("active.maxdist.microtransit.mgra"))); - this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; - } - - protected void createOriginZonalCentroidIdMap() - { - System.out.println("Creating MGRA Origin Zonal Centroid Id Map..."); - originZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.mgra > 0) - { - originZonalCentroidIdMap.put((int) n.mgra, n.getId()); - } - } - } - - protected void createDestinationZonalCentroidIdMap() - { - System.out.println("Creating MGRA Destination Zonal Centroid Id Map..."); - destinationZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.mgra > 0) - { - destinationZonalCentroidIdMap.put((int) n.mgra, n.getId()); - } - } - } - - public boolean isIntrazonalsNeeded() - { - return true; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index ee27177..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkMgraTapPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,62 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import org.sandag.abm.active.Network; - -public class SandagWalkMgraTapPathAlternativeListGenerationConfiguration - extends SandagWalkPathAlternativeListGenerationConfiguration -{ - - public SandagWalkMgraTapPathAlternativeListGenerationConfiguration( - Map propertyMap, - Network network) - { - super(propertyMap, network); - this.PROPERTIES_MAXDIST_ZONE = Math.max( - Math.max( - Double.parseDouble(propertyMap.get("active.maxdist.walk.tap")), - Double.parseDouble(propertyMap.get("active.maxdist.micromobility.tap"))), - Double.parseDouble(propertyMap.get("active.maxdist.microtransit.tap"))); - this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.mgra"; - } - - protected void createOriginZonalCentroidIdMap() - { - System.out.println("Creating MGRA Origin Zonal Centroid Id Map..."); - originZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.mgra > 0) - { - originZonalCentroidIdMap.put((int) n.mgra, n.getId()); - } - } - } - - protected void createDestinationZonalCentroidIdMap() - { - System.out.println("Creating TAP Destination Zonal Centroid Id Map..."); - destinationZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.tap > 0) - { - destinationZonalCentroidIdMap.put((int) n.tap, n.getId()); - } - } - } - - public boolean isIntrazonalsNeeded() - { - return false; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index 14fa304..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,310 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Set; -import org.sandag.abm.active.EdgeEvaluator; -import org.sandag.abm.active.Network; -import org.sandag.abm.active.NodePair; -import org.sandag.abm.active.ParallelSingleSourceDijkstra; -import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; -import org.sandag.abm.active.RepeatedSingleSourceDijkstra; -import org.sandag.abm.active.ShortestPathResultSet; -import org.sandag.abm.active.ShortestPathStrategy; -import org.sandag.abm.active.TraversalEvaluator; - -public abstract class SandagWalkPathAlternativeListGenerationConfiguration - implements - PathAlternativeListGenerationConfiguration -{ - public static final String PROPERTIES_SAMPLE_MAXCOST = "active.sample.maxcost"; - public static final String PROPERTIES_OUTPUT = "active.output.walk"; - public static final String PROPERTIES_TRACE_EXCLUSIVE = "active.trace.exclusive"; - public static final String PROPERTIES_WALK_MINUTES_PER_MILE = "active.walk.minutes.per.mile"; - - protected Map propertyMap; - protected PropertyParser propertyParser; - - protected double PROPERTIES_MAXDIST_ZONE; - protected String PROPERTIES_TRACE_ORIGINS; - - protected Map> nearbyZonalDistanceMap; - protected Map originZonalCentroidIdMap; - protected Map destinationZonalCentroidIdMap; - protected Network network; - private final double walkMinutesPerMile; - - public SandagWalkPathAlternativeListGenerationConfiguration(Map propertyMap, - Network network) - { - this.propertyMap = propertyMap; - this.propertyParser = new PropertyParser(propertyMap); - this.nearbyZonalDistanceMap = null; - this.originZonalCentroidIdMap = null; - this.destinationZonalCentroidIdMap = null; - this.network = network; - walkMinutesPerMile = Double.parseDouble(propertyMap.get(PROPERTIES_WALK_MINUTES_PER_MILE)); - } - - public Set getTraceOrigins() - { - return propertyMap.containsKey(PROPERTIES_TRACE_ORIGINS) ? new HashSet<>( - propertyParser.parseIntPropertyList(PROPERTIES_TRACE_ORIGINS)) - : new HashSet(); - } - - @Override - public Network getNetwork() - { - return network; - } - - public String getOutputDirectory() - { - return propertyMap.get(PROPERTIES_OUTPUT); - } - - static class SandagBikeDistanceEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.distance; - } - } - - static class SandagWalkAccessibleDistanceEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.distance + (edge.walkCost > 998 ? 999 : 0); - } - } - - static class ZeroTraversalEvaluator - implements TraversalEvaluator - { - public double evaluate(SandagBikeTraversal traversal) - { - return 999 * (traversal.thruCentroid ? 1 : 0); - } - } - - @Override - public EdgeEvaluator getEdgeLengthEvaluator() - { - return new SandagBikeDistanceEvaluator(); - } - - @Override - public EdgeEvaluator getEdgeCostEvaluator() - { - final class SandagWalkEdgeCostEvaluator - implements EdgeEvaluator - { - public double evaluate(SandagBikeEdge edge) - { - return edge.walkCost; - } - } - - return new SandagWalkEdgeCostEvaluator(); - } - - @Override - public TraversalEvaluator getTraversalCostEvaluator() - { - return new ZeroTraversalEvaluator(); - } - - @Override - public double getMaxCost() - { - return Double.parseDouble(propertyMap.get(PROPERTIES_SAMPLE_MAXCOST)); - } - - @Override - public double getDefaultMinutesPerMile() - { - return walkMinutesPerMile; - } - - @Override - public double[] getSampleDistanceBreaks() - { - return new double[] {99.0}; - } - - @Override - public double[] getSamplePathSizes() - { - return new double[] {1.0}; - } - - @Override - public double[] getSampleMinCounts() - { - return new double[] {1.0}; - } - - @Override - public double[] getSampleMaxCounts() - { - return new double[] {1.0}; - } - - @Override - public boolean isRandomCostSeeded() - { - return false; - } - - @Override - public Map> getNearbyZonalDistanceMap() - { - if (nearbyZonalDistanceMap == null) - { - nearbyZonalDistanceMap = new HashMap<>(); - ShortestPathStrategy sps = new ParallelSingleSourceDijkstra( - new RepeatedSingleSourceDijkstra( - network, new SandagWalkAccessibleDistanceEvaluator(), - new ZeroTraversalEvaluator()), - ParallelSingleSourceDijkstra.ParallelMethod.QUEUE); - if (originZonalCentroidIdMap == null) - { - createOriginZonalCentroidIdMap(); - } - if (destinationZonalCentroidIdMap == null) - { - createDestinationZonalCentroidIdMap(); - } - Set originNodes = new HashSet<>(); - Set destinationNodes = new HashSet<>(); - Map inverseOriginZonalCentroidIdMap = new HashMap<>(); - Map inverseDestinationZonalCentroidIdMap = new HashMap<>(); - SandagBikeNode n; - Map relevantOriginZonalCentroidIdMap = getOriginZonalCentroidIdMap(); - for (int zone : relevantOriginZonalCentroidIdMap.keySet()) - { - n = network.getNode(originZonalCentroidIdMap.get(zone)); - originNodes.add(n); - inverseOriginZonalCentroidIdMap.put(n, zone); - } - for (int zone : destinationZonalCentroidIdMap.keySet()) - { - n = network.getNode(destinationZonalCentroidIdMap.get(zone)); - destinationNodes.add(n); - inverseDestinationZonalCentroidIdMap.put(n, zone); - } - System.out.println("Calculating nearby Zonal Distance Map"); - ShortestPathResultSet resultSet = sps.getShortestPaths(originNodes, - destinationNodes, PROPERTIES_MAXDIST_ZONE); - int originZone, destinationZone; - for (NodePair odPair : resultSet) - { - originZone = inverseOriginZonalCentroidIdMap.get(odPair.getFromNode()); - destinationZone = inverseDestinationZonalCentroidIdMap.get(odPair.getToNode()); - if (!nearbyZonalDistanceMap.containsKey(originZone)) - { - nearbyZonalDistanceMap.put(originZone, new HashMap()); - } - nearbyZonalDistanceMap.get(originZone).put(destinationZone, - resultSet.getShortestPathResult(odPair).getCost()); - } - } - return nearbyZonalDistanceMap; - } - - @Override - public Map getOriginZonalCentroidIdMap() - { - if (originZonalCentroidIdMap == null) - { - createOriginZonalCentroidIdMap(); - } - - if (isTraceExclusive()) - { - Map m = new HashMap<>(); - for (int o : getTraceOrigins()) - { - m.put(o, originZonalCentroidIdMap.get(o)); - } - return m; - } else return originZonalCentroidIdMap; - } - - public Map getOriginZonalCentroidIdMapNonExclusiveOfTrace() - { - if (originZonalCentroidIdMap == null) - { - createOriginZonalCentroidIdMap(); - } - - return originZonalCentroidIdMap; - } - - @Override - public Map getDestinationZonalCentroidIdMap() - { - if (destinationZonalCentroidIdMap == null) - { - createDestinationZonalCentroidIdMap(); - } - - return destinationZonalCentroidIdMap; - } - - @Override - public Map getPropertyMap() - { - return propertyMap; - } - - protected abstract void createOriginZonalCentroidIdMap(); - - protected abstract void createDestinationZonalCentroidIdMap(); - - public Map getInverseOriginZonalCentroidIdMap() - { - HashMap newMap = new HashMap<>(); - Map origMap = getOriginZonalCentroidIdMap(); - for (Integer o : origMap.keySet()) - { - newMap.put(origMap.get(o), o); - } - return newMap; - } - - public Map getInverseDestinationZonalCentroidIdMap() - { - HashMap newMap = new HashMap<>(); - Map origMap = getDestinationZonalCentroidIdMap(); - for (Integer o : origMap.keySet()) - { - newMap.put(origMap.get(o), o); - } - return newMap; - } - - @Override - public boolean isTraceExclusive() - { - return Boolean.parseBoolean(propertyMap.get(PROPERTIES_TRACE_EXCLUSIVE)); - } - - public EdgeEvaluator getRandomizedEdgeCostEvaluator(int iter, long seed) - { - return getEdgeCostEvaluator(); - } - - @Override - public abstract boolean isIntrazonalsNeeded(); - - public boolean isAssignmentPathOutputNeeded() - { - return false; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java deleted file mode 100644 index 442c736..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkPathChoiceLogsumMatrixApplication.java +++ /dev/null @@ -1,232 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.io.IOException; -import java.io.PrintWriter; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.text.DecimalFormat; -import java.util.Arrays; -import java.util.List; -import java.util.Map; -import java.util.TreeSet; -import org.apache.log4j.Logger; -import org.sandag.abm.active.AbstractPathChoiceLogsumMatrixApplication; -import org.sandag.abm.active.IntrazonalCalculation; -import org.sandag.abm.active.IntrazonalCalculations; -import org.sandag.abm.active.Network; -import org.sandag.abm.active.NodePair; -import org.sandag.abm.active.PathAlternativeList; -import org.sandag.abm.active.PathAlternativeListGenerationConfiguration; -import com.pb.common.util.ResourceUtil; - -public class SandagWalkPathChoiceLogsumMatrixApplication - extends - AbstractPathChoiceLogsumMatrixApplication -{ - private static final Logger logger = Logger.getLogger(SandagWalkPathChoiceLogsumMatrixApplication.class); - - public static final String WALK_LOGSUM_SKIM_MGRA_MGRA_FILE_PROPERTY = "active.logsum.matrix.file.walk.mgra"; - public static final String WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY = "active.logsum.matrix.file.walk.mgratap"; - - private final PathAlternativeListGenerationConfiguration configuration; - - public SandagWalkPathChoiceLogsumMatrixApplication( - PathAlternativeListGenerationConfiguration configuration) - { - super(configuration); - this.configuration = configuration; - } - - @Override - protected double[] calculateMarketSegmentLogsums( - PathAlternativeList alternativeList) - { - if (alternativeList.getCount() > 1) - { - throw new UnsupportedOperationException( - "Walk logsums cannot be calculated for alternative lists containing multiple paths"); - } - - double utility = 0; - double distance = 0; - double gain = 0; - SandagBikeNode parent = null; - for (SandagBikeNode n : alternativeList.get(0)) - { - if (parent != null) - { - utility += configuration.getNetwork().getEdge(parent, n).walkCost; - distance += configuration.getNetwork().getEdge(parent, n).distance; - gain += configuration.getNetwork().getEdge(parent,n).gain; - } - parent = n; - } - - return new double[] {utility, distance * configuration.getDefaultMinutesPerMile(),gain}; - } - - @Override - protected List> getMarketSegmentIntrazonalCalculations() - { - IntrazonalCalculation intrazonalCalculation = IntrazonalCalculations - .minFactorIntrazonalCalculation( - IntrazonalCalculations.simpleFactorizer(0.5, 0), 1); - // do time then distance then gain - return Arrays.asList(intrazonalCalculation, intrazonalCalculation, intrazonalCalculation); - } - - public static void main(String... args) - { - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } - logger.info("loading property file: " - + ClassLoader.getSystemClassLoader().getResource(args[0] + ".properties").getFile() - .toString()); - - logger.info("Building walk skims"); - // String RESOURCE_BUNDLE_NAME = "sandag_abm_active_test"; - @SuppressWarnings("unchecked") - // this is ok - the map will be String->String - Map propertyMap = (Map) ResourceUtil - .getResourceBundleAsHashMap(args[0]); - - SandagBikeNetworkFactory factory = new SandagBikeNetworkFactory(propertyMap); - Network network = factory - .createNetwork(); - - DecimalFormat formatter = new DecimalFormat("#.###"); - - logger.info("Generating mgra->mgra walk skims"); - // mgra->mgra - PathAlternativeListGenerationConfiguration configuration = new SandagWalkMgraMgraPathAlternativeListGenerationConfiguration( - propertyMap, network); - SandagWalkPathChoiceLogsumMatrixApplication application = new SandagWalkPathChoiceLogsumMatrixApplication( - configuration); - Map, double[]> logsums = application - .calculateMarketSegmentLogsums(); - - Path outputDirectory = Paths.get(configuration.getOutputDirectory()); - Path outputFile = outputDirectory.resolve(propertyMap - .get(WALK_LOGSUM_SKIM_MGRA_MGRA_FILE_PROPERTY)); - - try - { - Files.createDirectories(outputDirectory); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - Map originCentroids = configuration.getInverseOriginZonalCentroidIdMap(); - Map destinationCentroids = configuration - .getInverseDestinationZonalCentroidIdMap(); - - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - writer.println("i,j,percieved,actual,gain"); - StringBuilder sb; - for (NodePair od : new TreeSet<>(logsums.keySet())) - { // sort them so the output "looks nice" - sb = new StringBuilder(); - sb.append(originCentroids.get(od.getFromNode().getId())).append(","); - sb.append(destinationCentroids.get(od.getToNode().getId())).append(","); - double[] values = logsums.get(od); - sb.append(formatter.format(values[0])).append(","); // percieved - // time - sb.append(formatter.format(values[1])).append(","); // actual time - sb.append(formatter.format(values[2])); //gain - writer.println(sb.toString()); - } - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("Generating mgra->tap walk skims"); - // mgra->tap - configuration = new SandagWalkMgraTapPathAlternativeListGenerationConfiguration( - propertyMap, network); - application = new SandagWalkPathChoiceLogsumMatrixApplication(configuration); - Map, double[]> mgraTapLogsums = application - .calculateMarketSegmentLogsums(); - - // for later - get from the first configuration - outputDirectory = Paths.get(configuration.getOutputDirectory()); - outputFile = outputDirectory.resolve(propertyMap - .get(WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY)); - originCentroids = configuration.getInverseOriginZonalCentroidIdMap(); - destinationCentroids = configuration.getInverseDestinationZonalCentroidIdMap(); - - // tap->mgra - configuration = new SandagWalkTapMgraPathAlternativeListGenerationConfiguration( - propertyMap, network); - application = new SandagWalkPathChoiceLogsumMatrixApplication(configuration); - Map, double[]> tapMgraLogsums = application - .calculateMarketSegmentLogsums(); - - // resolve if not a pair - int initialSize = mgraTapLogsums.size() + tapMgraLogsums.size(); - - for (NodePair mgraTapPair : mgraTapLogsums.keySet()) - { - NodePair tapMgraPair = new NodePair( - mgraTapPair.getToNode(), mgraTapPair.getFromNode()); - if (!tapMgraLogsums.containsKey(tapMgraPair)) - tapMgraLogsums.put(tapMgraPair, mgraTapLogsums.get(mgraTapPair)); - } - - for (NodePair tapMgraPair : tapMgraLogsums.keySet()) - { - NodePair mgraTapPair = new NodePair( - tapMgraPair.getToNode(), tapMgraPair.getFromNode()); - if (!mgraTapLogsums.containsKey(mgraTapPair)) - mgraTapLogsums.put(mgraTapPair, tapMgraLogsums.get(tapMgraPair)); - } - int asymmPairCount = initialSize - (mgraTapLogsums.size() + tapMgraLogsums.size()); - if (asymmPairCount > 0) - logger.info("Boarding or alighting times defaulted to transpose for " + asymmPairCount - + " mgra tap pairs with missing asymmetrical information"); - - try - { - Files.createDirectories(outputDirectory); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - try (PrintWriter writer = new PrintWriter(outputFile.toFile())) - { - writer.println("mgra,tap,boardingPerceived,boardingActual,alightingPerceived,alightingActual,boardingGain,alightingGain"); - StringBuilder sb; - for (NodePair od : new TreeSet<>(mgraTapLogsums.keySet())) - { // sort them so the output "looks nice" - sb = new StringBuilder(); - sb.append(originCentroids.get(od.getFromNode().getId())).append(","); - sb.append(destinationCentroids.get(od.getToNode().getId())).append(","); - double[] boardingValues = mgraTapLogsums.get(od); - sb.append(formatter.format(boardingValues[0])).append(","); // boarding - // percieved - sb.append(formatter.format(boardingValues[1])).append(","); // boarding - // actual - double[] alightingValues = tapMgraLogsums.get(new NodePair<>(od.getToNode(), od - .getFromNode())); - sb.append(formatter.format(alightingValues[0])).append(","); // alighting - // percieved - sb.append(formatter.format(alightingValues[1])).append(","); // alighting - // actual - sb.append(formatter.format(boardingValues[2])).append(","); // boarding gain - sb.append(formatter.format(alightingValues[2])); // alighting gain - writer.println(sb.toString()); - } - } catch (IOException e) - { - throw new RuntimeException(e); - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java deleted file mode 100644 index ead9201..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/SandagWalkTapMgraPathAlternativeListGenerationConfiguration.java +++ /dev/null @@ -1,62 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import org.sandag.abm.active.Network; - -public class SandagWalkTapMgraPathAlternativeListGenerationConfiguration - extends SandagWalkPathAlternativeListGenerationConfiguration -{ - - public SandagWalkTapMgraPathAlternativeListGenerationConfiguration( - Map propertyMap, - Network network) - { - super(propertyMap, network); - this.PROPERTIES_MAXDIST_ZONE = Math.max( - Math.max( - Double.parseDouble(propertyMap.get("active.maxdist.walk.tap")), - Double.parseDouble(propertyMap.get("active.maxdist.micromobility.tap"))), - Double.parseDouble(propertyMap.get("active.maxdist.microtransit.tap"))); - this.PROPERTIES_TRACE_ORIGINS = "active.trace.origins.tap"; - } - - protected void createOriginZonalCentroidIdMap() - { - System.out.println("Creating TAP Zonal Centroid Id Map..."); - originZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.tap > 0) - { - originZonalCentroidIdMap.put((int) n.tap, n.getId()); - } - } - } - - protected void createDestinationZonalCentroidIdMap() - { - System.out.println("Creating MGRA Zonal Centroid Id Map..."); - destinationZonalCentroidIdMap = new HashMap(); - Iterator nodeIterator = network.nodeIterator(); - SandagBikeNode n; - while (nodeIterator.hasNext()) - { - n = nodeIterator.next(); - if (n.mgra > 0) - { - destinationZonalCentroidIdMap.put((int) n.mgra, n.getId()); - } - } - } - - public boolean isIntrazonalsNeeded() - { - return false; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java b/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java deleted file mode 100644 index 4a63681..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/active/sandag/TurnType.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.abm.active.sandag; - -import java.util.HashMap; -import java.util.Map; - -public enum TurnType -{ - NONE(0), LEFT(1), RIGHT(2), REVERSAL(3); - - private int key; - private static Map map = new HashMap(); - - static - { - for (TurnType t : TurnType.values()) - { - map.put(t.key, t); - } - } - - private TurnType(final int key) - { - this.key = key; - } - - public static TurnType valueOf(int key) - { - return map.get(key); - } - - public int getKey() - { - return key; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java deleted file mode 100644 index ed1b4f1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDestChoiceModel.java +++ /dev/null @@ -1,581 +0,0 @@ -package org.sandag.abm.airport; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -public class AirportDestChoiceModel -{ - - private double[][] sizeTerms; // by - // segment, - // tazNumber - private double[][][] mgraProbabilities; // by - // segment, - // tazNumber, - // mgra - // index - // (sequential, - // 0-based) - private double[][] tazProbabilities; // by - // segment, - // origin - // taz - // 0-based - // (note - // since - // airport - // model, - // only - // 1 - // taz - // dimension - // is - // needed) - private int[] zipCodes; // by - // taz - - private TableDataSet alternativeData; // the - // alternatives, - // with - // a - // dest - // field - // indicating - // tazNumber - - private transient Logger logger = Logger.getLogger("airportModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication[] destModel; - private UtilityExpressionCalculator sizeTermUEC; - private Tracer tracer; - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - private boolean seek; - private HashMap rbMap; - private TableDataSet externalDataSet; - private int externalAirportColumn; - private int tazColumn; - private int mazOutColumn; - private int mazInbColumn; - - - private int airportMgra; - private int airportTaz; - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of airport model DMUs - */ - public AirportDestChoiceModel(HashMap rbMap, AirportDmuFactoryIf dmuFactory, String airportCode) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String airportDistUecFileName = Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".dc.uec.file"); - airportDistUecFileName = uecFileDirectory + airportDistUecFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".dc.data.page")); - int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".dc.size.page")); - - // read the model pages from the property file, create one choice model - // for each - destModel = new ChoiceModelApplication[AirportModelStructure.INTERNAL_PURPOSES]; - for (int i = 0; i < AirportModelStructure.INTERNAL_PURPOSES; ++i) - { - - // get page from property file - String purposeName = "airport."+airportCode+".dc.segment" + (i + 1) + ".page"; - String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); - purposeString.replaceAll(" ", ""); - int destModelPage = Integer.parseInt(purposeString); - - // initiate a DMU for each segment - AirportModelDMU dcDmu = dmuFactory.getAirportModelDMU(); - - // create a ChoiceModelApplication object for the filename, model - // page and data page. - destModel[i] = new ChoiceModelApplication(airportDistUecFileName, destModelPage, - dataPage, rbMap, (VariableTable) dcDmu); - } - - // get the alternative data from the first segment - UtilityExpressionCalculator uec = destModel[0].getUEC(); - alternativeData = uec.getAlternativeData(); - - // create a UEC to solve size terms for each MGRA - sizeTermUEC = new UtilityExpressionCalculator(new File(airportDistUecFileName), sizePage, - dataPage, rbMap, dmuFactory.getAirportModelDMU()); - - // set up the tracer object - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - airportMgra = Util.getIntegerValueFromPropertyMap(rbMap, "airport."+airportCode+".airportMgra"); - airportTaz = mgraManager.getTaz(airportMgra); - - // calculate the zip code array - calculateZipCodes(); - - //read the external station dataset into memory - String externalStationFile = Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".externalStationFile"); - readExternalPercentages(externalStationFile,airportCode); - } - - /** - * Iterate through the segments, calculating the mgra probabilities for each - */ - public void calculateMgraProbabilities(AirportDmuFactoryIf dmuFactory) - { - - logger.info("Calculating Airport Model Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int alternatives = sizeTermUEC.getNumberOfAlternatives(); - - double[][] mgraSizeTerms = new double[alternatives][maxMgra + 1]; - IndexValues iv = new IndexValues(); - AirportModelDMU aDmu = dmuFactory.getAirportModelDMU(); - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = sizeTermUEC.solve(iv, aDmu, null); - - // store the size terms - for (int segment = 0; segment < alternatives; ++segment) - { - mgraSizeTerms[segment][mgra] = utilities[segment]; - } - - // log - if (tracer.isTraceOn() && tracer.isTraceZone(taz)) - { - - logger.info("Size Term calculations for mgra " + mgra); - sizeTermUEC.logResultsArray(logger, 0, mgra); - - } - } - - // now iterate through tazs, calculate probabilities - int[] tazs = tazManager.getTazs(); - int maxTaz = tazManager.getMaxTaz(); - - // initialize arrays - mgraProbabilities = new double[alternatives][maxTaz + 1][]; - sizeTerms = new double[alternatives][maxTaz + 1]; - - // calculate arrays - for (int segment = 0; segment < alternatives; ++segment) - { - for (int taz = 0; taz < tazs.length; ++taz) - { - int tazNumber = tazs[taz]; - int[] mgraArray = tazManager.getMgraArray(tazNumber); - - // initialize the vector of mgras for this purpose-taz - mgraProbabilities[segment][tazNumber] = new double[mgraArray.length]; - - // first calculate the sum of size for all the mgras in the taz - double sum = 0; - for (int mgra = 0; mgra < mgraArray.length; ++mgra) - { - - int mgraNumber = mgraArray[mgra]; - - sum += mgraSizeTerms[segment][mgraNumber]; - } - // store the logsum in the size term array by taz - if (sum > 0.0) sizeTerms[segment][tazNumber] = Math.log(sum + 1.0); - - // now calculate the cumulative probability distribution - double lastProb = 0.0; - for (int mgra = 0; mgra < mgraArray.length; ++mgra) - { - - int mgraNumber = mgraArray[mgra]; - if (sum > 0.0) - mgraProbabilities[segment][tazNumber][mgra] = lastProb - + mgraSizeTerms[segment][mgraNumber] / sum; - lastProb = mgraProbabilities[segment][tazNumber][mgra]; - } - if (sum > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) - logger.info("Error: segment " + segment + " taz " + tazNumber - + " cum prob adds up to " + lastProb); - } - - } - logger.info("Finished Calculating Airport Model Size Terms"); - } - - /** - * Calculate the zip codes at a taz level from the mgra data file. This - * requires the mgra data to be specified as mgra.socec.file in the - * properties file. The mgra file must have four fields: zone, taz, pop, and - * zip The taz zip is coded based upon the highest population mgra within - * the taz. - * - * @return a zip code array dimensioned by taz numbers - */ - public void calculateZipCodes() - { - - logger.info("Calculating Airport Model TAZ Zip Code Array"); - - zipCodes = new int[tazManager.maxTaz + 1]; - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String fileName = directory + Util.getStringValueFromPropertyMap(rbMap, "mgra.socec.file"); - - logger.info("Begin reading the data in file " + fileName); - TableDataSet mgraTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - mgraTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - // iterate through TAZs and store zip codes in zipCodes array - int[] tazs = tazManager.getTazs(); - for (int i = 0; i < tazs.length; ++i) - { - - int tazNumber = tazs[i]; - int maxPop = 0; - int zip = 0; - for (int row = 1; row <= mgraTable.getRowCount(); ++row) - { - - if (mgraTable.getValueAt(row, "taz") == tazNumber) - { - int pop = (int) mgraTable.getValueAt(row, "pop"); - if (pop > maxPop) - { - maxPop = pop; - zip = (int) mgraTable.getValueAt(row, "ZIP09"); - } - } - } - // if iterate through mgra data, and no mgras with pop found, then - // choose zip of first mgra in taz - if (zip == 0) - { - for (int row = 1; row <= mgraTable.getRowCount(); ++row) - { - - if (mgraTable.getValueAt(row, "taz") == tazNumber) - { - zip = (int) mgraTable.getValueAt(row, "ZIP09"); - break; - } - } - } - // store zip in array - zipCodes[tazNumber] = zip; - } - logger.info("Finished Calculating Airport Model TAZ Zip Code Array"); - } - - /** - * Calculate taz probabilities. This method initializes and calculates the - * tazProbabilities array. - */ - public void calculateTazProbabilities(AirportDmuFactoryIf dmuFactory) - { - - if (sizeTerms == null) - { - logger.error("Error: attemping to execute airportmodel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); - throw new RuntimeException(); - } - - logger.info("Calculating Airport Model TAZ Probabilities Arrays"); - - // initialize taz probabilities array - int segments = sizeTerms.length; - int maxTaz = tazManager.getMaxTaz(); - tazProbabilities = new double[segments][maxTaz + 1]; - - // Note: this is an aggregate model to calculate utilities, but we need - // income so we need a party object - AirportParty airportParty = new AirportParty(1001); - - AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); - dmu.setZips(zipCodes); - dmu.setSizeTerms(sizeTerms); - dmu.setAirportParty(airportParty); - - int airportTaz = mgraManager.getTaz(airportMgra); - - // segments are combinations of 4 purposes and 8 income groups, which - // apply only to resident purposes - for (int segment = 0; segment < segments; ++segment) - { - - int purpose = AirportModelStructure.getPurposeFromDCSizeSegment(segment); - int income = AirportModelStructure.getIncomeFromDCSizeSegment(segment); - - airportParty.setPurpose((byte) purpose); - airportParty.setIncome((byte) income); - - // set airport taz as origin. Destination tazs controlled by - // alternative file. - IndexValues dmuIndex = dmu.getDmuIndex(); - dmuIndex.setOriginZone(airportTaz); - - // Calculate utilities & probabilities - destModel[purpose].computeUtilities(dmu, dmuIndex); - - // Store probabilities (by segment) - tazProbabilities[segment] = Arrays.copyOf( - destModel[purpose].getCumulativeProbabilities(), - destModel[purpose].getCumulativeProbabilities().length); - } - logger.info("Finished Calculating Airport Model TAZ Probabilities Arrays"); - } - - /** - * Choose an MGRA - * - * @param purpose - * Purpose - * @param income - * Income - * @param randomNumber - * Random number - * @return The chosen MGRA number - */ - public int chooseMGRA(int purpose, int income, double randomNumber) - { - - // first find a TAZ - int segment = AirportModelStructure.getDCSizeSegment(purpose, income); - int alt = 0; - double[] tazCumProb = tazProbabilities[segment]; - double altProb = 0; - double cumProb = 0; - for (int i = 0; i < tazCumProb.length; ++i) - { - if (tazCumProb[i] > randomNumber) - { - alt = i; - if (i != 0) - { - cumProb = tazCumProb[i - 1]; - altProb = tazCumProb[i] - tazCumProb[i - 1]; - } else - { - altProb = tazCumProb[i]; - } - break; - } - } - - // get the taz number of the alternative, and an array of mgras in that - // taz - int tazNumber = (int) alternativeData.getValueAt(alt + 1, "dest"); - int[] mgraArray = tazManager.getMgraArray(tazNumber); - - // now find an MGRA in the taz corresponding to the random number drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives probability - int mgraNumber = 0; - double[] mgraCumProb = mgraProbabilities[segment][tazNumber]; - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > randomNumber) - { - mgraNumber = mgraArray[i]; - } - } - // return the chosen MGRA number - return mgraNumber; - } - - /** - * Iterate through an array of AirportParty objects, choosing origin MGRAs - * for each and setting the result back in the objects. - * - * @param airportParties - * An array of AirportParty objects - */ - public void chooseOrigins(AirportParty[] airportParties) - { - - // iterate through the array, choosing mgras and setting them - for (AirportParty party : airportParties) - { - - int income = party.getIncome(); - int purpose = party.getPurpose(); - double random = party.getRandom(); - int mgra = -99; - if (purpose < AirportModelStructure.INTERNAL_PURPOSES){ - mgra = chooseMGRA(purpose, income, random); - - // if this is a departing travel party, the origin is the chosen - // mgra, and the destination is the airport terminal - if (party.getDirection() == AirportModelStructure.DEPARTURE) - { - party.setOriginMGRA(mgra); - party.setOriginTAZ(mgraManager.getTaz(mgra)); - party.setDestinationMGRA(airportMgra); - party.setDestinationTAZ(airportTaz); - } else - { - party.setOriginMGRA(airportMgra); - party.setOriginTAZ(airportTaz); - party.setDestinationMGRA(mgra); - party.setDestinationTAZ(mgraManager.getTaz(mgra)); - } - - }else{ - chooseExternalStation(party, random); - } - } - } - /** - * Read a csv file with probabilities by external station. Required fields in the file: - * - * taz The TAZ number of the external station - * mgraOut The MGRA number for trips leaving the region (closest MGRA to outbound external TAZ) - * mgraRet The MGRA number for trips returning to the region (closest MGRA to inbound external TAZ) - * AIRPORTNAME.pct The share of trips entering\exiting at the external station for the airport - * - * - * @param fileName The name of the external station file - */ - private void readExternalPercentages(String fileName, String airportCode){ - - logger.info("Begin reading the data in file " + fileName); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - externalDataSet = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - externalAirportColumn = externalDataSet.getColumnPosition(airportCode+".pct"); - tazColumn = externalDataSet.getColumnPosition("taz"); - mazOutColumn = externalDataSet.getColumnPosition("mgraOut"); - mazInbColumn = externalDataSet.getColumnPosition("mgraRet"); - - if(externalAirportColumn<=0|| tazColumn<=0 || mazOutColumn<=0 || mazInbColumn<=0){ - logger.error("Check fields in external station file "+fileName); - logger.error("File should have fields taz, mgraOut, mgraReg and "+airportCode+".pct"); - throw new RuntimeException(); - } - - logger.info("End reading the data in file " + fileName); - - } - - /** - * Choose an external TAZ and associated MAZ for the tour if it is external. The - * probabilities are in the external station file. - * - * @param airportParty The airport party - * @param randomNumber A uniform random number - */ - private void chooseExternalStation(AirportParty airportParty, double randomNumber){ - - double cumProb=0; - int taz = -1; - int maz = -1; - for(int row = 1; row <= externalDataSet.getRowCount();++row){ - - cumProb += externalDataSet.getValueAt(row,externalAirportColumn); - if(cumProb > randomNumber){ - taz = (int) externalDataSet.getValueAt(row, tazColumn); - - if(airportParty.getDirection() == AirportModelStructure.DEPARTURE){ - maz = (int) externalDataSet.getValueAt(row,mazInbColumn); - airportParty.setOriginTAZ(taz); - airportParty.setOriginMGRA(maz); - airportParty.setDestinationMGRA(airportMgra); - airportParty.setDestinationTAZ(airportTaz); - }else{ - maz = (int) externalDataSet.getValueAt(row,mazOutColumn); - airportParty.setDestinationTAZ(taz); - airportParty.setDestinationMGRA(maz); - airportParty.setOriginMGRA(airportMgra); - airportParty.setOriginTAZ(airportTaz); - - } - - } - - } - if(taz==-1||maz==-1){ - logger.fatal("Error: could not find external destination for airport tour"); - logger.fatal("Make sure probabilities add up to 1.0 in external station file"); - throw new RuntimeException(); - } - - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java deleted file mode 100644 index 9a79e69..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactory.java +++ /dev/null @@ -1,35 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.airport; - -import java.io.Serializable; - -/** - * ArcCtrampDmuFactory is a class that creates Airport Model DMU objects - * - * @author Joel Freedman - */ -public class AirportDmuFactory - implements AirportDmuFactoryIf, Serializable -{ - - //private AirportModelStructure airportModelStructure; - - public AirportDmuFactory()//AirportModelStructure modelStructure) - { - //this.airportModelStructure = modelStructure; - } - - public AirportModelDMU getAirportModelDMU() - { - return new AirportModelDMU(null); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java deleted file mode 100644 index 8f233cb..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportDmuFactoryIf.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.sandag.abm.airport; - -/** - * A DMU factory interface - */ -public interface AirportDmuFactoryIf -{ - - AirportModelDMU getAirportModelDMU(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java deleted file mode 100644 index b988d14..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModeChoiceModel.java +++ /dev/null @@ -1,378 +0,0 @@ -package org.sandag.abm.airport; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; -import org.sandag.abm.accessibilities.McLogsumsAppender; -import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -public class AirportModeChoiceModel -{ - private transient Logger logger = Logger.getLogger("airportModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication driveAloneModel; - private ChoiceModelApplication shared2Model; - private ChoiceModelApplication shared3Model; - private ChoiceModelApplication transitModel; - private ChoiceModelApplication accessModel; - - private Tracer tracer; - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - private boolean seek; - private HashMap rbMap; - - private McLogsumsCalculator logsumHelper; - private TripModeChoiceDMU mcDmuObject; - private AutoTazSkimsCalculator tazDistanceCalculator; - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of airport model DMUs - */ - public AirportModeChoiceModel(HashMap rbMap, AirportDmuFactoryIf dmuFactory, String airportCode) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String airportModeUecFileName = Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.uec.file"); - airportModeUecFileName = uecFileDirectory + airportModeUecFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.data.page")); - int daPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.da.page")); - int s2Page = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.s2.page")); - int s3Page = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.s3.page")); - int transitPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.transit.page")); - int accessPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".mc.accessMode.page")); - - logger.info("Creating Airport Model Mode Choice Application UECs"); - - // create a DMU - AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); - - // create a ChoiceModelApplication object for drive-alone mode choice - driveAloneModel = new ChoiceModelApplication(airportModeUecFileName, daPage, dataPage, - rbMap, (VariableTable) dmu); - - // create a ChoiceModelApplication object for shared 2 mode choice - shared2Model = new ChoiceModelApplication(airportModeUecFileName, s2Page, dataPage, rbMap, - (VariableTable) dmu); - - // create a ChoiceModelApplication object for shared 3+ mode choice - shared3Model = new ChoiceModelApplication(airportModeUecFileName, s3Page, dataPage, rbMap, - (VariableTable) dmu); - - // create a ChoiceModelApplication object for transit mode choice - transitModel = new ChoiceModelApplication(airportModeUecFileName, transitPage, dataPage, - rbMap, (VariableTable) dmu); - - // create a ChoiceModelApplication object for access mode choice - accessModel = new ChoiceModelApplication(airportModeUecFileName, accessPage, dataPage, - rbMap, (VariableTable) dmu); - - logger.info("Finished Creating Airport Model Mode Choice Application UECs"); - - // set up the tracer object - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(rbMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - SandagModelStructure modelStructure = new SandagModelStructure(); - mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); - - } - - - /** - * Choose airport arrival mode and trip mode for this party. Store results - * in the party object passed as argument. - * - * @param party - * The travel party - * @param dmu - * An airport model DMU - */ - public void chooseMode(AirportParty party, AirportModelDMU dmu) - { - - int origMgra = party.getOriginMGRA(); - int destMgra = party.getDestinationMGRA(); - int origTaz = mgraManager.getTaz(origMgra); - int destTaz = mgraManager.getTaz(destMgra); - int period = party.getDepartTime(); - - boolean inbound = false; - if (party.getDirection() == AirportModelStructure.ARRIVAL) inbound = true; - - dmu.setAirportParty(party); - dmu.setDmuIndexValues(party.getID(), origTaz, destTaz); - - // set trip mc dmu values for transit logsum (gets replaced below by uec values) - double c_ivt = -0.03; - double c_cost = - 0.0003; - - // Solve trip mode level utilities - mcDmuObject.setIvtCoeff(c_ivt); - mcDmuObject.setCostCoeff(c_cost); - double walkTransitLogsum = -999.0; - double driveTransitLogsum = -999.0; - - // if 1-person party, solve for the drive-alone and 2-person logsums - if (party.getSize() == 1) - { - driveAloneModel.computeUtilities(dmu, dmu.getDmuIndex()); - double driveAloneLogsum = driveAloneModel.getLogsum(); - dmu.setDriveAloneLogsum(driveAloneLogsum); - - c_ivt = driveAloneModel.getUEC().lookupVariableIndex("c_ivt"); - c_cost = driveAloneModel.getUEC().lookupVariableIndex("c_cost"); - - shared2Model.computeUtilities(dmu, dmu.getDmuIndex()); - double shared2Logsum = shared2Model.getLogsum(); - dmu.setShared2Logsum(shared2Logsum); - - } else if (party.getSize() == 2) - { // if 2-person party solve for the - // shared 2 and shared 3+ logsums - shared2Model.computeUtilities(dmu, dmu.getDmuIndex()); - double shared2Logsum = shared2Model.getLogsum(); - dmu.setShared2Logsum(shared2Logsum); - - shared3Model.computeUtilities(dmu, dmu.getDmuIndex()); - double shared3Logsum = shared3Model.getLogsum(); - dmu.setShared3Logsum(shared3Logsum); - - c_ivt = shared2Model.getUEC().lookupVariableIndex("c_ivt"); - c_cost = shared2Model.getUEC().lookupVariableIndex("c_cost"); - - } else - { // if 3+ person party, solve the shared 3+ logsums - shared3Model.computeUtilities(dmu, dmu.getDmuIndex()); - double shared3Logsum = shared3Model.getLogsum(); - dmu.setShared3Logsum(shared3Logsum); - - c_ivt = shared3Model.getUEC().lookupVariableIndex("c_ivt"); - c_cost = shared3Model.getUEC().lookupVariableIndex("c_cost"); - } - - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); - walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - if (party.getDirection() == AirportModelStructure.DEPARTURE) - { - logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); - } else - { - logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, period, party.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); - } - - dmu.setWalkTransitLogsum(walkTransitLogsum); - dmu.setDriveTransitLogsum(driveTransitLogsum); - - transitModel.computeUtilities(dmu, dmu.getDmuIndex()); - double transitLogsum = transitModel.getLogsum(); - dmu.setTransitLogsum(transitLogsum); - - // calculate access mode utility and choose access mode - accessModel.computeUtilities(dmu, dmu.getDmuIndex()); - int accessMode = accessModel.getChoiceResult(party.getRandom()); - party.setArrivalMode((byte) accessMode); - - // choose trip mode - int tripMode = 0; - int occupancy = AirportModelStructure.getOccupancy(accessMode, party.getSize()); - double randomNumber = party.getRandom(); - - float valueOfTime = 0; - - if ((accessMode != AirportModelStructure.TRANSIT) && (! AirportModelStructure.taxiTncMode(accessMode))) - { - if (occupancy == 1) - { - tripMode = occupancy; - - //following gets vot from UEC - UtilityExpressionCalculator uec = driveAloneModel.getUEC(); - int votIndex = uec.lookupVariableIndex("vot"); - valueOfTime = (float) uec.getValueForIndex(votIndex); - - } else if (occupancy == 2) - { - tripMode = occupancy; - - //following gets vot from UEC - UtilityExpressionCalculator uec = shared2Model.getUEC(); - int votIndex = uec.lookupVariableIndex("vot"); - valueOfTime = (float) uec.getValueForIndex(votIndex); - - } else if (occupancy > 2) - { - tripMode = 3; - - //following gets vot from UEC - UtilityExpressionCalculator uec = shared3Model.getUEC(); - int votIndex = uec.lookupVariableIndex("vot"); - valueOfTime = (float) uec.getValueForIndex(votIndex); - - } - } else if (accessMode == AirportModelStructure.TRANSIT) - { - int choice = transitModel.getChoiceResult(randomNumber); - double[][] bestTapPairs; - if (choice == 1){ - tripMode = AirportModelStructure.REALLOCATE_WLKTRN; //walk-transit - bestTapPairs = logsumHelper.getBestWtwTripTaps(); - } - else if (choice == 2){ - tripMode = AirportModelStructure.REALLOCATE_KNRPERTRN; //knr-personal tNCVehicle - if (party.getDirection() == AirportModelStructure.DEPARTURE) - bestTapPairs = logsumHelper.getBestDtwTripTaps(); - else - bestTapPairs = logsumHelper.getBestWtdTripTaps(); - } - else { - tripMode = AirportModelStructure.REALLOCATE_KNRTNCTRN; //knr-TNC - if (party.getDirection() == AirportModelStructure.DEPARTURE) - bestTapPairs = logsumHelper.getBestDtwTripTaps(); - else - bestTapPairs = logsumHelper.getBestWtdTripTaps(); - } - - //pick transit path from N-paths - double rn = party.getRandom(); - int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, party.getDebugChoiceModels(), logger); - int boardTap = (int) bestTapPairs[pathIndex][0]; - int alightTap = (int) bestTapPairs[pathIndex][1]; - int set = (int) bestTapPairs[pathIndex][2]; - party.setBoardTap(boardTap); - party.setAlightTap(alightTap); - party.setSet(set); - - //following gets vot from UEC - UtilityExpressionCalculator uec = transitModel.getUEC(); - int votIndex = uec.lookupVariableIndex("vot"); - valueOfTime = (float) uec.getValueForIndex(votIndex); - - }else if(accessMode == AirportModelStructure.TAXI){ - - tripMode=AirportModelStructure.REALLOCATE_TAXI; - } - else if(accessMode == AirportModelStructure.TNC_SINGLE){ - - tripMode=AirportModelStructure.REALLOCATE_TNCSINGLE; - - } - else if(accessMode == AirportModelStructure.TNC_SHARED){ - - tripMode=AirportModelStructure.REALLOCATE_TNCSHARED; - - } - - //set the VOT - if(AirportModelStructure.taxiTncMode(accessMode) ) { - UtilityExpressionCalculator uec = null; - - //following gets vot from UEC - if(occupancy==1) - uec = driveAloneModel.getUEC(); - else if (occupancy==2) - uec = shared2Model.getUEC(); - else - uec = shared3Model.getUEC(); - - int votIndex = uec.lookupVariableIndex("vot"); - valueOfTime = (float) uec.getValueForIndex(votIndex); - - } - party.setMode((byte) tripMode); - party.setValueOfTime(valueOfTime); - } - - /** - * Choose modes for internal trips. - * - * @param airportParties - * An array of travel parties, with destinations already chosen. - * @param dmuFactory - * A DMU Factory. - */ - public void chooseModes(AirportParty[] airportParties, AirportDmuFactoryIf dmuFactory) - { - - AirportModelDMU dmu = dmuFactory.getAirportModelDMU(); - // iterate through the array, choosing mgras and setting them - for (AirportParty party : airportParties) - { - - int ID = party.getID(); - - if ((ID <= 5) || (ID % 100) == 0) - logger.info("Choosing mode for party " + party.getID()); - - chooseMode(party, dmu); - - } - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java deleted file mode 100644 index f9c3159..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModel.java +++ /dev/null @@ -1,233 +0,0 @@ -package org.sandag.abm.airport; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -public class AirportModel -{ - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - - private MatrixDataServerRmi ms; - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - private HashMap rbMap; - private static float sampleRate; - private static int iteration; - private static String airportCode; - - /** - * Constructor - * - * @param rbMap - */ - public AirportModel(HashMap rbMap, float aSampleRate) - { - this.rbMap = rbMap; - this.sampleRate=aSampleRate; - } - - /** - * Run airport model. - */ - public void runModel() - { - Runtime gfg = Runtime.getRuntime(); - long memory1; - // checking the total memeory - System.out.println("Total memory is: "+ gfg.totalMemory()); - // checking free memory - memory1 = gfg.freeMemory(); - System.out.println("Initial free memory at Airport model: "+ memory1); - // calling the garbage collector on demand - gfg.gc(); - memory1 = gfg.freeMemory(); - System.out.println("Free memory after garbage "+ "collection: " + memory1); - - AirportDmuFactory dmuFactory = new AirportDmuFactory(); - - AirportPartyManager apm = new AirportPartyManager(rbMap, sampleRate, airportCode); - - apm.generateAirportParties(); - AirportParty[] parties = apm.getParties(); - - AirportDestChoiceModel destChoiceModel = new AirportDestChoiceModel(rbMap, dmuFactory,airportCode); - destChoiceModel.calculateMgraProbabilities(dmuFactory); - destChoiceModel.calculateTazProbabilities(dmuFactory); - destChoiceModel.chooseOrigins(parties); - - AirportModeChoiceModel modeChoiceModel = new AirportModeChoiceModel(rbMap, dmuFactory,airportCode); - modeChoiceModel.chooseModes(parties, dmuFactory); - - apm.writeOutputFile(rbMap); - - logger.info("Airport Model successfully completed!"); - - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @param args - */ - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Airport Model")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else { - propertiesFile = args[0]; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - if(args[i].equalsIgnoreCase("-airport")){ - airportCode = args[i+1]; - } - } - } - - logger.info("Airport Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - AirportModel airportModel = new AirportModel(pMap, sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = airportModel.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - airportModel.ms = matrixServer; - } else - { - airportModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - airportModel.ms.testRemote("AirportModel"); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(airportModel.ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - airportModel.runModel(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java deleted file mode 100644 index b564cd4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelDMU.java +++ /dev/null @@ -1,478 +0,0 @@ -package org.sandag.abm.airport; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class AirportModelDMU - implements Serializable, VariableTable -{ - protected IndexValues dmuIndex; - - private AirportParty airportParty; - private double[][] sizeTerms; // dimensioned - // by - // segment, - // taz - private int[] zips; // dimensioned - // by - // taz - public static final int OUT = 0; - public static final int IN = 1; - protected static final int NUM_DIR = 2; - - // estimation file defines time periods as: - // 1 | Early AM: 3:00 AM - 5:59 AM | - // 2 | AM Peak: 6:00 AM - 8:59 AM | - // 3 | Early MD: 9:00 AM - 11:59 PM | - // 4 | Late MD: 12:00 PM - 3:29 PM | - // 5 | PM Peak: 3:30 PM - 6:59 PM | - // 6 | Evening: 7:00 PM - 2:59 AM | - - protected static final int LAST_EA_INDEX = 3; - protected static final int LAST_AM_INDEX = 9; - protected static final int LAST_MD_INDEX = 22; - protected static final int LAST_PM_INDEX = 29; - - protected static final int EA = 1; - protected static final int AM = 2; - protected static final int MD = 3; - protected static final int PM = 4; - protected static final int EV = 5; - - protected static final int EA_D = 1; // 5am - protected static final int AM_D = 5; // 7am - protected static final int MD_D = 15; // 12pm - protected static final int PM_D = 27; // 6pm - protected static final int EV_D = 35; // 10pm - protected static final int[] DEFAULT_DEPART_INDICES = {-1, EA_D, AM_D, - MD_D, PM_D, EV_D }; - - protected static final int EA_A = 2; // 5:30am - protected static final int AM_A = 6; // 7:30am - protected static final int MD_A = 16; // 12:30pm - protected static final int PM_A = 28; // 6:30pm - protected static final int EV_A = 36; // 10:30pm - protected static final int[] DEFAULT_ARRIVE_INDICES = {-1, EA_A, AM_A, - MD_A, PM_A, EV_A }; - - protected String[][] departArriveCombinationLabels = { {"EA", "EA"}, - {"EA", "AM"}, {"EA", "MD"}, {"EA", "PM"}, {"EA", "EV"}, {"AM", "AM"}, {"AM", "MD"}, - {"AM", "PM"}, {"AM", "EV"}, {"MD", "MD"}, {"MD", "PM"}, {"MD", "EV"}, {"PM", "PM"}, - {"PM", "EV"}, {"EV", "EV"} }; - - protected int[][] departArriveCombinations = { {EA, EA}, {EA, AM}, - {EA, MD}, {EA, PM}, {EA, EV}, {AM, AM}, {AM, MD}, {AM, PM}, {AM, EV}, {MD, MD}, - {MD, PM}, {MD, EV}, {PM, PM}, {PM, EV}, {EV, EV} }; - - private double driveAloneLogsum; - private double shared2Logsum; - private double shared3Logsum; - private double transitLogsum; - - protected double walkTransitLogsum; - protected double driveTransitLogsum; - - protected Logger _logger = null; - - protected HashMap methodIndexMap; - - - public AirportModelDMU(Logger logger) - { - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - if (logger == null) - { - _logger = Logger.getLogger(AirportModelDMU.class); - } else _logger = logger; - } - - /** - * Set up the method index hashmap, where the key is the getter method for a - * data item and the value is the index. - */ - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - methodIndexMap.put("getDirection", 0); - methodIndexMap.put("getPurpose", 1); - methodIndexMap.put("getSize", 2); - methodIndexMap.put("getIncome", 3); - methodIndexMap.put("getDepartTime", 4); - methodIndexMap.put("getNights", 5); - methodIndexMap.put("getOriginMGRA", 6); - methodIndexMap.put("getLnDestChoiceSizeTazAlt", 7); - methodIndexMap.put("getDestZipAlt", 8); - - methodIndexMap.put("getWalkTransitLogsum", 10); - methodIndexMap.put("getDriveTransitLogsum", 11); - - methodIndexMap.put("getAvAvailable", 70); - - methodIndexMap.put("getDriveAloneLogsum", 90); - methodIndexMap.put("getShared2Logsum", 91); - methodIndexMap.put("getShared3Logsum", 92); - methodIndexMap.put("getTransitLogsum", 93); - - } - - /** - * Look up and return the value for the variable according to the index. - * - */ - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getDirection(); - break; - case 1: - returnValue = getPurpose(); - break; - case 2: - returnValue = getSize(); - break; - case 3: - returnValue = getIncome(); - break; - case 4: - returnValue = getDepartTime(); - break; - case 5: - returnValue = getNights(); - break; - case 6: - returnValue = getOriginMGRA(); - break; - case 7: - returnValue = getLnDestChoiceSizeTazAlt(arrayIndex); - break; - case 8: - returnValue = getDestZipAlt(arrayIndex); - break; - case 10: - returnValue = getWalkTransitLogsum(); - break; - case 11: - returnValue = getDriveTransitLogsum(); - break; - case 70: - returnValue = getAvAvailable(); - break; - case 90: - returnValue = getDriveAloneLogsum(); - break; - case 91: - returnValue = getShared2Logsum(); - break; - case 92: - returnValue = getShared3Logsum(); - break; - case 93: - returnValue = getTransitLogsum(); - break; - default: - _logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - - } - - - /** - * Get travel party direction. - * - * @return Travel party direction. - */ - public int getDirection() - { - return airportParty.getDirection(); - } - - /** - * Get travel party purpose. - * - * @return Travel party direction. - */ - public int getPurpose() - { - return airportParty.getPurpose(); - } - - /** - * Get travel party size. - * - * @return Travel party size. - */ - public int getSize() - { - return airportParty.getSize(); - } - - /** - * Get travel party income. - * - * @return Travel party income. - */ - public int getIncome() - { - return airportParty.getIncome(); - } - - /** - * Get the departure time for the trip - * - * @return Trip departure time. - */ - public int getDepartTime() - { - return airportParty.getDepartTime(); - } - - /** - * Get the number of nights - * - * @return Travel party number of nights. - */ - public int getNights() - { - return airportParty.getNights(); - } - - /** - * Get the origin(non-airport) MGRA - * - * @return Travel party origin MGRA - */ - public int getOriginMGRA() - { - return airportParty.getOriginMGRA(); - } - - public int getAvAvailable() { - - if(airportParty.getAvAvailable()) - return 1; - - return 0; - } - - - /** - * Set the index values for this DMU. - * - * @param id - * @param origTaz - * @param destTaz - */ - public void setDmuIndexValues(int id, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(id); - dmuIndex.setZoneIndex(origTaz); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (airportParty.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug Airport Model"); - } - - } - - /** - * Get the appropriate size term for this purpose and income market. - * - * @param tazNumber - * the number of the taz - * @return the size term - */ - public double getLnDestChoiceSizeTazAlt(int alt) - { - - int purpose = getPurpose(); - int income = getIncome(); - - int segment = AirportModelStructure.getDCSizeSegment(purpose, income); - - return sizeTerms[segment][alt]; - } - - /** - * Get the destination district for this alternative. - * - * @param tazNumber - * @return number of destination district. - */ - public int getDestZipAlt(int alt) - { - - return zips[alt]; - } - - /** - * Set size terms - * - * @param sizeTerms - * A double[][] array dimensioned by segments (purp\income - * groups) and taz numbers - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - /** - * set the zip codes - * - * @param zips - * int[] dimensioned by taz number - */ - public void setZips(int[] zips) - { - this.zips = zips; - } - - /** - * Set the airport party object. - * - * @param party - * The airport party. - */ - public void setAirportParty(AirportParty party) - { - - airportParty = party; - } - - /** - * @return the dmuIndex - */ - public IndexValues getDmuIndex() - { - return dmuIndex; - } - - /** - * @return the driveAloneLogsum - */ - public double getDriveAloneLogsum() - { - return driveAloneLogsum; - } - - /** - * @param driveAloneLogsum - * the driveAloneLogsum to set - */ - public void setDriveAloneLogsum(double driveAloneLogsum) - { - this.driveAloneLogsum = driveAloneLogsum; - } - - /** - * @return the shared2Logsum - */ - public double getShared2Logsum() - { - return shared2Logsum; - } - - /** - * @param shared2Logsum - * the shared2Logsum to set - */ - public void setShared2Logsum(double shared2Logsum) - { - this.shared2Logsum = shared2Logsum; - } - - /** - * @return the shared3Logsum - */ - public double getShared3Logsum() - { - return shared3Logsum; - } - - /** - * @param shared3Logsum - * the shared3Logsum to set - */ - public void setShared3Logsum(double shared3Logsum) - { - this.shared3Logsum = shared3Logsum; - } - - /** - * @return the transitLogsum - */ - public double getTransitLogsum() - { - return transitLogsum; - } - - /** - * @param transitLogsum - * the transitLogsum to set - */ - public void setTransitLogsum(double transitLogsum) - { - this.transitLogsum = transitLogsum; - } - - public double getWalkTransitLogsum() { - return walkTransitLogsum; - } - - public void setWalkTransitLogsum(double walkTransitLogsum) { - this.walkTransitLogsum = walkTransitLogsum; - } - - public double getDriveTransitLogsum() { - return driveTransitLogsum; - } - - public void setDriveTransitLogsum(double driveTransitLogsum) { - this.driveTransitLogsum = driveTransitLogsum; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java deleted file mode 100644 index 4b664ca..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportModelStructure.java +++ /dev/null @@ -1,239 +0,0 @@ -package org.sandag.abm.airport; - -public final class AirportModelStructure -{ - - public static final byte PURPOSES = 5; - public static final byte RESIDENT_BUSINESS = 0; - public static final byte RESIDENT_PERSONAL = 1; - public static final byte VISITOR_BUSINESS = 2; - public static final byte VISITOR_PERSONAL = 3; - public static final byte EXTERNAL = 4; - - public static final byte INTERNAL_PURPOSES = 4; - - public static final byte DEPARTURE = 0; - public static final byte ARRIVAL = 1; - - public static final byte INCOME_SEGMENTS = 8; - public static final byte DC_SIZE_SEGMENTS = INCOME_SEGMENTS * 2 + 2; - - public static final int AM = 0; - public static final int PM = 1; - public static final int OP = 2; - public static final int[] SKIM_PERIODS = {AM, PM, OP}; - public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; - public static final int UPPER_EA = 3; - public static final int UPPER_AM = 9; - public static final int UPPER_MD = 22; - public static final int UPPER_PM = 29; - public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; - - public static final byte ACCESS_MODES = 11; - - public static final byte PARK_TMNL = 1; - public static final byte PARK_SANOFF = 2; - public static final byte PARK_PVTOFF = 3; - public static final byte PUDO_ESC = 4; - public static final byte PUDO_CURB = 5; - public static final byte RENTAL = 6; - public static final byte TAXI = 7; - public static final byte TNC_SINGLE = 8; - public static final byte TNC_SHARED = 9; - public static final byte SHUTTLE_VAN = 10; - public static final byte TRANSIT = 11; - - //reallocate the trip modes from the access choice model to ones that the trip table and other code can read, consistent with - //resident models. - public static final byte REALLOCATE_WLKTRN = 6; //walk access - public static final byte REALLOCATE_KNRPERTRN = 8; //knr-personal tNCVehicle - public static final byte REALLOCATE_KNRTNCTRN = 9; //knr-TNC - public static final byte REALLOCATE_TAXI = 10; - public static final byte REALLOCATE_TNCSINGLE = 11; - public static final byte REALLOCATE_TNCSHARED = 12; - - private AirportModelStructure() - { - } - - /** - * Calculate and return the destination choice size term segment - * - * @param purpose - * @param income - * @return The dc size term segment, currently 0-17, where: 0-7 are 8 income - * groups for RES_BUS 8-15 are 8 income groups for RES_PER 16 is - * VIS_BUS 17 is VIS_PER - */ - public static int getDCSizeSegment(int purpose, int income) - { - - int segment = -1; - - // size terms for resident trips are dimensioned by income - if (purpose < 2) - { - segment = purpose * INCOME_SEGMENTS + income; - } else - { - segment = 2 * INCOME_SEGMENTS + purpose - 2; - } - return segment; - - } - - /** - * Calculate the purpose from the dc size segment. - * - * @param segment - * The dc size segment (0-17) - * @return The purpose - */ - public static int getPurposeFromDCSizeSegment(int segment) - { - - int purpose = -1; - - if (segment < INCOME_SEGMENTS) - { - purpose = 0; - } else if (segment < (AirportModelStructure.INCOME_SEGMENTS * 2)) - { - purpose = 1; - } else if (segment == (AirportModelStructure.INCOME_SEGMENTS * 2)) purpose = 2; - else purpose = 3; - - return purpose; - } - - /** - * Calculate the income from the dc size segment. - * - * @param segment - * The dc size segment (0-17) - * @return The income (defaults to 3 if not a resident purpose) - */ - public static int getIncomeFromDCSizeSegment(int segment) - { - - int income = 3; - - if (segment < AirportModelStructure.INCOME_SEGMENTS) - { - income = (byte) segment; - } else if (segment < (AirportModelStructure.INCOME_SEGMENTS * 2)) - income = ((byte) (segment - AirportModelStructure.INCOME_SEGMENTS)); - - return income; - } - - /** - * return the Skim period index 0=am, 1=pm, 2=off-peak - */ - public static int getSkimPeriodIndex(int departPeriod) - { - - int skimPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; - else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; - else skimPeriodIndex = OP; - - return skimPeriodIndex; - - } - - /** - * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV - */ - public static int getModelPeriodIndex(int departPeriod) - { - - int modelPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; - else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; - else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; - else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; - else modelPeriodIndex = 4; - - return modelPeriodIndex; - - } - - public static String getModelPeriodLabel(int period) - { - return MODEL_PERIOD_LABELS[period]; - } - - public static int getNumberModelPeriods() - { - return MODEL_PERIOD_LABELS.length; - } - - public static String getSkimMatrixPeriodString(int period) - { - int index = getSkimPeriodIndex(period); - return SKIM_PERIOD_STRINGS[index]; - } - - /** - * Get the tNCVehicle occupancy based upon the access mode and the party size. - * - * @param accessMode - * Access mode, 1-based, consistent with definitions above. - * @param partySize - * Number of passengers in travel party - * @return The (minimum) occupancy of the tNCVehicle trip to/from the airport. - */ - public static int getOccupancy(int accessMode, int partySize) - { - - switch (accessMode) - { - case PARK_TMNL: - return partySize; - case PARK_SANOFF: - return partySize; - case PARK_PVTOFF: - return partySize; - case PUDO_ESC: - return partySize + 1; - case PUDO_CURB: - return partySize + 1; - case RENTAL: - return partySize; - case TAXI: - return partySize + 1; - case TNC_SINGLE: - return partySize + 1; - case TNC_SHARED: - return partySize + 1; - case SHUTTLE_VAN: - return partySize + 1; - case TRANSIT: - return partySize; - - default: - throw new RuntimeException( - "Error: AccessMode not found in AirportModel.AirportModelStructure"); - - } - } - - public static boolean taxiTncMode(int accessMode) { - - switch (accessMode) { - case TAXI: - return true; - case TNC_SINGLE: - return true; - case TNC_SHARED: - return true; - } - - return false; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java deleted file mode 100644 index 4f96be2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportParty.java +++ /dev/null @@ -1,321 +0,0 @@ -package org.sandag.abm.airport; - -import java.io.Serializable; -import com.pb.common.math.MersenneTwister; - -public class AirportParty - implements Serializable -{ - - private MersenneTwister random; - private int ID; - - // following variables determined via simulation - private byte direction; - private byte purpose; - private byte size; - private byte income; - private int departTime; - private byte nights; - - private boolean debugChoiceModels; - - // following variables chosen via choice models - private int originMGRA; - private int destinationMGRA; - private int originTAZ; - private int destinationTAZ; - private byte mode; - private byte arrivalMode; - - private float valueOfTime; - - private int boardTap; - private int alightTap; - private int set = -1; - - private boolean avAvailable; - /** - * Public constructor. - * - * @param seed - * A seed for the random number generator. - */ - public AirportParty(long seed) - { - - random = new MersenneTwister(seed); - } - - /** - * @return the iD - */ - public int getID() - { - return ID; - } - - /** - * @param iD - * the iD to set - */ - public void setID(int iD) - { - ID = iD; - } - - /** - * @return the purpose - */ - public byte getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(byte purpose) - { - this.purpose = purpose; - } - - /** - * @return the size - */ - public byte getSize() - { - return size; - } - - /** - * @param size - * the size to set - */ - public void setSize(byte size) - { - this.size = size; - } - - /** - * @return the income - */ - public byte getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(byte income) - { - this.income = income; - } - - /** - * @return the departTime - */ - public int getDepartTime() - { - return departTime; - } - - /** - * @param departTime - * the departTime to set - */ - public void setDepartTime(int departTime) - { - this.departTime = departTime; - } - - /** - * @return the direction - */ - public byte getDirection() - { - return direction; - } - - /** - * @param direction - * the direction to set - */ - public void setDirection(byte direction) - { - this.direction = direction; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() - { - return originMGRA; - } - - /** - * @param originMGRA - * the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) - { - this.originMGRA = originMGRA; - } - - /** - * @return the trip mode - */ - public byte getMode() - { - return mode; - } - - /** - * @param mode - * the trip mode to set - */ - public void setMode(byte mode) - { - this.mode = mode; - } - - /** - * @return the arrivalMode - */ - public byte getArrivalMode() - { - return arrivalMode; - } - - /** - * @param arrivalMode - * the arrivalMode to set - */ - public void setArrivalMode(byte arrivalMode) - { - this.arrivalMode = arrivalMode; - } - - /** - * @return the nights - */ - public byte getNights() - { - return nights; - } - - /** - * @param nights - * the nights to set - */ - public void setNights(byte nights) - { - this.nights = nights; - } - - /** - * Get a random number from the parties random class. - * - * @return A random number. - */ - public double getRandom() - { - return random.nextDouble(); - } - - /** - * @return the debugChoiceModels - */ - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - /** - * @param debugChoiceModels - * the debugChoiceModels to set - */ - public void setDebugChoiceModels(boolean debugChoiceModels) - { - this.debugChoiceModels = debugChoiceModels; - } - - - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() - { - return destinationMGRA; - } - - /** - * @param destinationMGRA - * the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) - { - this.destinationMGRA = destinationMGRA; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public int getBoardTap() { - return boardTap; - } - - public void setBoardTap(int boardTap) { - this.boardTap = boardTap; - } - - public int getAlightTap() { - return alightTap; - } - - public void setAlightTap(int alightTap) { - this.alightTap = alightTap; - } - - public int getSet() { - return set; - } - - public void setSet(int set) { - this.set = set; - } - - public int getOriginTAZ() { - return originTAZ; - } - - public void setOriginTAZ(int originTAZ) { - this.originTAZ = originTAZ; - } - - public int getDestinationTAZ() { - return destinationTAZ; - } - - public void setDestinationTAZ(int destinationTAZ) { - this.destinationTAZ = destinationTAZ; - } - - public boolean getAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(boolean avAvailable) { - this.avAvailable = avAvailable; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java deleted file mode 100644 index 75d7794..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportPartyManager.java +++ /dev/null @@ -1,410 +0,0 @@ -package org.sandag.abm.airport; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -public class AirportPartyManager -{ - - private static Logger logger = Logger.getLogger("SandagTourBasedModel.class"); - - private AirportParty[] parties; - - private double[] purposeDistribution; - private double[][] sizeDistribution; - private double[][] durationDistribution; - private double[][] incomeDistribution; - private double[][] departureDistribution; - private double[][] arrivalDistribution; - - - SandagModelStructure sandagStructure; - private String airportCode; - - private float avShare; - - - /** - * Constructor. Reads properties file and opens/stores all probability - * distributions for sampling. Estimates number of airport travel parties - * and initializes parties[]. - * - * @param resourceFile - * Property file. - */ - public AirportPartyManager(HashMap rbMap, float sampleRate, String airportCode) - { - sandagStructure = new SandagModelStructure(); - this.airportCode = airportCode; - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String purposeFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".purpose.file"); - String sizeFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".size.file"); - String durationFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".duration.file"); - String incomeFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".income.file"); - String departFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".departureTime.file"); - String arriveFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".arrivalTime.file"); - - // Read the distributions - setPurposeDistribution(purposeFile); - sizeDistribution = setDistribution(sizeDistribution, sizeFile); - durationDistribution = setDistribution(durationDistribution, durationFile); - incomeDistribution = setDistribution(incomeDistribution, incomeFile); - departureDistribution = setDistribution(departureDistribution, departFile); - arrivalDistribution = setDistribution(arrivalDistribution, arriveFile); - - // calculate total number of parties - float enplanements = new Float(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".enplanements").replace(",", "")); - float connectingPassengers = new Float(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".connecting").replace(",", "")); - float annualFactor = new Float(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".annualizationFactor")); - float averageSize = new Float(Util.getStringValueFromPropertyMap(rbMap, - "airport."+airportCode+".averageSize")); - - - avShare = Util.getFloatValueFromPropertyMap(rbMap, "Mobility.AV.Share"); - - float directPassengers = (enplanements - connectingPassengers) / annualFactor; - int totalParties = (int) (directPassengers / averageSize) * 2; - parties = new AirportParty[(int)(totalParties*sampleRate)]; - - logger.info("Total airport parties: " + totalParties); - } - - /** - * Create parties based upon total parties (calculated in constructor). Fill - * parties[] with travel parties, assuming one-half are arriving and - * one-half are departing. Simulate party characteristics (income, size, - * duration, time departing from origin or arriving at airport) from - * distributions, also read in during constructor. - * - */ - public void generateAirportParties() - { - - int departures = parties.length / 2; - int arrivals = parties.length - departures; - int totalParties = 0; - int totalPassengers = 0; - for (int i = 0; i < departures; ++i) - { - - AirportParty party = new AirportParty(i * 101 + 1000); - - // simulate from distributions - party.setDirection(AirportModelStructure.DEPARTURE); - byte purpose = (byte) choosePurpose(party.getRandom()); - byte size = (byte) chooseFromDistribution(purpose, sizeDistribution, party.getRandom()); - byte nights = (byte) chooseFromDistribution(purpose, durationDistribution, - party.getRandom()); - byte income = (byte) chooseFromDistribution(purpose, incomeDistribution, - party.getRandom()); - byte period = (byte) chooseFromDistribution(purpose, departureDistribution, - party.getRandom()); - - if(party.getRandom() random) return alt; - } - return -99; - } - - /** - * Choose a purpose. - * - * @param random - * A uniform random number. - * @return the purpose. - */ - protected int choosePurpose(double random) - { - // iterate through the probability array and choose - for (int alt = 0; alt < purposeDistribution.length; ++alt) - { - if (purposeDistribution[alt] > random) return alt; - } - return -99; - } - - /** - * Create a text file and write all records to the file. - * - */ - public void writeOutputFile(HashMap rbMap) - { - - // Open file and print header - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String fileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".output.file"); - - PrintWriter writer = null; - try - { - writer = new PrintWriter(new BufferedWriter(new FileWriter(fileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + fileName + " for writing\n"); - throw new RuntimeException(); - } - String headerString = new String( - "id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ," - + "destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); - writer.print(headerString); - - // Iterate through the array, printing records to the file - for (int i = 0; i < parties.length; ++i) - { - - String record = new String(parties[i].getID() + "," + parties[i].getDirection() + "," - + parties[i].getPurpose() + "," + parties[i].getSize() + "," - + parties[i].getIncome() + "," + parties[i].getNights() + "," - + parties[i].getDepartTime() + "," + parties[i].getOriginMGRA() + "," - + parties[i].getDestinationMGRA() + "," - + parties[i].getOriginTAZ() + "," + parties[i].getDestinationTAZ() + "," - + parties[i].getMode() + "," - + (parties[i].getAvAvailable() ? 1 : 0) + "," - + parties[i].getArrivalMode() + "," + parties[i].getBoardTap() + "," + - + parties[i].getAlightTap() + "," + parties[i].getSet() + "," + - String.format("%9.2f", parties[i].getValueOfTime()) + "\n"); - writer.print(record); - } - writer.close(); - - } - - /** - * @return the parties - */ - public AirportParty[] getParties() - { - return parties; - } - - - /* - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Airport Model Party Manager")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - AirportPartyManager apm = new AirportPartyManager(pMap); - - apm.generateAirportParties(); - - apm.writeOutputFile(pMap); - - logger.info("Airport Model successfully completed!"); - - } -*/ -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java deleted file mode 100644 index b6fbab1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/airport/AirportTripTables.java +++ /dev/null @@ -1,707 +0,0 @@ -package org.sandag.abm.airport; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.crossborder.CrossBorderTripTables; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class AirportTripTables -{ - - private static Logger logger = Logger.getLogger("tripTables"); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TableDataSet tripData; - - // Some parameters - private int[] modeIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // total - // modes, - // returns - // 0=auto - // modes, - // 1=non-motor, - // 2=transit, - // 3= - // other - private int[] matrixIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // modes, - // returns - // the - // element - // of - // the - // matrix - // array - // to - // store - // value - - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private String[] purposeName = {"RES_BUS", "RES_PER", "VIS_BUS", - "VIS_PER" }; - private HashMap rbMap; - - // matrices are indexed by modes, vot bins, submodes - private Matrix[][][] matrix; - - private ResourceBundle rb; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - private String airportCode; - - private MatrixDataServerRmi ms; - private float sampleRate = 1; - private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; - private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; - private float valueOfTimeThresholdLow = 0; - private float valueOfTimeThresholdMed = 0; - //value of time bins by mode group - int[] votBins = {3,1,1,1}; - - public int numSkimSets; - - - public float getSampleRate() { - return sampleRate; - } - - public void setSampleRate(float sampleRate) { - this.sampleRate = sampleRate; - } - - public AirportTripTables(HashMap rbMap, String airportCode) - { - - this.rbMap = rbMap; - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTourModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - } else if (modelStructure.getTourModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - } else if (modelStructure.getTourModeIsWalkTransit(i) - || modelStructure.getTourModeIsDriveTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - } - } - //value of time thresholds - valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); - valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); - this.airportCode = airportCode; - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][][]; - for (int i = 0; i < numberOfModes; ++i) - { - - String modeName; - matrix[i] = new Matrix[votBins[i]][]; - - for(int j = 0; j< votBins[i];++j){ - if (i == 0) - { - matrix[i][j] = new Matrix[autoModes]; - for (int k = 0; k < autoModes; ++k) - { - modeName = modelStructure.getModeName(k + 1); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 1) - { - matrix[i][j] = new Matrix[nmotModes]; - for (int k = 0; k < nmotModes; ++k) - { - modeName = modelStructure.getModeName(k + 1 + autoModes); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 2) - { - matrix[i][j] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l1) - votBin = getValueOfTimeBin(valueOfTime); - - if (mode == 0) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); - } else if (mode == 1) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } else if (mode == 2) - { - - if (boardTap == 0 || alightTap == 0) continue; - - //store transit trips in matrices - mat = (matrixIndex[tripMode]*numSkimSets)+set; - float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); - matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); - - // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) - if (modelStructure.getTourModeIsDriveTransit(tripMode)) - { - - // add the tNCVehicle trip portion to the trip table - if (inbound == 0) - { // from origin to lot (boarding tap) - int PNRTAZ = tapManager.getTazForTap(boardTap); - value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); - matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); - - } else - { // from lot (alighting tap) to destination - int PNRTAZ = tapManager.getTazForTap(alightTap); - value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); - matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); - } - - } - } else - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } - - // generate another drive-alone trip in the opposite direction for - // pickup/dropoff - if (accMode == AirportModelStructure.PUDO_CURB - || accMode == AirportModelStructure.PUDO_CURB) - { - mode = 0; // auto mode - if (SandagModelStructure.getTripModeIsPay(tripMode)) // if the - // passenger - // chose - // pay, - // assume - // the - // driver - // will - // also - // pay - mat = 1; - else mat = 0; - float value = matrix[mode][votBin][mat].getValueAt(destinationTAZ, originTAZ); - matrix[mode][votBin][mat].setValueAt(destinationTAZ, originTAZ, (value + vehicleTrips)); - } - //logger.info("End creating trip tables for period " + timePeriod); - } - } - - /** - * Get the output trip table file names from the properties file, and write - * trip tables for all modes for the given time period. - * - * @param period - * Time period, which will be used to find the period time string - * to append to each trip table matrix file - */ - public void writeTrips(int period, MatrixType mt) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "scenario.path"); - String per = modelStructure.getModelPeriodLabel(period); - String[][] end = new String[4][]; - String[] fileName = new String[4]; - - fileName[0] = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.autoTripMatrix"); - fileName[1] = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.nMotTripMatrix"); - fileName[2] = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.tranTripMatrix"); - fileName[3] = directory - + Util.getStringValueFromPropertyMap(rbMap, "airport."+airportCode+".results.othrTripMatrix"); - - //the end of the name depends on whether there are multiple vot bins or not - String[] votBinName = {"low","med","high"}; - - for(int i = 0; i<4;++i){ - end[i] = new String[votBins[i]]; - for(int j = 0; j < votBins[i];++j){ - if(votBins[i]>1) - end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; - else - end[i][j] = "_" + per + ".omx"; - } - } - for (int i = 0; i < 4; ++i){ - for(int j = 0; j < votBins[i];++j){ - try - { - //Delete the file if it exists - File f = new File(fileName[i]+end[i][j]); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); - f.delete(); - } - - if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); - else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); - } catch (Exception e) - { - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName[i] +end[i][j] + ", for mode index = " + i, e); - throw new RuntimeException(); - } - } - } - - - } - - /** - * Return the value of time bin 0 through 2 based on the thresholds provided in the property map - * @param valueOfTime - * @return value of time bin 0 through 2 - */ - public int getValueOfTimeBin(float valueOfTime){ - - if(valueOfTime pMap; - String propertiesFile = null; - String airportCode = null; - - logger.info(String.format( - "SANDAG Airport Model Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - - float sampleRate = 1.0f; - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-airport")) - { - airportCode = args[i + 1]; - } - } - - AirportTripTables tripTables = new AirportTripTables(pMap, airportCode); - logger.info("Airport Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration+" -airport "+airportCode); - - tripTables.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - tripTables.ms = matrixServer; - } else - { - tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - tripTables.ms.testRemote("AirportTripTables"); - - // mdm = MatrixDataManager.getInstance(); - // mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - tripTables.createTripTables(mt); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java deleted file mode 100644 index 74dff9b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAppendMcLogsumDMU.java +++ /dev/null @@ -1,615 +0,0 @@ -package org.sandag.abm.application; - -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.util.HashMap; -import java.util.Iterator; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.TourModeChoiceDMU; - -import com.pb.common.calculator.IndexValues; - -public class SandagAppendMcLogsumDMU - extends TourModeChoiceDMU -{ - - private int departPeriod; - private int arrivePeriod; - - private int incomeInDollars; - private int adults; - private int autos; - private int hhSize; - private int personIsFemale; - private int age; - private int tourCategoryJoint; - private int tourCategoryEscort; - private int numberOfParticipantsInJointTour; - private int workTourModeIsHOV; - private int workTourModeIsSOV; - private int workTourModeIsBike; - private int tourCategorySubtour; - - public SandagAppendMcLogsumDMU(ModelStructure modelStructure, Logger aLogger) - { - super(modelStructure, aLogger); - setupMethodIndexMap(); - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - - public void setNonMotorizedWalkTimeOut(double walkTime) - { - nmWalkTimeOut = walkTime; - } - - public void setNonMotorizedWalkTimeIn(double walkTime) - { - nmWalkTimeIn = walkTime; - } - - public void setNonMotorizedBikeTimeOut(double bikeTime) - { - nmBikeTimeOut = bikeTime; - } - - public void setNonMotorizedBikeTimeIn(double bikeTime) - { - nmBikeTimeIn = bikeTime; - } - - public float getTimeOutbound() - { - return departPeriod; - } - - public float getTimeInbound() - { - return arrivePeriod; - } - - public void setDepartPeriod(int period) - { - departPeriod = period; - } - - public void setArrivePeriod(int period) - { - arrivePeriod = period; - } - - public void setHhSize(int arg) - { - hhSize = arg; - } - - public void setAge(int arg) - { - age = arg; - } - - public void setTourCategoryJoint(int arg) - { - tourCategoryJoint = arg; - } - - public void setTourCategoryEscort(int arg) - { - tourCategoryEscort = arg; - } - - public void setNumberOfParticipantsInJointTour(int arg) - { - numberOfParticipantsInJointTour = arg; - } - - public void setWorkTourModeIsSOV(int arg) - { - workTourModeIsSOV = arg; - } - - public void setWorkTourModeIsHOV(int arg) - { - workTourModeIsHOV = arg; - } - - public void setWorkTourModeIsBike(int arg) - { - workTourModeIsBike = arg; - } - - public void setPTazTerminalTime(float arg) - { - pTazTerminalTime = arg; - } - - public void setATazTerminalTime(float arg) - { - aTazTerminalTime = arg; - } - - public void setIncomeInDollars(int arg) - { - incomeInDollars = arg; - } - - public int getIncome() - { - return incomeInDollars; - } - - public void setAdults(int arg) - { - adults = arg; - } - - public int getAdults() - { - return adults; - } - - public void setAutos(int arg) - { - autos = arg; - } - - public int getAutos() - { - return autos; - } - - public int getAge() - { - return age; - } - - public int getHhSize() - { - return hhSize; - } - - public int getTourCategoryJoint() - { - return tourCategoryJoint; - } - - public int getTourCategoryEscort() - { - return tourCategoryEscort; - } - - public int getNumberOfParticipantsInJointTour() - { - return numberOfParticipantsInJointTour; - } - - public int getWorkTourModeIsSov() - { - return workTourModeIsSOV; - } - - public int getWorkTourModeIsHov() - { - return workTourModeIsHOV; - } - - public int getWorkTourModeIsBike() - { - return workTourModeIsBike; - } - - public void setPersonIsFemale(int arg) - { - personIsFemale = arg; - } - - public int getFemale() - { - return personIsFemale; - } - - public void setOrigDuDen(double arg) - { - origDuDen = arg; - } - - public void setOrigEmpDen(double arg) - { - origEmpDen = arg; - } - - public void setOrigTotInt(double arg) - { - origTotInt = arg; - } - - public void setDestDuDen(double arg) - { - destDuDen = arg; - } - - public void setDestEmpDen(double arg) - { - destEmpDen = arg; - } - - public void setDestTotInt(double arg) - { - destTotInt = arg; - } - - public double getODUDen() - { - return origDuDen; - } - - public double getOEmpDen() - { - return origEmpDen; - } - - public double getOTotInt() - { - return origTotInt; - } - - public double getDDUDen() - { - return destDuDen; - } - - public double getDEmpDen() - { - return destEmpDen; - } - - public double getDTotInt() - { - return destTotInt; - } - - public double getNm_walkTime_out() - { - return nmWalkTimeOut; - } - - public double getNm_walkTime_in() - { - return nmWalkTimeIn; - } - - public double getNm_bikeTime_out() - { - return nmBikeTimeOut; - } - - public double getNm_bikeTime_in() - { - return nmBikeTimeIn; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getIncome", 2); - methodIndexMap.put("getAdults", 3); - methodIndexMap.put("getFemale", 4); - methodIndexMap.put("getHhSize", 5); - methodIndexMap.put("getAutos", 6); - methodIndexMap.put("getAge", 7); - methodIndexMap.put("getTourCategoryJoint", 8); - methodIndexMap.put("getNumberOfParticipantsInJointTour", 9); - methodIndexMap.put("getWorkTourModeIsSov", 10); - methodIndexMap.put("getWorkTourModeIsBike", 11); - methodIndexMap.put("getWorkTourModeIsHov", 12); - methodIndexMap.put("getPTazTerminalTime", 14); - methodIndexMap.put("getATazTerminalTime", 15); - methodIndexMap.put("getODUDen", 16); - methodIndexMap.put("getOEmpDen", 17); - methodIndexMap.put("getOTotInt", 18); - methodIndexMap.put("getDDUDen", 19); - methodIndexMap.put("getDEmpDen", 20); - methodIndexMap.put("getDTotInt", 21); - methodIndexMap.put("getTourCategoryEscort", 22); - - methodIndexMap.put("getNm_walkTime_out", 90); - methodIndexMap.put("getNm_walkTime_in", 91); - methodIndexMap.put("getNm_bikeTime_out", 92); - methodIndexMap.put("getNm_bikeTime_in", 93); - methodIndexMap.put("getWtw_LB_ivt_out", 176); - methodIndexMap.put("getWtw_LB_ivt_in", 177); - methodIndexMap.put("getWtw_EB_ivt_out", 178); - methodIndexMap.put("getWtw_EB_ivt_in", 179); - methodIndexMap.put("getWtw_BRT_ivt_out", 180); - methodIndexMap.put("getWtw_BRT_ivt_in", 181); - methodIndexMap.put("getWtw_LRT_ivt_out", 182); - methodIndexMap.put("getWtw_LRT_ivt_in", 183); - methodIndexMap.put("getWtw_CR_ivt_out", 184); - methodIndexMap.put("getWtw_CR_ivt_in", 185); - methodIndexMap.put("getWtw_fwait_out", 186); - methodIndexMap.put("getWtw_fwait_in", 187); - methodIndexMap.put("getWtw_xwait_out", 188); - methodIndexMap.put("getWtw_xwait_in", 189); - methodIndexMap.put("getWtw_AccTime_out", 190); - methodIndexMap.put("getWtw_AccTime_in", 191); - methodIndexMap.put("getWtw_EgrTime_out", 192); - methodIndexMap.put("getWtw_EgrTime_in", 193); - methodIndexMap.put("getWtw_WalkAuxTime_out", 194); - methodIndexMap.put("getWtw_WalkAuxTime_in", 195); - methodIndexMap.put("getWtw_fare_out", 196); - methodIndexMap.put("getWtw_fare_in", 197); - methodIndexMap.put("getWtw_xfers_out", 198); - methodIndexMap.put("getWtw_xfers_in", 199); - - methodIndexMap.put("getWtd_LB_ivt_out", 276); - methodIndexMap.put("getWtd_LB_ivt_in", 277); - methodIndexMap.put("getWtd_EB_ivt_out", 278); - methodIndexMap.put("getWtd_EB_ivt_in", 279); - methodIndexMap.put("getWtd_BRT_ivt_out", 280); - methodIndexMap.put("getWtd_BRT_ivt_in", 281); - methodIndexMap.put("getWtd_LRT_ivt_out", 282); - methodIndexMap.put("getWtd_LRT_ivt_in", 283); - methodIndexMap.put("getWtd_CR_ivt_out", 284); - methodIndexMap.put("getWtd_CR_ivt_in", 285); - methodIndexMap.put("getWtd_fwait_out", 286); - methodIndexMap.put("getWtd_fwait_in", 287); - methodIndexMap.put("getWtd_xwait_out", 288); - methodIndexMap.put("getWtd_xwait_in", 289); - methodIndexMap.put("getWtd_AccTime_out", 290); - methodIndexMap.put("getWtd_AccTime_in", 291); - methodIndexMap.put("getWtd_EgrTime_out", 292); - methodIndexMap.put("getWtd_EgrTime_in", 293); - methodIndexMap.put("getWtd_WalkAuxTime_out", 294); - methodIndexMap.put("getWtd_WalkAuxTime_in", 295); - methodIndexMap.put("getWtd_fare_out", 296); - methodIndexMap.put("getWtd_fare_in", 297); - methodIndexMap.put("getWtd_xfers_out", 298); - methodIndexMap.put("getWtd_xfers_in", 299); - methodIndexMap.put("getDtw_LB_ivt_out", 376); - methodIndexMap.put("getDtw_LB_ivt_in", 377); - methodIndexMap.put("getDtw_EB_ivt_out", 378); - methodIndexMap.put("getDtw_EB_ivt_in", 379); - methodIndexMap.put("getDtw_BRT_ivt_out", 380); - methodIndexMap.put("getDtw_BRT_ivt_in", 381); - methodIndexMap.put("getDtw_LRT_ivt_out", 382); - methodIndexMap.put("getDtw_LRT_ivt_in", 383); - methodIndexMap.put("getDtw_CR_ivt_out", 384); - methodIndexMap.put("getDtw_CR_ivt_in", 385); - methodIndexMap.put("getDtw_fwait_out", 386); - methodIndexMap.put("getDtw_fwait_in", 387); - methodIndexMap.put("getDtw_xwait_out", 388); - methodIndexMap.put("getDtw_xwait_in", 389); - methodIndexMap.put("getDtw_AccTime_out", 390); - methodIndexMap.put("getDtw_AccTime_in", 391); - methodIndexMap.put("getDtw_EgrTime_out", 392); - methodIndexMap.put("getDtw_EgrTime_in", 393); - methodIndexMap.put("getDtw_WalkAuxTime_out", 394); - methodIndexMap.put("getDtw_WalkAuxTime_in", 395); - methodIndexMap.put("getDtw_fare_out", 396); - methodIndexMap.put("getDtw_fare_in", 397); - methodIndexMap.put("getDtw_xfers_out", 398); - methodIndexMap.put("getDtw_xfers_in", 399); - - - } - - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 2: - returnValue = getIncome(); - break; - case 3: - returnValue = getAdults(); - break; - case 4: - returnValue = getFemale(); - break; - case 5: - returnValue = getHhSize(); - break; - case 6: - returnValue = getAutos(); - break; - case 7: - returnValue = getAge(); - break; - case 8: - returnValue = getTourCategoryJoint(); - break; - case 9: - returnValue = getNumberOfParticipantsInJointTour(); - break; - case 10: - returnValue = getWorkTourModeIsSov(); - break; - case 11: - returnValue = getWorkTourModeIsBike(); - break; - case 12: - returnValue = getWorkTourModeIsHov(); - break; - case 14: - returnValue = getPTazTerminalTime(); - break; - case 15: - returnValue = getATazTerminalTime(); - break; - case 16: - returnValue = getODUDen(); - break; - case 17: - returnValue = getOEmpDen(); - break; - case 18: - returnValue = getOTotInt(); - break; - case 19: - returnValue = getDDUDen(); - break; - case 20: - returnValue = getDEmpDen(); - break; - case 21: - returnValue = getDTotInt(); - break; - case 22: - returnValue = getTourCategoryEscort(); - break; - case 90: - returnValue = getNm_walkTime_out(); - break; - case 91: - returnValue = getNm_walkTime_in(); - break; - case 92: - returnValue = getNm_bikeTime_out(); - break; - case 93: - returnValue = getNm_bikeTime_in(); - break; - /* TODO - case 176: - methodIndexMap.put("getWtw_LB_ivt_out", 176); - case 177: - methodIndexMap.put("getWtw_LB_ivt_in", 177); - case 178: - methodIndexMap.put("getWtw_EB_ivt_out", 178); - case 179: - methodIndexMap.put("getWtw_EB_ivt_in", 179); - case 180: - methodIndexMap.put("getWtw_BRT_ivt_out", 180); - case 181: - methodIndexMap.put("getWtw_BRT_ivt_in", 181); - case 182: - methodIndexMap.put("getWtw_LRT_ivt_out", 182); - case 183: - methodIndexMap.put("getWtw_LRT_ivt_in", 183); - methodIndexMap.put("getWtw_CR_ivt_out", 184); - methodIndexMap.put("getWtw_CR_ivt_in", 185); - methodIndexMap.put("getWtw_fwait_out", 186); - methodIndexMap.put("getWtw_fwait_in", 187); - methodIndexMap.put("getWtw_xwait_out", 188); - methodIndexMap.put("getWtw_xwait_in", 189); - methodIndexMap.put("getWtw_AccTime_out", 190); - methodIndexMap.put("getWtw_AccTime_in", 191); - methodIndexMap.put("getWtw_EgrTime_out", 192); - methodIndexMap.put("getWtw_EgrTime_in", 193); - methodIndexMap.put("getWtw_WalkAuxTime_out", 194); - methodIndexMap.put("getWtw_WalkAuxTime_in", 195); - methodIndexMap.put("getWtw_fare_out", 196); - methodIndexMap.put("getWtw_fare_in", 197); - methodIndexMap.put("getWtw_xfers_out", 198); - methodIndexMap.put("getWtw_xfers_in", 199); - - methodIndexMap.put("getWtd_LB_ivt_out", 276); - methodIndexMap.put("getWtd_LB_ivt_in", 277); - methodIndexMap.put("getWtd_EB_ivt_out", 278); - methodIndexMap.put("getWtd_EB_ivt_in", 279); - methodIndexMap.put("getWtd_BRT_ivt_out", 280); - methodIndexMap.put("getWtd_BRT_ivt_in", 281); - methodIndexMap.put("getWtd_LRT_ivt_out", 282); - methodIndexMap.put("getWtd_LRT_ivt_in", 283); - methodIndexMap.put("getWtd_CR_ivt_out", 284); - methodIndexMap.put("getWtd_CR_ivt_in", 285); - methodIndexMap.put("getWtd_fwait_out", 286); - methodIndexMap.put("getWtd_fwait_in", 287); - methodIndexMap.put("getWtd_xwait_out", 288); - methodIndexMap.put("getWtd_xwait_in", 289); - methodIndexMap.put("getWtd_AccTime_out", 290); - methodIndexMap.put("getWtd_AccTime_in", 291); - methodIndexMap.put("getWtd_EgrTime_out", 292); - methodIndexMap.put("getWtd_EgrTime_in", 293); - methodIndexMap.put("getWtd_WalkAuxTime_out", 294); - methodIndexMap.put("getWtd_WalkAuxTime_in", 295); - methodIndexMap.put("getWtd_fare_out", 296); - methodIndexMap.put("getWtd_fare_in", 297); - methodIndexMap.put("getWtd_xfers_out", 298); - methodIndexMap.put("getWtd_xfers_in", 299); - methodIndexMap.put("getDtw_LB_ivt_out", 376); - methodIndexMap.put("getDtw_LB_ivt_in", 377); - methodIndexMap.put("getDtw_EB_ivt_out", 378); - methodIndexMap.put("getDtw_EB_ivt_in", 379); - methodIndexMap.put("getDtw_BRT_ivt_out", 380); - methodIndexMap.put("getDtw_BRT_ivt_in", 381); - methodIndexMap.put("getDtw_LRT_ivt_out", 382); - methodIndexMap.put("getDtw_LRT_ivt_in", 383); - methodIndexMap.put("getDtw_CR_ivt_out", 384); - methodIndexMap.put("getDtw_CR_ivt_in", 385); - methodIndexMap.put("getDtw_fwait_out", 386); - methodIndexMap.put("getDtw_fwait_in", 387); - methodIndexMap.put("getDtw_xwait_out", 388); - methodIndexMap.put("getDtw_xwait_in", 389); - methodIndexMap.put("getDtw_AccTime_out", 390); - methodIndexMap.put("getDtw_AccTime_in", 391); - methodIndexMap.put("getDtw_EgrTime_out", 392); - methodIndexMap.put("getDtw_EgrTime_in", 393); - methodIndexMap.put("getDtw_WalkAuxTime_out", 394); - methodIndexMap.put("getDtw_WalkAuxTime_in", 395); - methodIndexMap.put("getDtw_fare_out", 396); - methodIndexMap.put("getDtw_fare_in", 397); - methodIndexMap.put("getDtw_xfers_out", 398); - methodIndexMap.put("getDtw_xfers_in", 399); - */ - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - return returnValue; - - } - - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java deleted file mode 100644 index 82ae86d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAtWorkSubtourFrequencyDMU.java +++ /dev/null @@ -1,64 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.AtWorkSubtourFrequencyDMU; -import org.sandag.abm.ctramp.ModelStructure; -import com.pb.common.calculator.VariableTable; - -public class SandagAtWorkSubtourFrequencyDMU - extends AtWorkSubtourFrequencyDMU - implements VariableTable -{ - - public SandagAtWorkSubtourFrequencyDMU(ModelStructure modelStructure) - { - super(modelStructure); - this.modelStructure = modelStructure; - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getIncomeInDollars", 0); - methodIndexMap.put("getPersonType", 1); - methodIndexMap.put("getFemale", 2); - methodIndexMap.put("getDrivers", 3); - methodIndexMap.put("getNumPreschoolChildren", 4); - methodIndexMap.put("getNumIndivEatOutTours", 5); - methodIndexMap.put("getNumTotalTours", 6); - methodIndexMap.put("getNmEatOutAccessibilityWorkplace", 7); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getIncomeInDollars(); - case 1: - return getPersonType(); - case 2: - return getFemale(); - case 3: - return getDrivers(); - case 4: - return getNumPreschoolChildren(); - case 5: - return getNumIndivEatOutTours(); - case 6: - return getNumTotalTours(); - case 7: - return getNmEatOutAccessibilityWorkplace(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java deleted file mode 100644 index 3c25f8d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagAutoOwnershipChoiceDMU.java +++ /dev/null @@ -1,104 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.AutoOwnershipChoiceDMU; - -public class SandagAutoOwnershipChoiceDMU - extends AutoOwnershipChoiceDMU -{ - - public SandagAutoOwnershipChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getDrivers", 1); - methodIndexMap.put("getNumFtWorkers", 2); - methodIndexMap.put("getNumPtWorkers", 3); - methodIndexMap.put("getNumPersons18to24", 4); - methodIndexMap.put("getNumPersons6to15", 5); - methodIndexMap.put("getNumPersons80plus", 6); - methodIndexMap.put("getNumPersons65to79", 7); - methodIndexMap.put("getHhIncomeInDollars", 8); - methodIndexMap.put("getNumHighSchoolGraduates", 9); - methodIndexMap.put("getDetachedDwellingType", 10); - methodIndexMap.put("getUseAccessibilities", 11); - methodIndexMap.put("getHomeTazNonMotorizedAccessibility", 12); - methodIndexMap.put("getHomeTazAutoAccessibility", 13); - methodIndexMap.put("getHomeTazTransitAccessibility", 14); - methodIndexMap.put("getWorkAutoDependency", 15); - methodIndexMap.put("getSchoolAutoDependency", 16); - methodIndexMap.put("getWorkersRailProportion", 17); - methodIndexMap.put("getStudentsRailProportion", 18); - methodIndexMap.put("getGq", 19); - methodIndexMap.put("getNumPersons18to35", 25); - methodIndexMap.put("getNumPersons65plus", 26); - methodIndexMap.put("getWorkAutoTime", 27); - methodIndexMap.put("getHomeTazMaasAccessibility", 28); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getDrivers(); - case 2: - return getNumFtWorkers(); - case 3: - return getNumPtWorkers(); - case 4: - return getNumPersons18to24(); - case 5: - return getNumPersons6to15(); - case 6: - return getNumPersons80plus(); - case 7: - return getNumPersons65to79(); - case 8: - return getHhIncomeInDollars(); - case 9: - return getNumHighSchoolGraduates(); - case 10: - return getDetachedDwellingType(); - case 11: - return getUseAccessibilities(); - case 12: - return getHomeTazNonMotorizedAccessibility(); - case 13: - return getHomeTazAutoAccessibility(); - case 14: - return getHomeTazTransitAccessibility(); - case 15: - return getWorkAutoDependency(); - case 16: - return getSchoolAutoDependency(); - case 17: - return getWorkersRailProportion(); - case 18: - return getStudentsRailProportion(); - case 19: - return getGq(); - case 25: - return getNumPersons18to35(); - case 26: - return getNumPersons65Plus(); - case 27: - return getWorkAutoTime(); - case 28: - return getHomeTazMaasAccessibility(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java deleted file mode 100644 index f69cfdc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCoordinatedDailyActivityPatternDMU.java +++ /dev/null @@ -1,176 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.CoordinatedDailyActivityPatternDMU; - -public class SandagCoordinatedDailyActivityPatternDMU - extends CoordinatedDailyActivityPatternDMU -{ - - public SandagCoordinatedDailyActivityPatternDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getFullTimeWorkerA", 0); - methodIndexMap.put("getFullTimeWorkerB", 1); - methodIndexMap.put("getFullTimeWorkerC", 2); - methodIndexMap.put("getPartTimeWorkerA", 3); - methodIndexMap.put("getPartTimeWorkerB", 4); - methodIndexMap.put("getPartTimeWorkerC", 5); - methodIndexMap.put("getUniversityStudentA", 6); - methodIndexMap.put("getUniversityStudentB", 7); - methodIndexMap.put("getUniversityStudentC", 8); - methodIndexMap.put("getNonWorkingAdultA", 9); - methodIndexMap.put("getNonWorkingAdultB", 10); - methodIndexMap.put("getNonWorkingAdultC", 11); - methodIndexMap.put("getRetiredA", 12); - methodIndexMap.put("getRetiredB", 13); - methodIndexMap.put("getRetiredC", 14); - methodIndexMap.put("getDrivingAgeSchoolChildA", 15); - methodIndexMap.put("getDrivingAgeSchoolChildB", 16); - methodIndexMap.put("getDrivingAgeSchoolChildC", 17); - methodIndexMap.put("getPreDrivingAgeSchoolChildA", 18); - methodIndexMap.put("getPreDrivingAgeSchoolChildB", 19); - methodIndexMap.put("getPreDrivingAgeSchoolChildC", 20); - methodIndexMap.put("getPreSchoolChildA", 21); - methodIndexMap.put("getPreSchoolChildB", 22); - methodIndexMap.put("getPreSchoolChildC", 23); - methodIndexMap.put("getAgeA", 24); - methodIndexMap.put("getFemaleA", 25); - methodIndexMap.put("getMoreCarsThanWorkers", 26); - methodIndexMap.put("getFewerCarsThanWorkers", 27); - methodIndexMap.put("getZeroCars", 28); - methodIndexMap.put("getHHIncomeInDollars", 29); - methodIndexMap.put("getHhDetach", 30); - methodIndexMap.put("getUsualWorkLocationIsHomeA", 31); - methodIndexMap.put("getNoUsualWorkLocationA", 32); - methodIndexMap.put("getNoUsualSchoolLocationA", 33); - methodIndexMap.put("getHhSize", 34); - methodIndexMap.put("getWorkLocationModeChoiceLogsumA", 35); - methodIndexMap.put("getSchoolLocationModeChoiceLogsumA", 36); - methodIndexMap.put("getRetailAccessibility", 37); - methodIndexMap.put("getNumAdultsWithNonMandatoryDap", 38); - methodIndexMap.put("getNumAdultsWithMandatoryDap", 39); - methodIndexMap.put("getNumKidsWithNonMandatoryDap", 40); - methodIndexMap.put("getNumKidsWithMandatoryDap", 41); - methodIndexMap.put("getAllAdultsAtHome", 42); - methodIndexMap.put("getWorkAccessForMandatoryDap", 43); - methodIndexMap.put("getTelecommuteFrequencyA", 44); - methodIndexMap.put("getTelecommuteFrequencyB", 45); - methodIndexMap.put("getTelecommuteFrequencyC", 46); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getFullTimeWorkerA(); - case 1: - return getFullTimeWorkerB(); - case 2: - return getFullTimeWorkerC(); - case 3: - return getPartTimeWorkerA(); - case 4: - return getPartTimeWorkerB(); - case 5: - return getPartTimeWorkerC(); - case 6: - return getUniversityStudentA(); - case 7: - return getUniversityStudentB(); - case 8: - return getUniversityStudentC(); - case 9: - return getNonWorkingAdultA(); - case 10: - return getNonWorkingAdultB(); - case 11: - return getNonWorkingAdultC(); - case 12: - return getRetiredA(); - case 13: - return getRetiredB(); - case 14: - return getRetiredC(); - case 15: - return getDrivingAgeSchoolChildA(); - case 16: - return getDrivingAgeSchoolChildB(); - case 17: - return getDrivingAgeSchoolChildC(); - case 18: - return getPreDrivingAgeSchoolChildA(); - case 19: - return getPreDrivingAgeSchoolChildB(); - case 20: - return getPreDrivingAgeSchoolChildC(); - case 21: - return getPreSchoolChildA(); - case 22: - return getPreSchoolChildB(); - case 23: - return getPreSchoolChildC(); - case 24: - return getAgeA(); - case 25: - return getFemaleA(); - case 26: - return getMoreCarsThanWorkers(); - case 27: - return getFewerCarsThanWorkers(); - case 28: - return getZeroCars(); - case 29: - return getHHIncomeInDollars(); - case 30: - return getHhDetach(); - case 31: - return getUsualWorkLocationIsHomeA(); - case 32: - return getNoUsualWorkLocationA(); - case 33: - return getNoUsualSchoolLocationA(); - case 34: - return getHhSize(); - case 35: - return getWorkLocationModeChoiceLogsumA(); - case 36: - return getSchoolLocationModeChoiceLogsumA(); - case 37: - return getRetailAccessibility(); - case 38: - return getNumAdultsWithNonMandatoryDap(); - case 39: - return getNumAdultsWithMandatoryDap(); - case 40: - return getNumKidsWithNonMandatoryDap(); - case 41: - return getNumKidsWithMandatoryDap(); - case 42: - return getAllAdultsAtHome(); - case 43: - return getWorkAccessForMandatoryDap(); - case 44: - return getTelecommuteFrequencyA(); - case 45: - return getTelecommuteFrequencyB(); - case 46: - return getTelecommuteFrequencyC(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java deleted file mode 100644 index f329b78..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCreateTripGenerationFiles.java +++ /dev/null @@ -1,1055 +0,0 @@ -package org.sandag.abm.application; - -import java.io.BufferedReader; -import java.io.File; -import java.io.FileNotFoundException; -import java.io.FileReader; -import java.io.IOException; -import java.util.HashMap; -import java.util.ResourceBundle; -import java.util.StringTokenizer; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.datafile.CSVFileWriter; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -public class SandagCreateTripGenerationFiles -{ - - private static Logger logger = Logger.getLogger(SandagCreateTripGenerationFiles.class); - - private static final String SANDAG_TRIP_GEN_FILE_KEY = "trip.model.trips.file"; - private static final String ABM_TRIP_GEN_FILE_KEY = "output.trips.file"; - private static final String ABM_INDIV_TRIP_FILE_KEY = "abm.individual.trip.file"; - private static final String TAZ_TDZ_CORRESP_KEY = "taz.tdz.corresp.file"; - private static final String SCALE_NHB_KEY = "scale.nhb"; - - private static final String TAZ_COLUMN_HEADING = "taz"; - private static final String TDZ_COLUMN_HEADING = "tdz"; - - private static final String TRIP_ORIG_PURPOSE_FIELD_NAME = "orig_purpose"; - private static final String TRIP_DEST_PURPOSE_FIELD_NAME = "dest_purpose"; - private static final String TRIP_ORIG_MGRA_FIELD_NAME = "orig_maz"; - private static final String TRIP_DEST_MGRA_FIELD_NAME = "dest_maz"; - private static final String[] HH_HEADINGS = { - TRIP_ORIG_PURPOSE_FIELD_NAME, TRIP_DEST_PURPOSE_FIELD_NAME, TRIP_ORIG_MGRA_FIELD_NAME, - TRIP_DEST_MGRA_FIELD_NAME }; - - private static final int MAX_PURPOSE_INDEX = 10; - - private static final int MIN_EXTERNAL_TDZ = 1; - private static final int MAX_EXTERNAL_TDZ = 12; - - private static final String TAZ_FIELD_HEADING = "zone"; - - private static final String[] TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS = {"a1", "a2", - "a3", "a4", "a5", "a8" }; - private static final String[] TRIP_MODEL_HOME_BASED_PRODUCTION_HEADINGS = {"p1", "p2", - "p3", "p4", "p5", "p8" }; - private static final String[] TRIP_MODEL_NON_HOME_BASED_ATTRACTION_HEADINGS = {"a6", "a7"}; - private static final String[] TRIP_MODEL_NON_HOME_BASED_PRODUCTION_HEADINGS = {"p6", "p7"}; - private static final String[] TRIP_MODEL_OTHER_BASED_PRODUCTION_HEADINGS = {"p9", "p10"}; - private static final String[] TRIP_MODEL_OTHER_BASED_ATTRACTION_HEADINGS = {"a9", "a10"}; - - private static final int[] AB_MODEL_HOME_BASED_PRODUCTION_INDICES = {1, 2, 3, 4, 5, 8}; - private static final int[] AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES = {6, 7}; - private static final int[] AB_MODEL_OTHER_PRODUCTION_INDICES = {9, 10}; - - private float[][] ieTrips; - - private static final String[] TABLE_HEADINGS = {"zone", "p1", - "p2", "p3", "p4", "p5", "p6", "p7", "p8", "p9", "p10", "a1", "a2", "a3", "a4", "a5", - "a6", "a7", "a8", "a9", "a10" }; - private static final String[] TABLE_HEADING_DESCRIPTIONS = {"zone", - "home based work", "home based university", "home based school", "home based shop", - "home based other", "non home based work related", "non home based other", - "home based escort", "home based visitor", "home based airport", "home based work", - "home based university", "home based school", "home based shop", "home based other", - "non home based work related", "non home based other", "home based escort", - "home based visitor", "home based airport" }; - - private MgraDataManager mgraManager; - private int maxTdz; - - public SandagCreateTripGenerationFiles(HashMap rbMap) - { - - mgraManager = MgraDataManager.getInstance(rbMap); - - } - - public void createTripGenFile(HashMap rbMap) - { - - String tgInputFile = rbMap.get(SANDAG_TRIP_GEN_FILE_KEY); - if (tgInputFile == null) - { - logger.error("Error getting the filename from the properties file for the input Sandag Trip Prods/Attrs by MDZ file."); - logger.error("Properties file target: " + SANDAG_TRIP_GEN_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + SANDAG_TRIP_GEN_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - String tgOutputFile = rbMap.get(ABM_TRIP_GEN_FILE_KEY); - if (tgOutputFile == null) - { - logger.error("Error getting the filename from the properties file to use for the new Trip Prods/Attrs by MDZ file created."); - logger.error("Properties file target: " + ABM_TRIP_GEN_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + ABM_TRIP_GEN_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - String abmTripFile = rbMap.get(ABM_INDIV_TRIP_FILE_KEY); - if (abmTripFile == null) - { - logger.error("Error getting the filename from the properties file to use for the ABM Model individual trips file."); - logger.error("Properties file target: " + ABM_INDIV_TRIP_FILE_KEY + " not found."); - logger.error("Please specify a filename value for the " + ABM_INDIV_TRIP_FILE_KEY - + " property."); - throw new RuntimeException(); - } - - String correspFile = rbMap.get(TAZ_TDZ_CORRESP_KEY); - if (correspFile == null) - { - logger.error("Error getting the filename from the properties file to use for the TAZ / TDZ correspondence file."); - logger.error("Properties file target: " + TAZ_TDZ_CORRESP_KEY + " not found."); - logger.error("Please specify a filename value for the " + TAZ_TDZ_CORRESP_KEY - + " property."); - throw new RuntimeException(); - } - - // default is false - boolean scaleNhbToAbm = false; - String scaleNhbToAbmString = rbMap.get(SCALE_NHB_KEY); - if (scaleNhbToAbmString != null && scaleNhbToAbmString.equalsIgnoreCase("true")) - scaleNhbToAbm = true; - logger.info("parameter to enable scaling NHB prods/attrs has a value of: " + scaleNhbToAbm); - - HashMap tazTdzMap = createTazTdzMap(correspFile); - - TableDataSet inTgTds = readInputTripGenFile(tgInputFile); - - int[][] tdzTrips = readInputAbmIndivTripFile(abmTripFile, tazTdzMap); - - TableDataSet outAbmTds = produceAbmTripTableDataSet(inTgTds, tdzTrips, scaleNhbToAbm); - - writeAbmTripGenFile(tgOutputFile, outAbmTds); - - logger.info(""); - logger.info(""); - logger.info("finished producing new trip generation files from the ABM trip data."); - } - - private TableDataSet readInputTripGenFile(String fileName) - { - - TableDataSet inTgTds = null; - - try - { - logger.info(""); - logger.info(""); - logger.info("reading input trip generation file."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - inTgTds = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading input trip generation data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - // create TDZ by purpose arrays for IE Prods and IE attrs from the trip - // model - // these will be added into the home-based AB model prods and attrs - // use the first dimension 0 element to accumulate totals by purpose for - // logging - ieTrips = new float[maxTdz + 1][2 * MAX_PURPOSE_INDEX + 1]; - for (int i = 0; i < inTgTds.getRowCount(); i++) - { - int tdz = (int) inTgTds.getValueAt(i + 1, TAZ_FIELD_HEADING); - for (int j = 1; j < inTgTds.getColumnCount(); j++) - { - if (tdz >= MIN_EXTERNAL_TDZ && tdz <= MAX_EXTERNAL_TDZ) - { - ieTrips[i + 1][j] = inTgTds.getValueAt(i + 1, j + 1); - ieTrips[0][j] += inTgTds.getValueAt(i + 1, j + 1); - } - } - } - - // log column totals - logger.info(""); - logger.info(""); - logger.info("\t" + inTgTds.getRowCount() + " rows in input file."); - logger.info("\t" + inTgTds.getColumnCount() + " columns in input file."); - logger.info(""); - logger.info(String.format("\t%-15s %-30s %15s %15s", "Column Name", "Column Purpose", - "Column Total", "Int-Ext")); - - String[] headings = inTgTds.getColumnLabels(); - logger.info(String.format("\t%-15s %-30s %15s %15s", headings[0], "N/A", "N/A", "N/A")); - float totProd = 0; - float totAttr = 0; - float totalIeProds = 0; - float totalIeAttrs = 0; - float columnSum = 0; - for (int i = 1; i < inTgTds.getColumnCount(); i++) - { - - columnSum = inTgTds.getColumnTotal(i + 1); - - // 1st 10 fields after zone are production fields, next 10 are - // attraction - // fields - if (i <= 10) - { - totProd += columnSum; - totalIeProds += ieTrips[0][i]; - } else - { - totAttr += columnSum; - totalIeAttrs += ieTrips[0][i]; - } - - logger.info(String.format("\t%-15s %-30s %15.1f %15.1f", headings[i], - TABLE_HEADING_DESCRIPTIONS[i], columnSum, ieTrips[0][i])); - - } - - logger.info(""); - logger.info(""); - logger.info(String.format("\ttotal productions = %15.1f", totProd)); - logger.info(String.format("\ttotal attractions = %15.1f", totAttr)); - logger.info(String.format("\ttotal IE productions = %12.1f", totalIeProds)); - logger.info(String.format("\ttotal IE attractions = %12.1f", totalIeAttrs)); - logger.info(""); - - return inTgTds; - } - - private HashMap createTazTdzMap(String correspFile) - { - - TableDataSet tazTdzTds = null; - - try - { - logger.info(""); - logger.info(""); - logger.info("reading input taz-tdz correspondence file."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - tazTdzTds = reader.readFile(new File(correspFile)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading input taz-tdz correspondence file: %s into TableDataSet object.", - correspFile)); - throw new RuntimeException(e); - } - - maxTdz = 0; - HashMap tazTdzMap = new HashMap(); - for (int r = 1; r <= tazTdzTds.getRowCount(); r++) - { - int taz = (int) tazTdzTds.getValueAt(r, TAZ_COLUMN_HEADING); - int tdz = (int) tazTdzTds.getValueAt(r, TDZ_COLUMN_HEADING); - tazTdzMap.put(taz, tdz); - - if (tdz > maxTdz) maxTdz = tdz; - } - - // a trip record with origin or destination mgra=0 or mgra=-1 (location - // not - // determined) should map to tdz=0 - // if a trip record mgra is not greater than 0, its taz will be 0 - then - // the - // following entry will map it to tdz=0. - tazTdzMap.put(0, 0); - - return tazTdzMap; - - } - - private int[][] readInputAbmIndivTripFile(String fileName, HashMap tazTdzMap) - { - - String origPurpose = ""; - String destPurpose = ""; - int origMgra = 0; - int destMgra = 0; - int homeTdz = 0; - int tripPurposeIndex = 0; - - // open the file for reading - String delimSet = ",\t\n\r\f\""; - BufferedReader inputStream = null; - try - { - inputStream = new BufferedReader(new FileReader(new File(fileName))); - } catch (FileNotFoundException e) - { - logger.fatal(String.format("Exception occurred reading input abm indiv trip file: %s.", - fileName)); - throw new RuntimeException(e); - } - - // first parse the trip file field names from the first record and - // associate - // column position with fields specified to be read - HashMap columnIndexHeadingMap = new HashMap(); - String line = ""; - try - { - line = inputStream.readLine(); - } catch (IOException e) - { - logger.fatal(String.format( - "Exception occurred reading header record of input abm indiv trip file: %s.", - fileName)); - logger.fatal(String.format("line = %s.", line)); - throw new RuntimeException(e); - } - StringTokenizer st = new StringTokenizer(line, delimSet); - int col = 0; - while (st.hasMoreTokens()) - { - String label = st.nextToken(); - for (String heading : HH_HEADINGS) - { - if (heading.equalsIgnoreCase(label)) - { - columnIndexHeadingMap.put(col, heading); - break; - } - } - col++; - } - - // dimension the array to hold trips summarized by tdz and trip model - // purpose - int[][] abmTdzTrips = new int[maxTdz + 1][MAX_PURPOSE_INDEX + 1]; - int[] abmTdzTotalTrips = new int[MAX_PURPOSE_INDEX + 1]; - - // read the trip records from the file - int lineCount = 0; - try - { - - while ((line = inputStream.readLine()) != null) - { - - lineCount++; - - // get the values for the fields specified. - col = 0; - st = new StringTokenizer(line, delimSet); - while (st.hasMoreTokens()) - { - String fieldValue = st.nextToken(); - if (columnIndexHeadingMap.containsKey(col)) - { - String fieldName = columnIndexHeadingMap.get(col++); - - if (fieldName.equalsIgnoreCase(TRIP_ORIG_PURPOSE_FIELD_NAME)) - { - origPurpose = fieldValue; - } else if (fieldName.equalsIgnoreCase(TRIP_DEST_PURPOSE_FIELD_NAME)) - { - destPurpose = fieldValue; - } else if (fieldName.equalsIgnoreCase(TRIP_ORIG_MGRA_FIELD_NAME)) - { - origMgra = Integer.parseInt(fieldValue); - } else if (fieldName.equalsIgnoreCase(TRIP_DEST_MGRA_FIELD_NAME)) - { - destMgra = Integer.parseInt(fieldValue); - - // don't need to process any more fields - break; - - } - - } else - { - col++; - } - - } - - int homeTaz = 0; - try - { - if (origPurpose.equalsIgnoreCase("Home") && origMgra > 0) homeTaz = mgraManager - .getTaz(origMgra); - else if (destPurpose.equalsIgnoreCase("Home") && destMgra > 0) - homeTaz = mgraManager.getTaz(destMgra); - } catch (Exception e) - { - logger.error("error getting home taz from mgraManager for origPurpose = " - + origPurpose + ", origMgra = " + origMgra + ", destPurpose = " - + destPurpose + ", destMgra = " + destMgra + ", lineCount = " - + lineCount); - throw new RuntimeException(e); - } - - try - { - homeTdz = tazTdzMap.get(homeTaz); - } catch (Exception e) - { - logger.error("error getting home tdz from tazTdzMap for homeTaz = " + homeTaz - + ", lineCount = " + lineCount); - throw new RuntimeException(e); - } - - try - { - // get the trip based model purpose index for this abm model - // trip - tripPurposeIndex = getTripModelPurposeForAbmTrip(origPurpose, destPurpose); - } catch (Exception e) - { - logger.error("error getting tripPurposeIndex for origPurpose = " + origPurpose - + ", destPurpose = " + destPurpose + ", lineCount = " + lineCount); - throw new RuntimeException(e); - } - - // accumulate trips in table - if (tripPurposeIndex >= 1 && tripPurposeIndex <= 5 || tripPurposeIndex == 8) - { - if (homeTdz > 0) abmTdzTrips[homeTdz][tripPurposeIndex]++; - else - { - logger.error("home tdz is le 0 for home-based trip."); - throw new RuntimeException(); - } - } else - { - if (homeTdz > 0) - { - logger.error("home tdz is gt 0 for non-home-based trip."); - throw new RuntimeException(); - } - abmTdzTrips[homeTdz][tripPurposeIndex]++; - } - abmTdzTotalTrips[tripPurposeIndex]++; - - } - - } catch (NumberFormatException e) - { - logger.fatal(String - .format("NumberFormatException occurred reading record of input abm indiv trip file: %s.", - fileName)); - logger.fatal(String.format("last record number read = %d.", lineCount)); - } catch (IOException e) - { - logger.fatal(String.format( - "IOException occurred reading record of input abm indiv trip file: %s.", - fileName)); - logger.fatal(String.format("last record number read = %d.", lineCount)); - } - - logger.info(lineCount + " trip records read from " + fileName); - - // log a summary report of trips by trip model purpose - logger.info(""); - logger.info(""); - logger.info("ABM Trip file trips by TM purpose"); - logger.info(String.format("\t%-15s %-30s %15s", "Column Name", "Column Purpose", - "Column Total")); - String[] headings = {"", "hbw", "hbu", "hbc", "hbs", "hbo", "nhw", "nho", "hbp"}; - int total = 0; - for (int i = 1; i < headings.length; i++) - { - logger.info(String.format("\t%-15s %-30s %15d", headings[i], - TABLE_HEADING_DESCRIPTIONS[i], abmTdzTotalTrips[i])); - total += abmTdzTotalTrips[i]; - } - logger.info(String.format("\t%-15s %-30s %15d", "Total", "", total)); - - return abmTdzTrips; - - } - - private int getTripModelPurposeForAbmTrip(String origPurpose, String destPurpose) - { - - /* - * assignment rules: replace tpurp4s = 1 if orig_purpose=="Home" & - * dest_purpose=="Work"; replace tpurp4s = 1 if orig_purpose=="Work" & - * dest_purpose=="Home"; replace tpurp4s = 2 if orig_purpose=="Home" & - * dest_purpose=="University"; replace tpurp4s = 2 if - * orig_purpose=="University" & dest_purpose=="Home"; replace tpurp4s = - * 3 if orig_purpose=="Home" & dest_purpose=="School"; replace tpurp4s = - * 3 if orig_purpose=="School" & dest_purpose=="Home"; replace tpurp4s = - * 4 if orig_purpose=="Home" & dest_purpose=="Shop"; replace tpurp4s = 4 - * if orig_purpose=="Shop" & dest_purpose=="Home"; replace tpurp4s = 5 - * if orig_purpose=="Home" & (dest_purpose=="Maintenance" | - * dest_purpose=="Eating Out" | dest_purpose=="Visiting" | - * dest_purpose=="Discretionary"); replace tpurp4s = 5 if - * dest_purpose=="Home" & (orig_purpose=="Maintenance" | - * orig_purpose=="Eating Out" | orig_purpose=="Visiting" | - * orig_purpose=="Discretionary"); replace tpurp4s = 8 if - * orig_purpose=="Home" & dest_purpose=="Escort"; replace tpurp4s = 8 if - * orig_purpose=="Escort" & dest_purpose=="Home"; replace tpurp4s = 6 if - * orig_purpose=="Work" & dest_purpose!="Home"; replace tpurp4s = 6 if - * orig_purpose!="Home" & dest_purpose=="Work"; replace tpurp4s = 6 if - * orig_purpose=="Work-Based" | dest_purpose=="Work-Based"; replace - * tpurp4s = 7 if tpurp4s==0; - */ - - int tripPurposeIndex = 0; - if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Work")) tripPurposeIndex = 1; - else if (origPurpose.equalsIgnoreCase("Work") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 1; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("University")) tripPurposeIndex = 2; - else if (origPurpose.equalsIgnoreCase("University") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 2; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("School")) tripPurposeIndex = 3; - else if (origPurpose.equalsIgnoreCase("School") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 3; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Shop")) tripPurposeIndex = 4; - else if (origPurpose.equalsIgnoreCase("Shop") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 4; - else if (origPurpose.equalsIgnoreCase("Home") - && destPurpose.equalsIgnoreCase("Maintenance")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Maintenance") - && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Eating Out")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Eating Out") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Visiting")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Visiting") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Home") - && destPurpose.equalsIgnoreCase("Discretionary")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Discretionary") - && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Home") - && destPurpose.equalsIgnoreCase("Work Related")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Work Related") - && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 5; - else if (origPurpose.equalsIgnoreCase("Home") && destPurpose.equalsIgnoreCase("Escort")) tripPurposeIndex = 8; - else if (origPurpose.equalsIgnoreCase("Escort") && destPurpose.equalsIgnoreCase("Home")) tripPurposeIndex = 8; - else if (origPurpose.equalsIgnoreCase("Work") && (!destPurpose.equalsIgnoreCase("Home"))) tripPurposeIndex = 6; - else if ((!destPurpose.equalsIgnoreCase("Home")) && destPurpose.equalsIgnoreCase("Work")) tripPurposeIndex = 6; - else if (origPurpose.equalsIgnoreCase("Work-Based") - || destPurpose.equalsIgnoreCase("Work-Based")) tripPurposeIndex = 6; - else tripPurposeIndex = 7; - - return tripPurposeIndex; - - } - - private TableDataSet produceAbmTripTableDataSet(TableDataSet inTgTds, int[][] tdzTrips, - boolean scaleNhbToAbm) - { - - float[][] newTrips = new float[maxTdz][2 * MAX_PURPOSE_INDEX + 1]; - - saveAbmHbProdsAndScaleTmAttrs(inTgTds, tdzTrips, newTrips); - - saveTmNhbProdsAsAbmProds(inTgTds, tdzTrips, scaleNhbToAbm, newTrips); - - saveTmNhbAttrsAsAbmProds(inTgTds, tdzTrips, scaleNhbToAbm, newTrips); - - addTmIeAttrsToAbmHbProds(inTgTds, tdzTrips, newTrips); - - addTmIeProdsToAbmHbAttrs(inTgTds, tdzTrips, newTrips); - - addTmAirportAndVisitorProdsToAbmProds(inTgTds, tdzTrips, newTrips); - - addTmAirportAndVisitorAttrsToAbmAttrs(inTgTds, tdzTrips, newTrips); - - saveZoneField(inTgTds, newTrips); - - TableDataSet abmTds = createFinalTableDataset(newTrips); - - return abmTds; - } - - private void saveAbmHbProdsAndScaleTmAttrs(TableDataSet inTgTds, int[][] tdzTrips, - float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("transferring ABM home-based productions and scaling TM attractions:"); - logger.info(String.format("%10s %-30s %15s %15s %15s %15s %15s", "TM Heading", - "TM Purpose", "TM Attrs", "ABM Attrs", "Scale Factor", "New ABM Prods", - "New ABM Attrs")); - int index = 0; - for (String heading : TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS) - { - - // get the trip model attractions, the TableDataSet returns a 0s - // based - // array - float[] values = inTgTds.getColumnAsFloat(heading); - - // add up the total TM attractions for this purpose - float total = 0; - for (int i = 0; i < values.length; i++) - total += values[i]; - - // add up the total ABM productions for this purpose - int abTotal = 0; - int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; - for (int i = 1; i < tdzTrips.length; i++) - abTotal += tdzTrips[i][abProdIndex]; - - // get the scale factor to scale trip model atractions to ABM - // productions - // by purpose - double scaleFactor = 0.0; - if (total > 0) - { - scaleFactor = abTotal / total; - } else - { - logger.error("attempting to scale an array which sums to 0.0."); - throw new RuntimeException(); - } - - // get the scaled attractions for the purpose - double[] scaledAttrs = getScaledValues(values, scaleFactor); - - // determine the final array column index into which to store the - // scaled - // attractions - int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; - - // save the scaled attractions in the final array - float abmAttrs = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][abAttrIndex] = (float) scaledAttrs[i]; - abmAttrs += newTrips[i][abAttrIndex]; - } - - // save the ABM productions in the final array - float abmProds = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][abProdIndex] = tdzTrips[i + 1][abProdIndex]; - abmProds += newTrips[i][abProdIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f %15.1f", - heading, TABLE_HEADING_DESCRIPTIONS[abAttrIndex], total, abTotal, scaleFactor, - abmProds, abmAttrs)); - - index++; - - } - } - - private void saveTmNhbProdsAsAbmProds(TableDataSet inTgTds, int[][] tdzTrips, - boolean scaleNhbToAbm, float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("non-home-based TM productions to total ABM productions:"); - logger.info(String.format("%10s %-30s %15s %15s %15s %15s", "TM Heading", - "TM Purpose", "TM Prods", "ABM Prods", "Scale Factor", "New ABM Prods")); - int index = 0; - for (String heading : TRIP_MODEL_NON_HOME_BASED_PRODUCTION_HEADINGS) - { - - // get the trip model nhb productions, the TableDataSet returns a 0s - // based array - float[] values = inTgTds.getColumnAsFloat(heading); - - // add up total TM productions by purpose - float total = 0; - for (int i = 0; i < values.length; i++) - total += values[i]; - - // get the total ab model productions for this purpose - int abProdIndex = AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES[index]; - int abTotal = tdzTrips[0][abProdIndex]; - - // get the scale factor to scale trip model productions to ABM - // productions by purpose - double scaleFactor = 0.0; - if (scaleNhbToAbm) - { - if (total > 0) - { - scaleFactor = abTotal / total; - } else - { - logger.error("attempting to scale an array which sums to 0.0."); - throw new RuntimeException(); - } - } else - { - scaleFactor = 1.0; - } - - // get the scaled productions for the purpose - double[] scaledProds = getScaledValues(values, scaleFactor); - - // save the scaled attractions in the final array - float abmProds = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][abProdIndex] = (float) scaledProds[i]; - abmProds += newTrips[i][abProdIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[abProdIndex], total, abTotal, scaleFactor, abmProds)); - - index++; - - } - } - - private void saveTmNhbAttrsAsAbmProds(TableDataSet inTgTds, int[][] tdzTrips, - boolean scaleNhbToAbm, float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("non-home-based TM attractions to total ABM productions:"); - logger.info(String.format("%10s %-30s %15s %15s %15s %15s", "TM Heading", - "TM Purpose", "TM Attrs", "ABM Prods", "Scale Factor", "New ABM Attrs")); - int index = 0; - for (String heading : TRIP_MODEL_NON_HOME_BASED_ATTRACTION_HEADINGS) - { - - // get the trip model nhb attractions, the TableDataSet returns a 0s - // based array - float[] values = inTgTds.getColumnAsFloat(heading); - - // add up total TM attracctions by purpose - float total = 0; - for (int i = 0; i < values.length; i++) - total += values[i]; - - // get the total ab model productions for this purpose - int abProdIndex = AB_MODEL_NON_HOME_BASED_PRODUCTION_INDICES[index]; - int abTotal = tdzTrips[0][abProdIndex]; - - // get the scale factor to scale trip model atractions to ABM - // productions - // by purpose - double scaleFactor = 0.0; - if (scaleNhbToAbm) - { - if (total > 0) - { - scaleFactor = abTotal / total; - } else - { - logger.error("attempting to scale an array which sums to 0.0."); - throw new RuntimeException(); - } - } else - { - scaleFactor = 1.0; - } - - // get the scaled attractions for the purpose - double[] scaledAttrs = getScaledValues(values, scaleFactor); - - // determine the final array column index into which to store the - // scaled - // attractions - int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; - - // save the scaled attractions in the final array - float abmAttrs = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][abAttrIndex] = (float) scaledAttrs[i]; - abmAttrs += newTrips[i][abAttrIndex]; - } - - index++; - - logger.info(String.format("%10s %-30s %15.1f %15d %15.6f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[abAttrIndex], total, abTotal, scaleFactor, abmAttrs)); - - } - } - - private void addTmIeAttrsToAbmHbProds(TableDataSet inTgTds, int[][] tdzTrips, float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("adding IE attrs from trip model to home-based ABM prods:"); - logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", - "ABM Prods", "IE attrs", "new ABM Prods")); - int index = 0; - for (String heading : TRIP_MODEL_HOME_BASED_PRODUCTION_HEADINGS) - { - - int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; - int abAttrIndex = abProdIndex + MAX_PURPOSE_INDEX; - - // save the new ABM productions in the final array - float abTotal = 0; - float newTotal = 0; - for (int i = 0; i < newTrips.length; i++) - { - abTotal += newTrips[i][abProdIndex]; - newTrips[i][abProdIndex] += ieTrips[i + 1][abAttrIndex]; - newTotal += newTrips[i][abProdIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15.1f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[abProdIndex], abTotal, ieTrips[0][abAttrIndex], - newTotal)); - - index++; - - } - } - - private void addTmIeProdsToAbmHbAttrs(TableDataSet inTgTds, int[][] tdzTrips, float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("adding IE prods from trip model to home-based ABM attrs:"); - logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", - "ABM Attrs", "IE prods", "new ABM Attrs")); - int index = 0; - for (String heading : TRIP_MODEL_HOME_BASED_ATTRACTION_HEADINGS) - { - - int abProdIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index]; - int abAttrIndex = AB_MODEL_HOME_BASED_PRODUCTION_INDICES[index] + MAX_PURPOSE_INDEX; - - // save the new ABM attractions in the final array - float abTotal = 0; - float newTotal = 0; - for (int i = 0; i < newTrips.length; i++) - { - abTotal += newTrips[i][abAttrIndex]; - newTrips[i][abAttrIndex] += ieTrips[i + 1][abProdIndex]; - newTotal += newTrips[i][abAttrIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15.1f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[abAttrIndex], abTotal, ieTrips[0][abProdIndex], - newTotal)); - - index++; - - } - } - - private void addTmAirportAndVisitorProdsToAbmProds(TableDataSet inTgTds, int[][] tdzTrips, - float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("airport and visitor TM to ABM productions:"); - logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", - "TM Prods", "Scale Factor", "New ABM Prods")); - int index = 0; - for (String heading : TRIP_MODEL_OTHER_BASED_PRODUCTION_HEADINGS) - { - - // get the trip model productions - float[] pValues = inTgTds.getColumnAsFloat(heading); - - // determine the final array column index into which to store the - // scaled - // attractions - int prodIndex = AB_MODEL_OTHER_PRODUCTION_INDICES[index]; - - // save the productions in the final array - // add up the original values - float pTotal = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][prodIndex] = pValues[i]; - pTotal += newTrips[i][prodIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15.6f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[prodIndex], pTotal, 1.0, pTotal)); - - index++; - - } - } - - private void addTmAirportAndVisitorAttrsToAbmAttrs(TableDataSet inTgTds, int[][] tdzTrips, - float[][] newTrips) - { - logger.info(""); - logger.info(""); - logger.info("airport and visitor TM to ABM attractions:"); - logger.info(String.format("%10s %-30s %15s %15s %15s", "TM Heading", "TM Purpose", - "TM Attrs", "Scale Factor", "New ABM Attrs")); - int index = 0; - for (String heading : TRIP_MODEL_OTHER_BASED_ATTRACTION_HEADINGS) - { - - // get the trip model attractions - float[] aValues = inTgTds.getColumnAsFloat(heading); - - // determine the final array column index into which to store the - // scaled - // attractions - int attrIndex = AB_MODEL_OTHER_PRODUCTION_INDICES[index] + MAX_PURPOSE_INDEX; - - // save the attractions in the final array - float aTotal = 0; - for (int i = 0; i < newTrips.length; i++) - { - newTrips[i][attrIndex] = aValues[i]; - aTotal += newTrips[i][attrIndex]; - } - - logger.info(String.format("%10s %-30s %15.1f %15.6f %15.1f", heading, - TABLE_HEADING_DESCRIPTIONS[attrIndex], aTotal, 1.0, aTotal)); - - index++; - - } - } - - private void saveZoneField(TableDataSet inTgTds, float[][] newTrips) - { - // save the zone field in final table - float[] values = inTgTds.getColumnAsFloat(TAZ_FIELD_HEADING); - int tazFieldIndex = inTgTds.getColumnPosition(TAZ_FIELD_HEADING) - 1; - for (int i = 0; i < newTrips.length; i++) - newTrips[i][tazFieldIndex] = values[i]; - } - - private TableDataSet createFinalTableDataset(float[][] newTrips) - { - TableDataSet abmTds = TableDataSet.create(newTrips, TABLE_HEADINGS); - - float[] newIeTrips = new float[2 * MAX_PURPOSE_INDEX + 1]; - for (int i = 0; i < abmTds.getRowCount(); i++) - { - int tdz = (int) abmTds.getValueAt(i + 1, TAZ_FIELD_HEADING); - for (int j = 1; j < abmTds.getColumnCount(); j++) - { - if (tdz >= MIN_EXTERNAL_TDZ && tdz <= MAX_EXTERNAL_TDZ) - { - newIeTrips[j] += abmTds.getValueAt(i + 1, j + 1); - } - } - } - - logger.info(""); - logger.info(""); - logger.info("summary of newly created trip generation file."); - logger.info("\t" + abmTds.getRowCount() + " rows in output file."); - logger.info("\t" + abmTds.getColumnCount() + " columns in output file."); - logger.info(""); - logger.info(String.format("\t%-15s %-30s %15s %15s", "Column Name", "Column Purpose", - "Column Total", "Int-Ext")); - - String[] headings = abmTds.getColumnLabels(); - logger.info(String.format("\t%-15s %-30s %15s %15s", headings[0], "N/A", "N/A", "N/A")); - float totProd = 0; - float totAttr = 0; - float totalIeProds = 0; - float totalIeAttrs = 0; - float columnSum = 0; - for (int i = 1; i < abmTds.getColumnCount(); i++) - { - - columnSum = abmTds.getColumnTotal(i + 1); - - // 1st 10 fields after zone are production fields, next 10 are - // attraction - // fields - if (i <= 10) - { - totProd += columnSum; - totalIeProds += newIeTrips[i]; - } else - { - totAttr += columnSum; - totalIeAttrs += newIeTrips[i]; - } - - logger.info(String.format("\t%-15s %-30s %15.1f %15.1f", headings[i], - TABLE_HEADING_DESCRIPTIONS[i], columnSum, newIeTrips[i])); - - } - - logger.info(""); - logger.info(""); - logger.info(String.format("\ttotal productions = %15.1f", totProd)); - logger.info(String.format("\ttotal attractions = %15.1f", totAttr)); - logger.info(String.format("\ttotal IE productions = %12.1f", totalIeProds)); - logger.info(String.format("\ttotal IE attractions = %12.1f", totalIeAttrs)); - logger.info(""); - - return abmTds; - } - - private void writeAbmTripGenFile(String tgOutputFile, TableDataSet outAbmTds) - { - - CSVFileWriter writer = new CSVFileWriter(); - try - { - writer.writeFile(outAbmTds, new File(tgOutputFile)); - } catch (IOException e) - { - logger.fatal(String - .format("Exception occurred writing new trip generation data file = %s from TableDataSet object.", - tgOutputFile)); - throw new RuntimeException(e); - } - } - - private double[] getScaledValues(float[] values, double scaleFactor) - { - - double[] scaledValues = new double[values.length]; - for (int i = 0; i < values.length; i++) - scaledValues[i] = values[i] * scaleFactor; - - return scaledValues; - } - - public static void main(String[] args) throws Exception - { - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else - { - - String baseName; - if (args[0].endsWith(".properties")) - { - int index = args[0].indexOf(".properties"); - baseName = args[0].substring(0, index); - } else - { - baseName = args[0]; - } - - ResourceBundle rb = ResourceBundle.getBundle(baseName); - HashMap rbMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - SandagCreateTripGenerationFiles mainObject = new SandagCreateTripGenerationFiles(rbMap); - - // pass true as an argument if NHB trips from the trip model are to - // be - // scaled to the number from the activity-based model - mainObject.createTripGenFile(rbMap); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java deleted file mode 100644 index 8dad984..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampApplication.java +++ /dev/null @@ -1,23 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import java.util.ResourceBundle; -import org.sandag.abm.ctramp.CtrampApplication; - -public class SandagCtrampApplication - extends CtrampApplication -{ - - public static final String PROGRAM_VERSION = "09June2008"; - public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; - - public SandagCtrampApplication(ResourceBundle rb, HashMap rbMap, - boolean calculateLandUseAccessibilities) - { - super(rb, rbMap, calculateLandUseAccessibilities); - - projectDirectory = rbMap.get(PROPERTIES_PROJECT_DIRECTORY); - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java deleted file mode 100644 index c473901..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagCtrampDmuFactory.java +++ /dev/null @@ -1,170 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.application; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Map; - -import org.sandag.abm.ctramp.AtWorkSubtourFrequencyDMU; -import org.sandag.abm.ctramp.AutoOwnershipChoiceDMU; -import org.sandag.abm.ctramp.BikeLogsum; -import org.sandag.abm.ctramp.CoordinatedDailyActivityPatternDMU; -import org.sandag.abm.ctramp.CtrampDmuFactoryIf; -import org.sandag.abm.ctramp.DcSoaDMU; -import org.sandag.abm.ctramp.DestChoiceDMU; -import org.sandag.abm.ctramp.DestChoiceTwoStageModelDMU; -import org.sandag.abm.ctramp.DestChoiceTwoStageSoaTazDistanceUtilityDMU; -import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyDMU; -import org.sandag.abm.ctramp.IndividualNonMandatoryTourFrequencyDMU; -import org.sandag.abm.ctramp.InternalExternalTripChoiceDMU; -import org.sandag.abm.ctramp.JointTourModelsDMU; -import org.sandag.abm.ctramp.MicromobilityChoiceDMU; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.ParkingChoiceDMU; -import org.sandag.abm.ctramp.ParkingProvisionChoiceDMU; -import org.sandag.abm.ctramp.StopFrequencyDMU; -import org.sandag.abm.ctramp.StopLocationDMU; -import org.sandag.abm.ctramp.TelecommuteDMU; -import org.sandag.abm.ctramp.TourDepartureTimeAndDurationDMU; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import org.sandag.abm.ctramp.TransponderChoiceDMU; -import org.sandag.abm.ctramp.TripModeChoiceDMU; - -/** - * ArcCtrampDmuFactory is a class that ... - * - * @author Kimberly Grommes - * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. - */ -public class SandagCtrampDmuFactory - implements CtrampDmuFactoryIf, Serializable -{ - - private ModelStructure modelStructure; - private Map propertyMap; - - public SandagCtrampDmuFactory(ModelStructure modelStructure, Map propertyMap) - { - this.modelStructure = modelStructure; - this.propertyMap = propertyMap; - } - - public AutoOwnershipChoiceDMU getAutoOwnershipDMU() - { - return new SandagAutoOwnershipChoiceDMU(); - } - - public TransponderChoiceDMU getTransponderChoiceDMU() - { - return new SandagTransponderChoiceDMU(); - } - - public TelecommuteDMU getTelecommuteDMU() - { - return new SandagTelecommuteDMU(); - } - - public InternalExternalTripChoiceDMU getInternalExternalTripChoiceDMU() - { - return new SandagInternalExternalTripChoiceDMU(); - } - - public ParkingProvisionChoiceDMU getFreeParkingChoiceDMU() - { - return new SandagParkingProvisionChoiceDMU(); - } - - public CoordinatedDailyActivityPatternDMU getCoordinatedDailyActivityPatternDMU() - { - return new SandagCoordinatedDailyActivityPatternDMU(); - } - - public DcSoaDMU getDcSoaDMU() - { - return new SandagDcSoaDMU(); - } - - public DestChoiceDMU getDestChoiceDMU() - { - return new SandagDestChoiceDMU(modelStructure); - } - - public DestChoiceTwoStageModelDMU getDestChoiceSoaTwoStageDMU() - { - return new SandagDestChoiceSoaTwoStageModelDMU(modelStructure); - } - - public DestChoiceTwoStageSoaTazDistanceUtilityDMU getDestChoiceSoaTwoStageTazDistUtilityDMU() - { - return new SandagDestChoiceSoaTwoStageTazDistUtilityDMU(); - } - - public TourModeChoiceDMU getModeChoiceDMU() - { - SandagTourModeChoiceDMU dmu = new SandagTourModeChoiceDMU(modelStructure,null); - dmu.setBikeLogsum(BikeLogsum.getBikeLogsum(propertyMap)); - return dmu; - } - - public IndividualMandatoryTourFrequencyDMU getIndividualMandatoryTourFrequencyDMU() - { - return new SandagIndividualMandatoryTourFrequencyDMU(); - } - - public TourDepartureTimeAndDurationDMU getTourDepartureTimeAndDurationDMU() - { - return new SandagTourDepartureTimeAndDurationDMU(modelStructure); - } - - public AtWorkSubtourFrequencyDMU getAtWorkSubtourFrequencyDMU() - { - return new SandagAtWorkSubtourFrequencyDMU(modelStructure); - } - - public JointTourModelsDMU getJointTourModelsDMU() - { - return new SandagJointTourModelsDMU(modelStructure); - } - - public IndividualNonMandatoryTourFrequencyDMU getIndividualNonMandatoryTourFrequencyDMU() - { - return new SandagIndividualNonMandatoryTourFrequencyDMU(); - } - - public StopFrequencyDMU getStopFrequencyDMU() - { - return new SandagStopFrequencyDMU(modelStructure); - } - - public StopLocationDMU getStopLocationDMU() - { - return new SandagStopLocationDMU(modelStructure,propertyMap); - } - - public TripModeChoiceDMU getTripModeChoiceDMU() - { - SandagTripModeChoiceDMU dmu = new SandagTripModeChoiceDMU(modelStructure,null); - dmu.setBikeLogsum(BikeLogsum.getBikeLogsum(propertyMap)); - return dmu; - } - - public ParkingChoiceDMU getParkingChoiceDMU() - { - return new SandagParkingChoiceDMU(); - } - - public MicromobilityChoiceDMU getMicromobilityChoiceDMU() - { - return new SandagMicromobilityChoiceDMU(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java deleted file mode 100644 index 8e17074..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDcSoaDMU.java +++ /dev/null @@ -1,89 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.DcSoaDMU; - -public class SandagDcSoaDMU - extends DcSoaDMU -{ - - public SandagDcSoaDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getLnDcSizeAlt", 0); - methodIndexMap.put("getOriginToMgraDistanceAlt", 1); - methodIndexMap.put("getTourPurposeIsEscort", 2); - methodIndexMap.put("getNumPreschool", 3); - methodIndexMap.put("getNumGradeSchoolStudents", 4); - methodIndexMap.put("getNumHighSchoolStudents", 5); - methodIndexMap.put("getDcSizeAlt", 6); - methodIndexMap.put("getHouseholdsDestAlt", 8); - methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 9); - methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 10); - methodIndexMap.put("getGradeSchoolDistrictDestAlt", 11); - methodIndexMap.put("getHomeMgraGradeSchoolDistrict", 12); - methodIndexMap.put("getHighSchoolDistrictDestAlt", 14); - methodIndexMap.put("getHomeMgraHighSchoolDistrict", 15); - methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); - //methodIndexMap.put("getHomeMgra", 17); - - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - public double getLnDcSizeAlt(int alt) - { - return getLnDcSize(alt); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getLnDcSizeAlt(arrayIndex); - case 1: - return getOriginToMgraDistanceAlt(arrayIndex); - case 2: - return getTourPurposeIsEscort(); - case 3: - return getNumPreschool(); - case 4: - return getNumGradeSchoolStudents(); - case 5: - return getNumHighSchoolStudents(); - case 6: - return getDcSizeAlt(arrayIndex); - case 8: - return getHouseholdsDestAlt(arrayIndex); - case 9: - return getGradeSchoolEnrollmentDestAlt(arrayIndex); - case 10: - return getHighSchoolEnrollmentDestAlt(arrayIndex); - case 11: - return getGradeSchoolDistrictDestAlt(arrayIndex); - case 12: - return getHomeMgraGradeSchoolDistrict(); - case 14: - return getHighSchoolDistrictDestAlt(arrayIndex); - case 15: - return getHomeMgraHighSchoolDistrict(); - case 16: - return getUniversityEnrollmentDestAlt(arrayIndex); - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java deleted file mode 100644 index 163c6b6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceDMU.java +++ /dev/null @@ -1,161 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.DestChoiceDMU; -import org.sandag.abm.ctramp.ModelStructure; - -public class SandagDestChoiceDMU - extends DestChoiceDMU -{ - - public SandagDestChoiceDMU(ModelStructure modelStructure) - { - super(modelStructure); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getMcLogsumDestAlt", 3); - methodIndexMap.put("getNumGradeSchoolStudents", 4); - methodIndexMap.put("getNumHighSchoolStudents", 5); - methodIndexMap.put("getHouseholdsDestAlt", 8); - methodIndexMap.put("getPopulationDestAlt", 9); - methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 10); - methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 11); - methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); - methodIndexMap.put("getPersonIsWorker", 20); - methodIndexMap.put("getPersonHasBachelors", 21); - methodIndexMap.put("getPersonType", 22); - methodIndexMap.put("getSubtourType", 23); - methodIndexMap.put("getDcSoaCorrectionsAlt", 24); - methodIndexMap.put("getNumberOfNonWorkingAdults", 25); - methodIndexMap.put("getNumPreschool", 26); - methodIndexMap.put("getFemale", 27); - methodIndexMap.put("getIncome", 28); - methodIndexMap.put("getFemaleWorker", 29); - methodIndexMap.put("getIncomeInDollars", 30); - methodIndexMap.put("getAutos", 31); - methodIndexMap.put("getWorkers", 32); - methodIndexMap.put("getNumChildrenUnder16", 33); - methodIndexMap.put("getNumChildrenUnder19", 34); - methodIndexMap.put("getAge", 35); - methodIndexMap.put("getFullTimeWorker", 36); - methodIndexMap.put("getWorkTaz", 37); - methodIndexMap.put("getWorkTourModeIsSOV", 38); - methodIndexMap.put("getTourIsJoint", 39); - methodIndexMap.put("getOpSovDistanceAlt", 42); - methodIndexMap.put("getLnDcSizeAlt", 43); - methodIndexMap.put("getWorkAccessibility", 44); - methodIndexMap.put("getNonMandatoryAccessibilityAlt", 45); - methodIndexMap.put("getToursLeft", 46); - methodIndexMap.put("getMaxWindow", 47); - methodIndexMap.put("getDcSizeAlt", 48); - } - - public void setMcLogsum(int mgra, double logsum) - { - modeChoiceLogsums[mgra] = logsum; - } - - public double getLogsumDestAlt(int alt) - { - return getMcLogsumDestAlt(alt); - } - - public int getPersonIsFullTimeWorker() - { - return person.getPersonIsFullTimeWorker(); - } - - /* - * public int getSubtourType() { if ( - * tour.getTourCategory().equalsIgnoreCase( ModelStructure.AT_WORK_CATEGORY - * ) ) return tour.getTourPurposeIndex(); else return 0; } - */ - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 3: - return getMcLogsumDestAlt(arrayIndex); - case 4: - return getNumGradeSchoolStudents(); - case 5: - return getNumHighSchoolStudents(); - case 8: - return getHouseholdsDestAlt(arrayIndex); - case 9: - return getPopulationDestAlt(arrayIndex); - case 10: - return getGradeSchoolEnrollmentDestAlt(arrayIndex); - case 11: - return getHighSchoolEnrollmentDestAlt(arrayIndex); - case 16: - return getUniversityEnrollmentDestAlt(arrayIndex); - case 20: - return getPersonIsWorker(); - case 21: - return getPersonHasBachelors(); - case 22: - return getPersonType(); - case 24: - return getDcSoaCorrectionsAlt(arrayIndex); - case 25: - return getNumberOfNonWorkingAdults(); - case 26: - return getNumPreschool(); - case 27: - return getFemale(); - case 28: - return getIncome(); - case 29: - return getFemaleWorker(); - case 30: - return getIncomeInDollars(); - case 31: - return getAutos(); - case 32: - return getWorkers(); - case 33: - return getNumChildrenUnder16(); - case 34: - return getNumChildrenUnder19(); - case 35: - return getAge(); - case 36: - return getFullTimeWorker(); - case 37: - return getWorkTaz(); - case 38: - return getWorkTourModeIsSOV(); - case 39: - return getTourIsJoint(); - case 42: - return getOpSovDistanceAlt(arrayIndex); - case 43: - return getLnDcSizeAlt(arrayIndex); - case 44: - return getWorkAccessibility(); - case 45: - return getNonMandatoryAccessibilityAlt(arrayIndex); - case 46: - return getToursLeftCount(); - case 47: - return getMaxContinuousAvailableWindow(); - case 48: - return getDcSizeAlt(arrayIndex); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java deleted file mode 100644 index aae0533..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageModelDMU.java +++ /dev/null @@ -1,158 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.DestChoiceTwoStageModelDMU; -import org.sandag.abm.ctramp.ModelStructure; - -public class SandagDestChoiceSoaTwoStageModelDMU - extends DestChoiceTwoStageModelDMU -{ - - public SandagDestChoiceSoaTwoStageModelDMU(ModelStructure modelStructure) - { - super(modelStructure); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getMcLogsumDestAlt", 3); - methodIndexMap.put("getNumGradeSchoolStudents", 4); - methodIndexMap.put("getNumHighSchoolStudents", 5); - methodIndexMap.put("getHouseholdsDestAlt", 8); - methodIndexMap.put("getPopulationDestAlt", 9); - methodIndexMap.put("getGradeSchoolEnrollmentDestAlt", 10); - methodIndexMap.put("getHighSchoolEnrollmentDestAlt", 11); - methodIndexMap.put("getUniversityEnrollmentDestAlt", 16); - methodIndexMap.put("getPersonIsWorker", 20); - methodIndexMap.put("getPersonHasBachelors", 21); - methodIndexMap.put("getPersonType", 22); - methodIndexMap.put("getSubtourType", 23); - methodIndexMap.put("getDcSoaCorrectionsAlt", 24); - methodIndexMap.put("getNumberOfNonWorkingAdults", 25); - methodIndexMap.put("getNumPreschool", 26); - methodIndexMap.put("getFemale", 27); - methodIndexMap.put("getIncome", 28); - methodIndexMap.put("getFemaleWorker", 29); - methodIndexMap.put("getIncomeInDollars", 30); - methodIndexMap.put("getAutos", 31); - methodIndexMap.put("getWorkers", 32); - methodIndexMap.put("getNumChildrenUnder16", 33); - methodIndexMap.put("getNumChildrenUnder19", 34); - methodIndexMap.put("getAge", 35); - methodIndexMap.put("getFullTimeWorker", 36); - methodIndexMap.put("getWorkTaz", 37); - methodIndexMap.put("getWorkTourModeIsSOV", 38); - methodIndexMap.put("getTourIsJoint", 39); - methodIndexMap.put("getOpSovDistanceAlt", 42); - methodIndexMap.put("getLnDcSizeAlt", 43); - methodIndexMap.put("getWorkAccessibility", 44); - methodIndexMap.put("getNonMandatoryAccessibilityAlt", 45); - methodIndexMap.put("getToursLeft", 46); - methodIndexMap.put("getMaxWindow", 47); - } - - public void setMcLogsum(int sampleIndex, double logsum) - { - modeChoiceLogsums[sampleIndex] = logsum; - } - - public double getLogsumDestAlt(int alt) - { - return getMcLogsumDestAlt(alt); - } - - public int getPersonIsFullTimeWorker() - { - return person.getPersonIsFullTimeWorker(); - } - - /* - * public int getSubtourType() { if ( - * tour.getTourCategory().equalsIgnoreCase( ModelStructure.AT_WORK_CATEGORY - * ) ) return tour.getTourPurposeIndex(); else return 0; } - */ - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 3: - return getMcLogsumDestAlt(arrayIndex); - case 4: - return getNumGradeSchoolStudents(); - case 5: - return getNumHighSchoolStudents(); - case 8: - return getHouseholdsDestAlt(arrayIndex); - case 9: - return getPopulationDestAlt(arrayIndex); - case 10: - return getGradeSchoolEnrollmentDestAlt(arrayIndex); - case 11: - return getHighSchoolEnrollmentDestAlt(arrayIndex); - case 16: - return getUniversityEnrollmentDestAlt(arrayIndex); - case 20: - return getPersonIsWorker(); - case 21: - return getPersonHasBachelors(); - case 22: - return getPersonType(); - case 24: - return getDcSoaCorrectionsAlt(arrayIndex); - case 25: - return getNumberOfNonWorkingAdults(); - case 26: - return getNumPreschool(); - case 27: - return getFemale(); - case 28: - return getIncome(); - case 29: - return getFemaleWorker(); - case 30: - return getIncomeInDollars(); - case 31: - return getAutos(); - case 32: - return getWorkers(); - case 33: - return getNumChildrenUnder16(); - case 34: - return getNumChildrenUnder19(); - case 35: - return getAge(); - case 36: - return getFullTimeWorker(); - case 37: - return getWorkTaz(); - case 38: - return getWorkTourModeIsSOV(); - case 39: - return getTourIsJoint(); - case 42: - return getOpSovDistanceAlt(arrayIndex); - case 43: - return getLnDcSizeAlt(arrayIndex); - case 44: - return getWorkAccessibility(); - case 45: - return getNonMandatoryAccessibilityAlt(arrayIndex); - case 46: - return getToursLeftCount(); - case 47: - return getMaxContinuousAvailableWindow(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java deleted file mode 100644 index 62e1688..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagDestChoiceSoaTwoStageTazDistUtilityDMU.java +++ /dev/null @@ -1,66 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.DestChoiceTwoStageSoaTazDistanceUtilityDMU; - -public class SandagDestChoiceSoaTwoStageTazDistUtilityDMU - extends DestChoiceTwoStageSoaTazDistanceUtilityDMU -{ - - public SandagDestChoiceSoaTwoStageTazDistUtilityDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getLnDestChoiceSizeTazAlt", 0); - methodIndexMap.put("getSizeTazAlt", 1); - methodIndexMap.put("getUniversityEnrollmentTazAlt", 2); - methodIndexMap.put("getGradeSchoolDistrictTazAlt", 3); - methodIndexMap.put("getHighSchoolDistrictTazAlt", 4); - methodIndexMap.put("getHomeTazGradeSchoolDistrict", 5); - methodIndexMap.put("getHomeTazHighSchoolDistrict", 6); - methodIndexMap.put("getGradeSchoolEnrollmentTazAlt", 7); - methodIndexMap.put("getHighSchoolEnrollmentTazAlt", 8); - methodIndexMap.put("getHouseholdsTazAlt", 9); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getLnDestChoiceSizeTazAlt(arrayIndex); - case 1: - return getSizeTazAlt(arrayIndex); - case 2: - return getUniversityEnrollmentTazAlt(arrayIndex); - case 3: - return getGradeSchoolDistrictTazAlt(arrayIndex); - case 4: - return getHighSchoolDistrictTazAlt(arrayIndex); - case 5: - return getHomeTazGradeSchoolDistrict(); - case 6: - return getHomeTazHighSchoolDistrict(); - case 7: - return getGradeSchoolEnrollmentTazAlt(arrayIndex); - case 8: - return getHighSchoolEnrollmentTazAlt(arrayIndex); - case 9: - return getHouseholdsTazAlt(arrayIndex); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java deleted file mode 100644 index 805df60..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager.java +++ /dev/null @@ -1,624 +0,0 @@ -package org.sandag.abm.application; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; -import java.io.BufferedReader; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileReader; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.HouseholdDataManager; -import org.sandag.abm.ctramp.Person; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * @author Jim Hicks - * - * Class for managing household and person object data read from - * synthetic population files. - */ -public class SandagHouseholdDataManager - extends HouseholdDataManager -{ - - public static final String HH_DATA_SERVER_NAME = SandagHouseholdDataManager.class - .getCanonicalName(); - public static final String HH_DATA_SERVER_ADDRESS = "127.0.0.1"; - public static final int HH_DATA_SERVER_PORT = 1139; - - public static final String PROPERTIES_OCCUP_CODES = "PopulationSynthesizer.OccupCodes"; - public static final String PROPERTIES_INDUSTRY_CODES = "PopulationSynthesizer.IndustryCodes"; - - public SandagHouseholdDataManager() - { - super(); - } - - /** - * Associate data in hh and person TableDataSets read from synthetic - * population files with Household objects and Person objects with - * Households. - * - */ - public void mapTablesToHouseholdObjects() - { - - logger.info("mapping popsyn household and person data records to objects."); - - int id = -1; - Household[] hhArray = new Household[hhTable.getRowCount()]; - - int invalidPersonTypeCount1 = 0; - int invalidPersonTypeCount2 = 0; - int invalidPersonTypeCount3 = 0; - - // read the corrrespondence files for mapping persons to occupation and - int[] occCodes = readOccupCorrespondenceData(); - int[] indCodes = readIndustryCorrespondenceData(); - - // get the maximum HH id value to use to dimension the hhIndex - // correspondence - // array. - // the hhIndex array will store the hhArray index number for the given - // hh - // index. - int maxHhId = 0; - for (int r = 1; r <= hhTable.getRowCount(); r++) - { - id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); - if (id > maxHhId) maxHhId = id; - } - hhIndexArray = new int[maxHhId + 1]; - int[] sortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); - - // for each household table record - for (int r = 1; r <= hhTable.getRowCount(); r++) - { - - try - { - - // create a Household object - Household hh = new Household(modelStructure); - - // get required values from table record and store in Household - // object - id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); - hh.setHhId(id, inputRandomSeed); - - // set the household in the hhIndexArray in random order - int index = sortedIndices[r - 1]; - hhIndexArray[hh.getHhId()] = index; - - int htaz = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); - hh.setHhTaz(htaz); - - int hmgra = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); - hh.setHhMgra(hmgra); - - double rn = hh.getHhRandom().nextDouble(); - int origWalkSubzone = getInitialOriginWalkSegment(htaz, rn); - hh.setHhWalkSubzone(origWalkSubzone); - - // autos could be modeled or from PUMA - int numAutos = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_AUTOS_FIELD_NAME)); - hh.setHhAutos(numAutos); - - // set the hhSize variable and create Person objects for each - // person - int numPersons = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_SIZE_FIELD_NAME)); - hh.setHhSize(numPersons); - - int numWorkers = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_WORKERS_FIELD_NAME)); - hh.setHhWorkers(numWorkers); - - int incomeCat = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_INCOME_CATEGORY_FIELD_NAME)); - hh.setHhIncomeCategory(incomeCat); - - int incomeInDollars = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_INCOME_DOLLARS_FIELD_NAME)); - hh.setHhIncomeInDollars(incomeInDollars); - - // 0=Housing unit, 1=Institutional group quarters, - // 2=Noninstitutional - // group quarters - int unitType = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_UNITTYPE_FIELD_NAME)); - hh.setUnitType(unitType); - - // 1=Family household:married-couple, 2=Family household:male - // householder,no wife present, 3=Family household:female - // householder,no - // husband present - // 4=Nonfamily household:male householder, living alone, - // 5=Nonfamily - // household:male householder, not living alone, - // 6=Nonfamily household:female householder, living alone, - // 7=Nonfamily household:female householder, not living alone - int type = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_TYPE_FIELD_NAME)); - hh.setHhType(type); - - // 1=mobile home, 2=one-family house detached from any other - // house, - // 3=one-family house attached to one or more houses, - // 4=building with 2 apartments, 5=building with 3 or 4 - // apartments, - // 6=building with 5 to 9 apartments, - // 7=building with 10 to 19 apartments, 8=building with 20 to 49 - // apartments, - // 9=building with 50 or more apartments, 10=Boat,RV,van,etc. - int bldgsz = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_BLDGSZ_FIELD_NAME)); - hh.setHhBldgsz(bldgsz); - - hh.initializeWindows(); - hhArray[index] = hh; - - } catch (Exception e) - { - - logger.fatal(String - .format("exception caught mapping household data record to a Household object, r=%d, id=%d.", - r, id)); - throw new RuntimeException(e); - - } - - } - - int hhid = -1; - int oldHhid = -1; - int i = -1; - int persNum = -1; - int persId = -1; - int fieldCount = 0; - - // for each person table record - for (int r = 1; r <= personTable.getRowCount(); r++) - { - - try - { - - // get the Household object for this person data to be stored in - hhid = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); - int index = hhIndexArray[hhid]; - Household hh = hhArray[index]; - fieldCount = 1; - - if (oldHhid < hhid) - { - oldHhid = hhid; - persNum = 1; - } - - // get the Person object for this person data to be stored in - persId = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_PERSON_ID_FIELD_NAME)); - Person person = hh.getPerson(persNum++); - person.setPersId(persId); - fieldCount++; - - // get required values from table record and store in Person - // object - int age = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_AGE_FIELD_NAME)); - person.setPersAge(age); - fieldCount++; - - int gender = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_GENDER_FIELD_NAME)); - person.setPersGender(gender); - fieldCount++; - - int occcen1 = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_OCCCEN1_FIELD_NAME)); - int pecasOccup = occCodes[occcen1]; - - if (pecasOccup == 0) logger.warn("pecasOccup==0 for occcen1=" + occcen1); - - int indcen = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_INDCEN_FIELD_NAME)); - int activityCode = indCodes[indcen]; - - if ((pecasOccup == 71) - && (activityCode == 2 || activityCode == 4 || activityCode == 6 - || activityCode == 8 || activityCode == 29)) activityCode++; - - if ((pecasOccup == 76) - && (activityCode == 3 || activityCode == 5 || activityCode == 7 - || activityCode == 9 || activityCode == 30)) activityCode--; - - if ((pecasOccup == 76) && (activityCode == 13)) activityCode = 14; - - if ((pecasOccup == 71) && (activityCode == 14)) activityCode = 13; - - if ((pecasOccup == 75) && (activityCode == 18)) activityCode = 22; - - if ((pecasOccup == 71) && (activityCode == 22)) activityCode = 18; - - if (activityCode == 28) pecasOccup = 77; - - person.setPersActivityCode(activityCode); - fieldCount++; - - person.setPersPecasOccup(pecasOccup); - fieldCount++; - - // Employment status (1-employed FT, 2-employed PT, 3-not - // employed, - // 4-under age 16) - int empCat = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME)); - person.setPersEmploymentCategory(empCat); - fieldCount++; - - // Student status (1 - student in grade or high school; 2 - - // student - // in college or higher; 3 - not a student) - int studentCat = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_STUDENT_CATEGORY_FIELD_NAME)); - person.setPersStudentCategory(studentCat); - fieldCount++; - - // Person type (1-FT worker age 16+, 2-PT worker nonstudent age - // 16+, - // 3-university student, 4-nonworker nonstudent age 16-64, - // 5-nonworker nonstudent age 65+, - // 6-"age 16-19 student, not FT wrkr or univ stud", 7-age 6-15 - // schpred, 8 under age 6 presch - int personType = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_TYPE_CATEGORY_FIELD_NAME)); - person.setPersonTypeCategory(personType); - fieldCount++; - - // Person educational attainment level to determine high school - // graduate status ( < 9 - not a graduate, 10+ - high school - // graduate - // and - // beyond) - int educ = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_EDUCATION_ATTAINMENT_FIELD_NAME)); - if (educ >= 9) person.setPersonIsHighSchoolGraduate(true); - else person.setPersonIsHighSchoolGraduate(false); - fieldCount++; - - // Person educational attainment level to determine higher - // education - // status ( > 12 - at least a bachelor's degree ) - if (educ >= 13) person.setPersonHasBachelors(true); - else person.setPersonHasBachelors(false); - fieldCount++; - - // Person grade enrolled in ( 0-"not enrolled", 1-"preschool", - // 2-"Kindergarten", 3-"Grade 1 to grade 4", - // 4-"Grade 5 to grade 8", 5-"Grade 9 to grade 12", - // 6-"College undergraduate", - // 7-"Graduate or professional school" ) - int grade = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_GRADE_ENROLLED_FIELD_NAME)); - person.setPersonIsGradeSchool(false); - person.setPersonIsHighSchool(false); - if (grade >= 2 && grade <= 4) person.setPersonIsGradeSchool(true); - else if (grade == 5) person.setPersonIsHighSchool(true); - fieldCount++; - - // if person is a university student but has school age student - // category value, reset student category value - if (personType == Person.PersonType.University_student.ordinal() - && studentCat != Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal()) - { - studentCat = Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount1++; - // if person is a student of any kind but has full-time - // employment - // status, reset student category value to non-student - } else if (studentCat != Person.StudentStatus.NON_STUDENT.ordinal() - && empCat == Person.EmployStatus.FULL_TIME.ordinal()) - { - studentCat = Person.StudentStatus.NON_STUDENT.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount2++; - } - fieldCount++; - - // check consistency of student category and person type - if (studentCat == Person.StudentStatus.NON_STUDENT.ordinal()) - { - - if (person.getPersonIsStudentNonDriving() == 1 - || person.getPersonIsStudentDriving() == 1) - { - studentCat = Person.StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount3++; - } - - } - fieldCount++; - - } catch (Exception e) - { - - logger.fatal("exception caught mapping person data record to a Person object, " - + String.format( - "r=%d, i=%d, hhid=%d, persid=%d, persnum=%d, fieldCount=%d.", r, i, - hhid, persId, persNum, fieldCount)); - throw new RuntimeException(e); - - } - - } // person loop - - hhs = hhArray; - - logger.warn(invalidPersonTypeCount1 - + " person type = university and student category = non-student person records" - + " had their student category changed to university or higher."); - logger.warn(invalidPersonTypeCount2 - + " Student category = student and employment category = full-time worker person records" - + " had their student category changed to non-student."); - logger.warn(invalidPersonTypeCount3 - + " Student category = non-student and person type = student person records" - + " had their student category changed to student high school or less."); - - } - - /** - * if called, must be called after readData so that the size of the full - * population is known. - * - * @param hhFileName - * @param persFileName - * @param numHhs - */ - public void createSamplePopulationFiles(String hhFileName, String persFileName, - String newHhFileName, String newPersFileName, int numHhs) - { - - int maximumHhId = 0; - for (int i = 0; i < hhs.length; i++) - { - int id = hhs[i].getHhId(); - if (id > maximumHhId) maximumHhId = id; - } - - int[] testHhs = new int[maximumHhId + 1]; - - int[] sortedIndices = getRandomOrderHhIndexArray(hhs.length); - - for (int i = 0; i < numHhs; i++) - { - int k = sortedIndices[i]; - int hhId = hhs[k].getHhId(); - testHhs[hhId] = 1; - } - - String hString = ""; - int hCount = 0; - try - { - - logger.info(String.format("writing sample household file for %d households", numHhs)); - - PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newHhFileName))); - BufferedReader in = new BufferedReader(new FileReader(hhFileName)); - - // read headers and write to output files - hString = in.readLine(); - out.write(hString + "\n"); - hCount++; - int count = 0; - - while ((hString = in.readLine()) != null) - { - hCount++; - int endOfField = hString.indexOf(','); - int hhId = Integer.parseInt(hString.substring(0, endOfField)); - - // if it's a sample hh, write the hh and the person records - if (testHhs[hhId] == 1) - { - out.write(hString + "\n"); - count++; - if (count == numHhs) break; - } - } - - out.close(); - - } catch (IOException e) - { - logger.fatal("IO Exception caught creating sample synpop household file."); - logger.fatal(String.format("reading hh file = %s, writing sample hh file = %s.", - hhFileName, newHhFileName)); - logger.fatal(String.format("hString = %s, hCount = %d.", hString, hCount)); - } - - String pString = ""; - int pCount = 0; - try - { - - logger.info(String.format("writing sample person file for selected households")); - - PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newPersFileName))); - BufferedReader in = new BufferedReader(new FileReader(persFileName)); - - // read headers and write to output files - pString = in.readLine(); - out.write(pString + "\n"); - pCount++; - int count = 0; - int oldId = 0; - while ((pString = in.readLine()) != null) - { - pCount++; - int endOfField = pString.indexOf(','); - int hhId = Integer.parseInt(pString.substring(0, endOfField)); - - // if it's a sample hh, write the hh and the person records - if (testHhs[hhId] == 1) - { - out.write(pString + "\n"); - if (hhId > oldId) count++; - } else - { - if (count == numHhs) break; - } - - oldId = hhId; - - } - - out.close(); - - } catch (IOException e) - { - logger.fatal("IO Exception caught creating sample synpop person file."); - logger.fatal(String.format( - "reading person file = %s, writing sample person file = %s.", persFileName, - newPersFileName)); - logger.fatal(String.format("pString = %s, pCount = %d.", pString, pCount)); - } - - } - - public static void main(String[] args) throws Exception - { - - String serverAddress = HH_DATA_SERVER_ADDRESS; - int serverPort = HH_DATA_SERVER_PORT; - - // optional arguments - for (int i = 0; i < args.length; i++) - { - if (args[i].equalsIgnoreCase("-hostname")) - { - serverAddress = args[i + 1]; - } - - if (args[i].equalsIgnoreCase("-port")) - { - serverPort = Integer.parseInt(args[i + 1]); - } - } - - Remote.config(serverAddress, HH_DATA_SERVER_PORT, null, 0); - - SandagHouseholdDataManager hhDataManager = new SandagHouseholdDataManager(); - - ItemServer.bind(hhDataManager, HH_DATA_SERVER_NAME); - - System.out.println(String.format( - "SandagHouseholdDataManager server class started on: %s:%d", serverAddress, - serverPort)); - - } - - public int[] getJointToursByHomeMgra(String purposeString) - { - // TODO Auto-generated method stub - return null; - } - - private int[] readOccupCorrespondenceData() - { - - TableDataSet occTable = null; - - // construct input household file name from properties file values - String occupFileName = propertyMap.get(PROPERTIES_OCCUP_CODES); - String fileName = projectDirectory + "/" + occupFileName; - - try - { - logger.info("reading occupation codes data file for creating occupation segments."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - occTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String.format( - "Exception occurred occupation codes data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - // get the array of indices from the TableDataSet - int[] occcen1Col = occTable.getColumnAsInt("occcen1"); - int[] occupCol = occTable.getColumnAsInt("pecas_occ"); - - // get the max index value, to use for array dimensions - int maxOcc = 0; - for (int occ : occcen1Col) - if (occ > maxOcc) maxOcc = occ; - - int[] occcen1Occup = new int[maxOcc + 1]; - for (int i = 0; i < occcen1Col.length; i++) - { - int index = occcen1Col[i]; - int value = occupCol[i]; - occcen1Occup[index] = value; - } - - return occcen1Occup; - } - - private int[] readIndustryCorrespondenceData() - { - - TableDataSet indTable = null; - - // construct input household file name from properties file values - String indFileName = propertyMap.get(PROPERTIES_INDUSTRY_CODES); - String fileName = projectDirectory + "/" + indFileName; - - try - { - logger.info("reading industry codes data file for creating industry segments."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - indTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading indistry codes data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - // get the array of indices from the TableDataSet - int[] indcenCol = indTable.getColumnAsInt("indcen"); - int[] activityCol = indTable.getColumnAsInt("activity_code"); - - // get the max index value, to use for array dimensions - int maxInd = 0; - for (int ind : indcenCol) - if (ind > maxInd) maxInd = ind; - - int[] indcenIndustry = new int[maxInd + 1]; - for (int i = 0; i < indcenCol.length; i++) - { - int index = indcenCol[i]; - int value = activityCol[i]; - indcenIndustry[index] = value; - } - - return indcenIndustry; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java deleted file mode 100644 index d6364c4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagHouseholdDataManager2.java +++ /dev/null @@ -1,788 +0,0 @@ -package org.sandag.abm.application; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.BufferedReader; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileReader; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; - -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.HouseholdDataManager; -import org.sandag.abm.ctramp.Person; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.IndexSort; -import com.pb.common.util.PropertyMap; - -/** - * @author Jim Hicks - * - * Class for managing household and person object data read from - * synthetic population files. - */ -public class SandagHouseholdDataManager2 - extends HouseholdDataManager -{ - - public static final String HH_DATA_SERVER_NAME = SandagHouseholdDataManager.class - .getCanonicalName(); - public static final String HH_DATA_SERVER_ADDRESS = "127.0.0.1"; - public static final int HH_DATA_SERVER_PORT = 1139; - - public static final String PROPERTIES_OCCUP_CODES = "PopulationSynthesizer.OccupCodes"; - public static final String PROPERTIES_INDUSTRY_CODES = "PopulationSynthesizer.IndustryCodes"; - public static final String PROPERTIES_MILITARY_INDUSTRY_RANGE = "PopulationSynthesizer.MilitaryIndustryRange"; - - - private int militaryIndustryLow; - private int militaryIndustryHigh; - - public SandagHouseholdDataManager2() - { - super(); - } - - /** - * Associate data in hh and person TableDataSets read from synthetic - * population files with Household objects and Person objects with - * Households. - * - */ - public void mapTablesToHouseholdObjects() - { - - logger.info("mapping popsyn household and person data records to objects."); - - int id = -1; - - int invalidPersonTypeCount1 = 0; - int invalidPersonTypeCount2 = 0; - int invalidPersonTypeCount3 = 0; - - // read the correspondence files for mapping persons to occupation and - HashMap occCodes = readOccupCorrespondenceData(); - int[] indCodes = readIndustryCorrespondenceData(); - - // get the maximum HH id value to use to dimension the hhIndex - // correspondence - // array. The hhIndex array will store the hhArray index number for the - // given - // hh index. - int maxHhId = 0; - int hhIDColumn = hhTable.getColumnPosition(HH_ID_FIELD_NAME); - - for (int r = 1; r <= hhTable.getRowCount(); r++) - { - id = (int) hhTable.getValueAt(r, hhIDColumn); - if (id > maxHhId) maxHhId = id; - } - hhIndexArray = new int[maxHhId + 1]; - - // get an index array for households sorted in random order - to remove - // the original order - int[] firstSortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); - - // get a second index array for households sorted in random order - to - // select a sample from the randomly ordered hhs - int[] randomSortedIndices = getRandomOrderHhIndexArray(hhTable.getRowCount()); - - hhs = null; - - int numHouseholdsInSample = (int) (hhTable.getRowCount() * sampleRate); - Household[] hhArray = new Household[numHouseholdsInSample]; - - // String outputFileName = "sample_hh_mgra_taz_seed_" + sampleSeed + - // ".csv"; - // PrintWriter outStream = null; - // try { - // outStream = new PrintWriter(new BufferedWriter(new FileWriter(new - // File(outputFileName)))); - // outStream.println("i,mgra,taz"); - // } - // catch (IOException e) { - // logger.fatal(String.format("Exception occurred opening output skims file: %s.", - // outputFileName)); - // throw new RuntimeException(e); - // } - - int[] tempFreqArray = new int[40000]; - int[] hhOriginSortArray = new int[numHouseholdsInSample]; - for (int i = 0; i < numHouseholdsInSample; i++) - { - int r = firstSortedIndices[randomSortedIndices[i]] + 1; - // int hhId = (int) hhTable.getValueAt(r, - // hhTable.getColumnPosition(HH_ID_FIELD_NAME)); - int hhMgra = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); - int hhTaz = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); - hhOriginSortArray[i] = hhMgra; - tempFreqArray[hhMgra]++; - - // outStream.println(i + "," + hhMgra + "," + hhTaz); - } - - // outStream.close(); - // System.exit(1); - - int mgrasInSample = 0; - for (int i = 0; i < tempFreqArray.length; i++) - { - if (tempFreqArray[i] > 0) mgrasInSample++; - } - logger.info(mgrasInSample + " unique MGRA values in the " + (sampleRate * 100) - + "% sample."); - - // get an index array for households sorted in order of home mgra - int[] newOrder = new int[numHouseholdsInSample]; - int[] sortedIndices = IndexSort.indexSort(hhOriginSortArray); - for (int i = 0; i < sortedIndices.length; i++) - { - int k = sortedIndices[i]; - newOrder[k] = i; - } - - // for each household in the sample - for (int i = 0; i < numHouseholdsInSample; i++) - { - int r = firstSortedIndices[randomSortedIndices[i]] + 1; - try - { - // create a Household object - Household hh = new Household(modelStructure); - - // get required values from table record and store in Household - // object - id = (int) hhTable.getValueAt(r, hhTable.getColumnPosition(HH_ID_FIELD_NAME)); - hh.setHhId(id, inputRandomSeed); - - // set the household in the hhIndexArray in random order - int newIndex = newOrder[i]; - hhIndexArray[hh.getHhId()] = newIndex; - - int htaz = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_TAZ_FIELD_NAME)); - hh.setHhTaz(htaz); - - int hmgra = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_HOME_MGRA_FIELD_NAME)); - hh.setHhMgra(hmgra); - - // autos could be modeled or from PUMA - int numAutos = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_AUTOS_FIELD_NAME)); - hh.setHhAutos(numAutos); - - // set the hhSize variable and create Person objects for each - // person - int numPersons = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_SIZE_FIELD_NAME)); - hh.setHhSize(numPersons); - - int numWorkers = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_WORKERS_FIELD_NAME)); - hh.setHhWorkers(numWorkers); - - int incomeCat = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_INCOME_CATEGORY_FIELD_NAME)); - hh.setHhIncomeCategory(incomeCat); - - int incomeInDollars = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_INCOME_DOLLARS_FIELD_NAME)); - hh.setHhIncomeInDollars(incomeInDollars); - - // 0=Housing unit, 1=Institutional group quarters, - // 2=Noninstitutional - // group quarters - int unitType = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_UNITTYPE_FIELD_NAME)); - hh.setUnitType(unitType); - - // 1=Family household:married-couple, 2=Family household:male - // householder,no wife present, 3=Family household:female - // householder,no - // husband present - // 4=Nonfamily household:male householder, living alone, - // 5=Nonfamily - // household:male householder, not living alone, - // 6=Nonfamily household:female householder, living alone, - // 7=Nonfamily household:female householder, not living alone - int type = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_TYPE_FIELD_NAME)); - hh.setHhType(type); - - // 1=mobile home, 2=one-family house detached from any other - // house, - // 3=one-family house attached to one or more houses, - // 4=building with 2 apartments, 5=building with 3 or 4 - // apartments, - // 6=building with 5 to 9 apartments, - // 7=building with 10 to 19 apartments, 8=building with 20 to 49 - // apartments, - // 9=building with 50 or more apartments, 10=Boat,RV,van,etc. - int bldgsz = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(HH_BLDGSZ_FIELD_NAME)); - hh.setHhBldgsz(bldgsz); - - hh.initializeWindows(); - hhArray[newIndex] = hh; - - } catch (Exception e) - { - - logger.fatal(String - .format("exception caught mapping household data record to a Household object, r=%d, id=%d.", - r, id)); - throw new RuntimeException(e); - - } - - } - - int[] personHhStart = new int[maxHhId + 1]; - int[] personHhEnd = new int[maxHhId + 1]; - - // get hhid for person record 1 - int hhid = (int) personTable.getValueAt(1, - personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); - personHhStart[hhid] = 1; - int oldHhid = hhid; - - for (int r = 1; r <= personTable.getRowCount(); r++) - { - - // get the Household object for this person data to be stored in - hhid = (int) personTable.getValueAt(r, - personTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); - - if (hhid != oldHhid) - { - personHhEnd[oldHhid] = r - 1; - oldHhid = hhid; - personHhStart[hhid] = r; - } - - } - personHhEnd[hhid] = personTable.getRowCount(); - - int r = 0; - int p = 0; - int persId = 0; - int persNum = 0; - int fieldCount = 0; - - for (int i = 0; i < numHouseholdsInSample; i++) - { - - try - { - - r = firstSortedIndices[randomSortedIndices[i]] + 1; - - hhid = (int) hhTable.getValueAt(r, - hhTable.getColumnPosition(PERSON_HH_ID_FIELD_NAME)); - - int index = hhIndexArray[hhid]; - Household hh = hhArray[index]; - - persNum = 1; - - for (p = personHhStart[hhid]; p <= personHhEnd[hhid]; p++) - { - - fieldCount = 0; - - // get the Person object for this person data to be stored - // in - persId = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_PERSON_ID_FIELD_NAME)); - Person person = hh.getPerson(persNum++); - person.setPersId(persId); - fieldCount++; - - // get required values from table record and store in Person - // object - int age = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_AGE_FIELD_NAME)); - person.setPersAge(age); - fieldCount++; - - int gender = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_GENDER_FIELD_NAME)); - person.setPersGender(gender); - fieldCount++; - /* - * int occcen1 = (int) personTable.getValueAt(p, - * personTable.getColumnPosition(PERSON_SOC_FIELD_NAME)); - * int pecasOccup = occCodes[occcen1]; - */ - int military = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_MILITARY_FIELD_NAME)); - int pecasOccup = 0; - - String occsoc = personTable.getStringValueAt(p, - personTable.getColumnPosition(PERSON_SOC_FIELD_NAME)); - - int indcen = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_INDCEN_FIELD_NAME)); - int activityCode = indCodes[indcen]; - - if (military == 1) // in active military - pecasOccup = 56; - else if (military != 1 && indcen >= militaryIndustryLow - && indcen <= militaryIndustryHigh) // not active - // military but - // military - // contractor - pecasOccup = 56; - else - { - if (!occCodes.containsKey(occsoc)) - { - logger.fatal("Error: Occupation code " + occsoc + " for hhid " + hhid - + " person " + p + " not found in occupation file"); - throw new RuntimeException(); - } - pecasOccup = occCodes.get(occsoc); // everyone else - if (pecasOccup == 0) logger.warn("pecasOccup==0 for occsoc==" + occsoc); - } - - person.setPersActivityCode(activityCode); - fieldCount++; - - person.setPersPecasOccup(pecasOccup); - fieldCount++; - - /* - * These are the old codes, based upon census occupation - * definitions if ((pecasOccup == 71) && (activityCode == 2 - * || activityCode == 4 || activityCode == 6 || activityCode - * == 8 || activityCode == 29)) activityCode++; - * - * if ((pecasOccup == 76) && (activityCode == 3 || - * activityCode == 5 || activityCode == 7 || activityCode == - * 9 || activityCode == 30)) activityCode--; - * - * if ((pecasOccup == 76) && (activityCode == 13)) - * activityCode = 14; - * - * if ((pecasOccup == 71) && (activityCode == 14)) - * activityCode = 13; - * - * if ((pecasOccup == 75) && (activityCode == 18)) - * activityCode = 22; - * - * if ((pecasOccup == 71) && (activityCode == 22)) - * activityCode = 18; - * - * if (activityCode == 28) pecasOccup = 77; - */ - - // Employment status (1-employed FT, 2-employed PT, 3-not - // employed, 4-under age 16) - int empCat = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME)); - - // recode PEMPLOY to 3 for persons whose age is 16 or - // greater and who have PEMPLOY set to 4 - if (empCat == 4 && age >= 16) empCat = 3; - person.setPersEmploymentCategory(empCat); - - fieldCount++; - - // Student status (1 - student in grade or high school; 2 - - // student in college or higher; 3 - not a student) - int studentCat = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_STUDENT_CATEGORY_FIELD_NAME)); - person.setPersStudentCategory(studentCat); - fieldCount++; - - // Person type (1-FT worker age 16+, 2-PT worker nonstudent - // age - // 16+, 3-university student, 4-nonworker nonstudent age - // 16-64, - // 5-nonworker nonstudent age 65+, - // 6-"age 16-19 student, not FT wrkr or univ stud", 7-age - // 6-15 - // schpred, 8 under age 6 presch - int personType = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_TYPE_CATEGORY_FIELD_NAME)); - person.setPersonTypeCategory(personType); - fieldCount++; - - // Person educational attainment level to determine high - // school - // graduate status ( < 9 - not a graduate, 10+ - high school - // graduate - // and beyond) - int educ = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_EDUCATION_ATTAINMENT_FIELD_NAME)); - if (educ >= 9) person.setPersonIsHighSchoolGraduate(true); - else person.setPersonIsHighSchoolGraduate(false); - fieldCount++; - - // Person educational attainment level to determine higher - // education status ( > 12 - at least a bachelor's degree ) - if (educ >= 13) person.setPersonHasBachelors(true); - else person.setPersonHasBachelors(false); - fieldCount++; - - // Person grade enrolled in ( 0-"not enrolled", - // 1-"preschool", - // 2-"Kindergarten", 3-"Grade 1 to grade 4", - // 4-"Grade 5 to grade 8", 5-"Grade 9 to grade 12", - // 6-"College undergraduate", - // 7-"Graduate or professional school" - // ) - int grade = (int) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_GRADE_ENROLLED_FIELD_NAME)); - person.setPersonIsGradeSchool(false); - person.setPersonIsHighSchool(false); - if (grade >= 2 && grade <= 4) - { - // change person type if person was 5 or under but - // enrolled in K-8. - if (person.getPersonIsPreschoolChild() == 1) - person.setPersonTypeCategory(Person.PersonType.Student_age_6_15_schpred - .ordinal()); - - person.setPersonIsGradeSchool(true); - } else if (grade == 5) - { - person.setPersonIsHighSchool(true); - } - fieldCount++; - - // if person is a university student but has school age - // student - // category value, reset student category value - if (personType == Person.PersonType.University_student.ordinal() - && studentCat != Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER - .ordinal()) - { - studentCat = Person.StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount1++; - } else if (studentCat != Person.StudentStatus.NON_STUDENT.ordinal() - && empCat == Person.EmployStatus.FULL_TIME.ordinal()) - { - // if person is a student of any kind but has full-time - // employment status, reset student category value to - // non-student - studentCat = Person.StudentStatus.NON_STUDENT.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount2++; - } - fieldCount++; - - // check consistency of student category and person type - if (studentCat == Person.StudentStatus.NON_STUDENT.ordinal()) - { - - if (person.getPersonIsStudentNonDriving() == 1 - || person.getPersonIsStudentDriving() == 1) - { - studentCat = Person.StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal(); - person.setPersStudentCategory(studentCat); - invalidPersonTypeCount3++; - } - - } - fieldCount++; - - double timeFactorWork = 1.0; - double timeFactorNonWork = 1.0; - if(readTimeFactors){ - timeFactorWork = (double) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_TIMEFACTOR_WORK_FIELD_NAME)); - timeFactorNonWork = (double) personTable.getValueAt(p, - personTable.getColumnPosition(PERSON_TIMEFACTOR_NONWORK_FIELD_NAME)); - } - person.setTimeFactorWork(timeFactorWork); - person.setTimeFactorNonWork(timeFactorNonWork); - - } - - } catch (Exception e) - { - - logger.fatal(String - .format("exception caught mapping person data record to a Person object, i=%d, r=%d, p=%d, hhid=%d, persid=%d, persnum=%d, fieldCount=%d.", - i, r, p, hhid, persId, persNum, fieldCount)); - throw new RuntimeException(e); - - } - - } // person loop - - hhs = hhArray; - - logger.warn(invalidPersonTypeCount1 - + " person type = university and student category = non-student person records had their student category changed to university or higher."); - logger.warn(invalidPersonTypeCount2 - + " Student category = student and employment category = full-time worker person records had their student category changed to non-student."); - logger.warn(invalidPersonTypeCount3 - + " Student category = non-student and person type = student person records had their student category changed to student high school or less."); - - // logger.info("Setting distributed values of time. "); - // setDistributedValuesOfTime(); - - } - - /** - * if called, must be called after readData so that the size of the full - * population is known. - * - * @param hhFileName - * @param persFileName - * @param numHhs - */ - public void createSamplePopulationFiles(String hhFileName, String persFileName, - String newHhFileName, String newPersFileName, int numHhs) - { - - int maximumHhId = 0; - for (int i = 0; i < hhs.length; i++) - { - int id = hhs[i].getHhId(); - if (id > maximumHhId) maximumHhId = id; - } - - int[] testHhs = new int[maximumHhId + 1]; - - int[] sortedIndices = getRandomOrderHhIndexArray(hhs.length); - - for (int i = 0; i < numHhs; i++) - { - int k = sortedIndices[i]; - int hhId = hhs[k].getHhId(); - testHhs[hhId] = 1; - } - - String hString = ""; - int hCount = 0; - try - { - - logger.info(String.format("writing sample household file for %d households", numHhs)); - - PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newHhFileName))); - BufferedReader in = new BufferedReader(new FileReader(hhFileName)); - - // read headers and write to output files - hString = in.readLine(); - out.write(hString + "\n"); - hCount++; - int count = 0; - - while ((hString = in.readLine()) != null) - { - hCount++; - int endOfField = hString.indexOf(','); - int hhId = Integer.parseInt(hString.substring(0, endOfField)); - - // if it's a sample hh, write the hh and the person records - if (testHhs[hhId] == 1) - { - out.write(hString + "\n"); - count++; - if (count == numHhs) break; - } - } - - out.close(); - - } catch (IOException e) - { - logger.fatal("IO Exception caught creating sample synpop household file."); - logger.fatal(String.format("reading hh file = %s, writing sample hh file = %s.", - hhFileName, newHhFileName)); - logger.fatal(String.format("hString = %s, hCount = %d.", hString, hCount)); - } - - String pString = ""; - int pCount = 0; - try - { - - logger.info(String.format("writing sample person file for selected households")); - - PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter(newPersFileName))); - BufferedReader in = new BufferedReader(new FileReader(persFileName)); - - // read headers and write to output files - pString = in.readLine(); - out.write(pString + "\n"); - pCount++; - int count = 0; - int oldId = 0; - while ((pString = in.readLine()) != null) - { - pCount++; - int endOfField = pString.indexOf(','); - int hhId = Integer.parseInt(pString.substring(0, endOfField)); - - // if it's a sample hh, write the hh and the person records - if (testHhs[hhId] == 1) - { - out.write(pString + "\n"); - if (hhId > oldId) count++; - } else - { - if (count == numHhs) break; - } - - oldId = hhId; - - } - - out.close(); - - } catch (IOException e) - { - logger.fatal("IO Exception caught creating sample synpop person file."); - logger.fatal(String.format( - "reading person file = %s, writing sample person file = %s.", persFileName, - newPersFileName)); - logger.fatal(String.format("pString = %s, pCount = %d.", pString, pCount)); - } - - } - - public static void main(String[] args) throws Exception - { - - String serverAddress = HH_DATA_SERVER_ADDRESS; - int serverPort = HH_DATA_SERVER_PORT; - - // optional arguments - for (int i = 0; i < args.length; i++) - { - if (args[i].equalsIgnoreCase("-hostname")) - { - serverAddress = args[i + 1]; - } - - if (args[i].equalsIgnoreCase("-port")) - { - serverPort = Integer.parseInt(args[i + 1]); - } - } - - Remote.config(serverAddress, serverPort, null, 0); - - SandagHouseholdDataManager2 hhDataManager = new SandagHouseholdDataManager2(); - - ItemServer.bind(hhDataManager, HH_DATA_SERVER_NAME); - - System.out.println(String.format( - "SandagHouseholdDataManager2 server class started on: %s:%d", serverAddress, - serverPort)); - - } - - public int[] getJointToursByHomeMgra(String purposeString) - { - // TODO Auto-generated method stub - return null; - } - - /** - * This method reads a cross-walk file between the occsoc code in Census and - * the PECAS occupation categories. It stores the result in a HashMap and - * returns it. - * - * @return - */ - private HashMap readOccupCorrespondenceData() - { - - int[] militaryRange = PropertyMap.getIntegerArrayFromPropertyMap(propertyMap, - PROPERTIES_MILITARY_INDUSTRY_RANGE); - militaryIndustryLow = militaryRange[0]; - militaryIndustryHigh = militaryRange[1]; - - TableDataSet occTable = null; - - // construct input household file name from properties file values - String occupFileName = propertyMap.get(PROPERTIES_OCCUP_CODES); - String fileName = projectDirectory + "/" + occupFileName; - - try - { - logger.info("reading occupation codes data file for creating occupation segments."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - occTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String.format( - "Exception occurred occupation codes data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - HashMap occMap = new HashMap(); - - for (int i = 1; i <= occTable.getRowCount(); ++i) - { - - String soc = occTable.getStringValueAt(i, "occsoc5"); - int occ = (int) occTable.getValueAt(i, "commodity_id"); - occMap.put(soc, occ); - } - - return occMap; - } - - private int[] readIndustryCorrespondenceData() - { - - TableDataSet indTable = null; - - // construct input household file name from properties file values - String indFileName = propertyMap.get(PROPERTIES_INDUSTRY_CODES); - String fileName = projectDirectory + "/" + indFileName; - - try - { - logger.info("reading industry codes data file for creating industry segments."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - indTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading indistry codes data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - // get the array of indices from the TableDataSet - int[] indcenCol = indTable.getColumnAsInt("indcen"); - int[] activityCol = indTable.getColumnAsInt("activity_code"); - - // get the max index value, to use for array dimensions - int maxInd = 0; - for (int ind : indcenCol) - if (ind > maxInd) maxInd = ind; - - int[] indcenIndustry = new int[maxInd + 1]; - for (int i = 0; i < indcenCol.length; i++) - { - int index = indcenCol[i]; - int value = activityCol[i]; - indcenIndustry[index] = value; - } - - return indcenIndustry; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java deleted file mode 100644 index 0a40886..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualMandatoryTourFrequencyDMU.java +++ /dev/null @@ -1,101 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyDMU; - -/** - * ArcIndividualMandatoryTourFrequencyDMU is a class that ... - * - * @author Kimberly Grommes - * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. - */ -public class SandagIndividualMandatoryTourFrequencyDMU - extends IndividualMandatoryTourFrequencyDMU -{ - - public SandagIndividualMandatoryTourFrequencyDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getDistanceToWorkLocation", 1); - methodIndexMap.put("getDistanceToSchoolLocation", 2); - methodIndexMap.put("getEscortAccessibility", 3); - methodIndexMap.put("getDrivers", 4); - methodIndexMap.put("getPreschoolChildren", 5); - methodIndexMap.put("getNumberOfChildren6To18WithoutMandatoryActivity", 6); - methodIndexMap.put("getNonFamilyHousehold", 7); - methodIndexMap.put("getIncomeInDollars", 8); - methodIndexMap.put("getPersonType", 9); - methodIndexMap.put("getFemale", 10); - methodIndexMap.put("getAutos", 11); - methodIndexMap.put("getAge", 12); - methodIndexMap.put("getBestTimeToWorkLocation", 13); - methodIndexMap.put("getNotEmployed", 14); - methodIndexMap.put("getWorkAtHome", 15); - methodIndexMap.put("getSchoolAtHome", 16); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getDistanceToWorkLocation(); - case 2: - return getDistanceToSchoolLocation(); - case 3: - return getEscortAccessibility(); - case 4: - return getDrivers(); - case 5: - return getPreschoolChildren(); - case 6: - return getNumberOfChildren6To18WithoutMandatoryActivity(); - case 7: - return getNonFamilyHousehold(); - case 8: - return getIncomeInDollars(); - case 9: - return getPersonType(); - case 10: - return getFemale(); - case 11: - return getAutos(); - case 12: - return getAge(); - case 13: - return getBestTimeToWorkLocation(); - case 14: - return getNotEmployed(); - case 15: - return getWorkAtHome(); - case 16: - return getSchoolAtHome(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java deleted file mode 100644 index 5e675e8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagIndividualNonMandatoryTourFrequencyDMU.java +++ /dev/null @@ -1,272 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.IndividualNonMandatoryTourFrequencyDMU; - -/** - * ArcIndividualNonMandatoryTourFrequencyDMU is a class that ... - * - */ -public class SandagIndividualNonMandatoryTourFrequencyDMU - extends IndividualNonMandatoryTourFrequencyDMU -{ - - public SandagIndividualNonMandatoryTourFrequencyDMU() - { - super(); - setupMethodIndexMap(); - - // set names used in SANDAG stop purpose file - TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME = "escort"; - TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME = "shopping"; - TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME = "othmaint"; - TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME = "eatout"; - TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME = "visit"; - TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME = "othdiscr"; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getIncomeInDollars", 0); - methodIndexMap.put("getHouseholdSize", 1); - methodIndexMap.put("getNumAutos", 2); - methodIndexMap.put("getCarsEqualsWorkers", 3); - methodIndexMap.put("getMoreCarsThanWorkers", 4); - methodIndexMap.put("getNumAdults", 5); - methodIndexMap.put("getNumChildren", 6); - methodIndexMap.put("getPersonIsAdult", 7); - methodIndexMap.put("getPersonIsChild", 8); - methodIndexMap.put("getPersonIsFullTimeWorker", 9); - methodIndexMap.put("getPersonIsPartTimeWorker", 10); - methodIndexMap.put("getPersonIsUniversity", 11); - methodIndexMap.put("getPersonIsNonworker", 12); - methodIndexMap.put("getPersonIsPreschool", 13); - methodIndexMap.put("getPersonIsStudentNonDriving", 14); - methodIndexMap.put("getPersonIsStudentDriving", 15); - methodIndexMap.put("getPersonStaysHome", 16); - methodIndexMap.put("getFemale", 17); - methodIndexMap.put("getFullTimeWorkers", 18); - methodIndexMap.put("getPartTimeWorkers", 19); - methodIndexMap.put("getUniversityStudents", 20); - methodIndexMap.put("getNonWorkers", 21); - methodIndexMap.put("getDrivingAgeStudents", 22); - methodIndexMap.put("getNonDrivingAgeStudents", 23); - methodIndexMap.put("getPreSchoolers", 24); - // methodIndexMap.put("getMaxAdultOverlaps", 26); - // methodIndexMap.put("getMaxChildOverlaps", 27); - // methodIndexMap.put("getMaxMixedOverlaps", 28); - // methodIndexMap.put("getMaxPairwiseOverlapAdult", 29); - // methodIndexMap.put("getMaxPairwiseOverlapChild", 30); - // methodIndexMap.put("getWindowBeforeFirstMandJointTour", 31); - // methodIndexMap.put("getWindowBetweenFirstLastMandJointTour", 32); - // methodIndexMap.put("getWindowAfterLastMandJointTour", 33); - methodIndexMap.put("getNumHhFtWorkers", 34); - methodIndexMap.put("getNumHhPtWorkers", 35); - methodIndexMap.put("getNumHhUnivStudents", 36); - methodIndexMap.put("getNumHhNonWorkAdults", 37); - methodIndexMap.put("getNumHhRetired", 38); - methodIndexMap.put("getNumHhDrivingStudents", 39); - methodIndexMap.put("getNumHhNonDrivingStudents", 40); - methodIndexMap.put("getNumHhPreschool", 41); - methodIndexMap.put("getTravelActiveAdults ", 42); - methodIndexMap.put("getTravelActiveChildren ", 43); - methodIndexMap.put("getNumMandatoryTours", 44); - methodIndexMap.put("getNumJointShoppingTours", 45); - methodIndexMap.put("getNumJointOthMaintTours", 46); - methodIndexMap.put("getNumJointEatOutTours", 47); - methodIndexMap.put("getNumJointSocialTours", 48); - methodIndexMap.put("getNumJointOthDiscrTours", 49); - methodIndexMap.put("getJTours", 50); - methodIndexMap.put("getPreDrivingAtHome", 51); - methodIndexMap.put("getPreschoolAtHome", 52); - methodIndexMap.put("getDistanceToWorkLocation", 53); - methodIndexMap.put("getDistanceToSchoolLocation", 54); - methodIndexMap.put("getEscortAccessibility", 55); - methodIndexMap.put("getShopAccessibility", 56); - methodIndexMap.put("getMaintAccessibility", 57); - methodIndexMap.put("getEatOutAccessibility", 58); - methodIndexMap.put("getVisitAccessibility", 59); - methodIndexMap.put("getDiscrAccessibility", 60); - methodIndexMap.put("getCdapIndex", 61); - methodIndexMap.put("getNonMotorizedDcLogsum", 62); - methodIndexMap.put("getNumPredrivingKidsGoOut", 63); - methodIndexMap.put("getNumPreschoolKidsGoOut", 64); - methodIndexMap.put("getCollegeEducation", 65); - methodIndexMap.put("getLowEducation", 66); - methodIndexMap.put("getDetachedHh", 67); - methodIndexMap.put("getWorksAtHome", 68); - methodIndexMap.put("getWorkAccessibility", 69); - methodIndexMap.put("getSchoolAccessibility", 70); - methodIndexMap.put("getTelecommuteFrequency", 71); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - - case 0: - return getIncomeInDollars(); - case 1: - return getHouseholdSize(); - case 2: - return getNumAutos(); - case 3: - return getCarsEqualsWorkers(); - case 4: - return getMoreCarsThanWorkers(); - case 5: - return getNumAdults(); - case 6: - return getNumChildren(); - case 7: - return getPersonIsAdult(); - case 8: - return getPersonIsChild(); - case 9: - return getPersonIsFullTimeWorker(); - case 10: - return getPersonIsPartTimeWorker(); - case 11: - return getPersonIsUniversity(); - case 12: - return getPersonIsNonworker(); - case 13: - return getPersonIsPreschool(); - case 14: - return getPersonIsStudentNonDriving(); - case 15: - return getPersonIsStudentDriving(); - case 16: - return getPersonStaysHome(); - case 17: - return getFemale(); - case 18: - return getFullTimeWorkers(); - case 19: - return getPartTimeWorkers(); - case 20: - return getUniversityStudents(); - case 21: - return getNonWorkers(); - case 22: - return getDrivingAgeStudents(); - case 23: - return getNonDrivingAgeStudents(); - case 24: - return getPreSchoolers(); - // case 26: - // return getMaxAdultOverlaps(); - // case 27: - // return getMaxChildOverlaps(); - // case 28: - // return getMaxMixedOverlaps(); - // case 29: - // return getMaxPairwiseOverlapAdult(); - // case 30: - // return getMaxPairwiseOverlapChild(); - // case 31: - // return getWindowBeforeFirstMandJointTour(); - // case 32: - // return getWindowBetweenFirstLastMandJointTour(); - // case 33: - // return getWindowAfterLastMandJointTour(); - case 34: - return getNumHhFtWorkers(); - case 35: - return getNumHhPtWorkers(); - case 36: - return getNumHhUnivStudents(); - case 37: - return getNumHhNonWorkAdults(); - case 38: - return getNumHhRetired(); - case 39: - return getNumHhDrivingStudents(); - case 40: - return getNumHhNonDrivingStudents(); - case 41: - return getNumHhPreschool(); - case 42: - return getTravelActiveAdults(); - case 43: - return getTravelActiveChildren(); - case 44: - return getNumMandatoryTours(); - case 45: - return getNumJointShoppingTours(); - case 46: - return getNumJointOthMaintTours(); - case 47: - return getNumJointEatOutTours(); - case 48: - return getNumJointSocialTours(); - case 49: - return getNumJointOthDiscrTours(); - case 50: - return getJTours(); - case 51: - return getPreDrivingAtHome(); - case 52: - return getPreschoolAtHome(); - case 53: - return getDistanceToWorkLocation(); - case 54: - return getDistanceToSchoolLocation(); - case 55: - return getEscortAccessibility(); - case 56: - return getShopAccessibility(); - case 57: - return getMaintAccessibility(); - case 58: - return getEatOutAccessibility(); - case 59: - return getVisitAccessibility(); - case 60: - return getDiscrAccessibility(); - case 61: - return getCdapIndex(); - case 62: - return getNonMotorizedDcLogsum(); - case 63: - return getNumPredrivingKidsGoOut(); - case 64: - return getNumPreschoolKidsGoOut(); - case 65: - return getCollegeEducation(); - case 66: - return getLowEducation(); - case 67: - return getDetachedHh(); - case 68: - return getWorksAtHome(); - case 69: - return getWorkAccessibility(); - case 70: - return getSchoolAccessibility(); - case 71: - return getTelecommuteFrequency(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java deleted file mode 100644 index fbab4ac..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagInternalExternalTripChoiceDMU.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.InternalExternalTripChoiceDMU; - -public class SandagInternalExternalTripChoiceDMU - extends InternalExternalTripChoiceDMU -{ - - private transient Logger logger = Logger.getLogger(SandagInternalExternalTripChoiceDMU.class); - - public SandagInternalExternalTripChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getDistanceToCordonsLogsum", 0); - methodIndexMap.put("getVehiclesPerHouseholdMember", 1); - methodIndexMap.put("getHhIncomeInDollars", 2); - methodIndexMap.put("getAge", 3); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getDistanceToCordonsLogsum(); - case 1: - return getVehiclesPerHouseholdMember(); - case 2: - return getHhIncomeInDollars(); - case 3: - return getAge(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java deleted file mode 100644 index f57b1d3..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagJointTourModelsDMU.java +++ /dev/null @@ -1,137 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.JointTourModelsDMU; -import org.sandag.abm.ctramp.ModelStructure; - -public class SandagJointTourModelsDMU - extends JointTourModelsDMU -{ - - public SandagJointTourModelsDMU(ModelStructure modelStructure) - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getActiveCountFullTimeWorkers", 1); - methodIndexMap.put("getActiveCountPartTimeWorkers", 2); - methodIndexMap.put("getActiveCountUnivStudents", 3); - methodIndexMap.put("getActiveCountNonWorkers", 4); - methodIndexMap.put("getActiveCountRetirees", 5); - methodIndexMap.put("getActiveCountDrivingAgeSchoolChildren", 6); - methodIndexMap.put("getActiveCountPreDrivingAgeSchoolChildren", 7); - methodIndexMap.put("getActiveCountPreSchoolChildren", 8); - methodIndexMap.put("getMaxPairwiseAdultOverlapsHh", 9); - methodIndexMap.put("getMaxPairwiseChildOverlapsHh", 10); - methodIndexMap.put("getMaxPairwiseMixedOverlapsHh", 11); - methodIndexMap.put("getMaxPairwiseOverlapOtherAdults", 12); - methodIndexMap.put("getMaxPairwiseOverlapOtherChildren", 13); - methodIndexMap.put("getTravelActiveAdults", 14); - methodIndexMap.put("getTravelActiveChildren", 15); - methodIndexMap.put("getPersonStaysHome", 16); - methodIndexMap.put("getIncomeLessThan30K", 17); - methodIndexMap.put("getIncome30Kto60K", 18); - methodIndexMap.put("getIncomeMoreThan100K", 19); - methodIndexMap.put("getNumAdults", 20); - methodIndexMap.put("getNumChildren", 21); - methodIndexMap.put("getHhWorkers", 22); - methodIndexMap.put("getAutoOwnership", 23); - methodIndexMap.put("getTourPurposeIsMaint", 24); - methodIndexMap.put("getTourPurposeIsEat", 25); - methodIndexMap.put("getTourPurposeIsVisit", 26); - methodIndexMap.put("getTourPurposeIsDiscr", 27); - methodIndexMap.put("getPersonType", 28); - methodIndexMap.put("getJointTourComposition", 29); - methodIndexMap.put("getJTours", 30); - methodIndexMap.put("getShopHOVAccessibility", 31); - methodIndexMap.put("getMaintHOVAccessibility", 32); - methodIndexMap.put("getDiscrHOVAccessibility", 33); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getActiveCountFullTimeWorkers(); - case 2: - return getActiveCountPartTimeWorkers(); - case 3: - return getActiveCountUnivStudents(); - case 4: - return getActiveCountNonWorkers(); - case 5: - return getActiveCountRetirees(); - case 6: - return getActiveCountDrivingAgeSchoolChildren(); - case 7: - return getActiveCountPreDrivingAgeSchoolChildren(); - case 8: - return getActiveCountPreSchoolChildren(); - case 9: - return getMaxPairwiseAdultOverlapsHh(); - case 10: - return getMaxPairwiseChildOverlapsHh(); - case 11: - return getMaxPairwiseMixedOverlapsHh(); - case 12: - return getMaxPairwiseOverlapOtherAdults(); - case 13: - return getMaxPairwiseOverlapOtherChildren(); - case 14: - return getTravelActiveAdults(); - case 15: - return getTravelActiveChildren(); - case 16: - return getPersonStaysHome(); - case 17: - return getIncomeLessThan30K(); - case 18: - return getIncome30Kto60K(); - case 19: - return getIncomeMoreThan100K(); - case 20: - return getNumAdults(); - case 21: - return getNumChildren(); - case 22: - return getHhWorkers(); - case 23: - return getAutoOwnership(); - case 24: - return getTourPurposeIsMaint(); - case 25: - return getTourPurposeIsEat(); - case 26: - return getTourPurposeIsVisit(); - case 27: - return getTourPurposeIsDiscr(); - case 28: - return getPersonType(); - case 29: - return getJointTourComposition(); - case 30: - return getJTours(); - case 31: - return getShopHOVAccessibility(); - case 32: - return getMaintHOVAccessibility(); - case 33: - return getDiscrHOVAccessibility(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java deleted file mode 100644 index 2c74056..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMGRAtoPNR.java +++ /dev/null @@ -1,287 +0,0 @@ -package org.sandag.abm.application; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileInputStream; -import java.io.FileWriter; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.MissingResourceException; -import java.util.Properties; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.active.sandag.PropertyParser; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.reporting.CsvRow; -import org.sandag.abm.reporting.DataExporter; -import org.sandag.abm.reporting.OMXMatrixDao; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -/** - * @author malinovskiyy - * Produces p&r and k&r access connections - * input files: - * impdan_xx.mtx: SOV non toll xx peak period skim matrix (3 cores: *SCST_XX (generalized cost), length(Skim) (distance in mile), and *STM_XX (Skim) (single driver xx time in minutes) - * tap.ptype: tap, lot id, parking type, taz, capacity, distance from lot to tap - * output files - * access.prp - */ -public class SandagMGRAtoPNR { - - private static Logger logger = Logger.getLogger("createPNRAccessFile"); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TapDataManager tapManager; - private int[] taps; - private int[] tazs; - private float[][][] tapParkingInfo; - private HashMap tapMap; - private HashMap> bestTapsMap; - private Matrix distanceMtx; - private Matrix timeMtx; - - private final Properties properties; - private final OMXMatrixDao mtxDao; - - private static final String FORMAL_PREMIUM = "taps.formal.premium.maxDist"; //8.0f; - private static final String FORMAL_EXPRESS = "taps.formal.express.maxDist"; //8.0f; - private static final String FORMAL_LOCAL = "taps.formal.local.maxDist"; //4.0f; - private static final String INFORMAL_PREMIUM = "taps.informal.premium.maxDist"; //4.0f; - private static final String INFORMAL_EXPRESS = "taps.informal.express.maxDist"; //4.0f; - private static final String INFORMAL_LOCAL = "taps.informal.local.maxDist"; //2.0f; - private static final String PREMIUM_MODES = "taps.premium.modes"; //new ArrayList(){{add(4); add(5); add(6); add(7);}}; - private static final String EXPRESS_MODES = "taps.express.modes"; //new ArrayList(){{add(8); add(9);}}; - private static final String LOCAL_MODES = "taps.local.modes"; //new ArrayList(){{add(10);}}; - - private static final String TAPS_SKIM = "taps.skim"; - private static final String TAPS_SKIM_DIST = "taps.skim.dist"; - private static final String TAPS_SKIM_TIME = "taps.skim.time"; - - private static final String EXTERNAL_TAZs = "external.tazs"; - - private double formalPremiumMaxD; - private double formalExpressMaxD; - private double formalLocalMaxD; - private double informalPremiumMaxD; - private double informalExpressMaxD; - private double informalLocalMaxD; - - private ArrayList premiumModes; - private ArrayList expressModes; - private ArrayList localModes; - private ArrayList externalTAZs; - - private static final String PROJECT_PATH_PROPERTY_TOKEN = "%project.folder%"; - - public SandagMGRAtoPNR(Properties theProperties, OMXMatrixDao aMtxDao, String projectPath, HashMap rbMap) - { - this.properties = theProperties; - this.mtxDao = aMtxDao; - - tapManager = TapDataManager.getInstance(rbMap); - //tazManager = TazDataManager.getInstance(rbMap); - this.taps = tapManager.getTaps(); - this.tapParkingInfo = tapManager.getTapParkingInfo(); - this.tapMap = getTAPMap(); - this.bestTapsMap = new HashMap>(); - this.distanceMtx = aMtxDao.getMatrix((String)properties.get(TAPS_SKIM),(String)properties.get(TAPS_SKIM_DIST)); - this.timeMtx = aMtxDao.getMatrix((String)properties.get(TAPS_SKIM),(String)properties.get(TAPS_SKIM_TIME)); - this.tazs = this.distanceMtx.getExternalRowNumbers(); - - - formalPremiumMaxD = Double.parseDouble((String) properties.get(FORMAL_PREMIUM)); - formalExpressMaxD = Double.parseDouble((String) properties.get(FORMAL_EXPRESS)); - formalLocalMaxD = Double.parseDouble((String) properties.get(FORMAL_LOCAL)); - informalPremiumMaxD = Double.parseDouble((String) properties.get(INFORMAL_PREMIUM)); - informalExpressMaxD = Double.parseDouble((String) properties.get(INFORMAL_EXPRESS)); - informalLocalMaxD = Double.parseDouble((String) properties.get(INFORMAL_LOCAL)); - - List stringList = Arrays.asList(((String) properties.get(PREMIUM_MODES)).split("\\s*,\\s*")); - premiumModes = new ArrayList(); - for (int i = 0; i < stringList.size(); i++){ - premiumModes.add(Integer.parseInt(stringList.get(i))); - } - - stringList = Arrays.asList(((String) properties.get(EXPRESS_MODES)).split("\\s*,\\s*")); - expressModes = new ArrayList(); - for (int i = 0; i < stringList.size(); i++){ - expressModes.add(Integer.parseInt(stringList.get(i))); - } - - stringList = Arrays.asList(((String) properties.get(LOCAL_MODES)).split("\\s*,\\s*")); - localModes = new ArrayList(); - for (int i = 0; i < stringList.size(); i++){ - localModes.add(Integer.parseInt(stringList.get(i))); - } - - stringList = Arrays.asList(((String) properties.get(EXTERNAL_TAZs)).split("\\s*,\\s*")); - externalTAZs = new ArrayList(); - for (int i = 0; i < stringList.size(); i++){ - externalTAZs.add(Integer.parseInt(stringList.get(i))); - } - } - - - public HashMap getTAPMap(){ - HashMap tm = new HashMap(); - for(int i = 0; i < taps.length; i++){ - int tap = taps[i]; - if(tap < tapParkingInfo.length && tapParkingInfo[tap] != null && tapParkingInfo[tap][0] != null){ - int taz = (int) tapParkingInfo[tap][1][0]; - tm.put(tap, taz); - } - } - return tm; - } - - public void nearestTAPs(){ - for(int i = 1; i < tazs.length; i++) { - int currTAZ = tazs[i]; - - //Skip externals - if(externalTAZs.contains(currTAZ)) - continue; - - ArrayList reachableTAPs = new ArrayList(); - bestTapsMap.put(currTAZ, reachableTAPs); - - HashMap modeMap = new HashMap(); - ArrayList addedTaps = new ArrayList(); - - for(Integer j : tapMap.keySet()){ - int tapTAZ = tapMap.get(j); - //distance to taz with the current tap - float dist = distanceMtx.getValueAt(currTAZ, tapTAZ); - float time = timeMtx.getValueAt(currTAZ, tapTAZ); - int lotType = (int) tapParkingInfo[j][3][0]; - int mode = (int) tapParkingInfo[j][5][0]; - //dist = (float) (dist + (tapParkingInfo[j][4][0] / 5280.0)); - if(!modeMap.containsKey(mode) || modeMap.get(mode)[2] > dist){ - float[] vals = new float[4]; - vals[0] = j; - vals[1] = time; - vals[2] = dist; - vals[3] = mode; - modeMap.put(mode, vals); - } - - //formal, premium, less than 8 miles - if( (lotType == 1 && premiumModes.contains(mode) && dist < formalPremiumMaxD) || - //formal, express, less than 8 miles - (lotType == 1 && expressModes.contains(mode) && dist < formalExpressMaxD) || - //formal, local, less than 4 miles - (lotType == 1 && localModes.contains(mode) && dist < formalLocalMaxD) || - //informal, premium, less than 4 miles - (lotType > 1 && premiumModes.contains(mode) && dist < informalPremiumMaxD) || - //informal, express, less than 4 miles - (lotType > 1 && expressModes.contains(mode) && dist < informalExpressMaxD) || - //informal, local, less than 2 miles - (lotType > 1 && localModes.contains(mode) && dist < informalLocalMaxD) ){ - float[] vals = new float[4]; - vals[0] = j; - vals[1] = time; - vals[2] = dist; - vals[3] = mode; - bestTapsMap.get(currTAZ).add(vals); - addedTaps.add(j); - } - } - for(Integer m : modeMap.keySet()){ - float[] closestTAPvals = modeMap.get(m); - int tap = (int) closestTAPvals[0]; - if(!addedTaps.contains(tap)){ //Put best taps by mode into bestTapsMap if they are not already there - bestTapsMap.get(currTAZ).add(closestTAPvals); - addedTaps.add(tap); - } - } - } - } - - /*The file has five columns: TAZ, TAP, travel time (min) *100, distance (mile) *100 and mode. - */ - public void writeResults(String filename) throws IOException{ - BufferedWriter writer = null; - try{ - writer = new BufferedWriter(new FileWriter(new File(filename))); - - for(int i = 1; i < tazs.length; i++) { - int currTAZ = tazs[i]; - - //Skip externals - if(externalTAZs.contains(currTAZ)) - continue; - - if(bestTapsMap.get(currTAZ).size() > 0){ - //NEW CSV FORMAT (tabular: TAZ, TAP, TIME, DIST, MODE) - for(int k = 0; k < bestTapsMap.get(currTAZ).size(); k++){ - writer.write( (int)(currTAZ) + "," + - (int)(bestTapsMap.get(currTAZ).get(k)[0]) + "," + - (double)(bestTapsMap.get(currTAZ).get(k)[1]) + "," + - (double)(bestTapsMap.get(currTAZ).get(k)[2]) + "," + - (int)(bestTapsMap.get(currTAZ).get(k)[3]) + "\n"); - } - } - } - }finally{ - if (writer != null) writer.close(); - } - } - - - /** - * @param args - */ - public static void main(String... args) throws Exception - { - HashMap pMap; - String propertiesFile = null; - - logger.info("Generating access**.prp files"); - if (args.length == 0) - { - logger.error(String.format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - - Properties properties = new Properties(); - properties.load(new FileInputStream("conf/sandag_abm.properties")); - - List definedTables = new ArrayList(); - for (String table : properties.getProperty("Report.tables").trim().split(",")) - definedTables.add(table.trim().toLowerCase()); - - String path = ClassLoader.getSystemResource("").getPath(); - path = path.substring(1, path.length() - 2); - String appPath = path.substring(0, path.lastIndexOf("/")); - - for (Object key : properties.keySet()) - { - String value = (String) properties.get(key); - properties.setProperty((String) key, value.replace(PROJECT_PATH_PROPERTY_TOKEN, appPath)); - } - - OMXMatrixDao mtxDao = new OMXMatrixDao(properties); - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - SandagMGRAtoPNR accessWriter = new SandagMGRAtoPNR(properties, mtxDao, appPath, pMap); - accessWriter.nearestTAPs(); - accessWriter.writeResults(properties.getProperty("taz.driveaccess.taps.file")); - } -} - diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java deleted file mode 100644 index ffe60cf..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagMicromobilityChoiceDMU.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; - -import org.sandag.abm.ctramp.MicromobilityChoiceDMU; - -public class SandagMicromobilityChoiceDMU - extends MicromobilityChoiceDMU -{ - - public SandagMicromobilityChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getIvtCoeff", 0); - methodIndexMap.put("getCostCoeff", 1); - methodIndexMap.put("getWalkTime", 2); - methodIndexMap.put("getIsTransit", 3); - methodIndexMap.put("getMicroTransitAvailable", 4); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getIvtCoeff(); - case 1: - return getCostCoeff(); - case 2: - return getWalkTime(); - case 3: - return isTransit()? 1 : 0; - case 4: - return isMicroTransitAvailable() ? 1 : 0; - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java deleted file mode 100644 index 774844c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagModelStructure.java +++ /dev/null @@ -1,1299 +0,0 @@ -package org.sandag.abm.application; - -import java.util.ArrayList; -import java.util.HashMap; - -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.Modes; - -public class SandagModelStructure - extends ModelStructure -{ - - public final String[] MANDATORY_DC_PURPOSE_NAMES = { - WORK_PURPOSE_NAME, UNIVERSITY_PURPOSE_NAME, SCHOOL_PURPOSE_NAME }; - public final String[] WORK_PURPOSE_SEGMENT_NAMES = { - "low", "med", "high", "very high", "part time" }; - public final String[] UNIVERSITY_PURPOSE_SEGMENT_NAMES = {}; - public final String[] SCHOOL_PURPOSE_SEGMENT_NAMES = { - "predrive", "drive" }; - - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_LO = 1; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_MD = 2; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_HI = 3; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_VHI = 4; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_PT = 5; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_UNIVERSITY_UNIVERSITY = 6; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_UNDER_SIXTEEN = 7; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_SIXTEEN_PLUS = 8; - - public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK = 1; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_UNIVERSITY = 2; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL = 3; - - public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK = 1; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_UNIVERSITY = 2; - public final int USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL = 3; - - public final int MANDATORY_STOP_FREQ_UEC_INDEX_WORK = 1; - public final int MANDATORY_STOP_FREQ_UEC_INDEX_UNIVERSITY = 2; - public final int MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL = 3; - - public final int MANDATORY_STOP_LOC_UEC_INDEX_WORK = 1; - public final int MANDATORY_STOP_LOC_UEC_INDEX_UNIVERSITY = 1; - public final int MANDATORY_STOP_LOC_UEC_INDEX_SCHOOL = 1; - - public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_WORK = 1; - public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_UNIVERSITY = 2; - public final int MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_SCHOOL = 3; - - public final String[] NON_MANDATORY_DC_PURPOSE_NAMES = { - "escort", "shopping", "eatOut", "othMaint", "visit", "othDiscr" }; - public final String[] ESCORT_PURPOSE_SEGMENT_NAMES = { - "kids", "no kids" }; - public final String[] SHOPPING_PURPOSE_SEGMENT_NAMES = {}; - public final String[] EAT_OUT_PURPOSE_SEGMENT_NAMES = {}; - public final String[] OTH_MAINT_PURPOSE_SEGMENT_NAMES = {}; - public final String[] SOCIAL_PURPOSE_SEGMENT_NAMES = {}; - public final String[] OTH_DISCR_PURPOSE_SEGMENT_NAMES = {}; - - /* - * public final int NON_MANDATORY_SOA_UEC_INDEX_ESCORT_KIDS = 9; public - * final int NON_MANDATORY_SOA_UEC_INDEX_ESCORT_NO_KIDS = 10; public final - * int NON_MANDATORY_SOA_UEC_INDEX_SHOPPING = 11; public final int - * NON_MANDATORY_SOA_UEC_INDEX_EAT_OUT = 12; public final int - * NON_MANDATORY_SOA_UEC_INDEX_OTHER_MAINT = 13; public final int - * NON_MANDATORY_SOA_UEC_INDEX_SOCIAL = 14; public final int - * NON_MANDATORY_SOA_UEC_INDEX_OTHER_DISCR = 15; - * - * public final int NON_MANDATORY_DC_UEC_INDEX_ESCORT_KIDS = 4; public final - * int NON_MANDATORY_DC_UEC_INDEX_ESCORT_NO_KIDS = 4; public final int - * NON_MANDATORY_DC_UEC_INDEX_SHOPPING = 5; public final int - * NON_MANDATORY_DC_UEC_INDEX_EAT_OUT = 6; public final int - * NON_MANDATORY_DC_UEC_INDEX_OTHER_MAINT = 7; public final int - * NON_MANDATORY_DC_UEC_INDEX_SOCIAL = 8; public final int - * NON_MANDATORY_DC_UEC_INDEX_OTHER_DISCR = 9; - * - * public final int NON_MANDATORY_MC_UEC_INDEX_ESCORT_KIDS = 4; public final - * int NON_MANDATORY_MC_UEC_INDEX_ESCORT_NO_KIDS = 4; public final int - * NON_MANDATORY_MC_UEC_INDEX_SHOPPING = 4; public final int - * NON_MANDATORY_MC_UEC_INDEX_EAT_OUT = 4; public final int - * NON_MANDATORY_MC_UEC_INDEX_OTHER_MAINT = 4; public final int - * NON_MANDATORY_MC_UEC_INDEX_SOCIAL = 4; public final int - * NON_MANDATORY_MC_UEC_INDEX_OTHER_DISCR = 4; - * - * public final int NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT = 4; public - * final int NON_MANDATORY_STOP_FREQ_UEC_INDEX_SHOPPING = 5; public final - * int NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_MAINT = 6; public final int - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_EAT_OUT = 7; public final int - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SOCIAL = 8; public final int - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_DISCR = 9; - * - * public final int NON_MANDATORY_STOP_LOC_UEC_INDEX_ESCORT = 2; public - * final int NON_MANDATORY_STOP_LOC_UEC_INDEX_SHOPPING = 3; public final int - * NON_MANDATORY_STOP_LOC_UEC_INDEX_EAT_OUT = 4; public final int - * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_MAINT = 5; public final int - * NON_MANDATORY_STOP_LOC_UEC_INDEX_SOCIAL = 6; public final int - * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_DISCR = 7; - * - * public final int NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX = 4; - */ - public final String[] AT_WORK_DC_PURPOSE_NAMES = {"atwork"}; - public final String[] AT_WORK_DC_SIZE_SEGMENT_NAMES = { - "cbd", "urban", "suburban", "rural" }; - - public final int AT_WORK_SOA_UEC_INDEX_EAT = 16; - public final int AT_WORK_SOA_UEC_INDEX_BUSINESS = 17; - public final int AT_WORK_SOA_UEC_INDEX_MAINT = 18; - - public final int AT_WORK_DC_UEC_INDEX_EAT = 10; - public final int AT_WORK_DC_UEC_INDEX_BUSINESS = 10; - public final int AT_WORK_DC_UEC_INDEX_MAINT = 10; - - public final int AT_WORK_MC_UEC_INDEX_EAT = 5; - public final int AT_WORK_MC_UEC_INDEX_BUSINESS = 5; - public final int AT_WORK_MC_UEC_INDEX_MAINT = 5; - - public final int SD_AT_WORK_PURPOSE_INDEX_EAT = 1; - public final int SD_AT_WORK_PURPOSE_INDEX_BUSINESS = 2; - public final int SD_AT_WORK_PURPOSE_INDEX_MAINT = 3; - - public final int AT_WORK_STOP_FREQ_UEC_INDEX_EAT = 9; - public final int AT_WORK_STOP_FREQ_UEC_INDEX_BUSINESS = 9; - public final int AT_WORK_STOP_FREQ_UEC_INDEX_MAINT = 9; - - // TODO: set these values from project specific code. - public static final int[] SOV_ALTS = { - 1 }; - public static final int[] HOV_ALTS = { - 2, 3 }; - public static final int[] HOV2_ALTS = { - 2 }; - public static final int[] HOV3_ALTS = { - 3 }; - public static final int[] WALK_ALTS = {4}; - public static final int[] BIKE_ALTS = {5}; - public static final int[] NON_MOTORIZED_ALTS = { - 4, 5 }; - public static final int[] TRANSIT_ALTS = { - 6, 7, 8, 9 }; - public static final int[] WALK_TRANSIT_ALTS = { - 6 }; - public static final int[] DRIVE_TRANSIT_ALTS = { - 7, 8, 9 }; - public static final int[] PNR_ALTS = { - 7 }; - public static final int[] KNR_ALTS = { - 8, 9 }; - - public static final int TNC_TRANSIT_ALT = 9; - - public static final int[] SCHOOL_BUS_ALTS = {13}; - public static final int[] TRIP_SOV_ALTS = { - 1 }; - public static final int[] TRIP_HOV_ALTS = { - 2,3 }; - - // public static final int[] PAY_ALTS = { - // 2, 4, 6 }; - - public static final int[] OTHER_ALTS = {10,11,12,13}; - - private static final int WALK = 4; - private static final int BIKE = 5; - - public static final int SCHOOL_BUS = 13; - public static final int TAXI = 10; - public static final int[] TNC_ALTS = {11,12}; - public static final int[] MAAS_ALTS = {10,11,12}; - - public static final String[] modeName = {"SOV","SR2","SR3", - "WALK","BIKE","WLK_SET","PNR_SET","KNR_SET","TNC_SET","TAXI","TNC_SINGLE","TNC_SHARED","SCHLBUS"}; - - public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 13; - - public final double[][] CDAP_6_PLUS_PROPORTIONS = { - {0.0, 0.0, 0.0}, {0.79647, 0.09368, 0.10985}, {0.61678, 0.25757, 0.12565}, - {0.69229, 0.15641, 0.15130}, {0.00000, 0.67169, 0.32831}, {0.00000, 0.54295, 0.45705}, - {0.77609, 0.06004, 0.16387}, {0.68514, 0.09144, 0.22342}, {0.14056, 0.06512, 0.79432} }; - - public static final String[] JTF_ALTERNATIVE_LABELS = { - "0_tours", "1_Shop", "1_Main", "1_Eat", "1_Visit", "1_Disc", "2_SS", "2_SM", "2_SE", - "2_SV", "2_SD", "2_MM", "2_ME", "2_MV", "2_MD", "2_EE", "2_EV", "2_ED", "2_VV", "2_VD", - "2_DD" }; - public static final String[] AWF_ALTERNATIVE_LABELS = { - "0_subTours", "1_eat", "1_business", "1_other", "2_business", "2 other", - "2_eat_business" }; - - public static final int MIN_DRIVING_AGE = 16; - - public static final int MAX_STOPS_PER_DIRECTION = 4; - - public SandagModelStructure() - { - super(); - - jtfAltLabels = JTF_ALTERNATIVE_LABELS; - awfAltLabels = AWF_ALTERNATIVE_LABELS; - - dcSizePurposeSegmentMap = new HashMap>(); - - dcSizeIndexSegmentMap = new HashMap(); - dcSizeSegmentIndexMap = new HashMap(); - dcSizeArrayIndexPurposeMap = new HashMap(); - dcSizeArrayPurposeIndexMap = new HashMap(); - - setMandatoryPurposeNameValues(); - - setUsualWorkAndSchoolLocationSoaUecSheetIndexValues(); - setUsualWorkAndSchoolLocationUecSheetIndexValues(); - setUsualWorkAndSchoolLocationModeChoiceUecSheetIndexValues(); - - setMandatoryStopFreqUecSheetIndexValues(); - setMandatoryStopLocUecSheetIndexValues(); - setMandatoryTripModeChoiceUecSheetIndexValues(); - - setNonMandatoryPurposeNameValues(); - - /* - * setNonMandatoryDcSoaUecSheetIndexValues(); - * setNonMandatoryDcUecSheetIndexValues(); - * setNonMandatoryModeChoiceUecSheetIndexValues(); - * - * setNonMandatoryStopFreqUecSheetIndexValues(); - * setNonMandatoryStopLocUecSheetIndexValues(); - * setNonMandatoryTripModeChoiceUecSheetIndexValues(); - */ - setAtWorkPurposeNameValues(); - - setAtWorkDcSoaUecSheetIndexValues(); - setAtWorkDcUecSheetIndexValues(); - setAtWorkModeChoiceUecSheetIndexValues(); - - setAtWorkStopFreqUecSheetIndexValues(); - - createDcSizePurposeSegmentMap(); - - // mapModelSegmentsToDcSizeArraySegments(); - - } - - /* - * private void mapModelSegmentsToDcSizeArraySegments() { - * - * Logger logger = Logger.getLogger(this.getClass()); - * - * dcSizeDcModelPurposeMap = new HashMap(); - * dcModelDcSizePurposeMap = new HashMap(); - * - * // loop over soa model names and map top dc size array indices for (int i - * = 0; i < dcModelPurposeIndexMap.size(); i++) { String modelSegment = - * dcModelIndexPurposeMap.get(i); - * - * // look for this modelSegment name in the dc size array names map, with - * // and without "_segment". if - * (dcSizeArrayPurposeIndexMap.containsKey(modelSegment)) { - * dcSizeDcModelPurposeMap.put(modelSegment, modelSegment); - * dcModelDcSizePurposeMap.put(modelSegment, modelSegment); } else { int - * underscoreIndex = modelSegment.indexOf('_'); if (underscoreIndex < 0) { - * if (dcSizeArrayPurposeIndexMap.containsKey(modelSegment + "_" + - * modelSegment)) { dcSizeDcModelPurposeMap .put(modelSegment + "_" + - * modelSegment, modelSegment); dcModelDcSizePurposeMap .put(modelSegment, - * modelSegment + "_" + modelSegment); } else { logger .error(String - * .format( - * "could not establish correspondence between DC SOA model purpose string = %s" - * , modelSegment)); - * logger.error(String.format("and a DC array purpose string:")); int j = 0; - * for (String key : dcSizeArrayPurposeIndexMap.keySet()) - * logger.error(String.format("%-2d: %s", ++j, key)); throw new - * RuntimeException(); } } else { // all at-work size segments should map to - * one model segment if (modelSegment.substring(0, - * underscoreIndex).equalsIgnoreCase( AT_WORK_PURPOSE_NAME)) { - * dcSizeDcModelPurposeMap.put(AT_WORK_PURPOSE_NAME + "_" + - * AT_WORK_PURPOSE_NAME, modelSegment); - * dcModelDcSizePurposeMap.put(modelSegment, AT_WORK_PURPOSE_NAME + "_" + - * AT_WORK_PURPOSE_NAME); } else { logger .error(String .format( - * "could not establish correspondence between DC SOA model purpose string = %s" - * , modelSegment)); - * logger.error(String.format("and a DC array purpose string:")); int j = 0; - * for (String key : dcSizeArrayPurposeIndexMap.keySet()) - * logger.error(String.format("%-2d: %s", ++j, key)); throw new - * RuntimeException(); } } } - * - * } - * - * } - */ - - public String getSchoolPurpose(int age) - { - if (age < MIN_DRIVING_AGE) return (schoolPurposeName + "_" + SCHOOL_PURPOSE_SEGMENT_NAMES[0]) - .toLowerCase(); - else return (schoolPurposeName + "_" + SCHOOL_PURPOSE_SEGMENT_NAMES[1]).toLowerCase(); - } - - public String getSchoolPurpose() - { - return schoolPurposeName.toLowerCase(); - } - - public String getUniversityPurpose() - { - return universityPurposeName.toLowerCase(); - } - - public String getWorkPurpose(int incomeCategory) - { - return getWorkPurpose(false, incomeCategory); - } - - public String getWorkPurpose(boolean isPtWorker, int incomeCategory) - { - if (isPtWorker) return (workPurposeName + "_" + WORK_PURPOSE_SEGMENT_NAMES[WORK_PURPOSE_SEGMENT_NAMES.length - 1]) - .toLowerCase(); - else return (workPurposeName + "_" + WORK_PURPOSE_SEGMENT_NAMES[incomeCategory - 1]) - .toLowerCase(); - } - - public boolean getTripModeIsSovOrHov(int tripMode) - { - - for (int i = 0; i < TRIP_SOV_ALTS.length; i++) - { - if (TRIP_SOV_ALTS[i] == tripMode) return true; - } - - for (int i = 0; i < TRIP_HOV_ALTS.length; i++) - { - if (TRIP_HOV_ALTS[i] == tripMode) return true; - } - - return false; - } - - public boolean getTourModeIsSov(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < SOV_ALTS.length; i++) - { - if (SOV_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsHov(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < HOV_ALTS.length; i++) - { - if (HOV_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsS2(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < HOV2_ALTS.length; i++) - { - if (HOV2_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsS3(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < HOV3_ALTS.length; i++) - { - if (HOV3_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsSovOrHov(int tourMode) - { - for (int i = 0; i < SOV_ALTS.length; i++) - { - if (SOV_ALTS[i] == tourMode) return true; - } - - for (int i = 0; i < HOV_ALTS.length; i++) - { - if (HOV_ALTS[i] == tourMode) return true; - } - - // if (tourMode == TAXI) return true; - - return false; - } - - public boolean getTourModeIsNonMotorized(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < NON_MOTORIZED_ALTS.length; i++) - { - if (NON_MOTORIZED_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsBike(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < BIKE_ALTS.length; i++) - { - if (BIKE_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsWalk(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < WALK_ALTS.length; i++) - { - if (WALK_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - - public boolean getTourModeIsTransit(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < TRANSIT_ALTS.length; i++) - { - if (TRANSIT_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsWalkTransit(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < WALK_TRANSIT_ALTS.length; ++i) - { - if (WALK_TRANSIT_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsDriveTransit(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < DRIVE_TRANSIT_ALTS.length; i++) - { - if (DRIVE_TRANSIT_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsPnr(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < PNR_ALTS.length; i++) - { - if (PNR_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsKnr(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < KNR_ALTS.length; i++) - { - if (KNR_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTourModeIsTncTransit(int tourMode) - { - if (TNC_TRANSIT_ALT == tourMode) - return true; - else - return false; - } - - public boolean getTourModeIsMaas(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < MAAS_ALTS.length; i++) - { - if (MAAS_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - public boolean getTourModeIsSchoolBus(int tourMode) - { - boolean returnValue = false; - for (int i = 0; i < SCHOOL_BUS_ALTS.length; i++) - { - if (SCHOOL_BUS_ALTS[i] == tourMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public static boolean getTripModeIsPay(int tripMode) - { - boolean returnValue = false; - /* for (int i = 0; i < PAY_ALTS.length; i++) - { - if (PAY_ALTS[i] == tripMode) - { - returnValue = true; - break; - } - } -*/ - return returnValue; - } - - public boolean getTripModeIsTransit(int tripMode){ - - boolean returnValue = false; - for( int i = 0; i < TRANSIT_ALTS.length;++i ) - if(tripMode==TRANSIT_ALTS[i]){ - returnValue = true; - break; - } - - return returnValue; - } - - /** - * Get the name of the mode - * - * @param mode - * The mode index - * @return The name of the mode - */ - public String getModeName(int mode) - { - - return modeName[mode - 1]; - - } - - private int createPurposeIndexMaps(String purposeName, String[] segmentNames, int index, - String categoryString) - { - - HashMap segmentMap = new HashMap(); - String key = ""; - if (segmentNames.length > 0) - { - for (int i = 0; i < segmentNames.length; i++) - { - segmentMap.put(segmentNames[i].toLowerCase(), i); - key = purposeName.toLowerCase() + "_" + segmentNames[i].toLowerCase(); - dcSizeIndexSegmentMap.put(index, key); - dcSizeSegmentIndexMap.put(key, index++); - } - } else - { - segmentMap.put(purposeName.toLowerCase(), 0); - key = purposeName.toLowerCase() + "_" + purposeName.toLowerCase(); - dcSizeIndexSegmentMap.put(index, key); - dcSizeSegmentIndexMap.put(key, index++); - } - dcSizePurposeSegmentMap.put(purposeName.toLowerCase(), segmentMap); - - return index; - - } - - /** - * This method defines the segmentation for which destination choice size - * variables are calculated. - */ - private void createDcSizePurposeSegmentMap() - { - - int index = 0; - - // put work purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(WORK_PURPOSE_NAME, WORK_PURPOSE_SEGMENT_NAMES, index, - MANDATORY_CATEGORY); - - // put university purpose segments, by which DC Size calculations are - // segmented, into a map to be stored by purpose name. - index = createPurposeIndexMaps(UNIVERSITY_PURPOSE_NAME, UNIVERSITY_PURPOSE_SEGMENT_NAMES, - index, MANDATORY_CATEGORY); - - // put school purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(SCHOOL_PURPOSE_NAME, SCHOOL_PURPOSE_SEGMENT_NAMES, index, - MANDATORY_CATEGORY); - - // put escort purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(ESCORT_PURPOSE_NAME, ESCORT_PURPOSE_SEGMENT_NAMES, index, - INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put shopping purpose segments, by which DC Size calculations are - // segmented, into a map to be stored by purpose name. - index = createPurposeIndexMaps(SHOPPING_PURPOSE_NAME, SHOPPING_PURPOSE_SEGMENT_NAMES, - index, INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put eat out purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(EAT_OUT_PURPOSE_NAME, EAT_OUT_PURPOSE_SEGMENT_NAMES, index, - INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put oth main purpose segments, by which DC Size calculations are - // segmented, into a map to be stored by purpose name. - index = createPurposeIndexMaps(OTH_MAINT_PURPOSE_NAME, OTH_MAINT_PURPOSE_SEGMENT_NAMES, - index, INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put social purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(SOCIAL_PURPOSE_NAME, SOCIAL_PURPOSE_SEGMENT_NAMES, index, - INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put oth discr purpose segments, by which DC Size calculations are - // segmented, into a map to be stored by purpose name. - index = createPurposeIndexMaps(OTH_DISCR_PURPOSE_NAME, OTH_DISCR_PURPOSE_SEGMENT_NAMES, - index, INDIVIDUAL_NON_MANDATORY_CATEGORY); - - // put at work purpose segments, by which DC Size calculations are - // segmented, - // into a map to be stored by purpose name. - index = createPurposeIndexMaps(AT_WORK_PURPOSE_NAME, AT_WORK_DC_SIZE_SEGMENT_NAMES, index, - AT_WORK_CATEGORY); - - } - - public HashMap> getDcSizePurposeSegmentMap() - { - return dcSizePurposeSegmentMap; - } - - private void setMandatoryPurposeNameValues() - { - - int index = 0; - - WORK_PURPOSE_NAME = "work"; - UNIVERSITY_PURPOSE_NAME = "university"; - SCHOOL_PURPOSE_NAME = "school"; - - int numDcSizePurposeSegments = 0; - if (WORK_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += WORK_PURPOSE_SEGMENT_NAMES.length; - else numDcSizePurposeSegments += 1; - if (UNIVERSITY_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += UNIVERSITY_PURPOSE_SEGMENT_NAMES.length; - else numDcSizePurposeSegments += 1; - if (SCHOOL_PURPOSE_SEGMENT_NAMES.length > 0) numDcSizePurposeSegments += SCHOOL_PURPOSE_SEGMENT_NAMES.length; - else numDcSizePurposeSegments += 1; - - mandatoryDcModelPurposeNames = new String[numDcSizePurposeSegments]; - - workPurposeName = WORK_PURPOSE_NAME.toLowerCase(); - workPurposeSegmentNames = new String[WORK_PURPOSE_SEGMENT_NAMES.length]; - if (workPurposeSegmentNames.length > 0) - { - for (int i = 0; i < WORK_PURPOSE_SEGMENT_NAMES.length; i++) - { - workPurposeSegmentNames[i] = WORK_PURPOSE_SEGMENT_NAMES[i].toLowerCase(); - mandatoryDcModelPurposeNames[index] = workPurposeName + "_" - + workPurposeSegmentNames[i]; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each work - // purpose_segment - dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - index++; - } - } else - { - mandatoryDcModelPurposeNames[index] = workPurposeName; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each work purpose_segment - String name = mandatoryDcModelPurposeNames[index] + "_" - + mandatoryDcModelPurposeNames[index]; - dcSizeArrayIndexPurposeMap.put(index, name); - dcSizeArrayPurposeIndexMap.put(name, index); - index++; - } - - universityPurposeName = UNIVERSITY_PURPOSE_NAME.toLowerCase(); - universityPurposeSegmentNames = new String[UNIVERSITY_PURPOSE_SEGMENT_NAMES.length]; - if (universityPurposeSegmentNames.length > 0) - { - for (int i = 0; i < universityPurposeSegmentNames.length; i++) - { - universityPurposeSegmentNames[i] = UNIVERSITY_PURPOSE_SEGMENT_NAMES[i] - .toLowerCase(); - mandatoryDcModelPurposeNames[index] = universityPurposeName + "_" - + universityPurposeSegmentNames[i]; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each university - // purpose_segment - dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - index++; - } - } else - { - mandatoryDcModelPurposeNames[index] = universityPurposeName; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each university - // purpose_segment - String name = mandatoryDcModelPurposeNames[index] + "_" - + mandatoryDcModelPurposeNames[index]; - dcSizeArrayIndexPurposeMap.put(index, name); - dcSizeArrayPurposeIndexMap.put(name, index); - index++; - } - - schoolPurposeName = SCHOOL_PURPOSE_NAME.toLowerCase(); - schoolPurposeSegmentNames = new String[SCHOOL_PURPOSE_SEGMENT_NAMES.length]; - if (schoolPurposeSegmentNames.length > 0) - { - for (int i = 0; i < schoolPurposeSegmentNames.length; i++) - { - schoolPurposeSegmentNames[i] = SCHOOL_PURPOSE_SEGMENT_NAMES[i].toLowerCase(); - mandatoryDcModelPurposeNames[index] = schoolPurposeName + "_" - + schoolPurposeSegmentNames[i]; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each school - // purpose_segment - dcSizeArrayIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - dcSizeArrayPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - index++; - } - } else - { - mandatoryDcModelPurposeNames[index] = schoolPurposeName; - dcModelPurposeIndexMap.put(mandatoryDcModelPurposeNames[index], index); - dcModelIndexPurposeMap.put(index, mandatoryDcModelPurposeNames[index]); - - // a separate size term is calculated for each school - // purpose_segment - String name = mandatoryDcModelPurposeNames[index] + "_" - + mandatoryDcModelPurposeNames[index]; - dcSizeArrayIndexPurposeMap.put(index, name); - dcSizeArrayPurposeIndexMap.put(name, index); - } - - } - - private void setUsualWorkAndSchoolLocationSoaUecSheetIndexValues() - { - dcSoaUecIndexMap.put("work_low", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_LO); - dcSoaUecIndexMap.put("work_med", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_MD); - dcSoaUecIndexMap.put("work_high", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_HI); - dcSoaUecIndexMap.put("work_very high", - USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_VHI); - dcSoaUecIndexMap - .put("work_part time", USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_WORK_PT); - dcSoaUecIndexMap.put("university", - USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_UNIVERSITY_UNIVERSITY); - dcSoaUecIndexMap.put("school_predrive", - USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_UNDER_SIXTEEN); - dcSoaUecIndexMap.put("school_drive", - USUAL_WORK_AND_SCHOOL_LOCATION_SOA_UEC_INDEX_SCHOOL_SIXTEEN_PLUS); - } - - private void setUsualWorkAndSchoolLocationUecSheetIndexValues() - { - dcUecIndexMap.put("work_low", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); - dcUecIndexMap.put("work_med", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); - dcUecIndexMap.put("work_high", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); - dcUecIndexMap.put("work_very high", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); - dcUecIndexMap.put("work_part time", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_WORK); - dcUecIndexMap.put("university", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_UNIVERSITY); - dcUecIndexMap.put("school_predrive", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL); - dcUecIndexMap.put("school_drive", USUAL_WORK_AND_SCHOOL_LOCATION_UEC_INDEX_SCHOOL); - } - - private void setUsualWorkAndSchoolLocationModeChoiceUecSheetIndexValues() - { - tourModeChoiceUecIndexMap.put("work_low", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); - tourModeChoiceUecIndexMap.put("work_med", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); - tourModeChoiceUecIndexMap.put("work_high", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); - tourModeChoiceUecIndexMap.put("work_very high", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); - tourModeChoiceUecIndexMap.put("work_part time", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_WORK); - tourModeChoiceUecIndexMap.put("university", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_UNIVERSITY); - tourModeChoiceUecIndexMap.put("school_predrive", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL); - tourModeChoiceUecIndexMap.put("school_drive", - USUAL_WORK_AND_SCHOOL_LOCATION_MODE_CHOICE_UEC_INDEX_SCHOOL); - } - - private void setMandatoryStopFreqUecSheetIndexValues() - { - stopFreqUecIndexMap.put("work_low", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); - stopFreqUecIndexMap.put("work_med", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); - stopFreqUecIndexMap.put("work_high", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); - stopFreqUecIndexMap.put("work_very high", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); - stopFreqUecIndexMap.put("work_part time", MANDATORY_STOP_FREQ_UEC_INDEX_WORK); - stopFreqUecIndexMap.put("university", MANDATORY_STOP_FREQ_UEC_INDEX_UNIVERSITY); - stopFreqUecIndexMap.put("school_predrive", MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL); - stopFreqUecIndexMap.put("school_drive", MANDATORY_STOP_FREQ_UEC_INDEX_SCHOOL); - } - - private void setMandatoryStopLocUecSheetIndexValues() - { - stopLocUecIndexMap.put(WORK_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); - stopLocUecIndexMap.put(UNIVERSITY_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); - stopLocUecIndexMap.put(SCHOOL_PURPOSE_NAME, MANDATORY_STOP_LOC_UEC_INDEX_WORK); - } - - private void setMandatoryTripModeChoiceUecSheetIndexValues() - { - tripModeChoiceUecIndexMap.put(WORK_PURPOSE_NAME, MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_WORK); - tripModeChoiceUecIndexMap.put(UNIVERSITY_PURPOSE_NAME, - MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_UNIVERSITY); - tripModeChoiceUecIndexMap.put(SCHOOL_PURPOSE_NAME, - MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX_SCHOOL); - } - - private void setNonMandatoryPurposeNameValues() - { - - ESCORT_PURPOSE_NAME = "escort"; - SHOPPING_PURPOSE_NAME = "shopping"; - EAT_OUT_PURPOSE_NAME = "eatout"; - OTH_MAINT_PURPOSE_NAME = "othmaint"; - SOCIAL_PURPOSE_NAME = "visit"; - OTH_DISCR_PURPOSE_NAME = "othdiscr"; - - // initialize index to the length of the mandatory names list already - // developed. - int index = dcSizeArrayPurposeIndexMap.size(); - - ESCORT_SEGMENT_NAMES = ESCORT_PURPOSE_SEGMENT_NAMES; - - // ESCORT is the only non-mandatory purpose with segments - ArrayList purposeNamesList = new ArrayList(); - for (int i = 0; i < NON_MANDATORY_DC_PURPOSE_NAMES.length; i++) - { - if (NON_MANDATORY_DC_PURPOSE_NAMES[i].equalsIgnoreCase(ESCORT_PURPOSE_NAME)) - { - for (int j = 0; j < ESCORT_SEGMENT_NAMES.length; j++) - { - String name = (ESCORT_PURPOSE_NAME + "_" + ESCORT_SEGMENT_NAMES[j]) - .toLowerCase(); - purposeNamesList.add(name); - dcModelPurposeIndexMap.put(name, index); - dcModelIndexPurposeMap.put(index, name); - - // a separate size term is calculated for each non-mandatory - // purpose_segment - dcSizeArrayIndexPurposeMap.put(index, name); - dcSizeArrayPurposeIndexMap.put(name, index); - index++; - } - } else - { - String name = NON_MANDATORY_DC_PURPOSE_NAMES[i].toLowerCase(); - purposeNamesList.add(name); - dcModelPurposeIndexMap.put(name, index); - dcModelIndexPurposeMap.put(index, name); - - // a separate size term is calculated for each non-mandatory - // purpose_segment - dcSizeArrayIndexPurposeMap.put(index, name + "_" + name); - dcSizeArrayPurposeIndexMap.put(name + "_" + name, index); - index++; - } - } - - int escortOffset = ESCORT_SEGMENT_NAMES.length; - - jointDcModelPurposeNames = new String[purposeNamesList.size() - escortOffset]; - nonMandatoryDcModelPurposeNames = new String[purposeNamesList.size()]; - for (int i = 0; i < purposeNamesList.size(); i++) - { - nonMandatoryDcModelPurposeNames[i] = purposeNamesList.get(i); - if (i > escortOffset - 1) - jointDcModelPurposeNames[i - escortOffset] = purposeNamesList.get(i); - } - - } - - /* - * private void setNonMandatoryDcSoaUecSheetIndexValues() { - * dcSoaUecIndexMap.put("escort_kids", - * NON_MANDATORY_SOA_UEC_INDEX_ESCORT_KIDS); - * dcSoaUecIndexMap.put("escort_no kids", - * NON_MANDATORY_SOA_UEC_INDEX_ESCORT_NO_KIDS); - * dcSoaUecIndexMap.put("shopping", NON_MANDATORY_SOA_UEC_INDEX_SHOPPING); - * dcSoaUecIndexMap.put("eatout", NON_MANDATORY_SOA_UEC_INDEX_EAT_OUT); - * dcSoaUecIndexMap.put("othmaint", - * NON_MANDATORY_SOA_UEC_INDEX_OTHER_MAINT); dcSoaUecIndexMap.put("social", - * NON_MANDATORY_SOA_UEC_INDEX_SOCIAL); dcSoaUecIndexMap.put("othdiscr", - * NON_MANDATORY_SOA_UEC_INDEX_OTHER_DISCR); } - * - * private void setNonMandatoryDcUecSheetIndexValues() { - * dcUecIndexMap.put("escort_kids", NON_MANDATORY_DC_UEC_INDEX_ESCORT_KIDS); - * dcUecIndexMap.put("escort_no kids", - * NON_MANDATORY_DC_UEC_INDEX_ESCORT_NO_KIDS); dcUecIndexMap.put("shopping", - * NON_MANDATORY_DC_UEC_INDEX_SHOPPING); dcUecIndexMap.put("eatout", - * NON_MANDATORY_DC_UEC_INDEX_EAT_OUT); dcUecIndexMap.put("othmaint", - * NON_MANDATORY_DC_UEC_INDEX_OTHER_MAINT); dcUecIndexMap.put("social", - * NON_MANDATORY_DC_UEC_INDEX_SOCIAL); dcUecIndexMap.put("othdiscr", - * NON_MANDATORY_DC_UEC_INDEX_OTHER_DISCR); } - * - * private void setNonMandatoryModeChoiceUecSheetIndexValues() { - * tourModeChoiceUecIndexMap.put("escort_kids", - * NON_MANDATORY_MC_UEC_INDEX_ESCORT_KIDS); - * tourModeChoiceUecIndexMap.put("escort_no kids", - * NON_MANDATORY_MC_UEC_INDEX_ESCORT_NO_KIDS); - * tourModeChoiceUecIndexMap.put("shopping", - * NON_MANDATORY_MC_UEC_INDEX_SHOPPING); - * tourModeChoiceUecIndexMap.put("eatout", - * NON_MANDATORY_MC_UEC_INDEX_EAT_OUT); - * tourModeChoiceUecIndexMap.put("othmaint", - * NON_MANDATORY_MC_UEC_INDEX_OTHER_MAINT); - * tourModeChoiceUecIndexMap.put("social", - * NON_MANDATORY_MC_UEC_INDEX_SOCIAL); - * tourModeChoiceUecIndexMap.put("othdiscr", - * NON_MANDATORY_MC_UEC_INDEX_OTHER_DISCR); } - * - * private void setNonMandatoryStopFreqUecSheetIndexValues() { - * stopFreqUecIndexMap.put("escort_kids", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT); - * stopFreqUecIndexMap.put("escort_no kids", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_ESCORT); - * stopFreqUecIndexMap.put("shopping", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SHOPPING); - * stopFreqUecIndexMap.put("eatout", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_EAT_OUT); - * stopFreqUecIndexMap.put("othmaint", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_MAINT); - * stopFreqUecIndexMap.put("social", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_SOCIAL); - * stopFreqUecIndexMap.put("othdiscr", - * NON_MANDATORY_STOP_FREQ_UEC_INDEX_OTHER_DISCR); } - * - * private void setNonMandatoryStopLocUecSheetIndexValues() { - * stopLocUecIndexMap.put(ESCORT_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_ESCORT); - * stopLocUecIndexMap.put(SHOPPING_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_SHOPPING); - * stopLocUecIndexMap.put(EAT_OUT_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_EAT_OUT); stopLocUecIndexMap - * .put(OTH_MAINT_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_MAINT); - * stopLocUecIndexMap.put(SOCIAL_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_SOCIAL); stopLocUecIndexMap - * .put(OTH_DISCR_PURPOSE_NAME, - * NON_MANDATORY_STOP_LOC_UEC_INDEX_OTHER_DISCR); } - * - * private void setNonMandatoryTripModeChoiceUecSheetIndexValues() { - * tripModeChoiceUecIndexMap .put(ESCORT_PURPOSE_NAME, - * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); - * tripModeChoiceUecIndexMap.put(SHOPPING_PURPOSE_NAME, - * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); - * tripModeChoiceUecIndexMap.put(EAT_OUT_PURPOSE_NAME, - * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); - * tripModeChoiceUecIndexMap.put(OTH_MAINT_PURPOSE_NAME, - * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); tripModeChoiceUecIndexMap - * .put(SOCIAL_PURPOSE_NAME, NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); - * tripModeChoiceUecIndexMap.put(OTH_DISCR_PURPOSE_NAME, - * NON_MANDATORY_TRIP_MODE_CHOICE_UEC_INDEX); } - */ - private void setAtWorkPurposeNameValues() - { - - AT_WORK_PURPOSE_NAME = "atwork"; - - AT_WORK_EAT_PURPOSE_NAME = "eat"; - AT_WORK_BUSINESS_PURPOSE_NAME = "business"; - AT_WORK_MAINT_PURPOSE_NAME = "other"; - - AT_WORK_PURPOSE_INDEX_EAT = SD_AT_WORK_PURPOSE_INDEX_EAT; - AT_WORK_PURPOSE_INDEX_BUSINESS = SD_AT_WORK_PURPOSE_INDEX_BUSINESS; - AT_WORK_PURPOSE_INDEX_MAINT = SD_AT_WORK_PURPOSE_INDEX_MAINT; - - AT_WORK_SEGMENT_NAMES = new String[3]; - AT_WORK_SEGMENT_NAMES[0] = AT_WORK_EAT_PURPOSE_NAME; - AT_WORK_SEGMENT_NAMES[1] = AT_WORK_BUSINESS_PURPOSE_NAME; - AT_WORK_SEGMENT_NAMES[2] = AT_WORK_MAINT_PURPOSE_NAME; - - // initialize index to the length of the home-based tour names list - // already - // developed. - int index = dcSizeArrayPurposeIndexMap.size(); - - // the same size term is used by each at-work soa model - dcSizeArrayIndexPurposeMap.put(index, AT_WORK_PURPOSE_NAME + "_" + AT_WORK_PURPOSE_NAME); - dcSizeArrayPurposeIndexMap.put(AT_WORK_PURPOSE_NAME + "_" + AT_WORK_PURPOSE_NAME, index); - - ArrayList purposeNamesList = new ArrayList(); - for (int j = 0; j < AT_WORK_SEGMENT_NAMES.length; j++) - { - String name = (AT_WORK_PURPOSE_NAME + "_" + AT_WORK_SEGMENT_NAMES[j]).toLowerCase(); - purposeNamesList.add(name); - dcModelPurposeIndexMap.put(name, index); - dcModelIndexPurposeMap.put(index, name); - index++; - } - - atWorkDcModelPurposeNames = new String[purposeNamesList.size()]; - for (int i = 0; i < purposeNamesList.size(); i++) - { - atWorkDcModelPurposeNames[i] = purposeNamesList.get(i); - } - - } - - private void setAtWorkDcSoaUecSheetIndexValues() - { - dcSoaUecIndexMap.put("atwork_eat", AT_WORK_SOA_UEC_INDEX_EAT); - dcSoaUecIndexMap.put("atwork_business", AT_WORK_SOA_UEC_INDEX_BUSINESS); - dcSoaUecIndexMap.put("atwork_other", AT_WORK_SOA_UEC_INDEX_MAINT); - } - - private void setAtWorkDcUecSheetIndexValues() - { - dcUecIndexMap.put("atwork_eat", AT_WORK_DC_UEC_INDEX_EAT); - dcUecIndexMap.put("atwork_business", AT_WORK_DC_UEC_INDEX_BUSINESS); - dcUecIndexMap.put("atwork_other", AT_WORK_DC_UEC_INDEX_MAINT); - } - - private void setAtWorkModeChoiceUecSheetIndexValues() - { - tourModeChoiceUecIndexMap.put("atwork_eat", AT_WORK_MC_UEC_INDEX_EAT); - tourModeChoiceUecIndexMap.put("atwork_business", AT_WORK_MC_UEC_INDEX_BUSINESS); - tourModeChoiceUecIndexMap.put("atwork_other", AT_WORK_MC_UEC_INDEX_MAINT); - } - - private void setAtWorkStopFreqUecSheetIndexValues() - { - stopFreqUecIndexMap.put("atwork_eat", AT_WORK_STOP_FREQ_UEC_INDEX_EAT); - stopFreqUecIndexMap.put("atwork_business", AT_WORK_STOP_FREQ_UEC_INDEX_BUSINESS); - stopFreqUecIndexMap.put("atwork_other", AT_WORK_STOP_FREQ_UEC_INDEX_MAINT); - } - - public double[][] getCdap6PlusProps() - { - return CDAP_6_PLUS_PROPORTIONS; - } - - public String getModelPeriodLabel(int period) - { - return MODEL_PERIOD_LABELS[period]; - } - - public int getNumberModelPeriods() - { - return MODEL_PERIOD_LABELS.length; - } - - public String getSkimMatrixPeriodString(int period) - { - int index = getSkimPeriodIndex(period); - return SKIM_PERIOD_STRINGS[index]; - } - - public int getDefaultAmPeriod() - { - return getTimePeriodIndexForTime(800); - } - - public int getDefaultPmPeriod() - { - return getTimePeriodIndexForTime(1700); - } - - public int getDefaultMdPeriod() - { - return getTimePeriodIndexForTime(1400); - } - - public int[] getSkimPeriodCombinationIndices() - { - return SKIM_PERIOD_COMBINATION_INDICES; - } - - public int getSkimPeriodCombinationIndex(int startPeriod, int endPeriod) - { - - int startPeriodIndex = getSkimPeriodIndex(startPeriod); - int endPeriodIndex = getSkimPeriodIndex(endPeriod); - - if (SKIM_PERIOD_COMBINATIONS[startPeriodIndex][endPeriodIndex] < 0) - { - String errorString = String - .format("startPeriod=%d, startPeriod=%d, endPeriod=%d, endPeriod=%d is invalid combination.", - startPeriod, startPeriodIndex, endPeriod, endPeriodIndex); - throw new RuntimeException(errorString); - } else - { - return SKIM_PERIOD_COMBINATIONS[startPeriodIndex][endPeriodIndex]; - } - - } - - public int getMaxTourModeIndex() - { - return MAXIMUM_TOUR_MODE_ALT_INDEX; - } - - public HashMap getWorkSegmentNameIndexMap() - { - return workSegmentNameIndexMap; - } - - public void setWorkSegmentNameIndexMap(HashMap argMap) - { - workSegmentNameIndexMap = argMap; - } - - public HashMap getSchoolSegmentNameIndexMap() - { - return schoolSegmentNameIndexMap; - } - - public void setSchoolSegmentNameIndexMap(HashMap argMap) - { - schoolSegmentNameIndexMap = argMap; - } - - public HashMap getWorkSegmentIndexNameMap() - { - return workSegmentIndexNameMap; - } - - public void setWorkSegmentIndexNameMap(HashMap argMap) - { - workSegmentIndexNameMap = argMap; - } - - public HashMap getSchoolSegmentIndexNameMap() - { - return schoolSegmentIndexNameMap; - } - - public void setSchoolSegmentIndexNameMap(HashMap argMap) - { - schoolSegmentIndexNameMap = argMap; - } - - public void setJtfAltLabels(String[] labels) - { - jtfAltLabels = labels; - } - - public String[] getJtfAltLabels() - { - return jtfAltLabels; - } - - public boolean getTripModeIsWalkTransit(int tripMode) - { - - for (int i = 0; i < WALK_TRANSIT_ALTS.length; i++) - { - if (WALK_TRANSIT_ALTS[i] == tripMode) return true; - } - - return false; - } - - public boolean getTripModeIsPnrTransit(int tripMode) - { - - for (int i = 0; i < PNR_ALTS.length; i++) - { - if (PNR_ALTS[i] == tripMode) return true; - } - - return false; - } - - public boolean getTripModeIsKnrTransit(int tripMode) - { - - for (int i = 0; i < KNR_ALTS.length; i++) - { - if (KNR_ALTS[i] == tripMode) return true; - } - - return false; - } - - public boolean getTripModeIsNonMotorized(int i) - { - - if (i == WALK || i == BIKE) return true; - else return false; - } - - public boolean getTripModeIsS2(int tripMode) - { - boolean returnValue = false; - for (int i = 0; i < HOV2_ALTS.length; i++) - { - if (HOV2_ALTS[i] == tripMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public boolean getTripModeIsS3(int tripMode) - { - boolean returnValue = false; - for (int i = 0; i < HOV3_ALTS.length; i++) - { - if (HOV3_ALTS[i] == tripMode) - { - returnValue = true; - break; - } - } - return returnValue; - } - - public int getMaxStopsPerDirection(){ - - return MAX_STOPS_PER_DIRECTION; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java deleted file mode 100644 index 99fb216..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingChoiceDMU.java +++ /dev/null @@ -1,94 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.ParkingChoiceDMU; - -public class SandagParkingChoiceDMU - extends ParkingChoiceDMU -{ - - public SandagParkingChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getParkMgraAlt", 1); - methodIndexMap.put("getDistanceTripOrigToParkAlt", 2); - methodIndexMap.put("getDistanceTripDestFromParkAlt", 3); - methodIndexMap.put("getDestSameAsParkAlt", 4); - methodIndexMap.put("getPersonType", 5); - methodIndexMap.put("getActivityIntervals", 6); - methodIndexMap.put("getTripDestPurpose", 7); - methodIndexMap.put("getLsWgtAvgCostM", 8); - methodIndexMap.put("getMstallsoth", 9); - methodIndexMap.put("getMstallssam", 10); - methodIndexMap.put("getMparkcost", 11); - methodIndexMap.put("getDstallsoth", 12); - methodIndexMap.put("getDstallssam", 13); - methodIndexMap.put("getDparkcost", 14); - methodIndexMap.put("getHstallsoth", 15); - methodIndexMap.put("getHstallssam", 16); - methodIndexMap.put("getHparkcost", 17); - methodIndexMap.put("getNumfreehrs", 18); - methodIndexMap.put("getReimbPct", 19); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - - case 1: - return getParkMgraAlt(arrayIndex); - case 2: - return getDistanceTripOrigToParkAlt(arrayIndex); - case 3: - return getDistanceTripDestFromParkAlt(arrayIndex); - case 4: - return getDestSameAsParkAlt(arrayIndex); - case 5: - return getPersonType(); - case 6: - return getActivityIntervals(); - case 7: - return getTripDestPurpose(); - case 8: - return getLsWgtAvgCostM(arrayIndex); - case 9: - return getMstallsoth(arrayIndex); - case 10: - return getMstallssam(arrayIndex); - case 11: - return getMparkcost(arrayIndex); - case 12: - return getDstallsoth(arrayIndex); - case 13: - return getDstallssam(arrayIndex); - case 14: - return getDparkcost(arrayIndex); - case 15: - return getHstallsoth(arrayIndex); - case 16: - return getHstallssam(arrayIndex); - case 17: - return getHparkcost(arrayIndex); - case 18: - return getNumfreehrs(arrayIndex); - case 19: - return getReimbPct(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java deleted file mode 100644 index a3c242e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagParkingProvisionChoiceDMU.java +++ /dev/null @@ -1,83 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.ParkingProvisionChoiceDMU; - -public class SandagParkingProvisionChoiceDMU - extends ParkingProvisionChoiceDMU -{ - - public SandagParkingProvisionChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getHhIncomeInDollars", 0); - methodIndexMap.put("getWorkLocMgra", 1); - methodIndexMap.put("getLsWgtAvgCostM", 2); - methodIndexMap.put("getLsWgtAvgCostD", 3); - methodIndexMap.put("getLsWgtAvgCostH", 4); - methodIndexMap.put("getMgraParkArea", 5); - methodIndexMap.put("getNumFreeHours", 6); - methodIndexMap.put("getMStallsOth", 7); - methodIndexMap.put("getMStallsSam", 8); - methodIndexMap.put("getMParkCost", 9); - methodIndexMap.put("getDStallsOth", 10); - methodIndexMap.put("getDStallsSam", 11); - methodIndexMap.put("getDParkCost", 12); - methodIndexMap.put("getHStallsOth", 13); - methodIndexMap.put("getHStallsSam", 14); - methodIndexMap.put("getHParkCost", 15); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getIncomeInDollars(); - case 1: - return getWorkLocMgra(); - case 2: - return getLsWgtAvgCostM(); - case 3: - return getLsWgtAvgCostD(); - case 4: - return getLsWgtAvgCostH(); - case 5: - return getMgraParkArea(); - case 6: - return getNumFreeHours(); - case 7: - return getMStallsOth(); - case 8: - return getMStallsSam(); - case 9: - return getMParkCost(); - case 10: - return getDStallsOth(); - case 11: - return getDStallsSam(); - case 12: - return getDParkCost(); - case 13: - return getHStallsOth(); - case 14: - return getHStallsSam(); - case 15: - return getHParkCost(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java deleted file mode 100644 index 1d61c60..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSamplePopulationGenerator.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import com.pb.common.util.ResourceUtil; - -public final class SandagSamplePopulationGenerator -{ - - private static Logger logger = Logger.getLogger(SandagSamplePopulationGenerator.class); - - public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; - private ResourceBundle rb; - - /** - * - * @param rb - * , java.util.ResourceBundle containing environment settings - * from a properties file specified on the command line - * @param baseName - * , String containing basename (without .properites) from which - * ResourceBundle was created. - * @param globalIterationNumber - * , int iteration number for which the model is run, set by - * another process controlling a model stream with feedback. - * @param iterationSampleRate - * , float percentage [0.0, 1.0] inicating the portion of all - * households to be modeled. - * - * This object defines the implementation of the ARC tour based, - * activity based travel demand model. - */ - private SandagSamplePopulationGenerator(ResourceBundle rb) - { - this.rb = rb; - } - - private void generateSampleFiles() - { - - // new a ctramp application object - HashMap propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - - // create a new local instance of the household array manager - SandagHouseholdDataManager householdDataManager = new SandagHouseholdDataManager(); - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population files - // and - // apply its tables to objects mapping method. - String inputHouseholdFileName = "data/inputs/hhfile.csv"; - String inputPersonFileName = "data/inputs/personfile.csv"; - householdDataManager.setHouseholdSampleRate(1.0f, 0); - - SandagModelStructure modelStructure = new SandagModelStructure(); - - householdDataManager.setModelStructure(modelStructure); - householdDataManager.readPopulationFiles(inputHouseholdFileName, inputPersonFileName); - householdDataManager.mapTablesToHouseholdObjects(); - - householdDataManager.createSamplePopulationFiles( - "/jim/projects/sandag/data/inputs/hhfile.csv", - "/jim/projects/sandag/data/inputs/personfile.csv", - "/jim/projects/sandag/data/inputs/hhfile_1000.csv", - "/jim/projects/sandag/data/inputs/personfile_1000.csv", 1000); - } - - public static void main(String[] args) - { - - ResourceBundle rb = null; - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - } - - // create an instance of this class for main() to use. - SandagSamplePopulationGenerator mainObject = new SandagSamplePopulationGenerator(rb); - - // run tour based models - try - { - mainObject.generateSampleFiles(); - } catch (RuntimeException e) - { - logger.error( - "RuntimeException caught in SandagSamplePopulationGenerator.main() -- exiting.", - e); - } - - System.exit(0); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java deleted file mode 100644 index e0c8a7c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopFrequencyDMU.java +++ /dev/null @@ -1,255 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.StopFrequencyDMU; - -public class SandagStopFrequencyDMU - extends StopFrequencyDMU -{ - - // the SANDAG UEC worksheet numbers defined below are used to associate - // worksheet - // pages to CTRAMP purpose indices - private static final int WORK_MODEL_SHEET = 1; - private static final int UNIVERSITY_MODEL_SHEET = 2; - private static final int SCHOOL_MODEL_SHEET = 3; - private static final int ESCORT_MODEL_SHEET = 4; - private static final int SHOPPING_MODEL_SHEET = 5; - private static final int MAINT_MODEL_SHEET = 6; - private static final int EAT_OUT_MODEL_SHEET = 7; - private static final int VISITING_MODEL_SHEET = 8; - private static final int DISCR_MODEL_SHEET = 9; - private static final int WORK_BASED_MODEL_SHEET = 10; - - private HashMap tourPurposeModelSheetMap; - private HashMap tourPurposeChoiceModelIndexMap; - private int[] modelSheetValues; - - public SandagStopFrequencyDMU(ModelStructure modelStructure) - { - super(modelStructure); - setupModelIndexMappings(); - setupMethodIndexMap(); - - // set names used in SANDAG stop purpose file - STOP_PURPOSE_FILE_WORK_NAME = "Work"; - STOP_PURPOSE_FILE_UNIVERSITY_NAME = "University"; - STOP_PURPOSE_FILE_SCHOOL_NAME = "School"; - STOP_PURPOSE_FILE_ESCORT_NAME = "Escort"; - STOP_PURPOSE_FILE_SHOPPING_NAME = "Shop"; - STOP_PURPOSE_FILE_MAINT_NAME = "Maintenance"; - STOP_PURPOSE_FILE_EAT_OUT_NAME = "Eating Out"; - STOP_PURPOSE_FILE_VISIT_NAME = "Visiting"; - STOP_PURPOSE_FILE_DISCR_NAME = "Discretionary"; - STOP_PURPOSE_FILE_WORK_BASED_NAME = "Work-Based"; - } - - private void setupModelIndexMappings() - { - - // setup the mapping from tour primary purpose indices to the worksheet - // page - // indices - tourPurposeModelSheetMap = new HashMap(); - tourPurposeModelSheetMap.put(ModelStructure.WORK_PRIMARY_PURPOSE_INDEX, WORK_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX, - UNIVERSITY_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX, - SCHOOL_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX, - ESCORT_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX, - SHOPPING_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX, - MAINT_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_INDEX, - EAT_OUT_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_INDEX, - VISITING_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_INDEX, - DISCR_MODEL_SHEET); - tourPurposeModelSheetMap.put(ModelStructure.WORK_BASED_PRIMARY_PURPOSE_INDEX, - WORK_BASED_MODEL_SHEET); - - // setup a mapping between primary tour purpose indices and - // ChoiceModelApplication array indices - // so that only as many ChoiceModelApplication objects are created in - // the - // Stop Frequency model implementation - // as there are worksheet model pages. - tourPurposeChoiceModelIndexMap = new HashMap(); - - int modelIndex = 0; - HashMap modelSheetIndexMap = new HashMap(); - for (int modelPurposeKey : tourPurposeModelSheetMap.keySet()) - { - - // get the sheet number associated with the tour purpose - int modelSheetKey = tourPurposeModelSheetMap.get(modelPurposeKey); - - // if the sheet number already exists in the sheet index to choice - // model - // index mapping, get that index - // and use it for the purpose to model index mapping - if (modelSheetIndexMap.containsKey(modelSheetKey)) - { - int index = modelSheetIndexMap.get(WORK_MODEL_SHEET); - tourPurposeChoiceModelIndexMap.put(modelPurposeKey, index); - } else - { - // otherwise add this sheet number to the model index mapping - // and use - // it - // for the purpose to model index mapping. - modelSheetIndexMap.put(modelSheetKey, modelIndex); - tourPurposeChoiceModelIndexMap.put(modelPurposeKey, modelIndex); - modelIndex++; - } - } - - modelSheetValues = new int[modelIndex]; - int i = 0; - for (int sheet : modelSheetIndexMap.keySet()) - modelSheetValues[i++] = sheet; - - } - - /** - * @return the array of unique worksheet model sheet values for whic a - * ChoiceModelApplication object will be created. The size of this - * array determines the number of ChoiceModelApplication objects. - */ - public int[] getModelSheetValuesArray() - { - return modelSheetValues; - } - - /** - * @return the HashMap that relates primary tour purpose - * indices to ChoiceModelApplication array indices. - */ - public HashMap getTourPurposeChoiceModelIndexMap() - { - return tourPurposeChoiceModelIndexMap; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getNumFtWorkers", 0); - methodIndexMap.put("getNumPtWorkers", 1); - methodIndexMap.put("getIncomeInDollars", 2); - methodIndexMap.put("getPersonType", 3); - methodIndexMap.put("getHhSize", 4); - methodIndexMap.put("getNumHhDrivingStudents", 5); - methodIndexMap.put("getNumHhNonDrivingStudents", 6); - methodIndexMap.put("getNumHhPreschool", 7); - methodIndexMap.put("getWorkTours", 8); - methodIndexMap.put("getTotalTours", 9); - methodIndexMap.put("getTotalHouseholdTours", 10); - methodIndexMap.put("getWorkLocationDistance", 11); - methodIndexMap.put("getSchoolLocationDistance", 12); - methodIndexMap.put("getAge", 13); - methodIndexMap.put("getSchoolTours", 14); - methodIndexMap.put("getEscortTours", 15); - methodIndexMap.put("getShoppingTours", 16); - methodIndexMap.put("getMaintenanceTours", 17); - methodIndexMap.put("getEatTours", 18); - methodIndexMap.put("getVisitTours", 19); - methodIndexMap.put("getDiscretionaryTours", 20); - methodIndexMap.put("getShoppingAccessibility", 21); - methodIndexMap.put("getMaintenanceAccessibility", 22); - methodIndexMap.put("getDiscretionaryAccessibility", 23); - methodIndexMap.put("getIsJoint", 24); - methodIndexMap.put("getTourDurationHours", 25); - methodIndexMap.put("getTourModeIsAuto", 26); - methodIndexMap.put("getTourModeIsTransit", 27); - methodIndexMap.put("getTourModeIsNonMotorized", 28); - methodIndexMap.put("getTourModeIsSchoolBus", 29); - methodIndexMap.put("getTourDepartPeriod", 30); - methodIndexMap.put("getTourArrivePeriod", 31); - methodIndexMap.put("getTelecommuteFrequency", 32); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getNumFtWorkers(); - case 1: - return getNumPtWorkers(); - case 2: - return getIncomeInDollars(); - case 3: - return getPersonType(); - case 4: - return getHhSize(); - case 5: - return getNumHhDrivingStudents(); - case 6: - return getNumHhNonDrivingStudents(); - case 7: - return getNumHhPreschool(); - case 8: - return getWorkTours(); - case 9: - return getTotalTours(); - case 10: - return getTotalHouseholdTours(); - case 11: - return getWorkLocationDistance(); - case 12: - return getSchoolLocationDistance(); - case 13: - return getAge(); - case 14: - return getSchoolTours(); - case 15: - return getEscortTours(); - case 16: - return getShoppingTours(); - case 17: - return getMaintenanceTours(); - case 18: - return getEatTours(); - case 19: - return getVisitTours(); - case 20: - return getDiscretionaryTours(); - case 21: - return getShoppingAccessibility(); - case 22: - return getMaintenanceAccessibility(); - case 23: - return getDiscretionaryAccessibility(); - case 24: - return getTourIsJoint(); - case 25: - return getTourDurationInHours(); - case 26: - return getTourModeIsAuto(); - case 27: - return getTourModeIsTransit(); - case 28: - return getTourModeIsNonMotorized(); - case 29: - return getTourModeIsSchoolBus(); - case 30: - return getTourDepartPeriod(); - case 31: - return getTourArrivePeriod(); - case 32: - return getTelecommuteFrequency(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java deleted file mode 100644 index 40f1885..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagStopLocationDMU.java +++ /dev/null @@ -1,132 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import java.util.Map; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.BikeLogsum; -import org.sandag.abm.ctramp.BikeLogsumSegment; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Stop; -import org.sandag.abm.ctramp.StopLocationDMU; -import org.sandag.abm.ctramp.Tour; - -public class SandagStopLocationDMU - extends StopLocationDMU -{ - public SandagStopLocationDMU(ModelStructure modelStructure, Map rbMap) - { - super(modelStructure); - setupMethodIndexMap(); - } - - public void setStopObject(Stop myStop) - { - super.setStopObject(myStop); - } - - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getSlcSoaCorrectionsAlt", 0); - methodIndexMap.put("getOrigToMgraDistanceAlt", 1); - methodIndexMap.put("getMgraToDestDistanceAlt", 2); - methodIndexMap.put("getOdDistance", 3); - methodIndexMap.put("getTourModeIsWalk", 4); - methodIndexMap.put("getTourModeIsBike", 5); - methodIndexMap.put("getTourModeIsWalkTransit", 6); - methodIndexMap.put("getWalkTransitAvailableAlt", 7); - methodIndexMap.put("getLnSlcSizeAlt", 8); - methodIndexMap.put("getStopPurpose", 9); - methodIndexMap.put("getTourPurpose", 10); - methodIndexMap.put("getTourMode", 11); - methodIndexMap.put("getStopNumber", 12); - methodIndexMap.put("getStopsOnHalfTour", 13); - methodIndexMap.put("getInboundStop", 14); - methodIndexMap.put("getTourIsJoint", 15); - methodIndexMap.put("getFemale", 16); - methodIndexMap.put("getAge", 17); - methodIndexMap.put("getTourOrigToMgraDistanceAlt", 18); - methodIndexMap.put("getMgraToTourDestDistanceAlt", 19); - methodIndexMap.put("getMcLogsumAlt", 20); - methodIndexMap.put("getSampleMgraAlt", 21); - methodIndexMap.put("getLnSlcSizeSampleAlt", 22); - methodIndexMap.put("getIncome", 23); - methodIndexMap.put("getOrigToMgraBikeLogsumAlt", 24); - methodIndexMap.put("getMgraToDestBikeLogsumAlt", 25); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getSlcSoaCorrectionsAlt(arrayIndex); - case 1: - return getOrigToMgraDistanceAlt(arrayIndex); - case 2: - return getMgraToDestDistanceAlt(arrayIndex); - case 3: - return getOdDistance(); - case 4: - return getTourModeIsWalk(); - case 5: - return getTourModeIsBike(); - case 6: - return getTourModeIsWalkTransit(); - case 7: - return getWalkTransitAvailableAlt(arrayIndex); - case 8: - return getLnSlcSizeAlt(arrayIndex); - case 9: - return getStopPurpose(); - case 10: - return getTourPurpose(); - case 11: - return getTourMode(); - case 12: - return getStopNumber(); - case 13: - return getStopsOnHalfTour(); - case 14: - return getInboundStop(); - case 15: - return getTourIsJoint(); - case 16: - return getFemale(); - case 17: - return getAge(); - case 18: - return getTourOrigToMgraDistanceAlt(arrayIndex); - case 19: - return getMgraToTourDestDistanceAlt(arrayIndex); - case 20: - return getMcLogsumAlt(arrayIndex); - case 21: - return getSampleMgraAlt(arrayIndex); - case 22: - return getLnSlcSizeSampleAlt(arrayIndex); - case 23: - return getIncomeInDollars(); - case 24: - return getOrigToMgraBikeLogsumAlt(arrayIndex); - case 25: - return getMgraToDestBikeLogsumAlt(arrayIndex); - - default: - Logger logger = Logger.getLogger(StopLocationDMU.class); - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java deleted file mode 100644 index dfa63ff..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagSummitFile.java +++ /dev/null @@ -1,820 +0,0 @@ -package org.sandag.abm.application; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import java.util.ResourceBundle; -import java.util.StringTokenizer; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.summit.ConcreteSummitRecord; -import com.pb.common.summit.SummitHeader; -import com.pb.common.summit.SummitRecordTable; -import com.pb.common.util.ResourceUtil; - -public class SandagSummitFile -{ - - private static Logger logger = Logger.getLogger(SandagSummitFile.class); - - private HashMap personsOver18; - private HashMap autosOwned; - - // Summit record table (one per file) - private SummitRecordTable summitRecordTable; - - private TableDataSet tourData; - - // Some parameters - private int modes; - private int upperEA; // Upper - // limit - // on - // time - // of - // day - // for - // the - // Early - // morning - // time - // period - private int upperAM; // Upper - // limit - // on - // time - // of - // day - // for - // the - // AM - // peak - // time - // period - private int upperMD; // Upper - // limit - // on - // time - // of - // day - // for - // the - // Midday - // time - // period - private int upperPM; // Upper - // limit - // on - // time - // of - // day - // for - // the - // PM - // time - // peak - // period - private int[] walkTransitModes; - private int[] driveTransitModes; - - // an array of file numbers, one per purpose - private int[] fileNumber; - private int numberOfFiles; - - private static final String[] PURPOSE_NAME = {"Work", "University", "School", "Escort", - "Shop", "Maintenance", "EatingOut", "Visiting", "Discretionary", "WorkBased"}; - private float[] ivtCoeff; - private String[] fileName; - - ResourceBundle rb; - MgraDataManager mdm; - TazDataManager tdm; - - public SandagSummitFile(String resourceFile) - { - - rb = ResourceUtil.getPropertyBundle(new File(resourceFile)); - mdm = MgraDataManager.getInstance(ResourceUtil.changeResourceBundleIntoHashMap(rb)); - tdm = TazDataManager.getInstance(ResourceUtil.changeResourceBundleIntoHashMap(rb)); - - // Time period limits - upperEA = Integer.valueOf(rb.getString("summit.upperEA")); - upperAM = Integer.valueOf(rb.getString("summit.upperAM")); - upperMD = Integer.valueOf(rb.getString("summit.upperMD")); - upperPM = Integer.valueOf(rb.getString("summit.upperPM")); - - // Find what file to store each purpose in - numberOfFiles = 0; - fileNumber = new int[PURPOSE_NAME.length]; - for (int i = 0; i < PURPOSE_NAME.length; ++i) - { - String fileString = "summit.purpose." + PURPOSE_NAME[i]; - fileNumber[i] = Integer.valueOf(rb.getString(fileString)) - 1; - numberOfFiles = Math.max(fileNumber[i] + 1, numberOfFiles); - } - - // Get the name of each file - fileName = new String[numberOfFiles]; - for (int i = 0; i < numberOfFiles; ++i) - { - String nameString = "summit.filename." + (i + 1); - fileName[i] = rb.getString(nameString); - } - - // Get the ivt coefficients for each file - ivtCoeff = new float[numberOfFiles]; - for (int i = 0; i < numberOfFiles; ++i) - { - String ivtString = "summit.ivt.file." + (i + 1); - ivtCoeff[i] = Float.valueOf(rb.getString(ivtString)); - } - - // set the arrays - modes = Integer.valueOf(rb.getString("summit.modes")); - walkTransitModes = new int[modes]; - driveTransitModes = new int[modes]; - - String modeArray = rb.getString("summit.mode.array").replace(" ", ""); - StringTokenizer inToken = new StringTokenizer(modeArray, ","); - int mode = 0; - while (inToken.hasMoreElements()) - { - int modeValue = Integer.valueOf(inToken.nextToken()); - logger.info("Mode " + mode + " value " + modeValue); - if (modeValue == 1) walkTransitModes[mode] = 1; - else if (modeValue == 2) driveTransitModes[mode] = 1; - - ++mode; - } - - } - - /** - * Create Summit files for all purposes and both individual and joint tour - * files. - * - */ - public void createSummitFiles() - { - - // Read the household file - String directory = rb.getString("Project.Directory"); - String hhFile = rb.getString("Results.HouseholdDataFile"); - readHouseholdFile(directory + hhFile); - - // Read the person file - String perFile = rb.getString("Results.PersonDataFile"); - readPersonFile(directory + perFile); - - // Open the individual tour file and start processing - String tourFile = rb.getString("Results.IndivTourDataFile"); - openTourFile(directory + tourFile); - - String outputDirectory = rb.getString("summit.output.directory"); - - for (int i = 0; i < getNumberOfFiles(); ++i) - { - - // Create the summit table - createSummitFile(i); - - // Write the summit output file - String purpose = getPurpose(i); - writeFile(outputDirectory + purpose + ".bin", i); - - } - - // Open the joint tour file and start processing - tourFile = rb.getString("Results.JointTourDataFile"); - openTourFile(directory + tourFile); - - for (int i = 0; i < getNumberOfFiles(); ++i) - { - - // Create the summit table - createSummitFile(i); - - // Write the summit output file - String purpose = getPurpose(i); - writeFile(outputDirectory + "jnt_" + purpose + ".bin", i); - - } - - } - - /** - * Get the number of SUMMIT Files as set in the properties file. - * - * @return The number of SUMMIT files. - */ - public int getNumberOfFiles() - { - return numberOfFiles; - } - - /** - * Read household records and store autos owned. - * - * @param fileName - * household file path/name. - */ - public void readHouseholdFile(String fileName) - { - - autosOwned = new HashMap(); - - logger.info("Begin reading the data in file " + fileName); - - TableDataSet hhData; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - hhData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - // iterate through the table and save number of autos - for (int i = 1; i <= hhData.getRowCount(); ++i) - { - long hhID = (long) hhData.getValueAt(i, "hh_ID"); - int autos = (int) hhData.getValueAt(i, "autos"); - autosOwned.put(hhID, autos); - } - logger.info("End reading the data in file " + fileName); - } - - /** - * Read person file and store persons >= 18. - * - * @param fileName - * Person file path/name. - */ - public void readPersonFile(String fileName) - { - personsOver18 = new HashMap(); - - logger.info("Begin reading the data in file " + fileName); - - TableDataSet personData; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - personData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - // iterate through the table and save number of persons>=18 - int personCount = 0; - long hhID_last = -99; - for (int i = 1; i <= personData.getRowCount(); ++i) - { - long hhID = (long) personData.getValueAt(i, "hh_ID"); - int age = (int) personData.getValueAt(i, "age"); - - // this record is a new household - if (hhID != hhID_last && i > 1) - { - personsOver18.put(hhID_last, personCount); - personCount = 0; - } - - if (age >= 18) ++personCount; - - hhID_last = hhID; - } - // save the last household - personsOver18.put(hhID_last, personCount); - - logger.info("End reading the data in file " + fileName); - - } - - /** - * Open a tour file for subsequent reading. - */ - public void openTourFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - tourData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - } - - /** - * This is the main workhorse method in this class. It creates a - * SummitRecordTable, and then iterates over records in the tour file. If - * the tour purpose for the record is mapped to the fileNumber argument, a - * ConcreteSummitRecord is created and the attributes are set, and the - * record is added to the SummitRecordTable. After all tour records have - * been read, the table is finalized. - * - * After this method is run, the file header information can be set and the - * SummitRecordTable can be written to a SUMMIT file. - * - * @param fileNumber - */ - public void createSummitFile(int fileNumber) - { - - String purpose = getPurpose(fileNumber); - logger.info("Begin creating SUMMIT record table for purpose " + purpose); - - // get column for start of utilities, probabilities - int start_util = tourData.getColumnPosition("util_1"); - int start_prob = tourData.getColumnPosition("prob_1"); - - boolean jointTour = tourData.containsColumn("tour_participants"); - int participantsCol = 0; - if (jointTour) - { - participantsCol = tourData.getColumnPosition("tour_participants"); - } - - // arrays of utilities, probabilities - float[] util = new float[modes]; - float[] prob = new float[modes]; - - // Instantiate a new SummitRecordTable - summitRecordTable = new SummitRecordTable(); - - // iterate through the tour data and save summit record tables - for (int i = 1; i <= tourData.getRowCount(); ++i) - { - - //if (i <= 5 || i % 1000 == 0) logger.info("Reading record " + i); - - String tourPurpose = tourData.getStringValueAt(i, "tour_purpose"); - int tableNumber = calculateSummitTable(tourPurpose); - - if (tableNumber != fileNumber) continue; - - long hhID = (long) tourData.getValueAt(i, "hh_id"); - - int originMGRA = (int) tourData.getValueAt(i, "orig_maz"); - int destinationMGRA = (int) tourData.getValueAt(i, "dest_maz"); - - int departPeriod = (int) tourData.getValueAt(i, "start_period"); - int arrivePeriod = (int) tourData.getValueAt(i, "end_period"); - - // get utilities, probabilities - for (int j = 0; j < modes; ++j) - { - util[j] = tourData.getValueAt(i, start_util + j); - prob[j] = tourData.getValueAt(i, start_prob + j); - } - - // calculate some necessary information for the SUMMIT record - int autoSufficiency = calculateAutoSufficiency(hhID); - short originTAZ = (short) mdm.getTaz(originMGRA); - short destinationTAZ = (short) mdm.getTaz(destinationMGRA); - int periodMarket = calculatePeriodMarket(departPeriod, arrivePeriod); - short marketSegment = (short) calculateMarketSegment(autoSufficiency, periodMarket); - float expUtility = calculateNonTransitExpUtility(util); // not used - float wtAvailShare = calculateWalkTransitAvailableShare(prob); - float dtAvailShare = calculateDriveTransitOnlyShare(prob, wtAvailShare); - float wtProb = calculateTransitShareOfWalkTransit(prob, wtAvailShare); - float dtProb = calculateTransitShareOfDriveTransitOnly(prob, dtAvailShare); - float aggExpUtility = calculateAggregateExpUtility(util); - - float participants = 1.0f; - if (jointTour) - { - String participantString = tourData.getStringValueAt(i, participantsCol); - for (int j = 0; j < participantString.length(); ++j) - if (participantString.charAt(j) == ' ') participants += 1; - } - // Create a new summit record, and set all attributes - ConcreteSummitRecord summitRecord = new ConcreteSummitRecord(); - - summitRecord.setPtaz(originTAZ); - summitRecord.setAtaz(destinationTAZ); - summitRecord.setMarket(marketSegment); - summitRecord.setTrips(participants); - summitRecord.setMotorizedTrips(participants); - summitRecord.setExpAuto(aggExpUtility); - summitRecord.setWalkTransitAvailableShare(wtAvailShare); - summitRecord.setDriveTransitOnlyShare(dtAvailShare); - summitRecord.setTransitShareOfWalkTransit(wtProb); - summitRecord.setTransitShareOfDriveTransitOnly(dtProb); - - // Insert the record into the record table - summitRecordTable.insertRecord(summitRecord); - - } - - logger.info("End creating SUMMIT record table for purpose " + purpose); - - logger.info("Begin finalizing table"); - summitRecordTable.finalizeTable(); - logger.info("End finalizing table"); - - } - - /** - * Calculate and return the market for the departure and arrival time - * periods. - * - * Market 0 = Departure & arrival in peak Market 1 = Departure & arrival in - * mixed periods (peak and off-peak) Market 2 = Departure & arrival in - * off-peak - * - * @param departPeriod - * 1-39 representing time in 30 min. increments, starting at 5 AM - * @param arrivePeriod - * 1-39 representing time in 30 min. increments, starting at 5 AM - * @return Market, as defined above. - */ - public int calculatePeriodMarket(int departPeriod, int arrivePeriod) - { - - int departPeak = 0; - int arrivePeak = 0; - - // check if departure is in peak period - if (departPeriod > upperEA && departPeriod <= upperAM) departPeak = 1; - else if (departPeriod > upperMD && departPeriod <= upperPM) departPeak = 1; - - // check if arrival is in peak period - if (arrivePeriod > upperEA && arrivePeriod <= upperAM) arrivePeak = 1; - else if (arrivePeriod > upperMD && arrivePeriod <= upperPM) arrivePeak = 1; - - /* - * Arrival & departure = peak, period = 0 Mixed peak & off-peak, period - * = 1 Arrival & departure = off-peal, period = 2 - */ - if (departPeak == 1 && arrivePeak == 1) return 0; - else if (departPeak == 0 && arrivePeak == 0) return 2; - - return 1; - } - - /** - * Calculate and return market segment based on auto sufficiency and time - * period combination. - * - * Market AutoSuff Period 1 0 0 Peak 2 0 1 Mixed 2 0 2 Off-Peak 3 1 0 Peak 4 - * 1 1 Mixed 4 1 2 Off-Peak 5 2 0 Peak 6 2 1 Mixed 6 2 2 Off-Peak - * - * - * @param autoSufficiency - * 0 = 0 autos, 1=autos < adults, 2 = autos >= adults - * @param periodMarket - * 0 = Peak, 1 = Mixed, 2 = Off-Peak - * @return Market for Summit record, as per above table. - */ - public int calculateMarketSegment(int autoSufficiency, int periodMarket) - { - - int market = 0; - - switch (autoSufficiency) - { - case 0: - market = (periodMarket == 0) ? 1 : 2; - break; - case 1: - market = (periodMarket == 0) ? 3 : 4; - break; - case 2: - market = (periodMarket == 0) ? 5 : 6; - break; - default: - logger.fatal("Error: Could not calculate market segment auto sufficiency " - + autoSufficiency); - throw new RuntimeException(); - } - - return market; - } - - /** - * Determine what table to use for tour purpose, based on fileNumber array - * set from properties file. - * - * @param tourPurpose - * @return Table for tour purpose - */ - public int calculateSummitTable(String tourPurpose) - { - - if (tourPurpose.contentEquals("Work")) return fileNumber[0]; - else if (tourPurpose.contentEquals("University")) return fileNumber[1]; - else if (tourPurpose.contentEquals("School")) return fileNumber[2]; - else if (tourPurpose.contentEquals("Escort")) return fileNumber[3]; - else if (tourPurpose.contentEquals("Shop")) return fileNumber[4]; - else if (tourPurpose.contentEquals("Maintenance")) return fileNumber[5]; - else if (tourPurpose.contentEquals("Eating Out")) return fileNumber[6]; - else if (tourPurpose.contentEquals("Visiting")) return fileNumber[7]; - else if (tourPurpose.contentEquals("Discretionary")) return fileNumber[8]; - else if (tourPurpose.contentEquals("Work-Based")) return fileNumber[9]; - else - { - logger.error("Error: Tour purpose " + tourPurpose + " not recognized"); - } - - return 99; - } - - /** - * Look up the purpose string based on the file number, for use in SUMMIT - * file header. - * - * @param fileNumber - * @return A string for the purpose (see above). - */ - public String getPurpose(int fileNumber) - { - - return fileName[fileNumber]; - } - - /** - * Calculate market segment (auto sufficiency) - * - * 0 = 0 autos owned 1 = autos > 0 & autos < adults (persons 18+) 2 = autos - * > adults - * - * @param hhID - * Household ID - * @return marketSegment - */ - public int calculateAutoSufficiency(long hhID) - { - - int drivers = personsOver18.get(hhID); - int autos = autosOwned.get(hhID); - - if (autos > 0) if (autos < drivers) return 1; - else return 2; - - return 0; - } - - /** - * Calculate the total non-transit exponentiated utility. The method uses - * the walkTransitModes array and the driveTransitModes array to determine - * which modes are non-transit, and the sum of their exponentiated utilities - * is calculated and returned. - * - * @param util - * An array of utilities, by mode. -999 indicates mode not - * available. - * @return Sum of exponentiated utilities of non-transit modes. - */ - public float calculateNonTransitExpUtility(float[] util) - { - - float expUtility = 0.0f; - - for (int i = 0; i < modes; ++i) - if (walkTransitModes[i] != 1 && driveTransitModes[i] != 1) - expUtility += (float) Math.exp(util[i]); - return expUtility; - } - - /** - * Calculate the share of walk-transit available: 1 if any walk-transit mode - * is available, as indicated by a non-zero probability, else 0. The method - * iterates through the probability array and returns a 1 if the probability - * is non-zero for any walk-transit mode, as indicated by the - * walkTransitModes array. - * - * @param prob - * An array of probabilities, dimensioned by modes. - * @return 1 if walk-transit is available for the record, else 0. - */ - public float calculateWalkTransitAvailableShare(float[] prob) - { - - // iterate through the probability array - for (int i = 0; i < modes; ++i) - if (walkTransitModes[i] == 1 && prob[i] > 0) return 1.0f; - - // no walk-transit modes with non-zero probability - return 0.0f; - } - - /** - * Calculate the share of drive-transit only available: 1 if any - * drive-transit mode is available, as indicated by a non-zero probability, - * and all walk-transit modes are not available. The method iterates through - * the probability array and returns a 1 if the probability is non-zero for - * any drive-transit mode, as indicated by the driveTransitModes array. - * - * @param prob - * An array of probabilities, dimensioned by modes. - * @param walkTransitAvailableShare - * 1 if walk-transit available, else 0. - * @return 1 if drive-transit only is available for the record, else 0. - */ - - public float calculateDriveTransitOnlyShare(float[] prob, float walkTransitAvailableShare) - { - - // if walk-transit is available, then drive-transit only share is 0. - if (walkTransitAvailableShare > 0) return 0.0f; - - // iterate through the probability array - for (int i = 0; i < modes; ++i) - if (driveTransitModes[i] == 1 && prob[i] > 0) return 1.0f; - - // no drive-transit modes with non-zero probability - return 0.0f; - - } - - /** - * Calculate the total transit probability for records with walk-transit - * available. The method returns 0 if walk-transit is not available. If - * walk-transit is available, the method iterates through the probability - * array, adding all transit mode probabilities. The sum is returned. - * - * @param prob - * An array of probabilities, one per mode. - * @param walkTransitAvailableShare - * 1 if walk-transit is available, else 0. - * @return The total transit probability if walk-transit is available, else - * 0. - */ - public float calculateTransitShareOfWalkTransit(float[] prob, float walkTransitAvailableShare) - { - - float transitShare = 0.0f; - - // if walk-transit is unavailable, then walk-transit share is 0. - if (walkTransitAvailableShare == 0) return transitShare; - - // iterate through the probability array - for (int i = 0; i < modes; ++i) - if (walkTransitModes[i] == 1 || driveTransitModes[i] == 1) transitShare += prob[i]; - - return transitShare; - - } - - /** - * Calculate the total transit probability for records where only - * drive-transit available. The method returns 0 if only drive-transit is - * not available. If only drive-transit is available, the method iterates - * through the probability array, adding all drive-transit mode - * probabilities. The sum is returned. - * - * @param prob - * An array of probabilities, one per mode. - * @param driveTransitOnlyAvailableShare - * 1 if only drive-transit is available, else 0. - * @return The total transit probability if only drive-transit is available, - * else 0. - */ - public float calculateTransitShareOfDriveTransitOnly(float[] prob, - float driveTransitOnlyAvailableShare) - { - - float transitShare = 0.0f; - - // if drive-transit is unavailable, then walk-transit share is 0. - if (driveTransitOnlyAvailableShare == 0) return transitShare; - - // iterate through the probability array - for (int i = 0; i < modes; ++i) - if (driveTransitModes[i] == 1) transitShare += prob[i]; - - return transitShare; - - } - - /** - * Calculate the logsum by taking ln[Sum (exp(utility)). - * - * @param util - * Array of utilities - * @return Logsum - */ - public float calculateAggregateExpUtility(float[] util) - { - - float aggExpUtility = 0.0f; - - for (int i = 0; i < util.length; ++i) - aggExpUtility += Math.exp(util[i]); - - return aggExpUtility; - } - - /** - * Get the in-tNCVehicle time coefficient for the file, based on the values - * read in the properties file. - * - * @param fileNumber - * @return The in-tNCVehicle time coefficient for the file. - */ - public float getIVTCoefficient(int fileNumber) - { - return ivtCoeff[fileNumber]; - } - - /** - * Create a Summit file header. - * - * @param fileNumber - * @return Header record for Summit file. - */ - public SummitHeader createSummitHeader(int fileNumber) - { - - SummitHeader header = new SummitHeader(); - - int zones = tdm.getMaxTaz(); - String purpose = getPurpose(fileNumber); - header.setZones(zones); - header.setMarketSegments(6); - - float ivt = getIVTCoefficient(fileNumber); - header.setTransitInVehicleTime(ivt); - header.setAutoInVehicleTime(ivt); - - header.setPurpose(purpose); - header.setTimeOfDay("ALL"); - header.setTitle("SANDAG CT-RAMP MODEL SUMMIT FILE"); - return header; - } - - public void writeFile(String fileName, int fileNumber) - { - - SummitHeader header = createSummitHeader(fileNumber); - summitRecordTable.writeTable(fileName, header); - - } - - /** - * @param args - */ - public static void main(String[] args) - { - - // Create a new SandagSummitFile - String propertiesFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\sandag_abm.properties"; - SandagSummitFile summitFile = new SandagSummitFile(propertiesFile); - - // Read the household file - String hhFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\householdData_1.csv"; - summitFile.readHouseholdFile(hhFile); - - // Read the person file - String perFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\personData_1.csv"; - summitFile.readPersonFile(perFile); - - // Open the individual tour file and start processing - String tourFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\indivTourData_1.csv"; - summitFile.openTourFile(tourFile); - - for (int i = 0; i < summitFile.getNumberOfFiles(); ++i) - { - - // Create the summit table - summitFile.createSummitFile(i); - - // Write the summit output file - String outputFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\"; - String purpose = summitFile.getPurpose(i); - summitFile.writeFile(outputFile + purpose + ".bin", i); - - } - - // Open the joint tour file and start processing - tourFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\jointTourData_1.csv"; - summitFile.openTourFile(tourFile); - - for (int i = 0; i < summitFile.getNumberOfFiles(); ++i) - { - - // Create the summit table - summitFile.createSummitFile(i); - - // Write the summit output file - String outputFile = "D:\\projects\\SANDAG\\AB_Model\\SUMMIT\\jnt_"; - String purpose = summitFile.getPurpose(i); - summitFile.writeFile(outputFile + purpose + ".bin", i); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java deleted file mode 100644 index e0f4051..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTelecommuteDMU.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.ParkingChoiceDMU; -import org.sandag.abm.ctramp.TelecommuteDMU; - -public class SandagTelecommuteDMU - extends TelecommuteDMU -{ - - public SandagTelecommuteDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getIncomeInDollars", 1); - methodIndexMap.put("getNumberOfAdults", 2); - methodIndexMap.put("getHasKids_0_5", 3); - methodIndexMap.put("getHasKids_6_12", 4); - methodIndexMap.put("getFemale", 5); - methodIndexMap.put("getPersonType", 6); - methodIndexMap.put("getNumberOfAutos", 7); - methodIndexMap.put("getOccupation", 8); - methodIndexMap.put("getPaysToPark", 9); - methodIndexMap.put("getWorkDistance", 10); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - - case 1: - return getIncomeInDollars(); - case 2: - return getNumberOfAdults(); - case 3: - return getHasKids_0_5(); - case 4: - return getHasKids_6_12(); - case 5: - return getFemale(); - case 6: - return getPersonType(); - case 7: - return getNumberOfAutos(); - case 8: - return getOccupation(); - case 9: - return getPaysToPark(); - case 10: - return getWorkDistance(); - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java deleted file mode 100644 index 92ba6ff..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTestSOA.java +++ /dev/null @@ -1,216 +0,0 @@ -package org.sandag.abm.application; - -import com.pb.common.matrix.Matrix; - -public class SandagTestSOA -{ - - private Matrix distanceMatrix; - private Matrix expDistanceMatrix; - private float[] size; - private float[] lnSize; - - public SandagTestSOA() - { - - } - - /** - * Create the distance matrix - * - * @param zones - * The number of zones - */ - public void createDistanceMatrix(int zones) - { - - distanceMatrix = new Matrix(zones, zones); - - for (int i = 1; i <= zones; ++i) - for (int j = 1; j <= zones; ++j) - { - float distance = (float) Math.random() * 100; - distanceMatrix.setValueAt(i, j, distance); - } - } - - /** - * Create the exponentiated distance matrix - * - * @param zones - * The number of zones - */ - public void createExpDistanceMatrix(float distParam, int zones) - { - - long createTime = -System.currentTimeMillis(); - expDistanceMatrix = new Matrix(zones, zones); - - for (int i = 1; i <= zones; ++i) - for (int j = 1; j <= zones; ++j) - { - float expDist = (float) distParam * distanceMatrix.getValueAt(i, j); - expDist = (float) Math.exp(expDist); - expDistanceMatrix.setValueAt(i, j, expDist); - } - createTime += System.currentTimeMillis(); - System.out.println("Time to exponentiate distance matrix " + createTime); - } - - /** - * Create the size terms - */ - public void createSizeTerms(int zones) - { - - size = new float[zones + 1]; - lnSize = new float[zones + 1]; - - for (int i = 1; i <= zones; ++i) - { - size[i] = (float) Math.random() * 1000; - lnSize[i] = (float) Math.log(size[i]); - } - } - - public void calculateProbabilitiesOldWay(int observations, int zones, float distParam) - { - - long oldWayTime = -System.currentTimeMillis(); - - float[] prob = new float[zones + 1]; - float[] expUtil = new float[zones + 1]; - float sumExp = 0; - - for (int obs = 0; obs < observations; ++obs) - { - - int origin = (int) (Math.random() * (zones - 1)) + 1; - int destination = (int) (Math.random() * (zones - 1)) + 1; - - float odDist = distanceMatrix.getValueAt(origin, destination); - - // calculate utilities - for (int stop = 1; stop <= zones; ++stop) - { - float osDist = distanceMatrix.getValueAt(origin, stop); - float sdDist = distanceMatrix.getValueAt(stop, destination); - - float util = distParam * (osDist + sdDist - odDist) + lnSize[stop]; - expUtil[stop] = (float) Math.exp(util); - sumExp += expUtil[stop]; - } - - // calculate probabilities - for (int stop = 1; stop <= zones; ++stop) - prob[stop] = expUtil[stop] / sumExp; - } - oldWayTime += System.currentTimeMillis(); - System.out.println("Time to calculate probabilities old way tazs " + oldWayTime); - - } - - public void calculateProbabilitiesOldWayMGRAs(int observations, int zones, float distParam, - int mgras) - { - - long oldWayMGRATime = -System.currentTimeMillis(); - - float[] prob = new float[mgras + 1]; - float[] expUtil = new float[mgras + 1]; - float sumExp = 0; - - for (int obs = 0; obs < observations; ++obs) - { - - int origin = (int) (Math.random() * (zones - 1)) + 1; - int destination = (int) (Math.random() * (zones - 1)) + 1; - - float odDist = distanceMatrix.getValueAt(origin, destination); - - int stopIndex = 1; - // calculate utilities - for (int stop = 1; stop <= mgras; ++stop) - { - - float osDist = distanceMatrix.getValueAt(origin, stopIndex); - float sdDist = distanceMatrix.getValueAt(stopIndex, destination); - - float util = distParam * (osDist + sdDist - odDist) + lnSize[stopIndex]; - expUtil[stopIndex] = (float) Math.exp(util); - sumExp += expUtil[stopIndex]; - - ++stopIndex; - if (stopIndex > zones) stopIndex = 1; - } - - // calculate probabilities - for (int stop = 1; stop <= mgras; ++stop) - prob[stop] = expUtil[stop] / sumExp; - - } - oldWayMGRATime += System.currentTimeMillis(); - System.out.println("Time to calculate probabilities old way mgras " + oldWayMGRATime); - - } - - public void calculateProbabilitiesNewWay(int observations, int zones) - { - - long newWayTime = -System.currentTimeMillis(); - - float[] prob = new float[zones + 1]; - float[] expUtil = new float[zones + 1]; - float sumExp = 0; - - for (int obs = 0; obs < observations; ++obs) - { - - int origin = (int) (Math.random() * (zones - 1)) + 1; - int destination = (int) (Math.random() * (zones - 1)) + 1; - - float odExpDist = expDistanceMatrix.getValueAt(origin, destination); - - // calculate utilities - for (int stop = 1; stop <= zones; ++stop) - { - float osExpDist = expDistanceMatrix.getValueAt(origin, stop); - float sdExpDist = expDistanceMatrix.getValueAt(stop, destination); - - expUtil[stop] = osExpDist * sdExpDist / odExpDist * size[stop]; - sumExp += expUtil[stop]; - } - - // calculate probabilities - for (int stop = 1; stop <= zones; ++stop) - prob[stop] = expUtil[stop] / sumExp; - } - - newWayTime += System.currentTimeMillis(); - System.out.println("Time to calculate probabilities new way tazs " + newWayTime); - } - - /** - * @param args - */ - public static void main(String[] args) - { - - int zones = 4600; - int observations = 500000; - float distParam = (float) -0.05; - int mgras = 32000; - - SandagTestSOA soa = new SandagTestSOA(); - - soa.createDistanceMatrix(zones); - soa.createExpDistanceMatrix(distParam, zones); - soa.createSizeTerms(zones); - - soa.calculateProbabilitiesOldWayMGRAs(observations, zones, distParam, mgras); - soa.calculateProbabilitiesOldWay(observations, zones, distParam); - soa.calculateProbabilitiesNewWay(observations, zones); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java deleted file mode 100644 index 86aaf7e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourBasedModel.java +++ /dev/null @@ -1,358 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; - -import org.sandag.abm.ctramp.HouseholdDataManager; -import org.sandag.abm.ctramp.HouseholdDataManagerIf; -import org.sandag.abm.ctramp.HouseholdDataManagerRmi; -import com.pb.common.util.ResourceUtil; - -public final class SandagTourBasedModel -{ - - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - - public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; - - private static final int DEFAULT_ITERATION_NUMBER = 1; - private static final float DEFAULT_SAMPLE_RATE = 1.0f; - private static final int DEFAULT_SAMPLE_SEED = 0; - - public static final int DEBUG_CHOICE_MODEL_HHID = 740151; - - private ResourceBundle rb; - - // values for these variables are set as command line arguments, or default - // vaues - // are used if no command line arguments are specified. - private int globalIterationNumber = 0; - private float iterationSampleRate = 0f; - private int sampleSeed = 0; - private boolean calculateLandUseAccessibilities = false; - - /** - * - * @param rb - * , java.util.ResourceBundle containing environment settings - * from a properties file specified on the command line - * @param globalIterationNumber - * , int iteration number for which the model is run, set by - * another process controlling a model stream with feedback. - * @param iterationSampleRate - * , float percentage [0.0, 1.0] inicating the portion of all - * households to be modeled. - * - * This object defines the implementation of the ARC tour based, - * activity based travel demand model. - */ - private SandagTourBasedModel(ResourceBundle aRb, HashMap aPropertyMap, - int aGlobalIterationNumber, float aIterationSampleRate, boolean aCalculateLandUseAccessibilities) - { - rb = aRb; - globalIterationNumber = aGlobalIterationNumber; - iterationSampleRate = aIterationSampleRate; - sampleSeed = Integer.parseInt(rb.getString("Model.Random.Seed")); - calculateLandUseAccessibilities = aCalculateLandUseAccessibilities; - } - - private void runTourBasedModel(HashMap propertyMap) - { - - // new a ctramp application object - SandagCtrampApplication ctrampApplication = new SandagCtrampApplication(rb, propertyMap, - calculateLandUseAccessibilities); - - // create modelStructure object - SandagModelStructure modelStructure = new SandagModelStructure(); - - boolean localHandlers = false; - - String hhHandlerAddress = ""; - int hhServerPort = 0; - try - { - // get household server address. if none is specified a local server - // in - // the current process will be started. - hhHandlerAddress = rb.getString("RunModel.HouseholdServerAddress"); - try - { - // get household server port. - hhServerPort = Integer.parseInt(rb.getString("RunModel.HouseholdServerPort")); - localHandlers = false; - } catch (MissingResourceException e) - { - // if no household data server address entry is found, the - // object - // will be created in the local process - localHandlers = true; - } - } catch (MissingResourceException e) - { - localHandlers = true; - } - - String testString; - // if ( localHandlers ) { - // tazDataHandler = new SandagTazDataHandler(rb, projectDirectory); - // } - // else { - // tazDataHandler = new TazDataHandlerRmi( - // ArcTazDataHandler.ZONAL_DATA_SERVER_ADDRESS, - // ArcTazDataHandler.ZONAL_DATA_SERVER_PORT, - // ArcTazDataHandler.ZONAL_DATA_SERVER_NAME ); - // testString = tazDataHandler.testRemote(); - // logger.info ( "TazDataHandler test: " + testString ); - // } - - // setup the ctramp application - ctrampApplication.setupModels(modelStructure); - - // generate the synthetic population - // ARCPopulationSynthesizer populationSynthesizer = new - // ARCPopulationSynthesizer( propertiesFileBaseName ); - // ctrampApplication.runPopulationSynthesizer( populationSynthesizer ); - - HouseholdDataManagerIf householdDataManager; - - try - { - - if (localHandlers) - { - - // create a new local instance of the household array manager - householdDataManager = new SandagHouseholdDataManager2(); - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - String inputHouseholdFileName = rb - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = rb - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, - inputHouseholdFileName, inputPersonFileName); - - } else - { - - householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, - SandagHouseholdDataManager2.HH_DATA_SERVER_NAME); - testString = householdDataManager.testRemote(); - logger.info("HouseholdDataManager test: " + testString); - - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - boolean restartHhServer = false; - try - { - // possible values for the following can be none, ao, cdap, - // imtf, - // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, - // inmtod, - // stf, stl - String restartModel = rb.getString("RunModel.RestartWithHhServer"); - if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; - else if (restartModel.equalsIgnoreCase("uwsl") - || restartModel.equalsIgnoreCase("ao") - || restartModel.equalsIgnoreCase("fp") - || restartModel.equalsIgnoreCase("cdap") - || restartModel.equalsIgnoreCase("imtf") - || restartModel.equalsIgnoreCase("imtod") - || restartModel.equalsIgnoreCase("awf") - || restartModel.equalsIgnoreCase("awl") - || restartModel.equalsIgnoreCase("awtod") - || restartModel.equalsIgnoreCase("jtf") - || restartModel.equalsIgnoreCase("jtl") - || restartModel.equalsIgnoreCase("jtod") - || restartModel.equalsIgnoreCase("inmtf") - || restartModel.equalsIgnoreCase("inmtl") - || restartModel.equalsIgnoreCase("inmtod") - || restartModel.equalsIgnoreCase("stf") - || restartModel.equalsIgnoreCase("stl")) restartHhServer = false; - } catch (MissingResourceException e) - { - restartHhServer = true; - } - - if (restartHhServer) - { - - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = rb - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = rb - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, - inputHouseholdFileName, inputPersonFileName); - - } else - { - - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setDebugHhIdsFromHashmap(); - householdDataManager.setTraceHouseholdSet(); - - // set the random number sequence for household objects - // accordingly based on which model components are - // assumed to have already run and are stored in the remote - // HouseholdDataManager object. - ctrampApplication.restartModels(householdDataManager); - - } - - } - - // create a factory object to pass to various model components from - // which - // they can create DMU objects - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - // run the models - ctrampApplication.runModels(householdDataManager, dmuFactory, globalIterationNumber, - iterationSampleRate); - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - } - - public static void main(String[] args) - { - Runtime gfg = Runtime.getRuntime(); - long memory1; - // checking the total memeory - System.out.println("Total memory is: "+ gfg.totalMemory()); - // checking free memory - memory1 = gfg.freeMemory(); - System.out.println("Initial free memory at Resident model: "+ memory1); - // calling the garbage collector on demand - gfg.gc(); - memory1 = gfg.freeMemory(); - System.out.println("Free memory after garbage "+ "collection: " + memory1); - - long startTime = System.currentTimeMillis(); - int globalIterationNumber = -1; - float iterationSampleRate = -1.0f; - //int sampleSeed = -1; - boolean calculateLandUseAccessibilities = false; - - ResourceBundle rb = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - pMap = ResourceUtil.getResourceBundleAsHashMap(args[0]); - - // optional arguments - for (int i = 1; i < args.length; i++) - { - - if (args[i].equalsIgnoreCase("-iteration")) - { - globalIterationNumber = Integer.parseInt(args[i + 1]); - logger.info(String.format("-iteration %d.", globalIterationNumber)); - } - - if (args[i].equalsIgnoreCase("-sampleRate")) - { - iterationSampleRate = Float.parseFloat(args[i + 1]); - logger.info(String.format("-sampleRate %.4f.", iterationSampleRate)); - } - - /* - if (args[i].equalsIgnoreCase("-sampleSeed")) - { - sampleSeed = Integer.parseInt(args[i + 1]); - logger.info(String.format("-sampleSeed %d.", sampleSeed)); - } - */ - - if (args[i].equalsIgnoreCase("-luAcc")) - { - calculateLandUseAccessibilities = Boolean.parseBoolean(args[i + 1]); - logger.info(String.format("-luAcc %s.", calculateLandUseAccessibilities)); - } - - } - - if (globalIterationNumber < 0) - { - globalIterationNumber = DEFAULT_ITERATION_NUMBER; - logger.info(String.format("no -iteration flag, default value %d used.", - globalIterationNumber)); - } - - if (iterationSampleRate < 0) - { - iterationSampleRate = DEFAULT_SAMPLE_RATE; - logger.info(String.format("no -sampleRate flag, default value %.4f used.", - iterationSampleRate)); - } - - /* - if (sampleSeed < 0) - { - sampleSeed = DEFAULT_SAMPLE_SEED; - logger.info(String - .format("no -sampleSeed flag, default value %d used.", sampleSeed)); - } - */ - - } - - // create an instance of this class for main() to use. - SandagTourBasedModel mainObject = new SandagTourBasedModel(rb, pMap, globalIterationNumber, - iterationSampleRate, calculateLandUseAccessibilities); - - // run tour based models - try - { - - logger.info(""); - logger.info("starting tour based model."); - mainObject.runTourBasedModel(pMap); - - } catch (RuntimeException e) - { - logger.error( - "RuntimeException caught in org.sandag.abm.application.SandagTourBasedModel.main() -- exiting.", - e); - } - - logger.info(""); - logger.info(""); - logger.info("SANDAG Activity Based Model finished in " - + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); - - System.exit(0); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java deleted file mode 100644 index 6971a98..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourDepartureTimeAndDurationDMU.java +++ /dev/null @@ -1,345 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.Definitions; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.TourDepartureTimeAndDurationDMU; - -/** - * ArcTourDepartureTimeAndDurationDMU is a class that ... - * - * @author Kimberly Grommes - * @version 1.0, Jul 17, 2008 Created by IntelliJ IDEA. - */ -public class SandagTourDepartureTimeAndDurationDMU - extends TourDepartureTimeAndDurationDMU -{ - - public SandagTourDepartureTimeAndDurationDMU(ModelStructure modelStructure) - { - super(modelStructure); - setupMethodIndexMap(); - } - - public double getDestinationEmploymentDensity() - { - return destEmpDen; - } - - public int getIncomeLessThan30k() - { - float incomeInDollars = (float) household.getIncomeInDollars(); - return (incomeInDollars < 30000) ? 1 : 0; - } - - public int getIncome30kTo60k() - { - float incomeInDollars = (float) household.getIncomeInDollars(); - return (incomeInDollars >= 30000 && incomeInDollars < 60000) ? 1 : 0; - } - - public int getIncomeHigherThan100k() - { - float incomeInDollars = (float) household.getIncomeInDollars(); - return (incomeInDollars >= 100000) ? 1 : 0; - } - - public int getAge() - { - return getPersonAge(); - } - - public int getFemale() - { - return getPersonIsFemale(); - } - - public int getFemaleWithPreschooler() - { - return ((getPersonIsFemale() == 1) && (getNumPreschoolChildrenInHh() > 1)) ? 1 : 0; - } - - public int getDrivingAgeStudent() - { - return (getStudentDrivingAge() == 1) ? 1 : 0; - } - - public int getSchoolChildWithMandatoryTour() - { - return (getStudentNonDrivingAge() == 1 && getPersonMandatoryTotal() > 0) ? 1 : 0; - } - - public int getUniversityWithMandatoryPattern() - { - return (getUniversityStudent() == 1 && person.getCdapActivity().equalsIgnoreCase( - Definitions.MANDATORY_PATTERN)) ? 1 : 0; - } - - public int getWorkerWithMandatoryPattern() - { - return ((getFullTimeWorker() == 1 || getPartTimeWorker() == 1) && person.getCdapActivity() - .equalsIgnoreCase(Definitions.MANDATORY_PATTERN)) ? 1 : 0; - } - - public int getPreschoolChildWithMandatoryTour() - { - return (getPreschool() == 1 && getPersonMandatoryTotal() > 0) ? 1 : 0; - } - - public int getNonWorkerInHH() - { - return (getNumNonWorkingAdultsInHh() > 0) ? 1 : 0; - } - - public int getJointTour() - { - return (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) ? 1 : 0; - } - - public int getIndividualTour() - { - return (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) ? 1 : 0; - } - - public int getJointTourInHH() - { - return (getHhJointTotal() > 0) ? 1 : 0; - } - - public int getSubsequentTourIsWorkTour() - { - return subsequentTourIsWork; - } - - public int getSubsequentTourIsSchoolTour() - { - return subsequentTourIsSchool; - } - - public int getNumberOfNonEscortingIndividualTours() - { - return getPersonNonMandatoryTotalNoEscort(); - } - - public int getNumberOfDiscretionaryTours() - { - return getPersonJointAndIndivDiscrToursTotal(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getDestinationEmploymentDensity", 1); - methodIndexMap.put("getIncomeLessThan30k", 2); - methodIndexMap.put("getIncome30kTo60k", 3); - methodIndexMap.put("getIncomeHigherThan100k", 4); - methodIndexMap.put("getAge", 5); - methodIndexMap.put("getFemale", 6); - methodIndexMap.put("getFemaleWithPreschooler", 7); - methodIndexMap.put("getFullTimeWorker", 8); - methodIndexMap.put("getPartTimeWorker", 9); - methodIndexMap.put("getUniversityStudent", 10); - methodIndexMap.put("getDrivingAgeStudent", 11); - methodIndexMap.put("getNonWorkerInHH", 12); - methodIndexMap.put("getJointTourInHH", 13); - methodIndexMap.put("getFirstTour", 14); - methodIndexMap.put("getSubsequentTour", 15); - methodIndexMap.put("getModeChoiceLogsumAlt", 16); - methodIndexMap.put("getSubsequentTourIsWorkTour", 17); - methodIndexMap.put("getSubsequentTourIsSchoolTour", 18); - methodIndexMap.put("getEndOfPreviousTour", 19); - methodIndexMap.put("getAllAdultsFullTimeWorkers", 20); - methodIndexMap.put("getNonWorker", 21); - methodIndexMap.put("getRetired", 22); - methodIndexMap.put("getSchoolChildWithMandatoryTour", 23); - methodIndexMap.put("getPreschoolChildWithMandatoryTour", 24); - methodIndexMap.put("getNumberOfNonEscortingIndividualTours", 25); - methodIndexMap.put("getNumberOfDiscretionaryTours", 26); - methodIndexMap.put("getIndividualTour", 27); - methodIndexMap.put("getJointTour", 28); - methodIndexMap.put("getHouseholdSize", 29); - methodIndexMap.put("getKidsOnJointTour", 30); - methodIndexMap.put("getAdditionalShoppingTours", 31); - methodIndexMap.put("getAdditionalMaintenanceTours", 32); - methodIndexMap.put("getAdditionalVisitingTours", 33); - methodIndexMap.put("getAdditionalDiscretionaryTours", 34); - methodIndexMap.put("getMaximumAvailableTimeWindow", 35); - methodIndexMap.put("getWorkerWithMandatoryPattern", 36); - methodIndexMap.put("getUnivStudentWithMandatoryPattern", 37); - methodIndexMap.put("getHhChildUnder16", 38); - methodIndexMap.put("getToursLeftToSchedule", 39); - methodIndexMap.put("getPreDrivingAgeChild", 40); - methodIndexMap.put("getJointTourPartySize", 41); - methodIndexMap.put("getSubtourPurposeIsEatOut", 42); - methodIndexMap.put("getSubtourPurposeIsBusiness", 43); - methodIndexMap.put("getSubtourPurposeIsOther", 44); - methodIndexMap.put("getMaxJointTimeWindow", 45); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - case 1: - returnValue = getDestinationEmploymentDensity(); - break; - case 2: - returnValue = getIncomeLessThan30k(); - break; - case 3: - returnValue = getIncome30kTo60k(); - break; - case 4: - returnValue = getIncomeHigherThan100k(); - break; - case 5: - returnValue = getAge(); - break; - case 6: - returnValue = getFemale(); - break; - case 7: - returnValue = getFemaleWithPreschooler(); - break; - case 8: - returnValue = getFullTimeWorker(); - break; - case 9: - returnValue = getPartTimeWorker(); - break; - case 10: - returnValue = getUniversityStudent(); - break; - case 11: - returnValue = getDrivingAgeStudent(); - break; - case 12: - returnValue = getNonWorkerInHH(); - break; - case 13: - returnValue = getJointTourInHH(); - break; - case 14: - returnValue = getFirstTour(); - break; - case 15: - returnValue = getSubsequentTour(); - break; - case 16: - returnValue = getModeChoiceLogsumAlt(arrayIndex); - break; - case 17: - returnValue = getSubsequentTourIsWorkTour(); - break; - case 18: - returnValue = getSubsequentTourIsSchoolTour(); - break; - case 19: - returnValue = getEndOfPreviousTour(); - break; - case 20: - returnValue = getAllAdultsFullTimeWorkers(); - break; - case 21: - returnValue = getNonWorker(); - break; - case 22: - returnValue = getRetired(); - break; - case 23: - returnValue = getSchoolChildWithMandatoryTour(); - break; - case 24: - returnValue = getPreschoolChildWithMandatoryTour(); - break; - case 25: - returnValue = getNumberOfNonEscortingIndividualTours(); - break; - case 26: - returnValue = getNumberOfDiscretionaryTours(); - break; - case 27: - returnValue = getIndividualTour(); - break; - case 28: - returnValue = getJointTour(); - break; - case 29: - returnValue = getHouseholdSize(); - break; - case 30: - returnValue = getKidsOnJointTour(); - break; - case 31: - returnValue = getNumIndivShopTours() - 1; - break; - case 32: - returnValue = getNumIndivMaintTours() - 1; - break; - case 33: - returnValue = getNumIndivVisitTours() - 1; - break; - case 34: - returnValue = getNumIndivDiscrTours() - 1; - break; - case 35: - returnValue = getMaximumAvailableTimeWindow(); - break; - case 36: - returnValue = getWorkerWithMandatoryPattern(); - break; - case 37: - returnValue = getUniversityWithMandatoryPattern(); - break; - case 38: - returnValue = getNumChildrenUnder16InHh() > 0 ? 1 : 0; - break; - case 39: - returnValue = getToursLeftToSchedule(); - break; - case 40: - returnValue = getPreDrivingAgeChild(); - break; - case 41: - returnValue = getJointTourPartySize(); - break; - case 42: - returnValue = getSubtourPurposeIsEatOut(); - break; - case 43: - returnValue = getSubtourPurposeIsBusiness(); - break; - case 44: - returnValue = getSubtourPurposeIsOther(); - break; - case 45: - returnValue = getMaxJointTimeWindow(); - break; - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java deleted file mode 100644 index 79e76ed..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTourModeChoiceDMU.java +++ /dev/null @@ -1,502 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import org.sandag.abm.ctramp.BikeLogsum; -import org.sandag.abm.ctramp.BikeLogsumSegment; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Tour; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import org.sandag.abm.ctramp.ModelStructure; - - -public class SandagTourModeChoiceDMU - extends TourModeChoiceDMU -{ - - private int setPersonHhTourCounter = 0; - private BikeLogsum bls; - protected double inboundFemaleBikeLogsum; - protected double outboundFemaleBikeLogsum; - protected double inboundMaleBikeLogsum; - protected double outboundMaleBikeLogsum; - protected double femaleInParty; - protected double maleInParty; - - public SandagTourModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) - { - super(modelStructure, aLogger); - setupMethodIndexMap(); - } - - public float getTimeOutbound() - { - return tour.getTourDepartPeriod(); - } - - public float getTimeInbound() - { - return tour.getTourArrivePeriod(); - } - - public int getIncome() - { - return hh.getIncomeInDollars(); - } - - public int getAdults() - { - return hh.getNumPersons18plus(); - } - - public int getFemale() - { - return person.getPersonIsFemale(); - } - - public void setOrigDuDen(double arg) - { - origDuDen = arg; - } - - public void setOrigEmpDen(double arg) - { - origEmpDen = arg; - } - - public void setOrigTotInt(double arg) - { - origTotInt = arg; - } - - public void setDestDuDen(double arg) - { - destDuDen = arg; - } - - public void setDestEmpDen(double arg) - { - destEmpDen = arg; - } - - public void setDestTotInt(double arg) - { - destTotInt = arg; - } - - public double getODUDen() - { - return origDuDen; - } - - public double getOEmpDen() - { - return origEmpDen; - } - - public double getOTotInt() - { - return origTotInt; - } - - public double getDDUDen() - { - return destDuDen; - } - - public double getDEmpDen() - { - return destEmpDen; - } - - public double getDTotInt() - { - return destTotInt; - } - - public double getNm_walkTime_out() - { - return getNmWalkTimeOut(); - } - - public double getNm_walkTime_in() - { - return getNmWalkTimeIn(); - } - - public double getNm_bikeTime_out() - { - return getNmBikeTimeOut(); - } - - public double getNm_bikeTime_in() - { - return getNmBikeTimeIn(); - } - - - public void setBikeLogsum(BikeLogsum bls) - { - this.bls = bls; - } - - public void setPersonObject(Person person) - { - super.setPersonObject(person); - checkSetPersonHhTour(); - } - - public void setHouseholdObject(Household hh) - { - super.setHouseholdObject(hh); - checkSetPersonHhTour(); - } - - public void setTourObject(Tour tour) - { - super.setTourObject(tour); - checkSetPersonHhTour(); - } - - private void checkSetPersonHhTour() - { - setPersonHhTourCounter = (setPersonHhTourCounter+1) % 3; - if (setPersonHhTourCounter == 0) { - setParty(person,tour,hh); - setBikeLogsum(); - } - } - - public double getFemaleInParty() - { - return femaleInParty; - } - - public double getMaleInParty() - { - return maleInParty; - } - - public void setParty(Person person, Tour tour, Household hh) - { - if (person != null) { - femaleInParty = person.getPersonIsFemale(); - maleInParty = femaleInParty == 0 ? 1 : 0; - } else { - femaleInParty = 0; - maleInParty = 0; - for (int participant : tour.getPersonNumArray()) { - if (hh.getPerson(participant).getPersonIsFemale() == 1) - femaleInParty = 1; - else - maleInParty = 1; - } - } - } - - public double getInboundFemaleBikeLogsum() - { - return inboundFemaleBikeLogsum; - } - - public double getOutboundFemaleBikeLogsum() - { - return outboundFemaleBikeLogsum; - } - - public double getInboundMaleBikeLogsum() - { - return inboundMaleBikeLogsum; - } - - public double getOutboundMaleBikeLogsum() - { - return outboundMaleBikeLogsum; - } - - - private void setBikeLogsum(double inboundFemaleBikeLogsum, double outboundFemaleBikeLogsum, - double inboundMaleBikeLogsum , double outboundMaleBikeLogsum) - { - this.inboundFemaleBikeLogsum = inboundFemaleBikeLogsum; - this.outboundFemaleBikeLogsum = outboundFemaleBikeLogsum; - this.inboundMaleBikeLogsum = inboundMaleBikeLogsum; - this.outboundMaleBikeLogsum = outboundMaleBikeLogsum; - } - - private void setBikeLogsum() - { - int origin = tour.getTourOrigMgra(); - int dest = tour.getTourDestMgra(); - boolean mandatory = tour.getTourPrimaryPurposeIndex() <= 3; - setBikeLogsum(bls.getLogsum(new BikeLogsumSegment(true,mandatory,true),dest,origin), - bls.getLogsum(new BikeLogsumSegment(true,mandatory,false),origin,dest), - bls.getLogsum(new BikeLogsumSegment(false,mandatory,true),dest,origin), - bls.getLogsum(new BikeLogsumSegment(false,mandatory,false),origin,dest)); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getIncomeCategory", 2); - methodIndexMap.put("getAdults", 3); - methodIndexMap.put("getFemale", 4); - methodIndexMap.put("getHhSize", 5); - methodIndexMap.put("getAutos", 6); - methodIndexMap.put("getAge", 7); - methodIndexMap.put("getTourCategoryJoint", 8); - methodIndexMap.put("getNumberOfParticipantsInJointTour", 9); - methodIndexMap.put("getWorkTourModeIsSov", 10); - methodIndexMap.put("getWorkTourModeIsBike", 11); - methodIndexMap.put("getWorkTourModeIsHov", 12); - methodIndexMap.put("getPTazTerminalTime", 14); - methodIndexMap.put("getATazTerminalTime", 15); - methodIndexMap.put("getODUDen", 16); - methodIndexMap.put("getOEmpDen", 17); - methodIndexMap.put("getOTotInt", 18); - methodIndexMap.put("getDDUDen", 19); - methodIndexMap.put("getDEmpDen", 20); - methodIndexMap.put("getDTotInt", 21); - methodIndexMap.put("getTourCategoryEscort", 22); - methodIndexMap.put("getMonthlyParkingCost", 23); - methodIndexMap.put("getDailyParkingCost", 24); - methodIndexMap.put("getHourlyParkingCost", 25); - methodIndexMap.put("getReimburseProportion", 26); - methodIndexMap.put("getPersonType", 27); - methodIndexMap.put("getFreeParkingEligibility", 28); - methodIndexMap.put("getParkingArea", 29); - - methodIndexMap.put("getWorkTimeFactor", 30); - methodIndexMap.put("getNonWorkTimeFactor", 31); - methodIndexMap.put("getJointTourTimeFactor", 32); - methodIndexMap.put("getTransponderOwnership", 33); - - methodIndexMap.put("getFemaleInParty", 50); - methodIndexMap.put("getMaleInParty", 51); - methodIndexMap.put("getInboundFemaleBikeLogsum", 52); - methodIndexMap.put("getOutboundFemaleBikeLogsum", 53); - methodIndexMap.put("getInboundMaleBikeLogsum", 54); - methodIndexMap.put("getOutboundMaleBikeLogsum", 55); - - methodIndexMap.put("getIvtCoeff", 56); - methodIndexMap.put("getCostCoeff", 57); - methodIndexMap.put("getIncomeInDollars", 58); - methodIndexMap.put("getWalkSetLogSum", 59); - methodIndexMap.put("getPnrSetLogSum", 60); - methodIndexMap.put("getKnrSetLogSum", 61); - - methodIndexMap.put( "getOrigTaxiWaitTime", 70 ); - methodIndexMap.put( "getDestTaxiWaitTime", 71 ); - methodIndexMap.put( "getOrigSingleTNCWaitTime", 72 ); - methodIndexMap.put( "getDestSingleTNCWaitTime", 73 ); - methodIndexMap.put( "getOrigSharedTNCWaitTime", 74 ); - methodIndexMap.put( "getDestSharedTNCWaitTime", 75 ); - methodIndexMap.put( "getUseOwnedAV", 76); - - methodIndexMap.put("getNm_walkTime_out", 90); - methodIndexMap.put("getNm_walkTime_in", 91); - methodIndexMap.put("getNm_bikeTime_out", 92); - methodIndexMap.put("getNm_bikeTime_in", 93); - - methodIndexMap.put("getOriginMgra", 96); - methodIndexMap.put("getDestMgra", 97); - - - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 2: - returnValue = getIncomeCategory(); - break; - case 3: - returnValue = getAdults(); - break; - case 4: - returnValue = getFemale(); - break; - case 5: - returnValue = getHhSize(); - break; - case 6: - returnValue = getAutos(); - break; - case 7: - returnValue = getAge(); - break; - case 8: - returnValue = getTourCategoryJoint(); - break; - case 9: - returnValue = getNumberOfParticipantsInJointTour(); - break; - case 10: - returnValue = getWorkTourModeIsSov(); - break; - case 11: - returnValue = getWorkTourModeIsBike(); - break; - case 12: - returnValue = getWorkTourModeIsHov(); - break; - case 14: - returnValue = getPTazTerminalTime(); - break; - case 15: - returnValue = getATazTerminalTime(); - break; - case 16: - returnValue = getODUDen(); - break; - case 17: - returnValue = getOEmpDen(); - break; - case 18: - returnValue = getOTotInt(); - break; - case 19: - returnValue = getDDUDen(); - break; - case 20: - returnValue = getDEmpDen(); - break; - case 21: - returnValue = getDTotInt(); - break; - case 22: - returnValue = getTourCategoryEscort(); - break; - case 23: - returnValue = getMonthlyParkingCost(); - break; - case 24: - returnValue = getDailyParkingCost(); - break; - case 25: - returnValue = getHourlyParkingCost(); - break; - case 26: - returnValue = getReimburseProportion(); - break; - case 27: - returnValue = getPersonType(); - break; - case 28: - returnValue = getFreeParkingEligibility(); - break; - case 29: - returnValue = getParkingArea(); - break; - case 30: - returnValue = getWorkTimeFactor(); - break; - case 31: - returnValue = getNonWorkTimeFactor(); - break; - case 32: - returnValue = getJointTourTimeFactor(); - break; - case 33: - returnValue = getTransponderOwnership(); - break; - case 50: - returnValue = getFemaleInParty(); - break; - case 51: - returnValue = getMaleInParty(); - break; - case 52: - returnValue = getInboundFemaleBikeLogsum(); - break; - case 53: - returnValue = getOutboundFemaleBikeLogsum(); - break; - case 54: - returnValue = getInboundMaleBikeLogsum(); - break; - case 55: - returnValue = getOutboundMaleBikeLogsum(); - break; - case 56: - returnValue = getIvtCoeff(); - break; - case 57: - returnValue = getCostCoeff(); - break; - case 58: - returnValue = getIncomeInDollars(); - break; - case 59: - returnValue = getTransitLogSum(WTW, true) + getTransitLogSum(WTW, false); - break; - case 60: - returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); - break; - case 61: - returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); - break; - - case 70: return getOrigTaxiWaitTime(); - case 71: return getDestTaxiWaitTime(); - case 72: return getOrigSingleTNCWaitTime(); - case 73: return getDestSingleTNCWaitTime(); - case 74: return getOrigSharedTNCWaitTime(); - case 75: return getDestSharedTNCWaitTime(); - case 76: return getUseOwnedAV(); - - - - - - - - - - case 90: - returnValue = getNmWalkTimeOut(); - break; - case 91: - returnValue = getNmWalkTimeIn(); - break; - case 92: - returnValue = getNmBikeTimeOut(); - break; - case 93: - returnValue = getNmBikeTimeIn(); - break; - case 96: - returnValue = getOriginMgra(); - break; - case 97: - returnValue = getDestMgra(); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - } - - return returnValue; - - } -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java deleted file mode 100644 index 0ac87da..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTransponderChoiceDMU.java +++ /dev/null @@ -1,56 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; -import org.sandag.abm.ctramp.TransponderChoiceDMU; - -public class SandagTransponderChoiceDMU - extends TransponderChoiceDMU -{ - - public SandagTransponderChoiceDMU() - { - super(); - setupMethodIndexMap(); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getAutoOwnership", 0); - methodIndexMap.put("getPctHighIncome", 1); - methodIndexMap.put("getPctMultipleAutos", 2); - methodIndexMap.put("getAvgtts", 3); - methodIndexMap.put("getDistanceFromFacility", 4); - methodIndexMap.put("getPctAltTimeCBD", 5); - methodIndexMap.put("getAvgTransitAccess", 6); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getAutoOwnership(); - case 1: - return getPctIncome100Kplus(); - case 2: - return getPctTazMultpleAutos(); - case 3: - return getExpectedTravelTimeSavings(); - case 4: - return getTransponderDistance(); - case 5: - return getPctDetour(); - case 6: - return getAccessibility(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java deleted file mode 100644 index a2a3b2f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripModeChoiceDMU.java +++ /dev/null @@ -1,572 +0,0 @@ -package org.sandag.abm.application; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.BikeLogsum; -import org.sandag.abm.ctramp.BikeLogsumSegment; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Tour; -import org.sandag.abm.ctramp.TripModeChoiceDMU; - -import com.pb.common.calculator.IndexValues; - -public class SandagTripModeChoiceDMU - extends TripModeChoiceDMU -{ - private int setPersonHhTourCounter = 0; - private BikeLogsum bls; - protected double femaleBikeLogsum; - protected double maleBikeLogsum; - protected double femaleInParty; - protected double maleInParty; - - public SandagTripModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) - { - super(modelStructure, aLogger); - setupMethodIndexMap(); - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU destination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getEscortTour() - { - return escortTour; - } - - @Override - public int getTourCategoryJoint() - { - return jointTour; - } - - @Override - public int getNumberOfParticipantsInJointTour() - { - return partySize; - } - - public int getAutos() - { - return autos; - } - - public int getAge() - { - return age; - } - - public int getAdults() - { - return adults; - } - - public int getHhSize() - { - return hhSize; - } - - public int getFemale() - { - return personIsFemale; - } - - public int getIncome() - { - return incomeInDollars; - } - - @Override - public float getTimeOutbound() - { - return departPeriod; - } - - @Override - public float getTimeInbound() - { - return arrivePeriod; - } - - public int getTimeTrip() - { - return tripPeriod; - } - - public int getOutboundStops() - { - return outboundStops; - } - - public int getReturnStops() - { - return inboundStops; - } - - public int getTourModeIsDA() - { - return tourModeIsDA; - } - - public int getTourModeIsS2() - { - return tourModeIsS2; - } - - public int getTourModeIsS3() - { - return tourModeIsS3; - } - - public int getTourModeIsWalk() - { - return tourModeIsWalk; - } - - public int getTourModeIsBike() - { - return tourModeIsBike; - } - - public int getTourModeIsWTran() - { - return tourModeIsWTran; - } - - public int getTourModeIsPNR() - { - return tourModeIsPnr; - } - - public int getTourModeIsKNR() - { - return tourModeIsKnr; - } - - public int getTourModeIsSchBus() - { - return tourModeIsSchBus; - } - - public void setBikeLogsum(BikeLogsum bls) - { - this.bls = bls; - } - - public void setPersonObject(Person person) - { - super.setPersonObject(person); - checkSetPersonHhTour(); - } - - public void setHouseholdObject(Household hh) - { - super.setHouseholdObject(hh); - checkSetPersonHhTour(); - } - - public void setTourObject(Tour tour) - { - super.setTourObject(tour); - checkSetPersonHhTour(); - } - - private void checkSetPersonHhTour() - { - setPersonHhTourCounter = (setPersonHhTourCounter+1) % 3; - if (setPersonHhTourCounter == 0) { - setParty(person,tour,hh); - } - } - - public double getFemaleInParty() - { - return femaleInParty; - } - - public double getMaleInParty() - { - return maleInParty; - } - - public void setParty(Person person, Tour tour, Household hh) - { - if (person != null) { - femaleInParty = person.getPersonIsFemale(); - maleInParty = femaleInParty == 0 ? 1 : 0; - } else { - femaleInParty = 0; - maleInParty = 0; - for (int participant : tour.getPersonNumArray()) { - if (hh.getPerson(participant).getPersonIsFemale() == 1) - femaleInParty = 1; - else - maleInParty = 1; - } - } - } - - public void setBikeLogsum(int origin, int dest, boolean inbound) { - boolean mandatory = tour.getTourPrimaryPurposeIndex() <= 3; - femaleBikeLogsum = bls.getLogsum(new BikeLogsumSegment(true,mandatory,inbound),origin,dest); - maleBikeLogsum = bls.getLogsum(new BikeLogsumSegment(false,mandatory,inbound),origin,dest); - } - - public double getFemaleBikeLogsum() { - return femaleBikeLogsum; - } - - public double getMaleBikeLogsum() { - return maleBikeLogsum; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getAutos", 1); - methodIndexMap.put("getAdults", 2); - methodIndexMap.put("getHhSize", 3); - methodIndexMap.put("getFemale", 4); - methodIndexMap.put("getIncomeCategory", 5); - methodIndexMap.put("getTimeOutbound", 6); - methodIndexMap.put("getTimeInbound", 7); - methodIndexMap.put("getTimeTrip", 8); - methodIndexMap.put("getTourCategoryJoint", 9); - methodIndexMap.put("getNumberOfParticipantsInJointTour", 10); - methodIndexMap.put("getOutboundStops", 11); - methodIndexMap.put("getReturnStops", 12); - methodIndexMap.put("getFirstTrip", 13); - methodIndexMap.put("getLastTrip", 14); - methodIndexMap.put("getTourModeIsDA", 15); - methodIndexMap.put("getTourModeIsS2", 16); - methodIndexMap.put("getTourModeIsS3", 17); - methodIndexMap.put("getTourModeIsWalk", 18); - methodIndexMap.put("getTourModeIsBike", 19); - methodIndexMap.put("getTourModeIsWTran", 20); - methodIndexMap.put("getTourModeIsPNR", 21); - methodIndexMap.put("getTourModeIsKNR", 22); - methodIndexMap.put("getODUDen", 23); - methodIndexMap.put("getOEmpDen", 24); - methodIndexMap.put("getOTotInt", 25); - methodIndexMap.put("getDDUDen", 26); - methodIndexMap.put("getDEmpDen", 27); - methodIndexMap.put("getDTotInt", 28); - methodIndexMap.put("getPTazTerminalTime", 30); - methodIndexMap.put("getATazTerminalTime", 31); - methodIndexMap.put("getAge", 32); - methodIndexMap.put("getTourModeIsSchBus", 33); - methodIndexMap.put("getEscortTour", 34); - methodIndexMap.put("getAutoModeAllowedForTripSegment", 35); - methodIndexMap.put("getWalkModeAllowedForTripSegment", 36); - methodIndexMap.put("getSegmentIsIk", 37); - methodIndexMap.put("getReimburseAmount", 38); - methodIndexMap.put("getMonthlyParkingCostTourDest", 39); - methodIndexMap.put("getDailyParkingCostTourDest", 40); - methodIndexMap.put("getHourlyParkingCostTourDest", 41); - methodIndexMap.put("getHourlyParkingCostTripOrig", 42); - methodIndexMap.put("getHourlyParkingCostTripDest", 43); - methodIndexMap.put("getTripOrigIsTourDest", 44); - methodIndexMap.put("getTripDestIsTourDest", 45); - methodIndexMap.put("getFreeOnsite", 46); - methodIndexMap.put("getPersonType", 47); - - methodIndexMap.put("getFemaleInParty", 50); - methodIndexMap.put("getMaleInParty", 51); - methodIndexMap.put("getFemaleBikeLogsum", 52); - methodIndexMap.put("getMaleBikeLogsum", 53); - - methodIndexMap.put("getTransponderOwnership", 54); - methodIndexMap.put("getWorkTimeFactor", 55); - methodIndexMap.put("getNonWorkTimeFactor", 56); - methodIndexMap.put("getJointTourTimeFactor", 57); - - methodIndexMap.put("getInbound",58); - - methodIndexMap.put("getIncomeInDollars",59); - methodIndexMap.put("getIvtCoeff", 60); - methodIndexMap.put("getCostCoeff", 61); - - methodIndexMap.put("getWalkSetLogSum", 62); - methodIndexMap.put("getPnrSetLogSum", 63); - methodIndexMap.put("getKnrSetLogSum", 64); - - methodIndexMap.put("getWaitTimeTaxi", 70); - methodIndexMap.put("getWaitTimeSingleTNC", 71); - methodIndexMap.put("getWaitTimeSharedTNC", 72); - methodIndexMap.put("getUseOwnedAV", 73); - - methodIndexMap.put("getNm_walkTime", 90); - methodIndexMap.put("getNm_bikeTime", 91); - - methodIndexMap.put("getOriginMgra", 93); - methodIndexMap.put("getDestMgra", 94); - - - methodIndexMap.put("getTourModeIsTncTransit", 95); - methodIndexMap.put("getTourModeIsMaas", 96); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 1: - returnValue = getAutos(); - break; - case 2: - returnValue = getAdults(); - break; - case 3: - returnValue = getHhSize(); - break; - case 4: - returnValue = getFemale(); - break; - case 5: - returnValue = getIncome(); - break; - case 6: - returnValue = getTimeOutbound(); - break; - case 7: - returnValue = getTimeInbound(); - break; - case 8: - returnValue = getTimeTrip(); - break; - case 9: - returnValue = getTourCategoryJoint(); - break; - case 10: - returnValue = getNumberOfParticipantsInJointTour(); - break; - case 11: - returnValue = getOutboundStops(); - break; - case 12: - returnValue = getReturnStops(); - break; - case 13: - returnValue = getFirstTrip(); - break; - case 14: - returnValue = getLastTrip(); - break; - case 15: - returnValue = getTourModeIsDA(); - break; - case 16: - returnValue = getTourModeIsS2(); - break; - case 17: - returnValue = getTourModeIsS3(); - break; - case 18: - returnValue = getTourModeIsWalk(); - break; - case 19: - returnValue = getTourModeIsBike(); - break; - case 20: - returnValue = getTourModeIsWTran(); - break; - case 21: - returnValue = getTourModeIsPnr(); - break; - case 22: - returnValue = getTourModeIsKnr(); - break; - case 23: - returnValue = getODUDen(); - break; - case 24: - returnValue = getOEmpDen(); - break; - case 25: - returnValue = getOTotInt(); - break; - case 26: - returnValue = getDDUDen(); - break; - case 27: - returnValue = getDEmpDen(); - break; - case 28: - returnValue = getDTotInt(); - break; - case 30: - returnValue = getPTazTerminalTime(); - break; - case 31: - returnValue = getATazTerminalTime(); - break; - case 32: - returnValue = getAge(); - break; - case 33: - returnValue = getTourModeIsSchBus(); - break; - case 34: - returnValue = getEscortTour(); - break; - case 35: - returnValue = getAutoModeAllowedForTripSegment(); - break; - case 36: - returnValue = getWalkModeAllowedForTripSegment(); - break; - case 37: - returnValue = getSegmentIsIk(); - break; - case 38: - returnValue = getReimburseAmount(); - break; - case 39: - returnValue = getMonthlyParkingCostTourDest(); - break; - case 40: - returnValue = getDailyParkingCostTourDest(); - break; - case 41: - returnValue = getHourlyParkingCostTourDest(); - break; - case 42: - returnValue = getHourlyParkingCostTripOrig(); - break; - case 43: - returnValue = getHourlyParkingCostTripDest(); - break; - case 44: - returnValue = getTripOrigIsTourDest(); - break; - case 45: - returnValue = getTripDestIsTourDest(); - break; - case 46: - returnValue = getFreeOnsite(); - break; - case 47: - returnValue = getPersonType(); - break; - case 50: - returnValue = getFemaleInParty(); - break; - case 51: - returnValue = getMaleInParty(); - break; - case 52: - returnValue = getFemaleBikeLogsum(); - break; - case 53: - returnValue = getMaleBikeLogsum(); - break; - case 54: - returnValue = getTransponderOwnership(); - break; - case 55: - returnValue = getWorkTimeFactor(); - break; - case 56: - returnValue = getNonWorkTimeFactor(); - break; - case 57: - returnValue = getJointTourTimeFactor(); - break; - case 58: - returnValue = getInbound(); - break; - case 59: - returnValue = getIncomeInDollars(); - break; - case 60: - returnValue = getIvtCoeff(); - break; - case 61: - returnValue = getCostCoeff(); - break; - case 62: - returnValue = getTransitLogSum(WTW); - break; - case 63: - if ( outboundHalfTourDirection == 1 ) - returnValue = getTransitLogSum(DTW); - else - returnValue = getTransitLogSum(WTD); - break; - case 64: - if ( outboundHalfTourDirection == 1 ) - returnValue = getTransitLogSum(DTW); - else - returnValue = getTransitLogSum(WTD); - break; - case 70: return getWaitTimeTaxi(); - case 71: return getWaitTimeSingleTNC(); - case 72: return getWaitTimeSharedTNC(); - case 73: return getUseOwnedAV(); - case 90: - returnValue = getNm_walkTime(); - break; - case 91: - returnValue = getNm_bikeTime(); - break; - case 93: - returnValue = getOriginMgra(); - break; - case 94: - returnValue = getDestMgra(); - break; - case 95: - returnValue = getTourModeIsTncTransit(); - break; - case 96: - returnValue = getTourModeIsMaas(); - break; - - - default: - logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - } -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java deleted file mode 100644 index a8c81f0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/application/SandagTripTables.java +++ /dev/null @@ -1,904 +0,0 @@ -package org.sandag.abm.application; - -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixIO32BitJvm; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class SandagTripTables -{ - - private static Logger logger = Logger.getLogger("tripTables"); - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; - private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; - - - private TableDataSet indivTripData; - private TableDataSet jointTripData; - - // Some parameters - private int[] modeIndex; // an - private int[] matrixIndex; // an - - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private String[] purposeName = {"Work", "University", "School", - "Escort", "Shop", "Maintenance", "EatingOut", "Visiting", "Discretionary", "WorkBased"}; - - // matrices are indexed by modes (auto, non-mot,tran,other), valueoftime bins, and sub-modes(shared2gp, etc). - private Matrix[][][] matrix; - - private HashMap rbMap; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - - - private float[][] CBDVehicles; // an // period - private float[][] PNRVehicles; // an - private float sampleRate; - private int iteration; - private MatrixType mt; - private MatrixDataServerRmi ms; - - private String[] indivColumns = {"stop_period", "orig_mgra", - "dest_mgra", "trip_mode", "inbound", "trip_board_tap", "trip_alight_tap", "set", - "parking_mgra", "tour_purpose", "valueOfTime", "transponder_avail" }; - - private String[] jointColumns = {"stop_period", "orig_mgra", - "dest_mgra", "trip_mode", "inbound", "trip_board_tap", "trip_alight_tap", "set", - "parking_mgra", "tour_purpose", "num_participants", "valueOfTime", "transponder_avail"}; - - private HashMap averageOcc3Plus; // a - - private float valueOfTimeThresholdLow = 0; - private float valueOfTimeThresholdMed = 0; - //value of time bins by mode group - int[] votBins = {3,1,1,1}; - - boolean segmentByTransponderOwnership; - - public int numSkimSets; - - - /** - * Constructor. - * - * @param rbMap - * HashMap formed from a property map, which includes environment - * variables and arguments passed in as -d to VM - * @param sampleRate - * Sample rate 0->1.0 - * @param iteration - * Iteration number, program will look for trip file names with - * _iteration appended - */ - public SandagTripTables(HashMap rbMap, float sampleRate, int iteration) - { - - this.rbMap = rbMap; - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - segmentByTransponderOwnership = Util.getBooleanValueFromPropertyMap(rbMap,"Results.segmentByTransponderOwnership"); - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTourModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - logger.info("Tour mode "+i+" is auto"); - } else if (modelStructure.getTourModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - logger.info("Tour mode "+i+" is non-motorized"); - } else if (modelStructure.getTourModeIsWalkTransit(i) - || modelStructure.getTourModeIsDriveTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - logger.info("Tour mode "+i+" is transit"); - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - logger.info("Tour mode "+i+" is other"); - } - } - logger.info("Total auto modes = "+autoModes); - logger.info("Total non-motorized modes = "+nmotModes); - logger.info("Total transit modes = "+tranModes); - logger.info("Total other modes = "+othrModes); - - readOccupancies(); - // Initialize arrays (need for all periods, so initialize here) - CBDVehicles = new float[mgraManager.getMaxMgra() + 1][numberOfPeriods]; - PNRVehicles = new float[tapManager.getMaxTap() + 1][numberOfPeriods]; - - setSampleRate(sampleRate); - setIteration(iteration); - - //value of time thresholds - valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); - valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); - - } - - /** - * Read occupancies from the properties file and store in the - * averageOcc3Plus HashMap - */ - public void readOccupancies() - { - - averageOcc3Plus = new HashMap(); - - for (int i = 0; i < purposeName.length; ++i) - { - String searchString = "occ3plus.purpose." + purposeName[i]; - float occupancy = new Float(Util.getStringValueFromPropertyMap(rbMap, searchString)); - averageOcc3Plus.put(purposeName[i], occupancy); - } - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) and value of time group - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][][]; - for (int i = 0; i < numberOfModes; ++i) - { - matrix[i] = new Matrix[votBins[i]][]; - - for(int j = 0; j< votBins[i];++j){ - - String modeName; - - - if (i == 0) - { - - int autoModeSegments = autoModes; - String transponderLabel = ""; - - if(segmentByTransponderOwnership) { - autoModeSegments *=2; //twice as many since segmentation would be by number of auto modes and by 0,1 for ownership - transponderLabel = "NOTRPDR"; - } - matrix[i][j] = new Matrix[autoModeSegments]; - - for (int k = 0; k < autoModes; ++k) - { - modeName = modelStructure.getModeName(k + 1); - matrix[i][j][k] = new Matrix(modeName + transponderLabel + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - - for (int k = autoModes; k < autoModeSegments; ++k) - { - modeName = modelStructure.getModeName((k + 1)-autoModes); - matrix[i][j][k] = new Matrix(modeName + "TRPDR"+ "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - - - - } else if (i == 1){ - - matrix[i][j] = new Matrix[nmotModes]; - for (int k = 0; k < nmotModes; ++k) - { - modeName = modelStructure.getModeName(k + 1 + autoModes); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - - } else if (i == 2){ - - matrix[i][j] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l1) - votBin = getValueOfTimeBin(valueOfTime); - - if (mode == 0) - { - if(segmentByTransponderOwnership) { - int ownsTransponder = (int) tripData.getValueAt(i, "transponder_avail"); - if(ownsTransponder==1) - mat = mat + SandagModelStructure.TRIP_SOV_ALTS.length + SandagModelStructure.TRIP_HOV_ALTS.length; - } - // look up what taz the parking mgra is in, and re-assign the - // trip destination to the parking taz - if (parkingMGRA > 0) - { - parkingTaz = mgraManager.getTaz(parkingMGRA); - destinationTAZ = parkingTaz; - CBDVehicles[parkingMGRA][period] = CBDVehicles[parkingMGRA][period] - + vehicleTrips; - } - - - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); - } else if (mode == 1) - { - - - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } else if (mode == 2) - { - - if (boardTap == 0 || alightTap == 0) continue; - - //store transit trips in matrices - mat = (matrixIndex[tripMode]*numSkimSets)+set; - - float value=0; - try{ - value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); - }catch(Exception e){ - - logger.fatal("Error trying to get transit trips from matrix"); - logger.fatal("boardTap,alightTap,set: "+boardTap+","+alightTap+","+set); - logger.fatal("tripMode,mode,votBin,mat: "+tripMode+","+mode+","+votBin+","+mat); - logger.fatal("number of skimsets: "+numSkimSets); - logger.fatal("total board taps in matrix:" + matrix[mode][votBin][mat].getRowCount()); - logger.fatal("total alight taps in matrix:" + matrix[mode][votBin][mat].getColumnCount()); - throw new RuntimeException(e); - } - matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); - - // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) - if (modelStructure.getTourModeIsDriveTransit(tripMode)) - { - - // add the tNCVehicle trip portion to the trip table - if (inbound == 0) - { // from origin to lot (boarding tap) - int PNRTAZ = tapManager.getTazForTap(boardTap); - - - value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); - matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); - - // and increment up the array of parked vehicles at the - // lot - ++PNRVehicles[boardTap][period]; - - } else - { // from lot (alighting tap) to destination - int PNRTAZ = tapManager.getTazForTap(alightTap); - - - value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); - matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); - } - - } - } else - { - - - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } - } - - //logger.info("End creating trip tables for period " + timePeriod); - } - - /** - * Return the value of time bin 0 through 2 based on the thresholds provided in the property map - * @param valueOfTime - * @return value of time bin 0 through 2 - */ - public int getValueOfTimeBin(float valueOfTime){ - - if(valueOfTime1) - end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; - else - end[i][j] = "_" + per + ".omx"; - } - } - - for (int i = 0; i < 4; ++i) - { - for(int j = 0; j < votBins[i];++j){ - try - { - //Delete the file if it exists - File f = new File(fileName[i]+end[i][j]); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); - f.delete(); - } - - if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); - else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); - } catch (Exception e) - { - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName[i] +end[i][j] + ", for mode index = " + i, e); - throw new RuntimeException(); - } - } - } - - } - - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - /** - * Connect to matrix server - */ - private void connectToMatrixServer() - { - - - // get matrix server address and port - String matrixServerAddress = Util.getStringValueFromPropertyMap(rbMap, "RunModel.MatrixServerAddress"); - int serverPort = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, "RunModel.MatrixServerPort")); - - ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - logger.info("connected to matrix data server"); - } - - /** - * Write a file of vehicles parking in parking-constrained areas by MGRA. - * - * @param fileName - * The name of the csv file to write to. - */ - public void writeCBDFile(String fileName) - { - - try - { - FileWriter writer = new FileWriter(fileName); - - // write header - writer.append("MGRA,"); - - for (int j = 0; j < numberOfPeriods; ++j) - writer.append(modelStructure.getModelPeriodLabel(j) + ","); - - writer.append("Total\n"); - - // iterate through mgras - for (int i = 0; i < CBDVehicles.length; ++i) - { - - float totalVehicles = 0; - for (int j = 0; j < numberOfPeriods; ++j) - { - totalVehicles += CBDVehicles[i][j]; - } - - // only write the mgra if there are vehicles parked there - if (totalVehicles > 0) - { - - writer.append(Integer.toString(i)); - - // iterate through periods - for (int j = 0; j < numberOfPeriods; ++j) - writer.append("," + Float.toString(CBDVehicles[i][j])); - - writer.append("," + Float.toString(totalVehicles) + "\n"); - writer.flush(); - } - } - writer.flush(); - writer.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - - } - - /** - * Write a file of vehicles parking in PNR lots by TAP. - * - * @param fileName - * The name of the csv file to write to. - */ - public void writePNRFile(String fileName) - { - - try - { - FileWriter writer = new FileWriter(fileName); - - // write header - writer.append("TAP,"); - - for (int j = 0; j < numberOfPeriods; ++j) - writer.append(modelStructure.getModelPeriodLabel(j) + ","); - - writer.append("Total\n"); - - // iterate through taps - for (int i = 0; i < PNRVehicles.length; ++i) - { - - float totalVehicles = 0; - for (int j = 0; j < numberOfPeriods; ++j) - { - totalVehicles += PNRVehicles[i][j]; - } - - // only write the tap if there are vehicles parked there - if (totalVehicles > 0) - { - - writer.append(Integer.toString(i)); - - // iterate through periods - for (int j = 0; j < numberOfPeriods; ++j) - writer.append("," + Float.toString(PNRVehicles[i][j])); - - writer.append("," + Float.toString(totalVehicles) + "\n"); - writer.flush(); - } - } - writer.flush(); - writer.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - - } - - /** - * Set the sample rate - * - * @param sampleRate - * The sample rate, used for expanding trips - */ - public void setSampleRate(float sampleRate) - { - this.sampleRate = sampleRate; - } - - /** - * Set the iteration number - * - * @param sampleRate - * The iteration number, should be appended to trip files as - * _iteration - */ - public void setIteration(int iteration) - { - this.iteration = iteration; - } - - - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - - float sampleRate = 1.0f; - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - - logger.info(String.format("-sampleRate %.4f.", sampleRate)); - logger.info("-iteration " + iteration); - - SandagTripTables tripTables = new SandagTripTables(pMap, sampleRate, iteration); - - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - tripTables.mt = MatrixType.lookUpMatrixType(matrixTypeName); - tripTables.createTripTables(tripTables.mt); - - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java deleted file mode 100644 index 00b2800..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactory.java +++ /dev/null @@ -1,51 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.crossborder; - -import java.io.Serializable; - -/** - * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects - * - * @author Joel Freedman - */ -public class CrossBorderDmuFactory - implements CrossBorderDmuFactoryIf, Serializable -{ - - private CrossBorderModelStructure crossBorderModelStructure; - - public CrossBorderDmuFactory(CrossBorderModelStructure modelStructure) - { - this.crossBorderModelStructure = modelStructure; - } - - public CrossBorderTourModeChoiceDMU getCrossBorderTourModeChoiceDMU() - { - return new CrossBorderTourModeChoiceDMU(crossBorderModelStructure); - } - - public CrossBorderTripModeChoiceDMU getCrossBorderTripModeChoiceDMU() - { - return new CrossBorderTripModeChoiceDMU(crossBorderModelStructure, null); - } - - public CrossBorderStationDestChoiceDMU getCrossBorderStationChoiceDMU() - { - return new CrossBorderStationDestChoiceDMU(crossBorderModelStructure); - } - - public CrossBorderStopLocationChoiceDMU getCrossBorderStopLocationChoiceDMU() - { - return new CrossBorderStopLocationChoiceDMU(crossBorderModelStructure); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java deleted file mode 100644 index 70b68da..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderDmuFactoryIf.java +++ /dev/null @@ -1,17 +0,0 @@ -package org.sandag.abm.crossborder; - -/** - * A DMU factory interface - */ -public interface CrossBorderDmuFactoryIf -{ - - CrossBorderTourModeChoiceDMU getCrossBorderTourModeChoiceDMU(); - - CrossBorderStationDestChoiceDMU getCrossBorderStationChoiceDMU(); - - CrossBorderTripModeChoiceDMU getCrossBorderTripModeChoiceDMU(); - - CrossBorderStopLocationChoiceDMU getCrossBorderStopLocationChoiceDMU(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java deleted file mode 100644 index 48f2c2b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModel.java +++ /dev/null @@ -1,525 +0,0 @@ -package org.sandag.abm.crossborder; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.concurrent.ForkJoinPool; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.crossborder.CrossBorderTourManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; -import com.pb.sawdust.util.concurrent.DnCRecursiveAction; - -public class CrossBorderModel -{ - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - public static final String RUN_MODEL_CONCURRENT_PROPERTY_KEY = "crossBorder.run.concurrent"; - public static final String CONCURRENT_PARALLELISM_PROPERTY_KEY = "crossBorder.concurrent.parallelism"; - - private static final Logger LOGGER = Logger.getLogger(SandagTourBasedModel.class); - private static final Object INITIALIZATION_LOCK = new Object(); - - private MatrixDataServerRmi ms; - private HashMap rbMap; - private AutoTazSkimsCalculator tazDistanceCalculator; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private CrossBorderTourManager tourManager; - - private boolean seek; - private int traceId; - private double sampleRate = 1; - - /** - * Constructor - * - * @param rbMap - */ - public CrossBorderModel(HashMap rbMap) - { - this.rbMap = rbMap; - - synchronized (INITIALIZATION_LOCK) - { // lock to make sure only one of - // these actually initializes - // things so we don't cross - // threads - mgraManager = MgraDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - tourManager = new CrossBorderTourManager(rbMap); - } - - seek = Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.seek")); - traceId = Integer.valueOf(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trace")); - - } - - // global variable used for reporting - private static final AtomicInteger TOUR_COUNTER = new AtomicInteger(0); - private final AtomicBoolean calculatorsInitialized = new AtomicBoolean(false); - - /** - * Run the model for a subset of tours in an array of tours. - * - * @param tours - * The array of tours. - * @param start - * The starting index of the tours to process. - * @param end - * The (exclusive) ending index of the tours to process. - */ - private void runModel(CrossBorderTour[] tours, int start, int end) - { - CrossBorderModelStructure modelStructure = new CrossBorderModelStructure(); - CrossBorderDmuFactoryIf dmuFactory = new CrossBorderDmuFactory(modelStructure); - - if (!calculatorsInitialized.get()) - { - // only let one thread in to initialize - synchronized (calculatorsInitialized) - { - // if still not initialized, then this is the first in so do the - // initialization (otherwise skip) - if (!calculatorsInitialized.get()) - { - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - calculatorsInitialized.set(true); - } - } - } - - CrossBorderTourTimeOfDayChoiceModel todChoiceModel = new CrossBorderTourTimeOfDayChoiceModel( - rbMap); - CrossBorderStationDestChoiceModel destChoiceModel = new CrossBorderStationDestChoiceModel( - rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - CrossBorderTourModeChoiceModel tourModeChoiceModel = new CrossBorderTourModeChoiceModel(rbMap, modelStructure, dmuFactory, - tazDistanceCalculator); - - CrossBorderTripModeChoiceModel tripModeChoiceModel = new CrossBorderTripModeChoiceModel(rbMap, modelStructure, - dmuFactory, tazDistanceCalculator); - destChoiceModel.calculateSizeTerms(dmuFactory); - destChoiceModel.calculateTazProbabilities(dmuFactory); - - CrossBorderStopFrequencyModel stopFrequencyModel = new CrossBorderStopFrequencyModel(rbMap); - CrossBorderStopPurposeModel stopPurposeModel = new CrossBorderStopPurposeModel(rbMap); - - CrossBorderStopTimeOfDayChoiceModel stopTodChoiceModel = new CrossBorderStopTimeOfDayChoiceModel( - rbMap); - CrossBorderStopLocationChoiceModel stopLocationChoiceModel = new CrossBorderStopLocationChoiceModel( - rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - - double[][] mgraSizeTerms = destChoiceModel.getMgraSizeTerms(); - double[][] tazSizeTerms = destChoiceModel.getTazSizeTerms(); - double[][][] mgraProbabilities = destChoiceModel.getMgraProbabilities(); - stopLocationChoiceModel.setMgraSizeTerms(mgraSizeTerms); - stopLocationChoiceModel.setTazSizeTerms(tazSizeTerms); - stopLocationChoiceModel.setMgraProbabilities(mgraProbabilities); - String purposeControlFileName = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory")+"input/crossBorder_tourPurpose_control.csv"; - - for (int i = start; i < end; i++) - { - CrossBorderTour tour = tours[i]; - - // sample tours - double rand = tour.getRandom(); - if (rand > sampleRate) continue; - - int tourCount = TOUR_COUNTER.incrementAndGet(); - if (tourCount % 1000 == 0) LOGGER.info("Processing tour " + tourCount); - - if (seek && tour.getID() != traceId) continue; - - if (tour.getID() == traceId) tour.setDebugChoiceModels(true); - - todChoiceModel.calculateTourTOD(tour); - destChoiceModel.chooseStationAndDestination(tour); - resetCrossingPurpose(purposeControlFileName,tour); - tourModeChoiceModel.chooseTourMode(tour); - stopFrequencyModel.calculateStopFrequency(tour); - stopPurposeModel.calculateStopPurposes(tour); - - int outboundStops = tour.getNumberOutboundStops(); - int inboundStops = tour.getNumberInboundStops(); - - // choose TOD for stops and location of each - if (outboundStops > 0) - { - CrossBorderStop[] stops = tour.getOutboundStops(); - for (CrossBorderStop stop : stops) - { - stopTodChoiceModel.chooseTOD(tour, stop); - stopLocationChoiceModel.chooseStopLocation(tour, stop); - } - } - if (inboundStops > 0) - { - CrossBorderStop[] stops = tour.getInboundStops(); - for (CrossBorderStop stop : stops) - { - stopTodChoiceModel.chooseTOD(tour, stop); - stopLocationChoiceModel.chooseStopLocation(tour, stop); - } - } - - // generate trips and choose mode for them - CrossBorderTrip[] trips = new CrossBorderTrip[outboundStops + inboundStops + 2]; - int tripNumber = 0; - - // outbound stops - if (outboundStops > 0) - { - CrossBorderStop[] stops = tour.getOutboundStops(); - for (CrossBorderStop stop : stops) - { - // generate a trip to the stop and choose a mode for it - trips[tripNumber] = new CrossBorderTrip(tour, stop, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - // generate a trip from the last stop to the tour destination - trips[tripNumber] = new CrossBorderTrip(tour, stops[stops.length - 1], false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - } else - { - // generate an outbound trip from the tour origin to the - // destination and choose a mode - trips[tripNumber] = new CrossBorderTrip(tour, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - - // inbound stops - if (inboundStops > 0) - { - CrossBorderStop[] stops = tour.getInboundStops(); - for (CrossBorderStop stop : stops) - { - // generate a trip to the stop and choose a mode for it - trips[tripNumber] = new CrossBorderTrip(tour, stop, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - // generate a trip from the last stop to the tour origin - trips[tripNumber] = new CrossBorderTrip(tour, stops[stops.length - 1], false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } else - { - - // generate an inbound trip from the tour destination to the - // origin and choose a mode - trips[tripNumber] = new CrossBorderTrip(tour, false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - - // set the trips in the tour object - tour.setTrips(trips); - - } - - } - - /** - * This class is the divide-and-conquer action (void return task) for - * running the cross-border model using the fork-join framework. The - * divisible problem is an array of tours, and the actual work is the - * {@link CrossBorderModel#runModel(CrossBorderTour[],int,int)} method, - * applied to a section of the array. - */ - private class CrossBorderModelAction - extends DnCRecursiveAction - { - private final HashMap rbMap; - private final CrossBorderTour[] tours; - - private CrossBorderModelAction(HashMap rbMap, CrossBorderTour[] tours) - { - super(0, tours.length); - this.rbMap = rbMap; - this.tours = tours; - } - - private CrossBorderModelAction(HashMap rbMap, CrossBorderTour[] tours, - long start, long length, DnCRecursiveAction next) - { - super(start, length, next); - this.rbMap = rbMap; - this.tours = tours; - } - - @Override - protected void computeAction(long start, long length) - { - runModel(tours, (int) start, (int) (start + length)); - } - - @Override - protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) - { - return new CrossBorderModelAction(rbMap, tours, start, length, next); - } - - @Override - protected boolean continueDividing(long length) - { - // if there are 3 extra tasks queued up, then start executing - // if there are 1000 or less tours to process, then start executing - // otherwise, keep dividing to build up tasks for the threads to - // process - return getSurplusQueuedTaskCount() < 3 && length > 1000; - } - } - - /** - * Run visitor model. - */ - public void runModel() - { - tourManager.generateCrossBorderTours(); - CrossBorderTour[] tours = tourManager.getTours(); - - // get new keys to see if we want to run in concurrent mode, and the - // parallelism - // (defaults to single threaded and parallelism = # of processors) - // note that concurrent can use up memory very quickly, so setting the - // parallelism might be prudent - boolean concurrent = rbMap.containsKey(RUN_MODEL_CONCURRENT_PROPERTY_KEY) - && Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, - RUN_MODEL_CONCURRENT_PROPERTY_KEY)); - int parallelism = rbMap.containsKey(CONCURRENT_PARALLELISM_PROPERTY_KEY) ? Integer - .valueOf(Util.getStringValueFromPropertyMap(rbMap, - CONCURRENT_PARALLELISM_PROPERTY_KEY)) : Runtime.getRuntime() - .availableProcessors(); - - if (concurrent) - { // use fork-join - CrossBorderModelAction action = new CrossBorderModelAction(rbMap, tours); - new ForkJoinPool(parallelism).execute(action); - action.getResult(); // wait for finish - } else - { // single-threaded: call the model runner in this thread - runModel(tours, 0, tours.length); - } - - tourManager.writeOutputFile(rbMap); - LOGGER.info("Cross Border Model successfully completed!"); - } - - /** - * @return the sampleRate - */ - public double getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(double sampleRate) - { - this.sampleRate = sampleRate; - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - LOGGER.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - LOGGER.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * Reset tour NB crossing purpose using purpose distribution by POE from 2011 survey - */ - private void resetCrossingPurpose(String purposeControlFile, CrossBorderTour tour) { - int poe=tour.getPoe(); - double [][] purpDistributionByPoe=new double[5][CrossBorderModelStructure.NUMBER_CROSSBORDER_PURPOSES]; - - // Read the distributions by poe - for (int i=0; i<5; i++) { - purpDistributionByPoe[i] = tourManager.setPurposeDistribution(purposeControlFile,purpDistributionByPoe[i],i+3); - } - int purpose = tourManager.choosePurpose(tour.getRandom(), purpDistributionByPoe[poe]); - tour.setPurpose((byte) purpose); - } - - /** - * @param args - */ - public static void main(String[] args) - { - Runtime gfg = Runtime.getRuntime(); - long memory1; - // checking the total memeory - System.out.println("Total memory is: "+ gfg.totalMemory()); - // checking free memory - memory1 = gfg.freeMemory(); - System.out.println("Initial free memory at Xborder model: "+ memory1); - // calling the garbage collector on demand - gfg.gc(); - memory1 = gfg.freeMemory(); - System.out.println("Free memory after garbage "+ "collection: " + memory1); - - String propertiesFile = null; - HashMap pMap; - - LOGGER.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - LOGGER.info(String.format("Running Cross-Border Model")); - - if (args.length == 0) - { - LOGGER.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - float sampleRate = 1.0f; - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - - LOGGER.info("Crossborder Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - CrossBorderModel crossBorderModel = new CrossBorderModel(pMap); - crossBorderModel.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = crossBorderModel.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - crossBorderModel.ms = matrixServer; - } else - { - crossBorderModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - crossBorderModel.ms.testRemote("CrossBorderModel"); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(crossBorderModel.ms); - } - - } - - } catch (Exception e) - { - - LOGGER.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - crossBorderModel.runModel(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java deleted file mode 100644 index d8baad1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderModelStructure.java +++ /dev/null @@ -1,106 +0,0 @@ -package org.sandag.abm.crossborder; - -import org.sandag.abm.application.SandagModelStructure; - -public class CrossBorderModelStructure - extends SandagModelStructure -{ - - public static final byte NUMBER_CROSSBORDER_PURPOSES = 6; - public static final byte WORK = 0; - public static final byte SCHOOL = 1; - public static final byte CARGO = 2; - public static final byte SHOP = 3; - public static final byte VISIT = 4; - public static final byte OTHER = 5; - - public static final String[] CROSSBORDER_PURPOSES = {"WORK", "SCHOOL", "CARGO", "SHOP", - "VISIT", "OTHER" }; - - public static final byte DEPARTURE = 0; - public static final byte ARRIVAL = 1; - - public static final int AM = 0; - public static final int PM = 1; - public static final int OP = 2; - public static final int[] SKIM_PERIODS = {AM, PM, OP}; - public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; - public static final int UPPER_EA = 3; - public static final int UPPER_AM = 9; - public static final int UPPER_MD = 22; - public static final int UPPER_PM = 29; - public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; - - public static final byte TOUR_MODES = 4; - - public static final byte DRIVEALONE = 1; - public static final byte SHARED2 = 2; - public static final byte SHARED3 = 3; - public static final byte WALK = 4; - - // note that time periods start at 1 and go to 40 - public static final byte TIME_PERIODS = 40; - - /** - * Calculate and return the destination choice size term segment - * - * @param purpose - * @return Right now, just the purpose is returned. - */ - public static int getDCSizeSegment(int purpose) - { - - return purpose; - - } - - /** - * Calculate the purpose from the dc size segment. - * - * @param segment - * The dc size segment (0-17) - * @return The purpose - */ - public static int getPurposeFromDCSizeSegment(int segment) - { - - return segment; - } - - /** - * return the Skim period index 0=am, 1=pm, 2=off-peak - */ - public static int getSkimPeriodIndex(int departPeriod) - { - - int skimPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; - else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; - else skimPeriodIndex = OP; - - return skimPeriodIndex; - - } - - /** - * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV - */ - public static int getModelPeriodIndex(int departPeriod) - { - - int modelPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; - else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; - else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; - else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; - else modelPeriodIndex = 4; - - return modelPeriodIndex; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java deleted file mode 100644 index b72dfc7..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceDMU.java +++ /dev/null @@ -1,405 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class CrossBorderStationDestChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("crossBorderModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected float tourDepartPeriod; - protected float tourArrivePeriod; - protected int purpose; - protected double[][] sizeTerms; // by - // purpose, - // alternative - // (station-taz - // or - // sampled - // station-mgra) - protected double[] stationSizeTerms; // by - // alternative - // (station-taz - // or - // sampled - // station-mgra) - protected double[] correctionFactors; // by - // alternative - // (sampled - // station-mgra - // pair, - // for - // full - // model - // only) - protected double[] tourModeLogsums; // by - // alternative - // (sampled - // station-mgra - // pair, - // for - // full - // model - // only) - protected int[] poeNumbers; // by - // alternative - // (station-taz - // or - // sampled - // station-mgra) - protected int[] originTazs; // by - // alternative - // (station-taz - // or - // sampled - // station-mgra) - protected int[] destinationTazs; // by - // alternative - // (station-taz - // or - // sampled - // station-mgra) - - protected double nmWalkTimeOut; - protected double nmWalkTimeIn; - protected double nmBikeTimeOut; - protected double nmBikeTimeIn; - protected double lsWgtAvgCostM; - protected double lsWgtAvgCostD; - protected double lsWgtAvgCostH; - - public CrossBorderStationDestChoiceDMU(CrossBorderModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Get the POE number for the alternative. - * - * @param alt - * Either station-taz or sampled station-mgra - * @return - */ - public int getPoe(int alt) - { - return poeNumbers[alt]; - } - - /** - * Set the poe number array - * - * @param poeNumbers - * An array of POE numbers, one for each alternative (either - * station-taz or sampled station-mgra) - */ - public void setPoeNumbers(int[] poeNumbers) - { - this.poeNumbers = poeNumbers; - } - - /** - * Get the tour mode choice logsum for the sampled station-mgra pair. - * - * @param alt - * Sampled station-mgra - * @return - */ - public double getTourModeLogsum(int alt) - { - return tourModeLogsums[alt]; - } - - /** - * Set the tour mode choice logsums - * - * @param poeNumbers - * An array of tour mode choice logsums, one for each alternative - * (sampled station-mgra) - */ - public void setTourModeLogsums(double[] logsums) - { - this.tourModeLogsums = logsums; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - /** - * @return the sizeTerms. The size term is the size of the alternative north - * of the border. It is indexed by alternative, where alternative is - * either taz-station pair or mgra-station pair, depending on - * whether the DMU is being used for the SOA model or the actual - * model. - */ - public double getSizeTerm(int alt) - { - return sizeTerms[purpose][alt]; - } - - /** - * @param sizeTerms - * the sizeTerms to set. The size term is the size of the - * alternative north of the border. It is indexed by alternative, - * where alternative is either taz-station pair or mgra-station - * pair, depending on whether the DMU is being used for the SOA - * model or the actual model. - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - /** - * @return the accessibility of the station to population south of the - * border. The size term is indexed by alternative, where - * alternative is either taz-station pair or mgra-station pair, - * depending on whether the DMU is being used for the SOA model or - * the actual model. - */ - public double getStationPopulationAccessibility(int alt) - { - return stationSizeTerms[alt]; - } - - /** - * @param accessibilities - * is the accessibility of the station to population south of the - * border. The size term is indexed by alternative, where - * alternative is either taz-station pair or mgra-station pair, - * depending on whether the DMU is being used for the SOA model - * or the actual model. - */ - public void setStationPopulationAccessibilities(double[] accessibilities) - { - this.stationSizeTerms = accessibilities; - } - - /** - * @return the correctionFactors - */ - public double getCorrectionFactor(int alt) - { - return correctionFactors[alt]; - } - - /** - * @param correctionFactors - * the correctionFactors to set - */ - public void setCorrectionFactors(double[] correctionFactors) - { - this.correctionFactors = correctionFactors; - } - - /** - * @return the origin taz - */ - public int getOriginTaz(int alt) - { - return originTazs[alt]; - } - - /** - * @param originTazs - * The origin tazs to set - */ - public void setOriginTazs(int[] originTazs) - { - this.originTazs = originTazs; - } - - /** - * @return the destination taz - */ - public int getDestinationTaz(int alt) - { - return destinationTazs[alt]; - } - - /** - * @param stopTazs - * The destination tazs to set - */ - public void setDestinationTazs(int[] destinationTazs) - { - this.destinationTazs = destinationTazs; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the purpose - */ - public int getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(int purpose) - { - this.purpose = purpose; - } - - public float getTimeOutbound() - { - return tourDepartPeriod; - } - - public float getTimeInbound() - { - return tourArrivePeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(float tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(float tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getStationPopulationAccessibility", 2); - methodIndexMap.put("getSizeTerm", 3); - methodIndexMap.put("getCorrectionFactor", 4); - methodIndexMap.put("getPoe", 5); - methodIndexMap.put("getPurpose", 6); - methodIndexMap.put("getTourModeLogsum", 7); - methodIndexMap.put("getOriginTaz", 8); - methodIndexMap.put("getDestinationTaz", 9); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 2: - returnValue = getStationPopulationAccessibility(arrayIndex); - break; - case 3: - returnValue = getSizeTerm(arrayIndex); - break; - case 4: - returnValue = getCorrectionFactor(arrayIndex); - break; - case 5: - returnValue = getPoe(arrayIndex); - break; - case 6: - returnValue = getPurpose(); - break; - case 7: - returnValue = getTourModeLogsum(arrayIndex); - break; - case 8: - returnValue = getOriginTaz(arrayIndex); - break; - case 9: - returnValue = getDestinationTaz(arrayIndex); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java deleted file mode 100644 index 14a6600..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStationDestChoiceModel.java +++ /dev/null @@ -1,818 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -/** - * This class is used for both the sample of alternatives and the full - * destination choice model for border crossing tours. - * - * The model first calculates a set of station-level logsums which represent the - * attractiveness of each station based upon the accessibility to Mexico - * populations (taking into account population size of Colonias and distance - * between each station and each Colonia. The model then creates a sample of - * alternatives where each alternative is a pair of border crossing station - * (entry MGRA) and destination MGRA in San Diego County, by tour purpose. This - * is sampled from for each tour, and a mode choice logsum is calculated for - * each station-MGRA pair. The full destination choice model is run on the - * sample with the mode choice logsums influencing station - destination choice, - * and a station-MGRA pair is chosen for each tour. - * - * @author Freedman - * - */ -public class CrossBorderStationDestChoiceModel -{ - - private double[][] mgraSizeTerms; // by purpose, MGRA - private double[][] tazSizeTerms; // by purpose, TAZ - private double[][] tazStationProbabilities; // by purpose, station-TAZ - // alternative - private double[][][] mgraProbabilities; // by purpose, TAZ, MGRA - private double[] stationLogsums; // by entry station, logsum - // from colonia to - // station - private double[] soaStationLogsums; // by station-TAZ - // alternative, station - // logsums - private double[][] soaSizeTerms; // by purpose, station-TAZ - // alternative, - // size terms for tazs - private int[] soaOriginTazs; // by station-TAZ - // alternative, origin Taz - private int[] soaDestinationTazs; // by station-TAZ - // alternative, destination - // Taz - - private int[] sampledDestinationMgra; // destination mgra for each - // of n - // samples - private int[] sampledEntryMgra; // entry mgra for each of n - // samples - private double[][] sampledSizeTerms; // size term for each of n - // samples (1st - // dimension is purpose) - private double[] sampledStationLogsums; // station logsum for each of - // n - // samples - private int[] sampledStations; // POE for each of n samples - private int[] sampledOriginTazs; // Origin Taz for each of n - // samples - private int[] sampledDestinationTazs; // Destination Taz for each - // of n - // samples - - private double[] sampledCorrectionFactors; // correction factor for each - // of - // n samples - private double[] tourModeChoiceLogsums; // mode choice logsum for - // each of n - // samples - - private class KeyClass - { - int station; - int mgra; - - @Override - public boolean equals(Object obj) - { - if (obj instanceof KeyClass) - { - return station == (((KeyClass) obj).station) && mgra == (((KeyClass) obj).mgra); - } - return false; - } - - @Override - public int hashCode() - { - return station * 10000 + mgra; - } - - } - - private KeyClass key; - private HashMap frequencyChosen; // by - // alternative, - // number - // of - // times - // chosen - - private TableDataSet alternativeData; // the - // alternatives, - // with - // the - // following - // fields: - // "EntryMGRA" - // - - // indicating - // border - // crossing - // entry - // MGRA - // "dest" - // - - // indicating - // the - // destination - // TAZ - // in - // San - // Diego - // County - - private int stations; // number - // of - // stations - private int sampleRate; - - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication soaModel; - private ChoiceModelApplication destModel; - private CrossBorderTourModeChoiceModel tourModeChoiceModel; - CrossBorderStationDestChoiceDMU dmu; - McLogsumsCalculator logsumHelper; - - private UtilityExpressionCalculator sizeTermUEC; - private Tracer tracer; - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - private boolean seek; - private HashMap rbMap; - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of cross border model DMUs - */ - public CrossBorderStationDestChoiceModel(HashMap rbMap, - CrossBorderModelStructure myStructure, CrossBorderDmuFactoryIf dmuFactory, - AutoTazSkimsCalculator tazDistanceCalculator) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String crossBorderDCSoaFileName = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.soa.uec.file"); - crossBorderDCSoaFileName = uecFileDirectory + crossBorderDCSoaFileName; - - int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.soa.data.page")); - int soaSizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.soa.size.page")); - int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.soa.model.page")); - - String crossBorderDCFileName = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.uec.file"); - crossBorderDCFileName = uecFileDirectory + crossBorderDCFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.data.page")); - int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.model.page")); - - // read the model pages from the property file, create one choice model - // for each - CrossBorderStationDestChoiceDMU dcDmu = dmuFactory.getCrossBorderStationChoiceDMU(); - - // create a ChoiceModelApplication object for the SOA model. - soaModel = new ChoiceModelApplication(crossBorderDCSoaFileName, soaModelPage, soaDataPage, - rbMap, (VariableTable) dcDmu); - - // create a ChoiceModelApplication object for the full model. - destModel = new ChoiceModelApplication(crossBorderDCFileName, modelPage, dataPage, rbMap, - (VariableTable) dcDmu); - sampleRate = destModel.getAlternativeNames().length; - - // get the alternative data from the model - UtilityExpressionCalculator uec = soaModel.getUEC(); - alternativeData = uec.getAlternativeData(); - - // create a UEC to solve size terms for each MGRA - sizeTermUEC = new UtilityExpressionCalculator(new File(crossBorderDCSoaFileName), - soaSizePage, soaDataPage, rbMap, dmuFactory.getCrossBorderStationChoiceDMU()); - // set up the tracer object - trace = Util.getBooleanValueFromPropertyMap(rbMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbMap, "Trace.dtaz"); - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - seek = Util.getBooleanValueFromPropertyMap(rbMap, "Seek"); - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String coloniaDistanceFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.colonia.file"); - coloniaDistanceFile = directory + coloniaDistanceFile; - - // calculate logsums for each station (based on Super-Colonia population - // and distance to station) - float distanceParam = new Float(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.dc.colonia.distance.parameter")); - calculateStationLogsum(coloniaDistanceFile, distanceParam); - - // arrays of sampled station-mgra pairs - sampledDestinationMgra = new int[sampleRate + 1]; - sampledEntryMgra = new int[sampleRate + 1]; - sampledCorrectionFactors = new double[sampleRate + 1]; - frequencyChosen = new HashMap(); - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(rbMap); - - // this sets by thread, so do it outside of initialization - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - // set up a tour mode choice model for calculation of tour mode - // probabilities - tourModeChoiceModel = new CrossBorderTourModeChoiceModel(rbMap, myStructure, dmuFactory, - tazDistanceCalculator); - - tourModeChoiceLogsums = new double[sampleRate + 1]; - sampledSizeTerms = new double[myStructure.CROSSBORDER_PURPOSES.length][sampleRate + 1]; - sampledStationLogsums = new double[sampleRate + 1]; - sampledStations = new int[sampleRate + 1]; - - sampledOriginTazs = new int[sampleRate + 1]; - sampledDestinationTazs = new int[sampleRate + 1]; - - - } - - /** - * Calculate the station logsum. Station logsums are based on distance from - * supercolonia to station and population of supercolonia, as follows: - * - * stationLogsum_i = LN [ Sum( exp(distanceParam * distance) * population) ] - * - * supercolonia population and distances are stored in @param fileName. - * Fields in file include: - * - * Population Population of supercolonia Distance_MGRANumber where - * MGRANumber is the number of the MGRA corresponding to the entry station, - * with one field for each possible entry station. - * - * @param fileName - * Name of file containing supercolonia population and distance - * @param distanceParameter - * Parameter for distance. - */ - private void calculateStationLogsum(String fileName, float distanceParameter) - { - - logger.info("Calculating Station Logsum"); - - logger.info("Begin reading the data in file " + fileName); - TableDataSet coloniaTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - coloniaTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - stations = 0; - // iterate through columns in table, calculate number of station - // alternatives (entry stations) - String[] columnLabels = coloniaTable.getColumnLabels(); - for (int i = 0; i < columnLabels.length; ++i) - { - String label = columnLabels[i]; - if (label.contains("Distance_")) - { - ++stations; - } - } - - // iterate through stations and calculate logsum - stationLogsums = new double[stations]; - int colonias = coloniaTable.getRowCount(); - for (int i = 0; i < colonias; ++i) - { - - float population = coloniaTable.getValueAt(i + 1, "Population"); - - for (int j = 0; j < stations; ++j) - { - - float distance = coloniaTable.getValueAt(i + 1, "Distance_poe" + j); - if (population > 0) - stationLogsums[j] += Math.exp(distanceParameter * distance) * population; - } - } - - // take natural log - for (int i = 0; i < stations; ++i) - stationLogsums[i] = Math.log(stationLogsums[i]); - - logger.info("Finished Calculating Border Crossing Station Logsum"); - - } - - /** - * Calculate size terms - */ - public void calculateSizeTerms(CrossBorderDmuFactoryIf dmuFactory) - { - - logger.info("Calculating Cross Border Model MGRA Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int maxTaz = tazManager.getMaxTaz(); - int purposes = sizeTermUEC.getNumberOfAlternatives(); - - mgraSizeTerms = new double[purposes][maxMgra + 1]; - tazSizeTerms = new double[purposes][maxTaz + 1]; - IndexValues iv = new IndexValues(); - CrossBorderStationDestChoiceDMU aDmu = dmuFactory.getCrossBorderStationChoiceDMU(); - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = sizeTermUEC.solve(iv, aDmu, null); - - // store the size terms - for (int purpose = 0; purpose < purposes; ++purpose) - { - - mgraSizeTerms[purpose][mgra] = utilities[purpose]; - tazSizeTerms[purpose][taz] += utilities[purpose]; - } - - // log - if (tracer.isTraceOn() && tracer.isTraceZone(taz)) - { - - logger.info("Size Term calculations for mgra " + mgra); - sizeTermUEC.logResultsArray(logger, 0, mgra); - - } - } - - // now calculate probability of selecting each MGRA within each TAZ for - // SOA - mgraProbabilities = new double[purposes][maxTaz + 1][]; - int[] tazs = tazManager.getTazs(); - - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazs.length; ++taz) - { - int tazNumber = tazs[taz]; - int[] mgraArray = tazManager.getMgraArray(tazNumber); - - // initialize the vector of mgras for this purpose-taz - mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; - - // now calculate the cumulative probability distribution - double lastProb = 0.0; - for (int mgra = 0; mgra < mgraArray.length; ++mgra) - { - - int mgraNumber = mgraArray[mgra]; - if (tazSizeTerms[purpose][tazNumber] > 0.0) - mgraProbabilities[purpose][tazNumber][mgra] = lastProb - + mgraSizeTerms[purpose][mgraNumber] - / tazSizeTerms[purpose][tazNumber]; - lastProb = mgraProbabilities[purpose][tazNumber][mgra]; - } - if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) - logger.info("Error: purpose " + purpose + " taz " + tazNumber - + " cum prob adds up to " + lastProb); - } - - } - - // calculate logged size terms for mgra and taz vectors to be used in - // dmu - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) - if (tazSizeTerms[purpose][taz] > 0.0) - tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); - - for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) - if (mgraSizeTerms[purpose][mgra] > 0.0) - mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); - - } - logger.info("Finished Calculating Cross Border Model MGRA Size Terms"); - } - - /** - * Calculate taz probabilities. This method initializes and calculates the - * tazProbabilities array. - */ - public void calculateTazProbabilities(CrossBorderDmuFactoryIf dmuFactory) - { - - if (tazSizeTerms == null) - { - logger.error("Error: attemping to execute CrossBorderStationDestChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); - throw new RuntimeException(); - } - - logger.info("Calculating Cross Border Model TAZ-Station Probabilities Arrays"); - - // initialize taz probabilities array - int purposes = tazSizeTerms.length; - - // initialize the index for station population accessibility and taz - // size term - int alternatives = soaModel.getNumberOfAlternatives(); - soaStationLogsums = new double[alternatives + 1]; // by station-TAZ - // alternative - - // station logsums - soaSizeTerms = new double[purposes][alternatives + 1]; // by purpose, - // station-TAZ - // alternative - - // size terms - // for tazs - soaOriginTazs = new int[alternatives + 1]; - soaDestinationTazs = new int[alternatives + 1]; - - // iterate through the alternatives in the alternatives file and set the - // size term and station logsum for each alternative - UtilityExpressionCalculator soaModelUEC = soaModel.getUEC(); - TableDataSet altData = soaModelUEC.getAlternativeData(); - - int rowCount = altData.getRowCount(); - for (int row = 1; row <= rowCount; ++row) - { - - int entryMgra = (int) altData.getValueAt(row, "mgra_entry"); - int poe = (int) altData.getValueAt(row, "poe"); - int destinationTaz = (int) altData.getValueAt(row, "dest"); - - soaStationLogsums[row] = stationLogsums[poe]; - - for (int purpose = 0; purpose < purposes; ++purpose) - soaSizeTerms[purpose][row] = tazSizeTerms[purpose][destinationTaz]; - - // set the origin taz - soaOriginTazs[row] = mgraManager.getTaz(entryMgra); - - // set the destination taz - soaDestinationTazs[row] = destinationTaz; - - } - - dmu = dmuFactory.getCrossBorderStationChoiceDMU(); - - // set size terms for each taz - dmu.setSizeTerms(soaSizeTerms); - - // set population accessibility for each station - dmu.setStationPopulationAccessibilities(soaStationLogsums); - - // set the stations for each alternative - int poeField = altData.getColumnPosition("poe"); - int[] poeNumbers = altData.getColumnAsInt(poeField, 1); // return field - // as 1-based - - dmu.setPoeNumbers(poeNumbers); - - // set origin and destination tazs - dmu.setOriginTazs(soaOriginTazs); - dmu.setDestinationTazs(soaDestinationTazs); - - // initialize array to hold taz-station probabilities - tazStationProbabilities = new double[purposes][alternatives + 1]; - - // iterate through purposes, calculate probabilities for each and store - // in array - for (int purpose = 0; purpose < purposes; ++purpose) - { - - dmu.setPurpose(purpose); - - // Calculate utilities & probabilities - soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - // Store probabilities (by purpose) - tazStationProbabilities[purpose] = Arrays.copyOf(soaModel.getCumulativeProbabilities(), - soaModel.getCumulativeProbabilities().length); - } - logger.info("Finished Calculating Cross Border Model TAZ-Station Probabilities Arrays"); - } - - /** - * Choose a Station-MGRA alternative for sampling - * - * @param tour - * CrossBorderTour with purpose and Random - * @return An array of station-mgra pairs - */ - private void chooseStationMgraSample(CrossBorderTour tour) - { - - frequencyChosen.clear(); - - // choose sample, set station logsums and mgra size terms - int purpose = tour.getPurpose(); - for (int sample = 1; sample <= sampleRate; ++sample) - { - - // first find a TAZ and station - int alt = 0; - double[] tazCumProb = tazStationProbabilities[purpose]; - double altProb = 0; - double cumProb = 0; - double random = tour.getRandom(); - for (int i = 0; i < tazCumProb.length; ++i) - { - if (tazCumProb[i] > random) - { - alt = i; - if (i != 0) - { - cumProb = tazCumProb[i - 1]; - altProb = tazCumProb[i] - tazCumProb[i - 1]; - } else - { - altProb = tazCumProb[i]; - } - break; - } - } - - // get the taz number of the alternative, and an array of mgras in - // that taz - int destinationTaz = (int) alternativeData.getValueAt(alt + 1, "dest"); - int poe = (int) alternativeData.getValueAt(alt + 1, "poe"); - int entryMgra = (int) alternativeData.getValueAt(alt + 1, "mgra_entry"); - sampledEntryMgra[sample] = entryMgra; - int[] mgraArray = tazManager.getMgraArray(destinationTaz); - - // set the origin taz - sampledOriginTazs[sample] = (int) alternativeData.getValueAt(alt + 1, "poe_taz"); - - // set the destination taz - sampledDestinationTazs[sample] = destinationTaz; - - // now find an MGRA in the taz corresponding to the random number - // drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives - // probability - int mgraNumber = 0; - double[] mgraCumProb = mgraProbabilities[purpose][destinationTaz]; - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > random && mgraCumProb[i] > 0) - { - mgraNumber = mgraArray[i]; - sampledDestinationMgra[sample] = mgraNumber; - - // for now, store the probability in the correction factors - // array - sampledCorrectionFactors[sample] = mgraCumProb[i] * altProb; - - break; - } - } - - // store frequency chosen - key = new KeyClass(); - key.mgra = mgraNumber; - key.station = entryMgra; - if (!frequencyChosen.containsKey(key)) - { - frequencyChosen.put(key, 1); - } else - { - int freq = frequencyChosen.get(key); - frequencyChosen.put(key, freq + 1); - } - - // set station logsums - sampledStationLogsums[sample] = stationLogsums[poe]; - - // set the size terms for the sample - sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; - - // set the sampled station number - sampledStations[sample] = poe; - - } - // calculate correction factors - for (int sample = 1; sample <= sampleRate; ++sample) - { - key = new KeyClass(); - key.mgra = sampledDestinationMgra[sample]; - key.station = sampledEntryMgra[sample]; - int freq = frequencyChosen.get(key); - sampledCorrectionFactors[sample] = (float) Math.log((double) freq - / sampledCorrectionFactors[sample]); - - } - - } - - /** - * Use the tour mode choice model to calculate the logsum for each sampled - * station-mgra pair and store in the array. - * - * @param tour - * The tour attributes used are tour purpose, depart and arrive - * periods, and sentri availability. - */ - private void calculateLogsumsForSample(CrossBorderTour tour) - { - - for (int sample = 1; sample <= sampleRate; ++sample) - { - - if (sampledEntryMgra[sample] > 0) - { - - int originMgra = sampledEntryMgra[sample]; - int destinationMgra = sampledDestinationMgra[sample]; - - tour.setOriginMGRA(originMgra); - tour.setOriginTAZ(sampledOriginTazs[sample]); - tour.setDestinationMGRA(destinationMgra); - tour.setDestinationTAZ(mgraManager.getTaz(destinationMgra)); - tour.setPoe(sampledStations[sample]); - - double logsum = tourModeChoiceModel.getLogsum(tour, logger, "Sample logsum " - + sample, "tour " + tour.getID()); - tourModeChoiceLogsums[sample] = logsum; - } else tourModeChoiceLogsums[sample] = 0; - - } - - } - - /** - * Choose a station and internal destination MGRA for the tour. - * - * @param tour - * A cross border tour with a tour purpose and departure\arrival - * time and SENTRI availability members. - */ - public void chooseStationAndDestination(CrossBorderTour tour) - { - - chooseStationMgraSample(tour); - calculateLogsumsForSample(tour); - - double random = tour.getRandom(); - dmu.setPurpose(tour.getPurpose()); - - // set size terms for each sampled station-mgra pair corresponding to - // mgra - dmu.setSizeTerms(sampledSizeTerms); - - // set population accessibility for each station-mgra pair corresponding - // to station - dmu.setStationPopulationAccessibilities(sampledStationLogsums); - - // set the sampled stations - dmu.setPoeNumbers(sampledStations); - - // set the correction factors - dmu.setCorrectionFactors(sampledCorrectionFactors); - - // set the tour mode choice logsums - dmu.setTourModeLogsums(tourModeChoiceLogsums); - - // set the origin and destination tazs - dmu.setOriginTazs(sampledOriginTazs); - dmu.setDestinationTazs(sampledDestinationTazs); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Choosing station-destination alternative from sample"); - tour.logTourObject(logger, 1000); - - // log the sample - logSample(); - destModel.choiceModelUtilityTraceLoggerHeading("Station-destination model", "tour " - + tour.getID()); - } - - destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - destModel.logUECResults(logger, "Station-destination model"); - } - int alt = destModel.getChoiceResult(random); - - int entryMgra = sampledEntryMgra[alt]; - int primaryDestination = sampledDestinationMgra[alt]; - int poe = sampledStations[alt]; - int taz = sampledOriginTazs[alt]; - - tour.setOriginMGRA(entryMgra); - tour.setOriginTAZ(taz); - tour.setDestinationMGRA(primaryDestination); - tour.setDestinationTAZ(mgraManager.getTaz(primaryDestination)); - tour.setPoe(poe); - - } - - public CrossBorderTourModeChoiceModel getTourModeChoiceModel() - { - return tourModeChoiceModel; - } - - public void logSample() - { - - logger.info("Sampled station-destination alternatives"); - - logger.info("\nAlt POE EntryMgra DestMgra"); - for (int i = 1; i <= sampleRate; ++i) - { - logger.info(i + " " + sampledStations[i] + " " + sampledEntryMgra[i] + " " - + sampledDestinationMgra[i]); - } - logger.info(""); - } - - /** - * @return the mgraSizeTerms - */ - public double[][] getMgraSizeTerms() - { - return mgraSizeTerms; - } - - /** - * @return the tazSizeTerms - */ - public double[][] getTazSizeTerms() - { - return tazSizeTerms; - } - - /** - * @return the mgraProbabilities - */ - public double[][][] getMgraProbabilities() - { - return mgraProbabilities; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java deleted file mode 100644 index 6382a30..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStop.java +++ /dev/null @@ -1,187 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import org.apache.log4j.Logger; - -public class CrossBorderStop - implements Serializable -{ - - private int id; - private int mode; - private int period; - private boolean inbound; - private int mgra; - private int TAZ; - private byte purpose; - - CrossBorderTour parentTour; - - public CrossBorderStop(CrossBorderTour parentTour, int id, boolean inbound) - { - this.parentTour = parentTour; - this.id = id; - this.inbound = inbound; - } - - /** - * @return the mgra - */ - public int getMgra() - { - return mgra; - } - - /** - * @param mgra - * the mgra to set - */ - public void setMgra(int mgra) - { - this.mgra = mgra; - } - - public int getTAZ() - { - return TAZ; - } - - public void setTAZ(int tAZ) - { - TAZ = tAZ; - } - - public void setMode(int mode) - { - this.mode = mode; - } - - public void setPeriod(int period) - { - this.period = period; - } - - /** - * @return the id - */ - public int getId() - { - return id; - } - - /** - * @param id - * the id to set - */ - public void setId(int id) - { - this.id = id; - } - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the parentTour - */ - public CrossBorderTour getParentTour() - { - return parentTour; - } - - /** - * @param parentTour - * the parentTour to set - */ - public void setParentTour(CrossBorderTour parentTour) - { - this.parentTour = parentTour; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(byte stopPurposeIndex) - { - this.purpose = stopPurposeIndex; - } - - public byte getPurpose() - { - return purpose; - } - - public int getMode() - { - return mode; - } - - public int getStopPeriod() - { - return period; - } - - public CrossBorderTour getTour() - { - return parentTour; - } - - public int getStopId() - { - return id; - } - - public void logStopObject(Logger logger, int totalChars) - { - - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - String purposeString = CrossBorderModelStructure.CROSSBORDER_PURPOSES[purpose]; - logHelper(logger, "stopId: ", id, totalChars); - logHelper(logger, "mgra: ", mgra, totalChars); - logHelper(logger, "mode: ", mode, totalChars); - logHelper(logger, "purpose: ", purposeString, totalChars); - logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); - logHelper(logger, inbound ? "outbound departPeriod: " : "inbound arrivePeriod: ", period, - totalChars); - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public static void logHelper(Logger logger, String label, int value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } - - public static void logHelper(Logger logger, String label, String value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java deleted file mode 100644 index b546780..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopFrequencyModel.java +++ /dev/null @@ -1,316 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the stop frequency model for cross border tours. It is - * currently based on a static probability distribution stored in an input file, - * and indexed into by tour purpose and duration. - * - * @author Freedman - * - */ -public class CrossBorderStopFrequencyModel -{ - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private double[][] cumProbability; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - private int[][] lowerBoundDurationHours; // by - // purpose, - // alternative: - // lower - // bound - // in - // hours - private int[][] upperBoundDurationHours; // by - // purpose, - // alternative: - // upper - // bound - // in - // hours - private int[][] outboundStops; // by - // purpose, - // alternative: - // number - // of - // outbound - // stops - private int[][] inboundStops; // by - // purpose, - // alternative: - // number - // of - // inbound - // stops - CrossBorderModelStructure modelStructure; - - /** - * Constructor. - */ - public CrossBorderStopFrequencyModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.stop.frequency.file"); - stopFrequencyFile = directory + stopFrequencyFile; - - modelStructure = new CrossBorderModelStructure(); - - readStopFrequencyFile(stopFrequencyFile); - - } - - /** - * Read the stop frequency distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readStopFrequencyFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating stop frequency probability distribution"); - - int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 - - int[] alts = new int[purposes]; - - // take a pass through the data and see how many alternatives there are - // for each purpose - int rowCount = probabilityTable.getRowCount(); - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - ++alts[purpose]; - } - - // initialize all the arrays - cumProbability = new double[purposes][]; - lowerBoundDurationHours = new int[purposes][]; - upperBoundDurationHours = new int[purposes][]; - outboundStops = new int[purposes][]; - inboundStops = new int[purposes][]; - - for (int i = 0; i < purposes; ++i) - { - cumProbability[i] = new double[alts[i]]; - lowerBoundDurationHours[i] = new int[alts[i]]; - upperBoundDurationHours[i] = new int[alts[i]]; - outboundStops[i] = new int[alts[i]]; - inboundStops[i] = new int[alts[i]]; - } - - // fill up arrays - int lastPurpose = 0; - int lastLowerBound = 0; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - int lowerBound = (int) probabilityTable.getValueAt(row, "DurationLo"); - int upperBound = (int) probabilityTable.getValueAt(row, "DurationHi"); - int outStops = (int) probabilityTable.getValueAt(row, "Outbound"); - int inbStops = (int) probabilityTable.getValueAt(row, "Inbound"); - - // reset cumulative probability if new purpose or lower-bound - if (purpose != lastPurpose || lowerBound != lastLowerBound) - { - - // log cumulative probability just in case - /* - logger.info("Cumulative probability for purpose " + purpose + " lower bound " - + lowerBound + " is " + cumProb); - */ - cumProb = 0; - } - - if (purpose != lastPurpose) alt = 0; - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - lowerBoundDurationHours[purpose][alt] = lowerBound; - upperBoundDurationHours[purpose][alt] = upperBound; - outboundStops[purpose][alt] = outStops; - inboundStops[purpose][alt] = inbStops; - - ++alt; - - lastPurpose = purpose; - lastLowerBound = lowerBound; - } - - logger.info("End calculating stop frequency probability distribution"); -/* - for (int purp = 0; purp < purposes; ++purp) - { - for (int a = 0; a < cumProbability[purp].length; ++a) - { - logger.info("Purpose " + purp + " lower " + lowerBoundDurationHours[purp][a] - + " upper " + upperBoundDurationHours[purp][a] + " cumProb " - + cumProbability[purp][a]); - } - } -*/ - } - - /** - * Calculate tour time of day for the tour. - * - * @param tour - * A cross border tour (with purpose) - */ - public void calculateStopFrequency(CrossBorderTour tour) - { - - int purpose = tour.getPurpose(); - double random = tour.getRandom(); - - if (tour.getDebugChoiceModels()) - { - logger.info("Choosing stop frequency for purpose " - + modelStructure.CROSSBORDER_PURPOSES[purpose] + " using random number " - + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - if (!tourIsInRange(tour, lowerBoundDurationHours[purpose][i], - upperBoundDurationHours[purpose][i])) continue; - - if (random < cumProbability[purpose][i]) - { - int outStops = outboundStops[purpose][i]; - int inbStops = inboundStops[purpose][i]; - - if (outStops > 0) - { - CrossBorderStop[] stops = generateOutboundStops(tour, outStops); - tour.setOutboundStops(stops); - } - - if (inbStops > 0) - { - CrossBorderStop[] stops = generateInboundStops(tour, inbStops); - tour.setInboundStops(stops); - } - if (tour.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose " + outStops + " outbound stops and " + inbStops - + " inbound stops"); - logger.info(""); - } - break; - } - } - - } - - /** - * Check if the tour duration is in range - * - * @param tour - * @param lowerBound - * @param upperBound - * @return True if tour duration is greater than or equal to lower and - */ - private boolean tourIsInRange(CrossBorderTour tour, int lowerBound, int upperBound) - { - - float depart = (float) tour.getDepartTime(); - float arrive = (float) tour.getArriveTime(); - - float halfHours = arrive + 1 - depart; // at least 30 minutes - float tourDurationInHours = halfHours * (float) 0.5; - - if (tourDurationInHours >= lowerBound && tourDurationInHours <= upperBound) return true; - - return false; - } - - /** - * Generate an array of outbound stops, from tour origin to primary - * destination, in order. - * - * @param tour - * The parent tour. - * @param numberOfStops - * Number of stops from stop frequency model. - * @return The array of outbound stops. - */ - private CrossBorderStop[] generateOutboundStops(CrossBorderTour tour, int numberOfStops) - { - - CrossBorderStop[] stops = new CrossBorderStop[numberOfStops]; - - for (int i = 0; i < stops.length; ++i) - { - CrossBorderStop stop = new CrossBorderStop(tour, i, false); - stops[i] = stop; - stop.setInbound(false); - stop.setParentTour(tour); - } - - return stops; - } - - /** - * Generate an array of inbound stops, from primary dest back to tour - * origin, in order. - * - * @param tour - * Parent tour. - * @param numberOfStops - * Number of stops from stop frequency model. - * @return The array of inbound stops. - */ - private CrossBorderStop[] generateInboundStops(CrossBorderTour tour, int numberOfStops) - { - - CrossBorderStop[] stops = new CrossBorderStop[numberOfStops]; - - for (int i = 0; i < stops.length; ++i) - { - CrossBorderStop stop = new CrossBorderStop(tour, i, true); - stops[i] = stop; - stop.setInbound(true); - stop.setParentTour(tour); - - } - - return stops; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java deleted file mode 100644 index 1307d94..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceDMU.java +++ /dev/null @@ -1,406 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class CrossBorderStopLocationChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("crossBorderModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected int purpose; - protected int stopsOnHalfTour; - protected int stopNumber; - protected int inboundStop; - protected int tourDuration; - - protected double[][] sizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgra) - protected double[] correctionFactors; // by - // alternative - // (sampled - // mgra, - // for - // full - // model - // only) - - protected int[] sampleNumber; // by - // alternative - // (taz - // or - // sampled - // mgra) - - protected double[] osMcLogsumAlt; - protected double[] sdMcLogsumAlt; - - protected double[] tourOrigToStopDistanceAlt; - protected double[] stopToTourDestDistanceAlt; - - public CrossBorderStopLocationChoiceDMU(CrossBorderModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - /** - * @return the stopsOnHalfTour - */ - public int getStopsOnHalfTour() - { - return stopsOnHalfTour; - } - - /** - * @param stopsOnHalfTour - * the stopsOnHalfTour to set - */ - public void setStopsOnHalfTour(int stopsOnHalfTour) - { - this.stopsOnHalfTour = stopsOnHalfTour; - } - - /** - * @return the stopNumber - */ - public int getStopNumber() - { - return stopNumber; - } - - /** - * @param stopNumber - * the stopNumber to set - */ - public void setStopNumber(int stopNumber) - { - this.stopNumber = stopNumber; - } - - /** - * @return the inboundStop - */ - public int getInboundStop() - { - return inboundStop; - } - - /** - * @param inboundStop - * the inboundStop to set - */ - public void setInboundStop(int inboundStop) - { - this.inboundStop = inboundStop; - } - - /** - * @return the tourDuration - */ - public int getTourDuration() - { - return tourDuration; - } - - /** - * @param tourDuration - * the tourDuration to set - */ - public void setTourDuration(int tourDuration) - { - this.tourDuration = tourDuration; - } - - /** - * @return the sampleNumber - */ - public int getSampleNumber(int alt) - { - return sampleNumber[alt]; - } - - /** - * @param sampleNumber - * the sampleNumber to set - */ - public void setSampleNumber(int[] sampleNumber) - { - this.sampleNumber = sampleNumber; - } - - /** - * @return the osMcLogsumAlt - */ - public double getOsMcLogsumAlt(int alt) - { - return osMcLogsumAlt[alt]; - } - - /** - * @param osMcLogsumAlt - * the osMcLogsumAlt to set - */ - public void setOsMcLogsumAlt(double[] osMcLogsumAlt) - { - this.osMcLogsumAlt = osMcLogsumAlt; - } - - /** - * @return the sdMcLogsumAlt - */ - public double getSdMcLogsumAlt(int alt) - { - return sdMcLogsumAlt[alt]; - } - - /** - * @param sdMcLogsumAlt - * the sdMcLogsumAlt to set - */ - public void setSdMcLogsumAlt(double[] sdMcLogsumAlt) - { - this.sdMcLogsumAlt = sdMcLogsumAlt; - } - - /** - * @return the tourOrigToStopDistanceAlt - */ - public double getTourOrigToStopDistanceAlt(int alt) - { - return tourOrigToStopDistanceAlt[alt]; - } - - /** - * @param tourOrigToStopDistanceAlt - * the tourOrigToStopDistanceAlt to set - */ - public void setTourOrigToStopDistanceAlt(double[] tourOrigToStopDistanceAlt) - { - this.tourOrigToStopDistanceAlt = tourOrigToStopDistanceAlt; - } - - /** - * @return the stopToTourDestDistanceAlt - */ - public double getStopToTourDestDistanceAlt(int alt) - { - return stopToTourDestDistanceAlt[alt]; - } - - /** - * @param stopToTourDestDistanceAlt - * the stopToTourDestDistanceAlt to set - */ - public void setStopToTourDestDistanceAlt(double[] stopToTourDestDistanceAlt) - { - this.stopToTourDestDistanceAlt = stopToTourDestDistanceAlt; - } - - /** - * @return the sizeTerms. The size term is the size of the alternative north - * of the border. It is indexed by alternative, where alternative is - * either taz-station pair or mgra-station pair, depending on - * whether the DMU is being used for the SOA model or the actual - * model. - */ - public double getSizeTerm(int alt) - { - return sizeTerms[purpose][alt]; - } - - /** - * @param sizeTerms - * the sizeTerms to set. The size term is the size of the - * alternative north of the border. It is indexed by alternative, - * where alternative is either taz-station pair or mgra-station - * pair, depending on whether the DMU is being used for the SOA - * model or the actual model. - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - /** - * @return the correctionFactors - */ - public double getCorrectionFactor(int alt) - { - return correctionFactors[alt]; - } - - /** - * @param correctionFactors - * the correctionFactors to set - */ - public void setCorrectionFactors(double[] correctionFactors) - { - this.correctionFactors = correctionFactors; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the purpose - */ - public int getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(int purpose) - { - this.purpose = purpose; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - methodIndexMap.put("getPurpose", 0); - methodIndexMap.put("getStopsOnHalfTour", 1); - methodIndexMap.put("getStopNumber", 2); - methodIndexMap.put("getInboundStop", 3); - methodIndexMap.put("getTourDuration", 4); - - methodIndexMap.put("getSizeTerm", 5); - methodIndexMap.put("getCorrectionFactor", 6); - methodIndexMap.put("getSampleNumber", 7); - methodIndexMap.put("getOsMcLogsumAlt", 8); - methodIndexMap.put("getSdMcLogsumAlt", 9); - methodIndexMap.put("getTourOrigToStopDistanceAlt", 10); - methodIndexMap.put("getStopToTourDestDistanceAlt", 11); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getPurpose(); - break; - case 1: - returnValue = getStopsOnHalfTour(); - break; - case 2: - returnValue = getStopNumber(); - break; - case 3: - returnValue = getInboundStop(); - break; - case 4: - returnValue = getTourDuration(); - break; - case 5: - returnValue = getSizeTerm(arrayIndex); - break; - case 6: - returnValue = getCorrectionFactor(arrayIndex); - break; - case 7: - returnValue = getSampleNumber(arrayIndex); - break; - case 8: - returnValue = getOsMcLogsumAlt(arrayIndex); - break; - case 9: - returnValue = getSdMcLogsumAlt(arrayIndex); - break; - case 10: - returnValue = getTourOrigToStopDistanceAlt(arrayIndex); - break; - case 11: - returnValue = getStopToTourDestDistanceAlt(arrayIndex); - break; - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java deleted file mode 100644 index 4394466..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopLocationChoiceModel.java +++ /dev/null @@ -1,536 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.ConcreteAlternative; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class CrossBorderStopLocationChoiceModel -{ - - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - // private McLogsumsCalculator logsumHelper; - private CrossBorderModelStructure modelStructure; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private CrossBorderStopLocationChoiceDMU dmu; - private CrossBorderTripModeChoiceModel tripModeChoiceModel; - double logsum = 0; - private ChoiceModelApplication soaModel; - private ChoiceModelApplication destModel; - private McLogsumsCalculator logsumHelper; - - - // the following arrays are calculated in the station-destination choice - // model and passed in the constructor. - private double[][] mgraSizeTerms; // by - // purpose, - // MGRA - private double[][][] mgraProbabilities; // by - // purpose, - // TAZ, - // MGRA - - private TableDataSet alternativeData; // the - // alternatives, - // with - // a - // "dest" - // - - // indicating - // the - // destination - // TAZ - // in - // San - // Diego - // County - - // following are used for each taz alternative - private double[] soaTourOrigToStopDistanceAlt; // by - // TAZ - private double[] soaStopToTourDestDistanceAlt; // by - // TAZ - private double[][] tazSizeTerms; // by - // purpose, - // TAZ - // - - // set - // by - // constructor - - // following are used for sampled mgras - private int sampleRate; - private double[][] sampledSizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgra) - private double[] correctionFactors; // by - // alternative - // (sampled - // mgra, - // for - // full - // model - // only) - private int[] sampledTazs; // by - // alternative - // (sampled - // taz) - private int[] sampledMgras; // by - // alternative(sampled - // mgra) - private double[] tourOrigToStopDistanceAlt; - private double[] stopToTourDestDistanceAlt; - private double[] osMcLogsumAlt; - private double[] sdMcLogsumAlt; - - HashMap frequencyChosen; - - private CrossBorderTrip trip; - - private int originMgra; // the - // origin - // MGRA - // of - // the - // stop - // (originMgra - // -> - // stopMgra - // -> - // destinationMgra) - private int destinationMgra; // the - // destination - // MGRA - // of - // the - // stop - // (originMgra - // -> - // stopMgra - // -> - // destinationMgra) - private int originTAZ; - private int destinationTAZ; - private AutoTazSkimsCalculator tazDistanceCalculator; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public CrossBorderStopLocationChoiceModel(HashMap propertyMap, - CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - modelStructure = myModelStructure; - - this.tazDistanceCalculator = tazDistanceCalculator; - setupStopLocationChoiceModel(propertyMap, dmuFactory); - - frequencyChosen = new HashMap(); - - trip = new CrossBorderTrip(); - - } - - /** - * Read the UEC file and set up the stop destination choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupStopLocationChoiceModel(HashMap rbMap, - CrossBorderDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up cross border stop location choice model.")); - - dmu = dmuFactory.getCrossBorderStopLocationChoiceDMU(); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String crossBorderStopLocationSoaFileName = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.soa.uec.file"); - crossBorderStopLocationSoaFileName = uecFileDirectory + crossBorderStopLocationSoaFileName; - - int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.soa.data.page")); - int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.soa.model.page")); - - String crossBorderStopLocationFileName = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.uec.file"); - crossBorderStopLocationFileName = uecFileDirectory + crossBorderStopLocationFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.data.page")); - int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.slc.model.page")); - - // create a ChoiceModelApplication object for the SOA model. - soaModel = new ChoiceModelApplication(crossBorderStopLocationSoaFileName, soaModelPage, - soaDataPage, rbMap, (VariableTable) dmu); - - // create a ChoiceModelApplication object for the full model. - destModel = new ChoiceModelApplication(crossBorderStopLocationFileName, modelPage, - dataPage, rbMap, (VariableTable) dmu); - sampleRate = destModel.getAlternativeNames().length; - - // get the alternative data - UtilityExpressionCalculator uec = soaModel.getUEC(); - alternativeData = uec.getAlternativeData(); - int purposes = modelStructure.CROSSBORDER_PURPOSES.length; - - sampledSizeTerms = new double[purposes][sampleRate + 1]; // by purpose, - // alternative - // (taz or - // sampled - // mgra) - correctionFactors = new double[sampleRate + 1]; // by alternative - // (sampled mgra, for - // full model only) - sampledTazs = new int[sampleRate + 1]; // by alternative (sampled taz) - sampledMgras = new int[sampleRate + 1]; // by alternative (sampled mgra) - tourOrigToStopDistanceAlt = new double[sampleRate + 1]; - stopToTourDestDistanceAlt = new double[sampleRate + 1]; - osMcLogsumAlt = new double[sampleRate + 1]; - sdMcLogsumAlt = new double[sampleRate + 1]; - - tripModeChoiceModel = new CrossBorderTripModeChoiceModel(rbMap, modelStructure, - dmuFactory, tazDistanceCalculator); - logsumHelper = tripModeChoiceModel.getMcLogsumsCalculator(); - - - } - - /** - * Create a sample for the tour and stop. - * - * @param tour - * @param stop - */ - private void createSample(CrossBorderTour tour, CrossBorderStop stop) - { - - int purpose = tour.getPurpose(); - int period = modelStructure.AM; - - dmu.setPurpose(purpose); - boolean inbound = stop.isInbound(); - if (inbound) - { - dmu.setInboundStop(1); - dmu.setStopsOnHalfTour(tour.getNumberInboundStops()); - - // destination for inbound stops is always tour origin - destinationMgra = tour.getOriginMGRA(); - destinationTAZ = mgraManager.getTaz(destinationMgra); - - // origin for inbound stops is tour destination if first stop, or - // last chosen stop location - if (stop.getId() == 0) - { - originMgra = tour.getDestinationMGRA(); - originTAZ = mgraManager.getTaz(originMgra); - } else - { - CrossBorderStop[] stops = tour.getInboundStops(); - originMgra = stops[stop.getId() - 1].getMgra(); - originTAZ = mgraManager.getTaz(originMgra); - } - - } else - { - dmu.setInboundStop(0); - dmu.setStopsOnHalfTour(tour.getNumberOutboundStops()); - - // destination for outbound stops is always tour destination - destinationMgra = tour.getDestinationMGRA(); - destinationTAZ = mgraManager.getTaz(destinationMgra); - - // origin for outbound stops is tour origin if first stop, or last - // chosen stop location - if (stop.getId() == 0) - { - originMgra = tour.getOriginMGRA(); - originTAZ = mgraManager.getTaz(originMgra); - } else - { - CrossBorderStop[] stops = tour.getOutboundStops(); - originMgra = stops[stop.getId() - 1].getMgra(); - originTAZ = mgraManager.getTaz(originMgra); - } - } - dmu.setStopNumber(stop.getId() + 1); - dmu.setDmuIndexValues(originTAZ, originTAZ, originTAZ, 0, false); - - // distances - soaTourOrigToStopDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceFromTaz( - originTAZ, period); - soaStopToTourDestDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceToTaz( - destinationTAZ, period); - dmu.setTourOrigToStopDistanceAlt(soaTourOrigToStopDistanceAlt); - dmu.setStopToTourDestDistanceAlt(soaStopToTourDestDistanceAlt); - - dmu.setSizeTerms(tazSizeTerms); - - // solve for each sample - frequencyChosen.clear(); - for (int sample = 1; sample <= sampleRate; ++sample) - { - - // solve the UEC - soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - // choose a TAZ - double random = tour.getRandom(); - ConcreteAlternative[] alts = soaModel.getAlternatives(); - double cumProb = 0; - double altProb = 0; - int sampledTaz = -1; - for (int i = 0; i < alts.length; ++i) - { - cumProb += alts[i].getProbability(); - if (random < cumProb) - { - sampledTaz = (int) alternativeData.getValueAt(i + 1, "dest"); - altProb = alts[i].getProbability(); - break; - } - } - - // set the sampled taz in the array - sampledTazs[sample] = sampledTaz; - - // now find an MGRA in the taz corresponding to the random number - // drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives - // probability - int[] mgraArray = tazManager.getMgraArray(sampledTaz); - int mgraNumber = 0; - double[] mgraCumProb = mgraProbabilities[purpose][sampledTaz]; - - if (mgraCumProb == null) - { - logger.error("Error: mgraCumProb array is null for purpose " + purpose - + " sampledTaz " + sampledTaz + " hhID " + tour.getID()); - throw new RuntimeException(); - } - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > random && mgraCumProb[i] > 0) - { - mgraNumber = mgraArray[i]; - sampledMgras[sample] = mgraNumber; - - // for now, store the probability in the correction factors - // array - correctionFactors[sample] = mgraCumProb[i] * altProb; - - break; - } - } - - // store frequency chosen - if (!frequencyChosen.containsKey(mgraNumber)) - { - frequencyChosen.put(mgraNumber, 1); - } else - { - int freq = frequencyChosen.get(mgraNumber); - frequencyChosen.put(mgraNumber, freq + 1); - } - - // set the size terms for the sample - sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; - - // set the distances for the sample - tourOrigToStopDistanceAlt[sample] = soaTourOrigToStopDistanceAlt[sampledTaz]; - stopToTourDestDistanceAlt[sample] = soaStopToTourDestDistanceAlt[sampledTaz]; - - } - // calculate correction factors - for (int sample = 1; sample <= sampleRate; ++sample) - { - int mgra = sampledMgras[sample]; - int freq = frequencyChosen.get(mgra); - correctionFactors[sample] = (float) Math.log((double) freq / correctionFactors[sample]); - - } - - } - - /** - * Choose a stop location from the sample. - * - * @param tour - * The cross border tour. - * @param stop - * The cross border stop. - */ - public void chooseStopLocation(CrossBorderTour tour, CrossBorderStop stop) - { - - // create a sample of mgras and set all of the dmu properties - createSample(tour, stop); - dmu.setCorrectionFactors(correctionFactors); - dmu.setSizeTerms(sampledSizeTerms); - dmu.setTourOrigToStopDistanceAlt(stopToTourDestDistanceAlt); - dmu.setStopToTourDestDistanceAlt(stopToTourDestDistanceAlt); - dmu.setSampleNumber(sampledMgras); - - // calculate trip mode choice logsums to and from stop - for (int i = 1; i <= sampleRate; ++i) - { - - // to stop (originMgra -> stopMgra ) - trip.initializeFromStop(tour, stop, true); - trip.setOriginMgra(originMgra); - trip.setOriginTAZ(originTAZ); - trip.setDestinationMgra(sampledMgras[i]); - trip.setDestinationTAZ(mgraManager.getTaz(sampledMgras[i])); - double logsum = tripModeChoiceModel.computeUtilities(tour, trip); - osMcLogsumAlt[i] = logsum; - - // from stop (stopMgra -> destinationMgra) - trip.initializeFromStop(tour, stop, true); - trip.setOriginMgra(sampledMgras[i]); - trip.setOriginTAZ(mgraManager.getTaz(sampledMgras[i])); - trip.setDestinationMgra(destinationMgra); - trip.setDestinationTAZ(destinationTAZ); - logsum = tripModeChoiceModel.computeUtilities(tour, trip); - sdMcLogsumAlt[i] = logsum; - - } - dmu.setOsMcLogsumAlt(osMcLogsumAlt); - dmu.setSdMcLogsumAlt(sdMcLogsumAlt); - - // log headers to traceLogger - if (tour.getDebugChoiceModels()) - { - String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() - + " purpose " + modelStructure.CROSSBORDER_PURPOSES[stop.getPurpose()]; - destModel.choiceModelUtilityTraceLoggerHeading( - "Intermediate stop location choice model", decisionMakerLabel); - } - - destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - double random = tour.getRandom(); - int alt = destModel.getChoiceResult(random); - int destMgra = sampledMgras[alt]; - stop.setMgra(destMgra); - stop.setTAZ(mgraManager.getTaz(destMgra)); - - // write UEC calculation results and choice - if (tour.getDebugChoiceModels()) - { - String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() - + " purpose " + modelStructure.CROSSBORDER_PURPOSES[stop.getPurpose()]; - String loggingHeader = String.format("%s %s", - "Intermediate stop location choice model", decisionMakerLabel); - destModel.logUECResults(logger, loggingHeader); - logger.info("Chose alternative " + alt + " mgra " + destMgra + " with random number " - + random); - logger.info(""); - logger.info(""); - } - - } - - /** - * @return the mgraSizeTerms - */ - public double[][] getMgraSizeTerms() - { - return mgraSizeTerms; - } - - /** - * @return the mgraProbabilities - */ - public double[][][] getMgraProbabilities() - { - return mgraProbabilities; - } - - /** - * @return the tazSizeTerms - */ - public double[][] getTazSizeTerms() - { - return tazSizeTerms; - } - - /** - * Set mgra size terms: must call before choosing location. - * - * @param mgraSizeTerms - */ - public void setMgraSizeTerms(double[][] mgraSizeTerms) - { - - if (mgraSizeTerms == null) - { - logger.error("Error attempting to set MGRASizeTerms in CrossBorderStopLocationChoiceModel: MGRASizeTerms are null"); - throw new RuntimeException(); - } - this.mgraSizeTerms = mgraSizeTerms; - } - - /** - * Set taz size terms: must call before choosing location. - * - * @param tazSizeTerms - */ - public void setTazSizeTerms(double[][] tazSizeTerms) - { - if (tazSizeTerms == null) - { - logger.error("Error attempting to set TazSizeTerms in CrossBorderStopLocationChoiceModel: TazSizeTerms are null"); - throw new RuntimeException(); - } - this.tazSizeTerms = tazSizeTerms; - } - - /** - * Set the mgra probabilities. Must call before choosing location. - * - * @param mgraProbabilities - */ - public void setMgraProbabilities(double[][][] mgraProbabilities) - { - if (mgraProbabilities == null) - { - logger.error("Error attempting to set mgraProbabilities in CrossBorderStopLocationChoiceModel: mgraProbabilities are null"); - throw new RuntimeException(); - } - this.mgraProbabilities = mgraProbabilities; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java deleted file mode 100644 index a2cdfb5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopPurposeModel.java +++ /dev/null @@ -1,233 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the stop purpose choice model for cross border tours. It is - * currently based on a static probability distribution stored in an input file, - * and indexed into by purpose, tour leg direction (inbound or outbound), the - * stop number, and whether there is just one or multiple stops on the tour leg. - * - * @author Freedman - * - */ -public class CrossBorderStopPurposeModel -{ - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private double[][] cumProbability; // by - // alternative, - // stop - // purpose: - // cumulative - // probability - // distribution - CrossBorderModelStructure modelStructure; - - HashMap arrayElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - // tour - // purpose, - // tour - // leg - // direction, - // stop - // number, - // and - // stop - // complexity. - - /** - * Constructor. - */ - public CrossBorderStopPurposeModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.stop.purpose.file"); - stopFrequencyFile = directory + stopFrequencyFile; - - modelStructure = new CrossBorderModelStructure(); - - arrayElementMap = new HashMap(); - readStopPurposeFile(stopFrequencyFile); - - } - - /** - * Read the stop frequency distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readStopPurposeFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating stop purpose probability distribution"); - - // take a pass through the data and see how many alternatives there are - // for each purpose - int rowCount = probabilityTable.getRowCount(); - int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 - - cumProbability = new double[rowCount][purposes]; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "TourPurp"); - - int inbound = (int) probabilityTable.getValueAt(row, "Inbound"); - int stopNumber = (int) probabilityTable.getValueAt(row, "StopNum"); - int multiple = (int) probabilityTable.getValueAt(row, "Multiple"); - - // store cumulative probabilities - float cumProb = 0; - for (int p = 0; p < purposes; ++p) - { - String label = "StopPurp" + p; - cumProb += probabilityTable.getValueAt(row, label); - cumProbability[row - 1][p] += cumProb; - } - - if (Math.abs(cumProb - 1.0) > 0.00001) - logger.info("Cumulative probability for tour purpose " + purpose + " inbound " - + inbound + " stopNumber " + stopNumber + " multiple " + multiple + " is " - + cumProb); - - int key = getKey(purpose, inbound, stopNumber, multiple); - arrayElementMap.put(key, row - 1); - - } - - logger.info("End calculating stop purpose probability distribution"); - - } - - /** - * Get the key for the arrayElementMap. - * - * @param tourPurp - * Tour purpose - * @param isInbound - * 1 if the stop is on the inbound direction, else 0. - * @param stopNumber - * The number of the stop. - * @param multipleStopsOnLeg - * 1 if multiple stops on leg, else 0. - * @return arrayElementMap key. - */ - private int getKey(int tourPurp, int isInbound, int stopNumber, int multipleStopsOnLeg) - { - - return tourPurp * 1000 + isInbound * 100 + stopNumber * 10 + multipleStopsOnLeg; - } - - /** - * Calculate purposes all stops on the tour - * - * @param tour - * A cross border tour (with tour purpose) - */ - public void calculateStopPurposes(CrossBorderTour tour) - { - - // outbound stops first - if (tour.getNumberOutboundStops() != 0) - { - - int tourPurp = tour.getPurpose(); - CrossBorderStop[] stops = tour.getOutboundStops(); - int multiple = 0; - if (stops.length > 1) multiple = 1; - - // iterate through stop list and calculate purpose for each - for (int i = 0; i < stops.length; ++i) - { - int key = getKey(tourPurp, 0, i + 1, multiple); - int element = arrayElementMap.get(key); - double[] cumProb = cumProbability[element]; - double rand = tour.getRandom(); - int purpose = chooseFromDistribution(rand, cumProb); - stops[i].setPurpose((byte) purpose); - } - } - // inbound stops last - if (tour.getNumberInboundStops() != 0) - { - - int tourPurp = tour.getPurpose(); - CrossBorderStop[] stops = tour.getInboundStops(); - int multiple = 0; - if (stops.length > 1) multiple = 1; - - // iterate through stop list and calculate purpose for each - for (int i = 0; i < stops.length; ++i) - { - int key = getKey(tourPurp, 1, i + 1, multiple); - int element = arrayElementMap.get(key); - double[] cumProb = cumProbability[element]; - double rand = tour.getRandom(); - int purpose = chooseFromDistribution(rand, cumProb); - stops[i].setPurpose((byte) purpose); - } - } - } - - /** - * Choose purpose from the cumulative probability distribution - * - * @param random - * Uniformly distributed random number - * @param cumProb - * Cumulative probability distribution - * @return Stop purpose (0 init). - */ - private int chooseFromDistribution(double random, double[] cumProb) - { - - int choice = -1; - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - choice = i; - break; - } - - } - return choice; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java deleted file mode 100644 index 5e3d4a1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderStopTimeOfDayChoiceModel.java +++ /dev/null @@ -1,365 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the TOD choice model for cross border tours. It is currently - * based on a static probability distribution stored in an input file, and - * indexed into by purpose. - * - * @author Freedman - * - */ -public class CrossBorderStopTimeOfDayChoiceModel -{ - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private double[][] outboundCumProbability; // by - // alternative: - // outbound - // cumulative - // probability - // distribution - private int[] outboundOffsets; // by - // alternative: - // offsets - // for - // outbound - // stop - // duration - // choice - - private double[][] inboundCumProbability; // by - // alternative: - // inbound - // cumulative - // probability - // distribution - private int[] inboundOffsets; // by - // alternative: - // offsets - // for - // inbound - // stop - // duration - // choice - private CrossBorderModelStructure modelStructure; - - private HashMap outboundElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - // tour duration and stop number. - - private HashMap inboundElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - - // tour duration and stop number. - - /** - * Constructor. - */ - public CrossBorderStopTimeOfDayChoiceModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String outboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.stop.outbound.duration.file"); - String inboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.stop.inbound.duration.file"); - - outboundDurationFile = directory + outboundDurationFile; - inboundDurationFile = directory + inboundDurationFile; - - modelStructure = new CrossBorderModelStructure(); - - outboundElementMap = new HashMap(); - readOutboundFile(outboundDurationFile); - - inboundElementMap = new HashMap(); - readInboundFile(inboundDurationFile); - } - - /** - * Read the outbound stop duration file and store the cumulative probability - * distribution as well as the offsets and set the key map to index into the - * probability array. - * - * @param fileName - */ - public void readOutboundFile(String fileName) - { - TableDataSet outboundTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - outboundTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - int columns = outboundTable.getColumnCount(); - int rows = outboundTable.getRowCount(); - outboundCumProbability = new double[rows][columns - 3]; - - // first three columns are index fields, rest are offsets - outboundOffsets = new int[columns - 3]; - for (int i = 4; i <= columns; ++i) - { - String offset = outboundTable.getColumnLabel(i); - outboundOffsets[i - 4] = new Integer(offset); - } - - // now fill in cumulative probability array - for (int row = 1; row <= rows; ++row) - { - - int lowerBound = (int) outboundTable.getValueAt(row, "RemainingLow"); - int upperBound = (int) outboundTable.getValueAt(row, "RemainingHigh"); - int stopNumber = (int) outboundTable.getValueAt(row, "Stop"); - - for (int duration = lowerBound; duration <= upperBound; ++duration) - { - int key = getKey(stopNumber, duration); - outboundElementMap.put(key, row - 1); - } - - // cumulative probability distribution - double cumProb = 0; - for (int col = 4; col <= columns; ++col) - { - cumProb += outboundTable.getValueAt(row, col); - outboundCumProbability[row - 1][col - 4] = cumProb; - } - - } - - } - - /** - * Read the inbound stop duration file and store the cumulative probability - * distribution as well as the offsets and set the key map to index into the - * probability array. - * - * @param fileName - */ - public void readInboundFile(String fileName) - { - TableDataSet inboundTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - inboundTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - int columns = inboundTable.getColumnCount(); - int rows = inboundTable.getRowCount(); - inboundCumProbability = new double[rows][columns - 3]; - - // first three columns are index fields, rest are offsets - inboundOffsets = new int[columns - 3]; - for (int i = 4; i <= columns; ++i) - { - String offset = inboundTable.getColumnLabel(i); - inboundOffsets[i - 4] = new Integer(offset); - } - - // now fill in cumulative probability array - for (int row = 1; row <= rows; ++row) - { - - int lowerBound = (int) inboundTable.getValueAt(row, "RemainingLow"); - int upperBound = (int) inboundTable.getValueAt(row, "RemainingHigh"); - int stopNumber = (int) inboundTable.getValueAt(row, "Stop"); - - for (int duration = lowerBound; duration <= upperBound; ++duration) - { - int key = getKey(stopNumber, duration); - inboundElementMap.put(key, row - 1); - } - // cumulative probability distribution - double cumProb = 0; - for (int col = 4; col <= columns; ++col) - { - cumProb += inboundTable.getValueAt(row, col); - inboundCumProbability[row - 1][col - 4] = cumProb; - } - - } - - } - - /** - * Get the key for the arrayElementMap. - * - * @param stopNumber - * stop number - * @param periodsRemaining - * Remaining time periods - * @return arrayElementMap key. - */ - private int getKey(int stopNumber, int periodsRemaining) - { - - return periodsRemaining * 10 + stopNumber; - } - - /** - * Choose the stop time of day period. - * - * @param tour - * @param stop - */ - public void chooseTOD(CrossBorderTour tour, CrossBorderStop stop) - { - - boolean inbound = stop.isInbound(); - int stopNumber = stop.getId() + 1; - int arrivalPeriod = tour.getArriveTime(); - - if (!inbound) - { - - // find the departure time - int departPeriod = 0; - if (stop.getId() == 0) departPeriod = tour.getDepartTime(); - else - { - CrossBorderStop[] stops = tour.getOutboundStops(); - departPeriod = stops[stop.getId() - 1].getStopPeriod(); - } - - int periodsRemaining = arrivalPeriod - departPeriod; - - int key = getKey(stopNumber, periodsRemaining); - int element = outboundElementMap.get(key); - double[] cumProb = outboundCumProbability[element]; - double random = tour.getRandom(); - - // iterate through the offset distribution, choose an offset, and - // set in the stop - if (tour.getDebugChoiceModels()) - { - logger.info("Stop TOD Choice Model for tour " + tour.getID() + " outbound stop " - + stop.getId() + " periods remaining " + periodsRemaining); - logger.info(" random number " + random); - } - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - int offset = outboundOffsets[i]; - int period = departPeriod + offset; - stop.setPeriod(period); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Chose alt " + i + " offset " + offset + " from depart period " - + departPeriod); - logger.info("Stop period is " + stop.getStopPeriod()); - - } - break; - - } - } - } else - { - // inbound stop - - // find the departure time - int departPeriod = 0; - - // first inbound stop - if (stop.getId() == 0) - { - - // there were outbound stops - if (tour.getOutboundStops() != null) - { - CrossBorderStop[] outboundStops = tour.getOutboundStops(); - departPeriod = outboundStops[outboundStops.length - 1].getStopPeriod(); - } else - { - // no outbound stops - departPeriod = tour.getDepartTime(); - } - } else - { - // not first inbound stop - CrossBorderStop[] stops = tour.getInboundStops(); - departPeriod = stops[stop.getId() - 1].getStopPeriod(); - } - - int periodsRemaining = arrivalPeriod - departPeriod; - - int key = getKey(stopNumber, periodsRemaining); - int element = inboundElementMap.get(key); - double[] cumProb = inboundCumProbability[element]; - double random = tour.getRandom(); - if (tour.getDebugChoiceModels()) - { - logger.info("Stop TOD Choice Model for tour " + tour.getID() + " inbound stop " - + stop.getId() + " periods remaining " + periodsRemaining); - logger.info("Random number " + random); - } - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - int offset = inboundOffsets[i]; - int arrivePeriod = tour.getArriveTime(); - int period = arrivePeriod + offset; - stop.setPeriod(period); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Chose alt " + i + " offset " + offset + " from arrive period " - + arrivePeriod); - logger.info("Stop period is " + stop.getStopPeriod()); - - } - break; - } - } - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java deleted file mode 100644 index 59d9c2b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTour.java +++ /dev/null @@ -1,367 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Household; -import com.pb.common.math.MersenneTwister; - -public class CrossBorderTour - implements Serializable -{ - - private MersenneTwister random; - private int ID; - - // following variables determined via simulation - private byte purpose; - private boolean sentriAvailable; - - private CrossBorderStop[] outboundStops; - private CrossBorderStop[] inboundStops; - - private CrossBorderTrip[] trips; - - private int departTime; - private int arriveTime; - - private boolean debugChoiceModels; - - // following variables chosen via choice models - private int poe; - private int originMGRA; - private int destinationMGRA; - private int originTAZ; - private int destinationTAZ; - private byte tourMode; - private float workTimeFactor; - private float nonWorkTimeFactor; - private float valueOfTime; - - private boolean avAvailable; - - /** - * Public constructor. - * - * @param seed - * A seed for the random number generator. - */ - public CrossBorderTour(long seed) - { - - random = new MersenneTwister(seed); - } - - /** - * @return the iD - */ - public int getID() - { - return ID; - } - - /** - * @param iD - * the iD to set - */ - public void setID(int iD) - { - ID = iD; - } - - /** - * @return the sentriAvailable - */ - public boolean isSentriAvailable() - { - return sentriAvailable; - } - - /** - * @return the poe - */ - public int getPoe() - { - return poe; - } - - /** - * @param poe - * the poe to set - */ - public void setPoe(int poe) - { - this.poe = poe; - } - - /** - * @param sentriAvailable - * the sentriAvailable to set - */ - public void setSentriAvailable(boolean sentriAvailable) - { - this.sentriAvailable = sentriAvailable; - } - - /** - * @return the purpose - */ - public byte getPurpose() - { - return purpose; - } - - /** - * @return the outboundStops - */ - public CrossBorderStop[] getOutboundStops() - { - return outboundStops; - } - - /** - * @param outboundStops - * the outboundStops to set - */ - public void setOutboundStops(CrossBorderStop[] outboundStops) - { - this.outboundStops = outboundStops; - } - - /** - * @return the inboundStops - */ - public CrossBorderStop[] getInboundStops() - { - return inboundStops; - } - - /** - * @param inboundStops - * the inboundStops to set - */ - public void setInboundStops(CrossBorderStop[] inboundStops) - { - this.inboundStops = inboundStops; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(byte purpose) - { - this.purpose = purpose; - } - - /** - * @return the departTime - */ - public int getDepartTime() - { - return departTime; - } - - /** - * @param departTime - * the departTime to set - */ - public void setDepartTime(int departTime) - { - this.departTime = departTime; - } - - public CrossBorderTrip[] getTrips() - { - return trips; - } - - public void setTrips(CrossBorderTrip[] trips) - { - this.trips = trips; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() - { - return originMGRA; - } - - /** - * @param originMGRA - * the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) - { - this.originMGRA = originMGRA; - } - - public int getOriginTAZ() - { - return originTAZ; - } - - public void setOriginTAZ(int originTAZ) - { - this.originTAZ = originTAZ; - } - - public int getDestinationTAZ() - { - return destinationTAZ; - } - - public void setDestinationTAZ(int destinationTAZ) - { - this.destinationTAZ = destinationTAZ; - } - - /** - * @return the tour mode - */ - public byte getTourMode() - { - return tourMode; - } - - /** - * @param mode - * the tour mode to set - */ - public void setTourMode(byte mode) - { - this.tourMode = mode; - } - - /** - * Get a random number from the parties random class. - * - * @return A random number. - */ - public double getRandom() - { - return random.nextDouble(); - } - - /** - * @return the debugChoiceModels - */ - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - /** - * @param debugChoiceModels - * the debugChoiceModels to set - */ - public void setDebugChoiceModels(boolean debugChoiceModels) - { - this.debugChoiceModels = debugChoiceModels; - } - - - /** - * Get the number of outbound stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberOutboundStops() - { - if (outboundStops == null) return 0; - else return outboundStops.length; - - } - - /** - * Get the number of return stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberInboundStops() - { - if (inboundStops == null) return 0; - else return inboundStops.length; - - } - - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() - { - return destinationMGRA; - } - - /** - * @param destinationMGRA - * the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) - { - this.destinationMGRA = destinationMGRA; - } - - public void setArriveTime(int arriveTime) - { - this.arriveTime = arriveTime; - } - - public int getArriveTime() - { - return arriveTime; - } - - public double getWorkTimeFactor() { - return workTimeFactor; - } - - public void setWorkTimeFactor(float workTimeFactor) { - this.workTimeFactor = workTimeFactor; - } - - public double getNonWorkTimeFactor() { - return nonWorkTimeFactor; - } - - public void setNonWorkTimeFactor(float nonWorkTimeFactor) { - this.nonWorkTimeFactor = nonWorkTimeFactor; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public boolean isAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(boolean avAvailable) { - this.avAvailable = avAvailable; - } - - public void logTourObject(Logger logger, int totalChars) - { - - Household.logHelper(logger, "tourId: ", ID, totalChars); - Household.logHelper(logger, "tourPurpose: ", purpose, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); - Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); - Household.logHelper(logger, "tourMode: ", tourMode, totalChars); - Household.logHelper(logger, "avAvailable:", (avAvailable ? 0 : 1), totalChars); - - String tempString = null; - - logger.info(tempString); - - logger.info(tempString); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java deleted file mode 100644 index 97bfdb7..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourManager.java +++ /dev/null @@ -1,358 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.TimeCoefficientDistributions; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -public class CrossBorderTourManager -{ - - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - - private CrossBorderTour[] tours; - private int totalTours; - - private double sentriShare; - - private double[] sentriPurposeDistribution; - private double[] nonSentriPurposeDistribution; - - CrossBorderModelStructure modelStructure; - SandagModelStructure sandagStructure; - private boolean seek; - private int traceId; - - private float avShare; - - TimeCoefficientDistributions timeDistributions; - - boolean distributedTimeCoefficients = false; - - /** - * Constructor. Reads properties file and opens/stores all probability - * distributions for sampling. Estimates number of airport travel parties - * and initializes parties[]. - * - * @param resourceFile - * Property file. - * - * Creates the array of cross-border tours. - */ - public CrossBorderTourManager(HashMap rbMap) - { - - modelStructure = new CrossBorderModelStructure(); - sandagStructure = new SandagModelStructure(); - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String nonSentriPurposeFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.purpose.nonsentri.file"); - String sentriPurposeFile = directory - + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.purpose.sentri.file"); - - // the share of cross-border tours that are sentri is an input - sentriShare = new Double(Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.sentriShare")); - - // Read the distributions - sentriPurposeDistribution = setPurposeDistribution(sentriPurposeFile, - sentriPurposeDistribution,2); - nonSentriPurposeDistribution = setPurposeDistribution(nonSentriPurposeFile, - nonSentriPurposeDistribution,2); - totalTours = new Integer(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.tours") - .replace(",", "")); - - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trace")); - - distributedTimeCoefficients = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "distributedTimeCoefficients")); - - if(distributedTimeCoefficients) { - timeDistributions = new TimeCoefficientDistributions(); - timeDistributions.createTimeDistributions(rbMap); - } - - avShare = Util.getFloatValueFromPropertyMap(rbMap, "crossBorder.avShare"); - - } - - /** - * Generate and attribute cross border tours - */ - public void generateCrossBorderTours() - { - - // calculate total number of cross border tours - tours = new CrossBorderTour[totalTours]; - - logger.info("Total cross border tours: " + totalTours); - - for (int i = 0; i < tours.length; ++i) - { - - long seed = i * 10 + 1001; - CrossBorderTour tour = new CrossBorderTour(seed); - - tours[i] = tour; - - tour.setID(i + 1); - - // determine if tour is sentri, and calculate tour purpose - if (tour.getRandom() < sentriShare) - { - tour.setSentriAvailable(true); - int purpose = choosePurpose(tour.getRandom(), sentriPurposeDistribution); - tour.setPurpose((byte) purpose); - } else - { - tour.setSentriAvailable(false); - int purpose = choosePurpose(tour.getRandom(), nonSentriPurposeDistribution); - tour.setPurpose((byte) purpose); - } - - //set time factors - double workTimeFactor = 1.0; - double nonWorkTimeFactor = 1.0; - - if(distributedTimeCoefficients){ - double rnum = tour.getRandom(); - workTimeFactor = timeDistributions.sampleFromWorkDistribution(rnum); - nonWorkTimeFactor = timeDistributions.sampleFromNonWorkDistribution(rnum); - - } - tour.setWorkTimeFactor((float)workTimeFactor); - tour.setNonWorkTimeFactor((float)nonWorkTimeFactor); - - if(tour.getRandom() < avShare) - tour.setAvAvailable(true); - - } - } - - /** - * Read file containing probabilities by purpose. Store cumulative - * distribution in purposeDistribution. - * - * @param fileName - * Name of file containing two columns, one row for each purpose. - * First column has purpose number, second column has - * probability. - */ - protected double[] setPurposeDistribution(String fileName, double[] purposeDistribution, int position) - { - //logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; - purposeDistribution = new double[purposes]; - - double total_prob = 0.0; - // calculate and store cumulative probability distribution - for (int purp = 0; purp < purposes; ++purp) - { - - double probability = probabilityTable.getValueAt(purp + 1, position); - - total_prob += probability; - purposeDistribution[purp] = total_prob; - - } - //logger.info("End storing cumulative probabilies from file " + fileName); - - return purposeDistribution; - } - - /** - * Choose a purpose. - * - * @param random - * A uniform random number. - * @return the purpose. - */ - protected int choosePurpose(double random, double[] purposeDistribution) - { - // iterate through the probability array and choose - for (int alt = 0; alt < purposeDistribution.length; ++alt) - { - if (purposeDistribution[alt] > random) return alt; - } - return -99; - } - - /** - * Create a text file and write all records to the file. - * - */ - public void writeOutputFile(HashMap rbMap) - { - - // Open file and print header - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String tourFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.tour.output.file"); - String tripFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "crossBorder.trip.output.file"); - - logger.info("Writing cross border tours to file " + tourFileName); - logger.info("Writing cross border trips to file " + tripFileName); - - PrintWriter tourWriter = null; - try - { - tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tourFileName + " for writing\n"); - throw new RuntimeException(); - } - String tourHeaderString = new String( - "id,purpose,sentri,poe,departTime,arriveTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tourMode,avAvailable,workTimeFactor,nonWorkTimeFactor,valueOfTime\n"); - tourWriter.print(tourHeaderString); - - PrintWriter tripWriter = null; - try - { - tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tripFileName + " for writing\n"); - throw new RuntimeException(); - } - String tripHeaderString = new String( - "tourID,tripID,originPurp,destPurp,originMGRA,destinationMGRA,originTAZ,destinationTAZ,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,avAvailable,boardingTap,alightingTap,set,workTimeFactor,nonWorkTimeFactor,valueOfTime,parkingCost\n"); - tripWriter.print(tripHeaderString); - - // Iterate through the array, printing records to the file - for (int i = 0; i < tours.length; ++i) - { - - CrossBorderTour tour = tours[i]; - - if (seek && tour.getID() != traceId) continue; - - CrossBorderTrip[] trips = tours[i].getTrips(); - - if (trips == null) continue; - - writeTour(tour, tourWriter); - - for (int j = 0; j < trips.length; ++j) - { - writeTrip(tour, trips[j], j + 1, tripWriter); - } - } - - tourWriter.close(); - tripWriter.close(); - - } - - /** - * Write the tour to the PrintWriter - * - * @param tour - * @param writer - */ - private void writeTour(CrossBorderTour tour, PrintWriter writer) - { - String record = new String(tour.getID() + "," + tour.getPurpose() + "," - + tour.isSentriAvailable() + "," + tour.getPoe() + "," + tour.getDepartTime() + "," - + tour.getArriveTime() + "," + tour.getOriginMGRA() + "," - + tour.getDestinationMGRA() + "," + tour.getOriginTAZ() + "," - + tour.getDestinationTAZ() + "," + tour.getTourMode() + "," - + (tour.isAvAvailable() ? 1 : 0) + "," - + String.format("%9.2f",tour.getWorkTimeFactor()) + "," - + String.format("%9.2f",tour.getNonWorkTimeFactor()) + "," - + String.format("%9.2f", tour.getValueOfTime()) +"\n"); - writer.print(record); - - } - - /** - * Write the trip to the PrintWriter - * - * @param tour - * @param trip - * @param tripNumber - * @param writer - */ - private void writeTrip(CrossBorderTour tour, CrossBorderTrip trip, int tripNumber, - PrintWriter writer) - { - - String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginPurpose() - + "," + trip.getDestinationPurpose() + "," + trip.getOriginMgra() + "," - + trip.getDestinationMgra() + "," + trip.getOriginTAZ() + "," - + trip.getDestinationTAZ() + "," + trip.isInbound() + "," - + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() - + "," + trip.getPeriod() + "," + trip.getTripMode() + "," - + (tour.isAvAvailable() ? 1 : 0) + "," - + trip.getBoardTap() + "," + trip.getAlightTap() + "," - + trip.getSet() + "," - + String.format("%9.2f",tour.getWorkTimeFactor()) + "," - + String.format("%9.2f",tour.getNonWorkTimeFactor()) + "," - + String.format("%9.2f", trip.getValueOfTime()) + "," - + String.format("%9.2f", trip.getParkingCost())+ "\n"); - writer.print(record); - } - /** - * @return the parties - */ - public CrossBorderTour[] getTours() - { - return tours; - } - - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Cross Border Model Tour Manager")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - CrossBorderTourManager apm = new CrossBorderTourManager(pMap); - apm.generateCrossBorderTours(); - apm.writeOutputFile(pMap); - - logger.info("Cross-Border Tour Manager successfully completed!"); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java deleted file mode 100644 index 9fbf960..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceDMU.java +++ /dev/null @@ -1,406 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class CrossBorderTourModeChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected double tourPurpose; - protected double tourModelsSentri; - protected double borderWaitStd; - protected double borderWaitPed; - protected double borderWaitSentri; - - protected double outboundTripMcLogsumDA; - protected double outboundTripMcLogsumSR2; - protected double outboundTripMcLogsumSR3; - protected double outboundTripMcLogsumWalk; - protected double inboundTripMcLogsumDA; - protected double inboundTripMcLogsumSR2; - protected double inboundTripMcLogsumSR3; - protected double inboundTripMcLogsumWalk; - - /** - * Constructor. - * - * @param modelStructure - */ - public CrossBorderTourModeChoiceDMU(CrossBorderModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * @return the tourPurpose - */ - public double getTourPurpose() - { - return tourPurpose; - } - - /** - * @param tourPurpose - * the tourPurpose to set - */ - public void setTourPurpose(double tourPurpose) - { - this.tourPurpose = tourPurpose; - } - - /** - * @return the tourModelsSentri - */ - public double getTourModelsSentri() - { - return tourModelsSentri; - } - - /** - * @param tourModelsSentri - * the tourModelsSentri to set - */ - public void setTourModelsSentri(double tourModelsSentri) - { - this.tourModelsSentri = tourModelsSentri; - } - - /** - * @return the borderWaitStd - */ - public double getBorderWaitStd() - { - return borderWaitStd; - } - - /** - * @param borderWaitStd - * the borderWaitStd to set - */ - public void setBorderWaitStd(double borderWaitStd) - { - this.borderWaitStd = borderWaitStd; - } - - /** - * @return the borderWaitPed - */ - public double getBorderWaitPed() - { - return borderWaitPed; - } - - /** - * @param borderWaitPed - * the borderWaitPed to set - */ - public void setBorderWaitPed(double borderWaitPed) - { - this.borderWaitPed = borderWaitPed; - } - - /** - * @return the borderWaitSentri - */ - public double getBorderWaitSentri() - { - return borderWaitSentri; - } - - /** - * @param borderWaitSentri - * the borderWaitSentri to set - */ - public void setBorderWaitSentri(double borderWaitSentri) - { - this.borderWaitSentri = borderWaitSentri; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the outboundTripMcLogsumDA - */ - public double getOutboundTripMcLogsumDA() - { - return outboundTripMcLogsumDA; - } - - /** - * @param outboundTripMcLogsumDA - * the outboundTripMcLogsumDA to set - */ - public void setOutboundTripMcLogsumDA(double outboundTripMcLogsumDA) - { - this.outboundTripMcLogsumDA = outboundTripMcLogsumDA; - } - - /** - * @return the outboundTripMcLogsumSR2 - */ - public double getOutboundTripMcLogsumSR2() - { - return outboundTripMcLogsumSR2; - } - - /** - * @param outboundTripMcLogsumSR2 - * the outboundTripMcLogsumSR2 to set - */ - public void setOutboundTripMcLogsumSR2(double outboundTripMcLogsumSR2) - { - this.outboundTripMcLogsumSR2 = outboundTripMcLogsumSR2; - } - - /** - * @return the outboundTripMcLogsumSR3 - */ - public double getOutboundTripMcLogsumSR3() - { - return outboundTripMcLogsumSR3; - } - - /** - * @param outboundTripMcLogsumSR3 - * the outboundTripMcLogsumSR3 to set - */ - public void setOutboundTripMcLogsumSR3(double outboundTripMcLogsumSR3) - { - this.outboundTripMcLogsumSR3 = outboundTripMcLogsumSR3; - } - - /** - * @return the outboundTripMcLogsumWalk - */ - public double getOutboundTripMcLogsumWalk() - { - return outboundTripMcLogsumWalk; - } - - /** - * @param outboundTripMcLogsumWalk - * the outboundTripMcLogsumWalk to set - */ - public void setOutboundTripMcLogsumWalk(double outboundTripMcLogsumWalk) - { - this.outboundTripMcLogsumWalk = outboundTripMcLogsumWalk; - } - - /** - * @return the inboundTripMcLogsumDA - */ - public double getInboundTripMcLogsumDA() - { - return inboundTripMcLogsumDA; - } - - /** - * @param inboundTripMcLogsumDA - * the inboundTripMcLogsumDA to set - */ - public void setInboundTripMcLogsumDA(double inboundTripMcLogsumDA) - { - this.inboundTripMcLogsumDA = inboundTripMcLogsumDA; - } - - /** - * @return the inboundTripMcLogsumSR2 - */ - public double getInboundTripMcLogsumSR2() - { - return inboundTripMcLogsumSR2; - } - - /** - * @param inboundTripMcLogsumSR2 - * the inboundTripMcLogsumSR2 to set - */ - public void setInboundTripMcLogsumSR2(double inboundTripMcLogsumSR2) - { - this.inboundTripMcLogsumSR2 = inboundTripMcLogsumSR2; - } - - /** - * @return the inboundTripMcLogsumSR3 - */ - public double getInboundTripMcLogsumSR3() - { - return inboundTripMcLogsumSR3; - } - - /** - * @param inboundTripMcLogsumSR3 - * the inboundTripMcLogsumSR3 to set - */ - public void setInboundTripMcLogsumSR3(double inboundTripMcLogsumSR3) - { - this.inboundTripMcLogsumSR3 = inboundTripMcLogsumSR3; - } - - /** - * @return the inboundTripMcLogsumWalk - */ - public double getInboundTripMcLogsumWalk() - { - return inboundTripMcLogsumWalk; - } - - /** - * @param inboundTripMcLogsumWalk - * the inboundTripMcLogsumWalk to set - */ - public void setInboundTripMcLogsumWalk(double inboundTripMcLogsumWalk) - { - this.inboundTripMcLogsumWalk = inboundTripMcLogsumWalk; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTourPurpose", 0); - methodIndexMap.put("getTourModelsSentri", 1); - methodIndexMap.put("getBorderWaitStd", 2); - methodIndexMap.put("getBorderWaitPed", 3); - methodIndexMap.put("getBorderWaitSentri", 4); - - methodIndexMap.put("getOutboundTripMcLogsumDA", 30); - methodIndexMap.put("getOutboundTripMcLogsumSR2", 31); - methodIndexMap.put("getOutboundTripMcLogsumSR3", 32); - methodIndexMap.put("getOutboundTripMcLogsumWalk", 33); - methodIndexMap.put("getInboundTripMcLogsumDA", 34); - methodIndexMap.put("getInboundTripMcLogsumSR2", 35); - methodIndexMap.put("getInboundTripMcLogsumSR3", 36); - methodIndexMap.put("getInboundTripMcLogsumWalk", 37); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTourPurpose(); - break; - case 1: - returnValue = getTourModelsSentri(); - break; - case 2: - returnValue = getBorderWaitStd(); - break; - case 3: - returnValue = getBorderWaitPed(); - break; - case 4: - returnValue = getBorderWaitSentri(); - break; - case 30: - returnValue = getOutboundTripMcLogsumDA(); - break; - case 31: - returnValue = getOutboundTripMcLogsumSR2(); - break; - case 32: - returnValue = getOutboundTripMcLogsumSR3(); - break; - case 33: - returnValue = getOutboundTripMcLogsumWalk(); - break; - case 34: - returnValue = getInboundTripMcLogsumDA(); - break; - case 35: - returnValue = getInboundTripMcLogsumSR2(); - break; - case 36: - returnValue = getInboundTripMcLogsumSR3(); - break; - case 37: - returnValue = getInboundTripMcLogsumWalk(); - break; - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java deleted file mode 100644 index aa15f6f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourModeChoiceModel.java +++ /dev/null @@ -1,590 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class CrossBorderTourModeChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - public static final boolean DEBUG_BEST_PATHS = false; - - private MgraDataManager mgraManager; - - /** - * A private class used as a key to store the wait times for a particular - * station. - * - * @author Freedman - * - */ - private class WaitTimeClass - { - int[] beginPeriod; // by time periods - int[] endPeriod; // by time periods - float[] StandardWait; // by time periods - float[] SENTRIWait; // by time periods - float[] PedestrianWait; // by time periods - } - - HashMap waitTimeMap; - - private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "crossBorder.tour.mc.uec.file"; - private static final String PROPERTIES_UEC_TOUR_DATA_SHEET = "crossBorder.tour.mc.data.page"; - private static final String PROPERTIES_UEC_MANDATORY_MODEL_SHEET = "crossBorder.tour.mc.mandatory.model.page"; - private static final String PROPERTIES_UEC_NONMANDATORY_MODEL_SHEET = "crossBorder.tour.mc.nonmandatory.model.page"; - private static final String PROPERTIES_POE_WAITTIMES = "crossBorder.poe.waittime.file"; - - private ChoiceModelApplication[] mcModel; // by - // segment - // - - // mandatory - // vs - // non-mandatory - // (each - // has - // different - // nesting - // coefficients) - private CrossBorderTripModeChoiceModel tripModeChoiceModel; - private CrossBorderTourModeChoiceDMU mcDmuObject; - private McLogsumsCalculator logsumHelper; - - private CrossBorderModelStructure modelStructure; - - private String tourCategory; - - private String[] modeAltNames; - - private boolean saveUtilsProbsFlag = false; - - double logsum = 0; - - // placeholders for calculation of logsums - private CrossBorderTour tour; - private CrossBorderTrip trip; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public CrossBorderTourModeChoiceModel(HashMap propertyMap, - CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, - AutoTazSkimsCalculator tazDistanceCalculator) - { - - mgraManager = MgraDataManager.getInstance(propertyMap); - modelStructure = myModelStructure; - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - mcDmuObject = dmuFactory.getCrossBorderTourModeChoiceDMU(); - setupModeChoiceModelApplicationArray(propertyMap); - - // Create a trip mode choice model object for calculation of logsums - tripModeChoiceModel = new CrossBorderTripModeChoiceModel(propertyMap, myModelStructure, - dmuFactory, tazDistanceCalculator); - - tour = new CrossBorderTour(0); - trip = new CrossBorderTrip(); - - String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); - String waitTimeFile = Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_POE_WAITTIMES); - waitTimeFile = directory + waitTimeFile; - - readWaitTimeFile(waitTimeFile); - } - - /** - * Read UECs and create model application objects. - * - * @param propertyMap - */ - private void setupModeChoiceModelApplicationArray(HashMap propertyMap) - { - - logger.info(String - .format("Setting up cross border tour (border crossing) mode choice model.")); - - // locate the mandatory tour mode choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - logger.info("Will read mcUECFile " + mcUecFile); - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_TOUR_DATA_SHEET)); - int mandatoryModelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_MANDATORY_MODEL_SHEET)); - int nonmandatoryModelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_NONMANDATORY_MODEL_SHEET)); - - // default is to not save the tour mode choice utils and probs for each - // tour - String saveUtilsProbsString = propertyMap - .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); - if (saveUtilsProbsString != null) - { - if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; - } - - mcModel = new ChoiceModelApplication[2]; - - mcModel[0] = new ChoiceModelApplication(mcUecFile, mandatoryModelPage, dataPage, - propertyMap, (VariableTable) mcDmuObject); - mcModel[1] = new ChoiceModelApplication(mcUecFile, nonmandatoryModelPage, dataPage, - propertyMap, (VariableTable) mcDmuObject); - - modeAltNames = mcModel[0].getAlternativeNames(); - - } - - /** - * Get the Logsum. - * - * @param tour - * @param modelLogger - * @param choiceModelDescription - * @param decisionMakerLabel - * @return - */ - public double getLogsum(CrossBorderTour tour, Logger modelLogger, - String choiceModelDescription, String decisionMakerLabel) - { - - // set all tour mode DMU attributes including calculation of trip mode - // choice logsums for inbound & outbound directions. - setDmuAttributes(tour); - - return getModeChoiceLogsum(tour, modelLogger, choiceModelDescription, decisionMakerLabel); - - } - - /** - * Set the tour mode choice attributes. - * - * @param tour - */ - public void setDmuAttributes(CrossBorderTour tour) - { - - codeWaitTime(tour); - setTripLogsums(tour); - } - - /** - * Code wait times in the mc dmu object. - * - * @param tour - * The tour with an origin MGRA and departure time period. - */ - public void codeWaitTime(CrossBorderTour tour) - { - - // get the wait time class from the waitTimeMap HashMap - int station = tour.getPoe(); - int period = tour.getDepartTime(); - WaitTimeClass wait = waitTimeMap.get(station); - int[] beginTime = wait.beginPeriod; - int[] endTime = wait.endPeriod; - - // iterate through time arrays, find corresponding row, and set wait - // times - for (int i = 0; i < beginTime.length; ++i) - { - if (period >= beginTime[i] && period <= endTime[i]) - { - mcDmuObject.borderWaitStd = wait.StandardWait[i]; - mcDmuObject.borderWaitSentri = wait.SENTRIWait[i]; - mcDmuObject.borderWaitPed = wait.PedestrianWait[i]; - break; - } - } - } - - /** - * Set trip mode choice logsums (outbound and inbound) for calculation of - * tour mode choice model. - * - * @param tour - * The tour with other attributes such as origin, destination, - * purpose coded. - */ - public void setTripLogsums(CrossBorderTour tour) - { - - // outbound - trip.initializeFromTour(tour, true); - - // DA logsum - tour.setTourMode(modelStructure.DRIVEALONE); - double logsumDAOut = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setOutboundTripMcLogsumDA(logsumDAOut); - - // S2 logsum - tour.setTourMode(modelStructure.SHARED2); - double logsumS2Out = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setOutboundTripMcLogsumSR2(logsumS2Out); - - // S2 logsum - tour.setTourMode(modelStructure.SHARED3); - double logsumS3Out = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setOutboundTripMcLogsumSR3(logsumS3Out); - - // walk logsum - tour.setTourMode(modelStructure.WALK); - double logsumWalkOut = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setOutboundTripMcLogsumWalk(logsumWalkOut); - - // inbound - trip.initializeFromTour(tour, false); - - // DA logsum - tour.setTourMode(modelStructure.DRIVEALONE); - double logsumDAIn = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setInboundTripMcLogsumDA(logsumDAIn); - - // S2 logsum - tour.setTourMode(modelStructure.SHARED2); - double logsumS2In = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setInboundTripMcLogsumSR2(logsumS2In); - - // S2 logsum - tour.setTourMode(modelStructure.SHARED3); - double logsumS3In = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setInboundTripMcLogsumSR3(logsumS3In); - - // walk logsum - tour.setTourMode(modelStructure.WALK); - double logsumWalkIn = tripModeChoiceModel.computeUtilities(tour, trip); - mcDmuObject.setInboundTripMcLogsumWalk(logsumWalkIn); - - } - - /** - * Get an index into the mcModel array for the tour purpose. - * - * @param tour - * @return The index. - */ - public int getModelIndex(CrossBorderTour tour) - { - int modelIndex = 1; - if (tour.getPurpose() == modelStructure.WORK || tour.getPurpose() == modelStructure.SCHOOL) - modelIndex = 0; - return modelIndex; - } - - /** - * Get the tour mode choice logsum. - * - * @param tour - * @param modelLogger - * @param choiceModelDescription - * @param decisionMakerLabel - * @return Tour mode choice logsum - */ - public double getModeChoiceLogsum(CrossBorderTour tour, Logger modelLogger, - String choiceModelDescription, String decisionMakerLabel) - { - setDmuAttributes(tour); - - int modelIndex = getModelIndex(tour); - - // log headers to traceLogger - if (tour.getDebugChoiceModels()) - { - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - } - - mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); - - double logsum = mcModel[modelIndex].getLogsum(); - - // write UEC calculation results to separate model specific log file - if (tour.getDebugChoiceModels()) - { - String loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); - modelLogger.info(""); - modelLogger.info(""); - } - - return logsum; - - } - - /** - * Use to choose tour mode and set result in tour object. Also set value of time in tour object. - * - * @param tour - * The crossborder tour - */ - public void chooseTourMode(CrossBorderTour tour) - { - - byte tourMode = (byte) getModeChoice(tour); - tour.setTourMode(tourMode); - - float valueOfTime = tripModeChoiceModel.getTourValueOfTime(tourMode); - tour.setValueOfTime(valueOfTime); - } - - /** - * Use to return the tour mode without setting in the tour object. - * - * @param tour - * The cross border tour whose mode to choose. - * @return An integer corresponding to the tour mode. - */ - public int getModeChoice(CrossBorderTour tour) - { - int modelIndex = getModelIndex(tour); - - Logger modelLogger = null; - modelLogger = logger; - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - String separator = ""; - - if (tour.getDebugChoiceModels()) - { - String purposeName = modelStructure.CROSSBORDER_PURPOSES[tour.getPurpose()]; - choiceModelDescription = String.format( - "%s Tour Mode Choice Model for: Purpose=%s, Origin=%d, Dest=%d", tourCategory, - purposeName, tour.getOriginMGRA(), tour.getDestinationMGRA()); - decisionMakerLabel = String.format(" tour ID =%d", tour.getID()); - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - tour.logTourObject(modelLogger, loggingHeader.length()); - } - - setDmuAttributes(tour); - - mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); - - double rn = tour.getRandom(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen; - if (mcModel[modelIndex].getAvailabilityCount() > 0) - { - - chosen = mcModel[modelIndex].getChoiceResult(rn); - - } else - { - - String purposeName = modelStructure.CROSSBORDER_PURPOSES[tour.getPurpose()]; - choiceModelDescription = String - .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", - tourCategory, purposeName, tour.getOriginMGRA(), - tour.getDestinationMGRA()); - decisionMakerLabel = String.format("TourId=%d", tour.getID()); - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - tour.logTourObject(modelLogger, loggingHeader.length()); - - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - modelLogger.info(""); - modelLogger.info(""); - - logger.error(String - .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", - tour.getID(), tourCategory)); - throw new RuntimeException(); - } - - // debug output - if (tour.getDebugChoiceModels()) - { - - double[] utilities = mcModel[modelIndex].getUtilities(); // 0s-indexing - double[] probabilities = mcModel[modelIndex].getProbabilities(); // 0s-indexing - boolean[] availabilities = mcModel[modelIndex].getAvailabilities(); // 1s-indexing - String[] altNames = mcModel[modelIndex].getAlternativeNames(); // 0s-indexing - - modelLogger.info("Tour Id: " + tour.getID()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("-------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < mcModel[modelIndex].getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %s", k + 1, altNames[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f", altString, rn)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to log file - mcModel[modelIndex].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - mcModel[modelIndex].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - mcModel[modelIndex].logLogitCalculations(choiceModelDescription, decisionMakerLabel); - - // write UEC calculation results to separate model specific log file - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - } - - if (saveUtilsProbsFlag) - { - - // get the utilities and probabilities arrays for the tour mode - // choice - // model for this tour and save them to the tour object - double[] dUtils = mcModel[modelIndex].getUtilities(); - double[] dProbs = mcModel[modelIndex].getProbabilities(); - - float[] utils = new float[dUtils.length]; - float[] probs = new float[dUtils.length]; - for (int k = 0; k < dUtils.length; k++) - { - utils[k] = (float) dUtils[k]; - probs[k] = (float) dProbs[k]; - } - - // tour.setTourModalUtilities(utils); - // tour.setTourModalProbabilities(probs); - - } - - return chosen; - - } - - /** - * Read wait time file and store wait times in waitTimeMap HashMap. - * - * @param fileName - * Name of file containing station, beginPeriod, endPeriod and - * wait time for standard, SENTRI, and pedestrians. - */ - protected void readWaitTimeFile(String fileName) - { - logger.info("Begin reading the data in file " + fileName); - TableDataSet waitTimeTable; - waitTimeMap = new HashMap(); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - waitTimeTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - int rowCount = waitTimeTable.getRowCount(); - - // iterate through file and fill up waitTimeMap - int lastStation = -999; - int index = 0; - WaitTimeClass wait = null; - for (int row = 1; row <= rowCount; ++row) - { - - int station = (int) waitTimeTable.getValueAt(row, "poe"); - if (station != lastStation) - { - wait = new WaitTimeClass(); - wait.beginPeriod = new int[modelStructure.TIME_PERIODS]; - wait.endPeriod = new int[modelStructure.TIME_PERIODS]; - wait.StandardWait = new float[modelStructure.TIME_PERIODS]; - wait.SENTRIWait = new float[modelStructure.TIME_PERIODS]; - wait.PedestrianWait = new float[modelStructure.TIME_PERIODS]; - index = 0; - lastStation = station; - } else - { - ++index; - } - - wait.beginPeriod[index] = (int) waitTimeTable.getValueAt(row, "StartPeriod"); - wait.endPeriod[index] = (int) waitTimeTable.getValueAt(row, "EndPeriod"); - wait.StandardWait[index] = waitTimeTable.getValueAt(row, "StandardWait"); - wait.SENTRIWait[index] = waitTimeTable.getValueAt(row, "SENTRIWait"); - wait.PedestrianWait[index] = waitTimeTable.getValueAt(row, "PedestrianWait"); - - waitTimeMap.put(station, wait); - - } - logger.info("End reading the data in file " + fileName); - - } - - public String[] getModeAltNames(int purposeIndex) - { - return modeAltNames; - } - - /** - * @return the tripModeChoiceModel - */ - public CrossBorderTripModeChoiceModel getTripModeChoiceModel() - { - return tripModeChoiceModel; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java deleted file mode 100644 index 31e4bb6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTourTimeOfDayChoiceModel.java +++ /dev/null @@ -1,188 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the TOD choice model for cross border tours. It is currently - * based on a static probability distribution stored in an input file, and - * indexed into by purpose. - * - * @author Freedman - * - */ -public class CrossBorderTourTimeOfDayChoiceModel -{ - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private double[][] cumProbability; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - private int[][] outboundPeriod; // by - // purpose, - // alternative: - // outbound - // period - private int[][] returnPeriod; // by - // purpose, - // alternative: - // return - // period - CrossBorderModelStructure modelStructure; - - /** - * Constructor. - */ - public CrossBorderTourTimeOfDayChoiceModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, - "crossBorder.tour.tod.file"); - stationDiurnalFile = directory + stationDiurnalFile; - - modelStructure = new CrossBorderModelStructure(); - - readTODFile(stationDiurnalFile); - - } - - /** - * Read the TOD distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readTODFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating tour TOD probability distribution"); - - int purposes = modelStructure.NUMBER_CROSSBORDER_PURPOSES; // start at 0 - int periods = modelStructure.TIME_PERIODS; // start at 1 - int periodCombinations = periods * (periods + 1) / 2; - - cumProbability = new double[purposes][periodCombinations]; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - outboundPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // outbound - // period - returnPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // return period - - // fill up arrays - int rowCount = probabilityTable.getRowCount(); - int lastPurpose = -99; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); - int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); - - // continue if return period before outbound period - if (retPer < outPer) continue; - - // reset if new purpose - if (purpose != lastPurpose) - { - - // log cumulative probability just in case - /* - if (lastPurpose != -99) - logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); - */ - cumProb = 0; - alt = 0; - } - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - outboundPeriod[purpose][alt] = outPer; - returnPeriod[purpose][alt] = retPer; - - //temporary - //logger.info("row="+row+" alt="+alt+" purpose="+purpose+" outPer="+outPer+" retPer="+retPer+" cumProb="+cumProb); - - ++alt; - - lastPurpose = purpose; - } - - logger.info("End calculating tour TOD probability distribution"); - - } - - /** - * Calculate tour time of day for the tour. - * - * @param tour - * A cross border tour (with purpose) - */ - public void calculateTourTOD(CrossBorderTour tour) - { - - int purpose = tour.getPurpose(); - double random = tour.getRandom(); - - if (tour.getDebugChoiceModels()) - { - logger.info("Choosing tour time of day for purpose " - + modelStructure.CROSSBORDER_PURPOSES[purpose] + " using random number " - + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - if (random < cumProbability[purpose][i]) - { - int depart = outboundPeriod[purpose][i]; - int arrive = returnPeriod[purpose][i]; - tour.setDepartTime(depart); - tour.setArriveTime(arrive); - break; - } - } - - if (tour.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " - + tour.getArriveTime()); - logger.info(""); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java deleted file mode 100644 index 02d205c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTrip.java +++ /dev/null @@ -1,502 +0,0 @@ -package org.sandag.abm.crossborder; - -public class CrossBorderTrip -{ - - private int originMgra; - private int destinationMgra; - private int originTAZ; - private int destinationTAZ; - private int tripMode; - private byte originPurpose; - private byte destinationPurpose; - private byte period; - private boolean inbound; - private boolean firstTrip; - private boolean lastTrip; - private boolean originIsTourDestination; - private boolean destinationIsTourDestination; - - private float parkingCost; - - private int boardTap; - private int alightTap; - private int set = -1; - float valueOfTime; - - /** - * Default constructor; nothing initialized. - */ - public CrossBorderTrip() - { - - } - - /** - * Create a cross border trip from a tour leg (no stops). - * - * @param tour - * The tour. - * @param outbound - * Outbound direction - */ - public CrossBorderTrip(CrossBorderTour tour, boolean outbound) - { - - initializeFromTour(tour, outbound); - - } - - /** - * Initilize from the tour. - * - * @param tour - * The tour. - * @param outbound - * Outbound direction. - */ - public void initializeFromTour(CrossBorderTour tour, boolean outbound) - { - // Note: mode is unknown - if (outbound) - { - this.originMgra = tour.getOriginMGRA(); - this.destinationMgra = tour.getDestinationMGRA(); - this.originTAZ = tour.getOriginTAZ(); - this.destinationTAZ = tour.getDestinationTAZ(); - this.originPurpose = -1; - this.destinationPurpose = tour.getPurpose(); - this.period = (byte) tour.getDepartTime(); - this.inbound = false; - this.firstTrip = true; - this.lastTrip = false; - this.originIsTourDestination = false; - this.destinationIsTourDestination = true; - } else - { - this.originMgra = tour.getDestinationMGRA(); - this.destinationMgra = tour.getOriginMGRA(); - this.originTAZ = tour.getDestinationTAZ(); - this.destinationTAZ = tour.getOriginTAZ(); - this.originPurpose = tour.getPurpose(); - this.destinationPurpose = -1; - this.period = (byte) tour.getArriveTime(); - this.inbound = true; - this.firstTrip = false; - this.lastTrip = true; - this.originIsTourDestination = true; - this.destinationIsTourDestination = false; - } - - } - - /** - * Create a cross border trip from a tour\stop. Note: trip mode is unknown. - * Stop period is only known for first, last stop on tour. - * - * @param tour - * The tour. - * @param stop - * The stop - */ - public CrossBorderTrip(CrossBorderTour tour, CrossBorderStop stop, boolean toStop) - { - - initializeFromStop(tour, stop, toStop); - } - - /** - * Initialize from stop attributes. A trip will be created to the stop if - * toStop is true, else a trip will be created from the stop. Use after all - * stop locations are known, or else reset the stop origin and destination - * mgras accordingly after using. - * - * @param tour - * @param stop - * @param toStop - */ - public void initializeFromStop(CrossBorderTour tour, CrossBorderStop stop, boolean toStop) - { - - this.inbound = stop.isInbound(); - this.destinationIsTourDestination = false; - this.originIsTourDestination = false; - - // if trip to stop, destination is stop mgra; else origin is stop mgra - if (toStop) - { - this.destinationMgra = stop.getMgra(); - this.destinationTAZ = stop.getTAZ(); - this.destinationPurpose = stop.getPurpose(); - } else - { - this.originMgra = stop.getMgra(); - this.originTAZ = stop.getTAZ(); - this.originPurpose = stop.getPurpose(); - } - CrossBorderStop[] stops; - - if (!inbound) stops = tour.getOutboundStops(); - else stops = tour.getInboundStops(); - - // if outbound, and trip is to stop - if (!inbound && toStop) - { - - // first trip on outbound journey, origin is tour origin - if (stop.getId() == 0) - { - this.originMgra = tour.getOriginMGRA(); - this.originTAZ = tour.getOriginTAZ(); - this.originPurpose = -1; - this.period = (byte) tour.getDepartTime(); - } else - { - // not first trip on outbound journey, origin is last stop - this.originMgra = stops[stop.getId() - 1].getMgra(); // last - // stop - // location - this.originTAZ = stops[stop.getId() - 1].getTAZ(); // last stop - // location - this.originPurpose = stops[stop.getId() - 1].getPurpose(); // last - // stop - // location - this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); - } - } else if (!inbound && !toStop) - { - // outbound and trip is from stop to either next stop or tour - // destination. - - // last trip on outbound journey, destination is tour destination - if (stop.getId() == (stops.length - 1)) - { - this.destinationMgra = tour.getDestinationMGRA(); - this.destinationTAZ = tour.getDestinationTAZ(); - this.destinationPurpose = tour.getPurpose(); - this.destinationIsTourDestination = true; - } else - { - // not last trip on outbound journey, destination is next stop - this.destinationMgra = stops[stop.getId() + 1].getMgra(); - this.destinationTAZ = stops[stop.getId() + 1].getTAZ(); - this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); - } - - // the period for the trip is the origin for the trip - if (stop.getId() == 0) this.period = (byte) tour.getDepartTime(); - else this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); - - } else if (inbound && toStop) - { - // inbound, trip is to stop from either tour destination or last - // stop. - - // first inbound trip; origin is tour destination - if (stop.getId() == 0) - { - this.originMgra = tour.getDestinationMGRA(); - this.originTAZ = tour.getDestinationTAZ(); - this.originPurpose = tour.getPurpose(); - this.originIsTourDestination = true; - } else - { - // not first inbound trip; origin is last stop - this.originMgra = stops[stop.getId() - 1].getMgra(); // last - // stop - // location - this.originTAZ = stops[stop.getId() - 1].getTAZ(); // last stop - // location - this.originPurpose = stops[stop.getId() - 1].getPurpose(); - } - - // the period for the trip is the destination for the trip - if (stop.getId() == stops.length - 1) this.period = (byte) tour.getArriveTime(); - else this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); - } else - { - // inbound, trip is from stop to either next stop or tour origin. - - // last trip, destination is back to tour origin - if (stop.getId() == (stops.length - 1)) - { - this.destinationMgra = tour.getOriginMGRA(); - this.destinationTAZ = tour.getOriginTAZ(); - this.destinationPurpose = -1; - this.period = (byte) tour.getArriveTime(); - } else - { - // not last trip, destination is next stop - this.destinationMgra = stops[stop.getId() + 1].getMgra(); - this.destinationTAZ = stops[stop.getId() + 1].getTAZ(); - this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); - this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); - } - } - - // code period for first trip on tour - if (toStop && !inbound && stop.getId() == 0) - { - this.firstTrip = true; - this.lastTrip = false; - this.period = (byte) tour.getDepartTime(); - } - // code period for last trip on tour - if (!toStop && inbound && stop.getId() == (stops.length - 1)) - { - this.firstTrip = false; - this.lastTrip = true; - this.period = (byte) tour.getArriveTime(); - } - - } - - /** - * @return the period - */ - public byte getPeriod() - { - return period; - } - - /** - * @param period - * the period to set - */ - public void setPeriod(byte period) - { - this.period = period; - } - - /** - * @return the origin purpose - */ - public byte getOriginPurpose() - { - return originPurpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setOriginPurpose(byte purpose) - { - this.originPurpose = purpose; - } - - /** - * @return the destination purpose - */ - public byte getDestinationPurpose() - { - return destinationPurpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setDestinationPurpose(byte purpose) - { - this.destinationPurpose = purpose; - } - - /** - * @return the originMgra - */ - public int getOriginMgra() - { - return originMgra; - } - - /** - * @param originMgra - * the originMgra to set - */ - public void setOriginMgra(int originMgra) - { - this.originMgra = originMgra; - } - - /** - * @return the destinationMgra - */ - public int getDestinationMgra() - { - return destinationMgra; - } - - /** - * @param destinationMgra - * the destinationMgra to set - */ - public void setDestinationMgra(int destinationMgra) - { - this.destinationMgra = destinationMgra; - } - - public int getOriginTAZ() - { - return originTAZ; - } - - public void setOriginTAZ(int originTAZ) - { - this.originTAZ = originTAZ; - } - - public int getDestinationTAZ() - { - return destinationTAZ; - } - - public void setDestinationTAZ(int destinationTAZ) - { - this.destinationTAZ = destinationTAZ; - } - - /** - * @return the tripMode - */ - public int getTripMode() - { - return tripMode; - } - - /** - * @param tripMode - * the tripMode to set - */ - public void setTripMode(int tripMode) - { - this.tripMode = tripMode; - } - - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the firstTrip - */ - public boolean isFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(boolean firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public boolean isLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(boolean lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the originIsTourDestination - */ - public boolean isOriginIsTourDestination() - { - return originIsTourDestination; - } - - /** - * @param originIsTourDestination - * the originIsTourDestination to set - */ - public void setOriginIsTourDestination(boolean originIsTourDestination) - { - this.originIsTourDestination = originIsTourDestination; - } - - /** - * @return the destinationIsTourDestination - */ - public boolean isDestinationIsTourDestination() - { - return destinationIsTourDestination; - } - - /** - * @param destinationIsTourDestination - * the destinationIsTourDestination to set - */ - public void setDestinationIsTourDestination(boolean destinationIsTourDestination) - { - this.destinationIsTourDestination = destinationIsTourDestination; - } - - public int getBoardTap() { - return boardTap; - } - - public void setBoardTap(int boardTap) { - this.boardTap = boardTap; - } - - public int getAlightTap() { - return alightTap; - } - - public void setAlightTap(int alightTap) { - this.alightTap = alightTap; - } - - public int getSet() { - return set; - } - - public void setSet(int set) { - this.set = set; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public float getParkingCost() { - return parkingCost; - } - - public void setParkingCost(float parkingCost) { - this.parkingCost = parkingCost; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java deleted file mode 100644 index da3854f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceDMU.java +++ /dev/null @@ -1,762 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.McLogsumsCalculator; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class CrossBorderTripModeChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(CrossBorderTripModeChoiceDMU.class); - - protected static final int WTW = McLogsumsCalculator.WTW; - protected static final int WTD = McLogsumsCalculator.WTD; - protected static final int DTW = McLogsumsCalculator.DTW; - protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - protected int tourDepartPeriod; - protected int tourArrivePeriod; - protected int tripPeriod; - protected int workTour; - protected int outboundStops; - protected int returnStops; - protected int firstTrip; - protected int lastTrip; - protected int tourModeIsDA; - protected int tourModeIsS2; - protected int tourModeIsS3; - protected int tourModeIsWalk; - protected int tourCrossingIsSentri; - protected float hourlyParkingCostTourDest; - protected float dailyParkingCostTourDest; - protected float monthlyParkingCostTourDest; - protected int tripOrigIsTourDest; - protected int tripDestIsTourDest; - protected float hourlyParkingCostTripOrig; - protected float hourlyParkingCostTripDest; - protected float workTimeFactor; - protected float nonWorkTimeFactor; - protected int avAvailable; - - protected double nmWalkTime; - protected double nmBikeTime; - protected HashMap methodIndexMap; - protected double ivtCoeff; - protected double costCoeff; - protected double walkTransitLogsum; - protected double pnrTransitLogsum; - protected double knrTransitLogsum; - - protected IndexValues dmuIndex; - protected int outboundHalfTourDirection; - - protected float waitTimeTaxi; - protected float waitTimeSingleTNC; - protected float waitTimeSharedTNC; - - public CrossBorderTripModeChoiceDMU(CrossBorderModelStructure modelStructure, Logger aLogger) - { - if (aLogger == null) - { - aLogger = Logger.getLogger("crossBorderModel"); - } - logger = aLogger; - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the tripPeriod - */ - public int getTripPeriod() - { - return tripPeriod; - } - - /** - * @param tripPeriod - * the tripPeriod to set - */ - public void setTripPeriod(int tripPeriod) - { - this.tripPeriod = tripPeriod; - } - - /** - * @return the workTour - */ - public int getWorkTour() - { - return workTour; - } - - /** - * @param workTour - * the workTour to set - */ - public void setWorkTour(int workTour) - { - this.workTour = workTour; - } - - /** - * @return the outboundStops - */ - public int getOutboundStops() - { - return outboundStops; - } - - /** - * @param outboundStops - * the outboundStops to set - */ - public void setOutboundStops(int outboundStops) - { - this.outboundStops = outboundStops; - } - - /** - * @return the returnStops - */ - public int getReturnStops() - { - return returnStops; - } - - /** - * @param returnStops - * the returnStops to set - */ - public void setReturnStops(int returnStops) - { - this.returnStops = returnStops; - } - - /** - * @return the firstTrip - */ - public int getFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(int firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public int getLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(int lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the tourModeIsDA - */ - public int getTourModeIsDA() - { - return tourModeIsDA; - } - - /** - * @param tourModeIsDA - * the tourModeIsDA to set - */ - public void setTourModeIsDA(int tourModeIsDA) - { - this.tourModeIsDA = tourModeIsDA; - } - - /** - * @return the tourModeIsS2 - */ - public int getTourModeIsS2() - { - return tourModeIsS2; - } - - /** - * @param tourModeIsS2 - * the tourModeIsS2 to set - */ - public void setTourModeIsS2(int tourModeIsS2) - { - this.tourModeIsS2 = tourModeIsS2; - } - - /** - * @return the tourModeIsS3 - */ - public int getTourModeIsS3() - { - return tourModeIsS3; - } - - /** - * @param tourModeIsS3 - * the tourModeIsS3 to set - */ - public void setTourModeIsS3(int tourModeIsS3) - { - this.tourModeIsS3 = tourModeIsS3; - } - - /** - * @return the tourModeIsWalk - */ - public int getTourModeIsWalk() - { - return tourModeIsWalk; - } - - /** - * @param tourModeIsWalk - * the tourModeIsWalk to set - */ - public void setTourModeIsWalk(int tourModeIsWalk) - { - this.tourModeIsWalk = tourModeIsWalk; - } - - /** - * @return the tourModeIsSentri - */ - public int getTourCrossingIsSentri() - { - return tourCrossingIsSentri; - } - - /** - * @param tourModeIsSentri - * the tourModeIsSentri to set - */ - public void setTourCrossingIsSentri(int tourCrossingIsSentri) - { - this.tourCrossingIsSentri = tourCrossingIsSentri; - } - - /** - * @return the hourlyParkingCostTourDest - */ - public float getHourlyParkingCostTourDest() - { - return hourlyParkingCostTourDest; - } - - /** - * @param hourlyParkingCostTourDest - * the hourlyParkingCostTourDest to set - */ - public void setHourlyParkingCostTourDest(float hourlyParkingCostTourDest) - { - this.hourlyParkingCostTourDest = hourlyParkingCostTourDest; - } - - /** - * @return the dailyParkingCostTourDest - */ - public float getDailyParkingCostTourDest() - { - return dailyParkingCostTourDest; - } - - /** - * @param dailyParkingCostTourDest - * the dailyParkingCostTourDest to set - */ - public void setDailyParkingCostTourDest(float dailyParkingCostTourDest) - { - this.dailyParkingCostTourDest = dailyParkingCostTourDest; - } - - /** - * @return the monthlyParkingCostTourDest - */ - public float getMonthlyParkingCostTourDest() - { - return monthlyParkingCostTourDest; - } - - /** - * @param monthlyParkingCostTourDest - * the monthlyParkingCostTourDest to set - */ - public void setMonthlyParkingCostTourDest(float monthlyParkingCostTourDest) - { - this.monthlyParkingCostTourDest = monthlyParkingCostTourDest; - } - - /** - * @return the tripOrigIsTourDest - */ - public int getTripOrigIsTourDest() - { - return tripOrigIsTourDest; - } - - /** - * @param tripOrigIsTourDest - * the tripOrigIsTourDest to set - */ - public void setTripOrigIsTourDest(int tripOrigIsTourDest) - { - this.tripOrigIsTourDest = tripOrigIsTourDest; - } - - /** - * @return the tripDestIsTourDest - */ - public int getTripDestIsTourDest() - { - return tripDestIsTourDest; - } - - /** - * @param tripDestIsTourDest - * the tripDestIsTourDest to set - */ - public void setTripDestIsTourDest(int tripDestIsTourDest) - { - this.tripDestIsTourDest = tripDestIsTourDest; - } - - /** - * @return the hourlyParkingCostTripOrig - */ - public float getHourlyParkingCostTripOrig() - { - return hourlyParkingCostTripOrig; - } - - /** - * @param hourlyParkingCostTripOrig - * the hourlyParkingCostTripOrig to set - */ - public void setHourlyParkingCostTripOrig(float hourlyParkingCostTripOrig) - { - this.hourlyParkingCostTripOrig = hourlyParkingCostTripOrig; - } - - /** - * @return the hourlyParkingCostTripDest - */ - public float getHourlyParkingCostTripDest() - { - return hourlyParkingCostTripDest; - } - - /** - * @param hourlyParkingCostTripDest - * the hourlyParkingCostTripDest to set - */ - public void setHourlyParkingCostTripDest(float hourlyParkingCostTripDest) - { - this.hourlyParkingCostTripDest = hourlyParkingCostTripDest; - } - - /** - * @return the outboundHalfTourDirection - */ - public int getOutboundHalfTourDirection() - { - return outboundHalfTourDirection; - } - - /** - * @param outboundHalfTourDirection - * the outboundHalfTourDirection to set - */ - public void setOutboundHalfTourDirection(int outboundHalfTourDirection) - { - this.outboundHalfTourDirection = outboundHalfTourDirection; - } - - /** - * @return the tourDepartPeriod - */ - public int getTourDepartPeriod() - { - return tourDepartPeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(int tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(int tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - /** - * @return the tourArrivePeriod - */ - public int getTourArrivePeriod() - { - return tourArrivePeriod; - } - - public double getNm_walkTime() - { - return nmWalkTime; - } - - public void setNonMotorizedWalkTime(double nmWalkTime) - { - this.nmWalkTime = nmWalkTime; - } - - public void setNonMotorizedBikeTime(double nmBikeTime) - { - this.nmBikeTime = nmBikeTime; - } - - public double getNm_bikeTime() - { - return nmBikeTime; - } - - - public float getWorkTimeFactor() { - return workTimeFactor; - } - - public void setWorkTimeFactor(float workTimeFactor) { - this.workTimeFactor = workTimeFactor; - } - - public float getNonWorkTimeFactor() { - return nonWorkTimeFactor; - } - - public void setNonWorkTimeFactor(float nonWorkTimeFactor) { - this.nonWorkTimeFactor = nonWorkTimeFactor; - } - - public double getIvtCoeff() { - return ivtCoeff; - } - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public double getCostCoeff() { - return costCoeff; - } - - public double getWalkTransitLogsum() { - return walkTransitLogsum; - } - - public void setWalkTransitLogsum(double walkTransitLogsum) { - this.walkTransitLogsum = walkTransitLogsum; - } - - public double getPnrTransitLogsum() { - return pnrTransitLogsum; - } - - public void setPnrTransitLogsum(double pnrTransitLogsum) { - this.pnrTransitLogsum = pnrTransitLogsum; - } - - public double getKnrTransitLogsum() { - return knrTransitLogsum; - } - - public void setKnrTransitLogsum(double knrTransitLogsum) { - this.knrTransitLogsum = knrTransitLogsum; - } - - - - - public int getAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(int avAvailable) { - this.avAvailable = avAvailable; - } - - public float getWaitTimeTaxi() { - return waitTimeTaxi; - } - - public void setWaitTimeTaxi(float waitTimeTaxi) { - this.waitTimeTaxi = waitTimeTaxi; - } - - public float getWaitTimeSingleTNC() { - return waitTimeSingleTNC; - } - - public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { - this.waitTimeSingleTNC = waitTimeSingleTNC; - } - - public float getWaitTimeSharedTNC() { - return waitTimeSharedTNC; - } - - public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { - this.waitTimeSharedTNC = waitTimeSharedTNC; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTourDepartPeriod", 0); - methodIndexMap.put("getTourArrivePeriod", 1); - methodIndexMap.put("getTripPeriod", 2); - - methodIndexMap.put("getWorkTour", 4); - methodIndexMap.put("getOutboundStops", 5); - methodIndexMap.put("getReturnStops", 6); - methodIndexMap.put("getFirstTrip", 7); - methodIndexMap.put("getLastTrip", 8); - methodIndexMap.put("getTourModeIsDA", 9); - methodIndexMap.put("getTourModeIsS2", 10); - methodIndexMap.put("getTourModeIsS3", 11); - methodIndexMap.put("getTourModeIsWalk", 12); - methodIndexMap.put("getTourCrossingIsSentri", 13); - methodIndexMap.put("getHourlyParkingCostTourDest", 14); - methodIndexMap.put("getDailyParkingCostTourDest", 15); - methodIndexMap.put("getMonthlyParkingCostTourDest", 16); - methodIndexMap.put("getTripOrigIsTourDest", 17); - methodIndexMap.put("getTripDestIsTourDest", 18); - methodIndexMap.put("getHourlyParkingCostTripOrig", 19); - methodIndexMap.put("getHourlyParkingCostTripDest", 20); - - methodIndexMap.put("getWorkTimeFactor", 50); - methodIndexMap.put("getNonWorkTimeFactor", 51); - - methodIndexMap.put("getIvtCoeff", 60); - methodIndexMap.put("getCostCoeff", 61); - - methodIndexMap.put("getWalkSetLogSum", 62); - methodIndexMap.put("getPnrSetLogSum", 63); - methodIndexMap.put("getKnrSetLogSum", 64); - - methodIndexMap.put("getWaitTimeTaxi", 70); - methodIndexMap.put("getWaitTimeSingleTNC", 71); - methodIndexMap.put("getWaitTimeSharedTNC", 72); - - methodIndexMap.put("getNm_walkTime", 90); - methodIndexMap.put("getNm_bikeTime", 91); - - methodIndexMap.put("getAvAvailable", 95); - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - case 0: - returnValue = getTourDepartPeriod(); - break; - case 1: - returnValue = getTourArrivePeriod(); - break; - case 2: - returnValue = getTripPeriod(); - break; - case 4: - returnValue = getWorkTour(); - break; - case 5: - returnValue = getOutboundStops(); - break; - case 6: - returnValue = getReturnStops(); - break; - case 7: - returnValue = getFirstTrip(); - break; - case 8: - returnValue = getLastTrip(); - break; - case 9: - returnValue = getTourModeIsDA(); - break; - case 10: - returnValue = getTourModeIsS2(); - break; - case 11: - returnValue = getTourModeIsS3(); - break; - case 12: - returnValue = getTourModeIsWalk(); - break; - case 13: - returnValue = getTourCrossingIsSentri(); - break; - case 14: - returnValue = getHourlyParkingCostTourDest(); - break; - case 15: - returnValue = getDailyParkingCostTourDest(); - break; - case 16: - returnValue = getMonthlyParkingCostTourDest(); - break; - case 17: - returnValue = getTripOrigIsTourDest(); - break; - case 18: - returnValue = getTripDestIsTourDest(); - break; - case 19: - returnValue = getHourlyParkingCostTripOrig(); - break; - case 20: - returnValue = getHourlyParkingCostTripDest(); - break; - case 50: - returnValue = getWorkTimeFactor(); - break; - case 51: - returnValue = getNonWorkTimeFactor(); - break; - case 60: - returnValue = getIvtCoeff(); - break; - case 61: - returnValue = getCostCoeff(); - break; - case 62: - returnValue = getWalkTransitLogsum(); - break; - case 63: - returnValue = getPnrTransitLogsum(); - break; - case 64: - returnValue = getKnrTransitLogsum(); - break; - case 70: return getWaitTimeTaxi(); - case 71: return getWaitTimeSingleTNC(); - case 72: return getWaitTimeSharedTNC(); - - case 90: - returnValue = getNm_walkTime(); - break; - case 91: - returnValue = getNm_bikeTime(); - break; - case 95: - returnValue = getAvAvailable(); - break; - - default: - logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java deleted file mode 100644 index 1d40c89..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripModeChoiceModel.java +++ /dev/null @@ -1,369 +0,0 @@ -package org.sandag.abm.crossborder; - -import java.util.HashMap; -import java.util.Random; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class CrossBorderTripModeChoiceModel -{ - - private transient Logger logger = Logger.getLogger("crossBorderModel"); - - private AutoAndNonMotorizedSkimsCalculator anm; - private McLogsumsCalculator logsumHelper; - private CrossBorderModelStructure modelStructure; - private SandagModelStructure sandagModelStructure; - private TazDataManager tazs; - private MgraDataManager mgraManager; - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - private CrossBorderTripModeChoiceDMU dmu; - private ChoiceModelApplication tripModeChoiceModel; - double logsum = 0; - - private TripModeChoiceDMU mcDmuObject; - private AutoTazSkimsCalculator tazDistanceCalculator; - - - private static final String PROPERTIES_UEC_DATA_SHEET = "crossBorder.trip.mc.data.page"; - private static final String PROPERTIES_UEC_MODEL_SHEET = "crossBorder.trip.mc.model.page"; - private static final String PROPERTIES_UEC_FILE = "crossBorder.trip.mc.uec.file"; - - private TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public CrossBorderTripModeChoiceModel(HashMap propertyMap, - CrossBorderModelStructure myModelStructure, CrossBorderDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - tazs = TazDataManager.getInstance(propertyMap); - mgraManager = MgraDataManager.getInstance(propertyMap); - - lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); - lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); - lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); - - modelStructure = myModelStructure; - sandagModelStructure = new SandagModelStructure(); - - this.tazDistanceCalculator = tazDistanceCalculator; - - setupTripModeChoiceModel(propertyMap, dmuFactory); - - } - - /** - * Read the UEC file and set up the trip mode choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupTripModeChoiceModel(HashMap propertyMap, - CrossBorderDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up cross border trip mode choice model.")); - - dmu = dmuFactory.getCrossBorderTripModeChoiceDMU(); - - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_DATA_SHEET)); - int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_MODEL_SHEET)); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); - tripModeUecFile = uecPath + tripModeUecFile; - - tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, - propertyMap, (VariableTable) dmu); - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - SandagModelStructure modelStructure = new SandagModelStructure(); - mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); - - tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); - tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); - - } - - /** - * Calculate utilities and return logsum for the tour and stop. - * - * @param tour - * @param trip - */ - public double computeUtilities(CrossBorderTour tour, CrossBorderTrip trip) - { - - setDmuAttributes(tour, trip); - - tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - tour.logTourObject(logger, 100); - tripModeChoiceModel.logUECResults(logger, "Cross border trip mode choice model"); - - } - - logsum = tripModeChoiceModel.getLogsum(); - - if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); - - return logsum; - - } - - /** - * Choose a mode and store in the trip object. - * - * @param tour - * CrossBorderTour - * @param trip - * CrossBorderTrip - * - */ - public void chooseMode(CrossBorderTour tour, CrossBorderTrip trip) - { - computeUtilities(tour, trip); - - double rand = tour.getRandom(); - int mode=0; - try{ - mode = tripModeChoiceModel.getChoiceResult(rand); - trip.setTripMode(mode); - float vot = getTripValueOfTime(mode); - trip.setValueOfTime(vot); - float parkingCost = getTripParkingCost(mode); - trip.setParkingCost(parkingCost); - - if(sandagModelStructure.getTripModeIsTransit(mode)){ - double[][] bestTapPairs = logsumHelper.getBestWtwTripTaps(); - //pick transit path from N-paths - double rn = tour.getRandom(); - int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); - int boardTap = (int) bestTapPairs[pathIndex][0]; - int alightTap = (int) bestTapPairs[pathIndex][1]; - int set = (int) bestTapPairs[pathIndex][2]; - trip.setBoardTap(boardTap); - trip.setAlightTap(alightTap); - trip.setSet(set); - } - }catch(Exception e){ - logger.info("rand="+rand); - tour.logTourObject(logger, 100); - logger.error(e.getMessage()); - } - - } - - /** - * Return parking cost from UEC if auto trip, else return 0. - * - * @param tripMode - * @return Parking cost if auto mode, else 0 - */ - public float getTripParkingCost(int tripMode) { - - float parkingCost=0; - - if(sandagModelStructure.getTripModeIsSovOrHov(tripMode)) { - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - int parkingCostIndex = uec.lookupVariableIndex("parkingCost"); - parkingCost = (float) uec.getValueForIndex(parkingCostIndex); - return parkingCost; - } - return parkingCost; - } - - /** - * This method looks up the value of time from the last call to the UEC and returns - * it based on the occupancy of the mode passed in as an argument. this method ensures - * that the value of time at a tour level is the same for all trips on the tour (even - * though the actual trip level VOT might vary based on the trip occupancy). - * - * @param tourMode - * @return The value of time - */ - public float getTourValueOfTime(int tourMode){ - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - - double vot = 0.0; - - if(tourMode== modelStructure.SHARED2){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = uec.getValueForIndex(votIndex); - }else if (tourMode== modelStructure.SHARED3){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = uec.getValueForIndex(votIndex); - } - return (float) (vot * 0.5); //take half the VOT (assumed for tours) - - - } - - /** - * This method looks up the value of time from the last call to the UEC and returns - * it based on the occupancy of the mode passed in as an argument. this method ensures - * that the value of time at a tour level is the same for all trips on the tour (even - * though the actual trip level VOT might vary based on the trip occupancy). - * - * @param tripMode - * @return The value of time - */ - public float getTripValueOfTime(int tripMode){ - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - - double vot = 0.0; - - if(modelStructure.getTripModeIsS2(tripMode)){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = uec.getValueForIndex(votIndex); - }else if (modelStructure.getTripModeIsS3(tripMode)){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = uec.getValueForIndex(votIndex); - } - return (float) vot; - - - } - - /** - * Set DMU attributes. - * - * @param tour - * @param trip - */ - public void setDmuAttributes(CrossBorderTour tour, CrossBorderTrip trip) - { - - int tourDestinationMgra = tour.getDestinationMGRA(); - int tripOriginMgra = trip.getOriginMgra(); - int tripDestinationMgra = trip.getDestinationMgra(); - - int tripOriginTaz = trip.getOriginTAZ(); - int tripDestinationTaz = trip.getDestinationTAZ(); - - dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, - tour.getDebugChoiceModels()); - - dmu.setTourDepartPeriod(tour.getDepartTime()); - dmu.setTourArrivePeriod(tour.getArriveTime()); - dmu.setTripPeriod(trip.getPeriod()); - - dmu.setWorkTimeFactor((float)tour.getWorkTimeFactor()); - dmu.setNonWorkTimeFactor((float)tour.getNonWorkTimeFactor()); - - // set trip mc dmu values for transit logsum (gets replaced below by uec values) - double c_ivt = -0.03; - double c_cost = - 0.0003; - - // Solve trip mode level utilities - mcDmuObject.setIvtCoeff(c_ivt); - mcDmuObject.setCostCoeff(c_cost); - double walkTransitLogsum = -999.0; - - logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); - dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); - - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, tripOriginMgra, tripDestinationMgra, trip.getPeriod(), tour.getDebugChoiceModels()); - walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - dmu.setWalkTransitLogsum(walkTransitLogsum); - - if (tour.getPurpose() == modelStructure.WORK) dmu.setWorkTour(1); - else dmu.setWorkTour(0); - - dmu.setOutboundStops(tour.getNumberInboundStops()); - dmu.setReturnStops(tour.getNumberInboundStops()); - - if (trip.isFirstTrip()) dmu.setFirstTrip(1); - else dmu.setFirstTrip(0); - - if (trip.isLastTrip()) dmu.setLastTrip(1); - else dmu.setLastTrip(0); - - if (tour.getTourMode() == modelStructure.DRIVEALONE) dmu.setTourModeIsDA(1); - else dmu.setTourModeIsDA(0); - - if (tour.getTourMode() == modelStructure.SHARED2) dmu.setTourModeIsS2(1); - else dmu.setTourModeIsS2(0); - - if (tour.getTourMode() == modelStructure.SHARED3) dmu.setTourModeIsS3(1); - else dmu.setTourModeIsS3(0); - - if (tour.getTourMode() == modelStructure.WALK) dmu.setTourModeIsWalk(1); - else dmu.setTourModeIsWalk(0); - - if (tour.isSentriAvailable()) dmu.setTourCrossingIsSentri(1); - else dmu.setTourCrossingIsSentri(0); - - if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); - else dmu.setTripOrigIsTourDest(0); - - if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); - else dmu.setTripDestIsTourDest(0); - - dmu.setHourlyParkingCostTourDest((float) lsWgtAvgCostH[tourDestinationMgra]); - dmu.setDailyParkingCostTourDest((float) lsWgtAvgCostD[tourDestinationMgra]); - dmu.setMonthlyParkingCostTourDest((float) lsWgtAvgCostM[tourDestinationMgra]); - dmu.setHourlyParkingCostTripOrig((float) lsWgtAvgCostH[tripOriginMgra]); - dmu.setHourlyParkingCostTripDest((float) lsWgtAvgCostH[tripDestinationMgra]); - - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(trip.getOriginMgra()); - - double rnum = tour.getRandom(); - float waitTimeSingleTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - float waitTimeSharedTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - float waitTimeTaxi = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - dmu.setWaitTimeSingleTNC(waitTimeSingleTNC); - dmu.setWaitTimeSharedTNC(waitTimeSharedTNC); - dmu.setWaitTimeTaxi(waitTimeTaxi); - - - } - - public McLogsumsCalculator getMcLogsumsCalculator(){ - return logsumHelper; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java deleted file mode 100644 index 3aca0c5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/crossborder/CrossBorderTripTables.java +++ /dev/null @@ -1,704 +0,0 @@ -package org.sandag.abm.crossborder; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class CrossBorderTripTables -{ - - private static Logger logger = Logger.getLogger("tripTables"); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TableDataSet tripData; - - // Some parameters - private int[] modeIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // total - // modes, - // returns - // 0=auto - // modes, - // 1=non-motor, - // 2=transit, - // 3= - // other - private int[] matrixIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // modes, - // returns - // the - // element - // of - // the - // matrix - // array - // to - // store - // value - - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private HashMap rbMap; - - // matrices are indexed by modes, vot bins, submodes - private Matrix[][][] matrix; - - private ResourceBundle rb; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - - private float averageOcc3Plus = 3.5f; - private float sampleRate = 1; - private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; - private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; - private float valueOfTimeThresholdLow = 0; - private float valueOfTimeThresholdMed = 0; - //value of time bins by mode group - int[] votBins = {3,1,1,1}; - - public int numSkimSets; - - - /** - * @return the sampleRate - */ - public float getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(float sampleRate) - { - this.sampleRate = sampleRate; - } - - private MatrixDataServerRmi ms; - - public CrossBorderTripTables(HashMap rbMap) - { - - this.rbMap = rbMap; - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTripModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - } else if (modelStructure.getTripModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - } else if (modelStructure.getTripModeIsWalkTransit(i) - || modelStructure.getTripModeIsPnrTransit(i) - || modelStructure.getTripModeIsKnrTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - } - } - //value of time thresholds - valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); - valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); - - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][][]; - for (int i = 0; i < numberOfModes; ++i) - { - - String modeName; - - matrix[i] = new Matrix[votBins[i]][]; - - for(int j = 0; j< votBins[i];++j){ - if (i == 0) - { - matrix[i][j] = new Matrix[autoModes]; - for (int k = 0; k < autoModes; ++k) - { - modeName = modelStructure.getModeName(k + 1); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 1) - { - matrix[i][j] = new Matrix[nmotModes]; - for (int k = 0; k < nmotModes; ++k) - { - modeName = modelStructure.getModeName(k + 1 + autoModes); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 2) - { - matrix[i][j] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l1) - votBin = getValueOfTimeBin(valueOfTime); - - if (mode == 0) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); - } else if (mode == 1) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } else if (mode == 2) - { - - if (boardTap == 0 || alightTap == 0) continue; - - //store transit trips in matrices - mat = (matrixIndex[tripMode]*numSkimSets)+set; - float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); - matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); - - // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) - if (modelStructure.getTourModeIsDriveTransit(tripMode)) - { - - boolean inbound = tripData.getBooleanValueAt(i, "inbound"); - - // add the tNCVehicle trip portion to the trip table - if (!inbound) - { // from origin to lot (boarding tap) - int PNRTAZ = tapManager.getTazForTap(boardTap); - value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); - matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); - - } else - { // from lot (alighting tap) to destination - int PNRTAZ = tapManager.getTazForTap(alightTap); - value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); - matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); - } - - } - } else - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } - - //logger.info("End creating trip tables for period " + timePeriod); - } - } - - /** - * Return the value of time bin 0 through 2 based on the thresholds provided in the property map - * @param valueOfTime - * @return value of time bin 0 through 2 - */ - public int getValueOfTimeBin(float valueOfTime){ - - if(valueOfTime1) - end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; - else - end[i][j] = "_" + per + ".omx"; - } - } - for (int i = 0; i < 4; ++i){ - for(int j = 0; j < votBins[i];++j){ - try - { - //Delete the file if it exists - File f = new File(fileName[i]+end[i][j]); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); - f.delete(); - } - - if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); - else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); - } catch (Exception e) - { - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName[i] +end[i][j] + ", for mode index = " + i, e); - throw new RuntimeException(); - } - } - } - - } - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - /** - * Start matrix server - * - * @param serverAddress - * @param serverPort - * @param mt - * @return - */ - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @param args - */ - public static void main(String[] args) - { - - HashMap pMap; - String propertiesFile = null; - - logger.info(String.format( - "SANDAG Cross-Border Model Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - CrossBorderTripTables tripTables = new CrossBorderTripTables(pMap); - float sampleRate = 1.0f; - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - - logger.info("Crossborder Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - tripTables.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - tripTables.ms = matrixServer; - } else - { - tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - tripTables.ms.testRemote("CrossBorderTripTables"); - - // mdm = MatrixDataManager.getInstance(); - // mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - tripTables.createTripTables(mt); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java deleted file mode 100644 index fa50023..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AtWorkSubtourFrequencyDMU.java +++ /dev/null @@ -1,204 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class AtWorkSubtourFrequencyDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(AtWorkSubtourFrequencyDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Person person; - protected Tour tour; - protected IndexValues dmuIndex; - - protected double nmEatOutAccessibillity; - - protected ModelStructure modelStructure; - - public AtWorkSubtourFrequencyDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - dmuIndex = new IndexValues(); - } - - public Household getHouseholdObject() - { - return hh; - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setPersonObject(Person persObject) - { - person = persObject; - } - - public void setTourObject(Tour tourObject) - { - tour = tourObject; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug INMTF UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return household income category - */ - public int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - /** - * @return person type category index - */ - public int getPersonType() - { - return person.getPersonTypeNumber(); - } - - /** - * @return person type category index - */ - public int getFemale() - { - if (person.getPersonIsFemale() == 1) return 1; - else return 0; - } - - /** - * @return number of driving age people in household - */ - public int getDrivers() - { - return hh.getDrivers(); - } - - /** - * @return number of people of preschool person type in household - */ - public int getNumPreschoolChildren() - { - return hh.getNumPreschool(); - } - - /** - * @return number of individual non-mandatory eat-out tours for the person. - */ - public int getNumIndivEatOutTours() - { - int numTours = 0; - ArrayList tourList = person.getListOfIndividualNonMandatoryTours(); - if (tourList != null) - { - for (Tour t : tourList) - { - String tourPurpose = t.getTourPurpose(); - if (tourPurpose.equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) - { - numTours++; - } - } - } - return numTours; - } - - /** - * @return total mandatory and non-mandatory tours for the person. - */ - public int getNumTotalTours() - { - int numTours = 0; - - ArrayList wTourList = person.getListOfWorkTours(); - if (wTourList != null) numTours += wTourList.size(); - - ArrayList sTourList = person.getListOfSchoolTours(); - if (sTourList != null) numTours += sTourList.size(); - - ArrayList nmTourList = person.getListOfIndividualNonMandatoryTours(); - if (nmTourList != null) numTours += nmTourList.size(); - - return numTours; - } - - public double getNmEatOutAccessibilityWorkplace() - { - return nmEatOutAccessibillity; - } - - /** - * set the value of the non-mandatory eat out accessibility for this - * decision maker - */ - public void setNmEatOutAccessibilityWorkplace(double nmEatOutAccessibillity) - { - this.nmEatOutAccessibillity = nmEatOutAccessibillity; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java deleted file mode 100644 index 1708b43..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/AutoOwnershipChoiceDMU.java +++ /dev/null @@ -1,267 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class AutoOwnershipChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(AutoOwnershipChoiceDMU.class); - - protected HashMap methodIndexMap; - - private Household hh; - private IndexValues dmuIndex; - - private boolean useAccessibility = false; - - private double workAutoDependency = 0.0; - private double schoolAutoDependency = 0.0; - private double workAutoTime = 0.0; - - private double workersRailProportion = 0.0; - private double studentsRailProportion = 0.0; - - private double homeTazAutoAccessibility = 0.0; - private double homeTazTransitAccessibility = 0.0; - private double homeTazNonMotorizedAccessibility = 0.0; - private double homeTazMaasAccessibility = 0.0; - - public AutoOwnershipChoiceDMU() - { - dmuIndex = new IndexValues(); - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug AO UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public Household getHouseholdObject() - { - return hh; - } - - public int getGq() - { - return hh.getIsGroupQuarters(); - } - - public int getDrivers() - { - return hh.getDrivers(); - } - - public int getNumFtWorkers() - { - return hh.getNumFtWorkers(); - } - - public int getNumPtWorkers() - { - return hh.getNumPtWorkers(); - } - - public int getNumPersons18to24() - { - return hh.getNumPersons18to24(); - } - - public int getNumPersons18to35(){ - return hh.getNumPersons18to35(); - } - - public int getNumPersons6to15() - { - return hh.getNumPersons6to15(); - } - - public int getNumPersons65Plus() - { - return (hh.getNumPersons65to79()+hh.getNumPersons80plus()); - } - - public int getNumPersons80plus() - { - return hh.getNumPersons80plus(); - } - - public int getNumPersons65to79() - { - return hh.getNumPersons65to79(); - } - - public int getHhIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - public int getNumHighSchoolGraduates() - { - return hh.getNumHighSchoolGraduates(); - } - - public int getDetachedDwellingType() - { - return hh.getHhBldgsz(); - } - - public double getUseAccessibilities() - { - return useAccessibility ? 1 : 0; - } - - public double getHomeTazAutoAccessibility() - { - return homeTazAutoAccessibility; - } - - public double getHomeTazTransitAccessibility() - { - return homeTazTransitAccessibility; - } - - public double getHomeTazMaasAccessibility() - { - return homeTazMaasAccessibility; - } - - public double getHomeTazNonMotorizedAccessibility() - { - return homeTazNonMotorizedAccessibility; - } - - public double getWorkAutoDependency() - { - return workAutoDependency; - } - - public double getSchoolAutoDependency() - { - return schoolAutoDependency; - } - - public double getWorkAutoTime() { - return workAutoTime; - } - - public void setWorkAutoTime(double workAutoTime) { - this.workAutoTime = workAutoTime; - } - - public double getWorkersRailProportion() - { - return workersRailProportion; - } - - public double getStudentsRailProportion() - { - return studentsRailProportion; - } - - public void setUseAccessibilities(boolean flag) - { - useAccessibility = flag; - } - - public void setHomeTazAutoAccessibility(double acc) - { - homeTazAutoAccessibility = acc; - } - - public void setHomeTazTransitAccessibility(double acc) - { - homeTazTransitAccessibility = acc; - } - - public void setHomeTazMaasAccessibility(double acc) - { - homeTazMaasAccessibility = acc; - } - - public void setHomeTazNonMotorizedAccessibility(double acc) - { - homeTazNonMotorizedAccessibility = acc; - } - - public void setWorkAutoDependency(double value) - { - workAutoDependency = value; - } - - public void setSchoolAutoDependency(double value) - { - schoolAutoDependency = value; - } - - public void setWorkersRailProportion(double proportion) - { - workersRailProportion = proportion; - } - - public void setStudentsRailProportion(double proportion) - { - studentsRailProportion = proportion; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java deleted file mode 100644 index 7105499..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsum.java +++ /dev/null @@ -1,259 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedReader; -import java.io.FileReader; -import java.io.IOException; -import java.io.ObjectStreamException; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Set; - -import org.apache.log4j.Level; -import org.apache.log4j.LogManager; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.util.ResourceUtil; - -/** - * The {@code BikeLogsum} class holds bike logsums for use in the SANDAG model. This class is intended to be used as a singleton, and so the only way to - * access it is via the {@code getBikeLogsum} static method. It is constructed on demand and safe for concurrent access. - *

    - * Internally, the logsums are held in a mapping using node-pairs as keys. The taz and mgra pairs are held in the same mapping, with the tazs multiplied - * by -1 to avoid conflicts. To ensure good performance when building the object, a good guess as to the number of node pairs (maz pairs plus taz pairs) - * can be provided (a default value of 26 million will be used otherwise) via the {@code BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY} property. - */ -public class BikeLogsum implements SegmentedSparseMatrix,Serializable { - private static final long serialVersionUID = 660793106399818667L; - private static Logger logger = Logger.getLogger(BikeLogsum.class); - - public static final String BIKE_LOGSUM_OUTPUT_PROPERTY = "active.output.bike"; - public static final String BIKE_LOGSUM_MGRA_FILE_PROPERTY = "active.logsum.matrix.file.bike.mgra"; - public static final String BIKE_LOGSUM_TAZ_FILE_PROPERTY = "active.logsum.matrix.file.bike.taz"; - public static final String BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY = "active.logsum.matrix.node.pair.count"; - /** - * The default logsum node pair count. - */ - public static final int DEFAULT_BIKE_LOGSUM_NODE_PAIR_COUNT = 26_000_000; //testing found 18_880_631, so this should be good enough to start - - - private Map logsum; - private int[] mgraIndex; - - private static volatile BikeLogsum instance = null; - - /** - * Get the {@code BikeLogsum} instance. - * - * @param rbMap - * The model property mapping. - * - * @return the {@code BikeLogsum} instance. - */ - public static BikeLogsum getBikeLogsum(Map rbMap) { - if (instance == null) { - synchronized (BikeLogsum.class) { - if (instance == null) { //check again to see if we waited for another thread to do the initialization already - int nodePairCount = rbMap.containsKey(BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY) ? - Integer.parseInt(rbMap.get(BIKE_LOGSUM_NODE_PAIR_COUNT_PROPERTY)) : DEFAULT_BIKE_LOGSUM_NODE_PAIR_COUNT; - String mgraFile = Paths.get(rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY),rbMap.get(MgraDataManager.PROPERTIES_MGRA_DATA_FILE)).toString(); - String tazLogsumFile = Paths.get(rbMap.get(BIKE_LOGSUM_OUTPUT_PROPERTY),rbMap.get(BIKE_LOGSUM_TAZ_FILE_PROPERTY)).toString(); - String mgraLogsumFile = Paths.get(rbMap.get(BIKE_LOGSUM_OUTPUT_PROPERTY),rbMap.get(BIKE_LOGSUM_MGRA_FILE_PROPERTY)).toString(); - instance = new BikeLogsum(tazLogsumFile,mgraLogsumFile,nodePairCount,mgraFile); - } - } - } - return instance; - } - - private BikeLogsum(String tazLogsumFile, String mgraLogsumFile, int nodePairCount, String mgraFile) { - logsum = new HashMap(nodePairCount,1.01f); //capacity of nodepairs, plus a little buffer just in case - Map> tazMgraMapping = loadTazMgraMapping(mgraFile); - mgraIndex = buildMgraIndex(tazMgraMapping); - loadLogsum(tazLogsumFile,true); - loadLogsum(mgraLogsumFile,false); - } - - private int[] buildMgraIndex(Map> tazMgraMapping) { - int maxMgra = 0; - for (Set mgras : tazMgraMapping.values()) - for (int mgra : mgras) - if (maxMgra < mgra) - maxMgra = mgra; - - int[] mgraIndex = new int[maxMgra+1]; - for (int taz : tazMgraMapping.keySet()) - for (int mgra : tazMgraMapping.get(taz)) - mgraIndex[mgra] = -1*taz; - - return mgraIndex; - } - - private Map> loadTazMgraMapping(String mgraFile) { - Map> tazMgraMapping = new HashMap<>(); - boolean first = true; - String mgraColumnName = MgraDataManager.MGRA_FIELD_NAME.toLowerCase(); - String tazColumnName = MgraDataManager.MGRA_TAZ_FIELD_NAME.toLowerCase(); - int mgraColumn = -1; - int tazColumn = -1; - try (BufferedReader reader = new BufferedReader(new FileReader(mgraFile))) { - String line; - while ((line = reader.readLine()) != null) { - String[] lineData = line.trim().split(","); - if (first) { - for (int i = 0; i < lineData.length; i++) { - String column = lineData[i].toLowerCase(); - if (column.equals(mgraColumnName)) - mgraColumn = i; - if (column.equals(tazColumnName)) - tazColumn = i; - } - first = false; - continue; - } - if (lineData.length < 2) - continue; - int mgra = Integer.parseInt(lineData[mgraColumn]); - int taz = Integer.parseInt(lineData[tazColumn]); - if (!tazMgraMapping.containsKey(taz)) - tazMgraMapping.put(taz,new HashSet()); - tazMgraMapping.get(taz).add(mgra); - } - - } catch (IOException e) { - throw new RuntimeException(e); - } - return tazMgraMapping; - } - - private void loadLogsum(String logsumFile, boolean taz) { - logger.info("Processing bike logsum from " + logsumFile); - int counter = 0; - long startTime = System.currentTimeMillis(); - - int segmentWidth = BikeLogsumSegment.segmentWidth(); - try (BufferedReader reader = new BufferedReader(new FileReader(logsumFile))) { - int logsumIndex = -1; - int timeIndex = -1; - boolean first = true; - - String line; - while ((line = reader.readLine()) != null) { - String[] lineData = line.trim().split(","); - for (int i = 0; i < lineData.length; i++) - lineData[i] = lineData[i].trim(); - if (first) { - for (int i = 2; i < lineData.length; i++) { //first two are for row and column - String columnName = lineData[i].toLowerCase(); - if (columnName.contains("logsum")) - logsumIndex = i; - if (columnName.contains("time")) - timeIndex = i; - } - first = false; - continue; - } - if (++counter % 100_000 == 0) - logger.debug("Finished processing " + counter + " node pairs (logsum lookup size: " + logsum.size() + ")"); - //if we ever bring back segmented logsums, then this will be a bit more complicated - // the basic idea is all logsums first, then times (in same order) so lookups are straightforward - // without having to replicate the hashmap, which is a big data structure - double[] data = new double[] {Double.parseDouble(lineData[logsumIndex]),Double.parseDouble(lineData[timeIndex])}; - - int fromZone = Integer.parseInt(lineData[0]); - int toZone = Integer.parseInt(lineData[1]); - int indexFactor = taz ? -1 : 1; - MatrixLookup ml = new MatrixLookup(indexFactor*fromZone,indexFactor*toZone); - logsum.put(ml,data); - - } - } catch (IOException e) { - throw new RuntimeException(e); - } - logger.info("Finished processing " + counter + " node pairs (logsum lookup size: " + logsum.size() + ") in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes"); - } - - private double[] getLogsums(int rowId, int columnId) { - double[] logsums = logsum.get(new MatrixLookup(rowId,columnId)); - if (logsums == null) - logsums = logsum.get(new MatrixLookup(mgraIndex[rowId],mgraIndex[columnId])); - return logsums; - } - - @Override - public double getValue(BikeLogsumSegment segment, int rowId, int columnId) { - double[] logsums = getLogsums(rowId,columnId); - return logsums == null ? -999 : logsums[segment.getSegmentId()]; - } - - public double getLogsum(BikeLogsumSegment segment, int rowId, int columnId) { - return getValue(segment,rowId,columnId); - } - - public double getTime(BikeLogsumSegment segment, int rowId, int columnId) { - double[] logsums = getLogsums(rowId,columnId); - return logsums == null ? Double.POSITIVE_INFINITY : logsums[segment.getSegmentId()+BikeLogsumSegment.segmentWidth()]; - } - - private static class MatrixLookup implements Serializable { - private static final long serialVersionUID = -5048040835197200584L; - private final int row; - private final int column; - - private MatrixLookup(int row, int column) { - this.row = row; - this.column = column; - } - - public boolean equals(Object o) { - if ((o == null) || (!(o instanceof MatrixLookup))) - return false; - MatrixLookup ml = (MatrixLookup) o; - return (row == ml.row) && (column == ml.column); - } - - public int hashCode() { - return row + 37*column; - } - } - - private void writeObject(java.io.ObjectOutputStream out) throws IOException { - out.writeObject(logsum); - out.writeObject(mgraIndex); - } - - @SuppressWarnings("unchecked") - private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { - logsum = (Map) in.readObject(); - mgraIndex = (int[]) in.readObject(); - synchronized (BikeLogsum.class) { - //ensures singleton - readResolve will ensure all get this single value - //we need to allow the above reading of fields, though, so that deserialization is aligned correctly - if (instance == null) - instance = this; - } - } - - private Object readResolve() throws ObjectStreamException { - return instance; //ensures singelton - } - - public static void main(String ... args) { - org.apache.log4j.BasicConfigurator.configure(); - LogManager.getRootLogger().setLevel(Level.INFO); - logger.info("usage: org.sandag.abm.ctramp.BikeLogsum properties origin_mgra dest_mgra"); - Map properties = ResourceUtil.getResourceBundleAsHashMap(args[0]); - int originMgra = Integer.parseInt(args[1]); - int destMgra = Integer.parseInt(args[2]); - BikeLogsum bls = BikeLogsum.getBikeLogsum(properties); - - BikeLogsumSegment defaultSegment = new BikeLogsumSegment(true,true,true); - double logsum = bls.getLogsum(new BikeLogsumSegment(true,true,true),originMgra,destMgra); - double time = bls.getTime(new BikeLogsumSegment(true,true,true),originMgra,destMgra); - logger.info(String.format("omgra: %s, dmgra: %s, logsum: %s, time: %s",originMgra,destMgra,logsum,time)); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java deleted file mode 100644 index 9c2efc0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/BikeLogsumSegment.java +++ /dev/null @@ -1,117 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.IOException; -import java.io.Serializable; - -/** - * The {@code BikeLogsumSegment} class provides a segmentation for bicycle logsums used in the SANDAG model. The segmentation is currently - * based on three variables: - *

    - *

      - *
    • sex (male/female)
    • - *
    • tour type (mandatory/non-mandatory)
    • - *
    • trip direction (outbound/inbound) - only used if tour type is mandatory
    • - *
    - *

    - * This segmentation maps the 6 possible unique combinations into the integer indices 0 through 5, which can then be used as a lookup to a - * zero-based array or list data structure. - */ -public class BikeLogsumSegment implements Serializable { - private static final long serialVersionUID = -8429882786837391491L; - - private int segmentId; - - /** - * Constructor specifying the segment parameters. - * - * @param isFemale - * {@code true} if the sex is female. - * - * @param mandatory - * {@code true} if the tour type is mandatory. - * - * @param inbound - * {@code true} if the trip direction is inbound. - */ - public BikeLogsumSegment(boolean isFemale, boolean mandatory, boolean inbound) { - segmentId = formSegmentId(isFemale,mandatory,inbound); - } - - private int formSegmentId(boolean isFemale, boolean mandatory, boolean inbound) { - return 0; //only one segment now, but leaving in this structure in case it is needed in the future - } - - /** - * Get the segment index (0-5) for this segment. - * - * @return return this segment's index id. - */ - public int getSegmentId() { - return segmentId; - } - - /** - * Get the total number of segments available from this class. All segment index ids will be less than this value. - * - * @return the number of unique segments provided by this class. - */ - public static int segmentWidth() { - return 1; - } - - /** - * Get the segments corresponding to the tour-specific segment values. This will give all permutations which can then be combined to form a - * tour-level (as opposed to trip-level) composite logsum. - * - * @param isFemale - * {@code true} if the sex is female. - * - * @param mandatory - * {@code true} if the tour type is mandatory. - * - * @return an array of all possible segments with {@code isFemale} and {@code mandatory}. - */ - public static BikeLogsumSegment[] getTourSegments(boolean isFemale, boolean mandatory) { - return new BikeLogsumSegment[] {new BikeLogsumSegment(isFemale,mandatory,true), - new BikeLogsumSegment(isFemale,mandatory,false)}; - } - - /** - * Get the segments corresponding to the tour-specific, person-inspecific segment values. This will give all permutations which can then be - * combined to form a tour- and household-level (as opposed to trip-level) composite logsum. - * - * @param mandatory - * {@code true} if the tour type is mandatory. - * - * @return an array of all possible segments with {@code mandatory}. - */ - public static BikeLogsumSegment[] getTourSegments(boolean mandatory) { - return new BikeLogsumSegment[] {new BikeLogsumSegment(true,mandatory,true), - new BikeLogsumSegment(true,mandatory,false), - new BikeLogsumSegment(false,mandatory,true), - new BikeLogsumSegment(false,mandatory,false)}; - } - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(" 0; - boolean isMandatory = (segmentId & 2) > 0; - boolean inbound = (segmentId & 4) > 0; - sb.append(isFemale ? "female" : "male"); - sb.append(",").append(isMandatory ? "mandatory" : "non-mandatory"); - if (isMandatory) - sb.append(",").append(inbound ? "inbound" : "outbound"); - sb.append(">"); - return sb.toString(); - } - - private void writeObject(java.io.ObjectOutputStream out) throws IOException { - out.writeInt(segmentId); - } - - private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { - segmentId = in.readInt(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java deleted file mode 100644 index 7084678..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ConnectionHelper.java +++ /dev/null @@ -1,55 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.sql.Connection; -import java.sql.DriverManager; -import java.sql.SQLException; - -public final class ConnectionHelper -{ - - private String url; - private static ConnectionHelper instance; - - private ConnectionHelper(String fileName) - { - try - { - Class.forName("org.sqlite.JDBC"); - // url = "jdbc:sqlite:/c:/jim/projects/baylanta/data/status.db"; - url = "jdbc:sqlite:/" + fileName; - } catch (Exception e) - { - e.printStackTrace(); - } - } - - public static Connection getConnection(String fileName) throws SQLException - { - if (instance == null) - { - instance = new ConnectionHelper(fileName); - } - try - { - return DriverManager.getConnection(instance.url); - } catch (SQLException e) - { - throw e; - } - } - - public static void close(Connection connection) - { - try - { - if (connection != null) - { - connection.close(); - } - } catch (SQLException e) - { - e.printStackTrace(); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java deleted file mode 100644 index 93d988c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CoordinatedDailyActivityPatternDMU.java +++ /dev/null @@ -1,442 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * Decision making unit object for the Coordinated Daily Activity Pattern Model. - * This DMU contains all the getters specified in the UEC, i.e. all the "@" - * variables. - * - * @author D. Ory - * - */ -public class CoordinatedDailyActivityPatternDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(CoordinatedDailyActivityPatternDMU.class); - - protected HashMap methodIndexMap; - - protected IndexValues dmuIndex; - - protected Household householdObject; - protected Person personA, personB, personC; - protected double workModeChoiceLogsumA; - protected double schoolModeChoiceLogsumA; - protected double retailAccessibility; - - protected double workAccessForMandatoryDap; - - protected int numAdultsWithNonMandatoryDap; - protected int numAdultsWithMandatoryDap; - protected int numKidsWithNonMandatoryDap; - protected int numKidsWithMandatoryDap; - protected int allAdultsAtHome; - - public CoordinatedDailyActivityPatternDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int zoneId) - { - dmuIndex.setZoneIndex(zoneId); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (householdObject.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug CDAP UEC"); - } - - } - - public IndexValues getIndexValues() - { - return dmuIndex; - } - - public void setHousehold(Household passedInHouseholdObject) - { - - householdObject = passedInHouseholdObject; - - // set the household index - dmuIndex.setHHIndex(passedInHouseholdObject.getHhId()); - - // set the home zone as the zone index - dmuIndex.setZoneIndex(passedInHouseholdObject.getHhMgra()); - } - - public void setPersonA(Person passedInPersonA) - { - this.personA = passedInPersonA; - } - - public void setPersonB(Person passedInPersonB) - { - this.personB = passedInPersonB; - } - - public void setPersonC(Person passedInPersonC) - { - this.personC = passedInPersonC; - } - - // full-time worker - public int getFullTimeWorkerA() - { - return (personA.getPersonTypeIsFullTimeWorker()); - } - - public int getFullTimeWorkerB() - { - return (personB.getPersonTypeIsFullTimeWorker()); - } - - public int getFullTimeWorkerC() - { - return (personC.getPersonTypeIsFullTimeWorker()); - } - - // part-time worker - public int getPartTimeWorkerA() - { - return (personA.getPersonTypeIsPartTimeWorker()); - } - - public int getPartTimeWorkerB() - { - return (personB.getPersonTypeIsPartTimeWorker()); - } - - public int getPartTimeWorkerC() - { - return (personC.getPersonTypeIsPartTimeWorker()); - } - - // university student - public int getUniversityStudentA() - { - return (personA.getPersonIsUniversityStudent()); - } - - public int getUniversityStudentB() - { - return (personB.getPersonIsUniversityStudent()); - } - - public int getUniversityStudentC() - { - return (personC.getPersonIsUniversityStudent()); - } - - // non-working adult - public int getNonWorkingAdultA() - { - return (personA.getPersonIsNonWorkingAdultUnder65()); - } - - public int getNonWorkingAdultB() - { - return (personB.getPersonIsNonWorkingAdultUnder65()); - } - - public int getNonWorkingAdultC() - { - return (personC.getPersonIsNonWorkingAdultUnder65()); - } - - // retired - public int getRetiredA() - { - return (personA.getPersonIsNonWorkingAdultOver65()); - } - - public int getRetiredB() - { - return (personB.getPersonIsNonWorkingAdultOver65()); - } - - public int getRetiredC() - { - return (personC.getPersonIsNonWorkingAdultOver65()); - } - - // driving age school child - public int getDrivingAgeSchoolChildA() - { - return (personA.getPersonIsStudentDriving()); - } - - public int getDrivingAgeSchoolChildB() - { - return (personB.getPersonIsStudentDriving()); - } - - public int getDrivingAgeSchoolChildC() - { - return (personC.getPersonIsStudentDriving()); - } - - // non-driving school-age child - public int getPreDrivingAgeSchoolChildA() - { - return (personA.getPersonIsStudentNonDriving()); - } - - public int getPreDrivingAgeSchoolChildB() - { - return (personB.getPersonIsStudentNonDriving()); - } - - public int getPreDrivingAgeSchoolChildC() - { - return (personC.getPersonIsStudentNonDriving()); - } - - // pre-school child - public int getPreSchoolChildA() - { - return (personA.getPersonIsPreschoolChild()); - } - - public int getPreSchoolChildB() - { - return (personB.getPersonIsPreschoolChild()); - } - - public int getPreSchoolChildC() - { - return (personC.getPersonIsPreschoolChild()); - } - - // age - public int getAgeA() - { - return (personA.getAge()); - } - - // female - public int getFemaleA() - { - return (personA.getPersonIsFemale()); - } - - // household more cars than workers - public int getMoreCarsThanWorkers() - { - - int workers = householdObject.getWorkers(); - int autos = householdObject.getAutosOwned(); - - if (autos > workers) return 1; - return 0; - - } - - // household fewer cars than workers - public int getFewerCarsThanWorkers() - { - - int workers = householdObject.getWorkers(); - int autos = householdObject.getAutosOwned(); - - if (autos < workers) return 1; - return 0; - - } - - // household with zero cars - public int getZeroCars() - { - int autos = householdObject.getAutosOwned(); - if (autos == 0) return 1; - return 0; - - } - - // household income - public int getHHIncomeInDollars() - { - return householdObject.getIncomeInDollars(); - } - - public int getUsualWorkLocationIsHomeA() - { - if (personA.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 1; - else return 0; - } - - public int getNoUsualWorkLocationA() - { - if (personA.getWorkLocation() > 0 - && personA.getWorkLocation() != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 0; - else return 1; - } - - // no usual school location is 1 if person is a student, location is home - // mgra, - // and distance to school is 0. - public int getNoUsualSchoolLocationA() - { - if (personA.getPersonIsStudent() == 1 - && personA.getUsualSchoolLocation() == personA.getHouseholdObject().getHhMgra() - && personA.getSchoolLocationDistance() == 0) return 0; - else return 1; - } - - public int getHhSize() - { - - int hhSize = Math.min(HouseholdCoordinatedDailyActivityPatternModel.MAX_MODEL_HH_SIZE, - householdObject.getSize()); - - return (hhSize); - } - - public int getHhDetach() - { - return householdObject.getHhBldgsz(); - } - - public int getNumAdultsWithNonMandatoryDap() - { - return numAdultsWithNonMandatoryDap; - } - - public int getNumAdultsWithMandatoryDap() - { - return numAdultsWithMandatoryDap; - } - - public int getNumKidsWithNonMandatoryDap() - { - return numKidsWithNonMandatoryDap; - } - - public int getNumKidsWithMandatoryDap() - { - return numKidsWithMandatoryDap; - } - - public void setNumAdultsWithNonMandatoryDap(int value) - { - numAdultsWithNonMandatoryDap = value; - } - - public void setNumAdultsWithMandatoryDap(int value) - { - numAdultsWithMandatoryDap = value; - } - - public void setNumKidsWithNonMandatoryDap(int value) - { - numKidsWithNonMandatoryDap = value; - } - - public void setNumKidsWithMandatoryDap(int value) - { - numKidsWithMandatoryDap = value; - } - - public int getAllAdultsAtHome() - { - return allAdultsAtHome; - } - - public void setAllAdultsAtHome(int value) - { - allAdultsAtHome = value; - } - - public void setWorkAccessForMandatoryDap(double logsum) - { - workAccessForMandatoryDap = logsum; - } - - public double getWorkAccessForMandatoryDap() - { - return workAccessForMandatoryDap; - } - - public void setWorkLocationModeChoiceLogsumA(double logsum) - { - workModeChoiceLogsumA = logsum; - } - - public double getWorkLocationModeChoiceLogsumA() - { - return workModeChoiceLogsumA; - } - - public void setSchoolLocationModeChoiceLogsumA(double logsum) - { - schoolModeChoiceLogsumA = logsum; - } - - public double getSchoolLocationModeChoiceLogsumA() - { - return schoolModeChoiceLogsumA; - } - - public void setRetailAccessibility(double logsum) - { - retailAccessibility = logsum; - } - - public double getRetailAccessibility() - { - return retailAccessibility; - } - - public int getTelecommuteFrequencyA() { - return personA.getTelecommuteChoice(); - } - - public int getTelecommuteFrequencyB() { - return personB.getTelecommuteChoice(); - } - - public int getTelecommuteFrequencyC() { - return personC.getTelecommuteChoice(); - } - - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java deleted file mode 100644 index bfb4032..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampApplication.java +++ /dev/null @@ -1,2481 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.NotSerializableException; -import java.io.PrintWriter; -import java.io.Serializable; -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.MissingResourceException; -import java.util.ResourceBundle; -import java.util.Set; -import java.util.TreeMap; -import org.apache.log4j.Logger; -import org.jppf.client.JPPFClient; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.StoredUtilityData; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.DataFile; -import com.pb.common.datafile.DataReader; -import com.pb.common.datafile.DataWriter; -import com.pb.common.matrix.MatrixIO32BitJvm; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; -// import org.sandag.abm.accessibilities.BuildAccessibilities; -// import -// org.sandag.abm.ctramp.HouseholdDataWriter; -// import org.sandag.abm.ctramp.IndividualMandatoryTourFrequencyModel; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -// 1.0.1 - 09/21/09 - starting point for SANDAG AB model implementation - AO -// model - -public class CtrampApplication - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(CtrampApplication.class); - - public static final String VERSION = "2.0.0"; - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - - public static final String PROPERTIES_BASE_NAME = "ctramp"; - public static final String PROPERTIES_PROJECT_DIRECTORY = "Project.Directory"; - - public static final String PROPERTIES_UEC_PATH = "uec.path"; - public static final String SQLITE_DATABASE_FILENAME = "Sqlite.DatabaseFileName"; - - public static final String PROPERTIES_RUN_POPSYN = "RunModel.PopulationSynthesizer"; - public static final String PROPERTIES_RUN_PRE_AUTO_OWNERSHIP = "RunModel.PreAutoOwnership"; - public static final String PROPERTIES_RUN_WORKSCHOOL_CHOICE = "RunModel.UsualWorkAndSchoolLocationChoice"; - public static final String PROPERTIES_RUN_AUTO_OWNERSHIP = "RunModel.AutoOwnership"; - public static final String PROPERTIES_RUN_TRANSPONDER_CHOICE = "RunModel.TransponderChoice"; - public static final String PROPERTIES_RUN_FREE_PARKING_AVAILABLE = "RunModel.FreeParking"; - public static final String PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP = "RunModel.InternalExternal"; - public static final String PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN = "RunModel.CoordinatedDailyActivityPattern"; - public static final String PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ = "RunModel.IndividualMandatoryTourFrequency"; - public static final String PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR = "RunModel.MandatoryTourDepartureTimeAndDuration"; - public static final String PROPERTIES_RUN_SCHOOL_ESCORT_MODEL = "RunModel.SchoolEscortModel"; - public static final String PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE = "RunModel.MandatoryTourModeChoice"; - public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ = "RunModel.AtWorkSubTourFrequency"; - public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE = "RunModel.AtWorkSubTourLocationChoice"; - public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE = "RunModel.AtWorkSubTourModeChoice"; - public static final String PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR = "RunModel.AtWorkSubTourDepartureTimeAndDuration"; - public static final String PROPERTIES_RUN_JOINT_TOUR_FREQ = "RunModel.JointTourFrequency"; - public static final String PROPERTIES_RUN_JOINT_LOCATION_CHOICE = "RunModel.JointTourLocationChoice"; - public static final String PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE = "RunModel.JointTourModeChoice"; - public static final String PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR = "RunModel.JointTourDepartureTimeAndDuration"; - public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ = "RunModel.IndividualNonMandatoryTourFrequency"; - public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE = "RunModel.IndividualNonMandatoryTourLocationChoice"; - public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE = "RunModel.IndividualNonMandatoryTourModeChoice"; - public static final String PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR = "RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration"; - public static final String PROPERTIES_RUN_STOP_FREQUENCY = "RunModel.StopFrequency"; - public static final String PROPERTIES_RUN_STOP_LOCATION = "RunModel.StopLocation"; - - public static final String PROPERTIES_RUN_WORK_LOC_CHOICE_KEY = "uwsl.run.workLocChoice"; - public static final String PROPERTIES_RUN_SCHOOL_LOC_CHOICE_KEY = "uwsl.run.schoolLocChoice"; - public static final String PROPERTIES_WRITE_WORK_SCHOOL_LOC_RESULTS_KEY = "uwsl.write.results"; - - public static final String PROPERTIES_UEC_AUTO_OWNERSHIP = "UecFile.AutoOwnership"; - public static final String PROPERTIES_UEC_DAILY_ACTIVITY_PATTERN = "UecFile.CoordinatedDailyActivityPattern"; - public static final String PROPERTIES_UEC_INDIV_MANDATORY_TOUR_FREQ = "UecFile.IndividualMandatoryTourFrequency"; - public static final String PROPERTIES_UEC_MAND_TOUR_DEP_TIME_AND_DUR = "UecFile.TourDepartureTimeAndDuration"; - public static final String PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ = "UecFile.IndividualNonMandatoryTourFrequency"; - - public static final String PROPERTIES_CLEAR_MATRIX_MANAGER_ON_START = "RunModel.Clear.MatrixMgr.At.Start"; - - public static final String READ_ACCESSIBILITIES = "acc.read.input.file"; - - // TODO eventually move to model-specific structure object - public static final int TOUR_MODE_CHOICE_WORK_MODEL_UEC_PAGE = 1; - public static final int TOUR_MODE_CHOICE_UNIVERSITY_MODEL_UEC_PAGE = 2; - public static final int TOUR_MODE_CHOICE_HIGH_SCHOOL_MODEL_UEC_PAGE = 3; - public static final int TOUR_MODE_CHOICE_GRADE_SCHOOL_MODEL_UEC_PAGE = 4; - - // TODO eventually move to model-specific model structure object - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_MODEL_UEC_PAGE = 1; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_DEPARTURE_UEC_PAGE = 2; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_DURATION_UEC_PAGE = 3; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_WORK_ARRIVAL_UEC_PAGE = 4; - - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_MODEL_UEC_PAGE = 5; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_DEPARTURE_UEC_PAGE = 6; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_DURATION_UEC_PAGE = 7; - public static final int MANDATORY_TOUR_DEP_TIME_AND_DUR_SCHOOL_ARRIVAL_UEC_PAGE = 8; - - public static final String PROPERTIES_SCHEDULING_NUMBER_OF_TIME_PERIODS = "Scheduling.NumberOfTimePeriods"; - public static final String PROPERTIES_SCHEDULING_FIRST_TIME_PERIOD = "Scheduling.FirstTimePeriod"; - - static final String PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER = "RunModel.RestartWithHhServer"; - - static final String PROPERTIES_HOUSEHOLD_DISK_OBJECT_FILE_NAME = "Households.disk.object.base.name"; - static final String PROPERTIES_HOUSEHOLD_DISK_OBJECT_KEY = "Read.HouseholdDiskObjectFile"; - - public static final String PROPERTIES_RESULTS_AUTO_OWNERSHIP = "Results.AutoOwnership"; - public static final String PROPERTIES_RESULTS_CDAP = "Results.CoordinatedDailyActivityPattern"; - - public static final String PROPERTIES_OUTPUT_WRITE_SWITCH = "CTRAMP.Output.WriteToDiskSwitch"; - public static final String PROPERTIES_OUTPUT_HOUSEHOLD_FILE = "CTRAMP.Output.HouseholdFile"; - public static final String PROPERTIES_OUTPUT_PERSON_FILE = "CTRAMP.Output.PersonFile"; - - public static final String PROPERTIES_WRITE_DATA_TO_FILE = "Results.WriteDataToFiles"; - public static final String PROPERTIES_WRITE_DATA_TO_DATABASE = "Results.WriteDataToDatabase"; - - public static final String PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS = "TourModeChoice.Save.UtilsAndProbs"; - - public static final String PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE = "UsualWorkLocationChoice.ShadowPrice.Input.File"; - public static final String PROPERTIES_SCHOOL_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE = "UsualSchoolLocationChoice.ShadowPrice.Input.File"; - - public static final String PROPERTIES_NUMBER_OF_GLOBAL_ITERATIONS = "Global.iterations"; - - public static final String ALT_FIELD_NAME = "a"; - public static final String START_FIELD_NAME = "depart"; - public static final String END_FIELD_NAME = "arrive"; - - private static final int NUM_WRITE_PACKETS = 1000; - - private ResourceBundle resourceBundle; - private HashMap propertyMap; - - private MatrixDataServerIf ms; - private MatrixDataManager mdm; - - private ModelStructure modelStructure; - protected String projectDirectory; - - private HashMap> cdapByHhSizeAndPattern; - private HashMap> cdapByPersonTypeAndActivity; - - private BuildAccessibilities aggAcc; - private boolean calculateLandUseAccessibilities; - - public CtrampApplication(ResourceBundle rb, HashMap rbMap, - boolean calculateLandUseAccessibilities) - { - resourceBundle = rb; - propertyMap = rbMap; - this.calculateLandUseAccessibilities = calculateLandUseAccessibilities; - } - - public void setupModels(ModelStructure modelStructure) - { - - this.modelStructure = modelStructure; - - } - - // public void runPopulationSynthesizer( SANDAGPopSyn populationSynthesizer - // ){ - // - // // run population synthesizer - // boolean runModelPopulationSynthesizer = - // ResourceUtil.getBooleanProperty(resourceBundle, PROPERTIES_RUN_POPSYN); - // if(runModelPopulationSynthesizer){ - // populationSynthesizer.run(); - // } - // - // } - - public void runModels(HouseholdDataManagerIf householdDataManager, - CtrampDmuFactoryIf dmuFactory, int globalIterationNumber, float iterationSampleRate) - { - - logger.info("Running JPPF CtrampApplication.runModels() for " - + householdDataManager.getNumHouseholds() + " households."); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(propertyMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(propertyMap, - "RunModel.MatrixServerPort"); - } catch (RuntimeException e) - { - // if no matrix server address entry is found, leave undefined -- - // it's eithe not needed or show could create an error. - } - } catch (RuntimeException e) - { - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServer matrixServer = null; - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = startMatrixServerProcess(matrixServerAddress, serverPort); - ms = matrixServer; - } else - { - ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error(String - .format("exception caught running ctramp model components -- exiting."), e); - throw new RuntimeException(e); - - } - - // get the property that indicates if the MatrixDataServer should be - // cleared. - // default is to clear the manager - boolean clearMatrixMgr = true; - try - { - clearMatrixMgr = Util.getBooleanValueFromPropertyMap(propertyMap, - PROPERTIES_CLEAR_MATRIX_MANAGER_ON_START); - } catch (RuntimeException e) - { - // catch the RuntimeExcption that's thrown if the property key - // is not found in the properties file. - // no need to anything - the boolean clearMatrixMgr was - // initialized to the default action. - } - - // if the property to clear matrices is true and a remote - // MatrixDataServer is being used, clear the matrices. - // if matrices are being read directly into the current process, no - // need to clear. - if (clearMatrixMgr && !matrixServerAddress.equalsIgnoreCase("localhost")) ms.clear(); - - // run core activity based model for the specified iteration - runModelSequence(globalIterationNumber, householdDataManager, dmuFactory); - - } - - /** - * This method maintains the sequencing of the various AB Model choice model - * components - * - * @param iteration - * is the global iteration number in the sequence of AB Model - * runs during feedback - * @param householdDataManager - * is the handle to the household object data manager - * @param dmuFactory - * is the factory object for creating DMU objects used in choice - * models - */ - private void runModelSequence(int iteration, HouseholdDataManagerIf householdDataManager, - CtrampDmuFactoryIf dmuFactory) - { - - String restartModel = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER); - boolean logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - if (restartModel == null) restartModel = "none"; - if (!restartModel.equalsIgnoreCase("none")) restartModels(householdDataManager); - - JPPFClient jppfClient = null; - - boolean runPreAutoOwnershipChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_PRE_AUTO_OWNERSHIP); - if (runPreAutoOwnershipChoiceModel) - { - - logger.info("creating Accessibilities Object for Pre-AO."); - buildNonMandatoryAccessibilities(calculateLandUseAccessibilities); - - logger.info("starting Pre-Auto Ownership Model."); - HashMap propertyMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - - householdDataManager.resetPreAoRandom(); - - HouseholdAutoOwnershipModel aoModel = new HouseholdAutoOwnershipModel(propertyMap, - dmuFactory, aggAcc.getAccessibilitiesTableObject(), null); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - for (int i = 0; i < householdArray.length; ++i) - { - - try - { - aoModel.applyModel(householdArray[i], true); - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught running pre-AO for i=%d, startIndex=%d, endIndex=%d, hhId=%d.", - i, startIndex, endIndex, householdArray[i].getHhId())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - householdDataManager.setHhArray(householdArray, startIndex); - - } - - saveAoResults(householdDataManager, projectDirectory, true); - - // clear the zonal data used in the AO UEC so a different zonal data - // file - // (MGRA data) can be used later by other UECs. - // TableDataSetManager tableDataManager = - // TableDataSetManager.getInstance(); - // tableDataManager.clearData(); - } - logger.info("flag to run pre-AO was set to: " + runPreAutoOwnershipChoiceModel); - logAoResults(householdDataManager, true); - - boolean runUsualWorkSchoolChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_WORKSCHOOL_CHOICE); - if (runUsualWorkSchoolChoiceModel) - { - - boolean runWorkLocationChoice = false; - try - { - String stringValue = resourceBundle.getString(PROPERTIES_RUN_WORK_LOC_CHOICE_KEY); - runWorkLocationChoice = stringValue.equalsIgnoreCase("true"); - } catch (MissingResourceException e) - { - // default value is true if property was not defined - runWorkLocationChoice = true; - } - logger.info("flag to run work location choice was set to: " + runWorkLocationChoice); - - boolean runSchoolLocationChoice = false; - try - { - String stringValue = resourceBundle.getString(PROPERTIES_RUN_SCHOOL_LOC_CHOICE_KEY); - runSchoolLocationChoice = stringValue.equalsIgnoreCase("true"); - } catch (MissingResourceException e) - { - // default value is true if property was not defined - runSchoolLocationChoice = true; - } - logger.info("flag to run school location choice was set to: " + runSchoolLocationChoice); - - boolean writeLocationChoiceResultsFile = false; - try - { - String stringValue = resourceBundle - .getString(PROPERTIES_WRITE_WORK_SCHOOL_LOC_RESULTS_KEY); - writeLocationChoiceResultsFile = stringValue.equalsIgnoreCase("true"); - } catch (MissingResourceException e) - { - // default value is true if property was not defined - writeLocationChoiceResultsFile = true; - } - logger.info("flag to write uwsl result was set to: " + writeLocationChoiceResultsFile); - - if (aggAcc == null) - { - logger.info("creating Accessibilities Object for UWSL."); - buildNonMandatoryAccessibilities(calculateLandUseAccessibilities); - } - - // new the usual school and location choice model object - jppfClient = new JPPFClient(); - UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( - resourceBundle, restartModel, jppfClient, modelStructure, ms, dmuFactory, - aggAcc); - - if (runWorkLocationChoice) - { - // calculate and get the array of worker size terms table - - // MGRAs by - // occupations - aggAcc.createWorkSegmentNameIndices(); - aggAcc.calculateWorkerSizeTerms(); - double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); - - // run the model - logger.info("starting usual work location choice."); - usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, - workerSizeTerms); - logger.info("finished with usual work location choice."); - } - - if (runSchoolLocationChoice) - { - aggAcc.createSchoolSegmentNameIndices(); - aggAcc.calculateSchoolSizeTerms(); - double[][] schoolFactors = aggAcc.calculateSchoolSegmentFactors(); - double[][] schoolSizeTerms = aggAcc.getSchoolSizeTerms(); - - logger.info("starting usual school location choice."); - usualWorkSchoolLocationChoiceModel.runSchoolLocationChoiceModel( - householdDataManager, schoolSizeTerms, schoolFactors); - logger.info("finished with usual school location choice."); - } - - if (writeLocationChoiceResultsFile) - { - logger.info("writing work/school location choice results file; may take a few minutes ..."); - usualWorkSchoolLocationChoiceModel.saveResults(householdDataManager, - projectDirectory, iteration); - logger.info(String - .format("finished writing work/school location choice results file.")); - } - - usualWorkSchoolLocationChoiceModel = null; - - } - logger.info("flag to run UWSL was set to: " + runUsualWorkSchoolChoiceModel); - - boolean runAutoOwnershipChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_AUTO_OWNERSHIP); - boolean runTransponderChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_TRANSPONDER_CHOICE); - boolean runFreeParkingChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_FREE_PARKING_AVAILABLE); - boolean runInternalExternalTripChoiceModel = ResourceUtil.getBooleanProperty( - resourceBundle, PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP); - boolean runCoordinatedDailyActivityPatternChoiceModel = ResourceUtil.getBooleanProperty( - resourceBundle, PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN); - boolean runMandatoryTourFreqChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ); - boolean runJointTourFreqChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_JOINT_TOUR_FREQ); - boolean runIndivNonManTourFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ); - boolean runAtWorkSubTourFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ); - boolean runStopFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_STOP_FREQUENCY); - - if (runAutoOwnershipChoiceModel || runTransponderChoiceModel || runFreeParkingChoiceModel - || runInternalExternalTripChoiceModel - || runCoordinatedDailyActivityPatternChoiceModel || runMandatoryTourFreqChoiceModel - || runIndivNonManTourFrequencyModel || runAtWorkSubTourFrequencyModel - || runStopFrequencyModel) - { - - // We're resetting the random number sequence used by pre-AO for the - // primary AO - if (runAutoOwnershipChoiceModel) householdDataManager.resetPreAoRandom(); - - if (runTransponderChoiceModel) - householdDataManager.computeTransponderChoiceTazPercentArrays(); - - logger.info("starting HouseholdChoiceModelRunner."); - HashMap propertyMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner(propertyMap, - jppfClient, restartModel, householdDataManager, ms, modelStructure, dmuFactory); - runner.runHouseholdChoiceModels(); - - if (runAutoOwnershipChoiceModel) - { - saveAoResults(householdDataManager, projectDirectory, false); - if (logResults) logAoResults(householdDataManager, false); - } - - if (runTransponderChoiceModel) - { - if (logResults) logTpResults(householdDataManager); - } - - if (runFreeParkingChoiceModel) - { - if (logResults) logFpResults(householdDataManager); - } - - if (runInternalExternalTripChoiceModel) - { - if (logResults) logIeResults(householdDataManager); - } - - if (runCoordinatedDailyActivityPatternChoiceModel) - { - saveCdapResults(householdDataManager, projectDirectory); - if (logResults) logCdapResults(householdDataManager); - } - - if (runMandatoryTourFreqChoiceModel) - { - if (logResults) logImtfResults(householdDataManager); - } - - if (runJointTourFreqChoiceModel) - { - if (logResults) logJointModelResults(householdDataManager, dmuFactory); - } - - if (runAtWorkSubTourFrequencyModel) - { - if (logResults) logAtWorkSubtourFreqResults(householdDataManager); - } - - if (runStopFrequencyModel) - { - if (logResults) logIndivStfResults(householdDataManager); - } - - } - - jppfClient.close(); - - boolean writeTextFileFlag = false; - boolean writeSqliteFlag = false; - try - { - writeTextFileFlag = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_WRITE_DATA_TO_FILE); - } catch (MissingResourceException e) - { - // if exception is caught while getting property file value, then - // boolean - // flag remains false - } - try - { - writeSqliteFlag = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_WRITE_DATA_TO_DATABASE); - } catch (MissingResourceException e) - { - // if exception is caught while getting property file value, then - // boolean - // flag remains false - } - - HouseholdDataWriter dataWriter = null; - if (writeTextFileFlag || writeSqliteFlag) - { - dataWriter = new HouseholdDataWriter(propertyMap, modelStructure, iteration); - - if (writeTextFileFlag) dataWriter.writeDataToFiles(householdDataManager); - - if (writeSqliteFlag) - { - String dbFilename = ""; - try - { - String baseDir = resourceBundle.getString(PROPERTIES_PROJECT_DIRECTORY); - dbFilename = baseDir + resourceBundle.getString(SQLITE_DATABASE_FILENAME) + "_" - + iteration; - dataWriter.writeDataToDatabase(householdDataManager, dbFilename); - } catch (MissingResourceException e) - { - // if exception is caught while getting property file value, - // then - // boolean flag remains false - } - } - } - - } - - /** - * Build the mandatory accessibilities object used by the usual work and - * school location choice models - * - * @return BuildAccessibilities object containing mandatory size term and - * logsum information private void buildMandatoryAccessibilities() { - * - * HashMap propertyMap = - * ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle); - * - * if ( aggAcc == null ) aggAcc = new BuildAccessibilities( - * propertyMap ); - * - * MatrixDataManager mdm = MatrixDataManager.getInstance(); - * mdm.setMatrixDataServerObject( ms ); - * - * aggAcc.setupBuildAccessibilities( propertyMap ); - * aggAcc.calculateConstants(); - * - * // do this in dest choice model - * //aggAcc.buildAccessibilityComponents( propertyMap ); - * - * } - */ - - /** - * Build the non-mandatory accessibilities object used by the auto ownership - * model - * - * @return BuildAccessibilities object containing non-mandatory size term - * and logsum information - */ - private void buildNonMandatoryAccessibilities(boolean calculateLandUseAccessibilities) - { - - HashMap propertyMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - - aggAcc = BuildAccessibilities.getInstance(); - aggAcc.setupBuildAccessibilities(propertyMap, calculateLandUseAccessibilities); - - - if(calculateLandUseAccessibilities) - aggAcc.setCalculatedLandUseAccessibilities(); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateWorkerSizeTerms(); - aggAcc.createSchoolSegmentNameIndices(); - aggAcc.calculateSchoolSizeTerms(); - aggAcc.calculateConstants(); - // aggAcc.buildAccessibilityComponents(propertyMap); - - boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, - READ_ACCESSIBILITIES); - if (readAccessibilities) - { - - // output data - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - if (isJppfRunningDistributed()) { - // release the memory used to store the access-tap, tap-egress, and - // tap-tap utilities while calculating accessibilities for the - // client program - // don't do this if we are running jppf in local mode - HashMap rbMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - StoredUtilityData.getInstance(MgraDataManager.getInstance(rbMap).getMaxMgra(), - MgraDataManager.getInstance(rbMap).getMaxTap(), - TazDataManager.getInstance(rbMap).getMaxTaz(), - BestTransitPathCalculator.ACC_EGR, ModelStructure.PERIODCODES) - .deallocateArrays(); - - MatrixDataManager.getInstance().clearData(); - } - } - - } - - private boolean isJppfRunningDistributed() { - //note: this assumes that the jppf config file is being entered in through a system property, and that it is a property file - String jppfConfigFile = System.getProperty("jppf.config"); - ResourceBundle jppfConfig = ResourceUtil.getResourceBundle(jppfConfigFile.replace(".properties","")); - try { - return !jppfConfig.getString("jppf.local.execution.enabled").equalsIgnoreCase("true"); - } catch (MissingResourceException e) { - return false; - } - } - - /* - * method used in original ARC implementation private void runIteration( int - * iteration, HouseholdDataManagerIf householdDataManager, - * CtrampDmuFactoryIf dmuFactory ) { String restartModel = ""; if ( - * hhDiskObjectKey != null && ! hhDiskObjectKey.equalsIgnoreCase("none") ) { - * String doFileName = hhDiskObjectFile + "_" + hhDiskObjectKey; - * householdDataManager.createHhArrayFromSerializedObjectInFile( doFileName, - * hhDiskObjectKey ); restartModel = hhDiskObjectKey; restartModels ( - * householdDataManager ); } else { restartModel = ResourceUtil.getProperty( - * resourceBundle, PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); if ( - * restartModel == null ) restartModel = "none"; if ( ! - * restartModel.equalsIgnoreCase("none") ) restartModels ( - * householdDataManager ); } JPPFClient jppfClient = new JPPFClient(); - * boolean runUsualWorkSchoolChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_WORKSCHOOL_CHOICE); if(runUsualWorkSchoolChoiceModel){ // - * create an object for calculating destination choice attraction size terms - * and managing shadow price calculations. DestChoiceSize dcSizeObj = new - * DestChoiceSize( modelStructure, tazDataManager ); // new the usual school - * and location choice model object UsualWorkSchoolLocationChoiceModel - * usualWorkSchoolLocationChoiceModel = new - * UsualWorkSchoolLocationChoiceModel(resourceBundle, restartModel, - * jppfClient, modelStructure, ms, tazDataManager, dcSizeObj, dmuFactory ); - * // run the model logger.info ( - * "starting usual work and school location choice."); - * usualWorkSchoolLocationChoiceModel - * .runSchoolAndLocationChoiceModel(householdDataManager); logger.info ( - * "finished with usual work and school location choice."); logger.info ( - * "writing work/school location choice results file; may take a few minutes ..." - * ); usualWorkSchoolLocationChoiceModel.saveResults( householdDataManager, - * projectDirectory, iteration ); logger.info ( - * String.format("finished writing results file.") ); - * usualWorkSchoolLocationChoiceModel = null; dcSizeObj = null; System.gc(); - * // write a disk object fle for the householdDataManager, in case we want - * to restart from the next step. if ( hhDiskObjectFile != null ) { - * logger.info ( - * "writing household disk object file after work/school location choice; may take a long time ..." - * ); String hhFileName = String.format( "%s_%d_ao", hhDiskObjectFile, - * iteration ); - * householdDataManager.createSerializedHhArrayInFileFromObject( hhFileName, - * "ao" ); logger.info (String.format( - * "finished writing household disk object file = %s after uwsl; continuing to household choice models ..." - * , hhFileName) ); } } boolean runAutoOwnershipChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_AUTO_OWNERSHIP ); boolean runFreeParkingChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_FREE_PARKING_AVAILABLE ); boolean - * runCoordinatedDailyActivityPatternChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN ); boolean - * runMandatoryTourFreqChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ ); boolean - * runMandatoryTourTimeOfDayChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR ); boolean - * runMandatoryTourModeChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE ); boolean - * runJointTourFrequencyModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_JOINT_TOUR_FREQ ); boolean runJointTourLocationChoiceModel - * = ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_JOINT_LOCATION_CHOICE ); boolean - * runJointTourModeChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE ); boolean - * runJointTourDepartureTimeAndDurationModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR ); boolean - * runIndivNonManTourFrequencyModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ ); boolean - * runIndivNonManTourLocationChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE ); boolean - * runIndivNonManTourModeChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE ); boolean - * runIndivNonManTourDepartureTimeAndDurationModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR ); boolean - * runAtWorkSubTourFrequencyModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ ); boolean - * runAtWorkSubtourLocationChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE ); boolean - * runAtWorkSubtourModeChoiceModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE ); boolean - * runAtWorkSubtourDepartureTimeAndDurationModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR ); boolean - * runStopFrequencyModel = ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_STOP_FREQUENCY ); boolean runStopLocationModel = - * ResourceUtil.getBooleanProperty(resourceBundle, - * PROPERTIES_RUN_STOP_LOCATION ); boolean runHouseholdModels = false; if ( - * runAutoOwnershipChoiceModel || runFreeParkingChoiceModel || - * runCoordinatedDailyActivityPatternChoiceModel || - * runMandatoryTourFreqChoiceModel || runMandatoryTourModeChoiceModel || - * runMandatoryTourTimeOfDayChoiceModel || runJointTourFrequencyModel || - * runJointTourLocationChoiceModel || runJointTourModeChoiceModel || - * runJointTourDepartureTimeAndDurationModel || - * runIndivNonManTourFrequencyModel || runIndivNonManTourLocationChoiceModel - * || runIndivNonManTourModeChoiceModel || - * runIndivNonManTourDepartureTimeAndDurationModel || - * runAtWorkSubTourFrequencyModel || runAtWorkSubtourLocationChoiceModel || - * runAtWorkSubtourModeChoiceModel || - * runAtWorkSubtourDepartureTimeAndDurationModel || runStopFrequencyModel || - * runStopLocationModel ) runHouseholdModels = true; // disk object file is - * labeled with the next component eligible to be run if model restarted - * String lastComponent = "uwsl"; String nextComponent = "ao"; if( - * runHouseholdModels ) { logger.info ( - * "starting HouseholdChoiceModelRunner." ); HashMap - * propertyMap = - * ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle); - * HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( - * propertyMap, jppfClient, restartModel, householdDataManager, ms, - * modelStructure, tazDataManager, dmuFactory ); - * runner.runHouseholdChoiceModels(); if( runAutoOwnershipChoiceModel ){ - * saveAoResults( householdDataManager, projectDirectory ); logAoResults( - * householdDataManager ); lastComponent = "ao"; nextComponent = "fp"; } if( - * runFreeParkingChoiceModel ){ logFpResults( householdDataManager ); - * lastComponent = "fp"; nextComponent = "cdap"; } if( - * runCoordinatedDailyActivityPatternChoiceModel ){ saveCdapResults( - * householdDataManager, projectDirectory ); logCdapResults( - * householdDataManager ); lastComponent = "cdap"; nextComponent = "imtf"; } - * if( runMandatoryTourFreqChoiceModel ){ logImtfResults( - * householdDataManager ); lastComponent = "imtf"; nextComponent = "imtod"; - * } if( runMandatoryTourTimeOfDayChoiceModel || - * runMandatoryTourModeChoiceModel ){ lastComponent = "imtod"; nextComponent - * = "jtf"; } if( runJointTourFrequencyModel ){ logJointModelResults( - * householdDataManager ); lastComponent = "jtf"; nextComponent = "jtl"; } - * if( runJointTourLocationChoiceModel ){ lastComponent = "jtl"; - * nextComponent = "jtod"; } if( runJointTourDepartureTimeAndDurationModel - * || runJointTourModeChoiceModel ){ lastComponent = "jtod"; nextComponent = - * "inmtf"; } if( runIndivNonManTourFrequencyModel ){ lastComponent = - * "inmtf"; nextComponent = "inmtl"; } if( - * runIndivNonManTourLocationChoiceModel ){ lastComponent = "inmtl"; - * nextComponent = "inmtod"; } if( - * runIndivNonManTourDepartureTimeAndDurationModel || - * runIndivNonManTourModeChoiceModel ){ lastComponent = "inmtod"; - * nextComponent = "awf"; } if( runAtWorkSubTourFrequencyModel ){ - * logAtWorkSubtourFreqResults( householdDataManager ); lastComponent = - * "awf"; nextComponent = "awl"; } if( runAtWorkSubtourLocationChoiceModel - * ){ lastComponent = "awl"; nextComponent = "awtod"; } if( - * runAtWorkSubtourDepartureTimeAndDurationModel || - * runAtWorkSubtourModeChoiceModel ){ lastComponent = "awtod"; nextComponent - * = "stf"; } if( runStopFrequencyModel ){ lastComponent = "stf"; - * nextComponent = "stl"; } if( runStopLocationModel ){ lastComponent = - * "stl"; nextComponent = "done"; } // write a disk object fle for the - * householdDataManager, in case we want to restart from the next step. if ( - * hhDiskObjectFile != null && ! lastComponent.equalsIgnoreCase("uwsl") ) { - * logger.info (String.format( - * "writing household disk object file after %s choice model; may take a long time ..." - * , lastComponent) ); String hhFileName = hhDiskObjectFile + "_" + - * nextComponent; - * householdDataManager.createSerializedHhArrayInFileFromObject( hhFileName, - * nextComponent ); logger.info ( - * String.format("finished writing household disk object file = %s.", - * hhFileName) ); } logger.info ( - * "finished with HouseholdChoiceModelRunner." ); } - */ - - public String getProjectDirectoryName() - { - return projectDirectory; - } - - private MatrixDataServer startMatrixServerProcess(String serverAddress, int serverPort) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServer matrixServer = new MatrixDataServer(); - - // bind this concrete object with the cajo library objects for managing RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - public void restartModels(HouseholdDataManagerIf householdDataManager) - { - - // if no filename was specified for the previous shadow price info, - // restartIter == -1, and random counts will be reset to 0. - int restartIter = -1; - String fileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); - if (fileName != null) - { - fileName = projectDirectory + fileName; - int underScoreIndex = fileName.lastIndexOf('_'); - int dotIndex = fileName.lastIndexOf('.'); - restartIter = Integer.parseInt(fileName.substring(underScoreIndex + 1, dotIndex)); - } - - boolean runPreAutoOwnershipModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_PRE_AUTO_OWNERSHIP); - if (runPreAutoOwnershipModel) - { - householdDataManager.resetPreAoRandom(); - } else - { - boolean runUsualWorkSchoolChoiceModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_WORKSCHOOL_CHOICE); - if (runUsualWorkSchoolChoiceModel) - { - householdDataManager.resetUwslRandom(restartIter + 1); - } else - { - boolean runAutoOwnershipModel = ResourceUtil.getBooleanProperty(resourceBundle, - PROPERTIES_RUN_AUTO_OWNERSHIP); - if (runAutoOwnershipModel) - { - // We're resetting the random number sequence used by pre-AO - // for - // the primary AO - householdDataManager.resetPreAoRandom(); - // householdDataManager.resetAoRandom( restartIter+1 ); - } else - { - // boolean runFreeParkingAvailableModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_FREE_PARKING_AVAILABLE); - // if ( runFreeParkingAvailableModel ) { - // householdDataManager.resetFpRandom(); - // } - // else { - boolean runCoordinatedDailyActivityPatternModel = ResourceUtil - .getBooleanProperty(resourceBundle, - PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN); - if (runCoordinatedDailyActivityPatternModel) - { - householdDataManager.resetCdapRandom(); - } else - { - boolean runIndividualMandatoryTourFrequencyModel = ResourceUtil - .getBooleanProperty(resourceBundle, - PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ); - if (runIndividualMandatoryTourFrequencyModel) - { - householdDataManager.resetImtfRandom(); - } else - { - // boolean - // runIndividualMandatoryTourDepartureAndDurationModel - // = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR); - // if ( - // runIndividualMandatoryTourDepartureAndDurationModel - // ) - // { - // householdDataManager.resetImtodRandom(); - // } - // else { - // boolean runJointTourFrequencyModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_JOINT_TOUR_FREQ); - // if ( runJointTourFrequencyModel ) { - // householdDataManager.resetJtfRandom(); - // } - // else { - // boolean runJointTourLocationModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_JOINT_LOCATION_CHOICE); - // if ( runJointTourLocationModel ) { - // householdDataManager.resetJtlRandom(); - // } - // else { - // boolean runJointTourDepartureAndDurationModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR); - // if ( runJointTourDepartureAndDurationModel ) { - // householdDataManager.resetJtodRandom(); - // } - // else { - boolean runIndividualNonMandatoryTourFrequencyModel = ResourceUtil - .getBooleanProperty(resourceBundle, - PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ); - if (runIndividualNonMandatoryTourFrequencyModel) - { - householdDataManager.resetInmtfRandom(); - } - // else { - // boolean - // runIndividualNonMandatoryTourLocationModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE); - // if ( runIndividualNonMandatoryTourLocationModel ) - // { - // householdDataManager.resetInmtlRandom(); - // } - // else { - // boolean - // runIndividualNonMandatoryTourDepartureAndDurationModel - // = ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR); - // if ( - // runIndividualNonMandatoryTourDepartureAndDurationModel - // ) { - // householdDataManager.resetInmtodRandom(); - // } - // else { - boolean runAtWorkSubTourFrequencyModel = ResourceUtil - .getBooleanProperty(resourceBundle, - PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ); - if (runAtWorkSubTourFrequencyModel) - { - householdDataManager.resetAwfRandom(); - } - // else { - // boolean runAtWorkSubtourLocationChoiceModel = - // ResourceUtil.getBooleanProperty( resourceBundle, - // PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE ); - // if ( runAtWorkSubtourLocationChoiceModel ) { - // householdDataManager.resetAwlRandom(); - // } - // else { - // boolean - // runAtWorkSubtourDepartureTimeAndDurationModel - // = ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR); - // if ( - // runAtWorkSubtourDepartureTimeAndDurationModel ) { - // householdDataManager.resetAwtodRandom(); - // } - // else { - boolean runStopFrequencyModel = ResourceUtil.getBooleanProperty( - resourceBundle, PROPERTIES_RUN_STOP_FREQUENCY); - if (runStopFrequencyModel) - { - householdDataManager.resetStfRandom(); - } - // else { - // boolean runStopLocationModel = - // ResourceUtil.getBooleanProperty(resourceBundle, - // PROPERTIES_RUN_STOP_LOCATION); - // if ( runStopLocationModel ) { - // householdDataManager.resetStlRandom(); - // } - // } - // } - // } - // } - // } - // } - // } - // } - // } - // } - // } - // } - } - } - } - } - } - } - - /** - * private void createSerializedObjectInFileFromObject( Object - * objectToSerialize, String serializedObjectFileName, String - * serializedObjectKey ){ try{ DataFile dataFile = new DataFile( - * serializedObjectFileName, 1 ); DataWriter dw = new DataWriter( - * serializedObjectKey ); dw.writeObject( objectToSerialize ); - * dataFile.insertRecord( dw ); dataFile.close(); } - * catch(NotSerializableException e) { logger.error( String.format( - * "NotSerializableException for %s. Trying to create serialized object with key=%s, in filename=%s." - * , objectToSerialize.getClass().getName(), serializedObjectKey, - * serializedObjectFileName ), e ); throw new RuntimeException(); } - * catch(IOException e) { logger.error( String.format( - * "IOException trying to write disk object file=%s, with key=%s for writing." - * , serializedObjectFileName, serializedObjectKey ), e ); throw new - * RuntimeException(); } } - * - * - * private Object createObjectFromSerializedObjectInFile( Object newObject, - * String serializedObjectFileName, String serializedObjectKey ){ try{ - * DataFile dataFile = new DataFile( serializedObjectFileName, "r" ); - * DataReader dr = dataFile.readRecord( serializedObjectKey ); newObject = - * dr.readObject(); dataFile.close(); return newObject; } catch(IOException - * e) { logger.error( String.format( - * "IOException trying to read disk object file=%s, with key=%s.", - * serializedObjectFileName, serializedObjectKey ), e ); throw new - * RuntimeException(); } catch(ClassNotFoundException e) { logger.error( - * String.format - * ("could not instantiate %s object, with key=%s from filename=%s.", - * newObject.getClass().getName(), serializedObjectFileName, - * serializedObjectKey ), e ); throw new RuntimeException(); } } - **/ - /** - * Loops through the households in the HouseholdDataManager, gets the auto - * ownership result for each household, and writes a text file with hhid and - * auto ownership. - * - * @param householdDataManager - * is the object from which the array of household objects can be - * retrieved. - * @param projectDirectory - * is the root directory for the output file named - */ - private void saveAoResults(HouseholdDataManagerIf householdDataManager, - String projectDirectory, boolean preModel) - { - - String aoResultsFileName; - try - { - - aoResultsFileName = resourceBundle.getString(PROPERTIES_RESULTS_AUTO_OWNERSHIP); - - // change the filename property value to include "_pre" at the end - // of the - // name before the extension, if this is a pre-auto ownership run - if (preModel) - { - int dotIndex = aoResultsFileName.indexOf('.'); - if (dotIndex > 0) - { - String beforeDot = aoResultsFileName.substring(0, dotIndex); - String afterDot = aoResultsFileName.substring(dotIndex); - aoResultsFileName = beforeDot + "_pre" + afterDot; - } else - { - aoResultsFileName += "_pre"; - } - } - - } catch (MissingResourceException e) - { - // if filename not specified in properties file, don't need to write - // it. - return; - } - - FileWriter writer; - PrintWriter outStream = null; - if (aoResultsFileName != null) - { - - aoResultsFileName = projectDirectory + aoResultsFileName; - - try - { - writer = new FileWriter(new File(aoResultsFileName)); - outStream = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening AO results file: %s.", - aoResultsFileName)); - throw new RuntimeException(e); - } - - outStream.println("HHID,AO"); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - int hhid = household.getHhId(); - int ao = household.getAutosOwned(); - - outStream.println(String.format("%d,%d", hhid, ao)); - - } - - } - - outStream.close(); - - } - - } - - private void logAoResults(HouseholdDataManagerIf householdDataManager, boolean preModel) - { - - String[] aoRowCategoryLabel = {"0 autos", "1 auto", "2 autos", "3 autos", "4 or more autos"}; - String[] aoColCategoryLabel = {"Non-GQ HHs", "GQ HHs",}; - - logger.info(""); - logger.info(""); - logger.info((preModel ? "Pre-" : "") + "Auto Ownership Model Results"); - String header = String.format("%-16s", "Category"); - for (String label : aoColCategoryLabel) - header += String.format("%15s", label); - header += String.format("%15s", "Total HHs"); - logger.info(header); - - // track the results - int[][] hhsByAutoOwnership = new int[aoRowCategoryLabel.length][aoColCategoryLabel.length]; - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - int ao = household.getAutosOwned(); - if (ao > hhsByAutoOwnership.length - 1) ao = hhsByAutoOwnership.length - 1; - - int gq = household.getIsGroupQuarters(); - hhsByAutoOwnership[ao][gq]++; - - } - - } - - int[] colTotals = new int[aoColCategoryLabel.length]; - for (int i = 0; i < hhsByAutoOwnership.length; i++) - { - - int rowTotal = 0; - String logString = String.format("%-16s", aoRowCategoryLabel[i]); - for (int j = 0; j < hhsByAutoOwnership[i].length; j++) - { - int value = hhsByAutoOwnership[i][j]; - logString += String.format("%15d", value); - rowTotal += value; - colTotals[j] += value; - } - logString += String.format("%15d", rowTotal); - logger.info(logString); - - } - - int total = 0; - String colTotalsString = String.format("%-16s", "Total"); - for (int j = 0; j < colTotals.length; j++) - { - colTotalsString += String.format("%15d", colTotals[j]); - total += colTotals[j]; - } - colTotalsString += String.format("%15d", total); - logger.info(colTotalsString); - - } - - private void logTpResults(HouseholdDataManagerIf householdDataManager) - { - - logger.info(""); - logger.info(""); - logger.info("Transponder Choice Model Results"); - logger.info(String.format("%-16s %20s", "Category", "Num Households")); - logger.info(String.format("%-16s %20s", "----------", "------------------")); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - int numYes = 0; - int numNo = 0; - int numOther = 0; - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - if (household.getTpChoice() + 1 == TransponderChoiceModel.TP_MODEL_NO_ALT) numNo++; - else if (household.getTpChoice() + 1 == TransponderChoiceModel.TP_MODEL_YES_ALT) numYes++; - else numOther++; - - } - - } - - logger.info(String.format("%-16s %20d", "No", numNo)); - logger.info(String.format("%-16s %20d", "Yes", numYes)); - logger.info(String.format("%-16s %20d", "Other", numOther)); - - logger.info(String.format("%-16s %20s", "----------", "------------------")); - logger.info(String.format("%-16s %20d", "Total", (numNo + numYes + numOther))); - - } - - private void logFpResults(HouseholdDataManagerIf householdDataManager) - { - - String[] fpCategoryLabel = {"No Choice Made", "Free Available", "Must Pay", "Reimbursed"}; - - logger.info(""); - logger.info(""); - logger.info("Free Parking Choice Model Results"); - logger.info(String.format("%-16s %20s %20s %20s %20s", "Category", "Workers in area 1", - "Workers in area 2", "Workers in area 3", "Workers in area 4")); - logger.info(String.format("%-16s %20s %20s %20s %20s", "----------", - "------------------", "------------------", "------------------", - "------------------")); - - // track the results by 4 work areas - only workers in area 1 should - // have made choices - int numParkAreas = 4; - int[][] workLocationsByFreeParking; - workLocationsByFreeParking = new int[fpCategoryLabel.length][numParkAreas]; - - // get the correspndence between mgra and park area to associate work - // locations with areas - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - int[] parkAreas = mgraManager.getMgraParkAreas(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - Person[] persons = household.getPersons(); - for (int p = 1; p < persons.length; p++) - { - int workLocation = persons[p].getWorkLocation(); - if (workLocation > 0 - && workLocation != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - int area = parkAreas[workLocation]; - int areaIndex = area - 1; - - int fp = persons[p].getFreeParkingAvailableResult(); - int freqIndex = 0; - if (fp > 0) freqIndex = fp; - - workLocationsByFreeParking[freqIndex][areaIndex]++; - } - } - - } - - } - - int[] total = new int[numParkAreas]; - for (int i = 0; i < workLocationsByFreeParking.length; i++) - { - logger.info(String.format("%-16s %20d %20d %20d %20d", fpCategoryLabel[i], - workLocationsByFreeParking[i][0], workLocationsByFreeParking[i][1], - workLocationsByFreeParking[i][2], workLocationsByFreeParking[i][3])); - for (int j = 0; j < numParkAreas; j++) - total[j] += workLocationsByFreeParking[i][j]; - } - logger.info(String.format("%-16s %20s %20s %20s %20s", "----------", - "------------------", "------------------", "------------------", - "------------------")); - logger.info(String.format("%-16s %20d %20d %20d %20d", "Totals", total[0], total[1], - total[2], total[3])); - - } - - private void logIeResults(HouseholdDataManagerIf householdDataManager) - { - - String[] ieCategoryLabel = {"No IE Trip", "Yes IE Trip"}; - - logger.info(""); - logger.info(""); - logger.info("Internal-External Trip Choice Model Results"); - logger.info(String.format("%-30s %20s %20s %20s", "Person Type", ieCategoryLabel[0], - ieCategoryLabel[1], "Total")); - logger.info(String.format("%-30s %20s %20s %20s", "-------------", "-------------", - "-------------", "---------")); - - // summarize yes/no choice by person type - int[][] personTypeByIeChoice; - personTypeByIeChoice = new int[Person.PERSON_TYPE_NAME_ARRAY.length][2]; - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - Person[] persons = household.getPersons(); - for (int p = 1; p < persons.length; p++) - { - - int ie = persons[p].getInternalExternalTripChoiceResult(); - - // ie = 1 means no, 2 means yes. Get index by subtracting 1. - // person typ indices are 1 based, so subtract 1 for array - // index - try - { - personTypeByIeChoice[persons[p].getPersonTypeNumber() - 1][ie - 1]++; - } catch (ArrayIndexOutOfBoundsException e) - { - logger.error("array index error"); - logger.error("hhid=" + household.getHhId() + ", p=" + p + ", ie=" + ie - + ", personType=" + persons[p].getPersonTypeNumber(), e); - } - } - - } - - } - - int[] totals = new int[2]; - for (int i = 0; i < personTypeByIeChoice.length; i++) - { - int total = personTypeByIeChoice[i][0] + personTypeByIeChoice[i][1]; - logger.info(String.format("%-30s %20d %20d %20d", Person.PERSON_TYPE_NAME_ARRAY[i], - personTypeByIeChoice[i][0], personTypeByIeChoice[i][1], total)); - totals[0] += personTypeByIeChoice[i][0]; - totals[1] += personTypeByIeChoice[i][1]; - } - logger.info(String.format("%-30s %20s %20s %20s", "-------------", "-------------", - "-------------", "---------")); - logger.info(String.format("%-30s %20d %20d %20d", "Totals", totals[0], totals[1], - (totals[0] + totals[1]))); - - } - - /** - * Records the coordinated daily activity pattern model results to the - * logger. A household-level summary simply records each pattern type and a - * person-level summary summarizes the activity choice by person type - * (full-time worker, university student, etc). - * - */ - public void logCdapResults(HouseholdDataManagerIf householdDataManager) - { - - String[] activityNameArray = {Definitions.MANDATORY_PATTERN, - Definitions.NONMANDATORY_PATTERN, Definitions.HOME_PATTERN}; - - getLogReportSummaries(householdDataManager); - - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info("Coordinated Daily Activity Pattern Model Results"); - - // count of activities by person type - logger.info(" "); - logger.info("CDAP Results: Count of activities by person type"); - String firstHeader = "Person type "; - String secondHeader = "----------------------------- "; - for (int i = 0; i < activityNameArray.length; ++i) - { - firstHeader += " " + activityNameArray[i] + " "; - secondHeader += "--------- "; - } - - firstHeader += " Total"; - secondHeader += "---------"; - - logger.info(firstHeader); - logger.info(secondHeader); - - int[] columnTotals = new int[activityNameArray.length]; - - for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) - { - String personType = Person.PERSON_TYPE_NAME_ARRAY[i]; - String stringToLog = String.format("%-30s", personType); - int lineTotal = 0; - - if (cdapByPersonTypeAndActivity.containsKey(personType)) - { - - for (int j = 0; j < activityNameArray.length; ++j) - { - int count = 0; - if (cdapByPersonTypeAndActivity.get(personType).containsKey( - activityNameArray[j])) - { - count = cdapByPersonTypeAndActivity.get(personType).get( - activityNameArray[j]); - } - stringToLog += String.format("%10d", count); - - lineTotal += count; - columnTotals[j] += count; - } // j - - } // if key - - stringToLog += String.format("%10d", lineTotal); - logger.info(stringToLog); - - } // i - - logger.info(secondHeader); - - String stringToLog = String.format("%-30s", "Total"); - int lineTotal = 0; - for (int j = 0; j < activityNameArray.length; ++j) - { - stringToLog += String.format("%10d", columnTotals[j]); - lineTotal += columnTotals[j]; - } // j - - stringToLog += String.format("%10d", lineTotal); - logger.info(stringToLog); - - // count of patterns - logger.info(" "); - logger.info(" "); - logger.info("CDAP Results: Count of patterns"); - logger.info("Pattern Count"); - logger.info("------------------ ---------"); - - // sort the map by hh size first - Set hhSizeKeySet = cdapByHhSizeAndPattern.keySet(); - Integer[] hhSizeKeyArray = new Integer[hhSizeKeySet.size()]; - hhSizeKeySet.toArray(hhSizeKeyArray); - Arrays.sort(hhSizeKeyArray); - - int total = 0; - for (int i = 0; i < hhSizeKeyArray.length; ++i) - { - - // sort the patterns alphabetically - HashMap patternMap = cdapByHhSizeAndPattern.get(hhSizeKeyArray[i]); - Set patternKeySet = patternMap.keySet(); - String[] patternKeyArray = new String[patternKeySet.size()]; - patternKeySet.toArray(patternKeyArray); - Arrays.sort(patternKeyArray); - for (int j = 0; j < patternKeyArray.length; ++j) - { - int count = patternMap.get(patternKeyArray[j]); - total += count; - logger.info(String.format("%-18s%10d", patternKeyArray[j], count)); - } - - } - - logger.info("------------------ ---------"); - logger.info(String.format("%-18s%10d", "Total", total)); - logger.info(" "); - - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info(" "); - logger.info(" "); - - } - - /** - * Logs the results of the individual mandatory tour frequency model. - * - */ - public void logImtfResults(HouseholdDataManagerIf householdDataManager) - { - - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info("Individual Mandatory Tour Frequency Model Results"); - - // count of model results - logger.info(" "); - String firstHeader = "Person type "; - String secondHeader = "----------------------------- "; - - String[] choiceResults = HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_RESULTS; - - // summarize results - HashMap countByPersonType = new HashMap(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Person[] personArray = householdArray[i].getPersons(); - for (int j = 1; j < personArray.length; j++) - { - - // only summarize persons with mandatory pattern - String personActivity = personArray[j].getCdapActivity(); - if (personActivity != null - && personArray[j].getCdapActivity().equalsIgnoreCase("M")) - { - - String personTypeString = personArray[j].getPersonType(); - int choice = personArray[j].getImtfChoice(); - - if (choice == 0) - { - - // there are 5 IMTF alts, so it's the offset for the - // extra at home categories - if (personArray[j].getPersonEmploymentCategoryIndex() < Person.EmployStatus.NOT_EMPLOYED - .ordinal() - && personArray[j].getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) choice = 5 + 1; - else if (personArray[j].getPersonEmploymentCategoryIndex() < Person.EmployStatus.NOT_EMPLOYED - .ordinal() - && personArray[j].getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) choice = 5 + 2; - else if (personArray[j].getPersonIsStudent() < Person.EmployStatus.NOT_EMPLOYED - .ordinal() - && personArray[j].getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) choice = 5 + 3; - else if (personArray[j].getPersonIsStudent() < Person.EmployStatus.NOT_EMPLOYED - .ordinal() - && personArray[j].getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) - choice = 5 + 4; - - } - - // count the results - if (countByPersonType.containsKey(personTypeString)) - { - - int[] counterArray = countByPersonType.get(personTypeString); - counterArray[choice - 1]++; - countByPersonType.put(personTypeString, counterArray); - - } else - { - - int[] counterArray = new int[choiceResults.length]; - counterArray[choice - 1]++; - countByPersonType.put(personTypeString, counterArray); - - } - } - - } - - } - - } - - for (int i = 0; i < choiceResults.length; ++i) - { - firstHeader += String.format("%12s", choiceResults[i]); - secondHeader += "----------- "; - } - - firstHeader += String.format("%12s", "Total"); - secondHeader += "-----------"; - - logger.info(firstHeader); - logger.info(secondHeader); - - int[] columnTotals = new int[choiceResults.length]; - - int lineTotal = 0; - for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) - { - String personTypeString = Person.PERSON_TYPE_NAME_ARRAY[i]; - String stringToLog = String.format("%-30s", personTypeString); - - if (countByPersonType.containsKey(personTypeString)) - { - - lineTotal = 0; - int[] countArray = countByPersonType.get(personTypeString); - for (int j = 0; j < choiceResults.length; ++j) - { - stringToLog += String.format("%12d", countArray[j]); - columnTotals[j] += countArray[j]; - lineTotal += countArray[j]; - } // j - } else - { - // if key - // log zeros - lineTotal = 0; - for (int j = 0; j < choiceResults.length; ++j) - { - stringToLog += String.format("%12d", 0); - } - } - - stringToLog += String.format("%12d", lineTotal); - - logger.info(stringToLog); - - } // i - - String stringToLog = String.format("%-30s", "Total"); - lineTotal = 0; - for (int j = 0; j < choiceResults.length; ++j) - { - stringToLog += String.format("%12d", columnTotals[j]); - lineTotal += columnTotals[j]; - } // j - - logger.info(secondHeader); - stringToLog += String.format("%12d", lineTotal); - logger.info(stringToLog); - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info(" "); - logger.info(" "); - - } - - private void logJointModelResults(HouseholdDataManagerIf householdDataManager, - CtrampDmuFactoryIf dmuFactory) - { - - String uecFileDirectory = ResourceUtil.getProperty(resourceBundle, PROPERTIES_UEC_PATH); - String uecFileName = ResourceUtil.getProperty(resourceBundle, - JointTourModels.UEC_FILE_PROPERTIES_TARGET); - uecFileName = uecFileDirectory + uecFileName; - - int dataSheet = ResourceUtil.getIntegerProperty(resourceBundle, - JointTourModels.UEC_DATA_PAGE_TARGET); - int freqCompSheet = ResourceUtil.getIntegerProperty(resourceBundle, - JointTourModels.UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE); - - // get the alternative names - JointTourModelsDMU dmuObject = dmuFactory.getJointTourModelsDMU(); - ChoiceModelApplication jointTourFrequencyModel = new ChoiceModelApplication(uecFileName, - freqCompSheet, dataSheet, - ResourceUtil.changeResourceBundleIntoHashMap(resourceBundle), - (VariableTable) dmuObject); - String[] altLabels = jointTourFrequencyModel.getAlternativeNames(); - - // this is the first index in the summary array for choices made by - // eligible households - int[] jointTourChoiceFreq = new int[altLabels.length + 1]; - - TreeMap partySizeFreq = new TreeMap(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Tour[] jt = householdArray[i].getJointTourArray(); - int jtfAlt = householdArray[i].getJointTourFreqChosenAlt(); - - if (jt == null) - { - - if (jtfAlt > 0) - { - logger.error(String - .format("HHID=%d, joint tour array is null, but a valid alternative=%d is recorded for the household.", - householdArray[i].getHhId(), jtfAlt)); - throw new RuntimeException(); - } - - jointTourChoiceFreq[0]++; - - } else - { - - if (jtfAlt < 1) - { - logger.error(String - .format("HHID=%d, joint tour array is not null, but an invalid alternative=%d is recorded for the household.", - householdArray[i].getHhId(), jtfAlt)); - throw new RuntimeException(); - } - - jointTourChoiceFreq[jtfAlt]++; - - // determine party size frequency for joint tours generated - Person[] persons = householdArray[i].getPersons(); - for (int j = 0; j < jt.length; j++) - { - - int compAlt = jt[j].getJointTourComposition(); - - // determine number of children and adults in tour - int adults = 0; - int children = 0; - int[] participants = jt[j].getPersonNumArray(); - for (int k = 0; k < participants.length; k++) - { - int index = participants[k]; - Person person = persons[index]; - if (person.getPersonIsAdult() == 1) adults++; - else children++; - } - - // create a key to use for a frequency map for - // "JointTourPurpose_Composition_NumAdults_NumChildren" - String key = String.format("%s_%d_%d_%d", jt[j].getTourPurpose(), compAlt, - adults, children); - - int value = 0; - if (partySizeFreq.containsKey(key)) value = partySizeFreq.get(key); - partySizeFreq.put(key, ++value); - - } - - } - - } - - } - - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info("Joint Tour Frequency and Joint Tour Composition Model Results"); - - logger.info(" "); - logger.info("Frequency Table of Households by Joint Tour Frequency Choice"); - logger.info(String.format("%-5s %-26s %12s", "Alt", "Alt Name", "Households")); - - int rowTotal = jointTourChoiceFreq[0]; - logger.info(String.format("%-5d %-26s %12d", 0, "None", jointTourChoiceFreq[0])); - for (int i = 1; i <= altLabels.length; i++) - { - logger.info(String.format("%-5d %-26s %12d", i, altLabels[i - 1], - jointTourChoiceFreq[i])); - rowTotal += jointTourChoiceFreq[i]; - } - logger.info(String.format("%-34s %12d", "Total Households", rowTotal)); - - logger.info(" "); - logger.info(" "); - logger.info(" "); - - logger.info("Frequency Table of Joint Tours by All Parties Generated"); - logger.info(String.format("%-5s %-20s %-15s %10s %10s %10s", "N", "Purpose", - "Type", "Adults", "Children", "Freq")); - - int count = 1; - for (String key : partySizeFreq.keySet()) - { - - int start = 0; - int end = 0; - int compIndex = 0; - int adults = 0; - int children = 0; - String indexString = ""; - String purpose = ""; - - start = 0; - end = key.indexOf('_', start); - purpose = key.substring(start, end); - - start = end + 1; - end = key.indexOf('_', start); - indexString = key.substring(start, end); - compIndex = Integer.parseInt(indexString); - - start = end + 1; - end = key.indexOf('_', start); - indexString = key.substring(start, end); - adults = Integer.parseInt(indexString); - - start = end + 1; - indexString = key.substring(start); - children = Integer.parseInt(indexString); - - logger.info(String.format("%-5d %-20s %-15s %10d %10d %10d", count++, - purpose, JointTourModels.JOINT_TOUR_COMPOSITION_NAMES[compIndex], adults, - children, partySizeFreq.get(key))); - } - - logger.info(" "); - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info(" "); - - } - - private void getLogReportSummaries(HouseholdDataManagerIf householdDataManager) - { - - // summary collections - cdapByHhSizeAndPattern = new HashMap>(); - cdapByPersonTypeAndActivity = new HashMap>(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] partialHhArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (Household hhObject : partialHhArray) - { - - // get the household's activity pattern choice - String pattern = hhObject.getCoordinatedDailyActivityPattern(); - if (pattern == null) continue; - - Person[] personArray = hhObject.getPersons(); - for (int j = 1; j < personArray.length; j++) - { - - // get person's activity string - String activityString = personArray[j].getCdapActivity(); - - // get the person type to simmarize results by - String personTypeString = personArray[j].getPersonType(); - - // check if the person type is in the map - if (cdapByPersonTypeAndActivity.containsKey(personTypeString)) - { - - HashMap activityCountMap = cdapByPersonTypeAndActivity - .get(personTypeString); - - // check if the activity is in the activity map - int currentCount = 1; - if (activityCountMap.containsKey(activityString)) - currentCount = activityCountMap.get(activityString) + 1; - - activityCountMap.put(activityString, currentCount); - cdapByPersonTypeAndActivity.put(personTypeString, activityCountMap); - - } else - { - - HashMap activityCountMap = new HashMap(); - activityCountMap.put(activityString, 1); - cdapByPersonTypeAndActivity.put(personTypeString, activityCountMap); - - } // is personType in map if - - } // j (person loop) - - // count each type of pattern string by hhSize - if ((!cdapByHhSizeAndPattern.isEmpty()) - && cdapByHhSizeAndPattern.containsKey(pattern.length())) - { - - HashMap patternCountMap = cdapByHhSizeAndPattern.get(pattern - .length()); - - int currentCount = 1; - if (patternCountMap.containsKey(pattern)) - currentCount = patternCountMap.get(pattern) + 1; - patternCountMap.put(pattern, currentCount); - cdapByHhSizeAndPattern.put(pattern.length(), patternCountMap); - - } else - { - - HashMap patternCountMap = new HashMap(); - patternCountMap.put(pattern, 1); - cdapByHhSizeAndPattern.put(pattern.length(), patternCountMap); - - } // is personType in map if - - } - - } - - } - - /** - * Loops through the households in the HouseholdDataManager, gets the - * coordinated daily activity pattern for each person in the household, and - * writes a text file with hhid, personid, persnum, and activity pattern. - * - * @param householdDataManager - */ - public void saveCdapResults(HouseholdDataManagerIf householdDataManager, String projectDirectory) - { - - String cdapResultsFileName; - try - { - cdapResultsFileName = resourceBundle.getString(PROPERTIES_RESULTS_CDAP); - } catch (MissingResourceException e) - { - // if filename not specified in properties file, don't need to write - // it. - return; - } - - FileWriter writer; - PrintWriter outStream = null; - if (cdapResultsFileName != null) - { - - cdapResultsFileName = projectDirectory + cdapResultsFileName; - - try - { - writer = new FileWriter(new File(cdapResultsFileName)); - outStream = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening CDAP results file: %s.", - cdapResultsFileName)); - throw new RuntimeException(e); - } - - outStream.println("HHID,PersonID,PersonNum,PersonType,ActivityString"); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - int hhid = household.getHhId(); - - // get the pattern for each person - Person[] personArray = household.getPersons(); - for (int j = 1; j < personArray.length; j++) - { - - Person person = personArray[j]; - - int persId = person.getPersonId(); - int persNum = person.getPersonNum(); - int persType = person.getPersonTypeNumber(); - String activityString = person.getCdapActivity(); - - outStream.println(String.format("%d,%d,%d,%d,%s", hhid, persId, persNum, - persType, activityString)); - - } // j (person loop) - - } - - } - - outStream.close(); - - } - - } - - /** - * Logs the results of the model. - * - */ - public void logAtWorkSubtourFreqResults(HouseholdDataManagerIf householdDataManager) - { - - String[] alternativeNames = modelStructure.getAwfAltLabels(); - HashMap awfByPersonType = new HashMap(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - for (int i = 0; i < householdArray.length; ++i) - { - - // get this household's person array - Person[] personArray = householdArray[i].getPersons(); - - // loop through the person array (1-based) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - // loop through the work tours for this person - ArrayList tourList = person.getListOfWorkTours(); - if (tourList == null || tourList.size() == 0) continue; - - // count the results by person type - String personTypeString = person.getPersonType(); - - for (Tour workTour : tourList) - { - - int choice = 0; - if (person.getListOfAtWorkSubtours().size() == 0) choice = 1; - else - { - choice = workTour.getSubtourFreqChoice(); - if (choice == 0) choice++; - } - - int dummy = 0; - if (person.getPersonTypeNumber() == 7) - { - dummy = 1; - } - - // count the results by person type - if (awfByPersonType.containsKey(personTypeString)) - { - int[] counterArray = awfByPersonType.get(personTypeString); - counterArray[choice - 1]++; - awfByPersonType.put(personTypeString, counterArray); - - } else - { - int[] counterArray = new int[alternativeNames.length]; - counterArray[choice - 1]++; - awfByPersonType.put(personTypeString, counterArray); - } - - } - - } - - } - - } - - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info("At-Work Subtour Frequency Model Results"); - - // count of model results - logger.info(" "); - String firstHeader = "Person type "; - String secondHeader = "--------------------------- "; - - for (int i = 0; i < alternativeNames.length; ++i) - { - firstHeader += String.format("%16s", alternativeNames[i]); - secondHeader += "------------ "; - } - - firstHeader += String.format("%16s", "Total"); - secondHeader += "------------"; - - logger.info(firstHeader); - logger.info(secondHeader); - - int[] columnTotals = new int[alternativeNames.length]; - - int lineTotal = 0; - for (int i = 0; i < Person.PERSON_TYPE_NAME_ARRAY.length; ++i) - { - String personTypeString = Person.PERSON_TYPE_NAME_ARRAY[i]; - String stringToLog = String.format("%-28s", personTypeString); - - if (awfByPersonType.containsKey(personTypeString)) - { - - lineTotal = 0; - int[] countArray = awfByPersonType.get(personTypeString); - for (int j = 0; j < alternativeNames.length; ++j) - { - stringToLog += String.format("%16d", countArray[j]); - columnTotals[j] += countArray[j]; - lineTotal += countArray[j]; - } // j - - } else - { - // if key - // log zeros - lineTotal = 0; - for (int j = 0; j < alternativeNames.length; ++j) - { - stringToLog += String.format("%16d", 0); - } - } - - stringToLog += String.format("%16d", lineTotal); - - logger.info(stringToLog); - - } // i - - String stringToLog = String.format("%-28s", "Total"); - lineTotal = 0; - for (int j = 0; j < alternativeNames.length; ++j) - { - stringToLog += String.format("%16d", columnTotals[j]); - lineTotal += columnTotals[j]; - } // j - - logger.info(secondHeader); - stringToLog += String.format("%16d", lineTotal); - logger.info(stringToLog); - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info(" "); - logger.info(" "); - - } - - /** - * Logs the results of the individual tour stop frequency model. - * - */ - public void logIndivStfResults(HouseholdDataManagerIf householdDataManager) - { - - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info("Individual Tour Stop Frequency Model Results"); - - // count of model results - logger.info(" "); - String firstHeader = "Tour Purpose "; - String secondHeader = "--------------- "; - - int[] obStopsAlt = StopFrequencyDMU.NUM_OB_STOPS_FOR_ALT; - int[] ibStopsAlt = StopFrequencyDMU.NUM_IB_STOPS_FOR_ALT; - - // 10 purposes - int[][] chosen = new int[obStopsAlt.length][11]; - HashMap indexPurposeMap = modelStructure.getIndexPrimaryPurposeNameMap(); - HashMap purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Person[] personArray = householdArray[i].getPersons(); - for (int j = 1; j < personArray.length; j++) - { - - List tourList = new ArrayList(); - - // apply stop frequency for all person tours - tourList.addAll(personArray[j].getListOfWorkTours()); - tourList.addAll(personArray[j].getListOfSchoolTours()); - tourList.addAll(personArray[j].getListOfIndividualNonMandatoryTours()); - tourList.addAll(personArray[j].getListOfAtWorkSubtours()); - - for (Tour t : tourList) - { - - int index = t.getTourPrimaryPurposeIndex(); - int choice = t.getStopFreqChoice(); - chosen[choice][index]++; - - } - - } - - } - - } - - for (int i = 1; i < chosen[1].length; ++i) - { - firstHeader += String.format("%18s", indexPurposeMap.get(i)); - secondHeader += " --------------- "; - } - - firstHeader += String.format("%18s", "Total"); - secondHeader += " --------------- "; - - logger.info(firstHeader); - logger.info(secondHeader); - - int[] columnTotals = new int[chosen[1].length]; - - int lineTotal = 0; - for (int i = 1; i < chosen.length; ++i) - { - String stringToLog = String.format("%d out, %d in ", obStopsAlt[i], ibStopsAlt[i]); - - lineTotal = 0; - int[] countArray = chosen[i]; - for (int j = 1; j < countArray.length; ++j) - { - stringToLog += String.format("%18d", countArray[j]); - columnTotals[j] += countArray[j]; - lineTotal += countArray[j]; - } // j - - stringToLog += String.format("%18d", lineTotal); - - logger.info(stringToLog); - - } // i - - String stringToLog = String.format("%-17s", "Total"); - lineTotal = 0; - for (int j = 1; j < chosen[1].length; ++j) - { - stringToLog += String.format("%18d", columnTotals[j]); - lineTotal += columnTotals[j]; - } // j - - logger.info(secondHeader); - stringToLog += String.format("%18d", lineTotal); - logger.info(stringToLog); - logger.info(" "); - logger.info("~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~"); - logger.info(" "); - logger.info(" "); - - } - - private void createSerializedObjectInFileFromObject(Object objectToSerialize, - String serializedObjectFileName, String serializedObjectKey) - { - try - { - DataFile dataFile = new DataFile(serializedObjectFileName, 1); - DataWriter dw = new DataWriter(serializedObjectKey); - dw.writeObject(objectToSerialize); - dataFile.insertRecord(dw); - dataFile.close(); - } catch (NotSerializableException e) - { - logger.error( - String.format( - "NotSerializableException for %s. Trying to create serialized object with key=%s, in filename=%s.", - objectToSerialize.getClass().getName(), serializedObjectKey, - serializedObjectFileName), e); - throw new RuntimeException(); - } catch (IOException e) - { - logger.error(String.format( - "IOException trying to write disk object file=%s, with key=%s for writing.", - serializedObjectFileName, serializedObjectKey), e); - throw new RuntimeException(); - } - } - - private Object createObjectFromSerializedObjectInFile(Object newObject, - String serializedObjectFileName, String serializedObjectKey) - { - try - { - DataFile dataFile = new DataFile(serializedObjectFileName, "r"); - DataReader dr = dataFile.readRecord(serializedObjectKey); - newObject = dr.readObject(); - dataFile.close(); - return newObject; - } catch (IOException e) - { - logger.error(String.format( - "IOException trying to read disk object file=%s, with key=%s.", - serializedObjectFileName, serializedObjectKey), e); - throw new RuntimeException(); - } catch (ClassNotFoundException e) - { - logger.error(String.format( - "could not instantiate %s object, with key=%s from filename=%s.", newObject - .getClass().getName(), serializedObjectFileName, serializedObjectKey), - e); - throw new RuntimeException(); - } - } - - private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) - { - - ArrayList startEndIndexList = new ArrayList(); - - int startIndex = 0; - int endIndex = 0; - - while (endIndex < numberOfHouseholds - 1) - { - endIndex = startIndex + NUM_WRITE_PACKETS - 1; - if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) - endIndex = numberOfHouseholds - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += NUM_WRITE_PACKETS; - } - - return startEndIndexList; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java deleted file mode 100644 index 95f7801..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/CtrampDmuFactoryIf.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.sandag.abm.ctramp; - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 9, 2008 Time: 3:13:17 PM To - * change this template use File | Settings | File Templates. - */ -public interface CtrampDmuFactoryIf -{ - - AutoOwnershipChoiceDMU getAutoOwnershipDMU(); - - ParkingProvisionChoiceDMU getFreeParkingChoiceDMU(); - - TelecommuteDMU getTelecommuteDMU(); - - TransponderChoiceDMU getTransponderChoiceDMU(); - - InternalExternalTripChoiceDMU getInternalExternalTripChoiceDMU(); - - CoordinatedDailyActivityPatternDMU getCoordinatedDailyActivityPatternDMU(); - - DcSoaDMU getDcSoaDMU(); - - DestChoiceDMU getDestChoiceDMU(); - - DestChoiceTwoStageModelDMU getDestChoiceSoaTwoStageDMU(); - - DestChoiceTwoStageSoaTazDistanceUtilityDMU getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - TourModeChoiceDMU getModeChoiceDMU(); - - IndividualMandatoryTourFrequencyDMU getIndividualMandatoryTourFrequencyDMU(); - - TourDepartureTimeAndDurationDMU getTourDepartureTimeAndDurationDMU(); - - AtWorkSubtourFrequencyDMU getAtWorkSubtourFrequencyDMU(); - - JointTourModelsDMU getJointTourModelsDMU(); - - IndividualNonMandatoryTourFrequencyDMU getIndividualNonMandatoryTourFrequencyDMU(); - - StopFrequencyDMU getStopFrequencyDMU(); - - StopLocationDMU getStopLocationDMU(); - - TripModeChoiceDMU getTripModeChoiceDMU(); - - ParkingChoiceDMU getParkingChoiceDMU(); - - MicromobilityChoiceDMU getMicromobilityChoiceDMU(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java deleted file mode 100644 index d2fc746..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DAOException.java +++ /dev/null @@ -1,23 +0,0 @@ -package org.sandag.abm.ctramp; - -public class DAOException - extends RuntimeException -{ - static final long serialVersionUID = -1881205326938716446L; - - public DAOException(String message) - { - super(message); - } - - public DAOException(Throwable cause) - { - super(cause); - } - - public DAOException(String message, Throwable cause) - { - super(message, cause); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java deleted file mode 100644 index 769f85d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DcSoaDMU.java +++ /dev/null @@ -1,215 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class DcSoaDMU - implements SoaDMU, Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(DcSoaDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Person person; - protected Tour tour; - - protected IndexValues dmuIndex = null; - protected String dmuLabel = "Origin Location"; - - protected double[] dcSize; - protected double[] distance; - - protected BuildAccessibilities aggAcc; - - public DcSoaDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug DC SOA UEC"); - } - - } - - public void setAggAcc(BuildAccessibilities myAggAcc) - { - aggAcc = myAggAcc; - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setPersonObject(Person personObject) - { - person = personObject; - } - - public void setTourObject(Tour tourObject) - { - tour = tourObject; - } - - public void setDestChoiceSize(double[] dcSize) - { - this.dcSize = dcSize; - } - - public void setDestDistance(double[] distance) - { - this.distance = distance; - } - - public double[] getDestDistance() - { - return distance; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public Household getHouseholdObject() - { - return hh; - } - - public int getTourPurposeIsEscort() - { - return tour.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME) ? 1 : 0; - } - - public int getNumPreschool() - { - return hh.getNumPreschool(); - } - - public int getNumGradeSchoolStudents() - { - return hh.getNumGradeSchoolStudents(); - } - - public int getNumHighSchoolStudents() - { - return hh.getNumHighSchoolStudents(); - } - - protected double getLnDcSize(int alt) - { - - double size = dcSize[alt]; - - double logSize = 0.0; - logSize = Math.log(size + 1); - - return logSize; - - } - - protected double getDcSizeAlt(int alt) - { - return dcSize[alt]; - } - - protected double getHouseholdsDestAlt(int mgra) - { - return aggAcc.getMgraHouseholds(mgra); - } - - protected double getGradeSchoolEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraGradeSchoolEnrollment(mgra); - } - - protected double getHighSchoolEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraHighSchoolEnrollment(mgra); - } - - public int getGradeSchoolDistrictDestAlt(int mgra) - { - return aggAcc.getMgraGradeSchoolDistrict(mgra); - } - - public int getHomeMgraGradeSchoolDistrict() - { - return aggAcc.getMgraGradeSchoolDistrict(hh.getHhMgra()); - } - - public double getHighSchoolDistrictDestAlt(int mgra) - { - return aggAcc.getMgraHighSchoolDistrict(mgra); - } - - public double getHomeMgraHighSchoolDistrict() - { - return aggAcc.getMgraHighSchoolDistrict(hh.getHhMgra()); - } - - public double getOriginToMgraDistanceAlt(int alt) - { - return distance[alt]; - } - - public double getUniversityEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraUniversityEnrollment(mgra); - } - - public String getDmuLabel() - { - return dmuLabel; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java deleted file mode 100644 index 496426e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Definitions.java +++ /dev/null @@ -1,22 +0,0 @@ -package org.sandag.abm.ctramp; - -/** - * This class holds definitions that are inherant in CT-RAMP based models - * including: person types tour category types purposes activity types - * - * @author Jim - * - */ -public final class Definitions -{ - - // Coordinated daily activity pattern type definitions - public static final String MANDATORY_PATTERN = "M"; - public static final String NONMANDATORY_PATTERN = "N"; - public static final String HOME_PATTERN = "H"; - - private Definitions() - { - // Not implemented in utility classes - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java deleted file mode 100644 index be69e84..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceDMU.java +++ /dev/null @@ -1,394 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public abstract class DestChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(DestChoiceDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Person person; - protected Tour tour; - protected IndexValues dmuIndex = null; - - protected double workAccessibility; - protected double nonMandatoryAccessibility; - - protected double[] homeMgraNonMandatoryAccessibilityArray; - protected double[] homeMgraTotalEmploymentAccessibilityArray; - protected double[] homeMgraSizeArray; - protected double[] homeMgraDistanceArray; - protected double[] modeChoiceLogsums; - protected double[] dcSoaCorrections; - - protected int toursLeftCount; - - protected ModelStructure modelStructure; - protected MgraDataManager mgraManager; - protected BuildAccessibilities aggAcc; - protected AccessibilitiesTable accTable; - - public DestChoiceDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - initDmuObject(); - } - - public abstract void setMcLogsum(int mgra, double logsum); - - private void initDmuObject() - { - - dmuIndex = new IndexValues(); - - // create default objects - some choice models use these as place - // holders for values - person = new Person(null, -1, modelStructure); - hh = new Household(modelStructure); - - mgraManager = MgraDataManager.getInstance(); - - int maxMgra = mgraManager.getMaxMgra(); - - modeChoiceLogsums = new double[maxMgra + 1]; - dcSoaCorrections = new double[maxMgra + 1]; - - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setPersonObject(Person personObject) - { - person = personObject; - } - - public void setTourObject(Tour tour) - { - this.tour = tour; - } - - public void setAggAcc(BuildAccessibilities aggAcc) - { - this.aggAcc = aggAcc; - } - - public void setAccTable(AccessibilitiesTable myAccTable) - { - accTable = myAccTable; - } - - public void setDestChoiceSize(double[] homeMgraSizeArray) - { - this.homeMgraSizeArray = homeMgraSizeArray; - } - - public void setDestChoiceDistance(double[] homeMgraDistanceArray) - { - this.homeMgraDistanceArray = homeMgraDistanceArray; - } - - public void setDcSoaCorrections(int mgra, double correction) - { - dcSoaCorrections[mgra] = correction; - } - - public void setNonMandatoryAccessibility(double nonMandatoryAccessibility) - { - this.nonMandatoryAccessibility = nonMandatoryAccessibility; - } - - public void setToursLeftCount(int count) - { - toursLeftCount = count; - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug DC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public Household getHouseholdObject() - { - return hh; - } - - public Person getPersonObject() - { - return person; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - protected int getToursLeftCount() - { - return toursLeftCount; - } - - protected int getMaxContinuousAvailableWindow() - { - - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return hh - .getMaxJointTimeWindow(tour); - else return person.getMaximumContinuousAvailableWindow(); - } - - protected double getDcSoaCorrectionsAlt(int alt) - { - return dcSoaCorrections[alt]; - } - - protected double getMcLogsumDestAlt(int mgra) - { - return modeChoiceLogsums[mgra]; - } - - protected double getPopulationDestAlt(int mgra) - { - return aggAcc.getMgraPopulation(mgra); - } - - protected double getHouseholdsDestAlt(int mgra) - { - return aggAcc.getMgraHouseholds(mgra); - } - - protected double getGradeSchoolEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraGradeSchoolEnrollment(mgra); - } - - protected double getHighSchoolEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraHighSchoolEnrollment(mgra); - } - - protected double getUniversityEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraUniversityEnrollment(mgra); - } - - protected double getOtherCollegeEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraOtherCollegeEnrollment(mgra); - } - - protected double getAdultSchoolEnrollmentDestAlt(int mgra) - { - return aggAcc.getMgraAdultSchoolEnrollment(mgra); - } - - protected int getIncome() - { - return hh.getIncomeCategory(); - } - - protected int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - protected int getAutos() - { - return hh.getAutosOwned(); - } - - protected int getWorkers() - { - return hh.getWorkers(); - } - - protected int getNumberOfNonWorkingAdults() - { - return hh.getNumberOfNonWorkingAdults(); - } - - protected int getNumPreschool() - { - return hh.getNumPreschool(); - } - - public int getNumGradeSchoolStudents() - { - return hh.getNumGradeSchoolStudents(); - } - - public int getNumHighSchoolStudents() - { - return hh.getNumHighSchoolStudents(); - } - - protected int getNumChildrenUnder16() - { - return hh.getNumChildrenUnder16(); - } - - protected int getNumChildrenUnder19() - { - return hh.getNumChildrenUnder19(); - } - - protected int getAge() - { - return person.getAge(); - } - - protected int getFemaleWorker() - { - if (person.getPersonIsFemale() == 1) return 1; - else return 0; - } - - protected int getFemale() - { - if (person.getPersonIsFemale() == 1) return 1; - else return 0; - } - - protected int getFullTimeWorker() - { - if (person.getPersonIsFullTimeWorker() == 1) return 1; - else return 0; - } - - protected int getTypicalUniversityStudent() - { - return person.getPersonIsTypicalUniversityStudent(); - } - - protected int getPersonType() - { - return person.getPersonTypeNumber(); - } - - protected int getPersonHasBachelors() - { - return person.getHasBachelors(); - } - - protected int getPersonIsWorker() - { - return person.getPersonIsWorker(); - } - - protected int getWorkTaz() - { - return person.getWorkLocation(); - } - - protected int getWorkTourModeIsSOV() - { - boolean tourModeIsSov = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); - if (tourModeIsSov) return 1; - else return 0; - } - - protected int getTourIsJoint() - { - return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 - : 0; - } - - protected double getTotEmpAccessibilityAlt(int alt) - { - return homeMgraTotalEmploymentAccessibilityArray[alt]; - } - - protected double getNonMandatoryAccessibilityAlt(int alt) - { - return accTable.getAggregateAccessibility("nonmotor", alt); - } - - protected double getOpSovDistanceAlt(int alt) - { - return homeMgraDistanceArray[alt]; - } - - protected double getLnDcSizeAlt(int alt) - { - return Math.log(homeMgraSizeArray[alt] + 1); - } - - protected double getDcSizeAlt(int alt) - { - return homeMgraSizeArray[alt]; - } - - protected void setWorkAccessibility(double accessibility) - { - workAccessibility = accessibility; - } - - protected double getWorkAccessibility() - { - return workAccessibility; - } - - protected double getNonMandatoryAccessibility() - { - return nonMandatoryAccessibility; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java deleted file mode 100644 index bd4c33f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceModelManager.java +++ /dev/null @@ -1,1082 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.Serializable; -import java.util.Date; -import java.util.HashMap; -import java.util.LinkedList; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; - -public class DestChoiceModelManager - implements Serializable -{ - - private static String PROPERTIES_WORK_DC_SOA_UEC_FILE = "work.soa.uec.file"; - private static String PROPERTIES_WORK_DC_SOA_UEC_MODEL_PAGE = "work.soa.uec.model"; - private static String PROPERTIES_WORK_DC_SOA_UEC_DATA_PAGE = "work.soa.uec.data"; - - private static String PROPERTIES_UNIV_DC_SOA_UEC_FILE = "univ.soa.uec.file"; - private static String PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE = "univ.soa.uec.model"; - private static String PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE = "univ.soa.uec.data"; - - private static String PROPERTIES_HS_DC_SOA_UEC_FILE = "hs.soa.uec.file"; - private static String PROPERTIES_HS_DC_SOA_UEC_MODEL_PAGE = "hs.soa.uec.model"; - private static String PROPERTIES_HS_DC_SOA_UEC_DATA_PAGE = "hs.soa.uec.data"; - - private static String PROPERTIES_GS_DC_SOA_UEC_FILE = "gs.soa.uec.file"; - private static String PROPERTIES_GS_DC_SOA_UEC_MODEL_PAGE = "gs.soa.uec.model"; - private static String PROPERTIES_GS_DC_SOA_UEC_DATA_PAGE = "gs.soa.uec.data"; - - private static String PROPERTIES_PS_DC_SOA_UEC_FILE = "ps.soa.uec.file"; - private static String PROPERTIES_PS_DC_SOA_UEC_MODEL_PAGE = "ps.soa.uec.model"; - private static String PROPERTIES_PS_DC_SOA_UEC_DATA_PAGE = "ps.soa.uec.data"; - - private static final int PRESCHOOL_ALT_INDEX = BuildAccessibilities.PRESCHOOL_ALT_INDEX; - private static final int GRADE_SCHOOL_ALT_INDEX = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; - private static final int HIGH_SCHOOL_ALT_INDEX = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; - private static final int UNIV_TYPICAL_ALT_INDEX = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; - private static final int UNIV_NONTYPICAL_ALT_INDEX = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; - private static final int NUMBER_OF_SCHOOL_SEGMENT_TYPES = 5; - - private static transient Logger logger = Logger.getLogger(DestChoiceModelManager.class); - - private static DestChoiceModelManager objInstance = null; - - private LinkedList modelQueueWorkLoc = null; - private LinkedList modelQueueSchoolLoc = null; - private LinkedList modelQueueWork = null; - private LinkedList modelQueueSchool = null; - - private MgraDataManager mgraManager; - private TazDataManager tdm; - - private int maxTaz; - - private BuildAccessibilities aggAcc; - - private HashMap propertyMap; - private String dcUecFileName; - private String soaUecFileName; - private int soaSampleSize; - private String modeChoiceUecFileName; - private CtrampDmuFactoryIf dmuFactory; - - private int modelIndexWork; - private int modelIndexSchool; - private int currentIteration; - - private DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu; - private DestChoiceTwoStageSoaProbabilitiesCalculator workLocSoaDistProbsObject; - private DestChoiceTwoStageSoaProbabilitiesCalculator psLocSoaDistProbsObject; - private DestChoiceTwoStageSoaProbabilitiesCalculator gsLocSoaDistProbsObject; - private DestChoiceTwoStageSoaProbabilitiesCalculator hsLocSoaDistProbsObject; - private DestChoiceTwoStageSoaProbabilitiesCalculator univLocSoaDistProbsObject; - - // the first dimension on these arrays is work location segments (worker - // occupations) - private double[][][] workSizeProbs; - private double[][][] workTazDistProbs; - - // the first dimension on these arrays is school location segment type (ps, - // gs, hs, univTypical, univNonTypical) - private double[][][] schoolSizeProbs; - private double[][][] schoolTazDistProbs; - - private AutoTazSkimsCalculator tazDistanceCalculator; - - private boolean managerIsSetup = false; - - private int completedHouseholdsWork; - private int completedHouseholdsSchool; - private boolean logResults=false; - - private DestChoiceModelManager() - { - } - - public static synchronized DestChoiceModelManager getInstance() - { - // logger.info( - // "beginning of DestChoiceModelManager.getInstance() - objInstance address = " - // + objInstance ); - if (objInstance == null) - { - objInstance = new DestChoiceModelManager(); - // logger.info( - // "after new DestChoiceModelManager() - objInstance address = " + - // objInstance ); - return objInstance; - } else - { - // logger.info( - // "returning current DestChoiceModelManager() - objInstance address = " - // + objInstance ); - return objInstance; - } - } - - // the task instances should call needToInitialize() first, then this method - // if necessary. - public synchronized void managerSetup(HashMap propertyMap, - ModelStructure modelStructure, MatrixDataServerIf ms, String dcUecFileName, - String soaUecFileName, int soaSampleSize, CtrampDmuFactoryIf dmuFactory, - String restartModelString) - { - - if (managerIsSetup) return; - - // get the HouseholdChoiceModelsManager instance and clear the objects - // that hold large memory references - HouseholdChoiceModelsManager.getInstance().clearHhModels(); - - modelIndexWork = 0; - modelIndexSchool = 0; - completedHouseholdsWork = 0; - completedHouseholdsSchool = 0; - - System.out.println(String.format("initializing DC ModelManager: thread=%s.", Thread - .currentThread().getName())); - - this.propertyMap = propertyMap; - this.dcUecFileName = dcUecFileName; - this.soaUecFileName = soaUecFileName; - this.soaSampleSize = soaSampleSize; - this.dmuFactory = dmuFactory; - - logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - - mgraManager = MgraDataManager.getInstance(propertyMap); - tdm = TazDataManager.getInstance(propertyMap); - maxTaz = tdm.getMaxTaz(); - - modelQueueWorkLoc = new LinkedList(); - modelQueueSchoolLoc = new LinkedList(); - modelQueueWork = new LinkedList(); - modelQueueSchool = new LinkedList(); - - // Initialize the MatrixDataManager to use the MatrixDataServer instance - // passed in, unless ms is null. - if (ms == null) - { - - logger.info(Thread.currentThread().getName() - + ": No remote MatrixServer being used, MatrixDataManager will get created when needed by DestChoiceModelManager."); - } else - { - - String testString = ms.testRemote(Thread.currentThread().getName()); - logger.info(String.format(Thread.currentThread().getName() - + ": DestChoiceModelManager connecting to remote MatrixDataServer.")); - logger.info(String.format("MatrixDataServer connection test: %s", testString)); - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - } - - aggAcc = BuildAccessibilities.getInstance(); - - // assume that if the filename exists, at was created previously, either - // in another model run, or by the main client - // if the filename doesn't exist, then calculate the accessibilities - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - boolean accFileReadFlag = Util.getBooleanValueFromPropertyMap(propertyMap, - CtrampApplication.READ_ACCESSIBILITIES); - - if ((new File(accFileName)).canRead()) - { - - logger.info("filling Accessibilities Object in DestChoiceModelManager by reading file: " - + accFileName + "."); - aggAcc.readAccessibilityTableFromFile(accFileName); - - aggAcc.setupBuildAccessibilities(propertyMap, false); - aggAcc.createSchoolSegmentNameIndices(); - - aggAcc.calculateSizeTerms(); - - } else - { - - aggAcc.setupBuildAccessibilities(propertyMap, false); - aggAcc.createSchoolSegmentNameIndices(); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - - logger.info("filling Accessibilities Object in DestChoiceModelManager by calculating them."); - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - - // compute the array of cumulative taz distance based SOA probabilities - // for each origin taz. - locChoiceDistSoaDmu = dmuFactory.getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - managerIsSetup = true; - - } - - public synchronized void returnWorkLocModelObject(WorkLocationChoiceModel dcModel, - int taskIndex, int startIndex, int endIndex) - { - modelQueueWorkLoc.add(dcModel); - completedHouseholdsWork += (endIndex - startIndex + 1); - if(logResults){ - logger.info(String - .format("returned workLocationChoice[%d,%d] to workQueueLoc, task=%d, thread=%s, completedHouseholds=%d.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread - .currentThread().getName(), completedHouseholdsWork)); - } - } - - public synchronized void returnDcWorkModelObject(MandatoryDestChoiceModel dcModel, - int taskIndex, int startIndex, int endIndex) - { - modelQueueWork.add(dcModel); - completedHouseholdsWork += (endIndex - startIndex + 1); - if(logResults){ - logger.info(String - .format("returned dcModelWork[%d,%d] to workQueue, task=%d, thread=%s, completedHouseholds=%d.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread - .currentThread().getName(), completedHouseholdsWork)); - } - } - - public synchronized void returnSchoolLocModelObject(SchoolLocationChoiceModel dcModel, - int taskIndex, int startIndex, int endIndex) - { - modelQueueSchoolLoc.add(dcModel); - completedHouseholdsSchool += (endIndex - startIndex + 1); - if(logResults){ - logger.info(String - .format("returned schoolLocationChoice[%d,%d] to schoolQueueLoc, task=%d, thread=%s, completedHouseholds=%d.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread - .currentThread().getName(), completedHouseholdsSchool)); - } - } - - public synchronized void returnDcSchoolModelObject(MandatoryDestChoiceModel dcModel, - int taskIndex, int startIndex, int endIndex) - { - modelQueueSchool.add(dcModel); - completedHouseholdsSchool += (endIndex - startIndex + 1); - if(logResults){ - logger.info(String - .format("returned dcModelSchool[%d,%d] to schoolQueue, task=%d, thread=%s, completedHouseholds=%d.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread - .currentThread().getName(), completedHouseholdsSchool)); - } - } - - public synchronized WorkLocationChoiceModel getWorkLocModelObject(int taskIndex, int iteration, - DestChoiceSize dcSizeObj, int[] uecIndices, int[] soaUecIndices) - { - - // can release memory for the school location choice probabilities - // before running school location choice - clearSchoolProbabilitiesArrys(); - - WorkLocationChoiceModel dcModel = null; - - if (!modelQueueWorkLoc.isEmpty()) - { - - // the first task processed with an iteration parameter greater than - // the manager's - // current iteration updates the manager's SOA size and dist - // probabilities arrays and - // updates the iteration count. - if (iteration > currentIteration) - { - - // update the arrays of cumulative probabilities based on mgra - // size for mgras within each origin taz. - double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); - updateWorkSoaProbabilities(workLocSoaDistProbsObject, dcSizeObj, workSizeProbs, - workTazDistProbs, dcSizeArray); - - currentIteration = iteration; - completedHouseholdsWork = 0; - } - - dcModel = modelQueueWorkLoc.remove(); - dcModel.setDcSizeObject(dcSizeObj); - - if(logResults){ - logger.info(String.format( - "removed workLocationChoice[%d,%d] from workQueueLoc, task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - } - - } else - { - - if (modelIndexWork == 0 && iteration == 0) - { - - // compute the arrays of cumulative probabilities based on mgra - // size for mgras within each origin taz. - logger.info("pre-computing work SOA Distance and Size probabilities."); - workLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_WORK_DC_SOA_UEC_FILE, - PROPERTIES_WORK_DC_SOA_UEC_MODEL_PAGE, PROPERTIES_WORK_DC_SOA_UEC_DATA_PAGE); - double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); - workSizeProbs = new double[dcSizeArray.length][maxTaz][]; - workTazDistProbs = new double[dcSizeArray.length][][]; - updateWorkSoaProbabilities(workLocSoaDistProbsObject, dcSizeObj, workSizeProbs, - workTazDistProbs, dcSizeArray); - - currentIteration = 0; - completedHouseholdsWork = 0; - } - - modelIndexWork++; - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - // pass null in instead if modelStructure, since it's not available - // and won't be needed for logsum calculation. - TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, - TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - - dcModel = new WorkLocationChoiceModel(modelIndexWork, propertyMap, dcSizeObj, aggAcc, - dcUecFileName, soaUecFileName, soaSampleSize, modeChoiceUecFileName, - dmuFactory, immcModel, workSizeProbs, workTazDistProbs); - - dcModel.setupWorkSegments(uecIndices, soaUecIndices); - dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, - soaSampleSize); - - logger.info(String.format("created workLocationChoice[%d,%d], task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - - } - - return dcModel; - - } - - public synchronized MandatoryDestChoiceModel getDcWorkModelObject(int taskIndex, int iteration, - DestChoiceSize dcSizeObj, int[] uecIndices, int[] soaUecIndices) - { - - MandatoryDestChoiceModel dcModel = null; - - if (!modelQueueWork.isEmpty()) - { - - // the first task processed with an iteration parameter greater than - // the manager's - // current iteration updates the manager's SOA size and dist - // probabilities arrays and - // updates the iteration count. - if (iteration > currentIteration) - { - - currentIteration = iteration; - completedHouseholdsWork = 0; - } - - dcModel = modelQueueWork.remove(); - dcModel.setDcSizeObject(dcSizeObj); - - if(logResults){ - logger.info(String.format( - "removed dcModelWork[%d,%d] from workQueue, task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - } - - } else - { - - if (modelIndexWork == 0 && iteration == 0) - { - - currentIteration = 0; - completedHouseholdsWork = 0; - } - - modelIndexWork++; - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - // pass null in instead if modelStructure, since it's not available - // and won't be needed for logsum calculation. - TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, - TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - - dcModel = new MandatoryDestChoiceModel(modelIndexWork, propertyMap, dcSizeObj, aggAcc, - mgraManager, dcUecFileName, soaUecFileName, soaSampleSize, - modeChoiceUecFileName, dmuFactory, immcModel); - - dcModel.setupWorkSegments(uecIndices, soaUecIndices); - dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, - soaSampleSize); - - logger.info(String.format("created dcModelWork[%d,%d], task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - - } - - return dcModel; - - } - - public synchronized SchoolLocationChoiceModel getSchoolLocModelObject(int taskIndex, - int iteration, DestChoiceSize dcSizeObj) - { - // can release memory for the work location choice probabilities before - // running school location choice - clearWorkProbabilitiesArrys(); - clearWorkLocModels(); - - SchoolLocationChoiceModel dcModel = null; - - int[] gsDistrict = new int[maxTaz + 1]; - int[] hsDistrict = new int[maxTaz + 1]; - double[] univEnrollment = new double[maxTaz + 1]; - - if (!modelQueueSchoolLoc.isEmpty()) - { - - // the first task processed with an iteration parameter greater than - // the - // manager's current iteration count clears the dcModel cache and - // updates the iteration count. - if (iteration > currentIteration) - { - - // compute the exponentiated distance utilities that all - // segments of this tour purpose will share - double[][] tazDistExpUtils = null; - - logger.info("updating pre-school SOA Distance and Size probabilities."); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(psLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getPsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[PRESCHOOL_ALT_INDEX], - schoolTazDistProbs[PRESCHOOL_ALT_INDEX]); - - logger.info("updating grade school SOA Distance and Size probabilities."); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(gsLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getGsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[GRADE_SCHOOL_ALT_INDEX], - schoolTazDistProbs[GRADE_SCHOOL_ALT_INDEX]); - - logger.info("updating high school SOA Distance and Size probabilities."); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(hsLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getHsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[HIGH_SCHOOL_ALT_INDEX], - schoolTazDistProbs[HIGH_SCHOOL_ALT_INDEX]); - - logger.info("updating university-typical school SOA Distance and Size probabilities."); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(univLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getUnivTypicalSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[UNIV_TYPICAL_ALT_INDEX], - schoolTazDistProbs[UNIV_TYPICAL_ALT_INDEX]); - - logger.info("updating university-non-typical school SOA Distance and Size probabilities."); - updateSchoolSoaProbabilities(aggAcc.getUnivNonTypicalSegmentNameIndexMap(), - dcSizeObj, tazDistExpUtils, schoolSizeProbs[UNIV_NONTYPICAL_ALT_INDEX], - schoolTazDistProbs[UNIV_NONTYPICAL_ALT_INDEX]); - - currentIteration = iteration; - completedHouseholdsSchool = 0; - - } - - dcModel = modelQueueSchoolLoc.remove(); - dcModel.setDcSizeObject(dcSizeObj); - if(logResults){ - logger.info(String.format( - "removed schoolLocationChoice[%d,%d] from schoolQueueLoc, task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - } - - } else - { - - if (modelIndexSchool == 0 && iteration == 0) - { - - // if the schoolSizeProbs array is null, no task has yet - // initialized the probabilities arrays, so enter the block. - // if not null, the arrays have been computed, so it's ok to - // skip. - if (schoolSizeProbs == null) - { - - // compute the exponentiated distance utilities that all - // segments of this tour purpose will share - double[][] tazDistExpUtils = null; - - int[] gsDistrictByMgra = aggAcc.getMgraGsDistrict(); - int[] hsDistrictByMgra = aggAcc.getMgraHsDistrict(); - - // determine university enrollment by TAZs - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray != null) - { - for (int mgra : mgraArray) - { - univEnrollment[taz] = aggAcc.getMgraUniversityEnrollment(mgra); - } - } - } - locChoiceDistSoaDmu.setTazUnivEnrollment(univEnrollment); - - // determine grade school and high school districts by TAZs - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray != null) - { - for (int mgra : mgraArray) - { - gsDistrict[taz] = gsDistrictByMgra[mgra]; - hsDistrict[taz] = hsDistrictByMgra[mgra]; - break; - } - } - } - locChoiceDistSoaDmu.setTazGsDistricts(gsDistrict); - locChoiceDistSoaDmu.setTazHsDistricts(hsDistrict); - - schoolSizeProbs = new double[NUMBER_OF_SCHOOL_SEGMENT_TYPES][maxTaz][]; - schoolTazDistProbs = new double[NUMBER_OF_SCHOOL_SEGMENT_TYPES][maxTaz][maxTaz]; - - // compute the arrays of cumulative probabilities based on - // mgra size for mgras within each origin taz. - try - { - logger.info("pre-computing pre-school SOA Distance and Size probabilities."); - psLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_PS_DC_SOA_UEC_FILE, - PROPERTIES_PS_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_PS_DC_SOA_UEC_DATA_PAGE); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(psLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getPsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[PRESCHOOL_ALT_INDEX], - schoolTazDistProbs[PRESCHOOL_ALT_INDEX]); - } catch (Exception e) - { - logger.error("exception caught updating pre-school SOA probabilities", e); - System.exit(-1); - } - - try - { - logger.info("pre-computing grade school SOA Distance and Size probabilities."); - gsLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_GS_DC_SOA_UEC_FILE, - PROPERTIES_GS_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_GS_DC_SOA_UEC_DATA_PAGE); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(gsLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getGsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[GRADE_SCHOOL_ALT_INDEX], - schoolTazDistProbs[GRADE_SCHOOL_ALT_INDEX]); - } catch (Exception e) - { - logger.error("exception caught updating grade school SOA probabilities", e); - System.exit(-1); - } - - try - { - logger.info("pre-computing high school SOA Distance and Size probabilities."); - hsLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_HS_DC_SOA_UEC_FILE, - PROPERTIES_HS_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_HS_DC_SOA_UEC_DATA_PAGE); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(hsLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getHsSegmentNameIndexMap(), dcSizeObj, - tazDistExpUtils, schoolSizeProbs[HIGH_SCHOOL_ALT_INDEX], - schoolTazDistProbs[HIGH_SCHOOL_ALT_INDEX]); - } catch (Exception e) - { - logger.error("exception caught updating high school SOA probabilities", e); - System.exit(-1); - } - - try - { - logger.info("pre-computing university-typical SOA Distance and Size probabilities."); - univLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_UNIV_DC_SOA_UEC_FILE, - PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE); - tazDistExpUtils = computeTazDistanceExponentiatedUtilities(univLocSoaDistProbsObject); - updateSchoolSoaProbabilities(aggAcc.getUnivTypicalSegmentNameIndexMap(), - dcSizeObj, tazDistExpUtils, - schoolSizeProbs[UNIV_TYPICAL_ALT_INDEX], - schoolTazDistProbs[UNIV_TYPICAL_ALT_INDEX]); - } catch (Exception e) - { - logger.error("exception caught updating university SOA probabilities", e); - System.exit(-1); - } - - try - { - logger.info("pre-computing university-non-typical SOA Distance and Size probabilities."); - univLocSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_UNIV_DC_SOA_UEC_FILE, - PROPERTIES_UNIV_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_UNIV_DC_SOA_UEC_DATA_PAGE); - updateSchoolSoaProbabilities(aggAcc.getUnivNonTypicalSegmentNameIndexMap(), - dcSizeObj, tazDistExpUtils, - schoolSizeProbs[UNIV_NONTYPICAL_ALT_INDEX], - schoolTazDistProbs[UNIV_NONTYPICAL_ALT_INDEX]); - } catch (Exception e) - { - logger.error("exception caught updating university SOA probabilities", e); - System.exit(-1); - } - - currentIteration = 0; - completedHouseholdsSchool = 0; - - } - - } - - modelIndexSchool++; - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - // pass null in instead if modelStructure, since it's not available - // and won't be needed for logsum calculation. - TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, - TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - - dcModel = new SchoolLocationChoiceModel(modelIndexSchool, propertyMap, dcSizeObj, - aggAcc, dcUecFileName, soaUecFileName, soaSampleSize, modeChoiceUecFileName, - dmuFactory, immcModel, schoolSizeProbs, schoolTazDistProbs); - - dcModel.setupSchoolSegments(); - dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, - soaSampleSize); - - logger.info(String.format("created schoolLocationChoice[%d,%d], task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - - } - - return dcModel; - - } - - public synchronized MandatoryDestChoiceModel getDcSchoolModelObject(int taskIndex, - int iteration, DestChoiceSize dcSizeObj) - { - - MandatoryDestChoiceModel dcModel = null; - if (!modelQueueSchool.isEmpty()) - { - - // the first task processed with an iteration parameter greater than - // the - // manager's current iteration count clears the dcModel cache and - // updates the iteration count. - if (iteration > currentIteration) - { - - currentIteration = iteration; - completedHouseholdsSchool = 0; - - } - - dcModel = modelQueueSchool.remove(); - dcModel.setDcSizeObject(dcSizeObj); - if(logResults){ - logger.info(String.format( - "removed dcModelSchool[%d,%d] from schoolQueue, task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - } - - } else - { - - if (modelIndexSchool == 0 && iteration == 0) - { - currentIteration = 0; - completedHouseholdsSchool = 0; - } - - modelIndexSchool++; - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - // pass null in instead if modelStructure, since it's not available - // and won't be needed for logsum calculation. - TourModeChoiceModel immcModel = new TourModeChoiceModel(propertyMap, null, - TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - - dcModel = new MandatoryDestChoiceModel(modelIndexSchool, propertyMap, dcSizeObj, - aggAcc, mgraManager, dcUecFileName, soaUecFileName, soaSampleSize, - modeChoiceUecFileName, dmuFactory, immcModel); - - dcModel.setupSchoolSegments(); - dcModel.setupDestChoiceModelArrays(propertyMap, dcUecFileName, soaUecFileName, - soaSampleSize); - - logger.info(String.format("created dcModelSchool[%d,%d], task=%d, thread=%s.", - currentIteration, dcModel.getModelIndex(), taskIndex, Thread.currentThread() - .getName())); - - } - - return dcModel; - - } - - public synchronized void clearDcModels() - { - - clearWorkLocModels(); - clearSchoolLocModels(); - clearWorkProbabilitiesArrys(); - clearSchoolProbabilitiesArrys(); - - if (tazDistanceCalculator != null) - { - tazDistanceCalculator.clearStoredTazsDistanceSkims(); - tazDistanceCalculator = null; - } - - logger.info("DestChoiceModelManager elements cleared."); - - } - - private void clearWorkLocModels() - { - - if (modelQueueWorkLoc != null && !modelQueueWorkLoc.isEmpty()) - { - - logger.info(String.format( - "%s: clearing dc choice models modelQueueWorkLoc, thread=%s.", new Date(), - Thread.currentThread().getName())); - - while (!modelQueueWorkLoc.isEmpty()) - modelQueueWorkLoc.remove(); - modelIndexWork = 0; - completedHouseholdsWork = 0; - - } - - if (modelQueueWork != null && !modelQueueWork.isEmpty()) - { - - logger.info(String.format("%s: clearing dc choice models modelQueueWork, thread=%s.", - new Date(), Thread.currentThread().getName())); - while (!modelQueueWork.isEmpty()) - modelQueueWork.remove(); - - modelIndexWork = 0; - completedHouseholdsWork = 0; - - } - - } - - private void clearSchoolLocModels() - { - - if (modelQueueSchoolLoc != null && !modelQueueSchoolLoc.isEmpty()) - { - - logger.info(String.format( - "%s: clearing dc choice models modelQueueSchoolLoc, thread=%s.", new Date(), - Thread.currentThread().getName())); - while (!modelQueueSchoolLoc.isEmpty()) - modelQueueSchoolLoc.remove(); - - modelIndexSchool = 0; - completedHouseholdsSchool = 0; - - } - - if (modelQueueSchool != null && !modelQueueSchool.isEmpty()) - { - - logger.info(String.format( - "%s: clearing dc choice models modelQueueSchool, thread=%s.", new Date(), - Thread.currentThread().getName())); - while (!modelQueueSchool.isEmpty()) - modelQueueSchool.remove(); - - modelIndexSchool = 0; - completedHouseholdsSchool = 0; - - } - - } - - private void clearWorkProbabilitiesArrys() - { - - // null out the cache of probabilities arrays for work location choice - if (workSizeProbs != null) - { - for (int i = 0; i < workSizeProbs.length; i++) - { - if (workSizeProbs[i] != null) - { - for (int j = 0; j < workSizeProbs[i].length; j++) - workSizeProbs[i][j] = null; - } - workSizeProbs[i] = null; - } - workSizeProbs = null; - } - - if (workTazDistProbs != null) - { - for (int i = 0; i < workTazDistProbs.length; i++) - { - if (workTazDistProbs[i] != null) - { - for (int j = 0; j < workTazDistProbs[i].length; j++) - workTazDistProbs[i][j] = null; - } - workTazDistProbs[i] = null; - } - workTazDistProbs = null; - } - - } - - private void clearSchoolProbabilitiesArrys() - { - - // null out the cache of probabilities arrays for work location choice - if (schoolSizeProbs != null) - { - for (int i = 0; i < schoolSizeProbs.length; i++) - { - if (schoolSizeProbs[i] != null) - { - for (int j = 0; j < schoolSizeProbs[i].length; j++) - schoolSizeProbs[i][j] = null; - } - schoolSizeProbs[i] = null; - } - schoolSizeProbs = null; - } - - if (schoolTazDistProbs != null) - { - for (int i = 0; i < schoolTazDistProbs.length; i++) - { - if (schoolTazDistProbs[i] != null) - { - for (int j = 0; j < schoolTazDistProbs[i].length; j++) - schoolTazDistProbs[i][j] = null; - } - schoolTazDistProbs[i] = null; - } - schoolTazDistProbs = null; - } - - } - - private void updateWorkSoaProbabilities( - DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, - DestChoiceSize dcSizeObj, double[][][] sizeProbs, double[][][] tazDistProbs, - double[][] dcSizeArray) - { - - HashMap segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); - - for (String segmentName : segmentNameIndexMap.keySet()) - { - - int segmentIndex = segmentNameIndexMap.get(segmentName); - - // compute the TAZ size values from the mgra values and the - // correspondence between mgras and tazs. - double[] tazSize = computeTazSize(dcSizeArray[segmentIndex]); - locChoiceDistSoaDmu.setDestChoiceTazSize(tazSize); - - // tazDistProbs[segmentIndex] = - // locChoiceSoaDistProbsObject.computeDistanceProbabilities( 3737, - // locChoiceDistSoaDmu ); - tazDistProbs[segmentIndex] = locChoiceSoaDistProbsObject - .computeDistanceProbabilities(locChoiceDistSoaDmu); - - computeSizeSegmentProbabilities(sizeProbs[segmentIndex], dcSizeArray[segmentIndex]); - - } - - } - - private void updateSchoolSoaProbabilities(HashMap segmentNameIndexMap, - DestChoiceSize dcSizeObj, double[][] tazDistExpUtils, double[][] sizeProbs, - double[][] tazDistProbs) - { - - double[][] dcSizeArray = dcSizeObj.getDcSizeArray(); - - double[] tempExpUtils = new double[tazDistExpUtils.length]; - - // compute an array of SOA probabilities for each segment - for (String segmentName : segmentNameIndexMap.keySet()) - { - - // compute the TAZ size values from the mgra values and the - // correspondence between mgras and tazs. - int segmentIndex = segmentNameIndexMap.get(segmentName); - double[] tazSize = computeTazSize(dcSizeArray[segmentIndex]); - - // compute the taz dist probabilities from the exponentiated - // utilities for this segmnet and the taz size terms - for (int i = 0; i < tazDistExpUtils.length; i++) - { - - // compute the final exponentiated utilities by multiplying with - // taz size, and accumulate total exponentiated utility. - double totalExpUtil = 0; - for (int j = 0; j < tempExpUtils.length; j++) - { - tempExpUtils[j] = tazDistExpUtils[i][j] * tazSize[j + 1]; - totalExpUtil += tempExpUtils[j]; - } - - if (totalExpUtil > 0) - { - - // compute the SOA cumulative probabilities - tazDistProbs[i][0] = tempExpUtils[0] / totalExpUtil; - for (int j = 1; j < tempExpUtils.length - 1; j++) - { - double prob = tempExpUtils[j] / totalExpUtil; - tazDistProbs[i][j] = tazDistProbs[i][j - 1] + prob; - } - tazDistProbs[i][tempExpUtils.length - 1] = 1.0; - - } - - } - - computeSizeSegmentProbabilities(sizeProbs, dcSizeArray[segmentIndex]); - - } - - } - - private double[][] computeTazDistanceExponentiatedUtilities( - DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject) - { - - double[][] tazDistExpUtils = locChoiceSoaDistProbsObject - .computeDistanceUtilities(locChoiceDistSoaDmu); - for (int i = 0; i < tazDistExpUtils.length; i++) - for (int j = 0; j < tazDistExpUtils[i].length; j++) - { - if (tazDistExpUtils[i][j] < -500) tazDistExpUtils[i][j] = 0; - else tazDistExpUtils[i][j] = Math.exp(tazDistExpUtils[i][j]); - } - - return tazDistExpUtils; - - } - - private double[] computeTazSize(double[] size) - { - - double[] tazSize = new double[maxTaz + 1]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray != null) - { - for (int mgra : mgraArray) - { - tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); - } - } - - } - - return tazSize; - - } - - private void computeSizeSegmentProbabilities(double[][] sizeProbs, double[] size) - { - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - - if (mgraArray == null) - { - sizeProbs[taz - 1] = new double[0]; - } else - { - double totalSize = 0; - for (int mgra : mgraArray) - totalSize += size[mgra] + (size[mgra] > 0 ? 1 : 0); - - if (totalSize > 0) - { - sizeProbs[taz - 1] = new double[mgraArray.length]; - for (int i = 0; i < mgraArray.length; i++) - { - double mgraSize = size[mgraArray[i]]; - if (mgraSize > 0) mgraSize += 1; - sizeProbs[taz - 1][i] = mgraSize / totalSize; - } - } else if (sizeProbs[taz - 1] == null) - { - sizeProbs[taz - 1] = new double[0]; - } - } - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java deleted file mode 100644 index 695b2fc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceSize.java +++ /dev/null @@ -1,965 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.text.DecimalFormat; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.datafile.CSVFileWriter; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * Handles building and storing destination choice size variables - * - */ - -public class DestChoiceSize - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(DestChoiceSize.class); - private transient Logger convergeLogger = Logger.getLogger("converge"); - - public static final String PROPERTIES_DC_SHADOW_OUTPUT = "uwsl.ShadowPricing.OutputFile"; - public static final String PROPERTIES_WORK_DC_SHADOW_NITER = "uwsl.ShadowPricing.Work.MaximumIterations"; - public static final String PROPERTIES_SCHOOL_DC_SHADOW_NITER = "uwsl.ShadowPricing.School.MaximumIterations"; - - private int numSegments; - private double[][] segmentSizeTerms; - private HashMap segmentIndexNameMap; - private HashMap segmentNameIndexMap; - private HashSet noShadowPriceSchoolSegmentIndices; - private MgraDataManager mgraManager; - - // 1st dimension is an index for the set of DC Size variables used in Sample - // of - // Alternative choice and destination choice, - // 2nd dimension is zone number (1,...,numZones), 3rd dimension walk subzone - // index is 0: no walk %, 1: shrt %, 2: long %. - protected double[][] dcSize; - protected double[][] originalSize; - protected double[][] originalAdjSize; - protected double[][] scaledSize; - protected double[][] balanceSize; - protected double[][] previousSize; - protected double[][] shadowPrice; - - protected double[][] externalFactors; - - protected int maxShadowPriceIterations; - - protected String dcShadowOutputFileName; - - protected boolean dcSizeCalculated = false; - - /** - * - * @param propertyMap - * is the model properties file key:value pairs - * @param segmentNameIndexMap - * is a map from segment name to size term array index. - * @param segmentIndexNameMap - * is a map from size term array index to segment name. - * @param segmentSizeTerms - * is an array by segment index and MGRA index - */ - public DestChoiceSize(HashMap propertyMap, - HashMap segmentIndexNameMap, - HashMap segmentNameIndexMap, double[][] segmentSizeTerms, - int maxIterations) - { - - this.segmentIndexNameMap = segmentIndexNameMap; - this.segmentNameIndexMap = segmentNameIndexMap; - this.segmentSizeTerms = segmentSizeTerms; - - // get the number of segments from the segmentIndexNameMap - numSegments = segmentIndexNameMap.size(); - - maxShadowPriceIterations = maxIterations; - - String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap, - CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - dcShadowOutputFileName = projectDirectory + propertyMap.get(PROPERTIES_DC_SHADOW_OUTPUT); - - mgraManager = MgraDataManager.getInstance(); - - // set default external factors array (all 1.0s) - // an object creating a destChoice object can override these valuse by - // calling setExternalFactors(). - externalFactors = new double[numSegments][mgraManager.getMaxMgra() + 1]; - for (int i = 0; i < numSegments; i++) - Arrays.fill(externalFactors[i], 1.0); - - } - - /** - * @return a boolean for whether or not the size terms in this object have - * been calculated - */ - public boolean getDcSizeCalculated() - { - return dcSizeCalculated; - } - - public HashMap getSegmentIndexNameMap() - { - return segmentIndexNameMap; - } - - public HashMap getSegmentNameIndexMap() - { - return segmentNameIndexMap; - } - - /** - * @return the maximum number of shadow price iterations set in the - * properties file - */ - public int getMaxShadowPriceIterations() - { - return maxShadowPriceIterations; - } - - public void setExternalFactors(double[][] factors) - { - externalFactors = factors; - } - - public void setNoShadowPriceSchoolSegmentIndices(HashSet indexSet) - { - noShadowPriceSchoolSegmentIndices = indexSet; - } - - /** - * Scale the destination choice size values so that the total modeled - * destinations by segment match the total origins. total Origin/Destination - * constraining usuallu done for home oriented mandatory tours, e.g. work - * university, school. This method also has the capability to read a file of - * destination size adjustments and apply them during the balancing - * procedure. This capability was used in the Morpc model and was - * transferred to the Baylanta project, but may or may not be used. - * - * @param originsByHomeZone - * - total long term choice origin locations (i.e. number of - * workers, university students, or school age students) in - * residence zone, subzone, by segment. - * - */ - public void balanceSizeVariables(int[][] originsByHomeMgra) - { - - // store the original size variable values. - // set the initial sizeBalance values to the original size variable - // values. - originalSize = duplicateDouble2DArray(segmentSizeTerms); - balanceSize = duplicateDouble2DArray(segmentSizeTerms); - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - // create the shadow price array - num - shadowPrice = new double[numSegments][maxMgra + 1]; - - // create the total origin locations array to store total tours by - // segment - double[] totalOriginLocations = new double[numSegments]; - - // create the total destination choice size array to store total tours - // by - // segment - double[] totalDestSize = new double[numSegments]; - - // initialize shadow prices with 1.0 - // accumulate total tours and size by segment. - for (int i = 0; i < numSegments; i++) - { - for (int j = 1; j <= maxMgra; j++) - { - shadowPrice[i][j] = 1.0; - totalOriginLocations[i] += originsByHomeMgra[i][j]; - totalDestSize[i] += (segmentSizeTerms[i][j] * externalFactors[i][j]); - } // j (mgra) - } // i (segment) - - // log a report of total origin locations by segment - logger.info(""); - logger.info("total origin locations by segment before any balancing, destination size adjustments, or shadow price scaling:"); - double segmentSum = 0.0; - for (int i = 0; i < numSegments; i++) - { - String segmentString = segmentIndexNameMap.get(i); - segmentSum += totalOriginLocations[i]; - logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, - totalOriginLocations[i])); - } // i - logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); - logger.info(""); - - // log a report of total destination choice size calculated by segment - logger.info(""); - logger.info("total destination choice size by segment before any balancing, destination choice size adjustments, or shadow price scaling:"); - segmentSum = 0.0; - for (int i = 0; i < numSegments; i++) - { - String segmentString = segmentIndexNameMap.get(i); - segmentSum += totalDestSize[i]; - logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, - totalDestSize[i])); - } - logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); - logger.info(""); - - // save original adjusted size variable arrays prior to balancing - used - // in - // reporting size variable calculations to output files. - originalAdjSize = duplicateDouble2DArray(balanceSize); - - // Balance destination choice size variables to equal total origin - // locations by segment. - // The scaledSize calculated is what is adjusted by shadow pricing - // adjustments, and dcSize, the array referenced - // by UEC DMUs is a duplicate copy of this array after the shadow - // pricing - // calculations are made. - scaledSize = new double[balanceSize.length][maxMgra + 1]; - double tot = 0.0; - for (int i = 0; i < numSegments; i++) - { - - tot = 0.0; - for (int j = 1; j <= maxMgra; j++) - { - - if (totalDestSize[i] > 0.0) scaledSize[i][j] = (balanceSize[i][j] - * externalFactors[i][j] * totalOriginLocations[i]) - / totalDestSize[i]; - else scaledSize[i][j] = 0.0f; - - tot += scaledSize[i][j]; - - } - - } - - // set destination choice size variables for the first iteration of - // shadow - // pricing to calculated scaled values - dcSize = duplicateDouble2DArray(scaledSize); - - // sum scaled destination size values by segment for reporting - double[] sumScaled = new double[numSegments]; - for (int i = 0; i < numSegments; i++) - { - for (int j = 1; j <= maxMgra; j++) - sumScaled[i] += scaledSize[i][j]; - } - - // log a report of total destination locations by segment - logger.info(""); - logger.info("total destination choice size by segment after destination choice size adjustments, after shadow price scaling:"); - segmentSum = 0.0; - for (int i = 0; i < numSegments; i++) - { - String segmentString = segmentIndexNameMap.get(i); - segmentSum += sumScaled[i]; - logger.info(String.format(" %-6d %-55s: %10.1f", i, segmentString, sumScaled[i])); - } - logger.info(String.format(" %-6s %-55s: %10.1f", " ", "Total", segmentSum)); - logger.info(""); - - // save scaled size variables used in shadow price adjustmnents for - // reporting - // to output file - previousSize = new double[numSegments][]; - for (int i = 0; i < numSegments; i++) - previousSize[i] = duplicateDouble1DArray(scaledSize[i]); - - } - - public double getDcSize(int segmentIndex, int mgra) - { - return dcSize[segmentIndex][mgra]; - } - - public double getDcSize(String segmentName, int mgra) - { - int segmentIndex = segmentNameIndexMap.get(segmentName); - return dcSize[segmentIndex][mgra]; - } - - public double[][] getDcSizeArray() - { - return dcSize; - } - - public int getNumberOfSegments() - { - return dcSize.length; - } - - public void updateSizeVariables() - { - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - for (int i = 0; i < numSegments; i++) - { - for (int j = 1; j <= maxMgra; j++) - { - dcSize[i][j] = scaledSize[i][j] * shadowPrice[i][j]; - if (dcSize[i][j] < 0.0f) dcSize[i][j] = 0.0f; - } - } - - } - - public void updateShadowPrices(int[][] modeledDestinationLocationsByDestMgra) - { - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - for (int i = 0; i < numSegments; i++) - { - if (noShadowPriceSchoolSegmentIndices != null - && noShadowPriceSchoolSegmentIndices.contains(i)) continue; - - for (int j = 1; j <= maxMgra; j++) - { - if (modeledDestinationLocationsByDestMgra[i][j] > 0) - shadowPrice[i][j] *= (scaledSize[i][j] / modeledDestinationLocationsByDestMgra[i][j]); - // else - // shadowPrice[i][j] *= scaledSize[i][j]; - } - } - - } - - public void reportMaxDiff(int iteration, int[][] modeledDestinationLocationsByDestMgra) - { - - double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; - double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, Double.MAX_VALUE}; - - int[] nObs = new int[maxSize.length]; - double[] sse = new double[maxSize.length]; - double[] sumObs = new double[maxSize.length]; - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - logger.info("Shadow Price Iteration " + iteration); - - double minRange = 0.0; - for (int r = 0; r < maxSize.length; r++) - { - - logger.info(String - .format("Frequency of chosen mgra locations with non-zero DC Size < %s by range of relative error", - (maxSize[r] < 1000000 ? String.format("%.1f", maxSize[r]) : "+Inf"))); - logger.info(String.format("%-6s %-55s %15s %15s %15s %15s %15s %15s %15s %8s", - "index", "segment", "0 DCs", "< 5%", "< 10%", "< 25%", "< 50%", "< 100%", - "100% +", "Total")); - - int tot = 0; - int[] tots = new int[maxDeltas.length + 1]; - String logRecord = ""; - for (int i = 0; i < numSegments; i++) - { - - tot = 0; - int[] freqs = new int[maxDeltas.length + 1]; - int nonZeroSizeLocs = 0; - for (int j = 1; j <= maxMgra; j++) - { - - if (scaledSize[i][j] > minRange && scaledSize[i][j] <= maxSize[r]) - { - - nonZeroSizeLocs++; - - if (modeledDestinationLocationsByDestMgra[i][j] == 0.0) - { - // store the number of DC alternatives where DC Size - // > 0, - // but alternative was not chosen. - // relative error measure is not meaningful for this - // case, so report number of cases separately. - freqs[0]++; - - // calculations for %RMSE - sse[r] += scaledSize[i][j] * scaledSize[i][j]; - } else - { - - double relDiff = Math.abs(scaledSize[i][j] - - modeledDestinationLocationsByDestMgra[i][j]) - / scaledSize[i][j]; - for (int k = 0; k < maxDeltas.length; k++) - { - if (relDiff < maxDeltas[k]) - { - // store number of DC alternatives chosen - // where - // DC Size > 0, by relative error range. - freqs[k + 1]++; - break; - } - } - - // calculations for %RMSE - sse[r] += relDiff * relDiff; - } - - // calculations for %RMSE - sumObs[r] += scaledSize[i][j]; - nObs[r]++; - - } - - } - - for (int k = 0; k < freqs.length; k++) - { - tots[k] += freqs[k]; - tot += freqs[k]; - } - - String segmentString = segmentIndexNameMap.get(i); - logRecord = String.format("%-6d %-55s", i, segmentString); - - for (int k = 0; k < freqs.length; k++) - { - float pct = 0.0f; - if (tot > 0) pct = (float) (100.0 * freqs[k] / tot); - logRecord += String.format(" %6d (%5.1f%%)", freqs[k], pct); - } - - logRecord += String.format(" %8d", tot); - logger.info(logRecord); - - } - - tot = 0; - for (int k = 0; k < tots.length; k++) - { - tot += tots[k]; - } - - logRecord = String.format("%-6s %-55s", " ", "Total"); - String underline = String.format("------------------------"); - - for (int k = 0; k < tots.length; k++) - { - float pct = 0.0f; - if (tot > 0) pct = (float) (100.0 * tots[k] / tot); - logRecord += String.format(" %6d (%5.1f%%)", tots[k], pct); - underline += String.format("----------------"); - } - - logRecord += String.format(" %8d", tot); - underline += String.format("---------"); - - logger.info(underline); - logger.info(logRecord); - - double rmse = -1.0; - if (nObs[r] > 1) - rmse = 100.0 * (Math.sqrt(sse[r] / (nObs[r] - 1)) / (sumObs[r] / nObs[r])); - - logger.info("%RMSE = " - + (rmse < 0 ? "N/A, no observations" : String.format( - "%.1f, with mean %.1f, for %d observations.", rmse, - (sumObs[r] / nObs[r]), nObs[r]))); - - logger.info(""); - - minRange = maxSize[r]; - - } - - logger.info(""); - logger.info(""); - - } - - public void saveSchoolMaxDiffValues(int iteration, int[][] modeledDestinationLocationsByDestMgra) - { - - // define labels for the schoolsegment categories - String[] segmentRangelabels = {"Pre-School", "K-8", "9-12", "Univ"}; - - // define the highest index value for the range of segments for the - // school segment category - int[] segmentRange = {0, 36, 54, 56}; - - double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; - double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, 999.9}; - - int[][][] freqs = new int[segmentRangelabels.length][maxSize.length][maxDeltas.length]; - - int[][] nObs = new int[segmentRangelabels.length][maxSize.length]; - double[][] sse = new double[segmentRangelabels.length][maxSize.length]; - double[][] sumObs = new double[segmentRangelabels.length][maxSize.length]; - - double[][] rmse = new double[segmentRangelabels.length][maxSize.length]; - double[][] meanSize = new double[segmentRangelabels.length][maxSize.length]; - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - convergeLogger.info("School Shadow Price Iteration " + iteration); - - double[] minRange = new double[segmentRangelabels.length]; - - int minS = 0; - for (int s = 0; s < segmentRangelabels.length; s++) - { - - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(segmentRangelabels[s] + " convergence statistics"); - - if (s > 0) minS = segmentRange[s - 1] + 1; - - for (int r = 0; r < maxSize.length; r++) - { - - for (int i = minS; i <= segmentRange[s]; i++) - { - - for (int j = 1; j <= maxMgra; j++) - { - - if (scaledSize[i][j] > minRange[s] && scaledSize[i][j] <= maxSize[r]) - { - - if (modeledDestinationLocationsByDestMgra[i][j] > 0.0) - { - - int delta = maxDeltas.length - 1; - double diff = Math.abs(scaledSize[i][j] - - modeledDestinationLocationsByDestMgra[i][j]); - double relDiff = diff / scaledSize[i][j]; - for (int k = 0; k < maxDeltas.length; k++) - { - if (relDiff < maxDeltas[k]) - { - delta = k; - break; - } - } - - freqs[s][r][delta]++; - - // calculations for %RMSE - sse[s][r] += (diff * diff); - - } - - // calculations for %RMSE - sumObs[s][r] += scaledSize[i][j]; - nObs[s][r]++; - - } - - } - - } - - rmse[s][r] = -1.0; - if (nObs[s][r] > 1) - { - meanSize[s][r] = sumObs[s][r] / nObs[s][r]; - rmse[s][r] = 100.0 * (Math.sqrt((sse[s][r] / (nObs[s][r] - 1))) / meanSize[s][r]); - } - - minRange[s] = maxSize[r]; - - } - - convergeLogger.info("%RMSE by DC Size Range Category"); - for (int i = 0; i < maxSize.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], rmse[s][i])); - convergeLogger.info(String - .format("%-8s %14.2f", " 1000+", rmse[s][maxSize.length - 1])); - - convergeLogger.info(""); - - convergeLogger.info("%Mean DC Size by DC Size Range Category"); - for (int i = 0; i < maxSize.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], meanSize[s][i])); - convergeLogger.info(String.format("%-8s %14.2f", " 1000+", - meanSize[s][maxSize.length - 1])); - - convergeLogger.info(""); - - convergeLogger.info("Freq of MGRAs by DC Size Range Category and Relative Error"); - for (int r = 0; r < maxSize.length - 1; r++) - { - - convergeLogger.info(String.format("Size < %-8.0f", maxSize[r])); - for (int i = 0; i < maxDeltas.length - 1; i++) - convergeLogger.info(String - .format("< %-8.2f %12d", maxDeltas[i], freqs[s][r][i])); - convergeLogger.info(String.format("%-8s %14d", " 1.0+", - freqs[s][r][maxDeltas.length - 1])); - - convergeLogger.info(""); - } - - convergeLogger.info(String.format("Size >= 1000")); - for (int i = 0; i < maxDeltas.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12d", maxDeltas[i], - freqs[s][maxSize.length - 1][i])); - convergeLogger.info(String.format("%-8s %14d", " 1.0+", - freqs[s][maxSize.length - 1][maxDeltas.length - 1])); - - convergeLogger.info(""); - - } - - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(""); - - } - - public void saveWorkMaxDiffValues(int iteration, int[][] modeledDestinationLocationsByDestMgra) - { - - // define labels for the schoolsegment categories - String[] segmentRangelabels = {"White Collar", "Services", "Health", "Retail and Food", - "Blue Collar", "Military"}; - - // define the highest index value for the range of segments for the - // school segment category - int[] segmentRange = {0, 1, 2, 3, 4, 5}; - - double[] maxSize = {10, 100, 1000, Double.MAX_VALUE}; - double[] maxDeltas = {0.05, 0.10, 0.25, 0.50, 1.0, 999.9}; - - int[][][] freqs = new int[segmentRangelabels.length][maxSize.length][maxDeltas.length]; - - int[][] nObs = new int[segmentRangelabels.length][maxSize.length]; - double[][] sse = new double[segmentRangelabels.length][maxSize.length]; - double[][] sumObs = new double[segmentRangelabels.length][maxSize.length]; - - double[][] rmse = new double[segmentRangelabels.length][maxSize.length]; - double[][] meanSize = new double[segmentRangelabels.length][maxSize.length]; - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - convergeLogger.info("Work Shadow Price Iteration " + iteration); - - double[] minRange = new double[segmentRangelabels.length]; - - int minS = 0; - for (int s = 0; s < segmentRangelabels.length; s++) - { - - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(segmentRangelabels[s] + " convergence statistics"); - - if (s > 0) minS = segmentRange[s - 1] + 1; - - for (int r = 0; r < maxSize.length; r++) - { - - for (int i = minS; i <= segmentRange[s]; i++) - { - - for (int j = 1; j <= maxMgra; j++) - { - - if (scaledSize[i][j] > minRange[s] && scaledSize[i][j] <= maxSize[r]) - { - - if (modeledDestinationLocationsByDestMgra[i][j] > 0.0) - { - - int delta = maxDeltas.length - 1; - double diff = Math.abs(scaledSize[i][j] - - modeledDestinationLocationsByDestMgra[i][j]); - double relDiff = diff / scaledSize[i][j]; - for (int k = 0; k < maxDeltas.length; k++) - { - if (relDiff < maxDeltas[k]) - { - delta = k; - break; - } - } - - freqs[s][r][delta]++; - - // calculations for %RMSE - sse[s][r] += (diff * diff); - - } - - // calculations for %RMSE - sumObs[s][r] += scaledSize[i][j]; - nObs[s][r]++; - - } - - } - - } - - rmse[s][r] = -1.0; - if (nObs[s][r] > 1) - { - meanSize[s][r] = sumObs[s][r] / nObs[s][r]; - rmse[s][r] = 100.0 * (Math.sqrt((sse[s][r] / (nObs[s][r] - 1))) / meanSize[s][r]); - } - - minRange[s] = maxSize[r]; - - } - - convergeLogger.info("%RMSE by DC Size Range Category"); - for (int i = 0; i < maxSize.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], rmse[s][i])); - convergeLogger.info(String - .format("%-8s %14.2f", " 1000+", rmse[s][maxSize.length - 1])); - - convergeLogger.info(""); - - convergeLogger.info("%Mean DC Size by DC Size Range Category"); - for (int i = 0; i < maxSize.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12.2f", maxSize[i], meanSize[s][i])); - convergeLogger.info(String.format("%-8s %14.2f", " 1000+", - meanSize[s][maxSize.length - 1])); - - convergeLogger.info(""); - - convergeLogger.info("Freq of MGRAs by DC Size Range Category and Relative Error"); - for (int r = 0; r < maxSize.length - 1; r++) - { - - convergeLogger.info(String.format("Size < %-8.0f", maxSize[r])); - for (int i = 0; i < maxDeltas.length - 1; i++) - convergeLogger.info(String - .format("< %-8.2f %12d", maxDeltas[i], freqs[s][r][i])); - convergeLogger.info(String.format("%-8s %14d", " 1.0+", - freqs[s][r][maxDeltas.length - 1])); - - convergeLogger.info(""); - } - - convergeLogger.info(String.format("Size >= 1000")); - for (int i = 0; i < maxDeltas.length - 1; i++) - convergeLogger.info(String.format("< %-8.2f %12d", maxDeltas[i], - freqs[s][maxSize.length - 1][i])); - convergeLogger.info(String.format("%-8s %14d", " 1.0+", - freqs[s][maxSize.length - 1][maxDeltas.length - 1])); - - convergeLogger.info(""); - } - - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(""); - convergeLogger.info(""); - - } - - public boolean getSegmentIsInSkipSegmentSet(int segment) - { - return noShadowPriceSchoolSegmentIndices.contains(segment); - } - - public void updateShadowPricingInfo(int iteration, int[][] originsByHomeMgra, - int[][] modeledDestinationLocationsByDestMgra, String mandatoryType) - { - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - ArrayList tableHeadings = new ArrayList(); - tableHeadings.add("alt"); - tableHeadings.add("mgra"); - - for (int i = 0; i < numSegments; i++) - { - - String segmentString = segmentIndexNameMap.get(i); - - tableHeadings.add(String.format("%s_origins", segmentString)); - tableHeadings.add(String.format("%s_sizeOriginal", segmentString)); - tableHeadings.add(String.format("%s_sizeAdjOriginal", segmentString)); - tableHeadings.add(String.format("%s_sizeScaled", segmentString)); - tableHeadings.add(String.format("%s_sizePrevious", segmentString)); - tableHeadings.add(String.format("%s_modeledDests", segmentString)); - tableHeadings.add(String.format("%s_sizeFinal", segmentString)); - tableHeadings.add(String.format("%s_shadowPrices", segmentString)); - - } - - // define a TableDataSet for use in writing output file - float[][] tableData = new float[maxMgra + 1][tableHeadings.size()]; - - int alt = 0; - for (int i = 1; i <= maxMgra; i++) - { - - tableData[alt][0] = alt + 1; - tableData[alt][1] = i; - - int index = 2; - - for (int p = 0; p < numSegments; p++) - { - tableData[alt][index++] = (float) originsByHomeMgra[p][i]; - tableData[alt][index++] = (float) originalSize[p][i]; - tableData[alt][index++] = (float) originalAdjSize[p][i]; - tableData[alt][index++] = (float) scaledSize[p][i]; - tableData[alt][index++] = (float) previousSize[p][i]; - tableData[alt][index++] = (float) modeledDestinationLocationsByDestMgra[p][i]; - tableData[alt][index++] = (float) dcSize[p][i]; - tableData[alt][index++] = (float) shadowPrice[p][i]; - } - alt++; - - } - - TableDataSet outputTable = TableDataSet.create(tableData, tableHeadings); - - // write outputTable to new output file - try - { - String newFilename = this.dcShadowOutputFileName.replaceFirst(".csv", "_" - + mandatoryType + "_" + iteration + ".csv"); - CSVFileWriter writer = new CSVFileWriter(); - writer.writeFile(outputTable, new File(newFilename), - new DecimalFormat("#.000000000000")); - // writer.writeFile( outputTable, new File(newFilename) ); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - // save scaled size variables used in shadow price adjustmnents for - // reporting - // to output file - for (int i = 0; i < numSegments; i++) - previousSize[i] = duplicateDouble1DArray(dcSize[i]); - - } - - public void restoreShadowPricingInfo(String fileName) - { - - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - - TableDataSet tds = null; - try - { - tds = reader.readFileAsDouble(new File(fileName)); - } catch (IOException e) - { - logger.error("exception reading saved shadow price file: " + fileName - + " from previous model run.", e); - } - - // the following are based on format used to write the shadow pricing - // file - // first three columns are indices: ALT, ZONE, SUBZONE. - int columnIndex = 2; - int numberOfColumnsPerPurpose = 8; - int scaledSizeColumnOffset = 3; - int previousSizeColumnOffset = 4; - int finalSizeColumnOffset = 6; - int finalShadowPriceOffset = 7; - - // get the number of MGRAs - int maxMgra = mgraManager.getMaxMgra(); - - for (int i = 0; i < numSegments; i++) - { - - // first restore the scaled size values; getColumnAsFloat(column) - // takes a - // 1s based column value, returns a 0s based array of values - int column = columnIndex + i * numberOfColumnsPerPurpose + scaledSizeColumnOffset + 1; - double[] columnData = tds.getColumnAsDoubleFromDouble(column); - for (int z = 1; z <= maxMgra; z++) - scaledSize[i][z] = columnData[z - 1]; - - // next restore the final size values - column = columnIndex + i * numberOfColumnsPerPurpose + finalSizeColumnOffset + 1; - columnData = tds.getColumnAsDoubleFromDouble(column); - for (int z = 1; z <= maxMgra; z++) - dcSize[i][z] = columnData[z - 1]; - - // next restore the previous size values from the final size of the - // previous iteration - column = columnIndex + i * numberOfColumnsPerPurpose + finalSizeColumnOffset + 1; - columnData = tds.getColumnAsDoubleFromDouble(column); - for (int z = 1; z <= maxMgra; z++) - previousSize[i][z] = columnData[z - 1]; - - // finally restore the final shadow price values - column = columnIndex + i * numberOfColumnsPerPurpose + finalShadowPriceOffset + 1; - columnData = tds.getColumnAsDoubleFromDouble(column); - for (int z = 1; z <= maxMgra; z++) - shadowPrice[i][z] = columnData[z - 1]; - - } - - } - - /** - * Create a new double[], dimension it exactly as the argument array, and - * copy the element values from the argument array to the new one. - * - * @param in - * a 1-dimension double array to be duplicated - * @return an exact duplicate of the argument array - */ - private double[] duplicateDouble1DArray(double[] in) - { - double[] out = new double[in.length]; - for (int i = 0; i < in.length; i++) - { - out[i] = in[i]; - } - return out; - } - - /** - * Create a new double[][], dimension it exactly as the argument array, and - * copy the element values from the argument array to the new one. - * - * @param in - * a 2-dimensional double array to be duplicated - * @return an exact duplicate of the argument array - */ - private double[][] duplicateDouble2DArray(double[][] in) - { - double[][] out = new double[in.length][]; - for (int i = 0; i < in.length; i++) - { - out[i] = new double[in[i].length]; - for (int j = 0; j < in[i].length; j++) - { - out[i][j] = in[i][j]; - } - } - return out; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java deleted file mode 100644 index afb76d0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModel.java +++ /dev/null @@ -1,625 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -public class DestChoiceTwoStageModel -{ - - private transient Logger logger = Logger.getLogger(DestChoiceTwoStageModel.class); - - // dimensioned for maxMgra, holds the number of times a specific mgra was - // selected to be in the the sample - private int[] mgraSampleFreq; - - // these arrays are dimesnioned to the maxSampleSize and hold values mgra - // selected for the sample. - private int[] sampleMgras; - private double[] sampleProbabilities; - - // these arrays are dimesnioned to the maxSampleSize, but hold values up to - // the number of unique mgras selected for the sample. - // array values after the number of unique selected mgras are default array - // values. - private int[] uniqueMgraSample; - private double[] uniqueCorrectionFactors; - - // this array holds the sampleIndex associated with a unique mgra, and is - // used to lookup the sample probability for the unique mgra. - private int[] uniqueSampleIndices; - - // use this variable to keep track of the number of unique sampled mgras - // while choosing the sample - private int uniqueIndex; - - private TazDataManager tdm; - private MgraDataManager mgraManager; - - // for each purpose index, the 2D array is 0-based on origTaz and is 0-based - // on destTaz, giving cumulative taz distance probabilities. - private double[][][] tazDistCumProbs; - - // for each purpose index, the 2D array is 0-based on TAZ and is 0-based on - // MGRAs in the taz, giving size probabilities for MGRAs in the TAZ. - private double[][][] mgraSizeProbs; - - private double[][][] slcSizeProbs; - private double[][] slcTazSize; - private double[][] slcTazDistExpUtils; - - // create an array to re-use to hold cumulative probabilities for selecting - // an MGRA from a TAZ. - private double[] tempMgraCumProbs = new double[200]; - - private double[] slcTazProbs; - private double[] slcTazCumProbs; - private int maxTaz; - - private long soaRunTime; - - public DestChoiceTwoStageModel(HashMap propertyMap, int soaMaxSampleSize) - { - - mgraManager = MgraDataManager.getInstance(propertyMap); - int maxMgra = mgraManager.getMaxMgra(); - - tdm = TazDataManager.getInstance(propertyMap); - maxTaz = tdm.getMaxTaz(); - - slcTazProbs = new double[maxTaz]; - slcTazCumProbs = new double[maxTaz]; - - mgraSampleFreq = new int[maxMgra + 1]; - - sampleMgras = new int[soaMaxSampleSize]; - sampleProbabilities = new double[soaMaxSampleSize]; - - uniqueMgraSample = new int[soaMaxSampleSize]; - uniqueCorrectionFactors = new double[soaMaxSampleSize]; - - uniqueSampleIndices = new int[soaMaxSampleSize]; - } - - private void resetSampleArrays() - { - Arrays.fill(mgraSampleFreq, 0); - Arrays.fill(sampleMgras, 0); - Arrays.fill(sampleProbabilities, 0); - Arrays.fill(uniqueMgraSample, 0); - Arrays.fill(uniqueSampleIndices, -1); - Arrays.fill(uniqueCorrectionFactors, 0); - uniqueIndex = 0; - } - - /** - * get the array of unique mgras selected in the sample. The number of - * unique mgras may be fewer than the number selected for the sample - the - * overall sample size. If so, values in this array from - * 0,...,numUniqueMgras-1 will be the selected unique mgras, and values from - * numUniqueMgras,...,maxSampleSize-1 will be 0. - * - * @return uniqueMgraSample array. - */ - public int[] getUniqueSampleMgras() - { - return uniqueMgraSample; - } - - /** - * get the number of unique mgra values in the sample. It gives the - * upperbound of unique values in uniqueMgraSample[0,...,numUniqueMgras-1]. - * - * @return number of unique mgra values in the sample - */ - public int getNumberofUniqueMgrasInSample() - { - return uniqueIndex; - } - - public double[] getUniqueSampleMgraCorrectionFactors() - { - - for (int i = 0; i < uniqueIndex; i++) - { - int chosenMgra = uniqueMgraSample[i]; - int freq = mgraSampleFreq[chosenMgra]; - - int sampleIndex = uniqueSampleIndices[i]; - double prob = sampleProbabilities[sampleIndex]; - - uniqueCorrectionFactors[i] = (float) Math.log((double) freq / prob); - } - - return uniqueCorrectionFactors; - } - - public void computeSoaProbabilities(int origTaz, int segmentTypeIndex) - { - - double[][] sizeProbs = mgraSizeProbs[segmentTypeIndex]; - double[] probs = new double[mgraManager.getMaxMgra() + 1]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray == null) continue; - - if (sizeProbs[taz - 1].length == 0) continue; - - double tazProb = 0; - if (taz > 1) tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][taz - 1] - - tazDistCumProbs[segmentTypeIndex][origTaz - 1][taz - 2]; - else tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; - - for (int mgraIndex = 0; mgraIndex < mgraArray.length; mgraIndex++) - { - double mgraProb = sizeProbs[taz - 1][mgraIndex]; - probs[mgraArray[mgraIndex]] = tazProb * mgraProb; - } - - } - - PrintWriter out = null; - try - { - out = new PrintWriter(new BufferedWriter(new FileWriter(new File("distSoaProbs.csv")))); - - for (int i = 1; i < probs.length; i++) - { - out.println(i + "," + probs[i]); - } - } catch (IOException e) - { - // TODO Auto-generated catch block - e.printStackTrace(); - } - - out.close(); - - } - - public void chooseSampleMgra(int sampleIndex, int origTaz, int sizePurposeIndex, - int segmentTypeIndex, double rn, boolean debug) - { - - // get the chosen TAZ array index for the 0-based cumulative TAZ - // distance probabilities array - int chosenTazIndex = Util.binarySearchDouble( - tazDistCumProbs[segmentTypeIndex][origTaz - 1], rn); - - if (mgraSizeProbs[segmentTypeIndex][chosenTazIndex].length == 0) - { - logger.error("The MGRA size probabilities array for chosen TAZ index = " - + chosenTazIndex + " has 0 length."); - logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); - logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex - + ", origTaz=" + origTaz + ", sizePurposeIndex=" + sizePurposeIndex - + ", segmentTypeIndex=" + segmentTypeIndex); - throw new RuntimeException(); - } - - // get the chosen TAZ distance probability from the taz distance - // cumulative probabilities array - // also initialize the 0 index cumulative MGRA probability to the - // cumulative taz distance propbaility - double tazProb = 0; - double cumProbabilityLowerBound = 0; - if (chosenTazIndex > 0) - { - tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex] - - tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; - cumProbabilityLowerBound = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; - } else - { - tazProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; - cumProbabilityLowerBound = 0; - } - - // get the array of MGRAs for the chosen TAZ (the chosen index + 1) - int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); - - // get the unscaled MGRA size probability, scale by the TAZ distance - // probability, and accumulate cumulative probabilities - tempMgraCumProbs[0] = cumProbabilityLowerBound - + (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][0] * tazProb); - for (int i = 1; i < mgraArray.length; i++) - tempMgraCumProbs[i] = tempMgraCumProbs[i - 1] - + (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][i] * tazProb); - - // get the chosen array index for the 0-based cumulative probabilities - // array - int chosenMgraIndex = Util.binarySearchDouble(cumProbabilityLowerBound, tempMgraCumProbs, - mgraArray.length, rn); - - // use the chosen mgra index to get the chosenMgra value from the - // 0-based array of MGRAs associated with the chosen TAZ - int chosenMgra = mgraArray[chosenMgraIndex]; - - // store the sampled mgra and its selection probability - sampleMgras[sampleIndex] = chosenMgra; - sampleProbabilities[sampleIndex] = (mgraSizeProbs[segmentTypeIndex][chosenTazIndex][chosenMgraIndex] * tazProb); - - // if the sample freq is 0, this mgra has not been selected yet, so add - // it to the array of unique sampled mgras. - if (mgraSampleFreq[chosenMgra] == 0) - { - uniqueMgraSample[uniqueIndex] = chosenMgra; - uniqueSampleIndices[uniqueIndex] = sampleIndex; - uniqueIndex++; - } - - // increment the frequency of times this mgra was selected for the - // sample - mgraSampleFreq[chosenMgra]++; - - if (debug) - { - - double cumDistProb = 0; - double prevDistCumProb = 0; - if (chosenTazIndex > 1) - { - cumDistProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex]; - prevDistCumProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][chosenTazIndex - 1]; - } else - { - cumDistProb = tazDistCumProbs[segmentTypeIndex][origTaz - 1][0]; - prevDistCumProb = 0; - } - - double cumSizeProb = 0; - double prevSizeCumProb = 0; - if (chosenMgraIndex > 0) - { - cumSizeProb = tempMgraCumProbs[chosenMgraIndex]; - prevSizeCumProb = tempMgraCumProbs[chosenMgraIndex - 1]; - } else - { - cumSizeProb = tempMgraCumProbs[0]; - prevSizeCumProb = 0; - } - - logger.info(String.format( - "%-12d %10d %10.6f %16.8f %16.8f %18d %18.8f %18.8f %12d %18.8f", sampleIndex, - chosenTazIndex, rn, prevDistCumProb, cumDistProb, chosenMgraIndex, - prevSizeCumProb, cumSizeProb, chosenMgra, - ((cumSizeProb - prevSizeCumProb) * (cumDistProb - prevDistCumProb)))); - } - - } - - private void chooseSlcSampleMgraBinarySearch(int sampleIndex, int slcOrigTaz, int slcDestTaz, - int slcSizeSegmentIndex, double rn, boolean debug) - { - - // compute stop location sample probabilities from the pre-computed - // sample exponentiated utilities and taz size terms. - // first compute exponentiated utilites for each alternative from the - // pre-computed component exponentiated utilities - double totalExponentiatedUtility = 0; - for (int k = 0; k < maxTaz; k++) - { - slcTazProbs[k] = (slcTazDistExpUtils[slcOrigTaz - 1][k] - * slcTazDistExpUtils[k][slcDestTaz - 1] / slcTazDistExpUtils[slcOrigTaz - 1][slcDestTaz - 1]) - * slcTazSize[slcSizeSegmentIndex][k + 1]; - totalExponentiatedUtility += slcTazProbs[k]; - } - - // now compute alterantive probabilities and determine selected - // alternative - slcTazCumProbs[0] = slcTazProbs[0] / totalExponentiatedUtility; - for (int k = 1; k < maxTaz - 1; k++) - slcTazCumProbs[k] = slcTazCumProbs[k - 1] - + (slcTazProbs[k] / totalExponentiatedUtility); - slcTazCumProbs[maxTaz - 1] = 1.0; - - // get the chosen TAZ array index for the 0-based cumulative TAZ - // distance probabilities array - int chosenTazIndex = Util.binarySearchDouble(slcTazCumProbs, rn); - - /* - * // now compute alterantive probabilities and determine selected - * alternative int chosenTazIndex0 = -1; double sum = slcTazProbs[0] / - * totalExponentiatedUtility; if ( rn < sum ) { chosenTazIndex0 = 0; } - * else { for ( int k=1; k < maxTaz; k++ ) { slcTazProbs[k] /= - * totalExponentiatedUtility; sum += slcTazProbs[k]; if ( rn < sum ) { - * chosenTazIndex0 = k; break; } } } - * - * - * if ( chosenTazIndex0 != chosenTazIndex ) { logger.error ( - * "error - inconsistent choices made by two alternative monte carlo methods. " - * ); System.exit(-1); } - */ - - if (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex].length == 0) - { - logger.error("The MGRA size probabilities array for chosen stop location TAZ index = " - + chosenTazIndex + " has 0 length."); - logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); - logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex - + ", slcOrigTaz=" + slcOrigTaz + ", slcDestTaz=" + slcDestTaz - + ", slcSizeSegmentIndex=" + slcSizeSegmentIndex); - throw new RuntimeException(); - } - - // get the chosen SLC TAZ distance probability from the taz distance - // cumulative probabilities array - // also initialize the 0 index cumulative MGRA probability to the - // cumulative taz distance propbaility - double tazProb = 0; - double cumProbabilityLowerBound = 0; - if (chosenTazIndex > 0) - { - tazProb = slcTazCumProbs[chosenTazIndex] - slcTazCumProbs[chosenTazIndex - 1]; - cumProbabilityLowerBound = slcTazCumProbs[chosenTazIndex - 1]; - } else - { - tazProb = slcTazCumProbs[0]; - cumProbabilityLowerBound = 0; - } - - // get the array of MGRAs for the chosen TAZ (the chosen index + 1) - int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); - - // get the unscaled MGRA size probability, scale by the TAZ distance - // probability, and accumulate cumulative probabilities - tempMgraCumProbs[0] = cumProbabilityLowerBound - + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb); - for (int i = 1; i < mgraArray.length; i++) - tempMgraCumProbs[i] = tempMgraCumProbs[i - 1] - + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][i] * tazProb); - - // get the chosen array index for the 0-based cumulative probabilities - // array - int chosenMgraIndex = Util.binarySearchDouble(cumProbabilityLowerBound, tempMgraCumProbs, - mgraArray.length, rn); - - // use the chosen mgra index to get the chosenMgra value from the - // 0-based array of MGRAs associated with the chosen TAZ - int chosenMgra = mgraArray[chosenMgraIndex]; - - // store the sampled mgra and its selection probability - sampleMgras[sampleIndex] = chosenMgra; - sampleProbabilities[sampleIndex] = (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][chosenMgraIndex] * tazProb); - - // if the sample freq is 0, this mgra has not been selected yet, so add - // it to the array of unique sampled mgras. - if (mgraSampleFreq[chosenMgra] == 0) - { - uniqueMgraSample[uniqueIndex] = chosenMgra; - uniqueSampleIndices[uniqueIndex] = sampleIndex; - uniqueIndex++; - } - - // increment the frequency of times this mgra was selected for the - // sample - mgraSampleFreq[chosenMgra]++; - - } - - private void chooseSlcSampleMgraLinearWalk(int sampleIndex, int slcOrigTaz, int slcDestTaz, - int slcSizeSegmentIndex, double rn, boolean debug) - { - - // compute stop location sample probabilities from the pre-computed - // sample exponentiated utilities and taz size terms. - // first compute exponentiated utilites for each alternative from the - // pre-computed component exponentiated utilities - double totalExponentiatedUtility = 0; - for (int k = 0; k < maxTaz; k++) - { - slcTazProbs[k] = (slcTazDistExpUtils[slcOrigTaz - 1][k] - * slcTazDistExpUtils[k][slcDestTaz - 1] / slcTazDistExpUtils[slcOrigTaz - 1][slcDestTaz - 1]) - * slcTazSize[slcSizeSegmentIndex][k + 1]; - totalExponentiatedUtility += slcTazProbs[k]; - } - - /* - * // now compute alterantive probabilities and determine selected - * alternative slcTazCumProbs[0] = slcTazProbs[0] / - * totalExponentiatedUtility; for ( int k=1; k < maxTaz - 1; k++ ) - * slcTazCumProbs[k] = slcTazCumProbs[k-1] + (slcTazProbs[k] / - * totalExponentiatedUtility); slcTazCumProbs[maxTaz - 1] = 1.0; - * - * - * /* // get the chosen TAZ array index for the 0-based cumulative TAZ - * distance probabilities array int chosenTazIndex0 = - * Util.binarySearchDouble( slcTazCumProbs, rn ); - */ - - // now compute alterantive probabilities and determine selected - // alternative - int chosenTazIndex = -1; - double sum = slcTazProbs[0] / totalExponentiatedUtility; - double cumProbabilityLowerBound = 0; - double tazProb = 0; - if (rn < sum) - { - chosenTazIndex = 0; - tazProb = sum; - } else - { - for (int k = 1; k < maxTaz; k++) - { - tazProb = slcTazProbs[k] / totalExponentiatedUtility; - cumProbabilityLowerBound = sum; - sum += tazProb; - if (rn < sum) - { - chosenTazIndex = k; - break; - } - } - } - - /* - * if ( chosenTazIndex0 != chosenTazIndex ) { logger.error ( - * "error - inconsistent choices made by two alternative monte carlo methods. " - * ); System.exit(-1); } - */ - - if (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex].length == 0) - { - logger.error("The MGRA size probabilities array for chosen stop location TAZ index = " - + chosenTazIndex + " has 0 length."); - logger.error("This should not be the case. If a TAZ was chosen, its TAZ Size > 0, so there should be at least one MGRA with size > 0 in the TAZ."); - logger.error("Likely cause is an indexing bug. sampleIndex=" + sampleIndex - + ", slcOrigTaz=" + slcOrigTaz + ", slcDestTaz=" + slcDestTaz - + ", slcSizeSegmentIndex=" + slcSizeSegmentIndex); - throw new RuntimeException(); - } - - /* - * // get the chosen SLC TAZ distance probability from the taz distance - * cumulative probabilities array // also initialize the 0 index - * cumulative MGRA probability to the cumulative taz distance - * propbaility double tazProb = 0; double cumProbabilityLowerBound = 0; - * if ( chosenTazIndex > 0 ) { tazProb = slcTazCumProbs[chosenTazIndex] - * - slcTazCumProbs[chosenTazIndex-1]; cumProbabilityLowerBound = - * slcTazCumProbs[chosenTazIndex-1]; } else { tazProb = - * slcTazCumProbs[0]; cumProbabilityLowerBound = 0; } - */ - - // get the array of MGRAs for the chosen TAZ (the chosen index + 1) - int[] mgraArray = tdm.getMgraArray(chosenTazIndex + 1); - - /* - * // get the unscaled MGRA size probability, scale by the TAZ distance - * probability, and accumulate cumulative probabilities - * tempMgraCumProbs[0] = cumProbabilityLowerBound + ( - * slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb ); for - * ( int i=1; i < mgraArray.length; i++ ) tempMgraCumProbs[i] = - * tempMgraCumProbs[i-1] + ( - * slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][i] * tazProb ); - */ - - // now compute alterantive probabilities and determine selected - // alternative - int chosenMgraIndex = -1; - sum = cumProbabilityLowerBound - + (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][0] * tazProb); - if (rn < sum) - { - chosenMgraIndex = 0; - } else - { - for (int k = 1; k < mgraArray.length; k++) - { - sum += (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][k] * tazProb); - if (rn < sum) - { - chosenMgraIndex = k; - break; - } - } - } - - /* - * // get the chosen array index for the 0-based cumulative - * probabilities array int chosenMgraIndex = Util.binarySearchDouble( - * cumProbabilityLowerBound, tempMgraCumProbs, mgraArray.length, rn ); - */ - - // use the chosen mgra index to get the chosenMgra value from the - // 0-based array of MGRAs associated with the chosen TAZ - int chosenMgra = mgraArray[chosenMgraIndex]; - - // store the sampled mgra and its selection probability - sampleMgras[sampleIndex] = chosenMgra; - sampleProbabilities[sampleIndex] = (slcSizeProbs[slcSizeSegmentIndex][chosenTazIndex][chosenMgraIndex] * tazProb); - - // if the sample freq is 0, this mgra has not been selected yet, so add - // it to the array of unique sampled mgras. - if (mgraSampleFreq[chosenMgra] == 0) - { - uniqueMgraSample[uniqueIndex] = chosenMgra; - uniqueSampleIndices[uniqueIndex] = sampleIndex; - uniqueIndex++; - } - - // increment the frequency of times this mgra was selected for the - // sample - mgraSampleFreq[chosenMgra]++; - - } - - public void chooseSample(int origTaz, int sizeSegmentIndex, int segmentTypeIndex, - int numInSample, Random rand, boolean debug) - { - - long timeCheck = System.nanoTime(); - - if (debug) - { - computeSoaProbabilities(origTaz, segmentTypeIndex); - } - - resetSampleArrays(); - for (int i = 0; i < numInSample; i++) - { - chooseSampleMgra(i, origTaz, sizeSegmentIndex, segmentTypeIndex, rand.nextDouble(), - debug); - } - - soaRunTime += (System.nanoTime() - timeCheck); - - } - - public void chooseSlcSample(int origTaz, int destTaz, int sizeSegmentIndex, int numInSample, - Random rand, boolean debug) - { - - long timeCheck = System.nanoTime(); - - resetSampleArrays(); - for (int i = 0; i < numInSample; i++) - { - // chooseSlcSampleMgraBinarySearch( i, origTaz, destTaz, - // sizeSegmentIndex, rand.nextDouble(), debug ); - chooseSlcSampleMgraLinearWalk(i, origTaz, destTaz, sizeSegmentIndex, rand.nextDouble(), - debug); - } - - soaRunTime += (System.nanoTime() - timeCheck); - - } - - public void setSlcSoaProbsAndUtils(double[][] slcTazDistExpUtils, double[][][] slcSizeProbs, - double[][] slcTazSize) - { - this.slcSizeProbs = slcSizeProbs; - this.slcTazSize = slcTazSize; - this.slcTazDistExpUtils = slcTazDistExpUtils; - } - - public void setMgraSizeProbs(double[][][] probs) - { - mgraSizeProbs = probs; - } - - public void setTazDistProbs(double[][][] probs) - { - tazDistCumProbs = probs; - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java deleted file mode 100644 index a23453c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageModelDMU.java +++ /dev/null @@ -1,408 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public abstract class DestChoiceTwoStageModelDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(DestChoiceTwoStageModelDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Person person; - protected Tour tour; - protected IndexValues dmuIndex = null; - - protected double workAccessibility; - protected double nonMandatoryAccessibility; - - protected double[] homeMgraNonMandatoryAccessibilityArray; - protected double[] homeMgraTotalEmploymentAccessibilityArray; - - protected int[] sampleMgras; - protected double[] modeChoiceLogsums; - protected double[] dcSoaCorrections; - - protected double[] mgraSizeArray; - protected double[] mgraDistanceArray; - - protected int toursLeftCount; - - protected ModelStructure modelStructure; - protected MgraDataManager mgraManager; - protected BuildAccessibilities aggAcc; - protected AccessibilitiesTable accTable; - - public DestChoiceTwoStageModelDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - initDmuObject(); - } - - public abstract void setMcLogsum(int mgra, double logsum); - - private void initDmuObject() - { - - dmuIndex = new IndexValues(); - - // create default objects - some choice models use these as place - // holders for values - person = new Person(null, -1, modelStructure); - hh = new Household(modelStructure); - - mgraManager = MgraDataManager.getInstance(); - - int maxMgra = mgraManager.getMaxMgra(); - - modeChoiceLogsums = new double[maxMgra + 1]; - dcSoaCorrections = new double[maxMgra + 1]; - - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setPersonObject(Person personObject) - { - person = personObject; - } - - public void setTourObject(Tour tour) - { - this.tour = tour; - } - - public void setAggAcc(BuildAccessibilities aggAcc) - { - this.aggAcc = aggAcc; - } - - public void setAccTable(AccessibilitiesTable myAccTable) - { - accTable = myAccTable; - } - - public void setMgraSizeArray(double[] mgraSizeArray) - { - this.mgraSizeArray = mgraSizeArray; - } - - public void setMgraDistanceArray(double[] mgraDistanceArray) - { - this.mgraDistanceArray = mgraDistanceArray; - } - - public void setSampleArray(int[] sampleArray) - { - sampleMgras = sampleArray; - } - - public void setDcSoaCorrections(double[] sampleCorrections) - { - dcSoaCorrections = sampleCorrections; - } - - public void setNonMandatoryAccessibility(double nonMandatoryAccessibility) - { - this.nonMandatoryAccessibility = nonMandatoryAccessibility; - } - - public void setToursLeftCount(int count) - { - toursLeftCount = count; - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug DC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public Household getHouseholdObject() - { - return hh; - } - - public Person getPersonObject() - { - return person; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - protected int getToursLeftCount() - { - return toursLeftCount; - } - - protected int getMaxContinuousAvailableWindow() - { - - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return hh - .getMaxJointTimeWindow(tour); - else return person.getMaximumContinuousAvailableWindow(); - } - - protected double getDcSoaCorrectionsAlt(int alt) - { - return dcSoaCorrections[alt - 1]; - } - - protected double getMcLogsumDestAlt(int alt) - { - return modeChoiceLogsums[alt - 1]; - } - - protected double getPopulationDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraPopulation(mgra); - } - - protected double getHouseholdsDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraHouseholds(mgra); - } - - protected double getGradeSchoolEnrollmentDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraGradeSchoolEnrollment(mgra); - } - - protected double getHighSchoolEnrollmentDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraHighSchoolEnrollment(mgra); - } - - protected double getUniversityEnrollmentDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraUniversityEnrollment(mgra); - } - - protected double getOtherCollegeEnrollmentDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraOtherCollegeEnrollment(mgra); - } - - protected double getAdultSchoolEnrollmentDestAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return aggAcc.getMgraAdultSchoolEnrollment(mgra); - } - - protected int getIncome() - { - return hh.getIncomeCategory(); - } - - protected int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - protected int getAutos() - { - return hh.getAutosOwned(); - } - - protected int getWorkers() - { - return hh.getWorkers(); - } - - protected int getNumberOfNonWorkingAdults() - { - return hh.getNumberOfNonWorkingAdults(); - } - - protected int getNumPreschool() - { - return hh.getNumPreschool(); - } - - public int getNumGradeSchoolStudents() - { - return hh.getNumGradeSchoolStudents(); - } - - public int getNumHighSchoolStudents() - { - return hh.getNumHighSchoolStudents(); - } - - protected int getNumChildrenUnder16() - { - return hh.getNumChildrenUnder16(); - } - - protected int getNumChildrenUnder19() - { - return hh.getNumChildrenUnder19(); - } - - protected int getAge() - { - return person.getAge(); - } - - protected int getFemaleWorker() - { - if (person.getPersonIsFemale() == 1) return 1; - else return 0; - } - - protected int getFemale() - { - if (person.getPersonIsFemale() == 1) return 1; - else return 0; - } - - protected int getFullTimeWorker() - { - if (person.getPersonIsFullTimeWorker() == 1) return 1; - else return 0; - } - - protected int getTypicalUniversityStudent() - { - return person.getPersonIsTypicalUniversityStudent(); - } - - protected int getPersonType() - { - return person.getPersonTypeNumber(); - } - - protected int getPersonHasBachelors() - { - return person.getHasBachelors(); - } - - protected int getPersonIsWorker() - { - return person.getPersonIsWorker(); - } - - protected int getWorkTaz() - { - return person.getWorkLocation(); - } - - protected int getWorkTourModeIsSOV() - { - boolean tourModeIsSov = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); - if (tourModeIsSov) return 1; - else return 0; - } - - protected int getTourIsJoint() - { - return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 - : 0; - } - - protected double getTotEmpAccessibilityAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return homeMgraTotalEmploymentAccessibilityArray[mgra]; - } - - protected double getNonMandatoryAccessibilityAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return accTable.getAggregateAccessibility("nonmotor", mgra); - } - - protected double getOpSovDistanceAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return mgraDistanceArray[mgra]; - } - - protected double getLnDcSizeAlt(int alt) - { - int mgra = sampleMgras[alt - 1]; - return Math.log(mgraSizeArray[mgra] + 1); - } - - protected void setWorkAccessibility(double accessibility) - { - workAccessibility = accessibility; - } - - protected double getWorkAccessibility() - { - return workAccessibility; - } - - protected double getNonMandatoryAccessibility() - { - return nonMandatoryAccessibility; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java deleted file mode 100644 index c04c83e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaProbabilitiesCalculator.java +++ /dev/null @@ -1,159 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.Arrays; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class DestChoiceTwoStageSoaProbabilitiesCalculator -{ - - private transient Logger soaTwoStageProbsLogger = Logger.getLogger("soaTwoStageProbsLogger"); - - private TazDataManager tdm; - private int maxTaz; - - private ChoiceModelApplication cm; - - public DestChoiceTwoStageSoaProbabilitiesCalculator(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory, String soaDistUECPropertyKey, - String soaDistUECModelSheetPropertyKey, String soaDistUECDataSheetPropertyKey) - { - - tdm = TazDataManager.getInstance(propertyMap); - maxTaz = tdm.getMaxTaz(); - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String soaDistUecFileName = propertyMap.get(soaDistUECPropertyKey); - soaDistUecFileName = uecFileDirectory + soaDistUecFileName; - int soaModelPage = Integer.parseInt(propertyMap.get(soaDistUECModelSheetPropertyKey)); - int soaDataPage = Integer.parseInt(propertyMap.get(soaDistUECDataSheetPropertyKey)); - - DestChoiceTwoStageSoaTazDistanceUtilityDMU tazDistUtilityDmu = dmuFactory - .getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - // create a ChoiceModelApplication object for the filename, model page - // and data page. - cm = new ChoiceModelApplication(soaDistUecFileName, soaModelPage, soaDataPage, propertyMap, - (VariableTable) tazDistUtilityDmu); - - } - - /** - * @param dmuObject - * is the distance utility DMU object - */ - public double[][] computeDistanceUtilities(DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) - { - - double[][] tazDistUtils = new double[maxTaz][maxTaz]; - IndexValues iv = new IndexValues(); - dmuObject.setIndexValuesObject(iv); - - // Loop through combinations of orig/dest TAZs and compute OD utilities - for (int i = 0; i < tazDistUtils.length; i++) - { - iv.setOriginZone(i + 1); - iv.setZoneIndex(i + 1); - cm.computeUtilities(dmuObject, iv); - tazDistUtils[i] = Arrays.copyOf(cm.getUtilities(), tazDistUtils.length); - } - - return tazDistUtils; - } - - /** - * @param dmuObject - * is the distance utility DMU object - * @param distUtilityIndex - * is the distance utility segment index This method signature is - * the default, assuming that no distance probabilities logging - * is required - */ - public double[][] computeDistanceProbabilities( - DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) - { - - double[][] tazDistProbs = new double[maxTaz][maxTaz]; - IndexValues iv = new IndexValues(); - dmuObject.setIndexValuesObject(iv); - - // Loop through combinations of orig/dest TAZs and compute OD utilities - for (int i = 0; i < tazDistProbs.length; i++) - { - iv.setOriginZone(i + 1); - iv.setZoneIndex(i + 1); - cm.computeUtilities(dmuObject, iv); - double[] tempArray = Arrays - .copyOf(cm.getCumulativeProbabilities(), tazDistProbs.length); - tazDistProbs[i] = tempArray; - } - - return tazDistProbs; - } - - /** - * @param dmuObject - * is the distance utility DMU object - * @param distUtilityIndex - * is the distance utility segment index This alternative method - * signature allows distance probabilities logging to be written - */ - public double[][] computeDistanceProbabilities(int traceOrig, - DestChoiceTwoStageSoaTazDistanceUtilityDMU dmuObject) - { - - double[][] tazDistProbs = new double[maxTaz][maxTaz]; - IndexValues iv = new IndexValues(); - dmuObject.setIndexValuesObject(iv); - - // Loop through combinations of orig/dest TAZs and compute OD utilities - for (int i = 0; i < tazDistProbs.length; i++) - { - - iv.setOriginZone(i + 1); - iv.setZoneIndex(i + 1); - cm.computeUtilities(dmuObject, iv); - - if (i == traceOrig - 1) - { - int[] altsToLog = {0, 500, 1000, 2000, 2500, 3736, 3737, 3738, 3739, 3500, 4000}; - cm.logUECResultsSpecificAlts(soaTwoStageProbsLogger, - "Two stage SOA Dist Utilities from TAZ = " + (i + 1), altsToLog); - - double[] probs = cm.getProbabilities(); - double[] utils = cm.getUtilities(); - double total = 0; - for (int k = 0; k < probs.length; k++) - total += Math.exp(utils[k]); - - soaTwoStageProbsLogger.info(""); - for (int k = 1; k < altsToLog.length; k++) - soaTwoStageProbsLogger.info("alt=" + (altsToLog[k] - 1) + ", util=" - + utils[altsToLog[k] - 1] + ", prob=" + probs[altsToLog[k] - 1]); - - soaTwoStageProbsLogger.info("total exponentiated utility = " + total); - soaTwoStageProbsLogger.info(""); - soaTwoStageProbsLogger.info(""); - - } - - tazDistProbs[i] = Arrays.copyOf(cm.getCumulativeProbabilities(), tazDistProbs.length); - - } - - for (int i = 0; i < tazDistProbs.length; i++) - { - soaTwoStageProbsLogger.info("orig=" + (i + 1) + ", dest=3738, cumProb[3737]=" - + tazDistProbs[i][3736] + ", cumProb[3738]=" + tazDistProbs[i][3737] - + ", prob[3738]=" + (tazDistProbs[i][3737] - tazDistProbs[i][3736])); - } - - return tazDistProbs; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java deleted file mode 100644 index 24299ac..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestChoiceTwoStageSoaTazDistanceUtilityDMU.java +++ /dev/null @@ -1,156 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class DestChoiceTwoStageSoaTazDistanceUtilityDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(DestChoiceTwoStageSoaTazDistanceUtilityDMU.class); - - protected HashMap methodIndexMap; - - protected IndexValues dmuIndex = null; - - protected double[] dcSize; - protected double[] univEnrollment; - protected double[] gsEnrollment; - protected double[] hsEnrollment; - protected double[] numHhs; - protected int[] gsDistricts; - protected int[] hsDistricts; - - public DestChoiceTwoStageSoaTazDistanceUtilityDMU() - { - } - - public void setIndexValuesObject(IndexValues index) - { - dmuIndex = index; - } - - public void setDestChoiceTazSize(double[] size) - { - dcSize = size; - } - - public void setTazUnivEnrollment(double[] enrollment) - { - univEnrollment = enrollment; - } - - public void setTazGsEnrollment(double[] enrollment) - { - gsEnrollment = enrollment; - } - - public void setTazHsEnrollment(double[] enrollment) - { - hsEnrollment = enrollment; - } - - public void setNumHhs(double[] hhs) - { - numHhs = hhs; - } - - public void setTazGsDistricts(int[] districts) - { - gsDistricts = districts; - } - - public void setTazHsDistricts(int[] districts) - { - hsDistricts = districts; - } - - public double getLnDestChoiceSizeTazAlt(int taz) - { - return dcSize[taz] == 0 ? -999 : Math.log(dcSize[taz]); - } - - public double getSizeTazAlt(int taz) - { - return dcSize[taz]; - } - - public double getUniversityEnrollmentTazAlt(int taz) - { - return univEnrollment[taz]; - } - - public double getGradeSchoolEnrollmentTazAlt(int taz) - { - return gsEnrollment[taz]; - } - - public double getHighSchoolEnrollmentTazAlt(int taz) - { - return hsEnrollment[taz]; - } - - public double getHouseholdsTazAlt(int taz) - { - return numHhs[taz]; - } - - public int getHomeTazGradeSchoolDistrict() - { - return gsDistricts[dmuIndex.getZoneIndex()]; - } - - public int getGradeSchoolDistrictTazAlt(int taz) - { - return gsDistricts[taz]; - } - - public int getHomeTazHighSchoolDistrict() - { - return hsDistricts[dmuIndex.getZoneIndex()]; - } - - public int getHighSchoolDistrictTazAlt(int taz) - { - return hsDistricts[taz]; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java deleted file mode 100644 index 9bb1941..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/DestinationSampleOfAlternativesModel.java +++ /dev/null @@ -1,612 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class DestinationSampleOfAlternativesModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(DestinationSampleOfAlternativesModel.class); - private transient Logger dcSoaLogger = Logger.getLogger("tourDcSoa"); - - // set to false to store probabilities in cache for re-use; true to disable - // probabilities cache. - private static final boolean ALWAYS_COMPUTE_PROBABILITIES = false; - private static final boolean ALLOW_DEBUG = true; - - private static final int DC_SOA_DATA_SHEET = 0; - private String dcSoaUecFileName; - private int sampleSize; - - private MgraDataManager mgraManager; - - private int currentOrigMgra; - private double[][] probabilitiesCache; - private double[][] cumProbabilitiesCache; - private int currentWorkMgra; - private double[][] subtourProbabilitiesCache; - private double[][] subtourCumProbabilitiesCache; - - // destsSample[] and destsAvailable[] are indexed by purpose and alternative - private boolean[] escortAvailable; - private int[] escortSample; - private boolean[][] destsAvailable; - private int[][] destsSample; - - private int[] sample; - private float[] corrections; - - private int[] dcSoaModelIndices; - private ChoiceModelApplication[] choiceModel; - - private int numberOfSoaChoiceAlternatives; - private int[] numberOfSoaChoiceAlternativesAvailable; - - private int soaProbabilitiesCalculationCount = 0; - private long soaRunTime = 0; - - public DestinationSampleOfAlternativesModel(String soaUecFile, int sampleSize, - HashMap propertyMap, MgraDataManager mgraManager, - double[][] dcSizeArray, DcSoaDMU dcSoaDmuObject, int[] soaUecIndices) - { - - this.sampleSize = sampleSize; - this.dcSoaUecFileName = soaUecFile; - this.mgraManager = mgraManager; - - // create an array of sample of alternative ChoiceModelApplication - // objects - // for each purpose - setupSampleOfAlternativesChoiceModelArrays(propertyMap, dcSizeArray, dcSoaDmuObject, - soaUecIndices); - - } - - private void setupSampleOfAlternativesChoiceModelArrays(HashMap propertyMap, - double[][] dcSizeArray, DcSoaDMU dcSoaDmuObject, int[] soaUecIndices) - { - - // create a HashMap to map purpose index to model index - dcSoaModelIndices = new int[soaUecIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int soaModelIndex = 0; - int sizeSegmentIndex = 0; - for (int uecIndex : soaUecIndices) - { - // if the uec sheet for the size segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, soaModelIndex); - dcSoaModelIndices[sizeSegmentIndex] = soaModelIndex++; - } else - { - dcSoaModelIndices[sizeSegmentIndex] = modelIndexMap.get(uecIndex); - } - - sizeSegmentIndex++; - } - // the value of soaModelIndex is the number of ChoiceModelApplication - // objects to create - // the modelIndexMap keys are the uec sheets to use in building - // ChoiceModelApplication objects - - choiceModel = new ChoiceModelApplication[modelIndexMap.size()]; - probabilitiesCache = new double[sizeSegmentIndex][]; - cumProbabilitiesCache = new double[sizeSegmentIndex][]; - subtourProbabilitiesCache = new double[sizeSegmentIndex][]; - subtourCumProbabilitiesCache = new double[sizeSegmentIndex][]; - - int i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - int modelIndex = -1; - try - { - modelIndex = modelIndexMap.get(uecIndex); - choiceModel[modelIndex] = new ChoiceModelApplication(dcSoaUecFileName, uecIndex, - DC_SOA_DATA_SHEET, propertyMap, (VariableTable) dcSoaDmuObject); - i++; - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up DC SOA ChoiceModelApplication[%d] for modelIndex=%d of %d models", - i, modelIndex, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - setAvailabilityForSampleOfAlternatives(dcSizeArray); - - } - - /** - * This method is called initially when the SOA choice models array is - * created. It would be called subsequently if a shadow pricing methodology - * is applied to reset the scaled size terms and corresponding - * availabilities and sample arrays. - */ - public void setAvailabilityForSampleOfAlternatives(double[][] dcSizeArray) - { - - int maxMgra = mgraManager.getMaxMgra(); - - // declare dimensions for the alternative availability array by purpose - // and - // number of alternaives - escortAvailable = new boolean[maxMgra + 1]; - escortSample = new int[maxMgra + 1]; - destsAvailable = new boolean[dcSizeArray.length][maxMgra + 1]; - destsSample = new int[dcSizeArray.length][maxMgra + 1]; - - numberOfSoaChoiceAlternativesAvailable = new int[dcSizeArray.length]; - - for (int i = 0; i < dcSizeArray.length; i++) - { - for (int k = 1; k <= maxMgra; k++) - { - if (dcSizeArray[i][k] > 0.0) - { - destsAvailable[i][k] = true; - destsSample[i][k] = 1; - numberOfSoaChoiceAlternativesAvailable[i]++; - } - } // k - } - - Arrays.fill(escortAvailable, true); - Arrays.fill(escortSample, 1); - - numberOfSoaChoiceAlternatives = maxMgra; - } - - public int getNumberOfAlternatives() - { - return numberOfSoaChoiceAlternatives; - } - - public void computeDestinationSampleOfAlternatives(DcSoaDMU dcSoaDmuObject, Tour tour, - Person person, String segmentName, int segmentIndex, int origMgra) - { - - long timeCheck = System.nanoTime(); - - // these will be dimensioned with the number of unique alternatives - // determined for the decision makers - int[] altList; - int[] altListFreq; - HashMap altFreqMap = new HashMap(); - - int modelIndex = dcSoaModelIndices[segmentIndex]; - - // if the flag is set to compute sample of alternative probabilities for - // every work/school location choice, - // or the tour's origin taz is different from the currentOrigTaz, reset - // the currentOrigTaz and clear the stored probabilities. - if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) - { - - if (ALWAYS_COMPUTE_PROBABILITIES || origMgra != currentWorkMgra) - { - - // clear the probabilities stored for the current origin mgra, - // for each DC segment - for (int i = 0; i < subtourProbabilitiesCache.length; i++) - { - subtourProbabilitiesCache[i] = null; - subtourCumProbabilitiesCache[i] = null; - } - currentWorkMgra = origMgra; - - } - - // If the sample of alternatives choice probabilities have not been - // computed for the current origin mgra - // and segment specified, compute them. - if (subtourProbabilitiesCache[segmentIndex] == null) - { - computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, - segmentName, segmentIndex, origMgra); - soaProbabilitiesCalculationCount++; - } - - } else if (tour != null - && tour.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) - { - - // always update probabilities for Escort tours - // clear the probabilities stored for the current origin mgra, for - // the Escort DC segment - probabilitiesCache[segmentIndex] = null; - cumProbabilitiesCache[segmentIndex] = null; - - destsAvailable[segmentIndex] = escortAvailable; - destsSample[segmentIndex] = escortSample; - - currentOrigMgra = origMgra; - - computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, - segmentName, segmentIndex, origMgra); - soaProbabilitiesCalculationCount++; - - } else - { - - if (ALWAYS_COMPUTE_PROBABILITIES || origMgra != currentOrigMgra) - { - - // clear the probabilities stored for the current origin mgra, - // for each DC segment - for (int i = 0; i < probabilitiesCache.length; i++) - { - probabilitiesCache[i] = null; - cumProbabilitiesCache[i] = null; - } - currentOrigMgra = origMgra; - - } - - // If the sample of alternatives choice probabilities have not been - // computed for the current origin taz - // and purpose specified, compute them. - if (probabilitiesCache[segmentIndex] == null) - { - computeSampleOfAlternativesChoiceProbabilities(dcSoaDmuObject, tour, person, - segmentName, segmentIndex, origMgra); - soaProbabilitiesCalculationCount++; - } - - } - - Household hhObj = person.getHouseholdObject(); - Random hhRandom = hhObj.getHhRandom(); - int rnCount = hhObj.getHhRandomCount(); - // when household.getHhRandom() was applied, the random count was - // incremented, assuming a random number would be drawn right away. - // so let's decrement by 1, then increment the count each time a random - // number is actually drawn in this method. - rnCount--; - - // select sampleSize alternatives based on probabilitiesList[origTaz], - // and - // count frequency of alternatives chosen. - // final sample may include duplicate alternative selections. - for (int i = 0; i < sampleSize; i++) - { - - double rn = hhRandom.nextDouble(); - rnCount++; - - int chosenAlt = -1; - if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) chosenAlt = Util - .binarySearchDouble(subtourCumProbabilitiesCache[segmentIndex], rn) + 1; - else chosenAlt = Util.binarySearchDouble(cumProbabilitiesCache[segmentIndex], rn) + 1; - - // write choice model alternative info to log file - if (hhObj.getDebugChoiceModels()) - { - choiceModel[modelIndex] - .logSelectionInfo( - String.format( - "Sample Of Alternatives Choice for segmentName=%s, segmentIndex=%d, modelIndex=%d, origMgra=%d", - segmentName, segmentIndex, modelIndex, origMgra), String - .format("HHID=%d, rn=%.8f, rnCount=%d", hhObj.getHhId(), - rn, (rnCount + i)), rn, chosenAlt); - } - - int freq = 0; - if (altFreqMap.containsKey(chosenAlt)) freq = altFreqMap.get(chosenAlt); - altFreqMap.put(chosenAlt, (freq + 1)); - - } - - // sampleSize random number draws were made from the Random object for - // the - // current household, - // so update the count in the hh's Random. - hhObj.setHhRandomCount(rnCount); - - // create arrays of the unique chosen alternatives and the frequency - // with - // which those alternatives were chosen. - int numUniqueAlts = altFreqMap.keySet().size(); - altList = new int[numUniqueAlts]; - altListFreq = new int[numUniqueAlts]; - Iterator it = altFreqMap.keySet().iterator(); - int k = 0; - while (it.hasNext()) - { - int key = (Integer) it.next(); - int value = (Integer) altFreqMap.get(key); - altList[k] = key; - altListFreq[k] = value; - k++; - } - - // loop through these arrays, construct final sample[] and - // corrections[]. - sample = new int[numUniqueAlts + 1]; - corrections = new float[numUniqueAlts + 1]; - for (k = 0; k < numUniqueAlts; k++) - { - int alt = altList[k]; - int freq = altListFreq[k]; - - double prob = 0; - if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) prob = subtourProbabilitiesCache[segmentIndex][alt - 1]; - else prob = probabilitiesCache[segmentIndex][alt - 1]; - - sample[k + 1] = alt; - corrections[k + 1] = (float) Math.log((double) freq / prob); - } - - soaRunTime += (System.nanoTime() - timeCheck); - - } - - /** - * This method is used if the tour/person decision maker is the first one - * encountered for the purpose and origin taz. Once the sample of - * alternatives choice probabilities for a purpose and origin taz are - * computed, they are stored in an array and used by other decision makers - * with the same purpose and origin taz. - * - * @param probabilitiesList - * is the probabilities array for the given purpose in which - * choice probabilities will be saved for the origin of the - * tour/place to be selected. - * @param cumProbabilitiesList - * is the probabilities array for the given purpose in which - * choice cumulative probabilities will be saved for the origin - * of the tour/place to be selected. - * @param choiceModel - * the ChoiceModelApplication object for the purpose - * @param tour - * the tour object for whic destination choice is required, or - * null if a usual work/school location is being chosen - * @param person - * the person object for whom the choice is being made - * @param segmentName - * the name of the segment the choice is being made for - for - * logging - * @param segmentindex - * the index associated with the segment - * @param origMgra - * the index associated with the segment - * @param distances - * array frpm origin MGRA to all MGRAs - */ - private void computeSampleOfAlternativesChoiceProbabilities(DcSoaDMU dcSoaDmuObject, Tour tour, - Person person, String segmentName, int segmentIndex, int origMgra) - { - - Household hhObj = person.getHouseholdObject(); - - // set the hh, person, and tour objects for this DMU object - dcSoaDmuObject.setHouseholdObject(hhObj); - dcSoaDmuObject.setPersonObject(person); - dcSoaDmuObject.setTourObject(tour); - - // set sample of alternatives choice DMU attributes - dcSoaDmuObject.setDmuIndexValues(hhObj.getHhId(), hhObj.getHhMgra(), origMgra, 0); - - // prepare a trace log header that the choiceModel object will write - // prior to - // UEC trace logging - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - - int choiceModelIndex = dcSoaModelIndices[segmentIndex]; - - // If the person making the choice is from a household requesting trace - // information, - // create a trace logger header and write prior to the choiceModel - // computing - // utilities - if (hhObj.getDebugChoiceModels()) - { - - if (tour == null) - { - // null tour means the SOA choice is for a mandatory usual - // location choice - choiceModelDescription = String.format( - "Usual Location Sample of Alternatives Choice Model for: Segment=%s", - segmentName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person - .getHouseholdObject().getHhId(), person.getPersonNum(), person - .getPersonType()); - } else - { - choiceModelDescription = String.format( - "Destination Choice Model for: Segment=%s, TourId=%d", segmentName, - tour.getTourId()); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person - .getHouseholdObject().getHhId(), person.getPersonNum(), person - .getPersonType()); - } - - // log headers to traceLogger if the person making the choice is - // from a - // household requesting trace information - choiceModel[choiceModelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - } - - try - { - choiceModel[choiceModelIndex].computeUtilities(dcSoaDmuObject, - dcSoaDmuObject.getDmuIndexValues(), destsAvailable[segmentIndex], - destsSample[segmentIndex]); - } catch (Exception e) - { - logger.error("exception caught in DC SOA model for:"); - choiceModelDescription = String.format( - "Destination Choice Model for: Segment=%s, TourId=%d", segmentName, - tour.getTourId()); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person - .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); - logger.error("choiceModelDescription:" + choiceModelDescription); - logger.error("decisionMakerLabel:" + decisionMakerLabel); - throw new RuntimeException(e); - } - - // TODO: debug - if (choiceModel[choiceModelIndex].getAvailabilityCount() == 0) - { - - int j = 0; - int[] debugAlts = new int[5]; - for (int i = 0; i < destsSample[segmentIndex].length; i++) - { - if (destsSample[segmentIndex][i] == 1) - { - debugAlts[j++] = i; - if (j == 5) break; - } - } - - choiceModel[choiceModelIndex].logUECResultsSpecificAlts(dcSoaLogger, "debugging", - debugAlts); - } - - // the following order of assignment is important in mult-threaded - // context. - // probabilitiesCache[][] is a trigger variable - if it is not null for - // any thread, the cumProbabilitiesCache[][] values - // are used immediately, so the cumProbabilitiesCache values must be - // assigned before the probabilitiesCache - // are assigned, which indicates cumProbabilitiesCache[][] values are - // ready to be used. - if (tour != null && tour.getTourCategory().equals(ModelStructure.AT_WORK_CATEGORY)) - { - - subtourCumProbabilitiesCache[segmentIndex] = Arrays.copyOf( - choiceModel[choiceModelIndex].getCumulativeProbabilities(), - choiceModel[choiceModelIndex].getNumberOfAlternatives()); - subtourProbabilitiesCache[segmentIndex] = Arrays.copyOf( - choiceModel[choiceModelIndex].getProbabilities(), - choiceModel[choiceModelIndex].getNumberOfAlternatives()); - } else - { - - cumProbabilitiesCache[segmentIndex] = Arrays.copyOf( - choiceModel[choiceModelIndex].getCumulativeProbabilities(), - choiceModel[choiceModelIndex].getNumberOfAlternatives()); - probabilitiesCache[segmentIndex] = Arrays.copyOf( - choiceModel[choiceModelIndex].getProbabilities(), - choiceModel[choiceModelIndex].getNumberOfAlternatives()); - - if (hhObj.getDebugChoiceModels()) - { - PrintWriter out = null; - try - { - out = new PrintWriter(new BufferedWriter(new FileWriter( - new File("soaProbs.csv")))); - - out.println("choiceModelDescription:" + choiceModelDescription); - out.println("decisionMakerLabel:" + decisionMakerLabel); - - for (int i = 0; i < probabilitiesCache[segmentIndex].length; i++) - { - out.println((i + 1) + "," + probabilitiesCache[segmentIndex][i]); - } - } catch (IOException e) - { - // TODO Auto-generated catch block - e.printStackTrace(); - } - - out.close(); - - } - - } - - // If the person making the choice is from a household requesting trace - // information, - // write choice model alternative info to the debug log file - if (hhObj.getDebugChoiceModels() && ALLOW_DEBUG) - { - // if ( dcSoaLogger.isDebugEnabled() ){ - int[] altsToLog = {0, 77, 78, 79, 80}; - choiceModel[choiceModelIndex].logAlternativesInfo(String.format( - "%s Sample Of Alternatives Choice for origTaz=%d", segmentName, origMgra), - String.format("HHID=%d", hhObj.getHhId()), dcSoaLogger); - choiceModel[choiceModelIndex].logUECResultsSpecificAlts(dcSoaLogger, - choiceModelDescription + ", " + decisionMakerLabel, altsToLog); - - double[] probs = choiceModel[choiceModelIndex].getProbabilities(); - double[] utils = choiceModel[choiceModelIndex].getUtilities(); - double total = 0; - for (int i = 0; i < probs.length; i++) - total += Math.exp(utils[i]); - - dcSoaLogger.info(""); - for (int i = 1; i < altsToLog.length; i++) - dcSoaLogger.info("alt=" + (altsToLog[i] + 1) + ", util=" + utils[altsToLog[i]] - + ", prob=" + probs[altsToLog[i]]); - - dcSoaLogger.info("total exponentiated utility = " + total); - dcSoaLogger.info(""); - dcSoaLogger.info(""); - // } - } - - } - - public int getSoaProbabilitiesCalculationCount() - { - return soaProbabilitiesCalculationCount; - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - - public int getCurrentOrigMgra() - { - return currentOrigMgra; - } - - public int[] getSampleOfAlternatives() - { - return sample; - } - - public float[] getSampleOfAlternativesCorrections() - { - return corrections; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java deleted file mode 100644 index ba5b011..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Household.java +++ /dev/null @@ -1,1847 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Random; - -import org.apache.log4j.Logger; - -public class Household - implements java.io.Serializable -{ - - private boolean debugChoiceModels; - - private int hhId; - private short hhIncomeCategory; - private int hhIncomeInDollars; - private short hhSize; - private short hhType; - private short unitType; - private short hhBldgsz; - private short hhWorkers; - - private int homeTaz; - private int homeMgra; - private int homeWalkSubzone; - - //added for BCA tool - private float autoOwnershipLogsum; - private float transponderLogsum; - private float cdapLogsum; - private float jtfLogsum; - - private Person[] persons; - - private Tour[] jointTours; - - private short aoModelAutos; - private short automatedVehicles; - private short conventionalVehicles; - private String cdapModelPattern; - private short imtfModelPattern; - private String jtfModelPattern; - private short tpChoice; - private short outboundEscortChoice; - private short inboundEscortChoice; - - - private Random hhRandom; - private int randomCount = 0; - private HashMap uwslRandomCountList; - private int preAoRandomCount; - private int aoRandomCount; - private int tpRandomCount; - private int fpRandomCount; - private int ieRandomCount; - private int cdapRandomCount; - private int imtfRandomCount; - private int imtodRandomCount; - private int awfRandomCount; - private int awlRandomCount; - private int awtodRandomCount; - private int jtfRandomCount; - private int jtlRandomCount; - private int jtodRandomCount; - private int inmtfRandomCount; - private int inmtlRandomCount; - private int inmtodRandomCount; - private int stfRandomCount; - private int stlRandomCount; - - private int maxAdultOverlaps; - private int maxChildOverlaps; - private int maxMixedOverlaps; - - private ModelStructure modelStructure; - - private long seed; - public Household(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - hhRandom = new Random(); - uwslRandomCountList = new HashMap(); - } - - /** - * Returns a 1-based array of persons in the household - * @return Person array - */ - public Person[] getPersons() - { - return persons; - } - - public void initializeWindows() - { - - // loop through the person array (1-based) - for (int i = 1; i < persons.length; ++i) - { - persons[i].initializeWindows(); - } - - } - - public void setDebugChoiceModels(boolean value) - { - debugChoiceModels = value; - } - - public void setHhId(int id, int baseSeed) - { - hhId = id; - randomCount = 0; - hhRandom.setSeed(baseSeed + hhId); - } - - public void setRandomObject(Random r) - { - hhRandom = r; - } - - public void setHhRandomCount(int count) - { - randomCount = count; - } - - // work/school location choice uses shadow pricing, so save randomCount per - // iteration - public void setUwslRandomCount(int iter, int count) - { - uwslRandomCountList.put(iter, count); - } - - public void setPreAoRandomCount(int count) - { - preAoRandomCount = count; - } - - public void setAoRandomCount(int count) - { - aoRandomCount = count; - } - - public void setTpRandomCount(int count) - { - tpRandomCount = count; - } - - public void setFpRandomCount(int count) - { - fpRandomCount = count; - } - - public void setIeRandomCount(int count) - { - ieRandomCount = count; - } - - public void setCdapRandomCount(int count) - { - cdapRandomCount = count; - } - - public void setImtfRandomCount(int count) - { - imtfRandomCount = count; - } - - public void setImtodRandomCount(int count) - { - imtodRandomCount = count; - } - - public void setAwfRandomCount(int count) - { - awfRandomCount = count; - } - - public void setAwlRandomCount(int count) - { - awlRandomCount = count; - } - - public void setAwtodRandomCount(int count) - { - awtodRandomCount = count; - } - - public void setJtfRandomCount(int count) - { - jtfRandomCount = count; - } - - public void setJtlRandomCount(int count) - { - jtlRandomCount = count; - } - - public void setJtodRandomCount(int count) - { - jtodRandomCount = count; - } - - public void setInmtfRandomCount(int count) - { - inmtfRandomCount = count; - } - - public void setInmtlRandomCount(int count) - { - inmtlRandomCount = count; - } - - public void setInmtodRandomCount(int count) - { - inmtodRandomCount = count; - } - - public void setStfRandomCount(int count) - { - stfRandomCount = count; - } - - public void setStlRandomCount(int count) - { - stlRandomCount = count; - } - - public void setHhTaz(int taz) - { - homeTaz = taz; - } - - public void setHhMgra(int mgra) - { - homeMgra = mgra; - } - - public void setHhWalkSubzone(int subzone) - { - homeWalkSubzone = (short) subzone; - } - - public void setHhAutos(int autos) - { - // this sets the variable that will be used in work/school location - // choice. - // after auto ownership runs, this variable gets updated with number of - // autos - // for result. - aoModelAutos = (short) autos; - } - - public void setTpChoice(int value) - { - tpChoice = (short) value; - } - - /** - * auto sufficiency: 1 if cars < workers, 2 if cars equal workers, 3 if cars - * > workers - * - * @return auto sufficiency value - */ - public int getAutoSufficiency() - { - if (aoModelAutos < hhWorkers) return 1; - else if (aoModelAutos == hhWorkers) return 2; - else return 3; - } - - public int getAutosOwned() - { - return (int) aoModelAutos; - } - - public int getAutomatedVehicles() { - return (int) automatedVehicles; - } - - public void setAutomatedVehicles(int automatedVehicles) { - this.automatedVehicles = (short) automatedVehicles; - } - - public int getConventionalVehicles() { - return (int) conventionalVehicles; - } - - public void setConventionalVehicles(int conventionalVehicles) { - this.conventionalVehicles = (short) conventionalVehicles; - } - - public int getTpChoice() - { - return (int) tpChoice; - } - - public void setCoordinatedDailyActivityPatternResult(String pattern) - { - cdapModelPattern = pattern; - } - - public String getCoordinatedDailyActivityPattern() - { - return cdapModelPattern; - } - - public void setJointTourFreqResult(int altIndex, String altName) - { - jtfModelPattern = String.format("%d_%s", altIndex, altName); - } - - public int getJointTourFreqChosenAlt() - { - int returnValue = 0; - if (jtfModelPattern == null) - { - returnValue = 0; - } else - { - int endIndex = jtfModelPattern.indexOf('_'); - returnValue = Integer.parseInt(jtfModelPattern.substring(0, endIndex)); - } - return returnValue; - } - - public String getJointTourFreqChosenAltName() - { - String returnValue = "none"; - if (jtfModelPattern != null) - { - int startIndex = jtfModelPattern.indexOf('_') + 1; - returnValue = jtfModelPattern.substring(startIndex); - } - return returnValue; - } - - public void setHhBldgsz(int code) - { - hhBldgsz = (short) code; - } - - public int getHhBldgsz() - { - return (int) hhBldgsz; - } - - public void setHhSize(int numPersons) - { - hhSize = (short) numPersons; - persons = new Person[numPersons + 1]; - for (int i = 1; i <= numPersons; i++) - persons[i] = new Person(this, i, modelStructure); - - } - - public void setHhIncomeCategory(int category) - { - hhIncomeCategory = (short) category; - } - - public void setHhIncomeInDollars(int dollars) - { - hhIncomeInDollars = dollars; - } - - public void setHhWorkers(int numWorkers) - { - hhWorkers = (short) numWorkers; - } - - public void setHhType(int type) - { - hhType = (short) type; - } - - // 0=Housing unit, 1=Institutional group quarters, 2=Noninstitutional group - // quarters - public void setUnitType(int type) - { - unitType = (short) type; - } - - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - public int getHhSize() - { - return (int) hhSize; - } - - public int getNumTotalIndivTours() - { - int count = 0; - for (int i = 1; i < persons.length; i++) - count += persons[i].getNumTotalIndivTours(); - return count; - } - - public int getNumberOfNonWorkingAdults() - { - int count = 0; - for (int i = 1; i < persons.length; i++) - count += persons[i].getPersonIsNonWorkingAdultUnder65() - + persons[i].getPersonIsNonWorkingAdultOver65(); - return count; - } - - public int getIsNonFamilyHousehold() - { - - if (hhType == HouseholdType.NON_FAMILY_MALE_ALONE.ordinal()) return (1); - if (hhType == HouseholdType.NON_FAMILY_MALE_NOT_ALONE.ordinal()) return (1); - if (hhType == HouseholdType.NON_FAMILY_FEMALE_ALONE.ordinal()) return (1); - if (hhType == HouseholdType.NON_FAMILY_FEMALE_NOT_ALONE.ordinal()) return (1); - - return (0); - } - - /** - * unitType: 0=Housing unit, 1=Institutional group quarters, - * 2=Noninstitutional group quarters - * - * @return 1 if household is group quarters, 0 for non-group quarters - */ - public int getIsGroupQuarters() - { - if (unitType == 0) return 0; - else return 1; - } - - public int getNumStudents() - { - int count = 0; - for (int i = 1; i < persons.length; ++i) - { - count += persons[i].getPersonIsStudent(); - } - return (count); - } - - public int getNumGradeSchoolStudents() - { - int count = 0; - for (int i = 1; i < persons.length; ++i) - { - count += persons[i].getPersonIsGradeSchool(); - } - return (count); - } - - public int getNumHighSchoolStudents() - { - int count = 0; - for (int i = 1; i < persons.length; ++i) - { - count += persons[i].getPersonIsHighSchool(); - } - return (count); - } - - public int getNumberOfChildren6To18WithoutMandatoryActivity() - { - - int count = 0; - - for (int i = 1; i < persons.length; ++i) - { - count += persons[i].getPersonIsChild6To18WithoutMandatoryActivity(); - } - - return (count); - } - - public int getNumberOfPreDrivingWithNonHomeActivity() - { - - int count = 0; - for (int i = 1; i < persons.length; ++i) - { - // count only predrving kids - if (persons[i].getPersonIsStudentDriving() == 1) - { - // count only if CDAP is M or N (i.e. not H) - if (!persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) - count++; - } - } - - return count; - } - - public int getNumberOfPreschoolWithNonHomeActivity() - { - - int count = 0; - for (int i = 1; i < persons.length; ++i) - { - // count only predrving kids - if (persons[i].getPersonIsPreschoolChild() == 1) - { - // count only if CDAP is M or N (i.e. not H) - if (!persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) - count++; - } - } - - return count; - } - - /** - * return the number of school age students this household has for the - * purpose index. - * - * @param purposeIndex - * is the DC purpose index to be compared to the usual school - * location index saved for this person upon reading synthetic - * population file. - * @return num, a value of the number of school age students in the - * household for this purpose index. - */ - public int getNumberOfDrivingAgedStudentsWithDcPurposeIndex(int segmentIndex) - { - int num = 0; - for (int j = 1; j < persons.length; j++) - { - if (persons[j].getPersonIsStudentDriving() == 1 - && persons[j].getSchoolLocationSegmentIndex() == segmentIndex) num++; - } - return num; - } - - public int getNumberOfNonDrivingAgedStudentsWithDcPurposeIndex(int segmentIndex) - { - int num = 0; - for (int j = 1; j < persons.length; j++) - { - if (persons[j].getPersonIsStudentNonDriving() == 1 - || persons[j].getPersonIsPreschoolChild() == 1 - && persons[j].getSchoolLocationSegmentIndex() == segmentIndex) num++; - } - return num; - } - - public Person getPerson(int persNum) - { - if (persNum < 1 || persNum > hhSize) - { - throw new RuntimeException(String.format( - "persNum value = %d is out of range for hhSize = %d", persNum, hhSize)); - } - - return persons[persNum]; - } - - // methods DMU will use to get info from household object - - public int getHhId() - { - return hhId; - } - - public Random getHhRandom() - { - randomCount++; - return hhRandom; - } - - public int getHhRandomCount() - { - return randomCount; - } - - public int getUwslRandomCount(int iter) - { - return uwslRandomCountList.get(iter); - } - - public int getPreAoRandomCount() - { - return preAoRandomCount; - } - - public int getAoRandomCount() - { - return aoRandomCount; - } - - public int getTpRandomCount() - { - return tpRandomCount; - } - - public int getFpRandomCount() - { - return fpRandomCount; - } - - public int getIeRandomCount() - { - return ieRandomCount; - } - - public int getCdapRandomCount() - { - return cdapRandomCount; - } - - public int getImtfRandomCount() - { - return imtfRandomCount; - } - - public int getImtodRandomCount() - { - return imtodRandomCount; - } - - public int getJtfRandomCount() - { - return jtfRandomCount; - } - - public int getAwfRandomCount() - { - return awfRandomCount; - } - - public int getAwlRandomCount() - { - return awlRandomCount; - } - - public int getAwtodRandomCount() - { - return awtodRandomCount; - } - - public int getJtlRandomCount() - { - return jtlRandomCount; - } - - public int getJtodRandomCount() - { - return jtodRandomCount; - } - - public int getInmtfRandomCount() - { - return inmtfRandomCount; - } - - public int getInmtlRandomCount() - { - return inmtlRandomCount; - } - - public int getInmtodRandomCount() - { - return inmtodRandomCount; - } - - public int getStfRandomCount() - { - return stfRandomCount; - } - - public int getStlRandomCount() - { - return stlRandomCount; - } - - public int getHhTaz() - { - return homeTaz; - } - - public int getHhMgra() - { - return homeMgra; - } - - public int getHhWalkSubzone() - { - return homeWalkSubzone; - } - - public int getIncomeCategory() - { - return (int) hhIncomeCategory; - } - - public int getIncomeInDollars() - { - return hhIncomeInDollars; - } - - public int getWorkers() - { - return (int) hhWorkers; - } - - public int getDrivers() - { - return getNumPersons16plus(); - } - - public int getSize() - { - return (int) hhSize; - } - - public int getChildunder16() - { - if (getNumChildrenUnder16() > 0) return 1; - else return 0; - } - - public int getChild16plus() - { - if (getNumPersons16plus() > 0) return 1; - else return 0; - } - - public int getNumChildrenUnder16() - { - int numChildrenUnder16 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() < 16) numChildrenUnder16++; - } - return numChildrenUnder16; - } - - public int getNumChildrenUnder19() - { - int numChildrenUnder19 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() < 19) numChildrenUnder19++; - } - return numChildrenUnder19; - } - - public int getNumPersons0to4() - { - int numPersons0to4 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() < 5) numPersons0to4++; - } - return numPersons0to4; - } - - /** - * used in AO choice utility - * - * @return number of persons age 6 to 15, inclusive - */ - public int getNumPersons6to15() - { - int numPersons6to15 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 6 && persons[i].getAge() <= 15) numPersons6to15++; - } - return numPersons6to15; - } - - /** - * used in Stop Frequency choice utility - * - * @return number of persons age 5 to 15, inclusive - */ - public int getNumPersons5to15() - { - int numPersons5to15 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 5 && persons[i].getAge() <= 15) numPersons5to15++; - } - return numPersons5to15; - } - - public int getNumPersons16to17() - { - int numPersons16to17 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 16 && persons[i].getAge() <= 17) numPersons16to17++; - } - return numPersons16to17; - } - - public int getNumPersons18to35(){ - - int numPersons18to35 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 18 && persons[i].getAge() <= 35) numPersons18to35++; - } - return numPersons18to35; - } - - public int getNumPersons16plus() - { - int numPersons16plus = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 16) numPersons16plus++; - } - return numPersons16plus; - } - - public int getNumPersons18plus() - { - int numPersons18plus = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 18) numPersons18plus++; - } - return numPersons18plus; - } - - public int getNumPersons80plus() - { - int numPersons80plus = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 80) numPersons80plus++; - } - return numPersons80plus; - } - - public int getNumPersons18to24() - { - int numPersons18to24 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 18 && persons[i].getAge() <= 24) numPersons18to24++; - } - return numPersons18to24; - } - - public int getNumPersons65to79() - { - int numPersons65to79 = 0; - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getAge() >= 65 && persons[i].getAge() <= 79) numPersons65to79++; - } - return numPersons65to79; - } - - public int getNumFtWorkers() - { - int numFtWorkers = 0; - for (int i = 1; i < persons.length; i++) - numFtWorkers += persons[i].getPersonIsFullTimeWorker(); - return numFtWorkers; - } - - public int getNumPtWorkers() - { - int numPtWorkers = 0; - for (int i = 1; i < persons.length; i++) - numPtWorkers += persons[i].getPersonIsPartTimeWorker(); - return numPtWorkers; - } - - public int getNumUnivStudents() - { - int numUnivStudents = 0; - for (int i = 1; i < persons.length; i++) - numUnivStudents += persons[i].getPersonIsUniversityStudent(); - return numUnivStudents; - } - - public int getNumNonWorkAdults() - { - int numNonWorkAdults = 0; - for (int i = 1; i < persons.length; i++) - numNonWorkAdults += persons[i].getPersonIsNonWorkingAdultUnder65(); - return numNonWorkAdults; - } - - public int getNumAdults() - { - int numAdults = 0; - for (int i = 1; i < persons.length; i++) - numAdults += (persons[i].getPersonIsFullTimeWorker() - + persons[i].getPersonIsPartTimeWorker() - + persons[i].getPersonIsUniversityStudent() - + persons[i].getPersonIsNonWorkingAdultUnder65() + persons[i] - .getPersonIsNonWorkingAdultOver65()); - return numAdults; - } - - public int getNumRetired() - { - int numRetired = 0; - for (int i = 1; i < persons.length; i++) - numRetired += persons[i].getPersonIsNonWorkingAdultOver65(); - return numRetired; - } - - public int getNumDrivingStudents() - { - int numDrivingStudents = 0; - for (int i = 1; i < persons.length; i++) - numDrivingStudents += persons[i].getPersonIsStudentDriving(); - return numDrivingStudents; - } - - public int getNumNonDrivingStudents() - { - int numNonDrivingStudents = 0; - for (int i = 1; i < persons.length; i++) - numNonDrivingStudents += persons[i].getPersonIsStudentNonDriving(); - return numNonDrivingStudents; - } - - public int getNumPreschool() - { - int numPreschool = 0; - for (int i = 1; i < persons.length; i++) - numPreschool += persons[i].getPersonIsPreschoolChild(); - return numPreschool; - } - - public int getNumHighSchoolGraduates() - { - int numGrads = 0; - for (int i = 1; i < persons.length; i++) - numGrads += persons[i].getPersonIsHighSchoolGraduate(); - return numGrads; - } - - /** - * Iterates through person array, adds up and returns number of children with school tours for preschool children, non-driving age students, and driving age students - * @return Number of children with school tours. - */ - public int getNumChildrenWithSchoolTours(){ - - int numChildrenWithSchoolTours = 0; - - for(int i=1; i < persons.length;++i){ - Person p = persons[i]; - if((p.getPersonIsPreschoolChild() == 1) || (p.getPersonIsStudentNonDriving() == 1)|| (p.getPersonIsStudentDriving() == 1)){ - if(p.getNumSchoolTours()>0) - ++numChildrenWithSchoolTours; - } - } - - return numChildrenWithSchoolTours; - } - /** - * joint tour frequency choice is not applied to a household unless it has: - * 2 or more persons, each with at least one out-of home activity, and at - * least 1 of the persons not a pre-schooler. - * */ - public int getValidHouseholdForJointTourFrequencyModel() - { - - // return one of the following condition codes for this household - // producing - // joint tours: - // 1: household eligible for joint tour production - // 2: household ineligible due to 1 person hh. - // 3: household ineligible due to fewer than 2 persons traveling out of - // home - // 4: household ineligible due to fewer than 1 non-preschool person - // traveling - // out of home - - // no joint tours for single person household - if (hhSize == 1) return 2; - - int leavesHome = 0; - int nonPreSchoolerLeavesHome = 0; - for (int i = 1; i < persons.length; i++) - { - if (!persons[i].getCdapActivity().equalsIgnoreCase("H")) - { - leavesHome++; - if (persons[i].getPersonIsPreschoolChild() == 0) nonPreSchoolerLeavesHome++; - } - } - - // if the number of persons leaving home during the day is not at least - // 2, no - // joint tours - if (leavesHome < 2) return 3; - - // if the number of non-preschool persons leaving home during the day is - // not - // at least 1, no joint tours - if (nonPreSchoolerLeavesHome < 1) return 4; - - // if all conditions are met, we can apply joint tour frequency model to - // this - // household - return 1; - - } - - /** - * return maximum periods of overlap between this person and other adult - * persons in the household. - * - * @return the most number of periods mutually available between this person - * and other adult household members - */ - public int getMaxAdultOverlaps() - { - return maxAdultOverlaps; - } - - /** - * return maximum periods of overlap between this person and other children - * in the household. - * - * @return the most number of periods mutually available between this person - * and other child household members - */ - public int getMaxChildOverlaps() - { - return maxChildOverlaps; - } - - /** - * return maximum periods of overlap between this person(adult/child) and - * other persons(child/adult) in the household. - * - * @return the most number of periods mutually available between this person - * and other type household members - */ - public int getMaxMixedOverlaps() - { - return maxMixedOverlaps; - } - - public int getMaxJointTimeWindow(Tour t) - { - // get array of person array indices participating in joint tour - int[] participatingPersonIndices = t.getPersonNumArray(); - - // create an array to hold time window arrays for each participant - short[][] personWindows = new short[participatingPersonIndices.length][]; - - // get time window arrays for each participant - int k = 0; - for (int i : participatingPersonIndices) - personWindows[k++] = persons[i].getTimeWindows(); - - int count = 0; - - int maxCount = 0; - // loop over time window intervals - for (int w = 1; w < personWindows[0].length; w++) - { - - // loop over party; determine if interval is available for everyone - // in party; - boolean available = true; - for (k = 0; k < personWindows.length; k++) - { - if (personWindows[k][w] > 0) - { - available = false; - break; - } - } - - // if available for whole party, increment count; determine maximum - // continous time window available to whole party. - if (available) - { - count++; - if (count > maxCount) maxCount = count; - } else - { - count = 0; - } - - } - - return maxCount; - } - - /** - * @return number of adults in household with "M" or "N" activity pattern - - * that is, traveling adults. - */ - public int getTravelActiveAdults() - { - - int adultsStayingHome = 0; - int adults = 0; - for (int p = 1; p < persons.length; p++) - { - // person is an adult - if (persons[p].getPersonIsAdult() == 1) - { - adults++; - if (persons[p].getCdapActivity().equalsIgnoreCase("H")) adultsStayingHome++; - } - } - - // return the number of adults traveling = number of adults minus the - // number - // of adults staying home. - return adults - adultsStayingHome; - - } - - /** - * @return number of children in household with "M" or "N" activity pattern - * - that is, traveling children. - */ - public int getTravelActiveChildren() - { - - int childrenStayingHome = 0; - int children = 0; - for (int p = 1; p < persons.length; p++) - { - // person is not an adult - if (persons[p].getPersonIsAdult() == 0) - { - children++; - if (persons[p].getCdapActivity().equalsIgnoreCase("H")) childrenStayingHome++; - } - } - - // return the number of adults traveling = number of adults minus the - // number - // of adults staying home. - return children - childrenStayingHome; - - } - - public int getOutboundEscortChoice() { - return outboundEscortChoice; - } - public void setOutboundEscortChoice(int outboundEscortChoice) { - this.outboundEscortChoice = (short) outboundEscortChoice; - } - public int getInboundEscortChoice() { - return inboundEscortChoice; - } - public void setInboundEscortChoice(int inboundEscortChoice) { - this.inboundEscortChoice = (short) inboundEscortChoice; - } - public void calculateTimeWindowOverlaps() - { - - boolean pAdult; - boolean qAdult; - - maxAdultOverlaps = 0; - maxChildOverlaps = 0; - maxMixedOverlaps = 0; - - int[] maxAdultOverlapsP = new int[persons.length]; - int[] maxChildOverlapsP = new int[persons.length]; - - // loop over persons in the household and count available time windows - for (int p = 1; p < persons.length; p++) - { - - // determine if person p is an adult -- that is, person is not any - // of the - // three child types - pAdult = persons[p].getPersonIsPreschoolChild() == 0 - && persons[p].getPersonIsStudentNonDriving() == 0 - && persons[p].getPersonIsStudentDriving() == 0; - - // loop over person indices to compute length of pairwise available - // time windows. - for (int q = 1; q < persons.length; q++) - { - - if (p == q) continue; - - // determine if person q is an adult -- that is, person is not - // any of the three child types - qAdult = persons[q].getPersonIsPreschoolChild() == 0 - && persons[q].getPersonIsStudentNonDriving() == 0 - && persons[q].getPersonIsStudentDriving() == 0; - - // get the length of the maximum pairwise available time window - // between persons p and q. - int maxWindow = persons[p].getMaximumContinuousPairwiseAvailableWindow(persons[q] - .getTimeWindows()); - - // determine max time window overlap between adult pairs, - // children pairs, and mixed pairs in the household - // for max windows in all pairs in hh, don't need to check q,p - // once we'alread done p,q, so skip q <= p. - if (q > p) - { - if (pAdult && qAdult) - { - if (maxWindow > maxAdultOverlaps) maxAdultOverlaps = maxWindow; - } else if (!pAdult && !qAdult) - { - if (maxWindow > maxChildOverlaps) maxChildOverlaps = maxWindow; - } else - { - if (maxWindow > maxMixedOverlaps) maxMixedOverlaps = maxWindow; - } - } - - // determine the max time window overlap between this person and - // other household adults and children. - if (qAdult) - { - if (maxWindow > maxAdultOverlapsP[p]) maxAdultOverlapsP[p] = maxWindow; - } else - { - if (maxWindow > maxChildOverlapsP[p]) maxChildOverlapsP[p] = maxWindow; - } - - } // end of person q - - // set person attributes - persons[p].setMaxAdultOverlaps(maxAdultOverlapsP[p]); - persons[p].setMaxChildOverlaps(maxChildOverlapsP[p]); - - } // end of person p - - } - - public boolean[] getAvailableJointTourTimeWindows(Tour t, int[] altStarts, int[] altEnds) - { - int[] participatingPersonIndices = t.getPersonNumArray(); - - // availability array for each person - boolean[][] availability = new boolean[participatingPersonIndices.length][]; - - for (int i = 0; i < participatingPersonIndices.length; i++) - { - - int personNum = participatingPersonIndices[i]; - Person person = persons[personNum]; - - // availability array is 1-based indexing - availability[i] = new boolean[altStarts.length + 1]; - - for (int k = 1; k <= altStarts.length; k++) - { - int start = altStarts[k - 1]; - int end = altEnds[k - 1]; - availability[i][k] = person.isWindowAvailable(start, end); - } - - } - - boolean[] jointAvailability = new boolean[availability[0].length]; - - for (int k = 0; k < jointAvailability.length; k++) - { - jointAvailability[k] = true; - for (int i = 0; i < participatingPersonIndices.length; i++) - { - if (!availability[i][k]) - { - jointAvailability[k] = false; - break; - } - } - } - - return jointAvailability; - - } - - public void scheduleJointTourTimeWindows(Tour t, int start, int end) - { - int[] participatingPersonIndices = t.getPersonNumArray(); - for (int i : participatingPersonIndices) - { - Person person = persons[i]; - person.scheduleWindow(start, end); - } - } - - public void createJointTourArray() - { - jointTours = new Tour[0]; - } - - public void createJointTourArray(Tour tour1) - { - jointTours = new Tour[1]; - tour1.setTourOrigMgra(homeMgra); - tour1.setTourDestMgra(0); - jointTours[0] = tour1; - } - - public void createJointTourArray(Tour tour1, Tour tour2) - { - jointTours = new Tour[2]; - tour1.setTourOrigMgra(homeMgra); - tour1.setTourDestMgra(0); - tour1.setTourId(0); - tour2.setTourOrigMgra(homeMgra); - tour2.setTourDestMgra(0); - tour2.setTourId(1); - jointTours[0] = tour1; - jointTours[1] = tour2; - } - - public Tour[] getJointTourArray() - { - return jointTours; - } - - public void initializeForAoRestart() - { - jointTours = null; - - aoModelAutos = 0; - cdapModelPattern = null; - imtfModelPattern = 0; - jtfModelPattern = null; - - tpRandomCount = 0; - fpRandomCount = 0; - ieRandomCount = 0; - cdapRandomCount = 0; - imtfRandomCount = 0; - imtodRandomCount = 0; - awfRandomCount = 0; - awlRandomCount = 0; - awtodRandomCount = 0; - jtfRandomCount = 0; - jtlRandomCount = 0; - jtodRandomCount = 0; - inmtfRandomCount = 0; - inmtlRandomCount = 0; - inmtodRandomCount = 0; - stfRandomCount = 0; - stlRandomCount = 0; - - maxAdultOverlaps = 0; - maxChildOverlaps = 0; - maxMixedOverlaps = 0; - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForAoRestart(); - - } - - public void initializeForImtfRestart() - { - jointTours = null; - - imtfModelPattern = 0; - jtfModelPattern = null; - - imtodRandomCount = 0; - jtfRandomCount = 0; - jtlRandomCount = 0; - jtodRandomCount = 0; - inmtfRandomCount = 0; - inmtlRandomCount = 0; - inmtodRandomCount = 0; - awfRandomCount = 0; - awlRandomCount = 0; - awtodRandomCount = 0; - stfRandomCount = 0; - stlRandomCount = 0; - - maxAdultOverlaps = 0; - maxChildOverlaps = 0; - maxMixedOverlaps = 0; - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForImtfRestart(); - - } - - public void initializeForJtfRestart() - { - - jtfModelPattern = null; - - jtfRandomCount = 0; - jtlRandomCount = 0; - jtodRandomCount = 0; - inmtfRandomCount = 0; - inmtlRandomCount = 0; - inmtodRandomCount = 0; - awfRandomCount = 0; - awlRandomCount = 0; - awtodRandomCount = 0; - stfRandomCount = 0; - stlRandomCount = 0; - - initializeWindows(); - - if (jointTours != null) - { - for (Tour t : jointTours) - { - t.clearStopModelResults(); - } - } - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForJtfRestart(); - - jointTours = null; - - } - - public void initializeForInmtfRestart() - { - - inmtfRandomCount = 0; - inmtlRandomCount = 0; - inmtodRandomCount = 0; - awfRandomCount = 0; - awlRandomCount = 0; - awtodRandomCount = 0; - stfRandomCount = 0; - stlRandomCount = 0; - - initializeWindows(); - - if (jointTours != null) - { - for (Tour t : jointTours) - { - for (int i : t.getPersonNumArray()) - persons[i].scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - } - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForInmtfRestart(); - - } - - public void initializeForAwfRestart() - { - - awfRandomCount = 0; - awlRandomCount = 0; - awtodRandomCount = 0; - stfRandomCount = 0; - stlRandomCount = 0; - - initializeWindows(); - - if (jointTours != null) - { - for (Tour t : jointTours) - { - for (int i : t.getPersonNumArray()) - persons[i].scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - } - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForAwfRestart(); - - } - - public void initializeForStfRestart() - { - - stfRandomCount = 0; - stlRandomCount = 0; - - for (int i = 1; i < persons.length; i++) - persons[i].initializeForStfRestart(); - - } - - public long getSeed() { - return seed; - } - public void setSeed(long seed) { - this.seed = seed; - } - - public float getAutoOwnershipLogsum() { - return autoOwnershipLogsum; - } - - public void setAutoOwnershipLogsum(float autoOwnershipLogsum) { - this.autoOwnershipLogsum = autoOwnershipLogsum; - } - - public float getTransponderLogsum() { - return transponderLogsum; - } - - public void setTransponderLogsum(float transponderLogsum) { - this.transponderLogsum = transponderLogsum; - } - - public float getCdapLogsum() { - return cdapLogsum; - } - - public void setCdapLogsum(float cdapLogsum) { - this.cdapLogsum = cdapLogsum; - } - - public float getJtfLogsum() { - return jtfLogsum; - } - - public void setJtfLogsum(float jtfLogsum) { - this.jtfLogsum = jtfLogsum; - } - - public void logHouseholdObject(String titleString, Logger logger) - { - - int totalChars = 72; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "H"; - - logger.info(separater); - logger.info(titleString); - logger.info(separater); - - Household.logHelper(logger, "hhId: ", hhId, totalChars); - Household.logHelper(logger, "debugChoiceModels: ", debugChoiceModels ? "True" : "False", - totalChars); - Household.logHelper(logger, "hhIncome: ", hhIncomeCategory, totalChars); - Household.logHelper(logger, "hhIncomeInDollars: ", hhIncomeInDollars, totalChars); - Household.logHelper(logger, "hhSize: ", hhSize, totalChars); - Household.logHelper(logger, "hhType: ", hhType, totalChars); - Household.logHelper(logger, "hhWorkers: ", hhWorkers, totalChars); - Household.logHelper(logger, "homeTaz: ", homeTaz, totalChars); - Household.logHelper(logger, "homeMgra: ", homeMgra, totalChars); - Household.logHelper(logger, "homeWalkSubzone: ", homeWalkSubzone, totalChars); - Household.logHelper(logger, "aoModelAutos: ", aoModelAutos, totalChars); - Household.logHelper(logger, "cdapModelPattern: ", cdapModelPattern, totalChars); - Household.logHelper(logger, "imtfModelPattern: ", imtfModelPattern, totalChars); - Household.logHelper(logger, "outboundEscortChoice: ", outboundEscortChoice, totalChars); - Household.logHelper(logger, "inboundEscortChoice: ", inboundEscortChoice, totalChars); - Household.logHelper(logger, "jtfModelPattern: ", jtfModelPattern, totalChars); - Household.logHelper(logger, "randomCount: ", randomCount, totalChars); - if (uwslRandomCountList.size() > 0) - { - for (int i : uwslRandomCountList.keySet()) - Household.logHelper(logger, String.format("uwslRandomCount[%d]: ", i), - uwslRandomCountList.get(i), totalChars); - } else - { - Household.logHelper(logger, "uwslRandomCount[0]: ", 0, totalChars); - } - Household.logHelper(logger, "aoRandomCount: ", aoRandomCount, totalChars); - Household.logHelper(logger, "tpRandomCount: ", tpRandomCount, totalChars); - Household.logHelper(logger, "fpRandomCount: ", fpRandomCount, totalChars); - Household.logHelper(logger, "ieRandomCount: ", ieRandomCount, totalChars); - Household.logHelper(logger, "cdapRandomCount: ", cdapRandomCount, totalChars); - Household.logHelper(logger, "imtfRandomCount: ", imtfRandomCount, totalChars); - Household.logHelper(logger, "imtodRandomCount: ", imtodRandomCount, totalChars); - Household.logHelper(logger, "awfRandomCount: ", awfRandomCount, totalChars); - Household.logHelper(logger, "awlRandomCount: ", awlRandomCount, totalChars); - Household.logHelper(logger, "awtodRandomCount: ", awtodRandomCount, totalChars); - Household.logHelper(logger, "jtfRandomCount: ", jtfRandomCount, totalChars); - Household.logHelper(logger, "jtlRandomCount: ", jtlRandomCount, totalChars); - Household.logHelper(logger, "jtodRandomCount: ", jtodRandomCount, totalChars); - Household.logHelper(logger, "inmtfRandomCount: ", inmtfRandomCount, totalChars); - Household.logHelper(logger, "inmtlRandomCount: ", inmtlRandomCount, totalChars); - Household.logHelper(logger, "inmtodRandomCount: ", inmtodRandomCount, totalChars); - Household.logHelper(logger, "stfRandomCount: ", stfRandomCount, totalChars); - Household.logHelper(logger, "stlRandomCount: ", stlRandomCount, totalChars); - Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); - Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); - Household.logHelper(logger, "maxMixedOverlaps: ", maxMixedOverlaps, totalChars); - - String tempString = String.format("Joint Tours[%s]:", - jointTours == null ? "" : String.valueOf(jointTours.length)); - logger.info(tempString); - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public void logPersonObject(String titleString, Logger logger, Person person) - { - - int totalChars = 114; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "P"; - - logger.info(separater); - logger.info(titleString); - logger.info(separater); - - person.logPersonObject(logger, totalChars); - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public void logTourObject(String titleString, Logger logger, Person person, Tour tour) - { - - int totalChars = 119; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "T"; - - logger.info(separater); - logger.info(titleString); - logger.info(separater); - - person.logTourObject(logger, totalChars, tour); - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public void logStopObject(String titleString, Logger logger, Stop stop, - ModelStructure modelStructure) - { - - int totalChars = 119; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "S"; - - logger.info(separater); - logger.info(titleString); - logger.info(separater); - - stop.logStopObject(logger, totalChars); - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public void logEntireHouseholdObject(String titleString, Logger logger) - { - - int totalChars = 60; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "="; - - logger.info(separater); - logger.info(titleString); - logger.info(separater); - - separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - Household.logHelper(logger, "hhId: ", hhId, totalChars); - Household.logHelper(logger, "debugChoiceModels: ", debugChoiceModels ? "True" : "False", - totalChars); - Household.logHelper(logger, "hhIncome: ", hhIncomeCategory, totalChars); - Household.logHelper(logger, "hhIncomeInDollars: ", hhIncomeInDollars, totalChars); - Household.logHelper(logger, "hhSize: ", hhSize, totalChars); - Household.logHelper(logger, "hhType: ", hhType, totalChars); - Household.logHelper(logger, "hhWorkers: ", hhWorkers, totalChars); - Household.logHelper(logger, "homeTaz: ", homeTaz, totalChars); - Household.logHelper(logger, "homeMgra: ", homeMgra, totalChars); - Household.logHelper(logger, "homeWalkSubzone: ", homeWalkSubzone, totalChars); - Household.logHelper(logger, "aoModelAutos: ", aoModelAutos, totalChars); - Household.logHelper(logger, "cdapModelPattern: ", cdapModelPattern, totalChars); - Household.logHelper(logger, "imtfModelPattern: ", imtfModelPattern, totalChars); - Household.logHelper(logger, "outboundEscortChoice: ", outboundEscortChoice, totalChars); - Household.logHelper(logger, "inboundEscortChoice: ", inboundEscortChoice, totalChars); - Household.logHelper(logger, "jtfModelPattern: ", jtfModelPattern, totalChars); - Household.logHelper(logger, "randomCount: ", randomCount, totalChars); - if (uwslRandomCountList.size() > 0) - { - for (int i : uwslRandomCountList.keySet()) - Household.logHelper(logger, String.format("uwslRandomCount[%d]: ", i), - uwslRandomCountList.get(i), totalChars); - } else - { - Household.logHelper(logger, "uwslRandomCount[0]: ", 0, totalChars); - } - Household.logHelper(logger, "aoRandomCount: ", aoRandomCount, totalChars); - Household.logHelper(logger, "tpRandomCount: ", tpRandomCount, totalChars); - Household.logHelper(logger, "fpRandomCount: ", fpRandomCount, totalChars); - Household.logHelper(logger, "ieRandomCount: ", ieRandomCount, totalChars); - Household.logHelper(logger, "cdapRandomCount: ", cdapRandomCount, totalChars); - Household.logHelper(logger, "imtfRandomCount: ", imtfRandomCount, totalChars); - Household.logHelper(logger, "imtodRandomCount: ", imtodRandomCount, totalChars); - Household.logHelper(logger, "awfRandomCount: ", awfRandomCount, totalChars); - Household.logHelper(logger, "awlRandomCount: ", awlRandomCount, totalChars); - Household.logHelper(logger, "awtodRandomCount: ", awtodRandomCount, totalChars); - Household.logHelper(logger, "jtfRandomCount: ", jtfRandomCount, totalChars); - Household.logHelper(logger, "jtlRandomCount: ", jtlRandomCount, totalChars); - Household.logHelper(logger, "jtodRandomCount: ", jtodRandomCount, totalChars); - Household.logHelper(logger, "inmtfRandomCount: ", inmtfRandomCount, totalChars); - Household.logHelper(logger, "inmtlRandomCount: ", inmtlRandomCount, totalChars); - Household.logHelper(logger, "inmtodRandomCount: ", inmtodRandomCount, totalChars); - Household.logHelper(logger, "stfRandomCount: ", stfRandomCount, totalChars); - Household.logHelper(logger, "stlRandomCount: ", stlRandomCount, totalChars); - Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); - Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); - Household.logHelper(logger, "maxMixedOverlaps: ", maxMixedOverlaps, totalChars); - - if (jointTours != null) - { - logger.info("Joint Tours:"); - if (jointTours.length > 0) - { - for (int i = 0; i < jointTours.length; i++) - jointTours[i].logEntireTourObject(logger); - } else logger.info(" No joint tours"); - } else logger.info(" No joint tours"); - - logger.info("Person Objects:"); - for (int i = 1; i < persons.length; i++) - persons[i].logEntirePersonObject(logger); - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public static void logHelper(Logger logger, String label, int value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } - - public static void logHelper(Logger logger, String label, String value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } - - public static void logHelper(Logger logger, String label, float value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%df", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } - public enum HouseholdType - { - nul, FAMILY_MARRIED, FAMILY_MALE_NO_WIFE, FAMILY_FEMALE_NO_HUSBAND, NON_FAMILY_MALE_ALONE, NON_FAMILY_MALE_NOT_ALONE, NON_FAMILY_FEMALE_ALONE, NON_FAMILY_FEMALE_NOT_ALONE - } - - /** - * Iterate through persons in household grab all adults and add them to an ArrayList of adults. Return the list. - * - * @return An ArrayList of adult household members. - */ - public List getAdultPersons() { - - List adultList = new ArrayList(); - - for(int i = 1; i < persons.length; ++i){ - Person p = persons[i]; - if((p.getPersonIsFullTimeWorker()==1)||(p.getPersonIsPartTimeWorker()==1)||(p.getPersonIsNonWorkingAdultUnder65()==1) - ||(p.getPersonIsUniversityStudent()==1)||(p.getPersonIsNonWorkingAdultOver65()==1)) - adultList.add(p); - - } - return adultList; - } - /** - * Iterate through persons in household grab all adults and add them to an ArrayList of adults. Return the list. - * - * @return An ArrayList of adult household members. - */ - public List getActiveAdultPersons() { - - List adultList = new ArrayList(); - - for(int i = 1; i < persons.length; ++i){ - Person p = persons[i]; - if(p.isActiveAdult()) - adultList.add(p); - - } - return adultList; - } - - /** - * Count and return the number of active adults in the household. - * - * @return The number of active adults. - */ - public int getNumberActiveAdults(){ - - int numberActiveAdults=0; - for(int i = 1; i < persons.length; ++i){ - if(persons[i].isActiveAdult()) - ++numberActiveAdults; - - } - return numberActiveAdults; - } - - - /** - * Iterate through persons in household grab all children and add them to an ArrayList of children. Return the list. - * - * @return An ArrayList of adult household members. - */ - public List getChildPersons() { - - List childList = new ArrayList(); - - for(int i = 1; i < persons.length; ++i){ - Person p = persons[i]; - if((p.getPersonIsPreschoolChild()==1)||(p.getPersonIsStudentNonDriving()==1)||(p.getPersonIsStudentDriving()==1)) - childList.add(p); - - } - return childList; - } - } diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java deleted file mode 100644 index 11daef6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAtWorkSubtourFrequencyModel.java +++ /dev/null @@ -1,334 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -// - -public class HouseholdAtWorkSubtourFrequencyModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdAtWorkSubtourFrequencyModel.class); - private transient Logger tourFreq = Logger.getLogger("tourFreq"); - - private static final String AWTF_CONTROL_FILE_TARGET = "awtf.uec.file"; - private static final String AWTF_DATA_SHEET_TARGET = "awtf.data.page"; - private static final String AWTF_MODEL_SHEET_TARGET = "awtf.model.page"; - - // model results - // private static final int NO_SUBTOURS = 1; - private static final int ONE_EAT = 2; - private static final int ONE_BUSINESS = 3; - private static final int ONE_OTHER = 4; - private static final int TWO_BUSINESS = 5; - private static final int TWO_OTHER = 6; - private static final int ONE_EAT_ONE_BUSINESS = 7; - - private AtWorkSubtourFrequencyDMU dmuObject; - private ChoiceModelApplication choiceModelApplication; - - private ModelStructure modelStructure; - private String[] alternativeNames; - - /** - * Constructor establishes the ChoiceModelApplication, which applies the - * logit model via the UEC spreadsheet. - * - * @param dmuObject - * is the UEC dmu object for this choice model - * @param uecFileName - * is the UEC control file name - * @param resourceBundle - * is the application ResourceBundle, from which a properties - * file HashMap will be created for the UEC - * @param tazDataManager - * is the object used to interact with the zonal data table - * @param modelStructure - * is the ModelStructure object that defines segmentation and - * other model structure relate atributes - */ - public HouseholdAtWorkSubtourFrequencyModel(HashMap propertyMap, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) - { - - this.modelStructure = modelStructure; - setUpModels(propertyMap, dmuFactory); - - } - - private void setUpModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up %s tour frequency choice model.", - ModelStructure.AT_WORK_CATEGORY)); - - // locate the individual mandatory tour frequency choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String awtfUecFile = propertyMap.get(AWTF_CONTROL_FILE_TARGET); - awtfUecFile = uecPath + awtfUecFile; - - int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, AWTF_DATA_SHEET_TARGET); - int modelPage = Util.getIntegerValueFromPropertyMap(propertyMap, AWTF_MODEL_SHEET_TARGET); - - dmuObject = dmuFactory.getAtWorkSubtourFrequencyDMU(); - - // set up the model - choiceModelApplication = new ChoiceModelApplication(awtfUecFile, modelPage, dataPage, - propertyMap, (VariableTable) dmuObject); - - } - - /** - * Applies the model for the array of households that are stored in the - * HouseholdDataManager. The results are summarized by person type. - * - * @param householdDataManager - * is the object containg the Household objects for which this - * model is to be applied. - */ - public void applyModel(Household household) - { - - int choice = -1; - String personTypeString = ""; - - Logger modelLogger = tourFreq; - if (household.getDebugChoiceModels()) - household.logHouseholdObject( - "Pre AtWork Subtour Frequency Choice HHID=" + household.getHhId() + " Object", - modelLogger); - - // get this household's person array - Person[] personArray = household.getPersons(); - - // set the household id, origin taz, hh taz, and debugFlag=false in the - // dmu - dmuObject.setHouseholdObject(household); - - // loop through the person array (1-based) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - // count the results by person type - personTypeString = person.getPersonType(); - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), personTypeString); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - // loop through the work tours for this person - ArrayList tourList = person.getListOfWorkTours(); - if (tourList == null) continue; - - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - int workTourIndex = 0; - for (Tour workTour : tourList) - { - - try - { - - // set the person and tour object - dmuObject.setPersonObject(person); - dmuObject.setTourObject(workTour); - - // write debug header - if (household.getDebugChoiceModels() || person.getPersonTypeNumber() == 7) - { - - choiceModelDescription = String - .format("At-work Subtour Frequency Choice Model:"); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, workTourId=%d", person - .getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), workTour.getTourId()); - choiceModelApplication.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = choiceModelDescription + " for " + decisionMakerLabel - + "."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - } - - // set the availability array for the tour frequency model - alternativeNames = choiceModelApplication.getAlternativeNames(); - int numberOfAlternatives = alternativeNames.length; - boolean[] availabilityArray = new boolean[numberOfAlternatives + 1]; - Arrays.fill(availabilityArray, true); - - // create the sample array - int[] sampleArray = new int[availabilityArray.length]; - Arrays.fill(sampleArray, 1); - - // compute the utilities - IndexValues index = dmuObject.getDmuIndexValues(); - index.setHHIndex(household.getHhId()); - index.setZoneIndex(household.getHhTaz()); - index.setOriginZone(workTour.getTourOrigMgra()); - index.setDestZone(workTour.getTourDestMgra()); - index.setDebug(household.getDebugChoiceModels()); - - if (household.getDebugChoiceModels()) - { - household.logTourObject(loggingHeader, modelLogger, person, workTour); - } - - float logsum = (float) choiceModelApplication.computeUtilities(dmuObject, index, availabilityArray, - sampleArray); - - workTour.setSubtourFreqLogsum(logsum); - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, - // make choice. - if (choiceModelApplication.getAvailabilityCount() > 0) choice = choiceModelApplication - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for j=%d, tourNum=%d, HHID=%d, no available at-work frequency alternatives to choose from in choiceModelApplication.", - j, workTourIndex, person.getHouseholdObject().getHhId())); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = choiceModelApplication.getUtilities(); - double[] probabilities = choiceModelApplication.getProbabilities(); - - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " - + personTypeString); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("-------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < alternativeNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %-16s", k + 1, - alternativeNames[k]); - modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %-16s", choice, - alternativeNames[choice - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - choiceModelApplication.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - choiceModelApplication.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, choice); - - // write UEC calculation results to separate model - // specific - // log file - choiceModelApplication.logUECResults(modelLogger, loggingHeader); - - } - - workTour.setSubtourFreqChoice(choice); - - // set the person choices - if (choice == ONE_EAT) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkEatPurposeName()); - } else if (choice == ONE_BUSINESS) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkBusinessPurposeName()); - } else if (choice == ONE_OTHER) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkMaintPurposeName()); - } else if (choice == TWO_BUSINESS) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkBusinessPurposeName()); - id = workTourIndex * 10 + 2; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkBusinessPurposeName()); - } else if (choice == TWO_OTHER) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkMaintPurposeName()); - id = workTourIndex * 10 + 2; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkMaintPurposeName()); - } else if (choice == ONE_EAT_ONE_BUSINESS) - { - int id = workTourIndex * 10 + 1; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkEatPurposeName()); - id = workTourIndex * 10 + 2; - person.createAtWorkSubtour(id, choice, workTour.getTourDestMgra(), - modelStructure.getAtWorkBusinessPurposeName()); - } - - } catch (Exception e) - { - logger.error(String.format("Exception caught for j=%d, tourNum=%d, HHID=%d.", - j, workTourIndex, household.getHhId())); - throw new RuntimeException(); - } - - workTourIndex++; - - } - - } // j (person loop) - - household.setAwfRandomCount(household.getHhRandomCount()); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java deleted file mode 100644 index cfa8483..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdAutoOwnershipModel.java +++ /dev/null @@ -1,343 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import com.pb.common.calculator.VariableTable; -import com.pb.common.model.ModelException; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class HouseholdAutoOwnershipModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdAutoOwnershipModel.class); - private transient Logger aoLogger = Logger.getLogger("ao"); - - private static final String AO_CONTROL_FILE_TARGET = "ao.uec.file"; - private static final String AO_MODEL_SHEET_TARGET = "ao.model.page"; - private static final String AO_DATA_SHEET_TARGET = "ao.data.page"; - - private static final int AUTO_SOV_TIME_INDEX = 10; - private static final int AUTO_LOGSUM_INDEX = 6; - private static final int TRANSIT_LOGSUM_INDEX = 8; - private static final int DT_RAIL_PROP_INDEX = 10; - - private AccessibilitiesTable accTable; - private MandatoryAccessibilitiesCalculator mandAcc; - private ChoiceModelApplication aoModel; - private AutoOwnershipChoiceDMU aoDmuObject; - - private int[] totalAutosByAlt; - private int[] automatedVehiclesByAlt; - private int[] conventionalVehiclesByAlt; - - public HouseholdAutoOwnershipModel(HashMap rbMap, - CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, - MandatoryAccessibilitiesCalculator myMandAcc) - { - - logger.info("setting up AO choice model."); - - // set the aggAcc class variable, which will serve as a flag: null -> no - // accessibilities, !null -> set accessibilities. - // if the BuildAccessibilities object is null, the AO utility does not - // need - // to use the accessibilities components. - accTable = myAccTable; - mandAcc = myMandAcc; - - // locate the auto ownership UEC - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String autoOwnershipUecFile = rbMap.get(AO_CONTROL_FILE_TARGET); - autoOwnershipUecFile = uecPath + autoOwnershipUecFile; - - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, AO_DATA_SHEET_TARGET); - int modelPage = Util.getIntegerValueFromPropertyMap(rbMap, AO_MODEL_SHEET_TARGET); - - // create the auto ownership choice model DMU object. - aoDmuObject = dmuFactory.getAutoOwnershipDMU(); - - // create the auto ownership choice model object - aoModel = new ChoiceModelApplication(autoOwnershipUecFile, modelPage, dataPage, rbMap, - (VariableTable) aoDmuObject); - - String[] alternativeNames = aoModel.getAlternativeNames(); - calculateAlternativeArrays(alternativeNames); - - } - - /** - * Set the dmu attributes, compute the pre-AO or AO utilities, and select an - * alternative - * - * @param hhObj - * for which to apply thye model - * @param preAutoOwnership - * is true if running pre-auto ownership, or false to run primary - * auto ownership model. - */ - - public void applyModel(Household hhObj, boolean preAutoOwnership) - { - - // update the AO dmuObject for this hh - aoDmuObject.setHouseholdObject(hhObj); - aoDmuObject.setDmuIndexValues(hhObj.getHhId(), hhObj.getHhMgra(), hhObj.getHhMgra(), 0); - - // set the non-mandatory accessibility values for the home MGRA. - // values used by both pre-ao and ao models. - aoDmuObject.setHomeTazAutoAccessibility(accTable.getAggregateAccessibility("auto", - hhObj.getHhMgra())); - aoDmuObject.setHomeTazTransitAccessibility(accTable.getAggregateAccessibility("transit", - hhObj.getHhMgra())); - aoDmuObject.setHomeTazNonMotorizedAccessibility(accTable.getAggregateAccessibility( - "nonmotor", hhObj.getHhMgra())); - aoDmuObject.setHomeTazMaasAccessibility(accTable.getAggregateAccessibility("maas", - hhObj.getHhMgra())); - - - - if (preAutoOwnership) - { - - aoDmuObject.setUseAccessibilities(false); - - } else - { - - aoDmuObject.setUseAccessibilities(true); - - // compute the disaggregate accessibilities for the home MGRA to - // work and - // school MGRAs summed accross workers and students - double workAutoDependency = 0.0; - double schoolAutoDependency = 0.0; - double workRailProp = 0.0; - double schoolRailProp = 0.0; - double workAutoTime = 0.0; - Person[] persons = hhObj.getPersons(); - for (int i = 1; i < persons.length; i++) - { - - // sum over all workers (full time or part time) - if (persons[i].getPersonIsWorker() == 1) - { - - int workMgra = persons[i].getWorkLocation(); - if (workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - - // Non-Motorized Factor = 0.5*MIN(MAX(DIST,1),3)-0.5 - // if 0 <= dist < 1, nmFactor = 0 - // if 1 <= dist <= 3, nmFactor = [0.0, 1.0] - // if 3 <= dist, nmFactor = 1.0 - double nmFactor = 0.5 * (Math.min( - Math.max(persons[i].getWorkLocationDistance(), 1.0), 3.0)) - 0.5; - - // if auto logsum < transit logsum, do not accumulate - // auto - // dependency - double[] workerAccessibilities = mandAcc - .calculateWorkerMandatoryAccessibilities(hhObj.getHhMgra(), - workMgra); - workAutoTime += workerAccessibilities[AUTO_SOV_TIME_INDEX]; - if (workerAccessibilities[AUTO_LOGSUM_INDEX] >= workerAccessibilities[TRANSIT_LOGSUM_INDEX]) - { - double logsumDiff = workerAccessibilities[AUTO_LOGSUM_INDEX] - - workerAccessibilities[TRANSIT_LOGSUM_INDEX]; - - // need to scale and cap logsum difference - logsumDiff = Math.min(logsumDiff / 3.0, 1.0); - workAutoDependency += (logsumDiff * nmFactor); - } - - workRailProp += workerAccessibilities[DT_RAIL_PROP_INDEX]; - - } - - } - - // sum over all students of driving age - if (persons[i].getPersonIsUniversityStudent() == 1 - || persons[i].getPersonIsStudentDriving() == 1) - { - - int schoolMgra = persons[i].getUsualSchoolLocation(); - if (schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) - { - - // Non-Motorized Factor = 0.5*MIN(MAX(DIST,1),3)-0.5 - // if 0 <= dist < 1, nmFactor = 0 - // if 1 <= dist <= 3, nmFactor = [0.0, 1.0] - // if 3 <= dist, nmFactor = 1.0 - double nmFactor = 0.5 * (Math.min( - Math.max(persons[i].getWorkLocationDistance(), 1.0), 3.0)) - 0.5; - - // if auto logsum < transit logsum, do not accumulate - // auto - // dependency - double[] studentAccessibilities = mandAcc - .calculateStudentMandatoryAccessibilities(hhObj.getHhMgra(), - schoolMgra); - if (studentAccessibilities[AUTO_LOGSUM_INDEX] >= studentAccessibilities[TRANSIT_LOGSUM_INDEX]) - { - double logsumDiff = studentAccessibilities[AUTO_LOGSUM_INDEX] - - studentAccessibilities[TRANSIT_LOGSUM_INDEX]; - - // need to scale and cap logsum difference - logsumDiff = Math.min(logsumDiff / 3.0, 1.0); - schoolAutoDependency += (logsumDiff * nmFactor); - } - - schoolRailProp += studentAccessibilities[DT_RAIL_PROP_INDEX]; - - } - } - - } - - aoDmuObject.setWorkAutoDependency(workAutoDependency); - aoDmuObject.setSchoolAutoDependency(schoolAutoDependency); - - aoDmuObject.setWorkersRailProportion(workRailProp); - aoDmuObject.setStudentsRailProportion(schoolRailProp); - - aoDmuObject.setWorkAutoTime(workAutoTime); - - } - - // compute utilities and choose auto ownership alternative. - float logsum = (float) aoModel.computeUtilities(aoDmuObject, aoDmuObject.getDmuIndexValues()); - - hhObj.setAutoOwnershipLogsum(logsum); - Random hhRandom = hhObj.getHhRandom(); - int randomCount = hhObj.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt = -1; - if (aoModel.getAvailabilityCount() > 0) - { - try - { - chosenAlt = aoModel.getChoiceResult(rn); - } catch (ModelException e) - { - logger.error(String.format( - "exception caught for HHID=%d in choiceModelApplication.", hhObj.getHhId())); - } - } else - { - logger.error(String - .format("error: HHID=%d has no available auto ownership alternatives to choose from in choiceModelApplication.", - hhObj.getHhId())); - throw new RuntimeException(); - } - - // write choice model alternative info to log file - if (hhObj.getDebugChoiceModels() || chosenAlt < 0) - { - - String loggerString = (preAutoOwnership ? "Pre-AO without" : "AO with") - + " accessibilities, Household " + hhObj.getHhId() + " Object"; - hhObj.logHouseholdObject(loggerString, aoLogger); - - double[] utilities = aoModel.getUtilities(); - double[] probabilities = aoModel.getProbabilities(); - - aoLogger.info("Alternative Utility Probability CumProb"); - aoLogger.info("-------------------- --------------- ------------ ------------"); - - double cumProb = 0.0; - for (int k = 0; k < aoModel.getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - aoLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", k + " autos", utilities[k], - probabilities[k], cumProb)); - } - - aoLogger.info(" "); - aoLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", chosenAlt, rn, - randomCount)); - - aoLogger.info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - aoLogger.info(""); - aoLogger.info(""); - - // write choice model alternative info to debug log file - aoModel.logAlternativesInfo("Household Auto Ownership Choice", - String.format("HH_%d", hhObj.getHhId())); - aoModel.logSelectionInfo("Household Auto Ownership Choice", - String.format("HH_%d", hhObj.getHhId()), rn, chosenAlt); - - // write UEC calculation results to separate model specific log file - aoModel.logUECResults(aoLogger, - String.format("Household Auto Ownership Choice, HH_%d", hhObj.getHhId())); - } - - if (preAutoOwnership) hhObj.setPreAoRandomCount(hhObj.getHhRandomCount()); - else hhObj.setAoRandomCount(hhObj.getHhRandomCount()); - - int autos = totalAutosByAlt[chosenAlt-1]; - int AVs = automatedVehiclesByAlt[chosenAlt-1]; - int CVs = conventionalVehiclesByAlt[chosenAlt-1]; - hhObj.setHhAutos(autos); - hhObj.setAutomatedVehicles(AVs); - hhObj.setConventionalVehicles(CVs); - - } - - - /** - * This is a helper method that iterates through the alternative names - * in the auto ownership UEC and searches through each name to collect - * the total number of autos (in the first position of the name character - * array), the number of AVs for the alternative (preceded by the "AV" substring) - * and the number of CVs for the alternative (preceded by the "CV" substring). The - * results are stored in the arrays: - * - * totalAutosByAlt - * automatedVehiclesByAlt - * conventionalVehiclesByAlt - * - * @param alternativeNames The array of alternative names. - */ - private void calculateAlternativeArrays(String[] alternativeNames){ - - totalAutosByAlt = new int[alternativeNames.length]; - automatedVehiclesByAlt = new int[alternativeNames.length]; - conventionalVehiclesByAlt = new int[alternativeNames.length]; - - - //iterate thru names - for(int i = 0; i < alternativeNames.length;++i){ - - String altName = alternativeNames[i]; - - //find the number of cars; first element of name (e.g. 0_CARS) - int autos = new Integer(altName.substring(0,1)).intValue(); - int AVs=0; - int HVs=0; - int AVPosition = altName.indexOf("AV"); - if(AVPosition>=0) - AVs = new Integer(altName.substring(AVPosition-1, AVPosition)).intValue(); - int HVPosition = altName.indexOf("HV"); - if(HVPosition>=0) - HVs = new Integer(altName.substring(HVPosition-1, HVPosition)).intValue(); - - totalAutosByAlt[i] = autos; - automatedVehiclesByAlt[i] = AVs; - conventionalVehiclesByAlt[i] = HVs; - - } - - } - - -} - diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java deleted file mode 100644 index 3a5d92a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelRunner.java +++ /dev/null @@ -1,254 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; - -import org.apache.log4j.Logger; -import org.jppf.client.JPPFClient; -import org.jppf.client.JPPFJob; -import org.jppf.node.protocol.DataProvider; -import org.jppf.node.protocol.MemoryMapDataProvider; -import org.jppf.node.protocol.Task; - -import com.pb.common.calculator.MatrixDataServerIf; - -public class HouseholdChoiceModelRunner -{ - - private Logger logger = Logger.getLogger(HouseholdChoiceModelRunner.class); - - private static int PACKET_SIZE = 0; - - private static String PROPERTIES_NUM_INITIALIZATION_PACKETS = "number.initialization.packets"; - private static String PROPERTIES_INITIALIZATION_PACKET_SIZE = "initialization.packet.size"; - private static int NUM_INITIALIZATION_PACKETS = 0; - private static int INITIALIZATION_PACKET_SIZE = 0; - - private int ONE_HH_ID = -1; - - private static final String HOUSEHOLD_CHOICE_PACKET_SIZE = "distributed.task.packet.size"; - private static final String RUN_THIS_HOUSEHOLD_ONLY = "run.this.household.only"; - - private HashMap propertyMap; - private String restartModelString; - private MatrixDataServerIf ms; - private HouseholdDataManagerIf hhDataManager; - private ModelStructure modelStructure; - private CtrampDmuFactoryIf dmuFactory; - - private JPPFClient jppfClient; - private boolean logResults=false; - - // The number of initialization packets are the number of "small" packets - // submited at the beginning of a - // distributed task to minimize synchronization issues that significantly - // slow - // down model object setup. - // It is assumed that after theses small packets have run, all the model - // objects - // will have been setup, - // and the task objects can process much bigger chuncks of households. - - public HouseholdChoiceModelRunner(HashMap propertyMap, JPPFClient jppfClient, - String restartModelString, HouseholdDataManagerIf hhDataManager, MatrixDataServerIf ms, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) - { - setupHouseholdChoiceModelRunner(propertyMap, jppfClient, restartModelString, hhDataManager, - ms, modelStructure, dmuFactory); - } - - private void setupHouseholdChoiceModelRunner(HashMap propertyMap, - JPPFClient jppfClient, String restartModelString, HouseholdDataManagerIf hhDataManager, - MatrixDataServerIf ms, ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) - { - - this.propertyMap = propertyMap; - this.restartModelString = restartModelString; - this.hhDataManager = hhDataManager; - this.ms = ms; - this.modelStructure = modelStructure; - this.dmuFactory = dmuFactory; - this.jppfClient = jppfClient; - - String oneHhString = propertyMap.get(RUN_THIS_HOUSEHOLD_ONLY); - if (oneHhString != null) ONE_HH_ID = Integer.parseInt(oneHhString); - - String propertyValue = propertyMap.get(HOUSEHOLD_CHOICE_PACKET_SIZE); - if (propertyValue == null) PACKET_SIZE = 0; - else PACKET_SIZE = Integer.parseInt(propertyValue); - - propertyValue = propertyMap.get(PROPERTIES_NUM_INITIALIZATION_PACKETS); - if (propertyValue == null) NUM_INITIALIZATION_PACKETS = 0; - else NUM_INITIALIZATION_PACKETS = Integer.parseInt(propertyValue); - - propertyValue = propertyMap.get(PROPERTIES_INITIALIZATION_PACKET_SIZE); - if (propertyValue == null) INITIALIZATION_PACKET_SIZE = 0; - else INITIALIZATION_PACKET_SIZE = Integer.parseInt(propertyValue); - - logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - } - - /** - * - * JPPF framework based method - */ - public void runHouseholdChoiceModels() - { - - long initTime = System.currentTimeMillis(); - - submitTasks(); - - logger.info(String.format("household model runner finished %d households in %d minutes.", - hhDataManager.getNumHouseholds(), - ((System.currentTimeMillis() - initTime) / 1000) / 60)); - - } - - /** - * @param client - * is a JPPFClient object which is used to establish a connection - * to a computing node, submit tasks, and receive results. - */ - private void submitTasks() - { - - // if PACKET_SIZE was not specified, create a single task to use for all - // households - if (PACKET_SIZE == 0) PACKET_SIZE = hhDataManager.getNumHouseholds(); - - // Create a setup task object and submit it to the computing node. - // This setup task creates the HouseholdChoiceModelManager and causes it - // to - // create the necessary numuber - // of HouseholdChoiceModels objects which will operate in parallel on - // the - // computing node. - try - { - - JPPFJob job = new JPPFJob(); - job.setName("Household Choice Job"); - - DataProvider dataProvider = new MemoryMapDataProvider(); - dataProvider.setParameter("propertyMap", propertyMap); - dataProvider.setParameter("ms", ms); - dataProvider.setParameter("hhDataManager", hhDataManager); - dataProvider.setParameter("modelStructure", modelStructure); - dataProvider.setParameter("dmuFactory", dmuFactory); - dataProvider.setParameter("restartModelString", restartModelString); - job.setDataProvider(dataProvider); - - ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(hhDataManager - .getNumHouseholds()); - - int startIndex = 0; - int endIndex = 0; - int taskIndex = 1; - for (int[] startEndIndices : startEndTaskIndicesList) - { - startIndex = startEndIndices[0]; - endIndex = startEndIndices[1]; - - HouseholdChoiceModelsTaskJppf task = new HouseholdChoiceModelsTaskJppf(taskIndex, - startIndex, endIndex); - job.add(task); - taskIndex++; - } - - List> results = jppfClient.submitJob(job); - for (Task task : results) - { - if (task.getThrowable() != null) throw new Exception(task.getThrowable()); - - try - { - if(logResults){ - logger.info(String.format("HH TASK: %s returned: %s, maxAlts: %d.", - task.getId(), (String) task.getResult(), - ((HouseholdChoiceModelsTaskJppf) task).getMaxAlts())); - } - } catch (Exception e) - { - logger.error( - "Exception returned by computing node caught in HouseholdChoiceModelsTaskJppf.", - e); - throw new RuntimeException(); - } - - } - - } catch (Exception e) - { - logger.error( - "Exception caught creating/submitting/receiving HouseholdChoiceModelsTaskJppf.", - e); - throw new RuntimeException(); - } - - } - - private ArrayList getTaskHouseholdRanges(int numberOfHouseholds) - { - - ArrayList startEndIndexList = new ArrayList(); - - if (ONE_HH_ID < 0) - { - - int numInitializationHouseholds = NUM_INITIALIZATION_PACKETS - * INITIALIZATION_PACKET_SIZE; - - int startIndex = 0; - int endIndex = 0; - if (numInitializationHouseholds < numberOfHouseholds) - { - - while (endIndex < numInitializationHouseholds) - { - endIndex = startIndex + INITIALIZATION_PACKET_SIZE - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += INITIALIZATION_PACKET_SIZE; - } - - } - - while (endIndex < numberOfHouseholds - 1) - { - endIndex = startIndex + PACKET_SIZE - 1; - if (endIndex + PACKET_SIZE > numberOfHouseholds) endIndex = numberOfHouseholds - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += PACKET_SIZE; - } - - return startEndIndexList; - - } else - { - - // create a single task packet high one household id - int[] startEndIndices = new int[2]; - int index = hhDataManager.getArrayIndex(ONE_HH_ID); - startEndIndices[0] = index; - startEndIndices[1] = index; - startEndIndexList.add(startEndIndices); - - return startEndIndexList; - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java deleted file mode 100644 index 8bd78d8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModels.java +++ /dev/null @@ -1,980 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.Arrays; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.util.ObjectUtil; - -public class HouseholdChoiceModels -{ - - private transient Logger logger = Logger.getLogger(HouseholdChoiceModels.class); - - private static final String GLOBAL_MODEL_SEED_PROPERTY = "Model.Random.Seed"; - private static final int AO_SEED_OFFSET = 0; - private static final int TP_SEED_OFFSET = 1; - private static final int PP_SEED_OFFSET = 2; - private static final int CDAP_SEED_OFFSET = 3; - private static final int IMTF_SEED_OFFSET = 4; - private static final int IMTOD_SEED_OFFSET = 5; - private static final int JTF_SEED_OFFSET = 6; - private static final int JTDC_SEED_OFFSET = 7; - private static final int JTOD_SEED_OFFSET = 8; - private static final int INMTF_SEED_OFFSET = 9; - private static final int INMDC_SEED_OFFSET = 10; - private static final int INMTOD_SEED_OFFSET = 11; - private static final int AWTF_SEED_OFFSET = 12; - private static final int AWDC_SEED_OFFSET = 13; - private static final int AWTOD_SEED_OFFSET = 14; - private static final int STF_SEED_OFFSET = 15; - private static final int SLC_SEED_OFFSET = 16; - private static final int IE_SEED_OFFSET = 17; - - private static final String USE_NEW_SLC_SOA_METHOD_PROPERTY_KEY = "slc.use.new.soa"; - - private boolean runAutoOwnershipModel; - private boolean runTransponderModel; - private boolean runInternalExternalModel; - private boolean runParkingProvisionModel; - private boolean runCoordinatedDailyActivityPatternModel; - private boolean runIndividualMandatoryTourFrequencyModel; - private boolean runMandatoryTourModeChoiceModel; - private boolean runMandatoryTourDepartureTimeAndDurationModel; - private boolean runEscortModel; - private boolean runAtWorkSubTourFrequencyModel; - private boolean runAtWorkSubtourLocationChoiceModel; - private boolean runAtWorkSubtourModeChoiceModel; - private boolean runAtWorkSubtourDepartureTimeAndDurationModel; - private boolean runJointTourFrequencyModel; - private boolean runJointTourLocationChoiceModel; - private boolean runJointTourDepartureTimeAndDurationModel; - private boolean runJointTourModeChoiceModel; - private boolean runIndividualNonMandatoryTourFrequencyModel; - private boolean runIndividualNonMandatoryTourLocationChoiceModel; - private boolean runIndividualNonMandatoryTourModeChoiceModel; - private boolean runIndividualNonMandatoryTourDepartureTimeAndDurationModel; - private boolean runStopFrequencyModel; - private boolean runStopLocationModel; - - private String restartModelString; - - private HouseholdAutoOwnershipModel aoModel; - private TourVehicleTypeChoiceModel tvtcModel; - private TransponderChoiceModel tcModel; - private InternalExternalTripChoiceModel ieModel; - private ParkingProvisionModel ppModel; - private TelecommuteModel teModel; - private HouseholdCoordinatedDailyActivityPatternModel cdapModel; - private HouseholdIndividualMandatoryTourFrequencyModel imtfModel; - private HouseholdIndividualNonMandatoryTourFrequencyModel inmtfModel; - private SchoolEscortingModel escortModel; - private HouseholdAtWorkSubtourFrequencyModel awfModel; - private StopFrequencyModel stfModel; - private TourModeChoiceModel immcModel; - private HouseholdIndividualMandatoryTourDepartureAndDurationTime imtodModel; - private JointTourModels jtfModel; - private TourModeChoiceModel nmmcModel; - private NonMandatoryDestChoiceModel nmlcModel; - private NonMandatoryTourDepartureAndDurationTime nmtodModel; - private TourModeChoiceModel awmcModel; - private SubtourDestChoiceModel awlcModel; - private SubtourDepartureAndDurationTime awtodModel; - private IntermediateStopChoiceModels stlmcModel; - private MicromobilityChoiceModel mmModel; - - private long aoTime; - private long fpTime; - private long ieTime; - private long cdapTime; - private long escortTime; - private long imtfTime; - private long imtodTime; - private long imtmcTime; - private long jtfTime; - private long jtdcTime; - private long jtodTime; - private long jtmcTime; - private long inmtfTime; - private long inmtdcTime; - private long inmtdcSoaTime; - private long inmtodTime; - private long inmtmcTime; - private long awtfTime; - private long awtdcTime; - private long awtdcSoaTime; - private long awtodTime; - private long awtmcTime; - private long stfTime; - private long stdtmTime; - private long[] returnPartialTimes = new long[IntermediateStopChoiceModels.NUM_CPU_TIME_VALUES]; - - private int maxAlts; - private int modelIndex; - - private int globalSeed; - - private boolean useNewSlcSoaMethod; - - private double[][][] slcSizeProbs; - private double[][] slcTazSize; - private double[][] slcTazDistExpUtils; - - private double[] distanceToCordonsLogsums; - - private MgraDataManager mgraManager; - private TazDataManager tdm; - - public HouseholdChoiceModels(int modelIndex, String restartModelString, - HashMap propertyMap, ModelStructure modelStructure, - CtrampDmuFactoryIf dmuFactory, BuildAccessibilities aggAcc, - McLogsumsCalculator logsumHelper, MandatoryAccessibilitiesCalculator mandAcc, - double[] pctHighIncome, double[] pctMultipleAutos, double[] avgtts, - double[] transpDist, double[] pctDetour, double[][][] nonManSoaDistProbs, - double[][][] nonManSoaSizeProbs, double[][][] subTourSoaDistProbs, - double[][][] subTourSoaSizeProbs, double[] distanceToCordonsLogsums,AutoTazSkimsCalculator tazDistanceCalculator) - { - - this.modelIndex = modelIndex; - this.restartModelString = restartModelString; - - this.distanceToCordonsLogsums = distanceToCordonsLogsums; - - globalSeed = Integer.parseInt(propertyMap.get(GLOBAL_MODEL_SEED_PROPERTY)); - - mgraManager = MgraDataManager.getInstance(propertyMap); - tdm = TazDataManager.getInstance(propertyMap); - - runAutoOwnershipModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_AUTO_OWNERSHIP)); - runTransponderModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_TRANSPONDER_CHOICE)); - runInternalExternalModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INTERNAL_EXTERNAL_TRIP)); - runParkingProvisionModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_FREE_PARKING_AVAILABLE)); - runCoordinatedDailyActivityPatternModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_DAILY_ACTIVITY_PATTERN)); - runIndividualMandatoryTourFrequencyModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INDIV_MANDATORY_TOUR_FREQ)); - runMandatoryTourModeChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_MAND_TOUR_MODE_CHOICE)); - runMandatoryTourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_MAND_TOUR_DEP_TIME_AND_DUR)); - runEscortModel = Boolean.parseBoolean(propertyMap.get(CtrampApplication.PROPERTIES_RUN_SCHOOL_ESCORT_MODEL)); - runAtWorkSubTourFrequencyModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_FREQ)); - runAtWorkSubtourLocationChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_LOCATION_CHOICE)); - runAtWorkSubtourModeChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_MODE_CHOICE)); - runAtWorkSubtourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_AT_WORK_SUBTOUR_DEP_TIME_AND_DUR)); - runJointTourFrequencyModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_FREQ)); - runJointTourLocationChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_JOINT_LOCATION_CHOICE)); - runJointTourModeChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_MODE_CHOICE)); - runJointTourDepartureTimeAndDurationModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_JOINT_TOUR_DEP_TIME_AND_DUR)); - runIndividualNonMandatoryTourFrequencyModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_FREQ)); - runIndividualNonMandatoryTourLocationChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_LOCATION_CHOICE)); - runIndividualNonMandatoryTourModeChoiceModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_MODE_CHOICE)); - runIndividualNonMandatoryTourDepartureTimeAndDurationModel = Boolean - .parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_INDIV_NON_MANDATORY_TOUR_DEP_TIME_AND_DUR)); - runStopFrequencyModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_STOP_FREQUENCY)); - runStopLocationModel = Boolean.parseBoolean(propertyMap - .get(CtrampApplication.PROPERTIES_RUN_STOP_LOCATION)); - - boolean measureObjectSizes = false; - - try - { - useNewSlcSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, - USE_NEW_SLC_SOA_METHOD_PROPERTY_KEY); - - AccessibilitiesTable accTable = aggAcc.getAccessibilitiesTableObject(); - - // create the auto ownership choice model application object - if (runAutoOwnershipModel) - { - aoModel = new HouseholdAutoOwnershipModel(propertyMap, dmuFactory, accTable, - mandAcc); - tvtcModel = new TourVehicleTypeChoiceModel(propertyMap); - if ( measureObjectSizes ) logger.info ( "AO size: " + ObjectUtil.sizeOf( aoModel ) + ObjectUtil.sizeOf(tvtcModel)); - } - - if (runTransponderModel) - { - tcModel = new TransponderChoiceModel(propertyMap, dmuFactory, accTable, - pctHighIncome, pctMultipleAutos, avgtts, transpDist, pctDetour); - if (measureObjectSizes) - logger.info("TC size: " + ObjectUtil.sizeOf(tcModel)); - } - - if (runParkingProvisionModel) - { - ppModel = new ParkingProvisionModel(propertyMap, dmuFactory); - teModel = new TelecommuteModel(propertyMap, dmuFactory); - if (measureObjectSizes) { - logger.info("PP size: " + ObjectUtil.sizeOf(ppModel)); - logger.info("TE size: " + ObjectUtil.sizeOf(teModel)); - } - } - - if (runInternalExternalModel) - { - ieModel = new InternalExternalTripChoiceModel(propertyMap, modelStructure,dmuFactory); - if (measureObjectSizes) - logger.info("IE size: " + ObjectUtil.sizeOf(ieModel)); - } - - if (runCoordinatedDailyActivityPatternModel) - { - cdapModel = new HouseholdCoordinatedDailyActivityPatternModel(propertyMap, - modelStructure, dmuFactory, accTable); - if (measureObjectSizes) - logger.info("CDAP size: " + ObjectUtil.sizeOf(cdapModel)); - } - - if (runIndividualMandatoryTourFrequencyModel) - { - imtfModel = new HouseholdIndividualMandatoryTourFrequencyModel(propertyMap, - modelStructure, dmuFactory, accTable, mandAcc); - if (measureObjectSizes) - logger.info("IMTF size: " + ObjectUtil.sizeOf(imtfModel)); - } - - if (runMandatoryTourDepartureTimeAndDurationModel || runMandatoryTourModeChoiceModel) - { - immcModel = new TourModeChoiceModel(propertyMap, modelStructure, - TourModeChoiceModel.MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - if (measureObjectSizes) - logger.info("IMMC size: " + ObjectUtil.sizeOf(immcModel)); - - imtodModel = new HouseholdIndividualMandatoryTourDepartureAndDurationTime( - propertyMap, modelStructure, aggAcc.getWorkSegmentNameList(), dmuFactory, - immcModel); - if (measureObjectSizes) - logger.info("IMTOD size: " + ObjectUtil.sizeOf(imtodModel)); - } - - if(runEscortModel){ - escortModel = new SchoolEscortingModel(propertyMap,mgraManager,tazDistanceCalculator); - if ( measureObjectSizes ) logger.info ( "SEM size: " + ObjectUtil.sizeOf( escortModel ) ); - - } - - if (runJointTourFrequencyModel) - { - jtfModel = new JointTourModels(propertyMap, accTable, modelStructure, dmuFactory); - if (measureObjectSizes) - logger.info("JTF size: " + ObjectUtil.sizeOf(jtfModel)); - } - - if (runIndividualNonMandatoryTourFrequencyModel) - { - inmtfModel = new HouseholdIndividualNonMandatoryTourFrequencyModel(propertyMap, - dmuFactory, accTable, mandAcc); - if (measureObjectSizes) - logger.info("INMTF size: " + ObjectUtil.sizeOf(inmtfModel)); - } - - if (runIndividualNonMandatoryTourLocationChoiceModel || runJointTourLocationChoiceModel - || runIndividualNonMandatoryTourDepartureTimeAndDurationModel - || runJointTourDepartureTimeAndDurationModel - || runIndividualNonMandatoryTourModeChoiceModel || runJointTourModeChoiceModel) - { - nmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - TourModeChoiceModel.NON_MANDATORY_MODEL_INDICATOR, dmuFactory, logsumHelper); - if (measureObjectSizes) - logger.info("INMMC size: " + ObjectUtil.sizeOf(nmmcModel)); - } - - if (runIndividualNonMandatoryTourLocationChoiceModel || runJointTourLocationChoiceModel) - { - nmlcModel = new NonMandatoryDestChoiceModel(propertyMap, modelStructure, aggAcc, - dmuFactory, nmmcModel); - nmlcModel.setNonMandatorySoaProbs(nonManSoaDistProbs, nonManSoaSizeProbs); - if (measureObjectSizes) - logger.info("INMLC size: " + ObjectUtil.sizeOf(nmlcModel)); - } - - if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel - || runJointTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) - { - nmtodModel = new NonMandatoryTourDepartureAndDurationTime(propertyMap, - modelStructure, dmuFactory, nmmcModel); - if (measureObjectSizes) - logger.info("INMTOD size: " + ObjectUtil.sizeOf(nmtodModel)); - } - - if (runAtWorkSubTourFrequencyModel) - { - awfModel = new HouseholdAtWorkSubtourFrequencyModel(propertyMap, modelStructure, - dmuFactory); - if (measureObjectSizes) - logger.info("AWTF size: " + ObjectUtil.sizeOf(awfModel)); - } - - if (runAtWorkSubtourLocationChoiceModel - || runAtWorkSubtourDepartureTimeAndDurationModel - || runAtWorkSubtourModeChoiceModel) - { - awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - TourModeChoiceModel.AT_WORK_SUBTOUR_MODEL_INDICATOR, dmuFactory, - logsumHelper); - if (measureObjectSizes) - logger.info("AWMC size: " + ObjectUtil.sizeOf(awmcModel)); - } - - if (runAtWorkSubtourLocationChoiceModel) - { - awlcModel = new SubtourDestChoiceModel(propertyMap, modelStructure, aggAcc, - dmuFactory, awmcModel); - awlcModel.setNonMandatorySoaProbs(subTourSoaDistProbs, subTourSoaSizeProbs); - if (measureObjectSizes) - logger.info("AWLC size: " + ObjectUtil.sizeOf(awlcModel)); - } - - if (runAtWorkSubtourDepartureTimeAndDurationModel) - { - awtodModel = new SubtourDepartureAndDurationTime(propertyMap, modelStructure, - dmuFactory, awmcModel); - if (measureObjectSizes) - logger.info("AWTOD size: " + ObjectUtil.sizeOf(awtodModel)); - } - - if (runStopFrequencyModel) - { - stfModel = new StopFrequencyModel(propertyMap, dmuFactory, modelStructure, accTable); - if (measureObjectSizes) - logger.info("STF size: " + ObjectUtil.sizeOf(stfModel)); - } - - if (runStopLocationModel) - { - stlmcModel = new IntermediateStopChoiceModels(propertyMap, modelStructure, - dmuFactory, logsumHelper); - - mmModel = new MicromobilityChoiceModel(propertyMap,modelStructure,dmuFactory); - - // if the slcTazDistProbs are not null, they have been already - // computed, and it is - // not necessary for the thread creating this - // HouseholdChoiceModels object to - // compute them also. If slcTazDistProbs is null, compute them. - if (useNewSlcSoaMethod && slcSizeProbs == null) - { - - // compute the array of cumulative taz distance based SOA - // probabilities for each origin taz. - DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu = dmuFactory - .getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - DestChoiceTwoStageSoaProbabilitiesCalculator slcSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, - dmuFactory, - IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY, - IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_MODEL_PAGE, - IntermediateStopChoiceModels.PROPERTIES_UEC_SLC_SOA_DISTANCE_DATA_PAGE); - - computeSlcSoaProbabilities(slcSoaDistProbsObject, locChoiceDistSoaDmu, - stlmcModel.getSizeSegmentNameIndexMap(), - stlmcModel.getSizeSegmentArray()); - - stlmcModel.setupSlcDistanceBaseSoaModel(propertyMap, slcTazDistExpUtils, - slcSizeProbs, slcTazSize); - } - - if (measureObjectSizes) - logger.info("SLMT size: " + ObjectUtil.sizeOf(stlmcModel)); - } - - } catch (RuntimeException e) - { - - String lastModel = ""; - if (runAutoOwnershipModel && aoModel != null) lastModel += " ao"; - - if (runParkingProvisionModel && ppModel != null) lastModel += " fp"; - - if (runInternalExternalModel && ieModel != null) lastModel += " ie"; - - if (runCoordinatedDailyActivityPatternModel && cdapModel != null) lastModel += " cdap"; - - if (runIndividualMandatoryTourFrequencyModel && imtfModel != null) - lastModel += " imtf"; - - if (runMandatoryTourModeChoiceModel && immcModel != null) lastModel += " immc"; - - if (runMandatoryTourDepartureTimeAndDurationModel && imtodModel != null) - lastModel += " imtod"; - - if (runJointTourFrequencyModel && jtfModel != null) lastModel += " jtf"; - - if (runJointTourModeChoiceModel && nmmcModel != null) lastModel += " jmc"; - - if (runJointTourLocationChoiceModel && nmlcModel != null) lastModel += " jlc"; - - if (runJointTourDepartureTimeAndDurationModel && nmtodModel != null) - lastModel += " jtod"; - - if (runIndividualNonMandatoryTourFrequencyModel && inmtfModel != null) - lastModel += " inmtf"; - - if (runIndividualNonMandatoryTourModeChoiceModel && nmmcModel != null) - lastModel += " inmmc"; - - if (runIndividualNonMandatoryTourLocationChoiceModel && nmlcModel != null) - lastModel += " inmlc"; - - if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel && nmtodModel != null) - lastModel += " inmtod"; - - if (runAtWorkSubTourFrequencyModel && awfModel != null) lastModel += " awf"; - - if (runAtWorkSubtourModeChoiceModel && awmcModel != null) lastModel += " awmc"; - - if (runAtWorkSubtourLocationChoiceModel && awlcModel != null) lastModel += " awlc"; - - if (runAtWorkSubtourDepartureTimeAndDurationModel && awtodModel != null) - lastModel += " awtod"; - - if (runStopFrequencyModel && stfModel != null) lastModel += " stf"; - - if (runStopLocationModel && stlmcModel != null) lastModel += " stlmc"; - - logger.error("RuntimeException setting up HouseholdChoiceModels."); - logger.error("Models setup = " + lastModel); - logger.error("", e); - - throw new RuntimeException(); - } - - } - - public void runModels(Household hhObject) - { - - // check to see if restartModel was set and reset random number sequence - // appropriately if so. - checkRestartModel(hhObject); - - if (runAutoOwnershipModel) aoModel.applyModel(hhObject, false); - - if (runTransponderModel) tcModel.applyModel(hhObject); - - if (runParkingProvisionModel) { - ppModel.applyModel(hhObject); - teModel.applyModel(hhObject); - } - - if (runInternalExternalModel) ieModel.applyModel(hhObject, distanceToCordonsLogsums); - - if (runCoordinatedDailyActivityPatternModel) cdapModel.applyModel(hhObject); - - if (runIndividualMandatoryTourFrequencyModel) { - imtfModel.applyModel(hhObject); - tvtcModel.applyModelToMandatoryTours(hhObject); - } - - if (runMandatoryTourDepartureTimeAndDurationModel||runMandatoryTourModeChoiceModel) - imtodModel.applyModel(hhObject, runMandatoryTourDepartureTimeAndDurationModel,runMandatoryTourModeChoiceModel); - - if(runEscortModel){ - try { - escortModel.applyModel(hhObject); - } catch (Exception e) { - logger.fatal("Error Attempting to run escort model for household "+hhObject.getHhId()); - throw new RuntimeException(e); - } - } - - if (runJointTourFrequencyModel) { - jtfModel.applyModel(hhObject); - tvtcModel.applyModelToJointTours(hhObject); - } - - if (runJointTourLocationChoiceModel) nmlcModel.applyJointModel(hhObject); - - if (runJointTourDepartureTimeAndDurationModel) - nmtodModel.applyJointModel(hhObject, runJointTourDepartureTimeAndDurationModel, runJointTourModeChoiceModel); - - if (runIndividualNonMandatoryTourFrequencyModel) { - inmtfModel.applyModel(hhObject); - tvtcModel.applyModelToNonMandatoryTours(hhObject); - } - - if (runIndividualNonMandatoryTourLocationChoiceModel) nmlcModel.applyIndivModel(hhObject); - - if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) - nmtodModel.applyIndivModel(hhObject, runIndividualNonMandatoryTourDepartureTimeAndDurationModel, runIndividualNonMandatoryTourModeChoiceModel); - - if (runAtWorkSubTourFrequencyModel) { - awfModel.applyModel(hhObject); - tvtcModel.applyModelToAtWorkSubTours(hhObject); - } - - if (runAtWorkSubtourLocationChoiceModel) awlcModel.applyModel(hhObject); - - if (runAtWorkSubtourDepartureTimeAndDurationModel) - awtodModel.applyModel(hhObject, runAtWorkSubtourModeChoiceModel); - - if (runStopFrequencyModel) stfModel.applyModel(hhObject); - - if (runStopLocationModel) { - stlmcModel.applyModel(hhObject, false); - mmModel.applyModel(hhObject); - } - - } - - public void runModelsWithTiming(Household hhObject) - { - - // check to see if restartModel was set and reset random number sequence - // appropriately if so. - checkRestartModel(hhObject); - - if (runAutoOwnershipModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + AO_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - aoModel.applyModel(hhObject, false); - aoTime += (System.nanoTime() - check); - } - - if (runTransponderModel) - { - // long hhSeed = globalSeed + hhObject.getHhId() + TP_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - tcModel.applyModel(hhObject); - } - - if (runParkingProvisionModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + PP_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - ppModel.applyModel(hhObject); - teModel.applyModel(hhObject); - fpTime += (System.nanoTime() - check); - } - - if (runInternalExternalModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + PP_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - ieModel.applyModel(hhObject, distanceToCordonsLogsums); - ieTime += (System.nanoTime() - check); - } - - if (runCoordinatedDailyActivityPatternModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + CDAP_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - cdapModel.applyModel(hhObject); - cdapTime += (System.nanoTime() - check); - } - - if (runIndividualMandatoryTourFrequencyModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + IMTF_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - imtfModel.applyModel(hhObject); - tvtcModel.applyModelToMandatoryTours(hhObject); - imtfTime += (System.nanoTime() - check); - } - - if (runMandatoryTourDepartureTimeAndDurationModel||runMandatoryTourModeChoiceModel); - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + - // IMTOD_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - imtodModel.applyModel(hhObject, runMandatoryTourDepartureTimeAndDurationModel,runMandatoryTourModeChoiceModel); - long mcTime = imtodModel.getModeChoiceTime(); - imtodTime += (System.nanoTime() - check - mcTime); - imtmcTime += mcTime; - } - if(runEscortModel){ - long check = System.nanoTime(); - try { - escortModel.applyModel(hhObject); - } catch (Exception e) { - logger.fatal("Error Attempting to run escort model for household "+hhObject.getHhId()); - e.printStackTrace(); - } - escortTime += ( System.nanoTime() - check ); - } - - - if (runJointTourFrequencyModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + JTF_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - jtfModel.applyModel(hhObject); - tvtcModel.applyModelToJointTours(hhObject); - jtfTime += (System.nanoTime() - check); - } - - if (runJointTourLocationChoiceModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + JTDC_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - nmlcModel.applyJointModel(hhObject); - jtdcTime += (System.nanoTime() - check); - } - - if (runJointTourDepartureTimeAndDurationModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + JTOD_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - nmtodModel.applyJointModel(hhObject, runJointTourDepartureTimeAndDurationModel,runJointTourModeChoiceModel); - long mcTime = nmtodModel.getJointModeChoiceTime(); - jtodTime += (System.nanoTime() - check - mcTime); - jtmcTime += mcTime; - } - - if (runIndividualNonMandatoryTourFrequencyModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + - // INMTF_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - inmtfModel.applyModel(hhObject); - tvtcModel.applyModelToNonMandatoryTours(hhObject); - inmtfTime += (System.nanoTime() - check); - } - - if (runIndividualNonMandatoryTourLocationChoiceModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + - // INMDC_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - nmlcModel.resetSoaRunTime(); - nmlcModel.applyIndivModel(hhObject); - inmtdcSoaTime += nmlcModel.getSoaRunTime(); - inmtdcTime += (System.nanoTime() - check); - } - - if (runIndividualNonMandatoryTourDepartureTimeAndDurationModel||runIndividualNonMandatoryTourModeChoiceModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + - // INMTOD_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - nmtodModel.applyIndivModel(hhObject, runIndividualNonMandatoryTourDepartureTimeAndDurationModel,runIndividualNonMandatoryTourModeChoiceModel); - long mcTime = nmtodModel.getIndivModeChoiceTime(); - inmtodTime += (System.nanoTime() - check - mcTime); - inmtmcTime += mcTime; - } - - if (runAtWorkSubTourFrequencyModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + AWTF_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - awfModel.applyModel(hhObject); - tvtcModel.applyModelToAtWorkSubTours(hhObject); - awtfTime += (System.nanoTime() - check); - } - - if (runAtWorkSubtourLocationChoiceModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + AWDC_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - awlcModel.applyModel(hhObject); - awtdcSoaTime += awlcModel.getSoaRunTime(); - awtdcTime += (System.nanoTime() - check); - } - - if (runAtWorkSubtourDepartureTimeAndDurationModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + - // AWTOD_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - awtodModel.applyModel(hhObject, runAtWorkSubtourModeChoiceModel); - long mcTime = awtodModel.getModeChoiceTime(); - awtodTime += (System.nanoTime() - check - mcTime); - awtmcTime += mcTime; - } - - if (runStopFrequencyModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + STF_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - stfModel.applyModel(hhObject); - stfTime += (System.nanoTime() - check); - } - - if (runStopLocationModel) - { - long check = System.nanoTime(); - // long hhSeed = globalSeed + hhObject.getHhId() + SLC_SEED_OFFSET; - // hhObject.getHhRandom().setSeed( hhSeed ); - stlmcModel.applyModel(hhObject, true); - stdtmTime += (System.nanoTime() - check); - - long[] partials = stlmcModel.getStopTimes(); - for (int i = 0; i < returnPartialTimes.length; i++) - returnPartialTimes[i] += partials[i]; - - if (stlmcModel.getMaxAltsInSample() > maxAlts) - maxAlts = stlmcModel.getMaxAltsInSample(); - - mmModel.applyModel(hhObject); - } - - } - - private void checkRestartModel(Household hhObject) - { - - // none, ao, cdap, imtf, imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, - // inmtl, inmtod, stf, stl - // version 1.0.8.22 - changed model restart options - possible values - // for - // restart are now: none, uwsl, ao, imtf, jtf, inmtf, stf - - // if restartModel was specified, reset the random number sequence - // based on the cumulative count of random numbers drawn by the - // component - // preceding the one specified. - if (restartModelString.equalsIgnoreCase("") || restartModelString.equalsIgnoreCase("none")) return; - else if (restartModelString.equalsIgnoreCase("ao")) - { - hhObject.initializeForAoRestart(); - } else if (restartModelString.equalsIgnoreCase("imtf")) - { - hhObject.initializeForImtfRestart(); - } else if (restartModelString.equalsIgnoreCase("jtf")) - { - hhObject.initializeForJtfRestart(); - } else if (restartModelString.equalsIgnoreCase("inmtf")) - { - hhObject.initializeForInmtfRestart(); - } else if (restartModelString.equalsIgnoreCase("awf")) - { - hhObject.initializeForAwfRestart(); - } else if (restartModelString.equalsIgnoreCase("stf")) - { - hhObject.initializeForStfRestart(); - } - - } - - public int getModelIndex() - { - return modelIndex; - } - - public void zeroTimes() - { - aoTime = 0; - fpTime = 0; - ieTime = 0; - cdapTime = 0; - imtfTime = 0; - imtodTime = 0; - imtmcTime = 0; - jtfTime = 0; - jtdcTime = 0; - jtodTime = 0; - jtmcTime = 0; - inmtfTime = 0; - inmtdcTime = 0; - inmtdcSoaTime = 0; - inmtodTime = 0; - inmtmcTime = 0; - awtfTime = 0; - awtdcTime = 0; - awtdcSoaTime = 0; - awtodTime = 0; - awtmcTime = 0; - stfTime = 0; - stdtmTime = 0; - - Arrays.fill(returnPartialTimes, 0); - } - - public long[] getPartialStopTimes() - { - return returnPartialTimes; - } - - public long[] getTimes() - { - long[] returnTimes = new long[23]; - returnTimes[0] = aoTime; - returnTimes[1] = fpTime; - returnTimes[2] = ieTime; - returnTimes[3] = cdapTime; - returnTimes[4] = imtfTime; - returnTimes[5] = imtodTime; - returnTimes[6] = imtmcTime; - returnTimes[7] = jtfTime; - returnTimes[8] = jtdcTime; - returnTimes[9] = jtodTime; - returnTimes[10] = jtmcTime; - returnTimes[11] = inmtfTime; - returnTimes[12] = inmtdcSoaTime; - returnTimes[13] = inmtdcTime; - returnTimes[14] = inmtodTime; - returnTimes[15] = inmtmcTime; - returnTimes[16] = awtfTime; - returnTimes[17] = awtdcSoaTime; - returnTimes[18] = awtdcTime; - returnTimes[19] = awtodTime; - returnTimes[20] = awtmcTime; - returnTimes[21] = stfTime; - returnTimes[22] = stdtmTime; - return returnTimes; - } - - public int getMaxAlts() - { - return maxAlts; - } - - private void computeSlcSoaProbabilities( - DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, - DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu, - HashMap segmentNameIndexMap, double[][] dcSizeArray) - { - - // compute the exponentiated distance utilities that all segments of - // this tour purpose will share - slcTazDistExpUtils = computeTazDistanceExponentiatedUtilities(locChoiceSoaDistProbsObject, - locChoiceDistSoaDmu); - - slcTazSize = new double[dcSizeArray.length][]; - slcSizeProbs = new double[dcSizeArray.length][][]; - - // compute an array of SOA size probabilities for each segment - for (String segmentName : segmentNameIndexMap.keySet()) - { - - // compute the TAZ size values from the mgra values and the - // correspondence between mgras and tazs. - int segmentIndex = segmentNameIndexMap.get(segmentName); - slcTazSize[segmentIndex] = computeTazSize(dcSizeArray[segmentIndex]); - - slcSizeProbs[segmentIndex] = computeSizeSegmentProbabilities(dcSizeArray[segmentIndex], - slcTazSize[segmentIndex]); - - } - - } - - private double[][] computeSizeSegmentProbabilities(double[] size, double[] totalTazSize) - { - - int maxTaz = tdm.getMaxTaz(); - - // this is a 0-based array of cumulative probabilities - double[][] sizeProbs = new double[maxTaz][]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - - if (mgraArray == null) - { - sizeProbs[taz - 1] = new double[0]; - } else - { - - if (totalTazSize[taz] > 0) - { - sizeProbs[taz - 1] = new double[mgraArray.length]; - for (int i = 0; i < mgraArray.length; i++) - { - double mgraSize = size[mgraArray[i]]; - if (mgraSize > 0) mgraSize += 1; - sizeProbs[taz - 1][i] = mgraSize / totalTazSize[taz]; - } - } else - { - sizeProbs[taz - 1] = new double[0]; - } - } - - } - - return sizeProbs; - - } - - private double[][] computeTazDistanceExponentiatedUtilities( - DestChoiceTwoStageSoaProbabilitiesCalculator locChoiceSoaDistProbsObject, - DestChoiceTwoStageSoaTazDistanceUtilityDMU locChoiceDistSoaDmu) - { - - // compute the TAZ x TAZ exponentiated utilities array for sample - // selection utilities. - double[][] tazDistExpUtils = locChoiceSoaDistProbsObject - .computeDistanceUtilities(locChoiceDistSoaDmu); - for (int i = 0; i < tazDistExpUtils.length; i++) - for (int j = 0; j < tazDistExpUtils[i].length; j++) - { - if (tazDistExpUtils[i][j] < -500) tazDistExpUtils[i][j] = 0; - else tazDistExpUtils[i][j] = Math.exp(tazDistExpUtils[i][j]); - } - - return tazDistExpUtils; - - } - - private double[] computeTazSize(double[] size) - { - - int maxTaz = tdm.getMaxTaz(); - - double[] tazSize = new double[maxTaz + 1]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray != null) - { - for (int mgra : mgraArray) - { - tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); - } - } - - } - - return tazSize; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java deleted file mode 100644 index ef4223d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsManager.java +++ /dev/null @@ -1,682 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Date; -import java.util.HashMap; -import java.util.LinkedList; -import java.util.StringTokenizer; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import org.sandag.abm.accessibilities.NonTransitUtilities; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MathUtil; - -public final class HouseholdChoiceModelsManager - implements Serializable -{ - - private static transient Logger logger = Logger.getLogger(HouseholdChoiceModelsManager.class); - - private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; - - private static final String TAZ_FIELD_NAME = "TAZ"; - private static final String TP_CHOICE_AVG_TTS_FILE = "tc.choice.avgtts.file"; - private static final String AVGTTS_COLUMN_NAME = "AVGTTS"; - private static final String TRANSP_DIST_COLUMN_NAME = "DIST"; - private static final String PCT_DETOUR_COLUMN_NAME = "PCTDETOUR"; - - private static final String IE_EXTERNAL_TAZS_KEY = "external.tazs"; - private static final String IE_DISTANCE_LOGSUM_COEFF_KEY = "ie.logsum.distance.coeff"; - - private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE = "nonSchool.soa.uec.file"; - private static String PROPERTIES_ESCORT_DC_SOA_UEC_MODEL_PAGE = "escort.soa.uec.model"; - private static String PROPERTIES_ESCORT_DC_SOA_UEC_DATA_PAGE = "escort.soa.uec.data"; - private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_MODEL_PAGE = "other.nonman.soa.uec.model"; - private static String PROPERTIES_NON_MANDATORY_DC_SOA_UEC_DATA_PAGE = "other.nonman.soa.uec.data"; - private static String PROPERTIES_ATWORK_DC_SOA_UEC_MODEL_PAGE = "atwork.soa.uec.model"; - private static String PROPERTIES_ATWORK_DC_SOA_UEC_DATA_PAGE = "atwork.soa.uec.data"; - - private static HouseholdChoiceModelsManager objInstance = null; - - private LinkedList modelQueue = null; - - private HashMap propertyMap; - private String restartModelString; - private ModelStructure modelStructure; - private CtrampDmuFactoryIf dmuFactory; - - private MgraDataManager mgraManager; - private TazDataManager tdm; - - private int maxMgra; - private int maxTaz; - - private BuildAccessibilities aggAcc; - - private int completedHouseholds; - private int modelIndex; - - // store taz-taz exponentiated utilities (period, from taz, to taz) - private double[][][] sovExpUtilities; - private double[][][] hovExpUtilities; - private double[][][] nMotorExpUtilities; - private double[][][] maasExpUtilities; - - private double[] pctHighIncome; - private double[] pctMultipleAutos; - - private double[] avgtts; - private double[] transpDist; - private double[] pctDetour; - - private double[][][] nonMandatorySizeProbs; - private double[][][] nonMandatoryTazDistProbs; - private double[][][] subTourSizeProbs; - private double[][][] subTourTazDistProbs; - - private AutoTazSkimsCalculator tazDistanceCalculator; - - private boolean useNewSoaMethod; - private boolean logResults=false; - - private HouseholdChoiceModelsManager() - { - } - - public static synchronized HouseholdChoiceModelsManager getInstance() - { - // logger.info( - // "beginning of HouseholdChoiceModelsManager() - objInstance address = " - // + objInstance ); - if (objInstance == null) - { - objInstance = new HouseholdChoiceModelsManager(); - // logger.info( - // "after new HouseholdChoiceModelsManager() - objInstance address = " - // + objInstance ); - return objInstance; - } else - { - // logger.info( - // "returning current HouseholdChoiceModelsManager() - objInstance address = " - // + objInstance ); - return objInstance; - } - } - - // the task instances should call needToInitialize() first, then this method - // if necessary. - public synchronized void managerSetup(MatrixDataServerIf ms, - HouseholdDataManagerIf hhDataManager, HashMap propertyMap, - String restartModelString, ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) - { - - if (modelQueue != null) return; - - // get the DestChoiceModelManager instance and clear the objects that - // hold large memory references - DestChoiceModelManager.getInstance().clearDcModels(); - - modelIndex = 0; - completedHouseholds = 0; - - this.propertyMap = propertyMap; - this.restartModelString = restartModelString; - this.modelStructure = modelStructure; - this.dmuFactory = dmuFactory; - - logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - - mgraManager = MgraDataManager.getInstance(propertyMap); - maxMgra = mgraManager.getMaxMgra(); - - tdm = TazDataManager.getInstance(propertyMap); - maxTaz = tdm.getMaxTaz(); - - pctHighIncome = hhDataManager.getPercentHhsIncome100Kplus(); - pctMultipleAutos = hhDataManager.getPercentHhsMultipleAutos(); - readTpChoiceAvgTtsFile(); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - aggAcc = BuildAccessibilities.getInstance(); - if (!aggAcc.getAccessibilitiesAreBuilt()) - { - logger.info("creating Accessibilities Object for Household Choice Models."); - - aggAcc.setupBuildAccessibilities(propertyMap, false); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - - // assume that if the filename exists, at was created previously, - // either in another model run, or by the main client - // if the filename doesn't exist, then calculate the accessibilities - String projectDirectory = propertyMap - .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - boolean accFileReadFlag = Util.getBooleanValueFromPropertyMap(propertyMap, - CtrampApplication.READ_ACCESSIBILITIES); - - if (accFileReadFlag && (new File(accFileName)).canRead()) - { - - logger.info("filling Accessibilities Object in HouseholdChoiceModelManager by reading file: " - + accFileName + "."); - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - logger.info("filling Accessibilities Object HouseholdChoiceModelManager by calculating them."); - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - - } - - useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, - USE_NEW_SOA_METHOD_PROPERTY_KEY); - - if (useNewSoaMethod) - { - // compute the arrays of cumulative probabilities based on mgra size - // for mgras within each origin taz. - logger.info("pre-computing non-mandatory purpose SOA Distance and Size probabilities."); - computeNonMandatorySegmentSizeArrays(dmuFactory); - - logger.info("pre-computing at-work sub-tour purpose SOA Distance and Size probabilities."); - computeSubtourSegmentSizeArrays(modelStructure, dmuFactory); - } - - tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - // the first thread to reach this method initializes the modelQueue used - // to - // recycle hhChoiceModels objects. - modelQueue = new LinkedList(); - - mgraManager = MgraDataManager.getInstance(propertyMap); - - } - - /** - * @return DestChoiceModel object created if none is available from the - * queue. - * - */ - public synchronized HouseholdChoiceModels getHouseholdChoiceModelsObject(int taskIndex) - { - - String message = ""; - HouseholdChoiceModels hhChoiceModels = null; - - if (modelQueue.isEmpty()) - { - - NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, - hovExpUtilities, nMotorExpUtilities, maasExpUtilities); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( - propertyMap, ntUtilities, aggAcc.getExpConstants(), - logsumHelper.getBestTransitPathCalculator()); - - // calculate array of distanceToExternalCordon logsums by taz for - // use by internal-external model - double[] distanceToCordonsLogsums = computeTazDistanceToExternalCordonLogsums(); - - // create choice model object - hhChoiceModels = new HouseholdChoiceModels(++modelIndex, restartModelString, - propertyMap, modelStructure, dmuFactory, aggAcc, logsumHelper, mandAcc, - pctHighIncome, pctMultipleAutos, avgtts, transpDist, pctDetour, - nonMandatoryTazDistProbs, nonMandatorySizeProbs, subTourTazDistProbs, - subTourSizeProbs, distanceToCordonsLogsums, tazDistanceCalculator); - if(logResults){ - message = String.format("created hhChoiceModels=%d, task=%d, thread=%s.", modelIndex, - taskIndex, Thread.currentThread().getName()); - logger.info(message); - logger.info(""); - } - - } else - { - hhChoiceModels = modelQueue.remove(); - if(logResults){ - message = String.format("removed hhChoiceModels=%d from queue, task=%d, thread=%s.", - hhChoiceModels.getModelIndex(), taskIndex, Thread.currentThread().getName()); - logger.info(message); - logger.info(""); - } - } - - return hhChoiceModels; - - } - - /** - * return the HouseholdChoiceModels object to the manager's queue so that it - * may be used by another thread without it having to create one. - * - * @param hhModels - */ - public void returnHouseholdChoiceModelsObject(HouseholdChoiceModels hhModels, int startIndex, - int endIndex) - { - modelQueue.add(hhModels); - completedHouseholds += (endIndex - startIndex + 1); - if(logResults){ - logger.info("returned hhChoiceModels=" + hhModels.getModelIndex() + " to queue: thread=" - + Thread.currentThread().getName() + ", completedHouseholds=" + completedHouseholds - + "."); - } - } - - public synchronized void clearHhModels() - { - - if (modelQueue == null) return; - - logger.info(String.format("%s: clearing household choice models modelQueue, thread=%s.", - new Date(), Thread.currentThread().getName())); - while (!modelQueue.isEmpty()) - modelQueue.remove(); - - modelIndex = 0; - completedHouseholds = 0; - - modelQueue = null; - - } - - private void readTpChoiceAvgTtsFile() - { - - // construct input household file name from properties file values - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - - String inputFileName = propertyMap.get(TP_CHOICE_AVG_TTS_FILE); - String fileName = projectDirectory + inputFileName; - - TableDataSet table; - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - table = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading tp choice avgtts data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - int[] tazField = table.getColumnAsInt(TAZ_FIELD_NAME); - double[] avgttsField = table.getColumnAsDouble(AVGTTS_COLUMN_NAME); - double[] transpDistField = table.getColumnAsDouble(TRANSP_DIST_COLUMN_NAME); - double[] pctDetourField = table.getColumnAsDouble(PCT_DETOUR_COLUMN_NAME); - - avgtts = new double[tdm.getMaxTaz() + 1]; - transpDist = new double[tdm.getMaxTaz() + 1]; - pctDetour = new double[tdm.getMaxTaz() + 1]; - - // loop over the number of mgra records in the TableDataSet. - for (int k = 0; k < tdm.getMaxTaz(); k++) - { - - // get the mgra value for TableDataSet row k from the mgra field. - int taz = tazField[k]; - - avgtts[taz] = avgttsField[k]; - transpDist[taz] = transpDistField[k]; - pctDetour[taz] = pctDetourField[k]; - - } - - } - - private void computeNonMandatorySegmentSizeArrays(CtrampDmuFactoryIf dmuFactory) - { - - // compute the array of cumulative taz distance based SOA probabilities - // for each origin taz. - DestChoiceTwoStageSoaTazDistanceUtilityDMU dcDistSoaDmu = dmuFactory - .getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - // the size term array in aggAcc gives mgra*purpose - need an array of - // all mgras for one purpose - double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); - - String[] tourPurposeNames = {ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME}; - - int[] sizeSheetIndices = {BuildAccessibilities.ESCORT_INDEX, - BuildAccessibilities.SHOP_INDEX, BuildAccessibilities.OTH_MAINT_INDEX, - BuildAccessibilities.EATOUT_INDEX, BuildAccessibilities.VISIT_INDEX, - BuildAccessibilities.OTH_DISCR_INDEX}; - - HashMap nonMandatorySegmentNameIndexMap = new HashMap(); - HashMap nonMandatorySizeSegmentNameIndexMap = new HashMap(); - for (int k = 0; k < tourPurposeNames.length; k++) - { - nonMandatorySegmentNameIndexMap.put(tourPurposeNames[k], k); - nonMandatorySizeSegmentNameIndexMap.put(tourPurposeNames[k], sizeSheetIndices[k]); - } - - double[][] dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; - for (int i = 0; i < aggAccDcSizeArray.length; i++) - { - for (int m : nonMandatorySegmentNameIndexMap.values()) - { - int s = sizeSheetIndices[m]; - dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; - } - } - - // compute the arrays of cumulative probabilities based on mgra size for - // mgras within each origin taz. - nonMandatorySizeProbs = new double[tourPurposeNames.length][][]; - nonMandatoryTazDistProbs = new double[tourPurposeNames.length][][]; - - DestChoiceTwoStageSoaProbabilitiesCalculator nonManSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, - PROPERTIES_NON_MANDATORY_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_NON_MANDATORY_DC_SOA_UEC_DATA_PAGE); - - for (String tourPurpose : tourPurposeNames) - { - - int purposeSizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurpose); - - // compute the TAZ size values from the mgra values and the - // correspondence between mgras and tazs. - if (tourPurpose.equalsIgnoreCase(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) - { - - double[] mgraData = new double[maxMgra + 1]; - double[] tazData = null; - - // aggregate TAZ grade school enrollment and set array in DMU - for (int i = 1; i <= maxMgra; i++) - mgraData[i] = aggAcc.getMgraGradeSchoolEnrollment(i); - tazData = computeTazSize(mgraData); - dcDistSoaDmu.setTazGsEnrollment(tazData); - - // aggregate TAZ high school enrollment and set array in DMU - for (int i = 1; i <= maxMgra; i++) - mgraData[i] = aggAcc.getMgraHighSchoolEnrollment(i); - tazData = computeTazSize(mgraData); - dcDistSoaDmu.setTazHsEnrollment(tazData); - - // aggregate TAZ households and set array in DMU - for (int i = 1; i <= maxMgra; i++) - mgraData[i] = aggAcc.getMgraHouseholds(i); - tazData = computeTazSize(mgraData); - dcDistSoaDmu.setNumHhs(tazData); - - DestChoiceTwoStageSoaProbabilitiesCalculator escortSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, - PROPERTIES_ESCORT_DC_SOA_UEC_MODEL_PAGE, - PROPERTIES_ESCORT_DC_SOA_UEC_DATA_PAGE); - - logger.info(" " + tourPurpose + " probabilities"); - nonMandatoryTazDistProbs[purposeSizeIndex] = escortSoaDistProbsObject - .computeDistanceProbabilities(dcDistSoaDmu); - - } else - { - - // aggregate TAZ size for the non-mandatoy purpose and set array - // in DMU - double[] tazSize = computeTazSize(dcSizeArray[purposeSizeIndex]); - dcDistSoaDmu.setDestChoiceTazSize(tazSize); - - logger.info(" " + tourPurpose + " probabilities"); - nonMandatoryTazDistProbs[purposeSizeIndex] = nonManSoaDistProbsObject - .computeDistanceProbabilities(dcDistSoaDmu); - - } - - nonMandatorySizeProbs[purposeSizeIndex] = computeSizeSegmentProbabilities(dcSizeArray[purposeSizeIndex]); - - } - - } - - private void computeSubtourSegmentSizeArrays(ModelStructure modelStructure, - CtrampDmuFactoryIf dmuFactory) - { - - // compute the array of cumulative taz distance based SOA probabilities - // for each origin taz. - DestChoiceTwoStageSoaTazDistanceUtilityDMU dcDistSoaDmu = dmuFactory - .getDestChoiceSoaTwoStageTazDistUtilityDMU(); - - // the size term array in aggAcc gives mgra*purpose - need an array of - // all mgras for one purpose - double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); - - String[] tourPurposeNames = {modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME, - modelStructure.AT_WORK_EAT_PURPOSE_NAME, modelStructure.AT_WORK_MAINT_PURPOSE_NAME}; - - int[] sizeSheetIndices = {SubtourDestChoiceModel.PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET, - SubtourDestChoiceModel.PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET, - SubtourDestChoiceModel.PROPERTIES_AT_WORK_OTHER_SIZE_SHEET}; - - HashMap segmentNameIndexMap = new HashMap(); - HashMap sizeSegmentNameIndexMap = new HashMap(); - for (int k = 0; k < tourPurposeNames.length; k++) - { - segmentNameIndexMap.put(tourPurposeNames[k], k); - sizeSegmentNameIndexMap.put(tourPurposeNames[k], sizeSheetIndices[k]); - } - - double[][] dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; - for (int i = 0; i < aggAccDcSizeArray.length; i++) - { - for (int m : segmentNameIndexMap.values()) - { - int s = sizeSheetIndices[m]; - dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; - } - } - - // compute the arrays of cumulative probabilities based on mgra size for - // mgras within each origin taz. - subTourSizeProbs = new double[tourPurposeNames.length][][]; - subTourTazDistProbs = new double[tourPurposeNames.length][][]; - - DestChoiceTwoStageSoaProbabilitiesCalculator subTourSoaDistProbsObject = new DestChoiceTwoStageSoaProbabilitiesCalculator( - propertyMap, dmuFactory, PROPERTIES_NON_MANDATORY_DC_SOA_UEC_FILE, - PROPERTIES_ATWORK_DC_SOA_UEC_MODEL_PAGE, PROPERTIES_ATWORK_DC_SOA_UEC_DATA_PAGE); - - for (String tourPurpose : tourPurposeNames) - { - - int purposeSizeIndex = segmentNameIndexMap.get(tourPurpose); - - // aggregate TAZ size for the non-mandatoy purpose and set array in - // DMU - double[] tazSize = computeTazSize(dcSizeArray[purposeSizeIndex]); - dcDistSoaDmu.setDestChoiceTazSize(tazSize); - - logger.info(" " + tourPurpose + " probabilities"); - subTourTazDistProbs[purposeSizeIndex] = subTourSoaDistProbsObject - .computeDistanceProbabilities(dcDistSoaDmu); - - subTourSizeProbs[purposeSizeIndex] = computeSizeSegmentProbabilities(dcSizeArray[purposeSizeIndex]); - - } - - } - - private double[] computeTazSize(double[] size) - { - - // this is a 0-based array of cumulative probabilities - double[] tazSize = new double[maxTaz + 1]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - if (mgraArray != null) - { - for (int mgra : mgraArray) - { - tazSize[taz] += size[mgra] + (size[mgra] > 0 ? 1 : 0); - } - } - - } - - return tazSize; - - } - - private double[][] computeSizeSegmentProbabilities(double[] size) - { - - // this is a 0-based array of cumulative probabilities - double[][] sizeProbs = new double[maxTaz][]; - - for (int taz = 1; taz <= tdm.getMaxTaz(); taz++) - { - - int[] mgraArray = tdm.getMgraArray(taz); - - if (mgraArray == null) - { - sizeProbs[taz - 1] = new double[0]; - } else - { - double totalSize = 0; - for (int mgra : mgraArray) - totalSize += size[mgra] + (size[mgra] > 0 ? 1 : 0); - - if (totalSize > 0) - { - sizeProbs[taz - 1] = new double[mgraArray.length]; - for (int i = 0; i < mgraArray.length; i++) - { - double mgraSize = size[mgraArray[i]]; - if (mgraSize > 0) mgraSize += 1; - sizeProbs[taz - 1][i] = mgraSize / totalSize; - } - } else - { - sizeProbs[taz - 1] = new double[0]; - } - } - - } - - return sizeProbs; - - } - - private double[] computeTazDistanceToExternalCordonLogsums() - { - - int maxTaz = tdm.getMaxTaz(); - String uecPath = propertyMap.get("uec.path"); - String altFileName = uecPath + propertyMap.get("internalExternal.dc.uec.alts.file"); - TableDataSet altData = readFile(altFileName); - - int tazCol = altData.getColumnPosition("taz"); - int ieCol = altData.getColumnPosition("iePct"); - altData.buildIndex(tazCol); - - // get parameters used to develop distance to cordon logsums for IE - // model - String coeffString = propertyMap.get(IE_DISTANCE_LOGSUM_COEFF_KEY); - double coeff = Double.parseDouble(coeffString); - - ArrayList tazList = new ArrayList(); - String externalTazListString = propertyMap.get(IE_EXTERNAL_TAZS_KEY); - StringTokenizer st = new StringTokenizer(externalTazListString, ","); - while (st.hasMoreTokens()) - { - String listValue = st.nextToken(); - int tazValue = Integer.parseInt(listValue.trim()); - tazList.add(tazValue); - } - int[] externalTazs = new int[tazList.size()]; - for (int i = 0; i < externalTazs.length; i++) - externalTazs[i] = tazList.get(i); - - // get stored distance arrays - double[][][] periodDistanceMatrices = tazDistanceCalculator - .getStoredFromTazToAllTazsDistanceSkims(); - - // compute the TAZ x EXTERNAL TAZ distance based logsums. - double[] tazDistLogsums = new double[maxTaz + 1]; - for (int i = 1; i <= maxTaz; i++) - { - - double sum = 0; - for (int j = 0; j < externalTazs.length; j++) - { - double distanceToExternal = periodDistanceMatrices[ModelStructure.MD_SKIM_PERIOD_INDEX][i][externalTazs[j]]; - double iePct = altData.getValueAt(externalTazs[j], ieCol); - sum += iePct * MathUtil.exp(coeff * distanceToExternal); - } - - tazDistLogsums[i] = MathUtil.log(sum); - } - - return tazDistLogsums; - - } - - /** - * Read the file and return the TableDataSet. - * - * @param fileName - * @return data - */ - private TableDataSet readFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet data; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - data = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return data; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java deleted file mode 100644 index 5ca5dcc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdChoiceModelsTaskJppf.java +++ /dev/null @@ -1,207 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.net.UnknownHostException; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.jppf.node.protocol.AbstractTask; -import org.jppf.node.protocol.DataProvider; -import org.jppf.node.protocol.JPPFTask; - -import com.pb.common.calculator.MatrixDataServerIf; - -public class HouseholdChoiceModelsTaskJppf - extends AbstractTask -{ - - private transient HashMap propertyMap; - private transient MatrixDataServerIf ms; - private transient HouseholdDataManagerIf hhDataManager; - private transient ModelStructure modelStructure; - private transient CtrampDmuFactoryIf dmuFactory; - private transient String restartModelString; - - private int startIndex; - private int endIndex; - private int taskIndex; - - private int maxAlts; - - private boolean runWithTiming; - private boolean logResults=false; - - public HouseholdChoiceModelsTaskJppf(int taskIndex, int startIndex, int endIndex) - { - this.startIndex = startIndex; - this.endIndex = endIndex; - this.taskIndex = taskIndex; - runWithTiming = true; - } - - public void run() - { - - long startTime = System.nanoTime(); - - Logger logger = Logger.getLogger(this.getClass()); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " - + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - try - { - - DataProvider dataProvider = getDataProvider(); - - propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); - logResults = Util.getStringValueFromPropertyMap(propertyMap, "RunModel.LogResults") - .equalsIgnoreCase("true"); - ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); - hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); - modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); - dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); - restartModelString = (String) dataProvider.getParameter("restartModelString"); - - } catch (Exception e) - { - e.printStackTrace(); - } - - // get the factory object used to create and recycle - // HouseholdChoiceModels objects. - HouseholdChoiceModelsManager modelManager = HouseholdChoiceModelsManager.getInstance(); - modelManager.managerSetup(ms, hhDataManager, propertyMap, restartModelString, - modelStructure, dmuFactory); - - HouseholdChoiceModels hhModel = modelManager.getHouseholdChoiceModelsObject(taskIndex); - - long setup1 = 0; - long setup2 = 0; - long setup3 = 0; - long setup4 = 0; - long setup5 = 0; - - setup1 = (System.nanoTime() - startTime) / 1000000; - - Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); - - setup2 = (System.nanoTime() - startTime) / 1000000; - - boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, - HouseholdDataManager.DEBUG_HHS_ONLY_KEY); - - if (runWithTiming) hhModel.zeroTimes(); - for (int i = 0; i < householdArray.length; i++) - { - - // for debugging only - process only household objects specified for - // debugging, if property key was set to true - if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; - - try - { - if (runWithTiming) hhModel.runModelsWithTiming(householdArray[i]); - else hhModel.runModels(householdArray[i]); - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught in taskIndex=%d hhModel index=%d applying hh model for i=%d, hhId=%d.", - taskIndex, hhModel.getModelIndex(), i, householdArray[i].getHhId())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - } - - long[] componentTimes = hhModel.getTimes(); - long[] partialStopTimes = hhModel.getPartialStopTimes(); - - if (hhModel.getMaxAlts() > maxAlts) maxAlts = hhModel.getMaxAlts(); - - setup3 = (System.nanoTime() - startTime) / 1000000; - - hhDataManager.setHhArray(householdArray, startIndex); - - setup4 = (System.nanoTime() - startTime) / 1000000; - - logger.info(String - .format("end of household choice model thread=%s, task[%d], hhModel[%d], startIndex=%d, endIndex=%d", - threadName, taskIndex, hhModel.getModelIndex(), startIndex, endIndex)); - - setResult(String.format("taskIndex=%d, hhModelInstance=%d, startIndex=%d, endIndex=%d", - taskIndex, hhModel.getModelIndex(), startIndex, endIndex)); - - setup5 = (System.nanoTime() - startTime) / 1000000; - - if(logResults){ - logger.info("task=" + taskIndex + ", setup=" + setup1 + ", getHhs=" + (setup2 - setup1) - + ", processHhs=" + (setup3 - setup2) + ", putHhs=" + (setup4 - setup3) - + ", return model=" + (setup5 - setup4) + "."); - } - - if (runWithTiming) - logModelComponentTimes(componentTimes, partialStopTimes, logger, - hhModel.getModelIndex()); - - // this has to be the last statement in this method. - // add this DestChoiceModel instance to the static queue shared by other - // tasks of this type - modelManager.returnHouseholdChoiceModelsObject(hhModel, startIndex, endIndex); - - } - - private void logModelComponentTimes(long[] componentTimes, long[] partialStopTimes, - Logger logger, int modelIndex) - { - - String[] label1 = {"AO", "FP", "IE", "CDAP", "IMTF", "IMTOD", "IMMC", "JTF", "JTDC", - "JTTOD", "JTMC", "INMTF", "INMTDCSOA", "INMTDCTOT", "INMTTOD", "INMTMC", "AWTF", - "AWTDCSOA", "AWTDCTOT", "AWTTOD", "AWTMC", "STF", "STDTM"}; - - logger.info("Household choice model component runtimes (in milliseconds) for task: " - + taskIndex + ", modelIndex: " + modelIndex + ", startIndex: " + startIndex - + ", endIndex: " + endIndex); - - float total = 0; - for (int i = 0; i < componentTimes.length; i++) - { - float time = (componentTimes[i] / 1000000); - logger.info(String.format("%-6d%30s:%15.1f", (i + 1), label1[i], time)); - total += time; - } - logger.info(String.format("%-6s%30s:%10.1f", "Total", "Total all components", total)); - logger.info(""); - - String[] label2 = {"SLC SOA AUTO", "SLC SOA OTHER", "SLC LS", "SLC DIST", "SLC", "SLC TOT", - "S TOD", "S MC", "TOTAL"}; - - logger.info(""); - logger.info("Times for parts of intermediate stop models:"); - for (int i = 0; i < partialStopTimes.length; i++) - { - float time = (partialStopTimes[i] / 1000000); - logger.info(String.format("%-6d%30s:%15.1f", (i + 1), label2[i], time)); - } - - } - - public String getId() - { - return Integer.toString(taskIndex); - } - - public int getMaxAlts() - { - return maxAlts; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java deleted file mode 100644 index 2b26011..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdCoordinatedDailyActivityPatternModel.java +++ /dev/null @@ -1,1423 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import com.pb.common.calculator.VariableTable; -import com.pb.common.model.Alternative; -import com.pb.common.model.ConcreteAlternative; -import com.pb.common.model.LogitModel; -import com.pb.common.model.ModelException; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * Implements a coordinated daily activity pattern model, which is a joint - * choice of activity types of each member of a household. The class builds and - * applies separate choice models for households of sizes 1, 2, 3, 4, and 5. For - * households larger than 5, the persons in the household are ordered such that - * the first 5 members include up to 2 workers and 3 children (youngest to - * oldest), the 5-person model is applied for these 5 household members, than a - * separate, simple cross-sectional distribution is looked up for the remaining - * household members. - * - * The utilities are computed using four separate UEC spreadsheets. The first - * computes the activity utility for each person individually; the second - * computes the activity utility for each person when paired with each other - * person; the third computes the activity utility for each person when paired - * with each group of two other people in the household; and the fourth computes - * the activity utility considering all the members of the household. These - * utilities are then aggregated to represent each possible activity pattern for - * the household, and the choice is made. For households larger than 5, a second - * model is applied after the first, which selects a pattern for the 5+ - * household members from a predefined distribution. - * - * @author D. Ory - * - */ -public class HouseholdCoordinatedDailyActivityPatternModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdCoordinatedDailyActivityPatternModel.class); - private transient Logger cdapLogger = Logger.getLogger("cdap"); - private transient Logger cdapUecLogger = Logger.getLogger("cdap_uec"); - private transient Logger cdapLogsumLogger = Logger.getLogger("cdap_logsum"); - - private static final String UEC_FILE_NAME_PROPERTY = "cdap.uec.file"; - private static final String UEC_DATA_PAGE_PROPERTY = "cdap.data.page"; - private static final String UEC_ONE_PERSON_UTILITY_PAGE_PROPERTY = "cdap.one.person.page"; - private static final String UEC_TWO_PERSON_UTILITY_PAGE_PROPERTY = "cdap.two.person.page"; - private static final String UEC_THREE_PERSON_UTILITY_PAGE_PROPERTY = "cdap.three.person.page"; - private static final String UEC_ALL_PERSON_UTILITY_PAGE_PROPERTY = "cdap.all.person.page"; - private static final String UEC_JOINT_UTILITY_PAGE_PROPERTY = "cdap.joint.page"; - - public static final int MAX_MODEL_HH_SIZE = 5; - - private static final String MANDATORY_PATTERN = Definitions.MANDATORY_PATTERN; - private static final String NONMANDATORY_PATTERN = Definitions.NONMANDATORY_PATTERN; - private static final String HOME_PATTERN = Definitions.HOME_PATTERN; - private static final String[] ACTIVITY_NAME_ARRAY = { - MANDATORY_PATTERN, NONMANDATORY_PATTERN, HOME_PATTERN }; - - private ModelStructure modelStructure; - private double[][] fixedCumulativeProportions; - - // collection of logit models - one for each household size - private ArrayList logitModelList; - - private AccessibilitiesTable accTable; - - // DMU for the UEC - private CoordinatedDailyActivityPatternDMU cdapDmuObject; - - // re-ordered collection of households - private Person[] cdapPersonArray; - - // Five separate UECs to compute segments of the utility - private UtilityExpressionCalculator onePersonUec, twoPeopleUec, threePeopleUec, - allMemberInteractionUec, jointUec; - - public HouseholdCoordinatedDailyActivityPatternModel(HashMap propertyMap, - ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory, - AccessibilitiesTable myAccTable) - { - - modelStructure = myModelStructure; - accTable = myAccTable; - - // setup the coordinated daily activity pattern choice model objects - createLogitModels(); - setupCoordinatedDailyActivityPatternModelApplication(propertyMap, dmuFactory); - - } - - private void setupCoordinatedDailyActivityPatternModelApplication( - HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - - logger.info("setting up CDAP choice model."); - - // locate the coordinated daily activity pattern choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String cdapUecFile = propertyMap.get(UEC_FILE_NAME_PROPERTY); - cdapUecFile = uecPath + cdapUecFile; - - int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, UEC_DATA_PAGE_PROPERTY); - int onePersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, - UEC_ONE_PERSON_UTILITY_PAGE_PROPERTY); - int twoPersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, - UEC_TWO_PERSON_UTILITY_PAGE_PROPERTY); - int threePersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, - UEC_THREE_PERSON_UTILITY_PAGE_PROPERTY); - int allPersonPage = Util.getIntegerValueFromPropertyMap(propertyMap, - UEC_ALL_PERSON_UTILITY_PAGE_PROPERTY); - int jointPage = Util.getIntegerValueFromPropertyMap(propertyMap, - UEC_JOINT_UTILITY_PAGE_PROPERTY); - - // create the coordinated daily activity pattern choice model DMU - // object. - cdapDmuObject = dmuFactory.getCoordinatedDailyActivityPatternDMU(); - - // create the uecs - onePersonUec = new UtilityExpressionCalculator(new File(cdapUecFile), onePersonPage, - dataPage, propertyMap, (VariableTable) cdapDmuObject); - twoPeopleUec = new UtilityExpressionCalculator(new File(cdapUecFile), twoPersonPage, - dataPage, propertyMap, (VariableTable) cdapDmuObject); - threePeopleUec = new UtilityExpressionCalculator(new File(cdapUecFile), threePersonPage, - dataPage, propertyMap, (VariableTable) cdapDmuObject); - allMemberInteractionUec = new UtilityExpressionCalculator(new File(cdapUecFile), - allPersonPage, dataPage, propertyMap, (VariableTable) cdapDmuObject); - jointUec = new UtilityExpressionCalculator(new File(cdapUecFile), jointPage, dataPage, - propertyMap, (VariableTable) cdapDmuObject); - - // get the proportions by person type - double[][] fixedRelativeProportions = modelStructure.getCdap6PlusProps(); - fixedCumulativeProportions = new double[fixedRelativeProportions.length][]; - - // i loops over personTypes, 0 not used. - for (int i = 1; i < fixedRelativeProportions.length; i++) - { - fixedCumulativeProportions[i] = new double[fixedRelativeProportions[i].length]; - - // j loops over cdap patterns, can skip index 0. - fixedCumulativeProportions[i][0] = fixedRelativeProportions[i][0]; - for (int j = 1; j < fixedRelativeProportions[i].length; j++) - fixedCumulativeProportions[i][j] = fixedCumulativeProportions[i][j - 1] - + fixedRelativeProportions[i][j]; - - // calculate the difference between 1.0 and the cumulative - // proportion and - // add to the Mandatory category (j==0) - // to make sure the cumulative propbabilities sum to exactly 1.0. - double diff = 1.0 - fixedCumulativeProportions[i][fixedRelativeProportions[i].length - 1]; - fixedCumulativeProportions[i][0] += diff; - } - - } - - public void applyModel(Household hhObject) - { - - if (hhObject.getDebugChoiceModels()) - hhObject.logHouseholdObject("Pre CDAP Household " + hhObject.getHhId() + " Object", - cdapLogger); - - // get the activity pattern choice - String pattern = getCoordinatedDailyActivityPatternChoice(hhObject); - - // set the pattern for the household - hhObject.setCoordinatedDailyActivityPatternResult(pattern); - - // set the pattern for each person and count by person type - Person[] personArray = hhObject.getPersons(); - for (int j = 1; j < personArray.length; ++j) - { - String activityString = pattern.substring(j - 1, j); - personArray[j].setDailyActivityResult(activityString); - } // j (person loop) - - // log results for debug households - if (hhObject.getDebugChoiceModels()) - { - - cdapLogger.info(" "); - cdapLogger.info("CDAP Chosen Pattern by Person Type"); - cdapLogger - .info("(* indicates person was involved in coordinated choice; no * indicates choice by fixed proportions)"); - cdapLogger.info("CDAP # Type FT W PT W UNIV NONW RETR SCHD SCHN PRES"); - cdapLogger.info("------ ---- ---- ---- ---- ---- ---- ---- ---- ----"); - - String bString = ""; - for (int j = 1; j < personArray.length; ++j) - { - - Person[] tempPersonArray = getPersonsNotModeledByCdap(MAX_MODEL_HH_SIZE); - - boolean persNumMatch = false; - for (int jj = 1; jj < tempPersonArray.length; jj++) - { - if (tempPersonArray[jj].getPersonNum() == personArray[j].getPersonNum()) - persNumMatch = true; - } - - String persNumString = ""; - if (persNumMatch) persNumString = String.format("%d ", j); - else persNumString = String.format("%d *", j); - - String pString = pattern.substring(j - 1, j); - String stringToLog = ""; - - if (personArray[j].getPersonTypeIsFullTimeWorker() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "FT W", pString, bString, bString, bString, bString, bString, bString, - bString); - } else if (personArray[j].getPersonTypeIsPartTimeWorker() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "PT W", bString, pString, bString, bString, bString, bString, bString, - bString); - } else if (personArray[j].getPersonIsUniversityStudent() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "UNIV", bString, bString, pString, bString, bString, bString, bString, - bString); - } else if (personArray[j].getPersonIsNonWorkingAdultUnder65() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "NONW", bString, bString, bString, pString, bString, bString, bString, - bString); - } else if (personArray[j].getPersonIsNonWorkingAdultOver65() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "RETR", bString, bString, bString, bString, pString, bString, bString, - bString); - } else if (personArray[j].getPersonIsStudentDriving() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "SCHD", bString, bString, bString, bString, bString, pString, bString, - bString); - } else if (personArray[j].getPersonIsStudentNonDriving() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "SCHN", bString, bString, bString, bString, bString, bString, pString, - bString); - } else if (personArray[j].getPersonIsPreschoolChild() == 1) - { - stringToLog = String.format("%6s%5s%5s%5s%5s%5s%5s%5s%5s%5s", persNumString, - "PRES", bString, bString, bString, bString, bString, bString, bString, - pString); - } - - cdapLogger.info(stringToLog); - - } // j (person loop) - - cdapLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - cdapLogger.info(""); - cdapLogger.info(""); - - } // if traceMe - - hhObject.setCdapRandomCount(hhObject.getHhRandomCount()); - - } - - /** - * Prepares a separate logit model for households of size 1, 2, 3, 4, and 5. - * Each model has 3^n alternatives, where n is the household size. The - * models are cleared and re-used for each household of the specified size. - * - */ - private void createLogitModels() - { - - // new the collection of logit models - logitModelList = new ArrayList(MAX_MODEL_HH_SIZE); - - // build a model for each HH size - for (int i = 0; i < MAX_MODEL_HH_SIZE; ++i) - { - - int hhSize = i + 1; - - // create the working model - LogitModel workingLogitModel = new LogitModel(hhSize + " Person HH"); - - // compute the number of alternatives - int numberOfAlternatives = 1; - for (int j = 0; j < hhSize; ++j) - numberOfAlternatives *= ACTIVITY_NAME_ARRAY.length; - - // create a counter for each of the people in the hh - int[] counterForEachPerson = new int[hhSize]; - Arrays.fill(counterForEachPerson, 0); - - // create the alternatives and add them to the logit model - int numberOfAltsCounter = 0; - int totalAltsCounter = 0; - while (numberOfAltsCounter < numberOfAlternatives) - { - - // set the string for the alternative - String alternativeName = ""; - int numOutOfHomeActivites = 0; - for (int j = 0; j < hhSize; ++j) - { - alternativeName += ACTIVITY_NAME_ARRAY[counterForEachPerson[j]]; - if (!ACTIVITY_NAME_ARRAY[counterForEachPerson[j]] - .equalsIgnoreCase(HOME_PATTERN)) numOutOfHomeActivites++; - } - - // create the alternative and add it to the model - if (numOutOfHomeActivites < 2) - { - ConcreteAlternative tempAlt = new ConcreteAlternative(alternativeName + "0", - totalAltsCounter); - workingLogitModel.addAlternative(tempAlt); - numberOfAltsCounter++; - totalAltsCounter++; - - } else - { - - ConcreteAlternative tempAlt = new ConcreteAlternative(alternativeName + "0", - totalAltsCounter); - workingLogitModel.addAlternative(tempAlt); - numberOfAltsCounter++; - totalAltsCounter++; - - tempAlt = new ConcreteAlternative(alternativeName + "j", totalAltsCounter); - workingLogitModel.addAlternative(tempAlt); - totalAltsCounter++; - - } - - // check increment the counters - for (int j = 0; j < hhSize; ++j) - { - counterForEachPerson[j]++; - if (counterForEachPerson[j] == ACTIVITY_NAME_ARRAY.length) counterForEachPerson[j] = 0; - else break; - } - - } - - // add the model to the array list - logitModelList.add(i, workingLogitModel); - - } // for i max hh size - - } - - /** - * Selects the coordinated daily activity pattern choice for the passed in - * Household. The method works for households of all sizes, though two - * separate models are applied for households with more than 5 members. - * - * @param householdObject - * @return a string of length household size, where each character in the - * string represents the activity pattern for that person, in order - * (see Household.reOrderPersonsForCdap method). - */ - public String getCoordinatedDailyActivityPatternChoice(Household householdObject) - { - - // set all household level dmu variables - cdapDmuObject.setHousehold(householdObject); - - // set the hh size (cap modeled size at MAX_MODEL_HH_SIZE) - int actualHhSize = householdObject.getSize(); - int modelHhSize = Math.min(MAX_MODEL_HH_SIZE, actualHhSize); - - // reorder persons for large households if need be - reOrderPersonsForCdap(householdObject); - - // get the logit model we need and clear it of any lingering probilities - LogitModel workingLogitModel = logitModelList.get(modelHhSize - 1); - workingLogitModel.clear(); - - // get the alternatives and reset the utilities to zero - ArrayList alternativeList = workingLogitModel.getAlternatives(); - for (int i = 0; i < alternativeList.size(); ++i) - { - Alternative tempAlt = (Alternative) alternativeList.get(i); - tempAlt.setUtility(0.0); - } - - // write the debug header if we have a trace household - if (householdObject.getDebugChoiceModels()) - { - - LogitModel.setLogger(cdapLogsumLogger); - - cdapLogger.info(" "); - cdapLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - cdapLogger.info("CDAP Model: Debug Statement for Household ID: " - + householdObject.getHhId()); - String firstHeader = "Utility Segment PersonA PersonB PersonC"; - String secondHeader = "------------------------------ -------- -------- --------"; - for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) - { - firstHeader += " " + ACTIVITY_NAME_ARRAY[j] + " util"; - secondHeader += " ---------"; - } - - cdapLogger.info(firstHeader); - cdapLogger.info(secondHeader); - - } - - // all the alternatives are available for all households (1-based, - // ignore 0 - // index and set other three to 1.) - int[] availability = {-1, 1, 1, 1}; - - String[] accStrings = {"", "hov0", "hov1", "hov2"}; - float retAccess = accTable.getAggregateAccessibility( - accStrings[householdObject.getAutoSufficiency()], householdObject.getHhMgra()); - - cdapDmuObject.setRetailAccessibility(retAccess); - - // loop through each person - for (int i = 0; i < modelHhSize; ++i) - { - - // get personA - Person personA = getCdapPerson(i + 1); - - // set the person level dmu variables - cdapDmuObject.setPersonA(personA); - - int workMgra = personA.getWorkLocation(); - if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) cdapDmuObject - .setWorkLocationModeChoiceLogsumA(personA.getWorkLocationLogsum()); - else cdapDmuObject.setWorkLocationModeChoiceLogsumA(0.0); - - int schoolMgra = personA.getPersonSchoolLocationZone(); - int studentType = getStudentTypeForThisCdapPerson(personA); - if (studentType > 0 && schoolMgra > 0) - { - cdapDmuObject.setSchoolLocationModeChoiceLogsumA(personA.getSchoolLocationLogsum()); - } else - { - cdapDmuObject.setSchoolLocationModeChoiceLogsumA(0.0); - } - - // compute the single person utilities - double[] firstPersonUtilities = onePersonUec.solve(cdapDmuObject.getIndexValues(), - cdapDmuObject, availability); - - // log these utilities for trace households - if (householdObject.getDebugChoiceModels()) - { - - String stringToLog = String.format("%-30s%9d%9s%9s", "OnePerson", (i + 1), "--", - "--"); - - for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) - { - stringToLog += String.format("%10.4f", firstPersonUtilities[j]); - } - cdapLogger.info(stringToLog); - - cdapUecLogger.info("PersonA:"); - personA.logEntirePersonObject(cdapUecLogger); - onePersonUec.logAnswersArray(cdapUecLogger, "ONE PERSON, personA personNum=" - + personA.getPersonNum()); - - } // debug trace - - // align the one person utilities with the alternatives for person i and calculate their individual logsum - float individualLogsum = 0; - for (int j = 0; j < alternativeList.size(); ++j) - { - - // get the name of the alternative - Alternative tempAlt = (Alternative) alternativeList.get(j); - String altName = tempAlt.getName(); - - // get the name of the activity for this person in the - // alternative - // string - String altNameForPersonA = altName.substring(i, i + 1); - - // align the utility results with this activity - for (int k = 0; k < ACTIVITY_NAME_ARRAY.length; ++k) - { - - if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[k])) - { - double currentUtility = tempAlt.getUtility(); - tempAlt.setUtility(currentUtility + firstPersonUtilities[k]); - individualLogsum += Math.exp(firstPersonUtilities[k]); - } - } // k - - } // j - - individualLogsum = (float) Math.log(individualLogsum); - personA.setCdapLogsum(individualLogsum); - - // loop through all possible person Bs - for (int j = 0; j < modelHhSize; ++j) - { - - // skip if same as person A - if (i == j) continue; - - Person personB = getCdapPerson(j + 1); - - // skip if i>j because if we have 1,2 for person 1, we don't - // also - // want 2,1; that's the - // same combination of two people - if (i > j) continue; - - // set the two person level dmu variables - cdapDmuObject.setPersonB(personB); - - // compute the two people utilities - double[] twoPersonUtilities = twoPeopleUec.solve(cdapDmuObject.getIndexValues(), - cdapDmuObject, availability); - - // log these utilities for trace households - if (householdObject.getDebugChoiceModels()) - { - - String stringToLog = String.format("%-30s%9d%9d%9s", "TwoPeople", (i + 1), - (j + 1), "--"); - - for (int k = 0; k < ACTIVITY_NAME_ARRAY.length; ++k) - { - stringToLog += String.format("%10.4f", twoPersonUtilities[k]); - } - cdapLogger.info(stringToLog); - - cdapUecLogger.info("PersonA:"); - personA.logEntirePersonObject(cdapUecLogger); - cdapUecLogger.info("PersonB:"); - personB.logEntirePersonObject(cdapUecLogger); - twoPeopleUec.logAnswersArray(cdapUecLogger, - "TWO PERSON, personA personNum=" + personA.getPersonNum() - + " personB personNum=" + personB.getPersonNum()); - - } // debug trace - - // align the two person utilities with the alternatives for - // person i - for (int k = 0; k < alternativeList.size(); ++k) - { - Alternative tempAlt = (Alternative) alternativeList.get(k); - String altName = tempAlt.getName(); - - // get the name of the activity for this person in the - // alternative string - String altNameForPersonA = altName.substring(i, i + 1); - String altNameForPersonB = altName.substring(j, j + 1); - - for (int l = 0; l < ACTIVITY_NAME_ARRAY.length; ++l) - { - if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[l]) - && altNameForPersonB.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[l])) - { - double currentUtility = tempAlt.getUtility(); - tempAlt.setUtility(currentUtility + twoPersonUtilities[l]); - } - } // l - } // k - - // loop through all possible person Cs - for (int k = 0; k < modelHhSize; ++k) - { - - // skip if same as person A - if (i == k) continue; - - // skip if same as person B - if (j == k) continue; - - // skip if j>k because if we have 1,2,3 for person 1, we - // don't - // also want 1,3,2; that's the - // same combination of three people - if (j > k) continue; - - Person personC = getCdapPerson(k + 1); - - // set the three level dmu variables - cdapDmuObject.setPersonC(personC); - - // compute the three person utilities - double[] threePersonUtilities = threePeopleUec.solve( - cdapDmuObject.getIndexValues(), cdapDmuObject, availability); - - // log these utilities for trace households - if (householdObject.getDebugChoiceModels()) - { - - String stringToLog = String.format("%-30s%9d%9d%9d", "ThreePeople", - (i + 1), (j + 1), (k + 1)); - - for (int l = 0; l < ACTIVITY_NAME_ARRAY.length; ++l) - { - stringToLog += String.format("%10.4f", threePersonUtilities[l]); - } - cdapLogger.info(stringToLog); - - cdapUecLogger.info("PersonA:"); - personA.logEntirePersonObject(cdapUecLogger); - cdapUecLogger.info("PersonB:"); - personB.logEntirePersonObject(cdapUecLogger); - cdapUecLogger.info("PersonC:"); - personC.logEntirePersonObject(cdapUecLogger); - threePeopleUec.logAnswersArray(cdapUecLogger, - "THREE PERSON, personA personNum=" + personA.getPersonNum() - + " personB personNum=" + personB.getPersonNum() - + " personC personNum=" + personC.getPersonNum()); - - } // debug trace - - // align the three person utilities with the alternatives - // for - // person i - for (int l = 0; l < alternativeList.size(); ++l) - { - Alternative tempAlt = (Alternative) alternativeList.get(l); - String altName = tempAlt.getName(); - - // get the name of the activity for this person in the - // alternative string - String altNameForPersonA = altName.substring(i, i + 1); - String altNameForPersonB = altName.substring(j, j + 1); - String altNameForPersonC = altName.substring(k, k + 1); - - for (int m = 0; m < ACTIVITY_NAME_ARRAY.length; ++m) - { - if (altNameForPersonA.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m]) - && altNameForPersonB.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m]) - && altNameForPersonC.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[m])) - { - double currentUtility = tempAlt.getUtility(); - tempAlt.setUtility(currentUtility + threePersonUtilities[m]); - } - } // m - } // l - - } // k (person C loop) - - } // j (person B loop) - - } // i (person A loop) - - // compute the interaction utilities - double[] allMemberInteractionUtilities = allMemberInteractionUec.solve( - cdapDmuObject.getIndexValues(), cdapDmuObject, availability); - - // log these utilities for trace households - if (householdObject.getDebugChoiceModels()) - { - - String stringToLog = String.format("%-30s%9s%9s%9s", "AllMembers", "--", "--", "--"); - - for (int i = 0; i < ACTIVITY_NAME_ARRAY.length; ++i) - { - stringToLog += String.format("%10.4f", allMemberInteractionUtilities[i]); - } - cdapLogger.info(stringToLog); - - allMemberInteractionUec.logAnswersArray(cdapUecLogger, "ALL MEMBER INTERACTIONS"); - - } // debug trace - - // align the utilities with the proper alternatives - for (int i = 0; i < alternativeList.size(); ++i) - { - Alternative tempAlt = (Alternative) alternativeList.get(i); - String altName = tempAlt.getName(); - - for (int j = 0; j < ACTIVITY_NAME_ARRAY.length; ++j) - { - - boolean samePattern = true; - for (int k = 0; k < modelHhSize; ++k) - { - - // alternative should have pattern j for each member k - String altNameForThisPerson = altName.substring(k, k + 1); - if (altNameForThisPerson.equalsIgnoreCase(ACTIVITY_NAME_ARRAY[j])) continue; - else - { - samePattern = false; - break; - } - } // k - - // if all have the same pattern, add the new utilities - if (samePattern) - { - double currentUtility = tempAlt.getUtility(); - tempAlt.setUtility(currentUtility + allMemberInteractionUtilities[j]); - } - } // j - - } // i - - // compute the joint utilities to be added to alternatives with joint - // tour - // indicator - - int adultsWithMand = 0; - int adultsWithNonMand = 0; - int kidsWithMand = 0; - int kidsWithNonMand = 0; - int adultsLeaveHome = 0; - int workMgra = 0; - double workLocationAccessibilityForWorkers = 0.0; - for (int i = 0; i < alternativeList.size(); ++i) - { - - Alternative tempAlt = (Alternative) alternativeList.get(i); - String altName = tempAlt.getName(); - - adultsWithMand = 0; - adultsWithNonMand = 0; - kidsWithMand = 0; - kidsWithNonMand = 0; - adultsLeaveHome = 0; - workMgra = 0; - workLocationAccessibilityForWorkers = 0.0; - for (int k = 0; k < modelHhSize; ++k) - { - - // alternative should have pattern j for each member k - String altNameForThisPerson = altName.substring(k, k + 1); - if (altNameForThisPerson.equalsIgnoreCase(MANDATORY_PATTERN)) - { - if (isThisCdapPersonAnAdult(k + 1)) - { - adultsWithMand++; - adultsLeaveHome++; - } else - { - kidsWithMand++; - } - - workMgra = getWorkLocationForThisCdapPerson(k + 1); - if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - Person tempPerson = getCdapPerson(k + 1); - workLocationAccessibilityForWorkers += tempPerson.getWorkLocationLogsum(); - } - - } else if (altNameForThisPerson.equalsIgnoreCase(NONMANDATORY_PATTERN)) - { - if (isThisCdapPersonAnAdult(k + 1)) - { - adultsWithNonMand++; - adultsLeaveHome++; - } else - { - kidsWithNonMand++; - } - } - - } // k - - cdapDmuObject.setNumAdultsWithNonMandatoryDap(adultsWithNonMand); - cdapDmuObject.setNumAdultsWithMandatoryDap(adultsWithMand); - cdapDmuObject.setNumKidsWithNonMandatoryDap(kidsWithNonMand); - cdapDmuObject.setNumKidsWithMandatoryDap(kidsWithMand); - cdapDmuObject.setAllAdultsAtHome(adultsLeaveHome == 0 ? 1 : 0); - cdapDmuObject.setWorkAccessForMandatoryDap(workLocationAccessibilityForWorkers); - - double[] jointUtilities = jointUec.solve(cdapDmuObject.getIndexValues(), cdapDmuObject, - availability); - - // log these utilities for trace households - if (householdObject.getDebugChoiceModels()) - { - String stringToLog = String.format("%-13s%4d %-12s%9s%9s%9s", "Joint", (i + 1), - altName, "--", "--", "--"); - stringToLog += String.format("%10.4f", - (altName.indexOf("j") > 0 ? jointUtilities[0] : 0.0)); - cdapLogger.info(stringToLog); - - jointUec.logAnswersArray(cdapUecLogger, "JOINT Utility for CDAP Alt index = " - + (i + 1) + ", Alt name = " + altName); - } // debug trace - - // align the utilities with the proper alternatives - if (altName.indexOf("j") > 0) - { - double currentUtility = tempAlt.getUtility(); - tempAlt.setUtility(currentUtility + jointUtilities[0]); - } - - } // i - - // TODO: check this out - use computeAvailabilty() checks that an - // alternative - // is available - // all utilities are set - compute probabilities - // workingLogitModel.setAvailability(true); - workingLogitModel.computeAvailabilities(); - - // compute the exponentiated utility, logging debug if need be - if (householdObject.getDebugChoiceModels()) - { - workingLogitModel.setDebug(true); - workingLogitModel.writeUtilityHeader(); - } - float logsum = (float) workingLogitModel.getUtility(); - householdObject.setCdapLogsum(logsum); - - // compute the probabilities, logging debug if need be - if (householdObject.getDebugChoiceModels()) - { - workingLogitModel.writeProbabilityHeader(); - } - workingLogitModel.calculateProbabilities(); - - // turn debug off for the next guy - workingLogitModel.setDebug(false); - - // make a choice for the first five - Random hhRandom = householdObject.getHhRandom(); - - double randomNumber = hhRandom.nextDouble(); - - if (householdObject.getDebugChoiceModels()) - { - cdapLogger.info("randomNumber = " + randomNumber); - } - - String firstFiveChosenName = ""; - try - { - ConcreteAlternative chosenAlternative = (ConcreteAlternative) workingLogitModel - .chooseElementalAlternative(randomNumber); - firstFiveChosenName = chosenAlternative.getName(); - - if (householdObject.getDebugChoiceModels()) - { - - int chosenIndex = 0; - // get the index number for the alternative chosen - for (int i = 0; i < alternativeList.size(); ++i) - { - Alternative tempAlt = (Alternative) alternativeList.get(i); - String altName = tempAlt.getName(); - if (altName.equalsIgnoreCase(firstFiveChosenName)) - { - chosenIndex = i + 1; - break; - } - } - - cdapLogger.info("chosen pattern (5 or fewer hh members): Alt index = " - + chosenIndex + ", Alt name = " + firstFiveChosenName); - cdapLogger.info(""); - cdapLogger.info(""); - } - } catch (ModelException e) - { - logger.error(String.format( - "Exception caught for HHID=%d, no available CDAP alternatives to choose.", - householdObject.getHhId()), e); - throw new RuntimeException(); - } - - // make a choice for additional hh members if need be - if (actualHhSize > MAX_MODEL_HH_SIZE) - { - - String allMembersChosenPattern = applyModelForExtraHhMembers(householdObject, - firstFiveChosenName); - - String extraChar = ""; - int extraCharIndex = allMembersChosenPattern.indexOf("0"); - if (extraCharIndex > 0) - { - extraChar = "0"; - } else - { - extraCharIndex = allMembersChosenPattern.indexOf("j"); - if (extraCharIndex > 0) extraChar = "j"; - } - - allMembersChosenPattern = allMembersChosenPattern.substring(0, extraCharIndex) - + allMembersChosenPattern.substring(extraCharIndex + 1); - - // re-order the activities by the original order of persons - String finalHhPattern = ""; - String[] finalHhPatternActivities = new String[cdapPersonArray.length]; - for (int i = 1; i < cdapPersonArray.length; i++) - { - int k = cdapPersonArray[i].getPersonNum(); - finalHhPatternActivities[k] = allMembersChosenPattern.substring(i - 1, i); - } - - for (int i = 1; i < cdapPersonArray.length; i++) - finalHhPattern += finalHhPatternActivities[i]; - - String finalString = finalHhPattern + extraChar; - if (householdObject.getDebugChoiceModels()) - { - cdapLogger.info("final pattern (more than 5 hh members) = " + finalString); - cdapLogger.info(""); - cdapLogger.info(""); - } - return finalString; - - } - - if (householdObject.getDebugChoiceModels()) - { - // reset the logger to what it was before we changed it - LogitModel.setLogger(Logger.getLogger(LogitModel.class)); - } - - // no need to re-order the activities - hhsize <= MAX_MODEL_HH_SIZE have - // original order of persons - return firstFiveChosenName; - - } - - /** - * Applies a simple choice from fixed proportions by person type for members - * of households with more than 5 people who are not included in the CDAP - * model. The choices of the additional household members are independent of - * each other. - * - * @param householdObject - * @param patternStringForOtherHhMembers - * @return the pattern for the entire household, including the 5-member - * pattern chosen by the logit model and the additional members - * chosen by the fixed-distribution model. - * - */ - private String applyModelForExtraHhMembers(Household householdObject, - String patternStringForOtherHhMembers) - { - - String allMembersPattern = patternStringForOtherHhMembers; - - // get the persons not yet modeled - Person[] personArray = getPersonsNotModeledByCdap(MAX_MODEL_HH_SIZE); - - // person array is 1-based to be consistent with other person arrays - for (int i = 1; i < personArray.length; i++) - { - - int personType = personArray[i].getPersonTypeNumber(); - - // get choice index from fixed proportions for 6 plus persons - int chosen = ChoiceModelApplication.getMonteCarloSelection( - fixedCumulativeProportions[personType], householdObject.getHhRandom() - .nextDouble()); - - allMembersPattern += ACTIVITY_NAME_ARRAY[chosen]; - - } - - return allMembersPattern; - - } - - /** - * Method reorders the persons in the household for use with the CDAP model, - * which only explicitly models the interaction of five persons in a HH. - * Priority in the reordering is first given to full time workers (up to - * two), then to part time workers (up to two workers, of any type), then to - * children (youngest to oldest, up to three). If the method is called for a - * household with less than 5 people, the cdapPersonArray is the same as the - * person array. - * - */ - public void reOrderPersonsForCdap(Household household) - { - - // set the person array - Person[] persons = household.getPersons(); - - // if hh is not too big, set cdap equal to persons and return - int hhSize = household.getSize(); - if (hhSize <= MAX_MODEL_HH_SIZE) - { - cdapPersonArray = persons; - return; - } - - // create the end game array - cdapPersonArray = new Person[persons.length]; - - // keep track of which persons you count - boolean[] iCountedYou = new boolean[persons.length]; - Arrays.fill(iCountedYou, false); - - // define the persons we want to find among the five - int firstWorkerIndex = -99; - int secondWorkerIndex = -99; - - int youngestChildIndex = -99; - int secondYoungestChildIndex = -99; - int thirdYoungestChildIndex = -99; - - int youngestChildAge = 99; - int secondYoungestChildAge = 99; - int thirdYoungestChildAge = 99; - - // first: full-time workers (persons is 1-based array) - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - // is the person a full-time worker - if (persons[i].getPersonIsFullTimeWorker() == 1) - { - - // check our indices - if (firstWorkerIndex == -99) - { - firstWorkerIndex = i; - iCountedYou[i] = true; - } else if (secondWorkerIndex == -99) - { - secondWorkerIndex = i; - iCountedYou[i] = true; - } - } - - } // i (full time workers) - - // second: part-time workers (only if we don't have two workers) - if (firstWorkerIndex == -99 || secondWorkerIndex == -99) - { - - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - // is the person part-time worker - if (persons[i].getPersonIsPartTimeWorker() == 1) - { - - // check our indices - if (firstWorkerIndex == -99) - { - firstWorkerIndex = i; - iCountedYou[i] = true; - } else if (secondWorkerIndex == -99) - { - secondWorkerIndex = i; - iCountedYou[i] = true; - } - - } - - } // i (part-time workers) - } - - // third: youngest child loop - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - if (persons[i].getPersonIsPreschoolChild() == 1 - || persons[i].getPersonIsStudentNonDriving() == 1 - || persons[i].getPersonIsStudentDriving() == 1) - { - - // check our indices - if (youngestChildIndex == -99) - { - youngestChildIndex = i; - youngestChildAge = persons[i].getAge(); - iCountedYou[i] = true; - } else - { - - // see if this child is younger than the one on record - int age = persons[i].getAge(); - if (age < youngestChildAge) - { - - // reset iCountedYou for previous child - iCountedYou[youngestChildIndex] = false; - - // set variables for this child - youngestChildIndex = i; - youngestChildAge = age; - iCountedYou[i] = true; - - } - } - - } // if person is child - - } // i (youngest child loop) - - // fourth: second youngest child loop (skip if youngest child is not - // filled) - if (youngestChildIndex != -99) - { - - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - if (persons[i].getPersonIsPreschoolChild() == 1 - || persons[i].getPersonIsStudentNonDriving() == 1 - || persons[i].getPersonIsStudentDriving() == 1) - { - - // check our indices - if (secondYoungestChildIndex == -99) - { - secondYoungestChildIndex = i; - secondYoungestChildAge = persons[i].getAge(); - iCountedYou[i] = true; - } else - { - - // see if this child is younger than the one on record - int age = persons[i].getAge(); - if (age < secondYoungestChildAge) - { - - // reset iCountedYou for previous child - iCountedYou[secondYoungestChildIndex] = false; - - // set variables for this child - secondYoungestChildIndex = i; - secondYoungestChildAge = age; - iCountedYou[i] = true; - - } - } - - } // if person is child - - } // i (second youngest child loop) - } - - // fifth: third youngest child loop (skip if second kid not included) - if (secondYoungestChildIndex != -99) - { - - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - if (persons[i].getPersonIsPreschoolChild() == 1 - || persons[i].getPersonIsStudentNonDriving() == 1 - || persons[i].getPersonIsStudentDriving() == 1) - { - - // check our indices - if (thirdYoungestChildIndex == -99) - { - thirdYoungestChildIndex = i; - thirdYoungestChildAge = persons[i].getAge(); - iCountedYou[i] = true; - } else - { - - // see if this child is younger than the one on record - int age = persons[i].getAge(); - if (age < thirdYoungestChildAge) - { - - // reset iCountedYou for previous child - iCountedYou[thirdYoungestChildIndex] = false; - - // set variables for this child - thirdYoungestChildIndex = i; - thirdYoungestChildAge = age; - iCountedYou[i] = true; - - } - } - - } // if person is child - - } // i (third youngest child loop) - } - - // assign any missing spots among the top 5 to random members - int cdapPersonIndex; - - Random hhRandom = household.getHhRandom(); - - int randomCount = household.getHhRandomCount(); - // when household.getHhRandom() was applied, the random count was - // incremented, assuming a random number would be drawn right away. - // so let's decrement by 1, then increment the count each time a random - // number is actually drawn in this method. - randomCount--; - - // first worker - cdapPersonIndex = 1; // persons and cdapPersonArray are 1-based - if (firstWorkerIndex == -99) - { - - int randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - while (iCountedYou[randomIndex] || randomIndex == 0) - { - randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - } - - cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; - iCountedYou[randomIndex] = true; - - } else - { - cdapPersonArray[cdapPersonIndex] = persons[firstWorkerIndex]; - } - - // second worker - cdapPersonIndex = 2; - if (secondWorkerIndex == -99) - { - - int randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - while (iCountedYou[randomIndex] || randomIndex == 0) - { - randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - } - - cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; - iCountedYou[randomIndex] = true; - - } else - { - cdapPersonArray[cdapPersonIndex] = persons[secondWorkerIndex]; - } - - // youngest child - cdapPersonIndex = 3; - if (youngestChildIndex == -99) - { - - int randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - while (iCountedYou[randomIndex] || randomIndex == 0) - { - randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - } - - cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; - iCountedYou[randomIndex] = true; - - } else - { - cdapPersonArray[cdapPersonIndex] = persons[youngestChildIndex]; - } - - // second youngest child - cdapPersonIndex = 4; - if (secondYoungestChildIndex == -99) - { - - int randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - while (iCountedYou[randomIndex] || randomIndex == 0) - { - randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - } - - cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; - iCountedYou[randomIndex] = true; - - } else - { - cdapPersonArray[cdapPersonIndex] = persons[secondYoungestChildIndex]; - } - - // third youngest child - cdapPersonIndex = 5; - if (thirdYoungestChildIndex == -99) - { - - int randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - while (iCountedYou[randomIndex] || randomIndex == 0) - { - randomIndex = (int) (hhRandom.nextDouble() * hhSize); - randomCount++; - } - - cdapPersonArray[cdapPersonIndex] = persons[randomIndex]; - iCountedYou[randomIndex] = true; - - } else - { - cdapPersonArray[cdapPersonIndex] = persons[thirdYoungestChildIndex]; - } - - // fill spots outside the top 5 - cdapPersonIndex = 6; - for (int i = 1; i < persons.length; ++i) - { - - if (iCountedYou[i]) continue; - - cdapPersonArray[cdapPersonIndex] = persons[i]; - cdapPersonIndex++; - } - - household.setHhRandomCount(randomCount); - - } - - protected Person getCdapPerson(int persNum) - { - if (persNum < 1 || persNum > cdapPersonArray.length - 1) - { - logger.fatal(String.format("persNum value = %d is out of range for hhSize = %d", - persNum, cdapPersonArray.length - 1)); - throw new RuntimeException(); - } - - return cdapPersonArray[persNum]; - } - - /** - * Method returns an array of persons not modeled by the CDAP model (i.e. - * persons 6 to X, when ordered by the reOrderPersonsForCdap method - * - * @param personsModeledByCdap - * @return - */ - public Person[] getPersonsNotModeledByCdap(int personsModeledByCdap) - { - - // create a 1-based person array to be consistent - Person[] personArray = null; - if (cdapPersonArray.length > personsModeledByCdap + 1) personArray = new Person[cdapPersonArray.length - - personsModeledByCdap]; - else personArray = new Person[0]; - - for (int i = 1; i < personArray.length; ++i) - { - personArray[i] = cdapPersonArray[personsModeledByCdap + i]; - } - - return personArray; - - } - - /** - * Returns true if this CDAP person number (meaning the number of persons - * for the purposes of the CDAP model) is an adult; false if not. - * - * @param cdapPersonNumber - * @return - */ - public boolean isThisCdapPersonAnAdult(int cdapPersonNumber) - { - - if (cdapPersonArray[cdapPersonNumber].getPersonIsAdult() == 1) return true; - return false; - } - - public int getStudentTypeForThisCdapPerson(int cdapPersonNumber) - { - - Person p = cdapPersonArray[cdapPersonNumber]; - int type = 0; - if (p.getPersonIsPreschoolChild() == 1) type = 1; - else if (p.getPersonIsGradeSchool() == 1) type = 2; - else if (p.getPersonIsHighSchool() == 1) type = 3; - else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) type = 4; - else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) type = 5; - - return type; - } - - public int getStudentTypeForThisCdapPerson(Person p) - { - - int type = 0; - if (p.getPersonIsPreschoolChild() == 1) type = 1; - else if (p.getPersonIsGradeSchool() == 1) type = 2; - else if (p.getPersonIsHighSchool() == 1) type = 3; - else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) type = 4; - else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) type = 5; - - return type; - } - - public int getWorkLocationForThisCdapPerson(int cdapPersonNumber) - { - int loc = cdapPersonArray[cdapPersonNumber].getWorkLocation(); - if (loc > 0 && loc != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return loc; - else return 0; - } - - public int getSchoolLocationForThisCdapPerson(int cdapPersonNumber) - { - return cdapPersonArray[cdapPersonNumber].getUsualSchoolLocation(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java deleted file mode 100644 index eb6ebe6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManager.java +++ /dev/null @@ -1,2039 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedReader; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileNotFoundException; -import java.io.FileReader; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Random; -import java.util.StringTokenizer; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import umontreal.iro.lecuyer.probdist.LognormalDist; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.IndexSort; -import com.pb.common.util.ObjectUtil; -import com.pb.common.util.SeededRandom; - -/** - * @author Jim Hicks - * - * Class for managing household and person object data read from - * synthetic population files. - */ -public abstract class HouseholdDataManager - implements HouseholdDataManagerIf, Serializable -{ - - protected transient Logger logger = Logger.getLogger(HouseholdDataManager.class); - - public static final String PROPERTIES_SYNPOP_INPUT_HH = "PopulationSynthesizer.InputToCTRAMP.HouseholdFile"; - public static final String PROPERTIES_SYNPOP_INPUT_PERS = "PopulationSynthesizer.InputToCTRAMP.PersonFile"; - - public static final String RANDOM_SEED_NAME = "Model.Random.Seed"; - - public static final String OUTPUT_HH_DATA_FILE_TARGET = "outputHouseholdData.file"; - public static final String OUTPUT_PERSON_DATA_FILE_TARGET = "outputPersonData.file"; - - public static final String READ_UWSL_RESULTS_FILE = "read.uwsl.results"; - public static final String READ_UWSL_RESULTS_FILENAME = "read.uwsl.filename"; - public static final String READ_PRE_AO_RESULTS_FILE = "read.pre.ao.results"; - public static final String READ_PRE_AO_RESULTS_FILENAME = "read.pre.ao.filename"; - - public static final String PROPERTIES_DISTRIBUTED_TIME = "distributedTimeCoefficients"; - - - // HHID,household_serial_no,TAZ,MGRA,VEH,PERSONS,HWORKERS,HINCCAT1,HINC,UNITTYPE,HHT,BLDGSZ - public static final String HH_ID_FIELD_NAME = "HHID"; - public static final String HH_HOME_TAZ_FIELD_NAME = "TAZ"; - public static final String HH_HOME_MGRA_FIELD_NAME = "MGRA"; - public static final String HH_INCOME_CATEGORY_FIELD_NAME = "HINCCAT1"; - public static final String HH_INCOME_DOLLARS_FIELD_NAME = "HINC"; - public static final String HH_WORKERS_FIELD_NAME = "HWORKERS"; - public static final String HH_AUTOS_FIELD_NAME = "VEH"; - public static final String HH_SIZE_FIELD_NAME = "PERSONS"; - public static final String HH_TYPE_FIELD_NAME = "HHT"; - public static final String HH_BLDGSZ_FIELD_NAME = "BLDGSZ"; - public static final String HH_UNITTYPE_FIELD_NAME = "UNITTYPE"; - - // HHID,PERID,AGE,SEX,OCCCEN1,INDCEN,PEMPLOY,PSTUDENT,PTYPE,EDUC,GRADE - public static final String PERSON_HH_ID_FIELD_NAME = "HHID"; - public static final String PERSON_PERSON_ID_FIELD_NAME = "PERID"; - public static final String PERSON_AGE_FIELD_NAME = "AGE"; - public static final String PERSON_GENDER_FIELD_NAME = "SEX"; - public static final String PERSON_MILITARY_FIELD_NAME = "MILTARY"; - public static final String PERSON_EMPLOYMENT_CATEGORY_FIELD_NAME = "PEMPLOY"; - public static final String PERSON_STUDENT_CATEGORY_FIELD_NAME = "PSTUDENT"; - public static final String PERSON_TYPE_CATEGORY_FIELD_NAME = "PTYPE"; - public static final String PERSON_EDUCATION_ATTAINMENT_FIELD_NAME = "EDUC"; - public static final String PERSON_GRADE_ENROLLED_FIELD_NAME = "GRADE"; - public static final String PERSON_OCCCEN1_FIELD_NAME = "OCCCEN1"; - public static final String PERSON_SOC_FIELD_NAME = "OCCSOC5"; - public static final String PERSON_INDCEN_FIELD_NAME = "INDCEN"; - - public static final String PERSON_TIMEFACTOR_WORK_FIELD_NAME = "timeFactorWork"; - public static final String PERSON_TIMEFACTOR_NONWORK_FIELD_NAME = "timeFactorNonWork"; - - public static final String PROPERTIES_HOUSEHOLD_TRACE_LIST = "Debug.Trace.HouseholdIdList"; - public static final String DEBUG_HHS_ONLY_KEY = "Process.Debug.HHs.Only"; - - private static final String PROPERTIES_MIN_VALUE_OF_TIME_KEY = "HouseholdManager.MinValueOfTime"; - private static final String PROPERTIES_MAX_VALUE_OF_TIME_KEY = "HouseholdManager.MaxValueOfTime"; - private static final String PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY = "HouseholdManager.MeanValueOfTime.Values"; - private static final String PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY = "HouseholdManager.MeanValueOfTime.Income.Limits"; - private static final String PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY = "HouseholdManager.HH.ValueOfTime.Multiplier.Under18"; - private static final String PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY = "HouseholdManager.Mean.ValueOfTime.Multiplier.Mu"; - private static final String PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY = "HouseholdManager.ValueOfTime.Lognormal.Sigma"; - - private HashMap schoolSegmentNameIndexMap; - private HashMap gsDistrictSegmentMap; - private HashMap hsDistrictSegmentMap; - private int[] mgraGsDistrict; - private int[] mgraHsDistrict; - - // these are not used for sandag; instead sandag uses distributed time coefficients read in the person file - protected float hhValueOfTimeMultiplierForPersonUnder18; - protected double meanValueOfTimeMultiplierBeforeLogForMu; - protected double valueOfTimeLognormalSigma; - protected float minValueOfTime; - protected float maxValueOfTime; - protected float[] meanValueOfTime; - protected int[] incomeDollarLimitsForValueOfTime; - protected LognormalDist[] valueOfTimeDistribution; - - protected HashMap propertyMap; - - protected String projectDirectory; - protected String outputHouseholdFileName; - protected String outputPersonFileName; - - protected ModelStructure modelStructure; - - protected TableDataSet hhTable; - protected TableDataSet personTable; - - protected HashSet householdTraceSet; - - protected Household[] hhs; - protected int[] hhIndexArray; - - protected int inputRandomSeed; - protected int numPeriods; - protected int firstPeriod; - - protected float sampleRate; - protected int sampleSeed; - - protected int maximumNumberOfHouseholdsPerFile = 0; - protected int numberOfHouseholdDiskObjectFiles = 0; - - protected MgraDataManager mgraManager; - protected TazDataManager tazManager; - - protected double[] percentHhsIncome100Kplus; - protected double[] percentHhsMultipleAutos; - - protected boolean readTimeFactors; - public HouseholdDataManager() - { - } - - /** - * Associate data in hh and person TableDataSets read from synthetic - * population files with Household objects and Person objects with - * Households. - */ - protected abstract void mapTablesToHouseholdObjects(); - - public String testRemote() - { - System.out.println("testRemote() called by remote process."); - return String.format("testRemote() method in %s called.", this.getClass() - .getCanonicalName()); - } - - public void setDebugHhIdsFromHashmap() - { - - householdTraceSet = new HashSet(); - - // get the household ids for which debug info is required - String householdTraceStringList = propertyMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); - - if (householdTraceStringList != null) - { - StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); - while (householdTokenizer.hasMoreTokens()) - { - String listValue = householdTokenizer.nextToken(); - int idValue = Integer.parseInt(listValue.trim()); - householdTraceSet.add(idValue); - } - } - - } - - public void readPopulationFiles(String inputHouseholdFileName, String inputPersonFileName) - { - - TimeCoefficientDistributions timeDistributions = new TimeCoefficientDistributions(); - timeDistributions.createTimeDistributions(propertyMap); - timeDistributions.appendTimeDistributionsOnPersonFile(propertyMap); - - // read synthetic population files - readHouseholdData(inputHouseholdFileName); - - readPersonData(inputPersonFileName); - } - - public void setModelStructure(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - } - - public void setupHouseholdDataManager(ModelStructure modelStructure, - String inputHouseholdFileName, String inputPersonFileName) - { - - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - setModelStructure(modelStructure); - readPopulationFiles(inputHouseholdFileName, inputPersonFileName); - - // Set the seed for the JVM default SeededRandom object - should only be - // used - // to set the order for the - // HH index array so that hhs can be processed in an arbitrary order as - // opposed to the order imposed by - // the synthetic population generator. - // The seed was set as a command line argument for the model run, or the - // default if no argument supplied - SeededRandom.setSeed(sampleSeed); - - // the seed read from the properties file controls seeding the Household - // object random number generator objects. - inputRandomSeed = Integer.parseInt(propertyMap.get(HouseholdDataManager.RANDOM_SEED_NAME)); - - // map synthetic population table data to objects to be used by CT-RAMP - mapTablesToHouseholdObjects(); - hhTable = null; - personTable = null; - - logPersonSummary(); - - setTraceHouseholdSet(); - - // if read pre-ao results flag is set, read the results file and - // populate the - // household object ao result field from these values. - String readPreAoResultsString = propertyMap.get(READ_PRE_AO_RESULTS_FILE); - if (readPreAoResultsString != null) - { - boolean readResults = Boolean.valueOf(readPreAoResultsString); - if (readResults) readPreAoResults(); - } - - // if read uwsl results flag is set, read the results file and populate - // the - // household object work/school location result fields from these - // values. - String readUwslResultsString = propertyMap.get(READ_UWSL_RESULTS_FILE); - if (readUwslResultsString != null) - { - boolean readResults = Boolean.valueOf(readUwslResultsString); - if (readResults) readWorkSchoolLocationResults(); - } - - //check if we want to read distributed time factors from the person file - String readTimeFactorsString = propertyMap.get(PROPERTIES_DISTRIBUTED_TIME); - if (readTimeFactorsString != null) - { - readTimeFactors = Boolean.valueOf(readTimeFactorsString); - logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); - } - - - - } - - public void setPropertyFileValues(HashMap propertyMap) - { - - String propertyValue = ""; - this.propertyMap = propertyMap; - - // save the project specific parameters in class attributes - this.projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - - outputHouseholdFileName = propertyMap - .get(CtrampApplication.PROPERTIES_OUTPUT_HOUSEHOLD_FILE); - outputPersonFileName = propertyMap.get(CtrampApplication.PROPERTIES_OUTPUT_PERSON_FILE); - - setDebugHhIdsFromHashmap(); - - propertyValue = propertyMap - .get(CtrampApplication.PROPERTIES_SCHEDULING_NUMBER_OF_TIME_PERIODS); - if (propertyValue == null) numPeriods = 0; - else numPeriods = Integer.parseInt(propertyValue); - - propertyValue = propertyMap.get(CtrampApplication.PROPERTIES_SCHEDULING_FIRST_TIME_PERIOD); - if (propertyValue == null) firstPeriod = 0; - else firstPeriod = Integer.parseInt(propertyValue); - - //check if we want to read distributed time factors from the person file - String readTimeFactorsString = propertyMap.get(PROPERTIES_DISTRIBUTED_TIME); - if (readTimeFactorsString != null) - { - readTimeFactors = Boolean.valueOf(readTimeFactorsString); - logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); - } - - } - - public int[] getRandomOrderHhIndexArray(int numHhs) - { - - Random myRandom = new Random(); - myRandom.setSeed(numHhs + 1); - - int[] data = new int[numHhs]; - for (int i = 0; i < numHhs; i++) - data[i] = (int) (10000000 * myRandom.nextDouble()); - - int[] index = IndexSort.indexSort(data); - - return index; - } - - // this is called at the end of UsualWorkSchoolLocation model step. - public void setUwslRandomCount(int iter) - { - - for (int r = 0; r < hhs.length; r++) - hhs[r].setUwslRandomCount(iter, hhs[r].getHhRandomCount()); - - } - - private void resetRandom(Household h, int count) - { - // get the household's Random - Random r = h.getHhRandom(); - - int seed = inputRandomSeed + h.getHhId(); - r.setSeed(seed); - - // select count Random draws to reset this household's Random to it's - // state - // prior to - // the model run for which model results were stored in - // HouseholdDataManager. - for (int i = 0; i < count; i++) - r.nextDouble(); - - // reset the randomCount for the household's Random - h.setHhRandomCount(count); - } - - public void resetPreAoRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // The Pre Auto Ownership model is the first model component, so - // reset - // counts to 0. - resetRandom(hhs[i], 0); - } - } - - public void resetUwslRandom(int iter) - { - for (int i = 0; i < hhs.length; i++) - { - // get the current random count for the end of the shadow price - // iteration - // passed in. - // this value was set at the end of UsualWorkSchoolLocation model - // step - // for the given iter. - // if < 0, random count should be set to the count at end of pre - // auto - // ownership. - int uwslCount = hhs[i].getPreAoRandomCount(); - if (iter >= 0) - { - uwslCount = hhs[i].getUwslRandomCount(iter); - } - - // draw uwslCount random numbers from the household's Random - resetRandom(hhs[i], uwslCount); - } - } - - public void resetAoRandom(int iter) - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Auto Ownership model from the - // Household - // object. - // this value was set at the end of UsualWorkSchoolLocation model - // step. - - int aoCount = hhs[i].getUwslRandomCount(iter); - - // draw aoCount random numbers from the household's Random - resetRandom(hhs[i], aoCount); - } - } - - public void resetTpRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Auto Ownership model from the - // Household - // object. - // this value was set at the end of UsualWorkSchoolLocation model - // step. - int tpCount = hhs[i].getAoRandomCount(); - - // draw aoCount random numbers from the household's Random - resetRandom(hhs[i], tpCount); - } - } - - public void resetFpRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Auto Ownership model from the - // Household - // object. - // this value was set at the end of UsualWorkSchoolLocation model - // step. - int fpCount = hhs[i].getTpRandomCount(); - - // draw aoCount random numbers from the household's Random - resetRandom(hhs[i], fpCount); - } - } - - public void resetIeRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Auto Ownership model from the - // Household - // object. - // this value was set at the end of UsualWorkSchoolLocation model - // step. - int ieCount = hhs[i].getFpRandomCount(); - - // draw aoCount random numbers from the household's Random - resetRandom(hhs[i], ieCount); - } - } - - public void resetCdapRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Coordinated Daily Activity Pattern - // model from the Household object. - // this value was set at the end of Auto Ownership model step. - int cdapCount = hhs[i].getIeRandomCount(); - - // draw cdapCount random numbers - resetRandom(hhs[i], cdapCount); - } - } - - public void resetImtfRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Individual Mandatory Tour - // Frequency - // model from the Household object. - // this value was set at the end of Coordinated Daily Activity - // Pattern - // model step. - int imtfCount = hhs[i].getCdapRandomCount(); - - // draw imtfCount random numbers - resetRandom(hhs[i], imtfCount); - } - } - - public void resetImtodRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Individual Mandatory Tour - // Departure and - // duration model from the Household object. - // this value was set at the end of Individual Mandatory Tour - // Frequency - // model step. - int imtodCount = hhs[i].getImtfRandomCount(); - - // draw imtodCount random numbers - resetRandom(hhs[i], imtodCount); - } - } - - public void resetJtfRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Joint Tour Frequency model from - // the - // Household object. - // this value was set at the end of Individual Mandatory departure - // time - // Choice model step. - int jtfCount = hhs[i].getImtodRandomCount(); - - // draw jtfCount random numbers - resetRandom(hhs[i], jtfCount); - } - } - - public void resetJtlRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Joint Tour Location model from the - // Household object. - // this value was set at the end of Joint Tour Frequency model step. - int jtlCount = hhs[i].getJtfRandomCount(); - - // draw jtlCount random numbers - resetRandom(hhs[i], jtlCount); - } - } - - public void resetJtodRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Joint Tour Departure and duration - // model - // from the Household object. - // this value was set at the end of Joint Tour Location model step. - int jtodCount = hhs[i].getJtlRandomCount(); - - // draw jtodCount random numbers - resetRandom(hhs[i], jtodCount); - } - } - - public void resetInmtfRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Individual non-mandatory tour - // frequency - // model from the Household object. - // this value was set at the end of Joint Tour Departure and - // duration - // model step. - int inmtfCount = hhs[i].getJtodRandomCount(); - - // draw inmtfCount random numbers - resetRandom(hhs[i], inmtfCount); - } - } - - public void resetInmtlRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Individual non-mandatory tour - // location - // model from the Household object. - // this value was set at the end of Individual non-mandatory tour - // frequency model step. - int inmtlCount = hhs[i].getInmtfRandomCount(); - - // draw inmtlCount random numbers - resetRandom(hhs[i], inmtlCount); - } - } - - public void resetInmtodRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to Individual non-mandatory tour - // departure - // and duration model from the Household object. - // this value was set at the end of Individual non-mandatory tour - // location model step. - int inmtodCount = hhs[i].getInmtlRandomCount(); - - // draw inmtodCount random numbers - resetRandom(hhs[i], inmtodCount); - } - } - - public void resetAwfRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to At-work Subtour Frequency model - // from - // the Household object. - // this value was set at the end of Individual Non-Mandatory Tour - // Departure and duration model step. - int awfCount = hhs[i].getInmtodRandomCount(); - - // draw awfCount random numbers - resetRandom(hhs[i], awfCount); - } - } - - public void resetAwlRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to At-work Subtour Location Choice - // model - // from the Household object. - // this value was set at the end of At-work Subtour Frequency model - // step. - int awlCount = hhs[i].getAwfRandomCount(); - - // draw awlCount random numbers - resetRandom(hhs[i], awlCount); - } - } - - public void resetAwtodRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to At-work Subtour Time-of-day and - // mode - // choice model from the Household object. - // this value was set at the end of At-work Subtour Location Choice - // model - // step. - int awtodCount = hhs[i].getAwlRandomCount(); - - // draw awtodCount random numbers - resetRandom(hhs[i], awtodCount); - } - } - - public void resetStfRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to stop frequency model from the - // Household - // object. - // this value was set at the end of At-work Subtour Time-of-day and - // mode - // choice model step. - int stfCount = hhs[i].getAwtodRandomCount(); - - // draw stfCount random numbers - resetRandom(hhs[i], stfCount); - } - } - - public void resetStlRandom() - { - for (int i = 0; i < hhs.length; i++) - { - // get the current count prior to stop location model from the - // Household - // object. - // this value was set at the end of stop frequency model step. - int stlCount = hhs[i].getStfRandomCount(); - - // draw stlCount random numbers - resetRandom(hhs[i], stlCount); - } - } - - /** - * Sets the HashSet used to trace households for debug purposes and sets the - * debug switch for each of the listed households. Also sets - */ - public void setTraceHouseholdSet() - { - - // loop through the households in the set and set the trace switches - for (int i = 0; i < hhs.length; i++) - hhs[i].setDebugChoiceModels(false); - - for (int id : householdTraceSet) - { - int index = hhIndexArray[id]; - hhs[index].setDebugChoiceModels(true); - } - - } - - /** - * Sets the sample rate used to run the model for a portion of the - * households. - * - * @param sampleRate - * , proportion of total households for which to run the model - * [0.0, 1.0]. - */ - public void setHouseholdSampleRate(float sampleRate, int sampleSeed) - { - this.sampleRate = sampleRate; - this.sampleSeed = sampleSeed; - } - - public void setHhArray(Household[] hhArray) - { - hhs = hhArray; - } - - public void setHhArray(Household[] tempHhs, int startIndex) - { - // long startTime = System.currentTimeMillis(); - // logger.info(String.format("start setHhArray for startIndex=%d, startTime=%d.", - // startIndex, - // startTime)); - - synchronized(hhs) { - for (int i = 0; i < tempHhs.length; i++) - { - hhs[startIndex + i] = tempHhs[i]; - } - } - // long endTime = System.currentTimeMillis(); - // logger.info(String.format( - // "end setHhArray for startIndex=%d, endTime=%d, elapsed=%d millisecs.", - // startIndex, - // endTime, (endTime - startTime))); - } - - /** - * return the array of Household objects holding the synthetic population - * and choice model outcomes. - * - * @return hhs - */ - public Household[] getHhArray() - { - return hhs; - } - - public Household[] getHhArray(int first, int last) - { - // long startTime = System.currentTimeMillis(); - // logger.info(String.format("start getHhArray for first=%d, last=%d, startTime=%d.", - // first, last, startTime)); - Household[] tempHhs = new Household[last - first + 1]; - for (int i = 0; i < tempHhs.length; i++) - { - tempHhs[i] = hhs[first + i]; - } - // long endTime = System.currentTimeMillis(); - // logger.info(String.format( - // "end getHhArray for first=%d, last=%d, endTime=%d, elapsed=%d millisecs.", - // first, last, endTime, (endTime - startTime))); - return tempHhs; - } - - public int getArrayIndex(int hhId) - { - int i = hhIndexArray[hhId]; - return i; - } - - /** - * return the number of household objects read from the synthetic - * population. - * - * @return - */ - public int getNumHouseholds() - { - // hhs is dimesioned to number of households + 1. - return hhs.length; - } - - /** - * set walk segment (0-none, 1-short, 2-long walk to transit access) for the - * origin for this tour - */ - public int getInitialOriginWalkSegment(int taz, double randomNumber) - { - // double[] proportions = tazDataManager.getZonalWalkPercentagesForTaz( - // taz - // ); - // return ChoiceModelApplication.getMonteCarloSelection(proportions, - // randomNumber); - return 0; - } - - private void readHouseholdData(String inputHouseholdFileName) - { - - // construct input household file name from properties file values - String fileName = projectDirectory + "/" + inputHouseholdFileName; - - try - { - logger.info("reading popsyn household data file."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - hhTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading synthetic household data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - } - - private void readPersonData(String inputPersonFileName) - { - - // construct input person file name from properties file values - String fileName = projectDirectory + "/" + inputPersonFileName; - - try - { - logger.info("reading popsyn person data file."); - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet("," + reader.getDelimSet()); - personTable = reader.readFile(new File(fileName)); - } catch (Exception e) - { - logger.fatal(String - .format("Exception occurred reading synthetic person data file: %s into TableDataSet object.", - fileName)); - throw new RuntimeException(e); - } - - } - - public void logPersonSummary() - { - - HashMap> summaryResults; - - summaryResults = new HashMap>(); - - for (int i = 0; i < hhs.length; ++i) - { - - Household household = hhs[i]; - - Person[] personArray = household.getPersons(); - for (int j = 1; j < personArray.length; ++j) - { - Person person = personArray[j]; - String personType = person.getPersonType(); - - String employmentStatus = person.getPersonEmploymentCategory(); - String studentStatus = person.getPersonStudentCategory(); - int age = person.getAge(); - int ageCategory; - if (age <= 5) - { - ageCategory = 0; - } else if (age <= 15) - { - ageCategory = 1; - } else if (age <= 18) - { - ageCategory = 2; - } else if (age <= 24) - { - ageCategory = 3; - } else if (age <= 44) - { - ageCategory = 4; - } else if (age <= 64) - { - ageCategory = 5; - } else - { - ageCategory = 6; - } - - if (summaryResults.containsKey(personType)) - { - // have person type - if (summaryResults.get(personType).containsKey(employmentStatus)) - { - // have employment category - summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; - } else - { - // don't have employment category - summaryResults.get(personType).put(employmentStatus, new int[7]); - summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; - } - if (summaryResults.get(personType).containsKey(studentStatus)) - { - // have student category - summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; - } else - { - // don't have student category - summaryResults.get(personType).put(studentStatus, new int[7]); - summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; - } - } else - { - // don't have person type - summaryResults.put(personType, new HashMap()); - summaryResults.get(personType).put(studentStatus, new int[7]); - summaryResults.get(personType).get(studentStatus)[ageCategory] += 1; - summaryResults.get(personType).put(employmentStatus, new int[7]); - summaryResults.get(personType).get(employmentStatus)[ageCategory] += 1; - } - } - } - String headerRow = String.format("%5s\t", "Age\t"); - for (String empCategory : Person.EMPLOYMENT_CATEGORY_NAME_ARRAY) - { - headerRow += String.format("%16s\t", empCategory); - } - for (String stuCategory : Person.STUDENT_CATEGORY_NAME_ARRAY) - { - headerRow += String.format("%16s\t", stuCategory); - } - String[] ageCategories = {"0-5", "6-15", "16-18", "19-24", "25-44", "45-64", "65+"}; - - for (String personType : summaryResults.keySet()) - { - - logger.info("Summary for person type: " + personType); - - logger.info(headerRow); - String row = ""; - - HashMap personTypeSummary = summaryResults.get(personType); - - for (int j = 0; j < ageCategories.length; ++j) - { - row = String.format("%5s\t", ageCategories[j]); - for (String empCategory : Person.EMPLOYMENT_CATEGORY_NAME_ARRAY) - { - if (personTypeSummary.containsKey(empCategory)) - { - row += String.format("%16d\t", personTypeSummary.get(empCategory)[j]); - } else row += String.format("%16d\t", 0); - } - for (String stuCategory : Person.STUDENT_CATEGORY_NAME_ARRAY) - { - if (personTypeSummary.containsKey(stuCategory)) - { - row += String.format("%16d\t", personTypeSummary.get(stuCategory)[j]); - } else row += String.format("%16d\t", 0); - } - logger.info(row); - } - - } - - } - - public int[][] getTourPurposePersonsByHomeMgra(String[] purposeList) - { - - int maxMgra = mgraManager.getMaxMgra(); - int[][] personsWithMandatoryPurpose = new int[purposeList.length][maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < hhs.length; r++) - { - - Person[] persons = hhs[r].getPersons(); - - int homeMgra = hhs[r].getHhMgra(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - int purposeIndex = -1; - try - { - - if (person.getPersonIsWorker() == 1) - { - - purposeIndex = person.getWorkLocationSegmentIndex(); - personsWithMandatoryPurpose[purposeIndex][homeMgra]++; - - } - - if (person.getPersonIsPreschoolChild() == 1 - || person.getPersonIsStudentDriving() == 1 - || person.getPersonIsStudentNonDriving() == 1 - || person.getPersonIsUniversityStudent() == 1) - { - - purposeIndex = person.getSchoolLocationSegmentIndex(); - personsWithMandatoryPurpose[purposeIndex][homeMgra]++; - - } - - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught summing workers/students by origin zone for household table record r=%d.", - r)); - throw e; - } - - } - - } // r (households) - - return personsWithMandatoryPurpose; - - } - - public double[] getPercentHhsIncome100Kplus() - { - return percentHhsIncome100Kplus; - } - - public double[] getPercentHhsMultipleAutos() - { - return percentHhsMultipleAutos; - } - - public void computeTransponderChoiceTazPercentArrays() - { - - PrintWriter out = null; - try - { - out = new PrintWriter(new BufferedWriter(new FileWriter(new File( - "./transpChoiceArrays.csv")))); - } catch (IOException e) - { - // TODO Auto-generated catch block - e.printStackTrace(); - } - - int maxTaz = tazManager.maxTaz; - int[] numHhs = new int[maxTaz + 1]; - - // for percent of households in TAZ with income > $100K - percentHhsIncome100Kplus = new double[maxTaz + 1]; - // for percent og households in TAZ with multiple autos - percentHhsMultipleAutos = new double[maxTaz + 1]; - - for (int r = 0; r < getNumHouseholds(); r++) - { - int homeMgra = hhs[r].getHhMgra(); - int homeTaz = mgraManager.getTaz(homeMgra); - numHhs[homeTaz]++; - if (hhs[r].getIncomeInDollars() > 100000) percentHhsIncome100Kplus[homeTaz]++; - if (hhs[r].getAutosOwned() > 1) percentHhsMultipleAutos[homeTaz]++; - } - - out.println("taz,numHhsTaz,numHhsIncome100KplusTaz,numHhsMultipleAutosTaz,proportionHhsIncome100KplusTaz,proportionHhsMultipleAutosTaz"); - - for (int i = 0; i <= maxTaz; i++) - { - - out.print(i + "," + numHhs[i] + "," + percentHhsIncome100Kplus[i] + "," - + percentHhsMultipleAutos[i]); - if (numHhs[i] > 0) - { - percentHhsIncome100Kplus[i] /= numHhs[i]; - percentHhsMultipleAutos[i] /= numHhs[i]; - out.println("," + percentHhsIncome100Kplus[i] + "," + percentHhsMultipleAutos[i]); - } else - { - out.println("," + 0.0 + "," + 0.0); - } - } - - out.close(); - } - - public int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap) - { - - int maxMgra = mgraManager.getMaxMgra(); - - int[][] workersByHomeMgra = new int[segmentValueIndexMap.size()][maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < getNumHouseholds(); r++) - { - - Person[] persons = hhs[r].getPersons(); - - int homeMgra = hhs[r].getHhMgra(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - if (person.getPersonIsFullTimeWorker() == 1 - || person.getPersonIsPartTimeWorker() == 1) - { - - int occup = -1; - int segmentIndex = -1; - try - { - - occup = person.getPersPecasOccup(); - segmentIndex = segmentValueIndexMap.get(occup); - workersByHomeMgra[segmentIndex][homeMgra]++; - - } catch (Exception e) - { - logger.error( - String.format( - "exception caught summing workers by origin MGRA for household table record r=%d, person=%d, homeMgra=%d, occup=%d, segmentIndex=%d.", - r, person.getPersonNum(), homeMgra, occup, segmentIndex), e); - throw new RuntimeException(); - } - - } - - } - - } // r (households) - - return workersByHomeMgra; - - } - - public int[][] getStudentsByHomeMgra() - { - - int maxMgra = mgraManager.getMaxMgra(); - - // there are 5 school types - preschool, K-8, HS, University with - // typical - // students, University with non-typical students. - int[][] studentsByHomeMgra = new int[schoolSegmentNameIndexMap.size()][maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < getNumHouseholds(); r++) - { - - Person[] persons = hhs[r].getPersons(); - - int homeMgra = hhs[r].getHhMgra(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - if (person.getPersonIsPreschoolChild() == 1 - || person.getPersonIsStudentNonDriving() == 1 - || person.getPersonIsStudentDriving() == 1 - || person.getPersonIsUniversityStudent() == 1) - { - - int segmentIndex = -1; - try - { - - if (person.getPersonIsPreschoolChild() == 1) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); - } else if (person.getPersonIsGradeSchool() == 1) - { - int gsDistrict = mgraGsDistrict[homeMgra]; - segmentIndex = gsDistrictSegmentMap.get(gsDistrict); - } else if (person.getPersonIsHighSchool() == 1) - { - int hsDistrict = mgraHsDistrict[homeMgra]; - segmentIndex = hsDistrictSegmentMap.get(hsDistrict); - } else if (person.getPersonIsUniversityStudent() == 1 - && person.getAge() < 30) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); - } else if (person.getPersonIsUniversityStudent() == 1 - && person.getAge() >= 30) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); - } - - // if person type is a student but segment index is -1, - // the person is not enrolled; assume home schooled and - // don't add to sum by home mgra - if (segmentIndex >= 0) studentsByHomeMgra[segmentIndex][homeMgra]++; - - } catch (Exception e) - { - logger.error( - String.format( - "exception caught summing students by origin MGRA for household table record r=%d, person=%d, homeMgra=%d, segmentIndex=%d.", - r, person.getPersonNum(), homeMgra, segmentIndex), e); - throw new RuntimeException(); - } - - } - - } - - } // r (households) - - return studentsByHomeMgra; - - } - - public int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString) - { - - // dimension the array - int maxMgra = mgraManager.getMaxMgra(); - int[] individualNonMandatoryTours = new int[maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - int count = 0; - for (int r = 0; r < hhs.length; r++) - { - - Person[] persons = hhs[r].getPersons(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - ArrayList it = person.getListOfIndividualNonMandatoryTours(); - - try - { - - if (it.size() == 0) continue; - - for (Tour tour : it) - { - // increment the segment count if it's the right purpose - String tourPurpose = tour.getTourPurpose(); - if (purposeString.startsWith(tourPurpose)) - { - int homeMgra = hhs[r].getHhMgra(); - individualNonMandatoryTours[homeMgra]++; - count++; - } - } - - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught counting number of individualNonMandatory tours for purpose: %s, for household table record r=%d, personNum=%d.", - purposeString, r, person.getPersonNum())); - throw e; - } - - } - - } // r (households) - - return individualNonMandatoryTours; - } - - public int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap) - { - - // dimension the array - int maxMgra = mgraManager.getMaxMgra(); - int destMgra = 0; - - int[][] workTours = new int[segmentValueIndexMap.size()][maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < getNumHouseholds(); r++) - { - - Person[] persons = hhs[r].getPersons(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - int occup = -1; - int segmentIndex = -1; - try - { - - if (person.getPersonIsWorker() == 1) - { - - destMgra = person.getWorkLocation(); - - if (destMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - occup = person.getPersPecasOccup(); - segmentIndex = segmentValueIndexMap.get(occup); - workTours[segmentIndex][destMgra]++; - } - - } - - } catch (Exception e) - { - logger.error( - String.format( - "exception caught summing workers by work location MGRA for household table record r=%d, person=%d, workMgra=%d, occup=%d, segmentIndex=%d.", - r, person.getPersonNum(), destMgra, occup, segmentIndex), e); - throw new RuntimeException(); - } - - } - - } // r (households) - - return workTours; - - } - - public int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap) - { - - int destMgra = 0; - - int[] workAtHome = new int[segmentValueIndexMap.size()]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < getNumHouseholds(); r++) - { - - Person[] persons = hhs[r].getPersons(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - int occup = -1; - int segmentIndex = -1; - try - { - - if (person.getPersonIsWorker() == 1) - { - - destMgra = person.getWorkLocation(); - - if (destMgra == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - occup = person.getPersPecasOccup(); - segmentIndex = segmentValueIndexMap.get(occup); - workAtHome[segmentIndex]++; - } - - } - - } catch (Exception e) - { - logger.error( - String.format( - "exception caught summing workers by work location MGRA for household table record r=%d, person=%d, workMgra=%d, occup=%d, segmentIndex=%d.", - r, person.getPersonNum(), destMgra, occup, segmentIndex), e); - throw new RuntimeException(); - } - - } - - } // r (households) - - return workAtHome; - - } - - public void setSchoolDistrictMappings(HashMap segmentNameIndexMap, - int[] mgraGsDist, int[] mgraHsDist, HashMap gsDistSegMap, - HashMap hsDistSegMap) - { - - schoolSegmentNameIndexMap = segmentNameIndexMap; - gsDistrictSegmentMap = gsDistSegMap; - hsDistrictSegmentMap = hsDistSegMap; - mgraGsDistrict = mgraGsDist; - mgraHsDistrict = mgraHsDist; - } - - public int[][] getSchoolToursByDestMgra() - { - - // dimension the array - int maxMgra = mgraManager.getMaxMgra(); - int destMgra = 0; - - int[][] schoolTours = new int[schoolSegmentNameIndexMap.size()][maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - for (int r = 0; r < getNumHouseholds(); r++) - { - - Person[] persons = hhs[r].getPersons(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - destMgra = person.getPersonSchoolLocationZone(); - if (destMgra == 0) continue; - - int segmentIndex = -1; - try - { - - if (person.getPersonIsPreschoolChild() == 1) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); - } else if (person.getPersonIsGradeSchool() == 1) - { - int gsDistrict = mgraGsDistrict[destMgra]; - segmentIndex = gsDistrictSegmentMap.get(gsDistrict); - } else if (person.getPersonIsHighSchool() == 1) - { - int hsDistrict = mgraHsDistrict[destMgra]; - segmentIndex = hsDistrictSegmentMap.get(hsDistrict); - } else if (person.getPersonIsUniversityStudent() == 1 && person.getAge() < 30) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); - } else if (person.getPersonIsUniversityStudent() == 1 && person.getAge() >= 30) - { - segmentIndex = schoolSegmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); - } - - // if person type is a student but segment index is -1, the - // person is not enrolled; assume home schooled and don't - // add to sum by home mgra - if (segmentIndex >= 0) schoolTours[segmentIndex][destMgra]++; - - } catch (Exception e) - { - logger.error( - String.format( - "exception caught summing students by origin MGRA for household table record r=%d, person=%d, schoolMgra=%d, segmentIndex=%d.", - r, person.getPersonNum(), destMgra, segmentIndex), e); - throw new RuntimeException(); - } - - } - - } // r (households) - - return schoolTours; - - } - - public int[] getJointToursByHomeZoneSubZone(String purposeString) - { - - // dimension the array - int maxMgra = mgraManager.getMaxMgra(); - - int[] jointTours = new int[maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - int count = 0; - for (int r = 0; r < hhs.length; r++) - { - - try - { - - Tour[] jt = hhs[r].getJointTourArray(); - - if (jt == null) continue; - - for (int i = 0; i < jt.length; i++) - { - // increment the segment count if it's the right purpose - if (jt[i].getTourPurpose().equalsIgnoreCase(purposeString)) - { - int homeMgra = hhs[r].getHhMgra(); - jointTours[homeMgra]++; - count++; - } - } - - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught counting number of joint tours for purpose: %s, for household table record r=%d.", - purposeString, r)); - throw e; - } - - } // r (households) - - return jointTours; - } - - public int[] getAtWorkSubtoursByWorkMgra(String purposeString) - { - - // dimension the array - int maxMgra = mgraManager.getMaxMgra(); - - int[] subtours = new int[maxMgra + 1]; - - // hhs is dimesioned to number of households + 1. - int count = 0; - for (int r = 0; r < hhs.length; r++) - { - - Person[] persons = hhs[r].getPersons(); - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - ArrayList subtourList = person.getListOfAtWorkSubtours(); - - try - { - - if (subtourList.size() == 0) continue; - - for (Tour tour : subtourList) - { - // increment the segment count if it's the right purpose - String tourPurpose = tour.getTourPurpose(); - if (tourPurpose.startsWith(purposeString)) - { - int workZone = tour.getTourOrigMgra(); - subtours[workZone]++; - count++; - } - } - - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught counting number of at-work subtours for purpose: %s, for household table record r=%d, personNum=%d, count=0%d.", - purposeString, r, person.getPersonNum(), count)); - throw e; - } - - } - - } // r (households) - - return subtours; - } - - public void readWorkSchoolLocationResults() - { - - String[] headings = {"HHID", "HomeMGRA", "Income", "PersonID", "PersonNum", "PersonType", - "PersonAge", "EmploymentCategory", "StudentCategory", "WorkSegment", - "SchoolSegment", "WorkLocation", "WorkLocationDistance", "WorkLocationLogsum", - "SchoolLocation", "SchoolLocationDistance", "SchoolLocationLogsum"}; - - String wsLocResultsFileName = propertyMap.get(READ_UWSL_RESULTS_FILENAME); - - // open the input stream - String delimSet = ","; - BufferedReader uwslStream = null; - String fileName = projectDirectory + wsLocResultsFileName; - try - { - uwslStream = new BufferedReader(new FileReader(new File(fileName))); - } catch (FileNotFoundException e) - { - e.printStackTrace(); - System.exit(-1); - } - - // first parse the results file field names from the first record and - // associate column position with fields used in model - HashMap indexHeadingMap = new HashMap(); - - String line = ""; - try - { - line = uwslStream.readLine(); - } catch (IOException e) - { - e.printStackTrace(); - System.exit(-1); - } - StringTokenizer st = new StringTokenizer(line, delimSet); - int col = 0; - while (st.hasMoreTokens()) - { - String label = st.nextToken(); - for (String heading : headings) - { - if (heading.equalsIgnoreCase(label)) - { - indexHeadingMap.put(col, heading); - break; - } - } - col++; - } - - Household hh = null; - Person person = null; - - try - { - - while ((line = uwslStream.readLine()) != null) - { - - // set the line number for the next line in the sample of - // households - // int sortedSampleIndex = - // sortedIndices[sortedSample[sampleCount]]; - - // get the household id and personNum first, before parsing - // other - // fields. Skip to next record if not in the sample. - col = 0; - int id = -1; - int personNum = -1; - int workLocation = -1; - int schoolLocation = -1; - st = new StringTokenizer(line, delimSet); - while (st.hasMoreTokens()) - { - String fieldValue = st.nextToken(); - if (indexHeadingMap.containsKey(col)) - { - String fieldName = indexHeadingMap.get(col++); - - if (fieldName.equalsIgnoreCase("HHID")) - { - id = Integer.parseInt(fieldValue); - - int index = getArrayIndex(id); - hh = hhs[index]; - } else if (fieldName.equalsIgnoreCase("PersonNum")) - { - personNum = Integer.parseInt(fieldValue); - person = hh.getPerson(personNum); - } else if (fieldName.equalsIgnoreCase("WorkSegment")) - { - int workSegment = Integer.parseInt(fieldValue); - person.setWorkLocationSegmentIndex(workSegment); - } else if (fieldName.equalsIgnoreCase("workLocation")) - { - workLocation = Integer.parseInt(fieldValue); - person.setWorkLocation(workLocation); - } else if (fieldName.equalsIgnoreCase("WorkLocationDistance")) - { - float distance = Float.parseFloat(fieldValue); - person.setWorkLocDistance(distance); - } else if (fieldName.equalsIgnoreCase("WorkLocationLogsum")) - { - float logsum = Float.parseFloat(fieldValue); - person.setWorkLocLogsum(logsum); - } else if (fieldName.equalsIgnoreCase("SchoolSegment")) - { - int schoolSegment = Integer.parseInt(fieldValue); - person.setSchoolLocationSegmentIndex(schoolSegment); - } else if (fieldName.equalsIgnoreCase("SchoolLocation")) - { - schoolLocation = Integer.parseInt(fieldValue); - person.setSchoolLoc(schoolLocation); - } else if (fieldName.equalsIgnoreCase("SchoolLocationDistance")) - { - float distance = Float.parseFloat(fieldValue); - person.setSchoolLocDistance(distance); - } else if (fieldName.equalsIgnoreCase("SchoolLocationLogsum")) - { - float logsum = Float.parseFloat(fieldValue); - person.setSchoolLocLogsum(logsum); - break; - } - - } else - { - col++; - } - - } - - } - - } catch (NumberFormatException e) - { - e.printStackTrace(); - System.exit(-1); - } catch (IOException e) - { - e.printStackTrace(); - System.exit(-1); - } - - } - - public void readPreAoResults() - { - - String[] headings = {"HHID", "AO"}; - - String preAoResultsFileName = propertyMap.get(READ_PRE_AO_RESULTS_FILENAME); - - // open the input stream - String delimSet = ","; - BufferedReader inStream = null; - String fileName = projectDirectory + preAoResultsFileName; - try - { - inStream = new BufferedReader(new FileReader(new File(fileName))); - } catch (FileNotFoundException e) - { - e.printStackTrace(); - System.exit(-1); - } - - // first parse the results file field names from the first record and - // associate column position with fields used in model - HashMap indexHeadingMap = new HashMap(); - - String line = ""; - try - { - line = inStream.readLine(); - } catch (IOException e) - { - e.printStackTrace(); - System.exit(-1); - } - StringTokenizer st = new StringTokenizer(line, delimSet); - int col = 0; - while (st.hasMoreTokens()) - { - String label = st.nextToken(); - for (String heading : headings) - { - if (heading.equalsIgnoreCase(label)) - { - indexHeadingMap.put(col, heading); - break; - } - } - col++; - } - - Household hh = null; - - try - { - - while ((line = inStream.readLine()) != null) - { - - // set the line number for the next line in the sample of - // households - // int sortedSampleIndex = - // sortedIndices[sortedSample[sampleCount]]; - - // get the household id first, before parsing other fields. Skip - // to - // next record if not in the sample. - col = 0; - int id = -1; - int ao = -1; - st = new StringTokenizer(line, delimSet); - while (st.hasMoreTokens()) - { - String fieldValue = st.nextToken(); - if (indexHeadingMap.containsKey(col)) - { - String fieldName = indexHeadingMap.get(col++); - - if (fieldName.equalsIgnoreCase("HHID")) - { - id = Integer.parseInt(fieldValue); - - int index = getArrayIndex(id); - hh = hhs[index]; - } else if (fieldName.equalsIgnoreCase("AO")) - { - ao = Integer.parseInt(fieldValue); - // pass in the ao model alternative number to this - // method - hh.setHhAutos(ao + 1); - break; - } - - } else - { - col++; - } - - } - - } - - } catch (NumberFormatException e) - { - e.printStackTrace(); - System.exit(-1); - } catch (IOException e) - { - e.printStackTrace(); - System.exit(-1); - } - - } - - public long getBytesUsedByHouseholdArray() - { - - long numBytes = 0; - for (int i = 0; i < hhs.length; i++) - { - Household hh = hhs[i]; - long size = ObjectUtil.sizeOf(hh); - numBytes += size; - } - - return numBytes; - } - - /** - * Assigns each individual person their own value of time, drawing from a - * lognormal distribution as a function of income. - */ - protected void setDistributedValuesOfTime() - { - - // read in values from property file - setValueOfTimePropertyFileValues(); - - // set up the probability distributions - for (int i = 0; i < valueOfTimeDistribution.length; i++) - { - double mu = Math.log(meanValueOfTime[i] * meanValueOfTimeMultiplierBeforeLogForMu); - valueOfTimeDistribution[i] = new LognormalDist(mu, valueOfTimeLognormalSigma); - } - - for (int i = 0; i < hhs.length; ++i) - { - Household household = hhs[i]; - - // each HH gets a VOT for consistency - double rnum = household.getHhRandom().nextDouble(); - int incomeCategory = getIncomeIndexForValueOfTime(household.getIncomeInDollars()); - double hhValueOfTime = valueOfTimeDistribution[incomeCategory - 1].inverseF(rnum); - - // constrain to logical min and max values - if (hhValueOfTime < minValueOfTime) hhValueOfTime = minValueOfTime; - if (hhValueOfTime > maxValueOfTime) hhValueOfTime = maxValueOfTime; - - // adults get the full value, and children 2/3 (1-based) - Person[] personArray = household.getPersons(); - for (int j = 1; j < personArray.length; ++j) - { - Person person = personArray[j]; - - int age = person.getAge(); - if (age < 18) person.setValueOfTime((float) hhValueOfTime - * hhValueOfTimeMultiplierForPersonUnder18); - else person.setValueOfTime((float) hhValueOfTime); - } - } - } - - /** - * Sets additional properties specific to MTC, included distributed - * value-of-time information - */ - private void setValueOfTimePropertyFileValues() - { - - boolean errorFlag = false; - String propertyValue = ""; - - propertyValue = propertyMap.get(PROPERTIES_MIN_VALUE_OF_TIME_KEY); - if (propertyValue == null) - { - logger.error("property file key missing: " + PROPERTIES_MIN_VALUE_OF_TIME_KEY - + ", not able to set min value of time value."); - errorFlag = true; - } else minValueOfTime = Float.parseFloat(propertyValue); - - propertyValue = propertyMap.get(PROPERTIES_MAX_VALUE_OF_TIME_KEY); - if (propertyValue == null) - { - logger.error("property file key missing: " + PROPERTIES_MAX_VALUE_OF_TIME_KEY - + ", not able to set max value of time value."); - errorFlag = true; - } else maxValueOfTime = Float.parseFloat(propertyValue); - - // mean values of time by income category are specified as a - // "comma-sparated" list of float values - // the number of mean values in the lsit determines the number of income - // categories for value of time - // the number of upper limit income dollar values is expected to be - // number of mean values - 1. - int numIncomeCategories = -1; - String meanValueOfTimesPropertyValue = propertyMap - .get(PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY); - if (meanValueOfTimesPropertyValue == null) - { - logger.error("property file key missing: " + PROPERTIES_MEAN_VALUE_OF_TIME_VALUES_KEY - + ", not able to set mean value of time values."); - errorFlag = true; - } else - { - - ArrayList valueList = new ArrayList(); - StringTokenizer valueTokenizer = new StringTokenizer(meanValueOfTimesPropertyValue, ","); - while (valueTokenizer.hasMoreTokens()) - { - String listValue = valueTokenizer.nextToken(); - float value = Float.parseFloat(listValue.trim()); - valueList.add(value); - } - - numIncomeCategories = valueList.size(); - meanValueOfTime = new float[numIncomeCategories]; - valueOfTimeDistribution = new LognormalDist[numIncomeCategories]; - - for (int i = 0; i < numIncomeCategories; i++) - meanValueOfTime[i] = valueList.get(i); - } - - // read the upper limit values for value of time income ranges. - // there should be exactly 1 less than the number of mean value of time - // values - any other value is an error. - String valueOfTimeIncomesPropertyValue = propertyMap - .get(PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY); - if (valueOfTimeIncomesPropertyValue == null) - { - logger.error("property file key missing: " - + PROPERTIES_MEAN_VALUE_OF_TIME_INCOME_LIMITS_KEY - + ", not able to set upper limits for value of time income ranges."); - errorFlag = true; - } else - { - - ArrayList valueList = new ArrayList(); - StringTokenizer valueTokenizer = new StringTokenizer(valueOfTimeIncomesPropertyValue, - ","); - while (valueTokenizer.hasMoreTokens()) - { - String listValue = valueTokenizer.nextToken(); - int value = Integer.parseInt(listValue.trim()); - valueList.add(value); - } - - int numIncomeValues = valueList.size(); - if (numIncomeValues != (numIncomeCategories - 1)) - { - Exception e = new RuntimeException(); - logger.error("an error occurred reading properties file values for distributed value of time calculations."); - logger.error("the mean value of time values property=" - + meanValueOfTimesPropertyValue + " specifies " + numIncomeCategories - + " mean values, thus " + numIncomeCategories + " income ranges."); - logger.error("the value of time income range values property=" - + valueOfTimeIncomesPropertyValue + " specifies " + numIncomeValues - + " income range limit values."); - logger.error("there should be exactly " + (numIncomeCategories - 1) - + " income range limit values for " + numIncomeCategories - + " mean value of time values.", e); - System.exit(-1); - } - - // set the income dollar value upper limits for value of time income - // ranges - incomeDollarLimitsForValueOfTime = new int[numIncomeValues + 1]; - for (int i = 0; i < numIncomeValues; i++) - incomeDollarLimitsForValueOfTime[i] = valueList.get(i); - - incomeDollarLimitsForValueOfTime[numIncomeValues] = Integer.MAX_VALUE; - } - - propertyValue = propertyMap.get(PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY); - if (propertyValue == null) - { - logger.error("property file key missing: " - + PROPERTIES_HH_VALUE_OF_TIME_MULTIPLIER_FOR_UNDER_18_KEY - + ", not able to set hh value of time multiplier for kids in hh under age 18."); - errorFlag = true; - } else hhValueOfTimeMultiplierForPersonUnder18 = Float.parseFloat(propertyValue); - - propertyValue = propertyMap.get(PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY); - if (propertyValue == null) - { - logger.error("property file key missing: " - + PROPERTIES_MEAN_VALUE_OF_TIME_MULTIPLIER_FOR_MU_KEY - + ", not able to set lognormal distribution mu parameter multiplier."); - errorFlag = true; - } else meanValueOfTimeMultiplierBeforeLogForMu = Float.parseFloat(propertyValue); - - propertyValue = propertyMap.get(PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY); - if (propertyValue == null) - { - logger.error("property file key missing: " - + PROPERTIES_VALUE_OF_TIME_LOGNORMAL_SIGMA_KEY - + ", not able to set lognormal distribution sigma parameter."); - errorFlag = true; - } else valueOfTimeLognormalSigma = Float.parseFloat(propertyValue); - - if (errorFlag) - { - Exception e = new RuntimeException(); - logger.error( - "errors occurred reading properties file values for distributed value of time calculations.", - e); - System.exit(-1); - } - - } - - private int getIncomeIndexForValueOfTime(int incomeInDollars) - { - int returnValue = -1; - for (int i = 0; i < incomeDollarLimitsForValueOfTime.length; i++) - { - if (incomeInDollars < incomeDollarLimitsForValueOfTime[i]) - { - // return a 1s based index value - returnValue = i + 1; - break; - } - } - - return returnValue; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java deleted file mode 100644 index c31368f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerIf.java +++ /dev/null @@ -1,137 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.HashMap; - -/** - * @author Jim Hicks - * - * Class for managing household and person object data read from - * synthetic population files. - */ -public interface HouseholdDataManagerIf -{ - - String testRemote(); - - void setPropertyFileValues(HashMap propertyMap); - - void setDebugHhIdsFromHashmap(); - - void computeTransponderChoiceTazPercentArrays(); - - double[] getPercentHhsIncome100Kplus(); - - double[] getPercentHhsMultipleAutos(); - - int[] getRandomOrderHhIndexArray(int numHhs); - - int getArrayIndex(int hhId); - - void setHhArray(Household[] hhs); - - void setHhArray(Household[] tempHhs, int startIndex); - - void setSchoolDistrictMappings(HashMap segmentNameIndexMap, int[] mgraGsDist, - int[] mgraHsDist, HashMap gsDistSegMap, - HashMap hsDistSegMap); - - void setupHouseholdDataManager(ModelStructure modelStructure, String inputHouseholdFileName, - String inputPersonFileName); - - int[][] getTourPurposePersonsByHomeMgra(String[] purposeList); - - int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap); - - int[][] getStudentsByHomeMgra(); - - int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap); - - int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap); - - int[][] getSchoolToursByDestMgra(); - - int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString); - - int[] getJointToursByHomeMgra(String purposeString); - - int[] getAtWorkSubtoursByWorkMgra(String purposeString); - - void logPersonSummary(); - - void setUwslRandomCount(int iter); - - void resetUwslRandom(int iter); - - void resetPreAoRandom(); - - void resetAoRandom(int iter); - - void resetFpRandom(); - - void resetCdapRandom(); - - void resetImtfRandom(); - - void resetImtodRandom(); - - void resetAwfRandom(); - - void resetAwlRandom(); - - void resetAwtodRandom(); - - void resetJtfRandom(); - - void resetJtlRandom(); - - void resetJtodRandom(); - - void resetInmtfRandom(); - - void resetInmtlRandom(); - - void resetInmtodRandom(); - - void resetStfRandom(); - - void resetStlRandom(); - - /** - * Sets the HashSet used to trace households for debug purposes and sets the - * debug switch for each of the listed households. Also sets - */ - void setTraceHouseholdSet(); - - /** - * Sets the HashSet used to trace households for debug purposes and sets the - * debug switch for each of the listed households. Also sets - */ - void setHouseholdSampleRate(float sampleRate, int sampleSeed); - - /** - * return the array of Household objects holding the synthetic population - * and choice model outcomes. - * - * @return hhs - */ - Household[] getHhArray(); - - Household[] getHhArray(int firstHhIndex, int lastHhIndex); - - /** - * return the number of household objects read from the synthetic - * population. - * - * @return - */ - int getNumHouseholds(); - - /** - * set walk segment (0-none, 1-short, 2-long walk to transit access) for the - * origin for this tour - */ - int getInitialOriginWalkSegment(int taz, double randomNumber); - - long getBytesUsedByHouseholdArray(); - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java deleted file mode 100644 index e98bdab..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataManagerRmi.java +++ /dev/null @@ -1,373 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -/** - * @author Jim Hicks - * - * Class for managing household and person object data read from - * synthetic population files. - */ -public class HouseholdDataManagerRmi - implements HouseholdDataManagerIf, Serializable -{ - - UtilRmi remote; - String connectString; - - public HouseholdDataManagerRmi(String hostname, int port, String className) - { - - connectString = String.format("//%s:%d/%s", hostname, port, className); - remote = new UtilRmi(connectString); - - } - - public void setPropertyFileValues(HashMap propertyMap) - { - Object[] objArray = {propertyMap}; - remote.method("setPropertyFileValues", objArray); - } - - public void setDebugHhIdsFromHashmap() - { - Object[] objArray = {}; - remote.method("setDebugHhIdsFromHashmap", objArray); - } - - public void setupHouseholdDataManager(ModelStructure modelStructure, - String inputHouseholdFileName, String inputPersonFileName) - { - Object[] objArray = {modelStructure, inputHouseholdFileName, inputPersonFileName}; - remote.method("setupHouseholdDataManager", objArray); - } - - public void setSchoolDistrictMappings(HashMap segmentNameIndexMap, - int[] mgraGsDist, int[] mgraHsDist, HashMap gsDistSegMap, - HashMap hsDistSegMap) - { - Object[] objArray = {segmentNameIndexMap, mgraGsDist, mgraHsDist, gsDistSegMap, - hsDistSegMap}; - remote.method("setSchoolDistrictMappings", objArray); - } - - public void computeTransponderChoiceTazPercentArrays() - { - Object[] objArray = {}; - remote.method("computeTransponderChoiceTazPercentArrays", objArray); - } - - public double[] getPercentHhsIncome100Kplus() - { - Object[] objArray = {}; - return (double[]) remote.method("getPercentHhsIncome100Kplus", objArray); - } - - public double[] getPercentHhsMultipleAutos() - { - Object[] objArray = {}; - return (double[]) remote.method("getPercentHhsMultipleAutos", objArray); - } - - public void logPersonSummary() - { - Object[] objArray = {}; - remote.method("logPersonSummary", objArray); - } - - public int getArrayIndex(int hhId) - { - Object[] objArray = {hhId}; - return (Integer) remote.method("getArrayIndex", objArray); - } - - public int[] getWorksAtHomeBySegment(HashMap segmentValueIndexMap) - { - Object[] objArray = {segmentValueIndexMap}; - return (int[]) remote.method("getWorksAtHomeBySegment", objArray); - } - - public int[][] getWorkToursByDestMgra(HashMap segmentValueIndexMap) - { - Object[] objArray = {segmentValueIndexMap}; - return (int[][]) remote.method("getWorkToursByDestMgra", objArray); - } - - public int[][] getSchoolToursByDestMgra() - { - Object[] objArray = {}; - return (int[][]) remote.method("getSchoolToursByDestMgra", objArray); - } - - public int[][] getWorkersByHomeMgra(HashMap segmentValueIndexMap) - { - Object[] objArray = {segmentValueIndexMap}; - return (int[][]) remote.method("getWorkersByHomeMgra", objArray); - } - - public int[][] getStudentsByHomeMgra() - { - Object[] objArray = {}; - return (int[][]) remote.method("getStudentsByHomeMgra", objArray); - } - - public int[][] getTourPurposePersonsByHomeMgra(String[] purposeList) - { - Object[] objArray = {purposeList}; - return (int[][]) remote.method("getTourPurposePersonsByHomeMgra", objArray); - } - - public int[] getIndividualNonMandatoryToursByHomeMgra(String purposeString) - { - Object[] objArray = {purposeString}; - return (int[]) remote.method("getIndividualNonMandatoryToursByHomeMgra", objArray); - } - - public int[] getJointToursByHomeMgra(String purposeString) - { - Object[] objArray = {purposeString}; - return (int[]) remote.method("getJointToursByHomeMgra", objArray); - } - - public int[] getAtWorkSubtoursByWorkMgra(String purposeString) - { - Object[] objArray = {purposeString}; - return (int[]) remote.method("getAtWorkSubtoursByWorkMgra", objArray); - } - - public String testRemote() - { - Object[] objArray = {}; - return (String) remote.method("testRemote", objArray); - } - - public void mapTablesToHouseholdObjects() - { - Object[] objArray = {}; - remote.method("mapTablesToHouseholdObjects", objArray); - } - - public void writeResultData() - { - Object[] objArray = {}; - remote.method("writeResultData", objArray); - } - - public int[] getRandomOrderHhIndexArray(int numHhs) - { - Object[] objArray = {numHhs}; - return (int[]) remote.method("getRandomOrderHhIndexArray", objArray); - } - - /** - * set the hh id for which debugging info from choice models applied to this - * household will be logged if debug logging. - */ - public void setDebugHouseholdId(int debugHhId, boolean value) - { - Object[] objArray = {debugHhId, value}; - remote.method("setDebugHouseholdId", objArray); - } - - /** - * Sets the HashSet used to trace households for debug purposes and sets the - * debug switch for each of the listed households. Also sets - */ - public void setTraceHouseholdSet() - { - Object[] objArray = {}; - remote.method("setTraceHouseholdSet", objArray); - } - - public void setHouseholdSampleRate(float sampleRate, int sampleSeed) - { - Object[] objArray = {sampleRate, sampleSeed}; - remote.method("setHouseholdSampleRate", objArray); - } - - public void resetUwslRandom(int iter) - { - Object[] objArray = {iter}; - remote.method("resetUwslRandom", objArray); - } - - public void resetPreAoRandom() - { - Object[] objArray = {}; - remote.method("resetPreAoRandom", objArray); - } - - public void setUwslRandomCount(int iter) - { - Object[] objArray = {iter}; - remote.method("setUwslRandomCount", objArray); - } - - public void resetAoRandom(int iter) - { - Object[] objArray = {iter}; - remote.method("resetAoRandom", objArray); - } - - public void resetFpRandom() - { - Object[] objArray = {}; - remote.method("resetFpRandom", objArray); - } - - public void resetCdapRandom() - { - Object[] objArray = {}; - remote.method("resetCdapRandom", objArray); - } - - public void resetImtfRandom() - { - Object[] objArray = {}; - remote.method("resetImtfRandom", objArray); - } - - public void resetImtodRandom() - { - Object[] objArray = {}; - remote.method("resetImtodRandom", objArray); - } - - public void resetAwfRandom() - { - Object[] objArray = {}; - remote.method("resetAwfRandom", objArray); - } - - public void resetAwlRandom() - { - Object[] objArray = {}; - remote.method("resetAwlRandom", objArray); - } - - public void resetAwtodRandom() - { - Object[] objArray = {}; - remote.method("resetAwtodRandom", objArray); - } - - public void resetJtfRandom() - { - Object[] objArray = {}; - remote.method("resetJtfRandom", objArray); - } - - public void resetJtlRandom() - { - Object[] objArray = {}; - remote.method("resetJtlRandom", objArray); - } - - public void resetJtodRandom() - { - Object[] objArray = {}; - remote.method("resetJtodRandom", objArray); - } - - public void resetInmtfRandom() - { - Object[] objArray = {}; - remote.method("resetInmtfRandom", objArray); - } - - public void resetInmtlRandom() - { - Object[] objArray = {}; - remote.method("resetInmtlRandom", objArray); - } - - public void resetInmtodRandom() - { - Object[] objArray = {}; - remote.method("resetInmtodRandom", objArray); - } - - public void resetStfRandom() - { - Object[] objArray = {}; - remote.method("resetStfRandom", objArray); - } - - public void resetStlRandom() - { - Object[] objArray = {}; - remote.method("resetStlRandom", objArray); - } - - /** - * return the array of Household objects holding the synthetic population - * and choice model outcomes. - * - * @return hhs - */ - public Household[] getHhArray() - { - Object[] objArray = {}; - return (Household[]) remote.method("getHhArray", objArray); - } - - public Household[] getHhArray(int first, int last) - { - Object[] objArray = {first, last}; - return (Household[]) remote.method("getHhArray", objArray); - } - - public void setHhArray(Household[] hhs) - { - Object[] objArray = {hhs}; - remote.method("setHhArray", objArray); - } - - public void setHhArray(Household[] tempHhs, int startIndex) - { - Object[] objArray = {tempHhs, startIndex}; - remote.method("setHhArray", objArray); - } - - /** - * return the array of Household objects holding the synthetic population - * and choice model outcomes. - * - * @return hhs - */ - public int[] getHhIndexArray() - { - Object[] objArray = {}; - return (int[]) remote.method("getHhIndexArray", objArray); - } - - /** - * return the number of household objects read from the synthetic - * population. - * - * @return number of households in synthetic population - */ - public int getNumHouseholds() - { - Object[] objArray = {}; - return (Integer) remote.method("getNumHouseholds", objArray); - } - - /** - * set walk segment (0-none, 1-short, 2-long walk to transit access) for the - * origin for this tour - */ - public int getInitialOriginWalkSegment(int taz, double randomNumber) - { - Object[] objArray = {taz, randomNumber}; - return (Integer) remote.method("getInitialOriginWalkSegment", objArray); - } - - public long getBytesUsedByHouseholdArray() - { - Object[] objArray = {}; - return (Long) remote.method("getBytesUsedByHouseholdArray", objArray); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java deleted file mode 100644 index 4a304b6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdDataWriter.java +++ /dev/null @@ -1,1871 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.IOException; -import java.io.PrintWriter; -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.SQLException; -import java.sql.Statement; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Iterator; -import java.util.LinkedList; -import java.util.List; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesDMU; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -// import com.pb.common.util.ObjectUtil; - -/** - * @author crf
    - * Started: Dec 31, 2008 11:46:36 AM - */ -public class HouseholdDataWriter -{ - - private transient Logger logger = Logger.getLogger(HouseholdDataWriter.class); - - private static final String PROPERTIES_HOUSEHOLD_DATA_FILE = "Results.HouseholdDataFile"; - private static final String PROPERTIES_PERSON_DATA_FILE = "Results.PersonDataFile"; - private static final String PROPERTIES_INDIV_TOUR_DATA_FILE = "Results.IndivTourDataFile"; - private static final String PROPERTIES_JOINT_TOUR_DATA_FILE = "Results.JointTourDataFile"; - private static final String PROPERTIES_INDIV_TRIP_DATA_FILE = "Results.IndivTripDataFile"; - private static final String PROPERTIES_JOINT_TRIP_DATA_FILE = "Results.JointTripDataFile"; - - private static final String PROPERTIES_HOUSEHOLD_TABLE = "Results.HouseholdTable"; - private static final String PROPERTIES_PERSON_TABLE = "Results.PersonTable"; - private static final String PROPERTIES_INDIV_TOUR_TABLE = "Results.IndivTourTable"; - private static final String PROPERTIES_JOINT_TOUR_TABLE = "Results.JointTourTable"; - private static final String PROPERTIES_INDIV_TRIP_TABLE = "Results.IndivTripTable"; - private static final String PROPERTIES_JOINT_TRIP_TABLE = "Results.JointTripTable"; - - private static final String PROPERTIES_WRITE_LOGSUMS = "Results.WriteLogsums"; - - private static final int NUM_WRITE_PACKETS = 2000; - - private final String intFormat = "%d"; - private final String floatFormat = "%f"; - private final String doubleFormat = "%f"; - private final String fileStringFormat = "%s"; - private final String databaseStringFormat = "'%s'"; - private String stringFormat = fileStringFormat; - - private boolean saveUtilsProbsFlag = false; - private boolean writeLogsums = false; - private int setNA = -1; - - - private HashMap rbMap; - - private MandatoryAccessibilitiesDMU dmu; - private UtilityExpressionCalculator autoSkimUEC; - private IndexValues iv; - private MgraDataManager mgraManager; - - private ModelStructure modelStructure; - private int iteration; - - private HashMap purposeIndexNameMap; - - public HouseholdDataWriter(HashMap rbMap, ModelStructure modelStructure, - int iteration) - { - logger.info("Writing data structures to files."); - this.modelStructure = modelStructure; - this.iteration = iteration; - this.rbMap = rbMap; - - // create a UEC to get highway distance traveled for tours - String uecFileName = rbMap.get("acc.mandatory.uec.file"); - int dataPage = Integer.parseInt(rbMap.get("acc.mandatory.data.page")); - int autoSkimPage = Integer.parseInt(rbMap.get("acc.mandatory.auto.page")); - File uecFile = new File(uecFileName); - dmu = new MandatoryAccessibilitiesDMU(); - autoSkimUEC = new UtilityExpressionCalculator(uecFile, autoSkimPage, dataPage, rbMap, dmu); - iv = new IndexValues(); - mgraManager = MgraDataManager.getInstance(rbMap); - - purposeIndexNameMap = this.modelStructure.getIndexPrimaryPurposeNameMap(); - - // default is to not save the tour mode choice utils and probs for each - // tour - String saveUtilsProbsString = rbMap - .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); - if (saveUtilsProbsString != null) - { - if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; - } - - String writeLogsumsString = rbMap.get(PROPERTIES_WRITE_LOGSUMS); - writeLogsums = Boolean.valueOf(writeLogsumsString); - - } - - // NOTE - this method should not be called simultaneously with the file one - // one - // as the string format is changed - public void writeDataToDatabase(HouseholdDataManagerIf householdData, String dbFileName) - { - logger.info("Writing data structures to database."); - long t = System.currentTimeMillis(); - stringFormat = databaseStringFormat; - writeData(householdData, new DatabaseDataWriter(dbFileName)); - float delta = ((Long) (System.currentTimeMillis() - t)).floatValue() / 60000.0f; - logger.info("Finished writing data structures to database (" + delta + " minutes)."); - } - - // NOTE - this method should not be called simultaneously with the database - // one - // one as the string format is changed - public void writeDataToFiles(HouseholdDataManagerIf householdData) - { - logger.info("Writing data structures to csv file."); - stringFormat = fileStringFormat; - FileDataWriter fdw = new FileDataWriter(); - writeData(householdData, fdw); - } - - private void writeData(HouseholdDataManagerIf householdDataManager, DataWriter writer) - { - int hhid = 0; - int persNum = 0; - int tourid = 0; - try - { - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - long maxSize = 0; - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (Household hh : householdArray) - { - if (hh == null) continue; - hhid = hh.getHhId(); - - // long size = ObjectUtil.sizeOf(hh); - // if (size > maxSize) maxSize = size; - - writer.writeHouseholdData(formHouseholdDataEntry(hh)); - for (Person p : hh.getPersons()) - { - if (p == null) continue; - persNum = p.getPersonNum(); - - writer.writePersonData(formPersonDataEntry(p)); - for (Tour t : p.getListOfWorkTours()) - writeIndivTourData(t, writer); - for (Tour t : p.getListOfSchoolTours()) - writeIndivTourData(t, writer); - for (Tour t : p.getListOfIndividualNonMandatoryTours()) - writeIndivTourData(t, writer); - for (Tour t : p.getListOfAtWorkSubtours()) - writeIndivTourData(t, writer); - } - Tour[] jointTours = hh.getJointTourArray(); - if (jointTours != null) for (Tour t : jointTours) - { - if (t == null) continue; - writeJointTourData(t, writer); - } - } - } - - // logger.info("max size for all Household objects after writing output files is " - // + maxSize + " bytes."); - - } catch (RuntimeException e) - { - logger.error(String.format("error writing hh=%d, persNum=%d", hhid, persNum), e); - throw new RuntimeException(); - } finally - { - writer.finishActions(); - } - } - - private void writeIndivTourData(Tour t, DataWriter writer) - { - writer.writeIndivTourData(formIndivTourDataEntry(t)); - - Stop[] outboundStops = t.getOutboundStops(); - if (outboundStops != null) - { - for (int i = 0; i < outboundStops.length; i++) - { - writer.writeIndivTripData(formIndivTripDataEntry(outboundStops[i])); - } - } else - { - writer.writeIndivTripData(formTourAsIndivTripDataEntry(t, false)); - } - - Stop[] inboundStops = t.getInboundStops(); - if (inboundStops != null) - { - for (Stop s : inboundStops) - writer.writeIndivTripData(formIndivTripDataEntry(s)); - } else - { - writer.writeIndivTripData(formTourAsIndivTripDataEntry(t, true)); - } - - } - - private void writeJointTourData(Tour t, DataWriter writer) - { - writer.writeJointTourData(formJointTourDataEntry(t)); - - Stop[] outboundStops = t.getOutboundStops(); - if (outboundStops != null) - { - for (Stop s : outboundStops) - writer.writeJointTripData(formJointTripDataEntry(s)); - } else - { - writer.writeJointTripData(formTourAsJointTripDataEntry(t, false)); - } - - Stop[] inboundStops = t.getInboundStops(); - if (inboundStops != null) - { - for (Stop s : inboundStops) - writer.writeJointTripData(formJointTripDataEntry(s)); - } else - { - writer.writeJointTripData(formTourAsJointTripDataEntry(t, true)); - } - - } - - private String string(int value) - { - return String.format(intFormat, value); - } - - private String string(float value) - { - return String.format(floatFormat, value); - } - - private String string(double value) - { - return String.format(doubleFormat, value); - } - - private String string(String value) - { - return String.format(stringFormat, value); - } - - private List formHouseholdColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("home_mgra"); - data.add("income"); - data.add("autos"); - data.add("HVs"); - data.add("AVs"); - data.add("transponder"); - data.add("cdap_pattern"); - data.add("out_escort_choice"); - data.add("inb_escort_choice"); - data.add("jtf_choice"); - - if(writeLogsums){ - data.add("aoLogsum"); - data.add("transponderLogsum"); - data.add("cdapLogsum"); - data.add("jtfLogsum"); - } - - return data; - } - - private List formHouseholdColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - - if(writeLogsums){ - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - } - return data; - } - - private List formHouseholdDataEntry(Household hh) - { - List data = new LinkedList(); - data.add(string(hh.getHhId())); - data.add(string(hh.getHhMgra())); - data.add(string(hh.getIncomeInDollars())); - data.add(string(hh.getAutosOwned())); - data.add(string(hh.getConventionalVehicles())); - data.add(string(hh.getAutomatedVehicles())); - data.add(string(hh.getTpChoice())); - data.add(string(hh.getCoordinatedDailyActivityPattern())); - data.add(string(hh.getOutboundEscortChoice())); - data.add(string(hh.getInboundEscortChoice())); - data.add(string(hh.getJointTourFreqChosenAlt())); - - if(writeLogsums){ - data.add(string(hh.getAutoOwnershipLogsum())); - data.add(string(hh.getTransponderLogsum())); - data.add(string(hh.getCdapLogsum())); - data.add(string(hh.getJtfLogsum())); - } - return data; - } - - private List formPersonColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("person_id"); - data.add("person_num"); - data.add("age"); - data.add("gender"); - data.add("type"); - data.add("value_of_time"); - data.add("activity_pattern"); - data.add("imf_choice"); - data.add("inmf_choice"); - data.add("fp_choice"); - data.add("reimb_pct"); - data.add("tele_choice"); - data.add("ie_choice"); - data.add("timeFactorWork"); - data.add("timeFactorNonWork"); - - if(writeLogsums){ - data.add("wfhLogsum"); - data.add("wlLogsum"); - data.add("slLogsum"); - data.add("fpLogsum"); - data.add("tcLogsum"); - data.add("ieLogsum"); - data.add("cdapLogsum"); - data.add("imtfLogsum"); - data.add("inmtfLogsum"); - } - - return data; - } - - private List formPersonColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - - if(writeLogsums){ - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - - } - return data; - } - - private List formPersonDataEntry(Person p) - { - List data = new LinkedList(); - data.add(string(p.getHouseholdObject().getHhId())); - data.add(string(p.getPersonId())); - data.add(string(p.getPersonNum())); - data.add(string(p.getAge())); - data.add(string(p.getPersonIsMale() == 1 ? "m" : "f")); - data.add(string(p.getPersonType())); - data.add(string(p.getValueOfTime())); - data.add(string(p.getCdapActivity())); - data.add(string(p.getImtfChoice())); - data.add(string(p.getInmtfChoice())); - data.add(string(p.getFreeParkingAvailableResult())); - data.add(string(p.getParkingReimbursement())); - data.add(string(p.getTelecommuteChoice())); - data.add(string(p.getInternalExternalTripChoiceResult())); - data.add(string(p.getTimeFactorWork())); - data.add(string(p.getTimeFactorNonWork())); - - if(writeLogsums){ - data.add(string(p.getWorksFromHomeLogsum())); - data.add(string(p.getWorkLocationLogsum())); - data.add(string(p.getSchoolLocationLogsum())); - data.add(string(p.getParkingProvisionLogsum())); - data.add(string(p.getTelecommuteLogsum())); - data.add(string(p.getIeLogsum())); - data.add(string(p.getCdapLogsum())); - data.add(string(p.getImtfLogsum())); - data.add(string(p.getInmtfLogsum())); - - } - return data; - } - - private List formIndivTourColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("person_id"); - data.add("person_num"); - data.add("person_type"); - data.add("tour_id"); - data.add("tour_category"); - data.add("tour_purpose"); - data.add("orig_mgra"); - data.add("dest_mgra"); - data.add("start_period"); - data.add("end_period"); - data.add("tour_mode"); - data.add("av_avail"); - data.add("tour_distance"); - data.add("atWork_freq"); - data.add("num_ob_stops"); - data.add("num_ib_stops"); - data.add("valueOfTime"); - - data.add("escort_type_out"); - data.add("escort_type_in"); - data.add("driver_num_out"); - data.add("driver_num_in"); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - for (int i = 1; i <= numModeAlts; i++) - { - String colName = String.format("util_%d", i); - data.add(colName); - } - - for (int i = 1; i <= numModeAlts; i++) - { - String colName = String.format("prob_%d", i); - data.add(colName); - } - } - - if(writeLogsums){ - data.add("timeOfDayLogsum"); - data.add("tourModeLogsum"); - data.add("subtourFreqLogsum"); - data.add("tourDestinationLogsum"); - data.add("stopFreqLogsum"); - - int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; - for(int i = 1; i<= numStopAlts;++i){ - String colName = String.format("outStopDCLogsum_%d", i); - data.add(colName); - } - for(int i = 1; i<= numStopAlts;++i){ - String colName = String.format("inbStopDCLogsum_%d", i); - data.add(colName); - } - } - - return data; - } - - private List formJointTourColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("tour_id"); - data.add("tour_category"); - data.add("tour_purpose"); - data.add("tour_composition"); - data.add("tour_participants"); - data.add("orig_mgra"); - data.add("dest_mgra"); - data.add("start_period"); - data.add("end_period"); - data.add("tour_mode"); - data.add("av_avail"); - data.add("tour_distance"); - data.add("num_ob_stops"); - data.add("num_ib_stops"); - data.add("valueOfTime"); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - for (int i = 1; i <= numModeAlts; i++) - { - String colName = String.format("util_%d", i); - data.add(colName); - } - - for (int i = 1; i <= numModeAlts; i++) - { - String colName = String.format("prob_%d", i); - data.add(colName); - } - } - if(writeLogsums){ - data.add("timeOfDayLogsum"); - data.add("tourModeLogsum"); - data.add("subtourFreqLogsum"); - data.add("tourDestinationLogsum"); - data.add("stopFreqLogsum"); - - int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; - for(int i = 1; i<= numStopAlts;++i){ - String colName = String.format("outStopDCLogsum_%d", i); - data.add(colName); - } - for(int i = 1; i<= numStopAlts;++i){ - String colName = String.format("inbStopDCLogsum_%d", i); - data.add(colName); - } - } - - return data; - } - - private List formIndivTourColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - for (int i = 1; i <= numModeAlts; i++) - { - data.add(SqliteDataTypes.REAL); - } - - for (int i = 1; i <= numModeAlts; i++) - { - data.add(SqliteDataTypes.REAL); - } - } - if(writeLogsums){ - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - - int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; - for(int i = 1; i<= numStopAlts;++i){ - data.add(SqliteDataTypes.REAL); - } - for(int i = 1; i<= numStopAlts;++i){ - data.add(SqliteDataTypes.REAL); - } - } - - - return data; - } - - private List formJointTourColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - for (int i = 1; i <= numModeAlts; i++) - { - data.add(SqliteDataTypes.REAL); - } - - for (int i = 1; i <= numModeAlts; i++) - { - data.add(SqliteDataTypes.REAL); - } - } - - if(writeLogsums){ - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - - int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; - for(int i = 1; i<= numStopAlts;++i){ - data.add(SqliteDataTypes.REAL); - } - for(int i = 1; i<= numStopAlts;++i){ - data.add(SqliteDataTypes.REAL); - } - } - return data; - } - - private List formIndivTourDataEntry(Tour t) - { - - List data = new LinkedList(); - data.add(string(t.getHhId())); - data.add(string(t.getPersonObject().getPersonId())); - data.add(string(t.getPersonObject().getPersonNum())); - data.add(string(t.getPersonObject().getPersonTypeNumber())); - data.add(string(t.getTourId())); - data.add(string(t.getTourCategory())); - data.add(string(t.getTourPurpose())); - data.add(string(t.getTourOrigMgra())); - data.add(string(t.getTourDestMgra())); - data.add(string(t.getTourDepartPeriod())); - data.add(string(t.getTourArrivePeriod())); - data.add(string(t.getTourModeChoice())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - data.add(string(calculateDistancesForAllMgras(t.getTourOrigMgra(), t.getTourDestMgra()))); - data.add(string(t.getSubtourFreqChoice())); - data.add(string(t.getNumOutboundStops() == 0 ? 0 : t.getNumOutboundStops() - 1)); - data.add(string(t.getNumInboundStops() == 0 ? 0 : t.getNumInboundStops() - 1)); - data.add(string(t.getValueOfTime())); - - data.add(string(t.getEscortTypeOutbound())); - data.add(string(t.getEscortTypeInbound())); - data.add(string(t.getDriverPnumOutbound())); - data.add(string(t.getDriverPnumInbound())); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - float[] utils = t.getTourModalUtilities(); - - if (utils != null){ - - for (int i = 0; i < utils.length; i++) - data.add(string(utils[i])); - for (int i = utils.length; i < numModeAlts; i++) - data.add("-999"); - - }else{ - for(int i =0;i outboundStopDCLogsums = t.getOutboundStopDestinationLogsums(); - for(int i = 0; i inboundStopDCLogsums = t.getInboundStopDestinationLogsums(); - for(int i = 0; i formJointTourDataEntry(Tour t) - { - List data = new LinkedList(); - data.add(string(t.getHhId())); - data.add(string(t.getTourId())); - data.add(string(t.getTourCategory())); - data.add(string(t.getTourPurpose())); - data.add(string(t.getJointTourComposition())); - data.add(string(formTourParticipationEntry(t))); - data.add(string(t.getTourOrigMgra())); - data.add(string(t.getTourDestMgra())); - data.add(string(t.getTourDepartPeriod())); - data.add(string(t.getTourArrivePeriod())); - data.add(string(t.getTourModeChoice())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - data.add(string(calculateDistancesForAllMgras(t.getTourOrigMgra(), t.getTourDestMgra()))); - data.add(string(t.getNumOutboundStops() == 0 ? 0 : t.getNumOutboundStops() - 1)); - data.add(string(t.getNumInboundStops() == 0 ? 0 : t.getNumInboundStops() - 1)); - data.add(string(t.getValueOfTime())); - - if (saveUtilsProbsFlag) - { - int numModeAlts = modelStructure.getMaxTourModeIndex(); - float[] utils = t.getTourModalUtilities(); - - int dummy = 0; - if (utils == null) dummy = 1; - - for (int i = 0; i < utils.length; i++) - data.add(string(utils[i])); - for (int i = utils.length; i < numModeAlts; i++) - data.add("-999"); - - float[] probs = t.getTourModalProbabilities(); - for (int i = 0; i < probs.length; i++) - data.add(string(probs[i])); - for (int i = probs.length; i < numModeAlts; i++) - data.add("0.0"); - } - - if(writeLogsums){ - data.add(string(t.getTimeOfDayLogsum())); - data.add(string(t.getTourModeLogsum())); - data.add(string(t.getSubtourFreqLogsum())); - data.add(string(t.getTourDestinationLogsum())); - data.add(string(t.getStopFreqLogsum())); - - int numStopAlts = modelStructure.MAX_STOPS_PER_DIRECTION; - ArrayList outboundStopDCLogsums = t.getOutboundStopDestinationLogsums(); - for(int i = 0; i inboundStopDCLogsums = t.getInboundStopDestinationLogsums(); - for(int i = 0; i formIndivTripColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("person_id"); - data.add("person_num"); - data.add("tour_id"); - data.add("stop_id"); - data.add("inbound"); - data.add("tour_purpose"); - data.add("orig_purpose"); - data.add("dest_purpose"); - data.add("orig_mgra"); - data.add("dest_mgra"); - data.add("parking_mgra"); - data.add("stop_period"); - data.add("trip_mode"); - data.add("av_avail"); - data.add("trip_board_tap"); - data.add("trip_alight_tap"); - data.add("set"); - data.add("tour_mode"); - data.add("driver_pnum"); - data.add("orig_escort_stoptype"); - data.add("orig_escortee_pnum"); - data.add("dest_escort_stoptype"); - data.add("dest_escortee_pnum"); - data.add("valueOfTime"); - data.add("transponder_avail"); - data.add("micro_walkMode"); - data.add("micro_trnAcc"); - data.add("micro_trnEgr"); - data.add("parkingCost"); - - if(writeLogsums) { - data.add("tripModeLogsum"); - data.add("microWalkModeLogsum"); - data.add("microTrnAccLogsum"); - data.add("microTrnEgrLogsum"); - } - return data; - } - - private List formJointTripColumnNames() - { - List data = new LinkedList(); - data.add("hh_id"); - data.add("tour_id"); - data.add("stop_id"); - data.add("inbound"); - data.add("tour_purpose"); - data.add("orig_purpose"); - data.add("dest_purpose"); - data.add("orig_mgra"); - data.add("dest_mgra"); - data.add("parking_mgra"); - data.add("stop_period"); - data.add("trip_mode"); - data.add("av_avail"); - data.add("num_participants"); - data.add("trip_board_tap"); - data.add("trip_alight_tap"); - data.add("set"); - data.add("tour_mode"); - data.add("valueOfTime"); - data.add("transponder_avail"); - //wsu remove micromobility columns, not applicable to joint trips - //data.add("micro_walkMode"); - //data.add("micro_trnAcc"); - //data.add("micro_trnEgr"); - data.add("parkingCost"); - - if(writeLogsums) { - data.add("tripModeLogsum"); - data.add("microWalkModeLogsum"); - data.add("microTrnAccLogsum"); - data.add("microTrnEgrLogsum"); - } - - return data; - } - - private List formIndivTripColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - - if(writeLogsums) { - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - } - return data; - } - - private List formJointTripColumnTypes() - { - List data = new LinkedList(); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.TEXT); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.INTEGER); - data.add(SqliteDataTypes.REAL); - - if(writeLogsums) { - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - data.add(SqliteDataTypes.REAL); - } - return data; - } - - private List formIndivTripDataEntry(Stop s) - { - Tour t = s.getTour(); - Household h = t.getPersonObject().getHouseholdObject(); - - List data = new LinkedList(); - data.add(string(t.getHhId())); - data.add(string(t.getPersonObject().getPersonId())); - data.add(string(t.getPersonObject().getPersonNum())); - data.add(string(t.getTourId())); - data.add(string(s.getStopId())); - data.add(string(s.isInboundStop() ? 1 : 0)); - data.add(string(t.getTourPurpose())); - - if (s.getStopId() == 0) - { - if (s.isInboundStop()) - { - // first trip on inbound half-tour with stops - data.add(s.getOrigPurpose()); - data.add(s.getDestPurpose()); - data.add(string(t.getTourDestMgra())); - data.add(string(s.getDest())); - } else - { - // first trip on outbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add("Work"); - data.add(s.getDestPurpose()); - } else - { - data.add("Home"); - data.add(s.getDestPurpose()); - } - data.add(string(t.getTourOrigMgra())); - data.add(string(s.getDest())); - } - } else if (s.isInboundStop() && s.getStopId() == t.getNumInboundStops() - 1) - { - // last trip on inbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(s.getOrigPurpose()); - data.add("Work"); - } else - { - data.add(s.getOrigPurpose()); - data.add("Home"); - } - data.add(string(s.getOrig())); - data.add(string(t.getTourOrigMgra())); - } else if (!s.isInboundStop() && s.getStopId() == t.getNumOutboundStops() - 1) - { - // last trip on outbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(s.getOrigPurpose()); - data.add(t.getTourPurpose()); - } else - { - data.add(s.getOrigPurpose()); - data.add(t.getTourPurpose()); - } - data.add(string(s.getOrig())); - data.add(string(t.getTourDestMgra())); - } else - { - data.add(s.getOrigPurpose()); - data.add(s.getDestPurpose()); - data.add(string(s.getOrig())); - data.add(string(s.getDest())); - } - - data.add(string(s.getPark())); - data.add(string(s.getStopPeriod())); - data.add(string(s.getMode())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - data.add(string(s.getBoardTap())); - data.add(string(s.getAlightTap())); - int set = setNA; - if(modelStructure.getTripModeIsTransit(s.getMode())) { - set = s.getSet(); - } - data.add(string(set)); - data.add(string(t.getTourModeChoice())); - data.add(string(s.isInboundStop() ? t.getDriverPnumInbound() : t.getDriverPnumOutbound())); - data.add(string(s.getEscortStopTypeOrig())); - data.add(string(s.getEscorteePnumOrig())); - data.add(string(s.getEscortStopTypeDest())); - data.add(string(s.getEscorteePnumDest())); - data.add(string(s.getValueOfTime())); - data.add(string(h.getTpChoice())); - data.add(string(s.getMicromobilityWalkMode())); - data.add(string(s.getMicromobilityAccessMode())); - data.add(string(s.getMicromobilityEgressMode())); - data.add(string(s.getParkingCost())); - - if(writeLogsums) { - data.add(string(s.getModeLogsum())); - data.add(string(s.getMicromobilityWalkLogsum())); - data.add(string(s.getMicromobilityAccessLogsum())); - data.add(string(s.getMicromobilityEgressLogsum())); - - } - return data; - } - - private List formJointTripDataEntry(Stop s) - { - Tour t = s.getTour(); - Household h = t.getPersonObject().getHouseholdObject(); - List data = new LinkedList(); - data.add(string(t.getHhId())); - data.add(string(t.getTourId())); - data.add(string(s.getStopId())); - data.add(string(s.isInboundStop() ? 1 : 0)); - data.add(string(t.getTourPurpose())); - - if (s.getStopId() == 0) - { - if (s.isInboundStop()) - { - // first trip on inbound half-tour with stops - data.add(s.getOrigPurpose()); - data.add(s.getDestPurpose()); - } else - { - // first trip on outbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add("Work"); - data.add(s.getDestPurpose()); - } else - { - data.add("Home"); - data.add(s.getDestPurpose()); - } - } - } else if (s.isInboundStop() && s.getStopId() == t.getNumInboundStops() - 1) - { - // last trip on inbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(s.getOrigPurpose()); - data.add("Work"); - } else - { - data.add(s.getOrigPurpose()); - data.add("Home"); - } - } else if (!s.isInboundStop() && s.getStopId() == t.getNumOutboundStops() - 1) - { - // last trip on outbound half-tour with stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(s.getOrigPurpose()); - data.add(t.getTourPurpose()); - } else - { - data.add(s.getOrigPurpose()); - data.add(t.getTourPurpose()); - } - } else - { - data.add(s.getOrigPurpose()); - data.add(s.getDestPurpose()); - } - - data.add(string(s.getOrig())); - data.add(string(s.getDest())); - data.add(string(s.getPark())); - data.add(string(s.getStopPeriod())); - data.add(string(s.getMode())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - - int[] participants = t.getPersonNumArray(); - if (participants == null) - { - logger.error("tour participants array is null, hhid=" + t.getHhId() + "."); - throw new RuntimeException(); - } - if (participants.length < 2) - { - logger.error("length of tour participants array is not null, but is < 2; should be >= 2 for joint tour, hhid=" - + t.getHhId() + "."); - throw new RuntimeException(); - } - - data.add(string(participants.length)); - data.add(string(s.getBoardTap())); - data.add(string(s.getAlightTap())); - int set = setNA; - if(modelStructure.getTripModeIsTransit(s.getMode())) { - set = s.getSet(); - } - data.add(string(set)); - data.add(string(t.getTourModeChoice())); - data.add(string(s.getValueOfTime())); - data.add(string(h.getTpChoice())); - //wsu, remove micromobility columns, not applicable to joint trips - //data.add(string(s.getMicromobilityWalkMode())); - //data.add(string(s.getMicromobilityAccessMode())); - //data.add(string(s.getMicromobilityEgressMode())); - data.add(string(s.getParkingCost())); - - if(writeLogsums) { - data.add(string(s.getModeLogsum())); - data.add(string(s.getMicromobilityWalkLogsum())); - data.add(string(s.getMicromobilityAccessLogsum())); - data.add(string(s.getMicromobilityEgressLogsum())); - - } - - - if(writeLogsums) - data.add(string(s.getModeLogsum())); - - return data; - } - - private List formTourAsIndivTripDataEntry(Tour t, boolean inbound) - { - List data = new LinkedList(); - Household h = t.getPersonObject().getHouseholdObject(); - - data.add(string(t.getHhId())); - data.add(string(t.getPersonObject().getPersonId())); - data.add(string(t.getPersonObject().getPersonNum())); - data.add(string(t.getTourId())); - data.add(string(-1)); - data.add(string((inbound ? 1 : 0))); - data.add(string(t.getTourPurpose())); - - if (inbound) - { - // inbound trip on half-tour with no stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(t.getTourPurpose()); - data.add("Work"); - } else - { - data.add(t.getTourPurpose()); - data.add("Home"); - } - } else - { - // outbound trip on half-tour with no stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add("Work"); - data.add(t.getTourPurpose()); - } else - { - data.add("Home"); - data.add(t.getTourPurpose()); - } - } - - data.add(string((inbound ? t.getTourDestMgra() : t.getTourOrigMgra()))); - data.add(string((inbound ? t.getTourOrigMgra() : t.getTourDestMgra()))); - data.add(string(t.getTourParkMgra())); - data.add(string(0)); - data.add(string(inbound ? t.getTourArrivePeriod() : t.getTourDepartPeriod())); - data.add(string(t.getTourModeChoice())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - data.add(string(inbound ? t.getDriverPnumInbound() : t.getDriverPnumOutbound())); - - /* //outbound chauffeured school tour no stops; origin stop type and escortee is zero, dest stop type is dropoff, escortee is pnum. - if(!inbound && t.getTourPurpose().equals("School") && t.getDriverPnumOutbound()>0){ - - data.add(string(0)); - data.add(string(0)); - data.add(string(ModelStructure.ESCORT_STOP_TYPE_DROPOFF)); - data.add(string(t.getPersonObject().getPersonNum())); - - }else if(inbound && t.getTourPurpose().equals("School") && t.getDriverPnumInbound()>0){ - - data.add(string(ModelStructure.ESCORT_STOP_TYPE_PICKUP)); - data.add(string(t.getPersonObject().getPersonNum())); - data.add(string(0)); - data.add(string(0)); - }else - */ - if (!inbound && t.getDriverPnumOutbound()>0){ //outbound - data.add(string(0)); //origin = home - data.add(string(0)); //origin = home - Stop[] stops = t.getInboundStops(); //there must be stops in inbound direction - int stopType = stops[0].getEscortStopTypeOrig(); //first inbound stop - int pnum = stops[0].getEscorteePnumOrig(); //first inbound stop - data.add(string(stopType)); //destination - data.add(string(pnum)); //destination - }else if (inbound && t.getDriverPnumInbound()>0){ //inbound - Stop[] stops = t.getOutboundStops(); - int stopType = stops[stops.length-1].getEscortStopTypeOrig(); //last outbound stop - int pnum = stops[stops.length-1].getEscorteePnumOrig(); //last outbound stop - data.add(string(stopType)); //origin - data.add(string(pnum)); //origin - data.add(string(0)); //destination = home - data.add(string(0)); //destination = home - } - else{ - data.add(string(0)); - data.add(string(0)); - data.add(string(0)); - data.add(string(0)); - } - - data.add(string(t.getTourModeChoice())); - - /* if(true){logger.error("Trying to write a tour as a trip"); - - logger.info("HHID: " +t.getHhId()); - logger.info("PERSNUM: "+ t.getPersonObject().getPersonNum()); - logger.info("TOURID: "+t.getTourId()); - logger.info(inbound ? "inbound" : "outbound"); - } - */ - - data.add(string(t.getValueOfTime())); - data.add(string(h.getTpChoice())); - - if(writeLogsums) - data.add(string(t.getTourModeLogsum())); - - return data; - } - - private List formTourAsJointTripDataEntry(Tour t, boolean inbound) - { - List data = new LinkedList(); - data.add(string(t.getHhId())); - data.add(string(t.getTourId())); - data.add(string(-1)); - data.add(string((inbound ? 1 : 0))); - data.add(string(t.getTourPurpose())); - - if (inbound) - { - // inbound trip on half-tour with no stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add(t.getTourPurpose()); - data.add("Work"); - } else - { - data.add(t.getTourPurpose()); - data.add("Home"); - } - } else - { - // outbound trip on half-tour with no stops - if (t.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - data.add("Work"); - data.add(t.getTourPurpose()); - } else - { - data.add("Home"); - data.add(t.getTourPurpose()); - } - } - - data.add(string((inbound ? t.getTourDestMgra() : t.getTourOrigMgra()))); - data.add(string((inbound ? t.getTourOrigMgra() : t.getTourDestMgra()))); - data.add(string(t.getTourParkMgra())); - data.add(string(inbound ? t.getTourArrivePeriod() : t.getTourDepartPeriod())); - data.add(string(t.getTourModeChoice())); - data.add(string(t.getUseOwnedAV() ? 1 : 0)); - - int[] participants = t.getPersonNumArray(); - if (participants == null) - { - logger.error("tour participants array is null, hhid=" + t.getHhId() + "."); - throw new RuntimeException(); - } - if (participants.length < 2) - { - logger.error("length of tour participants array is not null, but is < 2; should be >= 2 for joint tour, hhid=" - + t.getHhId() + "."); - throw new RuntimeException(); - } - - data.add(string(participants.length)); - data.add(string(t.getTourModeChoice())); - data.add(string(t.getValueOfTime())); - - if(writeLogsums) - data.add(string(t.getTourModeLogsum())); - - return data; - } - - private static enum SqliteDataTypes - { - INTEGER, TEXT, REAL - } - - private interface DataWriter - { - void writeHouseholdData(List data); - - void writePersonData(List data); - - void writeIndivTourData(List data); - - void writeJointTourData(List data); - - void writeIndivTripData(List data); - - void writeJointTripData(List data); - - void finishActions(); - } - - private class DatabaseDataWriter - implements DataWriter - { - private final String householdTable = rbMap.get(PROPERTIES_HOUSEHOLD_TABLE); - private final String personTable = rbMap.get(PROPERTIES_PERSON_TABLE); - private final String indivTourTable = rbMap.get(PROPERTIES_INDIV_TOUR_TABLE); - private final String jointTourTable = rbMap.get(PROPERTIES_JOINT_TOUR_TABLE); - private final String indivTripTable = rbMap.get(PROPERTIES_INDIV_TRIP_TABLE); - private final String jointTripTable = rbMap.get(PROPERTIES_JOINT_TRIP_TABLE); - private Connection connection = null; - private PreparedStatement hhPreparedStatement = null; - private PreparedStatement personPreparedStatement = null; - private PreparedStatement indivTourPreparedStatement = null; - private PreparedStatement jointTourPreparedStatement = null; - private PreparedStatement indivTripPreparedStatement = null; - private PreparedStatement jointTripPreparedStatement = null; - - public DatabaseDataWriter(String dbFileName) - { - initializeTables(dbFileName); - } - - private void initializeTables(String dbFileName) - { - Statement s = null; - try - { - connection = ConnectionHelper.getConnection(dbFileName); - s = connection.createStatement(); - s.addBatch(getTableInitializationString(householdTable, formHouseholdColumnNames(), - formHouseholdColumnTypes())); - s.addBatch(getTableInitializationString(personTable, formPersonColumnNames(), - formPersonColumnTypes())); - s.addBatch(getTableInitializationString(indivTourTable, formIndivTourColumnNames(), - formIndivTourColumnTypes())); - s.addBatch(getTableInitializationString(jointTourTable, formJointTourColumnNames(), - formJointTourColumnTypes())); - s.addBatch(getTableInitializationString(indivTripTable, formIndivTripColumnNames(), - formIndivTripColumnTypes())); - s.addBatch(getTableInitializationString(jointTripTable, formJointTripColumnNames(), - formJointTripColumnTypes())); - s.addBatch(getClearTableString(householdTable)); - s.addBatch(getClearTableString(personTable)); - s.addBatch(getClearTableString(indivTourTable)); - s.addBatch(getClearTableString(jointTourTable)); - s.addBatch(getClearTableString(indivTripTable)); - s.addBatch(getClearTableString(jointTripTable)); - s.executeBatch(); - } catch (SQLException e) - { - try - { - if (connection != null) connection.close(); - } catch (SQLException ee) - { - // swallow - } - throw new RuntimeException(e); - } finally - { - closeStatement(s); - } - setupPreparedStatements(); - } - - private void setupPreparedStatements() - { - String psStart = "INSERT INTO "; - String psMiddle = " VALUES (?"; - StringBuilder hhp = new StringBuilder(psStart); - hhp.append(householdTable).append(psMiddle); - for (int i = 1; i < formHouseholdColumnNames().size(); i++) - hhp.append(",?"); - hhp.append(");"); - StringBuilder pp = new StringBuilder(psStart); - pp.append(personTable).append(psMiddle); - for (int i = 1; i < formPersonColumnNames().size(); i++) - pp.append(",?"); - pp.append(");"); - StringBuilder itp = new StringBuilder(psStart); - itp.append(indivTourTable).append(psMiddle); - for (int i = 1; i < formIndivTourColumnNames().size(); i++) - itp.append(",?"); - itp.append(");"); - StringBuilder jtp = new StringBuilder(psStart); - jtp.append(jointTourTable).append(psMiddle); - for (int i = 1; i < formJointTourColumnNames().size(); i++) - jtp.append(",?"); - jtp.append(");"); - StringBuilder itp2 = new StringBuilder(psStart); - itp2.append(indivTripTable).append(psMiddle); - for (int i = 1; i < formIndivTripColumnNames().size(); i++) - itp2.append(",?"); - itp2.append(");"); - StringBuilder jtp2 = new StringBuilder(psStart); - jtp2.append(jointTripTable).append(psMiddle); - for (int i = 1; i < formJointTripColumnNames().size(); i++) - jtp2.append(",?"); - jtp2.append(");"); - try - { - hhPreparedStatement = connection.prepareStatement(hhp.toString()); - personPreparedStatement = connection.prepareStatement(pp.toString()); - indivTourPreparedStatement = connection.prepareStatement(itp.toString()); - jointTourPreparedStatement = connection.prepareStatement(jtp.toString()); - indivTripPreparedStatement = connection.prepareStatement(itp2.toString()); - jointTripPreparedStatement = connection.prepareStatement(jtp2.toString()); - connection.setAutoCommit(false); - } catch (SQLException e) - { - throw new RuntimeException(e); - } - } - - private String getTableInitializationString(String table, List columns, - List types) - { - StringBuilder sb = new StringBuilder("CREATE TABLE IF NOT EXISTS "); - sb.append(table).append(" ("); - Iterator cols = columns.iterator(); - Iterator tps = types.iterator(); - sb.append(cols.next()).append(" ").append(tps.next().name()); - while (cols.hasNext()) - sb.append(",").append(cols.next()).append(" ").append(tps.next().name()); - sb.append(");"); - return sb.toString(); - } - - private String getClearTableString(String table) - { - return "DELETE FROM " + table + ";"; - } - - private void writeToTable(PreparedStatement ps, List values) - { - try - { - int counter = 1; - for (String value : values) - ps.setString(counter++, value); - ps.executeUpdate(); - } catch (SQLException e) - { - throw new RuntimeException(e); - } - } - - // private void writeToTable(String table, List values) { - // StringBuilder sb = new StringBuilder("INSERT INTO"); - // sb.append(" ").append(table).append(" VALUES("); - // Iterator vls = values.iterator(); - // sb.append(vls.next()); - // while (vls.hasNext()) - // sb.append(",").append(vls.next()); - // sb.append(");"); - // try { - // s.addBatch(sb.toString()); - // } catch (SQLException e) { - // try { - // throw new RuntimeException(e); - // } finally { - // try { - // if (s != null) - // s.close(); - // } catch (SQLException ee) { - // //swallow - // } - // try { - // if (connection != null) - // connection.close(); - // } catch (SQLException ee) { - // //swallow - // } - // } - // } - // } - - public void writeHouseholdData(List data) - { - writeToTable(hhPreparedStatement, data); - } - - public void writePersonData(List data) - { - writeToTable(personPreparedStatement, data); - } - - public void writeIndivTourData(List data) - { - writeToTable(indivTourPreparedStatement, data); - } - - public void writeJointTourData(List data) - { - writeToTable(jointTourPreparedStatement, data); - } - - public void writeIndivTripData(List data) - { - writeToTable(indivTripPreparedStatement, data); - } - - public void writeJointTripData(List data) - { - writeToTable(jointTripPreparedStatement, data); - } - - public void finishActions() - { - - try - { - connection.commit(); - } catch (SQLException e) - { - throw new RuntimeException(e); - } finally - { - closeStatement(hhPreparedStatement); - closeStatement(personPreparedStatement); - closeStatement(indivTourPreparedStatement); - closeStatement(jointTourPreparedStatement); - closeStatement(indivTripPreparedStatement); - closeStatement(jointTripPreparedStatement); - try - { - if (connection != null) connection.close(); - } catch (SQLException ee) - { - // swallow - } - } - } - - private void closeStatement(Statement s) - { - try - { - if (s != null) s.close(); - } catch (SQLException e) - { - // swallow - } - } - } - - private class FileDataWriter - implements DataWriter - { - private final PrintWriter hhWriter; - private final PrintWriter personWriter; - private final PrintWriter indivTourWriter; - private final PrintWriter jointTourWriter; - private final PrintWriter indivTripWriter; - private final PrintWriter jointTripWriter; - - public FileDataWriter() - { - String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - - String hhFile = formFileName(rbMap.get(PROPERTIES_HOUSEHOLD_DATA_FILE), iteration); - String personFile = formFileName(rbMap.get(PROPERTIES_PERSON_DATA_FILE), iteration); - String indivTourFile = formFileName(rbMap.get(PROPERTIES_INDIV_TOUR_DATA_FILE), - iteration); - String jointTourFile = formFileName(rbMap.get(PROPERTIES_JOINT_TOUR_DATA_FILE), - iteration); - String indivTripFile = formFileName(rbMap.get(PROPERTIES_INDIV_TRIP_DATA_FILE), - iteration); - String jointTripFile = formFileName(rbMap.get(PROPERTIES_JOINT_TRIP_DATA_FILE), - iteration); - - try - { - hhWriter = new PrintWriter(new File(baseDir + hhFile)); - personWriter = new PrintWriter(new File(baseDir + personFile)); - indivTourWriter = new PrintWriter(new File(baseDir + indivTourFile)); - jointTourWriter = new PrintWriter(new File(baseDir + jointTourFile)); - indivTripWriter = new PrintWriter(new File(baseDir + indivTripFile)); - jointTripWriter = new PrintWriter(new File(baseDir + jointTripFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - writeHouseholdData(formHouseholdColumnNames()); - writePersonData(formPersonColumnNames()); - writeIndivTourData(formIndivTourColumnNames()); - writeJointTourData(formJointTourColumnNames()); - writeIndivTripData(formIndivTripColumnNames()); - writeJointTripData(formJointTripColumnNames()); - } - - private String formFileName(String originalFileName, int iteration) - { - int lastDot = originalFileName.lastIndexOf('.'); - - String returnString = ""; - if (lastDot > 0) - { - String base = originalFileName.substring(0, lastDot); - String ext = originalFileName.substring(lastDot); - returnString = String.format("%s_%d%s", base, iteration, ext); - } else - { - returnString = String.format("%s_%d.csv", originalFileName, iteration); - } - - logger.info("writing household csv file to " + returnString); - - return returnString; - } - - public void writeHouseholdData(List data) - { - writeEntryToCsv(hhWriter, data); - } - - public void writePersonData(List data) - { - writeEntryToCsv(personWriter, data); - } - - public void writeIndivTourData(List data) - { - writeEntryToCsv(indivTourWriter, data); - } - - public void writeJointTourData(List data) - { - writeEntryToCsv(jointTourWriter, data); - } - - public void writeIndivTripData(List data) - { - writeEntryToCsv(indivTripWriter, data); - } - - public void writeJointTripData(List data) - { - writeEntryToCsv(jointTripWriter, data); - } - - private void writeEntryToCsv(PrintWriter pw, List data) - { - pw.println(formCsvString(data)); - } - - private String formCsvString(List data) - { - char delimiter = ','; - Iterator it = data.iterator(); - StringBuilder sb = new StringBuilder(it.next()); - while (it.hasNext()) - sb.append(delimiter).append(it.next()); - return sb.toString(); - } - - public void finishActions() - { - try - { - hhWriter.flush(); - personWriter.flush(); - indivTourWriter.flush(); - jointTourWriter.flush(); - indivTripWriter.flush(); - jointTripWriter.flush(); - } finally - { - hhWriter.close(); - personWriter.close(); - indivTourWriter.close(); - jointTourWriter.close(); - indivTripWriter.close(); - jointTripWriter.close(); - } - - } - } - - private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) - { - - ArrayList startEndIndexList = new ArrayList(); - - int startIndex = 0; - int endIndex = 0; - - while (endIndex < numberOfHouseholds - 1) - { - endIndex = startIndex + NUM_WRITE_PACKETS - 1; - if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) - endIndex = numberOfHouseholds - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += NUM_WRITE_PACKETS; - } - - return startEndIndexList; - - } - - /** - * Calculate auto skims for a given origin to all destination mgras, and - * return auto distance. - * - * @param oMgra - * The origin mgra - * @return An array of distances - */ - private double calculateDistancesForAllMgras(int oMgra, int dMgra) - { - - int oTaz = mgraManager.getTaz(oMgra); - int dTaz = mgraManager.getTaz(dMgra); - - iv.setOriginZone(oTaz); - iv.setDestZone(dTaz); - - // sov time in results[0] and distance in resuls[1] - double[] results = autoSkimUEC.solve(iv, dmu, null); - - return results[1]; - } -} - diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java deleted file mode 100644 index 369b6e9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourDepartureAndDurationTime.java +++ /dev/null @@ -1,1672 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; - -import java.io.Serializable; -import java.util.*; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import com.pb.common.newmodel.ChoiceModelApplication; - -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - - - - - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To - * change this template use File | Settings | File Templates. - */ -public class HouseholdIndividualMandatoryTourDepartureAndDurationTime - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdIndividualMandatoryTourDepartureAndDurationTime.class); - private transient Logger todLogger = Logger.getLogger("todLogger"); - private transient Logger tourMCManLogger = Logger.getLogger("tourMcMan"); - - private static final String IMTOD_UEC_FILE_TARGET = "departTime.uec.file"; - private static final String IMTOD_UEC_DATA_TARGET = "departTime.data.page"; - private static final String IMTOD_UEC_WORK_MODEL_TARGET = "departTime.work.page"; - private static final String IMTOD_UEC_SCHOOL_MODEL_TARGET = "departTime.school.page"; - private static final String IMTOD_UEC_UNIV_MODEL_TARGET = "departTime.univ.page"; - - private int[] workTourDepartureTimeChoiceSample; - private int[] schoolTourDepartureTimeChoiceSample; - - // DMU for the UEC - private TourDepartureTimeAndDurationDMU imtodDmuObject; - private TourModeChoiceDMU mcDmuObject; - - private String tourCategory = ModelStructure.MANDATORY_CATEGORY; - - private ModelStructure modelStructure; - - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private ChoiceModelApplication workTourChoiceModel; - private ChoiceModelApplication schoolTourChoiceModel; - private ChoiceModelApplication univTourChoiceModel; - private TourModeChoiceModel mcModel; - - private boolean[] needToComputeLogsum; - private double[] modeChoiceLogsums; - - private int[] altStarts; - private int[] altEnds; - - private int noAvailableWorkWindowCount = 0; - private int noAvailableSchoolWindowCount = 0; - - private int noUsualWorkLocationForMandatoryActivity = 0; - private int noUsualSchoolLocationForMandatoryActivity = 0; - - private HashMap rbMap; - - private long mcTime; - - public HouseholdIndividualMandatoryTourDepartureAndDurationTime( - HashMap propertyMap, ModelStructure modelStructure, - String[] tourPurposeList, CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) - { - - setupHouseholdIndividualMandatoryTourDepartureAndDurationTime(propertyMap, modelStructure, - tourPurposeList, dmuFactory, mcModel); - - } - - private void setupHouseholdIndividualMandatoryTourDepartureAndDurationTime( - HashMap propertyMap, ModelStructure modelStructure, - String[] tourPurposeList, CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) - { - - logger.info(String.format("setting up %s time-of-day choice model.", tourCategory)); - - // set the model structure - this.modelStructure = modelStructure; - this.mcModel = mcModel; - rbMap = propertyMap; - - tazs = TazDataManager.getInstance(); - mgraManager = MgraDataManager.getInstance(); - - // locate the individual mandatory tour frequency choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String imtodUecFile = propertyMap.get(IMTOD_UEC_FILE_TARGET); - imtodUecFile = uecPath + imtodUecFile; - - int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTOD_UEC_DATA_TARGET); - int workModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, - IMTOD_UEC_WORK_MODEL_TARGET); - int schoolModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, - IMTOD_UEC_SCHOOL_MODEL_TARGET); - int univModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, - IMTOD_UEC_UNIV_MODEL_TARGET); - - // get the dmu objects from the factory - imtodDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); - mcDmuObject = dmuFactory.getModeChoiceDMU(); - - // set up the models - workTourChoiceModel = new ChoiceModelApplication(imtodUecFile, workModelPage, dataPage, - propertyMap, (VariableTable) imtodDmuObject); - schoolTourChoiceModel = new ChoiceModelApplication(imtodUecFile, schoolModelPage, dataPage, - propertyMap, (VariableTable) imtodDmuObject); - univTourChoiceModel = new ChoiceModelApplication(imtodUecFile, univModelPage, dataPage, - propertyMap, (VariableTable) imtodDmuObject); - - // get the alternatives table from the work tod UEC. - TableDataSet altsTable = workTourChoiceModel.getUEC().getAlternativeData(); - altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); - altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); - altsTable = null; - - imtodDmuObject.setTodAlts(altStarts, altEnds); - - int numWorkDepartureTimeChoiceAlternatives = workTourChoiceModel.getNumberOfAlternatives(); - workTourDepartureTimeChoiceSample = new int[numWorkDepartureTimeChoiceAlternatives + 1]; - Arrays.fill(workTourDepartureTimeChoiceSample, 1); - - int numSchoolDepartureTimeChoiceAlternatives = schoolTourChoiceModel - .getNumberOfAlternatives(); - schoolTourDepartureTimeChoiceSample = new int[numSchoolDepartureTimeChoiceAlternatives + 1]; - Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); - - int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; - needToComputeLogsum = new boolean[numLogsumIndices]; - - modeChoiceLogsums = new double[numLogsumIndices]; - - } - - public void applyModel(Household household, boolean runTODChoice, boolean runModeChoice) - { - mcTime = 0; - - Logger modelLogger = todLogger; - if (household.getDebugChoiceModels()) - { - household.logHouseholdObject( - "Pre Individual Mandatory Departure Time Choice Model HHID=" - + household.getHhId(), modelLogger); - if (runModeChoice) - household.logHouseholdObject( - "Pre Individual Mandatory Tour Mode Choice Model HHID=" - + household.getHhId(), tourMCManLogger); - } - - // set the household id, origin taz, hh taz, and debugFlag=false in the - // dmu - imtodDmuObject.setHousehold(household); - - // get the array of persons for this household - Person[] personArray = household.getPersons(); - - - if(!runTODChoice) { - // loop through the persons (1-based array) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - if (runModeChoice) - household.logPersonObject(decisionMakerLabel, tourMCManLogger, person); - } - - try { - ArrayList workTours = person.getListOfWorkTours(); - if(workTours!=null) - if(workTours.size()>0) { - for(Tour tour: workTours) { - runModeChoice(household,person,tour,tour.getTourDepartPeriod(),tour.getTourArrivePeriod()); - } - } - ArrayList schoolTours = person.getListOfSchoolTours(); - if(schoolTours!=null) - if(schoolTours.size()>0) { - for(Tour tour: schoolTours) { - runModeChoice(household,person,tour,tour.getTourDepartPeriod(),tour.getTourArrivePeriod()); - } - } - }catch(Exception e) { - logger.error(String - .format("error mandatory mode choice model for j=%d, hhId=%d, persId=%d, persNum=%d, personType=%s.", - j, person.getHouseholdObject().getHhId(), person.getPersonId(), - person.getPersonNum(), person.getPersonType())); - throw new RuntimeException(e); - - } - - } - return; - } - - // loop through the persons (1-based array) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - person.resetTimeWindow(); - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - if (runModeChoice) - household.logPersonObject(decisionMakerLabel, tourMCManLogger, person); - } - - // mandatory tour departure time and dureation choice models for - // each - // worker/student require a specific order: - // 1. Work tours made by workers, school/university tours made by - // students. - // 2. Work tours made by students, school/university tours made by - // workers. - // TODO: check consistency of these definitions - - // TODO: workers can also be students (school-age and university)?, - // non-driving students can be workers?, - // TODO: cannot be school-age student and university? etc... - - try - { - - if (person.getPersonIsWorker() == 1) - { - applyDepartureTimeChoiceForWorkTours(person, runModeChoice); - if (person.getListOfSchoolTours().size() > 0) - { - if (person.getPersonIsUniversityStudent() == 1) - { - applyDepartureTimeChoiceForUnivTours(person, runModeChoice); - } else - { - applyDepartureTimeChoiceForSchoolTours(person, runModeChoice); - } - } - } else if (person.getPersonIsStudent() == 1 - || person.getPersonIsPreschoolChild() == 1) - { - if (person.getPersonIsUniversityStudent() == 1) - { - applyDepartureTimeChoiceForUnivTours(person, runModeChoice); - } else - { - applyDepartureTimeChoiceForSchoolTours(person, runModeChoice); - } - if (person.getListOfWorkTours().size() > 0) - applyDepartureTimeChoiceForWorkTours(person, runModeChoice); - } else - { - if (person.getListOfWorkTours().size() > 0 - || person.getListOfSchoolTours().size() > 0) - { - logger.error(String - .format("error mandatory departure time choice model for j=%d, hhId=%d, persNum=%d, personType=%s.", - j, person.getHouseholdObject().getHhId(), - person.getPersonNum(), person.getPersonType())); - logger.error(String - .format("person with type other than worker or student has %d work tours and %d school tours.", - person.getListOfWorkTours().size(), person - .getListOfSchoolTours().size())); - throw new RuntimeException(); - } - } - - } catch (Exception e) - { - logger.error(String - .format("error mandatory departure time choice model for j=%d, hhId=%d, persId=%d, persNum=%d, personType=%s.", - j, person.getHouseholdObject().getHhId(), person.getPersonId(), - person.getPersonNum(), person.getPersonType())); - throw new RuntimeException(e); - } - - } - - household.setImtodRandomCount(household.getHhRandomCount()); - - } - - /** - * - * @param person - * object for which time choice should be made - * @return the number of work tours this person had scheduled. - */ - private int applyDepartureTimeChoiceForWorkTours(Person person, boolean runModeChoice) - { - - Logger modelLogger = todLogger; - - // set the dmu object - imtodDmuObject.setPerson(person); - - Household household = person.getHouseholdObject(); - - ArrayList workTours = person.getListOfWorkTours(); - ArrayList schoolTours = person.getListOfSchoolTours(); - - for (int i = 0; i < workTours.size(); i++) - { - - Tour t = workTours.get(i); - t.setTourDepartPeriod(-1); - t.setTourArrivePeriod(-1); - - // dest taz was set from result of usual school location choice when - // tour - // object was created in mandatory tour frequency model. - // TODO: if the destMgra value is -1, then this mandatory tour was - // created for a non-student (retired probably) - // TODO: and we have to resolve this somehow - either genrate a - // work/school location for retired, or change activity type for - // person. - // TODO: for now, we'll just skip the tour, and keep count of them. - int destMgra = t.getTourDestMgra(); - if (destMgra <= 0) - { - noUsualWorkLocationForMandatoryActivity++; - continue; - } - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Mandatory Work Tour Departure Time Choice Model for: Purpose=%s", - t.getTourPurpose()); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), - person.getPersonNum(), person.getPersonType(), t.getTourId(), - workTours.size()); - - workTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "Individual Mandatory Work Tour Departure Time Choice Model: Debug Statement for Household ID: " - + household.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " - + person.getPersonType() - + ", Work Tour Id: " - + t.getTourId() + " of " + workTours.size() + " work tours."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - } - - imtodDmuObject.setDestinationZone(destMgra); - imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); - - // set the dmu object - imtodDmuObject.setTour(t); - - int origMgra = t.getTourOrigMgra(); - imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); - imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); - - // set the choice availability and initialize sample array - - // choicemodelapplication will change sample[] according to - // availability[] - boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, - altEnds); - Arrays.fill(workTourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != workTourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in work departure time choice model for hhId=%d, persId=%d, persNum=%d, work tour %d of %d.", - person.getHouseholdObject().getHhId(), person.getPersonId(), - person.getPersonNum(), i, workTours.size())); - logger.error(String - .format("length of the availability array determined by the number of alternatiuves set in the person scheduler=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the work tour UEC=%d.", - workTourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if no time window is available for the tour, make the first and - // last - // alternatives available - // for that alternative, and keep track of the number of times this - // condition occurs. - boolean noAlternativeAvailable = true; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - noAlternativeAvailable = false; - break; - } - } - - if (noAlternativeAvailable) - { - noAvailableWorkWindowCount++; - departureTimeChoiceAvailability[1] = true; - departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; - } - - // check for multiple tours for this person - // set the first or second switch if multiple tours for person - if (workTours.size() == 1 && person.getListOfSchoolTours().size() == 0) - { - // not a multiple tour pattern - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else if (workTours.size() > 1 && person.getListOfSchoolTours().size() == 0) - { - // Two work tour multiple tour pattern - if (i == 0) - { - // first of 2 work tours - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(i + 1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(1); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else - { - // second of 2 work tours - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = workTours.get(0).getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this second work tour with depart - // <= first work tour departure AND arrive >= first work - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= workTours.get(0).getTourDepartPeriod() - && endPeriod >= workTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } else if (workTours.size() == 1 && schoolTours.size() == 1) - { - // One work tour, one school tour multiple tour pattern - if (person.getPersonIsWorker() == 1) - { - // worker, so work tour is first scheduled, school tour - // comes later. - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(1); - - } else - { - // student, so school tour was already scheduled, this work - // tour is the second. - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = person.getListOfSchoolTours().get(0) - .getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this work tour with depart <= - // first school tour departure AND arrive >= first school - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() - && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } - - // calculate and store the mode choice logsum for the usual work - // location - // for this worker at the various - // departure time and duration alternativees - setWorkTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, - departureTimeChoiceAvailability); - - if (household.getDebugChoiceModels()) - { - household.logTourObject(loggingHeader, modelLogger, person, t); - } - - float logsum=0; - try - { - logsum = (float) workTourChoiceModel.computeUtilities(imtodDmuObject, - imtodDmuObject.getIndexValues(), departureTimeChoiceAvailability, - workTourDepartureTimeChoiceSample); - } catch (Exception e) - { - logger.error("exception caught computing work tour TOD choice utilities."); - throw new RuntimeException(); - } - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has no available alternatives, choose between - // the - // first and last alternative. - int chosen; - if (workTourChoiceModel.getAvailabilityCount() > 0) chosen = workTourChoiceModel - .getChoiceResult(rn); - else chosen = rn < 0.5 ? 1 : altStarts.length; - - // schedule the chosen alternative - int chosenStartPeriod = altStarts[chosen - 1]; - int chosenEndPeriod = altEnds[chosen - 1]; - try - { - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - } catch (Exception e) - { - logger.error("exception caught updating work tour TOD choice time windows."); - throw new RuntimeException(); - } - - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = workTourChoiceModel.getUtilities(); - double[] probabilities = workTourChoiceModel.getProbabilities(); - boolean[] availabilities = workTourChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString - + ", Tour Id: " + t.getTourId()); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < workTourChoiceModel.getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], - altEnds[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, - altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - workTourChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - workTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, - rn, chosen); - - // write UEC calculation results to separate model specific log - // file - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - workTourChoiceModel.logUECResults(modelLogger, loggingHeader); - - } - - if (runModeChoice) - { - runModeChoice(household, person, t, chosenStartPeriod, chosenEndPeriod); - } - - } - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format( - "Final Work Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - return workTours.size(); - - } - - private void runModeChoice(Household household, Person person, Tour t, int chosenStartPeriod, int chosenEndPeriod) { - - long check = System.nanoTime(); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); - - // use the mcModel object already setup for computing logsums - // and get - // the mode choice, where the selected - // worklocation and subzone an departure time and duration are - // set - // for this work tour. - int chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); - t.setTourModeChoice(chosenMode); - - mcTime += (System.nanoTime() - check); - } - - - - private void setWorkTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Person person, - Tour tour, boolean[] altAvailable) - { - - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todLogger; - String choiceModelDescription = String.format( - "Work Tour Mode Choice Logsum calculation for %s Departure Time Choice", - tour.getTourPurpose()); - String decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person - .getPersonNum(), person.getPersonType(), tour.getTourId(), person - .getListOfWorkTours().size()); - String loggingHeader = String - .format("%s %s", choiceModelDescription, decisionMakerLabel); - - for (int a = 1; a <= altStarts.length; a++) - { - - // if the depart/arrive alternative is unavailable, no need to check - // to see if a logsum has been calculated - if (!altAvailable[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) - + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " work tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - throw new RuntimeException(e); - } - needToComputeLogsum[index] = false; - } - - } - - imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); - - mcDmuObject.getTourObject().setTourDepartPeriod(0); - mcDmuObject.getTourObject().setTourArrivePeriod(0); - } - - private void setSchoolTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives( - Person person, Tour tour, boolean[] altAvailable) - { - - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todLogger; - String choiceModelDescription = String.format( - "School Tour Mode Choice Logsum calculation for %s Departure Time Choice", - tour.getTourPurpose()); - String decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person - .getPersonNum(), person.getPersonType(), tour.getTourId(), person - .getListOfSchoolTours().size()); - String loggingHeader = String - .format("%s %s", choiceModelDescription, decisionMakerLabel); - - for (int a = 1; a <= altStarts.length; a++) - { - - // if the depart/arrive alternative is unavailable, no need to check - // to see if a logsum has been calculated - if (!altAvailable[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) - + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.error(e); - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " school tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - throw new RuntimeException(); - } - needToComputeLogsum[index] = false; - } - - } - - imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); - - } - - /** - * - * @param person - * object for which time choice should be made - * @return the number of school tours this person had scheduled. - */ - private int applyDepartureTimeChoiceForSchoolTours(Person person, boolean runModeChoice) - { - - Logger modelLogger = todLogger; - - // set the dmu object - imtodDmuObject.setPerson(person); - - Household household = person.getHouseholdObject(); - - ArrayList workTours = person.getListOfWorkTours(); - ArrayList schoolTours = person.getListOfSchoolTours(); - - for (int i = 0; i < schoolTours.size(); i++) - { - - Tour t = schoolTours.get(i); - t.setTourDepartPeriod(-1); - t.setTourArrivePeriod(-1); - - // dest taz was set from result of usual school location choice when - // tour - // object was created in mandatory tour frequency model. - // TODO: if the destMgra value is -1, then this mandatory tour was - // created for a non-student (retired probably) - // TODO: and we have to resolve this somehow - either genrate a - // work/school location for retired, or change activity type for - // person. - // TODO: for now, we'll just skip the tour, and keep count of them. - int destMgra = t.getTourDestMgra(); - if (destMgra <= 0) - { - noUsualSchoolLocationForMandatoryActivity++; - continue; - } - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Mandatory School Tour Departure Time Choice Model for: Purpose=%s", - t.getTourPurpose()); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), - person.getPersonNum(), person.getPersonType(), t.getTourId(), - schoolTours.size()); - - schoolTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "Individual Mandatory School Tour Departure Time Choice Model: Debug Statement for Household ID: " - + household.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " - + person.getPersonType() - + ", Tour Id: " - + t.getTourId() + " of " + schoolTours.size() + " school tours."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - imtodDmuObject.setDestinationZone(destMgra); - imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); - - // set the dmu object - imtodDmuObject.setTour(t); - - int origMgra = t.getTourOrigMgra(); - imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); - imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); - - // set the choice availability and sample - boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, - altEnds); - Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != schoolTourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in school departure time choice model for hhId=%d, persId=%d, persNum=%d, school tour %d of %d.", - person.getHouseholdObject().getHhId(), person.getPersonId(), - person.getPersonNum(), i, schoolTours.size())); - logger.error(String - .format("length of the availability array determined by the number of alternatiuves set in the person scheduler=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the school tour UEC=%d.", - schoolTourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if no time window is available for the tour, make the first and - // last - // alternatives available - // for that alternative, and keep track of the number of times this - // condition occurs. - boolean noAlternativeAvailable = true; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - noAlternativeAvailable = false; - break; - } - } - - if (noAlternativeAvailable) - { - noAvailableSchoolWindowCount++; - departureTimeChoiceAvailability[1] = true; - schoolTourDepartureTimeChoiceSample[1] = 1; - departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; - schoolTourDepartureTimeChoiceSample[schoolTourDepartureTimeChoiceSample.length - 1] = 1; - } - - // check for multiple tours for this person - // set the first or second switch if multiple tours for person - if (schoolTours.size() == 1 && person.getListOfWorkTours().size() == 0) - { - // not a multiple tour pattern - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else if (schoolTours.size() > 1 && person.getListOfWorkTours().size() == 0) - { - // Two school tour multiple tour pattern - if (i == 0) - { - // first of 2 school tours - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(i + 1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(1); - } else - { - // second of 2 school tours - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = schoolTours.get(0).getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this 2nd school tour with depart - // <= first school tour departure AND arrive >= first school - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() - && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } else if (schoolTours.size() == 1 && workTours.size() == 1) - { - // One school tour, one work tour multiple tour pattern - if (person.getPersonIsStudent() == 1) - { - // student, so school tour is first scheduled, work comes - // later. - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(1); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else - { - // worker, so work tour was already scheduled, this school - // tour is the second. - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = person.getListOfWorkTours().get(0) - .getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this 2nd school tour with depart - // <= first work tour departure AND arrive >= first work - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= workTours.get(0).getTourDepartPeriod() - && endPeriod >= workTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } - - // calculate and store the mode choice logsum for the usual school - // location for this student at the various - // departure time and duration alternativees - setSchoolTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, - departureTimeChoiceAvailability); - - if (household.getDebugChoiceModels()) - { - household.logTourObject(loggingHeader, modelLogger, person, t); - } - - float logsum = (float) schoolTourChoiceModel.computeUtilities(imtodDmuObject, imtodDmuObject.getIndexValues(), - departureTimeChoiceAvailability, schoolTourDepartureTimeChoiceSample); - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has no available alternatives, choose between - // the - // first and last alternative. - int chosen; - if (schoolTourChoiceModel.getAvailabilityCount() > 0) chosen = schoolTourChoiceModel - .getChoiceResult(rn); - else chosen = rn < 0.5 ? 1 : altStarts.length; - - // schedule the chosen alternative - int chosenStartPeriod = altStarts[chosen - 1]; - int chosenEndPeriod = altEnds[chosen - 1]; - try - { - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - } catch (Exception e) - { - logger.error("exception caught updating school tour TOD choice time windows."); - throw new RuntimeException(); - } - - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = schoolTourChoiceModel.getUtilities(); - double[] probabilities = schoolTourChoiceModel.getProbabilities(); - boolean[] availabilities = schoolTourChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString - + ", Tour Id: " + t.getTourId()); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < schoolTourChoiceModel.getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], - altEnds[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, - altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - schoolTourChoiceModel.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - schoolTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, - rn, chosen); - - // write UEC calculation results to separate model specific log - // file - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - schoolTourChoiceModel.logUECResults(modelLogger, loggingHeader, 200); - - } - - if (runModeChoice) - { - - long check = System.nanoTime(); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); - - // use the mcModel object already setup for computing logsums - // and get - // the mode choice, where the selected - // school location and subzone and departure time and duration - // are - // set for this school tour. - int chosenMode = -1; - chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); - - t.setTourModeChoice(chosenMode); - - mcTime += (System.nanoTime() - check); - } - - } - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String - .format("Final School Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - return schoolTours.size(); - - } - - private void setUnivTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Person person, - Tour tour, boolean[] altAvailable) - { - - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todLogger; - String choiceModelDescription = String.format( - "University Tour Mode Choice Logsum calculation for %s Departure Time Choice", - tour.getTourPurpose()); - String decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person - .getPersonNum(), person.getPersonType(), tour.getTourId(), person - .getListOfSchoolTours().size()); - String loggingHeader = String - .format("%s %s", choiceModelDescription, decisionMakerLabel); - - for (int a = 1; a <= altStarts.length; a++) - { - - // if the depart/arrive alternative is unavailable, no need to check - // to see if a logsum has been calculated - if (!altAvailable[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getSkimMatrixPeriodString(startPeriod) - + " to " + modelStructure.getSkimMatrixPeriodString(endPeriod); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.error(e); - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " university tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - throw new RuntimeException(); - } - needToComputeLogsum[index] = false; - } - - } - - imtodDmuObject.setModeChoiceLogsums(modeChoiceLogsums); - - } - - /** - * - * @param person - * object for which time choice should be made - * @return the number of school tours this person had scheduled. - */ - private int applyDepartureTimeChoiceForUnivTours(Person person, boolean runModeChoice) - { - - Logger modelLogger = todLogger; - - // set the dmu object - imtodDmuObject.setPerson(person); - - Household household = person.getHouseholdObject(); - - ArrayList workTours = person.getListOfWorkTours(); - ArrayList schoolTours = person.getListOfSchoolTours(); - - for (int i = 0; i < schoolTours.size(); i++) - { - - Tour t = schoolTours.get(i); - t.setTourDepartPeriod(-1); - t.setTourArrivePeriod(-1); - - // dest taz was set from result of usual school location choice when - // tour - // object was created in mandatory tour frequency model. - // TODO: if the destMgra value is -1, then this mandatory tour was - // created for a non-student (retired probably) - // TODO: and we have to resolve this somehow - either genrate a - // work/school location for retired, or change activity type for - // person. - // TODO: for now, we'll just skip the tour, and keep count of them. - int destMgra = t.getTourDestMgra(); - if (destMgra <= 0) - { - noUsualSchoolLocationForMandatoryActivity++; - continue; - } - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Mandatory University Tour Departure Time Choice Model for: Purpose=%s", - t.getTourPurpose()); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), - person.getPersonNum(), person.getPersonType(), t.getTourId(), - schoolTours.size()); - - univTourChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "Individual Mandatory University Tour Departure Time Choice Model: Debug Statement for Household ID: " - + household.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " - + person.getPersonType() - + ", Tour Id: " - + t.getTourId() + " of " + schoolTours.size() + " school tours."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - imtodDmuObject.setDestinationZone(destMgra); - imtodDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); - - // set the dmu object - imtodDmuObject.setTour(t); - - int origMgra = t.getTourOrigMgra(); - imtodDmuObject.setOriginZone(mgraManager.getTaz(origMgra)); - imtodDmuObject.setDestinationZone(mgraManager.getTaz(destMgra)); - - // set the choice availability and sample - boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows(altStarts, - altEnds); - Arrays.fill(schoolTourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != schoolTourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in university departure time choice model for hhId=%d, persId=%d, persNum=%d, school tour %d of %d.", - person.getHouseholdObject().getHhId(), person.getPersonId(), - person.getPersonNum(), i, schoolTours.size())); - logger.error(String - .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the university tour UEC=%d.", - schoolTourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if no time window is available for the tour, make the first and - // last - // alternatives available - // for that alternative, and keep track of the number of times this - // condition occurs. - boolean noAlternativeAvailable = true; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - noAlternativeAvailable = false; - break; - } - } - - if (noAlternativeAvailable) - { - noAvailableSchoolWindowCount++; - departureTimeChoiceAvailability[1] = true; - schoolTourDepartureTimeChoiceSample[1] = 1; - departureTimeChoiceAvailability[departureTimeChoiceAvailability.length - 1] = true; - schoolTourDepartureTimeChoiceSample[schoolTourDepartureTimeChoiceSample.length - 1] = 1; - } - - // check for multiple tours for this person - // set the first or second switch if multiple tours for person - if (schoolTours.size() == 1 && person.getListOfWorkTours().size() == 0) - { - // not a multiple tour pattern - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else if (schoolTours.size() > 1 && person.getListOfWorkTours().size() == 0) - { - // Two school tour multiple tour pattern - if (i == 0) - { - // first of 2 school tours - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(i + 1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(1); - } else - { - // second of 2 school tours - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = schoolTours.get(0).getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this 2nd school tour with depart - // <= first school tour departure AND arrive >= first school - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= schoolTours.get(0).getTourDepartPeriod() - && endPeriod >= schoolTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } else if (schoolTours.size() == 1 && workTours.size() == 1) - { - // One school tour, one work tour multiple tour pattern - if (person.getPersonIsStudent() == 1) - { - // student, so school tour is first scheduled, work comes - // later. - imtodDmuObject.setFirstTour(1); - imtodDmuObject.setSubsequentTour(0); - imtodDmuObject.setTourNumber(1); - imtodDmuObject.setEndOfPreviousScheduledTour(0); - imtodDmuObject.setSubsequentTourIsWork(1); - imtodDmuObject.setSubsequentTourIsSchool(0); - } else - { - // worker, so work tour was already scheduled, this school - // tour is the second. - imtodDmuObject.setFirstTour(0); - imtodDmuObject.setSubsequentTour(1); - imtodDmuObject.setTourNumber(i + 1); - int otherTourArrivePeriod = person.getListOfWorkTours().get(0) - .getTourArrivePeriod(); - imtodDmuObject.setEndOfPreviousScheduledTour(otherTourArrivePeriod); - imtodDmuObject.setSubsequentTourIsWork(0); - imtodDmuObject.setSubsequentTourIsSchool(0); - - // block alternatives for this 2nd school tour with depart - // <= first work tour departure AND arrive >= first work - // tour arrival. - for (int a = 1; a <= altStarts.length; a++) - { - // if the depart/arrive alternative is unavailable, no - // need to check to see if a logsum has been calculated - if (!departureTimeChoiceAvailability[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - if (startPeriod <= workTours.get(0).getTourDepartPeriod() - && endPeriod >= workTours.get(0).getTourArrivePeriod()) - departureTimeChoiceAvailability[a] = false; - } - } - } - - // calculate and store the mode choice logsum for the usual school - // location for this student at the various - // departure time and duration alternativees - setUnivTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(person, t, - departureTimeChoiceAvailability); - - if (household.getDebugChoiceModels()) - { - household.logTourObject(loggingHeader, modelLogger, person, t); - } - - float logsum = (float) univTourChoiceModel.computeUtilities(imtodDmuObject, imtodDmuObject.getIndexValues(), - departureTimeChoiceAvailability, schoolTourDepartureTimeChoiceSample); - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = imtodDmuObject.getDmuHouseholdObject().getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has no available alternatives, choose between - // the - // first and last alternative. - int chosen; - if (univTourChoiceModel.getAvailabilityCount() > 0) chosen = univTourChoiceModel - .getChoiceResult(rn); - else chosen = rn < 0.5 ? 1 : altStarts.length; - - // schedule the chosen alternative - int chosenStartPeriod = altStarts[chosen - 1]; - int chosenEndPeriod = altEnds[chosen - 1]; - try - { - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - } catch (Exception e) - { - logger.error("exception caught updating school tour TOD choice time windows."); - throw new RuntimeException(); - } - - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = univTourChoiceModel.getUtilities(); - double[] probabilities = univTourChoiceModel.getProbabilities(); - boolean[] availabilities = univTourChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString - + ", Tour Id: " + t.getTourId()); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < schoolTourChoiceModel.getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", k + 1, altStarts[k], - altEnds[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, - altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - univTourChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - univTourChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, - rn, chosen); - - // write UEC calculation results to separate model specific log - // file - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - univTourChoiceModel.logUECResults(modelLogger, loggingHeader, 200); - - } - - if (runModeChoice) - { - long check = System.nanoTime(); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, t, chosenStartPeriod, chosenEndPeriod); - - // use the mcModel object already setup for computing logsums - // and get - // the mode choice, where the selected - // school location and subzone and departure time and duration - // are - // set for this school tour. - int chosenMode = -1; - chosenMode = mcModel.getModeChoice(mcDmuObject, t.getTourPurpose()); - - t.setTourModeChoice(chosenMode); - - mcTime += (System.nanoTime() - check); - } - - } - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String - .format("Final University Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - return schoolTours.size(); - - } - - private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, - int startPeriod, int endPeriod) - { - - t.setTourDepartPeriod(startPeriod); - t.setTourArrivePeriod(endPeriod); - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(t); - mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), - t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); - - - - mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(t.getTourOrigMgra())); - mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(t.getTourOrigMgra())); - mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(t.getTourOrigMgra())); - - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(t.getTourDestMgra())); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(t.getTourDestMgra())); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(t.getTourDestMgra())); - - mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t - .getTourOrigMgra()))); - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t - .getTourDestMgra()))); - - mcDmuObject.setOriginMgra(t.getTourOrigMgra()); - mcDmuObject.setDestMgra(t.getTourDestMgra()); - - } - - public long getModeChoiceTime() - { - return mcTime; - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - TourModeChoiceModel mcModel = null; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - String[] tourPurposeList = {"White Collar", "Services", "Health", "Retail and Food", - "Blue Collar", "Military"}; - - HouseholdIndividualMandatoryTourDepartureAndDurationTime testObject = new HouseholdIndividualMandatoryTourDepartureAndDurationTime( - propertyMap, modelStructure, tourPurposeList, dmuFactory, mcModel); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java deleted file mode 100644 index bce9e2c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualMandatoryTourFrequencyModel.java +++ /dev/null @@ -1,414 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -/** - * Implements an invidual mandatory tour frequency model, which selects the - * number of work, school, or work and school tours for each person who selects - * a mandatory activity. There are essentially seven separate models, one for - * each person type (full-time worker, part-time worker, university student, non - * working adults, retired, driving students, and non-driving students), except - * pre-school students. The choices are one work tour, two work tours, one - * school tour, two school tours, and one work and school tour. Availability - * arrays are defined for each person type. - * - * The UEC for the model has two additional matrix calcuation tabs, which - * computes the one-way walk distance and the round-trip auto time to work - * and/or school for the model. This allows us to compute the work and/or school - * time, by setting the DMU destination index, just using the UEC. - * - * @author D. Ory - * - */ -public class HouseholdIndividualMandatoryTourFrequencyModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdIndividualMandatoryTourFrequencyModel.class); - private transient Logger tourFreq = Logger.getLogger("tourFreq"); - - private static final String IMTF_CONTROL_FILE_TARGET = "imtf.uec.file"; - private static final String IMTF_DATA_SHEET_TARGET = "imtf.data.page"; - private static final String IMTF_MODEL_SHEET_TARGET = "imtf.model.page"; - - private static final String MANDATORY_ACTIVITY = Definitions.MANDATORY_PATTERN; - - // model results - public static final int CHOICE_ONE_WORK = 1; - public static final int CHOICE_TWO_WORK = 2; - public static final int CHOICE_ONE_SCHOOL = 3; - public static final int CHOICE_TWO_SCHOOL = 4; - public static final int CHOICE_WORK_AND_SCHOOL = 5; - - public static final String[] CHOICE_RESULTS = {"1 Work", "2 Work", - "1 School", "2 School", "Wrk & Schl", "Worker Works At Home", "Student Works At Home", - "Worker School At Home", "Student School At Home" }; - - private IndividualMandatoryTourFrequencyDMU imtfDmuObject; - private ChoiceModelApplication choiceModelApplication; - - private AccessibilitiesTable accTable; - private MandatoryAccessibilitiesCalculator mandAcc; - - /** - * Constructor establishes the ChoiceModelApplication, which applies the - * logit model via the UEC spreadsheet, and it also establishes the UECs - * used to compute the one-way walk distance to work and/or school and the - * round-trip auto time to work and/or school. The model must be the first - * UEC tab, the one-way distance calculations must be the second UEC tab, - * round-trip time must be the third UEC tab. - * - * @param dmuObject - * is the UEC dmu object for this choice model - * @param uecFileName - * is the UEC control file name - * @param resourceBundle - * is the application ResourceBundle, from which a properties - * file HashMap will be created for the UEC - * @param tazDataManager - * is the object used to interact with the zonal data table - * @param modelStructure - * is the ModelStructure object that defines segmentation and - * other model structure relate atributes - */ - public HouseholdIndividualMandatoryTourFrequencyModel(HashMap propertyMap, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, - AccessibilitiesTable accTable, MandatoryAccessibilitiesCalculator myMandAcc) - { - - setupHouseholdIndividualMandatoryTourFrequencyModel(propertyMap, modelStructure, - dmuFactory, accTable, myMandAcc); - - } - - private void setupHouseholdIndividualMandatoryTourFrequencyModel( - HashMap propertyMap, ModelStructure modelStructure, - CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, - MandatoryAccessibilitiesCalculator myMandAcc) - { - - logger.info("setting up IMTF choice model."); - - accTable = myAccTable; - - // locate the individual mandatory tour frequency choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String imtfUecFile = propertyMap.get(IMTF_CONTROL_FILE_TARGET); - imtfUecFile = uecPath + imtfUecFile; - - int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTF_DATA_SHEET_TARGET); - int modelPage = Util.getIntegerValueFromPropertyMap(propertyMap, IMTF_MODEL_SHEET_TARGET); - - // get the dmu object from the factory - imtfDmuObject = dmuFactory.getIndividualMandatoryTourFrequencyDMU(); - - // set up the model - choiceModelApplication = new ChoiceModelApplication(imtfUecFile, modelPage, dataPage, - propertyMap, (VariableTable) imtfDmuObject); - - mandAcc = myMandAcc; - - } - - /** - * Applies the model for the array of households that are stored in the - * HouseholdDataManager. The results are summarized by person type. - * - * @param householdDataManager - * is the object containg the Household objects for which this - * model is to be applied. - */ - public void applyModel(Household household) - { - - Logger modelLogger = tourFreq; - if (household.getDebugChoiceModels()) - household.logHouseholdObject("Pre Individual Mandatory Tour Frequency Choice HHID=" - + household.getHhId() + " Object", modelLogger); - - int choice = -1; - - // get this household's person array - Person[] personArray = household.getPersons(); - - // set the household id, origin taz, hh taz, and debugFlag=false in the - // dmu - imtfDmuObject.setHousehold(household); - - // set the auto sufficiency dependent escort accessibility value for the - // household - String[] types = {"", "escort0", "escort1", "escort2"}; - int autoSufficiency = household.getAutoSufficiency(); - float accessibility = accTable.getAggregateAccessibility(types[autoSufficiency], - household.getHhMgra()); - imtfDmuObject.setEscortAccessibility(accessibility); - - // loop through the person array (1-based) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - String activity = person.getCdapActivity(); - - try - { - - // only apply the model for those with mandatory activities and - // not - // preschool children - if (person.getPersonIsPreschoolChild() == 0 - && activity.equalsIgnoreCase(MANDATORY_ACTIVITY)) - { - - // set the person - imtfDmuObject.setPerson(person); - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Mandatory Tour Frequency Choice Model:"); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s.", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - - choiceModelApplication.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "Individual Mandatory Tour Frequency Choice Model: Debug Statement for Household ID: " - + household.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " + person.getPersonType() + "."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - } - - double distance = 999.0; - double time = 999.0; - if (person.getPersonIsWorker() == 1) - { - - int workMgra = person.getWorkLocation(); - if (workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - - double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( - household.getHhMgra(), workMgra, - household.getDebugChoiceModels(), tourFreq); - - distance = person.getWorkLocationDistance(); - time = accessibilities[0]; // sov time - // wt time - if (accessibilities[2] > 0.0 && accessibilities[2] < time) - time = accessibilities[2]; - // dt time - if (accessibilities[3] > 0.0 && accessibilities[3] < time) - time = accessibilities[3]; - - } else - { - // no work location; skip the rest if no school - // location. - int schoolMgra = person.getUsualSchoolLocation(); - if (schoolMgra <= 0 - || schoolMgra == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) - continue; - } - - } - imtfDmuObject.setDistanceToWorkLoc(distance); - imtfDmuObject.setBestTimeToWorkLoc(time); - - distance = 999.0; - if (person.getPersonIsUniversityStudent() == 1 - || person.getPersonIsStudentDriving() == 1 - || person.getPersonIsStudentNonDriving() == 1) - { - - int schoolMgra = person.getUsualSchoolLocation(); - if (schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) - { - distance = person.getSchoolLocationDistance(); - } else - { - // no school location; skip the rest if no work - // location. - int workMgra = person.getWorkLocation(); - if (workMgra <= 0 - || workMgra == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - continue; - } - - } - imtfDmuObject.setDistanceToSchoolLoc(distance); - - // compute the utilities - IndexValues index = imtfDmuObject.getIndexValues(); - float logsum = (float) choiceModelApplication.computeUtilities(imtfDmuObject, index); - person.setImtfLogsum(logsum); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, - // make choice. - if (choiceModelApplication.getAvailabilityCount() > 0) choice = choiceModelApplication - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for j=%d, activity=%s, HHID=%d, no available alternatives to choose from in choiceModelApplication.", - j, activity, household.getHhId())); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = choiceModelApplication.getUtilities(); - double[] probabilities = choiceModelApplication.getProbabilities(); - - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " - + person.getPersonType()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < probabilities.length; ++k) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %10s", k + 1, CHOICE_RESULTS[k]); - modelLogger.info(String.format("%-15s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %10s", choice, - CHOICE_RESULTS[choice - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - choiceModelApplication.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - choiceModelApplication.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, choice); - - // write UEC calculation results to separate model - // specific - // log file - choiceModelApplication.logUECResults(modelLogger, loggingHeader); - - } - - person.setImtfChoice(choice); - - // set the person choices - if (choice == CHOICE_ONE_WORK) - { - person.createWorkTours(1, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, - ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); - } else if (choice == CHOICE_TWO_WORK) - { - person.createWorkTours(2, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, - ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); - } else if (choice == CHOICE_ONE_SCHOOL) - { - if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(1, - 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, - ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); - else person.createSchoolTours(1, 0, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); - } else if (choice == CHOICE_TWO_SCHOOL) - { - if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(2, - 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, - ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); - else person.createSchoolTours(2, 0, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); - } else if (choice == CHOICE_WORK_AND_SCHOOL) - { - person.createWorkTours(1, 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, - ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); - if (person.getPersonIsUniversityStudent() == 1) person.createSchoolTours(1, - 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, - ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); - else person.createSchoolTours(1, 0, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); - } - - } else if (activity.equalsIgnoreCase(MANDATORY_ACTIVITY) - && person.getPersonIsPreschoolChild() == 1) - { - // mandatory activity if - // pre-school child with mandatory activity type is assigned - // choice = 3 (1 school tour). - choice = 3; - - person.setImtfChoice(choice); - - // get the school purpose name for a non-driving age person - // to - // use for preschool tour purpose - person.createSchoolTours(1, 0, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); - } - - } catch (Exception e) - { - logger.error(String.format("Exception caught for j=%d, activity=%s, HHID=%d", j, - activity, household.getHhId())); - throw new RuntimeException(); - } - - } // j (person loop) - - household.setImtfRandomCount(household.getHhRandomCount()); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java deleted file mode 100644 index a0cf96e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdIndividualNonMandatoryTourFrequencyModel.java +++ /dev/null @@ -1,823 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Map; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -/** - * Implements an invidual mandatory tour frequency model, which selects the - * number of work, school, or work and school tours for each person who selects - * a mandatory activity. There are essentially seven separate models, one for - * each person type (full-time worker, part-time worker, university student, non - * working adults, retired, driving students, and non-driving students), except - * pre-school students. The choices are one work tour, two work tours, one - * school tour, two school tours, and one work and school tour. Availability - * arrays are defined for each person type. - * - * The UEC for the model has two additional matrix calcuation tabs, which - * computes the one-way walk distance and the round-trip auto time to work - * and/or school for the model. This allows us to compute the work and/or school - * time, by setting the DMU destination index, just using the UEC. - * - * @author D. Ory - * - */ -public class HouseholdIndividualNonMandatoryTourFrequencyModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(HouseholdIndividualNonMandatoryTourFrequencyModel.class); - private transient Logger tourFreq = Logger.getLogger("tourFreq"); - - private static final String UEC_DATA_PAGE_KEY = "inmtf.data.page"; - private static final String UEC_PERSONTYPE_1_PAGE_KEY = "inmtf.perstype1.page"; - private static final String UEC_PERSONTYPE_2_PAGE_KEY = "inmtf.perstype2.page"; - private static final String UEC_PERSONTYPE_3_PAGE_KEY = "inmtf.perstype3.page"; - private static final String UEC_PERSONTYPE_4_PAGE_KEY = "inmtf.perstype4.page"; - private static final String UEC_PERSONTYPE_5_PAGE_KEY = "inmtf.perstype5.page"; - private static final String UEC_PERSONTYPE_6_PAGE_KEY = "inmtf.perstype6.page"; - private static final String UEC_PERSONTYPE_7_PAGE_KEY = "inmtf.perstype7.page"; - private static final String UEC_PERSONTYPE_8_PAGE_KEY = "inmtf.perstype8.page"; - - private static final String HOME_ACTIVITY = Definitions.HOME_PATTERN; - - private static final String PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ = "inmtf.uec.file"; - - private static final String PROPERTIES_TOUR_FREQUENCY_EXTENSION_PROBABILITIES_FILE = "inmtf.FrequencyExtension.ProbabilityFile"; - - private static final int AUTO_LOGSUM_INDEX = 6; - - private AccessibilitiesTable accTable; - private MandatoryAccessibilitiesCalculator mandAcc; - - private HashMap purposeIndexToNameMap; - private HashMap personTypeIndexToModelIndexMap; - - private IndividualNonMandatoryTourFrequencyDMU dmuObject; - private ChoiceModelApplication[] choiceModelApplication; - private TableDataSet alternativesTable; - - private Map tourFrequencyIncreaseProbabilityMap; - private int[] maxTourFrequencyChoiceList; - - private static String[] escortTypes = { - "", "escort0", "escort1", "escort2" }; - private static String[] shopTypes = { - "", "shop0", "shop1", "shop2" }; - private static String[] maintTypes = { - "", "maint0", "maint1", "maint2" }; - private static String[] discrTypes = { - "", "discr0", "discr1", "discr2" }; - private static String[] eatOutTypes = { - "", "eatOut0", "eatOut1", "eatOut2" }; - private static String[] visitTypes = { - "", "visit0", "visit1", "visit2" }; - - /** - * Constructor establishes the ChoiceModelApplication, which applies the - * logit model via the UEC spreadsheet, and it also establishes the UECs - * used to compute the one-way walk distance to work and/or school and the - * round-trip auto time to work and/or school. The model must be the first - * UEC tab, the one-way distance calculations must be the second UEC tab, - * round-trip time must be the third UEC tab. - * - * @param dmuObject - * is the UEC dmu object for this choice model - * @param uecFileName - * is the UEC control file name - * @param resourceBundle - * is the application ResourceBundle, from which a properties - * file HashMap will be created for the UEC - * @param tazDataManager - * is the object used to interact with the zonal data table - * @param modelStructure - * is the ModelStructure object that defines segmentation and - * other model structure relate atributes - */ - - public HouseholdIndividualNonMandatoryTourFrequencyModel(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable myAccTable, - MandatoryAccessibilitiesCalculator myMandAcc) - { - - accTable = myAccTable; - mandAcc = myMandAcc; - - setUpModels(propertyMap, dmuFactory); - - } - - private void setUpModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up %s tour frequency choice model.", - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String inmtfUecFile = propertyMap.get(PROPERTIES_UEC_INDIV_NON_MANDATORY_TOUR_FREQ); - String uecFileName = uecPath + inmtfUecFile; - - dmuObject = dmuFactory.getIndividualNonMandatoryTourFrequencyDMU(); - - personTypeIndexToModelIndexMap = new HashMap(); - - int dataSheet = Integer.parseInt(propertyMap.get(UEC_DATA_PAGE_KEY)); - int sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_1_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(1, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_2_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(2, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_3_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(3, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_4_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(4, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_5_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(5, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_6_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(6, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_7_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(7, sheet); - sheet = Integer.parseInt(propertyMap.get(UEC_PERSONTYPE_8_PAGE_KEY)); - personTypeIndexToModelIndexMap.put(8, sheet); - - choiceModelApplication = new ChoiceModelApplication[personTypeIndexToModelIndexMap.size() + 1]; // one - // choice - // model - // for - // each - // person - // type - // that - // has model specified; Ones indexing for - // personType. - - // set up the model - for (int i : personTypeIndexToModelIndexMap.keySet()) - choiceModelApplication[i] = new ChoiceModelApplication(uecFileName, - personTypeIndexToModelIndexMap.get(i), dataSheet, propertyMap, - (VariableTable) dmuObject); - - // the alternatives are the same for each person type; use the first - // choiceModelApplication to get its uec and from it, get the - // TableDataSet - // of alternatives - // to use to determine which tour purposes should be generated for the - // chose - // alternative. - int ftIndex = personTypeIndexToModelIndexMap.get(1); - alternativesTable = choiceModelApplication[ftIndex].getUEC().getAlternativeData(); - - // check the field names in the alternatives table; make sure the their - // order - // is as expected. - String[] fieldNames = alternativesTable.getColumnLabels(); - - // create a mapping between names used in lookup file and purpose names - // used - // in model - HashMap primaryPurposeMap = new HashMap(); - primaryPurposeMap.put(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME); - primaryPurposeMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME); - primaryPurposeMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME); - primaryPurposeMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME); - primaryPurposeMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME); - primaryPurposeMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME, - dmuObject.TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME); - - purposeIndexToNameMap = new HashMap(); - purposeIndexToNameMap.put(1, ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); - purposeIndexToNameMap.put(2, ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); - purposeIndexToNameMap.put(3, ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); - purposeIndexToNameMap.put(4, ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); - purposeIndexToNameMap.put(5, ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); - purposeIndexToNameMap.put(6, ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); - - if (!fieldNames[0].equalsIgnoreCase("a") && !fieldNames[0].equalsIgnoreCase("alt")) - { - logger.error("error while checking order of fields in IndividualNonMandatoryTourFrequencyModel alternatives file."); - logger.error(String - .format("first field expected to be 'a' or 'alt' (case insensitive), but %s was found instead.", - fieldNames[0])); - throw new RuntimeException(); - } else - { - - for (int i : purposeIndexToNameMap.keySet()) - { - String primaryName = purposeIndexToNameMap.get(i).trim(); - String name = primaryPurposeMap.get(primaryName).trim(); - if (!fieldNames[i].equalsIgnoreCase(name)) - { - logger.error("error while checking order of fields in IndividualNonMandatoryTourFrequencyModel alternatives file."); - logger.error(String - .format("field %d expected to be '%s' (case insensitive), but %s was found instead.", - i, name, fieldNames[i])); - throw new RuntimeException(); - } - } - } - - // load data used for tour frequency extension model - loadIndividualNonMandatoryIncreaseModelData(uecPath - + propertyMap.get(PROPERTIES_TOUR_FREQUENCY_EXTENSION_PROBABILITIES_FILE)); - - } - - /** - * Applies the model for the array of households that are stored in the - * HouseholdDataManager. The results are summarized by person type. - * - * @param householdDataManager - * is the object containg the Household objects for which this - * model is to be applied. - */ - public void applyModel(Household household) - { - - int modelIndex = -1; - int choice = -1; - String personTypeString = "Missing"; - - Logger modelLogger = tourFreq; - if (household.getDebugChoiceModels()) - household.logHouseholdObject("Pre Individual Non-Mandatory Tour Frequency Choice HHID=" - + household.getHhId() + " Object", modelLogger); - - // this will be an array with values 1 -> tours.length being the number - // of - // non-mandatory tours in each category - // this keeps it consistent with the way the alternatives are held in - // the - // alternatives file/arrays - float[] tours = null; - - // get this household's person array - Person[] personArray = household.getPersons(); - - // set the household id, origin taz, hh taz, and debugFlag=false in the - // dmu - dmuObject.setHouseholdObject(household); - - // set the auto sufficiency dependent non-mandatory accessibility values - // for - // the household - int autoSufficiency = household.getAutoSufficiency(); - dmuObject.setShopAccessibility(accTable.getAggregateAccessibility( - shopTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setMaintAccessibility(accTable.getAggregateAccessibility( - maintTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setEatOutAccessibility(accTable.getAggregateAccessibility( - eatOutTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setVisitAccessibility(accTable.getAggregateAccessibility( - visitTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setDiscrAccessibility(accTable.getAggregateAccessibility( - discrTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setEscortAccessibility(accTable.getAggregateAccessibility( - escortTypes[autoSufficiency], household.getHhMgra())); - - dmuObject.setNmDcLogsum(accTable.getAggregateAccessibility("nonmotor", - household.getHhMgra())); - - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - // loop through the person array (1-based) - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - String activity = person.getCdapActivity(); - - try - { - - // only apply the model if person does not have H daily activity - // pattern - if (!activity.equalsIgnoreCase(HOME_ACTIVITY)) - { - - // set the person - dmuObject.setPersonObject(person); - - if (household.getDebugChoiceModels()) - { - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - // set the availability array for the tour frequency model - // same number of alternatives for each person type, so use - // person type 1 to get num alts. - int numberOfAlternatives = choiceModelApplication[1].getNumberOfAlternatives(); - boolean[] availabilityArray = new boolean[numberOfAlternatives + 1]; - Arrays.fill(availabilityArray, true); - - modelIndex = personTypeIndexToModelIndexMap.get(person.getPersonTypeNumber()); - personTypeString = person.getPersonType(); - - int workMgra = person.getWorkLocation(); - double accessibility = 0.0; - if (workMgra > 0 && workMgra != ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( - household.getHhMgra(), workMgra, household.getDebugChoiceModels(), - tourFreq); - accessibility = accessibilities[AUTO_LOGSUM_INDEX]; - } - dmuObject.setWorkAccessibility(accessibility); - - int schoolMgra = person.getUsualSchoolLocation(); - accessibility = 0.0; - if (schoolMgra > 0 && schoolMgra != ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) - { - double[] accessibilities = mandAcc.calculateAccessibilitiesForMgraPair( - household.getHhMgra(), schoolMgra, - household.getDebugChoiceModels(), tourFreq); - accessibility = accessibilities[AUTO_LOGSUM_INDEX]; - } - dmuObject.setSchoolAccessibility(accessibility); - - // person.computeIdapResidualWindows(); - - // create the sample array - int[] sampleArray = new int[availabilityArray.length]; - Arrays.fill(sampleArray, 1); - - // compute the utilities - dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), - household.getHhTaz(), -1); - - if (household.getDebugChoiceModels()) - { - - // write debug header - choiceModelDescription = String - .format("Individual Non-Mandatory Tour Frequency Orignal Choice Model:"); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType()); - choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = choiceModelDescription + " for " + decisionMakerLabel - + "."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, - dmuObject.getDmuIndexValues(), availabilityArray, sampleArray); - person.setInmtfLogsum(logsum); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, - // make choice. - if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for j=%d, activity=%s, HHID=%d, no Non-Mandatory Tour Frequency alternatives available to choose from in choiceModelApplication.", - j, activity, person.getHouseholdObject().getHhId())); - throw new RuntimeException(); - } - - // create the non-mandatory tour objects for the choice - // made. - // createIndividualNonMandatoryTours ( person, choice ); - tours = runIndividualNonMandatoryToursIncreaseModel(person, choice); - createIndividualNonMandatoryTours_new(person, tours); - - // debug output - if (household.getDebugChoiceModels()) - { - - String[] alternativeNames = choiceModelApplication[modelIndex] - .getAlternativeNames(); - double[] utilities = choiceModelApplication[modelIndex].getUtilities(); - double[] probabilities = choiceModelApplication[modelIndex] - .getProbabilities(); - - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " - + personTypeString); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("--------------------------------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < alternativeNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %-66s", k + 1, - getAlternativeNameFromChoice(k + 1)); - modelLogger.info(String.format("%-72s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %s", choice, - getAlternativeNameFromChoice(choice)); - modelLogger.info(String.format( - "Original Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - altString = String.format("%-3d %s", choice, - getAlternativeNameFromModifiedChoice(tours)); - modelLogger.info(String.format("Revised Choice After Increase: %s", - altString)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - choiceModelApplication[modelIndex].logAlternativesInfo( - choiceModelDescription, decisionMakerLabel); - choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, choice); - - loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; - - // write UEC calculation results to separate model - // specific - // log file - choiceModelApplication[modelIndex] - .logUECResults(modelLogger, loggingHeader); - - } - - person.setInmtfChoice(choice); - - } - - } catch (Exception e) - { - logger.error(String.format("Exception caught for j=%d, activity=%s, HHID=%d", j, - activity, household.getHhId())); - throw new RuntimeException(e); - } - - } // j (person loop) - - household.setInmtfRandomCount(household.getHhRandomCount()); - - } - - private String getAlternativeNameFromChoice(int choice) - { - - // use the 1s based choice value as the table row number - float[] rowValues = alternativesTable.getRowValues(choice); - - String altName = ""; - - // rowValues is a 0s based indexed array, but the first field is the - // alternative number, - // and subsequent fields indicate the number of tours to be generated - // for the - // purpose corresponding to the field. - for (int i = 1; i < rowValues.length; i++) - { - - int numTours = (int) rowValues[i]; - if (numTours == 0) continue; - - String purposeName = purposeIndexToNameMap.get(i); - if (altName.length() == 0) altName = String.format(", %d %s", numTours, purposeName); - else altName += String.format(", %d %s", numTours, purposeName); - } - - if (altName.length() == 0) altName = "no tours"; - - return altName; - } - - private String getAlternativeNameFromModifiedChoice(float[] rowValues) - { - - String altName = ""; - - // rowValues is a 0s based indexed array, but the first field is the - // alternative number, - // and subsequent fields indicate the nuimber of tours to be generated - // for - // the purpose corresponding to the field. - for (int i = 1; i < rowValues.length; i++) - { - - int numTours = (int) rowValues[i]; - if (numTours == 0) continue; - - String purposeName = purposeIndexToNameMap.get(i); - if (altName.length() == 0) altName = String.format(", %d %s", numTours, purposeName); - else altName += String.format(", %d %s", numTours, purposeName); - } - - if (altName.length() == 0) altName = "no tours"; - - return altName; - } - - /** - * Logs the results of the model. - * - * public void logResults(){ - * - * logger.info(" "); logger.info( - * "~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~" - * ); logger.info("Individual Non-Mandatory Tour Frequency Model Results"); - * - * // count of model results logger.info(" "); String firstHeader = - * "Person type "; String secondHeader = - * "----------------------------- "; - * - * - * - * String[] purposeNames = alternativesTable.getColumnLabels(); int[] - * columnTotals = new int[purposeNames.length-1]; - * - * for( int i=0; i < columnTotals.length; i++ ) { firstHeader += - * String.format("%12s", purposeNames[i+1]); secondHeader += "----------- "; - * } - * - * firstHeader += String.format("%12s","Total"); secondHeader += - * "-----------"; - * - * logger.info(firstHeader); logger.info(secondHeader); - * - * - * - * int lineTotal = 0; for(int i=0;i(); - TableDataSet probabilityTable; - try - { - probabilityTable = (new CSVFileReader().readFile(new File(filePath))); - } catch (IOException e) - { - logger.error( - "Exception caught reading Individual Non-Mandatory Tour Frequency extension probability table.", - e); - throw new RuntimeException(e); - } - - String personTypeColumnName = "person_type"; - String mandatoryTourParticipationColumnName = "mandatory_tour"; - String jointTourParticipationColumnName = "joint_tour"; - String nonMandatoryTourTypeColumn = "nonmandatory_tour_type"; - String zeroAdditionalToursColumnName = "0_tours"; - String oneAdditionalToursColumnName = "1_tours"; - String twoAdditionalToursColumnName = "2_tours"; - - for (int i = 1; i <= probabilityTable.getRowCount(); i++) - { - int key = getTourIncreaseTableKey( - (int) probabilityTable.getValueAt(i, nonMandatoryTourTypeColumn), - (int) probabilityTable.getValueAt(i, personTypeColumnName), - ((int) probabilityTable.getValueAt(i, mandatoryTourParticipationColumnName)) == 1, - ((int) probabilityTable.getValueAt(i, jointTourParticipationColumnName)) == 1); - tourFrequencyIncreaseProbabilityMap.put(key, - new float[] {probabilityTable.getValueAt(i, zeroAdditionalToursColumnName), - probabilityTable.getValueAt(i, oneAdditionalToursColumnName), - probabilityTable.getValueAt(i, twoAdditionalToursColumnName)}); - } - } - - private float[] runIndividualNonMandatoryToursIncreaseModel(Person person, int choice) - { - // use the 1s based choice value as the table row number - // rowValues is a 0s based indexed array, but the first field is the - // alternative number, - // and subsequent fields indicate the nuimber of tours to be generated - // for - // the purpose corresponding to the field. - - Household household = person.getHouseholdObject(); - - int personType = person.getPersonTypeNumber(); - boolean participatedInMandatoryTour = person.getListOfWorkTours().size() > 0 - || person.getListOfSchoolTours().size() > 0; - - boolean participatedInJointTour = false; - Tour[] jointTours = person.getHouseholdObject().getJointTourArray(); - if (jointTours != null) - { - for (Tour t : jointTours) - { - if (t.getPersonInJointTour(person)) - { - participatedInJointTour = true; - break; - } - } - } - - float[] rowValues = null; - - boolean notDone = true; - while (notDone) - { - - rowValues = alternativesTable.getRowValues(choice); - - int firstCount = tourCountSum(rowValues); - if (firstCount == 0 || firstCount >= 5) // if 0 or 5+ tours already, - // we - // are done - break; - - for (int i = 1; i < rowValues.length; i++) - { - - if (rowValues[i] < maxTourFrequencyChoiceList[i]) continue; - - int newChoice = -1; - int key = getTourIncreaseTableKey(i, personType, participatedInMandatoryTour, - participatedInJointTour); - float[] probabilities = tourFrequencyIncreaseProbabilityMap.get(key); - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - for (int j = 0; j < probabilities.length; j++) - { - if (rn <= probabilities[j]) - { - rowValues[i] += j; - newChoice = j; - break; - } - } - - // debug output - if (household.getDebugChoiceModels()) - { - - Logger modelLogger = tourFreq; - String[] alternativeNames = {"no additional", "1 additional", "2 additional"}; - - modelLogger - .info("Individual Non-Mandatory Tour Frequency Increase Choice for tour purposeName=" - + purposeIndexToNameMap.get(i) + ", purposeIndex=" + i); - modelLogger.info("Alternative Probability CumProb"); - modelLogger.info("--------------- -------------- --------------"); - - // probabilities array has tour frequency extension - // probabilities alread stored as cumulative probabilities. - // calculate alternative probabilities from cumulative for - // logging. - double prob = probabilities[0]; - double cumProb = probabilities[0]; - for (int k = 1; k < alternativeNames.length; k++) - { - cumProb = probabilities[k]; - prob = probabilities[k] - probabilities[k - 1]; - String altString = String.format("%-3d %-15s", k + 1, alternativeNames[k]); - modelLogger.info(String.format("%-20s%18.6e%18.6e", altString, prob, - cumProb)); - } - - modelLogger.info(String.format("choice: %s, with rn=%.8f, randomCount=%d", - newChoice + 1, rn, randomCount)); - - modelLogger.info(""); - modelLogger.info(""); - - } - - } - notDone = tourCountSum(rowValues) > 5; - } - - return rowValues; - } - - private int tourCountSum(float[] tours) - { - // tours are located in indices 1 -> tours.length - int sum = 0; - for (int i = 1; i < tours.length; i++) - sum += tours[i]; - return sum; - } - - private int getTourIncreaseTableKey(int nonMandatoryTourType, int personType, - boolean participatedInMandatoryTour, boolean participatedInJointTour) - { - return nonMandatoryTourType + 100 * personType + 10000 - * (participatedInMandatoryTour ? 1 : 0) + 100000 - * (participatedInJointTour ? 1 : 0); - } - - private void createIndividualNonMandatoryTours_new(Person person, float[] tours) - { - // person.clearIndividualNonMandatoryToursArray(); - - for (int i = 1; i < tours.length; i++) - { - int numTours = (int) tours[i]; - if (numTours > 0) - person.createIndividualNonMandatoryTours(numTours, purposeIndexToNameMap.get(i)); - } - } - - // private void createIndividualNonMandatoryTours ( Person person, int - // choice ) { - // - // // use the 1s based choice value as the table row number - // float[] rowValues = alternativesTable.getRowValues( choice ); - // - // // rowValues is a 0s based indexed array, but the first field is the - // alternative number, - // // and subsequent fields indicate the nuimber of tours to be generated - // for the - // purpose corresponding to the field. - // for ( int i=1; i < rowValues.length; i++ ) { - // - // int numTours = (int)rowValues[i]; - // if ( numTours == 0 ) - // continue; - // - // String purposeName = purposeIndexToNameMap.get(i); - // - // person.createIndividualNonMandatoryTours( numTours, i, purposeName ); - // } - // - // } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java deleted file mode 100644 index 9b6d466..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/HouseholdValidator.java +++ /dev/null @@ -1,82 +0,0 @@ -package org.sandag.abm.ctramp; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagTourBasedModel; - -public class HouseholdValidator -{ - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - - public static boolean vailidateWorkLocChoices(Household hh) - { - boolean result = true; - - for (int j = 1; j <= hh.getHhSize(); j++) - { - Person p = hh.getPerson(j); - int windex = p.getWorkLocationSegmentIndex(); - int wmgra = p.getWorkLocation(); - if (windex != -1 && wmgra == 0) - { - result = false; - logger.info("Invalid work location choice for " + p.getPersonId() + " a " - + p.getAge() + " years old with work segment index " - + p.getWorkLocationSegmentIndex() + " resubmitting job......"); - break; - } - } - return result; - } - - public static boolean vailidateSchoolLocChoices(Household hh) - { - boolean result = true; - - for (int j = 1; j <= hh.getHhSize(); j++) - { - Person p = hh.getPerson(j); - int sindex = p.getSchoolLocationSegmentIndex(); - int smgra = p.getPersonSchoolLocationZone(); - if (sindex != -1 && smgra == 0) - { - result = false; - logger.fatal("Invalid school location choice for " + p.getPersonId() + " a " - + p.getAge() + " years old with school segment index " - + p.getSchoolLocationSegmentIndex() + " resubmitting job....."); - break; - } - } - return result; - } - - public static boolean validateMandatoryDestinationChoices(Household[] householdArray, - String type) - { - boolean result = true; - if (type.equalsIgnoreCase("work")) - { - for (int i = 0; i < householdArray.length; i++) - { - if (!vailidateWorkLocChoices(householdArray[i])) - { - result = false; - break; - } - } - } else if (type.equalsIgnoreCase("school")) - { - for (int i = 0; i < householdArray.length; i++) - { - if (!vailidateSchoolLocChoices(householdArray[i])) - { - result = false; - break; - } - } - } else - { - logger.fatal("invalide mandatory destination choice type:" + type); - } - return result; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java deleted file mode 100644 index 6098701..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualMandatoryTourFrequencyDMU.java +++ /dev/null @@ -1,301 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class IndividualMandatoryTourFrequencyDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(IndividualMandatoryTourFrequencyDMU.class); - - protected HashMap methodIndexMap; - - protected IndexValues dmuIndex; - protected Household household; - protected Person person; - - protected double walkDistanceToWork, walkDistanceToSchool; - protected double roundTripAutoTimeToWork, roundTripAutoTimeToSchool; - - protected double distanceToWork; - protected double timeToWork; - protected double distanceToSchool; - protected double escortAccessibility; - - private int homeTazAreaType; - - public IndividualMandatoryTourFrequencyDMU() - { - dmuIndex = new IndexValues(); - } - - public IndexValues getIndexValues() - { - return dmuIndex; - } - - public void setHousehold(Household passedInHousehold) - { - household = passedInHousehold; - - // set the origin and zone indices - dmuIndex.setOriginZone(household.getHhMgra()); - dmuIndex.setZoneIndex(household.getHhMgra()); - dmuIndex.setHHIndex(household.getHhId()); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (household.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug IMTF UEC"); - } - - } - - public void setHomeTazAreaType(int at) - { - homeTazAreaType = at; - } - - public void setDestinationZone(int destinationZone) - { - dmuIndex.setDestZone(destinationZone); - } - - public void setPerson(Person passedInPerson) - { - person = passedInPerson; - } - - public void setDistanceToWorkLoc(double distance) - { - distanceToWork = distance; - } - - public void setBestTimeToWorkLoc(double time) - { - timeToWork = time; - } - - public void setDistanceToSchoolLoc(double distance) - { - distanceToSchool = distance; - } - - public void setEscortAccessibility(double accessibility) - { - escortAccessibility = accessibility; - } - - public double getDistanceToWorkLocation() - { - return distanceToWork; - } - - public double getBestTimeToWorkLocation() - { - return timeToWork; - } - - public double getDistanceToSchoolLocation() - { - return distanceToSchool; - } - - public double getEscortAccessibility() - { - return escortAccessibility; - } - - public int getFullTimeWorker() - { - return (person.getPersonTypeIsFullTimeWorker()); - } - - public int getPartTimeWorker() - { - return (person.getPersonTypeIsPartTimeWorker()); - } - - public int getUniversityStudent() - { - return (person.getPersonIsUniversityStudent()); - } - - public int getNonWorkingAdult() - { - return (person.getPersonIsNonWorkingAdultUnder65()); - } - - public int getRetired() - { - return (person.getPersonIsNonWorkingAdultOver65()); - } - - public int getDrivingAgeSchoolChild() - { - return (person.getPersonIsStudentDriving()); - } - - public int getPreDrivingAgeSchoolChild() - { - return (person.getPersonIsStudentNonDriving()); - } - - public int getFemale() - { - return (person.getPersonIsFemale()); - } - - public int getPersonType() - { - return person.getPersonTypeNumber(); - } - - public int getAge() - { - return (person.getAge()); - } - - public int getStudentIsEmployed() - { - - if (person.getPersonIsUniversityStudent() == 1 || person.getPersonIsStudentDriving() == 1) - { - return (person.getPersonIsWorker()); - } - - return (0); - } - - public int getNonStudentGoesToSchool() - { - - if (person.getPersonTypeIsFullTimeWorker() == 1 - || person.getPersonTypeIsPartTimeWorker() == 1 - || person.getPersonIsNonWorkingAdultUnder65() == 1 - || person.getPersonIsNonWorkingAdultOver65() == 1) - { - - return (person.getPersonIsStudent()); - } - - return (0); - - } - - public int getNotEmployed() - { - return person.notEmployed(); - } - - public int getNumberOfChildren6To18WithoutMandatoryActivity() - { - return household.getNumberOfChildren6To18WithoutMandatoryActivity(); - } - - public int getAutos() - { - return (household.getAutosOwned()); - } - - public int getDrivers() - { - return (household.getDrivers()); - } - - public int getPreschoolChildren() - { - return household.getNumPreschool(); - } - - public int getNonWorkers() - { - return (household.getNumberOfNonWorkingAdults()); - } - - public int getIncomeInDollars() - { - return (household.getIncomeInDollars()); - } - - public int getIncomeHigherThan50k() - { - if (household.getIncomeCategory() > 2) return (1); - return (0); - } - - public int getNonFamilyHousehold() - { - if (household.getIsNonFamilyHousehold() == 1 || household.getIsGroupQuarters() == 1) return 1; - else return 0; - } - - public int getAreaType() - { - return homeTazAreaType; - } - - public int getUsualWorkLocation() - { - return person.getWorkLocation(); - } - - public int getWorkAtHome() - { - return person.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR ? 1 - : 0; - } - - public int getSchoolAtHome() - { - return person.getPersonSchoolLocationZone() == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX ? 1 - : 0; - } - - public int getTelecommuteFrequency() { - return person.getTelecommuteChoice(); - } - - - public int getUsualSchoolLocation() - { - return person.getUsualSchoolLocation(); - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java deleted file mode 100644 index 317e940..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IndividualNonMandatoryTourFrequencyDMU.java +++ /dev/null @@ -1,773 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class IndividualNonMandatoryTourFrequencyDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(IndividualNonMandatoryTourFrequencyDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Person person; - protected IndexValues dmuIndex; - - protected int homeTazIsUrban; - protected double distanceToWork; - protected double distanceToSchool; - protected double escortAccessibility; - protected double shopAccessibility; - protected double maintAccessibility; - protected double eatOutAccessibility; - protected double visitAccessibility; - protected double discrAccessibility; - protected double nmDcLogsum; - protected double workAccessibility; - protected double schoolAccessibility; - - public String TOUR_FREQ_ALTERNATIVES_FILE_ESCORT_NAME = "Escort"; - public String TOUR_FREQ_ALTERNATIVES_FILE_SHOPPING_NAME = "Shopping"; - public String TOUR_FREQ_ALTERNATIVES_FILE_MAINT_NAME = "Maint"; - public String TOUR_FREQ_ALTERNATIVES_FILE_EAT_OUT_NAME = "EatOut"; - public String TOUR_FREQ_ALTERNATIVES_FILE_VISIT_NAME = "Visit"; - public String TOUR_FREQ_ALTERNATIVES_FILE_DISCR_NAME = "Discr"; - - public IndividualNonMandatoryTourFrequencyDMU() - { - dmuIndex = new IndexValues(); - } - - public Household getHouseholdObject() - { - return hh; - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setPersonObject(Person persObject) - { - person = persObject; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug INMTF UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return household income in dollars - */ - public int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - /** - * @return the number of persons in the "decision making" household. - */ - public int getHouseholdSize() - { - // 1-based indexing, so the array is dimensioned 1 more than the number - // of - // persons. - return hh.getPersons().length - 1; - } - - public int getNumAutos() - { - return hh.getAutosOwned(); - } - - /** - * @return 1 if household has at least 1 car, and the number of cars equals - * the number of workers - */ - public int getCarsEqualsWorkers() - { - int numAutos = hh.getAutosOwned(); - int numWorkers = hh.getWorkers(); - - // household must have at least 1 car, otherwise return 0. - if (numAutos > 0) - { - // if at least one car and numWorkers == numAutos, return 1, - // otherwise 0. - if (numAutos == numWorkers) return 1; - else return 0; - } else - { - return 0; - } - } - - /** - * @return 1 if household has at least 1 car, and the number of cars equals - * the number of workers - */ - public int getMoreCarsThanWorkers() - { - int numAutos = hh.getAutosOwned(); - int numWorkers = hh.getWorkers(); - - if (numAutos > numWorkers) return 1; - else return 0; - } - - public int getNumAdults() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - num += persons[i].getPersonIsAdult(); - return num; - } - - public int getNumChildren() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - num += (persons[i].getPersonIsAdult() == 0 ? 1 : 0); - return num; - } - - public int getPersonIsAdult() - { - return person.getPersonIsAdult(); - } - - public int getPersonIsChild() - { - return person.getPersonIsAdult() == 0 ? 1 : 0; - } - - public int getPersonIsFullTimeWorker() - { - return person.getPersonIsFullTimeWorker(); - } - - public int getPersonIsPartTimeWorker() - { - return person.getPersonIsPartTimeWorker(); - } - - public int getPersonIsUniversity() - { - return person.getPersonIsUniversityStudent(); - } - - public int getPersonIsNonworker() - { - return person.getPersonIsNonWorkingAdultUnder65() - + person.getPersonIsNonWorkingAdultOver65(); - } - - public int getPersonIsPreschool() - { - return person.getPersonIsPreschoolChild(); - } - - public int getPersonIsStudentNonDriving() - { - return person.getPersonIsStudentNonDriving(); - } - - public int getPersonIsStudentDriving() - { - return person.getPersonIsStudentDriving(); - } - - public int getPersonStaysHome() - { - return person.getCdapActivity().equalsIgnoreCase("H") ? 1 : 0; - } - - public int getWorksAtHome() - { - if (person.getPersonIsWorker() == 1 - && person.getWorkLocation() == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) return 1; - else return 0; - } - - public int getFemale() - { - return person.getPersonIsFemale(); - } - - /** - * determines the number of persons in the "decision making" household of - * type: full-time worker. returns the count, or 3, if count is 3 or more. - * - * @return count (up to a max of 3) of the number of full-time workers. - */ - public int getFullTimeWorkers() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: full time worker; if more than 3, - // return - // 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsFullTimeWorker(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: part-time worker. returns the count, or 3, if count is 3 or more. - * - * @return count (up to a max of 3) of the number of part-time workers. - */ - public int getPartTimeWorkers() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: part-time worker; if more than 3, - // return - // 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsPartTimeWorker(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: university student. returns the count, or 3, if count is 3 or more. - * - * @return count (up to a max of 3) of the number of university students. - */ - public int getUniversityStudents() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: university student; if more than 3, - // return 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsUniversityStudent(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: non-worker. returns the count, or 3, if count is 3 or more. - * - * @return count (up to a max of 3) of the number of non-workers. - */ - public int getNonWorkers() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: nonworker + retired; if more than - // 3, - // return 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsNonWorkingAdultUnder65() - + p[i].getPersonIsNonWorkingAdultOver65(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: driving-age student. returns the count, or 3, if count is 3 or - * more. - * - * @return count (up to a max of 3) of the number of driving-age students. - */ - public int getDrivingAgeStudents() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: driving-age student; if more than - // 3, - // return 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsStudentDriving(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: non-driving-age student. returns the count, or 3, if count is 3 or - * more. - * - * @return count (up to a max of 3) of the number of non-driving-age - * students. - */ - public int getNonDrivingAgeStudents() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: non-driving-age student; if more - // than 3, - // return 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsStudentNonDriving(); - if (count == 3) break; - } - - return count; - } - - /** - * determines the number of persons in the "decision making" household of - * type: pre-school age. returns the count, or 3, if count is 3 or more. - * - * @return count (up to a max of 3) of the number of pre-school age - * children. - */ - public int getPreSchoolers() - { - Person[] p = hh.getPersons(); - - // get the count of persons of type: pre-school; if more than 3, return - // 3. - int count = 0; - for (int i = 1; i < p.length; i++) - { - count += p[i].getPersonIsPreschoolChild(); - if (count == 3) break; - } - - return count; - } - - public void setEscortAccessibility(double accessibility) - { - escortAccessibility = accessibility; - } - - public void setShopAccessibility(double accessibility) - { - shopAccessibility = accessibility; - } - - public void setMaintAccessibility(double accessibility) - { - maintAccessibility = accessibility; - } - - public void setEatOutAccessibility(double accessibility) - { - eatOutAccessibility = accessibility; - } - - public void setVisitAccessibility(double accessibility) - { - visitAccessibility = accessibility; - } - - public void setDiscrAccessibility(double accessibility) - { - discrAccessibility = accessibility; - } - - public void setNmDcLogsum(double logsum) - { - nmDcLogsum = logsum; - } - - public void setWorkAccessibility(double accessibility) - { - workAccessibility = accessibility; - } - - public void setSchoolAccessibility(double accessibility) - { - schoolAccessibility = accessibility; - } - - /** - * called by methods invoked by UEC.solve() - * - * @return maximum number of hours mutually available between pairs of - * adults in household - */ - public int getMaxAdultOverlaps() - { - return hh.getMaxAdultOverlaps(); - } - - /** - * called by methods invoked by UEC.solve() - * - * @return maximum number of hours mutually available between pairs of - * children in household - */ - public int getMaxChildOverlaps() - { - return hh.getMaxChildOverlaps(); - } - - /** - * called by methods invoked by UEC.solve() - * - * @return maximum number of hours mutually available between pairs or - * adults/children where pairs consist of different types in - * household - */ - public int getMaxMixedOverlaps() - { - return hh.getMaxMixedOverlaps(); - } - - public int getMaxPairwiseOverlapAdult() - { - int maxOverlap = 0; - - // get array of person objects for the decision making household - Person[] dmuPersons = hh.getPersons(); - - for (int i = 1; i < dmuPersons.length; i++) - { - if (dmuPersons[i].getPersonIsAdult() == 1) - { - int overlap = getOverlap(person, dmuPersons[i]); - if (overlap > maxOverlap) maxOverlap = overlap; - } - } - - return maxOverlap; - } - - public int getMaxPairwiseOverlapChild() - { - int maxOverlap = 0; - - // get array of person objects for the decision making household - Person[] dmuPersons = hh.getPersons(); - - for (int i = 1; i < dmuPersons.length; i++) - { - if (dmuPersons[i].getPersonIsAdult() == 0) - { - int overlap = getOverlap(person, dmuPersons[i]); - if (overlap > maxOverlap) maxOverlap = overlap; - } - } - - return maxOverlap; - } - - // TODO: find out if this is suposed to be total pairwise available hours, - // or - // largest consecutive hours available for persons. - // TODO: right now, assuming total pairwise available hours - private int getOverlap(Person dmuPerson, Person otherPerson) - { - short[] dmuWindow = dmuPerson.getTimeWindows(); - short[] otherWindow = otherPerson.getTimeWindows(); - - int overlap = 0; - for (int i = 0; i < dmuWindow.length; i++) - { - if (dmuWindow[i] == 0 && otherWindow[i] == 0) overlap++; - } - - return overlap; - } - - public int getWindowBeforeFirstMandJointTour() - { - return person.getWindowBeforeFirstMandJointTour(); - } - - public int getWindowBetweenFirstLastMandJointTour() - { - return person.getWindowBetweenFirstLastMandJointTour(); - } - - public int getWindowAfterLastMandJointTour() - { - return person.getWindowAfterLastMandJointTour(); - } - - public int getNumHhFtWorkers() - { - return hh.getNumFtWorkers(); - } - - public int getNumHhPtWorkers() - { - return hh.getNumPtWorkers(); - } - - public int getNumHhUnivStudents() - { - return hh.getNumUnivStudents(); - } - - public int getNumHhNonWorkAdults() - { - return hh.getNumNonWorkAdults(); - } - - public int getNumHhRetired() - { - return hh.getNumRetired(); - } - - public int getNumHhDrivingStudents() - { - return hh.getNumDrivingStudents(); - } - - public int getNumHhNonDrivingStudents() - { - return hh.getNumNonDrivingStudents(); - } - - public int getNumHhPreschool() - { - return hh.getNumPreschool(); - } - - public int getTravelActiveAdults() - { - return hh.getTravelActiveAdults(); - } - - public int getTravelActiveChildren() - { - return hh.getTravelActiveChildren(); - } - - public int getNumMandatoryTours() - { - return person.getNumMandatoryTours(); - } - - public int getNumJointShoppingTours() - { - return person.getNumJointShoppingTours(); - } - - public int getNumJointOthMaintTours() - { - return person.getNumJointOthMaintTours(); - } - - public int getNumJointEatOutTours() - { - return person.getNumJointEatOutTours(); - } - - public int getNumJointSocialTours() - { - return person.getNumJointSocialTours(); - } - - public int getNumJointOthDiscrTours() - { - return person.getNumJointOthDiscrTours(); - } - - public int getJTours() - { - return hh.getJointTourArray().length; - } - - public int getPreDrivingAtHome() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getPersonIsStudentNonDriving() == 1 - && persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) - num++; - } - return num; - } - - public int getPreschoolAtHome() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - { - if (persons[i].getPersonIsPreschoolChild() == 1 - && persons[i].getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) - num++; - } - return num; - } - - public double getDistanceToWorkLocation() - { - return person.getWorkLocationDistance(); - } - - public double getDistanceToSchoolLocation() - { - return person.getSchoolLocationDistance(); - } - - public double getEscortAccessibility() - { - return escortAccessibility; - } - - public double getShopAccessibility() - { - return shopAccessibility; - } - - public double getMaintAccessibility() - { - return maintAccessibility; - } - - public double getEatOutAccessibility() - { - return eatOutAccessibility; - } - - public double getVisitAccessibility() - { - return visitAccessibility; - } - - public double getDiscrAccessibility() - { - return discrAccessibility; - } - - public int getCdapIndex() - { - return person.getCdapIndex(); - } - - public double getNonMotorizedDcLogsum() - { - return nmDcLogsum; - } - - public int getNumPredrivingKidsGoOut() - { - return hh.getNumberOfPreDrivingWithNonHomeActivity(); - } - - public int getNumPreschoolKidsGoOut() - { - return hh.getNumberOfPreschoolWithNonHomeActivity(); - } - - public int getCollegeEducation() - { - return person.getHasBachelors(); - } - - public int getLowEducation() - { - return person.getPersonIsHighSchoolGraduate() == 1 ? 0 : 1; - } - - public int getDetachedHh() - { - return hh.getHhBldgsz() == 1 ? 1 : 0; - } - - public double getWorkAccessibility() - { - return workAccessibility; - } - - public double getSchoolAccessibility() - { - return schoolAccessibility; - } - - public int getTelecommuteFrequency() { - return person.getTelecommuteChoice(); - } - - - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java deleted file mode 100644 index bd1e1db..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/IntermediateStopChoiceModels.java +++ /dev/null @@ -1,5075 +0,0 @@ -package org.sandag.abm.ctramp; - -import org.sandag.abm.ctramp.CtrampDmuFactoryIf; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Stop; -import org.sandag.abm.ctramp.StopLocationDMU; -import org.sandag.abm.ctramp.Tour; -import org.sandag.abm.ctramp.TripModeChoiceDMU; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Random; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * This class will be used for determining the trip departure time for outbound - * stops, trip arrival time for inbound stops, location of stops, and trip mode - * for trips between stops on individual mandatory, individual non-mandatory and - * joint tours. - * - * @author Jim Hicks - * @version Oct 2010 - */ -public class IntermediateStopChoiceModels - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(IntermediateStopChoiceModels.class); - private transient Logger slcLogger = Logger.getLogger("slcLogger"); - private transient Logger slcSoaLogger = Logger.getLogger("slcSoaLogger"); - private transient Logger smcLogger = Logger.getLogger("tripMcLog"); - private transient Logger tripDepartLogger = Logger.getLogger("tripDepartLog"); - private transient Logger parkLocLogger = Logger.getLogger("parkLocLog"); - - public static final int WTW = 0; - public static final int WTD = 1; - public static final int DTW = 2; - public static final int[] ACC_EGR = {WTW,WTD,DTW}; - private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "slc.use.new.soa"; - - public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; - - private static final String PROPERTIES_UEC_SLC_CHOICE = "slc.uec.file"; - private static final String PROPERTIES_UEC_SLC_DATA_PAGE = "slc.uec.data.page"; - private static final String PROPERTIES_UEC_MAND_SLC_MODEL_PAGE = "slc.mandatory.uec.model.page"; - private static final String PROPERTIES_UEC_MAINT_SLC_MODEL_PAGE = "slc.maintenance.uec.model.page"; - private static final String PROPERTIES_UEC_DISCR_SLC_MODEL_PAGE = "slc.discretionary.uec.model.page"; - - public static final String PROPERTIES_UEC_SLC_SOA_CHOICE = "slc.soa.uec.file"; - public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY = "auto.slc.soa.distance.uec.file"; - public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_DATA_PAGE = "auto.slc.soa.distance.data.page"; - public static final String PROPERTIES_UEC_SLC_SOA_DISTANCE_MODEL_PAGE = "auto.slc.soa.distance.model.page"; - - private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE = "slc.soa.size.uec.file"; - private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE_DATA = "slc.soa.size.uec.data.page"; - private static final String PROPERTIES_UEC_STOP_LOCATION_SIZE_MODEL = "slc.soa.size.uec.model.page"; - - private static final String PROPERTIES_UEC_PARKING_LOCATION_CHOICE = "plc.uec.file"; - private static final String PROPERTIES_UEC_PLC_DATA_PAGE = "plc.uec.data.page"; - private static final String PROPERTIES_UEC_PLC_MODEL_PAGE = "plc.uec.model.page"; - - private static final String PROPERTIES_UEC_PARKING_LOCATION_CHOICE_ALTERNATIVES = "plc.alts.corresp.file"; - - public static final int WORK_SHEET = 1; - public static final int UNIVERSITY_SHEET = 2; - public static final int SCHOOL_SHEET = 3; - public static final int MAINTENANCE_SHEET = 4; - public static final int DISCRETIONARY_SHEET = 5; - public static final int SUBTOUR_SHEET = 6; - public static final int[] MC_PURPOSE_SHEET_INDICES = { - -1, WORK_SHEET, UNIVERSITY_SHEET, SCHOOL_SHEET, MAINTENANCE_SHEET, MAINTENANCE_SHEET, - MAINTENANCE_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, DISCRETIONARY_SHEET, - SUBTOUR_SHEET }; - - public static final int WORK_CATEGORY = 0; - public static final int UNIVERSITY_CATEGORY = 1; - public static final int SCHOOL_CATEGORY = 2; - public static final int MAINTENANCE_CATEGORY = 3; - public static final int DISCRETIONARY_CATEGORY = 4; - public static final int SUBTOUR_CATEGORY = 5; - public static final String[] PURPOSE_CATEGORY_LABELS = { - "work", "university", "school", "maintenance", "discretionary", "subtour" }; - public static final int[] PURPOSE_CATEGORIES = { - -1, WORK_CATEGORY, UNIVERSITY_CATEGORY, SCHOOL_CATEGORY, MAINTENANCE_CATEGORY, - MAINTENANCE_CATEGORY, MAINTENANCE_CATEGORY, DISCRETIONARY_CATEGORY, - DISCRETIONARY_CATEGORY, DISCRETIONARY_CATEGORY, SUBTOUR_CATEGORY }; - - private static final String PARK_MGRA_COLUMN = "mgra"; - private static final String PARK_AREA_COLUMN = "parkarea"; - private static final int MAX_PLC_SAMPLE_SIZE = 1200; - - private static final int WORK_STOP_PURPOSE_INDEX = 1; - private static final int UNIV_STOP_PURPOSE_INDEX = 2; - private static final int ESCORT_STOP_PURPOSE_INDEX = 4; - private static final int SHOP_STOP_PURPOSE_INDEX = 5; - private static final int MAINT_STOP_PURPOSE_INDEX = 6; - private static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; - private static final int VISIT_STOP_PURPOSE_INDEX = 8; - private static final int DISCR_STOP_PURPOSE_INDEX = 9; - - private static final int OTHER_STOP_LOC_SOA_SHEET_INDEX = 2; - private static final int WALK_STOP_LOC_SOA_SHEET_INDEX = 3; - private static final int BIKE_STOP_LOC_SOA_SHEET_INDEX = 4; - private static final int MAX_STOP_LOC_SOA_SHEET_INDEX = 4; - - private static final int WORK_STOP_PURPOSE_SOA_SIZE_INDEX = 0; - private static final int UNIV_STOP_PURPOSE_SOA_SIZE_INDEX = 1; - private static final int ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX = 2; - private static final int ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX = 3; - private static final int ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 4; - private static final int ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 5; - private static final int ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX = 6; - private static final int ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 7; - private static final int ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 8; - private static final int ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX = 9; - private static final int SHOP_STOP_PURPOSE_SOA_SIZE_INDEX = 10; - private static final int MAINT_STOP_PURPOSE_SOA_SIZE_INDEX = 11; - private static final int EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX = 12; - private static final int VISIT_STOP_PURPOSE_SOA_SIZE_INDEX = 13; - private static final int DISCR_STOP_PURPOSE_SOA_SIZE_INDEX = 14; - - public static final int[] SLC_SIZE_SEGMENT_INDICES = { - WORK_STOP_PURPOSE_SOA_SIZE_INDEX, UNIV_STOP_PURPOSE_SOA_SIZE_INDEX, - ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX, - ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX, - ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, - ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX, - SHOP_STOP_PURPOSE_SOA_SIZE_INDEX, MAINT_STOP_PURPOSE_SOA_SIZE_INDEX, - EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX, VISIT_STOP_PURPOSE_SOA_SIZE_INDEX, - DISCR_STOP_PURPOSE_SOA_SIZE_INDEX }; - - public static final String[] SLC_SIZE_SEGMENT_NAMES = { - "work", "univ", "escort_0", "escort_ps", "escort_gs", "escort_hs", "escort_ps_gs", - "escort_ps_hs", "escort_gs_hs", "escort_ps_gs_hs", "shop", "maint", "eatout", "visit", - "discr" }; - - private static final int MAND_SLC_MODEL_INDEX = 0; - private static final int MAINT_SLC_MODEL_INDEX = 1; - private static final int DISCR_SLC_MODEL_INDEX = 2; - - private static final String PROPERTIES_TRIP_UTILITY_IVT_COEFFS = "trip.utility.ivt.coeffs"; - private static final String PROPERTIES_TRIP_UTILITY_INCOME_COEFFS = "trip.utility.income.coeffs"; - private static final String PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS = "trip.utility.income.exponents"; - - - private boolean[] sampleAvailability; - private int[] inSample; - private boolean[] soaAvailability; - private int[] soaSample; - private boolean[] soaAvailabilityBackup; - private int[] soaSampleBackup; - private int[] finalSample; - private double[] sampleCorrectionFactors; - private double[] tripModeChoiceLogsums; - private boolean[] sampleMgraInBoardingTapShed; - private boolean[] sampleMgraInAlightingTapShed; - private boolean earlierTripWasLocatedInAlightingTapShed; - - private double[] tourOrigToAllMgraDistances; - private double[] tourDestToAllMgraDistances; - private double[] ikDistance; - private double[] kjDistance; - private double[] okDistance; - private double[] kdDistance; - - private AutoAndNonMotorizedSkimsCalculator anm; - private McLogsumsCalculator logsumHelper; - private ModelStructure modelStructure; - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private int sampleSize; - private HashMap altFreqMap; - private StopLocationDMU stopLocDmuObj; - private TripModeChoiceDMU mcDmuObject; - private ParkingChoiceDMU parkingChoiceDmuObj; - - private double[][] slcSizeTerms; - private int[][] slcSizeSample; - private boolean[][] slcSizeAvailable; - - private double[] distanceFromStopOrigToAllMgras; - private double[] distanceToFinalDestFromAllMgras; - - private final BikeLogsum bls; - private BikeLogsumSegment segment; - - private double[] bikeLogsumFromStopOrigToAllMgras; - private double[] bikeLogsumToFinalDestFromAllMgras; - - private double[][] mcCumProbsSegmentIk; - private double[][] mcCumProbsSegmentKj; - - private double[] mcLogsumsSegmentIk; - private double[] mcLogsumsSegmentKj; - - private double[][] mcVOTsSegmentIk; //by sample and occupancy (0=non-SR,1=S2,2=S3) - private double[][] mcVOTsSegmentKj; - - private float[] parkingCostSegmentIk; //by sample - private float[] parkingCostSegmentKj; - - private double[][][][] segmentIkBestTapPairs; - private double[][][][] segmentKjBestTapPairs; - - - private ChoiceModelApplication[] mcModelArray; - private ChoiceModelApplication[] slcSoaModel; - private ChoiceModelApplication[] slcModelArray; - private ChoiceModelApplication plcModel; - - private int[] altMgraIndices; - private double[] altOsDistances; - private double[] altSdDistances; - private boolean[] altParkAvail; - private int[] altParkSample; - - private int numAltsInSample; - private int maxAltsInSample; - - private TableDataSet plcAltsTable; - private HashMap mgraAltLocationIndex; - private HashMap mgraAltParkArea; - private int[] parkMgras; - private int[] parkAreas; - - private int[] mgraParkArea; - private int[] numfreehrs; - private int[] hstallsoth; - private int[] hstallssam; - private float[] hparkcost; - private int[] dstallsoth; - private int[] dstallssam; - private float[] dparkcost; - private int[] mstallsoth; - private int[] mstallssam; - private float[] mparkcost; - - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - - private double[] altParkingCostsM; - private double[] altParkingCostsD; - private double[] altParkingCostsH; - private int[] altMstallsoth; - private int[] altMstallssam; - private float[] altMparkcost; - private int[] altDstallsoth; - private int[] altDstallssam; - private float[] altDparkcost; - private int[] altHstallsoth; - private int[] altHstallssam; - private float[] altHparkcost; - private int[] altNumfreehrs; - - private HashMap sizeSegmentNameIndexMap; - - private StopDepartArrivePeriodModel stopTodModel; - - private int availAltsToLog = 55; - // this - private TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator; - // logging - // private int availAltsToLog = 5; - - // set this constant for checking the number of times depart/arrive period - // selection is made so that no infinite loop occurs. - private static final int MAX_INVALID_FIRST_ARRIVAL_COUNT = 100; - - public static final int NUM_CPU_TIME_VALUES = 9; - private long soaAutoTime; - private long soaOtherTime; - private long slsTime; - private long sldTime; - private long slcTime; - private long todTime; - private long smcTime; - private long[] hhTimes = new long[NUM_CPU_TIME_VALUES]; - - private DestChoiceTwoStageModel dcTwoStageModelObject; - - private boolean useNewSoaMethod; - - private String loggerSeparator = ""; - - // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose - double[] ivtCoeffs; - double[] incomeCoeffs; - double[] incomeExponents; - - /** - * Constructor that will be used to set up the ChoiceModelApplications for - * each type of tour - * - * @param projectDirectory - * - name of root level project directory - * @param resourceBundle - * - properties file with paths identified - * @param dmuObject - * - decision making unit for stop frequency - * @param modelStructure - * - holds the model structure info - */ - public IntermediateStopChoiceModels(HashMap propertyMap, - ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory, - McLogsumsCalculator myLogsumHelper) - { - - tazs = TazDataManager.getInstance(propertyMap); - mgraManager = MgraDataManager.getInstance(propertyMap); - - mgraParkArea = mgraManager.getMgraParkAreas(); - numfreehrs = mgraManager.getNumFreeHours(); - lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); - lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); - lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); - mstallsoth = mgraManager.getMStallsOth(); - mstallssam = mgraManager.getMStallsSam(); - mparkcost = mgraManager.getMParkCost(); - dstallsoth = mgraManager.getDStallsOth(); - dstallssam = mgraManager.getDStallsSam(); - dparkcost = mgraManager.getDParkCost(); - hstallsoth = mgraManager.getHStallsOth(); - hstallssam = mgraManager.getHStallsSam(); - hparkcost = mgraManager.getHParkCost(); - - modelStructure = myModelStructure; - logsumHelper = myLogsumHelper; - - setupStopLocationChoiceModels(propertyMap, dmuFactory); - setupParkingLocationModel(propertyMap, dmuFactory); - - bls = BikeLogsum.getBikeLogsum(propertyMap); - - //get the coefficients for ivt and the coefficients to calculate the cost coefficient - ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_IVT_COEFFS); - incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_COEFFS); - incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS); - - tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); - tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); - - - } - - private void setupStopLocationChoiceModels(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up stop location choice models.")); - - useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, - USE_NEW_SOA_METHOD_PROPERTY_KEY); - - stopLocDmuObj = dmuFactory.getStopLocationDMU(); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String slcSoaUecFile = propertyMap.get(PROPERTIES_UEC_SLC_SOA_CHOICE); - slcSoaUecFile = uecPath + slcSoaUecFile; - - String slcUecFile = propertyMap.get(PROPERTIES_UEC_SLC_CHOICE); - slcUecFile = uecPath + slcUecFile; - - slcSoaModel = new ChoiceModelApplication[MAX_STOP_LOC_SOA_SHEET_INDEX + 1]; - slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, - OTHER_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); - slcSoaModel[WALK_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, - WALK_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); - slcSoaModel[BIKE_STOP_LOC_SOA_SHEET_INDEX] = new ChoiceModelApplication(slcSoaUecFile, - BIKE_STOP_LOC_SOA_SHEET_INDEX, 0, propertyMap, (VariableTable) stopLocDmuObj); - - int numSlcSoaAlternatives = slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX] - .getNumberOfAlternatives(); - stopLocDmuObj = dmuFactory.getStopLocationDMU(); - - sizeSegmentNameIndexMap = new HashMap(); - for (int k = 0; k < SLC_SIZE_SEGMENT_INDICES.length; k++) - { - sizeSegmentNameIndexMap.put(SLC_SIZE_SEGMENT_NAMES[k], k); - sizeSegmentNameIndexMap.put(SLC_SIZE_SEGMENT_NAMES[k], SLC_SIZE_SEGMENT_INDICES[k]); - } - - // set the second argument to a positive, non-zero mgra value to get - // logging for the size term calculation for the specified mgra. - slcSizeTerms = calculateLnSlcSizeTerms(propertyMap, -1); - - String mcUecFile = propertyMap.get(PROPERTIES_UEC_TRIP_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - mcDmuObject = dmuFactory.getTripModeChoiceDMU(); - - // logsumHelper.setupSkimCalculators(propertyMap); - anm = logsumHelper.getAnmSkimCalculator(); - mcDmuObject.setParkingCostInfo(mgraParkArea, lsWgtAvgCostM, lsWgtAvgCostD, lsWgtAvgCostH); - - mcModelArray = new ChoiceModelApplication[5 + 1]; - mcModelArray[WORK_CATEGORY] = new ChoiceModelApplication(mcUecFile, WORK_SHEET, 0, - propertyMap, (VariableTable) mcDmuObject); - mcModelArray[UNIVERSITY_CATEGORY] = new ChoiceModelApplication(mcUecFile, UNIVERSITY_SHEET, - 0, propertyMap, (VariableTable) mcDmuObject); - mcModelArray[SCHOOL_CATEGORY] = new ChoiceModelApplication(mcUecFile, SCHOOL_SHEET, 0, - propertyMap, (VariableTable) mcDmuObject); - mcModelArray[MAINTENANCE_CATEGORY] = new ChoiceModelApplication(mcUecFile, - MAINTENANCE_SHEET, 0, propertyMap, (VariableTable) mcDmuObject); - mcModelArray[DISCRETIONARY_CATEGORY] = new ChoiceModelApplication(mcUecFile, - DISCRETIONARY_SHEET, 0, propertyMap, (VariableTable) mcDmuObject); - mcModelArray[SUBTOUR_CATEGORY] = new ChoiceModelApplication(mcUecFile, SUBTOUR_SHEET, 0, - propertyMap, (VariableTable) mcDmuObject); - - // set up the stop location choice model object - int slcDataPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_SLC_DATA_PAGE)); - int slcMandModelPage = Integer - .parseInt(propertyMap.get(PROPERTIES_UEC_MAND_SLC_MODEL_PAGE)); - int slcMaintModelPage = Integer.parseInt(propertyMap - .get(PROPERTIES_UEC_MAINT_SLC_MODEL_PAGE)); - int slcDiscrModelPage = Integer.parseInt(propertyMap - .get(PROPERTIES_UEC_DISCR_SLC_MODEL_PAGE)); - slcModelArray = new ChoiceModelApplication[3]; - slcModelArray[MAND_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, - slcMandModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); - slcModelArray[MAINT_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, - slcMaintModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); - slcModelArray[DISCR_SLC_MODEL_INDEX] = new ChoiceModelApplication(slcUecFile, - slcDiscrModelPage, slcDataPage, propertyMap, (VariableTable) stopLocDmuObj); - - sampleSize = slcModelArray[MAND_SLC_MODEL_INDEX].getNumberOfAlternatives(); - altFreqMap = new HashMap(sampleSize); - finalSample = new int[sampleSize + 1]; - sampleCorrectionFactors = new double[sampleSize + 1]; - - // decalre the arrays for storing stop location choice ik and kj segment - // mode choice probability arrays - mcCumProbsSegmentIk = new double[sampleSize + 1][]; - mcCumProbsSegmentKj = new double[sampleSize + 1][]; - - //declare the arrays for storing the VOTs calculated by trip mode choice for ik and kj segment - mcVOTsSegmentIk = new double[sampleSize+1][3]; - mcVOTsSegmentKj = new double[sampleSize+1][3]; - - //declare arrays for storing the trip mode choice parking cost for ik and kj segment - parkingCostSegmentIk = new float[sampleSize+1]; - parkingCostSegmentKj = new float[sampleSize+1]; - - //declare the arrays for storing the stop location choice ik and kj segment - //mode choice logsum arrays - mcLogsumsSegmentIk = new double[sampleSize + 1]; - mcLogsumsSegmentKj = new double[sampleSize + 1]; - - // declare the arrays for storing stop location choice ik and kj segment best tap pair arrays - segmentIkBestTapPairs = new double[sampleSize+1][ACC_EGR.length][][]; - segmentKjBestTapPairs = new double[sampleSize+1][ACC_EGR.length][][]; - - // declare the arrays that holds ik and kj segment logsum values for - // each location choice sample alternative - tripModeChoiceLogsums = new double[sampleSize + 1]; - - // declare the arrays that holds ik and kj distance values for each - // location choice sample alternative - // these are set as stop location dmu attributes for stop orig to stop - // alt, and stop alt to half-tour final dest. - ikDistance = new double[sampleSize + 1]; - kjDistance = new double[sampleSize + 1]; - - // declare the arrays that holds ik and kj distance values for each - // location choice sample alternative - // these are set as stop location dmu attributes for tour orig to stop - // alt, and stop alt to tour dest. - okDistance = new double[sampleSize + 1]; - kdDistance = new double[sampleSize + 1]; - - // this array has elements with values of 0 if utility is not to be - // computed for alternative, or 1 if utility is to be computed. - inSample = new int[sampleSize + 1]; - - // this array has elements that are boolean that set availability of - // alternative - unavailable altrnatives do not get utility computed. - sampleAvailability = new boolean[sampleSize + 1]; - - soaSample = new int[numSlcSoaAlternatives + 1]; - soaAvailability = new boolean[numSlcSoaAlternatives + 1]; - soaSampleBackup = new int[numSlcSoaAlternatives + 1]; - soaAvailabilityBackup = new boolean[numSlcSoaAlternatives + 1]; - - sampleMgraInBoardingTapShed = new boolean[mgraManager.getMaxMgra() + 1]; - sampleMgraInAlightingTapShed = new boolean[mgraManager.getMaxMgra() + 1]; - - distanceFromStopOrigToAllMgras = new double[mgraManager.getMaxMgra() + 1]; - distanceToFinalDestFromAllMgras = new double[mgraManager.getMaxMgra() + 1]; - - bikeLogsumFromStopOrigToAllMgras = new double[mgraManager.getMaxMgra() + 1]; - bikeLogsumToFinalDestFromAllMgras = new double[mgraManager.getMaxMgra() + 1]; - - tourOrigToAllMgraDistances = new double[mgraManager.getMaxMgra() + 1]; - tourDestToAllMgraDistances = new double[mgraManager.getMaxMgra() + 1]; - - // create the array of 1s for MGRAs that have a non-empty set of TAPs - // within walk egress distance of them - // for the setting walk transit available for teh stop location - // alternatives. - // createWalkTransitAvailableArray(); - - setupTripDepartTimeModel(propertyMap, dmuFactory); - - loggerSeparator += "-"; - - } - - public void setupSlcDistanceBaseSoaModel(HashMap propertyMap, - double[][] soaExpUtils, double[][][] soaSizeProbs, double[][] soaTazSize) - { - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String slcSoaDistanceUecFile = propertyMap.get(PROPERTIES_UEC_SLC_SOA_DISTANCE_UTILITY); - slcSoaDistanceUecFile = uecPath + slcSoaDistanceUecFile; - - dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, sampleSize); - dcTwoStageModelObject.setSlcSoaProbsAndUtils(soaExpUtils, soaSizeProbs, soaTazSize); - - } - - private void setupTripDepartTimeModel(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - stopTodModel = new StopDepartArrivePeriodModel(propertyMap, modelStructure); - } - - private void setupParkingLocationModel(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - logger.info("setting up parking location choice models."); - - // locate the UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String plcUecFile = propertyMap.get(PROPERTIES_UEC_PARKING_LOCATION_CHOICE); - plcUecFile = uecPath + plcUecFile; - - int plcDataPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_PLC_DATA_PAGE)); - int plcModelPage = Integer.parseInt(propertyMap.get(PROPERTIES_UEC_PLC_MODEL_PAGE)); - - altMgraIndices = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altOsDistances = new double[MAX_PLC_SAMPLE_SIZE + 1]; - altSdDistances = new double[MAX_PLC_SAMPLE_SIZE + 1]; - altParkingCostsM = new double[MAX_PLC_SAMPLE_SIZE + 1]; - altParkingCostsD = new double[MAX_PLC_SAMPLE_SIZE + 1]; - altParkingCostsH = new double[MAX_PLC_SAMPLE_SIZE + 1]; - altMstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altMstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altMparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; - altDstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altDstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altDparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; - altHstallsoth = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altHstallssam = new int[MAX_PLC_SAMPLE_SIZE + 1]; - altHparkcost = new float[MAX_PLC_SAMPLE_SIZE + 1]; - altNumfreehrs = new int[MAX_PLC_SAMPLE_SIZE + 1]; - - altParkAvail = new boolean[MAX_PLC_SAMPLE_SIZE + 1]; - altParkSample = new int[MAX_PLC_SAMPLE_SIZE + 1]; - - parkingChoiceDmuObj = dmuFactory.getParkingChoiceDMU(); - - plcModel = new ChoiceModelApplication(plcUecFile, plcModelPage, plcDataPage, propertyMap, - (VariableTable) parkingChoiceDmuObj); - - // read the parking choice alternatives data file to get alternatives - // names - String plcAltsFile = propertyMap.get(PROPERTIES_UEC_PARKING_LOCATION_CHOICE_ALTERNATIVES); - plcAltsFile = uecPath + plcAltsFile; - - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - plcAltsTable = reader.readFile(new File(plcAltsFile)); - } catch (IOException e) - { - logger.error("problem reading table of cbd zones for parking location choice model.", e); - System.exit(1); - } - - parkMgras = plcAltsTable.getColumnAsInt(PARK_MGRA_COLUMN); - parkAreas = plcAltsTable.getColumnAsInt(PARK_AREA_COLUMN); - - parkingChoiceDmuObj.setParkAreaMgraArray(parkMgras); - parkingChoiceDmuObj.setSampleIndicesArray(altMgraIndices); - parkingChoiceDmuObj.setDistancesOrigAlt(altOsDistances); - parkingChoiceDmuObj.setDistancesAltDest(altSdDistances); - parkingChoiceDmuObj.setParkingCostsM(altParkingCostsM); - parkingChoiceDmuObj.setMstallsoth(altMstallsoth); - parkingChoiceDmuObj.setMstallssam(altMstallssam); - parkingChoiceDmuObj.setMparkCost(altMparkcost); - parkingChoiceDmuObj.setDstallsoth(altDstallsoth); - parkingChoiceDmuObj.setDstallssam(altDstallssam); - parkingChoiceDmuObj.setDparkCost(altDparkcost); - parkingChoiceDmuObj.setHstallsoth(altHstallsoth); - parkingChoiceDmuObj.setHstallssam(altHstallssam); - parkingChoiceDmuObj.setHparkCost(altHparkcost); - parkingChoiceDmuObj.setNumfreehrs(altNumfreehrs); - - mgraAltLocationIndex = new HashMap(); - mgraAltParkArea = new HashMap(); - - for (int i = 0; i < parkMgras.length; i++) - { - mgraAltLocationIndex.put(parkMgras[i], i); - mgraAltParkArea.put(parkMgras[i], parkAreas[i]); - } - - } - - private double[][] calculateLnSlcSizeTerms(HashMap rbMap, int logMgra) - { - - logger.info("calculating Stop Location SOA Size Terms"); - - String uecPath = rbMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String slcSizeUecFile = rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE); - slcSizeUecFile = uecPath + slcSizeUecFile; - int slcSizeUecData = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE_DATA)); - int slcSizeUecModel = Integer.parseInt(rbMap.get(PROPERTIES_UEC_STOP_LOCATION_SIZE_MODEL)); - - IndexValues iv = new IndexValues(); - UtilityExpressionCalculator slcSizeUec = new UtilityExpressionCalculator(new File( - slcSizeUecFile), slcSizeUecModel, slcSizeUecData, rbMap, null); - - ArrayList mgras = mgraManager.getMgras(); - int maxMgra = mgraManager.getMaxMgra(); - int numSizeSegments = slcSizeUec.getNumberOfAlternatives(); - - // create the array for storing logged size term values - to be returned - // by this method - double[][] lnSlcSoaSize = new double[numSizeSegments][maxMgra + 1]; - slcSizeSample = new int[numSizeSegments][maxMgra + 1]; - slcSizeAvailable = new boolean[numSizeSegments][maxMgra + 1]; - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - iv.setZoneIndex(mgra); - double[] size = slcSizeUec.solve(iv, null, null); - - // if a logMgra values > 0 was specified, log the size term utility - // calculation for that mgra - if (mgra == logMgra) - slcSizeUec.logAnswersArray(slcSoaLogger, "Stop Location SOA Size Terms, MGRA = " - + mgra); - - // store the logged size terms - for (int i = 0; i < numSizeSegments; i++) - { - lnSlcSoaSize[i][mgra] = Math.log(size[i] + 1); - if (size[i] > 0) - { - slcSizeSample[i][mgra] = 1; - slcSizeAvailable[i][mgra] = true; - } - } - - } - - return lnSlcSoaSize; - - } - - private double[] getLnSlcSizeTermsForStopPurpose(int stopPurpose, Household hh) - { - - double[] lnSlcSizeTerms = null; - - switch (stopPurpose) - { - - case WORK_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[WORK_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case UNIV_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[UNIV_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case ESCORT_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = getLnSlcSizeTermsForEscortStopPurpose(hh); - break; - case SHOP_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[SHOP_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case MAINT_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[MAINT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case EAT_OUT_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[EAT_OUT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case VISIT_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[VISIT_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - case DISCR_STOP_PURPOSE_INDEX: - lnSlcSizeTerms = slcSizeTerms[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[DISCR_STOP_PURPOSE_SOA_SIZE_INDEX]; - break; - } - - // save backup arrays with oroginal availability and sample values. - // the procedure to generate availabilty for transit tours overwrites - // the arrays used by the UECs, - // so they need to be restored after that happens. - for (int i = 0; i < soaSample.length; i++) - { - soaSampleBackup[i] = soaSample[i]; - soaAvailabilityBackup[i] = soaAvailability[i]; - } - - return lnSlcSizeTerms; - - } - - private double[] getLnSlcSizeTermsForEscortStopPurpose(Household hh) - { - - double[] lnSlcSizeTermsForEscort = null; - - // set booleans for presence of preschool, grade school, and high school - // students in the hh - boolean psInHh = hh.getNumPreschool() > 0; - boolean gsInHh = hh.getNumGradeSchoolStudents() > 0; - boolean hsInHh = hh.getNumHighSchoolStudents() > 0; - - if (!psInHh && !gsInHh && !hsInHh) - { - // if hh has no preschool, grade school or high school children, set - // the array to that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_0_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (psInHh && !gsInHh && !hsInHh) - { - // if hh has a preschool child and no gs or hs, set the array to - // that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_PS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (!psInHh && gsInHh && !hsInHh) - { - // if hh has a grade school child and no ps or hs, set the array to - // that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (!psInHh && !gsInHh && hsInHh) - { - // if hh has a high school child and no ps or gs, set the array to - // that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (psInHh && gsInHh && !hsInHh) - { - // if hh has a preschool and a grade school child and no hs, set the - // array to that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_PS_GS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (psInHh && !gsInHh && hsInHh) - { - // if hh has a preschool and a high school child and no gs, set the - // array to that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_PS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (!psInHh && gsInHh && hsInHh) - { - // if hh has a grade school and a high school child and no ps, set - // the array to that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } else if (psInHh && gsInHh && hsInHh) - { - // if hh has a preschool a grade school and a high school child, set - // the array to that specific size term field - lnSlcSizeTermsForEscort = slcSizeTerms[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaSample = slcSizeSample[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - soaAvailability = slcSizeAvailable[ESCORT_PS_GS_HS_STOP_PURPOSE_SOA_SIZE_INDEX]; - } - - return lnSlcSizeTermsForEscort; - } - - public void applyModel(Household household, boolean withTiming) - { - - if (withTiming) zeroOutCpuTimes(); - - if (household.getDebugChoiceModels()) - { - slcLogger.info("applying SLC model for hhid=" + household.getHhId()); - } - - // get this household's person array - Person[] personArray = household.getPersons(); - - // loop through the person array (1-based) - for (int j = 1; j < personArray.length; ++j) - { - - ArrayList tours = new ArrayList(); - - Person person = personArray[j]; - - // apply stop location and mode choice for all individual tours. - tours.addAll(person.getListOfWorkTours()); - tours.addAll(person.getListOfSchoolTours()); - tours.addAll(person.getListOfIndividualNonMandatoryTours()); - tours.addAll(person.getListOfAtWorkSubtours()); - - for (Tour tour : tours) - { - - if (withTiming) applyForOutboundStopsWithTiming(tour, person, household); - else applyForOutboundStops(tour, person, household); - - if (withTiming) applyForInboundStopsWithTiming(tour, person, household); - else applyForInboundStops(tour, person, household); - - } // tour loop - - } // j (person loop) - - // apply stop location and mode choice for all joint tours. - Tour[] jointTours = household.getJointTourArray(); - if (jointTours != null) - { - - for (Tour tour : jointTours) - { - - if (withTiming) applyForOutboundStopsWithTiming(tour, null, household); - else applyForOutboundStops(tour, null, household); - - if (withTiming) applyForInboundStopsWithTiming(tour, null, household); - else applyForInboundStops(tour, null, household); - - } // tour loop - - } - - household.setStlRandomCount(household.getHhRandomCount()); - - } - - private void applyForOutboundStops(Tour tour, Person person, Household household) - { - - //don't apply if the outbound direction is escort - if((tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE)) - return; - - Stop[] stops = tour.getOutboundStops(); - - // select trip depart periods for outbound stops - if (stops != null) - { - setOutboundTripDepartTimes(stops); - } - - int origMgra = tour.getTourOrigMgra(); - int destMgra = tour.getTourDestMgra(); - - applySlcModel(household, person, tour, stops, origMgra, destMgra, false); - - } - - private void applyForOutboundStopsWithTiming(Tour tour, Person person, Household household) - { - - //don't apply if the outbound direction is escort - if((tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE)) - return; - - long check = System.nanoTime(); - - Stop[] stops = tour.getOutboundStops(); - - // select trip depart periods for outbound stops - if (stops != null) - { - setOutboundTripDepartTimes(stops); - } - - int origMgra = tour.getTourOrigMgra(); - int destMgra = tour.getTourDestMgra(); - - todTime += (System.nanoTime() - check); - - applySlcModelWithTiming(household, person, tour, stops, origMgra, destMgra, false); - - } - - private void applyForInboundStops(Tour tour, Person person, Household household) - { - - //don't apply if the inbound direction is escort - if((tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE)) - return; - - Stop[] stops = tour.getInboundStops(); - - // select trip arrive periods for inbound stops - if (stops != null) - { - int lastOutboundTripDeparts = -1; - - // get the outbound stops array - note, if there were no outbound - // stops for half-tour, one stop object would have been generated - // to hold information about the half-tour trip, so this array - // should never be null. - Stop[] obStops = tour.getOutboundStops(); - if (obStops == null) - { - logger.error("error getting last outbound stop object for setting lastOutboundTripDeparts attribute for inbound stop arrival time choice."); - logger.error("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); - throw new RuntimeException(); - } else - { - Stop lastStop = obStops[obStops.length - 1]; - lastOutboundTripDeparts = lastStop.getStopPeriod(); - } - - setInboundTripDepartTimes(stops, lastOutboundTripDeparts); - } - - int origMgra = tour.getTourDestMgra(); - int destMgra = tour.getTourOrigMgra(); - - applySlcModel(household, person, tour, stops, origMgra, destMgra, true); - - } - - private void applyForInboundStopsWithTiming(Tour tour, Person person, Household household) - { - - //don't apply if the inbound direction is escort - if((tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE)||(tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE)) - return; - - long check = System.nanoTime(); - - Stop[] stops = tour.getInboundStops(); - - // select trip arrive periods for inbound stops - if (stops != null) - { - int lastOutboundTripDeparts = -1; - - // get the outbound stops array - note, if there were no outbound - // stops for half-tour, one stop object would have been generated - // to hold information about the half-tour trip, so this array - // should never be null. - Stop[] obStops = tour.getOutboundStops(); - if (obStops == null) - { - logger.error("error getting last outbound stop object for setting lastOutboundTripDeparts attribute for inbound stop arrival time choice."); - logger.error("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); - throw new RuntimeException(); - } else - { - Stop lastStop = obStops[obStops.length - 1]; - lastOutboundTripDeparts = lastStop.getStopPeriod(); - } - - setInboundTripDepartTimes(stops, lastOutboundTripDeparts); - } - - int origMgra = tour.getTourDestMgra(); - int destMgra = tour.getTourOrigMgra(); - - todTime += (System.nanoTime() - check); - - applySlcModelWithTiming(household, person, tour, stops, origMgra, destMgra, true); - - } - - private void applySlcModel(Household household, Person person, Tour tour, Stop[] stops, - int origMgra, int destMgra, boolean directionIsInbound) - { - - int lastDest = -1; - int newOrig = -1; - - // get the array of distances from the tour origin mgra to all MGRAs. - // use these distances for tour orig to stop alt distances - anm.getDistancesFromMgra(origMgra, tourOrigToAllMgraDistances, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - anm.getDistancesFromMgra(destMgra, tourDestToAllMgraDistances, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - // anm.getDistancesToMgra( destMgra, tourDestToAllMgraDistances, - // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) ); - - // if there are stops on this half-tour, determine their destinations, - // depart hours, trip modes, and parking tazs. - if (stops != null) - { - - int oldSelectedIndex = -1; - for (int i = 0; i < stops.length; i++) - { - - Stop stop = stops[i]; - - // if i is 0, the stop origin is set to origMgra; otherwise stop - // orig is the chosen dest from the previous stop. - if (i == 0){ - newOrig = origMgra; - } - else{ - newOrig = lastDest; - } - stop.setOrig(newOrig); - - stopLocDmuObj.setStopObject(stop); - stopLocDmuObj.setDmuIndexValues(household.getHhId(), household.getHhMgra(), - newOrig, destMgra); - - int choice = -1; - int selectedIndex = -1; - int modeAlt = -1; - float modeLogsum = 0; - double vot = -1.0; - // if not the last stop object, make a destination choice and a - // mode choice from IK MC probabilities; - // otherwise stop dest is set to destMgra, and make a mode - // choice from KJ MC probabilities. - if (i < stops.length - 1) - { - - //new code - depart period to and from stop - int departPeriodToStop = stop.getStopPeriod(); - int departPeriodFromStop = -1; - if(!directionIsInbound) - departPeriodFromStop = tour.getOutboundStops()[i+1].getStopPeriod(); - else - departPeriodFromStop = tour.getInboundStops()[i+1].getStopPeriod(); - - try - { - - selectedIndex = selectDestination(stop, departPeriodToStop, departPeriodFromStop); - choice = finalSample[selectedIndex]; - stop.setDest(choice); - lastDest = choice; - - if (household.getDebugChoiceModels()) - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" - + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - } - - modeAlt = selectModeFromProbabilities(stop, - mcCumProbsSegmentIk[selectedIndex]); - modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; - - if (modeAlt < 0) - { - logger.info("error getting trip mode choice for IK proportions, i=" + i); - logger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - logger.info("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" - + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - throw new RuntimeException(); - } - - //value of time; lookup vot, votS2, or votS3 depending on chosen mode - if(modelStructure.getTripModeIsS2(modeAlt)){ - vot = mcVOTsSegmentIk[selectedIndex][1]; - }else if (modelStructure.getTripModeIsS3(modeAlt)){ - vot = mcVOTsSegmentIk[selectedIndex][2]; - }else{ - vot = mcVOTsSegmentIk[selectedIndex][0]; - } - - int park = -1; - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - if (park > 0) lastDest = park; - float parkingCost = parkingCostSegmentIk[selectedIndex]; - stop.setParkingCost(parkingCost); - - } - - } catch (Exception e) - { - logger.error(String - .format("Exception caught processing %s stop location choice model for %s type tour %s stop: HHID=%d, personNum=%s, stop=%d.", - (stopLocDmuObj.getInboundStop() == 1 ? "inbound" - : "outbound"), - tour.getTourCategory(), - tour.getTourPurpose(), - household.getHhId(), - (person == null ? "N/A" : Integer.toString(person - .getPersonNum())), (i + 1))); - throw new RuntimeException(e); - } - - stop.setMode(modeAlt); - stop.setModeLogsum(modeLogsum); - stop.setValueOfTime(vot); - - // if the trip is a transit mode, set the boarding and - // alighting tap pairs in the stop object based on the ik - // segment pairs - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - - int accEgr = -1; - if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { - accEgr = WTW; - } else { - if (stop.isInboundStop()) { - accEgr = WTD; - } else { - accEgr = DTW; - } - } - - if ( segmentIkBestTapPairs[selectedIndex][accEgr] == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - //pick transit path from N-paths - double rn = household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, segmentIkBestTapPairs[selectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][0] ); - stop.setAlightTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][1] ); - stop.setSet( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][2] ); - - } - - } - - oldSelectedIndex = selectedIndex; - - } else - { - // last stop on half-tour, so dest is the half-tour final - // dest, and oldSelectedIndex was - // the selectedIndex determined for the previous stop - // location choice. - stop.setDest(destMgra); - - if (household.getDebugChoiceModels()) - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=End of " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " half-tour, stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" - + stop.getOrig() + ", stopDest=" + stop.getDest()); - } - - modeAlt = selectModeFromProbabilities(stop, - mcCumProbsSegmentKj[oldSelectedIndex]); - modeLogsum = (float) mcLogsumsSegmentKj[oldSelectedIndex]; - - if (modeAlt < 0) - { - logger.error("error getting trip mode choice for KJ proportions, i=" + i); - logger.error("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - logger.error("StopID=End of " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " half-tour, stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" - + stop.getOrig() + ", stopDest=" + stop.getDest()); - throw new RuntimeException(); - } - - //value of time; lookup vot, votS2, or votS3 depending on chosen mode - if(modelStructure.getTripModeIsS2(modeAlt)){ - vot = mcVOTsSegmentKj[oldSelectedIndex][1]; - }else if (modelStructure.getTripModeIsS3(modeAlt)){ - vot = mcVOTsSegmentKj[oldSelectedIndex][2]; - }else{ - vot = mcVOTsSegmentKj[oldSelectedIndex][0]; - } - - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) { - float parkingCost = parkingCostSegmentKj[oldSelectedIndex]; - stop.setParkingCost(parkingCost); - } - // last stop on tour, so if inbound, only need park location - // choice if tour is work-based subtour; - // otherwise dest is home. - int park = -1; - if (directionIsInbound) - { - if (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - } else - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - - stop.setMode(modeAlt); - stop.setModeLogsum(modeLogsum); - stop.setValueOfTime(vot); - - // if the last trip is a transit mode, set the boarding and - // alighting tap pairs in the stop object based on the kj - // segment pairs - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - - int accEgr = -1; - if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { - accEgr = WTW; - } else { - if (stop.isInboundStop()) { - accEgr = WTD; - } else { - accEgr = DTW; - } - } - - if ( segmentKjBestTapPairs[oldSelectedIndex][accEgr] == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - - //pick transit path from N-paths - float rn = (float)household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, segmentKjBestTapPairs[oldSelectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][0] ); - stop.setAlightTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][1] ); - stop.setSet( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][2] ); - } - - } - - } - - } - - } else - { - // create a stop object to hold attributes for orig, dest, mode, - // departtime, etc. - // for the half-tour with no stops. - - // create a Stop object for use in applying trip mode choice for - // this half tour without stops - String origStopPurpose = ""; - String destStopPurpose = ""; - if (!directionIsInbound) - { - origStopPurpose = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - destStopPurpose = tour.getTourPrimaryPurpose(); - } else - { - origStopPurpose = tour.getTourPrimaryPurpose(); - destStopPurpose = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - } - - Stop stop = null; - try - { - stop = tour.createStop(origStopPurpose, destStopPurpose, - directionIsInbound, - tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)); - } catch (Exception e) - { - logger.info("exception creating stop."); - throw new RuntimeException(e); - } - - // set stop origin and destination mgra, the stop period based on - // direction, then calculate the half-tour trip mode choice - stop.setOrig(origMgra); - stop.setDest(destMgra); - - if (directionIsInbound) stop.setStopPeriod(tour.getTourArrivePeriod()); - else stop.setStopPeriod(tour.getTourDepartPeriod()); - - int modeAlt = getHalfTourModeChoice(stop); - if (modeAlt < 0) - { - logger.info("error getting trip mode choice for half-tour with no stops."); - logger.info("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId()); - logger.info("StopID=" + (stop.getStopId() + 1) + " of no stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - throw new RuntimeException(); - } - - stop.setMode(modeAlt); - - double[][] bestTaps = null; - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) ) { - if ( directionIsInbound ) - bestTaps = tour.getBestWtwTapPairsIn(); - else - bestTaps = tour.getBestWtwTapPairsOut(); - } - else if ( modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - if ( directionIsInbound ) - bestTaps = tour.getBestWtdTapPairsIn(); - else - bestTaps = tour.getBestDtwTapPairsOut(); - } - - if ( bestTaps == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - //pick transit path from N-paths - float rn = (float)household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, bestTaps, household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)bestTaps[pathindex][0] ); - stop.setAlightTap( (int)bestTaps[pathindex][1] ); - stop.setSet( (int)bestTaps[pathindex][2] ); - } - - // inbound half-tour, only need park location choice if tour is - // work-based subtour; - // otherwise dest is home. - int park = -1; - if (directionIsInbound) - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - } else - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - - } - - } - - private void applySlcModelWithTiming(Household household, Person person, Tour tour, - Stop[] stops, int origMgra, int destMgra, boolean directionIsInbound) - { - - int lastDest = -1; - int newOrig = -1; - - // get the array of distances from the tour origin mgra to all MGRAs. - // use these distances for tour orig to stop alt distances - long check = System.nanoTime(); - anm.getDistancesFromMgra(origMgra, tourOrigToAllMgraDistances, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - anm.getDistancesFromMgra(destMgra, tourDestToAllMgraDistances, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - // anm.getDistancesToMgra( destMgra, tourDestToAllMgraDistances, - // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) ); - sldTime += (System.nanoTime() - check); - - // if there are stops on this half-tour, determine their destinations, - // depart hours, trip modes, and parking tazs. - if (stops != null) - { - - int oldSelectedIndex = -1; - earlierTripWasLocatedInAlightingTapShed = false; - - for (int i = 0; i < stops.length; i++) - { - - Stop stop = stops[i]; - - // if i is 0, the stop origin is set to origMgra; otherwise stop - // orig is the chosen dest from the previous stop. - if (i == 0) newOrig = origMgra; - else newOrig = lastDest; - stop.setOrig(newOrig); - - stopLocDmuObj.setStopObject(stop); - stopLocDmuObj.setDmuIndexValues(household.getHhId(), household.getHhMgra(), - newOrig, destMgra); - - int choice = -1; - int selectedIndex = -1; - int modeAlt = -1; - float modeLogsum = 0; - double vot= -1; - - // if not the last stop object, make a destination choice and a - // mode choice from IK MC probabilities; - // otherwise stop dest is set to destMgra, and make a mode - // choice from KJ MC probabilities. - if (i < stops.length - 1) - { - - //new code - depart period to and from stop - int departPeriodToStop = stop.getStopPeriod(); - int departPeriodFromStop = -1; - if(!directionIsInbound) - departPeriodFromStop = tour.getOutboundStops()[i+1].getStopPeriod(); - else - departPeriodFromStop = tour.getInboundStops()[i+1].getStopPeriod(); - - try - { - - check = System.nanoTime(); - - selectedIndex = selectDestinationWithTiming(stop,departPeriodToStop,departPeriodFromStop); - //close small probability logical hole, reset stop destination as intrazonal stop, log out reset cases - if(selectedIndex<0) { - selectedIndex=0; - choice=origMgra; - stop.setDest(choice); - modeAlt=tour.getTourModeChoice(); - modeLogsum=0; - logger.warn("Stop ID"+stop.id+" :destination set as intrazonal stop"); - }else { - choice = finalSample[selectedIndex]; - stop.setDest(choice); - modeAlt = selectModeFromProbabilities(stop, - mcCumProbsSegmentIk[selectedIndex]); - modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; - } - - if (sampleMgraInAlightingTapShed[choice]) - earlierTripWasLocatedInAlightingTapShed = true; - lastDest = choice; - slcTime += (System.nanoTime() - check); - - if (household.getDebugChoiceModels()) - { - if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " for joint tour stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + "N/A" - + ", tourPurpose=" + tour.getTourPrimaryPurpose() - + ", tourId=" + tour.getTourId() + ", tourMode=" - + tour.getTourModeChoice()); - smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" - + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - } else - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from IK Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" - + tour.getTourModeChoice()); - smcLogger.info("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" - + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - } - } - - check = System.nanoTime(); - /* - modeAlt = selectModeFromProbabilities(stop, - mcCumProbsSegmentIk[selectedIndex]); - modeLogsum = (float) mcLogsumsSegmentIk[selectedIndex]; - */ - - if (modeAlt < 0) - { - logger.info("error getting trip mode choice for IK proportions, i=" + i); - logger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - logger.info("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" - + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - throw new RuntimeException(); - } - - //value of time; lookup vot, votS2, or votS3 depending on chosen mode - if(modelStructure.getTripModeIsS2(modeAlt)){ - vot = mcVOTsSegmentIk[selectedIndex][1]; - }else if (modelStructure.getTripModeIsS3(modeAlt)){ - vot = mcVOTsSegmentIk[selectedIndex][2]; - }else{ - vot = mcVOTsSegmentIk[selectedIndex][0]; - } - - int park = -1; - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - if (park > 0) lastDest = park; - float parkingCost = parkingCostSegmentIk[selectedIndex]; - stop.setParkingCost(parkingCost); - - } - - smcTime += (System.nanoTime() - check); - } catch (Exception e) - { - logger.error(String.format( - "Exception caught processing %s stop location choice model.", - (stopLocDmuObj.getInboundStop() == 1 ? "inbound" : "outbound"))); - logger.error("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tour category=" - + tour.getTourCategory() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - logger.error("StopID=" + (stop.getStopId() + 1) + " of " - + (stops.length - 1) + " stops, inbound=" + stop.isInboundStop() - + ", stopPurpose=" + stop.getDestPurpose() + ", stopDepart=" - + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - logger.error(String - .format("origMgra=%d, destMgra=%d, newOrig=%d, lastDest=%d, modeAlt=%d, selectedIndex=%d, choice=%d.", - origMgra, destMgra, newOrig, lastDest, modeAlt, - selectedIndex, choice)); - throw new RuntimeException(e); - } - - stop.setMode(modeAlt); - stop.setModeLogsum(modeLogsum); - stop.setValueOfTime(vot); - - // if the trip is a transit mode, set the boarding and - // alighting tap pairs in the stop object based on the ik - // segment pairs - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) | modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - - int accEgr = -1; - if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { - accEgr = WTW; - } else { - if (stop.isInboundStop()) { - accEgr = WTD; - } else { - accEgr = DTW; - } - } - - if ( segmentIkBestTapPairs[selectedIndex] == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - //pick transit path from N-paths - double rn = household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, segmentIkBestTapPairs[selectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][0] ); - stop.setAlightTap( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][1] ); - stop.setSet( (int)segmentIkBestTapPairs[selectedIndex][accEgr][pathindex][2] ); - - } - - } - - oldSelectedIndex = selectedIndex; - - } else - { - - // last stop on half-tour, so dest is the half-tour final - // dest, and oldSelectedIndex was - // the selectedIndex determined for the previous stop - // location choice. - stop.setDest(destMgra); - - if (household.getDebugChoiceModels()) - { - if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " joint tour stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + "N/A" - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=End of " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " half-tour, stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" - + stop.getOrig() + ", stopDest=" + stop.getDest()); - } else - { - smcLogger - .info("Monte Carlo selection for determining Mode Choice from KJ Probabilities for " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " stop."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=End of " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " half-tour, stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" - + stop.getOrig() + ", stopDest=" + stop.getDest()); - } - } - check = System.nanoTime(); - //Wu added - if(mcCumProbsSegmentKj[oldSelectedIndex]!=null&&mcLogsumsSegmentKj!=null) { - modeAlt = selectModeFromProbabilities(stop, - mcCumProbsSegmentKj[oldSelectedIndex]); - modeLogsum = (float) mcLogsumsSegmentKj[oldSelectedIndex]; - }else { - modeAlt = tour.getTourModeChoice(); - modeLogsum = 0; - logger.warn("Stop ID"+stop.id+" :mode and mode logsum reset."); - } - - if (modeAlt < 0) - { - logger.error("error getting trip mode choice for KJ proportions, i=" + i); - logger.error("HHID=" + household.getHhId() + ", persNum=" - + person.getPersonNum() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId() - + ", tourMode=" + tour.getTourModeChoice()); - logger.error("StopID=End of " - + (stop.isInboundStop() ? "INBOUND" : "OUTBOUND") - + " half-tour, stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" - + stop.getOrig() + ", stopDest=" + stop.getDest()); - throw new RuntimeException(); - } - - //value of time; lookup vot, votS2, or votS3 depending on chosen mode - if(modelStructure.getTripModeIsS2(modeAlt)){ - vot = mcVOTsSegmentKj[oldSelectedIndex][1]; - }else if (modelStructure.getTripModeIsS3(modeAlt)){ - vot = mcVOTsSegmentKj[oldSelectedIndex][2]; - }else{ - vot = mcVOTsSegmentKj[oldSelectedIndex][0]; - } - - - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) { - float parkingCost = parkingCostSegmentKj[oldSelectedIndex]; - stop.setParkingCost(parkingCost); - } - // last stop on tour, so if inbound, only need park location - // choice if tour is work-based subtour; - // otherwise dest is home. - int park = -1; - if (directionIsInbound) - { - if (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - } else - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - - smcTime += (System.nanoTime() - check); - - stop.setMode(modeAlt); - stop.setModeLogsum(modeLogsum); - stop.setValueOfTime(vot); - - // if the last trip is a transit mode, set the boarding and - // alighting tap pairs in the stop object based on the kj - // segment pairs - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) || modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - - int accEgr = -1; - if(modelStructure.getTripModeIsWalkTransit(modeAlt)) { - accEgr = WTW; - } else { - if (stop.isInboundStop()) { - accEgr = WTD; - } else { - accEgr = DTW; - } - } - - if ( segmentKjBestTapPairs[oldSelectedIndex] == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - //pick transit path from N-paths - float rn = (float)household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, segmentKjBestTapPairs[oldSelectedIndex][accEgr], household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][0] ); - stop.setAlightTap( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][1] ); - stop.setSet( (int)segmentKjBestTapPairs[oldSelectedIndex][accEgr][pathindex][2] ); - - } - - } - - } - - } - - } else - { // create a stop object to hold attributes for orig, dest, mode, - // departtime, etc. - // for the half-tour with no stops. - - check = System.nanoTime(); - - // create a Stop object for use in applying trip mode choice for - // this half tour without stops - String origStopPurpose = ""; - String destStopPurpose = ""; - if (!directionIsInbound) - { - origStopPurpose = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - destStopPurpose = tour.getTourPrimaryPurpose(); - } else - { - origStopPurpose = tour.getTourPrimaryPurpose(); - destStopPurpose = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - } - - Stop stop = null; - try - { - stop = tour.createStop(origStopPurpose, destStopPurpose, - directionIsInbound, - tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)); - } catch (Exception e) - { - logger.info("exception creating stop."); - } - - stop.setOrig(origMgra); - stop.setDest(destMgra); - - if (directionIsInbound) stop.setStopPeriod(tour.getTourArrivePeriod()); - else stop.setStopPeriod(tour.getTourDepartPeriod()); - - int modeAlt = getHalfTourModeChoice(stop); - if (modeAlt < 0) - { - logger.info("error getting trip mode choice for half-tour with no stops."); - logger.info("HHID=" + household.getHhId() + ", tourPurpose=" - + tour.getTourPrimaryPurpose() + ", tourId=" + tour.getTourId()); - logger.info("StopID=" + (stop.getStopId() + 1) + " of no stops, inbound=" - + stop.isInboundStop() + ", stopPurpose=" + stop.getDestPurpose() - + ", stopDepart=" + stop.getStopPeriod() + ", stopOrig=" + stop.getOrig() - + ", stopDest=" + stop.getDest()); - - modeAlt = stop.getTour().getTourModeChoice(); - // throw new RuntimeException(); - } - - stop.setMode(modeAlt); - - double[][] bestTaps = null; - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) ) { - if ( directionIsInbound ) - bestTaps = tour.getBestWtwTapPairsIn(); - else - bestTaps = tour.getBestWtwTapPairsOut(); - } - else if ( modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - if ( directionIsInbound ) - bestTaps = tour.getBestWtdTapPairsIn(); - else - bestTaps = tour.getBestDtwTapPairsOut(); - } - - if ( bestTaps == null ) { - stop.setBoardTap( 0 ); - stop.setAlightTap( 0 ); - stop.setSet( 0 ); - } - else { - - // set taps - if ( modelStructure.getTripModeIsWalkTransit(modeAlt) || modelStructure.getTripModeIsPnrTransit(modeAlt) || modelStructure.getTripModeIsKnrTransit(modeAlt) ) { - - //pick transit path from N-paths - float rn = (float)household.getHhRandom().nextDouble(); - int pathindex = logsumHelper.chooseTripPath(rn, bestTaps, household.getDebugChoiceModels(), smcLogger); - - stop.setBoardTap( (int)bestTaps[pathindex][0] ); - stop.setAlightTap( (int)bestTaps[pathindex][1] ); - stop.setSet( (int)bestTaps[pathindex][2] ); - - } - - } - - // inbound half-tour, only need park location choice if tour is - // work-based subtour; - // otherwise dest is home. - int park = -1; - if (directionIsInbound) - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - } else - { - if (modelStructure.getTripModeIsSovOrHov(modeAlt)) - { - park = selectParkingLocation(household, tour, stop); - stop.setPark(park); - } - } - - smcTime += (System.nanoTime() - check); - } - - } - - private int selectDestination(Stop s, int departPeriodToStop, int departPeriodFromStop) - { - - Logger modelLogger = slcLogger; - - int[] loggingSample = null; - int[] debugLoggingSample = null; - // int[] debugLoggingSample = { 0, 16569 }; - // int[] debugLoggingSample = { 0, 4886, 16859, 18355, 3222, 14879, - // 26894, 16512, 9908, 18287, 14989 }; - - Tour tour = s.getTour(); - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - - if (household.getDebugChoiceModels()) - { - household.logHouseholdObject( - "Pre Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" - + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" - + tour.getTourPurpose() + ", Tour_" + tour.getTourId() - + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" - + (s.getStopId() + 1), modelLogger); - household.logPersonObject("Pre Stop Location Choice for person " - + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); - household.logTourObject("Pre Stop Location Choice for tour " + tour.getTourId(), - modelLogger, tour.getPersonObject(), tour); - household.logStopObject("Pre Stop Location Choice for stop " + (s.getStopId() + 1), - modelLogger, s, modelStructure); - - loggingSample = debugLoggingSample; - } - - int numAltsInSample = -1; - - stopLocDmuObj.setTourModeIndex(tour.getTourModeChoice()); - - // set the size terms array for the stop purpose in the dmu object - stopLocDmuObj - .setLogSize(getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household)); - - // get the array of distances from the stop origin mgra to all MGRAs and - // set in the dmu object - anm.getDistancesFromMgra(s.getOrig(), distanceFromStopOrigToAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesFromOrigMgra(distanceFromStopOrigToAllMgras); - - - // bike logsums from origin to all destinations - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumFromStopOrigToAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int dMgra = 1; dMgra <= mgraManager.getMaxMgra(); dMgra++) - { - bikeLogsumFromStopOrigToAllMgras[dMgra] = bls.getLogsum(segment,s.getOrig(),dMgra); - } - stopLocDmuObj.setBikeLogsumsFromOrigMgra(bikeLogsumFromStopOrigToAllMgras); - } - - - - // if tour mode is transit, set availablity of location alternatives - // based on transit accessibility relative to best transit TAP pair for - // tour - if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) - { - - Arrays.fill(sampleMgraInBoardingTapShed, false); - Arrays.fill(sampleMgraInAlightingTapShed, false); - - int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, tour, household.getDebugChoiceModels()); - if (numAvailableAlternatives == 0) - { - logger.error("no available locations - empty sample."); - logger.error("best tap pair which is empty: " + Arrays.deepToString(s.isInboundStop() ? tour.getBestWtwTapPairsIn() : tour.getBestWtwTapPairsOut())); - throw new RuntimeException(); - } - } - - // get the array of distances to the half-tour final destination mgra - // from all MGRAs and set in the dmu object - if (s.isInboundStop()) - { - // if inbound, final half-tour destination is the tour origin - anm.getDistancesToMgra(tour.getTourOrigMgra(), distanceToFinalDestFromAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); - - - // set the distance from the stop origin to the final half-tour - // destination - stopLocDmuObj - .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()]); - - - // bike logsums from all MGRAs back to tour origin - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) - { - bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourOrigMgra()); - } - stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); - } - - - - - - } else - { - // if outbound, final half-tour destination is the tour destination - anm.getDistancesToMgra(tour.getTourDestMgra(), distanceToFinalDestFromAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); - - // set the distance from the stop origin to the final half-tour - // destination - stopLocDmuObj - .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); - - // bike logsums from all MGRAs back to tour origin - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) - { - bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourDestMgra()); - } - stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); - } - - - - - } - - if (useNewSoaMethod) - { - if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) selectSampleOfAlternativesAutoTourNew( - s, tour, person, household, loggingSample); - else selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); - - numAltsInSample = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); - } else - { - if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) selectSampleOfAlternativesAutoTour( - s, tour, person, household, loggingSample); - else selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); - - numAltsInSample = altFreqMap.size(); - } - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - String separator = ""; - - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = "Stop Location Choice"; - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1)); - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - setupStopLocationChoiceAlternativeArrays(numAltsInSample, s, departPeriodToStop,departPeriodFromStop); - - int slcModelIndex = -1; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) slcModelIndex = MAND_SLC_MODEL_INDEX; - else if (tour.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX - || tour.getTourPrimaryPurposeIndex() == ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX - || tour.getTourPrimaryPurposeIndex() == ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX) slcModelIndex = MAINT_SLC_MODEL_INDEX; - else slcModelIndex = DISCR_SLC_MODEL_INDEX; - - float logsum = (float) slcModelArray[slcModelIndex].computeUtilities(stopLocDmuObj, - stopLocDmuObj.getDmuIndexValues(), sampleAvailability, inSample); - - if(s.isInboundStop()) - tour.addInboundStopDestinationLogsum(logsum); - else - tour.addOutboundStopDestinationLogsum(logsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - int selectedIndex = -1; - if (slcModelArray[slcModelIndex].getAvailabilityCount() > 0) - { - chosen = slcModelArray[slcModelIndex].getChoiceResult(rn); - selectedIndex = chosen; - }else{ - //wu's tempory fix to set chosen stop alternative to origin mgra if no alternative is available-8/27/2014 - //instead of this method, seems selectDestinationWithTiming(Stop s) is called (similar change made there too) - //tempory fix is put in here just in case. - chosen=tour.getTourOrigMgra(); - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels() || chosen < 0) - { - - if (chosen < 0) - { - - modelLogger - .error("ERROR selecting stop location choice due to no alternatives available."); - modelLogger - .error("setting debug to true and recomputing sample of alternatives selection."); - modelLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourPurpose(), - tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1), s.getOrig())); - - choiceModelDescription = "Stop Location Choice"; - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1), s.getOrig()); - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - - modelLogger.error(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.error(loggingHeader); - modelLogger.error(separator); - modelLogger.error(""); - modelLogger.error(""); - - // utilities and probabilities are 0 based. - double[] utilities = slcModelArray[slcModelIndex].getUtilities(); - double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); - - // availabilities is 1 based. - boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger - .error("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .error("Alternative Availability Utility Probability CumProb"); - modelLogger - .error("--------------------- ------------ ----------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 1; j <= numAltsInSample; j++) - { - - int alt = finalSample[j]; - - if (j == chosen) selectedIndex = j; - - cumProb += probabilities[j - 1]; - String altString = String.format("%-3d %5d", j, alt); - modelLogger.error(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); - } - - modelLogger.error(" "); - String altString = String.format("%-3d %5d", selectedIndex, -1); - modelLogger.error(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.error(separator); - modelLogger.error(""); - modelLogger.error(""); - - slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log - // file - slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); - - logger.error(String - .format("Error for HHID=%d, PersonNum=%d, no available %s stop destination choice alternatives to choose from in choiceModelApplication.", - tour.getHhId(), tour.getPersonObject().getPersonNum(), - tour.getTourPurpose())); - throw new RuntimeException(); - - } - - // utilities and probabilities are 0 based. - double[] utilities = slcModelArray[slcModelIndex].getUtilities(); - double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); - - // availabilities is 1 based. - boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- ------------ ----------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 1; j <= numAltsInSample; j++) - { - - int alt = finalSample[j]; - - if (j == chosen) selectedIndex = j; - - cumProb += probabilities[j - 1]; - String altString = String.format("%-3d %5d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %5d", selectedIndex, finalSample[selectedIndex]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); - - } - - return selectedIndex; - } - - private int selectDestinationWithTiming(Stop s,int departPeriodToStop,int departPeriodFromStop) - { - - Logger modelLogger = slcLogger; - - int[] loggingSample = null; - int[] debugLoggingSample = null; - // int[] debugLoggingSample = { 0, 16569 }; - // int[] debugLoggingSample = { 0, 4886, 16859, 18355, 3222, 14879, - // 26894, 16512, 9908, 18287, 14989 }; - - Tour tour = s.getTour(); - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - - if (household.getDebugChoiceModels()) - { - household.logHouseholdObject( - "Pre Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" - + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" - + tour.getTourPurpose() + ", Tour_" + tour.getTourId() - + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" - + (s.getStopId() + 1), modelLogger); - household.logPersonObject("Pre Stop Location Choice for person " - + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); - household.logTourObject("Pre Stop Location Choice for tour " + tour.getTourId(), - modelLogger, tour.getPersonObject(), tour); - household.logStopObject("Pre Stop Location Choice for stop " + (s.getStopId() + 1), - modelLogger, s, modelStructure); - - loggingSample = debugLoggingSample; - } - - int numAltsInSample = -1; - - stopLocDmuObj.setTourModeIndex(tour.getTourModeChoice()); - - // set the size terms array for the stop purpose in the dmu object - stopLocDmuObj - .setLogSize(getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household)); - - // get the array of distances from the stop origin mgra to all MGRAs and - // set in the dmu object - anm.getDistancesFromMgra(s.getOrig(), distanceFromStopOrigToAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesFromOrigMgra(distanceFromStopOrigToAllMgras); - - - // bike logsums from origin to all destinations - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumFromStopOrigToAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int dMgra = 1; dMgra <= mgraManager.getMaxMgra(); dMgra++) - { - bikeLogsumFromStopOrigToAllMgras[dMgra] = bls.getLogsum(segment,s.getOrig(),dMgra); - } - stopLocDmuObj.setBikeLogsumsFromOrigMgra(bikeLogsumFromStopOrigToAllMgras); - } - - // if tour mode is transit, set availablity of location alternatives - // based on transit accessibility relative to best transit TAP pair for - // tour - if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) - { - - Arrays.fill(sampleMgraInBoardingTapShed, false); - Arrays.fill(sampleMgraInAlightingTapShed, false); - - int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, tour, household.getDebugChoiceModels()); - if (numAvailableAlternatives == 0) - { - logger.error("no available locations - empty sample."); - logger.error("best tap pair which is empty: " + Arrays.deepToString(s.isInboundStop() ? tour.getBestWtwTapPairsIn() : tour.getBestWtwTapPairsOut())); - throw new RuntimeException(); - } - } - - // get the array of distances to the half-tour final destination mgra - // from all MGRAs and set in the dmu object - if (s.isInboundStop()) - { - // if inbound, final half-tour destination is the tour origin - anm.getDistancesToMgra(tour.getTourOrigMgra(), distanceToFinalDestFromAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); - - // set the distance from the stop origin to the final half-tour - // destination - stopLocDmuObj - .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()]); - - // bike logsums from all MGRAs back to tour origin - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) - { - bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourOrigMgra()); - } - stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); - } - - - } else - { - // if outbound, final half-tour destination is the tour destination - anm.getDistancesToMgra(tour.getTourDestMgra(), distanceToFinalDestFromAllMgras, - modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())); - stopLocDmuObj.setDistancesToDestMgra(distanceToFinalDestFromAllMgras); - - // set the distance from the stop origin to the final half-tour - // destination - stopLocDmuObj - .setOrigDestDistance(distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); - - // bike logsums from all MGRAs back to tour origin - if(modelStructure.getTourModeIsBike(tour.getTourModeChoice())){ - - Arrays.fill(bikeLogsumToFinalDestFromAllMgras, 0); - segment = new BikeLogsumSegment(person.getPersonIsFemale() == 1,tour.getTourPrimaryPurposeIndex() <= 3,s.isInboundStop()); - - for (int oMgra = 1; oMgra <= mgraManager.getMaxMgra(); oMgra++) - { - bikeLogsumToFinalDestFromAllMgras[oMgra] = bls.getLogsum(segment,oMgra,tour.getTourDestMgra()); - } - stopLocDmuObj.setBikeLogsumsToDestMgra(bikeLogsumToFinalDestFromAllMgras); - } - - - } - - long check = System.nanoTime(); - if (useNewSoaMethod) - { - if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) - { - selectSampleOfAlternativesAutoTourNew(s, tour, person, household, loggingSample); - soaAutoTime += (System.nanoTime() - check); - numAltsInSample = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); - } else - { - selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); - soaOtherTime += (System.nanoTime() - check); - numAltsInSample = altFreqMap.size(); - } - } else - { - if (modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice())) - { - selectSampleOfAlternativesAutoTour(s, tour, person, household, loggingSample); - soaAutoTime += (System.nanoTime() - check); - } else - { - selectSampleOfAlternativesOther(s, tour, person, household, loggingSample); - soaOtherTime += (System.nanoTime() - check); - } - numAltsInSample = altFreqMap.size(); - } - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - String separator = ""; - - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = "Stop Location Choice"; - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1)); - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - check = System.nanoTime(); - setupStopLocationChoiceAlternativeArrays(numAltsInSample, s,departPeriodToStop,departPeriodFromStop); - slsTime += (System.nanoTime() - check); - - int slcModelIndex = -1; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) slcModelIndex = MAND_SLC_MODEL_INDEX; - else if (tour.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX - || tour.getTourPrimaryPurposeIndex() == ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX - || tour.getTourPrimaryPurposeIndex() == ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX) slcModelIndex = MAINT_SLC_MODEL_INDEX; - else slcModelIndex = DISCR_SLC_MODEL_INDEX; - - float logsum = (float) slcModelArray[slcModelIndex].computeUtilities(stopLocDmuObj, - stopLocDmuObj.getDmuIndexValues(), sampleAvailability, inSample); - if(s.isInboundStop()) - tour.addInboundStopDestinationLogsum(logsum); - else - tour.addOutboundStopDestinationLogsum(logsum); - - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - int selectedIndex = -1; - if (slcModelArray[slcModelIndex].getAvailabilityCount() > 0) - { - selectedIndex = slcModelArray[slcModelIndex].getChoiceResult(rn); - chosen = finalSample[selectedIndex]; - }else{ - //wu's tempory fix to set chosen stop alternative to origin mgra if no alternative is available-8/27/2014 - chosen=tour.getTourOrigMgra(); - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels() || chosen < 0) - { - - if (chosen < 0) - { - - modelLogger - .error("ERROR selecting stop location choice due to no alternatives available."); - modelLogger - .error("setting debug to true and recomputing sample of alternatives selection."); - modelLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourPurpose(), - tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1), s.getOrig())); - - choiceModelDescription = "Stop Location Choice"; - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopId=%d, StopOrig=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourPurpose(), tour.getTourModeChoice(), tour.getTourId(), - s.getDestPurpose(), (s.getStopId() + 1), s.getOrig()); - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - - modelLogger.error(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.error(loggingHeader); - modelLogger.error(separator); - modelLogger.error(""); - modelLogger.error(""); - - // utilities and probabilities are 0 based. - double[] utilities = slcModelArray[slcModelIndex].getUtilities(); - double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); - - // availabilities is 1 based. - boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger - .error("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .error("Alternative Availability Utility Probability CumProb"); - modelLogger - .error("--------------------- ------------ ----------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 1; j <= numAltsInSample; j++) - { - - int alt = finalSample[j]; - - if (j == chosen) selectedIndex = j; - - cumProb += probabilities[j - 1]; - String altString = String.format("%-3d %5d", j, alt); - modelLogger.error(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); - } - - modelLogger.error(" "); - String altString = String.format("%-3d %5d", selectedIndex, -1); - modelLogger.error(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.error(separator); - modelLogger.error(""); - modelLogger.error(""); - - slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log - // file - slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); - - logger.error(String - .format("Error for HHID=%d, PersonNum=%d, no available %s stop destination choice alternatives to choose from in choiceModelApplication.", - tour.getHhId(), tour.getPersonObject().getPersonNum(), - tour.getTourPurpose())); - throw new RuntimeException(); - - } - - // utilities and probabilities are 0 based. - double[] utilities = slcModelArray[slcModelIndex].getUtilities(); - double[] probabilities = slcModelArray[slcModelIndex].getProbabilities(); - - // availabilities is 1 based. - boolean[] availabilities = slcModelArray[slcModelIndex].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- ------------ ----------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 1; j <= numAltsInSample; j++) - { - - int alt = finalSample[j]; - - if (j == chosen) selectedIndex = j; - - cumProb += probabilities[j - 1]; - String altString = String.format("%-3d %5d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j], utilities[j - 1], probabilities[j - 1], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %5d", selectedIndex, finalSample[selectedIndex]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - slcModelArray[slcModelIndex].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - slcModelArray[slcModelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - slcModelArray[slcModelIndex].logUECResults(modelLogger, loggingHeader); - - } - - return selectedIndex; - } - - private void setupStopLocationChoiceAlternativeArrays(int numAltsInSample, Stop s,int departPeriodToStop, int departPeriodFromStop) - { - - stopLocDmuObj.setNumberInSample(numAltsInSample); - stopLocDmuObj.setSampleOfAlternatives(finalSample); - stopLocDmuObj.setSlcSoaCorrections(sampleCorrectionFactors); - - // create arrays for ik and kj mode choice logsums for the stop origin, - // the sample stop location, and the half-tour final destination. - setupLogsumCalculation(s); - - int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; - ChoiceModelApplication mcModel = mcModelArray[category]; - - Household household = s.getTour().getPersonObject().getHouseholdObject(); - double income = (double) household.getIncomeInDollars(); - double ivtCoeff = ivtCoeffs[category]; - double incomeCoeff = incomeCoeffs[category]; - double incomeExpon = incomeExponents[category]; - double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); - double timeFactor = 1.0f; - if(s.getTour().getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - timeFactor = mcDmuObject.getJointTourTimeFactor(); - else if(s.getTour().getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) - timeFactor = mcDmuObject.getWorkTimeFactor(); - else - timeFactor = mcDmuObject.getNonWorkTimeFactor(); - - mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); - mcDmuObject.setCostCoeff(costCoeff); - - int halfTourFinalDest = s.isInboundStop() ? s.getTour().getTourOrigMgra() : s.getTour() - .getTourDestMgra(); - - // set the land use data items in the DMU for the stop origin - mcDmuObject.setOrigDuDen(mgraManager.getDuDenValue(s.getOrig())); - mcDmuObject.setOrigEmpDen(mgraManager.getEmpDenValue(s.getOrig())); - mcDmuObject.setOrigTotInt(mgraManager.getTotIntValue(s.getOrig())); - - for (int i = 1; i <= numAltsInSample; i++) - { - - int altMgra = finalSample[i]; - mcDmuObject.getDmuIndexValues().setDestZone(altMgra); - - // set distances to/from stop anchor points to stop location alternative. - ikDistance[i] = distanceFromStopOrigToAllMgras[altMgra]; - kjDistance[i] = distanceToFinalDestFromAllMgras[altMgra]; - - // set distances from tour anchor points to stop location - // alternative. - okDistance[i] = tourOrigToAllMgraDistances[altMgra]; - kdDistance[i] = tourDestToAllMgraDistances[altMgra]; - - // set the land use data items in the DMU for the sample location - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(altMgra)); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(altMgra)); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(altMgra)); - - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(altMgra))); - - // for walk-transit tours - if half-tour direction is outbound and - // stop alternative is in the walk shed, walk and walk-transit - // should be allowed for ik segments - // if half-tour direction is inbound and stop alternative is in the - // walk shed, walk and walk-transit should be allowed for kj - // segments - // if half-tour direction is outbound and stop alternative is in the - // walk shed, walk and walk-transit should be allowed for kj - // segments - // if half-tour direction is inbound and stop alternative is in the - // walk shed, walk and walk-transit should be allowed for ik - // segments - - // for drive-transit tours - if half-tour direction is outbound and - // stop alternative is in the drive shed, auto should be allowed for - // ik segments - // if half-tour direction is inbound and stop alternative is in the - // drive shed, auto should be allowed for kj segments - // if half-tour direction is outbound and stop alternative is in the - // walk shed, walk and walk-transit should be allowed for kj - // segments - // if half-tour direction is inbound and stop alternative is in the - // walk shed, walk and walk-transit should be allowed for ik - // segments - - // set values for walk-transit and drive-transit tours according to - // logic for IK segments - mcDmuObject.setAutoModeRequiredForTripSegment(false); - mcDmuObject.setWalkModeAllowedForTripSegment(false); - - mcDmuObject.setSegmentIsIk(true); - - double ikSegment = -999; - // drive transit tours are handled differently than walk transit - // tours - if (modelStructure.getTourModeIsDriveTransit(s.getTour().getTourModeChoice())) - { - - // if the direction is outbound - if (!s.isInboundStop()) - { - - // if the sampled mgra is in the outbound half-tour boarding tap shed (near tour origin) - if ( sampleMgraInBoardingTapShed[altMgra] ) { - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - } - - // if the sampled mgra is in the outbound half-tour alighting tap shed (near tour primary destination) - if ( sampleMgraInAlightingTapShed[altMgra] ) { - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - logsumHelper.setDtwTripMcDmuAttributes(mcDmuObject,s.getOrig(),altMgra,departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); - } - - - // if the trip origin and sampled mgra are in the outbound half-tour alighting tap shed (near tour origin) - if ( sampleMgraInAlightingTapShed[s.getOrig()] && sampleMgraInAlightingTapShed[altMgra] ) { - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, s.getOrig(), altMgra, departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - } - - } else - { - // if the sampled mgra is in the inbound half-tour boarding tap shed (near tour primary destination) - if ( sampleMgraInBoardingTapShed[altMgra] ) { - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, s.getOrig(), altMgra, departPeriodToStop, s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels() ); - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - } - - // if the sampled mgra is in the inbound half-tour alighting tap shed (near tour origin) - if ( sampleMgraInAlightingTapShed[altMgra] ) { - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - logsumHelper.setWtdTripMcDmuAttributes(mcDmuObject,s.getOrig(),altMgra,departPeriodToStop,s.getTour().getPersonObject().getHouseholdObject().getDebugChoiceModels()); - } - - // if the trip origin and sampled mgra are in the inbound half-tour alighting tap shed (near tour origin) - if ( sampleMgraInAlightingTapShed[s.getOrig()] && sampleMgraInAlightingTapShed[altMgra] ) { - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - } - - } - - } else if (modelStructure.getTourModeIsWalkTransit(s.getTour().getTourModeChoice())) - { // tour mode is walk-transit - - logsumHelper.setWtwTripMcDmuAttributes(mcDmuObject, s.getOrig(), altMgra, - departPeriodToStop, s.getTour().getPersonObject().getHouseholdObject() - .getDebugChoiceModels()); - - } else - { - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - } - ikSegment = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, departPeriodToStop, - mcModel, mcDmuObject, slcLogger); - - if (ikSegment < -900) - { - slcLogger.error("ERROR calculating trip mode choice logsum for " - + (s.isInboundStop() ? "inbound" : "outbound") - + " stop location choice - ikLogsum = " + ikSegment + "."); - slcLogger - .error("setting debug to true and recomputing ik segment logsum in order to log utility expression results."); - - if (s.isInboundStop()) slcLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, TourOrigMGRA=%d, TourDestMGRA=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumIBStops=%d, StopOrig=%d, AltStopLoc=%d", - s.getTour().getPersonObject().getHouseholdObject() - .getHhId(), s.getTour().getPersonObject() - .getPersonNum(), s.getTour().getPersonObject() - .getPersonType(), s.getTour().getTourPurpose(), s - .getTour().getTourModeChoice(), s.getTour() - .getTourId(), s.getTour().getTourOrigMgra(),s.getTour().getTourDestMgra(),s.getDestPurpose(), "inbound", (s - .getStopId() + 1), - s.getTour().getNumInboundStops() - 1, s.getOrig(), altMgra)); - else slcLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d,TourOrigMGRA=%d,TourDestMGRA=%d,StopPurpose=%s, StopDirection=%s, StopId=%d, NumOBStops=%d, StopOrig=%d, AltStopLoc=%d", - s.getTour().getPersonObject().getHouseholdObject() - .getHhId(), s.getTour().getPersonObject() - .getPersonNum(), s.getTour().getPersonObject() - .getPersonType(), s.getTour().getTourPurpose(), s - .getTour().getTourModeChoice(), s.getTour() - .getTourId(),s.getTour().getTourOrigMgra(),s.getTour().getTourDestMgra(),s.getDestPurpose(), "outbound", (s - .getStopId() + 1), s.getTour() - .getNumOutboundStops() - 1, s.getOrig(), altMgra)); - - mcDmuObject.getDmuIndexValues().setDebug(true); - mcDmuObject.getHouseholdObject().setDebugChoiceModels(true); - /* suppress log: Wu - mcDmuObject.getHouseholdObject().setDebugChoiceModels(true); - */ - mcDmuObject.getDmuIndexValues().setHHIndex( - s.getTour().getPersonObject().getHouseholdObject().getHhId()); - ikSegment = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, - departPeriodToStop, mcModel, mcDmuObject, slcLogger); - mcDmuObject.getDmuIndexValues().setDebug(false); - mcDmuObject.getHouseholdObject().setDebugChoiceModels(false); - - } - - // store the mode choice probabilities for the segment - mcCumProbsSegmentIk[i] = logsumHelper.getStoredSegmentCumulativeProbabilities(); - mcVOTsSegmentIk[i] = logsumHelper.getStoredSegmentVOTs(); - parkingCostSegmentIk[i] = logsumHelper.getTripModeChoiceSegmentStoredParkingCost(); - - // Store the mode choice logsum for the segment - mcLogsumsSegmentIk[i] = ikSegment; - - // store the best tap pairs for the segment - for(int j=0; j it = altFreqMap.keySet().iterator(); - int k = 0; - while (it.hasNext()) - { - - int alt = it.next(); - int freq = altFreqMap.get(alt); - - double prob = 0; - prob = probabilitiesList[alt - 1]; - - finalSample[k + 1] = alt; - sampleCorrectionFactors[k + 1] = Math.log((double) freq / prob); - - k++; - } - - while (k < sampleSize) - { - finalSample[k + 1] = -1; - sampleCorrectionFactors[k + 1] = Double.NaN; - sampleAvailability[k + 1] = false; - inSample[k + 1] = 0; - k++; - } - - } - - private void selectSampleOfAlternativesOther(Stop s, Tour tour, Person person, - Household household, int[] loggingSample) - { - - Logger soaLogger = Logger.getLogger("slcSoaLogger"); - - altFreqMap.clear(); - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - ChoiceModelApplication cm; - if (modelStructure.getTourModeIsWalk(tour.getTourModeChoice())) cm = slcSoaModel[WALK_STOP_LOC_SOA_SHEET_INDEX]; - else if (modelStructure.getTourModeIsBike(tour.getTourModeChoice())) cm = slcSoaModel[BIKE_STOP_LOC_SOA_SHEET_INDEX]; - else cm = slcSoaModel[OTHER_STOP_LOC_SOA_SHEET_INDEX]; - - if (household.getDebugChoiceModels()) - { - choiceModelDescription = String - .format("Stop Location SOA Choice Model for: stop purpose=%s, direction=%s, stopId=%d, stopOrig=%d", - s.getDestPurpose(), s.isInboundStop() ? "inbound" : "outbound", - (s.getStopId() + 1), s.getOrig()); - decisionMakerLabel = String - .format("HH=%d, persNum=%d, persType=%s, tourId=%d, tourPurpose=%s, tourOrig=%d, tourDest=%d, tourMode=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourId(), tour.getTourPrimaryPurpose(), tour.getTourOrigMgra(), - tour.getTourDestMgra(), tour.getTourModeChoice()); - cm.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, decisionMakerLabel); - } - - IndexValues dmuIndex = stopLocDmuObj.getDmuIndexValues(); - dmuIndex.setDebug(household.getDebugChoiceModels()); - - // stopLocDmuObj.setTourModeIndex( tour.getTourModeChoice() ); - // - // // set the size terms array for the stop purpose in the dmu object - // stopLocDmuObj.setLogSize( - // getLnSlcSizeTermsForStopPurpose(s.getStopPurposeIndex(), household) - // ); - // - // // get the array of distances from the stop origin mgra to all MGRAs - // and set in the dmu object - // anm.getDistancesFromMgra( s.getOrig(), - // distanceFromStopOrigToAllMgras, modelStructure.getTourModeIsSovOrHov( - // tour.getTourModeChoice() ) ); - // stopLocDmuObj.setDistancesFromOrigMgra( - // distanceFromStopOrigToAllMgras ); - // - // // if tour mode is transit, set availablity of location alternatives - // based on transit accessibility relative to best transit TAP pair for - // tour - // if ( modelStructure.getTourModeIsTransit( tour.getTourModeChoice() ) - // ) { - // int numAvailableAlternatives = setSoaAvailabilityForTransitTour(s, - // tour); - // if ( numAvailableAlternatives == 0 ) { - // logger.error( "no available locations - empty sample." ); - // throw new RuntimeException(); - // } - // } - // - // // get the array of distances to the half-tour final destination mgra - // from all MGRAs and set in the dmu object - if (s.isInboundStop()) - { - // // if inbound, final half-tour destination is the tour origin - // anm.getDistancesToMgra( tour.getTourOrigMgra(), - // distanceToFinalDestFromAllMgras, - // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) - // ); - // stopLocDmuObj.setDistancesToDestMgra( - // distanceToFinalDestFromAllMgras ); - // - // // set the distance from the stop origin to the final half-tour - // destination - // stopLocDmuObj.setOrigDestDistance( - // distanceFromStopOrigToAllMgras[tour.getTourOrigMgra()] ); - // - // // not used in UEC to reference matrices, but may be for - // debugging using $ORIG and $DEST as an expression - dmuIndex.setOriginZone(mgraManager.getTaz(s.getOrig())); - dmuIndex.setDestZone(mgraManager.getTaz(tour.getTourOrigMgra())); - } else - { - // // if outbound, final half-tour destination is the tour - // destination - // anm.getDistancesToMgra( tour.getTourDestMgra(), - // distanceToFinalDestFromAllMgras, - // modelStructure.getTourModeIsSovOrHov( tour.getTourModeChoice() ) - // ); - // stopLocDmuObj.setDistancesToDestMgra( - // distanceToFinalDestFromAllMgras ); - // - // // set the distance from the stop origin to the final half-tour - // destination - // stopLocDmuObj.setOrigDestDistance( - // distanceFromStopOrigToAllMgras[tour.getTourDestMgra()]); - // - // // not used in UEC to reference matrices, but may be for - // debugging using $ORIG and $DEST as an expression - dmuIndex.setOriginZone(mgraManager.getTaz(s.getOrig())); - dmuIndex.setDestZone(mgraManager.getTaz(tour.getTourDestMgra())); - } - - cm.computeUtilities(stopLocDmuObj, dmuIndex, soaAvailability, soaSample); - double[] probabilitiesList = cm.getProbabilities(); - double[] cumProbabilitiesList = cm.getCumulativeProbabilities(); - - // debug output - if (household.getDebugChoiceModels()) - { - - // write choice model alternative info to debug log file - cm.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - - // write UEC calculation results to separate model specific log file - loggingHeader = choiceModelDescription + ", " + decisionMakerLabel; - - if (loggingSample == null) - { - cm.logUECResultsSpecificAlts(soaLogger, loggingHeader, new int[] {0, s.getOrig(), - tour.getTourOrigMgra(), tour.getTourDestMgra()}); - // cm.logUECResults( soaLogger, loggingHeader, 10 ); - } else - { - cm.logUECResultsSpecificAlts(soaLogger, loggingHeader, loggingSample); - } - - } - - // loop over sampleSize, select alternatives based on probabilitiesList, - // and count frequency of alternatives chosen. - // may include duplicate alternative selections. - - Random hhRandom = household.getHhRandom(); - int rnCount = household.getHhRandomCount(); - // when household.getHhRandom() was applied, the random count was - // incremented, assuming a random number would be drawn right away. - // so let's decrement by 1, then increment the count each time a random - // number is actually drawn in this method. - rnCount--; - - // log degenerative cases - if (cm.getAvailabilityCount() == 0) - { - Logger badSlcLogger = Logger.getLogger("badSlc"); - - choiceModelDescription = String - .format("Stop Location SOA Choice Model for: stop purpose=%s, direction=%s, stopId=%d, stopOrig=%d", - s.getDestPurpose(), s.isInboundStop() ? "inbound" : "outbound", - (s.getStopId() + 1), s.getOrig()); - decisionMakerLabel = String - .format("HH=%d, persNum=%d, persType=%s, tourId=%d, tourPurpose=%s, tourOrig=%d, tourDest=%d, tourMode=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourId(), tour.getTourPrimaryPurpose(), tour.getTourOrigMgra(), - tour.getTourDestMgra(), tour.getTourModeChoice()); - loggingHeader = choiceModelDescription + ", " + decisionMakerLabel; - - badSlcLogger.info("....... Start Logging ......."); - badSlcLogger - .info("setting stop location sample to be an array with 1 element - just the stop origin mgra."); - badSlcLogger.info(""); - - household.logHouseholdObject( - "Stop Location Choice for trip: HH_" + household.getHhId() + ", Pers_" - + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" - + tour.getTourPurpose() + ", Tour_" + tour.getTourId() - + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" - + (s.getStopId() + 1), badSlcLogger); - household.logPersonObject("Stop Location Choice for person " - + tour.getPersonObject().getPersonNum(), badSlcLogger, tour.getPersonObject()); - household.logTourObject("Stop Location Choice for tour " + tour.getTourId(), - badSlcLogger, tour.getPersonObject(), tour); - household.logStopObject("Stop Location Choice for stop " + (s.getStopId() + 1), - badSlcLogger, s, modelStructure); - - badSlcLogger.info(decisionMakerLabel + " has no available alternatives for " - + choiceModelDescription + "."); - badSlcLogger.info("Logging StopLocation SOA Choice utility calculations for: stopOrig=" - + s.getOrig() + ", tourOrig=" + tour.getTourOrigMgra() + ", and tourDest=" - + tour.getTourDestMgra() + "."); - cm.logUECResultsSpecificAlts(badSlcLogger, loggingHeader, new int[] {0, s.getOrig(), - tour.getTourOrigMgra(), tour.getTourDestMgra()}); - - int chosenAlt = s.getOrig(); - probabilitiesList[chosenAlt - 1] = 1.0; - for (int j = chosenAlt - 1; j < cumProbabilitiesList.length; j++) - cumProbabilitiesList[j] = 1.0; - - double sum = 0; - double epsilon = .0000001; - for (int j = 0; j < probabilitiesList.length; j++) - { - sum += probabilitiesList[j]; - if (!(Math.abs(sum - cumProbabilitiesList[j]) < epsilon) || sum > 1.0) - { - badSlcLogger.info("error condition found! sum=" + sum + ", j=" + j - + ", cumProbabilitiesList[j]=" + cumProbabilitiesList[j]); - badSlcLogger.info("....... End Logging ......."); - throw new RuntimeException(); - } - } - - badSlcLogger.info("....... End Logging ......."); - badSlcLogger.info(""); - badSlcLogger.info(""); - } - - int chosenAlt = -1; - for (int i = 0; i < sampleSize; i++) - { - - double rn = hhRandom.nextDouble(); - rnCount++; - chosenAlt = Util.binarySearchDouble(cumProbabilitiesList, rn) + 1; - - // write choice model alternative info to log file - if (household.getDebugChoiceModels()) - { - cm.logSelectionInfo(loggingHeader, String.format("rnCount=%d", rnCount), rn, - chosenAlt); - } - - int freq = 0; - if (altFreqMap.containsKey(chosenAlt)) freq = altFreqMap.get(chosenAlt); - altFreqMap.put(chosenAlt, (freq + 1)); - - } - - // sampleSize random number draws were made from this Random object, so - // update the count in the hh's Random. - household.setHhRandomCount(rnCount); - - Arrays.fill(sampleAvailability, true); - Arrays.fill(inSample, 1); - - // create arrays of the unique chosen alternatives and the frequency - // with which those alternatives were chosen. - Iterator it = altFreqMap.keySet().iterator(); - int k = 0; - while (it.hasNext()) - { - - int alt = it.next(); - int freq = altFreqMap.get(alt); - - double prob = 0; - prob = probabilitiesList[alt - 1]; - - finalSample[k + 1] = alt; - sampleCorrectionFactors[k + 1] = Math.log((double) freq / prob); - - k++; - } - - while (k < sampleSize) - { - finalSample[k + 1] = -1; - sampleCorrectionFactors[k + 1] = Double.NaN; - sampleAvailability[k + 1] = false; - inSample[k + 1] = 0; - k++; - } - - // if the sample was determined for a transit tour, the sample and - // availability arrays for the full set of SOA alternatives need to be - // restored. - if (modelStructure.getTourModeIsTransit(tour.getTourModeChoice())) - { - for (int i = 0; i < soaSample.length; i++) - { - soaSample[i] = soaSampleBackup[i]; - soaAvailability[i] = soaAvailabilityBackup[i]; - } - } - - } - - private void setupLogsumCalculation(Stop s) - { - - Tour t = s.getTour(); - Person p = t.getPersonObject(); - Household hh = p.getHouseholdObject(); - - mcDmuObject.setHouseholdObject(hh); - mcDmuObject.setPersonObject(p); - mcDmuObject.setTourObject(t); - - int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; - double income = (double) hh.getIncomeInDollars(); - double ivtCoeff = ivtCoeffs[category]; - double incomeCoeff = incomeCoeffs[category]; - double incomeExpon = incomeExponents[category]; - double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); - double timeFactor = 1.0f; - if(t.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - timeFactor = mcDmuObject.getJointTourTimeFactor(); - else if(t.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) - timeFactor = mcDmuObject.getWorkTimeFactor(); - else - timeFactor = mcDmuObject.getNonWorkTimeFactor(); - - mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); - mcDmuObject.setCostCoeff(costCoeff); - - int tourMode = t.getTourModeChoice(); - int origMgra = s.getOrig(); - - mcDmuObject.getDmuIndexValues().setHHIndex(hh.getHhId()); - mcDmuObject.getDmuIndexValues().setZoneIndex(hh.getHhMgra()); - mcDmuObject.getDmuIndexValues().setOriginZone(origMgra); - mcDmuObject.getDmuIndexValues().setDebug(hh.getDebugChoiceModels()); - - mcDmuObject.setOutboundStops(t.getOutboundStops() == null ? 0 - : t.getOutboundStops().length - 1); - mcDmuObject.setInboundStops(t.getInboundStops() == null ? 0 - : t.getInboundStops().length - 1); - - mcDmuObject.setTripOrigIsTourDest(s.isInboundStop() && s.getStopId() == 0 ? 1 : 0); - mcDmuObject.setTripDestIsTourDest(!s.isInboundStop() - && ((s.getStopId() + 1) == (t.getNumOutboundStops() - 1)) ? 1 : 0); - - mcDmuObject.setInbound(s.isInboundStop()); - - mcDmuObject.setFirstTrip(0); - mcDmuObject.setLastTrip(0); - if (s.isInboundStop()) - { - mcDmuObject.setOutboundHalfTourDirection(0); - // compare stopId (0-based, so add 1) with number of stops (stops - // array length - 1); if last stop, set flag to 1, otherwise 0. - mcDmuObject.setLastTrip(((s.getStopId() + 1) == (t.getNumInboundStops() - 1)) ? 1 : 0); - } else - { - mcDmuObject.setOutboundHalfTourDirection(1); - // if first stopId (0-based), set flag to 1, otherwise 0. - mcDmuObject.setFirstTrip(s.getStopId() == 0 ? 1 : 0); - } - - mcDmuObject.setJointTour(t.getTourCategory().equalsIgnoreCase( - ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 : 0); - mcDmuObject - .setEscortTour(t.getTourPrimaryPurposeIndex() == ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX ? 1 - : 0); - - mcDmuObject.setIncomeInDollars(hh.getIncomeInDollars()); - mcDmuObject.setAdults(hh.getNumPersons18plus()); - mcDmuObject.setAutos(hh.getAutosOwned()); - mcDmuObject.setAge(p.getAge()); - mcDmuObject.setHhSize(hh.getHhSize()); - mcDmuObject.setPersonIsFemale(p.getPersonIsFemale()); - - mcDmuObject.setTourModeIsDA(modelStructure.getTourModeIsSov(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsS2(modelStructure.getTourModeIsS2(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsS3(modelStructure.getTourModeIsS3(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsWalk(modelStructure.getTourModeIsWalk(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsBike(modelStructure.getTourModeIsBike(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsWTran(modelStructure.getTourModeIsWalkTransit(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsPnr(modelStructure.getTourModeIsPnr(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsKnr(modelStructure.getTourModeIsKnr(tourMode) ? 1 : 0); - mcDmuObject.setTourModeIsSchBus(modelStructure.getTourModeIsSchoolBus(tourMode) ? 1 : 0); - - mcDmuObject - .setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(origMgra))); - - mcDmuObject.setDepartPeriod(t.getTourDepartPeriod()); - mcDmuObject.setArrivePeriod(t.getTourArrivePeriod()); - mcDmuObject.setTripPeriod(s.getStopPeriod()); - - double reimbursePct = mcDmuObject.getPersonObject().getParkingReimbursement(); - mcDmuObject.setReimburseProportion( reimbursePct ); - - - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(origMgra); - float waitTimeSingleTNC=0; - float waitTimeSharedTNC=0; - float waitTimeTaxi=0; - - Random hhRandom = hh.getHhRandom(); - double rnum = hhRandom.nextDouble(); - waitTimeSingleTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - waitTimeSharedTNC = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - waitTimeTaxi = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - mcDmuObject.setWaitTimeSingleTNC(waitTimeSingleTNC); - mcDmuObject.setWaitTimeSharedTNC(waitTimeSharedTNC); - mcDmuObject.setWaitTimeTaxi(waitTimeTaxi); - - - } - - /** - * determine if each indexed mgra has transit access to the best tap pairs - * for the tour create an array with 1 if the mgra indexed has at least one - * TAP within walk egress distance of the mgra or zero if no walk TAPS exist - * for the mgra. - */ - private int setSoaAvailabilityForTransitTour(Stop s, Tour t, boolean debug) - { - - int availableCount = 0; - double[][] bestTaps = null; - - if (s.isInboundStop()) - { - - if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) - bestTaps = t.getBestWtwTapPairsIn(); - else - bestTaps = t.getBestWtdTapPairsIn(); - - // loop through mgras and determine if they are available as a stop - // location - ArrayList mgras = mgraManager.getMgras(); - for (int alt : mgras) - { - // if alternative mgra is unavailable because it has no size, no - // need to check its accessibility - // if ( ! soaAvailability[alt] ) - // continue; - - boolean accessible = false; - int i=-1; - for (double[] tapPair : bestTaps) - { - if (tapPair == null) continue; - - ++i; - if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) { - // if alternative location mgra is accessible by walk to any of the best inbound boarding taps, AND it's accessible by walk to the stop origin, it's available. - if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) - && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) - && earlierTripWasLocatedInAlightingTapShed == false ) { - accessible = true; - sampleMgraInBoardingTapShed[alt] = true; - } - // if alternative location mgra is accessible by walk to any of the best inbound alighting taps, AND it's accessible by walk to the tour origin, it's available. - else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourOrigMgra()) ) { - accessible = true; - sampleMgraInAlightingTapShed[alt] = true; - } - } - else { - // if alternative location mgra is accessible by walk to any of the best origin taps, AND it's accessible by walk to the stop origin, it's available. - if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) - && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) - && earlierTripWasLocatedInAlightingTapShed == false ) { - accessible = true; - sampleMgraInBoardingTapShed[alt] = true; - } - // if alternative location mgra is accessible by drive to any of the best destination taps it's available. - else if ( mgraManager.getTapIsDriveAccessibleFromMgra(alt, (int)tapPair[1]) ) { - accessible = true; - sampleMgraInAlightingTapShed[alt] = true; - } - } - - if ( accessible ){ - if(debug){ - slcSoaLogger.info(""); - if(sampleMgraInBoardingTapShed[alt]==true){ - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in boarding shed of TAP "+tapPair[0]); - }else if(sampleMgraInAlightingTapShed[alt]==true){ - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in alighting shed of TAP "+tapPair[1]); - } - if((sampleMgraInBoardingTapShed[alt]==true) && (sampleMgraInAlightingTapShed[alt]==true)) //should not happen - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in both boarding shed of TAP "+tapPair[0]+" and alighting shed of TAP "+tapPair[1]); - } - - break; - } - } - - if (accessible) - { - availableCount++; - } else - { - soaSample[alt] = 0; - soaAvailability[alt] = false; - } - - } - - } else - { - if (modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice())) - bestTaps = t.getBestWtwTapPairsOut(); - else - bestTaps = t.getBestDtwTapPairsOut(); - - // loop through mgras and determine if they have walk egress - ArrayList mgras = mgraManager.getMgras(); - for (int alt : mgras) - { - // if alternative mgra is unavailable because it has no size, no - // need to check its accessibility - // if ( ! soaAvailability[alt] ) - // continue; - - // check whether any of the outbound dtw boarding taps or best - // wtw alighting taps are in the set of walk accessible TAPs for - // the alternative mgra. - // if not, the alternative is not available. - boolean accessible = false; - int i=-1; - for (double[] tapPair : bestTaps) - { - if (tapPair == null) continue; - - ++i; - if ( modelStructure.getTourModeIsWalkTransit(t.getTourModeChoice() ) ) { - // if alternative location mgra is accessible by walk to any of the best origin taps, AND it's accessible by walk to the stop origin, it's available. - if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[0]) - && mgraManager.getMgrasAreWithinWalkDistance(s.getOrig(), alt) - && earlierTripWasLocatedInAlightingTapShed == false ) { - accessible = true; - sampleMgraInBoardingTapShed[alt] = true; - } - // if alternative location mgra is accessible by walk to any of the best destination taps, AND it's accessible by walk to the tour primary destination, it's available. - else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourDestMgra()) ) { - accessible = true; - sampleMgraInAlightingTapShed[alt] = true; - } - } - else { - // if alternative location mgra is accessible by drive to any of the best origin taps, it's available. - if ( mgraManager.getTapIsDriveAccessibleFromMgra(alt, (int)tapPair[0]) - && earlierTripWasLocatedInAlightingTapShed == false ) { - accessible = true; - sampleMgraInBoardingTapShed[alt] = true; - } - // if alternative location mgra is accessible by walk to any of the best destination taps, AND it's accessible by walk to the tour primary destination, it's available. - else if ( mgraManager.getTapIsWalkAccessibleFromMgra(alt, (int)tapPair[1]) && mgraManager.getMgrasAreWithinWalkDistance(alt, t.getTourDestMgra()) ) { - accessible = true; - sampleMgraInAlightingTapShed[alt] = true; - } - } - - if ( accessible ){ - if(debug){ - slcSoaLogger.info(""); - if(sampleMgraInBoardingTapShed[alt]==true){ - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in boarding shed of TAP "+tapPair[0]); - }else if(sampleMgraInAlightingTapShed[alt]==true){ - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in alighting shed of TAP "+tapPair[1]); - } - if((sampleMgraInBoardingTapShed[alt]==true) && (sampleMgraInAlightingTapShed[alt]==true)) //should not happen - slcSoaLogger.info("Stop alternative MGRA "+alt+" is accessible for TapPair "+i+" in both boarding shed of TAP "+tapPair[0]+" and alighting shed of TAP "+tapPair[1]); - } - - break; - } - - } - if (accessible) - { - availableCount++; - } else - { - soaSample[alt] = 0; - soaAvailability[alt] = false; - } - - } - - } - - return availableCount; - } - - /** - * create an array with 1 if the mgra indexed has at least one TAP within - * walk egress distance of the mgra or zero if no walk TAPS exist for the - * mgra. private void createWalkTransitAvailableArray() { - * - * ArrayList mgras = mgraManager.getMgras(); int maxMgra = - * mgraManager.getMaxMgra(); - * - * walkTransitAvailable = new int[maxMgra+1]; - * - * // loop through mgras and determine if they have walk egress for (int alt - * : mgras) { - * - * // get the TAP set within walk egress distance of the stop location - * alternative. int[] aMgraSet = - * mgraManager.getMgraWlkTapsDistArray()[alt][0]; - * - * // set to 1 if the list of TAPS with walk accessible egress to alt is not - * empty; 0 otherwise if ( aMgraSet != null && aMgraSet.length > 0 ) - * walkTransitAvailable[alt] = 1; - * - * } - * - * stopLocDmuObj.setWalkTransitAvailable( walkTransitAvailable ); } - */ - - /** - * Do a monte carlo selection from the array of stored mode choice - * cumulative probabilities (0 based array). The probabilities were saved at - * the time the stop location alternative segment mode choice logsums were - * calculated. If the stop is not the last stop for the half-tour, the IK - * segment probabilities are passed in. If the stop is the last stop, the KJ - * probabilities are passeed in. - * - * @param household - * object frim which to get the Random object. - * @param props - * is the array of stored mode choice probabilities - 0s based - * array. - * - * @return the selected mode choice alternative from [1,...,numMcAlts]. - */ - private int selectModeFromProbabilities(Stop s, double[] cumProbs) - { - - Household household = s.getTour().getPersonObject().getHouseholdObject(); - - int selectedModeAlt = -1; - double rn = household.getHhRandom().nextDouble(); - int randomCount = household.getHhRandomCount(); - - int numAvailAlts = 0; - double sumProb = 0.0; - for (int i = 0; i < cumProbs.length; i++) - { - double tempProb = cumProbs[i] - sumProb; - sumProb += tempProb; - - if (tempProb > 0) numAvailAlts++; - - if (rn < cumProbs[i]) - { - selectedModeAlt = i + 1; - break; - } - } - - if (household.getDebugChoiceModels() || selectedModeAlt < 0 - || numAvailAlts >= availAltsToLog) - { - - // set the number of available alts to log value to a large number - // so no more get logged. - if (numAvailAlts >= availAltsToLog) - { - Person person = s.getTour().getPersonObject(); - Tour tour = s.getTour(); - smcLogger - .info("Monte Carlo selection for determining Mode Choice from Probabilities for stop with more than " - + availAltsToLog + " mode alts available."); - smcLogger.info("HHID=" + household.getHhId() + ", persNum=" + person.getPersonNum() - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=" - + (s.getStopId() + 1) - + " of " - + (s.isInboundStop() ? tour.getNumInboundStops() - 1 : tour - .getNumOutboundStops() - 1) + " stops, inbound=" - + s.isInboundStop() + ", stopPurpose=" + s.getDestPurpose() - + ", stopDepart=" + s.getStopPeriod() + ", stopOrig=" + s.getOrig() - + ", stopDest=" + s.getDest()); - availAltsToLog = 9999; - } - - smcLogger.info(""); - smcLogger.info(""); - String separator = ""; - for (int k = 0; k < 60; k++) - separator += "+"; - smcLogger.info(separator); - - smcLogger - .info("Alternative Availability Utility Probability CumProb"); - smcLogger - .info("--------------------- ------------ ----------- -------------- --------------"); - - sumProb = 0.0; - for (int j = 0; j < cumProbs.length; j++) - { - String altString = String.format("%-3d %-25s", j + 1, ""); - double tempProb = cumProbs[j] - sumProb; - smcLogger.info(String.format("%-30s%15s%18s%18.6e%18.6e", altString, "", "", - tempProb, cumProbs[j])); - sumProb += tempProb; - } - - if (selectedModeAlt < 0) - { - smcLogger.info(" "); - String altString = String.format("%-3d %-25s", selectedModeAlt, - "no MC alt available"); - smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - throw new RuntimeException(); - } else - { - smcLogger.info(" "); - String altString = String.format("%-3d %-25s", selectedModeAlt, ""); - smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - } - - smcLogger.info(separator); - smcLogger.info(""); - smcLogger.info(""); - - } - - // if this statement is reached, there's a problem with the cumulative - // probabilities array, so return -1. - return selectedModeAlt; - } - - /** - * This method is taken from the setupStopLocationChoiceAlternativeArrays(), - * except that the stop location choice dmu attributes are not set and the - * logsum calculation setup is done only for the selected stop location - * alternative. - * - * @param stop - * object representing the half-tour. - * - * @return the selected mode choice alternative from [1,...,numMcAlts]. - */ - private int getHalfTourModeChoice(Stop s) - { - - Household hh = s.getTour().getPersonObject().getHouseholdObject(); - - // create arrays for ik and kj mode choice logsums for the stop origin, - // the sample stop location, and the half-tour final destination. - setupLogsumCalculation(s); - - int category = PURPOSE_CATEGORIES[s.getTour().getTourPrimaryPurposeIndex()]; - ChoiceModelApplication mcModel = mcModelArray[category]; - - int altMgra = s.getDest(); - mcDmuObject.getDmuIndexValues().setDestZone(altMgra); - - // set the mode choice attributes for the sample location - mcDmuObject.setDestDuDen(mgraManager.getDuDenValue(altMgra)); - mcDmuObject.setDestEmpDen(mgraManager.getEmpDenValue(altMgra)); - mcDmuObject.setDestTotInt(mgraManager.getTotIntValue(altMgra)); - - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(altMgra))); - - mcDmuObject.setAutoModeRequiredForTripSegment(false); - mcDmuObject.setWalkModeAllowedForTripSegment(false); - - if (hh.getDebugChoiceModels()) - { - smcLogger.info("LOGSUM calculation for determining Mode Choice Probabilities for " - + (s.isInboundStop() ? "INBOUND" : "OUTBOUND") + " half-tour with no stops."); - - hh.logHouseholdObject( - "Half Tour Mode Choice: HH_" + hh.getHhId() + ", Pers_" - + s.getTour().getPersonObject().getPersonNum() + ", Tour Purpose_" - + s.getTour().getTourPurpose() + ", Tour_" + s.getTour().getTourId() - + ", Tour Purpose_" + s.getTour().getTourPurpose() + ", Stop_" - + (s.getStopId() + 1), smcLogger); - hh.logPersonObject("Half Tour Mode Choice for person " - + s.getTour().getPersonObject().getPersonNum(), smcLogger, s.getTour().getPersonObject()); - hh.logTourObject("Half Tour Mode Choice for tour " + s.getTour().getTourId(), - smcLogger, s.getTour().getPersonObject(), s.getTour()); - hh.logStopObject("Half Tour Mode Choice for stop " + (s.getStopId() + 1), - smcLogger, s, modelStructure); - } - - if (modelStructure.getTourModeIsDriveTransit(s.getTour().getTourModeChoice())) - { - - logsumHelper.setWalkTransitLogSumUnavailable( mcDmuObject ); - - if (s.isInboundStop()) logsumHelper.setWtdTripMcDmuAttributesForBestTapPairs( - mcDmuObject, s.getOrig(), altMgra, s.getStopPeriod(), s.getTour() - .getBestWtdTapPairsIn(), s.getTour().getPersonObject() - .getHouseholdObject().getDebugChoiceModels()); - else logsumHelper.setDtwTripMcDmuAttributesForBestTapPairs(mcDmuObject, s.getOrig(), - altMgra, s.getStopPeriod(), s.getTour().getBestDtwTapPairsOut(), s.getTour() - .getPersonObject().getHouseholdObject().getDebugChoiceModels()); - - } else - { - - logsumHelper.setDriveTransitLogSumUnavailable( mcDmuObject, s.isInboundStop() ); - - if (s.isInboundStop()) logsumHelper.setWtwTripMcDmuAttributesForBestTapPairs( - mcDmuObject, s.getOrig(), altMgra, s.getStopPeriod(), s.getTour() - .getBestWtwTapPairsIn(), s.getTour().getPersonObject() - .getHouseholdObject().getDebugChoiceModels()); - else logsumHelper.setWtwTripMcDmuAttributesForBestTapPairs(mcDmuObject, s.getOrig(), - altMgra, s.getStopPeriod(), s.getTour().getBestWtwTapPairsOut(), s.getTour() - .getPersonObject().getHouseholdObject().getDebugChoiceModels()); - - } - double logsum = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, s.getStopPeriod(), - mcModel, mcDmuObject, smcLogger); - - s.setModeLogsum((float) logsum); - - double rn = hh.getHhRandom().nextDouble(); - int randomCount = hh.getHhRandomCount(); - - int selectedModeAlt = -1; - if (mcModel.getAvailabilityCount() > 0) - { - selectedModeAlt = mcModel.getChoiceResult(rn); - } - - if (hh.getDebugChoiceModels() || selectedModeAlt < 0 - || mcModel.getAvailabilityCount() >= availAltsToLog) - { - - // set the number of available alts to log value to a large number - // so no more get logged. - if (selectedModeAlt < 0 || mcModel.getAvailabilityCount() >= availAltsToLog) - { - Person person = s.getTour().getPersonObject(); - Tour tour = s.getTour(); - if (mcModel.getAvailabilityCount() >= availAltsToLog) - { - availAltsToLog = 9999; - smcLogger - .info("Logsum calculation for determining Mode Choice for half-tour more than " - + availAltsToLog + " mode alts available."); - } else - { - smcLogger - .info("Logsum calculation for determining Mode Choice for half-tour with no stops."); - } - smcLogger.info("HHID=" + hh.getHhId() + ", persNum=" + person.getPersonNum() - + ", tourPurpose=" + tour.getTourPrimaryPurpose() + ", tourId=" - + tour.getTourId() + ", tourMode=" + tour.getTourModeChoice()); - smcLogger.info("StopID=" - + (s.getStopId() + 1) - + " of " - + (s.isInboundStop() ? tour.getNumInboundStops() - 1 : tour - .getNumOutboundStops() - 1) + " stops, inbound=" - + s.isInboundStop() + ", stopPurpose=" + s.getDestPurpose() - + ", stopDepart=" + s.getStopPeriod() + ", stopOrig=" + s.getOrig() - + ", stopDest=" + s.getDest()); - } - - // altNames, utilities and probabilities are 0 based. - String[] altNames = mcModel.getAlternativeNames(); - double[] utilities = mcModel.getUtilities(); - double[] probabilities = mcModel.getProbabilities(); - - // availabilities is 1 based. - boolean[] availabilities = mcModel.getAvailabilities(); - - smcLogger.info(""); - smcLogger.info(""); - String separator = ""; - for (int k = 0; k < 60; k++) - separator += "+"; - smcLogger.info(separator); - - smcLogger - .info("Alternative Availability Utility Probability CumProb"); - smcLogger - .info("--------------------- ------------ ----------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < utilities.length; j++) - { - cumProb += probabilities[j]; - String altString = String.format("%-3d %-25s", j + 1, altNames[j]); - smcLogger.info(String.format("%-30s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - if (selectedModeAlt < 0) - { - smcLogger.info(" "); - String altString = String.format("%-3d %-25s", selectedModeAlt, - "no MC alt available"); - smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - } else - { - smcLogger.info(" "); - String altString = String.format("%-3d %-25s", selectedModeAlt, - altNames[selectedModeAlt - 1]); - smcLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - } - - smcLogger.info(separator); - smcLogger.info(""); - smcLogger.info(""); - - if (logsum < -900 || selectedModeAlt < 0) - { - if (logsum < -900 || selectedModeAlt < 0) smcLogger - .error("ERROR calculating trip mode choice logsum for " - + (s.isInboundStop() ? "inbound" : "outbound") - + " half-tour with no stops - ikLogsum = " + logsum + "."); - else smcLogger.error("No half-tour mode choice alternatives available " - + (s.isInboundStop() ? "inbound" : "outbound") - + " half-tour with no stops - ikLogsum = " + logsum + "."); - smcLogger - .error("setting debug to true and recomputing half-tour logsum in order to log utility expression results."); - - if (s.isInboundStop()) smcLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumIBStops=%d, StopOrig=%d, AltStopLoc=%d", - s.getTour().getPersonObject().getHouseholdObject() - .getHhId(), s.getTour().getPersonObject() - .getPersonNum(), s.getTour().getPersonObject() - .getPersonType(), s.getTour().getTourPurpose(), s - .getTour().getTourModeChoice(), s.getTour() - .getTourId(), s.getDestPurpose(), "inbound", (s - .getStopId() + 1), - s.getTour().getNumInboundStops() - 1, s.getOrig(), altMgra)); - else smcLogger - .error(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourMode=%d, TourId=%d, StopPurpose=%s, StopDirection=%s, StopId=%d, NumOBStops=%d, StopOrig=%d, AltStopLoc=%d", - s.getTour().getPersonObject().getHouseholdObject() - .getHhId(), s.getTour().getPersonObject() - .getPersonNum(), s.getTour().getPersonObject() - .getPersonType(), s.getTour().getTourPurpose(), s - .getTour().getTourModeChoice(), s.getTour() - .getTourId(), s.getDestPurpose(), "outbound", (s - .getStopId() + 1), s.getTour() - .getNumOutboundStops() - 1, s.getOrig(), altMgra)); - - mcDmuObject.getDmuIndexValues().setDebug(true); - mcDmuObject.getDmuIndexValues().setHHIndex( - s.getTour().getPersonObject().getHouseholdObject().getHhId()); - logsum = logsumHelper.calculateTripMcLogsum(s.getOrig(), altMgra, - s.getStopPeriod(), mcModel, mcDmuObject, smcLogger); - mcDmuObject.getDmuIndexValues().setDebug(false); - - // throw new RuntimeException(); - - } - - } - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = mcModel.getUEC(); - - double vot = 0.0; - - if(modelStructure.getTripModeIsS2(selectedModeAlt)){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = uec.getValueForIndex(votIndex); - }else if (modelStructure.getTripModeIsS3(selectedModeAlt)){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = uec.getValueForIndex(votIndex); - } - s.setValueOfTime(vot); - - return selectedModeAlt; - } - - private void setOutboundTripDepartTimes(Stop[] stops) - { - - // these stops are in outbound direction - int halfTourDirection = 0; - - for (int i = 0; i < stops.length; i++) - { - - // if tour depart and arrive periods are the same, set same values - // for the stops - Stop stop = stops[i]; - Tour tour = stop.getTour(); - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - if (tour.getTourArrivePeriod() == tour.getTourDepartPeriod()) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Depart Time Model Not Run Since Tour Depart and Arrive Periods are Equal; Stop Depart Period set to Tour Depart Period = " - + tour.getTourDepartPeriod() + " for outbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourModeChoice(), - tour.getTourCategory(), tour.getTourPrimaryPurpose(), - tour.getTourId(), stop.getOrigPurpose(), - stop.getDestPurpose(), (stop.getStopId() + 1), - stops.length)); - tripDepartLogger.info(String.format("tourDepartPeriod=%d, tourArrivePeriod=%d", - tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourDepartPeriod()); - - } else - { - - int tripIndex = i + 1; - - if (tripIndex == 1) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Depart Time Model Not Run Since Trip is first trip in sequence, departing from " - + stop.getOrigPurpose() - + "; Stop Depart Period set to Tour Depart Period = " - + tour.getTourDepartPeriod() + " for outbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), tour.getTourId(), - stop.getOrigPurpose(), stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger.info(String.format( - "tourDepartPeriod=%d, tourArrivePeriod=%d", - tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourDepartPeriod()); - - } else - { - - int prevTripPeriod = stops[i - 1].getStopPeriod(); - - if (prevTripPeriod == tour.getTourArrivePeriod()) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Depart Time Model Not Run Since Previous Trip Depart and Tour Arrive Periods are Equal; Stop Depart Period set to Tour Arrive Period = " - + tour.getTourArrivePeriod() - + " for outbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), tour.getTourId(), - stop.getOrigPurpose(), stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger.info(String.format( - "prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", - prevTripPeriod, tour.getTourDepartPeriod(), - tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourDepartPeriod()); - - } else - { - - int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); - - double[] proportions = stopTodModel.getStopTodIntervalProportions( - tourPrimaryPurposeIndex, halfTourDirection, prevTripPeriod, - tripIndex); - - // for inbound trips, the first trip cannot arrive - // earlier than the last outbound trip departs - // if such a case is chosen, re-select. - int invalidCount = 0; - boolean validTripDepartPeriodSet = false; - while (validTripDepartPeriodSet == false) - { - - double rn = household.getHhRandom().nextDouble(); - int choice = getMonteCarloSelection(proportions, rn); - - // check that this stop depart time departs at same - // time or later than the stop object preceding this - // one in the stop sequence. - if (choice >= prevTripPeriod && choice <= tour.getTourArrivePeriod()) - { - validTripDepartPeriodSet = true; - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Depart Time Model for outbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), - person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), - tour.getTourId(), - stop.getOrigPurpose(), - stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger - .info(String - .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", - prevTripPeriod, - tour.getTourDepartPeriod(), - tour.getTourArrivePeriod())); - tripDepartLogger.info("tourPrimaryPurposeIndex=" - + tourPrimaryPurposeIndex + ", halfTourDirection=" - + halfTourDirection + ", tripIndex=" + tripIndex); - tripDepartLogger.info(""); - - tripDepartLogger.info(loggerSeparator); - tripDepartLogger.info(String.format("%-4s %-8s %10s %10s", - "alt", "time", "prob", "cumProb")); - double cumProb = 0.0; - for (int p = 1; p < proportions.length; p++) - { - int hr = 4 + (p / 2); - int min = (p % 2) * 30; - cumProb += proportions[p]; - String timeString = ((hr < 10) ? ("0" + hr) - : ("" + hr + ":")) + ((min == 30) ? min : "00"); - tripDepartLogger.info(String.format( - "%-4d %-8s %10.8f %10.8f", p, timeString, - proportions[p], cumProb)); - } - tripDepartLogger.info(loggerSeparator); - tripDepartLogger.info("rn=" + rn + ", choice=" + choice - + ", try=" + invalidCount); - tripDepartLogger.info(""); - } - stop.setStopPeriod(choice); - - } else - { - invalidCount++; - } - - if (invalidCount > MAX_INVALID_FIRST_ARRIVAL_COUNT) - { - tripDepartLogger.warn("Problem in Outbound Trip Depart Time Model."); - tripDepartLogger - .warn("Outbound trip depart time less than previous trip depart time for " - + invalidCount + " times."); - tripDepartLogger.warn("Possible infinite loop?"); - tripDepartLogger - .warn(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), - tour.getTourId(), stop.getOrigPurpose(), - stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger - .warn(String - .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d, last choice=%d", - prevTripPeriod, tour.getTourDepartPeriod(), - tour.getTourArrivePeriod(), choice)); - tripDepartLogger.warn("=" + invalidCount + " times."); - - //throw new RuntimeException(); - //instead of throwing an exception, set the stop period to the same period as the last stop - stop.setStopPeriod(prevTripPeriod); - } - - } - - } - - } - - } - - } - - } - - private void setInboundTripDepartTimes(Stop[] stops, int lastOutboundTripDeparts) - { - - // these stops are in inbound direction - int halfTourDirection = 1; - - for (int i = stops.length - 1; i >= 0; i--) - { - - // if tour depart and arrive periods are the same, set same values - // for the stops - Stop stop = stops[i]; - Tour tour = stop.getTour(); - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - if (tour.getTourArrivePeriod() == tour.getTourDepartPeriod()) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Arrive Time Model Not Run Since Tour Depart and Arrive Periods are Equal; Stop Arrive Period set to Tour Arrive Period = " - + tour.getTourDepartPeriod() + " for inbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourModeChoice(), - tour.getTourCategory(), tour.getTourPrimaryPurpose(), - tour.getTourId(), stop.getOrigPurpose(), - stop.getDestPurpose(), (stop.getStopId() + 1), - stops.length)); - tripDepartLogger.info(String.format("tourDepartPeriod=%d, tourArrivePeriod=%d", - tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourArrivePeriod()); - - } else - { - - int tripIndex = stops.length - i; - - if (tripIndex == 1) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Arrive Time Model Not Run Since Trip is last trip in sequence, arriving at " - + stop.getDestPurpose() - + "; Stop Arrive Period set to Tour Arrive Period = " - + tour.getTourArrivePeriod() + " for inbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), tour.getTourId(), - stop.getOrigPurpose(), stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger.info(String.format( - "tourDepartPeriod=%d, tourArrivePeriod=%d", - tour.getTourDepartPeriod(), tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourArrivePeriod()); - - } else - { - - int prevTripPeriod = stops[i + 1].getStopPeriod(); - - if (prevTripPeriod == tour.getTourArrivePeriod()) - { - - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Arrive Time Model Not Run Since Previous Trip Arrive and Tour Arrive Periods are Equal; Stop Arrive Period set to Tour Arrive Period = " - + tour.getTourArrivePeriod() - + " for intbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), tour.getTourId(), - stop.getOrigPurpose(), stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger.info(String.format( - "prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", - prevTripPeriod, tour.getTourDepartPeriod(), - tour.getTourArrivePeriod())); - tripDepartLogger.info(""); - } - stop.setStopPeriod(tour.getTourArrivePeriod()); - - } else - { - - int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); - - double[] proportions = stopTodModel.getStopTodIntervalProportions( - tourPrimaryPurposeIndex, halfTourDirection, prevTripPeriod, - tripIndex); - - // for inbound trips, the first trip cannot arrive - // earlier than the last outbound trip departs - // if such a case is chosen, re-select. - int invalidCount = 0; - boolean validTripArrivePeriodSet = false; - while (validTripArrivePeriodSet == false) - { - - double rn = household.getHhRandom().nextDouble(); - int choice = getMonteCarloSelection(proportions, rn); - - // check that this stop arrival time arrives at same - // time or earlier than the stop object following - // this one in the stop sequence. - // also check that this stop arrival is after the - // depart time for the last outbound stop. - if (choice <= prevTripPeriod && choice >= lastOutboundTripDeparts) - { - validTripArrivePeriodSet = true; - if (household.getDebugChoiceModels()) - { - tripDepartLogger - .info("Trip Arrive Time Model for inbound half-tour."); - tripDepartLogger - .info(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), - person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), - tour.getTourId(), - stop.getOrigPurpose(), - stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger.info("tourPrimaryPurposeIndex=" - + tourPrimaryPurposeIndex + ", halfTourDirection=" - + halfTourDirection + ", tripIndex=" + tripIndex - + ", prevTripPeriod=" + prevTripPeriod); - tripDepartLogger - .info(String - .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d", - prevTripPeriod, - tour.getTourDepartPeriod(), - tour.getTourArrivePeriod())); - tripDepartLogger.info(loggerSeparator); - tripDepartLogger.info(""); - - tripDepartLogger.info(String.format("%-4s %-8s %10s %10s", - "alt", "time", "prob", "cumProb")); - double cumProb = 0.0; - for (int p = 1; p < proportions.length; p++) - { - int hr = 4 + (p / 2); - int min = (p % 2) * 30; - cumProb += proportions[p]; - String timeString = ((hr < 10) ? ("0" + hr) - : ("" + hr + ":")) + ((min == 30) ? min : "00"); - tripDepartLogger.info(String.format( - "%-4d %-8s %10.8f %10.8f", p, timeString, - proportions[p], cumProb)); - } - tripDepartLogger.info(loggerSeparator); - tripDepartLogger.info("rn=" + rn + ", choice=" + choice - + ", try=" + invalidCount); - tripDepartLogger.info(""); - } - stop.setStopPeriod(choice); - } else - { - invalidCount++; - } - - if (invalidCount > MAX_INVALID_FIRST_ARRIVAL_COUNT) - { - tripDepartLogger.warn("Problem in Inbound Trip Arrival Time Model."); - tripDepartLogger - .warn("Inbound trip arrive time greater than tour arrive time chosen for " - + invalidCount + " times."); - tripDepartLogger.warn("Possible infinite loop?"); - tripDepartLogger - .warn(String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourMode=%d, TourCategory=%s, TourPurpose=%s, TourId=%d, StopOrigPurpose=%s, StopDestPurpose=%s, StopId=%d, outboundStopsArray Length=%d", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), - tour.getTourModeChoice(), - tour.getTourCategory(), - tour.getTourPrimaryPurpose(), - tour.getTourId(), stop.getOrigPurpose(), - stop.getDestPurpose(), - (stop.getStopId() + 1), stops.length)); - tripDepartLogger - .warn(String - .format("prevTripPeriod=%d, tourDepartPeriod=%d, tourArrivePeriod=%d, last choice=%d", - prevTripPeriod, tour.getTourDepartPeriod(), - tour.getTourArrivePeriod(), choice)); - tripDepartLogger.warn("=" + invalidCount + " times."); - - //throw new RuntimeException(); - //instead of throwing an exception, set the stop period to the same period as the previous stop - stop.setStopPeriod(lastOutboundTripDeparts); - } - - } - - } - - } - - } - - } - - } - - /** - * - * @param probabilities - * has 1s based indexing - * @param randomNumber - * @return - */ - private int getMonteCarloSelection(double[] probabilities, double randomNumber) - { - - int returnValue = 0; - double sum = probabilities[1]; - // probabilities array passded into this method is 1s based. - for (int i = 1; i < probabilities.length - 1; i++) - { - if (randomNumber <= sum) - { - returnValue = i; - break; - } else - { - sum += probabilities[i + 1]; - returnValue = i + 1; - } - } - return returnValue; - } - - private void zeroOutCpuTimes() - { - soaAutoTime = 0; - soaOtherTime = 0; - slsTime = 0; - sldTime = 0; - slcTime = 0; - todTime = 0; - smcTime = 0; - } - - public long[] getStopTimes() - { - hhTimes[0] = soaAutoTime; - hhTimes[1] = soaOtherTime; - hhTimes[2] = slsTime; - hhTimes[3] = sldTime; - hhTimes[4] = slcTime - (soaAutoTime + soaOtherTime + slsTime); - hhTimes[5] = slcTime; - hhTimes[6] = todTime; - hhTimes[7] = smcTime; - hhTimes[8] = slcTime + sldTime + todTime + smcTime; - - return hhTimes; - } - - // this method is called to determine the parking mgra location if the stop - // location is in parkarea 1 and chosen mode is sov or hov. - private int selectParkingLocation(Household household, Tour tour, Stop stop) - { - - Logger modelLogger = parkLocLogger; - - // if the trip destination mgra is not in parking area 1, it's not - // necessary to make a parking location choice - if (mgraAltLocationIndex.containsKey(stop.getDest()) == false - || mgraAltParkArea.get(stop.getDest()) != 1) return -1; - - // if person worked at home, no reason to make a parking location choice - if (tour.getPersonObject().getFreeParkingAvailableResult() == ParkingProvisionModel.FP_MODEL_NO_REIMBURSEMENT_CHOICE) - return -1; - - // if the person has free parking, set the parking location - if (tour.getPersonObject().getFreeParkingAvailableResult() == 1) return stop.getDest(); - - parkingChoiceDmuObj.setDmuIndexValues(household.getHhId(), stop.getOrig(), stop.getDest(), - household.getDebugChoiceModels()); - - parkingChoiceDmuObj.setPersonType(tour.getPersonObject().getPersonTypeNumber()); - - Stop[] stops = null; - if (stop.isInboundStop()) stops = tour.getInboundStops(); - else stops = tour.getOutboundStops(); - - // determine activity duration in number od departure time intervals - // if no stops on halftour, activity duration is tour duration - int activityIntervals = 0; - if (stops.length == 1) - { - activityIntervals = tour.getTourArrivePeriod() - tour.getTourDepartPeriod(); - } else - { - int stopId = stop.getStopId(); - if (stopId == stops.length - 1) activityIntervals = tour.getTourArrivePeriod() - - stop.getStopPeriod(); - else activityIntervals = stops[stopId + 1].getStopPeriod() - stop.getStopPeriod(); - } - - parkingChoiceDmuObj.setActivityIntervals(activityIntervals); - - parkingChoiceDmuObj.setDestPurpose(stop.getStopPurposeIndex()); - - parkingChoiceDmuObj.setReimbPct(tour.getPersonObject().getParkingReimbursement()); - - int[] sampleIndices = setupParkLocationChoiceAlternativeArrays(stop.getOrig(), - stop.getDest()); - - // if no alternatives in the sample, it's not necessary to make a - // parking location choice - if (sampleIndices == null) return -1; - - if (household.getDebugChoiceModels()) - { - household.logHouseholdObject( - "Pre Parking Location Choice for trip: HH_" + household.getHhId() + ", Pers_" - + tour.getPersonObject().getPersonNum() + ", Tour Purpose_" - + tour.getTourPurpose() + ", Tour_" + tour.getTourId() - + ", Tour Purpose_" + tour.getTourPurpose() + ", Stop_" - + stop.getStopId(), modelLogger); - household.logPersonObject("Pre Parking Location Choice for person " - + tour.getPersonObject().getPersonNum(), modelLogger, tour.getPersonObject()); - household.logTourObject("Pre Parking Location Choice for tour " + tour.getTourId(), - modelLogger, tour.getPersonObject(), tour); - household.logStopObject("Pre Parking Location Choice for stop " + stop.getStopId(), - modelLogger, stop, modelStructure); - } - - Person person = tour.getPersonObject(); - - String choiceModelDescription = ""; - String separator = ""; - String loggerString = ""; - String decisionMakerLabel = ""; - - // log headers to traceLogger if the person making the destination - // choice is from a household requesting trace information - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = "Parking Location Choice Model for trip"; - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourPurpose=%s, TourId=%d, StopPurpose=%s, StopId=%d", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourPurpose(), tour.getTourId(), tour.getTourPurpose(), - stop.getStopId()); - - modelLogger.info(" "); - loggerString = choiceModelDescription + " for " + decisionMakerLabel + "."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - plcModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - plcModel.computeUtilities(parkingChoiceDmuObj, parkingChoiceDmuObj.getDmuIndexValues(), - altParkAvail, altParkSample); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - int chosenIndex = -1; - int parkMgra = 0; - if (plcModel.getAvailabilityCount() > 0) - { - // get the mgra number associated with the chosen alternative - chosen = plcModel.getChoiceResult(rn); - // sampleIndices is 1-based, but the values returned are 0-based, - // parkMgras is 0-based - chosenIndex = sampleIndices[chosen]; - parkMgra = parkMgras[chosenIndex]; - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels() || chosen < 0) - { - - double[] utilities = plcModel.getUtilities(); - double[] probabilities = plcModel.getProbabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("-------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - - for (int k = 1; k <= numAltsInSample; k++) - { - int index = sampleIndices[k]; - int altMgra = parkMgras[index]; - cumProb += probabilities[k - 1]; - String altString = String.format("k=%d, index=%d, altMgra=%d", k, index, altMgra); - modelLogger.info(String.format("%-35s%18.6e%18.6e%18.6e", altString, - utilities[k - 1], probabilities[k - 1], cumProb)); - } - - modelLogger.info(" "); - if (chosen < 0) - { - modelLogger.info(String.format("No Alternatives Available For Choice !!!")); - } else - { - String altString = String.format("chosen=%d, chosenIndex=%d, chosenMgra=%d", - chosen, chosenIndex, parkMgra); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - } - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - plcModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - plcModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - plcModel.logUECResults(modelLogger, loggerString); - - } - - if (chosen > 0) return parkMgra; - else - { - logger.error(String - .format("Exception caught for HHID=%d, personNum=%d, no available parking location alternatives in tourId=%d to choose from in plcModelApplication.", - household.getHhId(), person.getPersonNum(), tour.getTourId())); - return stop.getDest(); - //throw new RuntimeException(); - } - - } - - // this method is called for trips that require a park location choice -- - // trip destination in parkarea 1 and not a work trip with free onsite - // parking - // return false if no parking location alternatives are in walk distance of - // trip destination; true otherwise. - private int[] setupParkLocationChoiceAlternativeArrays(int tripOrigMgra, int tripDestMgra) - { - - // get the array of mgras within walking distance of the trip - // destination - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceTo(tripDestMgra); - - // set the distance values for the mgras walkable to the destination - if (walkMgras != null) - { - - // get distances, in feet, and convert to miles - // get distances from destMgra since this is the direction of - // distances read from the data file - int altCount = 0; - for (int wMgra : walkMgras) - { - // if wMgra is in the set of parkarea==1 MGRAs, add to list of - // alternatives for this park location choice - if (mgraAltLocationIndex.containsKey(wMgra)) - { - - double curWalkDist = mgraManager.getMgraToMgraWalkDistTo(wMgra, tripDestMgra) / 5280.0; - - if (curWalkDist > MgraDataManager.MAX_PARKING_WALK_DISTANCE) continue; - - // the hashMap stores a 0-based index - int altIndex = mgraAltLocationIndex.get(wMgra); - //int m = wMgra - 1; - int m=wMgra; - - altSdDistances[altCount + 1] = curWalkDist; - altMgraIndices[altCount + 1] = altIndex; - - altParkingCostsM[altCount + 1] = lsWgtAvgCostM[m]; - altParkingCostsD[altCount + 1] = lsWgtAvgCostD[m]; - altParkingCostsH[altCount + 1] = lsWgtAvgCostH[m]; - altMstallsoth[altCount + 1] = mstallsoth[m]; - altMstallssam[altCount + 1] = mstallssam[m]; - altMparkcost[altCount + 1] = mparkcost[m]; - altDstallsoth[altCount + 1] = dstallsoth[m]; - altDstallssam[altCount + 1] = dstallssam[m]; - altDparkcost[altCount + 1] = dparkcost[m]; - altHstallsoth[altCount + 1] = hstallsoth[m]; - altHstallssam[altCount + 1] = hstallssam[m]; - altHparkcost[altCount + 1] = hparkcost[m]; - altNumfreehrs[altCount + 1] = numfreehrs[m]; - - altParkAvail[altCount + 1] = true; - altParkSample[altCount + 1] = 1; - - altCount++; - } - } - - if (altCount > 0) - { - - for (int i = altCount; i < MAX_PLC_SAMPLE_SIZE; i++) - { - altOsDistances[i + 1] = Double.NaN; - altSdDistances[i + 1] = Double.NaN; - altMgraIndices[i + 1] = Integer.MAX_VALUE; - - altParkingCostsM[i + 1] = Double.NaN; - altParkingCostsD[i + 1] = Double.NaN; - altParkingCostsH[i + 1] = Double.NaN; - altMstallsoth[i + 1] = Integer.MAX_VALUE; - altMstallssam[i + 1] = Integer.MAX_VALUE; - altMparkcost[i + 1] = Float.MAX_VALUE; - altDstallsoth[i + 1] = Integer.MAX_VALUE; - altDstallssam[i + 1] = Integer.MAX_VALUE; - altDparkcost[i + 1] = Float.MAX_VALUE; - altHstallsoth[i + 1] = Integer.MAX_VALUE; - altHstallssam[i + 1] = Integer.MAX_VALUE; - altHparkcost[i + 1] = Float.MAX_VALUE; - altNumfreehrs[i + 1] = Integer.MAX_VALUE; - - altParkAvail[i + 1] = false; - altParkSample[i + 1] = 0; - } - numAltsInSample = altCount; - if (numAltsInSample > maxAltsInSample) maxAltsInSample = numAltsInSample; - } - - return altMgraIndices; - - } - - return null; - - } - - public int getMaxAltsInSample() - { - return maxAltsInSample; - } - - public HashMap getSizeSegmentNameIndexMap() - { - return sizeSegmentNameIndexMap; - } - - public double[][] getSizeSegmentArray() - { - return slcSizeTerms; - } - /** - * This method calculates a cost coefficient based on the following formula: - * - * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) - * - * - * @param incomeCoeff - * @param incomeExponent - * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice - */ - public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ - - return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java deleted file mode 100644 index 3b00f69..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceDMU.java +++ /dev/null @@ -1,113 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * @author crf
    - * Started: Apr 14, 2009 11:09:58 AM - */ -public class InternalExternalTripChoiceDMU - implements Serializable, VariableTable -{ - - protected HashMap methodIndexMap; - - private Household hh; - private Person person; - - private double distanceToCordonsLogsum; - private double vehiclesPerHouseholdMember; - - private IndexValues iv; - - public InternalExternalTripChoiceDMU() - { - iv = new IndexValues(); - } - - public void setDmuIndexValues(int hhid, int hhtaz) - { - iv.setHHIndex(hhid); - iv.setZoneIndex(hhtaz); - iv.setDebug(hh.getDebugChoiceModels()); - } - - public IndexValues getDmuIndexValues() - { - return iv; - } - - public void setHouseholdObject(Household hhObj) - { - hh = hhObj; - } - - public void setPersonObject(Person persObj) - { - person = persObj; - } - - public void setDistanceToCordonsLogsum(double value) - { - distanceToCordonsLogsum = value; - } - - public double getDistanceToCordonsLogsum() - { - return distanceToCordonsLogsum; - } - - public void setVehiclesPerHouseholdMember(double value) - { - vehiclesPerHouseholdMember = value; - } - - public double getVehiclesPerHouseholdMember() - { - return vehiclesPerHouseholdMember; - } - - public int getHhIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - public int getAge() - { - return person.getAge(); - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java deleted file mode 100644 index eb856d0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/InternalExternalTripChoiceModel.java +++ /dev/null @@ -1,120 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; - -import org.apache.log4j.Logger; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class InternalExternalTripChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("ie"); - - private static final String IE_CONTROL_FILE_TARGET = "ie.uec.file"; - private static final String IE_DATA_SHEET_TARGET = "ie.data.page"; - private static final String IE_MODEL_SHEET_TARGET = "ie.model.page"; - - public static final int IE_MODEL_NO_ALT = 1; - public static final int IE_MODEL_YES_ALT = 2; - - private ChoiceModelApplication ieModel; - private InternalExternalTripChoiceDMU ieDmuObject; - private ModelStructure modelStructure; - - public InternalExternalTripChoiceModel(HashMap propertyMap,ModelStructure myModelStructure, - CtrampDmuFactoryIf dmuFactory) - { - - logger.info("setting up internal-external trip choice model."); - - - modelStructure = myModelStructure; - // locate the IE choice UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String ieUecFile = uecFileDirectory + propertyMap.get(IE_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, IE_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, IE_MODEL_SHEET_TARGET); - - // create the choice model DMU object. - ieDmuObject = dmuFactory.getInternalExternalTripChoiceDMU(); - - // create the choice model object - ieModel = new ChoiceModelApplication(ieUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) ieDmuObject); - - } - - public void applyModel(Household hhObject, double[] distanceToCordonsLogsums) - { - - int homeTaz = hhObject.getHhTaz(); - ieDmuObject.setDistanceToCordonsLogsum(distanceToCordonsLogsums[homeTaz]); - - ieDmuObject.setHouseholdObject(hhObject); - double vehiclesPerHouseholdMember = hhObject.getAutosOwned() - / hhObject.getHhSize(); - ieDmuObject.setVehiclesPerHouseholdMember(vehiclesPerHouseholdMember); - - Random hhRandom = hhObject.getHhRandom(); - - // person array is 1-based - Person[] person = hhObject.getPersons(); - for (int i = 1; i < person.length; i++) - { - - ieDmuObject.setPersonObject(person[i]); - ieDmuObject.setDmuIndexValues(hhObject.getHhId(), hhObject.getHhTaz()); - - double randomNumber = hhRandom.nextDouble(); - - // compute utilities and choose alternative. - float logsum = (float) ieModel.computeUtilities(ieDmuObject, ieDmuObject.getDmuIndexValues()); - person[i].setIeLogsum(logsum); - - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - if (ieModel.getAvailabilityCount() > 0) - { - chosenAlt = ieModel.getChoiceResult(randomNumber); - } else - { - String decisionMaker = String.format("HHID=%d, PersonNum=%d", hhObject.getHhId(), - person[i].getPersonNum()); - String errorMessage = String - .format("Exception caught for %s, no available internal-external trip choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - ieModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - // write choice model alternative info to log file - if (hhObject.getDebugChoiceModels()) - { - String decisionMaker = String.format("HHID=%d, PersonNum=%d", hhObject.getHhId(), - person[i].getPersonNum()); - ieModel.logAlternativesInfo("Internal-External Trip Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d with rn %.8f", - "Internal-External Trip Choice", decisionMaker, chosenAlt, randomNumber)); - ieModel.logUECResults(logger, decisionMaker); - } - - person[i].setInternalExternalTripChoiceResult(chosenAlt); - - - } - - hhObject.setIeRandomCount(hhObject.getHhRandomCount()); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java deleted file mode 100644 index c886c22..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModels.java +++ /dev/null @@ -1,916 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.net.UnknownHostException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To - * change this template use File | Settings | File Templates. - */ -public class JointTourModels - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(JointTourModels.class); - private transient Logger tourFreq = Logger.getLogger("tourFreq"); - - // these are public because CtrampApplication creates a - // ChoiceModelAppplication in order to get the alternatives for a log report - public static final String UEC_FILE_PROPERTIES_TARGET = "jtfcp.uec.file"; - public static final String UEC_DATA_PAGE_TARGET = "jtfcp.data.page"; - public static final String UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE = "jtfcp.freq.comp.page"; - - private static final String UEC_JOINT_TOUR_PARTIC_MODEL_PAGE = "jtfcp.participate.page"; - - public static final int JOINT_TOUR_COMPOSITION_ADULTS = 1; - public static final int JOINT_TOUR_COMPOSITION_CHILDREN = 2; - public static final int JOINT_TOUR_COMPOSITION_MIXED = 3; - - public static final String[] JOINT_TOUR_COMPOSITION_NAMES = {"", "adult", "child", - "mixed" }; - - public static final int PURPOSE_1_FIELD = 2; - public static final int PURPOSE_2_FIELD = 3; - public static final int PARTY_1_FIELD = 4; - public static final int PARTY_2_FIELD = 5; - - // DMU for the UEC - private JointTourModelsDMU dmuObject; - private AccessibilitiesTable accTable; - - private ChoiceModelApplication jointTourFrequencyModel; - private ChoiceModelApplication jointTourParticipation; - private TableDataSet jointModelsAltsTable; - private HashMap purposeIndexNameMap; - - private int[] invalidCount = new int[5]; - - private String threadName = null; - - public JointTourModels(HashMap propertyMap, AccessibilitiesTable myAccTable, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory) - { - - accTable = myAccTable; - - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + ": " - + Thread.currentThread().getName() + "]"; - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - setUpModels(propertyMap, modelStructure, dmuFactory); - } - - public void setUpModels(HashMap propertyMap, ModelStructure modelStructure, - CtrampDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up %s tour frequency model on %s", - ModelStructure.JOINT_NON_MANDATORY_CATEGORY, threadName)); - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - String uecFileName = propertyMap.get(UEC_FILE_PROPERTIES_TARGET); - uecFileName = uecFileDirectory + uecFileName; - - dmuObject = dmuFactory.getJointTourModelsDMU(); - - purposeIndexNameMap = new HashMap(); - purposeIndexNameMap.put(ModelStructure.SHOPPING_PRIMARY_PURPOSE_INDEX, - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); - purposeIndexNameMap.put(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_INDEX, - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); - purposeIndexNameMap.put(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_INDEX, - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); - purposeIndexNameMap.put(ModelStructure.VISITING_PRIMARY_PURPOSE_INDEX, - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); - purposeIndexNameMap.put(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_INDEX, - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); - - int dataSheet = Integer.parseInt(propertyMap.get(UEC_DATA_PAGE_TARGET)); - int freqCompSheet = Integer.parseInt(propertyMap.get(UEC_JOINT_TOUR_FREQ_COMP_MODEL_PAGE)); - int particSheet = Integer.parseInt(propertyMap.get(UEC_JOINT_TOUR_PARTIC_MODEL_PAGE)); - - // set up the models - jointTourFrequencyModel = new ChoiceModelApplication(uecFileName, freqCompSheet, dataSheet, - propertyMap, (VariableTable) dmuObject); - jointModelsAltsTable = jointTourFrequencyModel.getUEC().getAlternativeData(); - modelStructure.setJtfAltLabels(jointTourFrequencyModel.getAlternativeNames()); - - jointTourParticipation = new ChoiceModelApplication(uecFileName, particSheet, dataSheet, - propertyMap, (VariableTable) dmuObject); - } - - public void applyModel(Household household) - { - - // this household does not make joint tours if the CDAP pattern does not - // contain "j". - if (!household.getCoordinatedDailyActivityPattern().contains("j")) return; - - household.calculateTimeWindowOverlaps(); - - try - { - - // joint tour frequency choice is not applied to a household unless - // it has: - // 2 or more persons, each with at least one out-of home activity, - // and at - // least 1 of the persons not a pre-schooler. - - Logger modelLogger = tourFreq; - if (household.getDebugChoiceModels()) - household.logHouseholdObject( - "Pre Joint Tour Frequency Choice HHID=" + household.getHhId() + " Object", - modelLogger); - - // if it's not a valid household for joint tour frequency, keep - // track of - // count for logging later, and return. - int validIndex = household.getValidHouseholdForJointTourFrequencyModel(); - if (validIndex != 1) - { - invalidCount[validIndex]++; - switch (validIndex) - { - case 2: - household.setJointTourFreqResult(-2, "-2_1 person"); - break; - case 3: - household.setJointTourFreqResult(-3, "-3_< 2 travel"); - break; - case 4: - household.setJointTourFreqResult(-4, "-4_only preschool travel"); - break; - } - return; - } - - // set the household id, origin taz, hh taz, and debugFlag=false in - // the dmu - dmuObject.setHouseholdObject(household); - - // set the accessibility values needed based on auto sufficiency - // category for the hh. - if (household.getAutoSufficiency() == 1) - { - dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov0", - household.getHhMgra())); - dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov0", - household.getHhMgra())); - dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov0", - household.getHhMgra())); - } else if (household.getAutoSufficiency() == 2) - { - dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov1", - household.getHhMgra())); - dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov1", - household.getHhMgra())); - dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov1", - household.getHhMgra())); - } else if (household.getAutoSufficiency() == 3) - { - dmuObject.setShopHOVAccessibility(accTable.getAggregateAccessibility("shopHov2", - household.getHhMgra())); - dmuObject.setMaintHOVAccessibility(accTable.getAggregateAccessibility("maintHov2", - household.getHhMgra())); - dmuObject.setDiscrHOVAccessibility(accTable.getAggregateAccessibility("discrHov2", - household.getHhMgra())); - } - - IndexValues index = dmuObject.getDmuIndexValues(); - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Joint Non-Mandatory Tour Frequency Choice Model:"); - decisionMakerLabel = String.format("HH=%d, hhSize=%d.", household.getHhId(), - household.getHhSize()); - - jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - } - - float logsum = (float) jointTourFrequencyModel.computeUtilities(dmuObject, index); - household.setJtfLogsum(logsum); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosenFreqAlt = -1; - if (jointTourFrequencyModel.getAvailabilityCount() > 0) - { - chosenFreqAlt = jointTourFrequencyModel.getChoiceResult(rn); - household.setJointTourFreqResult(chosenFreqAlt, - jointTourFrequencyModel.getAlternativeNames()[chosenFreqAlt - 1]); - } else - { - logger.error(String - .format("Exception caught for HHID=%d, no available joint tour frequency alternatives to choose from in choiceModelApplication.", - household.getHhId())); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - String[] altNames = jointTourFrequencyModel.getAlternativeNames(); - - double[] utilities = jointTourFrequencyModel.getUtilities(); - double[] probabilities = jointTourFrequencyModel.getProbabilities(); - - modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " - + household.getHhSize()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %10s", k + 1, altNames[k]); - modelLogger.info(String.format("%-15s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %10s", chosenFreqAlt, - altNames[chosenFreqAlt - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - jointTourFrequencyModel.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - jointTourFrequencyModel.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosenFreqAlt); - - // write UEC calculation results to separate model specific log - // file - jointTourFrequencyModel.logUECResults(modelLogger, loggingHeader); - - } - - createJointTours(household, chosenFreqAlt); - - } catch (Exception e) - { - logger.error(String.format("error joint tour choices model for hhId=%d.", - household.getHhId())); - throw new RuntimeException(); - } - - household.setJtfRandomCount(household.getHhRandomCount()); - - } - - private void jointTourParticipation(Tour jointTour) - { - - // get the Household object for this joint tour - Household household = dmuObject.getHouseholdObject(); - - // get the array of Person objects for this hh - Person[] persons = household.getPersons(); - - // define an ArrayList to hold indices of person objects participating - // in the joint tour - ArrayList jointTourPersonList = null; - - // make sure each joint tour has a valid composition before going to the - // next one. - boolean validParty = false; - - int adults = 0; - int children = 0; - - Logger modelLogger = tourFreq; - - int loopCount = 0; - while (!validParty) - { - - dmuObject.setTourObject(jointTour); - - adults = 0; - children = 0; - - jointTourPersonList = new ArrayList(); - - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - jointTour.setPersonObject(persons[p]); - - if (household.getDebugChoiceModels() || loopCount == 1000) - { - decisionMakerLabel = String.format( - "HH=%d, hhSize=%d, PersonNum=%d, PersonType=%s, tourId=%d.", - household.getHhId(), household.getHhSize(), person.getPersonNum(), - person.getPersonType(), jointTour.getTourId()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - // if person type is inconsistent with tour composition, - // person's - // participation is by definition no, - // so skip making the choice and go to next person - switch (jointTour.getJointTourComposition()) - { - - // adults only in joint tour - case 1: - if (persons[p].getPersonIsAdult() == 1) - { - - // write debug header - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - choiceModelDescription = String - .format("Adult Party Joint Tour Participation Choice Model:"); - jointTourParticipation.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " - + decisionMakerLabel + "."; - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - } - - jointTourParticipation.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - int chosen = -1; - if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for HHID=%d, person p=%d, no available adults only joint tour participation alternatives to choose from in choiceModelApplication.", - jointTour.getHhId(), p)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - String[] altNames = jointTourParticipation.getAlternativeNames(); - - double[] utilities = jointTourParticipation.getUtilities(); - double[] probabilities = jointTourParticipation.getProbabilities(); - - modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " - + household.getHhSize() + ", tourId: " - + jointTour.getTourId() + ", jointFreqChosen: " - + household.getJointTourFreqChosenAltName()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %13s", k + 1, - altNames[k]); - modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", - altString, utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %13s", chosen, - altNames[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug - // log - // file - jointTourParticipation.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - jointTourParticipation.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate - // model - // specific log file - jointTourParticipation.logUECResults(modelLogger, loggingHeader); - - if (loopCount == 1000) - { - jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - jointTourFrequencyModel.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - jointTourFrequencyModel.logUECResults(modelLogger, - loggingHeader); - } - - } - - // particpate is alternative 1, not participating is - // alternative 2. - if (chosen == 1) - { - jointTourPersonList.add(p); - adults++; - } - } - break; - - // children only in joint tour - case 2: - if (persons[p].getPersonIsAdult() == 0) - { - - // write debug header - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - choiceModelDescription = String - .format("Child Party Joint Tour Participation Choice Model:"); - jointTourParticipation.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " - + decisionMakerLabel + "."; - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - } - - jointTourParticipation.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - int chosen = -1; - if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for HHID=%d, person p=%d, no available children only joint tour participation alternatives to choose from in choiceModelApplication.", - jointTour.getHhId(), p)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - String[] altNames = jointTourParticipation.getAlternativeNames(); - - double[] utilities = jointTourParticipation.getUtilities(); - double[] probabilities = jointTourParticipation.getProbabilities(); - - modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " - + household.getHhSize() + ", tourId: " - + jointTour.getTourId()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %13s", k + 1, - altNames[k]); - modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", - altString, utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %13s", chosen, - altNames[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug - // log - // file - jointTourParticipation.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - jointTourParticipation.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate - // model - // specific log file - jointTourParticipation.logUECResults(modelLogger, loggingHeader); - - if (loopCount == 1000) - { - jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - jointTourFrequencyModel.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - jointTourFrequencyModel.logUECResults(modelLogger, - loggingHeader); - } - } - - // particpate is alternative 1, not participating is - // alternative 2. - if (chosen == 1) - { - jointTourPersonList.add(p); - children++; - } - } - break; - - // mixed, adults and children in joint tour - case 3: - - // write debug header - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - choiceModelDescription = String - .format("Mixed Party Joint Tour Participation Choice Model:"); - jointTourParticipation.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " + decisionMakerLabel - + "."; - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - } - - jointTourParticipation.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - int chosen = -1; - if (jointTourParticipation.getAvailabilityCount() > 0) chosen = jointTourParticipation - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught for HHID=%d, person p=%d, no available mixed adult/children joint tour participation alternatives to choose from in choiceModelApplication.", - jointTour.getHhId(), p)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels() || loopCount == 1000) - { - - String[] altNames = jointTourParticipation.getAlternativeNames(); - - double[] utilities = jointTourParticipation.getUtilities(); - double[] probabilities = jointTourParticipation.getProbabilities(); - - modelLogger.info("HHID: " + household.getHhId() + ", HHSize: " - + household.getHhSize() + ", tourId: " + jointTour.getTourId()); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %13s", k + 1, altNames[k]); - modelLogger.info(String.format("%-18s%18.6e%18.6e%18.6e", - altString, utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %13s", chosen, - altNames[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log - // file - jointTourParticipation.logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - jointTourParticipation.logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model - // specific log file - jointTourParticipation.logUECResults(modelLogger, loggingHeader); - - if (loopCount == 1000) - { - jointTourFrequencyModel.choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - jointTourFrequencyModel.computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - jointTourFrequencyModel.logUECResults(modelLogger, loggingHeader); - } - } - - // particpate is alternative 1, not participating is - // alternative 2. - if (chosen == 1) - { - jointTourPersonList.add(p); - if (persons[p].getPersonIsAdult() == 1) adults++; - else children++; - } - break; - - } - - } - - // done with all persons, so see if the chosen participation is a - // valid - // composition, and if not, repeat the participation choice. - switch (jointTour.getJointTourComposition()) - { - - case 1: - if (adults > 1 && children == 0) validParty = true; - break; - - case 2: - if (adults == 0 && children > 1) validParty = true; - break; - - case 3: - if (adults > 0 && children > 0) validParty = true; - break; - - } - - if (!validParty) loopCount++; - - if (loopCount > 1000) - { - logger.warn("loop count in joint tour participation model is " + loopCount); - if (loopCount > 2000) - { - logger.warn("joint tour party composition-terminating on excessive loop count."); - //logger.error("terminating on excessive loop count."); - //throw new RuntimeException(); - } - } - - } // end while - - // create an array of person indices for participation in the tour - int[] personNums = new int[jointTourPersonList.size()]; - for (int i = 0; i < personNums.length; i++) - personNums[i] = jointTourPersonList.get(i); - jointTour.setPersonNumArray(personNums); - - if (household.getDebugChoiceModels()) - { - for (int i = 0; i < personNums.length; i++) - { - Person person = household.getPersons()[personNums[i]]; - String decisionMakerLabel = String - .format("Person in Party, HH=%d, hhSize=%d, PersonNum=%d, PersonType=%s, tourId=%d.", - household.getHhId(), household.getHhSize(), person.getPersonNum(), - person.getPersonType(), jointTour.getTourId()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - } - - } - - /** - * creates the tour objects in the Household object given the chosen joint - * tour frequency alternative. - * - * @param chosenAlt - */ - private void createJointTours(Household household, int chosenAlt) - { - - int purposeIndex1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PURPOSE_1_FIELD); - int purposeIndex2 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PURPOSE_2_FIELD); - - if (purposeIndex1 > 0 && purposeIndex2 > 0) - { - - Tour t1 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex1), - ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex1); - int party1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_1_FIELD); - t1.setJointTourComposition(party1); - - Tour t2 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex2), - ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex2); - int party2 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_2_FIELD); - t2.setJointTourComposition(party2); - - household.createJointTourArray(t1, t2); - - jointTourParticipation(t1); - jointTourParticipation(t2); - - } else - { - - Tour t1 = new Tour(household, (String) purposeIndexNameMap.get(purposeIndex1), - ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purposeIndex1); - int party1 = (int) jointModelsAltsTable.getValueAt(chosenAlt, PARTY_1_FIELD); - t1.setJointTourComposition(party1); - - household.createJointTourArray(t1); - - jointTourParticipation(t1); - - } - - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - /* - * - */ - String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); - - MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, - Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TazDataManager tazManager = TazDataManager.getInstance(propertyMap); - - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - AccessibilitiesTable accTable = new AccessibilitiesTable(accFileName); - - String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); - int hhServerPort = Integer.parseInt((String) propertyMap - .get("RunModel.HouseholdServerPort")); - - HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, - hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - - householdDataManager.setPropertyFileValues(propertyMap); - householdDataManager.setHouseholdSampleRate(1.0f, 0); - householdDataManager.setDebugHhIdsFromHashmap(); - householdDataManager.setTraceHouseholdSet(); - - int id = householdDataManager.getArrayIndex(423804); - Household[] hh = householdDataManager.getHhArray(id, id); - - JointTourModels jtfModel = new JointTourModels(propertyMap, accTable, modelStructure, - dmuFactory); - jtfModel.applyModel(hh[0]); - - /* - * Use this block to instantiate a UEC for the joint freq/comp model and - * a UEC for the joint participate model. After checking the UECs are - * instantiated correctly (spelling/typos/dmu methods implemented/etc.), - * test model implementation. String uecFileDirectory = propertyMap.get( - * CtrampApplication.PROPERTIES_UEC_PATH ); - * - * ModelStructure modelStructure = new SandagModelStructure(); - * SandagCtrampDmuFactory dmuFactory = new - * SandagCtrampDmuFactory(modelStructure); - * - * JointTourModelsDMU dmuObject = dmuFactory.getJointTourModelsDMU(); - * File uecFile = new File(uecFileDirectory + propertyMap.get( - * UEC_FILE_PROPERTIES_TARGET )); UtilityExpressionCalculator uec = new - * UtilityExpressionCalculator(uecFile, 1, 0, propertyMap, - * (VariableTable)dmuObject); - * System.out.println("Jount tour freq/comp choice UEC passed."); - * - * uec = new UtilityExpressionCalculator(uecFile, 2, 0, propertyMap, - * (VariableTable)dmuObject); - * System.out.println("Joint tour participation choice UEC passed."); - */ - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java deleted file mode 100644 index efadb79..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/JointTourModelsDMU.java +++ /dev/null @@ -1,310 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class JointTourModelsDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); - - protected HashMap methodIndexMap; - - protected Household hh; - protected Tour tour; - protected IndexValues dmuIndex; - - private float shopHOVAccessibility; - private float maintHOVAccessibility; - private float discrHOVAccessibility; - - public JointTourModelsDMU() - { - dmuIndex = new IndexValues(); - } - - public Household getHouseholdObject() - { - return hh; - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public void setTourObject(Tour tourObject) - { - tour = tourObject; - } - - // DMU methods - define one of these for every @var in the mode choice - // control - // file. - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getActiveCountFullTimeWorkers() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsFullTimeWorker() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountPartTimeWorkers() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsPartTimeWorker() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountUnivStudents() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsUniversityStudent() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountNonWorkers() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsNonWorkingAdultUnder65() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountRetirees() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsNonWorkingAdultOver65() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountDrivingAgeSchoolChildren() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsStudentDriving() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountPreDrivingAgeSchoolChildren() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsStudentNonDriving() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getActiveCountPreSchoolChildren() - { - int count = 0; - for (Person p : hh.getPersons()) - if (p != null && p.getPersonIsPreschoolChild() == 1) - if (!p.getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) count++; - return count; - } - - public int getMaxPairwiseAdultOverlapsHh() - { - return hh.getMaxAdultOverlaps(); - } - - public int getMaxPairwiseChildOverlapsHh() - { - return hh.getMaxChildOverlaps(); - } - - public int getMaxPairwiseMixedOverlapsHh() - { - return hh.getMaxMixedOverlaps(); - } - - public int getMaxPairwiseOverlapOtherAdults() - { - return tour.getPersonObject().getMaxAdultOverlaps(); - } - - public int getMaxPairwiseOverlapOtherChildren() - { - return tour.getPersonObject().getMaxChildOverlaps(); - } - - public int getTravelActiveAdults() - { - return hh.getTravelActiveAdults(); - } - - public int getTravelActiveChildren() - { - return hh.getTravelActiveChildren(); - } - - public int getPersonStaysHome() - { - Person p = tour.getPersonObject(); - return p.getCdapActivity().equalsIgnoreCase("H") ? 1 : 0; - } - - public int getIncomeLessThan30K() - { - return hh.getIncomeInDollars() < 30000 ? 1 : 0; - } - - public int getIncome30Kto60K() - { - int income = hh.getIncomeInDollars(); - return (income >= 30000 && income < 60000) ? 1 : 0; - } - - public int getIncomeMoreThan100K() - { - return hh.getIncomeInDollars() >= 100000 ? 1 : 0; - } - - public int getNumAdults() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - num += (persons[i].getPersonIsAdult() == 1 ? 1 : 0); - return num; - } - - public int getNumChildren() - { - int num = 0; - Person[] persons = hh.getPersons(); - for (int i = 1; i < persons.length; i++) - num += (persons[i].getPersonIsAdult() == 0 ? 1 : 0); - return num; - } - - public int getHhWorkers() - { - return hh.getWorkers(); - } - - public int getAutoOwnership() - { - return hh.getAutosOwned(); - } - - public int getTourPurposeIsMaint() - { - return tour.getTourPurpose() - .equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME) ? 1 : 0; - } - - public int getTourPurposeIsEat() - { - return tour.getTourPurpose().equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) ? 1 - : 0; - } - - public int getTourPurposeIsVisit() - { - return tour.getTourPurpose().equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) ? 1 - : 0; - } - - public int getTourPurposeIsDiscr() - { - return tour.getTourPurpose() - .equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME) ? 1 : 0; - } - - public int getPersonType() - { - return tour.getPersonObject().getPersonTypeNumber(); - } - - public int getJointTourComposition() - { - return tour.getJointTourComposition(); - } - - public int getJTours() - { - return hh.getJointTourArray().length; - } - - public void setShopHOVAccessibility(float accessibility) - { - shopHOVAccessibility = accessibility; - } - - public float getShopHOVAccessibility() - { - return shopHOVAccessibility; - } - - public void setMaintHOVAccessibility(float accessibility) - { - maintHOVAccessibility = accessibility; - } - - public float getMaintHOVAccessibility() - { - return maintHOVAccessibility; - } - - public void setDiscrHOVAccessibility(float accessibility) - { - discrHOVAccessibility = accessibility; - } - - public float getDiscrHOVAccessibility() - { - return discrHOVAccessibility; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java deleted file mode 100644 index ebc8c62..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MandatoryDestChoiceModel.java +++ /dev/null @@ -1,885 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.IndexSort; - -public class MandatoryDestChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(MandatoryDestChoiceModel.class); - private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); - - // this constant used as a dimension for saving distance and logsums for - // alternatives in samples - private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; - - private static final int DC_DATA_SHEET = 0; - private static final int DC_WORK_AT_HOME_SHEET = 1; - - private MgraDataManager mgraManager; - private DestChoiceSize dcSizeObj; - - private DestChoiceDMU dcDmuObject; - private DcSoaDMU dcSoaDmuObject; - - private TourModeChoiceModel mcModel; - private DestinationSampleOfAlternativesModel dcSoaModel; - - private String[] segmentNameList; - private HashMap segmentNameIndexMap; - private HashMap workOccupValueSegmentIndexMap; - - private int[] dcModelIndices; - - // A ChoiceModelApplication object and modeAltsAvailable[] is needed for - // each purpose - private ChoiceModelApplication[] locationChoiceModels; - private ChoiceModelApplication locationChoiceModel; - private ChoiceModelApplication worksAtHomeModel; - - private int[] uecSheetIndices; - private int[] soaUecSheetIndices; - - int origMgra; - - private int modelIndex; - private int shadowPricingIteration; - - private double[] sampleAlternativeDistances; - private double[] sampleAlternativeLogsums; - - private double[] mgraDistanceArray; - - private BuildAccessibilities aggAcc; - - public MandatoryDestChoiceModel(int index, HashMap propertyMap, - DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, MgraDataManager mgraManager, - String dcUecFileName, String soaUecFile, int soaSampleSize, String modeChoiceUecFile, - CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel) - { - - // set the model structure and the tour purpose list - this.mgraManager = mgraManager; - this.aggAcc = aggAcc; - this.dcSizeObj = dcSizeObj; - this.mcModel = mcModel; - - modelIndex = index; - - dcDmuObject = dmuFactory.getDestChoiceDMU(); - dcDmuObject.setAggAcc(aggAcc); - - dcSoaDmuObject = dmuFactory.getDcSoaDMU(); - dcSoaDmuObject.setAggAcc(aggAcc); - - shadowPricingIteration = 0; - - sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - - workOccupValueSegmentIndexMap = aggAcc.getWorkOccupValueIndexMap(); - - } - - public void setupWorkSegments(int[] myUecSheetIndices, int[] mySoaUecSheetIndices) - { - uecSheetIndices = myUecSheetIndices; - soaUecSheetIndices = mySoaUecSheetIndices; - segmentNameList = aggAcc.getWorkSegmentNameList(); - } - - public void setupSchoolSegments() - { - aggAcc.createSchoolSegmentNameIndices(); - uecSheetIndices = aggAcc.getSchoolDcUecSheets(); - soaUecSheetIndices = aggAcc.getSchoolDcSoaUecSheets(); - segmentNameList = aggAcc.getSchoolSegmentNameList(); - } - - public void setupDestChoiceModelArrays(HashMap propertyMap, - String dcUecFileName, String soaUecFile, int soaSampleSize) - { - - segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); - - // create a sample of alternatives choice model object for use in - // selecting a sample - // of all possible destination choice alternatives. - dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFile, soaSampleSize, - propertyMap, mgraManager, dcSizeObj.getDcSizeArray(), dcSoaDmuObject, - soaUecSheetIndices); - - // create the works-at-home ChoiceModelApplication object - worksAtHomeModel = new ChoiceModelApplication(dcUecFileName, DC_WORK_AT_HOME_SHEET, - DC_DATA_SHEET, propertyMap, (VariableTable) dcDmuObject); - - dcSoaModel.setAvailabilityForSampleOfAlternatives(dcSizeObj.getDcSizeArray()); - - // create a lookup array to map purpose index to model index - dcModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int dcModelIndex = 0; - int dcSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, dcModelIndex); - dcModelIndices[dcSegmentIndex] = dcModelIndex++; - } else - { - dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); - } - - dcSegmentIndex++; - } - // the value of dcModelIndex is the number of ChoiceModelApplication - // objects to create - // the modelIndexMap keys are the uec sheets to use in building - // ChoiceModelApplication objects - - locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; - - int i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - - int modelIndex = -1; - try - { - modelIndex = modelIndexMap.get(uecIndex); - locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, - uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcDmuObject); - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", - i, modelIndex, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - throw new RuntimeException(); - } - - } - - mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; - - } - - public void applyWorkLocationChoice(Household hh) - { - - if (hh.getDebugChoiceModels()) - { - String label = String.format("Pre Work Location Choice HHId=%d Object", hh.getHhId()); - hh.logHouseholdObject(label, dcManLogger); - } - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - int occupSegmentIndex = -1; - int occup = -1; - String occupSegmentName = ""; - - Person[] persons = hh.getPersons(); - - int tourNum = 0; - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - // skip person if they are not a worker - if (p.getPersonIsWorker() != 1) - { - p.setWorkLocationSegmentIndex(-1); - p.setWorkLocation(0); - p.setWorkLocDistance(0); - p.setWorkLocLogsum(-999); - continue; - } - - // skip person if their work at home choice was work in the home - // (alternative 2 in choice model) - int worksAtHomeChoice = selectWorksAtHomeChoice(dcDmuObject, hh, p); - if (worksAtHomeChoice == ModelStructure.WORKS_AT_HOME_ALTERNATUVE_INDEX) - { - p.setWorkLocationSegmentIndex(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); - p.setWorkLocation(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); - p.setWorkLocDistance(0); - p.setWorkLocLogsum(-999); - continue; - } - - // save person information in decision maker label, and log person - // object - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p - .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); - hh.logPersonObject(decisionMakerLabel, dcManLogger, p); - } - - double[] results = null; - try - { - - int homeMgra = hh.getHhMgra(); - origMgra = homeMgra; - - occup = p.getPersPecasOccup(); - occupSegmentIndex = workOccupValueSegmentIndexMap.get(occup); - occupSegmentName = segmentNameList[occupSegmentIndex]; - - p.setWorkLocationSegmentIndex(occupSegmentIndex); - - // update the DC dmuObject for this person - dcDmuObject.setHouseholdObject(hh); - dcDmuObject.setPersonObject(p); - dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); - - double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[occupSegmentIndex]; - mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, - mgraDistanceArray); - - // set size array for the tour segment and distance array from - // the home mgra to all destinaion mgras. - dcSoaDmuObject.setDestChoiceSize(homeMgraSizeArray); - dcSoaDmuObject.setDestDistance(mgraDistanceArray); - - dcDmuObject.setDestChoiceSize(homeMgraSizeArray); - dcDmuObject.setDestChoiceDistance(mgraDistanceArray); - - int choiceModelIndex = dcModelIndices[occupSegmentIndex]; - locationChoiceModel = locationChoiceModels[choiceModelIndex]; - - // get the work location alternative chosen from the sample - results = selectLocationFromSampleOfAlternatives("work", -1, p, occupSegmentName, - occupSegmentIndex, tourNum++, homeMgraSizeArray, mgraDistanceArray); - - } catch (RuntimeException e) - { - logger.fatal(String - .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in work location choice, occup=%d, segmentIndex=%d, segmentName=%s", - i, hh.getHhId(), i, occup, occupSegmentIndex, occupSegmentName)); - logger.fatal("Exception caught:", e); - logger.fatal("calling System.exit(-1) to terminate."); - System.exit(-1); - } - - p.setWorkLocation((int) results[0]); - p.setWorkLocDistance((float) results[1]); - p.setWorkLocLogsum((float) results[2]); - - } - - } - - public void applySchoolLocationChoice(Household hh) - { - - if (hh.getDebugChoiceModels()) - { - String label = String.format("Pre school Location Choice HHId=%d Object", hh.getHhId()); - hh.logHouseholdObject(label, dcManLogger); - } - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - - int homeMgra = hh.getHhMgra(); - Person[] persons = hh.getPersons(); - - int tourNum = 0; - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - int segmentIndex = -1; - int segmentType = -1; - if (p.getPersonIsPreschoolChild() == 1 || p.getPersonIsStudentNonDriving() == 1 - || p.getPersonIsStudentDriving() == 1 || p.getPersonIsUniversityStudent() == 1) - { - - if (p.getPersonIsPreschoolChild() == 1) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.PRESCHOOL_ALT_INDEX; - } else if (p.getPersonIsGradeSchool() == 1) - { - segmentIndex = aggAcc.getMgraGradeSchoolSegmentIndex(homeMgra); - segmentType = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; - } else if (p.getPersonIsHighSchool() == 1) - { - segmentIndex = aggAcc.getMgraHighSchoolSegmentIndex(homeMgra); - segmentType = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; - } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; - } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; - } - - // if person type is a student but segment index is -1, the - // person is not enrolled - // assume home schooled - if (segmentIndex < 0) - { - p.setSchoolLocationSegmentIndex(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); - p.setSchoolLoc(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); - p.setSchoolLocDistance(0); - p.setSchoolLocLogsum(-999); - continue; - } else - { - // if the segment is in the skip shadow pricing set, and the - // iteration is > 0, dont' compute new choice - if (shadowPricingIteration == 0 - || !dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) - p.setSchoolLocationSegmentIndex(segmentIndex); - } - - if (segmentType < 0) - { - segmentType = ModelStructure.NOT_ENROLLED_SEGMENT_INDEX; - } - } else // not a student person type - { - p.setSchoolLocationSegmentIndex(-1); - p.setSchoolLoc(0); - p.setSchoolLocDistance(0); - p.setSchoolLocLogsum(-999); - continue; - } - - // save person information in decision maker label, and log person - // object - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p - .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); - hh.logPersonObject(decisionMakerLabel, dcManLogger, p); - } - - // if the segment is in the skip shadow pricing set, and the - // iteration is > 0, dont' compute new choice - if (shadowPricingIteration > 0 && dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) - continue; - - double[] results = null; - int modelIndex = 0; - try - { - - origMgra = homeMgra; - - // update the DC dmuObject for this person - dcDmuObject.setHouseholdObject(hh); - dcDmuObject.setPersonObject(p); - dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); - - /** - * remove following - don't need non-mandatory accessibility - * since we're doing shadow pricing for school tours // set the - * auto sufficiency dependent non-mandatory accessibility value - * for the household int autoSufficiency = - * hh.getAutoSufficiency(); float accessibility = - * aggAcc.getAggregateAccessibility( - * nonMandatoryAccessibilityTypes[autoSufficiency], - * hh.getHhMgra() ); dcDmuObject.setNonMandatoryAccessibility( - * accessibility ); - */ - - double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[segmentIndex]; - mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, - mgraDistanceArray); - - // set size array for the tour segment and distance array from - // the home mgra to all destinaion mgras. - dcSoaDmuObject.setDestChoiceSize(homeMgraSizeArray); - dcSoaDmuObject.setDestDistance(mgraDistanceArray); - - dcDmuObject.setDestChoiceSize(homeMgraSizeArray); - dcDmuObject.setDestChoiceDistance(mgraDistanceArray); - - modelIndex = dcModelIndices[segmentIndex]; - locationChoiceModel = locationChoiceModels[modelIndex]; - - // get the school location alternative chosen from the sample - results = selectLocationFromSampleOfAlternatives("school", segmentType, p, - segmentNameList[segmentIndex], segmentIndex, tourNum++, homeMgraSizeArray, - mgraDistanceArray); - - } catch (RuntimeException e) - { - logger.fatal(String - .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in school location choice, modelIndex=%d, segmentIndex=%d, segmentName=%s", - i, hh.getHhId(), i, modelIndex, segmentIndex, - segmentNameList[segmentIndex])); - logger.fatal("Exception caught:", e); - logger.fatal("calling System.exit(-1) to terminate."); - System.exit(-1); - } - - p.setSchoolLoc((int) results[0]); - p.setSchoolLocDistance((float) results[1]); - p.setSchoolLocLogsum((float) results[2]); - - } - - } - - /** - * - * @return an array with chosen mgra, distance to chosen mgra, and logsum to - * chosen mgra. - */ - private double[] selectLocationFromSampleOfAlternatives(String segmentType, - int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, - int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcManLogger; - - Household household = person.getHouseholdObject(); - - // compute the sample of alternatives set for the person - dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, null, person, - segmentName, sizeSegmentIndex, household.getHhMgra()); - - // get sample of locations and correction factors for sample - int[] finalSample = dcSoaModel.getSampleOfAlternatives(); - float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); - - int numAlts = locationChoiceModel.getNumberOfAlternatives(); - - // set the destAltsAvailable array to true for all destination choice - // alternatives for each purpose - boolean[] destAltsAvailable = new boolean[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsAvailable[k] = false; - - // set the destAltsSample array to 1 for all destination choice - // alternatives - // for each purpose - int[] destAltsSample = new int[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsSample[k] = 0; - - int[] sampleValues = new int[finalSample.length]; - - dcDmuObject.setDestChoiceSize(homeMgraSizeArray); - dcDmuObject.setDestChoiceDistance(homeMgraDistanceArray); - - // for the destinations and sub-zones in the sample, compute mc logsums - // and - // save in DC dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 1; i < finalSample.length; i++) - { - - int destMgra = finalSample[i]; - sampleValues[i] = finalSample[i]; - - // get the mode choice logsum for the destination choice sample - // alternative - double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); - - sampleAlternativeLogsums[i] = logsum; - sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - dcDmuObject.setMcLogsum(destMgra, logsum); - - // set sample of alternatives correction factor used in destination - // choice utility for the sampled alternative. - dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); - - // set availaibility and sample values for the purpose, dcAlt. - destAltsAvailable[finalSample[i]] = true; - destAltsSample[finalSample[i]] = 1; - - } - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tourNum); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" - + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " - + person.getPersonType() + ", TourNum=" + tourNum); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - locationChoiceModel.computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), - destAltsAvailable, destAltsSample); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (locationChoiceModel.getAvailabilityCount() > 0) - { - chosen = locationChoiceModel.getChoiceResult(rn); - } else - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), segmentName)); - } - - // write choice model alternative info to log file - int selectedIndex = -1; - for (int j = 1; j < finalSample.length; j++) - { - if (finalSample[j] == chosen) - { - selectedIndex = j; - break; - } - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = locationChoiceModel.getUtilities(); - double[] probabilities = locationChoiceModel.getProbabilities(); - boolean[] availabilities = locationChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb Distance Logsum"); - modelLogger - .info("--------------------- -------------- -------------- -------------- -------------- -------------- --------------"); - - int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); - - int sortedSelectedIndex = 0; - double cumProb = 0.0; - for (int j = 1; j < finalSample.length; j++) - { - int k = sortedSampleValueIndices[j]; - int alt = finalSample[k]; - - if (alt == chosen) sortedSelectedIndex = j; - - cumProb += probabilities[alt - 1]; - String altString = String.format("j=%-2d, k=%-2d, mgra=%-5d", j, k, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e%18.6e%18.6e", - altString, availabilities[alt], utilities[alt - 1], probabilities[alt - 1], - cumProb, sampleAlternativeDistances[k], sampleAlternativeLogsums[k])); - } - - if (sortedSelectedIndex >= 0) - { - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", sortedSelectedIndex, chosen); - modelLogger.info(String.format( - "Choice: %s, dist=%.6e, logsum=%.6e with rn=%.8f, randomCount=%d", - altString, sampleAlternativeDistances[selectedIndex], - sampleAlternativeLogsums[selectedIndex], rn, randomCount)); - } else - { - modelLogger.info(" "); - modelLogger.info(String.format( - "j=%d, mgra=None selected, no alternatives available", selectedIndex)); - modelLogger.info(String.format("Choice: %s, rn=%.8f, randomCount=%d", "N/A", rn, - randomCount)); - } - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - - // write UEC calculation results to separate model specific log file - locationChoiceModel.logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.fatal(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), segmentName)); - logger.fatal("calling System.exit(-1) to terminate."); - System.exit(-1); - } - - } - - double[] returnArray = new double[3]; - - returnArray[0] = chosen; - returnArray[1] = sampleAlternativeDistances[selectedIndex]; - returnArray[2] = sampleAlternativeLogsums[selectedIndex]; - - return returnArray; - - } - - private int selectWorksAtHomeChoice(DestChoiceDMU dcDmuObject, Household household, - Person person) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcManLogger; - - dcDmuObject.setHouseholdObject(household); - dcDmuObject.setPersonObject(person); - dcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), origMgra, 0); - - double accessibility = aggAcc.getAccessibilitiesTableObject().getAggregateAccessibility( - "totEmp", household.getHhMgra()); - dcDmuObject.setWorkAccessibility(accessibility); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format("Usual Work Location Is At Home Choice Model"); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person - .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Usual Work Location Is At Home Choice Model: Person Num: " - + person.getPersonNum() + ", Person Type: " + person.getPersonType()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - worksAtHomeModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float logsum = (float) worksAtHomeModel.computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues()); - person.setWorksFromHomeLogsum(logsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (worksAtHomeModel.getAvailabilityCount() > 0) - { - chosen = worksAtHomeModel.getChoiceResult(rn); - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels() || chosen < 0) - { - - double[] utilities = worksAtHomeModel.getUtilities(); - double[] probabilities = worksAtHomeModel.getProbabilities(); - boolean[] availabilities = worksAtHomeModel.getAvailabilities(); - - String[] altNames = worksAtHomeModel.getAlternativeNames(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < utilities.length; j++) - { - cumProb += probabilities[j]; - String altString = String.format("%d, %s", j + 1, altNames[j]); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, alt=%s", chosen, - (chosen < 0 ? "N/A, no available alternatives" : altNames[chosen - 1])); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - worksAtHomeModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - worksAtHomeModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - - // write UEC calculation results to separate model specific log file - worksAtHomeModel.logUECResults(modelLogger, loggingHeader); - - } - - if (chosen < 0) - { - logger.fatal(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available works at home alternatives to choose from in choiceModelApplication.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum())); - logger.fatal("calling System.exit(-1) to terminate."); - System.exit(-1); - } - - return chosen; - - } - - private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, - int segmentTypeIndex) - { - - int purposeIndex = 0; - String purpose = ""; - if (segmentTypeIndex < 0) - { - purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } - - // create a temporary tour to use to calculate mode choice logsum - Tour mcLogsumTour = new Tour(person, 0, purposeIndex); - mcLogsumTour.setTourPurpose(purpose); - mcLogsumTour.setTourOrigMgra(household.getHhMgra()); - mcLogsumTour.setTourDestMgra(sampleDestMgra); - mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); - mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - - if (household.getDebugChoiceModels()) - { - dcManLogger.info(""); - dcManLogger.info(""); - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, - dcManLogger, person); - } - - double logsum = -1; - try - { - logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, - choiceModelDescription, decisionMakerLabel); - } catch (Exception e) - { - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - logger.fatal("Exception caught:", e); - System.exit(-1); - } - - return logsum; - } - - public int getModelIndex() - { - return modelIndex; - } - - public void setDcSizeObject(DestChoiceSize dcSizeObj) - { - this.dcSizeObj = dcSizeObj; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java deleted file mode 100644 index aa90422..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServer.java +++ /dev/null @@ -1,209 +0,0 @@ -package org.sandag.abm.ctramp; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagTourBasedModel; - -import com.pb.common.calculator.DataEntry; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; - -/** - * @author Jim Hicks - * - * Class for managing matrix data in a remote process and accessed by - * UECs using RMI. - */ -public class MatrixDataServer - implements MatrixDataServerIf, Serializable -{ - - private static Logger logger = Logger.getLogger(MatrixDataServer.class); - - private Object objectLock; - - private static final String VERSION = "2.3_OMX_Only"; - - // These are used if the server is started manually by running this class's - // main(). If so, these must be defined consistently with - // any class that acts as a client to the server, i.e. the client must know - // the - // address and port as well. - private static final String MATRIX_DATA_SERVER_ADDRESS = "127.0.0.1"; - private static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final String MATRIX_DATA_SERVER_NAME = MatrixDataServer.class - .getCanonicalName(); - private static final String MATRIX_DATA_SERVER_LABEL = "matrix server"; - - private HashMap matrixEntryMap; - private HashMap matrixMap; - - public MatrixDataServer() - { - - // create the HashMap objects to keep track of matrix data read by the server - matrixEntryMap = new HashMap(); - matrixMap = new HashMap(); - - objectLock = new Object(); - } - - public String testRemote(String remoteObjectName) - { - logger.info("testRemote() called by remote process: " + remoteObjectName + "."); - return String.format("testRemote() method in %s called by %s.", this.getClass() - .getCanonicalName(), remoteObjectName); - } - - public String testRemote() - { - logger.info("testRemote() called by remote process."); - return String.format("testRemote() method in %s called.", this.getClass() - .getCanonicalName()); - } - - /* - * Read a matrix. - * - * @param matrixEntry a DataEntry describing the matrix to read - * - * @return a Matrix - */ - public Matrix getMatrix(DataEntry matrixEntry) - { - - Matrix matrix; - - synchronized (objectLock) - { - - String name = matrixEntry.name; - - if (matrixEntryMap.containsKey(name)) - { - matrix = matrixMap.get(name); - } else - { - - //create 64bit matrix reader - String fileName = matrixEntry.fileName; - MatrixReader mr = MatrixReader.createReader(MatrixType.OMX, new File(fileName)); - matrix = mr.readMatrix(matrixEntry.matrixName); - logger.info("Read " + matrixEntry.matrixName + " as " + name + " from " + fileName); - - // Use token name from control file for matrix name (not name - // from underlying matrix) - matrix.setName(matrixEntry.name); - - matrixMap.put(name, matrix); - matrixEntryMap.put(name, matrixEntry); - } - - } - - return matrix; - } - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m) - { - - File outFile = new File(fileName); - MatrixWriter writer = MatrixWriter.createWriter(MatrixType.OMX, outFile); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - } - - writer.writeMatrices(names, m); - } - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m, MatrixType mt) - { - writeMatrixFile(fileName, m); - } - - public void clear() - { - if (matrixMap != null) - { - matrixMap = new HashMap(); - logger.info("MatrixDataServer matrixMap object is cleared."); - } else - { - logger.info("MatrixDataServer.clear() called, but matrixMap object is null."); - } - - if (matrixEntryMap != null) - { - matrixEntryMap = new HashMap(); - logger.info("MatrixDataServer matrixEntryMap object is cleared."); - } else - { - logger.info("MatrixDataServer.clear() called, but matrixEntryMap object is null."); - } - } - - //Empty methods to maintain compatibility - public void start32BitMatrixIoServer(MatrixType mType) {} - public void stop32BitMatrixIoServer() {} - public void setRam(int ram) {} - - public static void main(String[] args) throws Exception - { - - String serverAddress = MATRIX_DATA_SERVER_ADDRESS; - int serverPort = MATRIX_DATA_SERVER_PORT; - String className = MATRIX_DATA_SERVER_NAME; - String serverLabel = MATRIX_DATA_SERVER_LABEL; - int ram = -1; - - for (int i = 0; i < args.length; i++) - { - if (args[i].equalsIgnoreCase("-hostname")) serverAddress = args[i + 1]; - else if (args[i].equalsIgnoreCase("-port")) serverPort = Integer.parseInt(args[i + 1]); - else if (args[i].equalsIgnoreCase("-label")) serverLabel = args[i + 1]; - else if (args[i].equalsIgnoreCase("-ram")) ram = Integer.parseInt(args[i + 1]); - } - - MatrixDataServer matrixServer = new MatrixDataServer(); - - // bind this concrete object with the cajo library objects for managing RMI - Remote.config(serverAddress, serverPort, null, 0); - ItemServer.bind(matrixServer, className); - - // log that the server started - logger.info("server starting on " + (System.getProperty("os.arch")) - + " operating system."); - logger.info(String.format("%s version %s started on: %s:%d", serverLabel, VERSION, - serverAddress, serverPort)); - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java deleted file mode 100644 index 767b869..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MatrixDataServerRmi.java +++ /dev/null @@ -1,98 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; - -import com.pb.common.calculator.DataEntry; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; - -/** - * @author Jim Hicks - * - * Class for managing matrix data in a remote process and accessed by - * UECs using RMI. - */ -public class MatrixDataServerRmi - implements MatrixDataServerIf, Serializable -{ - - // protected static Logger logger = - // Logger.getLogger(MatrixDataServerRmi.class); - - UtilRmi remote; - String connectString; - - public MatrixDataServerRmi(String hostname, int port, String className) - { - - connectString = String.format("//%s:%d/%s", hostname, port, className); - remote = new UtilRmi(connectString); - - } - - public void clear() - { - Object[] objArray = {}; - remote.method("clear", objArray); - } - - public void writeMatrixFile(String fileName, Matrix[] m, MatrixType mt) - { - Object[] objArray = {fileName, m, mt}; - remote.method("writeMatrixFile", objArray); - } - - public Matrix getMatrix(DataEntry dataEntry) - { - Object[] objArray = {dataEntry}; - return (Matrix) remote.method("getMatrix", objArray); - } - - public void start32BitMatrixIoServer(MatrixType mType) - { - Object[] objArray = {mType}; - remote.method("start32BitMatrixIoServer", objArray); - } - - public void start32BitMatrixIoServer(MatrixType mType, String label) - { - Object[] objArray = {mType, label}; - remote.method("start32BitMatrixIoServer", objArray); - } - - public void stop32BitMatrixIoServer() - { - Object[] objArray = {}; - remote.method("stop32BitMatrixIoServer", objArray); - } - - public void stop32BitMatrixIoServer(String label) - { - Object[] objArray = {label}; - remote.method("stop32BitMatrixIoServer", objArray); - } - - public String testRemote(String remoteObjectName) - { - Object[] objArray = {remoteObjectName}; - return (String) remote.method("testRemote", objArray); - } - - public String testRemote() - { - Object[] objArray = {}; - return (String) remote.method("testRemote", objArray); - } - - /** - * This method is included in the Interface this class implements, but is not used anywhere by the SANDAG model. - * It is included hear to satisfy the interface only. - */ - @Override - public void writeMatrixFile(String fileName, Matrix[] m) { - // TODO Auto-generated method stub - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java deleted file mode 100644 index 2699a7b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/McLogsumsCalculator.java +++ /dev/null @@ -1,754 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; - -import com.pb.common.newmodel.Alternative; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.ConcreteAlternative; -import com.pb.common.newmodel.LogitModel; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.calculator.IndexValues; - - -public class McLogsumsCalculator implements Serializable -{ - - private transient Logger autoSkimLogger = Logger.getLogger(McLogsumsCalculator.class); - - public static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; - public static final String PROPERTIES_UEC_TRIP_MODE_CHOICE = "tripModeChoice.uec.file"; - - - public static final int WTW = 0; - public static final int WTD = 1; - public static final int DTW = 2; - public static final int NUM_ACC_EGR = 3; - - public static final int OUT = 0; - public static final int IN = 1; - public static final int NUM_DIR = 2; - - private BestTransitPathCalculator bestPathUEC; - private double[] tripModeChoiceSegmentStoredProbabilities; - private double[] tripModeChoiceSegmentStoredVOTs; - private float tripModeChoiceSegmentStoredParkingCost; - - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - private int[] parkingArea; - - private double[][] bestWtwTapPairsOut; - private double[][] bestWtwTapPairsIn; - private double[][] bestWtdTapPairsOut; - private double[][] bestWtdTapPairsIn; - private double[][] bestDtwTapPairsOut; - private double[][] bestDtwTapPairsIn; - - private double[][] bestWtwTripTapPairs; - private double[][] bestWtdTripTapPairs; - private double[][] bestDtwTripTapPairs; - - private AutoAndNonMotorizedSkimsCalculator anm; - - private int setTourMcLogsumDmuAttributesTotalTime = 0; - private int setTripMcLogsumDmuAttributesTotalTime = 0; - - - - public McLogsumsCalculator() - { - if (mgraManager == null) - mgraManager = MgraDataManager.getInstance(); - - if (tazManager == null) - tazManager = TazDataManager.getInstance(); - - this.lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); - this.lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); - this.lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); - this.parkingArea = mgraManager.getMgraParkAreas(); - - tripModeChoiceSegmentStoredVOTs = new double[3]; - } - - - public BestTransitPathCalculator getBestTransitPathCalculator() - { - return bestPathUEC; - } - - - public void setupSkimCalculators(HashMap rbMap) - { - bestPathUEC = new BestTransitPathCalculator(rbMap); - anm = new AutoAndNonMotorizedSkimsCalculator(rbMap); - } - - public void setTazDistanceSkimArrays( double[][][] storedFromTazDistanceSkims, double[][][] storedToTazDistanceSkims ) { - anm.setTazDistanceSkimArrays( storedFromTazDistanceSkims, storedToTazDistanceSkims ); - } - - - public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() - { - return anm; - } - - public void setTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean debug ) - { - - setNmTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); - setWtwTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); - setDtwTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); - setWtdTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, debug ); - - // set the land use data items in the DMU for the origin - mcDmuObject.setOrigDuDen( mgraManager.getDuDenValue( origMgra ) ); - mcDmuObject.setOrigEmpDen( mgraManager.getEmpDenValue( origMgra ) ); - mcDmuObject.setOrigTotInt( mgraManager.getTotIntValue( origMgra ) ); - - // set the land use data items in the DMU for the destination - mcDmuObject.setDestDuDen( mgraManager.getDuDenValue( destMgra ) ); - mcDmuObject.setDestEmpDen( mgraManager.getEmpDenValue( destMgra ) ); - mcDmuObject.setDestTotInt( mgraManager.getTotIntValue( destMgra ) ); - - mcDmuObject.setLsWgtAvgCostM( lsWgtAvgCostM[destMgra] ); - mcDmuObject.setLsWgtAvgCostD( lsWgtAvgCostD[destMgra] ); - mcDmuObject.setLsWgtAvgCostH( lsWgtAvgCostH[destMgra] ); - - int tourOrigTaz = mgraManager.getTaz(origMgra); - int tourDestTaz = mgraManager.getTaz(destMgra); - - mcDmuObject.setPTazTerminalTime( tazManager.getOriginTazTerminalTime(tourOrigTaz) ); - mcDmuObject.setATazTerminalTime( tazManager.getDestinationTazTerminalTime(tourDestTaz) ); - - Person person = mcDmuObject.getPersonObject(); - - double reimbursePct=0; - if(person!=null) { - reimbursePct = person.getParkingReimbursement(); - } - - mcDmuObject.setReimburseProportion( reimbursePct ); - mcDmuObject.setParkingArea(parkingArea[destMgra]); - - - } - - - public double calculateTourMcLogsum(int origMgra, int destMgra, int departPeriod, int arrivePeriod, - ChoiceModelApplication mcModel, TourModeChoiceDMU mcDmuObject) - { - - long currentTime = System.currentTimeMillis(); - setTourMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, arrivePeriod, mcDmuObject.getDmuIndexValues().getDebug() ); - setTourMcLogsumDmuAttributesTotalTime += ( System.currentTimeMillis() - currentTime ); - - // mode choice UEC references highway skim matrices directly, so set index orig/dest to O/D TAZs. - IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); - int tourOrigTaz = mgraManager.getTaz(origMgra); - int tourDestTaz = mgraManager.getTaz(destMgra); - mcDmuIndex.setOriginZone(tourOrigTaz); - mcDmuIndex.setDestZone(tourDestTaz); - mcDmuObject.setOriginMgra(origMgra); - mcDmuObject.setDestMgra(destMgra); - - mcModel.computeUtilities(mcDmuObject, mcDmuIndex); - double logsum = mcModel.getLogsum(); - - return logsum; - - } - - public void setWalkTransitLogSumUnavailable( TripModeChoiceDMU tripMcDmuObject ) { - tripMcDmuObject.setTransitLogSum( WTW, bestPathUEC.NA ); - } - - public void setDriveTransitLogSumUnavailable( TripModeChoiceDMU tripMcDmuObject, boolean isInbound ) { - - // set drive transit skim attributes to unavailable - if ( ! isInbound ) { - tripMcDmuObject.setTransitLogSum( DTW, bestPathUEC.NA); - } - else { - tripMcDmuObject.setTransitLogSum( WTD, bestPathUEC.NA); - } - - } - - - public double calculateTripMcLogsum(int origMgra, int destMgra, int departPeriod, ChoiceModelApplication mcModel, TripModeChoiceDMU mcDmuObject, Logger myLogger) - { - long currentTime = System.currentTimeMillis(); - setNmTripMcDmuAttributes( mcDmuObject, origMgra, destMgra, departPeriod, mcDmuObject.getHouseholdObject().getDebugChoiceModels() ); - - mcDmuObject.setTripPeriod(departPeriod); - - // set the land use data items in the DMU for the origin - mcDmuObject.setOrigDuDen( mgraManager.getDuDenValue( origMgra ) ); - mcDmuObject.setOrigEmpDen( mgraManager.getEmpDenValue( origMgra ) ); - mcDmuObject.setOrigTotInt( mgraManager.getTotIntValue( origMgra ) ); - - // set the land use data items in the DMU for the destination - mcDmuObject.setDestDuDen( mgraManager.getDuDenValue( destMgra ) ); - mcDmuObject.setDestEmpDen( mgraManager.getEmpDenValue( destMgra ) ); - mcDmuObject.setDestTotInt( mgraManager.getTotIntValue( destMgra ) ); - - // mode choice UEC references highway skim matrices directly, so set index orig/dest to O/D TAZs. - IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); - mcDmuIndex.setOriginZone(mgraManager.getTaz(origMgra)); - mcDmuIndex.setDestZone(mgraManager.getTaz(destMgra)); - mcDmuObject.setOriginMgra(origMgra); - mcDmuObject.setDestMgra(destMgra); - - setTripMcLogsumDmuAttributesTotalTime += ( System.currentTimeMillis() - currentTime ); - mcDmuObject.setPTazTerminalTime( tazManager.getOriginTazTerminalTime(mgraManager.getTaz(origMgra)) ); - mcDmuObject.setATazTerminalTime( tazManager.getDestinationTazTerminalTime(mgraManager.getTaz(destMgra)) ); - - - mcModel.computeUtilities(mcDmuObject, mcDmuIndex); - double logsum = mcModel.getLogsum(); - tripModeChoiceSegmentStoredProbabilities = Arrays.copyOf( mcModel.getCumulativeProbabilities(), mcModel.getNumberOfAlternatives() ); - - //also save the VOTs from the model - UtilityExpressionCalculator uec = mcModel.getUEC(); - - ModelStructure modelStructure = mcDmuObject.modelStructure; - - tripModeChoiceSegmentStoredVOTs[0] = uec.getValueForIndex(uec.lookupVariableIndex("vot")); - tripModeChoiceSegmentStoredVOTs[1] = uec.getValueForIndex(uec.lookupVariableIndex("votS2")); - tripModeChoiceSegmentStoredVOTs[2] = uec.getValueForIndex(uec.lookupVariableIndex("votS3")); - - tripModeChoiceSegmentStoredParkingCost = (float) uec.getValueForIndex(uec.lookupVariableIndex("parkingCost")); - - if ( mcDmuObject.getHouseholdObject().getDebugChoiceModels() ) - mcModel.logUECResults(myLogger, "Trip Mode Choice Utility Expressions for mgras: " + origMgra + " to " + destMgra + " for HHID: " + mcDmuIndex.getHHIndex() ); - - return logsum; - - } - - - /** - * return the array of mode choice model cumulative probabilities determined while - * computing the mode choice logsum for the trip segmen during stop location choice. - * These probabilities arrays are stored for each sampled stop location so that when - * the selected sample stop location is known, the mode choice can be drawn from the - * already computed probabilities. - * - * @return mode choice cumulative probabilities array - */ - public double[] getStoredSegmentCumulativeProbabilities() { - return tripModeChoiceSegmentStoredProbabilities; - } - - public double[] getStoredSegmentVOTs() { - return tripModeChoiceSegmentStoredVOTs; - } - public double[][] getBestWtwTapsOut() - { - return bestWtwTapPairsOut; - } - - public double[][] getBestWtwTapsIn() - { - return bestWtwTapPairsIn; - } - - public double[][] getBestWtdTapsOut() - { - return bestWtdTapPairsOut; - } - - public double[][] getBestWtdTapsIn() - { - return bestWtdTapPairsIn; - } - - public double[][] getBestDtwTapsOut() - { - return bestDtwTapPairsOut; - } - - public double[][] getBestDtwTapsIn() - { - return bestDtwTapPairsIn; - } - - public double[][] getBestWtwTripTaps() - { - return bestWtwTripTapPairs; - } - - public double[][] getBestDtwTripTaps() - { - return bestDtwTripTapPairs; - } - - public double[][] getBestWtdTripTaps() - { - return bestWtdTripTapPairs; - } - - - private void setNmTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) - { - // non-motorized, outbound then inbound - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - departPeriod = skimPeriodIndex; - double[] nmSkimsOut = anm.getNonMotorizedSkims(origMgra, destMgra, departPeriod, loggingEnabled, autoSkimLogger); - if (loggingEnabled) - anm.logReturnedSkims(origMgra, destMgra, departPeriod, nmSkimsOut, "non-motorized outbound", autoSkimLogger); - - skimPeriodIndex = ModelStructure.getSkimPeriodIndex(arrivePeriod); - arrivePeriod = skimPeriodIndex; - double[] nmSkimsIn = anm.getNonMotorizedSkims(destMgra, origMgra, arrivePeriod, loggingEnabled, autoSkimLogger); - if (loggingEnabled) anm.logReturnedSkims(destMgra, origMgra, arrivePeriod, nmSkimsIn, "non-motorized inbound", autoSkimLogger); - - int walkIndex = anm.getNmWalkTimeSkimIndex(); - mcDmuObject.setNmWalkTimeOut( nmSkimsOut[walkIndex] ); - mcDmuObject.setNmWalkTimeIn( nmSkimsIn[walkIndex] ); - - int bikeIndex = anm.getNmBikeTimeSkimIndex(); - mcDmuObject.setNmBikeTimeOut( nmSkimsOut[bikeIndex] ); - mcDmuObject.setNmBikeTimeIn( nmSkimsIn[bikeIndex] ); - - } - - private void setWtwTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) - { - - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // walk access, walk egress transit, outbound - int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - bestWtwTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); - - if (bestWtwTapPairsOut[0] == null) { - mcDmuObject.setTransitLogSum( WTW, false, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(mcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); - } - - bestWtwTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestWtwTapPairsOut, walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); - double logsumOut = bestPathUEC.calcTripLogSum(bestWtwTapPairsOut, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( WTW, false, logsumOut); - } - - //setup best path dmu variables - walkDmu = new TransitWalkAccessDMU(); - driveDmu = new TransitDriveAccessDMU(); - - // walk access, walk egress transit, inbound - int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); - bestWtwTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); - - if (bestWtwTapPairsIn[0] == null) { - mcDmuObject.setTransitLogSum( WTW, true, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(mcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); - } - - bestWtwTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestWtwTapPairsIn, walkDmu, driveDmu, WTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); - double logsumIn = bestPathUEC.calcTripLogSum(bestWtwTapPairsIn, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( WTW, true, logsumIn); - } - } - - private void setWtdTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) - { - - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // logsum for WTD outbound is never used -> set to NA - mcDmuObject.setTransitLogSum( WTD, false, bestPathUEC.NA ); - /* TODO: - remove this section of code after successful testing - // walk access, drive egress transit, outbound - int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); - bestWtdTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger); - - if (bestWtdTapPairsOut[0] == null) { - mcDmuObject.setTransitLogSum( WTD, false, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.personType : mcDmuObject.getPersonType()); - walkDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.personType : mcDmuObject.getPersonType()); - driveDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); - - bestWtdTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestWtdTapPairsOut, walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger); - double logsumOut = bestPathUEC.calcTripLogSum(bestWtdTapPairsOut, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( WTD, false, logsumOut); - } - */ - - // walk access, drive egress transit, inbound - int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - - bestWtdTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); - - if (bestWtdTapPairsIn[0] == null) { - mcDmuObject.setTransitLogSum( WTD, true, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(mcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); - } - bestWtdTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestWtdTapPairsIn, walkDmu, driveDmu, WTD, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger, odDistance); - double logsumIn = bestPathUEC.calcTripLogSum(bestWtdTapPairsIn, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( WTD, true, logsumIn); - } - } - - private void setDtwTourMcDmuAttributes( TourModeChoiceDMU mcDmuObject, int origMgra, int destMgra, int departPeriod, int arrivePeriod, boolean loggingEnabled ) - { - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // drive access, walk egress transit, outbound - int skimPeriodIndexOut = ModelStructure.getSkimPeriodIndex(departPeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - - bestDtwTapPairsOut = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); - - if (bestDtwTapPairsOut[0] == null) { - mcDmuObject.setTransitLogSum( DTW, false, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setIvtCoeff( (float) mcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) mcDmuObject.getCostCoeff()); - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(mcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : mcDmuObject.getPersonType()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : mcDmuObject.getPersonType()); - } - - bestDtwTapPairsOut = bestPathUEC.calcPersonSpecificUtilities(bestDtwTapPairsOut, walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndexOut, loggingEnabled, autoSkimLogger, odDistance); - double logsumOut = bestPathUEC.calcTripLogSum(bestDtwTapPairsOut, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( DTW, false, logsumOut); - } - - // logsum for DTW inbound is never used -> set to NA - mcDmuObject.setTransitLogSum( DTW, true, bestPathUEC.NA ); - - /* TODO: remove this section of code after successful testing - //setup best path dmu variables - walkDmu = new TransitWalkAccessDMU(); - driveDmu = new TransitDriveAccessDMU(); - - // drive access, walk egress transit, inbound - int skimPeriodIndexIn = ModelStructure.getSkimPeriodIndex(arrivePeriod); - bestDtwTapPairsIn = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger); - - if (bestDtwTapPairsIn[0] == null) { - mcDmuObject.setTransitLogSum( DTW, true, bestPathUEC.NA ); - } else { - // calculate logsum - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - walkDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); - walkDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? walkDmu.personType : mcDmuObject.getPersonType()); - walkDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TOURMC); - driveDmu.setTourCategoryIsJoint(mcDmuObject.getTourCategoryJoint()); - driveDmu.setPersonType(mcDmuObject.getTourCategoryJoint()==1 ? driveDmu.personType : mcDmuObject.getPersonType()); - driveDmu.setValueOfTime((float)mcDmuObject.getValueOfTime()); - - bestDtwTapPairsIn = bestPathUEC.calcPersonSpecificUtilities(bestDtwTapPairsIn, walkDmu, driveDmu, DTW, destMgra, origMgra, skimPeriodIndexIn, loggingEnabled, autoSkimLogger); - double logsumIn = bestPathUEC.calcTripLogSum(bestDtwTapPairsIn, loggingEnabled, autoSkimLogger); - mcDmuObject.setTransitLogSum( DTW, true, logsumIn); - } - */ - } - - public void setNmTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) - { - - double[] nmSkims = null; - - // non-motorized, outbound then inbound - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - departPeriod = skimPeriodIndex; - nmSkims = anm.getNonMotorizedSkims(origMgra, destMgra, departPeriod, loggingEnabled, autoSkimLogger); - if (loggingEnabled) - anm.logReturnedSkims(origMgra, destMgra, departPeriod, nmSkims, "non-motorized trip mode choice skims", autoSkimLogger); - - int walkIndex = anm.getNmWalkTimeSkimIndex(); - tripMcDmuObject.setNonMotorizedWalkTime(nmSkims[walkIndex] ); - - int bikeIndex = anm.getNmBikeTimeSkimIndex(); - tripMcDmuObject.setNonMotorizedBikeTime(nmSkims[bikeIndex] ); - - } - - public void setWtwTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled) - { - - if (bestTapPairs == null) { - if(loggingEnabled){ - autoSkimLogger.info("Attempting to set WTW Trip MC DMU Attributes for null best TAP pairs array"); - } - tripMcDmuObject.setTransitLogSum( WTW, bestPathUEC.NA ); - bestWtwTripTapPairs = bestTapPairs; - return; - } - - // calculate logsum - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); - tripMcDmuObject.setTransitLogSum( WTW, logsum); - bestWtwTripTapPairs = bestTapPairs; - - } - - public void setDtwTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled ) - { - - if (bestTapPairs == null) { - if(loggingEnabled){ - autoSkimLogger.info("Attempting to set DTW Trip MC DMU Attributes for null best TAP pairs array"); - } - tripMcDmuObject.setTransitLogSum( DTW, bestPathUEC.NA ); - bestDtwTripTapPairs = bestTapPairs; - return; - } - - // calculate logsum - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); - - if(loggingEnabled) - autoSkimLogger.info("Setting DTW logsum in trip MC DMU object to "+logsum); - - tripMcDmuObject.setTransitLogSum( DTW, logsum); - bestDtwTripTapPairs = bestTapPairs; - - } - - public void setWtdTripMcDmuAttributesForBestTapPairs( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, double[][] bestTapPairs, boolean loggingEnabled ) - { - - if (bestTapPairs == null) { - if(loggingEnabled){ - autoSkimLogger.info("Attempting to set WTD Trip MC DMU Attributes for null best TAP pairs array"); - } - tripMcDmuObject.setTransitLogSum( WTD, bestPathUEC.NA ); - bestWtdTripTapPairs = bestTapPairs; - return; - } - - // calculate logsum - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - double logsum = bestPathUEC.calcTripLogSum(bestTapPairs, loggingEnabled, autoSkimLogger); - tripMcDmuObject.setTransitLogSum( WTD, logsum); - bestWtdTripTapPairs = bestTapPairs; - - } - - public void setWtwTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) - { - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // walk access and walk egress for transit segment - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - - // store best tap pairs for walk-transit-walk - bestWtwTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance ); - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(tripMcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); - driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); - } - // calculate logsum - bestWtwTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestWtwTripTapPairs, walkDmu, driveDmu, WTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); - double logsum = bestPathUEC.calcTripLogSum(bestWtwTripTapPairs, loggingEnabled, autoSkimLogger); - tripMcDmuObject.setTransitLogSum( WTW, logsum); - - } - - public void setWtdTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) - { - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // walk access, drive egress transit, outbound - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - - // store best tap pairs using outbound direction array - bestWtdTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(tripMcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); - driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); - } - - // calculate logsum - bestWtdTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestWtdTripTapPairs, walkDmu, driveDmu, WTD, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); - double logsum = bestPathUEC.calcTripLogSum(bestWtdTripTapPairs, loggingEnabled, autoSkimLogger); - tripMcDmuObject.setTransitLogSum( WTD, logsum); - - } - - public void setDtwTripMcDmuAttributes( TripModeChoiceDMU tripMcDmuObject, int origMgra, int destMgra, int departPeriod, boolean loggingEnabled ) - { - //setup best path dmu variables - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - - // drive access, walk egress transit, outbound - int skimPeriodIndex = ModelStructure.getSkimPeriodIndex(departPeriod); - int pTaz = mgraManager.getTaz(origMgra); - int aTaz = mgraManager.getTaz(destMgra); - float odDistance = (float) anm.getTazDistanceFromTaz(pTaz, ModelStructure.AM_SKIM_PERIOD_INDEX)[aTaz]; - - // store best tap pairs using outbound direction array - bestDtwTripTapPairs = bestPathUEC.getBestTapPairs(walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); - - //set person specific variables and re-calculate best tap pair utilities - walkDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - walkDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - walkDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - driveDmu.setApplicationType(bestPathUEC.APP_TYPE_TRIPMC); - driveDmu.setIvtCoeff( (float) tripMcDmuObject.getIvtCoeff()); - driveDmu.setCostCoeff( (float) tripMcDmuObject.getCostCoeff()); - - //catch issues where the trip mode choice DMU was set up without a household or person object - if(tripMcDmuObject.getHouseholdObject()!=null){ - walkDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? walkDmu.getPersonType() : tripMcDmuObject.getPersonType()); - driveDmu.setPersonType(tripMcDmuObject.getTourCategoryJoint()==1 ? driveDmu.getPersonType() : tripMcDmuObject.getPersonType()); - } - // calculate logsum - bestDtwTripTapPairs = bestPathUEC.calcPersonSpecificUtilities(bestDtwTripTapPairs, walkDmu, driveDmu, DTW, origMgra, destMgra, skimPeriodIndex, loggingEnabled, autoSkimLogger, odDistance); - double logsum = bestPathUEC.calcTripLogSum(bestDtwTripTapPairs, loggingEnabled, autoSkimLogger); - tripMcDmuObject.setTransitLogSum( DTW, logsum); - - } - - //select best transit path from N-path for trip - public int chooseTripPath(double rnum, double[][] bestTapPairs, boolean myTrace, Logger myLogger) { - return bestPathUEC.chooseTripPath(rnum, bestTapPairs, myTrace, myLogger); - } - - - public float getTripModeChoiceSegmentStoredParkingCost() { - return tripModeChoiceSegmentStoredParkingCost; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java deleted file mode 100644 index 356b914..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceDMU.java +++ /dev/null @@ -1,119 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - */ -public class MicromobilityChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(MicromobilityChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - protected double ivtCoeff; - protected double costCoeff; - protected float walkTime; - protected boolean isTransit; - protected boolean microTransitAvailable; - - - public MicromobilityChoiceDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - } - - - public double getIvtCoeff() { - return ivtCoeff; - } - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public double getCostCoeff() { - return costCoeff; - } - - public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; - } - - public float getWalkTime() { - return walkTime; - } - - public void setWalkTime(float walkTime) { - this.walkTime = walkTime; - } - - public boolean isTransit() { - return isTransit; - } - - public void setTransit(boolean isTransit) { - this.isTransit = isTransit; - } - - public boolean isMicroTransitAvailable() { - return microTransitAvailable; - } - - public void setMicroTransitAvailable(boolean microTransitAvailable) { - this.microTransitAvailable = microTransitAvailable; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java deleted file mode 100644 index 724da7a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MicromobilityChoiceModel.java +++ /dev/null @@ -1,449 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class MicromobilityChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("micromobility"); - - private static final String MM_CONTROL_FILE_TARGET = "micromobility.uec.file"; - private static final String MM_DATA_SHEET_TARGET = "micromobility.data.page"; - private static final String MM_MODEL_SHEET_TARGET = "micromobility.model.page"; - private static final String MT_TAP_FILE_TARGET = "active.microtransit.tap.file"; - private static final String MT_MAZ_FILE_TARGET = "active.microtransit.mgra.file"; - - public static final int MM_MODEL_WALK_ALT = 0; - public static final int MM_MODEL_MICROMOBILITY_ALT = 1; - public static final int MM_MODEL_MICROTRANSIT_ALT = 2; - - - private ChoiceModelApplication mmModel; - private MicromobilityChoiceDMU mmDmuObject; - - // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose - double[] ivtCoeffs; - double[] incomeCoeffs; - double[] incomeExponents; - - private static final String PROPERTIES_TRIP_UTILITY_IVT_COEFFS = "trip.utility.ivt.coeffs"; - private static final String PROPERTIES_TRIP_UTILITY_INCOME_COEFFS = "trip.utility.income.coeffs"; - private static final String PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS = "trip.utility.income.exponents"; - private ModelStructure modelStructure; - private MgraDataManager mgraDataManager; - - private HashSet microtransitTaps; - private HashSet microtransitMazs; - - - public MicromobilityChoiceModel(HashMap propertyMap, - ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory) - { - - setupMicromobilityChoiceModelApplication(propertyMap, myModelStructure, dmuFactory); - } - - private void setupMicromobilityChoiceModelApplication(HashMap propertyMap, - ModelStructure myModelStructure, CtrampDmuFactoryIf dmuFactory) - { - logger.info("setting up micromobility choice model."); - - modelStructure = myModelStructure; - - // locate the micromobility choice UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mmUecFile = uecFileDirectory + propertyMap.get(MM_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_MODEL_SHEET_TARGET); - - // create the micromobility choice model DMU object. - mmDmuObject = dmuFactory.getMicromobilityChoiceDMU(); - - // create the transponder choice model object - mmModel = new ChoiceModelApplication(mmUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) mmDmuObject); - - - //get the coefficients for ivt and the coefficients to calculate the cost coefficient - ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_IVT_COEFFS); - incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_COEFFS); - incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TRIP_UTILITY_INCOME_EXPONENTS); - - mgraDataManager = MgraDataManager.getInstance(); - - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String microTransitTapFile = projectDirectory + propertyMap.get(MT_TAP_FILE_TARGET); - String microTransitMazFile = projectDirectory + propertyMap.get(MT_MAZ_FILE_TARGET); - - TableDataSet microTransitTapData = Util.readTableDataSet(microTransitTapFile); - TableDataSet microTransitMazData = Util.readTableDataSet(microTransitMazFile); - - microtransitTaps = new HashSet(); - microtransitMazs = new HashSet(); - - for(int i=1;i<=microTransitTapData.getRowCount();++i) { - - int tap = (int) microTransitTapData.getValueAt(i,"TAP"); - microtransitTaps.add(tap); - } - - for(int i=1;i<=microTransitMazData.getRowCount();++i) { - - int maz = (int) microTransitMazData.getValueAt(i,"MGRA"); - microtransitMazs.add(maz); - } - - - } - - - /** - * Apply model to all trips for the household. - * - * @param household - */ - public void applyModel(Household household) { - - for(Person person : household.getPersons()) { - - if(person==null) - continue; - - //work tours - if(person.getListOfWorkTours()!=null) { - - for(Tour tour:person.getListOfWorkTours()) - applyModel(household, person, tour); - } - - //school tours - if(person.getListOfSchoolTours()!=null) { - - for(Tour tour:person.getListOfSchoolTours()) - applyModel(household, person, tour); - } - - //non-mandatory tours - if(person.getListOfIndividualNonMandatoryTours()!=null) { - - for(Tour tour:person.getListOfIndividualNonMandatoryTours()) - applyModel(household, person, tour); - } - - //at-work sub tours - if(person.getListOfAtWorkSubtours()!=null) { - - for(Tour tour:person.getListOfAtWorkSubtours()) - applyModel(household, person, tour); - } - - } - - - } - - public void applyModel(Household household, Person person, Tour tour) { - - //apply to outbound stops - if(tour.getOutboundStops()!=null) { - - for(Stop s: tour.getOutboundStops()) - applyModel(household, person, tour, s); - } - - //apply to inbound stops - if(tour.getInboundStops()!=null) { - - for(Stop s: tour.getInboundStops()) - applyModel(household, person, tour, s); - } - - - } - - public void applyModel(Household household, Person person, Tour tour, Stop s) - { - - if(tour==null) - return; - - - if(!modelStructure.getTourModeIsWalk(s.getMode()) && !modelStructure.getTourModeIsWalkTransit(s.getMode())&& !modelStructure.getTourModeIsDriveTransit(s.getMode())) - return; - - int homeMaz = household.getHhMgra(); - double income = (double) household.getIncomeInDollars(); - - int category = IntermediateStopChoiceModels.PURPOSE_CATEGORIES[tour.getTourPrimaryPurposeIndex()]; - double ivtCoeff = ivtCoeffs[category]; - double incomeCoeff = incomeCoeffs[category]; - double incomeExpon = incomeExponents[category]; - double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); - double timeFactor = 1.0f; - if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - timeFactor = tour.getJointTourTimeFactor(); - else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) - timeFactor = person.getTimeFactorWork(); - else - timeFactor = person.getTimeFactorNonWork(); - - mmDmuObject.setIvtCoeff(ivtCoeff * timeFactor); - mmDmuObject.setCostCoeff(costCoeff); - int originMaz = s.getOrig(); - int destMaz = s.getDest(); - - if(modelStructure.getTourModeIsWalk(s.getMode())) - mmDmuObject.setTransit(false); - else - mmDmuObject.setTransit(true); - - if(modelStructure.getTourModeIsWalk(s.getMode())) { - - float walkTime = mgraDataManager.getMgraToMgraWalkTime(originMaz, destMaz); - mmDmuObject.setWalkTime(walkTime); - - if(microtransitMazs.contains(originMaz) && microtransitMazs.contains(destMaz)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - s.setMicromobilityWalkLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice(household, person, tour, s); - s.setMicromobilityWalkMode(chosenAlt); - - // write choice model alternative info to log file - if (household.getDebugChoiceModels()) - { - String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Micromobility Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - - }else if(modelStructure.getTourModeIsWalkTransit(s.getMode())) { - - //access - int tapPosition = mgraDataManager.getTapPosition(originMaz, s.boardTap); - if(tapPosition==-1) { - logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); - logger.warn("Can't find walk connection from origin to board TAP; skipping micromobility choice"); - return; - } - float walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); - mmDmuObject.setWalkTime(walkTime); - - if(microtransitMazs.contains(originMaz) && microtransitTaps.contains(s.boardTap)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - s.setMicromobilityAccessLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice(household, person, tour, s); - s.setMicromobilityAccessMode(chosenAlt); - - // write choice model alternative info to log file - if (household.getDebugChoiceModels()) - { - String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()+ " access choice"); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Micromobility Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - //egress - tapPosition = mgraDataManager.getTapPosition(destMaz, s.alightTap); - if(tapPosition==-1) { - logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); - logger.warn("Can't find walk connection from alight TAP to destination; skipping micromobility choice"); - return; - } - walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); - mmDmuObject.setWalkTime(walkTime); - - if(microtransitMazs.contains(destMaz) && microtransitTaps.contains(s.alightTap)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC - int closestMazToAlightTap = mgraDataManager.getClosestMgra(s.alightTap); - mmDmuObject.setDmuIndexValues(household.getHhId(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); - - // compute utilities and choose micromobility choice alternative. - logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - s.setMicromobilityEgressLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - chosenAlt = (byte) getChoice(household, person, tour, s); - s.setMicromobilityEgressMode(chosenAlt); - - // write choice model alternative info to log file - if (household.getDebugChoiceModels()) - { - String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()+ " egress choice"); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Micromobility Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - - - } else if( modelStructure.getTourModeIsDriveTransit(s.getMode()) ) { //drive-transit. Choose non-drive direction - - int tapPosition = 0; - float walkTime = 9999; - - if(s.isInboundStop()) { //inbound, so access mode is walk - tapPosition = mgraDataManager.getTapPosition(originMaz, s.boardTap); - if(tapPosition==-1) { - logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); - logger.warn("Can't find walk connection from origin to board TAP; skipping micromobility choice"); - return; - } - - walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(household.getHhId(), originMaz, originMaz, originMaz); - - if(microtransitMazs.contains(originMaz) && microtransitTaps.contains(s.boardTap)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - }else { //outbound so egress mode is walk. - tapPosition = mgraDataManager.getTapPosition(destMaz, s.alightTap); - if(tapPosition==-1) { - logger.warn("Problem with hh "+household.getHhId()+" Person "+person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - logger.warn("Origin MAZ "+originMaz+ " Board TAP "+s.boardTap+ " Alight TAP "+s.alightTap+" Destination MAZ "+destMaz); - logger.warn("Can't find walk connection from destination MAZ to alight TAP; skipping micromobility choice"); - return; - } - walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); - //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC - int closestMazToAlightTap = mgraDataManager.getClosestMgra(s.alightTap); - mmDmuObject.setDmuIndexValues(household.getHhId(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); - - if(microtransitMazs.contains(destMaz) && microtransitTaps.contains(s.alightTap)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - } - mmDmuObject.setWalkTime(walkTime); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice(household, person, tour, s); - - if(s.isInboundStop()) { //inbound, set access - s.setMicromobilityAccessMode(chosenAlt); - s.setMicromobilityAccessLogsum(logsum); - }else { //outound, set egress - s.setMicromobilityEgressMode(chosenAlt); - s.setMicromobilityEgressLogsum(logsum); - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels()) - { - String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - //String decisionMaker = String.format("Household %d", household.getHhId()+ "Person %d", person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Micromobility Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - - } - - } - - - /** - * Select the micromobility mode from the UEC. This is helper code for applyModel(), where utilities have already been calculated. - * - * @param household - * @param person - * @param tour - * @param s - * @return The micromobility mode. - */ - private int getChoice(Household household, Person person, Tour tour, Stop s) { - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - Random hhRandom = household.getHhRandom(); - if (mmModel.getAvailabilityCount() > 0) - { - double randomNumber = hhRandom.nextDouble(); - chosenAlt = mmModel.getChoiceResult(randomNumber); - return chosenAlt; - } else - { - String decisionMaker = String.format("Household " + household.getHhId()+ "Person " + person.getPersonNum()+" "+tour.getTourCategory()+" tour ID "+tour.getTourId()+ "stop "+s.getStopId()+ " mode " +s.getMode()); - String errorMessage = String - .format("Exception caught for %s, no available micromobility choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.info(errorMessage); - logger.info("Setting mode to walk"); - - mmModel.logUECResults(logger, decisionMaker); - return MM_MODEL_WALK_ALT; - } - - } - - /** - * This method calculates a cost coefficient based on the following formula: - * - * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) - * - * - * @param incomeCoeff - * @param incomeExponent - * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice - */ - public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ - - return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java deleted file mode 100644 index db7b60e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ModelStructure.java +++ /dev/null @@ -1,600 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -/** - * Holds the tour purpose list as well as the market segments for each tour. - * - * @author D. Ory - * - */ -public abstract class ModelStructure - implements Serializable -{ - - public static final String[] DC_SIZE_AREA_TYPE_BASED_SEGMENTS = {"CBD", - "URBAN", "SUBURBAN", "RURAL" }; - - public static final String MANDATORY_CATEGORY = "MANDATORY"; - public static final String JOINT_NON_MANDATORY_CATEGORY = "JOINT_NON_MANDATORY"; - public static final String INDIVIDUAL_NON_MANDATORY_CATEGORY = "INDIVIDUAL_NON_MANDATORY"; - public static final String AT_WORK_CATEGORY = "AT_WORK"; - - public static final String MANDATORY_PATTERN = "M"; - public static final String NONMANDATORY_PATTERN = "N"; - public static final String HOME_PATTERN = "H"; - - public static final int FIRST_DEPART_HOUR = 4; - public static final int LAST_DEPART_HOUR = 24; - public static final int FIRST_TOD_INTERVAL_HOUR = 430; - public static final int LAST_TOD_INTERVAL_HOUR = 2400; - public static final float TOD_INTERVAL_IN_MINUTES = 30.0f; - public static final int MAX_TOD_INTERVAL = 40; - protected String[] TOD_INTERVAL_LABELS; - - public static final int EA_SKIM_PERIOD_INDEX = 0; - public static final int AM_SKIM_PERIOD_INDEX = 1; - public static final int MD_SKIM_PERIOD_INDEX = 2; - public static final int PM_SKIM_PERIOD_INDEX = 3; - public static final int EV_SKIM_PERIOD_INDEX = 4; - public static final int[] SKIM_PERIOD_INDICES = { - EA_SKIM_PERIOD_INDEX, AM_SKIM_PERIOD_INDEX, MD_SKIM_PERIOD_INDEX, PM_SKIM_PERIOD_INDEX, - EV_SKIM_PERIOD_INDEX }; - - public static final int[] PERIODCODES = { EA_SKIM_PERIOD_INDEX, AM_SKIM_PERIOD_INDEX, - MD_SKIM_PERIOD_INDEX, PM_SKIM_PERIOD_INDEX, EV_SKIM_PERIOD_INDEX }; - - public static final String[] SKIM_PERIOD_STRINGS = {"EA", - "AM", "MD", "PM", "EV" }; - - // define indices associated with valid skim period combinations - public static final int EA_EA = 0; - public static final int EA_AM = 1; - public static final int EA_MD = 2; - public static final int EA_PM = 3; - public static final int EA_EV = 4; - // AM cannot be before EA - public static final int AM_EA = -1; - public static final int AM_AM = 5; - public static final int AM_MD = 6; - public static final int AM_PM = 7; - public static final int AM_EV = 8; - // MD cannot be before EA or AM - public static final int MD_EA = -1; - public static final int MD_AM = -1; - public static final int MD_MD = 9; - public static final int MD_PM = 10; - public static final int MD_EV = 11; - // PM cannot be before EA, AM or PM - public static final int PM_EA = -1; - public static final int PM_AM = -1; - public static final int PM_MD = -1; - public static final int PM_PM = 12; - public static final int PM_EV = 13; - // EV cannot be before EA, AM, MD or PM - public static final int EV_EA = -1; - public static final int EV_AM = -1; - public static final int EV_MD = -1; - public static final int EV_PM = -1; - public static final int EV_EV = 14; - - // define an array that contains the set of the valid skim period - // combination indices - public static final int[] SKIM_PERIOD_COMBINATION_INDICES = {EA_EA, - EA_AM, EA_MD, EA_PM, EA_EV, AM_AM, AM_MD, AM_PM, AM_EV, MD_MD, MD_PM, MD_EV, PM_PM, - PM_EV, EV_EV }; - - // define a 2-D array for the set of skim period combinations associatedf - // with each skim period index value - public static final int[][] SKIM_PERIOD_COMBINATIONS = { - {EA_EA, EA_AM, EA_MD, EA_PM, EA_EV}, {AM_EA, AM_AM, AM_MD, AM_PM, AM_EV}, - {MD_EA, MD_AM, MD_MD, MD_PM, MD_EV}, {PM_EA, PM_AM, PM_MD, PM_PM, PM_EV}, - {EV_EA, EV_AM, EV_MD, EV_PM, EV_EV} }; - - // define model period labels associated with each model period index - public static final String[] MODEL_PERIOD_LABELS = {"EA", - "AM", "MD", "PM", "EV" }; - - // the upper TOD interval index for each model period (EA:1-3, AM:6-9, - // MD:10-22, PM:23-29, EV:30-40) - public static final int UPPER_EA = 3; - public static final int UPPER_AM = 9; - public static final int UPPER_MD = 22; - public static final int UPPER_PM = 29; - - public static final int[] PERIOD_ENDS = {UPPER_EA,UPPER_AM,UPPER_MD,UPPER_PM, 40}; - - private HashMap indexTimePeriodMap; - private HashMap timePeriodIndexMap; - - public static final int WORKS_AT_HOME_ALTERNATUVE_INDEX = 2; - public static final int WORKS_AT_HOME_LOCATION_INDICATOR = 99999; - public static final int NOT_ENROLLED_SEGMENT_INDEX = 88888; - - private HashMap primaryTourPurposeNameIndexMap = new HashMap(); - private HashMap indexPrimaryTourPurposeNameMap = new HashMap(); - - public static final String WORK_PRIMARY_PURPOSE_NAME = "Work"; - public static final String UNIVERSITY_PRIMARY_PURPOSE_NAME = "University"; - public static final String SCHOOL_PRIMARY_PURPOSE_NAME = "School"; - public static final String ESCORT_PRIMARY_PURPOSE_NAME = "Escort"; - public static final String SHOPPING_PRIMARY_PURPOSE_NAME = "Shop"; - public static final String OTH_MAINT_PRIMARY_PURPOSE_NAME = "Maintenance"; - public static final String EAT_OUT_PRIMARY_PURPOSE_NAME = "Eating Out"; - public static final String VISITING_PRIMARY_PURPOSE_NAME = "Visiting"; - public static final String OTH_DISCR_PRIMARY_PURPOSE_NAME = "Discretionary"; - public static final String WORK_BASED_PRIMARY_PURPOSE_NAME = "Work-Based"; - - public static final int WORK_PRIMARY_PURPOSE_INDEX = 1; - public static final int UNIVERSITY_PRIMARY_PURPOSE_INDEX = 2; - public static final int SCHOOL_PRIMARY_PURPOSE_INDEX = 3; - public static final int ESCORT_PRIMARY_PURPOSE_INDEX = 4; - public static final int SHOPPING_PRIMARY_PURPOSE_INDEX = 5; - public static final int OTH_MAINT_PRIMARY_PURPOSE_INDEX = 6; - public static final int EAT_OUT_PRIMARY_PURPOSE_INDEX = 7; - public static final int VISITING_PRIMARY_PURPOSE_INDEX = 8; - public static final int OTH_DISCR_PRIMARY_PURPOSE_INDEX = 9; - public static final int WORK_BASED_PRIMARY_PURPOSE_INDEX = 10; - public static final int NUM_PRIMARY_PURPOSES = 10; - - public static final int WORK_STOP_PURPOSE_INDEX = 1; - public static final int UNIV_STOP_PURPOSE_INDEX = 2; - public static final int ESCORT_STOP_PURPOSE_INDEX = 4; - public static final int SHOP_STOP_PURPOSE_INDEX = 5; - public static final int MAINT_STOP_PURPOSE_INDEX = 6; - public static final int EAT_OUT_STOP_PURPOSE_INDEX = 7; - public static final int VISIT_STOP_PURPOSE_INDEX = 8; - public static final int DISCR_STOP_PURPOSE_INDEX = 9; - - public static final byte ESCORT_STOP_TYPE_DROPOFF = 1; - public static final byte ESCORT_STOP_TYPE_PICKUP = 2; - public static final int RIDE_SHARING_TYPE = 1; - public static final int PURE_ESCORTING_TYPE = 2; - - public static final int MAX_STOPS_PER_DIRECTION = 4; - - public String WORK_PURPOSE_NAME; - public String UNIVERSITY_PURPOSE_NAME; - public String SCHOOL_PURPOSE_NAME; - public String ESCORT_PURPOSE_NAME; - public String SHOPPING_PURPOSE_NAME; - public String EAT_OUT_PURPOSE_NAME; - public String OTH_MAINT_PURPOSE_NAME; - public String SOCIAL_PURPOSE_NAME; - public String OTH_DISCR_PURPOSE_NAME; - public String AT_WORK_PURPOSE_NAME; - public String AT_WORK_EAT_PURPOSE_NAME; - public String AT_WORK_BUSINESS_PURPOSE_NAME; - public String AT_WORK_MAINT_PURPOSE_NAME; - - public int AT_WORK_PURPOSE_INDEX_EAT; - public int AT_WORK_PURPOSE_INDEX_BUSINESS; - public int AT_WORK_PURPOSE_INDEX_MAINT; - - public String[] ESCORT_SEGMENT_NAMES; - public String[] AT_WORK_SEGMENT_NAMES; - - protected HashMap workSegmentNameIndexMap; - protected HashMap schoolSegmentNameIndexMap; - protected HashMap workSegmentIndexNameMap; - protected HashMap schoolSegmentIndexNameMap; - - // TODO: Determine which of the following can be eliminated - protected HashMap dcSoaUecIndexMap; - protected HashMap dcUecIndexMap; - protected HashMap tourModeChoiceUecIndexMap; - - protected HashMap dcSizeDcModelPurposeMap; - protected HashMap dcModelDcSizePurposeMap; - - protected HashMap dcModelPurposeIndexMap; // segments - // for - // which - // dc - // soa alternative models - // are applied - protected HashMap dcModelIndexPurposeMap; // segments - // for - // which - // dc - // soa alternative models - // are applied - - protected HashMap dcSizeSegmentIndexMap; // segments - // for - // which - // separate dc size - // coefficients are - // specified - protected HashMap dcSizeIndexSegmentMap; - protected HashMap dcSizeArrayPurposeIndexMap; // segments - // for - // which - // dc - // size terms are stored - protected HashMap dcSizeArrayIndexPurposeMap; - protected HashMap> dcSizePurposeSegmentMap; - - private String dcSizeCoeffPurposeFieldName = "purpose"; - private String dcSizeCoeffSegmentFieldName = "segment"; - - // TODO meld with what jim is doing on this front - protected String[] mandatoryDcModelPurposeNames; - protected String[] jointDcModelPurposeNames; - protected String[] nonMandatoryDcModelPurposeNames; - protected String[] atWorkDcModelPurposeNames; - - protected String workPurposeName; - protected String universityPurposeName; - protected String schoolPurposeName; - - protected String[] workPurposeSegmentNames; - protected String[] universityPurposeSegmentNames; - protected String[] schoolPurposeSegmentNames; - - protected HashMap stopFreqUecIndexMap; - protected HashMap stopLocUecIndexMap; - protected HashMap tripModeChoiceUecIndexMap; - - protected String[] jtfAltLabels; - protected String[] awfAltLabels; - - /** - * Assume name of the columns in the destination size coefficients file that - * contain the purpose strings is "purpose" and the column that contains the - * segment strings is "segment" - */ - public ModelStructure() - { - - workSegmentNameIndexMap = new HashMap(); - schoolSegmentNameIndexMap = new HashMap(); - workSegmentIndexNameMap = new HashMap(); - schoolSegmentIndexNameMap = new HashMap(); - - dcModelPurposeIndexMap = new HashMap(); - dcModelIndexPurposeMap = new HashMap(); - dcSoaUecIndexMap = new HashMap(); - dcUecIndexMap = new HashMap(); - tourModeChoiceUecIndexMap = new HashMap(); - stopFreqUecIndexMap = new HashMap(); - stopLocUecIndexMap = new HashMap(); - tripModeChoiceUecIndexMap = new HashMap(); - - // create a mapping between primary purpose - // names and purpose indices - createPrimaryPurposeMappings(); - - createIndexTimePeriodMap(); - - } - - public abstract HashMap getWorkSegmentNameIndexMap(); - - public abstract HashMap getSchoolSegmentNameIndexMap(); - - public abstract HashMap getWorkSegmentIndexNameMap(); - - public abstract HashMap getSchoolSegmentIndexNameMap(); - - // a derived class must implement these methods to retrieve purpose names - // for - // various personTypes making mandatory tours. - public abstract String getWorkPurpose(int incomeCategory); - - public abstract String getWorkPurpose(boolean isPtWorker, int incomeCategory); - - public abstract String getUniversityPurpose(); - - public abstract String getSchoolPurpose(int age); - - public abstract boolean getTourModeIsSov(int tourMode); - - public abstract boolean getTourModeIsSovOrHov(int tourMode); - - public abstract boolean getTourModeIsS2(int tourMode); - - public abstract boolean getTourModeIsS3(int tourMode); - - public abstract boolean getTourModeIsHov(int tourMode); - - public abstract boolean getTourModeIsNonMotorized(int tourMode); - - public abstract boolean getTourModeIsBike(int tourMode); - - public abstract boolean getTourModeIsWalk(int tourMode); - - public abstract boolean getTourModeIsTransit(int tourMode); - - public abstract boolean getTourModeIsWalkTransit(int tourMode); - - public abstract boolean getTourModeIsDriveTransit(int tourMode); - - public abstract boolean getTourModeIsPnr(int tourMode); - - public abstract boolean getTourModeIsKnr(int tourMode); - - public abstract boolean getTourModeIsSchoolBus(int tourMode); - - public abstract boolean getTourModeIsTncTransit(int tripMode); - - public abstract boolean getTourModeIsMaas(int tripMode); - - public abstract boolean getTripModeIsSovOrHov(int tripMode); - - public abstract boolean getTripModeIsWalkTransit(int tripMode); - - public abstract boolean getTripModeIsPnrTransit(int tripMode); - - public abstract boolean getTripModeIsKnrTransit(int tripMode); - - public abstract boolean getTripModeIsTransit(int tripMode); - - public abstract boolean getTripModeIsS2(int tripMode); - - public abstract boolean getTripModeIsS3(int tripMode); - - public abstract double[][] getCdap6PlusProps(); - - public abstract int getDefaultAmPeriod(); - - public abstract int getDefaultPmPeriod(); - - public abstract int getDefaultMdPeriod(); - - public abstract int getMaxTourModeIndex(); - - public abstract String getModelPeriodLabel(int period); - - public abstract int[] getSkimPeriodCombinationIndices(); - - public abstract int getSkimPeriodCombinationIndex(int startPeriod, int endPeriod); - - public abstract String getSkimMatrixPeriodString(int period); - - public abstract HashMap> getDcSizePurposeSegmentMap(); - - public abstract String[] getJtfAltLabels(); - - public abstract void setJtfAltLabels(String[] labels); - - private void createPrimaryPurposeMappings() - { - - primaryTourPurposeNameIndexMap.put(WORK_PRIMARY_PURPOSE_NAME, WORK_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(WORK_PRIMARY_PURPOSE_INDEX, WORK_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(UNIVERSITY_PRIMARY_PURPOSE_NAME, - UNIVERSITY_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(UNIVERSITY_PRIMARY_PURPOSE_INDEX, - UNIVERSITY_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(SCHOOL_PRIMARY_PURPOSE_NAME, - SCHOOL_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(SCHOOL_PRIMARY_PURPOSE_INDEX, - SCHOOL_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(ESCORT_PRIMARY_PURPOSE_NAME, - ESCORT_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(ESCORT_PRIMARY_PURPOSE_INDEX, - ESCORT_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(SHOPPING_PRIMARY_PURPOSE_NAME, - SHOPPING_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(SHOPPING_PRIMARY_PURPOSE_INDEX, - SHOPPING_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(OTH_MAINT_PRIMARY_PURPOSE_NAME, - OTH_MAINT_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(OTH_MAINT_PRIMARY_PURPOSE_INDEX, - OTH_MAINT_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(EAT_OUT_PRIMARY_PURPOSE_NAME, - EAT_OUT_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(EAT_OUT_PRIMARY_PURPOSE_INDEX, - EAT_OUT_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(VISITING_PRIMARY_PURPOSE_NAME, - VISITING_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(VISITING_PRIMARY_PURPOSE_INDEX, - VISITING_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(OTH_DISCR_PRIMARY_PURPOSE_NAME, - OTH_DISCR_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(OTH_DISCR_PRIMARY_PURPOSE_INDEX, - OTH_DISCR_PRIMARY_PURPOSE_NAME); - primaryTourPurposeNameIndexMap.put(WORK_BASED_PRIMARY_PURPOSE_NAME, - WORK_BASED_PRIMARY_PURPOSE_INDEX); - indexPrimaryTourPurposeNameMap.put(WORK_BASED_PRIMARY_PURPOSE_INDEX, - WORK_BASED_PRIMARY_PURPOSE_NAME); - - } - - /** - * @return the HashMap object that maps primary tour purpose - * names common to all CTRAMP implementations to indices (1-10). - */ - public HashMap getPrimaryPurposeNameIndexMap() - { - return primaryTourPurposeNameIndexMap; - } - - /** - * @return the HashMap object that maps indices (1-10) to - * primary tour purpose names common to all CTRAMP implementations. - */ - public HashMap getIndexPrimaryPurposeNameMap() - { - return indexPrimaryTourPurposeNameMap; - } - - /** - * @param purposeKey - * is the "purpose" name used as a key for the map to get the - * associated UEC tab number. - * @return the tab number of the UEC control file for the purpose - */ - public int getSoaUecIndexForPurpose(String purposeKey) - { - return dcSoaUecIndexMap.get(purposeKey); - } - - /** - * @param purposeKey - * is the "purpose" name used as a key for the map to get the - * associated UEC tab number. - * @return the tab number of the UEC control file for the purpose - */ - public int getDcUecIndexForPurpose(String purposeKey) - { - return dcUecIndexMap.get(purposeKey); - } - - /** - * @param purposeKey - * is the "purpose" name used as a key for the map to get the - * associated UEC tab number. - * @return the tab number of the UEC control file for the purpose - */ - public int getTourModeChoiceUecIndexForPurpose(String purposeKey) - { - return tourModeChoiceUecIndexMap.get(purposeKey); - } - - public String[] getDcModelPurposeList(String tourCategory) - { - if (tourCategory.equalsIgnoreCase(MANDATORY_CATEGORY)) return mandatoryDcModelPurposeNames; - else if (tourCategory.equalsIgnoreCase(JOINT_NON_MANDATORY_CATEGORY)) return jointDcModelPurposeNames; - else if (tourCategory.equalsIgnoreCase(INDIVIDUAL_NON_MANDATORY_CATEGORY)) return nonMandatoryDcModelPurposeNames; - else if (tourCategory.equalsIgnoreCase(AT_WORK_CATEGORY)) return atWorkDcModelPurposeNames; - else return null; - } - - public String getDcSizeCoeffPurposeFieldName() - { - return dcSizeCoeffPurposeFieldName; - } - - public String getDcSizeCoeffSegmentFieldName() - { - return this.dcSizeCoeffSegmentFieldName; - } - - public String getAtWorkEatPurposeName() - { - return AT_WORK_EAT_PURPOSE_NAME; - } - - public String[] getAtWorkSegmentNames() - { - return AT_WORK_SEGMENT_NAMES; - } - - public String getAtWorkBusinessPurposeName() - { - return AT_WORK_BUSINESS_PURPOSE_NAME; - } - - public String getAtWorkMaintPurposeName() - { - return AT_WORK_MAINT_PURPOSE_NAME; - } - - public int getAtWorkEatPurposeIndex() - { - return AT_WORK_PURPOSE_INDEX_EAT; - } - - public int getAtWorkBusinessPurposeIndex() - { - return AT_WORK_PURPOSE_INDEX_BUSINESS; - } - - public int getAtWorkMaintPurposeIndex() - { - return AT_WORK_PURPOSE_INDEX_MAINT; - } - - /** - * @param departPeriod - * is the model TOD interval for the departure period (for tour - * or trip) - * @return the skim period index associated with the departure interval - */ - public static int getSkimPeriodIndex(int departPeriod) - { - - int skimPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) skimPeriodIndex = EA_SKIM_PERIOD_INDEX; - else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM_SKIM_PERIOD_INDEX; - else if (departPeriod <= UPPER_MD) skimPeriodIndex = MD_SKIM_PERIOD_INDEX; - else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM_SKIM_PERIOD_INDEX; - else skimPeriodIndex = EV_SKIM_PERIOD_INDEX; - - return skimPeriodIndex; - - } - - /** - * @param departPeriod - * is the model TOD interval for the departure period (for tour - * or trip) - * @return the model period index associated with the departure interval - * Model periods: 0=EA, 1=AM, 2=MD, 3=PM, 4=EV - */ - public static int getModelPeriodIndex(int departPeriod) - { - - int modelPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; - else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; - else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; - else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; - else modelPeriodIndex = 4; - - return modelPeriodIndex; - - } - - private void createIndexTimePeriodMap() - { - indexTimePeriodMap = new HashMap(); - timePeriodIndexMap = new HashMap(); - - int numHours = LAST_DEPART_HOUR - FIRST_DEPART_HOUR; - int numHalfHours = numHours * 2; - - TOD_INTERVAL_LABELS = new String[numHalfHours + 1]; - - for (int i = 1; i <= numHalfHours; i++) - { - int time = ((int) (i / 2) + FIRST_DEPART_HOUR) * 100 + (i % 2) * 30; - indexTimePeriodMap.put(i, time); - timePeriodIndexMap.put(time, i); - TOD_INTERVAL_LABELS[i] = Integer.toString(time); - } - } - - public String[] getTimePeriodLabelArray() - { - return TOD_INTERVAL_LABELS; - } - - public String getTimePeriodLabel(int timePeriodIndex) - { - return TOD_INTERVAL_LABELS[timePeriodIndex]; - } - - // time argument is specified as: 500 for 5 am, 530 for 5:30 am, 1530 for - // 3:30 pm, etc. - public int getTimePeriodIndexForTime(int time) - { - return timePeriodIndexMap.get(time); - } - - public int getNumberOfTimePeriods() - { - return TOD_INTERVAL_LABELS.length - 1; - } - - public String[] getAwfAltLabels() - { - return awfAltLabels; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java deleted file mode 100644 index 29fc670..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/MyLogit.java +++ /dev/null @@ -1,115 +0,0 @@ -package org.sandag.abm.ctramp; - -import com.pb.common.math.MathUtil; -import com.pb.common.model.Alternative; -import com.pb.common.model.LogitModel; -import com.pb.common.model.ModelException; - -public class MyLogit - extends LogitModel -{ - - private static final int MAX_EXP_ARGUMENT = 400; - - private double[] utilities; - private double[] util; - private double[] constant; - private String[] altName; - - public MyLogit(String n, int numberOfAlternatives) - { - super(n, numberOfAlternatives); - - utilities = new double[numberOfAlternatives]; - util = new double[numberOfAlternatives]; - constant = new double[numberOfAlternatives]; - altName = new String[numberOfAlternatives]; - - nf.setMaximumFractionDigits(8); - nf.setMinimumFractionDigits(8); - } - - /** - * Overrides the base class getUtility() method to call a method to return - * the array of exponentiated utilities, having passed to it an array of - * utilities. - * - * @return The composite utility (logsum value) of all the alternatives. - */ - public double getUtility() throws ModelException - { - - double sum = 0; - double base = 0; - - // get the array of utility values to be exponentiated from the - // alternatives - // objects. - int i = 0; - for (int alt = 0; alt < alternatives.size(); ++alt) - { - Alternative thisAlt = (Alternative) alternatives.get(alt); - if (thisAlt.isAvailable()) - { - - // assign attributes of the alternatives - util[i] = thisAlt.getUtility(); - constant[i] = thisAlt.getConstant(); - altName[i] = thisAlt.getName(); - - // if alternative has a very large negative utility, it isn't - // available - if (util[i] + constant[i] < -MAX_EXP_ARGUMENT) - { - utilities[i] = -MAX_EXP_ARGUMENT; - } else - { - utilities[i] = dispersionParameter * (util[i] + constant[i]); - setAvailability(true); - } - - i++; - } else - { - utilities[i++] = -MAX_EXP_ARGUMENT; - } - } - - // exponentiate the utilities array and save result in expUtilities. - MathUtil.expArray(utilities, expUtilities); - - // sum the exponentiated utilities - for (i = 0; i < expUtilities.length; i++) - sum += expUtilities[i]; - - // if debug, and the alternatives is elemental, log the utility values - if (debug) - { - for (i = 0; i < expUtilities.length; i++) - { - Boolean elemental = (Boolean) isElementalAlternative.get(i); - if (elemental.equals(Boolean.TRUE)) - logger.info(String.format("%-20s", altName[i]) + "\t\t" + nf.format(util[i]) - + "\t\t\t" + nf.format(constant[i]) + "\t\t\t" - + nf.format(Math.exp(utilities[i]))); - } - } - - if (isAvailable()) - { - base = (1 / dispersionParameter) * MathUtil.log(sum); - - if (Double.isNaN(base)) throw new ModelException(ModelException.INVALID_UTILITY); - - if (debug) - logger.info(String.format("%-20s", getName() + " logsum:") + "\t\t" - + nf.format(base)); - - return base; - } - - // if nothing avaiable, return a bad utilty - return -999; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java deleted file mode 100644 index 04a5ecc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryDestChoiceModel.java +++ /dev/null @@ -1,1527 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.Random; -import java.util.ResourceBundle; - -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.IndexSort; -import com.pb.common.util.ResourceUtil; -import com.pb.common.newmodel.ChoiceModelApplication; -import org.apache.log4j.Logger; - -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import org.sandag.abm.accessibilities.NonTransitUtilities; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampDmuFactoryIf; -import org.sandag.abm.ctramp.DcSoaDMU; -import org.sandag.abm.ctramp.DestChoiceDMU; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Tour; -import org.sandag.abm.ctramp.DestinationSampleOfAlternativesModel; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import org.sandag.abm.ctramp.TourModeChoiceModel; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.visitor.VisitorTour; -public class NonMandatoryDestChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(NonMandatoryDestChoiceModel.class); - private transient Logger dcNonManLogger = Logger.getLogger("tourDcNonMan"); - private transient Logger todMcLogger = Logger.getLogger("todMcLogsum"); - - // TODO eventually remove this target - private static final String PROPERTIES_DC_UEC_FILE = "nmdc.uec.file"; - private static final String PROPERTIES_DC_UEC_FILE2 = "nmdc.uec.file2"; - private static final String PROPERTIES_DC_SOA_UEC_FILE = "nmdc.soa.uec.file"; - - private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; - - private static final String PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY = "nmdc.soa.SampleSize"; - - private static final String PROPERTIES_DC_DATA_SHEET = "nmdc.data.page"; - - private static final String PROPERTIES_DC_ESCORT_MODEL_SHEET = "nmdc.escort.model.page"; - private static final String PROPERTIES_DC_SHOP_MODEL_SHEET = "nmdc.shop.model.page"; - private static final String PROPERTIES_DC_MAINT_MODEL_SHEET = "nmdc.maint.model.page"; - private static final String PROPERTIES_DC_EATOUT_MODEL_SHEET = "nmdc.eat.model.page"; - private static final String PROPERTIES_DC_VISIT_MODEL_SHEET = "nmdc.visit.model.page"; - private static final String PROPERTIES_DC_DISCR_MODEL_SHEET = "nmdc.discr.model.page"; - - private static final String PROPERTIES_DC_SOA_ESCORT_MODEL_SHEET = "nmdc.soa.escort.model.page"; - private static final String PROPERTIES_DC_SOA_SHOP_MODEL_SHEET = "nmdc.soa.shop.model.page"; - private static final String PROPERTIES_DC_SOA_MAINT_MODEL_SHEET = "nmdc.soa.maint.model.page"; - private static final String PROPERTIES_DC_SOA_EATOUT_MODEL_SHEET = "nmdc.soa.eat.model.page"; - private static final String PROPERTIES_DC_SOA_VISIT_MODEL_SHEET = "nmdc.soa.visit.model.page"; - private static final String PROPERTIES_DC_SOA_DISCR_MODEL_SHEET = "nmdc.soa.discr.model.page"; - - private static final String PROPERTIES_DC_SAMPLE_TOD_PERIOD = "nmdc.SampleTODPeriod"; - private static final String PROPERTIES_SAMPLE_TOD_PERIOD_FILE = "nmdc.SampleTODPeriod.file"; - private boolean sampleTODPeriod = false; - private double[][] cumProbability; // by purpose, alternative: cumulative probability distribution - private int[][] outboundPeriod; // by purpose, alternative: outbound period - private int[][] returnPeriod; // by purpose, alternative: return period - - - private static final String[] TOUR_PURPOSE_NAMES = { - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME }; - - // set priority ranking for non-mandatory purposes - include 0 values for 0 - // element and mandatory purposes - private static final int[] TOUR_PURPOSE_PRIORITIES = {0, - 0, 0, 0, 1, 3, 2, 6, 4, 5 }; - - private static final String[] DC_MODEL_SHEET_KEYS = { - PROPERTIES_DC_ESCORT_MODEL_SHEET, PROPERTIES_DC_SHOP_MODEL_SHEET, - PROPERTIES_DC_MAINT_MODEL_SHEET, PROPERTIES_DC_EATOUT_MODEL_SHEET, - PROPERTIES_DC_VISIT_MODEL_SHEET, PROPERTIES_DC_DISCR_MODEL_SHEET }; - - private static final String[] DC_SOA_MODEL_SHEET_KEYS = { - PROPERTIES_DC_SOA_ESCORT_MODEL_SHEET, PROPERTIES_DC_SOA_SHOP_MODEL_SHEET, - PROPERTIES_DC_SOA_MAINT_MODEL_SHEET, PROPERTIES_DC_SOA_EATOUT_MODEL_SHEET, - PROPERTIES_DC_SOA_VISIT_MODEL_SHEET, PROPERTIES_DC_SOA_DISCR_MODEL_SHEET }; - - // all three subtour purposes use the same SOA sheet - private final int[] sizeSheetIndices = { - BuildAccessibilities.ESCORT_INDEX, BuildAccessibilities.SHOP_INDEX, - BuildAccessibilities.OTH_MAINT_INDEX, BuildAccessibilities.EATOUT_INDEX, - BuildAccessibilities.VISIT_INDEX, BuildAccessibilities.OTH_DISCR_INDEX }; - - // set default depart periods that represents each model period - private static final int EA = 1; - private static final int AM = 8; - private static final int MD = 16; - private static final int PM = 26; - private static final int EV = 36; - - private static final int[][][] PERIOD_COMBINATIONS = { - { {AM, AM}, {MD, MD}, {PM, PM}}, { {MD, MD}, {PM, PM}, {EV, EV}}, - { {AM, MD}, {MD, PM}, {PM, EV}}, { {MD, MD}, {PM, PM}, {EV, EV}}, - { {MD, MD}, {PM, PM}, {EV, EV}}, { {AM, MD}, {MD, PM}, {PM, EV}} }; - - private static final double[][] PERIOD_COMBINATION_COEFFICIENTS = { - {-1.065820, -0.871051, -1.439514}, {-0.467154, -1.411351, -2.044826}, - {-0.941865, -0.813977, -1.789714}, {-1.007316, -0.968856, -1.365375}, - {-1.081531, -1.121260, -1.093461}, {-1.258919, -1.155085, -0.913773} }; - - private ModelStructure modelStructure; - - private int[] dcModelIndices; - private HashMap purposeNameIndexMap; - HashMap nonMandatorySegmentNameIndexMap; - HashMap nonMandatorySizeSegmentNameIndexMap; - - private double[][] dcSizeArray; - - private TourModeChoiceDMU mcDmuObject; - private DestChoiceDMU dcDmuObject; - private DestChoiceTwoStageModelDMU dcDistSoaDmuObject; - private DcSoaDMU dcSoaDmuObject; - - private boolean[] needToComputeLogsum; - private double[] modeChoiceLogsums; - - private TourModeChoiceModel mcModel; - private DestinationSampleOfAlternativesModel dcSoaModel; - private ChoiceModelApplication[] dcModel; - private ChoiceModelApplication[] dcModel2; - - private boolean[] dcModel2AltsAvailable; - private int[] dcModel2AltsSample; - private int[] dcModel2SampleValues; - - private double[] mgraDistanceArray; - - private BuildAccessibilities aggAcc; - - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private DestChoiceTwoStageModel dcSoaTwoStageObject; - - private boolean useNewSoaMethod; - - private int soaSampleSize; - - private long soaRunTime; - - public NonMandatoryDestChoiceModel(HashMap propertyMap, - ModelStructure myModelStructure, BuildAccessibilities myAggAcc, - CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel myMcModel) - { - - logger.info("setting up Non-Mandatory tour destination choice model."); - - // set the model structure and the tour purpose list - this.modelStructure = myModelStructure; - this.mcModel = myMcModel; - aggAcc = myAggAcc; - - mgraManager = MgraDataManager.getInstance(); - tazs = TazDataManager.getInstance(); - - soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, - PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); - - useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, - USE_NEW_SOA_METHOD_PROPERTY_KEY); - - if (useNewSoaMethod) - dcSoaTwoStageObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); - - // create an array of ChoiceModelApplication objects for each choice - // purpose - setupDestChoiceModelArrays(propertyMap, dmuFactory); - - sampleTODPeriod = Util.getBooleanValueFromPropertyMap(propertyMap, PROPERTIES_DC_SAMPLE_TOD_PERIOD); - String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); - String diurnalFile = Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_SAMPLE_TOD_PERIOD_FILE); - diurnalFile = directory + diurnalFile; - - if(sampleTODPeriod) - readTODFile(diurnalFile); - } - - /** - * Read the TOD distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readTODFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating tour TOD probability distribution"); - - int purposes = PERIOD_COMBINATIONS.length; // start at 0 - int periods = ModelStructure.MAX_TOD_INTERVAL; // start at 1 - int periodCombinations = periods * (periods + 1) / 2; - - cumProbability = new double[purposes][periodCombinations]; - outboundPeriod = new int[purposes][periodCombinations]; - returnPeriod = new int[purposes][periodCombinations]; - - // fill up arrays - int rowCount = probabilityTable.getRowCount(); - int lastPurpose = -99; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose") - 4; //4 mandatory purposes, first non-mand purpose is escort - 4 - int outPer = (int) probabilityTable.getValueAt(row, "OutboundPeriod"); - int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); - - // continue if return period before outbound period - if (retPer < outPer) continue; - - // reset if new purpose - if (purpose != lastPurpose) - { - - // log cumulative probability just in case - if (lastPurpose != -99) - logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); - cumProb = 0; - alt = 0; - } - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - outboundPeriod[purpose][alt] = outPer; - returnPeriod[purpose][alt] = retPer; - - ++alt; - - lastPurpose = purpose; - } - - logger.info("End calculating tour TOD probability distribution"); - - } - - /** - * Calculate tour time of day for the tour. - * - * @param tour - * A tour (with purpose) - */ - public double sampleTODPeriodAndCalculateDCLogsum(Person person, Tour tour, int sampleDestMgra) - { - - Logger modelLogger = todMcLogger; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - Household household = person.getHouseholdObject(); - - if (household.getDebugChoiceModels()) - { - choiceModelDescription = String - .format("Non-Mandatory sample TOD logsum calculations for %s Location Choice", - tour.getTourPurpose()); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourId(), person.getListOfIndividualNonMandatoryTours().size()); - - } - - - double random = household.getHhRandom().nextDouble(); - int purpose = purposeNameIndexMap.get(tour.getTourPurpose()); - - int depart = -1; - int arrive = -1; - if (household.getDebugChoiceModels()) - { - logger.info("Choosing tour time of day for purpose " - + tour.getTourPurpose() + " using random number " + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - //Wu added to prevent large random number resulting in invalid choice - if (random>0.999999) { - depart = outboundPeriod[purpose][cumProbability[purpose].length-1]; - arrive = returnPeriod[purpose][cumProbability[purpose].length-1]; - break; - } - if (random < cumProbability[purpose][i]) - { - depart = outboundPeriod[purpose][i]; - arrive = returnPeriod[purpose][i]; - break; - } - } - if((depart ==-1)||(arrive==-1)){ - logger.fatal("Error: did not find outbound or return period for tour"); - logger.fatal("Depart period, arrive period = "+depart+","+arrive); - logger.fatal("Random number: "+random); - tour.logTourObject(logger,100); - throw new RuntimeException(); - } - - String periodString = modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(depart)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(arrive)); - - if (household.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose depart period " + depart + " and arrival period " - + arrive); - logger.info(""); - } - - // set the mode choice attributes needed by @variables in the UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, depart, arrive, sampleDestMgra); - - double logsum = -999; - try - { - logsum = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel+","+periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + tour.getTourPrimaryPurpose() + " tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - e.printStackTrace(); - throw new RuntimeException(e); - } - - if (household.getDebugChoiceModels()) - modelLogger.info("Mode choice logsum for sampled mgra = " + logsum); - - return logsum; - - } - - - private void setupDestChoiceModelArrays(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - String dcUecFileName = propertyMap.get(PROPERTIES_DC_UEC_FILE); - dcUecFileName = uecFileDirectory + dcUecFileName; - - String dcUecFileName2 = propertyMap.get(PROPERTIES_DC_UEC_FILE2); - dcUecFileName2 = uecFileDirectory + dcUecFileName2; - - String soaUecFileName = propertyMap.get(PROPERTIES_DC_SOA_UEC_FILE); - soaUecFileName = uecFileDirectory + soaUecFileName; - - int dcModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, - PROPERTIES_DC_DATA_SHEET); - - dcDmuObject = dmuFactory.getDestChoiceDMU(); - dcDmuObject.setAggAcc(aggAcc); - dcDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); - - if (useNewSoaMethod) - { - dcDistSoaDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); - dcDistSoaDmuObject.setAggAcc(aggAcc); - dcDistSoaDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); - } - - dcSoaDmuObject = dmuFactory.getDcSoaDMU(); - dcSoaDmuObject.setAggAcc(aggAcc); - - mcDmuObject = dmuFactory.getModeChoiceDMU(); - - int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; - needToComputeLogsum = new boolean[numLogsumIndices]; - modeChoiceLogsums = new double[numLogsumIndices]; - - // create the arrays of dc model and soa model indices - int[] uecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; - int[] soaUecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; - - purposeNameIndexMap = new HashMap(TOUR_PURPOSE_NAMES.length); - - int i = 0; - for (String purposeName : TOUR_PURPOSE_NAMES) - { - int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); - int soaUecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, - DC_SOA_MODEL_SHEET_KEYS[i]); - purposeNameIndexMap.put(purposeName, i); - uecSheetIndices[i] = uecIndex; - soaUecSheetIndices[i] = soaUecIndex; - i++; - } - - // create a lookup array to map purpose index to model index - dcModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int dcModelIndex = 0; - int dcSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, dcModelIndex); - dcModelIndices[dcSegmentIndex] = dcModelIndex++; - } else - { - dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); - } - - dcSegmentIndex++; - } - - // the size term array in aggAcc gives mgra*purpose - need an array of - // all mgras for one purpose - double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); - nonMandatorySegmentNameIndexMap = new HashMap(); - nonMandatorySizeSegmentNameIndexMap = new HashMap(); - for (int k = 0; k < TOUR_PURPOSE_NAMES.length; k++) - { - nonMandatorySegmentNameIndexMap.put(TOUR_PURPOSE_NAMES[k], k); - nonMandatorySizeSegmentNameIndexMap.put(TOUR_PURPOSE_NAMES[k], sizeSheetIndices[k]); - } - - dcSizeArray = new double[TOUR_PURPOSE_NAMES.length][aggAccDcSizeArray.length]; - for (i = 0; i < aggAccDcSizeArray.length; i++) - { - for (int m : nonMandatorySegmentNameIndexMap.values()) - { - int s = sizeSheetIndices[m]; - dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; - } - } - - dcModel = new ChoiceModelApplication[modelIndexMap.size()]; - - if (useNewSoaMethod) - { - dcModel2 = new ChoiceModelApplication[modelIndexMap.size()]; - dcModel2AltsAvailable = new boolean[soaSampleSize + 1]; - dcModel2AltsSample = new int[soaSampleSize + 1]; - dcModel2SampleValues = new int[soaSampleSize]; - } else - { - // create a sample of alternatives choice model object for use in - // selecting a sample - // of all possible destination choice alternatives. - dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFileName, soaSampleSize, - propertyMap, mgraManager, dcSizeArray, dcSoaDmuObject, soaUecSheetIndices); - } - - i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - - try - { - dcModel[i] = new ChoiceModelApplication(dcUecFileName, uecIndex, dcModelDataSheet, - propertyMap, (VariableTable) dcDmuObject); - - if (useNewSoaMethod) - { - dcModel2[i] = new ChoiceModelApplication(dcUecFileName2, uecIndex, - dcModelDataSheet, propertyMap, (VariableTable) dcDistSoaDmuObject); - } - - i++; - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up DC ChoiceModelApplication[%d] for model index=%d of %d models", - i, i, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; - } - - public void applyIndivModel(Household hh) - { - - soaRunTime = 0; - - if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); - else dcSoaModel.resetSoaRunTime(); - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - - Person[] persons = hh.getPersons(); - - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - // get the individual non-mandatory tours for this person and choose - // a destination for each. - ArrayList tourList = getPriorityOrderedTourList(p - .getListOfIndividualNonMandatoryTours()); - - int currentTourNum = 0; - for (Tour tour : tourList) - { - - if(tour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE||tour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE|| - tour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE||tour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE) - continue; - - int chosen = -1; - try - { - - int homeTaz = hh.getHhTaz(); - int origMgra = tour.getTourOrigMgra(); - - // update the MC dmuObject for this person - mcDmuObject.setHouseholdObject( hh ); - mcDmuObject.setPersonObject( p ); - mcDmuObject.setTourObject( tour ); - mcDmuObject.setDmuIndexValues( hh.getHhId(), homeTaz, origMgra, 0, hh.getDebugChoiceModels() ); - mcDmuObject.setOriginMgra(origMgra); - - // update the DC dmuObject for this person - dcDmuObject.setHouseholdObject(hh); - dcDmuObject.setPersonObject(p); - dcDmuObject.setTourObject(tour); - dcDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); - - if (useNewSoaMethod) - { - dcDistSoaDmuObject.setHouseholdObject(hh); - dcDistSoaDmuObject.setPersonObject(p); - dcDistSoaDmuObject.setTourObject(tour); - dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); - } - - // for individual non-mandatory DC, just count remaining - // individual non-mandatory tours - int toursLeftCount = tourList.size() - currentTourNum; - dcDmuObject.setToursLeftCount(toursLeftCount); - if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); - - // get the tour location alternative chosen from the sample - if (useNewSoaMethod) - { - chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); - soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); - } else - { - chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, - dcSoaDmuObject, mcDmuObject); - soaRunTime += dcSoaModel.getSoaRunTime(); - } - - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught selecting individual non-mandatory tour destination choice for hh.hhid=%d, personNum=%d, tourId=%d, purposeName=%s", - hh.getHhId(), p.getPersonNum(), tour.getTourId(), - tour.getTourPurpose())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - // set chosen values in tour object - tour.setTourDestMgra(chosen); - - currentTourNum++; - } - - } - - hh.setInmtlRandomCount(hh.getHhRandomCount()); - - } - - public void applyJointModel(Household hh) - { - - soaRunTime = 0; - - if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); - else dcSoaModel.resetSoaRunTime(); - - // if no joint non-mandatory tours, nothing to do for this household. - Tour[] jointTours = hh.getJointTourArray(); - if (jointTours == null || jointTours.length == 0) return; - - // get the individual non-mandatory tours for this person and choose a - // destination for each. - ArrayList tourList = getPriorityOrderedTourList(jointTours); - - int currentTourNum = 0; - for (Tour tour : tourList) - { - - int chosen = -1; - try - { - - int homeTaz = hh.getHhTaz(); - int origMgra = tour.getTourOrigMgra(); - - // update the MC dmuObject for this person - mcDmuObject.setHouseholdObject( hh ); - mcDmuObject.setPersonObject( null ); - mcDmuObject.setTourObject( tour ); - mcDmuObject.setDmuIndexValues( hh.getHhId(), homeTaz, origMgra, 0, hh.getDebugChoiceModels() ); - mcDmuObject.setOriginMgra(origMgra); - - // update the DC dmuObject for this person - dcDmuObject.setHouseholdObject(hh); - dcDmuObject.setPersonObject(null); - dcDmuObject.setTourObject(tour); - dcDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); - - if (useNewSoaMethod) - { - dcDistSoaDmuObject.setHouseholdObject(hh); - dcDistSoaDmuObject.setPersonObject(null); - dcDistSoaDmuObject.setTourObject(tour); - dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); - } - - // for individual non-mandatory DC, just count remaining - // individual non-mandatory tours - int toursLeftCount = tourList.size() - currentTourNum; - dcDmuObject.setToursLeftCount(toursLeftCount); - if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); - - // get the tour location alternative chosen from the sample - if (useNewSoaMethod) - { - chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); - soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); - } else - { - chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, - dcSoaDmuObject, mcDmuObject); - soaRunTime += dcSoaModel.getSoaRunTime(); - } - - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught selecting joint non-mandatory tour destination choice for hh.hhid=%d, tourId=%d, purposeName=%s", - hh.getHhId(), tour.getTourId(), tour.getTourPurpose())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - // set chosen values in tour object - tour.setTourDestMgra(chosen); - - currentTourNum++; - } - - hh.setJtlRandomCount(hh.getHhRandomCount()); - - } - - /** - * - * @return chosen mgra. - */ - private int selectLocationFromSampleOfAlternatives(Tour tour, DestChoiceDMU dcDmuObject, - DcSoaDMU dcSoaDmuObject, TourModeChoiceDMU mcDmuObject) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcNonManLogger; - - // get the Household object for the person making this non-mandatory - // tour - Person person = tour.getPersonObject(); - - // get the Household object for the person making this non-mandatory - // tour - Household household = person.getHouseholdObject(); - - // get the tour purpose name - String tourPurposeName = tour.getTourPurpose(); - int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); - - int sizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurposeName); - dcSoaDmuObject.setDestChoiceSize(dcSizeArray[sizeIndex]); - - // double[] homeMgraDistanceArray = - // mandAcc.calculateDistancesForAllMgras( household.getHhMgra() ); - mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(household.getHhMgra(), - mgraDistanceArray); - dcSoaDmuObject.setDestDistance(mgraDistanceArray); - - dcDmuObject.setDestChoiceSize(dcSizeArray[sizeIndex]); - dcDmuObject.setDestChoiceDistance(mgraDistanceArray); - - // compute the sample of alternatives set for the person - dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, tour, person, - tourPurposeName, tourPurposeIndex, household.getHhMgra()); - - // get sample of locations and correction factors for sample - int[] finalSample = dcSoaModel.getSampleOfAlternatives(); - float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); - - int m = dcModelIndices[tourPurposeIndex]; - int numAlts = dcModel[m].getNumberOfAlternatives(); - - // set the destAltsAvailable array to true for all destination choice - // alternatives for each purpose - boolean[] destAltsAvailable = new boolean[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsAvailable[k] = false; - - // set the destAltsSample array to 1 for all destination choice - // alternatives - // for each purpose - int[] destAltsSample = new int[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsSample[k] = 0; - - int[] sampleValues = new int[finalSample.length]; - - // for the destinations and sub-zones in the sample, compute mc logsums - // and - // save in DC dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 1; i < finalSample.length; i++) - { - - int destMgra = finalSample[i]; - sampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = -999; - if(sampleTODPeriod) - logsum = sampleTODPeriodAndCalculateDCLogsum(person, tour, destMgra); - else - logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); - - dcDmuObject.setMcLogsum(destMgra, logsum); - - // set sample of alternatives correction factor used in destination - // choice utility for the sampled alternative. - dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); - - // set availaibility and sample values for the purpose, dcAlt. - destAltsAvailable[finalSample[i]] = true; - destAltsSample[finalSample[i]] = 1; - - } - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" - + tourPurposeName + ", Person Num: " + person.getPersonNum() - + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - dcModel[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float modelLogsum = (float) dcModel[m].computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), - destAltsAvailable, destAltsSample); - - tour.setTourDestinationLogsum(modelLogsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (dcModel[m].getAvailabilityCount() > 0) - { - try - { - chosen = dcModel[m].getChoiceResult(rn); - } catch (Exception e) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), tour.getTourId(), - tourPurposeName)); - throw new RuntimeException(); - } - } - - // write choice model alternative info to log file - int selectedIndex = -1; - for (int j = 1; j < finalSample.length; j++) - { - if (finalSample[j] == chosen) - { - selectedIndex = j; - break; - } - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = dcModel[m].getUtilities(); - double[] probabilities = dcModel[m].getProbabilities(); - boolean[] availabilities = dcModel[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); - - double cumProb = 0.0; - for (int j = 1; j < finalSample.length; j++) - { - int k = sortedSampleValueIndices[j]; - int alt = finalSample[k]; - - if (finalSample[k] == chosen) selectedIndex = j; - - cumProb += probabilities[alt - 1]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[alt], utilities[alt - 1], probabilities[alt - 1], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", selectedIndex, chosen); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - dcModel[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - dcModel[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - dcModel[m].logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), tour.getTourId(), - tourPurposeName)); - throw new RuntimeException(); - } - - } - - return chosen; - - } - - /** - * - * @return chosen mgra. - */ - private int selectLocationFromTwoStageSampleOfAlternatives(Tour tour, - TourModeChoiceDMU mcDmuObject) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcNonManLogger; - - // get the Household object for the person making this non-mandatory - // tour - Person person = tour.getPersonObject(); - - // get the Household object for the person making this non-mandatory - // tour - Household household = person.getHouseholdObject(); - - // get the tour purpose name - String tourPurposeName = tour.getTourPurpose(); - int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); - - // get sample of locations and correction factors for sample using the - // alternate method - // for non-mandatory tour destination choice, the sizeSegmentType INdex - // and sizeSegmentIndex are the same values. - dcSoaTwoStageObject.chooseSample(mgraManager.getTaz(tour.getTourOrigMgra()), - tourPurposeIndex, tourPurposeIndex, soaSampleSize, household.getHhRandom(), - household.getDebugChoiceModels()); - int[] finalSample = dcSoaTwoStageObject.getUniqueSampleMgras(); - double[] sampleCorrectionFactors = dcSoaTwoStageObject - .getUniqueSampleMgraCorrectionFactors(); - int numUniqueAlts = dcSoaTwoStageObject.getNumberofUniqueMgrasInSample(); - - int m = dcModelIndices[tourPurposeIndex]; - int numAlts = dcModel2[m].getNumberOfAlternatives(); - - Arrays.fill(dcModel2AltsAvailable, false); - Arrays.fill(dcModel2AltsSample, 0); - Arrays.fill(dcModel2SampleValues, 999999); - - mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(household.getHhMgra(), - mgraDistanceArray); - dcDistSoaDmuObject.setMgraDistanceArray(mgraDistanceArray); - - int sizeIndex = nonMandatorySizeSegmentNameIndexMap.get(tourPurposeName); - dcDistSoaDmuObject.setMgraSizeArray(dcSizeArray[sizeIndex]); - - // set sample of alternatives correction factors used in destination - // choice utility for the sampled alternatives. - dcDistSoaDmuObject.setDcSoaCorrections(sampleCorrectionFactors); - - // for the destination mgras in the sample, compute mc logsums and save - // in dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 0; i < numUniqueAlts; i++) - { - - int destMgra = finalSample[i]; - dcModel2SampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = -999; - if(sampleTODPeriod) - logsum = sampleTODPeriodAndCalculateDCLogsum(person, tour, destMgra); - else - logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); - dcDistSoaDmuObject.setMcLogsum(i, logsum); - - // set availaibility and sample values for the purpose, dcAlt. - dcModel2AltsAvailable[i + 1] = true; - dcModel2AltsSample[i + 1] = 1; - - } - - dcDistSoaDmuObject.setSampleArray(dcModel2SampleValues); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" - + tourPurposeName + ", Person Num: " + person.getPersonNum() - + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - dcModel2[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float logsum = (float) dcModel2[m].computeUtilities(dcDistSoaDmuObject, dcDistSoaDmuObject.getDmuIndexValues(), - dcModel2AltsAvailable, dcModel2AltsSample); - - tour.setTourDestinationLogsum(logsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (dcModel2[m].getAvailabilityCount() > 0) - { - try - { - chosen = dcModel2[m].getChoiceResult(rn); - } catch (Exception e) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", - dcDistSoaDmuObject.getHouseholdObject().getHhId(), - dcDistSoaDmuObject.getPersonObject().getPersonNum(), - tour.getTourId(), tourPurposeName)); - throw new RuntimeException(); - } - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = dcModel2[m].getUtilities(); - double[] probabilities = dcModel2[m].getProbabilities(); - boolean[] availabilities = dcModel2[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < finalSample.length; j++) - { - int alt = finalSample[j]; - cumProb += probabilities[j]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - dcModel2[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - dcModel2[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - dcModel2[m].logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", - dcDistSoaDmuObject.getHouseholdObject().getHhId(), - dcDistSoaDmuObject.getPersonObject().getPersonNum(), - tour.getTourId(), tourPurposeName)); - throw new RuntimeException(); - } - - } - - return chosen; - - } - - private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, - int startPeriod, int endPeriod, int sampleDestMgra) - { - - t.setTourDestMgra(sampleDestMgra); - t.setTourDepartPeriod(startPeriod); - t.setTourArrivePeriod(endPeriod); - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(t); - mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), - t.getTourOrigMgra(), sampleDestMgra, household.getDebugChoiceModels()); - - mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t - .getTourOrigMgra()))); - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(sampleDestMgra))); - mcDmuObject.setOriginMgra(t.getTourOrigMgra()); - mcDmuObject.setDestMgra(t.getTourDestMgra()); - - } - - - /** - * This method calculates TOD choice logsum for the person, tour and sampled destination. - * @param person - * @param tour - * @param sampleDestMgra - * @param sampleNum - * @return The logsum. - */ - private double calculateSimpleTODChoiceLogsum(Person person, Tour tour, int sampleDestMgra, - int sampleNum) - { - - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todMcLogger; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - choiceModelDescription = String - .format("Non-Mandatory Simplified TOD logsum calculations for %s Location Choice, Sample Number %d", - tour.getTourPurpose(), sampleNum); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourId(), person.getListOfIndividualNonMandatoryTours().size()); - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - - } - - int i = 0; - int tourPurposeIndex = purposeNameIndexMap.get(tour.getTourPurpose()); - double totalExpUtility = 0.0; - for (int[] combo : PERIOD_COMBINATIONS[tourPurposeIndex]) - { - int startPeriod = combo[0]; - int endPeriod = combo[1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod, - sampleDestMgra); - - if (household.getDebugChoiceModels()) - { - modelLogger.info(""); - modelLogger.info(""); - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, tour); - } - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " " + tour.getTourPrimaryPurpose() + " tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - e.printStackTrace(); - //System.exit(-1); - throw new RuntimeException(e); - } - needToComputeLogsum[index] = false; - } - - double expUtil = Math.exp(modeChoiceLogsums[index] - + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]); - totalExpUtility += expUtil; - - if (household.getDebugChoiceModels()) - modelLogger - .info("i = " - + i - + ", purpose = " - + tourPurposeIndex - + ", " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)) - + " MCLS = " - + modeChoiceLogsums[index] - + ", ASC = " - + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i] - + ", (MCLS + ASC) = " - + (modeChoiceLogsums[index] + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]) - + ", exp(MCLS + ASC) = " + expUtil + ", cumExpUtility = " - + totalExpUtility); - - i++; - } - - double logsum = Math.log(totalExpUtility); - - if (household.getDebugChoiceModels()) - modelLogger.info("final simplified TOD logsum = " + logsum); - - return logsum; - } - - /** - * takes an ArrayList of tours - * - * @return a new ArrayList ordered by priority - */ - private ArrayList getPriorityOrderedTourList(ArrayList toursIn) - { - - int[] tourPriorities = new int[toursIn.size()]; - - int i = 0; - for (Tour tour : toursIn) - { - String purposeName = tour.getTourPurpose(); - int purposeIndex = purposeNameIndexMap.get(purposeName); - int purposePriority = TOUR_PURPOSE_PRIORITIES[purposeIndex]; - tourPriorities[i] = purposePriority; - } - - int[] sortedIndices = IndexSort.indexSort(tourPriorities); - ArrayList toursOut = new ArrayList(toursIn.size()); - - for (i = 0; i < toursIn.size(); i++) - toursOut.add(toursIn.get(sortedIndices[i])); - - return toursOut; - } - - /** - * takes an ArrayList of tours - * - * @return a new ArrayList ordered by priority - */ - private ArrayList getPriorityOrderedTourList(Tour[] toursIn) - { - - int[] tourPriorities = new int[toursIn.length]; - - int i = 0; - for (Tour tour : toursIn) - { - String purposeName = tour.getTourPurpose(); - int purposeIndex = purposeNameIndexMap.get(purposeName); - int purposePriority = TOUR_PURPOSE_PRIORITIES[purposeIndex]; - tourPriorities[i] = purposePriority; - } - - int[] sortedIndices = IndexSort.indexSort(tourPriorities); - ArrayList toursOut = new ArrayList(toursIn.length); - - for (i = 0; i < toursIn.length; i++) - toursOut.add(toursIn[sortedIndices[i]]); - - return toursOut; - } - - public void setNonMandatorySoaProbs(double[][][] soaDistProbs, double[][][] soaSizeProbs) - { - if (useNewSoaMethod) - { - dcSoaTwoStageObject.setTazDistProbs(soaDistProbs); - dcSoaTwoStageObject.setMgraSizeProbs(soaSizeProbs); - } - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); - - MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, - Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TazDataManager tazManager = TazDataManager.getInstance(propertyMap); - - BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); - if (!aggAcc.getAccessibilitiesAreBuilt()) - { - aggAcc.setupBuildAccessibilities(propertyMap, false); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - // aggAcc.buildAccessibilityComponents(propertyMap); - - boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, - CtrampApplication.READ_ACCESSIBILITIES); - if (readAccessibilities) - { - - // output data - String projectDirectory = propertyMap - .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - - } - - double[][] expConstants = aggAcc.getExpConstants(); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - - double[][][] sovExpUtilities = null; - double[][][] hovExpUtilities = null; - double[][][] nMotorExpUtilities = null; - double[][][] maasExpUtilities = null; - - NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, - hovExpUtilities, nMotorExpUtilities,maasExpUtilities); - - MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( - propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); - - HouseholdIndividualNonMandatoryTourFrequencyModel inmtfModel = new HouseholdIndividualNonMandatoryTourFrequencyModel( - propertyMap, dmuFactory, aggAcc.getAccessibilitiesTableObject(), mandAcc); - - TourModeChoiceModel inmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - "Non-Mandatory", dmuFactory, logsumHelper); - - NonMandatoryDestChoiceModel testObject = new NonMandatoryDestChoiceModel(propertyMap, - modelStructure, aggAcc, dmuFactory, inmmcModel); - - String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); - int hhServerPort = Integer.parseInt((String) propertyMap - .get("RunModel.HouseholdServerPort")); - - HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, - hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - boolean restartHhServer = false; - try - { - // possible values for the following can be none, ao, cdap, imtf, - // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, - // stf, stl - String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); - if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; - else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") - || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") - || restartModel.equalsIgnoreCase("imtf") - || restartModel.equalsIgnoreCase("imtod") - || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") - || restartModel.equalsIgnoreCase("awtod") - || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") - || restartModel.equalsIgnoreCase("jtod") - || restartModel.equalsIgnoreCase("inmtf") - || restartModel.equalsIgnoreCase("inmtl") - || restartModel.equalsIgnoreCase("inmtod") - || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) - restartHhServer = false; - } catch (MissingResourceException e) - { - restartHhServer = true; - } - - if (restartHhServer) - { - - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(0.2f, 0); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, - inputPersonFileName); - - } else - { - - householdDataManager.setHouseholdSampleRate(0.2f, 0); - householdDataManager.setDebugHhIdsFromHashmap(); - householdDataManager.setTraceHouseholdSet(); - - } - - int id = householdDataManager.getArrayIndex(1033380); - Household[] hh = householdDataManager.getHhArray(id, id); - - testObject.applyIndivModel(hh[0]); - testObject.applyJointModel(hh[0]); - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java deleted file mode 100644 index fb4cfec..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/NonMandatoryTourDepartureAndDurationTime.java +++ /dev/null @@ -1,1448 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To - * change this template use File | Settings | File Templates. - */ -public class NonMandatoryTourDepartureAndDurationTime - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(NonMandatoryTourDepartureAndDurationTime.class); - private transient Logger todLogger = Logger.getLogger("todLogger"); - private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); - - private static final String IMTOD_UEC_FILE_TARGET = "departTime.uec.file"; - private static final String IMTOD_UEC_DATA_TARGET = "departTime.data.page"; - private static final String IMTOD_UEC_ESCORT_MODEL_TARGET = "departTime.escort.page"; - private static final String IMTOD_UEC_SHOP_MODEL_TARGET = "departTime.shop.page"; - private static final String IMTOD_UEC_MAINT_MODEL_TARGET = "departTime.maint.page"; - private static final String IMTOD_UEC_EAT_MODEL_TARGET = "departTime.eat.page"; - private static final String IMTOD_UEC_VISIT_MODEL_TARGET = "departTime.visit.page"; - private static final String IMTOD_UEC_DISCR_MODEL_TARGET = "departTime.discr.page"; - - private static final String[] TOUR_PURPOSE_NAMES = { - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME, - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME, - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME, - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME, - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME }; - - private static final String[] DC_MODEL_SHEET_KEYS = { - IMTOD_UEC_ESCORT_MODEL_TARGET, IMTOD_UEC_SHOP_MODEL_TARGET, - IMTOD_UEC_MAINT_MODEL_TARGET, IMTOD_UEC_EAT_MODEL_TARGET, IMTOD_UEC_VISIT_MODEL_TARGET, - IMTOD_UEC_DISCR_MODEL_TARGET }; - - // process non-mandatory tours in order by priority purpose: - // 4=escort, 6=oth maint, 5=shop, 8=visiting, 9=oth discr, 7=eat out, - private static final int[] TOUR_PURPOSE_INDEX_ORDER = {4, 6, 5, 8, 9, 7}; - - private ArrayList[] purposeTourLists; - - private int[] todModelIndices; - private HashMap purposeNameIndexMap; - - private int[] tourDepartureTimeChoiceSample; - - // DMU for the UEC - private TourDepartureTimeAndDurationDMU todDmuObject; - private TourModeChoiceDMU mcDmuObject; - - // model structure to compare the .properties time of day with the UECs - private ModelStructure modelStructure; - - // private double[][] dcSizeArray; - - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private ChoiceModelApplication[] todModels; - private TourModeChoiceModel mcModel; - - private int[] altStarts; - private int[] altEnds; - - private boolean[] needToComputeLogsum; - private double[] modeChoiceLogsums; - - private int noAltChoice = 1; - - private long jointModeChoiceTime; - private long indivModeChoiceTime; - - public NonMandatoryTourDepartureAndDurationTime(HashMap propertyMap, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, - TourModeChoiceModel mcModel) - { - - // set the model structure - this.modelStructure = modelStructure; - this.mcModel = mcModel; - - logger.info("setting up Non-Mandatory time-of-day choice model."); - - setupTodChoiceModels(propertyMap, dmuFactory); - } - - private void setupTodChoiceModels(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - tazs = TazDataManager.getInstance(); - mgraManager = MgraDataManager.getInstance(); - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - String todUecFileName = propertyMap.get(IMTOD_UEC_FILE_TARGET); - todUecFileName = uecFileDirectory + todUecFileName; - - todDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); - - mcDmuObject = dmuFactory.getModeChoiceDMU(); - - int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; - needToComputeLogsum = new boolean[numLogsumIndices]; - modeChoiceLogsums = new double[numLogsumIndices]; - - // create the array of tod model indices - int[] uecSheetIndices = new int[TOUR_PURPOSE_NAMES.length]; - - purposeNameIndexMap = new HashMap(TOUR_PURPOSE_NAMES.length); - - int i = 0; - for (String purposeName : TOUR_PURPOSE_NAMES) - { - int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); - purposeNameIndexMap.put(purposeName, i); - uecSheetIndices[i] = uecIndex; - i++; - } - - // create a lookup array to map purpose index to model index - todModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int todModelIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, todModelIndex); - todModelIndices[todModelIndex] = todModelIndex++; - } else - { - todModelIndices[todModelIndex++] = modelIndexMap.get(uecIndex); - } - } - - todModels = new ChoiceModelApplication[modelIndexMap.size()]; - int todModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, - IMTOD_UEC_DATA_TARGET); - - for (int uecIndex : modelIndexMap.keySet()) - { - int modelIndex = modelIndexMap.get(uecIndex); - try - { - todModels[modelIndex] = new ChoiceModelApplication(todUecFileName, uecIndex, - todModelDataSheet, propertyMap, (VariableTable) todDmuObject); - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up NonMandatory TOD ChoiceModelApplication[%d] for modelIndex=%d, num choice models=%d", - modelIndex, modelIndex, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - // get the alternatives table from the work tod UEC. - TableDataSet altsTable = todModels[0].getUEC().getAlternativeData(); - - altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); - altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); - todDmuObject.setTodAlts(altStarts, altEnds); - - int numDepartureTimeChoiceAlternatives = todModels[0].getNumberOfAlternatives(); - tourDepartureTimeChoiceSample = new int[numDepartureTimeChoiceAlternatives + 1]; - Arrays.fill(tourDepartureTimeChoiceSample, 1); - - // allocate an array of ArrayList objects to hold tour lists by purpose - // - tour lists will be processed - // in priority purpose order. - int maxPurposeIndex = 0; - for (i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) - if (TOUR_PURPOSE_INDEX_ORDER[i] > maxPurposeIndex) - maxPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; - - purposeTourLists = new ArrayList[maxPurposeIndex + 1]; - for (i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) - { - int index = TOUR_PURPOSE_INDEX_ORDER[i]; - purposeTourLists[index] = new ArrayList(); - } - - } - - public void applyIndivModel(Household hh, boolean runTODChoice, boolean runModeChoice) - { - - indivModeChoiceTime = 0; - - Logger modelLogger = todLogger; - - // get the person objects for this household - Person[] persons = hh.getPersons(); - - if(!runTODChoice) { - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - // if no individual non-mandatory tours, nothing to do. - if (person.getListOfIndividualNonMandatoryTours().size() == 0) continue; - - // arrange the individual non-mandatory tours for this person in an - // array of ArrayLists by purpose - getPriorityOrderedTourList(person.getListOfIndividualNonMandatoryTours()); - - for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) { - int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; - for (Tour t : purposeTourLists[tourPurposeIndex]) - try { - runModeChoice(hh,person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); - - }catch(Exception e) { - String errorMessage = String - .format("Exception caught for HHID=%d, personNum=%d, individual non-mandatory mode choice, tour ArrayList index=%d.", - hh.getHhId(), person.getPersonNum(), tourPurposeIndex); - String decisionMakerLabel = String - .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), person.getPersonNum(), person.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, person); - logger.error(errorMessage, e); - throw new RuntimeException(e); - - } - } - } - return; - - } - - - - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - // if no individual non-mandatory tours, nothing to do. - if (person.getListOfIndividualNonMandatoryTours().size() == 0) continue; - - // arrange the individual non-mandatory tours for this person in an - // array of ArrayLists by purpose - getPriorityOrderedTourList(person.getListOfIndividualNonMandatoryTours()); - - // define variables to hold depart/arrive periods selected for the - // most recent tour. - // if a tour has no non-overlapping period available, set the - // periods to either the depart or arrive of the most recently - // determined - // if none has been selected yet, set to the first of last interval - int previouslySelectedDepartPeriod = -1; - int previouslySelectedArrivePeriod = -1; - - for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) - { - - int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; - - // process each individual non-mandatory tour from the list - int m = -1; - int tourPurpNum = 1; - int noWindowCountFirstTemp = 0; - int noWindowCountLastTemp = 0; - int noLaterAlternativeCountTemp = 0; - - for (Tour t : purposeTourLists[tourPurposeIndex]) - { - - //store the tour depart time and arrival time if it is an escort tour; that way mode choice - //logsums can be calculated for the tour and stored when the actual tour dep/arr period isn't chosen. - int escortTourDepartPeriod=0; - int escortTourArrivePeriod=0; - if(t.getEscortTypeOutbound()>0 || t.getEscortTypeInbound()>0){ - escortTourDepartPeriod = t.getTourDepartPeriod(); - escortTourArrivePeriod = t.getTourArrivePeriod(); - continue; - } - try - { - - // get the choice model for the tour purpose - String tourPurposeName = t.getTourPurpose(); - m = todModelIndices[purposeNameIndexMap.get(tourPurposeName)]; - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (hh.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Non-Mandatory Tour Departure Time Choice Model for: Purpose=%s", - tourPurposeName); - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, tourId=%d, num=%d of %d %s tours", - hh.getHhId(), person.getPersonNum(), - person.getPersonType(), t.getTourId(), tourPurpNum, - purposeTourLists[tourPurposeIndex].size(), - tourPurposeName); - - todModels[m].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "Individual Non-Mandatory Tour Departure Time Choice Model: Debug Statement for Household ID: " - + hh.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " - + person.getPersonType() - + ", Tour Id: " - + t.getTourId() - + ", num " - + tourPurpNum - + " of " - + purposeTourLists[tourPurposeIndex].size() - + " " - + tourPurposeName + " tours."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - - } - - // set the dmu object - todDmuObject.setHousehold(hh); - todDmuObject.setPerson(person); - todDmuObject.setTour(t); - - // check for multiple tours for this person, by purpose - // set the first or second switch if multiple tours for - // person, by purpose - if (purposeTourLists[tourPurposeIndex].size() == 1) - { - // not a multiple tour pattern - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(1); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else if (purposeTourLists[tourPurposeIndex].size() > 1) - { - // Two-plus tour multiple tour pattern - if (tourPurpNum == 1) - { - // first of 2+ tours - todDmuObject.setFirstTour(1); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(tourPurpNum); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else - { - // 2nd or greater tours - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(1); - todDmuObject.setTourNumber(tourPurpNum); - // the ArrayList is 0-based, and we want the - // previous tour, so subtract 2 from tourPurpNum - // to get the right index - int otherTourEndHour = purposeTourLists[tourPurposeIndex].get( - tourPurpNum - 2).getTourArrivePeriod(); - todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); - } - } - - // set the choice availability and sample - boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows( - altStarts, altEnds); - Arrays.fill(tourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in individual non-mandatory departure time choice model for hhId=%d, personNum=%d, tour purpose index=%d, tour ArrayList index=%d.", - hh.getHhId(), person.getPersonNum(), tourPurposeIndex, - tourPurpNum - 1)); - logger.error(String - .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the individual non-mandatory tour UEC=%d.", - tourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if all time windows for this person have already been - // scheduled, choose either the first and last - // alternatives and keep track of the number of times - // this condition occurs. - int alternativeAvailable = -1; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - alternativeAvailable = a; - break; - } - } - - int chosen = -1; - int chosenStartPeriod = -1; - int chosenEndPeriod = -1; - - // alternate making the first and last periods chosen if - // no alternatives are available - if (alternativeAvailable < 0) - { - - if (noAltChoice == 1) - { - if (previouslySelectedDepartPeriod < 0) - { - chosenStartPeriod = altStarts[noAltChoice - 1]; - chosenEndPeriod = altEnds[noAltChoice - 1]; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, and no non-mandatory tour scheduled yet, depart AND arrive set to first period=" - + chosenStartPeriod - + ", " - + chosenEndPeriod + "."); - } else - { - chosenStartPeriod = previouslySelectedArrivePeriod; - chosenEndPeriod = previouslySelectedArrivePeriod; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled non-mandatory tour=" - + chosenStartPeriod - + ", " - + chosenEndPeriod + "."); - } - noWindowCountFirstTemp++; - noAltChoice = departureTimeChoiceAvailability.length - 1; - } else - { - if (previouslySelectedDepartPeriod < 0) - { - chosenStartPeriod = altStarts[noAltChoice - 1]; - chosenEndPeriod = altEnds[noAltChoice - 1]; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, and no non-mandatory tour scheduled yet, depart AND arrive set to last period=" - + chosenStartPeriod - + ", " - + chosenEndPeriod + "."); - } else - { - chosenStartPeriod = previouslySelectedArrivePeriod; - chosenEndPeriod = previouslySelectedArrivePeriod; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled non-mandatory tour=" - + chosenStartPeriod - + ", " - + chosenEndPeriod + "."); - } - noWindowCountLastTemp++; - noAltChoice = 1; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to work tour arrive period=" - + chosenEndPeriod + "."); - } - - // schedule the chosen alternative - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - previouslySelectedDepartPeriod = chosenStartPeriod; - previouslySelectedArrivePeriod = chosenEndPeriod; - - if (runModeChoice) - { - - runModeChoice( hh, person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); - } - - } else - { - - // calculate and store the mode choice logsum for - // the usual work location for this worker at the - // various - // departure time and duration alternativees - setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, - departureTimeChoiceAvailability); - - if (hh.getDebugChoiceModels()) - { - hh.logTourObject(loggingHeader, modelLogger, person, t); - } - - todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); - todDmuObject - .setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); - - float logsum = (float) todModels[m].computeUtilities(todDmuObject, - todDmuObject.getIndexValues(), departureTimeChoiceAvailability, - tourDepartureTimeChoiceSample); - - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = hh.getHhRandom(); - int randomCount = hh.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - if (todModels[m].getAvailabilityCount() > 0) - { - chosen = todModels[m].getChoiceResult(rn); - - // debug output - if (hh.getDebugChoiceModels()) - { - - double[] utilities = todModels[m].getUtilities(); - double[] probabilities = todModels[m].getProbabilities(); - boolean[] availabilities = todModels[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " - + personTypeString + ", Tour Id: " + t.getTourId() - + ", Tour num: " + tourPurpNum + " of " - + purposeTourLists[tourPurposeIndex].size() + " " - + tourPurposeName + " tours."); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", - k + 1, altStarts[k], altEnds[k]); - modelLogger.info(String.format( - "%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], - probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", - chosen, altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to - // debug log file - todModels[m].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - todModels[m].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate - // model specific log file - loggingHeader = String.format("%s for %s", - choiceModelDescription, decisionMakerLabel); - todModels[m].logUECResults(modelLogger, loggingHeader); - - } - - } else - { - - // since there were no alternatives with valid - // utility, assuming previous - // tour of this type scheduled up to the last - // period, so no periods left - // for this tour. - - // TODO: do a formal check for this so we can - // still flag other reasons why there's - // no valid utility for any alternative - chosen = departureTimeChoiceAvailability.length - 1; - noLaterAlternativeCountTemp++; - - } - - // schedule the chosen alternative - chosenStartPeriod = altStarts[chosen - 1]; - chosenEndPeriod = altEnds[chosen - 1]; - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - previouslySelectedDepartPeriod = chosenStartPeriod; - previouslySelectedArrivePeriod = chosenEndPeriod; - - if (runModeChoice) - { - - runModeChoice(hh,person,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); - } - - } - - } catch (Exception e) - { - String errorMessage = String - .format("Exception caught for HHID=%d, personNum=%d, individual non-mandatory Departure time choice, tour ArrayList index=%d.", - hh.getHhId(), person.getPersonNum(), tourPurpNum - 1); - String decisionMakerLabel = String - .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), person.getPersonNum(), person.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, person); - todModels[m].logUECResults(modelLogger, errorMessage); - - logger.error(errorMessage, e); - throw new RuntimeException(e); - } - - tourPurpNum++; - - } - - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String - .format("Final Individual Non-Mandatory Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), person.getPersonNum(), person.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - } - - } - - hh.setInmtodRandomCount(hh.getHhRandomCount()); - - } - - - public void runModeChoice(Household hh, Person person, Tour t, int chosenStartPeriod, int chosenEndPeriod) { - if (hh.getDebugChoiceModels()) - hh.logHouseholdObject( - "Pre Non-Mandatory Tour Mode Choice Household " - + hh.getHhId() - + ", Tour " - + t.getTourId() - + " of " - + person.getListOfIndividualNonMandatoryTours() - .size(), tourMCNonManLogger); - - // set the mode choice attributes needed by - // @variables in the UEC spreadsheets - setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, - chosenEndPeriod); - - // use the mcModel object already setup for - // computing logsums and get - // the mode choice, where the selected - // worklocation and subzone an departure time - // and duration are set - // for this work tour. - int chosenMode = mcModel.getModeChoice(mcDmuObject, - t.getTourPrimaryPurpose()); - t.setTourModeChoice(chosenMode); - - } - public void applyJointModel(Household hh,boolean runTODChoice, boolean runModeChoice) - { - - jointModeChoiceTime = 0; - - // if no joint non-mandatory tours, nothing to do for this household. - Tour[] jointTours = hh.getJointTourArray(); - if (jointTours == null || jointTours.length == 0) return; - - Logger modelLogger = todLogger; - - // arrange the joint non-mandatory tours for this househol in an array - // of ArrayLists by purpose - getPriorityOrderedTourList(jointTours); - - // process tour lists by priority purpose - if(!runTODChoice) { - for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) - { - - int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; - for (Tour t : purposeTourLists[tourPurposeIndex]) - runModeChoice(hh,t,t.getTourDepartPeriod(),t.getTourArrivePeriod()); - } - return; - } - - - // define variables to hold depart/arrive periods selected for the most - // recent tour. - // if a tour has no non-overlapping period available, set the periods to - // either the depart or arrive of the most recently determined - // if none has been selected yet, set to the first of last interval - int previouslySelectedDepartPeriod = -1; - int previouslySelectedArrivePeriod = -1; - - for (int i = 0; i < TOUR_PURPOSE_INDEX_ORDER.length; i++) - { - - int tourPurposeIndex = TOUR_PURPOSE_INDEX_ORDER[i]; - - // process each individual non-mandatory tour from the list - int m = -1; - int tourPurpNum = 1; - int noWindowCountFirstTemp = 0; - int noWindowCountLastTemp = 0; - int noLaterAlternativeCountTemp = 0; - for (Tour t : purposeTourLists[tourPurposeIndex]) - { - - try - { - - // get the choice model for the tour purpose - String tourPurposeName = t.getTourPurpose(); - m = todModelIndices[purposeNameIndexMap.get(tourPurposeName)]; - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (hh.getDebugChoiceModels()) - { - - String personNumsInJointTour = "Person Nums: ["; - for (int n : t.getPersonNumArray()) - personNumsInJointTour += " " + n; - personNumsInJointTour += " ]"; - - choiceModelDescription = String - .format("Joint Non-Mandatory Tour Departure Time Choice Model for: Purpose=%s", - tourPurposeName); - decisionMakerLabel = String.format( - "HH=%d, tourId=%d, %s, num=%d of %d %s tours", hh.getHhId(), - t.getTourId(), personNumsInJointTour, tourPurpNum, - purposeTourLists[tourPurposeIndex].size(), tourPurposeName); - - todModels[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - // set the dmu object - todDmuObject.setHousehold(hh); - todDmuObject.setTour(t); - - // check for multiple tours for this person, by purpose - // set the first or second switch if multiple tours for - // person, by purpose - if (purposeTourLists[tourPurposeIndex].size() == 1) - { - // not a multiple tour pattern - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(1); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else if (purposeTourLists[tourPurposeIndex].size() > 1) - { - // Two-plus tour multiple tour pattern - if (tourPurpNum == 1) - { - // first of 2+ tours - todDmuObject.setFirstTour(1); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(tourPurpNum); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else - { - // 2nd or greater tours - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(1); - todDmuObject.setTourNumber(tourPurpNum); - // the ArrayList is 0-based, and we want the - // previous tour, so subtract 2 from tourPurpNum to - // get the right index - int otherTourEndHour = purposeTourLists[tourPurposeIndex].get( - tourPurpNum - 2).getTourArrivePeriod(); - todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); - } - } - - // set the choice availability and sample - boolean[] departureTimeChoiceAvailability = hh - .getAvailableJointTourTimeWindows(t, altStarts, altEnds); - Arrays.fill(tourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in joint non-mandatory departure time choice model for hhId=%d, tour purpose index=%d, tour ArrayList index=%d.", - hh.getHhId(), tourPurposeIndex, tourPurpNum - 1)); - logger.error(String - .format("length of the availability array determined by the number of alternatives set in the person schedules=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the joint non-mandatory tour UEC=%d.", - tourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if all time windows for this person have already been - // scheduled, choose either the first and last - // alternatives and keep track of the number of times this - // condition occurs. - int alternativeAvailable = -1; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - alternativeAvailable = a; - break; - } - } - - int chosen = -1; - int chosenStartPeriod = -1; - int chosenEndPeriod = -1; - - // alternate making the first and last periods chosen if no - // alternatives are available - if (alternativeAvailable < 0) - { - - if (noAltChoice == 1) - { - if (previouslySelectedDepartPeriod < 0) - { - chosenStartPeriod = altStarts[noAltChoice - 1]; - chosenEndPeriod = altEnds[noAltChoice - 1]; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, and no joint non-mandatory tour scheduled yet, depart AND arrive set to first period=" - + chosenStartPeriod - + ", " - + chosenEndPeriod - + "."); - } else - { - chosenStartPeriod = previouslySelectedArrivePeriod; - chosenEndPeriod = previouslySelectedArrivePeriod; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled joint non-mandatory tour=" - + chosenStartPeriod - + ", " - + chosenEndPeriod - + "."); - } - noWindowCountFirstTemp++; - noAltChoice = departureTimeChoiceAvailability.length - 1; - } else - { - if (previouslySelectedDepartPeriod < 0) - { - chosenStartPeriod = altStarts[noAltChoice - 1]; - chosenEndPeriod = altEnds[noAltChoice - 1]; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, and no joint non-mandatory tour scheduled yet, depart AND arrive set to last period=" - + chosenStartPeriod - + ", " - + chosenEndPeriod - + "."); - } else - { - chosenStartPeriod = previouslySelectedArrivePeriod; - chosenEndPeriod = previouslySelectedArrivePeriod; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to arrive period of most recent scheduled joint non-mandatory tour=" - + chosenStartPeriod - + ", " - + chosenEndPeriod - + "."); - } - noWindowCountLastTemp++; - noAltChoice = 1; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to work tour arrive period=" - + chosenEndPeriod + "."); - } - - // schedule the chosen alternative - hh.scheduleJointTourTimeWindows(t, chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - previouslySelectedDepartPeriod = chosenStartPeriod; - previouslySelectedArrivePeriod = chosenEndPeriod; - - if (runModeChoice) - { - runModeChoice(hh,t,chosenStartPeriod,chosenEndPeriod); - - } - - } else - { - - // calculate and store the mode choice logsum for the - // usual work location for this worker at the various - // departure time and duration alternativees - setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, - departureTimeChoiceAvailability); - - if (hh.getDebugChoiceModels()) - { - for (int p = 1; p < hh.getPersons().length; p++) - { - Person pers = hh.getPersons()[p]; - hh.logTourObject(loggingHeader, modelLogger, pers, t); - } - } - - todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); - todDmuObject.setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); - - float logsum = (float) todModels[m].computeUtilities(todDmuObject, todDmuObject.getIndexValues(), - departureTimeChoiceAvailability, tourDepartureTimeChoiceSample); - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = hh.getHhRandom(); - int randomCount = hh.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - if (todModels[m].getAvailabilityCount() > 0) - { - chosen = todModels[m].getChoiceResult(rn); - - // debug output - if (hh.getDebugChoiceModels()) - { - - double[] utilities = todModels[m].getUtilities(); - double[] probabilities = todModels[m].getProbabilities(); - boolean[] availabilities = todModels[m].getAvailabilities(); - - modelLogger.info("Tour Id: " + t.getTourId() + ", Tour num: " - + tourPurpNum + " of " - + purposeTourLists[tourPurposeIndex].size() + " " - + tourPurposeName + " tours."); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", - k + 1, altStarts[k], altEnds[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", - altString, availabilities[k + 1], utilities[k], - probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, - altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug - // log file - todModels[m].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - todModels[m].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate - // model specific log file - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - todModels[m].logUECResults(modelLogger, loggingHeader); - - } - - } else - { - - // since there were no alternatives with valid - // utility, assuming previous - // tour of this type scheduled up to the last - // period, so no periods left - // for this tour. - - // TODO: do a formal check for this so we can still - // flag other reasons why there's - // no valid utility for any alternative - chosen = departureTimeChoiceAvailability.length - 1; - noLaterAlternativeCountTemp++; - - } - - // schedule the chosen alternative - chosenStartPeriod = altStarts[chosen - 1]; - chosenEndPeriod = altEnds[chosen - 1]; - hh.scheduleJointTourTimeWindows(t, chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - previouslySelectedDepartPeriod = chosenStartPeriod; - previouslySelectedArrivePeriod = chosenEndPeriod; - - if (runModeChoice) - { - - runModeChoice(hh,t,chosenStartPeriod,chosenEndPeriod); - - } - - } - - } catch (Exception e) - { - String errorMessage = String - .format("Exception caught for HHID=%d, joint non-mandatory Departure time choice, tour ArrayList index=%d.", - hh.getHhId(), tourPurpNum - 1); - String decisionMakerLabel = "Final Joint Non-Mandatory Departure Time Person Objects:"; - for (int p = 1; p < hh.getPersons().length; p++) - { - Person pers = hh.getPersons()[p]; - hh.logPersonObject(decisionMakerLabel, modelLogger, pers); - todModels[m].logUECResults(modelLogger, errorMessage); - } - - logger.error(errorMessage, e); - throw new RuntimeException(); - } - - tourPurpNum++; - - } - - if (hh.getDebugChoiceModels()) - { - for (int p = 1; p < hh.getPersons().length; p++) - { - Person pers = hh.getPersons()[p]; - String decisionMakerLabel = String - .format("Final Joint Non-Mandatory Departure Time Person Objects: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), pers.getPersonNum(), pers.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, pers); - } - } - - } - - hh.setJtodRandomCount(hh.getHhRandomCount()); - - } - - /** - * For joint tours - * @param hh - * @param t - * @param chosenStartPeriod - * @param chosenEndPeriod - */ - private void runModeChoice(Household hh, Tour t, int chosenStartPeriod, int chosenEndPeriod) { - - long check = System.nanoTime(); - - if (hh.getDebugChoiceModels()) - hh.logHouseholdObject( - "Pre Joint Non-Mandatory Tour Mode Choice Household " - + hh.getHhId() + ", Tour " + t.getTourId()+1 + " of " - + hh.getJointTourArray().length, - tourMCNonManLogger); - - // set the mode choice attributes needed by - // @variables in the UEC spreadsheets - setModeChoiceDmuAttributes(hh, null, t, chosenStartPeriod, - chosenEndPeriod); - - // use the mcModel object already setup for - // computing logsums and get - // the mode choice, where the selected - // worklocation and subzone an departure time and - // duration are set - // for this work tour. - int chosenMode = mcModel.getModeChoice(mcDmuObject, - t.getTourPrimaryPurpose()); - t.setTourModeChoice(chosenMode); - jointModeChoiceTime += (System.nanoTime() - check); - - } - - - - - - private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, - int startPeriod, int endPeriod) - { - - t.setTourDepartPeriod(startPeriod); - t.setTourArrivePeriod(endPeriod); - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(t); - mcDmuObject.setDmuIndexValues(household.getHhId(), t.getTourOrigMgra(), - t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); - - - - mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t - .getTourOrigMgra()))); - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t - .getTourDestMgra()))); - - mcDmuObject.setOriginMgra(t.getTourOrigMgra()); - mcDmuObject.setDestMgra(t.getTourDestMgra()); - - } - - private void setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Tour tour, - boolean[] altAvailable) - { - - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todLogger; - String choiceModelDescription = String.format( - "NonMandatory Tour Mode Choice Logsum calculation for %s Departure Time Choice", - tour.getTourPurpose()); - String decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person - .getPersonNum(), person.getPersonType(), tour.getTourId(), person - .getListOfWorkTours().size()); - String loggingHeader = String - .format("%s %s", choiceModelDescription, decisionMakerLabel); - - for (int a = 1; a <= altStarts.length; a++) - { - - // if the depart/arrive alternative is unavailable, no need to check - // to see if a logsum has been calculated - if (!altAvailable[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " work tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - throw new RuntimeException(e); - } - needToComputeLogsum[index] = false; - } - - } - - todDmuObject.setModeChoiceLogsums(modeChoiceLogsums); - - mcDmuObject.getTourObject().setTourDepartPeriod(0); - mcDmuObject.getTourObject().setTourArrivePeriod(0); - } - - /** - * takes an ArrayList of individual non-mandatory tours creates an array of - * ArrayLists of tours by purpose - */ - private void getPriorityOrderedTourList(ArrayList toursIn) - { - - // clear the ArrayLists - for (int i = 0; i < purposeTourLists.length; i++) - { - if (purposeTourLists[i] != null) purposeTourLists[i].clear(); - } - - // go through the list of non-mandatory tours, and put each into an - // array of ArrayLists, by purpose. - for (Tour tour : toursIn) - { - int purposeIndex = tour.getTourPrimaryPurposeIndex(); - purposeTourLists[purposeIndex].add(tour); - } - - } - - /** - * takes an array of joint non-mandatory tours creates an array of - * ArrayLists of tours by purpose - */ - private void getPriorityOrderedTourList(Tour[] toursIn) - { - - // clear the ArrayLists - for (int i = 0; i < purposeTourLists.length; i++) - { - if (purposeTourLists[i] != null) purposeTourLists[i].clear(); - } - - // go through the list of non-mandatory tours, and put each into an - // array of ArrayLists, by purpose. - for (Tour tour : toursIn) - { - int purposeIndex = tour.getTourPrimaryPurposeIndex(); - purposeTourLists[purposeIndex].add(tour); - } - - } - - public long getJointModeChoiceTime() - { - return jointModeChoiceTime; - } - - public long getIndivModeChoiceTime() - { - return indivModeChoiceTime; - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); - - MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, - Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TazDataManager tazManager = TazDataManager.getInstance(propertyMap); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - - String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); - int hhServerPort = Integer.parseInt((String) propertyMap - .get("RunModel.HouseholdServerPort")); - - HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, - hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - boolean restartHhServer = false; - try - { - // possible values for the following can be none, ao, cdap, imtf, - // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, - // stf, stl - String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); - if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; - else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") - || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") - || restartModel.equalsIgnoreCase("imtf") - || restartModel.equalsIgnoreCase("imtod") - || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") - || restartModel.equalsIgnoreCase("awtod") - || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") - || restartModel.equalsIgnoreCase("jtod") - || restartModel.equalsIgnoreCase("inmtf") - || restartModel.equalsIgnoreCase("inmtl") - || restartModel.equalsIgnoreCase("inmtod") - || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) - restartHhServer = false; - } catch (MissingResourceException e) - { - restartHhServer = true; - } - - if (restartHhServer) - { - - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(1.0f, 0); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, - inputPersonFileName); - - } else - { - - householdDataManager.setHouseholdSampleRate(1.0f, 0); - householdDataManager.setDebugHhIdsFromHashmap(); - householdDataManager.setTraceHouseholdSet(); - - } - - // int id = householdDataManager.getArrayIndex( 1033380 ); - // int id = householdDataManager.getArrayIndex( 1033331 ); - int id = householdDataManager.getArrayIndex(423804); - Household[] hh = householdDataManager.getHhArray(id, id); - - TourModeChoiceModel inmmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, dmuFactory, logsumHelper); - - NonMandatoryTourDepartureAndDurationTime testObject = new NonMandatoryTourDepartureAndDurationTime( - propertyMap, modelStructure, dmuFactory, inmmcModel); - - testObject.applyIndivModel(hh[0], true, true); - testObject.applyJointModel(hh[0], true, true); - - /** - * used this block of code to test for typos and implemented dmu - * methiods in the TOD choice UECs - * - * String uecFileDirectory = propertyMap.get( - * CtrampApplication.PROPERTIES_UEC_PATH ); String todUecFileName = - * propertyMap.get( IMTOD_UEC_FILE_TARGET ); todUecFileName = - * uecFileDirectory + todUecFileName; - * - * ModelStructure modelStructure = new SandagModelStructure(); - * SandagCtrampDmuFactory dmuFactory = new - * SandagCtrampDmuFactory(modelStructure); - * TourDepartureTimeAndDurationDMU todDmuObject = - * dmuFactory.getTourDepartureTimeAndDurationDMU(); - * - * int[] indices = { 0, 1, 2, 3, 4, 5 }; for (int i : indices ) { int - * uecIndex = i + 4; File uecFile = new File(todUecFileName); - * UtilityExpressionCalculator uec = new - * UtilityExpressionCalculator(uecFile, uecIndex, 0, propertyMap, - * (VariableTable) todDmuObject); } - */ - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java deleted file mode 100644 index 8cbfa08..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingChoiceDMU.java +++ /dev/null @@ -1,332 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * @author crf
    - * Started: Apr 14, 2009 1:34:03 PM - */ -public class ParkingChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(ParkingChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - - private int personType; - private int activityIntervals; - private int destPurpose; - private double reimbPct; - - private double[] distancesOrigAlt; - private double[] distancesAltDest; - - private double[] altParkingCostsM; - private int[] altMstallsoth; - private int[] altMstallssam; - private float[] altMparkcost; - private int[] altDstallsoth; - private int[] altDstallssam; - private float[] altDparkcost; - private int[] altHstallsoth; - private int[] altHstallssam; - private float[] altHparkcost; - private int[] altNumfreehrs; - - private int[] parkAreaMgras; - private int[] altMgraIndices; - - public ParkingChoiceDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int hhId, int origMgra, int destMgra, boolean hhDebug) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setOriginZone(origMgra); - dmuIndex.setDestZone(destMgra); - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - - if (hhDebug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug Parking Choice UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public void setPersonType(int value) - { - personType = value; - } - - public void setActivityIntervals(int value) - { - activityIntervals = value; - } - - public void setDestPurpose(int value) - { - destPurpose = value; - } - - public void setReimbPct(double value) - { - reimbPct = value; - } - - /** - * @param mgras - * is the array of MGRAs in parking area from - * "plc.alts.corresp.file". This is a 0-based array - */ - public void setParkAreaMgraArray(int[] mgras) - { - parkAreaMgras = mgras; - } - - /** - * @param indices - * is an array of indices representing this person's park - * location choice sample. the index value in this array points - * to the parkAreaMgras element, and the corresponding mgra - * value. This is a 0-based array - */ - public void setSampleIndicesArray(int[] indices) - { - altMgraIndices = indices; - } - - public void setDistancesOrigAlt(double[] distances) - { - distancesOrigAlt = distances; - } - - public void setDistancesAltDest(double[] distances) - { - distancesAltDest = distances; - } - - public void setParkingCostsM(double[] values) - { - altParkingCostsM = values; - } - - public void setMstallsoth(int[] values) - { - altMstallsoth = values; - } - - public void setMstallssam(int[] values) - { - altMstallssam = values; - } - - public void setMparkCost(float[] values) - { - altMparkcost = values; - } - - public void setDstallsoth(int[] values) - { - altDstallsoth = values; - } - - public void setDstallssam(int[] values) - { - altDstallssam = values; - } - - public void setDparkCost(float[] values) - { - altDparkcost = values; - } - - public void setHstallsoth(int[] values) - { - altHstallsoth = values; - } - - public void setHstallssam(int[] values) - { - altHstallssam = values; - } - - public void setHparkCost(float[] values) - { - altHparkcost = values; - } - - public void setNumfreehrs(int[] values) - { - altNumfreehrs = values; - } - - public int getPersonType() - { - return personType; - } - - public int getActivityIntervals() - { - return activityIntervals; - } - - public int getTripDestPurpose() - { - return destPurpose; - } - - public double getReimbPct() - { - return reimbPct; - } - - /** - * @param alt - * is the index value in the alternatives array (0,...,num alts). - * @return the distance between the trip origin mgra and the alternative - * park mgra. - */ - public double getDistanceTripOrigToParkAlt(int alt) - { - return distancesOrigAlt[alt]; - } - - /** - * @param alt - * is the index value in the alternatives array (0,...,num alts). - * @return the distance between the alternative park mgra and the trip - * destination mgra. - */ - public double getDistanceTripDestFromParkAlt(int alt) - { - return distancesAltDest[alt]; - } - - /** - * @param alt - * is the index value in the alternatives array (0,...,num alts). - * @return the cost for this person to park at the alternative park mgra. - */ - public double getLsWgtAvgCostM(int alt) - { - return altParkingCostsM[alt]; - } - - public int getMstallsoth(int alt) - { - return altMstallsoth[alt]; - } - - public int getMstallssam(int alt) - { - return altMstallssam[alt]; - } - - public float getMparkcost(int alt) - { - return altMparkcost[alt]; - } - - public int getDstallsoth(int alt) - { - return altDstallsoth[alt]; - } - - public int getDstallssam(int alt) - { - return altDstallssam[alt]; - } - - public float getDparkcost(int alt) - { - return altDparkcost[alt]; - } - - public int getHstallsoth(int alt) - { - return altHstallsoth[alt]; - } - - public int getHstallssam(int alt) - { - return altHstallssam[alt]; - } - - public float getHparkcost(int alt) - { - return altHparkcost[alt]; - } - - public int getNumfreehrs(int alt) - { - return altNumfreehrs[alt]; - } - - /** - * @return 1 if the altMgra attribute that was set equals the trip - * destination - */ - public int getDestSameAsParkAlt(int alt) - { - int index = altMgraIndices[alt]; - int altMgra = parkAreaMgras[index]; - return altMgra == dmuIndex.getDestZone() ? 1 : 0; - } - - /** - * @return the altMgra attribute for this alternative - */ - public int getParkMgraAlt(int alt) - { - int index = altMgraIndices[alt]; - int altMgra = parkAreaMgras[index]; - return altMgra; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java deleted file mode 100644 index cde7ed7..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionChoiceDMU.java +++ /dev/null @@ -1,256 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * @author crf
    - * Started: Apr 14, 2009 11:09:58 AM - */ -public class ParkingProvisionChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(ParkingProvisionChoiceDMU.class); - - protected HashMap methodIndexMap; - - private Household hh; - private Person person; - private IndexValues dmuIndex; - - private int mgraParkArea; - private int numFreeHours; - private int mstallsoth; - private int mstallssam; - private float mparkcost; - private int dstallsoth; - private int dstallssam; - private float dparkcost; - private int hstallsoth; - private int hstallssam; - private float hparkcost; - - private double lsWgtAvgCostM; - private double lsWgtAvgCostD; - private double lsWgtAvgCostH; - - public ParkingProvisionChoiceDMU() - { - dmuIndex = new IndexValues(); - } - - /** need to set hh and home taz before using **/ - public void setPersonObject(Person person) - { - hh = person.getHouseholdObject(); - this.person = person; - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug Free Parking UEC"); - } - } - - public void setMgraParkArea(int value) - { - mgraParkArea = value; - } - - public void setNumFreeHours(int value) - { - numFreeHours = value; - } - - public void setLsWgtAvgCostM(double cost) - { - lsWgtAvgCostM = cost; - } - - public void setLsWgtAvgCostD(double cost) - { - lsWgtAvgCostD = cost; - } - - public void setLsWgtAvgCostH(double cost) - { - lsWgtAvgCostH = cost; - } - - public void setMStallsOth(int value) - { - mstallsoth = value; - } - - public void setMStallsSam(int value) - { - mstallssam = value; - } - - public void setMParkCost(float value) - { - mparkcost = value; - } - - public void setDStallsOth(int value) - { - dstallsoth = value; - } - - public void setDStallsSam(int value) - { - dstallssam = value; - } - - public void setDParkCost(float value) - { - dparkcost = value; - } - - public void setHStallsOth(int value) - { - hstallsoth = value; - } - - public void setHStallsSam(int value) - { - hstallssam = value; - } - - public void setHParkCost(float value) - { - hparkcost = value; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /* dmu @ functions */ - - public int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - public double getLsWgtAvgCostM() - { - return lsWgtAvgCostM; - } - - public double getLsWgtAvgCostD() - { - return lsWgtAvgCostD; - } - - public double getLsWgtAvgCostH() - { - return lsWgtAvgCostH; - } - - public int getMgraParkArea() - { - return mgraParkArea; - } - - public int getNumFreeHours() - { - return numFreeHours; - } - - public int getMStallsOth() - { - return mstallsoth; - } - - public int getMStallsSam() - { - return mstallssam; - } - - public float getMParkCost() - { - return mparkcost; - } - - public int getDStallsOth() - { - return dstallsoth; - } - - public int getDStallsSam() - { - return dstallssam; - } - - public float getDParkCost() - { - return dparkcost; - } - - public int getHStallsOth() - { - return hstallsoth; - } - - public int getHStallsSam() - { - return hstallssam; - } - - public float getHParkCost() - { - return hparkcost; - } - - public int getWorkLocMgra() - { - return person.getWorkLocation(); - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java deleted file mode 100644 index e283792..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/ParkingProvisionModel.java +++ /dev/null @@ -1,216 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class ParkingProvisionModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("fp"); - - private static final String FP_CONTROL_FILE_TARGET = "fp.uec.file"; - private static final String FP_DATA_SHEET_TARGET = "fp.data.page"; - private static final String FP_MODEL_SHEET_TARGET = "fp.model.page"; - - public static final int FP_MODEL_NO_REIMBURSEMENT_CHOICE = -1; - public static final int FP_MODEL_FREE_ALT = 1; - public static final int FP_MODEL_PAY_ALT = 2; - public static final int FP_MODEL_REIMB_ALT = 3; - - private static final String REIMBURSEMENT_MEAN = "park.cost.reimb.mean"; - private static final String REIMBURSEMENT_STD_DEV = "park.cost.reimb.std.dev"; - - private MgraDataManager mgraManager; - - private double meanReimb; - private double stdDevReimb; - - private int[] mgraParkArea; - private int[] numfreehrs; - private int[] hstallsoth; - private int[] hstallssam; - private float[] hparkcost; - private int[] dstallsoth; - private int[] dstallssam; - private float[] dparkcost; - private int[] mstallsoth; - private int[] mstallssam; - private float[] mparkcost; - - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - - private ChoiceModelApplication fpModel; - private ParkingProvisionChoiceDMU fpDmuObject; - - public ParkingProvisionModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - mgraManager = MgraDataManager.getInstance(propertyMap); - setupFreeParkingChoiceModelApplication(propertyMap, dmuFactory); - } - - private void setupFreeParkingChoiceModelApplication(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - logger.info("setting up free parking choice model."); - - // locate the free parking UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String fpUecFile = uecFileDirectory + propertyMap.get(FP_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, FP_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, FP_MODEL_SHEET_TARGET); - - // create the auto ownership choice model DMU object. - fpDmuObject = dmuFactory.getFreeParkingChoiceDMU(); - - // create the auto ownership choice model object - fpModel = new ChoiceModelApplication(fpUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) fpDmuObject); - - meanReimb = Float.parseFloat(propertyMap.get(REIMBURSEMENT_MEAN)); - stdDevReimb = Float.parseFloat(propertyMap.get(REIMBURSEMENT_STD_DEV)); - - mgraParkArea = mgraManager.getMgraParkAreas(); - numfreehrs = mgraManager.getNumFreeHours(); - lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); - lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); - lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); - mstallsoth = mgraManager.getMStallsOth(); - mstallssam = mgraManager.getMStallsSam(); - mparkcost = mgraManager.getMParkCost(); - dstallsoth = mgraManager.getDStallsOth(); - dstallssam = mgraManager.getDStallsSam(); - dparkcost = mgraManager.getDParkCost(); - hstallsoth = mgraManager.getHStallsOth(); - hstallssam = mgraManager.getHStallsSam(); - hparkcost = mgraManager.getHParkCost(); - - } - - public void applyModel(Household hhObject) - { - - Random hhRandom = hhObject.getHhRandom(); - - // person array is 1-based - Person[] person = hhObject.getPersons(); - for (int i = 1; i < person.length; i++) - { - - int workLoc = person[i].getWorkLocation(); - if (workLoc == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - - person[i].setFreeParkingAvailableResult(FP_MODEL_NO_REIMBURSEMENT_CHOICE); - person[i].setParkingReimbursement(FP_MODEL_NO_REIMBURSEMENT_CHOICE); - - } else if (workLoc > 0 && mgraParkArea[workLoc] == MgraDataManager.PARK_AREA_ONE) - { - - double randomNumber = hhRandom.nextDouble(); - int chosen = getParkingChoice(person[i], randomNumber); - person[i].setFreeParkingAvailableResult(chosen); - - if (chosen == FP_MODEL_REIMB_ALT) - { - double logReimbPct = meanReimb + hhRandom.nextGaussian() * stdDevReimb; - person[i].setParkingReimbursement(Math.exp(logReimbPct)); - } else if (chosen == FP_MODEL_FREE_ALT) - { - person[i].setParkingReimbursement(0.0); - } else if (chosen == FP_MODEL_PAY_ALT) - { - person[i].setParkingReimbursement(0.0); - } - - } else - { - - person[i].setFreeParkingAvailableResult(FP_MODEL_NO_REIMBURSEMENT_CHOICE); - person[i].setParkingReimbursement(0.0); - - } - } - - hhObject.setFpRandomCount(hhObject.getHhRandomCount()); - } - - private int getParkingChoice(Person personObj, double randomNumber) - { - - // get the corresponding household object - Household hhObj = personObj.getHouseholdObject(); - fpDmuObject.setPersonObject(personObj); - - fpDmuObject.setMgraParkArea(mgraParkArea[personObj.getWorkLocation()]); - fpDmuObject.setNumFreeHours(numfreehrs[personObj.getWorkLocation()]); - fpDmuObject.setLsWgtAvgCostM(lsWgtAvgCostM[personObj.getWorkLocation()]); - fpDmuObject.setLsWgtAvgCostD(lsWgtAvgCostD[personObj.getWorkLocation()]); - fpDmuObject.setLsWgtAvgCostH(lsWgtAvgCostH[personObj.getWorkLocation()]); - fpDmuObject.setMStallsOth(mstallsoth[personObj.getWorkLocation()]); - fpDmuObject.setMStallsSam(mstallssam[personObj.getWorkLocation()]); - fpDmuObject.setMParkCost(mparkcost[personObj.getWorkLocation()]); - fpDmuObject.setDStallsSam(dstallssam[personObj.getWorkLocation()]); - fpDmuObject.setDStallsOth(dstallsoth[personObj.getWorkLocation()]); - fpDmuObject.setDParkCost(dparkcost[personObj.getWorkLocation()]); - fpDmuObject.setHStallsOth(hstallsoth[personObj.getWorkLocation()]); - fpDmuObject.setHStallsSam(hstallssam[personObj.getWorkLocation()]); - fpDmuObject.setHParkCost(hparkcost[personObj.getWorkLocation()]); - - // set the zone and dest attributes to the person's work location - fpDmuObject.setDmuIndexValues(hhObj.getHhId(), personObj.getWorkLocation(), - hhObj.getHhTaz(), personObj.getWorkLocation()); - - // compute utilities and choose auto ownership alternative. - float logsum = (float) fpModel.computeUtilities(fpDmuObject, fpDmuObject.getDmuIndexValues()); - personObj.setParkingProvisionLogsum(logsum); - - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - if (fpModel.getAvailabilityCount() > 0) - { - try { - chosenAlt = fpModel.getChoiceResult(randomNumber); - }catch(Exception e) { - - logger.fatal("Error trying to get parking location for HHID="+hhObj.getHhId()+", PERSID="+ - personObj.getPersonId() +" Destination MGRA="+personObj.getWorkLocation()); - throw new RuntimeException(e); - } - } else - { - String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), - personObj.getPersonId()); - String errorMessage = String - .format("Exception caught for %s, no available free parking alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - fpModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - // write choice model alternative info to log file - if (hhObj.getDebugChoiceModels()) - { - String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), - personObj.getPersonId()); - fpModel.logAlternativesInfo("Free parking Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d with rn %.8f", - "Free parking Choice", decisionMaker, chosenAlt, randomNumber)); - fpModel.logUECResults(logger, decisionMaker); - } - - return chosenAlt; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java deleted file mode 100644 index 9ce7486..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Person.java +++ /dev/null @@ -1,2032 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.util.ArrayList; -import org.apache.log4j.Logger; - -public class Person - implements java.io.Serializable -{ - - // 8 am default departure period - public static final int DEFAULT_MANDATORY_START_PERIOD = 8; - // 10 am default departure period - public static final int DEFAULT_NON_MANDATORY_START_PERIOD = 12; - // 12 pm default departure period - public static final int DEFAULT_AT_WORK_SUBTOUR_START_PERIOD = 16; - // 5 pm default arrival period - public static final int DEFAULT_MANDATORY_END_PERIOD = 26; - // 3 pm default arrival period - public static final int DEFAULT_NON_MANDATORY_END_PERIOD = 22; - // 2 pm default arrival period - public static final int DEFAULT_AT_WORK_SUBTOUR_END_PERIOD = 20; - - public static final int MIN_ADULT_AGE = 19; - public static final int MIN_STUDENT_AGE = 5; - - - // person type strings used for data summaries - public static final String PERSON_TYPE_FULL_TIME_WORKER_NAME = "Full-time worker"; - public static final String PERSON_TYPE_PART_TIME_WORKER_NAME = "Part-time worker"; - public static final String PERSON_TYPE_UNIVERSITY_STUDENT_NAME = "University student"; - public static final String PERSON_TYPE_NON_WORKER_NAME = "Non-worker"; - public static final String PERSON_TYPE_RETIRED_NAME = "Retired"; - public static final String PERSON_TYPE_STUDENT_DRIVING_NAME = "Student of driving age"; - public static final String PERSON_TYPE_STUDENT_NON_DRIVING_NAME = "Student of non-driving age"; - public static final String PERSON_TYPE_PRE_SCHOOL_CHILD_NAME = "Child too young for school"; - public static final int PERSON_TYPE_FULL_TIME_WORKER_INDEX = 1; - public static final int PERSON_TYPE_PART_TIME_WORKER_INDEX = 2; - public static final int PERSON_TYPE_UNIVERSITY_STUDENT_INDEX = 3; - public static final int PERSON_TYPE_NON_WORKER_INDEX = 4; - public static final int PERSON_TYPE_RETIRED_INDEX = 5; - public static final int PERSON_TYPE_STUDENT_DRIVING_INDEX = 6; - public static final int PERSON_TYPE_STUDENT_NON_DRIVING_INDEX = 7; - public static final int PERSON_TYPE_PRE_SCHOOL_CHILD_INDEX = 8; - public static final int MALE_INDEX = 1; - public static final int FEMALE_INDEX = 2; - public static final String[] PERSON_TYPE_NAME_ARRAY = { - PERSON_TYPE_FULL_TIME_WORKER_NAME, PERSON_TYPE_PART_TIME_WORKER_NAME, - PERSON_TYPE_UNIVERSITY_STUDENT_NAME, PERSON_TYPE_NON_WORKER_NAME, - PERSON_TYPE_RETIRED_NAME, PERSON_TYPE_STUDENT_DRIVING_NAME, - PERSON_TYPE_STUDENT_NON_DRIVING_NAME, PERSON_TYPE_PRE_SCHOOL_CHILD_NAME}; - - // Employment category (1-employed FT, 2-employed PT, 3-not employed, - // 4-under age - // 16) - // Student category (1 - student in grade or high school; 2 - student in - // college - // or higher; 3 - not a student) - - public static final String EMPLOYMENT_CATEGORY_FULL_TIME_WORKER_NAME = "Full-time worker"; - public static final String EMPLOYMENT_CATEGORY_PART_TIME_WORKER_NAME = "Part-time worker"; - public static final String EMPLOYMENT_CATEGORY_NOT_EMPLOYED_NAME = "Not employed"; - public static final String EMPLOYMENT_CATEGORY_UNDER_AGE_16_NAME = "Under age 16"; - - public static final String[] EMPLOYMENT_CATEGORY_NAME_ARRAY = { - EMPLOYMENT_CATEGORY_FULL_TIME_WORKER_NAME, EMPLOYMENT_CATEGORY_PART_TIME_WORKER_NAME, - EMPLOYMENT_CATEGORY_NOT_EMPLOYED_NAME, EMPLOYMENT_CATEGORY_UNDER_AGE_16_NAME}; - - public static final String STUDENT_CATEGORY_GRADE_OR_HIGH_SCHOOL_NAME = "Grade or high school"; - public static final String STUDENT_CATEGORY_COLLEGE_OR_HIGHER_NAME = "College or higher"; - public static final String STUDENT_CATEGORY_NOT_STUDENT_NAME = "Not student"; - - public static final String[] STUDENT_CATEGORY_NAME_ARRAY = { - STUDENT_CATEGORY_GRADE_OR_HIGH_SCHOOL_NAME, STUDENT_CATEGORY_COLLEGE_OR_HIGHER_NAME, - STUDENT_CATEGORY_NOT_STUDENT_NAME }; - - private Household hhObj; - - private int persNum; - private int persId; - private short persAge; - private short persGender; - private short persPecasOccup; - private short persActivityCode; - private short persEmploymentCategory; - private short persStudentCategory; - private short personType; - private boolean gradeSchool; - private boolean highSchool; - private boolean highSchoolGraduate; - private boolean hasBachelors; - - // individual value-of-time in $/hr - private float persValueOfTime; - - private int workLocation; - private int workLocSegmentIndex; - private float workLocDistance; - private float workLocLogsum; - private int schoolLoc; - private int schoolLocSegmentIndex; - private float schoolLocDistance; - private float schoolLocLogsum; - - private float timeFactorWork; - private float timeFactorNonWork; - - private short freeParkingAvailable; - private short internalExternalTripChoice = 1; - private float reimbursePercent; - - private float worksFromHomeLogsum; - private float parkingProvisionLogsum; - private float telecommuteLogsum; - private float ieLogsum; - private float cdapLogsum; - private float imtfLogsum; - private float inmtfLogsum; - - private String cdapActivity; - private short imtfChoice; - private short inmtfChoice; - - private int maxAdultOverlaps; - private int maxChildOverlaps; - - private short telecommuteChoice; - - private ArrayList workTourArrayList; - private ArrayList schoolTourArrayList; - private ArrayList indNonManTourArrayList; - private ArrayList atWorkSubtourArrayList; - - // private Scheduler scheduler; - // windows[] is 1s based - indexed from 1 to number of intervals. - private short[] windows; - - private int windowBeforeFirstMandJointTour; - private int windowBetweenFirstLastMandJointTour; - private int windowAfterLastMandJointTour; - - private ModelStructure modelStructure; - - public Person(Household hhObj, int persNum, ModelStructure modelStructure) - { - this.hhObj = hhObj; - this.persNum = persNum; - this.workTourArrayList = new ArrayList(); - this.schoolTourArrayList = new ArrayList(); - this.indNonManTourArrayList = new ArrayList(); - this.atWorkSubtourArrayList = new ArrayList(); - this.modelStructure = modelStructure; - - initializeWindows(); - - freeParkingAvailable = ParkingProvisionModel.FP_MODEL_REIMB_ALT; - reimbursePercent = 0.43f; - } - - public Household getHouseholdObject() - { - return hhObj; - } - - public ArrayList getListOfWorkTours() - { - return workTourArrayList; - } - - public ArrayList getListOfSchoolTours() - { - return schoolTourArrayList; - } - - public ArrayList getListOfIndividualNonMandatoryTours() - { - return indNonManTourArrayList; - } - - public ArrayList getListOfAtWorkSubtours() - { - return atWorkSubtourArrayList; - } - - public short[] getTimeWindows() - { - return windows; - } - - public String getTimePeriodLabel(int windowIndex) - { - return modelStructure.getTimePeriodLabel(windowIndex); - } - - public void initializeWindows() - { - windows = new short[modelStructure.getNumberOfTimePeriods() + 1]; - } - - public void resetTimeWindow(int startPeriod, int endPeriod) - { - for (int i = startPeriod; i <= endPeriod; i++) - { - windows[i] = 0; - } - } - - public void resetTimeWindow() - { - for (int i = 0; i < windows.length; i++) - { - windows[i] = 0; - } - } - - /** - * code the time window array for this tour being scheduled. 0: unscheduled, - * 1: scheduled, middle of tour, 2: scheduled, start of tour, 3: scheduled, - * end of tour, 4: scheduled, end of previous tour, start of current tour or - * end of current tour, start of subsequent tour; or current tour start/end - * same period. - * - * @param start - * is the departure period index for the tour - * @param end - * is the arrival period index for the tour - */ - public void scheduleWindow(int start, int end) - { - - /* - * This is the logic used in ARC/MTC, but for SANDAG, we don't allow - * overlapping tours - * - * - * if (start == end) { windows[start] = 4; } else { if (windows[start] - * == 3) windows[start] = 4; else if (windows[start] == 0) - * windows[start] = 2; - * - * if (windows[end] == 2) windows[end] = 4; else if (windows[end] == 0) - * windows[end] = 3; } - * - * for (int h = start + 1; h < end; h++) { windows[h] = 1; } - */ - - for (int h = start; h <= end; h++) - { - windows[h] = 1; - } - - } - - public boolean[] getAvailableTimeWindows(int[] altStarts, int[] altEnds) - { - - // availability array is used by UEC based choice model, which uses - // 1-based - // indexing - boolean[] availability = new boolean[altStarts.length + 1]; - - for (int i = 1; i <= altStarts.length; i++) - { - int start = altStarts[i - 1]; - int end = altEnds[i - 1]; - availability[i] = isWindowAvailable(start, end); - } - - return availability; - } - - public boolean isWindowAvailable(int start, int end) - { - - /* - * This is the logic used in ARC/MTC, but for SANDAG, we don't allow - * overlapping tours - * - * - * // check start period, if window is 0, it is unscheduled; // if - * window is 3, it is the last period of another tour, and available // - * as the first period of this tour. if (windows[start] == 1) return - * false; else if (windows[start] == 2 && start != end) return false; - * - * // check end period, if window is 0, it is unscheduled; // if window - * is 2, it is the first period of another tour, and available // as the - * last period of this tour. if (windows[end] == 1) return false; else - * if (windows[end] == 3 && start != end) return false; - * - * // the alternative is available if start and end are available, and - * all periods // from start+1,...,end-1 are available. for (int h = - * start + 1; h < end; h++) { if (windows[h] > 0) return false; } - * - * return true; - */ - - // the alternative is available if all intervals between start and end, - // inclusive, are available - for (int h = start; h <= end; h++) - { - if (windows[h] > 0) return false; - } - - return true; - - } - - /** - * @return true if the window for the argument is the end of a previously - * scheduled tour and this period does not overlap with any other - * tour. - */ - public boolean isPreviousArrival(int period) - { - - if (windows[period] == 3 || windows[period] == 4) return true; - else return false; - - } - - /** - * @return true if the window for the argument is the start of a previously - * scheduled tour and this period does not overlap with any other - * tour. - */ - public boolean isPreviousDeparture(int period) - { - - if (windows[period] == 2 || windows[period] == 4) return true; - else return false; - - } - - public boolean isPeriodAvailable(int period) - { - // if windows[index] == 0, the period is available. - - // if window[index] is 0 (available), 2 (start of another tour), 3 (end - // of - // another tour), 4 available for this period only, the period is - // available; - // otherwise, if window[index] is 1 (middle of another tour), it is not - // available. - if (windows[period] == 1) return false; - else return true; - } - - public void setPersId(int id) - { - persId = id; - } - - public void setFreeParkingAvailableResult(int chosenAlt) - { - freeParkingAvailable = (short) chosenAlt; - } - - /** - * set the chosen alternative number: 1=no, 2=yes - * - * @param chosenAlt - */ - public void setInternalExternalTripChoiceResult(int chosenAlt) - { - internalExternalTripChoice = (short) chosenAlt; - } - - public void setParkingReimbursement(double pct) - { - reimbursePercent = (float) pct; - } - - public void setWorkLocationSegmentIndex(int workSegment) - { - workLocSegmentIndex = workSegment; - } - - public void setSchoolLocationSegmentIndex(int schoolSegment) - { - schoolLocSegmentIndex = schoolSegment; - } - - public void setPersAge(int age) - { - persAge = (short) age; - } - - public void setPersGender(int gender) - { - persGender = (short) gender; - } - - public void setPersPecasOccup(int occup) - { - persPecasOccup = (short) occup; - } - - public void setPersActivityCode(int actCode) - { - persActivityCode = (short) actCode; - } - - public void setPersEmploymentCategory(int category) - { - persEmploymentCategory = (short) category; - } - - public void setPersStudentCategory(int category) - { - persStudentCategory = (short) category; - } - - public void setPersonTypeCategory(int personTypeCategory) - { - personType = (short) personTypeCategory; - } - - public void setValueOfTime(float vot) - { - persValueOfTime = vot; - } - - public void setWorkLocation(int aWorkLocationMgra) - { - workLocation = aWorkLocationMgra; - } - - public void setWorkLocDistance(float distance) - { - workLocDistance = distance; - } - - public void setWorkLocLogsum(float logsum) - { - workLocLogsum = logsum; - } - - public void setSchoolLoc(int loc) - { - schoolLoc = loc; - } - - public void setSchoolLocDistance(float distance) - { - schoolLocDistance = distance; - } - - public void setSchoolLocLogsum(float logsum) - { - schoolLocLogsum = logsum; - } - - public void setImtfChoice(int choice) - { - imtfChoice = (short) choice; - } - - public void setInmtfChoice(int choice) - { - inmtfChoice = (short) choice; - } - - public int getImtfChoice() - { - return imtfChoice; - } - - public int getInmtfChoice() - { - return inmtfChoice; - } - - public void clearIndividualNonMandatoryToursArray() - { - indNonManTourArrayList.clear(); - } - - public void createIndividualNonMandatoryTours(int numberOfTours, String primaryPurposeName) - { - - /* - * // if purpose is escort, need to determine if household has kids or - * not String purposeName = primaryPurposeName; if ( - * purposeName.equalsIgnoreCase( modelStructure.ESCORT_PURPOSE_NAME ) ) - * { if ( hhObj.getNumChildrenUnder19() > 0 ) purposeName += "_" + - * modelStructure.ESCORT_SEGMENT_NAMES[0]; else purposeName += "_" + - * modelStructure.ESCORT_SEGMENT_NAMES[1]; } int purposeIndex = - * modelStructure.getDcModelPurposeIndex( purposeName ); - */ - - int id = indNonManTourArrayList.size(); - - int primaryIndex = modelStructure.getPrimaryPurposeNameIndexMap().get(primaryPurposeName); - - for (int i = 0; i < numberOfTours; i++) - { - Tour tempTour = new Tour(id++, this.hhObj, this, primaryPurposeName, - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, primaryIndex); - - tempTour.setTourOrigMgra(this.hhObj.getHhMgra()); - tempTour.setTourDestMgra(0); - - tempTour.setTourPurpose(primaryPurposeName); - - tempTour.setTourDepartPeriod(DEFAULT_NON_MANDATORY_START_PERIOD); - tempTour.setTourArrivePeriod(DEFAULT_NON_MANDATORY_END_PERIOD); - - indNonManTourArrayList.add(tempTour); - } - - } - - public void createWorkTours(int numberOfTours, int startId, String tourPurpose, - int tourPurposeIndex) - { - - workTourArrayList.clear(); - - for (int i = 0; i < numberOfTours; i++) - { - int id = startId + i; - Tour tempTour = new Tour(this, id, tourPurposeIndex); - - tempTour.setTourOrigMgra(hhObj.getHhMgra()); - - if (workLocation == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) tempTour - .setTourDestMgra(hhObj.getHhMgra()); - else tempTour.setTourDestMgra(workLocation); - - tempTour.setTourPurpose(tourPurpose); - - tempTour.setTourDepartPeriod(-1); - tempTour.setTourArrivePeriod(-1); - // tempTour.setTourDepartPeriod(DEFAULT_MANDATORY_START_PERIOD); - // tempTour.setTourArrivePeriod(DEFAULT_MANDATORY_END_PERIOD); - - workTourArrayList.add(tempTour); - } - - } - - public void clearAtWorkSubtours() - { - - atWorkSubtourArrayList.clear(); - - } - - public void createAtWorkSubtour(int id, int choice, int workMgra, String subtourPurpose) - { - - /* - * String segmentedPurpose = modelStructure.AT_WORK_PURPOSE_NAME + "_" + - * tourPurpose; int purposeIndex = - * modelStructure.getDcModelPurposeIndex( segmentedPurpose ); - */ - - Tour tempTour = new Tour(id, this.hhObj, this, - ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME, ModelStructure.AT_WORK_CATEGORY, - ModelStructure.WORK_BASED_PRIMARY_PURPOSE_INDEX); - - tempTour.setTourOrigMgra(workMgra); - tempTour.setTourDestMgra(0); - - tempTour.setTourPurpose(ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME); - tempTour.setSubTourPurpose(subtourPurpose); - - tempTour.setTourDepartPeriod(DEFAULT_AT_WORK_SUBTOUR_START_PERIOD); - tempTour.setTourArrivePeriod(DEFAULT_AT_WORK_SUBTOUR_END_PERIOD); - - atWorkSubtourArrayList.add(tempTour); - - } - - public void createSchoolTours(int numberOfTours, int startId, String tourPurpose, - int tourPurposeIndex) - { - - schoolTourArrayList.clear(); - - for (int i = 0; i < numberOfTours; i++) - { - int id = startId + i; - Tour tempTour = new Tour(this, id, tourPurposeIndex); - - tempTour.setTourOrigMgra(this.hhObj.getHhMgra()); - - if (schoolLoc == ModelStructure.NOT_ENROLLED_SEGMENT_INDEX) tempTour - .setTourDestMgra(hhObj.getHhMgra()); - else tempTour.setTourDestMgra(schoolLoc); - - tempTour.setTourPurpose(tourPurpose); - - tempTour.setTourDepartPeriod(-1); - tempTour.setTourArrivePeriod(-1); - // tempTour.setTourDepartPeriod(DEFAULT_MANDATORY_START_PERIOD); - // tempTour.setTourArrivePeriod(DEFAULT_MANDATORY_END_PERIOD); - - schoolTourArrayList.add(tempTour); - } - } - - public int getWorkLocationSegmentIndex() - { - return workLocSegmentIndex; - } - - public int getSchoolLocationSegmentIndex() - { - return schoolLocSegmentIndex; - } - - public void setDailyActivityResult(String activity) - { - this.cdapActivity = activity; - } - - public int getPersonIsChildUnder16WithHomeOrNonMandatoryActivity() - { - - // check the person type - if (persIsStudentNonDrivingAge() == 1 || persIsPreschoolChild() == 1) - { - - // check the activity type - if (cdapActivity.equalsIgnoreCase(ModelStructure.HOME_PATTERN)) return (1); - - if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return (1); - - } - - return (0); - } - - /** - * @return 1 if M, 2 if N, 3 if H - */ - public int getCdapIndex() - { - - // return the activity type - if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return 1; - - if (cdapActivity.equalsIgnoreCase(ModelStructure.NONMANDATORY_PATTERN)) return 2; - - if (cdapActivity.equalsIgnoreCase(ModelStructure.HOME_PATTERN)) return 3; - - return (0); - } - - public int getPersonIsChild6To18WithoutMandatoryActivity() - { - - // check the person type - if (persIsStudentDrivingAge() == 1 || persIsStudentNonDrivingAge() == 1) - { - - // check the activity type - if (cdapActivity.equalsIgnoreCase(ModelStructure.MANDATORY_PATTERN)) return 0; - else return 1; - - } - - return 0; - } - - // methods DMU will use to get info from household object - - public int getAge() - { - return (int) persAge; - } - - public int getHomemaker() - { - return persIsHomemaker(); - } - - public int getGender() - { - return (int) persGender; - } - - public int getPersonIsFemale() - { - if (persGender == 2) return 1; - return 0; - } - - public int getPersonIsMale() - { - if (persGender == 1) return 1; - return 0; - } - - public int getPersonId() - { - return this.persId; - } - - public int getPersonNum() - { - return this.persNum; - } - - public String getPersonType() - { - return PERSON_TYPE_NAME_ARRAY[personType - 1]; - } - - public void setPersonIsHighSchool(boolean flag) - { - highSchool = flag; - } - - public int getPersonIsHighSchool() - { - return highSchool ? 1 : 0; - } - - public void setPersonIsGradeSchool(boolean flag) - { - gradeSchool = flag; - } - - public int getPersonIsGradeSchool() - { - return gradeSchool ? 1 : 0; - } - - public int getPersonIsHighSchoolGraduate() - { - return highSchoolGraduate ? 1 : 0; - } - - public void setPersonIsHighSchoolGraduate(boolean hsGrad) - { - highSchoolGraduate = hsGrad; - } - - public void setPersonHasBachelors(boolean hasBS) - { - hasBachelors = hasBS; - } - - public int getPersonTypeNumber() - { - return personType; - } - - public int getPersPecasOccup() - { - return (int) persPecasOccup; - } - - public int getPersActivityCode() - { - return (int) persActivityCode; - } - - public int getPersonEmploymentCategoryIndex() - { - return (int) persEmploymentCategory; - } - - public String getPersonEmploymentCategory() - { - return EMPLOYMENT_CATEGORY_NAME_ARRAY[persEmploymentCategory - 1]; - } - - public int getPersonStudentCategoryIndex() - { - return persStudentCategory; - } - - public String getPersonStudentCategory() - { - return STUDENT_CATEGORY_NAME_ARRAY[persStudentCategory - 1]; - } - - public float getValueOfTime() - { - return persValueOfTime; - } - - public int getWorkLocation() - { - return workLocation; - } - - public int getPersonSchoolLocationZone() - { - return schoolLoc; - } - - public int getFreeParkingAvailableResult() - { - return freeParkingAvailable; - } - - public int getInternalExternalTripChoiceResult() - { - return internalExternalTripChoice; - } - - public double getParkingReimbursement() - { - return reimbursePercent; - } - - public String getCdapActivity() - { - return cdapActivity; - } - - public float getWorkLocationDistance() - { - return workLocDistance; - } - - public float getWorkLocationLogsum() - { - return workLocLogsum; - } - - public int getUsualSchoolLocation() - { - return schoolLoc; - } - - public float getSchoolLocationDistance() - { - return schoolLocDistance; - } - - public float getSchoolLocationLogsum() - { - return schoolLocLogsum; - } - - public int getHasBachelors() - { - return hasBachelors ? 1 : 0; - } - - public int getNumWorkTours() - { - ArrayList workTours = getListOfWorkTours(); - if (workTours != null) return workTours.size(); - else return 0; - } - - public int getNumSchoolTours() - { - ArrayList schoolTours = getListOfSchoolTours(); - if (schoolTours != null) return schoolTours.size(); - else return 0; - } - - public int getNumIndividualEscortTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.ESCORT_PURPOSE_NAME)) num++; - return num; - } - - public int getNumIndividualShoppingTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SHOPPING_PURPOSE_NAME)) - num++; - return num; - } - - public int getNumIndividualEatOutTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.EAT_OUT_PURPOSE_NAME)) num++; - return num; - } - - public int getNumIndividualOthMaintTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_MAINT_PURPOSE_NAME)) - num++; - return num; - } - - public int getNumIndividualSocialTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SOCIAL_PURPOSE_NAME)) num++; - return num; - } - - public int getNumIndividualOthDiscrTours() - { - int num = 0; - for (Tour tour : getListOfIndividualNonMandatoryTours()) - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_DISCR_PURPOSE_NAME)) - num++; - return num; - } - - public int getNumMandatoryTours() - { - int numTours = 0; - ArrayList workTours = getListOfWorkTours(); - if (workTours != null) numTours += workTours.size(); - - ArrayList schoolTours = getListOfSchoolTours(); - if (schoolTours != null) numTours += schoolTours.size(); - - return numTours; - } - - public int getNumNonMandatoryTours() - { - ArrayList nonMandTours = getListOfIndividualNonMandatoryTours(); - if (nonMandTours == null) return 0; - else return nonMandTours.size(); - } - - public int getNumSubtours() - { - ArrayList subtours = getListOfAtWorkSubtours(); - if (subtours == null) return 0; - else return subtours.size(); - } - - public int getNumTotalIndivTours() - { - return getNumMandatoryTours() + getNumNonMandatoryTours() + getNumSubtours(); - } - - public int getNumJointShoppingTours() - { - return getNumJointToursForPurpose(modelStructure.SHOPPING_PURPOSE_NAME); - } - - public int getNumJointOthMaintTours() - { - return getNumJointToursForPurpose(modelStructure.OTH_MAINT_PURPOSE_NAME); - } - - public int getNumJointEatOutTours() - { - return getNumJointToursForPurpose(modelStructure.EAT_OUT_PURPOSE_NAME); - } - - public int getNumJointSocialTours() - { - return getNumJointToursForPurpose(modelStructure.SOCIAL_PURPOSE_NAME); - } - - public int getNumJointOthDiscrTours() - { - return getNumJointToursForPurpose(modelStructure.OTH_DISCR_PURPOSE_NAME); - } - - private int getNumJointToursForPurpose(String purposeName) - { - int count = 0; - Tour[] jt = hhObj.getJointTourArray(); - if (jt == null) return 0; - - for (int i = 0; i < jt.length; i++) - { - if (jt[i] == null) continue; - String jtPurposeName = jt[i].getTourPurpose(); - int[] personNumsParticipating = jt[i].getPersonNumArray(); - for (int p : personNumsParticipating) - { - if (p == persNum) - { - if (jtPurposeName.equalsIgnoreCase(purposeName)) count++; - break; - } - } - } - - return count; - } - - public void computeIdapResidualWindows() - { - - // find the start of the earliest mandatory or joint tour for this - // person - // and end of last one. - int firstTourStart = 9999; - int lastTourEnd = -1; - int firstTourEnd = 0; - int lastTourStart = 0; - - // first check mandatory tours - for (Tour tour : workTourArrayList) - { - int tourDeparts = tour.getTourDepartPeriod(); - int tourArrives = tour.getTourArrivePeriod(); - - if (tourDeparts < firstTourStart) - { - firstTourStart = tourDeparts; - firstTourEnd = tourArrives; - } - - if (tourArrives > lastTourEnd) - { - lastTourStart = tourDeparts; - lastTourEnd = tourArrives; - } - } - - for (Tour tour : schoolTourArrayList) - { - int tourDeparts = tour.getTourDepartPeriod(); - int tourArrives = tour.getTourArrivePeriod(); - - if (tourDeparts < firstTourStart) - { - firstTourStart = tourDeparts; - firstTourEnd = tourArrives; - } - - if (tourArrives > lastTourEnd) - { - lastTourStart = tourDeparts; - lastTourEnd = tourArrives; - } - } - - // now check joint tours - Tour[] jointTourArray = hhObj.getJointTourArray(); - if (jointTourArray != null) - { - for (Tour tour : jointTourArray) - { - - if (tour == null) continue; - - // see if this person is in the joint tour or not - if (tour.getPersonInJointTour(this)) - { - - int tourDeparts = tour.getTourDepartPeriod(); - int tourArrives = tour.getTourArrivePeriod(); - - if (tourDeparts < firstTourStart) - { - firstTourStart = tourDeparts; - firstTourEnd = tourArrives; - } - - if (tourArrives > lastTourEnd) - { - lastTourStart = tourDeparts; - lastTourEnd = tourArrives; - } - - } - - } - } - - if (firstTourStart > modelStructure.getNumberOfTimePeriods() - 1 && lastTourEnd < 0) - { - int numPeriods = windows.length; - windowBeforeFirstMandJointTour = numPeriods; - windowAfterLastMandJointTour = numPeriods; - windowBetweenFirstLastMandJointTour = numPeriods; - } else - { - - // since first tour first period and last tour last period are - // available, - // account for them. - windowBeforeFirstMandJointTour = firstTourStart + 1; - windowAfterLastMandJointTour = modelStructure.getNumberOfTimePeriods() - lastTourEnd; - - // find the number of unscheduled periods between end of first tour - // and - // start of last tour - windowBetweenFirstLastMandJointTour = 0; - for (int i = firstTourEnd; i <= lastTourStart; i++) - { - if (isPeriodAvailable(i)) windowBetweenFirstLastMandJointTour++; - } - } - - } - - public int getWindowBeforeFirstMandJointTour() - { - return windowBeforeFirstMandJointTour; - } - - public int getWindowBetweenFirstLastMandJointTour() - { - return windowBetweenFirstLastMandJointTour; - } - - public int getWindowAfterLastMandJointTour() - { - return windowAfterLastMandJointTour; - } - - // public int getNumberOfMandatoryWorkTours( String workPurposeName ){ - // - // int numberOfTours = 0; - // for(int i=0;i= MIN_ADULT_AGE - && persEmploymentCategory == EmployStatus.NOT_EMPLOYED.ordinal()) return 1; - else return 0; - } - - public int notEmployed() - { - if (persEmploymentCategory == EmployStatus.NOT_EMPLOYED.ordinal()) return 1; - else return 0; - } - - private int persIsWorker() - { - if (persEmploymentCategory == EmployStatus.FULL_TIME.ordinal() - || persEmploymentCategory == EmployStatus.PART_TIME.ordinal()) return 1; - else return 0; - } - - private int persIsStudent() - { - if (persStudentCategory == StudentStatus.STUDENT_HIGH_SCHOOL_OR_LESS.ordinal() - || persStudentCategory == StudentStatus.STUDENT_COLLEGE_OR_HIGHER.ordinal()) - { - return 1; - } else - { - return 0; - } - } - - private int persIsFullTimeWorker() - { - if (persEmploymentCategory == EmployStatus.FULL_TIME.ordinal()) return 1; - else return 0; - } - - private int persIsPartTimeWorker() - { - if (persEmploymentCategory == EmployStatus.PART_TIME.ordinal()) return 1; - else return 0; - } - - private int persTypeIsFullTimeWorker() - { - if (personType == PersonType.FT_worker_age_16plus.ordinal()) return 1; - else return 0; - } - - private int persTypeIsPartTimeWorker() - { - if (personType == PersonType.PT_worker_nonstudent_age_16plus.ordinal()) return 1; - else return 0; - } - - private int persIsUniversity() - { - if (personType == PersonType.University_student.ordinal()) return 1; - else return 0; - } - - private int persIsStudentDrivingAge() - { - if (personType == PersonType.Student_age_16_19_not_FT_wrkr_or_univ_stud.ordinal()) return 1; - else return 0; - } - - private int persIsStudentNonDrivingAge() - { - if (personType == PersonType.Student_age_6_15_schpred.ordinal()) return 1; - else return 0; - } - - private int persIsPreschoolChild() - { - if (personType == PersonType.Preschool_under_age_6.ordinal()) return 1; - else return 0; - - } - - private int persIsNonWorkingAdultUnder65() - { - if (personType == PersonType.Nonworker_nonstudent_age_16_64.ordinal()) return 1; - else return 0; - } - - private int persIsNonWorkingAdultOver65() - { - if (personType == PersonType.Nonworker_nonstudent_age_65plus.ordinal()) - { - return 1; - } else - { - return 0; - } - } - - /** - * return maximum periods of overlap between this person and other adult - * persons in the household. - * - * @return the most number of periods mutually available between this person - * and other adult household members - */ - public int getMaxAdultOverlaps() - { - return maxAdultOverlaps; - } - - /** - * set maximum periods of overlap between this person and other adult - * persons in the household. - * - * @param overlaps - * are the most number of periods mutually available between this - * person and other adult household members - */ - public void setMaxAdultOverlaps(int overlaps) - { - maxAdultOverlaps = overlaps; - } - - /** - * return maximum periods of overlap between this person and other children - * in the household. - * - * @return the most number of periods mutually available between this person - * and other child household members - */ - public int getMaxChildOverlaps() - { - return maxChildOverlaps; - } - - /** - * set maximum periods of overlap between this person and other children in - * the household. - * - * @param overlaps - * are the most number of periods mutually available between this - * person and other child household members - */ - public void setMaxChildOverlaps(int overlaps) - { - maxChildOverlaps = overlaps; - } - - /** - * return available time window for this person in the household. - * - * @return the total number of periods available for this person - */ - public int getAvailableWindow() - { - int numPeriodsAvailable = 0; - for (int i = 1; i < windows.length; i++) - if (windows[i] != 1) numPeriodsAvailable++; - - return numPeriodsAvailable; - } - - /** - * determine the maximum consecutive available time window for the person - * - * @return the length of the maximum available window in units of time - * intervals - */ - public int getMaximumContinuousAvailableWindow() - { - int maxWindow = 0; - int currentWindow = 0; - for (int i = 1; i < windows.length; i++) - { - if (windows[i] == 0) - { - currentWindow++; - } else - { - if (currentWindow > maxWindow) maxWindow = currentWindow; - currentWindow = 0; - } - } - if (currentWindow > maxWindow) maxWindow = currentWindow; - - return maxWindow; - } - - /** - * determine the maximum consecutive pairwise available time window for this - * person and the person for which a window was passed - * - * @return the length of the maximum pairwise available window in units of - * time intervals - */ - public int getMaximumContinuousPairwiseAvailableWindow(short[] otherWindow) - { - int maxWindow = 0; - int currentWindow = 0; - for (int i = 1; i < windows.length; i++) - { - if (windows[i] == 0 && otherWindow[i] == 0) - { - currentWindow++; - } else - { - if (currentWindow > maxWindow) maxWindow = currentWindow; - currentWindow = 0; - } - } - if (currentWindow > maxWindow) maxWindow = currentWindow; - - return maxWindow; - } - - public void setTimeWindows(short[] win) - { - windows = win; - } - - public void initializeForAoRestart() - { - - cdapActivity = "-"; - imtfChoice = 0; - inmtfChoice = 0; - - maxAdultOverlaps = 0; - maxChildOverlaps = 0; - - workTourArrayList.clear(); - schoolTourArrayList.clear(); - indNonManTourArrayList.clear(); - atWorkSubtourArrayList.clear(); - - initializeWindows(); - - windowBeforeFirstMandJointTour = 0; - windowBetweenFirstLastMandJointTour = 0; - windowAfterLastMandJointTour = 0; - - } - - public void initializeForImtfRestart() - { - - imtfChoice = 0; - inmtfChoice = 0; - - maxAdultOverlaps = 0; - maxChildOverlaps = 0; - - workTourArrayList.clear(); - schoolTourArrayList.clear(); - indNonManTourArrayList.clear(); - atWorkSubtourArrayList.clear(); - - initializeWindows(); - - windowBeforeFirstMandJointTour = 0; - windowBetweenFirstLastMandJointTour = 0; - windowAfterLastMandJointTour = 0; - - } - - /** - * initialize the person attributes and tour objects for restarting the - * model at joint tour frequency - */ - public void initializeForJtfRestart() - { - - inmtfChoice = 0; - - indNonManTourArrayList.clear(); - atWorkSubtourArrayList.clear(); - - for (int i = 0; i < workTourArrayList.size(); i++) - { - Tour t = workTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - for (int i = 0; i < schoolTourArrayList.size(); i++) - { - Tour t = schoolTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - - windowBeforeFirstMandJointTour = 0; - windowBetweenFirstLastMandJointTour = 0; - windowAfterLastMandJointTour = 0; - - } - - /** - * initialize the person attributes and tour objects for restarting the - * model at individual non-mandatory tour frequency. - */ - public void initializeForInmtfRestart() - { - - inmtfChoice = 0; - - indNonManTourArrayList.clear(); - atWorkSubtourArrayList.clear(); - - for (int i = 0; i < workTourArrayList.size(); i++) - { - Tour t = workTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - for (int i = 0; i < schoolTourArrayList.size(); i++) - { - Tour t = schoolTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - - windowBeforeFirstMandJointTour = 0; - windowBetweenFirstLastMandJointTour = 0; - windowAfterLastMandJointTour = 0; - - } - - /** - * initialize the person attributes and tour objects for restarting the - * model at at-work sub-tour frequency. - */ - public void initializeForAwfRestart() - { - - atWorkSubtourArrayList.clear(); - - for (int i = 0; i < workTourArrayList.size(); i++) - { - Tour t = workTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - for (int i = 0; i < schoolTourArrayList.size(); i++) - { - Tour t = schoolTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - for (int i = 0; i < indNonManTourArrayList.size(); i++) - { - Tour t = indNonManTourArrayList.get(i); - scheduleWindow(t.getTourDepartPeriod(), t.getTourArrivePeriod()); - t.clearStopModelResults(); - } - - } - - /** - * initialize the person attributes and tour objects for restarting the - * model at stop frequency. - */ - public void initializeForStfRestart() - { - - for (int i = 0; i < workTourArrayList.size(); i++) - { - Tour t = workTourArrayList.get(i); - t.clearStopModelResults(); - } - for (int i = 0; i < schoolTourArrayList.size(); i++) - { - Tour t = schoolTourArrayList.get(i); - t.clearStopModelResults(); - } - for (int i = 0; i < atWorkSubtourArrayList.size(); i++) - { - Tour t = atWorkSubtourArrayList.get(i); - t.clearStopModelResults(); - } - for (int i = 0; i < indNonManTourArrayList.size(); i++) - { - Tour t = indNonManTourArrayList.get(i); - t.clearStopModelResults(); - } - - } - - public float getParkingProvisionLogsum() { - return parkingProvisionLogsum; - } - - public void setParkingProvisionLogsum(float parkingProvisionLogsum) { - this.parkingProvisionLogsum = parkingProvisionLogsum; - } - - public float getIeLogsum() { - return ieLogsum; - } - - public void setIeLogsum(float ieLogsum) { - this.ieLogsum = ieLogsum; - } - - public float getCdapLogsum() { - return cdapLogsum; - } - - public void setCdapLogsum(float cdapLogsum) { - this.cdapLogsum = cdapLogsum; - } - - - public float getImtfLogsum() { - return imtfLogsum; - } - - public void setImtfLogsum(float imtfLogsum) { - this.imtfLogsum = imtfLogsum; - } - - public float getInmtfLogsum() { - return inmtfLogsum; - } - - public void setInmtfLogsum(float inmtfLogsum) { - this.inmtfLogsum = inmtfLogsum; - } - - public float getWorksFromHomeLogsum() { - return worksFromHomeLogsum; - } - - public void setWorksFromHomeLogsum(float worksFromHomeLogsum) { - this.worksFromHomeLogsum = worksFromHomeLogsum; - } - - public void logPersonObject(Logger logger, int totalChars) - { - - Household.logHelper(logger, "persNum: ", persNum, totalChars); - Household.logHelper(logger, "persId: ", persId, totalChars); - Household.logHelper(logger, "persAge: ", persAge, totalChars); - Household.logHelper(logger, "persGender: ", persGender, totalChars); - Household.logHelper(logger, "persEmploymentCategory: ", persEmploymentCategory, totalChars); - Household.logHelper(logger, "persStudentCategory: ", persStudentCategory, totalChars); - Household.logHelper(logger, "personType: ", personType, totalChars); - Household.logHelper(logger, "workLoc: ", workLocation, totalChars); - Household.logHelper(logger, "schoolLoc: ", schoolLoc, totalChars); - Household.logHelper(logger, "workLocSegmentIndex: ", workLocSegmentIndex, totalChars); - Household.logHelper(logger, "schoolLocSegmentIndex: ", schoolLocSegmentIndex, totalChars); - - Household.logHelper(logger, "timeFactorWork: ", String.format("%.2f",timeFactorWork), totalChars); - Household.logHelper(logger, "timeFactorNonWork: ", String.format("%.2f",timeFactorNonWork), totalChars); - Household.logHelper(logger, "freeParkingAvailable: ", freeParkingAvailable, totalChars); - Household.logHelper(logger, "reimbursementPct: ", - String.format("%.2f%%", (100 * reimbursePercent)), totalChars); - Household.logHelper(logger, "cdapActivity: ", cdapActivity, totalChars); - Household.logHelper(logger, "imtfChoice: ", imtfChoice, totalChars); - Household.logHelper(logger, "inmtfChoice: ", inmtfChoice, totalChars); - Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); - Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); - Household.logHelper(logger, "windowBeforeFirstMandJointTour: ", - windowBeforeFirstMandJointTour, totalChars); - Household.logHelper(logger, "windowBetweenFirstLastMandJointTour: ", - windowBetweenFirstLastMandJointTour, totalChars); - Household.logHelper(logger, "windowAfterLastMandJointTour: ", windowAfterLastMandJointTour, - totalChars); - - String header1 = " Index: |"; - String header2 = " Period: |"; - String windowString = " Window: |"; - String periodString = ""; - for (int i = 1; i < windows.length; i++) - { - header1 += String.format(" %2d |", i); - header2 += String.format("%4s|", modelStructure.getTimePeriodLabel(i)); - switch (windows[i]) - { - case 0: - periodString = " "; - break; - case 1: - periodString = "XXXX"; - break; - } - windowString += String.format("%4s|", periodString); - } - - logger.info(header1); - logger.info(header2); - logger.info(windowString); - - if (workTourArrayList.size() > 0) - { - for (Tour tour : workTourArrayList) - { - int id = tour.getTourId(); - logger.info(tour.getTourWindow(String.format("W%d", id))); - } - } - if (atWorkSubtourArrayList.size() > 0) - { - for (Tour tour : atWorkSubtourArrayList) - { - int id = tour.getTourId(); - String alias = ""; - String purposeName = tour.getSubTourPurpose(); - if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME)) alias = "aB"; - else if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_EAT_PURPOSE_NAME)) alias = "aE"; - else if (purposeName.equalsIgnoreCase(modelStructure.AT_WORK_MAINT_PURPOSE_NAME)) - alias = "aM"; - logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); - } - } - if (schoolTourArrayList.size() > 0) - { - for (Tour tour : schoolTourArrayList) - { - int id = tour.getTourId(); - String alias = "S"; - logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); - } - } - if (hhObj.getJointTourArray() != null && hhObj.getJointTourArray().length > 0) - { - for (Tour tour : hhObj.getJointTourArray()) - { - if (tour == null) continue; - - // log this persons time window if they are in the joint tour - // party. - int[] persNumArray = tour.getPersonNumArray(); - if (persNumArray != null) - { - for (int num : persNumArray) - { - if (num == persNum) - { - - Person person = hhObj.getPersons()[num]; - tour.setPersonObject(person); - - int id = tour.getTourId(); - String alias = ""; - if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) alias = "jE"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) alias = "jS"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) alias = "jM"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) alias = "jV"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) alias = "jD"; - logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); - } - } - } - } - } - if (indNonManTourArrayList.size() > 0) - { - for (Tour tour : indNonManTourArrayList) - { - int id = tour.getTourId(); - String alias = ""; - if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) alias = "ie"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME)) alias = "iE"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) alias = "iS"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) alias = "iM"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) alias = "iV"; - else if (tour.getTourPurpose().equalsIgnoreCase( - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) alias = "iD"; - logger.info(tour.getTourWindow(String.format("%s%d", alias, id))); - } - } - - } - - public void logTourObject(Logger logger, int totalChars, Tour tour) - { - tour.logTourObject(logger, totalChars); - } - - public void logEntirePersonObject(Logger logger) - { - - int totalChars = 60; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - Household.logHelper(logger, "persNum: ", persNum, totalChars); - Household.logHelper(logger, "persId: ", persId, totalChars); - Household.logHelper(logger, "persAge: ", persAge, totalChars); - Household.logHelper(logger, "persGender: ", persGender, totalChars); - Household.logHelper(logger, "persEmploymentCategory: ", persEmploymentCategory, totalChars); - Household.logHelper(logger, "persStudentCategory: ", persStudentCategory, totalChars); - Household.logHelper(logger, "personType: ", personType, totalChars); - Household.logHelper(logger, "workLoc: ", workLocation, totalChars); - Household.logHelper(logger, "schoolLoc: ", schoolLoc, totalChars); - Household.logHelper(logger, "workLocSegmentIndex: ", workLocSegmentIndex, totalChars); - Household.logHelper(logger, "schoolLocSegmentIndex: ", schoolLocSegmentIndex, totalChars); - Household.logHelper(logger, "freeParkingAvailable: ", freeParkingAvailable, totalChars); - Household.logHelper(logger, "reimbursementPct: ", - String.format("%.2f%%", (100 * reimbursePercent)), totalChars); - Household.logHelper(logger, "cdapActivity: ", cdapActivity, totalChars); - Household.logHelper(logger, "imtfChoice: ", imtfChoice, totalChars); - Household.logHelper(logger, "inmtfChoice: ", inmtfChoice, totalChars); - Household.logHelper(logger, "maxAdultOverlaps: ", maxAdultOverlaps, totalChars); - Household.logHelper(logger, "maxChildOverlaps: ", maxChildOverlaps, totalChars); - Household.logHelper(logger, "windowBeforeFirstMandJointTour: ", - windowBeforeFirstMandJointTour, totalChars); - Household.logHelper(logger, "windowBetweenFirstLastMandJointTour: ", - windowBetweenFirstLastMandJointTour, totalChars); - Household.logHelper(logger, "windowAfterLastMandJointTour: ", windowAfterLastMandJointTour, - totalChars); - - String header = " Period: |"; - String windowString = " Window: |"; - for (int i = 1; i < windows.length; i++) - { - header += String.format("%4s|", modelStructure.getTimePeriodLabel(i)); - windowString += String.format("%4s|", windows[i] == 0 ? " " : "XXXX"); - } - - logger.info(header); - logger.info(windowString); - - if (workTourArrayList.size() > 0) - { - for (Tour tour : workTourArrayList) - { - int id = tour.getTourId(); - logger.info(tour.getTourWindow(String.format("W%d", id))); - } - } - if (schoolTourArrayList.size() > 0) - { - for (Tour tour : schoolTourArrayList) - { - logger.info(tour - .getTourWindow(tour.getTourPurpose().equalsIgnoreCase("university") ? "U" - : "S")); - } - } - if (indNonManTourArrayList.size() > 0) - { - for (Tour tour : indNonManTourArrayList) - { - logger.info(tour.getTourWindow("N")); - } - } - if (atWorkSubtourArrayList.size() > 0) - { - for (Tour tour : atWorkSubtourArrayList) - { - logger.info(tour.getTourWindow("A")); - } - } - if (hhObj.getJointTourArray() != null && hhObj.getJointTourArray().length > 0) - { - for (Tour tour : hhObj.getJointTourArray()) - { - if (tour != null) logger.info(tour.getTourWindow("J")); - } - } - - logger.info(separater); - - logger.info("Work Tours:"); - if (workTourArrayList.size() > 0) - { - for (Tour tour : workTourArrayList) - { - tour.logEntireTourObject(logger); - } - } else - { - logger.info(" No work tours"); - } - - logger.info("School Tours:"); - if (schoolTourArrayList.size() > 0) - { - for (Tour tour : schoolTourArrayList) - { - tour.logEntireTourObject(logger); - } - } else - { - logger.info(" No school tours"); - } - - logger.info("Individual Non-mandatory Tours:"); - if (indNonManTourArrayList.size() > 0) - { - for (Tour tour : indNonManTourArrayList) - { - tour.logEntireTourObject(logger); - } - } else - { - logger.info(" No individual non-mandatory tours"); - } - - logger.info("Work based subtours Tours:"); - if (atWorkSubtourArrayList.size() > 0) - { - for (Tour tour : atWorkSubtourArrayList) - { - tour.logEntireTourObject(logger); - } - } else - { - logger.info(" No work based subtours"); - } - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public double getTimeFactorWork() { - return (double) timeFactorWork; - } - - public void setTimeFactorWork(double timeFactorWork) { - this.timeFactorWork = (float) timeFactorWork; - } - - public double getTimeFactorNonWork() { - return (double) timeFactorNonWork; - } - - public void setTimeFactorNonWork(double timeFactorNonWork) { - this.timeFactorNonWork = (float) timeFactorNonWork; - } - - public enum EmployStatus - { - nul, FULL_TIME, PART_TIME, NOT_EMPLOYED, UNDER16 - } - - public enum StudentStatus - { - nul, STUDENT_HIGH_SCHOOL_OR_LESS, STUDENT_COLLEGE_OR_HIGHER, NON_STUDENT - } - - public enum PersonType - { - nul, FT_worker_age_16plus, PT_worker_nonstudent_age_16plus, University_student, Nonworker_nonstudent_age_16_64, Nonworker_nonstudent_age_65plus, Student_age_16_19_not_FT_wrkr_or_univ_stud, Student_age_6_15_schpred, Preschool_under_age_6 - } - - /** - * Returns true if this person is an active adult, else returns false. Active adult - * is defined as full-time worker, part-time worker, university student, - * non-working adult or retired person who has an activity pattern other than H. - * - * @return true if active adult, else false. - */ - public boolean isActiveAdult(){ - boolean activeAdult=false; - if((getPersonTypeNumber()==Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX)||(getPersonTypeNumber()==Person.PERSON_TYPE_PART_TIME_WORKER_INDEX)|| - (getPersonTypeNumber()==Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX)||(getPersonTypeNumber()==Person.PERSON_TYPE_NON_WORKER_INDEX)|| - (getPersonTypeNumber()==Person.PERSON_TYPE_RETIRED_INDEX)) - if(!getCdapActivity().equalsIgnoreCase(ModelStructure.HOME_PATTERN)) - activeAdult=true; - return activeAdult; - } - - public short getTelecommuteChoice() { - return telecommuteChoice; - } - - public void setTelecommuteChoice(short telecommuteChoice) { - this.telecommuteChoice = telecommuteChoice; - } - - public float getTelecommuteLogsum() { - return telecommuteLogsum; - } - - public void setTelecommuteLogsum(float telecommuteLogsum) { - this.telecommuteLogsum = telecommuteLogsum; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java deleted file mode 100644 index 32f93ef..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChauffeurResult.java +++ /dev/null @@ -1,67 +0,0 @@ -/* -* The school-escort model was designed by PB (Gupta, Vovsha, et al) -* as part of the Maricopa Association of Governments (MAG) -* Activity-based Travel Model Development project. -* -* This source code, which implements the school escort model, -* was written exclusively for and funded by MAG as part of the -* same project; therefore, per their contract, the -* source code belongs to MAG and can only be used with their -* permission. -* -* It is being adapted for the Southern Oregon ABM by PB & RSG -* with permission from MAG and all references to -* the school escort model as well as source code adapted from this -* original code should credit MAG's role in its development. -* -* The escort model and source code should not be transferred to or -* adapted for other agencies or used in other projects without -* expressed permission from MAG. -* -* The source code has been substantially revised to fit within the -* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). -*/ - -package org.sandag.abm.ctramp; - -import java.io.Serializable; - -public class SchoolEscortChauffeurResult implements Serializable { - - private static final long serialVersionUID = 1L; - - private final int pid; - private final short dir; - private final short bundle; - private final short escortType; - private final short[] childPnums; - - public SchoolEscortChauffeurResult( int pid, short dir, short bundle, short escortType, short[] childPnums ) { - this.pid = pid; - this.dir = dir; - this.bundle = bundle; - this.escortType = escortType; - this.childPnums = childPnums; - } - - public int getPid() { - return pid; - } - - public short getDirection() { - return dir; - } - - public short getBundle() { - return bundle; - } - - public short getEscortType() { - return escortType; - } - - public short[] getChildPnums() { - return childPnums; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java deleted file mode 100644 index cfa6098..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortChildResult.java +++ /dev/null @@ -1,68 +0,0 @@ -/* -* The school-escort model was designed by PB (Gupta, Vovsha, et al) -* as part of the Maricopa Association of Governments (MAG) -* Activity-based Travel Model Development project. -* -* This source code, which implements the school escort model, -* was written exclusively for and funded by MAG as part of the -* same project; therefore, per their contract, the -* source code belongs to MAG and can only be used with their -* permission. -* -* It is being adapted for the Southern Oregon ABM by PB & RSG -* with permission from MAG and all references to -* the school escort model as well as source code adapted from this -* original code should credit MAG's role in its development. -* -* The escort model and source code should not be transferred to or -* adapted for other agencies or used in other projects without -* expressed permission from MAG. -* -* The source code has been substantially revised to fit within the -* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). -* -*/ - -package org.sandag.abm.ctramp; - -import java.io.Serializable; - -public class SchoolEscortChildResult implements Serializable { - - private static final long serialVersionUID = 1L; - - private final int pid; - private final short dir; - private final short bundle; - private final short escortType; - private final short adultPnum; - - public SchoolEscortChildResult( int pid, short dir, short bundle, short escortType, short adultPnum ) { - this.pid = pid; - this.dir = dir; - this.bundle = bundle; - this.escortType = escortType; - this.adultPnum = adultPnum; - } - - public int getPid() { - return pid; - } - - public short getDirection() { - return dir; - } - - public short getBundle() { - return bundle; - } - - public short getEscortType() { - return escortType; - } - - public short getAdultPnum() { - return adultPnum; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java deleted file mode 100644 index ee37952..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingBundle.java +++ /dev/null @@ -1,499 +0,0 @@ -/* -* The school-escort model was designed by PB (Gupta, Vovsha, et al) -* as part of the Maricopa Association of Governments (MAG) -* Activity-based Travel Model Development project. -* -* This source code, which implements the school escort model, -* was written exclusively for and funded by MAG as part of the -* same project; therefore, per their contract, the -* source code belongs to MAG and can only be used with their -* permission. -* -* It is being adapted for the Southern Oregon ABM by PB & RSG -* with permission from MAG and all references to -* the school escort model as well as source code adapted from this -* original code should credit MAG's role in its development. -* -* The escort model and source code should not be transferred to or -* adapted for other agencies or used in other projects without -* expressed permission from MAG. -* -* The source code has been substantially revised to fit within the -* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). -*/ - -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.List; -import java.util.StringTokenizer; - -import org.apache.log4j.Logger; - - -public class SchoolEscortingBundle implements Serializable { - - private static final long serialVersionUID = 1L; - - private int id; - private int dir; - private final int alt; - private final int bundle; - private final int type; - private final int chaufId; - private int chaufPnum; - private int chaufPersType; - private int chaufPid; - private int[] childIds; - private int[] childPnums; - private int[] schoolMazs; - private float[] schoolDists; - private int workOrSchoolMaz; - private int departHome; - private int arriveWork; - private int departWork; - private int arriveHome; - private int departPrimaryInterval = -1; - - private SchoolEscortingBundle( int alt, int bundle, int chaufId, int type, int[] childIds, int[] childPnums ) { - this.alt = alt; - this.bundle = bundle; - this.chaufId = chaufId; - this.type = type; - this.childIds = childIds; - this.childPnums = childPnums; - } - - - /** - * Get an Arraylist of SchoolEscortingBundles, dimensioned by: - * 0: max chauffeurs (2) - * 1: max bundles (3) - * - * @param alt The alternative number - * @param altBundleIncidence - * @return An array of SchoolEscortingBundles (dimensioned by 2, for each chauffeur) - */ - public static List[] constructAltBundles( int alt, int[][] altBundleIncidence ) { - - //first dimension of results array is dimensioned by number of chauffeurs + 1 - List[] results = new List[ SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1 ]; - - //for each potential bundle (3) - for ( int i=1; i <= SchoolEscortingModel.NUM_BUNDLES; i++ ) { - - //for each potential chauffeur (2) - for ( int j=1; j <= SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH; j++ ) { - - // for each escort type (rideshare vs pure escort) - for ( int k=1; k <= SchoolEscortingModel.NUM_ESCORT_TYPES; k++ ) { - - //if an arraylist hasn't been created for this chauffeur, create one. - if ( results[j] == null ) - results[j] = new ArrayList(SchoolEscortingModel.NUM_BUNDLES); - - //childIdList initial capacity is max escortees (3); for each potential escortee - List childIdList = new ArrayList(SchoolEscortingModel.NUM_ESCORTEES_PER_HH); - for ( int l=1; l <= SchoolEscortingModel.NUM_ESCORTEES_PER_HH; l++ ) { - int columnIndex = (i-1) * (SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH * SchoolEscortingModel.NUM_ESCORT_TYPES * SchoolEscortingModel.NUM_ESCORTEES_PER_HH) - + (j-1) * (SchoolEscortingModel.NUM_ESCORT_TYPES * SchoolEscortingModel.NUM_ESCORTEES_PER_HH) - + (k-1) * (SchoolEscortingModel.NUM_ESCORTEES_PER_HH) - + (l-1) + 1; - //if child number l belongs to this bundle\chauffeur\escort type combination, add l to the childIdList - if ( altBundleIncidence[alt][columnIndex] > 0 ) - childIdList.add( l ); - } - - //if children are in this bundle\chauffeur\escort type combination - if ( childIdList.size() > 0 ) { - int[] childIds = new int[childIdList.size()]; - int[] childPnums = new int[childIdList.size()]; - for ( int l=0; l < childIdList.size(); l++ ) { - childIds[l] = childIdList.get( l ); - } - //add a new bundle to the chauffeur element. The bundle contains the number of the bundle, the number of the chauffeur, the - //escort type (rideshare versus pure), the child ids (1 through 3) and an empty array of person numbers for each child. - results[j].add( new SchoolEscortingBundle( alt, i, j, k, childIds, childPnums ) ); - } - - } - - } - - } - - return results; - - } - - public void setId( int id ) { - this.id = id; - } - - public int getId() { - return id; - } - - public void setDir( int dir ) { - this.dir = dir; - } - - public int getDir() { - return dir; - } - - public int getAlt() { - return alt; - } - - public int getBundle() { - return bundle; - } - - public int getChaufId() { - return chaufId; - } - - public void setChaufPnum( int pnum ) { - chaufPnum = pnum; - } - - public int getChaufPnum() { - return chaufPnum; - } - - public void setChaufPersType( int ptype ) { - chaufPersType = ptype; - } - - public int getChaufPersType() { - return chaufPersType; - } - - public void setChaufPid( int pid ) { - chaufPid = pid; - } - - public int getChaufPid() { - return chaufPid; - } - - public int getEscortType() { - return type; - } - - public void setSchoolMazs( int[] schoolMazs ) { - this.schoolMazs = schoolMazs; - } - - public int[] getSchoolMazs() { - return schoolMazs; - } - - public void setSchoolDists( float[] schoolDists ) { - this.schoolDists = schoolDists; - } - - public float[] getSchoolDists() { - return schoolDists; - } - - public void setChildIds( int[] childIds ) { - this.childIds = childIds; - } - - public int[] getChildIds() { - return childIds; - } - - public void setChildPnums( int[] childPnums ) { - this.childPnums = childPnums; - } - - public int[] getChildPnums() { - return childPnums; - } - - public void setDepartHome( int depart ) { - departHome = depart; - } - - public int getDepartHome() { - return departHome; - } - - /* - public void setArriveHome( int arrive ) { - arriveHome = Math.min( arrive, TourTodDmu.NUM_TOD_INTERVALS ); - } -*/ - /** - * Arrive home; modified JEF to remove taking the minimum of arrive and number of TOD intervals. - * @param arrive - */ - public void setArriveHome( int arrive ) { - arriveHome = arrive; - } - - - public int getArriveHome() { - return arriveHome; - } - - public void setDepartWork( int depart ) { - departWork = depart; - } - - public int getDepartWork() { - return departWork; - } - - public void setArriveWork( int arrive ) { - arriveWork = arrive; - } - - public int getArriveWork() { - return arriveWork; - } - - public void setWorkOrSchoolMaz( int maz ) { - workOrSchoolMaz = maz; - } - - public int getWorkOrSchoolMaz() { - return workOrSchoolMaz; - } - - public void setDepartPrimaryInterval( int interval ) { - departPrimaryInterval = interval; - } - - public int getDepartPrimaryInterval() { - return departPrimaryInterval; - } - - - public String toString() { - - String childIdString = "["; - String childPnumString = "["; - String schoolString = "["; - String distsString = "["; - if ( childIds.length > 0 ) { - childIdString += childIds[0]; - childPnumString += childPnums[0]; - schoolString += schoolMazs[0]; - distsString += String.format( "%.5f", schoolDists[0] ); - for ( int i=1; i < childIds.length; i++ ) { - childIdString += "," + childIds[i]; - childPnumString += "," + childPnums[i]; - schoolString += "," + schoolMazs[i]; - distsString += "," + String.format( "%.5f", schoolDists[i] ); - } - } - childIdString += "]"; - childPnumString += "]"; - schoolString += "]"; - distsString += "]"; - - String outputString = - "\tid = " + id + "\n" + - "\tdir = " + (dir == SchoolEscortingModel.DIR_OUTBOUND ? "outbound" : "inbound" ) + "\n" + - "\talt = " + alt + "\n" + - "\tbundle = " + bundle + "\n" + - "\tchaufPnum = " + chaufPnum + "\n" + - "\tchaufPid = " + chaufPid + "\n" + - "\tchaufPtype = " + chaufPersType + "\n" + - "\tescort type = " + (type == ModelStructure.RIDE_SHARING_TYPE ? "ride sharing" : "pure escort" ) + "\n" + - "\tchildIds = " + childIdString + "\n" + - "\tchildPnums = " + childPnumString + "\n" + - "\tschoolMazs = " + schoolString + "\n" + - "\tschoolDists = " + distsString + "\n" + - "\tdepartHome = " + departHome + "\n" + - "\tarriveHome = " + arriveHome + "\n" + - "\tdepartWork = " + departWork + "\n" + - "\tarriveWork = " + arriveWork + "\n\n"; - - return outputString; - - } - - public static String getExportHeaderString() { - String header = "id,dir,alt,bundle,type,chaufId,chaufPnum,chaufPersType,chaufPid,departHome,arriveHome,departWork,arriveWork,childIds,childPnums,schoolMazs,schoolDists"; - return header; - } - - public String getExportString() { - - String childIdString = "["; - String childPnumString = "["; - String schoolString = "["; - String distsString = "["; - if ( childIds.length > 0 ) { - childIdString += childIds[0]; - childPnumString += childPnums[0]; - schoolString += schoolMazs[0]; - distsString += String.format( "%.5f", schoolDists[0] ); - for ( int i=1; i < childIds.length; i++ ) { - childIdString += "," + childIds[i]; - childPnumString += "," + childPnums[i]; - schoolString += "," + schoolMazs[i]; - distsString += "," + String.format( "%.5f", schoolDists[i] ); - } - } - childIdString += "]"; - childPnumString += "]"; - schoolString += "]"; - distsString += "]"; - - String outputString = - id + "," + - dir + "," + - alt + "," + - bundle + "," + - type + "," + - chaufId + "," + - chaufPnum + "," + - chaufPersType + "," + - chaufPid + "," + - departHome + "," + - arriveHome + "," + - departWork + "," + - arriveWork + "," + - childIdString + "," + - childPnumString + "," + - schoolString + "," + - distsString; - - return outputString; - - } - - public void logBundle(Logger logger){ - - String childIdString = "["; - String childPnumString = "["; - String schoolString = "["; - String distsString = "["; - if ( childIds.length > 0 ) { - childIdString += childIds[0]; - childPnumString += childPnums[0]; - schoolString += schoolMazs[0]; - distsString += String.format( "%.5f", schoolDists[0] ); - for ( int i=1; i < childIds.length; i++ ) { - childIdString += "," + childIds[i]; - childPnumString += "," + childPnums[i]; - schoolString += "," + schoolMazs[i]; - distsString += "," + String.format( "%.5f", schoolDists[i] ); - } - } - childIdString += "]"; - childPnumString += "]"; - schoolString += "]"; - distsString += "]"; - logger.info("***********************************************"); - logger.info("id = " + id); - logger.info("dir = " + (dir == SchoolEscortingModel.DIR_OUTBOUND ? "outbound" : "inbound" ) ); - logger.info("alt = " + alt ); - logger.info("bundle = " + bundle ); - logger.info("chaufPnum = " + chaufPnum ); - logger.info("chaufPid = " + chaufPid ); - logger.info("chaufPtype = " + chaufPersType ); - logger.info("escort type = " + (type == ModelStructure.RIDE_SHARING_TYPE ? "ride sharing" : "pure escort" ) ); - logger.info("childIds = " + childIdString ); - logger.info("childPnums = " + childPnumString ); - logger.info("schoolMazs = " + schoolString ); - logger.info("schoolDists = " + distsString ); - logger.info("departHome = " + departHome ); - logger.info("arriveHome = " + arriveHome ); - logger.info("departWork = " + departWork ); - logger.info("arriveWork = " + arriveWork ); - logger.info("***********************************************"); - - - } - -/* - public static SchoolEscortingBundle restoreSchoolEscortingBundleFromExportString( String exportString ) throws Exception { - - StringTokenizer st = new StringTokenizer( exportString, "," ); - - String stringValue = st.nextToken().trim(); - int idValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int dirValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int altValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int bundleValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int typeValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int chaufIdValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int chaufPnumValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int chaufPtypeValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int chaufPidValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int departHomeValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int arriveHomeValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int departWorkValue = Integer.parseInt( stringValue ); - - stringValue = st.nextToken().trim(); - int arriveWorkValue = Integer.parseInt( stringValue ); - - int startCharIndex = exportString.indexOf("[") + 1; - int endCharIndex = exportString.indexOf("]"); - String valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); - int[] childIdValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); - Integer.par - startCharIndex = exportString.indexOf("[", endCharIndex) + 1; - endCharIndex = exportString.indexOf("]", startCharIndex); - valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); - int[] childPnumValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); - - startCharIndex = exportString.indexOf("[", endCharIndex) + 1; - endCharIndex = exportString.indexOf("]", startCharIndex); - valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); - int[] schoolMazsValues = Parsing.getOneDimensionalIntArrayValuesFromExportString( valuesOnlyString ); - - startCharIndex = exportString.indexOf("[", endCharIndex) + 1; - endCharIndex = exportString.indexOf("]", startCharIndex); - valuesOnlyString = exportString.substring( startCharIndex, endCharIndex ); - float[] schoolDistValues = Parsing.getOneDimensionalFloatArrayValuesFromExportString( valuesOnlyString ); - - - SchoolEscortingBundle newBundle = new SchoolEscortingBundle( altValue, bundleValue, chaufIdValue, typeValue, childIdValues, childPnumValues ); - newBundle.setId( idValue ); - newBundle.setDir( dirValue ); - newBundle.setChaufPnum( chaufPnumValue ); - newBundle.setChaufPersType( chaufPtypeValue ); - newBundle.setChaufPid( chaufPidValue ); - newBundle.setSchoolMazs( schoolMazsValues ); - newBundle.setSchoolDists( schoolDistValues ); - newBundle.setDepartHome( departHomeValue ); - newBundle.setArriveHome( arriveHomeValue ); - newBundle.setDepartWork( departWorkValue ); - newBundle.setArriveWork( arriveWorkValue ); - - return newBundle; - - } - */ -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java deleted file mode 100644 index f8c76fb..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingDmu.java +++ /dev/null @@ -1,1716 +0,0 @@ -/* -* The school-escort model was designed by PB (Gupta, Vovsha, et al) -* as part of the Maricopa Association of Governments (MAG) -* Activity-based Travel Model Development project. -* -* This source code, which implements the school escort model, -* was written exclusively for and funded by MAG as part of the -* same project; therefore, per their contract, the -* source code belongs to MAG and can only be used with their -* permission. -* -* It is being adapted for the Southern Oregon ABM by PB & RSG -* with permission from MAG and all references to -* the school escort model as well as source code adapted from this -* original code should credit MAG's role in its development. -* -* The escort model and source code should not be transferred to or -* adapted for other agencies or used in other projects without -* expressed permission from MAG. -*/ - -package org.sandag.abm.ctramp; - -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.matrix.Matrix; -import com.pb.common.util.IndexSort; - - - - -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; - - - -public class SchoolEscortingDmu implements VariableTable { - - private Logger logger = Logger.getLogger( SchoolEscortingDmu.class ); - - private static final float DROP_OFF_DURATION = 5.0f; - private static final float PICK_UP_DURATION = 10.0f; - private static final float MINUTES_PER_MILE = 2.0f; - - - private Household hhObj; - - private float[] distHomeSchool; - private float[] timeHomeSchool; - private float[] distSchoolHome; - private float[] timeSchoolHome; - - //for each cheaffeur - private float[] distHomeMandatory; - private float[] timeHomeMandatory; - private float[] distMandatoryHome; - private float[] timeMandatoryHome; - - private float[][] distSchoolSchool; - private float[][] distSchoolMandatory; - private float[][] distMandatorySchool; - - private Person[] escortees; - - - private int[] escorteeIds; - private int[] escorteePnums; - private int[] escorteeAge; - - private int[] escorteeSchoolLoc; - private int[] escorteeSchoolAtHome; - private int[] escorteeDepartForSchool; - private int[] escorteeDepartFromSchool; - private int numChildrenTravelingToSchool; - - private Person[] chauffers; - private int[] chauffeurPnums; - private int[] chauffeurPids; - private int[] chauffeurAge; - private int[] chauffeurGender; - private int[] chauffeurPersonType; - private int[] chauffeurDap; - private int[] chauffeurMandatoryLoc; - private int[] chauffeurDepartForMandatory; - private int[] chauffeurDepartFromMandatory; - private int numPotentialChauffeurs; - - private int[][][] chaufExtents; - - private int chosenObEscortType1; - private int chosenObEscortType2; - private int chosenObEscortType3; - private int chosenObChauf1; - private int chosenObChauf2; - private int chosenObChauf3; - private int potentialObChauf1; - private int potentialObChauf2; - - private int chosenIbEscortType1; - private int chosenIbEscortType2; - private int chosenIbEscortType3; - private int chosenIbChauf1; - private int chosenIbChauf2; - private int chosenIbChauf3; - private int potentialIbChauf1; - private int potentialIbChauf2; - - private MgraDataManager mgraDataManager; - - private double[][] distanceArray; - - private int[][] altBundleIncidence; - - private Map methodIndexMap; - - - - /** - * Create the DMU by passing in... - * @param MgraDataManager mgraDataManager - * @param Matrix distanceMatrix - */ - public SchoolEscortingDmu(MgraDataManager mgraDataManager, double[][] distanceArray) { - - this.mgraDataManager = mgraDataManager; - this.distanceArray = distanceArray; - - chauffeurPnums = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurPids = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurAge = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurGender = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurPersonType = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurDap = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurMandatoryLoc = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurDepartForMandatory = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - chauffeurDepartFromMandatory = new int[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - distHomeMandatory = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - timeHomeMandatory = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - distMandatoryHome = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - timeMandatoryHome = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - - escorteeIds = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteePnums = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteeAge = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteeSchoolLoc = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteeSchoolAtHome = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteeDepartForSchool = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - escorteeDepartFromSchool = new int[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - distHomeSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - timeHomeSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - distSchoolHome = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - timeSchoolHome = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - - distSchoolSchool = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1][SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - distSchoolMandatory = new float[SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1][SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1]; - distMandatorySchool = new float[SchoolEscortingModel.NUM_CHAUFFEURS_PER_HH+1][SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1]; - - setupMethodIndexMap(); - } - - - public void setAltTableBundleIncidence( int[][] altBundleIncidence ) { - - this.altBundleIncidence = altBundleIncidence; - - } - - /** - * Set attributes of the potential chauffeurs - * @param numPotential Number of potential chauffeurs (size of adults array) - * @param adults Ordered array of chauffeurs - * @param mandatoryMazs Array size of chauffeurs, holding MAZ of last mandatory tour - * @param mandatoryDeparts Array size of chauffeurs, holding home departure period of last mandatory tour - * @param mandatoryReturns Array size of chauffeurs, holding work departure period of last mandatory tour - * @param chaufExtents Array size of chauffeurs, - */ - public void setChaufferAttributes( int numPotential, Person[] adults, int[] mandatoryMazs, int[] mandatoryDeparts, int[] mandatoryReturns, int[][][] chaufExtents ) { - - this.chaufExtents = chaufExtents; - - chauffers = adults; - numPotentialChauffeurs = numPotential; - - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) { - chauffeurAge[i] = 0; - chauffeurGender[i] = 0; - chauffeurPersonType[i] = 0; - chauffeurDap[i] = 0; - chauffeurMandatoryLoc[i] = 0; - chauffeurDepartForMandatory[i] = 0; - chauffeurDepartFromMandatory[i] = 0; - chauffeurPnums[i] = 0; - chauffeurPids[i] = 0; - } - else { - chauffeurPnums[i] = chauffers[i].getPersonNum(); - chauffeurPids[i] = chauffers[i].getPersonId(); - chauffeurAge[i] = chauffers[i].getAge(); - chauffeurGender[i] = chauffers[i].getGender(); - chauffeurPersonType[i] = chauffers[i].getPersonTypeNumber(); - chauffeurDap[i] = chauffers[i].getCdapIndex(); - if ( mandatoryMazs[i] == 0 ) { - chauffeurMandatoryLoc[i] = 0; - chauffeurDepartForMandatory[i] = 0; - chauffeurDepartFromMandatory[i] = 0; - } - else { - chauffeurMandatoryLoc[i] = mandatoryMazs[i]; - chauffeurDepartForMandatory[i] = mandatoryDeparts[i]; - chauffeurDepartFromMandatory[i] = mandatoryReturns[i]; - } - } - } - } - - /** - * Set attributes for escortees. - * - * @param numPotential Number of potential escortees (children traveling to school). - * @param children Person array of potential escortees - * @param schoolAtHome An array for each person indicating if they are schooled at home - * @param schoolMazs An array of school MAZs for each person - * @param schoolDeparts An array of school tour outbound periods - * @param schoolReturns an array of school tour return periods - */ - public void setEscorteeAttributes( int numPotential, Person[] children, int[] schoolAtHome, int[] schoolMazs, int[] schoolDeparts, int[] schoolReturns ) { - - escortees = children; - numChildrenTravelingToSchool = numPotential; - - for ( int i=1; i < escortees.length; i++ ) { - if ( escortees[i] == null || schoolMazs[i] == 0 ) { - escorteeIds[i] = 0; - escorteePnums[i] = 0; - escorteeAge[i] = 0; - escorteeSchoolLoc[i] = 0; - escorteeSchoolAtHome[i] = 0; - escorteeDepartForSchool[i] = 0; - escorteeDepartFromSchool[i] = 0; - } - else { - escorteeIds[i] = i; - escorteePnums[i] = escortees[i].getPersonNum(); - escorteeAge[i] = escortees[i].getAge(); - escorteeSchoolLoc[i] = schoolMazs[i]; - escorteeSchoolAtHome[i] = schoolAtHome[i]; - escorteeDepartForSchool[i] = schoolDeparts[i]; - escorteeDepartFromSchool[i] = schoolReturns[i]; - } - - } - - } - - /** - * Sets distance time attributes for combinations of chauffeur mandatory locations and escortee school locations. - * @param hhObj - * @param distanceArray - */ - public void setDistanceTimeAttributes( Household hhObj, double[][] distanceArray ) { - - this.hhObj = hhObj; - - int homeMaz = hhObj.getHhTaz(); - int homeTaz = mgraDataManager.getTaz(homeMaz); - - // compute times and distances from "home to work" and from "work to home" by traversing the chain of business locations for work tours - // to/from the primary work location for work tours and the chain that may include a work location for school tours. - - //for each chauffeur - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) { - distHomeMandatory[i] = 0; - timeHomeMandatory[i] = 0; - distMandatoryHome[i] = 0; - timeMandatoryHome[i] = 0; - } - else { - if ( chauffeurMandatoryLoc[i] > 0 ) { - - distHomeMandatory[i] = 0; - timeHomeMandatory[i] = 0; - distMandatoryHome[i] = 0; - timeMandatoryHome[i] = 0; - - //the MAG model would traverse all the activities on the tour, skipping non-work and non-school tours, and skipping - //non-work activities, and sum up the distance from home to each work activity. In the case of ORRAMP, only the work - //primary destination is known, so the method has been re-written accordingly to use work primary destination for workers. - - int mandatoryMaz = chauffeurMandatoryLoc[i]; - int mandatoryTaz = mgraDataManager.getTaz(mandatoryMaz); - - distHomeMandatory[i] = (float) distanceArray[homeTaz][ mandatoryTaz]; - timeHomeMandatory[i] = MINUTES_PER_MILE * (float) distanceArray[homeTaz][mandatoryTaz]; - distMandatoryHome[i] = (float) distanceArray[mandatoryTaz][homeTaz]; - timeMandatoryHome[i] = MINUTES_PER_MILE * (float) distanceArray[mandatoryTaz][homeTaz]; - - } - else { - distHomeMandatory[i] = 0; - timeHomeMandatory[i] = 0; - distMandatoryHome[i] = 0; - timeMandatoryHome[i] = 0; - } - } - } - - //iterating through potential escortees (i) - for ( int i=1; i < escortees.length; i++ ) { - if ( escortees[i] == null ) { - distHomeSchool[i] = 0; - timeHomeSchool[i] = 0; - distSchoolHome[i] = 0; - timeSchoolHome[i] = 0; - for ( int j=1; j < chauffeurPnums.length; j++ ) { - distSchoolMandatory[i][j] = 0; - distMandatorySchool[j][i] = 0; - } - for ( int j=1; j < escorteePnums.length; j++ ) - distSchoolSchool[i][j] = 0; - } - else { - if ( escorteeSchoolLoc[i] > 0 ) { - - int schoolTaz = mgraDataManager.getTaz(escorteeSchoolLoc[i]); - - distHomeSchool[i] = (float) distanceArray[homeTaz][schoolTaz]; - timeHomeSchool[i] = MINUTES_PER_MILE * (float) distanceArray[homeTaz][schoolTaz]; - distSchoolHome[i] = (float) distanceArray[schoolTaz][ homeTaz]; - timeSchoolHome[i] = MINUTES_PER_MILE * (float) distanceArray[schoolTaz][homeTaz]; - - //iterating through potential chauffeurs (j) - for ( int j=1; j < chauffeurPnums.length; j++ ) { - distSchoolMandatory[i][j] = 0; - distMandatorySchool[j][i] = 0; - if ( chauffeurMandatoryLoc[j] > 0 ) { - int mandatoryMaz = chauffeurMandatoryLoc[j]; - int mandatoryTaz = mgraDataManager.getTaz(mandatoryMaz); - distSchoolMandatory[i][j] = (float) distanceArray[schoolTaz][mandatoryTaz]; - distMandatorySchool[j][i] = (float) distanceArray[mandatoryTaz][schoolTaz]; - } - } - - for ( int j=1; j < escorteePnums.length; j++ ) { - distSchoolSchool[i][j] = 0; - if ( escorteeSchoolLoc[j] > 0 && escorteeSchoolLoc[j] != escorteeSchoolLoc[i] ) { - int schoolTazJ = mgraDataManager.getTaz(escorteeSchoolLoc[j]); - distSchoolSchool[i][j] = (float) distanceArray[schoolTaz][schoolTazJ]; - } - } - } - else { - distHomeSchool[i] = 0; - timeHomeSchool[i] = 0; - distSchoolHome[i] = 0; - timeSchoolHome[i] = 0; - for ( int j=1; j < chauffeurPnums.length; j++ ) { - distSchoolMandatory[i][j] = 0; - distMandatorySchool[j][i] = 0; - } - for ( int j=1; j < escorteePnums.length; j++ ) - distSchoolSchool[i][j] = 0; - } - } - - } - - } - - - public void setOutboundEscortType1( int chosenObEscortType ) { - chosenObEscortType1 = chosenObEscortType; - } - - public void setOutboundEscortType2( int chosenObEscortType ) { - chosenObEscortType2 = chosenObEscortType; - } - - public void setOutboundEscortType3( int chosenObEscortType ) { - chosenObEscortType3 = chosenObEscortType; - } - - public void setOutboundChauffeur1( int chosenObChauf ) { - chosenObChauf1 = chosenObChauf; - } - - public void setOutboundChauffeur2( int chosenObChauf ) { - chosenObChauf2 = chosenObChauf; - } - - public void setOutboundChauffeur3( int chosenObChauf ) { - chosenObChauf3 = chosenObChauf; - } - - public void setOutboundPotentialChauffeur1( int chaufPnum ) { - potentialObChauf1 = chaufPnum; - } - - public void setOutboundPotentialChauffeur2( int chaufPnum ) { - potentialObChauf2 = chaufPnum; - } - - - - public void setInboundEscortType1( int chosenIbEscortType ) { - chosenIbEscortType1 = chosenIbEscortType; - } - - public void setInboundEscortType2( int chosenIbEscortType ) { - chosenIbEscortType2 = chosenIbEscortType; - } - - public void setInboundEscortType3( int chosenIbEscortType ) { - chosenIbEscortType3 = chosenIbEscortType; - } - - public void setInboundChauffeur1( int chosenIbChauf ) { - chosenIbChauf1 = chosenIbChauf; - } - - public void setInboundChauffeur2( int chosenIbChauf ) { - chosenIbChauf2 = chosenIbChauf; - } - - public void setInboundChauffeur3( int chosenIbChauf ) { - chosenIbChauf3 = chosenIbChauf; - } - - public void setInboundPotentialChauffeur1( int chaufPnum ) { - potentialIbChauf1 = chaufPnum; - } - - public void setInboundPotentialChauffeur2( int chaufPnum ) { - potentialIbChauf2 = chaufPnum; - } - - public int[] getChauffeurPnums() { - return chauffeurPnums; - } - - public int[] getChauffeurDepartForMandatory() { - return chauffeurDepartForMandatory; - } - - public int[] getChauffeurDepartFromMandatory() { - return chauffeurDepartFromMandatory; - } - - public int[] getEscorteePnums() { - return escorteePnums; - } - - public int[] getEscorteeDepartForSchool() { - return escorteeDepartForSchool; - } - - public int[] getEscorteeDepartFromSchool() { - return escorteeDepartFromSchool; - } - - public int[] getEscorteeSchoolAtHome() { - return escorteeSchoolAtHome; - } - - public int[] getEscorteeDistToSchool() { - int[] tempDist = new int[distHomeSchool.length]; - for ( int i=1; i < distHomeSchool.length; i++ ) - tempDist[i] = (int)(distHomeSchool[i] * 100); - return tempDist; - } - - public int[] getEscorteeDistFromSchool() { - int[] tempDist = new int[distSchoolHome.length]; - for ( int i=1; i < distSchoolHome.length; i++ ) - tempDist[i] = (int)(distSchoolHome[i] * 100); - return tempDist; - } - - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put( "getChild1Pnum", 1 ); - methodIndexMap.put( "getChild2Pnum", 2 ); - methodIndexMap.put( "getChild3Pnum", 3 ); - methodIndexMap.put( "getAdult1Pnum", 4 ); - methodIndexMap.put( "getAdult2Pnum", 5 ); - methodIndexMap.put( "getAgeChild1", 6 ); - methodIndexMap.put( "getAgeChild2", 7 ); - methodIndexMap.put( "getAgeChild3", 8 ); - methodIndexMap.put( "getSchoolMazChild1", 9 ); - methodIndexMap.put( "getSchoolMazChild2", 10 ); - methodIndexMap.put( "getSchoolMazChild3", 11 ); - methodIndexMap.put( "getDistHomeSchool1", 12 ); - methodIndexMap.put( "getDistHomeSchool2", 13 ); - methodIndexMap.put( "getDistHomeSchool3", 14 ); - methodIndexMap.put( "getDistSchoolHome1", 15 ); - methodIndexMap.put( "getDistSchoolHome2", 16 ); - methodIndexMap.put( "getDistSchoolHome3", 17 ); - methodIndexMap.put( "getTimeHomeSchool1", 18 ); - methodIndexMap.put( "getTimeHomeSchool2", 19 ); - methodIndexMap.put( "getTimeHomeSchool3", 20 ); - methodIndexMap.put( "getTimeSchoolHome1", 21 ); - methodIndexMap.put( "getTimeSchoolHome2", 22 ); - methodIndexMap.put( "getTimeSchoolHome3", 23 ); - methodIndexMap.put( "getDepartHomeSchool1", 24 ); - methodIndexMap.put( "getDepartHomeSchool2", 25 ); - methodIndexMap.put( "getDepartHomeSchool3", 26 ); - methodIndexMap.put( "getDepartSchoolHome1", 27 ); - methodIndexMap.put( "getDepartSchoolHome2", 28 ); - methodIndexMap.put( "getDepartSchoolHome3", 29 ); - methodIndexMap.put( "getGenderAdult1", 30 ); - methodIndexMap.put( "getGenderAdult2", 31 ); - methodIndexMap.put( "getPersonTypeAdult1", 32 ); - methodIndexMap.put( "getPersonTypeAdult2", 33 ); - methodIndexMap.put( "getAgeAdult1", 34 ); - methodIndexMap.put( "getAgeAdult2", 35 ); - methodIndexMap.put( "getDepartHomeWorkAdult1", 36 ); - methodIndexMap.put( "getDepartHomeWorkAdult2", 37 ); - methodIndexMap.put( "getDepartWorkHomeAdult1", 38 ); - methodIndexMap.put( "getDepartWorkHomeAdult2", 39 ); - methodIndexMap.put( "getDapAdult1", 40 ); - methodIndexMap.put( "getDapAdult2", 41 ); - methodIndexMap.put( "getDistHomeWork1", 42 ); - methodIndexMap.put( "getDistHomeWork2", 43 ); - methodIndexMap.put( "getTimeHomeWork1", 44 ); - methodIndexMap.put( "getTimeHomeWork2", 45 ); - methodIndexMap.put( "getDistWorkHome1", 46 ); - methodIndexMap.put( "getDistWorkHome2", 47 ); - methodIndexMap.put( "getTimeWorkHome1", 48 ); - methodIndexMap.put( "getTimeWorkHome2", 49 ); - methodIndexMap.put( "getDistSchool1School2", 50 ); - methodIndexMap.put( "getDistSchool1School3", 51 ); - methodIndexMap.put( "getDistSchool2School3", 52 ); - methodIndexMap.put( "getDistSchool1Work1", 53 ); - methodIndexMap.put( "getDistSchool1Work2", 54 ); - methodIndexMap.put( "getDistSchool2Work1", 55 ); - methodIndexMap.put( "getDistSchool2Work2", 56 ); - methodIndexMap.put( "getDistSchool3Work1", 57 ); - methodIndexMap.put( "getDistSchool3Work2", 58 ); - methodIndexMap.put( "getDistWork1School1", 59 ); - methodIndexMap.put( "getDistWork2School1", 60 ); - methodIndexMap.put( "getDistWork1School2", 61 ); - methodIndexMap.put( "getDistWork2School2", 62 ); - methodIndexMap.put( "getDistWork1School3", 63 ); - methodIndexMap.put( "getDistWork2School3", 64 ); - methodIndexMap.put( "getIncome", 65 ); - methodIndexMap.put( "getNumAutosInHH", 66 ); - methodIndexMap.put( "getNumWorkersInHH", 67 ); - methodIndexMap.put( "getNumChildrenWithSchoolOutsideOfHomeAndDap1", 68 ); - methodIndexMap.put( "getNumAdultsinHHDap12", 69 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild1Chauffeur1", 70 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild1Chauffeur2", 71 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild2Chauffeur1", 72 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild2Chauffeur2", 73 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild3Chauffeur1", 74 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild3Chauffeur2", 75 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild12Chauffeur1", 76 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild12Chauffeur2", 77 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild13Chauffeur1", 78 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild13Chauffeur2", 79 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild23Chauffeur1", 80 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild23Chauffeur2", 81 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild123Chauffeur1", 82 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceOutboundChild123Chauffeur2", 83 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild1Chauffeur1", 84 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild1Chauffeur2", 85 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild2Chauffeur1", 86 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild2Chauffeur2", 87 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild3Chauffeur1", 88 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild3Chauffeur2", 89 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild12Chauffeur1", 90 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild12Chauffeur2", 91 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild13Chauffeur1", 92 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild13Chauffeur2", 93 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild23Chauffeur1", 94 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild23Chauffeur2", 95 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild123Chauffeur1", 96 ); - methodIndexMap.put( "getAbsoluteDeviationDistanceInboundChild123Chauffeur2", 97 ); - methodIndexMap.put( "getInboundEscortType1", 98 ); - methodIndexMap.put( "getInboundEscortType2", 99 ); - methodIndexMap.put( "getInboundEscortType3", 100 ); - methodIndexMap.put( "getInboundChauffeur1", 101 ); - methodIndexMap.put( "getInboundChauffeur2", 102 ); - methodIndexMap.put( "getInboundChauffeur3", 103 ); - methodIndexMap.put( "getOutboundEscortType1", 104 ); - methodIndexMap.put( "getOutboundEscortType2", 105 ); - methodIndexMap.put( "getOutboundEscortType3", 106 ); - methodIndexMap.put( "getOutboundChauffeur1", 107 ); - methodIndexMap.put( "getOutboundChauffeur2", 108 ); - methodIndexMap.put( "getOutboundChauffeur3", 109 ); - methodIndexMap.put( "getInboundPotentialChauffeur1", 110 ); - methodIndexMap.put( "getInboundPotentialChauffeur2", 111 ); - methodIndexMap.put( "getOutboundPotentialChauffeur1", 112 ); - methodIndexMap.put( "getOutboundPotentialChauffeur2", 113 ); - methodIndexMap.put( "getTravelTimeWork1School1", 114 ); - methodIndexMap.put( "getTravelTimeWork2School1", 115 ); - methodIndexMap.put( "getTravelTimeWork1School2", 116 ); - methodIndexMap.put( "getTravelTimeWork2School2", 117 ); - methodIndexMap.put( "getTravelTimeWork1School3", 118 ); - methodIndexMap.put( "getTravelTimeWork2School3", 119 ); - methodIndexMap.put( "getTravelTimeWork1Home", 120 ); - methodIndexMap.put( "getTravelTimeWork2Home", 121 ); - methodIndexMap.put( "getAvailabilityForMultipleBundlesOutbound", 122 ); - methodIndexMap.put( "getAvailabilityForMultipleBundlesInbound", 123 ); - methodIndexMap.put( "getAvailabilityForInboundChauf1WithOutboundBundles", 124); - methodIndexMap.put( "getAvailabilityForInboundChauf2WithOutboundBundles", 125); - methodIndexMap.put( "getTravelTimeHomeSchool1", 126 ); - methodIndexMap.put( "getTravelTimeHomeSchool2", 127 ); - methodIndexMap.put( "getTravelTimeHomeSchool3", 128 ); - methodIndexMap.put( "getTravelTimeSchool1Home", 129 ); - methodIndexMap.put( "getTravelTimeSchool2Home", 130 ); - methodIndexMap.put( "getTravelTimeSchool3Home", 131 ); - methodIndexMap.put( "getAvailabilityForOutboundChauf1WithInboundBundles", 132); - methodIndexMap.put( "getAvailabilityForOutboundChauf2WithInboundBundles", 133); - - } - - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return escorteePnums[1]; - case 2: - return escorteePnums[2]; - case 3: - return escorteePnums[3]; - case 4: - return chauffeurPnums[1]; - case 5: - return chauffeurPnums[2]; - case 6: - return escorteeAge[1]; - case 7: - return escorteeAge[2]; - case 8: - return escorteeAge[3]; - case 9: - return escorteeSchoolLoc[1]; - case 10: - return escorteeSchoolLoc[2]; - case 11: - return escorteeSchoolLoc[3]; - case 12: - return distHomeSchool[1]; - case 13: - return distHomeSchool[2]; - case 14: - return distHomeSchool[3]; - case 15: - return distSchoolHome[1]; - case 16: - return distSchoolHome[2]; - case 17: - return distSchoolHome[3]; - case 18: - return timeHomeSchool[1]; - case 19: - return timeHomeSchool[2]; - case 20: - return timeHomeSchool[3]; - case 21: - return timeSchoolHome[1]; - case 22: - return timeSchoolHome[2]; - case 23: - return timeSchoolHome[3]; - case 24: - return escorteeDepartForSchool[1]; - case 25: - return escorteeDepartForSchool[2]; - case 26: - return escorteeDepartForSchool[3]; - case 27: - return escorteeDepartFromSchool[1]; - case 28: - return escorteeDepartFromSchool[2]; - case 29: - return escorteeDepartFromSchool[3]; - case 30: - return chauffeurGender[1]; - case 31: - return chauffeurGender[2]; - case 32: - return chauffeurPersonType[1]; - case 33: - return chauffeurPersonType[2]; - case 34: - return chauffeurAge[1]; - case 35: - return chauffeurAge[2]; - case 36: - return chauffeurDepartForMandatory[1]; - case 37: - return chauffeurDepartForMandatory[2]; - case 38: - return chauffeurDepartFromMandatory[1]; - case 39: - return chauffeurDepartFromMandatory[2]; - case 40: - return chauffeurDap[1]; - case 41: - return chauffeurDap[2]; - case 42: - return distHomeMandatory[1]; - case 43: - return distHomeMandatory[2]; - case 44: - return timeHomeMandatory[1]; - case 45: - return timeHomeMandatory[2]; - case 46: - return distMandatoryHome[1]; - case 47: - return distMandatoryHome[2]; - case 48: - return timeMandatoryHome[1]; - case 49: - return timeMandatoryHome[2]; - case 50: - return distSchoolSchool[1][2]; - case 51: - return distSchoolSchool[1][3]; - case 52: - return distSchoolSchool[2][3]; - case 53: - return distSchoolMandatory[1][1]; - case 54: - return distSchoolMandatory[1][2]; - case 55: - return distSchoolMandatory[2][1]; - case 56: - return distSchoolMandatory[2][2]; - case 57: - return distSchoolMandatory[3][1]; - case 58: - return distSchoolMandatory[3][2]; - case 59: - return distMandatorySchool[1][1]; - case 60: - return distMandatorySchool[1][2]; - case 61: - return distMandatorySchool[1][3]; - case 62: - return distMandatorySchool[2][1]; - case 63: - return distMandatorySchool[2][2]; - case 64: - return distMandatorySchool[2][3]; - case 65: - return hhObj.getIncomeInDollars(); - case 66: - return hhObj.getAutosOwned(); - case 67: - return hhObj.getWorkers(); - case 68: - return numChildrenTravelingToSchool; - case 69: - return numPotentialChauffeurs; - case 70: - return Math.max( distHomeSchool[1] + distSchoolMandatory[1][1] - distHomeMandatory[1], 0 ); - case 71: - return Math.max( distHomeSchool[1] + distSchoolMandatory[1][2] - distHomeMandatory[2], 0 ); - case 72: - return Math.max( distHomeSchool[2] + distSchoolMandatory[2][1] - distHomeMandatory[1], 0 ); - case 73: - return Math.max( distHomeSchool[2] + distSchoolMandatory[2][2] - distHomeMandatory[2], 0 ); - case 74: - return Math.max( distHomeSchool[3] + distSchoolMandatory[3][1] - distHomeMandatory[1], 0 ); - case 75: - return Math.max( distHomeSchool[3] + distSchoolMandatory[3][2] - distHomeMandatory[2], 0 ); - case 76: - return getAbsoluteDeviationDistanceOutboundChild12Chauffeur1(); - case 77: - return getAbsoluteDeviationDistanceOutboundChild12Chauffeur2(); - case 78: - return getAbsoluteDeviationDistanceOutboundChild13Chauffeur1(); - case 79: - return getAbsoluteDeviationDistanceOutboundChild13Chauffeur2(); - case 80: - return getAbsoluteDeviationDistanceOutboundChild23Chauffeur1(); - case 81: - return getAbsoluteDeviationDistanceOutboundChild23Chauffeur2(); - case 82: - return getAbsoluteDeviationDistanceOutboundChild123Chauffeur1(); - case 83: - return getAbsoluteDeviationDistanceOutboundChild123Chauffeur2(); - case 84: - return Math.max( distMandatorySchool[1][1] + distSchoolHome[1] - distMandatoryHome[1], 0 ); - case 85: - return Math.max( distMandatorySchool[2][1] + distSchoolHome[1] - distMandatoryHome[2], 0 ); - case 86: - return Math.max( distMandatorySchool[1][2] + distSchoolHome[2] - distMandatoryHome[1], 0 ); - case 87: - return Math.max( distMandatorySchool[2][2] + distSchoolHome[2] - distMandatoryHome[2], 0 ); - case 88: - return Math.max( distMandatorySchool[1][3] + distSchoolHome[3] - distMandatoryHome[1], 0 ); - case 89: - return Math.max( distMandatorySchool[2][3] + distSchoolHome[3] - distMandatoryHome[2], 0 ); - case 90: - return getAbsoluteDeviationDistanceInboundChild12Chauffeur1(); - case 91: - return getAbsoluteDeviationDistanceInboundChild12Chauffeur2(); - case 92: - return getAbsoluteDeviationDistanceInboundChild13Chauffeur1(); - case 93: - return getAbsoluteDeviationDistanceInboundChild13Chauffeur2(); - case 94: - return getAbsoluteDeviationDistanceInboundChild23Chauffeur1(); - case 95: - return getAbsoluteDeviationDistanceInboundChild23Chauffeur2(); - case 96: - return getAbsoluteDeviationDistanceInboundChild123Chauffeur1(); - case 97: - return getAbsoluteDeviationDistanceInboundChild123Chauffeur2(); - case 98: - return chosenIbEscortType1; - case 99: - return chosenIbEscortType2; - case 100: - return chosenIbEscortType3; - case 101: - return chosenIbChauf1; - case 102: - return chosenIbChauf2; - case 103: - return chosenIbChauf3; - case 104: - return chosenObEscortType1; - case 105: - return chosenObEscortType2; - case 106: - return chosenObEscortType3; - case 107: - return chosenObChauf1; - case 108: - return chosenObChauf2; - case 109: - return chosenObChauf3; - case 110: - return potentialIbChauf1; - case 111: - return potentialIbChauf2; - case 112: - return potentialObChauf1; - case 113: - return potentialObChauf2; - case 114: - return (int)( ( ( distMandatorySchool[1][1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 115: - return (int)( ( ( distMandatorySchool[2][1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 116: - return (int)( ( ( distMandatorySchool[1][2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 117: - return (int)( ( ( distMandatorySchool[2][2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 118: - return (int)( ( ( distMandatorySchool[1][3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 119: - return (int)( ( ( distMandatorySchool[2][3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 120: - return (int)( ( ( distMandatoryHome[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 121: - return (int)( ( ( distMandatoryHome[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 122: - return getAvailabilityForMultipleBundlesOutbound( arrayIndex ); - case 123: - return getAvailabilityForMultipleBundlesInbound( arrayIndex ); - case 124: - return getAvailabilityForInboundChauf1WithOutboundBundles( arrayIndex ); - case 125: - return getAvailabilityForInboundChauf2WithOutboundBundles( arrayIndex ); - case 126: - return (int)( ( ( distHomeSchool[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 127: - return (int)( ( ( distHomeSchool[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 128: - return (int)( ( ( distHomeSchool[3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 129: - return (int)( ( ( distSchoolHome[1] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 130: - return (int)( ( ( distSchoolHome[2] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 131: - return (int)( ( ( distSchoolHome[3] * MINUTES_PER_MILE ) / ModelStructure.TOD_INTERVAL_IN_MINUTES ) + 0.99999 ); - case 132: - return getAvailabilityForOutboundChauf1WithInboundBundles( arrayIndex ); - case 133: - return getAvailabilityForOutboundChauf2WithInboundBundles( arrayIndex ); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - - - private float getAbsoluteDeviationDistanceOutboundChild12Chauffeur1() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolMandatory[2][1]; - float d2 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolMandatory[1][1]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceInboundChild12Chauffeur1() { - float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][2] + distSchoolHome[2]; - float d2 = distMandatorySchool[1][2] + distSchoolSchool[2][1] + distSchoolHome[1]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceOutboundChild12Chauffeur2() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolMandatory[2][2]; - float d2 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolMandatory[1][2]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceInboundChild12Chauffeur2() { - float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][2] + distSchoolHome[2]; - float d2 = distMandatorySchool[2][2] + distSchoolSchool[2][1] + distSchoolHome[1]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceOutboundChild13Chauffeur1() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolMandatory[3][1]; - float d2 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolMandatory[1][1]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceInboundChild13Chauffeur1() { - float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[1][3] + distSchoolSchool[3][1] + distSchoolHome[1]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceOutboundChild13Chauffeur2() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolMandatory[3][2]; - float d2 = distHomeSchool[3] + distSchoolSchool[3][1] - distSchoolMandatory[1][2]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceInboundChild13Chauffeur2() { - float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceOutboundChild23Chauffeur1() { - float d1 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolMandatory[3][1]; - float d2 = distHomeSchool[3] + distSchoolSchool[3][2] - distSchoolMandatory[2][1]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceInboundChild23Chauffeur1() { - float d1 = distMandatorySchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; - return Math.min( d1, d2 ) - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceOutboundChild23Chauffeur2() { - float d1 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolMandatory[3][2]; - float d2 = distHomeSchool[3] + distSchoolSchool[3][2] - distSchoolMandatory[2][2]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceInboundChild23Chauffeur2() { - float d1 = distMandatorySchool[2][2] + distSchoolSchool[2][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[2][3] + distSchoolSchool[3][2] + distSchoolHome[2]; - return Math.min( d1, d2 ) - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceOutboundChild123Chauffeur1() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolMandatory[3][1]; - float d2 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolMandatory[2][1]; - float d3 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolMandatory[3][1]; - float d4 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolMandatory[1][1]; - float d5 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolMandatory[2][1]; - float d6 = distHomeSchool[3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolMandatory[1][1]; - float d = Math.min( d1, d2 ); - d = Math.min( d, d3 ); - d = Math.min( d, d4 ); - d = Math.min( d, d5 ); - d = Math.min( d, d6 ); - return d - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceInboundChild123Chauffeur1() { - float d1 = distMandatorySchool[1][1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[1][1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; - float d3 = distMandatorySchool[1][2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; - float d4 = distMandatorySchool[1][2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; - float d5 = distMandatorySchool[1][3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolHome[2]; - float d6 = distMandatorySchool[1][3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolHome[1]; - float d = Math.min( d1, d2 ); - d = Math.min( d, d3 ); - d = Math.min( d, d4 ); - d = Math.min( d, d5 ); - d = Math.min( d, d6 ); - return d - distHomeMandatory[1]; - } - - private float getAbsoluteDeviationDistanceOutboundChild123Chauffeur2() { - float d1 = distHomeSchool[1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolMandatory[3][2]; - float d2 = distHomeSchool[1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolMandatory[2][2]; - float d3 = distHomeSchool[2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolMandatory[3][2]; - float d4 = distHomeSchool[2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolMandatory[1][2]; - float d5 = distHomeSchool[3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolMandatory[2][2]; - float d6 = distHomeSchool[3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolMandatory[1][2]; - float d = Math.min( d1, d2 ); - d = Math.min( d, d3 ); - d = Math.min( d, d4 ); - d = Math.min( d, d5 ); - d = Math.min( d, d6 ); - return d - distHomeMandatory[2]; - } - - private float getAbsoluteDeviationDistanceInboundChild123Chauffeur2() { - float d1 = distMandatorySchool[2][1] + distSchoolSchool[1][2] + distSchoolSchool[2][3] + distSchoolHome[3]; - float d2 = distMandatorySchool[2][1] + distSchoolSchool[1][3] + distSchoolSchool[3][2] + distSchoolHome[2]; - float d3 = distMandatorySchool[2][2] + distSchoolSchool[2][1] + distSchoolSchool[1][3] + distSchoolHome[3]; - float d4 = distMandatorySchool[2][2] + distSchoolSchool[2][3] + distSchoolSchool[3][1] + distSchoolHome[1]; - float d5 = distMandatorySchool[2][3] + distSchoolSchool[3][1] + distSchoolSchool[1][2] + distSchoolHome[2]; - float d6 = distMandatorySchool[2][3] + distSchoolSchool[3][2] + distSchoolSchool[2][1] + distSchoolHome[1]; - float d = Math.min( d1, d2 ); - d = Math.min( d, d3 ); - d = Math.min( d, d4 ); - d = Math.min( d, d5 ); - d = Math.min( d, d6 ); - return d - distHomeMandatory[2]; - } - - - /** - * This method should only be called for relevant alternatives - those with multiple bundles for a single chauffeur. - */ - private int getAvailabilityForMultipleBundlesOutbound( int alt ) { - - // set availability to 0 if unavailable, or 1 if available - int availabilityForMultipleBundles = 1; - - List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); - - //check the number of bundles - int chaufIndex = 0; - if ( altChaufBundles[1].size() > 1 ) //more than one bundle for the first chauffeur - chaufIndex = 1; - else if ( altChaufBundles[2].size() > 1 ) //more than one bundle for the second chauffeur - chaufIndex = 2; - else { - logger.fatal( "UEC method getAvailabilityForMultipleBundlesOutbound( alt=" + alt + " ) was called, but neither chauf has multiple escort bundles." ); - logger.fatal("Size of altChaufBundles[1] = "+altChaufBundles[1].size()); - logger.fatal("Size of altChaufBundles[2] = "+altChaufBundles[2].size()); - throw new RuntimeException( ); - } - - - // set the bundle depart intervals for all bundles and arrive back home intervals for pure escort only - for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { - - int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; - Arrays.fill( sortData, 999999999 ); - int[] children = bundleObj.getChildIds(); - for ( int j=0; j < children.length; j++ ) - sortData[children[j]] = escorteeDepartForSchool[children[j]]; - - int[] childrenOrder = IndexSort.indexSort( sortData ); - - int departHomeInterval = escorteeDepartForSchool[ childrenOrder[0] ]; - bundleObj.setDepartHome( departHomeInterval ); - - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( childrenOrder, children.length, DROP_OFF_DURATION ); - float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); - bundleObj.setArriveHome( arriveHomeInterval ); - } - - } - - - int[] chaufBundlesOrder = getChaufBundlesOrderOutbound( altChaufBundles[chaufIndex] ); - - for( int j=1; j < chaufBundlesOrder.length; j++ ) { - SchoolEscortingBundle bundleSubsequent = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); - SchoolEscortingBundle bundlePrevious = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j-1] ); - if ( bundleSubsequent.getDepartHome() <= bundlePrevious.getArriveHome() ) { - availabilityForMultipleBundles = 0; - break; - } - } - - return availabilityForMultipleBundles; - - } - - float convertIntervalToMinutes(int interval){ - - float minutes = interval * ModelStructure.TOD_INTERVAL_IN_MINUTES; - return minutes; - } - - int convertMinutesToInterval(float minutes){ - - int interval = (int) (minutes/ModelStructure.TOD_INTERVAL_IN_MINUTES); - interval = Math.min(interval,ModelStructure.MAX_TOD_INTERVAL); - return interval; - - } - - - /** - * This method should only be called for relevant alternatives - those with multiple bundles for a single chauffeur. - */ - private int getAvailabilityForMultipleBundlesInbound( int alt ) { - - // set availability to 0 if unavailable, or 1 if available - int availabilityForMultipleBundles = 1; - - List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); - - int chaufIndex = 0; - if ( altChaufBundles[1].size() > 1 ) - chaufIndex = 1; - else if ( altChaufBundles[2].size() > 1 ) - chaufIndex = 2; - else { - logger.error( "UEC method getAvailabilityForMultipleBundlesInbound( alt=" + alt + " ) was called, but neither chauf has multiple escort bundles." ); - throw new RuntimeException( ); - } - - - // set the bundle arrive intervals for all bundles and depart from home intervals for pure escort only - for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { - - int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; - Arrays.fill( sortData, 999999999 ); - int[] children = bundleObj.getChildIds(); - for ( int j=0; j < children.length; j++ ) - sortData[children[j]] = escorteeDepartFromSchool[children[j]]; - - int[] childrenOrder = IndexSort.indexSort( sortData ); - - int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; - - float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); - float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); - bundleObj.setArriveHome( arriveHomeInterval ); - - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[childrenOrder[0]] ); - float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; - int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); - bundleObj.setDepartHome( departHomeInterval ); - } - - } - - - int[] chaufBundlesOrder = getChaufBundlesOrderInbound( altChaufBundles[chaufIndex] ); - - for( int j=1; j < chaufBundlesOrder.length; j++ ) { - SchoolEscortingBundle bundleSubsequent = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); - SchoolEscortingBundle bundlePrevious = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j-1] ); - if ( bundleSubsequent.getDepartHome() <= bundlePrevious.getArriveHome() ) { - availabilityForMultipleBundles = 0; - break; - } - } - - return availabilityForMultipleBundles; - - } - - private int getAvailabilityForInboundChauf1WithOutboundBundles( int alt ) { - return getAvailabilityForChaufWithPreviousDirectionBundles( 1, SchoolEscortingModel.DIR_INBOUND, alt ); - } - - private int getAvailabilityForInboundChauf2WithOutboundBundles( int alt ) { - return getAvailabilityForChaufWithPreviousDirectionBundles( 2, SchoolEscortingModel.DIR_INBOUND, alt ); - } - - private int getAvailabilityForOutboundChauf1WithInboundBundles( int alt ) { - return getAvailabilityForChaufWithPreviousDirectionBundles( 1, SchoolEscortingModel.DIR_OUTBOUND, alt ); - } - - private int getAvailabilityForOutboundChauf2WithInboundBundles( int alt ) { - return getAvailabilityForChaufWithPreviousDirectionBundles( 2, SchoolEscortingModel.DIR_OUTBOUND, alt ); - } - - /** - * This method should only be called for for relevant alternatives - those with multiple bundles for a single chauffeur. - */ - private int getAvailabilityForChaufWithPreviousDirectionBundles( int chaufid, int dir, int alt ) { - - // set availability to 0 if unavailable, or 1 if available - int availability = 1; - - List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); - - // set the bundle depart and arrive intervals for all bundles for the alternative - for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufid] ) { - - // order the children by earliest pickup time and set arriveHome - int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; - Arrays.fill( sortData, 999999999 ); - int[] children = bundleObj.getChildIds(); - for ( int j=0; j < children.length; j++ ) { - sortData[children[j]] = escorteeDepartFromSchool[children[j]]; - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) - sortData[children[j]] = escorteeDepartForSchool[children[j]]; - else - sortData[children[j]] = escorteeDepartFromSchool[children[j]]; - } - - int[] childrenOrder = IndexSort.indexSort( sortData ); - - int chaufPnum = chauffeurPnums[chaufid]; - - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - - // set OB depart to earliest child's depart from home or either pure escort or ride sharing - int departHomeInterval = escorteeDepartForSchool[ childrenOrder[0] ]; - bundleObj.setDepartHome( departHomeInterval ); - - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - - float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( childrenOrder, children.length, DROP_OFF_DURATION ); - float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); - bundleObj.setArriveHome( arriveHomeInterval ); - - // neither end of the alternative window can overlap the reserved window - if ( ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) || - ( arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) ) - availability = 0; - // if the start of the alternative window is before the start of the reserved window, the end of the alternative window must also be before the start of the reserved window. - else if ( departHomeInterval < chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] ) - availability = 0; - // the start of the alternative window cannot be after the start of the reserved window - else if ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] ) - availability = 0; - } - else { - - if ( chauffeurDap[chaufid] != 1 ) { - availability = 0; - } - else { - - float numMinutes = getTimeInMinutesFromHomeThruAllSchoolsToWork( chauffeurMandatoryLoc[chaufid], childrenOrder, children.length, DROP_OFF_DURATION ); - float arriveWorkMinute = convertIntervalToMinutes( departHomeInterval ) + numMinutes; - int arriveWorkInterval = convertMinutesToInterval( arriveWorkMinute ); - - if ( ( departHomeInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) || - ( arriveWorkInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && arriveWorkInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) ) - availability = 0; - } - } - - } - else { - - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - - int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; - - float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[childrenOrder[0]] ); - float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; - int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); - - float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); - float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); - - // neither end of the alternative window can overlap the reserved window - if ( ( departHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) || - ( arriveHomeInterval >= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) ) - availability = 0; - // the start of the alternative window cannot be before the end of the reserved window - else if ( departHomeInterval <= chaufExtents[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] ) - availability = 0; - } - else { - - if ( chauffeurDap[chaufid] != 1 ) { - availability = 0; - } - else { - - int departFromFirstSchoolInterval = escorteeDepartFromSchool[childrenOrder[0]]; - - float workToFirstSchoolMinutes = getMazToMazTimeInMinutes( chauffeurMandatoryLoc[chaufid], escorteeSchoolLoc[childrenOrder[0]] ); - float departWorkMinute = convertIntervalToMinutes( departFromFirstSchoolInterval ) - workToFirstSchoolMinutes; - int departWorkInterval = convertMinutesToInterval( departWorkMinute ); - - float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( childrenOrder, children.length, PICK_UP_DURATION ); - float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); - - if ( ( departWorkInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && departWorkInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) || - ( arriveHomeInterval >= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] && arriveHomeInterval <= chaufExtents[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] ) ) - availability = 0; - } - } - - } - - } - - - return availability; - - } - - - /** - * This method creates and returns the chosen bundles given the attributes of the chauffeur and escortees on the bundle and the type of bundle selected (pure escort vs rideshare) - * and the direction for the bundle (outbound versus return). - * - * @param alt The chosen alternative. - * @param chaufIndex The chauffeur to get the bundle for. - * @param dir Outbound or inbound. - * @return A fully coded bundle for the chauffeur given the chosen alternative. - */ - public SchoolEscortingBundle[] getChosenBundles( int alt, int chaufIndex, int dir ) { - - //an arraylist of school escorting bundles, dimensioned by each chauffeur (2) - List[] altChaufBundles = SchoolEscortingBundle.constructAltBundles( alt, altBundleIncidence ); - - // set the bundle depart intervals for all bundles and arrive back home intervals for pure escort only - for ( SchoolEscortingBundle bundleObj : altChaufBundles[chaufIndex] ) { - - //if the bundle direction is outbound, sort the escortees by departure period, else sort by arrival period - int[] sortData = new int[ SchoolEscortingModel.NUM_ESCORTEES_PER_HH+1 ]; - Arrays.fill( sortData, 999999999 ); - int[] altBundleChildIds = bundleObj.getChildIds(); - for ( int j=0; j < altBundleChildIds.length; j++ ) { - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) - sortData[altBundleChildIds[j]] = escorteeDepartForSchool[altBundleChildIds[j]]; - else - sortData[altBundleChildIds[j]] = escorteeDepartFromSchool[altBundleChildIds[j]]; - } - int[] altBundleChildrenOrder = IndexSort.indexSort( sortData ); - - //set the school locations and person numbers for each escortee in the order of escortees set above - int[] altBundleChildSchools = new int[altBundleChildIds.length]; - int[] altBundleChildPnums = new int[altBundleChildIds.length]; - for ( int j=0; j < altBundleChildIds.length; j++ ) { - int k = altBundleChildrenOrder[j]; - altBundleChildSchools[j] = escorteeSchoolLoc[k]; - altBundleChildPnums[j] = escorteePnums[k]; - } - - //set other elements of the bundle for this choice\household - bundleObj.setDir( dir ); - bundleObj.setChaufPnum( chauffeurPnums[chaufIndex] ); - bundleObj.setChaufPid( chauffeurPids[chaufIndex] ); - bundleObj.setChaufPersType( chauffeurPersonType[chaufIndex] ); - bundleObj.setChildPnums( altBundleChildPnums ); - bundleObj.setSchoolMazs( altBundleChildSchools ); - bundleObj.setWorkOrSchoolMaz( chauffeurMandatoryLoc[chaufIndex] ); - - - //if the bundle is outbound - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - - //get an array of distances to each child's school starting from home. - float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( hhObj.getHhMgra() ), altBundleChildrenOrder, altBundleChildIds.length ); - bundleObj.setSchoolDists( altBundleSchoolDistances ); - - // set OB depart to earliest child's depart from home for either pure escort or ride sharing - int departHomeInterval = escorteeDepartForSchool[ altBundleChildrenOrder[0] ]; - bundleObj.setDepartHome( departHomeInterval ); - - //if the bundle is pure escort, then the total trip time in minutes is from home through all passengers back to home and the arrival time back at home is the departure - //period plus the time and some time for stops. Otherwise the time arriving to work is the time period departing home + travel time through all escortees plus dwell. - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - - float roundTripMinutes = getRoundTripMinutesFromHomeThruAllSchoolsToHome( altBundleChildrenOrder, altBundleChildIds.length, DROP_OFF_DURATION ); - float arriveBackHomeMinute = convertIntervalToMinutes( departHomeInterval ) + roundTripMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveBackHomeMinute ); - bundleObj.setArriveHome( arriveHomeInterval ); - } - else { - float numMinutes = getTimeInMinutesFromHomeThruAllSchoolsToWork( chauffeurMandatoryLoc[chaufIndex], altBundleChildrenOrder, altBundleChildIds.length, DROP_OFF_DURATION ); - float arriveWorkMinute = convertIntervalToMinutes( departHomeInterval ) + numMinutes; - int arriveWorkInterval = convertMinutesToInterval( arriveWorkMinute ); - bundleObj.setArriveWork( arriveWorkInterval ); - } - - } - else { - - //on the return direction, the departure time from the first school is the time for the first child being escorted. - int departFromFirstSchoolInterval = escorteeDepartFromSchool[altBundleChildrenOrder[0]]; - - // if the bundle is a pure escort, set the departure time period from home to the time that the first child departs from school minus the time required to get to school. - if ( bundleObj.getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - float homeToFirstSchoolMinutes = getMazToMazTimeInMinutes( hhObj.getHhMgra(), escorteeSchoolLoc[altBundleChildrenOrder[0]] ); - float departFromHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) - homeToFirstSchoolMinutes; - int departHomeInterval = convertMinutesToInterval( departFromHomeMinutes ); - bundleObj.setDepartHome( departHomeInterval ); - - float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( hhObj.getHhMgra() ), altBundleChildrenOrder, altBundleChildIds.length ); - bundleObj.setSchoolDists( altBundleSchoolDistances ); - } - else { //else if the bundle is rideshare, the time departing work is the time the first child leave school minus the time required to get to school from work. - float workToFirstSchoolMinutes = getMazToMazTimeInMinutes( chauffeurMandatoryLoc[chaufIndex], escorteeSchoolLoc[altBundleChildrenOrder[0]] ); - float departWorkMinute = convertIntervalToMinutes( departFromFirstSchoolInterval ) - workToFirstSchoolMinutes; - int departWorkInterval = convertMinutesToInterval( departWorkMinute ); - bundleObj.setDepartWork( departWorkInterval ); - - float[] altBundleSchoolDistances = getDistancesToSchools( mgraDataManager.getTaz( chauffeurMandatoryLoc[chaufIndex] ), altBundleChildrenOrder, altBundleChildIds.length ); - bundleObj.setSchoolDists( altBundleSchoolDistances ); - } - - //on the return direction, the arrival time back home is the time from the first child's departure time plus the time required to get home plus dwell time for each escortee. - float firstSchoolToHomeMinutes = getMinutesFromFirstSchoolToHome( altBundleChildrenOrder, altBundleChildIds.length, PICK_UP_DURATION ); - float arriveHomeMinutes = convertIntervalToMinutes( departFromFirstSchoolInterval ) + firstSchoolToHomeMinutes; - int arriveHomeInterval = convertMinutesToInterval( arriveHomeMinutes ); - bundleObj.setArriveHome( arriveHomeInterval ); - - bundleObj.setDepartPrimaryInterval( departFromFirstSchoolInterval ); - } - - } - - - int[] chaufBundlesOrder = null; - if ( altChaufBundles[chaufIndex].size() > 1 ) { - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) - chaufBundlesOrder = getChaufBundlesOrderOutbound( altChaufBundles[chaufIndex] ); - else - chaufBundlesOrder = getChaufBundlesOrderInbound( altChaufBundles[chaufIndex] ); - } - else { - chaufBundlesOrder = new int[]{ 0 }; - } - - SchoolEscortingBundle[] result = new SchoolEscortingBundle[chaufBundlesOrder.length]; - for( int j=0; j < chaufBundlesOrder.length; j++ ) - result[j] = altChaufBundles[chaufIndex].get( chaufBundlesOrder[j] ); - - - return result; - - } - - - /** - * Get an array of distances to each child's school. - * - * @param origTaz The originTaz is the origin of the first child's trip (home for outbound direction, primary destination for return direction) - * @param childrenOrder The order of each escortee, can be by departure time - * @param numChildren Number of children - * @return A float array of distances to each child's school. - */ - private float[] getDistancesToSchools( int origTaz, int[] childrenOrder, int numChildren ) { - - float[] distances = new float[numChildren]; - - for ( int j=0; j < numChildren; j++ ) { - int k = childrenOrder[j]; - int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); - distances[j] = (float) distanceArray[origTaz][schoolTaz]; - origTaz = schoolTaz; - } - - return distances; - - } - - - private float getMinutesFromFirstSchoolToHome( int[] childrenOrder, int numChildren, float minDuration ) { - - int homeTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); - int firstSchool = escorteeSchoolLoc[ childrenOrder[0] ]; - int firstSchoolTaz = mgraDataManager.getTaz( firstSchool ); - int originTaz = firstSchoolTaz; - - // distance is the cumulative distance from the school where the child is picked up to home, through other schools . - float distance = 0; - float duration = 0; - - for ( int j=1; j < numChildren; j++ ) { - int k = childrenOrder[j]; - int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); - distance += (float) distanceArray[originTaz][ schoolTaz ]; - originTaz = schoolTaz; - - duration += minDuration; - } - - distance += (float) distanceArray[originTaz][ homeTaz ]; - - float timeInMinutes = distance * MINUTES_PER_MILE + duration; - - return timeInMinutes; - - } - - - private float getMazToMazTimeInMinutes( int fromMaz, int toMaz ) { - - int fromTaz = mgraDataManager.getTaz( fromMaz ); - int toTaz = mgraDataManager.getTaz( toMaz ); - - float timeInMinutes = MINUTES_PER_MILE * (float) distanceArray[fromTaz][ toTaz]; - return timeInMinutes; - - } - - - private float getRoundTripMinutesFromHomeThruAllSchoolsToHome( int[] childrenOrder, int numChildren, float minDuration ) { - - int originTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); - - // cumulative distance and duration - float distance = 0; - float duration = 0; - - for ( int j=0; j < numChildren; j++ ) { - int k = childrenOrder[j]; - int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); - distance += (float) distanceArray[originTaz][ schoolTaz]; - originTaz = schoolTaz ; - - duration += minDuration; - } - int destTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); - distance += (float) distanceArray[originTaz][ destTaz]; - - float timeInMinutes = MINUTES_PER_MILE * distance + duration; - return timeInMinutes; - - } - - - private float getTimeInMinutesFromHomeThruAllSchoolsToWork( int workMaz, int[] childrenOrder, int numChildren, float minDuration ) { - - int homeTaz = mgraDataManager.getTaz( hhObj.getHhMgra() ); - int workTaz = mgraDataManager.getTaz( workMaz ); - - int originTaz = homeTaz; - - float distance = 0; - float duration = 0; - - for ( int j=0; j < numChildren; j++ ) { - int k = childrenOrder[j]; - int schoolTaz = mgraDataManager.getTaz( escorteeSchoolLoc[k] ); - distance += distanceArray[originTaz][ schoolTaz ]; - originTaz = schoolTaz; - duration += minDuration; - } - - distance += (float) distanceArray[originTaz][ workTaz ]; - - float timeInMinutes = MINUTES_PER_MILE * distance + duration; - return timeInMinutes; - - } - - - // Create an array to hold the list indices for the order in which escort activities should be performed. - // This method only gets called while checking availability for a chauffeur to have multiple bundles, so the - // list altChaufBundles has either 2 or 3 escort activities. - private int[] getChaufBundlesOrderOutbound( List altChaufBundles ) { - - int[] chaufBundlesOrder = null; - - // number of escort activities is 2. - if ( altChaufBundles.size() == 2 ) { - // [RS,PE]: if the first activity for the alternative is ride sharing, the second must be pure escort, and should be ordered before the ride sharing escort activity; - if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - chaufBundlesOrder = new int[]{ 1, 0 }; - // [PE,RS]: likewise if the second activity is ride sharing - the first must be pure escort, and must be ordered first - else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - chaufBundlesOrder = new int[]{ 0, 1 }; - // [PE,PE]: otherwise, both activities are pure escort, and the depart times will determine the order - else if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 1, 0 }; - else - chaufBundlesOrder = new int[]{ 0, 1 }; - } - // number of escort activities is 3. - else { - // [RS,PE,PE]: if the first activity for the alternative is ride sharing, the second and third must also be pure escort, and should be ordered before the ride sharing escort activity; - if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 1 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 2, 1, 0 }; - else - chaufBundlesOrder = new int[]{ 1, 2, 0 }; - } - // [PE,RS,PE]: likewise if the second activity is ride sharing - the first and third must be pure escort, and must be ordered before ride sharing - else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 2, 0, 1 }; - else - chaufBundlesOrder = new int[]{ 0, 2, 1 }; - } - // [PE,PE,RS]: likewise if the third activity is ride sharing - the first and second must be pure escort, and must be ordered before ride sharing - else if ( altChaufBundles.get( 2 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 1, 0, 2 }; - else - chaufBundlesOrder = new int[]{ 0, 1, 2 }; - } - // [PE,PE,PE]: otherwise, all three activities are pure escort, and the depart times will determine the order - else { - int[] sortData = new int[]{ altChaufBundles.get( 0 ).getDepartHome(), altChaufBundles.get( 1 ).getDepartHome(), altChaufBundles.get( 2 ).getDepartHome() }; - chaufBundlesOrder = IndexSort.indexSort( sortData ); - } - } - - return chaufBundlesOrder; - - } - - - // Create an array to hold the list indices for the order in which escort activities should be performed. - // This method only gets called while checking availability for a chauffeur to have multiple bundles, so the - // list altChaufBundles has either 2 or 3 escort activities. - private int[] getChaufBundlesOrderInbound( List altChaufBundles ) { - - int[] chaufBundlesOrder = null; - - // number of escort activities is 2. - if ( altChaufBundles.size() == 2 ) { - // [RS,PE]: if the first activity for the alternative is ride sharing, the second must be pure escort, and should be ordered after the ride sharing escort activity; - if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - chaufBundlesOrder = new int[]{ 0, 1 }; - // [PE,RS]: likewise if the second activity is ride sharing - the first must be pure escort, and must be ordered second - else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - chaufBundlesOrder = new int[]{ 1, 0 }; - // [PE,PE]: otherwise, both activities are pure escort, and the depart times will determine the order - else if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 1, 0 }; - else - chaufBundlesOrder = new int[]{ 0, 1 }; - } - // number of escort activities is 3. - else { - // [RS,PE,PE]: if the first activity for the alternative is ride sharing, the second and third must also be pure escort, and should be ordered after the ride sharing escort activity; - if ( altChaufBundles.get( 0 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 1 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 0, 2, 1 }; - else - chaufBundlesOrder = new int[]{ 0, 1, 2 }; - } - // [PE,RS,PE]: likewise if the second activity is ride sharing - the first and third must be pure escort, and must be ordered after ride sharing - else if ( altChaufBundles.get( 1 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 2 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 1, 2, 0 }; - else - chaufBundlesOrder = new int[]{ 1, 0, 2 }; - } - // [PE,PE,RS]: likewise if the third activity is ride sharing - the first and second must be pure escort, and must be ordered after ride sharing - else if ( altChaufBundles.get( 2 ).getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - if ( altChaufBundles.get( 1 ).getDepartHome() < altChaufBundles.get( 0 ).getDepartHome() ) - chaufBundlesOrder = new int[]{ 2, 1, 0 }; - else - chaufBundlesOrder = new int[]{ 2, 0, 1 }; - } - // [PE,PE,PE]: otherwise, all three activities are pure escort, and the depart times will determine the order - else { - int[] sortData = new int[]{ altChaufBundles.get( 0 ).getDepartHome(), altChaufBundles.get( 1 ).getDepartHome(), altChaufBundles.get( 2 ).getDepartHome() }; - chaufBundlesOrder = IndexSort.indexSort( sortData ); - } - } - - return chaufBundlesOrder; - - } - - - - - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java deleted file mode 100644 index a79ed0b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolEscortingModel.java +++ /dev/null @@ -1,2154 +0,0 @@ -/* -* The school-escort model was designed by PB (Gupta, Vovsha, et al) -* as part of the Maricopa Association of Governments (MAG) -* Activity-based Travel Model Development project. -* -* This source code, which implements the school escort model, -* was written exclusively for and funded by MAG as part of the -* same project; therefore, per their contract, the -* source code belongs to MAG and can only be used with their -* permission. -* -* It is being adapted for the Southern Oregon ABM by PB & RSG -* with permission from MAG and all references to -* the school escort model as well as source code adapted from this -* original code should credit MAG's role in its development. -* -* The escort model and source code should not be transferred to or -* adapted for other agencies or used in other projects without -* expressed permission from MAG. -* -* The source code has been substantially revised to fit within the -* SANDAG\MTC\ODOT CT-RAMP model structure by RSG (2015). -*/ - -package org.sandag.abm.ctramp; -import java.util.ArrayList; -import java.util.Collections; -import java.util.Comparator; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Random; -import java.util.Set; -import java.util.TreeSet; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.calculator.IndexValues; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.IndexSort; -import com.pb.common.util.PropertyMap; - - -public class SchoolEscortingModel { - - public static final String ALT_TABLE_BUNDLE1_NAME = "bundle1"; - public static final String ALT_TABLE_BUNDLE2_NAME = "bundle2"; - public static final String ALT_TABLE_BUNDLE3_NAME = "bundle3"; - public static final String ALT_TABLE_CHAUF1_NAME = "chauf1"; - public static final String ALT_TABLE_CHAUF2_NAME = "chauf2"; - public static final String ALT_TABLE_CHAUF3_NAME = "chauf3"; - public static final String ALT_TABLE_NBUNDLES_NAME = "nbundles"; - public static final String ALT_TABLE_NBUNDLES_RS1_NAME = "nrs1"; - public static final String ALT_TABLE_NBUNDLES_ES1_NAME = "npe1"; - public static final String ALT_TABLE_NBUNDLES_RS2_NAME = "nrs2"; - public static final String ALT_TABLE_NBUNDLES_ES2_NAME = "npe2"; - public static final String ALT_TABLE_BUNDLE_INCIDENCE_FIRST_COLUMN_NAME = "b1_rs1_1"; - - - public static final int NUM_ESCORTEES_PER_HH = 3; - public static final int NUM_CHAUFFEURS_PER_HH = 2; - public static final int NUM_ESCORT_TYPES = 2; - public static final int NUM_BUNDLES = 3; - - - public static final int ESCORT_ELIGIBLE = 1; - - public static final int CHAUFFEUR_1 = 1; - public static final int CHAUFFEUR_2 = 2; - - public static final int DIR_OUTBOUND = 1; - public static final int DIR_INBOUND = 2; - - public static final int RIDE_SHARING_CHAUFFEUR_1 = 1; - public static final int PURE_ESCORTING_CHAUFFEUR_1 = 2; - public static final int RIDE_SHARING_CHAUFFEUR_2 = 3; - public static final int PURE_ESCORTING_CHAUFFEUR_2 = 4; - - - - // chauffeur priority lookup values determined by 100*PersonTypesOld(1-8) + 10*gender(1-2) + 1*age > 25 - private static final int PT_WEIGHT = 100; - private static final int G_WEIGHT = 10; - private static final int A_WEIGHT = 1; - - public static final int RESULT_CHILD_DIRECTION_FIELD = 0; - public static final int RESULT_CHILD_CHOSEN_ALT_FIELD = 1; - public static final int RESULT_CHILD_HHID_FIELD = 2; - public static final int RESULT_CHILD_PNUM_FIELD = 3; - public static final int RESULT_CHILD_PID_FIELD = 4; - public static final int RESULT_CHILD_PERSON_TYPE_FIELD = 5; - public static final int RESULT_CHILD_AGE_FIELD = 6; - public static final int RESULT_CHILD_CDAP_FIELD = 7; - public static final int RESULT_CHILD_SCHOOL_AT_HOME_FIELD = 8; - public static final int RESULT_CHILD_SCHOOL_LOC_FIELD = 9; - public static final int RESULT_CHILD_ESCORT_ELIGIBLE_FIELD = 10; - public static final int RESULT_CHILD_DEPART_FROM_HOME_FIELD = 11; - public static final int RESULT_CHILD_DEPART_TO_HOME_FIELD = 12; - public static final int RESULT_CHILD_DIST_TO_SCHOOL_FIELD = 13; - public static final int RESULT_CHILD_DIST_FROM_SCHOOL_FIELD = 14; - public static final int RESULT_CHILD_ADULT1_DEPART_FROM_HOME_FIELD = 15; - public static final int RESULT_CHILD_ADULT1_DEPART_TO_HOME_FIELD = 16; - public static final int RESULT_CHILD_ADULT2_DEPART_FROM_HOME_FIELD = 17; - public static final int RESULT_CHILD_ADULT2_DEPART_TO_HOME_FIELD = 18; - public static final int RESULT_CHILD_ESCORT_TYPE_FIELD = 19; - public static final int RESULT_CHILD_BUNDLE_ID_FIELD = 20; - public static final int RESULT_CHILD_CHILD_ID_FIELD = 21; - public static final int RESULT_CHILD_CHAUFFEUR_ID_FIELD = 22; - public static final int RESULT_CHILD_CHAUFFEUR_PNUM_FIELD = 23; - public static final int RESULT_CHILD_CHAUFFEUR_PID_FIELD = 24; - public static final int RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD = 25; - public static final int RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD = 26; - public static final int RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD = 27; - public static final int RESULT_CHILD_RANDOM_NUM_FIELD = 28; - public static final int NUM_RESULTS_BY_CHILD_FIELDS = 29; - - - public static final int RESULT_CHAUF_BUNDLE_ID_FIELD = 0; - public static final int RESULT_CHAUF_DIRECTION_FIELD = 1; - public static final int RESULT_CHAUF_CHOSEN_ALT_FIELD = 2; - public static final int RESULT_CHAUF_HHID_FIELD = 3; - public static final int RESULT_CHAUF_PNUM_FIELD = 4; - public static final int RESULT_CHAUF_PID_FIELD = 5; - public static final int RESULT_CHAUF_PERSON_TYPE_FIELD = 6; - public static final int RESULT_CHAUF_AGE_FIELD = 7; - public static final int RESULT_CHAUF_CDAP_FIELD = 8; - public static final int RESULT_CHAUF_ESCORT_ELIGIBLE_FIELD = 9; - public static final int RESULT_CHAUF_DEPART_HOME_FIELD = 10; - public static final int RESULT_CHAUF_ID_FIELD = 11; - public static final int RESULT_CHAUF_ESCORT_TYPE_FIELD = 12; - public static final int RESULT_CHAUF_CHILD1_PNUM_FIELD = 13; - public static final int RESULT_CHAUF_CHILD1_PERSON_TYPE_FIELD = 14; - public static final int RESULT_CHAUF_CHILD1_DEPART_HOME_FIELD = 15; - public static final int RESULT_CHAUF_CHILD2_PNUM_FIELD = 16; - public static final int RESULT_CHAUF_CHILD2_PERSON_TYPE_FIELD = 17; - public static final int RESULT_CHAUF_CHILD2_DEPART_HOME_FIELD = 18; - public static final int RESULT_CHAUF_CHILD3_PNUM_FIELD = 19; - public static final int RESULT_CHAUF_CHILD3_PERSON_TYPE_FIELD = 20; - public static final int RESULT_CHAUF_CHILD3_DEPART_HOME_FIELD = 21; - public static final int NUM_RESULTS_BY_CHAUF_FIELDS = 22; - - public static final int DRIVE_ALONE_MODE = 1; - public static final int SHARED_RIDE_2_MODE = 2; - public static final int SHARED_RIDE_3_MODE = 3; - - private Map chauffeurPriorityOutboundMap; - private Map chauffeurPriorityInboundMap; - - - private TableDataSet altTable; - private String[] altTableNames; - private int[] altTableBundle1; - private int[] altTableBundle2; - private int[] altTableBundle3; - private int[] altTableChauf1; - private int[] altTableChauf2; - private int[] altTableChauf3; - private int[] altTableNumBundles; - private int[] altTableNumRideSharing1Bundles; - private int[] altTableNumPureEscort1Bundles; - private int[] altTableNumRideSharing2Bundles; - private int[] altTableNumPureEscort2Bundles; - - private float defaultTourVOT = (float)7.5; //default VOT for pure escort tours generated by this model. - private float defaultTripVOT = (float)15.0; //default VOT for pure escort trips generated by this model. - - private int[][] altBundleIncidence; - - private int[] previousChoiceChauffeurs; - - private SchoolEscortingDmu decisionMaker; - private transient Logger logger = Logger.getLogger(SchoolEscortingModel.class); - - private static final String UEC_PATH_KEY = "uec.path"; - private static final String OUTBOUND_UEC_FILENAME_KEY = "school.escort.uec.filename"; - private static final String OUTBOUND_UEC_MODEL_SHEET_KEY = "school.escort.outbound.model.sheet"; - private static final String OUTBOUND_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; - private static final String OUTBOUND_CHOICE_MODEL_DESCRIPTION = "School Escorting - Outbound unconditional Choice"; - - private static final String INBOUND_CONDITIONAL_UEC_FILENAME_KEY = "school.escort.uec.filename"; - private static final String INBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY = "school.escort.inbound.conditonal.model.sheet"; - private static final String INBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; - private static final String INBOUND_CONDITIONAL_CHOICE_MODEL_DESCRIPTION = "School Escorting - inbound Conditional Choice"; - - private static final String OUTBOUND_CONDITIONAL_UEC_FILENAME_KEY = "school.escort.uec.filename"; - private static final String OUTBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY = "school.escort.outbound.conditonal.model.sheet"; - private static final String OUTBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY = "school.escort.data.sheet"; - private static final String OUTBOUND_CONDITIONAL_CHOICE_MODEL_DESCRIPTION = "School Escorting - Outbound Conditional Choice"; - - private static final String PROPERTIES_MODEL_OFFSET = "school.escort.RNG.offset"; - - private ChoiceModelApplication outboundModel; - private ChoiceModelApplication inboundConditionalModel; - private ChoiceModelApplication outboundConditionalModel; - - private MgraDataManager mgraDataManager; - private double[][] distanceArray; - - private IndexValues indexValues; - - private long randomOffset = 110001; - private MersenneTwister random; - - - public SchoolEscortingModel( HashMap propertyMap, MgraDataManager mgraDataManager, AutoTazSkimsCalculator tazDistanceCalculator ) { - - this.mgraDataManager = mgraDataManager; - - random = new MersenneTwister(); - randomOffset = PropertyMap.getLongValueFromPropertyMap(propertyMap,PROPERTIES_MODEL_OFFSET); - - //to do: set distance and time matrix - - createChauffeurPriorityOutboundMap(); - createChauffeurPriorityInboundMap(); - - double[][][] storedFromTazToAllTazsDistanceSkims = tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(); - distanceArray = storedFromTazToAllTazsDistanceSkims[ModelStructure.AM_SKIM_PERIOD_INDEX]; - decisionMaker = new SchoolEscortingDmu( mgraDataManager, distanceArray ); - - - // Create the choice model applications - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - //1. outbound model - String outboundUecFile = uecPath + propertyMap.get(OUTBOUND_UEC_FILENAME_KEY); - int outboundDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_UEC_DATA_SHEET_KEY); - int outboundModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_UEC_MODEL_SHEET_KEY); - - // create the outbound choice model object - outboundModel = new ChoiceModelApplication(outboundUecFile, outboundModelPage, outboundDataPage, propertyMap, - (VariableTable) decisionMaker); - - //2. inbound conditional model - String inboundConditionalUecFile = uecPath + propertyMap.get(INBOUND_CONDITIONAL_UEC_FILENAME_KEY); - int inboundConditionalDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, INBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY); - int inboundConditionalModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, INBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY); - - // create the inbound conditional choice model object - inboundConditionalModel = new ChoiceModelApplication(inboundConditionalUecFile, inboundConditionalModelPage, - inboundConditionalDataPage, propertyMap, - (VariableTable) decisionMaker); - - //3. outbound conditional model - String outboundConditionalUecFile = uecPath + propertyMap.get(OUTBOUND_CONDITIONAL_UEC_FILENAME_KEY); - int outboundConditionalDataPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_CONDITIONAL_UEC_DATA_SHEET_KEY); - int outboundConditionalModelPage = Util.getIntegerValueFromPropertyMap(propertyMap, OUTBOUND_CONDITIONAL_UEC_MODEL_SHEET_KEY); - - // create the outbound conditional choice model object - outboundConditionalModel = new ChoiceModelApplication(outboundConditionalUecFile, outboundConditionalModelPage, - outboundConditionalDataPage, propertyMap, - (VariableTable) decisionMaker); - - indexValues = new IndexValues(); - - // get the 0-index columns from the alternatives table, and save in 1-base indexed arrays for lookup by alternative number (1,...,numAlts). - UtilityExpressionCalculator outboundUEC = outboundModel.getUEC(); - altTable = outboundUEC.getAlternativeData(); - altTableNames = outboundUEC.getAlternativeNames(); - - int[] temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE1_NAME ); - altTableBundle1 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableBundle1[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE2_NAME ); - altTableBundle2 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableBundle2[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_BUNDLE3_NAME ); - altTableBundle3 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableBundle3[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF1_NAME ); - altTableChauf1 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableChauf1[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF2_NAME ); - altTableChauf2 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableChauf2[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_CHAUF3_NAME ); - altTableChauf3 = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableChauf3[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_NAME ); - altTableNumBundles = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableNumBundles[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_RS1_NAME ); - altTableNumRideSharing1Bundles = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableNumRideSharing1Bundles[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_ES1_NAME ); - altTableNumPureEscort1Bundles = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableNumPureEscort1Bundles[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_RS2_NAME ); - altTableNumRideSharing2Bundles = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableNumRideSharing2Bundles[i+1] = temp[i]; - - temp = altTable.getColumnAsInt( ALT_TABLE_NBUNDLES_ES2_NAME ); - altTableNumPureEscort2Bundles = new int[temp.length+1]; - for ( int i=0; i < temp.length; i++ ) - altTableNumPureEscort2Bundles[i+1] = temp[i]; - - int index = altTable.getColumnPosition( ALT_TABLE_BUNDLE_INCIDENCE_FIRST_COLUMN_NAME ); - int numColumns = NUM_BUNDLES * NUM_CHAUFFEURS_PER_HH * NUM_ESCORT_TYPES * NUM_ESCORTEES_PER_HH; - altBundleIncidence = new int[temp.length+1][ numColumns + 1 ]; - for ( int i=0; i < numColumns; i++ ) { - temp = altTable.getColumnAsInt( index+i ); - for ( int j=0; j < temp.length; j++ ) - altBundleIncidence[j+1][i+1] = temp[j]; - } - - decisionMaker.setAltTableBundleIncidence( altBundleIncidence ); - - previousChoiceChauffeurs = new int[NUM_CHAUFFEURS_PER_HH+1]; - - } - - - - /** - * Solve the school escort model for an array of households. The method sets DMU objects, calls the outbound choice model, the - * inbound conditional choice model, and another outbound conditional choice model. The method returns an ArrayList of results with - * 3 member lists: - * 0: results for escortees - * 1: results for chauffeurs - * 2: - * @param logger - * @param hhs - * @throws Exception - */ - public void applyModel( Household household ) throws Exception { - - List childResultList = new ArrayList(); - List chaufResultList = new ArrayList(); - - // apply model only if at least 1 child goes to school - // output: - // child - each direction, escortType(0=none, 1=ride sharing, 2=pure escort), pnum of chauffeur, bundle id, preferred departure time - // chauffeur - each direction, each bundle, pnums of children, preferred departure times - // household - bundles - - //TODO: apply schedule synchronization step after choice according to rules - - - int bundleListId = 0; - List escortBundleList = new ArrayList(); - List obPidList = new ArrayList(); - List obPeList = new ArrayList(); - List obRsList = new ArrayList(); - List ibPidList = new ArrayList(); - List ibPeList = new ArrayList(); - List ibRsList = new ArrayList(); - - try { - //there has to be at least one child with a school tour and one active adult - if ( (household.getNumChildrenWithSchoolTours() > 0) && (household.getNumberActiveAdults() > 0)) { - - boolean debug = false; - if ( household.getDebugChoiceModels() ) { - household.logEntireHouseholdObject("Escort Model trace for Household "+household.getHhId(), logger); - debug = true; - } - - long seed = household.getSeed() + randomOffset; - random.setSeed(seed); - - previousChoiceChauffeurs[1] = 0; - previousChoiceChauffeurs[2] = 0; - - setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_OUTBOUND ); - setDmuAttributesForAdultsOutbound( household, null ); - int[][][] ob0BundleResults = applyOutboundChoiceModel( logger, household, random, debug ); - - - int[][][] chaufExtentsReservedForIb = getEscortBundlesExtent( ob0BundleResults[1], SchoolEscortingModel.DIR_OUTBOUND, household.getSize() ); - setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_INBOUND ); - setDmuAttributesForAdultsInbound( household, chaufExtentsReservedForIb ); - - //note: first dimension = escortees versus chauffeurs. Second dimension = size of household or size * bundles. Third dimension = results - int[][][] ibBundleResults = applyInboundConditionalChoiceModel( logger, household, random, debug ); - - try { - bundleListId = createInboundEscortBundleObjects( ibBundleResults[1], bundleListId, escortBundleList, ibPidList, ibPeList, ibRsList ); - } - catch (Exception e) { - logger.error ( "exception caught saving inbound school escort bundle objects for hhid = " + household.getHhId() + ".", e ); - throw new RuntimeException(e); - } - - - int[][][] chaufExtentsReservedForOb = getEscortBundlesExtent( ibBundleResults[1], SchoolEscortingModel.DIR_INBOUND, household.getHhSize() ); - setDmuAttributesForChildren( household, SchoolEscortingModel.DIR_OUTBOUND ); - setDmuAttributesForAdultsOutbound( household, chaufExtentsReservedForOb ); - //note: first dimension = escortees versus chauffeurs. Second dimension = size of household or size * bundles. Third dimension = results - int[][][] obBundleResults = applyOutboundConditionalChoiceModel( logger, household, debug ); - - try { - bundleListId = createOutboundEscortBundleObjects( obBundleResults[1], bundleListId, escortBundleList, obPidList, obPeList, obRsList ); - } - catch (Exception e) { - logger.error ( "exception caught saving outbound school escort bundle objects for hhid = " + household.getHhId() + ".", e ); - throw new RuntimeException(e); - } - - for ( int j=1; j < ibBundleResults[0].length; j++ ) { - if ( ( ibBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 4 || ibBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 5 ) && ibBundleResults[0][j][RESULT_CHILD_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) - logger.info( "inbound child has ridesharing with non-working chauffeur, j=" + j ); - else if ( ( obBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 4 || obBundleResults[0][j][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] == 5 ) && obBundleResults[0][j][RESULT_CHILD_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) - logger.info( "outbound child has ridesharing with non-working chauffeur, j=" + j ); - else if ( ( ibBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 4 || ibBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 5 ) && ibBundleResults[1][j][RESULT_CHAUF_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) - logger.info( "inbound non-working chauffeur has ridesharing, j=" + j ); - else if ( ( obBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 4 || obBundleResults[1][j][RESULT_CHAUF_PERSON_TYPE_FIELD] == 5 ) && obBundleResults[1][j][RESULT_CHAUF_ESCORT_TYPE_FIELD] == ModelStructure.RIDE_SHARING_TYPE ) - logger.info( "outbound non-working chauffeur has ridesharing, j=" + j ); - } - - childResultList.add( obBundleResults[0] ); //outbound results for escortees - childResultList.add( ibBundleResults[0] ); //inbound results for escortees - chaufResultList.add( obBundleResults[1] ); //outbound results for chauffeurs - chaufResultList.add( ibBundleResults[1] ); //inbound results for chauffeurs - - - } - createTours(household, escortBundleList); - recodeSchoolTours(household); - - } - catch ( Exception e ) { - logger.error( "exception caught applying escort choice model for hh id:" + household.getHhId(), e ); - household.logEntireHouseholdObject("Escort model trace for problem household", logger); - int bundleNumber=0; - for(SchoolEscortingBundle bundle : escortBundleList){ - logger.error("School Escort Bundle "+bundleNumber); - bundle.logBundle(logger); - ++bundleNumber; - } - throw new RuntimeException(e); - } - - if(household.getDebugChoiceModels()){ - logger.info("Logging escort model results for household: "+household.getHhId()); - household.logEntireHouseholdObject("Escort model logging", logger); - } - - } - - - - private int[][][] getEscortBundlesExtent( int[][] bundleResults, int dir, int numPersons ) { - - Set processedChauffSet = new TreeSet(); - - int[][][] chaufExtent = new int[NUM_ESCORT_TYPES+1][numPersons+1][2]; - - for ( int j=1; j < bundleResults.length; j++ ) { - - int chaufid = bundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; - if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { - - int chaufPnum = bundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; - SchoolEscortingBundle[] escortBundles = decisionMaker.getChosenBundles( bundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, dir ); - - if ( escortBundles[0].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - chaufExtent[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][0] = escortBundles[0].getDepartHome(); - chaufExtent[ModelStructure.RIDE_SHARING_TYPE][chaufPnum][1] = escortBundles[0].getArriveWork(); - } - else if ( escortBundles[0].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - chaufExtent[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][0] = escortBundles[0].getDepartHome(); - - int lastBundleIndex = 0; - for ( int i=0; i < escortBundles.length; i++ ) - if ( escortBundles[i].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) - lastBundleIndex = i; - chaufExtent[ModelStructure.PURE_ESCORTING_TYPE][chaufPnum][1] = escortBundles[lastBundleIndex].getArriveHome(); - } - - processedChauffSet.add( chaufid ); - - } - - } - - return chaufExtent; - - } - - - - /** - * Create outbound escort bundle objects and return the bundle list id incremented by the number of new bundles created. The method also adds the bundles to the escortBundleList and - * increments bundleListId by the number of outbound escort bundles. - * - * @param obBundleResults An integer array of results for escortees dimensioned by: household size + 1, escort results fields. - * @param bundleListId The starting bundle ID; will be incremented for each additional bundle, first for rideshare bundles, then escort bundles. - * @param escortBundleList A List of SchoolEscortingBundle objects that will be added to by this method. - * @param obPidList A List of chauffeur IDs that will be added to by this method, one for each outbound bundle - * @param obPeList A List of the outbound pure escort bundle IDs, one for each outbound pure escort bundle - * @param obRsList A List of the outbound rideshare bundle IDs, one for each outbound rideshare bundle - * @return The updated number of chosen bundles (last ID set in the ID lists) - */ - private int createOutboundEscortBundleObjects( int[][] obBundleResults, int bundleListId, List escortBundleList, List obPidList, List obPeList, List obRsList ) { - - int[] obRsIds = null; - int[] obPeIds = null; - - Set processedChauffSet = new TreeSet(); - - for ( int j=1; j < obBundleResults.length; j++ ) { - - int chaufid = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; - if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { - - int chaufPid = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PID_FIELD]; - int chaufPnum = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; - int chaufPtype = obBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PERSON_TYPE_FIELD]; - - SchoolEscortingBundle[] obEscortBundles = decisionMaker.getChosenBundles( obBundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, SchoolEscortingModel.DIR_OUTBOUND ); - - int numRs = 0; - int numPe = 0; - for ( int k=0; k < obEscortBundles.length; k++ ) { - if ( obEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - numRs++; - else if ( obEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) - numPe++; - } - - obRsIds = new int[numRs]; - obPeIds = new int[numPe]; - - int n = 0; - for ( int k=0; k < obEscortBundles.length; k++ ) { - if ( obEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - obEscortBundles[k].setId(bundleListId); - obEscortBundles[k].setChaufPnum( chaufPnum ); - obEscortBundles[k].setChaufPersType( chaufPtype ); - obEscortBundles[k].setChaufPid( chaufPid ); - escortBundleList.add( obEscortBundles[k] ); - obRsIds[n++] = bundleListId++; - } - } - n = 0; - for ( int k=0; k < obEscortBundles.length; k++ ) { - if ( obEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - obEscortBundles[k].setId(bundleListId); - obEscortBundles[k].setChaufPnum( chaufPnum ); - obEscortBundles[k].setChaufPersType( chaufPtype ); - obEscortBundles[k].setChaufPid( chaufPid ); - escortBundleList.add( obEscortBundles[k] ); - obPeIds[n++] = bundleListId++; - } - } - - obPidList.add( chaufPid ); - obPeList.add( obPeIds ); - obRsList.add( obRsIds ); - - processedChauffSet.add( chaufid ); - - } - - } - - return bundleListId; - - } - - - /** - * Create inbound escort bundle objects and return the bundle list id incremented by the number of new bundles created. The method also adds the bundles to the escortBundleList and - * increments bundleListId by the number of inbound escort bundles. - * - * @param ibBundleResults An integer array of results for escortees dimensioned by: household size + 1, escort results fields. - * @param bundleListId The starting bundle ID; will be incremented for each additional bundle, first for rideshare bundles, then escort bundles. - * @param escortBundleList A List of SchoolEscortingBundle objects that will be added to by this method. - * @param ibPidList A List of chauffeur IDs that will be added to by this method, one for each inbound bundle - * @param ibPeList A List of the inbound pure escort bundle IDs, one for each inbound pure escort bundle - * @param ibRsList A List of the inbound rideshare bundle IDs, one for each inbound rideshare bundle - * @return The updated number of chosen bundles (last ID in the ID lists) - */ - private int createInboundEscortBundleObjects( int[][] ibBundleResults, int bundleListId, List escortBundleList, List ibPidList, List ibPeList, List ibRsList ) { - - int[] ibRsIds = null; - int[] ibPeIds = null; - - Set processedChauffSet = new TreeSet(); - - //for each person in the household - for ( int j=1; j < ibBundleResults.length; j++ ) { - - //the id of the chauffeur for this escortee - int chaufid = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_ID_FIELD]; - - // If this household member is escorted and we haven't created a bundle for the chauffeur yet - if ( chaufid > 0 && !processedChauffSet.contains( chaufid ) ) { - - int chaufPid = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PID_FIELD]; - int chaufPnum = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PNUM_FIELD]; - int chaufPtype = ibBundleResults[j][SchoolEscortingModel.RESULT_CHAUF_PERSON_TYPE_FIELD]; - - //get the chosen bundles for this chauffeur in the inbound direction - SchoolEscortingBundle[] ibEscortBundles = decisionMaker.getChosenBundles( ibBundleResults[j][RESULT_CHAUF_CHOSEN_ALT_FIELD], chaufid, SchoolEscortingModel.DIR_INBOUND ); - - int numRs = 0; //number rideshare bundles - int numPe = 0; //number pure escort bundles - for ( int k=0; k < ibEscortBundles.length; k++ ) { - if ( ibEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) - numRs++; - else if ( ibEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) - numPe++; - } - - ibRsIds = new int[numRs]; //id of each inbound rideshare bundle - ibPeIds = new int[numPe]; //id of each inbound pure escort bundle - - //for each inbound bundle for this chauffeur, set the bundle chauffeur attributes - int n = 0; - for ( int k=0; k < ibEscortBundles.length; k++ ) { - if ( ibEscortBundles[k].getEscortType() == ModelStructure.RIDE_SHARING_TYPE ) { - ibEscortBundles[k].setId(bundleListId); - ibEscortBundles[k].setChaufPnum( chaufPnum ); - ibEscortBundles[k].setChaufPersType( chaufPtype ); - ibEscortBundles[k].setChaufPid( chaufPid ); - escortBundleList.add( ibEscortBundles[k] ); - ibRsIds[n++] = bundleListId++; - } - } - n = 0; - for ( int k=0; k < ibEscortBundles.length; k++ ) { - if ( ibEscortBundles[k].getEscortType() == ModelStructure.PURE_ESCORTING_TYPE ) { - ibEscortBundles[k].setId(bundleListId); - ibEscortBundles[k].setChaufPnum( chaufPnum ); - ibEscortBundles[k].setChaufPersType( chaufPtype ); - ibEscortBundles[k].setChaufPid( chaufPid ); - escortBundleList.add( ibEscortBundles[k] ); - ibPeIds[n++] = bundleListId++; - } - } - - ibPidList.add( chaufPid ); - ibPeList.add( ibPeIds ); - ibRsList.add( ibRsIds ); - - processedChauffSet.add( chaufid ); - - } - - } - - return bundleListId; - - } - - - /** - * Apply the outbound choice model for the household. Returns a three-dimensional integer array where: - * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs - * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 - * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. - * Ever hear of an object? - * @param logger A logger to write messages and debug statements to. - * @param hh The household to solve the outbound model for. - * @param debug If true calculations will be logged to the logger. - * @return A ragged 3d integer array containing results for the outbound choice model for both escortees and chauffeurs. - * @throws Exception Will be thrown if no alternative is chosen. - */ - private int[][][] applyOutboundChoiceModel( Logger logger, Household hh, Random randObject, boolean debug ) throws Exception { - - - double rn = randObject.nextDouble(); - outboundModel.computeUtilities(decisionMaker, indexValues); - //double[] utilities = outboundModel. - int chosenObAlt = outboundModel.getChoiceResult(rn); - if ( chosenObAlt < 0 || debug ) { - - logger.info("Logging Escort Outbound Model Results for household "+hh.getHhId()); - - //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; - int[] altvaluesToLog = new int[50]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+1; - outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+51; - outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+101; - outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); - altvaluesToLog = new int[altTableNames.length - 150]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+151; - outboundModel.logUECResultsSpecificAlts( logger, "Escort model outbound UEC", altvaluesToLog ); - if ( chosenObAlt < 0 ) { - logger.error( hh.toString() ); - throw new Exception( "chosenObAlt = " + chosenObAlt + " for hhid=" + hh.getHhId() ); - } - - //outboundModel.logInfo("outbound unconditional model", "HHID "+ hh.getHhId(), logger); - - if(debug) - logger.info("Chose outbound unconditional alternative "+ chosenObAlt); - } - - // get field valuess from alternatives table associated with chosen alternative - int chosenObBundle1 = altTableBundle1[chosenObAlt]; - int chosenObBundle2 = altTableBundle2[chosenObAlt]; - int chosenObBundle3 = altTableBundle3[chosenObAlt]; - int chosenObChauf1 = altTableChauf1[chosenObAlt]; - int chosenObChauf2 = altTableChauf2[chosenObAlt]; - int chosenObChauf3 = altTableChauf3[chosenObAlt]; - - - // set the dmu attributes associated with the chosen alternative needed by the Inbound Conditional Choice model - decisionMaker.setOutboundEscortType1( chosenObChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setOutboundEscortType2( chosenObChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setOutboundEscortType3( chosenObChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenObChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenObChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenObChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setOutboundChauffeur1( chosenObChauf1 ); - decisionMaker.setOutboundChauffeur2( chosenObChauf2 ); - decisionMaker.setOutboundChauffeur3( chosenObChauf3 ); - - int[][][] results = new int[2][][]; - results[0] = getResultsByChildArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, (int)(rn*1000000000), chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); - results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); - - return results; - - } - - - /** - * Apply the inbound conditional choice model for the household. Returns a three-dimensional integer array where: - * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs - * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 - * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. - * Ever hear of an object? - * @param logger A logger to write messages and debug statements to. - * @param hh The household to solve the inbound conditional model for. - * @param debug If true calculations will be logged to the logger. - * @return A ragged 3d integer array containing results for the inbound conditional choice model for both escortees and chauffeurs. - * @throws Exception Will be thrown if no alternative is chosen. - */ - private int[][][] applyInboundConditionalChoiceModel( Logger logger, Household hh, Random randObject, boolean debug ) throws Exception { - - double rn = randObject.nextDouble(); - inboundConditionalModel.computeUtilities(decisionMaker, indexValues); - int chosenIbAlt = inboundConditionalModel.getChoiceResult(rn); - if ( chosenIbAlt < 0 || debug ) { - - logger.info("Logging Escort Inbound Conditional Model Results for household "+hh.getHhId()); - - //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; - int[] altvaluesToLog = new int[50]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+1; - inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+51; - inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+101; - inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); - altvaluesToLog = new int[altTableNames.length - 150]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+151; - inboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model inbound conditional UEC", altvaluesToLog ); - if ( chosenIbAlt < 0 ) { - logger.error( hh.toString() ); - throw new Exception( "chosenIbAlt = " + chosenIbAlt + " for hhid=" + hh.getHhId() ); - } - - //inboundConditionalModel.logInfo("inbound conditional model", "HHID "+ hh.getHhId(), logger); - - if(debug) - logger.info("Chose inbound conditional alternative "+ chosenIbAlt); - } - - hh.setInboundEscortChoice(chosenIbAlt); - - // get field values from alternatives table associated with chosen alternative - int chosenIbBundle1 = altTableBundle1[chosenIbAlt]; - int chosenIbBundle2 = altTableBundle2[chosenIbAlt]; - int chosenIbBundle3 = altTableBundle3[chosenIbAlt]; - int chosenIbChauf1 = altTableChauf1[chosenIbAlt]; - int chosenIbChauf2 = altTableChauf2[chosenIbAlt]; - int chosenIbChauf3 = altTableChauf3[chosenIbAlt]; - - if ( debug ) { - int[] escortees = decisionMaker.getEscorteePnums(); - String escorteeString = String.format( "[%s%s%s]", String.valueOf(escortees[1]), (escortees[2] > 0 ? "," + String.valueOf(escortees[2]) : ""), (escortees[3] > 0 ? "," + String.valueOf(escortees[3]) : "") ); - int[] chaufs = decisionMaker.getChauffeurPnums(); - String chaufString = String.format( "[%s%s]", String.valueOf(chaufs[1]), (chaufs[2] > 0 ? "," + String.valueOf(chaufs[2]) : "") ); - logger.info( "hhid=" + hh.getHhId() + ", escortees=" + escorteeString + ", chaufs=" + chaufString ); - logger.info( "chosenIbAlt=" + chosenIbAlt + ", chosenIbChauf1=" + chosenIbChauf1 + ", chosenIbChauf2=" + chosenIbChauf2 + ", chosenIbChauf3=" + chosenIbChauf3 ); - logger.info( "chosenIbBundle1=" + chosenIbBundle1 + ", chosenIbBundle2=" + chosenIbBundle2 + ", chosenIbBundle3=" + chosenIbBundle3 ); - } - - - // set the dmu attributes associated with the chosen alternative needed by the Outbound Conditional Choice model - decisionMaker.setInboundEscortType1( chosenIbChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setInboundEscortType2( chosenIbChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setInboundEscortType3( chosenIbChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenIbChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenIbChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenIbChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0 ); - decisionMaker.setInboundChauffeur1( chosenIbChauf1 ); - decisionMaker.setInboundChauffeur2( chosenIbChauf2 ); - decisionMaker.setInboundChauffeur3( chosenIbChauf3 ); - - - int[][][] results = new int[2][][]; - results[0] = getResultsByChildArray( hh, decisionMaker, DIR_INBOUND, chosenIbAlt, (int)(rn*1000000000), chosenIbBundle1, chosenIbBundle2, chosenIbBundle3, chosenIbChauf1, chosenIbChauf2, chosenIbChauf3 ); - results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_INBOUND, chosenIbAlt, chosenIbBundle1, chosenIbBundle2, chosenIbBundle3, chosenIbChauf1, chosenIbChauf2, chosenIbChauf3 ); - - return results; - - } - - - /** - * Apply the outbound conditional choice model for the household. Returns a three-dimensional integer array where: - * dimension 1: sized 2, 0 = results for escortees in household, 1 = results for chauffeurs - * dimension 2: if d1 = 0, sized by household size + 1, if d1 = 1, sized by household size * max bundles (3) + 1 - * dimension 3: if d1 = 0, size by number of result fields for escortees, if d1 = 1, sized by number of result fields for chauffeurs. - * Ever hear of an object? - * @param logger A logger to write messages and debug statements to. - * @param hh The household to solve the outbound conditional model for. - * @param debug If true calculations will be logged to the logger. - * @return A ragged 3d integer array containing results for the outbound conditional choice model for both escortees and chauffeurs. - * @throws Exception Will be thrown if no alternative is chosen. - */ - private int[][][] applyOutboundConditionalChoiceModel( Logger logger, Household hh, boolean debug ) throws Exception { - - double rn = random.nextDouble(); - outboundConditionalModel.computeUtilities(decisionMaker, indexValues); - int chosenObAlt = outboundConditionalModel.getChoiceResult(rn); - if ( chosenObAlt < 0 || debug ) { - - logger.info("Logging Escort Outbound Conditional Model Results for household "+hh.getHhId()); - - //int[] altvaluesToLog = new int[]{ 1, 7, 40, 70, 105, 138, 161, 188 }; - int[] altvaluesToLog = new int[50]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+1; - outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+51; - outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+101; - outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); - altvaluesToLog = new int[altTableNames.length - 150]; - for ( int i=0; i < altvaluesToLog.length; i++ ) - altvaluesToLog[i] = i+151; - outboundConditionalModel.logUECResultsSpecificAlts( logger, "Escort model outbound conditional UEC", altvaluesToLog ); - if ( chosenObAlt < 0 ) { - logger.error( hh.toString() ); - throw new Exception( "chosenObAlt = " + chosenObAlt + " for hhid=" + hh.getHhId() ); - } - - //outboundConditionalModel.logInfo("outbound conditional model", "HHID "+ hh.getHhId(), logger); - - if(debug) - logger.info("Chose outbound conditional alternative "+ chosenObAlt); - } - - hh.setOutboundEscortChoice(chosenObAlt); - - // get field valuess from alternatives table associated with chosen alternative - int chosenObBundle1 = altTableBundle1[chosenObAlt]; - int chosenObBundle2 = altTableBundle2[chosenObAlt]; - int chosenObBundle3 = altTableBundle3[chosenObAlt]; - int chosenObChauf1 = altTableChauf1[chosenObAlt]; - int chosenObChauf2 = altTableChauf2[chosenObAlt]; - int chosenObChauf3 = altTableChauf3[chosenObAlt]; - - if ( debug ) { - int[] escortees = decisionMaker.getEscorteePnums(); - String escorteeString = String.format( "[%s%s%s]", String.valueOf(escortees[1]), (escortees[2] > 0 ? "," + String.valueOf(escortees[2]) : ""), (escortees[3] > 0 ? "," + String.valueOf(escortees[3]) : "") ); - int[] chaufs = decisionMaker.getChauffeurPnums(); - String chaufString = String.format( "[%s%s]", String.valueOf(chaufs[1]), (chaufs[2] > 0 ? "," + String.valueOf(chaufs[2]) : "") ); - logger.info( "hhid=" + hh.getHhId() + ", escortees=" + escorteeString + ", chaufs=" + chaufString ); - logger.info( "chosenObAlt=" + chosenObAlt + ", chosenObChauf1=" + chosenObChauf1 + ", chosenObChauf2=" + chosenObChauf2 + ", chosenObChauf3=" + chosenObChauf3 ); - logger.info( "chosenObBundle1=" + chosenObBundle1 + ", chosenObBundle2=" + chosenObBundle2 + ", chosenObBundle3=" + chosenObBundle3 ); - } - - int[][][] results = new int[2][][]; - results[0] = getResultsByChildArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, (int)(rn*1000000000), chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); - results[1] = getResultsByChauffeurArray( hh, decisionMaker, DIR_OUTBOUND, chosenObAlt, chosenObBundle1, chosenObBundle2, chosenObBundle3, chosenObChauf1, chosenObChauf2, chosenObChauf3 ); - - return results; - - } - - - private void setDmuAttributesForChildren( Household hh, int dir ) { - - List children = hh.getChildPersons(); - List cList = new ArrayList(); - for ( Person child : children ) { - if (child.getUsualSchoolLocation() > 0 && child.getNumSchoolTours() > 0 ) - cList.add( child ); - } - Person[] escortees = getOrderedSchoolChildrenForEscorting( cList, dir ); - - int[] schoolAtHome = new int[NUM_ESCORTEES_PER_HH+1]; - int[] schoolMazs = new int[NUM_ESCORTEES_PER_HH+1]; - int[] schoolDeparts = new int[NUM_ESCORTEES_PER_HH+1]; - int[] schoolReturns = new int[NUM_ESCORTEES_PER_HH+1]; - for ( int i=1; i < escortees.length; i++ ) { - if ( escortees[i] == null ) - continue; - ArrayList schoolTours = escortees[i].getListOfSchoolTours(); - if ( schoolTours != null ) { - Tour tour = schoolTours.get(0); - schoolAtHome[i] = 0; - schoolMazs[i] = tour.getTourDestMgra(); - schoolDeparts[i] = tour.getTourDepartPeriod(); - schoolReturns[i] = tour.getTourArrivePeriod(); - } - } - - decisionMaker.setEscorteeAttributes( cList.size(), escortees, schoolAtHome, schoolMazs, schoolDeparts, schoolReturns ); - - } - - - private void setDmuAttributesForAdultsInbound( Household hh, int[][][] chaufExtents ) { - - List activeAdults = hh.getActiveAdultPersons(); - - Person[] chauffers = getOrderedAdultsForChauffeuringInbound( activeAdults ); - - int[] mandatoryMazs = new int[chauffers.length]; - int[] mandatoryDeparts = new int[chauffers.length]; - int[] mandatoryReturns = new int[chauffers.length]; - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) - continue; - - ArrayList mandatoryTours = getMandatoryTours(chauffers[i]); - - if ( ! mandatoryTours.isEmpty() ) { - Tour tour2 = mandatoryTours.size() == 2 ? mandatoryTours.get(1) : mandatoryTours.get(0); - mandatoryMazs[i] = tour2.getTourDestMgra(); - mandatoryDeparts[i] = tour2.getTourDepartPeriod(); - mandatoryReturns[i] = tour2.getTourArrivePeriod(); - } - } - - decisionMaker.setChaufferAttributes( activeAdults.size(), chauffers, mandatoryMazs, mandatoryDeparts, mandatoryReturns, chaufExtents ); - decisionMaker.setDistanceTimeAttributes( hh, distanceArray ); - - // set the potential chauffeur pnums from the outbound unconditional choice for use with the inbound conditional choice - decisionMaker.setOutboundPotentialChauffeur1( previousChoiceChauffeurs[1] ); - decisionMaker.setOutboundPotentialChauffeur2( previousChoiceChauffeurs[2] ); - - // set the potential chauffeur pnum values in the inbound direction for eventual use in the outbound conditional choice - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) - previousChoiceChauffeurs[i] = 0; - else - previousChoiceChauffeurs[i] = chauffers[i].getPersonNum(); - } - - } - - - /** - * Get a list of mandatory tours for this person. - * - * @param p A person. - * @return An ArrayList of mandatory tours. Empty if no mandatory tours. - */ - ArrayList getMandatoryTours(Person p){ - - ArrayList mandatoryTours = new ArrayList(); - - if(p.getListOfWorkTours()!=null) - mandatoryTours.addAll(p.getListOfWorkTours()); - - if(p.getListOfSchoolTours()!=null) - mandatoryTours.addAll(p.getListOfSchoolTours()); - - return mandatoryTours; - - } - /** - * Set DMU attributes for adults in the outbound direction. - * - * @param hh - * @param chaufExtents - */ - private void setDmuAttributesForAdultsOutbound( Household hh, int[][][] chaufExtents ) { - - List activeAdults = hh.getActiveAdultPersons(); - - Person[] chauffers = getOrderedAdultsForChauffeuringOutbound( activeAdults ); - - int[] mandatoryMazs = new int[NUM_CHAUFFEURS_PER_HH+1]; - int[] mandatoryDeparts = new int[NUM_CHAUFFEURS_PER_HH+1]; - int[] mandatoryReturns = new int[NUM_CHAUFFEURS_PER_HH+1]; - - - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) - continue; - - ArrayList mandatoryTours = getMandatoryTours(chauffers[i]); - - if ( ! mandatoryTours.isEmpty() ) { - Tour tour1 = mandatoryTours.get(0); - Tour tour2 = mandatoryTours.size() == 2 ? mandatoryTours.get(1) : mandatoryTours.get(0); - mandatoryMazs[i] = tour1.getTourDestMgra(); - mandatoryDeparts[i] = tour1.getTourDepartPeriod(); - mandatoryReturns[i] = tour2.getTourArrivePeriod(); - } - - } - - decisionMaker.setChaufferAttributes( activeAdults.size(), chauffers, mandatoryMazs, mandatoryDeparts, mandatoryReturns, chaufExtents ); - decisionMaker.setDistanceTimeAttributes( hh, distanceArray ); - - // set the potential chauffeur pnums from the inbound conditional choice for use with the outbound conditional choice - decisionMaker.setInboundPotentialChauffeur1( previousChoiceChauffeurs[1] ); - decisionMaker.setInboundPotentialChauffeur2( previousChoiceChauffeurs[2] ); - - // set the potential chauffeur pnum values in the outbound direction for eventual use in the inbound conditional choice - for ( int i=1; i < chauffers.length; i++ ) { - if ( chauffers[i] == null ) - previousChoiceChauffeurs[i] = 0; - else - previousChoiceChauffeurs[i] = chauffers[i].getPersonNum(); - } - - } - - - /** - * Order children in list according to age, distance and time period, and return the result in an array. - * - * @param childList List of children to escort - * @param dir Direction of travel (outbound or return) - * @return - */ - private Person[] getOrderedSchoolChildrenForEscorting( List childList, int dir) { - - Household hh = childList.get( 0 ).getHouseholdObject(); - int homeTaz = mgraDataManager.getTaz( hh.getHhMgra() ); - - // sort the eligible children by age so that the 3 youngest are the final candidates - Collections.sort( childList, - new Comparator() { - @Override - public int compare( Person p1, Person p2 ) { - return p1.getAge() - p2.getAge(); - } - } - ); - - Person[] returnArray = new Person[NUM_ESCORTEES_PER_HH+1]; - - //if there is only one child to escort, its easy - just return him/her. - if ( childList.size() == 1 ) { - returnArray[1] = childList.get( 0 ); - } - //if there are two or three children sort them according to distance and departure time time - else if ( childList.size() == 2 ) { - - Person child0 = childList.get( 0 ); - ArrayList schoolTours = child0.getListOfSchoolTours(); - Tour tour0 = schoolTours.get(0); - int timeInterval0 = 0; - int schoolTaz = mgraDataManager.getTaz( tour0.getTourDestMgra()); - float dist0 = (float) distanceArray[homeTaz][ schoolTaz]; - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - timeInterval0 = tour0.getTourDepartPeriod(); - } - else { - timeInterval0 = tour0.getTourArrivePeriod(); - } - - Person child1 = childList.get( 1 ); - schoolTours = child1.getListOfSchoolTours(); - Tour tour1 = schoolTours.get(0); - int timeInterval1 = 0; - schoolTaz = mgraDataManager.getTaz( tour1.getTourDestMgra()); - float dist1 = (float) distanceArray[homeTaz][ schoolTaz]; - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - timeInterval1 = tour1.getTourDepartPeriod(); - } - else { - timeInterval1 = tour1.getTourArrivePeriod(); - } - - int[] sortData = new int[2]; - sortData[0] = timeInterval0*10000000 + (int)(dist0*1000); - sortData[1] = timeInterval1*10000000 + (int)(dist1*1000); - int[] sortIndices = IndexSort.indexSort( sortData ); - - returnArray[1] = childList.get( sortIndices[0] ); - returnArray[2] = childList.get( sortIndices[1] ); - - } - else if ( childList.size() >= 3 ) { - - Person child0 = childList.get( 0 ); - ArrayList schoolTours = child0.getListOfSchoolTours(); - Tour tour0 = schoolTours.get(0); - int timeInterval0 = 0; - int schoolTaz = mgraDataManager.getTaz( tour0.getTourDestMgra()); - float dist0 = (float) distanceArray[homeTaz][ schoolTaz]; - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - timeInterval0 = tour0.getTourDepartPeriod(); - } - else { - timeInterval0 = tour0.getTourArrivePeriod(); - } - - Person child1 = childList.get( 1 ); - schoolTours = child1.getListOfSchoolTours(); - Tour tour1 = schoolTours.get(0); - int timeInterval1 = 0; - schoolTaz = mgraDataManager.getTaz( tour1.getTourDestMgra()); - float dist1 = (float) distanceArray[homeTaz][ schoolTaz]; - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - timeInterval1 = tour1.getTourDepartPeriod(); - } - else { - timeInterval1 = tour1.getTourArrivePeriod(); - } - - Person child2 = childList.get( 2 ); - schoolTours = child2.getListOfSchoolTours(); - Tour tour2 = schoolTours.get(0); - int timeInterval2 = 0; - schoolTaz = mgraDataManager.getTaz( tour2.getTourDestMgra()); - float dist2 = (float) distanceArray[homeTaz][ schoolTaz]; - - if ( dir == SchoolEscortingModel.DIR_OUTBOUND ) { - timeInterval2 = tour2.getTourDepartPeriod(); - } - else { - timeInterval2 = tour2.getTourArrivePeriod(); - } - - int[] sortData = new int[3]; - sortData[0] = timeInterval0*10000000 + (int)(dist0*1000); - sortData[1] = timeInterval1*10000000 + (int)(dist1*1000); - sortData[2] = timeInterval2*10000000 + (int)(dist2*1000); - int[] sortIndices = IndexSort.indexSort( sortData ); - - returnArray[1] = childList.get( sortIndices[0] ); - returnArray[2] = childList.get( sortIndices[1] ); - returnArray[3] = childList.get( sortIndices[2] ); - - } - - return returnArray; - - } - - - private Person[] getOrderedAdultsForChauffeuringOutbound( List adultList ) { - - Collections.sort( adultList, - new Comparator() { - @Override - public int compare( Person p1, Person p2 ) { - int p1LookupValue = PT_WEIGHT*p1.getPersonTypeNumber() + G_WEIGHT*p1.getGender() + A_WEIGHT*(p1.getAge() > 25 ? 1 : 0); - int p2LookupValue = PT_WEIGHT*p2.getPersonTypeNumber() + G_WEIGHT*p2.getGender() + A_WEIGHT*(p2.getAge() > 25 ? 1 : 0); - if(!chauffeurPriorityOutboundMap.containsKey(p1LookupValue)){ - logger.fatal("Cannot find lookup value "+p1LookupValue+" in outbound chauffeur priority map"); - throw new RuntimeException(); - } - if(!chauffeurPriorityOutboundMap.containsKey(p2LookupValue)){ - logger.fatal("Cannot find lookup value "+p2LookupValue+" in outbound chauffeur priority map"); - throw new RuntimeException(); - } - int p1Score = chauffeurPriorityOutboundMap.get(p1LookupValue); - int p2Score = chauffeurPriorityOutboundMap.get(p2LookupValue); - return p1Score - p2Score; - } - } - ); - - Person[] returnArray = new Person[NUM_CHAUFFEURS_PER_HH+1]; - for ( int i=0; i < adultList.size() && i < NUM_CHAUFFEURS_PER_HH; i++ ) - returnArray[i+1] = adultList.get( i ); - - return returnArray; - - } - - - /** - * Orders the list of adults passed to the method by person type, gender, and age combination. - * - * @param adultList A list of potential chauffeurs - * @return An ordered array of the chauffeurs. - */ - private Person[] getOrderedAdultsForChauffeuringInbound( List adultList ) { - - Collections.sort( adultList, - new Comparator() { - @Override - public int compare( Person p1, Person p2 ) { - int p1LookupValue = PT_WEIGHT*p1.getPersonTypeNumber() + G_WEIGHT*p1.getGender() + A_WEIGHT*(p1.getAge() >= 25 ? 1 : 0); - int p2LookupValue = PT_WEIGHT*p2.getPersonTypeNumber() + G_WEIGHT*p2.getGender() + A_WEIGHT*(p2.getAge() >= 25 ? 1 : 0); - return chauffeurPriorityInboundMap.get(p1LookupValue) - chauffeurPriorityInboundMap.get(p2LookupValue); - } - } - ); - - Person[] returnArray = new Person[NUM_CHAUFFEURS_PER_HH+1]; - for ( int i=0; i < adultList.size() && i < NUM_CHAUFFEURS_PER_HH; i++ ) - returnArray[i+1] = adultList.get( i ); - - return returnArray; - - } - - /** - * Create the priority map for outbound chauffeurs, according to person type, gender, and age bin. - */ - private void createChauffeurPriorityOutboundMap() { - - chauffeurPriorityOutboundMap = new HashMap(); - - int lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 1 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 2 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 3 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 4 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 5 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 6 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 7 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 8 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 9 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 10 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 11 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 12 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 13 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 13 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 14 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 14 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 15 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 15 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityOutboundMap.put( lookupValue, 16 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityOutboundMap.put( lookupValue, 16 ); - - } - - - /** - * Create the priority map for inbound chauffeurs, according to person type, gender, and age bin. - */ - private void createChauffeurPriorityInboundMap() { - - chauffeurPriorityInboundMap = new HashMap(); - - int lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 1 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 1 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 2 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 2 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 3 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 3 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 4 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 5 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_PART_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 5 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 6 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_FULL_TIME_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 6 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 7 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 8 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_NON_WORKER_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 8 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 9 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_UNIVERSITY_STUDENT_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 9 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 10 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.FEMALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 10 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*0; - chauffeurPriorityInboundMap.put( lookupValue, 11 ); - - lookupValue = PT_WEIGHT*Person.PERSON_TYPE_RETIRED_INDEX + G_WEIGHT*Person.MALE_INDEX + A_WEIGHT*1; - chauffeurPriorityInboundMap.put( lookupValue, 11 ); - - } - - /** - * Gets the results for each child in the household. The method returns a two dimensional array where the first - * dimension is sized by persons in household + 1, and the second dimension is the total number of fields for child results. - * The first dimension is indexed into by person number and the second dimension is indexed into by result field number. - * Results include household and person attributes for the child being escorted and the chauffeur who is escorting them, - * as well as the attributes of the choice. - * - * @param hh - * @param decisionMaker - * @param direction - * @param chosenAlt Number of chosen alternative. - * @param intRandNum - * @param chosenBundle1 The bundle for child 1: 0 = not escorted. Max 3 bundles - * @param chosenBundle2 The bundle for child 2: 0 = not escorted. Max 3 bundles - * @param chosenBundle3 The bundle for child 3: 0 = not escorted. Max 3 bundles - * @param chosenChauf1 The chauffeur for child 1: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @param chosenChauf2 The chauffeur for child 2: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @param chosenChauf3 The chauffeur for child 3: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @return An integer array of results for each person in the household. - */ - private int[][] getResultsByChildArray( Household hh, SchoolEscortingDmu decisionMaker, int direction, int chosenAlt, int intRandNum, int chosenBundle1, int chosenBundle2, int chosenBundle3, int chosenChauf1, int chosenChauf2, int chosenChauf3 ) { - - int[][] resultsByChild = new int[hh.getHhSize()+1][NUM_RESULTS_BY_CHILD_FIELDS]; - - - int[] adultPnums = decisionMaker.getChauffeurPnums(); - int[] childPnums = decisionMaker.getEscorteePnums(); - - - // set result attributes for all children in the Household - for ( int pnum=1; pnum <= hh.getHhSize(); pnum++ ) { - Person person = hh.getPerson( pnum ); - resultsByChild[pnum][RESULT_CHILD_HHID_FIELD] = hh.getHhId(); - resultsByChild[pnum][RESULT_CHILD_PNUM_FIELD] = pnum; - resultsByChild[pnum][RESULT_CHILD_PID_FIELD] = person.getPersonId(); - resultsByChild[pnum][RESULT_CHILD_PERSON_TYPE_FIELD] = person.getPersonTypeNumber(); - resultsByChild[pnum][RESULT_CHILD_AGE_FIELD] = person.getAge(); - resultsByChild[pnum][RESULT_CHILD_CDAP_FIELD] = person.getCdapIndex(); - resultsByChild[pnum][RESULT_CHILD_SCHOOL_AT_HOME_FIELD] = 0; - resultsByChild[pnum][RESULT_CHILD_SCHOOL_LOC_FIELD] = person.getPersonSchoolLocationZone(); - } - - // set result attributes for children that are to be escorted - for ( Person child : hh.getChildPersons() ) { - int pnum = child.getPersonNum(); - resultsByChild[pnum][RESULT_CHILD_DIRECTION_FIELD] = direction; - resultsByChild[pnum][RESULT_CHILD_CHOSEN_ALT_FIELD] = chosenAlt; - resultsByChild[pnum][RESULT_CHILD_RANDOM_NUM_FIELD] = intRandNum; - } - - // for each escorted child - for ( int i=1; i < childPnums.length; i++ ) { - if ( childPnums[i] > 0 ) { - resultsByChild[childPnums[i]][RESULT_CHILD_ESCORT_ELIGIBLE_FIELD] = ESCORT_ELIGIBLE; - resultsByChild[childPnums[i]][RESULT_CHILD_DEPART_FROM_HOME_FIELD] = decisionMaker.getEscorteeDepartForSchool()[i]; - resultsByChild[childPnums[i]][RESULT_CHILD_DEPART_TO_HOME_FIELD] = decisionMaker.getEscorteeDepartFromSchool()[i]; - resultsByChild[childPnums[i]][RESULT_CHILD_DIST_TO_SCHOOL_FIELD] = decisionMaker.getEscorteeDistToSchool()[i]; - resultsByChild[childPnums[i]][RESULT_CHILD_DIST_FROM_SCHOOL_FIELD] = decisionMaker.getEscorteeDistFromSchool()[i]; - if ( adultPnums[1] > 0 ) { - resultsByChild[childPnums[i]][RESULT_CHILD_ADULT1_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[1]; - resultsByChild[childPnums[i]][RESULT_CHILD_ADULT1_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[1]; - } - if ( adultPnums[2] > 0 ) { - resultsByChild[childPnums[i]][RESULT_CHILD_ADULT2_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[2]; - resultsByChild[childPnums[i]][RESULT_CHILD_ADULT2_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[2]; - } - } - } - - // for each adult - for ( int i=1; i < adultPnums.length; i++ ) { - if ( adultPnums[i] > 0 ) { - resultsByChild[adultPnums[i]][RESULT_CHILD_DEPART_FROM_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[i]; - resultsByChild[adultPnums[i]][RESULT_CHILD_DEPART_TO_HOME_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[i]; - } - } - - if ( chosenChauf1 > 0 ) { - int childid = 1; - int chaufid = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; - resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle1; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; - } - - if ( chosenChauf2 > 0 ) { - int childid = 2; - int chaufid = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; - resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle2; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; - } - - if ( chosenChauf3 > 0 ) { - int childid = 3; - int chaufid = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - resultsByChild[childPnums[childid]][RESULT_CHILD_ESCORT_TYPE_FIELD] = escortType; - resultsByChild[childPnums[childid]][RESULT_CHILD_BUNDLE_ID_FIELD] = chosenBundle3; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHILD_ID_FIELD] = childid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_ID_FIELD] = chaufid; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PNUM_FIELD] = adultPnums[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PID_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonId(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_PERSON_TYPE_FIELD] = hh.getPerson( adultPnums[chaufid] ).getPersonTypeNumber(); - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_HOME_FIELD] = decisionMaker.getChauffeurDepartForMandatory()[chaufid]; - resultsByChild[childPnums[childid]][RESULT_CHILD_CHAUFFEUR_DEPART_WORK_FIELD] = decisionMaker.getChauffeurDepartFromMandatory()[chaufid]; - } - - return resultsByChild; - - } - - - /** - * Get the results for each chauffeur in the household and escort bundles undertaken. The method returns a two dimensional integer array - * where the first dimension is sized by household size * total possible bundles (3) + 1, and the second dimension is sized by number of - * results fields for chauffeurs. - * - * @param hh - * @param decisionMaker - * @param direction - * @param chosenAlt - * @param chosenBundle1 The bundle for child 1: 0 = not escorted. Max 3 bundles - * @param chosenBundle2 The bundle for child 2: 0 = not escorted. Max 3 bundles - * @param chosenBundle3 The bundle for child 3: 0 = not escorted. Max 3 bundles - * @param chosenChauf1 The chauffeur for child 1: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @param chosenChauf2 The chauffeur for child 2: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @param chosenChauf3 The chauffeur for child 3: 0 = not escorted; 1 = driver 1, rideshare; 2 = driver 1, pure escort; 3 = driver 2, rideshare; 4 = driver 2, pure escort - * @return - */ - private int[][] getResultsByChauffeurArray( Household hh, SchoolEscortingDmu decisionMaker, int direction, int chosenAlt, int chosenBundle1, int chosenBundle2, int chosenBundle3, int chosenChauf1, int chosenChauf2, int chosenChauf3 ) { - - int[][] resultsByChauffeurBundle = new int[hh.getHhSize()*NUM_BUNDLES+1][NUM_RESULTS_BY_CHAUF_FIELDS]; - - - int[] adultPnums = decisionMaker.getChauffeurPnums(); - int[] childPnums = decisionMaker.getEscorteePnums(); - - - // set result attributes for chauffeurs - for ( int pnum=1; pnum <= hh.getHhSize(); pnum++ ) { - Person person = hh.getPerson( pnum ); - for ( int i=1; i <= NUM_BUNDLES; i++ ) { - int rowIndex = (pnum-1)*NUM_BUNDLES+i; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_BUNDLE_ID_FIELD] = i; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_HHID_FIELD] = hh.getHhId(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PNUM_FIELD] = pnum; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PID_FIELD] = person.getPersonId(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_PERSON_TYPE_FIELD] = person.getPersonTypeNumber(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_AGE_FIELD] = person.getAge(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CDAP_FIELD] = person.getCdapIndex(); - } - } - - for ( Person adult : hh.getAdultPersons() ) { - int pnum = adult.getPersonNum(); - for ( int i=1; i <= NUM_BUNDLES; i++ ) { - int rowIndex = (pnum-1)*NUM_BUNDLES+i; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_DIRECTION_FIELD] = direction; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHOSEN_ALT_FIELD] = chosenAlt; - } - } - - for ( int i=1; i < adultPnums.length; i++ ) { - if ( adultPnums[i] > 0 ) { - for ( int j=1; j <= NUM_BUNDLES; j++ ) { - int rowIndex = (adultPnums[i]-1)*NUM_BUNDLES+j; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_ELIGIBLE_FIELD] = ESCORT_ELIGIBLE; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getChauffeurDepartForMandatory()[i] : decisionMaker.getChauffeurDepartFromMandatory()[i]; - } - } - } - - - if ( chosenChauf1 > 0 ) { - int childid = 1; - int chaufid = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf1 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf1 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf1 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - int bundle = chosenBundle1; - int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; - - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_PNUM_FIELD] = childPnums[childid]; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD1_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; - } - - if ( chosenChauf2 > 0 ) { - int childid = 2; - int chaufid = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf2 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf2 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf2 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - int bundle = chosenBundle2; - int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; - - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_PNUM_FIELD] = childPnums[childid]; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD2_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; - } - - if ( chosenChauf3 > 0 ) { - int childid = 3; - int chaufid = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 ? CHAUFFEUR_1 : chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? CHAUFFEUR_2 : 0; - int escortType = chosenChauf3 == RIDE_SHARING_CHAUFFEUR_1 || chosenChauf3 == RIDE_SHARING_CHAUFFEUR_2 ? ModelStructure.RIDE_SHARING_TYPE : chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_1 || chosenChauf3 == PURE_ESCORTING_CHAUFFEUR_2 ? ModelStructure.PURE_ESCORTING_TYPE : 0; - int bundle = chosenBundle3; - int rowIndex = (adultPnums[chaufid]-1)*NUM_BUNDLES + bundle; - - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ID_FIELD] = chaufid; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_ESCORT_TYPE_FIELD] = escortType; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_PNUM_FIELD] = childPnums[childid]; - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_PERSON_TYPE_FIELD] = hh.getPerson( childPnums[childid] ).getPersonTypeNumber(); - resultsByChauffeurBundle[rowIndex][RESULT_CHAUF_CHILD3_DEPART_HOME_FIELD] = direction == DIR_OUTBOUND ? decisionMaker.getEscorteeDepartForSchool()[childid] : decisionMaker.getEscorteeDepartFromSchool()[childid]; - } - - return resultsByChauffeurBundle; - - } - - /** - * Modify and/or create tours based on the results of the school escort model. - * - * @param household - * @param escortBundleList An array of escort bundles. - */ - public void createTours(Household household, List escortBundleList){ - - - //for each bundle - for(SchoolEscortingBundle escortBundle : escortBundleList){ - - //get chauffeur - int chauffeurPnum = escortBundle.getChaufPnum(); - if(chauffeurPnum==0) - continue; - Person chauffeur = household.getPerson(chauffeurPnum); - - //get the list of ordered children in the bundle - int[] childPnums = escortBundle.getChildPnums(); - int[] schoolMAZs = escortBundle.getSchoolMazs(); - - Tour chauffeurTour = null; - int escortType = escortBundle.getEscortType(); - int numStops = 0; - //************************************************************************************************************** - // - // Pure escort tour : Need to create the chauffeur tour - // Also, number of stops(trips) is equal to children - // - //************************************************************************************************************** - if(escortType==ModelStructure.PURE_ESCORTING_TYPE){ - - ArrayList existingTours = chauffeur.getListOfIndividualNonMandatoryTours(); - int id=0; - if(existingTours!=null) - id=existingTours.size(); - else - existingTours = new ArrayList(); - - //generate a non-mandatory escort tour - chauffeurTour = new Tour(id++, household, chauffeur, "Escort", - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY, ModelStructure.ESCORT_PRIMARY_PURPOSE_INDEX); - - chauffeurTour.setTourOrigMgra(household.getHhMgra()); - chauffeurTour.setTourPurpose("Escort"); - - if(escortBundle.getDir() == DIR_OUTBOUND){ - //the destination of the outbound tour is the school MAZ of the last child to drop off - int destMAZ = schoolMAZs[schoolMAZs.length-1]; - chauffeurTour.setTourDestMgra(destMAZ); - }else{ - //the destination of the inbound tour is the school MAZ of the first child to pick up - int destMAZ = schoolMAZs[0]; - chauffeurTour.setTourDestMgra(destMAZ); - } - - int departPeriod = escortBundle.getDepartHome(); - int arrivePeriod = escortBundle.getArriveHome(); - chauffeurTour.setTourDepartPeriod(departPeriod); - chauffeurTour.setTourArrivePeriod(arrivePeriod); - chauffeur.scheduleWindow(departPeriod, arrivePeriod); - chauffeurTour.setValueOfTime(defaultTourVOT); - existingTours.add(chauffeurTour); - numStops = escortBundle.getChildPnums().length; - - } - //************************************************************************************************************** - // - // Ridesharing tour: Need to find chauffeur tour in existing mandatory tour array - // Also, number of stops is equal to children + 1 - // - //************************************************************************************************************** - if(escortType==ModelStructure.RIDE_SHARING_TYPE){ - - // ******************************************************************************************** - // Change the mandatory tour of the chauffeur - // ******************************************************************************************** - ArrayList chauffeurMandatoryTours = getMandatoryTours(chauffeur); - if(chauffeurMandatoryTours.isEmpty()){ - logger.fatal("Error: trying to get mandatory tours for person "+chauffeurPnum+" in household "+household.getHhId()+" for ride-sharing bundle"); - household.logEntireHouseholdObject("Escort model debug", logger); - throw new RuntimeException(); - } - - //number of stops is number of children needing to be dropped off plus one for the primary destination/tour origin - numStops = childPnums.length+1; - - //get tour and find existing tour mode (if already set by this method) - if(escortBundle.getDir()==DIR_OUTBOUND) - chauffeurTour = chauffeurMandatoryTours.get(0); - else{ - chauffeurTour = chauffeurMandatoryTours.get(chauffeurMandatoryTours.size() - 1 ); - } - } // end if rideshare type - - //set tour mode to max of existing tour mode and mode for occupancy - int occupancy = childPnums.length + 1; - - //check for stops in the opposite direction, in order to set the occupancy to the max for the tour - int occupancyInOppositeDirection = 0; - if(escortBundle.getDir()==DIR_OUTBOUND){ - Stop[] stops = chauffeurTour.getInboundStops(); - if(stops!=null) - occupancyInOppositeDirection = stops.length; - }else{ - Stop[] stops = chauffeurTour.getOutboundStops(); - if(stops!=null) - occupancyInOppositeDirection = stops.length; - } - - if(Math.max(occupancy,occupancyInOppositeDirection)==2) - chauffeurTour.setTourModeChoice(SHARED_RIDE_2_MODE); - else - chauffeurTour.setTourModeChoice(SHARED_RIDE_3_MODE); - - - //create stops for each child on the chauffeurs tour if ride-share type, or children - 1 if pure escort - if(escortBundle.getDir()==DIR_OUTBOUND){ - - chauffeurTour.setEscortTypeOutbound(escortType); - chauffeurTour.setDriverPnumOutbound(chauffeurPnum); - - //if this is a pure rideshare tour, and there's only one child, create one stop for the outbound direction and move on. - if(escortType==ModelStructure.PURE_ESCORTING_TYPE && numStops == 1){ - Stop stop = chauffeurTour.createStop( "Home", "Escort", false, false); - stop.setOrig(household.getHhMgra()); - stop.setDest(chauffeurTour.getTourDestMgra()); - stop.setStopPeriod(chauffeurTour.getTourDepartPeriod()); - stop.setMode(SHARED_RIDE_2_MODE); - stop.setValueOfTime(defaultTripVOT); - stop.setEscorteePnumDest(childPnums[0]); - stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); - }//more than one stop on the outbound direction on pure escort or its a rideshare tour - else{ - //insert stops on tour for each child to be escorted - String[] stopOrigPurposes = new String[numStops]; - String[] stopDestPurposes = new String[numStops]; - int[] stopPurposeIndices = new int[numStops]; - stopOrigPurposes[0] = "Home"; - - for(int i = 0; i < numStops-1; ++i){ - if (i > 0) - stopOrigPurposes[i] = stopDestPurposes[i - 1]; - stopPurposeIndices[i] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; - stopDestPurposes[i] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; - - } - stopOrigPurposes[numStops-1] = stopDestPurposes[numStops - 2]; - stopDestPurposes[numStops-1] = chauffeurTour.getTourPrimaryPurpose(); - chauffeurTour.createOutboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); - - Stop[] stops = chauffeurTour.getOutboundStops(); - int origMAZ = household.getHhMgra(); - int escorteePnumOrig=0; - byte escortStopTypeOrig=0; - for (int i = 0; i < stops.length; ++i) { - Stop stop = stops[i]; - - //mode to stop is the occupancy - number of stops so far - int mode = 0; - if(occupancy==1) - mode = DRIVE_ALONE_MODE; - else if(occupancy==2) - mode = SHARED_RIDE_2_MODE; - else - mode = SHARED_RIDE_3_MODE; - stop.setMode(mode); - stop.setValueOfTime(defaultTripVOT); - - //decrement the occupancy - occupancy--; - - //set other information for the stop (really the trip to the school) - stop.setEscorteePnumOrig(escorteePnumOrig); - stop.setEscortStopTypeOrig(escortStopTypeOrig); - stop.setOrig(origMAZ); - stop.setStopPeriod(chauffeurTour.getTourDepartPeriod()); - - if(i 0) - stopOrigPurposes[i] = stopDestPurposes[i - 1]; - stopPurposeIndices[i] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; - stopDestPurposes[i] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; - - } - stopOrigPurposes[numStops-1] = stopDestPurposes[numStops - 2]; - stopDestPurposes[numStops-1] = "HOME"; - chauffeurTour.createInboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); - - Stop[] stops = chauffeurTour.getInboundStops(); - int origMAZ = chauffeurTour.getTourDestMgra(); - - - - //if it is a pure escort tour, the origin of the first inbound stop is escort - int escorteePnumOrig=0; - byte escortStopTypeOrig=0; - if(escortType==ModelStructure.PURE_ESCORTING_TYPE){ - escorteePnumOrig = childPnums[0]; - escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; - occupancy = 2; //driver + child - }else{ - occupancy=1; //driver - } - for (int i = 0; i < stops.length; ++i) { - Stop stop = stops[i]; - int mode = 0; - if(occupancy==1) - mode = DRIVE_ALONE_MODE; - else if(occupancy==2) - mode = SHARED_RIDE_2_MODE; - else - mode = SHARED_RIDE_3_MODE; - stop.setMode(mode); - stop.setValueOfTime(defaultTripVOT); - - //increment the occupancy - occupancy++; - - //set the person being escorted - stop.setEscorteePnumOrig(escorteePnumOrig); - stop.setEscortStopTypeOrig(escortStopTypeOrig); - stop.setStopPeriod(chauffeurTour.getTourArrivePeriod()); - escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; //origin stop type of next stop to this stop type - stop.setOrig(origMAZ); - - - int childIndex = i; - if(escortType==ModelStructure.PURE_ESCORTING_TYPE) - ++childIndex; - - if(i adults = household.getActiveAdultPersons(); - List children = household.getChildPersons(); - - if(adults == null || children == null) - return; - - //cycle thru adults in household - for(Person adult : adults){ - ArrayList mandatoryTours = getMandatoryTours(adult); - ArrayList nonMandatoryTours = adult.getListOfIndividualNonMandatoryTours(); - - ArrayList chauffeurTours = new ArrayList(); - - //add mandatory tours to the potential list of chauffeur tours - if(mandatoryTours!=null) - if(mandatoryTours.size()>0) - chauffeurTours.addAll(mandatoryTours); - - //add non-mandatory tours to the potential list of chauffeur tours - if(nonMandatoryTours!=null) - if(nonMandatoryTours.size()>0) - chauffeurTours.addAll(nonMandatoryTours); - - //if there are no possible chauffeur tours, we're done. - if(chauffeurTours.size()==0){ - return; - } - - //cycle thru mandatory tours for each adult - for(Tour chauffeurTour : chauffeurTours){ - - if(chauffeurTour.getEscortTypeOutbound()==ModelStructure.RIDE_SHARING_TYPE||chauffeurTour.getEscortTypeOutbound()==ModelStructure.PURE_ESCORTING_TYPE){ - - //cycle thru children in household - for(Person child : children){ - - if(child.getListOfSchoolTours() !=null) - recodeSchoolTour(household, child.getPersonNum(), chauffeurTour, DIR_OUTBOUND); - } - } - if(chauffeurTour.getEscortTypeInbound()==ModelStructure.RIDE_SHARING_TYPE||chauffeurTour.getEscortTypeInbound()==ModelStructure.PURE_ESCORTING_TYPE){ - - //cycle thru children in household - for(Person child : children){ - - if(child.getListOfSchoolTours() !=null) - recodeSchoolTour(household, child.getPersonNum(), chauffeurTour, DIR_INBOUND); - } - } - - } - } - } - - /** - * Recode the child's school tour to be consistent with the chauffeurTour for the given direction. If the - * child does not have any school tours, the method will simply return. If there is a school tour, and the - * chauffeur is escorting the child, then the child's stop sequence, tour and trip modes, and other - * relevant data is made consistent with the chauffeur's tour. - * - * @param household Household object for the given child pnum. - * @param childPnum The person number of the child. - * @param chauffeurTour The chauffeurTour to use for checking\coding. - * @param direction Outbound or inbound direction. - */ - public void recodeSchoolTour(Household household, int childPnum, Tour chauffeurTour, int direction){ - - //get child's school tour - Person child = household.getPerson(childPnum); - ArrayList schoolTours = child.getListOfSchoolTours(); - - // no school tours for this child - if(schoolTours.isEmpty()){ - return; - } - Tour schoolTour = schoolTours.get(0); - schoolTour.setTourModeChoice(SHARED_RIDE_2_MODE); - - if(direction==DIR_OUTBOUND){ - Stop[] chauffeurStops = chauffeurTour.getOutboundStops(); - - int driverPnum = chauffeurTour.getDriverPnumOutbound(); - - //loop through chauffeur tour stops - for(int i = 0; i < chauffeurStops.length; ++i){ - Stop chauffeurStop = chauffeurStops[i]; - - int occupancy=0; - if(chauffeurTour.getTourPurpose().equals("Escort")) - occupancy = chauffeurStops.length+1; //occupancy of last trip is equal to number of stops + 1 for first child picked up - else - occupancy = chauffeurStops.length; - - if(chauffeurStop.getEscorteePnumDest()==childPnum){ - - int existingTourMode = schoolTour.getTourModeChoice(); - schoolTour.setTourModeChoice(Math.max(existingTourMode,chauffeurTour.getTourModeChoice())); - schoolTour.setDriverPnumOutbound(driverPnum); - schoolTour.setEscortTypeOutbound(chauffeurTour.getEscortTypeOutbound()); - - //child is first stop; no intermediate stops on this child's school tour in the outbound direction - if(i==0){ - Stop stop = schoolTour.createStop( "Home", "School", false, false); - stop.setOrig(household.getHhMgra()); - stop.setDest(schoolTour.getTourDestMgra()); - stop.setStopPeriod(schoolTour.getTourDepartPeriod()); - if(occupancy==2) - stop.setMode(SHARED_RIDE_2_MODE); - else - stop.setMode(SHARED_RIDE_3_MODE); - stop.setValueOfTime(defaultTripVOT); - stop.setEscorteePnumDest(childPnum); - stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); - break; - } - - //child is second or third stop; create outbound stops array with one or two previous stops. - if(i>0){ - //insert stops on tour for each child to be escorted - String[] stopOrigPurposes = new String[i + 1]; - String[] stopDestPurposes = new String[i + 1]; - int[] stopPurposeIndices = new int[i + 1]; - stopOrigPurposes[0] = "Home"; - - for(int j = 0; j < i; ++j){ - if (j > 0) - stopOrigPurposes[j] = stopDestPurposes[j - 1]; - stopPurposeIndices[j] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; - stopDestPurposes[j] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; - } - stopOrigPurposes[i] = stopDestPurposes[i - 1]; - stopDestPurposes[i] = schoolTour.getTourPrimaryPurpose(); - schoolTour.createOutboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); - - Stop[] stops = schoolTour.getOutboundStops(); - int origMAZ = household.getHhMgra(); - int escorteePnumOrig=0; - byte escortStopTypeOrig=0; - for (int j = 0; j < stops.length; ++j) { - Stop stop = stops[j]; - stop.setOrig(origMAZ); - stop.setDest(chauffeurStops[j].getDest()); - origMAZ = stop.getDest(); - stop.setStopPeriod(schoolTour.getTourDepartPeriod()); - if(occupancy==2) - stop.setMode(SHARED_RIDE_2_MODE); - else - stop.setMode(SHARED_RIDE_3_MODE); - stop.setValueOfTime(defaultTripVOT); - stop.setEscorteePnumOrig(escorteePnumOrig); - stop.setEscortStopTypeOrig(escortStopTypeOrig); - stop.setEscorteePnumDest(chauffeurStops[j].getEscorteePnumDest()); - stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_DROPOFF); - escorteePnumOrig = chauffeurStops[j].getEscorteePnumDest(); - escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_DROPOFF; - } - break; - } - - } //end if found child in chauffeur stop array - --occupancy; - - } //end cycling through stops in outbound direction - - - } //end if in outbound direction - - // things are more complicated in the inbound direction. in this case, the child who is the last to be - // picked up has the simple tour. - if(direction==DIR_INBOUND){ - Stop[] chauffeurStops = chauffeurTour.getInboundStops(); - - int driverPnum = chauffeurTour.getDriverPnumInbound(); - - //loop through chauffeur tour stops from last to first - for(int i = chauffeurStops.length - 1; i >=0; --i){ - Stop chauffeurStop = chauffeurStops[i]; - if(chauffeurStop.getEscorteePnumOrig()==childPnum){ - - int existingTourMode = schoolTour.getTourModeChoice(); - schoolTour.setTourModeChoice(Math.max(existingTourMode,chauffeurTour.getTourModeChoice())); - schoolTour.setDriverPnumInbound(driverPnum); - schoolTour.setEscortTypeInbound(chauffeurTour.getEscortTypeInbound()); - - //child is last stop; no intermediate stops on this child's school tour in the inbound direction - if(i==chauffeurStops.length-1){ - Stop stop = schoolTour.createStop( "School", "Home", true, false); - stop.setOrig(schoolTour.getTourDestMgra()); - stop.setDest(household.getHhMgra()); - stop.setStopPeriod(schoolTour.getTourArrivePeriod()); - int occupancy=0; - if(chauffeurTour.getTourPurpose().equals("Escort")) - occupancy = chauffeurStops.length+1; //occupancy of last trip is equal to number of stops + 1 for first child picked up - else - occupancy = chauffeurStops.length; - if(occupancy==2) - stop.setMode(SHARED_RIDE_2_MODE); - else - stop.setMode(SHARED_RIDE_3_MODE); - stop.setValueOfTime(defaultTripVOT); - stop.setEscorteePnumOrig(childPnum); - stop.setEscortStopTypeOrig(ModelStructure.ESCORT_STOP_TYPE_PICKUP); - break; - } - - //child is not last stop; create inbound stops array with one or two subsequent stops. - if(i 0) - stopOrigPurposes[j] = stopDestPurposes[j - 1]; - stopPurposeIndices[j] = ModelStructure.ESCORT_STOP_PURPOSE_INDEX; - stopDestPurposes[j] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; - } - stopOrigPurposes[numberOfOtherChildrenToPickup] = stopDestPurposes[numberOfOtherChildrenToPickup - 1]; - stopDestPurposes[numberOfOtherChildrenToPickup] = "Home"; - schoolTour.createInboundStops(stopOrigPurposes, stopDestPurposes, stopPurposeIndices); - - Stop[] stops = schoolTour.getInboundStops(); - int origMAZ = chauffeurTour.getTourDestMgra(); - int escorteePnumOrig=0; - byte escortStopTypeOrig=0; - for (int j = 0; j < stops.length; ++j) { - Stop stop = stops[j]; - stop.setOrig(origMAZ); - stop.setDest(chauffeurStops[j].getDest()); - origMAZ = stop.getDest(); - stop.setStopPeriod(schoolTour.getTourDepartPeriod()); - stop.setMode(chauffeurStops[j].getMode()); - stop.setValueOfTime(defaultTripVOT); - stop.setEscorteePnumOrig(escorteePnumOrig); - stop.setEscortStopTypeOrig(escortStopTypeOrig); - stop.setEscorteePnumDest(chauffeurStops[j].getEscorteePnumDest()); - stop.setEscortStopTypeDest(ModelStructure.ESCORT_STOP_TYPE_PICKUP); - escorteePnumOrig = chauffeurStops[j].getEscorteePnumDest(); - escortStopTypeOrig = ModelStructure.ESCORT_STOP_TYPE_PICKUP; - } - break; - } - - } //end if found child in chauffeur stop array - } //end cycling through stops in inbound direction - - } //end if inbound direction - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java deleted file mode 100644 index deb2c80..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceModel.java +++ /dev/null @@ -1,603 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class SchoolLocationChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(SchoolLocationChoiceModel.class); - private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); - - // this constant used as a dimension for saving distance and logsums for - // alternatives in samples - private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; - - private static final int DC_DATA_SHEET = 0; - - private MgraDataManager mgraManager; - private DestChoiceSize dcSizeObj; - - private DestChoiceTwoStageModelDMU dcTwoStageDmuObject; - - private DestChoiceTwoStageModel dcTwoStageModelObject; - private TourModeChoiceModel mcModel; - - private String[] segmentNameList; - private HashMap segmentNameIndexMap; - - private int[] dcModelIndices; - - // A ChoiceModelApplication object and modeAltsAvailable[] is needed for - // each purpose - private ChoiceModelApplication[] locationChoiceModels; - private ChoiceModelApplication locationChoiceModel; - - private boolean[] dcModelAltsAvailable; - private int[] dcModelAltsSample; - private int[] dcModelSampleValues; - - private int[] uecSheetIndices; - - int origMgra; - - private int modelIndex; - private int shadowPricingIteration; - - private double[] sampleAlternativeDistances; - private double[] sampleAlternativeLogsums; - - private double[] mgraDistanceArray; - - private BuildAccessibilities aggAcc; - - private int soaSampleSize; - - private long soaRunTime; - - public SchoolLocationChoiceModel(int index, HashMap propertyMap, - DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, String dcUecFileName, - String soaUecFile, int soaSampleSize, String modeChoiceUecFile, - CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel, - double[][][] schoolSizeProbs, double[][][] schoolTazDistProbs) - { - - this.aggAcc = aggAcc; - this.dcSizeObj = dcSizeObj; - this.mcModel = mcModel; - this.soaSampleSize = soaSampleSize; - - modelIndex = index; - - mgraManager = MgraDataManager.getInstance(); - - dcTwoStageDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); - dcTwoStageDmuObject.setAggAcc(this.aggAcc); - - dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); - dcTwoStageModelObject.setTazDistProbs(schoolTazDistProbs); - dcTwoStageModelObject.setMgraSizeProbs(schoolSizeProbs); - - shadowPricingIteration = 0; - - sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - - } - - public void setupSchoolSegments() - { - aggAcc.createSchoolSegmentNameIndices(); - uecSheetIndices = aggAcc.getSchoolDcUecSheets(); - segmentNameList = aggAcc.getSchoolSegmentNameList(); - } - - public void setupDestChoiceModelArrays(HashMap propertyMap, - String dcUecFileName, String soaUecFile, int soaSampleSize) - { - - segmentNameIndexMap = dcSizeObj.getSegmentNameIndexMap(); - - // create a lookup array to map purpose index to model index - dcModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int dcModelIndex = 0; - int dcSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, dcModelIndex); - dcModelIndices[dcSegmentIndex] = dcModelIndex++; - } else - { - dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); - } - - dcSegmentIndex++; - } - // the value of dcModelIndex is the number of ChoiceModelApplication - // objects to create - // the modelIndexMap keys are the uec sheets to use in building - // ChoiceModelApplication objects - - locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; - - int i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - - int modelIndex = -1; - try - { - modelIndex = modelIndexMap.get(uecIndex); - locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, - uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", - i, modelIndex, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - dcModelAltsAvailable = new boolean[soaSampleSize + 1]; - dcModelAltsSample = new int[soaSampleSize + 1]; - dcModelSampleValues = new int[soaSampleSize]; - - mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; - - } - - public void applySchoolLocationChoice(Household hh) - { - - if (hh.getDebugChoiceModels()) - { - String label = String.format("Pre school Location Choice HHId=%d Object", hh.getHhId()); - hh.logHouseholdObject(label, dcManLogger); - } - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - - int homeMgra = hh.getHhMgra(); - Person[] persons = hh.getPersons(); - - int tourNum = 0; - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - int segmentIndex = -1; - int segmentType = -1; - if (p.getPersonIsPreschoolChild() == 1 || p.getPersonIsStudentNonDriving() == 1 - || p.getPersonIsStudentDriving() == 1 || p.getPersonIsUniversityStudent() == 1) - { - - if (p.getPersonIsPreschoolChild() == 1) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.PRESCHOOL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.PRESCHOOL_ALT_INDEX; - } else if (p.getPersonIsGradeSchool() == 1) - { - segmentIndex = aggAcc.getMgraGradeSchoolSegmentIndex(homeMgra); - segmentType = BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX; - } else if (p.getPersonIsHighSchool() == 1) - { - segmentIndex = aggAcc.getMgraHighSchoolSegmentIndex(homeMgra); - segmentType = BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX; - } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() < 30) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_TYPICAL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX; - } else if (p.getPersonIsUniversityStudent() == 1 && p.getAge() >= 30) - { - segmentIndex = segmentNameIndexMap - .get(BuildAccessibilities.SCHOOL_DC_SIZE_SEGMENT_NAME_LIST[BuildAccessibilities.UNIV_NONTYPICAL_SEGMENT_GROUP_INDEX]); - segmentType = BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX; - } - - // if person type is a student but segment index is -1, the - // person is not enrolled - // assume home schooled - if (segmentIndex < 0) - { - p.setSchoolLocationSegmentIndex(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); - p.setSchoolLoc(ModelStructure.NOT_ENROLLED_SEGMENT_INDEX); - p.setSchoolLocDistance(0); - p.setSchoolLocLogsum(-999); - continue; - } else - { - // if the segment is in the skip shadow pricing set, and the - // iteration is > 0, dont' compute new choice - if (shadowPricingIteration == 0 - || !dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) - p.setSchoolLocationSegmentIndex(segmentIndex); - } - - if (segmentType < 0) - { - segmentType = ModelStructure.NOT_ENROLLED_SEGMENT_INDEX; - } - } else // not a student person type - { - p.setSchoolLocationSegmentIndex(-1); - p.setSchoolLoc(0); - p.setSchoolLocDistance(0); - p.setSchoolLocLogsum(-999); - continue; - } - - // save person information in decision maker label, and log person - // object - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p - .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); - hh.logPersonObject(decisionMakerLabel, dcManLogger, p); - } - - // if the segment is in the skip shadow pricing set, and the - // iteration is > 0, dont' compute new choice - if (shadowPricingIteration > 0 && dcSizeObj.getSegmentIsInSkipSegmentSet(segmentIndex)) - continue; - - double[] results = null; - int modelIndex = 0; - try - { - - origMgra = homeMgra; - - // update the DC dmuObject for this person - dcTwoStageDmuObject.setHouseholdObject(hh); - dcTwoStageDmuObject.setPersonObject(p); - dcTwoStageDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); - - double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[segmentIndex]; - mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, - mgraDistanceArray); - - // set size array for the tour segment and distance array from - // the home mgra to all destinaion mgras. - dcTwoStageDmuObject.setMgraSizeArray(homeMgraSizeArray); - dcTwoStageDmuObject.setMgraDistanceArray(mgraDistanceArray); - - modelIndex = dcModelIndices[segmentIndex]; - locationChoiceModel = locationChoiceModels[modelIndex]; - - // get the school location alternative chosen from the sample - results = selectLocationFromSampleOfAlternatives("school", segmentType, p, - segmentNameList[segmentIndex], segmentIndex, tourNum++, homeMgraSizeArray, - mgraDistanceArray); - - } catch (RuntimeException e) - { - logger.fatal(String - .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in school location choice, modelIndex=%d, segmentType=%d, segmentIndex=%d, segmentName=%s", - i, hh.getHhId(), i, modelIndex, segmentType, segmentIndex, - segmentNameList[segmentIndex])); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - p.setSchoolLoc((int) results[0]); - p.setSchoolLocDistance((float) results[1]); - p.setSchoolLocLogsum((float) results[2]); - - } - - } - - /** - * - * @return an array with chosen mgra, distance to chosen mgra, and logsum to - * chosen mgra. - */ - private double[] selectLocationFromSampleOfAlternatives(String segmentType, - int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, - int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcManLogger; - - Household household = person.getHouseholdObject(); - - // get sample of locations and correction factors for sample using the - // alternate method - dcTwoStageModelObject.chooseSample(household.getHhTaz(), sizeSegmentIndex, - segmentTypeIndex, soaSampleSize, household.getHhRandom(), - household.getDebugChoiceModels()); - int[] finalSample = dcTwoStageModelObject.getUniqueSampleMgras(); - double[] sampleCorrectionFactors = dcTwoStageModelObject - .getUniqueSampleMgraCorrectionFactors(); - int numUniqueAlts = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); - - Arrays.fill(dcModelAltsAvailable, false); - Arrays.fill(dcModelAltsSample, 0); - Arrays.fill(dcModelSampleValues, 999999); - - // set sample of alternatives correction factors used in destination - // choice utility for the sampled alternatives. - dcTwoStageDmuObject.setDcSoaCorrections(sampleCorrectionFactors); - - // for the destination mgras in the sample, compute mc logsums and save - // in dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 0; i < numUniqueAlts; i++) - { - - int destMgra = finalSample[i]; - dcModelSampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); - dcTwoStageDmuObject.setMcLogsum(i, logsum); - - sampleAlternativeLogsums[i] = logsum; - sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; - - // set availaibility and sample values for the purpose, dcAlt. - dcModelAltsAvailable[i + 1] = true; - dcModelAltsSample[i + 1] = 1; - - } - - dcTwoStageDmuObject.setSampleArray(dcModelSampleValues); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tourNum); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" - + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " - + person.getPersonType() + ", TourNum=" + tourNum); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - locationChoiceModel.computeUtilities(dcTwoStageDmuObject, - dcTwoStageDmuObject.getDmuIndexValues(), dcModelAltsAvailable, dcModelAltsSample); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (locationChoiceModel.getAvailabilityCount() > 0) - { - try - { - chosen = locationChoiceModel.getChoiceResult(rn); - } catch (Exception e) - { - } - } else - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", - dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject - .getPersonObject().getPersonNum(), segmentName)); - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = locationChoiceModel.getUtilities(); - double[] probabilities = locationChoiceModel.getProbabilities(); - boolean[] availabilities = locationChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb Distance Logsum"); - modelLogger - .info("--------------------- -------------- -------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 1; j < finalSample.length; j++) - { - int alt = finalSample[j]; - cumProb += probabilities[j]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - if (chosen > 0) - { - String altString = String.format("j=%d, mgra=%d", chosen - 1, - finalSample[chosen - 1]); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - } else - { - String altString = String.format("No Chosen Alt, availability count = %d", - locationChoiceModel.getAvailabilityCount()); - modelLogger.info(altString); - } - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - - // write UEC calculation results to separate model specific log file - locationChoiceModel.logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", - dcTwoStageDmuObject.getHouseholdObject().getHhId(), - dcTwoStageDmuObject.getPersonObject().getPersonNum(), segmentName)); - System.exit(-1); - } - - } - - double[] returnArray = new double[3]; - - returnArray[0] = finalSample[chosen - 1]; - returnArray[1] = sampleAlternativeDistances[chosen - 1]; - returnArray[2] = sampleAlternativeLogsums[chosen - 1]; - - return returnArray; - - } - - private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, - int segmentTypeIndex) - { - - int purposeIndex = 0; - String purpose = ""; - if (segmentTypeIndex < 0) - { - purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } - - // create a temporary tour to use to calculate mode choice logsum - Tour mcLogsumTour = new Tour(person, 0, purposeIndex); - mcLogsumTour.setTourPurpose(purpose); - mcLogsumTour.setTourOrigMgra(household.getHhMgra()); - mcLogsumTour.setTourDestMgra(sampleDestMgra); - mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); - mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - - if (household.getDebugChoiceModels()) - { - dcManLogger.info(""); - dcManLogger.info(""); - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, - dcManLogger, person); - } - - double logsum = -1; - try - { - logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, - choiceModelDescription, decisionMakerLabel); - } catch (Exception e) - { - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - e.printStackTrace(); - System.exit(-1); - } - - return logsum; - } - - public int getModelIndex() - { - return modelIndex; - } - - public void setDcSizeObject(DestChoiceSize dcSizeObj) - { - this.dcSizeObj = dcSizeObj; - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java deleted file mode 100644 index b162282..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SchoolLocationChoiceTaskJppf.java +++ /dev/null @@ -1,238 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.net.UnknownHostException; -import java.util.Date; -import java.util.HashMap; -import org.jppf.node.protocol.AbstractTask; -import org.jppf.node.protocol.DataProvider; -import com.pb.common.calculator.MatrixDataServerIf; - -public class SchoolLocationChoiceTaskJppf - extends AbstractTask -{ - - private static String VERSION = "Task.1.0.3"; - - private transient HashMap propertyMap; - private transient MatrixDataServerIf ms; - private transient HouseholdDataManagerIf hhDataManager; - private transient ModelStructure modelStructure; - private transient String tourCategory; - private transient DestChoiceSize dcSizeObj; - private transient String dcUecFileName; - private transient String soaUecFileName; - private transient int soaSampleSize; - private transient CtrampDmuFactoryIf dmuFactory; - private transient String restartModelString; - - private int iteration; - private int startIndex; - private int endIndex; - private int taskIndex = -1; - - public SchoolLocationChoiceTaskJppf(int taskIndex, int startIndex, int endIndex, int iteration) - { - this.startIndex = startIndex; - this.endIndex = endIndex; - this.taskIndex = taskIndex; - this.iteration = iteration; - } - - public void run() - { - - String start = (new Date()).toString(); - long startTime = System.currentTimeMillis(); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " - + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - // logger.info( String.format( - // "startTime=%d, task=%d run(), thread=%s, start=%d, end=%d.", - // startTime, - // taskIndex, threadName, startIndex, - // endIndex ) ); - - try - { - DataProvider dataProvider = getDataProvider(); - - this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); - this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); - this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); - this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); - this.tourCategory = (String) dataProvider.getParameter("tourCategory"); - this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); - this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); - this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); - this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); - this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); - this.restartModelString = (String) dataProvider.getParameter("restartModelString"); - - } catch (Exception e) - { - e.printStackTrace(); - } - - // HouseholdChoiceModelsManager hhModelManager = - // HouseholdChoiceModelsManager.getInstance( - // propertyMap, restartModelString, modelStructure, dmuFactory); - // hhModelManager.clearHhModels(); - // hhModelManager = null; - - // get the factory object used to create and recycle dcModel objects. - DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); - - // one of tasks needs to initialize the manager object by passing - // attributes - // needed to create a destination choice model object. - modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, - soaSampleSize, dmuFactory, restartModelString); - - // get a dcModel object from manager, which either creates one or - // returns one - // for re-use. - MandatoryDestChoiceModel dcModel = modelManager.getDcSchoolModelObject(taskIndex, - iteration, dcSizeObj); - - // logger.info( String.format( - // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, - // taskIndex, - // threadName, startIndex, endIndex ) ); - System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", - new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); - - long setup1 = (System.currentTimeMillis() - startTime) / 1000; - - Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); - - long setup2 = (System.currentTimeMillis() - startTime) / 1000; - // logger.info( String.format( - // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", - // taskIndex, startIndex, endIndex, - // threadName, setup1, setup2 ) ); - System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", - new Date(), taskIndex, startIndex, endIndex, threadName)); - - int i = -1; - try - { - - boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, - HouseholdDataManager.DEBUG_HHS_ONLY_KEY); - - for (i = 0; i < householdArray.length; i++) - { - // for debugging only - process only household objects specified - // for debugging, if property key was set to true - if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; - - dcModel.applySchoolLocationChoice(householdArray[i]); - } - - hhDataManager.setHhArray(householdArray, startIndex); - - //check to make sure hh array got set in hhDataManager - boolean allHouseholdsAreSame = false; - while(!allHouseholdsAreSame) { - Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); - for(int j = 0; j< householdArrayRemote.length;++j) { - - Household remoteHousehold = householdArrayRemote[j]; - Household localHousehold = householdArray[j]; - - allHouseholdsAreSame = checkIfSameSchoolLocationResults(remoteHousehold, localHousehold); - - if(!allHouseholdsAreSame) - break; - } - if(!allHouseholdsAreSame) { - System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with school location choice results; updating"); - hhDataManager.setHhArray(householdArray, startIndex); - - } - } - - - } catch (Exception e) - { - if (i >= 0 && i < householdArray.length) System.out - .println(String - .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", - taskIndex, i, householdArray[i].getHhId(), startIndex)); - else System.out.println(String.format( - "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", - taskIndex, i, startIndex)); - System.out.println("Exception caught:"); - e.printStackTrace(); - System.out.println("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; - long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; - // logger.info( String.format( - // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, - // threadName, getHhs, processHhs ) ); - System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, - threadName)); - - long total = (System.currentTimeMillis() - startTime) / 1000; - String resultString = String - .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", - threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, - setup2, getHhs, processHhs, total); - // logger.info( resultString ); - setResult(resultString); - - modelManager.returnDcSchoolModelObject(dcModel, taskIndex, startIndex, endIndex); - - } - - /** - * Returns true if school location results are the same, else returns false. - * - * @param thisHousehold - * @param thatHousehold - * @return true or false - */ - public boolean checkIfSameSchoolLocationResults(Household thisHousehold, Household thatHousehold) { - - Person[] thisPersons = thisHousehold.getPersons(); - Person[] thatPersons = thatHousehold.getPersons(); - - if(thisPersons.length!=thatPersons.length) - return false; - - for(int k=1;k -{ - - private static String VERSION = "Task.1.0.3"; - - private transient HashMap propertyMap; - private transient MatrixDataServerIf ms; - private transient HouseholdDataManagerIf hhDataManager; - private transient ModelStructure modelStructure; - private transient String tourCategory; - private transient DestChoiceSize dcSizeObj; - private transient String dcUecFileName; - private transient String soaUecFileName; - private transient int soaSampleSize; - private transient CtrampDmuFactoryIf dmuFactory; - private transient String restartModelString; - - private int iteration; - private int startIndex; - private int endIndex; - private int taskIndex = -1; - - public SchoolLocationChoiceTaskJppfNew(int taskIndex, int startIndex, int endIndex, - int iteration) - { - this.startIndex = startIndex; - this.endIndex = endIndex; - this.taskIndex = taskIndex; - this.iteration = iteration; - } - - public void run() - { - - String start = (new Date()).toString(); - long startTime = System.currentTimeMillis(); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " - + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - // logger.info( String.format( - // "startTime=%d, task=%d run(), thread=%s, start=%d, end=%d.", - // startTime, - // taskIndex, threadName, startIndex, - // endIndex ) ); - - try - { - DataProvider dataProvider = getDataProvider(); - - this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); - this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); - this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); - this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); - this.tourCategory = (String) dataProvider.getParameter("tourCategory"); - this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); - this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); - this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); - this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); - this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); - this.restartModelString = (String) dataProvider.getParameter("restartModelString"); - - } catch (Exception e) - { - e.printStackTrace(); - } - - // HouseholdChoiceModelsManager hhModelManager = - // HouseholdChoiceModelsManager.getInstance( - // propertyMap, restartModelString, modelStructure, dmuFactory); - // hhModelManager.clearHhModels(); - // hhModelManager = null; - - // get the factory object used to create and recycle dcModel objects. - DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); - - // one of tasks needs to initialize the manager object by passing - // attributes - // needed to create a destination choice model object. - modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, - soaSampleSize, dmuFactory, restartModelString); - - // get a dcModel object from manager, which either creates one or - // returns one - // for re-use. - SchoolLocationChoiceModel dcModel = modelManager.getSchoolLocModelObject(taskIndex, - iteration, dcSizeObj); - - // logger.info( String.format( - // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, - // taskIndex, - // threadName, startIndex, endIndex ) ); - System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", - new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); - - long setup1 = (System.currentTimeMillis() - startTime) / 1000; - - Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); - - long setup2 = (System.currentTimeMillis() - startTime) / 1000; - // logger.info( String.format( - // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", - // taskIndex, startIndex, endIndex, - // threadName, setup1, setup2 ) ); - System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", - new Date(), taskIndex, startIndex, endIndex, threadName)); - - int i = -1; - try - { - - boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, - HouseholdDataManager.DEBUG_HHS_ONLY_KEY); - - for (i = 0; i < householdArray.length; i++) - { - // for debugging only - process only household objects specified - // for debugging, if property key was set to true - if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; - - dcModel.applySchoolLocationChoice(householdArray[i]); - } - - hhDataManager.setHhArray(householdArray, startIndex); - - //check to make sure hh array got set in hhDataManager - boolean allHouseholdsAreSame = false; - while(!allHouseholdsAreSame) { - Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); - for(int j = 0; j< householdArrayRemote.length;++j) { - - Household remoteHousehold = householdArrayRemote[j]; - Household localHousehold = householdArray[j]; - - allHouseholdsAreSame = checkIfSameSchoolLocationResults(remoteHousehold, localHousehold); - - if(!allHouseholdsAreSame) - break; - } - if(!allHouseholdsAreSame) { - System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with school location choice results; updating"); - hhDataManager.setHhArray(householdArray, startIndex); - - } - } - - } catch (Exception e) - { - if (i >= 0 && i < householdArray.length) System.out - .println(String - .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", - taskIndex, i, householdArray[i].getHhId(), startIndex)); - else System.out.println(String.format( - "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", - taskIndex, i, startIndex)); - System.out.println("Exception caught:"); - e.printStackTrace(); - System.out.println("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; - long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; - // logger.info( String.format( - // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, - // threadName, getHhs, processHhs ) ); - System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, - threadName)); - - long total = (System.currentTimeMillis() - startTime) / 1000; - String resultString = String - .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", - threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, - setup2, getHhs, processHhs, total); - // logger.info( resultString ); - setResult(resultString); - - modelManager.returnSchoolLocModelObject(dcModel, taskIndex, startIndex, endIndex); - - clearClassAttributes(); - } - - /** - * Returns true if school location results are the same, else returns false. - * - * @param thisHousehold - * @param thatHousehold - * @return true or false - */ - public boolean checkIfSameSchoolLocationResults(Household thisHousehold, Household thatHousehold) { - - Person[] thisPersons = thisHousehold.getPersons(); - Person[] thatPersons = thatHousehold.getPersons(); - - if(thisPersons.length!=thatPersons.length) - return false; - - for(int k=1;k - * The type that the matrices are segmented against. - */ -public interface SegmentedSparseMatrix { - /** - * Get the value of the matrix for a specified segment and row/column ids. - * - * @param segment - * The segment. - * - * @param rowId - * The row id. - * - * @param columnId - * The column id. - * - * @return the matrix value at (rowId,columnId) for {@code segment}. - */ - double getValue(S segment, int rowId, int columnId); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java deleted file mode 100644 index 740cbb0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SoaDMU.java +++ /dev/null @@ -1,10 +0,0 @@ -package org.sandag.abm.ctramp; - -/** - * @author crf
    - * Started: Nov 15, 2008 3:25:49 PM - */ -public interface SoaDMU -{ - Household getHouseholdObject(); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java deleted file mode 100644 index 73c7178..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SqliteService.java +++ /dev/null @@ -1,118 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.sql.Connection; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Statement; - -public class SqliteService -{ - - Connection c = null; - String databaseFile; - - public void connect(String fileName, String tableName) throws DAOException - { - - try - { - c = ConnectionHelper.getConnection(fileName); - Statement s = c.createStatement(); - - s.execute("CREATE TABLE IF NOT EXISTS " + tableName + " (" + " id INTEGER, " - + " numProcessed INTEGER, " + " totalToProcess INTEGER, " - + " startUp INTEGER, " + " runTime INTEGER, " + " shutDown INTEGER " - + ")"); - - s.execute("DELETE FROM " + tableName); - - } catch (SQLException e) - { - e.printStackTrace(); - throw new DAOException(e); - } - - } - - public void listRecords(String tableName) throws DAOException - { - - try - { - - Statement s = c.createStatement(); - - ResultSet rs = s - .executeQuery("SELECT id, numProcessed, totalToProcess, startUp, runTime, shutDown FROM " - + tableName + " ORDER BY id"); - while (rs.next()) - { - System.out.println(rs.getInt("id") + ", " + rs.getInt("numProcessed") + ", " - + rs.getInt("totalToProcess") + ", " + rs.getInt("startUp") + ", " - + rs.getInt("runTime") + ", " + rs.getInt("shutDown")); - } - - } catch (SQLException e) - { - e.printStackTrace(); - throw new DAOException(e); - } - - } - - public void insertRecord(String tableName, int id, int numProcessed, int totalToProcess, - int startUp, int runTime, int shutDown) throws DAOException - { - - try - { - - Statement s = c.createStatement(); - String query = String - .format("INSERT INTO %s (id, numProcessed, totalToProcess, startUp, runTime, shutDown) VALUES (%d, %d, %d, %d, %d, %d)", - tableName, id, numProcessed, totalToProcess, startUp, runTime, shutDown); - s.execute(query); - - } catch (SQLException e) - { - e.printStackTrace(); - throw new DAOException(e); - } - - } - - public void updateRecord(String tableName, int id, int numProcessed, int totalToProcess, - int startUp, int runTime, int shutDown) throws DAOException - { - - try - { - - Statement s = c.createStatement(); - String query = String - .format("UPDATE %s SET numProcessed=%d, totalToProcess=%d, startUp=%d, runTime=%d, shutDown=%d WHERE id=%d", - tableName, numProcessed, totalToProcess, startUp, runTime, shutDown, id); - s.execute(query); - - } catch (SQLException e) - { - e.printStackTrace(); - throw new DAOException(e); - } - - } - - public static void main(String[] args) - { - - SqliteService s = new SqliteService(); - s.connect("c:/jim/status.db", "uwsl"); - - s.insertRecord("uwsl", 0, 27, 1250, 99, 102, 10); - s.insertRecord("uwsl", 2, 29, 1250, 58, 101, 9); - s.insertRecord("uwsl", 1, 32, 1250, 77, 99, 8); - s.listRecords("uwsl"); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java deleted file mode 100644 index 882b5b8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Stop.java +++ /dev/null @@ -1,304 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import org.apache.log4j.Logger; - -public class Stop - implements Serializable -{ - - static final byte STOP_TYPE_PICKUP = 1; - static final byte STOP_TYPE_DROPOFF = 2; - static final byte STOP_TYPE_OTHER = 3; - int id; - int orig; - int dest; - int park; - int mode; - byte micromobilityWalkMode; - byte micromobilityAccessMode; - byte micromobilityEgressMode; - float micromobilityWalkLogsum; - float micromobilityAccessLogsum; - float micromobilityEgressLogsum; - private float modeLogsum; - private float parkingCost; - - int stopPeriod; - int boardTap; - int alightTap; - boolean inbound; - int set = -1; - - private int escorteePnumOrig; - private byte escortStopTypeOrig; - private int escorteePnumDest; - private byte escortStopTypeDest; - - String origPurpose; - String destPurpose; - int stopPurposeIndex; - - Tour parentTour; - - private double valueOfTime; - - public Stop(Tour parentTour, String origPurpose, String destPurpose, int id, boolean inbound, - int stopPurposeIndex) - { - this.parentTour = parentTour; - this.origPurpose = origPurpose; - this.destPurpose = destPurpose; - this.stopPurposeIndex = stopPurposeIndex; - this.id = id; - this.inbound = inbound; - } - - public void setOrig(int orig) - { - this.orig = orig; - } - - public void setDest(int dest) - { - this.dest = dest; - } - - public void setPark(int park) - { - this.park = park; - } - - public void setMode(int mode) - { - this.mode = mode; - } - - public void setSet(int Skimset) - { - set = Skimset; - } - - public void setBoardTap(int tap) - { - boardTap = tap; - } - - public void setAlightTap(int tap) - { - alightTap = tap; - } - - public void setStopPeriod(int period) - { - stopPeriod = period; - } - - public int getOrig() - { - return orig; - } - - public int getDest() - { - return dest; - } - - public int getPark() - { - return park; - } - - public String getOrigPurpose() - { - return origPurpose; - } - - public String getDestPurpose() - { - return destPurpose; - } - - public void setOrigPurpose(String purpose) - { - origPurpose=purpose; - } - public void setDestPurpose(String purpose){ - destPurpose = purpose; - } - public int getStopPurposeIndex() - { - return stopPurposeIndex; - } - - public int getMode() - { - return mode; - } - public int getSet() - { - return set; - } - - public int getBoardTap() - { - return boardTap; - } - - public int getAlightTap() - { - return alightTap; - } - - public int getStopPeriod() - { - return stopPeriod; - } - - public Tour getTour() - { - return parentTour; - } - - public boolean isInboundStop() - { - return inbound; - } - - public int getStopId() - { - return id; - } - - public int getEscorteePnumOrig() { - return escorteePnumOrig; - } - - public void setEscorteePnumOrig(int escorteePnum) { - this.escorteePnumOrig = escorteePnum; - } - - public byte getEscortStopTypeOrig() { - return escortStopTypeOrig; - } - - public void setEscortStopTypeOrig(byte stopType) { - this.escortStopTypeOrig = stopType; - } - - public int getEscorteePnumDest() { - return escorteePnumDest; - } - - public void setEscorteePnumDest(int escorteePnum) { - this.escorteePnumDest = escorteePnum; - } - - public byte getEscortStopTypeDest() { - return escortStopTypeDest; - } - - public void setEscortStopTypeDest(byte stopType) { - this.escortStopTypeDest = stopType; - } - - public float getModeLogsum() { - return modeLogsum; - } - - public void setModeLogsum(float modeLogsum) { - this.modeLogsum = modeLogsum; - } - public double getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(double valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public void setMicromobilityWalkMode(byte micromobilityWalkMode) { - this.micromobilityWalkMode=micromobilityWalkMode; - } - - public byte getMicromobilityWalkMode() { - return micromobilityWalkMode; - } - public float getMicromobilityWalkLogsum() { - return micromobilityWalkLogsum; - } - - public void setMicromobilityWalkLogsum(float micromobilityWalkLogsum) { - this.micromobilityWalkLogsum = micromobilityWalkLogsum; - } - - public byte getMicromobilityAccessMode() { - return micromobilityAccessMode; - } - - public void setMicromobilityAccessMode(byte micromobilityAccessMode) { - this.micromobilityAccessMode = micromobilityAccessMode; - } - - public byte getMicromobilityEgressMode() { - return micromobilityEgressMode; - } - - public void setMicromobilityEgressMode(byte micromobilityEgressMode) { - this.micromobilityEgressMode = micromobilityEgressMode; - } - - public float getMicromobilityAccessLogsum() { - return micromobilityAccessLogsum; - } - - public void setMicromobilityAccessLogsum(float micromobilityAccessLogsum) { - this.micromobilityAccessLogsum = micromobilityAccessLogsum; - } - - public float getMicromobilityEgressLogsum() { - return micromobilityEgressLogsum; - } - - public void setMicromobilityEgressLogsum(float micromobilityEgressLogsum) { - this.micromobilityEgressLogsum = micromobilityEgressLogsum; - } - - public float getParkingCost() { - return parkingCost; - } - - public void setParkingCost(float parkingCost) { - this.parkingCost = parkingCost; - } - - public void logStopObject(Logger logger, int totalChars) - { - - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - Household.logHelper(logger, "stopId: ", id, totalChars); - Household.logHelper(logger, "origPurpose: ", origPurpose, totalChars); - Household.logHelper(logger, "destPurpose: ", destPurpose, totalChars); - Household.logHelper(logger, "orig: ", orig, totalChars); - Household.logHelper(logger, "dest: ", dest, totalChars); - Household.logHelper(logger, "mode: ", mode, totalChars); - Household.logHelper(logger, "value of time: ", ((float)valueOfTime), totalChars); - Household.logHelper(logger, "boardTap: ", boardTap, totalChars); - Household.logHelper(logger, "alightTap: ", alightTap, totalChars); - Household.logHelper(logger, "TapSet: ", set, totalChars); - Household.logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); - Household.logHelper( logger, "stopPeriod: ", stopPeriod, totalChars ); - Household.logHelper( logger, "orig escort stop type: ",escortStopTypeOrig, totalChars); - Household.logHelper( logger, "orig escortee pnum: ",escorteePnumOrig, totalChars); - Household.logHelper( logger, "dest escort stop type: ",escortStopTypeDest, totalChars); - Household.logHelper( logger, "dest escortee pnum: ",escorteePnumDest, totalChars); - logger.info(separater); - logger.info(""); - logger.info(""); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java deleted file mode 100644 index ab675da..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDCSoaDMU.java +++ /dev/null @@ -1,167 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import com.pb.common.calculator.VariableTable; - -/** - * @author crf
    - * Started: Nov 14, 2008 3:32:58 PM - */ -public class StopDCSoaDMU - implements Serializable, VariableTable -{ - - protected HashMap methodIndexMap; - - protected int tourModeIndex; - protected int[] walkTransitAvailableAtMgra; - protected double origDestDistance; - protected double[] distancesFromOrigMgra; - protected double[] distancesToDestMgra; - protected double[] logSizeTerms; - protected ModelStructure modelStructure; - - public StopDCSoaDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - } - - /** - * set the array of distance values from the origin MGRA of the stop to all - * MGRAs. - * - * @param distances - */ - public void setDistancesFromOrigMgra(double[] distances) - { - distancesFromOrigMgra = distances; - } - - /** - * set the array of distance values from all MGRAs to the final destination - * MGRA of the stop. - * - * @param distances - */ - public void setDistancesToDestMgra(double[] distances) - { - distancesToDestMgra = distances; - } - - /** - * set the OD distance value from the stop origin MGRA to the final - * destination MGRA of the stop. - * - * @param distances - */ - public void setOrigDestDistance(double distance) - { - origDestDistance = distance; - } - - /** - * set the tour mode index value for the tour of the stop being located - * - * @param tour - */ - public void setTourModeIndex(int index) - { - tourModeIndex = index; - } - - /** - * set the array of attributes for all MGRAs that says their is walk transit - * access for the indexed mgra - * - * @param tour - */ - public void setWalkTransitAvailable(int[] avail) - { - walkTransitAvailableAtMgra = avail; - } - - /** - * set the array of logged size terms for all MGRAs for the stop being - * located - * - * @param size - */ - public void setLnSlcSizeAlt(double[] size) - { - logSizeTerms = size; - } - - public double getOrigToMgraDistanceAlt(int alt) - { - return distancesFromOrigMgra[alt]; - } - - public double getMgraToDestDistanceAlt(int alt) - { - return distancesToDestMgra[alt]; - } - - public double getOdDistance() - { - return origDestDistance; - } - - public int getTourModeIsWalk() - { - boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tourModeIndex); - return tourModeIsWalk ? 1 : 0; - } - - public int getTourModeIsBike() - { - boolean tourModeIsBike = modelStructure.getTourModeIsBike(tourModeIndex); - return tourModeIsBike ? 1 : 0; - } - - public int getTourModeIsWalkTransit() - { - return (modelStructure.getTourModeIsWalkTransit(tourModeIndex) ? 1 : 0); - } - - public int getWalkTransitAvailableAlt(int alt) - { - return walkTransitAvailableAtMgra[alt]; - } - - public double getLnSlcSizeAlt(int alt) - { - return logSizeTerms[alt]; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java deleted file mode 100644 index 3c1e437..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDepartArrivePeriodModel.java +++ /dev/null @@ -1,144 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class will be used for determining the number of stops on individual - * mandatory, individual non-mandatory and joint tours. - * - * @author Christi Willison - * @version Nov 4, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public class StopDepartArrivePeriodModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(StopDepartArrivePeriodModel.class); - - private static final String PROPERTIES_STOP_TOD_LOOKUP_FILE = "stop.depart.arrive.proportions"; - - // define names used in lookup file - private static final String TOUR_PURPOSE_INDEX_COLUMN_HEADING = "tourpurp"; - private static final String HALF_TOUR_DIRECTION_COLUMN_HEADING = "isInbound"; - private static final String TOUR_TOD_PERIOD_HEADING = "interval"; - private static final String TRIP_NUMBER_COLUMN_HEADING = "trip"; - private static final String INTERVAL_1_PROPORTION_COLUMN_HEADING = "p1"; - - private static final int NUM_DIRECTIONS = 2; - private static final int NUM_TRIPS = 4; - - private double[][][][][] proportions; - - private ModelStructure modelStructure; - - /** - * Constructor - * - * @param propertyMap - * - properties HashMap - * @param modelStructure - * - model definitions helper class - */ - public StopDepartArrivePeriodModel(HashMap propertyMap, - ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - setupModels(propertyMap); - } - - private void setupModels(HashMap propertyMap) - { - - logger.info(String.format("setting up stop depart/arrive choice model.")); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String propsFile = uecPath + propertyMap.get(PROPERTIES_STOP_TOD_LOOKUP_FILE); - - // read the stop purpose lookup table data and populate the maps used to - // assign stop purposes - readLookupProportions(propsFile); - - } - - private void readLookupProportions(String propsLookupFilename) - { - - // read the stop purpose proportions into a TableDataSet - TableDataSet propsLookupTable = null; - String fileName = ""; - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - propsLookupTable = reader.readFile(new File(propsLookupFilename)); - } catch (Exception e) - { - logger.error(String.format( - "Exception occurred reading stop purpose lookup proportions file: %s.", - fileName), e); - throw new RuntimeException(); - } - - // allocate an array for storing proportions arrays. - int lastInterval = modelStructure - .getTimePeriodIndexForTime(ModelStructure.LAST_TOD_INTERVAL_HOUR); - proportions = new double[ModelStructure.NUM_PRIMARY_PURPOSES + 1][NUM_DIRECTIONS][lastInterval + 1][NUM_TRIPS + 1][lastInterval + 1]; - - // fields in lookup file are: - // tourpurp isInbound interval trip p1-p40 (alternative interval - // proportions) - - // populate the outProportionsMaps and inProportionsMaps arrays of maps - // from data in the TableDataSet. - // when stops are generated, they can lookup the proportions for stop - // depart or arrive interval determined - // by tour purpose, outbound/inbound direction and interval of previous - // trip. From these proportions, - // a stop tod interval can be drawn. - - // loop over rows in the TableDataSet - for (int i = 0; i < propsLookupTable.getRowCount(); i++) - { - - // get the tour primary purpose index (1-10) - int tourPrimaryPurposeIndex = (int) propsLookupTable.getValueAt(i + 1, - TOUR_PURPOSE_INDEX_COLUMN_HEADING); - - // get the half tour direction (0 for outbound or 1 for inbound) - int direction = (int) propsLookupTable.getValueAt(i + 1, - HALF_TOUR_DIRECTION_COLUMN_HEADING); - - // get the tod interval (1-40) - int todInterval = (int) propsLookupTable.getValueAt(i + 1, TOUR_TOD_PERIOD_HEADING); - - // get the trip number (1-4) - int tripNumber = (int) propsLookupTable.getValueAt(i + 1, TRIP_NUMBER_COLUMN_HEADING); - - // get the index of the first alternative TOD interval proportion. - int firstPropColumn = propsLookupTable - .getColumnPosition(INTERVAL_1_PROPORTION_COLUMN_HEADING); - - // starting at this column, read the proportions for all TOD - // interval proportions. - // Create the array of proportions for this table record. - for (int j = 1; j <= lastInterval; j++) - proportions[tourPrimaryPurposeIndex][direction][todInterval][tripNumber][j] = propsLookupTable - .getValueAt(i + 1, firstPropColumn + j - 1); - - } - - } - - public double[] getStopTodIntervalProportions(int tourPrimaryPurposeIndex, int direction, - int prevTripTodInterval, int tripNumber) - { - return proportions[tourPrimaryPurposeIndex][direction][prevTripTodInterval][tripNumber]; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java deleted file mode 100644 index a09ac84..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopDestChoiceSize.java +++ /dev/null @@ -1,214 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.Serializable; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Set; -import org.apache.log4j.Logger; -import com.pb.common.datafile.CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * @author crf
    - * Started: Nov 15, 2008 4:17:57 PM - */ -public class StopDestChoiceSize - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(StopDestChoiceSize.class); - - public static final String PROPERTIES_STOP_DC_SIZE_INPUT = "StopDestinationChoice.SizeCoefficients.InputFile"; - - private final Map>> sizeMap; // map - // of - // purpose,purpose - // segment, - // and - // zone/subzone - // to - // size - private final TazDataIf tazDataManager; - private final ModelStructure modelStructure; - private Map>> sizeCoefficients; - - public StopDestChoiceSize(HashMap propertyMap, TazDataIf tazDataManager, - ModelStructure modelStructure) - { - this.tazDataManager = tazDataManager; - this.modelStructure = modelStructure; - sizeMap = new HashMap>>(); - - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String coeffsFileName = propertyMap.get(PROPERTIES_STOP_DC_SIZE_INPUT); - coeffsFileName = projectDirectory + coeffsFileName; - - loadSizeData(coeffsFileName); - } - - public double getDcSize(String purpose, String purposeSegment, int zone, int subzone) - { - return sizeMap.get(purpose).get(purposeSegment).get(getZoneSubzoneMapping(zone, subzone)); - } - - private int getZoneSubzoneMapping(int zone, int subzone) - { - return zone * 10 + subzone; - } - - private void loadSizeData(String coeffsFileName) - { - loadSizeCoefficientTableInformation(readSizeCoefficientTable(coeffsFileName)); - determineSizeCoefficients(); - } - - private TableDataSet readSizeCoefficientTable(String coeffsFileName) - { - try - { - CSVFileReader reader = new CSVFileReader(); - return reader.readFile(new File(coeffsFileName)); - } catch (Exception e) - { - logger.fatal(String.format( - "Exception occurred reading DC Stop Size coefficients data file = %s.", - coeffsFileName), e); - throw new RuntimeException(); - } - } - - private Set getValidPurposes() - { - Set validPurposes = new HashSet(); - validPurposes.add(modelStructure.WORK_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.ESCORT_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.SHOPPING_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.EAT_OUT_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.OTH_MAINT_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.SOCIAL_PURPOSE_NAME.toLowerCase()); - validPurposes.add(modelStructure.OTH_DISCR_PURPOSE_NAME.toLowerCase()); - return validPurposes; - } - - private Set getValidSegments(String purpose) - { - Set validSegments = new HashSet(); - validSegments.add(purpose); - if (purpose.equals(modelStructure.ESCORT_PURPOSE_NAME.toLowerCase())) - for (String segment : modelStructure.ESCORT_SEGMENT_NAMES) - validSegments.add(segment.toLowerCase()); - return validSegments; - } - - private void loadSizeCoefficientTableInformation(TableDataSet coefficients) - { - Set sizeTazColumns = new HashSet(); - String[] coefficientTableColumns = coefficients.getColumnLabels(); - String purposeColumn = modelStructure.getDcSizeCoeffPurposeFieldName(); - String segmentColumn = modelStructure.getDcSizeCoeffSegmentFieldName(); - boolean foundPurposeColumn = false; - boolean foundSegmentColumn = false; - boolean errors = false; - for (String label : coefficientTableColumns) - { - if (label.equals(purposeColumn)) - { - foundPurposeColumn = true; - continue; - } - if (label.equals(segmentColumn)) - { - foundSegmentColumn = true; - continue; - } - - if (!tazDataManager.isValidZoneTableField(label)) - { - logger.fatal("Stop size coefficient table column does not correspond to taz data column: " - + label); - errors = true; - } - sizeTazColumns.add(label); - } - if (!foundPurposeColumn) - { - logger.fatal("Purpose column (" + purposeColumn - + ") not found in stop size coefficient table"); - errors = true; - } - if (!foundSegmentColumn) - { - logger.fatal("Purpose segment column (" + segmentColumn - + ") not found in stop size coefficient table"); - errors = true; - } - - if (!errors) - { - sizeCoefficients = new HashMap>>(); - Set validPurposes = getValidPurposes(); - for (int i = 1; i <= coefficients.getRowCount(); i++) - { - String purpose = coefficients.getStringValueAt(i, purposeColumn).toLowerCase(); - String segment = coefficients.getStringValueAt(i, segmentColumn).toLowerCase(); - if (validPurposes.contains(purpose)) - { - if (!sizeCoefficients.containsKey(purpose)) - sizeCoefficients.put(purpose, new HashMap>()); - if (getValidSegments(purpose).contains(segment)) - { - Map coefficientMap = new HashMap(); - for (String column : sizeTazColumns) - coefficientMap.put(column, (double) coefficients.getValueAt(i, column)); - sizeCoefficients.get(purpose).put(segment, coefficientMap); - } else - { - logger.fatal("Invalid segment for purpose " + purpose - + " found in stop destination choice size coefficient table: " - + segment); - errors = true; - } - - } else - { - logger.fatal("Invalid purpose found in stop destination choice size coefficient table: " - + purpose); - errors = true; - } - } - } - - if (errors) - { - throw new RuntimeException( - "Errors in stop destination choice size coefficient file; see log file for details."); - } - } - - private void determineSizeCoefficients() - { - sizeMap.clear(); - for (String purpose : sizeCoefficients.keySet()) - { - sizeMap.put(purpose, new HashMap>()); - for (String segment : sizeCoefficients.get(purpose).keySet()) - { - Map zoneSizeMap = new HashMap(); - for (int i = 1; i <= tazDataManager.getNumberOfZones(); i++) - { - double size = 0.0d; - Map coefficients = sizeCoefficients.get(purpose).get(segment); - for (String column : sizeCoefficients.get(purpose).get(segment).keySet()) - size += tazDataManager.getZoneTableValue(i, column) - * coefficients.get(column); - double[] walkPercentages = tazDataManager.getZonalWalkPercentagesForTaz(i); - for (int j = 0; j < tazDataManager.getNumberOfSubZones(); j++) - zoneSizeMap.put(getZoneSubzoneMapping(i, j), size * walkPercentages[j]); - } - sizeMap.get(purpose).put(segment, zoneSizeMap); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java deleted file mode 100644 index 51403ef..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyDMU.java +++ /dev/null @@ -1,428 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Nov 4, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public abstract class StopFrequencyDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(StopFrequencyDMU.class); - - protected HashMap methodIndexMap; - - public static final int[] NUM_OB_STOPS_FOR_ALT = {-99999999, 0, 0, 0, 0, - 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3 }; - public static final int[] NUM_IB_STOPS_FOR_ALT = {-99999999, 0, 1, 2, 3, - 0, 1, 2, 3, 0, 1, 2, 3, 0, 1, 2, 3 }; - - public String STOP_PURPOSE_FILE_WORK_NAME = "Work"; - public String STOP_PURPOSE_FILE_UNIVERSITY_NAME = "University"; - public String STOP_PURPOSE_FILE_SCHOOL_NAME = "School"; - public String STOP_PURPOSE_FILE_ESCORT_NAME = "Escort"; - public String STOP_PURPOSE_FILE_SHOPPING_NAME = "Shopping"; - public String STOP_PURPOSE_FILE_MAINT_NAME = "Maint"; - public String STOP_PURPOSE_FILE_EAT_OUT_NAME = "EatOut"; - public String STOP_PURPOSE_FILE_VISIT_NAME = "Visit"; - public String STOP_PURPOSE_FILE_DISCR_NAME = "Discr"; - public String STOP_PURPOSE_FILE_WORK_BASED_NAME = "WorkBased"; - - protected IndexValues dmuIndex; - protected Household household; - protected Person person; - protected Tour tour; - - protected ModelStructure modelStructure; - - private double shoppingAccessibility; - private double maintenanceAccessibility; - private double discretionaryAccessibility; - - public StopFrequencyDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - dmuIndex = new IndexValues(); - } - - public abstract HashMap getTourPurposeChoiceModelIndexMap(); - - public abstract int[] getModelSheetValuesArray(); - - public void setDmuIndexValues(int hhid, int homeTaz, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhid); - dmuIndex.setZoneIndex(homeTaz); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (household.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug SF UEC"); - } - - } - - public void setHouseholdObject(Household household) - { - this.household = household; - } - - public void setPersonObject(Person person) - { - this.person = person; - } - - public void setTourObject(Tour tour) - { - this.tour = tour; - } - - public int getTourIsJoint() - { - return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 - : 0; - } - - public void setShoppingAccessibility(double shoppingAccessibility) - { - this.shoppingAccessibility = shoppingAccessibility; - } - - public void setMaintenanceAccessibility(double maintenanceAccessibility) - { - this.maintenanceAccessibility = maintenanceAccessibility; - } - - public void setDiscretionaryAccessibility(double discretionaryAccessibility) - { - this.discretionaryAccessibility = discretionaryAccessibility; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the household income in dollars - */ - public int getIncomeInDollars() - { - return household.getIncomeInDollars(); - } - - /** - * @return the number of full time workers in the household - */ - public int getNumFtWorkers() - { - return household.getNumFtWorkers(); - } - - /** - * @return the number of part time workers in the household - */ - public int getNumPtWorkers() - { - return household.getNumPtWorkers(); - } - - /** - * @return the person type index for the person making the tour - */ - public int getPersonType() - { - return person.getPersonTypeNumber(); - } - - /** - * @return the number of persons in the household - */ - public int getHhSize() - { - return household.getHhSize(); - } - - /** - * @return the number of driving age students in the household - */ - public int getNumHhDrivingStudents() - { - return household.getNumDrivingStudents(); - } - - /** - * @return the number of non-driving age students in the household - */ - public int getNumHhNonDrivingStudents() - { - return household.getNumNonDrivingStudents(); - } - - /** - * @return the number of preschool age students in the household - */ - public int getNumHhPreschool() - { - return household.getNumPreschool(); - } - - /** - * @return the number of work tours made by this person - */ - public int getWorkTours() - { - return person.getNumWorkTours(); - } - - /** - * - * @return 1 if the outbound portion of the tour is escort, in which case stops are already determined, else 0 - */ - public int getOutboundIsEscort(){ - - return ((tour.getEscortTypeOutbound() == ModelStructure.RIDE_SHARING_TYPE) ||(tour.getEscortTypeOutbound() == ModelStructure.PURE_ESCORTING_TYPE)) - ? 1 : 0; - } - - /** - * - * @return 1 if the inbound portion of the tour is escort, in which case stops are already determined, else 0 - */ - public int getInboundIsEscort(){ - - return ((tour.getEscortTypeInbound() == ModelStructure.RIDE_SHARING_TYPE) ||(tour.getEscortTypeInbound() == ModelStructure.PURE_ESCORTING_TYPE)) - ? 1 : 0; - } - - /** - * @return the total number of tours made by this person - */ - public int getTotalTours() - { - return person.getNumTotalIndivTours(); - } - - /** - * @return the total number of tours made by this person - */ - public int getTotalHouseholdTours() - { - return household.getNumTotalIndivTours(); - } - - /** - * @return the distance from the home mgra to the work mgra for this person - */ - public double getWorkLocationDistance() - { - return person.getWorkLocationDistance(); - } - - /** - * @return the distance from the home mgra to the school mgra for this - * person - */ - public double getSchoolLocationDistance() - { - return person.getSchoolLocationDistance(); - } - - /** - * @return the age of this person - */ - public int getAge() - { - return person.getAge(); - } - - /** - * @return the number of school tours made by this person - */ - public int getSchoolTours() - { - return person.getNumSchoolTours(); - } - - /** - * @return the number of escort tours made by this person - */ - public int getEscortTours() - { - return person.getNumIndividualEscortTours(); - } - - /** - * @return the number of shopping tours made by this person - */ - public int getShoppingTours() - { - return person.getNumIndividualShoppingTours(); - } - - /** - * @return the number of maintenance tours made by this person - */ - public int getMaintenanceTours() - { - return person.getNumIndividualOthMaintTours(); - } - - /** - * @return the number of eating out tours made by this person - */ - public int getEatTours() - { - return person.getNumIndividualEatOutTours(); - } - - /** - * @return the number of visit tours made by this person - */ - public int getVisitTours() - { - return person.getNumIndividualSocialTours(); - } - - /** - * @return the number of discretionary tours made by this person - */ - public int getDiscretionaryTours() - { - return person.getNumIndividualOthDiscrTours(); - } - - /** - * @return the shopping accessibility for the household (alts 28-30) - */ - public double getShoppingAccessibility() - { - return shoppingAccessibility; - } - - /** - * @return the maintenance accessibility for the household (alts 31-33) - */ - public double getMaintenanceAccessibility() - { - return maintenanceAccessibility; - } - - /** - * @return the discretionary accessibility for the household (alts 40-42) - */ - public double getDiscretionaryAccessibility() - { - return discretionaryAccessibility; - } - - /** - * @return the number of inbound stops that correspond to the chosen stop - * frequency alternative - */ - public int getNumIbStopsAlt(int alt) - { - return NUM_IB_STOPS_FOR_ALT[alt]; - } - - /** - * @return the number of outbound stops that correspond to the chosen stop - * frequency alternative - */ - public int getNumObStopsAlt(int alt) - { - return NUM_OB_STOPS_FOR_ALT[alt]; - } - - /** - * get the tour duration, measured in hours - * - * @return duration of tour in hours - number of half-hour intervals - - * arrive period - depart period divided by 2. - */ - public float getTourDurationInHours() - { - return (tour.getTourArrivePeriod() - tour.getTourDepartPeriod()) / 2; - } - - public int getTourModeIsAuto() - { - return modelStructure.getTourModeIsSovOrHov(tour.getTourModeChoice()) ? 1 : 0; - } - - public int getTourModeIsTransit() - { - return modelStructure.getTourModeIsTransit(tour.getTourModeChoice()) ? 1 : 0; - } - - public int getTourModeIsNonMotorized() - { - return modelStructure.getTourModeIsNonMotorized(tour.getTourModeChoice()) ? 1 : 0; - } - - public int getTourModeIsSchoolBus() - { - return modelStructure.getTourModeIsSchoolBus(tour.getTourModeChoice()) ? 1 : 0; - } - - public int getTourDepartPeriod() - { - return tour.getTourDepartPeriod(); - } - - public int getTourArrivePeriod() - { - return tour.getTourArrivePeriod(); - } - - public int getTelecommuteFrequency() { - return person.getTelecommuteChoice(); - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java deleted file mode 100644 index a5778c5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopFrequencyModel.java +++ /dev/null @@ -1,811 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.File; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Random; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -/** - * This class will be used for determining the number of stops on individual - * mandatory, individual non-mandatory and joint tours. - * - * @author Christi Willison - * @version Nov 4, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public class StopFrequencyModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(StopFrequencyModel.class); - private transient Logger stopFreqLogger = Logger.getLogger("stopFreqLog"); - - private static final String PROPERTIES_UEC_STOP_FREQ = "stf.uec.file"; - private static final String PROPERTIES_STOP_PURPOSE_LOOKUP_FILE = "stf.purposeLookup.proportions"; - - private static String[] shopTypes = {"", - "shopSov0", "shopSov1", "shopSov2" }; - private static String[] maintTypes = {"", - "maintSov0", "maintSov1", "maintSov2" }; - private static String[] discrTypes = {"", - "discrSov0", "discrSov1", "discrSov2" }; - - private static final int UEC_DATA_PAGE = 0; - - // define names used in lookup file - private static final String TOUR_PRIMARY_PURPOSE_COLUMN_HEADING = "PrimPurp"; - private static final String HALF_TOUR_DIRECTION_COLUMN_HEADING = "Direction"; - private static final String TOUR_DEPARTURE_START_RANGE_COLUMN_HEADING = "DepartRangeStart"; - private static final String TOUR_DEPARTURE_END_RANGE_COLUMN_HEADING = "DepartRangeEnd"; - private static final String PERSON_TYPE_COLUMN_HEADING = "Ptype"; - - private static final String OUTBOUND_DIRECTION_NAME = "Outbound"; - private static final String INBOUND_DIRECTION_NAME = "Inbound"; - - private static final String FT_WORKER_PERSON_TYPE_NAME = "FT Worker"; - private static final String PT_WORKER_PERSON_TYPE_NAME = "PT Worker"; - private static final String UNIVERSITY_PERSON_TYPE_NAME = "University Student"; - private static final String NONWORKER_PERSON_TYPE_NAME = "Homemaker"; - private static final String RETIRED_PERSON_TYPE_NAME = "Retired"; - private static final String DRIVING_STUDENT_PERSON_TYPE_NAME = "Driving-age Child"; - private static final String NONDRIVING_STUDENT_PERSON_TYPE_NAME = "Pre-Driving Child"; - private static final String PRESCHOOL_PERSON_TYPE_NAME = "Preschool"; - private static final String ALL_PERSON_TYPE_NAME = "All"; - - private StopFrequencyDMU dmuObject; - private ChoiceModelApplication[] choiceModelApplication; - - HashMap tourPurposeModelIndexMap; - HashMap tourPrimaryPurposeIndexNameMap; - - private HashMap indexPurposeMap; - private HashMap[] outProportionsMaps; - private HashMap[] inProportionsMaps; - - private AccessibilitiesTable accTable; - private ModelStructure modelStructure; - private MgraDataManager mgraManager; - - /** - * Constructor that will be used to set up the ChoiceModelApplications for - * each type of tour - * - * @param projectDirectory - * - name of root level project directory - * @param resourceBundle - * - properties file with paths identified - * @param dmuObject - * - decision making unit for stop frequency - * @param tazDataManager - * - holds information about TAZs in the model. - */ - public StopFrequencyModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory, - ModelStructure myModelStructure, AccessibilitiesTable myAccTable) - { - accTable = myAccTable; - modelStructure = myModelStructure; - setupModels(propertyMap, dmuFactory); - } - - private void setupModels(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - - mgraManager = MgraDataManager.getInstance(propertyMap); - - logger.info(String.format("setting up stop frequency choice models.")); - - // String projectDirectory = propertyMap.get( - // CtrampApplication.PROPERTIES_PROJECT_DIRECTORY ); - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String stfUecFile = propertyMap.get(PROPERTIES_UEC_STOP_FREQ); - String uecFileName = uecPath + stfUecFile; - - dmuObject = dmuFactory.getStopFrequencyDMU(); - - tourPrimaryPurposeIndexNameMap = modelStructure.getIndexPrimaryPurposeNameMap(); - - tourPurposeModelIndexMap = dmuObject.getTourPurposeChoiceModelIndexMap(); - int[] modelSheetsArray = dmuObject.getModelSheetValuesArray(); - - // one choice model for each model sheet specified - choiceModelApplication = new ChoiceModelApplication[modelSheetsArray.length]; - for (int i = 0; i < modelSheetsArray.length; i++) - choiceModelApplication[i] = new ChoiceModelApplication(uecFileName, - modelSheetsArray[i], UEC_DATA_PAGE, propertyMap, (VariableTable) dmuObject); - - String purposeLookupFileName = uecPath - + propertyMap.get(PROPERTIES_STOP_PURPOSE_LOOKUP_FILE); - - // read the stop purpose lookup table data and populate the maps used to - // assign stop purposes - readPurposeLookupProportionsTable(purposeLookupFileName); - - } - - public void applyModel(Household household) - { - - int totalStops = 0; - int totalTours = 0; - - Logger modelLogger = stopFreqLogger; - if (household.getDebugChoiceModels()) - household.logHouseholdObject("Pre Stop Frequency Choice: HH=" + household.getHhId(), - stopFreqLogger); - - // get this household's person array - Person[] personArray = household.getPersons(); - - // set the household id, origin taz, hh taz, and debugFlag=false in the - // dmu - dmuObject.setHouseholdObject(household); - - // set the auto sufficiency dependent non-mandatory accessibility values - // for - // the household - int autoSufficiency = household.getAutoSufficiency(); - dmuObject.setShoppingAccessibility(accTable.getAggregateAccessibility( - shopTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setMaintenanceAccessibility(accTable.getAggregateAccessibility( - maintTypes[autoSufficiency], household.getHhMgra())); - dmuObject.setDiscretionaryAccessibility(accTable.getAggregateAccessibility( - discrTypes[autoSufficiency], household.getHhMgra())); - - // process the joint tours for the household first - Tour[] jt = household.getJointTourArray(); - if (jt != null) - { - - List tourList = new ArrayList(); - for (Tour t : jt) - tourList.add(t); - - int tourCount = 0; - for (Tour tour : tourList) - { - - try - { - - //tour.clearStopModelResults(); - - int modelIndex = tourPurposeModelIndexMap - .get(tour.getTourPrimaryPurposeIndex()); - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - if (household.getDebugChoiceModels()) - { - choiceModelDescription = String - .format("Joint Tour Stop Frequency Choice Model:"); - decisionMakerLabel = String.format( - "HH=%d, TourType=%s, TourId=%d, TourPurpose=%s.", - household.getHhId(), tour.getTourCategory(), tour.getTourId(), - tour.getTourPurpose()); - choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; - - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - } - - // set the tour object - dmuObject.setTourObject(tour); - - // set the tour orig/dest TAZs associated with the tour - // orig/dest MGRAs in the IndexValues object. - dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), - mgraManager.getTaz(tour.getTourOrigMgra()), - mgraManager.getTaz(tour.getTourDestMgra())); - - // compute the utilities - float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - tour.setStopFreqLogsum(logsum); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - int choice = -1; - if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught applying joint tour stop frequency choice model for %s type tour: HHID=%d, tourCount=%d, randomCount=%f -- no avaialable stop frequency alternative to choose.", - tour.getTourCategory(), household.getHhId(), tourCount, - randomCount)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = choiceModelApplication[modelIndex].getUtilities(); - double[] probabilities = choiceModelApplication[modelIndex] - .getProbabilities(); - String[] altNames = choiceModelApplication[modelIndex] - .getAlternativeNames(); - - // 0s-indexing - modelLogger.info(decisionMakerLabel); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %15s", k + 1, altNames[k]); - modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %s", choice, altNames[choice - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - choiceModelApplication[modelIndex].logAlternativesInfo( - choiceModelDescription, decisionMakerLabel); - choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, choice); - - // write UEC calculation results to separate model - // specific log file - choiceModelApplication[modelIndex] - .logUECResults(modelLogger, loggingHeader); - } - - // save the chosen alternative and create and populate the - // arrays of inbound/outbound - // stops in the tour object - totalStops += setStopFreqChoice(tour, choice); - - totalTours++; - tourCount++; - - } catch (Exception e) - { - logger.error(String - .format("Exception caught processing joint tour stop frequency choice model for %s type tour: HHID=%d, tourCount=%d.", - tour.getTourCategory(), household.getHhId(), tourCount)); - throw new RuntimeException(e); - } - - } - - } - - // now loop through the person array (1-based), and process all tours - // for - // each person - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - if (household.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", - household.getHhId(), person.getPersonNum(), person.getPersonType()); - household.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - // set the person - dmuObject.setPersonObject(person); - - List tourList = new ArrayList(); - - // apply stop frequency for all person tours - tourList.addAll(person.getListOfWorkTours()); - tourList.addAll(person.getListOfSchoolTours()); - tourList.addAll(person.getListOfIndividualNonMandatoryTours()); - tourList.addAll(person.getListOfAtWorkSubtours()); - - int tourCount = 0; - for (Tour tour : tourList) - { - - try - { - - //tour.clearStopModelResults(); - - int modelIndex = tourPurposeModelIndexMap - .get(tour.getTourPrimaryPurposeIndex()); - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - choiceModelDescription = String - .format("Individual Tour Stop Frequency Choice Model:"); - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, TourType=%s, TourId=%d, TourPurpose=%s, modelIndex=%d.", - household.getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourCategory(), - tour.getTourId(), tour.getTourPurpose(), modelIndex); - - choiceModelApplication[modelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - loggingHeader = choiceModelDescription + " for " + decisionMakerLabel; - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - } - - // set the tour object - dmuObject.setTourObject(tour); - - // compute the utilities - dmuObject.setDmuIndexValues(household.getHhId(), household.getHhTaz(), - mgraManager.getTaz(tour.getTourOrigMgra()), - mgraManager.getTaz(tour.getTourDestMgra())); - - float logsum = (float) choiceModelApplication[modelIndex].computeUtilities(dmuObject, - dmuObject.getDmuIndexValues()); - tour.setStopFreqLogsum(logsum); - - // get the random number from the household - Random random = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = random.nextDouble(); - - // if the choice model has at least one available - // alternative, - // make choice. - int choice = -1; - if (choiceModelApplication[modelIndex].getAvailabilityCount() > 0) choice = choiceModelApplication[modelIndex] - .getChoiceResult(rn); - else - { - logger.error(String - .format("Exception caught applying Individual Tour stop frequency choice model for %s type tour: j=%d, HHID=%d, personNum=%d, tourCount=%d, randomCount=%f -- no avaialable stop frequency alternative to choose.", - tour.getTourCategory(), j, household.getHhId(), - person.getPersonNum(), tourCount, randomCount)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = choiceModelApplication[modelIndex].getUtilities(); - double[] probabilities = choiceModelApplication[modelIndex] - .getProbabilities(); - String[] altNames = choiceModelApplication[modelIndex] - .getAlternativeNames(); // 0s-indexing - - modelLogger.info(decisionMakerLabel); - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("------------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < altNames.length; ++k) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %15s", k + 1, altNames[k]); - modelLogger.info(String.format("%-20s%18.6e%18.6e%18.6e", altString, - utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %s", choice, altNames[choice - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", - altString, rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug log file - choiceModelApplication[modelIndex].logAlternativesInfo( - choiceModelDescription, decisionMakerLabel); - choiceModelApplication[modelIndex].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, choice); - - // write UEC calculation results to separate model - // specific - // log file - choiceModelApplication[modelIndex] - .logUECResults(modelLogger, loggingHeader); - - } - - // choiceResultsFreq[choice][modelIndex]++; - - // save the chosen alternative and create and populate the - // arrays - // of inbound/outbound stops in the tour object - totalStops += setStopFreqChoice(tour, choice); - totalTours++; - - tourCount++; - - } catch (Exception e) - { - logger.error(String - .format("Exception caught processing Individual Tour stop frequency choice model for %s type tour: j=%d, HHID=%d, personNum=%d, tourCount=%d.", - tour.getTourCategory(), j, household.getHhId(), - person.getPersonNum(), tourCount)); - throw new RuntimeException(e); - } - - } - - } // j (person loop) - - household.setStfRandomCount(household.getHhRandomCount()); - - } - - private int setStopFreqChoice(Tour tour, int stopFreqChoice) - { - - tour.setStopFreqChoice(stopFreqChoice); - - // set argument values for method call to get stop purpose - Household hh = tour.getPersonObject().getHouseholdObject(); - int tourDepartPeriod = tour.getTourDepartPeriod(); - int tourArrivePeriod = tour.getTourArrivePeriod(); - - //log out tour details if invalid tour departure and arrival time periods are found - if(tourDepartPeriod==-1||tourArrivePeriod==-1) tour.logTourObject(logger, 100); - - int tourPrimaryPurposeIndex = tour.getTourPrimaryPurposeIndex(); - String tourPrimaryPurpose = tourPrimaryPurposeIndexNameMap.get(tourPrimaryPurposeIndex); - String personType = tour.getPersonObject().getPersonType(); - - int numObStops = dmuObject.getNumObStopsAlt(stopFreqChoice); - if ((numObStops > 0) && (tour.getEscortTypeOutbound()!=ModelStructure.RIDE_SHARING_TYPE) && (tour.getEscortTypeOutbound()!=ModelStructure.PURE_ESCORTING_TYPE)) - { - // get a stop purpose for each outbound stop generated, plus the - // stop at - // the primary destination - String[] obStopOrigPurposes = new String[numObStops + 1]; - String[] obStopDestPurposes = new String[numObStops + 1]; - int[] obStopPurposeIndices = new int[numObStops + 1]; - obStopOrigPurposes[0] = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - for (int i = 0; i < numObStops; i++) - { - if (i > 0) obStopOrigPurposes[i] = obStopDestPurposes[i - 1]; - obStopPurposeIndices[i] = getStopPurpose(hh, OUTBOUND_DIRECTION_NAME, - tourDepartPeriod, tourPrimaryPurpose, personType); - obStopDestPurposes[i] = indexPurposeMap.get(obStopPurposeIndices[i]); - } - obStopOrigPurposes[numObStops] = obStopDestPurposes[numObStops - 1]; - obStopDestPurposes[numObStops] = tourPrimaryPurpose; - // the last stop record is for the trip from stop to destination - - // pass in the array of stop purposes; length of array determines - // number - // of outbound stop objects created. - if (tour.getOutboundStops() != null) - { - Exception e = new RuntimeException(); - logger.error("outbound stops array for hhid=" + tour.getHhId() + ", person=" - + tour.getPersonObject().getPersonNum() + ", tour=" + tour.getTourId() - + ", purpose=" + tour.getTourPurpose(), e); - try - { - throw e; - } catch (Exception e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - } - tour.createOutboundStops(obStopOrigPurposes, obStopDestPurposes, obStopPurposeIndices); - } - - int numIbStops = dmuObject.getNumIbStopsAlt(stopFreqChoice); - if ((numIbStops > 0) && (tour.getEscortTypeInbound()!=ModelStructure.RIDE_SHARING_TYPE) && (tour.getEscortTypeInbound()!=ModelStructure.PURE_ESCORTING_TYPE)) - { - // get a stop purpose for each inbound stop generated - String[] ibStopOrigPurposes = new String[numIbStops + 1]; - String[] ibStopDestPurposes = new String[numIbStops + 1]; - int[] ibStopPurposeIndices = new int[numIbStops + 1]; - ibStopOrigPurposes[0] = tour.getTourPrimaryPurpose(); - for (int i = 0; i < numIbStops; i++) - { - if (i > 0) ibStopOrigPurposes[i] = ibStopDestPurposes[i - 1]; - ibStopPurposeIndices[i] = getStopPurpose(hh, INBOUND_DIRECTION_NAME, - tourArrivePeriod, tourPrimaryPurpose, personType); - ibStopDestPurposes[i] = indexPurposeMap.get(ibStopPurposeIndices[i]); - } - ibStopOrigPurposes[numIbStops] = ibStopDestPurposes[numIbStops - 1]; - ibStopDestPurposes[numIbStops] = tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY) ? "Work" : "Home"; - // the last stop record is for the trip from stop to home or work - - // pass in the array of stop purposes; length of array determines - // number - // of inbound stop objects created. - if (tour.getInboundStops() != null) - { - Exception e = new RuntimeException(); - logger.error("inbound stops array for hhid=" + tour.getHhId() + ", person=" - + tour.getPersonObject().getPersonNum() + ", tour=" + tour.getTourId() - + ", purpose=" + tour.getTourPurpose(), e); - try - { - throw e; - } catch (Exception e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - } - - tour.createInboundStops(ibStopOrigPurposes, ibStopDestPurposes, ibStopPurposeIndices); - } - - return numObStops + numIbStops; - - } - - private void readPurposeLookupProportionsTable(String purposeLookupFilename) - { - - // read the stop purpose proportions into a TableDataSet - TableDataSet purposeLookupTable = null; - String fileName = ""; - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - purposeLookupTable = reader.readFile(new File(purposeLookupFilename)); - } catch (Exception e) - { - logger.error(String.format( - "Exception occurred reading stop purpose lookup proportions file: %s.", - fileName), e); - throw new RuntimeException(); - } - - // allocate a HashMap array for each direction, dimensioned to maximum - // departure hour, to map keys determined by combination of categories - // to - // proportions arrays. - int numDepartPeriods = modelStructure.getNumberOfTimePeriods(); - outProportionsMaps = new HashMap[numDepartPeriods + 1]; - inProportionsMaps = new HashMap[numDepartPeriods + 1]; - for (int i = 0; i <= numDepartPeriods; i++) - { - outProportionsMaps[i] = new HashMap(); - inProportionsMaps[i] = new HashMap(); - } - - // create a mapping between names used in lookup file and purpose names - // used - // in model - HashMap primaryPurposeMap = new HashMap(); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_WORK_NAME, - ModelStructure.WORK_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_UNIVERSITY_NAME, - ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_SCHOOL_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_ESCORT_NAME, - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_SHOPPING_NAME, - ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_EAT_OUT_NAME, - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_MAINT_NAME, - ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_VISIT_NAME, - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_DISCR_NAME, - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); - primaryPurposeMap.put(dmuObject.STOP_PURPOSE_FILE_WORK_BASED_NAME, - ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME); - - // create a mapping between stop purpose alternative indices selected - // from - // monte carlo process and stop purpose names used in model - // the indices are the order of the proportions columns in the table - indexPurposeMap = new HashMap(); - indexPurposeMap.put(1, "work related"); - indexPurposeMap.put(2, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(3, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(4, ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(5, ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(6, ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(7, ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(8, ModelStructure.VISITING_PRIMARY_PURPOSE_NAME); - indexPurposeMap.put(9, ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME); - - // create a mapping between names used in lookup file and person type - // names - // used in model - HashMap personTypeMap = new HashMap(); - personTypeMap.put(FT_WORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_FULL_TIME_WORKER_NAME); - personTypeMap.put(PT_WORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_PART_TIME_WORKER_NAME); - personTypeMap.put(UNIVERSITY_PERSON_TYPE_NAME, Person.PERSON_TYPE_UNIVERSITY_STUDENT_NAME); - personTypeMap.put(NONWORKER_PERSON_TYPE_NAME, Person.PERSON_TYPE_NON_WORKER_NAME); - personTypeMap.put(RETIRED_PERSON_TYPE_NAME, Person.PERSON_TYPE_RETIRED_NAME); - personTypeMap - .put(DRIVING_STUDENT_PERSON_TYPE_NAME, Person.PERSON_TYPE_STUDENT_DRIVING_NAME); - personTypeMap.put(NONDRIVING_STUDENT_PERSON_TYPE_NAME, - Person.PERSON_TYPE_STUDENT_NON_DRIVING_NAME); - personTypeMap.put(PRESCHOOL_PERSON_TYPE_NAME, Person.PERSON_TYPE_PRE_SCHOOL_CHILD_NAME); - personTypeMap.put(ALL_PERSON_TYPE_NAME, ALL_PERSON_TYPE_NAME); - - // fields in lookup file are: - // PrimPurp Direction DepartRangeStart DepartRangeEnd Ptype Work - // University - // School Escort Shop Maintenance Eating Out Visiting Discretionary - - // populate the outProportionsMaps and inProportionsMaps arrays of maps - // from - // data in the TableDataSet. - // when stops are generated, they can lookup the proportions for stop - // purpose - // selection from a map determined - // by tour purpose, person type, outbound/inbound direction and tour - // departure time. From these proportions, - // a stop purpose can be drawn. - - // loop over rows in the TableDataSet - for (int i = 0; i < purposeLookupTable.getRowCount(); i++) - { - - // get the tour primary purpose - String tourPrimPurp = primaryPurposeMap.get(purposeLookupTable.getStringValueAt(i + 1, - TOUR_PRIMARY_PURPOSE_COLUMN_HEADING)); - - // get the half tour direction - String direction = purposeLookupTable.getStringValueAt(i + 1, - HALF_TOUR_DIRECTION_COLUMN_HEADING); - - // get the beginning of the range of departure hours - int departPeriodRangeStart = (int) purposeLookupTable.getValueAt(i + 1, - TOUR_DEPARTURE_START_RANGE_COLUMN_HEADING); - - // get the end of the range of departure hours - int arriveperiodRangeEnd = (int) purposeLookupTable.getValueAt(i + 1, - TOUR_DEPARTURE_END_RANGE_COLUMN_HEADING); - - int startRange = modelStructure.getTimePeriodIndexForTime(departPeriodRangeStart); - int endRange = modelStructure.getTimePeriodIndexForTime(arriveperiodRangeEnd); - - // get the person type - String personType = personTypeMap.get(purposeLookupTable.getStringValueAt(i + 1, - PERSON_TYPE_COLUMN_HEADING)); - - // columns following person type are proportions by stop purpose. - // Get the - // index of the first stop purpose proportion. - int firstPropColumn = purposeLookupTable.getColumnPosition(PERSON_TYPE_COLUMN_HEADING) + 1; - - // starting at this column, read the proportions for all stop - // purposes. - // Create the array of proportions for this table record. - double[] props = new double[indexPurposeMap.size()]; - for (int j = 0; j < props.length; j++) - { - props[j] = purposeLookupTable.getValueAt(i + 1, firstPropColumn + j); - } - - // get a HashMap for the direction and each hour in the start/end - // range, - // and store the proportions in that map for the key. - // the key to use for any of these HashMaps is created consisting of - // "TourPrimPurp_PersonType" - // if the person type for the record is "All", a key is defined for - // each - // person type, and the proportions stored for each key. - if (personType.equalsIgnoreCase(ALL_PERSON_TYPE_NAME)) - { - for (String ptype : personTypeMap.values()) - { - String key = tourPrimPurp + "_" + ptype; - if (direction.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) - { - for (int k = startRange; k <= endRange; k++) - { - outProportionsMaps[k].put(key, props); - } - } else if (direction.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) - { - for (int k = startRange; k <= endRange; k++) - inProportionsMaps[k].put(key, props); - } - } - } else - { - String key = tourPrimPurp + "_" + personType; - if (direction.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) - { - for (int k = startRange; k <= endRange; k++) - outProportionsMaps[k].put(key, props); - } else if (direction.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) - { - for (int k = startRange; k <= endRange; k++) - inProportionsMaps[k].put(key, props); - } - } - - } - - } - - private int getStopPurpose(Household household, String halfTourDirection, int tourDepartPeriod, - String tourPrimaryPurpose, String personType) - { - - double[] props = null; - String key = tourPrimaryPurpose + "_" + personType; - - try - { - if (halfTourDirection.equalsIgnoreCase(OUTBOUND_DIRECTION_NAME)) props = outProportionsMaps[tourDepartPeriod] - .get(key); - else if (halfTourDirection.equalsIgnoreCase(INBOUND_DIRECTION_NAME)) - props = inProportionsMaps[tourDepartPeriod].get(key); - - double rn = household.getHhRandom().nextDouble(); - int choice = ChoiceModelApplication.getMonteCarloSelection(props, rn); - - return (choice + 1); - - } catch (Exception e) - { - logger.error("exception caught trying to determine stop purpose."); - logger.error("key=" + key + ", tourPrimaryPurpose=" + tourPrimaryPurpose - + ", personType=" + personType + ", halfTourDirection=" + halfTourDirection - + ", tourDepartPeriod=" + tourDepartPeriod); - throw new RuntimeException(); - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java deleted file mode 100644 index 180f181..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/StopLocationDMU.java +++ /dev/null @@ -1,411 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Nov 4, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public class StopLocationDMU - implements Serializable, VariableTable -{ - - protected HashMap methodIndexMap; - - protected IndexValues dmuIndex; - protected Household household; - protected Person person; - protected Tour tour; - protected Stop stop; - protected ModelStructure modelStructure; - - protected int numberInSample; - protected int tourModeIndex; - protected double origDestDistance; - - // these arrays are dimensioned to the total number of location choice - // alternatives (number of MGRAs) - protected int[] walkTransitAvailableAtMgra; - protected double[] distancesFromOrigMgra; - protected double[] distancesFromTourOrigMgra; - protected double[] distancesToDestMgra; - protected double[] distancesToTourDestMgra; - protected double[] logSizeTerms; - - protected double[] bikeLogsumsFromOrigMgra; - protected double[] bikeLogsumsToDestMgra; - - - - // these arrays are dimensioned to the maximum number of alternatives in the - // sample - protected double[] mcLogsums; - protected double[] slcSoaCorrections; - protected int[] sampleArray; - - public StopLocationDMU(ModelStructure modelStructure) - { - dmuIndex = new IndexValues(); - - this.modelStructure = modelStructure; - } - - public void setDmuIndexValues(int hhid, int homeTaz, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhid); - dmuIndex.setZoneIndex(homeTaz); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (household.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug SL UEC"); - } - - } - - public void setStopObject(Stop myStop) - { - stop = myStop; - tour = stop.getTour(); - person = tour.getPersonObject(); - household = person.getHouseholdObject(); - } - - /** - * set the value for the number of unique alternatives in the sample. - * sampleArray can be indexed as i=1; i <= numberInSample. - * sampleArray.length - 1 is the maximum number of locations in the sample. - * - * @param num - * - number of unique alternatives in the sample. - */ - public void setNumberInSample(int num) - { - numberInSample = num; - } - - /** - * set the array of sample MGRA values from which the stop location MGRA - * will be selected. - * - * @param sample - * - the sample array of MGRA location choice alternatives. - use - * numberInSample as upperbound of relevant choices in sample - */ - public void setSampleOfAlternatives(int[] sample) - { - sampleArray = sample; - } - - public void setSlcSoaCorrections(double[] corrections) - { - slcSoaCorrections = corrections; - } - - public void setMcLogsums(double[] logsums) - { - mcLogsums = logsums; - } - - public void setLogSize(double[] size) - { - logSizeTerms = size; - } - - /** - * set the array of distance values from the origin MGRA of the stop to all - * MGRAs. - * - * @param distances - */ - public void setDistancesFromOrigMgra(double[] distances) - { - distancesFromOrigMgra = distances; - } - - /** - * set the array of distance values from the tour origin MGRA to all MGRAs. - * - * @param distances - */ - public void setDistancesFromTourOrigMgra(double[] distances) - { - distancesFromTourOrigMgra = distances; - } - - /** - * set the array of distance values from all MGRAs to the final destination - * MGRA of the stop. - * - * @param distances - */ - public void setDistancesToDestMgra(double[] distances) - { - distancesToDestMgra = distances; - } - - /** - * set the array of distance values from all MGRAs to the tour destination - * MGRA. - * - * @param distances - */ - public void setDistancesToTourDestMgra(double[] distances) - { - distancesToTourDestMgra = distances; - } - - /** - * @param bikeLogsumsFromOrigMgra the bikeLogsumsFromOrigMgra to set - */ - public void setBikeLogsumsFromOrigMgra(double[] bikeLogsumsFromOrigMgra) { - this.bikeLogsumsFromOrigMgra = bikeLogsumsFromOrigMgra; - } - - /** - * @param bikeLogsumsToDestMgra the bikeLogsumsToDestMgra to set - */ - public void setBikeLogsumsToDestMgra(double[] bikeLogsumsToDestMgra) { - this.bikeLogsumsToDestMgra = bikeLogsumsToDestMgra; - } - - /** - * set the OD distance value from the stop origin MGRA to the final - * destination MGRA of the stop. - * - * @param distances - */ - public void setOrigDestDistance(double distance) - { - origDestDistance = distance; - } - - /** - * set the tour mode index value for the tour of the stop being located - * - * @param tour - */ - public void setTourModeIndex(int index) - { - tourModeIndex = index; - } - - /** - * set the array of attributes for all MGRAs that says their is walk transit - * access for the indexed mgra - * - * @param tour - */ - public void setWalkTransitAvailable(int[] avail) - { - walkTransitAvailableAtMgra = avail; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getSampleMgraAlt(int alt) - { - return sampleArray[alt]; - } - - public double getSlcSoaCorrectionsAlt(int alt) - { - return slcSoaCorrections[alt]; - } - - public double getMcLogsumAlt(int alt) - { - return mcLogsums[alt]; - } - - /** - * get the logged size term from the full set of size terms for all mgra - * associated with the sample alternative - * - * @param alt - * - element number for the sample array - * @return logged size term for mgra associated with the sample element - */ - public double getLnSlcSizeSampleAlt(int alt) - { - int mgra = sampleArray[alt]; - return logSizeTerms[mgra]; - } - - /** - * get the logged size term ffrom the full set of size terms for all mgra - * alternatives - * - * @param mgra - * - mgra location alternive - * @return logged size term for mgra - */ - public double getLnSlcSizeAlt(int mgra) - { - return logSizeTerms[mgra]; - } - - protected int getTourIsJoint() - { - return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 - : 0; - } - - public int getTourMode() - { - return tour.getTourModeChoice(); - } - - public int getTourPurpose() - { - return tour.getTourPrimaryPurposeIndex(); - } - - public int getFemale() - { - return person.getPersonIsFemale(); - } - - public int getIncomeInDollars() - { - return household.getIncomeInDollars(); - } - - public int getAge() - { - return person.getAge(); - } - - public int getStopPurpose() - { - return stop.getStopPurposeIndex(); - } - - public int getStopNumber() - { - return (stop.getStopId() + 1); - } - - public int getInboundStop() - { - return stop.isInboundStop() ? 1 : 0; - } - - public int getStopsOnHalfTour() - { - return stop.isInboundStop() ? tour.getInboundStops().length - : tour.getOutboundStops().length; - } - - public double getOrigToMgraDistanceAlt(int alt) - { - // int dummy=0; - // double dist = Math.abs(distancesFromOrigMgra[alt] - - // distancesToDestMgra[alt]); - // double maxSegDist = Math.max(distancesFromOrigMgra[alt], - // distancesToDestMgra[alt]); - // if ( dist > 0 && dist < 1 && origDestDistance > 40 ) - // dummy = 1; - - return distancesFromOrigMgra[alt]; - } - - public double getTourOrigToMgraDistanceAlt(int alt) - { - return distancesFromTourOrigMgra[alt]; - } - - public double getMgraToDestDistanceAlt(int alt) - { - return distancesToDestMgra[alt]; - } - - public double getMgraToTourDestDistanceAlt(int alt) - { - return distancesToTourDestMgra[alt]; - } - - public double getOrigToMgraBikeLogsumAlt(int alt) - { - return bikeLogsumsFromOrigMgra[alt]; - } - - public double getMgraToDestBikeLogsumAlt(int alt) - { - return bikeLogsumsToDestMgra[alt]; - } - - - - public double getOdDistance() - { - return origDestDistance; - } - - public int getTourModeIsWalk() - { - boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tourModeIndex); - return tourModeIsWalk ? 1 : 0; - } - - public int getTourModeIsBike() - { - boolean tourModeIsBike = modelStructure.getTourModeIsBike(tourModeIndex); - return tourModeIsBike ? 1 : 0; - } - - public int getTourModeIsWalkTransit() - { - return (modelStructure.getTourModeIsWalkTransit(tourModeIndex) ? 1 : 0); - } - - public int getWalkTransitAvailableAlt(int alt) - { - return walkTransitAvailableAtMgra[alt]; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java deleted file mode 100644 index 736fd49..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDepartureAndDurationTime.java +++ /dev/null @@ -1,956 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; - -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import org.sandag.abm.accessibilities.NonTransitUtilities; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.ResourceUtil; - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 11, 2008 Time: 9:25:30 AM To - * change this template use File | Settings | File Templates. - */ -public class SubtourDepartureAndDurationTime - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(SubtourDepartureAndDurationTime.class); - private transient Logger todLogger = Logger.getLogger("todLogger"); - private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); - - private static final String TOD_UEC_FILE_TARGET = "departTime.uec.file"; - private static final String TOD_UEC_DATA_TARGET = "departTime.data.page"; - private static final String TOD_UEC_AT_WORK_MODEL_TARGET = "departTime.atwork.page"; - - private static String[] tourPurposeNames; - - private int[] todModelIndices; - private HashMap purposeNameIndexMap; - - private static final String[] DC_MODEL_SHEET_KEYS = { - TOD_UEC_AT_WORK_MODEL_TARGET, TOD_UEC_AT_WORK_MODEL_TARGET, - TOD_UEC_AT_WORK_MODEL_TARGET }; - - private int[] tourDepartureTimeChoiceSample; - - // DMU for the UEC - private TourDepartureTimeAndDurationDMU todDmuObject; - private TourModeChoiceDMU mcDmuObject; - - // model structure to compare the .properties time of day with the UECs - private ModelStructure modelStructure; - - private TazDataManager tazs; - private MgraDataManager mgraManager; - - // private double[][] dcSizeArray; - - private ChoiceModelApplication[] todModels; - private TourModeChoiceModel mcModel; - - private int[] altStarts; - private int[] altEnds; - - private boolean[] needToComputeLogsum; - private double[] modeChoiceLogsums; - - private short[] tempWindow; - - // create an array to count the subtours propcessed within work tours - // there are at most 2 work tours per person - private int[] subtourNumForWorkTours = new int[2]; - - private int noAltChoice = 1; - - private long mcTime; - - public SubtourDepartureAndDurationTime(HashMap propertyMap, - ModelStructure modelStructure, CtrampDmuFactoryIf dmuFactory, - TourModeChoiceModel mcModel) - { - - // set the model structure - this.modelStructure = modelStructure; - this.mcModel = mcModel; - - logger.info(String.format("setting up %s time-of-day choice model.", - ModelStructure.AT_WORK_CATEGORY)); - - setupTodChoiceModels(propertyMap, dmuFactory); - } - - private void setupTodChoiceModels(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - tazs = TazDataManager.getInstance(); - mgraManager = MgraDataManager.getInstance(); - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - String todUecFileName = propertyMap.get(TOD_UEC_FILE_TARGET); - todUecFileName = uecFileDirectory + todUecFileName; - - todDmuObject = dmuFactory.getTourDepartureTimeAndDurationDMU(); - - mcDmuObject = dmuFactory.getModeChoiceDMU(); - - int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; - needToComputeLogsum = new boolean[numLogsumIndices]; - modeChoiceLogsums = new double[numLogsumIndices]; - - tourPurposeNames = new String[3]; - tourPurposeNames[0] = modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME; - tourPurposeNames[1] = modelStructure.AT_WORK_EAT_PURPOSE_NAME; - tourPurposeNames[2] = modelStructure.AT_WORK_MAINT_PURPOSE_NAME; - - // create the array of tod model indices - int[] uecSheetIndices = new int[tourPurposeNames.length]; - - purposeNameIndexMap = new HashMap(tourPurposeNames.length); - - int i = 0; - for (String purposeName : tourPurposeNames) - { - int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); - purposeNameIndexMap.put(purposeName, i); - uecSheetIndices[i] = uecIndex; - i++; - } - - // create a lookup array to map purpose index to model index - todModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int todModelIndex = 0; - int todSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, todModelIndex); - todModelIndices[todSegmentIndex] = todModelIndex++; - } else - { - todModelIndices[todSegmentIndex] = modelIndexMap.get(uecIndex); - } - - todSegmentIndex++; - } - - todModels = new ChoiceModelApplication[modelIndexMap.size()]; - int todModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, - TOD_UEC_DATA_TARGET); - - for (int uecIndex : modelIndexMap.keySet()) - { - int modelIndex = modelIndexMap.get(uecIndex); - try - { - todModels[modelIndex] = new ChoiceModelApplication(todUecFileName, uecIndex, - todModelDataSheet, propertyMap, (VariableTable) todDmuObject); - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up At-work Subtour TOD ChoiceModelApplication[%d] for model index=%d of %d models", - i, i, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - // get the alternatives table from the work tod UEC. - TableDataSet altsTable = todModels[0].getUEC().getAlternativeData(); - - altStarts = altsTable.getColumnAsInt(CtrampApplication.START_FIELD_NAME); - altEnds = altsTable.getColumnAsInt(CtrampApplication.END_FIELD_NAME); - todDmuObject.setTodAlts(altStarts, altEnds); - - int numDepartureTimeChoiceAlternatives = todModels[0].getNumberOfAlternatives(); - tourDepartureTimeChoiceSample = new int[numDepartureTimeChoiceAlternatives + 1]; - Arrays.fill(tourDepartureTimeChoiceSample, 1); - - tempWindow = new short[modelStructure.getNumberOfTimePeriods() + 1]; - - } - - public void applyModel(Household hh, boolean runModeChoice) - { - - mcTime = 0; - - Logger modelLogger = todLogger; - - // get the peron objects for this household - Person[] persons = hh.getPersons(); - for (int p = 1; p < persons.length; p++) - { - - Person person = persons[p]; - - // get the work tours for the person - ArrayList subtourList = person.getListOfAtWorkSubtours(); - - // if no work subtours for person, nothing to do. - if (subtourList.size() == 0) continue; - - ArrayList workTourList = person.getListOfWorkTours(); - int numWorkTours = workTourList.size(); - - // save a copy of this person's original time windows - short[] personWindow = person.getTimeWindows(); - for (int w = 0; w < personWindow.length; w++) - tempWindow[w] = personWindow[w]; - - for (int i = 0; i < subtourNumForWorkTours.length; i++) - subtourNumForWorkTours[i] = 0; - - int m = -1; - int tourPurpNum = 1; - int noWindowCountFirstTemp = 0; - int noWindowCountLastTemp = 0; - int noLaterAlternativeCountTemp = 0; - for (Tour t : subtourList) - { - - Tour workTour = null; - int workTourIndex = 0; - - try - { - - workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); - subtourNumForWorkTours[workTourIndex]++; - workTour = workTourList.get(workTourIndex); - - // if the first subtour for a work tour, make window of work - // tour available, and other windows not available - if (subtourNumForWorkTours[workTourIndex] == 1) - { - person.resetTimeWindow(workTour.getTourDepartPeriod(), - workTour.getTourArrivePeriod()); - person.scheduleWindow(0, workTour.getTourDepartPeriod() - 1); - person.scheduleWindow(workTour.getTourArrivePeriod() + 1, - modelStructure.getNumberOfTimePeriods()); - } else if (subtourNumForWorkTours[workTourIndex] > 2) - { - logger.error("too many subtours for a work tour. workTourIndex=" - + workTourIndex + ", subtourNumForWorkTours[workTourIndex]" - + subtourNumForWorkTours[workTourIndex]); - logger.error("hhid=" + hh.getHhId() + ", persNum=" + person.getPersonNum()); - throw new RuntimeException(); - } - - // get the choice model for the tour purpose - String tourPurposeName = t.getSubTourPurpose(); - - int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); - m = todModelIndices[tourPurposeIndex]; - - // write debug header - String separator = ""; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (hh.getDebugChoiceModels()) - { - - choiceModelDescription = String.format( - "AtWork Subtour Departure Time Choice Model for: Purpose=%s", - tourPurposeName); - decisionMakerLabel = String - .format("HH=%d, PersonNum=%d, PersonType=%s, tourId=%d, num=%d of %d %s tours", - hh.getHhId(), person.getPersonNum(), - person.getPersonType(), t.getTourId(), tourPurpNum, - subtourList.size(), tourPurposeName); - todModels[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - String loggerString = "AtWork Subtour Departure Time Choice Model: Debug Statement for Household ID: " - + hh.getHhId() - + ", Person Num: " - + person.getPersonNum() - + ", Person Type: " - + person.getPersonType() - + ", Tour Id: " - + t.getTourId() - + ", num " - + tourPurpNum - + " of " - + subtourList.size() + " " + tourPurposeName + " tours."; - for (int k = 0; k < loggerString.length(); k++) - separator += "+"; - modelLogger.info(loggerString); - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - - } - - // set the dmu object - todDmuObject.setHousehold(hh); - todDmuObject.setPerson(person); - todDmuObject.setTour(t); - - int otherTourEndHour = -1; - - // check for multiple tours for this person, by purpose - // set the first or second switch if multiple tours for - // person, by purpose - if (subtourList.size() == 1) - { - // not a multiple tour pattern - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(1); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else if (subtourList.size() > 1) - { - // Two-plus tour multiple tour pattern - if (tourPurpNum == 1) - { - // first of 2+ tours - todDmuObject.setFirstTour(1); - todDmuObject.setSubsequentTour(0); - todDmuObject.setTourNumber(tourPurpNum); - todDmuObject.setEndOfPreviousScheduledTour(0); - } else - { - // 2nd or greater tours - todDmuObject.setFirstTour(0); - todDmuObject.setSubsequentTour(1); - todDmuObject.setTourNumber(tourPurpNum); - // the ArrayList is 0-based, and we want the - // previous tour, so subtract 2 from tourPurpNum to - // get the right index - otherTourEndHour = subtourList.get(tourPurpNum - 2) - .getTourArrivePeriod(); - todDmuObject.setEndOfPreviousScheduledTour(otherTourEndHour); - } - } - - // set the choice availability and sample - boolean[] departureTimeChoiceAvailability = person.getAvailableTimeWindows( - altStarts, altEnds); - Arrays.fill(tourDepartureTimeChoiceSample, 1); - - if (departureTimeChoiceAvailability.length != tourDepartureTimeChoiceSample.length) - { - logger.error(String - .format("error in at-work subtour departure time choice model for hhId=%d, personNum=%d, tour purpose index=%d, tour ArrayList index=%d.", - hh.getHhId(), person.getPersonNum(), tourPurposeIndex, - tourPurpNum - 1)); - logger.error(String - .format("length of the availability array determined by the number of alternatives set in the person scheduler=%d", - departureTimeChoiceAvailability.length)); - logger.error(String - .format("does not equal the length of the sample array determined by the number of alternatives in the at-work subtour UEC=%d.", - tourDepartureTimeChoiceSample.length)); - throw new RuntimeException(); - } - - // if no time window is available for the tour, make the - // first and last alternatives available - // for that alternative, and keep track of the number of - // times this condition occurs. - int alternativeAvailable = -1; - for (int a = 0; a < departureTimeChoiceAvailability.length; a++) - { - if (departureTimeChoiceAvailability[a]) - { - alternativeAvailable = a; - break; - } - } - - int chosen = -1; - int chosenStartPeriod = -1; - int chosenEndPeriod = -1; - - // alternate making the first and last periods chosen if no - // alternatives are available - if (alternativeAvailable < 0) - { - - if (noAltChoice == 1) - { - if (subtourList.size() > 1 && tourPurpNum > 1) - { - chosenStartPeriod = otherTourEndHour; - chosenEndPeriod = otherTourEndHour; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to previous sub-tour arrive period=" - + chosenStartPeriod + "."); - } else - { - chosenStartPeriod = workTour.getTourDepartPeriod(); - chosenEndPeriod = workTour.getTourDepartPeriod(); - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled and no previous sub-tour, depart AND arrive set to work tour depart period=" - + chosenStartPeriod + "."); - } - noWindowCountFirstTemp++; - noAltChoice = departureTimeChoiceAvailability.length - 1; - } else - { - if (subtourList.size() > 1 && tourPurpNum > 1) - { - chosenStartPeriod = otherTourEndHour; - chosenEndPeriod = otherTourEndHour; - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled, depart AND arrive set to previous sub-tour arrive period=" - + chosenStartPeriod + "."); - } else - { - chosenStartPeriod = workTour.getTourArrivePeriod(); - chosenEndPeriod = workTour.getTourArrivePeriod(); - if (hh.getDebugChoiceModels()) - modelLogger - .info("All alternatives already scheduled and no previous sub-tour, depart AND arrive set to work tour arrive period=" - + chosenStartPeriod + "."); - } - noWindowCountLastTemp++; - noAltChoice = 1; - } - - // schedule the chosen alternative - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - - if (runModeChoice) - { - - long check = System.nanoTime(); - - if (hh.getDebugChoiceModels()) - hh.logHouseholdObject( - "Pre At-work Subtour Tour Mode Choice Household " - + hh.getHhId() + ", Tour " + tourPurpNum + " of " - + subtourList.size(), tourMCNonManLogger); - - // set the mode choice attributes needed by - // @variables in the UEC spreadsheets - setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, - chosenEndPeriod); - - // use the mcModel object already setup for - // computing logsums and get - // the mode choice, where the selected - // worklocation and subzone an departure time and - // duration are set - // for this work tour. - int chosenMode = mcModel.getModeChoice(mcDmuObject, - t.getTourPrimaryPurpose()); - t.setTourModeChoice(chosenMode); - - mcTime += (System.nanoTime() - check); - - } - - } else - { - - // calculate and store the mode choice logsum for the - // usual work location for this worker at the various - // departure time and duration alternativees - setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(t, - departureTimeChoiceAvailability); - - if (hh.getDebugChoiceModels()) - { - hh.logTourObject(loggingHeader, modelLogger, person, t); - } - - todDmuObject.setOriginZone(mgraManager.getTaz(t.getTourOrigMgra())); - todDmuObject.setDestinationZone(mgraManager.getTaz(t.getTourDestMgra())); - - float logsum = (float) todModels[m].computeUtilities(todDmuObject, todDmuObject.getIndexValues(), - departureTimeChoiceAvailability, tourDepartureTimeChoiceSample); - t.setTimeOfDayLogsum(logsum); - - Random hhRandom = hh.getHhRandom(); - int randomCount = hh.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available - // alternative, make choice. - if (todModels[m].getAvailabilityCount() > 0) - { - chosen = todModels[m].getChoiceResult(rn); - - // debug output - if (hh.getDebugChoiceModels()) - { - - double[] utilities = todModels[m].getUtilities(); - double[] probabilities = todModels[m].getProbabilities(); - boolean[] availabilities = todModels[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " - + personTypeString + ", Tour Id: " + t.getTourId() - + ", Tour num: " + tourPurpNum + " of " - + subtourList.size() + " " + tourPurposeName + " tours."); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("-------------------- ------------ -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < todModels[m].getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d out=%-3d, in=%-3d", - k + 1, altStarts[k], altEnds[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", - altString, availabilities[k + 1], utilities[k], - probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d out=%-3d, in=%-3d", chosen, - altStarts[chosen - 1], altEnds[chosen - 1]); - modelLogger.info(String.format( - "Choice: %s, with rn=%.8f, randomCount=%d", altString, rn, - randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to debug - // log file - todModels[m].logAlternativesInfo(choiceModelDescription, - decisionMakerLabel); - todModels[m].logSelectionInfo(choiceModelDescription, - decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate - // model specific log file - loggingHeader = String.format("%s for %s", choiceModelDescription, - decisionMakerLabel); - todModels[m].logUECResults(modelLogger, loggingHeader); - - } - - } else - { - - // since there were no alternatives with valid - // utility, assuming previous - // tour of this type scheduled up to the last - // period, so no periods left - // for this tour. - - // TODO: do a formal check for this so we can still - // flag other reasons why there's - // no valid utility for any alternative - chosen = departureTimeChoiceAvailability.length - 1; - noLaterAlternativeCountTemp++; - - } - - // schedule the chosen alternative - chosenStartPeriod = altStarts[chosen - 1]; - chosenEndPeriod = altEnds[chosen - 1]; - person.scheduleWindow(chosenStartPeriod, chosenEndPeriod); - t.setTourDepartPeriod(chosenStartPeriod); - t.setTourArrivePeriod(chosenEndPeriod); - - if (runModeChoice) - { - - if (hh.getDebugChoiceModels()) - hh.logHouseholdObject( - "Pre At-work Subtour Tour Mode Choice Household " - + hh.getHhId() + ", Tour " + tourPurpNum + " of " - + subtourList.size(), tourMCNonManLogger); - - // set the mode choice attributes needed by - // @variables in the UEC spreadsheets - setModeChoiceDmuAttributes(hh, person, t, chosenStartPeriod, - chosenEndPeriod); - - // use the mcModel object already setup for - // computing logsums and get - // the mode choice, where the selected - // worklocation and subzone an departure time and - // duration are set - // for this work tour. - int chosenMode = mcModel.getModeChoice(mcDmuObject, - t.getTourPrimaryPurpose()); - t.setTourModeChoice(chosenMode); - - } - - } - - } catch (Exception e) - { - String errorMessage = String - .format("Exception caught for HHID=%d, personNum=%d, At-work Subtour Departure time choice, tour ArrayList index=%d.", - hh.getHhId(), person.getPersonNum(), tourPurpNum - 1); - String decisionMakerLabel = String - .format("Final At-work Subtour Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), person.getPersonNum(), person.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, person); - todModels[m].logUECResults(modelLogger, errorMessage); - - logger.error(errorMessage, e); - throw new RuntimeException(); - } - - tourPurpNum++; - - } - - for (int w = 0; w < person.getTimeWindows().length; w++) - person.getTimeWindows()[w] = tempWindow[w]; - - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String - .format("Final At-work Subtour Departure Time Person Object: HH=%d, PersonNum=%d, PersonType=%s", - hh.getHhId(), person.getPersonNum(), person.getPersonType()); - hh.logPersonObject(decisionMakerLabel, modelLogger, person); - } - - } - - hh.setAwtodRandomCount(hh.getHhRandomCount()); - - } - - private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, - int startPeriod, int endPeriod) - { - - t.setTourDepartPeriod(startPeriod); - t.setTourArrivePeriod(endPeriod); - - int workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); - Tour workTour = person.getListOfWorkTours().get(workTourIndex); - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(t); - mcDmuObject.setWorkTourObject(workTour); - mcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), - t.getTourOrigMgra(), t.getTourDestMgra(), household.getDebugChoiceModels()); - - - mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t - .getTourOrigMgra()))); - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager.getTaz(t - .getTourDestMgra()))); - - mcDmuObject.setOriginMgra(t.getTourOrigMgra()); - mcDmuObject.setDestMgra(t.getTourDestMgra()); - - } - - private void setTourModeChoiceLogsumsForDepartureTimeAndDurationAlternatives(Tour tour, - boolean[] altAvailable) - { - - Person person = tour.getPersonObject(); - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todLogger; - String choiceModelDescription = String.format( - "At-work Subtour Mode Choice Logsum calculation for %s Departure Time Choice", - tour.getTourPurpose()); - String decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d", household.getHhId(), person - .getPersonNum(), person.getPersonType(), tour.getTourId(), person - .getListOfWorkTours().size()); - String loggingHeader = String - .format("%s %s", choiceModelDescription, decisionMakerLabel); - - for (int a = 1; a <= altStarts.length; a++) - { - - // if the depart/arrive alternative is unavailable, no need to check - // to see if a logsum has been calculated - if (!altAvailable[a]) continue; - - int startPeriod = altStarts[a - 1]; - int endPeriod = altEnds[a - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " work tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - throw new RuntimeException(e); - } - needToComputeLogsum[index] = false; - } - - } - - todDmuObject.setModeChoiceLogsums(modeChoiceLogsums); - - mcDmuObject.getTourObject().setTourDepartPeriod(0); - mcDmuObject.getTourObject().setTourArrivePeriod(0); - } - - public long getModeChoiceTime() - { - return mcTime; - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); - - MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, - Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - /* - */ - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TazDataManager tazManager = TazDataManager.getInstance(propertyMap); - - BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); - if (!aggAcc.getAccessibilitiesAreBuilt()) - { - aggAcc.setupBuildAccessibilities(propertyMap, false); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - // aggAcc.buildAccessibilityComponents(propertyMap); - - boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, - CtrampApplication.READ_ACCESSIBILITIES); - if (readAccessibilities) - { - - // output data - String projectDirectory = propertyMap - .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - - } - - double[][] expConstants = aggAcc.getExpConstants(); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - - double[][][] sovExpUtilities = null; - double[][][] hovExpUtilities = null; - double[][][] nMotorExpUtilities = null; - double[][][] maasExpUtilities = null; - NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, - hovExpUtilities, nMotorExpUtilities, maasExpUtilities); - - MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( - propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); - - TourModeChoiceModel awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - ModelStructure.AT_WORK_CATEGORY, dmuFactory, logsumHelper); - - SubtourDestChoiceModel testObject = new SubtourDestChoiceModel(propertyMap, modelStructure, - aggAcc, dmuFactory, awmcModel); - System.out.println("SubtourDestChoiceModel object creation passed."); - - SubtourDepartureAndDurationTime testObject2 = new SubtourDepartureAndDurationTime( - propertyMap, modelStructure, dmuFactory, awmcModel); - System.out.println("SubtourDepartureAndDurationTime object creation passed."); - - // String hhHandlerAddress = (String) - // propertyMap.get("RunModel.HouseholdServerAddress"); - // int hhServerPort = Integer.parseInt((String) - // propertyMap.get("RunModel.HouseholdServerPort")); - // - // HouseholdDataManagerIf householdDataManager = new - // HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, - // SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - // - // - // householdDataManager.setPropertyFileValues(propertyMap); - // - // // have the household data manager read the synthetic population - // // files and apply its tables to objects mapping method. - // boolean restartHhServer = false; - // try - // { - // // possible values for the following can be none, ao, cdap, imtf, - // // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, - // // stf, stl - // String restartModel = (String) - // propertyMap.get("RunModel.RestartWithHhServer"); - // if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; - // else if (restartModel.equalsIgnoreCase("uwsl") - // || restartModel.equalsIgnoreCase("ao") - // || restartModel.equalsIgnoreCase("fp") - // || restartModel.equalsIgnoreCase("cdap") - // || restartModel.equalsIgnoreCase("imtf") - // || restartModel.equalsIgnoreCase("imtod") - // || restartModel.equalsIgnoreCase("awf") - // || restartModel.equalsIgnoreCase("awl") - // || restartModel.equalsIgnoreCase("awtod") - // || restartModel.equalsIgnoreCase("jtf") - // || restartModel.equalsIgnoreCase("jtl") - // || restartModel.equalsIgnoreCase("jtod") - // || restartModel.equalsIgnoreCase("inmtf") - // || restartModel.equalsIgnoreCase("inmtl") - // || restartModel.equalsIgnoreCase("inmtod") - // || restartModel.equalsIgnoreCase("stf") - // || restartModel.equalsIgnoreCase("stl")) restartHhServer = false; - // } catch (MissingResourceException e) - // { - // restartHhServer = true; - // } - // - // if (restartHhServer) - // { - // - // householdDataManager.setDebugHhIdsFromHashmap(); - // - // String inputHouseholdFileName = (String) - // propertyMap.get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - // String inputPersonFileName = (String) - // propertyMap.get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - // householdDataManager.setHouseholdSampleRate( 1.0f, 0 ); - // householdDataManager.setupHouseholdDataManager(modelStructure, null, - // inputHouseholdFileName, inputPersonFileName); - // - // } else - // { - // - // householdDataManager.setHouseholdSampleRate( 1.0f, 0 ); - // householdDataManager.setDebugHhIdsFromHashmap(); - // householdDataManager.setTraceHouseholdSet(); - // - // } - - // int id = householdDataManager.getArrayIndex( 1033380 ); - // int id = householdDataManager.getArrayIndex( 1033331 ); - // int id = householdDataManager.getArrayIndex( 423804 ); - // Household[] hh = householdDataManager.getHhArray( id, id ); - // testObject.applyModel( hh[0] ); - // testObject2.applyModel( hh[0], true ); - - /** - * used this block of code to test for typos and implemented dmu - * methiods in the TOD choice UECs - * - * String uecFileDirectory = propertyMap.get( - * CtrampApplication.PROPERTIES_UEC_PATH ); String todUecFileName = - * propertyMap.get( TOD_UEC_FILE_TARGET ); todUecFileName = - * uecFileDirectory + todUecFileName; - * - * ModelStructure modelStructure = new SandagModelStructure(); - * SandagCtrampDmuFactory dmuFactory = new - * SandagCtrampDmuFactory(modelStructure); - * TourDepartureTimeAndDurationDMU todDmuObject = - * dmuFactory.getTourDepartureTimeAndDurationDMU(); - * - * File uecFile = new File(todUecFileName); UtilityExpressionCalculator - * uec = new UtilityExpressionCalculator(uecFile, 10, 0, propertyMap, - * (VariableTable) todDmuObject); - * System.out.println("Subtour departure and duration UEC passed"); - */ - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java deleted file mode 100644 index 1d44542..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/SubtourDestChoiceModel.java +++ /dev/null @@ -1,1185 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.Random; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; - -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.accessibilities.MandatoryAccessibilitiesCalculator; -import org.sandag.abm.accessibilities.NonTransitUtilities; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.util.IndexSort; -import com.pb.common.util.ResourceUtil; - -public class SubtourDestChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(SubtourDestChoiceModel.class); - private transient Logger dcNonManLogger = Logger.getLogger("tourDcNonMan"); - private transient Logger todMcLogger = Logger.getLogger("todMcLogsum"); - - // TODO eventually remove this target - private static final String PROPERTIES_DC_UEC_FILE = "nmdc.uec.file"; - private static final String PROPERTIES_DC_UEC_FILE2 = "nmdc.uec.file2"; - private static final String PROPERTIES_DC_SOA_UEC_FILE = "nmdc.soa.uec.file"; - - private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "nmdc.use.new.soa"; - - private static final String PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY = "nmdc.soa.SampleSize"; - - private static final String PROPERTIES_DC_DATA_SHEET = "nmdc.data.page"; - - private static final String PROPERTIES_DC_AT_WORK_MODEL_SHEET = "nmdc.atwork.model.page"; - - private static final String PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET = "nmdc.soa.atwork.model.page"; - - // size term array indices for purposes are 0-based - public static final int PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET = 11; - public static final int PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET = 10; - public static final int PROPERTIES_AT_WORK_OTHER_SIZE_SHEET = 9; - - private static String[] tourPurposeNames; - private static int[] tourPurposeIndices; - - // all three subtour purposes use the same DC sheet - private static final String[] DC_MODEL_SHEET_KEYS = { - PROPERTIES_DC_AT_WORK_MODEL_SHEET, PROPERTIES_DC_AT_WORK_MODEL_SHEET, - PROPERTIES_DC_AT_WORK_MODEL_SHEET }; - - // all three subtour purposes use the same SOA sheet - private static final String[] DC_SOA_MODEL_SHEET_KEYS = { - PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET, PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET, - PROPERTIES_DC_SOA_AT_WORK_MODEL_SHEET }; - - // all three subtour purposes use the same SOA sheet - private final int[] sizeSheetKeys = { - PROPERTIES_AT_WORK_BUSINESS_SIZE_SHEET, PROPERTIES_AT_WORK_EAT_OUT_SIZE_SHEET, - PROPERTIES_AT_WORK_OTHER_SIZE_SHEET }; - - // set default depart periods that represents each model period - private static final int EA = 1; - private static final int AM = 8; - private static final int MD = 16; - private static final int PM = 26; - private static final int EV = 36; - - private static final int[][][] PERIOD_COMBINATIONS = { - { {AM, AM}, {MD, MD}, {PM, PM}}, { {AM, AM}, {MD, MD}, {PM, PM}}, - { {AM, AM}, {MD, MD}, {PM, PM}} }; - - private static final double[][] PERIOD_COMBINATION_COEFFICIENTS = { - {-3.1453, -0.1029, -2.9056}, {-3.1453, -0.1029, -2.9056}, {-3.1453, -0.1029, -2.9056}}; - - private String tourCategory; - private ModelStructure modelStructure; - - private int[] dcModelIndices; - private HashMap purposeNameIndexMap; - - private HashMap subtourSegmentNameIndexMap; - - private double[][] dcSizeArray; - - private TourModeChoiceDMU mcDmuObject; - private DestChoiceDMU dcDmuObject; - private DestChoiceTwoStageModelDMU dcDistSoaDmuObject; - private DcSoaDMU dcSoaDmuObject; - - private boolean[] needToComputeLogsum; - private double[] modeChoiceLogsums; - - private TourModeChoiceModel mcModel; - private DestinationSampleOfAlternativesModel dcSoaModel; - private ChoiceModelApplication[] dcModel; - private ChoiceModelApplication[] dcModel2; - - private boolean[] dcModel2AltsAvailable; - private int[] dcModel2AltsSample; - private int[] dcModel2SampleValues; - - private BuildAccessibilities aggAcc; - - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private double[] mgraDistanceArray; - - private DestChoiceTwoStageModel dcSoaTwoStageObject; - - private boolean useNewSoaMethod; - - private int soaSampleSize; - - private long soaRunTime; - - public SubtourDestChoiceModel(HashMap propertyMap, - ModelStructure myModelStructure, BuildAccessibilities myAggAcc, - CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel myMcModel) - { - - tourCategory = ModelStructure.AT_WORK_CATEGORY; - modelStructure = myModelStructure; - mcModel = myMcModel; - aggAcc = myAggAcc; - - tourPurposeIndices = new int[3]; - tourPurposeIndices[0] = modelStructure.AT_WORK_PURPOSE_INDEX_BUSINESS; - tourPurposeIndices[1] = modelStructure.AT_WORK_PURPOSE_INDEX_EAT; - tourPurposeIndices[2] = modelStructure.AT_WORK_PURPOSE_INDEX_MAINT; - - tourPurposeNames = new String[3]; - tourPurposeNames[0] = modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME; - tourPurposeNames[1] = modelStructure.AT_WORK_EAT_PURPOSE_NAME; - tourPurposeNames[2] = modelStructure.AT_WORK_MAINT_PURPOSE_NAME; - - logger.info(String.format("creating %s subtour dest choice mode instance", tourCategory)); - - mgraManager = MgraDataManager.getInstance(); - tazs = TazDataManager.getInstance(); - - soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, - PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); - - useNewSoaMethod = Util.getBooleanValueFromPropertyMap(propertyMap, - USE_NEW_SOA_METHOD_PROPERTY_KEY); - - if (useNewSoaMethod) - dcSoaTwoStageObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); - - // create an array of ChoiceModelApplication objects for each choice - // purpose - setupDestChoiceModelArrays(propertyMap, dmuFactory); - - } - - private void setupDestChoiceModelArrays(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - - String dcUecFileName = propertyMap.get(PROPERTIES_DC_UEC_FILE); - dcUecFileName = uecFileDirectory + dcUecFileName; - - String dcUecFileName2 = propertyMap.get(PROPERTIES_DC_UEC_FILE2); - dcUecFileName2 = uecFileDirectory + dcUecFileName2; - - String soaUecFileName = propertyMap.get(PROPERTIES_DC_SOA_UEC_FILE); - soaUecFileName = uecFileDirectory + soaUecFileName; - - int dcModelDataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, - PROPERTIES_DC_DATA_SHEET); - int soaSampleSize = Util.getIntegerValueFromPropertyMap(propertyMap, - PROPERTIES_DC_SOA_NON_MAND_SAMPLE_SIZE_KEY); - - dcDmuObject = dmuFactory.getDestChoiceDMU(); - dcDmuObject.setAggAcc(aggAcc); - dcDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); - - if (useNewSoaMethod) - { - dcDistSoaDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); - dcDistSoaDmuObject.setAggAcc(aggAcc); - dcDistSoaDmuObject.setAccTable(aggAcc.getAccessibilitiesTableObject()); - } - - dcSoaDmuObject = dmuFactory.getDcSoaDMU(); - dcSoaDmuObject.setAggAcc(aggAcc); - - mcDmuObject = dmuFactory.getModeChoiceDMU(); - - int numLogsumIndices = modelStructure.getSkimPeriodCombinationIndices().length; - needToComputeLogsum = new boolean[numLogsumIndices]; - modeChoiceLogsums = new double[numLogsumIndices]; - - // create the arrays of dc model and soa model indices - int[] uecSheetIndices = new int[tourPurposeNames.length]; - int[] soaUecSheetIndices = new int[tourPurposeNames.length]; - - purposeNameIndexMap = new HashMap(tourPurposeNames.length); - - int i = 0; - for (String purposeName : tourPurposeNames) - { - int uecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, DC_MODEL_SHEET_KEYS[i]); - int soaUecIndex = Util.getIntegerValueFromPropertyMap(propertyMap, - DC_SOA_MODEL_SHEET_KEYS[i]); - purposeNameIndexMap.put(purposeName, i); - uecSheetIndices[i] = uecIndex; - soaUecSheetIndices[i] = soaUecIndex; - i++; - } - - // create a lookup array to map purpose index to model index - dcModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int dcModelIndex = 0; - int dcSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, dcModelIndex); - dcModelIndices[dcSegmentIndex] = dcModelIndex++; - } else - { - dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); - } - - dcSegmentIndex++; - } - - // the size term array in aggAcc gives mgra*purpose - need an array of - // all mgras for one purpose - double[][] aggAccDcSizeArray = aggAcc.getSizeTerms(); - subtourSegmentNameIndexMap = new HashMap(); - for (int k = 0; k < tourPurposeIndices.length; k++) - { - subtourSegmentNameIndexMap.put(tourPurposeNames[k], k); - } - - dcSizeArray = new double[tourPurposeNames.length][aggAccDcSizeArray.length]; - for (i = 0; i < aggAccDcSizeArray.length; i++) - { - for (int m : subtourSegmentNameIndexMap.values()) - { - int s = sizeSheetKeys[m]; - dcSizeArray[m][i] = aggAccDcSizeArray[i][s]; - } - } - - // create a sample of alternatives choice model object for use in - // selecting a sample - // of all possible destination choice alternatives. - dcSoaModel = new DestinationSampleOfAlternativesModel(soaUecFileName, soaSampleSize, - propertyMap, mgraManager, dcSizeArray, dcSoaDmuObject, soaUecSheetIndices); - - dcModel = new ChoiceModelApplication[modelIndexMap.size()]; - - if (useNewSoaMethod) - { - dcModel2 = new ChoiceModelApplication[modelIndexMap.size()]; - dcModel2AltsAvailable = new boolean[soaSampleSize + 1]; - dcModel2AltsSample = new int[soaSampleSize + 1]; - dcModel2SampleValues = new int[soaSampleSize]; - } - - i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - - try - { - dcModel[i] = new ChoiceModelApplication(dcUecFileName, uecIndex, dcModelDataSheet, - propertyMap, (VariableTable) dcDmuObject); - - if (useNewSoaMethod) - { - dcModel2[i] = new ChoiceModelApplication(dcUecFileName2, uecIndex, - dcModelDataSheet, propertyMap, (VariableTable) dcDistSoaDmuObject); - } - - i++; - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up ATWork DC ChoiceModelApplication[%d] for model index=%d of %d models", - i, i, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; - - } - - public void applyModel(Household hh) - { - - soaRunTime = 0; - - if (useNewSoaMethod) dcSoaTwoStageObject.resetSoaRunTime(); - else dcSoaModel.resetSoaRunTime(); - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - - Logger modelLogger = dcNonManLogger; - if (hh.getDebugChoiceModels()) - hh.logHouseholdObject("Pre Subtour Location Choice Household " + hh.getHhId() - + " Object", modelLogger); - - Person[] persons = hh.getPersons(); - - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - // get the at-work subtours for this person and choose a destination - // for each. - ArrayList tourList = p.getListOfAtWorkSubtours(); - - int currentTourNum = 0; - for (Tour tour : tourList) - { - - Tour workTour = null; - int workTourIndex = 0; - workTourIndex = tour.getWorkTourIndexFromSubtourId(tour.getTourId()); - workTour = p.getListOfWorkTours().get(workTourIndex); - - int chosen = -1; - try - { - - int homeMgra = hh.getHhMgra(); - int homeTaz = hh.getHhTaz(); - int origMgra = workTour.getTourDestMgra(); - tour.setTourOrigMgra(origMgra); - - // update the MC dmuObject for this person - mcDmuObject.setHouseholdObject(hh); - mcDmuObject.setPersonObject(p); - mcDmuObject.setTourObject(tour); - mcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0, - hh.getDebugChoiceModels()); - mcDmuObject.setOriginMgra(origMgra); - - // update the DC dmuObject for this person - dcDmuObject.setHouseholdObject(hh); - dcDmuObject.setPersonObject(p); - dcDmuObject.setTourObject(tour); - dcDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); - - if (useNewSoaMethod) - { - dcDistSoaDmuObject.setHouseholdObject(hh); - dcDistSoaDmuObject.setPersonObject(p); - dcDistSoaDmuObject.setTourObject(tour); - dcDistSoaDmuObject.setDmuIndexValues(hh.getHhId(), homeTaz, origMgra, 0); - } - - // for At-work Subtour DC, just count remaining At-work - // Subtour tours - int toursLeftCount = tourList.size() - currentTourNum; - dcDmuObject.setToursLeftCount(toursLeftCount); - if (useNewSoaMethod) dcDistSoaDmuObject.setToursLeftCount(toursLeftCount); - - // get the tour location alternative chosen from the sample - if (useNewSoaMethod) - { - chosen = selectLocationFromTwoStageSampleOfAlternatives(tour, mcDmuObject); - soaRunTime += dcSoaTwoStageObject.getSoaRunTime(); - } else - { - chosen = selectLocationFromSampleOfAlternatives(tour, dcDmuObject, - dcSoaDmuObject, mcDmuObject); - soaRunTime += dcSoaModel.getSoaRunTime(); - } - - } catch (RuntimeException e) - { - logger.fatal(String - .format("exception caught selecting %s tour destination choice for hh.hhid=%d, personNum=%d, tourId=%d, purposeName=%s", - tourCategory, hh.getHhId(), p.getPersonNum(), tour.getTourId(), - tour.getSubTourPurpose())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - // set chosen values in tour object - tour.setTourDestMgra(chosen); - - currentTourNum++; - } - - } - - hh.setAwlRandomCount(hh.getHhRandomCount()); - - } - - /** - * - * @return chosen mgra. - */ - private int selectLocationFromSampleOfAlternatives(Tour tour, DestChoiceDMU dcDmuObject, - DcSoaDMU dcSoaDmuObject, TourModeChoiceDMU mcDmuObject) - { - - Logger modelLogger = dcNonManLogger; - - // get the Household object for the person making this subtour - Person person = tour.getPersonObject(); - - // get the Household object for the person making this subtour - Household household = person.getHouseholdObject(); - - // get the tour purpose name - String tourPurposeName = tour.getSubTourPurpose(); - int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); - - dcSoaDmuObject.setDestChoiceSize(dcSizeArray[tourPurposeIndex]); - - // the originMgra in the tour object is already set to the work tour - // dest mgra - // double[] workMgraDistanceArray = - // mandAcc.calculateDistancesForAllMgras( tour.getTourOrigMgra() ); - mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(tour.getTourOrigMgra(), - mgraDistanceArray); - dcSoaDmuObject.setDestDistance(mgraDistanceArray); - - dcDmuObject.setDestChoiceSize(dcSizeArray[tourPurposeIndex]); - dcDmuObject.setDestChoiceDistance(mgraDistanceArray); - - // compute the sample of alternatives set for the person - dcSoaModel.computeDestinationSampleOfAlternatives(dcSoaDmuObject, tour, person, - tourPurposeName, tourPurposeIndex, tour.getTourOrigMgra()); - - // get sample of locations and correction factors for sample - int[] finalSample = dcSoaModel.getSampleOfAlternatives(); - float[] sampleCorrectionFactors = dcSoaModel.getSampleOfAlternativesCorrections(); - - int m = dcModelIndices[tourPurposeIndex]; - int numAlts = dcModel[m].getNumberOfAlternatives(); - - // set the destAltsAvailable array to true for all destination choice - // alternatives for each purpose - boolean[] destAltsAvailable = new boolean[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsAvailable[k] = false; - - // set the destAltsSample array to 1 for all destination choice - // alternatives - // for each purpose - int[] destAltsSample = new int[numAlts + 1]; - for (int k = 0; k <= numAlts; k++) - destAltsSample[k] = 0; - - int[] sampleValues = new int[finalSample.length]; - - // for the destinations and sub-zones in the sample, compute mc logsums - // and - // save in DC dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 1; i < finalSample.length; i++) - { - - int destMgra = finalSample[i]; - sampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); - dcDmuObject.setMcLogsum(destMgra, logsum); - - // set sample of alternatives correction factor used in destination - // choice utility for the sampled alternative. - dcDmuObject.setDcSoaCorrections(destMgra, sampleCorrectionFactors[i]); - - // set availaibility and sample values for the purpose, dcAlt. - destAltsAvailable[finalSample[i]] = true; - destAltsSample[finalSample[i]] = 1; - - } - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "At-work Subtour Location Choice Model for: tour purpose=%s", tourPurposeName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("At-work Subtour Location Choice Model for tour purpose=" - + tourPurposeName + ", Person Num: " + person.getPersonNum() - + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - dcModel[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float logsum = (float) dcModel[m].computeUtilities(dcDmuObject, dcDmuObject.getDmuIndexValues(), - destAltsAvailable, destAltsSample); - - tour.setTourDestinationLogsum(logsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (dcModel[m].getAvailabilityCount() > 0) - { - try - { - chosen = dcModel[m].getChoiceResult(rn); - } catch (Exception e) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), tour.getTourId(), - tourPurposeName)); - throw new RuntimeException(); - } - } - - // write choice model alternative info to log file - int selectedIndex = -1; - for (int j = 1; j < finalSample.length; j++) - { - if (finalSample[j] == chosen) - { - selectedIndex = j; - break; - } - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = dcModel[m].getUtilities(); - double[] probabilities = dcModel[m].getProbabilities(); - boolean[] availabilities = dcModel[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - int[] sortedSampleValueIndices = IndexSort.indexSort(sampleValues); - - double cumProb = 0.0; - for (int j = 1; j < finalSample.length; j++) - { - int k = sortedSampleValueIndices[j]; - int alt = finalSample[k]; - - if (finalSample[k] == chosen) selectedIndex = j; - - cumProb += probabilities[alt - 1]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[alt], utilities[alt - 1], probabilities[alt - 1], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", selectedIndex, chosen); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - dcModel[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - dcModel[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - dcModel[m].logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", - dcDmuObject.getHouseholdObject().getHhId(), dcDmuObject - .getPersonObject().getPersonNum(), tour.getTourId(), - tourPurposeName)); - throw new RuntimeException(); - } - - } - - return chosen; - - } - - /** - * - * @return chosen mgra. - */ - private int selectLocationFromTwoStageSampleOfAlternatives(Tour tour, - TourModeChoiceDMU mcDmuObject) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcNonManLogger; - - // get the Household object for the person making this non-mandatory - // tour - Person person = tour.getPersonObject(); - - // get the Household object for the person making this non-mandatory - // tour - Household household = person.getHouseholdObject(); - - // get the tour purpose name - String tourPurposeName = tour.getSubTourPurpose(); - int tourPurposeIndex = purposeNameIndexMap.get(tourPurposeName); - - // get sample of locations and correction factors for sample using the - // alternate method - // for non-mandatory tour destination choice, the sizeSegmentType INdex - // and sizeSegmentIndex are the same values. - dcSoaTwoStageObject.chooseSample(mgraManager.getTaz(tour.getTourOrigMgra()), - tourPurposeIndex, tourPurposeIndex, soaSampleSize, household.getHhRandom(), - household.getDebugChoiceModels()); - int[] finalSample = dcSoaTwoStageObject.getUniqueSampleMgras(); - double[] sampleCorrectionFactors = dcSoaTwoStageObject - .getUniqueSampleMgraCorrectionFactors(); - int numUniqueAlts = dcSoaTwoStageObject.getNumberofUniqueMgrasInSample(); - - int m = dcModelIndices[tourPurposeIndex]; - int numAlts = dcModel2[m].getNumberOfAlternatives(); - - Arrays.fill(dcModel2AltsAvailable, false); - Arrays.fill(dcModel2AltsSample, 0); - Arrays.fill(dcModel2SampleValues, 999999); - - mcModel.getAnmSkimCalculator().getOpSkimDistancesFromMgra(tour.getTourOrigMgra(), - mgraDistanceArray); - dcDistSoaDmuObject.setMgraDistanceArray(mgraDistanceArray); - - int sizeIndex = subtourSegmentNameIndexMap.get(tourPurposeName); - dcDistSoaDmuObject.setMgraSizeArray(dcSizeArray[sizeIndex]); - - // set sample of alternatives correction factors used in destination - // choice utility for the sampled alternatives. - dcDistSoaDmuObject.setDcSoaCorrections(sampleCorrectionFactors); - - // for the destination mgras in the sample, compute mc logsums and save - // in dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 0; i < numUniqueAlts; i++) - { - - int destMgra = finalSample[i]; - dcModel2SampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = calculateSimpleTODChoiceLogsum(person, tour, destMgra, i); - dcDistSoaDmuObject.setMcLogsum(i, logsum); - - // set availaibility and sample values for the purpose, dcAlt. - dcModel2AltsAvailable[i + 1] = true; - dcModel2AltsSample[i + 1] = 1; - - } - - dcDistSoaDmuObject.setSampleArray(dcModel2SampleValues); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Non-Mandatory Location Choice Model for: tour purpose=%s", tourPurposeName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Non-Mandatory Location Choice Model for tour purpose=" - + tourPurposeName + ", Person Num: " + person.getPersonNum() - + ", Person Type: " + person.getPersonType() + ", TourId=" + tour.getTourId()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - dcModel2[m].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float modelLogsum = (float) dcModel2[m].computeUtilities(dcDistSoaDmuObject, dcDistSoaDmuObject.getDmuIndexValues(), - dcModel2AltsAvailable, dcModel2AltsSample); - tour.setTourDestinationLogsum(modelLogsum); - - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (dcModel2[m].getAvailabilityCount() > 0) - { - try - { - chosen = dcModel2[m].getChoiceResult(rn); - } catch (Exception e) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, in %s destination choice.", - dcDistSoaDmuObject.getHouseholdObject().getHhId(), - dcDistSoaDmuObject.getPersonObject().getPersonNum(), - tour.getTourId(), tourPurposeName)); - throw new RuntimeException(); - } - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = dcModel2[m].getUtilities(); - double[] probabilities = dcModel2[m].getProbabilities(); - boolean[] availabilities = dcModel2[m].getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < finalSample.length; j++) - { - int alt = finalSample[j]; - cumProb += probabilities[j]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - dcModel2[m].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - dcModel2[m].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - - // write UEC calculation results to separate model specific log file - dcModel2[m].logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, tourId=%d, tourPurpose=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", - dcDistSoaDmuObject.getHouseholdObject().getHhId(), - dcDistSoaDmuObject.getPersonObject().getPersonNum(), - tour.getTourId(), tourPurposeName)); - throw new RuntimeException(); - } - - } - - return chosen; - - } - - private void setModeChoiceDmuAttributes(Household household, Person person, Tour t, - int startPeriod, int endPeriod, int sampleDestMgra) - { - - t.setTourDestMgra(sampleDestMgra); - t.setTourDepartPeriod(startPeriod); - t.setTourArrivePeriod(endPeriod); - - int workTourIndex = t.getWorkTourIndexFromSubtourId(t.getTourId()); - Tour workTour = person.getListOfWorkTours().get(workTourIndex); - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(t); - mcDmuObject.setWorkTourObject(workTour); - mcDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), - t.getTourOrigMgra(), sampleDestMgra, household.getDebugChoiceModels()); - - - mcDmuObject.setPTazTerminalTime(tazs.getOriginTazTerminalTime(mgraManager.getTaz(t - .getTourOrigMgra()))); - mcDmuObject.setATazTerminalTime(tazs.getDestinationTazTerminalTime(mgraManager - .getTaz(sampleDestMgra))); - - mcDmuObject.setOriginMgra(t.getTourOrigMgra()); - mcDmuObject.setDestMgra(t.getTourDestMgra()); - - } - - private double calculateSimpleTODChoiceLogsum(Person person, Tour tour, int sampleDestMgra, - int sampleNum) - { - - Household household = person.getHouseholdObject(); - - Arrays.fill(needToComputeLogsum, true); - Arrays.fill(modeChoiceLogsums, -999); - - Logger modelLogger = todMcLogger; - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - choiceModelDescription = String - .format("At-work Subtour Simplified TOD logsum calculations for %s Location Choice, Sample Number %d", - tour.getSubTourPurpose(), sampleNum); - decisionMakerLabel = String.format( - "HH=%d, PersonNum=%d, PersonType=%s, tourId=%d of %d non-mand tours", - household.getHhId(), person.getPersonNum(), person.getPersonType(), - tour.getTourId(), person.getListOfAtWorkSubtours().size()); - loggingHeader = String.format("%s %s", choiceModelDescription, decisionMakerLabel); - } - - int i = 0; - int tourPurposeIndex = purposeNameIndexMap.get(tour.getSubTourPurpose()); - double totalExpUtility = 0.0; - for (int[] combo : PERIOD_COMBINATIONS[tourPurposeIndex]) - { - int startPeriod = combo[0]; - int endPeriod = combo[1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - if (needToComputeLogsum[index]) - { - - String periodString = modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)); - - // set the mode choice attributes needed by @variables in the - // UEC spreadsheets - setModeChoiceDmuAttributes(household, person, tour, startPeriod, endPeriod, - sampleDestMgra); - - if (household.getDebugChoiceModels()) - household.logTourObject(loggingHeader + ", " + periodString, modelLogger, - person, mcDmuObject.getTourObject()); - - try - { - modeChoiceLogsums[index] = mcModel.getModeChoiceLogsum(mcDmuObject, tour, - modelLogger, choiceModelDescription, decisionMakerLabel + ", " - + periodString); - } catch (Exception e) - { - logger.fatal("exception caught applying mcModel.getModeChoiceLogsum() for " - + periodString + " " + tour.getTourPrimaryPurpose() + " tour."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - e.printStackTrace(); - System.exit(-1); - // throw new RuntimeException(e); - } - needToComputeLogsum[index] = false; - } - - double expUtil = Math.exp(modeChoiceLogsums[index] - + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]); - totalExpUtility += expUtil; - - if (household.getDebugChoiceModels()) - modelLogger - .info("i = " - + i - + ", purpose = " - + tourPurposeIndex - + ", " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(startPeriod)) - + " to " - + modelStructure.getModelPeriodLabel(modelStructure - .getModelPeriodIndex(endPeriod)) - + " MCLS = " - + modeChoiceLogsums[index] - + ", ASC = " - + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i] - + ", (MCLS + ASC) = " - + (modeChoiceLogsums[index] + PERIOD_COMBINATION_COEFFICIENTS[tourPurposeIndex][i]) - + ", exp(MCLS + ASC) = " + expUtil + ", cumExpUtility = " - + totalExpUtility); - - i++; - } - - double logsum = Math.log(totalExpUtility); - - if (household.getDebugChoiceModels()) - modelLogger.info("final simplified TOD logsum = " + logsum); - - return logsum; - } - - public void setNonMandatorySoaProbs(double[][][] soaDistProbs, double[][][] soaSizeProbs) - { - if (useNewSoaMethod) - { - dcSoaTwoStageObject.setTazDistProbs(soaDistProbs); - dcSoaTwoStageObject.setMgraSizeProbs(soaSizeProbs); - } - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - - public static void main(String[] args) - { - - // set values for these arguments so an object instance can be created - // and setup run to test integrity of UEC files before running full - // model. - HashMap propertyMap; - - if (args.length == 0) - { - System.out - .println("no properties file base name (without .properties extension) was specified as an argument."); - return; - } else - { - ResourceBundle rb = ResourceBundle.getBundle(args[0]); - propertyMap = ResourceUtil.changeResourceBundleIntoHashMap(rb); - } - - String matrixServerAddress = (String) propertyMap.get("RunModel.MatrixServerAddress"); - String matrixServerPort = (String) propertyMap.get("RunModel.MatrixServerPort"); - - MatrixDataServerIf ms = new MatrixDataServerRmi(matrixServerAddress, - Integer.parseInt(matrixServerPort), MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(ms); - - MgraDataManager mgraManager = MgraDataManager.getInstance(propertyMap); - TazDataManager tazManager = TazDataManager.getInstance(propertyMap); - - /* - * - */ - ModelStructure modelStructure = new SandagModelStructure(); - SandagCtrampDmuFactory dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - BuildAccessibilities aggAcc = BuildAccessibilities.getInstance(); - if (!aggAcc.getAccessibilitiesAreBuilt()) - { - aggAcc.setupBuildAccessibilities(propertyMap, false); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - // aggAcc.buildAccessibilityComponents(propertyMap); - - boolean readAccessibilities = Util.getBooleanValueFromPropertyMap(propertyMap, - CtrampApplication.READ_ACCESSIBILITIES); - if (readAccessibilities) - { - - // output data - String projectDirectory = propertyMap - .get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String accFileName = projectDirectory - + Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file"); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - - } - - double[][] expConstants = aggAcc.getExpConstants(); - - McLogsumsCalculator logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - - double[][][] sovExpUtilities = null; - double[][][] hovExpUtilities = null; - double[][][] nMotorExpUtilities = null; - double[][][] maasExpUtilities = null; - NonTransitUtilities ntUtilities = new NonTransitUtilities(propertyMap, sovExpUtilities, - hovExpUtilities, nMotorExpUtilities, maasExpUtilities); - - MandatoryAccessibilitiesCalculator mandAcc = new MandatoryAccessibilitiesCalculator( - propertyMap, ntUtilities, expConstants, logsumHelper.getBestTransitPathCalculator()); - - String hhHandlerAddress = (String) propertyMap.get("RunModel.HouseholdServerAddress"); - int hhServerPort = Integer.parseInt((String) propertyMap - .get("RunModel.HouseholdServerPort")); - - HouseholdDataManagerIf householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, - hhServerPort, SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - boolean restartHhServer = false; - try - { - // possible values for the following can be none, ao, cdap, imtf, - // imtod, awf, awl, awtod, jtf, jtl, jtod, inmtf, inmtl, inmtod, - // stf, stl - String restartModel = (String) propertyMap.get("RunModel.RestartWithHhServer"); - if (restartModel.equalsIgnoreCase("none")) restartHhServer = true; - else if (restartModel.equalsIgnoreCase("uwsl") || restartModel.equalsIgnoreCase("ao") - || restartModel.equalsIgnoreCase("fp") || restartModel.equalsIgnoreCase("cdap") - || restartModel.equalsIgnoreCase("imtf") - || restartModel.equalsIgnoreCase("imtod") - || restartModel.equalsIgnoreCase("awf") || restartModel.equalsIgnoreCase("awl") - || restartModel.equalsIgnoreCase("awtod") - || restartModel.equalsIgnoreCase("jtf") || restartModel.equalsIgnoreCase("jtl") - || restartModel.equalsIgnoreCase("jtod") - || restartModel.equalsIgnoreCase("inmtf") - || restartModel.equalsIgnoreCase("inmtl") - || restartModel.equalsIgnoreCase("inmtod") - || restartModel.equalsIgnoreCase("stf") || restartModel.equalsIgnoreCase("stl")) - restartHhServer = false; - } catch (MissingResourceException e) - { - restartHhServer = true; - } - - if (restartHhServer) - { - - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = (String) propertyMap - .get(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(1.0f, 0); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, - inputPersonFileName); - - } else - { - - householdDataManager.setHouseholdSampleRate(1.0f, 0); - householdDataManager.setDebugHhIdsFromHashmap(); - householdDataManager.setTraceHouseholdSet(); - - } - - // int id = householdDataManager.getArrayIndex( 1033380 ); - // int id = householdDataManager.getArrayIndex( 1033331 ); - int id = householdDataManager.getArrayIndex(423804); - Household[] hh = householdDataManager.getHhArray(id, id); - - TourModeChoiceModel awmcModel = new TourModeChoiceModel(propertyMap, modelStructure, - ModelStructure.AT_WORK_CATEGORY, dmuFactory, logsumHelper); - - SubtourDestChoiceModel testObject = new SubtourDestChoiceModel(propertyMap, modelStructure, - aggAcc, dmuFactory, awmcModel); - - testObject.applyModel(hh[0]); - - /** - * used this block of code to test for typos and implemented dmu methods - * in the TOD choice UECs - * - * String uecFileDirectory = propertyMap.get( - * CtrampApplication.PROPERTIES_UEC_PATH ); - * - * ModelStructure modelStructure = new SandagModelStructure(); - * SandagCtrampDmuFactory dmuFactory = new - * SandagCtrampDmuFactory(modelStructure); - * - * String dcUecFileName = propertyMap.get( PROPERTIES_DC_UEC_FILE ); - * DestChoiceDMU dcDmuObject = dmuFactory.getDestChoiceDMU(); File - * uecFile = new File(uecFileDirectory + dcUecFileName); - * UtilityExpressionCalculator uec = new - * UtilityExpressionCalculator(uecFile, 13, 0, propertyMap, - * (VariableTable) dcDmuObject); - * System.out.println("Subtour destination choice UEC passed"); - * - * String soaUecFileName = propertyMap.get( PROPERTIES_DC_SOA_UEC_FILE - * ); DcSoaDMU dcSoaDmuObject = dmuFactory.getDcSoaDMU(); uecFile = new - * File(uecFileDirectory + soaUecFileName); uec = new - * UtilityExpressionCalculator(uecFile, 7, 0, propertyMap, - * (VariableTable) dcSoaDmuObject); - * System.out.println("Subtour destination choice SOA UEC passed"); - */ - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java deleted file mode 100644 index 3731e65..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TNCAndTaxiWaitTimeCalculator.java +++ /dev/null @@ -1,264 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; - -import umontreal.iro.lecuyer.probdist.LognormalDist; - - - -public class TNCAndTaxiWaitTimeCalculator implements Serializable{ - - private static Logger logger = Logger.getLogger(TNCAndTaxiWaitTimeCalculator.class); - - private float[] startPopEmpPerSqMi; - - private LognormalDist[] TNCSingleWaitTimeDistribution; - private LognormalDist[] TNCSharedWaitTimeDistribution; - private LognormalDist[] TaxiWaitTimeDistribution; - float[] meanTNCSingleWaitTime ; - float[] meanTNCSharedWaitTime ; - float[] meanTaxiWaitTime ; - - /** - * Constructor; doesn't do anything (call @createTimeDistributions method next) - */ - public TNCAndTaxiWaitTimeCalculator(){ - - } - - /** - * Reads the propertyMap and finds values for property arrays - * TNC.waitTime.mean, Taxi.waitTime.mean, - * TNC.waitTime.sd, and Taxi.waitTime.sd - containing arrays - * of wait time and standard deviations for TNCs and Taxis by area type, - * plus an array of end ranges for area type (pop+emp)/sq miles - * with an implied start value of 0. - * Creates and stores umontreal.iro.lecuyer.probdist.LognormalDist - * TNCWaitTimeDistribution[] and TaxiWaitTimeDistribution[] where - * each element of the distribution corresponds to the areatype range. - * @param propertyMap - */ - - public void createWaitTimeDistributions(HashMap propertyMap){ - - //read properties - meanTNCSingleWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.single.waitTime.mean"); - float[] sdTNCSingleWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.single.waitTime.sd"); - - meanTNCSharedWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.shared.waitTime.mean"); - float[] sdTNCSharedWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "TNC.shared.waitTime.sd"); - - meanTaxiWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "Taxi.waitTime.mean"); - float[] sdTaxiWaitTime = Util.getFloatArrayFromPropertyMap(propertyMap, "Taxi.waitTime.sd"); - - startPopEmpPerSqMi = Util.getFloatArrayFromPropertyMap(propertyMap, "WaitTimeDistribution.EndPopEmpPerSqMi"); - - // create the distribution arrays - TNCSingleWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; - TNCSharedWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; - TaxiWaitTimeDistribution = new LognormalDist[startPopEmpPerSqMi.length]; - - //iterate through area types - for(int i = 0; i< startPopEmpPerSqMi.length;++i){ - - // calculate the location and scale parameters from the mean and standard deviations - double locationTNCSingleWaitTime = calculateLocation(meanTNCSingleWaitTime[i], sdTNCSingleWaitTime[i]); - double scaleTNCSingleWaitTime = calculateScale(meanTNCSingleWaitTime[i], sdTNCSingleWaitTime[i]); - - - double locationTNCSharedWaitTime = calculateLocation(meanTNCSharedWaitTime[i], sdTNCSharedWaitTime[i]); - double scaleTNCSharedWaitTime = calculateScale(meanTNCSharedWaitTime[i], sdTNCSharedWaitTime[i]); - - // create the TNC wait time distribution for this area type - TNCSingleWaitTimeDistribution[i] = new LognormalDist(locationTNCSingleWaitTime, scaleTNCSingleWaitTime); - TNCSharedWaitTimeDistribution[i] = new LognormalDist(locationTNCSharedWaitTime, scaleTNCSharedWaitTime); - - double locationTaxiWaitTime = calculateLocation(meanTaxiWaitTime[i], sdTaxiWaitTime[i]); - double scaleTaxiWaitTime = calculateScale(meanTaxiWaitTime[i], sdTaxiWaitTime[i]); - - TaxiWaitTimeDistribution[i] = new LognormalDist(locationTaxiWaitTime, scaleTaxiWaitTime); - - } - } - - /** - * Calculate the lognormal distribution location given - * the mean and standard deviation of the distribution - * according to the formula: - * - * location = ln(mean/sqrt(1 + variance/mean^2)) - * - * @param mean - * @param standardDeviation - * @return Location variable (u) - */ - public double calculateLocation(double mean, double standardDeviation){ - - double variance = standardDeviation * standardDeviation; - double meanSquared = mean * mean; - double denom = Math.sqrt(1.0 + (variance/meanSquared)); - double location = mean/denom; - if(location<=0){ - logger.error("Error: Trying to calculation location for mean "+mean - +" and standard deviation "+standardDeviation); - throw new RuntimeException(); - } - - return Math.log(location); - - } - - /** - * Calculate the lognormal distribution scale given - * the mean and standard deviation of the distribution - * according to the formula: - * - * scale = sqrt(ln(1 + variance/mean^2)); - * - * @param mean - * @param standardDeviation - * @return Scale variable (sigma) - */ - public double calculateScale(double mean, double standardDeviation){ - - double variance = standardDeviation * standardDeviation; - double meanSquared = mean * mean; - return Math.sqrt(Math.log(1 + variance/meanSquared)); - } - - /** - * Sample from the Single TNC wait time distribution and return the wait time. - * @param rnum A unit-distributed random number. - * @param popEmpPerSqMi The population plus employment divided by square miles - * @return The sampled TNC wait time. - */ - public double sampleFromSingleTNCWaitTimeDistribution(double rnum, double popEmpPerSqMi){ - - for(int i = 0; i < startPopEmpPerSqMi.length;++i){ - - if(popEmpPerSqMi zoneValues = new HashMap(); - - for (int i = 1; i <= zoneTable.getRowCount(); i++) - { - int zone = (int) zoneTable.getValueAt(i, zoneCol); - if (zoneValues.containsKey(zone)) - { - logger.fatal(String - .format("zone employment table read from %s has duplicate value for ZONE=%d in column %d at record number %d", - zoneDataFileName, zone, zoneCol, (i + 1))); - throw new RuntimeException(); - } else - { - zoneValues.put(zone, i); - numZones++; - if (zone > maxZone) maxZone = zone; - } - - } - - } catch (IOException e) - { - logger.error(String - .format("Exception occurred reading zonal employment data file: %s into TableDataSet object.", - zoneDataFileName)); - throw new RuntimeException(); - } - - NUM_ZONES = numZones; - - // store the row numbers for each zone so that zonal attributes can be - // retrieved later using given a zone number - indexToZone = new int[numZones + 1]; - zoneTableRow = new int[maxZone + 1]; - for (int i = 1; i <= zoneTable.getRowCount(); i++) - { - int zone = (int) zoneTable.getValueAt(i, zoneCol); - zoneTableRow[zone] = i; - indexToZone[i] = zone; - } - - return zoneTable; - - } - - private void readWalkPercentagesFile(String fileName) - { - - int taz; - float[] shrtArray = new float[NUM_ZONES + 1]; - float[] longArray = new float[NUM_ZONES + 1]; - zonalWalkPctArray = new float[3][NUM_ZONES + 1]; - Arrays.fill(zonalWalkPctArray[0], 1.0f); - Arrays.fill(zonalWalkPctArray[1], 0.0f); - Arrays.fill(zonalWalkPctArray[2], 0.0f); - - if (fileName != null) - { - - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet(" " + reader.getDelimSet()); - TableDataSet wa = reader.readFile(new File(fileName)); - - int tazPosition = wa.getColumnPosition(walkPctZoneFieldName); - if (tazPosition <= 0) - { - logger.fatal(String - .format("expected zone field name=%s was not a field in the walk access file: %s.", - WALK_PERCENTAGE_FILE_ZONE_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int shrtPosition = wa.getColumnPosition(walkPctShortFieldName); - if (shrtPosition <= 0) - { - logger.fatal(String - .format("expected short field name=%s was not a field in the walk access file: %s.", - WALK_PERCENTAGE_FILE_SHORT_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int longPosition = wa.getColumnPosition(walkPctLongFieldName); - if (longPosition <= 0) - { - logger.fatal(String - .format("expected long field name=%s was not a field in the walk access file: %s.", - WALK_PERCENTAGE_FILE_LONG_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - for (int j = 1; j <= wa.getRowCount(); j++) - { - taz = (int) wa.getValueAt(j, tazPosition); - shrtArray[taz] = wa.getValueAt(j, shrtPosition); - longArray[taz] = wa.getValueAt(j, longPosition); - zonalWalkPctArray[1][taz] = shrtArray[taz]; - zonalWalkPctArray[2][taz] = longArray[taz]; - zonalWalkPctArray[0][taz] = (float) (1.0 - (shrtArray[taz] + longArray[taz])); - } - - } catch (IOException e) - { - logger.fatal( - String.format("exception caught reading walk access file: %s", fileName), e); - } - - } else - { - - logger.fatal("no zonal walk access data file was named in properties file with target: 'WalkPercentages.file ='."); - throw new RuntimeException(); - - } - - } - - private void readZonalAccessibilitiesFile(String fileName) - { - - int taz; - pkAutoRetail = new float[NUM_ZONES + 1]; - pkAutoTotal = new float[NUM_ZONES + 1]; - opAutoRetail = new float[NUM_ZONES + 1]; - opAutoTotal = new float[NUM_ZONES + 1]; - pkTransitRetail = new float[NUM_ZONES + 1]; - pkTransitTotal = new float[NUM_ZONES + 1]; - opTransitRetail = new float[NUM_ZONES + 1]; - opTransitTotal = new float[NUM_ZONES + 1]; - nonMotorizedRetail = new float[NUM_ZONES + 1]; - nonMotorizedTotal = new float[NUM_ZONES + 1]; - - if (fileName != null) - { - - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - reader.setDelimSet(" " + reader.getDelimSet()); - TableDataSet acc = reader.readFile(new File(fileName)); - - int tazPosition = acc.getColumnPosition(ACCESSIBILITIES_FILE_ZONE_FIELD_NAME); - if (tazPosition <= 0) - { - logger.fatal(String - .format("expected zone field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_FILE_ZONE_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int pkAutoRetailPosition = acc - .getColumnPosition(ACCESSIBILITIES_PEAK_AUTO_RETAIL_FIELD_NAME); - if (pkAutoRetailPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_PEAK_AUTO_RETAIL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int pkAutoTotalPosition = acc - .getColumnPosition(ACCESSIBILITIES_PEAK_AUTO_TOTAL_FIELD_NAME); - if (pkAutoTotalPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_PEAK_AUTO_TOTAL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int pkTransitRetailPosition = acc - .getColumnPosition(ACCESSIBILITIES_PEAK_TRANSIT_RETAIL_FIELD_NAME); - if (pkTransitRetailPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_PEAK_TRANSIT_RETAIL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int pkTransitTotalPosition = acc - .getColumnPosition(ACCESSIBILITIES_PEAK_TRANSIT_TOTAL_FIELD_NAME); - if (pkTransitTotalPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_PEAK_TRANSIT_TOTAL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int opAutoRetailPosition = acc - .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_AUTO_RETAIL_FIELD_NAME); - if (opAutoRetailPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_OFF_PEAK_AUTO_RETAIL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int opAutoTotalPosition = acc - .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_AUTO_TOTAL_FIELD_NAME); - if (opAutoTotalPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_OFF_PEAK_AUTO_TOTAL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int opTransitRetailPosition = acc - .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_TRANSIT_RETAIL_FIELD_NAME); - if (opTransitRetailPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_OFF_PEAK_TRANSIT_RETAIL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int opTransitTotalPosition = acc - .getColumnPosition(ACCESSIBILITIES_OFF_PEAK_TRANSIT_TOTAL_FIELD_NAME); - if (opTransitTotalPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_OFF_PEAK_TRANSIT_TOTAL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int nonMotorizedRetailPosition = acc - .getColumnPosition(ACCESSIBILITIES_NON_MOTORIZED_RETAIL_FIELD_NAME); - if (nonMotorizedRetailPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_NON_MOTORIZED_RETAIL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - int nonMotorizedTotalPosition = acc - .getColumnPosition(ACCESSIBILITIES_NON_MOTORIZED_TOTAL_FIELD_NAME); - if (nonMotorizedTotalPosition <= 0) - { - logger.fatal(String - .format("expected field name=%s was not a field in the zonal accessibilities file: %s.", - ACCESSIBILITIES_NON_MOTORIZED_TOTAL_FIELD_NAME, fileName)); - throw new RuntimeException(); - } - - for (int i = 1; i <= acc.getRowCount(); i++) - { - taz = (int) acc.getValueAt(i, tazPosition); - pkAutoRetail[taz] = acc.getValueAt(i, pkAutoRetailPosition); - pkAutoTotal[taz] = acc.getValueAt(i, pkAutoTotalPosition); - pkTransitRetail[taz] = acc.getValueAt(i, pkTransitRetailPosition); - pkTransitTotal[taz] = acc.getValueAt(i, pkTransitTotalPosition); - opAutoRetail[taz] = acc.getValueAt(i, opAutoRetailPosition); - opAutoTotal[taz] = acc.getValueAt(i, opAutoTotalPosition); - opTransitRetail[taz] = acc.getValueAt(i, opTransitRetailPosition); - opTransitTotal[taz] = acc.getValueAt(i, opTransitTotalPosition); - nonMotorizedRetail[taz] = acc.getValueAt(i, nonMotorizedRetailPosition); - nonMotorizedTotal[taz] = acc.getValueAt(i, nonMotorizedTotalPosition); - } - - } catch (IOException e) - { - logger.fatal(String.format("exception caught reading accessibilities file: %s", - fileName), e); - } - - } else - { - - logger.fatal("no zonal accessibilities data file was named in properties file with target: " - + fileName); - throw new RuntimeException(); - - } - - } - - /** - * @param alt - * is the DC alternaive number - * @return zone number for the DC alt. - */ - private int getZoneFromAlt(int alt) - { - int zone = (int) ((alt - 1) / NUM_SUBZONES) + 1; - if (zone < 1 || zone > NUM_ZONES) - { - logger.fatal(String.format( - "invalid value for zone index = %d, determined for alt = %d.", zone, alt)); - logger.fatal(String.format("NUM_ZONES = %d, NUM_SUBZONES = %d.", NUM_ZONES, - NUM_SUBZONES)); - throw new RuntimeException(); - } - return zone; - } - - /** - * @param alt - * is the DC alternaive number - * @return walk subzone index for the DC alt. - */ - private int getWalkSubzoneFromAlt(int alt) - { - int zone = getZoneFromAlt(alt); - int subzone = alt - (zone - 1) * NUM_SUBZONES - 1; - if (subzone < 0 || subzone >= NUM_SUBZONES) - { - logger.fatal(String - .format("invalid value for walk subzone index = %d, zone = %d, determined for alt = %d.", - subzone, zone, alt)); - logger.fatal(String.format("NUM_ZONES = %d, NUM_SUBZONES = %d.", NUM_ZONES, - NUM_SUBZONES)); - throw new RuntimeException(); - } - return subzone; - } - - public String testRemote() - { - return String.format("testRemote() method in %s called.", this.getClass() - .getCanonicalName()); - } - - public int[] getAltToZoneArray() - { - return altToZone; - } - - public int[] getAltToSubZoneArray() - { - return altToSubZone; - } - - public int[] getIndexToZoneArray() - { - return indexToZone; - } - - public int[] getZoneTableRowArray() - { - return zoneTableRow; - } - - /** - * - * @param field - * is the field name to be checked against the column names in - * the zone data table. - * @return true if field matches one of the zone data table column names, - * otherwise false. - */ - public boolean isValidZoneTableField(String field) - { - return zoneDataTable.getColumnPosition(field) >= 0; - } - - public String[] getZoneDataTableColumnLabels() - { - return zoneDataTable.getColumnLabels(); - } - - public int getNumberOfZones() - { - return NUM_ZONES; - } - - public int getNumberOfSubZones() - { - return NUM_SUBZONES; - } - - public String[] getSubZoneNames() - { - return subZoneNames; - } - - public double[] getZonalWalkPercentagesForTaz(int taz) - { - double[] percentages = new double[NUM_SUBZONES]; - for (int i = 0; i < NUM_SUBZONES; i++) - percentages[i] = zonalWalkPctArray[i][taz]; - return percentages; - } - - public float getZoneTableValue(int taz, String fieldName) - { - // get the table row number for the TAZ passed in - int rowIndex = zoneTableRow[taz]; - - // get the table value from the rowIndex and fieldname passed in - return zoneDataTable.getValueAt(rowIndex, fieldName); - } - - // get the table column from the fieldname passed in - public int[] getZoneTableIntColumn(String fieldName) - { - return zoneDataTable.getColumnAsInt(fieldName); - } - - // get the table column from the fieldname passed in - public float[] getZoneTableFloatColumn(String fieldName) - { - return zoneDataTable.getColumnAsFloat(fieldName); - } - - /** - * @param tableRowNumber - * is the zone table row number - * @return zone number for the table row. - */ - public int getTazNumber(int tableRowNumber) - { - return (int) zoneDataTable.getValueAt(tableRowNumber, tazDataZoneFieldName); - } - - /** - * @return area type array from the zone data table. - */ - public int[] getZonalAreaType() - { - int atFieldPosition = zoneDataTable.getColumnPosition(tazDataAtFieldName); - if (atFieldPosition < 0) - { - logger.error(String - .format("The area type field name = %s defined in %s is not found as a field name in the zone data table.", - tazDataAtFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsInt(atFieldPosition); - } - - /** - * @return district array from the zone data table. - */ - public int[] getZonalDistrict() - { - int districtFieldPosition = zoneDataTable.getColumnPosition(tazDataDistFieldName); - if (districtFieldPosition < 0) - { - logger.error(String - .format("The district field name = %s defined in %s is not found as a field name in the zone data table.", - tazDataDistFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsInt(districtFieldPosition); - } - - /** - * @return county array from the zone data table. - */ - public int[] getZonalCounty() - { - int countyFieldPosition = zoneDataTable.getColumnPosition(tazDataCountyFieldName); - if (countyFieldPosition < 0) - { - logger.error(String - .format("The county field name = %s defined in %s is not found as a field name in the zone data table.", - tazDataCountyFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsInt(countyFieldPosition); - } - - public int getZoneIsCbd(int taz) - { - return getZoneIsInAreaType(taz, areaTypes[cbdAreaTypesArrayIndex]); - } - - public int getZoneIsUrban(int taz) - { - return getZoneIsInAreaType(taz, areaTypes[urbanAreaTypesArrayIndex]); - } - - public int getZoneIsSuburban(int taz) - { - return getZoneIsInAreaType(taz, areaTypes[suburbanAreaTypesArrayIndex]); - } - - public int getZoneIsRural(int taz) - { - return getZoneIsInAreaType(taz, areaTypes[ruralAreaTypesArrayIndex]); - } - - private int getZoneIsInAreaType(int taz, int[] areaTypes) - { - int returnValue = 0; - int tazAreaType = (int) getZoneTableValue(taz, tazDataAtFieldName); - for (int atIndex : areaTypes) - { - if (tazAreaType == atIndex) - { - returnValue = 1; - break; - } - } - return returnValue; - } - - /** - * @return parkTot array from the zone data table. - */ - public int[] getZonalParkTot() - { - int parkTotFieldPosition = zoneDataTable.getColumnPosition(parkTotFieldName); - if (parkTotFieldPosition < 0) - { - logger.error(String - .format("The parkTot field name = %s defined in %s is not found as a field name in the zone data table.", - parkTotFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsInt(parkTotFieldPosition); - } - - /** - * @return parkLong array from the zone data table. - */ - public int[] getZonalParkLong() - { - int parkLongFieldPosition = zoneDataTable.getColumnPosition(parkLongFieldName); - if (parkLongFieldPosition < 0) - { - logger.error(String - .format("The parkLong field name = %s defined in %s is not found as a field name in the zone data table.", - parkLongFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsInt(parkLongFieldPosition); - } - - /** - * @return propFree array from the zone data table. - */ - public float[] getZonalPropFree() - { - int propFreeFieldPosition = zoneDataTable.getColumnPosition(propFreeFieldName); - if (propFreeFieldPosition < 0) - { - logger.error(String - .format("The propFree field name = %s defined in %s is not found as a field name in the zone data table.", - propFreeFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsFloat(propFreeFieldPosition); - } - - /** - * @return parkRate array from the zone data table. - */ - public float[] getZonalParkRate() - { - int parkRateFieldPosition = zoneDataTable.getColumnPosition(parkRateFieldName); - if (parkRateFieldPosition < 0) - { - logger.error(String - .format("The parkRate field name = %s defined in %s is not found as a field name in the zone data table.", - parkRateFieldName, this.getClass().getName())); - throw new RuntimeException(); - } - return zoneDataTable.getColumnAsFloat(parkRateFieldPosition); - } - - public float[] getPkAutoRetailAccessibity() - { - return pkAutoRetail; - } - - public float[] getPkAutoTotalAccessibity() - { - return pkAutoTotal; - } - - public float[] getPkTransitRetailAccessibity() - { - return pkTransitRetail; - } - - public float[] getPkTransitTotalAccessibity() - { - return pkTransitTotal; - } - - public float[] getOpAutoRetailAccessibity() - { - return opAutoRetail; - } - - public float[] getOpAutoTotalAccessibity() - { - return opAutoTotal; - } - - public float[] getOpTransitRetailAccessibity() - { - return opTransitRetail; - } - - public float[] getOpTransitTotalAccessibity() - { - return opTransitTotal; - } - - public float[] getNonMotorizedRetailAccessibity() - { - return nonMotorizedRetail; - } - - public float[] getNonMotorizedTotalAccessibity() - { - return nonMotorizedTotal; - } - - public enum AreaType - { - CBD, URBAN, SUBURBAN, RURAL - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java deleted file mode 100644 index 0e051c5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataHandlerRmi.java +++ /dev/null @@ -1,297 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; - -/** - * This class provides methods defined in the TazDataIf interface for accessing - * zonal data stored in its TazDataManager object. - * - * A CT-RAMP tour based model application could create an instance of a subclass - * of this class, where additional project specific varaible definitions and - * methods are defined and pass that instance to its model component objects. - * - * Alternatively, an application could use TazDataHandlerRmi as the base class - * instead and create a "remoteable" subclass. The TazDataHandlerRmi class - * implements the same interface, so the model component classes can be unaware - * of whether the taz data handler object accesses zonal data from its member - * object or remotely from a server. Those methods in the rmi class access zonal - * data from a TazDataManager object contained in a "taz data server" object - * which must exist in a separate JVM on the same machine or on another - * addressable machine over the network. - * - * The flexibility provided by this design is intended to allow the "local" - * instance to be declared and passed within a single JVM to model components - * for possibly greater performance (yet to be tested and proven) at production - * run time. The "rmi" instance however allows the model components to access - * zonal data from a "long-running process" (the server class may execute for - * weeks or months). This approach aids in model development as during - * development, model applications can be written to skip startup procedures for - * reading zonal data, and access them directly from the server that is already - * running. - * - * A similar approach is planned for managing objects such as Household objects - * and ModelResults objects so that model components, for example individual - * non-mandatory tour related models which occur well into the tour based model - * stream, can be run in a "hot-start" fasion, where the model component of - * interest is executed immediately where all the preliminary data and prior - * model results it requires are stored in long-running server objects. Testing - * and debugging of these model components can occur without the time required - * to run through all preliminary steps. - * - * - */ - -public class TazDataHandlerRmi - implements TazDataIf, Serializable -{ - - UtilRmi remote; - String connectString; - - public TazDataHandlerRmi(String hostname, int port, String className) - { - - connectString = String.format("//%s:%d/%s", hostname, port, className); - remote = new UtilRmi(connectString); - - } - - public String testRemote() - { - Object[] objArray = {}; - return (String) remote.method("testRemote", objArray); - } - - public int[] getAltToZoneArray() - { - Object[] objArray = {}; - return (int[]) remote.method("getAltToZoneArray", objArray); - } - - public int[] getAltToSubZoneArray() - { - Object[] objArray = {}; - return (int[]) remote.method("getAltToSubZoneArray", objArray); - } - - public int[] getIndexToZoneArray() - { - Object[] objArray = {}; - return (int[]) remote.method("getIndexToZoneArray", objArray); - } - - public int[] getZoneTableRowArray() - { - Object[] objArray = {}; - return (int[]) remote.method("getZoneTableRowArray", objArray); - } - - /** - * @param field - * is the field name to be checked against the column names in - * the zone data table. - * @return true if field matches one of the zone data table column names, - * otherwise false. - */ - public boolean isValidZoneTableField(String field) - { - Object[] objArray = {field}; - return (Boolean) remote.method("isValidZoneTableField", objArray); - } - - public String[] getZoneDataTableColumnLabels() - { - Object[] objArray = {}; - return (String[]) remote.method("getZoneDataTableColumnLabels", objArray); - } - - public int getNumberOfZones() - { - Object[] objArray = {}; - return (Integer) remote.method("getNumberOfZones", objArray); - } - - public int getNumberOfSubZones() - { - Object[] objArray = {}; - return (Integer) remote.method("getNumberOfSubZones", objArray); - } - - public String[] getSubZoneNames() - { - Object[] objArray = {}; - return (String[]) remote.method("getSubZoneNames", objArray); - } - - public double[] getZonalWalkPercentagesForTaz(int taz) - { - Object[] objArray = {taz}; - return (double[]) remote.method("getZonalWalkPercentagesForTaz", objArray); - } - - public float getZoneTableValue(int taz, String fieldName) - { - Object[] objArray = {taz, fieldName}; - return (Float) remote.method("getZoneTableValue", objArray); - } - - public int[] getZoneTableIntColumn(String fieldName) - { - Object[] objArray = {fieldName}; - return (int[]) remote.method("getZoneTableIntColumn", objArray); - } - - // get the table column from the fieldname passed in - public float[] getZoneTableFloatColumn(String fieldName) - { - Object[] objArray = {fieldName}; - return (float[]) remote.method("getZoneTableFloatColumn", objArray); - } - - /** - * @param tableRowNumber - * is the zone table row number - * @return zone number for the table row. - */ - public int getTazNumber(int tableRowNumber) - { - Object[] objArray = {tableRowNumber}; - return (Integer) remote.method("getTazNumber", objArray); - } - - /** - * @return area type from the zone data table for the zone. - */ - public int[] getZonalAreaType() - { - Object[] objArray = {}; - return (int[]) remote.method("getZonalAreaType", objArray); - } - - /** - * @return district from the zone data table for the zone. - */ - public int[] getZonalDistrict() - { - Object[] objArray = {}; - return (int[]) remote.method("getZonalDistrict", objArray); - } - - public int[] getZonalParkTot() - { - Object[] objArray = {}; - return (int[]) remote.method("getZonalParkTot", objArray); - } - - public int[] getZonalParkLong() - { - Object[] objArray = {}; - return (int[]) remote.method("getZonalParkLong", objArray); - } - - public float[] getZonalPropFree() - { - Object[] objArray = {}; - return (float[]) remote.method("getZonalPropFree", objArray); - } - - public float[] getZonalParkRate() - { - Object[] objArray = {}; - return (float[]) remote.method("getZonalParkRate", objArray); - } - - /** - * @return integer county value from the zone data table for the zone. - */ - public int[] getZonalCounty() - { - Object[] objArray = {}; - return (int[]) remote.method("getZonalCounty", objArray); - } - - public int getZoneIsCbd(int taz) - { - Object[] objArray = {taz}; - return (Integer) remote.method("getZoneIsCbd", objArray); - } - - public int getZoneIsUrban(int taz) - { - Object[] objArray = {taz}; - return (Integer) remote.method("getZoneIsUrban", objArray); - } - - public int getZoneIsSuburban(int taz) - { - Object[] objArray = {taz}; - return (Integer) remote.method("getZoneIsSuburban", objArray); - } - - public int getZoneIsRural(int taz) - { - Object[] objArray = {taz}; - return (Integer) remote.method("getZoneIsRural", objArray); - } - - public float[] getPkAutoRetailAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getPkAutoRetailAccessibity", objArray); - } - - public float[] getPkAutoTotalAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getPkAutoTotalAccessibity", objArray); - } - - public float[] getPkTransitRetailAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getPkTransitRetailAccessibity", objArray); - } - - public float[] getPkTransitTotalAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getPkTransitTotalAccessibity", objArray); - } - - public float[] getOpAutoRetailAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getOpAutoRetailAccessibity", objArray); - } - - public float[] getOpAutoTotalAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getOpAutoTotalAccessibity", objArray); - } - - public float[] getOpTransitRetailAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getOpTransitRetailAccessibity", objArray); - } - - public float[] getOpTransitTotalAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getOpTransitTotalAccessibity", objArray); - } - - public float[] getNonMotorizedRetailAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getNonMotorizedRetailAccessibity", objArray); - } - - public float[] getNonMotorizedTotalAccessibity() - { - Object[] objArray = {}; - return (float[]) remote.method("getNonMotorizedTotalAccessibity", objArray); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java deleted file mode 100644 index f43c846..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TazDataIf.java +++ /dev/null @@ -1,152 +0,0 @@ -package org.sandag.abm.ctramp; - -/** - * Created by IntelliJ IDEA. User: Jim Date: Jul 1, 2008 Time: 9:58:21 AM - * - * Interface for accessing zonal information used by CT-RAMP modules - */ -public interface TazDataIf -{ - - String testRemote(); - - int[] getAltToZoneArray(); - - int[] getAltToSubZoneArray(); - - int[] getIndexToZoneArray(); - - int[] getZoneTableRowArray(); - - int getZoneIsCbd(int taz); - - int getZoneIsUrban(int taz); - - int getZoneIsSuburban(int taz); - - int getZoneIsRural(int taz); - - float[] getPkAutoRetailAccessibity(); - - float[] getPkAutoTotalAccessibity(); - - float[] getPkTransitRetailAccessibity(); - - float[] getPkTransitTotalAccessibity(); - - float[] getOpAutoRetailAccessibity(); - - float[] getOpAutoTotalAccessibity(); - - float[] getOpTransitRetailAccessibity(); - - float[] getOpTransitTotalAccessibity(); - - float[] getNonMotorizedRetailAccessibity(); - - float[] getNonMotorizedTotalAccessibity(); - - /** - * - * @param field - * is the field name to be checked against the column names in - * the zone data table. - * @return true if field matches one of the zone data table column names, - * otherwise false. - */ - boolean isValidZoneTableField(String field); - - /** - * @return a String[] of the column labels in the zone data table - */ - String[] getZoneDataTableColumnLabels(); - - /** - * @return an int value for the number of zones, i.e. rows in the zone data - * table - */ - int getNumberOfZones(); - - /** - * @return an int value for the number of subZones, i.e. number of - * walkTransit accessible segments defined in model for zones. - * Typical value might be 3, "no walk access", "short walk access", - * "long walk access". - */ - int getNumberOfSubZones(); - - /** - * @return a String[] for the subZone names, e.g. "no walk access", - * "short walk access", "long walk access". - */ - String[] getSubZoneNames(); - - /** - * @param taz - * is the taz index for the zonalWalkPctArray which is - * dimensioned to ZONES+1, assuming taz index values range from 1 - * to NUM_ZONES. - * @return a double[], dimensioned to NUM_SIBZONES, with the subzone - * proportions for the TAZ passed in - */ - double[] getZonalWalkPercentagesForTaz(int taz); - - /** - * @param taz - * is the taz index for the zone data table which is dimensioned - * to ZONES+1, assuming taz index values range from 1 to - * NUM_ZONES. - * @param fieldName - * is the column label in the zone data table. - * @return a float value from the zone data table at the specified row index - * and column label. - */ - float getZoneTableValue(int taz, String fieldName); - - int[] getZoneTableIntColumn(String fieldName); - - float[] getZoneTableFloatColumn(String fieldName); - - /** - * @param tableRowNumber - * is the zone table row number - * @return zone number for the table row. - */ - int getTazNumber(int tableRowNumber); - - /** - * @return area type from the zone data table for the zone index. - */ - int[] getZonalAreaType(); - - /** - * @return district from the zone data table for the zone index. - */ - int[] getZonalDistrict(); - - /** - * @return integer county value from the zone data table for the zone index. - */ - int[] getZonalCounty(); - - /** - * @return the parking rate array - */ - float[] getZonalParkRate(); - - /** - * @return the proportion of free parking array - */ - float[] getZonalPropFree(); - - /** - * @return the number of long parking spots array - */ - int[] getZonalParkLong(); - - /** - * @return the number of parking spots array - */ - int[] getZonalParkTot(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java deleted file mode 100644 index 5a1aed8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteDMU.java +++ /dev/null @@ -1,172 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - * @author jef
    - * Started: Jun 2019 - */ -public class TelecommuteDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TelecommuteDMU.class); - - protected HashMap methodIndexMap; - - private Household hh; - private Person person; - private IndexValues dmuIndex; - - public TelecommuteDMU() - { - dmuIndex = new IndexValues(); - } - - /** need to set hh and home taz before using **/ - public void setPersonObject(Person person) - { - this.hh = person.getHouseholdObject(); - this.person = person; - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug Telecommute UEC"); - } - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /* dmu @ functions */ - - public int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - public int getNumberOfAdults() - { - Person[] persons = hh.getPersons(); - int adults=0; - for(int i=1;i=18) - ++adults; - } - return adults; - } - - public int getHasKids_0_5() - { - Person[] persons = hh.getPersons(); - int hasKids_0_5=0; - for(int i=1;i=0) && persons[i].getAge()<=5) { - hasKids_0_5=1; - break; - } - } - return hasKids_0_5; - } - - - public int getHasKids_6_12() - { - Person[] persons = hh.getPersons(); - int hasKids_6_12=0; - for(int i=1;i=6) && persons[i].getAge()<=12) { - hasKids_6_12=1; - break; - } - } - return hasKids_6_12; - } - - public int getFemale() - { - return person.getPersonIsFemale(); - } - - public int getPersonType() - { - return person.getPersonTypeNumber(); - } - - public int getNumberOfAutos() - { - return hh.getAutosOwned(); - - } - - public int getOccupation() - { - return person.getPersPecasOccup(); - } - - public int getPaysToPark() - { - - int freeParkingChoice = person.getFreeParkingAvailableResult(); - if((freeParkingChoice==ParkingProvisionModel.FP_MODEL_PAY_ALT)|| - (freeParkingChoice==ParkingProvisionModel.FP_MODEL_REIMB_ALT)) - - return 1; - return 0; - } - - public float getWorkDistance() { - - return person.getWorkLocationDistance(); - } - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java deleted file mode 100644 index 5f47355..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TelecommuteModel.java +++ /dev/null @@ -1,144 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class TelecommuteModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("tc"); - - private static final String TC_CONTROL_FILE_TARGET = "te.uec.file"; - private static final String TC_DATA_SHEET_TARGET = "te.data.page"; - private static final String TC_MODEL_SHEET_TARGET = "te.model.page"; - - public static final short TC_MODEL_NO_TC_CHOICE = -1; - public static final short TC_MODEL_NO_TELECOMMUTE = 0; - public static final short TC_MODEL_1_DAY_WEEK_CHOICE = 1; - public static final short TC_MODEL_2_3_DAYS_WEEK_CHOICE = 2; - public static final short TC_MODEL_4P_DAYS_WEEK_CHOICE = 3; - public static final short WORK_AT_HOME_CHOICE = 9; - - - private MgraDataManager mgraManager; - - private ChoiceModelApplication tcModel; - private TelecommuteDMU tcDmuObject; - - public TelecommuteModel(HashMap propertyMap, CtrampDmuFactoryIf dmuFactory) - { - mgraManager = MgraDataManager.getInstance(propertyMap); - setupTelecommuteChoiceModelApplication(propertyMap, dmuFactory); - } - - private void setupTelecommuteChoiceModelApplication(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - logger.info("Setting up telecommute choice model."); - - // locate the telecommute UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tcUecFile = uecFileDirectory + propertyMap.get(TC_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TC_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TC_MODEL_SHEET_TARGET); - - // create the telecommute model DMU object. - tcDmuObject = dmuFactory.getTelecommuteDMU(); - - // create the telecommute model object - tcModel = new ChoiceModelApplication(tcUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) tcDmuObject); - - } - - public void applyModel(Household hhObject) - { - - Random hhRandom = hhObject.getHhRandom(); - - // person array is 1-based - Person[] person = hhObject.getPersons(); - for (int i = 1; i < person.length; i++) - { - - int workLoc = person[i].getWorkLocation(); - if (workLoc == ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR) - { - - person[i].setTelecommuteChoice(WORK_AT_HOME_CHOICE); - - } else if (workLoc > 0 ) - { - - double randomNumber = hhRandom.nextDouble(); - short chosen = (short) ((short) getTelecommuteChoice(person[i], randomNumber) - (short)1); - person[i].setTelecommuteChoice(chosen ); - - - } else - { - - person[i].setTelecommuteChoice(TC_MODEL_NO_TC_CHOICE); - - } - } - - hhObject.setFpRandomCount(hhObject.getHhRandomCount()); - } - - private int getTelecommuteChoice(Person personObj, double randomNumber) - { - - // get the corresponding household object - Household hhObj = personObj.getHouseholdObject(); - tcDmuObject.setPersonObject(personObj); - - // set the zone and dest attributes to the person's work location - tcDmuObject.setDmuIndexValues(hhObj.getHhId(), personObj.getWorkLocation(), - hhObj.getHhTaz(), personObj.getWorkLocation()); - - // compute utilities and choose telecommute alternative. - float logsum = (float) tcModel.computeUtilities(tcDmuObject, tcDmuObject.getDmuIndexValues()); - personObj.setTelecommuteLogsum(logsum); - - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - if (tcModel.getAvailabilityCount() > 0) - { - chosenAlt = tcModel.getChoiceResult(randomNumber); - } else - { - String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), - personObj.getPersonId()); - String errorMessage = String - .format("Exception caught for %s, no available telecommute alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - tcModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - // write choice model alternative info to log file - if (hhObj.getDebugChoiceModels()) - { - String decisionMaker = String.format("HHID=%d, PERSID=%d", hhObj.getHhId(), - personObj.getPersonId()); - tcModel.logAlternativesInfo("Telecommute Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d with rn %.8f", - "Telecommute Choice", decisionMaker, chosenAlt, randomNumber)); - tcModel.logUECResults(logger, decisionMaker); - } - - return chosenAlt; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java deleted file mode 100644 index b2fb242..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeCoefficientDistributions.java +++ /dev/null @@ -1,284 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedWriter; -import java.nio.charset.Charset; -import java.nio.charset.StandardCharsets; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ResourceBundle; -import java.util.Scanner; -import java.util.Set; - -import org.apache.log4j.Logger; - -import com.pb.common.math.MersenneTwister; -import com.pb.common.util.ResourceUtil; - -import umontreal.iro.lecuyer.probdist.LognormalDist; - -public class TimeCoefficientDistributions { - - private static Logger logger = Logger.getLogger(TimeCoefficientDistributions.class); - protected LognormalDist timeDistributionWork; - protected LognormalDist timeDistributionNonWork; - - - /** - * Constructor; doesn't do anything (call @createTimeDistributions method next) - */ - public TimeCoefficientDistributions(){ - - } - - /** - * Reads the propertyMap and finds values for properties - * timeDistributionMean.work, and - * timeDistributionStandardDistribution.work. - * Creates and stores umontreal.iro.lecuyer.probdist.LognormalDist - * workDistribution for work tours & trips. - * Reads the propertyMap and finds values for properties - * timeDistributionMean.nonwork, - * timeDistributionStandardDistribution.nonwork. - * Creates and stores a umontreal.iro.lecuyer.probdist.LognormalDist - * nonworkDistribution for non-work tours & trips. - * @param propertyMap - */ - - public void createTimeDistributions(HashMap propertyMap){ - - double meanWork = new Double(propertyMap.get("timeDistribution.mean.work" )); - double sdWork = new Double(propertyMap.get("timeDistribution.standardDeviation.work" )); - - double locationWork = calculateLocation(meanWork, sdWork); - double scaleWork = calculateScale(meanWork, sdWork); - - timeDistributionWork = new LognormalDist(locationWork, scaleWork); - - double meanNonWork = new Double(propertyMap.get("timeDistribution.mean.nonWork" )); - double sdNonWork = new Double(propertyMap.get("timeDistribution.standardDeviation.nonWork" )); - - double locationNonWork = calculateLocation(meanNonWork, sdNonWork); - double scaleNonWork = calculateScale(meanNonWork, sdNonWork); - - timeDistributionNonWork = new LognormalDist(locationNonWork, scaleNonWork); - - } - - /** - * Calculate the lognormal distribution location given - * the mean and standard deviation of the distribution - * according to the formula: - * - * location = ln(mean/sqrt(1 + variance/mean^2)) - * - * @param mean - * @param standardDeviation - * @return Location variable (u) - */ - public double calculateLocation(double mean, double standardDeviation){ - - double variance = standardDeviation * standardDeviation; - double meanSquared = mean * mean; - double denom = Math.sqrt(1.0 + (variance/meanSquared)); - double location = mean/denom; - if(location<=0){ - logger.error("Error: Trying to calculation location for mean "+mean - +" and standard deviation "+standardDeviation); - throw new RuntimeException(); - } - - return Math.log(location); - - } - - /** - * Calculate the lognormal distribution scale given - * the mean and standard deviation of the distribution - * according to the formula: - * - * scale = sqrt(ln(1 + variance/mean^2)); - * - * @param mean - * @param standardDeviation - * @return Scale variable (sigma) - */ - public double calculateScale(double mean, double standardDeviation){ - - double variance = standardDeviation * standardDeviation; - double meanSquared = mean * mean; - return Math.sqrt(Math.log(1 + variance/meanSquared)); - - - } - - /** - * Sample from the work distribution and return the factor to apply to work - * travel time coefficient. - * @param rnum A unit-distributed random number. - * @return The sampled time factor for work tours & trips. - */ - public double sampleFromWorkDistribution(double rnum){ - - return timeDistributionWork.inverseF(rnum); - } - - /** - * Sample from the non-work distribution and return the factor to apply - * to the non non-work travel time coefficient. - * - * @param rnum A unit-distributed random number. - * @return The sampled time factor for non-work tours and trips. - */ - public double sampleFromNonWorkDistribution(double rnum){ - - return timeDistributionNonWork.inverseF(rnum); - } - - /** - * Get the time distribution for work. - * - * @return The lognormal distribution for work. - */ - public LognormalDist getTimeDistributionWork() { - return timeDistributionWork; - } - - /** - * Get the time distribution for non-work. - * - * @return The lognormal distribution for non-work. - */ - public LognormalDist getTimeDistributionNonWork() { - return timeDistributionNonWork; - } - - - /** - * This method reads the input person file, samples from the lognormal - * time distributions for work and for non-work tours and trips, and - * appends the two fields for each person on the person file, over-writing the - * input person file with the results. If the fields already exist, nothing is - * done. The fields added to the person file are: - * - * timeFactorWork - * timeFactorNonWork - * - * @param propertyMap A property map with the following properties: - * Project.Directory: the path to directory to read the person file from. - * PopulationSynthesizer.InputToCTRAMP.PersonFile: the input person file - * timeDistribution.randomSeed: a random seed for sampling from the distributions for each person - */ - public void appendTimeDistributionsOnPersonFile(HashMap propertyMap){ - - logger.info("Appending time factors to person file"); - String directory = propertyMap.get("Project.Directory"); - String personFile = directory + propertyMap.get("PopulationSynthesizer.InputToCTRAMP.PersonFile"); - - long seed = new Long(propertyMap.get("timeDistribution.randomSeed")); - MersenneTwister random = new MersenneTwister(seed); - - Charset ENCODING = StandardCharsets.UTF_8; - - logger.info(""); - logger.info("Reading person file "+personFile); - Path path = Paths.get(personFile); - ArrayList personData = new ArrayList(5000000); - String header = null; - - - int persons = 0; - - - try (Scanner scanner = new Scanner(path, ENCODING.name())){ - - header = scanner.nextLine(); - - // does first row of person file contain field names for time factors? - if(header.contains("timeFactorWork")){ - - logger.info("File "+personFile+ " contains time factor fields already"); - scanner.close(); - return; - - }else{ - while (scanner.hasNextLine()){ - //add the row to the person data array list - personData.add(scanner.nextLine()); - ++persons; - - if(persons % 100000 == 0) - logger.info("Reading person file line "+persons); - } - } - scanner.close(); - }catch(Exception e){ - logger.fatal("Error while reading "+personFile); - throw new RuntimeException(); - } - - logger.info("Appending time factors to person file "+personFile); - header = header +",timeFactorWork,timeFactorNonWork"; - - try (BufferedWriter writer = Files.newBufferedWriter(path, ENCODING)){ - - //write the header with the additional fields - writer.write(header); - writer.newLine(); - - //write each person line, sampling from the work and non-work distributions for each and - //appending the results onto the initial data - for(int person = 0; person pMap; - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else - { - rb = ResourceBundle.getBundle(args[0]); - pMap = ResourceUtil.getResourceBundleAsHashMap(args[0]); - } - - TimeCoefficientDistributions timeDistributions = new TimeCoefficientDistributions(); - timeDistributions.createTimeDistributions(pMap); - timeDistributions.appendTimeDistributionsOnPersonFile(pMap); - - logger.info("All done!"); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java deleted file mode 100644 index 65f09c5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TimeDMU.java +++ /dev/null @@ -1,105 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class TimeDMU - implements Serializable, VariableTable -{ - - IndexValues dmuIndex = null; - - // switches used in the Individual Mandatory Tour Frequency Model - int imtfWorkSwitch, imtfSchoolSwitch; - - public TimeDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz, boolean debugUec) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debugUec) - { - dmuIndex.setDebug(true); - // dmuIndex.setDebugLabel ( "Debug IMTF Time UEC" ); - dmuIndex.setDebugLabel("Debug AO Time UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - // /** - // * Used in the Individual Mandatory Tour Frequency model; set to true - // * when the model is applied for a worker (to get round trip time to work, - // * which uses peak skims) - // * @param workOn - // */ - // public void setImtfWorkSwitch(int workOn){ - // imtfWorkSwitch = workOn; - // } - // - // /** - // * Used in the Individual Mandatory Tour Frequency model; set to true - // * when the model is applied for a student (to get round trip time to - // school, - // * which uses peak skims in the o/d direction and off-peak skims in the - // d/o - // * direction) - // * @param schoolOn - // */ - // public void setImtfSchoolSwitch(int schoolOn){ - // imtfSchoolSwitch = schoolOn; - // } - // - // public int getImtfWorkSwitch(){ - // return this.imtfWorkSwitch; - // } - // - // public int getImtfSchoolSwitch(){ - // return this.imtfSchoolSwitch; - // } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public int getIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java deleted file mode 100644 index 0646c5c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/Tour.java +++ /dev/null @@ -1,875 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.ArrayList; - -import org.apache.log4j.Logger; - -public class Tour - implements Serializable -{ - - private Person perObj; - private Household hhObj; - - private String tourCategory; - private String tourPurpose; - private String subtourPurpose; - - // use this array to hold personNum (i.e. index values for Household.persons - // array) for persons in tour. - // for individual tour types, this array is null. - // for joint tours, there will be an entry for each participating person. - private int[] personNumArray; - - // alternative number chosen by the joint tour composition model ( 1=adults, - // 2=children, 3=mixed ). - private int jointTourComposition; - - private int tourId; - private int tourOrigMgra; - private int tourDestMgra; - private int tourOrigWalkSubzone; - private int tourDestWalkSubzone; - private int tourDepartPeriod; - private int tourArrivePeriod; - private int tourMode; - private int subtourFreqChoice; - private int tourParkMgra; - - private float timeOfDayLogsum; - private float tourModeLogsum; - private float subtourFreqLogsum; - private float tourDestinationLogsum; - private float stopFreqLogsum; - - - private int tourPrimaryPurposeIndex; - - private float[] tourModalProbabilities; - private float[] tourModalUtilities; - - private int stopFreqChoice; - private Stop[] outboundStops; - private Stop[] inboundStops; - - private ArrayList outboundStopDestinationLogsums; - private ArrayList inboundStopDestinationLogsums; - - //Dimension N-path by 3 - 0=btap, 1=atap, 2=skim set , 3=utility - private double[][] bestWtwTapPairsOut; - - - private double[][] bestWtwTapPairsIn; - private double[][] bestWtdTapPairsOut; - private double[][] bestWtdTapPairsIn; - private double[][] bestDtwTapPairsOut; - private double[][] bestDtwTapPairsIn; - - private int choosenTransitPathIn; - - private int choosenTransitPathOut; - - private boolean useOwnedAV; - - private double valueOfTime; - - private int escortTypeOutbound; - private int escortTypeInbound; - private int driverPnumOutbound; - private int driverPnumInbound; - - // this constructor used for mandatory tour creation - public Tour(Person perObj, int tourId, int primaryIndex) - { - hhObj = perObj.getHouseholdObject(); - this.perObj = perObj; - this.tourId = tourId; - tourCategory = ModelStructure.MANDATORY_CATEGORY; - tourPrimaryPurposeIndex = primaryIndex; - - outboundStopDestinationLogsums = new ArrayList(); - inboundStopDestinationLogsums = new ArrayList(); - } - - // this constructor used for joint tour creation - public Tour(Household hhObj, String tourPurpose, String category, int primaryIndex) - { - this.hhObj = hhObj; - this.tourPurpose = tourPurpose; - tourCategory = category; - tourPrimaryPurposeIndex = primaryIndex; - outboundStopDestinationLogsums = new ArrayList(); - inboundStopDestinationLogsums = new ArrayList(); - } - - // this constructor used for individual non-mandatory or at-work subtour - // creation - public Tour(int id, Household hhObj, Person persObj, String tourPurpose, String category, - int primaryIndex) - { - this.hhObj = hhObj; - this.perObj = persObj; - tourId = id; - this.tourPurpose = tourPurpose; - tourCategory = category; - tourPrimaryPurposeIndex = primaryIndex; - outboundStopDestinationLogsums = new ArrayList(); - inboundStopDestinationLogsums = new ArrayList(); - } - - public Person getPersonObject() - { - return perObj; - } - - public void setPersonObject(Person p) - { - perObj = p; - } - - public void setPersonNumArray(int[] personNums) - { - personNumArray = personNums; - } - - public int[] getPersonNumArray() - { - return personNumArray; - } - - public boolean getPersonInJointTour(Person person) - { - boolean inTour = false; - for (int num : personNumArray) - { - if (person.getPersonNum() == num) - { - inTour = true; - break; - } - } - return inTour; - } - - public void setJointTourComposition(int compositionAlternative) - { - jointTourComposition = compositionAlternative; - } - - public int getJointTourComposition() - { - return jointTourComposition; - } - - public void setTourPurpose(String name) - { - tourPurpose = name; - } - - public void setSubTourPurpose(String name) - { - subtourPurpose = name; - } - - public String getSubTourPurpose() - { - return subtourPurpose; - } - - public String getTourCategory() - { - return tourCategory; - } - - public String getTourPurpose() - { - return tourPurpose; - } - - public String getTourPrimaryPurpose() - { - int index = tourPurpose.indexOf('_'); - if (index < 0) return tourPurpose; - else return tourPurpose.substring(0, index); - } - - // public int getTourPurposeIndex() { - // return tourPurposeIndex; - // } - - public int getTourPrimaryPurposeIndex() - { - return tourPrimaryPurposeIndex; - } - - public int getTourModeChoice() - { - return tourMode; - } - - public void setTourId(int id) - { - tourId = id; - } - - public void setTourOrigMgra(int origMgra) - { - tourOrigMgra = origMgra; - } - - public void setTourDestMgra(int destMgra) - { - tourDestMgra = destMgra; - } - - public void setTourOrigWalkSubzone(int subzone) - { - tourOrigWalkSubzone = subzone; - } - - public void setTourDestWalkSubzone(int subzone) - { - tourDestWalkSubzone = subzone; - } - - public void setTourDepartPeriod(int departPeriod) - { - tourDepartPeriod = departPeriod; - } - - public void setTourArrivePeriod(int arrivePeriod) - { - tourArrivePeriod = arrivePeriod; - } - - public void setTourModeChoice(int modeIndex) - { - tourMode = modeIndex; - } - - public void setTourParkMgra(int parkMgra) - { - tourParkMgra = parkMgra; - } - - // methods DMU will use to get info from household object - - public int getTourOrigMgra() - { - return tourOrigMgra; - } - - public int getTourDestMgra() - { - return tourDestMgra; - } - - public int getTourOrigWalkSubzone() - { - return tourOrigWalkSubzone; - } - - public int getTourDestWalkSubzone() - { - return tourDestWalkSubzone; - } - - public int getTourDepartPeriod() - { - return tourDepartPeriod; - } - - public int getTourArrivePeriod() - { - return tourArrivePeriod; - } - - public int getTourParkMgra() - { - return tourParkMgra; - } - - public int getHhId() - { - return hhObj.getHhId(); - } - - public int getHhMgra() - { - return hhObj.getHhMgra(); - } - - public int getTourId() - { - return tourId; - } - - public int getWorkTourIndexFromSubtourId(int subtourIndex) - { - // when subtour was created, it's purpose index was set to 10*work - // purpose - // index + at-work subtour index - return subtourIndex / 10; - } - - public int getSubtourIndexFromSubtourId(int subtourIndex) - { - // when subtour was created, it's purpose index was set to 10*work - // purpose - // index + at-work subtour index - int workTourIndex = subtourIndex / 10; - return subtourIndex - 10 * workTourIndex; - } - - public void setSubtourFreqChoice(int choice) - { - subtourFreqChoice = choice; - } - - public int getSubtourFreqChoice() - { - return subtourFreqChoice; - } - - public void setStopFreqChoice(int chosenAlt) - { - stopFreqChoice = chosenAlt; - } - - public int getStopFreqChoice() - { - return stopFreqChoice; - } - - public void createOutboundStops(String[] stopOrigPurposes, String[] stopDestPurposes, - int[] stopPurposeIndex) - { - outboundStops = new Stop[stopOrigPurposes.length]; - for (int i = 0; i < stopOrigPurposes.length; i++) - outboundStops[i] = new Stop(this, stopOrigPurposes[i], stopDestPurposes[i], i, false, - stopPurposeIndex[i]); - } - - public void createInboundStops(String[] stopOrigPurposes, String[] stopDestPurposes, - int[] stopPurposeIndex) - { - // needs outbound stops to be created first to get id numbering correct - - inboundStops = new Stop[stopOrigPurposes.length]; - for (int i = 0; i < stopOrigPurposes.length; i++) - inboundStops[i] = new Stop(this, stopOrigPurposes[i], stopDestPurposes[i], i, true, - stopPurposeIndex[i]); - } - - /** - * Create a Stop object to represent a half-tour where no stops were - * generated. The id for the stop is set to -1 so that trips for half-tours - * without stops can be distinguished in the output trip files from turs - * that have stops. Trips for these tours come from stop objects with ids in - * the range 0,...,3. - * - * @param origPurp - * is "home" or "work" (for at-work subtours) if outbound, or the - * primary tour purpose if inbound - * @param destPurp - * is "home" or "work" (for at-work subtours) if inbound, or the - * primary tour purpose if outbound - * @param inbound - * is true if the half-tour is inbound, or false if outbound. - * @return the created Stop object. - */ - public Stop createStop(String origPurp, String destPurp, - boolean inbound, boolean subtour) - { - Stop stop = null; - int id = -1; - if (inbound) - { - inboundStops = new Stop[1]; - inboundStops[0] = new Stop(this, origPurp, destPurp, id, inbound, 0); - stop = inboundStops[0]; - } else - { - outboundStops = new Stop[1]; - outboundStops[0] = new Stop(this, origPurp, destPurp, id, inbound, 0); - stop = outboundStops[0]; - } - return stop; - } - - public int getNumOutboundStops() - { - if (outboundStops == null) return 0; - else return outboundStops.length; - } - - public int getNumInboundStops() - { - if (inboundStops == null) return 0; - else return inboundStops.length; - } - - public Stop[] getOutboundStops() - { - return outboundStops; - } - - public Stop[] getInboundStops() - { - return inboundStops; - } - - public void clearStopModelResults() - { - stopFreqChoice = 0; - outboundStops = null; - inboundStops = null; - } - - public String getTourWindow(String purposeAbbreviation) - { - String returnString = String.format(" %5s: |", purposeAbbreviation); - short[] windows = perObj.getTimeWindows(); - for (int i = 1; i < windows.length; i++) - { - String tempString = String.format("%s", - i >= tourDepartPeriod && i <= tourArrivePeriod ? purposeAbbreviation : " "); - if (tempString.length() == 2 || tempString.length() == 3) - tempString = " " + tempString; - returnString += String.format("%4s|", tempString); - } - return returnString; - } - - public int getEscortTypeOutbound() { - return escortTypeOutbound; - } - public void setEscortTypeOutbound(int escortType) { - this.escortTypeOutbound = escortType; - } - public int getEscortTypeInbound() { - return escortTypeInbound; - } - public void setEscortTypeInbound(int escortType) { - this.escortTypeInbound = escortType; - } - public int getDriverPnumOutbound() { - return driverPnumOutbound; - } - public void setDriverPnumOutbound(int driverPnum) { - this.driverPnumOutbound = driverPnum; - } - public int getDriverPnumInbound() { - return driverPnumInbound; - } - public void setDriverPnumInbound(int driverPnum) { - this.driverPnumInbound = driverPnum; - } - public void logTourObject(Logger logger, int totalChars) - { - - String personNumArrayString = "-"; - if (personNumArray != null) - { - personNumArrayString = "[ "; - personNumArrayString += String.format("%d", personNumArray[0]); - for (int i = 1; i < personNumArray.length; i++) - personNumArrayString += String.format(", %d", personNumArray[i]); - personNumArrayString += " ]"; - } - - Household.logHelper(logger, "tourId: ", tourId, totalChars); - Household.logHelper(logger, "tourCategory: ", tourCategory, totalChars); - Household.logHelper(logger, "tourPurpose: ", tourPurpose, totalChars); - Household.logHelper(logger, "tourPurposeIndex: ", tourPrimaryPurposeIndex, totalChars); - Household.logHelper(logger, "personNumArray: ", personNumArrayString, totalChars); - Household.logHelper(logger, "jointTourComposition: ", jointTourComposition, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", tourOrigMgra, totalChars); - Household.logHelper(logger, "tourDestMgra: ", tourDestMgra, totalChars); - Household.logHelper(logger, "tourOrigWalkSubzone: ", tourOrigWalkSubzone, totalChars); - Household.logHelper(logger, "tourDestWalkSubzone: ", tourDestWalkSubzone, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", tourDepartPeriod, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", tourArrivePeriod, totalChars); - Household.logHelper(logger, "tourMode: ", tourMode, totalChars); - Household.logHelper(logger, "escortTypeOutbound: ", escortTypeOutbound, totalChars); - Household.logHelper(logger, "driverPnumOutbound: ", driverPnumOutbound, totalChars); - Household.logHelper(logger, "escortTypeInbound: ", escortTypeInbound, totalChars); - Household.logHelper(logger, "driverPnumInbound: ", driverPnumInbound, totalChars); - Household.logHelper(logger, "stopFreqChoice: ", stopFreqChoice, totalChars); - - String tempString = String.format("outboundStops[%s]:", - outboundStops == null ? "" : String.valueOf(outboundStops.length)); - logger.info(tempString); - - tempString = String.format("inboundStops[%s]:", - inboundStops == null ? "" : String.valueOf(inboundStops.length)); - logger.info(tempString); - - if ( bestWtwTapPairsOut == null ) { - tempString = "bestWtwTapPairsOut: no tap pairs saved"; - } - else { - if ( bestWtwTapPairsOut[0] == null ) - tempString = "bestWtwTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestWtwTapPairsOut: " + 0 + "[" + bestWtwTapPairsOut[0][0] + "," + bestWtwTapPairsOut[0][1] + "," + bestWtwTapPairsOut[0][2] + "]"; - for (int i=1; i < bestWtwTapPairsOut.length; i++) - if ( bestWtwTapPairsOut[i] == null ) - tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestWtwTapPairsOut[i][0] + "," + bestWtwTapPairsOut[i][1] + "," + bestWtwTapPairsOut[0][2] + "]"; - - tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; - } - logger.info(tempString); - - if ( bestWtwTapPairsIn == null ) { - tempString = "bestWtwTapPairsIn: no tap pairs saved"; - } - else { - if ( bestWtwTapPairsIn[0] == null ) - tempString = "bestWtwTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestWtwTapPairsIn: " + 0 + "[" + bestWtwTapPairsIn[0][0] + "," + bestWtwTapPairsIn[0][1] + "," + bestWtwTapPairsIn[0][2] + "]"; - for (int i=1; i < bestWtwTapPairsIn.length; i++) - if ( bestWtwTapPairsIn[i] == null ) - tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestWtwTapPairsIn[i][0] + "," + bestWtwTapPairsIn[i][1] + "," + bestWtwTapPairsIn[0][2] + "]"; - - tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; - } - logger.info(tempString); - - if ( bestWtdTapPairsOut == null ) { - tempString = "bestWtdTapPairsOut: no tap pairs saved"; - } - else { - if ( bestWtdTapPairsOut[0] == null ) - tempString = "bestWtdTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestWtdTapPairsOut: " + 0 + "[" + bestWtdTapPairsOut[0][0] + "," + bestWtdTapPairsOut[0][1] + "," + bestWtdTapPairsOut[0][2] + "]"; - for (int i=1; i < bestWtdTapPairsOut.length; i++) - if ( bestWtdTapPairsOut[i] == null ) - tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestWtdTapPairsOut[i][0] + "," + bestWtdTapPairsOut[i][1] + "," + bestWtdTapPairsOut[0][2] + "]"; - - tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; - } - logger.info(tempString); - - if ( bestWtdTapPairsIn == null ) { - tempString = "bestWtdTapPairsIn: no tap pairs saved"; - } - else { - if ( bestWtdTapPairsIn[0] == null ) - tempString = "bestWtdTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestWtdTapPairsIn: " + 0 + "[" + bestWtdTapPairsIn[0][0] + "," + bestWtdTapPairsIn[0][1] + "," + bestWtdTapPairsIn[0][2] + "]"; - for (int i=1; i < bestWtdTapPairsIn.length; i++) - if ( bestWtdTapPairsIn[i] == null ) - tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestWtdTapPairsIn[i][0] + "," + bestWtdTapPairsIn[i][1] + "," + bestWtdTapPairsIn[0][2] + "]"; - - tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; - } - logger.info(tempString); - - if ( bestDtwTapPairsOut == null ) { - tempString = "bestDtwTapPairsOut: no tap pairs saved"; - } - else { - if ( bestDtwTapPairsOut[0] == null ) - tempString = "bestDtwTapPairsOut: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestDtwTapPairsOut: " + 0 + "[" + bestDtwTapPairsOut[0][0] + "," + bestDtwTapPairsOut[0][1] + "," + bestDtwTapPairsOut[0][2] + "]"; - for (int i=1; i < bestDtwTapPairsOut.length; i++) - if ( bestDtwTapPairsOut[i] == null ) - tempString += ", " + i + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestDtwTapPairsOut[i][0] + "," + bestDtwTapPairsOut[i][1] + "," + bestDtwTapPairsOut[0][2] + "]"; - - tempString += ", choosenTransitPathOut: " + choosenTransitPathOut; - } - logger.info(tempString); - - if ( bestDtwTapPairsIn == null ) { - tempString = "bestDtwTapPairsIn: no tap pairs saved"; - } - else { - if ( bestDtwTapPairsIn[0] == null ) - tempString = "bestDtwTapPairsIn: " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString = "bestDtwTapPairsIn: " + 0 + "[" + bestDtwTapPairsIn[0][0] + "," + bestDtwTapPairsIn[0][1] + "," + bestDtwTapPairsIn[0][2] + "]"; - for (int i=1; i < bestDtwTapPairsIn.length; i++) - if ( bestDtwTapPairsIn[i] == null ) - tempString += ", " + 0 + "[" + "none" + "," + "none" + "," + "none" + "]"; - else - tempString += ", " + i + "[" + bestDtwTapPairsIn[i][0] + "," + bestDtwTapPairsIn[i][1] + "," + bestDtwTapPairsIn[0][2] + "]"; - - tempString += ", choosenTransitPathIn: " + choosenTransitPathIn; - } - logger.info(tempString); - - } - - public void logEntireTourObject(Logger logger) - { - - int totalChars = 60; - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - String personNumArrayString = "-"; - if (personNumArray != null) - { - personNumArrayString = "[ "; - personNumArrayString += String.format("%d", personNumArray[0]); - for (int i = 1; i < personNumArray.length; i++) - personNumArrayString += String.format(", %d", personNumArray[i]); - personNumArrayString += " ]"; - } - - Household.logHelper(logger, "tourId: ", tourId, totalChars); - Household.logHelper(logger, "tourCategory: ", tourCategory, totalChars); - Household.logHelper(logger, "tourPurpose: ", tourPurpose, totalChars); - Household.logHelper(logger, "tourPurposeIndex: ", tourPrimaryPurposeIndex, totalChars); - Household.logHelper(logger, "personNumArray: ", personNumArrayString, totalChars); - Household.logHelper(logger, "jointTourComposition: ", jointTourComposition, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", tourOrigMgra, totalChars); - Household.logHelper(logger, "tourDestMgra: ", tourDestMgra, totalChars); - Household.logHelper(logger, "tourOrigWalkSubzone: ", tourOrigWalkSubzone, totalChars); - Household.logHelper(logger, "tourDestWalkSubzone: ", tourDestWalkSubzone, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", tourDepartPeriod, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", tourArrivePeriod, totalChars); - Household.logHelper(logger, "driverPnumOutbound: ", driverPnumOutbound, totalChars); - Household.logHelper(logger, "escortTypeInbound: ", escortTypeInbound, totalChars); - Household.logHelper(logger, "driverPnumInbound: ", driverPnumInbound, totalChars); - Household.logHelper(logger, "tourMode: ", tourMode, totalChars); - Household.logHelper(logger, "stopFreqChoice: ", stopFreqChoice, totalChars); - - if (outboundStops != null) - { - logger.info("Outbound Stops:"); - if (outboundStops.length > 0) - { - for (int i = 0; i < outboundStops.length; i++) - outboundStops[i].logStopObject(logger, totalChars); - } else - { - logger.info(" No outbound stops"); - } - } else - { - logger.info(" No outbound stops"); - } - - if (inboundStops != null) - { - logger.info("Inbound Stops:"); - if (inboundStops.length > 0) - { - for (int i = 0; i < inboundStops.length; i++) - inboundStops[i].logStopObject(logger, totalChars); - } else - { - logger.info(" No inbound stops"); - } - } else - { - logger.info(" No inbound stops"); - } - - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public void setTourModalUtilities(float[] utils) - { - tourModalUtilities = utils; - } - - public float[] getTourModalUtilities() - { - return tourModalUtilities; - } - - public void setTourModalProbabilities(float[] probs) - { - tourModalProbabilities = probs; - } - - public float[] getTourModalProbabilities() - { - return tourModalProbabilities; - } - - public void setBestWtwTapPairsOut(double[][] tapPairArray) - { - bestWtwTapPairsOut = tapPairArray; - } - - public void setBestWtwTapPairsIn(double[][] tapPairArray) - { - bestWtwTapPairsIn = tapPairArray; - } - - public void setBestWtdTapPairsOut(double[][] tapPairArray) - { - bestWtdTapPairsOut = tapPairArray; - } - - public void setBestWtdTapPairsIn(double[][] tapPairArray) - { - bestWtdTapPairsIn = tapPairArray; - } - - public void setBestDtwTapPairsOut(double[][] tapPairArray) - { - bestDtwTapPairsOut = tapPairArray; - } - - public void setBestDtwTapPairsIn(double[][] tapPairArray) - { - bestDtwTapPairsIn = tapPairArray; - } - - public void setChoosenTransitPathIn( int path ) - { - choosenTransitPathIn = path; - } - public void setChoosenTransitPathOut( int path ) - { - choosenTransitPathOut = path; - } - public double[][] getBestWtwTapPairsOut() - { - return bestWtwTapPairsOut; - } - - public double[][] getBestWtwTapPairsIn() - { - return bestWtwTapPairsIn; - } - - public double[][] getBestWtdTapPairsOut() - { - return bestWtdTapPairsOut; - } - - public double[][] getBestWtdTapPairsIn() - { - return bestWtdTapPairsIn; - } - - public double[][] getBestDtwTapPairsOut() - { - return bestDtwTapPairsOut; - } - - public double[][] getBestDtwTapPairsIn() - { - return bestDtwTapPairsIn; - } - - public double getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(double valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public float getTimeOfDayLogsum() { - return timeOfDayLogsum; - } - - public void setTimeOfDayLogsum(float timeOfDayLogsum) { - this.timeOfDayLogsum = timeOfDayLogsum; - } - - public float getTourModeLogsum() { - return tourModeLogsum; - } - - public void setTourModeLogsum(float tourModeLogsum) { - this.tourModeLogsum = tourModeLogsum; - } - - public float getSubtourFreqLogsum() { - return subtourFreqLogsum; - } - - public void setSubtourFreqLogsum(float subtourFreqLogsum) { - this.subtourFreqLogsum = subtourFreqLogsum; - } - - public float getTourDestinationLogsum() { - return tourDestinationLogsum; - } - - public void setTourDestinationLogsum(float tourDestinationLogsum) { - this.tourDestinationLogsum = tourDestinationLogsum; - } - - public float getStopFreqLogsum() { - return stopFreqLogsum; - } - - public void setStopFreqLogsum(float stopFreqLogsum) { - this.stopFreqLogsum = stopFreqLogsum; - } - - public ArrayList getOutboundStopDestinationLogsums(){ - return outboundStopDestinationLogsums; - } - public ArrayList getInboundStopDestinationLogsums(){ - return inboundStopDestinationLogsums; - } - - public void addOutboundStopDestinationLogsum(float logsum){ - outboundStopDestinationLogsums.add(logsum); - } - - public void addInboundStopDestinationLogsum(float logsum){ - inboundStopDestinationLogsums.add(logsum); - } - - public boolean getUseOwnedAV() { - return useOwnedAV; - } - - public void setUseOwnedAV(boolean useOwnedAV) { - this.useOwnedAV = useOwnedAV; - } - - /** - * Iterate through persons on tour and return non-work time factor - * for oldest person. If the person array is null then return 1.0. - * - * @return Time factor for oldest person on joint tour. - */ - public double getJointTourTimeFactor() { - int[] personNumArray = getPersonNumArray(); - int oldestAge = -999; - Person oldestPerson = null; - for (int num : personNumArray){ - Person p = hhObj.getPerson(num); - if(p.getAge() > oldestAge){ - oldestPerson = p; - oldestAge = p.getAge(); - } - } - if(oldestPerson != null) - return oldestPerson.getTimeFactorNonWork(); - - return 1.0; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java deleted file mode 100644 index 6e55920..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourDepartureTimeAndDurationDMU.java +++ /dev/null @@ -1,864 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class TourDepartureTimeAndDurationDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TourDepartureTimeAndDurationDMU.class); - - protected HashMap methodIndexMap; - - protected IndexValues dmuIndex; - - protected Person person; - protected Household household; - protected Tour tour; - - protected double destEmpDen; - protected int subsequentTourIsWork; - protected int subsequentTourIsSchool; - - protected double[] modeChoiceLogsums; - - private int[] altStarts; - private int[] altEnds; - - protected int originAreaType, destinationAreaType; - - protected int tourNumber; - - protected int firstTour; - protected int subsequentTour; - protected int endOfPreviousScheduledTour; - - protected ModelStructure modelStructure; - - public TourDepartureTimeAndDurationDMU(ModelStructure modelStructure) - { - this.modelStructure = modelStructure; - dmuIndex = new IndexValues(); - } - - public void setPerson(Person passedInPerson) - { - person = passedInPerson; - } - - public void setHousehold(Household passedInHousehold) - { - household = passedInHousehold; - - // set the origin and zone indices - dmuIndex.setZoneIndex(household.getHhMgra()); - dmuIndex.setHHIndex(household.getHhId()); - - // set the debug flag that can be used in the UEC - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (household.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug DepartTime UEC"); - } - - } - - public void setTour(Tour passedInTour) - { - tour = passedInTour; - } - - public void setOriginZone(int zone) - { - dmuIndex.setOriginZone(zone); - } - - public void setDestinationZone(int zone) - { - dmuIndex.setDestZone(zone); - } - - public void setOriginAreaType(int areaType) - { - originAreaType = areaType; - } - - public void setDestinationAreaType(int areaType) - { - destinationAreaType = areaType; - } - - public void setDestEmpDen(double arg) - { - destEmpDen = arg; - } - - public void setFirstTour(int trueOrFalse) - { - firstTour = trueOrFalse; - } - - public void setSubsequentTour(int trueOrFalse) - { - subsequentTour = trueOrFalse; - } - - public void setSubsequentTourIsWork(int trueOrFalse) - { - subsequentTourIsWork = trueOrFalse; - } - - public void setSubsequentTourIsSchool(int trueOrFalse) - { - subsequentTourIsSchool = trueOrFalse; - } - - /** - * Set the sequence number of this tour among all scheduled - * - * @param tourNum - */ - public void setTourNumber(int tourNum) - { - tourNumber = tourNum; - } - - public void setEndOfPreviousScheduledTour(int endHr) - { - endOfPreviousScheduledTour = endHr; - } - - public void setModeChoiceLogsums(double[] logsums) - { - modeChoiceLogsums = logsums; - } - - public void setTodAlts(int[] altStarts, int[] altEnds) - { - this.altStarts = altStarts; - this.altEnds = altEnds; - } - - public IndexValues getIndexValues() - { - return (dmuIndex); - } - - public Household getDmuHouseholdObject() - { - return household; - } - - public int getOriginZone() - { - return (dmuIndex.getOriginZone()); - } - - public int getDestinationZone() - { - return (dmuIndex.getDestZone()); - } - - public int getOriginAreaType() - { - return (originAreaType); - } - - public int getDestinationAreaType() - { - return (destinationAreaType); - } - - public int getPreDrivingAgeChild() - { - return (person.getPersonIsStudentNonDriving() == 1 || person.getPersonIsPreschoolChild() == 1) ? 1 - : 0; - } - - public int getPersonAge() - { - return person.getAge(); - } - - public int getPersonIsFemale() - { - return person.getGender() == 2 ? 1 : 0; - } - - public int getHouseholdSize() - { - return household.getHhSize(); - } - - public int getNumPreschoolChildrenInHh() - { - return household.getNumPreschool(); - } - - public int getNumChildrenUnder16InHh() - { - return household.getNumChildrenUnder16(); - } - - public int getNumNonWorkingAdultsInHh() - { - return household.getNumberOfNonWorkingAdults(); - } - - public int getFullTimeWorker() - { - return (this.person.getPersonTypeIsFullTimeWorker()); - } - - public int getPartTimeWorker() - { - return (this.person.getPersonTypeIsPartTimeWorker()); - } - - public int getUniversityStudent() - { - return (this.person.getPersonIsUniversityStudent()); - } - - public int getStudentDrivingAge() - { - return (this.person.getPersonIsStudentDriving()); - } - - public int getStudentNonDrivingAge() - { - return (this.person.getPersonIsStudentNonDriving()); - } - - public int getNonWorker() - { - return (this.person.getPersonIsNonWorkingAdultUnder65()); - } - - public int getRetired() - { - return (this.person.getPersonIsNonWorkingAdultOver65()); - } - - public int getAllAdultsFullTimeWorkers() - { - Person[] p = household.getPersons(); - boolean allAdultsAreFullTimeWorkers = true; - for (int i = 1; i < p.length; i++) - { - if (p[i].getPersonIsAdult() == 1 && p[i].getPersonIsFullTimeWorker() == 0) - { - allAdultsAreFullTimeWorkers = false; - break; - } - } - - if (allAdultsAreFullTimeWorkers) return 1; - else return 0; - } - - public int getSubtourPurposeIsEatOut() - { - if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_EAT_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getSubtourPurposeIsBusiness() - { - if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_BUSINESS_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getSubtourPurposeIsOther() - { - if (tour.getSubTourPurpose().equalsIgnoreCase(modelStructure.AT_WORK_MAINT_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourPurposeIsShopping() - { - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SHOPPING_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourPurposeIsEatOut() - { - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.EAT_OUT_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourPurposeIsMaint() - { - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.OTH_MAINT_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourPurposeIsVisit() - { - if (tour.getTourPurpose().equalsIgnoreCase(modelStructure.SOCIAL_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourPurposeIsDiscr() - { - if (tour.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getNumIndivShopTours() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME)) - count++; - - return count; - } - - public int getNumIndivMaintTours() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) - count++; - - return count; - } - - public int getNumIndivVisitTours() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME)) - count++; - - return count; - } - - public int getNumIndivDiscrTours() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (t.getTourPurpose().equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) - count++; - - return count; - } - - /* - * if ( tour.getTourCategory() == ModelStructure.AT_WORK_CATEGORY ) { return - * tour.getTourPurposeIndex(); } else { return 0; } } - */ - - public int getAdultsInTour() - { - - int count = 0; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person[] persons = household.getPersons(); - - int[] personNums = tour.getPersonNumArray(); - for (int i = 0; i < personNums.length; i++) - { - int p = personNums[i]; - if (persons[p].getPersonIsAdult() == 1) count++; - } - } else if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) - { - if (person.getPersonIsAdult() == 1) count = 1; - } - - return count; - } - - public int getJointTourPartySize() - { - int count = 0; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - count = tour.getPersonNumArray().length; - - return count; - } - - public int getKidsOnJointTour() - { - - int count = 0; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person[] persons = household.getPersons(); - - int[] personNums = tour.getPersonNumArray(); - for (int i = 0; i < personNums.length; i++) - { - int p = personNums[i]; - if ((persons[p].getPersonIsPreschoolChild() - + persons[p].getPersonIsStudentNonDriving() + persons[p] - .getPersonIsStudentDriving()) > 0) count++; - } - } - - return count > 0 ? 1 : 0; - - } - - // return 1 if at least one preschool or pre-driving child is in joint tour, - // otherwise 0. - public int getPreschoolPredrivingInTour() - { - - int count = 0; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person[] persons = household.getPersons(); - int[] personNums = tour.getPersonNumArray(); - for (int i = 0; i < personNums.length; i++) - { - int p = personNums[i]; - if (persons[p].getPersonIsPreschoolChild() == 1 - || persons[p].getPersonIsStudentNonDriving() == 1) return 1; - } - } else if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) - { - if (person.getPersonIsPreschoolChild() == 1 - || person.getPersonIsStudentNonDriving() == 1) count = 1; - } - - return count; - - } - - // return 1 if the person is preschool - public int getPreschool() - { - return person.getPersonIsPreschoolChild() == 1 ? 1 : 0; - - } - - // return 1 if at least one university student is in joint tour, otherwise - // 0. - public int getUnivInTour() - { - - int count = 0; - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person[] persons = household.getPersons(); - int[] personNums = tour.getPersonNumArray(); - for (int i = 0; i < personNums.length; i++) - { - int p = personNums[i]; - if (persons[p].getPersonIsUniversityStudent() == 1) return 1; - } - } else if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.INDIVIDUAL_NON_MANDATORY_CATEGORY)) - { - if (person.getPersonIsUniversityStudent() == 1) count = 1; - } - - return count; - - } - - // return 1 if all adults in joint tour are fulltime workers, 0 otherwise; - public int getAllWorkFull() - { - - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - int adultCount = 0; - int ftWorkerAdultCount = 0; - - Person[] persons = household.getPersons(); - int[] personNums = tour.getPersonNumArray(); - for (int i = 0; i < personNums.length; i++) - { - int p = personNums[i]; - if (persons[p].getPersonIsAdult() == 1) - { - adultCount++; - if (persons[p].getPersonIsFullTimeWorker() == 1) ftWorkerAdultCount++; - } - } - - if (adultCount > 0 && adultCount == ftWorkerAdultCount) return 1; - else return 0; - } - - return 0; - - } - - public int getPartyComp() - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - return tour.getJointTourComposition(); - } else - { - return 0; - } - } - - /** - * @return number of individual non-mandatory tours, including escort, for - * the person - */ - public int getPersonNonMandatoryTotalWithEscort() - { - return person.getListOfIndividualNonMandatoryTours().size(); - } - - /** - * @return number of individual non-mandatory tours, excluding escort, for - * the person - */ - public int getPersonNonMandatoryTotalNoEscort() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (!t.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) count++; - return count; - } - - /** - * @return number of individual non-mandatory discretionary tours for the - * person - */ - public int getPersonDiscrToursTotal() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - { - if (t.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) - || t.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) - || t.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) count++; - } - return count; - } - - /** - * @return number of individual non-mandatory tours, excluding escort, for - * the person - */ - public int getPersonEscortTotal() - { - int count = 0; - for (Tour t : person.getListOfIndividualNonMandatoryTours()) - if (t.getTourPurpose().startsWith("escort")) count++; - return count; - } - - public int getHhJointTotal() - { - Tour[] jt = household.getJointTourArray(); - if (jt == null) return 0; - else return jt.length; - } - - public int getPersonMandatoryTotal() - { - return person.getListOfWorkTours().size() + person.getListOfSchoolTours().size(); - } - - public int getPersonJointTotal() - { - Tour[] jtArray = household.getJointTourArray(); - if (jtArray == null) - { - return 0; - } else - { - int numJtParticipations = 0; - for (Tour jt : jtArray) - { - int[] personJtIndices = jt.getPersonNumArray(); - for (int pNum : personJtIndices) - { - if (pNum == person.getPersonNum()) - { - numJtParticipations++; - break; - } - } - } - return numJtParticipations; - } - } - - public int getPersonJointAndIndivDiscrToursTotal() - { - - int totDiscr = getPersonDiscrToursTotal(); - - Tour[] jtArray = household.getJointTourArray(); - if (jtArray == null) - { - return totDiscr; - } else - { - // count number of joint discretionary tours person participates in - int numJtParticipations = 0; - for (Tour jt : jtArray) - { - if (jt.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) - || jt.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) - || jt.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) - { - int[] personJtIndices = jt.getPersonNumArray(); - for (int pNum : personJtIndices) - { - if (pNum == person.getPersonNum()) - { - numJtParticipations++; - break; - } - } - } - } - return numJtParticipations + totDiscr; - } - } - - public int getFirstTour() - { - return firstTour; - } - - public int getSubsequentTour() - { - return subsequentTour; - } - - public int getWorkAndSchoolToursByWorker() - { - int returnValue = 0; - if (person.getPersonIsWorker() == 1) - { - if (person.getImtfChoice() == HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_WORK_AND_SCHOOL) - returnValue = 1; - } - return returnValue; - } - - public int getWorkAndSchoolToursByStudent() - { - int returnValue = 0; - if (person.getPersonIsStudent() == 1) - { - if (person.getImtfChoice() == HouseholdIndividualMandatoryTourFrequencyModel.CHOICE_WORK_AND_SCHOOL) - returnValue = 1; - } - return returnValue; - } - - public double getModeChoiceLogsumAlt(int alt) - { - - int startPeriod = altStarts[alt - 1]; - int endPeriod = altEnds[alt - 1]; - - int index = modelStructure.getSkimPeriodCombinationIndex(startPeriod, endPeriod); - - return modeChoiceLogsums[index]; - - } - - public int getPrevTourEndsThisDeparturePeriodAlt(int alt) - { - - // get the departure period for the current alternative - int thisTourStartsPeriod = altStarts[alt - 1]; - - if (person.isPreviousArrival(thisTourStartsPeriod)) return 1; - else return 0; - - } - - public int getPrevTourBeginsThisArrivalPeriodAlt(int alt) - { - - // get the arrival period for the current alternative - int thisTourEndsPeriod = altStarts[alt - 1]; - - if (person.isPreviousDeparture(thisTourEndsPeriod)) return 1; - else return 0; - - } - - public int getAdjWindowBeforeThisPeriodAlt(int alt) - { - - int thisTourStartsPeriod = altStarts[alt - 1]; - - int numAdjacentPeriodsAvailable = 0; - for (int i = thisTourStartsPeriod - 1; i >= 0; i--) - { - if (person.isPeriodAvailable(i)) numAdjacentPeriodsAvailable++; - else break; - } - - return numAdjacentPeriodsAvailable; - - } - - public int getAdjWindowAfterThisPeriodAlt(int alt) - { - - int thisTourEndsPeriod = altEnds[alt - 1]; - - int numAdjacentPeriodsAvailable = 0; - for (int i = thisTourEndsPeriod + 1; i < modelStructure.getNumberOfTimePeriods(); i++) - { - if (person.isPeriodAvailable(i)) numAdjacentPeriodsAvailable++; - else break; - } - - return numAdjacentPeriodsAvailable; - - } - - public int getRemainingPeriodsAvailableAlt(int alt) - { - - int periodsAvail = person.getAvailableWindow(); - - int start = altStarts[alt - 1]; - int end = altEnds[alt - 1]; - - // determine the availabilty of each period after the alternative time - // window - // is hypothetically scheduled - // if start == end, the availability won't change, so no need to - // compute. - if (start != end) - { - - // the start and end periods will always be available after - // scheduling, so - // don't need to check them. - // the periods between start/end must be 0 or the alternative could - // not - // have been available, - // so count them all as unavailable after scheduling this window. - periodsAvail -= (end - start - 1); - - } - - return periodsAvail; - - } - - public float getRemainingInmToursToAvailablePeriodsRatioAlt(int alt) - { - int periodsAvail = getRemainingPeriodsAvailableAlt(alt); - if (periodsAvail > 0) - { - float ratio = (float) (person.getListOfIndividualNonMandatoryTours().size() - tourNumber) - / periodsAvail; - return ratio; - } else return -999; - } - - public int getMaximumAvailableTimeWindow() - { - return person.getMaximumContinuousAvailableWindow(); - } - - public int getMaxJointTimeWindow() - { - return household.getMaxJointTimeWindow(tour); - } - - /** - * get the number of tours left to be scheduled, including the current tour - * - * @return number of tours left to be scheduled, including the current tour - */ - public int getToursLeftToSchedule() - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Tour[] jt = household.getJointTourArray(); - return jt.length - tourNumber + 1; - } else return person.getListOfIndividualNonMandatoryTours().size() - tourNumber + 1; - } - - public int getEndOfPreviousTour() - { - return endOfPreviousScheduledTour; - } - - public int getTourCategoryIsJoint() - { - return tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY) ? 1 - : 0; - } - - public float getOpSovTimeOd() - { - return 1; - } - - public float getOpSovTimeDo() - { - return 1; - } - - public int getDestMgraInCbd() - { - return 0; - } - - public int getOrigMgraInRural() - { - return 0; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java deleted file mode 100644 index 7e83278..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceDMU.java +++ /dev/null @@ -1,523 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class TourModeChoiceDMU implements Serializable, - VariableTable { - - protected transient Logger logger = Logger.getLogger(TourModeChoiceDMU.class); - - public static final int WTW = McLogsumsCalculator.WTW; - public static final int WTD = McLogsumsCalculator.WTD; - public static final int DTW = McLogsumsCalculator.DTW; - protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - protected float origTaxiWaitTime; - protected float destTaxiWaitTime; - protected float origSingleTNCWaitTime; - protected float destSingleTNCWaitTime; - protected float origSharedTNCWaitTime; - protected float destSharedTNCWaitTime; - - - protected Household hh; - protected Tour tour; - protected Tour workTour; - protected Person person; - - protected ModelStructure modelStructure; - - protected double origDuDen; - protected double origEmpDen; - protected double origTotInt; - protected double destDuDen; - protected double destEmpDen; - protected double destTotInt; - - protected double lsWgtAvgCostM; - protected double lsWgtAvgCostD; - protected double lsWgtAvgCostH; - protected double reimburseProportion; - protected int parkingArea; - - protected float pTazTerminalTime; - protected float aTazTerminalTime; - - protected double nmWalkTimeOut; - protected double nmWalkTimeIn; - protected double nmBikeTimeOut; - protected double nmBikeTimeIn; - - protected int originMgra; - protected int destMgra; - - protected double ivtCoeff; - protected double costCoeff; - - protected double[][] transitLogSum; - - - public TourModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) { - this.modelStructure = modelStructure; - dmuIndex = new IndexValues(); - - //accEgr by in/outbound - transitLogSum = new double[NUM_ACC_EGR][NUM_DIR]; - - } - - public void setHouseholdObject(Household hhObject) { - hh = hhObject; - } - - - public Household getHouseholdObject() { - return hh; - } - - public void setPersonObject(Person personObject) { - person = personObject; - } - - public Person getPersonObject() { - return person; - } - - public void setWorkTourObject(Tour tourObject) { - workTour = tourObject; - } - - public void setTourObject(Tour tourObject) { - tour = tourObject; - } - - public Tour getTourObject() { - return tour; - } - - public int getParkingArea() { - return parkingArea; - } - - public void setParkingArea(int parkingArea) { - this.parkingArea = parkingArea; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, - int destIndex, boolean debug) { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public int getPersonType() { - return person.getPersonTypeNumber(); - } - - public void setOrigDuDen(double arg) { - origDuDen = arg; - } - - public void setOrigEmpDen(double arg) { - origEmpDen = arg; - } - - public void setOrigTotInt(double arg) { - origTotInt = arg; - } - - public void setDestDuDen(double arg) { - destDuDen = arg; - } - - public void setDestEmpDen(double arg) { - destEmpDen = arg; - } - - public void setDestTotInt(double arg) { - destTotInt = arg; - } - - public void setReimburseProportion(double proportion) { - reimburseProportion = proportion; - } - - public void setLsWgtAvgCostM(double cost) { - lsWgtAvgCostM = cost; - } - - public void setLsWgtAvgCostD(double cost) { - lsWgtAvgCostD = cost; - } - - public void setLsWgtAvgCostH(double cost) { - lsWgtAvgCostH = cost; - } - - public void setPTazTerminalTime(float time) { - pTazTerminalTime = time; - } - - public void setATazTerminalTime(float time) { - aTazTerminalTime = time; - } - - public IndexValues getDmuIndexValues() { - return dmuIndex; - } - - public void setIndexDest(int d) { - dmuIndex.setDestZone(d); - } - - public void setTransitLogSum(int accEgr, boolean inbound, double value){ - transitLogSum[accEgr][inbound == true ? 1 : 0] = value; - } - - protected double getTransitLogSum(int accEgr,boolean inbound){ - return transitLogSum[accEgr][inbound == true ? 1 : 0]; - } - - public int getWorkTourModeIsSov() { - boolean tourModeIsSov = modelStructure.getTourModeIsSov(workTour - .getTourModeChoice()); - return tourModeIsSov ? 1 : 0; - } - - public int getWorkTourModeIsHov() { - boolean tourModeIsHov = modelStructure.getTourModeIsHov(workTour - .getTourModeChoice()); - return tourModeIsHov ? 1 : 0; - } - - public int getWorkTourModeIsBike() { - boolean tourModeIsBike = modelStructure.getTourModeIsBike(workTour - .getTourModeChoice()); - return tourModeIsBike ? 1 : 0; - } - - public int getTourCategoryJoint() { - if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - return 1; - else - return 0; - } - - public int getTourCategoryEscort() { - if (tour.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) - return 1; - else - return 0; - } - - public int getTourCategorySubtour() { - if (tour.getTourCategory().equalsIgnoreCase( - ModelStructure.AT_WORK_CATEGORY)) - return 1; - else - return 0; - } - - public int getNumberOfParticipantsInJointTour() { - int[] participants = tour.getPersonNumArray(); - int returnValue = 0; - if (participants != null) - returnValue = participants.length; - return returnValue; - } - - public int getHhSize() { - return hh.getHhSize(); - } - - public int getAutos() { - return hh.getAutosOwned(); - } - - public int getAge() { - return person.getAge(); - } - - public int getIncomeCategory() { - return hh.getIncomeCategory(); - } - - public int getIncomeInDollars() { - return hh.getIncomeInDollars(); - } - - - public void setNmWalkTimeOut(double nmWalkTime) { - nmWalkTimeOut = nmWalkTime; - } - - public double getNmWalkTimeOut() { - return nmWalkTimeOut; - } - - public void setNmWalkTimeIn(double nmWalkTime) { - nmWalkTimeIn = nmWalkTime; - } - - public double getNmWalkTimeIn() { - return nmWalkTimeIn; - } - - public void setNmBikeTimeOut(double nmBikeTime) { - nmBikeTimeOut = nmBikeTime; - } - - public double getNmBikeTimeOut() { - return nmBikeTimeOut; - } - - public void setNmBikeTimeIn(double nmBikeTime) { - nmBikeTimeIn = nmBikeTime; - } - - public double getNmBikeTimeIn() { - return nmBikeTimeIn; - } - - public double getWorkTimeFactor() { - return person.getTimeFactorWork(); - } - - public double getNonWorkTimeFactor(){ - return person.getTimeFactorNonWork(); - } - - /** - * Iterate through persons on tour and return non-work time factor - * for oldest person. If the person array is null then return 1.0. - * - * @return Time factor for oldest person on joint tour. - */ - public double getJointTourTimeFactor() { - int[] personNumArray = tour.getPersonNumArray(); - int oldestAge = -999; - Person oldestPerson = null; - for (int num : personNumArray){ - Person p = hh.getPerson(num); - if(p.getAge() > oldestAge){ - oldestPerson = p; - oldestAge = p.getAge(); - } - } - if(oldestPerson != null) - return oldestPerson.getTimeFactorNonWork(); - - return 1.0; - } - - - public int getFreeParkingEligibility() { - return person.getFreeParkingAvailableResult(); - } - - public double getReimburseProportion() { - return reimburseProportion; - } - - public double getMonthlyParkingCost() { - return lsWgtAvgCostM; - } - - public double getDailyParkingCost() { - return lsWgtAvgCostD; - } - - public double getHourlyParkingCost() { - return lsWgtAvgCostH; - } - - public double getPTazTerminalTime() { - return pTazTerminalTime; - } - - public double getATazTerminalTime() { - return aTazTerminalTime; - } - - public void setOriginMgra( int value ) { - originMgra = value; - } - - public void setDestMgra( int value ) { - destMgra = value; - } - - public int getOriginMgra() { - return originMgra; - } - - public int getDestMgra() { - return destMgra; - } - - /** - * 1 if household owns transponder, else 0 - * @return 1 if household owns transponder, else 0 - */ - public int getTransponderOwnership(){ - return hh.getTpChoice(); - } - - public double getIvtCoeff() { - return ivtCoeff; -} - -public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; -} - -public double getCostCoeff() { - return costCoeff; -} - -public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; -} - - public int getIndexValue(String variableName) { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) { - throw new UnsupportedOperationException(); - } - public int getUseOwnedAV(){ - - if(tour==null) - return 0; - - return (tour.getUseOwnedAV() ? 1: 0); - } - - - - public float getOrigTaxiWaitTime() { - return origTaxiWaitTime; - } - - - - public void setOrigTaxiWaitTime(float origTaxiWaitTime) { - this.origTaxiWaitTime = origTaxiWaitTime; - } - - - - public float getDestTaxiWaitTime() { - return destTaxiWaitTime; - } - - - - public void setDestTaxiWaitTime(float destTaxiWaitTime) { - this.destTaxiWaitTime = destTaxiWaitTime; - } - - - - public float getOrigSingleTNCWaitTime() { - return origSingleTNCWaitTime; - } - - - - public void setOrigSingleTNCWaitTime(float origSingleTNCWaitTime) { - this.origSingleTNCWaitTime = origSingleTNCWaitTime; - } - - - - public float getDestSingleTNCWaitTime() { - return destSingleTNCWaitTime; - } - - - - public void setDestSingleTNCWaitTime(float destSingleTNCWaitTime) { - this.destSingleTNCWaitTime = destSingleTNCWaitTime; - } - - - - public float getOrigSharedTNCWaitTime() { - return origSharedTNCWaitTime; - } - - - - public void setOrigSharedTNCWaitTime(float origSharedTNCWaitTime) { - this.origSharedTNCWaitTime = origSharedTNCWaitTime; - } - - - - public float getDestSharedTNCWaitTime() { - return destSharedTNCWaitTime; - } - - - - public void setDestSharedTNCWaitTime(float destSharedTNCWaitTime) { - this.destSharedTNCWaitTime = destSharedTNCWaitTime; - } - - - -} - diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java deleted file mode 100644 index 48122f6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourModeChoiceModel.java +++ /dev/null @@ -1,762 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; - - - - - - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; -import org.sandag.abm.modechoice.MgraDataManager; - -public class TourModeChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(TourModeChoiceModel.class); - private transient Logger tourMCManLogger = Logger.getLogger("tourMcMan"); - private transient Logger tourMCNonManLogger = Logger.getLogger("tourMcNonMan"); - - public static final String MANDATORY_MODEL_INDICATOR = ModelStructure.MANDATORY_CATEGORY; - public static final String NON_MANDATORY_MODEL_INDICATOR = "Non-Mandatory"; - public static final String AT_WORK_SUBTOUR_MODEL_INDICATOR = ModelStructure.AT_WORK_CATEGORY; - - public static final boolean DEBUG_BEST_PATHS = true; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - private static final int MC_DATA_SHEET = 0; - private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "tourModeChoice.uec.file"; - private static final String PROPERTIES_UEC_MAINT_TOUR_MODE_SHEET = "tourModeChoice.maint.model.page"; - private static final String PROPERTIES_UEC_DISCR_TOUR_MODE_SHEET = "tourModeChoice.discr.model.page"; - private static final String PROPERTIES_UEC_AT_WORK_TOUR_MODE_SHEET = "tourModeChoice.atwork.model.page"; - - - private static final String PROPERTIES_TOUR_UTILITY_IVT_COEFFS = "tour.utility.ivt.coeffs"; - private static final String PROPERTIES_TOUR_UTILITY_INCOME_COEFFS = "tour.utility.income.coeffs"; - private static final String PROPERTIES_TOUR_UTILITY_INCOME_EXPONENTS = "tour.utility.income.exponents"; - - // A MyChoiceModelApplication object and modeAltsAvailable[] is needed for - // each purpose - private ChoiceModelApplication[] mcModel; - private TourModeChoiceDMU mcDmuObject; - private McLogsumsCalculator logsumHelper; - - private ModelStructure modelStructure; - - private String tourCategory; - private String[] tourPurposeList; - - private HashMap purposeModelIndexMap; - - private String[][] modeAltNames; - - private boolean saveUtilsProbsFlag = false; - - // following arrays used to store ivt coefficients, and income coefficients, income exponents to calculate cost coefficient, by tour purpose - double[] ivtCoeffs; - double[] incomeCoeffs; - double[] incomeExponents; - - private MgraDataManager mgraManager; - - //added for TNC and Taxi modes - TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; - - public TourModeChoiceModel(HashMap propertyMap, ModelStructure myModelStructure, - String myTourCategory, CtrampDmuFactoryIf dmuFactory, McLogsumsCalculator myLogsumHelper) - { - - modelStructure = myModelStructure; - tourCategory = myTourCategory; - logsumHelper = myLogsumHelper; - // logsumHelper passed in, but if it were instantiated here, it woul be - // as follows - // logsumHelper = new McLogsumsAppender(); - // logsumHelper.setupSkimCalculators(propertyMap); - - mcDmuObject = dmuFactory.getModeChoiceDMU(); - setupModeChoiceModelApplicationArray(propertyMap, tourCategory); - - mgraManager = MgraDataManager.getInstance(); - - //get the coefficients for ivt and the coefficients to calculate the cost coefficient - ivtCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_IVT_COEFFS); - incomeCoeffs = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_INCOME_COEFFS); - incomeExponents = Util.getDoubleArrayFromPropertyMap(propertyMap, PROPERTIES_TOUR_UTILITY_INCOME_EXPONENTS); - - tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); - tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); - - } - - public AutoAndNonMotorizedSkimsCalculator getAnmSkimCalculator() - { - return logsumHelper.getAnmSkimCalculator(); - } - - private void setupModeChoiceModelApplicationArray(HashMap propertyMap, - String tourCategory) - { - - logger.info(String.format("setting up %s tour mode choice model.", tourCategory)); - - // locate the individual mandatory tour mode choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = propertyMap.get(PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - // default is to not save the tour mode choice utils and probs for each - // tour - String saveUtilsProbsString = propertyMap - .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); - if (saveUtilsProbsString != null) - { - if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; - } - - // get the number of purposes and declare the array dimension to be this - // size. - HashMap modelIndexMap = new HashMap(); - - // create a HashMap to map purposeName to model index - purposeModelIndexMap = new HashMap(); - - if (tourCategory.equalsIgnoreCase(MANDATORY_MODEL_INDICATOR)) - { - tourPurposeList = new String[3]; - tourPurposeList[0] = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; - tourPurposeList[1] = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - tourPurposeList[2] = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - - int uecIndex = 1; - int mcModelIndex = 0; - for (String purposeName : tourPurposeList) - { - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, mcModelIndex); - purposeModelIndexMap.put(purposeName, mcModelIndex++); - } else - { - purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); - } - uecIndex++; - } - - } else if (tourCategory.equalsIgnoreCase(NON_MANDATORY_MODEL_INDICATOR)) - { - tourPurposeList = new String[6]; - tourPurposeList[0] = ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME; - tourPurposeList[1] = ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME; - tourPurposeList[2] = ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME; - tourPurposeList[3] = ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME; - tourPurposeList[4] = ModelStructure.VISITING_PRIMARY_PURPOSE_NAME; - tourPurposeList[5] = ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME; - - int maintSheet = Integer - .parseInt(propertyMap.get(PROPERTIES_UEC_MAINT_TOUR_MODE_SHEET)); - int discrSheet = Integer - .parseInt(propertyMap.get(PROPERTIES_UEC_DISCR_TOUR_MODE_SHEET)); - - int uecIndex = 1; - int mcModelIndex = 0; - int i = 0; - for (String purposeName : tourPurposeList) - { - - uecIndex = -1; - if (purposeName.equalsIgnoreCase(ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME) - || purposeName - .equalsIgnoreCase(ModelStructure.SHOPPING_PRIMARY_PURPOSE_NAME) - || purposeName - .equalsIgnoreCase(ModelStructure.OTH_MAINT_PRIMARY_PURPOSE_NAME)) uecIndex = maintSheet; - else if (purposeName.equalsIgnoreCase(ModelStructure.EAT_OUT_PRIMARY_PURPOSE_NAME) - || purposeName - .equalsIgnoreCase(ModelStructure.VISITING_PRIMARY_PURPOSE_NAME) - || purposeName - .equalsIgnoreCase(ModelStructure.OTH_DISCR_PRIMARY_PURPOSE_NAME)) - uecIndex = discrSheet; - - // if the uec sheet for the model segment is not in the map, add - // it, otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, mcModelIndex); - purposeModelIndexMap.put(purposeName, mcModelIndex++); - } else - { - purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); - } - i++; - } - - } else if (tourCategory.equalsIgnoreCase(AT_WORK_SUBTOUR_MODEL_INDICATOR)) - { - tourPurposeList = new String[1]; - tourPurposeList[0] = ModelStructure.WORK_BASED_PRIMARY_PURPOSE_NAME; - - int[] uecSheets = new int[1]; - uecSheets[0] = Integer - .parseInt(propertyMap.get(PROPERTIES_UEC_AT_WORK_TOUR_MODE_SHEET)); - - int mcModelIndex = 0; - int i = 0; - for (String purposeName : tourPurposeList) - { - int uecIndex = uecSheets[i]; - - // if the uec sheet for the model segment is not in the map, add - // it, otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, mcModelIndex); - purposeModelIndexMap.put(purposeName, mcModelIndex++); - } else - { - purposeModelIndexMap.put(purposeName, modelIndexMap.get(uecIndex)); - } - i++; - } - - } - - mcModel = new ChoiceModelApplication[modelIndexMap.size()]; - - // declare dimensions for the array of choice alternative availability - // by - // purpose - modeAltNames = new String[purposeModelIndexMap.size()][]; - - // for each unique model index, create the ChoiceModelApplication object - // and - // the availabilty array - int i = 0; - for (int m : modelIndexMap.keySet()) - { - mcModel[i] = new ChoiceModelApplication(mcUecFile, m, MC_DATA_SHEET, propertyMap, - (VariableTable) mcDmuObject); - modeAltNames[i] = mcModel[i].getAlternativeNames(); - i++; - } - - - } - - public double getModeChoiceLogsum(Household household, Person person, Tour tour, - Logger modelLogger, String choiceModelDescription, String decisionMakerLabel) - { - - // update the MC dmuObjects for this person - mcDmuObject.setHouseholdObject(household); - mcDmuObject.setPersonObject(person); - mcDmuObject.setTourObject(tour); - mcDmuObject.setDmuIndexValues(household.getHhId(), tour.getTourDestMgra(), - tour.getTourOrigMgra(), tour.getTourDestMgra(), household.getDebugChoiceModels()); - mcDmuObject.setOriginMgra(tour.getTourOrigMgra()); - mcDmuObject.setDestMgra(tour.getTourDestMgra()); - - float SingleTNCWaitTimeOrig = 0; - float SingleTNCWaitTimeDest = 0; - float SharedTNCWaitTimeOrig = 0; - float SharedTNCWaitTimeDest = 0; - float TaxiWaitTimeOrig = 0; - float TaxiWaitTimeDest = 0; - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); - float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); - - if(household!=null){ - Random hhRandom = household.getHhRandom(); - double rnum = hhRandom.nextDouble(); - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); - }else{ - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); - } - - mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); - mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); - mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); - mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); - mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); - mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); - - return getModeChoiceLogsum(mcDmuObject, tour, modelLogger, choiceModelDescription, - decisionMakerLabel); - } - - public double getModeChoiceLogsum(TourModeChoiceDMU mcDmuObject, Tour tour, Logger modelLogger, - String choiceModelDescription, String decisionMakerLabel) - { - - int modelIndex = purposeModelIndexMap.get(tour.getTourPrimaryPurpose()); - - Household household = tour.getPersonObject().getHouseholdObject(); - double income = (double) household.getIncomeInDollars(); - double timeFactor = 1.0f; - if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - timeFactor = mcDmuObject.getJointTourTimeFactor(); - else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) - timeFactor = mcDmuObject.getWorkTimeFactor(); - else - timeFactor = mcDmuObject.getNonWorkTimeFactor(); - - double ivtCoeff = ivtCoeffs[modelIndex]; - double incomeCoeff = incomeCoeffs[modelIndex]; - double incomeExpon = incomeExponents[modelIndex]; - double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); - - mcDmuObject.setIvtCoeff(ivtCoeff*timeFactor); - mcDmuObject.setCostCoeff(costCoeff); - - float SingleTNCWaitTimeOrig = 0; - float SingleTNCWaitTimeDest = 0; - float SharedTNCWaitTimeOrig = 0; - float SharedTNCWaitTimeDest = 0; - float TaxiWaitTimeOrig = 0; - float TaxiWaitTimeDest = 0; - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); - float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); - - if(household!=null){ - Random hhRandom = household.getHhRandom(); - double rnum = hhRandom.nextDouble(); - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); - }else{ - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); - } - - mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); - mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); - mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); - mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); - mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); - mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); - - // log headers to traceLogger - if (household.getDebugChoiceModels()) - { - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - } - - double logsum = logsumHelper.calculateTourMcLogsum(tour.getTourOrigMgra(), - tour.getTourDestMgra(), tour.getTourDepartPeriod(), tour.getTourArrivePeriod(), - mcModel[modelIndex], mcDmuObject); - - // write UEC calculation results to separate model specific log file - if (household.getDebugChoiceModels()) - { - String loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); - modelLogger.info(""); - modelLogger.info(""); - } - - return logsum; - - } - - public int getModeChoice(TourModeChoiceDMU mcDmuObject, String purposeName) - { - - int modelIndex = purposeModelIndexMap.get(purposeName); - - Household household = mcDmuObject.getHouseholdObject(); - Tour tour = mcDmuObject.getTourObject(); - double income = (double) household.getIncomeInDollars(); - double ivtCoeff = ivtCoeffs[modelIndex]; - double incomeCoeff = incomeCoeffs[modelIndex]; - double incomeExpon = incomeExponents[modelIndex]; - double costCoeff = calculateCostCoefficient(income, incomeCoeff,incomeExpon); - double timeFactor = 1.0f; - if(tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - timeFactor = mcDmuObject.getJointTourTimeFactor(); - else if(tour.getTourPrimaryPurposeIndex()==ModelStructure.WORK_PRIMARY_PURPOSE_INDEX) - timeFactor = mcDmuObject.getWorkTimeFactor(); - else - timeFactor = mcDmuObject.getNonWorkTimeFactor(); - - mcDmuObject.setIvtCoeff(ivtCoeff * timeFactor); - mcDmuObject.setCostCoeff(costCoeff); - - float SingleTNCWaitTimeOrig = 0; - float SingleTNCWaitTimeDest = 0; - float SharedTNCWaitTimeOrig = 0; - float SharedTNCWaitTimeDest = 0; - float TaxiWaitTimeOrig = 0; - float TaxiWaitTimeDest = 0; - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getTourOrigMgra()); - float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getTourDestMgra()); - - if(household!=null){ - Random hhRandom = household.getHhRandom(); - double rnum = hhRandom.nextDouble(); - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); - }else{ - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSingleTNCWaitTime( popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanSharedTNCWaitTime( popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime( popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.getMeanTaxiWaitTime(popEmpDenDest); - } - - mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); - mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); - mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); - mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); - mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); - mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); - - Logger modelLogger = null; - if (tourCategory.equalsIgnoreCase(ModelStructure.MANDATORY_CATEGORY)) modelLogger = tourMCManLogger; - else modelLogger = tourMCNonManLogger; - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - String separator = ""; - - - if (household.getDebugChoiceModels()) - { - - if (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person person = null; - Person[] persons = mcDmuObject.getHouseholdObject().getPersons(); - int[] personNums = tour.getPersonNumArray(); - for (int n = 0; n < personNums.length; n++) - { - int p = personNums[n]; - person = persons[p]; - - choiceModelDescription = String.format( - "%s Tour Mode Choice Model for: Purpose=%s, Home=%d, Dest=%d", - tourCategory, purposeName, household.getHhMgra(), - tour.getTourDestMgra()); - decisionMakerLabel = String - .format("HH=%d, person record %d of %d in joint tour, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), p, personNums.length, - person.getPersonNum(), person.getPersonType(), tour.getTourId()); - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - household.logTourObject(loggingHeader, modelLogger, person, tour); - } - } else - { - Person person = mcDmuObject.getPersonObject(); - - choiceModelDescription = String.format( - "%s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", - tourCategory, purposeName, tour.getTourOrigMgra(), tour.getTourDestMgra()); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - household.logTourObject(loggingHeader, modelLogger, person, tour); - } - - } - - logsumHelper.setTourMcDmuAttributes(mcDmuObject, tour.getTourOrigMgra(), - tour.getTourDestMgra(), tour.getTourDepartPeriod(), tour.getTourArrivePeriod(), - (household.getDebugChoiceModels() && DEBUG_BEST_PATHS)); - - // mode choice UEC references highway skim matrices directly, so set - // index orig/dest to O/D TAZs. - IndexValues mcDmuIndex = mcDmuObject.getDmuIndexValues(); - mcDmuIndex.setOriginZone(mgraManager.getTaz(tour.getTourOrigMgra())); - mcDmuIndex.setDestZone(mgraManager.getTaz(tour.getTourDestMgra())); - mcDmuIndex.setZoneIndex(tour.getTourDestMgra()); - mcDmuObject.setOriginMgra(tour.getTourOrigMgra()); - mcDmuObject.setDestMgra(tour.getTourDestMgra()); - - float logsum = (float) mcModel[modelIndex].computeUtilities(mcDmuObject, mcDmuIndex); - tour.setTourModeLogsum(logsum); - - mcDmuIndex.setOriginZone(tour.getTourOrigMgra()); - mcDmuIndex.setDestZone(tour.getTourDestMgra()); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen; - if (mcModel[modelIndex].getAvailabilityCount() > 0) - { - - chosen = mcModel[modelIndex].getChoiceResult(rn); - - // best tap pairs were determined and saved in mcDmuObject while - // setting dmu skim attributes - // if chosen mode is a transit mode, save these tap pairs in the - // tour object; if not transit tour attributes remain null. - if (modelStructure.getTourModeIsTransit(chosen)) - { - tour.setBestWtwTapPairsOut(logsumHelper.getBestWtwTapsOut()); - tour.setBestWtwTapPairsIn(logsumHelper.getBestWtwTapsIn()); - tour.setBestWtdTapPairsOut(logsumHelper.getBestWtdTapsOut()); - tour.setBestWtdTapPairsIn(logsumHelper.getBestWtdTapsIn()); - tour.setBestDtwTapPairsOut(logsumHelper.getBestDtwTapsOut()); - tour.setBestDtwTapPairsIn(logsumHelper.getBestDtwTapsIn()); - } - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = mcModel[modelIndex].getUEC(); - - double vot = 0.0; - - if(modelStructure.getTourModeIsS2(chosen)){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = uec.getValueForIndex(votIndex); - }else if (modelStructure.getTourModeIsS3(chosen)){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = uec.getValueForIndex(votIndex); - } - tour.setValueOfTime(vot); - - - } else - { - - if (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - Person person = null; - Person[] persons = mcDmuObject.getHouseholdObject().getPersons(); - int[] personNums = tour.getPersonNumArray(); - for (int n = 0; n < personNums.length; n++) - { - int p = personNums[n]; - person = persons[p]; - - choiceModelDescription = String - .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Home=%d, Dest=%d", - tourCategory, purposeName, household.getHhMgra(), - tour.getTourDestMgra()); - decisionMakerLabel = String - .format("HH=%d, person record %d of %d in joint tour, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), p, personNums.length, - person.getPersonNum(), person.getPersonType(), tour.getTourId()); - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading( - choiceModelDescription, decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - household.logTourObject(loggingHeader, modelLogger, person, tour); - } - } else - { - Person person = mcDmuObject.getPersonObject(); - - choiceModelDescription = String - .format("No alternatives available for %s Tour Mode Choice Model for: Purpose=%s, Orig=%d, Dest=%d", - tourCategory, purposeName, tour.getTourOrigMgra(), - tour.getTourDestMgra()); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourId=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tour.getTourId()); - loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - - mcModel[modelIndex].choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - modelLogger.info(" "); - for (int k = 0; k < loggingHeader.length(); k++) - separator += "+"; - modelLogger.info(loggingHeader); - modelLogger.info(separator); - - household.logTourObject(loggingHeader, modelLogger, person, tour); - } - - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - modelLogger.info(""); - modelLogger.info(""); - - logger.error(String - .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", - household.getHhId(), tourCategory)); - throw new RuntimeException(); - } - - // debug output - if (household.getDebugChoiceModels()) - { - - double[] utilities = mcModel[modelIndex].getUtilities(); // 0s-indexing - double[] probabilities = mcModel[modelIndex].getProbabilities(); // 0s-indexing - boolean[] availabilities = mcModel[modelIndex].getAvailabilities(); // 1s-indexing - String[] altNames = mcModel[modelIndex].getAlternativeNames(); // 0s-indexing - - if (tour.getTourCategory() - .equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) - { - modelLogger.info("Joint Tour Id: " + tour.getTourId()); - } else - { - Person person = mcDmuObject.getPersonObject(); - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString - + ", Tour Id: " + tour.getTourId()); - } - modelLogger - .info("Alternative Utility Probability CumProb"); - modelLogger - .info("-------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < mcModel[modelIndex].getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %s", k + 1, altNames[k]); - modelLogger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger.info(separator); - modelLogger.info(""); - modelLogger.info(""); - - // write choice model alternative info to log file - mcModel[modelIndex].logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - mcModel[modelIndex].logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - mcModel[modelIndex].logLogitCalculations(choiceModelDescription, decisionMakerLabel); - - // write UEC calculation results to separate model specific log file - mcModel[modelIndex].logUECResults(modelLogger, loggingHeader); - } - - if (saveUtilsProbsFlag) - { - - // get the utilities and probabilities arrays for the tour mode - // choice - // model for this tour and save them to the tour object - double[] dUtils = mcModel[modelIndex].getUtilities(); - double[] dProbs = mcModel[modelIndex].getProbabilities(); - - float[] utils = new float[dUtils.length]; - float[] probs = new float[dUtils.length]; - for (int k = 0; k < dUtils.length; k++) - { - utils[k] = (float) dUtils[k]; - probs[k] = (float) dProbs[k]; - } - - tour.setTourModalUtilities(utils); - tour.setTourModalProbabilities(probs); - - } - - return chosen; - - } - - public String[] getModeAltNames(int purposeIndex) - { - int modelIndex = purposeModelIndexMap.get(tourPurposeList[purposeIndex]); - return modeAltNames[modelIndex]; - } - - /** - * This method calculates a cost coefficient based on the following formula: - * - * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) - * - * - * @param incomeCoeff - * @param incomeExponent - * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice - */ - public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ - - return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java deleted file mode 100644 index bd6ef9b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TourVehicleTypeChoiceModel.java +++ /dev/null @@ -1,189 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import com.pb.common.calculator.VariableTable; -import com.pb.common.model.ModelException; - -import org.apache.log4j.Logger; - -public class TourVehicleTypeChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(TourVehicleTypeChoiceModel.class); - float probabilityBoostAutosLTDrivers = 0; - float probabilityBoostAutosGEDrivers = 0; - - - public TourVehicleTypeChoiceModel(HashMap rbMap) - { - - logger.info("setting up tour tNCVehicle type choice model."); - probabilityBoostAutosLTDrivers = Util.getFloatValueFromPropertyMap(rbMap,"Mobility.AV.ProbabilityBoost.AutosLTDrivers"); - probabilityBoostAutosGEDrivers = Util.getFloatValueFromPropertyMap(rbMap,"Mobility.AV.ProbabilityBoost.AutosGEDrivers"); - - - } - - /** - * Calculate the probability of the tour using the AV in the household. If AVs owned =0, returns 0, else - * returns a probability equal to the share of AVs in the household, boosted by the parameters in the properties file. - * The parameters are named Mobility.AV.ProbabilityBoost.AutosLTDrivers and Mobility.AV.ProbabilityBoost.AutosGEDrivers - * and are read in the object constructor. - * - * @param hhObj - * @return The probability of using one of the household AVs for the tour. - */ - public double calculateProbability(Household hhObj){ - - float numberOfAVs = (float) hhObj.getAutomatedVehicles(); - - if(numberOfAVs==0) - return 0; - - float numberOfCVs = (float) hhObj.getConventionalVehicles(); - float numberOfDrivers = (float) hhObj.getDrivers(); - float totalVehicles = numberOfAVs + numberOfCVs; - float probability = numberOfAVs/totalVehicles; - if(totalVehicles - * Started: Apr 14, 2009 11:09:58 AM - */ -public class TransponderChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TransponderChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - - private Household hh; - - private double percentTazIncome100Kplus; - private double percentTazMultpleAutos; - private double expectedTravelTimeSavings; - private double transpDist; - private double pctDetour; - private double accessibility; - - public TransponderChoiceDMU() - { - dmuIndex = new IndexValues(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (hh.getDebugChoiceModels()) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug Free Parking UEC"); - } - } - - public void setHouseholdObject(Household hhObj) - { - hh = hhObj; - } - - public void setPctIncome100Kplus(double value) - { - percentTazIncome100Kplus = value; - } - - public void setPctTazMultpleAutos(double value) - { - percentTazMultpleAutos = value; - } - - public void setExpectedTravelTimeSavings(double value) - { - expectedTravelTimeSavings = value; - } - - public void setTransponderDistance(double value) - { - transpDist = value; - } - - public void setPctDetour(double value) - { - pctDetour = value; - } - - public void setAccessibility(double value) - { - accessibility = value; - } - - public double getPctIncome100Kplus() - { - return percentTazIncome100Kplus; - } - - public double getPctTazMultpleAutos() - { - return percentTazMultpleAutos; - } - - public double getExpectedTravelTimeSavings() - { - return expectedTravelTimeSavings; - } - - public double getTransponderDistance() - { - return transpDist; - } - - public double getPctDetour() - { - return pctDetour; - } - - public double getAccessibility() - { - return accessibility; - } - - public int getAutoOwnership() - { - return hh.getAutosOwned(); - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java deleted file mode 100644 index 35eda9f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TransponderChoiceModel.java +++ /dev/null @@ -1,131 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AccessibilitiesTable; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class TransponderChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("tp"); - - private static final String TP_CONTROL_FILE_TARGET = "tc.uec.file"; - private static final String TP_DATA_SHEET_TARGET = "tc.data.page"; - private static final String TP_MODEL_SHEET_TARGET = "tc.model.page"; - - public static final int TP_MODEL_NO_ALT = 1; - public static final int TP_MODEL_YES_ALT = 2; - - private ChoiceModelApplication tpModel; - private TransponderChoiceDMU tpDmuObject; - - private AccessibilitiesTable accTable; - - private double[] pctHighIncome; - private double[] pctMultipleAutos; - private double[] avgtts; - private double[] transpDist; - private double[] pctDetour; - - public TransponderChoiceModel(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory, AccessibilitiesTable accTable, double[] pctHighIncome, - double[] pctMultipleAutos, double[] avgtts, double[] transpDist, double[] pctDetour) - { - this.accTable = accTable; - this.pctHighIncome = pctHighIncome; - this.pctMultipleAutos = pctMultipleAutos; - this.avgtts = avgtts; - this.transpDist = transpDist; - this.pctDetour = pctDetour; - - setupTransponderChoiceModelApplication(propertyMap, dmuFactory); - } - - private void setupTransponderChoiceModelApplication(HashMap propertyMap, - CtrampDmuFactoryIf dmuFactory) - { - logger.info("setting up transponder choice model."); - - // locate the transponder choice UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tpUecFile = uecFileDirectory + propertyMap.get(TP_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TP_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, TP_MODEL_SHEET_TARGET); - - // create the transponder choice model DMU object. - tpDmuObject = dmuFactory.getTransponderChoiceDMU(); - - // create the transponder choice model object - tpModel = new ChoiceModelApplication(tpUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) tpDmuObject); - - } - - public void applyModel(Household hhObject) - { - - int homeTaz = hhObject.getHhTaz(); - - tpDmuObject.setHouseholdObject(hhObject); - - // set the zone, orig and dest attributes - tpDmuObject.setDmuIndexValues(hhObject.getHhId(), hhObject.getHhTaz(), hhObject.getHhTaz(), - 0); - - tpDmuObject.setPctIncome100Kplus(pctHighIncome[homeTaz]); - tpDmuObject.setPctTazMultpleAutos(pctMultipleAutos[homeTaz]); - tpDmuObject.setExpectedTravelTimeSavings(avgtts[homeTaz]); - tpDmuObject.setTransponderDistance(transpDist[homeTaz]); - tpDmuObject.setPctDetour(pctDetour[homeTaz]); - - float accessibility = accTable.getAggregateAccessibility("transit", hhObject.getHhMgra()); - tpDmuObject.setAccessibility(accessibility); - - Random hhRandom = hhObject.getHhRandom(); - double randomNumber = hhRandom.nextDouble(); - - // compute utilities and choose transponder choice alternative. - float logsum = (float) tpModel.computeUtilities(tpDmuObject, tpDmuObject.getDmuIndexValues()); - - hhObject.setTransponderLogsum(logsum); - - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - if (tpModel.getAvailabilityCount() > 0) - { - chosenAlt = tpModel.getChoiceResult(randomNumber); - } else - { - String decisionMaker = String.format("HHID=%d", hhObject.getHhId()); - String errorMessage = String - .format("Exception caught for %s, no available transponder choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - tpModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - // write choice model alternative info to log file - if (hhObject.getDebugChoiceModels()) - { - String decisionMaker = String.format("HHID=%d", hhObject.getHhId()); - tpModel.logAlternativesInfo("Transponder Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d with rn %.8f", - "Transponder Choice", decisionMaker, chosenAlt, randomNumber)); - tpModel.logUECResults(logger, decisionMaker); - } - - hhObject.setTpChoice(chosenAlt - 1); - - hhObject.setTpRandomCount(hhObject.getHhRandomCount()); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java deleted file mode 100644 index b8ca3e7..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/TripModeChoiceDMU.java +++ /dev/null @@ -1,846 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class TripModeChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TripModeChoiceDMU.class); - - protected static final int WTW = McLogsumsCalculator.WTW; - protected static final int WTD = McLogsumsCalculator.WTD; - protected static final int DTW = McLogsumsCalculator.DTW; - protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - protected HashMap methodIndexMap; - - protected Tour tour; - protected Person person; - protected Household hh; - protected IndexValues dmuIndex; - - protected double nmWalkTime; - protected double nmBikeTime; - - protected ModelStructure modelStructure; - - protected double origDuDen; - protected double origEmpDen; - protected double origTotInt; - protected double destDuDen; - protected double destEmpDen; - protected double destTotInt; - - protected int tripOrigIsTourDest; - protected int tripDestIsTourDest; - - protected int tripTime; - protected int firstTrip; - protected int lastTrip; - protected int outboundStops; - protected int inboundStops; - - protected int incomeInDollars; - protected int age; - protected int adults; - protected int autos; - protected int hhSize; - protected int personIsFemale; - - protected int departPeriod; - protected int arrivePeriod; - protected int tripPeriod; - - protected int escortTour; - protected int jointTour; - protected int partySize; - - protected int outboundHalfTourDirection; - protected float waitTimeTaxi; - protected float waitTimeSingleTNC; - protected float waitTimeSharedTNC; - - protected int tourModeIsDA; - protected int tourModeIsS2; - protected int tourModeIsS3; - protected int tourModeIsWalk; - protected int tourModeIsBike; - protected int tourModeIsWTran; - protected int tourModeIsPnr; - protected int tourModeIsKnr; - protected int tourModeIsSchBus; - - protected double reimburseAmount; - - protected float pTazTerminalTime; - protected float aTazTerminalTime; - - protected int[] mgraParkArea; - - protected double[] lsWgtAvgCostM; - protected double[] lsWgtAvgCostD; - protected double[] lsWgtAvgCostH; - - protected boolean segmentIsIk; - protected boolean autoModeRequiredForDriveTransit; - protected boolean walkModeAllowedForDriveTransit; - - protected double ivtCoeff; - protected double costCoeff; - - protected double[] transitLogSum; - - protected boolean inbound; - - protected int originMgra; - protected int destMgra; - - - public TripModeChoiceDMU(ModelStructure modelStructure, Logger aLogger) - { - this.modelStructure = modelStructure; - dmuIndex = new IndexValues(); - - transitLogSum = new double[McLogsumsCalculator.NUM_ACC_EGR]; - } - - - - public void setParkingCostInfo(int[] mgraParkArea, double[] lsWgtAvgCostM, - double[] lsWgtAvgCostD, double[] lsWgtAvgCostH) - { - this.mgraParkArea = mgraParkArea; - this.lsWgtAvgCostM = lsWgtAvgCostM; - this.lsWgtAvgCostD = lsWgtAvgCostD; - this.lsWgtAvgCostH = lsWgtAvgCostH; - } - - public void setHouseholdObject(Household hhObject) - { - hh = hhObject; - } - - public Household getHouseholdObject() - { - return hh; - } - - public void setPersonObject(Person personObject) - { - person = personObject; - } - - public Person getPersonObject() - { - return person; - } - - public void setTourObject(Tour tourObject) - { - tour = tourObject; - } - - public Tour getTourObject() - { - return tour; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public float getTimeOutbound() - { - return tour.getTourDepartPeriod(); - } - - public float getTimeInbound() - { - return tour.getTourArrivePeriod(); - } - - public void setSegmentIsIk(boolean flag) - { - segmentIsIk = flag; - } - - public int getSegmentIsIk() - { - return segmentIsIk ? 1 : 0; - } - - public void setIncomeInDollars(int arg) - { - incomeInDollars = arg; - } - - public void setAutos(int arg) - { - autos = arg; - } - - public void setAdults(int arg) - { - adults = arg; - } - - public void setHhSize(int arg) - { - hhSize = arg; - } - - public void setAge(int arg) - { - age = arg; - } - - public void setPersonIsFemale(int arg) - { - personIsFemale = arg; - } - - public void setEscortTour(int arg) - { - escortTour = arg; - } - - public void setJointTour(int arg) - { - jointTour = arg; - } - - public void setPartySize(int arg) - { - partySize = arg; - } - - public void setOutboundHalfTourDirection(int arg) - { - outboundHalfTourDirection = arg; - } - - public void setDepartPeriod(int period) - { - departPeriod = period; - } - - public void setArrivePeriod(int period) - { - arrivePeriod = period; - } - - public void setTripPeriod(int period) - { - tripPeriod = period; - } - - public void setOutboundStops(int stops) - { - outboundStops = stops; - } - - public void setInboundStops(int stops) - { - inboundStops = stops; - } - - public void setFirstTrip(int trip) - { - firstTrip = trip; - } - - public void setLastTrip(int trip) - { - lastTrip = trip; - } - - public void setTourModeIsDA(int arg) - { - tourModeIsDA = arg; - } - - public void setTourModeIsS2(int arg) - { - tourModeIsS2 = arg; - } - - public void setTourModeIsS3(int arg) - { - tourModeIsS3 = arg; - } - - public void setTourModeIsWalk(int arg) - { - tourModeIsWalk = arg; - } - - public void setTourModeIsBike(int arg) - { - tourModeIsBike = arg; - } - - public void setTourModeIsWTran(int arg) - { - tourModeIsWTran = arg; - } - - public void setTourModeIsPnr(int arg) - { - tourModeIsPnr = arg; - } - - public void setTourModeIsKnr(int arg) - { - tourModeIsKnr = arg; - } - - public void setTourModeIsSchBus(int arg) - { - tourModeIsSchBus = arg; - } - - public void setOrigDuDen(double arg) - { - origDuDen = arg; - } - - public void setOrigEmpDen(double arg) - { - origEmpDen = arg; - } - - public void setOrigTotInt(double arg) - { - origTotInt = arg; - } - - public void setDestDuDen(double arg) - { - destDuDen = arg; - } - - public void setDestEmpDen(double arg) - { - destEmpDen = arg; - } - - public void setDestTotInt(double arg) - { - destTotInt = arg; - } - - public void setReimburseProportion(double prop) - { - reimburseAmount = prop; - } - - public void setPTazTerminalTime(float time) - { - pTazTerminalTime = time; - } - - public void setATazTerminalTime(float time) - { - aTazTerminalTime = time; - } - - public void setTripOrigIsTourDest(int value) - { - tripOrigIsTourDest = value; - } - - public void setTripDestIsTourDest(int value) - { - tripDestIsTourDest = value; - } - - public void setBikeLogsum(int origin, int dest, boolean inbound) { - //do nothing - this is a stub to allow SANDAG to work correctly - // see SandagTripModeChoiceModelDMU for actual implementation - } - - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public void setAutoModeRequiredForTripSegment(boolean flag) - { - autoModeRequiredForDriveTransit = flag; - } - - public void setWalkModeAllowedForTripSegment(boolean flag) - { - walkModeAllowedForDriveTransit = flag; - } - - public void setIndexDest(int d) - { - dmuIndex.setDestZone(d); - } - - public void setNonMotorizedWalkTime(double walkTime) - { - nmWalkTime = walkTime; - } - - public void setNonMotorizedBikeTime(double bikeTime) - { - nmBikeTime = bikeTime; - } - - public int getAutoModeAllowedForTripSegment() - { - return autoModeRequiredForDriveTransit ? 1 : 0; - } - - public int getWalkModeAllowedForTripSegment() - { - return walkModeAllowedForDriveTransit ? 1 : 0; - } - - public int getTourModeIsDA() - { - boolean tourModeIsDa = modelStructure.getTourModeIsSov(tour.getTourModeChoice()); - return tourModeIsDa ? 1 : 0; - } - - public int getTourModeIsS2() - { - boolean tourModeIsS2 = modelStructure.getTourModeIsS2(tour.getTourModeChoice()); - return tourModeIsS2 ? 1 : 0; - } - - public int getTourModeIsS3() - { - boolean tourModeIsS3 = modelStructure.getTourModeIsS3(tour.getTourModeChoice()); - return tourModeIsS3 ? 1 : 0; - } - - public int getTourModeIsSchBus() - { - boolean tourModeIsSchBus = modelStructure.getTourModeIsSchoolBus(tour.getTourModeChoice()); - return tourModeIsSchBus ? 1 : 0; - } - - public int getTourModeIsWalk() - { - boolean tourModeIsWalk = modelStructure.getTourModeIsWalk(tour.getTourModeChoice()); - return tourModeIsWalk ? 1 : 0; - } - - public int getTourModeIsBike() - { - boolean tourModeIsBike = modelStructure.getTourModeIsBike(tour.getTourModeChoice()); - return tourModeIsBike ? 1 : 0; - } - - public int getTourModeIsWTran() - { - boolean tourModeIsWTran = modelStructure.getTourModeIsWalkTransit(tour.getTourModeChoice()); - return tourModeIsWTran ? 1 : 0; - } - - public int getTourModeIsPnr() - { - boolean tourModeIsPnr = modelStructure.getTourModeIsPnr(tour.getTourModeChoice()); - return tourModeIsPnr ? 1 : 0; - } - - public int getTourModeIsKnr() - { - boolean tourModeIsKnr = modelStructure.getTourModeIsKnr(tour.getTourModeChoice()); - return tourModeIsKnr ? 1 : 0; - } - - public int getTourModeIsTncTransit() - { - boolean tourModeIsTncTransit = modelStructure.getTourModeIsTncTransit(tour.getTourModeChoice()); - return tourModeIsTncTransit ? 1 : 0; - } - - public int getTourModeIsMaas() - { - boolean tourModeIsMaas = modelStructure.getTourModeIsMaas(tour.getTourModeChoice()); - return tourModeIsMaas ? 1 : 0; - } - - public void setTransitLogSum(int accEgr, double value){ - transitLogSum[accEgr] = value; - } - - public double getTransitLogSum(int accEgr){ - return transitLogSum[accEgr]; - } - - - public double getODUDen() - { - return origDuDen; - } - - public double getOEmpDen() - { - return origEmpDen; - } - - public double getOTotInt() - { - return origTotInt; - } - - public double getDDUDen() - { - return destDuDen; - } - - public double getDEmpDen() - { - return destEmpDen; - } - - public double getDTotInt() - { - return destTotInt; - } - - public int getFirstTrip() - { - return firstTrip; - } - - public int getLastTrip() - { - return lastTrip; - } - - public int getTourCategoryJoint() - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.JOINT_NON_MANDATORY_CATEGORY)) return 1; - else return 0; - } - - public int getTourCategoryEscort() - { - if (tour.getTourPrimaryPurpose().equalsIgnoreCase( - ModelStructure.ESCORT_PRIMARY_PURPOSE_NAME)) return 1; - else return 0; - } - - public int getTourCategorySubtour() - { - if (tour.getTourCategory().equalsIgnoreCase(ModelStructure.AT_WORK_CATEGORY)) return 1; - else return 0; - } - - public int getNumberOfParticipantsInJointTour() - { - int[] participants = tour.getPersonNumArray(); - int returnValue = 0; - if (participants != null) returnValue = participants.length; - return returnValue; - } - - public int getHhSize() - { - return hh.getHhSize(); - } - - public int getAutos() - { - return hh.getAutosOwned(); } - - public int getAge() - { - return person.getAge(); - } - - public int getFemale() - { - return person.getPersonIsFemale(); - } - - public int getIncomeCategory() - { - return hh.getIncomeCategory(); - } - - public double getNm_walkTime() - { - return nmWalkTime; - } - - public double getNm_bikeTime() - { - return nmBikeTime; - } - - public double getReimburseAmount() - { - return reimburseAmount; - } - - public double getMonthlyParkingCostTourDest() - { - return lsWgtAvgCostM[tour.getTourDestMgra()]; - } - - public double getDailyParkingCostTourDest() - { - return lsWgtAvgCostD[tour.getTourDestMgra()]; - } - - public double getHourlyParkingCostTourDest() - { - return lsWgtAvgCostH[tour.getTourDestMgra()]; - } - - public double getHourlyParkingCostTripOrig() - { - return lsWgtAvgCostH[originMgra]; - } - - public double getHourlyParkingCostTripDest() - { - return lsWgtAvgCostH[destMgra]; - } - - public int getTripOrigIsTourDest() - { - return tripOrigIsTourDest; - } - - public int getTripDestIsTourDest() - { - return tripDestIsTourDest; - } - - public void setOriginMgra( int value ) { - originMgra = value; - } - - public void setDestMgra( int value ) { - destMgra = value; - } - - public int getFreeOnsite() - { - return person.getFreeParkingAvailableResult() == ParkingProvisionModel.FP_MODEL_FREE_ALT ? 1 - : 0; - } - - public int getPersonType() - { - return person.getPersonTypeNumber(); - } - - public double getWorkTimeFactor() { - return person.getTimeFactorWork(); - } - - public double getNonWorkTimeFactor(){ - return person.getTimeFactorNonWork(); - } - - /** - * Iterate through persons on tour and return non-work time factor - * for oldest person. If the person array is null then return 1.0. - * - * @return Time factor for oldest person on joint tour. - */ - public double getJointTourTimeFactor() { - int[] personNumArray = tour.getPersonNumArray(); - int oldestAge = -999; - Person oldestPerson = null; - for (int num : personNumArray){ - Person p = hh.getPerson(num); - if(p.getAge() > oldestAge){ - oldestPerson = p; - oldestAge = p.getAge(); - } - } - if(oldestPerson != null) - return oldestPerson.getTimeFactorNonWork(); - - return 1.0; - } - - public double getPTazTerminalTime() - { - return pTazTerminalTime; - } - - public double getATazTerminalTime() - { - return aTazTerminalTime; - } - - /** - * @return the originMgra - */ - public int getOriginMgra() { - return originMgra; - } - - /** - * @return the destMgra - */ - public int getDestMgra() { - return destMgra; - } - - public boolean isInbound() { - return inbound; - } - - public int getInbound() { - return inbound ? 1 : 0 ; - } - - - public void setInbound(boolean inbound) { - this.inbound = inbound; - } - - /** - * 1 if household owns transponder, else 0 - * @return 1 if household owns transponder, else 0 - */ - public int getTransponderOwnership(){ - return hh.getTpChoice(); - } - - - - - public double getIvtCoeff() { - return ivtCoeff; - } - - - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - - - public double getCostCoeff() { - return costCoeff; - } - - - - public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; - } - - public int getIncomeInDollars() - { - return hh.getIncomeInDollars(); - } - - public int getUseOwnedAV(){ - - if(tour==null) - return 0; - - return (tour.getUseOwnedAV() ? 1: 0); - } - - - - - public float getWaitTimeTaxi() { - return waitTimeTaxi; - } - - public void setWaitTimeTaxi(float waitTimeTaxi) { - this.waitTimeTaxi = waitTimeTaxi; - } - - public float getWaitTimeSingleTNC() { - return waitTimeSingleTNC; - } - - public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { - this.waitTimeSingleTNC = waitTimeSingleTNC; - } - - public float getWaitTimeSharedTNC() { - return waitTimeSharedTNC; - } - - public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { - this.waitTimeSharedTNC = waitTimeSharedTNC; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java deleted file mode 100644 index 5b04c22..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UsualWorkSchoolLocationChoiceModel.java +++ /dev/null @@ -1,1003 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.MissingResourceException; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; -import org.jppf.client.JPPFClient; -import org.jppf.client.JPPFJob; -import org.jppf.node.protocol.DataProvider; -import org.jppf.node.protocol.JPPFTask; -import org.jppf.node.protocol.MemoryMapDataProvider; -import org.jppf.node.protocol.Task; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; - -public class UsualWorkSchoolLocationChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(UsualWorkSchoolLocationChoiceModel.class); - - private static final String USE_NEW_SOA_METHOD_PROPERTY_KEY = "uwsl.use.new.soa"; - - private static final String PROPERTIES_DC_SOA_WORK_SAMPLE_SIZE = "uwsl.work.soa.SampleSize"; - private static final String PROPERTIES_DC_SOA_SCHOOL_SAMPLE_SIZE = "uwsl.school.soa.SampleSize"; - private static final String PROPERTIES_UEC_USUAL_LOCATION = "uwsl.dc.uec.file"; - private static final String PROPERTIES_UEC_USUAL_LOCATION_NEW = "uwsl.dc2.uec.file"; - private static final String PROPERTIES_UEC_USUAL_LOCATION_SOA = "uwsl.soa.uec.file"; - - private static final String PROPERTIES_RESULTS_WORK_SCHOOL_LOCATION_CHOICE = "Results.UsualWorkAndSchoolLocationChoice"; - - private static final String PROPERTIES_WORK_SCHOOL_LOCATION_CHOICE_PACKET_SIZE = "distributed.task.packet.size"; - - private static final String WORK_SCHOOL_SEGMENTS_FILE_NAME = "workSchoolSegments.definitions"; - - private static final int[] WORK_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX = {1, 1, 1, - 1, 1, 1 }; - private static final int[] WORK_LOC_SEGMENT_TO_UEC_SHEET_INDEX = {2, 2, 2, - 2, 2, 2 }; - private static int PACKET_SIZE = 0; - - // TODO: see if we can eliminate the setup synchronization issues - - // otherwise the - // number of these small - // packets can be fine-tuned and set in properties file.. - - // The number of initialization packets are the number of "small" packets - // submited at the beginning of a - // distributed task to minimize synchronization issues that significantly - // slow - // down model object setup. - // It is assumed that after theses small packets have run, all the model - // objects - // will have been setup, - // and the task objects can process much bigger chuncks of households. - private static String PROPERTIES_NUM_INITIALIZATION_PACKETS = "number.initialization.packets"; - private static String PROPERTIES_INITIALIZATION_PACKET_SIZE = "initialization.packet.size"; - private static int NUM_INITIALIZATION_PACKETS = 0; - private static int INITIALIZATION_PACKET_SIZE = 0; - - private static final int NUM_WRITE_PACKETS = 1000; - - private String wsLocResultsFileName; - - private transient ResourceBundle resourceBundle; - - private MgraDataManager mgraManager; - private TazDataManager tdm; - - private int maxTaz; - - private MatrixDataServerIf ms; - private ModelStructure modelStructure; - private CtrampDmuFactoryIf dmuFactory; - - private String workLocUecFileName; - private String schoolLocUecFileName; - private String soaUecFileName; - private int soaWorkSampleSize; - private int soaSchoolSampleSize; - - private HashSet skipSegmentIndexSet; - - private BuildAccessibilities aggAcc; - private DestChoiceSize workerDcSizeObj; - private DestChoiceSize schoolDcSizeObj; - - private String restartModelString; - - private JPPFClient jppfClient = null; - - private boolean useNewSoaMethod; - - public UsualWorkSchoolLocationChoiceModel(ResourceBundle resourceBundle, - String restartModelString, JPPFClient jppfClient, ModelStructure modelStructure, - MatrixDataServerIf ms, CtrampDmuFactoryIf dmuFactory, BuildAccessibilities aggAcc) - { - - // set the local variables - this.resourceBundle = resourceBundle; - this.modelStructure = modelStructure; - this.dmuFactory = dmuFactory; - this.ms = ms; - this.jppfClient = jppfClient; - this.restartModelString = restartModelString; - this.aggAcc = aggAcc; - - try - { - PACKET_SIZE = Integer.parseInt(resourceBundle - .getString(PROPERTIES_WORK_SCHOOL_LOCATION_CHOICE_PACKET_SIZE)); - } catch (MissingResourceException e) - { - PACKET_SIZE = 0; - } - - try - { - NUM_INITIALIZATION_PACKETS = Integer.parseInt(resourceBundle - .getString(PROPERTIES_NUM_INITIALIZATION_PACKETS)); - } catch (MissingResourceException e) - { - NUM_INITIALIZATION_PACKETS = 0; - } - - try - { - INITIALIZATION_PACKET_SIZE = Integer.parseInt(resourceBundle - .getString(PROPERTIES_INITIALIZATION_PACKET_SIZE)); - } catch (MissingResourceException e) - { - INITIALIZATION_PACKET_SIZE = 0; - } - - try - { - wsLocResultsFileName = resourceBundle - .getString(PROPERTIES_RESULTS_WORK_SCHOOL_LOCATION_CHOICE); - } catch (MissingResourceException e) - { - wsLocResultsFileName = null; - } - - String uecPath = ResourceUtil.getProperty(resourceBundle, - CtrampApplication.PROPERTIES_UEC_PATH); - - // get the sample-of-alternatives sample size - soaWorkSampleSize = ResourceUtil.getIntegerProperty(resourceBundle, - PROPERTIES_DC_SOA_WORK_SAMPLE_SIZE); - soaSchoolSampleSize = ResourceUtil.getIntegerProperty(resourceBundle, - PROPERTIES_DC_SOA_SCHOOL_SAMPLE_SIZE); - - useNewSoaMethod = ResourceUtil.getBooleanProperty(resourceBundle, - USE_NEW_SOA_METHOD_PROPERTY_KEY); - - // locate the UECs for destination choice, sample of alts, and mode - // choice - String usualWorkLocationUecFileName; - String usualSchoolLocationUecFileName; - if (useNewSoaMethod) - { - usualWorkLocationUecFileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_UEC_USUAL_LOCATION_NEW); - usualSchoolLocationUecFileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_UEC_USUAL_LOCATION_NEW); - } else - { - usualWorkLocationUecFileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_UEC_USUAL_LOCATION); - usualSchoolLocationUecFileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_UEC_USUAL_LOCATION); - } - workLocUecFileName = uecPath + usualWorkLocationUecFileName; - schoolLocUecFileName = uecPath + usualSchoolLocationUecFileName; - - String usualLocationSoaUecFileName = ResourceUtil.getProperty(resourceBundle, - PROPERTIES_UEC_USUAL_LOCATION_SOA); - soaUecFileName = uecPath + usualLocationSoaUecFileName; - - mgraManager = MgraDataManager.getInstance(); - - } - - public void runWorkLocationChoiceModel(HouseholdDataManagerIf householdDataManager, - double[][] workerSizeTerms) - { - - HashMap propertyMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - - // get the map of size term segment values to names - HashMap occupValueIndexMap = aggAcc.getWorkOccupValueIndexMap(); - - HashMap workSegmentIndexNameMap = aggAcc.getWorkSegmentIndexNameMap(); - HashMap workSegmentNameIndexMap = aggAcc.getWorkSegmentNameIndexMap(); - - int maxShadowPriceIterations = Integer.parseInt(propertyMap - .get(DestChoiceSize.PROPERTIES_WORK_DC_SHADOW_NITER)); - - // create an object for calculating destination choice attraction size - // terms - // and managing shadow price calculations. - workerDcSizeObj = new DestChoiceSize(propertyMap, workSegmentIndexNameMap, - workSegmentNameIndexMap, workerSizeTerms, maxShadowPriceIterations); - - int[][] originLocationsByHomeMgra = householdDataManager - .getWorkersByHomeMgra(occupValueIndexMap); - - // balance the size variables - workerDcSizeObj.balanceSizeVariables(originLocationsByHomeMgra); - - if (PACKET_SIZE == 0) PACKET_SIZE = householdDataManager.getNumHouseholds(); - - int currentIter = 0; - String fileName = propertyMap - .get(CtrampApplication.PROPERTIES_WORK_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); - if (fileName != null) - { - if (fileName.length() > 2) - { - String projectDirectory = ResourceUtil.getProperty(resourceBundle, - CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - workerDcSizeObj.restoreShadowPricingInfo(projectDirectory + fileName); - int underScoreIndex = fileName.lastIndexOf('_'); - int dotIndex = fileName.lastIndexOf('.'); - currentIter = Integer.parseInt(fileName.substring(underScoreIndex + 1, dotIndex)); - currentIter++; - } - } - - // String restartFlag = propertyMap.get( - // CtrampApplication.PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); - // if ( restartFlag == null ) - // restartFlag = "none"; - // if ( restartFlag.equalsIgnoreCase("none") ) - // currentIter = 0; - - long initTime = System.currentTimeMillis(); - - // shadow pricing iterations - for (int iter = 0; iter < workerDcSizeObj.getMaxShadowPriceIterations(); iter++) - { - - logger.info("Work location choice shadow pricing iteration "+iter); - int innerLoop=0; - - while(true) { - ++innerLoop; - try - { - JPPFJob job = new JPPFJob(); - job.setName("Work Location Choice Job"); - - ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(householdDataManager - .getNumHouseholds()); - - DataProvider dataProvider = new MemoryMapDataProvider(); - dataProvider.setParameter("propertyMap", propertyMap); - dataProvider.setParameter("ms", ms); - dataProvider.setParameter("hhDataManager", householdDataManager); - dataProvider.setParameter("modelStructure", modelStructure); - dataProvider.setParameter("uecIndices", WORK_LOC_SEGMENT_TO_UEC_SHEET_INDEX); - dataProvider.setParameter("soaUecIndices", WORK_LOC_SOA_SEGMENT_TO_UEC_SHEET_INDEX); - dataProvider.setParameter("tourCategory", ModelStructure.MANDATORY_CATEGORY); - dataProvider.setParameter("dcSizeObj", workerDcSizeObj); - dataProvider.setParameter("dcUecFileName", workLocUecFileName); - dataProvider.setParameter("soaUecFileName", soaUecFileName); - dataProvider.setParameter("soaSampleSize", soaWorkSampleSize); - dataProvider.setParameter("dmuFactory", dmuFactory); - dataProvider.setParameter("restartModelString", restartModelString); - - job.setDataProvider(dataProvider); - - int startIndex = 0; - int endIndex = 0; - int taskIndex = 1; - WorkLocationChoiceTaskJppf myTask = null; - WorkLocationChoiceTaskJppfNew myTaskNew = null; - for (int[] startEndIndices : startEndTaskIndicesList) - { - startIndex = startEndIndices[0]; - endIndex = startEndIndices[1]; - - if (innerLoop == 1) { - resetWorkLocationResults(startIndex,endIndex,householdDataManager); - } - - //check if there's work to do (JEF) - if(workLocationResultsOK(startIndex, endIndex, householdDataManager)) - continue; - - if (useNewSoaMethod) - { - myTaskNew = new WorkLocationChoiceTaskJppfNew(taskIndex, startIndex, - endIndex, iter); - job.add(myTaskNew); - } else - { - myTask = new WorkLocationChoiceTaskJppf(taskIndex, startIndex, endIndex, - iter); - job.add(myTask); - } - taskIndex++; - } - //nothing to do, so break out of this loop (JEF) - if(job.getJobTasks().isEmpty()) { - logger.info("Work location choice tasks completed successfully after "+(innerLoop-1)+" loops"); - break; - }else { - logger.info("Work location choice tasks need to be executed in loop "+innerLoop); - } - logger.info("Usual work location choice model submitting tasks to jppf job"); - List> results = jppfClient.submitJob(job); - for (Task task : results) - { - //if (task.getException() != null) throw task.getException(); - //wu modefied for jppf 6.1.4 - if (task.getThrowable() != null) { - Throwable t = task.getThrowable(); - t.printStackTrace(); - } - try - { - String stringResult = (String) task.getResult(); - logger.info(stringResult); - System.out.println(stringResult); - } catch (Exception e) - { - logger.error("", e); - throw new RuntimeException(); - } - - } - - } catch (Exception e) - { - e.printStackTrace(); - } - } - - // sum the chosen destinations by purpose, dest zone and subzone for - // shadow pricing adjustment - int[][] finalModeledDestChoiceLocationsByDestMgra = householdDataManager - .getWorkToursByDestMgra(occupValueIndexMap); - - int[] numChosenDests = new int[workSegmentNameIndexMap.size()]; - - for (int i = 0; i < numChosenDests.length; i++) - { - for (int j = 1; j <= mgraManager.getMaxMgra(); j++) - numChosenDests[i] += finalModeledDestChoiceLocationsByDestMgra[i][j]; - } - - logger.info(String - .format("Usual work location choice tasks completed for shadow price iteration %d in %d seconds.", - iter, ((System.currentTimeMillis() - initTime) / 1000))); - logger.info(String.format("Chosen dests by segment:")); - double total = 0; - for (int i = 0; i < numChosenDests.length; i++) - { - String segmentString = workSegmentIndexNameMap.get(i); - logger.info(String.format("\t%-8d%-15s = %10d", i + 1, segmentString, - numChosenDests[i])); - total += numChosenDests[i]; - } - logger.info(String.format("\t%-8s%-15s = %10.0f", "total", "", total)); - - // apply the shadow price adjustments - workerDcSizeObj.reportMaxDiff(iter, finalModeledDestChoiceLocationsByDestMgra); - workerDcSizeObj.saveWorkMaxDiffValues(iter, finalModeledDestChoiceLocationsByDestMgra); - workerDcSizeObj.updateShadowPrices(finalModeledDestChoiceLocationsByDestMgra); - workerDcSizeObj.updateSizeVariables(); - workerDcSizeObj.updateShadowPricingInfo(currentIter, originLocationsByHomeMgra, - finalModeledDestChoiceLocationsByDestMgra, "work"); - - householdDataManager.setUwslRandomCount(currentIter); - - currentIter++; - - } // iter - - logger.info("Usual work location choices computed in " - + ((System.currentTimeMillis() - initTime) / 1000) + " seconds."); - - } - - public void runSchoolLocationChoiceModel(HouseholdDataManagerIf householdDataManager, - double[][] schoolSizeTerms, double[][] schoolFactors) - { - - HashMap propertyMap = ResourceUtil - .changeResourceBundleIntoHashMap(resourceBundle); - - // get the maps of segment names and indices for school location choice - // size - HashMap schoolSegmentIndexNameMap = aggAcc.getSchoolSegmentIndexNameMap(); - HashMap schoolSegmentNameIndexMap = aggAcc.getSchoolSegmentNameIndexMap(); - - int maxShadowPriceIterations = Integer.parseInt(propertyMap - .get(DestChoiceSize.PROPERTIES_SCHOOL_DC_SHADOW_NITER)); - - // create an object for calculating destination choice attraction size - // terms - // and managing shadow price calculations. - schoolDcSizeObj = new DestChoiceSize(propertyMap, schoolSegmentIndexNameMap, - schoolSegmentNameIndexMap, schoolSizeTerms, maxShadowPriceIterations); - - // get the set of segment indices for which shadow pricing should be - // skipped. - skipSegmentIndexSet = aggAcc.getNoShadowPriceSchoolSegmentIndexSet(); - schoolDcSizeObj.setNoShadowPriceSchoolSegmentIndices(skipSegmentIndexSet); - - // set the school segment external factors calculated for university - // segment in the method that called this one. - schoolDcSizeObj.setExternalFactors(schoolFactors); - - householdDataManager.setSchoolDistrictMappings(schoolSegmentNameIndexMap, - aggAcc.getMgraGsDistrict(), aggAcc.getMgraHsDistrict(), - aggAcc.getGsDistrictIndexMap(), aggAcc.getHsDistrictIndexMap()); - - int[][] originLocationsByHomeMgra = householdDataManager.getStudentsByHomeMgra(); - - // balance the size variables - schoolDcSizeObj.balanceSizeVariables(originLocationsByHomeMgra); - - if (PACKET_SIZE == 0) PACKET_SIZE = householdDataManager.getNumHouseholds(); - - int currentIter = 0; - String fileName = propertyMap - .get(CtrampApplication.PROPERTIES_SCHOOL_LOCATION_CHOICE_SHADOW_PRICE_INPUT_FILE); - if (fileName != null) - if (fileName.length() > 2) - { - { - String projectDirectory = ResourceUtil.getProperty(resourceBundle, - CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - schoolDcSizeObj.restoreShadowPricingInfo(projectDirectory + fileName); - int underScoreIndex = fileName.lastIndexOf('_'); - int dotIndex = fileName.lastIndexOf('.'); - currentIter = Integer.parseInt(fileName - .substring(underScoreIndex + 1, dotIndex)); - currentIter++; - } - } - // String restartFlag = propertyMap.get( - // CtrampApplication.PROPERTIES_RESTART_WITH_HOUSEHOLD_SERVER ); - // if ( restartFlag == null ) - // restartFlag = "none"; - // if ( restartFlag.equalsIgnoreCase("none") ) - // currentIter = 0; - - long initTime = System.currentTimeMillis(); - - // shadow pricing iterations - for (int iter = 0; iter < schoolDcSizeObj.getMaxShadowPriceIterations(); iter++) - { - - // logger.info( String.format( "Size of Household[] in bytes = %d.", - // householdDataManager.getBytesUsedByHouseholdArray() ) ); - logger.info("School location choice shadow pricing iteration "+iter); - int innerLoop=0; - - while(true) { - ++innerLoop; - - try - { - JPPFJob job = new JPPFJob(); - job.setName("School Location Choice Job"); - - ArrayList startEndTaskIndicesList = getTaskHouseholdRanges(householdDataManager - .getNumHouseholds()); - - DataProvider dataProvider = new MemoryMapDataProvider(); - dataProvider.setParameter("propertyMap", propertyMap); - dataProvider.setParameter("ms", ms); - dataProvider.setParameter("hhDataManager", householdDataManager); - dataProvider.setParameter("modelStructure", modelStructure); - dataProvider.setParameter("tourCategory", ModelStructure.MANDATORY_CATEGORY); - dataProvider.setParameter("dcSizeObj", schoolDcSizeObj); - dataProvider.setParameter("dcUecFileName", schoolLocUecFileName); - dataProvider.setParameter("soaUecFileName", soaUecFileName); - dataProvider.setParameter("soaSampleSize", soaSchoolSampleSize); - dataProvider.setParameter("dmuFactory", dmuFactory); - dataProvider.setParameter("restartModelString", restartModelString); - - job.setDataProvider(dataProvider); - - int startIndex = 0; - int endIndex = 0; - int taskIndex = 1; - SchoolLocationChoiceTaskJppf myTask = null; - SchoolLocationChoiceTaskJppfNew myTaskNew = null; - for (int[] startEndIndices : startEndTaskIndicesList) - { - startIndex = startEndIndices[0]; - endIndex = startEndIndices[1]; - - if (innerLoop == 1) { - resetSchoolLocationResults(startIndex,endIndex,householdDataManager); - } - //check if there's work to do (JEF) - if( schoolLocationResultsOK(startIndex, endIndex, householdDataManager)) - continue; - - if (useNewSoaMethod) - { - myTaskNew = new SchoolLocationChoiceTaskJppfNew(taskIndex, startIndex, - endIndex, iter); - job.add(myTaskNew); - } else - { - myTask = new SchoolLocationChoiceTaskJppf(taskIndex, startIndex, endIndex, - iter); - job.add(myTask); - } - taskIndex++; - } - - //nothing to do, so break out of this loop (JEF) - if(job.getJobTasks().isEmpty()) { - logger.info("School location choice tasks completed successfully after "+(innerLoop-1)+" loops"); - break; - }else { - logger.info("School location choice tasks need to be executed in loop "+innerLoop); - } - - List> results = jppfClient.submitJob(job); - for (Task task : results) - { - //if (task.getException() != null) throw task.getException(); - //wu modefied for jppf 6.1.4 - if (task.getThrowable() != null) { - Throwable t = task.getThrowable(); - t.printStackTrace(); - } - try - { - String stringResult = (String) task.getResult(); - logger.info(stringResult); - System.out.println(stringResult); - } catch (Exception e) - { - logger.error("", e); - throw new RuntimeException(); - } - - } - - } catch (Exception e) - { - e.printStackTrace(); - } - } - // sum the chosen destinations by purpose, dest zone and subzone for - // shadow pricing adjustment - int[][] finalModeledDestChoiceLocationsByDestMgra = householdDataManager - .getSchoolToursByDestMgra(); - - int[] numChosenDests = new int[schoolSegmentIndexNameMap.size()]; - - for (int i = 0; i < numChosenDests.length; i++) - { - for (int j = 1; j <= mgraManager.getMaxMgra(); j++) - numChosenDests[i] += finalModeledDestChoiceLocationsByDestMgra[i][j]; - } - - logger.info(String.format( - "Usual school location choice tasks completed for shadow price iteration %d.", - iter)); - logger.info(String.format("Chosen dests by segment:")); - double total = 0; - for (int i = 0; i < numChosenDests.length; i++) - { - String segmentString = schoolSegmentIndexNameMap.get(i); - logger.info(String.format("\t%-8d%-20s = %10d", i + 1, segmentString, - numChosenDests[i])); - total += numChosenDests[i]; - } - logger.info(String.format("\t%-8s%-20s = %10.0f", "total", "", total)); - - logger.info(String - .format("Usual school location choice tasks completed for shadow price iteration %d in %d seconds.", - iter, ((System.currentTimeMillis() - initTime) / 1000))); - - // apply the shadow price adjustments - schoolDcSizeObj.reportMaxDiff(iter, finalModeledDestChoiceLocationsByDestMgra); - schoolDcSizeObj - .saveSchoolMaxDiffValues(iter, finalModeledDestChoiceLocationsByDestMgra); - schoolDcSizeObj.updateShadowPrices(finalModeledDestChoiceLocationsByDestMgra); - schoolDcSizeObj.updateSizeVariables(); - schoolDcSizeObj.updateShadowPricingInfo(currentIter, originLocationsByHomeMgra, - finalModeledDestChoiceLocationsByDestMgra, "school"); - - householdDataManager.setUwslRandomCount(currentIter); - - currentIter++; - - } // iter - - logger.info("Usual school location choices computed in " - + ((System.currentTimeMillis() - initTime) / 1000) + " seconds."); - - } - - /** - * Loops through the households in the HouseholdDataManager, gets the - * households and persons and writes a row with detail on each of these in a - * file. - * - * @param householdDataManager - * is the object from which the array of household objects can be - * retrieved. - * @param projectDirectory - * is the root directory for the output file named - */ - public void saveResults(HouseholdDataManagerIf householdDataManager, String projectDirectory, - int globalIteration) - { - - HashMap workSegmentNameIndexMap = modelStructure - .getWorkSegmentNameIndexMap(); - HashMap schoolSegmentNameIndexMap = modelStructure - .getSchoolSegmentNameIndexMap(); - - FileWriter writer; - PrintWriter outStream = null; - - if (wsLocResultsFileName != null) - { - - // insert '_' and the global iteration number at end of filename or - // before '.' if there is a file extension in the name. - int dotIndex = wsLocResultsFileName.indexOf('.'); - if (dotIndex < 0) - { - wsLocResultsFileName = String - .format("%s_%d", wsLocResultsFileName, globalIteration); - } else - { - String base = wsLocResultsFileName.substring(0, dotIndex); - String extension = wsLocResultsFileName.substring(dotIndex); - wsLocResultsFileName = String.format("%s_%d%s", base, globalIteration, extension); - } - - wsLocResultsFileName = projectDirectory + wsLocResultsFileName; - - try - { - writer = new FileWriter(new File(wsLocResultsFileName)); - outStream = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format("Exception occurred opening wsLoc results file: %s.", - wsLocResultsFileName)); - throw new RuntimeException(e); - } - - // write header - outStream - .println("HHID,HomeMGRA,Income,PersonID,PersonNum,PersonType,PersonAge,EmploymentCategory,StudentCategory,WorkSegment,SchoolSegment,WorkLocation,WorkLocationDistance,WorkLocationLogsum,SchoolLocation,SchoolLocationDistance,SchoolLocationLogsum"); - - ArrayList startEndTaskIndicesList = getWriteHouseholdRanges(householdDataManager - .getNumHouseholds()); - - for (int[] startEndIndices : startEndTaskIndicesList) - { - - int startIndex = startEndIndices[0]; - int endIndex = startEndIndices[1]; - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - for (int i = 0; i < householdArray.length; ++i) - { - - Household household = householdArray[i]; - - int hhId = household.getHhId(); - int homeMgra = household.getHhMgra(); - int income = household.getIncomeInDollars(); - - Person[] personArray = household.getPersons(); - - for (int j = 1; j < personArray.length; ++j) - { - - Person person = personArray[j]; - - int personId = person.getPersonId(); - int personNum = person.getPersonNum(); - int personType = person.getPersonTypeNumber(); - int personAge = person.getAge(); - int employmentCategory = person.getPersonEmploymentCategoryIndex(); - int studentCategory = person.getPersonStudentCategoryIndex(); - - int schoolSegmentIndex = person.getSchoolLocationSegmentIndex(); - int workSegmentIndex = person.getWorkLocationSegmentIndex(); - - int workLocation = person.getWorkLocation(); - int schoolLocation = person.getUsualSchoolLocation(); - - // write data record - outStream.println(String.format( - "%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%.5e,%.5e,%d,%.5e,%.5e", hhId, - homeMgra, income, personId, personNum, personType, personAge, - employmentCategory, studentCategory, workSegmentIndex, - schoolSegmentIndex, workLocation, person.getWorkLocationDistance(), - person.getWorkLocationLogsum(), schoolLocation, - person.getSchoolLocationDistance(), - person.getSchoolLocationLogsum())); - - } - - } - - } - - outStream.close(); - - } - - // save the mappings between segment index and segment labels to a file - // for - // workers and students - String fileName = projectDirectory + WORK_SCHOOL_SEGMENTS_FILE_NAME; - - try - { - writer = new FileWriter(new File(fileName)); - outStream = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) - { - logger.fatal(String.format( - "Exception occurred opening work/school segment definitions file: %s.", - fileName)); - throw new RuntimeException(e); - } - - outStream - .println("Correspondence table for work location segment indices and work location segment names"); - outStream.println(String.format("%-15s %-20s", "Index", "Segment Name")); - - String[] names = new String[workSegmentNameIndexMap.size() + 1]; - for (String key : workSegmentNameIndexMap.keySet()) - { - int index = workSegmentNameIndexMap.get(key); - names[index] = key; - } - - for (int i = 0; i < names.length; i++) - { - if (names[i] != null) outStream.println(String.format("%-15d %-20s", i, names[i])); - } - - outStream.println(""); - outStream.println(""); - outStream.println(""); - - outStream - .println("Correspondence table for school location segment indices and school location segment names"); - outStream.println(String.format("%-15s %-20s", "Index", "Segment Name")); - - names = new String[schoolSegmentNameIndexMap.size() + 1]; - for (String key : schoolSegmentNameIndexMap.keySet()) - { - int index = schoolSegmentNameIndexMap.get(key); - names[index] = key; - } - - for (int i = 0; i < names.length; i++) - { - if (names[i] != null) outStream.println(String.format("%-15d %-20s", i, names[i])); - } - - outStream.println(""); - - outStream.close(); - - } - - private ArrayList getTaskHouseholdRanges(int numberOfHouseholds) - { - - ArrayList startEndIndexList = new ArrayList(); - - int numInitializationHouseholds = NUM_INITIALIZATION_PACKETS * INITIALIZATION_PACKET_SIZE; - - int startIndex = 0; - int endIndex = 0; - if (numInitializationHouseholds < numberOfHouseholds) - { - - while (endIndex < numInitializationHouseholds) - { - endIndex = startIndex + INITIALIZATION_PACKET_SIZE - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += INITIALIZATION_PACKET_SIZE; - } - - } - - while (endIndex < numberOfHouseholds - 1) - { - endIndex = startIndex + PACKET_SIZE - 1; - if (endIndex + PACKET_SIZE > numberOfHouseholds) endIndex = numberOfHouseholds - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += PACKET_SIZE; - } - - return startEndIndexList; - - } - - private ArrayList getWriteHouseholdRanges(int numberOfHouseholds) - { - - ArrayList startEndIndexList = new ArrayList(); - - int startIndex = 0; - int endIndex = 0; - - while (endIndex < numberOfHouseholds - 1) - { - endIndex = startIndex + NUM_WRITE_PACKETS - 1; - if (endIndex + NUM_WRITE_PACKETS > numberOfHouseholds) - endIndex = numberOfHouseholds - 1; - - int[] startEndIndices = new int[2]; - startEndIndices[0] = startIndex; - startEndIndices[1] = endIndex; - startEndIndexList.add(startEndIndices); - - startIndex += NUM_WRITE_PACKETS; - } - - return startEndIndexList; - - } - - - /** - * Returns true if work location results are OK, else returns false. - * - * @param startIndex starting range of households - * @param endIndex ending range of households - * @param HouseholdDataManagerIf household data manager - * @return true or false - */ - public boolean workLocationResultsOK(int startIndex, int endIndex, HouseholdDataManagerIf householdDataManager) { - - - // get the array of households - Household[] householdArray = householdDataManager.getHhArray(startIndex, endIndex); - - //iterate through hhs - for(Household thisHousehold:householdArray) { - - Person[] persons = thisHousehold.getPersons(); - - //iterate through persons - for(int k=1;k rbMap, String key) - { - boolean returnValue; - String value = rbMap.get(key); - if (value.equalsIgnoreCase("true") || value.equalsIgnoreCase("false")) - { - returnValue = Boolean.parseBoolean(value); - } else - { - logger.info("property file key: " + key + " = " + value - + " should be either 'true' or 'false'."); - throw new RuntimeException(); - } - - return returnValue; - } - - public static String getStringValueFromPropertyMap(HashMap rbMap, String key) - { - String returnValue = rbMap.get(key); - if (returnValue == null) returnValue = ""; - - return returnValue; - } - - public static String[] getStringArrayFromPropertyMap(HashMap rbMap, String key) { - String[] values = getStringValueFromPropertyMap(rbMap,key).split(","); - return values; - } - - - public static int getIntegerValueFromPropertyMap(HashMap rbMap, String key) - { - String value = rbMap.get(key); - if (value != null) - { - return Integer.parseInt(value); - } else - { - logger.info("property file key: " + key - + " missing. No integer value can be determined."); - throw new RuntimeException(); - } - } - - public static float getFloatValueFromPropertyMap(HashMap rbMap, String key) - { - String value = rbMap.get(key); - if (value != null) - { - return Float.parseFloat(value); - } else - { - logger.info("property file key: " + key - + " missing. No float value can be determined."); - throw new RuntimeException(); - } - } - - public static int[] getIntegerArrayFromPropertyMap(HashMap rbMap, String key) - { - - int[] returnArray; - String valueList = rbMap.get(key); - if (valueList != null) - { - - ArrayList valueSet = new ArrayList(); - - if (valueSet != null) - { - StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); - while (valueTokenizer.hasMoreTokens()) - { - String listValue = valueTokenizer.nextToken(); - int intValue = Integer.parseInt(listValue.trim()); - valueSet.add(intValue); - } - } - - returnArray = new int[valueSet.size()]; - int i = 0; - for (int v : valueSet) - returnArray[i++] = v; - - } else - { - logger.info("property file key: " + key - + " missing. No integer value can be determined."); - throw new RuntimeException(); - } - - return returnArray; - - } - - public static float[] getFloatArrayFromPropertyMap(HashMap rbMap, String key) - { - - float[] returnArray; - String valueList = rbMap.get(key); - if (valueList != null) - { - - ArrayList valueSet = new ArrayList(); - - StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); - while (valueTokenizer.hasMoreTokens()) - { - String listValue = valueTokenizer.nextToken(); - float floatValue = Float.parseFloat(listValue.trim()); - valueSet.add(floatValue); - } - - returnArray = new float[valueSet.size()]; - int i = 0; - for (float v : valueSet) - returnArray[i++] = v; - - } else - { - logger.info("property file key: " + key - + " missing. No float value can be determined."); - throw new RuntimeException(); - } - - return returnArray; - - } - - public static double[] getDoubleArrayFromPropertyMap(HashMap rbMap, String key) - { - - double[] returnArray; - String valueList = rbMap.get(key); - if (valueList != null) - { - - ArrayList valueSet = new ArrayList(); - - StringTokenizer valueTokenizer = new StringTokenizer(valueList, ","); - while (valueTokenizer.hasMoreTokens()) - { - String listValue = valueTokenizer.nextToken(); - double doubleValue = Double.parseDouble(listValue.trim()); - valueSet.add(doubleValue); - } - - returnArray = new double[valueSet.size()]; - int i = 0; - for (double v : valueSet) - returnArray[i++] = v; - - } else - { - logger.info("property file key: " + key - + " missing. No double value can be determined."); - throw new RuntimeException(); - } - - return returnArray; - - } - - /** - * - * @param cumProbabilities - * cumulative probabilities array - * @param entry - * target to search for in array - * @return the array index i where cumProbabilities[i] < entry and - * cumProbabilities[i-1] <= entry. - */ - public static int binarySearchDouble(double[] cumProbabilities, double entry) - { - - // lookup index for 0 <= entry < 1.0 in cumProbabilities - // cumProbabilities values are assumed to be in range: [0,1], and - // cumProbabilities[cumProbabilities.length-1] must equal 1.0 - - // if entry is outside the allowed range, return -1 - if (entry < 0 || entry >= 1.0) - { - System.out.println("entry = " + entry - + " is outside of allowable range for cumulative distribution [0,...,1.0)"); - return -1; - } - - // if cumProbabilities[cumProbabilities.length-1] is not equal to 1.0, - // return -1 - double epsilon = .0000001; - if (!(Math.abs(cumProbabilities[cumProbabilities.length - 1] - 1.0) < epsilon)) - { - System.out.println("cumProbabilities[cumProbabilities.length-1] = " - + cumProbabilities[cumProbabilities.length - 1] + " must equal 1.0"); - return -1; - } - - int hi = cumProbabilities.length; - int lo = 0; - int mid = (hi - lo) / 2; - - int safetyCount = 0; - - // if mid is 0, - if (mid == 0) - { - if (entry < cumProbabilities[0]) return 0; - else return 1; - } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) - { - return mid; - } - - while (true) - { - - if (entry < cumProbabilities[mid]) - { - hi = mid; - mid = (hi + lo) / 2; - } else - { - lo = mid; - mid = (hi + lo) / 2; - } - - // if mid is 0, - if (mid == 0) - { - if (entry < cumProbabilities[0]) return 0; - else return 1; - } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) - { - return mid; - } - - if (safetyCount++ > cumProbabilities.length) - { - logger.info("binary search stuck in the while loop"); - throw new RuntimeException("binary search stuck in the while loop"); - } - - } - - } - - /** - * - * @param cumProbabilities - * cumulative probabilities array - * @param numIndices - * are the number of probability values to consider in the - * cumulative probabilities array - * @param entry - * target to search for in array between indices 1 and numValues. - * @return the array index i where cumProbabilities[i] < entry and - * cumProbabilities[i-1] <= entry. - */ - public static int binarySearchDouble(double cumProbabilityLowerBound, - double[] cumProbabilities, int numIndices, double entry) - { - - // search for 0-based index i for cumProbabilities such that - // cumProbabilityLowerBound <= entry < cumProbabilities[0], i = 0; - // or - // cumProbabilities[i-1] <= entry < cumProbabilities[i], for i = - // 1,...numIndices-1; - - // if entry is outside the allowed range, return -1 - if (entry < cumProbabilityLowerBound || entry >= cumProbabilities[numIndices - 1]) - { - logger.info("entry = " + entry - + " is outside of allowable range of cumulative probabilities."); - logger.info("cumProbabilityLowerBound = " + cumProbabilityLowerBound - + ", cumProbabilities[numIndices-1] = " + cumProbabilities[numIndices - 1] - + ", numIndices = " + numIndices); - return -1; - } - - int hi = numIndices; - int lo = 0; - int mid = (hi - lo) / 2; - - int safetyCount = 0; - - // if mid is 0, - if (mid == 0) - { - if (entry < cumProbabilities[0]) return 0; - else return 1; - } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) - { - return mid; - } - - while (true) - { - - if (entry < cumProbabilities[mid]) - { - hi = mid; - mid = (hi + lo) / 2; - } else - { - lo = mid; - mid = (hi + lo) / 2; - } - - // if mid is 0, - if (mid == 0) - { - if (entry < cumProbabilities[0]) return 0; - else return 1; - } else if (entry < cumProbabilities[mid] && entry >= cumProbabilities[mid - 1]) - { - return mid; - } - - if (safetyCount++ > numIndices) - { - logger.info("binary search stuck in the while loop"); - throw new RuntimeException("binary search stuck in the while loop"); - } - - } - - } - - /** - * REad a tabledataset from a CSV file and return it. - * - * @param fileName - * @return - */ - public static TableDataSet readTableDataSet(String fileName) { - - - TableDataSet tableData; - - try - { - CSVFileReader csvFile = new CSVFileReader(); - tableData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - logger.fatal("Error trying to read table data set from csv file: "+ fileName); - throw new RuntimeException(e); - } - - return tableData; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java deleted file mode 100644 index a3f10a0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/UtilRmi.java +++ /dev/null @@ -1,137 +0,0 @@ -package org.sandag.abm.ctramp; - -import gnu.cajo.invoke.Remote; -import java.io.IOException; -import java.lang.reflect.InvocationTargetException; -import java.net.MalformedURLException; -import java.rmi.ConnectIOException; -import java.rmi.NotBoundException; -import java.rmi.RemoteException; -import org.apache.log4j.Logger; - -/** - * User: Jim Date: Jul 3, 2008 Time: 2:27:02 PM - * - * Utility class for applying remote methods of various types - * - */ - -public class UtilRmi - implements java.io.Serializable -{ - - private transient Logger logger = Logger.getLogger(UtilRmi.class); - private String connectString; - - private static int MAX_RETRY_COUNT = 100; - private static int MAX_RETRY_TIME = 1000; // milliseconds - - public UtilRmi(String connectString) - { - this.connectString = connectString; - } - - public Object method(String name, Object[] args) - { - - int connectExceptionCount = 0; - - Object itemObject = null; - Object returnObject = null; - - while (connectExceptionCount < MAX_RETRY_COUNT) - { - - try - { - itemObject = Remote.getItem(connectString); - break; - } catch (ConnectIOException e) - { - - try - { - Thread.currentThread().wait(MAX_RETRY_TIME); - } catch (InterruptedException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - connectExceptionCount++; - - } catch (RemoteException e) - { - logger.error("RemoteException exception making RMI method call: " + connectString - + "." + name + "().", e); - throw new RuntimeException(); - } catch (MalformedURLException e) - { - logger.error("MalformedURLException exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } catch (NotBoundException e) - { - logger.error("NotBoundException exception making RMI method call: " + connectString - + "." + name + "().", e); - throw new RuntimeException(); - } catch (IOException e) - { - logger.error("IOException exception making RMI method call: " + connectString + "." - + name + "().", e); - throw new RuntimeException(); - } catch (ClassNotFoundException e) - { - logger.error("ClassNotFoundException exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } catch (InstantiationException e) - { - logger.error("InstantiationException exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } catch (IllegalAccessException e) - { - logger.error("IllegalAccessException exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } catch (UnsatisfiedLinkError e) - { - logger.error("UnsatisfiedLinkError exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } - - } - - if (connectExceptionCount > 0) - { - logger.warn("UtilRmi.method() timed out " + connectExceptionCount - + "times connecting to: " + connectString + "." + name + "()."); - } - if (connectExceptionCount == MAX_RETRY_COUNT) - { - logger.error("UtilRmi.method() connection was never made."); - throw new RuntimeException(); - } - - try - { - returnObject = Remote.invoke(itemObject, name, args); - } catch (InvocationTargetException e) - { - logger.error("InvocationTargetException exception making RMI method call: " - + connectString + "." + name + "().", e); - throw new RuntimeException(); - } catch (Exception e) - { - logger.error("Exception exception making RMI method call: " + connectString + "." - + name + "().", e); - throw new RuntimeException(); - } - - return returnObject; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java deleted file mode 100644 index 18af3b5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceModel.java +++ /dev/null @@ -1,692 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import java.util.Random; -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.modechoice.MgraDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class WorkLocationChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(MandatoryDestChoiceModel.class); - private transient Logger dcManLogger = Logger.getLogger("tourDcMan"); - - // this constant used as a dimension for saving distance and logsums for - // alternatives in samples - private static final int MAXIMUM_SOA_ALTS_FOR_ANY_MODEL = 200; - - private static final int DC_DATA_SHEET = 0; - private static final int DC_WORK_AT_HOME_SHEET = 1; - - // private TazDataManager tazs; - private MgraDataManager mgraManager; - private DestChoiceSize dcSizeObj; - - private DestChoiceTwoStageModelDMU dcTwoStageDmuObject; - - private DestChoiceTwoStageModel dcTwoStageModelObject; - private TourModeChoiceModel mcModel; - - private String[] segmentNameList; - private HashMap workOccupValueSegmentIndexMap; - - private int[] dcModelIndices; - - // A ChoiceModelApplication object and modeAltsAvailable[] is needed for - // each purpose - private ChoiceModelApplication[] locationChoiceModels; - private ChoiceModelApplication locationChoiceModel; - private ChoiceModelApplication worksAtHomeModel; - - private boolean[] dcModelAltsAvailable; - private int[] dcModelAltsSample; - private int[] dcModelSampleValues; - - private int[] uecSheetIndices; - - int origMgra; - - private int modelIndex; - - private double[] sampleAlternativeDistances; - private double[] sampleAlternativeLogsums; - - private BuildAccessibilities aggAcc; - - private double[] mgraDistanceArray; - - private int soaSampleSize; - - private long soaRunTime; - - public WorkLocationChoiceModel(int index, HashMap propertyMap, - DestChoiceSize dcSizeObj, BuildAccessibilities aggAcc, String dcUecFileName, - String soaUecFile, int soaSampleSize, String modeChoiceUecFile, - CtrampDmuFactoryIf dmuFactory, TourModeChoiceModel mcModel, double[][][] workSizeProbs, - double[][][] workTazDistProbs) - { - - this.aggAcc = aggAcc; - this.dcSizeObj = dcSizeObj; - this.mcModel = mcModel; - this.soaSampleSize = soaSampleSize; - - modelIndex = index; - - mgraManager = MgraDataManager.getInstance(); - - dcTwoStageDmuObject = dmuFactory.getDestChoiceSoaTwoStageDMU(); - dcTwoStageDmuObject.setAggAcc(this.aggAcc); - - dcTwoStageModelObject = new DestChoiceTwoStageModel(propertyMap, soaSampleSize); - dcTwoStageModelObject.setTazDistProbs(workTazDistProbs); - dcTwoStageModelObject.setMgraSizeProbs(workSizeProbs); - - sampleAlternativeDistances = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - sampleAlternativeLogsums = new double[MAXIMUM_SOA_ALTS_FOR_ANY_MODEL]; - - workOccupValueSegmentIndexMap = aggAcc.getWorkOccupValueIndexMap(); - - } - - public void setupWorkSegments(int[] myUecSheetIndices, int[] mySoaUecSheetIndices) - { - uecSheetIndices = myUecSheetIndices; - segmentNameList = aggAcc.getWorkSegmentNameList(); - } - - public void setupDestChoiceModelArrays(HashMap propertyMap, - String dcUecFileName, String soaUecFile, int soaSampleSize) - { - - // create the works-at-home ChoiceModelApplication object - worksAtHomeModel = new ChoiceModelApplication(dcUecFileName, DC_WORK_AT_HOME_SHEET, - DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); - - // create a lookup array to map purpose index to model index - dcModelIndices = new int[uecSheetIndices.length]; - - // get a set of unique model sheet numbers so that we can create - // ChoiceModelApplication objects once for each model sheet used - // also create a HashMap to relate size segment index to SOA Model - // objects - HashMap modelIndexMap = new HashMap(); - int dcModelIndex = 0; - int dcSegmentIndex = 0; - for (int uecIndex : uecSheetIndices) - { - // if the uec sheet for the model segment is not in the map, add it, - // otherwise, get it from the map - if (!modelIndexMap.containsKey(uecIndex)) - { - modelIndexMap.put(uecIndex, dcModelIndex); - dcModelIndices[dcSegmentIndex] = dcModelIndex++; - } else - { - dcModelIndices[dcSegmentIndex] = modelIndexMap.get(uecIndex); - } - - dcSegmentIndex++; - } - // the value of dcModelIndex is the number of ChoiceModelApplication - // objects to create - // the modelIndexMap keys are the uec sheets to use in building - // ChoiceModelApplication objects - - locationChoiceModels = new ChoiceModelApplication[modelIndexMap.size()]; - - int i = 0; - for (int uecIndex : modelIndexMap.keySet()) - { - - int modelIndex = -1; - try - { - modelIndex = modelIndexMap.get(uecIndex); - locationChoiceModels[modelIndex] = new ChoiceModelApplication(dcUecFileName, - uecIndex, DC_DATA_SHEET, propertyMap, (VariableTable) dcTwoStageDmuObject); - } catch (RuntimeException e) - { - logger.error(String - .format("exception caught setting up DC ChoiceModelApplication[%d] for modelIndex=%d of %d models", - i, modelIndex, modelIndexMap.size())); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - } - - dcModelAltsAvailable = new boolean[soaSampleSize + 1]; - dcModelAltsSample = new int[soaSampleSize + 1]; - dcModelSampleValues = new int[soaSampleSize]; - - mgraDistanceArray = new double[mgraManager.getMaxMgra() + 1]; - - } - - public boolean applyWorkLocationChoice(Household hh) - { - boolean result=true; - - if (hh.getDebugChoiceModels()) - { - String label = String.format("Pre Work Location Choice HHId=%d Object", hh.getHhId()); - hh.logHouseholdObject(label, dcManLogger); - } - - // declare these variables here so their values can be logged if a - // RuntimeException occurs. - int i = -1; - int occupSegmentIndex = -1; - int occup = -1; - String occupSegmentName = ""; - - int homeMgra = hh.getHhMgra(); - Person[] persons = hh.getPersons(); - - int tourNum = 0; - for (i = 1; i < persons.length; i++) - { - - Person p = persons[i]; - - // skip person if they are not a worker - if (p.getPersonIsWorker() != 1) - { - p.setWorkLocationSegmentIndex(-1); - p.setWorkLocation(0); - p.setWorkLocDistance(0); - p.setWorkLocLogsum(-999); - continue; - } - - // skip person if their work at home choice was work in the home - // (alternative 2 in choice model) - int worksAtHomeChoice = selectWorksAtHomeChoice(dcTwoStageDmuObject, hh, p); - if (worksAtHomeChoice == ModelStructure.WORKS_AT_HOME_ALTERNATUVE_INDEX) - { - p.setWorkLocationSegmentIndex(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); - p.setWorkLocation(ModelStructure.WORKS_AT_HOME_LOCATION_INDICATOR); - p.setWorkLocDistance(0); - p.setWorkLocLogsum(-999); - continue; - } - - // save person information in decision maker label, and log person - // object - if (hh.getDebugChoiceModels()) - { - String decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", p - .getHouseholdObject().getHhId(), p.getPersonNum(), p.getPersonType()); - hh.logPersonObject(decisionMakerLabel, dcManLogger, p); - } - - double[] results = null; - int modelIndex = 0; - try - { - - origMgra = homeMgra; - - occup = p.getPersPecasOccup(); - occupSegmentIndex = workOccupValueSegmentIndexMap.get(occup); - occupSegmentName = segmentNameList[occupSegmentIndex]; - - p.setWorkLocationSegmentIndex(occupSegmentIndex); - - // update the DC dmuObject for this person - dcTwoStageDmuObject.setHouseholdObject(hh); - dcTwoStageDmuObject.setPersonObject(p); - dcTwoStageDmuObject.setDmuIndexValues(hh.getHhId(), homeMgra, origMgra, 0); - - double[] homeMgraSizeArray = dcSizeObj.getDcSizeArray()[occupSegmentIndex]; - mcModel.getAnmSkimCalculator().getAmPkSkimDistancesFromMgra(homeMgra, - mgraDistanceArray); - - // set size array for the tour segment and distance array from - // the home mgra to all destinaion mgras. - dcTwoStageDmuObject.setMgraSizeArray(homeMgraSizeArray); - dcTwoStageDmuObject.setMgraDistanceArray(mgraDistanceArray); - - modelIndex = dcModelIndices[occupSegmentIndex]; - locationChoiceModel = locationChoiceModels[modelIndex]; - - // get the work location alternative chosen from the sample - results = selectLocationFromSampleOfAlternatives("work", -1, p, occupSegmentName, - occupSegmentIndex, tourNum++, homeMgraSizeArray, mgraDistanceArray); - - soaRunTime += dcTwoStageModelObject.getSoaRunTime(); - - } catch (RuntimeException e) - { - logger.fatal(String - .format("Exception caught in dcModel selecting location for i=%d, hh.hhid=%d, person i=%d, in work location choice, modelIndex=%d, occup=%d, segmentIndex=%d, segmentName=%s", - i, hh.getHhId(), i, modelIndex, occup, occupSegmentIndex, - occupSegmentName)); - logger.fatal("Exception caught:", e); - logger.fatal("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(); - } - - p.setWorkLocation((int) results[0]); - p.setWorkLocDistance((float) results[1]); - p.setWorkLocLogsum((float) results[2]); - - if(p.getWorkLocationSegmentIndex()>-1 && p.getWorkLocation()==0){ - logger.error("***********************************************************************************************************************************"); - logger.error("!!!!!!!Worker in worksegmetn "+p.getWorkLocationSegmentIndex()+" can't find a work location!!!!! RESTART Work location choice!!!!"); - result=false; - } - - } - return result; - - } - /** - * - * @return an array with chosen mgra, distance to chosen mgra, and logsum to - * chosen mgra. - */ - private double[] selectLocationFromSampleOfAlternatives(String segmentType, - int segmentTypeIndex, Person person, String segmentName, int sizeSegmentIndex, - int tourNum, double[] homeMgraSizeArray, double[] homeMgraDistanceArray) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcManLogger; - - Household household = person.getHouseholdObject(); - - // get sample of locations and correction factors for sample using the - // alternate method - // for work location, the sizeSegmentType INdex and sizeSegmentIndex are - // the same values. - dcTwoStageModelObject.chooseSample(household.getHhTaz(), sizeSegmentIndex, - sizeSegmentIndex, soaSampleSize, household.getHhRandom(), - household.getDebugChoiceModels()); - int[] finalSample = dcTwoStageModelObject.getUniqueSampleMgras(); - double[] sampleCorrectionFactors = dcTwoStageModelObject - .getUniqueSampleMgraCorrectionFactors(); - int numUniqueAlts = dcTwoStageModelObject.getNumberofUniqueMgrasInSample(); - - Arrays.fill(dcModelAltsAvailable, false); - Arrays.fill(dcModelAltsSample, 0); - Arrays.fill(dcModelSampleValues, 999999); - - // set sample of alternatives correction factors used in destination - // choice utility for the sampled alternatives. - dcTwoStageDmuObject.setDcSoaCorrections(sampleCorrectionFactors); - - // for the destination mgras in the sample, compute mc logsums and save - // in dmuObject. - // also save correction factor and set availability and sample value for - // the - // sample alternative to true. 1, respectively. - for (int i = 0; i < numUniqueAlts; i++) - { - - int destMgra = finalSample[i]; - dcModelSampleValues[i] = finalSample[i]; - - // set logsum value in DC dmuObject for the logsum index, sampled - // zone and subzone. - double logsum = getModeChoiceLogsum(household, person, destMgra, segmentTypeIndex); - dcTwoStageDmuObject.setMcLogsum(i, logsum); - - sampleAlternativeLogsums[i] = logsum; - sampleAlternativeDistances[i] = homeMgraDistanceArray[finalSample[i]]; - - // set availaibility and sample values for the purpose, dcAlt. - dcModelAltsAvailable[i + 1] = true; - dcModelAltsSample[i + 1] = 1; - - } - - dcTwoStageDmuObject.setSampleArray(dcModelSampleValues); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format( - "Usual %s Location Choice Model for: Segment=%s", segmentType, segmentName); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s, TourNum=%d", - person.getHouseholdObject().getHhId(), person.getPersonNum(), - person.getPersonType(), tourNum); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Usual " + segmentType + " Location Choice Model for: Segment=" - + segmentName + ", Person Num: " + person.getPersonNum() + ", Person Type: " - + person.getPersonType() + ", TourNum=" + tourNum); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - locationChoiceModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - locationChoiceModel.computeUtilities(dcTwoStageDmuObject, - dcTwoStageDmuObject.getDmuIndexValues(), dcModelAltsAvailable, dcModelAltsSample); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (locationChoiceModel.getAvailabilityCount() > 0) - { - try - { - chosen = locationChoiceModel.getChoiceResult(rn); - } catch (Exception e) - { - } - } else - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available %s destination choice alternatives to choose from in choiceModelApplication.", - dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject - .getPersonObject().getPersonNum(), segmentName)); - } - - if (household.getDebugChoiceModels() || chosen <= 0) - { - - double[] utilities = locationChoiceModel.getUtilities(); - double[] probabilities = locationChoiceModel.getProbabilities(); - boolean[] availabilities = locationChoiceModel.getAvailabilities(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < finalSample.length; j++) - { - int alt = finalSample[j]; - cumProb += probabilities[j]; - String altString = String.format("j=%d, mgra=%d", j, alt); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, mgra=%d", chosen - 1, finalSample[chosen - 1]); - modelLogger.info(String.format("Choice: %s with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - locationChoiceModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - locationChoiceModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - - // write UEC calculation results to separate model specific log file - locationChoiceModel.logUECResults(modelLogger, loggingHeader); - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, workSegment=%d, no available %s destination choice alternatives to choose from in ChoiceModelApplication.", - dcTwoStageDmuObject.getHouseholdObject().getHhId(), - dcTwoStageDmuObject.getPersonObject().getPersonNum(), segmentName)); - throw new RuntimeException(); - } - - } - - double[] returnArray = new double[3]; - - returnArray[0] = finalSample[chosen - 1]; - returnArray[1] = sampleAlternativeDistances[chosen - 1]; - returnArray[2] = sampleAlternativeLogsums[chosen - 1]; - - return returnArray; - - } - - private int selectWorksAtHomeChoice(DestChoiceTwoStageModelDMU dcTwoStageDmuObject, - Household household, Person person) - { - - // set tour origin taz/subzone and start/end times for calculating mode - // choice logsum - Logger modelLogger = dcManLogger; - - dcTwoStageDmuObject.setHouseholdObject(household); - dcTwoStageDmuObject.setPersonObject(person); - dcTwoStageDmuObject.setDmuIndexValues(household.getHhId(), household.getHhMgra(), origMgra, - 0); - - double accessibility = aggAcc.getAccessibilitiesTableObject().getAggregateAccessibility( - "totEmp", household.getHhMgra()); - dcTwoStageDmuObject.setWorkAccessibility(accessibility); - - // log headers to traceLogger if the person making the destination - // choice is - // from a household requesting trace information - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - if (household.getDebugChoiceModels()) - { - - // null tour means the DC is a mandatory usual location choice - choiceModelDescription = String.format("Usual Work Location Is At Home Choice Model"); - decisionMakerLabel = String.format("HH=%d, PersonNum=%d, PersonType=%s", person - .getHouseholdObject().getHhId(), person.getPersonNum(), person.getPersonType()); - - modelLogger.info(" "); - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info("Usual Work Location Is At Home Choice Model: Person Num: " - + person.getPersonNum() + ", Person Type: " + person.getPersonType()); - - loggingHeader = String.format("%s for %s", choiceModelDescription, decisionMakerLabel); - - worksAtHomeModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, - decisionMakerLabel); - - } - - // compute destination choice proportions and choose alternative - float logsum = (float) worksAtHomeModel.computeUtilities(dcTwoStageDmuObject, - dcTwoStageDmuObject.getDmuIndexValues()); - person.setWorksFromHomeLogsum(logsum); - - Random hhRandom = household.getHhRandom(); - int randomCount = household.getHhRandomCount(); - double rn = hhRandom.nextDouble(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen = -1; - if (worksAtHomeModel.getAvailabilityCount() > 0) - { - chosen = worksAtHomeModel.getChoiceResult(rn); - } - - // write choice model alternative info to log file - if (household.getDebugChoiceModels() || chosen < 0) - { - - double[] utilities = worksAtHomeModel.getUtilities(); - double[] probabilities = worksAtHomeModel.getProbabilities(); - boolean[] availabilities = worksAtHomeModel.getAvailabilities(); - - String[] altNames = worksAtHomeModel.getAlternativeNames(); - - String personTypeString = person.getPersonType(); - int personNum = person.getPersonNum(); - - modelLogger.info("Person num: " + personNum + ", Person type: " + personTypeString); - modelLogger - .info("Alternative Availability Utility Probability CumProb"); - modelLogger - .info("--------------------- -------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int j = 0; j < utilities.length; j++) - { - cumProb += probabilities[j]; - String altString = String.format("%d, %s", j + 1, altNames[j]); - modelLogger.info(String.format("%-21s%15s%18.6e%18.6e%18.6e", altString, - availabilities[j + 1], utilities[j], probabilities[j], cumProb)); - } - - modelLogger.info(" "); - String altString = String.format("j=%d, alt=%s", chosen, - (chosen < 0 ? "N/A, no available alternatives" : altNames[chosen - 1])); - modelLogger.info(String.format("Choice: %s, with rn=%.8f, randomCount=%d", altString, - rn, randomCount)); - - modelLogger - .info("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++"); - modelLogger.info(" "); - - worksAtHomeModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - worksAtHomeModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, - chosen); - - // write UEC calculation results to separate model specific log file - worksAtHomeModel.logUECResults(modelLogger, loggingHeader); - - } - - if (chosen < 0) - { - logger.error(String - .format("Exception caught for HHID=%d, PersonNum=%d, no available works at home alternatives to choose from in choiceModelApplication.", - dcTwoStageDmuObject.getHouseholdObject().getHhId(), dcTwoStageDmuObject - .getPersonObject().getPersonNum())); - throw new RuntimeException(); - } - - return chosen; - - } - - private double getModeChoiceLogsum(Household household, Person person, int sampleDestMgra, - int segmentTypeIndex) - { - - int purposeIndex = 0; - String purpose = ""; - if (segmentTypeIndex < 0) - { - purposeIndex = ModelStructure.WORK_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.WORK_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.PRESCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.GRADE_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.HIGH_SCHOOL_ALT_INDEX) - { - purposeIndex = ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_TYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } else if (segmentTypeIndex == BuildAccessibilities.UNIV_NONTYPICAL_ALT_INDEX) - { - purposeIndex = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX; - purpose = ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME; - } - - // create a temporary tour to use to calculate mode choice logsum - Tour mcLogsumTour = new Tour(person, 0, purposeIndex); - mcLogsumTour.setTourPurpose(purpose); - mcLogsumTour.setTourOrigMgra(household.getHhMgra()); - mcLogsumTour.setTourDestMgra(sampleDestMgra); - mcLogsumTour.setTourDepartPeriod(Person.DEFAULT_MANDATORY_START_PERIOD); - mcLogsumTour.setTourArrivePeriod(Person.DEFAULT_MANDATORY_END_PERIOD); - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - - if (household.getDebugChoiceModels()) - { - dcManLogger.info(""); - dcManLogger.info(""); - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - household.logPersonObject(choiceModelDescription + ", " + decisionMakerLabel, - dcManLogger, person); - } - - double logsum = -1; - try - { - logsum = mcModel.getModeChoiceLogsum(household, person, mcLogsumTour, dcManLogger, - choiceModelDescription, decisionMakerLabel); - } catch (Exception e) - { - choiceModelDescription = "location choice logsum for segmentTypeIndex=" - + segmentTypeIndex + ", temp tour PurposeIndex=" + purposeIndex; - decisionMakerLabel = "HHID: " + household.getHhId() + ", PersNum: " - + person.getPersonNum(); - logger.fatal("exception caught calculating ModeChoiceLogsum for usual work/school location choice."); - logger.fatal("choiceModelDescription = " + choiceModelDescription); - logger.fatal("decisionMakerLabel = " + decisionMakerLabel); - e.printStackTrace(); - System.exit(-1); - } - - return logsum; - } - - public int getModelIndex() - { - return modelIndex; - } - - public void setDcSizeObject(DestChoiceSize dcSizeObj) - { - this.dcSizeObj = dcSizeObj; - } - - public long getSoaRunTime() - { - return soaRunTime; - } - - public void resetSoaRunTime() - { - soaRunTime = 0; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java b/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java deleted file mode 100644 index 99ccb09..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/ctramp/WorkLocationChoiceTaskJppf.java +++ /dev/null @@ -1,246 +0,0 @@ -package org.sandag.abm.ctramp; - -import java.net.UnknownHostException; -import java.util.Date; -import java.util.HashMap; -import org.jppf.node.protocol.AbstractTask; -import org.jppf.node.protocol.DataProvider; -import com.pb.common.calculator.MatrixDataServerIf; - -import nl.tudelft.simulation.logger.Logger; - -public class WorkLocationChoiceTaskJppf - extends AbstractTask -{ - - private static String VERSION = "Task.1.0.3"; - - private transient HashMap propertyMap; - private transient MatrixDataServerIf ms; - private transient ModelStructure modelStructure; - private transient HouseholdDataManagerIf hhDataManager; - private transient String tourCategory; - private transient DestChoiceSize dcSizeObj; - private transient int[] uecIndices; - private transient int[] soaUecIndices; - private transient String dcUecFileName; - private transient String soaUecFileName; - private transient int soaSampleSize; - private transient CtrampDmuFactoryIf dmuFactory; - private transient String restartModelString; - - private int iteration; - private int startIndex; - private int endIndex; - private int taskIndex = -1; - - public WorkLocationChoiceTaskJppf(int taskIndex, int startIndex, int endIndex, int iteration) - { - this.startIndex = startIndex; - this.endIndex = endIndex; - this.taskIndex = taskIndex; - this.iteration = iteration; - } - - public void run() - { - - String start = (new Date()).toString(); - long startTime = System.currentTimeMillis(); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " - + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - try - { - DataProvider dataProvider = getDataProvider(); - - this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); - this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); - this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); - this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); - this.uecIndices = (int[]) dataProvider.getParameter("uecIndices"); - this.soaUecIndices = (int[]) dataProvider.getParameter("soaUecIndices"); - this.tourCategory = (String) dataProvider.getParameter("tourCategory"); - this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); - this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); - this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); - this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); - this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); - this.restartModelString = (String) dataProvider.getParameter("restartModelString"); - - } catch (Exception e) - { - e.printStackTrace(); - throw new RuntimeException(e); - } - - // get the factory object used to create and recycle dcModel objects. - DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); - - // one of tasks needs to initialize the manager object by passing - // attributes - // needed to create a destination choice model object. - modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, - soaSampleSize, dmuFactory, restartModelString); - - // get a dcModel object from manager, which either creates one or - // returns one - // for re-use. - MandatoryDestChoiceModel dcModel = modelManager.getDcWorkModelObject(taskIndex, iteration, - dcSizeObj, uecIndices, soaUecIndices); - - // logger.info( String.format( - // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, - // taskIndex, - // threadName, startIndex, endIndex ) ); - System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", - new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); - - long setup1 = (System.currentTimeMillis() - startTime) / 1000; - - Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); - - long setup2 = (System.currentTimeMillis() - startTime) / 1000; - // logger.info( String.format( - // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", - // taskIndex, startIndex, endIndex, - // threadName, setup1, setup2 ) ); - System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", - new Date(), taskIndex, startIndex, endIndex, threadName)); - - int i = -1; - try - { - - boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, - HouseholdDataManager.DEBUG_HHS_ONLY_KEY); - - for (i = 0; i < householdArray.length; i++) - { - // for debugging only - process only household objects specified - // for debugging, if property key was set to true - if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) continue; - - dcModel.applyWorkLocationChoice(householdArray[i]); - } - - boolean worked=false; - int tries=0; - do{ - ++tries; - try{ - hhDataManager.setHhArray(householdArray, startIndex); - worked=true; - }catch(Exception e) { - System.out.println("Error trying to set households in hh manager for start index "+startIndex+" (tried "+tries+" times"); - if(tries<1000) - System.out.println("Trying again!"); - } - }while(!worked && (tries<1000)); - - //check to make sure hh array got set in hhDataManager - boolean allHouseholdsAreSame = false; - while(!allHouseholdsAreSame) { - Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); - for(int j = 0; j< householdArrayRemote.length;++j) { - - Household remoteHousehold = householdArrayRemote[j]; - Household localHousehold = householdArray[j]; - - allHouseholdsAreSame = checkIfSameWorkLocationResults(remoteHousehold, localHousehold); - - if(!allHouseholdsAreSame) - break; - } - if(!allHouseholdsAreSame) { - System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with work location choice results; updating"); - hhDataManager.setHhArray(householdArray, startIndex); - - } - } - - - } catch (Exception e) - { - if (i >= 0 && i < householdArray.length) System.out - .println(String - .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", - taskIndex, i, householdArray[i].getHhId(), startIndex)); - else System.out.println(String.format( - "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", - taskIndex, i, startIndex)); - System.out.println("Exception caught:"); - e.printStackTrace(); - System.out.println("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; - long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; - // logger.info( String.format( - // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, - // threadName, getHhs, processHhs ) ); - System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, - threadName)); - - long total = (System.currentTimeMillis() - startTime) / 1000; - String resultString = String - .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", - threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, - setup2, getHhs, processHhs, total); - // logger.info( resultString ); - setResult(resultString); - - modelManager.returnDcWorkModelObject(dcModel, taskIndex, startIndex, endIndex); - - } - - /** - * Returns true if work location results are the same, else returns false. - * - * @param thisHousehold - * @param thatHousehold - * @return true or false - */ - public boolean checkIfSameWorkLocationResults(Household thisHousehold, Household thatHousehold) { - - Person[] thisPersons = thisHousehold.getPersons(); - Person[] thatPersons = thatHousehold.getPersons(); - - if(thisPersons.length!=thatPersons.length) - return false; - - for(int k=1;k -{ - - private static String VERSION = "Task.1.0.3"; - - private transient HashMap propertyMap; - private transient MatrixDataServerIf ms; - private transient ModelStructure modelStructure; - private transient HouseholdDataManagerIf hhDataManager; - private transient String tourCategory; - private transient DestChoiceSize dcSizeObj; - private transient int[] uecIndices; - private transient int[] soaUecIndices; - private transient String dcUecFileName; - private transient String soaUecFileName; - private transient int soaSampleSize; - private transient CtrampDmuFactoryIf dmuFactory; - private transient String restartModelString; - - private int iteration; - private int startIndex; - private int endIndex; - private int taskIndex = -1; - - public WorkLocationChoiceTaskJppfNew(int taskIndex, int startIndex, int endIndex, int iteration) - { - this.startIndex = startIndex; - this.endIndex = endIndex; - this.taskIndex = taskIndex; - this.iteration = iteration; - } - - public void run() - { - - String start = (new Date()).toString(); - long startTime = System.currentTimeMillis(); - - String threadName = null; - try - { - threadName = "[" + java.net.InetAddress.getLocalHost().getHostName() + "] " - + Thread.currentThread().getName(); - } catch (UnknownHostException e1) - { - // TODO Auto-generated catch block - e1.printStackTrace(); - } - - try - { - DataProvider dataProvider = getDataProvider(); - - this.propertyMap = (HashMap) dataProvider.getParameter("propertyMap"); - this.ms = (MatrixDataServerIf) dataProvider.getParameter("ms"); - this.hhDataManager = (HouseholdDataManagerIf) dataProvider.getParameter("hhDataManager"); - this.modelStructure = (ModelStructure) dataProvider.getParameter("modelStructure"); - this.uecIndices = (int[]) dataProvider.getParameter("uecIndices"); - this.soaUecIndices = (int[]) dataProvider.getParameter("soaUecIndices"); - this.tourCategory = (String) dataProvider.getParameter("tourCategory"); - this.dcSizeObj = (DestChoiceSize) dataProvider.getParameter("dcSizeObj"); - this.dcUecFileName = (String) dataProvider.getParameter("dcUecFileName"); - this.soaUecFileName = (String) dataProvider.getParameter("soaUecFileName"); - this.soaSampleSize = (Integer) dataProvider.getParameter("soaSampleSize"); - this.dmuFactory = (CtrampDmuFactoryIf) dataProvider.getParameter("dmuFactory"); - this.restartModelString = (String) dataProvider.getParameter("restartModelString"); - - } catch (Exception e) - { - e.printStackTrace(); - } - - // get the factory object used to create and recycle dcModel objects. - DestChoiceModelManager modelManager = DestChoiceModelManager.getInstance(); - - // one of tasks needs to initialize the manager object by passing - // attributes - // needed to create a destination choice model object. - modelManager.managerSetup(propertyMap, modelStructure, ms, dcUecFileName, soaUecFileName, - soaSampleSize, dmuFactory, restartModelString); - - // get a dcModel object from manager, which either creates one or - // returns one for re-use. - WorkLocationChoiceModel dcModel = modelManager.getWorkLocModelObject(taskIndex, iteration, - dcSizeObj, uecIndices, soaUecIndices); - - // logger.info( String.format( - // "%s, task=%d run(), thread=%s, start=%d, end=%d.", VERSION, - // taskIndex, - // threadName, startIndex, endIndex ) ); - System.out.println(String.format("%s: %s, task=%d run(), thread=%s, start=%d, end=%d.", - new Date(), VERSION, taskIndex, threadName, startIndex, endIndex)); - - long setup1 = (System.currentTimeMillis() - startTime) / 1000; - - Household[] householdArray = hhDataManager.getHhArray(startIndex, endIndex); - - long setup2 = (System.currentTimeMillis() - startTime) / 1000; - // logger.info( String.format( - // "task=%d processing households[%d:%d], thread=%s, setup1=%d, setup2=%d.", - // taskIndex, startIndex, endIndex, - // threadName, setup1, setup2 ) ); - System.out.println(String.format("%s: task=%d processing households[%d:%d], thread=%s.", - new Date(), taskIndex, startIndex, endIndex, threadName)); - - int i = -1; - try - { - - boolean runDebugHouseholdsOnly = Util.getBooleanValueFromPropertyMap(propertyMap, - HouseholdDataManager.DEBUG_HHS_ONLY_KEY); - - for (i = 0; i < householdArray.length; i++) - { - // for debugging only - process only household objects specified - // for debugging, if property key was set to true - if (runDebugHouseholdsOnly && !householdArray[i].getDebugChoiceModels()) - // if ( householdArray[i].getHhTaz() % 200 != 0 ) - continue; - - if(!dcModel.applyWorkLocationChoice(householdArray[i])){ - i=0; - String restartMsg="A Worker in this HH batch didn't get valid work location. REPROCESSING HH batch, startIndex:"+startIndex+" endIndex="+endIndex; - setResult(restartMsg); - System.out.println(restartMsg); - } - } - - hhDataManager.setHhArray(householdArray, startIndex); - - //check to make sure hh array got set in hhDataManager - boolean allHouseholdsAreSame = false; - while(!allHouseholdsAreSame) { - Household[] householdArrayRemote = hhDataManager.getHhArray(startIndex, endIndex); - for(int j = 0; j< householdArrayRemote.length;++j) { - - Household remoteHousehold = householdArrayRemote[j]; - Household localHousehold = householdArray[j]; - - allHouseholdsAreSame = checkIfSameWorkLocationResults(remoteHousehold, localHousehold); - - if(!allHouseholdsAreSame) - break; - } - if(!allHouseholdsAreSame) { - System.out.println("Warning: found households in household manager (starting array index "+startIndex+") not updated with work location choice results; updating"); - hhDataManager.setHhArray(householdArray, startIndex); - } - } - - - } catch (Exception e) - { - if (i >= 0 && i < householdArray.length) System.out - .println(String - .format("exception caught in taskIndex=%d applying dc model for i=%d, hhId=%d, startIndex=%d.", - taskIndex, i, householdArray[i].getHhId(), startIndex)); - else System.out.println(String.format( - "exception caught in taskIndex=%d applying dc model for i=%d, startIndex=%d.", - taskIndex, i, startIndex)); - System.out.println("Exception caught:"); - e.printStackTrace(); - System.out.println("Throwing new RuntimeException() to terminate."); - throw new RuntimeException(e); - } - - long getHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup1; - long processHhs = ((System.currentTimeMillis() - startTime) / 1000) - setup2 - getHhs; - // logger.info( String.format( - // "task=%d finished, thread=%s, getHhs=%d, processHhs=%d.", taskIndex, - // threadName, getHhs, processHhs ) ); - System.out.println(String.format("%s: task=%d finished, thread=%s.", new Date(), taskIndex, - threadName)); - - long total = (System.currentTimeMillis() - startTime) / 1000; - String resultString = String - .format("result for thread=%s, task=%d, startIndex=%d, endIndex=%d, startTime=%s, endTime=%s, setup1=%d, setup2=%d, getHhs=%d, run=%d, total=%d.", - threadName, taskIndex, startIndex, endIndex, start, new Date(), setup1, - setup2, getHhs, processHhs, total); - // logger.info( resultString ); - setResult(resultString); - - modelManager.returnWorkLocModelObject(dcModel, taskIndex, startIndex, endIndex); - - clearClassAttributes(); - } - - /** - * Returns true if work location results are the same, else returns false. - * - * @param thisHousehold - * @param thatHousehold - * @return true or false - */ - public boolean checkIfSameWorkLocationResults(Household thisHousehold, Household thatHousehold) { - - Person[] thisPersons = thisHousehold.getPersons(); - Person[] thatPersons = thatHousehold.getPersons(); - - if(thisPersons.length!=thatPersons.length) - return false; - - for(int k=1;k rbMap; - public HashSet householdTraceSet; - public HashSet originTraceSet; - - public String outputsPath; - public String disaggTODPath; - public String todType; - public String outputFile; - - public String inputFile; - public String marketSegment; - public double SampleRate; - - public PrintWriter tripWriter; - - private static Logger logger = Logger.getLogger("postprocessModel"); - - - /** - * Default constructor. - */ - public PostprocessModel(HashMap rbMap, String timeType, double sampleRate, String inputFile, String marketSegment){ - - this.rbMap = rbMap; - this.SampleRate = sampleRate; - this.inputFile = inputFile; - this.marketSegment = marketSegment; - this.todType = timeType; - - } - - - public void runModel(){ - - disaggTODPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_DISAGGPATHTOD); - outputFile = disaggTODPath + Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_TRIPOUT); - outputsPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_OUTPUTSPATH); - outputFile = outputsPath + Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_TRIPOUT); - - setDebugHouseholdsFromPropertyMap(); - setDebugOrigZonesFromPropertyMap(); - - logger.info("Trip file being written to "+outputFile); - // Write the trip header to the output file - boolean fileExists = new File(outputFile).isFile(); - - FileWriter writer; - - if(fileExists){ - logger.info("Output file already exists. New data will be appended."); - try { - writer = new FileWriter(new File(outputFile), true); - tripWriter = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) { - logger.fatal(String.format("Exception occurred opening Postprocessing output file: %s.", - outputFile)); - throw new RuntimeException(e); - } - }else{ - logger.info("Output file does not exist. New file being created."); - try { - writer = new FileWriter(new File(outputFile)); - tripWriter = new PrintWriter(new BufferedWriter(writer)); - } catch (IOException e) { - logger.fatal(String.format("Exception occurred opening Postprocessing output file: %s.", - outputFile)); - throw new RuntimeException(e); - } - dtaTrip Trip = new dtaTrip(); - Trip.writeHeader(tripWriter); - } - - if(todType.equalsIgnoreCase("broad")){ - logger.info("Processing Broad TOD Model"); - TableDataSet broadFiles = TableDataSet.readFile(disaggTODPath+inputFile); - int numFiles = broadFiles.getRowCount(); - for (int i=0; i(); - - // get the household ids for which debug info is required - String householdTraceStringList = rbMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); - - if (householdTraceStringList != null) - { - StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); - while(householdTokenizer.hasMoreTokens()) - { - String listValue = householdTokenizer.nextToken(); - int idValue = Integer.parseInt(listValue.trim()); - householdTraceSet.add(idValue); - } - } - - } - - /** - * Set the HashSet for debugging households, which contains the IDs of the households to debug. - */ - private void setDebugOrigZonesFromPropertyMap() - { - originTraceSet = new HashSet(); - - // get the household ids for which debug info is required - String originTraceStringList = rbMap.get(PROPERTIES_ORIGIN_TRACE_LIST); - - if (originTraceStringList != null) - { - StringTokenizer originTokenizer = new StringTokenizer(originTraceStringList, ","); - while(originTokenizer.hasMoreTokens()) - { - String listValue = originTokenizer.nextToken(); - int idValue = Integer.parseInt(listValue.trim()); - originTraceSet.add(idValue); - } - } - - } - - /** - * Check if this is a trace household. - * - * @param householdId - * @return True if a trace household, else false - */ - public boolean isTraceHousehold(int householdId){ - - return householdTraceSet.contains(householdId); - - } - - /** - * Check if this is a trace origin. - * - * @param householdId - * @return True if a trace household, else false - */ - public boolean isTraceOrigin(int origTAZ){ - - return originTraceSet.contains(origTAZ); - - } - - /** - * @param args - */ - public static void main(String[] args) { - - String propertiesFile = null; - HashMap pMap; - String todType = null; - String inputFile = null; - String marketSegment = null; - double sampleRate = 1.0; - - if (args.length == 0) { - logger.error( String.format("no properties file base name (without .properties extension) was specified as an argument.") ); - return; - } else{ - propertiesFile = args[0]; - } - for (int i = 1; i< args.length;++i){ - if (args[i].equalsIgnoreCase("-todType")){ - todType = (String) args[i + 1]; - } - if (args[i].equalsIgnoreCase("-sampleRate")){ - sampleRate = new Double(args[i+1]); - } - if (args[i].equalsIgnoreCase("-inputFile")){ - inputFile = (String) args[i+1]; - } - if (args[i].equalsIgnoreCase("-marketSegment")){ - marketSegment = (String) args[i+1]; - } - } - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - - logger.info("Running SANDAG Trip TOD Disaggregation Model"); - - logger.info("todType = "+todType); - logger.info("Sample Rate = "+sampleRate); - logger.info("Input File = "+inputFile); - logger.info("Market Segment = "+marketSegment); - - PostprocessModel postprocessingModel = new PostprocessModel(pMap,todType,sampleRate,inputFile,marketSegment); - postprocessingModel.runModel(); - } - -} - - diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java deleted file mode 100644 index 855582c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/broadTODProcessing.java +++ /dev/null @@ -1,224 +0,0 @@ -package org.sandag.abm.dta.postprocessing; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; - -import java.io.File; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.HashSet; - -import org.apache.log4j.Logger; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; - -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.dta.postprocessing.dtaTrip; -import org.sandag.abm.dta.postprocessing.todDisaggregationModel; -import org.sandag.abm.dta.postprocessing.PostprocessModel; -import org.sandag.abm.dta.postprocessing.spatialDisaggregationModel; - -public class broadTODProcessing { - - private static final String PROPERTIES_DISAGGPATHTOD = "dta.postprocessing.disaggregateTOD.path"; - private static final String PROPERTIES_DISAGGPATHZONE = "dta.postprocessing.disaggregateZone.path"; - private static final String PROPERTIES_DISAGGPATHNODE = "dta.postprocessing.disaggregateNode.path"; - private static final String PROPERTIES_BROADTODPROBABILITIES = "dta.postprocessing.BroadTODFile"; - private static final String PROPERTIES_ZONEPROBABILITIES = "dta.postprocessing.ZoneFile"; - private static final String PROPERTIES_NODEPROBABILITIES = "dta.postprocessing.NodeFile"; - private static final String PROPERTIES_RANDOMSEED = "dta.postprocessing.RandomSeed"; - - private HashMap rbMap; - private HashSet originTraceSet; - private MersenneTwister random; - private long randomSeed; - - private String disaggTODPath; - private String disaggZonePath; - private String disaggNodePath; - - private double sampleRate; - private String inputFile; - private String marketSegment; - - private int[] broadTODMap; - private int[] tazMap; - private int[] mgraTAZMap; - private int[] mgraNodeMap; - private int[] nodeMap; - private double[] broadProbabilities; - private double[] mgraProdProbabilities; - private double[] mgraAttrProbabilities; - private double[] nodeProbabilities; - - private todDisaggregationModel todDisaggregationModel; - private spatialDisaggregationModel spatialDisaggregationModel; - - private dtaTrip Trip; - private PrintWriter tripWriter; - - - private transient Logger logger = Logger.getLogger("postprocessModel"); - - - /** - * Default constructor. - */ - public broadTODProcessing(HashMap rbMap, double sampleRate, String inputFile, String marketSegment, HashSet originTraceSet, PrintWriter tripWriter){ - - this.rbMap = rbMap; - this.sampleRate = sampleRate; - this.inputFile = inputFile; - this.marketSegment = marketSegment; - this.tripWriter = tripWriter; - this.originTraceSet = originTraceSet; - - todDisaggregationModel = new todDisaggregationModel(rbMap); - spatialDisaggregationModel = new spatialDisaggregationModel(rbMap); - - randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); - random = new MersenneTwister(); - random.setSeed(randomSeed); - - //Read in factors and maps to aggregate time periods - String broadFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_BROADTODPROBABILITIES); - - TableDataSet BroadData = TableDataSet.readFile(broadFactorsFile); - int numPeriods = BroadData.getRowCount(); - broadProbabilities = todDisaggregationModel.getTODProbabilities(BroadData, numPeriods, marketSegment); - broadTODMap = todDisaggregationModel.getTODMap(BroadData, numPeriods); - - String mgraFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_ZONEPROBABILITIES); - TableDataSet MGRAData = TableDataSet.readFile(mgraFactorsFile); - int numMGRAs = MGRAData.getRowCount(); - mgraProdProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Prods", marketSegment); - mgraAttrProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Attrs", marketSegment); - tazMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "taz"); - mgraTAZMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "mgra"); - - String nodeFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_NODEPROBABILITIES); - TableDataSet NodeData = TableDataSet.readFile(nodeFactorsFile); - int numNodes = NodeData.getRowCount(); - nodeProbabilities = spatialDisaggregationModel.getSpatialProbabilities(NodeData, numNodes, "Probability", null); - nodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "NodeId"); - mgraNodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "MGRA"); - - } - - - /** - * Create trip record from and disaggregate tod, mgra, and node for broad tod files - */ - public void createBroadTODTrips(String inputFileName,String MarketSegment,String matrixName,int broadTOD,String vehType,int occ,int toll){ - - //Read TransCAD matrix and create a trip for each record in each cell - File inputFile = new File(inputFileName); - Matrix m = MatrixReader.readMatrix(inputFile,matrixName); - int intTrips = 0; - double expansionFactor=1.0; - int totalTrips = 0; - boolean debug=false; - - logger.info("*************************************"); - logger.info("Summary info for TransCAD Matrix"); - logger.info("Market Segment = "+MarketSegment); - logger.info("TNCVehicle Type = "+vehType); - logger.info("TNCVehicle Occupancy = "+occ); - logger.info("Toll Eligibility = "+toll); - logger.info("Number of Trips = "+m.getSum()); - logger.info("*************************************"); - - - for (int i=1; i<=m.getRowCount();++i){ - for (int j=1; j<=m.getColumnCount();++j){ - - double numTrips = m.getValueAt(i,j); - - if (numTrips==0.0) - continue; - - intTrips = (int) Math.floor(numTrips); - double fracTrips = numTrips - intTrips; - double rn = random.nextDouble(); - // Check if a trip should be created for the fractional trip value - if (rn rbMap; - private HashSet householdTraceSet; - private HashSet originTraceSet; - private MgraDataManager mgraManager; - //private final AutoAndNonMotorizedSkimsCalculator autoNonMotSkims; - private MersenneTwister random; - private long randomSeed; - - private todDisaggregationModel todDisaggregationModel; - private spatialDisaggregationModel spatialDisaggregationModel; - - private String disaggTODPath; - private String disaggNodePath; - private String outputsPath; - - private double sampleRate; - private String inputFile; - private String marketSegment; - - private int[] detailTODMap; - private int[] mgraNodeMap; - private int[] nodeMap; - private int[] tazMap; - private int[] mgraTAZMap; - - private double[] detailProbabilities; - private double[] nodeProbabilities; - private double[] mgraProdProbabilities; - private double[] mgraAttrProbabilities; - - private dtaTrip Trip; - private PrintWriter tripWriter; - - - private transient Logger logger = Logger.getLogger("postprocessModel"); - - - /** - * Default constructor. - */ - public detailedTODProcessing(HashMap rbMap, double sampleRate, String inputFile, String marketSegment, HashSet householdTraceSet, HashSet originTraceSet, PrintWriter tripWriter){ - - this.sampleRate = sampleRate; - this.rbMap = rbMap; - this.inputFile = inputFile; - this.marketSegment = marketSegment; - this.tripWriter = tripWriter; - this.householdTraceSet = householdTraceSet; - this.originTraceSet = originTraceSet; - - todDisaggregationModel = new todDisaggregationModel(rbMap); - - spatialDisaggregationModel = new spatialDisaggregationModel(rbMap); - - mgraManager = MgraDataManager.getInstance(rbMap); - //autoNonMotSkims = new AutoAndNonMotorizedSkimsCalculator(rbMap); - - outputsPath = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_OUTPUTSPATH); - - //Read in factors and maps to aggregate time periods - String detailedFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_DETAILPROBABILITIES); - - TableDataSet DetailedData = TableDataSet.readFile(detailedFactorsFile); - int numDetailedPeriods = DetailedData.getRowCount(); - detailProbabilities = todDisaggregationModel.getTODProbabilities(DetailedData, numDetailedPeriods, null); - detailTODMap = todDisaggregationModel.getTODMap(DetailedData, numDetailedPeriods); - - String nodeFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_NODEPROBABILITIES); - TableDataSet NodeData = TableDataSet.readFile(nodeFactorsFile); - int numNodes = NodeData.getRowCount(); - nodeProbabilities = spatialDisaggregationModel.getSpatialProbabilities(NodeData, numNodes, "Probability", null); - nodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "NodeId"); - mgraNodeMap = spatialDisaggregationModel.getSpatialMap(NodeData, numNodes, "MGRA"); - - randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); - random = new MersenneTwister(); - random.setSeed(randomSeed); - - // read skims - AshishK - String skimPath = Util.getStringValueFromPropertyMap(rbMap, "skims.path"); - String skimPrefix = Util.getStringValueFromPropertyMap(rbMap, "dta.skims.prefix"); - String skimSuffix = Util.getStringValueFromPropertyMap(rbMap, "dta.skims.mat.name.suffix"); - skimMatrix = new Matrix[ModelStructure.MODEL_PERIOD_LABELS.length]; - - for(int p=0; p rbMap){ - SandagModelStructure modelStructure = new SandagModelStructure(); - TableDataSet tripRecords = TableDataSet.readFile(inputFile); - int numRecords = tripRecords.getRowCount(); - int period = -1; - int periodLast = -1; - int hhidLast = -1; - int persidLast = -1; - int touridLast = -1; - int modeLast = -1; - - double expansionFactor=1.0; - boolean debug = false; - boolean addSOVTrip = false; - int tripExp=0; - int offset = 0; - int scheduleCount = 0; - int tripsCount = 0; - int [] dtaTimes; - int [] dtaTimesPrev; - dtaTimes = new int[10]; - dtaTimesPrev = new int[10]; - - Arrays.fill(dtaTimesPrev, 0); - - if(inputFile.contains("TripMatrices.csv")) { - String mgraFactorsFile = Util.getStringValueFromPropertyMap(rbMap, PROPERTIES_ZONEPROBABILITIES); - TableDataSet MGRAData = TableDataSet.readFile(mgraFactorsFile); - int numMGRAs = MGRAData.getRowCount(); - mgraProdProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Prods", marketSegment); - mgraAttrProbabilities = spatialDisaggregationModel.getSpatialProbabilities(MGRAData, numMGRAs, "Attrs", marketSegment); - tazMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "taz"); - mgraTAZMap = spatialDisaggregationModel.getSpatialMap(MGRAData, numMGRAs, "mgra"); - - } - - - logger.info("Reading "+numRecords+" trip records from disaggregate file: "+inputFile); - - // Create a trip record for each record in the input file - for (int i=0; i0){ - destMGRA = parkingMGRA; - } - } - if (tripRecords.containsColumn("originMGRA")){ - origMGRA = (int) tripRecords.getValueAt(i+1, "originMGRA"); - } - if (tripRecords.containsColumn("destinationMGRA")){ - destMGRA = (int) tripRecords.getValueAt(i+1, "destinationMGRA"); - } - - if (tripRecords.containsColumn("originTAZ")){ - origTAZ = (int) tripRecords.getValueAt(i+1, "originTAZ"); - }else{ - origTAZ = mgraManager.getTaz(origMGRA); - } - if (tripRecords.containsColumn("destinationTAZ")){ - destTAZ = (int) tripRecords.getValueAt(i+1, "destinationTAZ"); - }else{ - destTAZ = mgraManager.getTaz(destMGRA); - } - - if(origMGRA==0) - origMGRA = spatialDisaggregationModel.selectMGRAfromTAZ(Trip.getOriginTaz(),mgraTAZMap,tazMap,mgraProdProbabilities,debug); - - if(destMGRA==0) - destMGRA = spatialDisaggregationModel.selectMGRAfromTAZ(Trip.getOriginTaz(),mgraTAZMap,tazMap,mgraAttrProbabilities,debug); - - if (origMGRA==0){ - origMGRA = 50000+Trip.getOriginTaz(); - } - if (destMGRA==0){ - destMGRA = 50000+Trip.getDestinationTaz(); - } - - //JEF 2021-04-21: not sure what the following code intended - removing - /* - if (tripRecords.containsColumn("arrivalMode")){ - int arriveMode = (int) tripRecords.getValueAt(i+1, "arrivalMode"); - if((occ>1 && arriveMode==5)||(mode>=16 && mode<26)){ - addSOVTrip = true; - } - } - */ - if (tripRecords.containsColumn("driver")){ - tourDriver = (int) tripRecords.getValueAt(i+1, "driver"); - } - - if (tripRecords.containsColumn("stop_period")){ - period = (int) tripRecords.getValueAt(i+1,"stop_period"); - } - if (tripRecords.containsColumn("period")){ - period = (int) tripRecords.getValueAt(i+1,"period"); - } - if (tripRecords.containsColumn("departTime")){ - period = (int) tripRecords.getValueAt(i+1,"departTime"); - } - if(tripRecords.containsColumn("departTimeAbmHalfHour")) { - period = (int) tripRecords.getValueAt(i+1,"departTimeAbmHalfHour"); - - } - - if(period==0) - period=1; - - //Calculate number of trips to generate from the record (at a tour level where possible) - - if (tourid!=touridLast || persid!=persidLast || hhid!=hhidLast || (tourid==0 && hhid==0)){ - tripExp = (int) Math.floor(expansionFactor); - double tripsFrac = expansionFactor - tripExp; - double rn = random.nextDouble(); - if (rnfractionalTrips) - tripExp += 0; - }else if(fractionalTrips<1.0 && mode==modeLast){ - double rn = random.nextDouble(); - if (rn>fractionalTrips) - tripExp += 0; - }else if((fractionalTrips<1.0 && mode!=modeLast) || (fractionalTrips==1.0 && modelStructure.getTourModeIsHov(modeLast))){ - tripExp = (int) Math.floor(expansionFactor); - double tripsFrac = expansionFactor - tripExp; - double rn = random.nextDouble(); - if (rnfractionalTrips) - tripExp += 0; - } - - //logger.info("expansionFactor " + expansionFactor + ", fractionalTrips " + fractionalTrips + ", tripExp " + tripExp); - // Reset the dtaTimes array - Arrays.fill(dtaTimes,0); - - // Create a number of integer trip instances based on the expansion factor - for (int k=0; k=1 & dtaPer<=36){ - tod = EA_SKIM_PERIOD_INDEX; - }else if(dtaPer>36 & dtaPer<=72){ - tod = AM_SKIM_PERIOD_INDEX; - }else if(dtaPer>72 & dtaPer<=150){ - tod = MD_SKIM_PERIOD_INDEX; - }else if(dtaPer>150 & dtaPer<=192){ - tod = PM_SKIM_PERIOD_INDEX; - }else{ - tod = EV_SKIM_PERIOD_INDEX; - } - - //dtaPeriod = todDisaggregationModel.calculateDisaggregateTOD(period, detailTODMap, detailProbabilities,debug); - - //double[] autoSkims = autoNonMotSkims.getAutoSkims(tripOrig, tripDest, tod, false, logger); - //double travTime = autoSkims[DA_TIME_INDEX]; - - //changed the code to directly read the values from skim matrices - AshishK - double travTime = skimMatrix[tod].getValueAt(origTAZ, destTAZ); - - int travPer = (int) Math.ceil(travTime/5.0); - - if (direction==1){ - dtaPeriod = dtaPer + travPer + 2; - }else{ - dtaPeriod = dtaPer - (travPer + 2); - } - - // limit the dta period between 1 and 288 - AshishK - if(dtaPeriod < 1) { - dtaPeriod = 1; - } - if(dtaPeriod > detailProbabilities.length) { - dtaPeriod = detailProbabilities.length; - } - - int origNode = spatialDisaggregationModel.selectNodeFromMGRA(tripOrig, nodeMap, mgraNodeMap, nodeProbabilities, debug); - int destNode = spatialDisaggregationModel.selectNodeFromMGRA(tripDest, nodeMap, mgraNodeMap, nodeProbabilities, debug); - - Trip = new dtaTrip(); - Trip.setMarketSegment(marketSegment); - Trip.setOriginMGRA(tripOrig); - Trip.setDestinationMGRA(tripDest); - Trip.setOriginTaz(origTAZ); - Trip.setDestinationTaz(destTAZ); - Trip.setOriginNode(origNode); - Trip.setDestinationNode(destNode); - Trip.setDetailedPeriod(period); - Trip.setDTAPeriod(dtaPeriod); - Trip.setVehicleType("passengerCar"); - Trip.setVehicleOccupancy(1); - Trip.setTollEligible(toll); - Trip.setExpansionFactor(1.0); - tripWriter.print("\r\n"); - Trip.writeTrip(tripWriter); - - } - - /** - * Check if this is a trace household. - * - * @param householdId - * @return True if a trace household, else false - */ - public boolean isTraceHousehold(int householdId){ - - return householdTraceSet.contains(householdId); - - } - - /** - * Check if this is a trace origin. - * - * @param householdId - * @return True if a trace household, else false - */ - public boolean isTraceOrigin(int origTAZ){ - - return originTraceSet.contains(origTAZ); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java deleted file mode 100644 index 44b6e34..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/dtaTrip.java +++ /dev/null @@ -1,386 +0,0 @@ -package org.sandag.abm.dta.postprocessing; - -import java.io.PrintWriter; -import java.io.Serializable; - -public class dtaTrip implements Serializable { - - private int id; - private int hhid; - private int persid; - private int tourid; - private int originTaz; - private int destinationTaz; - private int originMGRA; - private int destinationMGRA; - private int originNode; - private int destinationNode; - private String vehicleType; - private int vehOccupancy; - private int tollEligible; - private String marketSegment; - private int broadPeriod; - private int detailedPeriod; - private int dtaPeriod; - private int driver; - private double expansionFactor; - - - - /** - * Default constructor; nothing initialized. - */ - public dtaTrip(){ - - } - /** - * Initialize a trip will zero values for all fields - */ - public void initializeTrip() { - this.id=0; - this.hhid=0; - this.persid=0; - this.tourid=0; - this.originTaz=0; - this.destinationTaz=0; - this.originMGRA=0; - this.destinationMGRA=0; - this.originNode=0; - this.destinationNode=0; - this.vehicleType="na"; - this.vehOccupancy=0; - this.tollEligible=0; - this.marketSegment="na"; - this.broadPeriod=0; - this.detailedPeriod=0; - this.dtaPeriod=0; - this.driver=-1; - this.expansionFactor=1.0; - - } - /** - * @return the household id - */ - public int getHHId() { - return hhid; - } - /** - * @param hhid the household id to set - */ - public void setHHId(int hhid) { - this.hhid = hhid; - } - /** - * @return the person id - */ - public int getPersonId() { - return persid; - } - /** - * @param persid the person id to set - */ - public void setPersonId(int persid) { - this.persid = persid; - } - /** - * @return the tour id - */ - public int getTourId() { - return tourid; - } - /** - * @param tourid the tour id to set - */ - public void setTourId(int tourid) { - this.tourid = tourid; - } - /** - * @return the id - */ - public int getId() { - return id; - } - /** - * @param id the id to set - */ - public void setId(int id) { - this.id = id; - } - - /** - * @return the originTaz - */ - public int getOriginTaz() { - return originTaz; - } - /** - * @param originTaz the originTaz to set - */ - public void setOriginTaz(int originTaz) { - this.originTaz = originTaz; - } - /** - * @return the destinationTaz - */ - public int getDestinationTaz() { - return destinationTaz; - } - /** - * @param destinationTaz the destinationTaz to set - */ - public void setDestinationTaz(int destinationTaz) { - this.destinationTaz = destinationTaz; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() { - return originMGRA; - } - /** - * @param originMGRA the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) { - this.originMGRA = originMGRA; - } - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() { - return destinationMGRA; - } - /** - * @param destinationMGRA the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) { - this.destinationMGRA = destinationMGRA; - } - - /** - * @return the originNode - */ - public int getOriginNode() { - return originNode; - } - /** - * @param originNode the originNode to set - */ - public void setOriginNode(int originNode) { - this.originNode = originNode; - } - /** - * @return the destinationNode - */ - public int getDestinationNode() { - return destinationNode; - } - /** - * @param destinationMGRA the destinationMGRA to set - */ - public void setDestinationNode(int destinationNode) { - this.destinationNode = destinationNode; - } - /** - * set trip mode values based on trip mode in input file - */ - public void setTripMode(int mode) { - if(mode<=8||mode>=27){ - setVehicleType("passengerCar"); - setVehicleOccupancy(1); - setTollEligible(0); - if(mode==2){ - setTollEligible(1); - } - if((mode>=3 && mode<=5)||mode==27){ - setVehicleOccupancy(2); - } - if(mode==5){ - setTollEligible(1); - } - if(mode>=6 && mode<=8){ - setVehicleOccupancy(3); - } - if(mode==8){ - setTollEligible(1); - } - } - if(mode>8 && mode<11){ - setVehicleType("nonMotorized"); - setVehicleOccupancy(0); - setTollEligible(0); - } - if(mode>=11 && mode<16){ - setVehicleType("WalkTransit"); - setVehicleOccupancy(0); - setTollEligible(0); - } - if(mode>=16 && mode<26){ - setVehicleType("DriveTransit"); - setVehicleOccupancy(1); - setTollEligible(0); - } - if(mode==26){ - setVehicleType("SchoolBus"); - } - } - /** - * @return the person number of the driver - */ - public int getTourDriver() { - return driver; - } - /** - * @param driver the tour driver to set - */ - public void setTourDriver(int tourDriver) { - this.driver = tourDriver; - } - - /** - * @return the vehicleType - */ - public String getVehicleType() { - return vehicleType; - } - /** - * @param vehicleType the vehicleType to set - */ - public void setVehicleType(String vehicleType) { - this.vehicleType = vehicleType; - } - /** - * @return the vehicleOccupancy - */ - public int getVehicleOccupancy() { - return vehOccupancy; - } - /** - * @param vecOccupancy the vehOccupancy to set - */ - public void setVehicleOccupancy(int vehOccupancy) { - this.vehOccupancy = vehOccupancy; - } - /** - * @return the tollEligibility - */ - public int getTollEligible() { - return tollEligible; - } - /** - * @param tollEligible the tollEligible to set - */ - public void setTollEligible(int tollEligible) { - this.tollEligible = tollEligible; - } - /** - * @return the market segment - */ - public String getMarketSegment() { - return marketSegment; - } - - /** - * @param marketSegment the marketSegment to set - */ - public void setMarketSegment(String marketSegment){ - this.marketSegment = marketSegment; - } - - /** - * @return the broad time period - */ - public int getBroadPeriod() { - return broadPeriod; - } - /** - * @param broadPeriod the broadPeriod to set - */ - public void setBroadPeriod(int Period) { - this.broadPeriod = Period; - } - - /** - * @return the detailed time period - */ - public int getDetailedPeriod() { - return detailedPeriod; - } - /** - * @param detailedPeriod the detailedPeriod to set - */ - public void setDetailedPeriod(int Period) { - this.detailedPeriod = Period; - } - - /** - * @return the dta time period - */ - public int getDTAPeriod() { - return dtaPeriod; - } - /** - * @param dtaPeriod the dtaPeriod to set - */ - public void setDTAPeriod(int Period) { - this.dtaPeriod = Period; - } - /** - * @return the trip expansion factor - */ - public double getExpansionFactor() { - return expansionFactor; - } - /** - * @param dtaPeriod the dtaPeriod to set - */ - public void setExpansionFactor(double expansionFactor) { - this.expansionFactor = expansionFactor; - } - - /** - * Write the trip - * - * @param writer - */ - public void writeTrip(PrintWriter writer){ - String record = new String( - hhid + "," + - persid + "," + - tourid + "," + - id + "," + - originTaz + "," + - destinationTaz + "," + - originMGRA + "," + - destinationMGRA + "," + - originNode + "," + - destinationNode + "," + - vehicleType + "," + - vehOccupancy + "," + - tollEligible + "," + - marketSegment + "," + - detailedPeriod + "," + - broadPeriod + "," + - dtaPeriod + "," + - driver + "," + - expansionFactor - ); - writer.print(record); - } - - /** - * Write a header record - * - * @param writer - */ - /** - * Write a header record - * - * @param writer - */ - public void writeHeader(PrintWriter writer){ - String header = "hh_id,person_id,tour_id,trip_id,originTaz,destinationTaz,originMGRA,destinationMGRA,originNode,destinationNode,vehicleType,vehicleOccupancy,tollEligibility,marketSegment,detailedPeriod,broadPeriod,dtaPeriod,driver,expansionFactor"; - writer.print(header); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java b/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java deleted file mode 100644 index fd22ba1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/dta/postprocessing/spatialDisaggregationModel.java +++ /dev/null @@ -1,144 +0,0 @@ -package org.sandag.abm.dta.postprocessing; - -import java.io.PrintWriter; -import java.util.HashMap; - -import org.apache.log4j.Logger; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import org.sandag.abm.ctramp.Util; - -public class spatialDisaggregationModel { - - private static final String PROPERTIES_RANDOMSEED = "dta.postprocessing.RandomSeed"; - - private HashMap rbMap; - private MersenneTwister random; - private long randomSeed; - - public PrintWriter tripWriter; - - - private transient Logger logger = Logger.getLogger("postprocessModel"); - - - /** - * Default constructor. - */ - public spatialDisaggregationModel(HashMap rbMap){ - - this.rbMap = rbMap; - randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); - random = new MersenneTwister(); - random.setSeed(randomSeed); - } - - /** - * Read the probability by spatial data file, return an array of probabilities. - */ - public double[] getSpatialProbabilities(TableDataSet SpatialData, int numRecords, String inputField, String marketSegment){ - - // read the spatial factors file - double [] probabilities; - probabilities = new double [numRecords]; - - String fieldName = null; - if (marketSegment==null){ - fieldName = inputField; - }else{ - fieldName = marketSegment+inputField; - } - //fill in probabilities array - for(int i = 0;i rbMap; - private MersenneTwister random; - private long randomSeed; - - - public PrintWriter tripWriter; - - - private transient Logger logger = Logger.getLogger("postprocessModel"); - - - /** - * Default constructor. - */ - public todDisaggregationModel(HashMap rbMap){ - this.rbMap = rbMap; - randomSeed = Util.getIntegerValueFromPropertyMap(rbMap, PROPERTIES_RANDOMSEED); - random = new MersenneTwister(); - random.setSeed(randomSeed); - } - - /** - * Read the probability by tod data file, return an array of probabilities. - */ - public double[] getTODProbabilities(TableDataSet TODData, int numPeriods, String marketSegment){ - - // read the tod factors file - double [] probabilities; - probabilities = new double [numPeriods]; - - //fill in probabilities array - for(int i = 0;i1) - startLoc = (period-1)*6 + 18; - - // loop through the array of probabilities - for (int i=startLoc;i rbMap; - private McLogsumsCalculator logsumsCalculator; - private AutoTazSkimsCalculator tazDistanceCalculator; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - - private boolean seek; - private int traceId; - - private static float sampleRate=0; - private int iteration = 1; - - /** - * Constructor - * - * @param rbMap - */ - public InternalExternalModel(HashMap rbMap) - { - this.rbMap = rbMap; - mgraManager = MgraDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trace")); - - } - - public int getIteration() - { - return iteration; - } - - public void setIteration(int iteration) - { - this.iteration = iteration; - } - - /** - * Run InternalExternal model. - */ - public void runModel() - { - - InternalExternalModelStructure modelStructure = new InternalExternalModelStructure(); - - InternalExternalDmuFactoryIf dmuFactory = new InternalExternalDmuFactory(modelStructure); - - InternalExternalTourManager tourManager = new InternalExternalTourManager(rbMap, iteration); - - tourManager.generateTours(); - - InternalExternalTour[] tours = tourManager.getTours(); - - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - logsumsCalculator = new McLogsumsCalculator(); - logsumsCalculator.setupSkimCalculators(rbMap); - logsumsCalculator.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - InternalExternalTourTimeOfDayChoiceModel todChoiceModel = new InternalExternalTourTimeOfDayChoiceModel( - rbMap); - InternalExternalTourDestChoiceModel destChoiceModel = new InternalExternalTourDestChoiceModel( - rbMap, modelStructure, dmuFactory); - destChoiceModel.calculateTazProbabilities(dmuFactory); - - InternalExternalTripModeChoiceModel tripModeChoiceModel = new InternalExternalTripModeChoiceModel( - rbMap, modelStructure, dmuFactory); - - // Run models for array of tours - for (int i = 0; i < tours.length; ++i) - { - - InternalExternalTour tour = tours[i]; - - if (i < 10 || i % 1000 == 0) logger.info("Processing tour " + i); - - if (seek && tour.getID() != traceId) continue; - - if (tour.getID() == traceId) tour.setDebugChoiceModels(true); - - todChoiceModel.calculateTourTOD(tour); - destChoiceModel.chooseDestination(tour); - - // generate trips and choose mode for them - note this assumes two - // trips per tour - InternalExternalTrip[] trips = new InternalExternalTrip[2]; - int tripNumber = 0; - - // generate an outbound trip from the tour origin to the destination - // and choose a mode - trips[tripNumber] = new InternalExternalTrip(tour, true, mgraManager); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - // generate an inbound trip from the tour destination to the origin - // and choose a mode - trips[tripNumber] = new InternalExternalTrip(tour, false, mgraManager); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - // set the trips in the tour object - tour.setTrips(trips); - - } - - tourManager.writeOutputFile(rbMap); - - logger.info("Internal-External Model successfully completed!"); - - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @param args - */ - public static void main(String[] args) - { - Runtime gfg = Runtime.getRuntime(); - long memory1; - // checking the total memeory - System.out.println("Total memory is: "+ gfg.totalMemory()); - // checking free memory - memory1 = gfg.freeMemory(); - System.out.println("Initial free memory at IE model: "+ memory1); - // calling the garbage collector on demand - gfg.gc(); - memory1 = gfg.freeMemory(); - System.out.println("Free memory after garbage "+ "collection: " + memory1); - - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running InternalExternal Model")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - - // sampleRate is not relevant for internal-external model, since - // sampling - // would have been applied in CT-RAMP model - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - - logger.info("IE Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - InternalExternalModel internalExternalModel = new InternalExternalModel(pMap); - internalExternalModel.setIteration(iteration); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = internalExternalModel.startMatrixServerProcess( - matrixServerAddress, serverPort, mt); - internalExternalModel.ms = matrixServer; - } else - { - internalExternalModel.ms = new MatrixDataServerRmi(matrixServerAddress, - serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - internalExternalModel.ms.testRemote("InternalExternalModel"); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(internalExternalModel.ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - internalExternalModel.runModel(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java deleted file mode 100644 index f97bed3..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalModelStructure.java +++ /dev/null @@ -1,95 +0,0 @@ -package org.sandag.abm.internalexternal; - -import org.sandag.abm.application.SandagModelStructure; - -public class InternalExternalModelStructure - extends SandagModelStructure -{ - - public static final byte NUMBER_VISITOR_PURPOSES = 6; - public static final byte WORK = 0; - public static final byte RECREATION = 1; - public static final byte DINING = 2; - - public static final String[] VISITOR_PURPOSES = {"WORK", "RECREATE", "DINING"}; - - // override on max tour mode, since we have taxi in this model. - public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 27; - - public static final byte NUMBER_VISITOR_SEGMENTS = 2; - public static final byte BUSINESS = 0; - public static final byte PERSONAL = 1; - - public static final String[] VISITOR_SEGMENTS = {"BUSINESS", "PERSONAL"}; - public static final byte DEPARTURE = 0; - public static final byte ARRIVAL = 1; - - public static final byte INCOME_SEGMENTS = 5; - - // note that time periods start at 1 and go to 40 - public static final byte TIME_PERIODS = 40; - - public static final int AM = 0; - public static final int PM = 1; - public static final int OP = 2; - public static final int[] SKIM_PERIODS = {AM, PM, OP}; - public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; - public static final int UPPER_EA = 3; - public static final int UPPER_AM = 9; - public static final int UPPER_MD = 22; - public static final int UPPER_PM = 29; - public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; - - public static final byte TAXI = 13; - - /** - * Taxi tour mode - * - * @param tourMode - * @return - */ - public boolean getTourModeIsTaxi(int tourMode) - { - - if (tourMode == TAXI) return true; - else return false; - - } - - /** - * return the Skim period index 0=am, 1=pm, 2=off-peak - */ - public static int getSkimPeriodIndex(int departPeriod) - { - - int skimPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; - else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; - else skimPeriodIndex = OP; - - return skimPeriodIndex; - - } - - /** - * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV - */ - public static int getModelPeriodIndex(int departPeriod) - { - - int modelPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; - else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; - else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; - else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; - else modelPeriodIndex = 4; - - return modelPeriodIndex; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java deleted file mode 100644 index 97a7dc8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTour.java +++ /dev/null @@ -1,307 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.Serializable; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Household; -import com.pb.common.math.MersenneTwister; - -public class InternalExternalTour - implements Serializable -{ - - private MersenneTwister random; - private int ID; - private int hhID; - private int personID; - private int pnum; - - // following variables set from household and person objects - private int income; - private int autos; - private int age; - private int female; - private double nonWorkTimeFactor; - - private boolean avAvailable; - - // private InternalExternalStop[] outboundStops; - // private InternalExternalStop[] inboundStops; - - private InternalExternalTrip[] trips; - - private int departTime; - private int arriveTime; - - private boolean debugChoiceModels; - - // following variables chosen via choice models - private int originMGRA; - private int destinationMGRA; - private int destinationTAZ; // the external TAZ may be - // different from the - // external MGRA - - /** - * Public constructor. - * - * @param seed - * A seed for the random number generator. - */ - public InternalExternalTour(long seed) - { - - random = new MersenneTwister(seed); - } - - /** - * @return the destinationTAZ - */ - public int getDestinationTAZ() - { - return destinationTAZ; - } - - /** - * @param destinationTAZ - * the destinationTAZ to set - */ - public void setDestinationTAZ(int destinationTAZ) - { - this.destinationTAZ = destinationTAZ; - } - - /** - * @return the iD - */ - public int getID() - { - return ID; - } - - /** - * @param iD - * the iD to set - */ - public void setID(int iD) - { - ID = iD; - } - - /** - * @return the departTime - */ - public int getDepartTime() - { - return departTime; - } - - /** - * @param departTime - * the departTime to set - */ - public void setDepartTime(int departTime) - { - this.departTime = departTime; - } - - public InternalExternalTrip[] getTrips() - { - return trips; - } - - public void setTrips(InternalExternalTrip[] trips) - { - this.trips = trips; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() - { - return originMGRA; - } - - /** - * @param originMGRA - * the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) - { - this.originMGRA = originMGRA; - } - - /** - * Get a random number from the parties random class. - * - * @return A random number. - */ - public double getRandom() - { - return random.nextDouble(); - } - - /** - * @return the debugChoiceModels - */ - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - /** - * @param debugChoiceModels - * the debugChoiceModels to set - */ - public void setDebugChoiceModels(boolean debugChoiceModels) - { - this.debugChoiceModels = debugChoiceModels; - } - - /** - * Get the number of outbound stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberOutboundStops() - { - return 0; - - } - - /** - * Get the number of return stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberInboundStops() - { - return 0; - - } - - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() - { - return destinationMGRA; - } - - /** - * @param destinationMGRA - * the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) - { - this.destinationMGRA = destinationMGRA; - } - - public void setArriveTime(int arriveTime) - { - this.arriveTime = arriveTime; - } - - public int getArriveTime() - { - return arriveTime; - } - - /** - * @return the income - */ - public int getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(int income) - { - this.income = income; - } - - public int getAutos() - { - return autos; - } - - public void setAutos(int autos) - { - this.autos = autos; - } - - public int getAge() - { - return age; - } - - public void setAge(int age) - { - this.age = age; - } - - public int getFemale() - { - return female; - } - - public void setFemale(int female) - { - this.female = female; - } - - public int getHhID() { - return hhID; - } - - public void setHhID(int hhID) { - this.hhID = hhID; - } - - public int getPersonID() { - return personID; - } - - public void setPersonID(int personID) { - this.personID = personID; - } - - public int getPnum() { - return pnum; - } - - public void setPnum(int pnum) { - this.pnum = pnum; - } - - public double getNonWorkTimeFactor() { - return nonWorkTimeFactor; - } - - public void setNonWorkTimeFactor(double nonWorkTimeFactor) { - this.nonWorkTimeFactor = nonWorkTimeFactor; - } - - public boolean isAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(boolean avAvailable) { - this.avAvailable = avAvailable; - } - - public void logTourObject(Logger logger, int totalChars) - { - - Household.logHelper(logger, "tourId: ", ID, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); - Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java deleted file mode 100644 index 05886cc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceDMU.java +++ /dev/null @@ -1,114 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class InternalExternalTourDestChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("internalExternalModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - public InternalExternalTourDestChoiceDMU(InternalExternalModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - // methodIndexMap.put("getTimeOutbound", 0); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - /* - * - * switch (variableIndex) { - * - * case 0: returnValue = getTimeOutbound(); break; - * - * default: logger.error("method number = " + variableIndex + - * " not found"); throw new RuntimeException("method number = " + - * variableIndex + " not found"); - * - * } - */ - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java deleted file mode 100644 index 83e952a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourDestChoiceModel.java +++ /dev/null @@ -1,172 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * This class is used for external station destination choice model for IE - * tours. - * - * - * @author Freedman - * - */ -public class InternalExternalTourDestChoiceModel -{ - - private transient Logger logger = Logger.getLogger("internalExternalModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication destModel; - - private HashMap rbMap; - - private InternalExternalTourDestChoiceDMU dcDmu; - - private Matrix tazProbabilities; - private TableDataSet altData; - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of airport model DMUs - */ - public InternalExternalTourDestChoiceModel(HashMap rbMap, - InternalExternalModelStructure modelStructure, InternalExternalDmuFactoryIf dmuFactory) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - - // initiate a DMU - dcDmu = dmuFactory.getInternalExternalTourDestChoiceDMU(); - - // create the full model UECs - // read the model pages from the property file, create one choice model - // for each full model - String internalExternalDCFileName = Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.dc.uec.file"); - internalExternalDCFileName = uecFileDirectory + internalExternalDCFileName; - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, - "internalExternal.dc.uec.data.page"); - int destModelPage = Util.getIntegerValueFromPropertyMap(rbMap, - "internalExternal.dc.uec.model.page"); - destModel = new ChoiceModelApplication(internalExternalDCFileName, destModelPage, dataPage, - rbMap, (VariableTable) dcDmu); - - } - - /** - * Calculate taz probabilities. This method initializes and calculates the - * tazProbabilities array. - */ - public void calculateTazProbabilities(InternalExternalDmuFactoryIf dmuFactory) - { - - logger.info("Calculating IE Model TAZ Probabilities Arrays"); - - // iterate through the alternatives in the alternatives file and set the - // size term and station logsum for each alternative - UtilityExpressionCalculator soaModelUEC = destModel.getUEC(); - altData = soaModelUEC.getAlternativeData(); - - // initialize the arrays - int maxTaz = tazManager.getMaxTaz(); - - tazProbabilities = new Matrix("Prob_Matrix", "Probability Matrix", maxTaz + 1, maxTaz + 1); - - // iterate through origin zones, solve the UEC and store the results in - // the matrix - for (int taz = 1; taz <= maxTaz; ++taz) - { - - int originTaz = taz; - - // set origin taz in dmu (destination set in UEC by alternative) - dcDmu.setDmuIndexValues(originTaz, originTaz, originTaz, originTaz, false); - - // Calculate utilities & probabilities - destModel.computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); - - // Store probabilities (by purpose) - double[] probabilities = destModel.getCumulativeProbabilities(); - - for (int i = 0; i < probabilities.length; ++i) - { - - double cumProb = probabilities[i]; - int destTaz = (int) altData.getValueAt(i + 1, "taz"); - tazProbabilities.setValueAt(originTaz, destTaz, (float) cumProb); - } - } - logger.info("Finished Calculating IE Model TAZ Probabilities Arrays"); - } - - /** - * Choose a destination TAZ and MGRA for the tour. - * - * @param tour - * An IE tour with a tour origin. - */ - public void chooseDestination(InternalExternalTour tour) - { - - double random = tour.getRandom(); - int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Choosing destination alternative"); - tour.logTourObject(logger, 1000); - - } - - // cycle through probability array for origin taz and find destination - // station & corresponding MGRA - int chosenTaz = -1; - int chosenMgra = -1; - for (int i = 1; i <= altData.getRowCount(); ++i) - { - int destTaz = (int) altData.getValueAt(i, "taz"); - if (random < tazProbabilities.getValueAt(originTaz, destTaz)) - { - chosenTaz = destTaz; - chosenMgra = (int) altData.getValueAt(i, "mgraOut"); - break; - } - } - - if (chosenTaz == -1) - { - logger.error("Error: IE Tour Destination Choice Model for tour " + tour.getID()); - throw new RuntimeException(); - } - - tour.setDestinationMGRA(chosenMgra); - tour.setDestinationTAZ(chosenTaz); - - if (tour.getDebugChoiceModels()) - logger.info("Chose taz " + chosenTaz + " mgra " + chosenMgra + " with random " + random); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java deleted file mode 100644 index 3d5d669..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourManager.java +++ /dev/null @@ -1,377 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.util.ResourceUtil; - -public class InternalExternalTourManager -{ - - private static Logger logger = Logger.getLogger("internalExternalModel"); - - private InternalExternalTour[] tours; - public static final String PROPERTIES_DISTRIBUTED_TIME = "distributedTimeCoefficients"; - protected boolean readTimeFactors; - public static final String PERSON_TIMEFACTOR_NONWORK_FIELD_NAME = "timeFactorNonWork"; - - InternalExternalModelStructure modelStructure; - - TableDataSet personData; - - private boolean seek; - private int traceId; - - private MersenneTwister random; - - private class HouseholdClass - { - - int autos; - int income; - int homeMGRA; - int autonomousVehicles; - } - - private HashMap householdData; - - /** - * Constructor. Reads properties file and opens/stores all probability - * distributions for sampling. Estimates number of airport travel parties - * and initializes parties[]. - * - * @param resourceFile - * Property file. - * - * Creates the array of cross-border tours. - */ - public InternalExternalTourManager(HashMap rbMap, int iteration) - { - - modelStructure = new InternalExternalModelStructure(); - - // append _iteration to file - String iterationString = "_" + new Integer(iteration).toString(); - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - - String personFile = Util.getStringValueFromPropertyMap(rbMap, "Results.PersonDataFile"); - // Remove extension from filename - String extension = getFileExtension(personFile); - personFile = removeFileExtension(personFile) + iterationString + extension; - - personFile = directory + personFile; - - String householdFile = Util.getStringValueFromPropertyMap(rbMap, - "Results.HouseholdDataFile"); - - householdFile = directory + householdFile; - // Remove extension from filename - extension = getFileExtension(householdFile); - householdFile = removeFileExtension(householdFile) + iterationString + extension; - - readHouseholdFile(householdFile); - personData = readFile(personFile); - - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trace")); - - random = new MersenneTwister(1000001); - //check if we want to read distributed time factors from the person file - String readTimeFactorsString = rbMap.get(PROPERTIES_DISTRIBUTED_TIME); - if (readTimeFactorsString != null) - { - readTimeFactors = Boolean.valueOf(readTimeFactorsString); - logger.info("Distributed time coefficients = "+Boolean.toString(readTimeFactors)); - } - - } - - /** - * Get the file extension - * - * @param fileName - * with the extension - * @return The extension - */ - public String getFileExtension(String fileName) - { - - int index = fileName.lastIndexOf("."); - int length = fileName.length(); - - String extension = fileName.substring(index, length); - - return extension; - - } - - /** - * Get the file name without the extension - * - * @param fileName - * The filename with the extension - * @return The filename without the extension - */ - public String removeFileExtension(String fileName) - { - int index = fileName.lastIndexOf("."); - String name = fileName.substring(0, index); - - return name; - - } - - /** - * Read household records and store autos owned. - * - * @param fileName - * household file path/name. - */ - public void readHouseholdFile(String fileName) - { - - householdData = new HashMap(); - - logger.info("Begin reading the data in file " + fileName); - - TableDataSet hhData; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - hhData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - // iterate through the table and save number of autos - for (int i = 1; i <= hhData.getRowCount(); ++i) - { - long hhID = (long) hhData.getValueAt(i, "hh_id"); - int autos = (int) hhData.getValueAt(i, "autos"); - int income = (int) hhData.getValueAt(i, "income"); - int mgra = (int) hhData.getValueAt(i, "home_mgra"); - - int AVs = (int) hhData.getValueAt(i,"AVs"); - - // new household - HouseholdClass hh = new HouseholdClass(); - hh.autos = autos; - hh.income = income; - hh.homeMGRA = mgra; - hh.autonomousVehicles = AVs; - - // store in HashMap - householdData.put(hhID, hh); - } - logger.info("End reading the data in file " + fileName); - } - - /** - * Read the file and return the TableDataSet. - * - * @param fileName - * @return data - */ - private TableDataSet readFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet data; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - data = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return data; - } - - /** - * Generate and attribute IE tours - */ - public void generateTours() - { - - ArrayList tourList = new ArrayList(); - - int rows = personData.getRowCount(); - - int tourCount = 0; - for (int i = 1; i <= rows; ++i) - { - - // TODO: generate IE tours here - if (((int) personData.getValueAt(i, "ie_choice")) == 2) - { - - InternalExternalTour tour = new InternalExternalTour(i + 100001); - tour.setID(i); - - // get the household for the person - long ID = (long) personData.getValueAt(i, "hh_id"); - HouseholdClass hh = householdData.get(ID); - tour.setHhID((int)ID); - - int pID = (int) personData.getValueAt(i, "person_id"); - tour.setPersonID(pID); - - int pnum=(int) personData.getValueAt(i, "person_num"); - tour.setPnum(pnum); - - int age = (int) personData.getValueAt(i, "age"); - String gender = (String) personData.getStringValueAt(i, "gender"); - - tour.setOriginMGRA(hh.homeMGRA); - tour.setIncome(hh.income); - tour.setAutos(hh.autos); - tour.setAge(age); - - if(hh.autonomousVehicles>0) - tour.setAvAvailable(true); - else - tour.setAvAvailable(false); - - if (gender.equals("f")) tour.setFemale(1); - else tour.setFemale(0); - - double timeFactorNonWork = 1.0; - if(readTimeFactors){ - timeFactorNonWork = (double) personData.getValueAt(i, - personData.getColumnPosition(PERSON_TIMEFACTOR_NONWORK_FIELD_NAME)); - } - tour.setNonWorkTimeFactor(timeFactorNonWork); - - tourList.add(tour); - - ++tourCount; - } - - } - if (tourList.isEmpty()) - { - logger.error("Internal-external tour list is empty!!"); - throw new RuntimeException(); - } - - tours = new InternalExternalTour[tourList.size()]; - for (int i = 0; i < tours.length; ++i) - tours[i] = tourList.get(i); - - logger.info("Total IE tours: " + tourCount); - - } - - /** - * Create a text file and write all records to the file. - * - */ - public void writeOutputFile(HashMap rbMap) - { - - // Open file and print header - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String tripFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "internalExternal.trip.output.file"); - - logger.info("Writing IE trips to file " + tripFileName); - - PrintWriter tripWriter = null; - try - { - tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tripFileName + " for writing\n"); - throw new RuntimeException(); - } - String tripHeaderString = new String( - "hhID,pnum,personID,tourID,originMGRA,destinationMGRA,originTAZ,destinationTAZ,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,av_avail,boardingTap,alightingTap,set,valueOfTime\n"); - tripWriter.print(tripHeaderString); - - for (int i = 0; i < tours.length; ++i) - { - InternalExternalTrip[] trips = tours[i].getTrips(); - for (int j = 0; j < trips.length; ++j) - writeTrip(tours[i].getHhID(), tours[i].getPnum(),tours[i].getPersonID(), tours[i].getID(), tours[i],trips[j], tripWriter); - } - - tripWriter.close(); - - } - - /** - * Write the trip to the PrintWriter - * - * @param tour - * @param trip - * @param tripNumber - * @param writer - */ - private void writeTrip(int hhID, int pnum, int personID, int tourID, InternalExternalTour tour, InternalExternalTrip trip, PrintWriter writer) - { - - String record = new String(hhID+","+pnum+","+personID+","+tourID+","+trip.getOriginMgra() + "," + trip.getDestinationMgra() + "," - + trip.getOriginTaz() + "," + trip.getDestinationTaz() + "," + trip.isInbound() - + "," + trip.isOriginIsTourDestination() + "," - + trip.isDestinationIsTourDestination() + "," + trip.getPeriod() + "," - + trip.getTripMode() + "," + (tour.isAvAvailable() ? 1 : 0) + "," - + trip.getBoardTap() + "," + trip.getAlightTap() + "," + trip.getSet()+ "," - +String.format("%9.2f",trip.getValueOfTime()) + "\n"); - writer.print(record); - } - - /** - * @return the trips - */ - public InternalExternalTour[] getTours() - { - return tours; - } - - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running IE Model Trip Manager")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - InternalExternalTourManager apm = new InternalExternalTourManager(pMap, 1); - apm.generateTours(); - apm.writeOutputFile(pMap); - - logger.info("IE Trip Manager successfully completed!"); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java deleted file mode 100644 index 189a548..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTourTimeOfDayChoiceModel.java +++ /dev/null @@ -1,182 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the TOD choice model for IE tours. It is currently based on a - * static probability distribution stored in an input file, and indexed into by - * purpose. Since there are no IE purposes, the purpose is 0. - * - * @author Freedman - * - */ -public class InternalExternalTourTimeOfDayChoiceModel -{ - private transient Logger logger = Logger.getLogger("internalExternalModel"); - - private double[][] cumProbability; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - private int[][] outboundPeriod; // by - // purpose, - // alternative: - // outbound - // period - private int[][] returnPeriod; // by - // purpose, - // alternative: - // return - // period - InternalExternalModelStructure modelStructure; - - /** - * Constructor. - */ - public InternalExternalTourTimeOfDayChoiceModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.tour.tod.file"); - stationDiurnalFile = directory + stationDiurnalFile; - - modelStructure = new InternalExternalModelStructure(); - - readTODFile(stationDiurnalFile); - - } - - /** - * Read the TOD distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readTODFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating tour TOD probability distribution"); - - int purposes = 1; // start at 0 - int periods = modelStructure.TIME_PERIODS; // start at 1 - int periodCombinations = periods * (periods + 1) / 2; - - cumProbability = new double[purposes][periodCombinations]; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - outboundPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // outbound - // period - returnPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // return period - - // fill up arrays - int rowCount = probabilityTable.getRowCount(); - int lastPurpose = -99; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); - int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); - - // continue if return period before outbound period - if (retPer < outPer) continue; - - // reset if new purpose - if (purpose != lastPurpose) - { - - // log cumulative probability just in case - if (lastPurpose != -99) - logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); - cumProb = 0; - alt = 0; - } - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - outboundPeriod[purpose][alt] = outPer; - returnPeriod[purpose][alt] = retPer; - - ++alt; - - lastPurpose = purpose; - } - - logger.info("End calculating tour TOD probability distribution"); - - } - - /** - * Calculate tour time of day for the tour. - * - * @param tour - * An IE tour - */ - public void calculateTourTOD(InternalExternalTour tour) - { - - int purpose = 0; - double random = tour.getRandom(); - - if (tour.getDebugChoiceModels()) - { - logger.info("Choosing tour time of day for tour ID " + tour.getID() - + " using random number " + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - if (random < cumProbability[purpose][i]) - { - int depart = outboundPeriod[purpose][i]; - int arrive = returnPeriod[purpose][i]; - tour.setDepartTime(depart); - tour.setArriveTime(arrive); - break; - } - } - - if (tour.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " - + tour.getArriveTime()); - logger.info(""); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java deleted file mode 100644 index c347392..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTrip.java +++ /dev/null @@ -1,313 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.Serializable; -import org.sandag.abm.modechoice.MgraDataManager; - -public class InternalExternalTrip - implements Serializable -{ - - private int originMgra; - private int destinationMgra; - - - private int originTaz; - private int destinationTaz; - - private int tripMode; - private byte period; - private boolean inbound; - private boolean firstTrip; - private boolean lastTrip; - private boolean originIsTourDestination; - private boolean destinationIsTourDestination; - - private int boardTap; - private int alightTap; - private int set = -1; - - private double valueOfTime; - - /** - * Default constructor; nothing initialized. - */ - public InternalExternalTrip() - { - - } - - /** - * Create a cross border trip from a tour leg (no stops). - * - * @param tour - * The tour. - * @param outbound - * Outbound direction - */ - public InternalExternalTrip(InternalExternalTour tour, boolean outbound, - MgraDataManager mgraManager) - { - - initializeFromTour(tour, outbound, mgraManager); - } - - /** - * Initilize from the tour. - * - * @param tour - * The tour. - * @param outbound - * Outbound direction. - */ - public void initializeFromTour(InternalExternalTour tour, boolean outbound, - MgraDataManager mgraManager) - { - // Note: mode is unknown - if (outbound) - { - this.originMgra = tour.getOriginMGRA(); - this.originTaz = mgraManager.getTaz(tour.getOriginMGRA()); - this.destinationMgra = tour.getDestinationMGRA(); - this.destinationTaz = tour.getDestinationTAZ(); - this.period = (byte) tour.getDepartTime(); - this.inbound = false; - this.firstTrip = true; - this.lastTrip = false; - this.originIsTourDestination = false; - this.destinationIsTourDestination = true; - } else - { - this.originMgra = tour.getDestinationMGRA(); - this.originTaz = tour.getDestinationTAZ(); - this.destinationMgra = tour.getOriginMGRA(); - this.destinationTaz = mgraManager.getTaz(tour.getOriginMGRA()); - this.period = (byte) tour.getArriveTime(); - this.inbound = true; - this.firstTrip = false; - this.lastTrip = true; - this.originIsTourDestination = true; - this.destinationIsTourDestination = false; - } - - } - /** - * @param destinationTaz - * the destinationTaz to set - */ - public void setDestinationTaz(int destinationTaz) - { - this.destinationTaz = destinationTaz; - } - - - /** - * @return the period - */ - public byte getPeriod() - { - return period; - } - - /** - * @param period - * the period to set - */ - public void setPeriod(byte period) - { - this.period = period; - } - - /** - * @return the originMgra - */ - public int getOriginMgra() - { - return originMgra; - } - - /** - * @param originMgra - * the originMgra to set - */ - public void setOriginMgra(int originMgra) - { - this.originMgra = originMgra; - } - - /** - * @return the destinationMgra - */ - public int getDestinationMgra() - { - return destinationMgra; - } - - /** - * @param destinationMgra - * the destinationMgra to set - */ - public void setDestinationMgra(int destinationMgra) - { - this.destinationMgra = destinationMgra; - } - - /** - * @return the tripMode - */ - public int getTripMode() - { - return tripMode; - } - - /** - * @param tripMode - * the tripMode to set - */ - public void setTripMode(int tripMode) - { - this.tripMode = tripMode; - } - public int getBoardTap() { - return boardTap; - } - - public void setBoardTap(int boardTap) { - this.boardTap = boardTap; - } - - public int getAlightTap() { - return alightTap; - } - - public void setAlightTap(int alightTap) { - this.alightTap = alightTap; - } - - public int getSet() { - return set; - } - - public void setSet(int set) { - this.set = set; - } - - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the firstTrip - */ - public boolean isFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(boolean firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public boolean isLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(boolean lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the originIsTourDestination - */ - public boolean isOriginIsTourDestination() - { - return originIsTourDestination; - } - - /** - * @param originIsTourDestination - * the originIsTourDestination to set - */ - public void setOriginIsTourDestination(boolean originIsTourDestination) - { - this.originIsTourDestination = originIsTourDestination; - } - - /** - * @return the destinationIsTourDestination - */ - public boolean isDestinationIsTourDestination() - { - return destinationIsTourDestination; - } - - /** - * @param destinationIsTourDestination - * the destinationIsTourDestination to set - */ - public void setDestinationIsTourDestination(boolean destinationIsTourDestination) - { - this.destinationIsTourDestination = destinationIsTourDestination; - } - - /** - * @return the originTaz - */ - public int getOriginTaz() - { - return originTaz; - } - - /** - * @param originTaz - * the originTaz to set - */ - public void setOriginTaz(int originTaz) - { - this.originTaz = originTaz; - } - - /** - * @return the destinationTaz - */ - public int getDestinationTaz() - { - return destinationTaz; - } - - public double getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(double valueOfTime) { - this.valueOfTime = valueOfTime; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java deleted file mode 100644 index 1f1f864..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceDMU.java +++ /dev/null @@ -1,551 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class InternalExternalTripModeChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(InternalExternalTripModeChoiceDMU.class); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - - protected int tourDepartPeriod; - protected int tourArrivePeriod; - protected int tripPeriod; - protected int outboundStops; - protected int returnStops; - protected int firstTrip; - protected int lastTrip; - - protected int income; - protected int female; - protected int age; - protected int autos; - protected int hhSize; - protected int tripOrigIsTourDest; - protected int tripDestIsTourDest; - - protected double nonWorkTimeFactor; - - protected double nmWalkTime; - protected double nmBikeTime; - - - protected double ivtCoeff; - protected double costCoeff; - - protected double walkTransitLogsum; - protected double pnrTransitLogsum; - protected double knrTransitLogsum; - - protected int outboundHalfTourDirection; - - protected int avAvailable; - - public InternalExternalTripModeChoiceDMU(InternalExternalModelStructure modelStructure, - Logger aLogger) - { - if (aLogger == null) - { - aLogger = Logger.getLogger("internalExternalModel"); - } - logger = aLogger; - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the tripPeriod - */ - public int getTripPeriod() - { - return tripPeriod; - } - - /** - * @param tripPeriod - * the tripPeriod to set - */ - public void setTripPeriod(int tripPeriod) - { - this.tripPeriod = tripPeriod; - } - - /** - * @return the outboundStops - */ - public int getOutboundStops() - { - return outboundStops; - } - - /** - * @param outboundStops - * the outboundStops to set - */ - public void setOutboundStops(int outboundStops) - { - this.outboundStops = outboundStops; - } - - /** - * @return the returnStops - */ - public int getReturnStops() - { - return returnStops; - } - - /** - * @param returnStops - * the returnStops to set - */ - public void setReturnStops(int returnStops) - { - this.returnStops = returnStops; - } - - /** - * @return the firstTrip - */ - public int getFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(int firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public int getLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(int lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the tripOrigIsTourDest - */ - public int getTripOrigIsTourDest() - { - return tripOrigIsTourDest; - } - - /** - * @param tripOrigIsTourDest - * the tripOrigIsTourDest to set - */ - public void setTripOrigIsTourDest(int tripOrigIsTourDest) - { - this.tripOrigIsTourDest = tripOrigIsTourDest; - } - - /** - * @return the tripDestIsTourDest - */ - public int getTripDestIsTourDest() - { - return tripDestIsTourDest; - } - - /** - * @param tripDestIsTourDest - * the tripDestIsTourDest to set - */ - public void setTripDestIsTourDest(int tripDestIsTourDest) - { - this.tripDestIsTourDest = tripDestIsTourDest; - } - - /** - * @return the outboundHalfTourDirection - */ - public int getOutboundHalfTourDirection() - { - return outboundHalfTourDirection; - } - - /** - * @param outboundHalfTourDirection - * the outboundHalfTourDirection to set - */ - public void setOutboundHalfTourDirection(int outboundHalfTourDirection) - { - this.outboundHalfTourDirection = outboundHalfTourDirection; - } - - /** - * @return the tourDepartPeriod - */ - public int getTourDepartPeriod() - { - return tourDepartPeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(int tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(int tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - /** - * @return the tourArrivePeriod - */ - public int getTourArrivePeriod() - { - return tourArrivePeriod; - } - - public double getNm_walkTime() - { - return nmWalkTime; - } - - public void setNonMotorizedWalkTime(double nmWalkTime) - { - this.nmWalkTime = nmWalkTime; - } - - public void setNonMotorizedBikeTime(double nmBikeTime) - { - this.nmBikeTime = nmBikeTime; - } - - public double getNm_bikeTime() - { - return nmBikeTime; - } - - /** - * @return the income - */ - public int getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(int income) - { - this.income = income; - } - - public int getFemale() - { - return female; - } - - public void setFemale(int female) - { - this.female = female; - } - - public int getAge() - { - return age; - } - - public void setAge(int age) - { - this.age = age; - } - - public int getAutos() - { - return autos; - } - - public void setAutos(int autos) - { - this.autos = autos; - } - - public int getHhSize() - { - return hhSize; - } - - public void setHhSize(int hhSize) - { - this.hhSize = hhSize; - } - public double getNonWorkTimeFactor(){ - return nonWorkTimeFactor; - } - - public void setNonWorkTimeFactor(double nonWorkTimeFactor){ - this.nonWorkTimeFactor=nonWorkTimeFactor; - } - - public double getIvtCoeff() { - return ivtCoeff; - } - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public double getCostCoeff() { - return costCoeff; - } - - public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; - } - - public double getWalkTransitLogsum() { - return walkTransitLogsum; - } - - public void setWalkTransitLogsum(double walkTransitLogsum) { - this.walkTransitLogsum = walkTransitLogsum; - } - - public double getPnrTransitLogsum() { - return pnrTransitLogsum; - } - - public void setPnrTransitLogsum(double pnrTransitLogsum) { - this.pnrTransitLogsum = pnrTransitLogsum; - } - - public double getKnrTransitLogsum() { - return knrTransitLogsum; - } - - public void setKnrTransitLogsum(double knrTransitLogsum) { - this.knrTransitLogsum = knrTransitLogsum; - } - - - public int getAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(int avAvailable) { - this.avAvailable = avAvailable; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTourDepartPeriod", 0); - methodIndexMap.put("getTourArrivePeriod", 1); - methodIndexMap.put("getTripPeriod", 2); - methodIndexMap.put("getOutboundStops", 5); - methodIndexMap.put("getReturnStops", 6); - methodIndexMap.put("getFirstTrip", 7); - methodIndexMap.put("getLastTrip", 8); - methodIndexMap.put("getIncome", 9); - methodIndexMap.put("getFemale", 10); - methodIndexMap.put("getAutos", 11); - methodIndexMap.put("getHhSize", 12); - methodIndexMap.put("getAge", 13); - methodIndexMap.put("getNonWorkTimeFactor", 14); - - methodIndexMap.put("getTripOrigIsTourDest", 23); - methodIndexMap.put("getTripDestIsTourDest", 24); - - methodIndexMap.put("getIvtCoeff", 60); - methodIndexMap.put("getCostCoeff", 61); - - methodIndexMap.put("getWalkSetLogSum", 62); - methodIndexMap.put("getPnrSetLogSum", 63); - methodIndexMap.put("getKnrSetLogSum", 64); - - methodIndexMap.put("getAvAvailable",70); - - methodIndexMap.put("getNm_walkTime", 90); - methodIndexMap.put("getNm_bikeTime", 91); - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTourDepartPeriod(); - break; - case 1: - returnValue = getTourArrivePeriod(); - break; - case 2: - returnValue = getTripPeriod(); - break; - case 5: - returnValue = getOutboundStops(); - break; - case 6: - returnValue = getReturnStops(); - break; - case 7: - returnValue = getFirstTrip(); - break; - case 8: - returnValue = getLastTrip(); - break; - case 9: - returnValue = getIncome(); - break; - case 10: - returnValue = getFemale(); - break; - case 11: - returnValue = getAutos(); - break; - case 12: - returnValue = getHhSize(); - break; - case 13: - returnValue = getAge(); - break; - case 14: - returnValue = getNonWorkTimeFactor(); - break; - case 23: - returnValue = getTripOrigIsTourDest(); - break; - case 24: - returnValue = getTripDestIsTourDest(); - break; - - case 60: - returnValue = getIvtCoeff(); - break; - case 61: - returnValue = getCostCoeff(); - break; - case 62: - returnValue = getWalkTransitLogsum(); - break; - case 63: - returnValue = getPnrTransitLogsum(); - break; - case 64: - returnValue = getKnrTransitLogsum(); - break; - case 70: - returnValue = getAvAvailable(); - break; - case 90: - returnValue = getNm_walkTime(); - break; - case 91: - returnValue = getNm_bikeTime(); - break; - default: - logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - } - - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java deleted file mode 100644 index f9dc39d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripModeChoiceModel.java +++ /dev/null @@ -1,277 +0,0 @@ -package org.sandag.abm.internalexternal; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.airport.AirportModelStructure; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class InternalExternalTripModeChoiceModel -{ - - private transient Logger logger = Logger.getLogger("internalExternalModel"); - - private AutoTazSkimsCalculator tazDistanceCalculator; - - private McLogsumsCalculator logsumHelper; - private InternalExternalModelStructure modelStructure; - private TazDataManager tazs; - private MgraDataManager mgraManager; - private InternalExternalTripModeChoiceDMU dmu; - private ChoiceModelApplication tripModeChoiceModel; - double logsum = 0; - - private static final String PROPERTIES_UEC_DATA_SHEET = "internalExternal.trip.mc.data.page"; - private static final String PROPERTIES_UEC_MODEL_SHEET = "internalExternal.trip.mc.model.page"; - private static final String PROPERTIES_UEC_FILE = "internalExternal.trip.mc.uec.file"; - private TripModeChoiceDMU mcDmuObject; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public InternalExternalTripModeChoiceModel(HashMap propertyMap, - InternalExternalModelStructure myModelStructure, - InternalExternalDmuFactoryIf dmuFactory) - { - tazs = TazDataManager.getInstance(propertyMap); - mgraManager = MgraDataManager.getInstance(propertyMap); - - modelStructure = myModelStructure; - - tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - SandagModelStructure modelStructure = new SandagModelStructure(); - mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); - - setupTripModeChoiceModel(propertyMap, dmuFactory); - - } - - /** - * Read the UEC file and set up the trip mode choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupTripModeChoiceModel(HashMap propertyMap, - InternalExternalDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up IE trip mode choice model.")); - - dmu = dmuFactory.getInternalExternalTripModeChoiceDMU(); - - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_DATA_SHEET)); - int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_MODEL_SHEET)); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); - tripModeUecFile = uecPath + tripModeUecFile; - - tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, - propertyMap, (VariableTable) dmu); - - } - - /** - * Calculate utilities and return logsum for the tour and stop. - * - * @param tour - * @param trip - */ - public double computeUtilities(InternalExternalTour tour, InternalExternalTrip trip) - { - - setDmuAttributes(tour, trip); - - tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - tour.logTourObject(logger, 100); - tripModeChoiceModel.logUECResults(logger, "IE trip mode choice model"); - - } - - logsum = tripModeChoiceModel.getLogsum(); - - if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); - - return logsum; - - } - - /** - * Choose a mode and store in the trip object. - * - * @param tour - * InternalExternalTour - * @param trip - * InternalExternalTrip - * - */ - public void chooseMode(InternalExternalTour tour, InternalExternalTrip trip) - { - - computeUtilities(tour, trip); - - double rand = tour.getRandom(); - int mode = tripModeChoiceModel.getChoiceResult(rand); - - trip.setTripMode(mode); - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - - double vot = 0.0; - - if(modelStructure.getTripModeIsS2(mode)){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = uec.getValueForIndex(votIndex); - }else if (modelStructure.getTripModeIsS3(mode)){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = uec.getValueForIndex(votIndex); - } - trip.setValueOfTime(vot); - - - if(modelStructure.getTripModeIsTransit(mode)){ - double[][] bestTapPairs = null; - - if (modelStructure.getTripModeIsWalkTransit(mode)){ - bestTapPairs = logsumHelper.getBestWtwTripTaps(); - } - else if (modelStructure.getTripModeIsPnrTransit(mode)||modelStructure.getTripModeIsKnrTransit(mode)){ - if (!trip.isInbound()) - bestTapPairs = logsumHelper.getBestDtwTripTaps(); - else - bestTapPairs = logsumHelper.getBestWtdTripTaps(); - } - double rn = tour.getRandom(); - int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); - int boardTap = (int) bestTapPairs[pathIndex][0]; - int alightTap = (int) bestTapPairs[pathIndex][1]; - int set = (int) bestTapPairs[pathIndex][2]; - trip.setBoardTap(boardTap); - trip.setAlightTap(alightTap); - trip.setSet(set); - } - - - - - - } - - /** - * Set DMU attributes. - * - * @param tour - * @param trip - */ - public void setDmuAttributes(InternalExternalTour tour, InternalExternalTrip trip) - { - - int tripOriginTaz = trip.getOriginTaz(); - int tripDestinationTaz = trip.getDestinationTaz(); - - dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, - tour.getDebugChoiceModels()); - - dmu.setTourDepartPeriod(tour.getDepartTime()); - dmu.setTourArrivePeriod(tour.getArriveTime()); - dmu.setTripPeriod(trip.getPeriod()); - - dmu.setAutos(tour.getAutos()); - dmu.setIncome(tour.getIncome()); - dmu.setAge(tour.getAge()); - dmu.setFemale(tour.getFemale()); - - dmu.setNonWorkTimeFactor(tour.getNonWorkTimeFactor()); - - // set trip mc dmu values for transit logsum (gets replaced below by uec values) - double c_ivt = -0.03; - double c_cost = - 0.003; - - // Solve trip mode level utilities - mcDmuObject.setIvtCoeff(c_ivt * tour.getNonWorkTimeFactor()); - mcDmuObject.setCostCoeff(c_cost); - - dmu.setIvtCoeff(c_ivt * tour.getNonWorkTimeFactor()); - dmu.setCostCoeff(c_cost); - double walkTransitLogsum = -999.0; - double driveTransitLogsum = -999.0; - - logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); - dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); - - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); - walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - dmu.setWalkTransitLogsum(walkTransitLogsum); - if (!trip.isInbound()) - { - logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); - } else - { - logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); - } - - dmu.setPnrTransitLogsum(driveTransitLogsum); - dmu.setKnrTransitLogsum(driveTransitLogsum); - - dmu.setOutboundStops(tour.getNumberInboundStops()); - dmu.setReturnStops(tour.getNumberInboundStops()); - - if (trip.isFirstTrip()) dmu.setFirstTrip(1); - else dmu.setFirstTrip(0); - - if (trip.isLastTrip()) dmu.setLastTrip(1); - else dmu.setLastTrip(0); - - if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); - else dmu.setTripOrigIsTourDest(0); - - if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); - else dmu.setTripDestIsTourDest(0); - - if(tour.isAvAvailable()) - dmu.setAvAvailable(1); - else - dmu.setAvAvailable(0); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java deleted file mode 100644 index 3594539..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/internalexternal/InternalExternalTripTables.java +++ /dev/null @@ -1,698 +0,0 @@ -package org.sandag.abm.internalexternal; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class InternalExternalTripTables -{ - - private static Logger logger = Logger.getLogger("tripTables"); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TableDataSet tripData; - - // Some parameters - private int[] modeIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // total - // modes, - // returns - // 0=auto - // modes, - // 1=non-motor, - // 2=transit, - // 3= - // other - private int[] matrixIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // modes, - // returns - // the - // element - // of - // the - // matrix - // array - // to - // store - // value - - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private HashMap rbMap; - private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; - private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; - - // matrices are indexed by modes, votbins, tables - private Matrix[][][] matrix; - private float averageOcc3Plus = 3.5f; - - private ResourceBundle rb; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - - private MatrixDataServerRmi ms; - private float sampleRate = 1; - private static int iteration=1; - private float valueOfTimeThresholdLow = 0; - private float valueOfTimeThresholdMed = 0; - //value of time bins by mode group - int[] votBins = {3,1,1,1}; - - public int numSkimSets; - - - /** - * @return the sampleRate - */ - public float getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(float sampleRate) - { - this.sampleRate = sampleRate; - } - - public InternalExternalTripTables(HashMap rbMap) - { - - this.rbMap = rbMap; - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTourModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - } else if (modelStructure.getTourModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - } else if (modelStructure.getTourModeIsWalkTransit(i) - || modelStructure.getTourModeIsDriveTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - } - } - //value of time thresholds - valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); - valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][][]; - for (int i = 0; i < numberOfModes; ++i) - { - matrix[i] = new Matrix[votBins[i]][]; - - String modeName; - for(int j = 0; j< votBins[i];++j){ - - if (i == 0) - { - matrix[i][j] = new Matrix[autoModes]; - for (int k = 0; k < autoModes; ++k) - { - modeName = modelStructure.getModeName(k + 1); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 1) - { - matrix[i][j] = new Matrix[nmotModes]; - for (int k = 0; k < nmotModes; ++k) - { - modeName = modelStructure.getModeName(k + 1 + autoModes); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 2) - { - matrix[i][j] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l1) - votBin = getValueOfTimeBin(valueOfTime); - - if (mode == 0) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); - } else if (mode == 1) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } else if (mode == 2) - { - - if (boardTap == 0 || alightTap == 0) continue; - - //store transit trips in matrices - mat = (matrixIndex[tripMode]*numSkimSets)+set; - float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); - matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); - - // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) - if (modelStructure.getTourModeIsDriveTransit(tripMode)) - { - - // add the tNCVehicle trip portion to the trip table - if (!inbound) - { // from origin to lot (boarding tap) - int PNRTAZ = tapManager.getTazForTap(boardTap); - value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); - matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); - - } else - { // from lot (alighting tap) to destination - int PNRTAZ = tapManager.getTazForTap(alightTap); - value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); - matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); - } - - } - } else - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } - - //logger.info("End creating trip tables for period " + timePeriod); - } - } - - /** - * Get the output trip table file names from the properties file, and write - * trip tables for all modes for the given time period. - * - * @param period - * Time period, which will be used to find the period time string - * to append to each trip table matrix file - */ - public void writeTrips(int period, MatrixType mt) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "scenario.path"); - String per = modelStructure.getModelPeriodLabel(period); - String[][] end = new String[4][]; - String[] fileName = new String[4]; - - fileName[0] = directory - + Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.results.autoTripMatrix"); - fileName[1] = directory - + Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.results.nMotTripMatrix"); - fileName[2] = directory - + Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.results.tranTripMatrix"); - fileName[3] = directory - + Util.getStringValueFromPropertyMap(rbMap, - "internalExternal.results.othrTripMatrix"); - - //the end of the name depends on whether there are multiple vot bins or not - String[] votBinName = {"low","med","high"}; - - for(int i = 0; i<4;++i){ - end[i] = new String[votBins[i]]; - for(int j = 0; j < votBins[i];++j){ - if(votBins[i]>1) - end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; - else - end[i][j] = "_" + per + ".omx"; - } - } - - for (int i = 0; i < 4; ++i) - { - for(int j = 0; j < votBins[i];++j){ - try - { - //Delete the file if it exists - File f = new File(fileName[i]+end[i][j]); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); - f.delete(); - } - - if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); - else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); - } catch (Exception e) - { - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName[i] +end[i][j] + ", for mode index = " + i, e); - throw new RuntimeException(); - } - } - } - } - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - /** - * Start matrix server - * - * @param serverAddress - * @param serverPort - * @param mt - * @return - */ - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @param args - */ - public static void main(String[] args) - { - - HashMap pMap; - String propertiesFile = null; - - logger.info(String.format( - "SANDAG IE Model Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - InternalExternalTripTables tripTables = new InternalExternalTripTables(pMap); - float sampleRate = 1.0f; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - - logger.info("IE Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - tripTables.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - tripTables.ms = matrixServer; - } else - { - tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - tripTables.ms.testRemote("InternalExternalTripTables"); - - // mdm = MatrixDataManager.getInstance(); - // mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - tripTables.createTripTables(mt); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java deleted file mode 100644 index 120dbef..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationManager.java +++ /dev/null @@ -1,1496 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Iterator; -import java.util.Set; -import java.util.StringTokenizer; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.PropertyMap; - -public class HouseholdAVAllocationManager { - - HashMap householdMap; //by hh_id - private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModelRunner.class); - protected HashMap propertyMap = null; - protected MersenneTwister random; - protected ModelStructure modelStructure; - protected int iteration; - protected static final String ModelSeedProperty = "Model.Random.Seed"; - protected static final String DirectoryProperty = "Project.Directory"; - protected static final String HouseholdDataFileProperty = "Results.HouseholdDataFile"; - protected static final String PersonDataFileProperty = "Results.PersonDataFile"; - protected static final String IndivTripDataFileProperty = "Results.IndivTripDataFile"; - protected static final String JointTripDataFileProperty = "Results.JointTripDataFile"; - protected static final String VEHICLETRIP_OUTPUT_FILE_PROPERTY = "Maas.AVAllocationModel.vehicletrip.output.file"; - protected static final String VEHICLETRIP_OUTPUT_MATRIX_PROPERTY = "Maas.AVAllocationModel.vehicletrip.output.matrix"; - protected static final String REMOTE_PARKING_COST_PROPERTY = "Mobility.AV.RemoteParkingCostPerHour"; - - protected HashSet householdTraceSet; - public static final String PROPERTIES_HOUSEHOLD_TRACE_LIST = "Debug.Trace.HouseholdIdList"; - // one file per time period - // matrices are indexed by periods - private Matrix[] emptyVehicleTripMatrix; - MgraDataManager mazManager; - TazDataManager tazManager; - - protected static final int[] AutoModes = {1,2,3}; - protected static final int MaxAutoMode = 3; - private long randomSeed = 198761; - protected String vehicleTripOutputFile; - - protected float remoteParkingCostAtDest; - - boolean sortByPerson; - - HashMap personTypeMap; - String[] personTypes = {"Full-time worker","Part-time worker","University student", - "Non-worker", "Retired","Student of driving age","Student of non-driving age", - "Child too young for school"}; - - class Household { - - HashMap personMap; //by person_num - int id; - int homeMaz; - int income; - int autos; - int HVs; - int AVs; - ArrayList trips; - ArrayList autonomousVehicles; - int seed; - boolean debug; - - public void writeDebug(Logger logger, boolean logAVs) { - - logger.info("******** HH DEBUG **************"); - logger.info("HH ID: "+ id); - logger.info("Home MAZ: "+homeMaz); - logger.info("Income: "+income); - logger.info("Autos: "+ autos); - logger.info("HVs: "+HVs); - logger.info("AVs: "+AVs); - logger.info("Seed: "+seed); - - //log persons - if(personMap.size()==0) { - logger.info(" No persons to log"); - }else { - Set keySet = personMap.keySet(); - for(Integer key: keySet) { - Person person = personMap.get(key); - person.writeDebug(logger); - } - } - - //log trips - if(trips.size()==0) { - logger.info(" No trips to log"); - }else { - - for(int i=0;i0 && logAVs) { - for(int i=0;i(); - personMap = new HashMap(); - autonomousVehicles = new ArrayList(); - - } - - public ArrayList getTrips() { - return trips; - } - public void setTrips(ArrayList trips) { - this.trips = trips; - } - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public int getHomeMaz() { - return homeMaz; - } - - public void setHomeMaz(int homeMaz) { - this.homeMaz = homeMaz; - } - - public int getIncome() { - return income; - } - - public void setIncome(int income) { - this.income = income; - } - - public int getAutos() { - return autos; - } - - public void setAutos(int autos) { - this.autos = autos; - } - - public int getHVs() { - return HVs; - } - - public void setHVs(int hVs) { - HVs = hVs; - } - - public int getAVs() { - return AVs; - } - - public void setAVs(int aVs) { - AVs = aVs; - } - - public ArrayList getAutonomousVehicles() { - return autonomousVehicles; - } - - public void setAutonomousVehicles(ArrayList autonomousVehicles) { - this.autonomousVehicles = autonomousVehicles; - } - - public int getSeed() { - return seed; - } - - public void setSeed(int seed) { - this.seed = seed; - } - - public boolean isDebug() { - return debug; - } - - public void setDebug(boolean debug) { - this.debug = debug; - } - - - } - - class Person { - - int hh_id; - int person_id; - int person_num; - int age; - int gender; - int type; - float value_of_time; - float reimb_pct; - float timeFactorWork; - float timeFactorNonWork; - int fp_choice; - - public void writeDebug(Logger logger) { - - logger.info("******** PERSON DEBUG **************"); - logger.info("HH ID: "+ hh_id); - logger.info("Person ID: "+person_id); - logger.info("Person Num: "+person_num); - logger.info("Age: "+age); - logger.info("Gender: "+gender); - logger.info("Type: "+type); - logger.info("Value of time: "+value_of_time); - logger.info("Reimb percent: "+reimb_pct); - logger.info("Time factor work: "+timeFactorWork); - logger.info("Time factor nonwork: "+timeFactorNonWork); - logger.info("Free parking choice: "+fp_choice); - - } - - - public int getHh_id() { - return hh_id; - } - public void setHh_id(int hh_id) { - this.hh_id = hh_id; - } - public int getPerson_id() { - return person_id; - } - public void setPerson_id(int person_id) { - this.person_id = person_id; - } - public int getPerson_num() { - return person_num; - } - public void setPerson_num(int person_num) { - this.person_num = person_num; - } - public int getAge() { - return age; - } - public void setAge(int age) { - this.age = age; - } - public int getGender() { - return gender; - } - public void setGender(int gender) { - this.gender = gender; - } - public int getType() { - return type; - } - public void setType(int type) { - this.type = type; - } - public float getValue_of_time() { - return value_of_time; - } - public void setValue_of_time(float value_of_time) { - this.value_of_time = value_of_time; - } - public float getReimb_pct() { - return reimb_pct; - } - public void setReimb_pct(float reimb_pct) { - this.reimb_pct = reimb_pct; - } - public float getTimeFactorWork() { - return timeFactorWork; - } - public void setTimeFactorWork(float timeFactorWork) { - this.timeFactorWork = timeFactorWork; - } - public float getTimeFactorNonWork() { - return timeFactorNonWork; - } - public void setTimeFactorNonWork(float timeFactorNonWork) { - this.timeFactorNonWork = timeFactorNonWork; - } - public int getFp_choice() { - return fp_choice; - } - public void setFp_choice(int fp_choice) { - this.fp_choice = fp_choice; - } - } - - class Trip implements Comparable{ - - int hh_id; - int person_id; - int person_num; - int tour_id; - int stop_id; - int inbound; - String tour_purpose; - String orig_purpose; - String dest_purpose; - int orig_maz; - int dest_maz; - int parking_maz; - int stop_period; - int periodsUntilNextTrip; - int trip_mode; - int av_avail; - int tour_mode; - int driver_pnum; - float valueOfTime; - int transponder_avail; - int num_participants; //for joint trips - ArrayList persons; - int veh_used; //the number of the vehicle used (1,2,3 or 0 for no AV used) - - public Trip() { - - tour_purpose=""; - orig_purpose=""; - dest_purpose=""; - - } - public void writeDebug(Logger logger) { - - logger.info("******** TRIP DEBUG *******"); - logger.info("HH ID: "+ hh_id); - logger.info("Person ID: "+person_id); - logger.info("Person Num: "+person_num); - logger.info("Tour ID: "+tour_id); - logger.info("Stop ID: "+stop_id); - logger.info("Inbound: "+inbound); - logger.info("Tour purpose: "+tour_purpose); - logger.info("Orig purpose: "+orig_purpose); - logger.info("Dest purpose: "+dest_purpose); - logger.info("Orig MAZ : "+orig_maz); - logger.info("Dest MAZ : "+dest_maz); - logger.info("Parking MAZ: "+parking_maz); - logger.info("Stop Period: "+stop_period); - logger.info("Periods Until Next Trip: "+periodsUntilNextTrip); - logger.info("Trip mode: "+trip_mode); - logger.info("AV avail: "+av_avail); - logger.info("Tour mode: "+tour_mode); - logger.info("Driver pnum: "+driver_pnum); - logger.info("Value Of Time: "+valueOfTime); - logger.info("Transponder Avail: "+transponder_avail); - logger.info("Num participants: "+num_participants); //for joint trips - logger.info("Veh used: "+veh_used); //the number of the vehicle used (1,2,3 or 0 for no AV used) - - } - - - /** - * Return true if its the same person and the same tour id and purpose - * @param thatTrip - * @return true or false - */ - public boolean sameTour(Trip thatTrip) { - - if((person_id==thatTrip.getPerson_id()) && - (tour_id==thatTrip.getTour_id()) && - (num_participants==thatTrip.getNum_participants()) && - (tour_purpose.compareTo(thatTrip.getTour_purpose())==0)) - return true; - return false; - } - - @Override - public int compareTo(Trip thatTrip) { - - if (this == thatTrip) return 0; - - if(sortByPerson) { - - if(person_idthatTrip.getPerson_id()) - return 1; - else if(person_id==thatTrip.getPerson_id()) { - if(stop_periodthatTrip.getStop_period()) - return 1; - else if(stop_period==thatTrip.getStop_period()) { - if(tour_purpose.compareTo("Work")==0 && thatTrip.getTour_purpose().compareTo("Work-Based")==0) { - if(inbound==0) - return -1; - else - return 1; - }else if(tour_purpose.compareTo("Work-based")==0 && thatTrip.getTour_purpose().compareTo("Work")==0) { - if(thatTrip.getInbound()==0) - return 1; - else - return -1; - } - if(tour_idthatTrip.getTour_id()) - return 1; - if(inbound==0 && thatTrip.getInbound()==1) - return -1; - } - - } - return 0; - } - - /* - //if its the same person and the same tour, use the stop ID - if(sameTour(thatTrip)){ - int thisTourTripSeq = inbound * 1000+stop_id; - int thatTourTripSeq = thatTrip.getInbound() * 1000+thatTrip.getStop_id(); - - if(thisTourTripSeqthatTourTripSeq) - return 1; - else - return 0; - } - - //its not the same tour - */ if(stop_periodthatTrip.getStop_period()) - return 1; - else if(stop_period==thatTrip.getStop_period()) { //its the same stop period - if((person_id==thatTrip.getPerson_id())) { //same person - if(tour_purpose.compareTo("Work")==0 && thatTrip.getTour_purpose().compareTo("Work-Based")==0) { - if(inbound==0) - return -1; - else - return 1; - }else if(tour_purpose.compareTo("Work-based")==0 && thatTrip.getTour_purpose().compareTo("Work")==0) { - if(thatTrip.getInbound()==0) - return 1; - else - return -1; - } - - } - /* - * if(periodsUntilNextTripthatTrip.getPeriodsUntilNextTrip()) - return 1; - */ - } - - return 0; - } - - public int getHh_id() { - return hh_id; - } - public void setHh_id(int hh_id) { - this.hh_id = hh_id; - } - public int getPerson_id() { - return person_id; - } - public void setPerson_id(int person_id) { - this.person_id = person_id; - } - public int getPerson_num() { - return person_num; - } - public void setPerson_num(int person_num) { - this.person_num = person_num; - } - public int getTour_id() { - return tour_id; - } - public void setTour_id(int tour_id) { - this.tour_id = tour_id; - } - public int getStop_id() { - return stop_id; - } - public void setStop_id(int stop_id) { - this.stop_id = stop_id; - } - public int getInbound() { - return inbound; - } - public void setInbound(int inbound) { - this.inbound = inbound; - } - public String getTour_purpose() { - return tour_purpose; - } - public void setTour_purpose(String tour_purpose) { - this.tour_purpose = tour_purpose; - } - public String getOrig_purpose() { - return orig_purpose; - } - public void setOrig_purpose(String orig_purpose) { - this.orig_purpose = orig_purpose; - } - public String getDest_purpose() { - return dest_purpose; - } - public void setDest_purpose(String dest_purpose) { - this.dest_purpose = dest_purpose; - } - public int getOrig_maz() { - return orig_maz; - } - public void setOrig_maz(int orig_maz) { - this.orig_maz = orig_maz; - } - public int getDest_maz() { - return dest_maz; - } - public void setDest_maz(int dest_maz) { - this.dest_maz = dest_maz; - } - public int getParking_maz() { - return parking_maz; - } - public void setParking_maz(int parking_maz) { - this.parking_maz = parking_maz; - } - public int getStop_period() { - return stop_period; - } - public void setStop_period(int stop_period) { - this.stop_period = stop_period; - } - public int getTrip_mode() { - return trip_mode; - } - public void setTrip_mode(int trip_mode) { - this.trip_mode = trip_mode; - } - public int getAv_avail() { - return av_avail; - } - public void setAv_avail(int av_avail) { - this.av_avail = av_avail; - } - public int getTour_mode() { - return tour_mode; - } - public void setTour_mode(int tour_mode) { - this.tour_mode = tour_mode; - } - public int getDriver_pnum() { - return driver_pnum; - } - public void setDriver_pnum(int driver_pnum) { - this.driver_pnum = driver_pnum; - } - public float getValueOfTime() { - return valueOfTime; - } - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - public int getTransponder_avail() { - return transponder_avail; - } - public void setTransponder_avail(int transponder_avail) { - this.transponder_avail = transponder_avail; - } - public int getNum_participants() { - return num_participants; - } - public void setNum_participants(int num_participants) { - this.num_participants = num_participants; - } - public ArrayList getPersons() { - return persons; - } - public void setPersons(ArrayList persons) { - this.persons = persons; - } - - public int getPeriodsUntilNextTrip() { - return periodsUntilNextTrip; - } - - public void setPeriodsUntilNextTrip(int periodsUntilNextTrip) { - this.periodsUntilNextTrip = periodsUntilNextTrip; - } - - public int getVeh_used() { - return veh_used; - } - - public void setVeh_used(int veh_used) { - this.veh_used = veh_used; - } - - } - - public class VehicleTrip{ - - int origMaz; - int destMaz; - int period; - int occupants; - boolean originIsHome; - boolean destinationIsHome; - boolean originIsRemoteParking; - boolean destinationIsRemoteParking; - int parkingChoiceAtDestination; - - Trip tripServed; - - public void writeDebug(Logger logger) { - - logger.info("*** VEHICLE TRIP DEBUG ***"); - logger.info("Orig MAZ: "+origMaz); - logger.info("Dest MAZ: "+destMaz); - logger.info("Period: "+period); - logger.info("Occupants: "+occupants); - logger.info("Orig is home: "+originIsHome); - logger.info("Dest is home: "+destinationIsHome); - logger.info("Orig is remote park: "+originIsRemoteParking); - logger.info("Dest is remote park: "+destinationIsRemoteParking); - logger.info("Parking choice at dest: "+parkingChoiceAtDestination); - - } - - public VehicleTrip() { - } - - public int getOrigMaz() { - return origMaz; - } - - public void setOrigMaz(int origMaz) { - this.origMaz = origMaz; - } - - public int getDestMaz() { - return destMaz; - } - - public void setDestMaz(int destMaz) { - this.destMaz = destMaz; - } - - public int getPeriod() { - return period; - } - - public void setPeriod(int period) { - this.period = period; - } - - public int getOccupants() { - return occupants; - } - - public void setOccupants(int occupants) { - this.occupants = occupants; - } - - public boolean isOriginIsHome() { - return originIsHome; - } - - public void setOriginIsHome(boolean originIsHome) { - this.originIsHome = originIsHome; - } - - public boolean isDestinationIsHome() { - return destinationIsHome; - } - - public void setDestinationIsHome(boolean destinationIsHome) { - this.destinationIsHome = destinationIsHome; - } - - public boolean isOriginIsRemoteParking() { - return originIsRemoteParking; - } - - public void setOriginIsRemoteParking(boolean originIsRemoteParking) { - this.originIsRemoteParking = originIsRemoteParking; - } - - public boolean isDestinationIsRemoteParking() { - return destinationIsRemoteParking; - } - - public void setDestinationIsRemoteParking(boolean destinationIsRemoteParking) { - this.destinationIsRemoteParking = destinationIsRemoteParking; - } - - public int getParkingChoiceAtDestination() { - return parkingChoiceAtDestination; - } - - public void setParkingChoiceAtDestination(int parkingChoiceAtDestination) { - this.parkingChoiceAtDestination = parkingChoiceAtDestination; - } - - public Trip getTripServed() { - return tripServed; - } - - public void setTripServed(Trip tripServed) { - this.tripServed = tripServed; - } - - } - - /** - * This method writes AV vehicle trips to the output file. - * - */ - public void writeVehicleTrips(float sampleRate){ - - logger.info("Writing AV trips to file " + vehicleTripOutputFile); - PrintWriter printWriter = null; - try - { - printWriter = new PrintWriter(new BufferedWriter(new FileWriter(vehicleTripOutputFile))); - } catch (IOException e) - { - logger.fatal("Could not open file " + vehicleTripOutputFile + " for writing\n"); - throw new RuntimeException(); - } - - printHeader(printWriter); - Set keySet = householdMap.keySet(); - for(Integer key: keySet) { - Household hh = householdMap.get(key); - printVehicleTrips(printWriter,hh, sampleRate); - printWriter.flush(); - } - - printWriter.close(); - - } - - - public void printHeader(PrintWriter writer) { - - writer.println("hh_id,veh_id,vehicleTrip_id,orig_mgra,dest_gra,period,occupants," - + "originIsHome,destinationIsHome,originIsRemoteParking,destinationIsRemoteParking," - + "parkingChoiceAtDestination,remoteParkingCostAtDest," - + "person_id,person_num,tour_id,stop_id,inbound,tour_purpose,orig_purpose,dest_purpose," - + "trip_orig_mgra,trip_dest_mgra,stop_period,periodsUntilNextTrip,trip_mode"); - - } - - /** - * Write output to the printwriter. - * - * @param writer - * @param hh - */ - public void printVehicleTrips(PrintWriter writer, Household hh, float sampleRate) { - - int hhid=hh.getId(); - ArrayList vehicles = hh.getAutonomousVehicles(); - if(vehicles==null) - return; - for(int i=0;i vehicleTrips = vehicle.getVehicleTrips(); - - if(vehicleTrips==null) - continue; - - if(vehicleTrips.size()==0) - continue; - - for(int j=0;j vehicleTrips; - - public Vehicle() { - - vehicleTrips = new ArrayList(); - } - - public void writeDebug(Logger logger) { - - logger.info("*** Vehicle debug **"); - logger.info("MAZ: "+maz); - logger.info("Is home: "+isHome); - logger.info("Period available: "+periodAvailable); - - if(vehicleTrips.size()>0) { - for(int i =0;i0) { - VehicleTrip lastTrip = vehicleTrips.get(vehicleTrips.size()-1); - vehicleTrip.setOrigMaz(lastTrip.getDestMaz()); - vehicleTrip.setOriginIsHome(lastTrip.isDestinationIsHome()); - }else { - vehicleTrip.setOriginIsHome(true); - } - - vehicleTrips.add(vehicleTrip); - - return vehicleTrip; - } - - public VehicleTrip createNewVehicleTrip() { - return new VehicleTrip(); - } - - public int getMaz() { - return maz; - } - public void setMaz(int maz) { - this.maz = maz; - } - public boolean isHome() { - return isHome; - } - public void setHome(boolean isHome) { - this.isHome = isHome; - } - public int getPeriodAvailable() { - return periodAvailable; - } - public void setPeriodAvailable(int periodAvailable) { - this.periodAvailable = periodAvailable; - } - - public ArrayList getVehicleTrips() { - return vehicleTrips; - } - - public void setVehicleTrips(ArrayList vehicleTrips) { - this.vehicleTrips = vehicleTrips; - } - - public int getWithPersonId() { - return withPersonId; - } - - public void setWithPersonId(int withPersonId) { - this.withPersonId = withPersonId; - } - - } - - public HouseholdAVAllocationManager(HashMap propertyMap, int iteration,MgraDataManager mazManager,TazDataManager tazManager){ - this.iteration = iteration; - this.propertyMap = propertyMap; - this.tazManager = tazManager; - this.mazManager = mazManager; - modelStructure = new SandagModelStructure(); - - } - - - public void setup() { - - random = new MersenneTwister(); - random.setSeed(randomSeed); - String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); - vehicleTripOutputFile = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_FILE_PROPERTY); - - householdMap = new HashMap(); - - personTypeMap = new HashMap(); - - for(int i = 0;i keySet = householdMap.keySet(); - - for(Integer key: keySet) { - - Household hh = householdMap.get(key); - if(hh.isDebug()) { - - logger.info("***********************************************"); - logger.info("AV allocation model trace (After reading) for household "+hh.getId()); - logger.info(""); - hh.writeDebug(logger, false); - logger.info("***********************************************"); - - } - } - - - } - - /* - * Drop households from the map that don't have AVs - */ - public void dropHouseholdsWithoutAVs() { - - logger.info("Dropping non-AV households"); - - Set keys = householdMap.keySet(); - ArrayList hhIdsToRemove = new ArrayList(); - - for(Integer key: keys) { - - Household hh = householdMap.get(key); - if(hh.getAVs()<=0) { - hhIdsToRemove.add(key); - } - } - if(hhIdsToRemove.size()>0) { - for(Integer hhId : hhIdsToRemove) { - householdMap.remove(hhId); - } - } - logger.info("Completed dropping non-AV households"); - - } - - /* - * Drop trips from the map that aren't auto trips with AVs - */ - public void dropNonAVTrips() { - - logger.info("Dropping non-AV trips from households"); - Set keys = householdMap.keySet(); - - for(Integer key: keys) { - - Household hh = householdMap.get(key); - ArrayList trips = hh.getTrips(); - if(trips.size()==0) - continue; - - Iterator itr = trips.iterator(); - while (itr.hasNext()){ - Trip trip = itr.next(); - - if(trip.getAv_avail()==0) - itr.remove(); - - else if(trip.trip_mode>MaxAutoMode) - itr.remove(); - } - } - logger.info("Completed dropping non-AV trips from households"); - - } - - - public void sortTrips() { - - Set keys = householdMap.keySet(); - for(Integer key: keys) { - - Household hh = householdMap.get(key); - ArrayList trips = hh.getTrips(); - if(trips.size()==0) - continue; - - sortByPerson=true; - Collections.sort(trips); - - //first calculate time before next AV trip made by same person - for(int i = 0 ; i< trips.size();++i) { - Trip trip = trips.get(i); - if(i<(trips.size()-1)) { - Trip nextTrip = trips.get(i+1); - if(trip.getPerson_id()==nextTrip.getPerson_id()) { - int periods = nextTrip.getStop_period()-trip.getStop_period(); - trip.setPeriodsUntilNextTrip(periods); - }else - trip.setPeriodsUntilNextTrip(99);//last trip of the day - }else - trip.setPeriodsUntilNextTrip(99); //last trip of the household - - } - sortByPerson=false; - - Collections.sort(trips); - - } - - } - - - public void readHouseholds() { - - - setDebugHhIdsFromHashmap(); - - String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); - String householdFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, HouseholdDataFileProperty); - householdFile = insertIterationNumber(householdFile,iteration); - - //get the household table and fill up the householdMap with households. - TableDataSet householdDataSet = readTableData(householdFile); - - for(int row=1;row<=householdDataSet.getRowCount();++row) { - - int seed = Math.abs(random.nextInt()); - - //read data - int hhId = (int) householdDataSet.getValueAt(row, "hh_id"); - int hhMgra = (int) householdDataSet.getValueAt(row,"home_mgra"); - int income = (int) householdDataSet.getValueAt(row,"income"); - int autos = (int) householdDataSet.getValueAt(row,"autos"); - int HVs = (int) householdDataSet.getValueAt(row,"HVs"); - int AVs = (int) householdDataSet.getValueAt(row,"AVs"); - - //create household object - Household hh = new Household(); - hh.setId(hhId); - hh.setHomeMaz(hhMgra); - hh.setIncome(income); - hh.setAutos(autos); - hh.setHVs(HVs); - hh.setAVs(AVs); - hh.setSeed(seed); - - if(householdTraceSet.contains(hhId)) - hh.setDebug(true); - else - hh.setDebug(false); - - //generate a set of vehicles and store in h - if(AVs>0) { - for(int i=0;i vehicles = hh.getAutonomousVehicles(); - vehicles.add(AV); - } - } - - //put hh in map - householdMap.put(hhId, hh); - - } - - - } - - public void readPersons() { - - - String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); - String personFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, PersonDataFileProperty); - personFile = insertIterationNumber(personFile,iteration); - - //get the household table and fill up the householdMap with households. - TableDataSet personDataSet = readTableData(personFile); - - for(int row=1;row<=personDataSet.getRowCount();++row) { - - - int hh_id = (int) personDataSet.getValueAt(row,"hh_id"); - int person_id = (int) personDataSet.getValueAt(row,"person_id"); - int person_num = (int) personDataSet.getValueAt(row,"person_num"); - int age = (int) personDataSet.getValueAt(row,"age"); - String gender = personDataSet.getStringValueAt(row,"gender"); - String type = personDataSet.getStringValueAt(row,"type"); - float value_of_time = personDataSet.getValueAt(row,"value_of_time"); - float reimb_pct = personDataSet.getValueAt(row,"reimb_pct"); - int parkingChoice = (int) personDataSet.getValueAt(row, "fp_choice"); - float timeFactorWork = personDataSet.getValueAt(row,"timeFactorWork"); - float timeFactorNonWork = personDataSet.getValueAt(row,"timeFactorNonWork"); - - Person person = new Person(); - person.setHh_id(hh_id); - person.setPerson_id(person_id); - person.setPerson_num(person_num); - person.setAge(age); - person.setGender(gender.compareToIgnoreCase("m")==0 ? 1 : 2); - person.setType(personTypeMap.get(type)); - person.setValue_of_time(value_of_time); - person.setReimb_pct(reimb_pct); - person.setFp_choice(parkingChoice); - person.setTimeFactorWork(timeFactorWork); - person.setTimeFactorNonWork(timeFactorNonWork); - - if(householdMap.containsKey(hh_id)) { - Household hh = householdMap.get(hh_id); - hh.personMap.put(person_num,person); - }else { - logger.fatal("Error: No household ID "+hh_id+" in householdMap. Cannot add person object"); - throw new RuntimeException(); - } - - } - - } - - public void readTrips(String filename, boolean isJoint) { - - - //get the household table and fill up the householdMap with households. - TableDataSet tripDataSet = readTableData(filename); - - for(int row=1;row<=tripDataSet.getRowCount();++row) { - - - int hh_id = (int) tripDataSet.getValueAt(row, "hh_id"); - int person_id=-9; - int person_num=-9; - int num_participants = 1; - int driver_pnum=-9; - if(!isJoint) { - person_id = (int) tripDataSet.getValueAt(row,"person_id"); - person_num = (int) tripDataSet.getValueAt(row,"person_num"); - driver_pnum = (int) tripDataSet.getValueAt(row,"driver_pnum"); - }else { - num_participants = (int) tripDataSet.getValueAt(row,"num_participants"); - person_id = hh_id*100+num_participants; - } - int tour_id = (int) tripDataSet.getValueAt(row,"tour_id"); - int stop_id = (int) tripDataSet.getValueAt(row,"stop_id"); - int inbound = (int) tripDataSet.getValueAt(row,"inbound"); - String tour_purpose = tripDataSet.getStringValueAt(row,"tour_purpose"); - String orig_purpose = tripDataSet.getStringValueAt(row,"orig_purpose"); - String dest_purpose = tripDataSet.getStringValueAt(row,"dest_purpose"); - int orig_maz= (int) tripDataSet.getValueAt(row,"orig_mgra"); - int dest_maz= (int) tripDataSet.getValueAt(row,"dest_mgra"); - int parking_maz = (int) tripDataSet.getValueAt(row,"parking_mgra"); - int stop_period = (int) tripDataSet.getValueAt(row,"stop_period"); - int trip_mode = (int) tripDataSet.getValueAt(row,"trip_mode"); - int av_avail = (int) tripDataSet.getValueAt(row,"av_avail"); - int tour_mode = (int) tripDataSet.getValueAt(row,"tour_mode"); - float valueOfTime = tripDataSet.getValueAt(row,"valueOfTime"); - int transponder_avail = (int) tripDataSet.getValueAt(row,"transponder_avail"); - - Trip trip = new Trip(); - trip.setHh_id(hh_id); - trip.setPerson_id(person_id); - trip.setPerson_num(person_num); - trip.setTour_id(tour_id); - trip.setStop_id(stop_id); - trip.setInbound(inbound); - trip.setTour_purpose(tour_purpose); - trip.setOrig_purpose(orig_purpose); - trip.setDest_purpose(dest_purpose); - trip.setOrig_maz(orig_maz); - trip.setDest_maz(dest_maz); - trip.setParking_maz(parking_maz); - trip.setStop_period(stop_period); - trip.setTrip_mode(trip_mode); - trip.setAv_avail(av_avail); - trip.setTour_mode(tour_mode); - trip.setDriver_pnum(driver_pnum); - trip.setValueOfTime(valueOfTime); - trip.setTransponder_avail(transponder_avail); - trip.setNum_participants(num_participants); - - if(householdMap.containsKey(hh_id)){ - Household hh = householdMap.get(hh_id); - - //following code only handles individual trips right now - //TODO: Revise trip file to include participants, and modify code to - //add all participants. - if(person_num!=-99) { - HashMap personMap = hh.personMap; - Person person = personMap.get(person_num); - ArrayList personsOnTrip = trip.getPersons(); - if(personsOnTrip==null) - personsOnTrip = new ArrayList(); - personsOnTrip.add(person); - } - hh.trips.add(trip); - }else { - logger.fatal("Error: No household ID "+hh_id+" in householdMap. Cannot add trip object"); - throw new RuntimeException(); - - } - - } - - - - } - - /** - * Read data into inputDataTable tabledataset. - * - */ - private TableDataSet readTableData(String inputFile){ - - TableDataSet tableDataSet = null; - - logger.info("Begin reading the data in file " + inputFile); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - tableDataSet = csvFile.readFile(new File(inputFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End reading the data in file " + inputFile); - - return tableDataSet; - } - /** - * A simple helper function to insert the iteration number into the file name. - * - * @param filename The input file name (ex: inputFile.csv) - * @param iteration The iteration number (ex: 3) - * @return The new string (ex: inputFile_3.csv) - */ - private String insertIterationNumber(String filename, int iteration){ - - String newFileName = filename.replace(".csv", "_"+new Integer(iteration).toString()+".csv"); - return newFileName; - } - - public HashMap getHouseholdMap() { - return householdMap; - } - - public void setDebugHhIdsFromHashmap() - { - - householdTraceSet = new HashSet(); - - // get the household ids for which debug info is required - String householdTraceStringList = propertyMap.get(PROPERTIES_HOUSEHOLD_TRACE_LIST); - - if (householdTraceStringList != null) - { - StringTokenizer householdTokenizer = new StringTokenizer(householdTraceStringList, ","); - while (householdTokenizer.hasMoreTokens()) - { - String listValue = householdTokenizer.nextToken(); - int idValue = Integer.parseInt(listValue.trim()); - householdTraceSet.add(idValue); - } - } - - } - - /** - * Get the output trip table file names from the properties file, and write - * trip tables for all modes for the given time period. - * - * @param period - * Time period, which will be used to find the period time string - * to append to each trip table matrix file - */ - public void writeTripTable(MatrixDataServerRmi ms) - { - - String directory = Util.getStringValueFromPropertyMap(propertyMap, "scenario.path"); - String matrixTypeName = Util.getStringValueFromPropertyMap(propertyMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - String fileName = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_MATRIX_PROPERTY) + ".omx"; - try{ - //Delete the file if it exists - File f = new File(fileName); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName); - f.delete(); - } - - if (ms != null) - ms.writeMatrixFile(fileName, emptyVehicleTripMatrix, mt); - else - writeMatrixFile(fileName, emptyVehicleTripMatrix); - } catch (Exception e){ - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName, e); - throw new RuntimeException(); - } - - } - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - private void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java deleted file mode 100644 index 8101a57..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModel.java +++ /dev/null @@ -1,671 +0,0 @@ -package org.sandag.abm.maas; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ListIterator; -import java.util.Random; -import java.util.Set; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.MicromobilityChoiceDMU; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Household; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Person; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Trip; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Vehicle; -import org.sandag.abm.maas.HouseholdAVAllocationManager.VehicleTrip; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.ResourceUtil; - -public class HouseholdAVAllocationModel { - - private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModel.class); - private HashMap propertyMap = null; - HouseholdAVAllocationManager avManager; - private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; - private static final String MINUTES_PER_SIMULATION_PERIOD_PROPERTY = "Maas.RoutingModel.minutesPerSimulationPeriod"; - - private static final String AV_CONTROL_FILE_TARGET = "Maas.AVAllocation.uec.file"; - private static final String AV_DATA_SHEET_TARGET = "Maas.AVAllocation.data.page"; - private static final String AV_VEHICLECHOICE_SHEET_TARGET = "Maas.AVAllocation.vehiclechoice.model.page"; - private static final String AV_PARKINGCHOICE_SHEET_TARGET = "Maas.AVAllocation.parkingchoice.model.page"; - private static final String AV_TRIPUTILITY_SHEET_TARGET = "Maas.AVAllocation.triputility.model.page"; - - private static final int parkingChoiceStay =1; - private static final int parkingChoiceRemote=2; - private static final int parkingChoiceHome=3; - - private static final int vehicleChoiceOther=4; - - private MersenneTwister random; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - - private ChoiceModelApplication parkingChoiceModel; - private ChoiceModelApplication vehicleChoiceModel; - private ChoiceModelApplication tripUtilityModel; - - private HouseholdAVAllocationModelParkingChoiceDMU parkingChoiceDMU; - private HouseholdAVAllocationModelVehicleChoiceDMU vehicleChoiceDMU; - private HouseholdAVAllocationModelTripUtilityDMU tripUtilityDMU; - - int vehicleChoiceOffset = 23942345; - int parkingChoiceOffset =984388432; - int[] closestRemoteLotToMaz; - - - /** - * Constructor. - * - * @param propertyMap - * @param iteration - */ - public HouseholdAVAllocationModel(HashMap propertyMap, MgraDataManager mgraManager, - TazDataManager tazManager, int[] closestRemoteLotToMaz){ - this.propertyMap = propertyMap; - this.mgraManager = mgraManager; - this.tazManager = tazManager; - this.closestRemoteLotToMaz=closestRemoteLotToMaz; - } - - /** - * Initialize all the data members. - * - */ - public void initialize(){ - - - //seed the random number generator so that results can be replicated if desired. - int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); - - random = new MersenneTwister(seed + 4292); - - //create the model UECs - // locate the micromobility choice UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String avUecFile = uecFileDirectory + propertyMap.get(AV_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_DATA_SHEET_TARGET); - int vehicleModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_VEHICLECHOICE_SHEET_TARGET); - int parkingModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_PARKINGCHOICE_SHEET_TARGET); - int tripModelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, AV_TRIPUTILITY_SHEET_TARGET); - - // create the DMU objects. - vehicleChoiceDMU = new HouseholdAVAllocationModelVehicleChoiceDMU(); - parkingChoiceDMU = new HouseholdAVAllocationModelParkingChoiceDMU(); - tripUtilityDMU = new HouseholdAVAllocationModelTripUtilityDMU(); - - // create the choice model objects - vehicleChoiceModel= new ChoiceModelApplication(avUecFile, vehicleModelSheet, dataSheet, propertyMap, - (VariableTable) vehicleChoiceDMU); - parkingChoiceModel= new ChoiceModelApplication(avUecFile, parkingModelSheet, dataSheet, propertyMap, - (VariableTable) parkingChoiceDMU); - tripUtilityModel= new ChoiceModelApplication(avUecFile, tripModelSheet, dataSheet, propertyMap, - (VariableTable) tripUtilityDMU); - - - } - - /** - * Run the AV allocation model for all households in HashMap. - * - * @param hhMap A hashmap of households - * @return the completed hashmap - */ - public HashMap runModel(HashMap hhMap){ - - //iterate through map - Set keySet = hhMap.keySet(); - for(Integer key : keySet) { - - Household hh = hhMap.get(key); - ArrayList trips = hh.getTrips(); - if(trips==null) - continue; - if(trips.size()==0) - continue; - - ArrayList hhVehicles = hh.getAutonomousVehicles(); - - //iterate through trips, choose vehicle - for(int i =0;i vehicleTrips = vehicle.getVehicleTrips(); - VehicleTrip newVehicleTrip = vehicle.createNewVehicleTrip(); - - if(trip.getOrig_purpose().compareToIgnoreCase("Home")==0) - newVehicleTrip.setOriginIsHome(true); - else - newVehicleTrip.setOriginIsHome(false); - - if(trip.getDest_purpose().compareToIgnoreCase("Home")==0) - newVehicleTrip.setDestinationIsHome(true); - else - newVehicleTrip.setDestinationIsHome(false); - - newVehicleTrip.setOrigMaz(trip.getOrig_maz()); - newVehicleTrip.setDestMaz(trip.getDest_maz()); - newVehicleTrip.setOccupants(trip.getNum_participants()); - newVehicleTrip.setPeriod(trip.getStop_period()); - newVehicleTrip.setTripServed(trip); - vehicleTrips.add(newVehicleTrip); - - //update the time for which the vehicle will be available if it - //is an AV, and update the vehicle location - int period = trip.getStop_period(); - if(period>1 && period<40) { - float timeInMinutes = getTravelTime(hh.getId(),trip.getOrig_maz(),trip.getDest_maz(),period); - int additionalPeriods = (int) (timeInMinutes/30); - vehicle.setPeriodAvailable(period + additionalPeriods); - }else { - vehicle.setPeriodAvailable(period); - } - vehicle.setMaz(trip.getDest_maz()); - if(trip.getDest_purpose().compareToIgnoreCase("Home")==0) - vehicle.setHome(true); - else - vehicle.setHome(false); - - vehicle.setWithPersonId(trip.getPerson_id()); - - } - - if(hh.isDebug()) { - - logger.info("***********************************************"); - logger.info("AV allocation model trace (After vehicle choice) for household "+hh.getId()); - logger.info(""); - hh.writeDebug(logger, true); - logger.info("***********************************************"); - - } - - //iterate through vehicles, choose parking location for each trip in each vehicle. - for(int i =0;i vehicleTrips = veh.getVehicleTrips(); - if(vehicleTrips.size()==0) - continue; - - for(int j =0; j persons = trip.getPersons(); - Person person=null; - if(persons!=null) - person = persons.get(0); - - setParkingChoiceDMUAttributes(hh,person,trip, nextTrip); - parkingChoiceModel.computeUtilities(parkingChoiceDMU, parkingChoiceDMU.getDmuIndexValues()); - int parkingChoice = getParkingChoice(hh,trip); - vehicleTrip.setParkingChoiceAtDestination(parkingChoice); - } - } - - if(hh.isDebug()) { - - logger.info("***********************************************"); - logger.info("AV allocation model trace (After parking choice) for household "+hh.getId()); - logger.info(""); - hh.writeDebug(logger, true); - logger.info("***********************************************"); - - } - - //iterate through vehicles, generate empty vehicle trips. - for(int i =0;i vehicleTrips = vehicle.getVehicleTrips(); - if(vehicleTrips.size()==0) - return; - - int homeMaz = hh.getHomeMaz(); - - ArrayList newVehicleTrips = new ArrayList(); - for(int i =0; i < vehicleTrips.size();++i) { - VehicleTrip vehicleTrip = vehicleTrips.get(i); - Trip tripServed = vehicleTrip.getTripServed(); - int personId = tripServed.getPerson_id(); - - //first vehicle trip - if(i==0) { - - //vehicle is home, but first trip is NOT home, create a trip to it - if(tripServed.orig_purpose.compareToIgnoreCase("Home")!=0) { - VehicleTrip newTrip = vehicle.createNewVehicleTrip(); - newTrip.setOriginIsHome(true); - newTrip.setOrigMaz(homeMaz); - newTrip.setDestMaz(tripServed.getOrig_maz()); - newTrip.setPeriod(tripServed.getStop_period()); - newTrip.setOccupants(0); - newTrip.setOriginIsRemoteParking(false); - newTrip.setDestinationIsRemoteParking(false); - newTrip.setDestinationIsHome(false); - newTrip.setOccupants(0); - newVehicleTrips.add(newTrip); - } - } - newVehicleTrips.add(vehicleTrip); - - //generate empty trip to remote lot if parking is remote lot - int parkingChoice = vehicleTrip.getParkingChoiceAtDestination(); - if(parkingChoice==parkingChoiceRemote) { - VehicleTrip newTrip = vehicle.createNewVehicleTrip(); - newTrip.setOriginIsHome(false); - newTrip.setOrigMaz(vehicleTrip.getDestMaz()); - newTrip.setDestMaz(closestRemoteLotToMaz[vehicleTrip.getDestMaz()]); - newTrip.setPeriod(tripServed.getStop_period()); - newTrip.setDestinationIsHome(false); - newTrip.setOccupants(0); - newTrip.setOriginIsRemoteParking(false); - newTrip.setDestinationIsRemoteParking(true); - newTrip.setDestinationIsHome(false); - newVehicleTrips.add(newTrip); - //or a trip to home if parking choice is home - }else if(parkingChoice==parkingChoiceHome) { - VehicleTrip newTrip = vehicle.createNewVehicleTrip(); - newTrip.setOriginIsHome(false); - newTrip.setOrigMaz(vehicleTrip.getDestMaz()); - newTrip.setDestMaz(homeMaz); - newTrip.setPeriod(tripServed.getStop_period()); - newTrip.setDestinationIsHome(true); - newTrip.setOccupants(0); - newTrip.setOriginIsRemoteParking(false); - newTrip.setDestinationIsRemoteParking(false); - newTrip.setDestinationIsHome(true); - newVehicleTrips.add(newTrip); - } - //next trip - if(i<(vehicleTrips.size()-1)) { - VehicleTrip nextTrip = vehicleTrips.get(i+1); - Trip nextTripServed = nextTrip.getTripServed(); - VehicleTrip lastTrip = newVehicleTrips.get(newVehicleTrips.size()-1); - //if the trip is not already in the same MAZ generate an empty trip - if(lastTrip.getDestMaz()!=nextTrip.getOrigMaz()) { - VehicleTrip newTrip = vehicle.createNewVehicleTrip(); - newTrip.setOriginIsHome(lastTrip.isDestinationIsHome()); - newTrip.setOrigMaz(lastTrip.getDestMaz()); - newTrip.setDestMaz(nextTripServed.getOrig_maz()); - newTrip.setPeriod(nextTripServed.getStop_period()); - newTrip.setDestinationIsHome(nextTripServed.getOrig_purpose().compareToIgnoreCase("home")==0); - newTrip.setOccupants(0); - newTrip.setOriginIsRemoteParking(lastTrip.isDestinationIsRemoteParking()); - newTrip.setDestinationIsRemoteParking(false); - newTrip.setDestinationIsHome(nextTripServed.orig_purpose.compareToIgnoreCase("Home")==0); - newVehicleTrips.add(newTrip); - } - } - } - vehicle.setVehicleTrips(newVehicleTrips); - } - - - /** - * Set attributes for the vehicle allocation model. - * @param hh - * @param thisTrip - */ - public void setVehicleChoiceDMUAttributes(Household hh,Trip thisTrip) { - int[] vehicleIsAvailable= {0,0,0}; - float[] travelUtilityToPerson= {0,0,0}; - int[] vehicleIsWithPerson = {0,0,0}; - - int[] avail = {1,1,1}; - - ArrayList vehicles = hh.getAutonomousVehicles(); - for(int i = 0;iveh.getPeriodAvailable()) - vehicleIsAvailable[i]=1; - else { - vehicleIsAvailable[i]=0; - continue; - } - - //if the vehicle is with the person - if(thisTrip.getPerson_id()==veh.getWithPersonId()) - vehicleIsWithPerson[i]=1; - - int origMgra=veh.getMaz(); - int origTaz = mgraManager.getTaz(origMgra); - int destMgra=thisTrip.getDest_maz(); - int destTaz=mgraManager.getTaz(destMgra); - vehicleChoiceDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); - - //the utility to the person is 0, so don't calculate anything - if(veh.isHome() && thisTrip.orig_purpose.compareToIgnoreCase("Home")==0) - continue; - - float utility= getTravelUtility(hh, origMgra,destMgra,period); - - travelUtilityToPerson[i]=utility; - - } - - vehicleChoiceDMU.setVehicle1IsAvailable(vehicleIsAvailable[0]); - vehicleChoiceDMU.setVehicle2IsAvailable(vehicleIsAvailable[1]); - vehicleChoiceDMU.setVehicle3IsAvailable(vehicleIsAvailable[2]); - - vehicleChoiceDMU.setPersonWithVehicle1(vehicleIsWithPerson[0]); - vehicleChoiceDMU.setPersonWithVehicle2(vehicleIsWithPerson[1]); - vehicleChoiceDMU.setPersonWithVehicle3(vehicleIsWithPerson[2]); - - vehicleChoiceDMU.setTravelUtilityToPersonVeh1(travelUtilityToPerson[0]); - vehicleChoiceDMU.setTravelUtilityToPersonVeh2(travelUtilityToPerson[1]); - vehicleChoiceDMU.setTravelUtilityToPersonVeh3(travelUtilityToPerson[2]); - - vehicleChoiceDMU.setMinutesUntilNextTrip(thisTrip.getPeriodsUntilNextTrip()*30); - - - } - - /** - * After the trip is complete, a parking choice is made for the vehicle used for - * the trip. This should only be called for trips for which an AV was used. - * - * @param hh - * @param person - * @param thisTrip - * @param nextTrip - */ - public void setParkingChoiceDMUAttributes(Household hh, Person person, Trip thisTrip, Trip nextTrip) { - - int[] parkArea = mgraManager.getMgraParkAreas(); - float[] monthlyCosts =mgraManager.getMParkCost(); - float[] dailyCosts = mgraManager.getDParkCost(); - float[] hourlyCosts = mgraManager.getHParkCost(); - - int id = hh.getId(); - int hhMaz = hh.getHomeMaz(); - int destMaz = thisTrip.getDest_maz(); - int period = thisTrip.getStop_period(); - - float durationBeforeNextTrip=8*60; //assume 8 hrs before the next trip if there isn't one - if(nextTrip!=null) - durationBeforeNextTrip=(nextTrip.getStop_period()- thisTrip.getStop_period())*30; - - int personType=0; - int atWork=0; - float reimburseProportion=0; - int freeParkingEligibility=0; - if(person!=null) { - personType = person.getType(); - atWork = thisTrip.getDest_purpose().compareToIgnoreCase("Work")==0? 1 : 0; - reimburseProportion = person.getReimb_pct(); - freeParkingEligibility = person.getFp_choice()==1 ? 1 : 0; - } - int parkingArea= parkArea[destMaz]; - float dailyParkingCost= dailyCosts[destMaz]; - float hourlyParkingCost=hourlyCosts[destMaz]; - float monthlyParkingCost=monthlyCosts[destMaz]; - - float utilityToClosestRemoteLot = getTravelUtility(hh, destMaz,destMaz,period); - float utilityToHome = getTravelUtility(hh, destMaz,hhMaz,period); - - float utilityFromHomeToNextTrip =0; - if(nextTrip!=null) { - int nextMaz = nextTrip.getOrig_maz(); - utilityFromHomeToNextTrip = getTravelUtility(hh, destMaz,nextMaz,period); - } - - parkingChoiceDMU.setDurationBeforeNextTrip(durationBeforeNextTrip); - parkingChoiceDMU.setPersonType(personType); - parkingChoiceDMU.setAtWork(atWork); - parkingChoiceDMU.setFreeParkingEligibility(freeParkingEligibility); - parkingChoiceDMU.setReimburseProportion(reimburseProportion); - parkingChoiceDMU.setDailyParkingCost(dailyParkingCost); - parkingChoiceDMU.setHourlyParkingCost(hourlyParkingCost); - parkingChoiceDMU.setMonthlyParkingCost(monthlyParkingCost); - parkingChoiceDMU.setParkingArea(parkingArea); - parkingChoiceDMU.setUtilityToClosestRemoteLot(utilityToClosestRemoteLot); - parkingChoiceDMU.setUtilityToHome(utilityToHome); - parkingChoiceDMU.setUtilityFromHomeToNextTrip(utilityFromHomeToNextTrip); - - } - - /** - * Get the travel time for the origin and destination. - * - * @param id An ID for the trip - * @param originMaz origin MAZ - * @param destMaz destination MAZ - * @param period departure period - * @return Time from the tripUtilityUEC (2nd alternative) - */ - public float getTravelTime(int id, int originMaz,int destMaz,int period) { - - int[] avail = {1,1,1}; - - //calculate when the vehicle would be available for the next trip given the current - //period and the travel time to the next trip. - int origTaz = mgraManager.getTaz(originMaz); - int destTaz=mgraManager.getTaz(destMaz); - tripUtilityDMU.setDmuIndexValues(id, origTaz, origTaz, destTaz); - vehicleChoiceDMU.setDmuIndexValues(id, origTaz, origTaz, destTaz); - tripUtilityDMU.setTimeTrip(period); - - UtilityExpressionCalculator tripUtilityUEC = tripUtilityModel.getUEC(); - double[] util = tripUtilityUEC.solve(tripUtilityDMU.getDmuIndexValues(), tripUtilityDMU, avail); - - return (float) util[1]; - } - - /** - * Get the travel utility for the origin and destination. - * - * @param id An ID for the trip - * @param originMaz origin MAZ - * @param destMaz destination MAZ - * @param period departure period - * @return Utility from the tripUtilityUEC (1st alternative) - */ - public float getTravelUtility(Household hh, int originMaz,int destMaz,int period) { - - int[] avail = {1,1,1}; - - //calculate when the vehicle would be available for the next trip given the current - //period and the travel time to the next trip. - int origTaz = mgraManager.getTaz(originMaz); - int destTaz=mgraManager.getTaz(destMaz); - tripUtilityDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); - vehicleChoiceDMU.setDmuIndexValues(hh.getId(), origTaz, origTaz, destTaz); - tripUtilityDMU.setTimeTrip(period); - - UtilityExpressionCalculator tripUtilityUEC = tripUtilityModel.getUEC(); - double[] util = tripUtilityUEC.solve(tripUtilityDMU.getDmuIndexValues(), tripUtilityDMU, avail); - - // write choice model alternative info to log file - if (hh.isDebug()) - { - logger.info("Calculating travel utility calculation for household " + hh.getId()+ " trip in period " + period +" from MAZ "+originMaz+" to MAZ "+destMaz); - tripUtilityUEC.logAnswersArray( logger, "Trip utility"); - } - - - return (float) util[0]; - } - - /** - * Select the vehicle choice from the UEC. This is helper code for applyModel(), where utilities have already been calculated. - - * @return The vehicle alternative 1,2,3,4.1-3 are AV veh numbers, 4 is other - */ - private int getVehicleChoice(Household hh, Trip trip) { - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - long seed = hh.getSeed() + (vehicleChoiceOffset + trip.getStop_id()*23 +trip.getPerson_id()*34 + trip.getStop_period()*23); - random.setSeed(seed); - - if (vehicleChoiceModel.getAvailabilityCount() > 0) - { - double randomNumber = random.nextDouble(); - chosenAlt = vehicleChoiceModel.getChoiceResult(randomNumber); - - // write choice model alternative info to log file - if (hh.isDebug()) - { - String decisionMaker = String.format("Household " + hh.getId()+ " trip in period " + trip.stop_period +" from MAZ "+trip.getOrig_maz()+" to MAZ "+trip.getDest_maz()); - vehicleChoiceModel.logAlternativesInfo("Vehicle Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Vehicle Choice", decisionMaker, chosenAlt)); - vehicleChoiceModel.logUECResults(logger, decisionMaker); - } - - - return chosenAlt; - } else - { - String decisionMaker = String.format("Household " + hh.getId()+" "+trip.getStop_id()); - String errorMessage = String - .format("Exception caught for %s, no available vehicle choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - vehicleChoiceModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - } - - - /** - * Select the parking choice from the UEC. This is helper code for applyModel(), where utilities have already been calculated. - - * @return The parking alternative - */ - private int getParkingChoice(Household hh, Trip trip) { - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - long seed = hh.getSeed() + (parkingChoiceOffset + trip.getStop_id()*123 +trip.getPerson_id()*23 + trip.getStop_period()*18); - random.setSeed(seed); - - if (parkingChoiceModel.getAvailabilityCount() > 0) - { - double randomNumber = random.nextDouble(); - chosenAlt = parkingChoiceModel.getChoiceResult(randomNumber); - - // write choice model alternative info to log file - if (hh.isDebug()) - { - String decisionMaker = String.format("Household " + hh.getId()+ " trip in period " + trip.stop_period +" from MAZ "+trip.getOrig_maz()+" to MAZ "+trip.getDest_maz()); - parkingChoiceModel.logAlternativesInfo("Parking Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Parking Choice", decisionMaker, chosenAlt)); - parkingChoiceModel.logUECResults(logger, decisionMaker); - } - - - - return chosenAlt; - } else - { - String decisionMaker = String.format("Household " + hh.getId()+" "+trip.getStop_id()); - String errorMessage = String - .format("Exception caught for %s, no available vehicle choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.error(errorMessage); - - parkingChoiceModel.logUECResults(logger, decisionMaker); - throw new RuntimeException(); - } - - } - - - - - /** - * Main run method - * @param args - */ - public static void main(String[] args) { - - - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java deleted file mode 100644 index f9d0f26..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelParkingChoiceDMU.java +++ /dev/null @@ -1,239 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - */ -public class HouseholdAVAllocationModelParkingChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelParkingChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - float durationBeforeNextTrip; - int personType; - int atWork; - int freeParkingEligibility; - float reimburseProportion; - float dailyParkingCost; - float hourlyParkingCost; - float monthlyParkingCost; - int parkingArea; - float utilityToClosestRemoteLot; - float utilityToHome; - float utilityFromHomeToNextTrip; - - - public HouseholdAVAllocationModelParkingChoiceDMU() - { - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - } - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getDurationBeforeNextTrip",1); - methodIndexMap.put("getPersonType",2); - methodIndexMap.put("getAtWork",3); - methodIndexMap.put("getFreeParkingEligibility",4); - methodIndexMap.put("getReimburseProportion",5); - methodIndexMap.put("getDailyParkingCost",6); - methodIndexMap.put("getHourlyParkingCost",7); - methodIndexMap.put("getMonthlyParkingCost",8); - methodIndexMap.put("getParkingArea",9); - methodIndexMap.put("getUtilityToClosestRemoteLot",10); - methodIndexMap.put("getUtilityToHome",11); - methodIndexMap.put("getUtilityFromHomeToNextTrip",12); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getDurationBeforeNextTrip(); - case 2: - return getPersonType(); - case 3: - return getAtWork(); - case 4: - return getFreeParkingEligibility(); - case 5: - return getReimburseProportion(); - case 6: - return getDailyParkingCost(); - case 7: - return getHourlyParkingCost(); - case 8: - return getMonthlyParkingCost(); - case 9: - return getParkingArea(); - case 10: - return getUtilityToClosestRemoteLot(); - case 11: - return getUtilityToHome(); - case 12: - return getUtilityFromHomeToNextTrip(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public float getDurationBeforeNextTrip() { - return durationBeforeNextTrip; - } - - public void setDurationBeforeNextTrip(float durationBeforeNextTrip) { - this.durationBeforeNextTrip = durationBeforeNextTrip; - } - - public int getPersonType() { - return personType; - } - - public void setPersonType(int personType) { - this.personType = personType; - } - - public int getAtWork() { - return atWork; - } - - public void setAtWork(int atWork) { - this.atWork = atWork; - } - - public int getFreeParkingEligibility() { - return freeParkingEligibility; - } - - public void setFreeParkingEligibility(int freeParkingEligibility) { - this.freeParkingEligibility = freeParkingEligibility; - } - - public float getReimburseProportion() { - return reimburseProportion; - } - - public void setReimburseProportion(float reimburseProportion) { - this.reimburseProportion = reimburseProportion; - } - - public float getDailyParkingCost() { - return dailyParkingCost; - } - - public void setDailyParkingCost(float dailyParkingCost) { - this.dailyParkingCost = dailyParkingCost; - } - - public float getHourlyParkingCost() { - return hourlyParkingCost; - } - - public void setHourlyParkingCost(float hourlyParkingCost) { - this.hourlyParkingCost = hourlyParkingCost; - } - - public float getMonthlyParkingCost() { - return monthlyParkingCost; - } - - public void setMonthlyParkingCost(float monthlyParkingCost) { - this.monthlyParkingCost = monthlyParkingCost; - } - - public int getParkingArea() { - return parkingArea; - } - - public void setParkingArea(int parkingArea) { - this.parkingArea = parkingArea; - } - - public float getUtilityToClosestRemoteLot() { - return utilityToClosestRemoteLot; - } - - public void setUtilityToClosestRemoteLot(float utilityToClosestRemoteLot) { - this.utilityToClosestRemoteLot = utilityToClosestRemoteLot; - } - - public float getUtilityToHome() { - return utilityToHome; - } - - public void setUtilityToHome(float utilityToHome) { - this.utilityToHome = utilityToHome; - } - - public float getUtilityFromHomeToNextTrip() { - return utilityFromHomeToNextTrip; - } - - public void setUtilityFromHomeToNextTrip(float utilityFromHomeToNextTrip) { - this.utilityFromHomeToNextTrip = utilityFromHomeToNextTrip; - } - - - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java deleted file mode 100644 index cc4c03f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelRunner.java +++ /dev/null @@ -1,260 +0,0 @@ -package org.sandag.abm.maas; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Set; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.MicromobilityChoiceDMU; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Household; -import org.sandag.abm.maas.HouseholdAVAllocationManager.Trip; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.matrix.MatrixType; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.ResourceUtil; - -public class HouseholdAVAllocationModelRunner { - - private static final Logger logger = Logger.getLogger(HouseholdAVAllocationModelRunner.class); - private HashMap propertyMap = null; - HouseholdAVAllocationManager avManager; - private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; - - private int iteration; - private float sampleRate; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private HouseholdAVAllocationModel AVAllocationModel; //one for now, can multi-thread later - private int[] closestMazWithRemoteLot; - MatrixDataServerRmi ms; - private UtilityExpressionCalculator distanceUEC; - protected VariableTable dmu = null; - protected float[][] tazDistanceSkims; //travel distance - - /** - * Constructor. - * - * @param propertyMap - * @param iteration - */ - public HouseholdAVAllocationModelRunner(HashMap propertyMap, int iteration, float sampleRate){ - this.propertyMap = propertyMap; - this.iteration = iteration; - this.sampleRate = sampleRate; - } - - /** - * Initialize all the data members. - * - */ - public void initialize(){ - - startMatrixServer(propertyMap); - - //managers for MAZ and TAZ data - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - - //create a household AV manager, read trips - avManager = new HouseholdAVAllocationManager(propertyMap, iteration, mgraManager, tazManager); - avManager.setup(); - avManager.readInputFiles(); - - calculateDistanceSkims(); - calculateClosestRemoteLotMazs(); - - } - - /** - * Creates a midday distance UEC, solves it for all zones, stores results in tazDistanceSkims[][]. - */ - public void calculateDistanceSkims() { - - logger.info("Calculating distance skims"); - // Create the distance UEC - String uecPath = Util.getStringValueFromPropertyMap(propertyMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = uecPath - + Util.getStringValueFromPropertyMap(propertyMap, "taz.distance.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(propertyMap, "taz.distance.data.page"); - int distancePage = Util.getIntegerValueFromPropertyMap(propertyMap, "taz.od.distance.md.page"); - distanceUEC = new UtilityExpressionCalculator(new File(uecFileName), distancePage, dataPage, - propertyMap, dmu); - IndexValues iv = new IndexValues(); - - int maxTaz = tazManager.getMaxTaz(); - tazDistanceSkims = new float[maxTaz+1][maxTaz+1]; - - for (int oTaz = 1; oTaz <= maxTaz; oTaz++){ - - iv.setOriginZone(oTaz); - - double[] autoDist = distanceUEC.solve(iv, dmu, null); - for (int d = 0; d < maxTaz; d++){ - tazDistanceSkims[oTaz][d + 1] = (float) autoDist[d]; - } - } - logger.info("Completed calculating distance skims"); - - } - - public void runModel(){ - - //iterate through map - HashMap hhMap = avManager.getHouseholdMap(); - AVAllocationModel = new HouseholdAVAllocationModel(propertyMap, mgraManager,tazManager,closestMazWithRemoteLot); - AVAllocationModel.initialize(); - - AVAllocationModel.runModel(hhMap); - - avManager.writeVehicleTrips(sampleRate); - - avManager.writeTripTable(ms); - - } - - - - /** - * Start a matrix server - * - * @param properties - */ - private void startMatrixServer(HashMap properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - logger.error("could not connect to matrix server", e); - throw new RuntimeException(e); - - } - - } - - - /** - * Iterate through the zones for each MAZ and find the closest MAZ with a remote parking lot. - * - */ - public void calculateClosestRemoteLotMazs() { - - int maxMaz = mgraManager.getMaxMgra(); - - //initialize the array - closestMazWithRemoteLot = new int[maxMaz+1]; - - //iterate through origin MAZs - for(int originMaz=1;originMaz<=maxMaz;++originMaz) { - - float minDist = 99999; //initialize to a really high value - - int originTaz = mgraManager.getTaz(originMaz); - if(originTaz<=0) - continue; - - //iterate through destination MAZs - for(int destinationMaz=1;destinationMaz<=maxMaz;++destinationMaz) { - - //no refueling stations in the destination, keep going - if(mgraManager.getRemoteParkingLot(originMaz)==0) - continue; - - int destinationTaz = mgraManager.getTaz(destinationMaz); - if(destinationTaz<=0) - continue; - - float dist = getDistance(originTaz, destinationTaz); - - //lowest distance, so reset the closest MAZ - if(dist pMap; - - logger.info(String.format("Household AV Fleet Allocation Program using CT-RAMP version ", - CtrampApplication.VERSION)); - - int iteration=0; - float sampleRate=1; - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else { - propertiesFile = args[0]; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.valueOf(args[i + 1]); - } - - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.valueOf(args[i + 1]); - } - - } - } - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - HouseholdAVAllocationModelRunner householdAVModel = new HouseholdAVAllocationModelRunner(pMap, iteration, sampleRate); - householdAVModel.initialize(); - householdAVModel.runModel(); - - - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java deleted file mode 100644 index fa1c6bb..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelTripUtilityDMU.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - */ -public class HouseholdAVAllocationModelTripUtilityDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelTripUtilityDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - int timeTrip; //trip period - - public HouseholdAVAllocationModelTripUtilityDMU() - { - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - } - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - methodIndexMap.put("getTimeTrip", 1); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getTimeTrip(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public int getTimeTrip() { - return timeTrip; - } - - public void setTimeTrip(int timeTrip) { - this.timeTrip = timeTrip; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java deleted file mode 100644 index 91373a0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/HouseholdAVAllocationModelVehicleChoiceDMU.java +++ /dev/null @@ -1,212 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - */ -public class HouseholdAVAllocationModelVehicleChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(HouseholdAVAllocationModelVehicleChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - int vehicle1IsAvailable; - int vehicle2IsAvailable; - int vehicle3IsAvailable; - int personWithVehicle1; - int personWithVehicle2; - int personWithVehicle3; - float travelUtilityToPersonVeh1; - float travelUtilityToPersonVeh2; - float travelUtilityToPersonVeh3; - float minutesUntilNextTrip; - - public HouseholdAVAllocationModelVehicleChoiceDMU() - { - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - } - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - methodIndexMap.put("getVehicle1IsAvailable",1); - methodIndexMap.put("getVehicle2IsAvailable",2); - methodIndexMap.put("getVehicle3IsAvailable",3); - methodIndexMap.put("getPersonWithVehicle1",4); - methodIndexMap.put("getPersonWithVehicle2",5); - methodIndexMap.put("getPersonWithVehicle3",6); - methodIndexMap.put("getTravelUtilityToPersonVeh1",7); - methodIndexMap.put("getTravelUtilityToPersonVeh2",8); - methodIndexMap.put("getTravelUtilityToPersonVeh3",9); - methodIndexMap.put("getMinutesUntilNextTrip",10); - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 1: - return getVehicle1IsAvailable(); - case 2: - return getVehicle2IsAvailable(); - case 3: - return getVehicle3IsAvailable(); - case 4: - return getPersonWithVehicle1(); - case 5: - return getPersonWithVehicle2(); - case 6: - return getPersonWithVehicle3(); - case 7: - return getTravelUtilityToPersonVeh1(); - case 8: - return getTravelUtilityToPersonVeh2(); - case 9: - return getTravelUtilityToPersonVeh3(); - case 10: - return getMinutesUntilNextTrip(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - - - public float getTravelUtilityToPersonVeh1() { - return travelUtilityToPersonVeh1; - } - - public void setTravelUtilityToPersonVeh1(float travelUtilityToPersonVeh1) { - this.travelUtilityToPersonVeh1 = travelUtilityToPersonVeh1; - } - - public float getTravelUtilityToPersonVeh2() { - return travelUtilityToPersonVeh2; - } - - public void setTravelUtilityToPersonVeh2(float travelUtilityToPersonVeh2) { - this.travelUtilityToPersonVeh2 = travelUtilityToPersonVeh2; - } - - public float getTravelUtilityToPersonVeh3() { - return travelUtilityToPersonVeh3; - } - - public void setTravelUtilityToPersonVeh3(float travelUtilityToPersonVeh3) { - this.travelUtilityToPersonVeh3 = travelUtilityToPersonVeh3; - } - - public int getVehicle1IsAvailable() { - return vehicle1IsAvailable; - } - - public void setVehicle1IsAvailable(int vehicle1IsAvailable) { - this.vehicle1IsAvailable = vehicle1IsAvailable; - } - - public int getVehicle2IsAvailable() { - return vehicle2IsAvailable; - } - - public void setVehicle2IsAvailable(int vehicle2IsAvailable) { - this.vehicle2IsAvailable = vehicle2IsAvailable; - } - - public int getVehicle3IsAvailable() { - return vehicle3IsAvailable; - } - - public void setVehicle3IsAvailable(int vehicle3IsAvailable) { - this.vehicle3IsAvailable = vehicle3IsAvailable; - } - - public int getPersonWithVehicle1() { - return personWithVehicle1; - } - - public void setPersonWithVehicle1(int personWithVehicle1) { - this.personWithVehicle1 = personWithVehicle1; - } - - public int getPersonWithVehicle2() { - return personWithVehicle2; - } - - public void setPersonWithVehicle2(int personWithVehicle2) { - this.personWithVehicle2 = personWithVehicle2; - } - - public int getPersonWithVehicle3() { - return personWithVehicle3; - } - - public void setPersonWithVehicle3(int personWithVehicle3) { - this.personWithVehicle3 = personWithVehicle3; - } - - public float getMinutesUntilNextTrip() { - return minutesUntilNextTrip; - } - - public void setMinutesUntilNextTrip(float minutesUntilNextTrip) { - this.minutesUntilNextTrip = minutesUntilNextTrip; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java deleted file mode 100644 index b1e0c6a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTrip.java +++ /dev/null @@ -1,277 +0,0 @@ -package org.sandag.abm.maas; - -/** - * A holder class for trips. - * @author joel.freedman - * - */ -class PersonTrip implements Comparable, Cloneable{ - protected String uniqueId; - protected long hhid; - protected long personId; - protected int personNumber; - protected int tourid; - protected int stopid; - protected int inbound; - protected int joint; - - protected int originMaz; - protected int destinationMaz; - protected short departPeriod; - protected float departTime; //minutes after 3 AM - protected float sampleRate; - protected int mode; - protected int parkingMaz; - protected byte avAvailable; - protected boolean rideSharer; - protected int pickupMaz; - protected int dropoffMaz; - - public PersonTrip(String uniqueId,long hhid,long personId,int personNumber, int tourid,int stopid,int inbound,int joint,int originMaz, int destinationMaz, int departPeriod, float departTime, float sampleRate, int mode, int boardingTap, int alightingTap, int set, boolean rideSharer){ - this.uniqueId = uniqueId; - this.hhid = hhid; - this.personId = personId; - this.personNumber = personNumber; - this.tourid = tourid; - this.stopid = stopid; - this.inbound = inbound; - this.joint = joint; - - this.originMaz = originMaz; - this.destinationMaz = destinationMaz; - this.departPeriod = (short) departPeriod; - this.departTime = departTime; - this.sampleRate = sampleRate; - this.mode = mode; - this.rideSharer = rideSharer; - - //set the pickup MAZ to the originMaz and dropoff MAZ to the destinationMaz - this.pickupMaz = originMaz; - this.dropoffMaz = destinationMaz; - - - - - } - - public String getUniqueId() { - return uniqueId; - } - - public void setUniqueId(String uniqueId) { - this.uniqueId = uniqueId; - } - public long getHhid() { - return hhid; - } - - public void setHhid(long hhid) { - this.hhid = hhid; - } - - public long getPersonId() { - return personId; - } - - public void setPersonId(long personId) { - this.personId = personId; - } - - public int getPersonNumber() { - return personNumber; - } - - public void setPersonNumber(int personNumber) { - this.personNumber = personNumber; - } - - public int getTourid() { - return tourid; - } - - public void setTourid(int tourid) { - this.tourid = tourid; - } - - public int getStopid() { - return stopid; - } - - public void setStopid(int stopid) { - this.stopid = stopid; - } - - public int getInbound() { - return inbound; - } - - public void setInbound(int inbound) { - this.inbound = inbound; - } - - public int getJoint() { - return joint; - } - - public void setJoint(int joint) { - this.joint = joint; - } - - public int getOriginMaz() { - return originMaz; - } - - public void setOriginMaz(int originMaz) { - this.originMaz = originMaz; - } - - public int getPickupMaz() { - return pickupMaz; - } - - public void setPickupMaz(int pickupMaz) { - this.pickupMaz = pickupMaz; - } - - public int getDestinationMaz() { - return destinationMaz; - } - - public void setDestinationMaz(int destinationMaz) { - this.destinationMaz = destinationMaz; - } - - public int getDropoffMaz() { - return dropoffMaz; - } - - public void setDropoffMaz(int dropoffMaz) { - this.dropoffMaz = dropoffMaz; - } - - public short getDepartPeriod() { - return departPeriod; - } - - public void setDepartPeriod(short departPeriod) { - this.departPeriod = departPeriod; - } - - public float getDepartTime() { - return departTime; - } - - public void setDepartTime(float departTime) { - this.departTime = departTime; - } - - public float getSampleRate() { - return sampleRate; - } - - public void setSampleRate(float sampleRate) { - this.sampleRate = sampleRate; - } - - public int getMode() { - return mode; - } - - public void setMode(int mode) { - this.mode = mode; - } - - - public int getParkingMaz() { - return parkingMaz; - } - - public void setParkingMaz(int parkingMaz) { - this.parkingMaz = parkingMaz; - } - - public int getAvAvailable() { - return avAvailable; - } - - public void setAvAvailable(byte avAvailable) { - this.avAvailable = avAvailable; - } - - public boolean isRideSharer() { - return rideSharer; - } - - public void setRideSharer(boolean rideSharer) { - this.rideSharer=rideSharer; - } - - - /** - * Compare based on departure time. - */ - public int compareTo(Object aThat) { - final int BEFORE = -1; - final int EQUAL = 0; - final int AFTER = 1; - - final PersonTrip that = (PersonTrip)aThat; - - //primitive numbers follow this form - if (this.departTime < that.departTime) return BEFORE; - if (this.departTime > that.departTime) return AFTER; - - return EQUAL; - } - - /** - * Override equals. - */ - public boolean equals(Object aThat){ - final PersonTrip that = (PersonTrip)aThat; - - if(this.uniqueId.compareTo(that.getUniqueId())==0) - return true; - - return false; - } - - - /** - * If this trip and thatTrip are both joint trips from the same travel party, return true, else return false - * the comparison is done based on uniqueID, where the end of the unique id is the participant number, - * for example "_1" versus "_2". The method compares the part of the unique ID before the participant number, - * and returns true if they are the same. - * - * @param thatTrip - * @return True if both from same party, else false - */ - public boolean sameParty(PersonTrip thatTrip) { - - if(joint==0) - return false; - - if(thatTrip.getJoint()==0) - return false; - - int lastIndex = uniqueId.lastIndexOf("_"); - String thisUniqueIdMinusParticipantID = uniqueId.substring(0, lastIndex); - - lastIndex = thatTrip.getUniqueId().lastIndexOf("_"); - String thatUniqueIdMinusParticipantID = thatTrip.getUniqueId().substring(0, lastIndex); - - if(thisUniqueIdMinusParticipantID.compareTo(thatUniqueIdMinusParticipantID)==0) - return true; - - return false; - - } - - public Object clone() throws - CloneNotSupportedException - { - PersonTrip trip = (PersonTrip) super.clone(); - trip.uniqueId= new String(); - return trip; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java deleted file mode 100644 index 5cffb2c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/PersonTripManager.java +++ /dev/null @@ -1,1087 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Collection; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; - -public class PersonTripManager { - - protected static final Logger logger = Logger.getLogger(PersonTripManager.class); - protected HashMap propertyMap = null; - protected MersenneTwister random; - protected ModelStructure modelStructure; - protected HashMap personTripMap; - protected ArrayList[][] personTripArrayByDepartureBinAndMaz; //an array of PersonTrips by departure time increment and origin MAZ. - protected ArrayList[] personTripArrayByDepartureBin; //an array of PersonTrips by departure time increment - protected double[] endTimeMinutes; // the period end time in number of minutes past 3 AM , starting in period 1 (index 1) - protected int iteration; - protected MgraDataManager mgraManager; - protected TazDataManager tazManager; - protected int idNumber; - protected int[] modesToKeep; - protected int[] rideShareEligibleModes; - protected int numberOfTimeBins; - protected int periodLengthInMinutes; - protected int minTaz; //the minimum taz number with mazs; any origin or destination person trip less than this will be skipped. - protected float maxWalkDistance; - - protected static final String ModelSeedProperty = "Model.Random.Seed"; - protected static final String DirectoryProperty = "Project.Directory"; - protected static final String IndivTripDataFileProperty = "Results.IndivTripDataFile"; - protected static final String JointTripDataFileProperty = "Results.JointTripDataFile"; - protected static final String ModesToKeepProperty = "Maas.RoutingModel.Modes"; - protected static final String SharedEligibleProperty = "Maas.RoutingModel.SharedEligible"; - protected static final String MaxWalkDistance = "Maas.RoutingModel.maxWalkDistance"; - - protected static final String MexResTripDataFileProperty ="crossBorder.trip.output.file"; - protected static final String VisitorTripDataFileProperty ="visitor.trip.output.file"; - protected static final String AirportSANTripDataFileProperty ="airport.SAN.output.file"; - protected static final String AirportCBXTripDataFileProperty ="airport.CBX.output.file"; - protected static final String IETripDataFileProperty ="internalExternal.trip.output.file"; - - - /** - * Constructor. - * - * @param propertyMap - * @param iteration - */ - public PersonTripManager(HashMap propertyMap, int iteration){ - this.iteration = iteration; - this.propertyMap = propertyMap; - - modelStructure = new SandagModelStructure(); - - } - - /** - * Initialize (not done by default). - * Initializes array to simulate actual departure time - * Reads in individual and joint person trips - */ - public void initialize(int periodLengthInMinutes){ - logger.info("Initializing PersonTripManager"); - - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - //find minimum TAZ with mazs - int maxTaz = tazManager.getMaxTaz(); - minTaz = -1; - for(int i=1;i<=maxTaz;++i){ - int[] mazs = tazManager.getMgraArray(i); - - if(mazs==null|| mazs.length==0) - minTaz = Math.max(i, minTaz); - } - - logger.info("Minimum TAZ number is "+minTaz); - logger.info("Maximum TAZ number is "+maxTaz); - - //initialize the end time in minutes (stored in double so no overlap between periods) - endTimeMinutes = new double[40+1]; - endTimeMinutes[1]=119.999999; //first period is 3-3:59:99:99 - for(int period=2;period 0, implement hotspots - if(maxWalkDistance>0) - moveRidesharersToHotspots(); - - logger.info("Completed Initializing PersonTripManager"); - - - } - - /** - * Read the input individual and joint trip files. This function calls the method - * @readTripList for each table. This method is called from {@initialize()} - */ - private void readInputFiles(){ - - String directory = Util.getStringValueFromPropertyMap(propertyMap, DirectoryProperty); - String indivTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, IndivTripDataFileProperty); - indivTripFile = insertIterationNumber(indivTripFile,iteration); - String jointTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, JointTripDataFileProperty); - jointTripFile = insertIterationNumber(jointTripFile,iteration); - - //start with individual trips - TableDataSet indivTripDataSet = readTableData(indivTripFile); - personTripMap = readResidentTripList(personTripMap, indivTripDataSet, false); - int tripsSoFar=personTripMap.size(); - - logger.info("Read "+tripsSoFar+" individual person trips"); - - //now read joint trip data - TableDataSet jointTripDataSet = readTableData(jointTripFile); - personTripMap = readResidentTripList(personTripMap, jointTripDataSet, true); - - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" joint person trips"); - tripsSoFar=personTripMap.size(); - - - String mexicanResidentTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, MexResTripDataFileProperty); - TableDataSet mexicanResidentTripDataSet = readTableData(mexicanResidentTripFile); - personTripMap = readMexicanResidentTripList(personTripMap, mexicanResidentTripDataSet); - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" mexican resident person trips"); - tripsSoFar=personTripMap.size(); - - - String visitorTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, VisitorTripDataFileProperty); - TableDataSet visitorTripDataSet = readTableData(visitorTripFile); - personTripMap = readVisitorTripList(personTripMap, visitorTripDataSet); - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" visitor person trips"); - tripsSoFar=personTripMap.size(); - - String airportSANTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, AirportSANTripDataFileProperty); - TableDataSet airportSANTripDataSet = readTableData(airportSANTripFile); - personTripMap = readAirportTripList(personTripMap, airportSANTripDataSet, -6,"SAN"); - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" SAN airport person trips"); - tripsSoFar=personTripMap.size(); - - String airportCBXTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, AirportCBXTripDataFileProperty); - TableDataSet airportCBXTripDataSet = readTableData(airportCBXTripFile); - personTripMap = readAirportTripList(personTripMap, airportCBXTripDataSet, -5,"CBX"); - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" CBX airport person trips"); - tripsSoFar=personTripMap.size(); - - String ieTripFile = directory - + Util.getStringValueFromPropertyMap(propertyMap, IETripDataFileProperty); - TableDataSet ieTripDataSet = readTableData(ieTripFile); - personTripMap = readIETripList(personTripMap, ieTripDataSet); - logger.info("Read "+(personTripMap.size()-tripsSoFar)+" IE person trips"); - tripsSoFar=personTripMap.size(); - - logger.info("Read "+personTripMap.size()+" total person trips"); - - } - - /** - * Read the CTRAMP trip list in the TableDataSet. - * - * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. - * @param inputTripTableData The TableDataSet containing the CT-RAMP output trip file. - * @param jointTripData A boolean indicating whether the data is for individual or joint trips. - */ - public HashMap readResidentTripList(HashMap personTripMap, TableDataSet inputTripTableData, boolean jointTripData){ - - if(personTripMap==null) - personTripMap = new HashMap(); - - for(int row = 1; row <= inputTripTableData.getRowCount();++row){ - - - int mode = (int) inputTripTableData.getValueAt(row,"trip_mode"); - if(modesToKeep[mode]!=1) - continue; - - boolean rideShare=false; - if(rideShareEligibleModes[mode]==1) - rideShare=true; - - int oMaz = (int) inputTripTableData.getValueAt(row,"orig_mgra"); - int dMaz = (int) inputTripTableData.getValueAt(row,"dest_mgra"); - - int oTaz = mgraManager.getTaz(oMaz); - int dTaz = mgraManager.getTaz(dMaz); - - if((oTaz1) { - personTrip.setJoint(1); - personTrip.setUniqueId(uniqueID+"_1"); - } - personTripMap.put(idNumber, personTrip); - - //replicate joint trips - if(num_participants>1) - for(int i=2;i<=num_participants;++i){ - ++idNumber; - PersonTrip newTrip = null; - try { - newTrip = (PersonTrip) personTrip.clone(); - }catch(Exception e) { - - logger.fatal("Error attempting to clone joint trip object "+uniqueID); - throw new RuntimeException(e); - } - newTrip.setUniqueId(uniqueID+"_"+i); - personTripMap.put(idNumber, newTrip); - } - } - - return personTripMap; - } - - - /** - * Read the visitor trip list in the TableDataSet. - * - * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. - * @param inputTripTableData The TableDataSet containing the visitor output trip file. - */ - public HashMap readVisitorTripList(HashMap personTripMap, TableDataSet inputTripTableData){ - - if(personTripMap==null) - personTripMap = new HashMap(); - - for(int row = 1; row <= inputTripTableData.getRowCount();++row){ - - - int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); - if(modesToKeep[mode]!=1) - continue; - - boolean rideShare=false; - if(rideShareEligibleModes[mode]==1) - rideShare=true; - - int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); - int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); - - int oTaz = mgraManager.getTaz(oMaz); - int dTaz = mgraManager.getTaz(dMaz); - - if((oTaz1) { - personTrip.setJoint(1); - personTrip.setUniqueId(uniqueID+"_1"); - } - personTripMap.put(idNumber, personTrip); - - //replicate joint trips - if(num_participants>1) - for(int i=2;i<=num_participants;++i){ - ++idNumber; - PersonTrip newTrip = null; - try { - newTrip = (PersonTrip) personTrip.clone(); - }catch(Exception e) { - - logger.fatal("Error attempting to clone joint trip object "+uniqueID); - throw new RuntimeException(e); - } - newTrip.setUniqueId(uniqueID+"_"+i); - personTripMap.put(idNumber, newTrip); - } - - } - - return personTripMap; - } - - /** - * Read the Mexican resident trip list in the TableDataSet. - * - * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. - * @param inputTripTableData The TableDataSet containing the visitor output trip file. - */ - public HashMap readMexicanResidentTripList(HashMap personTripMap, TableDataSet inputTripTableData){ - - if(personTripMap==null) - personTripMap = new HashMap(); - - for(int row = 1; row <= inputTripTableData.getRowCount();++row){ - - int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); - if(modesToKeep[mode]!=1) - continue; - - boolean rideShare=false; - if(rideShareEligibleModes[mode]==1) - rideShare=true; - - int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); - int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); - - int oTaz = mgraManager.getTaz(oMaz); - int dTaz = mgraManager.getTaz(dMaz); - - if((oTaz1) { - personTrip.setJoint(1); - personTrip.setUniqueId(uniqueID+"_1"); - } - personTripMap.put(idNumber, personTrip); - - //replicate joint trips - if(num_participants>1) - for(int i=2;i<=num_participants;++i){ - ++idNumber; - PersonTrip newTrip = null; - try { - newTrip = (PersonTrip) personTrip.clone(); - }catch(Exception e) { - - logger.fatal("Error attempting to clone joint trip object "+uniqueID); - throw new RuntimeException(e); - } - newTrip.setUniqueId(uniqueID+"_"+i); - personTripMap.put(idNumber, newTrip); - } - - - } - - return personTripMap; - } - - /** - * Read the airport trip list in the TableDataSet. - * - * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. - * @param inputTripTableData The TableDataSet containing the visitor output trip file. - */ - public HashMap readAirportTripList(HashMap personTripMap, TableDataSet inputTripTableData, int default_id, String airportCode){ - - if(personTripMap==null) - personTripMap = new HashMap(); - - for(int row = 1; row <= inputTripTableData.getRowCount();++row){ - - int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); - if(modesToKeep[mode]!=1) - continue; - - boolean rideShare=false; - if(rideShareEligibleModes[mode]==1) - rideShare=true; - - int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); - int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); - - int oTaz = mgraManager.getTaz(oMaz); - int dTaz = mgraManager.getTaz(dMaz); - - if((oTaz1) { - personTrip.setJoint(1); - personTrip.setUniqueId(uniqueID+"_1"); - - } - personTripMap.put(idNumber, personTrip); - - //replicate joint trips - if(num_participants>1) - for(int i=2;i<=num_participants;++i){ - ++idNumber; - PersonTrip newTrip = null; - try { - newTrip = (PersonTrip) personTrip.clone(); - }catch(Exception e) { - - logger.fatal("Error attempting to clone joint trip object "+uniqueID); - throw new RuntimeException(e); - } - newTrip.setUniqueId(uniqueID+"_"+i); - personTripMap.put(idNumber, newTrip); - } - - - } - - return personTripMap; - } - - - /** - * Read the IE trip list in the TableDataSet. - * - * @param personTripList A HashMap of PersonTrips. If null will be instantiated in this method. - * @param inputTripTableData The TableDataSet containing the visitor output trip file. - */ - public HashMap readIETripList(HashMap personTripMap, TableDataSet inputTripTableData){ - - if(personTripMap==null) - personTripMap = new HashMap(); - - for(int row = 1; row <= inputTripTableData.getRowCount();++row){ - - int mode = (int) inputTripTableData.getValueAt(row,"tripMode"); - if(modesToKeep[mode]!=1) - continue; - - boolean rideShare=false; - if(rideShareEligibleModes[mode]==1) - rideShare=true; - - int oMaz = (int) inputTripTableData.getValueAt(row,"originMGRA"); - int dMaz = (int) inputTripTableData.getValueAt(row,"destinationMGRA"); - - int oTaz = (int) inputTripTableData.getValueAt(row,"originTAZ"); - int dTaz = (int) inputTripTableData.getValueAt(row,"destinationTAZ"); - - if((oTaz1) { - personTrip.setJoint(1); - personTrip.setUniqueId(uniqueID+"_1"); - } - personTripMap.put(idNumber, personTrip); - - //replicate joint trips - if(num_participants>1) - for(int i=2;i<=num_participants;++i){ - ++idNumber; - PersonTrip newTrip = null; - try { - newTrip = (PersonTrip) personTrip.clone(); - }catch(Exception e) { - - logger.fatal("Error attempting to clone joint trip object "+uniqueID); - throw new RuntimeException(e); - } - newTrip.setUniqueId(uniqueID+"_"+i); - personTripMap.put(idNumber, newTrip); - } - - } - - return personTripMap; - } - /** - * Simulate the exact time for the period. - * - * @param period The time period (1->40) - * @return The exact time in double precision (number of minutes past 3 AM) - */ - public float simulateExactTime(int period){ - - double lowerEnd = endTimeMinutes[period-1]; - double upperEnd = endTimeMinutes[period]; - double randomNumber = random.nextDouble(); - - float time = (float) ((upperEnd - lowerEnd) * randomNumber + lowerEnd); - - return time; - } - - /** - * A simple helper function to insert the iteration number into the file name. - * - * @param filename The input file name (ex: inputFile.csv) - * @param iteration The iteration number (ex: 3) - * @return The new string (ex: inputFile_3.csv) - */ - private String insertIterationNumber(String filename, int iteration){ - - String newFileName = filename.replace(".csv", "_"+new Integer(iteration).toString()+".csv"); - return newFileName; - } - - /** - * Read data into inputDataTable tabledataset. - * - */ - private TableDataSet readTableData(String inputFile){ - - TableDataSet tableDataSet = null; - - logger.info("Begin reading the data in file " + inputFile); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - tableDataSet = csvFile.readFile(new File(inputFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End reading the data in file " + inputFile); - - return tableDataSet; - } - - /** - * Go through the person trip list, sort the person trips by departure time and MAZ. - * - */ - @SuppressWarnings("unchecked") - public void groupPersonTripsByDepartureTimePeriodAndOrigin(){ - - numberOfTimeBins = ((24*60)/periodLengthInMinutes); - int maxMaz = mgraManager.getMaxMgra(); - - logger.info("Calculated "+numberOfTimeBins+" simulation periods using a period length of "+periodLengthInMinutes+" minutes"); - personTripArrayByDepartureBinAndMaz = new ArrayList[numberOfTimeBins][maxMaz+1]; - personTripArrayByDepartureBin = new ArrayList[numberOfTimeBins]; - - //initialize - for(int i = 0; i < numberOfTimeBins;++i){ - personTripArrayByDepartureBin[i] = new ArrayList(); - for(int j = 0; j <=maxMaz;++j){ - personTripArrayByDepartureBinAndMaz[i][j] = new ArrayList(); - } - - } - - Collection personTripList = personTripMap.values(); - for(PersonTrip personTrip : personTripList){ - - int originMaz = personTrip.getPickupMaz(); - - float departTime = personTrip.getDepartTime(); - int bin = (int) Math.floor(departTime/((float) periodLengthInMinutes)); - - personTripArrayByDepartureBinAndMaz[bin][originMaz].add(personTrip); - personTripArrayByDepartureBin[bin].add(personTrip); - - - } - } - - - /** - * Get the person trips for the period bin (indexed from 0) and the origin MAZ. - * - * @param periodBin The number of the departure time period bin based on the period length used to group person trips. - * @param maz The number of the origin MAZ. - * - * @return An arraylist of person trips. - */ - public ArrayList getPersonTripsByDepartureTimePeriodAndMaz(int periodBin, int maz){ - - return personTripArrayByDepartureBinAndMaz[periodBin][maz]; - - } - - /** - * Sample a person trip from the array for the given period. REMOVE IT from the person trip arrays. - * - * @param simulationPeriod The simulation period to sample a trip from. - * @param rnum A random number to be used in sampling. - * @return A person trip, or null if the ArrayList is null or empty. - */ - PersonTrip samplePersonTrip(int simulationPeriod, double rnum){ - - ArrayList personTripArray = personTripArrayByDepartureBin[simulationPeriod]; - - if(personTripArray==null) - return null; - - int listSize = personTripArray.size(); - - if(listSize==0) - return null; - - int element = (int) Math.floor(rnum * listSize); - PersonTrip personTrip = personTripArray.get(element); - personTripArrayByDepartureBin[simulationPeriod].remove(personTrip); - personTripArrayByDepartureBinAndMaz[simulationPeriod][personTrip.getPickupMaz()].remove(personTrip); - - return personTrip; - } - - /** - * Sample a person trip from the array for the given period. REMOVE IT from the array. - * - * @param simulationPeriod The period to sample a trip from. - * @param maz The maz to sample a trip from. - * @param rnum A random number to be used in sampling. - * @return A person trip, or null if the ArrayList is null or empty. - */ - PersonTrip samplePersonTrip(int simulationPeriod, int maz, double rnum){ - - ArrayList personTripArray = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; - - if(personTripArray==null) - return null; - - int listSize = personTripArray.size(); - - if(listSize==0) - return null; - - int element = (int) Math.floor(rnum * listSize); - PersonTrip personTrip = personTripArray.get(element); - personTripArrayByDepartureBinAndMaz[simulationPeriod][maz].remove(personTrip); - personTripArrayByDepartureBin[simulationPeriod].remove(personTrip); - - return personTrip; - } - - /** - * Check if there are more person trips in this simulation period. - * - * @param simulationPeriod - * @return true if there are more person trips, false if not. - */ - public boolean morePersonTripsInSimulationPeriod(int simulationPeriod){ - ArrayList personTripArray = personTripArrayByDepartureBin[simulationPeriod]; - - if(personTripArray==null) - return false; - - int listSize = personTripArray.size(); - - if(listSize==0) - return false; - - return true; - - } - - /** - * Check if there are more person trips in this simulation period and maz. - * - * @param simulationPeriod - * @return true if there are more person trips, false if not. - */ - public boolean morePersonTripsInSimulationPeriodAndMaz(int simulationPeriod, int maz){ - ArrayList personTripArray = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; - - if(personTripArray==null) - return false; - - int listSize = personTripArray.size(); - - if(listSize==0) - return false; - - return true; - - } - /** - * Remove the person trip. - * - * @param trip - * @param simulationPeriod - */ - public void removePersonTrip(PersonTrip trip, int simulationPeriod){ - - int originMaz = trip.getPickupMaz(); - personTripArrayByDepartureBin[simulationPeriod].remove(trip); - personTripArrayByDepartureBinAndMaz[simulationPeriod][originMaz].remove(trip); - - } - - /** - * Pre-process the array of person trips by moving nearby ridesharers to hotspots. - * The algorithm finds the maz in each TAZ and simulation period with the most ridesharers. - * It moves the ridesharers within the maximum walking distance to that MAZ. - */ - public void moveRidesharersToHotspots() { - - logger.info("Hotspots - moving ride-sharers to high demand MAZs"); - - int[] tazs = tazManager.getTazs(); - int maxTaz = tazManager.getMaxTaz(); - - //store the hotspot Maz for each period and taz - int[][] hotspotMazs = new int[numberOfTimeBins][maxTaz+1]; - - //track the number of ridesharers moved - int totalRidesharersMoved = 0; - - for(int simulationPeriod=0;simulationPeriod personTrips = personTripArrayByDepartureBinAndMaz[simulationPeriod][maz]; - if(personTrips==null) - continue; - - //set maz and max ridesharers - if(personTrips.size()>maxRideSharers) { - maxRideSharers= personTrips.size(); - hotspotMaz = maz; - hotspotMazs[simulationPeriod][taz] = hotspotMaz; - } - } // end mazs - - //no mazs with ridesharers in this taz and simulation period - if(maxRideSharers==-1) - continue; - else { - - //get nearby ridesharers and move them to hotspot - ArrayList nearbySharers = findNearbyRideSharersByOriginMaz(hotspotMaz, simulationPeriod, mgraManager.getTaz(hotspotMaz)); - - if(nearbySharers==null) - continue; - - if(nearbySharers.size()==0) - continue; - - // logger.info("TAZ "+taz+": found "+nearbySharers.size()+" to move to hotspot MAZ "+hotspotMaz+" in period "+simulationPeriod); - - //add each ridesharer to the person trip array at the hotspot maz, and remove them from their origin - for(PersonTrip personTrip : nearbySharers) { - int originMaz = personTrip.getOriginMaz(); - personTrip.setPickupMaz(hotspotMaz); - personTripArrayByDepartureBinAndMaz[simulationPeriod][originMaz].remove(personTrip); - personTripArrayByDepartureBinAndMaz[simulationPeriod][hotspotMaz].add(personTrip); - ++ridesharersMoved; - ++totalRidesharersMoved; - } - } - } //end for zones - - - - logger.info("Simulation period "+ simulationPeriod+" moved "+ridesharersMoved+" ridesharers"); - } // end for simulation periods - - //now move dropoffs to hotspot locations - int movedDropoffs = 0; - Collection personTripList = personTripMap.values(); - for(PersonTrip personTrip : personTripList){ - - //skip non-ride shareres - if(!personTrip.isRideSharer()) - continue; - - int destinationMaz = personTrip.getDestinationMaz(); - int destinationTaz = mgraManager.getTaz(destinationMaz); - float departTime = personTrip.getDepartTime(); - int departBin = (int) Math.floor(departTime/((float) periodLengthInMinutes)); - int hotspotMaz = hotspotMazs[departBin][destinationTaz]; - - //no hotspot for this person's destination taz - if(hotspotMaz==0) - continue; - - float distance = ((float) mgraManager.getMgraToMgraWalkDistFrom(destinationMaz,hotspotMaz))/((float)5280.0); - - if(distance==0) - continue; - - //distance between destination and hotspot is less than max walk distance, so move this person - if(distance<=maxWalkDistance) { - personTrip.setDropoffMaz(hotspotMaz); - ++movedDropoffs; - } - } - - logger.info("Hotspots moved "+totalRidesharersMoved+" ride-share pickups and "+movedDropoffs+" dropoffs"); - } - - - - /** - * Cycle through all the MAZs within maximum walk distance of the origin MAZ, and find - * rideshare passengers departing within the same period. Add them to an ArrayList and - * return it. - * - * @param originMaz The origin for searching - * @param simulationPeriod The simulation period - * @return The ArrayList of ridesharers. - */ - public ArrayList findNearbyRideSharersByOriginMaz(int originMaz, int simulationPeriod, int constraintTaz) { - - int[] walkMgras = mgraManager.getMgrasWithinWalkDistanceFrom(originMaz); - - if(walkMgras==null) - return null; - - ArrayList nearbyRideSharers = new ArrayList(); - - //cycle through walk mgras - for(int walkMgra : walkMgras) { - - //skip intrazonal - if(walkMgra==originMaz) - continue; - - if(constraintTaz>0) - if(mgraManager.getTaz(walkMgra)!=constraintTaz) - continue; - - //walk mgra is less than max walk distance - float distance = ((float) mgraManager.getMgraToMgraWalkDistFrom(originMaz,walkMgra))/((float)5280.0); - - if(distance==0) - continue; - if(distance<=maxWalkDistance) { - - ArrayList personTrips = personTripArrayByDepartureBinAndMaz[simulationPeriod][walkMgra]; - - //cycle through person trips in this mgra and add them to the array if they are willing to rideshare - for(PersonTrip personTrip : personTrips) { - - if(personTrip.isRideSharer()) { - nearbyRideSharers.add(personTrip); - } - } - } - } - - return nearbyRideSharers; - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java deleted file mode 100644 index 66e2cd1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCFleetModel.java +++ /dev/null @@ -1,448 +0,0 @@ -package org.sandag.abm.maas; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -public class TNCFleetModel { - - private static final Logger logger = Logger.getLogger(TNCFleetModel.class); - private HashMap propertyMap = null; - TransportCostManager transportCostManager; //manages transport costs! - PersonTripManager personTripManager; //manages person trips! - TNCVehicleManager tNCVehicleManager; //manages vehicles! - - private int iteration; - private float sampleRate; - private int minutesPerSimulationPeriod; - private int numberOfSimulationPeriods; - private MersenneTwister random; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private byte maxSharedTNCPassengers; - - MatrixDataServerRmi ms; - private byte[] skimPeriodLookup; //an array indexed by number of periods that corresponds to the skim period - - private boolean routeIntrazonal; - - private static final String MAX_PICKUP_DISTANCE_PROPERTY = "Maas.RoutingModel.maxDistanceForPickup"; - private static final String MAX_PICKUP_DIVERSON_TIME_PROPERTY = "Maas.RoutingModel.maxDiversionTimeForPickup"; - private static final String MINUTES_PER_SIMULATION_PERIOD_PROPERTY = "Maas.RoutingModel.minutesPerSimulationPeriod"; - private static final String MAX_SHARED_TNC_PASSENGERS_PROPERTY = "Maas.RoutingModel.maxPassengers"; - private static final String ROUTE_INTRAZONAL_PROPERTY = "Maas.RoutingModel.routeIntrazonal"; - private static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; - - int vehicleDebug; - /** - * Constructor. - * - * @param propertyMap - * @param iteration - */ - public TNCFleetModel(HashMap propertyMap, int iteration, float sampleRate){ - this.propertyMap = propertyMap; - this.iteration = iteration; - this.sampleRate = sampleRate; - } - - /** - * Initialize all the data members. - * - */ - public void initialize(){ - - startMatrixServer(propertyMap); - - //managers for MAZ and TAZ data - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - //some controlling properties - float maxPickupDistance = Util.getFloatValueFromPropertyMap(propertyMap, MAX_PICKUP_DISTANCE_PROPERTY); - float maxDiversionTime = Util.getFloatValueFromPropertyMap(propertyMap, MAX_PICKUP_DIVERSON_TIME_PROPERTY); - maxSharedTNCPassengers = (byte) Util.getIntegerValueFromPropertyMap(propertyMap, MAX_SHARED_TNC_PASSENGERS_PROPERTY); - routeIntrazonal = Util.getBooleanValueFromPropertyMap(propertyMap, ROUTE_INTRAZONAL_PROPERTY); - - //set the length of a simulation period - minutesPerSimulationPeriod = Util.getIntegerValueFromPropertyMap(propertyMap, MINUTES_PER_SIMULATION_PERIOD_PROPERTY); - numberOfSimulationPeriods = ((24*60)/minutesPerSimulationPeriod); - logger.info("Running "+numberOfSimulationPeriods+" simulation periods using a period length of "+minutesPerSimulationPeriod+" minutes"); - calculateSkimPeriods(); - - //create a new transport cost manager and create data structures - transportCostManager = new TransportCostManager(propertyMap,maxDiversionTime,maxPickupDistance); - transportCostManager.initialize(); - transportCostManager.calculateTazsByTimeFromOrigin(); - - //create a person trip manager, read person trips - personTripManager = new PersonTripManager(propertyMap, iteration); - personTripManager.initialize(minutesPerSimulationPeriod); - - - //create a tNCVehicle manager - tNCVehicleManager = new TNCVehicleManager(propertyMap, transportCostManager, maxSharedTNCPassengers, minutesPerSimulationPeriod); - tNCVehicleManager.initialize(); - vehicleDebug = tNCVehicleManager.vehicleDebug; - - //seed the random number generator so that results can be replicated if desired. - int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); - random = new MersenneTwister(seed + 4292); - - } - - /** - * Relate simulation periods to skim periods. - * - */ - public void calculateSkimPeriods(){ - - skimPeriodLookup = new byte[numberOfSimulationPeriods]; - int numberSkimPeriods = ModelStructure.SKIM_PERIOD_INDICES.length; - int[] endSkimPeriod = new int[numberSkimPeriods]; - - int lastPeriodEnd = 0; - int lastEndSkimPeriod = 0; - for(int skimPeriod = 0;skimPeriod=0;--skimPeriod){ - if(period pMap; - - logger.info(String.format("TNC Fleet Simulation Program using CT-RAMP version ", - CtrampApplication.VERSION)); - - int iteration=0; - float sampleRate=1; - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else { - propertiesFile = args[0]; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.valueOf(args[i + 1]); - } - - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.valueOf(args[i + 1]); - } - - - - } - } - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - TNCFleetModel fleetModel = new TNCFleetModel(pMap, iteration, sampleRate); - fleetModel.initialize(); - fleetModel.runModel(); - - - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java deleted file mode 100644 index 837d3f9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicle.java +++ /dev/null @@ -1,156 +0,0 @@ -package org.sandag.abm.maas; - -import java.util.ArrayList; - -public class TNCVehicle { - - - protected ArrayList personTripList; - - protected ArrayList tNCVehicleTrips; - protected byte maxPassengers; - protected short generationTaz; - protected short generationPeriod; - protected int id; - protected float maxDistanceBeforeRefuel; - protected float distanceSinceRefuel; - protected int periodsRefueling; - - /** - * Create a new tNCVehicle. - * - * @param id - * @param maxPassengers - * @param maxDistanceBeforeRefuel - */ - public TNCVehicle(int id, byte maxPassengers, float maxDistanceBeforeRefuel){ - this.id= id; - this.maxPassengers = maxPassengers; - personTripList = new ArrayList(); - tNCVehicleTrips = new ArrayList(); - this.maxDistanceBeforeRefuel = maxDistanceBeforeRefuel; - } - - /** - * Add a tNCVehicle trip to this tNCVehicle. - * - * @param tNCVehicleTrip - */ - public void addVehicleTrip(TNCVehicleTrip tNCVehicleTrip){ - tNCVehicleTrips.add(tNCVehicleTrip); - } - - /** - * Add an ArrayList of tNCVehicle trips to this tNCVehicle. - * - * @param tNCVehicleTrips - */ - public void addVehicleTrips(ArrayList tNCVehicleTrips){ - this.tNCVehicleTrips.addAll(tNCVehicleTrips); - } - - /** - * Clear all the person trips from this tNCVehicle. Used after routing the tNCVehicle. - * - */ - public void clearPersonTrips(){ - this.personTripList.clear(); - } - - /** - * Get all the tNCVehicle trips for this tNCVehicle. - * - * @return VehicleTrips - */ - public ArrayList getVehicleTrips(){ - - return tNCVehicleTrips; - } - - /** - * Get the tNCVehicle ID. - * - * @return - */ - public int getId(){ - return this.id; - } - - /** - * Add a passenger to the tNCVehicle. - * - * @param personTrip - */ - public void addPersonTrip(PersonTrip personTrip){ - - personTripList.add(personTrip); - - } - - /** - * Remove one person trip from the tNCVehicle. Used after routing. - * - * @param personTrip - */ - public void removePersonTrip(PersonTrip personTrip){ - - personTripList.remove(personTrip); - } - - /** - * Get number of passengers. - * - * @return The number of passengers - */ - public byte getNumberPassengers(){ - - return (byte) personTripList.size(); - } - - public byte getMaxPassengers() { - return maxPassengers; - } - - public void setMaxPassengers(byte maxPassengers) { - this.maxPassengers = maxPassengers; - } - - public short getGenerationTaz() { - return generationTaz; - } - - public void setGenerationTaz(short generationTaz) { - this.generationTaz = generationTaz; - } - - public short getGenerationPeriod() { - return generationPeriod; - } - - public void setGenerationPeriod(short generationPeriod) { - this.generationPeriod = generationPeriod; - } - - public ArrayList getPersonTripList() { - return personTripList; - } - - public float getDistanceSinceRefuel() { - return distanceSinceRefuel; - } - - public void setDistanceSinceRefuel(float distanceSinceRefuel) { - this.distanceSinceRefuel = distanceSinceRefuel; - } - - public int getPeriodsRefueling() { - return periodsRefueling; - } - - public void setPeriodsRefueling(int periodsRefueling) { - this.periodsRefueling = periodsRefueling; - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java deleted file mode 100644 index 78e411f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/TNCVehicleManager.java +++ /dev/null @@ -1,878 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.maas.TNCVehicleTrip.Purpose; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.math.MersenneTwister; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; - -public class TNCVehicleManager { - - protected static final Logger logger = Logger.getLogger(TNCVehicleManager.class); - protected HashMap propertyMap = null; - protected ArrayList[] emptyVehicleList; //by taz - protected ArrayList vehiclesToRouteList; - protected ArrayList activeVehicleList; - protected ArrayList refuelingVehicleList; - - protected MersenneTwister random; - protected static final String MODEL_SEED_PROPERTY = "Model.Random.Seed"; - protected static final String VEHICLETRIP_OUTPUT_FILE_PROPERTY = "Maas.RoutingModel.vehicletrip.output.file"; - protected static final String VEHICLETRIP_OUTPUT_MATRIX_PROPERTY = "Maas.RoutingModel.vehicletrip.output.matrix"; - protected static final String MAX_DISTANCE_BEFORE_REFUEL_PROPERTY = "Maas.RoutingModel.maxDistanceBeforeRefuel"; - protected static final String TIME_REQUIRED_FOR_REFUEL_PROPERTY = "Maas.RoutingModel.timeRequiredForRefuel"; - - - protected String vehicleTripOutputFile; - protected TazDataManager tazManager; - protected MgraDataManager mazManager; - protected int maxTaz; - protected int maxMaz; - protected byte maxPassengers; - protected TransportCostManager transportCostManager; - protected int totalVehicles; - protected int minutesPerSimulationPeriod; - protected int vehicleDebug; - protected int totalVehicleTrips; - private byte[] skimPeriodLookup; //an array indexed by number of periods that corresponds to the skim period - private int numberOfSimulationPeriods; - protected float maxDistanceBeforeRefuel; - protected float timeRequiredForRefuel; - protected int periodsRequiredForRefuel; - protected int[] closestMazWithRefeulingStation ; //the closest MAZ with a refueling station - - // one file per time period - // matrices are indexed by periods, occupants - private Matrix[][] TNCTripMatrix; - - - - - /** - * - * @param propertyMap - * @param transportCostManager - */ - public TNCVehicleManager(HashMap propertyMap, TransportCostManager transportCostManager, byte maxPassengers, int minutesPerSimulationPeriod){ - - this.propertyMap = propertyMap; - this.transportCostManager = transportCostManager; - this.maxPassengers = maxPassengers; - this.minutesPerSimulationPeriod = minutesPerSimulationPeriod; - numberOfSimulationPeriods = ((24*60)/minutesPerSimulationPeriod); - - } - - @SuppressWarnings("unchecked") - public void initialize(){ - - int seed = Util.getIntegerValueFromPropertyMap(propertyMap, MODEL_SEED_PROPERTY); - random = new MersenneTwister(seed + 234324); - - tazManager = TazDataManager.getInstance(); - maxTaz = tazManager.getMaxTaz(); - - mazManager = MgraDataManager.getInstance(); - maxMaz = mazManager.getMaxMgra(); - - emptyVehicleList = new ArrayList[maxTaz+1]; - - activeVehicleList = new ArrayList(); - vehiclesToRouteList = new ArrayList(); - refuelingVehicleList = new ArrayList(); - - String directory = Util.getStringValueFromPropertyMap(propertyMap, "Project.Directory"); - vehicleTripOutputFile = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_FILE_PROPERTY); - - maxDistanceBeforeRefuel = Util.getFloatValueFromPropertyMap(propertyMap, MAX_DISTANCE_BEFORE_REFUEL_PROPERTY); - timeRequiredForRefuel = Util.getFloatValueFromPropertyMap(propertyMap, TIME_REQUIRED_FOR_REFUEL_PROPERTY); - periodsRequiredForRefuel = (int) Math.ceil(timeRequiredForRefuel/minutesPerSimulationPeriod); - - vehicleDebug = 1; - - calculateClosestRefuelingMazs(); - - calculateSkimPeriods(); - - //initialize the matrices for writing trips - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - TNCTripMatrix = new Matrix[transportCostManager.NUM_PERIODS][]; - - for(int i =0;i0){ - - TNCVehicle tNCVehicle = null; - double rnum = random.nextDouble(); - tNCVehicle = getRandomVehicleFromList(emptyVehicleList[taz], rnum); - - //generate an empty trip for the tNCVehicle - ArrayList trips = tNCVehicle.getVehicleTrips(); - - if(trips.size()==0){ - logger.warn("Weird: got an empty tNCVehicle (id:"+tNCVehicle.getId()+") but no trips in tNCVehicle trip list"); - return tNCVehicle; - } - - TNCVehicleTrip lastTrip = trips.get(trips.size()-1); - int originTaz = lastTrip.getDestinationTaz(); - int originMaz = lastTrip.getDestinationMaz(); - ArrayList pickupIdsAtOrigin = lastTrip.getPickupIdsAtDestination(); - ArrayList dropoffIdsAtOrigin = lastTrip.getDropoffIdsAtDestination(); - - ++totalVehicleTrips; - TNCVehicleTrip newTrip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); - - newTrip.setOriginMaz(originMaz); - newTrip.setOriginTaz((short) originTaz); - newTrip.setDestinationMaz(departureMaz); - newTrip.setDestinationTaz((short)departureTaz); - newTrip.setStartPeriod(simulationPeriod); - newTrip.setEndPeriod(simulationPeriod); //instantaneous arrivals? Need traveling tNCVehicle queue... - if(lastTrip.getDestinationPurpose()==TNCVehicleTrip.Purpose.REFUEL) - newTrip.setOriginPurpose(TNCVehicleTrip.Purpose.REFUEL); - else { - newTrip.setPickupIdsAtOrigin(pickupIdsAtOrigin); - newTrip.setDropoffIdsAtOrigin(dropoffIdsAtOrigin); - } - newTrip.setDestinationPurpose(TNCVehicleTrip.Purpose.PICKUP_ONLY); - tNCVehicle.addVehicleTrip(newTrip); - - return tNCVehicle; - } - - } - - //Iterated through all TAZs, could not find an empty tNCVehicle. Return a new tNCVehicle. - return generateVehicle(simulationPeriod, departureTaz); - } - - /** - * Get a tNCVehicle at random from the arraylist of vehicles, remove it from the list, and return it. - * - * @param emptyVehicleList - * @param rnum a random number used to draw a tNCVehicle from the list. - * @return The tNCVehicle chosen. - */ - private TNCVehicle getRandomVehicleFromList(ArrayList vehicleList, double rnum){ - - - if(vehicleList==null) - return null; - - int listSize = vehicleList.size(); - int element = (int) Math.floor(rnum * listSize); - TNCVehicle tNCVehicle = vehicleList.get(element); - vehicleList.remove(element); - return tNCVehicle; - - } - - /** - * Encapsulating in method so that vehicles and some statistics can be tracked. - */ - private synchronized TNCVehicle generateVehicle(int simulationPeriod, int taz){ - ++totalVehicles; - TNCVehicle tNCVehicle = new TNCVehicle(totalVehicles, maxPassengers, maxDistanceBeforeRefuel); - tNCVehicle.setGenerationPeriod((short)simulationPeriod); - tNCVehicle.setGenerationTaz((short) taz); - return tNCVehicle; - - } - - public int getTotalVehicles(){ - return totalVehicles; - } - - /** - * Add empty tNCVehicle to the empty tNCVehicle list. - * - * @param tNCVehicle - * @param taz - */ - public void storeEmptyVehicle(TNCVehicle tNCVehicle, int taz){ - - if(emptyVehicleList[taz] == null) - emptyVehicleList[taz] = new ArrayList(); - - emptyVehicleList[taz].add(tNCVehicle); - - } - - public void addActiveVehicle(TNCVehicle tNCVehicle){ - - activeVehicleList.add(tNCVehicle); - } - - public void addVehicleToRoute(TNCVehicle tNCVehicle){ - - ArrayList personTrips = tNCVehicle.getPersonTripList(); - if(personTrips.size()==0){ - logger.info("Adding tNCVehicle "+tNCVehicle.getId()+" to vehicles to route list but no person trips"); - throw new RuntimeException(); - } - vehiclesToRouteList.add(tNCVehicle); - } - - /** - * All active vehicles are assigned passengers, now they must be routed through all pickups and dropoffs. - * THe method iterates through the vehiclesToRouteList and adds passengers based on the out-direction - * time required to pick them up and drop them off. - * - * @param skimPeriod - * @param simulationPeriod - * @param transportCostManager - */ - public synchronized void routeActiveVehicles(int skimPeriod, int simulationPeriod, TransportCostManager transportCostManager){ - - logger.info("Routing "+vehiclesToRouteList.size()+" vehicles in period "+simulationPeriod); - ArrayList vehiclesToRemove = new ArrayList(); - - - //iterate through vehicles to route list - for(TNCVehicle tNCVehicle: vehiclesToRouteList){ - - // get the person list, if it is empty throw a warning (should never be empty) - ArrayList personTrips = tNCVehicle.getPersonTripList(); - if(personTrips==null||personTrips.size()==0){ - logger.error("Attempting to route empty tNCVehicle "+tNCVehicle.getId()); - } - - if(tNCVehicle.getId()==vehicleDebug){ - logger.info("***********************************************************************************"); - logger.info("Debugging Vehicle routing for vehicle ID "+tNCVehicle.getId()); - logger.info("***********************************************************************************"); - logger.info("There are "+personTrips.size()+" person trips in vehicle ID "+tNCVehicle.getId()); - for(PersonTrip pTrip: personTrips){ - logger.info("Vehicle ID "+tNCVehicle.getId()+" person trip id: "+pTrip.getUniqueId()+" from pickup MAZ: "+pTrip.getPickupMaz()+ " to dropoff MAZ "+pTrip.getDropoffMaz()); - } - } - //some information on the first passenger - PersonTrip firstTrip = personTrips.get(0); - int firstOriginMaz = firstTrip.getPickupMaz(); - int firstOriginTaz = mazManager.getTaz(firstOriginMaz); - int firstDestinationMaz = firstTrip.getDropoffMaz(); - int firstDestinationTaz = mazManager.getTaz(firstDestinationMaz); - - // get the arraylist of tNCVehicle trips for this tNCVehicle - ArrayList existingVehicleTrips = tNCVehicle.getVehicleTrips(); - ArrayList newVehicleTrips = new ArrayList(); - - if(tNCVehicle.getId()==vehicleDebug) - logger.info("There are "+existingVehicleTrips.size()+" existing vehicle trips in vehicle ID "+tNCVehicle.getId()); - - //iterate through person list and save HashMap of other passenger pickups and dropoffs by MAZ - HashMap> pickupsByMaz = new HashMap>(); - HashMap> dropoffsByMaz = new HashMap>(); - - //save the dropoff location of the first passenger in the dropoffsByMaz array (the pickup location must be the trip origin) - ArrayList firstDropoffArray = new ArrayList(); - firstDropoffArray.add(personTrips.get(0)); - dropoffsByMaz.put(firstDestinationMaz, firstDropoffArray); - - //iterate through the rest of the person trips other than the first passenger - for(int i = 1; i < personTrips.size();++i){ - - PersonTrip personTrip = personTrips.get(i); - - int pickupMaz = personTrip.getPickupMaz(); - int dropoffMaz = personTrip.getDropoffMaz(); - - //only add pickup maz for passengers other than first passenger - if(!pickupsByMaz.containsKey(pickupMaz) ){ - ArrayList pickups = new ArrayList(); - pickups.add(personTrip); - pickupsByMaz.put(pickupMaz,pickups); - }else{ - ArrayList pickups = pickupsByMaz.get(pickupMaz); - pickups.add(personTrip); - pickupsByMaz.put(pickupMaz,pickups); - } - - if(!dropoffsByMaz.containsKey(dropoffMaz)){ - ArrayList dropoffs = new ArrayList(); - dropoffs.add(personTrip); - dropoffsByMaz.put(dropoffMaz,dropoffs); - }else{ - ArrayList dropoffs = dropoffsByMaz.get(dropoffMaz); - dropoffs.add(personTrip); - dropoffsByMaz.put(dropoffMaz,dropoffs); - } - } - - if(tNCVehicle.getId()==vehicleDebug){ - logger.info("There are "+pickupsByMaz.size()+" pickup mazs in vehicle ID "+tNCVehicle.getId()); - logger.info("There are "+dropoffsByMaz.size()+" dropoff mazs in vehicle ID "+tNCVehicle.getId()); - } - - // the list of TAZs in order from closest to furthest, that will determine tNCVehicle routing. - // any TAZ in the list with an origin or destination by a passenger will be visited. - short[] tazs = transportCostManager.getZonesWithinMaxDiversionTime(skimPeriod, firstOriginTaz, firstDestinationTaz); - - //create a new tNCVehicle trip, and populate it with information from the first passenger - ++totalVehicleTrips; - TNCVehicleTrip trip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); - trip.setStartPeriod(simulationPeriod); - trip.addPickupAtOrigin(firstTrip.getUniqueId()); - trip.setOriginMaz(firstOriginMaz); - trip.setOriginTaz((short) firstOriginTaz); - trip.setPassengers(1); - - //iterate through tazs sorted by time from first passenger's origin, and - //assign person trips to pickup and dropoff arrays based on diversion time. - for(int i=0;i pickups = pickupsByMaz.get(maz); - for(int p = 0; p< pickups.size();++p){ - PersonTrip pTrip = pickups.get(p); - trip.addPickupAtDestination(pTrip.getUniqueId()); - } - } - - //there are dropoffs in this maz - if(dropoffsByMaz.containsKey(maz)){ - ArrayList dropoffs = dropoffsByMaz.get(maz); - for(int p = 0; p< dropoffs.size();++p){ - PersonTrip pTrip = dropoffs.get(p); - trip.addDropoffAtDestination(pTrip.getUniqueId()); - - //remove this person trip from the list of persons in this tNCVehicle since they are getting dropped off. - tNCVehicle.removePersonTrip(pTrip); - - } - } - - // this is not the first tNCVehicle trip for this tNCVehicle. So we need to find the last tNCVehicle trip - // occupancy and destination pickups and dropoffs to set the trip occupancy and origin pickups & dropoffs accordingly. - int lastTripPassengers=0; - TNCVehicleTrip lastTrip = null; - if(newVehicleTrips.size()==0 && existingVehicleTrips.size()>0){ - lastTrip = existingVehicleTrips.get(existingVehicleTrips.size()-1); - }else if(newVehicleTrips.size()>0){ - lastTrip = newVehicleTrips.get(newVehicleTrips.size()-1); - } - //set the origin and other values for the trip - if(lastTrip!=null){ - lastTripPassengers = lastTrip.getPassengers(); - ArrayList dropoffsAtDestinationOfLastTrip = lastTrip.getDropoffIdsAtDestination(); - ArrayList pickupsAtDestinationOfLastTrip = lastTrip.getPickupIdsAtDestination(); - - //add pickups and dropoffs at origin from last trip - trip.addDropoffIdsAtOrigin(dropoffsAtDestinationOfLastTrip); - trip.addPickupIdsAtOrigin(pickupsAtDestinationOfLastTrip); - - //add pickup and dropoffs at origin of this trip to destination of last trip. (commenting to test write problem) - //lastTrip.addDropoffIdsAtDestination(trip.getDropoffIdsAtOrigin()); - //lastTrip.addPickupIdsAtDestination(trip.getPickupIdsAtOrigin()); - - - - } - - int passengers = lastTripPassengers + trip.getNumberOfPickupsAtOrigin() - trip.getNumberOfDropoffsAtOrigin(); - trip.setPassengers(passengers); - trip.setDestinationMaz(maz); - trip.setDestinationTaz((short) tazs[i]); - - //measure time from first trip to destination (current) or track time in tNCVehicle explicitly for each trip? - float time = transportCostManager.getTime(skimPeriod, firstOriginTaz, trip.getDestinationTaz()); - float periods = time/(float)minutesPerSimulationPeriod; - int endPeriod = (int) Math.floor(simulationPeriod + periods); //currently measuring time as simulation period + straight time to dest. - trip.setEndPeriod(endPeriod); - - //measure distance for current trip origin and destination - float distance = transportCostManager.getDistance(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz()); - trip.setDistance(distance); - tNCVehicle.setDistanceSinceRefuel(tNCVehicle.getDistanceSinceRefuel()+distance); - - if(tNCVehicle.getId()==vehicleDebug){ - logger.info("Vehicle ID "+tNCVehicle.getId()+" now has vehicle trip ID "+trip.getId()); - trip.writeTrace(); - } - - newVehicleTrips.add(trip); - - //more trips to go! - if(tNCVehicle.getPersonTripList().size()>0){ - ++totalVehicleTrips; - TNCVehicleTrip newTrip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips); - newTrip.setOriginMaz(maz); - newTrip.setOriginTaz((short) tazs[i]); - newTrip.setStartPeriod(trip.getEndPeriod()); - trip = newTrip; - } - - } //end mazs - - } //end tazs - - //add tNCVehicle trips to tNCVehicle - tNCVehicle.addVehicleTrips(newVehicleTrips); - - //add tNCVehicle to active vehicles - activeVehicleList.add(tNCVehicle); - - //track tNCVehicle in vehicles to remove from route list. - vehiclesToRemove.add(tNCVehicle); - } //end vehicles - - //Remove vehicles that have been routed - vehiclesToRouteList.removeAll(vehiclesToRemove); - } - - /** - * Free vehicles from the active tNCVehicle list and put them in the free tNCVehicle list if - * the last trip in the tNCVehicle ends in the current simulation period. - * - * @param simulationPeriod - */ - public void freeVehicles(int simulationPeriod){ - - int freedVehicles=0; - - //no active vehicles in the simulation period - if(activeVehicleList.size()==0){ - logger.warn("Trying to free vehicles from active vehicle list in simulation period "+simulationPeriod+" but there are no active vehicles."); - }else{ - logger.info("There are "+activeVehicleList.size()+" active vehicles in period "+simulationPeriod); - } - - //track the vehicles to remove - ArrayList vehiclesToRemove = new ArrayList(); - // go through active vehicles (vehicles that have been routed and are picking up/dropping off passengers) - for(int i = 0; i< activeVehicleList.size();++i){ - TNCVehicle tNCVehicle = activeVehicleList.get(i); - - ArrayList trips = tNCVehicle.getVehicleTrips(); - - //this tNCVehicle has no trips; why is it in the active tNCVehicle list?? - if(trips.size()==0){ - logger.error("Vehicle ID "+tNCVehicle.getId()+" has no vehicle trips but is in active vehicle list"); - continue; - } - - //Find out when the last dropoff occurs (the end period of the last trip) - TNCVehicleTrip lastTrip = trips.get(trips.size()-1); - if(lastTrip.endPeriod==simulationPeriod){ - int taz = lastTrip.getDestinationTaz(); - vehiclesToRemove.add(tNCVehicle); - ++freedVehicles; - - //store the empty tNCVehicle in the last dropoff location (the last trip destination TAZ) - if(emptyVehicleList[taz]==null) - emptyVehicleList[taz]= new ArrayList(); - - emptyVehicleList[taz].add(tNCVehicle); - } - } - activeVehicleList.removeAll(vehiclesToRemove); - logger.info("Freed "+freedVehicles+" vehicles from active tNCVehicle list"); - logger.info("There are now "+activeVehicleList.size()+" vehicles in the active tNCVehicle list"); - } - - - /** - * First find vehicles that need to refuel, generate a trip to the closest refueling station, then - * remove them from the empty tNCVehicle list, and add them to the refueling tNCVehicle list. - * Next, for all refueling vehicles, check if they are done refueling, and if so, remove them - * from the refueling list and add them to the empty tNCVehicle list. - * - * @param skimPeriod - * @param simulationPeriod - */ - public synchronized void checkForRefuelingVehicles(int skimPeriod, int simulationPeriod) { - - //iterate through zones - for(int i = 1; i <= maxTaz; ++ i){ - if(emptyVehicleList[i]==null) - continue; - - //track the vehicles to remove - ArrayList vehiclesToRemove = new ArrayList(); - - //iterate through vehicles in this zone - for(TNCVehicle tNCVehicle : emptyVehicleList[i]) { - - //if distance since refueling is greater than max, generate a new trip to the closest refueling station. - if(tNCVehicle.getDistanceSinceRefuel()>=maxDistanceBeforeRefuel) { - - ArrayList currentTrips = tNCVehicle.getVehicleTrips(); - TNCVehicleTrip lastTrip = currentTrips.get(currentTrips.size()-1); - - TNCVehicleTrip trip = new TNCVehicleTrip(tNCVehicle,totalVehicleTrips+1); - trip.setStartPeriod(lastTrip.endPeriod); - trip.setOriginMaz(lastTrip.destinationMaz); - trip.setOriginTaz(lastTrip.originTaz); - trip.setPassengers(0); - trip.setOriginPurpose(lastTrip.destinationPurpose); - trip.setDestinationPurpose(Purpose.REFUEL); - - int refeulingMaz = closestMazWithRefeulingStation[trip.getOriginMaz()]; - trip.setDestinationMaz(refeulingMaz); - trip.setDestinationTaz((short) mazManager.getTaz(refeulingMaz)); - float time = transportCostManager.getTime(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz() ); - float distance = transportCostManager.getDistance(skimPeriod, trip.getOriginTaz(), trip.getDestinationTaz()); - float periods = time/(float)minutesPerSimulationPeriod; - int endPeriod = (int) Math.floor(simulationPeriod + periods); - trip.setEndPeriod(endPeriod); - trip.setDistance(distance); - - //add the tNCVehicle trip to the tNCVehicle - tNCVehicle.addVehicleTrip(trip); - - vehiclesToRemove.add(tNCVehicle); - - if(tNCVehicle.getId()==vehicleDebug) { - logger.info("*************"); - logger.info("Vehicle ID "+tNCVehicle.getId()+" refueling trip generated"); - logger.info("From origin MAZ "+trip.getOriginMaz()+" to refueling MAZ "+trip.getDestinationMaz()+" in TAZ "+trip.getDestinationTaz()); - logger.info("Start period "+trip.getStartPeriod()+" end period "+trip.getEndPeriod()); - logger.info("*************"); - - } - - } - - } - //remove all the refueling vehicles from the empty tNCVehicle list - emptyVehicleList[i].removeAll(vehiclesToRemove); - - //add them to the refueling tNCVehicle list - refuelingVehicleList.addAll(vehiclesToRemove); - } - - //track the vehicles to remove - ArrayList vehiclesToRemove = new ArrayList(); - - //iterate through the refueling vehicles - for(TNCVehicle tNCVehicle : refuelingVehicleList ) { - - ArrayList currentTrips = tNCVehicle.getVehicleTrips(); - TNCVehicleTrip lastTrip = currentTrips.get(currentTrips.size()-1); - - //trip is not refueling - if(lastTrip.destinationPurpose!=Purpose.REFUEL) - continue; - - //trip is still en-route to refueling - if(lastTrip.endPeriod>simulationPeriod) - continue; - - - //if its been refueling for appropriate periods, add to empty tNCVehicle list and remove it from the refueling tNCVehicle list - if(tNCVehicle.periodsRefueling==periodsRequiredForRefuel) { - tNCVehicle.setDistanceSinceRefuel(0); - vehiclesToRemove.add(tNCVehicle); - short refuelTaz = lastTrip.destinationTaz; - - if(emptyVehicleList[refuelTaz] == null) - emptyVehicleList[refuelTaz] = new ArrayList(); - - emptyVehicleList[refuelTaz].add(tNCVehicle); - - if(tNCVehicle.getId()==vehicleDebug) { - logger.info("*************"); - logger.info("Vehicle ID "+tNCVehicle.getId()+" has completed refueling in period "+simulationPeriod); - logger.info("Distance to refuel is reset to 0 and vehicle added to empty vehicle list in TAZ "+refuelTaz); - logger.info("*************"); - } - - - // else increment up the number of periods refueling - }else { - tNCVehicle.setPeriodsRefueling(tNCVehicle.getPeriodsRefueling()+1); - if(tNCVehicle.getId()==vehicleDebug) { - logger.info("*************"); - logger.info("Vehicle ID "+tNCVehicle.getId()+" is still refueling in period "+simulationPeriod); - logger.info("*************"); - } - - } - } - - refuelingVehicleList.removeAll(vehiclesToRemove); - } - - - /** - * This method writes tNCVehicle trips to the output file. - * - */ - public void writeVehicleTrips(float sampleRate){ - - logger.info("Writing tNCVehicle trips to file " + vehicleTripOutputFile); - PrintWriter printWriter = null; - try - { - printWriter = new PrintWriter(new BufferedWriter(new FileWriter(vehicleTripOutputFile))); - } catch (IOException e) - { - logger.fatal("Could not open file " + vehicleTripOutputFile + " for writing\n"); - throw new RuntimeException(); - } - - TNCVehicleTrip.printHeader(printWriter); - - //count the total empty vehicles - int totalEmptyVehicles =0; - for(int i = 1; i <= maxTaz; ++ i){ - if(emptyVehicleList[i]==null) - continue; - - totalEmptyVehicles+=emptyVehicleList[i].size(); - } - logger.info("Writing "+totalEmptyVehicles+" total vehicles to file"); - - //reset trip id;wsu - int tripid=0; - for(int i = 1; i <= maxTaz; ++ i){ - if(emptyVehicleList[i]==null) - continue; - - if(emptyVehicleList[i].size()==0) - continue; - - for(TNCVehicle tNCVehicle : emptyVehicleList[i] ){ - if(tNCVehicle.getId()==vehicleDebug) { - logger.info("Writing "+tNCVehicle.getVehicleTrips().size()+" vehicle trips for vehicle ID "+tNCVehicle.getId()); - } - - for(TNCVehicleTrip tNCVehicleTrip : tNCVehicle.getVehicleTrips()){ - - tripid++; - //reorder trip id by wsu - tNCVehicleTrip.setId(tripid); - tNCVehicleTrip.printData(printWriter); - - //save the data in the trip matrix - int startPeriod = tNCVehicleTrip.getStartPeriod(); - int skimPeriod = skimPeriodLookup[startPeriod]; - int origTaz = tNCVehicleTrip.getOriginTaz(); - int destTaz = tNCVehicleTrip.getDestinationTaz(); - int occ = Math.min(tNCVehicleTrip.getPassengers(),3); - - float existingTrips = TNCTripMatrix[skimPeriod][occ].getValueAt(origTaz,destTaz); - TNCTripMatrix[skimPeriod][occ].setValueAt(origTaz,destTaz,existingTrips + (1*(1/sampleRate))); - - - } - } - } - printWriter.close(); - } - - /** - * Get the output trip table file names from the properties file, and write - * trip tables for all modes for the given time period. - * - * @param period - * Time period, which will be used to find the period time string - * to append to each trip table matrix file - */ - public void writeTripTable(MatrixDataServerRmi ms) - { - - String directory = Util.getStringValueFromPropertyMap(propertyMap, "scenario.path"); - String matrixTypeName = Util.getStringValueFromPropertyMap(propertyMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - - for(int i =0;i< transportCostManager.NUM_PERIODS;++i) { - String fileName = directory + Util.getStringValueFromPropertyMap(propertyMap, VEHICLETRIP_OUTPUT_MATRIX_PROPERTY) + "_"+transportCostManager.PERIODS[i]+".omx"; - try{ - //Delete the file if it exists - File f = new File(fileName); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName); - f.delete(); - } - - if (ms != null) - ms.writeMatrixFile(fileName, TNCTripMatrix[i], mt); - else - writeMatrixFile(fileName, TNCTripMatrix[i]); - } catch (Exception e){ - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName, e); - throw new RuntimeException(); - } - } - - } - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - private void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - /** - * Relate simulation periods to skim periods. - * - */ - public void calculateSkimPeriods(){ - - skimPeriodLookup = new byte[numberOfSimulationPeriods]; - int numberSkimPeriods = ModelStructure.SKIM_PERIOD_INDICES.length; - int[] endSkimPeriod = new int[numberSkimPeriods]; - - int lastPeriodEnd = 0; - int lastEndSkimPeriod = 0; - for(int skimPeriod = 0;skimPeriod=0;--skimPeriod){ - if(period pickupIdsAtOrigin; - protected ArrayList dropoffIdsAtOrigin; - protected ArrayList pickupIdsAtDestination; - protected ArrayList dropoffIdsAtDestination; - protected Purpose originPurpose; - protected Purpose destinationPurpose; - protected float distance; - - protected enum Purpose { HOME, PICKUP_ONLY, DROPOFF_ONLY, PICKUP_AND_DROPOFF, REFUEL } - - - public TNCVehicleTrip(TNCVehicle tNCVehicle, int id){ - - this.id=id; - this.tNCVehicle = tNCVehicle; - pickupIdsAtOrigin = new ArrayList(); - dropoffIdsAtOrigin = new ArrayList(); - pickupIdsAtDestination = new ArrayList(); - dropoffIdsAtDestination = new ArrayList(); - originPurpose=Purpose.HOME; - destinationPurpose=Purpose.HOME; - - } - - public ArrayList getPickupIdsAtOrigin() { - return pickupIdsAtOrigin; - } - - public void setPickupIdsAtOrigin(ArrayList pickupIdsAtOrigin) { - this.pickupIdsAtOrigin = pickupIdsAtOrigin; - - if(pickupIdsAtOrigin.isEmpty()) - return; - - if(originPurpose==Purpose.DROPOFF_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.PICKUP_ONLY; - } - - public void addPickupIdsAtOrigin(ArrayList pickupIdsAtOrigin) { - this.pickupIdsAtOrigin.addAll(pickupIdsAtOrigin); - - if(pickupIdsAtOrigin.isEmpty()) - return; - - if(originPurpose==Purpose.DROPOFF_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.PICKUP_ONLY; - - } - public void addPickupIdsAtDestination(ArrayList pickupIdsAtDestination) { - this.pickupIdsAtDestination.addAll(pickupIdsAtDestination); - - if(pickupIdsAtDestination.isEmpty()) - return; - - if(destinationPurpose==Purpose.DROPOFF_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.PICKUP_ONLY; - - } - public ArrayList getDropoffIdsAtOrigin() { - return dropoffIdsAtOrigin; - } - - public void setDropoffIdsAtOrigin(ArrayList dropoffIdsAtOrigin) { - this.dropoffIdsAtOrigin = dropoffIdsAtOrigin; - - if(dropoffIdsAtOrigin.isEmpty()) - return; - - if(originPurpose==Purpose.PICKUP_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.DROPOFF_ONLY; - - } - - public void addDropoffIdsAtOrigin(ArrayList dropoffIdsAtOrigin) { - this.dropoffIdsAtOrigin.addAll(dropoffIdsAtOrigin); - - if(dropoffIdsAtOrigin.isEmpty()) - return; - - if(originPurpose==Purpose.PICKUP_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.DROPOFF_ONLY; - - } - public void addDropoffIdsAtDestination(ArrayList dropoffIdsAtDestination) { - this.dropoffIdsAtDestination.addAll(dropoffIdsAtDestination); - - if(dropoffIdsAtDestination.isEmpty()) - return; - - if(destinationPurpose==Purpose.PICKUP_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.DROPOFF_ONLY; - - } - public ArrayList getPickupIdsAtDestination() { - return pickupIdsAtDestination; - } - - public void setPickupIdsAtDestination(ArrayList pickupIdsAtDestination) { - this.pickupIdsAtDestination = pickupIdsAtDestination; - - if(pickupIdsAtDestination.isEmpty()) - return; - - if(destinationPurpose==Purpose.DROPOFF_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.PICKUP_ONLY; - - } - - public ArrayList getDropoffIdsAtDestination() { - return dropoffIdsAtDestination; - } - - public void setDropoffIdsAtDestination( - ArrayList dropoffIdsAtDestination) { - this.dropoffIdsAtDestination = dropoffIdsAtDestination; - - if(dropoffIdsAtDestination.isEmpty()) - return; - - if(destinationPurpose==Purpose.PICKUP_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.DROPOFF_ONLY; - - } - - - - public void addPickupAtOrigin(String id){ - pickupIdsAtOrigin.add(id); - - if(originPurpose==Purpose.DROPOFF_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.PICKUP_ONLY; - } - - public void addPickupAtDestination(String id){ - pickupIdsAtDestination.add(id); - - if(destinationPurpose==Purpose.DROPOFF_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.PICKUP_ONLY; - } - - public void addDropoffAtOrigin(String id){ - dropoffIdsAtOrigin.add(id); - - if(originPurpose==Purpose.PICKUP_ONLY) - originPurpose = Purpose.PICKUP_AND_DROPOFF; - else - originPurpose = Purpose.DROPOFF_ONLY; - - - } - - public void addDropoffAtDestination(String id){ - dropoffIdsAtDestination.add(id); - - if(destinationPurpose==Purpose.PICKUP_ONLY) - destinationPurpose = Purpose.PICKUP_AND_DROPOFF; - else - destinationPurpose = Purpose.DROPOFF_ONLY; -} - - public int getNumberOfPickupsAtOrigin(){ - return pickupIdsAtOrigin.size(); - } - - public int getNumberOfDropoffsAtOrigin(){ - return dropoffIdsAtOrigin.size(); - } - - public int getNumberOfPickupsAtDestination(){ - return pickupIdsAtDestination.size(); - } - - public int getNumberOfDropoffsAtDestination(){ - return dropoffIdsAtDestination.size(); - } - - public TNCVehicle getVehicle() { - return tNCVehicle; - } - - - public void setVehicle(TNCVehicle tNCVehicle) { - this.tNCVehicle = tNCVehicle; - } - - - public short getOriginTaz() { - return originTaz; - } - - - public void setOriginTaz(short originTaz) { - this.originTaz = originTaz; - } - - - public short getDestinationTaz() { - return destinationTaz; - } - - - public void setDestinationTaz(short destinationTaz) { - this.destinationTaz = destinationTaz; - } - - - public int getOriginMaz() { - return originMaz; - } - - - public void setOriginMaz(int originMaz) { - this.originMaz = originMaz; - } - - - public int getDestinationMaz() { - return destinationMaz; - } - - - public void setDestinationMaz(int destinationMaz) { - this.destinationMaz = destinationMaz; - } - - - public int getPassengers() { - return passengers; - } - - - public void setPassengers(int passengers) { - this.passengers = passengers; - } - - public int getStartPeriod() { - return startPeriod; - } - - public void setStartPeriod(int startPeriod) { - this.startPeriod = startPeriod; - } - - public int getEndPeriod() { - return endPeriod; - } - - public void setEndPeriod(int endPeriod) { - this.endPeriod = endPeriod; - } - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public static void printHeader(PrintWriter writer){ - String record = new String("trip_ID,vehicle_ID,originTaz,destinationTaz,originMgra,destinationMgra,totalPassengers,startPeriod,endPeriod,pickupIdsAtOrigin,dropoffIdsAtOrigin,pickupIdsAtDestination,dropoffIdsAtDestination, originPurpose, destinationPurpose"); - writer.println(record); - writer.flush(); - } - - public void printData(PrintWriter writer){ - - String pickupIdsAtOriginString = ""; - String dropoffIdsAtOriginString = ""; - String pickupIdsAtDestinationString = ""; - String dropoffIdsAtDestinationString = ""; - - if(pickupIdsAtOrigin.size()>0) - for(String pid : pickupIdsAtOrigin) - pickupIdsAtOriginString += (pid + " "); - - if(dropoffIdsAtOrigin.size()>0) - for(String pid : dropoffIdsAtOrigin) - dropoffIdsAtOriginString += (pid + " "); - - if(pickupIdsAtDestination.size()>0) - for(String pid : pickupIdsAtDestination) - pickupIdsAtDestinationString += (pid + " "); - - if(dropoffIdsAtDestination.size()>0) - for(String pid : dropoffIdsAtDestination) - dropoffIdsAtDestinationString += (pid + " "); - - String record = new String( - id + "," + - tNCVehicle.getId() +"," + - originTaz + "," + - destinationTaz + "," + - originMaz + "," + - destinationMaz + "," + - passengers + "," + - startPeriod + "," + - endPeriod + "," + - pickupIdsAtOriginString + "," + - dropoffIdsAtOriginString + "," + - pickupIdsAtDestinationString + "," + - dropoffIdsAtDestinationString + "," + - originPurpose.ordinal() + "," + - destinationPurpose.ordinal()); - - writer.println(record); - writer.flush(); - } - - public void writeTrace(){ - - String pickupIdsAtOriginString = ""; - String dropoffIdsAtOriginString = ""; - String pickupIdsAtDestinationString = ""; - String dropoffIdsAtDestinationString = ""; - - if(pickupIdsAtOrigin.size()>0) - for(String pid : pickupIdsAtOrigin) - pickupIdsAtOriginString += (pid + " "); - - if(dropoffIdsAtOrigin.size()>0) - for(String pid : dropoffIdsAtOrigin) - dropoffIdsAtOriginString += (pid + " "); - - if(pickupIdsAtDestination.size()>0) - for(String pid : pickupIdsAtDestination) - pickupIdsAtDestinationString += (pid + " "); - - if(dropoffIdsAtDestination.size()>0) - for(String pid : dropoffIdsAtDestination) - dropoffIdsAtDestinationString += (pid + " "); - - logger.info("*********************************************************"); - logger.info("Trace for tNCVehicle trip "+id+" in tNCVehicle "+tNCVehicle.getId()); - logger.info("Trip ID: " + id); - logger.info("TNCVehicle ID: "+tNCVehicle.getId()); - logger.info("Origin TAZ: "+originTaz); - logger.info("Destination TAZ: "+destinationTaz); - logger.info("Origin MAZ: "+originMaz); - logger.info("Destination MAZ: "+destinationMaz); - logger.info("Passengers: "+passengers); - logger.info("Start period: "+startPeriod); - logger.info("End period: "+endPeriod); - logger.info("Pickups at Origin: "+ pickupIdsAtOriginString); - logger.info("Dropoffs at Origin: "+ dropoffIdsAtOriginString); - logger.info("Pickups at Destination: "+ pickupIdsAtDestinationString); - logger.info("Dropoffs at Destination: "+ dropoffIdsAtDestinationString); - logger.info("Origin Purpose: "+ originPurpose); - logger.info("Destination Purpose: "+ destinationPurpose); - - logger.info("*********************************************************"); - - - } - - public Purpose getOriginPurpose() { - return originPurpose; - } - - public void setOriginPurpose(Purpose originPurpose) { - this.originPurpose = originPurpose; - } - - public Purpose getDestinationPurpose() { - return destinationPurpose; - } - - public void setDestinationPurpose(Purpose destinationPurpose) { - this.destinationPurpose = destinationPurpose; - } - - - public float getDistance() { - return distance; - } - - public void setDistance(float distance) { - this.distance = distance; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java b/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java deleted file mode 100644 index e0f64db..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/maas/TransportCostManager.java +++ /dev/null @@ -1,438 +0,0 @@ -package org.sandag.abm.maas; - -import java.io.File; -import java.util.ArrayList; -import java.util.Collection; -import java.util.Collections; -import java.util.HashMap; -import java.util.concurrent.Executors; -import java.util.concurrent.ThreadPoolExecutor; -import java.util.concurrent.TimeUnit; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -import drasys.or.util.Array; - -public class TransportCostManager { - - protected transient Logger logger = Logger.getLogger(TransportCostManager.class); - - protected static final int EA = ModelStructure.EA_SKIM_PERIOD_INDEX; - protected static final int AM = ModelStructure.AM_SKIM_PERIOD_INDEX; - protected static final int MD = ModelStructure.MD_SKIM_PERIOD_INDEX; - protected static final int PM = ModelStructure.PM_SKIM_PERIOD_INDEX; - protected static final int EV = ModelStructure.EV_SKIM_PERIOD_INDEX; - public static final int NUM_PERIODS = ModelStructure.SKIM_PERIOD_INDICES.length; - protected static final String[] PERIODS = ModelStructure.SKIM_PERIOD_STRINGS; - - protected static int TAZ_CALCULATOR_THREADS = 20; //default - - //by period, origin, destination - ragged array of zone numbers of zones within max time diversion - //sorted by time from origin (assuming pickups would be en-route) - protected short[][][][] tazsWithinOriginAndDestination; - // private float[][][][] addTimeWithinOriginAndDestination; - - //by period, origin, destination - protected float[][][] tazTimeSkims; //travel time - protected float[][][] tazDistanceSkims; //travel distance - - protected short[][][] tazsByTimeFromOrigin; //array of TAZs sorted by time from origin, by period and origin TAZ - - protected float maxTimeDiversion; - protected float maxDistanceToPickup; - protected int maxTaz; - - // declare an array of UEC objects, 1 for each time period - protected UtilityExpressionCalculator[] autoDistOD_UECs; - protected UtilityExpressionCalculator[] autoTimeOD_UECs; - - // The simple auto skims UEC does not use any DMU variables - protected VariableTable dmu = null; - protected TazDataManager tazManager; - int totalThreads; - - - /** - * Instantiate transport cost manager. - * - * @param rbMap - * @param maxTimeDiversion - */ - public TransportCostManager(HashMap rbMap, float maxTimeDiversion, float maxDistanceToPickup) - { - - this.maxTimeDiversion=maxTimeDiversion; - this.maxDistanceToPickup=maxDistanceToPickup; - - // Create the UECs - String uecPath = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String uecFileName = uecPath - + Util.getStringValueFromPropertyMap(rbMap, "taz.distance.uec.file"); - int dataPage = Util.getIntegerValueFromPropertyMap(rbMap, "taz.distance.data.page"); - - - //iterate thru settings in properties file and create time and distance UECs - autoDistOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; - autoTimeOD_UECs = new UtilityExpressionCalculator[NUM_PERIODS]; - File uecFile = new File(uecFileName); - - for(int i =0; i stopTazList = new ArrayList(); - - for (int oTaz = startOriginTaz; oTaz <= endOriginTaz; oTaz++){ - - if((oTaz==startOriginTaz)||(oTaz % 100 == 0)) - logger.info("Thread "+threadName + " Period "+period+" Origin TAZ "+oTaz); - - for (int dTaz = 1; dTaz <= maxTaz; dTaz++){ - - stopTazList.clear(); - - //Stop TAZs - for(int kTaz = 1; kTaz <= maxTaz; ++kTaz){ - - //Calculate additional time to stop - float ikTime = tazTimeSkims[period][oTaz][kTaz]; - float kjTime = tazTimeSkims[period][kTaz][dTaz]; - float totalIKJTime = ikTime + kjTime; - float divertTime = totalIKJTime - tazTimeSkims[period][oTaz][dTaz]; - - //if time is less than max diversion time (or the stop zone is the origin or destination zone), add zone and time to arraylist - if( (divertTime < maxTimeDiversion) || (kTaz==oTaz) || (kTaz==dTaz)){ - StopTaz stopTaz = new StopTaz(); - stopTaz.tazNumber = kTaz; - stopTaz.diversionTime = divertTime; - stopTaz.originStopTime = ikTime; - stopTazList.add(stopTaz); - } - - } //end for stops - - //initialize arrays for saving tazs, time and set the values in the ragged arrays - if(!stopTazList.isEmpty()){ - Collections.sort(stopTazList); - int numberOfStops = stopTazList.size(); - tazsWithinOriginAndDestination[period][oTaz][dTaz] = new short[numberOfStops]; - //addTimeWithinOriginAndDestination[period][oTaz][dTaz] = new float[numberOfStops]; - - for(int k = 0; k < numberOfStops; ++k){ - StopTaz stopTaz = stopTazList.get(k); - tazsWithinOriginAndDestination[period][oTaz][dTaz][k] = (short) stopTaz.tazNumber; - //addTimeWithinOriginAndDestination[period][oTaz][dTaz][k] = stopTaz.diversionTime; - } - } - } - - } - - - - - } - } - - /** - * This method finds stop zones for each origin-destination zone pair and saves the zone number - * and diversion time, sorted by distance from origin. - * - */ - private void calculateTazsWithinDistanceThreshold(){ - - - tazsWithinOriginAndDestination = new short[NUM_PERIODS][maxTaz+1][maxTaz+1][]; - //addTimeWithinOriginAndDestination = new float[NUM_PERIODS][maxTaz+1][maxTaz+1][]; - int processors = Runtime.getRuntime().availableProcessors(); - //use 80% of the machine's processing power - TAZ_CALCULATOR_THREADS = totalThreads; - int chunkSize = (int) Math.floor(maxTaz / TAZ_CALCULATOR_THREADS); - - logger.info("...Calculating TAZs within distance thresholds with "+TAZ_CALCULATOR_THREADS+ " threads ("+processors+" processors)"); - - for( int period = 0; period < NUM_PERIODS;++period ){ - - int endZone = 0; - - ThreadPoolExecutor executor = (ThreadPoolExecutor) Executors.newFixedThreadPool(TAZ_CALCULATOR_THREADS); - - for(int i = 0; i < TAZ_CALCULATOR_THREADS; ++ i){ - - int startZone = endZone + 1; - - if(i==(TAZ_CALCULATOR_THREADS-1)) - endZone = maxTaz; - else - endZone = startZone+chunkSize; - - executor.execute(new TazDistanceCalculatorThread( "Thread-"+i,period,startZone,endZone)); - - } - executor.shutdown(); - try{ - executor.awaitTermination(60, TimeUnit.MINUTES); - }catch(InterruptedException e){ - throw new RuntimeException(e); - } - } - } - - /** - * Calculate zones sorted by time from origin. Always include intrazonal as within the maximum distance range. - * - */ - public void calculateTazsByTimeFromOrigin(){ - - ArrayList stopTazList = new ArrayList(); - - tazsByTimeFromOrigin = new short[NUM_PERIODS][maxTaz+1][]; - - for(int period = 0; period that.originStopTime) return AFTER; - - return EQUAL; - } - - - } - - /** - * Get the array of zones that are within the diversion time from the origin to the - * destination, sorted by time from origin. - * - * @param skimPeriod - * @param origTaz - * @param destTaz - * @return The array of zones, or null if there are no zones within the max diversion time. - */ - public short[] getZonesWithinMaxDiversionTime(int skimPeriod, int origTaz, int destTaz){ - - return tazsWithinOriginAndDestination[skimPeriod][origTaz][destTaz]; - - } - - /** - * Is the zone within the set of zones that is within maximum diversion time from the origin to the destination? - * - * @param skimPeriod - * @param origTaz The origin TAZ - * @param destTaz The destination TAZ - * @param taz The stop TAZ - * @return A boolean indicating whether the zone is within the maximum deviation time from the origin to the destination. - */ - public boolean stopZoneIsWithinMaxDiversionTime(int skimPeriod, int origTaz, int destTaz, int taz){ - - short[] tazArray = getZonesWithinMaxDiversionTime(skimPeriod, origTaz, destTaz); - for(int i = 0; i < tazArray.length; ++i) - if(tazArray[i]==taz) - return true; - return false; - - } - - - - /** - * Get the diversion times for the zones that are within the diversion time from the origin to the - * destination, sorted by time from origin. - * - * @param period - * @param origTaz - * @param destTaz - * @return The array of diversion times, or null if there are no zones within the max diversion time. - */ - public float[] getDiversionTimes(int period, int origTaz, int destTaz){ - - //return addTimeWithinOriginAndDestination[period][origTaz][destTaz]; - logger.fatal("Error trying to call getDiversionTimes when additional time array not initialized"); - throw new RuntimeException(); - } - - /** - * Get a ragged array of zone numbers sorted by time from the origin. The array is ragged - * because it is capped by the maximum distance for hailing a TNC\TAXI. - * - * @param period - * @param origTaz - * @return A sorted array of zone numbers, or null if there are no zones within the maximum distance. - */ - public short[] getZoneNumbersSortedByTime(int period, int origTaz){ - - return tazsByTimeFromOrigin[period][origTaz]; - } - - public float getTime(int period, int origTaz, int destTaz){ - - return tazTimeSkims[period][origTaz][destTaz]; - } - - public float getDistance(int period, int origTaz, int destTaz){ - - return tazDistanceSkims[period][origTaz][destTaz]; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java deleted file mode 100644 index 59823a0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoDMU.java +++ /dev/null @@ -1,136 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Mar 9, 2009 - *

    - * Created by IntelliJ IDEA. - */ -public class AutoDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(AutoDMU.class); - - protected HashMap methodIndexMap; - - private double avgHourlyParkingCostAtDestTaz; - private float pTazTerminalTime; - private float aTazTerminalTime; - - public AutoDMU() - { - setupMethodIndexMap(); - } - - public double getAvgHourlyParkingCostAtDestTaz() - { - return avgHourlyParkingCostAtDestTaz; - } - - public void setAvgHourlyParkingCostAtDestTaz(double cost) - { - avgHourlyParkingCostAtDestTaz = cost; - } - - public float getPTazTerminalTime() - { - return pTazTerminalTime; - } - - public void setPTazTerminalTime(float pTazTerminalTime) - { - this.pTazTerminalTime = pTazTerminalTime; - } - - public float getATazTerminalTime() - { - return aTazTerminalTime; - } - - public void setATazTerminalTime(float aTazTerminalTime) - { - this.aTazTerminalTime = aTazTerminalTime; - } - - /** - * Log the DMU values. - * - * @param localLogger - * The logger to use. - */ - public void logValues(Logger localLogger) - { - - localLogger.info(""); - localLogger.info("Auto DMU Values:"); - localLogger.info(""); - localLogger.info(String.format("Average TAZ Parking cost at destination: %9f", - avgHourlyParkingCostAtDestTaz)); - localLogger.info(String.format("Production/Origin Terminal Time: %9.4f", pTazTerminalTime)); - localLogger.info(String.format("Attraction/Destin Terminal Time: %9.4f", aTazTerminalTime)); - - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getAvgHourlyParkingCostAtDestTaz", 0); - methodIndexMap.put("getATazTerminalTime", 1); - methodIndexMap.put("getPTazTerminalTime", 2); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getAvgHourlyParkingCostAtDestTaz(); - case 1: - return getATazTerminalTime(); - case 2: - return getPTazTerminalTime(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java deleted file mode 100644 index 2481944..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/AutoUEC.java +++ /dev/null @@ -1,138 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.LogitModel; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Mar 9, 2009 - *

    - * Created by IntelliJ IDEA. - */ -public class AutoUEC - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(AutoUEC.class); - private TazDataManager tazs; - private UtilityExpressionCalculator uec; - private LogitModel model; - private ChoiceModelApplication modelApp; - - private IndexValues index = new IndexValues(); - private int[] availFlag; - private AutoDMU dmu; - - // seek and trace - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - protected Tracer tracer; - - /** - * Constructor. - * - * @param rb - * ResourceBundle - * @param UECFileName - * The path/name of the UEC containing the auto model. - * @param modelSheet - * The sheet (0-indexed) containing the model specification. - * @param dataSheet - * The sheet (0-indexed) containing the data specification. - */ - public AutoUEC(HashMap rbHashMap, String uecFileName, int modelSheet, - int dataSheet) - { - - dmu = new AutoDMU(); - - // use the choice model application to set up the model structure - modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); - - // but return the logit model itself, so we can use compound utilities - model = modelApp.getRootLogitModel(); - uec = modelApp.getUEC(); - - tazs = TazDataManager.getInstance(); - trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - } - - /** - * Solve auto utilities for a given zone-pair - * - * @param pTaz - * Production/Origin TAZ. - * @param aTaz - * Attraction/Destination TAZ. - * @return The root utility. - */ - public double calculateUtilitiesForTazPair(int pTaz, int aTaz, double avgTazHourlyParkingCost) - { - - trace = false; - if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) - { - trace = true; - } - index.setOriginZone(pTaz); - index.setDestZone(aTaz); - availFlag = new int[uec.getNumberOfAlternatives() + 1]; - Arrays.fill(availFlag, 1); - - dmu.setAvgHourlyParkingCostAtDestTaz(avgTazHourlyParkingCost); - dmu.setPTazTerminalTime(tazs.getOriginTazTerminalTime(pTaz)); - dmu.setATazTerminalTime(tazs.getDestinationTazTerminalTime(aTaz)); - - // log DMU values - if (trace) - { - TapDataManager tapManager = TapDataManager.getInstance(); - if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 - && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) - uec.logDataValues(logger, pTaz, aTaz, aTaz); - dmu.logValues(logger); - } - - modelApp.computeUtilities(dmu, index); - double utility = modelApp.getLogsum(); - - // logging - if (trace) - { - uec.logAnswersArray(logger, "Auto UEC"); - uec.logResultsArray(logger, pTaz, aTaz); - modelApp.logLogitCalculations("Auto UEC", "Trace"); - logger.info("Logsum = " + utility); - trace = false; - } - - return utility; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java deleted file mode 100644 index 46c933d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Constants.java +++ /dev/null @@ -1,56 +0,0 @@ -package org.sandag.abm.modechoice; - -/** - * This class is used for storing constants. Many of these are listed in the - * sandag.inc file associated with the FORTRAN code. We should eventually move - * these into a properties file and have this class set them from the prop file. - * - * I am just trying to not get bogged down in the details. - * - * @author Christi Willison - * @version Nov 6, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public final class Constants -{ - - public static int MAX_EXTERNAL = 12; - public static float AutoCostPerMile = 10.0f; - - public static float[][] parkingCost = { {0.0f, 50.0f, 200.0f, 300.0f, 400.0f}, - {0.0f, 50.0f, 125.0f, 200.0f, 400.0f}, {0.0f, 50.0f, 100.0f, 200.0f, 400.0f}}; - - public static float walkMinutesPerMile = 20.0f; // 20 - // minutes - // per - // mile - // (dist - // is - // in - // feet) - // or - // 3 - // mph. - public static float bikeMinutesPerMile = 5.0f; // 5 - - // minutes - // per - // mile - // (dist - // is - // in - // feet) - // or - // 12 - // mph. - - public static float feetPerMile = 5280.0f; - public static double walkMinutesPerFoot = walkMinutesPerMile/feetPerMile; - public static double bikeMinutesPerFoot = bikeMinutesPerMile/feetPerMile; - - private Constants() - { - // Not Implemented - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java deleted file mode 100644 index 52cd382..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasDMU.java +++ /dev/null @@ -1,128 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -/** - * This class is the DMU object for MAAS - * joel freedman - * RSG 2019-07-08 - **/ - -public class MaasDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(MaasDMU.class); - - protected HashMap methodIndexMap; - - protected float waitTimeTaxi; - protected float waitTimeSingleTNC; - protected float waitTimeSharedTNC; - - public MaasDMU() - { - setupMethodIndexMap(); - } - - public float getWaitTimeTaxi() { - return waitTimeTaxi; - } - - public void setWaitTimeTaxi(float waitTimeTaxi) { - this.waitTimeTaxi = waitTimeTaxi; - } - - public float getWaitTimeSingleTNC() { - return waitTimeSingleTNC; - } - - public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { - this.waitTimeSingleTNC = waitTimeSingleTNC; - } - - public float getWaitTimeSharedTNC() { - return waitTimeSharedTNC; - } - - public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { - this.waitTimeSharedTNC = waitTimeSharedTNC; - } - - - /** - * Log the DMU values. - * - * @param localLogger - * The logger to use. - */ - public void logValues(Logger localLogger) - { - - localLogger.info(""); - localLogger.info("Maas DMU Values:"); - localLogger.info(""); - localLogger.info(String.format("Taxi wait time: %9.2f", waitTimeTaxi)); - localLogger.info(String.format("Single TNC wait time: %9.2f", waitTimeSingleTNC)); - localLogger.info(String.format("Shared TNC wait time: %9.2f", waitTimeSharedTNC)); - - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getWaitTimeTaxi", 0); - methodIndexMap.put("getWaitTimeSingleTNC", 1); - methodIndexMap.put("getWaitTimeSharedTNC", 2); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getWaitTimeTaxi(); - case 1: - return getWaitTimeSingleTNC(); - case 2: - return getWaitTimeSharedTNC(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java deleted file mode 100644 index 682f7b8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MaasUEC.java +++ /dev/null @@ -1,141 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; -import org.sandag.abm.ctramp.Util; -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.LogitModel; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -/** - * This class is the UEC for MAAS - * Joel Freedman - * RSG 2019-07-08 - */ -public class MaasUEC - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(MaasUEC.class); - private TazDataManager tazs; - private UtilityExpressionCalculator uec; - private LogitModel model; - private ChoiceModelApplication modelApp; - - private IndexValues index = new IndexValues(); - private int[] availFlag; - private MaasDMU dmu; - - // seek and trace - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - protected Tracer tracer; - - - - - /** - * Constructor. - * - * @param rb - * ResourceBundle - * @param UECFileName - * The path/name of the UEC containing the auto model. - * @param modelSheet - * The sheet (0-indexed) containing the model specification. - * @param dataSheet - * The sheet (0-indexed) containing the data specification. - */ - public MaasUEC(HashMap rbHashMap, String uecFileName, int modelSheet, - int dataSheet) - { - - dmu = new MaasDMU(); - - // use the choice model application to set up the model structure - modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); - - // but return the logit model itself, so we can use compound utilities - model = modelApp.getRootLogitModel(); - uec = modelApp.getUEC(); - - tazs = TazDataManager.getInstance(); - trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - if (trace) - { - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - } - } - } - - - } - - /** - * Solve auto utilities for a given zone-pair - * - * @param pTaz - * Production/Origin TAZ. - * @param aTaz - * Attraction/Destination TAZ. - * @return The root utility. - */ - public double calculateUtilitiesForTazPair(int pTaz, int aTaz, float avgTaxiWaitTime, float avgSingleTNCWaitTime,float avgSharedTNCWaitTime) - { - - trace = false; - if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) - { - trace = true; - } - index.setOriginZone(pTaz); - index.setDestZone(aTaz); - availFlag = new int[uec.getNumberOfAlternatives() + 1]; - Arrays.fill(availFlag, 1); - - dmu.setWaitTimeTaxi(avgTaxiWaitTime); - dmu.setWaitTimeSingleTNC(avgSingleTNCWaitTime); - dmu.setWaitTimeSharedTNC(avgSharedTNCWaitTime); - - // log DMU values - if (trace) - { - TapDataManager tapManager = TapDataManager.getInstance(); - if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 - && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) - uec.logDataValues(logger, pTaz, aTaz, aTaz); - dmu.logValues(logger); - } - - modelApp.computeUtilities(dmu, index); - double utility = modelApp.getLogsum(); - - // logging - if (trace) - { - uec.logAnswersArray(logger, "Maas UEC"); - uec.logResultsArray(logger, pTaz, aTaz); - modelApp.logLogitCalculations("Maas UEC", "Trace"); - logger.info("Logsum = " + utility); - trace = false; - } - - return utility; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java deleted file mode 100644 index 06a2bee..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/MgraDataManager.java +++ /dev/null @@ -1,1551 +0,0 @@ -package org.sandag.abm.modechoice; - - -import org.sandag.abm.active.sandag.SandagWalkPathAlternativeListGenerationConfiguration; -import org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication; -import org.sandag.abm.ctramp.BikeLogsum; -import org.sandag.abm.ctramp.BikeLogsumSegment; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.Modes.AccessMode; - -import java.io.BufferedReader; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileNotFoundException; -import java.io.FileReader; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.io.Serializable; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import java.util.ResourceBundle; -import java.util.Set; -import java.util.StringTokenizer; -import java.util.TreeSet; - -import org.apache.log4j.Logger; - -import com.pb.common.datafile.CSVFileReader; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Sep 4, 2008 - *

    - * Created by IntelliJ IDEA. - * - * Edited JEF May 2009 - */ -public final class MgraDataManager - implements Serializable -{ - - private static MgraDataManager instance; - protected transient Logger logger = Logger.getLogger(MgraDataManager.class); - - public static final double MAX_PARKING_WALK_DISTANCE = 0.75; - - private static final int LOG_MGRA = -4502; - private static final String LOG_MGRA_FILE = LOG_MGRA - + "debug"; - private static final String MGRA_MGRA_WALK_FILE_PROPERTY = "active.micromobility.file.walk.mgra"; - private static final String MGRA_TAP_WALK_FILE_PROPERTY = "active.micromobility.file.walk.mgratap"; - - - // create Strubg variables for the 4D land use data file field names - public static final String MGRA_4DDENSITY_DU_DEN_FIELD = "DUDen"; - public static final String MGRA_4DDENSITY_EMP_DEN_FIELD = "EmpDen"; - public static final String MGRA_4DDENSITY_TOT_INT_FIELD = "TotInt"; - - // public static final String MGRA_FIELD_NAME = "MGRASR10"; - public static final String MGRA_FIELD_NAME = "mgra"; - public static final String MGRA_TAZ_FIELD_NAME = "TAZ"; - public static final String MGRA_LUZ_FIELD_NAME = "luz_id"; - private static final String MGRA_POPULATION_FIELD_NAME = "pop"; - private static final String MGRA_HOUSEHOLDS_FIELD_NAME = "hh"; - private static final String MGRA_GRADE_SCHOOL_ENROLLMENT_FIELD_NAME = "EnrollGradeKto8"; - private static final String MGRA_HIGH_SCHOOL_ENROLLMENT_FIELD_NAME = "EnrollGrade9to12"; - private static final String MGRA_UNIVERSITY_ENROLLMENT_FIELD_NAME = "collegeEnroll"; - private static final String MGRA_OTHER_COLLEGE_ENROLLMENT_FIELD_NAME = "otherCollegeEnroll"; - private static final String MGRA_ADULT_SCHOOL_ENROLLMENT_FIELD_NAME = "AdultSchEnrl"; - private static final String MGRA_GRADE_SCHOOL_DISTRICT_FIELD_NAME = "ech_dist"; - private static final String MGRA_HIGH_SCHOOL_DISTRICT_FIELD_NAME = "hch_dist"; - private static final String MGRA_REFUELING_STATIONS_FIELD_NAME = "refueling_stations"; - private static final String MGRA_REMOTE_PARKING_LOT_FIELD_NAME = "remoteAVParking"; - - private static final String PROPERTIES_PARKING_COST_OUTPUT_FILE = "mgra.avg.cost.output.file"; - - - public static final String PROPERTIES_MGRA_DATA_FILE = "mgra.socec.file"; - private static final String MGRA_DISTANCE_COEFF_WORK = "mgra.avg.cost.dist.coeff.work"; - private static final String MGRA_DISTANCE_COEFF_OTHER = "mgra.avg.cost.dist.coeff.other"; - - public static final int PARK_AREA_ONE = 1; - private static final String MGRA_PARKAREA_FIELD = "parkarea"; - private static final String MGRA_HSTALLSOTH_FIELD = "hstallsoth"; - private static final String MGRA_HSTALLSSAM_FIELD = "hstallssam"; - private static final String MGRA_HPARKCOST_FIELD = "hparkcost"; - private static final String MGRA_NUMFREEHRS_FIELD = "numfreehrs"; - private static final String MGRA_DSTALLSOTH_FIELD = "dstallsoth"; - private static final String MGRA_DSTALLSSAM_FIELD = "dstallssam"; - private static final String MGRA_DPARKCOST_FIELD = "dparkcost"; - private static final String MGRA_MSTALLSOTH_FIELD = "mstallsoth"; - private static final String MGRA_MSTALLSSAM_FIELD = "mstallssam"; - private static final String MGRA_MPARKCOST_FIELD = "mparkcost"; - - //for TNC and Taxi wait time calculations - private static final String MGRA_POPEMPPERSQMI_FIELD = "PopEmpDenPerMi"; - private ArrayList mgras = new ArrayList(); - private int maxMgra; - private int maxLuz; - - private int maxTap; - private int nMgrasWithWlkTaps; - // [mgra], [0=tapID, 1=Distance], [tap number (0-number of taps)] - private int[][][] mgraWlkTapsDistArray; - private int[] mgraTaz; - private int[] mgraLuz; - - // An array of Hashmaps dimensioned by origin mgra, with distance in feet, - // in a ragged - // array (no key for mgra means no other mgras in walk distance) - private HashMap[] oMgraWalkDistance; - - // An array of Hashmaps dimensioned by destination mgra, with distance in - // feet, in a ragged - // array (no key for mgra means no other mgras in walk distance) - private HashMap[] dMgraWalkDistance; - - private BikeLogsum bls; - //segment doesn't matter as it is now just a passthrough - private BikeLogsumSegment defaultSegment = new BikeLogsumSegment(true,true,true); - - // An array dimensioned to maxMgra of ragged arrays of lists of TAPs - // accessible by driving - private Set[] driveAccessibleTaps; - private Set[] walkAccessibleTaps; - - //by TAP, closest mgra to the tap by walking distance. - private int[] closestMgraToTap; - - private TableDataSet mgraTableDataSet; - - private HashMap mgraDataTableMgraRowMap; - - private double[] duDen; - private double[] empDen; - private double[] totInt; - private double[] popEmpDenPerSqMi; - - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - - private int[] mgraParkArea; - private int[] numfreehrs; - private int[] hstallsoth; - private int[] hstallssam; - private float[] hparkcost; - private int[] dstallsoth; - private int[] dstallssam; - private float[] dparkcost; - private int[] mstallsoth; - private int[] mstallssam; - private float[] mparkcost; - - private TableDataSet tapLinesTable; - private HashMap taplines; - - /** - * Constructor. - * - * @param rbMap - * A HashMap created from a resourcebundle with model properties. - * - */ - private MgraDataManager(HashMap rbMap) - { - System.out.println("I'm the MgraDataManager"); - readMgraTableData(rbMap); - readMgraWlkTaps(rbMap); - readMgraWlkDist(rbMap); - - readTapLines(rbMap); - trimTapSet(); - - - - bls = BikeLogsum.getBikeLogsum(rbMap); - - // pre-process the list of TAPS reachable by drive access for each MGRA - mapDriveAccessTapsToMgras(TazDataManager.getInstance(rbMap)); - - // create arrays from 4ddensity fields added to MGRA table used by - // TourModeChoice DMU methods - process4ddensityData(rbMap); - - calculateMgraAvgParkingCosts(rbMap); - - calculateClosestMgraToTap(); - - printMgraStats(); - } - - /** - * Find the closest mgra by walk time to the tap. - */ - public void calculateClosestMgraToTap() { - - closestMgraToTap = new int[maxTap+1]; - float[] minTimeToTap = new float[maxTap+1]; - Arrays.fill(minTimeToTap, 999999); - - for(int mgra = 1; mgra taps = walkAccessibleTaps[mgra]; - for(int tap : taps) { - int pos = getTapPosition(mgra, tap); - float time = getMgraToTapWalkTime(mgra,pos); - if(time rbMap) - { - if (instance == null) - { - instance = new MgraDataManager(rbMap); - return instance; - } else return instance; - } - - /** - * This method should only be used after the getInstance(ResourceBundle rb) - * method has been called since the rb is needed to read in all the data and - * populate the object. This method will return the instance that has - * already been populated. - * - * @return instance - * @throws RuntimeException - */ - public static MgraDataManager getInstance() - { - if (instance == null) - { - throw new RuntimeException( - "Must instantiate MgraDataManager with the getInstance(rb) method first"); - } else - { - return instance; - } - } - - /** - * Read the walk-transit taps for mgras. - * - * @param rb - * The resourcebundle with the scenario.path and - * mgra.wlkacc.taps.and.distance.file properties. - */ - public void readMgraWlkTaps(HashMap rbMap) - { - String mgraWlkTapTimeFile = rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT) - +rbMap.get(MGRA_TAP_WALK_FILE_PROPERTY); - - TableDataSet mgraTapData = Util.readTableDataSet(mgraWlkTapTimeFile); - - Map> mgraWlkTapList = new HashMap<>(); //mgra -> tap -> distance - - //mgra,tap,walkTime,dist,mmTime,mmCost,mtTime,mtCost,mmGenTime,mtGenTime,minTime - for(int row = 1; row <= mgraTapData.getRowCount();++row) { - - int mgra = (int) mgraTapData.getValueAt(row, "mgra"); - int tap = (int) mgraTapData.getValueAt(row, "tap"); - if (tap > maxTap) maxTap = tap; - float minTime = mgraTapData.getValueAt(row,"minTime"); - - int distance = Math.round(minTime / Constants.walkMinutesPerMile * Constants.feetPerMile); - - //reset 0 distances to 0.1 miles, and log potential error - if(distance==0){ - //logger.info("Potential error: Distance from mgra "+mgra+" to tap "+tap+" is 0; resetting to 0.1 miles"); - distance = Math.round(Constants.feetPerMile * (float)0.1); - } - - if (!mgraWlkTapList.containsKey(mgra)) - mgraWlkTapList.put(mgra,new HashMap()); - mgraWlkTapList.get(mgra).put(tap,distance); - } - - // now go thru the array of ArrayLists and convert the lists to arrays - // and - // store in the class variable mgraWlkTapsDistArrays. - mgraWlkTapsDistArray = new int[maxMgra + 1][2][]; - nMgrasWithWlkTaps = mgraWlkTapList.size(); - for (int mgra : mgraWlkTapList.keySet()) { - Map wlkTapList = mgraWlkTapList.get(mgra); - mgraWlkTapsDistArray[mgra][0] = new int[wlkTapList.size()]; - mgraWlkTapsDistArray[mgra][1] = new int[wlkTapList.size()]; - int counter = 0; - for (int tap : new TreeSet(wlkTapList.keySet())) { //get the taps in ascending order - not sure if this matters, but it is cleaner - int distance = wlkTapList.get(tap); - mgraWlkTapsDistArray[mgra][0][counter] = tap; - mgraWlkTapsDistArray[mgra][1][counter] = distance; - counter++; - } - } - } - - /** - * read tap lines table (tap, line names served) - * @param rbMap - */ - public void readTapLines(HashMap rbMap) { - - File tapLinesTableFile = Paths.get(Util.getStringValueFromPropertyMap(rbMap, "scenario.path"), - Util.getStringValueFromPropertyMap(rbMap, "maz.tap.tapLines")).toFile(); - try { - CSVFileReader csvReader = new CSVFileReader(); - tapLinesTable = csvReader.readFile( tapLinesTableFile); - } catch (IOException e) { - throw new RuntimeException(); - } - - //get tap lines table field names - int[] tapLinesTapIds = tapLinesTable.getColumnAsInt("TAP"); - String[] linesForTap = tapLinesTable.getColumnAsString("LINES"); - - //create lookups - taplines = new HashMap(); - for(int i=0; i maz2TapData = new ArrayList(); - for (int j=0; j linesServed = new HashMap(); - for (Maz2Tap m2t : maz2TapData) { - - //skip if no lines served - if(m2t.lines != null) { - - for (int k=0; k tapsToRemove = new ArrayList(); - for (Maz2Tap m2t : maz2TapData) { - mazToTaps = mazToTaps + 1; - if( m2t.servesNewLines == false) { - tapsToRemove.add(m2t.tap); - trimmedTaps = trimmedTaps + 1; - } - } - - int[] finalTaps = new int[taps.length-tapsToRemove.size()]; - int[] finalDistances = new int[taps.length-tapsToRemove.size()]; - - int tapCounter = 0; - for (int m=0; m rbMap) - { - String mgraWlkTimeFile = rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT) - + rbMap.get(MGRA_MGRA_WALK_FILE_PROPERTY); - oMgraWalkDistance = new HashMap[maxMgra + 1]; - dMgraWalkDistance = new HashMap[maxMgra + 1]; - - TableDataSet mgraWalkData = Util.readTableDataSet(mgraWlkTimeFile); - - //i,j,walkTime,dist,mmTime,mmCost,mtTime,mtCost,mmGenTime,mtGenTime,minTime - for(int row = 1; row <= mgraWalkData.getRowCount();++row) { - - int oMgra = (int) mgraWalkData.getValueAt(row, "i"); - int dMgra = (int) mgraWalkData.getValueAt(row, "j"); - int distance = Math.round( mgraWalkData.getValueAt(row, "minTime") / Constants.walkMinutesPerMile * Constants.feetPerMile); - - if (oMgraWalkDistance[oMgra] == null) - oMgraWalkDistance[oMgra] = new HashMap(); - oMgraWalkDistance[oMgra].put(dMgra, distance); - - if (dMgraWalkDistance[dMgra] == null) - dMgraWalkDistance[dMgra] = new HashMap(); - dMgraWalkDistance[dMgra].put(oMgra, distance); - } - - } - - /** - * Return an int array of mgras within walking distance of this mgra. - * - * @param mgra - * The mgra to look up - * @return The mgras within walking distance. Null is returned if no mgras - * are within walk distance. - */ - public int[] getMgrasWithinWalkDistanceFrom(int mgra) - { - - if (oMgraWalkDistance[mgra] == null) return null; - - Set keySet = oMgraWalkDistance[mgra].keySet(); - int[] walkMgras = new int[keySet.size()]; - Iterator it = keySet.iterator(); - int i = 0; - while (it.hasNext()) - { - walkMgras[i] = it.next(); - ++i; - } - return walkMgras; - - } - - /** - * Return an int array of mgras within walking distance of this mgra. - * - * @param mgra - * The mgra to look up - * @return The mgras within walking distance. Null is returned if no mgras - * are within walk distance. - */ - public int[] getMgrasWithinWalkDistanceTo(int mgra) - { - - if (dMgraWalkDistance[mgra] == null) return null; - - Set keySet = dMgraWalkDistance[mgra].keySet(); - int[] walkMgras = new int[keySet.size()]; - Iterator it = keySet.iterator(); - int i = 0; - while (it.hasNext()) - { - walkMgras[i] = it.next(); - ++i; - } - return walkMgras; - - } - - /** - * Return true if mgras are within walking distance of each other. - * - * @param oMgra - * The from mgra - * @param dMgra - * The to mgra - * @return The mgras are within walking distance - true or false. - */ - public boolean getMgrasAreWithinWalkDistance(int oMgra, int dMgra) - { - - if (dMgraWalkDistance[dMgra] == null) return false; - - return dMgraWalkDistance[dMgra].containsKey(oMgra); - - } - - - - /** - * Get the position of the tap in the mgra walk tap array. - * - * @param mgra - * The mgra to lookup - * @param tap - * The tap to lookup - * @return The position of the tap in the mgra array. -1 is returned if it - * is an invalid tap for the mgra, or if the tap is not within - * walking distance. - */ - public int getTapPosition(int mgra, int tap) - { - - if (mgraWlkTapsDistArray[mgra] != null) - { - if (mgraWlkTapsDistArray[mgra][0] != null) - { - for (int i = 0; i < mgraWlkTapsDistArray[mgra][0].length; ++i) - if (mgraWlkTapsDistArray[mgra][0][i] == tap) return i; - } - } - - return -1; - - } - - /** - * Get the walk board time from an MGRA to a TAP. - * - * @param mgra - * The number of the destination MGRA. - * @param pos - * The position of the TAP in the MGRA array (0+) - * @return The walk time in minutes. - */ - public float getMgraToTapWalkBoardTime(int mgra, int pos) - { - float distanceInFeet = (float) mgraWlkTapsDistArray[mgra][1][pos]; - float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; - return time; - } - - - //todo: delete this method: currently retained for compatibility (namely: abm_reports) - /** - * Get the walk time from an MGRA to a TAP. - * - * @param mgra The number of the destination MGRA. - * @param pos The position of the TAP in the MGRA array (0+) - * @return The walk time in minutes. - */ - public float getMgraToTapWalkTime(int mgra, int pos) - { - float distanceInFeet = (float) mgraWlkTapsDistArray[mgra][1][pos]; - float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; - return time; - } - - /** - * Get the walk distance from an MGRA to an MGRA. Return 0 if not within walking - * distance. - * - * @param oMgra - * The number of the production/origin MGRA. - * @param dMgra - * The number of the attraction/destination MGRA. - * @return The walk distance in feet. - */ - public int getMgraToMgraWalkDistFrom(int oMgra, int dMgra) - { - - if (oMgraWalkDistance[oMgra] == null) return 0; - else if (oMgraWalkDistance[oMgra].containsKey(dMgra)) - //return oMgraWalkDistance[oMgra].get(dMgra)[0]; - - return oMgraWalkDistance[oMgra].get(dMgra); - - return 0; - } - - /** - * Get the walk time from an MGRA to a TAP. - * - * @param mgra The MGRA - * @param tap The TAP - * @return The walk time in minutes, else -1 if there is no walk link between the MGRA and the TAP. - */ - public float getWalkTimeFromMgraToTap(int mgra, int tap){ - - int tapPosition = getTapPosition(mgra, tap); - float time = 0; - - if(tapPosition==-1){ - logger.info("Bad Tap Position for Walk Access From MAZ: "+mgra+" to TAP: "+tap); - return -1; - } - else{ - time = (float) (mgraWlkTapsDistArray[mgra][1][tapPosition] * Constants.walkMinutesPerFoot); - } - return time; - } - /** - * Get the walk distance from an MGRA to a TAP. - * - * @param mgra The MGRA - * @param tap The TAP - * @return The walk distance in miles, else -1 if there is no walk link between the MGRA and the TAP. - */ - public float getWalkDistanceFromMgraToTap(int mgra, int tap){ - - int tapPosition = getTapPosition(mgra, tap); - float distance = 0; - - if(tapPosition==-1){ - logger.info("Bad Tap Position for Walk Access From MAZ: "+mgra+" to TAP: "+tap); - return -1; - } - else{ - distance = (float) (mgraWlkTapsDistArray[mgra][1][tapPosition]/Constants.feetPerMile); - } - return distance; - } - /** - * Get the walk distance from an MGRA to an MGRA. Return 0 if not within - * walking distance. - * - * @param oMgra - * The number of the production/origin MGRA. - * @param dMgra - * The number of the attraction/destination MGRA. - * @return The walk distance in feet. - */ - public int getMgraToMgraWalkDistTo(int oMgra, int dMgra) - { - - if (dMgraWalkDistance[dMgra] == null) return 0; - else if (dMgraWalkDistance[dMgra].containsKey(oMgra)) - - return dMgraWalkDistance[dMgra].get(oMgra); - - return 0; - } - - /** - * Get the walk time from an MGRA to an MGRA. Return 0 if not within walking - * distance. - * - * @param oMgra - * The number of the production/origin MGRA. - * @param dMgra - * The number of the attraction/destination MGRA. - * @return The walk time in minutes. - */ - public float getMgraToMgraWalkTime(int oMgra, int dMgra) - { - - if (oMgraWalkDistance[oMgra] == null) return 0f; - else if (oMgraWalkDistance[oMgra].containsKey(dMgra)){ - float distanceInFeet = (float) oMgraWalkDistance[oMgra].get(dMgra); - float time = distanceInFeet/Constants.feetPerMile * Constants.walkMinutesPerMile; - return time; - } - return 0f; - } - - /** - * Get the bike time from an MGRA to an MGRA. Return 0 if not within walking - * distance. - * - * @param oMgra - * The number of the production/origin MGRA. - * @param dMgra - * The number of the attraction/destination MGRA. - * @return The bike time in minutes. - */ - public float getMgraToMgraBikeTime(int oMgra, int dMgra) - { - double time = bls.getTime(defaultSegment,oMgra,dMgra); - return (time == Double.POSITIVE_INFINITY) ? 0f : (float) time; - } - - /** - * Print mgra data to the log file for debugging purposes. - * - */ - public void printMgraStats() - { - logger.info("Number of MGRAs: " + mgras.size()); - logger.info("Max MGRA: " + maxMgra); - - // logger.info("Number of MGRAs with WalkAccessTaps: " + - // nMgrasWithWlkTaps); - // logger.info("Number of TAPs in MGRA 18 (should be 3): " - // + mgraWlkTapsDistArray[18][0].length); - // logger.info("Distance between MGRA 18 and TAP 1648 (should be 2728): " - // + mgraWlkTapsDistArray[18][1][1]); - // logger.info("MGRA 28435 is in what TAZ? (Should be 995)" + - // mgraTaz[28435]); - // logger.info("Number of mgras within walk distance of mgra 22573 (Should be 67)" - // + getMgrasWithinWalkDistanceFrom(22573).length); - - } - - /** - * - * @param mgra - * - the zone - * @return the taz that the tmgra is contained in - */ - public int getTaz(int mgra) - { - return mgraTaz[mgra]; - } - - /** - * - * @param mgra - * - the zone - * @return the luz that the mgra is contained in - */ - public int getMgraLuz(int mgra) - { - return mgraLuz[mgra]; - } - - /** - * Get the maximum LUZ. - * - * @return The highest LUZ number - */ - public int getMaxLuz() - { - return maxLuz; - } - - /** - * Get the maximum MGRA. - * - * @return The highest MGRA number - */ - public int getMaxMgra() - { - return maxMgra; - } - - /** - * Get the maximum TAP. - * - * @return The highest TAP number - */ - public int getMaxTap() - { - return maxTap; - } - - /** - * Get the ArrayList of MGRAs - * - * @return ArrayList mgras. - */ - public ArrayList getMgras() - { - return mgras; - } - - /** - * Get the MgraTaz correspondence array. Given an MGRA, returns its TAZ. - * - * @return int[] mgraTaz correspondence array. - */ - public int[] getMgraTaz() - { - return mgraTaz; - } - - /** - * Get the array of Taps within walk distance - * - * @return The int[][][] array of Taps within walk distance of MGRAs - */ - public int[][][] getMgraWlkTapsDistArray() - { - return mgraWlkTapsDistArray; - } - - /** - * get arrays of drive accessible TAPS for each MGRA and populate an array - * of sets so that later one can determine, for a given mgra, if a tap is - * contained in the set. - * - * @param args - * TazDataManager to get TAPs with drive access from TAZs - */ - public void mapDriveAccessTapsToMgras(TazDataManager tazDataManager) - { - - walkAccessibleTaps = new TreeSet[maxMgra + 1]; - driveAccessibleTaps = new TreeSet[maxMgra + 1]; - - for (int mgra = 1; mgra <= maxMgra; mgra++) - { - - // get the TAZ associated with this MGRA - int taz = getTaz(mgra); - - // store the array of walk accessible TAPS for this MGRA as a set so - // that contains can be called on it later - // to determine, for a given mgra, if a tap is contained in the set. - int[] mgraSet = getMgraWlkTapsDistArray()[mgra][0]; - if (mgraSet != null) - { - walkAccessibleTaps[mgra] = new TreeSet(); - for (int i = 0; i < mgraSet.length; i++) - walkAccessibleTaps[mgra].add(mgraSet[i]); - } - - // store the array of drive accessible TAPS for this MGRA as a set - // so that contains can be called on it later - // to determine, for a given mgra, if a tap is contained in the set. - int[] tapItems = tazDataManager.getParkRideOrKissRideTapsForZone(taz, - AccessMode.PARK_N_RIDE); - driveAccessibleTaps[mgra] = new TreeSet(); - for (int item : tapItems) - driveAccessibleTaps[mgra].add(item); - - } - - } - - /** - * @param mgra - * for which we want to know if TAP can be reached by drive - * access - * @param tap - * for which we want to know if the mgra can reach it by drive - * access - * @return true if reachable; false otherwise - */ - public boolean getTapIsDriveAccessibleFromMgra(int mgra, int tap) - { - if (driveAccessibleTaps[mgra] == null) return false; - else return driveAccessibleTaps[mgra].contains(tap); - } - - /** - * @param mgra - * for which we want to know if TAP can be reached by walk access - * @param tap - * for which we want to know if the mgra can reach it by walk - * access - * @return true if reachable; false otherwise - */ - public boolean getTapIsWalkAccessibleFromMgra(int mgra, int tap) - { - if (walkAccessibleTaps[mgra] == null) return false; - else return walkAccessibleTaps[mgra].contains(tap); - } - - /** - * return the duDen value for the mgra - * - * @param mgra - * is the MGRA value for which the duDen value is needed - * @return duDen[mgra] - */ - public double getDuDenValue(int mgra) - { - return duDen[mgra]; - } - - /** - * return the empDen value for the mgra - * - * @param mgra - * is the MGRA value for which the empDen value is needed - * @return empDen[mgra] - */ - public double getEmpDenValue(int mgra) - { - return empDen[mgra]; - } - - /** - * return the totInt value for the mgra - * - * @param mgra - * is the MGRA value for which the totInt value is needed - * @return totInt[mgra] - */ - public double getTotIntValue(int mgra) - { - return totInt[mgra]; - } - - - public double getPopEmpPerSqMi( int mgra ) { - return popEmpDenPerSqMi[mgra]; - } - - /** - * Process the 4D density land use data file and store the selected fields - * as arrays indexed by the mgra value. The data fields are in the mgra - * TableDataSet read from the MGRA csv file. - * - * @param rbMap - * is a HashMap for the resource bundle generated from the - * properties file. - */ - public void process4ddensityData(HashMap rbMap) - { - - try - { - - // allocate arrays for the land use data fields - duDen = new double[maxMgra + 1]; - empDen = new double[maxMgra + 1]; - totInt = new double[maxMgra + 1]; - - //added for Taxi/TNC - popEmpDenPerSqMi = new double[maxMgra+1]; - - // get the data fields needed for the mode choice utilities as - // 0-based double[] - double[] duDenField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_DU_DEN_FIELD); - double[] empDenField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_EMP_DEN_FIELD); - double[] totIntField = mgraTableDataSet.getColumnAsDouble(MGRA_4DDENSITY_TOT_INT_FIELD); - double[] popEmpField = mgraTableDataSet.getColumnAsDouble( MGRA_POPEMPPERSQMI_FIELD ); - - // create a HashMap to convert MGRA values to array indices for the - // data - // arrays above - int mgraCol = mgraTableDataSet.getColumnPosition(MGRA_FIELD_NAME); - - for (int row = 1; row <= mgraTableDataSet.getRowCount(); row++) - { - - int mgra = (int) mgraTableDataSet.getValueAt(row, mgraCol); - duDen[mgra] = duDenField[row - 1]; - empDen[mgra] = empDenField[row - 1]; - totInt[mgra] = totIntField[row - 1]; - popEmpDenPerSqMi [mgra] = popEmpField[row-1]; - - } - - } catch (Exception e) - { - logger.error( - String.format("Exception occurred processing 4ddensity data file from mgraData TableDataSet object."), - e); - throw new RuntimeException(e); - } - - } - - private void readMgraTableData(HashMap rbMap) - { - - // get the mgra data table from one of these UECs. - String projectPath = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String mgraFile = rbMap.get(PROPERTIES_MGRA_DATA_FILE); - mgraFile = projectPath + mgraFile; - - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - mgraTableDataSet = reader.readFile(new File(mgraFile)); - } catch (IOException e) - { - logger.error("problem reading mgra data table for MgraDataManager.", e); - System.exit(1); - } - - HashMap tazs = new HashMap(); - HashMap luzs = new HashMap(); - - // create a HashMap between mgra values and the corresponding row number - // in the mgra TableDataSet. - mgraDataTableMgraRowMap = new HashMap(); - maxMgra = 0; - maxLuz = 0; - for (int i = 1; i <= mgraTableDataSet.getRowCount(); i++) - { - int mgra = (int) mgraTableDataSet.getValueAt(i, MGRA_FIELD_NAME); - int taz = (int) mgraTableDataSet.getValueAt(i, MGRA_TAZ_FIELD_NAME); - - int luz = (int) mgraTableDataSet.getValueAt(i, MGRA_LUZ_FIELD_NAME); - - mgraDataTableMgraRowMap.put(mgra, i); - - if (mgra > maxMgra) maxMgra = mgra; - mgras.add(mgra); - - tazs.put(mgra, taz); - - if (luz > 0) - { - if (luz > maxLuz) maxLuz = luz; - luzs.put(mgra, luz); - } - } - - mgraTaz = new int[maxMgra + 1]; - for (int mgra : mgras) - mgraTaz[mgra] = tazs.get(mgra); - - mgraLuz = new int[maxMgra + 1]; - for (int mgra : mgras) - mgraLuz[mgra] = luzs.get(mgra); - - } - - /** - * @param mgra - * for which table data is desired - * @return population for the specified mgra. - */ - public double getMgraPopulation(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_POPULATION_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return households for the specified mgra. - */ - public double getMgraHouseholds(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_HOUSEHOLDS_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return grade school enrollment for the specified mgra. - */ - public double getMgraGradeSchoolEnrollment(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_GRADE_SCHOOL_ENROLLMENT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return high school enrollment for the specified mgra. - */ - public double getMgraHighSchoolEnrollment(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_HIGH_SCHOOL_ENROLLMENT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return university enrollment for the specified mgra. - */ - public double getMgraUniversityEnrollment(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_UNIVERSITY_ENROLLMENT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return other college enrollment for the specified mgra. - */ - public double getMgraOtherCollegeEnrollment(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_OTHER_COLLEGE_ENROLLMENT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return adult school enrollment for the specified mgra. - */ - public double getMgraAdultSchoolEnrollment(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_ADULT_SCHOOL_ENROLLMENT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return grade school district for the specified mgra. - */ - public int getMgraGradeSchoolDistrict(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return (int) mgraTableDataSet.getValueAt(row, MGRA_GRADE_SCHOOL_DISTRICT_FIELD_NAME); - } - - /** - * @param mgra - * for which table data is desired - * @return high school district for the specified mgra. - */ - public int getMgraHighSchoolDistrict(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return (int) mgraTableDataSet.getValueAt(row, MGRA_HIGH_SCHOOL_DISTRICT_FIELD_NAME); - } - - public HashMap getMgraDataTableMgraRowMap() - { - return mgraDataTableMgraRowMap; - } - - private void calculateMgraAvgParkingCosts(HashMap propertyMap) - { - - // open output file to write average parking costs for each mgra - PrintWriter out = null; - - String projectPath = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String outFile = propertyMap.get(PROPERTIES_PARKING_COST_OUTPUT_FILE); - outFile = projectPath + outFile; - - try - { - out = new PrintWriter(new BufferedWriter(new FileWriter(new File(outFile)))); - } catch (IOException e) - { - logger.error("Exception caught trying to create file " + outFile); - System.out.println("Exception caught trying to create file " + outFile); - e.printStackTrace(); - throw new RuntimeException(); - } - - // write the header record - out.println("mgra,mgraParkArea,lsWgtAvgCostM,lsWgtAvgCostD,lsWgtAvgCostH"); - - // open output files for writing debug info for a specific mgra - PrintWriter outM = null; - PrintWriter outD = null; - PrintWriter outH = null; - - if (LOG_MGRA > 0) - { - try - { - outM = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath - + "output/" + LOG_MGRA_FILE + "M.csv")))); - outD = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath - + "output/" + LOG_MGRA_FILE + "D.csv")))); - outH = new PrintWriter(new BufferedWriter(new FileWriter(new File(projectPath - + "output/" + LOG_MGRA_FILE + "H.csv")))); - } catch (IOException e) - { - // TODO Auto-generated catch block - e.printStackTrace(); - } - } - - float workDistCoeff = Float.parseFloat(propertyMap.get(MGRA_DISTANCE_COEFF_WORK)); - float otherDistCoeff = Float.parseFloat(propertyMap.get(MGRA_DISTANCE_COEFF_OTHER)); - - int[] mgraField = mgraTableDataSet.getColumnAsInt(MGRA_FIELD_NAME); - int[] mgraParkAreaField = mgraTableDataSet.getColumnAsInt(MGRA_PARKAREA_FIELD); - int[] hstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_HSTALLSOTH_FIELD); - int[] hstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_HSTALLSSAM_FIELD); - float[] hparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_HPARKCOST_FIELD); - int[] numfreehrsField = mgraTableDataSet.getColumnAsInt(MGRA_NUMFREEHRS_FIELD); - int[] dstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_DSTALLSOTH_FIELD); - int[] dstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_DSTALLSSAM_FIELD); - float[] dparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_DPARKCOST_FIELD); - int[] mstallsothField = mgraTableDataSet.getColumnAsInt(MGRA_MSTALLSOTH_FIELD); - int[] mstallssamField = mgraTableDataSet.getColumnAsInt(MGRA_MSTALLSSAM_FIELD); - float[] mparkcostField = mgraTableDataSet.getColumnAsFloat(MGRA_MPARKCOST_FIELD); - - mgraParkArea = new int[maxMgra + 1]; - numfreehrs = new int[maxMgra + 1]; - hstallsoth = new int[maxMgra + 1]; - hstallssam = new int[maxMgra + 1]; - hparkcost = new float[maxMgra + 1]; - dstallsoth = new int[maxMgra + 1]; - dstallssam = new int[maxMgra + 1]; - dparkcost = new float[maxMgra + 1]; - mstallsoth = new int[maxMgra + 1]; - mstallssam = new int[maxMgra + 1]; - mparkcost = new float[maxMgra + 1]; - - lsWgtAvgCostM = new double[maxMgra + 1]; - lsWgtAvgCostD = new double[maxMgra + 1]; - lsWgtAvgCostH = new double[maxMgra + 1]; - - // loop over the number of mgra records in the TableDataSet. - for (int k = 0; k < maxMgra; k++) - { - - // get the mgra value for TableDataSet row k from the mgra field. - int mgra = mgraField[k]; - - mgraParkArea[mgra] = mgraParkAreaField[k]; - numfreehrs[mgra] = numfreehrsField[k]; - hstallsoth[mgra] = hstallsothField[k]; - hstallssam[mgra] = hstallssamField[k]; - hparkcost[mgra] = hparkcostField[k]; - dstallsoth[mgra] = dstallsothField[k]; - dstallssam[mgra] = dstallssamField[k]; - dparkcost[mgra] = dparkcostField[k]; - mstallsoth[mgra] = mstallsothField[k]; - mstallssam[mgra] = mstallssamField[k]; - mparkcost[mgra] = mparkcostField[k]; - - // get the array of mgras within walking distance of m - int[] walkMgras = getMgrasWithinWalkDistanceFrom(mgra); - - // park area 1. - if (mgraParkArea[mgra] == PARK_AREA_ONE) - { - - // calculate weighted average cost from monthly costs - double dist = getMgraToMgraWalkDistFrom(mgra, mgra) / 5280.0; - - double numeratorM = mstallssam[mgra] * Math.exp(workDistCoeff * dist) - * mparkcost[mgra]; - double denominatorM = mstallssam[mgra] * Math.exp(workDistCoeff * dist); - - double numeratorD = dstallssam[mgra] * Math.exp(workDistCoeff * dist) - * dparkcost[mgra]; - double denominatorD = dstallssam[mgra] * Math.exp(workDistCoeff * dist); - - double discountFactor = Math.max(1 - (numfreehrs[mgra] / 4), 0); - double numeratorH = hstallssam[mgra] * Math.exp(workDistCoeff * dist) - * discountFactor * hparkcost[mgra]; - double denominatorH = hstallssam[mgra] * Math.exp(workDistCoeff * dist); - - if (mgra == LOG_MGRA) - { - // log the file header - outM.println("wMgra" + "," + "mgraParkArea" + "," + "workDistCoeff*dist" + "," - + "exp(workDistCoeff*dist)" + "," + "mstallsoth" + "," + "mparkcost" - + "," + "numeratorM" + "," + "denominatorM"); - outD.println("wMgra" + "," + "mgraParkArea" + "," + "otherDistCoeff*dist" + "," - + "exp(otherDistCoeff*dist)" + "," + "dstallsoth" + "," + "dparkcost" - + "," + "numeratorD" + "," + "denominatorD"); - outH.println("wMgra" + "," + "mgraParkArea" + "," + "otherDistCoeff*dist" + "," - + "exp(otherDistCoeff*dist)" + "," + "discountFactor" + "," - + "hstallsoth" + "," + "hparkcost" + "," + "numeratorH" + "," - + "denominatorH"); - - outM.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," - + Math.exp(workDistCoeff * dist) + "," + mstallsoth[mgra] + "," - + mparkcost[mgra] + "," + numeratorM + "," + denominatorM); - outD.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," - + Math.exp(workDistCoeff * dist) + "," + dstallsoth[mgra] + "," - + dparkcost[mgra] + "," + numeratorD + "," + denominatorD); - outH.println(mgra + "," + mgraParkArea[mgra] + "," + workDistCoeff * dist + "," - + Math.exp(workDistCoeff * dist) + "," + discountFactor + "," - + hstallsoth[mgra] + "," + hparkcost[mgra] + "," + numeratorH + "," - + denominatorH); - } - - if (walkMgras != null) - { - - for (int wMgra : walkMgras) - { - - // skip mgra if not in park area 1 or 2. - if (mgraParkArea[wMgra] > 2) - { - if (mgra == LOG_MGRA) - { - outM.println(wMgra + "," + mgraParkArea[wMgra]); - outD.println(wMgra + "," + mgraParkArea[wMgra]); - outH.println(wMgra + "," + mgraParkArea[wMgra]); - } - continue; - } - - if (wMgra != mgra) - { - dist = getMgraToMgraWalkDistFrom(mgra, wMgra) / 5280.0; - - if (dist > MAX_PARKING_WALK_DISTANCE) - { - if (mgra == LOG_MGRA) - { - outM.println(wMgra + "," + mgraParkArea[wMgra]); - outD.println(wMgra + "," + mgraParkArea[wMgra]); - outH.println(wMgra + "," + mgraParkArea[wMgra]); - } - continue; - } - - numeratorM += mstallsoth[wMgra] * Math.exp(workDistCoeff * dist) - * mparkcost[wMgra]; - denominatorM += mstallsoth[wMgra] * Math.exp(workDistCoeff * dist); - - numeratorD += dstallsoth[wMgra] * Math.exp(otherDistCoeff * dist) - * dparkcost[wMgra]; - denominatorD += dstallsoth[wMgra] * Math.exp(otherDistCoeff * dist); - - discountFactor = Math.max(1 - (numfreehrs[wMgra] / 4), 0); - numeratorH += hstallsoth[wMgra] * Math.exp(otherDistCoeff * dist) - * discountFactor * hparkcost[wMgra]; - denominatorH += hstallsoth[wMgra] * Math.exp(otherDistCoeff * dist); - - if (mgra == LOG_MGRA) - { - outM.println(wMgra + "," + mgraParkArea[wMgra] + "," - + workDistCoeff * dist + "," - + Math.exp(workDistCoeff * dist) + "," + mstallsoth[wMgra] - + "," + mparkcost[wMgra] + "," + numeratorM + "," - + denominatorM); - outD.println(wMgra + "," + mgraParkArea[wMgra] + "," - + otherDistCoeff * dist + "," - + Math.exp(otherDistCoeff * dist) + "," + dstallsoth[wMgra] - + "," + dparkcost[wMgra] + "," + numeratorD + "," - + denominatorD); - outH.println(wMgra + "," + mgraParkArea[wMgra] + "," - + otherDistCoeff * dist + "," - + Math.exp(otherDistCoeff * dist) + "," + discountFactor - + "," + hstallsoth[wMgra] + "," + hparkcost[wMgra] + "," - + numeratorH + "," + denominatorH); - } - - } - - } - - } - // jef: storing by mgra since they are indexed into by mgra - // wsu added if clauses. If denominators are 0, read costs directly from input file - if(denominatorM>0) - lsWgtAvgCostM[mgra] = numeratorM / denominatorM; - else - lsWgtAvgCostM[mgra] = mparkcost[mgra]; - if(denominatorD>0) - lsWgtAvgCostD[mgra] = numeratorD / denominatorD; - else - lsWgtAvgCostD[mgra] = dparkcost[mgra]; - if(denominatorH>0) - lsWgtAvgCostH[mgra] = numeratorH / denominatorH; - else - lsWgtAvgCostH[mgra] = hparkcost[mgra]; - } else - { - - lsWgtAvgCostM[mgra] = mparkcost[mgra]; - lsWgtAvgCostD[mgra] = dparkcost[mgra]; - lsWgtAvgCostH[mgra] = hparkcost[mgra]; - - } - - // write the data record - out.println(mgra + "," + mgraParkArea[mgra] + "," + lsWgtAvgCostM[mgra] + "," - + lsWgtAvgCostD[mgra] + "," + lsWgtAvgCostH[mgra]); - } - - if (LOG_MGRA > 0) - { - outM.close(); - outD.close(); - outH.close(); - } - - out.close(); - - } - - public double[] getLsWgtAvgCostM() - { - return lsWgtAvgCostM; - } - - public double[] getLsWgtAvgCostD() - { - return lsWgtAvgCostD; - } - - public double[] getLsWgtAvgCostH() - { - return lsWgtAvgCostH; - } - - public int[] getMgraParkAreas() - { - return mgraParkArea; - } - - public int[] getNumFreeHours() - { - return numfreehrs; - } - - public int[] getMStallsOth() - { - return mstallsoth; - } - - public int[] getMStallsSam() - { - return mstallssam; - } - - public float[] getMParkCost() - { - return mparkcost; - } - - public int[] getDStallsOth() - { - return dstallsoth; - } - - public int[] getDStallsSam() - { - return dstallssam; - } - - public float[] getDParkCost() - { - return dparkcost; - } - - public int[] getHStallsOth() - { - return hstallsoth; - } - - public int[] getHStallsSam() - { - return hstallssam; - } - - public float[] getHParkCost() - { - return hparkcost; - } - - /** - * @param mgra - * for which table data is desired - * @return high school district for the specified mgra. - */ - public int getMgraHourlyParkingCost(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return (int) mgraTableDataSet.getValueAt(row, MGRA_HPARKCOST_FIELD); - } - - public float getRefeulingStations(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_REFUELING_STATIONS_FIELD_NAME); - } - - public float getRemoteParkingLot(int mgra) - { - int row = mgraDataTableMgraRowMap.get(mgra); - return mgraTableDataSet.getValueAt(row, MGRA_REMOTE_PARKING_LOT_FIELD_NAME); - } - private class Maz2Tap implements Comparable, Serializable - { - public int maz; - public int tap; - public double dist; - public String[] lines; - public boolean servesNewLines = false; - - @Override - public int compareTo(Maz2Tap o) { - if ( this.dist < o.dist ) { - return -1; - } else if (this.dist==o.dist) { - return 0; - } else { - return 1; - } - } - } - public TableDataSet getMgraTableDataSet() { - return mgraTableDataSet; - } - - public static void main(String[] args) - { - ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); - MgraDataManager mdm = MgraDataManager.getInstance(ResourceUtil - .changeResourceBundleIntoHashMap(rb)); - mdm.printMgraStats(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java deleted file mode 100644 index 2204762..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/Modes.java +++ /dev/null @@ -1,115 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; - -/** - * This class is used for specifying modes. - * - * @author Christi Willison - * @version Sep 8, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public final class Modes - implements Serializable -{ - - public enum AutoMode - { - DRIVE_ALONE_TOLL("dat"), DRIVE_ALONE_NONTOLL("dan"), TWO_NONTOLL("2nt"), TWO_TOLL( - "2t"), THREEPLUS_NONTOLL("3+nt"), THREEPLUS_TOLL( - "3+t"); - - private final String name; - - AutoMode(String s) - { - this.name = s; - } - - public AutoMode[] getAutoModes() - { - return AutoMode.values(); - } - - public String toString() - { - return name; - } - } - - public enum TransitMode - { - COMMUTER_RAIL("cr", true), // label and true = premium - LIGHT_RAIL("lr", true), BRT("brt", true), EXPRESS_BUS("eb", true), LOCAL_BUS("lb", false); - - private final String name; - private final boolean premium; - - TransitMode(String name, boolean premium) - { - this.name = name; - this.premium = premium; - } - - public TransitMode[] getTransitModes() - { - return TransitMode.values(); - } - - public boolean isPremiumMode(TransitMode transitMode) - { - return transitMode.premium; - } - - public String toString() - { - return name; - } - - } - - public enum AccessMode - { - WALK("WLK"), PARK_N_RIDE("PNR"), KISS_N_RIDE("KNR"); - private final String name; - - AccessMode(String name) - { - this.name = name; - } - - public AccessMode[] getAccessModes() - { - return AccessMode.values(); - } - - public String toString() - { - return name; - } - - } - - public enum NonMotorizedMode - { - WALK, BIKE - } - - public enum OtherMode - { - SCHOOL_BUS - } - - private Modes() - { - // Not implemented in utility classes - } - - - public static void main(String[] args) - { - System.out.println(AutoMode.DRIVE_ALONE_NONTOLL.toString()); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java deleted file mode 100644 index 01a7a4b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorDMU.java +++ /dev/null @@ -1,146 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -/** - * This class is used for non-motorized DMU attributes. - * - * @author Joel Freedman - * @version May 28,2009 - *

    - */ -public class NonMotorDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(NonMotorDMU.class); - - protected HashMap methodIndexMap; - - private float mgraWalkTime; - private float mgraBikeTime; - - public NonMotorDMU() - { - setupMethodIndexMap(); - } - - /** - * Get MGRA-MGRA walk time. - * - * @return Mgra-mgra walk time in minutes. - */ - public float getMgraWalkTime() - { - return mgraWalkTime; - } - - /** - * Set Mgra-Mgra walk time in minutes. - * - * @param mgraWalkTime - * Mgra walk time in minutes. - */ - public void setMgraWalkTime(float mgraWalkTime) - { - this.mgraWalkTime = mgraWalkTime; - } - - /** - * Get MGRA-MGRA bike time. - * - * @return Mgra-mgra bike time in minutes. - */ - public float getMgraBikeTime() - { - return mgraBikeTime; - } - - /** - * Set Mgra-Mgra bike time in minutes. - * - * @param mgraBikeTime - * Mgra bike time in minutes. - */ - public void setMgraBikeTime(float mgraBikeTime) - { - this.mgraBikeTime = mgraBikeTime; - } - - /** - * Log the DMU values. - * - * @param localLogger - * The logger to use. - */ - public void logValues(Logger localLogger) - { - - localLogger.info(""); - localLogger.info("Non-Motorized DMU Values:"); - localLogger.info(""); - localLogger.info(String.format("MGRA-MGRA Walk Time: %9.4f", mgraWalkTime)); - localLogger.info(String.format("MGRA-MGRA Bike Time: %9.4f", mgraBikeTime)); - - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getMgraBikeTime", 0); - methodIndexMap.put("getMgraWalkTime", 1); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = 0; - - switch (variableIndex) - { - case 0: - returnValue = getMgraBikeTime(); - break; - case 1: - returnValue = getMgraWalkTime(); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java deleted file mode 100644 index 4e80181..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/NonMotorUEC.java +++ /dev/null @@ -1,189 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.Arrays; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.calculator.IndexValues; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.LogitModel; -import com.pb.common.newmodel.UtilityExpressionCalculator; -import com.pb.common.util.Tracer; - -/** - * This class is used for ... - * - * @author Christi Willison - * @version Mar 9, 2009 - *

    - * Created by IntelliJ IDEA. - */ -public class NonMotorUEC - implements Serializable -{ - - private transient Logger logger = Logger.getLogger(NonMotorUEC.class); - private TazDataManager tazManager; - private MgraDataManager mgraManager; - private UtilityExpressionCalculator uec; - private IndexValues index = new IndexValues(); - private int[] availFlag; - private NonMotorDMU dmu; - private LogitModel model; - private ChoiceModelApplication modelApp; - - // seek and trace - private boolean trace; - private int[] traceOtaz; - private int[] traceDtaz; - protected Tracer tracer; - - /** - * Default Constructor. - * - * @param rb - * @param uecFileName - * @param modelSheet - * @param dataSheet - */ - public NonMotorUEC(HashMap rbHashMap, String uecFileName, int modelSheet, - int dataSheet) - { - - dmu = new NonMotorDMU(); - - // use the choice model application to set up the model structure - modelApp = new ChoiceModelApplication(uecFileName, modelSheet, dataSheet, rbHashMap, dmu); - - // but return the logit model itself, so we can use compound utilities - model = modelApp.getRootLogitModel(); - uec = modelApp.getUEC(); - availFlag = new int[uec.getNumberOfAlternatives() + 1]; - - tazManager = TazDataManager.getInstance(); - mgraManager = MgraDataManager.getInstance(); - - trace = Util.getBooleanValueFromPropertyMap(rbHashMap, "Trace"); - traceOtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.otaz"); - traceDtaz = Util.getIntegerArrayFromPropertyMap(rbHashMap, "Trace.dtaz"); - - // set up the tracer object - tracer = Tracer.getTracer(); - tracer.setTrace(trace); - for (int i = 0; i < traceOtaz.length; i++) - { - for (int j = 0; j < traceDtaz.length; j++) - { - tracer.traceZonePair(traceOtaz[i], traceDtaz[j]); - logger.info("Setting trace zone pair in NonMotorUEC Object for i: "+ traceOtaz[i] + " j: " + traceDtaz[j]); - } - } - } - - /** - * Calculate utilities for a given TAZ pair. - * - * @param pTaz - * Production/Origin TAZ. - * @param aTaz - * Attraction/Destination TAZ. - * @return The root utility. - */ - public double calculateUtilitiesForTazPair(int pTaz, int aTaz) - { - - index.setOriginZone(pTaz); - index.setDestZone(aTaz); - - Arrays.fill(availFlag, 1); - dmu.setMgraWalkTime(0); - dmu.setMgraBikeTime(0); - - trace = false; - if (tracer.isTraceOn() && tracer.isTraceZonePair(pTaz, aTaz)) - { - trace = true; - } - - // log DMU values - if (trace) - { - TapDataManager tapManager = TapDataManager.getInstance(); - if (Arrays.binarySearch(tapManager.getTaps(), pTaz) > 0 - && Arrays.binarySearch(tapManager.getTaps(), aTaz) > 0) - uec.logDataValues(logger, pTaz, aTaz, 0); - dmu.logValues(logger); - } - - modelApp.computeUtilities(dmu, index); - double utility = modelApp.getLogsum(); - if (utility == 0) utility = -999; - - // logging - if (trace) - { - uec.logAnswersArray(logger, "NonMotorized UEC"); - uec.logResultsArray(logger, pTaz, aTaz); - modelApp.logLogitCalculations("NonMotorized UEC", "Zone Trace"); - logger.info("Logsum = " + utility); - trace = false; - } - - return utility; - } - - /** - * Calculate utilities for a given TAZ pair. - * - * @param oMgra - * Production/Origin Mgra. - * @param dMgra - * Attraction/Destination Mgra. - * @return The root utility. - */ - public double calculateUtilitiesForMgraPair(int oMgra, int dMgra) - { - - Arrays.fill(availFlag, 1); - - trace = false; - int pTaz = mgraManager.getTaz(oMgra); - int aTaz = mgraManager.getTaz(dMgra); - index.setOriginZone(pTaz); - index.setDestZone(aTaz); - - if (tracer.isTraceOn() && tracer.isTraceZone(pTaz)) - { - trace = true; - } - - dmu.setMgraWalkTime(mgraManager.getMgraToMgraWalkTime(oMgra, dMgra)); - dmu.setMgraBikeTime(mgraManager.getMgraToMgraBikeTime(oMgra, dMgra)); - - // log DMU values - if (trace) - { - logger.info("MGRA-MGRA non-motorized calculations for " + oMgra + " to " + dMgra); - dmu.logValues(logger); - } - - modelApp.computeUtilities(dmu, index); - double utility = modelApp.getLogsum(); - if (utility == 0) utility = -999; - - // logging - if (trace) - { - uec.logAnswersArray(logger, "NonMotorized UEC"); - uec.logResultsArray(logger, pTaz, aTaz); - modelApp.logLogitCalculations("NonMotorized UEC", "Mgra Trace"); - logger.info("Logsum = " + utility); - trace = false; - } - dmu.setMgraWalkTime(0); - dmu.setMgraWalkTime(0); - return utility; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java deleted file mode 100644 index 5f42ed1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TapDataManager.java +++ /dev/null @@ -1,321 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.BufferedReader; -import java.io.File; -import java.io.FileReader; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Collections; -import java.util.HashMap; -import java.util.Iterator; -import java.util.List; -import java.util.Map; -import java.util.ResourceBundle; -import java.util.StringTokenizer; -import java.util.TreeMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.active.sandag.SandagWalkPathAlternativeListGenerationConfiguration; -import org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.util.ResourceUtil; - -public final class TapDataManager - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(TapDataManager.class); - private static volatile TapDataManager instance = null; - private static final Object LOCK = new Object(); - - // tapID, [tapNum, lotId, ??, taz][list of values] - private float[][][] tapParkingInfo; - - // an array that stores parking lot use by lot ID. - private int[] lotUse; - - // an array of taps - private int[] taps; - private int maxTap; - - public int getMaxTap() - { - return maxTap; - } - - private TapDataManager(HashMap rbMap) - { - - System.out.println("I'm the TapDataManager"); - readTap(rbMap); - getTapList(rbMap); - intializeLotUse(); - printStats(); - } - - /** - * This method should only be used after the getInstance(HashMap rbMap) method has been called since the rbMap is needed to read - * in all the data and populate the object. This method will return the - * instance that has already been populated. - * - * @return instance - * @throws RuntimeException - */ - public static TapDataManager getInstance() - { - if (instance == null) - { - throw new RuntimeException( - "Must instantiate TapDataManager with the getInstance(rbMap) method first"); - } else - { - return instance; - } - } - - /** - * This method will read in the tapParkingInfo.ptype file and store the info - * in a TreeMap where key equals the iTap and value equals an array of [][3] - * elements. The TreeMap will be passed to the populateTap function which - * will transpose the array of [][3] elements to an array of [4][] elements - * and attaches it to the this.tapParkingInfo[key] - * - * //TODO: Test this and see if there is only a single lot associated // - * TODO with each tap. - * - * The file has 6 columns - tap, lotId, parking type, taz, capacity and mode - * - * @param rb - * - the resource bundle that lists the tap.ptype file and - * scenario.path. - */ - private void readTap(HashMap rbMap) - { - - File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") - + Util.getStringValueFromPropertyMap(rbMap, "tap.ptype.file")); - String s; - TreeMap> map = new TreeMap>(); - StringTokenizer st; - try - { - BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile)); - while ((s = br.readLine()) != null) - { - st = new StringTokenizer(s, " "); - float[] tapList = new float[6]; - int key = Integer.parseInt(st.nextToken()); // tap number - tapList[0] = Float.parseFloat(st.nextToken()); // lot id - tapList[3] = Float.parseFloat(st.nextToken()); // ptype - tapList[1] = Float.parseFloat(st.nextToken()); // taz - tapList[2] = (Math.max(Float.parseFloat(st.nextToken()), 15)) * 2.5f; // lot capacity - tapList[4] = Float.parseFloat(st.nextToken()); // distance from lot to TAP - tapList[5] = Float.parseFloat(st.nextToken()); /* Transit mode {4: CR, - 5: LRT, - 6: BRT, - 7: BRT, - 8:Limited Express Bus, - 9:Express bus, - 10: local}*/ - - if (map.get(key) == null) - { - List newList = new ArrayList(); - newList.add(tapList); - map.put(key, newList); - } else - { - map.get(key).add(tapList); - } - } - br.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - populateTap(map); - } - - /** - * The function will get a TreeMap having with iTaps as keys and [][4] - * arrays. For each iTap in the TreeMap it will transpose the [][4] array - * associated with it and attach it to the this.tapParkingInfo[key] element. - * - * @param map - * - a TreeMap containing all the records of the - * tapParkingInfo.ptype file - */ - private void populateTap(TreeMap> map) - { - - this.tapParkingInfo = new float[map.lastKey() + 1][6][]; - Iterator iterKeys = map.keySet().iterator(); - while (iterKeys.hasNext()) - { - int key = iterKeys.next(); - int numElem = map.get(key).size(); - for (int i = 0; i < 6; i++) - this.tapParkingInfo[key][i] = new float[numElem]; - for (int i = 0; i < numElem; i++) - { - for (int j = 0; j < 6; j++) - { - this.tapParkingInfo[key][j][i] = map.get(key).get(i)[j]; - } - } - } - } - - // TODO: test this. - public void intializeLotUse() - { - - float maxLotId = 0; - for (int i = 0; i < tapParkingInfo.length; i++) - { - float[] lotIds = tapParkingInfo[i][0]; - if (lotIds != null) - { - for (int j = 0; j < tapParkingInfo[i][0].length; j++) - { - if (maxLotId < tapParkingInfo[i][0][j]) maxLotId = tapParkingInfo[i][0][j]; - - } - } - } - - lotUse = new int[(int) maxLotId + 1]; - } - - /** - * Set the array of tap numbers (taps[]), indexed at 1. - * - * @param rb - * A Resourcebundle with skims.path and tap.skim.file properties. - */ - public void getTapList(HashMap rbMap) - { - ArrayList tapList = new ArrayList(); - - File mgraWlkTapTimeFile = new File(rbMap.get(SandagWalkPathAlternativeListGenerationConfiguration.PROPERTIES_OUTPUT), - rbMap.get(SandagWalkPathChoiceLogsumMatrixApplication.WALK_LOGSUM_SKIM_MGRA_TAP_FILE_PROPERTY)); - Map> mgraWlkTapList = new HashMap<>(); //mgra -> tap -> [board dist,alight dist] - String s; - try ( BufferedReader br = new BufferedReader(new FileReader(mgraWlkTapTimeFile))) - { - // read the first data file line containing column names - s = br.readLine(); - - // read the data records - while ((s = br.readLine()) != null) - { - StringTokenizer st = new StringTokenizer(s, ","); - int mgra = Integer.parseInt(st.nextToken().trim()); - int tap = Integer.parseInt(st.nextToken().trim()); - if (tap > maxTap) maxTap = tap; - if (!tapList.contains(tap)) tapList.add(tap); - } - } catch (IOException e) { - logger.error(e); - throw new RuntimeException(e); - } - - // read taps from park-and-ride file - File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") - + Util.getStringValueFromPropertyMap(rbMap, "tap.ptype.file")); - - try (BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile))) - { - - while ((s = br.readLine()) != null) - { - StringTokenizer st = new StringTokenizer(s, " "); - int tap = Integer.parseInt(st.nextToken()); // tap number - if (!tapList.contains(tap)) tapList.add(tap); - } - br.close(); - } catch (IOException e) { - logger.error(e); - throw new RuntimeException(e); - } - - Collections.sort(tapList); - // now go thru the array of ArrayLists and convert the lists to arrays - // and - taps = new int[tapList.size() + 1]; - - for (int i = 0; i < tapList.size(); ++i) - taps[i + 1] = tapList.get(i); - - } - - public int getLotUse(int lotId) - { - return lotUse[lotId]; - } - - public void printStats() - { - /* - * logger.info("Tap 561 is in zone: " + tapParkingInfo[561][1][0]); - * logger.info("Tap 298 lot capacity: " + tapParkingInfo[298][2][0]); - */ - } - - public int getTazForTap(int tap) - { - return (int) tapParkingInfo[tap][1][0]; - } - - public static TapDataManager getInstance(HashMap rbMap) - { - if (instance == null) { - synchronized (LOCK) { - if (instance == null) { - instance = new TapDataManager(rbMap); - } - } - } - return instance; - } - - public float[][][] getTapParkingInfo() - { - if (instance != null) - { - return this.tapParkingInfo; - } else - { - throw new RuntimeException(); - } - } - - public float getCarToStationWalkTime(int tap) - { - return 0.0f; - } - - public float getEscalatorTime(int tap) - { - return 0.0f; - } - - public int[] getTaps() - { - return taps; - } - - public static void main(String[] args) - { - ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); - - TapDataManager tdm = TapDataManager.getInstance(ResourceUtil - .changeResourceBundleIntoHashMap(rb)); - tdm.printStats(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java deleted file mode 100644 index ab697d1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TazDataManager.java +++ /dev/null @@ -1,924 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.BufferedReader; -import java.io.File; -import java.io.FileReader; -import java.io.IOException; -import java.io.Serializable; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; -import java.util.ResourceBundle; -import java.util.StringTokenizer; -import java.util.TreeSet; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.Modes.AccessMode; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -/** - * This class is used for storing the TAZ data for the mode choice model. - * - * @author Christi Willison - * @version Sep 2, 2008 - *

    - * Created by IntelliJ IDEA. - */ -public final class TazDataManager - implements Serializable -{ - - protected transient Logger logger = Logger.getLogger(TazDataManager.class); - private static TazDataManager instance; - public int[] tazs; - protected int[] tazsOneBased; - - // arrays for the MGRA and TAZ fields from the MGRA table data file. - private int[] mgraTableMgras; - private int[] mgraTableTazs; - - // list of TAZs in numerical order - public TreeSet tazSet = new TreeSet(); - public int maxTaz; - - private int nTazsWithMgras; - private int[][] tazMgraArray; - // private int[][] tazXYCoordArray; - private float[] tazDestinationTerminalTime; - private float[] tazOriginTerminalTime; - // private int[] tazSuperDistrict; - // private int[] pmsa; - // private int[] cmsa; - // This might be a poor name for this array. - // Please change if you know better. - private int[] tazAreaType; - // they are being read from same file but might be different - - // [tdz][id,time,dist][tap position] - // in the future. - private float[][][] tazParkNRideTaps; - // They are floats because they store time and distance - private float[][][] tazKissNRideTaps; - - // added from tdz manager - private int[] tazParkingType; - - /** - * Get an array of tazs, indexed sequentially from 0 - * - * @return Taz array indexed from 0 - */ - public int[] getTazs() - { - return tazs; - } - - /** - * Get an array of tazs, indexed sequentially from 1 - * - * @return taz array indexed from 1 - */ - public int[] getTazsOneBased() - { - return tazsOneBased; - } - - private TazDataManager(HashMap rbMap) - { - System.out.println("I'm the TazDataManager"); - - // read the MGRA data file into a TableDataSet and get the MGRA and TAZ - // fields from it for setting TAZ correspondence. - readMgraTableData(rbMap); - setTazMgraCorrespondence(); - readTazTerminalTimeCorrespondence(rbMap); - readPnRTapsInfo(rbMap); - - //printTazStats(); - } - - /** - * This method reads in the taz.tdz file which has 2 columns. The first - * column is the taz and the second column is corresponding tdz. The - * correspondence will be stored in the tdz class. The only data captured - * here is the list of TAZs. - * - * This method will also set the maxTaz value. - * - * @param rb - * the properties file that lists the taz.tdz file and the - * generic.path. - */ - private void readTazs(HashMap rbMap) - { - File tazTdzCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") - + Util.getStringValueFromPropertyMap(rbMap, "taz.tdz.correspondence.file")); - String s; - int taz; - StringTokenizer st; - try - { - BufferedReader br = new BufferedReader(new FileReader(tazTdzCorresFile)); - while ((s = br.readLine()) != null) - { - st = new StringTokenizer(s, " "); - taz = Integer.parseInt(st.nextToken()); - tazSet.add(taz); - st.nextToken(); - } - br.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - maxTaz = tazSet.last(); - tazs = new int[tazSet.size()]; - tazsOneBased = new int[tazSet.size() + 1]; - int i = 0; - for (Integer tazNumber : tazSet) - { - tazs[i] = tazNumber; - tazsOneBased[i + 1] = tazNumber; - ++i; - } - - } - - /** - * This method will set the TAZ/MGRA correspondence. Two columns from the - * MGRA data table are used. The first column is the MGRA and the second - * column is the TAZ. The goal of this method is to populate the - * tazMgraArray array and the tazs treeset, plus set maxTaz. - * - */ - private void setTazMgraCorrespondence() - { - - HashMap> tazMgraMap = new HashMap>(); - - int mgra; - int taz; - - for (int i = 0; i < mgraTableMgras.length; i++) - { - - mgra = mgraTableMgras[i]; - taz = mgraTableTazs[i]; - if (!tazSet.contains(taz)) tazSet.add(taz); - - maxTaz = Math.max(taz, maxTaz); - - if (!tazMgraMap.containsKey(taz)) - { - ArrayList tazMgraList = new ArrayList(); - tazMgraList.add(mgra); - tazMgraMap.put(taz, tazMgraList); - } else - { - ArrayList tazMgraList = tazMgraMap.get(taz); - tazMgraList.add(mgra); - } - - } - - // now go thru the array of ArrayLists and convert the lists to arrays - // and - // store in the class variable tazMgraArrays. - tazMgraArray = new int[maxTaz + 1][]; - for (Iterator it = tazMgraMap.entrySet().iterator(); it.hasNext();) - { // elements - // in - // the - // array - // of - // arraylists - Map.Entry entry = (Map.Entry) it.next(); - taz = (Integer) entry.getKey(); - ArrayList tazMgraList = (ArrayList) entry.getValue(); - if (tazMgraList != null) - { // if the list isn't null - tazMgraArray[taz] = new int[tazMgraList.size()]; // initialize - // the class - // variable - for (int j = 0; j < tazMgraList.size(); j++) - tazMgraArray[taz][j] = (Integer) tazMgraList.get(j); - nTazsWithMgras++; - } - } - tazs = new int[tazSet.size()]; - - tazsOneBased = new int[tazSet.size() + 1]; - int i = 0; - for (Integer tazNumber : tazSet) - { - tazs[i] = tazNumber; - tazsOneBased[i + 1] = tazNumber; - ++i; - } - } - - /** - * This method will initialize the class variable tazSuperDistrict. The - * taz.district file has 2 columns, the first is the taz and the second is - * the superdistrict - * - * @param rb - * the resource bundle that specifies the taz.district file and - * the generic.path public void - * readTazDistrictCorrespondence(HashMap rbMap) { - * tazSuperDistrict = new int[maxTaz + 1]; File tazTdzCorresFile - * = new File(Util.getStringValueFromPropertyMap(rbMap, - * "generic.path") + Util.getStringValueFromPropertyMap(rbMap, - * "taz.district.correspondence.file")); String s; int taz; int - * sd; StringTokenizer st; try { BufferedReader br = new - * BufferedReader(new FileReader(tazTdzCorresFile)); while ((s = - * br.readLine()) != null) { st = new StringTokenizer(s, " "); - * taz = Integer.parseInt(st.nextToken()); sd = - * Integer.parseInt(st.nextToken()); tazSuperDistrict[taz] = sd; - * } br.close(); } catch (IOException e) { e.printStackTrace(); } - * } - */ - - /** - * This method will read the zone.avrzone file and store the location area - * (0-3) for each taz. - * - * - * @param rb - * - resourceBundle That specifies the zone.avrzone file and the - * generic.path private void - * readZoneAvrZoneCorrespondence(HashMap rbMap) { - * File zoneAvrZoneCorresFile = new - * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") - * + Util.getStringValueFromPropertyMap(rbMap, - * "taz.avrzone.correspondence.file")); tazAreaType = new - * int[maxTaz + 1]; - * - * // read the file to get the location area (0 - 3) for each TDZ - * String s; int taz; StringTokenizer st; int location; try { - * BufferedReader br = new BufferedReader(new - * FileReader(zoneAvrZoneCorresFile)); while ((s = br.readLine()) - * != null) { st = new StringTokenizer(s, " "); taz = - * Integer.parseInt(st.nextToken()); location = - * Integer.parseInt(st.nextToken()); tazAreaType[taz] = location; - * } br.close(); } catch (IOException e) { e.printStackTrace(); } - * - * } - */ - - /** - * This method reads in the zone.pmsa file which has 2 columns. The first - * column is the taz and the second column is corresponding pmsa. The - * correspondence will be stored in the pmsa list. The only data captured - * here is the list of pmsas. - * - * This method will also set the maxTaz value. - * - * @param rb - * the properties file that lists the taz.tdz file and the - * generic.path private void readZonePMSA(HashMap - * rbMap) { - * - * pmsa = new int[maxTaz + 1]; File zonePmsaFileName = new - * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") - * + Util.getStringValueFromPropertyMap(rbMap, "taz.pmsa.file")); - * String s; int taz; int tazPmsa; StringTokenizer st; try { - * BufferedReader br = new BufferedReader(new - * FileReader(zonePmsaFileName)); // BufferedReader br = new - * BufferedReader(new // - * FileReader("/Users/michalis/Documents/Fortran2Java/data/zone.pmsa" - * )); while ((s = br.readLine()) != null) { st = new - * StringTokenizer(s, " "); taz = - * Integer.parseInt(st.nextToken()); tazPmsa = - * Integer.parseInt(st.nextToken()); pmsa[taz] = tazPmsa; } - * br.close(); } catch (IOException e) { e.printStackTrace(); } - * - * } - */ - - /** - * This method reads in the zone.cmsa file which has 2 columns. The first - * column is the taz and the second column is corresponding cmsa. The - * correspondence will be stored in the cmsa list. The only data captured - * here is the list of cmsas. - * - * This method will also set the maxTaz value. - * - * @param rb - * the properties file that lists the zone.cmsa file and the - * generic.path - * - * private void readZoneCMSA(HashMap rbMap) { - * - * cmsa = new int[maxTaz + 1]; File zoneCmsaFile = new - * File(Util.getStringValueFromPropertyMap(rbMap, "generic.path") - * + Util.getStringValueFromPropertyMap(rbMap, "taz.cmsa.file")); - * String s; int taz; int tazCmsa; StringTokenizer st; try { - * BufferedReader br = new BufferedReader(new - * FileReader(zoneCmsaFile)); // BufferedReader br = new - * BufferedReader(new // - * FileReader("/Users/michalis/Documents/Fortran2Java/data/zone.cmsa" - * )); while ((s = br.readLine()) != null) { st = new - * StringTokenizer(s, " "); taz = - * Integer.parseInt(st.nextToken()); tazCmsa = - * Integer.parseInt(st.nextToken()); cmsa[taz] = tazCmsa; } - * br.close(); } catch (IOException e) { e.printStackTrace(); } - * - * } - * */ - - /** - * This method will read the zone.term file and store the terminal time for - * each taz. - * - * @param rb - * the properties file that lists the zone.term file and the - * scenario.path - */ - private void readTazTerminalTimeCorrespondence(HashMap rbMap) - { - File tdzTerminalTimeCorresFile = new File(Util.getStringValueFromPropertyMap(rbMap, - "scenario.path") - + Util.getStringValueFromPropertyMap(rbMap, "taz.terminal.time.file")); - - tazDestinationTerminalTime = new float[maxTaz + 1]; - tazOriginTerminalTime = new float[maxTaz + 1]; - - // read the file to get the terminal time for each TDZ - String s; - int taz; - StringTokenizer st; - float terminalTime; - try - { - BufferedReader br = new BufferedReader(new FileReader(tdzTerminalTimeCorresFile)); - while ((s = br.readLine()) != null) - { - st = new StringTokenizer(s, " "); - taz = Integer.parseInt(st.nextToken()); - terminalTime = Float.parseFloat(st.nextToken()); - tazDestinationTerminalTime[taz] = terminalTime; - tazOriginTerminalTime[taz] = terminalTime; - } - br.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - - } - - /** - * This method will read the zone.pterm file and store the production - * terminal time for each tdz. - * - * @param rb - * the properties file that lists the zone.pterm file and the - * scenario.path - * - * - * private void - * readTazProductionTerminalTimeCorrespondence(HashMap rbMap) { File tdzProductionTerminalTimeCorresFile = - * new File(Util.getStringValueFromPropertyMap( rbMap, - * "scenario.path") + Util.getStringValueFromPropertyMap(rbMap, - * "taz.prod.terminal.time.file")); - * - * tazOriginTerminalTime = new float[maxTaz + 1]; - * - * // read the file to get the production terminal time for each - * TDZ String s; int taz; StringTokenizer st; float - * productionTerminalTime; try { BufferedReader br = new - * BufferedReader(new FileReader( - * tdzProductionTerminalTimeCorresFile)); while ((s = - * br.readLine()) != null) { st = new StringTokenizer(s, " "); - * taz = Integer.parseInt(st.nextToken()); productionTerminalTime - * = Float.parseFloat(st.nextToken()); tazOriginTerminalTime[taz] - * = productionTerminalTime; } br.close(); } catch (IOException - * e) { e.printStackTrace(); } - * - * } - */ - - /** - * This method will read the zone.park file and store the parking type for - * each taz. Only types 2 - 5 are given. Rest assumed to be type 1. - * - * @param rb - * the properties file that lists the taz.parkingtype.file and - * the scenario.path private void - * readTAZParkingTypeCorrespondence(HashMap - * rbMap) { File tazParkingTypeCorresFile = new - * File(Util.getStringValueFromPropertyMap(rbMap, - * "scenario.path") + Util.getStringValueFromPropertyMap(rbMap, - * "taz.parkingtype.file")); - * - * tazParkingType = new int[maxTaz + 1]; - * Arrays.fill(tazParkingType, 1); - * - * // read the file to get the parking type (2 - 5) for each TAZ - * String s; int taz; StringTokenizer st; int parkingType; try { - * BufferedReader br = new BufferedReader(new - * FileReader(tazParkingTypeCorresFile)); while ((s = - * br.readLine()) != null) { st = new StringTokenizer(s, " "); - * taz = Integer.parseInt(st.nextToken()); parkingType = - * Integer.parseInt(st.nextToken()); tazParkingType[taz] = - * parkingType; } br.close(); } catch (IOException e) { - * e.printStackTrace(); } - * - * } - */ - - /** - * This method read in the access061.prp file that lists the taz and the # - * of taps that have drive access. Then the taps are listed along with the - * time and the distance to those taps from the taz. - * - * @param rb - * the properties file that lists the taz.driveaccess.taps.file - * and the scenario.path - */ - public void readPnRTapsInfo(HashMap rbMap) - { - File tdzDATapFile = new File(Util.getStringValueFromPropertyMap(rbMap, "scenario.path") - + Util.getStringValueFromPropertyMap(rbMap, "taz.driveaccess.taps.file")); - tazParkNRideTaps = new float[maxTaz + 1][3][]; // tapId, time, distance - tazKissNRideTaps = new float[maxTaz + 1][3][]; // tapId, time, distance - - String s, s1; - StringTokenizer st, st1; - int taz; - int tapId; - float tapTime; - float tapDist; - - //Shove into hash at first, then decompose into float array - HashMap< Integer, HashMap> tazTapMap = new HashMap>(); - - try - { - BufferedReader br = new BufferedReader(new FileReader(tdzDATapFile)); - while ((s = br.readLine()) != null) - { - st = new StringTokenizer(s, ","); - - taz = Integer.parseInt(st.nextToken()); - tapId = Integer.parseInt(st.nextToken()); - tapTime = Float.parseFloat(st.nextToken()); - tapDist = Float.parseFloat(st.nextToken()); - - if(tazTapMap.get(taz) != null){ - - HashMap< Integer, float[] > tapVals = tazTapMap.get(taz); - if(tapVals.get(tapId) != null){ - //something wrong, since there should only be unique taps for a taz - throw new RuntimeException("There should not be any duplicate TAPs for a TAZ"); - }else{ - float[] timeDist = new float[2]; - timeDist[0] = tapTime; - timeDist[1] = tapDist; - tapVals.put(tapId, timeDist); - } - - }else{ - HashMap< Integer, float[] > tapVals = new HashMap(); - float[] timeDist = new float[2]; - timeDist[0] = tapTime; - timeDist[1] = tapDist; - tapVals.put(tapId, timeDist); - tazTapMap.put(taz, tapVals); - } - } - - Iterator it = tazTapMap.entrySet().iterator(); - while (it.hasNext()) { - Map.Entry pair = (Map.Entry)it.next(); - taz = (int) pair.getKey(); - HashMap< Integer, float[] > tapVals = tazTapMap.get(taz); - int nTaps = tapVals.keySet().size(); - - tazParkNRideTaps[taz][0] = new float[nTaps]; - tazParkNRideTaps[taz][1] = new float[nTaps]; - tazParkNRideTaps[taz][2] = new float[nTaps]; - tazKissNRideTaps[taz][0] = new float[nTaps]; - tazKissNRideTaps[taz][1] = new float[nTaps]; - tazKissNRideTaps[taz][2] = new float[nTaps]; - - Iterator it2 = tapVals.entrySet().iterator(); - int i = 0; - while (it2.hasNext()) { - Map.Entry pair2 = (Map.Entry)it2.next(); - tapId = (int) pair2.getKey(); - float[] vals = (float[]) pair2.getValue(); - tazParkNRideTaps[taz][0][i] = tapId; - tazParkNRideTaps[taz][1][i] = vals[0]; - tazParkNRideTaps[taz][2][i] = vals[1]; - tazKissNRideTaps[taz][0][i] = tapId; - tazKissNRideTaps[taz][1][i] = vals[0]; - tazKissNRideTaps[taz][2][i] = vals[1]; - i++; - } - } - - br.close(); - } catch (IOException e) - { - e.printStackTrace(); - } - - } - - /** - * This method will return the Area Type (Location?) for the TAZ. - * - * - * @param taz - * - TAZ that AreaType is wanted for. - * @return area type that the taz corresponds to - */ - public int getTAZAreaType(int taz) - { - return tazAreaType[taz]; - } - - /** - * Write taz data manager data to logger for debugging. - * - */ - public void printTazStats() - { - - logger.info("Number of TAZs: " + tazSet.size()); - logger.info("Max TAZ: " + maxTaz); - logger.info("Number of TAZs with MGRAs: " + nTazsWithMgras); - } - - /** - * Get a static instance of the Taz Data Manager. One is created if it - * doesn't exist already. - * - * @param rb - * A resourcebundle with properties for the TazDataManager. - * @return A static instance of this class. - */ - public static TazDataManager getInstance(HashMap rbMap) - { - if (instance == null) - { - instance = new TazDataManager(rbMap); - return instance; - } else return instance; - } - - /** - * This method should only be used after the getInstance(HashMap rbMap) method has been called since the rbMap is needed to read - * in all the data and populate the object. This method will return the - * instance that has already been populated. - * - * @return instance - * @throws RuntimeException - */ - public static TazDataManager getInstance() throws RuntimeException - { - if (instance == null) - { - throw new RuntimeException( - "Must instantiate TazDataManager with the getInstance(rbMap) method first"); - } else - { - return instance; - } - } - - /** - * Get the number of TAZs with MGRAs. - * - * @return The number of TAZs with MGRAs. - */ - public int getNTazsWithMgras() - { - if (instance != null) - { - return nTazsWithMgras; - } else - { - throw new RuntimeException(); - } - } - - public int[][] getTazMgraArray() - { - if (instance != null) - { - return tazMgraArray; - } else - { - throw new RuntimeException(); - } - } - - /** - * Return the list of MGRAs within this TAZ. - * - * @param taz - * The TAZ number - * @return An array of MGRAs within the TAZ. - */ - public int[] getMgraArray(int taz) - { - if (instance != null) - { - return tazMgraArray[taz]; - } else - { - throw new RuntimeException(); - } - } - - /* - * public int[][] getTazXYCoordArray() { if (instance != null) { return - * tazXYCoordArray; } else { throw new RuntimeException(); } } - * - * public int[] getTazSuperDistrict() { if (instance != null) { return - * tazSuperDistrict; } else { throw new RuntimeException(); } - * - * } - * - * public int[] getPmsa() { if (instance != null) { return this.pmsa; } else - * { throw new RuntimeException(); } } - * - * public int[] getCmsa() { if (instance != null) { return this.cmsa; } else - * { throw new RuntimeException(); } } - */ - - /** - * This method will return the Parking Type for the TAZ. - * - * @param taz - * - TAZ that Parking Type is wanted for. - * @return Parking Type - */ - public int getTazParkingType(int taz) - { - return tazParkingType[taz]; - } - - /** - * Get the list of Park and Ride Taps for this TAZ. - * - * @param Taz - * @return An array of PNR taps for the TAZ. - */ - public int[] getParkRideTapsForZone(int taz) - { - if (tazParkNRideTaps[taz][0] == null) return null; - - int[] parkTaps = new int[tazParkNRideTaps[taz][0].length]; - for (int i = 0; i < tazParkNRideTaps[taz][0].length; i++) - { - parkTaps[i] = (int) tazParkNRideTaps[taz][0][i]; - } - return parkTaps; - } - - /** - * Get the list of Kiss and Ride Taps for this TAZ. - * - * @param Taz - * @return An array of KNR taps for the TAZ. - */ - public int[] getKissRideTapsForZone(int taz) - { - if (tazKissNRideTaps[taz][0] == null) return null; - int[] kissTaps = new int[tazKissNRideTaps[taz][0].length]; - for (int i = 0; i < tazKissNRideTaps[taz][0].length; i++) - { - kissTaps[i] = (int) tazKissNRideTaps[taz][0][i]; - } - return kissTaps; - } - - public int[] getParkRideOrKissRideTapsForZone(int taz, AccessMode aMode) - { - - switch (aMode) - { - case WALK: - return null; - case PARK_N_RIDE: - return getParkRideTapsForZone(taz); - case KISS_N_RIDE: - return getKissRideTapsForZone(taz); - default: - throw new RuntimeException( - "Error trying to get ParkRideOrKissRideTaps for unknown access mode: " - + aMode); - } - } - - /** - * Get the position of the tap in the taz tap array. - * - * @param taz - * The taz to lookup - * @param tap - * The tap to lookup - * @param aMode - * The access mode - * @return The position of the tap in the taz array. -1 is returned if it is - * an invalid tap for the taz. - */ - public int getTapPosition(int taz, int tap, AccessMode aMode) - { - - int[] taps = getParkRideOrKissRideTapsForZone(taz, aMode); - - if (taps == null) return -1; - - for (int i = 0; i < taps.length; ++i) - if (taps[i] == tap) return i; - - return -1; - - } - - /** - * Get the taz to tap time in minutes. - * - * @param taz - * Origin/Production TAZ - * @param pos - * Position of the TAP in this TAZ - * @param mode - * Park and Ride or Kiss and Ride - * @return The TAZ to TAP time in minutes. - */ - public float getTapTime(int taz, int pos, AccessMode aMode) - { - // only expecting this method for Park and Ride and Kiss and Ride modes. - switch (aMode) - { - case PARK_N_RIDE: - return (tazParkNRideTaps[taz][1][pos]); - case KISS_N_RIDE: - return (tazKissNRideTaps[taz][1][pos]); - default: - throw new RuntimeException( - "Error trying to get ParkRideOrKissRideTaps for invalid access mode: " - + aMode); - } - } - - /** - * Get the taz to tap distance in miles. - * - * @param taz - * Origin/Production TAZ - * @param pos - * Position of the TAP in this TAZ - * @param mode - * Park and Ride or Kiss and Ride - * @return The TAZ to TAP distance in miles. - */ - public float getTapDist(int taz, int pos, AccessMode aMode) - { - // only expecting this method for Park and Ride and Kiss and Ride modes. - switch (aMode) - { - case PARK_N_RIDE: - return (tazParkNRideTaps[taz][2][pos]); - case KISS_N_RIDE: - return (tazKissNRideTaps[taz][2][pos]); - default: - throw new RuntimeException( - "Error trying to get ParkRideOrKissRideTaps for invalid access mode: " - + aMode); - } - } - - /** - * Get the time from the TAZ to the TAP in minutes. - * - * @param taz The origin TAZ - * @param tap The destination TAP - * @param aMode The access model (PNR or KNR) - * @return The time in minutes, or -1 if there isn't an access link from the TAZ to the TAP. - */ - public float getTimeToTapFromTaz(int taz, int tap, AccessMode aMode){ - - int btapPosition = getTapPosition(taz,tap,aMode); - float time; - - if(btapPosition==-1){ - logger.info("Bad tap position for " + (aMode==Modes.AccessMode.PARK_N_RIDE ? "PNR" : "KNR") +" access board tap"); - return -1; - }else{ - time = getTapTime(taz,btapPosition,Modes.AccessMode.PARK_N_RIDE); - } - - return time; - - } - /** - * Get the distance from the TAZ to the TAP in miles. - * - * @param taz The origin TAZ - * @param tap The destination TAP - * @param aMode The access model (PNR or KNR) - * @return The distance in miles, or -1 if there isn't an access link from the TAZ to the TAP. - */ - public float getDistanceToTapFromTaz(int taz, int tap, AccessMode aMode){ - - int btapPosition = getTapPosition(taz,tap,aMode); - float distance; - - if(btapPosition==-1){ - logger.info("Bad tap position for " + (aMode==Modes.AccessMode.PARK_N_RIDE ? "PNR" : "KNR") +" access board tap"); - return -1; - }else{ - distance = getTapDist(taz,btapPosition,Modes.AccessMode.PARK_N_RIDE); - } - - return distance; - - } /** - * Returns the max TAZ value - * - * @return the max TAZ value - */ - public int getMaxTaz() - { - return maxTaz; - } - - private void readMgraTableData(HashMap rbMap) - { - - // get the mgra data table from one of these UECs. - String projectPath = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String mgraFile = rbMap.get(MgraDataManager.PROPERTIES_MGRA_DATA_FILE); - mgraFile = projectPath + mgraFile; - - TableDataSet mgraTableDataSet = null; - try - { - OLD_CSVFileReader reader = new OLD_CSVFileReader(); - mgraTableDataSet = reader.readFile(new File(mgraFile)); - } catch (IOException e) - { - logger.error("problem reading mgra data table for TazDataManager.", e); - System.exit(1); - } - - // get 0-based arrays from the specified fields in the MGRA table - mgraTableMgras = mgraTableDataSet.getColumnAsInt(MgraDataManager.MGRA_FIELD_NAME); - mgraTableTazs = mgraTableDataSet.getColumnAsInt(MgraDataManager.MGRA_TAZ_FIELD_NAME); - - } - - /** - * Test an instance of the class by instantiating and reporting. - * - * @param args - * [0] The properties file name/path. - */ - public static void main(String[] args) - { - ResourceBundle rb = ResourceUtil.getPropertyBundle(new File(args[0])); - - TazDataManager tdm = TazDataManager.getInstance(ResourceUtil - .changeResourceBundleIntoHashMap(rb)); - - } - - /** - * This method will return the Origin Terminal Time for the TDZ. - * - * @param taz - * - TAZ that Terminal Time is wanted for. - * @return Origin Terminal Time - */ - public float getOriginTazTerminalTime(int taz) - { - return tazOriginTerminalTime[taz]; - } - - /** - * This method will return the Destination Terminal Time for the TDZ. - * - * @param taz - * - TAZ that Terminal Time is wanted for. - * @return Destination Terminal Time - */ - public float getDestinationTazTerminalTime(int taz) - { - return tazDestinationTerminalTime[taz]; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java deleted file mode 100644 index 6e102b6..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitDriveAccessDMU.java +++ /dev/null @@ -1,465 +0,0 @@ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.modechoice.Modes.AccessMode; - -import com.pb.common.calculator.VariableTable; -/** - * This class is used for ... - * - * @author Joel Freedman - * @version Mar 20, 2009 - *

    - * Created by IntelliJ IDEA. - */ -public class TransitDriveAccessDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TransitDriveAccessDMU.class); - - protected HashMap methodIndexMap; - - double driveTimeToTap; - double driveDistToTap; - double driveDistFromTap; - double driveTimeFromTap; - double OrigDestDistance; - double tapToMgraWalkTime; - double mgraToTapWalkTime; - double carToStationWalkTime; - double escalatorTime; - int accessMode; - int period; - int set; - - //default values for generic application - int applicationType = 0; - int personType = 1; - float ivtCoeff; - float costCoeff; - int joint = 0; //added for consistency with MTC - float valueOfTime = 0; //added for consistency with MTC - - - public TransitDriveAccessDMU() - { - setupMethodIndexMap(); - } - - /** - * Set the joint indicator. - * - * @param joint - */ - public void setTourCategoryIsJoint(int joint){ - this.joint = joint; - } - - /** - * Get the joint indicator. - * - * @return joint - */ - public int getTourCategoryIsJoint(){ - return joint; - } - - /** - * Set the value of time. - * - * @param the value of time - */ - public void setValueOfTime(float valueOfTime){ - this.valueOfTime = valueOfTime; - } - - /** - * Get the value of time. - * - * @return value of time. - */ - public float getValueOfTime(){ - return valueOfTime; - } - - - /** - * Get the walk time from the alighting TAP to the destination MGRA. - * - * @return The walk time from the alighting TAP to the destination MGRA. - */ - public double getTapMgraWalkTime() - { - return tapToMgraWalkTime; - } - - /** - * Set the walk time from the alighting TAP to the destination MGRA. - * - * @param walkTime The walk time from the alighting TAP to the destination MGRA. - */ - public void setTapMgraWalkTime(double walkTime) - { - tapToMgraWalkTime = walkTime; - } - - /** - * Get the walk time to the boarding TAP from the origin MGRA. - * - * @return The walk time from the origin MGRA to the boarding TAP. - */ - public double getMgraTapWalkTime() - { - return mgraToTapWalkTime; - } - - /** - * Set the walk time to the boarding TAP from the origin MGRA - * - * @param walkTime The walk time to the boarding TAP from the origin MGRA. - */ - public void setMgraTapWalkTime(double walkTime) - { - mgraToTapWalkTime = walkTime; - } - - /** - * Get the walk time from the lot to the station. - * - * @return The time in minutes. - */ - public double getCarToStationWalkTime() - { - return carToStationWalkTime; - } - - /** - * Set the walk time from the lot to the station. - * - * @param carToStationWalkTime The time in minutes. - */ - public void setCarToStationWalkTime(double carToStationWalkTime) - { - this.carToStationWalkTime = carToStationWalkTime; - } - - /** - * Get the time to get to the platform. - * - * @return The time in minutes. - */ - public double getEscalatorTime() - { - return escalatorTime; - } - - /** - * Set the time to get to the platform. - * - * @param escalatorTime The time in minutes. - */ - public void setEscalatorTime(double escalatorTime) - { - this.escalatorTime = escalatorTime; - } - - /** - * Get the access mode for this DMU. - * - * @return The access mode. - */ - public int getAccessMode() - { - return accessMode; - } - - /** - * Set the access mode for this DMU. - * - * @param accessMode The access mode. - */ - public void setAccessMode(int accessMode) - { - this.accessMode = accessMode; - } - - /** - * Get the drive time from the origin/production TDZ/TAZ to the TAP. - * - * @return The drive time in minutes. - */ - public double getDriveTimeToTap() - { - return driveTimeToTap; - } - - /** - * Set the drive time from the origin/production TDZ/TAZ to the TAP. - * - * @param driveTimeToTap The drive time in minutes. - */ - public void setDriveTimeToTap(double driveTimeToTap) - { - this.driveTimeToTap = driveTimeToTap; - } - - /** - * Get the drive distance from the origin/production TDZ/TAZ to the TAP. - * - * @return The drive distance in miles. - */ - public double getDriveDistToTap() - { - return driveDistToTap; - } - - /** - * Set the drive distance from the origin/production TDZ/TAZ to the TAP. - * - * @param driveDistToTap The drive distance in miles. - */ - public void setDriveDistToTap(double driveDistToTap) - { - this.driveDistToTap = driveDistToTap; - } - - /** - * Get the drive time from the TAP to the destination/attraction TDZ/TAZ. - * - * @return The drive time in minutes. - */ - public double getDriveTimeFromTap() - { - return driveTimeFromTap; - } - - /** - * Set the drive time from the TAP to the destination/attraction TDZ/TAZ. - * - * @param driveTime The drive time in minutes. - */ - public void setDriveTimeFromTap(double driveTime) - { - driveTimeFromTap = driveTime; - } - - /** - * Get the drive distance from the TAP to the destination/attraction TDZ/TAZ. - * - * @return The drive distance in miles. - */ - public double getDriveDistFromTap() - { - return driveDistFromTap; - } - - /** - * Set the drive distance from the TAP to the destination/attraction TDZ/TAZ. - * - * @param driveDist The drive distance in miles. - */ - public void setDriveDistFromTap(double driveDist) - { - driveDistFromTap = driveDist; - } - - public double getOrigDestDistance() { - return OrigDestDistance; - } - - public void setOrigDestDistance(double origDestDistance) { - OrigDestDistance = origDestDistance; - } - - public void setTOD(int period) { - this.period = period; - } - - public int getTOD() { - return period; - } - - public void setSet(int set) { - this.set = set; - } - - public int getSet() { - return set; - } - - - public void setApplicationType(int applicationType) { - this.applicationType = applicationType; - } - - public int getApplicationType() { - return applicationType; - } - - public void setPersonType(int personType) { - this.personType = personType; - } - - public int getPersonType() { - return personType; - } - - public void setIvtCoeff(float ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public void setCostCoeff(float costCoeff) { - this.costCoeff = costCoeff; - } - - public float getIvtCoeff() { - return ivtCoeff; - } - - public float getCostCoeff() { - return costCoeff; - } - - /** - * Log the DMU values. - * - * @param localLogger The logger to use. - */ - public void logValues(Logger localLogger) - { - - localLogger.info(""); - localLogger.info("Drive-Transit Auto Access DMU Values:"); - localLogger.info(""); - localLogger.info(String.format("Drive Time To Tap: %9.4f", driveTimeToTap)); - localLogger.info(String.format("Drive Dist To Tap: %9.4f", driveDistToTap)); - localLogger.info(String.format("Drive Time From Tap: %9.4f", driveTimeFromTap)); - localLogger.info(String.format("Drive Dist From Tap: %9.4f", driveDistFromTap)); - localLogger.info(String.format("TAP to MGRA walk time: %9.4f", tapToMgraWalkTime)); - localLogger.info(String.format("MGRA to TAP walk time: %9.4f", mgraToTapWalkTime)); - localLogger.info(String.format("Car to station walk time: %9.4f", carToStationWalkTime)); - localLogger.info(String.format("Escalator time: %9.4f", escalatorTime)); - localLogger.info(String.format("Period: %9s", period)); - localLogger.info(String.format("Set: %9s", set)); - localLogger.info(String.format("applicationType: %9s", applicationType)); - localLogger.info(String.format("personType: %9s", personType)); - localLogger.info(String.format("ivtCoeff %9.4f", ivtCoeff)); - localLogger.info(String.format("costCoeff %9.4f", costCoeff)); - localLogger.info(String.format("origDestDistance %9.4f, origDestDistance")); - localLogger.info(String.format("joint : %9s", joint)); - localLogger.info(String.format("value of time : %9.4f", valueOfTime)); - - - AccessMode[] accessModes = AccessMode.values(); - localLogger.info(String.format("Access Mode: %5s", accessModes[accessMode] - .toString())); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getAccessMode", 0); - methodIndexMap.put("getCarToStationWalkTime", 1); - methodIndexMap.put("getDriveDistToTap", 2); - methodIndexMap.put("getDriveTimeToTap", 3); - methodIndexMap.put("getDriveDistFromTap", 4); - methodIndexMap.put("getDriveTimeFromTap", 5); - methodIndexMap.put("getEscalatorTime", 6); - methodIndexMap.put("getTapMgraWalkTime", 7); - methodIndexMap.put("getMgraTapWalkTime", 8); - methodIndexMap.put("getTOD", 9); - methodIndexMap.put("getSet", 10); - - methodIndexMap.put("getApplicationType", 12); - methodIndexMap.put("getTourCategoryIsJoint", 13); - methodIndexMap.put("getPersonType", 14); - methodIndexMap.put("getIvtCoeff", 15); - methodIndexMap.put("getCostCoeff", 16); - methodIndexMap.put("getOrigDestDistance",17); - methodIndexMap.put("getTourCategoryIsJoint", 18); - methodIndexMap.put("getValueOfTime", 19); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getAccessMode(); - case 1: - return getCarToStationWalkTime(); - case 2: - return getDriveDistToTap(); - case 3: - return getDriveTimeToTap(); - case 4: - return getDriveDistFromTap(); - case 5: - return getDriveTimeFromTap(); - case 6: - return getEscalatorTime(); - case 7: - return getTapMgraWalkTime(); - case 8: - return getMgraTapWalkTime(); - case 9: - return getTOD(); - case 10: - return getSet(); - - case 12: - return getApplicationType(); - case 14: - return getPersonType(); - case 15: - return getIvtCoeff(); - case 16: - return getCostCoeff(); - case 17: - return getOrigDestDistance(); - case 18: - return getTourCategoryIsJoint(); - case 19: - return getValueOfTime(); - - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java b/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java deleted file mode 100644 index 781d4d1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/modechoice/TransitWalkAccessDMU.java +++ /dev/null @@ -1,332 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.modechoice; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.VariableTable; - -/** - * WalkDMU is the Decision-Making Unit class for the Walk-transit choice. The class - * contains getter and setter methods for the variables used in the WalkPathUEC. - * - * @author Joel Freedman - * @version 1.0, March, 2009 - * - */ -public class TransitWalkAccessDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(TransitWalkAccessDMU.class); - - protected HashMap methodIndexMap; - - double tapToMgraWalkTime; - double mgraToTapWalkTime; - double escalatorTime; - int period; - int set; - - //default values for generic application - int applicationType = 0; - int personType = 1; //defaults to full-time worker - float ivtCoeff; - float costCoeff; - int accessEgressMode=0; //this is called a walk-access DMU but it is used in the TAP-to-TAP UEC, so it is - //possible that it is being called for a drive-access path! - int joint = 0; //added for consistency with MTC - float valueOfTime = 0; //added for consistency with MTC - public TransitWalkAccessDMU() - { - setupMethodIndexMap(); - } - - - /** - * Set the joint indicator. - * - * @param joint - */ - public void setTourCategoryIsJoint(int joint){ - this.joint = joint; - } - - /** - * Get the joint indicator. - * - * @return joint - */ - public int getTourCategoryIsJoint(){ - return joint; - } - - /** - * Set the value of time. - * - * @param the value of time - */ - public void setValueOfTime(float valueOfTime){ - this.valueOfTime = valueOfTime; - } - - /** - * Get the value of time. - * - * @return value of time. - */ - public float getValueOfTime(){ - return valueOfTime; - } - - - /** - * Set the access/egress mode - * - * @param accessEgressMode - */ - public void setAccessEgressMode(int accessEgressMode){ - this.accessEgressMode = accessEgressMode; - } - - - /** - * Get the access/egress mode - * - * @return accessEgressMode - */ - public int getAccessEgressMode(){ - return accessEgressMode; - } - - - /** - * Get the time from the production/origin MGRA to the boarding TAP. - * - * @return The time from the production/origin MGRA to the boarding TAP. - */ - public double getMgraTapWalkTime() - { - return mgraToTapWalkTime; - } - - /** - * Set the time from the production/origin MGRA to the boarding TAP. - * - * @param walkTime The time from the production/origin MGRA to the boarding TAP. - */ - public void setMgraTapWalkTime(double walkTime) - { - this.mgraToTapWalkTime = walkTime; - } - - /** - * Get the time from the alighting TAP to the attraction/destination MGRA. - * - * @return The time from the alighting TAP to the attraction/destination MGRA. - */ - public double getTapMgraWalkTime() - { - return tapToMgraWalkTime; - } - - /** - * Set the time from the alighting TAP to the attraction/destination MGRA. - * - * @param walkTime The time from the alighting TAP to the attraction/destination - * MGRA. - */ - public void setTapMgraWalkTime(double walkTime) - { - this.tapToMgraWalkTime = walkTime; - } - - /** - * Get the time to get to the platform. - * - * @return The time in minutes. - */ - public double getEscalatorTime() - { - return escalatorTime; - } - - /** - * Set the time to get to the platform. - * - * @param escalatorTime The time in minutes. - */ - public void setEscalatorTime(double escalatorTime) - { - this.escalatorTime = escalatorTime; - } - - public void setTOD(int period) { - this.period = period; - } - - public int getTOD() { - return period; - } - - public void setSet(int set) { - this.set = set; - } - - public int getSet() { - return set; - } - - - public void setApplicationType(int applicationType) { - this.applicationType = applicationType; - } - - public int getApplicationType() { - return applicationType; - } - - - public void setPersonType(int personType) { - this.personType = personType; - } - - public int getPersonType() { - return personType; - } - - public void setIvtCoeff(float ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public void setCostCoeff(float costCoeff) { - this.costCoeff = costCoeff; - } - - public float getIvtCoeff() { - return ivtCoeff; - } - - public float getCostCoeff() { - return costCoeff; - } - - /** - * Log the DMU values. - * - * @param localLogger The logger to use. - */ - public void logValues(Logger localLogger) - { - - localLogger.info(""); - localLogger.info("Walk DMU Values:"); - localLogger.info(""); - localLogger.info(String.format("MGRA to TAP walk time: %9.4f", mgraToTapWalkTime)); - localLogger.info(String.format("TAP to MGRA walk time: %9.4f", tapToMgraWalkTime)); - localLogger.info(String.format("Escalator time: %9.4f", escalatorTime)); - localLogger.info(String.format("Period: %9s", period)); - localLogger.info(String.format("Set: %9s", set)); - localLogger.info(String.format("applicationType: %9s", applicationType)); - localLogger.info(String.format("personType: %9s", personType)); - localLogger.info(String.format("ivtCoeff : %9.4f", ivtCoeff)); - localLogger.info(String.format("costCoeff : %9.4f", costCoeff)); - localLogger.info(String.format("accessEgressMode : %9s", accessEgressMode)); - localLogger.info(String.format("joint : %9s", joint)); - localLogger.info(String.format("value of time : %9.4f", valueOfTime)); - - - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getEscalatorTime", 0); - methodIndexMap.put("getMgraTapWalkTime", 1); - methodIndexMap.put("getTapMgraWalkTime", 2); - methodIndexMap.put("getTOD", 3); - methodIndexMap.put("getSet", 4); - - methodIndexMap.put("getApplicationType", 6); - methodIndexMap.put("getPersonType", 8); - methodIndexMap.put("getIvtCoeff", 9); - methodIndexMap.put("getCostCoeff", 10); - methodIndexMap.put("getAccessEgressMode", 11); - methodIndexMap.put("getTourCategoryIsJoint", 12); - methodIndexMap.put("getValueOfTime", 13); - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getEscalatorTime(); - case 1: - return getMgraTapWalkTime(); - case 2: - return getTapMgraWalkTime(); - case 3: - return getTOD(); - case 4: - return getSet(); - case 6: - return getApplicationType(); - case 8: - return getPersonType(); - case 9: - return getIvtCoeff(); - case 10: - return getCostCoeff(); - case 11: - return getAccessEgressMode(); - case 12: - return getTourCategoryIsJoint(); - case 13: - return getValueOfTime(); - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java deleted file mode 100644 index a757c38..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/AbstractCsvExporter.java +++ /dev/null @@ -1,32 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.util.Properties; -import org.apache.log4j.Logger; - -public abstract class AbstractCsvExporter - implements IExporter -{ - private final File file; - private final IMatrixDao matrixDao; - private final String reportFolder = "report.path"; - - protected static final Logger LOGGER = Logger.getLogger(AbstractCsvExporter.class); - - public AbstractCsvExporter(Properties properties, IMatrixDao aMatrixDao, String aBaseFileName) - { - this.file = new File(properties.getProperty(reportFolder), aBaseFileName + ".csv"); - this.matrixDao = aMatrixDao; - } - - public IMatrixDao getMatrixDao() - { - return this.matrixDao; - } - - public File getFile() - { - return this.file; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java deleted file mode 100644 index ecc5632..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMExporter.java +++ /dev/null @@ -1,304 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Properties; -import java.util.Set; - -import org.apache.log4j.Logger; - -import com.pb.common.datafile.DataTypes; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; - -public class CVMExporter { - private static final Logger logger = Logger.getLogger(DataExporter.class); - - public final String[] cvmPeriodNames = {"OE","AM","MD","PM","OL"}; - public final String[] periodNames = {"EA","AM","MD","PM","EV"}; - public final String[] cvmClassNames = {"L","M","I","H"}; - public final String[] skimTollClassNames = {"SOV_NT_H","TRK_L","TRK_M","TRK_H"}; - public final String[] skimNonTollClassNames = {"SOV_TR_H","TRK_L","TRK_M","TRK_H"}; - public final String[] nonTollSkims = {"TIME","DIST"}; - public final String[] tollSkims = {"TIME","DIST","TOLLCOST"}; - public final String[] cvmModeNames = {"NT","T"}; - public final String[] modelModeNames = {"GP","TOLL"}; - - public final String[] segmentNames = {"FA","RE","GO","IN","SV","WH","TH"}; - - protected Properties properties; - protected String projectPath; - protected String reportPath; - protected HashMap cvmSkimMap; - protected HashMap periodMap; //lookup cvm period, return model period - protected HashMap tollClassMap; //lookup cvm class, return toll skim class - protected HashMap nonTollClassMap; //lookup cvm class, return non-toll skim class - - protected HashMap modeMap; //lookup cvm mode, return model mode - - private final OMXMatrixDao mtxDao; - protected float autoOperatingCost; - - - public CVMExporter(Properties theProperties, OMXMatrixDao aMtxDao){ - this.properties = theProperties; - this.mtxDao = aMtxDao; - projectPath = properties.getProperty("scenario.path"); - reportPath = properties.getProperty("report.path"); - float fuelCost = new Float(properties.getProperty("aoc.fuel")); - float mainCost = new Float(properties.getProperty("aoc.maintenance")); - autoOperatingCost = (fuelCost + mainCost) * 0.01f; - - } - - private void createPeriodMap(){ - - periodMap = new HashMap(); - for(int i = 0; i(); - tollClassMap = new HashMap(); - for(int i = 0; i(); - for (int i = 0; i < cvmModeNames.length;++i) - modeMap.put(cvmModeNames[i], modelModeNames[i]); - } - - public void export(){ - createPeriodMap(); - createClassMap(); - createModeMap(); - readSkims(); - TableDataSet inputData = readCVMTrips(); - int totalRows = inputData.getRowCount(); - float[] timeCol = new float[totalRows]; - float[] distCol = new float[totalRows]; - float[] aocCol = new float[totalRows]; - float[] tollCol = new float[totalRows]; - - for(int row = 1; row<=totalRows;++row){ - - int otaz = (int) inputData.getValueAt(row, "I"); - int dtaz = (int) inputData.getValueAt(row, "J"); - String cvmPeriod = inputData.getStringValueAt(row,"OriginalTimePeriod"); - String cvmClass = inputData.getStringValueAt(row,"Mode"); - String cvmMode = inputData.getStringValueAt(row,"TripMode"); - - Matrix timeMatrix = null; - Matrix distMatrix = null; - Matrix tollMatrix = null; - - String modelPeriod = periodMap.get(cvmPeriod); - String modelClass = null; - if(cvmMode.equals("NT")){ - modelClass = nonTollClassMap.get(cvmClass); - - }else{ - modelClass = tollClassMap.get(cvmClass); - } - - timeMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"TIME"); - distMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"DIST"); - if(cvmSkimMap.containsKey(modelPeriod+"_"+modelClass+"_"+"TOLLCOST")) - tollMatrix = cvmSkimMap.get(modelPeriod+"_"+modelClass+"_"+"TOLLCOST"); - - timeCol[row-1] = timeMatrix.getValueAt(otaz, dtaz); - distCol[row-1] = distMatrix.getValueAt(otaz, dtaz); - aocCol[row-1] = distMatrix.getValueAt(otaz, dtaz) * autoOperatingCost; - if(tollMatrix != null) - tollCol[row-1] = tollMatrix.getValueAt(otaz, dtaz); - - } - - //append the columns - inputData.appendColumn(timeCol, "TIME"); - inputData.appendColumn(distCol, "DIST"); - inputData.appendColumn(aocCol, "AOC"); - inputData.appendColumn(tollCol, "TOLLCOST"); - - //write the data - TableDataSet.writeFile(reportPath+"cvm_trips.csv", inputData); - - } - - /** - * Read data into inputDataTable tabledataset. - * - */ - private TableDataSet readTableDataSet(String inputFile){ - - logger.info("Begin reading the data in file " + inputFile); - TableDataSet inputDataTable = null; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - inputDataTable = csvFile.readFile(new File(inputFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End reading the data in file " + inputFile); - - return inputDataTable; - } - - private void readSkims(){ - - cvmSkimMap = new HashMap(); - - for(String period: periodNames){ - - String fileName = "traffic_skims_"+period+".omx"; - - for(String nonTollClass: skimNonTollClassNames){ - - for(String skim:nonTollSkims){ - - String skimName = period+"_"+nonTollClass+"_"+skim; - - Matrix m = mtxDao.getMatrix(fileName, skimName); - cvmSkimMap.put(skimName, m); - - } - } - for(String tollClass: skimTollClassNames){ - for(String skim:tollSkims){ - String skimName = period+"_"+tollClass+"_"+skim; - - Matrix m = mtxDao.getMatrix(fileName, skimName); - cvmSkimMap.put(skimName, m); - - } - } - } - - - - - } - - /** - * Helper method to read in all the CVM files and concatenate into one TableDataSet. - * - * @return the concatenated data - */ - private TableDataSet readCVMTrips(){ - - int tables = 0; - String[] header = null; - int[] columnType = null; - HashMap> floatCols = new HashMap>(); - HashMap> stringCols = new HashMap>(); - - //first read all the data, and store arraylists of data in the two hashmaps - for(String segment: segmentNames){ - - for(String period:cvmPeriodNames){ - - String fileName = projectPath+"output\\Trip_"+segment+"_"+period+".csv"; - TableDataSet inData = readTableDataSet(fileName); - if(tables==0){ - columnType = inData.getColumnType(); - header = inData.getColumnLabels(); - } - ++tables; - for(int i = 0; i< inData.getColumnCount();++i){ - - String colName = header[i]; - if(columnType[i]==DataTypes.NUMBER){ - float[] data = inData.getColumnAsFloat(colName); - ArrayList colArray = null; - if(floatCols.containsKey(colName)) - colArray = floatCols.get(colName); - else - colArray = new ArrayList(); - - for(int j=0;j colArray = null; - if(stringCols.containsKey(colName)) - colArray = stringCols.get(colName); - else - colArray = new ArrayList(); - - for(int j=0;j keySet = floatCols.keySet(); - String[] colNames = new String[keySet.size()]; - float[][] data = null; - int colNumber=0; - for(String colName:keySet){ - colNames[colNumber] = colName; - ArrayList col = floatCols.get(colName); - if(colNumber==0){ - - data = new float[col.size()][colNames.length]; - } - for(int i = 0; i < col.size();++i){ - data[i][colNumber] = col.get(i); - } - ++colNumber; - } - - TableDataSet allTrips = TableDataSet.create(data,colNames); - - //logger.info("Created table data set with "+ allTrips.getColumnCount()+" columns"); - //String outputColNames = ""; - //for(String colName : allTrips.getColumnLabels()) - // outputColNames += (colName + " "); - //logger.info("Columns: "+outputColNames); - - //now append the string columns - keySet = stringCols.keySet(); - colNames = new String[keySet.size()]; - colNumber=0; - for(String colName:keySet){ - colNames[colNumber] = colName; - ArrayList col = stringCols.get(colName); - String[] stringData = new String[col.size()]; - stringData = col.toArray(stringData); - logger.info("Appending string column "+colName+" with "+stringData.length+" elements"); - allTrips.appendColumn(stringData, colName); - } - - //logger.info("Final table data set has "+ allTrips.getColumnCount()+" columns"); - //outputColNames = ""; - //for(String colName : allTrips.getColumnLabels()) - // outputColNames += (colName + " "); - //logger.info("Columns: "+outputColNames); - - return allTrips; - - - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java deleted file mode 100644 index d7d595e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/CVMScaler.java +++ /dev/null @@ -1,179 +0,0 @@ -package org.sandag.abm.reporting; -import java.io.File; -import java.io.FileInputStream; -import java.io.FileNotFoundException; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.List; -import java.util.Properties; - -import org.apache.log4j.Logger; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -public class CVMScaler { - private static final Logger logger= Logger.getLogger(DataExporter.class); - protected Properties properties; - protected String projectPath; - protected String reportPath; - protected String [] lightscalers; - protected String [] mediumscalers; - protected String [] heavyscalers; - protected float lightshare; - protected float mediumshare; - protected float heavyshare; - - public CVMScaler(Properties theProperties){ - this.properties = theProperties; - projectPath = properties.getProperty("scenario.path"); - reportPath = properties.getProperty("report.path"); - String delims = "[,]"; - lightscalers = properties.getProperty("cvm.scale_light").split(delims); - mediumscalers = properties.getProperty("cvm.scale_medium").split(delims); - heavyscalers = properties.getProperty("cvm.scale_heavy").split(delims); - lightshare = new Float(properties.getProperty("cvm.share.light")); - mediumshare = new Float(properties.getProperty("cvm.share.medium")); - heavyshare = new Float(properties.getProperty("cvm.share.heavy")); - } - - public void scale(){ - logger.info("Running CVM scaler ... "); - String fileName = reportPath+"cvm_trips.csv"; - TableDataSet inData = readTableDataSet(fileName); - int totalRows = inData.getRowCount(); - String[] timeCol=new String[totalRows]; - - int tod=lightscalers.length; - float [] lscaler=new float[tod]; - float [] mscaler=new float[tod]; - float [] hscaler=new float[tod]; - - for(int i=0; i0) { - if (colname.equals("Mode")){ - value = value.replaceAll(vehicle, "I"); - }else if (colname.equals("TripTime")){ - value = value.replaceAll(":"+vehicle, ":I"); - } - - if (line_new==null) line_new = value; - else line_new = line_new + "," + value; - } - } - - //write existing lines - scaler = getScaler(scalerArray, str); - value_new = scaler * (1-share); - line = line + "," + Float.toString(value_new); - writer.println(line.trim()); - - //write new lines - if (share>0) { - value_new = scaler * (share); - line_new = line_new + "," + Float.toString(value_new); - writer.println(line_new.trim()); - } - } - - private float getScaler(float [] scalerArray, String str) { - float scaler = 1.0f; - - if(str.contains("_EA")){ - scaler = scalerArray[0]; - }else if(str.contains("_AM")){ - scaler = scalerArray[1]; - }else if(str.contains("_MD")){ - scaler = scalerArray[2]; - }else if(str.contains("_PM")){ - scaler = scalerArray[3]; - }else if(str.contains("_EV")){ - scaler = scalerArray[4]; - }else { - scaler = 1.0f; - } - - return scaler; - - } - - private TableDataSet readTableDataSet(String inputFile){ - - logger.info("Begin reading the data in file " + inputFile); - TableDataSet inputDataTable = null; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - inputDataTable = csvFile.readFile(new File(inputFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End reading the data in file " + inputFile); - - return inputDataTable; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java deleted file mode 100644 index fb93a89..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvRow.java +++ /dev/null @@ -1,22 +0,0 @@ -package org.sandag.abm.reporting; - -public class CsvRow -{ - private final String row; - - public CsvRow(String[] values) - { - StringBuilder sb = new StringBuilder(32); - sb.append(values[0]); - for (int i = 1; i < values.length; i++) - sb.append(',').append(values[i]); - sb.append(System.lineSeparator()); - - row = sb.toString(); - } - - public String getRow() - { - return row; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java deleted file mode 100644 index 9ee01e2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/CsvWriterThread.java +++ /dev/null @@ -1,86 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.io.FileOutputStream; -import java.io.IOException; -import java.nio.ByteBuffer; -import java.nio.channels.FileChannel; -import java.util.concurrent.BlockingQueue; -import org.apache.log4j.Logger; - -public class CsvWriterThread - implements Runnable -{ - private final BlockingQueue queue; - private final File file; - private final String[] header; - - private static final Logger LOGGER = Logger.getLogger(CsvWriterThread.class); - - private int maxBuffer = 1024 * 1024 * 1024; - private static final String ENCODING = "UTF-8"; - - public static final CsvRow POISON_PILL = new CsvRow(new String[] {"ALL_DONE"}); - - public CsvWriterThread(BlockingQueue aRowQueue, File anOutputLocation, String[] aHeader) - { - this.queue = aRowQueue; - this.file = anOutputLocation; - this.header = aHeader; - } - - public int getMaxBuffer() - { - return this.maxBuffer; - } - - public void setMaxBuffer(int aMaxBuffer) - { - this.maxBuffer = aMaxBuffer; - } - - @Override - public void run() - { - FileOutputStream outStream = null; - try - { - outStream = new FileOutputStream(file, false); - FileChannel outChannel = outStream.getChannel(); - ByteBuffer buffer = ByteBuffer.allocateDirect(getMaxBuffer()); - - CsvRow headerRow = new CsvRow(header); - buffer.put(headerRow.getRow().getBytes(ENCODING)); - - CsvRow row = null; - while ((row = queue.take()) != CsvWriterThread.POISON_PILL) - { - byte[] rowBytes = row.getRow().getBytes(ENCODING); - if ((buffer.position() + rowBytes.length) > buffer.capacity()) - { - buffer.flip(); - outChannel.write(buffer); - buffer.clear(); - } - buffer.put(rowBytes); - } - LOGGER.info("End of records found. Clearing Buffer and Writing Remains."); - buffer.flip(); - outChannel.write(buffer); - } catch (IOException | InterruptedException e) - { - LOGGER.fatal(e); - throw new RuntimeException(e); - } finally - { - if (null != outStream) try - { - outStream.close(); - LOGGER.info("CSV Writer Stream Closed."); - } catch (IOException e) - { - LOGGER.error(e); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java deleted file mode 100644 index 30bc37e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/DataExporter.java +++ /dev/null @@ -1,2688 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.BufferedReader; -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileInputStream; -import java.io.FileReader; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.rmi.RemoteException; -import java.text.DecimalFormat; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.LinkedHashMap; -import java.util.LinkedHashSet; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Properties; -import java.util.Set; -import java.util.TreeSet; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.LinkedBlockingQueue; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.matrix.OMXMatrixWriter; -import com.pb.common.util.ResourceUtil; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -/** - * The {@code DataExporter} ... - * - * @author crf Started 9/20/12 8:36 AM - */ -public final class DataExporter -{ - private static final Logger LOGGER = Logger.getLogger(DataExporter.class); - - private static final String NUMBER_FORMAT_NAME = "NUMBER"; - private static final String STRING_FORMAT_NAME = "STRING"; - private static final String PROJECT_PATH_PROPERTY_TOKEN = "%project.folder%"; - private static final String TOD_TOKEN = "%TOD%"; - - private final Properties properties; - private final OMXMatrixDao mtxDao; - private final File projectPathFile; - private final int feedbackIterationNumber; - private final Set tables; - private final String[] timePeriods = ModelStructure.MODEL_PERIOD_LABELS; - private final String FUEL_COST_PROPERTY = "aoc.fuel"; - private final String MAINTENANCE_COST_PROPERTY = "aoc.maintenance"; - private static final String WRITE_LOGSUMS_PROPERTY = "Results.WriteLogsums"; - private static final String WRITE_TRANSIT_IVT_PROPERTY = "Report.writeTransitIVT"; - private static final String WRITE_UTILS_PROPERTY = "TourModeChoice.Save.UtilsAndProbs"; - - private float autoOperatingCost; - - private boolean writeCSV = false; - private boolean writeLogsums = false; - private boolean writeUtilities = true; - //private boolean writeTransitIVTs = false; - private MatrixDataServerIf ms; - - - public DataExporter(Properties theProperties, OMXMatrixDao aMtxDao, String projectPath, - int feedbackIterationNumber) - { - this.properties = theProperties; - this.mtxDao = aMtxDao; - - projectPathFile = new File(theProperties.getProperty("Project.Directory")); - this.feedbackIterationNumber = feedbackIterationNumber; - - float fuelCost = new Float(theProperties.getProperty(FUEL_COST_PROPERTY)); - float mainCost = new Float(theProperties.getProperty(MAINTENANCE_COST_PROPERTY)); - writeLogsums = new Boolean(theProperties.getProperty(WRITE_LOGSUMS_PROPERTY)); - writeUtilities = new Boolean(theProperties.getProperty(WRITE_UTILS_PROPERTY)); - //writeTransitIVTs = new Boolean(theProperties.getProperty(WRITE_TRANSIT_IVT_PROPERTY)); - - autoOperatingCost = (fuelCost + mainCost) * 0.01f; - - tables = new LinkedHashSet(); - - } - - private void addTable(String table) - { - tables.add(table); - LOGGER.info("exporting data: " + table); - } - - private String getPath(String path) - { - if (properties.containsKey(path)) return getPathFromProperty(path); - File ff = new File(path); - if (!ff.exists()) ff = new File(projectPathFile, path); - return ff.getAbsolutePath(); - } - - private String getPathFromProperty(String propertyToken) - { - String path = (String) properties.get(propertyToken); - if (!path.startsWith(projectPathFile.getAbsolutePath())) - path = new File(projectPathFile, path).getAbsolutePath(); - return path; - } - - /** - * Takes an input file name, returns the path to the directory - * with that name. - * - * @param The name of the file to create - * @return The path of the file. - */ - private String getOutputPath(String file) - { - return new File(properties.getProperty("report.path"), file).getAbsolutePath(); - } - - private String getData(TableDataSet data, int row, int column, FieldType type) - { - switch (type) - { - case INT: - return "" + Math.round(data.getValueAt(row, column)); - case FLOAT: - return "" + data.getValueAt(row, column); - case STRING: - return data.getStringValueAt(row, column); - case BIT: - return Boolean.parseBoolean(data.getStringValueAt(row, column)) ? "1" : "0"; - default: - throw new IllegalStateException("undefined field type: " + type); - } - } - - private String getPreferredColumnName(String columnName) - { - if (columnName.equalsIgnoreCase("hh_id")) return "HH_ID"; - if (columnName.equalsIgnoreCase("person_id")) return "PERSON_ID"; - if (columnName.toLowerCase().contains("maz")) - return columnName.toLowerCase().replace("maz", "mgra").toUpperCase(); - return columnName.toUpperCase(); - } - - private void exportData(TableDataSet data, String outputFileName, - Map outputMapping, Map outputTypes) - { - int[] outputIndices = new int[outputMapping.size()]; - FieldType[] outputFieldTypes = new FieldType[outputIndices.length]; - String[] header = new String[outputMapping.size()]; - - int counter = 0; - for (String column : outputMapping.keySet()) - { - header[counter] = column; - outputIndices[counter] = data.getColumnPosition(outputMapping.get(column)); - outputFieldTypes[counter++] = outputTypes.get(column); - } - - BlockingQueue queue = new LinkedBlockingQueue(); - Thread writerProcess = null; - try - { - CsvWriterThread writerThread = new CsvWriterThread(queue, new File( - getOutputPath(outputFileName + ".csv")), header); - writerProcess = new Thread(writerThread); - writerProcess.start(); - - for (int i = 1; i <= data.getRowCount(); i++) - { - String[] row = new String[outputMapping.size()]; - row[0] = getData(data, i, outputIndices[0], outputFieldTypes[0]); - - for (int j = 1; j < outputIndices.length; j++) - { - row[j] = getData(data, i, outputIndices[j], outputFieldTypes[j]); - } - queue.add(new CsvRow(row)); - } - } finally - { - queue.add(CsvWriterThread.POISON_PILL); - if (null != writerProcess) - { - try - { - writerProcess.join(); - } catch (InterruptedException e) - { - LOGGER.error(e); - System.exit(-1); - } - } - } - } - - private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, - boolean includeFeedbackIteration, String[] formats, Set floatColumns, - Set stringColumns, Set intColumns, Set bitColumns, - FieldType defaultFieldType, Set primaryKey, - TripStructureDefinition tripStructureDefinition,boolean isCB) - { - return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, - formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, - primaryKey, tripStructureDefinition, null,isCB); - } - - private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, - boolean includeFeedbackIteration, String[] formats, Set floatColumns, - Set stringColumns, Set intColumns, Set bitColumns, - FieldType defaultFieldType, Set primaryKey, - TripStructureDefinition tripStructureDefinition, JoinData joinData,boolean isCB) - { - return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, - formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, - primaryKey, new HashMap(), tripStructureDefinition, joinData,isCB); - } - - private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, - boolean includeFeedbackIteration, String[] formats, Set floatColumns, - Set stringColumns, Set intColumns, Set bitColumns, - FieldType defaultFieldType, Set primaryKey, - Map overridingFieldMappings, - TripStructureDefinition tripStructureDefinition,boolean isCB) - { - return exportDataGeneric(outputFileBase, filePropertyToken, includeFeedbackIteration, - formats, floatColumns, stringColumns, intColumns, bitColumns, defaultFieldType, - primaryKey, overridingFieldMappings, tripStructureDefinition, null,isCB); - } - - private TableDataSet exportDataGeneric(String outputFileBase, String filePropertyToken, - boolean includeFeedbackIteration, String[] formats, Set floatColumns, - Set stringColumns, Set intColumns, Set bitColumns, - FieldType defaultFieldType, Set primaryKey, - Map overridingFieldMappings, - TripStructureDefinition tripStructureDefinition, JoinData joinData,boolean isCB) - { - TableDataSet table; - try - { - String f = includeFeedbackIteration ? getPath(filePropertyToken).replace(".csv", - "_" + feedbackIterationNumber + ".csv") : getPath(filePropertyToken); - table = formats == null ? new CSVFileReader().readFile(new File(f)) - : new CSVFileReader().readFileWithFormats(new File(f), formats); - } catch (IOException e) - { - throw new RuntimeException(e); - } - if (joinData != null) joinData.joinDataToTable(table); - exportDataGeneric(table, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, defaultFieldType, primaryKey, overridingFieldMappings, - tripStructureDefinition,isCB); - return table; - } - - private class JoinData - { - private final Map> data; - private final Map dataType; - private final String idColumn; - - public JoinData(String idColumn) - { - this.idColumn = idColumn; - data = new LinkedHashMap>(); - dataType = new HashMap(); - } - - public void addJoinData(Map joinData, FieldType type, String columnName) - { - data.put(columnName, joinData); - dataType.put(columnName, type); - } - - public void joinDataToTable(TableDataSet table) - { - int[] ids = table.getColumnAsInt(idColumn); - for (String column : data.keySet()) - table.appendColumn(getData(ids, column), column); - } - - private Object getData(int[] ids, String column) - { - switch (dataType.get(column)) - { - case INT: - { - int[] columnData = new int[ids.length]; - @SuppressWarnings("unchecked") - // this is correct - Map dataMap = (Map) data.get(column); - for (int i = 0; i < ids.length; i++) - columnData[i] = dataMap.get(ids[i]); - return columnData; - } - case FLOAT: - { - float[] columnData = new float[ids.length]; - @SuppressWarnings("unchecked") - // this is correct - Map dataMap = (Map) data.get(column); - for (int i = 0; i < ids.length; i++) - columnData[i] = dataMap.get(ids[i]); - return columnData; - } - case STRING: - { - String[] columnData = new String[ids.length]; - @SuppressWarnings("unchecked") - // this is correct - Map dataMap = (Map) data.get(column); - for (int i = 0; i < ids.length; i++) - columnData[i] = dataMap.get(ids[i]); - return columnData; - } - case BIT: - { - boolean[] columnData = new boolean[ids.length]; - @SuppressWarnings("unchecked") - // this is correct - Map dataMap = (Map) data.get(column); - for (int i = 0; i < ids.length; i++) - columnData[i] = dataMap.get(ids[i]); - return columnData; - } - default: - throw new IllegalStateException("shouldn't be here: " + dataType.get(column)); - } - } - } - - private void exportDataGeneric(TableDataSet table, String outputFileBase, - Set floatColumns, Set stringColumns, Set intColumns, - Set bitColumns, FieldType defaultFieldType, Set primaryKey, - TripStructureDefinition tripStructureDefinition,boolean isCB) - { - exportDataGeneric(table, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, defaultFieldType, primaryKey, new HashMap(), - tripStructureDefinition,isCB); - - } - - private void exportDataGeneric(TableDataSet table, String outputFileBase, - Set floatColumns, Set stringColumns, Set intColumns, - Set bitColumns, FieldType defaultFieldType, Set primaryKey, - Map overridingFieldMappings, - TripStructureDefinition tripStructureDefinition,boolean isCB) - { - Map fieldMappings = new LinkedHashMap(); - Map fieldTypes = new HashMap(); - - if (tripStructureDefinition != null) - { - appendTripData(table, tripStructureDefinition,isCB); - floatColumns.add("AUTO_IVT"); - floatColumns.add("AUTO_AOC"); - floatColumns.add("AUTO_STD"); - floatColumns.add("AUTO_TOLL"); - floatColumns.add("TRAN_IVT"); - floatColumns.add("TRAN_WAIT"); - floatColumns.add("TRAN_WALK"); - floatColumns.add("TRAN_FARE"); - floatColumns.add("TRAN_ACCDIST"); - floatColumns.add("TRAN_EGRDIST"); - floatColumns.add("TRAN_AUXTIME"); - floatColumns.add("TRAN_ACCTIME"); - floatColumns.add("TRAN_EGRTIME"); - floatColumns.add("TRAN_TRANSFERS"); - floatColumns.add("WALK_TIME"); - floatColumns.add("BIKE_TIME"); - floatColumns.add("TRIP_DIST"); - stringColumns.add("TRIP_PURPOSE_NAME"); - stringColumns.add("TRIP_MODE_NAME"); - intColumns.add("RECID"); - floatColumns.add("LOC_IVT"); - floatColumns.add("EXP_IVT"); - floatColumns.add("BRT_IVT"); - floatColumns.add("LRT_IVT"); - floatColumns.add("CR_IVT"); - floatColumns.add("TRAN_DIST"); - floatColumns.add("PARK_WALK_TIME"); - floatColumns.add("PARK_WALK_DIST"); - - } - - if (primaryKey.size() == 0) - { - // have to add in a key - call it ID - int[] id = new int[table.getRowCount()]; - for (int i = 0; i < id.length; i++) - id[i] = i + 1; - table.appendColumn(id, "ID"); - - primaryKey.add("ID"); - intColumns.add("ID"); - } - - outer: for (String column : table.getColumnLabels()) - { - String c = overridingFieldMappings.containsKey(column) ? overridingFieldMappings - .get(column) : getPreferredColumnName(column); - fieldMappings.put(c, column); - for (String fc : floatColumns) - { - if (fc.equalsIgnoreCase(column)) - { - fieldTypes.put(c, FieldType.FLOAT); - continue outer; - } - } - for (String sc : stringColumns) - { - if (sc.equalsIgnoreCase(column)) - { - fieldTypes.put(c, FieldType.STRING); - continue outer; - } - } - for (String sc : intColumns) - { - if (sc.equalsIgnoreCase(column)) - { - fieldTypes.put(c, FieldType.INT); - continue outer; - } - } - for (String sc : bitColumns) - { - if (sc.equalsIgnoreCase(column)) - { - fieldTypes.put(c, FieldType.BIT); - continue outer; - } - } - fieldTypes.put(c, defaultFieldType); - } - Set pKey = new LinkedHashSet(); - for (String column : primaryKey) - pKey.add(getPreferredColumnName(column)); - exportData(table, outputFileBase, fieldMappings, fieldTypes); - } - - private PrintWriter getBufferedPrintWriter(String fileName) throws IOException - { - return new PrintWriter(new BufferedWriter(new FileWriter(fileName))); - } - - /** - * Appends trip data to table including skim attributes. - * - * @param table - * @param tripStructureDefinition - */ - private void appendTripData(TableDataSet table, TripStructureDefinition tripStructureDefinition, boolean isCB) - { - // id triptype recid partysize orig_maz dest_maz trip_board_tap - // trip_alight_tap trip_depart_time trip_time trip_distance trip_cost - // trip_purpose_name trip_mode_name vot - int rowCount = table.getRowCount(); - - float[] autoInVehicleTime = new float[rowCount]; - float[] autoOperatingCost = new float[rowCount]; - float[] autoStandardDeviation = new float[rowCount]; - float[] autoTollCost = new float[rowCount]; - float[] transitInVehicleTime = new float[rowCount]; - float[] transitWaitTime = new float[rowCount]; - float[] transitWalkTime = new float[rowCount]; - float[] transitFare = new float[rowCount]; - float[] walkModeTime = new float[rowCount]; - float[] bikeModeTime = new float[rowCount]; - float[] tripDistance = new float[rowCount]; - String[] tripPurpose = new String[rowCount]; - String[] tripMode = new String[rowCount]; - int[] tripId = new int[rowCount]; - int[] tripDepartTime = new int[rowCount]; - int[] tripBoardTaz = new int[rowCount]; - int[] tripAlightTaz = new int[rowCount]; - float[] tripParkingTime = new float[rowCount]; - float[] tripParkingDistance = new float[rowCount]; - - - float[] transitAccessDist = new float[rowCount]; - float[] transitEgressDist = new float[rowCount]; - float[] transitAuxTime = new float[rowCount]; - float[] transitAccTime = new float[rowCount]; - float[] transitEgrTime = new float[rowCount]; - float[] transitTransfers = new float[rowCount]; - - //these are only set if writeTransitIVTs is true - float[] locIVT = new float[rowCount]; - float[] expIVT = new float[rowCount]; - float[] brtIVT = new float[rowCount]; - float[] lrtIVT = new float[rowCount]; - float[] crIVT = new float[rowCount]; - - float[] tranDist = new float[rowCount]; - - SkimBuilder skimBuilder = new SkimBuilder(properties); - boolean hasPurposeColumn = tripStructureDefinition.originPurposeColumn > -1; - for (int i = 0; i < rowCount; i++) - { - int row = i + 1; - - double epsilon = .000001; - boolean inbound = tripStructureDefinition.booleanIndicatorVariables ? table - .getBooleanValueAt(row, tripStructureDefinition.inboundColumn) : Math.abs(table - .getValueAt(row, tripStructureDefinition.inboundColumn) - 1.0) < epsilon; - - int transponderOwnership=0; - if(tripStructureDefinition.transponderOwnershipColumn>0) - transponderOwnership = (int) table.getValueAt(row, tripStructureDefinition.transponderOwnershipColumn); - - SkimBuilder.TripAttributes attributes = skimBuilder.getTripAttributes( - (int) table.getValueAt(row, tripStructureDefinition.originMgraColumn), - (int) table.getValueAt(row, tripStructureDefinition.destMgraColumn), - (int) table.getValueAt(row, tripStructureDefinition.modeColumn), - (int) table.getValueAt(row, tripStructureDefinition.boardTapColumn), - (int) table.getValueAt(row, tripStructureDefinition.alightTapColumn), - (int) table.getValueAt(row, tripStructureDefinition.todColumn), - inbound, - table.getValueAt(row,tripStructureDefinition.valueOfTimeColumn), - (int) table.getValueAt(row, tripStructureDefinition.setColumn),isCB,transponderOwnership); - - autoInVehicleTime[i] = attributes.getAutoInVehicleTime(); - autoOperatingCost[i] = attributes.getAutoOperatingCost(); - autoStandardDeviation[i] = attributes.getAutoStandardDeviationTime(); - autoTollCost[i] = attributes.getAutoTollCost(); - transitInVehicleTime[i] = attributes.getTransitInVehicleTime(); - transitWaitTime[i] = attributes.getTransitWaitTime(); - transitWalkTime[i] = attributes.getTransitWalkTime(); - transitFare[i] = attributes.getTransitFare(); - walkModeTime[i] = attributes.getWalkModeTime(); - bikeModeTime[i] = attributes.getBikeModeTime(); - tripDistance[i] = attributes.getTripDistance(); - transitAccessDist[i] = attributes.getTransitAccessDistance(); - transitEgressDist[i] = attributes.getTransitEgressDistance(); - transitAuxTime[i] = attributes.getTransitAuxiliaryTime(); - transitAccTime[i] = attributes.getTransitAccessTime(); - transitEgrTime[i] = attributes.getTransitEgressTime(); - transitTransfers[i] = attributes.getTransitTransfers(); - - - //get parking walk time - if(tripStructureDefinition.parkingMazColumn>-1) { - int parkingMaz = (int) table.getValueAt(row, tripStructureDefinition.parkingMazColumn); - - if(parkingMaz>0) { - - int destMaz = (int) table.getValueAt(row, tripStructureDefinition.destMgraColumn); - float parkingWalkTime = skimBuilder.getLotWalkTime(parkingMaz,destMaz); - float parkingWalkDistance = skimBuilder.getLotWalkDistance(parkingMaz,destMaz); - tripParkingTime[i]= parkingWalkTime; - tripParkingDistance[i] = parkingWalkDistance; - } - } - - - if (hasPurposeColumn) - { - tripPurpose[i] = table.getStringValueAt(row, - tripStructureDefinition.destinationPurposeColumn); - } else - { - if (!inbound) // going out - tripPurpose[i] = tripStructureDefinition.destinationName; - else tripPurpose[i] = tripStructureDefinition.homeName; - } - tripMode[i] = attributes.getTripModeName(); - tripId[i] = i; - tripDepartTime[i] = attributes.getTripStartTime(); - tripBoardTaz[i] = attributes.getTripBoardTaz(); - tripAlightTaz[i] = attributes.getTripAlightTaz(); - - locIVT[i] = attributes.getLocTime(); - expIVT[i] = attributes.getExpTime(); - brtIVT[i] = attributes.getBrtTime(); - lrtIVT[i] = attributes.getLrtTime(); - crIVT[i] = attributes.getCrTime(); - - tranDist[i] = attributes.getTransitDistance(); - } - table.appendColumn(autoInVehicleTime, "AUTO_IVT"); - table.appendColumn(autoOperatingCost, "AUTO_AOC"); - table.appendColumn(autoStandardDeviation, "AUTO_STD"); - table.appendColumn(autoTollCost, "AUTO_TOLL"); - table.appendColumn(transitInVehicleTime, "TRAN_IVT"); - table.appendColumn(transitWaitTime, "TRAN_WAIT"); - table.appendColumn(transitWalkTime, "TRAN_WALK"); - table.appendColumn(transitFare, "TRAN_FARE"); - table.appendColumn(transitAccessDist, "TRAN_ACCDIST"); - table.appendColumn(transitEgressDist, "TRAN_EGRDIST"); - table.appendColumn(transitAuxTime, "TRAN_AUXTIME"); - table.appendColumn(transitAccTime, "TRAN_ACCTIME"); - table.appendColumn(transitEgrTime, "TRAN_EGRTIME"); - table.appendColumn(transitTransfers, "TRAN_TRANSFERS"); - - table.appendColumn(walkModeTime, "WALK_TIME"); - table.appendColumn(bikeModeTime, "BIKE_TIME"); - table.appendColumn(tripDistance, "TRIP_DIST"); - table.appendColumn(tripPurpose, "TRIP_PURPOSE_NAME"); - table.appendColumn(tripMode, "TRIP_MODE_NAME"); - table.appendColumn(tripId, "RECID"); - table.appendColumn(tripBoardTaz, "TRIP_BOARD_TAZ"); - table.appendColumn(tripAlightTaz, "TRIP_ALIGHT_TAZ"); - - table.appendColumn(locIVT, "LOC_IVT"); - table.appendColumn(expIVT, "EXP_IVT"); - table.appendColumn(brtIVT, "BRT_IVT"); - table.appendColumn(lrtIVT, "LRT_IVT"); - table.appendColumn(crIVT, "CR_IVT"); - - table.appendColumn(tranDist, "TRAN_DIST"); - table.appendColumn(tripParkingTime, "PARK_WALK_TIME"); - table.appendColumn(tripParkingDistance,"PARK_WALK_DIST"); - -// table.appendColumn(valueOfTime, "VALUE_OF_TIME"); - } - - private void exportAccessibilities(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(Arrays.asList("mgra")); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("mgra")); - exportDataGeneric(outputFileBase, "acc.output.file", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private void exportMazData(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(Arrays.asList("mgra", "TAZ", "ZIP09")); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("mgra")); - exportDataGeneric(outputFileBase, "mgra.socec.file", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private void nullifyFile(String file) - { - String tempFile = file + ".temp"; - File f = new File(file); - if (!f.renameTo(new File(tempFile))) - throw new RuntimeException("Couldn't rename to file: " + f); - BufferedReader reader = null; - PrintWriter writer = null; - try - { - reader = new BufferedReader(new FileReader(tempFile)); - writer = getBufferedPrintWriter(file); - String line; - while ((line = reader.readLine()) != null) - writer.println(line.replace(NULL_VALUE, "")); - } catch (IOException e) - { - throw new RuntimeException(e); - } finally - { - if (reader != null) - { - try - { - reader.close(); - } catch (IOException e) - { - // ignore - } - } - if (writer != null) writer.close(); - } - new File(tempFile).delete(); - } - - public static int NULL_INT_VALUE = -98765; - public static float NULL_FLOAT_VALUE = NULL_INT_VALUE; - public static String NULL_VALUE = "" + NULL_FLOAT_VALUE; - - private void exportTapData(String outputFileBase) - { - addTable(outputFileBase); - Map ptype = readSpaceDelimitedData(getPath("tap.ptype.file"), - Arrays.asList("TAP", "LOTID", "PTYPE", "TAZ", "CAPACITY", "DISTANCE")); - Map pelev = readSpaceDelimitedData( - getPath("tap.ptype.file").replace("ptype", "elev"), Arrays.asList("TAP", "ELEV")); - float[] taps = ptype.get("TAP"); - float[] etaps = pelev.get("TAP"); - ptype.put("ELEV", getPartialData(taps, etaps, pelev.get("ELEV"))); - - TableDataSet finalData = new TableDataSet(); - for (String columnName : ptype.keySet()) - finalData.appendColumn(ptype.get(columnName), columnName); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("TAP")); - exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, FieldType.INT, primaryKey, null,false); - nullifyFile(getOutputPath(outputFileBase + ".csv")); - } - - private void exportMgraToTapData(String outputFileBase) - { - addTable(outputFileBase); - String walkdistanceFile=PROJECT_PATH_PROPERTY_TOKEN+"\\input\\"+properties.getProperty("active.logsum.matrix.file.walk.mgratap"); - Map mgraToTap = readSpaceDelimitedData(walkdistanceFile, Arrays.asList("MGRA", "TAP", "DISTANCE")); - TableDataSet finalData = new TableDataSet(); - for (String columnName : mgraToTap.keySet()) - finalData.appendColumn(mgraToTap.get(columnName), columnName); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("MGRA", "TAP")); - exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, FieldType.INT, primaryKey, null,false); - nullifyFile(getOutputPath(outputFileBase + ".csv")); - } - - private void exportMgraToMgraData(String outputFileBase) - { - addTable(outputFileBase); - //wu modified to get the updated walk distance between MGRAs - String walkdistanceFile=PROJECT_PATH_PROPERTY_TOKEN+"\\input\\"+properties.getProperty("active.logsum.matrix.file.walk.mgra"); - Map mgraToMgra = readSpaceDelimitedData(walkdistanceFile, Arrays.asList("ORIG_MGRA", "DEST_MGRA", "DISTANCE")); - Map> actualData = new LinkedHashMap>(); - for (String column : Arrays.asList("TAZ", "ORIG_MGRA", "DEST_MGRA", "DISTANCE")) - actualData.put(column, new LinkedList()); - float[] dcolumn = mgraToMgra.get("DISTANCE"); - float[] origColumn = mgraToMgra.get("ORIG_MGRA"); - float[] destColumn = mgraToMgra.get("DEST_MGRA"); - for (int i = 0; i < dcolumn.length; i++) - { - int count = 0; - if (dcolumn[i] < 0) count = (int) destColumn[i]; - int taz = (int) origColumn[i]; - while (count-- > 0) - { - i++; - actualData.get("TAZ").add(taz); - actualData.get("ORIG_MGRA").add((int) origColumn[i]); - actualData.get("DEST_MGRA").add((int) destColumn[i]); - actualData.get("DISTANCE").add(dcolumn[i]); - } - } - - TableDataSet finalData = new TableDataSet(); - for (String columnName : actualData.keySet()) - { - Object data; - if (columnName.equals("DISTANCE")) - { - float[] dd = new float[actualData.get(columnName).size()]; - int counter = 0; - for (Number n : actualData.get(columnName)) - dd[counter++] = n.floatValue(); - data = dd; - } else - { - int[] dd = new int[actualData.get(columnName).size()]; - int counter = 0; - for (Number n : actualData.get(columnName)) - dd[counter++] = n.intValue(); - data = dd; - } - finalData.appendColumn(data, columnName); - } - - Set intColumns = new HashSet(Arrays.asList("TAZ", "ORIG_MGRA", "DEST_MGRA")); - Set floatColumns = new HashSet(Arrays.asList("DISTANCE")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("ORIG_MGRA", "DEST_MGRA")); - exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, FieldType.INT, primaryKey, null,false); - nullifyFile(getOutputPath(outputFileBase + ".csv")); - } - - private void exportTazToTapData(String outputFileBase) - { - addTable(outputFileBase); - Map tazToTap = readSpaceDelimitedData( - getPath("taz.driveaccess.taps.file"), Arrays.asList("TAZ", "TAP", "TIME", "DISTANCE", "MODE")); - - Map> actualData = new LinkedHashMap>(); - for (String column : Arrays.asList("TAZ", "TAP", "TIME", "DISTANCE", "MODE")) - actualData.put(column, new LinkedList()); - - float[] taz = tazToTap.get("TAZ"); - float[] tap = tazToTap.get("TAP"); - float[] time = tazToTap.get("TIME"); - float[] dist = tazToTap.get("DISTANCE"); - float[] mode = tazToTap.get("MODE"); - - for (int i = 0; i < taz.length; i++) - { - actualData.get("TAZ").add((int) taz[i]); - actualData.get("TAP").add((int) tap[i]); - actualData.get("TIME").add(time[i]); - actualData.get("DISTANCE").add(dist[i]); - actualData.get("MODE").add(mode[i]); - } - - TableDataSet finalData = new TableDataSet(); - for (String columnName : actualData.keySet()) - { - Object data; - if (columnName.equals("DISTANCE") || columnName.equals("TIME")) - { - float[] dd = new float[actualData.get(columnName).size()]; - int counter = 0; - for (Number n : actualData.get(columnName)) - dd[counter++] = n.floatValue(); - data = dd; - } else - { - int[] dd = new int[actualData.get(columnName).size()]; - int counter = 0; - for (Number n : actualData.get(columnName)) - dd[counter++] = n.intValue(); - data = dd; - } - finalData.appendColumn(data, columnName); - } - - Set intColumns = new HashSet(Arrays.asList("TAZ", "TAP")); - Set floatColumns = new HashSet(Arrays.asList("TIME", "DISTANCE")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("TAZ", "TAP")); - exportDataGeneric(finalData, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, FieldType.INT, primaryKey, null,false); - nullifyFile(getOutputPath(outputFileBase + ".csv")); - } - - private float[] toFloatArray(int[] data) - { - float[] f = new float[data.length]; - for (int i = 0; i < f.length; i++) - f[i] = data[i]; - return f; - } - - private float[] getPartialData(float[] fullKey, float[] partialKey, float[] partialData) - { - float[] data = new float[fullKey.length]; - Arrays.fill(data, NULL_FLOAT_VALUE); - int counter = 0; - for (float key : fullKey) - { - for (int i = 0; i < partialKey.length; i++) - { - if (partialKey[i] == key) - { - data[counter] = partialData[i]; - } - } - counter++; - } - return data; - } - - private void exportTazData(String outputFileBase) - { - addTable(outputFileBase); - int[] tazs = getTazList(); - TableDataSet data = new TableDataSet(); - data.appendColumn(tazs, "TAZ"); - Map term = readSpaceDelimitedData(getPath("taz.terminal.time.file"), - Arrays.asList("TAZ", "TERM")); - Map park = readSpaceDelimitedData(getPath("taz.parkingtype.file"), - Arrays.asList("TAZ", "PARK")); - data.appendColumn(getPartialData(toFloatArray(tazs), term.get("TAZ"), term.get("TERM")), - "TERM"); - data.appendColumn(getPartialData(toFloatArray(tazs), park.get("TAZ"), park.get("PARK")), - "PARK"); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("TAZ")); - exportDataGeneric(data, outputFileBase, floatColumns, stringColumns, intColumns, - bitColumns, FieldType.INT, primaryKey, null,false); - nullifyFile(getOutputPath(outputFileBase + ".csv")); - } - - private int[] getTazList() - { - Set tazs = new TreeSet(); - TableDataSet mgraData; - try - { - mgraData = new CSVFileReader().readFile(new File(getPath("mgra.socec.file"))); - } catch (IOException e) - { - throw new RuntimeException(e); - } - boolean first = true; - for (int taz : mgraData.getColumnAsInt("taz")) - { - if (first) - { - first = false; - continue; - } - tazs.add(taz); - } - int[] finalTazs = new int[tazs.size()]; - int counter = 0; - for (int taz : tazs) - finalTazs[counter++] = taz; - return finalTazs; - } - - private Map readSpaceDelimitedData(String location, List columnNames) - { - Map> data = new LinkedHashMap>(); - for (String columnName : columnNames) - data.put(columnName, new LinkedList()); - BufferedReader reader = null; - try - { - reader = new BufferedReader(new FileReader(location)); - String line; - while ((line = reader.readLine()) != null) - { - String[] d = line.trim().split(","); - int counter = 0; - for (String columnName : columnNames) - { - if (counter < d.length) - { - data.get(columnName).add(Float.parseFloat(d[counter++])); - } else - { - data.get(columnName).add(NULL_FLOAT_VALUE); // if missing - // entry/entries, - // then put in - // null value - } - } - } - } catch (IOException e) - { - throw new RuntimeException(e); - } finally - { - if (reader != null) - { - try - { - reader.close(); - } catch (IOException e) - { - // ignore - } - } - } - Map d = new LinkedHashMap(); - for (String columnName : columnNames) - { - float[] f = new float[data.get(columnName).size()]; - int counter = 0; - for (Float i : data.get(columnName)) - f[counter++] = i; - d.put(columnName, f); - } - return d; - } - - private void exportHouseholdData(String outputFileBase) - { - addTable(outputFileBase); - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // hh_id - formatList.add(NUMBER_FORMAT_NAME); // home_mgra - formatList.add(NUMBER_FORMAT_NAME); // income - formatList.add(NUMBER_FORMAT_NAME); // autos - formatList.add(NUMBER_FORMAT_NAME); // HVs - formatList.add(NUMBER_FORMAT_NAME); // AVs - formatList.add(NUMBER_FORMAT_NAME); // transponder - formatList.add(STRING_FORMAT_NAME); // cdap_pattern - formatList.add(NUMBER_FORMAT_NAME); // jtf_choice - - - if(writeLogsums){ - formatList.add(NUMBER_FORMAT_NAME); //aoLogsum - formatList.add(NUMBER_FORMAT_NAME); //transponderLogsum - formatList.add(NUMBER_FORMAT_NAME); //cdapLogsum - formatList.add(NUMBER_FORMAT_NAME); //jtfLogsum - } - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - - if(writeLogsums){ - floatColumns.add("aoLogsum"); //aoLogsum - floatColumns.add("transponderLogsum"); //transponderLogsum - floatColumns.add("cdapLogsum"); //cdapLogsum - floatColumns.add("jtfLogsum"); //jtfLogsum - } - - Set stringColumns = new HashSet(Arrays.asList("cdap_pattern")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id")); - exportDataGeneric(outputFileBase, "Results.HouseholdDataFile", true, formats, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, null,false); - } - - private void exportPersonData(String outputFileBase) - { - addTable(outputFileBase); - - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // hh_id - formatList.add(NUMBER_FORMAT_NAME); // person_id - formatList.add(NUMBER_FORMAT_NAME); // person_num - formatList.add(NUMBER_FORMAT_NAME); // age - formatList.add(STRING_FORMAT_NAME); // gender - formatList.add(STRING_FORMAT_NAME); // type - formatList.add(NUMBER_FORMAT_NAME); // value_of_time (float) - formatList.add(STRING_FORMAT_NAME); // activity_pattern - formatList.add(NUMBER_FORMAT_NAME); // imf_choice - formatList.add(NUMBER_FORMAT_NAME); // inmf_choice - formatList.add(NUMBER_FORMAT_NAME); // fp_choice - formatList.add(NUMBER_FORMAT_NAME); // reimb_pct (float) - formatList.add(NUMBER_FORMAT_NAME); // ie_choice - formatList.add(NUMBER_FORMAT_NAME); // timeFactorWork - formatList.add(NUMBER_FORMAT_NAME); // timeFactorNonWork - - if(writeLogsums){ - formatList.add(NUMBER_FORMAT_NAME); //wfhLogsum - formatList.add(NUMBER_FORMAT_NAME); //wlLogsum - formatList.add(NUMBER_FORMAT_NAME); //slLogsum - formatList.add(NUMBER_FORMAT_NAME); //fpLogsum - formatList.add(NUMBER_FORMAT_NAME); //ieLogsum - formatList.add(NUMBER_FORMAT_NAME); //cdapLogsum - formatList.add(NUMBER_FORMAT_NAME); //imtfLogsum - formatList.add(NUMBER_FORMAT_NAME);//inmtfLogsum - } - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("value_of_time", "reimb_pct")); - - if(writeLogsums){ - floatColumns.add("wfhLogsum"); //wfhLogsum - floatColumns.add("wlLogsum"); //wlLogsum - floatColumns.add("slLogsum"); //slLogsum - floatColumns.add("fpLogsum"); //fpLogsum - floatColumns.add("ieLogsum"); //ieLogsum - floatColumns.add("cdapLogsum"); //cdapLogsum - floatColumns.add("imtfLogsum"); //imtfLogsum - floatColumns.add("inmtfLogsum");//inmtfLogsum - } - - Set stringColumns = new HashSet(Arrays.asList("gender", "type", - "activity_pattern")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("person_id")); - exportDataGeneric(outputFileBase, "Results.PersonDataFile", true, formats, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, null,false); - } - - private void exportSyntheticHouseholdData(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("HHID")); - exportDataGeneric(outputFileBase, "PopulationSynthesizer.InputToCTRAMP.HouseholdFile", - false, null, floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, - primaryKey, null,false); - } - - private void exportSyntheticPersonData(String outputFileBase) - { - addTable(outputFileBase); - String[] formats = {NUMBER_FORMAT_NAME, // HHID - NUMBER_FORMAT_NAME, // PERID - NUMBER_FORMAT_NAME, // household_serial_no - NUMBER_FORMAT_NAME, // PNUM - NUMBER_FORMAT_NAME, // AGE - NUMBER_FORMAT_NAME, // SEX - NUMBER_FORMAT_NAME, // MILTARY - NUMBER_FORMAT_NAME, // PEMPLOY - NUMBER_FORMAT_NAME, // PSTUDENT - NUMBER_FORMAT_NAME, // PTYPE - NUMBER_FORMAT_NAME, // EDUC - NUMBER_FORMAT_NAME, // GRADE - NUMBER_FORMAT_NAME, // OCCCEN5 - STRING_FORMAT_NAME, // OCCSOC5 - NUMBER_FORMAT_NAME, // INDCEN - NUMBER_FORMAT_NAME, // WEEKS - NUMBER_FORMAT_NAME, // HOURS - }; - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(Arrays.asList("OCCSOC5")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("PERID")); - exportDataGeneric(outputFileBase, "PopulationSynthesizer.InputToCTRAMP.PersonFile", false, - formats, floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, - primaryKey, null,false); - } - - private void exportWorkSchoolLocation(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("WorkLocationDistance", - "WorkLocationLogsum", "SchoolLocation", "SchoolLocationDistance", - "SchoolLocationLogsum")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("PERSON_ID")); - Map overridingNames = new HashMap(); - overridingNames.put("PersonID", "PERSON_ID"); - exportDataGeneric(outputFileBase, "Results.UsualWorkAndSchoolLocationChoice", true, null, - floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - overridingNames, null,false); - } - - private void exportIndivToursData(String outputFileBase) - { - addTable(outputFileBase); - - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // hh_id - formatList.add(NUMBER_FORMAT_NAME); // person_id - formatList.add(NUMBER_FORMAT_NAME); // person_num - formatList.add(NUMBER_FORMAT_NAME); // person_type - formatList.add(NUMBER_FORMAT_NAME); // tour_id - formatList.add(STRING_FORMAT_NAME); // tour_category - formatList.add(STRING_FORMAT_NAME); // tour_purpose - formatList.add(NUMBER_FORMAT_NAME); // orig_maz - formatList.add(NUMBER_FORMAT_NAME); // dest_maz - formatList.add(NUMBER_FORMAT_NAME); // start_period - formatList.add(NUMBER_FORMAT_NAME); // end_period - formatList.add(NUMBER_FORMAT_NAME); // tour_mode - formatList.add(NUMBER_FORMAT_NAME); // av_avail - formatList.add(NUMBER_FORMAT_NAME); // tour_distance - formatList.add(NUMBER_FORMAT_NAME); // atWork_freq - formatList.add(NUMBER_FORMAT_NAME); // num_ob_stops - formatList.add(NUMBER_FORMAT_NAME); // num_ib_stops - formatList.add(NUMBER_FORMAT_NAME); // valueOfTime - - if(writeUtilities){ - for(int i=1;i<=26;++i) - formatList.add(NUMBER_FORMAT_NAME); // util_i - for(int i=1;i<=26;++i) - formatList.add(NUMBER_FORMAT_NAME); // prob_i - } - - if(writeLogsums){ - formatList.add(NUMBER_FORMAT_NAME); //timeOfDayLogsum - formatList.add(NUMBER_FORMAT_NAME);//tourModeLogsum - formatList.add(NUMBER_FORMAT_NAME);//subtourFreqLogsum - formatList.add(NUMBER_FORMAT_NAME);//tourDestinationLogsum - formatList.add(NUMBER_FORMAT_NAME);//stopFreqLogsum - - for(int i = 1; i<=4;++i) - formatList.add(NUMBER_FORMAT_NAME);//outStopDCLogsum_i - - for(int i = 1; i<=4;++i) - formatList.add(NUMBER_FORMAT_NAME);//inbStopDCLogsum_i - } - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(Arrays.asList("hh_id", "person_id", - "person_num", "person_type", "tour_id", "orig_mgra", "dest_mgra", "start_period", - "end_period", "tour_mode", "av_avail", "atWork_freq", "num_ob_stops", "num_ib_stops")); - - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - - if(writeUtilities){ - for(int i=1;i<=12;++i) - floatColumns.add("util_"+i); // util_i - for(int i=1;i<=12;++i) - floatColumns.add("prob_"+i); // prob_i - } - if(writeLogsums){ - floatColumns.add("timeOfDayLogsum"); - floatColumns.add("tourModeLogsum"); - floatColumns.add("subtourFreqLogsum"); - floatColumns.add("tourDestinationLogsum"); - floatColumns.add("stopFreqLogsum"); - - for(int i = 1; i<=4;++i) - floatColumns.add("outStopDCLogsum_"+i);//outStopDCLogsum_i - - for(int i = 1; i<=4;++i) - floatColumns.add("inbStopDCLogsum_"+i);//inbStopDCLogsum_i - } - - Set stringColumns = new HashSet(Arrays.asList("tour_category", - "tour_purpose")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "person_id", - "tour_category", "tour_id", "tour_purpose")); - exportDataGeneric(outputFileBase, "Results.IndivTourDataFile", true, formats, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private void exportJointToursData(String outputFileBase) - { - addTable(outputFileBase); - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // hh_id - formatList.add(NUMBER_FORMAT_NAME); // tour_id - formatList.add(STRING_FORMAT_NAME); // tour_category - formatList.add(STRING_FORMAT_NAME); // tour_purpose - formatList.add(NUMBER_FORMAT_NAME); // tour_composition - formatList.add(STRING_FORMAT_NAME); // tour_participants - formatList.add(NUMBER_FORMAT_NAME); // orig_maz - formatList.add(NUMBER_FORMAT_NAME); // dest_maz - formatList.add(NUMBER_FORMAT_NAME); // start_period - formatList.add(NUMBER_FORMAT_NAME); // end_period - formatList.add(NUMBER_FORMAT_NAME); // tour_mode - formatList.add(NUMBER_FORMAT_NAME); // av_avail - formatList.add(NUMBER_FORMAT_NAME); // tour_distance - formatList.add(NUMBER_FORMAT_NAME); // num_ob_stops - formatList.add(NUMBER_FORMAT_NAME); // num_ib_stops - formatList.add(NUMBER_FORMAT_NAME); // valueOfTime - - if(writeUtilities){ - for(int i=1;i<=26;++i) - formatList.add(NUMBER_FORMAT_NAME); // util_i - for(int i=1;i<=26;++i) - formatList.add(NUMBER_FORMAT_NAME); // prob_i - } - - if(writeLogsums){ - formatList.add(NUMBER_FORMAT_NAME); //timeOfDayLogsum - formatList.add(NUMBER_FORMAT_NAME);//tourModeLogsum - formatList.add(NUMBER_FORMAT_NAME);//subtourFreqLogsum - formatList.add(NUMBER_FORMAT_NAME);//tourDestinationLogsum - formatList.add(NUMBER_FORMAT_NAME);//stopFreqLogsum - - for(int i = 1; i<=4;++i) - formatList.add(NUMBER_FORMAT_NAME);//outStopDCLogsum_i - - for(int i = 1; i<=4;++i) - formatList.add(NUMBER_FORMAT_NAME);//inbStopDCLogsum_i - } - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(Arrays.asList("hh_id", "tour_id", - "tour_composition", "orig_mgra", "dest_mgra", "start_period", "end_period", - "tour_mode", "av_avail", "num_ob_stops", "num_ib_stops")); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - - if(writeUtilities){ - for(int i=1;i<=12;++i) - floatColumns.add("util_"+i); // util_i - for(int i=1;i<=12;++i) - floatColumns.add("prob_"+i); // prob_i - } - if(writeLogsums){ - floatColumns.add("timeOfDayLogsum"); - floatColumns.add("tourModeLogsum"); - floatColumns.add("subtourFreqLogsum"); - floatColumns.add("tourDestinationLogsum"); - floatColumns.add("stopFreqLogsum"); - - for(int i = 1; i<=4;++i) - floatColumns.add("outStopDCLogsum_"+i);//outStopDCLogsum_i - - for(int i = 1; i<=4;++i) - floatColumns.add("inbStopDCLogsum_"+i);//inbStopDCLogsum_i - } - Set stringColumns = new HashSet(Arrays.asList("tour_category", - "tour_purpose", "tour_participants")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "tour_category", - "tour_id", "tour_purpose")); - exportDataGeneric(outputFileBase, "Results.JointTourDataFile", true, formats, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private void exportIndivTripData(String outputFileBase) - { - addTable(outputFileBase); - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // 1 hh_id - formatList.add(NUMBER_FORMAT_NAME); // 2 person_id - formatList.add(NUMBER_FORMAT_NAME); // 3 person_num - formatList.add(NUMBER_FORMAT_NAME); // 4 tour_id - formatList.add(NUMBER_FORMAT_NAME); // 5 stop_id - formatList.add(NUMBER_FORMAT_NAME); // 6 inbound - formatList.add(STRING_FORMAT_NAME); // 7 tour_purpose - formatList.add(STRING_FORMAT_NAME); // 8 orig_purpose - formatList.add(STRING_FORMAT_NAME); // 9 dest_purpose - formatList.add(NUMBER_FORMAT_NAME); // 10 orig_maz - formatList.add(NUMBER_FORMAT_NAME); // 11 dest_maz - formatList.add(NUMBER_FORMAT_NAME); // 12 parking_maz - formatList.add(NUMBER_FORMAT_NAME); // 13 stop_period - formatList.add(NUMBER_FORMAT_NAME); // 14 trip_mode - formatList.add(NUMBER_FORMAT_NAME); // 15 av_avail - formatList.add(NUMBER_FORMAT_NAME); // 16 trip_board_tap - formatList.add(NUMBER_FORMAT_NAME); // 17 trip_alight_tap - formatList.add(NUMBER_FORMAT_NAME); // 18 set - formatList.add(NUMBER_FORMAT_NAME); // 19 tour_mode - formatList.add(NUMBER_FORMAT_NAME); // 20 driver_pnum - formatList.add(NUMBER_FORMAT_NAME); // 21 orig_escort_stoptype - formatList.add(NUMBER_FORMAT_NAME); // 22 orig_escortee_pnum - formatList.add(NUMBER_FORMAT_NAME); // 23 dest_escort_stoptype - formatList.add(NUMBER_FORMAT_NAME); // 24 dest_escortee_pnum - formatList.add(NUMBER_FORMAT_NAME); // 25 value of time - formatList.add(NUMBER_FORMAT_NAME); // 26 transponder availability - formatList.add(NUMBER_FORMAT_NAME); // 27 micro_walkMode - formatList.add(NUMBER_FORMAT_NAME); // 28 micro_trnAcc - formatList.add(NUMBER_FORMAT_NAME); // 29 micro_trnEgr - - if(writeLogsums) - formatList.add(NUMBER_FORMAT_NAME);//tripModeLogsum - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(); - - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - - if(writeLogsums) - floatColumns = new HashSet(Arrays.asList("valueOfTime","tripModeLogsum")); - - Set stringColumns = new HashSet(Arrays.asList("tour_purpose", - "orig_purpose", "dest_purpose")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "person_id", - "tour_id", "tour_purpose", "inbound", "stop_id")); - exportDataGeneric( - outputFileBase, - "Results.IndivTripDataFile", - true, - formats, - floatColumns, - stringColumns, - intColumns, - bitColumns, - FieldType.INT, - primaryKey, - new TripStructureDefinition(10, 11, 8, 9, 13, 14, 16, 17, 12, -1, 29, "INDIV", 6, false, 25, 18,26),false); - } - - private void exportJointTripData(String outputFileBase) - { - addTable(outputFileBase); - ArrayList formatList = new ArrayList(); - - formatList.add(NUMBER_FORMAT_NAME); // 1 hh_id - formatList.add(NUMBER_FORMAT_NAME); // 2 tour_id - formatList.add(NUMBER_FORMAT_NAME); // 3 stop_id - formatList.add(NUMBER_FORMAT_NAME); // 4 inbound - formatList.add(STRING_FORMAT_NAME); // 5 tour_purpose - formatList.add(STRING_FORMAT_NAME); // 6 orig_purpose - formatList.add(STRING_FORMAT_NAME); // 7 dest_purpose - formatList.add(NUMBER_FORMAT_NAME); // 8 orig_maz - formatList.add(NUMBER_FORMAT_NAME); // 9 dest_maz - formatList.add(NUMBER_FORMAT_NAME); // 10 parking_maz - formatList.add(NUMBER_FORMAT_NAME); // 11 stop_period - formatList.add(NUMBER_FORMAT_NAME); // 12 trip_mode - formatList.add(NUMBER_FORMAT_NAME); // 13 av_avail - formatList.add(NUMBER_FORMAT_NAME); // 14 num_participants - formatList.add(NUMBER_FORMAT_NAME); // 15 trip_board_tap - formatList.add(NUMBER_FORMAT_NAME); // 16 trip_alight_tap - formatList.add(NUMBER_FORMAT_NAME); // 17 set - formatList.add(NUMBER_FORMAT_NAME); // 18 tour_mode - formatList.add(NUMBER_FORMAT_NAME); // 19 value of time - formatList.add(NUMBER_FORMAT_NAME); // 20 transponder availability - formatList.add(NUMBER_FORMAT_NAME); // 21 micro_walkMode - formatList.add(NUMBER_FORMAT_NAME); // 22 micro_trnAcc - formatList.add(NUMBER_FORMAT_NAME); // 23 micro_trnEgr - - if(writeLogsums) - formatList.add(NUMBER_FORMAT_NAME);//tripModeLogsum - - String[] formats = new String[formatList.size()]; - formats = formatList.toArray(formats); - - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - - if(writeLogsums) - floatColumns = new HashSet(Arrays.asList("valueOfTime","tripModeLogsum")); - - Set stringColumns = new HashSet(Arrays.asList("tour_purpose", - "orig_purpose", "dest_purpose")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("hh_id", "tour_id", - "tour_purpose", "inbound", "stop_id")); - exportDataGeneric(outputFileBase, "Results.JointTripDataFile", true, formats, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - new TripStructureDefinition(8, 9, 6, 7, 11, 12, 15, 16, 10, 14, 23, "JOINT", 4, false, 19, 17,20),false); - } - - private void exportAirportTripsSAN(String outputFileBase) - { - - //id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); - - addTable(outputFileBase); - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("id")); - Map overridingNames = new HashMap(); - // overridingNames.put("id","PARTYID"); - exportDataGeneric(outputFileBase, "airport.SAN.output.file", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, overridingNames, - new TripStructureDefinition(8, 9, 7, 12, 15, 16, -1, 4, 18, "AIRPORT", "HOME", - "AIRPORT", 2, false, 18, 17,-1),false); - } - - private void exportAirportTripsCBX(String outputFileBase) - { - - //id,direction,purpose,size,income,nights,departTime,originMGRA,destinationMGRA,originTAZ,destinationTAZ,tripMode,av_avail,arrivalMode,boardingTAP,alightingTAP,set,valueOfTime\n"); - addTable(outputFileBase); - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("id")); - Map overridingNames = new HashMap(); - // overridingNames.put("id","PARTYID"); - exportDataGeneric(outputFileBase, "airport.CBX.output.file", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, overridingNames, - new TripStructureDefinition(8, 9, 7, 12, 15, 16, -1, 4, 18, "AIRPORT", "HOME", - "AIRPORT", 2, false, 18, 17,-1),false); - } - - private void exportCrossBorderTourData(String outputFileBase) - { - addTable(outputFileBase); - String[] formats = {NUMBER_FORMAT_NAME, // id - NUMBER_FORMAT_NAME, // purpose - STRING_FORMAT_NAME, // sentri - NUMBER_FORMAT_NAME, // poe - NUMBER_FORMAT_NAME, // departTime - NUMBER_FORMAT_NAME, // arriveTime - NUMBER_FORMAT_NAME, // originMGRA - NUMBER_FORMAT_NAME, // destinationMGRA - NUMBER_FORMAT_NAME, // origTaz - NUMBER_FORMAT_NAME, // destTaz - NUMBER_FORMAT_NAME, // tourMode - NUMBER_FORMAT_NAME, // av_avail - NUMBER_FORMAT_NAME, // workTimeFactor - NUMBER_FORMAT_NAME, // nonWorkTimeFactor - NUMBER_FORMAT_NAME // valueOfTime - }; - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(Arrays.asList("sentri")); - Set primaryKey = new LinkedHashSet(Arrays.asList("TOURID")); - Map overridingNames = new HashMap(); - overridingNames.put("id", "TOURID"); - exportDataGeneric(outputFileBase, "crossBorder.tour.output.file", false, formats, - floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - overridingNames, null,true); - } - - private void exportCrossBorderTripData(String outputFileBase) - { - addTable(outputFileBase); - String[] formats = { - NUMBER_FORMAT_NAME, // 1 tourID - NUMBER_FORMAT_NAME, // 2 tripID - NUMBER_FORMAT_NAME, // 3 originPurp - NUMBER_FORMAT_NAME, // 4 destPurp - NUMBER_FORMAT_NAME, // 5 originMGRA - NUMBER_FORMAT_NAME, // 6 destinationMGRA - NUMBER_FORMAT_NAME, // 7 originTAZ - NUMBER_FORMAT_NAME, // 8 destinationTAZ - STRING_FORMAT_NAME, // 9 inbound - STRING_FORMAT_NAME, // 10 originIsTourDestination - STRING_FORMAT_NAME, // 11 destinationIsTourDestination - NUMBER_FORMAT_NAME, // 12 period - NUMBER_FORMAT_NAME, // 13 tripMode - NUMBER_FORMAT_NAME, // 14 av_avail - NUMBER_FORMAT_NAME, // 15 boardingTap - NUMBER_FORMAT_NAME, // 16 alightingTap - NUMBER_FORMAT_NAME, // 17 set - NUMBER_FORMAT_NAME, // 18 workTimeFactor - NUMBER_FORMAT_NAME, // 19 nonWorkTimeFactor - NUMBER_FORMAT_NAME // 20 valueOfTime - }; - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("workTimeFactor","nonWorkTimeFactor","valueOfTime")); - Set stringColumns = new HashSet(Arrays.asList("inbound", - "originIsTourDestination", "destinationIsTourDestination")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("tourID", "tripID")); - Map overridingNames = new HashMap(); - overridingNames.put("id", "TOURID"); - exportDataGeneric(outputFileBase, "crossBorder.trip.output.file", false, formats, - floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - overridingNames, new TripStructureDefinition(5, 6, 3, 4, 12, 13, 15, 16, -1, -1, 20, - "CB", 9, true, 20, 17,-1),true); - } - - private void exportVisitorData(String outputTourFileBase, String outputTripFileBase) - { - TableDataSet tourData = exportVisitorTourData(outputTourFileBase); - String tourIdField = "id"; - String partySizeField = "partySize"; - Map tourIdToPartySize = new HashMap(); - int[] ids = tourData.getColumnAsInt(tourIdField); - int[] partySize = tourData.getColumnAsInt(partySizeField); - for (int i = 0; i < ids.length; i++) - tourIdToPartySize.put(ids[i], partySize[i]); - exportVisitorTripData(outputTripFileBase, tourIdToPartySize); - } - - private TableDataSet exportVisitorTourData(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("id", "segment")); - Map overridingNames = new HashMap(); - // overridingNames.put("id","PARTYID"); - return exportDataGeneric(outputFileBase, "visitor.tour.output.file", false, null, - floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - overridingNames, null,false); - } - - private void exportVisitorTripData(String outputFileBase, Map tourIdToPartyMap) - { - addTable(outputFileBase); - String[] formats = { - NUMBER_FORMAT_NAME, // 1 tourID - NUMBER_FORMAT_NAME, // 2 tripID - NUMBER_FORMAT_NAME, // 3 originPurp - NUMBER_FORMAT_NAME, // 4 destPurp - NUMBER_FORMAT_NAME, // 5 originMGRA - NUMBER_FORMAT_NAME, // 6 destinationMGRA - STRING_FORMAT_NAME, // 7 inbound - STRING_FORMAT_NAME, // 8 originIsTourDestination - STRING_FORMAT_NAME, // 9 destinationIsTourDestination - NUMBER_FORMAT_NAME, // 10 period - NUMBER_FORMAT_NAME, // 11 tripMode - NUMBER_FORMAT_NAME, // 12 avAvailable - NUMBER_FORMAT_NAME, // 13 boardingTap - NUMBER_FORMAT_NAME, // 14 alightingTap - NUMBER_FORMAT_NAME, // 15 set - NUMBER_FORMAT_NAME, // 16 valueOfTime - NUMBER_FORMAT_NAME, // 17 partySize (added) - NUMBER_FORMAT_NAME, // 18 micro_walkMode - NUMBER_FORMAT_NAME, // 19 micro_trnAcc - NUMBER_FORMAT_NAME // 20 micro_trnEgr - - }; - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(Arrays.asList("inbound", - "originIsTourDestination", "destinationIsTourDestination")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("tourID", "tripId")); - primaryKey = new LinkedHashSet(Arrays.asList("RECID")); // todo: temporary until bugfix - //JoinData joinData = new JoinData("tourID"); - //joinData.addJoinData(tourIdToPartyMap, FieldType.INT, "partySize"); - exportDataGeneric( - outputFileBase, - "visitor.trip.output.file", - false, - formats, - floatColumns, - stringColumns, - intColumns, - bitColumns, - FieldType.INT, - primaryKey, - new TripStructureDefinition(5, 6, 3, 4, 10, 11, 13, 14, -1, 17, 20, "VISITOR", 7, true, 16,15,-1),false); - //, joinData); - } - - private void exportInternalExternalTripData(String outputFileBase) - { - addTable(outputFileBase); - String[] formats = { - NUMBER_FORMAT_NAME, // 1 hh_id - NUMBER_FORMAT_NAME, // 2 pnum - NUMBER_FORMAT_NAME, // 3 person_id - NUMBER_FORMAT_NAME, // 4 tour_id - NUMBER_FORMAT_NAME, // 5 originMGRA - NUMBER_FORMAT_NAME, // 6 destinationMGRA - NUMBER_FORMAT_NAME, // 7 originTAZ - NUMBER_FORMAT_NAME, // 8 destinationTAZ - STRING_FORMAT_NAME, // 9 inbound - STRING_FORMAT_NAME, // 10 originIsTourDestination - STRING_FORMAT_NAME, // 11 destinationIsTourDestination - NUMBER_FORMAT_NAME, // 12 period - NUMBER_FORMAT_NAME, // 13 tripMode - NUMBER_FORMAT_NAME, // 14 av_avail - NUMBER_FORMAT_NAME, // 15 boardingTap - NUMBER_FORMAT_NAME, // 16 alightingTap - NUMBER_FORMAT_NAME, // 17 set - NUMBER_FORMAT_NAME // 18 value of time - }; - Set intColumns = new HashSet(); - Set floatColumns = new HashSet(Arrays.asList("valueOfTime")); - Set stringColumns = new HashSet(Arrays.asList("inbound", - "originIsTourDestination", "destinationIsTourDestination")); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(); - exportDataGeneric(outputFileBase, "internalExternal.trip.output.file", false, formats, - floatColumns, stringColumns, intColumns, bitColumns, FieldType.INT, primaryKey, - new TripStructureDefinition(5, 6, 12, 13, 15, 16, -1, -1, 18, "IE", "HOME", "EXTERNAL", - 9, true,18,17,-1),false); - } - - private Set getExternalZones() - { - Set externalZones = new LinkedHashSet(); - for (String zone : ((String) properties.get("external.tazs")).trim().split(",")) - externalZones.add(Integer.parseInt(zone.trim())); - return externalZones; - } - - /** - * Export commercial tNCVehicle data. - * - * @param outputFileBase - * @throws IOException - */ - private void exportCommVehData(String outputFileBase) throws IOException - { - addTable(outputFileBase); - Set internalZones = new LinkedHashSet(); - DecimalFormat formatter = new DecimalFormat("#.######"); - - BufferedWriter writer = null; - try - { - writer = new BufferedWriter(new FileWriter(new File(getOutputPath(outputFileBase - + ".csv"))), 1024 * 1024 * 1024); - - CsvRow headerRow = new CsvRow(new String[] {"ORIG_TAZ", "DEST_TAZ", "TOD", - "TRIPS_COMMVEH"}); - writer.write(headerRow.getRow()); - - for (String period : timePeriods) - { - Matrix matrixData = mtxDao.getMatrix("commVehTODTrips", period + " Trips"); - - // This doesn't make sense - if (internalZones.isEmpty()) for (int zone : matrixData.getExternalColumnNumbers()) - internalZones.add(zone); - - for (int i : internalZones) - { - for (int j : internalZones) - { - float value = matrixData.getValueAt(i, j); - if (value > .00001) - { - String[] rowValue = new String[4]; - rowValue[0] = String.valueOf(i); - rowValue[1] = String.valueOf(j); - rowValue[2] = period; - rowValue[3] = formatter.format(value); - CsvRow dataRow = new CsvRow(rowValue); - writer.write(dataRow.getRow()); - } - } - } - } - } finally - { - if (writer != null) writer.close(); - } - } - - /** - * Export commercial tNCVehicle data to OMX Format. - * - * @param outputFileBase - * @throws IOException - */ - private void exportCommVehDataToOmx(String outputFileBase) throws IOException - { - String[] modes = {"Toll","NonToll"}; - - addTable(outputFileBase); - for (String period : timePeriods){ - - Matrix[] matrices = new Matrix[modes.length]; - int counter = 0; - for(String mode : modes){ - - matrices[counter] = mtxDao.getMatrix("commVehTODTrips", period + " " + mode); - ++counter; - } - File outMatrixFile = new File(getOutputPath("commVeh_" + period + ".omx")); - MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX,outMatrixFile); - matrixWriter.writeMatrices(modes,matrices); - } - - } - private void exportExternalInternalTripData(String outputFileBase) - { - addTable(outputFileBase); - Set internalZones = new LinkedHashSet(); - Set externalZones = getExternalZones(); - List cores = Arrays.asList("DAN", "S2N", "S3N", "DAT", "S2T", "S3T"); - Map purposeMap = new HashMap(); - purposeMap.put("WORK", "Wrk"); - purposeMap.put("NONWORK", "Non"); - - Matrix[] matrixData = new Matrix[cores.size()]; - - PrintWriter writer = null; - try - { - writer = getBufferedPrintWriter(getOutputPath(outputFileBase + ".csv")); - - StringBuilder sb = new StringBuilder(); - sb.append("ORIG_TAZ,DEST_TAZ,TOD,PURPOSE"); - for (String core : cores) - sb.append(",").append("TRIPS_").append(core); - writer.println(sb.toString()); - - for (String period : timePeriods) - { - for (String purpose : purposeMap.keySet()) - { - int counter = 0; - for (String core : cores) - matrixData[counter++] = mtxDao.getMatrix("usSd" + purposeMap.get(purpose) - + "_" + period, core); - - if (internalZones.size() == 0) - { // only need to form internal zones once - for (int zone : matrixData[0].getExternalColumnNumbers()) - internalZones.add(zone); - internalZones.removeAll(externalZones); - } - - for (int i : internalZones) - { - for (int e : externalZones) - { - StringBuilder sbie = new StringBuilder(); - StringBuilder sbei = new StringBuilder(); - sbie.append(i).append(",").append(e).append(",").append(period) - .append(",").append(purpose); - sbei.append(e).append(",").append(i).append(",").append(period) - .append(",").append(purpose); - float ie = 0; - float ei = 0; - - for (Matrix matrix : matrixData) - { - float vie = matrix.getValueAt(i, e); - float vei = matrix.getValueAt(e, i); - ie += vie; - ei += vei; - sbie.append(",").append(vie); - sbei.append(",").append(vei); - } - if (ie > 0) writer.println(sbie.toString()); - if (ei > 0) writer.println(sbei.toString()); - } - } - } - } - - } catch (IOException e) - { - throw new RuntimeException(e); - } finally - { - if (writer != null) writer.close(); - } - } - - /** - * Export the external-internal trips to OMX format. Collapse out purposes. - * @param outputFileBase - */ - private void exportExternalInternalTripDataToOMX(String outputFileBase) - { - addTable(outputFileBase); - String[] cores = {"DAN", "S2N", "S3N", "DAT", "S2T", "S3T"}; - String[] purposes = {"Wrk","Non"}; - - Matrix[] outMatrixData = new Matrix[cores.length]; - - for (String period : timePeriods) - { - for(int p = 0; p externalZones = getExternalZones(); - - BufferedWriter writer = null; - MatrixWriter matrixWriter = null; - try - { - if(writeCSV){ - writer = new BufferedWriter(new FileWriter(new File(getOutputPath(outputFileBase - + ".csv"))), 1024 * 1024 * 1024); - - CsvRow headerRow = new CsvRow(new String[] {"ORIG_TAZ", "DEST_TAZ", "TRIPS_EE"}); - writer.write(headerRow.getRow()); - }else{ - matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, new File(getOutputPath(outputFileBase + ".omx"))); - } - - Matrix m = mtxDao.getMatrix("externalExternalTrips", "Trips"); - - if(writeCSV){ - for (int o : externalZones) - { - for (int d : externalZones) - { - String[] values = new String[3]; - values[0] = String.valueOf(o); - values[1] = String.valueOf(d); - values[2] = String.valueOf(m.getValueAt(o, d)); - CsvRow dataRow = new CsvRow(values); - writer.write(dataRow.getRow()); - } - } - }else{ - matrixWriter.writeMatrix(m); - } - } finally - { - if (writer != null) writer.close(); - } - - } - - /** - * A private helper class to organize skims - * - * @author joel.freedman - * - */ - private class AutoSkimSet{ - - String fileName; - String[] skimNames; - - AutoSkimSet(String fileName, String[] skimNames){ - this.fileName = fileName; - this.skimNames = skimNames; - } - } - - /** - * Return a map containing a number of elements where key is the name of the skim file and - * value is the name of a matrix core in the skim file. The map includes length and time for - * "free" path skims and length, time and toll for toll skims. - * - * @return The map. - */ - private HashMap getVehicleSkimFileCoreNameMapping() - { - HashMap map = new HashMap(); - - String[] votBins = {"L","M","H"}; - - for(int i = 1; i< votBins.length;++i){ - - // DA Non-Toll - AutoSkimSet SOVGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_SOVGP"+votBins[i]+"_DIST", - TOD_TOKEN+"_SOVGP"+votBins[i]+"_TIME", - TOD_TOKEN+"_SOVGP"+votBins[i]+"_REL"}); - map.put("SOVGP"+votBins[i], SOVGP); - - // DA Toll - AutoSkimSet SOVTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_DIST", - TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_TIME", - TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_TOLLCOST", - TOD_TOKEN+"_SOVTOLL"+votBins[i]+"_REL"}); - map.put("SOTOLL"+votBins[i], SOVTOLL); - - // S2 Non-Toll - AutoSkimSet HOV2HOV = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_DIST", - TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_TIME", - TOD_TOKEN+"_HOV2HOV"+votBins[i]+"_REL"}); - map.put("HOV2HOV"+votBins[i], HOV2HOV); - - // S2 Toll - AutoSkimSet HOV2TOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_DIST", - TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_TIME", - TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_TOLLCOST", - TOD_TOKEN+"_HOV2TOLL"+votBins[i]+"_REL"}); - map.put("HOV2TOLL"+votBins[i], HOV2TOLL); - - // S3+ Non-Toll - AutoSkimSet HOV3HOV = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_DIST", - TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_TIME", - TOD_TOKEN+"_HOV3HOV"+votBins[i]+"_REL"}); - map.put("HOV3HOV"+votBins[i], HOV3HOV); - - // S3+ Toll - AutoSkimSet HOV3TOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_DIST", - TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_TIME", - TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_TOLLCOST", - TOD_TOKEN+"_HOV3TOLL"+votBins[i]+"_REL"}); - map.put("HOV3TOLL"+votBins[i], HOV3TOLL); - - } - - // Light Truck GP - AutoSkimSet TRKLGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKLGP_DIST", - TOD_TOKEN+"_TRKLGP_TIME"}); - map.put("TRKLGP", TRKLGP); - - // Light Truck Toll - AutoSkimSet TRKLTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKLTOLL_DIST", - TOD_TOKEN+"_TRKLTOLL_TIME", - TOD_TOKEN+"_TRKLTOLL_TOLLCOST"}); - map.put("TRKLTOLL", TRKLTOLL); - - - // Medium Truck GP - AutoSkimSet TRKMGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKMGP_DIST", - TOD_TOKEN+"_TRKMGP_TIME"}); - map.put("TRKMGP", TRKMGP); - - // Medium Truck Toll - AutoSkimSet TRKMTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKMTOLL_DIST", - TOD_TOKEN+"_TRKMTOLL_TIME", - TOD_TOKEN+"_TRKMTOLL_TOLLCOST"}); - map.put("TRKMTOLL", TRKMTOLL); - - // Heavy Truck GP - AutoSkimSet TRKHGP = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKHGP_DIST", - TOD_TOKEN+"_TRKHGP_TIME"}); - map.put("TRKHGP", TRKHGP); - - // Heavy Truck Toll - AutoSkimSet TRKHTOLL = new AutoSkimSet("traffic_skims_" + TOD_TOKEN, new String[] { - TOD_TOKEN+"_TRKHTOLL_DIST", - TOD_TOKEN+"_TRKHTOLL_TIME", - TOD_TOKEN+"_TRKHTOLL_TOLLCOST"}); - map.put("TRKHTOLL", TRKHTOLL); - - - return map; - } - - /** - * Export auto skims to the directory using both csv and omx formats. The CSV file will be - * written if writeCSV is true. Otherwise OMX files will be written. The OMX files will also - * contain an auto operating cost matrix. - * - * @param outputFileBase The name of output csv file to write to the reports directory. - */ - private void exportAutoSkims(String outputFileBase) - { - addTable(outputFileBase); - String[] includedTimePeriods = getTimePeriodsForSkims(); - Set internalZones = new LinkedHashSet(); - String path = properties.getProperty("report.path"); - - BlockingQueue queue = new LinkedBlockingQueue(); - try - { - - Map vehicleSkimCores = getVehicleSkimFileCoreNameMapping(); - - boolean first = true; - - ArrayList modeNames = new ArrayList(); - - for (String period : includedTimePeriods) - { - Map lengthMatrix = new LinkedHashMap(); - Map timeMatrix = new LinkedHashMap(); - Map tollMatrix = new LinkedHashMap(); - Map stdMatrix = new LinkedHashMap(); - - - //iterate through the auto modes - for (String key : vehicleSkimCores.keySet()) - { - // String name = vehicleSkimFiles.get(key); - AutoSkimSet skimSet = vehicleSkimCores.get(key); - String skimFileName = skimSet.fileName; - String[] inputMatrixNames = skimSet.skimNames; - - // the first skim is always distance. Remove it to get the name of the mode. - modeNames.add(key+"_"+TOD_TOKEN); //store all the modes - - int stdMatrixNumber=-1; - int tollMatrixNumber=-1; - - //need to replace the TOD token with the period name for matrices to output to OMX - String[] outCores = new String[inputMatrixNames.length+1]; - for(int i =0; i < (outCores.length-1);++i){ - outCores[i] = inputMatrixNames[i].replace(TOD_TOKEN, period); - if(inputMatrixNames[i].contains("REL")) - stdMatrixNumber=i; - if(inputMatrixNames[i].contains("TOLLCOST")) - tollMatrixNumber=i; - } - - skimFileName = skimFileName.replace(TOD_TOKEN,period); - Matrix length = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[0].replace(TOD_TOKEN, period)); - Matrix time = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[1].replace(TOD_TOKEN, period)); - Matrix aoc = length.multiply(autoOperatingCost); - - String aocName = outCores[0].replace("_DIST", "_AOC"); - outCores[outCores.length-1] = aocName; - - String outputFileName = path+key+"_"+period+".omx"; - - MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, new File(outputFileName)); - - Matrix[] matrices = new Matrix[inputMatrixNames.length+1]; - matrices[0] = length; - matrices[1] = time; - - lengthMatrix.put(inputMatrixNames[0],length); - timeMatrix.put(inputMatrixNames[1], time); - - - int matrixNumber=2; - if(stdMatrixNumber>-1){ - Matrix std = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[stdMatrixNumber].replace(TOD_TOKEN, period)); - matrices[matrixNumber]= std; - stdMatrix.put(inputMatrixNames[matrixNumber], std); - ++matrixNumber; - } - if(tollMatrixNumber>-1){ - Matrix cost = mtxDao.getMatrix(skimFileName+".omx", inputMatrixNames[tollMatrixNumber].replace(TOD_TOKEN, period)); - matrices[matrixNumber]= cost; - - ++matrixNumber; - } - - matrices[matrixNumber] = aoc; - - LOGGER.info("Writing "+outCores.length+" skims to file "+outputFileName); - matrixWriter.writeMatrices(outCores, matrices); - - if(writeCSV){ - if (internalZones.size() == 0) - { - boolean f = true; - for (int zone : lengthMatrix.get(inputMatrixNames[0]).getExternalColumnNumbers()) - { - if (f) - { - f = false; - continue; - } - internalZones.add(zone); - } - } - - // put data into arrays for faster access - Matrix[] orderedData = new Matrix[lengthMatrix.size() + timeMatrix.size() - + stdMatrix.size() + tollMatrix.size()]; - int counter = 0; - for (String mode : modeNames) - { - orderedData[counter++] = lengthMatrix.get(mode); - orderedData[counter++] = timeMatrix.get(mode); - orderedData[counter++] = stdMatrix.get(mode); - if (tollMatrix.containsKey(mode)) - orderedData[counter++] = tollMatrix.get(mode); - } - - if (first) - { - List header = new ArrayList(); - header.add("ORIG_TAZ"); - header.add("DEST_TAZ"); - header.add("TOD"); - - for (String modeName : modeNames) - { - header.add("DIST_" + modeName); - header.add("TIME_" + modeName); - header.add("STD_TIME_" + modeName); - if (tollMatrix.containsKey(modeName)) - { - header.add("COST_" + modeName); - } - } - - CsvWriterThread writerThread = new CsvWriterThread(queue, new File( - getOutputPath(outputFileBase + ".csv")), - header.toArray(new String[header.size()])); - new Thread(writerThread).start(); - first = false; - } - - int rowSize = 3 + orderedData.length; - - for (int i : internalZones) - { - for (int j : internalZones) - { - String[] values = new String[rowSize]; - values[0] = String.valueOf(i); - values[1] = String.valueOf(j); - values[2] = period; - int position = 3; - for (Matrix matrix : orderedData) - values[position++] = DoubleFormatUtil.formatDouble( - matrix.getValueAt(i, j), 4, 4); - queue.add(new CsvRow(values)); - } - } - } - } - } - } finally - { - queue.add(CsvWriterThread.POISON_PILL); - } - } - - private Map getTransitSkimFileNameMapping() - { - Map map = new LinkedHashMap(); - // map.put("implocl_" + TOD_TOKEN + "o", "LOCAL_TRANSIT"); - map.put("impprem_" + TOD_TOKEN + "o", "PREMIUM_TRANSIT"); - return map; - } - - private String getTransitSkimFileFareCoreName() - { - return "Fare"; - } - - private Map getTransitSkimFileInVehicleTimeCoreNameMapping() - { // distance,time,cost - Map map = new LinkedHashMap(); - map.put("impprem_" + TOD_TOKEN + "o", new String[] {"IVT:CR", "IVT:LR", "IVT:BRT", - "IVT:EXP", "IVT:LB"}); - return map; - } - - private String[] getTimePeriodsForSkims() - { - return IExporter.TOD_TOKENS; - } - - /** - * This method reads the transit skims and exports them to OMX format. It will also write - * csv file of skim values if the writeCSVSkims attribute is set to true. - * - * @param outputFileBase - */ - private void exportTransitSkims(String outputFileBase) - { - addTable(outputFileBase); - String[] includedTimePeriods = getTimePeriodsForSkims(); - - Set internalZones = new LinkedHashSet(); - - BlockingQueue queue = new LinkedBlockingQueue(); - try - { - Map transitSkimFiles = getTransitSkimFileNameMapping(); - Map transitSkimTimeCores = getTransitSkimFileInVehicleTimeCoreNameMapping(); - String fareCore = getTransitSkimFileFareCoreName(); - String initialWaitCore = "Initial Wait Time"; - String transferTimeCore = "Transfer Wait Time"; - String walkTimeCore = "Walk Time"; - Set modeNames = new LinkedHashSet(); - for (String n : transitSkimFiles.keySet()) - modeNames.add(transitSkimFiles.get(n)); - boolean first = true; - int numOfColumns = 3 + 5 * modeNames.size(); - for (String period : includedTimePeriods) - { - Map timeMatrix = new LinkedHashMap(); - Map fareMatrix = new LinkedHashMap(); - Map initialMatrix = new LinkedHashMap(); - Map transferMatrix = new LinkedHashMap(); - Map walkTimeMatrix = new LinkedHashMap(); - - for (String key : transitSkimFiles.keySet()) - { - String name = transitSkimFiles.get(key); - String[] timeCores = transitSkimTimeCores.get(key); - String file = key.replace(TOD_TOKEN, period); - Matrix[] timeMatrices = new Matrix[timeCores.length]; - for (int i = 0; i < timeCores.length; i++) - timeMatrices[i] = mtxDao.getMatrix(file, - timeCores[i].replace(TOD_TOKEN, period)); - timeMatrix.put(name, timeMatrices); - fareMatrix.put(name, - mtxDao.getMatrix(file, fareCore.replace(TOD_TOKEN, period))); - initialMatrix.put(name, mtxDao.getMatrix(file, initialWaitCore)); - transferMatrix.put(name, mtxDao.getMatrix(file, transferTimeCore)); - walkTimeMatrix.put(name, mtxDao.getMatrix(file, walkTimeCore)); - if (internalZones.size() == 0) - { - boolean f = true; - for (int zone : fareMatrix.get(name).getExternalColumnNumbers()) - { - if (f) - { - f = false; - continue; - } - internalZones.add(zone); - } - } - } - - // put data into arrays for faster access - Matrix[][] orderedTimeData = new Matrix[timeMatrix.size()][]; - Matrix[] fareData = new Matrix[orderedTimeData.length]; - Matrix[] initialWaitData = new Matrix[orderedTimeData.length]; - Matrix[] transferTimeData = new Matrix[orderedTimeData.length]; - Matrix[] walkTimeData = new Matrix[orderedTimeData.length]; - - int counter = 0; - for (String mode : modeNames) - { - orderedTimeData[counter] = timeMatrix.get(mode); - fareData[counter] = fareMatrix.get(mode); - initialWaitData[counter] = initialMatrix.get(mode); - transferTimeData[counter] = transferMatrix.get(mode); - walkTimeData[counter++] = walkTimeMatrix.get(mode); - } - - if (first) - { - String[] header = new String[numOfColumns]; - - header[0] = "ORIG_TAP"; - header[1] = "DEST_TAP"; - header[2] = "TOD"; - int column = 3; - - for (String modeName : modeNames) - { - header[column++] = "TIME_INIT_WAIT_" + modeName; - header[column++] = "TIME_IVT_TIME_" + modeName; - header[column++] = "TIME_WALK_TIME_" + modeName; - header[column++] = "TIME_TRANSFER_TIME_" + modeName; - header[column++] = "FARE_" + modeName; - } - - CsvWriterThread writerThread = new CsvWriterThread(queue, new File( - getOutputPath(outputFileBase + ".csv")), header); - new Thread(writerThread).start(); - - first = false; - } - - for (int i : internalZones) - { - for (int j : internalZones) - { - String[] values = new String[numOfColumns]; - values[0] = String.valueOf(i); - values[1] = String.valueOf(j); - values[2] = period; - - int column = 3; - float runningTotal = 0.0f; - - for (int m = 0; m < orderedTimeData.length; m++) - { - float time = 0.0f; - float initTime = initialWaitData[m].getValueAt(i, j); - for (Matrix tm : orderedTimeData[m]) - time += tm.getValueAt(i, j); - float walkTime = walkTimeData[m].getValueAt(i, j); - float transferTime = transferTimeData[m].getValueAt(i, j); - float fare = fareData[m].getValueAt(i, j); - runningTotal += fare + time; - values[column++] = DoubleFormatUtil.formatDouble(initTime, 4, 4); - values[column++] = DoubleFormatUtil.formatDouble(time, 4, 4); - values[column++] = DoubleFormatUtil.formatDouble(walkTime, 4, 4); - values[column++] = DoubleFormatUtil.formatDouble(transferTime, 4, 4); - values[column++] = DoubleFormatUtil.formatDouble(fare, 2, 2); - } - if (runningTotal > 0.0f) queue.add(new CsvRow(values)); - } - } - } - - } finally - { - queue.add(CsvWriterThread.POISON_PILL); - } - } - - private void exportDefinitions(String outputFileBase) - { - addTable(outputFileBase); - Map tripPurposes = new LinkedHashMap(); - Map modes = new LinkedHashMap(); - Map ejCategories = new LinkedHashMap(); - - PrintWriter writer = null; - try - { - writer = getBufferedPrintWriter(getOutputPath(outputFileBase + ".csv")); - writer.println("type,code,description"); - writer.println("nothing,placeholder,this describes nothing"); - for (String tripPurpose : tripPurposes.keySet()) - writer.println("trip_purpose," + tripPurpose + "," + tripPurposes.get(tripPurpose)); - for (String mode : modes.keySet()) - writer.println("mode," + mode + "," + modes.get(mode)); - for (String ejCategory : ejCategories.keySet()) - writer.println("ej_category," + ejCategory + "," + ejCategories.get(ejCategory)); - } catch (IOException e) - { - throw new RuntimeException(e); - } finally - { - if (writer != null) writer.close(); - } - } - - private void exportPnrVehicleData(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(Arrays.asList("TAP")); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("TAP")); - exportDataGeneric(outputFileBase, "Results.PNRFile", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private void exportCbdVehicleData(String outputFileBase) - { - addTable(outputFileBase); - Set intColumns = new HashSet(Arrays.asList("MGRA")); - Set floatColumns = new HashSet(); - Set stringColumns = new HashSet(); - Set bitColumns = new HashSet(); - Set primaryKey = new LinkedHashSet(Arrays.asList("MGRA")); - exportDataGeneric(outputFileBase, "Results.CBDFile", false, null, floatColumns, - stringColumns, intColumns, bitColumns, FieldType.FLOAT, primaryKey, null,false); - } - - private static enum FieldType - { - INT, FLOAT, STRING, BIT - } - - private final class TripStructureDefinition - { - private final int originMgraColumn; - private final int destMgraColumn; - private final int originPurposeColumn; - private final int destinationPurposeColumn; - private final int todColumn; - private final int modeColumn; - private final int boardTapColumn; - private final int alightTapColumn; - - private final int parkingMazColumn; - - private final String homeName; - private final String destinationName; - private final int inboundColumn; - private final boolean booleanIndicatorVariables; - private final int valueOfTimeColumn; - private final int setColumn; - private final int transponderOwnershipColumn; - - private TripStructureDefinition(int originMgraColumn, int destMgraColumn, - int originPurposeColumn, int destinationPurposeColumn, int todColumn, - int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, - int tripTimeColumn, int outVehicleTimeColumn, int tripDistanceColumn, - int tripCostColumn, int tripPurposeNameColumn, int tripModeNameColumn, - int recIdColumn, int boardTazColumn, int alightTazColumn, String tripType, - String homeName, String destinationName, int inboundColumn, - boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) - { - this.originMgraColumn = originMgraColumn; - this.destMgraColumn = destMgraColumn; - this.originPurposeColumn = originPurposeColumn; - this.destinationPurposeColumn = destinationPurposeColumn; - this.todColumn = todColumn; - this.modeColumn = modeColumn; - this.boardTapColumn = boardTapColumn; - this.alightTapColumn = alightTapColumn; - this.parkingMazColumn = parkingMazColumn; - this.homeName = homeName; - this.destinationName = destinationName; - this.inboundColumn = inboundColumn; - - this.booleanIndicatorVariables = booleanIndicatorVariables; - this.valueOfTimeColumn = valueOfTimeColumn; - this.setColumn = setColumn; - this.transponderOwnershipColumn = transponderOwnershipColumn; - } - - private TripStructureDefinition(int originMgraColumn, int destMgraColumn, - int originPurposeColumn, int destinationPurposeColumn, int todColumn, - int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, - int columnCount, String tripType, int inboundColumn, - boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) - { - this(originMgraColumn, destMgraColumn, originPurposeColumn, destinationPurposeColumn, - todColumn, modeColumn, boardTapColumn, alightTapColumn, parkingMazColumn, partySizeColumn, - columnCount + 1, columnCount + 2, columnCount + 3, columnCount + 4, - columnCount + 5, columnCount + 6, columnCount + 7, columnCount + 8, - columnCount + 9, tripType, "", "", inboundColumn, booleanIndicatorVariables, valueOfTimeColumn,setColumn,transponderOwnershipColumn); - } - - private TripStructureDefinition(int originMgraColumn, int destMgraColumn, int todColumn, - int modeColumn, int boardTapColumn, int alightTapColumn, int parkingMazColumn, int partySizeColumn, - int columnCount, String tripType, String homeName, String destinationName, - int inboundColumn, boolean booleanIndicatorVariables, int valueOfTimeColumn, int setColumn, int transponderOwnershipColumn) - { - this(originMgraColumn, destMgraColumn, -1, -1, todColumn, modeColumn, boardTapColumn, - alightTapColumn, parkingMazColumn, partySizeColumn, columnCount + 1, columnCount + 2, - columnCount + 3, columnCount + 4, columnCount + 5, columnCount + 6, - columnCount + 7, columnCount + 8, columnCount + 9, tripType, homeName, - destinationName, inboundColumn, booleanIndicatorVariables, valueOfTimeColumn,setColumn,transponderOwnershipColumn); - } - } - - public static void main(String... args) throws Exception - { - String propertiesFile = null; - propertiesFile = args[0]; - - Properties properties = new Properties(); - properties.load(new FileInputStream("conf/sandag_abm.properties")); - HashMap pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - - int feedbackIteration = Integer.valueOf(properties.getProperty("Report.iteration").trim()); - - List definedTables = new ArrayList(); - for (String table : properties.getProperty("Report.tables").trim().split(",")) - definedTables.add(table.trim().toLowerCase()); - - String path = ClassLoader.getSystemResource("").getPath(); - path = path.substring(1, path.length() - 2); - String appPath = path.substring(0, path.lastIndexOf("/")); - - for (Object key : properties.keySet()) - { - String value = (String) properties.get(key); - properties.setProperty((String) key, - value.replace(PROJECT_PATH_PROPERTY_TOKEN, appPath)); - } - - OMXMatrixDao mtxDao = new OMXMatrixDao(properties); - - DataExporter dataExporter = new DataExporter(properties, mtxDao, appPath, feedbackIteration); - dataExporter.startMatrixServer(pMap); - - if (definedTables.contains("accessibilities")) - dataExporter.exportAccessibilities("accessibilities"); - if (definedTables.contains("mgra")) dataExporter.exportMazData("mgra"); - if (definedTables.contains("taz")) dataExporter.exportTazData("taz"); - if (definedTables.contains("tap")) dataExporter.exportTapData("tap"); - if (definedTables.contains("mgratotap")) dataExporter.exportMgraToTapData("mgratotap"); - if (definedTables.contains("mgratomgra")) dataExporter.exportMgraToMgraData("mgratomgra"); - if (definedTables.contains("taztotap")) dataExporter.exportTazToTapData("taztotap"); - if (definedTables.contains("hhdata")) dataExporter.exportHouseholdData("hhdata"); - if (definedTables.contains("persondata")) dataExporter.exportPersonData("persondata"); - if (definedTables.contains("wslocation")) - dataExporter.exportWorkSchoolLocation("wslocation"); - if (definedTables.contains("synhh")) dataExporter.exportSyntheticHouseholdData("synhh"); - if (definedTables.contains("synperson")) - dataExporter.exportSyntheticPersonData("synperson"); - if (definedTables.contains("indivtours")) dataExporter.exportIndivToursData("indivtours"); - if (definedTables.contains("jointtours")) dataExporter.exportJointToursData("jointtours"); - if (definedTables.contains("indivtrips")) dataExporter.exportIndivTripData("indivtrips"); - if (definedTables.contains("jointtrips")) dataExporter.exportJointTripData("jointtrips"); - if (definedTables.contains("airporttripssan")) - dataExporter.exportAirportTripsSAN("airporttripssan"); - if (definedTables.contains("airporttripscbx")) - dataExporter.exportAirportTripsCBX("airporttripscbx"); - if (definedTables.contains("cbtours")) dataExporter.exportCrossBorderTourData("cbtours"); - if (definedTables.contains("cbtrips")) dataExporter.exportCrossBorderTripData("cbtrips"); - if (definedTables.contains("visitortours") && definedTables.contains("visitortrips")) - dataExporter.exportVisitorData("visitortours", "visitortrips"); - if (definedTables.contains("ietrip")) - dataExporter.exportInternalExternalTripData("ietrip"); - if (definedTables.contains("commtrip")){ - CVMExporter cvmExporter = new CVMExporter(properties,mtxDao); - cvmExporter.export(); - CVMScaler cvmScaler = new CVMScaler(properties); - cvmScaler.scale(); - } - - - if (definedTables.contains("trucktrip")) - { - if(dataExporter.writeCSV){ - IExporter truckExporter = new TruckCsvExporter(properties, mtxDao, "trucktrip"); - truckExporter.export(); - }else{ - IExporter truckExporter = new TruckOmxExporter(properties, mtxDao, "trucktrip"); - truckExporter.export(); - } - } - if (definedTables.contains("eetrip")) - dataExporter.exportExternalExternalTripData("eetrip"); - - if (definedTables.contains("eitrip")) - if(dataExporter.writeCSV) - dataExporter.exportExternalInternalTripData("eitrip"); - else - dataExporter.exportExternalInternalTripDataToOMX("eitrip"); - - if (definedTables.contains("tazskim")) dataExporter.exportAutoSkims("tazskim"); - if (definedTables.contains("tapskim")) dataExporter.exportTransitSkims("tapskim"); - if (definedTables.contains("definition")) dataExporter.exportDefinitions("definition"); - if (definedTables.contains("pnrvehicles")) - dataExporter.exportPnrVehicleData("pnrvehicles"); - if (definedTables.contains("cbdvehicles")) - dataExporter.exportCbdVehicleData("cbdvehicles"); - } - - private void startMatrixServer(HashMap properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - LOGGER.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - LOGGER.error("could not connect to matrix server"); - LOGGER.info("Running Data Exporter with internal matrix class"); - // throw new RuntimeException(e); - - } - - } - - /** - * Startup a connection to the matrix manager. - * - * @param serverAddress - * @param serverPort - * @param mt - * @return - */ - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - try - { - // create the concrete data server object - matrixServer.start32BitMatrixIoServer(mt); - } catch (RuntimeException e) - { - matrixServer.stop32BitMatrixIoServer(); - LOGGER.error( - "RuntimeException caught making remote method call to start 32 bit mitrix in remote MatrixDataServer.", - e); - } - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - LOGGER.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - matrixServer.stop32BitMatrixIoServer(); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - LOGGER.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - matrixServer.stop32BitMatrixIoServer(); - throw new RuntimeException(); - } - - return matrixServer; - - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java deleted file mode 100644 index 19f2c7d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/DoubleFormatUtil.java +++ /dev/null @@ -1,484 +0,0 @@ -package org.sandag.abm.reporting; -/* - * Licensed to the Apache Software Foundation (ASF) under one or more - * contributor license agreements. See the NOTICE file distributed with this - * work for additional information regarding copyright ownership. The ASF - * licenses this file to You 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. - */ - -/* $Id$ */ - -/** - * This class implements fast, thread-safe format of a double value with a given - * number of decimal digits. - *

    - * The contract for the format methods is this one: if the source is greater - * than or equal to 1 (in absolute value), use the decimals parameter to define - * the number of decimal digits; else, use the precision parameter to define the - * number of decimal digits. - *

    - * A few examples (consider decimals being 4 and precision being 8): - *

      - *
    • 0.0 should be rendered as "0" - *
    • 0.1 should be rendered as "0.1" - *
    • 1234.1 should be rendered as "1234.1" - *
    • 1234.1234567 should be rendered as "1234.1235" (note the trailing 5! - * Rounding!) - *
    • 1234.00001 should be rendered as "1234" - *
    • 0.00001 should be rendered as "0.00001" (here you see the effect of the - * "precision" parameter) - *
    • 0.00000001 should be rendered as "0.00000001" - *
    • 0.000000001 should be rendered as "0" - *
    - * - * Originally authored by Julien Aymé. - */ -public final class DoubleFormatUtil -{ - - private DoubleFormatUtil() - { - } - - public static String formatDouble(double source, int decimals, int precision) - { - StringBuffer target = new StringBuffer(); - int scale = (Math.abs(source) >= 1.0) ? decimals : precision; - if (tooManyDigitsUsed(source, scale) || tooCloseToRound(source, scale)) - { - formatDoublePrecise(source, decimals, precision, target); - } else - { - formatDoubleFast(source, decimals, precision, target); - } - - return target.toString(); - } - - /** - * Rounds the given source value at the given precision and writes the - * rounded value into the given target - * - * @param source - * the source value to round - * @param decimals - * the decimals to round at (use if abs(source) ≥ 1.0) - * @param precision - * the precision to round at (use if abs(source) < 1.0) - * @param target - * the buffer to write to - */ - public static void formatDouble(double source, int decimals, int precision, StringBuffer target) - { - int scale = (Math.abs(source) >= 1.0) ? decimals : precision; - if (tooManyDigitsUsed(source, scale) || tooCloseToRound(source, scale)) - { - formatDoublePrecise(source, decimals, precision, target); - } else - { - formatDoubleFast(source, decimals, precision, target); - } - } - - /** - * Rounds the given source value at the given precision and writes the - * rounded value into the given target - *

    - * This method internally uses the String representation of the source - * value, in order to avoid any double precision computation error. - * - * @param source - * the source value to round - * @param decimals - * the decimals to round at (use if abs(source) ≥ 1.0) - * @param precision - * the precision to round at (use if abs(source) < 1.0) - * @param target - * the buffer to write to - */ - public static void formatDoublePrecise(double source, int decimals, int precision, - StringBuffer target) - { - if (isRoundedToZero(source, decimals, precision)) - { - // Will always be rounded to 0 - target.append('0'); - return; - } else if (Double.isNaN(source) || Double.isInfinite(source)) - { - // Cannot be formated - target.append(Double.toString(source)); - return; - } - - boolean negative = source < 0.0; - if (negative) - { - source = -source; - // Done once and for all - target.append('-'); - } - int scale = (source >= 1.0) ? decimals : precision; - - // The only way to format precisely the double is to use the String - // representation of the double, and then to do mathematical integer - // operation on it. - String s = Double.toString(source); - if (source >= 1e-3 && source < 1e7) - { - // Plain representation of double: "intPart.decimalPart" - int dot = s.indexOf('.'); - String decS = s.substring(dot + 1); - int decLength = decS.length(); - if (scale >= decLength) - { - if ("0".equals(decS)) - { - // source is a mathematical integer - target.append(s.substring(0, dot)); - } else - { - target.append(s); - // Remove trailing zeroes - for (int l = target.length() - 1; l >= 0 && target.charAt(l) == '0'; l--) - { - target.setLength(l); - } - } - return; - } else if (scale + 1 < decLength) - { - // ignore unnecessary digits - decLength = scale + 1; - decS = decS.substring(0, decLength); - } - long intP = Long.parseLong(s.substring(0, dot)); - long decP = Long.parseLong(decS); - format(target, scale, intP, decP); - } else - { - // Scientific representation of double: "x.xxxxxEyyy" - int dot = s.indexOf('.'); - assert dot >= 0; - int exp = s.indexOf('E'); - assert exp >= 0; - int exposant = Integer.parseInt(s.substring(exp + 1)); - String intS = s.substring(0, dot); - String decS = s.substring(dot + 1, exp); - int decLength = decS.length(); - if (exposant >= 0) - { - int digits = decLength - exposant; - if (digits <= 0) - { - // no decimal part, - // no rounding involved - target.append(intS); - target.append(decS); - for (int i = -digits; i > 0; i--) - { - target.append('0'); - } - } else if (digits <= scale) - { - // decimal part precision is lower than scale, - // no rounding involved - target.append(intS); - target.append(decS.substring(0, exposant)); - target.append('.'); - target.append(decS.substring(exposant)); - } else - { - // decimalDigits > scale, - // Rounding involved - long intP = Long.parseLong(intS) * tenPow(exposant) - + Long.parseLong(decS.substring(0, exposant)); - long decP = Long.parseLong(decS.substring(exposant, exposant + scale + 1)); - format(target, scale, intP, decP); - } - } else - { - // Only a decimal part is supplied - exposant = -exposant; - int digits = scale - exposant + 1; - if (digits < 0) - { - target.append('0'); - } else if (digits == 0) - { - long decP = Long.parseLong(intS); - format(target, scale, 0L, decP); - } else if (decLength < digits) - { - long decP = Long.parseLong(intS) * tenPow(decLength + 1) + Long.parseLong(decS) - * 10; - format(target, exposant + decLength, 0L, decP); - } else - { - long subDecP = Long.parseLong(decS.substring(0, digits)); - long decP = Long.parseLong(intS) * tenPow(digits) + subDecP; - format(target, scale, 0L, decP); - } - } - } - } - - /** - * Returns true if the given source value will be rounded to zero - * - * @param source - * the source value to round - * @param decimals - * the decimals to round at (use if abs(source) ≥ 1.0) - * @param precision - * the precision to round at (use if abs(source) < 1.0) - * @return true if the source value will be rounded to zero - */ - private static boolean isRoundedToZero(double source, int decimals, int precision) - { - // Use 4.999999999999999 instead of 5 since in some cases, 5.0 / 1eN > - // 5e-N (e.g. for N = 37, 42, 45, 66, ...) - return source == 0.0 - || Math.abs(source) < 4.999999999999999 / tenPowDouble(Math - .max(decimals, precision) + 1); - } - - /** - * Most used power of ten (to avoid the cost of Math.pow(10, n) - */ - private static final long[] POWERS_OF_TEN_LONG = new long[19]; - private static final double[] POWERS_OF_TEN_DOUBLE = new double[30]; - static - { - POWERS_OF_TEN_LONG[0] = 1L; - for (int i = 1; i < POWERS_OF_TEN_LONG.length; i++) - { - POWERS_OF_TEN_LONG[i] = POWERS_OF_TEN_LONG[i - 1] * 10L; - } - for (int i = 0; i < POWERS_OF_TEN_DOUBLE.length; i++) - { - POWERS_OF_TEN_DOUBLE[i] = Double.parseDouble("1e" + i); - } - } - - /** - * Returns ten to the power of n - * - * @param n - * the nth power of ten to get - * @return ten to the power of n - */ - public static long tenPow(int n) - { - assert n >= 0; - return n < POWERS_OF_TEN_LONG.length ? POWERS_OF_TEN_LONG[n] : (long) Math.pow(10, n); - } - - private static double tenPowDouble(int n) - { - assert n >= 0; - return n < POWERS_OF_TEN_DOUBLE.length ? POWERS_OF_TEN_DOUBLE[n] : Math.pow(10, n); - } - - /** - * Helper method to do the custom rounding used within formatDoublePrecise - * - * @param target - * the buffer to write to - * @param scale - * the expected rounding scale - * @param intP - * the source integer part - * @param decP - * the source decimal part, truncated to scale + 1 digit - */ - private static void format(StringBuffer target, int scale, long intP, long decP) - { - if (decP != 0L) - { - // decP is the decimal part of source, truncated to scale + 1 digit. - // Custom rounding: add 5 - decP += 5L; - decP /= 10L; - if (decP >= tenPowDouble(scale)) - { - intP++; - decP -= tenPow(scale); - } - if (decP != 0L) - { - // Remove trailing zeroes - while (decP % 10L == 0L) - { - decP = decP / 10L; - scale--; - } - } - } - target.append(intP); - if (decP != 0L) - { - target.append('.'); - // Use tenPow instead of tenPowDouble for scale below 18, - // since the casting of decP to double may cause some imprecisions: - // E.g. for decP = 9999999999999999L and scale = 17, - // decP < tenPow(16) while (double) decP == tenPowDouble(16) - while (scale > 0 - && (scale > 18 ? decP < tenPowDouble(--scale) : decP < tenPow(--scale))) - { - // Insert leading zeroes - target.append('0'); - } - target.append(decP); - } - } - - /** - * Rounds the given source value at the given precision and writes the - * rounded value into the given target - *

    - * This method internally uses double precision computation and rounding, so - * the result may not be accurate (see formatDouble method for conditions). - * - * @param source - * the source value to round - * @param decimals - * the decimals to round at (use if abs(source) ≥ 1.0) - * @param precision - * the precision to round at (use if abs(source) < 1.0) - * @param target - * the buffer to write to - */ - public static void formatDoubleFast(double source, int decimals, int precision, - StringBuffer target) - { - if (isRoundedToZero(source, decimals, precision)) - { - // Will always be rounded to 0 - target.append('0'); - return; - } else if (Double.isNaN(source) || Double.isInfinite(source)) - { - // Cannot be formated - target.append(Double.toString(source)); - return; - } - - boolean isPositive = source >= 0.0; - source = Math.abs(source); - int scale = (source >= 1.0) ? decimals : precision; - - long intPart = (long) Math.floor(source); - double tenScale = tenPowDouble(scale); - double fracUnroundedPart = (source - intPart) * tenScale; - long fracPart = Math.round(fracUnroundedPart); - if (fracPart >= tenScale) - { - intPart++; - fracPart = Math.round(fracPart - tenScale); - } - if (fracPart != 0L) - { - // Remove trailing zeroes - while (fracPart % 10L == 0L) - { - fracPart = fracPart / 10L; - scale--; - } - } - - if (intPart != 0L || fracPart != 0L) - { - // non-zero value - if (!isPositive) - { - // negative value, insert sign - target.append('-'); - } - // append integer part - target.append(intPart); - if (fracPart != 0L) - { - // append fractional part - target.append('.'); - // insert leading zeroes - while (scale > 0 && fracPart < tenPowDouble(--scale)) - { - target.append('0'); - } - target.append(fracPart); - } - } else - { - target.append('0'); - } - } - - /** - * Returns the exponent of the given value - * - * @param value - * the value to get the exponent from - * @return the value's exponent - */ - public static int getExponant(double value) - { - // See Double.doubleToRawLongBits javadoc or IEEE-754 spec - // to have this algorithm - long exp = Double.doubleToRawLongBits(value) & 0x7ff0000000000000L; - exp = exp >> 52; - return (int) (exp - 1023L); - } - - /** - * Returns true if the rounding is considered to use too many digits of the - * double for a fast rounding - * - * @param source - * the source to round - * @param scale - * the scale to round at - * @return true if the rounding will potentially use too many digits - */ - private static boolean tooManyDigitsUsed(double source, int scale) - { - // if scale >= 308, 10^308 ~= Infinity - double decExp = Math.log10(source); - return scale >= 308 || decExp + scale >= 14.5; - } - - /** - * Returns true if the given source is considered to be too close of a - * rounding value for the given scale. - * - * @param source - * the source to round - * @param scale - * the scale to round at - * @return true if the source will be potentially rounded at the scale - */ - private static boolean tooCloseToRound(double source, int scale) - { - source = Math.abs(source); - long intPart = (long) Math.floor(source); - double fracPart = (source - intPart) * tenPowDouble(scale); - double decExp = Math.log10(source); - double range = decExp + scale >= 12 ? .1 : .001; - double distanceToRound1 = Math.abs(fracPart - Math.floor(fracPart)); - double distanceToRound2 = Math.abs(fracPart - Math.floor(fracPart) - 0.5); - return distanceToRound1 <= range || distanceToRound2 <= range; - // .001 range: Totally arbitrary range, - // I never had a failure in 10e7 random tests with this value - // May be JVM dependent or architecture dependent - } -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java deleted file mode 100644 index ef96f80..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/IExporter.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.IOException; - -public interface IExporter -{ - static final String[] TOD_TOKENS = {"EA", "AM", "MD", "PM", "EV"}; - static final String TOD_TOKEN = "${TOD_TOKEN}"; - - void export() throws IOException; -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java deleted file mode 100644 index 12567c9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/IMatrixDao.java +++ /dev/null @@ -1,8 +0,0 @@ -package org.sandag.abm.reporting; - -import com.pb.common.matrix.Matrix; - -public interface IMatrixDao -{ - Matrix getMatrix(String matrixName, String coreName); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java deleted file mode 100644 index 2ba5261..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/OMXMatrixDao.java +++ /dev/null @@ -1,36 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.util.HashMap; -import java.util.Properties; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.MatrixType; - -public class OMXMatrixDao - implements IMatrixDao -{ - private final String outputFolderToken = "skims.path"; - private final String matrixLocation; - - public OMXMatrixDao(Properties properties) - { - matrixLocation = properties.getProperty(outputFolderToken); - } - - public OMXMatrixDao(HashMap properties) - { - matrixLocation = (String)properties.get(outputFolderToken); - } - - - public Matrix getMatrix(String matrixName, String coreName) - { - String matrixPath = matrixLocation + File.separator + matrixName; - - MatrixReader mr = MatrixReader.createReader(MatrixType.OMX, new File(matrixPath)); - - return mr.readMatrix(coreName); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java deleted file mode 100644 index ff3e8df..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/SkimBuilder.java +++ /dev/null @@ -1,721 +0,0 @@ -package org.sandag.abm.reporting; - -import java.util.HashMap; -import java.util.Map; -import java.util.Properties; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -/** - * The {@code SkimBuilder} ... - * - * @author crf Started 10/17/12 9:12 AM - */ -public class SkimBuilder -{ - private static final Logger logger = Logger.getLogger(SkimBuilder.class); - - private static final int WALK_TIME_INDEX = 0; - private static final int BIKE_TIME_INDEX = 0; - - private static final int DA_NT_TIME_INDEX = 0; - private static final int DA_NT_FF_TIME_INDEX = 1; - private static final int DA_NT_DIST_INDEX = 2; - private static final int DA_NT_TOLL_INDEX = 3; - private static final int DA_NT_TOLLDIST_INDEX = 4; - private static final int DA_NT_STD_INDEX = 5; - private static final int DA_TR_TIME_INDEX = 6; - private static final int DA_TR_FF_TIME_INDEX = 7; - private static final int DA_TR_DIST_INDEX = 8; - private static final int DA_TR_TOLL_INDEX = 9; - private static final int DA_TR_TOLLDIST_INDEX = 10; - private static final int DA_TR_STD_INDEX = 11; - private static final int SR2_TIME_INDEX = 12; - private static final int SR2_FF_TIME_INDEX = 13; - private static final int SR2_DIST_INDEX = 14; - private static final int SR2_HOVDIST_INDEX = 15; - private static final int SR2_TOLL_INDEX = 16; - private static final int SR2_TOLLDIST_INDEX = 17; - private static final int SR2_STD_INDEX = 18; - private static final int SR3_TIME_INDEX = 19; - private static final int SR3_FF_TIME_INDEX = 20; - private static final int SR3_DIST_INDEX = 21; - private static final int SR3_HOVDIST_INDEX = 22; - private static final int SR3_TOLL_INDEX = 23; - private static final int SR3_TOLLDIST_INDEX = 24; - private static final int SR3_STD_INDEX = 25; - - - - private static final int TRANSIT_SET_ACCESS_TIME_INDEX = 0; - private static final int TRANSIT_SET_EGRESS_TIME_INDEX = 1; - private static final int TRANSIT_SET_AUX_WALK_TIME_INDEX = 2; - private static final int TRANSIT_SET_LOCAL_BUS_TIME_INDEX = 3; - private static final int TRANSIT_SET_EXPRESS_BUS_TIME_INDEX = 4; - private static final int TRANSIT_SET_BRT_TIME_INDEX = 5; - private static final int TRANSIT_SET_LRT_TIME_INDEX = 6; - private static final int TRANSIT_SET_CR_TIME_INDEX = 7; - private static final int TRANSIT_SET_FIRST_WAIT_TIME_INDEX = 8; - private static final int TRANSIT_SET_TRANSFER_WAIT_TIME_INDEX = 9; - private static final int TRANSIT_SET_FARE_INDEX = 10; - private static final int TRANSIT_SET_MAIN_MODE_INDEX = 11; - private static final int TRANSIT_SET_XFERS_INDEX = 12; - private static final int TRANSIT_SET_DIST_INDEX = 13; - - private static final double FEET_IN_MILE = 5280.0; - - private final TapDataManager tapManager; - private final TazDataManager tazManager; - private final MgraDataManager mgraManager; - private final AutoTazSkimsCalculator tazDistanceCalculator; - private final AutoAndNonMotorizedSkimsCalculator autoNonMotSkims; - private final WalkTransitWalkSkimsCalculator wtw; - private final WalkTransitDriveSkimsCalculator wtd; - private final DriveTransitWalkSkimsCalculator dtw; - - private final String FUEL_COST_PROPERTY = "aoc.fuel"; - private final String MAINTENANCE_COST_PROPERTY = "aoc.maintenance"; - private float autoOperatingCost; - - - public SkimBuilder(Properties properties) - { - - HashMap rbMap = new HashMap( - (Map) (Map) properties); - tapManager = TapDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - autoNonMotSkims = new AutoAndNonMotorizedSkimsCalculator(rbMap); - autoNonMotSkims.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - BestTransitPathCalculator bestPathUEC = new BestTransitPathCalculator(rbMap); - wtw = new WalkTransitWalkSkimsCalculator(rbMap); - wtw.setup(rbMap, logger, bestPathUEC); - wtd = new WalkTransitDriveSkimsCalculator(rbMap); - wtd.setup(rbMap, logger, bestPathUEC); - dtw = new DriveTransitWalkSkimsCalculator(rbMap); - dtw.setup(rbMap, logger, bestPathUEC); - - float fuelCost = new Float(properties.getProperty(FUEL_COST_PROPERTY)); - float mainCost = new Float(properties.getProperty(MAINTENANCE_COST_PROPERTY)); - autoOperatingCost = (fuelCost + mainCost) * 0.01f; - - } - - // todo: hard coding these next two lookups because it is convenient, but - // probably should move to a lookup file - private final String[] modeNameLookup = { - "UNKNOWN", // ids start at one - "DRIVEALONE", "SHARED2", "SHARED3", "WALK", "BIKE", "WALK_SET", "PNR_SET", - "KNR_TRN", "TNC_TRN", "TAXI", "TNC_SINGLE", "TNC_SHARED","SCHBUS"}; - - private final TripModeChoice[] modeChoiceLookup = {TripModeChoice.UNKNOWN, - TripModeChoice.DRIVE_ALONE, TripModeChoice.SR2,TripModeChoice.SR3, - TripModeChoice.WALK, TripModeChoice.BIKE, TripModeChoice.WALK_SET, - TripModeChoice.PNR_SET, TripModeChoice.KNR_SET, TripModeChoice.KNR_SET, - TripModeChoice.SR2, - TripModeChoice.SR2, TripModeChoice.SR2, TripModeChoice.SR2 }; - - private int getTod(int tripTimePeriod) - { - return ModelStructure.getSkimPeriodIndex(tripTimePeriod); - } - - private int getStartTime(int tripTimePeriod) - { - return (tripTimePeriod - 1) * 30 + 270; // starts at 4:30 and goes half - // hour intervals after that - } - - public TripAttributes getTripAttributes(int origin, int destination, int tripModeIndex, - int boardTap, int alightTap, int tripTimePeriod, boolean inbound, float valueOfTime, int set, boolean isCB, int transponderOwnership) - { - int tod = getTod(tripTimePeriod); - TripModeChoice tripMode = modeChoiceLookup[tripModeIndex < 0 ? 0 : tripModeIndex]; - - TripAttributes attributes = getTripAttributes(tripMode, origin, destination, boardTap, - alightTap, tod, inbound, valueOfTime, set,isCB,transponderOwnership); - attributes.setTripModeName(modeNameLookup[tripModeIndex < 0 ? 0 : tripModeIndex]); - attributes.setTripStartTime(getStartTime(tripTimePeriod)); - return attributes; - } - - private TripAttributes getTripAttributesUnknown() - { - return new TripAttributes(0,0,0,0,0,0,0,0,0,0,0,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); - } - - private final float DEFAULT_BIKE_SPEED = 12; - private final float DEFAULT_WALK_SPEED = 3; - private int omeMgra=7123; - - private TripAttributes getTripAttributes(TripModeChoice modeChoice, int origin, - int destination, int boardTap, int alightTap, int tod, boolean inbound, float vot, int set,boolean isCB, int transponderOwnership) - { - int timeIndex = -1; - int distIndex = -1; - int costIndex = -1; - int stdIndex = -1; - int otaz=-1; - int dtaz=-1; - - double [] autoSkims; - - switch (modeChoice) - { - case UNKNOWN: - return getTripAttributesUnknown(); - case DRIVE_ALONE: - { - if(transponderOwnership>0) { - timeIndex = DA_TR_TIME_INDEX; - distIndex = DA_TR_DIST_INDEX; - costIndex = DA_TR_TOLL_INDEX; - stdIndex = DA_TR_STD_INDEX; - }else { - timeIndex = DA_NT_TIME_INDEX; - distIndex = DA_NT_DIST_INDEX; - costIndex = DA_NT_TOLL_INDEX; - stdIndex = DA_NT_STD_INDEX; - - } - if(isCB && (origin==omeMgra||destination==omeMgra)) { - int[] tazPair=setTazOD(origin, destination); - otaz=tazPair[0]; - dtaz=tazPair[1]; - autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, - logger); - }else { - autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, - logger); - } - return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); - } - - case SR2: // wu added - { - timeIndex = SR2_TIME_INDEX; - distIndex = SR2_DIST_INDEX; - costIndex = SR2_TOLL_INDEX; - stdIndex = SR2_STD_INDEX; - if(isCB && (origin==omeMgra||destination==omeMgra)) { - int[] tazPair=setTazOD(origin, destination); - otaz=tazPair[0]; - dtaz=tazPair[1]; - autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, - logger); - }else { - autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, - logger); - } - - return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); - } - case SR3: - { - timeIndex = SR3_TIME_INDEX; - distIndex = SR3_DIST_INDEX; - costIndex = SR3_TOLL_INDEX; - stdIndex = SR3_STD_INDEX; - if(isCB && (origin==omeMgra||destination==omeMgra)) { - int[] tazPair=setTazOD(origin, destination); - otaz=tazPair[0]; - dtaz=tazPair[1]; - autoSkims = autoNonMotSkims.getAutoSkimsByTAZ(otaz, dtaz, tod, vot,false, - logger); - }else { - autoSkims = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, - logger); - } - return new TripAttributes(autoSkims[timeIndex], autoSkims[distIndex], autoSkims[distIndex]*autoOperatingCost, autoSkims[stdIndex], autoSkims[costIndex]); - } - case WALK: - { - // first, look in mgra manager, otherwise default to auto skims - double distance = mgraManager.getMgraToMgraWalkDistFrom(origin, destination) / FEET_IN_MILE; - double time =0; - if (distance > 0) - { - time = mgraManager.getMgraToMgraWalkTime(origin, destination); - }else{ - distance = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, logger)[DA_NT_DIST_INDEX]; - time = distance * 60 / DEFAULT_WALK_SPEED; - } - return new TripAttributes(0, 0, 0, 0, 0, 0, 0, 0, time, 0, distance, -1, -1, 0,0,0,0,0,0,0,0,0,0,0,0,0,0); - } - case BIKE: - { - double time = mgraManager.getMgraToMgraBikeTime(origin, destination); - double distance = 0; - if (time > 0) - { - distance = time * DEFAULT_BIKE_SPEED / 60; - - }else{ - distance = autoNonMotSkims.getAutoSkims(origin, destination, tod, vot,false, - logger)[DA_NT_DIST_INDEX]; - time = distance * 60 / DEFAULT_BIKE_SPEED; - } - return new TripAttributes(0, 0, 0, 0, 0, 0, 0, 0, 0, time, distance, -1, -1, 0,0,0,0,0,0,0,0,0,0,0,0,0,0); - } - case WALK_SET : - case PNR_SET : - case KNR_SET : { - boolean isDrive = modeChoice.isDrive; - double walkTime = 0.0; - double driveTime = 0.0; - - double[] skims; - int boardTaz = -1; - int alightTaz = -1; - double boardAccessTime = 0.0; - double alightEgressTime = 0.0; - double accessDistance = 0.0; - double egressDistance = 0.0; - int originTaz = mgraManager.getTaz(origin); - int destTaz = mgraManager.getTaz(destination); - if (isDrive) { - if (!inbound) { //outbound: drive to transit stop at origin, then transit to destination - boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - accessDistance = tazManager.getDistanceToTapFromTaz(originTaz,boardTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - alightEgressTime = mgraManager.getWalkTimeFromMgraToTap(destination,alightTap); - egressDistance = mgraManager.getWalkDistanceFromMgraToTap(destination,alightTap); - - if (boardAccessTime ==-1) { - logger.info("Error: TAP not accessible from origin TAZ by "+ (modeChoice==TripModeChoice.PNR_SET ? "PNR" : "KNR" )+" access"); - logger.info("mc: " + modeChoice); - logger.info("origin MAZ: " + origin); - logger.info("origin TAZ" + originTaz); - logger.info("dest MAZ: " + destination); - logger.info("board tap: " + boardTap); - logger.info("alight tap: " + alightTap); - logger.info("tod: " + tod); - logger.info("inbound: " + inbound); - logger.info("set: " + set); - } - - if (alightEgressTime == -1){ - logger.info("Error: TAP not accessible from destination MAZ by walk access"); - logger.info("mc: " + modeChoice); - logger.info("origin MAZ: " + origin); - logger.info("origin TAZ" + originTaz); - logger.info("dest MAZ: " + destination); - logger.info("board tap: " + boardTap); - logger.info("alight tap: " + alightTap); - logger.info("tod: " + tod); - logger.info("inbound: " + inbound); - logger.info("set: " + set); - - } - skims = dtw.getDriveTransitWalkSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); - walkTime = alightEgressTime; - driveTime= boardAccessTime; - - } else { //inbound: transit from origin to destination, then drive - boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(origin,boardTap); - accessDistance = mgraManager.getWalkDistanceFromMgraToTap(origin,boardTap); - alightEgressTime = tazManager.getTimeToTapFromTaz(destTaz,alightTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - egressDistance = tazManager.getDistanceToTapFromTaz(destTaz,alightTap,( modeChoice==TripModeChoice.PNR_SET ? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - if (boardAccessTime ==-1) { - logger.info("Error: TAP not accessible from origin MAZ by walk access"); - logger.info("mc: " + modeChoice); - logger.info("origin MAZ: " + origin); - logger.info("origin TAZ" + originTaz); - logger.info("dest MAZ: " + destination); - logger.info("board tap: " + boardTap); - logger.info("alight tap: " + alightTap); - logger.info("tod: " + tod); - logger.info("inbound: " + inbound); - logger.info("set: " + set); - } - - if (alightEgressTime == -1){ - logger.info("Error: TAP not accessible from destination TAZ by "+ (modeChoice==TripModeChoice.PNR_SET ? "PNR" : "KNR" )+" access"); - logger.info("mc: " + modeChoice); - logger.info("origin MAZ: " + origin); - logger.info("origin TAZ" + originTaz); - logger.info("dest MAZ: " + destination); - logger.info("board tap: " + boardTap); - logger.info("alight tap: " + alightTap); - logger.info("tod: " + tod); - logger.info("inbound: " + inbound); - logger.info("set: " + set); - - } - skims = wtd.getWalkTransitDriveSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); - walkTime = boardAccessTime ; - driveTime= alightEgressTime; - } - } else { - int bt = mgraManager.getTapPosition(origin,boardTap); - int at = mgraManager.getTapPosition(destination,alightTap); - if (bt < 0 || at < 0) { - logger.info("bad tap position: " + bt + " " + at); - logger.info("mc: " + modeChoice); - logger.info("origin: " + origin); - logger.info("dest: " + destination); - logger.info("board tap: " + boardTap); - logger.info("alight tap: " + alightTap); - logger.info("tod: " + tod); - logger.info("inbound: " + inbound); - logger.info("set: " + set); - logger.info("board tap position: " + bt); - logger.info("alight tap position: " + at); - } else { - boardAccessTime = mgraManager.getMgraToTapWalkTime(origin,bt); - accessDistance = mgraManager.getWalkDistanceFromMgraToTap(origin,boardTap); - alightEgressTime = mgraManager.getMgraToTapWalkTime(destination,at); - egressDistance = mgraManager.getWalkDistanceFromMgraToTap(destination,alightTap); - } - walkTime = boardAccessTime + alightEgressTime; - skims = wtw.getWalkTransitWalkSkims(set,boardAccessTime,alightEgressTime,boardTap,alightTap,tod,false); - } - - double transitInVehicleTime = 0.0; - - transitInVehicleTime += skims[TRANSIT_SET_CR_TIME_INDEX]; - transitInVehicleTime += skims[TRANSIT_SET_LRT_TIME_INDEX]; - transitInVehicleTime += skims[TRANSIT_SET_BRT_TIME_INDEX]; - transitInVehicleTime += skims[TRANSIT_SET_EXPRESS_BUS_TIME_INDEX]; - transitInVehicleTime += skims[TRANSIT_SET_LOCAL_BUS_TIME_INDEX]; - - double crTime = skims[TRANSIT_SET_CR_TIME_INDEX]; - double lrtTime = skims[TRANSIT_SET_LRT_TIME_INDEX]; - double brtTime = skims[TRANSIT_SET_BRT_TIME_INDEX]; - double expTime = skims[TRANSIT_SET_EXPRESS_BUS_TIME_INDEX]; - double locTime = skims[TRANSIT_SET_LOCAL_BUS_TIME_INDEX]; - - //wsu 9/17/18, walkTime already set - //walkTime += skims[TRANSIT_SET_ACCESS_TIME_INDEX]; - //walkTime += skims[TRANSIT_SET_EGRESS_TIME_INDEX ]; - walkTime += skims[TRANSIT_SET_AUX_WALK_TIME_INDEX]; - - double auxiliaryTime = skims[TRANSIT_SET_AUX_WALK_TIME_INDEX]; - - double waitTime = 0.0; - waitTime += skims[TRANSIT_SET_FIRST_WAIT_TIME_INDEX]; - waitTime += skims[TRANSIT_SET_TRANSFER_WAIT_TIME_INDEX]; - - double transfers = skims[TRANSIT_SET_XFERS_INDEX]; - - double transitFare = 0.0; - transitFare += skims[TRANSIT_SET_FARE_INDEX]; - - double transitDist = skims[TRANSIT_SET_DIST_INDEX]; - /* - int modeIndex = 0; - for(modeIndex = TRANSIT_SET_LOCAL_BUS_TIME_INDEX; modeIndex <= TRANSIT_SET_CR_TIME_INDEX; modeIndex++){ - if(skims[modeIndex] > 0) - break; - } - */ - double dist = autoNonMotSkims.getAutoSkims(origin,destination,tod,vot,false,logger)[DA_NT_DIST_INDEX]; //todo: is this correct enough? - return new TripAttributes(driveTime, driveTime/60*35*autoOperatingCost, 0, 0, transitInVehicleTime, - waitTime, walkTime, transitFare, 0, 0, dist, boardTaz, alightTaz, vot, set, - accessDistance,egressDistance,auxiliaryTime,boardAccessTime,alightEgressTime,transfers,locTime,expTime,brtTime,lrtTime,crTime,transitDist); - } - default: - throw new IllegalStateException("Should not be here: " + modeChoice); - } - } - - private int[] setTazOD(int omgra, int dmgra) { - int [] result=new int[2]; - //int omeMgra=7123; - result [0]=mgraManager.getTaz(omgra); - result [1]=mgraManager.getTaz(dmgra); - if (omgra==omeMgra) result[0]=3; - if (dmgra==omeMgra) result[1]=3; - return result; - } - - public static enum TripModeChoice - { - UNKNOWN(false), - DRIVE_ALONE(true), - SR2(true), - SR3(true), - WALK(false), - BIKE(false), - WALK_SET(false), - PNR_SET(true), - KNR_SET(true); - - private final boolean isDrive; - - private TripModeChoice(boolean drive) - { - isDrive = drive; - } - - } - - public static class TripAttributes - { - private final float autoInVehicleTime; - private final float autoOperatingCost; - private final float autoStandardDeviationTime; - private final float autoTollCost; - private final float transitInVehicleTime; - private final float transitWaitTime; - private final float transitWalkTime; - private final float transitFare; - private final float walkModeTime; - private final float bikeModeTime; - private final float tripDistance; - private final int tripBoardTaz; - private final int tripAlightTaz; - private final int set; - private final float valueOfTime; - private final float transitAccessDistance; - private final float transitEgressDistance; - private final float transitAuxiliaryTime; - private final float transitAccessTime; - private final float transitEgressTime; - private final float transitTransfers; - private final float locTime; - private final float expTime; - private final float brtTime; - private final float lrtTime; - private final float crTime; - private final float transitDistance; - - private String tripModeName; - - public int getTripStartTime() - { - return tripStartTime; - } - - public void setTripStartTime(int tripStartTime) - { - this.tripStartTime = tripStartTime; - } - - private int tripStartTime; - - public TripAttributes(double autoInVehicleTime, double autoOperatingCost, double autoStandardDeviationTime, double autoTollCost, double transitInVehicleTime, - double transitWaitTime, double transitWalkTime, double transitFare, double walkModeTime, double bikeModeTime, double tripDistance, - int tripBoardTaz, int tripAlightTaz, float valueOfTime, int set, double accessDistance, - double egressDistance, double auxiliaryTime, double accessTime,double egressTime, double transfers, double locTime, double expTime, double brtTime, double lrtTime, double crTime, double trnDist) - { - this.autoInVehicleTime = (float) autoInVehicleTime; - this.autoOperatingCost = (float) autoOperatingCost; - this.autoStandardDeviationTime = (float) autoStandardDeviationTime; - this.autoTollCost = (float) autoTollCost; - this.transitInVehicleTime = (float) transitInVehicleTime; - this.transitWaitTime = (float) transitWaitTime; - this.transitWalkTime = (float) transitWalkTime; - this.transitFare = (float) transitFare; - this.walkModeTime = (float) walkModeTime; - this.bikeModeTime = (float) bikeModeTime; - this.tripDistance = (float) tripDistance; - this.tripBoardTaz = tripBoardTaz; - this.tripAlightTaz = tripAlightTaz; - this.set = set; - this.valueOfTime = valueOfTime; - this.transitAccessDistance = (float) accessDistance; - this.transitEgressDistance = (float) egressDistance; - this.transitAuxiliaryTime = (float) auxiliaryTime; - this.transitAccessTime = (float) accessTime; - this.transitEgressTime = (float) egressTime; - this.transitTransfers = (float) transfers; - this.locTime = (float) locTime; - this.expTime = (float) expTime; - this.brtTime = (float) brtTime; - this.lrtTime = (float) lrtTime; - this.crTime = (float) crTime; - this.transitDistance = (float) trnDist; - - } - - - /** - * A method to set create trip attributes for a non-toll auto choice. - * - * @param autoInVehicleTime - * @param tripDistance - * @param autoOperatingCost - */ - public TripAttributes(double autoInVehicleTime, double tripDistance, double autoOperatingCost, double stdDevTime) - { - this(autoInVehicleTime, autoOperatingCost, stdDevTime, 0,0,0,0,0,0,0,tripDistance,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); - } - - /** - * A method to create trip attributes for a toll auto choice. - * - * @param autoInVehicleTime - * @param tripDistance - * @param autoOperatingCost - * @param tollCost - */ - public TripAttributes(double autoInVehicleTime, double tripDistance, double autoOperatingCost, double stdDevTime, double tollCost) - { - this(autoInVehicleTime, autoOperatingCost, stdDevTime, tollCost,0,0,0,0,0,0,tripDistance,-1,-1,0,0,0,0,0,0,0,0,0,0,0,0,0,0); - } - - - - public void setTripModeName(String tripModeName) - { - this.tripModeName = tripModeName; - } - - public float getAutoInVehicleTime() { - return autoInVehicleTime; - } - - public float getAutoOperatingCost() { - return autoOperatingCost; - } - - public float getAutoStandardDeviationTime() { - return autoStandardDeviationTime; - } - - public float getAutoTollCost() { - return autoTollCost; - } - - public float getTransitInVehicleTime() { - return transitInVehicleTime; - } - - public float getTransitWaitTime() { - return transitWaitTime; - } - - public float getTransitFare() { - return transitFare; - } - - public float getTransitWalkTime() { - return transitWalkTime; - } - - public float getWalkModeTime() { - return walkModeTime; - } - - public float getBikeModeTime() { - return bikeModeTime; - } - - public float getTripDistance() { - return tripDistance; - } - - public String getTripModeName() - { - return tripModeName; - } - - public int getTripBoardTaz() - { - return tripBoardTaz; - } - - public int getTripAlightTaz() - { - return tripAlightTaz; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public int getSet() { - return set; - } - - public float getTransitAccessDistance() { - return transitAccessDistance; - } - - public float getTransitEgressDistance() { - return transitEgressDistance; - } - - public float getTransitAuxiliaryTime() { - return transitAuxiliaryTime; - } - - public float getTransitAccessTime() { - return transitAccessTime; - } - - public float getTransitEgressTime() { - return transitEgressTime; - } - - public float getTransitTransfers() { - return transitTransfers; - } - - public float getLocTime() { - return locTime; - } - - public float getExpTime() { - return expTime; - } - - public float getBrtTime() { - return brtTime; - } - - public float getLrtTime() { - return lrtTime; - } - - public float getCrTime() { - return crTime; - } - - public float getTransitDistance(){ - return transitDistance; - } - } - - public float getLotWalkTime(int parkingLotMaz, int destinationMaz) { - - // first, look in mgra manager, otherwise default to auto skims - double distance = mgraManager.getMgraToMgraWalkDistFrom(parkingLotMaz, destinationMaz) / FEET_IN_MILE; - if (distance <= 0) { - distance = autoNonMotSkims.getAutoSkims(parkingLotMaz, destinationMaz, SandagModelStructure.EA_SKIM_PERIOD_INDEX +1, (float)15.0,false, logger)[DA_NT_DIST_INDEX]; - } - - return (float) (distance * 60 / DEFAULT_WALK_SPEED); - } - - - public float getLotWalkDistance(int parkingLotMaz, int destinationMaz) { - - // first, look in mgra manager, otherwise default to auto skims - double distance = mgraManager.getMgraToMgraWalkDistFrom(parkingLotMaz, destinationMaz) / FEET_IN_MILE; - if (distance <= 0) { - distance = autoNonMotSkims.getAutoSkims(parkingLotMaz, destinationMaz, SandagModelStructure.EA_SKIM_PERIOD_INDEX +1, (float)15.0,false, logger)[DA_NT_DIST_INDEX]; - } - - return (float) distance; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java deleted file mode 100644 index f5aafc2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/TranscadMatrixDao.java +++ /dev/null @@ -1,28 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.util.Properties; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.MatrixType; - -public class TranscadMatrixDao - implements IMatrixDao -{ - private final String outputFolderToken = "skims.path"; - private final String matrixLocation; - - public TranscadMatrixDao(Properties properties) - { - matrixLocation = properties.getProperty(outputFolderToken); - } - - public Matrix getMatrix(String matrixName, String coreName) - { - String matrixPath = matrixLocation + File.separator + matrixName + ".mtx"; - - MatrixReader mr = MatrixReader.createReader(MatrixType.TRANSCAD, new File(matrixPath)); - - return mr.readMatrix(coreName); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java deleted file mode 100644 index bb5628f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/TransitTimeReporter.java +++ /dev/null @@ -1,449 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.BufferedWriter; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.Modes; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; - -public class TransitTimeReporter { - private static final Logger logger = Logger.getLogger(TransitTimeReporter.class); - private BestTransitPathCalculator bestPathCalculator; - protected WalkTransitWalkSkimsCalculator wtw; - protected WalkTransitDriveSkimsCalculator wtd; - protected DriveTransitWalkSkimsCalculator dtw; - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - private MatrixDataServerRmi ms; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - AutoTazSkimsCalculator tazDistanceCalculator; - - //skim locations in WalkTransitWalkSkims UEC - private static final int WLK_WALKACCESSTIME = 0; - private static final int WLK_WALKEGRESSTIME = 1; - private static final int WLK_AUXWALKTIME = 2; - private static final int WLK_LOCALBUSIVT = 3; - private static final int WLK_EXPRESSBUSIVT = 4; - private static final int WLK_BRTIVT = 5; - private static final int WLK_LRTIVT = 6; - private static final int WLK_CRIVT = 7; - private static final int WLK_T1IVT = 8; - private static final int WLK_FIRSTWAITTIME = 9; - private static final int WLK_TRWAITTIME = 10; - private static final int WLK_FARE = 11; - private static final int WLK_TOTALIVT = 12; - private static final int WLK_XFERS = 13; - private static final int WLK_DIST = 14; - - //skim locations in DriveTransitWalkSkims UEC - private static final int DRV_DRIVEACCESSTIME = 0; - private static final int DRV_WALKEGRESSTIME = 1; - private static final int DRV_AUXWALKTIME = 2; - private static final int DRV_LOCALBUSIVT = 3; - private static final int DRV_EXPRESSBUSIVT = 4; - private static final int DRV_BRTIVT = 5; - private static final int DRV_LRTIVT = 6; - private static final int DRV_CRIVT = 7; - private static final int DRV_T1IVT = 8; - private static final int DRV_FIRSTWAITTIME = 9; - private static final int DRV_TRWAITTIME = 10; - private static final int DRV_FARE = 11; - private static final int DRV_TOTALIVT = 12; - private static final int DRV_XFERS = 13; - private static final int DRV_DIST = 14; - - String period; //should be "AM" or "MD" - float threshold; //tested at 30 minutes - boolean inbound = false; - - private PrintWriter walkAccessWriter; - private PrintWriter driveAccessWriter; - private String outWalkFile; - private String outDriveFile; - private boolean createDriveFile=false; - - public TransitTimeReporter(HashMap propertyMap, float threshold, String period,String outWalkFileName,String outDriveFileName){ - - startMatrixServer(propertyMap); - - this.threshold = threshold; - this.period = period; - this.outWalkFile = outWalkFileName; - this.outDriveFile= outDriveFileName; - if(outDriveFile!=null) { - this.outDriveFile = outDriveFileName; - createDriveFile=true; - } - - initialize(propertyMap); - } - - /** - * Initialize best path builders. - * - * @param propertyMap A property map with relevant properties. - */ - public void initialize(HashMap propertyMap){ - - String path=System.getProperty("user.dir"); - outWalkFile=path+"\\output\\"+outWalkFile; - if(createDriveFile) { - outDriveFile=path+"\\output\\"+outDriveFile; - } - - logger.info("Initializing Transit Time Reporter"); - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - bestPathCalculator = new BestTransitPathCalculator(propertyMap); - - tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - wtw = new WalkTransitWalkSkimsCalculator(propertyMap); - wtw.setup(propertyMap, logger, bestPathCalculator); - wtd = new WalkTransitDriveSkimsCalculator(propertyMap); - wtd.setup(propertyMap, logger, bestPathCalculator); - dtw = new DriveTransitWalkSkimsCalculator(propertyMap); - dtw.setup(propertyMap, logger, bestPathCalculator); - - walkAccessWriter = createOutputFile(outWalkFile); - if(createDriveFile) { - driveAccessWriter = createOutputFile(outDriveFile); - } - } - - /** - * Create the output file. - */ - private PrintWriter createOutputFile(String fileName){ - - logger.info("Creating file " + fileName); - PrintWriter writer; - try - { - writer = new PrintWriter(new BufferedWriter(new FileWriter(fileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + fileName + " for writing\n"); - throw new RuntimeException(); - } - - return writer; - - } - - private ArrayList getWalkTransitTimeComponents(HashMap propertyMap){ - String timeElements = (String) propertyMap.get("transitShed.walkTransitTimeComponents"); - String delims = "[,]"; - String[] elements = timeElements.split(delims); - ArrayList components=new ArrayList(); - for (int i=0; i getDriveTransitTimeComponents(HashMap propertyMap){ - String timeElements = (String) propertyMap.get("transitShed.driveTransitTimeComponents"); - String delims = "[,]"; - String[] elements = timeElements.split(delims); - ArrayList components=new ArrayList(); - for (int i=0; i pMap){ - - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - double boardAccessTime; - double alightAccessTime; - - int skimPeriod = -1; - - if(period.compareTo("EA")==0){ - skimPeriod=ModelStructure.EA_SKIM_PERIOD_INDEX; - inbound = false; - }else if(period.compareTo("AM")==0){ - skimPeriod=ModelStructure.AM_SKIM_PERIOD_INDEX; - inbound = false; - }else if(period.compareTo("MD")==0){ - skimPeriod=ModelStructure.MD_SKIM_PERIOD_INDEX; - inbound = false; - }else if(period.compareTo("PM")==0){ - skimPeriod=ModelStructure.PM_SKIM_PERIOD_INDEX; - inbound = true; - }else if(period.compareTo("EV")==0){ - skimPeriod=ModelStructure.EV_SKIM_PERIOD_INDEX; - inbound = true; - }else{ - logger.fatal("Skim period "+period+" not recognized"); - throw new RuntimeException(); - } - - //iterate through mazs and calculate time - ArrayList mazs = mgraManager.getMgras(); - - //origins - for(int originMaz: mazs ){ - - if((originMaz<=100) || ((originMaz % 100) == 0)) - logger.info("Processing origin mgra "+originMaz); - - int originTaz = mgraManager.getTaz(originMaz); - - //for saving results - String outWalkString = null; - String outDriveString = null; - - //destinations - for(int destinationMaz:mazs){ - - int destinationTaz = mgraManager.getTaz(destinationMaz); - - float odDistance = (float) tazDistanceCalculator.getTazToTazDistance(skimPeriod, originTaz, destinationTaz); - - //walk calculations - double[][] bestWalkTaps = bestPathCalculator.getBestTapPairs(walkDmu, driveDmu, bestPathCalculator.WTW, originMaz, destinationMaz, skimPeriod, false, logger, odDistance); - double[] bestWalkUtilities = bestPathCalculator.getBestUtilities(); - - //only look at best utility path; continue if MGRA isn't available by walk. - if(bestWalkUtilities[0]>-500){ - - //Best walk TAP pair - int boardTap = (int) bestWalkTaps[0][0]; - int alightTap = (int) bestWalkTaps[0][1]; - int set = (int) bestWalkTaps[0][2]; - - // get walk skims - boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); - alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); - double[] walkSkims = wtw.getWalkTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); - - //calculate total time - double totalTime=0; - ArrayList wtelements=getWalkTransitTimeComponents(pMap); - for (int i=0; i-500){ - - //best drive TAP pair - int boardTap = (int) bestDriveTaps[0][0]; - int alightTap = (int) bestDriveTaps[0][1]; - int set = (int) bestDriveTaps[0][2]; - - //skims for best drive pair - double[] driveSkims = null; - if(inbound==false){ - boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( Modes.AccessMode.PARK_N_RIDE )); - alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); - driveSkims = dtw.getDriveTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); - }else{ - boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); - alightAccessTime = tazManager.getTimeToTapFromTaz(destinationTaz,alightTap,( Modes.AccessMode.PARK_N_RIDE )); - driveSkims = wtd.getWalkTransitDriveSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, skimPeriod, false); - - } - //total drive-transit time - //calculate total time - double totalTime=0; - ArrayList dtelements=getDriveTransitTimeComponents(pMap); - for (int i=0; i properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - MatrixDataServerIf ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - logger.error("could not connect to matrix server", e); - throw new RuntimeException(e); - - } - - } - - /** - * Main run method - * @param args - */ - public static void main(String[] args) { - - String propertiesFile = null; - float threshold = 0; - String period = null; - String outWalkFileName = null; - String outDriveFileName = null; - String delims = "[.]"; - - HashMap pMap; - - logger.info(String.format("Report MAZs within transit time threshold. Using CT-RAMP version ", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else { - propertiesFile = args[0]; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-threshold")) - { - threshold = Float.valueOf(args[i + 1]); - } - - if (args[i].equalsIgnoreCase("-period")) - { - period = args[i + 1]; - } - - if (args[i].equalsIgnoreCase("-outWalkFileName")) - { - String[] elements = args[i + 1].split(delims); - outWalkFileName = elements[0]+"_"+period+".csv"; - } - if (args[i].equalsIgnoreCase("-outDriveFileName")) - { - String[] elements = args[i + 1].split(delims); - outDriveFileName = elements[0]+"_"+period+".csv"; - } - } - } - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - TransitTimeReporter transitTimeReporter = new TransitTimeReporter(pMap, threshold, period,outWalkFileName,outDriveFileName); - - - transitTimeReporter.run(pMap); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java deleted file mode 100644 index ebd5cf2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvExporter.java +++ /dev/null @@ -1,58 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.IOException; -import java.util.Properties; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.LinkedBlockingQueue; - -public class TruckCsvExporter - extends AbstractCsvExporter -{ - private static final String MATRIX_BASE_NAME = "dailyDistributionMatricesTruck" + TOD_TOKEN; - private static final String[] CORE_NAMES = {"lhdn", "lhdt", "mhdn", "mhdt", "hhdn", - "hhdt" }; - private static final String[] COLUMN_HEADERS = {"ORIG", "DEST", "TOD", "CLASS", "TRIPS"}; - - public TruckCsvExporter(Properties properties, IMatrixDao aMatrixServerWrapper, - String aBaseFileName) - { - super(properties, aMatrixServerWrapper, aBaseFileName); - } - - @Override - public void export() throws IOException - { - BlockingQueue queue = new LinkedBlockingQueue(); - - Thread[] threads = new Thread[TOD_TOKENS.length]; - - LOGGER.info("Initializing Truck Writer Thread. Output Location: " - + getFile().getAbsoluteFile()); - CsvWriterThread writerThread = new CsvWriterThread(queue, getFile(), COLUMN_HEADERS); - new Thread(writerThread).start(); - - for (int i = 0; i < TOD_TOKENS.length; i++) - { - String matrixName = MATRIX_BASE_NAME.replace(TOD_TOKEN, TOD_TOKENS[i]); - LOGGER.info("Initializing Truck Reader Thread. Matrix: " + matrixName); - TruckCsvPublisherThread publisherThread = new TruckCsvPublisherThread(queue, - getMatrixDao(), matrixName, TOD_TOKENS[i], CORE_NAMES); - threads[i] = new Thread(publisherThread); - threads[i].start(); - } - - for (Thread thread : threads) - { - try - { - thread.join(); - } catch (InterruptedException e) - { - e.printStackTrace(System.err); - } - } - - LOGGER.info("Initializing Truck Reader Threads Complete. Issuing Poison Pill to Writer."); - queue.add(CsvWriterThread.POISON_PILL); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java deleted file mode 100644 index 2120d6e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckCsvPublisherThread.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.sandag.abm.reporting; - -import java.text.DecimalFormat; -import java.util.concurrent.BlockingQueue; -import org.apache.log4j.Logger; -import com.pb.common.matrix.Matrix; - -public class TruckCsvPublisherThread - implements Runnable -{ - private static final Logger LOGGER = Logger.getLogger(TruckCsvPublisherThread.class); - - private BlockingQueue queue; - private IMatrixDao mtxDao; - private String matrixName; - private String tod; - private String[] cores; - - private final double sizeThreshold = 0.00001; - - private static final DecimalFormat FORMATTER = new DecimalFormat("#.######"); - - public TruckCsvPublisherThread(BlockingQueue aQueue, - IMatrixDao aMtxDao, String aMatrixName, String aTod, String[] theCores) - { - this.queue = aQueue; - this.mtxDao = aMtxDao; - this.matrixName = aMatrixName; - this.tod = aTod; - this.cores = theCores; - } - - @Override - public void run() - { - for (String core : cores) - { - Matrix matrix = mtxDao.getMatrix(matrixName, core); - try - { - addRowsToQueue(core, matrix); - } catch (InterruptedException e) - { - LOGGER.fatal(e); - throw new RuntimeException(e); - } - } - - } - - public void addRowsToQueue(String core, Matrix matrix) throws InterruptedException - { - for (int origin : matrix.getExternalNumbers()) - { - for (int dest : matrix.getExternalColumnNumbers()) - { - float trips = matrix.getValueAt(origin, dest); - if (trips > sizeThreshold) - { - CsvRow row = new CsvRow(new String[] {String.valueOf(origin), - String.valueOf(dest), tod, core, FORMATTER.format(trips)}); - queue.put(row); - } - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java deleted file mode 100644 index a7272ef..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/TruckOmxExporter.java +++ /dev/null @@ -1,56 +0,0 @@ -package org.sandag.abm.reporting; - -import java.io.File; -import java.io.IOException; -import java.util.Properties; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.LinkedBlockingQueue; - -import org.apache.log4j.Logger; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; - -public class TruckOmxExporter implements IExporter -{ - private static final String MATRIX_BASE_NAME = "DailyDistributionMatricesTruck" + TOD_TOKEN; - private static final String[] CORE_NAMES = {"lhdn", "lhdt", "mhdn", "mhdt", "hhdn", - "hhdt" }; - - private IMatrixDao matrixDao; - private String reportFolder = "report.path"; - private Properties properties; - - protected static final Logger LOGGER = Logger.getLogger(AbstractCsvExporter.class); - - - public TruckOmxExporter(Properties properties, IMatrixDao aMatrixServerWrapper, - String aBaseFileName) - { - this.matrixDao = aMatrixServerWrapper; - this.properties = properties; - } - - @Override - public void export() throws IOException - { - - for (int i = 0; i < TOD_TOKENS.length; i++) - { - String matrixName = MATRIX_BASE_NAME.replace(TOD_TOKEN, TOD_TOKENS[i]); - - File outMatrixFile = new File(properties.getProperty(reportFolder), matrixName+".omx"); - - MatrixWriter matrixWriter = MatrixWriter.createWriter(MatrixType.OMX, outMatrixFile); - Matrix[] inMatrix = new Matrix[CORE_NAMES.length]; - - for(int j = 0; j < CORE_NAMES.length; ++j) - inMatrix[j] = matrixDao.getMatrix(matrixName, CORE_NAMES[j]); - - - matrixWriter.writeMatrices(CORE_NAMES, inMatrix); - - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java deleted file mode 100644 index e8ec218..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/AquavisDataBuilder.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.pb.sawdust.util.property.PropertyDeluxe; - -/** - * The {@code SandagAquavisInputBuilder} ... - * - * @author Wu.Sun@sandag.org 1/21/2014 - */ -public class AquavisDataBuilder { - private static final Logger LOGGER = LoggerFactory - .getLogger(AquavisDataBuilder.class); - private final PropertyDeluxe properties; - private Emfac2011SqlUtil sqlUtil = null; - - public AquavisDataBuilder(PropertyDeluxe properties, - Emfac2011SqlUtil sqlUtil) { - this.sqlUtil = sqlUtil; - this.properties = properties; - } - - public void createAquavisInputs() { - String scenarioId = properties - .getString(Emfac2011Properties.SCENARIO_ID); - String scenarioToken = properties - .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY); - - LOGGER.info("Step 1.1: Creating intrazonal Aquavis table..."); - sqlUtil.detemplifyAndRunScript( - properties - .getPath(Emfac2011Properties.CREATE_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY), - scenarioId, scenarioToken); - LOGGER.info("Step 1.2: Creating network Aquavis table..."); - sqlUtil.detemplifyAndRunScript( - properties - .getPath(Emfac2011Properties.CREATE_AQUAVIS_NETWORK_TEMPLATE_PROPERTY), - scenarioId, scenarioToken); - LOGGER.info("Step 1.3: Creating trips Aquavis table..."); - sqlUtil.detemplifyAndRunScript( - properties - .getPath(Emfac2011Properties.CREATE_AQUAVIS_TRIPS_TEMPLATE_PROPERTY), - scenarioId, scenarioToken); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java deleted file mode 100644 index 81fb2b9..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011AquavisIntrazonal.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; -/** - * - * @author Wu.Sun@sandag.org - * - */ -public class Emfac2011AquavisIntrazonal { - protected int zone; - protected double distance; - protected double speed; - protected String region; - protected String aType; - public int getZone() { - return zone; - } - public void setZone(int zone) { - this.zone = zone; - } - public double getDistance() { - return distance; - } - public void setDistance(double distance) { - this.distance = distance; - } - public double getSpeed() { - return speed; - } - public void setSpeed(double speed) { - this.speed = speed; - } - public String getRegion() { - return region; - } - public void setRegion(String region) { - this.region = region; - } - public String getaType() { - return aType; - } - public void setaType(String aType) { - this.aType = aType; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java deleted file mode 100644 index 76a6fd4..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Data.java +++ /dev/null @@ -1,302 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import java.util.ArrayList; -import java.util.EnumMap; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Set; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.pb.sawdust.tabledata.DataRow; -import com.pb.sawdust.tabledata.DataTable; -import com.pb.sawdust.tabledata.TableIndex; -import com.pb.sawdust.tabledata.basic.BasicTableIndex; -import com.pb.sawdust.tabledata.basic.RowDataTable; -import com.pb.sawdust.tabledata.metadata.DataType; -import com.pb.sawdust.tabledata.metadata.TableSchema; -import com.pb.sawdust.util.property.PropertyDeluxe; - -/** - * The {@code AbstractEmfac2011Data} class is used to provide data used to - * modify the EMFAC2011 SG input file. It essentially reads in the generic - * AquaVis data (which represents the travel demand model results) and refactors - * it into a data table that is used (by {@link Emfac2011InputFileCreator}) to - * create an adjusted EMFAC2011 input file. - * - * @author crf Started 2/8/12 9:13 AM - * Modified by Wu.Sun@sandag.org 1/21/2014 - */ -public abstract class Emfac2011Data { - private static final Logger LOGGER = LoggerFactory - .getLogger(Emfac2011Data.class); - private final PropertyDeluxe properties; - private Emfac2011SqlUtil sqlUtil = null; - - /** - * Get a mapping from the tNCVehicle types listed in a model's AquaVis results - * to their corresponding EMFAC2011 tNCVehicle types. The returned map should - * have the (exact) names listed in the AquaVis results as keys, and the set - * of EMFAC2011 tNCVehicle types that the represent the AquaVis type. A single - * EMFAC2011 may be used in the mappings of multiple AquaVis types (there is - * no functional mapping requirement), and only mutable EMFAC2011 tNCVehicle - * types may be used in the mapping (see - * {@link com.pb.aquavis.emfac2011.Emfac2011VehicleType#getMutableVehicleTypes()} - * . Also, all aquavis tNCVehicle types must be represented in the map - * (even if it is an empty mapping). - * - * @return a map representing the relationship between the AquaVis and - * EMFAC2011 tNCVehicle types. - */ - protected abstract Map> getAquavisVehicleTypeToEmfacTypeMapping(); - - public Emfac2011Data(PropertyDeluxe properties, Emfac2011SqlUtil sqlUtil) { - this.sqlUtil = sqlUtil; - this.properties = properties; - } - - public DataTable processAquavisData(Emfac2011Properties properties) { - - String scenario = properties.getString(Emfac2011Properties.SCENARIO_ID); - ArrayList> network = queryNetwork(sqlUtil, scenario); - ArrayList> trips = queryTrips(sqlUtil, scenario); - ArrayList> intrazonal = queryIntrazonal(sqlUtil, - scenario); - - Map> areas = new HashMap<>( - properties - .> getMap(Emfac2011Properties.AREAS_PROPERTY)); - Map districtsToSubareas = new HashMap<>(); - for (String subarea : areas.keySet()) - for (String district : areas.get(subarea)) - districtsToSubareas.put(district, subarea); - - DataTable outputTable = buildEmfacDataTableShell(areas); - TableIndex index = new BasicTableIndex<>(outputTable, - Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD, - Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, - Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); - index.buildIndex(); - - // need to spread out speed fractions - by auto class - Map> aquavisVehicleTypeToEmfacTypeMapping = getAquavisVehicleTypeToEmfacTypeMapping(); - Map> vehicleFractions = buildVehicleFractioning(aquavisVehicleTypeToEmfacTypeMapping); - - LOGGER.info("Step 2.1: Aggregating aquavis network VMT data"); - for (ArrayList row : network) { - double len = new Double(row.get(3)).doubleValue(); - String vehicleType = row.get(6); - double speed = new Double(row.get(7)).doubleValue(); - double vol = new Double(row.get(8)).doubleValue(); - String district = row.get(9); - - if (districtsToSubareas.containsKey(district)) { - String subarea = districtsToSubareas.get(district); - for (Emfac2011VehicleType emfacVehicle : aquavisVehicleTypeToEmfacTypeMapping - .get(vehicleType)) { - double fraction = vehicleFractions.get(emfacVehicle).get( - vehicleType); - for (int r : index.getRowNumbers(Emfac2011SpeedCategory - .getSpeedCategory(speed).getName(), subarea, - emfacVehicle.getName())) - outputTable - .setCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, - (Double) outputTable - .getCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) - + fraction * vol * len); - } - } - } - - // need to collect intrazonal vmt and add it to network vmt - by auto - // class - LOGGER.info("Step 2.2: Aggregating aquavis intrazonal VMT data"); - HashMap intrazonalMap = convertIntrazonal(intrazonal); - for (ArrayList row : trips) { - int zone = new Integer(row.get(1)).intValue(); - String vClass = row.get(5); - int vol = new Integer(row.get(6)).intValue(); - String district = (String) intrazonalMap.get(zone).getRegion(); - if (districtsToSubareas.containsKey(district)) { - String subarea = districtsToSubareas.get(district); - double speed = intrazonalMap.get(zone).getSpeed(); - double vmt = intrazonalMap.get(zone).getDistance() * vol; - for (Emfac2011VehicleType emfacVehicle : aquavisVehicleTypeToEmfacTypeMapping - .get(vClass)) { - double fraction = vehicleFractions.get(emfacVehicle).get( - vClass); - for (int r : index.getRowNumbers(Emfac2011SpeedCategory - .getSpeedCategory(speed).getName(), subarea, - emfacVehicle.getName())) - outputTable - .setCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, - (Double) outputTable - .getCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) - + fraction * vmt); - } - } - } - - LOGGER.info("Step 2.3: Building speed fractions"); - // build fractions - index = new BasicTableIndex<>(outputTable, - Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, - Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); - index.buildIndex(); - for (Emfac2011VehicleType emfacVehicle : Emfac2011VehicleType - .getMutableVehicleTypes()) { - for (String subarea : areas.keySet()) { - double sum = 0.0; - int count = 0; - for (DataRow row : outputTable.getIndexedRows(index, subarea, - emfacVehicle.getName())) { - sum += row - .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD); - count++; - } - for (int r : index.getRowNumbers(subarea, - emfacVehicle.getName())) - outputTable - .setCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD, - sum == 0.0 ? 1.0 / count - : (Double) outputTable - .getCellValue( - r, - Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD) - / sum); - } - } - - return outputTable; - } - - private DataTable buildEmfacDataTableShell(Map> areas) { - LOGGER.debug("Building EMFAC data table shell"); - TableSchema schema = new TableSchema("Emfac Data"); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD, - DataType.STRING); - schema.addColumn( - Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD, - DataType.DOUBLE); - schema.addColumn( - Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD, - DataType.DOUBLE); - DataTable outputTable = new RowDataTable(schema); - - // first, add rows for everything - for (Emfac2011VehicleType vehicle : Emfac2011VehicleType - .getMutableVehicleTypes()) - for (String subArea : areas.keySet()) - for (Emfac2011SpeedCategory speed : Emfac2011SpeedCategory - .values()) - outputTable.addRow(speed.getName(), subArea, - vehicle.getName(), 0.0, -1.0); // -1 - // is - // for - // error - // checking - // - - // 0 - // might - // pass - // through - return outputTable; // unnoticed - } - - private Map> buildVehicleFractioning( - Map> aquavisVehicleTypeToEmfacTypeMapping) { - // returns a map which says for every emfac tNCVehicle type, what aquavis - // tNCVehicle types should have their vmt added - // to it, and by what fraction - - Map> vehicleFractionMap = new EnumMap<>( - Emfac2011VehicleType.class); - for (Emfac2011VehicleType type : Emfac2011VehicleType - .getMutableVehicleTypes()) - vehicleFractionMap.put(type, new HashMap()); - for (String aquavisVehicleType : aquavisVehicleTypeToEmfacTypeMapping - .keySet()) { - double fraction = 1.0 / aquavisVehicleTypeToEmfacTypeMapping.get( - aquavisVehicleType).size(); - for (Emfac2011VehicleType type : aquavisVehicleTypeToEmfacTypeMapping - .get(aquavisVehicleType)) { - if (!vehicleFractionMap.containsKey(type)) - throw new IllegalStateException( - "Emfac tNCVehicle type is not mutable (" - + type - + ") and should not be component for aquavis type " - + aquavisVehicleType); - vehicleFractionMap.get(type).put(aquavisVehicleType, fraction); - } - } - return vehicleFractionMap; - } - - private HashMap convertIntrazonal( - ArrayList> intrazonal) { - HashMap result = new HashMap(); - for (ArrayList row : intrazonal) { - Emfac2011AquavisIntrazonal rec = new Emfac2011AquavisIntrazonal(); - int zone = new Integer(row.get(1)).intValue(); - rec.setZone(zone); - rec.setDistance(new Double(row.get(2))); - rec.setSpeed(new Double(row.get(3))); - rec.setRegion(row.get(4)); - rec.setaType(row.get(5)); - result.put(zone, rec); - } - return result; - } - - private ArrayList> queryNetwork(Emfac2011SqlUtil sqlUtil, - String schema) { - ArrayList> result = sqlUtil - .queryAquavisTables( - properties - .getPath(Emfac2011Properties.QUERY_AQUAVIS_NETWORK_TEMPLATE_PROPERTY), - schema, - properties - .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); - return result; - } - - private ArrayList> queryTrips(Emfac2011SqlUtil sqlUtil, - String schema) { - ArrayList> result = sqlUtil - .queryAquavisTables( - properties - .getPath(Emfac2011Properties.QUERY_AQUAVIS_TRIPS_TEMPLATE_PROPERTY), - schema, - properties - .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); - return result; - } - - private ArrayList> queryIntrazonal( - Emfac2011SqlUtil sqlUtil, String schema) { - ArrayList> result = sqlUtil - .queryAquavisTables( - properties - .getPath(Emfac2011Properties.QUERY_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY), - schema, - properties - .getString(Emfac2011Properties.AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY)); - return result; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java deleted file mode 100644 index bdc94b2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Definitions.java +++ /dev/null @@ -1,86 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -/** - * - * @author Wu.Sun@sandag.org 1/20/2014 - * - */ - -public class Emfac2011Definitions { - - //Aquavis table content defintions - public static final String AQUAVIS_NETWORK_FROM_NODE_FIELD = "from_node"; - public static final String AQUAVIS_NETWORK_TO_NODE_FIELD = "to_node"; - public static final String AQUAVIS_NETWORK_LENGTH_FIELD = "length"; - public static final String AQUAVIS_NETWORK_LINK_CLASS_FIELD = "link_class"; - public static final String AQUAVIS_NETWORK_TIME_PERIOD_FIELD = "time_period"; - public static final String AQUAVIS_NETWORK_VEHICLE_CLASS_FIELD = "vehicle_class"; - public static final String AQUAVIS_NETWORK_SPEED_FIELD = "assigned_speed"; - public static final String AQUAVIS_NETWORK_VOLUME_FIELD = "volume"; - public static final String AQUAVIS_NETWORK_REGION_FIELD = "region"; - - public static final String AQUAVIS_INTRAZONAL_ZONE_FIELD = "zone"; - public static final String AQUAVIS_INTRAZONAL_LENGTH_FIELD = "distance"; - public static final String AQUAVIS_INTRAZONAL_SPEED_FIELD = "speed"; - public static final String AQUAVIS_INTRAZONAL_REGION_FIELD = "region"; - public static final String AQUAVIS_INTRAZONAL_AREA_TYPE_FIELD = "area_type"; - - public static final String AQUAVIS_TRIPS_ORIGIN_ZONE_FIELD = "origin_zone"; - public static final String AQUAVIS_TRIPS_DESTINATION_ZONE_FIELD = "destination_zone"; - public static final String AQUAVIS_TRIPS_HOUR_FIELD = "hour"; - public static final String AQUAVIS_TRIPS_TIME_PERIOD_FIELD = "time_period"; - public static final String AQUAVIS_TRIPS_VEHICLE_CLASS_FIELD = "vehicle_class"; - public static final String AQUAVIS_TRIPS_TRIPS_FIELD = "trips"; - - public static final String VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN = "EMFAC2011_MODE"; - - // Emfac2011 Data Table Definitions - public static final String EMFAC_2011_DATA_SUB_AREA_FIELD = "subarea"; - public static final String EMFAC_2011_DATA_SPEED_FIELD = "speed"; - public static final String EMFAC_2011_DATA_VEHICLE_TYPE_FIELD = "vehicle_type"; - public static final String EMFAC_2011_DATA_VMT_FIELD = "vmt"; - public static final String EMFAC_2011_DATA_SPEED_FRACTION_FIELD = "fraction"; - - // Emfac2011 Excel Input Sheets definitions - public static final String EMFAC_2011_SCENARIO_TABLE_NAME = "Regional_Scenarios"; - public static final String EMFAC_2011_SCENARIO_TABLE_GROUP_FIELD = "Group"; - public static final String EMFAC_2011_SCENARIO_TABLE_AREA_TYPE_FIELD = "Area Type"; - public static final String EMFAC_2011_SCENARIO_TABLE_AREA_FIELD = "Area"; - public static final String EMFAC_2011_SCENARIO_TABLE_YEAR_FIELD = "CalYr"; - public static final String EMFAC_2011_SCENARIO_TABLE_SEASON_FIELD = "Season"; - - public static final String EMFAC_2011_VMT_TABLE_NAME = "Scenario_Base_Inputs"; - public static final String EMFAC_2011_VMT_TABLE_GROUP_FIELD = "Group"; - public static final String EMFAC_2011_VMT_TABLE_AREA_FIELD = "Area"; - public static final String EMFAC_2011_VMT_TABLE_SCENARIO_FIELD = "Scenario"; - public static final String EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD = "Sub-Area"; - public static final String EMFAC_2011_VMT_TABLE_YEAR_FIELD = "CalYr"; - public static final String EMFAC_2011_VMT_TABLE_SEASON_FIELD = "Season"; - public static final String EMFAC_2011_VMT_TABLE_TITLE_FIELD = "Title"; - public static final String EMFAC_2011_VMT_TABLE_VMT_PROFILE_FIELD = "VMT Profile"; - public static final String EMFAC_2011_VMT_TABLE_VMT_BY_VEH_FIELD = "VMT by TNCVehicle Category"; - public static final String EMFAC_2011_VMT_TABLE_SPEED_PROFILE_FIELD = "Speed Profile"; - public static final String EMFAC_2011_VMT_TABLE_VMT_FIELD = "New Total VMT"; - - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_NAME = "Scenario_VMT_by_VehCat"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_GROUP_FIELD = "Group"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_AREA_FIELD = "Area"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SCENARIO_FIELD = "Scenario"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD = "Sub-Area"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_YEAR_FIELD = "CalYr"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_SEASON_FIELD = "Season"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_TITLE_FIELD = "Title"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD = "Veh & Tech"; - public static final String EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD = "New VMT"; - - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_NAME = "Scenario_Speed_Profiles"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_GROUP_FIELD = "Group"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_AREA_FIELD = "Area"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SCENARIO_FIELD = "Scenario"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD = "Sub-Area"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_YEAR_FIELD = "CalYr"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_SEASON_FIELD = "Season"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_TITLE_FIELD = "Title"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD = "Veh & Tech"; - public static final String EMFAC_2011_SPEED_FRACTION_TABLE_2007_VEHICLE_FIELD = "EMFAC2007 Veh & Tech"; -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java deleted file mode 100644 index 85a5cef..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011InputFileCreator.java +++ /dev/null @@ -1,577 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import static com.pb.sawdust.util.Range.range; -import java.io.IOException; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.Arrays; -import java.util.EnumMap; -import java.util.HashMap; -import java.util.LinkedHashMap; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Set; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import com.pb.sawdust.excel.tabledata.read.ExcelTableReader; -import com.pb.sawdust.excel.tabledata.write.ExcelTableWriter; -import com.pb.sawdust.tabledata.DataRow; -import com.pb.sawdust.tabledata.DataTable; -import com.pb.sawdust.tabledata.TableIndex; -import com.pb.sawdust.tabledata.basic.BasicTableIndex; -import com.pb.sawdust.tabledata.basic.RowDataTable; -import com.pb.sawdust.tabledata.metadata.DataType; -import com.pb.sawdust.tabledata.metadata.TableSchema; -import com.pb.sawdust.tabledata.write.TableWriter; -import com.pb.sawdust.util.ProcessUtil; -import com.pb.sawdust.util.exceptions.RuntimeIOException; - -/** - * The {@code InputTemplateCreator} class is used to build an adjusted input - * file used for running the EMFAC2011 SG model. - * - * - * @author crf Started 2/7/12 1:48 PM Modified by Wu.Sun@sandag.org 1/21/2014 - */ -public class Emfac2011InputFileCreator -{ - private static final Logger LOGGER = LoggerFactory.getLogger(Emfac2011Data.class); - - private static final String[] SEASONS = {"ANNUAL", "SUMMER", "WINTER"}; - - /** - * Create an input file that can be used with the EMFAC2011 SG model. The - * constructed input file will use the default EMFAC2011 - * parameters/specifications, with adjustments based on a travel demand - * model's results. - * - * @param properties - * The properties specific to the model run. - * - * @param emfacModelData - * A data table (obtained from - * {@link Emfac2011Data#processAquavisData(Emfac2011Properties)} - * )) holding the results of the travel demand model. - * - * @return an EMFAC2011 SG input file, adjusted using {@code emfacModelData} - * . - */ - public Path createInputFile(Emfac2011Properties properties, DataTable emfacModelData) - { - String areaType = properties.getString(Emfac2011Properties.AREA_TYPE_PROPERTY); - String region = properties.getString(Emfac2011Properties.REGION_NAME_PROPERTY); - // Set areas = new - // HashSet<>(properties.getList(Emfac2011Properties.AREAS_PROPERTY)); - // Map> areas = new - // HashMap<>(properties.>getMap(Emfac2011Properties.AREAS_PROPERTY)); - Set areas = new HashMap<>( - properties.>getMap(Emfac2011Properties.AREAS_PROPERTY)) - .keySet(); - int oriYear = properties.getInt(Emfac2011Properties.YEAR_PROPERTY); - int year = Math.min(oriYear, 2035); - String inventoryDir = Paths.get( - properties.getString(Emfac2011Properties.EMFAC2011_INSTALLATION_DIR_PROPERTY), - "Application Files/Inventory Files").toString(); - String outputDir = properties.getString(Emfac2011Properties.OUTPUT_DIR_PROPERTY); - String converterProgram = properties - .getString(Emfac2011Properties.XLS_CONVERTER_PROGRAM_PROPERTY); - boolean preserveEmfacVehicleFractions = properties - .getBoolean(Emfac2011Properties.PRESERVE_EMFAC_VEHICLE_FRACTIONS_PROPERTY); - boolean modelVmtIncludesNonMutableVehicleTypes = properties - .getBoolean(Emfac2011Properties.MODEL_VMT_INCLUDES_NON_MUTABLES_PROPERTY); - return createInputFile(areaType, region, areas, year, oriYear, emfacModelData, - preserveEmfacVehicleFractions, modelVmtIncludesNonMutableVehicleTypes, - inventoryDir, outputDir, converterProgram); - } - - private String formInventoryFileName(String area, String season, int year) - { - return "EMFAC2011-SG Inventory - " + area + " - " + year + " (" + season + ").xls"; - } - - private void convertFile(String inventoryFile, String outputInventoryFile, - String converterProgram) - { - ProcessUtil.runProcess(Arrays.asList(converterProgram, inventoryFile, outputInventoryFile)); - } - - private DataTable readInventoryTable(String inventoryFile) - { - return new RowDataTable(ExcelTableReader.excelTableReader(inventoryFile)); - } - - private Path createInputFile(String areaType, String region, Set areas, int year, - int oriYear, DataTable emfacModelData, boolean preserveEmfacVehicleFractions, - boolean modelVmtIncludesNonMutableVehicleTypes, String inventoryDir, String outputDir, - String converterProgram) - { - Path outputFile = Paths.get(outputDir, formOutputFileName(region, oriYear)); - try - { - Files.deleteIfExists(outputFile); - } catch (IOException e) - { - throw new RuntimeIOException(e); - } - - TableWriter writer = new ExcelTableWriter(outputFile.toFile()); - LOGGER.debug("Initializing input excel file: " + outputFile); - writer.writeTable(formScenarioTable(areaType, region, year, SEASONS)); - - DataTable masterVmtTable = initVmtTable(); - DataTable masterVehicleVmtTable = initVehicleVmtTable(); - DataTable masterVmtSpeedTable = initVmtSpeedTable(); - - for (int i=0;i inputTables = new LinkedHashMap<>(); - for (String area : areas) - { - String inventoryFile = Paths.get(inventoryDir, - formInventoryFileName(area, SEASONS[i], year)).toString(); - Path outputInventoryFile = Paths.get(outputDir, - formInventoryFileName(area, SEASONS[i], year)); - convertFile(inventoryFile, outputInventoryFile.toString(), converterProgram); - inputTables.put(area, readInventoryTable(outputInventoryFile.toString())); - try - { - Files.delete(outputInventoryFile); - } catch (IOException e) - { - throw new RuntimeIOException(e); - } - } - - LOGGER.debug("Building vmt table"); - DataTable vmtTable = extractVmtTables(inputTables, i+1, region, year, SEASONS[i]); - LOGGER.debug("Building vmt by tNCVehicle type table"); - DataTable vehicleVmtTable = extractVmtVehicleTables(inputTables, i+1, region, year, SEASONS[i]); - LOGGER.debug("Building speed fraction table"); - DataTable vmtSpeedTable = extractVmtSpeedTables(inputTables, i+1, region, year, SEASONS[i]); - LOGGER.debug("Shifting tables using model data"); - shiftVmtTables(vmtTable, vehicleVmtTable, areas, emfacModelData, - preserveEmfacVehicleFractions, modelVmtIncludesNonMutableVehicleTypes); - shiftSpeedFractionTable(vmtSpeedTable, emfacModelData); - LOGGER.debug("Writing tables"); - - appendDataTable(masterVmtTable, vmtTable); - appendDataTable(masterVehicleVmtTable, vehicleVmtTable); - appendDataTable(masterVmtSpeedTable, vmtSpeedTable); - } - - writer.writeTable(masterVmtTable); - writer.writeTable(masterVehicleVmtTable); - writer.writeTable(masterVmtSpeedTable); - return outputFile; - } - - private void appendDataTable(DataTable master, DataTable fragment) - { - for(DataRow row : fragment) - { - master.addRow(row); - } - } - - private String formOutputFileName(String region, int year) - { - return "EMFAC2011-" + region + "-" + year + ".xls"; - } - - private DataTable formScenarioTable(String areaType, String area, int year, String[] seasons) - { - TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_NAME); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_GROUP_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_AREA_TYPE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_AREA_FIELD, DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_YEAR_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SCENARIO_TABLE_SEASON_FIELD, - DataType.STRING); - - DataTable table = new RowDataTable(schema); - - for (int i = 0; i < seasons.length; i++) - { - List row = new LinkedList<>(); - row.add(i+1); - row.add(areaType); - row.add(area); - row.add(year); - row.add(seasons[i]); - table.addRow(row.toArray(new Object[row.size()])); - } - return table; - } - - private DataTable initVmtTable() - { - TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_NAME); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_GROUP_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_AREA_FIELD, DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SCENARIO_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD, DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_YEAR_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SEASON_FIELD, DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_TITLE_FIELD, DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_PROFILE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_BY_VEH_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SPEED_PROFILE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD, DataType.DOUBLE); - return new RowDataTable(schema); - } - - private DataTable initVehicleVmtTable() - { - TableSchema schema = new TableSchema(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_NAME); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_GROUP_FIELD, - DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_AREA_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SCENARIO_FIELD, - DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_YEAR_FIELD, DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SEASON_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_TITLE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD, - DataType.DOUBLE); - return new RowDataTable(schema); - } - - private DataTable initVmtSpeedTable() - { - TableSchema schema = new TableSchema( - Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_NAME); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_GROUP_FIELD, - DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_AREA_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SCENARIO_FIELD, - DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_YEAR_FIELD, - DataType.INT); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SEASON_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_TITLE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD, - DataType.STRING); - schema.addColumn(Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_2007_VEHICLE_FIELD, - DataType.STRING); - for (Emfac2011SpeedCategory category : Emfac2011SpeedCategory.values()) - schema.addColumn(category.getName(), DataType.DOUBLE); - return new RowDataTable(schema); - } - - private DataTable extractVmtTables(Map inputTables, int group, String area, int year, - String season) - { - DataTable vmtTable = initVmtTable(); - - int counter = 1; - for (String subArea : inputTables.keySet()) - { - List row = new LinkedList<>(); - row.add(group); - row.add(area); - row.add(counter); - row.add(subArea); - row.add(year); - row.add(season); - row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter++, - subArea, year, season)); - row.add("User"); - row.add("User"); - row.add("User"); - - double totalVmt = 0.0; - for (DataRow r : inputTables.get(subArea)) - if (r.getCellAsString("Tech").equals("TOT")) totalVmt += r.getCellAsDouble("VMT"); - row.add(totalVmt); - vmtTable.addRow(row.toArray(new Object[row.size()])); - } - return vmtTable; - } - - private DataTable extractVmtVehicleTables(Map inputTables, int group, String area, - int year, String season) - { - DataTable vehicleVmtTable = initVehicleVmtTable(); - - int counter = 1; - for (String subArea : inputTables.keySet()) - { - for (DataRow r : inputTables.get(subArea)) - { - String tech = r.getCellAsString("Tech"); - if (tech.equals("DSL") || tech.equals("GAS")) - { - List row = new LinkedList<>(); - row.add(group); - row.add(area); - row.add(counter); - row.add(subArea); - row.add(year); - row.add(season); - row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter, - subArea, year, season)); - row.add(r.getCellAsString("Veh & Tech")); - row.add(r.getCellAsDouble("VMT")); - vehicleVmtTable.addRow(row.toArray(new Object[row.size()])); - } - } - counter++; - } - return vehicleVmtTable; - } - - private DataTable extractVmtSpeedTables(Map inputTables, int group, String area, - int year, String season) - { - DataTable vmtSpeedTable = initVmtSpeedTable(); - - int counter = 1; - for (String subArea : inputTables.keySet()) - { - // loop over everything once to get sums and types, and then second - // time to generate fractions - Map techTotals = new LinkedHashMap<>(); - for (DataRow r : inputTables.get(subArea)) - { - String tech = r.getCellAsString("Tech"); - if (tech.startsWith("Spd") && !tech.endsWith("TOT")) - { - String vehNTech = r.getCellAsString("Veh & Tech").toLowerCase(); - if (!techTotals.containsKey(vehNTech)) techTotals.put(vehNTech, 0.0); - techTotals.put(vehNTech, techTotals.get(vehNTech) + r.getCellAsDouble("VMT")); - } - } - Map techRows = new HashMap<>(); - // loop over each type and add in the rows - for (Emfac2011VehicleType type : Emfac2011VehicleType.values()) - { - List row = new LinkedList<>(); - row.add(group); - row.add(area); - row.add(counter); - row.add(subArea); - row.add(year); - row.add(season); - row.add(String.format("Group #1 (%s), Scenario #%d - %s %d %s", area, counter, - subArea, year, season)); - row.add(type.getName()); - row.add(type.getEmfac2007Name()); - for (int i : range(5, 71, 5)) - row.add(0.0); - vmtSpeedTable.addRow(row.toArray(new Object[row.size()])); - techRows.put(type.getName().toLowerCase(), vmtSpeedTable.getRowCount() - 1); - } - // now reloop over table to get fractions - for (DataRow r : inputTables.get(subArea)) - { - String tech = r.getCellAsString("Tech"); - if (tech.startsWith("Spd") && !tech.endsWith("TOT")) - { - String vehNTech = r.getCellAsString("Veh & Tech").toLowerCase(); - double fraction = techTotals.get(vehNTech) == 0.0 ? 0.0 : r - .getCellAsDouble("VMT") / techTotals.get(vehNTech); - vmtSpeedTable.setCellValue(techRows.get(vehNTech), - Integer.parseInt(tech.substring(3, 5)) + "MPH", fraction); - } - } - counter++; - } - // ensure that we sum up to one - counter = 0; - for (DataRow r : vmtSpeedTable) - { - double sum = 1.0; - Emfac2011SpeedCategory[] speeds = Emfac2011SpeedCategory.values(); - for (int i : range(speeds.length - 1)) - sum -= r.getCellAsDouble(speeds[i].getName()); - vmtSpeedTable.setCellValue(counter++, Emfac2011SpeedCategory.SPEED_65_70plus.getName(), - sum); - } - return vmtSpeedTable; - } - - // private void shiftSpeedFractionTable(DataTable speedVmtTable, DataTable - // modelData) { - private void shiftSpeedFractionTable(DataTable speedVmtTable, DataTable modelData) - { - TableIndex index = new BasicTableIndex<>(speedVmtTable, - Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_SUB_AREA_FIELD, - Emfac2011Definitions.EMFAC_2011_SPEED_FRACTION_TABLE_VEHICLE_FIELD); - index.buildIndex(); - - for (DataRow row : modelData) - { - String subArea = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD); - String vehicleType = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD); - String category = Emfac2011SpeedCategory.getTypeForName( - row.getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FIELD)) - .getName(); - speedVmtTable.setCellValue(index.getRowNumbers(subArea, vehicleType).iterator().next(), - category, - row.getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_SPEED_FRACTION_FIELD)); - } - } - - private void shiftVmtTables(DataTable vmtTable, DataTable vehicleVmtTable, Set areas, - DataTable modelData, boolean preserveEmfacVehicleFractions, - boolean modelVmtIncludesNonMutableVehicleTypes) - { - Map> modelVmtByAreaAndVehicleType = new HashMap<>(); - Map> emfacMutableVmtByAreaAndVehicleType = new HashMap<>(); - Map> emfacImmutableVmtByAreaAndVehicleType = new HashMap<>(); - - Set mutableVehicleTypes = Emfac2011VehicleType - .getMutableVehicleTypes(); - - for (String subarea : areas) - { - Map m1 = new EnumMap<>(Emfac2011VehicleType.class); - Map m2 = new EnumMap<>(Emfac2011VehicleType.class); - Map m3 = new EnumMap<>(Emfac2011VehicleType.class); - for (Emfac2011VehicleType vehicleType : Emfac2011VehicleType.values()) - { - if (mutableVehicleTypes.contains(vehicleType)) - { - m1.put(vehicleType, 0.0); - m2.put(vehicleType, 0.0); - } else - { - if (modelVmtIncludesNonMutableVehicleTypes) m1.put(vehicleType, 0.0); - m3.put(vehicleType, 0.0); - } - } - modelVmtByAreaAndVehicleType.put(subarea, m1); - emfacMutableVmtByAreaAndVehicleType.put(subarea, m2); - emfacImmutableVmtByAreaAndVehicleType.put(subarea, m3); - } - - for (DataRow row : vehicleVmtTable) - { - String subArea = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD); - Emfac2011VehicleType vehicleType = Emfac2011VehicleType - .getVehicleType(row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD)); - double vmt = row - .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD); - if (mutableVehicleTypes.contains(vehicleType)) emfacMutableVmtByAreaAndVehicleType.get( - subArea).put(vehicleType, - emfacMutableVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); - else emfacImmutableVmtByAreaAndVehicleType.get(subArea).put(vehicleType, - emfacImmutableVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); - } - - for (DataRow row : modelData) - { - String subArea = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_SUB_AREA_FIELD); - Emfac2011VehicleType vehicleType = Emfac2011VehicleType.getVehicleType(row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_DATA_VEHICLE_TYPE_FIELD)); - double vmt = row.getCellAsDouble(Emfac2011Definitions.EMFAC_2011_DATA_VMT_FIELD); - modelVmtByAreaAndVehicleType.get(subArea).put(vehicleType, - modelVmtByAreaAndVehicleType.get(subArea).get(vehicleType) + vmt); - } - - if (preserveEmfacVehicleFractions) - { - // need to reshift data - for (String subarea : modelVmtByAreaAndVehicleType.keySet()) - { - double totalVmt = 0.0; - Map modelVmt = modelVmtByAreaAndVehicleType - .get(subarea); - for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) - totalVmt += modelVmt.get(vehicleType); - double totalEmfacVmt = 0.0; - Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType - .get(subarea); - Map emfacImutableVmt = emfacImmutableVmtByAreaAndVehicleType - .get(subarea); - for (Emfac2011VehicleType vehicleType : emfacMutableVmt.keySet()) - totalEmfacVmt += emfacMutableVmt.get(vehicleType); - if (modelVmtIncludesNonMutableVehicleTypes) - for (Emfac2011VehicleType vehicleType : emfacImutableVmt.keySet()) - totalEmfacVmt += emfacImutableVmt.get(vehicleType); - for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) - { - if (emfacMutableVmt.containsKey(vehicleType)) modelVmt.put(vehicleType, - totalVmt * emfacMutableVmt.get(vehicleType) / totalEmfacVmt); - else modelVmt.put(vehicleType, totalVmt * emfacImutableVmt.get(vehicleType) - / totalEmfacVmt); - } - } - } - - // shift overall vmt - vmtTable.setPrimaryKey(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD); - for (DataRow row : vmtTable) - { - String subArea = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_SUB_AREA_FIELD); - double originalVmt = row - .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD); - Map modelVmt = modelVmtByAreaAndVehicleType.get(subArea); - Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType - .get(subArea); - Map emfacImutableVmt = emfacImmutableVmtByAreaAndVehicleType - .get(subArea); - for (Emfac2011VehicleType vehicleType : modelVmt.keySet()) - { - originalVmt += modelVmt.get(vehicleType); // add corrected vmt - originalVmt -= emfacMutableVmt.containsKey(vehicleType) ? emfacMutableVmt - .get(vehicleType) : emfacImutableVmt.get(vehicleType); // subtract - // old - // vmt - } - vmtTable.setCellValueByKey(subArea, - Emfac2011Definitions.EMFAC_2011_VMT_TABLE_VMT_FIELD, originalVmt); - } - - // replace tNCVehicle vmts - TableIndex index = new BasicTableIndex<>(vehicleVmtTable, - Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD, - Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD); - index.buildIndex(); - for (DataRow row : vehicleVmtTable) - { - String subArea = row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_SUB_AREA_FIELD); - Map modelVmt = modelVmtByAreaAndVehicleType.get(subArea); - Map emfacMutableVmt = emfacMutableVmtByAreaAndVehicleType - .get(subArea); - Emfac2011VehicleType vehicleType = Emfac2011VehicleType - .getVehicleType(row - .getCellAsString(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VEHICLE_FIELD)); - if (modelVmt.containsKey(vehicleType)) - { - double vmt = row - .getCellAsDouble(Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD); - if (modelVmtIncludesNonMutableVehicleTypes) vmt = modelVmt.get(vehicleType); - else vmt += modelVmt.get(vehicleType) - emfacMutableVmt.get(vehicleType); - vehicleVmtTable.setCellValue(index.getRowNumbers(subArea, vehicleType.getName()) - .iterator().next(), - Emfac2011Definitions.EMFAC_2011_VEHICLE_VMT_TABLE_VMT_FIELD, vmt); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java deleted file mode 100644 index b3a2031..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Properties.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import com.pb.sawdust.util.property.PropertyDeluxe; - -/** - * The {@code Emfac2011Properties} holds the properties used by the various - * classes in this package. The property keys listed in this class (as - * {@code public static final} constants) must all be present in the - * instantiated properties object, or it will throw an error during - * construction. Additionally, {@link #AREAS_PROPERTY} must be typed correctly - * as a map from a string to a list holding strings (see its documentation for - * more information). - * - * @author crf Started 2/9/12 8:40 AM - * Modified by Wu.Sun@sanag.org 1/28/2014 - */ -public class Emfac2011Properties - extends PropertyDeluxe -{ - //database properties - public static final String REPORTS_DATABASE_IPADDRESS_PROPERTY = "reports.database.ipaddress"; - public static final String REPORTS_DATABASE_PORT_PROPERTY = "reports.database.port"; - public static final String REPORTS_DATABASE_NAME_PROPERTY = "reports.database.name"; - public static final String REPORTS_DATABASE_USERNAME_PROPERTY = "reports.database.username"; - public static final String REPORTS_DATABASE_PASSWORD_PROPERTY = "reports.database.password"; - public static final String REPORTS_DATABASE_INSTANCE_PROPERTY = "reports.database.instance"; - - //San Diego Emfac2011 properties - public static final String AREA_TYPE_PROPERTY = "emfac.2011.area.type"; - public static final String REGION_NAME_PROPERTY = "emfac.2011.region.name"; - public static final String AREAS_PROPERTY = "emfac.2011.area"; - public static final String SEASON_PROPERTY = "emfac.2011.season"; - public static final String YEAR_PROPERTY = "emfac.2011.year"; - public static final String EMFAC2011_INSTALLATION_DIR_PROPERTY = "emfac.2011.installation.dir"; - public static final String OUTPUT_DIR_PROPERTY = "emfac.2011.output.dir"; - public static final String AQUAVIS_INTRAZONAL_FILE_PROPERTY = "emfac.2011.aquavis.intrazonal"; - - // Aquavis table creation templates - public static final String CREATE_AQUAVIS_NETWORK_TEMPLATE_PROPERTY = "aquavis.network.sql.template"; - public static final String CREATE_AQUAVIS_TRIPS_TEMPLATE_PROPERTY = "aquavis.trips.sql.template"; - public static final String CREATE_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY = "aquavis.intrazonal.sql.template"; - - // Aquavis table query templates - public static final String QUERY_AQUAVIS_NETWORK_TEMPLATE_PROPERTY = "aquavis.network.query.template"; - public static final String QUERY_AQUAVIS_TRIPS_TEMPLATE_PROPERTY = "aquavis.trips.query.template"; - public static final String QUERY_AQUAVIS_INTRAZONAL_TEMPLATE_PROPERTY = "aquavis.intrazonal.query.template"; - - // inputs, outputs, and tokens - public static final String AQUAVIS_TEMPLATE_SCENARIOID_TOKEN_PROPERTY= "aquavis.template.scenarioId.token"; - public static final String SCENARIO_ID = "scenario.id"; - public static final String VEHICLE_CODE_MAPPING_FILE_PROPERTY = "emfac.2011.to.sandag.vehicle.code.mapping.file"; - - // switch if EMFAC is executed - public static final String EXECUTE_EMFAC="execute.emfac"; - - /** - * The property key for the boolean indicating if the (default) EMFAC - * tNCVehicle fractions should be preserved in the EMFAC input file (value is - * {@code true}), or if the model tNCVehicle fractions should be used (value is - * {@code false}). - */ - public static final String PRESERVE_EMFAC_VEHICLE_FRACTIONS_PROPERTY = "emfac.2011.preserve.emfac.vehicle.fractions"; - /** - * The property key for the boolean indicating whether or not the model - * (travel demand, not EMFAC) VMT includes totals for non-mutable tNCVehicle - * types. If it does, then the VMT will be scaled before adjusting the EMFAC - * input file. - */ - public static final String MODEL_VMT_INCLUDES_NON_MUTABLES_PROPERTY = "emfac.2011.model.vmt.includes.non.mutable.vehicles"; - - /** - * The property key for the location of the xls converter program. This - * program converts the malformed EMFAC2011 reference files to a (strictly) - * valid format, which can then be used by the rest of the model. - */ - public static final String XLS_CONVERTER_PROGRAM_PROPERTY = "emfac.2011.xls.converter.program"; - - /** - * The property key for the location (full path) of the AquaVis output - * intrazonal file. - */ - - /** - * Constructor specifying the resources used to build the properties. - * - * @param firstResource - * The first properties resource. - * - * @param additionalResources - * Any additional properties resources. - * - * @throws IllegalArgumentException - * if any of the required properties is missing, or if they are - * typed incorrectly. - */ - public Emfac2011Properties(String firstResource) - { - super(firstResource); - System.out.println("first property="+firstResource); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java deleted file mode 100644 index 4622be2..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011Runner.java +++ /dev/null @@ -1,216 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import java.io.IOException; -import java.nio.file.FileSystems; -import java.nio.file.FileVisitOption; -import java.nio.file.FileVisitResult; -import java.nio.file.FileVisitor; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.PathMatcher; -import java.nio.file.Paths; -import java.nio.file.SimpleFileVisitor; -import java.nio.file.attribute.BasicFileAttributes; -import java.util.Arrays; -import java.util.Collections; -import java.util.EnumMap; -import java.util.EnumSet; -import java.util.HashMap; -import java.util.LinkedHashSet; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Set; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.pb.sawdust.tabledata.DataRow; -import com.pb.sawdust.tabledata.DataTable; -import com.pb.sawdust.tabledata.basic.RowDataTable; -import com.pb.sawdust.tabledata.read.CsvTableReader; -import com.pb.sawdust.util.ProcessUtil; -import com.pb.sawdust.util.exceptions.RuntimeIOException; - -/** - * The {@code Emfac2011Runner} class is used to generate an EMFAC2011 SG input - * file adjusted for travel demand model results, and then run (via the end - * user) the EMFAC2011 SG model using those inputs. - * - * @author crf Started 2/9/12 9:17 AM - * Modified by Wu.Sun@sandag.org 1/24/2014 - */ -public class Emfac2011Runner { - private static final Logger LOGGER = LoggerFactory - .getLogger(Emfac2011Runner.class); - private final Emfac2011Properties properties; - private Emfac2011SqlUtil sqlUtil;; - - /** - * Constructor specifying the resources used to build the properties used - * for the EMFAC2011 SG run. - * - * @param propertyResource - * The first properties resource. - * - * @param additionalResources - * Any additional properties resources. - */ - public Emfac2011Runner(String propertyResource) { - properties = new Emfac2011Properties(propertyResource); - sqlUtil = new Emfac2011SqlUtil(properties); - } - - public void runEmfac2011() { - LOGGER.info("***************Running Emfac2011 for SANDAG***********************"); - LOGGER.info("Step 0: Setting up mutable tNCVehicle types"); - // have to call this first because it sets the mutable types, which are, - // used throughout the EMFAC2011 process - Path path = Paths - .get(properties.getString(Emfac2011Properties.VEHICLE_CODE_MAPPING_FILE_PROPERTY)); - - final Map> aquavisVehicleTypeToEmfacMapping = buildAquavisVehicleTypeToEmfacMapping(path); - - runEmfac2011(new Emfac2011Data(properties, sqlUtil) { - @Override - protected Map> getAquavisVehicleTypeToEmfacTypeMapping() { - return aquavisVehicleTypeToEmfacMapping; - } - }); - } - - /** - * Run the EMFAC2011 model. This method will process the model results (via - * AquaVis outputs), create an adjusted EMFAC2011 input file, and initiate - * the EMFAC2011 SG model. Because of the way it is set up, the user must - * actually set up and run the EMFAC2011 SG model, but this method will - * create a dialog window which will walk the user through the steps - * required to do that. - * - * @param emfac2011Data - * The {@code Emfac2011Data} instance corresponding to the model - * results/run. - */ - public void runEmfac2011(Emfac2011Data emfac2011Data) { - LOGGER.info("Step 1: Building Aquavis inputs from scenario: " - + properties.getString(Emfac2011Properties.SCENARIO_ID)); - AquavisDataBuilder builder = new AquavisDataBuilder(properties, sqlUtil); - builder.createAquavisInputs(); - LOGGER.info("Step 2: Processing aquavis data"); - DataTable data = emfac2011Data.processAquavisData(properties); - LOGGER.info("Step 3: Creating EMFAC2011 input file"); - Emfac2011InputFileCreator inputFileCreator = new Emfac2011InputFileCreator(); - Path inputfile = inputFileCreator.createInputFile(properties, data); - if((properties.getString(Emfac2011Properties.EXECUTE_EMFAC)).equalsIgnoreCase("true")){ - LOGGER.info("Step 4: Initiating EMFAC2011"); - RunEmfacDialog.createAndShowGUI(inputfile, this); - }else{ - LOGGER.info("Sipped--Step 4: Initiating EMFAC2011"); - } - LOGGER.info("EMFAC2011 run finished"); - } - - private Map> buildAquavisVehicleTypeToEmfacMapping( - Path vehicleCodeMappingFile) { - Map> mapping = new EnumMap<>( - SandagAutoModes.class); - for (SandagAutoModes type : SandagAutoModes.values()) - mapping.put(type, EnumSet.noneOf(Emfac2011VehicleType.class)); - - // file has one column = - // VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN - // the rest have names which, when made uppercase, should match - // VehicleType enum - DataTable vehicleCodeMapping = new RowDataTable(new CsvTableReader( - vehicleCodeMappingFile.toString())); - vehicleCodeMapping.setDataCoersion(true); - Set vehicleCodeColumns = new LinkedHashSet<>(); - for (String column : vehicleCodeMapping.getColumnLabels()) { - try { - SandagAutoModes.valueOf(column.toUpperCase()); - vehicleCodeColumns.add(column); - } catch (IllegalArgumentException e) { - // absorb - not a valid type column - } - } - Set mutableVehicleType = EnumSet - .noneOf(Emfac2011VehicleType.class); - for (DataRow row : vehicleCodeMapping) { - Emfac2011VehicleType emfac2011VehicleType = Emfac2011VehicleType - .getVehicleType(row - .getCellAsString(Emfac2011Definitions.VEHICLE_CODE_MAPPING_EMFAC2011_VEHICLE_NAME_COLUMN)); - // now dynamically setting mutable tNCVehicle types, so we need to not - // rely on the defaults - // if (!emfac2011VehicleType.isMutableType()) - // continue; //skip any non-mutable types, as they can't be used - for (String column : vehicleCodeColumns) { - if (row.getCellAsBoolean(column)) { - mutableVehicleType.add(emfac2011VehicleType); // if a - // mapping - // exists, - // then the - // EMFAC - // tNCVehicle - // type is - // assumed - // to - // be - // mutable - mapping.get(SandagAutoModes.valueOf(column.toUpperCase())) - .add(emfac2011VehicleType); - } - } - } - Emfac2011VehicleType.setMutableTypes(mutableVehicleType); - Map> finalMapping = new HashMap<>(); - for (SandagAutoModes type : mapping.keySet()) - finalMapping.put(type.name(), mapping.get(type)); - return finalMapping; - } - - void runEmfac2011Program() { - final Path emfacInstallationDir = Paths - .get(properties - .getString(Emfac2011Properties.EMFAC2011_INSTALLATION_DIR_PROPERTY)); - final PathMatcher matcher = FileSystems.getDefault().getPathMatcher( - "glob:*.lnk"); - final List link = new LinkedList<>(); - FileVisitor visitor = new SimpleFileVisitor() { - - @Override - public FileVisitResult visitFile(Path file, - BasicFileAttributes attrs) throws IOException { - Path name = file.getFileName(); - if (name != null && matcher.matches(name)) { - link.add(file); - return FileVisitResult.TERMINATE; - } - return FileVisitResult.CONTINUE; - } - }; - - try { - Files.walkFileTree(emfacInstallationDir, - Collections. emptySet(), 1, visitor); - } catch (IOException e) { - throw new RuntimeIOException(e); - } - - if (link.size() == 0) - throw new IllegalStateException( - "Cannot find Emfac2011 shortcut in " + emfacInstallationDir); - ProcessUtil.runProcess(Arrays.asList("cmd", "/c", link.get(0) - .toString())); - } - - public static void main(String... args) { - double startTime = System.currentTimeMillis(); - // do work - new Emfac2011Runner(args[0]).runEmfac2011(); - // time stamp - LOGGER.info("Emfac2011 completed in: " - + (float) (((System.currentTimeMillis() - startTime) / 1000.0) / 60.0) - + " minutes."); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java deleted file mode 100644 index 1d76b98..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SpeedCategory.java +++ /dev/null @@ -1,73 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -/** - * The {@code EmfacSpeedClass} enum represents the speed categories used in the - * EMFAC2011 model. - * - * @author crf Started 2/9/12 5:58 AM - */ -public enum Emfac2011SpeedCategory -{ - SPEED_0_5(5, "5MPH"), SPEED_5_10(10, "10MPH"), SPEED_10_15(15, "15MPH"), SPEED_15_20(20, - "20MPH"), SPEED_20_25(25, "25MPH"), SPEED_25_30(30, "30MPH"), SPEED_30_35(35, "35MPH"), SPEED_35_40( - 40, "40MPH"), SPEED_40_45(45, "45MPH"), SPEED_45_50(50, "50MPH"), SPEED_50_55(55, - "55MPH"), SPEED_55_60(60, "60MPH"), SPEED_60_65(65, "65MPH"), SPEED_65_70plus(1000, - "70MPH"); // big upper bound - - private final String name; - private final double upperBound; - - private Emfac2011SpeedCategory(double upperBound, String name) - { - this.upperBound = upperBound; - this.name = name; - } - - /** - * Get the EMFAC name for this speed category. - * - * @return this speed category's name. - */ - public String getName() - { - return name; - } - - /** - * Get the {@code SpeedCategory} for a given speed. - * - * @param speed - * The speed in question. - * - * @return the {@code SpeedCategory} for the given speed. - * - * @throws IllegalArgumentException - * if {@code speed} < 0. - */ - public static Emfac2011SpeedCategory getSpeedCategory(double speed) - { - if (speed < 0) - throw new IllegalArgumentException("Negative speeds are not allowed: " + speed); - for (Emfac2011SpeedCategory sc : values()) - if (speed <= sc.upperBound) return sc; - throw new IllegalStateException("Couldn't find speed category for " + speed); - } - - /** - * Get the {@code SpeedCategory} corresponding to an EMFAC name. - * - * @param name - * The EMFAC speed category name. - * - * @return the {@code SpeedCategory} corresponding to {@code name}. - * - * @throws IllegalArgumentException - * if {@code name} is not a valid EMFAC speed category name. - */ - public static Emfac2011SpeedCategory getTypeForName(String name) - { - for (Emfac2011SpeedCategory type : values()) - if (type.name.equals(name)) return type; - throw new IllegalArgumentException("Type for name not found: " + name); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java deleted file mode 100644 index 56a719e..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011SqlUtil.java +++ /dev/null @@ -1,145 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import java.io.FileReader; -import java.io.IOException; -import java.nio.file.Path; -import java.sql.Connection; -import java.sql.DriverManager; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Statement; -import java.util.ArrayList; - -import nl.tudelft.simulation.logger.Logger; - -import com.pb.sawdust.util.exceptions.RuntimeIOException; -import com.pb.sawdust.util.exceptions.RuntimeWrappingException; -import com.pb.sawdust.util.property.PropertyDeluxe; - -/** - * @author crf Started 2/9/12 9:17 AM - * Modified by Wu.Sun@sandag.org 1/21/2014 - */ - -public class Emfac2011SqlUtil { - - private final String connectionUrl; - - public Emfac2011SqlUtil(PropertyDeluxe properties) { - connectionUrl = formConnectionUrl( - properties - .getString(Emfac2011Properties.REPORTS_DATABASE_IPADDRESS_PROPERTY), - properties - .getInt(Emfac2011Properties.REPORTS_DATABASE_PORT_PROPERTY), - properties - .getString(Emfac2011Properties.REPORTS_DATABASE_NAME_PROPERTY), - properties - .hasKey(Emfac2011Properties.REPORTS_DATABASE_USERNAME_PROPERTY) ? properties - .getString(Emfac2011Properties.REPORTS_DATABASE_USERNAME_PROPERTY) - : null, - properties - .hasKey(Emfac2011Properties.REPORTS_DATABASE_PASSWORD_PROPERTY) ? properties - .getString(Emfac2011Properties.REPORTS_DATABASE_PASSWORD_PROPERTY) - : null, - properties - .hasKey(Emfac2011Properties.REPORTS_DATABASE_INSTANCE_PROPERTY) ? properties - .getString(Emfac2011Properties.REPORTS_DATABASE_INSTANCE_PROPERTY) - : null); - } - - public void detemplifyAndRunScript(Path script, String scenarioId, - String scenarioIdToken) { - String s = readFile(script).replace(scenarioIdToken, scenarioId); - try (Connection connection = getConnection(); - Statement statement = connection.createStatement()) { - connection.setAutoCommit(false); - statement.execute(s); - connection.commit(); - } catch (SQLException e) { - throw new RuntimeWrappingException(e); - } - } - - public ArrayList> queryAquavisTables(Path script, - String scenarioId, String scenarioIdToken) { - ArrayList> table = null; - String s = readFile(script).replace(scenarioIdToken, scenarioId); - try (Connection connection = getConnection(); - Statement statement = connection.createStatement()) { - System.out.println("query="+s); - table = extract(statement.executeQuery(s)); - connection.close(); - } catch (SQLException e) { - throw new RuntimeWrappingException(e); - } - return table; - } - - private ArrayList> extract(ResultSet resultSet) - throws SQLException { - ArrayList> table; - int columnCount = resultSet.getMetaData().getColumnCount(); - - if (resultSet.getType() == ResultSet.TYPE_FORWARD_ONLY) - table = new ArrayList>(); - else { - resultSet.last(); - table = new ArrayList>(resultSet.getRow()); - resultSet.beforeFirst(); - } - - for (ArrayList row; resultSet.next(); table.add(row)) { - row = new ArrayList(columnCount); - - for (int c = 1; c <= columnCount; ++c) - row.add(resultSet.getString(c).intern()); - } - return table; - } - - private String readFile(Path file) { - try (FileReader reader = new FileReader(file.toFile())) { - StringBuilder sb = new StringBuilder(); - char[] buffer = new char[8192]; - int readCount; - while ((readCount = reader.read(buffer, 0, buffer.length)) > 0) - sb.append(buffer, 0, readCount); - return sb.toString(); - } catch (IOException e) { - throw new RuntimeIOException(e); - } - } - - private String formConnectionUrl(String ipAddress, int port, - String databaseName, String username, String password, - String instance) { - String url = "jdbc:jtds:sqlserver://" + ipAddress + ":" + port + "/" - + databaseName; - if (username != null) - url += ";user=" + username + ";" + password; // not - // super - // secure, - // btu - // ok - // for - // now - // - - // probably - // will - // use - // SSO - // normally - if (instance != null) - url += ";instance=" + instance; - return url; - } - - private Connection getConnection() throws SQLException { - try { - Class.forName("net.sourceforge.jtds.jdbc.Driver"); - } catch (ClassNotFoundException e) { - throw new RuntimeWrappingException(e); - } - return DriverManager.getConnection(connectionUrl); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java deleted file mode 100644 index 0ec7a68..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/Emfac2011VehicleType.java +++ /dev/null @@ -1,146 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import java.util.Collections; -import java.util.EnumSet; -import java.util.Set; - -/** - * The {@code EmfacVehicleType} enum represents the EMFAC2011 tNCVehicle types. - * - * @author crf Started 2/8/12 8:54 AM - */ -public enum Emfac2011VehicleType -{ - OTHER_BUSES_DSL("All Other Buses - DSL", "OBUS - DSL", false), LDA_DSL("LDA - DSL", - "LDA - DSL", true), LDA_GAS("LDA - GAS", "LDA - GAS", true), LDT1_DSL("LDT1 - DSL", - "LDT1 - DSL", true), LDT1_GAS("LDT1 - GAS", "LDT1 - GAS", true), LDT2_DSL("LDT2 - DSL", - "LDT2 - DSL", true), LDT2_GAS("LDT2 - GAS", "LDT2 - GAS", true), LHD1_DSL("LHD1 - DSL", - "LHDT1 - DSL", true), LHE1_GAS("LHD1 - GAS", "LHDT1 - GAS", true), LHD2_DSL( - "LHD2 - DSL", "LHDT2 - DSL", true), LHD2_GAS("LHD2 - GAS", "LHDT2 - GAS", true), MCY_GAS( - "MCY - GAS", "MCY - GAS", true), MDV_DSL("MDV - DSL", "MDV - DSL", true), MDV_GAS( - "MDV - GAS", "MDV - GAS", true), MH_DSL("MH - DSL", "MH - DSL", false), MH_GAS( - "MH - GAS", "MH - GAS", false), MOTOR_COACH_DSL("Motor Coach - DSL", "OBUS - DSL", - false), OBUS_GAS("OBUS - GAS", "OBUS - GAS", false), PTO_DSL("PTO - DSL", "HHDT - DSL", - false), SBUS_DSL("SBUS - DSL", "SBUS - DSL", false), SBUS_GAS("SBUS - GAS", - "SBUS - GAS", false), T6_AG_DSL("T6 Ag - DSL", "MHDT - DSL", false), T6_CAIRP_HEAVY_DSL( - "T6 CAIRP heavy - DSL", "MHDT - DSL", false), T6_CAIRP_SMALL_DSL( - "T6 CAIRP small - DSL", "MHDT - DSL", false), T6_INSTATE_CONSTRUCTION_HEAVY_DSL( - "T6 instate construction heavy - DSL", "MHDT - DSL", false), T6_INSTATE_CONSTRUCTION_SMALL_DSL( - "T6 instate construction small - DSL", "MHDT - DSL", false), T6_INSTATE_HEAVY_DSL( - "T6 instate heavy - DSL", "MHDT - DSL", false), T6_INSTATE_SMALL_DSL( - "T6 instate small - DSL", "MHDT - DSL", false), T6_OOS_HEAVY_DSL("T6 OOS heavy - DSL", - "MHDT - DSL", false), T6_OOS_SMALL_DSL("T6 OOS small - DSL", "MHDT - DSL", false), T6_PUBLIC_DSL( - "T6 public - DSL", "MHDT - DSL", false), T6_UTILITY_DSL("T6 utility - DSL", - "MHDT - DSL", false), T6TS_GAS("T6TS - GAS", "MHDT - GAS", false), T7_AG_DSL( - "T7 Ag - DSL", "HHDT - DSL", false), T7_CAIRP_DSL("T7 CAIRP - DSL", "HHDT - DSL", false), T7_CAIRP_CONSTRUCTION_DSL( - "T7 CAIRP construction - DSL", "HHDT - DSL", false), T7_NNOOS_DSL("T7 NNOOS - DSL", - "HHDT - DSL", false), T7_NOOS_DSL("T7 NOOS - DSL", "HHDT - DSL", false), T7_OTHER_PORT_DSL( - "T7 other port - DSL", "HHDT - DSL", false), T7_POAK_DSL("T7 POAK - DSL", "HHDT - DSL", - false), T7_POLA_DSL("T7 POLA - DSL", "HHDT - DSL", false), T7_PUBLIC_DSL( - "T7 public - DSL", "HHDT - DSL", false), T7_SINGLE_DSL("T7 Single - DSL", "HHDT - DSL", - false), T7_SINGLE_CONSTRUCTION_DSL("T7 single construction - DSL", "HHDT - DSL", false), T7_SWCV_DSL( - "T7 SWCV - DSL", "HHDT - DSL", false), T7_TRACTOR_DSL("T7 tractor - DSL", "HHDT - DSL", - false), T7_TRACTOR_CONSTRUCTION_DSL("T7 tractor construction - DSL", "HHDT - DSL", - false), T7_UTILITY_DSL("T7 utility - DSL", "HHDT - DSL", false), T7IS_GAS("T7IS - GAS", - "HHDT - GAS", false), UBUS_DSL("UBUS - DSL", "UBUS - DSL", false), UBUS_GAS( - "UBUS - GAS", "UBUS - GAS", false); - - private final String name; - private final String emfac2007Name; - private final boolean isMutableType; - - private Emfac2011VehicleType(String name, String emfac2007Name, boolean isMutableType) - { - this.name = name; - this.emfac2007Name = emfac2007Name; - this.isMutableType = isMutableType; - } - - /** - * Get the name of the tNCVehicle type. This is the name used by the EMFAC2011 - * model. - * - * @return the name of the tNCVehicle type. - */ - public String getName() - { - return name; - } - - /** - * Get this tNCVehicle type's equivalent EMFAC2007 tNCVehicle type name. - * - * @return the EMFAC2007 tNCVehicle type corresponding to this tNCVehicle type. - */ - public String getEmfac2007Name() - { - return emfac2007Name; - } - - /** - * Determine if this tNCVehicle type is mutable. If a tNCVehicle type is mutable, - * then its EMFAC2011 SG inputs may be adjusted to account for travel demand - * model results. - * - * @return {@code true} if this tNCVehicle type is mutable, {@code false} if - * not. - */ - public boolean isMutableType() - { - return mutableTypes.contains(this); - } - - private static Set mutableTypes; - - static - { - Set set = EnumSet.noneOf(Emfac2011VehicleType.class); - for (Emfac2011VehicleType type : Emfac2011VehicleType.values()) - if (type.isMutableType) set.add(type); - setMutableTypes(set); - } - - /** - * Set the mutable tNCVehicle types. This need only be called if the default - * mutable types are unsatisfactory for the particular application; however, - * if it needs to be called, then it should be before any of the - * EMFAC2011/Aquavis processing commences. - * - * @param mutableTypes - * The set of mutable tNCVehicle types for processing. - */ - public static void setMutableTypes(Set mutableTypes) - { - Emfac2011VehicleType.mutableTypes = Collections.unmodifiableSet(mutableTypes); - } - - /** - * Get the tNCVehicle type corresponding to the given (EMFAC2011) name. - * - * @param name - * The tNCVehicle type name used in the EMFAC2011 tNCVehicle type. - * - * @return the tNCVehicle type corresponding to {@code name}. - * - * @throws IllegalArgumentException - * if {@code name} does not correspond to any tNCVehicle type. - */ - public static Emfac2011VehicleType getVehicleType(String name) - { - for (Emfac2011VehicleType type : values()) - if (type.getName().equals(name)) return type; - throw new IllegalArgumentException("No EMFAC tNCVehicle type corresponding to: " + name); - } - - /** - * Get the set of EMFAC2011 tNCVehicle types which are mutable. If a tNCVehicle - * type is mutable, then its EMFAC2011 SG inputs may be adjusted to account - * for travel demand model results. - * - * @return a set holding the mutable EMFAC2011 tNCVehicle types. - */ - public static Set getMutableVehicleTypes() - { - return mutableTypes; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java deleted file mode 100644 index a19b83d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/RunEmfacDialog.java +++ /dev/null @@ -1,174 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -import java.awt.GridBagConstraints; -import java.awt.GridBagLayout; -import java.awt.event.ActionEvent; -import java.awt.event.ActionListener; -import java.awt.event.WindowAdapter; -import java.awt.event.WindowEvent; -import java.nio.file.Path; -import javax.swing.JButton; -import javax.swing.JFrame; -import javax.swing.JLayeredPane; -import javax.swing.JPanel; -import javax.swing.JScrollPane; -import javax.swing.JTextArea; -import com.pb.sawdust.util.exceptions.RuntimeInterruptedException; - -/** - * The {@code HelpDialog} ... - * - * @author crf Started 2/9/12 3:57 PM - */ -class RunEmfacDialog - extends JPanel - implements ActionListener -{ - private static final long serialVersionUID = -3645537208340049132L; - private final JButton button; - private final JLayeredPane pane; - private final Emfac2011Runner emfac2011Runner; - - public RunEmfacDialog(Path outputPath, Emfac2011Runner emfac2011Runner) - { - super(new GridBagLayout()); - - StringBuilder sb = new StringBuilder(); - sb.append("\n ") - .append("Almost there! Press the button to start EMFAC2011, then follow these directions.") - .append("\n\n"); - sb.append(" ") - .append("EMFAC2011 should have booted up. If not, check that the installation directory is defined correctly in the properties file.") - .append("\n\n"); - sb.append(" ").append("The following steps will take you through running the program.") - .append("\n\n"); - sb.append(" ") - .append("1) In the \"Regional Scenarios\" box, hit the \"Load Regional Scenarios (External Files)\" button.") - .append("\n\n"); - sb.append(" ").append("2) Browse to and select: ").append(outputPath).append("\n\n"); - sb.append(" ") - .append("3) The EMFAC2011-SG-Scenario Builder window should appear. Press the \"Save and Continue\" button.") - .append("\n\n"); - sb.append(" ").append("4) A message box will appear. Click \"Yes\"").append("\n\n"); - sb.append(" ") - .append("5) In the EMFAC2011-SG model window, hit the \"Verify Speed Data Quality\" button.") - .append("\n\n"); - sb.append(" ") - .append("6) If there are no errors, hit the \"Continue\" button in the \"Verify Speed Inputs\" window.") - .append("\n\n"); - sb.append(" ") - .append("7) In the EMFAC2011-SG model window, hit the \"Save Scenarios\" button.") - .append("\n\n"); - sb.append(" ") - .append("8) Select the same file that we loaded in step (2). Say \"Yes\" to the question about replacing the file.") - .append("\n\n"); - sb.append(" ").append("9) EMFAC2011 will tell you it saved the input file. Click \"OK\"") - .append("\n\n"); - sb.append(" ") - .append("10) In the EMFAC2011-SG model window, hit the \"Execute Model\" button.") - .append("\n\n"); - sb.append(" ") - .append("11) In the EMFAC2011-SG-Model Execution Options window do the following") - .append("\n"); - sb.append(" ") - .append("\ta) In the \"Input Parameters\" box, the \"Export Default Input Parameters\" check box should NOT be checked.") - .append("\n"); - sb.append(" ") - .append("\tb) In the \"Model Outputs\" box, choose \"XLS\" as the output format, and check the \"Create Additional Summary Outputs\" \n\t checkbox. Leave the \"Create Separate Output Files for Each Regional Scenario\" checkbox unchecked.") - .append("\n"); - sb.append(" ").append("\tc) Hit the \"Start\" button.").append("\n\n"); - sb.append(" ") - .append("12) The EMFAC2011 model should run, and then pop up a dialog box saying it finished. Click \"OK\"") - .append("\n\n"); - sb.append(" ").append("13) Click the \"Exit EMFAC2011-SG\" button.").append("\n\n"); - sb.append(" ").append("14) All done! Close this window when you are finished."); - - JTextArea textArea = new JTextArea(40, 80); - textArea.setEditable(false); - textArea.setText(sb.toString()); - textArea.setCaretPosition(0); - textArea.setLineWrap(true); - textArea.setWrapStyleWord(true); - JScrollPane scrollPane = new JScrollPane(textArea); - GridBagConstraints c = new GridBagConstraints(); - c.gridwidth = GridBagConstraints.REMAINDER; - c.fill = GridBagConstraints.HORIZONTAL; - c.fill = GridBagConstraints.BOTH; - c.weightx = 1.0; - c.weighty = 1.0; - - pane = new JLayeredPane(); - scrollPane.setSize(850, 675); - button = new JButton("Start EMFAC2011"); - button.addActionListener(this); - button.setLocation(500, 15); - button.setSize(150, 20); - pane.add(button, 0, -1); - pane.add(scrollPane); - add(pane); - add(pane, c); - - this.emfac2011Runner = emfac2011Runner; - } - - public static void createAndShowGUI(Path inputFile, Emfac2011Runner emfac2011Runner) - { - final Object lock = new Object(); - final JFrame frame = new JFrame("Run EMFAC2011"); - frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); - frame.add(new RunEmfacDialog(inputFile, emfac2011Runner)); - frame.setSize(800, 700); - frame.setVisible(true); - - Thread thread = new Thread() - { - @Override - public void run() - { - synchronized (lock) - { - while (frame.isVisible()) - { - try - { - lock.wait(); - } catch (InterruptedException e) - { - throw new RuntimeInterruptedException(e); - } - } - } - } - }; - thread.start(); - - frame.addWindowListener(new WindowAdapter() - { - @Override - public void windowClosing(WindowEvent arg0) - { - synchronized (lock) - { - frame.setVisible(false); - lock.notify(); - } - } - }); - try - { - thread.join(); - } catch (InterruptedException e) - { - throw new RuntimeInterruptedException(e); - } - - } - - @Override - public void actionPerformed(ActionEvent e) - { - pane.remove(button); - pane.repaint(); - emfac2011Runner.runEmfac2011Program(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java b/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java deleted file mode 100644 index 4cdb52a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/reporting/emfac2011/SandagAutoModes.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.sandag.abm.reporting.emfac2011; - -/** - * The {@code VehicleType} ... - * - * @author crf Started 12/7/12 3:28 PM - */ -public enum SandagAutoModes -{ - SOV_GP, SOV_PAY, SR2_GP, SR2_HOV, SR2_PAY, SR3_GP, SR3_HOV, SR3_PAY, LHDN, MHDN, HHDN, LHDT, MHDT, HHDT, UBUS -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java deleted file mode 100644 index 2eb422a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactory.java +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.specialevent; - -import java.io.Serializable; -import org.sandag.abm.application.SandagModelStructure; - -/** - * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects - * - * @author Joel Freedman - */ -public class SpecialEventDmuFactory - implements SpecialEventDmuFactoryIf, Serializable -{ - - private SandagModelStructure sandagModelStructure; - - public SpecialEventDmuFactory(SandagModelStructure modelStructure) - { - this.sandagModelStructure = modelStructure; - } - - public SpecialEventTripModeChoiceDMU getSpecialEventTripModeChoiceDMU() - { - return new SpecialEventTripModeChoiceDMU(sandagModelStructure, null); - } - - public SpecialEventOriginChoiceDMU getSpecialEventOriginChoiceDMU() - { - return new SpecialEventOriginChoiceDMU(sandagModelStructure); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java deleted file mode 100644 index 33e5633..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventDmuFactoryIf.java +++ /dev/null @@ -1,13 +0,0 @@ -package org.sandag.abm.specialevent; - -/** - * A DMU factory interface - */ -public interface SpecialEventDmuFactoryIf -{ - - SpecialEventTripModeChoiceDMU getSpecialEventTripModeChoiceDMU(); - - SpecialEventOriginChoiceDMU getSpecialEventOriginChoiceDMU(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java deleted file mode 100644 index ddc0782..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventModel.java +++ /dev/null @@ -1,347 +0,0 @@ -package org.sandag.abm.specialevent; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -/** - * Trips for special events. - * - * This programs models trips to special events using the Event Model framework. - * The Event Model framework generates and distributes trips. These trips are - * put through a mode choice model. User benefits are optionally calculated and - * written to a SUMMIT formatted file. - * - */ -public final class SpecialEventModel -{ - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - - private MatrixDataServerRmi ms; - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - private HashMap rbMap; - private McLogsumsCalculator logsumsCalculator; - private AutoTazSkimsCalculator tazDistanceCalculator; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private boolean seek; - private int traceId; - - private TableDataSet eventData; - private double sampleRate = 1; - - /** - * Default Constructor. - */ - private SpecialEventModel(HashMap rbMap) - { - this.rbMap = rbMap; - mgraManager = MgraDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - - // read the event data - String eventFile = Util.getStringValueFromPropertyMap(rbMap, "specialEvent.event.file"); - eventFile = directory + eventFile; - eventData = readFile(eventFile); - - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "specialEvent.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "specialEvent.trace")); - - } - - /** - * Read the file and return the TableDataSet. - * - * @param fileName - * @return data - */ - private TableDataSet readFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet data; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - data = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return data; - } - - /** - * @return the sampleRate - */ - public double getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(double sampleRate) - { - this.sampleRate = sampleRate; - } - - /** - * Run the Event Model. - * - * The Event Model is run for each mgra specified in the events.file - */ - public void runModel() - { - - SandagModelStructure modelStructure = new SandagModelStructure(); - - SpecialEventDmuFactoryIf dmuFactory = new SpecialEventDmuFactory(modelStructure); - - SpecialEventTourManager tourManager = new SpecialEventTourManager(rbMap, eventData); - - tourManager.generateTours(); - SpecialEventTour[] tours = tourManager.getTours(); - - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - logsumsCalculator = new McLogsumsCalculator(); - logsumsCalculator.setupSkimCalculators(rbMap); - logsumsCalculator.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - SpecialEventOriginChoiceModel originChoiceModel = new SpecialEventOriginChoiceModel(rbMap, - dmuFactory, eventData); - originChoiceModel.calculateSizeTerms(dmuFactory); - originChoiceModel.calculateTazProbabilities(dmuFactory); - - SpecialEventTripModeChoiceModel tripModeChoiceModel = new SpecialEventTripModeChoiceModel( - rbMap, modelStructure, dmuFactory, logsumsCalculator, eventData); - - // Run models for array of tours - for (int i = 0; i < tours.length; ++i) - { - - SpecialEventTour tour = tours[i]; - - // Wu added for sampling tours - double rand = tour.getRandom(); - if (rand > sampleRate) continue; - - if (i < 10 || i % 1000 == 0) logger.info("Processing tour " + (i + 1)); - - if (seek && tour.getID() != traceId) continue; - - if (tour.getID() == traceId) tour.setDebugChoiceModels(true); - originChoiceModel.chooseOrigin(tour); - // generate trips and choose mode for them - SpecialEventTrip[] trips = new SpecialEventTrip[2]; - int tripNumber = 0; - - // generate an outbound trip from the tour origin to the destination - // and choose a mode - trips[tripNumber] = new SpecialEventTrip(tour, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - // generate an inbound trip from the tour destination to the origin - // and choose a mode - trips[tripNumber] = new SpecialEventTrip(tour, false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - // set the trips in the tour object - tour.setTrips(trips); - - } - - tourManager.writeOutputFile(rbMap); - - logger.info("Special Event Model successfully completed!"); - - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * Run Special Events Model. - * - * The Special Events Model generates, distributes, and chooses modes for - * attendees of sporting events and similar activities. - */ - public static void main(String[] args) - { - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Special Event Model")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - SpecialEventModel specialEventModel = new SpecialEventModel(pMap); - - //Wu added for sampling special event tours based on sample rate - float sampleRate = 1.0f; - int iteration = 1; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - logger.info("Special Event Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - specialEventModel.setSampleRate(sampleRate); - - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = specialEventModel.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - specialEventModel.ms = matrixServer; - } else - { - specialEventModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - specialEventModel.ms.testRemote("SpecialEventModel"); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(specialEventModel.ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - specialEventModel.runModel(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java deleted file mode 100644 index 0f418bc..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceDMU.java +++ /dev/null @@ -1,205 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class SpecialEventOriginChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("specialEventModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected float tourDepartPeriod; - protected float tourArrivePeriod; - protected int purpose; - protected double[][] sizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgras) - - protected double nmWalkTimeOut; - protected double nmWalkTimeIn; - protected double nmBikeTimeOut; - protected double nmBikeTimeIn; - protected double lsWgtAvgCostM; - protected double lsWgtAvgCostD; - protected double lsWgtAvgCostH; - - public SpecialEventOriginChoiceDMU(SandagModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - /** - * @return the sizeTerm. The size term is the size of the origin taz. - */ - public double getSizeTerm(int alt) - { - return sizeTerms[purpose][alt]; - } - - /** - * @param sizeTerms - * the sizeTerms to set. The size term is the array of origin taz - * sizes. - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the purpose - */ - public int getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(int purpose) - { - this.purpose = purpose; - } - - public float getTimeOutbound() - { - return tourDepartPeriod; - } - - public float getTimeInbound() - { - return tourArrivePeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(float tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(float tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getSizeTerm", 2); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 2: - returnValue = getSizeTerm(arrayIndex); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java deleted file mode 100644 index 690775a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventOriginChoiceModel.java +++ /dev/null @@ -1,384 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.io.File; -import java.util.ArrayList; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class SpecialEventOriginChoiceModel -{ - - private double[][] mgraSizeTerms; // by - // purpose, - // MGRA - private double[][] tazSizeTerms; // by - // purpose, - // TAZ - private double[][][] mgraProbabilities; // by - // purpose, - // tazNumber, - // mgra - // index - // (sequential, - // 0-based) - private Matrix[] tazProbabilities; // by - // purpose, - // origin - // TAZ, - // destination - // TAZ - - private TableDataSet alternativeData; // the - // alternatives, - // with - // a - // dest - // field - // indicating - // tazNumber - - private transient Logger logger = Logger.getLogger("specialEventModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication destModel; - private UtilityExpressionCalculator sizeTermUEC; - private HashMap rbMap; - private HashMap purposeMap; // string - // is - // purpose, - // int - // is - // alternative - // for - // size - // terms - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of airport model DMUs - */ - public SpecialEventOriginChoiceModel(HashMap rbMap, - SpecialEventDmuFactoryIf dmuFactory, TableDataSet eventData) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String destUecFileName = Util.getStringValueFromPropertyMap(rbMap, - "specialEvent.dc.uec.file"); - destUecFileName = uecFileDirectory + destUecFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "specialEvent.dc.data.page")); - int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "specialEvent.dc.size.page")); - int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "specialEvent.dc.model.page")); - - // read the model pages from the property file, create one choice model - // for each - // get page from property file - - SpecialEventOriginChoiceDMU dcDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); - - // create a ChoiceModelApplication object for the filename, model page - // and data page. - destModel = new ChoiceModelApplication(destUecFileName, modelPage, dataPage, rbMap, - (VariableTable) dcDmu); - - // get the alternative data from the first segment - UtilityExpressionCalculator uec = destModel.getUEC(); - alternativeData = uec.getAlternativeData(); - - // create a UEC to solve size terms for each MGRA - SpecialEventOriginChoiceDMU dmu = dmuFactory.getSpecialEventOriginChoiceDMU(); - sizeTermUEC = new UtilityExpressionCalculator(new File(destUecFileName), sizePage, - dataPage, rbMap, dmu); - - } - - /** - * Calculate size terms - */ - public void calculateSizeTerms(SpecialEventDmuFactoryIf dmuFactory) - { - - logger.info("Calculating Special Event Origin Choice Model Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int maxTaz = tazManager.getMaxTaz(); - int purposes = sizeTermUEC.getNumberOfAlternatives(); - - // set up the map of string purpose to integer purpose - String[] altNames = sizeTermUEC.getAlternativeNames(); - purposeMap = new HashMap(); - for (int i = 0; i < altNames.length; ++i) - { - purposeMap.put(altNames[i], i); - } - - mgraSizeTerms = new double[purposes][maxMgra + 1]; - tazSizeTerms = new double[purposes][maxTaz + 1]; - IndexValues iv = new IndexValues(); - SpecialEventOriginChoiceDMU aDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = sizeTermUEC.solve(iv, aDmu, null); - - // store the size terms - for (int purpose = 0; purpose < purposes; ++purpose) - { - - mgraSizeTerms[purpose][mgra] = utilities[purpose]; - tazSizeTerms[purpose][taz] += utilities[purpose]; - } - - } - - // now calculate probability of selecting each MGRA within each TAZ for - // SOA - mgraProbabilities = new double[purposes][maxTaz + 1][]; - int[] tazs = tazManager.getTazs(); - - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazs.length; ++taz) - { - int tazNumber = tazs[taz]; - int[] mgraArray = tazManager.getMgraArray(tazNumber); - - // initialize the vector of mgras for this purpose-taz - mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; - - // now calculate the cumulative probability distribution - double lastProb = 0.0; - for (int mgra = 0; mgra < mgraArray.length; ++mgra) - { - - int mgraNumber = mgraArray[mgra]; - if (tazSizeTerms[purpose][tazNumber] > 0.0) - mgraProbabilities[purpose][tazNumber][mgra] = lastProb - + mgraSizeTerms[purpose][mgraNumber] - / tazSizeTerms[purpose][tazNumber]; - lastProb = mgraProbabilities[purpose][tazNumber][mgra]; - } - if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) - logger.info("Error: purpose " + purpose + " taz " + tazNumber - + " cum prob adds up to " + lastProb); - } - - } - - // calculate logged size terms for mgra and taz vectors to be used in - // dmu - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) - if (tazSizeTerms[purpose][taz] > 0.0) - tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); - - for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) - if (mgraSizeTerms[purpose][mgra] > 0.0) - mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); - - } - logger.info("Finished Calculating Special Event Tour Origin Choice Model Size Terms"); - } - - /** - * Calculate taz probabilities. This method initializes and calculates the - * tazProbabilities array. - */ - public void calculateTazProbabilities(SpecialEventDmuFactoryIf dmuFactory) - { - - if (tazSizeTerms == null) - { - logger.error("Error: attemping to execute SpecialEventTourOriginChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); - throw new RuntimeException(); - } - - logger.info("Calculating Special Event Model TAZ Probabilities Arrays"); - - // initialize taz probabilities array - int purposes = tazSizeTerms.length; - - // initialize the arrays - tazProbabilities = new Matrix[purposes]; - - // iterate through the alternatives in the alternatives file and set the - // size term for each alternative - UtilityExpressionCalculator modelUEC = destModel.getUEC(); - TableDataSet altData = modelUEC.getAlternativeData(); - - SpecialEventOriginChoiceDMU dcDmu = dmuFactory.getSpecialEventOriginChoiceDMU(); - dcDmu.setSizeTerms(tazSizeTerms); - - // iterate through purposes - for (int purpose = 0; purpose < purposes; ++purpose) - { - - tazProbabilities[purpose] = new Matrix("Prob_Matrix", "Probability Matrix", - altData.getRowCount() + 1, altData.getRowCount() + 1); - int[] tazs = altData.getColumnAsInt("dest"); - tazProbabilities[purpose].setExternalNumbersZeroBased(tazs); - - // iterate through destination zones, solve the UEC for all origins - // and store the results in the matrix - for (int taz = 0; taz < tazs.length; ++taz) - { - - int destinationTaz = (int) tazs[taz]; - - // set origin taz in dmu (destination set in UEC by alternative) - dcDmu.setDmuIndexValues(0, 0, 0, destinationTaz, false); - - dcDmu.setPurpose(purpose); - - // Calculate utilities & probabilities - destModel.computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); - - // Store probabilities (by purpose) - double[] probabilities = destModel.getCumulativeProbabilities(); - - for (int i = 0; i < probabilities.length; ++i) - { - - double cumProb = probabilities[i]; - int originTaz = (int) altData.getValueAt(i + 1, "dest"); - tazProbabilities[purpose] - .setValueAt(originTaz, destinationTaz, (float) cumProb); - } - } - } - logger.info("Finished Calculating Special Event Model TAZ Probabilities Arrays"); - } - - /** - * Choose an MGRA - * - * @param eventType - * Event type corresponding to size term - * @param destinationMgra - * MGRA of destination - * @param random - * Random number - * @return The chosen MGRA number - */ - public int chooseMGRA(String eventType, int destinationMgra, double random, boolean debug) - { - - int destinationTaz = mgraManager.getTaz(destinationMgra); - int purpose = purposeMap.get(eventType); - - if (debug) - { - logger.info("Random number " + random); - logger.info("Purpose " + purpose); - logger.info("Destination TAZ " + destinationTaz); - - } - - // first find a TAZ and station - Matrix tazCumProb = tazProbabilities[purpose]; - double altProb = 0; - double cumProb = 0; - int originTaz = -1; - for (int i = 0; i < tazCumProb.getColumnCount(); ++i) - { - originTaz = (int) tazCumProb.getExternalColumnNumber(i); - if (tazCumProb.getValueAt(originTaz, destinationTaz) > random) - { // the - // probabilities - // are - // stored - // column-wise - if (i != 0) - { - cumProb = tazCumProb.getValueAt(originTaz, - tazCumProb.getExternalColumnNumber(i - 1)); - altProb = tazCumProb.getValueAt(originTaz, destinationTaz) - - tazCumProb.getValueAt(originTaz, - tazCumProb.getExternalColumnNumber(i - 1)); - } else - { - altProb = tazCumProb.getValueAt(originTaz, destinationTaz); - } - break; - } - } - - // get the taz number of the alternative, and an array of mgras in that - // taz - int[] mgraArray = tazManager.getMgraArray(originTaz); - - // now find an MGRA in the taz corresponding to the random number drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives probability - int mgraNumber = 0; - - if (debug) - { - logger.info("Chosen origin TAZ " + originTaz); - } - - double[] mgraCumProb = mgraProbabilities[purpose][originTaz]; - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > random) - { - mgraNumber = mgraArray[i]; - } - } - if (debug) logger.info("Chose origin MGRA " + mgraNumber); - // return the chosen MGRA number - return mgraNumber; - } - - /** - * Choose origin MGRAs for a special event tour. - * - * @param tour - * A Special Event tour - */ - public void chooseOrigin(SpecialEventTour tour) - { - - String eventType = tour.getEventType(); - double random = tour.getRandom(); - int destinationMgra = tour.getDestinationMGRA(); - int mgra = chooseMGRA(eventType, destinationMgra, random, tour.getDebugChoiceModels()); - tour.setOriginMGRA(mgra); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java deleted file mode 100644 index ba05b9b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTour.java +++ /dev/null @@ -1,291 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.io.Serializable; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Household; -import com.pb.common.math.MersenneTwister; - -public class SpecialEventTour - implements Serializable -{ - - private MersenneTwister random; - private int ID; - - private byte eventNumber; - private String eventType; - // following variables determined via simulation - private int income; - private int partySize; - - private SpecialEventTrip[] trips; - - private int departTime; - private int arriveTime; - - private boolean debugChoiceModels; - - // following variables chosen via choice models - private int originMGRA; - private int destinationMGRA; - private byte tourMode; - - private float valueOfTime; - - /** - * Public constructor. - * - * @param seed - * A seed for the random number generator. - */ - public SpecialEventTour(long seed) - { - - random = new MersenneTwister(seed); - } - - /** - * @return the iD - */ - public int getID() - { - return ID; - } - - /** - * @param iD - * the iD to set - */ - public void setID(int iD) - { - ID = iD; - } - - /** - * @return the eventNumber - */ - public byte getEventNumber() - { - return eventNumber; - } - - /** - * @param eventNumber - * the eventNumber to set - */ - public void setEventNumber(byte eventNumber) - { - this.eventNumber = eventNumber; - } - - /** - * @return the departTime - */ - public int getDepartTime() - { - return departTime; - } - - /** - * @param departTime - * the departTime to set - */ - public void setDepartTime(int departTime) - { - this.departTime = departTime; - } - - public SpecialEventTrip[] getTrips() - { - return trips; - } - - /** - * @return the eventType - */ - public String getEventType() - { - return eventType; - } - - /** - * @param eventType - * the eventType to set - */ - public void setEventType(String eventType) - { - this.eventType = eventType; - } - - /** - * @return the income - */ - public int getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(int income) - { - this.income = income; - } - - /** - * @return the partySize - */ - public int getPartySize() - { - return partySize; - } - - /** - * @param partySize - * the partySize to set - */ - public void setPartySize(int partySize) - { - this.partySize = partySize; - } - - public void setTrips(SpecialEventTrip[] trips) - { - this.trips = trips; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() - { - return originMGRA; - } - - /** - * @param originMGRA - * the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) - { - this.originMGRA = originMGRA; - } - - /** - * @return the tour mode - */ - public byte getTourMode() - { - return tourMode; - } - - /** - * @param mode - * the tour mode to set - */ - public void setTourMode(byte mode) - { - this.tourMode = mode; - } - - /** - * Get a random number from the parties random class. - * - * @return A random number. - */ - public double getRandom() - { - return random.nextDouble(); - } - - /** - * @return the debugChoiceModels - */ - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - /** - * @param debugChoiceModels - * the debugChoiceModels to set - */ - public void setDebugChoiceModels(boolean debugChoiceModels) - { - this.debugChoiceModels = debugChoiceModels; - } - - - - /** - * Get the number of outbound stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberOutboundStops() - { - return 0; - - } - - /** - * Get the number of return stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberInboundStops() - { - return 0; - - } - - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() - { - return destinationMGRA; - } - - /** - * @param destinationMGRA - * the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) - { - this.destinationMGRA = destinationMGRA; - } - - public void setArriveTime(int arriveTime) - { - this.arriveTime = arriveTime; - } - - public int getArriveTime() - { - return arriveTime; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public void logTourObject(Logger logger, int totalChars) - { - - Household.logHelper(logger, "tourId: ", ID, totalChars); - Household.logHelper(logger, "Event type: ", eventType, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); - Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); - Household.logHelper(logger, "tourMode: ", tourMode, totalChars); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java deleted file mode 100644 index 9e38021..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTourManager.java +++ /dev/null @@ -1,342 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; - -public class SpecialEventTourManager -{ - - private TableDataSet eventData; - private TableDataSet incomeData; - private TableDataSet partySizeData; - private HashMap rbMap; - private static Logger logger = Logger.getLogger("specialEventModel"); - - private SpecialEventTour[] tours; - private MersenneTwister randomGenerator; - private SandagModelStructure sandagStructure; - private boolean saveUtilsAndProbs; - - /** - * Default Constructor. - */ - public SpecialEventTourManager(HashMap rbMap, TableDataSet eventData) - { - this.rbMap = rbMap; - randomGenerator = new MersenneTwister(10001); - saveUtilsAndProbs = Util.getBooleanValueFromPropertyMap(rbMap, - "specialEvent.saveUtilsAndProbs"); - this.eventData = eventData; - sandagStructure = new SandagModelStructure(); - - } - - /** - * Generate special event tours. - */ - public void generateTours() - { - - // read the party size data - String partySizeFile = Util.getStringValueFromPropertyMap(rbMap, - "specialEvent.partySize.file"); - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - partySizeFile = directory + partySizeFile; - partySizeData = readFile(partySizeFile); - - // read the income data - String incomeFile = Util.getStringValueFromPropertyMap(rbMap, "specialEvent.income.file"); - incomeFile = directory + incomeFile; - incomeData = readFile(incomeFile); - - ArrayList eventTourList = new ArrayList(); - - int eventTours = 0; - for (int i = 1; i <= eventData.getRowCount(); ++i) - { - - int eventMgra = (int) eventData.getValueAt(i, "MGRA"); - int attendance = (int) eventData.getValueAt(i, "Attendance"); - String eventType = eventData.getStringValueAt(i, "EventType"); - int startPeriod = (int) eventData.getValueAt(i, "StartPeriod"); - int endPeriod = (int) eventData.getValueAt(i, "EndPeriod"); - - // generate tours for the event - for (int j = 0; j < attendance; ++j) - { - - ++eventTours; - - long randomSeed = getRandomSeed(eventTours); - SpecialEventTour tour = new SpecialEventTour(randomSeed); - tour.setID(eventTours); - tour.setEventNumber((byte) i); - tour.setDestinationMGRA(eventMgra); - tour.setDepartTime(startPeriod); - tour.setArriveTime(endPeriod); - tour.setEventType(eventType); - - // choose income and party size for the tour - int income = chooseIncome(randomGenerator.nextDouble(), eventType); - int partySize = choosePartySize(randomGenerator.nextDouble(), eventType); - - tour.setIncome(income); - tour.setPartySize(partySize); - - eventTourList.add(tour); - } - } - - // convert the ArrayList to an array - tours = new SpecialEventTour[eventTourList.size()]; - for (int i = 0; i < tours.length; ++i) - tours[i] = eventTourList.get(i); - - } - - /** - * Simulate income from the income data table. - * - * @param random - * a uniformly-distributed random number. - * @param eventType - * a string identifying the type of event, which should be a - * column in the income data table - * @return income chosen - */ - public int chooseIncome(double random, String eventType) - { - - int income = -1; - double cumProb = 0; - for (int i = 1; i <= incomeData.getRowCount(); ++i) - { - cumProb += incomeData.getValueAt(i, eventType); - if (random < cumProb) - { - income = (int) incomeData.getValueAt(i, "Income"); - break; - } - } - return income; - } - - /** - * Simulate party size from the party size data table. - * - * @param random - * a uniformly-distributed random number. - * @param eventType - * a string identifying the type of event, which should be a - * column in the party size data table - * @return party size chosen - */ - public int choosePartySize(double random, String eventType) - { - - int partySize = -1; - double cumProb = 0; - for (int i = 1; i <= partySizeData.getRowCount(); ++i) - { - cumProb += partySizeData.getValueAt(i, eventType); - if (random < cumProb) - { - partySize = (int) partySizeData.getValueAt(i, "PartySize"); - break; - } - } - return partySize; - } - - /** - * Read the file and return the TableDataSet. - * - * @param fileName - * @return data - */ - private TableDataSet readFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet data; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - data = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return data; - } - - /** - * Create a text file and write all records to the file. - * - */ - public void writeOutputFile(HashMap rbMap) - { - - // Open file and print header - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String tourFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "specialEvent.tour.output.file"); - String tripFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "specialEvent.trip.output.file"); - - logger.info("Writing special event tours to file " + tourFileName); - logger.info("Writing special event trips to file " + tripFileName); - - PrintWriter tourWriter = null; - try - { - tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tourFileName + " for writing\n"); - throw new RuntimeException(); - } - String tourHeaderString = new String( - "id,eventNumber,eventType,income,partySize,departTime,arriveTime,originMGRA,destinationMGRA,tourMode,valueOfTime\n"); - tourWriter.print(tourHeaderString); - - PrintWriter tripWriter = null; - try - { - tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tripFileName + " for writing\n"); - throw new RuntimeException(); - } - String tripHeaderString = new String( - "tourID,tripID,originMGRA,destinationMGRA,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,boardingTap,alightingTap,set,valueOfTime"); - - // Iterate through the array, printing records to the file - for (int i = 0; i < tours.length; ++i) - { - - SpecialEventTour tour = tours[i]; - - SpecialEventTrip[] trips = tours[i].getTrips(); - - if (trips == null) continue; - - writeTour(tour, tourWriter); - - // if this is the first record, and we are saving utils and probs, - // append a line to the trip file header - if (i == 0) - { - - if (saveUtilsAndProbs) - { - float[] utils = trips[0].getModeUtilities(); - String header = ""; - for (int j = 0; j < utils.length; ++j) - header += ",util_" + j; - for (int j = 0; j < utils.length; ++j) - header += ",prob_" + j; - tripWriter.print(tripHeaderString + header + "\n"); - } else tripWriter.print(tripHeaderString + "\n"); - } - for (int j = 0; j < trips.length; ++j) - { - writeTrip(tour, trips[j], j + 1, tripWriter); - } - } - - tourWriter.close(); - tripWriter.close(); - - } - - /** - * Write the tour to the PrintWriter - * - * @param tour - * @param writer - */ - private void writeTour(SpecialEventTour tour, PrintWriter writer) - { - String record = new String(tour.getID() + "," + tour.getEventNumber() + "," - + tour.getEventType() + "," + tour.getIncome() + "," + tour.getPartySize() + "," - + tour.getDepartTime() + "," + tour.getArriveTime() + "," + tour.getOriginMGRA() - + "," + tour.getDestinationMGRA() + "," + tour.getTourMode() + "," - + String.format("%9.2f",tour.getValueOfTime()) + "\n"); - writer.print(record); - - } - - /** - * Write the trip to the PrintWriter - * - * @param tour - * @param trip - * @param tripNumber - * @param writer - */ - private void writeTrip(SpecialEventTour tour, SpecialEventTrip trip, int tripNumber, - PrintWriter writer) - { - - String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginMgra() - + "," + trip.getDestinationMgra() + "," + trip.isInbound() + "," - + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() - + "," + trip.getPeriod() + "," + trip.getTripMode() + "," - + trip.getBoardTap() + "," + trip.getAlightTap() +"," + trip.getSet()+ "," - + String.format("%9.2f",tour.getValueOfTime())); - - - if (saveUtilsAndProbs) - { - - String utilRecord = new String(); - float[] utils = trip.getModeUtilities(); - for (int i = 0; i < utils.length; ++i) - utilRecord += ("," + String.format("%9.5f", utils[i])); - float[] probs = trip.getModeProbabilities(); - for (int i = 0; i < probs.length; ++i) - utilRecord += ("," + String.format("%9.5f", probs[i])); - record = record + utilRecord; - } - writer.print(record + "\n"); - } - /** - * get special event tours. - * - * @return - */ - public SpecialEventTour[] getTours() - { - return tours; - } - - /** - * Calculate and return a random number seed for the tour. - * - * @param eventID - * @return - */ - public long getRandomSeed(int eventID) - { - - long seed = (eventID * 10 + 100001); - return seed; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java deleted file mode 100644 index cc34fa1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTrip.java +++ /dev/null @@ -1,292 +0,0 @@ -package org.sandag.abm.specialevent; - -public class SpecialEventTrip -{ - - private int originMgra; - private int destinationMgra; - private int tripMode; - private byte period; - private boolean inbound; - private boolean firstTrip; - private boolean lastTrip; - private boolean originIsTourDestination; - private boolean destinationIsTourDestination; - - private float[] modeProbabilities; - private float[] modeUtilities; - - private int boardTap; - private int alightTap; - private int set = -1; - - /** - * Default constructor; nothing initialized. - */ - public SpecialEventTrip() - { - - } - - /** - * Create a cross border trip from a tour leg (no stops). - * - * @param tour - * The tour. - * @param outbound - * Outbound direction - */ - public SpecialEventTrip(SpecialEventTour tour, boolean outbound) - { - - initializeFromTour(tour, outbound); - - } - - /** - * Initilize from the tour. - * - * @param tour - * The tour. - * @param outbound - * Outbound direction. - */ - public void initializeFromTour(SpecialEventTour tour, boolean outbound) - { - // Note: mode is unknown - if (outbound) - { - this.originMgra = tour.getOriginMGRA(); - this.destinationMgra = tour.getDestinationMGRA(); - this.period = (byte) tour.getDepartTime(); - this.inbound = false; - this.firstTrip = true; - this.lastTrip = false; - this.originIsTourDestination = false; - this.destinationIsTourDestination = true; - } else - { - this.originMgra = tour.getDestinationMGRA(); - this.destinationMgra = tour.getOriginMGRA(); - this.period = (byte) tour.getArriveTime(); - this.inbound = true; - this.firstTrip = false; - this.lastTrip = true; - this.originIsTourDestination = true; - this.destinationIsTourDestination = false; - } - - } - - /** - * @return the period - */ - public byte getPeriod() - { - return period; - } - - /** - * @param period - * the period to set - */ - public void setPeriod(byte period) - { - this.period = period; - } - - /** - * @return the originMgra - */ - public int getOriginMgra() - { - return originMgra; - } - - /** - * @param originMgra - * the originMgra to set - */ - public void setOriginMgra(int originMgra) - { - this.originMgra = originMgra; - } - - /** - * @return the destinationMgra - */ - public int getDestinationMgra() - { - return destinationMgra; - } - - /** - * @param destinationMgra - * the destinationMgra to set - */ - public void setDestinationMgra(int destinationMgra) - { - this.destinationMgra = destinationMgra; - } - - /** - * @return the tripMode - */ - public int getTripMode() - { - return tripMode; - } - - /** - * @param tripMode - * the tripMode to set - */ - public void setTripMode(int tripMode) - { - this.tripMode = tripMode; - } - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the firstTrip - */ - public boolean isFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(boolean firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public boolean isLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(boolean lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the originIsTourDestination - */ - public boolean isOriginIsTourDestination() - { - return originIsTourDestination; - } - - /** - * @param originIsTourDestination - * the originIsTourDestination to set - */ - public void setOriginIsTourDestination(boolean originIsTourDestination) - { - this.originIsTourDestination = originIsTourDestination; - } - - /** - * @return the destinationIsTourDestination - */ - public boolean isDestinationIsTourDestination() - { - return destinationIsTourDestination; - } - - /** - * @param destinationIsTourDestination - * the destinationIsTourDestination to set - */ - public void setDestinationIsTourDestination(boolean destinationIsTourDestination) - { - this.destinationIsTourDestination = destinationIsTourDestination; - } - - /** - * @return the modeProbabilities - */ - public float[] getModeProbabilities() - { - return modeProbabilities; - } - - /** - * @param modeProbabilities - * the modeProbabilities to set - */ - public void setModeProbabilities(float[] modeProbabilities) - { - this.modeProbabilities = modeProbabilities; - } - - /** - * @return the modeUtilities - */ - public float[] getModeUtilities() - { - return modeUtilities; - } - - /** - * @param modeUtilities - * the modeUtilities to set - */ - public void setModeUtilities(float[] modeUtilities) - { - this.modeUtilities = modeUtilities; - } - - public int getBoardTap() { - return boardTap; - } - - public void setBoardTap(int boardTap) { - this.boardTap = boardTap; - } - - public int getAlightTap() { - return alightTap; - } - - public void setAlightTap(int alightTap) { - this.alightTap = alightTap; - } - - public int getSet() { - return set; - } - - public void setSet(int set) { - this.set = set; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java deleted file mode 100644 index 2ac573b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceDMU.java +++ /dev/null @@ -1,451 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class SpecialEventTripModeChoiceDMU - implements Serializable, VariableTable -{ - protected transient Logger logger = Logger.getLogger(SpecialEventTripModeChoiceDMU.class); - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected int tourDepartPeriod; - protected int tourArrivePeriod; - protected int tripPeriod; - protected int tripOrigIsTourDest; - protected int tripDestIsTourDest; - protected float parkingCost; - protected float parkingTime; - protected int income; - protected int partySize; - - protected double nmWalkTime; - protected double nmBikeTime; - - protected double ivtCoeff; - protected double costCoeff; - - protected double walkTransitLogsum; - protected double pnrTransitLogsum; - protected double knrTransitLogsum; - - protected int outboundHalfTourDirection; - - public SpecialEventTripModeChoiceDMU(SandagModelStructure modelStructure, Logger aLogger) - { - if (aLogger == null) - { - aLogger = Logger.getLogger("specialEventModel"); - } - logger = aLogger; - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the tripPeriod - */ - public int getTripPeriod() - { - return tripPeriod; - } - - /** - * @param tripPeriod - * the tripPeriod to set - */ - public void setTripPeriod(int tripPeriod) - { - this.tripPeriod = tripPeriod; - } - - /** - * @return the tripOrigIsTourDest - */ - public int getTripOrigIsTourDest() - { - return tripOrigIsTourDest; - } - - /** - * @param tripOrigIsTourDest - * the tripOrigIsTourDest to set - */ - public void setTripOrigIsTourDest(int tripOrigIsTourDest) - { - this.tripOrigIsTourDest = tripOrigIsTourDest; - } - - /** - * @return the tripDestIsTourDest - */ - public int getTripDestIsTourDest() - { - return tripDestIsTourDest; - } - - /** - * @param tripDestIsTourDest - * the tripDestIsTourDest to set - */ - public void setTripDestIsTourDest(int tripDestIsTourDest) - { - this.tripDestIsTourDest = tripDestIsTourDest; - } - - /** - * @return the outboundHalfTourDirection - */ - public int getOutboundHalfTourDirection() - { - return outboundHalfTourDirection; - } - - /** - * @param outboundHalfTourDirection - * the outboundHalfTourDirection to set - */ - public void setOutboundHalfTourDirection(int outboundHalfTourDirection) - { - this.outboundHalfTourDirection = outboundHalfTourDirection; - } - - /** - * @return the tourDepartPeriod - */ - public int getTourDepartPeriod() - { - return tourDepartPeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(int tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(int tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - /** - * @return the tourArrivePeriod - */ - public int getTourArrivePeriod() - { - return tourArrivePeriod; - } - - public double getNm_walkTime() - { - return nmWalkTime; - } - - public void setNonMotorizedWalkTime(double nmWalkTime) - { - this.nmWalkTime = nmWalkTime; - } - - public void setNonMotorizedBikeTime(double nmBikeTime) - { - this.nmBikeTime = nmBikeTime; - } - - public double getNm_bikeTime() - { - return nmBikeTime; - } - - /** - * @return the parkingCost - */ - public float getParkingCost() - { - return parkingCost; - } - - /** - * @param parkingCost - * the parkingCost to set - */ - public void setParkingCost(float parkingCost) - { - this.parkingCost = parkingCost; - } - - /** - * @return the parkingTime - */ - public float getParkingTime() - { - return parkingTime; - } - - /** - * @param parkingTime - * the parkingTime to set - */ - public void setParkingTime(float parkingTime) - { - this.parkingTime = parkingTime; - } - - /** - * @return the income - */ - public int getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(int income) - { - this.income = income; - } - - /** - * @return the partySize - */ - public int getPartySize() - { - return partySize; - } - - /** - * @param partySize - * the partySize to set - */ - public void setPartySize(int partySize) - { - this.partySize = partySize; - } - - public double getNmWalkTime() { - return nmWalkTime; - } - - public void setNmWalkTime(double nmWalkTime) { - this.nmWalkTime = nmWalkTime; - } - - public double getNmBikeTime() { - return nmBikeTime; - } - - public void setNmBikeTime(double nmBikeTime) { - this.nmBikeTime = nmBikeTime; - } - - public double getIvtCoeff() { - return ivtCoeff; - } - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public double getCostCoeff() { - return costCoeff; - } - - public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; - } - - public double getWalkTransitLogsum() { - return walkTransitLogsum; - } - - public void setWalkTransitLogsum(double walkTransitLogsum) { - this.walkTransitLogsum = walkTransitLogsum; - } - - public double getPnrTransitLogsum() { - return pnrTransitLogsum; - } - - public void setPnrTransitLogsum(double pnrTransitLogsum) { - this.pnrTransitLogsum = pnrTransitLogsum; - } - - public double getKnrTransitLogsum() { - return knrTransitLogsum; - } - - public void setKnrTransitLogsum(double knrTransitLogsum) { - this.knrTransitLogsum = knrTransitLogsum; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTourDepartPeriod", 0); - methodIndexMap.put("getTourArrivePeriod", 1); - methodIndexMap.put("getTripPeriod", 2); - methodIndexMap.put("getParkingCost", 3); - methodIndexMap.put("getParkingTime", 4); - methodIndexMap.put("getTripOrigIsTourDest", 5); - methodIndexMap.put("getTripDestIsTourDest", 6); - methodIndexMap.put("getIncome", 7); - methodIndexMap.put("getPartySize", 8); - - methodIndexMap.put("getIvtCoeff", 60); - methodIndexMap.put("getCostCoeff", 61); - - methodIndexMap.put("getWalkSetLogSum", 62); - methodIndexMap.put("getPnrSetLogSum", 63); - methodIndexMap.put("getKnrSetLogSum", 64); - - methodIndexMap.put("getNm_walkTime", 90); - methodIndexMap.put("getNm_bikeTime", 91); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTourDepartPeriod(); - break; - case 1: - returnValue = getTourArrivePeriod(); - break; - case 2: - returnValue = getTripPeriod(); - break; - case 3: - returnValue = getParkingCost(); - break; - case 4: - returnValue = getParkingTime(); - break; - case 5: - returnValue = getTripOrigIsTourDest(); - break; - case 6: - returnValue = getTripDestIsTourDest(); - break; - case 7: - returnValue = getIncome(); - break; - case 8: - returnValue = getPartySize(); - break; - case 60: - returnValue = getIvtCoeff(); - break; - case 61: - returnValue = getCostCoeff(); - break; - case 62: - returnValue = getWalkTransitLogsum(); - break; - case 63: - returnValue = getPnrTransitLogsum(); - break; - case 64: - returnValue = getKnrTransitLogsum(); - break; - case 90: - returnValue = getNm_walkTime(); - break; - case 91: - returnValue = getNm_bikeTime(); - break; - default: - logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - - - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java deleted file mode 100644 index 8680d17..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripModeChoiceModel.java +++ /dev/null @@ -1,284 +0,0 @@ -package org.sandag.abm.specialevent; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class SpecialEventTripModeChoiceModel -{ - - private transient Logger logger = Logger.getLogger("specialEventModel"); - - private AutoAndNonMotorizedSkimsCalculator anm; - private McLogsumsCalculator logsumHelper; - private ModelStructure modelStructure; - private TazDataManager tazs; - private MgraDataManager mgraManager; - - private SpecialEventTripModeChoiceDMU dmu; - private ChoiceModelApplication tripModeChoiceModel; - private boolean saveUtilsAndProbs; - double logsum = 0; - TableDataSet eventData; - private TripModeChoiceDMU mcDmuObject; - - private static final String PROPERTIES_UEC_DATA_SHEET = "specialEvent.trip.mc.data.page"; - private static final String PROPERTIES_UEC_MODEL_SHEET = "specialEvent.trip.mc.model.page"; - private static final String PROPERTIES_UEC_FILE = "specialEvent.trip.mc.uec.file"; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public SpecialEventTripModeChoiceModel(HashMap propertyMap, - ModelStructure myModelStructure, SpecialEventDmuFactoryIf dmuFactory, - McLogsumsCalculator myLogsumHelper, TableDataSet eventData) - { - tazs = TazDataManager.getInstance(propertyMap); - mgraManager = MgraDataManager.getInstance(propertyMap); - - modelStructure = myModelStructure; - logsumHelper = myLogsumHelper; - this.eventData = eventData; - - SandagModelStructure modelStructure = new SandagModelStructure(); - mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); - - setupTripModeChoiceModel(propertyMap, dmuFactory); - saveUtilsAndProbs = Util.getBooleanValueFromPropertyMap(propertyMap, - "specialEvent.saveUtilsAndProbs"); - - } - - /** - * Read the UEC file and set up the trip mode choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupTripModeChoiceModel(HashMap propertyMap, - SpecialEventDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up Special Event trip mode choice model.")); - - dmu = dmuFactory.getSpecialEventTripModeChoiceDMU(); - - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_DATA_SHEET)); - int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_MODEL_SHEET)); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); - tripModeUecFile = uecPath + tripModeUecFile; - logger.info(tripModeUecFile); - - tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, - propertyMap, (VariableTable) dmu); - logger.info(String.format("Finished setting up Special Event trip mode choice model.")); - } - - /** - * Calculate utilities and return logsum for the tour and stop. - * - * @param tour - * @param trip - */ - public double computeUtilities(SpecialEventTour tour, SpecialEventTrip trip) - { - - setDmuAttributes(tour, trip); - - tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - tour.logTourObject(logger, 100); - tripModeChoiceModel.logUECResults(logger, "Special Event trip mode choice model"); - - } - - logsum = tripModeChoiceModel.getLogsum(); - - if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); - - return logsum; - - } - - /** - * Choose a mode and store in the trip object. - * - * @param tour - * SpecialEventTour - * @param trip - * SpecialEventTrip - * - */ - public void chooseMode(SpecialEventTour tour, SpecialEventTrip trip) - { - - computeUtilities(tour, trip); - - double rand = tour.getRandom(); - int mode = tripModeChoiceModel.getChoiceResult(rand); - - trip.setTripMode(mode); - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - - float vot = 0.0f; - - if(modelStructure.getTourModeIsS2(mode)){ - int votIndex = uec.lookupVariableIndex("votS2"); - vot = (float) uec.getValueForIndex(votIndex); - }else if (modelStructure.getTourModeIsS3(mode)){ - int votIndex = uec.lookupVariableIndex("votS3"); - vot = (float) uec.getValueForIndex(votIndex); - }else{ - int votIndex = uec.lookupVariableIndex("vot"); - vot = (float) uec.getValueForIndex(votIndex); - } - tour.setValueOfTime(vot); - - if(mode>=9){ - double[][] bestTapPairs = null; - - if (mode == 9){ - bestTapPairs = logsumHelper.getBestWtwTripTaps(); - } - else if (mode==10||mode==11){ - if (!trip.isInbound()) - bestTapPairs = logsumHelper.getBestDtwTripTaps(); - else - bestTapPairs = logsumHelper.getBestWtdTripTaps(); - } - double rn = tour.getRandom(); - int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); - int boardTap = (int) bestTapPairs[pathIndex][0]; - int alightTap = (int) bestTapPairs[pathIndex][1]; - int set = (int) bestTapPairs[pathIndex][2]; - trip.setBoardTap(boardTap); - trip.setAlightTap(alightTap); - trip.setSet(set); - } - - - if (tour.getDebugChoiceModels()) - { - logger.info("Chose mode " + mode + " with random number " + rand); - } - - if (saveUtilsAndProbs) - { - double[] probs = tripModeChoiceModel.getProbabilities(); - float[] localProbs = new float[probs.length]; - for (int i = 0; i < probs.length; ++i) - localProbs[i] = (float) probs[i]; - - double[] utils = tripModeChoiceModel.getUtilities(); - float[] localUtils = new float[utils.length]; - for (int i = 0; i < utils.length; ++i) - localUtils[i] = (float) utils[i]; - - trip.setModeUtilities(localUtils); - trip.setModeProbabilities(localProbs); - } - - } - - /** - * Set DMU attributes. - * - * @param tour - * @param trip - */ - public void setDmuAttributes(SpecialEventTour tour, SpecialEventTrip trip) - { - - int tourDestinationMgra = tour.getDestinationMGRA(); - int tripOriginMgra = trip.getOriginMgra(); - int tripDestinationMgra = trip.getDestinationMgra(); - - int tripOriginTaz = mgraManager.getTaz(tripOriginMgra); - int tripDestinationTaz = mgraManager.getTaz(tripDestinationMgra); - - dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, - tour.getDebugChoiceModels()); - - dmu.setTourDepartPeriod(tour.getDepartTime()); - dmu.setTourArrivePeriod(tour.getArriveTime()); - dmu.setTripPeriod(trip.getPeriod()); - dmu.setIncome(tour.getIncome()); - dmu.setPartySize(tour.getPartySize()); - if (trip.isInbound()) dmu.setOutboundHalfTourDirection(0); - else dmu.setOutboundHalfTourDirection(1); - - // set trip mc dmu values for transit logsum (gets replaced below by uec values) - double c_ivt = -0.03; - double c_cost = - 0.0033; - - // Solve trip mode level utilities - mcDmuObject.setIvtCoeff(c_ivt); - mcDmuObject.setCostCoeff(c_cost); - double walkTransitLogsum = -999.0; - double driveTransitLogsum = -999.0; - - logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); - dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); - - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); - walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - dmu.setWalkTransitLogsum(walkTransitLogsum); - if (!trip.isInbound()) - { - logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); - } else - { - logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); - } - - dmu.setPnrTransitLogsum(driveTransitLogsum); - dmu.setKnrTransitLogsum(driveTransitLogsum); - - int eventNumber = tour.getEventNumber(); - - float parkingCost = eventData.getValueAt(eventNumber, "ParkingCost"); - float parkingTime = eventData.getValueAt(eventNumber, "ParkingTime"); - - dmu.setParkingCost(parkingCost); - dmu.setParkingTime(parkingTime); - - if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); - else dmu.setTripOrigIsTourDest(0); - - if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); - else dmu.setTripDestIsTourDest(0); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java deleted file mode 100644 index f3edf5a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/specialevent/SpecialEventTripTables.java +++ /dev/null @@ -1,595 +0,0 @@ -package org.sandag.abm.specialevent; -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class SpecialEventTripTables { - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TableDataSet tripData; - - // Some parameters - private int[] modeIndex; // an index array,dimensioned by number of total modes, returns 0=auto modes, 1=non-motor, 2=transit, 3=other - private int[] matrixIndex; // an index array, dimensioned by number of modes, returns the element of the matrix array to store value - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private HashMap rbMap; - - // matrices are indexed by modes - private Matrix[][] matrix; - - private ResourceBundle rb; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - - private MatrixDataServerRmi ms; - private float sampleRate; - private static int iteration=1; - public int numSkimSets; - - public SpecialEventTripTables(HashMap rbMap) - { - - this.rbMap = rbMap; - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTourModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - } else if (modelStructure.getTourModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - } else if (modelStructure.getTourModeIsWalkTransit(i) - || modelStructure.getTourModeIsDriveTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - } - } - - logger.info("autoModes="+autoModes+" nmotModes="+nmotModes+" tranModes="+tranModes+" othrModes="+othrModes); - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][]; - for (int i = 0; i < numberOfModes; ++i) - { - - String modeName; - - if (i == 0) - { - matrix[i] = new Matrix[autoModes]; - for (int j = 0; j < autoModes; ++j) - { - modeName = modelStructure.getModeName(j + 1); - matrix[i][j] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j].setExternalNumbers(tazIndex); - } - } else if (i == 1) - { - matrix[i] = new Matrix[nmotModes]; - for (int j = 0; j < nmotModes; ++j) - { - modeName = modelStructure.getModeName(j + 1 + autoModes); - matrix[i][j] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j].setExternalNumbers(tazIndex); - } - } else if (i == 2) - { - matrix[i] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l pMap; - String propertiesFile = null; - - logger.info(String.format( - "SANDAG Special Event Model Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - float sampleRate = 1.0f; - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - logger.info("Special Event Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - SpecialEventTripTables tripTables = new SpecialEventTripTables(pMap); - tripTables.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - tripTables.ms = matrixServer; - } else - { - tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - tripTables.ms.testRemote("SpecialEventTripTables"); - - // mdm = MatrixDataManager.getInstance(); - // mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - tripTables.createTripTables(mt); - - } - - /** - * @return the sampleRate - */ - public double getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(float sampleRate) - { - this.sampleRate = sampleRate; - } - -} - - diff --git a/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java b/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java deleted file mode 100644 index 912a87c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/survey/OutputTapPairs.java +++ /dev/null @@ -1,363 +0,0 @@ -package org.sandag.abm.survey; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.rmi.RemoteException; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.accessibilities.BestTransitPathCalculator; -import org.sandag.abm.accessibilities.DriveTransitWalkSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitDriveSkimsCalculator; -import org.sandag.abm.accessibilities.WalkTransitWalkSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; -import org.sandag.abm.modechoice.TransitDriveAccessDMU; -import org.sandag.abm.modechoice.TransitWalkAccessDMU; -import org.sandag.abm.modechoice.Modes; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -/** - * This class reads and processes on-board transit survey data. It writes out - * all TAP-pairs given the origin/destination MAZ, the time period, and the - * access/egress mode sequence for the observation. - * - * @author joel.freedman - * - */ -public class OutputTapPairs { - private static final Logger logger = Logger.getLogger(OutputTapPairs.class); - private BestTransitPathCalculator bestPathCalculator; - protected WalkTransitWalkSkimsCalculator wtw; - protected WalkTransitDriveSkimsCalculator wtd; - protected DriveTransitWalkSkimsCalculator dtw; - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - private MatrixDataServerRmi ms; - private String inputFile; - private String outputFile; - private TableDataSet inputDataTable; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private HashMap sequentialMaz; - - AutoTazSkimsCalculator tazDistanceCalculator; - - protected PrintWriter writer; - - - public OutputTapPairs(HashMap propertyMap, String inputFile, String outputFile){ - this.inputFile = inputFile; - this.outputFile = outputFile; - - startMatrixServer(propertyMap); - initialize(propertyMap); - } - - - /** - * Initialize best path builders. - * - * @param propertyMap A property map with relevant properties. - */ - public void initialize(HashMap propertyMap){ - - logger.info("Initializing OutputTapPairs"); - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - bestPathCalculator = new BestTransitPathCalculator(propertyMap); - - tazDistanceCalculator = new AutoTazSkimsCalculator(propertyMap); - tazDistanceCalculator.computeTazDistanceArrays(); - - wtw = new WalkTransitWalkSkimsCalculator(propertyMap); - wtw.setup(propertyMap, logger, bestPathCalculator); - wtd = new WalkTransitDriveSkimsCalculator(propertyMap); - wtd.setup(propertyMap, logger, bestPathCalculator); - dtw = new DriveTransitWalkSkimsCalculator(propertyMap); - dtw.setup(propertyMap, logger, bestPathCalculator); - - readData(); - createOutputFile(); - - } - - /** - * Read data into inputDataTable tabledataset. - * - */ - private void readData(){ - - logger.info("Begin reading the data in file " + inputFile); - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - inputDataTable = csvFile.readFile(new File(inputFile)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End reading the data in file " + inputFile); - } - - /** - * Create the output file and write a header record. - */ - private void createOutputFile(){ - - logger.info("Creating file " + outputFile); - - try - { - writer = new PrintWriter(new BufferedWriter(new FileWriter(outputFile))); - } catch (IOException e) - { - logger.fatal("Could not open file " + outputFile + " for writing\n"); - throw new RuntimeException(); - } - String headerString = new String( - "rownames,npath,access_mode_recode,period,set,boardTap,alightTap,bestUtility,accessTime,egressTime,auxWalkTime," - + "localBusIvt,expressBusIvt,brtIvt,lrtIvt,crIvt,firstWaitTime,trfWaitTime,fare,totalIVT,xfers\n"); - - - writer.print(headerString); - - } - - - /** - * Iterate through input data and process\write taps. - */ - private void run(){ - - TransitWalkAccessDMU walkDmu = new TransitWalkAccessDMU(); - TransitDriveAccessDMU driveDmu = new TransitDriveAccessDMU(); - double[][] bestTaps = null; - double[] skims = null; - double boardAccessTime; - double alightAccessTime; - //iterate through data and calculate - for(int row = 1; row<=inputDataTable.getRowCount();++row ){ - - if((row<=100) || ((row % 100) == 0)) - logger.info("Processing input record "+row); - - String label=inputDataTable.getStringValueAt(row, "id"); - int originMaz = (int) inputDataTable.getValueAt(row, "orig_maz"); - int destinationMaz = (int) inputDataTable.getValueAt(row, "dest_maz"); - int period = (int) inputDataTable.getValueAt(row, "period")-1; //Input is 1=EA, 2=AM, 3=MD, 4=PM, 5=EV - int accessMode = (int) inputDataTable.getValueAt(row, "accessEgress"); // 1 walk, 2 PNR, 3 KNR\bike - int inbound = (int) inputDataTable.getValueAt(row, "inbound"); // 1 if inbound, else 0 - - int accessEgressMode = -1; - - if(accessMode ==1) - accessEgressMode=bestPathCalculator.WTW; - else if ((accessMode == 2||accessMode==3) && inbound==0) - accessEgressMode = bestPathCalculator.DTW; - else if ((accessMode == 2||accessMode==3) && inbound==1) - accessEgressMode = bestPathCalculator.WTD; - - if(originMaz==0||destinationMaz==0||accessEgressMode==-1) - continue; - - - int originTaz = mgraManager.getTaz(originMaz); - int destinationTaz = mgraManager.getTaz(destinationMaz); - - float odDistance = (float) tazDistanceCalculator.getTazToTazDistance(ModelStructure.AM_SKIM_PERIOD_INDEX, originTaz, destinationTaz); - bestTaps = bestPathCalculator.getBestTapPairs(walkDmu, driveDmu, accessEgressMode, originMaz, destinationMaz, period, false, logger, odDistance); - double[] bestUtilities = bestPathCalculator.getBestUtilities(); - - //iterate through n-best paths - for (int i = 0; i < bestTaps.length; i++) - { - if(bestUtilities[i]<-500) - continue; - - writer.print(label); - - //write transit TAP pairs and utility - int boardTap = (int) bestTaps[i][0]; - int alightTap = (int) bestTaps[i][1]; - int set = (int) bestTaps[i][2]; - - writer.format(",%d,%d,%d,%d,%d,%d,%9.4f",i,accessMode,period,set,boardTap,alightTap,bestUtilities[i]); - - // System.out.println(label+String.format(",%d,%d,%d,%d,%d,%d,%9.4f",i,accessEgressMode,period,set,boardTap,alightTap,bestUtilities[i])); - //write skims - if(accessEgressMode==bestPathCalculator.WTW){ - boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); - alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); - skims = wtw.getWalkTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); - }else if (accessEgressMode==bestPathCalculator.DTW){ - boardAccessTime = tazManager.getTimeToTapFromTaz(originTaz,boardTap,( accessMode==2? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - alightAccessTime = mgraManager.getWalkTimeFromMgraToTap(destinationMaz,alightTap); - skims = dtw.getDriveTransitWalkSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); - }else if(accessEgressMode==bestPathCalculator.WTD){ - boardAccessTime = mgraManager.getWalkTimeFromMgraToTap(originMaz,boardTap); - alightAccessTime = tazManager.getTimeToTapFromTaz(destinationTaz,alightTap,( accessMode==2? Modes.AccessMode.PARK_N_RIDE : Modes.AccessMode.KISS_N_RIDE)); - skims = wtd.getWalkTransitDriveSkims(set, boardAccessTime, alightAccessTime, boardTap, alightTap, period, false); - } - - for(int j=0; j < skims.length; ++j) - writer.format(",%9.2f",skims[j]); - - writer.format("\n"); - - } - writer.flush(); - } - - - } - - /** - * Startup a connection to the matrix manager. - * - * @param serverAddress - * @param serverPort - * @param mt - * @return - */ - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - try - { - // create the concrete data server object - matrixServer.start32BitMatrixIoServer(mt); - } catch (RuntimeException e) - { - matrixServer.stop32BitMatrixIoServer(); - logger.error( - "RuntimeException caught making remote method call to start 32 bit mitrix in remote MatrixDataServer.", - e); - } - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - matrixServer.stop32BitMatrixIoServer(); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - matrixServer.stop32BitMatrixIoServer(); - throw new RuntimeException(); - } - - return matrixServer; - - } - - - /** - * Main run method - * @param args - */ - public static void main(String[] args) { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("Best Tap Pairs Program using CT-RAMP version ", - CtrampApplication.VERSION)); - - logger.info(String.format("Outputting TAP pairs and utilities for on-board survey data")); - - - String inputFile = null; - String outputFile = null; - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else { - propertiesFile = args[0]; - - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-inputFile")) - { - inputFile = args[i + 1]; - } - if (args[i].equalsIgnoreCase("-outputFile")) - { - outputFile = args[i + 1]; - } - } - } - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - OutputTapPairs outputTapPairs = new OutputTapPairs(pMap, inputFile, outputFile); - - - outputTapPairs.run(); - - - - - } - private void startMatrixServer(HashMap properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - MatrixDataServerIf ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - logger.error("could not connect to matrix server", e); - throw new RuntimeException(e); - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java deleted file mode 100644 index ae898e5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/utilities/CreateLogsums.java +++ /dev/null @@ -1,410 +0,0 @@ -package org.sandag.abm.utilities; - -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.jppf.client.JPPFClient; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.HouseholdChoiceModelRunner; -import org.sandag.abm.ctramp.HouseholdDataManager; -import org.sandag.abm.ctramp.HouseholdDataManagerIf; -import org.sandag.abm.ctramp.HouseholdDataManagerRmi; -import org.sandag.abm.ctramp.HouseholdDataWriter; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.UsualWorkSchoolLocationChoiceModel; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; - -public class CreateLogsums { - - - private BuildAccessibilities aggAcc; - private JPPFClient jppfClient; - private static Logger logger = Logger.getLogger(CreateLogsums.class); - private HouseholdDataManagerIf householdDataManager; - private HashMap propertyMap; - private ResourceBundle resourceBundle; - // are used if no command line arguments are specified. - private int globalIterationNumber = 0; - private float iterationSampleRate = 0f; - private int sampleSeed = 0; - private SandagModelStructure modelStructure; - private SandagCtrampDmuFactory dmuFactory; - private MatrixDataServerIf ms; - private ModelOutputReader modelOutputReader; - - - /** - * Constructor. - * - * @param propertiesFile - * @param globalIterationNumber - * @param globalSampleRate - * @param sampleSeed - */ - public CreateLogsums(String propertiesFile, int globalIterationNumber, float globalSampleRate, int sampleSeed){ - - this.resourceBundle = ResourceBundle.getBundle(propertiesFile); - propertyMap = ResourceUtil.getResourceBundleAsHashMap ( propertiesFile); - this.globalIterationNumber = globalIterationNumber; - this.iterationSampleRate = globalSampleRate; - this.sampleSeed = sampleSeed; - - } - - /** - * Initialize data members - */ - public void initialize(){ - - startMatrixServer(propertyMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - householdDataManager = getHouseholdDataManager(); - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after getting household manager"); - - // create a factory object to pass to various model components from which - // they can create DMU objects - dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - modelOutputReader = new ModelOutputReader(propertyMap,modelStructure, globalIterationNumber); - } - - - /** - * Run all components. - * - */ - public void run(){ - - initialize(); - readModelOutputsAndCreateTours(); - createWorkLogsums(); - createNonWorkLogsums(); - - HouseholdDataWriter dataWriter = new HouseholdDataWriter( propertyMap, modelStructure, globalIterationNumber ); - dataWriter.writeDataToFiles(householdDataManager); - - } - - /** - * Read the model outputs and create tours. - */ - public void readModelOutputsAndCreateTours(){ - - modelOutputReader.readHouseholdDataOutput(); - modelOutputReader.readPersonDataOutput(); - modelOutputReader.readTourDataOutput(); - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before reading model output"); - - Household[] households = householdDataManager.getHhArray(); - for(Household household : households){ - - modelOutputReader.setHouseholdAndPersonAttributes(household); - - if(modelOutputReader.hasJointTourFile()) - modelOutputReader.createJointTours(household); - - if(modelOutputReader.hasIndividualTourFile()) - modelOutputReader.createIndividualTours(household); - } - householdDataManager.setHhArray(households); - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after reading model output"); - - } - - - - /** - * Calculate and write work destination choice logsums for the synthetic population. - * - * @param propertyMap - */ - public void createWorkLogsums(){ - - jppfClient = new JPPFClient(); - - if (aggAcc == null) - { - logger.info("creating Accessibilities Object for UWSL."); - aggAcc = BuildAccessibilities.getInstance(); - aggAcc.setupBuildAccessibilities(propertyMap,false); -// aggAcc.setJPPFClient(jppfClient); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - - boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, "acc.read.input.file"); - if (readAccessibilities) - { - String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap,"Project.Directory"); - String accFileName = Paths.get(projectDirectory,Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file")).toString(); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - } - - // new the usual school and location choice model object - UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( - resourceBundle, "none", jppfClient, modelStructure, ms, dmuFactory, aggAcc); - - // calculate and get the array of worker size terms table - MGRAs by - // occupations - aggAcc.createWorkSegmentNameIndices(); - aggAcc.calculateWorkerSizeTerms(); - double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); - - // run the model - logger.info("Starting usual work location choice for logsum calculations."); - usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, workerSizeTerms); - logger.info("Finished with usual work location choice for logsum calculations."); - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after running school and work location choice"); - - } - - public void createNonWorkLogsums(){ - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before running non-work logsums"); - - HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( propertyMap, jppfClient, "False", householdDataManager, ms, modelStructure, dmuFactory ); - runner.runHouseholdChoiceModels(); - - } - - - - /** - * Create the household data manager. Based on the code in MTCTM2TourBasedModel.runTourBasedModel() - * @return The household data manager interface. - */ - public HouseholdDataManagerIf getHouseholdDataManager( ){ - - - boolean localHandlers = false; - - String testString; - - HouseholdDataManagerIf householdDataManager; - String hhHandlerAddress = ""; - int hhServerPort = 0; - try - { - // get household server address. if none is specified a local server in - // the current process will be started. - hhHandlerAddress = resourceBundle.getString("RunModel.HouseholdServerAddress"); - try - { - // get household server port. - hhServerPort = Integer.parseInt(resourceBundle.getString("RunModel.HouseholdServerPort")); - localHandlers = false; - } catch (MissingResourceException e) - { - // if no household data server address entry is found, the object - // will be created in the local process - localHandlers = true; - } - } catch (MissingResourceException e) - { - localHandlers = true; - } - - - try - { - - if (localHandlers) - { - - // create a new local instance of the household array manager - householdDataManager = new SandagHouseholdDataManager(); - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - String inputHouseholdFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); - - } else - { - - householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, - SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - testString = householdDataManager.testRemote(); - logger.info("HouseholdDataManager test: " + testString); - - householdDataManager.setPropertyFileValues(propertyMap); - } - - //always starting from scratch (RunModel.RestartWithHhServer=none) - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = resourceBundle - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = resourceBundle - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); - - }catch (Exception e) - { - - logger.error(String - .format("Exception caught setting up household data manager."), e); - throw new RuntimeException(); - - } - - return householdDataManager; - } - - - /** - * Start a new matrix server connection. - * - * @param properties - */ - private void startMatrixServer(HashMap properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - logger.error("could not connect to matrix server", e); - throw new RuntimeException(e); - - } - - } - - - public JPPFClient getJppfClient() { - return jppfClient; - } - - public void setJppfClient(JPPFClient jppfClient) { - this.jppfClient = jppfClient; - } - - public static void main(String[] args) { - - long startTime = System.currentTimeMillis(); - int globalIterationNumber = -1; - float iterationSampleRate = -1.0f; - int sampleSeed = -1; - - ResourceBundle rb = null; - - logger.info( String.format( "Generating Logsums from MTC Tour Based Model using CT-RAMP version %s, 22feb2011 build %s", CtrampApplication.VERSION, 2 ) ); - - if ( args.length == 0 ) { - logger.error( String.format( "no properties file base name (without .properties extension) was specified as an argument." ) ); - return; - } - else { - rb = ResourceBundle.getBundle( args[0] ); - - // optional arguments - for (int i=1; i < args.length; i++) { - - if (args[i].equalsIgnoreCase("-iteration")) { - globalIterationNumber = Integer.parseInt( args[i+1] ); - logger.info( String.format( "-iteration %d.", globalIterationNumber ) ); - } - - if (args[i].equalsIgnoreCase("-sampleRate")) { - iterationSampleRate = Float.parseFloat( args[i+1] ); - logger.info( String.format( "-sampleRate %.4f.", iterationSampleRate ) ); - } - - if (args[i].equalsIgnoreCase("-sampleSeed")) { - sampleSeed = Integer.parseInt( args[i+1] ); - logger.info( String.format( "-sampleSeed %d.", sampleSeed ) ); - } - - } - - if ( globalIterationNumber < 0 ) { - globalIterationNumber = 1; - logger.info( String.format( "no -iteration flag, default value %d used.", globalIterationNumber ) ); - } - - if ( iterationSampleRate < 0 ) { - iterationSampleRate = 1; - logger.info( String.format( "no -sampleRate flag, default value %.4f used.", iterationSampleRate ) ); - } - - if ( sampleSeed < 0 ) { - sampleSeed = 0; - logger.info( String.format( "no -sampleSeed flag, default value %d used.", sampleSeed ) ); - } - - } - - - String baseName; - if ( args[0].endsWith(".properties") ) { - int index = args[0].indexOf(".properties"); - baseName = args[0].substring(0, index); - } - else { - baseName = args[0]; - } - - - // create an instance of this class for main() to use. - CreateLogsums mainObject = new CreateLogsums( args[0], globalIterationNumber, iterationSampleRate, sampleSeed ); - - // Create logsums - try { - - logger.info ("Creating logsums."); - mainObject.run(); - - } - catch ( RuntimeException e ) { - logger.error ( "RuntimeException caught in com.pb.mtctm2.abm.reports.CreateLogsums.main() -- exiting.", e ); - System.exit(2); - } - - - logger.info (""); - logger.info (""); - logger.info ("CreateLogsums finished in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); - - System.exit(0); - - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java deleted file mode 100644 index 92ee34d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/utilities/ErrorLogging.java +++ /dev/null @@ -1,41 +0,0 @@ -package org.sandag.abm.utilities; - -import java.util.HashMap; -/** - * ErrorLogging definitions - * - * @author wsu - * @mail Wu.Sun@sandag.org - * @version 13.3.0 - * @since 2016-06-20 - * - */ -public class ErrorLogging { - public static final String AtTapNotInTransitNetwork = "A TAP in AT network is not in TRANSIT network!"; - public static final String TransitTapNotInAt = "A TAP in TRANSIT network is not in AT network!"; - public static final String InconsistentTapPostions = "Positions of a TAP are different in AT and TRANSIT networks!"; - public static final String NoAtTaps = "No valid TAPs in AT network!"; - public static final String NoTransitTaps = "No valid TAPs in TRANSIT network!"; - protected HashMap atErrorIndexMap; - - public ErrorLogging() - { - - atErrorIndexMap = new HashMap(); - createAtErrorIndexMap(); - } - private void createAtErrorIndexMap() - { - atErrorIndexMap.put("AT1", AtTapNotInTransitNetwork); - atErrorIndexMap.put("AT2", TransitTapNotInAt ); - atErrorIndexMap.put("AT3", InconsistentTapPostions); - atErrorIndexMap.put("AT4", NoAtTaps); - atErrorIndexMap.put("AT5", NoTransitTaps); - } - - public String getAtError(String index) - { - return atErrorIndexMap.get(index); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java deleted file mode 100644 index 23ef9fa..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/utilities/ModelOutputReader.java +++ /dev/null @@ -1,836 +0,0 @@ -package org.sandag.abm.utilities; - -import java.io.File; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.ModelStructure; -import org.sandag.abm.ctramp.Person; -import org.sandag.abm.ctramp.Tour; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -public class ModelOutputReader { - - private transient Logger logger = Logger.getLogger(ModelOutputReader.class); - - private static final String PROPERTIES_HOUSEHOLD_DATA_FILE = "Accessibilities.HouseholdDataFile"; - private static final String PROPERTIES_PERSON_DATA_FILE = "Accessibilities.PersonDataFile"; - private static final String PROPERTIES_INDIV_TOUR_DATA_FILE = "Accessibilities.IndivTourDataFile"; - private static final String PROPERTIES_JOINT_TOUR_DATA_FILE = "Accessibilities.JointTourDataFile"; - private static final String PROPERTIES_INDIV_TRIP_DATA_FILE = "Accessibilities.IndivTripDataFile"; - private static final String PROPERTIES_JOINT_TRIP_DATA_FILE = "Accessibilities.JointTripDataFile"; - private ModelStructure modelStructure; - private int iteration; - private HashMap rbMap; - private HashMap householdFileAttributesMap; - private HashMap personFileAttributesMap; - private HashMap> individualTourAttributesMap; //by person_id - private HashMap> jointTourAttributesMap; //by hh_id - - private boolean readIndividualTourFile = false; - private boolean readJointTourFile = false; - - /** - * Default constructor. - * @param rbMap Hashmap of properties - * @param modelStructure Model structure object - * @param iteration Iteration number used for file names - */ - public ModelOutputReader(HashMap rbMap, ModelStructure modelStructure, - int iteration) - { - logger.info("Writing data structures to files."); - this.modelStructure = modelStructure; - this.iteration = iteration; - this.rbMap = rbMap; - } - - - /** - * Read household data and store records in householdFileAttributesMap - */ - public void readHouseholdDataOutput(){ - - String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String hhFile = rbMap.get(PROPERTIES_HOUSEHOLD_DATA_FILE); - - TableDataSet householdData = readTableData(baseDir+hhFile); - - householdFileAttributesMap = new HashMap(); - - //hh_id,home_mgra,income,HVs,AVs,transponder,cdap_pattern,out_escort_choice,inb_escort_choice,jtf_choice - for(int row = 1; row<=householdData.getRowCount();++row){ - - long hhid = (long) householdData.getValueAt(row,"hh_id"); - int home_mgra = (int)householdData.getValueAt(row,"home_mgra"); - int income = (int) householdData.getValueAt(row,"income"); - int automated_vehicles = (int) householdData.getValueAt(row,"AVs"); - int human_vehicles = (int) householdData.getValueAt(row,"HVs"); - int autos = automated_vehicles + human_vehicles; - int transponder = (int) householdData.getValueAt(row,"transponder"); - String cdap_pattern = householdData.getStringValueAt(row,"cdap_pattern"); - int jtf_choice = (int) householdData.getValueAt(row,"jtf_choice"); - int out_escort_choice = (int) householdData.getValueAt(row,"out_escort_choice"); - int inb_escort_choice = (int) householdData.getValueAt(row,"inb_escort_choice"); - - - // float sampleRate = householdData.getValueAt(row,"sampleRate"); - HouseholdFileAttributes hhAttributes = new HouseholdFileAttributes(hhid, - home_mgra, income, autos, automated_vehicles, human_vehicles,transponder,cdap_pattern, - jtf_choice,out_escort_choice,inb_escort_choice); - - householdFileAttributesMap.put(hhid, hhAttributes); - - } - } - - - /** - * Read the data from the Results.PersonDataFile. - * Data is stored in HashMap personFileAttributesMap - * so that it can be retrieved quickly for a household object. - * - */ - public void readPersonDataOutput(){ - - //read person data - String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String personFile = baseDir + rbMap.get(PROPERTIES_PERSON_DATA_FILE); - TableDataSet personData = readTableData(personFile); - - personFileAttributesMap = new HashMap(); - //hh_id,person_id,person_num,age,gender,type,value_of_time,activity_pattern,imf_choice,inmf_choice, - // fp_choice,reimb_pct,tele_choice,ie_choice,timeFactorWork,timeFactorNonWork - - for(int row = 1; row<=personData.getRowCount();++row){ - - //get the values for this person - long hhid = (long) personData.getValueAt(row, "hh_id"); - long person_id = (long) personData.getValueAt(row,"person_id"); - long personNumber = (long) personData.getValueAt(row,"person_num"); - int age = (int) personData.getValueAt(row,"age"); - - String genderString = personData.getStringValueAt(row,"gender"); - int gender = (genderString.compareTo("m")==0 ? 1 : 2); - - float valueOfTime = personData.getValueAt(row,"value_of_time"); - String activityPattern = personData.getStringValueAt(row,"activity_pattern"); - String type = personData.getStringValueAt(row,"type"); - int personType = getPersonType(type); - - // int occup = (int) personData.getValueAt(row,"occp"); - - - int imfChoice = (int) personData.getValueAt(row, "imf_choice"); - int inmfChoice = (int) personData.getValueAt(row, "inmf_choice"); - int fp_choice = (int) personData.getValueAt(row,"fp_choice"); - float reimb_pct = personData.getValueAt(row,"reimb_pct"); - int tele_choice = (int) personData.getValueAt(row,"tele_choice"); - int ie_choice = (int) personData.getValueAt(row,"ie_choice"); - float timeFactorWork = personData.getValueAt(row,"timeFactorWork"); - float timeFactorNonWork = personData.getValueAt(row,"timeFactorNonWork"); - - //float sampleRate = personData.getValueAt(row,"sampleRate"); - - PersonFileAttributes personFileAttributes = new PersonFileAttributes(hhid,person_id,personNumber,age,gender,valueOfTime, - activityPattern,personType, imfChoice,inmfChoice,fp_choice,reimb_pct,tele_choice, - ie_choice, timeFactorWork, timeFactorNonWork); - - personFileAttributesMap.put(person_id,personFileAttributes); - - } - - } - - /** - * Read both tour files. - * - */ - public void readTourDataOutput(){ - - String baseDir = rbMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - - if(rbMap.containsKey(PROPERTIES_INDIV_TOUR_DATA_FILE)){ - String indivTourFile = rbMap.get(PROPERTIES_INDIV_TOUR_DATA_FILE); - if(indivTourFile != null){ - if(indivTourFile.length()>0){ - individualTourAttributesMap = readTourData(baseDir+indivTourFile, false, individualTourAttributesMap); - readIndividualTourFile = true; - } - } - } - if(rbMap.containsKey(PROPERTIES_JOINT_TOUR_DATA_FILE)){ - String jointTourFile = rbMap.get(PROPERTIES_JOINT_TOUR_DATA_FILE); - if(jointTourFile != null){ - if(jointTourFile.length()>0){ - jointTourAttributesMap = readTourData(baseDir+jointTourFile, true, jointTourAttributesMap); - readJointTourFile = true; - } - } - } - if(readIndividualTourFile==false){ - logger.info("No individual tour file to read in MtcModelOutputReader class"); - } - if(readJointTourFile==false){ - logger.info("No joint tour file to read in MtcModelOutputReader class"); - } - } - - - /** - * Read the data from the Results.IndivTourDataFile or Results.JointTourDataFile. - * Data is stored in HashMap passed into method as an argument. Method handles - * both individual and joint data. Joint tour data is indexed by hh_id - * so that it can be retrieved quickly for a household object. Individual tour data is - * indexed by person_id. - * - */ - public HashMap> readTourData(String filename, boolean isJoint, HashMap> tourFileAttributesMap ){ - - TableDataSet tourData = readTableData(filename); - - tourFileAttributesMap = new HashMap>(); - //hh_id,person_id,person_num,person_type,tour_id,tour_category,tour_purpose, - //orig_mgra,dest_mgra,start_period,end_period,tour_mode,av_avail,tour_distance,atwork_freq, - //num_ob_stops,num_ib_stops,valueOfTime,escort_type_out,escort_type_in,driver_num_out,driver_num_in - - for(int row = 1; row<=tourData.getRowCount();++row){ - - long hh_id = (long) tourData.getValueAt(row,"hh_id"); - long person_id = 0; - int person_num=0; - int person_type=0; - int escort_type_out=0; - int escort_type_in=0; - int driver_num_out=0; - int driver_num_in=0; - if(!isJoint){ - person_id = (long) tourData.getValueAt(row,"person_id");; - person_num = (int) tourData.getValueAt(row,"person_num"); - person_type = (int) tourData.getValueAt(row,"person_type"); - escort_type_out = (int) tourData.getValueAt(row,"escort_type_out"); - escort_type_in = (int) tourData.getValueAt(row,"escort_type_in"); - driver_num_out = (int) tourData.getValueAt(row,"driver_num_out"); - driver_num_in = (int) tourData.getValueAt(row,"driver_num_in"); - } - int tour_id = (int) tourData.getValueAt(row,"tour_id"); - String tour_category = tourData.getStringValueAt(row,"tour_category"); - String tour_purpose = tourData.getStringValueAt(row,"tour_purpose"); - - int tour_composition = 0; - String tour_participants = null; - if(isJoint){ - tour_composition = (int) tourData.getValueAt(row,"tour_composition"); - tour_participants = tourData.getStringValueAt(row,"tour_participants"); - } - - int orig_mgra = (int) tourData.getValueAt(row,"orig_mgra"); - int dest_mgra = (int) tourData.getValueAt(row,"dest_mgra"); - int start_period = (int) tourData.getValueAt(row,"start_period"); - int end_period = (int) tourData.getValueAt(row,"end_period"); - int tour_mode = (int) tourData.getValueAt(row,"tour_mode"); - int av_avail = (int) tourData.getValueAt(row,"av_avail"); - float tour_distance = tourData.getValueAt(row,"tour_distance"); - // float tour_time = tourData.getValueAt(row,"tour_time"); - int atWork_freq = (int) tourData.getValueAt(row,"atWork_freq"); - int num_ob_stops = (int) tourData.getValueAt(row,"num_ob_stops"); - int num_ib_stops = (int) tourData.getValueAt(row,"num_ib_stops"); - float valueOfTime = tourData.getValueAt(row, "valueOfTime"); - /* - int out_btap = (int) tourData.getValueAt(row,"out_btap"); - int out_atap = (int) tourData.getValueAt(row,"out_atap"); - int in_btap = (int) tourData.getValueAt(row,"in_btap"); - int in_atap = (int) tourData.getValueAt(row,"in_atap"); - int out_set = (int) tourData.getValueAt(row,"out_set"); - int in_set = (int) tourData.getValueAt(row,"in_set"); -// float sampleRate = tourData.getValueAt(row,"sampleRate"); -// int avAvailable = (int) tourData.getValueAt(row,"avAvailable"); - */ float[] util = new float[modelStructure.getMaxTourModeIndex()]; - float[] prob = new float[modelStructure.getMaxTourModeIndex()]; - - TourFileAttributes tourFileAttributes = new TourFileAttributes(hh_id, person_id, person_num, person_type, - tour_id, tour_category, tour_purpose, orig_mgra,dest_mgra, - start_period, end_period, tour_mode, av_avail, tour_distance, - atWork_freq, num_ob_stops, num_ib_stops, valueOfTime, - escort_type_out,escort_type_in,driver_num_out,driver_num_in, - tour_composition, tour_participants,util,prob); - - //if individual tour, map key is person_id, else it is hh_id - long key = -1; - if(!isJoint) - key = person_id; - else - key = hh_id; - - //if the not the first tour for this person or hh, add the tour to the existing - //arraylist; else create a new arraylist and add the tour attributes to it, - //then add the arraylist to the map - if(tourFileAttributesMap.containsKey(key)){ - ArrayList tourArray = tourFileAttributesMap.get(key); - tourArray.add(tourFileAttributes); - }else{ - ArrayList tourArray = new ArrayList(); - tourArray.add(tourFileAttributes); - tourFileAttributesMap.put(key, tourArray); - } - - } - - return tourFileAttributesMap; - } - - /** - * Create individual tour objects for all persons in the household object based - * on the data read in the individual tour file. - * - * @param household - */ - public void createIndividualTours(Household household){ - - HashMap purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); - Person[] persons = household.getPersons(); - for(int pnum=1;pnum tourAttributesArray = individualTourAttributesMap.get(personId); - - //store tours by type - ArrayList workTours = new ArrayList(); - ArrayList universityTours = new ArrayList(); - ArrayList schoolTours = new ArrayList(); - ArrayList atWorkSubtours = new ArrayList(); - ArrayList nonMandTours = new ArrayList(); - - for(int i=0;i0){ - p.createWorkTours(workTours.size(), 0, ModelStructure.WORK_PRIMARY_PURPOSE_NAME, - ModelStructure.WORK_PRIMARY_PURPOSE_INDEX); - ArrayList workTourArrayList = p.getListOfWorkTours(); - for(int i=0;i0){ - p.createSchoolTours(schoolTours.size(), 0, ModelStructure.SCHOOL_PRIMARY_PURPOSE_NAME, - ModelStructure.SCHOOL_PRIMARY_PURPOSE_INDEX); - ArrayList schoolTourArrayList = p.getListOfSchoolTours(); - for(int i=0;i0){ - p.createSchoolTours(universityTours.size(), 0, ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_NAME, - ModelStructure.UNIVERSITY_PRIMARY_PURPOSE_INDEX); - ArrayList universityTourArrayList = p.getListOfSchoolTours(); - for(int i=0;i nonMandTourArrayList = p.getListOfIndividualNonMandatoryTours(); - for(int i =0; i purposeIndexMap = modelStructure.getPrimaryPurposeNameIndexMap(); - - //joint tours - long hhid = household.getHhId(); - if(jointTourAttributesMap.containsKey(hhid)){ - ArrayList tourArray = jointTourAttributesMap.get(hhid); - int numberOfJointTours = tourArray.size(); - - //get the first joint tour - TourFileAttributes tourAttributes = tourArray.get(0); - String purposeString = tourAttributes.tour_purpose; - int purpose = purposeIndexMap.get(purposeString); - int composition = tourAttributes.tour_composition; - int[] tourParticipants = getTourParticipantsArray(tourAttributes.tour_participants); - - Tour t1 = new Tour(household,purposeString, ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purpose); - t1.setJointTourComposition(composition); - t1.setPersonNumArray(tourParticipants); - - //if the household has two joint tours, get the second - if(numberOfJointTours==2){ - tourAttributes = tourArray.get(2); - purposeString = tourAttributes.tour_purpose; - purpose = purposeIndexMap.get(purposeString); - composition = tourAttributes.tour_composition; - tourParticipants = getTourParticipantsArray(tourAttributes.tour_participants); - - Tour t2 = new Tour(household,purposeString, ModelStructure.JOINT_NON_MANDATORY_CATEGORY, purpose); - t2.setJointTourComposition(composition); - t2.setPersonNumArray(tourParticipants); - - //set in hh object - household.createJointTourArray(t1, t2); - tourAttributes.setModeledTourAttributes(t1); - tourAttributes.setModeledTourAttributes(t2); - }else{ - household.createJointTourArray(t1); - tourAttributes.setModeledTourAttributes(t1); - } - } - - - } - -// HELPER METHODS AND CLASSES - - /** - * Split the participants string around spaces and return the - * integer array of participant numbers. - * - * @param tourParticipants - * @return - */ - public int[] getTourParticipantsArray(String tourParticipants){ - - String[] values = tourParticipants.split(" "); - int[] array = new int[values.length]; - for (int i = 0; i < array.length; i++) - array[i] = Integer.parseInt(values[i]); - return array; - } - - /** - * Set household and person attributes for this household object. This method uses - * the data in the personFileAttributesMap to set the data members of the - * Person objects for all persons in the household. - * - * @param hhObject - */ - public void setHouseholdAndPersonAttributes(Household hhObject){ - - long hhid = (long) hhObject.getHhId(); - HouseholdFileAttributes hhAttributes = householdFileAttributesMap.get(hhid); - hhAttributes.setHouseholdAttributes(hhObject); - Person[] persons = hhObject.getPersons(); - for(int i=1;i 0) - { - String base = originalFileName.substring(0, lastDot); - String ext = originalFileName.substring(lastDot); - returnString = String.format("%s_%d%s", base, iteration, ext); - } else - { - returnString = String.format("%s_%d.csv", originalFileName, iteration); - } - - logger.info("writing " + originalFileName + " file to " + returnString); - - return returnString; - } - - public HashMap getHouseholdFileAttributesMap() { - return householdFileAttributesMap; - } - - public HashMap getPersonFileAttributesMap() { - return personFileAttributesMap; - } - - public HashMap> getIndividualTourAttributesMap() { - return individualTourAttributesMap; - } - - public HashMap> getJointTourAttributesMap() { - return jointTourAttributesMap; - } - - public boolean hasIndividualTourFile() { - return readIndividualTourFile; - } - - public boolean hasJointTourFile() { - return readJointTourFile; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java deleted file mode 100644 index 883786d..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/utilities/RunModeChoice.java +++ /dev/null @@ -1,402 +0,0 @@ -package org.sandag.abm.utilities; - -import java.nio.file.Paths; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.jppf.client.JPPFClient; -import org.sandag.abm.accessibilities.BuildAccessibilities; -import org.sandag.abm.application.SandagCtrampDmuFactory; -import org.sandag.abm.application.SandagHouseholdDataManager; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Household; -import org.sandag.abm.ctramp.HouseholdChoiceModelRunner; -import org.sandag.abm.ctramp.HouseholdDataManager; -import org.sandag.abm.ctramp.HouseholdDataManagerIf; -import org.sandag.abm.ctramp.HouseholdDataManagerRmi; -import org.sandag.abm.ctramp.HouseholdDataWriter; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.UsualWorkSchoolLocationChoiceModel; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.calculator.MatrixDataServerIf; -import com.pb.common.util.ResourceUtil; - -public class RunModeChoice { - - - private BuildAccessibilities aggAcc; - private JPPFClient jppfClient; - private static Logger logger = Logger.getLogger(RunModeChoice.class); - private HouseholdDataManagerIf householdDataManager; - private HashMap propertyMap; - private ResourceBundle resourceBundle; - // are used if no command line arguments are specified. - private int globalIterationNumber = 0; - private float iterationSampleRate = 0f; - private int sampleSeed = 0; - private SandagModelStructure modelStructure; - private SandagCtrampDmuFactory dmuFactory; - private MatrixDataServerIf ms; - private ModelOutputReader modelOutputReader; - - - /** - * Constructor. - * - * @param propertiesFile - * @param globalIterationNumber - * @param globalSampleRate - * @param sampleSeed - */ - public RunModeChoice(String propertiesFile, int globalIterationNumber, float globalSampleRate, int sampleSeed){ - - this.resourceBundle = ResourceBundle.getBundle(propertiesFile); - propertyMap = ResourceUtil.getResourceBundleAsHashMap ( propertiesFile); - this.globalIterationNumber = globalIterationNumber; - this.iterationSampleRate = globalSampleRate; - this.sampleSeed = sampleSeed; - - } - - /** - * Initialize data members - */ - public void initialize(){ - - startMatrixServer(propertyMap); - - // create modelStructure object - modelStructure = new SandagModelStructure(); - - householdDataManager = getHouseholdDataManager(); - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after getting household manager"); - - // create a factory object to pass to various model components from which - // they can create DMU objects - dmuFactory = new SandagCtrampDmuFactory(modelStructure,propertyMap); - - modelOutputReader = new ModelOutputReader(propertyMap,modelStructure, globalIterationNumber); - - jppfClient = new JPPFClient(); - - } - - - /** - * Run all components. - * - */ - public void run(){ - - initialize(); - readModelOutputsAndCreateTours(); - runHouseholdModels(); - HouseholdDataWriter dataWriter = new HouseholdDataWriter( propertyMap, modelStructure, globalIterationNumber ); - dataWriter.writeDataToFiles(householdDataManager); - - } - - /** - * Read the model outputs and create tours. - */ - public void readModelOutputsAndCreateTours(){ - - modelOutputReader.readHouseholdDataOutput(); - modelOutputReader.readPersonDataOutput(); - modelOutputReader.readTourDataOutput(); - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before reading model output"); - - Household[] households = householdDataManager.getHhArray(); - for(Household household : households){ - - modelOutputReader.setHouseholdAndPersonAttributes(household); - - if(modelOutputReader.hasJointTourFile()) - modelOutputReader.createJointTours(household); - - if(modelOutputReader.hasIndividualTourFile()) - modelOutputReader.createIndividualTours(household); - } - householdDataManager.setHhArray(households); - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after reading model output"); - - } - - - - /** - * Calculate and write work destination choice logsums for the synthetic population. - * - * @param propertyMap - */ - public void createWorkLogsums(){ - - - if (aggAcc == null) - { - logger.info("creating Accessibilities Object for UWSL."); - aggAcc = BuildAccessibilities.getInstance(); - aggAcc.setupBuildAccessibilities(propertyMap,false); -// aggAcc.setJPPFClient(jppfClient); - - aggAcc.calculateSizeTerms(); - aggAcc.calculateConstants(); - - boolean readAccessibilities = ResourceUtil.getBooleanProperty(resourceBundle, "acc.read.input.file"); - if (readAccessibilities) - { - String projectDirectory = Util.getStringValueFromPropertyMap(propertyMap,"Project.Directory"); - String accFileName = Paths.get(projectDirectory,Util.getStringValueFromPropertyMap(propertyMap, "acc.output.file")).toString(); - - aggAcc.readAccessibilityTableFromFile(accFileName); - - } else - { - - aggAcc.calculateDCUtilitiesDistributed(propertyMap); - - } - } - - // new the usual school and location choice model object - UsualWorkSchoolLocationChoiceModel usualWorkSchoolLocationChoiceModel = new UsualWorkSchoolLocationChoiceModel( - resourceBundle, "none", jppfClient, modelStructure, ms, dmuFactory, aggAcc); - - // calculate and get the array of worker size terms table - MGRAs by - // occupations - aggAcc.createWorkSegmentNameIndices(); - aggAcc.calculateWorkerSizeTerms(); - double[][] workerSizeTerms = aggAcc.getWorkerSizeTerms(); - - // run the model - logger.info("Starting usual work location choice for logsum calculations."); - usualWorkSchoolLocationChoiceModel.runWorkLocationChoiceModel(householdDataManager, workerSizeTerms); - logger.info("Finished with usual work location choice for logsum calculations."); - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager after running school and work location choice"); - - } - - public void runHouseholdModels(){ - - logger.info("There are " + householdDataManager.getNumHouseholds()+" households in hh manager before running non-work logsums"); - - HouseholdChoiceModelRunner runner = new HouseholdChoiceModelRunner( propertyMap, jppfClient, "False", householdDataManager, ms, modelStructure, dmuFactory ); - runner.runHouseholdChoiceModels(); - - } - - - - /** - * Create the household data manager. Based on the code in MTCTM2TourBasedModel.runTourBasedModel() - * @return The household data manager interface. - */ - public HouseholdDataManagerIf getHouseholdDataManager( ){ - - - boolean localHandlers = false; - - String testString; - - HouseholdDataManagerIf householdDataManager; - String hhHandlerAddress = ""; - int hhServerPort = 0; - try - { - // get household server address. if none is specified a local server in - // the current process will be started. - hhHandlerAddress = resourceBundle.getString("RunModel.HouseholdServerAddress"); - try - { - // get household server port. - hhServerPort = Integer.parseInt(resourceBundle.getString("RunModel.HouseholdServerPort")); - localHandlers = false; - } catch (MissingResourceException e) - { - // if no household data server address entry is found, the object - // will be created in the local process - localHandlers = true; - } - } catch (MissingResourceException e) - { - localHandlers = true; - } - - - try - { - - if (localHandlers) - { - - // create a new local instance of the household array manager - householdDataManager = new SandagHouseholdDataManager(); - householdDataManager.setPropertyFileValues(propertyMap); - - // have the household data manager read the synthetic population - // files and apply its tables to objects mapping method. - String inputHouseholdFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = resourceBundle.getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); - - } else - { - - householdDataManager = new HouseholdDataManagerRmi(hhHandlerAddress, hhServerPort, - SandagHouseholdDataManager.HH_DATA_SERVER_NAME); - testString = householdDataManager.testRemote(); - logger.info("HouseholdDataManager test: " + testString); - - householdDataManager.setPropertyFileValues(propertyMap); - } - - //always starting from scratch (RunModel.RestartWithHhServer=none) - householdDataManager.setDebugHhIdsFromHashmap(); - - String inputHouseholdFileName = resourceBundle - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_HH); - String inputPersonFileName = resourceBundle - .getString(HouseholdDataManager.PROPERTIES_SYNPOP_INPUT_PERS); - householdDataManager.setHouseholdSampleRate(iterationSampleRate, sampleSeed); - householdDataManager.setupHouseholdDataManager(modelStructure, inputHouseholdFileName, inputPersonFileName); - - }catch (Exception e) - { - - logger.error(String - .format("Exception caught setting up household data manager."), e); - throw new RuntimeException(); - - } - - return householdDataManager; - } - - - /** - * Start a new matrix server connection. - * - * @param properties - */ - private void startMatrixServer(HashMap properties) { - String serverAddress = (String) properties.get("RunModel.MatrixServerAddress"); - int serverPort = new Integer((String) properties.get("RunModel.MatrixServerPort")); - logger.info("connecting to matrix server " + serverAddress + ":" + serverPort); - - try{ - - MatrixDataManager mdm = MatrixDataManager.getInstance(); - ms = new MatrixDataServerRmi(serverAddress, serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - ms.testRemote(Thread.currentThread().getName()); - mdm.setMatrixDataServerObject(ms); - - } catch (Exception e) { - logger.error("could not connect to matrix server", e); - throw new RuntimeException(e); - - } - - } - - - public static void main(String[] args) { - - long startTime = System.currentTimeMillis(); - int globalIterationNumber = -1; - float iterationSampleRate = -1.0f; - int sampleSeed = -1; - - ResourceBundle rb = null; - - logger.info( String.format( "Generating Logsums from MTC Tour Based Model using CT-RAMP version %s, 22feb2011 build %s", CtrampApplication.VERSION, 2 ) ); - - if ( args.length == 0 ) { - logger.error( String.format( "no properties file base name (without .properties extension) was specified as an argument." ) ); - return; - } - else { - rb = ResourceBundle.getBundle( args[0] ); - - // optional arguments - for (int i=1; i < args.length; i++) { - - if (args[i].equalsIgnoreCase("-iteration")) { - globalIterationNumber = Integer.parseInt( args[i+1] ); - logger.info( String.format( "-iteration %d.", globalIterationNumber ) ); - } - - if (args[i].equalsIgnoreCase("-sampleRate")) { - iterationSampleRate = Float.parseFloat( args[i+1] ); - logger.info( String.format( "-sampleRate %.4f.", iterationSampleRate ) ); - } - - if (args[i].equalsIgnoreCase("-sampleSeed")) { - sampleSeed = Integer.parseInt( args[i+1] ); - logger.info( String.format( "-sampleSeed %d.", sampleSeed ) ); - } - - } - - if ( globalIterationNumber < 0 ) { - globalIterationNumber = 1; - logger.info( String.format( "no -iteration flag, default value %d used.", globalIterationNumber ) ); - } - - if ( iterationSampleRate < 0 ) { - iterationSampleRate = 1; - logger.info( String.format( "no -sampleRate flag, default value %.4f used.", iterationSampleRate ) ); - } - - if ( sampleSeed < 0 ) { - sampleSeed = 0; - logger.info( String.format( "no -sampleSeed flag, default value %d used.", sampleSeed ) ); - } - - } - - - String baseName; - if ( args[0].endsWith(".properties") ) { - int index = args[0].indexOf(".properties"); - baseName = args[0].substring(0, index); - } - else { - baseName = args[0]; - } - - - // create an instance of this class for main() to use. - RunModeChoice mainObject = new RunModeChoice( args[0], globalIterationNumber, iterationSampleRate, sampleSeed ); - - // Create logsums - try { - - logger.info ("Creating logsums."); - mainObject.run(); - - } - catch ( RuntimeException e ) { - logger.error ( "RuntimeException caught in com.pb.mtctm2.abm.reports.CreateLogsums.main() -- exiting.", e ); - System.exit(2); - } - - - logger.info (""); - logger.info (""); - logger.info ("CreateLogsums finished in " + ((System.currentTimeMillis() - startTime) / 60000.0) + " minutes."); - - System.exit(0); - - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java b/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java deleted file mode 100644 index f930cca..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/utilities/TapAtConsistencyCheck.java +++ /dev/null @@ -1,183 +0,0 @@ -package org.sandag.abm.utilities; - -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; - -import com.linuxense.javadbf.DBFException; -import com.linuxense.javadbf.DBFReader; - -/** - * The TapAtConsistencyCheck program implements consistency checkng between TAPs in AT and Transit networks. - * There are three types of inconsistencies: - * 1) A TAP in AT network is not in Tranist network - * 2) A TAP in Transit network is not in AT network - * 3) Positions of a TAP are different in AT and Transit networks. - * - * @author wsu - * @mail Wu.Sun@sandag.org - * @version 13.3.0 - * @since 2016-06-20 - * - */ -public class TapAtConsistencyCheck { - private static Logger logger = Logger.getLogger(TapAtConsistencyCheck.class); - private HashMap > atMap; - private HashMap > tranMap; - private float xThreshold; - private float yThreshold; - private String message; - - public TapAtConsistencyCheck(ResourceBundle aRb, String folder){ - xThreshold=Float.parseFloat(aRb.getString("AtTransitConsistency.xThreshold")); - yThreshold=Float.parseFloat(aRb.getString("AtTransitConsistency.yThreshold")); - readAtTaps(folder); - logger.info("Finished reading TAPs in AT network."); - readTranTaps(folder); - logger.info("Finished reading TAPs in transit network."); - } - - public boolean validate(){ - boolean result=false; - message=compareMap(atMap,tranMap); - if(message.equalsIgnoreCase("OK")){ - result=true; - } - return result; - } - - - private void readAtTaps(String folder){ - atMap=new HashMap>(); - Object [] atTapObjects; - - try { - InputStream inputStream = new FileInputStream(folder+"\\SANDAG_Bike_Node.dbf"); - DBFReader atTapReader = new DBFReader( inputStream); - while( (atTapObjects = atTapReader.nextRecord()) != null) { - double tap_at = (double)atTapObjects[3]; - if(tap_at>0){ - ArrayList xy=new ArrayList(); - xy.add((float)atTapObjects[4]); - xy.add((float)atTapObjects[5]); - atMap.put((int)tap_at, xy); - } - } - inputStream.close(); - }catch( DBFException e) { - System.out.println( e.getMessage()); - logger.fatal(e.getMessage()); - System.exit(-1); - }catch( IOException e) { - System.out.println( e.getMessage()); - logger.fatal(e.getMessage()); - System.exit(-1); - } - } - - private void readTranTaps(String folder){ - tranMap=new HashMap>(); - Object [] tranTapObjects; - - try { - InputStream inputStream = new FileInputStream(folder+"\\tapcov.dbf"); - DBFReader tranTapReader = new DBFReader( inputStream); - while( (tranTapObjects = tranTapReader.nextRecord()) != null) { - double tap_tran = (double)tranTapObjects[16]; - ArrayList xy=new ArrayList(); - xy.add((double)tranTapObjects[7]); - xy.add((double)tranTapObjects[8]); - tranMap.put((int)tap_tran, xy); - } - inputStream.close(); - }catch( DBFException e) { - System.out.println( e.getMessage()); - logger.fatal(e.getMessage()); - System.exit(-1); - }catch( IOException e) { - System.out.println( e.getMessage()); - logger.fatal(e.getMessage()); - System.exit(-1); - } - } - -public String compareMap(HashMap> map1, HashMap> map2) { - - String message="OK"; - - if (map1.size()==0){ - message=new ErrorLogging().getAtError("AT4"); - logger.fatal(message); - return message; - } - - if (map2.size()==0){ - message=new ErrorLogging().getAtError("AT5"); - logger.fatal(message); - return message; - } - - for (Integer ch1 : map1.keySet()) { - Float x1=map1.get(ch1).get(0); - Float y1=map1.get(ch1).get(1); - - if(map2.get(ch1)==null||map2.get(ch1)==null){ - message=new ErrorLogging().getAtError("AT1")+"(in SANDAG_Bike_Node.dbf "+"TAP="+ch1+" x_at="+x1+" y_at="+y1+")"; - logger.fatal(message); - }else{ - Double x2=map2.get(ch1).get(0); - Double y2=map2.get(ch1).get(1); - if((Math.abs(x1-x2)>xThreshold)&&(Math.abs(y1-y2)>yThreshold)){ - message=new ErrorLogging().getAtError("AT3")+"("+"TAP="+ch1+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2+")"; - logger.fatal(message); - } - } - - } - - for (Integer ch2 : map2.keySet()) { - Double x2=map2.get(ch2).get(0); - Double y2=map2.get(ch2).get(1); - - if(map1.get(ch2)==null||map1.get(ch2)==null){ - message=new ErrorLogging().getAtError("AT2")+"(in tapcov.dbf "+"TAP="+ch2+" x_tran="+x2+" y_tran="+y2+")"; - logger.fatal(message); - }else{ - Float x1=map1.get(ch2).get(0); - Float y1=map1.get(ch2).get(1); - //System.out.println("TAP="+ch2+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2); - if((Math.abs(x1-x2)>xThreshold)&&(Math.abs(y1-y2)>yThreshold)){ - message=new ErrorLogging().getAtError("AT3")+"("+"TAP="+ch2+" x_at="+x1+" y_at="+y1+" x_tran="+x2+" y_tran="+y2+")"; - logger.fatal(message); - } - } - } - - return message; -} - - public static void main(String[] args) - { - ResourceBundle rb = null; - logger.info("Checking AT and Transit Network Consistency..."); - - if (args.length == 0) - { - logger.error(String.format("no properties file base name (without .properties extension) was specified as an argument.")); - System.exit(-1); - } else - { - rb = ResourceBundle.getBundle(args[0]); - TapAtConsistencyCheck mainObject = new TapAtConsistencyCheck(rb, args[1]); - if(!mainObject.validate()){ - logger.fatal(mainObject.message); - System.exit(-1); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java b/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java deleted file mode 100644 index f596b5c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/validation/MainApplication.java +++ /dev/null @@ -1,437 +0,0 @@ -package org.sandag.abm.validation; - -import java.io.*; -import java.nio.file.Files; -import org.apache.poi.xssf.usermodel.XSSFCell; -import org.apache.poi.xssf.usermodel.XSSFRow; -import org.apache.poi.xssf.usermodel.XSSFSheet; -import org.apache.poi.xssf.usermodel.XSSFWorkbook; -import java.sql.Connection; -import java.sql.DriverManager; -import java.sql.ResultSet; -import java.sql.ResultSetMetaData; -import java.sql.SQLException; -import java.sql.Statement; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.ResourceBundle; - -public class MainApplication { - - public static List getHighwayFlowData(int scenario, ResourceBundle rb) throws SQLException { - - System.out.println("Getting highway flow data for scenario " + scenario + " from database ..."); - - String query = "SELECT " + - " flow.scenario_id," + - " link.hwycov_id " + - " ,sum(CONVERT(bigint,(flow.flow + link_ab_tod.preload / 3.0))) as total_flow " + - "FROM " + - " abm_13_2_3.abm.hwy_flow flow " + - "JOIN abm_13_2_3.abm.hwy_link_ab_tod link_ab_tod " + - " ON flow.scenario_id = link_ab_tod.scenario_id AND flow.hwy_link_ab_tod_id = link_ab_tod.hwy_link_ab_tod_id " + - "JOIN abm_13_2_3.abm.hwy_link_tod link_tod " + - " ON link_ab_tod.scenario_id = link_tod.scenario_id AND link_ab_tod.hwy_link_tod_id = link_tod.hwy_link_tod_id " + - "JOIN abm_13_2_3.abm.hwy_link link " + - " ON link_tod.scenario_id = link.scenario_id AND link_tod.hwy_link_id = link.hwy_link_id " + - "JOIN ref.ifc ifc " + - " ON link.ifc = ifc.ifc " + - "WHERE " + - " flow.scenario_id = " + scenario + - "GROUP BY " + - " flow.scenario_id, link.hwycov_id " + - "ORDER BY " + - " hwycov_id"; - - System.out.println(""); - return getSqlData(query, rb); - } - - public static List getTransitBoardingByModeData(int scenario, ResourceBundle rb) throws SQLException { - - System.out.println("Getting transit boarding by mode data for scenario " + scenario + " from database ..."); - - String query = "SELECT transit_mode_desc,transit_access_mode_desc,sum(boardings) " + - "FROM ws.dbo.transitBoardingSummary(" + scenario + ") " + - "GROUP BY transit_mode_desc,transit_access_mode_desc " + - "ORDER BY transit_mode_desc,transit_access_mode_desc"; - System.out.println(""); - return getSqlData(query, rb); - } - - public static List getTransitBoardingByRouteData(int scenario, ResourceBundle rb) throws SQLException { - - System.out.println("Getting transit boarding by route data for scenario " + scenario + " from database ..."); - - String query = "SELECT rtt, sum(boardings) as boardings " + - "FROM " + - "(SELECT (config/1000) as rtt, boardings " + - "FROM ws.dbo.transitBoardingSummary(" + scenario + ")" + " AS boarding " + - "JOIN abm_13_2_3.abm.transit_route route " + - " ON boarding.route_id = route.transit_route_id " + - "WHERE scenario_id = " + scenario + ") as t " + - "GROUP BY rtt " + - "ORDER BY rtt"; - System.out.println(""); - return getSqlData(query, rb); - } - - public static List getSqlData(String query, ResourceBundle rb) throws SQLException { - - String dbHost = getProperty(rb, "database.host", null); - String dbName = getProperty(rb, "database.name", null); - String user = getProperty(rb, "database.user", null); - String password = getProperty(rb, "database.pwd", null); - - // System.out.println( dbHost ); - // System.out.println( user ); - // System.out.println( password ); - // System.out.println( dbName ); - - Connection conn = null; - conn = getConnectionToDatabase(dbHost, dbName, user, password); - - Statement stmt = conn.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); - ResultSet rs = stmt.executeQuery(query); - ResultSetMetaData rsmd = rs.getMetaData(); - - int nCols = rsmd.getColumnCount(); - - List data = new ArrayList(); - StringBuilder row = null; - - while(rs.next()){ - row = new StringBuilder(); - for(int c = 1; c <= nCols; c++){ - row.append(rs.getString(c)); - - if(c < nCols) - row.append("|"); - } - data.add(row.toString()); - } - - conn.close(); - - return data; - } - - public static void printData(List data){ - for(String str: data) - System.out.println(str); - } - - public static void printRow(String[] vals) { - for (String i : vals) { - System.out.print(i); - System.out.print("\t"); - } - System.out.println(); - } - - //Wu modified to allow writing out files to a different location other than input file location - public static void writeHighwayDataToExcel(String input_file_name, String sheet_to_modify, List data, int scenario, String outputDir) throws IOException { - try { - InputStream input_file = new FileInputStream(new File(input_file_name)); - XSSFWorkbook wb = new XSSFWorkbook(input_file); - input_file.close(); - - XSSFSheet ws = wb.getSheet(sheet_to_modify); - int prevRows = ws.getLastRowNum(); - - XSSFRow row = null; - XSSFCell cell = null; - - // writing data to excel sheet - int r = 1; - for (String str : data) { - row = ws.getRow(r); - - if(row == null){ - // this will happen when the number of links in the template sheet - // are less than the number of highway links for the data obtained from database - ws.createRow(r); - row = ws.getRow(r); - } - - //System.out.println(str); - - String[] vals = str.split("\\|"); - - for(int c = 0; c < vals.length; c++){ - cell = row.getCell(c); - - if(cell == null){ - row.createCell(c); - cell = row.getCell(c); - } - - cell.setCellValue(Integer.valueOf(vals[c])); - } - r++; - } - - // delete additional existing rows, if any - // this is to remove excel rows if the highway links in the template sheet - // are more than the number of highway links for the data obtained from database) - for(int d = data.size() + 1; d < prevRows; d++){ - row = ws.getRow(d); - ws.removeRow(row); - } - - //update all formula calculation - wb.getCreationHelper().createFormulaEvaluator().evaluateAll(); - wb.setForceFormulaRecalculation(true); - - // FormulaEvaluator evaluator = wb.getCreationHelper().createFormulaEvaluator(); - // - // for (int i = 0; i < wb.getNumberOfSheets(); i++) - // { - // Sheet sheet = wb.getSheetAt(i); - // System.out.println("Sheet name " + sheet.getSheetName()); - // for (Row ro : sheet) { - // if(sheet.getSheetName().equalsIgnoreCase("all_nb")){ - // System.out.println("row num " + ro.getRowNum()); - // } - // for (Cell c : ro) { - // if (c.getCellTypeEnum() == CellType.FORMULA) { - // if(sheet.getSheetName().equalsIgnoreCase("all_nb")){ - // System.out.println("col index " + c.getColumnIndex()); - // } - // try { - // evaluator.evaluateFormulaCellEnum(c); - // } catch (Exception e) { - // // TODO Auto-generated catch block - // e.printStackTrace(); - // } - // } - // } - // } - // } - - //forming output file name - String name = input_file_name.substring(0, input_file_name.lastIndexOf(".")); - String ext = input_file_name.substring(input_file_name.lastIndexOf(".") + 1); - String output_file_name = outputDir+name + "_s" + String.valueOf(scenario) + "." + ext; - - //writing out revised excel file - System.out.println("Writing highway data to excel file : " + output_file_name); - - File file = new File(output_file_name); - Files.deleteIfExists(file.toPath()); - - OutputStream output_file = new FileOutputStream(new File(output_file_name)); - wb.write(output_file); - wb.close(); - output_file.close(); - System.out.println(""); - - } catch (FileNotFoundException e) { - e.printStackTrace(); - } catch (IOException e) { - e.printStackTrace(); - } - } - - public static void writeBoardingDataToExcel(String input_file_name, String sheet_to_modify, List data, int scenario, String output_file_name) throws IOException { - try { - InputStream input_file = new FileInputStream(new File(input_file_name)); - XSSFWorkbook wb = new XSSFWorkbook(input_file); - input_file.close(); - - XSSFSheet ws = wb.getSheet(sheet_to_modify); - int prevRows = ws.getLastRowNum(); - - XSSFRow row = null; - XSSFCell cell = null; - - boolean[] colIsNumeric = null; - - // writing data to excel sheet - int r = 1; - for (String str : data) { - row = ws.getRow(r); - - if(row == null){ - // this will happen when the number of routes/modes in the template sheet - // are less than the number of routes/modes for the data obtained from database - ws.createRow(r); - row = ws.getRow(r); - } - - String[] vals = str.split("\\|"); - - //identify columns as string or int/double value - if(r == 1){ - colIsNumeric = new boolean[vals.length]; - Arrays.fill(colIsNumeric, Boolean.TRUE); - - for(int c = 0; c < vals.length; c++) - colIsNumeric[c] = isNumeric(vals[c]); - } - - for(int c = 0; c < vals.length; c++){ - - cell = row.getCell(c); - - if(cell == null){ - row.createCell(c); - cell = row.getCell(c); - } - if(colIsNumeric[c]) - cell.setCellValue(Double.valueOf(vals[c]).intValue()); - else - cell.setCellValue(vals[c]); - } - r++; - } - - // delete additional existing rows, if any - // this is to remove excel rows if the transit routes/modes in the template sheet - // are more than the number of transit routes/modes for the data obtained from database) - for(int d = data.size() + 1; d < prevRows; d++){ - row = ws.getRow(d); - ws.removeRow(row); - } - - //update all formula calculation - wb.getCreationHelper().createFormulaEvaluator().evaluateAll(); - wb.setForceFormulaRecalculation(true); - - //writing out revised excel file - System.out.println("Writing boarding data to excel file : " + output_file_name); - - File file = new File(output_file_name); - Files.deleteIfExists(file.toPath()); - - OutputStream output_file = new FileOutputStream(new File(output_file_name)); - wb.write(output_file); - output_file.close(); - System.out.println(""); - - } catch (FileNotFoundException e) { - e.printStackTrace(); - - } catch (IOException e) { - e.printStackTrace(); - } - } - - public static boolean isNumeric(String str) - { - try{ - double d = Double.parseDouble(str); - } - catch(NumberFormatException nfe) { - return false; - } - return true; - } - - public static String getProperty(ResourceBundle rb, String keyName, String defaultValue) { - - String keyValue = defaultValue; - - try { - keyValue = rb.getString(keyName); - } catch (RuntimeException e) { - //key was not found or resource bundle is null - if(rb == null) throw new RuntimeException("ResourceBundle is null", e); - } - - if(keyValue == null) return keyValue; //you can't trim a null. - - return keyValue.trim(); - } - - /** - * @param dbHost is database server name - * @param dbName is the name of the database we want to use in the server - * @param user is the name of the user used to login to access the database. Note for MS_SQL, user can be null, - * which indicates that standard MS Windows authentication will be used. In this case sqljdbc_auth.dll must be in the java library path. - * @param password is the user's password - not necessary if user is null. - * @return the connection instance for the URL formed for the specific database server specified - */ - - public static Connection getConnectionToDatabase(String dbHost, String dbName, String user, String password) { - Connection conn = null; - - try { - Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver"); - } catch (ClassNotFoundException e1) { - throw new RuntimeException("Class not found ", e1); - } - - try { - String urlToConnect = "jdbc:sqlserver://" + dbHost + ";" + "database=" + dbName + ";" + - (user != null ? ("user=" + user + ";" + "password=" + password) : "integratedSecurity=True"); - - //System.out.println("urlToConnect " + urlToConnect); - conn = DriverManager.getConnection(urlToConnect); - } - catch (SQLException e) { - throw new RuntimeException("Cannot connect to database ", e); - } - return conn; - } - - public static void main(String[] args) throws Exception { - long startTime = System.currentTimeMillis(); - - - // Collection supportedFuncs = WorkbookEvaluator.getSupportedFunctionNames(); - // System.out.println(supportedFuncs); - // - // Collection unsupportedFuncs = WorkbookEvaluator.getNotSupportedFunctionNames(); - // System.err.println(unsupportedFuncs); - // System.exit(-1); - - if ( args.length < 2 ) { - System.out.println( "invalid number of command line arguments." ); - System.out.println( "two argument must be specified, 1) basename of the properties file and 2) scenario number"); - System.exit(-1); - } - - ResourceBundle rb = ResourceBundle.getBundle( args[0] ); - - int scenario = Integer.valueOf(args[1]); - - //Wu modified to allow writing out files to a different location other than input file location - String outputDir=args[2]; - - String sheet_to_modify = null; - - List flow_data = getHighwayFlowData(scenario, rb); - - // revise summary by class workbook - String summary_by_class_file = rb.getString("highway.summary.class.template"); - sheet_to_modify = rb.getString("sheet.to.modify.highway.summary"); - writeHighwayDataToExcel(summary_by_class_file, sheet_to_modify, flow_data, scenario,outputDir); - - // revise summary by corridor workbook - String summary_by_corridor_file = rb.getString("highway.summary.corridor.template"); - sheet_to_modify = rb.getString("sheet.to.modify.highway.summary"); - writeHighwayDataToExcel(summary_by_corridor_file, sheet_to_modify, flow_data, scenario, outputDir); - - // revise transit validation workbook - String transit_validation_file = rb.getString("transit.validation.template"); - - //forming transit output file name - String name = transit_validation_file.substring(0, transit_validation_file.lastIndexOf(".")); - String ext = transit_validation_file.substring(transit_validation_file.lastIndexOf(".") + 1); - String output_file_name = name + "_s" + String.valueOf(scenario) + "." + ext; - String transit_output_file =outputDir+output_file_name; - - List boarding_by_route_data = getTransitBoardingByRouteData(scenario, rb); - sheet_to_modify = rb.getString("sheet.to.modify.transit.boarding.route"); - writeBoardingDataToExcel(transit_validation_file, sheet_to_modify, boarding_by_route_data, scenario, transit_output_file); - - List boarding_by_mode_data = getTransitBoardingByModeData(scenario, rb); - sheet_to_modify = rb.getString("sheet.to.modify.transit.boarding.mode"); - writeBoardingDataToExcel(transit_output_file, sheet_to_modify, boarding_by_mode_data, scenario, transit_output_file); - - System.out.println( String.format( "%s%.1f%s", "Total Run Time - ", ( ( System.currentTimeMillis() - startTime ) / 1000.0 ), " seconds." ) ); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java deleted file mode 100644 index acf7f8c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactory.java +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - */ -package org.sandag.abm.visitor; - -import java.io.Serializable; - -/** - * ArcCtrampDmuFactory is a class that creates Visitor Model DMU objects - * - * @author Joel Freedman - */ -public class VisitorDmuFactory - implements VisitorDmuFactoryIf, Serializable -{ - - private VisitorModelStructure visitorModelStructure; - - public VisitorDmuFactory(VisitorModelStructure modelStructure) - { - this.visitorModelStructure = modelStructure; - } - - public VisitorTourModeChoiceDMU getVisitorTourModeChoiceDMU() - { - return new VisitorTourModeChoiceDMU(visitorModelStructure, null); - } - - public VisitorTourDestChoiceDMU getVisitorTourDestChoiceDMU() - { - return new VisitorTourDestChoiceDMU(visitorModelStructure); - } - - public VisitorStopLocationChoiceDMU getVisitorStopLocationChoiceDMU() - { - return new VisitorStopLocationChoiceDMU(visitorModelStructure); - } - - public VisitorTripModeChoiceDMU getVisitorTripModeChoiceDMU() - { - return new VisitorTripModeChoiceDMU(visitorModelStructure, null); - } - - public VisitorMicromobilityChoiceDMU getVisitorMicromobilityChoiceDMU() - { - return new VisitorMicromobilityChoiceDMU(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java deleted file mode 100644 index d25e087..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorDmuFactoryIf.java +++ /dev/null @@ -1,18 +0,0 @@ -package org.sandag.abm.visitor; - -/** - * A DMU factory interface - */ -public interface VisitorDmuFactoryIf -{ - - VisitorTourModeChoiceDMU getVisitorTourModeChoiceDMU(); - - VisitorTourDestChoiceDMU getVisitorTourDestChoiceDMU(); - - VisitorStopLocationChoiceDMU getVisitorStopLocationChoiceDMU(); - - VisitorTripModeChoiceDMU getVisitorTripModeChoiceDMU(); - - VisitorMicromobilityChoiceDMU getVisitorMicromobilityChoiceDMU(); -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java deleted file mode 100644 index 8634f41..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceDMU.java +++ /dev/null @@ -1,139 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -/** - */ -public class VisitorMicromobilityChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(VisitorMicromobilityChoiceDMU.class); - - protected HashMap methodIndexMap; - - private IndexValues dmuIndex; - protected int income; - protected float walkTime; - protected boolean isTransit; - protected boolean microTransitAvailable; - - - public VisitorMicromobilityChoiceDMU() - { - dmuIndex = new IndexValues(); - setupMethodIndexMap(); - - } - - public void setDmuIndexValues(int hhId, int zoneId, int origTaz, int destTaz) - { - dmuIndex.setHHIndex(hhId); - dmuIndex.setZoneIndex(zoneId); - dmuIndex.setOriginZone(origTaz); - dmuIndex.setDestZone(destTaz); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - } - - - public int getIncome() - { - return income; - } - - public void setIncome(int income) { - this.income = income; - } - - public float getWalkTime() { - return walkTime; - } - - public void setWalkTime(float walkTime) { - this.walkTime = walkTime; - } - - public boolean isTransit() { - return isTransit; - } - - public void setTransit(boolean isTransit) { - this.isTransit = isTransit; - } - - public boolean isMicroTransitAvailable() { - return microTransitAvailable; - } - - public void setMicroTransitAvailable(boolean microTransitAvailable) { - this.microTransitAvailable = microTransitAvailable; - } - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getIncome", 0); - methodIndexMap.put("getWalkTime", 1); - methodIndexMap.put("getIsTransit", 3); - methodIndexMap.put("getMicroTransitAvailable", 4); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - switch (variableIndex) - { - case 0: - return getIncome(); - case 1: - return getWalkTime(); - - case 3: - return isTransit()? 1 : 0; - case 4: - return isMicroTransitAvailable() ? 1 : 0; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java deleted file mode 100644 index 135bc8f..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorMicromobilityChoiceModel.java +++ /dev/null @@ -1,325 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.HashSet; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; - -public class VisitorMicromobilityChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("micromobility"); - - private static final String MM_CONTROL_FILE_TARGET = "visitor.micromobility.uec.file"; - private static final String MM_DATA_SHEET_TARGET = "visitor.micromobility.data.page"; - private static final String MM_MODEL_SHEET_TARGET = "visitor.micromobility.model.page"; - private static final String MT_TAP_FILE_TARGET = "active.microtransit.tap.file"; - private static final String MT_MAZ_FILE_TARGET = "active.microtransit.mgra.file"; - - public static final int MM_MODEL_WALK_ALT = 0; - public static final int MM_MODEL_MICROMOBILITY_ALT = 1; - public static final int MM_MODEL_MICROTRANSIT_ALT = 2; - - private ChoiceModelApplication mmModel; - private VisitorMicromobilityChoiceDMU mmDmuObject; - - private VisitorModelStructure modelStructure; - private MgraDataManager mgraDataManager; - - private HashSet microtransitTaps; - private HashSet microtransitMazs; - - public VisitorMicromobilityChoiceModel(HashMap propertyMap, - VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory) - { - - setupMicromobilityChoiceModelApplication(propertyMap, myModelStructure, dmuFactory); - } - - private void setupMicromobilityChoiceModelApplication(HashMap propertyMap, - VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory) - { - // logger.info("setting up micromobility choice model."); - - modelStructure = myModelStructure; - - // locate the micromobility choice UEC - String uecFileDirectory = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mmUecFile = uecFileDirectory + propertyMap.get(MM_CONTROL_FILE_TARGET); - - int dataSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_DATA_SHEET_TARGET); - int modelSheet = Util.getIntegerValueFromPropertyMap(propertyMap, MM_MODEL_SHEET_TARGET); - - // create the micromobility choice model DMU object. - mmDmuObject = dmuFactory.getVisitorMicromobilityChoiceDMU(); - - // create the transponder choice model object - mmModel = new ChoiceModelApplication(mmUecFile, modelSheet, dataSheet, propertyMap, - (VariableTable) mmDmuObject); - - mgraDataManager = MgraDataManager.getInstance(); - String projectDirectory = propertyMap.get(CtrampApplication.PROPERTIES_PROJECT_DIRECTORY); - String microTransitTapFile = projectDirectory + propertyMap.get(MT_TAP_FILE_TARGET); - String microTransitMazFile = projectDirectory + propertyMap.get(MT_MAZ_FILE_TARGET); - - TableDataSet microTransitTapData = Util.readTableDataSet(microTransitTapFile); - TableDataSet microTransitMazData = Util.readTableDataSet(microTransitMazFile); - - microtransitTaps = new HashSet(); - microtransitMazs = new HashSet(); - - for(int i=1;i<=microTransitTapData.getRowCount();++i) { - - int tap = (int) microTransitTapData.getValueAt(i,"TAP"); - microtransitTaps.add(tap); - } - - for(int i=1;i<=microTransitMazData.getRowCount();++i) { - - int maz = (int) microTransitMazData.getValueAt(i,"MGRA"); - microtransitMazs.add(maz); - } - - } - - - public void applyModel(VisitorTour tour) { - - //apply to trips on tour - if(tour.getTrips()!=null) { - - for(VisitorTrip trip: tour.getTrips()) - applyModel(tour, trip); - } - - - } - - public void applyModel(VisitorTour tour, VisitorTrip trip) - { - - - if(!modelStructure.getTourModeIsWalk(trip.getTripMode()) && !modelStructure.getTourModeIsWalkTransit(trip.getTripMode())&& !modelStructure.getTourModeIsDriveTransit(trip.getTripMode())) - return; - - mmDmuObject.setIncome(tour.getIncome()); - int originMaz = trip.getOriginMgra(); - int destMaz = trip.getDestinationMgra(); - if(modelStructure.getTourModeIsWalk(trip.getTripMode())) - mmDmuObject.setTransit(false); - else - mmDmuObject.setTransit(true); - - - if(modelStructure.getTourModeIsWalk(trip.getTripMode())) { - - float walkTime = mgraDataManager.getMgraToMgraWalkTime(originMaz, destMaz); - mmDmuObject.setWalkTime(walkTime); - - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); - - if(microtransitMazs.contains(originMaz) && microtransitMazs.contains(destMaz)) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - trip.setMicromobilityWalkLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice(tour, trip); - trip.setMicromobilityWalkMode(chosenAlt); - - // write choice model alternative info to log file - if (tour.getDebugChoiceModels()) - { - String decisionMaker = String.format("Tour "+tour.getID()+ " mode " +trip.getTripMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Transponder Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - - - }else if(modelStructure.getTourModeIsWalkTransit(trip.getTripMode())) { - - //access - int tapPosition = mgraDataManager.getTapPosition(originMaz, trip.getBoardTap()); - float walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); - mmDmuObject.setWalkTime(walkTime); - - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); - - if(microtransitTaps.contains(trip.getBoardTap())) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - trip.setMicromobilityAccessLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice( tour, trip); - trip.setMicromobilityAccessMode(chosenAlt); - - // write choice model alternative info to log file - if (tour.getDebugChoiceModels()) - { - String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Transponder Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - //egress - tapPosition = mgraDataManager.getTapPosition(destMaz, trip.getAlightTap()); - walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); - mmDmuObject.setWalkTime(walkTime); - - if(microtransitTaps.contains(trip.getAlightTap())) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC - int closestMazToAlightTap = mgraDataManager.getClosestMgra(trip.getAlightTap()); - mmDmuObject.setDmuIndexValues(tour.getID(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); - - // compute utilities and choose micromobility choice alternative. - logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - trip.setMicromobilityEgressLogsum(logsum); - - // if the choice model has at least one available alternative, make choice - chosenAlt = (byte) getChoice( tour, trip); - trip.setMicromobilityEgressMode(chosenAlt); - - // write choice model alternative info to log file - if (tour.getDebugChoiceModels()) - { - String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Transponder Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - } else if( modelStructure.getTourModeIsDriveTransit(trip.getTripMode()) ) { //drive-transit. Choose non-drive direction - - int tapPosition = 0; - float walkTime = 9999; - - if(trip.isInbound()) { //inbound, so access mode is walk - tapPosition = mgraDataManager.getTapPosition(originMaz, trip.getBoardTap()); - walkTime = mgraDataManager.getMgraToTapWalkTime(originMaz, tapPosition); - //set destination to origin so that Z can be used to find origin zone access to mode in mgra data file in UEC - mmDmuObject.setDmuIndexValues(tour.getID(), originMaz, originMaz, originMaz); - - if(microtransitTaps.contains(trip.getBoardTap())) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - }else { //outbound so egress mode is walk. - tapPosition = mgraDataManager.getTapPosition(destMaz, trip.getAlightTap()); - walkTime = mgraDataManager.getMgraToTapWalkTime(destMaz, tapPosition); - //set destination to closest mgra to alighting TAP so that Z can be used to find access to mode in mgra data file in UEC - int closestMazToAlightTap = mgraDataManager.getClosestMgra(trip.getAlightTap()); - mmDmuObject.setDmuIndexValues(tour.getID(), closestMazToAlightTap, closestMazToAlightTap, closestMazToAlightTap); - } - mmDmuObject.setWalkTime(walkTime); - - if(microtransitTaps.contains(trip.getAlightTap())) - mmDmuObject.setMicroTransitAvailable(true); - else - mmDmuObject.setMicroTransitAvailable(false); - - // compute utilities and choose micromobility choice alternative. - float logsum = (float) mmModel.computeUtilities(mmDmuObject, mmDmuObject.getDmuIndexValues()); - - // if the choice model has at least one available alternative, make choice - byte chosenAlt = (byte) getChoice(tour, trip); - - if(trip.isInbound()) { //inbound, set access - trip.setMicromobilityAccessMode(chosenAlt); - trip.setMicromobilityAccessLogsum(logsum); - }else { //outound, set egress - trip.setMicromobilityEgressMode(chosenAlt); - trip.setMicromobilityEgressLogsum(logsum); - } - - // write choice model alternative info to log file - if (tour.getDebugChoiceModels()) - { - String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); - mmModel.logAlternativesInfo("Micromobility Choice", decisionMaker, logger); - logger.info(String.format("%s result chosen for %s is %d", - "Transponder Choice", decisionMaker, chosenAlt)); - mmModel.logUECResults(logger, decisionMaker); - } - - } - - } - - - /** - * Select the micromobility mode from the UEC. This is helper code for applyModel(), where utilities have already been calculated. - * - * @param household - * @param person - * @param tour - * @param s - * @return The micromobility mode. - */ - private int getChoice(VisitorTour tour, VisitorTrip trip) { - // if the choice model has at least one available alternative, make - // choice. - int chosenAlt; - if (mmModel.getAvailabilityCount() > 0) - { - double randomNumber = tour.getRandom(); - chosenAlt = mmModel.getChoiceResult(randomNumber); - return chosenAlt; - } else - { - String decisionMaker = String.format("Tour %d", tour.getID()+ " mode " +trip.getTripMode()); - String errorMessage = String - .format("Exception caught for %s, no available micromobility choice alternatives to choose from in choiceModelApplication.", - decisionMaker); - logger.info(errorMessage); - - mmModel.logUECResults(logger, decisionMaker); - return MM_MODEL_WALK_ALT; - } - - } - - /** - * This method calculates a cost coefficient based on the following formula: - * - * costCoeff = incomeCoeff * 1/(max(income,1000)^incomeExponent) - * - * - * @param incomeCoeff - * @param incomeExponent - * @return A cost coefficent that should be multiplied by cost variables (cents) in tour mode choice - */ - public double calculateCostCoefficient(double income, double incomeCoeff, double incomeExponent){ - - return incomeCoeff * 1.0/(Math.pow(Math.max(income,1000.0),incomeExponent)); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java deleted file mode 100644 index a45b6f5..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModel.java +++ /dev/null @@ -1,499 +0,0 @@ -package org.sandag.abm.visitor; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.concurrent.ForkJoinPool; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.crossborder.CrossBorderTripModeChoiceModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; -import com.pb.sawdust.util.concurrent.DnCRecursiveAction; - - -public class VisitorModel -{ - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - public static final String RUN_MODEL_CONCURRENT_PROPERTY_KEY = "visitor.run.concurrent"; - public static final String CONCURRENT_PARALLELISM_PROPERTY_KEY = "visitor.concurrent.parallelism"; - - private MatrixDataServerRmi ms; - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - private static final Object INITIALIZATION_LOCK = new Object(); - - private HashMap rbMap; - private McLogsumsCalculator logsumsCalculator; - private AutoTazSkimsCalculator tazDistanceCalculator; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - - private boolean seek; - private int traceId; - private double sampleRate = 1; - private static int iteration=1; - - /** - * Constructor - * - * @param rbMap - */ - public VisitorModel(HashMap rbMap) - { - this.rbMap = rbMap; - synchronized (INITIALIZATION_LOCK) - { // lock to make sure only one of - // these actually initializes - // things so we don't cross - // threads - mgraManager = MgraDataManager.getInstance(rbMap); - tazManager = TazDataManager.getInstance(rbMap); - } - - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "visitor.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "visitor.trace")); - - } - - // global variable used for reporting - private static final AtomicInteger TOUR_COUNTER = new AtomicInteger(0); - private final AtomicBoolean calculatorsInitialized = new AtomicBoolean(false); - - /** - * Run visitor model. - */ - private void runModel(VisitorTour[] tours, int start, int end){ - - - VisitorModelStructure modelStructure = new VisitorModelStructure(); - - VisitorDmuFactoryIf dmuFactory = new VisitorDmuFactory(modelStructure); - - if (!calculatorsInitialized.get()) - { - // only let one thread in to initialize - synchronized (calculatorsInitialized) - { - // if still not initialized, then this is the first in so do the - // initialization (otherwise skip) - if (!calculatorsInitialized.get()) - { - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - calculatorsInitialized.set(true); - } - } - } - - - VisitorTourTimeOfDayChoiceModel todChoiceModel = new VisitorTourTimeOfDayChoiceModel(rbMap); - VisitorTourDestChoiceModel destChoiceModel = new VisitorTourDestChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - VisitorTourModeChoiceModel tourModeChoiceModel = destChoiceModel.getTourModeChoiceModel(); - //VisitorTripModeChoiceModel tripModeChoiceModel = tourModeChoiceModel.getTripModeChoiceModel(); - destChoiceModel.calculateSizeTerms(dmuFactory); - destChoiceModel.calculateTazProbabilities(dmuFactory); - - VisitorStopFrequencyModel stopFrequencyModel = new VisitorStopFrequencyModel(rbMap); - VisitorStopPurposeModel stopPurposeModel = new VisitorStopPurposeModel(rbMap); - VisitorStopTimeOfDayChoiceModel stopTodChoiceModel = new VisitorStopTimeOfDayChoiceModel(rbMap); - VisitorStopLocationChoiceModel stopLocationChoiceModel = new VisitorStopLocationChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - VisitorTripModeChoiceModel tripModeChoiceModel = new VisitorTripModeChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - VisitorMicromobilityChoiceModel micromobilityChoiceModel = new VisitorMicromobilityChoiceModel(rbMap,modelStructure, dmuFactory); - - double[][] mgraSizeTerms = destChoiceModel.getMgraSizeTerms(); - double[][] tazSizeTerms = destChoiceModel.getTazSizeTerms(); - double[][][] mgraProbabilities = destChoiceModel.getMgraProbabilities(); - stopLocationChoiceModel.setMgraSizeTerms(mgraSizeTerms); - stopLocationChoiceModel.setTazSizeTerms(tazSizeTerms); - stopLocationChoiceModel.setMgraProbabilities(mgraProbabilities); - stopLocationChoiceModel.setTripModeChoiceModel(tripModeChoiceModel); - - // Run models for array of tours - for (int i = start; i < end; i++) - { - VisitorTour tour = tours[i]; - - // sample tours - double rand = tour.getRandom(); - if (rand > sampleRate) continue; - - int tourCount = TOUR_COUNTER.incrementAndGet(); - if (tourCount % 1000 == 0) logger.info("Processing tour " + tourCount); - - if (seek && tour.getID() != traceId) continue; - - if (tour.getID() == traceId) - tour.setDebugChoiceModels(true); - - - todChoiceModel.calculateTourTOD(tour); - destChoiceModel.chooseDestination(tour); - tourModeChoiceModel.chooseTourMode(tour); - - stopFrequencyModel.calculateStopFrequency(tour); - stopPurposeModel.calculateStopPurposes(tour); - - int outboundStops = tour.getNumberOutboundStops(); - int inboundStops = tour.getNumberInboundStops(); - - // choose TOD for stops and location of each - if (outboundStops > 0) - { - VisitorStop[] stops = tour.getOutboundStops(); - for (VisitorStop stop : stops) - { - stopTodChoiceModel.chooseTOD(tour, stop); - stopLocationChoiceModel.chooseStopLocation(tour, stop); - } - } - if (inboundStops > 0) - { - VisitorStop[] stops = tour.getInboundStops(); - for (VisitorStop stop : stops) - { - stopTodChoiceModel.chooseTOD(tour, stop); - stopLocationChoiceModel.chooseStopLocation(tour, stop); - } - } - - // generate trips and choose mode for them - VisitorTrip[] trips = new VisitorTrip[outboundStops + inboundStops + 2]; - int tripNumber = 0; - - // outbound stops - if (outboundStops > 0) - { - VisitorStop[] stops = tour.getOutboundStops(); - for (VisitorStop stop : stops) - { - // generate a trip to the stop and choose a mode for it - trips[tripNumber] = new VisitorTrip(tour, stop, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - // generate a trip from the last stop to the tour destination - trips[tripNumber] = new VisitorTrip(tour, stops[stops.length - 1], false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - - } else - { - // generate an outbound trip from the tour origin to the - // destination and choose a mode - trips[tripNumber] = new VisitorTrip(tour, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - - // inbound stops - if (inboundStops > 0) - { - VisitorStop[] stops = tour.getInboundStops(); - for (VisitorStop stop : stops) - { - // generate a trip to the stop and choose a mode for it - trips[tripNumber] = new VisitorTrip(tour, stop, true); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - // generate a trip from the last stop to the tour origin - trips[tripNumber] = new VisitorTrip(tour, stops[stops.length - 1], false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } else - { - - // generate an inbound trip from the tour destination to the - // origin and choose a mode - trips[tripNumber] = new VisitorTrip(tour, false); - tripModeChoiceModel.chooseMode(tour, trips[tripNumber]); - ++tripNumber; - } - - // set the trips in the tour object - tour.setTrips(trips); - micromobilityChoiceModel.applyModel(tour); - - } - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @return the sampleRate - */ - public double getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(double sampleRate) - { - this.sampleRate = sampleRate; - } - - /** - * This class is the divide-and-conquer action (void return task) for - * running the visitor model using the fork-join framework. The - * divisible problem is an array of tours, and the actual work is the - * {@link VisitorModel#runModel(VisitorTour[],int,int)} method, - * applied to a section of the array. - */ - private class VisitorModelAction - extends DnCRecursiveAction - { - private final HashMap rbMap; - private final VisitorTour[] tours; - - private VisitorModelAction(HashMap rbMap, VisitorTour[] tours) - { - super(0, tours.length); - this.rbMap = rbMap; - this.tours = tours; - } - - private VisitorModelAction(HashMap rbMap, VisitorTour[] tours, - long start, long length, DnCRecursiveAction next) - { - super(start, length, next); - this.rbMap = rbMap; - this.tours = tours; - } - - @Override - protected void computeAction(long start, long length) - { - runModel(tours, (int) start, (int) (start + length)); - } - - @Override - protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) - { - return new VisitorModelAction(rbMap, tours, start, length, next); - } - - @Override - protected boolean continueDividing(long length) - { - // if there are 3 extra tasks queued up, then start executing - // if there are 1000 or less tours to process, then start executing - // otherwise, keep dividing to build up tasks for the threads to - // process - return getSurplusQueuedTaskCount() < 3 && length > 5000; - } - } - - /** - * Run visitor model. - */ - public void runModel() - { - VisitorTourManager tourManager = new VisitorTourManager(rbMap); - tourManager.generateVisitorTours(); - VisitorTour[] tours = tourManager.getTours(); - - // get new keys to see if we want to run in concurrent mode, and the - // parallelism - // (defaults to single threaded and parallelism = # of processors) - // note that concurrent can use up memory very quickly, so setting the - // parallelism might be prudent - boolean concurrent = rbMap.containsKey(RUN_MODEL_CONCURRENT_PROPERTY_KEY) - && Boolean.valueOf(Util.getStringValueFromPropertyMap(rbMap, - RUN_MODEL_CONCURRENT_PROPERTY_KEY)); - int parallelism = rbMap.containsKey(CONCURRENT_PARALLELISM_PROPERTY_KEY) ? Integer - .valueOf(Util.getStringValueFromPropertyMap(rbMap, - CONCURRENT_PARALLELISM_PROPERTY_KEY)) : Runtime.getRuntime() - .availableProcessors(); - - if (concurrent) - { // use fork-join - VisitorModelAction action = new VisitorModelAction(rbMap, tours); - new ForkJoinPool(parallelism).execute(action); - action.getResult(); // wait for finish - } else - { // single-threaded: call the model runner in this thread - runModel(tours, 0, tours.length); - } - - tourManager.writeOutputFile(rbMap); - logger.info("Visitor Model successfully completed!"); - } - - /** - * @param args - */ - public static void main(String[] args) - { - Runtime gfg = Runtime.getRuntime(); - long memory1; - // checking the total memeory - System.out.println("Total memory is: "+ gfg.totalMemory()); - // checking free memory - memory1 = gfg.freeMemory(); - System.out.println("Initial free memory at Visitor model: "+ memory1); - // calling the garbage collector on demand - gfg.gc(); - memory1 = gfg.freeMemory(); - System.out.println("Free memory after garbage "+ "collection: " + memory1); - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Visitor Model")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - VisitorModel visitorModel = new VisitorModel(pMap); - - float sampleRate = 1.0f; - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - logger.info("Visitor Model:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - visitorModel.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = visitorModel.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - visitorModel.ms = matrixServer; - } else - { - visitorModel.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - visitorModel.ms.testRemote("VisitorModel"); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(visitorModel.ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - visitorModel.runModel(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java deleted file mode 100644 index 920f59a..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorModelStructure.java +++ /dev/null @@ -1,95 +0,0 @@ -package org.sandag.abm.visitor; - -import org.sandag.abm.application.SandagModelStructure; - -public class VisitorModelStructure - extends SandagModelStructure -{ - - public static final byte NUMBER_VISITOR_PURPOSES = 6; - public static final byte WORK = 0; - public static final byte RECREATION = 1; - public static final byte DINING = 2; - - public static final String[] VISITOR_PURPOSES = {"WORK", "RECREATE", "DINING"}; - - // override on max tour mode, since we have taxi in this model. - public static final int MAXIMUM_TOUR_MODE_ALT_INDEX = 13; - - public static final byte NUMBER_VISITOR_SEGMENTS = 2; - public static final byte BUSINESS = 0; - public static final byte PERSONAL = 1; - - public static final String[] VISITOR_SEGMENTS = {"BUSINESS", "PERSONAL"}; - public static final byte DEPARTURE = 0; - public static final byte ARRIVAL = 1; - - public static final byte INCOME_SEGMENTS = 5; - - // note that time periods start at 1 and go to 40 - public static final byte TIME_PERIODS = 40; - - public static final int AM = 0; - public static final int PM = 1; - public static final int OP = 2; - public static final int[] SKIM_PERIODS = {AM, PM, OP}; - public static final String[] SKIM_PERIOD_STRINGS = {"AM", "PM", "OP"}; - public static final int UPPER_EA = 3; - public static final int UPPER_AM = 9; - public static final int UPPER_MD = 22; - public static final int UPPER_PM = 29; - public static final String[] MODEL_PERIOD_LABELS = {"EA", "AM", "MD", "PM", "EV"}; - - public static final byte TAXI = 13; - - /** - * Taxi tour mode - * - * @param tourMode - * @return - */ - public boolean getTourModeIsTaxi(int tourMode) - { - - if (tourMode == TAXI) return true; - else return false; - - } - - /** - * return the Skim period index 0=am, 1=pm, 2=off-peak - */ - public static int getSkimPeriodIndex(int departPeriod) - { - - int skimPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_AM) skimPeriodIndex = AM; - else if (departPeriod <= UPPER_MD) skimPeriodIndex = OP; - else if (departPeriod <= UPPER_PM) skimPeriodIndex = PM; - else skimPeriodIndex = OP; - - return skimPeriodIndex; - - } - - /** - * return the Model period index 0=EA, 1=AM, 2=MD, 3=PM, 4=EV - */ - public static int getModelPeriodIndex(int departPeriod) - { - - int modelPeriodIndex = 0; - - if (departPeriod <= UPPER_EA) modelPeriodIndex = 0; - else if (departPeriod <= UPPER_AM) modelPeriodIndex = 1; - else if (departPeriod <= UPPER_MD) modelPeriodIndex = 2; - else if (departPeriod <= UPPER_PM) modelPeriodIndex = 3; - else modelPeriodIndex = 4; - - return modelPeriodIndex; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java deleted file mode 100644 index 86e0499..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStop.java +++ /dev/null @@ -1,176 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import org.apache.log4j.Logger; - -public class VisitorStop - implements Serializable -{ - - private int id; - private int mode; - private int period; - private boolean inbound; - private int mgra; - private byte purpose; - - VisitorTour parentTour; - - public VisitorStop(VisitorTour parentTour, int id, boolean inbound) - { - this.parentTour = parentTour; - this.id = id; - this.inbound = inbound; - } - - /** - * @return the mgra - */ - public int getMgra() - { - return mgra; - } - - /** - * @param mgra - * the mgra to set - */ - public void setMgra(int mgra) - { - this.mgra = mgra; - } - - public void setMode(int mode) - { - this.mode = mode; - } - - public void setPeriod(int period) - { - this.period = period; - } - - /** - * @return the id - */ - public int getId() - { - return id; - } - - /** - * @param id - * the id to set - */ - public void setId(int id) - { - this.id = id; - } - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the parentTour - */ - public VisitorTour getParentTour() - { - return parentTour; - } - - /** - * @param parentTour - * the parentTour to set - */ - public void setParentTour(VisitorTour parentTour) - { - this.parentTour = parentTour; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(byte stopPurposeIndex) - { - this.purpose = stopPurposeIndex; - } - - public byte getPurpose() - { - return purpose; - } - - public int getMode() - { - return mode; - } - - public int getStopPeriod() - { - return period; - } - - public VisitorTour getTour() - { - return parentTour; - } - - public int getStopId() - { - return id; - } - - public void logStopObject(Logger logger, int totalChars) - { - - String separater = ""; - for (int i = 0; i < totalChars; i++) - separater += "-"; - - String purposeString = VisitorModelStructure.VISITOR_PURPOSES[purpose]; - logHelper(logger, "stopId: ", id, totalChars); - logHelper(logger, "mgra: ", mgra, totalChars); - logHelper(logger, "mode: ", mode, totalChars); - logHelper(logger, "purpose: ", purposeString, totalChars); - logHelper(logger, "direction: ", inbound ? "inbound" : "outbound", totalChars); - logHelper(logger, inbound ? "outbound departPeriod: " : "inbound arrivePeriod: ", period, - totalChars); - logger.info(separater); - logger.info(""); - logger.info(""); - - } - - public static void logHelper(Logger logger, String label, int value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%dd", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } - - public static void logHelper(Logger logger, String label, String value, int totalChars) - { - int labelChars = label.length() + 2; - int remainingChars = totalChars - labelChars - 4; - String formatString = String.format(" %%%ds %%%ds", label.length(), remainingChars); - String logString = String.format(formatString, label, value); - logger.info(logString); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java deleted file mode 100644 index ed33801..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopFrequencyModel.java +++ /dev/null @@ -1,325 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the stop frequency model for visitor tours. It is currently - * based on a static probability distribution stored in an input file, and - * indexed into by tour purpose and duration. - * - * @author Freedman - * - */ -public class VisitorStopFrequencyModel -{ - private transient Logger logger = Logger.getLogger("visitorModel"); - - private double[][] cumProbability; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - private int[][] lowerBoundDurationHours; // by - // purpose, - // alternative: - // lower - // bound - // in - // hours - private int[][] upperBoundDurationHours; // by - // purpose, - // alternative: - // upper - // bound - // in - // hours - private int[][] outboundStops; // by - // purpose, - // alternative: - // number - // of - // outbound - // stops - private int[][] inboundStops; // by - // purpose, - // alternative: - // number - // of - // inbound - // stops - VisitorModelStructure modelStructure; - - /** - * Constructor. - */ - public VisitorStopFrequencyModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.stop.frequency.file"); - stopFrequencyFile = directory + stopFrequencyFile; - - modelStructure = new VisitorModelStructure(); - - readStopFrequencyFile(stopFrequencyFile); - - } - - /** - * Read the stop frequency distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readStopFrequencyFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating stop frequency probability distribution"); - - int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 - - int[] alts = new int[purposes]; - - // take a pass through the data and see how many alternatives there are - // for each purpose - int rowCount = probabilityTable.getRowCount(); - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - ++alts[purpose]; - } - - // initialize all the arrays - cumProbability = new double[purposes][]; - lowerBoundDurationHours = new int[purposes][]; - upperBoundDurationHours = new int[purposes][]; - outboundStops = new int[purposes][]; - inboundStops = new int[purposes][]; - - for (int i = 0; i < purposes; ++i) - { - cumProbability[i] = new double[alts[i]]; - lowerBoundDurationHours[i] = new int[alts[i]]; - upperBoundDurationHours[i] = new int[alts[i]]; - outboundStops[i] = new int[alts[i]]; - inboundStops[i] = new int[alts[i]]; - } - - // fill up arrays - int lastPurpose = 0; - int lastLowerBound = 0; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - int lowerBound = (int) probabilityTable.getValueAt(row, "DurationLo"); - int upperBound = (int) probabilityTable.getValueAt(row, "DurationHi"); - int outStops = (int) probabilityTable.getValueAt(row, "Outbound"); - int inbStops = (int) probabilityTable.getValueAt(row, "Inbound"); - - // reset cumulative probability if new purpose or lower-bound - if (purpose != lastPurpose || lowerBound != lastLowerBound) - { - - // log cumulative probability just in case - logger.info("Cumulative probability for purpose " + purpose + " lower bound " - + lowerBound + " is " + cumProb); - cumProb = 0; - } - - if (purpose != lastPurpose) alt = 0; - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - lowerBoundDurationHours[purpose][alt] = lowerBound; - upperBoundDurationHours[purpose][alt] = upperBound; - outboundStops[purpose][alt] = outStops; - inboundStops[purpose][alt] = inbStops; - - ++alt; - - lastPurpose = purpose; - lastLowerBound = lowerBound; - } - - logger.info("End calculating stop frequency probability distribution"); - - for (int purp = 0; purp < purposes; ++purp) - { - for (int a = 0; a < cumProbability[purp].length; ++a) - { - logger.info("Purpose " + purp + " lower " + lowerBoundDurationHours[purp][a] - + " upper " + upperBoundDurationHours[purp][a] + " cumProb " - + cumProbability[purp][a]); - } - } - - } - - /** - * Calculate number of stops for the tour. - * - * @param tour - * A visitor tour (with purpose and mode chosen) - */ - public void calculateStopFrequency(VisitorTour tour) - { - - int purpose = tour.getPurpose(); - double random = tour.getRandom(); - - int tourMode = tour.getTourMode(); - - if (!modelStructure.getTourModeIsSovOrHov(tourMode) - && !modelStructure.getTourModeIsTaxi(tourMode)) return; - - if (tour.getDebugChoiceModels()) - { - logger.info("Choosing stop frequency for purpose " - + modelStructure.VISITOR_PURPOSES[purpose] + " using random number " + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - if (!tourIsInRange(tour, lowerBoundDurationHours[purpose][i], - upperBoundDurationHours[purpose][i])) continue; - - if (tour.getDebugChoiceModels()) - { - logger.info("lower bound " + lowerBoundDurationHours[purpose][i] + " upper bound " - + upperBoundDurationHours[purpose][i]); - } - - if (random < cumProbability[purpose][i]) - { - int outStops = outboundStops[purpose][i]; - int inbStops = inboundStops[purpose][i]; - - if (outStops > 0) - { - VisitorStop[] stops = generateOutboundStops(tour, outStops); - tour.setOutboundStops(stops); - } - - if (inbStops > 0) - { - VisitorStop[] stops = generateInboundStops(tour, inbStops); - tour.setInboundStops(stops); - } - if (tour.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose " + outStops + " outbound stops and " + inbStops - + " inbound stops"); - logger.info(""); - } - break; - } - } - - } - - /** - * Check if the tour duration is in range - * - * @param tour - * @param lowerBound - * @param upperBound - * @return True if tour duration is greater than or equal to lower and - */ - private boolean tourIsInRange(VisitorTour tour, int lowerBound, int upperBound) - { - - float depart = (float) tour.getDepartTime(); - float arrive = (float) tour.getArriveTime(); - - float halfHours = arrive + 1 - depart; // at least 30 minutes - float tourDurationInHours = halfHours * (float) 0.5; - - if ((tourDurationInHours >= (float) lowerBound) - && (tourDurationInHours <= (float) upperBound)) return true; - - return false; - } - - /** - * Generate an array of outbound stops, from tour origin to primary - * destination, in order. - * - * @param tour - * The parent tour. - * @param numberOfStops - * Number of stops from stop frequency model. - * @return The array of outbound stops. - */ - private VisitorStop[] generateOutboundStops(VisitorTour tour, int numberOfStops) - { - - VisitorStop[] stops = new VisitorStop[numberOfStops]; - - for (int i = 0; i < stops.length; ++i) - { - VisitorStop stop = new VisitorStop(tour, i, false); - stops[i] = stop; - stop.setInbound(false); - stop.setParentTour(tour); - } - - return stops; - } - - /** - * Generate an array of inbound stops, from primary dest back to tour - * origin, in order. - * - * @param tour - * Parent tour. - * @param numberOfStops - * Number of stops from stop frequency model. - * @return The array of inbound stops. - */ - private VisitorStop[] generateInboundStops(VisitorTour tour, int numberOfStops) - { - - VisitorStop[] stops = new VisitorStop[numberOfStops]; - - for (int i = 0; i < stops.length; ++i) - { - VisitorStop stop = new VisitorStop(tour, i, true); - stops[i] = stop; - stop.setInbound(true); - stop.setParentTour(tour); - - } - - return stops; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java deleted file mode 100644 index 8fabb9c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceDMU.java +++ /dev/null @@ -1,406 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class VisitorStopLocationChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("visitorModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected int purpose; - protected int stopsOnHalfTour; - protected int stopNumber; - protected int inboundStop; - protected int tourDuration; - - protected double[][] sizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgra) - protected double[] correctionFactors; // by - // alternative - // (sampled - // mgra, - // for - // full - // model - // only) - - protected int[] sampleNumber; // by - // alternative - // (taz - // or - // sampled - // mgra) - - protected double[] osMcLogsumAlt; - protected double[] sdMcLogsumAlt; - - protected double[] tourOrigToStopDistanceAlt; - protected double[] stopToTourDestDistanceAlt; - - public VisitorStopLocationChoiceDMU(VisitorModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - /** - * @return the stopsOnHalfTour - */ - public int getStopsOnHalfTour() - { - return stopsOnHalfTour; - } - - /** - * @param stopsOnHalfTour - * the stopsOnHalfTour to set - */ - public void setStopsOnHalfTour(int stopsOnHalfTour) - { - this.stopsOnHalfTour = stopsOnHalfTour; - } - - /** - * @return the stopNumber - */ - public int getStopNumber() - { - return stopNumber; - } - - /** - * @param stopNumber - * the stopNumber to set - */ - public void setStopNumber(int stopNumber) - { - this.stopNumber = stopNumber; - } - - /** - * @return the inboundStop - */ - public int getInboundStop() - { - return inboundStop; - } - - /** - * @param inboundStop - * the inboundStop to set - */ - public void setInboundStop(int inboundStop) - { - this.inboundStop = inboundStop; - } - - /** - * @return the tourDuration - */ - public int getTourDuration() - { - return tourDuration; - } - - /** - * @param tourDuration - * the tourDuration to set - */ - public void setTourDuration(int tourDuration) - { - this.tourDuration = tourDuration; - } - - /** - * @return the sampleNumber - */ - public int getSampleNumber(int alt) - { - return sampleNumber[alt]; - } - - /** - * @param sampleNumber - * the sampleNumber to set - */ - public void setSampleNumber(int[] sampleNumber) - { - this.sampleNumber = sampleNumber; - } - - /** - * @return the osMcLogsumAlt - */ - public double getOsMcLogsumAlt(int alt) - { - return osMcLogsumAlt[alt]; - } - - /** - * @param osMcLogsumAlt - * the osMcLogsumAlt to set - */ - public void setOsMcLogsumAlt(double[] osMcLogsumAlt) - { - this.osMcLogsumAlt = osMcLogsumAlt; - } - - /** - * @return the sdMcLogsumAlt - */ - public double getSdMcLogsumAlt(int alt) - { - return sdMcLogsumAlt[alt]; - } - - /** - * @param sdMcLogsumAlt - * the sdMcLogsumAlt to set - */ - public void setSdMcLogsumAlt(double[] sdMcLogsumAlt) - { - this.sdMcLogsumAlt = sdMcLogsumAlt; - } - - /** - * @return the tourOrigToStopDistanceAlt - */ - public double getTourOrigToStopDistanceAlt(int alt) - { - return tourOrigToStopDistanceAlt[alt]; - } - - /** - * @param tourOrigToStopDistanceAlt - * the tourOrigToStopDistanceAlt to set - */ - public void setTourOrigToStopDistanceAlt(double[] tourOrigToStopDistanceAlt) - { - this.tourOrigToStopDistanceAlt = tourOrigToStopDistanceAlt; - } - - /** - * @return the stopToTourDestDistanceAlt - */ - public double getStopToTourDestDistanceAlt(int alt) - { - return stopToTourDestDistanceAlt[alt]; - } - - /** - * @param stopToTourDestDistanceAlt - * the stopToTourDestDistanceAlt to set - */ - public void setStopToTourDestDistanceAlt(double[] stopToTourDestDistanceAlt) - { - this.stopToTourDestDistanceAlt = stopToTourDestDistanceAlt; - } - - /** - * @return the sizeTerms. The size term is the size of the alternative north - * of the border. It is indexed by alternative, where alternative is - * either taz-station pair or mgra-station pair, depending on - * whether the DMU is being used for the SOA model or the actual - * model. - */ - public double getSizeTerm(int alt) - { - return sizeTerms[purpose][alt]; - } - - /** - * @param sizeTerms - * the sizeTerms to set. The size term is the size of the - * alternative north of the border. It is indexed by alternative, - * where alternative is either taz-station pair or mgra-station - * pair, depending on whether the DMU is being used for the SOA - * model or the actual model. - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - /** - * @return the correctionFactors - */ - public double getCorrectionFactor(int alt) - { - return correctionFactors[alt]; - } - - /** - * @param correctionFactors - * the correctionFactors to set - */ - public void setCorrectionFactors(double[] correctionFactors) - { - this.correctionFactors = correctionFactors; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the purpose - */ - public int getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(int purpose) - { - this.purpose = purpose; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - methodIndexMap.put("getPurpose", 0); - methodIndexMap.put("getStopsOnHalfTour", 1); - methodIndexMap.put("getStopNumber", 2); - methodIndexMap.put("getInboundStop", 3); - methodIndexMap.put("getTourDuration", 4); - - methodIndexMap.put("getSizeTerm", 5); - methodIndexMap.put("getCorrectionFactor", 6); - methodIndexMap.put("getSampleNumber", 7); - methodIndexMap.put("getOsMcLogsumAlt", 8); - methodIndexMap.put("getSdMcLogsumAlt", 9); - methodIndexMap.put("getTourOrigToStopDistanceAlt", 10); - methodIndexMap.put("getStopToTourDestDistanceAlt", 11); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getPurpose(); - break; - case 1: - returnValue = getStopsOnHalfTour(); - break; - case 2: - returnValue = getStopNumber(); - break; - case 3: - returnValue = getInboundStop(); - break; - case 4: - returnValue = getTourDuration(); - break; - case 5: - returnValue = getSizeTerm(arrayIndex); - break; - case 6: - returnValue = getCorrectionFactor(arrayIndex); - break; - case 7: - returnValue = getSampleNumber(arrayIndex); - break; - case 8: - returnValue = getOsMcLogsumAlt(arrayIndex); - break; - case 9: - returnValue = getSdMcLogsumAlt(arrayIndex); - break; - case 10: - returnValue = getTourOrigToStopDistanceAlt(arrayIndex); - break; - case 11: - returnValue = getStopToTourDestDistanceAlt(arrayIndex); - break; - - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java deleted file mode 100644 index 4a6bde0..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopLocationChoiceModel.java +++ /dev/null @@ -1,538 +0,0 @@ -package org.sandag.abm.visitor; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.ConcreteAlternative; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class VisitorStopLocationChoiceModel -{ - - private transient Logger logger = Logger.getLogger("visitorModel"); - - private McLogsumsCalculator logsumHelper; - private VisitorModelStructure modelStructure; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private VisitorStopLocationChoiceDMU dmu; - private VisitorTripModeChoiceModel tripModeChoiceModel; - double logsum = 0; - private ChoiceModelApplication soaModel; - private ChoiceModelApplication destModel; - - // the following arrays are calculated in the station-destination choice - // model and passed in the constructor. - private double[][] mgraSizeTerms; // by - // purpose, - // MGRA - private double[][][] mgraProbabilities; // by - // purpose, - // TAZ, - // MGRA - - private TableDataSet alternativeData; // the - // alternatives, - // with - // a - // "dest" - // - - // indicating - // the - // destination - // TAZ - // in - // San - // Diego - // County - - // following are used for each taz alternative - private double[] soaTourOrigToStopDistanceAlt; // by - // TAZ - private double[] soaStopToTourDestDistanceAlt; // by - // TAZ - private double[][] tazSizeTerms; // by - // purpose, - // TAZ - // - - // set - // by - // constructor - - // following are used for sampled mgras - private int sampleRate; - private double[][] sampledSizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgra) - private double[] correctionFactors; // by - // alternative - // (sampled - // mgra, - // for - // full - // model - // only) - private int[] sampledTazs; // by - // alternative - // (sampled - // taz) - private int[] sampledMgras; // by - // alternative(sampled - // mgra) - private double[] tourOrigToStopDistanceAlt; - private double[] stopToTourDestDistanceAlt; - private double[] osMcLogsumAlt; - private double[] sdMcLogsumAlt; - - HashMap frequencyChosen; - - private VisitorTrip trip; - - private int originMgra; // the - // origin - // MGRA - // of - // the - // stop - // (originMgra - // -> - // stopMgra - // -> - // destinationMgra) - private int destinationMgra; // the - // destination - // MGRA - // of - // the - // stop - // (originMgra - // -> - // stopMgra - // -> - // destinationMgra) - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public VisitorStopLocationChoiceModel(HashMap propertyMap, - VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - mgraManager = MgraDataManager.getInstance(propertyMap); - tazManager = TazDataManager.getInstance(propertyMap); - - modelStructure = myModelStructure; - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - // this sets by thread, so do it outside of initialization - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - setupStopLocationChoiceModel(propertyMap, dmuFactory); - - frequencyChosen = new HashMap(); - - trip = new VisitorTrip(); - - } - - /** - * Read the UEC file and set up the stop destination choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupStopLocationChoiceModel(HashMap rbMap, - VisitorDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up visitor stop location choice model.")); - - dmu = dmuFactory.getVisitorStopLocationChoiceDMU(); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String visitorStopLocationSoaFileName = Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.soa.uec.file"); - visitorStopLocationSoaFileName = uecFileDirectory + visitorStopLocationSoaFileName; - - int soaDataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.soa.data.page")); - int soaModelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.soa.model.page")); - - String visitorStopLocationFileName = Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.uec.file"); - visitorStopLocationFileName = uecFileDirectory + visitorStopLocationFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.data.page")); - int modelPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.slc.model.page")); - - // create a ChoiceModelApplication object for the SOA model. - soaModel = new ChoiceModelApplication(visitorStopLocationSoaFileName, soaModelPage, - soaDataPage, rbMap, (VariableTable) dmu); - - // create a ChoiceModelApplication object for the full model. - destModel = new ChoiceModelApplication(visitorStopLocationFileName, modelPage, dataPage, - rbMap, (VariableTable) dmu); - sampleRate = destModel.getAlternativeNames().length; - - // get the alternative data - UtilityExpressionCalculator uec = soaModel.getUEC(); - alternativeData = uec.getAlternativeData(); - int purposes = modelStructure.VISITOR_PURPOSES.length; - - sampledSizeTerms = new double[purposes][sampleRate + 1]; // by purpose, - // alternative - // (taz or - // sampled - // mgra) - correctionFactors = new double[sampleRate + 1]; // by alternative - // (sampled mgra, for - // full model only) - sampledTazs = new int[sampleRate + 1]; // by alternative (sampled taz) - sampledMgras = new int[sampleRate + 1]; // by alternative (sampled mgra) - tourOrigToStopDistanceAlt = new double[sampleRate + 1]; - stopToTourDestDistanceAlt = new double[sampleRate + 1]; - osMcLogsumAlt = new double[sampleRate + 1]; - sdMcLogsumAlt = new double[sampleRate + 1]; - - } - - /** - * Create a sample for the tour and stop. - * - * @param tour - * @param stop - */ - private void createSample(VisitorTour tour, VisitorStop stop) - { - - int purpose = tour.getPurpose(); - int origTaz = 0; - int destTaz = 0; - int period = modelStructure.AM; - - dmu.setPurpose(purpose); - boolean inbound = stop.isInbound(); - if (inbound) - { - dmu.setInboundStop(1); - dmu.setStopsOnHalfTour(tour.getNumberInboundStops()); - - // destination for inbound stops is always tour origin - destinationMgra = tour.getOriginMGRA(); - destTaz = mgraManager.getTaz(destinationMgra); - - // origin for inbound stops is tour destination if first stop, or - // last chosen stop location - if (stop.getId() == 0) - { - originMgra = tour.getDestinationMGRA(); - origTaz = mgraManager.getTaz(originMgra); - } else - { - VisitorStop[] stops = tour.getInboundStops(); - originMgra = stops[stop.getId() - 1].getMgra(); - origTaz = mgraManager.getTaz(originMgra); - } - - } else - { - dmu.setInboundStop(0); - dmu.setStopsOnHalfTour(tour.getNumberOutboundStops()); - - // destination for outbound stops is always tour destination - destinationMgra = tour.getDestinationMGRA(); - destTaz = mgraManager.getTaz(destinationMgra); - - // origin for outbound stops is tour origin if first stop, or last - // chosen stop location - if (stop.getId() == 0) - { - originMgra = tour.getOriginMGRA(); - origTaz = mgraManager.getTaz(originMgra); - } else - { - VisitorStop[] stops = tour.getOutboundStops(); - originMgra = stops[stop.getId() - 1].getMgra(); - origTaz = mgraManager.getTaz(originMgra); - } - } - dmu.setStopNumber(stop.getId() + 1); - dmu.setDmuIndexValues(origTaz, origTaz, origTaz, 0, false); - - // distances - soaTourOrigToStopDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceFromTaz( - origTaz, period); - soaStopToTourDestDistanceAlt = logsumHelper.getAnmSkimCalculator().getTazDistanceToTaz( - destTaz, period); - dmu.setTourOrigToStopDistanceAlt(soaTourOrigToStopDistanceAlt); - dmu.setStopToTourDestDistanceAlt(soaStopToTourDestDistanceAlt); - - dmu.setSizeTerms(tazSizeTerms); - - // solve for each sample - frequencyChosen.clear(); - for (int sample = 1; sample <= sampleRate; ++sample) - { - - // solve the UEC - soaModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - // choose a TAZ - double random = tour.getRandom(); - ConcreteAlternative[] alts = soaModel.getAlternatives(); - double cumProb = 0; - double altProb = 0; - int sampledTaz = -1; - for (int i = 0; i < alts.length; ++i) - { - cumProb += alts[i].getProbability(); - if (random < cumProb) - { - sampledTaz = (int) alternativeData.getValueAt(i + 1, "dest"); - altProb = alts[i].getProbability(); - break; - } - } - - // set the sampled taz in the array - sampledTazs[sample] = sampledTaz; - - // now find an MGRA in the taz corresponding to the random number - // drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives - // probability - int[] mgraArray = tazManager.getMgraArray(sampledTaz); - int mgraNumber = 0; - double[] mgraCumProb = mgraProbabilities[purpose][sampledTaz]; - - if (mgraCumProb == null) - { - logger.error("Error: mgraCumProb array is null for purpose " + purpose - + " sampledTaz " + sampledTaz + " hhID " + tour.getID()); - throw new RuntimeException(); - } - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > random && mgraCumProb[i] > 0) - { - mgraNumber = mgraArray[i]; - sampledMgras[sample] = mgraNumber; - - // for now, store the probability in the correction factors - // array - correctionFactors[sample] = mgraCumProb[i] * altProb; - - break; - } - } - - // store frequency chosen - if (!frequencyChosen.containsKey(mgraNumber)) - { - frequencyChosen.put(mgraNumber, 1); - } else - { - int freq = frequencyChosen.get(mgraNumber); - frequencyChosen.put(mgraNumber, freq + 1); - } - - // set the size terms for the sample - sampledSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; - - // set the distances for the sample - tourOrigToStopDistanceAlt[sample] = soaTourOrigToStopDistanceAlt[sampledTaz]; - stopToTourDestDistanceAlt[sample] = soaStopToTourDestDistanceAlt[sampledTaz]; - - } - // calculate correction factors - for (int sample = 1; sample <= sampleRate; ++sample) - { - int mgra = sampledMgras[sample]; - int freq = frequencyChosen.get(mgra); - correctionFactors[sample] = (float) Math.log((double) freq / correctionFactors[sample]); - - } - - } - - /** - * Choose a stop location from the sample. - * - * @param tour - * The visitor tour. - * @param stop - * The visitor stop. - */ - public void chooseStopLocation(VisitorTour tour, VisitorStop stop) - { - - // create a sample of mgras and set all of the dmu properties - createSample(tour, stop); - dmu.setCorrectionFactors(correctionFactors); - dmu.setSizeTerms(sampledSizeTerms); - dmu.setTourOrigToStopDistanceAlt(stopToTourDestDistanceAlt); - dmu.setStopToTourDestDistanceAlt(stopToTourDestDistanceAlt); - dmu.setSampleNumber(sampledMgras); - - // calculate trip mode choice logsums to and from stop - for (int i = 1; i <= sampleRate; ++i) - { - - // to stop (originMgra -> stopMgra ) - trip.initializeFromStop(tour, stop, true); - trip.setOriginMgra(trip.getOriginMgra()); - trip.setDestinationMgra(sampledMgras[i]); - double logsum = tripModeChoiceModel.computeUtilities(tour, trip); - osMcLogsumAlt[i] = logsum; - - // from stop (stopMgra -> destinationMgra) - trip.initializeFromStop(tour, stop, false); - trip.setOriginMgra(sampledMgras[i]); - trip.setDestinationMgra(trip.getDestinationMgra()); - logsum = tripModeChoiceModel.computeUtilities(tour, trip); - sdMcLogsumAlt[i] = logsum; - - } - dmu.setOsMcLogsumAlt(osMcLogsumAlt); - dmu.setSdMcLogsumAlt(sdMcLogsumAlt); - - // log headers to traceLogger - if (tour.getDebugChoiceModels()) - { - String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() - + " purpose " + modelStructure.VISITOR_PURPOSES[stop.getPurpose()]; - destModel.choiceModelUtilityTraceLoggerHeading( - "Intermediate stop location choice model", decisionMakerLabel); - } - - destModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - double random = tour.getRandom(); - int alt = destModel.getChoiceResult(random); - int destMgra = sampledMgras[alt]; - stop.setMgra(destMgra); - - // write UEC calculation results and choice - if (tour.getDebugChoiceModels()) - { - String decisionMakerLabel = "Tour ID " + tour.getID() + " stop id " + stop.getId() - + " purpose " + modelStructure.VISITOR_PURPOSES[stop.getPurpose()]; - String loggingHeader = String.format("%s %s", - "Intermediate stop location choice model", decisionMakerLabel); - destModel.logUECResults(logger, loggingHeader); - logger.info("Chose alternative " + alt + " mgra " + destMgra + " with random number " - + random); - logger.info(""); - logger.info(""); - } - - } - - /** - * @return the mgraSizeTerms - */ - public double[][] getMgraSizeTerms() - { - return mgraSizeTerms; - } - - /** - * @return the mgraProbabilities - */ - public double[][][] getMgraProbabilities() - { - return mgraProbabilities; - } - - /** - * @return the tazSizeTerms - */ - public double[][] getTazSizeTerms() - { - return tazSizeTerms; - } - - /** - * Set mgra size terms: must call before choosing location. - * - * @param mgraSizeTerms - */ - public void setMgraSizeTerms(double[][] mgraSizeTerms) - { - - if (mgraSizeTerms == null) - { - logger.error("Error attempting to set MGRASizeTerms in VisitorStopLocationChoiceModel: MGRASizeTerms are null"); - throw new RuntimeException(); - } - this.mgraSizeTerms = mgraSizeTerms; - } - - /** - * Set taz size terms: must call before choosing location. - * - * @param tazSizeTerms - */ - public void setTazSizeTerms(double[][] tazSizeTerms) - { - if (tazSizeTerms == null) - { - logger.error("Error attempting to set TazSizeTerms in VisitorStopLocationChoiceModel: TazSizeTerms are null"); - throw new RuntimeException(); - } - this.tazSizeTerms = tazSizeTerms; - } - - /** - * Set the mgra probabilities. Must call before choosing location. - * - * @param mgraProbabilities - */ - public void setMgraProbabilities(double[][][] mgraProbabilities) - { - if (mgraProbabilities == null) - { - logger.error("Error attempting to set mgraProbabilities in VisitorStopLocationChoiceModel: mgraProbabilities are null"); - throw new RuntimeException(); - } - this.mgraProbabilities = mgraProbabilities; - } - - /** - * Set trip mode choice model. Must call before choosing location. - * - * @param tripModeChoiceModel - */ - public void setTripModeChoiceModel(VisitorTripModeChoiceModel tripModeChoiceModel) - { - this.tripModeChoiceModel = tripModeChoiceModel; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java deleted file mode 100644 index 2693a44..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopPurposeModel.java +++ /dev/null @@ -1,233 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the stop purpose choice model for visitor tours. It is - * currently based on a static probability distribution stored in an input file, - * and indexed into by purpose, tour leg direction (inbound or outbound), the - * stop number, and whether there is just one or multiple stops on the tour leg. - * - * @author Freedman - * - */ -public class VisitorStopPurposeModel -{ - private transient Logger logger = Logger.getLogger("visitorModel"); - - private double[][] cumProbability; // by - // alternative, - // stop - // purpose: - // cumulative - // probability - // distribution - VisitorModelStructure modelStructure; - - HashMap arrayElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - // tour - // purpose, - // tour - // leg - // direction, - // stop - // number, - // and - // stop - // complexity. - - /** - * Constructor. - */ - public VisitorStopPurposeModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stopFrequencyFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.stop.purpose.file"); - stopFrequencyFile = directory + stopFrequencyFile; - - modelStructure = new VisitorModelStructure(); - - arrayElementMap = new HashMap(); - readStopPurposeFile(stopFrequencyFile); - - } - - /** - * Read the stop frequency distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readStopPurposeFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating stop purpose probability distribution"); - - // take a pass through the data and see how many alternatives there are - // for each purpose - int rowCount = probabilityTable.getRowCount(); - int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 - - cumProbability = new double[rowCount][purposes]; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "TourPurp"); - - int inbound = (int) probabilityTable.getValueAt(row, "Inbound"); - int stopNumber = (int) probabilityTable.getValueAt(row, "StopNum"); - int multiple = (int) probabilityTable.getValueAt(row, "Multiple"); - - // store cumulative probabilities - float cumProb = 0; - for (int p = 0; p < purposes; ++p) - { - String label = "StopPurp" + p; - cumProb += probabilityTable.getValueAt(row, label); - cumProbability[row - 1][p] += cumProb; - } - - if (Math.abs(cumProb - 1.0) > 0.00001) - logger.info("Cumulative probability for tour purpose " + purpose + " inbound " - + inbound + " stopNumber " + stopNumber + " multiple " + multiple + " is " - + cumProb); - - int key = getKey(purpose, inbound, stopNumber, multiple); - arrayElementMap.put(key, row - 1); - - } - - logger.info("End calculating stop purpose probability distribution"); - - } - - /** - * Get the key for the arrayElementMap. - * - * @param tourPurp - * Tour purpose - * @param isInbound - * 1 if the stop is on the inbound direction, else 0. - * @param stopNumber - * The number of the stop. - * @param multipleStopsOnLeg - * 1 if multiple stops on leg, else 0. - * @return arrayElementMap key. - */ - private int getKey(int tourPurp, int isInbound, int stopNumber, int multipleStopsOnLeg) - { - - return tourPurp * 1000 + isInbound * 100 + stopNumber * 10 + multipleStopsOnLeg; - } - - /** - * Calculate purposes all stops on the tour - * - * @param tour - * A cross border tour (with tour purpose) - */ - public void calculateStopPurposes(VisitorTour tour) - { - - // outbound stops first - if (tour.getNumberOutboundStops() != 0) - { - - int tourPurp = tour.getPurpose(); - VisitorStop[] stops = tour.getOutboundStops(); - int multiple = 0; - if (stops.length > 1) multiple = 1; - - // iterate through stop list and calculate purpose for each - for (int i = 0; i < stops.length; ++i) - { - int key = getKey(tourPurp, 0, i + 1, multiple); - int element = arrayElementMap.get(key); - double[] cumProb = cumProbability[element]; - double rand = tour.getRandom(); - int purpose = chooseFromDistribution(rand, cumProb); - stops[i].setPurpose((byte) purpose); - } - } - // inbound stops last - if (tour.getNumberInboundStops() != 0) - { - - int tourPurp = tour.getPurpose(); - VisitorStop[] stops = tour.getInboundStops(); - int multiple = 0; - if (stops.length > 1) multiple = 1; - - // iterate through stop list and calculate purpose for each - for (int i = 0; i < stops.length; ++i) - { - int key = getKey(tourPurp, 1, i + 1, multiple); - int element = arrayElementMap.get(key); - double[] cumProb = cumProbability[element]; - double rand = tour.getRandom(); - int purpose = chooseFromDistribution(rand, cumProb); - stops[i].setPurpose((byte) purpose); - } - } - } - - /** - * Choose purpose from the cumulative probability distribution - * - * @param random - * Uniformly distributed random number - * @param cumProb - * Cumulative probability distribution - * @return Stop purpose (0 init). - */ - private int chooseFromDistribution(double random, double[] cumProb) - { - - int choice = -1; - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - choice = i; - break; - } - - } - return choice; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java deleted file mode 100644 index e5a2bd7..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorStopTimeOfDayChoiceModel.java +++ /dev/null @@ -1,365 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the TOD choice model for visitor tours. It is currently based - * on a static probability distribution stored in an input file, and indexed - * into by purpose. - * - * @author Freedman - * - */ -public class VisitorStopTimeOfDayChoiceModel -{ - private transient Logger logger = Logger.getLogger("visitorModel"); - - private double[][] outboundCumProbability; // by - // alternative: - // outbound - // cumulative - // probability - // distribution - private int[] outboundOffsets; // by - // alternative: - // offsets - // for - // outbound - // stop - // duration - // choice - - private double[][] inboundCumProbability; // by - // alternative: - // inbound - // cumulative - // probability - // distribution - private int[] inboundOffsets; // by - // alternative: - // offsets - // for - // inbound - // stop - // duration - // choice - private VisitorModelStructure modelStructure; - - private HashMap outboundElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - // tour duration and stop number. - - private HashMap inboundElementMap; // Hashmap - // used - // to - // get - // the - // element - // number - // of - // the - // cumProbability - // array - // based - // on - // the - - // tour duration and stop number. - - /** - * Constructor. - */ - public VisitorStopTimeOfDayChoiceModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String outboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.stop.outbound.duration.file"); - String inboundDurationFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.stop.inbound.duration.file"); - - outboundDurationFile = directory + outboundDurationFile; - inboundDurationFile = directory + inboundDurationFile; - - modelStructure = new VisitorModelStructure(); - - outboundElementMap = new HashMap(); - readOutboundFile(outboundDurationFile); - - inboundElementMap = new HashMap(); - readInboundFile(inboundDurationFile); - } - - /** - * Read the outbound stop duration file and store the cumulative probability - * distribution as well as the offsets and set the key map to index into the - * probability array. - * - * @param fileName - */ - public void readOutboundFile(String fileName) - { - TableDataSet outboundTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - outboundTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - int columns = outboundTable.getColumnCount(); - int rows = outboundTable.getRowCount(); - outboundCumProbability = new double[rows][columns - 3]; - - // first three columns are index fields, rest are offsets - outboundOffsets = new int[columns - 3]; - for (int i = 4; i <= columns; ++i) - { - String offset = outboundTable.getColumnLabel(i); - outboundOffsets[i - 4] = new Integer(offset); - } - - // now fill in cumulative probability array - for (int row = 1; row <= rows; ++row) - { - - int lowerBound = (int) outboundTable.getValueAt(row, "RemainingLow"); - int upperBound = (int) outboundTable.getValueAt(row, "RemainingHigh"); - int stopNumber = (int) outboundTable.getValueAt(row, "Stop"); - - for (int duration = lowerBound; duration <= upperBound; ++duration) - { - int key = getKey(stopNumber, duration); - outboundElementMap.put(key, row - 1); - } - - // cumulative probability distribution - double cumProb = 0; - for (int col = 4; col <= columns; ++col) - { - cumProb += outboundTable.getValueAt(row, col); - outboundCumProbability[row - 1][col - 4] = cumProb; - } - - } - - } - - /** - * Read the inbound stop duration file and store the cumulative probability - * distribution as well as the offsets and set the key map to index into the - * probability array. - * - * @param fileName - */ - public void readInboundFile(String fileName) - { - TableDataSet inboundTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - inboundTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - int columns = inboundTable.getColumnCount(); - int rows = inboundTable.getRowCount(); - inboundCumProbability = new double[rows][columns - 3]; - - // first three columns are index fields, rest are offsets - inboundOffsets = new int[columns - 3]; - for (int i = 4; i <= columns; ++i) - { - String offset = inboundTable.getColumnLabel(i); - inboundOffsets[i - 4] = new Integer(offset); - } - - // now fill in cumulative probability array - for (int row = 1; row <= rows; ++row) - { - - int lowerBound = (int) inboundTable.getValueAt(row, "RemainingLow"); - int upperBound = (int) inboundTable.getValueAt(row, "RemainingHigh"); - int stopNumber = (int) inboundTable.getValueAt(row, "Stop"); - - for (int duration = lowerBound; duration <= upperBound; ++duration) - { - int key = getKey(stopNumber, duration); - inboundElementMap.put(key, row - 1); - } - // cumulative probability distribution - double cumProb = 0; - for (int col = 4; col <= columns; ++col) - { - cumProb += inboundTable.getValueAt(row, col); - inboundCumProbability[row - 1][col - 4] = cumProb; - } - - } - - } - - /** - * Get the key for the arrayElementMap. - * - * @param stopNumber - * stop number - * @param periodsRemaining - * Remaining time periods - * @return arrayElementMap key. - */ - private int getKey(int stopNumber, int periodsRemaining) - { - - return periodsRemaining * 10 + stopNumber; - } - - /** - * Choose the stop time of day period. - * - * @param tour - * @param stop - */ - public void chooseTOD(VisitorTour tour, VisitorStop stop) - { - - boolean inbound = stop.isInbound(); - int stopNumber = stop.getId() + 1; - int arrivalPeriod = tour.getArriveTime(); - - if (!inbound) - { - - // find the departure time - int departPeriod = 0; - if (stop.getId() == 0) departPeriod = tour.getDepartTime(); - else - { - VisitorStop[] stops = tour.getOutboundStops(); - departPeriod = stops[stop.getId() - 1].getStopPeriod(); - } - - int periodsRemaining = arrivalPeriod - departPeriod; - - int key = getKey(stopNumber, periodsRemaining); - int element = outboundElementMap.get(key); - double[] cumProb = outboundCumProbability[element]; - double random = tour.getRandom(); - - // iterate through the offset distribution, choose an offset, and - // set in the stop - if (tour.getDebugChoiceModels()) - { - logger.info("Stop TOD Choice Model for tour " + tour.getID() + " outbound stop " - + stop.getId() + " periods remaining " + periodsRemaining); - logger.info(" random number " + random); - } - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - int offset = outboundOffsets[i]; - int period = departPeriod + offset; - stop.setPeriod(period); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Chose alt " + i + " offset " + offset + " from depart period " - + departPeriod); - logger.info("Stop period is " + stop.getStopPeriod()); - - } - break; - - } - } - } else - { - // inbound stop - - // find the departure time - int departPeriod = 0; - - // first inbound stop - if (stop.getId() == 0) - { - - // there were outbound stops - if (tour.getOutboundStops() != null) - { - VisitorStop[] outboundStops = tour.getOutboundStops(); - departPeriod = outboundStops[outboundStops.length - 1].getStopPeriod(); - } else - { - // no outbound stops - departPeriod = tour.getDepartTime(); - } - } else - { - // not first inbound stop - VisitorStop[] stops = tour.getInboundStops(); - departPeriod = stops[stop.getId() - 1].getStopPeriod(); - } - - int periodsRemaining = arrivalPeriod - departPeriod; - - int key = getKey(stopNumber, periodsRemaining); - int element = inboundElementMap.get(key); - double[] cumProb = inboundCumProbability[element]; - double random = tour.getRandom(); - if (tour.getDebugChoiceModels()) - { - logger.info("Stop TOD Choice Model for tour " + tour.getID() + " inbound stop " - + stop.getId() + " periods remaining " + periodsRemaining); - logger.info("Random number " + random); - } - for (int i = 0; i < cumProb.length; ++i) - { - if (random < cumProb[i]) - { - int offset = inboundOffsets[i]; - int arrivePeriod = tour.getArriveTime(); - int period = arrivePeriod + offset; - stop.setPeriod(period); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Chose alt " + i + " offset " + offset + " from arrive period " - + arrivePeriod); - logger.info("Stop period is " + stop.getStopPeriod()); - - } - break; - } - } - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java deleted file mode 100644 index 1de6057..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTour.java +++ /dev/null @@ -1,350 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Household; -import com.pb.common.math.MersenneTwister; - -public class VisitorTour - implements Serializable -{ - - private MersenneTwister random; - private int ID; - - // following variables determined via simulation - private byte segment; // 0 - // = - // business, - // 1 - // = - // personal - private byte purpose; - private byte numberOfParticipants; - private int income; - private int autoAvailable; - - private VisitorStop[] outboundStops; - private VisitorStop[] inboundStops; - - private VisitorTrip[] trips; - - private int departTime; - private int arriveTime; - - private boolean debugChoiceModels; - - // following variables chosen via choice models - private int originMGRA; - private int destinationMGRA; - private byte tourMode; - - private float valueOfTime; - /** - * Public constructor. - * - * @param seed - * A seed for the random number generator. - */ - public VisitorTour(long seed) - { - - random = new MersenneTwister(seed); - } - - /** - * @return the iD - */ - public int getID() - { - return ID; - } - - /** - * @param iD - * the iD to set - */ - public void setID(int iD) - { - ID = iD; - } - - /** - * @return the purpose - */ - public byte getPurpose() - { - return purpose; - } - - /** - * @return the outboundStops - */ - public VisitorStop[] getOutboundStops() - { - return outboundStops; - } - - /** - * @param outboundStops - * the outboundStops to set - */ - public void setOutboundStops(VisitorStop[] outboundStops) - { - this.outboundStops = outboundStops; - } - - /** - * @return the inboundStops - */ - public VisitorStop[] getInboundStops() - { - return inboundStops; - } - - /** - * @param inboundStops - * the inboundStops to set - */ - public void setInboundStops(VisitorStop[] inboundStops) - { - this.inboundStops = inboundStops; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(byte purpose) - { - this.purpose = purpose; - } - - /** - * @return the departTime - */ - public int getDepartTime() - { - return departTime; - } - - /** - * @param departTime - * the departTime to set - */ - public void setDepartTime(int departTime) - { - this.departTime = departTime; - } - - public VisitorTrip[] getTrips() - { - return trips; - } - - public void setTrips(VisitorTrip[] trips) - { - this.trips = trips; - } - - /** - * @return the originMGRA - */ - public int getOriginMGRA() - { - return originMGRA; - } - - /** - * @param originMGRA - * the originMGRA to set - */ - public void setOriginMGRA(int originMGRA) - { - this.originMGRA = originMGRA; - } - - /** - * @return the tour mode - */ - public byte getTourMode() - { - return tourMode; - } - - /** - * @param mode - * the tour mode to set - */ - public void setTourMode(byte mode) - { - this.tourMode = mode; - } - - /** - * Get a random number from the parties random class. - * - * @return A random number. - */ - public double getRandom() - { - return random.nextDouble(); - } - - /** - * @return the debugChoiceModels - */ - public boolean getDebugChoiceModels() - { - return debugChoiceModels; - } - - /** - * @param debugChoiceModels - * the debugChoiceModels to set - */ - public void setDebugChoiceModels(boolean debugChoiceModels) - { - this.debugChoiceModels = debugChoiceModels; - } - - /** - * Get the number of outbound stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberOutboundStops() - { - if (outboundStops == null) return 0; - else return outboundStops.length; - - } - - /** - * Get the number of return stops - * - * @return 0 if not initialized, else number of stops - */ - public int getNumberInboundStops() - { - if (inboundStops == null) return 0; - else return inboundStops.length; - - } - - /** - * @return the destinationMGRA - */ - public int getDestinationMGRA() - { - return destinationMGRA; - } - - /** - * @param destinationMGRA - * the destinationMGRA to set - */ - public void setDestinationMGRA(int destinationMGRA) - { - this.destinationMGRA = destinationMGRA; - } - - public void setArriveTime(int arriveTime) - { - this.arriveTime = arriveTime; - } - - public int getArriveTime() - { - return arriveTime; - } - - /** - * @return the numberOfParticipants - */ - public byte getNumberOfParticipants() - { - return numberOfParticipants; - } - - /** - * @param numberOfParticipants - * the numberOfParticipants to set - */ - public void setNumberOfParticipants(byte numberOfParticipants) - { - this.numberOfParticipants = numberOfParticipants; - } - - /** - * @return the income - */ - public int getIncome() - { - return income; - } - - /** - * @param income - * the income to set - */ - public void setIncome(int income) - { - this.income = income; - } - - /** - * @return the autoAvailable - */ - public int getAutoAvailable() - { - return autoAvailable; - } - - /** - * @param autoAvailable - * the autoAvailable to set - */ - public void setAutoAvailable(int autoAvailable) - { - this.autoAvailable = autoAvailable; - } - - /** - * @return the segment - */ - public byte getSegment() - { - return segment; - } - - /** - * @param segment - * the segment to set - */ - public void setSegment(byte segment) - { - this.segment = segment; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - - public void logTourObject(Logger logger, int totalChars) - { - - Household.logHelper(logger, "tourId: ", ID, totalChars); - Household.logHelper(logger, "tourPurpose: ", purpose, totalChars); - Household.logHelper(logger, "tourOrigMgra: ", originMGRA, totalChars); - Household.logHelper(logger, "tourDestMgra: ", destinationMGRA, totalChars); - Household.logHelper(logger, "tourDepartPeriod: ", departTime, totalChars); - Household.logHelper(logger, "tourArrivePeriod: ", arriveTime, totalChars); - Household.logHelper(logger, "tourMode: ", tourMode, totalChars); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java deleted file mode 100644 index 405ab33..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceDMU.java +++ /dev/null @@ -1,317 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; -import org.apache.log4j.Logger; -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class VisitorTourDestChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger("visitorModel"); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected float tourDepartPeriod; - protected float tourArrivePeriod; - protected int purpose; - protected double[][] sizeTerms; // by - // purpose, - // alternative - // (taz - // or - // sampled - // mgras) - protected double[] correctionFactors; // by - // alternative - // (sampled - // mgra, - // for - // full - // model - // only) - protected double[] tourModeLogsums; // by - // alternative - // (sampled - // mgra - // pair, - // for - // full - // model - // only) - protected int[] sampleMGRA; // by - // alternative - // (sampled - // mgra) - protected int[] sampleTAZ; // by - // alternative - // (sampled - // taz) - - protected double nmWalkTimeOut; - protected double nmWalkTimeIn; - protected double nmBikeTimeOut; - protected double nmBikeTimeIn; - protected double lsWgtAvgCostM; - protected double lsWgtAvgCostD; - protected double lsWgtAvgCostH; - - public VisitorTourDestChoiceDMU(VisitorModelStructure modelStructure) - { - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Get the tour mode choice logsum for the sampled station-mgra pair. - * - * @param alt - * Sampled station-mgra - * @return - */ - public double getTourModeLogsum(int alt) - { - return tourModeLogsums[alt]; - } - - /** - * Set the tour mode choice logsums - * - * @param poeNumbers - * An array of tour mode choice logsums, one for each alternative - * (sampled station-mgra) - */ - public void setTourModeLogsums(double[] logsums) - { - this.tourModeLogsums = logsums; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - /** - * @return the sizeTerms. The size term is the size of the alternative north - * of the border. It is indexed by alternative, where alternative is - * either taz-station pair or mgra-station pair, depending on - * whether the DMU is being used for the SOA model or the actual - * model. - */ - public double getSizeTerm(int alt) - { - return sizeTerms[purpose][alt]; - } - - /** - * @param sizeTerms - * the sizeTerms to set. The size term is the size of the - * alternative north of the border. It is indexed by alternative, - * where alternative is either taz-station pair or mgra-station - * pair, depending on whether the DMU is being used for the SOA - * model or the actual model. - */ - public void setSizeTerms(double[][] sizeTerms) - { - this.sizeTerms = sizeTerms; - } - - /** - * @return the correctionFactors - */ - public double getCorrectionFactor(int alt) - { - return correctionFactors[alt]; - } - - /** - * @param correctionFactors - * the correctionFactors to set - */ - public void setCorrectionFactors(double[] correctionFactors) - { - this.correctionFactors = correctionFactors; - } - - public int getSampleMgra(int alt) - { - return sampleMGRA[alt]; - } - - public void setSampleMgra(int[] sampleNumber) - { - this.sampleMGRA = sampleNumber; - } - - public int getSampleTaz(int alt) - { - return sampleTAZ[alt]; - } - - public void setSampleTaz(int[] sampleNumber) - { - this.sampleTAZ = sampleNumber; - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the purpose - */ - public int getPurpose() - { - return purpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setPurpose(int purpose) - { - this.purpose = purpose; - } - - public float getTimeOutbound() - { - return tourDepartPeriod; - } - - public float getTimeInbound() - { - return tourArrivePeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(float tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(float tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getSizeTerm", 2); - methodIndexMap.put("getCorrectionFactor", 3); - methodIndexMap.put("getPurpose", 4); - methodIndexMap.put("getTourModeLogsum", 5); - methodIndexMap.put("getSampleMgra", 6); - methodIndexMap.put("getSampleTaz", 7); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 2: - returnValue = getSizeTerm(arrayIndex); - break; - case 3: - returnValue = getCorrectionFactor(arrayIndex); - break; - case 4: - returnValue = getPurpose(); - break; - case 5: - returnValue = getTourModeLogsum(arrayIndex); - break; - case 6: - returnValue = getSampleMgra(arrayIndex); - break; - case 7: - returnValue = getSampleTaz(arrayIndex); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java deleted file mode 100644 index 800b310..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourDestChoiceModel.java +++ /dev/null @@ -1,586 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.File; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -/** - * This class is used for both the sample of alternatives and the full - * destination choice model for visitor tours. - * - * - * @author Freedman - * - */ -public class VisitorTourDestChoiceModel -{ - - private double[][] mgraSizeTerms; // by - // purpose, - // MGRA - private double[][] tazSizeTerms; // by - // purpose, - // TAZ - private double[][][] mgraProbabilities; // by - // purpose, - // tazNumber, - // mgra - // index - // (sequential, - // 0-based) - private Matrix[] tazProbabilities; // by - // purpose, - // origin - // TAZ, - // destination - // TAZ - private TableDataSet alternativeData; // the - // alternatives, - // with - // a - // dest - // field - // indicating - // tazNumber - private int[] sampleMgras; // numbers - // of - // mgra - // for - // the - // sample - private int[] sampleTazs; // numbers - // of - // taz - // for - // the - // sample - private double[] sampleCorrectionFactors; // correction - // factors - // for - // sample - private double[][] sampleSizeTerms; // size - // terms - // for - // sample - private double[] sampleLogsums; // tour - // mc - // logsums - - private transient Logger logger = Logger.getLogger("visitorModel"); - - private TazDataManager tazManager; - private MgraDataManager mgraManager; - - private ChoiceModelApplication[] soaModel; - private ChoiceModelApplication[] destModel; - private UtilityExpressionCalculator sizeTermUEC; - private HashMap rbMap; - - private VisitorTourDestChoiceDMU dcDmu; - private VisitorTourModeChoiceModel tourModeChoiceModel; - - private HashMap frequencyChosen; // by - // mgra, - // number - // of - // times - // chosen - private int sampleRate; - - /** - * Constructor - * - * @param propertyMap - * Resource properties file map. - * @param dmuFactory - * Factory object for creation of airport model DMUs - */ - public VisitorTourDestChoiceModel(HashMap rbMap, - VisitorModelStructure modelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - - this.rbMap = rbMap; - - tazManager = TazDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - String uecFileDirectory = Util.getStringValueFromPropertyMap(rbMap, - CtrampApplication.PROPERTIES_UEC_PATH); - String visitorDCSoaFileName = Util.getStringValueFromPropertyMap(rbMap, - "visitor.dc.soa.uec.file"); - visitorDCSoaFileName = uecFileDirectory + visitorDCSoaFileName; - - int dataPage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.dc.soa.data.page")); - int sizePage = Integer.parseInt(Util.getStringValueFromPropertyMap(rbMap, - "visitor.dc.soa.size.page")); - - // initiate a DMU - dcDmu = dmuFactory.getVisitorTourDestChoiceDMU(); - - // read the model pages from the property file, create one choice model - // for each soa model - soaModel = new ChoiceModelApplication[VisitorModelStructure.VISITOR_PURPOSES.length]; - for (int i = 0; i < soaModel.length; ++i) - { - - // get page from property file - String purpose = VisitorModelStructure.VISITOR_PURPOSES[i].toLowerCase(); - String purposeName = "visitor.dc.soa." + purpose + ".page"; - String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); - purposeString.replaceAll(" ", ""); - int destModelPage = Integer.parseInt(purposeString); - - // create a ChoiceModelApplication object for the filename, model - // page and data page. - soaModel[i] = new ChoiceModelApplication(visitorDCSoaFileName, destModelPage, dataPage, - rbMap, (VariableTable) dcDmu); - } - - // get the alternative data from the first segment - UtilityExpressionCalculator uec = soaModel[0].getUEC(); - alternativeData = uec.getAlternativeData(); - - // create a UEC to solve size terms for each MGRA - sizeTermUEC = new UtilityExpressionCalculator(new File(visitorDCSoaFileName), sizePage, - dataPage, rbMap, (VariableTable) dcDmu); - - // create the full model UECs - // read the model pages from the property file, create one choice model - // for each full model - String visitorDCFileName = Util.getStringValueFromPropertyMap(rbMap, "visitor.dc.uec.file"); - visitorDCFileName = uecFileDirectory + visitorDCFileName; - destModel = new ChoiceModelApplication[VisitorModelStructure.VISITOR_PURPOSES.length]; - for (int i = 0; i < destModel.length; ++i) - { - - // get page from property file - String purpose = VisitorModelStructure.VISITOR_PURPOSES[i].toLowerCase(); - String purposeName = "visitor.dc." + purpose + ".page"; - String purposeString = Util.getStringValueFromPropertyMap(rbMap, purposeName); - purposeString.replaceAll(" ", ""); - int destModelPage = Integer.parseInt(purposeString); - - // create a ChoiceModelApplication object for the filename, model - // page and data page. - destModel[i] = new ChoiceModelApplication(visitorDCFileName, destModelPage, dataPage, - rbMap, (VariableTable) dcDmu); - if (i == 0) sampleRate = destModel[i].getNumberOfAlternatives(); - } - - frequencyChosen = new HashMap(); - sampleMgras = new int[sampleRate + 1]; - sampleTazs = new int[sampleRate + 1]; - sampleCorrectionFactors = new double[sampleRate + 1]; - sampleSizeTerms = new double[destModel.length][sampleRate + 1]; - sampleLogsums = new double[sampleRate + 1]; - - tourModeChoiceModel = new VisitorTourModeChoiceModel(rbMap, modelStructure, dmuFactory, tazDistanceCalculator); - - } - - /** - * Calculate size terms - */ - public void calculateSizeTerms(VisitorDmuFactoryIf dmuFactory) - { - - logger.info("Calculating Visitor Tour Destination Choice Model MGRA Size Terms"); - - ArrayList mgras = mgraManager.getMgras(); - int[] mgraTaz = mgraManager.getMgraTaz(); - int maxMgra = mgraManager.getMaxMgra(); - int maxTaz = tazManager.getMaxTaz(); - int purposes = sizeTermUEC.getNumberOfAlternatives(); - - mgraSizeTerms = new double[purposes][maxMgra + 1]; - tazSizeTerms = new double[purposes][maxTaz + 1]; - IndexValues iv = new IndexValues(); - VisitorTourDestChoiceDMU aDmu = dmuFactory.getVisitorTourDestChoiceDMU(); - - // loop through mgras and calculate size terms - for (int mgra : mgras) - { - - int taz = mgraTaz[mgra]; - iv.setZoneIndex(mgra); - double[] utilities = sizeTermUEC.solve(iv, aDmu, null); - - // store the size terms - for (int purpose = 0; purpose < purposes; ++purpose) - { - - mgraSizeTerms[purpose][mgra] = utilities[purpose]; - tazSizeTerms[purpose][taz] += utilities[purpose]; - } - - } - - // now calculate probability of selecting each MGRA within each TAZ for - // SOA - mgraProbabilities = new double[purposes][maxTaz + 1][]; - int[] tazs = tazManager.getTazs(); - - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazs.length; ++taz) - { - int tazNumber = tazs[taz]; - int[] mgraArray = tazManager.getMgraArray(tazNumber); - - // initialize the vector of mgras for this purpose-taz - mgraProbabilities[purpose][tazNumber] = new double[mgraArray.length]; - - // now calculate the cumulative probability distribution - double lastProb = 0.0; - for (int mgra = 0; mgra < mgraArray.length; ++mgra) - { - - int mgraNumber = mgraArray[mgra]; - if (tazSizeTerms[purpose][tazNumber] > 0.0) - mgraProbabilities[purpose][tazNumber][mgra] = lastProb - + mgraSizeTerms[purpose][mgraNumber] - / tazSizeTerms[purpose][tazNumber]; - lastProb = mgraProbabilities[purpose][tazNumber][mgra]; - } - if (tazSizeTerms[purpose][tazNumber] > 0.0 && Math.abs(lastProb - 1.0) > 0.000001) - logger.info("Error: purpose " + purpose + " taz " + tazNumber - + " cum prob adds up to " + lastProb); - } - - } - - // calculate logged size terms for mgra and taz vectors to be used in - // dmu - for (int purpose = 0; purpose < purposes; ++purpose) - { - for (int taz = 0; taz < tazSizeTerms[purpose].length; ++taz) - if (tazSizeTerms[purpose][taz] > 0.0) - tazSizeTerms[purpose][taz] = Math.log(tazSizeTerms[purpose][taz] + 1.0); - - for (int mgra = 0; mgra < mgraSizeTerms[purpose].length; ++mgra) - if (mgraSizeTerms[purpose][mgra] > 0.0) - mgraSizeTerms[purpose][mgra] = Math.log(mgraSizeTerms[purpose][mgra] + 1.0); - - } - logger.info("Finished Calculating Visitor Tour Destination Choice Model MGRA Size Terms"); - } - - /** - * Calculate taz probabilities. This method initializes and calculates the - * tazProbabilities array. - */ - public void calculateTazProbabilities(VisitorDmuFactoryIf dmuFactory) - { - - if (tazSizeTerms == null) - { - logger.error("Error: attemping to execute VisitorTourDestChoiceModel.calculateTazProbabilities() before calling calculateMgraProbabilities()"); - throw new RuntimeException(); - } - - logger.info("Calculating Visitor Model TAZ Probabilities Arrays"); - - // initialize taz probabilities array - int purposes = tazSizeTerms.length; - - // initialize the arrays - tazProbabilities = new Matrix[purposes]; - - // iterate through the alternatives in the alternatives file and set the - // size term and station logsum for each alternative - UtilityExpressionCalculator soaModelUEC = soaModel[0].getUEC(); - TableDataSet altData = soaModelUEC.getAlternativeData(); - - dcDmu.setSizeTerms(tazSizeTerms); - - // iterate through purposes - for (int purpose = 0; purpose < soaModel.length; ++purpose) - { - - tazProbabilities[purpose] = new Matrix("Prob_Matrix", "Probability Matrix", - altData.getRowCount() + 1, altData.getRowCount() + 1); - int[] tazs = altData.getColumnAsInt("dest"); - tazProbabilities[purpose].setExternalNumbersZeroBased(tazs); - - // iterate through origin zones, solve the UEC and store the results - // in the matrix - for (int taz = 0; taz < tazs.length; ++taz) - { - - int originTaz = (int) tazs[taz]; - - // set origin taz in dmu (destination set in UEC by alternative) - dcDmu.setDmuIndexValues(originTaz, originTaz, originTaz, originTaz, false); - - dcDmu.setPurpose(purpose); - - // Calculate utilities & probabilities - soaModel[purpose].computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); - - // Store probabilities (by purpose) - double[] probabilities = soaModel[purpose].getCumulativeProbabilities(); - - for (int i = 0; i < probabilities.length; ++i) - { - - double cumProb = probabilities[i]; - int destTaz = (int) altData.getValueAt(i + 1, "dest"); - tazProbabilities[purpose].setValueAt(originTaz, destTaz, (float) cumProb); - } - } - } - logger.info("Finished Calculating Visitor Model TAZ Probabilities Arrays"); - } - - /** - * Choose a MGRA alternative for sampling - * - * @param tour - * VisitorTour with purpose and Random - */ - private void chooseMgraSample(VisitorTour tour) - { - - frequencyChosen.clear(); - - // choose sample, set station logsums and mgra size terms - int purpose = tour.getPurpose(); - int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); - - for (int sample = 1; sample <= sampleRate; ++sample) - { - - // first find a TAZ and station - int alt = 0; - Matrix tazCumProb = tazProbabilities[purpose]; - double altProb = 0; - double cumProb = 0; - double random = tour.getRandom(); - int destinationTaz = -1; - for (int i = 0; i < tazCumProb.getColumnCount(); ++i) - { - destinationTaz = (int) tazCumProb.getExternalColumnNumber(i); - if (tazCumProb.getValueAt(originTaz, destinationTaz) > random) - { - alt = i; - if (i != 0) - { - cumProb = tazCumProb.getValueAt(originTaz, - tazCumProb.getExternalColumnNumber(i - 1)); - altProb = tazCumProb.getValueAt(originTaz, destinationTaz) - - tazCumProb.getValueAt(originTaz, - tazCumProb.getExternalColumnNumber(i - 1)); - } else - { - altProb = tazCumProb.getValueAt(originTaz, destinationTaz); - } - break; - } - } - - // get the taz number of the alternative, and an array of mgras in - // that taz - - int[] mgraArray = tazManager.getMgraArray(destinationTaz); - - // now find an MGRA in the taz corresponding to the random number - // drawn: - // note that the indexing needs to be offset by the cumulative - // probability of the chosen taz and the - // mgra probabilities need to be scaled by the alternatives - // probability - int mgraNumber = 0; - double[] mgraCumProb = mgraProbabilities[purpose][destinationTaz]; - for (int i = 0; i < mgraCumProb.length; ++i) - { - cumProb += mgraCumProb[i] * altProb; - if (cumProb > random && mgraCumProb[i] > 0) - { - mgraNumber = mgraArray[i]; - sampleMgras[sample] = mgraNumber; - sampleTazs[sample] = mgraManager.getTaz(mgraNumber); - - // for now, store the probability in the correction factors - // array - sampleCorrectionFactors[sample] = mgraCumProb[i] * altProb; - - break; - } - } - // store frequency chosen - if (!frequencyChosen.containsKey(mgraNumber)) - { - frequencyChosen.put(mgraNumber, 1); - } else - { - int freq = frequencyChosen.get(mgraNumber); - frequencyChosen.put(mgraNumber, freq + 1); - } - // set the size terms for the sample - sampleSizeTerms[purpose][sample] = mgraSizeTerms[purpose][mgraNumber]; - } - // calculate correction factors - for (int sample = 1; sample <= sampleRate; ++sample) - { - int mgra = sampleMgras[sample]; - int freq = frequencyChosen.get(mgra); - sampleCorrectionFactors[sample] = (float) Math.log((double) freq - / sampleCorrectionFactors[sample]); - - } - - } - - /** - * Use the tour mode choice model to calculate the logsum for each sampled - * mgra and store in the array. - * - * @param tour - * The visitor tour. - */ - private void calculateLogsumsForSample(VisitorTour tour) - { - - for (int sample = 1; sample <= sampleRate; ++sample) - { - - if (sampleMgras[sample] > 0) - { - - int destinationMgra = sampleMgras[sample]; - tour.setDestinationMGRA(destinationMgra); - - double logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, - "Sample logsum " + sample, "tour " + tour.getID() + " dest " - + destinationMgra); - sampleLogsums[sample] = logsum; - } else sampleLogsums[sample] = 0; - - } - - } - - /** - * Choose a destination MGRA for the tour. - * - * @param tour - * A cross border tour with a tour origin, purpose, attributes, - * and departure\arrival time and SENTRI availability members. - */ - public void chooseDestination(VisitorTour tour) - { - - chooseMgraSample(tour); - calculateLogsumsForSample(tour); - - double random = tour.getRandom(); - int purpose = tour.getPurpose(); - dcDmu.setPurpose(purpose); - - // set origin taz in dmu (destination set in UEC by alternative) - int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); - dcDmu.setDmuIndexValues(0, 0, originTaz, 0, false); - - // set size terms for each sampled station-mgra pair corresponding to - // mgra - dcDmu.setSizeTerms(sampleSizeTerms); - - // set the correction factors - dcDmu.setCorrectionFactors(sampleCorrectionFactors); - - // set the tour mode choice logsums - dcDmu.setTourModeLogsums(sampleLogsums); - - // sampled mgra - dcDmu.setSampleMgra(sampleMgras); - - // sampled taz - dcDmu.setSampleTaz(sampleTazs); - - if (tour.getDebugChoiceModels()) - { - logger.info("***"); - logger.info("Choosing destination alternative from sample"); - tour.logTourObject(logger, 100); - - // log the sample - destModel[purpose].choiceModelUtilityTraceLoggerHeading( - "Visitor tour destination model", "tour " + tour.getID()); - } - - destModel[purpose].computeUtilities(dcDmu, dcDmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - destModel[purpose].logUECResults(logger, "Visitor tour destination model"); - } - int alt = destModel[purpose].getChoiceResult(random); - - int primaryDestination = sampleMgras[alt]; - - if (tour.getDebugChoiceModels()) - { - logger.info("Chose destination MGRA " + primaryDestination); - } - - tour.setDestinationMGRA(primaryDestination); - } - - /** - * @return the tourModeChoiceModel - */ - public VisitorTourModeChoiceModel getTourModeChoiceModel() - { - return tourModeChoiceModel; - } - - /** - * @param tourModeChoiceModel - * the tourModeChoiceModel to set - */ - public void setTourModeChoiceModel(VisitorTourModeChoiceModel tourModeChoiceModel) - { - this.tourModeChoiceModel = tourModeChoiceModel; - } - - /** - * @return the mgraSizeTerms - */ - public double[][] getMgraSizeTerms() - { - return mgraSizeTerms; - } - - /** - * @return the tazSizeTerms - */ - public double[][] getTazSizeTerms() - { - return tazSizeTerms; - } - - /** - * @return the mgraProbabilities - */ - public double[][][] getMgraProbabilities() - { - return mgraProbabilities; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java deleted file mode 100644 index 31acfca..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourEstimationFile.java +++ /dev/null @@ -1,382 +0,0 @@ -package org.sandag.abm.visitor; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.MatrixDataManager; -import com.pb.common.datafile.CSVFileWriter; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.MatrixType; -import com.pb.common.util.ResourceUtil; - -public class VisitorTourEstimationFile -{ - - public static final int MATRIX_DATA_SERVER_PORT = 1171; - public static final int MATRIX_DATA_SERVER_PORT_OFFSET = 0; - - private MatrixDataServerRmi ms; - private String inputFileName; - private String outputFileName; - private TableDataSet estimationData; - - private VisitorModelStructure myModelStructure; - private VisitorDmuFactoryIf dmuFactory; - private McLogsumsCalculator logsumsCalculator; - private VisitorTourModeChoiceModel tourModeChoiceModel; - private HashMap rbMap; - private AutoTazSkimsCalculator tazDistanceCalculator; - - private static Logger logger = Logger.getLogger(VisitorTourEstimationFile.class); - private static final int SAMPLE_SIZE = 30; - private MgraDataManager mgraManager; - - /** - * Default constructor - */ - public VisitorTourEstimationFile(HashMap propertyMap) - { - this.rbMap = propertyMap; - mgraManager = MgraDataManager.getInstance(propertyMap); - myModelStructure = new VisitorModelStructure(); - - dmuFactory = new VisitorDmuFactory(myModelStructure); - - } - - public void createEstimationFile() - { - tazDistanceCalculator = new AutoTazSkimsCalculator(rbMap); - tazDistanceCalculator.computeTazDistanceArrays(); - logsumsCalculator = new McLogsumsCalculator(); - logsumsCalculator.setupSkimCalculators(rbMap); - logsumsCalculator.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - tourModeChoiceModel = new VisitorTourModeChoiceModel(rbMap, myModelStructure, dmuFactory, tazDistanceCalculator); - - // open file - estimationData = openFile(inputFileName); - - // iterate through file and calculate logsums - calculateTourMCLogsums(); - - // write the file - writeFile(outputFileName, estimationData); - } - - /** - * Open a trip file and return the Tabledataset. - * - * @fileName The name of the trip file - * @return The tabledataset - */ - public TableDataSet openFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet tripData; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - tripData = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return tripData; - } - - /** - * Write the file to disk. - * - * @param fileName - * Name of file - * @param data - * TableDataSet to write - */ - public void writeFile(String fileName, TableDataSet data) - { - logger.info("Begin writing the data to file " + fileName); - - try - { - CSVFileWriter csvFile = new CSVFileWriter(); - csvFile.writeFile(data, new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - logger.info("End writing the data to file " + fileName); - - } - - /** - * Calculate mode choice logsums - * - * TOURNUM: Unique ID SURVNUM: Number associated with traveler PersonType: 0 - * - Business, 1 - Personal Persons: Number of persons (in addition to - * traveler) on tour HHIncome: 1= $0-29,999; 2 = 30,000-59,999; 3 = - * 60,000-99,999; 4 = 100,000-149,000; 5=$150,000 or more; 6 = DK/Refused - * AutoAvail: 0 - No auto, 1 - Auto Available PURPOSE: 1 - Work, 2 - Other - * (non-dining) 3 - Dining originMGRA - origin of tour (hotel/overnight - * location) destMGRA - primary destination MGRA PeriodDepart: Period of - * Tour Departure from hotel/overnight location PeriodArrive: Period of Tour - * Arrival at primary destination SAMPLE_: 1:30 sampled alternatives - * - */ - public void calculateTourMCLogsums() - { - - int[] sample = new int[SAMPLE_SIZE]; - float[][] sampleLogsum = new float[SAMPLE_SIZE][estimationData.getRowCount()]; - float[] chosenLogsum = new float[estimationData.getRowCount()]; - String fieldName = null; - - for (int i = 0; i < estimationData.getRowCount(); ++i) - { - - if ((i + 1) <= 10 || (i + 1) % 100 == 0) - { - logger.info("Processing record " + (i + 1)); - } - int ID = (int) estimationData.getValueAt(i + 1, "TOURNUM"); - byte segment = (byte) estimationData.getValueAt(i + 1, "PersonType"); - byte purpose = (byte) estimationData.getValueAt(i + 1, "PURPOSE"); - byte income = (byte) estimationData.getValueAt(i + 1, "HHIncome"); - byte autoAvailable = (byte) estimationData.getValueAt(i + 1, "AutoAvail"); - byte participants = (byte) (estimationData.getValueAt(i + 1, "Persons") + 1); - int departTime = (int) estimationData.getValueAt(i + 1, "PeriodDepart"); - int arriveTime = (int) estimationData.getValueAt(i + 1, "PeriodArrive"); - int originMGRA = (int) estimationData.getValueAt(i + 1, "originMGRA"); - int destinationMGRA = (int) estimationData.getValueAt(i + 1, "destMGRA"); - - for (int j = 0; j < SAMPLE_SIZE; ++j) - { - fieldName = "SAMPLE_" + new Integer(j + 1).toString(); - sample[j] = (int) estimationData.getValueAt(i + 1, fieldName); - } - - VisitorTour tour = new VisitorTour(ID + 10000); - tour.setID(ID); - tour.setSegment(segment); - tour.setPurpose(purpose); - tour.setIncome(income); - tour.setAutoAvailable(autoAvailable); - tour.setNumberOfParticipants(participants); - tour.setDepartTime(departTime); - tour.setArriveTime(arriveTime); - tour.setOriginMGRA(originMGRA); - - if ((i + 1) == 1 || (i + 1) == 500) tour.setDebugChoiceModels(true); - else tour.setDebugChoiceModels(false); - - double logsum = 0; - // for each sampled destination - for (int j = 0; j < SAMPLE_SIZE; ++j) - { - - tour.setDestinationMGRA(sample[j]); - - // some of the samples are 0 - if (sample[j] > 0 && originMGRA > 0) - { - logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, - "DCEstimationFileLogsum", "ID " + ID); - sampleLogsum[j][i] = (float) logsum; - } - } - - // for the chosen destination - tour.setDestinationMGRA(destinationMGRA); - if (originMGRA > 0 && destinationMGRA > 0) - logsum = tourModeChoiceModel.getModeChoiceLogsum(tour, logger, - "DCEstimationFileLogsum", "ID " + ID); - chosenLogsum[i] = (float) logsum; - - } - - // append the logsum fields to the tabledata - estimationData.appendColumn(chosenLogsum, "CHSN_LS"); - - for (int j = 0; j < SAMPLE_SIZE; ++j) - { - fieldName = "SAMPLE_" + (j + 1) + "_LS"; - estimationData.appendColumn(sampleLogsum[j], fieldName); - } - - } - - /** - * Write the destination choice estimation file with logsums appended. - */ - public void writeDCEstimationFile(String name) - { - - } - - /** - * @param args - */ - public static void main(String[] args) - { - // TODO Auto-generated method stub - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - 2.0)); - - logger.info(String.format("Running Visitor Tour Estimation File Model")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - String inFile = args[1]; - String outFile = args[2]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - VisitorTourEstimationFile visitorEstimationFile = new VisitorTourEstimationFile(pMap); - - visitorEstimationFile.inputFileName = inFile; - visitorEstimationFile.outputFileName = outFile; - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = visitorEstimationFile.startMatrixServerProcess( - matrixServerAddress, serverPort, mt); - visitorEstimationFile.ms = matrixServer; - } else - { - visitorEstimationFile.ms = new MatrixDataServerRmi(matrixServerAddress, - serverPort, MatrixDataServer.MATRIX_DATA_SERVER_NAME); - visitorEstimationFile.ms.testRemote(Thread.currentThread().getName()); - - // these methods need to be called to set the matrix data - // manager in the matrix data server - MatrixDataManager mdm = MatrixDataManager.getInstance(); - mdm.setMatrixDataServerObject(visitorEstimationFile.ms); - } - - } - - } catch (Exception e) - { - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - visitorEstimationFile.createEstimationFile(); - - if (args.length < 3) - { - System.out - .println("Error: please specifiy inputFileName and outputFileName on command line"); - throw new RuntimeException(); - } - - } - - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java deleted file mode 100644 index 3bf9f9b..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourManager.java +++ /dev/null @@ -1,531 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.BufferedWriter; -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.ArrayList; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.application.SandagTourBasedModel; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.Util; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.math.MersenneTwister; -import com.pb.common.util.ResourceUtil; - -public class VisitorTourManager -{ - - private static Logger logger = Logger.getLogger(SandagTourBasedModel.class); - - private VisitorTour[] tours; - - VisitorModelStructure modelStructure; - SandagModelStructure sandagStructure; - - TableDataSet businessTourFrequency; - TableDataSet personalTourFrequency; - TableDataSet partySizeFrequency; - TableDataSet autoAvailableFrequency; - TableDataSet incomeFrequency; - - TableDataSet mgraData; - - float occupancyRate; - float householdRate; - - float businessHotelPercent; - float businessHouseholdPercent; - - private boolean seek; - private int traceId; - - private MersenneTwister random; - - private boolean avAvailable; - - /** - * Constructor. Reads properties file and opens/stores all probability - * distributions for sampling. Estimates number of airport travel parties - * and initializes parties[]. - * - * @param resourceFile - * Property file. - * - * Creates the array of cross-border tours. - */ - public VisitorTourManager(HashMap rbMap) - { - - modelStructure = new VisitorModelStructure(); - sandagStructure = new SandagModelStructure(); - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String mgraFile = Util.getStringValueFromPropertyMap(rbMap, "mgra.socec.file"); - mgraFile = directory + mgraFile; - - occupancyRate = new Float(Util.getStringValueFromPropertyMap(rbMap, - "visitor.hotel.occupancyRate")); - householdRate = new Float(Util.getStringValueFromPropertyMap(rbMap, - "visitor.household.occupancyRate")); - - businessHotelPercent = new Float(Util.getStringValueFromPropertyMap(rbMap, - "visitor.hotel.businessPercent")); - businessHouseholdPercent = new Float(Util.getStringValueFromPropertyMap(rbMap, - "visitor.household.businessPercent")); - - String businessTourFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.business.tour.file"); - businessTourFile = directory + businessTourFile; - - String personalTourFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.personal.tour.file"); - personalTourFile = directory + personalTourFile; - - String partySizeFile = Util.getStringValueFromPropertyMap(rbMap, "visitor.partySize.file"); - partySizeFile = directory + partySizeFile; - - String autoAvailableFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.autoAvailable.file"); - autoAvailableFile = directory + autoAvailableFile; - - String incomeFile = Util.getStringValueFromPropertyMap(rbMap, "visitor.income.file"); - incomeFile = directory + incomeFile; - - businessTourFrequency = readFile(businessTourFile); - personalTourFrequency = readFile(personalTourFile); - partySizeFrequency = readFile(partySizeFile); - autoAvailableFrequency = readFile(autoAvailableFile); - incomeFrequency = readFile(incomeFile); - - mgraData = readFile(mgraFile); - - seek = new Boolean(Util.getStringValueFromPropertyMap(rbMap, "visitor.seek")); - traceId = new Integer(Util.getStringValueFromPropertyMap(rbMap, "visitor.trace")); - - float avShare = new Float(Util.getFloatValueFromPropertyMap(rbMap, "Mobility.AV.Share")); - if(avShare>0) - avAvailable=true; - - random = new MersenneTwister(1000001); - - } - - /** - * Read the file and return the TableDataSet. - * - * @param fileName - * @return data - */ - private TableDataSet readFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet data; - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - data = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - return data; - } - - /** - * Generate and attribute visitor tours - */ - public void generateVisitorTours() - { - - // calculate total number of cross border tours - ArrayList tourList = new ArrayList(); - - int rows = mgraData.getRowCount(); - - int tourCount = 0; - int personalCount = 0; - int businessCount = 0; - for (int i = 1; i <= rows; ++i) - { - - float hotelRooms = mgraData.getValueAt(i, "HotelRoomTotal"); - float households = mgraData.getValueAt(i, "hh"); - int mgraNumber = (int) mgraData.getValueAt(i, "mgra"); - - float hotelVisitorParties = hotelRooms * occupancyRate; - float householdVisitorParties = households * householdRate; - - int businessParties = Math.round(hotelVisitorParties * businessHotelPercent); - int personalParties = Math.round(hotelVisitorParties * (1.0f - businessHotelPercent)); - - businessParties += Math.round(householdVisitorParties * businessHouseholdPercent); - personalParties += Math.round(householdVisitorParties - * (1.0f - businessHouseholdPercent)); - - personalCount += personalParties; - businessCount += businessParties; - - // generate a tour for each business party - for (int j = 0; j < businessParties; ++j) - { - - int[] tourPurposes = simulateTours(businessTourFrequency); - - for (int k = 0; k < tourPurposes.length; ++k) - { - VisitorTour tour = new VisitorTour(tourCount + 1000001); - tour.setID(tourCount + 1); - tour.setOriginMGRA(mgraNumber); - tour.setSegment(modelStructure.BUSINESS); - tour.setPurpose((byte) tourPurposes[k]); - calculateSize(tour); - calculateAutoAvailability(tour); - calculateIncome(tour); - tourList.add(tour); - ++tourCount; - } - } - - // generate a tour for each personal party - for (int j = 0; j < personalParties; ++j) - { - - int[] tourPurposes = simulateTours(personalTourFrequency); - - for (int k = 0; k < tourPurposes.length; ++k) - { - VisitorTour tour = new VisitorTour(tourCount + 1000001); - tour.setID(tourCount + 1); - tour.setOriginMGRA(mgraNumber); - tour.setSegment(modelStructure.PERSONAL); - tour.setPurpose((byte) tourPurposes[k]); - calculateSize(tour); - calculateAutoAvailability(tour); - calculateIncome(tour); - tourList.add(tour); - ++tourCount; - } - } - - } - - if (tourList.isEmpty()) - { - logger.error("Visitor tour list is empty!!"); - throw new RuntimeException(); - } - - tours = new VisitorTour[tourList.size()]; - for (int i = 0; i < tours.length; ++i) - tours[i] = tourList.get(i); - - logger.info("Total personal parties: " + personalCount); - logger.info("Total business parties: " + businessCount); - - logger.info("Total visitor tours: " + tourCount); - - } - - /** - * Calculate the number of tours for this travel party, by purpose. Return - * an array whose length equals the number of tours, where each element is - * the purpose of the tour. - * - * @param tourFrequency - * A tableDataSet with the following fields - * WorkTours,RecreationTours,DiningTours,Percent - * @return An array dimensioned to number of tours to generate, with the - * purpose of each. - */ - private int[] simulateTours(TableDataSet tourFrequency) - { - - int[] tourPurposes; - double rand = random.nextDouble(); - - double cumProb = 0.0; - int row = -1; - for (int i = 0; i < tourFrequency.getRowCount(); ++i) - { - - float percent = tourFrequency.getValueAt(i + 1, "Percent"); - cumProb += percent; - if (rand < cumProb) - { - row = i + 1; - break; - } - } - int workTours = (int) tourFrequency.getValueAt(row, "WorkTours"); - int recTours = (int) tourFrequency.getValueAt(row, "RecreationTours"); - int diningTours = (int) tourFrequency.getValueAt(row, "DiningTours"); - - int totalTours = workTours + recTours + diningTours; - tourPurposes = new int[totalTours]; - - int workSet = 0; - int recSet = 0; - int diningSet = 0; - for (int j = 0; j < tourPurposes.length; ++j) - { - - if (workTours > 0 && workSet < workTours) - { - tourPurposes[j] = modelStructure.WORK; - ++workSet; - } else if (recTours > 0 && recSet < recTours) - { - tourPurposes[j] = modelStructure.RECREATION; - ++recSet; - } else if (diningTours > 0 && diningSet < diningTours) - { - tourPurposes[j] = modelStructure.DINING; - ++diningSet; - } - } - return tourPurposes; - } - - /** - * Calculate the size of the tour and store in tour object. - * - * @param tour - */ - private void calculateSize(VisitorTour tour) - { - - byte purp = tour.getPurpose(); - String purpString = modelStructure.VISITOR_PURPOSES[purp]; - String columnName = purpString.toLowerCase(); - - double cumProb = 0; - double rand = tour.getRandom(); - byte size = -1; - int rowCount = partySizeFrequency.getRowCount(); - for (int i = 1; i <= rowCount; ++i) - { - cumProb += partySizeFrequency.getValueAt(i, columnName); - if (rand < cumProb) - { - size = (byte) partySizeFrequency.getValueAt(i, "PartySize"); - break; - } - } - if (size == -1) - { - logger.error("Error attempting to choose party size for visitor tour " + tour.getID()); - throw new RuntimeException(); - } - tour.setNumberOfParticipants(size); - } - - /** - * Calculate whether autos are available for this tour. - * - * @param tour - */ - private void calculateAutoAvailability(VisitorTour tour) - { - - byte purp = tour.getPurpose(); - String purpString = modelStructure.VISITOR_PURPOSES[purp]; - String columnName = purpString.toLowerCase(); - - double rand = tour.getRandom(); - boolean autoAvailable = false; - double probability = autoAvailableFrequency.getValueAt(1, columnName); - if (rand < probability) autoAvailable = true; - - tour.setAutoAvailable(autoAvailable ? 1 : 0); - } - - /** - * Calculate the income of the tour - * - * @param tour - */ - private void calculateIncome(VisitorTour tour) - { - byte segment = tour.getSegment(); - String segmentString = modelStructure.VISITOR_SEGMENTS[segment]; - String columnName = segmentString.toLowerCase(); - - double rand = tour.getRandom(); - int income = -1; - double cumProb = 0; - int rowCount = incomeFrequency.getRowCount(); - for (int i = 1; i <= rowCount; ++i) - { - cumProb += incomeFrequency.getValueAt(i, columnName); - if (rand < cumProb) - { - income = (int) incomeFrequency.getValueAt(i, "Income"); - break; - } - } - if (income == -1) - { - logger.error("Error attempting to choose party size for visitor tour " + tour.getID()); - throw new RuntimeException(); - } - tour.setIncome(income); - - } - - /** - * Create a text file and write all records to the file. - * - */ - public void writeOutputFile(HashMap rbMap) - { - - // Open file and print header - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String tourFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "visitor.tour.output.file"); - String tripFileName = directory - + Util.getStringValueFromPropertyMap(rbMap, "visitor.trip.output.file"); - - logger.info("Writing visitor tours to file " + tourFileName); - logger.info("Writing visitor trips to file " + tripFileName); - - PrintWriter tourWriter = null; - try - { - tourWriter = new PrintWriter(new BufferedWriter(new FileWriter(tourFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tourFileName + " for writing\n"); - throw new RuntimeException(); - } - String tourHeaderString = new String( - "id,segment,purpose,autoAvailable,partySize,income,departTime,arriveTime,originMGRA,destinationMGRA,tourMode,avAvailable,outboundStops,inboundStops,valueOfTime\n"); - tourWriter.print(tourHeaderString); - - PrintWriter tripWriter = null; - try - { - tripWriter = new PrintWriter(new BufferedWriter(new FileWriter(tripFileName))); - } catch (IOException e) - { - logger.fatal("Could not open file " + tripFileName + " for writing\n"); - throw new RuntimeException(); - } - String tripHeaderString = new String( - "tourID,tripID,originPurp,destPurp,originMGRA,destinationMGRA,inbound,originIsTourDestination,destinationIsTourDestination,period,tripMode,avAvailable,boardingTap,alightingTap,set,valueOfTime,partySize," - +"micro_walkMode,micro_trnAcc,micro_trnEgr,parkingCost\n"); - tripWriter.print(tripHeaderString); - - // Iterate through the array, printing records to the file - for (int i = 0; i < tours.length; ++i) - { - - VisitorTour tour = tours[i]; - - if (seek && tour.getID() != traceId) continue; - - VisitorTrip[] trips = tours[i].getTrips(); - - if (trips == null) continue; - - writeTour(tour, tourWriter); - - for (int j = 0; j < trips.length; ++j) - { - writeTrip(tour, trips[j], j + 1, tripWriter); - } - } - - tourWriter.close(); - tripWriter.close(); - - } - - /** - * Write the tour to the PrintWriter - * - * @param tour - * @param writer - */ - private void writeTour(VisitorTour tour, PrintWriter writer) - { - String record = new String(tour.getID() + "," + tour.getSegment() + "," + tour.getPurpose() - + "," + tour.getAutoAvailable() + "," + tour.getNumberOfParticipants() + "," - + tour.getIncome() + "," + tour.getDepartTime() + "," + tour.getArriveTime() + "," - + tour.getOriginMGRA() + "," + tour.getDestinationMGRA() + "," + tour.getTourMode() + "," - + (avAvailable?1:0) - + "," + tour.getNumberOutboundStops() + "," + tour.getNumberInboundStops() - + "," + String.format("%9.2f",tour.getValueOfTime())+ "\n"); - writer.print(record); - - } - - /** - * Write the trip to the PrintWriter - * - * @param tour - * @param trip - * @param tripNumber - * @param writer - */ - private void writeTrip(VisitorTour tour, VisitorTrip trip, int tripNumber, PrintWriter writer) - { - String record = new String(tour.getID() + "," + tripNumber + "," + trip.getOriginPurpose() - + "," + trip.getDestinationPurpose() + "," + trip.getOriginMgra() + "," - + trip.getDestinationMgra() + "," + trip.isInbound() + "," - + trip.isOriginIsTourDestination() + "," + trip.isDestinationIsTourDestination() - + "," + trip.getPeriod() + "," + trip.getTripMode() + "," + (avAvailable?1:0) +"," - + trip.getBoardTap() + "," + trip.getAlightTap() + "," + trip.getSet() - + "," + String.format("%9.2f",trip.getValueOfTime())+ "," + tour.getNumberOfParticipants()+"," - +trip.getMicromobilityWalkMode()+"," +trip.getMicromobilityAccessMode()+"," +trip.getMicromobilityEgressMode() - + "," + String.format("%9.2f", trip.getParkingCost()) +"\n"); - writer.print(record); - } - - - /** - * @return the parties - */ - public VisitorTour[] getTours() - { - return tours; - } - - public static void main(String[] args) - { - - String propertiesFile = null; - HashMap pMap; - - logger.info(String.format("SANDAG Activity Based Model using CT-RAMP version %s", - CtrampApplication.VERSION)); - - logger.info(String.format("Running Cross Border Model Tour Manager")); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - VisitorTourManager apm = new VisitorTourManager(pMap); - apm.generateVisitorTours(); - apm.writeOutputFile(pMap); - - logger.info("Cross-Border Tour Manager successfully completed!"); - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java deleted file mode 100644 index 3fa9b3c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceDMU.java +++ /dev/null @@ -1,570 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.TourModeChoiceDMU; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class VisitorTourModeChoiceDMU - implements Serializable, VariableTable -{ - protected transient Logger logger = Logger.getLogger(VisitorTourModeChoiceDMU.class); - - public static final int WTW = McLogsumsCalculator.WTW; - public static final int WTD = McLogsumsCalculator.WTD; - public static final int DTW = McLogsumsCalculator.DTW; - protected static final int NUM_ACC_EGR = McLogsumsCalculator.NUM_ACC_EGR; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected float tourDepartPeriod; - protected float tourArrivePeriod; - protected double origDuDen; - protected double origEmpDen; - protected double origTotInt; - protected double destDuDen; - protected double destEmpDen; - protected double destTotInt; - - protected int partySize; - protected int autoAvailable; - protected int income; - protected int tourPurpose; - - protected float pTazTerminalTime; - protected float aTazTerminalTime; - - protected double nmWalkTimeOut; - protected double nmWalkTimeIn; - protected double nmBikeTimeOut; - protected double nmBikeTimeIn; - protected double lsWgtAvgCostM; - protected double lsWgtAvgCostD; - protected double lsWgtAvgCostH; - - protected double[][] transitLogSum; - protected float origTaxiWaitTime; - protected float destTaxiWaitTime; - protected float origSingleTNCWaitTime; - protected float destSingleTNCWaitTime; - protected float origSharedTNCWaitTime; - protected float destSharedTNCWaitTime; - - public VisitorTourModeChoiceDMU(VisitorModelStructure modelStructure, Logger aLogger) - { - if (aLogger == null) aLogger = Logger.getLogger(TourModeChoiceDMU.class); - logger = aLogger; - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - //accEgr by in/outbound - transitLogSum = new double[NUM_ACC_EGR][NUM_DIR]; - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - public void setLsWgtAvgCostM(double cost) - { - lsWgtAvgCostM = cost; - } - - public void setLsWgtAvgCostD(double cost) - { - lsWgtAvgCostD = cost; - } - - public void setLsWgtAvgCostH(double cost) - { - lsWgtAvgCostH = cost; - } - - public double getMonthlyParkingCost() - { - return lsWgtAvgCostM; - } - - public double getDailyParkingCost() - { - return lsWgtAvgCostD; - } - - public double getHourlyParkingCost() - { - return lsWgtAvgCostH; - } - - public float getTimeOutbound() - { - return tourDepartPeriod; - } - - public float getTimeInbound() - { - return tourArrivePeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(float tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(float tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - public void setOrigDuDen(double arg) - { - origDuDen = arg; - } - - public void setOrigEmpDen(double arg) - { - origEmpDen = arg; - } - - public void setOrigTotInt(double arg) - { - origTotInt = arg; - } - - public void setDestDuDen(double arg) - { - destDuDen = arg; - } - - public void setDestEmpDen(double arg) - { - destEmpDen = arg; - } - - public void setDestTotInt(double arg) - { - destTotInt = arg; - } - - public int getTourPurpose() - { - return tourPurpose; - } - - public void setTourPurpose(int tourPurpose) - { - this.tourPurpose = tourPurpose; - } - - public double getODUDen() - { - return origDuDen; - } - - public double getOEmpDen() - { - return origEmpDen; - } - - public double getOTotInt() - { - return origTotInt; - } - - public double getDDUDen() - { - return destDuDen; - } - - public double getDEmpDen() - { - return destEmpDen; - } - - public double getDTotInt() - { - return destTotInt; - } - - public void setNmWalkTimeOut(double nmWalkTime) - { - nmWalkTimeOut = nmWalkTime; - } - - public double getNm_walkTime_out() - { - return nmWalkTimeOut; - } - - public void setNmWalkTimeIn(double nmWalkTime) - { - nmWalkTimeIn = nmWalkTime; - } - - public double getNm_walkTime_in() - { - return nmWalkTimeIn; - } - - public void setNmBikeTimeOut(double nmBikeTime) - { - nmBikeTimeOut = nmBikeTime; - } - - public double getNm_bikeTime_out() - { - return nmBikeTimeOut; - } - - public void setNmBikeTimeIn(double nmBikeTime) - { - nmBikeTimeIn = nmBikeTime; - } - - public double getNm_bikeTime_in() - { - return nmBikeTimeIn; - } - - public void setPTazTerminalTime(float time) - { - pTazTerminalTime = time; - } - - public void setATazTerminalTime(float time) - { - aTazTerminalTime = time; - } - - public double getPTazTerminalTime() - { - return pTazTerminalTime; - } - - public double getATazTerminalTime() - { - return aTazTerminalTime; - } - - public int getPartySize() - { - return partySize; - } - - public void setPartySize(int partySize) - { - this.partySize = partySize; - } - - public int getAutoAvailable() - { - return autoAvailable; - } - - public void setAutoAvailable(int autoAvailable) - { - this.autoAvailable = autoAvailable; - } - - public int getIncome() - { - return income; - } - - public void setIncome(int income) - { - this.income = income; - } - - public void setTransitLogSum(int accEgr, boolean inbound, double value){ - transitLogSum[accEgr][inbound == true ? 1 : 0] = value; - } - - protected double getTransitLogSum(int accEgr,boolean inbound){ - return transitLogSum[accEgr][inbound == true ? 1 : 0]; - } - - - - public float getOrigTaxiWaitTime() { - return origTaxiWaitTime; - } - - - - public void setOrigTaxiWaitTime(float origTaxiWaitTime) { - this.origTaxiWaitTime = origTaxiWaitTime; - } - - - - public float getDestTaxiWaitTime() { - return destTaxiWaitTime; - } - - - - public void setDestTaxiWaitTime(float destTaxiWaitTime) { - this.destTaxiWaitTime = destTaxiWaitTime; - } - - - - public float getOrigSingleTNCWaitTime() { - return origSingleTNCWaitTime; - } - - - - public void setOrigSingleTNCWaitTime(float origSingleTNCWaitTime) { - this.origSingleTNCWaitTime = origSingleTNCWaitTime; - } - - - - public float getDestSingleTNCWaitTime() { - return destSingleTNCWaitTime; - } - - - - public void setDestSingleTNCWaitTime(float destSingleTNCWaitTime) { - this.destSingleTNCWaitTime = destSingleTNCWaitTime; - } - - - - public float getOrigSharedTNCWaitTime() { - return origSharedTNCWaitTime; - } - - - - public void setOrigSharedTNCWaitTime(float origSharedTNCWaitTime) { - this.origSharedTNCWaitTime = origSharedTNCWaitTime; - } - - - - public float getDestSharedTNCWaitTime() { - return destSharedTNCWaitTime; - } - - - - public void setDestSharedTNCWaitTime(float destSharedTNCWaitTime) { - this.destSharedTNCWaitTime = destSharedTNCWaitTime; - } - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTimeOutbound", 0); - methodIndexMap.put("getTimeInbound", 1); - methodIndexMap.put("getPTazTerminalTime", 14); - methodIndexMap.put("getATazTerminalTime", 15); - methodIndexMap.put("getODUDen", 16); - methodIndexMap.put("getOEmpDen", 17); - methodIndexMap.put("getOTotInt", 18); - methodIndexMap.put("getDDUDen", 19); - methodIndexMap.put("getDEmpDen", 20); - methodIndexMap.put("getDTotInt", 21); - methodIndexMap.put("getMonthlyParkingCost", 23); - methodIndexMap.put("getDailyParkingCost", 24); - methodIndexMap.put("getHourlyParkingCost", 25); - methodIndexMap.put("getPartySize", 30); - methodIndexMap.put("getAutoAvailable", 31); - methodIndexMap.put("getIncome", 32); - methodIndexMap.put("getTourPurpose", 33); - - methodIndexMap.put("getIvtCoeff", 56); - methodIndexMap.put("getCostCoeff", 57); - methodIndexMap.put("getWalkSetLogSum", 59); - methodIndexMap.put("getPnrSetLogSum", 60); - methodIndexMap.put("getKnrSetLogSum", 61); - - methodIndexMap.put( "getOrigTaxiWaitTime", 70 ); - methodIndexMap.put( "getDestTaxiWaitTime", 71 ); - methodIndexMap.put( "getOrigSingleTNCWaitTime", 72 ); - methodIndexMap.put( "getDestSingleTNCWaitTime", 73 ); - methodIndexMap.put( "getOrigSharedTNCWaitTime", 74 ); - methodIndexMap.put( "getDestSharedTNCWaitTime", 75 ); - - methodIndexMap.put("getNm_walkTime_out", 90); - methodIndexMap.put("getNm_walkTime_in", 91); - methodIndexMap.put("getNm_bikeTime_out", 92); - methodIndexMap.put("getNm_bikeTime_in", 93); - - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - case 0: - returnValue = getTimeOutbound(); - break; - case 1: - returnValue = getTimeInbound(); - break; - case 14: - returnValue = getPTazTerminalTime(); - break; - case 15: - returnValue = getATazTerminalTime(); - break; - case 16: - returnValue = getODUDen(); - break; - case 17: - returnValue = getOEmpDen(); - break; - case 18: - returnValue = getOTotInt(); - break; - case 19: - returnValue = getDDUDen(); - break; - case 20: - returnValue = getDEmpDen(); - break; - case 21: - returnValue = getDTotInt(); - break; - case 23: - returnValue = getMonthlyParkingCost(); - break; - case 24: - returnValue = getDailyParkingCost(); - break; - case 25: - returnValue = getHourlyParkingCost(); - break; - case 30: - returnValue = getPartySize(); - break; - case 31: - returnValue = getAutoAvailable(); - break; - case 32: - returnValue = getIncome(); - break; - case 33: - returnValue = getTourPurpose(); - break; - case 59: - returnValue = getTransitLogSum(WTW, true) + getTransitLogSum(WTW, false); - break; - case 60: - returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); - break; - case 61: - returnValue = getTransitLogSum(WTD, true) + getTransitLogSum(DTW, false); - break; - case 70: return getOrigTaxiWaitTime(); - case 71: return getDestTaxiWaitTime(); - case 72: return getOrigSingleTNCWaitTime(); - case 73: return getDestSingleTNCWaitTime(); - case 74: return getOrigSharedTNCWaitTime(); - case 75: return getDestSharedTNCWaitTime(); - case 90: - returnValue = getNm_walkTime_out(); - break; - case 91: - returnValue = getNm_walkTime_in(); - break; - case 92: - returnValue = getNm_bikeTime_out(); - break; - case 93: - returnValue = getNm_bikeTime_in(); - break; - default: - logger.error("method number = " + variableIndex + " not found"); - throw new RuntimeException("method number = " + variableIndex + " not found"); - } - - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java deleted file mode 100644 index 87572d1..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourModeChoiceModel.java +++ /dev/null @@ -1,399 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Random; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.TNCAndTaxiWaitTimeCalculator; -import org.sandag.abm.ctramp.TourModeChoiceDMU; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class VisitorTourModeChoiceModel - implements Serializable -{ - - private transient Logger logger = Logger.getLogger("visitorModel"); - - public static final boolean DEBUG_BEST_PATHS = false; - - private MgraDataManager mgraManager; - - protected static final int OUT = McLogsumsCalculator.OUT; - protected static final int IN = McLogsumsCalculator.IN; - protected static final int NUM_DIR = McLogsumsCalculator.NUM_DIR; - - private static final String PROPERTIES_UEC_TOUR_MODE_CHOICE = "visitor.mc.uec.file"; - private static final String PROPERTIES_UEC_TOUR_DATA_SHEET = "visitor.mc.data.page"; - private static final String PROPERTIES_UEC_TOUR_MODEL_SHEET = "visitor.mc.model.page"; - - private ChoiceModelApplication mcModel; - private VisitorTourModeChoiceDMU mcDmuObject; - private TripModeChoiceDMU tripDmuObject; - private McLogsumsCalculator logsumHelper; - - private VisitorModelStructure modelStructure; - - private String tourCategory; - - private String[] modeAltNames; - - private boolean saveUtilsProbsFlag = false; - private AutoTazSkimsCalculator tazDistanceCalculator; - - //added for TNC and Taxi modes - TNCAndTaxiWaitTimeCalculator tncTaxiWaitTimeCalculator = null; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public VisitorTourModeChoiceModel(HashMap propertyMap, - VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - - mgraManager = MgraDataManager.getInstance(propertyMap); - modelStructure = myModelStructure; - this.tazDistanceCalculator = tazDistanceCalculator; - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - tripDmuObject = new TripModeChoiceDMU(modelStructure,logger); - mcDmuObject = dmuFactory.getVisitorTourModeChoiceDMU(); - setupModeChoiceModelApplicationArray(propertyMap); - - tncTaxiWaitTimeCalculator = new TNCAndTaxiWaitTimeCalculator(); - tncTaxiWaitTimeCalculator.createWaitTimeDistributions(propertyMap); - - } - - /** - * Set up the mode choice model. - * - * @param propertyMap - */ - private void setupModeChoiceModelApplicationArray(HashMap propertyMap) - { - - logger.info(String.format("setting up visitor tour mode choice model.")); - - // locate the individual mandatory tour mode choice model UEC - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String mcUecFile = Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_TOUR_MODE_CHOICE); - mcUecFile = uecPath + mcUecFile; - - logger.info("Will read mcUECFile " + mcUecFile); - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_TOUR_DATA_SHEET)); - int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_TOUR_MODEL_SHEET)); - - // default is to not save the tour mode choice utils and probs for each - // tour - String saveUtilsProbsString = propertyMap - .get(CtrampApplication.PROPERTIES_SAVE_TOUR_MODE_CHOICE_UTILS); - if (saveUtilsProbsString != null) - { - if (saveUtilsProbsString.equalsIgnoreCase("true")) saveUtilsProbsFlag = true; - } - - mcModel = new ChoiceModelApplication(mcUecFile, modelPage, dataPage, propertyMap, - (VariableTable) mcDmuObject); - modeAltNames = mcModel.getAlternativeNames(); - - } - - public double getModeChoiceLogsum(VisitorTour tour, Logger modelLogger, - String choiceModelDescription, String decisionMakerLabel) - { - - setDmuAttributes(tour); - - // log headers to traceLogger - if (tour.getDebugChoiceModels()) - { - mcModel.choiceModelUtilityTraceLoggerHeading(choiceModelDescription, decisionMakerLabel); - } - - - mcModel.computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); - double logsum = mcModel.getLogsum(); - - // write UEC calculation results to separate model specific log file - if (tour.getDebugChoiceModels()) - { - String loggingHeader = String.format("%s %s", choiceModelDescription, - decisionMakerLabel); - mcModel.logUECResults(modelLogger, loggingHeader); - modelLogger.info(choiceModelDescription + " Logsum value: " + logsum); - modelLogger.info(""); - modelLogger.info(""); - } - - return logsum; - - } - - /** - * Set the DMU attributes for the tour. - * - * @param tour - */ - private void setDmuAttributes(VisitorTour tour) - { - - // update the MC dmuObjects for this person - int originTaz = mgraManager.getTaz(tour.getOriginMGRA()); - int destinationTaz = mgraManager.getTaz(tour.getDestinationMGRA()); - mcDmuObject.setDmuIndexValues(tour.getID(), originTaz, originTaz, destinationTaz, - tour.getDebugChoiceModels()); - - mcDmuObject.setTourDepartPeriod(tour.getDepartTime()); - mcDmuObject.setTourArrivePeriod(tour.getArriveTime()); - mcDmuObject.setIncome((byte) tour.getIncome()); - mcDmuObject.setAutoAvailable(tour.getAutoAvailable()); - mcDmuObject.setPartySize(tour.getNumberOfParticipants()); - mcDmuObject.setTourPurpose(tour.getPurpose()); - double ivtCoeff = -0.015; - double costCoeff = -0.0017; - tripDmuObject.setIvtCoeff(ivtCoeff); - tripDmuObject.setCostCoeff(costCoeff); - - logsumHelper.setNmTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), - tour.getDepartTime(),tour.getDebugChoiceModels()); - double nmWalkTimeOut = tripDmuObject.getNm_walkTime(); - double nmBikeTimeOut = tripDmuObject.getNm_bikeTime(); - mcDmuObject.setNmWalkTimeOut(nmWalkTimeOut); - mcDmuObject.setNmBikeTimeOut(nmBikeTimeOut); - logsumHelper.setNmTripMcDmuAttributes( tripDmuObject, tour.getDestinationMGRA(), tour.getOriginMGRA(), - tour.getArriveTime(),tour.getDebugChoiceModels()); - double nmWalkTimeIn = tripDmuObject.getNm_walkTime(); - double nmBikeTimeIn = tripDmuObject.getNm_bikeTime(); - mcDmuObject.setNmWalkTimeOut(nmWalkTimeIn); - mcDmuObject.setNmBikeTimeOut(nmBikeTimeIn); - - - double walkTransitLogsumOut = -999.0; - double driveTransitLogsumOut = -999.0; - double walkTransitLogsumIn = -999.0; - double driveTransitLogsumIn = -999.0; - - // walk-transit out logsum - logsumHelper.setWtwTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), - tour.getDepartTime(),tour.getDebugChoiceModels()); - - walkTransitLogsumOut = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - // walk-transit in logsum - logsumHelper.setWtwTripMcDmuAttributes( tripDmuObject,tour.getDestinationMGRA(), tour.getOriginMGRA(), - tour.getArriveTime(),tour.getDebugChoiceModels()); - - walkTransitLogsumIn = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - //drive-transit out logsum - logsumHelper.setDtwTripMcDmuAttributes( tripDmuObject, tour.getOriginMGRA(), tour.getDestinationMGRA(), - tour.getDepartTime(),tour.getDebugChoiceModels()); - - driveTransitLogsumOut = tripDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); - - //drive-transit in logsum - logsumHelper.setWtdTripMcDmuAttributes( tripDmuObject, tour.getDestinationMGRA(),tour.getOriginMGRA(), - tour.getArriveTime(),tour.getDebugChoiceModels()); - - driveTransitLogsumIn = tripDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); - - mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTW,false,walkTransitLogsumOut); - mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTW,true,walkTransitLogsumIn); - mcDmuObject.setTransitLogSum(McLogsumsCalculator.DTW,false,driveTransitLogsumOut); - mcDmuObject.setTransitLogSum(McLogsumsCalculator.WTD,true,driveTransitLogsumIn); - - float SingleTNCWaitTimeOrig = 0; - float SingleTNCWaitTimeDest = 0; - float SharedTNCWaitTimeOrig = 0; - float SharedTNCWaitTimeDest = 0; - float TaxiWaitTimeOrig = 0; - float TaxiWaitTimeDest = 0; - float popEmpDenOrig = (float) mgraManager.getPopEmpPerSqMi(tour.getOriginMGRA()); - float popEmpDenDest = (float) mgraManager.getPopEmpPerSqMi(tour.getDestinationMGRA()); - - double rnum = tour.getRandom(); - SingleTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SingleTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSingleTNCWaitTimeDistribution(rnum, popEmpDenDest); - SharedTNCWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenOrig); - SharedTNCWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromSharedTNCWaitTimeDistribution(rnum, popEmpDenDest); - TaxiWaitTimeOrig = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenOrig); - TaxiWaitTimeDest = (float) tncTaxiWaitTimeCalculator.sampleFromTaxiWaitTimeDistribution(rnum, popEmpDenDest); - - mcDmuObject.setOrigTaxiWaitTime(TaxiWaitTimeOrig); - mcDmuObject.setDestTaxiWaitTime(TaxiWaitTimeDest); - mcDmuObject.setOrigSingleTNCWaitTime(SingleTNCWaitTimeOrig); - mcDmuObject.setDestSingleTNCWaitTime(SingleTNCWaitTimeDest); - mcDmuObject.setOrigSharedTNCWaitTime(SharedTNCWaitTimeOrig); - mcDmuObject.setDestSharedTNCWaitTime(SharedTNCWaitTimeDest); - - - } - - /** - * Use to choose tour mode and set result in tour object. - * - * @param tour - * The crossborder tour - */ - public void chooseTourMode(VisitorTour tour) - { - - byte tourMode = (byte) getModeChoice(tour); - tour.setTourMode(tourMode); - } - - /** - * Get the choice of mode for the tour, and return as an integer (don't - * store in tour object) - * - * @param tour - * @return - */ - private int getModeChoice(VisitorTour tour) - { - - String choiceModelDescription = ""; - String decisionMakerLabel = ""; - String loggingHeader = ""; - String separator = ""; - String purposeName = modelStructure.VISITOR_PURPOSES[tour.getPurpose()]; - - if (tour.getDebugChoiceModels()) - { - - tour.logTourObject(logger, 100); - logger.info("Logging tour mode choice model"); - } - - setDmuAttributes(tour); - - mcModel.computeUtilities(mcDmuObject, mcDmuObject.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - mcModel.logUECResults(logger, "Visitor tour mode choice model"); - - double rn = tour.getRandom(); - - // if the choice model has at least one available alternative, make - // choice. - int chosen; - if (mcModel.getAvailabilityCount() > 0) - { - - chosen = mcModel.getChoiceResult(rn); - - //value of time; lookup vot from the UEC - UtilityExpressionCalculator uec = mcModel.getUEC(); - int votIndex = uec.lookupVariableIndex("vot"); - float vot = (float) uec.getValueForIndex(votIndex); - - tour.setValueOfTime(vot); - - } else - { - - tour.logTourObject(logger, loggingHeader.length()); - - mcModel.logUECResults(logger, loggingHeader); - logger.info(""); - logger.info(""); - - logger.error(String - .format("Exception caught for HHID=%d, no available %s tour mode alternatives to choose from in choiceModelApplication.", - tour.getID(), tourCategory)); - throw new RuntimeException(); - } - - // debug output - if (tour.getDebugChoiceModels()) - { - - double[] utilities = mcModel.getUtilities(); // 0s-indexing - double[] probabilities = mcModel.getProbabilities(); // 0s-indexing - boolean[] availabilities = mcModel.getAvailabilities(); // 1s-indexing - String[] altNames = mcModel.getAlternativeNames(); // 0s-indexing - - logger.info("Tour Id: " + tour.getID()); - logger.info("Alternative Utility Probability CumProb"); - logger.info("-------------------- -------------- -------------- --------------"); - - double cumProb = 0.0; - for (int k = 0; k < mcModel.getNumberOfAlternatives(); k++) - { - cumProb += probabilities[k]; - String altString = String.format("%-3d %s", k + 1, altNames[k]); - logger.info(String.format("%-20s%15s%18.6e%18.6e%18.6e", altString, - availabilities[k + 1], utilities[k], probabilities[k], cumProb)); - } - - logger.info(" "); - String altString = String.format("%-3d %s", chosen, altNames[chosen - 1]); - logger.info(String.format("Choice: %s, with rn=%.8f", altString, rn)); - - logger.info(separator); - logger.info(""); - logger.info(""); - - // write choice model alternative info to log file - mcModel.logAlternativesInfo(choiceModelDescription, decisionMakerLabel); - mcModel.logSelectionInfo(choiceModelDescription, decisionMakerLabel, rn, chosen); - mcModel.logLogitCalculations(choiceModelDescription, decisionMakerLabel); - - } - - if (saveUtilsProbsFlag) - { - - // get the utilities and probabilities arrays for the tour mode - // choice - // model for this tour and save them to the tour object - double[] dUtils = mcModel.getUtilities(); - double[] dProbs = mcModel.getProbabilities(); - - float[] utils = new float[dUtils.length]; - float[] probs = new float[dUtils.length]; - for (int k = 0; k < dUtils.length; k++) - { - utils[k] = (float) dUtils[k]; - probs[k] = (float) dProbs[k]; - } - - // tour.setTourModalUtilities(utils); - // tour.setTourModalProbabilities(probs); - - } - - return chosen; - - } - - public String[] getModeAltNames(int purposeIndex) - { - return modeAltNames; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java deleted file mode 100644 index f12d379..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTourTimeOfDayChoiceModel.java +++ /dev/null @@ -1,192 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.File; -import java.io.IOException; -import java.util.HashMap; -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.Util; -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; - -/** - * This class is the TOD choice model for cross border tours. It is currently - * based on a static probability distribution stored in an input file, and - * indexed into by purpose. - * - * @author Freedman - * - */ -public class VisitorTourTimeOfDayChoiceModel -{ - private transient Logger logger = Logger.getLogger("visitorModel"); - - private double[][] cumProbability; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - private int[][] outboundPeriod; // by - // purpose, - // alternative: - // outbound - // period - private int[][] returnPeriod; // by - // purpose, - // alternative: - // return - // period - VisitorModelStructure modelStructure; - - /** - * Constructor. - */ - public VisitorTourTimeOfDayChoiceModel(HashMap rbMap) - { - - String directory = Util.getStringValueFromPropertyMap(rbMap, "Project.Directory"); - String stationDiurnalFile = Util.getStringValueFromPropertyMap(rbMap, - "visitor.tour.tod.file"); - stationDiurnalFile = directory + stationDiurnalFile; - - modelStructure = new VisitorModelStructure(); - - readTODFile(stationDiurnalFile); - - } - - /** - * Read the TOD distribution in the file and populate the arrays. - * - * @param fileName - */ - private void readTODFile(String fileName) - { - - logger.info("Begin reading the data in file " + fileName); - TableDataSet probabilityTable; - - try - { - OLD_CSVFileReader csvFile = new OLD_CSVFileReader(); - probabilityTable = csvFile.readFile(new File(fileName)); - } catch (IOException e) - { - throw new RuntimeException(e); - } - - logger.info("End reading the data in file " + fileName); - - logger.info("Begin calculating tour TOD probability distribution"); - - int purposes = modelStructure.VISITOR_PURPOSES.length; // start at 0 - int periods = modelStructure.TIME_PERIODS; // start at 1 - int periodCombinations = periods * (periods + 1) / 2; - - cumProbability = new double[purposes][periodCombinations]; // by - // purpose, - // alternative: - // cumulative - // probability - // distribution - outboundPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // outbound - // period - returnPeriod = new int[purposes][periodCombinations]; // by purpose, - // alternative: - // return period - - // fill up arrays - int rowCount = probabilityTable.getRowCount(); - int lastPurpose = -99; - double cumProb = 0; - int alt = 0; - for (int row = 1; row <= rowCount; ++row) - { - - int purpose = (int) probabilityTable.getValueAt(row, "Purpose"); - int outPer = (int) probabilityTable.getValueAt(row, "EntryPeriod"); - int retPer = (int) probabilityTable.getValueAt(row, "ReturnPeriod"); - - // continue if return period before outbound period - if (retPer < outPer) continue; - - // reset if new purpose - if (purpose != lastPurpose) - { - - // log cumulative probability just in case - if (lastPurpose != -99) - logger.info("Cumulative probability for purpose " + purpose + " is " + cumProb); - cumProb = 0; - alt = 0; - } - - // calculate cumulative probability and store in array - cumProb += probabilityTable.getValueAt(row, "Percent"); - cumProbability[purpose][alt] = cumProb; - outboundPeriod[purpose][alt] = outPer; - returnPeriod[purpose][alt] = retPer; - - ++alt; - - lastPurpose = purpose; - } - - logger.info("End calculating tour TOD probability distribution"); - - } - - /** - * Calculate tour time of day for the tour. - * - * @param tour - * A cross border tour (with purpose) - */ - public void calculateTourTOD(VisitorTour tour) - { - - int purpose = tour.getPurpose(); - double random = tour.getRandom(); - - int depart = -1; - int arrive = -1; - if (tour.getDebugChoiceModels()) - { - logger.info("Choosing tour time of day for purpose " - + modelStructure.VISITOR_PURPOSES[purpose] + " using random number " + random); - tour.logTourObject(logger, 100); - } - - for (int i = 0; i < cumProbability[purpose].length; ++i) - { - - if (random < cumProbability[purpose][i]) - { - depart = outboundPeriod[purpose][i]; - arrive = returnPeriod[purpose][i]; - tour.setDepartTime(depart); - tour.setArriveTime(arrive); - break; - } - } - if((depart ==-1)||(arrive==-1)){ - logger.fatal("Error: did not find outbound or return period for tour"); - logger.fatal("Depart period, arrive period = "+depart+","+arrive); - logger.fatal("Random number: "+random); - tour.logTourObject(logger,100); - throw new RuntimeException(); - } - - - if (tour.getDebugChoiceModels()) - { - logger.info(""); - logger.info("Chose depart period " + tour.getDepartTime() + " and arrival period " - + tour.getArriveTime()); - logger.info(""); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java deleted file mode 100644 index 32f7dc8..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTrip.java +++ /dev/null @@ -1,518 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; - -public class VisitorTrip - implements Serializable -{ - - private int originMgra; - private int destinationMgra; - private int tripMode; - private byte originPurpose; - private byte destinationPurpose; - private byte period; - private boolean inbound; - private boolean firstTrip; - private boolean lastTrip; - private boolean originIsTourDestination; - private boolean destinationIsTourDestination; - byte micromobilityWalkMode; - byte micromobilityAccessMode; - byte micromobilityEgressMode; - float micromobilityWalkLogsum; - float micromobilityAccessLogsum; - float micromobilityEgressLogsum; - - float parkingCost; - - private int boardTap; - private int alightTap; - private int set = -1; - - private float valueOfTime; - - /** - * Default constructor; nothing initialized. - */ - public VisitorTrip() - { - - } - - /** - * Create a cross border trip from a tour leg (no stops). - * - * @param tour - * The tour. - * @param outbound - * Outbound direction - */ - public VisitorTrip(VisitorTour tour, boolean outbound) - { - - initializeFromTour(tour, outbound); - } - - /** - * Initilize from the tour. - * - * @param tour - * The tour. - * @param outbound - * Outbound direction. - */ - public void initializeFromTour(VisitorTour tour, boolean outbound) - { - // Note: mode is unknown - if (outbound) - { - this.originMgra = tour.getOriginMGRA(); - this.destinationMgra = tour.getDestinationMGRA(); - this.originPurpose = -1; - this.destinationPurpose = tour.getPurpose(); - this.period = (byte) tour.getDepartTime(); - this.inbound = false; - this.firstTrip = true; - this.lastTrip = false; - this.originIsTourDestination = false; - this.destinationIsTourDestination = true; - } else - { - this.originMgra = tour.getDestinationMGRA(); - this.destinationMgra = tour.getOriginMGRA(); - this.originPurpose = tour.getPurpose(); - this.destinationPurpose = -1; - this.period = (byte) tour.getArriveTime(); - this.inbound = true; - this.firstTrip = false; - this.lastTrip = true; - this.originIsTourDestination = true; - this.destinationIsTourDestination = false; - } - - } - - /** - * Create a visitor trip from a tour\stop. Note: trip mode is unknown. Stop - * period is only known for first, last stop on tour. - * - * @param tour - * The tour. - * @param stop - * The stop - */ - public VisitorTrip(VisitorTour tour, VisitorStop stop, boolean toStop) - { - - initializeFromStop(tour, stop, toStop); - } - - /** - * Initialize from stop attributes. A trip will be created to the stop if - * toStop is true, else a trip will be created from the stop. Use after all - * stop locations are known, or else reset the stop origin and destination - * mgras accordingly after using. - * - * @param tour - * @param stop - * @param toStop - */ - public void initializeFromStop(VisitorTour tour, VisitorStop stop, boolean toStop) - { - - this.inbound = stop.isInbound(); - this.destinationIsTourDestination = false; - this.originIsTourDestination = false; - - // if trip to stop, destination is stop mgra; else origin is stop mgra - if (toStop) - { - this.destinationMgra = stop.getMgra(); - this.destinationPurpose = stop.getPurpose(); - } else - { - this.originMgra = stop.getMgra(); - this.originPurpose = stop.getPurpose(); - } - VisitorStop[] stops; - - if (!inbound) stops = tour.getOutboundStops(); - else stops = tour.getInboundStops(); - - // if outbound, and trip is to stop - if (!inbound && toStop) - { - - // first trip on outbound journey, origin is tour origin - if (stop.getId() == 0) - { - this.originMgra = tour.getOriginMGRA(); - this.originPurpose = -1; - this.period = (byte) tour.getDepartTime(); - } else - { - // not first trip on outbound journey, origin is last stop - this.originMgra = stops[stop.getId() - 1].getMgra(); // last - // stop - // location - this.originPurpose = stops[stop.getId() - 1].getPurpose(); // last - // stop - // location - this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); - } - } else if (!inbound && !toStop) - { - // outbound and trip is from stop to either next stop or tour - // destination. - - // last trip on outbound journey, destination is tour destination - if (stop.getId() == (stops.length - 1)) - { - this.destinationMgra = tour.getDestinationMGRA(); - this.destinationPurpose = tour.getPurpose(); - this.destinationIsTourDestination = true; - } else - { - // not last trip on outbound journey, destination is next stop - this.destinationMgra = stops[stop.getId() + 1].getMgra(); - this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); - } - - // the period for the trip is the origin for the trip - if (stop.getId() == 0) this.period = (byte) tour.getDepartTime(); - else this.period = (byte) stops[stop.getId() - 1].getStopPeriod(); - - } else if (inbound && toStop) - { - // inbound, trip is to stop from either tour destination or last - // stop. - - // first inbound trip; origin is tour destination - if (stop.getId() == 0) - { - this.originMgra = tour.getDestinationMGRA(); - this.originPurpose = tour.getPurpose(); - this.originIsTourDestination = true; - } else - { - // not first inbound trip; origin is last stop - this.originMgra = stops[stop.getId() - 1].getMgra(); // last - // stop - // location - this.originPurpose = stops[stop.getId() - 1].getPurpose(); - } - - // the period for the trip is the destination for the trip - if (stop.getId() == stops.length - 1) this.period = (byte) tour.getArriveTime(); - else this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); - } else - { - // inbound, trip is from stop to either next stop or tour origin. - - // last trip, destination is back to tour origin - if (stop.getId() == (stops.length - 1)) - { - this.destinationMgra = tour.getOriginMGRA(); - this.destinationPurpose = -1; - this.period = (byte) tour.getArriveTime(); - } else - { - // not last trip, destination is next stop - this.destinationMgra = stops[stop.getId() + 1].getMgra(); - this.destinationPurpose = stops[stop.getId() + 1].getPurpose(); - this.period = (byte) stops[stop.getId() + 1].getStopPeriod(); - } - } - - // code period for first trip on tour - if (toStop && !inbound && stop.getId() == 0) - { - this.firstTrip = true; - this.lastTrip = false; - this.period = (byte) tour.getDepartTime(); - } - // code period for last trip on tour - if (!toStop && inbound && stop.getId() == (stops.length - 1)) - { - this.firstTrip = false; - this.lastTrip = true; - this.period = (byte) tour.getArriveTime(); - } - - } - - /** - * @return the period - */ - public byte getPeriod() - { - return period; - } - - /** - * @param period - * the period to set - */ - public void setPeriod(byte period) - { - this.period = period; - } - - /** - * @return the origin purpose - */ - public byte getOriginPurpose() - { - return originPurpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setOriginPurpose(byte purpose) - { - this.originPurpose = purpose; - } - - /** - * @return the destination purpose - */ - public byte getDestinationPurpose() - { - return destinationPurpose; - } - - /** - * @param purpose - * the purpose to set - */ - public void setDestinationPurpose(byte purpose) - { - this.destinationPurpose = purpose; - } - - /** - * @return the originMgra - */ - public int getOriginMgra() - { - return originMgra; - } - - /** - * @param originMgra - * the originMgra to set - */ - public void setOriginMgra(int originMgra) - { - this.originMgra = originMgra; - } - - /** - * @return the destinationMgra - */ - public int getDestinationMgra() - { - return destinationMgra; - } - - /** - * @param destinationMgra - * the destinationMgra to set - */ - public void setDestinationMgra(int destinationMgra) - { - this.destinationMgra = destinationMgra; - } - - /** - * @return the tripMode - */ - public int getTripMode() - { - return tripMode; - } - - /** - * @param tripMode - * the tripMode to set - */ - public void setTripMode(int tripMode) - { - this.tripMode = tripMode; - } - - public int getBoardTap() { - return boardTap; - } - - public void setBoardTap(int boardTap) { - this.boardTap = boardTap; - } - - public int getAlightTap() { - return alightTap; - } - - public void setAlightTap(int alightTap) { - this.alightTap = alightTap; - } - - public int getSet() { - return set; - } - - public void setSet(int set) { - this.set = set; - } - - /** - * @return the inbound - */ - public boolean isInbound() - { - return inbound; - } - - /** - * @param inbound - * the inbound to set - */ - public void setInbound(boolean inbound) - { - this.inbound = inbound; - } - - /** - * @return the firstTrip - */ - public boolean isFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(boolean firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public boolean isLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(boolean lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the originIsTourDestination - */ - public boolean isOriginIsTourDestination() - { - return originIsTourDestination; - } - - /** - * @param originIsTourDestination - * the originIsTourDestination to set - */ - public void setOriginIsTourDestination(boolean originIsTourDestination) - { - this.originIsTourDestination = originIsTourDestination; - } - - /** - * @return the destinationIsTourDestination - */ - public boolean isDestinationIsTourDestination() - { - return destinationIsTourDestination; - } - - /** - * @param destinationIsTourDestination - * the destinationIsTourDestination to set - */ - public void setDestinationIsTourDestination(boolean destinationIsTourDestination) - { - this.destinationIsTourDestination = destinationIsTourDestination; - } - - public float getValueOfTime() { - return valueOfTime; - } - - public void setValueOfTime(float valueOfTime) { - this.valueOfTime = valueOfTime; - } - public float getParkingCost() { - return parkingCost; - } - - public void setParkingCost(float parkingCost) { - this.parkingCost = parkingCost; - } - - public void setMicromobilityWalkMode(byte micromobilityWalkMode) { - this.micromobilityWalkMode=micromobilityWalkMode; - } - - public byte getMicromobilityWalkMode() { - return micromobilityWalkMode; - } - public float getMicromobilityWalkLogsum() { - return micromobilityWalkLogsum; - } - - public void setMicromobilityWalkLogsum(float micromobilityWalkLogsum) { - this.micromobilityWalkLogsum = micromobilityWalkLogsum; - } - - public byte getMicromobilityAccessMode() { - return micromobilityAccessMode; - } - - public void setMicromobilityAccessMode(byte micromobilityAccessMode) { - this.micromobilityAccessMode = micromobilityAccessMode; - } - - public byte getMicromobilityEgressMode() { - return micromobilityEgressMode; - } - - public void setMicromobilityEgressMode(byte micromobilityEgressMode) { - this.micromobilityEgressMode = micromobilityEgressMode; - } - - public float getMicromobilityAccessLogsum() { - return micromobilityAccessLogsum; - } - - public void setMicromobilityAccessLogsum(float micromobilityAccessLogsum) { - this.micromobilityAccessLogsum = micromobilityAccessLogsum; - } - - public float getMicromobilityEgressLogsum() { - return micromobilityEgressLogsum; - } - - public void setMicromobilityEgressLogsum(float micromobilityEgressLogsum) { - this.micromobilityEgressLogsum = micromobilityEgressLogsum; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java deleted file mode 100644 index c984730..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceDMU.java +++ /dev/null @@ -1,883 +0,0 @@ -package org.sandag.abm.visitor; - -import java.io.Serializable; -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.internalexternal.InternalExternalTripModeChoiceDMU; - -import com.pb.common.calculator.IndexValues; -import com.pb.common.calculator.VariableTable; - -public class VisitorTripModeChoiceDMU - implements Serializable, VariableTable -{ - - protected transient Logger logger = Logger.getLogger(InternalExternalTripModeChoiceDMU.class); - - protected HashMap methodIndexMap; - protected IndexValues dmuIndex; - - protected int tourDepartPeriod; - protected int tourArrivePeriod; - protected int tripPeriod; - protected int outboundStops; - protected int returnStops; - protected int firstTrip; - protected int lastTrip; - protected int tourPurpose; - protected int segment; - protected int partySize; - protected int autoAvailable; - protected int income; - - // tour mode - protected int tourModeIsDA; - protected int tourModeIsS2; - protected int tourModeIsS3; - protected int tourModeIsWalk; - protected int tourModeIsBike; - protected int tourModeIsWalkTransit; - protected int tourModeIsPNRTransit; - protected int tourModeIsKNRTransit; - protected int tourModeIsMaas; - protected int tourModeIsTNCTransit; - - - protected float hourlyParkingCostTourDest; - protected float dailyParkingCostTourDest; - protected float monthlyParkingCostTourDest; - protected int tripOrigIsTourDest; - protected int tripDestIsTourDest; - protected float hourlyParkingCostTripOrig; - protected float hourlyParkingCostTripDest; - - protected double nmWalkTime; - protected double nmBikeTime; - - protected double ivtCoeff; - protected double costCoeff; - protected double walkTransitLogsum; - protected double pnrTransitLogsum; - protected double knrTransitLogsum; - - protected float waitTimeTaxi; - protected float waitTimeSingleTNC; - protected float waitTimeSharedTNC; - - - protected int outboundHalfTourDirection; - - public VisitorTripModeChoiceDMU(VisitorModelStructure modelStructure, Logger aLogger) - { - if (aLogger == null) aLogger = Logger.getLogger("visitorModel"); - logger = aLogger; - setupMethodIndexMap(); - dmuIndex = new IndexValues(); - - } - - /** - * Set this index values for this tour mode choice DMU object. - * - * @param hhIndex - * is the DMU household index - * @param zoneIndex - * is the DMU zone index - * @param origIndex - * is the DMU origin index - * @param destIndex - * is the DMU desatination index - */ - public void setDmuIndexValues(int hhIndex, int zoneIndex, int origIndex, int destIndex, - boolean debug) - { - dmuIndex.setHHIndex(hhIndex); - dmuIndex.setZoneIndex(zoneIndex); - dmuIndex.setOriginZone(origIndex); - dmuIndex.setDestZone(destIndex); - - dmuIndex.setDebug(false); - dmuIndex.setDebugLabel(""); - if (debug) - { - dmuIndex.setDebug(true); - dmuIndex.setDebugLabel("Debug MC UEC"); - } - - } - - public IndexValues getDmuIndexValues() - { - return dmuIndex; - } - - /** - * @return the tripPeriod - */ - public int getTripPeriod() - { - return tripPeriod; - } - - /** - * @param tripPeriod - * the tripPeriod to set - */ - public void setTripPeriod(int tripPeriod) - { - this.tripPeriod = tripPeriod; - } - - /** - * @return the outboundStops - */ - public int getOutboundStops() - { - return outboundStops; - } - - /** - * @param outboundStops - * the outboundStops to set - */ - public void setOutboundStops(int outboundStops) - { - this.outboundStops = outboundStops; - } - - /** - * @return the returnStops - */ - public int getReturnStops() - { - return returnStops; - } - - /** - * @param returnStops - * the returnStops to set - */ - public void setReturnStops(int returnStops) - { - this.returnStops = returnStops; - } - - /** - * @return the firstTrip - */ - public int getFirstTrip() - { - return firstTrip; - } - - /** - * @param firstTrip - * the firstTrip to set - */ - public void setFirstTrip(int firstTrip) - { - this.firstTrip = firstTrip; - } - - /** - * @return the lastTrip - */ - public int getLastTrip() - { - return lastTrip; - } - - /** - * @param lastTrip - * the lastTrip to set - */ - public void setLastTrip(int lastTrip) - { - this.lastTrip = lastTrip; - } - - /** - * @return the tourModeIsDA - */ - public int getTourModeIsDA() - { - return tourModeIsDA; - } - - /** - * @param tourModeIsDA - * the tourModeIsDA to set - */ - public void setTourModeIsDA(int tourModeIsDA) - { - this.tourModeIsDA = tourModeIsDA; - } - - /** - * @return the tourModeIsS2 - */ - public int getTourModeIsS2() - { - return tourModeIsS2; - } - - /** - * @param tourModeIsS2 - * the tourModeIsS2 to set - */ - public void setTourModeIsS2(int tourModeIsS2) - { - this.tourModeIsS2 = tourModeIsS2; - } - - /** - * @return the tourModeIsS3 - */ - public int getTourModeIsS3() - { - return tourModeIsS3; - } - - /** - * @param tourModeIsS3 - * the tourModeIsS3 to set - */ - public void setTourModeIsS3(int tourModeIsS3) - { - this.tourModeIsS3 = tourModeIsS3; - } - - /** - * @return the tourModeIsWalk - */ - public int getTourModeIsWalk() - { - return tourModeIsWalk; - } - - /** - * @param tourModeIsWalk - * the tourModeIsWalk to set - */ - public void setTourModeIsWalk(int tourModeIsWalk) - { - this.tourModeIsWalk = tourModeIsWalk; - } - - /** - * @return the tourModeIsBike - */ - public int getTourModeIsBike() - { - return tourModeIsBike; - } - - /** - * @param tourModeIsBike - * the tourModeIsBike to set - */ - public void setTourModeIsBike(int tourModeIsBike) - { - this.tourModeIsBike = tourModeIsBike; - } - - /** - * @return the tourModeIsWalkTransit - */ - public int getTourModeIsWalkTransit() - { - return tourModeIsWalkTransit; - } - - /** - * @param tourModeIsWalkTransit - * the tourModeIsWalkTransit to set - */ - public void setTourModeIsWalkTransit(int tourModeIsWalkTransit) - { - this.tourModeIsWalkTransit = tourModeIsWalkTransit; - } - - /** - * @return the tourModeIsPNRTransit - */ - public int getTourModeIsPNRTransit() - { - return tourModeIsPNRTransit; - } - - /** - * @param tourModeIsPNRTransit - * the tourModeIsPNRTransit to set - */ - public void setTourModeIsPNRTransit(int tourModeIsPNRTransit) - { - this.tourModeIsPNRTransit = tourModeIsPNRTransit; - } - - /** - * @return the tourModeIsKNRTransit - */ - public int getTourModeIsKNRTransit() - { - return tourModeIsKNRTransit; - } - - /** - * @param tourModeIsKNRTransit - * the tourModeIsKNRTransit to set - */ - public void setTourModeIsKNRTransit(int tourModeIsKNRTransit) - { - this.tourModeIsKNRTransit = tourModeIsKNRTransit; - } - - /** - * @return the tourModeIsMaas - */ - public int getTourModeIsMaas() - { - return tourModeIsMaas; - } - - /** - * @param tourModeIsMaas - * the tourModeIsMaas to set - */ - public void setTourModeIsMaas(int tourModeIsMaas) - { - this.tourModeIsMaas = tourModeIsMaas; - } - - /** - * @return the hourlyParkingCostTourDest - */ - public float getHourlyParkingCostTourDest() - { - return hourlyParkingCostTourDest; - } - - /** - * @param hourlyParkingCostTourDest - * the hourlyParkingCostTourDest to set - */ - public void setHourlyParkingCostTourDest(float hourlyParkingCostTourDest) - { - this.hourlyParkingCostTourDest = hourlyParkingCostTourDest; - } - - /** - * @return the dailyParkingCostTourDest - */ - public float getDailyParkingCostTourDest() - { - return dailyParkingCostTourDest; - } - - /** - * @param dailyParkingCostTourDest - * the dailyParkingCostTourDest to set - */ - public void setDailyParkingCostTourDest(float dailyParkingCostTourDest) - { - this.dailyParkingCostTourDest = dailyParkingCostTourDest; - } - - /** - * @return the monthlyParkingCostTourDest - */ - public float getMonthlyParkingCostTourDest() - { - return monthlyParkingCostTourDest; - } - - /** - * @param monthlyParkingCostTourDest - * the monthlyParkingCostTourDest to set - */ - public void setMonthlyParkingCostTourDest(float monthlyParkingCostTourDest) - { - this.monthlyParkingCostTourDest = monthlyParkingCostTourDest; - } - - /** - * @return the tripOrigIsTourDest - */ - public int getTripOrigIsTourDest() - { - return tripOrigIsTourDest; - } - - /** - * @param tripOrigIsTourDest - * the tripOrigIsTourDest to set - */ - public void setTripOrigIsTourDest(int tripOrigIsTourDest) - { - this.tripOrigIsTourDest = tripOrigIsTourDest; - } - - /** - * @return the tripDestIsTourDest - */ - public int getTripDestIsTourDest() - { - return tripDestIsTourDest; - } - - /** - * @param tripDestIsTourDest - * the tripDestIsTourDest to set - */ - public void setTripDestIsTourDest(int tripDestIsTourDest) - { - this.tripDestIsTourDest = tripDestIsTourDest; - } - - /** - * @return the hourlyParkingCostTripOrig - */ - public float getHourlyParkingCostTripOrig() - { - return hourlyParkingCostTripOrig; - } - - /** - * @param hourlyParkingCostTripOrig - * the hourlyParkingCostTripOrig to set - */ - public void setHourlyParkingCostTripOrig(float hourlyParkingCostTripOrig) - { - this.hourlyParkingCostTripOrig = hourlyParkingCostTripOrig; - } - - /** - * @return the hourlyParkingCostTripDest - */ - public float getHourlyParkingCostTripDest() - { - return hourlyParkingCostTripDest; - } - - /** - * @param hourlyParkingCostTripDest - * the hourlyParkingCostTripDest to set - */ - public void setHourlyParkingCostTripDest(float hourlyParkingCostTripDest) - { - this.hourlyParkingCostTripDest = hourlyParkingCostTripDest; - } - - /** - * @return the outboundHalfTourDirection - */ - public int getOutboundHalfTourDirection() - { - return outboundHalfTourDirection; - } - - /** - * @param outboundHalfTourDirection - * the outboundHalfTourDirection to set - */ - public void setOutboundHalfTourDirection(int outboundHalfTourDirection) - { - this.outboundHalfTourDirection = outboundHalfTourDirection; - } - - /** - * @return the tourDepartPeriod - */ - public int getTourDepartPeriod() - { - return tourDepartPeriod; - } - - /** - * @param tourDepartPeriod - * the tourDepartPeriod to set - */ - public void setTourDepartPeriod(int tourDepartPeriod) - { - this.tourDepartPeriod = tourDepartPeriod; - } - - /** - * @param tourArrivePeriod - * the tourArrivePeriod to set - */ - public void setTourArrivePeriod(int tourArrivePeriod) - { - this.tourArrivePeriod = tourArrivePeriod; - } - - /** - * @return the tourArrivePeriod - */ - public int getTourArrivePeriod() - { - return tourArrivePeriod; - } - - public double getNm_walkTime() - { - return nmWalkTime; - } - - public void setNonMotorizedWalkTime(double nmWalkTime) - { - this.nmWalkTime = nmWalkTime; - } - - public void setNonMotorizedBikeTime(double nmBikeTime) - { - this.nmBikeTime = nmBikeTime; - } - - public double getNm_bikeTime() - { - return nmBikeTime; - } - - /** - * @return the tourPurpose - */ - public int getTourPurpose() - { - return tourPurpose; - } - - /** - * @param tourPurpose - * the tourPurpose to set - */ - public void setTourPurpose(int tourPurpose) - { - this.tourPurpose = tourPurpose; - } - - /** - * @return the segment - */ - public int getSegment() - { - return segment; - } - - /** - * @param segment - * the segment to set - */ - public void setSegment(int segment) - { - this.segment = segment; - } - - public int getPartySize() - { - return partySize; - } - - public void setPartySize(int partySize) - { - this.partySize = partySize; - } - - public int getAutoAvailable() - { - return autoAvailable; - } - - public void setAutoAvailable(int autoAvailable) - { - this.autoAvailable = autoAvailable; - } - - public int getIncome() - { - return income; - } - - public void setIncome(int income) - { - this.income = income; - } - public double getIvtCoeff() { - return ivtCoeff; - } - - public void setIvtCoeff(double ivtCoeff) { - this.ivtCoeff = ivtCoeff; - } - - public double getCostCoeff() { - return costCoeff; - } - - public void setCostCoeff(double costCoeff) { - this.costCoeff = costCoeff; - } - - public double getWalkTransitLogsum() { - return walkTransitLogsum; - } - - public void setWalkTransitLogsum(double walkTransitLogsum) { - this.walkTransitLogsum = walkTransitLogsum; - } - - public double getPnrTransitLogsum() { - return pnrTransitLogsum; - } - - public void setPnrTransitLogsum(double pnrTransitLogsum) { - this.pnrTransitLogsum = pnrTransitLogsum; - } - - public double getKnrTransitLogsum() { - return knrTransitLogsum; - } - - public void setKnrTransitLogsum(double knrTransitLogsum) { - this.knrTransitLogsum = knrTransitLogsum; - } - - public float getWaitTimeTaxi() { - return waitTimeTaxi; - } - - public void setWaitTimeTaxi(float waitTimeTaxi) { - this.waitTimeTaxi = waitTimeTaxi; - } - - public float getWaitTimeSingleTNC() { - return waitTimeSingleTNC; - } - - public void setWaitTimeSingleTNC(float waitTimeSingleTNC) { - this.waitTimeSingleTNC = waitTimeSingleTNC; - } - - public float getWaitTimeSharedTNC() { - return waitTimeSharedTNC; - } - - public void setWaitTimeSharedTNC(float waitTimeSharedTNC) { - this.waitTimeSharedTNC = waitTimeSharedTNC; - } - - - private void setupMethodIndexMap() - { - methodIndexMap = new HashMap(); - - methodIndexMap.put("getTourDepartPeriod", 0); - methodIndexMap.put("getTourArrivePeriod", 1); - methodIndexMap.put("getTripPeriod", 2); - methodIndexMap.put("getSegment", 3); - methodIndexMap.put("getTourPurpose", 4); - methodIndexMap.put("getOutboundStops", 5); - methodIndexMap.put("getReturnStops", 6); - methodIndexMap.put("getFirstTrip", 7); - methodIndexMap.put("getLastTrip", 8); - methodIndexMap.put("getTourModeIsDA", 9); - methodIndexMap.put("getTourModeIsS2", 10); - methodIndexMap.put("getTourModeIsS3", 11); - methodIndexMap.put("getTourModeIsWalk", 12); - methodIndexMap.put("getTourModeIsBike", 13); - methodIndexMap.put("getTourModeIsWalkTransit", 14); - methodIndexMap.put("getTourModeIsPNRTransit", 15); - methodIndexMap.put("getTourModeIsKNRTransit", 16); - methodIndexMap.put("getTourModeIsMaas", 17); - methodIndexMap.put("getTourModeIsTNCTransit", 18); - - methodIndexMap.put("getHourlyParkingCostTourDest", 20); - methodIndexMap.put("getDailyParkingCostTourDest", 21); - methodIndexMap.put("getMonthlyParkingCostTourDest", 22); - methodIndexMap.put("getTripOrigIsTourDest", 23); - methodIndexMap.put("getTripDestIsTourDest", 24); - methodIndexMap.put("getHourlyParkingCostTripOrig", 25); - methodIndexMap.put("getHourlyParkingCostTripDest", 26); - - methodIndexMap.put("getPartySize", 30); - methodIndexMap.put("getAutoAvailable", 31); - methodIndexMap.put("getIncome", 32); - - methodIndexMap.put("getIvtCoeff", 60); - methodIndexMap.put("getCostCoeff", 61); - - methodIndexMap.put("getWalkSetLogSum", 62); - methodIndexMap.put("getPnrSetLogSum", 63); - methodIndexMap.put("getKnrSetLogSum", 64); - - methodIndexMap.put("getWaitTimeTaxi", 70); - methodIndexMap.put("getWaitTimeSingleTNC", 71); - methodIndexMap.put("getWaitTimeSharedTNC", 72); - - methodIndexMap.put("getNm_walkTime", 90); - methodIndexMap.put("getNm_bikeTime", 91); - - } - - public double getValueForIndex(int variableIndex, int arrayIndex) - { - - double returnValue = -1; - - switch (variableIndex) - { - case 0: - returnValue = getTourDepartPeriod(); - break; - case 1: - returnValue = getTourArrivePeriod(); - break; - case 2: - returnValue = getTripPeriod(); - break; - case 3: - returnValue = getSegment(); - break; - case 4: - returnValue = getTourPurpose(); - break; - case 5: - returnValue = getOutboundStops(); - break; - case 6: - returnValue = getReturnStops(); - break; - case 7: - returnValue = getFirstTrip(); - break; - case 8: - returnValue = getLastTrip(); - break; - case 9: - returnValue = getTourModeIsDA(); - break; - case 10: - returnValue = getTourModeIsS2(); - break; - case 11: - returnValue = getTourModeIsS3(); - break; - case 12: - returnValue = getTourModeIsWalk(); - break; - case 13: - returnValue = getTourModeIsBike(); - break; - case 14: - returnValue = getTourModeIsWalkTransit(); - break; - case 15: - returnValue = getTourModeIsPNRTransit(); - break; - case 16: - returnValue = getTourModeIsKNRTransit(); - break; - case 17: - returnValue = getTourModeIsMaas(); - break; - case 18: - returnValue = getTourModeIsTNCTransit(); - break; - case 20: - returnValue = getHourlyParkingCostTourDest(); - break; - case 21: - returnValue = getDailyParkingCostTourDest(); - break; - case 22: - returnValue = getMonthlyParkingCostTourDest(); - break; - case 23: - returnValue = getTripOrigIsTourDest(); - break; - case 24: - returnValue = getTripDestIsTourDest(); - break; - case 25: - returnValue = getHourlyParkingCostTripOrig(); - break; - case 26: - returnValue = getHourlyParkingCostTripDest(); - break; - case 30: - returnValue = getPartySize(); - break; - case 31: - returnValue = getAutoAvailable(); - break; - case 32: - returnValue = getIncome(); - break; - case 60: - returnValue = getIvtCoeff(); - break; - case 61: - returnValue = getCostCoeff(); - break; - case 62: - returnValue = getWalkTransitLogsum(); - break; - case 63: - returnValue = getPnrTransitLogsum(); - break; - case 64: - returnValue = getKnrTransitLogsum(); - break; - case 70: return getWaitTimeTaxi(); - case 71: return getWaitTimeSingleTNC(); - case 72: return getWaitTimeSharedTNC(); - case 90: - returnValue = getNm_walkTime(); - break; - case 91: - returnValue = getNm_bikeTime(); - break; - default: - logger.error( "method number = " + variableIndex + " not found" ); - throw new RuntimeException( "method number = " + variableIndex + " not found" ); - } - return returnValue; - - } - - public int getIndexValue(String variableName) - { - return methodIndexMap.get(variableName); - } - - public int getAssignmentIndexValue(String variableName) - { - throw new UnsupportedOperationException(); - } - - public double getValueForIndex(int variableIndex) - { - throw new UnsupportedOperationException(); - } - - public void setValue(String variableName, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public void setValue(int variableIndex, double variableValue) - { - throw new UnsupportedOperationException(); - } - - public int getTourModeIsTNCTransit() { - return tourModeIsTNCTransit; - } - - public void setTourModeIsTNCTransit(int tourModeIsTNCTransit) { - this.tourModeIsTNCTransit = tourModeIsTNCTransit; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java deleted file mode 100644 index bd2196c..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripModeChoiceModel.java +++ /dev/null @@ -1,327 +0,0 @@ -package org.sandag.abm.visitor; - -import java.util.HashMap; - -import org.apache.log4j.Logger; -import org.sandag.abm.accessibilities.AutoAndNonMotorizedSkimsCalculator; -import org.sandag.abm.accessibilities.AutoTazSkimsCalculator; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.McLogsumsCalculator; -import org.sandag.abm.ctramp.TripModeChoiceDMU; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.calculator.VariableTable; -import com.pb.common.newmodel.ChoiceModelApplication; -import com.pb.common.newmodel.UtilityExpressionCalculator; - -public class VisitorTripModeChoiceModel -{ - - private transient Logger logger = Logger.getLogger("visitorModel"); - - private AutoAndNonMotorizedSkimsCalculator anm; - private McLogsumsCalculator logsumHelper; - private VisitorModelStructure modelStructure; - private TazDataManager tazs; - private MgraDataManager mgraManager; - private double[] lsWgtAvgCostM; - private double[] lsWgtAvgCostD; - private double[] lsWgtAvgCostH; - private VisitorTripModeChoiceDMU dmu; - private ChoiceModelApplication tripModeChoiceModel; - double logsum = 0; - - private static final String PROPERTIES_UEC_DATA_SHEET = "visitor.trip.mc.data.page"; - private static final String PROPERTIES_UEC_MODEL_SHEET = "visitor.trip.mc.model.page"; - private static final String PROPERTIES_UEC_FILE = "visitor.trip.mc.uec.file"; - private TripModeChoiceDMU mcDmuObject; - private AutoTazSkimsCalculator tazDistanceCalculator; - - /** - * Constructor. - * - * @param propertyMap - * @param myModelStructure - * @param dmuFactory - * @param myLogsumHelper - */ - public VisitorTripModeChoiceModel(HashMap propertyMap, - VisitorModelStructure myModelStructure, VisitorDmuFactoryIf dmuFactory, AutoTazSkimsCalculator tazDistanceCalculator) - { - tazs = TazDataManager.getInstance(propertyMap); - mgraManager = MgraDataManager.getInstance(propertyMap); - - lsWgtAvgCostM = mgraManager.getLsWgtAvgCostM(); - lsWgtAvgCostD = mgraManager.getLsWgtAvgCostD(); - lsWgtAvgCostH = mgraManager.getLsWgtAvgCostH(); - - modelStructure = myModelStructure; - this.tazDistanceCalculator = tazDistanceCalculator; - - logsumHelper = new McLogsumsCalculator(); - logsumHelper.setupSkimCalculators(propertyMap); - logsumHelper.setTazDistanceSkimArrays( - tazDistanceCalculator.getStoredFromTazToAllTazsDistanceSkims(), - tazDistanceCalculator.getStoredToTazFromAllTazsDistanceSkims()); - - SandagModelStructure modelStructure = new SandagModelStructure(); - mcDmuObject = new TripModeChoiceDMU(modelStructure, logger); - - setupTripModeChoiceModel(propertyMap, dmuFactory); - - } - - /** - * Read the UEC file and set up the trip mode choice model. - * - * @param propertyMap - * @param dmuFactory - */ - private void setupTripModeChoiceModel(HashMap propertyMap, - VisitorDmuFactoryIf dmuFactory) - { - - logger.info(String.format("setting up visitor trip mode choice model.")); - - dmu = dmuFactory.getVisitorTripModeChoiceDMU(); - - int dataPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_DATA_SHEET)); - int modelPage = new Integer(Util.getStringValueFromPropertyMap(propertyMap, - PROPERTIES_UEC_MODEL_SHEET)); - - String uecPath = propertyMap.get(CtrampApplication.PROPERTIES_UEC_PATH); - String tripModeUecFile = propertyMap.get(PROPERTIES_UEC_FILE); - tripModeUecFile = uecPath + tripModeUecFile; - - tripModeChoiceModel = new ChoiceModelApplication(tripModeUecFile, modelPage, dataPage, - propertyMap, (VariableTable) dmu); - - } - - /** - * Calculate utilities and return logsum for the tour and stop. - * - * @param tour - * @param trip - */ - public double computeUtilities(VisitorTour tour, VisitorTrip trip) - { - - setDmuAttributes(tour, trip); - - tripModeChoiceModel.computeUtilities(dmu, dmu.getDmuIndexValues()); - - if (tour.getDebugChoiceModels()) - { - tour.logTourObject(logger, 100); - tripModeChoiceModel.logUECResults(logger, "Visitor trip mode choice model"); - - } - - logsum = tripModeChoiceModel.getLogsum(); - - if (tour.getDebugChoiceModels()) logger.info("Returning logsum " + logsum); - - return logsum; - - } - - /** - * Choose a mode and store in the trip object. - * - * @param tour - * VisitorTour - * @param trip - * VisitorTrip - * - */ - public void chooseMode(VisitorTour tour, VisitorTrip trip) - { - - computeUtilities(tour, trip); - - double rand = tour.getRandom(); - try{ - int mode = tripModeChoiceModel.getChoiceResult(rand); - trip.setTripMode(mode); - - //value of time; lookup vot, votS2, or votS3 from the UEC depending on chosen mode - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - - int votIndex = uec.lookupVariableIndex("vot"); - double vot = uec.getValueForIndex(votIndex); - trip.setValueOfTime((float)vot); - - float parkingCost = getTripParkingCost(mode); - trip.setParkingCost(parkingCost); - - if(modelStructure.getTripModeIsTransit(mode)){ - double[][] bestTapPairs = null; - - if (modelStructure.getTripModeIsWalkTransit(mode)){ - bestTapPairs = logsumHelper.getBestWtwTripTaps(); - } - else if (modelStructure.getTripModeIsPnrTransit(mode)||modelStructure.getTripModeIsKnrTransit(mode)){ - if (!trip.isInbound()) - bestTapPairs = logsumHelper.getBestDtwTripTaps(); - else - bestTapPairs = logsumHelper.getBestWtdTripTaps(); - } - double rn = tour.getRandom(); - int pathIndex = logsumHelper.chooseTripPath(rn, bestTapPairs, tour.getDebugChoiceModels(), logger); - int boardTap = (int) bestTapPairs[pathIndex][0]; - int alightTap = (int) bestTapPairs[pathIndex][1]; - int set = (int) bestTapPairs[pathIndex][2]; - trip.setBoardTap(boardTap); - trip.setAlightTap(alightTap); - trip.setSet(set); - } - - }catch(Exception e){ - logger.info("Error calculating visitor trip mode choice with rand="+rand); - tour.logTourObject(logger, 100); - logger.error(e.getMessage()); - } - - } - - /** - * Return parking cost from UEC if auto trip, else return 0. - * - * @param tripMode - * @return Parking cost if auto mode, else 0 - */ - public float getTripParkingCost(int tripMode) { - - float parkingCost=0; - - if(modelStructure.getTripModeIsSovOrHov(tripMode)) { - UtilityExpressionCalculator uec = tripModeChoiceModel.getUEC(); - int parkingCostIndex = uec.lookupVariableIndex("parkingCost"); - parkingCost = (float) uec.getValueForIndex(parkingCostIndex); - return parkingCost; - } - return parkingCost; - } - - - /** - * Set DMU attributes. - * - * @param tour - * @param trip - */ - public void setDmuAttributes(VisitorTour tour, VisitorTrip trip) - { - - int tourDestinationMgra = tour.getDestinationMGRA(); - int tripOriginMgra = trip.getOriginMgra(); - int tripDestinationMgra = trip.getDestinationMgra(); - - int tripOriginTaz = mgraManager.getTaz(tripOriginMgra); - int tripDestinationTaz = mgraManager.getTaz(tripDestinationMgra); - - int tourMode = tour.getTourMode(); - - dmu.setDmuIndexValues(tripOriginTaz, tripDestinationTaz, tripOriginTaz, tripDestinationTaz, - tour.getDebugChoiceModels()); - - dmu.setTourDepartPeriod(tour.getDepartTime()); - dmu.setTourArrivePeriod(tour.getArriveTime()); - dmu.setTripPeriod(trip.getPeriod()); - - // set trip mc dmu values for transit logsum (gets replaced below by uec values) - double c_ivt = -0.03; - double c_cost = - 0.0033; - - // Solve trip mode level utilities - mcDmuObject.setIvtCoeff(c_ivt); - mcDmuObject.setCostCoeff(c_cost); - double walkTransitLogsum = -999.0; - double driveTransitLogsum = -999.0; - - logsumHelper.setNmTripMcDmuAttributes(mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - dmu.setNonMotorizedWalkTime(mcDmuObject.getNm_walkTime()); - dmu.setNonMotorizedBikeTime(mcDmuObject.getNm_bikeTime()); - - logsumHelper.setWtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(),tour.getDebugChoiceModels()); - walkTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTW); - - dmu.setWalkTransitLogsum(walkTransitLogsum); - if (!trip.isInbound()) - { - logsumHelper.setDtwTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.DTW); - } else - { - logsumHelper.setWtdTripMcDmuAttributes( mcDmuObject, trip.getOriginMgra(), trip.getDestinationMgra(), trip.getPeriod(), tour.getDebugChoiceModels()); - driveTransitLogsum = mcDmuObject.getTransitLogSum(McLogsumsCalculator.WTD); - } - - dmu.setPnrTransitLogsum(driveTransitLogsum); - dmu.setKnrTransitLogsum(driveTransitLogsum); - - dmu.setTourPurpose(tour.getPurpose()); - - dmu.setOutboundStops(tour.getNumberInboundStops()); - dmu.setReturnStops(tour.getNumberInboundStops()); - - if (trip.isFirstTrip()) dmu.setFirstTrip(1); - else dmu.setFirstTrip(0); - - if (trip.isLastTrip()) dmu.setLastTrip(1); - else dmu.setLastTrip(0); - - if (modelStructure.getTourModeIsSov(tourMode)) dmu.setTourModeIsDA(1); - else dmu.setTourModeIsDA(0); - - if (modelStructure.getTourModeIsS2(tourMode)) dmu.setTourModeIsS2(1); - else dmu.setTourModeIsS2(0); - - if (modelStructure.getTourModeIsS3(tourMode)) dmu.setTourModeIsS3(1); - else dmu.setTourModeIsS3(0); - - if (modelStructure.getTourModeIsWalk(tourMode)) dmu.setTourModeIsWalk(1); - else dmu.setTourModeIsWalk(0); - - if (modelStructure.getTourModeIsBike(tourMode)) dmu.setTourModeIsBike(1); - else dmu.setTourModeIsBike(0); - - if (modelStructure.getTourModeIsWalkTransit(tourMode)) dmu.setTourModeIsWalkTransit(1); - else dmu.setTourModeIsWalkTransit(0); - - if (modelStructure.getTourModeIsPnr(tourMode)) dmu.setTourModeIsPNRTransit(1); - else dmu.setTourModeIsPNRTransit(0); - - if (modelStructure.getTourModeIsKnr(tourMode)) dmu.setTourModeIsKNRTransit(1); - else dmu.setTourModeIsKNRTransit(0); - - if (modelStructure.getTourModeIsMaas(tourMode)) dmu.setTourModeIsMaas(1); - else dmu.setTourModeIsMaas(0); - - if (modelStructure.getTourModeIsTncTransit(tourMode)) dmu.setTourModeIsTNCTransit(1); - else dmu.setTourModeIsTNCTransit(0); - - if (trip.isOriginIsTourDestination()) dmu.setTripOrigIsTourDest(1); - else dmu.setTripOrigIsTourDest(0); - - if (trip.isDestinationIsTourDestination()) dmu.setTripDestIsTourDest(1); - else dmu.setTripDestIsTourDest(0); - - dmu.setIncome((byte) tour.getIncome()); - dmu.setAutoAvailable(tour.getAutoAvailable()); - dmu.setPartySize(tour.getNumberOfParticipants()); - - dmu.setHourlyParkingCostTourDest((float) lsWgtAvgCostH[tourDestinationMgra]); - dmu.setDailyParkingCostTourDest((float) lsWgtAvgCostD[tourDestinationMgra]); - dmu.setMonthlyParkingCostTourDest((float) lsWgtAvgCostM[tourDestinationMgra]); - dmu.setHourlyParkingCostTripOrig((float) lsWgtAvgCostH[tripOriginMgra]); - dmu.setHourlyParkingCostTripDest((float) lsWgtAvgCostH[tripDestinationMgra]); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java b/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java deleted file mode 100644 index 7a715bf..0000000 --- a/sandag_abm/src/main/java/org/sandag/abm/visitor/VisitorTripTables.java +++ /dev/null @@ -1,682 +0,0 @@ -package org.sandag.abm.visitor; - -import gnu.cajo.invoke.Remote; -import gnu.cajo.utils.ItemServer; - -import java.io.File; -import java.io.IOException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.MissingResourceException; -import java.util.ResourceBundle; - -import org.apache.log4j.Logger; -import org.sandag.abm.application.SandagModelStructure; -import org.sandag.abm.ctramp.CtrampApplication; -import org.sandag.abm.ctramp.MatrixDataServer; -import org.sandag.abm.ctramp.MatrixDataServerRmi; -import org.sandag.abm.ctramp.Util; -import org.sandag.abm.modechoice.MgraDataManager; -import org.sandag.abm.modechoice.TapDataManager; -import org.sandag.abm.modechoice.TazDataManager; - -import com.pb.common.datafile.OLD_CSVFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.MatrixWriter; -import com.pb.common.util.ResourceUtil; - -public class VisitorTripTables -{ - - private static Logger logger = Logger.getLogger("tripTables"); - public static final int MATRIX_DATA_SERVER_PORT = 1171; - - private TableDataSet tripData; - - // Some parameters - private int[] modeIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // total - // modes, - // returns - // 0=auto - // modes, - // 1=non-motor, - // 2=transit, - // 3= - // other - private int[] matrixIndex; // an - // index - // array, - // dimensioned - // by - // number - // of - // modes, - // returns - // the - // element - // of - // the - // matrix - // array - // to - // store - // value - - // array modes: AUTO, NON-MOTORIZED, TRANSIT, OTHER - private int autoModes = 0; - private int tranModes = 0; - private int nmotModes = 0; - private int othrModes = 0; - - // one file per time period - private int numberOfPeriods; - - private HashMap rbMap; - - // matrices are indexed by modes, vot bins, submodes - private Matrix[][][] matrix; - - private ResourceBundle rb; - private MgraDataManager mgraManager; - private TazDataManager tazManager; - private TapDataManager tapManager; - private SandagModelStructure modelStructure; - - private MatrixDataServerRmi ms; - private float sampleRate; - private static int iteration=1; - private static final String VOT_THRESHOLD_LOW = "valueOfTime.threshold.low"; - private static final String VOT_THRESHOLD_MED = "valueOfTime.threshold.med"; - private float valueOfTimeThresholdLow = 0; - private float valueOfTimeThresholdMed = 0; - //value of time bins by mode group - int[] votBins = {3,1,1,1}; - public int numSkimSets; - - public VisitorTripTables(HashMap rbMap) - { - - this.rbMap = rbMap; - tazManager = TazDataManager.getInstance(rbMap); - tapManager = TapDataManager.getInstance(rbMap); - mgraManager = MgraDataManager.getInstance(rbMap); - - modelStructure = new SandagModelStructure(); - - // Time period limits - numberOfPeriods = modelStructure.getNumberModelPeriods(); - - // number of modes - modeIndex = new int[modelStructure.MAXIMUM_TOUR_MODE_ALT_INDEX + 1]; - matrixIndex = new int[modeIndex.length]; - - numSkimSets = Util.getIntegerValueFromPropertyMap(rbMap,"utility.bestTransitPath.skim.sets"); - - // set the mode arrays - for (int i = 1; i < modeIndex.length; ++i) - { - if (modelStructure.getTourModeIsSovOrHov(i)) - { - modeIndex[i] = 0; - matrixIndex[i] = autoModes; - ++autoModes; - } else if (modelStructure.getTourModeIsNonMotorized(i)) - { - modeIndex[i] = 1; - matrixIndex[i] = nmotModes; - ++nmotModes; - } else if (modelStructure.getTourModeIsWalkTransit(i) - || modelStructure.getTourModeIsDriveTransit(i)) - { - modeIndex[i] = 2; - matrixIndex[i] = tranModes; - ++tranModes; - } else - { - modeIndex[i] = 3; - matrixIndex[i] = othrModes; - ++othrModes; - } - } - //value of time thresholds - valueOfTimeThresholdLow = new Float(rbMap.get(VOT_THRESHOLD_LOW)); - valueOfTimeThresholdMed = new Float(rbMap.get(VOT_THRESHOLD_MED)); - } - - /** - * Initialize all the matrices for the given time period. - * - * @param periodName - * The name of the time period. - */ - public void initializeMatrices(String periodName) - { - - /* - * This won't work because external stations aren't listed in the MGRA - * file int[] tazIndex = tazManager.getTazsOneBased(); int tazs = - * tazIndex.length-1; - */ - // Instead, use maximum taz number - int maxTaz = tazManager.getMaxTaz(); - int[] tazIndex = new int[maxTaz + 1]; - - // assume zone numbers are sequential - for (int i = 1; i < tazIndex.length; ++i) - tazIndex[i] = i; - - // get the tap index - int[] tapIndex = tapManager.getTaps(); - int taps = tapIndex.length - 1; - - // Initialize matrices; one for each mode group (auto, non-mot, tran, - // other) - // All matrices will be dimensioned by TAZs except for transit, which is - // dimensioned by TAPs - int numberOfModes = 4; - matrix = new Matrix[numberOfModes][][]; - for (int i = 0; i < numberOfModes; ++i) - { - - String modeName; - - matrix[i] = new Matrix[votBins[i]][]; - - for(int j = 0; j< votBins[i];++j){ - if (i == 0) - { - matrix[i][j] = new Matrix[autoModes]; - for (int k = 0; k < autoModes; ++k) - { - modeName = modelStructure.getModeName(k + 1); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 1) - { - matrix[i][j] = new Matrix[nmotModes]; - for (int k = 0; k < nmotModes; ++k) - { - modeName = modelStructure.getModeName(k + 1 + autoModes); - matrix[i][j][k] = new Matrix(modeName + "_" + periodName, "", maxTaz, maxTaz); - matrix[i][j][k].setExternalNumbers(tazIndex); - } - } else if (i == 2) - { - matrix[i][j] = new Matrix[tranModes*numSkimSets]; - for (int k = 0; k < tranModes; ++k) - { - for(int l=0;l1) - votBin = getValueOfTimeBin(valueOfTime); - - if (mode == 0) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + vehicleTrips)); - } else if (mode == 1) - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } else if (mode == 2) - { - - if (boardTap == 0 || alightTap == 0) continue; - - //store transit trips in matrices - mat = (matrixIndex[tripMode]*numSkimSets)+set; - float value = matrix[mode][votBin][mat].getValueAt(boardTap, alightTap); - matrix[mode][votBin][mat].setValueAt(boardTap, alightTap, (value + personTrips)); - - // Store PNR transit trips in SOV free mode skim (mode 0 mat 0) - if (modelStructure.getTourModeIsDriveTransit(tripMode)) - { - - // add the tNCVehicle trip portion to the trip table - if (!inbound) - { // from origin to lot (boarding tap) - int PNRTAZ = tapManager.getTazForTap(boardTap); - value = matrix[0][votBin][0].getValueAt(originTAZ, PNRTAZ); - matrix[0][votBin][0].setValueAt(originTAZ, PNRTAZ, (value + vehicleTrips)); - - } else - { // from lot (alighting tap) to destination - int PNRTAZ = tapManager.getTazForTap(alightTap); - value = matrix[0][votBin][0].getValueAt(PNRTAZ, destinationTAZ); - matrix[0][votBin][0].setValueAt(PNRTAZ, destinationTAZ, (value + vehicleTrips)); - } - - } - } else - { - float value = matrix[mode][votBin][mat].getValueAt(originTAZ, destinationTAZ); - matrix[mode][votBin][mat].setValueAt(originTAZ, destinationTAZ, (value + personTrips)); - } - - //logger.info("End creating trip tables for period " + timePeriod); - } - } - - /** - * Return the value of time bin 0 through 2 based on the thresholds provided in the property map - * @param valueOfTime - * @return value of time bin 0 through 2 - */ - public int getValueOfTimeBin(float valueOfTime){ - - if(valueOfTime1) - end[i][j] = "_" + per + "_"+ votBinName[j]+ ".omx"; - else - end[i][j] = "_" + per + ".omx"; - } - } - for (int i = 0; i < 4; ++i){ - for(int j = 0; j < votBins[i];++j){ - try - { - //Delete the file if it exists - File f = new File(fileName[i]+end[i][j]); - if(f.exists()){ - logger.info("Deleting existing trip file: "+fileName[i]+end[i][j]); - f.delete(); - } - if (ms != null) ms.writeMatrixFile(fileName[i]+end[i][j], matrix[i][j], mt); - else writeMatrixFile(fileName[i]+end[i][j], matrix[i][j]); - } catch (Exception e) - { - logger.error("exception caught writing " + mt.toString() + " matrix file = " - + fileName[i] +end[i][j] + ", for mode index = " + i, e); - throw new RuntimeException(); - } - } - } - - } - - /** - * Utility method to write a set of matrices to disk. - * - * @param fileName - * The file name to write to. - * @param m - * An array of matrices - */ - public void writeMatrixFile(String fileName, Matrix[] m) - { - - // auto trips - MatrixWriter writer = MatrixWriter.createWriter(fileName); - String[] names = new String[m.length]; - - for (int i = 0; i < m.length; i++) - { - names[i] = m[i].getName(); - logger.info(m[i].getName() + " has " + m[i].getRowCount() + " rows, " - + m[i].getColumnCount() + " cols, and a total of " + m[i].getSum()); - } - - writer.writeMatrices(names, m); - } - - /** - * Start matrix server - * - * @param serverAddress - * @param serverPort - * @param mt - * @return - */ - private MatrixDataServerRmi startMatrixServerProcess(String serverAddress, int serverPort, - MatrixType mt) - { - - String className = MatrixDataServer.MATRIX_DATA_SERVER_NAME; - MatrixDataServerRmi matrixServer = new MatrixDataServerRmi(serverAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - - // bind this concrete object with the cajo library objects for managing - // RMI - try - { - Remote.config(serverAddress, serverPort, null, 0); - } catch (Exception e) - { - logger.error(String.format( - "UnknownHostException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - try - { - ItemServer.bind(matrixServer, className); - } catch (RemoteException e) - { - logger.error(String.format( - "RemoteException. serverAddress = %s, serverPort = %d -- exiting.", - serverAddress, serverPort), e); - throw new RuntimeException(); - } - - return matrixServer; - - } - - /** - * @param args - */ - public static void main(String[] args) - { - - HashMap pMap; - String propertiesFile = null; - - logger.info(String.format( - "SANDAG Visitor Model Trip Table Generation Program using CT-RAMP version %s", - CtrampApplication.VERSION)); - - if (args.length == 0) - { - logger.error(String - .format("no properties file base name (without .properties extension) was specified as an argument.")); - return; - } else propertiesFile = args[0]; - - float sampleRate = 1.0f; - for (int i = 1; i < args.length; ++i) - { - if (args[i].equalsIgnoreCase("-sampleRate")) - { - sampleRate = Float.parseFloat(args[i + 1]); - } - if (args[i].equalsIgnoreCase("-iteration")) - { - iteration = Integer.parseInt(args[i + 1]); - } - } - logger.info("Visitor Model Trip Table:"+String.format("-sampleRate %.4f.", sampleRate)+"-iteration " + iteration); - pMap = ResourceUtil.getResourceBundleAsHashMap(propertiesFile); - VisitorTripTables tripTables = new VisitorTripTables(pMap); - tripTables.setSampleRate(sampleRate); - - String matrixServerAddress = ""; - int serverPort = 0; - try - { - // get matrix server address. if "none" is specified, no server will - // be - // started, and matrix io will ocurr within the current process. - matrixServerAddress = Util.getStringValueFromPropertyMap(pMap, - "RunModel.MatrixServerAddress"); - try - { - // get matrix server port. - serverPort = Util.getIntegerValueFromPropertyMap(pMap, "RunModel.MatrixServerPort"); - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, leave undefined - // -- - // it's eithe not needed or show could create an error. - } - } catch (MissingResourceException e) - { - // if no matrix server address entry is found, set to localhost, and - // a - // separate matrix io process will be started on localhost. - matrixServerAddress = "localhost"; - serverPort = MATRIX_DATA_SERVER_PORT; - } - - MatrixDataServerRmi matrixServer = null; - String matrixTypeName = Util.getStringValueFromPropertyMap(pMap, "Results.MatrixType"); - MatrixType mt = MatrixType.lookUpMatrixType(matrixTypeName); - - try - { - - if (!matrixServerAddress.equalsIgnoreCase("none")) - { - - if (matrixServerAddress.equalsIgnoreCase("localhost")) - { - matrixServer = tripTables.startMatrixServerProcess(matrixServerAddress, - serverPort, mt); - tripTables.ms = matrixServer; - } else - { - tripTables.ms = new MatrixDataServerRmi(matrixServerAddress, serverPort, - MatrixDataServer.MATRIX_DATA_SERVER_NAME); - tripTables.ms.testRemote("VisitorTripTables"); - - // mdm = MatrixDataManager.getInstance(); - // mdm.setMatrixDataServerObject(ms); - } - - } - - } catch (Exception e) - { - - logger.error( - String.format("exception caught running ctramp model components -- exiting."), - e); - throw new RuntimeException(); - - } - - tripTables.createTripTables(mt); - - } - - /** - * @return the sampleRate - */ - public double getSampleRate() - { - return sampleRate; - } - - /** - * @param sampleRate - * the sampleRate to set - */ - public void setSampleRate(float sampleRate) - { - this.sampleRate = sampleRate; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java deleted file mode 100644 index d6cda4b..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/AlternativeUsesMatrices.java +++ /dev/null @@ -1,42 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public interface AlternativeUsesMatrices extends CodedAlternative { - - void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError; - - - /** - * Method readMatrices. - * @param matrixCacheReader - */ - void readMatrices(MatrixCacheReader matrixCacheReader); - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java deleted file mode 100644 index f69c70c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ChangingTravelAttributeGetter.java +++ /dev/null @@ -1,52 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -/* - * An interface that determines the travel time between two points, given a time (represented by a double) and - * a tNCVehicle type (represented by a char). - */ -package org.sandag.cvm.activityTravel; - -/** - * @author jabraham - * - * An interface that determines a travel attribute between two points, given a time (represented by a double) and - * a tNCVehicle type (represented by a char). For instance if there are matrices of travel time for peak and off-peak, - * and implementation of this interface would be able to determine whether time was in the peak or off-peak, and - * then return the travel conditions appropriately. - */ -public interface ChangingTravelAttributeGetter { - /** - * - * This method returns the travel attribute associated with travelling from origin to destination at time time by - * tNCVehicle type vehicleType - * - * @param origin - * @param destination - * @param time - * @param vehicleType - * @return - */ - public abstract double getTravelAttribute( - int origin, - int destination, - double time, - char vehicleType); -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java deleted file mode 100644 index 738846d..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CodedAlternative.java +++ /dev/null @@ -1,35 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - - -import org.sandag.cvm.common.model.Alternative; - - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public interface CodedAlternative extends Alternative { - - public String getCode(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java deleted file mode 100644 index fc937a4..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/CoefficientFormatError.java +++ /dev/null @@ -1,61 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CoefficientFormatError extends Exception { - - /** - * Constructor for CoefficientFormatError. - */ - public CoefficientFormatError() { - super(); - } - - /** - * Constructor for CoefficientFormatError. - * @param message - */ - public CoefficientFormatError(String message) { - super(message); - } - - /** - * Constructor for CoefficientFormatError. - * @param message - * @param cause - */ - public CoefficientFormatError(String message, Throwable cause) { - super(message, cause); - } - - /** - * Constructor for CoefficientFormatError. - * @param cause - */ - public CoefficientFormatError(Throwable cause) { - super(cause); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java deleted file mode 100644 index 9009825..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/DurationModel.java +++ /dev/null @@ -1,91 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -//import org.sandag.cvm.calgary.commercial.GenerateCommercialTours; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class DurationModel implements ModelUsesMatrices, RealNumberDistribution { - - protected double a=0; - protected double b=0; - protected double c=0; - protected double d=0; - protected double e=0; - protected double f=0; - - - /** @return stop duration in hours - */ - public double sampleValue() { - double x = -Math.random(); - double y = a*Math.exp(b*x)+c*Math.exp(d*x)+f; - if (y<0) y=0; - if (y>24) y = 24; - return y; - } - - /** - * Method addCoefficient. - * @param alternative - * @param index1 - * @param index2 - * @param matrix - * @param coefficient - */ - public void addCoefficient ( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError { - if (index1.equals("a")) a = coefficient; - else if(index1.equals("b")) b = coefficient; - else if(index1.equals("c")) c = coefficient; - else if(index1.equals("d")) d = coefficient; - else if (index1.equals("e")) e = coefficient; - else if(index1.equals("f")) f = coefficient; - else throw new CoefficientFormatError("Duration model coefficients must have index1 as a,b,c or d"); - } - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader matrixReader) {} - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { -// readMatrices(GenerateCommercialTours.matrixReader); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java deleted file mode 100644 index 9449313..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/HouseholdInterface.java +++ /dev/null @@ -1,46 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on 24-Mar-2005 - * - */ -package org.sandag.cvm.activityTravel; - -import java.util.Collection; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public interface HouseholdInterface { - - /** - * @return the ID of the household - */ - public int getId(); - - /** - * @return a collection containing the Persons in the household - */ - public Collection getPersons(); - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java deleted file mode 100644 index 6bd9ac2..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/LoggingStopAlternative.java +++ /dev/null @@ -1,79 +0,0 @@ -package org.sandag.cvm.activityTravel; - -import org.apache.log4j.Logger; - -public class LoggingStopAlternative extends StopAlternative { - - - static Logger logger = Logger.getLogger(LoggingStopAlternative.class); - static double[] co = new double[10]; - - static boolean loggedParts = false; - -// @Override - public double getUtility() { - if (!loggedParts) { - logger.info("Parts of utility function are travel,destination,returnHomeTravel,returnHomeDisutility,travelDisutility,returnHomeTime,travelTime,angle,zoneType,sizeTerm"); - loggedParts=true; - } - - co[0] = c.travelUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); - co[1] = c.destinationUtilityFunction.calcForIndex(location,1); - // TODO should be logsum of trip mode - co[2] = c.returnHomeUtilityFunction.calcForIndex(location,c.myTour.getOriginZone()); - if (c.disutilityToOriginCoefficient!=0) { - co[3] = c.disutilityToOriginCoefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - } else co[3]=0; - if (c.disutilityToNextStopCoefficient!=0) { - co[4] = c.disutilityToNextStopCoefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - } else co[4] = 0; - if (c.timeToOriginCoefficient!=0) { - double timeToOriginUtility = c.timeToOriginCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - // Doug and Kevin Hack of Jan 5 2004 -// if (myTour.getTotalElapsedTime()>240.0) timeToOriginUtility*=3; - co[5] = timeToOriginUtility; - } else co[5]=0; - if (c.timeToNextStopCoefficient!=0) { - double timeToNextStopUtility = c.timeToNextStopCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - // Doug and Kevin Hack of Jan 5 2004 -// if (myTour.getTotalElapsedTime()>240.0) timeToNextStopUtility*=3; - co[6] = timeToNextStopUtility; - } else co[6]=0; - - if (c.xMatrix !=null && c.yMatrix != null) { - - double xOrig = c.xMatrix.getValueAt(c.myTour.getOriginZone(),1); - double yOrig = c.yMatrix.getValueAt(c.myTour.getOriginZone(),1); - double xNow = c.xMatrix.getValueAt(c.myTour.getCurrentLocation(),1); - double yNow = c.yMatrix.getValueAt(c.myTour.getCurrentLocation(),1); - double xMaybe = c.xMatrix.getValueAt(location,1); - double yMaybe = c.yMatrix.getValueAt(location,1); - double angle1 = Math.atan2(yNow-yOrig,xNow-xOrig); - double angle2 = Math.atan2(yMaybe-yNow,xMaybe-xNow); - double angle = (angle2-angle1)+Math.PI; - if (angle > Math.PI*2) angle -= Math.PI*2; - if (angle <0) angle += Math.PI*2; - if (angle > Math.PI) angle =2*Math.PI-angle; - co[7]= c.angleCoefficient*angle*180/Math.PI; - } else co[7]=0; - co[8]= c.zoneTypeUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); - if (c.sizeTermCoefficient !=0) { - double sizeTermValue = Math.log(c.sizeTerm.calcForIndex(location,1)); - co[9]= c.sizeTermCoefficient*sizeTermValue; - } else co[9]=0; - StringBuffer logStatement = new StringBuffer("OD "+c.getTour().getCurrentLocation()+","+location+" :"); - double uti=0; - for (int index=0;indexduration in hours - */ - public float duration; - /** - * location represents the location of the stop - */ - public int location; - /** - * purpose represents the purpose of the stop - */ - public int purpose; - public double travelTimeMinutes; - public String tripMode = "NA"; - public final int previousLocation; - public final double departureTimeFromPreviousStop; - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java deleted file mode 100644 index 9249773..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopAlternative.java +++ /dev/null @@ -1,106 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - - -public class StopAlternative implements CodedAlternative { - final StopChoice c; - public final int location; - public double getUtility() { - - if (c.maxDist >0) { - assert c.distanceMatrix!=null : "Distance Matrix didn't get initialized, yet maxDist set to "+c.maxDist; - double distance = c.distanceMatrix.getValueAt(c.myTour.getOriginZone(),location); - - if((distance<-99999) || (distance>99999)) - distance=0; - - if (distance >c.maxDist) { - return Double.NEGATIVE_INFINITY; - } - } - - boolean firstStop = (c.myTour.getStopCounts()[0]==0); - - double utility = c.travelUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); - utility += c.destinationUtilityFunction.calcForIndex(location,1); - utility += c.returnHomeUtilityFunction.calcForIndex(location,c.myTour.getOriginZone()); - - if (c.disutilityToOriginCoefficient!=0 || c.disutilityToOriginAdditionalCoefficientForStopGT1!=0) { - double coefficient = c.disutilityToOriginCoefficient + (firstStop ? 0 : c.disutilityToOriginAdditionalCoefficientForStopGT1); - utility += coefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - } - if (c.disutilityToNextStopCoefficient!=0 || c.disutilityToNextStopAdditionalCoefficientForStopGT1 !=0) { - double coefficient = c.disutilityToNextStopCoefficient + (firstStop ? 0 : c.disutilityToNextStopAdditionalCoefficientForStopGT1); - utility += coefficient*c.myTour.getTravelDisutilityTracker().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - } - if (c.timeToOriginCoefficient!=0) { - double timeToOriginUtility = c.timeToOriginCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(location,c.myTour.getOriginZone(),c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - // Doug and Kevin Hack of Jan 5 2004 -// if (myTour.getTotalElapsedTime()>240.0) timeToOriginUtility*=3; - utility += timeToOriginUtility; - } - if (c.timeToNextStopCoefficient!=0) { - double timeToNextStopUtility = c.timeToNextStopCoefficient*c.myTour.getElapsedTravelTimeCalculator().getTravelAttribute(c.myTour.getCurrentLocation(),location,c.myTour.getCurrentTimeHrs(),c.myTour.getMyVehicleTourType().vehicleType); - // Doug and Kevin Hack of Jan 5 2004 -// if (myTour.getTotalElapsedTime()>240.0) timeToNextStopUtility*=3; - utility += timeToNextStopUtility; - } - - - if (c.xMatrix !=null && c.yMatrix != null) { - // angle calculation and max distance - - double xOrig = c.xMatrix.getValueAt(c.myTour.getOriginZone(),1); - double yOrig = c.yMatrix.getValueAt(c.myTour.getOriginZone(),1); - double xNow = c.xMatrix.getValueAt(c.myTour.getCurrentLocation(),1); - double yNow = c.yMatrix.getValueAt(c.myTour.getCurrentLocation(),1); - double xMaybe = c.xMatrix.getValueAt(location,1); - double yMaybe = c.yMatrix.getValueAt(location,1); - - double angle1 = Math.atan2(yNow-yOrig,xNow-xOrig); - double angle2 = Math.atan2(yMaybe-yNow,xMaybe-xNow); - double angle = (angle2-angle1)+Math.PI; - if (angle > Math.PI*2) angle -= Math.PI*2; - if (angle <0) angle += Math.PI*2; - if (angle > Math.PI) angle =2*Math.PI-angle; - utility += c.angleCoefficient*angle*180/Math.PI; - } - utility += c.zoneTypeUtilityFunction.calcForIndex(c.myTour.getCurrentLocation(),location); - if (c.sizeTermCoefficient !=0) { - double sizeTermValue = Math.log(c.sizeTerm.calcForIndex(location,1)); - utility += c.sizeTermCoefficient*sizeTermValue; - } - return utility; - } - - public StopAlternative(StopChoice choice, int stopLocation) { - this.location = stopLocation; - this.c = choice; - } - - public String getCode() { - return String.valueOf(location); - } - - - - - } \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java deleted file mode 100644 index 38f419a..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/StopChoice.java +++ /dev/null @@ -1,230 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.common.emme2.IndexConditionFunction; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.LogitModel; -import com.pb.common.matrix.*; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - * Modified 2014 for trip mode choice - */ -public abstract class StopChoice extends LogitModel implements ModelUsesMatrices { - - static int angleCalculationCounter = 50; - public static MatrixCacheReader myMatrixCacheReader=null; - - static Logger logger = Logger.getLogger(StopChoice.class); - /*private static Boolean useTripModeChoice = null;*/ - - - - protected double angleCoefficient; - protected double maxDist; - - protected IndexLinearFunction destinationUtilityFunction = new IndexLinearFunction(); - - protected double disutilityToNextStopAdditionalCoefficientForStopGT1; - protected double disutilityToNextStopCoefficient; - protected double disutilityToOriginAdditionalCoefficientForStopGT1; - protected double disutilityToOriginCoefficient; - protected Tour myTour; - protected IndexLinearFunction returnHomeUtilityFunction = new IndexLinearFunction(); - protected IndexLinearFunction sizeTerm = null; - protected double sizeTermCoefficient = 0; - protected double timeToNextStopCoefficient; - protected double timeToOriginCoefficient; - protected IndexLinearFunction travelUtilityFunction = new IndexLinearFunction(); - protected Matrix xMatrix; - private String xMatrixName; - protected Matrix yMatrix; - private String yMatrixName; - protected IndexConditionFunction zoneTypeUtilityFunction = new IndexConditionFunction(); - public Matrix distanceMatrix; - private String distanceMatrixName; - /** - * Method addCoefficient. - * @param alternative - * @param index1 - * @param index2 - * @param matrix - * @param coefficient - */ - public void addCoefficient( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError { - if (!alternative.equals("zone")) throw new RuntimeException("StopAlternative coefficients must have \"zone\" as alternative"); - - //If Index1 is "cstop" then Index2 must equal "nstop", and the term in the utility function is the entry from the mf matrix identified in the Matrix field, indexed with i being the current stop location and j being the next stop location whose utility is being evaluated. - if(index1.equals("cstop")) { - if(index2.equals("nstop")) { - travelUtilityFunction.addCoefficient(matrix,coefficient); - } else throw new CoefficientFormatError("cstop coefficients for next stop location must index an mf matrix using nstop as the J"); - } - - - else if (index1.equals("nstop")) { - //If Index1 is "nstop" and Index2 is "origin" then Matrix identifies an mf matrix, and the term in the utility function is the matrix value indexed with i being the next stop location whose utility is being evaluated and j is the origin of the tour. - if(index2.equals("origin")) { - returnHomeUtilityFunction.addCoefficient(matrix,coefficient); - //If Index1 is "nstop" and Index2 is blank, then Matrix identifies an mo or md matrix. The next stop location under consideration is used to retrieve the appropriate entry from the matrix. - } else if (index2.equals("") ||index2.equals("none")) { - destinationUtilityFunction.addCoefficient(matrix,coefficient); - } else throw new CoefficientFormatError("nstop coefficients for next stop location must have index2=\"\" or index2 = origin"); - } - else if (index1.equals("angle")) { - angleCoefficient = coefficient; - setXYNames(matrix, index2); - } - else if (index1.equals("travelDisutility")) { - if (index2.equals("nstop")) { - disutilityToNextStopCoefficient+=coefficient; - } else if (index2.equals("origin")) { - disutilityToOriginCoefficient+=coefficient; - } else if (index2.equalsIgnoreCase("nstopx")){ - disutilityToNextStopAdditionalCoefficientForStopGT1 += coefficient; - } else if (index2.equalsIgnoreCase("originx")) { - disutilityToOriginAdditionalCoefficientForStopGT1 += coefficient; - } else { - throw new CoefficientFormatError("travelDisutility coefficients for next stop must have be to either \"origin\" or to \"nstop\""); - } - } - else if (index1.equals("travelTime")) { - if (index2.equals("nstop")) { - timeToNextStopCoefficient+=coefficient; - } else if (index2.equals("origin")) { - timeToOriginCoefficient+=coefficient; - } else { - throw new CoefficientFormatError("travelDisutility coefficients for next stop must have be to either \"origin\" or to \"nstop\""); - } - } - else if (index1.equals("sizeTerm1")) { - sizeTermCoefficient += coefficient; - if (sizeTerm== null) sizeTerm = new IndexLinearFunction(); - sizeTerm.addCoefficient(matrix,1.0); - } - else if (index1.equals("sizeTerm2") || index1.equals("sizeTermx")) { - if (sizeTerm== null) sizeTerm = new IndexLinearFunction(); - sizeTerm.addCoefficient(matrix,coefficient); - } - else if (index1.equals("maxDist")) { - maxDist = coefficient; - distanceMatrixName = matrix; - } - else { - int destinationCondition; - int originCondition; - try { - destinationCondition = Integer.valueOf(index1).intValue(); - } catch (NumberFormatException e) { - throw new CoefficientFormatError("Can't convert "+index1+" to a number, not allowed as an index type for stop location choice"); - } - boolean twoTypeCondition=false; - try { - originCondition = Integer.valueOf(index2).intValue(); - zoneTypeUtilityFunction.addCoefficient(matrix,destinationCondition,originCondition,coefficient); - twoTypeCondition= true; - } catch (NumberFormatException e) { - } - if (!twoTypeCondition) { - zoneTypeUtilityFunction.addCoefficient(matrix,destinationCondition,coefficient); - } - } - } - private void setXYNames(String xName, String yName) { - if (xMatrixName==null) { - xMatrixName = xName; - } else{ - if (!xMatrixName.equals(xName)) { - String msg = "xName for angle and maxdist needs to be the same, "+xMatrixName+"!="+xName; - logger.fatal(msg); - throw new RuntimeException(msg); - } - } - if (yMatrixName == null) { - yMatrixName = yName; - } else { - if (!yMatrixName.equals(yName)) { - String msg = "yName for angle and maxdist needs to be the same, "+yMatrixName+"!="+yName; - logger.fatal(msg); - throw new RuntimeException(msg); - } - } - } - /** - * Returns the myTour. - * @return CommercialTour - */ - public Tour getTour() { - return myTour; - } - public void init() { - if (myMatrixCacheReader==null) throw new RuntimeException("StopChoice needs an initialized Emme2MatrixReader before it can be initialized"); - readMatrices(myMatrixCacheReader); - } - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader mr) { - travelUtilityFunction.readMatrices(mr); - destinationUtilityFunction.readMatrices(mr); - returnHomeUtilityFunction.readMatrices(mr); - zoneTypeUtilityFunction.readMatrices(mr); - if (sizeTerm != null) sizeTerm.readMatrices(mr); - if (xMatrixName != null) xMatrix = mr.readMatrix(xMatrixName); - if (yMatrixName!=null) yMatrix = mr.readMatrix(yMatrixName); - if (distanceMatrixName!=null) { - distanceMatrix = mr.readMatrix(distanceMatrixName); - } else { - logger.warn("Distance matrix name not specified, no maximum distance set"); - } - } - - /** - * Sets the myTour. - * @param myTour The myTour to set - */ - public void setTour(Tour myTour) { - this.myTour = myTour; - } - /*public static Boolean getUseTripModeChoice() { - return useTripModeChoice; - } - public static void setUseTripModeChoice(Boolean useTripModeChoice) { - StopChoice.useTripModeChoice = useTripModeChoice; - }*/ - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java deleted file mode 100644 index 7ed78ef..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/Tour.java +++ /dev/null @@ -1,290 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on Feb 4, 2005 - * - */ -package org.sandag.cvm.activityTravel; - -import java.util.ArrayList; -import java.util.Iterator; - -import org.apache.log4j.Logger; - -//import org.sandag.cvm.calgary.commercial.TourStartTimeModel; -//import org.sandag.cvm.calgary.commercial.WeekendTravelTimeTracker; -import org.sandag.cvm.activityTravel.cvm.TourStartTimeModel; -import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; -import org.sandag.cvm.common.model.NoAlternativeAvailable; - - -/** - * @author jabraham - * - * This is a representation of a tour. - */ -public abstract class Tour implements TourInterface { - - public abstract ChangingTravelAttributeGetter getElapsedTravelTimeCalculator(); - public abstract TourStartTimeModel getTourStartTimeModel(); - -// @Override - public String toString() { - StringBuffer buff = new StringBuffer("Tour from "+originZone+" via "); - for (int i=0;istops is an ArrayList containing instances of Tour.Stop, one for each stop made on the tour - */ - protected ArrayList stops = new ArrayList(); - - private double tourStartTimeHrs = 0; - private double travelTimeMinutes; - - static Logger logger = Logger.getLogger(Tour.class); - - /** - * Adds a stop to the stops and alsu updates travelTimeMinutes and currentTimeHrs - * @param newStop - */ - protected void addStop(Stop newStop) { - int lastStopLocation = getOrigin(); - if (stops.size()!=0) lastStopLocation = ((Stop) stops.get(stops.size()-1)).location; - stops.add(newStop); - // TODO need to use trip mode not tour mode for travel time tracking. - // TODO should account for toll/non toll for travel time tracking (e.g. if toll is a trip mode.) - double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastStopLocation,newStop.location,currentTimeHrs,myVehicleTourType.getVehicleType()); - if (legTravelTime == 0 ) { - // a problem - logger.warn("Leg travel time is zero for "+lastStopLocation + " to " + newStop.location); - } - newStop.travelTimeMinutes=legTravelTime; - travelTimeMinutes+=legTravelTime; - currentTimeHrs += (legTravelTime/60 + newStop.duration); - } - - protected double calcCurrentTime() { - double travelTime =0; - double currentTimeCalc = getTourStartTimeHrs(); - Iterator stopIt = stops.iterator(); - int lastStop = getOriginZone(); - while (stopIt.hasNext()) { - Stop stop = (Stop) stopIt.next(); - // TODO need to use trip mode not tour mode for travel time tracking. - double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastStop,stop.location,currentTimeCalc,myVehicleTourType.getVehicleType()); - travelTime += legTravelTime; - currentTimeCalc += legTravelTime/60; - currentTimeCalc += stop.duration; - lastStop = stop.location; - } - return currentTimeCalc; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getCurrentTimeHrs() - */ - public double getCurrentTimeHrs() { - return currentTimeHrs; - } - - /** - * Method getLastStopType. - * @return int - */ - public int getLastStopType() { - if (stops.size()==0) return 0; - Stop theLastStop = (Stop) stops.get(stops.size()-1); - return theLastStop.purpose; - } - - /** - * @return Returns the myVehicleTourType. - */ - public TourType getMyVehicleTourType() { - return myVehicleTourType; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getOrigin() - */ - public int getOrigin() { - return getOriginZone(); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getOriginZone() - */ - public int getOriginZone() { - return originZone; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getStopCount() - */ - public int[] getStopCounts() { - int[] stopCounter = new int[getMaxTourTypes()+1]; - Iterator stopIt = stops.iterator(); - while (stopIt.hasNext()) { - Stop stop = (Stop) stopIt.next(); - stopCounter[0]++; - stopCounter[stop.purpose]++; - } - return stopCounter; - } - - public abstract int getMaxTourTypes(); - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getTotalElapsedTime() - */ - public double getTotalElapsedTimeHrs() { - return getCurrentTimeHrs()-getTourStartTimeHrs(); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getTotalTravelTimeMinutes() - */ - public double getTotalTravelTimeMinutes() { - return travelTimeMinutes; - } - - /** - * Method getTourStartTime. - * @return double - */ - protected double getTourStartTimeHrs() { - return tourStartTimeHrs; - } - - /** - * Method getTourTypeCode. - * @return A string grepresenting the type of tour -- perhaps representing the types - * of activities that occur in the tour. - */ - protected String getTourTypeCode() { - return myVehicleTourType.getCode().substring(1); - } - - /** - * Method getVehicleCode. - * @return a string representing the tNCVehicle code used for the tour - */ - protected String getVehicleCode() { - return myVehicleTourType.getCode().substring(0,1); - } - - /** - * @return Returns the vehicleTourTypeChoice. - */ - public abstract VehicleTourTypeChoice getVehicleTourTypeChoice(); - - /** - * Method sampleStartTime. - */ - public void sampleStartTime() { - tourStartTimeHrs = getTourStartTimeModel().sampleValue(); - if (stops.size()==0) currentTimeHrs = tourStartTimeHrs; - else currentTimeHrs = calcCurrentTime(); - - } - - /** - * This method uses a random number generator to sample the stops along the tour -- their location, - * duration and purpose. - */ - public abstract void sampleStops(); - - /** - * This method uses a random number generator to sample the type of tour, including - * the type of tNCVehicle(s) used for the tour and perhaps some information about the types - * of activities that occur along the tour. - */ - public void sampleVehicleAndTourType() { - getVehicleTourTypeChoice().setMyTour(this); - try { - myVehicleTourType = (TourType) getVehicleTourTypeChoice().monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - myVehicleTourType = null; - //Leave it null for an error to occur when it's actually needed - } - } - - /** - * @param currentTimeHrs The currentTimeHrs to set. - */ - protected void setCurrentTimeHrs(double currentTimeHrs) { - this.currentTimeHrs = currentTimeHrs; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#setOrigin(int) - */ - public void setOrigin(int z) { - setOriginZone(z); - } - - void setOriginZone(int originZone) { - this.originZone = originZone; - } - - /** - * @param tourStartTimeHrs The time when the tour starts. - */ - protected void setTourStartTimeHrs(double tourStartTimeHrs) { - this.tourStartTimeHrs = tourStartTimeHrs; - } - - /** - * @param vehicleTourTypeChoice The vehicleTourTypeChoice to set. - */ - public abstract void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoice); - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.TourInterface#getCurrentLocation() - */ - public int getCurrentLocation() { - if (stops.size()==0) return getOriginZone(); - Stop stop = (Stop) stops.get(stops.size()-1); - return stop.location; - } - - public abstract ChangingTravelAttributeGetter getTravelDisutilityTracker() ; - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java deleted file mode 100644 index c755573..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourInterface.java +++ /dev/null @@ -1,61 +0,0 @@ -package org.sandag.cvm.activityTravel; - -public interface TourInterface { - - /** - * Method getCurrentTime - * @deprecated - * @return double current time of day in hours - */ - public double getCurrentTimeHrs(); - - /** - * Method getCurrentTime - * @return double current time of day in minutes - */ - public double getCurrentTimeMinutes(); - - /** - * Method getOrigin. - * @return an int representing the origin location of the tour. - */ - public int getOrigin(); - - /** - * @return the integer representing the origin of the tour - */ - public int getOriginZone(); - - /** - * Method getStopCount. - * @return int[] an integer array counting the stops that occur by type. Element 0 - * is the total number of stops; other elements correspond to different stop purposes - */ - public int[] getStopCounts(); - - /** - * Method getTotalElapsedTime. - * @deprecated - * @return double total elapsed time in hours - */ - public double getTotalElapsedTimeHrs(); - - /** - * Method getTotalElapsedTime. - * @return double total elapsed time in hours - */ - public double getTotalElapsedTimeMinutes(); - - /** - * Method getTotalTravelTime. - * @return double total travel time in minutes - */ - public double getTotalTravelTimeMinutes(); - - /** - * Method getCurrentLocation. - * @return int - */ - public int getCurrentLocation(); - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java deleted file mode 100644 index 1ddaeaa..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourNextStopPurposeChoice.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on 25-Feb-2005 - * - */ -package org.sandag.cvm.activityTravel; - -import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; - - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public interface TourNextStopPurposeChoice extends DiscreteChoiceModelInterface { - /** - * sets the tour to be used for independent variables to influence the choice of next stop purpose - * @param myTour - */ - public abstract void setMyTour(Tour myTour); - /** - * @return the tour currently being used for information that influences the next stop purpose choice - */ - public abstract Tour getMyTour(); -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java deleted file mode 100644 index fefb6a1..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TourType.java +++ /dev/null @@ -1,115 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on Feb 4, 2005 - * - */ -package org.sandag.cvm.activityTravel; - -/** - * @author jabraham - * - * This class is a representation of a type of tour. Contained it in - * are counters for the number of instances of the type of tour and trips - * as well as a choice model for choosing a tNCVehicle type for the tour. - * The class is identified by a name which is a String. - */ -public abstract class TourType { - - /** - * @param tourType a string representing the type of tour, which may also have information on tNCVehicle types - * @param vehicleType a char representing the type of tNCVehicle - * @param theChoiceModel the choice model associated with the tour type - */ - public TourType(String tourType, char vehicleType, VehicleTourTypeChoice theChoiceModel){ - myChoice = theChoiceModel; - this.tourTypeName = tourType; - this.vehicleType= vehicleType; - } - - /** - * tourCount is a counter for the number of tours of this type. - */ - protected int tourCount = 0; - protected int tripCount = 0; - - /** - * This class can serve as a place to keep track of the number of tours and trips of different types. - * This method increments the number of tours and trips. - * @param tours the number of tours to increment the tour counter by - * @param trips the number of trips to increment the trip counter by - */ - public void incrementTourAndTripCount(int tours, int trips) { - tourCount += tours; - tripCount += trips; - } - - /** - * tourTypeName is the unique identifier of the type of tour - */ - public final String tourTypeName; - /** - * vehicleType is the identifier for the tNCVehicle type used in the tour - */ - public final char vehicleType; - - /** - * @return the tourTypeName - */ - public String getCode() { - return getTourTypeName(); - } - - /** - * @return the tourTypeName - */ - public String getTourTypeName() { - return tourTypeName; - } - - /** - * @return the char representing the tNCVehicle type - */ - public char getVehicleType() { - return vehicleType; - } - - /** - * @return the utility of this alternative - */ - public abstract double getUtility(); - - /** - * @return Returns the tourCount. - */ - public int getTourCount() { - return tourCount; - } - - /** - * @return Returns the tripCount. - */ - public int getTripCount() { - return tripCount; - } - - protected final VehicleTourTypeChoice myChoice; - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java deleted file mode 100644 index 14ef56a..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TravelTimeTracker.java +++ /dev/null @@ -1,117 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel; - -import java.util.ArrayList; -import java.util.Iterator; - -import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; -import org.sandag.cvm.activityTravel.ModelUsesMatrices; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public abstract class TravelTimeTracker - implements ChangingTravelAttributeGetter, ModelUsesMatrices { - - protected final ArrayList travelTimeMatrices = new ArrayList(); - - - - public static class TravelTimeMatrixSpec { - final String name; - Matrix matrix; - final float startTime; - final float endTime; - final char vehicleType; - - public TravelTimeMatrixSpec(String name, float startTime, float endTime, char vehicleType) { - this.name = name; - this.startTime = startTime; - this.endTime=endTime; - this.vehicleType = vehicleType; - } - - /** - * Method readMatrices. - * @param matrixReader - */ - void readMatrices(MatrixCacheReader matrixReader) { - matrix = matrixReader.readMatrix(name); - } - - /** - * Method getTimeFromMatrix. - * @param origin - * @param destination - * @return double - * - * Note: modified to return 0 if value is greater than 99999 - */ - double getTimeFromMatrix(int origin, int destination) { - double value= matrix.getValueAt(origin,destination); - if((value>(-99999)) && (value<99999)) - return value; - else - return 0; - } - - - - } - - /** - * Method get. - * @param lastStop - * @param i - * @param currentTime - * @return double - */ - public double getTravelAttribute(int origin, int destination, double timeOfDay, char vehicleType) { - TravelTimeMatrixSpec defaultMatrix = null; - while (timeOfDay>=24.00) timeOfDay -=24.00; - for (int i =0; i=s.startTime && timeOfDay < s.endTime && s.vehicleType == vehicleType) return s.getTimeFromMatrix(origin,destination); - else if (s.startTime <0) defaultMatrix = s; - } - } - if (defaultMatrix==null) throw new RuntimeException("no default travel time matrix for tNCVehicle type "+vehicleType); - return defaultMatrix.getTimeFromMatrix(origin,destination); - } - - public void readMatrices(MatrixCacheReader matrixReader) { - Iterator it = travelTimeMatrices.iterator(); - while (it.hasNext()) { - TravelTimeMatrixSpec s = (TravelTimeMatrixSpec) it.next(); - s.readMatrices(matrixReader); - } - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java deleted file mode 100644 index 4c0f205..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripMode.java +++ /dev/null @@ -1,74 +0,0 @@ -package org.sandag.cvm.activityTravel; - -import org.sandag.cvm.common.emme2.MatrixCacheReader; - -public abstract class TripMode { - - /** - * First part of string is tNCVehicle type aka tour mode. - * Followed by a colon and then the trip mode. - * e.g L:T means tour "Light" and trip-mode "toll" - */ - protected final String myType; - protected final TripModeChoice myChoiceModel; - public final char vehicleType; - public final String tripMode; - protected int origin; - protected int destination; - protected double timeOfDay; - - public TripMode( - TripModeChoice choiceModel, - String type) { - myType = type; - myChoiceModel = choiceModel; - vehicleType = myType.split(":")[0].charAt(0); - tripMode = myType.split(":")[1]; - - } - - - public abstract double getUtility(); - - - public String getCode() { - return myType; - } - - public abstract void readMatrices(MatrixCacheReader matrixReader); - - public abstract void addCoefficient(String index1, String index2, - String matrix, double coefficient) throws CoefficientFormatError; - - public int getDestination() { - return destination; - } - - public void setDestination(int destination) { - this.destination = destination; - } - - public int getOrigin() { - return origin; - } - - public void setOrigin(int origin) { - this.origin = origin; - } - - public void setTime(double time) { - timeOfDay = time; - - } - - public String logOriginDestination() { - return String.valueOf(origin)+" to "+destination; - } - - public String getTripMode() { - return tripMode; - } - - - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java deleted file mode 100644 index 391a6eb..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/TripModeChoice.java +++ /dev/null @@ -1,96 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on Feb 4, 2005 - * - */ -package org.sandag.cvm.activityTravel; - -import java.util.Iterator; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.activityTravel.cvm.CommercialTripMode; -import org.sandag.cvm.common.emme2.MatrixAndTAZTableCache; -import org.sandag.cvm.common.model.Alternative; -import org.sandag.cvm.common.model.ChoiceModelOverflowException; -import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; -import org.sandag.cvm.common.model.LogitModel; -import org.sandag.cvm.common.model.NoAlternativeAvailable; - - -/** - * @author jabraham - * - * A model of tNCVehicle type together with tour type - */ -public abstract class TripModeChoice implements ChangingTravelAttributeGetter { - - protected LogitModel myLogitModel = new LogitModel(); - - protected static Logger logger = Logger.getLogger(TripModeChoice.class); - - protected Tour theTour; - - /** - * @param myTour the tour to consider when making the tour type choice - */ - public void setMyTour(Tour myTour) { - theTour = myTour; - } - - - - /** - * @return the tour associated with the tour type choice - */ - public Tour getMyTour() { - return theTour; - } - - - - @Override - public double getTravelAttribute(int origin, int destination, double time, - char vehicleType) { - Iterator m = myLogitModel.getAlternativesIterator(); - while (m.hasNext()) { - TripMode tm = (TripMode) m.next(); - tm.setOrigin(origin); - tm.setDestination(destination); - tm.setTime(time); - } - return myLogitModel.getUtility(); - } - - - - public void readMatrices(MatrixAndTAZTableCache matrixReader) { - Iterator m = myLogitModel.getAlternativesIterator(); - while (m.hasNext()) { - CommercialTripMode tm = (CommercialTripMode) m.next(); - tm.readMatrices(matrixReader); - } - } - - - public abstract TripMode chooseTripModeForDestination(int location) throws ChoiceModelOverflowException, NoAlternativeAvailable; - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java deleted file mode 100644 index d87ec63..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/VehicleTourTypeChoice.java +++ /dev/null @@ -1,51 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -/* - * Created on Feb 4, 2005 - * - */ -package org.sandag.cvm.activityTravel; - -import org.sandag.cvm.common.model.DiscreteChoiceModelInterface; - - -/** - * @author jabraham - * - * A model of tNCVehicle type together with tour type - */ -public interface VehicleTourTypeChoice extends DiscreteChoiceModelInterface { - /** - * @param myTour the tour to consider when making the tour type choice - */ - public void setMyTour(Tour myTour); - - /** - * - */ - public void writeTourAndTripSummary(); - - /** - * @return the tour associated with the tour type choice - */ - Tour getMyTour(); - - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java deleted file mode 100644 index a2ffddd..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/ZonePairDisutility.java +++ /dev/null @@ -1,7 +0,0 @@ -package org.sandag.cvm.activityTravel; - -public interface ZonePairDisutility { - - public abstract double calcForIndex(int i, int j); - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java deleted file mode 100644 index 6c04cf1..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/AlogitLogitModelNest.java +++ /dev/null @@ -1,21 +0,0 @@ -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.common.model.LogitModel; - -public class AlogitLogitModelNest extends LogitModel { - - double nestingCoefficient = 1.0; - - public void setAlogitNestingCoefficient(double coefficient) { - nestingCoefficient = coefficient; - } - - @Override - public double getUtility() { - if (getDispersionParameter()!=1.0) { - throw new RuntimeException("Alogit nesting always needs a dispersion parameter of 1.0 in the lower level nests"); - } - return super.getUtility()*nestingCoefficient; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java deleted file mode 100644 index c937cad..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopChoice.java +++ /dev/null @@ -1,64 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.activityTravel.StopAlternative; -import org.sandag.cvm.activityTravel.StopChoice; -import com.pb.common.matrix.Matrix; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CommercialNextStopChoice extends StopChoice { - - private String segmentId; - - /** - * Constructor for VehicleTypeChoice. - */ - - public CommercialNextStopChoice(int[] zoneNums, int notEqualToOrLowerThan, int notEqualToOrHigherThan, String segmentID) { - super(); - this.segmentId = segmentID; - for (int z = 1;z notEqualToOrLowerThan && theNumber < notEqualToOrHigherThan) { - this.addAlternative(new StopAlternative(this, zoneNums[z])); - } - } - } - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateCommercialTours.matrixReader); - } - - @Override - public String toString() { - return "Stop choice for "+segmentId; - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java deleted file mode 100644 index 7409cc7..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialNextStopPurposeChoice.java +++ /dev/null @@ -1,249 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.LogitModel; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -import java.util.*; - -import org.apache.log4j.Logger; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CommercialNextStopPurposeChoice extends LogitModel implements ModelUsesMatrices, TourNextStopPurposeChoice { - - /** - * Constructor for VehicleTypeChoice. - */ - - final static int SERVICE = 1; - final static int GOODS = 2; - final static int OTHER = 3; - final static int RETURNTOORIGIN = 4; - private static Logger logger = Logger.getLogger(CommercialNextStopPurposeChoice.class); - - final char tourType; - - /** - * Constructor CommercialNextStopPurposeChoice. - * @param c - */ - public CommercialNextStopPurposeChoice(char c) { - if (c == 'S' || c == 's') this.addAlternative(new NextStopPurpose(SERVICE)); - if (c == 'G' || c == 'g') this.addAlternative(new NextStopPurpose(GOODS)); - this.addAlternative(new NextStopPurpose(OTHER)); - this.addAlternative(new NextStopPurpose(RETURNTOORIGIN)); - tourType = c; - } - - static String decodeStopPurpose(int s) { - if (s==SERVICE) return "Srv"; - if (s==GOODS) return "Gds"; - if (s==OTHER) return "Oth"; - if (s==RETURNTOORIGIN) return "Est"; - String msg = "Bad stop purpose code "+s; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - - private CommercialTour myTour; - - - public class NextStopPurpose implements AlternativeUsesMatrices { - double[] transitionConstants = {0,0,0,0}; - double[] stopCountCoefficients = {0,0,0,0}; - /** - * Constructor VehicleTypeAlternative. - * @param stopType - */ - public NextStopPurpose(int stopType) { - this.stopType = stopType; - } - - double constant = 0; - - final int stopType; - IndexLinearFunction previousStopUtility = new IndexLinearFunction(); - IndexLinearFunction originUtility = new IndexLinearFunction(); - IndexLinearFunction returnToOriginUtility = new IndexLinearFunction(); - double timeToOriginCoefficient = 0; - double disutilityToOriginCoefficient = 0; - double totalTravelTimeCoefficient = 0; - double totalTripTimeCoefficient = 0; - public double getUtility() { - double utility = previousStopUtility.calcForIndex(myTour.getCurrentLocation(),1); - utility += originUtility.calcForIndex(getMyTour().getOriginZone(),1); - int previousStopType= myTour.getLastStopType(); - utility += transitionConstants[previousStopType]; - int[] stopCounts = getMyTour().getStopCounts(); - // can't return home on first stop - if (stopCounts[0]==0 && stopType==RETURNTOORIGIN) utility += Double.NEGATIVE_INFINITY; - utility += stopCountCoefficients[0]*Math.log(stopCounts[0] +1) + - stopCountCoefficients[1]*Math.log(stopCounts[1]+1) + - stopCountCoefficients[2]*Math.log(stopCounts[2]+1) + - stopCountCoefficients[3]*Math.log(stopCounts[3]+1); - double returnHomeUtility = returnToOriginUtility.calcForIndex(myTour.getCurrentLocation(),getMyTour().getOriginZone()); - - // make people return home more -- Doug and Kevin Hack of Jan 5th - //if (myTour.getTotalElapsedTime()>240.0) returnHomeUtility *=3; - utility += returnHomeUtility; - - utility += totalTravelTimeCoefficient*getMyTour().getTotalTravelTimeMinutes(); - utility += totalTripTimeCoefficient*getMyTour().getTotalElapsedTimeHrs(); - utility += timeToOriginCoefficient*getMyTour().getElapsedTravelTimeCalculator().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); - utility += disutilityToOriginCoefficient*getMyTour().getTravelDisutilityTracker().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); - utility += constant; - return utility; - } - - /** - * Method addParameter. - * @param matrix - * @param coefficient - */ - public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { - if(index1.equals("origin")) { - originUtility.addCoefficient(matrix,coefficient); - } else if (index1.equals("cstop")) { - if (index2.equals("origin")) returnToOriginUtility.addCoefficient(matrix,coefficient); - else previousStopUtility.addCoefficient(matrix,coefficient); - } else if (index1.equals("prevStopType")) { - if (index2.equals("goods")) transitionConstants[GOODS] = coefficient; - else if (index2.equals("service")) transitionConstants[SERVICE] = coefficient; - else if (index2.equals("other")) transitionConstants[OTHER] = coefficient; - else if (index2.equals("return")) transitionConstants[RETURNTOORIGIN]= coefficient; - else throw new RuntimeException("previous stop type not known: "+index2); - } else if (index1.equals("logStopCount")) { - if (index2.equals("goods")) stopCountCoefficients[GOODS] = coefficient; - else if (index2.equals("service")) stopCountCoefficients[SERVICE] = coefficient; - else if (index2.equals("other")) stopCountCoefficients[OTHER] = coefficient; - else if (index2.equals("all")) stopCountCoefficients[0] = coefficient; - else throw new RuntimeException("stop count type not known: "+index2); - } else if (index1.equals("timeAccumulator")) { - totalTravelTimeCoefficient += coefficient; - } else if (index1.equals("totalAccumulator")) { - totalTripTimeCoefficient += coefficient; - } else if (index1.equals("travelDisutility") && index2.equals("origin")) { - disutilityToOriginCoefficient += coefficient; - } else if (index1.equals("travelTime") && index2.equals("origin")) { - timeToOriginCoefficient += coefficient; - } else if (index1.equals("") && index2.equals("")) { - constant += coefficient; - } else { - throw new CoefficientFormatError("invalid indexing "+index1+ ","+index2+" in matrix "+matrix +" for next stop purpose model "); - } - } - - - - /** - * Method readMatrices. - * @param mr - */ - public void readMatrices(MatrixCacheReader mr) { - previousStopUtility.readMatrices(mr); - originUtility.readMatrices(mr); - returnToOriginUtility.readMatrices(mr); - } - - /** - * Method getStopPurposeCode. - * @return String - */ - public String getCode() { - if (stopType == SERVICE) return "S"; - if (stopType == OTHER) return "O"; - if (stopType == GOODS) return "G"; - if (stopType == RETURNTOORIGIN) return "R"; - return null; - } - - @Override - public String toString() { - return "StopPurpose:"+getCode(); - } - - - -} - - /** - * Method addParameter. - * @param alternative - * @param matrix - * @param coefficient - */ - public void addCoefficient( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError { - Iterator alternativeIterator = alternatives.iterator(); - boolean found = false; - while (alternativeIterator.hasNext()) { - AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); - if (alternative.equals(alt.getCode())) { - alt.addCoefficient(index1,index2,matrix,coefficient); - found = true; - } - } - if (!found) throw new CoefficientFormatError("Bad alternative in next stop purpose choice model: "+alternative); - } - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader matrixCacheReader) { - Iterator alternativeIterator = alternatives.iterator(); - while (alternativeIterator.hasNext()) { - AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); - alt.readMatrices(matrixCacheReader); - } - } - - public void setMyTour(Tour myTour) { - this.myTour = (CommercialTour) myTour; - } - - public Tour getMyTour() { - return myTour; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateCommercialTours.matrixReader); - } - - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java deleted file mode 100644 index 279c3ba..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTour.java +++ /dev/null @@ -1,328 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel.cvm; -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.*; - -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; -import java.util.*; - -import org.apache.log4j.Logger; - -import com.pb.common.matrix.*; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CommercialTour extends Tour { - - static Logger logger = Logger.getLogger(CommercialTour.class); - - - private final PrintWriter tripLog; - private Hashtable models; - private final GenerateCommercialTours generator; - - - -// private String myTripOutputMatrixName; -// Matrix emme2OutputMatrix; - - static class TripOutputMatrixSpec { - - final String name; - boolean write; - Matrix matrix; - final float startTime; - final float endTime; - final char vehicleType; - public final String tripMode; - - TripOutputMatrixSpec(String name, String type, float startTime, float endTime, char vehicleType, String tripMode) { - if (type.equals("tripOut")) { - write = true; - } else if (type.equals("timeDef")) { - write = false; - } else { - String msg = "Invalid type in index1 in TripMatrix "+type; - logger.fatal(msg); - throw new RuntimeException(msg); - } - this.name = name; - this.startTime = startTime; - this.endTime=endTime; - this.vehicleType = vehicleType; - this.tripMode = tripMode; - } - - @Override - public String toString() { - return "Outmatrix "+name+":"+vehicleType+"("+startTime+":"+endTime+")"; - } - - /** - * Method readMatrices. - * @param matrixReader - */ - void readMatrices(MatrixCacheReader matrixReader) { - matrix = null; - if (write) matrix = matrixReader.readMatrix(name); - } - - public void createMatrix(int size,int[] externalNumbers) { - matrix = new Matrix(name, "Trips for "+vehicleType+" between "+startTime+" and "+endTime,size,size); - matrix.setExternalNumbers(externalNumbers); - } - - } - - - - public CommercialTour(Hashtable models, GenerateCommercialTours generator, PrintWriter tripLog, int tourNumber) { - super(); - this.tripLog = tripLog; - this.generator = generator; - this.models = models; - tourNum = tourNumber; - } - - /** - * Method sampleVehicleAndTourType. - */ - public void sampleVehicleAndTourType() { - synchronized (generator.vehicleTourTypeChoice) { - generator.vehicleTourTypeChoice.setMyTour(this); - try { - myVehicleTourType = (TourType) generator.vehicleTourTypeChoice.monteCarloElementalChoice(); - myNextStopPurposeChoice = (CommercialNextStopPurposeChoice) models.get(myVehicleTourType.getCode()+"StopType"); - ((CommercialNextStopPurposeChoice) myNextStopPurposeChoice).setMyTour(this); - } catch (NoAlternativeAvailable e) { - throw new RuntimeException(e); - } - } - } - - /** - * Method addTripsToMatrix. - */ - public void addTripsToMatrix() { - int trips = 0; - Iterator stopIterator = stops.iterator(); - int lastLocation = getOriginZone(); - double currentTime = getTourStartTimeHrs(); - String prevStopType = "Est"; - while (stopIterator.hasNext()) { - Stop s = (Stop) stopIterator.next(); - trips ++; - int location = s.location; - // FIXME should take into account trip mode. - double legTravelTime = getElapsedTravelTimeCalculator().getTravelAttribute(lastLocation,location,currentTime,myVehicleTourType.getVehicleType()); - float midPointOfTripTime = (float) (currentTime + legTravelTime/60/2); - - String newStopType = CommercialNextStopPurposeChoice.decodeStopPurpose(s.purpose); - - String mNameForTripLog = "None"; - - TripOutputMatrixSpec spec = generator.getTripOutputMatrixSpec(getVehicleCode().charAt(0),s.tripMode,midPointOfTripTime); - if (spec!=null) { - mNameForTripLog = spec.name; - if (spec.write) { - synchronized(spec.matrix) { - // increment stop count in the matrix if this matrix is going to be written out - float mTripCountEntry = spec.matrix.getValueAt(lastLocation,location); - mTripCountEntry++; - spec.matrix.setValueAt(lastLocation,location,mTripCountEntry); - } - } - } - // write to trip log - if (tripLog !=null) { - tripLog.print("3,"+tourNum+",1,"+trips+",1,"+getOriginZone()+","+generator.segmentString1+","+prevStopType+","+newStopType+","+lastLocation+","+location+","+mNameForTripLog+","+getVehicleCode()+","+currentTime+","); - } - boolean tollAvailable = isTollAvailable(lastLocation,location,currentTime); - if (s.tripMode.equals("T") && !tollAvailable) { - logger.error("No toll available but toll chosen for "+s); - } - - currentTime += legTravelTime/60; - currentTime += s.duration; - if (tripLog !=null) { - tripLog.println(currentTime+","+s.duration+","+getTourTypeCode()+","+generator.segmentString2+","+s.tripMode+","+tollAvailable); - } - lastLocation = location; - prevStopType = newStopType; - } - myVehicleTourType.incrementTourAndTripCount(1,trips); - } - - /** - * Method sampleStops. - */ - public void sampleStops() { - ((CommercialNextStopPurposeChoice) myNextStopPurposeChoice).setMyTour(this); - CommercialNextStopChoice myNextStopModel; - do { - Stop thisStop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); - Alternative temp; - try { - temp = myNextStopPurposeChoice.monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - logger.fatal("no valid purpose alternative available for "+this, e); - throw new RuntimeException("no valid purpose alternative available for "+this, e); - } catch (RuntimeException e) { - logger.fatal("Cannot sample stops for "+this, e); - throw new RuntimeException("Cannot sample stops for "+this, e); - } - CommercialNextStopPurposeChoice.NextStopPurpose nextStopPurpose = (CommercialNextStopPurposeChoice.NextStopPurpose) temp; - thisStop.purpose=nextStopPurpose.stopType; - if (thisStop.purpose==CommercialNextStopPurposeChoice.RETURNTOORIGIN) { - thisStop.location=getOriginZone(); - chooseTripMode(thisStop); - addStop(thisStop); - break; - } - String nextStopModelCode = getVehicleCode()+getTourTypeCode()+nextStopPurpose.getCode() + "StopLocation"; - myNextStopModel = (CommercialNextStopChoice) models.get(nextStopModelCode); - if (myNextStopModel == null) throw new RuntimeException("Can't find stop model "+nextStopModelCode); - myNextStopModel.setTour(this); - try { - temp = myNextStopModel.monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - logger.error("no valid location alternative available from "+this.toString()+" for "+generator.segmentString+" stop purpose "+nextStopModelCode); - throw new RuntimeException("no valid location alternative available from "+this.toString()); - } - thisStop.location = ((StopAlternative) temp).location; - DurationModel myDurationModel = (DurationModel) models.get(myVehicleTourType.getCode()+"Duration"); - thisStop.duration = (float) myDurationModel.sampleValue(); - chooseTripMode(thisStop); - addStop(thisStop); - } while (true); - } - - private void chooseTripMode(Stop thisStop) { - if (generator.isUseTripModes()) { - CommercialVehicleTripModeChoice tmc = (CommercialVehicleTripModeChoice) generator.getTravelDisutilityTracker(getVehicleCode()); - tmc.setMyTour(this); - TripMode m; - try { - m = (TripMode) tmc.chooseTripModeForDestination(thisStop.location); - } catch (ChoiceModelOverflowException | NoAlternativeAvailable e) { - String msg = "Can't sample trip mode for "+this; - logger.fatal(msg); - throw new RuntimeException(msg); - } - thisStop.tripMode=m.getCode().split(":")[1]; - } - } - - private boolean isTollAvailable(int origin, int destination, double timeOfDay) { - if (!generator.isUseTripModes()) { - return false; - } - CommercialVehicleTripModeChoice tmc = (CommercialVehicleTripModeChoice) generator.getTravelDisutilityTracker(getVehicleCode()); - tmc.setMyTour(this); - - return tmc.isTollAvailable(origin,destination,timeOfDay); - } - - /** - * Method getTourTypeCode. - * @return String - */ - public String getTourTypeCode() { - return myVehicleTourType.getCode().substring(1); - } - - - /** - * Method getVehicleCode. - */ - public String getVehicleCode() { - return myVehicleTourType.getCode().substring(0,1); - } - - - private int tourNum; - - - - - /** - * @return - */ - public int getOriginZoneType() { - return Math.round(generator.landUseTypeMatrix.getValueAt(getOriginZone(),1)); - } - - public ChangingTravelAttributeGetter getTravelDisutilityTracker() { - if (generator.isUseTripModes()) { - return generator.getTravelDisutilityTracker(getVehicleCode()); - } - return generator.getTravelDisutilityTracker(); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.Tour#getMaxTourTypes() - */ - public int getMaxTourTypes() { - return 4; - } - - public double getCurrentTimeMinutes() { - return getCurrentTimeHrs()*60; - } - - public double getTotalElapsedTimeMinutes() { - return getTotalElapsedTimeHrs()*60; - } - - @Override - public VehicleTourTypeChoice getVehicleTourTypeChoice() { - // TODO Auto-generated method stub - return null; - } - - @Override - public void setVehicleTourTypeChoice( - VehicleTourTypeChoice vehicleTourTypeChoice) { - // TODO Auto-generated method stub - - } - - @Override - public ChangingTravelAttributeGetter getElapsedTravelTimeCalculator() { - // FIXME should use trip modes, like getTravelDisutilityTracker() does. - return generator.getElapsedTravelTimeCalculator(); - } - - @Override - public TourStartTimeModel getTourStartTimeModel() { - return generator.getTourStartTimeModel(); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java deleted file mode 100644 index 3a4be51..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTravelTimeTracker.java +++ /dev/null @@ -1,77 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; -import org.sandag.cvm.activityTravel.CoefficientFormatError; -import org.sandag.cvm.activityTravel.ModelUsesMatrices; -import org.sandag.cvm.activityTravel.TravelTimeTracker; - - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CommercialTravelTimeTracker extends TravelTimeTracker implements ModelUsesMatrices, ChangingTravelAttributeGetter { - - /** - * Method addCoefficient. - * @param alternative - * @param index1 - * @param index2 - * @param matrix - * @param coefficient - */ - public void addCoefficient ( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError - { - if (alternative.length()!=1) { - throw new CoefficientFormatError("Alternative must be L, M, I or H for TravelTimeMatrix"); - } - char vehicleType = alternative.charAt(0); - if (vehicleType!='L' && vehicleType!='M' && vehicleType!='H' && vehicleType != 'I') { - throw new CoefficientFormatError("Alternative must be L, M, I or H for TravelTimeMatrix"); - } - if (index1.equals("default")) { - travelTimeMatrices.add(new TravelTimeMatrixSpec(matrix, -1, -1, vehicleType)); - } else if (index1.equals("")) { - travelTimeMatrices.add(new TravelTimeMatrixSpec(matrix, Float.valueOf(index2).floatValue(), (float) coefficient, vehicleType)); - } else throw new CoefficientFormatError("Index1 must be \"default\" or blank for TravelTimeMatrices"); - } - - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateCommercialTours.matrixReader); - } - - - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java deleted file mode 100644 index fb7816c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialTripMode.java +++ /dev/null @@ -1,256 +0,0 @@ -package org.sandag.cvm.activityTravel.cvm; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.activityTravel.AlternativeUsesMatrices; -import org.sandag.cvm.activityTravel.CodedAlternative; -import org.sandag.cvm.activityTravel.CoefficientFormatError; -import org.sandag.cvm.activityTravel.TripMode; -import org.sandag.cvm.activityTravel.TripModeChoice; -import org.sandag.cvm.activityTravel.TravelTimeTracker.TravelTimeMatrixSpec; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; - -public class CommercialTripMode extends TripMode implements CodedAlternative , AlternativeUsesMatrices { - - CommercialTravelTimeTracker noTollDistance = new CommercialTravelTimeTracker(); - CommercialTravelTimeTracker noTollTime = new CommercialTravelTimeTracker(); - CommercialTravelTimeTracker tollDistance = new CommercialTravelTimeTracker(); - CommercialTravelTimeTracker totalDistance = new CommercialTravelTimeTracker(); - CommercialTravelTimeTracker tollTime = new CommercialTravelTimeTracker(); - CommercialTravelTimeTracker tollCost = new CommercialTravelTimeTracker(); - private static Logger logger = Logger.getLogger(CommercialTripMode.class); - - CommercialVehicleTripModeChoice getMyCM() { - return (CommercialVehicleTripModeChoice) myChoiceModel; - } - - public CommercialTripMode( - CommercialVehicleTripModeChoice choiceModel, - String type) { - super(choiceModel, type); - try { - switch (myType) { - case "M:T": - case "M:NT": //uses light-heavy truck skims - noTollDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); - noTollDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); - noTollDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); - noTollTime.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TIME", 0); - noTollTime.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TIME", 9); - noTollTime.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TIME", 19); - tollDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); - tollDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); - tollDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); - totalDistance.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_DIST", 0); - totalDistance.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_DIST", 9); - totalDistance.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_DIST", 19); - tollTime.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TIME", 0); - tollTime.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TIME", 9); - tollTime.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TIME", 19); - tollCost.addCoefficient("M", "default", "", "traffic_skims_MD:MD_TRK_L_TOLLCOST", 0); - tollCost.addCoefficient("M", "", "6", "traffic_skims_AM:AM_TRK_L_TOLLCOST", 9); - tollCost.addCoefficient("M", "", "15.5", "traffic_skims_PM:PM_TRK_L_TOLLCOST", 19); - break; - case "H:T": - case "H:NT": //uses heavy-heavy truck skims - noTollDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); - noTollDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); - noTollDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); - noTollTime.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TIME", 0); - noTollTime.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TIME", 9); - noTollTime.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TIME", 19); - tollDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); - tollDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); - tollDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); - totalDistance.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_DIST", 0); - totalDistance.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_DIST", 9); - totalDistance.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_DIST", 19); - tollTime.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TIME", 0); - tollTime.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TIME", 9); - tollTime.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TIME", 19); - tollCost.addCoefficient("H", "default", "", "traffic_skims_MD:MD_TRK_H_TOLLCOST", 0); - tollCost.addCoefficient("H", "", "6", "traffic_skims_AM:AM_TRK_H_TOLLCOST", 9); - tollCost.addCoefficient("H", "", "15.5", "traffic_skims_PM:PM_TRK_H_TOLLCOST", 19); - break; - case "I:T": - case "I:NT": //use medium-heavy truck skims - noTollDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); - noTollDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); - noTollDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); - noTollTime.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TIME", 0); - noTollTime.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TIME", 9); - noTollTime.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TIME", 19); - tollDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); - tollDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); - tollDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); - totalDistance.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_DIST", 0); - totalDistance.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_DIST", 9); - totalDistance.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_DIST", 19); - tollTime.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TIME", 0); - tollTime.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TIME", 9); - tollTime.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TIME", 19); - tollCost.addCoefficient("I", "default", "", "traffic_skims_MD:MD_TRK_M_TOLLCOST", 0); - tollCost.addCoefficient("I", "", "6", "traffic_skims_AM:AM_TRK_M_TOLLCOST", 9); - tollCost.addCoefficient("I", "", "15.5", "traffic_skims_PM:PM_TRK_M_TOLLCOST", 19); - break; - case "L:T": - case "L:NT": //light vehicles use high-VOT auto skims - noTollDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_NT_H_DIST", 0); - noTollDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_NT_H_DIST", 9); - noTollDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_NT_H_DIST", 19); - noTollTime.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_NT_H_TIME", 0); - noTollTime.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_NT_H_TIME", 9); - noTollTime.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_NT_H_TIME", 19); - tollDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_DIST", 0); - tollDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_DIST", 9); - tollDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_DIST", 19); - totalDistance.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_DIST", 0); - totalDistance.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_DIST", 9); - totalDistance.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_DIST", 19); - tollTime.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_TIME", 0); - tollTime.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_TIME", 9); - tollTime.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_TIME", 19); - tollCost.addCoefficient("L", "default", "", "traffic_skims_MD:MD_SOV_TR_H_TOLLCOST", 0); - tollCost.addCoefficient("L", "", "6", "traffic_skims_AM:AM_SOV_TR_H_TOLLCOST", 9); - tollCost.addCoefficient("L", "", "15.5", "traffic_skims_PM:PM_SOV_TR_H_TOLLCOST", 19); - break; - default: - logger.fatal("Invalid tNCVehicle type in trip mode choice model "+myType); - throw new RuntimeException("Invalid tNCVehicle type in trip mode choice model "+myType); - } - } catch (CoefficientFormatError e) { - logger.fatal("Problem setting up coefficeint for trip modes", e); - throw new RuntimeException("Problem setting up coefficient for trip modes", e); - } - } - - @Override - public double getUtility() { - double tollOptTotalDistance = totalDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - double tollPortion = 0; - if (tollOptTotalDistance >0) { - //tollPortion = tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)/tollOptTotalDistance; - tollPortion=0; - } - double tollDisutility = 0; - switch (myType) { - // FIXME parameterize them in the .CSV file - case "L:NT": - return getMyCM().dispersionParam*( - -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.138*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); - - - case "I:NT": - case "M:NT": - return getMyCM().dispersionParam*( - -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.492*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); - - case "H:NT": - return getMyCM().dispersionParam*( - -0.313*noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.580*noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.01 *tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType)); - // return -999.0; - case "L:T": - - //if(toll<0.01 || toll>99999) - return -999.0; - /* - double toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - - tollDisutility = - -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.138*tollOptTotalDistance+ - -0.01 * toll; - return getMyCM().dispersionParam * tollDisutility + - getMyCM().portionParam * tollPortion; */ - case "I:T": - case "M:T": - - //if(toll<0.01 || toll>99999) - return -999.0; - /* - toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - - tollDisutility = - -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.492*tollOptTotalDistance+ - -0.01 * tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - return getMyCM().dispersionParam * tollDisutility + - getMyCM().portionParam * tollPortion; - */ - case "H:T": - - //if(toll<0.01 || toll>99999) - return -999.0; - /* toll = tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - - tollDisutility = - -0.313*tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - -0.580*tollOptTotalDistance+ - -0.01 * tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - return getMyCM().dispersionParam * tollDisutility + - getMyCM().portionParam * tollPortion; - */ - } - String msg = "Invalid tNCVehicle toll trip mode "+myType; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - @Override - public void readMatrices(MatrixCacheReader matrixReader) { - tollCost.readMatrices(matrixReader); - noTollTime.readMatrices(matrixReader); - noTollDistance.readMatrices(matrixReader); - tollDistance.readMatrices(matrixReader); - totalDistance.readMatrices(matrixReader); - tollTime.readMatrices(matrixReader); - } - - @Override - public void addCoefficient(String index1, String index2, String matrix, - double coefficient) throws CoefficientFormatError { - logger.warn("Ignoring coefficeiint "+index1+" "+index2+" "+matrix+" "+coefficient+", trip mode matrix coefficients are still hardcoded"); - } - - public double getTollDistance() { - return tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - } - - public Double getTollTime() { - return tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - } - - public Double getNonTollTime() { - return noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - } - - public String reportAttributes() { - switch (myType) { - // FIXME parameterize them in the .CSV file - case "L:NT": - case "I:NT": - case "M:NT": - case "H:NT": - return "time:"+noTollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - " dist:"+noTollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - case "L:T": - case "I:T": - case "M:T": - case "H:T": - return "time:"+tollTime.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - " dist:"+totalDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - " tolldist:"+tollDistance.getTravelAttribute(origin, destination, timeOfDay, vehicleType)+ - " toll:"+tollCost.getTravelAttribute(origin, destination, timeOfDay, vehicleType); - } - logger.fatal("Oops invalid trip mode for reportATtributes()"); - throw new RuntimeException("Oops invalid trip mode for reportATtributes()"); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java deleted file mode 100644 index 9a15850..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourType.java +++ /dev/null @@ -1,112 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -/* - * Created on Feb 4, 2005 - * - */ -package org.sandag.cvm.activityTravel.cvm; - -import java.util.Enumeration; -import java.util.Hashtable; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; - - -public class CommercialVehicleTourType extends TourType implements AlternativeUsesMatrices { - public CommercialVehicleTourType(CommercialVehicleTourTypeChoice choice, String vehicleTourType) { - super (vehicleTourType, vehicleTourType.charAt(0), (VehicleTourTypeChoice) choice); - myId = vehicleTourType; - } - - final String myId; - - - - @Override - public String toString() { - return "VehicleTourType:"+myId; - - } - - /** - * Method readMatrices. - * @param mr - */ - public void readMatrices(MatrixCacheReader mr) { - utilityFunction.readMatrices(mr); - Enumeration it = conditionalUtilityFunctions.elements(); - while (it.hasMoreElements()) { - IndexLinearFunction bob = (IndexLinearFunction) it.nextElement(); - bob.readMatrices(mr); - } - } - - - - - IndexLinearFunction utilityFunction = new IndexLinearFunction(); - Hashtable conditionalUtilityFunctions = new Hashtable(); - - class TripOutputMatrixSpec { - String name; - Matrix matrix; - float startTime; - float endTime; - } - - public double getUtility() { - double utility = utilityFunction.calcForIndex(this.myChoice.getMyTour().getOriginZone(),1); - Integer myZoneType = new Integer(((CommercialTour) this.myChoice.getMyTour()).getOriginZoneType()); - ZonePairDisutility uf = (ZonePairDisutility) conditionalUtilityFunctions.get(myZoneType); - if (uf != null) { - utility += uf.calcForIndex(this.myChoice.getMyTour().getOriginZone(),1); - } - return utility; - } - - public void addCoefficient(String index1, String index2, String matrix, double coefficient) { - if (index1.equals("origin")) { - utilityFunction.addCoefficient(matrix,coefficient); - } else if (index1.equals("originLU") || index1.equals("originConstant")){ - Integer luCondition = Integer.valueOf(index2); - IndexLinearFunction conditionalUtilityFunction = (IndexLinearFunction) conditionalUtilityFunctions.get(luCondition); - if (conditionalUtilityFunction == null) { - conditionalUtilityFunction = new IndexLinearFunction(); - conditionalUtilityFunctions.put(luCondition,conditionalUtilityFunction); - } - if (index1.equals("originConstant")) { - conditionalUtilityFunction.addConstant(coefficient); - } else if (index1.equals("originLU")) { - conditionalUtilityFunction.addCoefficient(matrix,coefficient); - } - }else if (index1.equals("")) { - utilityFunction.addConstant(coefficient); - } else { - throw new RuntimeException("Invalid index1 for alternative "+getCode()); - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java deleted file mode 100644 index 1715143..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/CommercialVehicleTourTypeChoice.java +++ /dev/null @@ -1,238 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.activityTravel.cvm; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Enumeration; -import java.util.Hashtable; -import java.util.Iterator; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.Alternative; -import org.sandag.cvm.common.model.LogitModel; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class CommercialVehicleTourTypeChoice extends LogitModel implements ModelUsesMatrices, VehicleTourTypeChoice { - - ArrayList alternativesWaitingToBeAddedToNestingStructure = new ArrayList(); - ArrayList allElementalAlternatives = new ArrayList(); - - private void addAlternativeInitially(CommercialVehicleTourType a) { - alternativesWaitingToBeAddedToNestingStructure.add(a); - allElementalAlternatives.add(a); - } - - /** - * Constructor for VehicleTypeChoice. - * @param types - */ - - // tour types are LS,LG,LO,MS,MG,MO,HS,HG,HO - public CommercialVehicleTourTypeChoice(String[] types, boolean nesting) { - super(); - for (String type : types) { - addAlternativeInitially(new CommercialVehicleTourType(this, type)); - } - if (!nesting) { - // no nesting structure specified - for (String type : types ) { - setUpNestingElement(type,"top",1.0); - } - } - - } - - private CommercialTour myTour; - private static Logger logger = Logger.getLogger(CommercialVehicleTourTypeChoice.class); - - /** - * Method addParameter. - * - * @param alternative - * @param matrix - * @param coefficient - * @throws CoefficientFormatError - */ - public void addCoefficient(String alternative, String index1, - String index2, String matrix, double coefficient) - throws CoefficientFormatError { - // if index1 is "nest" we will set up a nesting structure. - if (index1.equalsIgnoreCase("nest")) { - setUpNestingElement(alternative, matrix, coefficient); - } else { - boolean found = false; - Iterator myAltsIterator = allElementalAlternatives.iterator(); - while (myAltsIterator.hasNext()) { - Alternative alt = (Alternative) myAltsIterator.next(); - CommercialVehicleTourType vtt = (CommercialVehicleTourType) alt; - if (vtt.getCode().equals(alternative)) { - vtt.addCoefficient(index1, index2, matrix, coefficient); - found = true; - } - } - if (!found) throw new RuntimeException("can't find alternative "+alternative+" in vehicleTourType model"); - } - } - - private void setUpNestingElement(String alternativeName, String nestName, double coefficient) { - boolean found = false; - // first check if an elemental alternative; - for (int i=0;i myAltsIterator = myLogitModel.getAlternativesIterator(); - while (myAltsIterator.hasNext()) { - Alternative alt = myAltsIterator.next(); - CommercialTripMode tm = (CommercialTripMode) alt; - if (tm.getCode().equals(alternative)) { - tm.addCoefficient(index1, index2, matrix, coefficient); - found = true; - } - } - if (!found) { - if (index1.equalsIgnoreCase("dispersion")) { - dispersionParam = coefficient; - } else if (index1.equalsIgnoreCase("portion")) { - portionParam = coefficient; - } else { - throw new RuntimeException("can't find alternative "+alternative+" in TripMode model"); - } - } - } - - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader matrixCacheReader) { - Iterator myAltsIterator = myLogitModel.getAlternativesIterator(); - while (myAltsIterator.hasNext()) { - CommercialTripMode tm = (CommercialTripMode) myAltsIterator.next(); - tm.readMatrices(matrixCacheReader); - } - } - - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateCommercialTours.matrixReader); - } - - @Override - public TripMode chooseTripModeForDestination(int location) - throws ChoiceModelOverflowException, NoAlternativeAvailable { - CommercialTripMode choice = (CommercialTripMode) debugChooseTripModeForDestination(location); - if (choice.getTripMode().equals("T")) { - // going to check if any tolls are actually paid - double tollDistance = choice.getTollDistance(); - if (tollDistance ==0) { - // get the non toll alternative - Iterator it = myLogitModel.getAlternativesIterator(); - while (it.hasNext()) { - CommercialTripMode tm = (CommercialTripMode) it.next(); - if (tm.getTripMode().equals("NT")) { - choice = tm; - break; - } - } - } else { - // logTollTripChoice(); turn this off for now - } - } - return choice; - } - - - public TripMode debugChooseTripModeForDestination(int location) - throws ChoiceModelOverflowException, NoAlternativeAvailable { - /*// delete this debug stuff - Double tollTime = null; - Double nTollTime = null; - */ - - Iterator m = myLogitModel.getAlternativesIterator(); - while (m.hasNext()) { - CommercialTripMode tm = (CommercialTripMode) m.next(); - tm.setOrigin(getMyTour().getCurrentLocation()); - tm.setDestination(location); - tm.setTime(getMyTour().getCurrentTimeHrs()); - - /* delete this debug stuff - if (tm.getTripMode().equals("T")) { - tollTime = tm.getTollTime(); - } - if (tm.getTripMode().equals("NT")) { - nTollTime = tm.getNonTollTime(); - }*/ - } - - /* delete this debug stuff - if (tollTime==null || nTollTime==null) { - String msg = "Oops couldn't extract toll and non toll times for debugging purposes"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - if (nTollTime < tollTime ) { - logger.warn("nTollTime is less than tollTime for "+getMyTour().getCurrentLocation()+" to "+location); - } - if (nTollTime - tollTime > 5) { - // 5 minute savings! Check it out - logger.info("5 minute savings taking the toll road from "+getMyTour().getCurrentLocation()+" to "+location); - } - */ - return (TripMode) myLogitModel.monteCarloChoice(); - } - - - - protected void logTollTripChoice() { - - StringBuffer msg = new StringBuffer("Toll chosen:"); - msg.append(((CommercialTripMode) myLogitModel.alternativeAt(0)).logOriginDestination()); - msg.append(" for "); - msg.append(theTour.getMyVehicleTourType()); - msg.append(" "); - double utility1= ((CommercialTripMode) myLogitModel.alternativeAt(0)).getUtility(); - double utility2 = ((CommercialTripMode) myLogitModel.alternativeAt(1)).getUtility(); - double prob = Math.exp(utility1)/(Math.exp(utility2)+Math.exp(utility1)); - msg.append(((CommercialTripMode)myLogitModel.alternativeAt(0)).getTripMode()+" prob "+prob+" {"); - msg.append(((CommercialTripMode) myLogitModel.alternativeAt(0)).reportAttributes()); - msg.append(" or "); - msg.append(((CommercialTripMode) myLogitModel.alternativeAt(1)).reportAttributes()); - msg.append("}"); - logger.info(msg); - - } - - - public boolean isTollAvailable(int origin, int destination, double timeOfDay) { - Iterator it = myLogitModel.getAlternativesIterator(); - while (it.hasNext()) { - CommercialTripMode tm = (CommercialTripMode) it.next(); - tm.setOrigin(origin); - tm.setDestination(destination); - tm.setTime(timeOfDay); - if (tm.tripMode.equals("T")) { - if (tm.getTollDistance() ==0) return false; - } else { - return true; - } - } - throw new RuntimeException("No toll option for trip, can't report whether toll was actually available "); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java deleted file mode 100644 index 068aeb5..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/DurationModel2.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.activityTravel.CoefficientFormatError; -import org.sandag.cvm.activityTravel.DurationModel; - -public class DurationModel2 extends DurationModel { - - int functionalForm; - static boolean durationInMinutes = false; - - @Override - public void addCoefficient(String alternative, String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { - if(index1.equals("functionForm")) { - if (index2.equals("power")) functionalForm =1; - else if (index2.equals("cubic")) functionalForm = 2; - else if (index2.equals("exponential")) functionalForm = 3; - else if (index2.equals("addedexponential")) functionalForm = 0; - else throw new CoefficientFormatError("functionalForm for tour start model must have index2 as \"power\", \"cubic\" or \"exponential\""); - } else { - super.addCoefficient(alternative, index1, index2, matrix, coefficient); - } - } - - @Override - public double sampleValue() { - double y=0; - double x = Math.random(); - switch (functionalForm) { - case 0: - y = super.sampleValue(); - break; - case 1: - y = a * Math.pow(x,b)+c*Math.pow(x,d) + e * x + f; - break; - case 2: - y = a+b*x+c*x*x+d*x*x*x; - break; - case 3: - y = c*Math.exp(a*x+b)+d; - break; - default: - throw new RuntimeException("Functional form for duration model must be 0,1,2 or 3"); - } - if (durationInMinutes) y=y/60; - if (y<0) y =0; - if (y>24) y = 24; - return y; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java deleted file mode 100644 index 6be0c18..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/GenerateCommercialTours.java +++ /dev/null @@ -1,855 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - - */ - -package org.sandag.cvm.activityTravel.cvm; - -/** - * @author John Abraham - * - * (c) 2003-2010 John Abraham - */ - -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixAndTAZTableCache; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.skims.HDF5MatrixReader; -import org.sandag.cvm.common.skims.OMXMatrixCollectionReader; -import org.sandag.cvm.common.skims.TranscadMatrixCollectionReader; -import org.sandag.cvm.common.datafile.CSVFileReader; -import org.sandag.cvm.common.datafile.TableDataSet; - -import com.pb.common.matrix.*; -import com.pb.common.util.ResourceUtil; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.activityTravel.cvm.CommercialTour.TripOutputMatrixSpec; - -import java.io.*; -import java.sql.*; -import java.util.*; -import java.util.concurrent.Callable; -import java.util.concurrent.ExecutionException; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.Future; - -import org.apache.log4j.Logger; - - -public class GenerateCommercialTours implements Runnable, Callable { - - private static Logger logger = Logger.getLogger(GenerateCommercialTours.class); - public static MatrixAndTAZTableCache matrixReader; - public static ResourceBundle propsResource; - - final ArrayList tripOutputMatrices = new ArrayList(); - final String segmentString; - final String segmentString1; - final String segmentString2; - - - public GenerateCommercialTours(String genColumn, boolean useTripModes) { - segmentString = genColumn; - String[] strings = genColumn.split("_"); - if (strings.length>0) segmentString1 = strings[0]; - else segmentString1 = ""; - if (strings.length>1) segmentString2 = strings[1]; - else segmentString2 = ""; - this.useTripModes = useTripModes; - } - - /** - * Generate commercial tours for Calgary EMME/2 model - * @param args is six strings, location of databank, location of coefficients file, name of coefficients file, minZone,maxZone, anyOldMatrixName - */ - public static void main(String[] args) { - - // check arguments - if (args.length != 1) { - System.out.println("Usage: java "+GenerateCommercialTours.class.getCanonicalName()+" propertiesFileName"); - throw new RuntimeException("Usage: java "+GenerateCommercialTours.class.getCanonicalName()+" propertiesFileName"); - } - try { - - try { - // basic setup of connections, variables, readers and the like - setupStaticData(args); - - //DurationModel2.durationInMinutes=ResourceUtil.getBooleanProperty(propsResource,"DurationInMinutes", false); - //TourStartTimeModel.startTimeInMinutes=ResourceUtil.getBooleanProperty(propsResource, "StartTimeInMinutes", false); - DurationModel2.durationInMinutes=silentlyCheckBooleanProperty(propsResource,"DurationInMinutes",false); - TourStartTimeModel.startTimeInMinutes=silentlyCheckBooleanProperty(propsResource, "StartTimeInMinutes",false); - - zones = zonalAttributes.getColumnAsInt(zonalAttributes.checkColumnPosition("TAZ")); - - String[] vehTypes = ResourceUtil.checkAndGetProperty(propsResource, "FirstPart").split(","); - String[] timePeriods = ResourceUtil.checkAndGetProperty(propsResource, "SecondPart").split(","); - ArrayList segmentRunners = new ArrayList(); - - if (ResourceUtil.getProperty(propsResource, "RunZones")!=null) { - logger.info("Setting up Run Zones "+ResourceUtil.getProperty(propsResource, "RunZones")); - setUpValidZones(ResourceUtil.getProperty(propsResource, "RunZones")); - } - /*if (ResourceUtil.getProperty(propsResource, "ExcludeZones")!=null) { - logger.info("Excluding Zones "+ResourceUtil.getProperty(propsResource, "ExcludeZones"); - excludeZones(ResourceUtil.getProperty(propsResource, "ExcludeZones")); - }*/ - TreeMap createdMatrices = new TreeMap(); - for (String vehType : vehTypes) { - for (String timePeriod : timePeriods) { - String genColumn = vehType + "_"+ timePeriod; - if (timePeriod.equals("")) { - genColumn = vehType; - timePeriod = null; - } - GenerateCommercialTours generator = new GenerateCommercialTours(genColumn, ResourceUtil.getBooleanProperty(propsResource, "UseTripModes")); - segmentRunners.add(generator); - generator.buildModelStructure(zones,minZone,maxZone,propsResource); - generator.generationEquation = new IndexLinearFunction(); - generator.prevGenerationEquation = new IndexLinearFunction(); - - // read the coefficients and make sure all the data is read in. - try { - readCoefficientFiles(csvInputFileReader, vehType, timePeriod); - } catch (IOException e) { - String msg = "Error reading coefficeint files "+vehType+" and "+timePeriod+ " from "+ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation"); - logger.fatal(msg,e); - throw new RuntimeException(msg,e); - } - generator.setUpCoefficients(args, coefficients); - if (coefficients2!=null) generator.setUpCoefficients(args, coefficients2); - - // TODO move this to coefficent file - generator.setUpDisutilityGetters(); - - if (ResourceUtil.getBooleanProperty(propsResource, "UseSegmentNameInGeneration",true)) { - generator.generationEquation.addCoefficient(genColumn,1); - } - - String tripLogPath = ResourceUtil.getProperty(propsResource, "TripLogPath"); - if (tripLogPath !=null) { - if (!tripLogPath.equals("")) { - String tripLogFile = tripLogPath + - "Trip_"+genColumn+".csv"; - - generator.openTripLog(new File(tripLogFile)); - } - } - generator.readMatrices(); - if (ResourceUtil.getBooleanProperty(propsResource,"ReadOutputMatrices",true)) { - generator.readOutputMatrices(matrixReader); - } else { - generator.createEmptyOutputMatrices(matrixReader.getDim2MatrixSize(),matrixReader.getDim2MatrixExternalNumbers(),createdMatrices); - } - generator.landUseTypeMatrix = matrixReader.readMatrix(generator.landUseTypeMatrixName); - - - } - } - - int nThreads = ResourceUtil.getIntegerProperty(propsResource, "nThreads",8); - - threadpool = Executors.newFixedThreadPool(nThreads); - - List> runners; - try { - runners = threadpool.invokeAll(segmentRunners); - for (Future runner : runners) { - runner.get(); - } - } catch (InterruptedException e) { - logger.fatal("Thread was interrupted",e); - throw new RuntimeException("Thread was interrupted",e); - } catch (ExecutionException e) { - logger.fatal("Exception in one segment",e); - throw new RuntimeException("Exception in one segment",e); - } - - int totalTours = 0; - int totalTrips = 0; - for (GenerateCommercialTours generator : segmentRunners) { - totalTours += generator.totalTours; - totalTrips += generator.totalTrips; - logger.info("Trips for segment "+generator.segmentString); - generator.vehicleTourTypeChoice.writeTourAndTripSummary(); - } - - - logger.info("finished generating "+totalTours+" tours and "+totalTrips+" trips"); - - MatrixWriter writer = null; - if (ResourceUtil.getBooleanProperty(propsResource, "WriteEmmeApiMatrices",false)) { - try { - writer = setUpEmmeApiWriter(); - writeAllMatrices(writer,segmentRunners); - } finally { - if (writer !=null) { - try { - ((EmmeApiMatrixWriter) writer).close(); - } catch (IOException e) { - // oh well - } - } - } - } else if (ResourceUtil.getBooleanProperty(propsResource, "WriteDirectEmmeMatrices",false)) { - //writer = new Emme2MatrixWriter(file) - //writeMatrices(writer); - logger.error("WriteDirectEmmeMatrices is no longer supported, please use WriteEmmeApiMatrices instead. Matrices not written out!"); - } else if (ResourceUtil.getProperty(propsResource, "CSVOutputFileLocation")!=null) { - writer = new CSVMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "CSVOutputFileLocation"))); - writeAllMatrices(writer, segmentRunners); - } else if (ResourceUtil.getProperty(propsResource, "TranscadCVMMatrixFile")!=null) { - writer = new TranscadMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "TranscadCVMMatrixFile"))); - writeAllMatrices(writer, segmentRunners); - } else { - writer = new CSVMatrixWriter(new File(ResourceUtil.getProperty(propsResource, "CSVFileLocation")+File.pathSeparator+"TripMatrices.csv")); - writeAllMatrices(writer,segmentRunners); - } - - logger.info("Finished writing matrices"); - - - } catch (Throwable e) { - logger.fatal("Error in CVM program", e); - } - } finally { - if (threadpool!=null) { - threadpool.shutdown(); - } - if (matrixReader!=null) { - MatrixReader actualReader = matrixReader.getActualReader(); - if (actualReader instanceof EmmeApiMatrixReader) { - try { - ((EmmeApiMatrixReader) actualReader).close(); - } catch (IOException e) { - logger.warn("IOException trying to close EmmeApiMatrixReader",e); - } - } - } - } - } - - - - private static boolean silentlyCheckBooleanProperty( - ResourceBundle propsResource2, String name, boolean defaultVal) { - String str = ResourceUtil.getProperty(propsResource2, name); - if (str==null) return defaultVal; - if (str.equals("")) return defaultVal; - if (str.equalsIgnoreCase("True")) return true; - if (str.equalsIgnoreCase("False")) return false; - logger.error(name + "property should be boolean, it is "+name); - return defaultVal; - } - - private HashMap travelDisutilityTrackers = null; - - private void readMatrices() { - Iterator modelIterator = models.values().iterator(); - while (modelIterator.hasNext()) { - ModelUsesMatrices model = (ModelUsesMatrices) modelIterator.next(); - model.readMatrices(matrixReader); - } - generationEquation.readMatrices(matrixReader); - prevGenerationEquation.readMatrices(matrixReader); - } - - VehicleTourTypeChoice vehicleTourTypeChoice; - private final boolean useTripModes; - /** - * @return Returns the vehicleTourTypeChoice. - */ - public VehicleTourTypeChoice getVehicleTourTypeChoice() { - return vehicleTourTypeChoice; - } - - /** - * @param vehicleTourTypeChoice The vehicleTourTypeChoice to set. - */ - public void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoiceParam) { - vehicleTourTypeChoice = vehicleTourTypeChoiceParam; - } - - - - private void generateTheTours() { - vehicleTourTypeChoice=(CommercialVehicleTourTypeChoice) models.get("VehicleTourType"); - setTourStartTimeModel((TourStartTimeModel) models.get("TourStartTime")); - setElapsedTravelTimeCalculator((CommercialTravelTimeTracker) models.get("TravelTimeMatrix")); - totalTours = 0; - int oldTotalTours = 0; - totalTrips = 0; - System.out.println("Generating tours..."); - for (int z = 1;z 100) { - String msg = null; - if (totalTours - oldTotalTours == tours) - msg = "Generating "+tours+" "+segmentString+" tours in zone "+zone+" ... "; - else - msg = "Generated "+(totalTours - oldTotalTours)+" "+segmentString+"tours, including "+tours+" currently being generated in zone "+zone; - logger.info(msg); - oldTotalTours = totalTours; - } - int tripCount = 0; - for (int tour = 0; tour < tours; tour ++) { - CommercialTour t = new CommercialTour(models,this,tripLog,getNextTourNumber()); - t.setOrigin(zone); - t.sampleStartTime(); - t.sampleVehicleAndTourType(); - t.sampleStops(); - t.addTripsToMatrix(); - tripCount += t.getStopCounts()[0]; - } - if (logger.isDebugEnabled()) logger.debug(tripCount+" trips generated from "+segmentString+" tours in zone "+zone+" ... "); - totalTrips += tripCount; - } - - } - } - - private void setUpDisutilityGetters() { - if (isUseTripModes()) { - // TODO this shouldn't be hardcoded - HashMap trackers = new HashMap(); - trackers.put("L", (ChangingTravelAttributeGetter) models.get("TripModeL")); - trackers.put("I", (ChangingTravelAttributeGetter) models.get("TripModeI")); - trackers.put("M", (ChangingTravelAttributeGetter) models.get("TripModeM")); - trackers.put("H", (ChangingTravelAttributeGetter) models.get("TripModeH")); - setTravelDisutilityTrackers(trackers); - } else { - HashMap trackers = new HashMap(); - trackers.put("",((CommercialTravelTimeTracker) models.get("TravelDisutilityMatrix"))); - } - } - - static TreeSet zoneSet = null; - - private boolean isValidZone(int zone) { - if (zone maxZone) return false; - if (zoneSet==null) return true; - if (zoneSet.contains(zone)) return true; - return false; - } - - - private static void setUpValidZones(String property) { - String[] zoneIds = property.split(","); - zoneSet = new TreeSet(); - for (String zoneId : zoneIds) { - zoneSet.add(Integer.valueOf(zoneId)); - } - } - - - - private static void setupStaticData(String[] args) { - try { - propsResource = new PropertyResourceBundle(new FileInputStream(args[0])); - } catch (FileNotFoundException e2) { - logger.fatal("Can't find file "+args[0]); - throw new RuntimeException("Can't find file "+args[0], e2); - } catch (IOException e2) { - logger.fatal("Can't read file "+args[1]); - throw new RuntimeException("Can't read file "+args[0], e2); - } - csvInputFileReader = new CSVFileReader(); - csvInputFileReader.setPadNulls(true); - csvInputFileReader.setMyDirectory(ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation")); - try { - logger.info("Reading ZonalProperties"); - zonalAttributes = csvInputFileReader.readTable(ResourceUtil.getProperty(propsResource, "ZonalPropertiesFileName", "ZonalProperties.csv ")); - String file2 = ResourceUtil.getProperty(propsResource, "ZonalPropertiesFileName2"); - if (file2 !=null) { - logger.info("Reading ZonalProperties file 2"); - TableDataSet attributes2 = csvInputFileReader.readTable(file2); - appendNewDataSet(zonalAttributes, attributes2, "TAZ"); - } else { - logger.info("No ZonalPropertiesFileName2, just using one zonal properties file"); - } - } catch (IOException e1) { - String msg = "Error reading zonal properties files from "+ResourceUtil.checkAndGetProperty(propsResource,"CSVFileLocation")+" this might be ok if your zonal properties are stored with your matrices"; - logger.warn(msg,e1); - } - - minZone = ResourceUtil.getIntegerProperty(propsResource, "StartZone"); - maxZone = ResourceUtil.getIntegerProperty(propsResource, "EndZone"); - - if(ResourceUtil.getProperty(propsResource, "SkimDatabase") !=null) setUpDatabaseSkims(); - else if (ResourceUtil.getProperty(propsResource, "EmmeUserInitials") != null) setUpEmmeApiSkims(); - else if (ResourceUtil.getProperty(propsResource, "TranscadSkimLocation") != null) setUpTranscadSkims(); - else if (ResourceUtil.getProperty(propsResource, "OMXSkimLocation") != null) setUpOMXSkims(); - else setUpHDF5Skims(); - - } - - private static void setUpTranscadSkims() { - matrixReader = new MatrixAndTAZTableCache( - new TranscadMatrixCollectionReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "TranscadSkimLocation"))), zonalAttributes); - } - - private static void setUpOMXSkims() { - matrixReader = new MatrixAndTAZTableCache( - new OMXMatrixCollectionReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "OMXSkimLocation"))), zonalAttributes); - } - - private static void setUpHDF5Skims() { - String skimNameString = ResourceUtil.checkAndGetProperty(propsResource, "SkimNames"); - String[] skimNames = skimNameString.split(" *, *"); - - String nodeNameString = ResourceUtil.checkAndGetProperty(propsResource, "SkimFileNodeNames"); - String[] nodeNames = nodeNameString.split(" *, *"); - matrixReader = new MatrixAndTAZTableCache( - new HDF5MatrixReader(new File(ResourceUtil.checkAndGetProperty(propsResource, "SkimFile")), - nodeNames, - skimNames), zonalAttributes); - } - - private static void setUpEmmeApiSkims() { - String initials = ResourceUtil.checkAndGetProperty(propsResource, "EmmeUserInitials"); - String emmeBank = ResourceUtil.checkAndGetProperty(propsResource, "EmmeBank"); - String iks = ResourceUtil.getProperty(propsResource, "EmmeIKS"); - try { - if (iks==null) { - matrixReader = new MatrixAndTAZTableCache( - new EmmeApiMatrixReader(initials, emmeBank), zonalAttributes); - } else { - matrixReader = new MatrixAndTAZTableCache( - new EmmeApiMatrixReader(initials, emmeBank, iks, "", false), zonalAttributes); - } - } catch (IOException e) { - String msg = "Couldn't open the emme 2 databank "+emmeBank; - logger.fatal(msg,e); - throw new RuntimeException(msg,e); - } - } - - private static EmmeApiMatrixWriter setUpEmmeApiWriter() { - if (matrixReader.getActualReader() instanceof EmmeApiMatrixReader) { - return new EmmeApiMatrixWriter((EmmeApiMatrixReader) matrixReader.getActualReader()); - } else { - EmmeApiMatrixWriter writer; - String initials = ResourceUtil.checkAndGetProperty(propsResource, "EmmeUserInitials"); - String emmeBank = ResourceUtil.checkAndGetProperty(propsResource, "EmmeBank"); - String iks = ResourceUtil.getProperty(propsResource, "EmmeIKS"); - try { - if (iks==null) { - writer = new EmmeApiMatrixWriter(initials, emmeBank); - } else { - writer = new EmmeApiMatrixWriter(initials, emmeBank, iks, "", false); - } - } catch (IOException e) { - String msg = "Couldn't open the emme 2 databank "+emmeBank; - logger.fatal(msg,e); - throw new RuntimeException(msg,e); - } - return writer; - } - } - - - private static void setUpDatabaseSkims() { - Connection conn; - try { - conn = conn=DriverManager.getConnection( - ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabase"), - ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabaseUser"), - ResourceUtil.checkAndGetProperty(propsResource, "SkimDatabasePassword")); - } catch (SQLException e1) { - String msg = "Can't open skim database"; - logger.fatal(msg,e1); - throw new RuntimeException(msg,e1); - } - logger.info("Reading SQL Matrices"); - matrixReader=new MatrixAndTAZTableCache(new SQLMatrixReader(conn, - ResourceUtil.checkAndGetProperty(propsResource, "SkimQuery"), - ResourceUtil.checkAndGetProperty(propsResource, "OriginQuery"), - ResourceUtil.checkAndGetProperty(propsResource, "DestinationQuery") - ), zonalAttributes); - } - - private static void readCoefficientFiles(CSVFileReader reader, String name1, String name2) - throws IOException { - coefficients = reader.readTable(name1); - if (name2 != null) { - coefficients2 = reader.readTable(name2); - } else { - logger.info("No CoefficientFileName2, just using one coefficient file"); - } - } - - /** - * Takes the columns from one dataset and appends them to another dataset based on a common integer index column. - * This uses the indexing feature in the TableDataSet and so creates a new row index on the dataset using - * the common column name. If you are relying on the row indexing feature from a different column, you'll - * need to reindex it after. - * @param datasetToAppendTo the dataset to be modified by appending the new columns - * @param newData the dataset containing the new data to be appended to the other data set - * @param commonIndexColumn the name of the common index column - */ - private static void appendNewDataSet(TableDataSet datasetToAppendTo, TableDataSet newData, String commonIndexColumn) { - int originalTazColumnNum = datasetToAppendTo.checkColumnPosition(commonIndexColumn); - datasetToAppendTo.buildIndex(originalTazColumnNum); - int tazColumn = newData.checkColumnPosition(commonIndexColumn); - for (int column = 1; column <= newData.getColumnCount(); column++) { - if (column != tazColumn) { - String columnName = newData.getColumnLabel(column); - if (datasetToAppendTo.getColumnPosition(columnName)!=-1) { - String msg = "Duplicate column name "+columnName+" in first and second dataset"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - float[] newColumnVals = new float[datasetToAppendTo.getRowCount()]; - datasetToAppendTo.appendColumn(newColumnVals, columnName); - int newColumnNumber = datasetToAppendTo.checkColumnPosition(columnName); - for (int row = 1; row <= newData.getRowCount();row++) { - datasetToAppendTo.setIndexedValueAt(((int) newData.getValueAt(row, tazColumn)),newColumnNumber,newData.getValueAt(row,column)); - } - } - } - } - - private void setUpCoefficients(String[] args, - TableDataSet coefficients) { - try { - for (int row=1;row<=coefficients.getRowCount();row++) { - String modelName = coefficients.getStringValueAt(row,"Model"); - String matrix = coefficients.getStringValueAt(row,"Matrix"); - String alternative = coefficients.getStringValueAt(row,"Alternative"); - String index1 = coefficients.getStringValueAt(row,"Index1"); - String index2 = coefficients.getStringValueAt(row,"Index2"); - double coefficient = coefficients.getValueAt(row,"Value"); - if (modelName.equals("Generation")) { - if (index1.equalsIgnoreCase("origin")) { - generationEquation.addCoefficient(matrix,coefficient); - } else if (index1.equalsIgnoreCase("moms")) { - generationEquation.addCoefficient(matrix,index2); - } else { - throw new CoefficientFormatError("generation equations must have origin or MOMS in Index1"); - } - } else if (modelName.equalsIgnoreCase("PrevGeneration")) { - if (index1.equalsIgnoreCase("origin")) { - prevGenerationEquation.addCoefficient(matrix,coefficient); - } else if (index1.equalsIgnoreCase("moms")) { - prevGenerationEquation.addCoefficient(matrix,index2); - } else throw new CoefficientFormatError("generation equations must have origin or MOMS in Index1"); - - } else if (modelName.equals("TripMatrix")) { - if (alternative.contains(":")) { - String[] vehicleTour = alternative.split(":"); - tripOutputMatrices.add(new CommercialTour.TripOutputMatrixSpec(matrix, index1, Float.valueOf(index2).floatValue(), (float) coefficient, vehicleTour[0].charAt(0),vehicleTour[1])); - } else { - tripOutputMatrices.add(new CommercialTour.TripOutputMatrixSpec(matrix, index1, Float.valueOf(index2).floatValue(), (float) coefficient, alternative.charAt(0),"")); - } - } else if (modelName.equals("LandUseTypeCode")) { - setLandUseTypeMatrixName(matrix); - } else { - ModelUsesMatrices model = (ModelUsesMatrices) models.get(modelName); - if (model == null) throw new CoefficientFormatError("Bad model type in coefficients: "+modelName); - model.addCoefficient(alternative,index1,index2,matrix,coefficient); - } - - } - } catch (CoefficientFormatError e) { - System.out.println("Coefficient format error -- you have an invalid coefficient"); - System.out.println(e.toString()); - System.out.println("Aborting..."); - throw new RuntimeException("Coefficient format error -- you have an invalid coefficient",e); - } - } - - void setLandUseTypeMatrixName(String landUseTypeMatrixName) { - this.landUseTypeMatrixName = landUseTypeMatrixName; - } - - - /** - * Method buildModelStructure. - * @param propsResource2 - */ - private void buildModelStructure(int[] zoneNums, int lowNumber, int highNumber, ResourceBundle propsResource) { - models.put("VehicleTourType",new CommercialVehicleTourTypeChoice( - ResourceUtil.getProperty(propsResource,"VehicleTourTypes", - "LS,LG,LO,MS,MG,MO,IS,IG,IO,HS,HG,HO").split(","), - ResourceUtil.getBooleanProperty(propsResource,"NestingVTTChoice",true))); - - // don't use these anymore because tNCVehicle and tour type are joint in the same alternative. - // models.put("LTour", new TourTypeChoice()); - // models.put("MTour", new TourTypeChoice()); - // models.put("HTour", new TourTypeChoice()); - String stopTypeModelString = ResourceUtil.getProperty(propsResource, "StopTypeModels", - "LSStopType, LGStopType, LOStopType, MSStopType, MGStopType, MOStopType, ISStopType, IGStopType, IOStopType, HSStopType, HGStopType, HOStopType"); - String[] stopTypeModels = stopTypeModelString.split(","); - for (String stopTypeModel : stopTypeModels) { - models.put(stopTypeModel.trim(), new CommercialNextStopPurposeChoice(stopTypeModel.trim().charAt(1))); - } - - String[] stopLocationModels = ResourceUtil.getProperty(propsResource, "StopLocationModels", - "LSSStopLocation, LSOStopLocation, LGGStopLocation,LGOStopLocation,LOOStopLocation,MSSStopLocation,MSOStopLocation,MGGStopLocation,MGOStopLocation,MGOStopLocation,MOOStopLocation,ISSStopLocation,ISOStopLocation,IGGStopLocation,IGOStopLocation,IOOStopLocation,HSSStopLocation,HSOStopLocation,HGGStopLocation,HGOStopLocation,HOOStopLocation") - .split(","); - for (String stopLocationModel : stopLocationModels) { - models.put(stopLocationModel.trim(), new CommercialNextStopChoice(zoneNums, lowNumber-1, highNumber+1, stopLocationModel.trim())); - } - - String[] durationModels = ResourceUtil.getProperty(propsResource, "DurationModels", - "LODuration,LSDuration,LGDuration,MODuration,MSDuration,MGDuration,IODuration,ISDuration,IGDuration,HODuration,HSDuration,HGDuration") - .split(","); - for (String durationModel : durationModels) { - models.put(durationModel.trim(), new DurationModel2()); - } - - //TODO here is the place where we parameterize the trip mode choice - if (isUseTripModes()) { - /*String[] tripModeChoiceModels = ResourceUtil.getProperty(propsResource, "TripModeL,TripModelM,TripModeI,TripModeH") - .split(","); - for (String tripModeChoiceModel : tripModeChoiceModels) { - models.put(tripModeChoiceModel.trim(), new CommercialVehicleTripModeChoice(tripModeChoiceModels)); - }*/ - models.put("TripModeL", new CommercialVehicleTripModeChoice(new String[] {"L:T", "L:NT"})); - models.put("TripModeM", new CommercialVehicleTripModeChoice(new String[] {"M:T", "M:NT"})); - models.put("TripModeI", new CommercialVehicleTripModeChoice(new String[] {"I:T", "I:NT"})); - models.put("TripModeH", new CommercialVehicleTripModeChoice(new String[] {"H:T", "H:NT"})); - - } - - models.put("TourStartTime", new TourStartTimeModel()); - models.put("TravelTimeMatrix", new CommercialTravelTimeTracker()); - models.put("TravelDisutilityMatrix", new CommercialTravelTimeTracker()); - - ModelUsesMatrices dummyModel = new ModelUsesMatrices() { - public void readMatrices(MatrixCacheReader matrixReader) { - // nothing - } - public void addCoefficient(String alternative, String index1, - String index2, String matrixName, double coefficient) - throws CoefficientFormatError { - // nothing - } - public void init() { - // nothing - } - - }; - - // dummy models, placeholders for coefficients handled elsewhere (in python) - models.put("ShipNoShip", dummyModel); - models.put("GenPerEmployee", dummyModel); - models.put("TourTOD", dummyModel); - - } - - final Hashtable models = new Hashtable(); - private static int minZone; - private static int maxZone; - static TableDataSet coefficients; - static TableDataSet coefficients2; - static TableDataSet zonalAttributes; - private int totalTours; - private int totalTrips; - private IndexLinearFunction generationEquation; - private IndexLinearFunction prevGenerationEquation; - private static int globalTourNumber = 0; - private static int[] zones; - private static CSVFileReader csvInputFileReader; - private PrintWriter tripLog; - - - public static int getNextTourNumber() { - synchronized(GenerateCommercialTours.class){ - return globalTourNumber ++; - } - } - - void openTripLog(File tripLogFile) { - try { - logger.info("Opening log file "+tripLogFile); - tripLog = new PrintWriter(new FileWriter(tripLogFile)); - tripLog.println("Model,SerialNo,Person,Trip,Tour,HomeZone,ActorType,OPurp,DPurp,I,J,TripTime,Mode,StartTime,EndTime,StopDuration,TourType,OriginalTimePeriod,TripMode,TollAvailable"); - } catch (IOException e) { - logger.fatal("Can't open trip log file "+tripLogFile,e); - throw new RuntimeException("Can't open trip log file "+tripLogFile, e); - } - } - - void closeTripLog() { - if (tripLog !=null) { - tripLog.close(); - logger.info("Closing trip log file for "+segmentString); - } - } - - - String landUseTypeMatrixName; - Matrix landUseTypeMatrix; - private TourStartTimeModel tourStartTimeModel; - private ChangingTravelAttributeGetter elapsedTravelTimeCalculator; - private static ExecutorService threadpool; - - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readOutputMatrices(MatrixCacheReader matrixReader) { - for (int i =0; i createdOnes) { - for (TripOutputMatrixSpec s : tripOutputMatrices) { - if (s.write) { - Matrix m = createdOnes.get(s.name); - if (m==null) { - s.createMatrix(size, externalNumbers); - createdOnes.put(s.name, s.matrix); - } else { - s.matrix = m; - } - } - } - } - - /** - * Method writeMatrices. - * @param matrixWriter - */ - void writeMatrices(MatrixWriter matrixWriter) { - ArrayList names = new ArrayList(); - ArrayList m = new ArrayList(); - for (int i =0; i0) { - matrixWriter.writeMatrices(names.toArray(new String[names.size()]), m.toArray(new Matrix[m.size()])); - } - } - - static void writeAllMatrices(MatrixWriter matrixWriter, ArrayList segmentRunners ) { - ArrayList names = new ArrayList(); - ArrayList m = new ArrayList(); - for (GenerateCommercialTours segment : segmentRunners) { - ArrayList tripSpecs = segment.tripOutputMatrices; - for (int i =0; i0) { - StringBuffer msg = new StringBuffer("Writing out matrices "); - for (String name : names) msg.append(","+name); - logger.info(msg); - matrixWriter.writeMatrices(names.toArray(new String[names.size()]), m.toArray(new Matrix[m.size()])); - } - } - - - - TripOutputMatrixSpec getTripOutputMatrixSpec(char vehicleType, String tripMode, float time) { - while (time>=24.00) time-=24.00; - for (int i =0; i=s.startTime && time < s.endTime && s.vehicleType==vehicleType && s.tripMode.equals(tripMode)) return s; - } - return null; - } - - @Override - public void run() { - logger.info("Starting segment "+segmentString); - generateTheTours(); - closeTripLog(); - System.out.println("DonE segement "+segmentString +"!"); - logger.info("DonE segement "+segmentString +"!"); - if (tripLog!=null) tripLog.close(); - } - - private void setElapsedTravelTimeCalculator( - // TODO should use trip modes like getTravelDisutilityTracker does - ChangingTravelAttributeGetter elapsedTravelTimeCalculator) { - this.elapsedTravelTimeCalculator = elapsedTravelTimeCalculator; - } - - public ChangingTravelAttributeGetter getElapsedTravelTimeCalculator() { - return elapsedTravelTimeCalculator; - } - - private void setTourStartTimeModel(TourStartTimeModel tourStartTimeModel) { - this.tourStartTimeModel = tourStartTimeModel; - } - - public TourStartTimeModel getTourStartTimeModel() { - return tourStartTimeModel; - } - - @Override - public Object call() throws Exception { - run(); - return null; - } - - ChangingTravelAttributeGetter getTravelDisutilityTracker() { - if (!isUseTripModes()) - return travelDisutilityTrackers.get(""); - else { - String msg = "Trip mode is being used, no defualt travel disutility tracker"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - } - - void setTravelDisutilityTrackers(HashMap trackers) { - this.travelDisutilityTrackers = trackers; - } - - public ChangingTravelAttributeGetter getTravelDisutilityTracker( - String vehicleCode) { - return travelDisutilityTrackers.get(vehicleCode); - - } - - public boolean isUseTripModes() { - return useTripModes; - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java deleted file mode 100644 index 0dfc727..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/TourStartTimeModel.java +++ /dev/null @@ -1,115 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class TourStartTimeModel implements ModelUsesMatrices, RealNumberDistribution { - - /** returns tour start time in hours since midnight - */ - static boolean startTimeInMinutes = false; - - double periodStart; // base time for the TourStartTimeModel - int functionalForm; // power =1, cubic=2 or exponential =3 - double a; - double b; - double c; - double d; - double e; - double f; - - - public double sampleValue() { - double x = Math.random(); - double y=0; - switch (functionalForm) { - case 1: - y = a * Math.pow(x,b)+c*Math.pow(x,d) + e * x + f; - break; - case 2: - y = a+b*x+c*x*x+d*x*x*x; - break; - case 3: - y = c*Math.exp(a*x+b)+d; - break; - } - if (startTimeInMinutes) y = y/60; - if (y<0) y =0; - if (y>24) y = 24; - y += periodStart; - return y; - } - - - /** - * Method addCoefficient. - * @param alternative - * @param index1 - * @param index2 - * @param matrix - * @param coefficient - */ - public void addCoefficient ( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError { - if (index1.equals("a")) a = coefficient; - else if(index1.equals("b")) b = coefficient; - else if(index1.equals("c")) c = coefficient; - else if(index1.equals("d")) d = coefficient; - else if(index1.equals("e")) e = coefficient; - else if(index1.equals("f")) f = coefficient; - else if(index1.equals("functionForm")) { - if (index2.equals("power")) functionalForm =1; - else if (index2.equals("cubic")) functionalForm = 2; - else if (index2.equals("exponential")) functionalForm = 3; - else throw new CoefficientFormatError("functionalForm for tour start model must have index2 as \"power\", \"cubic\" or \"exponential\""); - } - else if (index1.equals("periodStart")) periodStart=coefficient; - else throw new CoefficientFormatError("Tour start time model model coefficients must have index1 as a,b,c,d,e,f, periodStart or functionalForm"); - } - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader matrixReader) {} - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateCommercialTours.matrixReader); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java b/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java deleted file mode 100644 index 25df5d7..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/activityTravel/cvm/VehicleTourTypeNest.java +++ /dev/null @@ -1,16 +0,0 @@ -package org.sandag.cvm.activityTravel.cvm; - -import org.sandag.cvm.common.model.LogitModel; - -public class VehicleTourTypeNest extends AlogitLogitModelNest { - final String myCode; - - public VehicleTourTypeNest(String code) { - myCode = code; - } - - public String getCode() { - return myCode; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java deleted file mode 100644 index 2c97491..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/GenerateWeekendTours.java +++ /dev/null @@ -1,292 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.calgary.weekend; - -import com.pb.common.matrix.*; -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.*; - -import java.io.*; -import java.sql.*; -import java.util.*; -import org.sandag.cvm.common.datafile.*; -import org.apache.log4j.Logger; - -public class GenerateWeekendTours { - - private static Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); - public static MatrixCacheReader matrixReader; - public static TableDataSetCollection inputData; - static Properties props = new Properties(); - - - /** - * Generate commercial tours for Calgary EMME/2 model - * @param args is six strings, location of databank, location of coefficients file, name of coefficients file, minZone,maxZone, anyOldMatrixName - */ - public static void main(String[] args) { - - FileInputStream fin = null; - try { - fin = new FileInputStream(args[0]); - } catch (FileNotFoundException e) { - System.out.println("error: be sure to put the location of the properties file on the command line"); - e.printStackTrace(); - System.exit(-1); - } - try { - props.load(fin); - } catch (IOException e) { - System.out.println("Error reading properties file "+args[0]); - System.exit(-1); - } - - // get emme/2 matrix - try { - matrixReader = new MatrixCacheReader(new Emme2MatrixReader(new File(props.getProperty("databank")))); - } catch (Exception e) { - System.out.println("Error opening emme2 databank \""+props.getProperty("databank")+"\""); - System.out.println(e); - e.printStackTrace(); - } - Matrix anyOldMatrix = matrixReader.readMatrix(props.getProperty("anyOldMatrixName")); - minZone = Integer.valueOf(props.getProperty("minZone")).intValue(); - maxZone = Integer.valueOf(props.getProperty("maxZone")).intValue(); - //int[] zones = anyOldMatrix.getExternalNumbers(); - buildModelStructure(matrixReader,anyOldMatrix,minZone,maxZone); - - // These next lines are just for testing in the case that you don't have an emme2 databank to load -// int[] zones = new int[10]; -// int minZone = 0; -// int maxZone = 10; -// buildModelStructure(null,minZone,maxZone); - - String inputLocation = props.getProperty("inputLocation"); - CSVFileReader inputDataReader = new CSVFileReader(); - inputDataReader.setMyDirectory(inputLocation); - // no output location for now - inputData = new TableDataSetCollection(inputDataReader, null); - - - TableDataSet coefficients = inputData.getTableDataSet(props.getProperty("coefficientsTable")); - - - try { - for (int cn= 1; cn <= coefficients.getRowCount(); cn++) { //cn is "coefficientNumber" - String modelName = coefficients.getStringValueAt(cn, "Model"); - String matrix = coefficients.getStringValueAt(cn,"Matrix"); - String alternative = coefficients.getStringValueAt(cn,"Alternative"); - String index1 = coefficients.getStringValueAt(cn,"Index1"); - String index2 = coefficients.getStringValueAt(cn,"Index2"); - double coefficient = coefficients.getValueAt(cn,"Value"); - ModelWithCoefficients model = (ModelWithCoefficients) models.get(modelName); - if (model == null) throw new CoefficientFormatError("Bad model type in coefficients: "+modelName); - model.addCoefficient(alternative,index1,index2,matrix,coefficient); - - } - } catch (CoefficientFormatError e) { - System.out.println("Coefficient format error -- you have an invalid coefficient"); - System.out.println(e.toString()); - e.printStackTrace(); - System.out.println("Aborting..."); - System.exit(-1); - } - - Iterator modelIterator = models.values().iterator(); - while (modelIterator.hasNext()) { - ModelWithCoefficients model = (ModelWithCoefficients) modelIterator.next(); - model.init(); - } - -// generationEquation.readMatrices(matrixReader); -// WeekendTour.readMatrices(matrixReader); -// -// CommercialTour.tourTypeChoiceModel=(CommercialVehicleTourTypeChoice) models.get("VehicleTourType"); -// CommercialTour.setTourStartTimeModel((TourStartTimeModel) models.get("TourStartTime")); -// CommercialTour.setElapsedTravelTimeCalculator((WeekendTravelTimeTracker) models.get("TravelTimeMatrix")); -// CommercialTour.travelDisutilityTracker = (WeekendTravelTimeTracker) models.get("TravelDisutilityMatrix"); - - //TODO the household dataset is probably too big to load in the whole thing, so instead step through the file one record at a time - TableDataSet households = inputData.getTableDataSet(props.getProperty("householdsTable")); - TableDataSet householdDetails = inputData.getTableDataSet(props.getProperty("householdDetailsTable")); - householdDetails.buildIndex(householdDetails.checkColumnPosition("hh_ID")); - // TODO read in person file - - int totalTours = 0; - int totalTrips = 0; - System.out.println("Generating tours..."); - for (int hhnum = 1;hhnum<=households.getRowCount();hhnum++) { - if (hhnum %10 == 0) System.out.print(hhnum+" "); - if (hhnum %200 == 0) System.out.println(); - - - WeekendHousehold household = new WeekendHousehold(households, hhnum, householdDetails); - // TODO should add people by reading them from a dataset - household.addPeople(); - - int zone = household.getHomeZone(); - - // TODO check if to generate tours from externals? Probably not. - if (zone>=minZone && zone <= maxZone) { - household.resetCurrentTime(); - WeekendTour tour = household.sampleNextWeekendTour(); - while (tour != null) { - totalTours++; - // TODO add the trips to the matrix; - // t.addTripsToMatrix(); - // TODO write the tour to the tour database - totalTrips += tour.getStopCounts()[0]; - tour = household.sampleNextWeekendTour(); - } - } - - } - System.out.println(); - - - System.out.println("finished generating "+totalTours+" tours and "+totalTrips+" trips , now writing trip matrices out to emme2 databank"); -// WeekendTour.tourTypeChoiceModel.writeTourAndTripSummary(); - Emme2MatrixWriter matrixWriter = new Emme2MatrixWriter(new File(props.getProperty("databank"))); - // TODO write out all of the trip matrices back into the emme2 databank -// WeekendTour.writeMatrices(matrixWriter); - System.out.println("done!"); - } - - /** - * Method buildModelStructure. - */ - private static void buildModelStructure(MatrixCacheReader matrixReader2, Matrix anyOldMatrix, int lowNumber, int highNumber) { - - TourInTimeBand titb = new TourInTimeBand(); - WeekendTour.setTourInTimeBand(titb); - models.put("tourInBand",titb); - NextWeekendTourStartTime ttnt = new NextWeekendTourStartTime(); - WeekendTour.setTourStartTimeModel(ttnt); - models.put("tourStart", ttnt); - WeekendTourTypeChoice wttc = new WeekendTourTypeChoice(); - wttc.setMatrixReader(matrixReader2); - models.put("tourType", wttc); - WeekendTour.tourTypeChoiceModel= wttc; - models.put("workPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); - models.put("schoolPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("relCivicPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("exercisePrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("outOfTownPrimaryStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("workIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); - models.put("schoolIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("relCivicIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("exerciseIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("outOfTownIntermediateStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("workReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone) ); - models.put("schoolReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("relCivicReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("exerciseReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("outOfTownReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("SELSEReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("chaufReturnStop", new WeekendStopChoice(anyOldMatrix,minZone,maxZone)); - models.put("workStopType", new WeekendStopPurposeChoice("workStopType")); - models.put("schoolStopType", new WeekendStopPurposeChoice("schoolStopType")); - models.put("exerciseStopType", new WeekendStopPurposeChoice("exerciseStopType")); - models.put("relCivicStopType", new WeekendStopPurposeChoice("relCivicStopType")); - models.put("outOfTownStopType", new WeekendStopPurposeChoice("outOfTownStopType")); - models.put("chaufStopType", new WeekendStopPurposeChoice("chaufStopType")); - models.put("SELSEStopType", new WeekendStopPurposeChoice("SELSEStopType")); - models.put("workDuration", new DurationModel()); - models.put("shopDuration", new DurationModel()); - models.put("relCivicDuration", new DurationModel()); - models.put("eatDuration", new DurationModel()); - models.put("entLeisureDuration", new DurationModel()); - models.put("socialDuration", new DurationModel()); - models.put("exerciseDuration", new DurationModel()); - models.put("schoolDuration", new DurationModel()); - models.put("outOfTownDuration", new DurationModel()); - models.put("pickUpDuration", new DurationModel()); - models.put("dropOffDuration", new DurationModel()); - - // models.put("VehicleTourType",new CommercialVehicleTourTypeChoice()); -//// models.put("LTour", new TourTypeChoice()); -//// models.put("MTour", new TourTypeChoice()); -//// models.put("HTour", new TourTypeChoice()); -// models.put("LSStopType", new CommercialNextStopPurposeChoice('S')); -// models.put("LGStopType", new CommercialNextStopPurposeChoice('G')); -// models.put("LOStopType", new CommercialNextStopPurposeChoice('O')); -// models.put("MSStopType", new CommercialNextStopPurposeChoice('S')); -// models.put("MGStopType", new CommercialNextStopPurposeChoice('G')); -// models.put("MOStopType", new CommercialNextStopPurposeChoice('O')); -// models.put("HSStopType", new CommercialNextStopPurposeChoice('S')); -// models.put("HGStopType", new CommercialNextStopPurposeChoice('G')); -// models.put("HOStopType", new CommercialNextStopPurposeChoice('O')); -// models.put("LSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("LSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("LGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("LGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("LOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("MSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("MSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("MGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("MGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("MOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("HSSStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("HSOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("HGGStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("HGOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("HOOStopLocation", new WeekendStopChoice(anyOldMatrix, lowNumber, highNumber)); -// models.put("LODuration", new DurationModel()); -// models.put("LSDuration", new DurationModel()); -// models.put("LGDuration", new DurationModel()); -// models.put("MODuration", new DurationModel()); -// models.put("MSDuration", new DurationModel()); -// models.put("MGDuration", new DurationModel()); -// models.put("HODuration", new DurationModel()); -// models.put("HSDuration", new DurationModel()); -// models.put("HGDuration", new DurationModel()); - - WeekendTravelTimeTracker timeTracker = new WeekendTravelTimeTracker(); - models.put("TravelTimeMatrix",timeTracker); - WeekendTour.setElapsedTravelTimeCalculator(timeTracker); - - WeekendTravelTimeTracker disutilTracker = new WeekendTravelTimeTracker(); - models.put("TravelDisutilityMatrix", disutilTracker); - WeekendTour.setTravelDisutilityTracker(disutilTracker); - } - - - static final Hashtable models = new Hashtable(); - private static int minZone; - private static int maxZone; - -// public static ResultSet readCoefficients(String parameterFileLocation, String tableName) { -// ResultSet results = null; -// try { -// Class.forName("org.relique.jdbc.csv.CsvDriver"); -// Connection conn = DriverManager.getConnection("jdbc:relique:csv:" + parameterFileLocation); -// Statement stmt = conn.createStatement(); -// results = stmt.executeQuery("SELECT * FROM "+tableName); -// return results; -// } -// catch(Exception e) -// { -// System.out.println("JDBC Error connecting to "+parameterFileLocation+" "+ e); -// e.printStackTrace(); -// System.exit(-1); -// } -// return results; -// } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java deleted file mode 100644 index 95d72e7..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/NextWeekendTourStartTime.java +++ /dev/null @@ -1,101 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.calgary.weekend; - -import org.sandag.cvm.activityTravel.*; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public class NextWeekendTourStartTime - implements ModelWithCoefficients, RealNumberDistribution { - - double startTime = 0; - double endTime = 1; - private WeekendHousehold myHousehold; - - /** - * - */ - public NextWeekendTourStartTime() { - super(); - // TODO Auto-generated constructor stub - } - - /* (non-Javadoc) - * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#addCoefficient(java.lang.String, java.lang.String, java.lang.String, java.lang.String, double) - */ - public void addCoefficient( - String alternative, - String index1, - String index2, - String matrixName, - double coefficient) - throws CoefficientFormatError { - if (alternative.equalsIgnoreCase("EndTime")) { - endTime = coefficient; - } else if (alternative.equalsIgnoreCase("StartTime")) { - startTime = coefficient; - } else { - throw new CoefficientFormatError("Valid coefficients for tour start time model are \"StartTime\" and \"EndTime\""); - } - - } - - /* (non-Javadoc) - * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#init() - */ - public void init() { - // Nothing to do here. - - } - - public void shiftTime(double shift) { - startTime+=shift; - endTime +=shift; - } - - public void resetStart() { - endTime = endTime-startTime; - startTime = 0; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.common.model.SingleValueRealDistribution#sampleValue() - */ - public double sampleValue() { - // TODO use a different distribution? - double timeToNextTour = Math.random()*(endTime-startTime); - return timeToNextTour+startTime; - } - - public void setMyHousehold(WeekendHousehold myHousehold) { - this.myHousehold = myHousehold; - } - - public WeekendHousehold getMyHousehold() { - return myHousehold; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java deleted file mode 100644 index 09a2f9a..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/TourInTimeBand.java +++ /dev/null @@ -1,185 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.calgary.weekend; - -import java.util.ArrayList; - -import org.sandag.cvm.activityTravel.AlternativeUsesMatrices; -import org.sandag.cvm.activityTravel.CoefficientFormatError; -import org.sandag.cvm.activityTravel.ModelWithCoefficients; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.Alternative; -import org.sandag.cvm.common.model.LogitModel; -import org.sandag.cvm.common.model.NoAlternativeAvailable; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -/** - * @author jabraham - * - * Controls the time bands, increments through them, and has the model as - * to whether another tour occurs in the time band - */ -public class TourInTimeBand extends LogitModel implements ModelWithCoefficients { - - ArrayList bandStarts = new ArrayList(); - double dayEnd = 24*60; - final Alternative noTour; - final Alternative makeATour; - int currentBand = 0; - double makeATourConstant = 0; - WeekendHousehold currentHousehold=null; - - double getCurrentBandStart() { - if (currentBand >= bandStarts.size()) { - return dayEnd; - } else { - return ((Double)bandStarts.get(currentBand)).doubleValue(); - } - } - - double getCurrentBandEnd() { - if (currentBand +1 >= bandStarts.size()) { - return dayEnd; - } else { - return ((Double)bandStarts.get(currentBand+1)).doubleValue(); - } - } - - /* (non-Javadoc) - * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#addCoefficient(java.lang.String, java.lang.String, java.lang.String, java.lang.String, double) - */ - public void addCoefficient(String alternative, String index1, String index2, String matrixName, double coefficient) throws CoefficientFormatError { - Double lastBandStart = null; - if (bandStarts.size() == 0) lastBandStart = new Double(Double.NEGATIVE_INFINITY); - else lastBandStart =(Double)(bandStarts.get(bandStarts.size()-1)); - if (alternative.equalsIgnoreCase("bandStart")) { - if (coefficient <= lastBandStart.doubleValue()) { - throw new CoefficientFormatError("Start bands for tour time band model need to be specified in increasing order"); - } - bandStarts.add(new Double(coefficient)); - } else if (alternative.equalsIgnoreCase("dayEnd")) { - if (coefficient <= lastBandStart.doubleValue()) { - throw new CoefficientFormatError("End of day for tour time band model need to be greater than last time band"); - } - dayEnd = coefficient; - } else if (alternative.equalsIgnoreCase("constant")) { - makeATourConstant = coefficient; - } else { - throw new CoefficientFormatError("Bad coefficient for tour band model "+alternative+" "+index1+" "+index2+" "+matrixName); - } - - // TODO other coefficients - - } - - /* (non-Javadoc) - * @see org.sandag.cvm.calgary.weekend.ModelWithCoefficients#init() - */ - public void init() { - // TODO read matrices if necessary - } - - public TourInTimeBand() { - super(); - noTour = new AlternativeUsesMatrices() { - public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { - throw new CoefficientFormatError("The NoTour alternative has no coefficients and baseline utility zero"); - } - - public void readMatrices(MatrixCacheReader mr) { - // nothing to do - } - - public String getCode() { - return "noTour"; - } - - public double getUtility() { - return 0; - } - }; - makeATour = new AlternativeUsesMatrices() { - public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { - throw new CoefficientFormatError("The NoTour alternative has no coefficients and baseline utility zero"); - } - - public void readMatrices(MatrixCacheReader mr) { - // read matrices if we need to. - } - - public String getCode() { - return "makeATour"; - } - - public double getUtility() { - if (getCurrentBandEnd()<=getCurrentBandStart()){ - return Double.NEGATIVE_INFINITY; - } - return makeATourConstant + Math.log(getCurrentBandEnd()-getCurrentBandStart()) + Math.log(currentHousehold.countPeopleAtHome()); - // TODO add in other parameters related to number of people at home and household size etc. - } - }; - addAlternative(noTour); - addAlternative(makeATour); - } - - public boolean beyondLastBand() { - if (currentBand>=bandStarts.size()) return true; - return false; - } - - /** - * @return - */ - public boolean tourStartsInBand() { - Alternative chosen=null; - try { - chosen = this.monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - e.printStackTrace(); - throw new RuntimeException("Error in TourStartsInBand module",e); - } - return chosen == makeATour; - } - - /** - * @param currentTime - */ - public void setBandBasedOnTime(double currentTime) { - if (bandStarts.size() == 0) { - currentBand = 0; - return; - } - if (currentTime > dayEnd) { - currentBand = bandStarts.size(); - return; - } - if (currentTime < ((Double)bandStarts.get(0)).doubleValue()) { - throw new RuntimeException("Current household time isn't in any time band"); - } - for (currentBand = bandStarts.size()-1;currentBand >=0;currentBand--) { - if (currentTime >= ((Double)bandStarts.get(currentBand)).doubleValue()) { - return; - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java deleted file mode 100644 index d27d19f..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendHousehold.java +++ /dev/null @@ -1,366 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.calgary.weekend; -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; - -import org.sandag.cvm.activityTravel.HouseholdInterface; -import org.sandag.cvm.activityTravel.PersonInterface; -import org.sandag.cvm.common.datafile.TableDataSet; - - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public class WeekendHousehold implements HouseholdInterface { - - ArrayList myPeople =null; - private static int incrementalId = 0; - int id; - private boolean incomeValid; - private float annualIncome; - private int numberOfVehicles; - private int adultOtherCount; // ao - private int adultWorkerNeedingCarCount; // awnc - private int adultWorkerNotNeedingCarCount; // awnnc - private int studentKto9Count; //kejs - private int postSecondaryStudentCount; //pss - private int seniorCount; // Sen - private int student10to12Count; //SHS - private int youthOtherCount; //YO - private double currentTime; - private boolean homelessHousehold = true; - private int homeZone = 0; - - static NextWeekendTourStartTime myTourStartTimeModel = null; - - public WeekendHousehold(TableDataSet populationHouseholds, int rowNum, TableDataSet sampleHouseholds) { - homeZone = (int) populationHouseholds.getValueAt(rowNum,"Zone"); - homelessHousehold = false; - int hhid = (int) populationHouseholds.getValueAt(rowNum,"HHID"); - int sampleRowNum = sampleHouseholds.getIndexedRowNumber(hhid); - fillInDataFromDataSet(sampleHouseholds, sampleRowNum); - id = incrementalId++; - } - - - - public WeekendHousehold(TableDataSet tds, int rowNum) { - // For integration with household synthesis, need to get household attributes - // and people information from another file. - id=(int) tds.getValueAt(rowNum,"hh_ID"); - fillInDataFromDataSet(tds,rowNum); - } - - void fillInDataFromDataSet(TableDataSet tds, int rowNum) { - //TODO check to see if value targets in synthesis deal with missing values properly - annualIncome = tds.getValueAt(rowNum,"Value"); - if (annualIncome < -100000) { - annualIncome = 0; - incomeValid = false; - } else { - incomeValid = true; - } - numberOfVehicles = (int) tds.getValueAt(rowNum,"CountOfveh_id"); - adultOtherCount = (int) tds.getValueAt(rowNum,"AO"); - adultWorkerNeedingCarCount = (int) tds.getValueAt(rowNum,"AWNC"); - adultWorkerNotNeedingCarCount = (int) tds.getValueAt(rowNum,"AWNNC"); - studentKto9Count = (int) tds.getValueAt(rowNum,"KEJS"); - postSecondaryStudentCount = (int) tds.getValueAt(rowNum,"PSS"); - seniorCount = (int)tds.getValueAt(rowNum,"Sen"); - student10to12Count = (int) tds.getValueAt(rowNum,"SHS"); - youthOtherCount = (int)tds.getValueAt(rowNum,"YO"); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.HouseholdInterface#getId() - */ - public int getId() { - return id; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.HouseholdInterface#getPersons() - */ - public Collection getPersons() { - if (myPeople == null) { - makeMyPeople(); - } - return myPeople; - } - - private void makeMyPeople() { - // TODO add in other attributes of people as necessary - // TODO get RID of this method, as people should be read from the database. use AddPeople instead - myPeople = new ArrayList(); - int peopleToMake = getPersonCount(); - for (int p=0;p person.returnTime) person.atHome = true; - } - } - } - - - - double getCurrentTime() { - return currentTime; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java deleted file mode 100644 index 2b4ee77..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendPerson.java +++ /dev/null @@ -1,105 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.calgary.weekend; - -import org.sandag.cvm.activityTravel.HouseholdInterface; -import org.sandag.cvm.activityTravel.PersonInterface; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public class WeekendPerson implements PersonInterface { - - WeekendHousehold myHousehold; - /** - * @param household - */ - public WeekendPerson(WeekendHousehold household) { - myHousehold = household; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.PersonInterface#getMyHousehold() - */ - public HouseholdInterface getMyHousehold() { - return myHousehold; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.PersonInterface#getAge() - */ - public int getAge() { - // TODO add age attribute - throw new RuntimeException("getAge() is not yet implemented for WeekendPerson"); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.PersonInterface#getPersonID() - */ - public long getPersonID() { - // TODO Auto-generated method stub - throw new RuntimeException("getPersonID() is not yet implemented for WeekendPerson"); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.PersonInterface#isFemale() - */ - public boolean isFemale() { - // TODO Auto-generated method stub - throw new RuntimeException("isFemale() is not yet implemented for WeekendPerson"); - } - - public boolean atHome= true; - - public double returnTime=0; - - /** - * @return Returns the atHome. - */ - public boolean isAtHome() { - return atHome; - } - - /** - * @param atHome The atHome to set. - */ - public void setAtHome(boolean atHome) { - this.atHome = atHome; - } - - /** - * @return Returns the returnTime. - */ - public double getReturnTime() { - return returnTime; - } - - /** - * @param returnTime The returnTime to set. - */ - public void setReturnTime(double returnTime) { - this.returnTime = returnTime; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java deleted file mode 100644 index 1b6ba0d..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopChoice.java +++ /dev/null @@ -1,54 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.calgary.weekend; - -import org.sandag.cvm.activityTravel.StopAlternative; -import org.sandag.cvm.activityTravel.StopChoice; -import com.pb.common.matrix.Matrix; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class WeekendStopChoice extends StopChoice { - - public WeekendStopChoice(Matrix r, int minZone, int maxZone) { - super(); - int[] zoneNums = r.getExternalNumbers(); - for (int z = 1;z= minZone && theNumber <= maxZone) { - this.addAlternative(new StopAlternative(this, zoneNums[z])); - } - } - } - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateWeekendTours.matrixReader); - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java deleted file mode 100644 index c6c238c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendStopPurposeChoice.java +++ /dev/null @@ -1,217 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - -package org.sandag.cvm.calgary.weekend; - -import org.sandag.cvm.activityTravel.*; -import org.sandag.cvm.common.emme2.IndexLinearFunction; -import org.sandag.cvm.common.emme2.MatrixCacheReader; -import org.sandag.cvm.common.model.LogitModel; -import org.sandag.cvm.common.model.NoAlternativeAvailable; -import com.pb.common.matrix.Emme2MatrixReader; -import com.pb.common.matrix.MatrixReader; - -import java.util.*; - -import org.apache.log4j.Logger; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2005 - */ -public class WeekendStopPurposeChoice extends LogitModel implements ModelUsesMatrices, TourNextStopPurposeChoice { - - - private static Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); - Tour myTour; - static final int RETURN=WeekendTour.convertPurposeToInt("return"); - - public class NextStopPurpose implements AlternativeUsesMatrices { - public NextStopPurpose(int stopType) { - this.stopType = stopType; - } - - double[] transitionConstants = new double[14]; - double[] stopCountCoefficients = new double[14]; - double constant = 0; - - final int stopType; - IndexLinearFunction previousStopUtility = new IndexLinearFunction(); - IndexLinearFunction originUtility = new IndexLinearFunction(); - IndexLinearFunction returnToOriginUtility = new IndexLinearFunction(); - double timeToOriginCoefficient = 0; - double disutilityToOriginCoefficient = 0; - double totalTravelTimeCoefficient = 0; - double totalTripTimeCoefficient = 0; - public double getUtility() { - double utility = previousStopUtility.calcForIndex(myTour.getCurrentLocation(),1); - utility += originUtility.calcForIndex(getMyTour().getOriginZone(),1); - int previousStopType= myTour.getLastStopType(); - // TODO make sure the stop count code works properly for weekend model; default implementation just counts total stops, not by type - int[] stopCounts = getMyTour().getStopCounts(); - // can't return home on first stop - if (stopCounts[0]==0 && stopType==RETURN) utility += Double.NEGATIVE_INFINITY; - for (int type =0;type < stopCountCoefficients.length;type++) { - utility += stopCountCoefficients[type]*Math.log(stopCounts[type]+1); - } - double returnHomeUtility = returnToOriginUtility.calcForIndex(myTour.getCurrentLocation(),getMyTour().getOriginZone()); - - // make people return home more -- Doug and Kevin Hack of Jan 5th - //if (myTour.getTotalElapsedTime()>240.0) returnHomeUtility *=3; - utility += returnHomeUtility; - - utility += totalTravelTimeCoefficient*getMyTour().getTotalTravelTimeMinutes(); - utility += totalTripTimeCoefficient*getMyTour().getTotalElapsedTimeHrs(); - utility += timeToOriginCoefficient*myTour.getElapsedTravelTimeCalculator().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); - utility += disutilityToOriginCoefficient*getMyTour().getTravelDisutilityTracker().getTravelAttribute(myTour.getCurrentLocation(),getMyTour().getOrigin(),getMyTour().getCurrentTimeHrs(),getMyTour().getMyVehicleTourType().vehicleType); - utility += constant; - return utility; - } - - /** - * Method addParameter. - * @param matrix - * @param coefficient - */ - public void addCoefficient(String index1, String index2, String matrix, double coefficient) throws CoefficientFormatError { - if(index1.equals("origin")) { - originUtility.addCoefficient(matrix,coefficient); - } else if (index1.equals("cstop")) { - if (index2.equals("origin")) returnToOriginUtility.addCoefficient(matrix,coefficient); - else previousStopUtility.addCoefficient(matrix,coefficient); - } else if (index1.equals("prevStopType")) { - int type2 = WeekendTour.convertPurposeToInt(index2); - transitionConstants[type2] = coefficient; - } else if (index1.equals("logStopCount")) { - int type2 = WeekendTour.convertPurposeToInt(index2); - stopCountCoefficients[type2] = coefficient; - } else if (index1.equals("timeAccumulator")) { - totalTravelTimeCoefficient += coefficient; - } else if (index1.equals("totalAccumulator")) { - totalTripTimeCoefficient += coefficient; - } else if (index1.equals("travelDisutility") && index2.equals("origin")) { - disutilityToOriginCoefficient += coefficient; - } else if (index1.equals("travelTime") && index2.equals("origin")) { - timeToOriginCoefficient += coefficient; - } else if ((index1.equals("") || index1.equals("none")) && (index2.equals("") ||index2.equals("none"))) { - constant += coefficient; - } else { - throw new CoefficientFormatError("invalid indexing "+index1+ ","+index2+" in matrix "+matrix +" for next stop purpose model "); - } - } - - - - /** - * Method readMatrices. - * @param mr - */ - public void readMatrices(MatrixCacheReader mr) { - previousStopUtility.readMatrices(mr); - originUtility.readMatrices(mr); - returnToOriginUtility.readMatrices(mr); - } - - /** - * Method getStopPurposeCode. - * @return String - */ - public String getCode() { - return WeekendTour.convertPurposeToString(stopType); - } - -} - - /** - * Method addParameter. - * @param alternative - * @param matrix - * @param coefficient - */ - public void addCoefficient( - String alternative, - String index1, - String index2, - String matrix, - double coefficient) throws CoefficientFormatError { - Iterator alternativeIterator = alternatives.iterator(); - boolean found = false; - while (alternativeIterator.hasNext()) { - AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); - if (alternative.equals(alt.getCode())) { - alt.addCoefficient(index1,index2,matrix,coefficient); - found = true; - } - } - if (!found) { - logger.info("adding alternative "+alternative+" to "+name); - NextStopPurpose newPurpose = new NextStopPurpose(WeekendTour.convertPurposeToInt(alternative)); - addAlternative(newPurpose); - newPurpose.addCoefficient(index1,index2,matrix,coefficient); - } - } - - /** - * Method readMatrices. - * @param matrixReader - */ - public void readMatrices(MatrixCacheReader mr) { - Iterator alternativeIterator = alternatives.iterator(); - while (alternativeIterator.hasNext()) { - AlternativeUsesMatrices alt = (AlternativeUsesMatrices) alternativeIterator.next(); - alt.readMatrices(mr); - } - } - - public void setMyTour(Tour myTour) { - this.myTour = myTour; - } - - public Tour getMyTour() { - return myTour; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.ModelWithCoefficients#init() - */ - public void init() { - readMatrices(GenerateWeekendTours.matrixReader); - } - - - final String name; - - public WeekendStopPurposeChoice(String myName) { - super(); - this.name = myName; - } - - int monteCarloSamplePurpose() { - NextStopPurpose purpose; - try { - purpose = (NextStopPurpose) monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - e.printStackTrace(); - throw new RuntimeException("No valid purposes available",e); - } - return purpose.stopType; - } -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java b/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java deleted file mode 100644 index 58e439c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/calgary/weekend/WeekendTour.java +++ /dev/null @@ -1,345 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.calgary.weekend; - -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.activityTravel.ChangingTravelAttributeGetter; -import org.sandag.cvm.activityTravel.RealNumberDistribution; -import org.sandag.cvm.activityTravel.Stop; -import org.sandag.cvm.activityTravel.StopAlternative; -import org.sandag.cvm.activityTravel.StopChoice; -import org.sandag.cvm.activityTravel.Tour; -import org.sandag.cvm.activityTravel.VehicleTourTypeChoice; -import org.sandag.cvm.activityTravel.cvm.TourStartTimeModel; -import org.sandag.cvm.common.model.NoAlternativeAvailable; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public class WeekendTour extends Tour { - - static final Logger logger = Logger.getLogger("org.sandag.cvm.calgary.weekend"); - - - /** - * The primaryPerson is the person who makes a full tour, from origin and back again. - */ - public WeekendPerson primaryPerson; - ArrayList otherPeople = new ArrayList(); // the other people in the tour - - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.Tour#sampleStops() - */ - public void sampleStops() { - String tourType = myVehicleTourType.getTourTypeName(); - if (tourType.equals("SELSE") || tourType.equals("chauf")) { - sampleReturnStops(); - } - else { - samplePrimaryStop(); - sampleIntermediateOutboundStop(); - samplePrimaryAndIntermediateDurationAndAddToStopList(); - sampleReturnStops(); - } - } - - private void samplePrimaryAndIntermediateDurationAndAddToStopList() { - if (intermediateStop !=null) { - String stopTypeCode = convertPurposeToString(intermediateStop.purpose); - RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(stopTypeCode+"Duration"); - if (myDurationModel == null) throw new RuntimeException("Can't find duration model "+stopTypeCode+"Duration for stop type number "+intermediateStop.purpose); - intermediateStop.duration = (float) myDurationModel.sampleValue(); - addStop(intermediateStop); - } - String stopTypeCode = convertPurposeToString(primaryStop.purpose); - RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(stopTypeCode+"Duration"); - if (myDurationModel == null) throw new RuntimeException("Can't find duration model "+stopTypeCode+"Duration for stop type number "+primaryStop.purpose); - primaryStop.duration = (float) myDurationModel.sampleValue(); - addStop(primaryStop); - } - - public static int convertPurposeToInt(String stopPurpose) { - if (stopPurpose.equals("work")) return 1; - if (stopPurpose.equals("school")) return 2; - if (stopPurpose.equals("exercise")) return 3; - if (stopPurpose.equals("relCivic")) return 4; - if (stopPurpose.equals("social")) return 5; - if (stopPurpose.equals("entLeisure")) return 6; - if (stopPurpose.equals("shop")) return 7; - if (stopPurpose.equals("eat")) return 8; - if (stopPurpose.equals("dropOff")) return 9; - if (stopPurpose.equals("outOfTown")) return 10; - if (stopPurpose.equals("return")) return 11; - if (stopPurpose.equals("pickUp")) return 12; - if (stopPurpose.equals("dropOff")) return 13; - throw new RuntimeException("stop purpose "+stopPurpose+" is not a valid stop purpose type"); - } - - - /** - * @return String representing the stop type - */ - public static String convertPurposeToString(int purpose) { - switch (purpose) { - case 1: - return "work"; - case 2: - return "school"; - case 3: - return "exercise"; - case 4: - return "relCivic"; - case 5: - return "social"; - case 6: - return "entLeisure"; - case 7: - return "shop"; - case 8: - return "eat"; - case 9: - return "dropOff"; - case 10: - return "outOfTown"; - case 11: - return "return"; - case 12: - return "pickUp"; - case 13: - return "dropOff"; - } - throw new RuntimeException("invalid stop purpose code "+purpose); - } - - Stop intermediateStop = null; - - private void sampleIntermediateOutboundStop() { - - //TODO smarter intermediate stop choice existance model - if (Math.random() > 0.3) return; // 70% chance of no intermediate stop - - StopChoice theModel; - String stopModelStringCode = getTourTypeCode()+"IntermediateStop"; - theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); - if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); - theModel.setTour(this); - intermediateStop = new Stop(this, getCurrentLocation(),getTotalElapsedTimeHrs()); - try { - intermediateStop.location = ((StopAlternative) theModel.monteCarloChoice()).location; - } catch (NoAlternativeAvailable e) { - e.printStackTrace(); - throw new RuntimeException("Can't find a viable intermediate stop alternative",e); - } - - WeekendStopPurposeChoice purposeModel = (WeekendStopPurposeChoice) GenerateWeekendTours.models.get(getTourTypeCode()+"StopType"); - if (purposeModel == null)throw new RuntimeException("Can't find stop purpose model for "+getTourTypeCode()); - purposeModel.setMyTour(this); - intermediateStop.purpose = purposeModel.monteCarloSamplePurpose(); - - } - - Stop primaryStop = null; - - private void samplePrimaryStop() { - StopChoice theModel; - String stopModelStringCode = getTourTypeCode()+"PrimaryStop"; - theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); - if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); - theModel.setTour(this); - // FIXME will need to rewrite start location and start time if there are any intermediate stops outbound - primaryStop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); - try { - primaryStop.location= ((StopAlternative) theModel.monteCarloChoice()).location; - } catch (NoAlternativeAvailable e) { - e.printStackTrace(); - throw new RuntimeException("Can't find a viable primary stop alternative",e); - } - primaryStop.purpose = convertPurposeToInt(getTourTypeCode()); // assume tour type code with primary stops are subset of stop type codes - } - - final int returnStopTypeCode = convertPurposeToInt("return"); - - private void sampleReturnStops() { - StopChoice theModel; - String stopModelStringCode = getTourTypeCode()+"ReturnStop"; - theModel = (StopChoice) GenerateWeekendTours.models.get(stopModelStringCode); - if (theModel == null) throw new RuntimeException("Can't find stop choice model for "+stopModelStringCode); - final int maxReturnStops = 100; - int returnStops = 0; - Stop stop; - do { - theModel.setTour(this); - stop = new Stop(this, getCurrentLocation(),getCurrentTimeHrs()); - WeekendStopPurposeChoice purposeModel = (WeekendStopPurposeChoice) GenerateWeekendTours.models.get(getTourTypeCode()+"StopType"); - if (purposeModel == null)throw new RuntimeException("Can't find stop purpose model for "+getTourTypeCode()); - purposeModel.setMyTour(this); - stop.purpose = purposeModel.monteCarloSamplePurpose(); - if (stop.purpose == returnStopTypeCode) { - stop.location = getOriginZone(); - } - try { - stop.location= ((StopAlternative) theModel.monteCarloChoice()).location; - } catch (NoAlternativeAvailable e) { - e.printStackTrace(); - throw new RuntimeException("Can't find a viable return stop alternative",e); - } - if (stop.purpose != returnStopTypeCode) { - RealNumberDistribution myDurationModel = (RealNumberDistribution) GenerateWeekendTours.models.get(WeekendTour.convertPurposeToString(stop.purpose)+"Duration"); - if (myDurationModel == null) throw new RuntimeException("no "+WeekendTour.convertPurposeToString(stop.purpose)+"Duration model"); - stop.duration = (float) myDurationModel.sampleValue(); - } - addStop(stop); - returnStops ++; - } while (stop.purpose != returnStopTypeCode && returnStops <= maxReturnStops); - if (returnStops >= maxReturnStops) { - logger.warn("Return stops hit maximum, "+maxReturnStops); - } - } - - /** - * tourTypeChoiceModel is the choice model for the tNCVehicle tour type. It is currently - * a static variable, but it is accesssed by getters and setters so could be an instance variable instead, if - * different tours need different models for choosing the tour type - */ - static WeekendTourTypeChoice tourTypeChoiceModel; - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.Tour#getVehicleTourTypeChoice() - */ - public VehicleTourTypeChoice getVehicleTourTypeChoice() { - return tourTypeChoiceModel; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.activityTravel.Tour#setVehicleTourTypeChoice(org.sandag.cvm.activityTravel.VehicleTourTypeChoice) - */ - public void setVehicleTourTypeChoice(VehicleTourTypeChoice vehicleTourTypeChoice) { - WeekendTour.tourTypeChoiceModel = (WeekendTourTypeChoice) vehicleTourTypeChoice; - - } - - - static TourInTimeBand tourInTimeBand; - - /** - * @param titb - */ - public static void setTourInTimeBand(TourInTimeBand titb) { - tourInTimeBand = titb; - - } - - /** - * - */ - public void buildTourGroupAndFlagPeopleOutOfHome() { - primaryPerson.atHome = false; - List peopleInHousehold = (List) primaryPerson.getMyHousehold().getPersons(); - for (int p=0;p0.5) { - person.atHome = false; - otherPeople.add(person); - } - } - } - - } - - /** - * - */ - public void setReturnHomeTimesForGroupMembers() { - primaryPerson.returnTime = getCurrentTimeHrs(); - for (int p=0;p MAX_KEY_LENGTH) { - throw new IllegalArgumentException("Key is larger than permitted size of " - + MAX_KEY_LENGTH + " bytes"); - } - - file.seek(indexPositionToKeyFp(currentNumRecords)); - temp.writeTo(file); - file.seek(indexPositionToDataHeaderFp(currentNumRecords)); - newRecord.write(file); - newRecord.setIndexPosition(currentNumRecords); - writeNumRecordsHeader(currentNumRecords + 1); - } - - - /** - * Removes the record from the index. Replaces the target with the entry at the - * end of the index. - */ - protected void deleteEntryFromIndex(String key, DataHeader header, int currentNumRecords) - throws IOException { - - if (header.indexPosition != (currentNumRecords - 1)) { - String lastKey = readKeyFromIndex(currentNumRecords - 1); - DataHeader last = keyToDataHeader(lastKey); - - last.setIndexPosition(header.indexPosition); - file.seek(indexPositionToKeyFp(last.indexPosition)); - file.writeUTF(lastKey); - file.seek(indexPositionToDataHeaderFp(last.indexPosition)); - last.write(file); - } - - writeNumRecordsHeader(currentNumRecords - 1); - } - - - /** - * Adds the given record to the database. - */ - public synchronized void insertRecord(DataWriter rw) throws IOException { - String key = rw.getKey(); - - if (recordExists(key)) { - throw new IllegalArgumentException("Key exists: " + key); - } - - insureIndexSpace(getNumRecords() + 1); - - DataHeader newRecord = allocateRecord(key, rw.getDataLength()); - - writeRecordData(newRecord, rw); - addEntryToIndex(key, newRecord, getNumRecords()); - } - - - /** - * Updates an existing record. If the new contents do not fit in the original record, - * then the update is handled by deleting the old record and adding the new. - */ - public synchronized void updateRecord(DataWriter rw) throws IOException { - DataHeader header = keyToDataHeader(rw.getKey()); - - if (rw.getDataLength() > header.dataCapacity) { - deleteRecord(rw.getKey()); - insertRecord(rw); - } else { - writeRecordData(header, rw); - writeDataHeaderToIndex(header); - } - } - - - /** - * Reads a record. - */ - public synchronized DataReader readRecord(String key) throws IOException { - byte[] data = readRecordData(key); - - return new DataReader(key, data); - } - - - /** - * Reads the data for the record with the given key. - */ - protected byte[] readRecordData(String key) throws IOException { - return readRecordData(keyToDataHeader(key)); - } - - - /** - * Reads the record data for the given record header. - */ - protected byte[] readRecordData(DataHeader header) throws IOException { - byte[] buf = new byte[header.dataCount]; - - file.seek(header.dataPointer); - file.readFully(buf); - - return buf; - } - - - /** - * Updates the contents of the given record. An IOException is thrown if the - * new data does not fit in the space allocated to the record. The header's - * data count is updated, but not written to the file. - */ - protected void writeRecordData(DataHeader header, DataWriter rw) throws IOException { - if (rw.getDataLength() > header.dataCapacity) { - throw new IOException("Record data does not fit, header.dataCapacity="+ - header.dataCapacity+ - ", dataLength="+rw.getDataLength()); - } - - header.dataCount = rw.getDataLength(); - file.seek(header.dataPointer); - rw.writeTo((DataOutput) file); - } - - - /** - * Updates the contents of the given record. A DataFileException is thrown if - * the new data does not fit in the space allocated to the record. The header's - * data count is updated, but not written to the file. - */ - protected void writeRecordData(DataHeader header, byte[] data) throws IOException { - if (data.length > header.dataCapacity) { - throw new IOException("Record data does not fit, header.dataCapacity="+ - header.dataCapacity+ - ", dataLength="+data.length); - } - - header.dataCount = data.length; - file.seek(header.dataPointer); - file.write(data, 0, data.length); - } - - - /** - * Deletes a record. - */ - public synchronized void deleteRecord(String key) throws IOException { - DataHeader delRec = keyToDataHeader(key); - int currentNumRecords = getNumRecords(); - - if (getFileLength() == (delRec.dataPointer + delRec.dataCapacity)) { - // shrink file since this is the last record in the file - setFileLength(delRec.dataPointer); - } else { - DataHeader previous = getRecordAt(delRec.dataPointer - 1); - - if (previous != null) { - // append space of deleted record onto previous record - previous.dataCapacity += delRec.dataCapacity; - writeDataHeaderToIndex(previous); - } else { - // target record is first in the file and is deleted by adding its - // space to the second record. - DataHeader secondRecord = getRecordAt(delRec.dataPointer + - (long) delRec.dataCapacity); - byte[] data = readRecordData(secondRecord); - - secondRecord.dataPointer = delRec.dataPointer; - secondRecord.dataCapacity += delRec.dataCapacity; - writeRecordData(secondRecord, data); - writeDataHeaderToIndex(secondRecord); - } - } - - deleteEntryFromIndex(key, delRec, currentNumRecords); - } - - - // Checks to see if there is space for and additional index entry. If - // not, space is created by moving records to the end of the file. - protected void insureIndexSpace(int requiredNumRecords) throws IOException { - int originalFirstDataCapacity; - int currentNumRecords = getNumRecords(); - long endIndexPtr = indexPositionToKeyFp(requiredNumRecords); - - if (endIndexPtr > getFileLength() && currentNumRecords == 0) { - setFileLength(endIndexPtr); - dataStartPtr = endIndexPtr; - writeDataStartPtrHeader(dataStartPtr); - - return; - } - - // If first.dataCapacity is set to the actual data count BEFORE resetting - // dataStartPtr, and there is free space in 'first', then dataStartPtr will - // not be reset to the start of the second record. Capture the capacity - // first and use it to perform the reset. - while (endIndexPtr > dataStartPtr) { - DataHeader first = getRecordAt(dataStartPtr); - byte[] data = readRecordData(first); - first.dataPointer = getFileLength(); - originalFirstDataCapacity = first.dataCapacity; - first.dataCapacity = data.length; - setFileLength(first.dataPointer + data.length); - writeRecordData(first, data); - writeDataHeaderToIndex(first); - dataStartPtr += originalFirstDataCapacity; - writeDataStartPtrHeader(dataStartPtr); - } - } - - - /** - * Closes the file. - */ - public synchronized void close() throws IOException { - try { - file.close(); - } finally { - file = null; - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java deleted file mode 100644 index 761b4b0..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileReader.java +++ /dev/null @@ -1,114 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; -import java.io.ObjectInput; -import java.io.ObjectInputStream; -import java.util.ArrayList; -import org.apache.log4j.Logger; - -/** - * Reads a binary file containing a seriazlied TableDataSet class and creates - * an instance of a TableDataSet class. - * - * @author Tim Heier - * @version 1.0, 5/08/2004 - * - */ -public class BinaryFileReader extends TableDataFileReader implements DataTypes { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - //These attributes are initialized on each call to readFile() - private int columnCount; - private int rowCount; - private ArrayList columnData = new ArrayList(); - private ArrayList columnLabels = new ArrayList(); - private int[] columnType; - - - public BinaryFileReader () { - } - - - /** - * Reads a binary file containing serialized objects that make up a TableDataSet - * object. - * - * @param file name of file which contains the binary data - * @return a TableDataSet object - * - * @throws IOException when the file is not found - */ - public TableDataSet readFile(File file) throws IOException { - - TableDataSet table = null; - - try { - logger.debug("Opening file: "+file); - - //Open the file - ObjectInput inStream = new ObjectInputStream(new FileInputStream(file)); - - //Read magic number - int magicNumber = inStream.readInt(); - - //Read number of columns - int nCols = inStream.readInt(); - columnType = new int[nCols]; - - //Read titles - for (int c=0; c < nCols; c++) { - columnLabels.add( inStream.readUTF() ); - } - - //Read column data - for (int c=0; c < nCols; c++) { - Object colObj = inStream.readObject(); - columnData.add( colObj ); - } - inStream.close(); - - table = new TableDataSet(); - table.setName(file.toString()); - for (int i=0; i < nCols; i++) { - table.appendColumn(columnData.get(i), (String) columnLabels.get(i)); - } - } - catch (ClassNotFoundException e) { - logger.error("", e); - } - - return table; - } - - - public TableDataSet readTable(String tableName) throws IOException { - File fileName = new File (getMyDirectory() + File.separator + tableName + ".binTable"); - TableDataSet myTable = readFile(fileName); - myTable.setName(tableName); - return myTable; - } - - - public void close() { - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java deleted file mode 100644 index 0d1f0fb..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/BinaryFileWriter.java +++ /dev/null @@ -1,106 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.ObjectOutput; -import java.io.ObjectOutputStream; -import java.util.ArrayList; -import org.apache.log4j.Logger; - - -/** - * Writes a TableDataSet class to a binary file. - * - * @author Tim Heier - * @version 1.0, 5/08/2004 - * - */ -public class BinaryFileWriter extends TableDataFileWriter implements DataTypes { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - - public BinaryFileWriter () { - } - - - /** - * Writes a binary file containing serialized objects that make up a TableDataSet - * object. - * - * @param tableData the tabledataset to be written out to file - * @param file the file where the data should be written - * - * @throws IOException is thrown when the file cannot be written to - */ - public void writeFile(TableDataSet tableData, File file) throws IOException { - - //Pull data out of tableData object for convenience - int nCols = tableData.getColumnCount(); - String[] columnLabels = tableData.getColumnLabels(); - ArrayList columnData = tableData.getColumnData(); - - try { - logger.debug("Opening file: "+file); - - //Create file - ObjectOutput outStream = new ObjectOutputStream(new FileOutputStream(file)); - - //Write magic number - outStream.writeInt(1); - - //Write number of columns - outStream.writeInt( nCols ); - - //Write titles - for (int c=0; c < nCols; c++) { - outStream.writeUTF( columnLabels[c] ); - } - - //Write column data - for (int c=0; c < nCols; c++) { - outStream.writeObject( columnData.get(c) ); - } - - outStream.close(); - } - catch (IOException e) { - throw e; - } - - } - - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataWriter#writeTable(org.sandag.cvm.common.datafile.TableDataSet, java.lang.String) - */ - public void writeTable(TableDataSet tableData, String tableName) throws IOException { - File file = new File (getMyDirectory().getPath() + File.separator + tableName + ".binTable"); - writeFile(tableData, file); - } - - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataWriter#close() - */ - public void close() { - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java deleted file mode 100644 index 7f5637b..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/CSVFileReader.java +++ /dev/null @@ -1,810 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import org.apache.log4j.Logger; - -import java.io.*; -import java.net.URL; -import java.net.URLConnection; -import java.util.ArrayList; -import java.util.List; -import java.util.regex.Pattern; - - -/** - * Creates a TableData class from a CSV file. The default delimiter character is a comma. - * - * @author Tim Heier - * @version 1.0, 2/07/2004 - * - */ -public class CSVFileReader extends TableDataFileReader implements DataTypes { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - //Can be set by caller - private char delimiter = ','; - - //Pattern composed of regular expression used to parse CSV fields - private String pattern = ",(?=(?:[^\"]*\"[^\"]*\")*(?![^\"]*\"))"; - private Pattern regexPattern = Pattern.compile(pattern); - - //These attributes are initialized on each call to readFile() - private int columnCount; - private int rowCount; - private List columnData; - private ArrayList columnLabels; - private int[] columnType; - - private boolean padNulls=false; - - - public boolean isPadNulls() { - return padNulls; - } - - - public void setPadNulls(boolean padNulls) { - this.padNulls = padNulls; - } - - - public CSVFileReader () { - } - - - /** - * Sets the delimiters used by the StringTokenizer when reading column values. - * - * @param delimiter character separating fields in CSV file, default is a comma - */ - public void setDelimiter(char delimiter) { - this.delimiter = delimiter; - - //Update the pattern string with the new delimiter - pattern = Character.toString(delimiter) + pattern.substring(1); - regexPattern = Pattern.compile(pattern); - } - - - /** - * - * @return the delimiter in use - */ - public char getDelimiter() { - return delimiter; - } - - /** - * - * @return the pattern string used to parse CSV fields - */ - public String getPattern() { - return pattern; - } - - /** - * - * @param pattern new pattern string to be used when parsing CSV fields - */ - public void setPattern(String pattern) { - this.pattern = pattern; - regexPattern = Pattern.compile(pattern); - } - - - public TableDataSet readFile(File file) throws IOException { - return readFile(file, true); - } - - public TableDataSet readFile(String urlString) throws IOException { - return readFile(urlString, true); - } - - - /** - * Convenience method to load a CSV file into a table data class. - * - * @param file name of file to read - * @param columnLabelsPresent determines whether first line is treated - * as column titles - * @throws IOException - * - */ - public TableDataSet readFile(File file, boolean columnLabelsPresent) throws IOException { - return readFile(file, columnLabelsPresent, null); - } - - public TableDataSet readFile(String urlString, boolean columnLabelsPresent) throws IOException { - return readFile(urlString, columnLabelsPresent, null); - } - - - - - /** - * Convenience method to load a CSV file into a table data class. - * - * @param file name of file to read - * @param columnsToRead list of column labels that should be read - all other - * columns will be dropped from the table data set - * @throws IOException - * - */ - public TableDataSet readFile(File file, String[] columnsToRead) throws IOException { - return readFile(file, true, columnsToRead); - } - - - /** - * Main method which loads a CSV file into a table data class. - * - * @param file name of file to read - * @param columnLabelsPresent determines whether first line is treated - * as column titles - * @param columnsToRead list of column labels that should be read - all other - * columns will be dropped from the table data set - * @throws IOException - * - */ - public TableDataSet readFile(File file, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { - - if ((columnsToRead != null) && (columnLabelsPresent == false)) { - throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); - } - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - BufferedReader inStream = openFile(file); - - boolean[] readColumnFlag = null; - - if (columnLabelsPresent) { - readColumnFlag = readColumnLabels(inStream, columnsToRead); - boolean readAColumn=false; - for (boolean b: readColumnFlag) { - readAColumn = readAColumn || b; - } - if (!readAColumn) { - logger.fatal("No columns read when reading file "+file); - throw new RuntimeException("No columns read when reading file "+file); - } - } - readData(file, inStream, columnLabelsPresent, readColumnFlag); - - TableDataSet tds = makeTableDataSet(); - tds.setName(file.toString()); - return tds; - } - - /** - * Main method which loads a CSV file into a table data class. - * - * @param urlString http address of file to read - * @param columnLabelsPresent determines whether first line is treated - * as column titles - * @param columnsToRead list of column labels that should be read - all other - * columns will be dropped from the table data set - * @throws IOException - * - */ - public TableDataSet readFile(String urlString, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { - - if ((columnsToRead != null) && (columnLabelsPresent == false)) { - throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); - } - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - URL url; - URLConnection urlConn; - DataInputStream dis; - - url = new URL(urlString); - urlConn = url.openConnection(); - dis = new DataInputStream(urlConn.getInputStream()); - BufferedReader inStream = new BufferedReader(new InputStreamReader(dis)); - - - boolean[] readColumnFlag = null; - - if (columnLabelsPresent) { - readColumnFlag = readColumnLabels(inStream, columnsToRead); - boolean readAColumn=false; - for (boolean b: readColumnFlag) { - readAColumn = readAColumn || b; - } - if (!readAColumn) { - logger.fatal("No columns read when reading file "+ urlString); - throw new RuntimeException("No columns read when reading file "+ urlString); - } - } - readData(urlString, inStream, columnLabelsPresent, readColumnFlag); - - TableDataSet tds = makeTableDataSet(); - tds.setName(urlString.substring((urlString.lastIndexOf("/"))+1, urlString.length())); - System.out.println("Table Name is: " + tds.getName()); - return tds; - } - - - /* - * Read the csv file with a String[] of specified column formats (NUMBER or STRING), - * where the format is specified for all columns, all columns are read, - * and column headings must be present on the first line. - */ - public TableDataSet readFileWithFormats(File file, String[] columnFormats) throws IOException { - - boolean columnLabelsPresent = true; - String[] columnsToRead = null; - - if ((columnsToRead != null) && (columnLabelsPresent == false)) { - throw new RuntimeException("Column lables provided as filter but there are no column labels in CSV file"); - } - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - BufferedReader inStream = openFile(file); - - boolean[] readColumnFlag = null; - - if (columnLabelsPresent) { - readColumnFlag = readColumnLabels(inStream, columnsToRead); - } - readData(file, inStream, columnLabelsPresent, readColumnFlag, columnFormats); - - TableDataSet tds = makeTableDataSet(); - tds.setName(file.toString()); - return tds; - } - - - private BufferedReader openFile(File file) throws IOException { - logger.debug("Opening file: "+file); - - BufferedReader inStream = null; - try { - inStream = new BufferedReader( new FileReader(file) ); - } - catch (IOException e) { - throw e; - } - - return inStream; - } - - - /** - * Read and parse the column titles from the first line of file. - */ - private boolean[] readColumnLabels(BufferedReader inStream, String[] columnsToRead) - throws IOException { - //Read the first line - String line = inStream.readLine(); - - //Test for an empty file - if (line == null) { - throw new IOException("Error: file looks like it's empty"); - } - - //Tokenize the first line - String[] tokens = parseTokens(line); - int count = tokens.length; - - boolean[] readColumnFlag = new boolean[count]; - - //Initialize the readColumnFlag to false if the caller has supplied a - //list of columns. It will be turned to true basedon a comparison of the - //column labels found in the file. Otherwise initialize to true. - for (int i=0; i < count; i++) { - if (columnsToRead != null) - readColumnFlag[i] = false; - else - readColumnFlag[i] = true; - } - - //Read column titles - int c = 0; - for (int i=0; i < count; i++) { - String column_name = tokens[i]; - - //Check if column should be read based on list supplied by caller - if (columnsToRead != null) { - for (int j=0; j < columnsToRead.length; j++) { - if (columnsToRead[j].equalsIgnoreCase(column_name)) { - readColumnFlag[c] = true; - - columnLabels.add(column_name); - columnCount++; - break; - } - } - } - else { - columnLabels.add(column_name); - columnCount++; - } - c++; //the actual columnn number in the file being read - } - - //Debugging output - String msg = "column read flag = "; - for (int i=0; i < readColumnFlag.length; i++) { - if (readColumnFlag[i] == true) - msg += "true"; - else - msg += "false"; - if (i < (readColumnFlag.length-1)) - msg += ", "; - } - msg += "\n"; - logger.debug(msg); - - - return readColumnFlag; - } - - - /** - * Read and parse data portion of file. - */ - private void readData(File file, BufferedReader inStream, boolean columnLabelsPresent, - boolean[] readColumnFlag) - throws IOException { - - int rowNumber = 0; - - //Determine the number of lines in the file - rowCount = findNumberOfLinesInFile(file); - - readRows(file.toString(), inStream, columnLabelsPresent, readColumnFlag, rowNumber); - inStream.close(); - } - - - private void readRows(String source, BufferedReader inStream, - boolean columnLabelsPresent, boolean[] readColumnFlag, int rowNumber) - throws IOException { - logger.debug("number of lines in file: " + rowCount); - if (columnLabelsPresent) { - rowCount--; - } - - //Process each line in the file - String line; - if (rowCount == 0) { - columnType = new int[columnCount]; - readColumnFlag = new boolean[columnCount]; - for (int col =0; col #.00 and fieldWidth = 8 - * %6.0f --> #.# and fieldWidth = 6 - * %.3f --> #.000 and fieldWidth = 3 - * - * @param tableData the TableDataSet to write - * @param file the destination file to write to - * @param fieldFormat an array of PaddedDecimalFormat objects, one for each column - * - */ - public void writeFile(TableDataSet tableData, File file, PaddedDecimalFormat[] fieldFormat) throws IOException { - String formatString; - PrintWriter outStream = null; - - //Pull data out of tableData object for convenience - int nCols = tableData.getColumnCount(); - int nRows = tableData.getRowCount(); - int[] columnType = tableData.getColumnType(); - String[] columnLabels = tableData.getColumnLabels(); - ArrayList columnData = tableData.getColumnData(); - - if (fieldFormat.length != nCols) { - throw new RuntimeException("Length of format array is " + fieldFormat.length + - " should be " + nCols); - } - - try { - outStream = new PrintWriter (new BufferedWriter( new FileWriter(file) ) ); - - //Print titles - for (int i = 0; i < columnLabels.length; i++) { - if (i != 0) - outStream.print(","); - outStream.print( columnLabels[i] ); - } - outStream.println(); - - //Print data - for (int r=0; r < nRows; r++) { - //float[] rowValues = getRowValues(r, 0); - - for (int c=0; c < nCols; c++) { - if (c != 0) - outStream.print(","); - - switch(columnType[c]) { - case STRING: - String[] s = (String[]) columnData.get(c); - if (quoteStrings) { - outStream.print("\""+ s[r]+"\"" ); - } else { - outStream.print(s[r]); - } - break; - case NUMBER: - float[] f = (float[]) columnData.get(c); - if (Float.isInfinite(f[r])) { - // don't want the infinity figure (sideways 8) in output file, want Infinity or -Infinity. - outStream.print(f[r]); - } else { - outStream.print( fieldFormat[c].format(f[r]) ); - } - break; - default: - throw new RuntimeException("unknown column data type: " + columnType[c]); - } - } - outStream.println(); - } - outStream.close(); - } - catch (IOException e) { - throw e; - } - - //Update dirty flag - tableData.setDirty(false); - } - - - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataWriter#writeTable(org.sandag.cvm.common.datafile.TableDataSet, java.lang.String) - */ - public void writeTable(TableDataSet tableData, String tableName) throws IOException { - File file = new File (getMyDirectory().getPath() + File.separator + tableName + ".csv"); - writeFile(tableData, file); - } - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataWriter#close() - */ - public void close() { - } - - public boolean isQuoteStrings() { - return quoteStrings; - } - - public void setQuoteStrings(boolean quoteStrings) { - this.quoteStrings = quoteStrings; - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java deleted file mode 100644 index 55b45ca..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D211FileReader.java +++ /dev/null @@ -1,304 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileReader; -import java.io.BufferedReader; -import java.io.IOException; -import java.io.Serializable; -import java.util.StringTokenizer; -import java.util.ArrayList; -import org.apache.log4j.Logger; - - -/** - * Reads a standard Emme/2 d211 text format file containing node and link records - * for a transportation network. - * - * @author Jim Hicks - * @version 1.0, 5/12/2004 - * - */ -public class D211FileReader implements Serializable { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - - - public D211FileReader () { - } - - - - public TableDataSet readNodeTable (File file) throws IOException { - - int record = 0; - boolean found_t_nodes_init = false; - boolean found_t_links_init = false; - - ArrayList nList = new ArrayList(); - ArrayList xList = new ArrayList(); - ArrayList yList = new ArrayList(); - - float[][] dataTable = null; - - TableDataSet table = null; - - - - try { - logger.debug( "Opening d211 file to read node records: " + file.getName() ); - - //Open the file - BufferedReader in = new BufferedReader(new FileReader(file)); - - String s = new String(); - while ((s = in.readLine()) != null) { - - record++; - - if ( s.indexOf("t nodes") >= 0 ) { - - found_t_nodes_init = true; - - } - else if ( s.indexOf("t links") >= 0 ) { - - found_t_links_init = true; - - } - else if (found_t_nodes_init && !found_t_links_init) { - - parseNode(s, nList, xList, yList); - - } - - } - - } - catch (Exception e) { - System.out.println ("IO Exception caught reading node table data from d211 format file: " + file.getName() + ", record number=" + record ); - e.printStackTrace(); - } - - - - dataTable = new float[nList.size()][3]; - for (int i=0; i < nList.size(); i++) { - try { - dataTable[i][0] = Integer.parseInt ( (String)nList.get(i) ); - dataTable[i][1] = Float.parseFloat ( (String)xList.get(i) ); - dataTable[i][2] = Float.parseFloat ( (String)yList.get(i) ); - } catch (Exception e) { - String msg = "Can't parse "+i+"th node"; - logger.fatal(msg,e); - throw new RuntimeException(msg,e); - } - } - - - ArrayList tableHeadings = new ArrayList(); - tableHeadings.add ("node"); - tableHeadings.add ("x"); - tableHeadings.add ("y"); - - - table = TableDataSet.create( dataTable, tableHeadings ); - - return table; - - } - - - - public TableDataSet readLinkTable (File file) throws IOException { - return readLinkTable ( file, 'a' ); - } - - - public TableDataSet readLinkTableMods (File file) throws IOException { - return readLinkTable ( file, 'm' ); - } - - - private TableDataSet readLinkTable (File file, char action) throws IOException { - - int record = 0; - boolean found_t_links_init = false; - - ArrayList anList = new ArrayList(); - ArrayList bnList = new ArrayList(); - ArrayList distList = new ArrayList(); - ArrayList modeList = new ArrayList(); - ArrayList typeList = new ArrayList(); - ArrayList lanesList = new ArrayList(); - ArrayList vdfList = new ArrayList(); - ArrayList ul1List = new ArrayList(); - ArrayList ul2List = new ArrayList(); - ArrayList ul3List = new ArrayList(); - ArrayList ul4List = new ArrayList(); - - float[][] dataTable = null; - String[] stringColumn = null; - - TableDataSet table = null; - - - - try { - logger.debug( "Opening d211 file to read link records: " + file.getName() ); - - //Open the file - BufferedReader in = new BufferedReader(new FileReader(file)); - - String s = new String(); - while ((s = in.readLine()) != null) { - - record++; - - if ( s.indexOf("t links") >= 0 ) { - - found_t_links_init = true; - - } - else if (found_t_links_init) { - - parseLink( s, action, anList, bnList, distList, modeList, typeList, lanesList, vdfList, ul1List, ul2List, ul3List, ul4List ); - - } - - } - - } - catch (Exception e) { - System.out.println ("IO Exception caught reading link table data from d211 format file: " + file.getName() + ", record number=" + record ); - e.printStackTrace(); - } - - - - dataTable = new float[anList.size()][10]; - stringColumn = new String[anList.size()]; - for (int i=0; i < anList.size(); i++) { - dataTable[i][0] = Integer.parseInt ( (String)anList.get(i) ); - dataTable[i][1] = Integer.parseInt ( (String)bnList.get(i) ); - dataTable[i][2] = Float.parseFloat ( (String)distList.get(i) ); - stringColumn[i] = (String)modeList.get(i); - dataTable[i][3] = Integer.parseInt ( (String)typeList.get(i) ); - dataTable[i][4] = Float.parseFloat ( (String)lanesList.get(i) ); - dataTable[i][5] = Integer.parseInt ( (String)vdfList.get(i) ); - - try { - dataTable[i][6] = Float.parseFloat ( (String)ul1List.get(i) ); - } - catch (Exception e) { - dataTable[i][6] = 0.0f; - } - try { - dataTable[i][7] = Float.parseFloat ( (String)ul2List.get(i) ); - } - catch (Exception e) { - dataTable[i][7] = 0.0f; - } - try { - dataTable[i][8] = Float.parseFloat ( (String)ul3List.get(i) ); - } - catch (Exception e) { - dataTable[i][8] = 0.0f; - } - try { - dataTable[i][9] = Float.parseFloat ( (String)ul4List.get(i) ); - } - catch (Exception e) { - dataTable[i][9] = 0.0f; - } - } - - - ArrayList tableHeadings = new ArrayList(); - tableHeadings.add ("anode"); - tableHeadings.add ("bnode"); - tableHeadings.add ("dist"); - tableHeadings.add ("type"); - tableHeadings.add ("lanes"); - tableHeadings.add ("vdf"); - tableHeadings.add ("ul1"); - tableHeadings.add ("ul2"); - tableHeadings.add ("ul3"); - tableHeadings.add ("ul4"); - - table = TableDataSet.create( dataTable, tableHeadings ); - table.appendColumn (stringColumn, "mode"); - - - - return table; - - } - - - - void parseNode ( String InputString, ArrayList n, ArrayList x, ArrayList y ) { - - StringTokenizer st = new StringTokenizer(InputString); - - if (st.hasMoreTokens()) { - - if ((st.nextToken()).charAt(0) == 'a') { // read only add records - - n.add ( st.nextToken() ); - x.add ( st.nextToken() ); - y.add ( st.nextToken() ); - - } - - } - - } - - - - - void parseLink ( String InputString, char action, ArrayList anList, ArrayList bnList, ArrayList distList, ArrayList modeList, ArrayList typeList, ArrayList lanesList, ArrayList vdfList, ArrayList ul1List, ArrayList ul2List, ArrayList ul3List, ArrayList ul4List ) { - - StringTokenizer st = new StringTokenizer(InputString); - int count = st.countTokens(); - - while (st.hasMoreTokens()) { - - if ( (st.nextToken()).charAt(0) == action ) { // process add or mod records as requested - - anList.add ( st.nextToken() ); - bnList.add ( st.nextToken() ); - distList.add ( st.nextToken() ); - modeList.add ( st.nextToken() ); - typeList.add ( st.nextToken() ); - lanesList.add ( st.nextToken() ); - vdfList.add ( st.nextToken() ); - ul1List.add ( st.nextToken() ); - if (st.hasMoreTokens()) ul2List.add ( st.nextToken() ); - if (st.hasMoreTokens()) ul3List.add ( st.nextToken() ); - if (st.hasMoreTokens()) ul4List.add ( st.nextToken() ); - - } - - } - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java deleted file mode 100644 index f991620..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/D231FileReader.java +++ /dev/null @@ -1,129 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileReader; -import java.io.BufferedReader; -import java.io.IOException; -import java.io.Serializable; -import java.util.StringTokenizer; -import java.util.ArrayList; -import org.apache.log4j.Logger; - - -/** - * Reads a standard Emme/2 d231 text format file containing turn definitions - * for a transportation network. - * - * @author Jim Hicks - * @version 1.0, 5/12/2004 - * - */ -public class D231FileReader implements Serializable { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - - - public D231FileReader () { - } - - - - public float[][] readTurnTable (File file) throws IOException { - - int record = 0; - boolean found_t_turns_init = false; - - ArrayList values = new ArrayList(); - - float[][] dataTable = null; - - - try { - logger.debug( "Opening d231 file to read turn records: " + file.getName() ); - - //Open the file - BufferedReader in = new BufferedReader(new FileReader(file)); - - String s = new String(); - while ((s = in.readLine()) != null) { - - record++; - - if ( s.indexOf("t turns") >= 0 ) { - - found_t_turns_init = true; - - } - else if (found_t_turns_init) { - - values.add ( parseRecord ( s ) ); - - } - - } - - } - catch (Exception e) { - System.out.println ("IO Exception caught reading node table data from d211 format file: " + file.getName() + ", record number=" + record ); - e.printStackTrace(); - } - - - - dataTable = new float[values.size()][((float[])values.get(0)).length]; - for (int i=0; i < values.size(); i++) { - dataTable[i] = (float[])values.get(i); - } - - - return dataTable; - - } - - - - private float[] parseRecord ( String InputString ) { - - float[] values = new float[8]; - - StringTokenizer st = new StringTokenizer(InputString); - - - int i = 0; - - if (st.hasMoreTokens()) { - - if ((st.nextToken()).charAt(0) == 'a') { // read only add records - - while (st.hasMoreTokens()) { - - values[i] = Float.parseFloat ( st.nextToken() ); - i++; - - } - } - - } - - return values; - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java deleted file mode 100644 index b27c645..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DBFFileReader.java +++ /dev/null @@ -1,225 +0,0 @@ -/* - * Copyright 2006 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.util.ArrayList; -import java.util.List; - -import org.apache.log4j.Logger; - -import com.linuxense.javadbf.DBFException; -import com.linuxense.javadbf.DBFField; -import com.linuxense.javadbf.DBFReader; -import org.sandag.cvm.common.datafile.TableDataFileReader; -import org.sandag.cvm.common.datafile.TableDataSet; - -import static org.sandag.cvm.common.datafile.DataTypes.*; - -/** - * This class reads DBF files and stores them as TableDataSets. It uses the third party - * classes from com.linuxense.javadbf - * - * @author Erhardt - * @version 1.0 Nov 27, 2006 - * - */ -public class DBFFileReader extends TableDataFileReader { - - protected static transient Logger logger = Logger.getLogger(DBFFileReader.class); - - //These attributes are initialized on each call to readFile() - private int columnCount; - private int rowCount; - private List columnData; - private ArrayList columnLabels; - private int[] columnType; - private DBFField[] columnDBF; - - /** - * Default constructor. - * - */ - public DBFFileReader() { - - } - - /** - * @see org.sandag.cvm.common.datafile.TableDataFileReader#readFile(java.io.File) - */ - @Override - public TableDataSet readFile(File file) throws IOException { - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - // open file - logger.debug("Opening file: "+file); - InputStream inStream = new FileInputStream(file); - DBFReader reader = new DBFReader(inStream); - - // read data - readHeader(reader); - readData(reader); - inStream.close(); - - // make table - TableDataSet tds = makeTableDataSet(); - tds.setName(file.toString()); - return tds; - } - - /** - * Reads the header information and sets up the table structure. - * - * @param reader - * @throws DBFException - */ - private void readHeader(DBFReader reader) throws DBFException { - - // Determine the number of columns in the file - columnCount = reader.getFieldCount(); - logger.debug("number of columns in file: " + columnCount); - - //Determine the number of lines in the file - rowCount = reader.getRecordCount(); - logger.debug("number of lines in file: " + rowCount); - - //Get the field descriptions - columnType = new int[columnCount]; - columnDBF = new DBFField[columnCount]; - for (int col=0; col 0) { - if (dataLength > maxElementSize) { - throw new IllegalArgumentException( - "Size of entry="+dataLength+" bytes is larger than maxElementSize=" + maxElementSize); - } - setFileLength(fp + maxElementSize); //all elements are maxElementSize in length - newRecord = new DataHeader(fp, maxElementSize); - } - else { - setFileLength(fp + dataLength); - newRecord = new DataHeader(fp, dataLength); - } - } - - return newRecord; - } - - - /** - * Returns the record to which the target file pointer belongs - meaning the specified location - * in the file is part of the record data of the DataHeader which is returned. Returns null if - * the location is not part of a record. (O(n) mem accesses) - */ - protected DataHeader getRecordAt(long targetFp) { - Enumeration e = memIndex.elements(); - - while (e.hasMoreElements()) { - DataHeader next = (DataHeader) e.nextElement(); - - if ((targetFp >= next.dataPointer) && (targetFp < (next.dataPointer + (long) next.dataCapacity))) { - return next; - } - } - - return null; - } - - - /** - * Closes the database. - */ - public synchronized void close() throws IOException { - try { - super.close(); - } finally { - memIndex.clear(); - memIndex = null; - } - } - - - /** - * Adds the new record to the in-memory index and calls the super class add - * the index entry to the file. - */ - protected void addEntryToIndex(String key, DataHeader newRecord, int currentNumRecords) throws IOException { - super.addEntryToIndex(key, newRecord, currentNumRecords); - memIndex.put(key, newRecord); - } - - - /** - * Removes the record from the index. Replaces the target with the entry at the - * end of the index. - */ - protected void deleteEntryFromIndex(String key, DataHeader header, int currentNumRecords) throws IOException { - super.deleteEntryFromIndex(key, header, currentNumRecords); - - DataHeader deleted = (DataHeader) memIndex.remove(key); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java deleted file mode 100644 index 091015c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataHeader.java +++ /dev/null @@ -1,114 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.DataInput; -import java.io.DataOutput; -import java.io.IOException; - -public class DataHeader { - - /** - * File pointer to the first byte of record data (8 bytes). - */ - protected long dataPointer; - - /** - * Actual number of bytes of data held in this record (4 bytes). - */ - protected int dataCount; - - /** - * Number of bytes of data that this record can hold (4 bytes). - */ - protected int dataCapacity; - - /** - * Indicates this header's position in the file index. - */ - protected int indexPosition; - - protected DataHeader() { - } - - - protected DataHeader(long dataPointer, int dataCapacity) { - if (dataCapacity < 1) { - throw new IllegalArgumentException("Bad record size: " + dataCapacity); - } - - this.dataPointer = dataPointer; - this.dataCapacity = dataCapacity; - this.dataCount = 0; - } - - protected int getIndexPosition() { - return indexPosition; - } - - - protected void setIndexPosition(int indexPosition) { - this.indexPosition = indexPosition; - } - - - protected int getDataCapacity() { - return dataCapacity; - } - - - protected int getFreeSpace() { - return dataCapacity - dataCount; - } - - - protected void read(DataInput in) throws IOException { - dataPointer = in.readLong(); - dataCapacity = in.readInt(); - dataCount = in.readInt(); - } - - - protected void write(DataOutput out) throws IOException { - out.writeLong(dataPointer); - out.writeInt(dataCapacity); - out.writeInt(dataCount); - } - - - protected static DataHeader readHeader(DataInput in) throws IOException { - DataHeader r = new DataHeader(); - - r.read(in); - - return r; - } - - - /** - * Returns a new record header which occupies the free space of this record. - * Shrinks this record size by the size of its free space. - */ - protected DataHeader split() { - long newFp = dataPointer + (long) dataCount; - DataHeader newData = new DataHeader(newFp, getFreeSpace()); - - dataCapacity = dataCount; - - return newData; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java deleted file mode 100644 index 58bb994..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataReader.java +++ /dev/null @@ -1,94 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.*; -import org.apache.log4j.Logger; - -public class DataReader { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - String key; - byte[] data; - ByteArrayInputStream in; - ObjectInputStream objIn; - - public DataReader(String key, byte[] data) { - this.key = key; - this.data = data; - in = new ByteArrayInputStream(data); - } - - public String getKey() { - return key; - } - - - public byte[] getData() { - return data; - } - - - public InputStream getInputStream() throws IOException { - return in; - } - - - public ObjectInputStream getObjectInputStream() throws IOException { - if (objIn == null) { - objIn = new ObjectInputStream(in); - } - - return objIn; - } - - - /** - * Reads the next object in the record using an ObjectInputStream. - */ - public Object readObject() throws IOException, OptionalDataException, ClassNotFoundException { - return getObjectInputStream().readObject(); - } - - - - /** - * Reads the serialized object from filename on disk using the specified key. - */ - public static Object readDiskObject ( String filename, String key ) { - - Object obj=null; - DataFile dataFile=null; - - try { - dataFile = new DataFile( filename, "r" ); - DataReader dr = dataFile.readRecord( key ); - obj = dr.readObject(); - } - catch (Exception e) { - logger.error("Exception thrown when reading DiskObject file: " + filename ); - e.printStackTrace(); - } - - return obj; - } - - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java deleted file mode 100644 index 2de42ff..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataTypes.java +++ /dev/null @@ -1,38 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.Serializable; - -/** - * Supported data types in table oriented data set classes. - * - * @author Tim Heier - * @version 1.0, 1/30/2003 - * - */ - -public interface DataTypes extends Serializable { - - //Supported data types - public final static int NULL = 0; - public final static int BOOLEAN = 1; - public final static int STRING = 2; - public final static int NUMBER = 3; - public final static int DOUBLE = 4; - public final static int OTHER = 1111; -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java deleted file mode 100644 index 6a4420a..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DataWriter.java +++ /dev/null @@ -1,105 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.DataOutput; -import java.io.IOException; -import java.io.ObjectOutputStream; -import org.apache.log4j.Logger; - -public class DataWriter { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - private String key; - private DbByteArrayOutputStream out; - private ObjectOutputStream objOut; - - - public DataWriter(String key) { - this.key = key; - out = new DbByteArrayOutputStream(); - try { - objOut = new ObjectOutputStream(out); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - - public String getKey() { - return key; - } - - - public void writeObject(Object o) throws IOException { - - //Reset the size of the underlying ByteArrayOutputStream so it can be reused - //out.reset(); - objOut.reset(); - objOut.writeObject(o); - objOut.flush(); - } - - - /** - * Returns the number of bytes in the data. - */ - public int getDataLength() { - return out.size(); - } - - - /** - * Writes the data out to the stream without re-allocating the buffer. - */ - public void writeTo(DataOutput str) throws IOException { - out.writeTo(str); - } - - - /** - * Writes the serialized contents of the object to filename on disk using the specified key. - */ - public static void writeDiskObject ( Object obj, String filename, String key ) { - - DataFile dataFile=null; - - try { - dataFile = new DataFile( filename, 1 ); - DataWriter dw = new DataWriter( key ); - dw.writeObject( obj ); - dataFile.insertRecord(dw); - } - catch (IOException e) { - logger.error( "IO Exception thrown when writing DiskObject file: " + filename ); - e.printStackTrace(); - } - - try { - dataFile.close(); - } - catch (IOException e) { - logger.error( "IO Exception thrown when closing DiskObject file: " + filename ); - e.printStackTrace(); - } - } - - -} - - diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java deleted file mode 100644 index 4e4133c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DbByteArrayOutputStream.java +++ /dev/null @@ -1,47 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.ByteArrayOutputStream; -import java.io.DataOutput; -import java.io.IOException; - -/** - * Extends ByteArrayOutputStream to provide a way of writing the buffer to - * a DataOutput without re-allocating it. - */ -public class DbByteArrayOutputStream extends ByteArrayOutputStream { - - public DbByteArrayOutputStream() { - super(); - } - - - public DbByteArrayOutputStream(int size) { - super(size); - } - - /** - * Writes the full contents of the buffer a DataOutput stream. - */ - public synchronized void writeTo(DataOutput dstr) throws IOException { - byte[] data = super.buf; - int l = super.size(); - - dstr.write(data, 0, l); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java deleted file mode 100644 index 5bbb092..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/DiskObjectArray.java +++ /dev/null @@ -1,227 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.FileNotFoundException; -import java.io.IOException; -import java.io.Serializable; - -/** - * - * @author Tim Heier - * @version 1.0, 8/13/2003 - * - */ - -public class DiskObjectArray implements Serializable { - - private String fileName; - private int arraySize; - private int maxElementSize; - - private String[] indexArray; - - private transient DataFile dataFile; - - private DiskObjectArray() { - } - - - /** - * Constructor called when creating/overwriting an existing file. - * - * @param fileName fully quallified file name of file used to - * @param arraySize number of elements in the array - * @param maxElementSize maximum size of an element that will be stored (in bytes) - * @throws IOException - */ - public DiskObjectArray(String fileName, int arraySize, int maxElementSize) throws IOException { - this.fileName = fileName; - this.arraySize = arraySize; - this.maxElementSize = maxElementSize; - - dataFile = new DataFile(fileName, arraySize, maxElementSize); - - //Number of elements in array - used when the data file is opened later - DataWriter dw = new DataWriter("arraySize"); - dw.writeObject( new Integer(arraySize) ); - dataFile.insertRecord(dw); - - //maximum size of an element that will be stored - used when the data file is opened later - dw = new DataWriter("maxElementSize"); - dw.writeObject( new Integer(maxElementSize) ); - dataFile.insertRecord(dw); - - indexArray = new String[arraySize+1]; - for (int i=0; i <=arraySize; i++) { - indexArray[i] = i + ""; - } - - } - - - /** - * Constructor called when opening an existing file. - * - * @param fileName fully qualified file name of data-file - * @throws IOException - */ - public DiskObjectArray(String fileName) throws IOException, FileNotFoundException { - this.fileName = fileName; - - File f = new File(fileName); - - if (! f.exists()) { - throw new FileNotFoundException("Database file could not be found: " + fileName); - } - - //Read size of array from data file and initialize indexArray - try { - dataFile = new DataFile(fileName, "rw"); - DataReader dr = dataFile.readRecord("arraySize"); - Integer i = (Integer) dr.readObject(); - this.arraySize = i.intValue(); - - dr = dataFile.readRecord("maxElementSize"); - i = (Integer) dr.readObject(); - this.maxElementSize = i.intValue(); - - //This is a small hack to set the max element size without calling a constructor - //dataFile.maxElementSize = this.maxElementSize; - - } - catch (IOException e) { - throw e; - } - catch (ClassNotFoundException e) { - e.printStackTrace(); - } - - indexArray = new String[arraySize+1]; - for (int i=0; i <=arraySize; i++) { - indexArray[i] = i + ""; - } - - } - - - public void add(int index, Object element) { - //Skip size check for maximum performance - //if (index > size || index < 0) { - // throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size); - //} - - DataWriter dw = new DataWriter( indexArray[index] ); - try { - dw.writeObject( element ); - } - catch (IOException e) { - e.printStackTrace(); - } - - //Check size of element against the maximum allowed record size - if (dw.getDataLength() > maxElementSize) { - throw new IllegalArgumentException( - "Size of entry="+dw.getDataLength()+" bytes is larger than maxElementSize=" + maxElementSize); - } - - try { - //Update record if it exists already - if (dataFile.recordExists(indexArray[index])) { - dataFile.updateRecord( dw ); - } - else { - dataFile.insertRecord( dw ); - } - } - catch (Exception e) { - e.printStackTrace(); - } - } - - - /** - * Returns an element from the array. - * - * @param index array index - */ - public Object get(int index) { - //Skip size check for maximum performance - //if (index > size || index < 0) { - // throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size); - //} - - DataReader dr = null; - Object obj = null; - try { - dr = dataFile.readRecord(indexArray[index]); - obj = dr.readObject(); - } - catch (Exception e) { - e.printStackTrace(); - } - - return obj; - } - - - /** - * Removes an element from the array. Can be an expensive operation so only - * delete when necessary. - * - * @param index array index - */ - public void remove(int index) { - try { - dataFile.deleteRecord( indexArray[index] ); - } - catch (IOException e) { - e.printStackTrace(); - } - - } - - - public String getFileName() { - return fileName; - } - - - public int getArraySize() { - return arraySize; - } - - - public int getMaxElementSize() { - return maxElementSize; - } - - /** - * Closes the underlying file. - * - */ - public void close() { - try { - dataFile.close(); - } - catch (IOException e) { - e.printStackTrace(); - } - } -} - diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java deleted file mode 100644 index f79a397..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/ExcelFileReader.java +++ /dev/null @@ -1,692 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.List; - -import jxl.CellType; -import jxl.NumberCell; -import jxl.NumberFormulaCell; -import jxl.Workbook; -import jxl.Cell; -import jxl.Sheet; - -import org.apache.log4j.Logger; - - -/** - * Creates a TableData class from an Excel file. - * - * @author Joel Freedman - * @author John Abraham - * @version 1.1, 10/10/2007 - * - * New version checks for cell types and throws an error - * if it can't get a number value from a cell when it thinks - * it should (J. Abraham, Sept-Oct 2007) - */ -public class ExcelFileReader extends TableDataFileReader implements DataTypes { - - /** - * - */ - private static final long serialVersionUID = 1L; - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - //These attributes are initialized on each call to readFile() - private int columnCount; - private int rowCount; - private List columnData; - private ArrayList columnLabels; - private int[] columnType; - private String worksheetName; - private File workbookFile=null; - - - public ExcelFileReader () { - } - - /** - * Set the name of the worksheet to read in this excel workbook. - * @param name THe name of the worksheet to read. - */ - public void setWorksheetName(String name){ - worksheetName = name; - } - - public TableDataSet readFile(File file) throws IOException { - - if(worksheetName == null){ - logger.fatal("Error: must set worksheet to read using setWorksheetName method before reading"); - throw new RuntimeException(); - } - - return readFile(file, worksheetName, true); - } - - - /** - * Convenience method to load a Excel file into a table data class. - * - * @param file name of file to read - * @param columnLabelsPresent determines whether first line is treated - * as column titles - * @throws IOException - * - */ - public TableDataSet readFile(File file, String worksheetName, boolean columnLabelsPresent) throws IOException { - return readFile(file, worksheetName, columnLabelsPresent, null); - } - - - /** - * Convenience method to load a Excel file into a table data class. - * - * @param file name of file to read - * @param columnsToRead list of column labels that should be read - all other - * columns will be dropped from the table data set - * @throws IOException - * - */ - public TableDataSet readFile(File file, String worksheetName, String[] columnsToRead) throws IOException { - return readFile(file, worksheetName,true, columnsToRead); - } - - - /** - * Main method which loads an Excel file into a table data class. - * - * @param file name of file to read - * @param columnLabelsPresent determines whether first line is treated - * as column titles - * @param columnsToRead list of column labels that should be read - all other - * columns will be dropped from the table data set - * @throws IOException - * - */ - public TableDataSet readFile(File file, String worksheetName, boolean columnLabelsPresent, String[] columnsToRead) throws IOException { - - if ((columnsToRead != null) && (columnLabelsPresent == false)) { - throw new RuntimeException("Column lables provided as filter but there are no column labels in Excel file"); - } - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - Workbook workbook = null; - //open workbook - try { - workbook = Workbook.getWorkbook( file); - } - catch (Throwable t) { - logger.error("Error attemting to open excel file " + file); - t.printStackTrace(); - } - - Sheet worksheet = workbook.getSheet(worksheetName); - if (worksheet==null) return null; - boolean[] readColumnFlag = null; - - if (columnLabelsPresent) { - readColumnFlag = readColumnLabels(worksheet, columnsToRead); - } - readData(worksheet, columnLabelsPresent, readColumnFlag); - - TableDataSet tds = makeTableDataSet(); - tds.setName(file.toString()); - return tds; - } - - - /** - * Read the excel file with a String[] of specified column formats (NUMBER or STRING), - * where the format is specified for all columns, all columns are read, - * and column headings must be present on the first line. - * - * @param file File object of excel workbook - * @param worksheetName the name of the worksheet to read - * @param columnFormats An array of column formats. - * - * @return A tableDataSet object containing the data in the worksheet. - */ - public TableDataSet readFileWithFormats(File file, String worksheetName, String[] columnFormats) throws IOException { - - boolean columnLabelsPresent = true; - String[] columnsToRead = null; - - if ((columnsToRead != null) && (columnLabelsPresent == false)) { - throw new RuntimeException("Column lables provided as filter but there are no column labels in Excel file"); - } - - //Initialize class attributes - columnCount = 0; - rowCount = 0; - columnData = new ArrayList(); - columnLabels = new ArrayList(); - columnType = null; - - Workbook workbook = openFile(file); - Sheet worksheet = workbook.getSheet(worksheetName); - - boolean[] readColumnFlag = null; - - if (columnLabelsPresent) { - readColumnFlag = readColumnLabels(worksheet, columnsToRead); - } - - readData( worksheet, columnLabelsPresent, readColumnFlag, columnFormats); - - TableDataSet tds = makeTableDataSet(); - tds.setName(file.toString()); - return tds; - } - - - /** - * Open file method. - * @param file The file. - * @return The workbook - * @throws IOException - */ - private Workbook openFile(File file) throws IOException { - logger.debug("Opening excel file: "+file); - - Workbook workbook = null; - //open workbook - try { - workbook = Workbook.getWorkbook( file ); - } - catch (Throwable t) { - logger.error("Error attemting to open excel file " + file); - t.printStackTrace(); - } - return workbook; - } - - /** - * Read and parse the column titles from the first line of file. - */ - private boolean[] readColumnLabels(Sheet worksheet, String[] columnsToRead) - throws IOException { - - - //Read the first cell - Cell cell = worksheet.getCell(0,0); - - //Test for an empty file - if (cell.getContents().length()==0) { - throw new IOException("Error: first row in sheet looks like it's empty"); - } - - int count = countNumberOfColumns(worksheet); - - boolean[] readColumnFlag = new boolean[count]; - - //Initialize the readColumnFlag to false if the caller has supplied a - //list of columns. It will be turned to true based on a comparison of the - //column labels found in the file. Otherwise initialize to true. - for (int i=0; i < count; i++) { - if (columnsToRead != null) - readColumnFlag[i] = false; - else - readColumnFlag[i] = true; - } - - //Read column titles - int c = 0; - for (int i=0; i < count; i++) { - cell = worksheet.getCell(i, 0); - String column_name = cell.getContents(); - - //Check if column should be read based on list supplied by caller - if (columnsToRead != null) { - for (int j=0; j < columnsToRead.length; j++) { - if (columnsToRead[j].equalsIgnoreCase(column_name)) { - readColumnFlag[c] = true; - - columnLabels.add(column_name); - columnCount++; - break; - } - } - } - else { - columnLabels.add(column_name); - columnCount++; - } - c++; //the actual columnn number in the file being read - } - - //Debugging output - String msg = "column read flag = "; - for (int i=0; i < readColumnFlag.length; i++) { - if (readColumnFlag[i] == true) - msg += "true"; - else - msg += "false"; - if (i < (readColumnFlag.length-1)) - msg += ", "; - } - msg += "\n"; - logger.debug(msg); - - - return readColumnFlag; - } - - - /** - * Read and parse data portion of file. - */ - private void readData(Sheet worksheet, boolean columnLabelsPresent, - boolean[] readColumnFlag) - throws IOException { - - //Determine the number of lines in the file - rowCount = countNumberOfRows(worksheet); - - logger.debug("number of rows in file: " + rowCount); - if (columnLabelsPresent) { - rowCount--; - } - - //Process each line in the file - if (rowCount == 0) { - columnType = new int[columnCount]; - readColumnFlag = new boolean[columnCount]; - for (int col =0; col 0) { - inStream.readLine(); - } - - // read the data - ArrayList[] columnData = readData(inStream, dictionary); - inStream.close(); - - // make the table - TableDataSet tds = makeTableDataSet(columnData, dictionary); - tds.setName(file.toString()); - - return tds; - } - - /** - * Opens the file, and creates a buffered file reader to that file. - * - * @param file The file to open. - * @return Reader for the file - * @throws IOException - */ - private BufferedReader openFile(File file) throws IOException { - logger.debug("Opening file: "+file); - - BufferedReader inStream = null; - try { - inStream = new BufferedReader( new FileReader(file) ); - } - catch (IOException e) { - throw e; - } - - return inStream; - } - - /** - * Read and parse data portion of file. - * - * @param inStream The stream of input data. - * @param dictionary The dictionary defining how the columns are set up. - * - * @return An ArrayList of data for each column read. - * - * @throws IOException - */ - private ArrayList[] readData(BufferedReader inStream, TableDataSet dictionary) throws IOException { - - // set up the start, end and type arrays - int start[] = new int[dictionary.getRowCount()]; - int end[] = new int[dictionary.getRowCount()]; - String type[] = new String[dictionary.getRowCount()]; - for (int i=0; i(); - } - if (type[i].equals("STRING")) { - columnData[i] = new ArrayList(); - } - } - - //Process each line in the file - String line; - int lineNum = 1; - while ((line = inStream.readLine()) != null) { - String[] lineData = new String[dictionary.getRowCount()]; - for (int i=0; i end) { - throw new RuntimeException("Start greater than end position: "+start+">"+end); - } - - String type = dictionary.getStringValueAt(i, "TYPE"); - if (!type.equals("NUMBER") && !type.equalsIgnoreCase("STRING")) { - throw new RuntimeException("Column type must be NUMBER or STRING."); - } - - int labelInFile = (int) dictionary.getValueAt(i, "LABELINFILE"); - if (labelInFile!=0 && labelInFile!=1) { - throw new RuntimeException("LABELINFILE must be 0 or 1."); - } - } - - return dictionary; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataReader#readTable(java.lang.String) - */ - public TableDataSet readTable(String tableName) throws IOException { - File fileName = new File (getMyDirectory() + File.separator + tableName + ".dat"); - TableDataSet me= readFile(fileName); - me.setName(tableName); - return me; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataReader#close() - */ - public void close() { - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java deleted file mode 100644 index 14dc2bb..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/GeneralDecimalFormat.java +++ /dev/null @@ -1,53 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.text.*; - -/** - * This class provides a way to use scientific notation when necessary. - */ -public class GeneralDecimalFormat extends DecimalFormat { - final DecimalFormat nonScientific; - double useScientificAbove; - double useScientificBelow; - - public GeneralDecimalFormat(String scientificString, double useScientificAbove, double useScientificBelow) { - super(scientificString); - this.useScientificAbove = useScientificAbove; - this.useScientificBelow = useScientificBelow; - String nonScientificString = scientificString.replaceAll("E0+",""); - nonScientific = new DecimalFormat(nonScientificString); - } - - public StringBuffer format(double val, StringBuffer buffer, FieldPosition f) { - if (val ==0.0 || (Math.abs(val) <= useScientificAbove && Math.abs(val) > useScientificBelow)) { - return nonScientific.format(val, buffer, f); - } else { - return super.format(val,buffer,f); - } - } - - public StringBuffer format(long val, StringBuffer buffer, FieldPosition f) { - if (val ==0.0 || (Math.abs(val) <= useScientificAbove && Math.abs(val) > useScientificBelow)) { - return nonScientific.format(val, buffer, f); - } else { - return super.format(val,buffer,f); - } - } - -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java deleted file mode 100644 index a010bfc..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableReader.java +++ /dev/null @@ -1,221 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.IOException; -import java.sql.*; -import java.util.ArrayList; -import java.util.List; -import org.apache.log4j.Logger; - -import com.pb.common.sql.JDBCConnection; -import com.pb.common.sql.SQLExecute; - -/** - * Creates a TableData class from a table in a JDBC data source. - * - * @author Tim Heier - * @version 1.0, 2/07/2004 - * - */ -public class JDBCTableReader extends TableDataReader { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - private JDBCConnection jdbcConn = null; - - private boolean mangleTableNamesForExcel = false; - - - public JDBCTableReader (JDBCConnection jdbcConn) { - if (jdbcConn == null) { - throw new RuntimeException("Database connection is null"); - } - - this.jdbcConn = jdbcConn; - } - - - /** - * Load all columns and rows of table. - */ - public TableDataSet readTable(String tableName) throws IOException { - String mangledTableName = tableName; - if (mangleTableNamesForExcel) { - logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); - mangledTableName = "["+tableName+"$]"; - } - TableDataSet theTable = null; - try { - theTable = loadTable(tableName, "SELECT * FROM " + mangledTableName); - theTable.setName(tableName); - } catch (RuntimeException e) { - logger.warn("Table "+tableName+" can not be read by JDBCTableReader, "+e.toString()); - // want to return null object if table doesn't exist, to be consistent with other table readers. - theTable = null; - } - return theTable; - } - - - /** - * Load table using specified query string. - */ - private TableDataSet loadTable(String tableName, String sqlString) throws IOException { - List columnData = new ArrayList(); - String[] columnLabels; - int columnCount; - - logger.debug("JDBCTableReader, table name: " + tableName); - logger.debug("JDBCTableReader, SQL String: " + sqlString); - - if (mangleTableNamesForExcel) { - logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); - tableName = "["+tableName+"$]"; - } - - Statement stmt = null; - ResultSet rs = null; - ResultSetMetaData metaData = null; - - SQLExecute sqlExecute = new SQLExecute(jdbcConn); - try { - int rowCount = sqlExecute.getRowCount(tableName); - - //Run main query - rs = sqlExecute.executeQuery(sqlString); - metaData = rs.getMetaData(); - - columnCount = sqlExecute.getColumnCount(); - columnLabels = sqlExecute.getColumnLabels(); - int[] columnType = new int[columnCount]; - - //Set up a vector of arrays to hold the result set. Store the - //column type at the same time. Each vector holds a column of data. - for (int c = 0; c < columnCount; c++) { - int type = metaData.getColumnType(c+1); - - switch(type) { - //Map these types to STRING - case Types.CHAR: - case Types.VARCHAR: - case Types.LONGVARCHAR: - case Types.DATE: - case Types.TIME: - case Types.TIMESTAMP: - case Types.BIT: - columnData.add(new String[rowCount]); - columnType[c] = DataTypes.STRING; - break; - //Map these types to NUMBER - case Types.TINYINT: - case Types.SMALLINT: - case Types.INTEGER: - case Types.BIGINT: - case Types.FLOAT: - case Types.REAL: - case Types.DOUBLE: - case Types.DECIMAL: - case Types.NUMERIC: - columnData.add(new float[rowCount]); - columnType[c] = DataTypes.NUMBER; - break; - default: - System.err.println("**error** unknown column data type, column=" + c + - ", type=" + type); - break; - } - } - - //Read result set and store the data into column-wise arrays. - //Column data in a ResultSet starts in 1,2... - int row = 0; - while (rs.next()) { - for (int c=0; c < columnCount; c++) { - int type = columnType[c]; - - switch(type) { - case DataTypes.STRING: - String[] s = (String[]) columnData.get(c); - s[row] = rs.getString(c+1); - break; - case DataTypes.NUMBER: - float[] f = (float[]) columnData.get(c); - f[row] = rs.getFloat(c+1); - break; - default: - System.err.println("**error** unknown column data type - should not be here"); - break; - } - } - row++; - } - } - catch (SQLException e) { - throw new IOException(e.getMessage()); - } -// finally { -// try { -// rs.close(); -// stmt.close(); -// } -// catch (SQLException e) { -// e.printStackTrace(); -// } -// } - - TableDataSet tds = makeTableDataSet(columnData, columnLabels, columnCount); - tds.setName(tableName); - return tds; - } - - - private TableDataSet makeTableDataSet(List columnData, String[] columnLabels, int columnCount) { - - TableDataSet table = new TableDataSet(); - - for (int i=0; i < columnCount; i++) { - table.appendColumn(columnData.get(i), columnLabels[i]); - } - - return table; - } - - - /** - * @return Returns the mangleTableNamesForExcel. - */ - public boolean isMangleTableNamesForExcel() { - return mangleTableNamesForExcel; - } - - /** - * @param mangleTableNamesForExcel The mangleTableNamesForExcel to set. - */ - public void setMangleTableNamesForExcel(boolean mangleTableNamesForExcel) { - this.mangleTableNamesForExcel = mangleTableNamesForExcel; - } - - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataReader#close() - */ - public void close() { - jdbcConn.close(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java deleted file mode 100644 index 7c4a487..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/JDBCTableWriter.java +++ /dev/null @@ -1,136 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.IOException; -import org.apache.log4j.Logger; - -import com.pb.common.sql.JDBCConnection; -import com.pb.common.sql.SQLExecute; - -/** - * @author jabraham - * - * This class writes TableDataSets to an SQL Database - */ -public class JDBCTableWriter extends TableDataWriter { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile"); - - private JDBCConnection jdbcConn = null; - - private boolean mangleTableNamesForExcel = false; - - - - public void writeTable(TableDataSet tableData, String tableName) - throws IOException { - String insertMangledTableName = tableName; - if (mangleTableNamesForExcel) { - logger.debug("Mangling table name "+tableName+" to ["+tableName+"$]"); - insertMangledTableName = "["+tableName+"$]"; - } - String createMangledTableName = tableName; - if (mangleTableNamesForExcel) { - logger.debug("Mangling table name "+tableName+" to ["+tableName+"]"); - createMangledTableName = "["+tableName+"]"; - } - saveTable(createMangledTableName, insertMangledTableName, tableData); - } - - - /** - * @param mangledTableName - * @param tableData - */ - private void saveTable(String createMangledTableName, String insertMangledTableName, TableDataSet tableData) { - - //TODO: First we need to check to see if the table already exists and then delete it. - - // first create the table - StringBuffer createStatement = new StringBuffer("CREATE TABLE "+createMangledTableName+ " ("); - int[] columnTypes = tableData.getColumnType(); - for (int c =0; c nCols)) { - String msg = "Column number out of range: " + column; - msg += (", number of columns: " + nCols); - - throw new RuntimeException(msg); - } - - if (columnType[column] != type) { - throw new RuntimeException("column " + column + " is type " + - columnType[column] + " not type " + type); - } - } - - /** - * Return the name of the column given a postion. Column numbers are 1-based. - * - * @return The name of the column or an empty string if the requested - * field is out of bounds. - */ - public String getColumnLabel(int column) { - //Data is 0 based so subtract 1 from what the user supplies - column = column - 1; - - if ((columnLabels != null) && (column >= 0) && - (column < columnLabels.size())) { - return (String) columnLabels.get(column); - } else { - return ""; - } - } - - /** - * Return the postion of a column given the name. Column numbers are 1-based. - * - * @return -1 if the requested column name is not found. - * - */ - public int getColumnPosition(String columnName) { - int position = -1; - - for (int col = 0; col < columnLabels.size(); col++) { - String currentColumn = (String) columnLabels.get(col); - if (currentColumn.equalsIgnoreCase(columnName)) { - position = col + 1; - break; - } - } - - return position; - } - - public int checkColumnPosition(String columnName) throws RuntimeException { - int position = getColumnPosition(columnName); - if (position <0) throw new RuntimeException("Column "+columnName+" does not exist in TableDataSet "+getName()); - return position; - } - - /** - * Indicates whether or not the table constains the specified column. - * - * @param columnName Name of the column to check. - * @return boolean indicating if the column is present. - */ - public boolean containsColumn(String columnName) { - int position = getColumnPosition(columnName); - if (position>=0) return true; - else return false; - } - - /** - * Return the values in a specified row as a float[] - * - * @param row row number to retrieve, values are 0-based - * - * @throws RuntimeException when one of the columns is of type STRING - * - */ - public float[] getRowValues(int row) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - - int startPosition = 0; //position to start in array - - float[] rowValues = new float[nCols + startPosition]; - - for (int c = 0; c < nCols; c++) { - if (columnType[c] == STRING) { - throw new RuntimeException("column " + c + 1 + - " is of type STRING"); - } - - float[] f = (float[]) columnData.get(c); - - rowValues[c + startPosition] = f[row]; - } - - return rowValues; - } - - /** - * Return a values from a specified row using the indexed column. - * - * @param row indexed row number - */ - public float[] getIndexedRowValuesAt(int row) { - if (columnIndex == null) { - throw new RuntimeException("No index defined."); - } - - row = columnIndex[row] + 1; // getRowValues will subtract 1 - - return getRowValues(row); - } - - /** - * Return the row number from the index. - * - * @param index indexed row number - * @return the row number associated with the index - */ - public int getIndexedRowNumber(int index) { - if (columnIndex == null) { - throw new RuntimeException("No index defined."); - } - - return columnIndex[index] + 1; // getRowValues will subtract 1 - } - - - /** - * Return the values in a specified row as a String[] - * - * @param row row number to retrieve, values are 0-based - * - * @throws RuntimeException when one of the columns is of type STRING - * - */ - public String[] getRowValuesAsString(int row) { - //TODO lookup based on indexed column - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - - int startPosition = 0; //position to start in array - - String[] rowValues = new String[nCols + startPosition]; - - for (int c = 0; c < nCols; c++) { - switch (columnType[c]) { - case STRING: - String[] s = (String[]) columnData.get(c); - rowValues[c + startPosition] = s[row]; - break; - case NUMBER: - float[] f = (float[]) columnData.get(c); - rowValues[c + startPosition] = valueFormat.format(f[row]); - break; - } - } - - return rowValues; - } - - /** - * Returns a copy of the values in the table as a float[][] - */ - public float[][] getValues() { - float[][] tableValues = new float[nRows][nCols]; - - for (int c = 0; c < nCols; c++) { - if (columnType[c] == STRING) { - throw new RuntimeException("column " + c + 1 + - " is of type STRING"); - } - - float[] f = (float[]) columnData.get(c); - - for (int r = 0; r < nRows; r++) { - tableValues[r][c] = f[r]; - } - } - - return tableValues; - } - - /** - * Return a value from a specified row and column. For speed, the column - * type is not checked. A RuntimeException will be thrown if the column is - * of type NUMBER. - * - */ - public float getValueAt(int row, int column) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - column = column - 1; - - - //TODO lookup based on indexed column - float[] f = null; - try { - f = (float[]) columnData.get(column); - } catch (ClassCastException e) { - throw new RuntimeException("Column "+column+" in TableDataSet is not float values",e); - } - - return f[row]; - } - - public float getValueAt(int row, String columnName) { - int columnNumber = getColumnPosition(columnName); - - if (columnNumber <= 0) { - logger.error("no column named " + columnName + " in TableDataSet"); - throw new RuntimeException("no column named " + columnName + - " in TableDataSet"); - } - - return getValueAt(row, columnNumber); - } - - /** - * Return a value from a specified row and column. If the column type is - * not STRING then the numeric value will be converted to string before - * it is returned. - * - */ - public String getStringValueAt(int row, int column) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - column = column - 1; - - String value; - - if (columnType[column] == NUMBER) { - float[] f = (float[]) columnData.get(column); - value = valueFormat.format(f[row]); - } else { - String[] s = (String[]) columnData.get(column); - value = s[row]; - } - - return value; - } - - public boolean getBooleanValueAt(int row, String columnName) { - return getBooleanValueAt(row,checkColumnPosition(columnName)); - } - /** - * Return a value from a specified row and column. - * - */ - public boolean getBooleanValueAt(int row, int column) { - String boolString = getStringValueAt(row,column); - boolString = boolString.trim(); - if (boolString == null) throw new RuntimeException("Boolean value in TableDataSet "+name+" is blank (null)"); - if (use1sAnd0sForTrueFalse ) { - if (boolString.equalsIgnoreCase("1")) return true; - if (boolString.equalsIgnoreCase("0")) return false; - } - /* ABDEL M.:I added boolString.equalsIgnoreCase("t") and boolString.equalsIgnoreCase("f") to the condition below. - The method getStringValueAt(row,column) above returns t or f rather than true or false. It is not the best way to fix the issue but this is a work around - for now till John A. comes and check it. - */ - if (boolString.equalsIgnoreCase("true") || boolString.equalsIgnoreCase("t")) return true; - if (boolString.equalsIgnoreCase("false") || boolString.equalsIgnoreCase("f")) return false; - throw new RuntimeException("Boolean value in table dataset "+name+" column "+ column+ " is neither true nor false, but ('"+boolString+"')."); - } - - public void setBooleanValueAt(int row, String columnName, boolean value) { - setBooleanValueAt(row,checkColumnPosition(columnName), value); - } - - public void setBooleanValueAt(int row, int column, boolean value) { - if (value) setStringValueAt(row,column,"true"); - else setStringValueAt(row,column,"false"); - } - - /** - * Return a value from a specified row and column. For speed, the column - * type is not checked. A RuntimeException will be thrown if the column is - * of type STRING. - * - */ - public String getStringValueAt(int row, String columnName) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - - int columnNumber = getColumnPosition(columnName); - - if (columnNumber <= 0) { - logger.error("no column named " + columnName + " in TableDataSet"); - - throw new RuntimeException("no column named " + columnName + - " in TableDataSet"); - } - - //Call with 1-based row and column numbers - return getStringValueAt(row + 1, columnNumber); - } - - /** - * Set at a value using the column name. - */ - public void setValueAt(int row, String colName, float newValue) { - int col = getColumnPosition(colName); - setValueAt(row, col, newValue); - } - - /** - * Return a value from a specified row and column. For speed, the column - * type is not checked. A RuntimeException will be thrown if the column is - * of type NUMBER. - * - */ - public void setValueAt(int row, int column, float newValue) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - column = column - 1; - - float[] f = (float[]) columnData.get(column); - - f[row] = newValue; - - // any TableDataSetIndex that uses this column will have to regenerate its index. - if (indexColumns[column]==true) { - fireIndexValuesChanged(); - } - setDirty(true); - } - - /** - * update the column specified with the int values specified. - * - */ - public void setColumnAsInt ( int column, int[] newValues ) { - //Data is 0 based so subtract 1 from what the user supplies - column = column - 1; - - float[] f = new float[newValues.length]; - for (int i=0; i < newValues.length; i++) - f[i] = (float)newValues[i]; - - columnData.set( column, f ); - if (indexColumns[column]==true) { - fireIndexValuesChanged(); - } - - setDirty(true); - } - - /** - * update the column specified with the float values specified. - * - */ - public void setColumnAsFloat ( int column, float[] newValues ) { - //Data is 0 based so subtract 1 from what the user supplies - column = column - 1; - - columnData.set( column, newValues ); - if (indexColumns[column]==true) { - fireIndexValuesChanged(); - } - - setDirty(true); - } - - /** - * update the column specified with the double values specified. - * - */ - public void setColumnAsDouble ( int column, double[] newValues ) { - //Data is 0 based so subtract 1 from what the user supplies - column = column - 1; - - float[] f = new float[newValues.length]; - for (int i=0; i < newValues.length; i++) - f[i] = (float)newValues[i]; - - columnData.set( column, f ); - if (indexColumns[column]==true) { - fireIndexValuesChanged(); - } - - setDirty(true); - } - - public void setIndexedValueAt(int row, String colName, float newValue) { - if (columnIndex == null) { - throw new RuntimeException("No index defined."); - } - row = columnIndex[row] + 1; - - int col = getColumnPosition(colName); - setValueAt(row, col, newValue); - } - - /** - * Return a value from an indexed row and column. For speed, the column - * type is not checked. A RuntimeException will be thrown if the column is - * of type NUMBER. - * - */ - public void setIndexedValueAt(int row, int column, float newValue) { - if (columnIndex == null) { - throw new RuntimeException("No index defined."); - } - - row = columnIndex[row] + 1; // getRowValues will subtract 1 - setValueAt(row, column, newValue); - } - - /** - * Return a value from a specified row and column. For speed, the column - * type is not checked. A RuntimeException will be thrown if the column is - * of type STRING. - * - */ - public void setStringValueAt(int row, int column, String newValue) { - //Data is 0 based so subtract 1 from what the user supplies - row = row - 1; - column = column - 1; - - //TODO lookup based on indexed column - String[] s = (String[]) columnData.get(column); - - s[row] = newValue; - setDirty(true); - if (indexColumns[column]==true) { - fireIndexValuesChanged(); - } - if (column == stringIndexColumn) - stringIndexDirty = true; - } - - /** - * Set the value of a {@code STRING} column at a specified row. - * - * @param row - * The (1-based) row number. - * - * @param column - * The name of the column. - * - * @param value - * The value to place in {@code column} at {@code row}. - * - * @throws ClassCastException if {@code column} is not a {@code STRING} column. - * @throws RuntimeException if {@code column} is not found in this table. - */ - public void setStringValueAt(int row, String column, String value) { - setStringValueAt(row,checkColumnPosition(column),value); - } - - public void setIndexColumnNames(String[] indexColumnNames) { - for (int i=0;i= 0 ) - continue; - - logger.info("Adding column " + tdsInHeadings[c] + " to TableDataSet due to merge"); - if ( tdsInTypes[c] == NUMBER ) { - float[] newColumn = new float[nRows]; - - for (int r = 0; r < nRows; r++) - newColumn[r] = tdsIn.getValueAt( r+1, c+1 ); - - appendColumn( newColumn, tdsInHeadings[c] ); - } - else if ( tdsInTypes[c] == STRING ) { - String[] newColumn = new String[nRows]; - - for (int r = 0; r < nRows; r++) - newColumn[r] = tdsIn.getStringValueAt( r+1, c+1 ); - - appendColumn( newColumn, tdsInHeadings[c] ); - } - } - setDirty(true); - - } - - /** - * Static method to log the frequency of values in - * a column specified by the user. - * - * @param tableName String identifying contents of TableDataSet - * @param tds containing column for creating frequency table - * @param columnPosition position of desired column - * - */ - public static void logColumnFreqReport(String tableName, TableDataSet tds, - int columnPosition) { - if (tds.getRowCount() == 0) { - logger.info(tableName + " Table is empty - no data to summarize"); - - return; - } - - float[] columnData = new float[tds.getRowCount()]; - int[] sortValues = new int[tds.getRowCount()]; - - for (int r = 1; r <= tds.getRowCount(); r++) { - columnData[r - 1] = tds.getValueAt(r, columnPosition); - sortValues[r - 1] = (int) (columnData[r - 1] * 10000); - } - - // sort the column elements - int[] index = IndexSort.indexSort(sortValues); - - ArrayList bucketValues = new ArrayList(); - ArrayList bucketSizes = new ArrayList(); - - // count the number of identical elements into buckets - float oldValue = columnData[index[0]]; - int count = 1; - - for (int r = 1; r < tds.getRowCount(); r++) { - if (columnData[index[r]] > oldValue) { - bucketValues.add(Float.toString(oldValue)); - bucketSizes.add(Integer.toString(count)); - count = 0; - oldValue = columnData[index[r]]; - } - - count++; - } - - bucketValues.add(Float.toString(oldValue)); - bucketSizes.add(Integer.toString(count)); - - // print a simple summary table - logger.info("Frequency Report table: " + tableName); - logger.info("Frequency for column " + columnPosition + ": " + - (tds.getColumnLabel(columnPosition))); - logger.info(String.format("%8s", "Value") + - String.format("%11s", "Frequency")); - - int total = 0; - - for (int i = 0; i < bucketValues.size(); i++) { - float value = Float.parseFloat((String) (bucketValues.get(i))); - logger.info(String.format("%8.0f", value) + - String.format("%11d", Integer.parseInt((String) (bucketSizes.get(i))))); - total += Integer.parseInt((String) (bucketSizes.get(i))); - } - - logger.info(String.format("%8s", "Total") + - String.format("%11d\n\n\n", total)); - } - - /** - * Logs the frequency of values in a column specified by the user. - * The array list argument can hold descriptions of the values - * if they are known. For example if a column lists alternatives 1-5, you - * might know that alt 1= 0_autos, alt2=1_auto, etc. - * - * @param tableName String identifying contents of TableDataSet - * @param tds containing column for creating frequency table - * @param columnPosition position of desired column - * - */ - public static void logColumnFreqReport(String tableName, TableDataSet tds, - int columnPosition, String[] descriptions) { - if (tds.getRowCount() == 0) { - logger.info(tableName + " Table is empty - no data to summarize"); - - return; - } - - float[] columnData = new float[tds.getRowCount()]; - int[] sortValues = new int[tds.getRowCount()]; - - for (int r = 1; r <= tds.getRowCount(); r++) { - columnData[r - 1] = tds.getValueAt(r, columnPosition); - sortValues[r - 1] = (int) (columnData[r - 1] * 10000); - } - - // sort the column elements - int[] index = IndexSort.indexSort(sortValues); - - ArrayList bucketValues = new ArrayList(); - ArrayList bucketSizes = new ArrayList(); - - // count the number of identical elements into buckets - float oldValue = columnData[index[0]]; - int count = 1; - - for (int r = 1; r < tds.getRowCount(); r++) { - if (columnData[index[r]] > oldValue) { - bucketValues.add(Float.toString(oldValue)); - bucketSizes.add(Integer.toString(count)); - count = 0; - oldValue = columnData[index[r]]; - } - - count++; - } - - bucketValues.add(Float.toString(oldValue)); - bucketSizes.add(Integer.toString(count)); - - // print a simple summary table - logger.info("Frequency Report table: " + tableName); - logger.info("Frequency for column " + columnPosition + ": " + - (tds.getColumnLabel(columnPosition))); - logger.info(String.format("%8s", "Value") + - String.format("%13s", "Description") + - String.format("%11s", "Frequency")); - - if(descriptions!=null) - if(bucketValues.size() != descriptions.length) - logger.fatal("The number of descriptions does not match the number of values in your data"); - - int total = 0; - - for (int i = 0; i < bucketValues.size(); i++) { - float value = Float.parseFloat((String) (bucketValues.get(i))); - String description = ""; //default value as sometime certain columns don't have descriptions - if(descriptions !=null) { - description = descriptions[i]; - } - logger.info(String.format("%8.0f", value) + " " + String.format("%-11s", description) + - String.format("%11d", Integer.parseInt((String) (bucketSizes.get(i))))); - total += Integer.parseInt((String) (bucketSizes.get(i))); - } - - logger.info(String.format("%23s", "Total") + - String.format("%9d\n\n\n", total)); - } - - - /** - * @param index - */ - public void removeChangeListener(ChangeListener index) { - changeListeners.remove(index); - } - - - public void setName(String name) { - this.name = name; - } - - - public String getName() { - return name; - } - - /* (non-Javadoc) - * @see java.lang.Object#toString() - */ - public String toString() { - return "TableDataSet "+name; - } - - - public interface TableDataSetWatcher { - public void isBeingForgotten(TableDataSet s); - public void isDirty(TableDataSet s); - } - - private ArrayList myWatchers = null; - - public void addFinalizingListener(TableDataSetWatcher watcher) { - if (myWatchers ==null) { - myWatchers = new ArrayList(); - } - myWatchers.add(watcher); - } - - - void tellWatchersImBeingForgotten() { - if (myWatchers != null) { - for (int w=0;w stringIndex = null; //map of string index keys to 1-based row numbers - private int stringIndexColumn = -1; - private boolean stringIndexDirty; - - /** - * Build a string index on the specified column. The values in the column must be unique. If the column is - * modified after the index is built, the index must be rebuilt (using this function) before it can be used again. - * - * @param column - * The column to build the index on. - * - * @throws IllegalStateException if the column contains repeated values. - * @throws RuntimeException if the column is not of type {@code STRING}. - */ - public void buildStringIndex(int column) { - checkColumnNumber(column, STRING); - column--; //changed to a zero based index - String[] columnValues = (String[]) columnData.get(column); - - stringIndex = new HashMap(columnValues.length); - Set repeatedValues = new HashSet(); - - for (int i = 0; i < columnValues.length; i++) - if (stringIndex.put(columnValues[i],i+1) != null) - repeatedValues.add(columnValues[i]); - - if (repeatedValues.size() > 0) { - //nulify the index, as it is invalid anyway, in case somebody catches this exception and tries to carry on - stringIndex = null; - stringIndexColumn = -1; - throw new IllegalStateException("String index cannot be built on a column with non-unique values." + - "The following values have been repeated: " + Arrays.toString(repeatedValues.toArray(new String[repeatedValues.size()]))); - } - - stringIndexColumn = column; - stringIndexDirty = false; - } - - private void checkStringIndexValue(String index) { - if (stringIndex == null) - throw new IllegalStateException("No string index exists for this table."); - if (stringIndexDirty) - throw new IllegalStateException("String index column changed, must be rebuilt."); - if (!stringIndex.containsKey(index)) - throw new IllegalArgumentException("String value not found in index: " + index); - } - - /** - * Get the (0-based) row number for the given string index value. - * - * @param index - * The string index value. - * - * @return the row number corresponding to index {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - */ - public int getStringIndexedRowNumber(String index) { - checkStringIndexValue(index); - return stringIndex.get(index)-1; - } - - /** - * Get the value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - * @throws RuntimeException if {@code column} does not hold {@code float}s. - */ - public float getStringIndexedValueAt(String index, int column) { - checkStringIndexValue(index); - return getValueAt(stringIndex.get(index),column); - } - - /** - * Get the value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table, or does not hold {@code float}s. - */ - public float getStringIndexedValueAt(String index, String column) { - checkStringIndexValue(index); - return getValueAt(stringIndex.get(index),column); - } - - /** - * Get the boolean value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - * @throws RuntimeException if {@code column} does not hold {@code boolean}s. - */ - public boolean getStringIndexedBooleanValueAt(String index, int column) { - checkStringIndexValue(index); - return getBooleanValueAt(stringIndex.get(index),column); - } - - /** - * Get the boolean value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table, or does not hold {@code boolean}s. - */ - public boolean getStringIndexedBooleanValueAt(String index, String column) { - checkStringIndexValue(index); - return getBooleanValueAt(stringIndex.get(index),column); - } - - /** - * Get the string value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - */ - public String getStringIndexedStringValueAt(String index, int column) { - checkStringIndexValue(index); - return getStringValueAt(stringIndex.get(index),column); - } - - /** - * Get the string value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @return the value in {@code column} at the row specified by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table. - */ - public String getStringIndexedStringValueAt(String index, String column) { - checkStringIndexValue(index); - return getStringValueAt(stringIndex.get(index),column); - } - - /** - * Set the value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - * @throws ClassCastException if {@code column} does not hold {@code float}s. - */ - public void setStringIndexedValueAt(String index, int column, float value) { - checkStringIndexValue(index); - setValueAt(stringIndex.get(index),column,value); - } - - /** - * Set the value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table. - * @throws ClassCastException if {@code column} does not hold {@code float}s. - */ - public void setStringIndexedValueAt(String index, String column, float value) { - checkStringIndexValue(index); - setValueAt(stringIndex.get(index),column,value); - } - - /** - * Set the boolean value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - * @throws ClassCastException if {@code column} does not hold {@code boolean}s. - */ - public void setStringIndexedBooleanValueAt(String index, int column, boolean value) { - checkStringIndexValue(index); - setBooleanValueAt(stringIndex.get(index),column,value); - } - - /** - * Set the boolean value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table. - * @throws ClassCastException if {@code column} does not hold {@code boolean}s. - */ - public void setStringIndexedBooleanValueAt(String index, String column, boolean value) { - checkStringIndexValue(index); - setBooleanValueAt(stringIndex.get(index),column,value); - } - - /** - * Set the string value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The (1-based) column number. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws IndexOutOfBoundsException if {@code column} is less than 1 or greater than the number of columns in this table. - * @throws ClassCastException if {@code column} does not hold {@code String}s. - */ - public void setStringIndexedStringValueAt(String index, int column, String value) { - checkStringIndexValue(index); - setStringValueAt(stringIndex.get(index),column,value); - } - - /** - * Set the string value for the specified column and row (specified by string index). - * - * @param index - * The index specifying the row. - * - * @param column - * The column name. - * - * @param value - * The value to place in {@code column} at the row indexed by {@code index}. - * - * @throws IllegalStateException if no string index has been built for this table, or if the index needs to be - * rebuilt (because the index column has been modified after the index was built). - * @throws IllegalArgumentException if {@code index} is not found in the index column. - * @throws RuntimeException if {@code column} is not found in the table. - * @throws ClassCastException if {@code column} does not hold {@code String}s. - */ - public void setStringIndexedStringValueAt(String index, String column, String value) { - checkStringIndexValue(index); - setStringValueAt(stringIndex.get(index),column,value); - } - - //The user must ensure that the keys in the HashMap correspond - //to the headers in the table and that each column in the table - //has a value in the HashMap that is of the correct type. This - //method will not do a lot of error checking or handling. - // - // - public void appendRow(HashMap rowData){ - - int type; - String[] headers = getColumnLabels(); - int columnNum = 1; - for(String header : headers){ - System.out.println("Header Value to Append: " + header); - type = getColumnType()[getColumnPosition(header)-1]; - if(type == DataTypes.NUMBER){ - float[] col = getColumnAsFloat(header); - float[] newCol = new float[col.length+1]; - System.arraycopy(col, 0, newCol, 0, col.length); - newCol[newCol.length-1] = (Float) rowData.get(header); - replaceFloatColumn(columnNum, newCol); - }else if(type == DataTypes.STRING){ - String[] col = getColumnAsString(header); - String[] newCol = new String[col.length+1]; - System.arraycopy(col, 0, newCol, 0, col.length); - newCol[newCol.length-1] = (String) rowData.get(header); - replaceStringColumn(columnNum, newCol); - } - columnNum++; - } - nRows++; - - } - - //column positions are 1-number of columns but columnData ArrayList - //is zero-based so subtract 1 from the supplied colNumber - public void replaceFloatColumn(int colNumber, float[] newData){ - columnData.remove(colNumber-1); - columnData.add(colNumber-1, newData); - } - - //column positions are 1-number of columns but columnData ArrayList - //is zero-based so subtract 1 from the supplied colNumber - public void replaceStringColumn(int colNumber, String[] newData){ - columnData.remove(colNumber-1); - columnData.add(colNumber-1, newData); - } - -} - - diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java deleted file mode 100644 index 1270832..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCacheCollection.java +++ /dev/null @@ -1,218 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.IOException; -import java.lang.ref.Reference; -import java.lang.ref.ReferenceQueue; -import java.lang.ref.SoftReference; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; - -import org.apache.log4j.Logger; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to Window - - * Preferences - Java - Code Generation - Code and Comments - */ -public class TableDataSetCacheCollection extends TableDataSetCollection implements TableDataSet.TableDataSetWatcher { - - private static Logger logger = Logger.getLogger(TableDataSetCollection.class); - private HashMap readDataSoftReferences = new HashMap(); - - private HashMap dirtyDataSetMap = new HashMap(); - - ReferenceQueue myReferenceQueue = new ReferenceQueue(); - - public TableDataSetCacheCollection(TableDataReader reader, TableDataWriter writer) { - super(reader,writer); -// Runnable cleanUp = new Runnable() { -// public void run() { -// while (true) { -// try { -// TableDataSetSoftReference fred = (TableDataSetSoftReference) myReferenceQueue.remove(); -// logger.info("TableDataSet "+fred.name+" is gone"); -// readDataSoftReferences.remove(fred.name); -// } catch (InterruptedException e) { -// throw new RuntimeException("TableDataSetCacheCollection cleanup thread is being interrupted",e); -// } -// } -// } -// }; -// Thread cleanUpThread = new Thread(cleanUp); -// cleanUpThread.start(); - } - - - - /** - * @param name - * @return the TableDataSet requested - */ - public synchronized TableDataSet getTableDataSet(String name) { - // first see if we can remove references to datasets that we don't use - TableDataSetSoftReference fred; - fred = (TableDataSetSoftReference) myReferenceQueue.poll(); - while (fred !=null) { - if (logger.isDebugEnabled()) logger.debug("Removing key "+fred.name+" from list of TableDataSets read"); - readDataSoftReferences.remove(fred.name); - fred = (TableDataSetSoftReference) myReferenceQueue.poll(); - } - SoftReference wr = (SoftReference) readDataSoftReferences.get(name); - TableDataSet theTable = null; - if (wr!=null) { - theTable = (TableDataSet) wr.get(); - } - // could be the case that the soft reference was cleared before it became dirty, check - // to see if we can find it in our dirty list. - if (theTable == null) { - theTable = (TableDataSet) dirtyDataSetMap.get(name); - if (theTable!=null) { - addTableToTempStorage(theTable); - } - } - if (theTable == null) { - try { - if (logger.isDebugEnabled()) logger.debug("reading table "+name); - theTable = getMyReader().readTable(name); - double freeMem = Runtime.getRuntime().freeMemory()/1000000.0; - if (logger.isDebugEnabled()) logger.debug("Memory is "+freeMem); - } catch (IOException e) { - e.printStackTrace(); - } - if (theTable == null) - throw new RuntimeException("Can't read in table " + name); - addTableToTempStorage(theTable); - theTable.addFinalizingListener(this); - } - return theTable; - } - - - - private void addTableToTempStorage(TableDataSet theTable) { - if (logger.isDebugEnabled()) logger.debug("Creating temporary references for "+theTable.getName()); - TableDataSetSoftReference fred = new TableDataSetSoftReference(theTable, myReferenceQueue); - readDataSoftReferences.put(theTable.getName(), fred); - } - - public synchronized void flushAndForget(TableDataSet me) { - if (me.isDirty()) { - writeTableToDisk(me); - } - //TODO remove this next line, shouldn't need to manually remove it from the SoftReferences. - readDataSoftReferences.remove(me.getName()); - dirtyDataSetMap.remove(me.getName()); - } - - - private void writeTableToDisk(TableDataSet me) { - try { - if (logger.isDebugEnabled()) logger.debug("writing table "+me.getName() + ". Table has " + me.getRowCount() + " rows"); - getMyWriter().writeTable(me, me.getName()); - me.setDirty(false); - dirtyDataSetMap.remove(me.getName()); - } catch (IOException e1) { - e1.printStackTrace(); - throw new RuntimeException("Can't write out table " + me.getName()); - } - } - - /* - * (non-Javadoc) - * - * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#flush() - */ - public synchronized void flush() { - Iterator it = dirtyDataSetMap.entrySet().iterator(); - while (it.hasNext()) { - Map.Entry e = (Map.Entry) it.next(); - TableDataSet t = (TableDataSet) e.getValue(); - if (t.isDirty()) { - writeTableToDisk(t); - } else { - // if (logger.isDebugEnabled()) logger.debug("*don't need to flush* table "+t.getName()+" as it's not dirty"); - } - } - dirtyDataSetMap.clear(); - it = readDataSoftReferences.entrySet().iterator(); - while (it.hasNext()) { - Map.Entry e = (Map.Entry) it.next(); - TableDataSet t = (TableDataSet) ((Reference) e.getValue()).get(); - if (t == null) { - if (logger.isDebugEnabled()) logger.debug("soft reference to table "+e.getKey()+" has been cleared"); - } else { - if (t.isDirty()) { - logger.error("Dirty table not in dirtyDataSetMap"); - writeTableToDisk(t); - } else { - // if (logger.isDebugEnabled()) logger.debug("*don't need to flush* table "+t.getName()+" as it's not dirty"); - } - } - } - } - - /* - * Call flush first if any changes need to be written out - * - * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#invalidate() - */ - public synchronized void invalidate() throws IOException { - //TODO should we do something to remove them from the reference queue? - readDataSoftReferences.clear(); - dirtyDataSetMap.clear(); - super.invalidate(); - } - /** - * @param aTable the TableDataSet to add to the colleciton - */ - public synchronized void addTableDataSet(TableDataSet aTable) { - addTableToTempStorage(aTable); - aTable.setDirty(true); - aTable.addFinalizingListener(this); - } - - - public synchronized void isBeingForgotten(TableDataSet t) { - if (logger.isDebugEnabled()) { - double freeMem = Runtime.getRuntime().freeMemory() / 1000000.0; - logger.debug("getting ready to forget about table " + t.getName() + " freeMem=" - + freeMem); - } - } - - public synchronized void isDirty(TableDataSet s) { - if (logger.isDebugEnabled()) logger.debug("Table "+s+" is now dirty, creating a new soft reference to it now"); - dirtyDataSetMap.put(s.getName(),s); - addTableToTempStorage(s); - } - - @Override - protected void finalize() throws Throwable { - // this could be called at the end of the run, before all of the TableDatasets have been finalized - // So we have to write out any dirty datasets. - try { - flush(); - } finally { - super.finalize(); - } - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java deleted file mode 100644 index 1c43810..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCollection.java +++ /dev/null @@ -1,200 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile; - -import org.apache.log4j.Logger; - -import java.io.IOException; -import java.lang.ref.WeakReference; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Iterator; -import java.util.Map; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to Window - - * Preferences - Java - Code Generation - Code and Comments - */ -public class TableDataSetCollection { - - private static Logger logger = Logger.getLogger(TableDataSetCollection.class); - - private HashMap readTableDataSets = new HashMap(); - private ArrayList cacheOfCurrentlyUsedIndices = new ArrayList(); - private TableDataReader myReader = null; - private TableDataWriter myWriter = null; - - public TableDataSetCollection(TableDataReader reader, TableDataWriter writer) { - setMyReader(reader); - setMyWriter(writer); - } - - public synchronized TableDataSetIndex getTableDataSetIndex(String tableName, String[] stringKeyColumnNames, String[] intKeyColumnNames) { - TableDataSetIndex theIndex = null; - TableDataSet theTableDataSet = getTableDataSet(tableName); - Iterator it = cacheOfCurrentlyUsedIndices.iterator(); - while (it.hasNext() && theIndex == null) { - WeakReference r = (WeakReference) it.next(); - TableDataSetIndex anIndex = (TableDataSetIndex) r.get(); - if (anIndex != null) { - if (anIndex.getTableName().equals(tableName)) { - theIndex = anIndex; // assume they match then prove - // otherwise - if (anIndex.getStringColumnNumbers().length != stringKeyColumnNames.length - || anIndex.getIntColumnNumbers().length != intKeyColumnNames.length) { - theIndex = null; - } else { - for (int j = 0; j < stringKeyColumnNames.length; j++) { - int column = theTableDataSet.getColumnPosition(stringKeyColumnNames[j]); - if (column != anIndex.getStringColumnNumbers()[j]) { - theIndex = null; - } - } - for (int j = 0; j < intKeyColumnNames.length; j++) { - int column = theTableDataSet.getColumnPosition(intKeyColumnNames[j]); - if (column != anIndex.getIntColumnNumbers()[j]) { - theIndex = null; - } - } - } - - } - } else { - // null weak reference - it.remove(); - } - } - if (theIndex == null) { - theIndex = new TableDataSetIndex(this, tableName); - theIndex.setIndexColumns(stringKeyColumnNames, intKeyColumnNames); - cacheOfCurrentlyUsedIndices.add(new WeakReference(theIndex)); - } - return theIndex; - } - - /** - * @param name - * @return the TableDataSet requested - */ - public synchronized TableDataSet getTableDataSet(String name) { - TableDataSet theTable = (TableDataSet) readTableDataSets.get(name); - if (theTable == null) { - try { - logger.info("reading table "+name); - theTable = getMyReader().readTable(name); - } catch (IOException e) { - e.printStackTrace(); - } - if (theTable == null) { - logger.fatal("Can't read in table " + name); - throw new RuntimeException("Can't read in table " + name); - } - readTableDataSets.put(name, theTable); - } - return theTable; - } - - public synchronized void flushAndForget(TableDataSet me) { - if (me.isDirty()) { - try { - logger.info("writing table "+me.getName() + ". Table has " + me.getRowCount() + " rows"); - getMyWriter().writeTable(me, me.getName()); - } catch (IOException e1) { - e1.printStackTrace(); - throw new RuntimeException("Can't write out table " + me.getName()); - } - - } - readTableDataSets.remove(me.getName()); - } - - /* - * (non-Javadoc) - * - * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#flush() - */ - public synchronized void flush() { - Iterator it = readTableDataSets.entrySet().iterator(); - while (it.hasNext()) { - Map.Entry e = (Map.Entry) it.next(); - TableDataSet t = (TableDataSet) e.getValue(); - if (t.isDirty()) { - try { - logger.info("writing table "+t.getName() + ". Table has " + t.getRowCount() + " rows"); - getMyWriter().writeTable(t, (String) e.getKey()); - } catch (IOException e1) { - e1.printStackTrace(); - throw new RuntimeException("Can't write out table " + e.getKey()); - } - } - } - getMyWriter().close(); - - } - - /* - * Call flush first if any changes need to be written out - * - * @see com.hbaspecto.calibrator.ModelInputsAndOutputs#invalidate() - */ - public synchronized void invalidate() throws IOException { - readTableDataSets.clear(); -// cacheOfCurrentlyUsedIndices.clear(); - Iterator it = cacheOfCurrentlyUsedIndices.iterator(); - while (it.hasNext()) { - TableDataSetIndex x = (TableDataSetIndex) ((WeakReference) it.next()).get(); - if (x!=null) x.tableDataSetShouldBeReloaded(); - else it.remove(); - } - } - - synchronized void setMyReader(TableDataReader myReader) { - this.myReader = myReader; - } - - TableDataReader getMyReader() { - return myReader; - } - - synchronized void setMyWriter(TableDataWriter myWriter) { - this.myWriter = myWriter; - } - - TableDataWriter getMyWriter() { - return myWriter; - } - - /** - * @param landInventoryTable - */ - public synchronized void addTableDataSet(TableDataSet aTable) { - readTableDataSets.put(aTable.getName(),aTable); - aTable.setDirty(true); - } - - /** - * - */ - public synchronized void close() { - myReader.close(); - myWriter.close(); - - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java deleted file mode 100644 index 8fbdee9..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/TableDataSetCrosstabber.java +++ /dev/null @@ -1,407 +0,0 @@ -/* - * Created on 13-Oct-2005 - * - * Copyright 2005 JE Abraham and others - * - * 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. - * - */ -package org.sandag.cvm.common.datafile; - -import java.io.File; -import java.util.Iterator; -import java.util.TreeSet; - -public class TableDataSetCrosstabber { - - - - public TableDataSetCrosstabber() { - super(); - } - - public static void main(String args[]) { - if (args.length!=5) { - System.out.println("usage: java org.sandag.cvm.common.datafile.TableDataSetCrosstabber directory table rows columns, values"); - System.exit(1); - } - CSVFileReader myReader= new CSVFileReader(); - myReader.setMyDirectory(args[0]); - CSVFileWriter myWriter = new CSVFileWriter(); - myWriter.setMyDecimalFormat(new GeneralDecimalFormat("0.#########E0",10000000,.001)); - myWriter.setMyDirectory(new File(args[0])); - TableDataSetCollection myCollection = new TableDataSetCollection(myReader,myWriter); - TableDataSet results = crossTabDataset(myCollection,args[1],args[2],args[3],args[4]); - results.setName(args[1]+"_"+args[2]+args[3]+"_"+args[4]+"_crossTab"); - myCollection.addTableDataSet(results); - myCollection.flush(); - } - -// this will be the new way to avoid code duplication -// public static TableDataSet crossTabDataset(TableDataSetCollection aCollection, String inputTableName, String rowColumnName, String columnColumnName, String valuesColumnName) { -// String[] rowNames = new String[1]; -// rowNames[0] = rowColumnName; -// return crossTabDataset(aCollection, inputTableName, rowNames, columnColumnName, valuesColumnName); -// } - -// this is the method that should be able to handle multiple row headers - // TODO test this - public static TableDataSet crossTabDataset(TableDataSetCollection aCollection, String inputTableName, String[] rowColumnNames, String columnColumnName, String valuesColumnName) { - // preliminaries - TableDataSet inputTable = aCollection.getTableDataSet(inputTableName); - int[] columnTypes = inputTable.getColumnType(); - TableDataSet out = new TableDataSet(); - - // set up rows - TreeSet[] rowHeaders = new TreeSet[rowColumnNames.length]; - int totalOutputRows = 1; - for (int rowColumnId=0;rowColumnId < rowColumnNames.length;rowColumnId++) { - int forRows = inputTable.checkColumnPosition(rowColumnNames[rowColumnId]); - rowHeaders[rowColumnId] = new TreeSet(); - if (columnTypes[forRows-1]==TableDataSet.STRING) { - String[] rowColumn = inputTable.getColumnAsString(forRows); - for (int i=0;ivalueMode controls behaviour when multiple records match the indices - */ - private int valueMode=SINGLE_VALUE_MODE; - /** - * With SUM_MODE the value returned is the sum of all the values - * that match the indices. When a value is put it is divided by the number of - * records that match the indices - */ - public static final int SUM_MODE=1; - /** - * With AVERAGE_MODE the value returned is the average of all the values - * that match the indices. When a value is put the value is put into all of the records - * that match the indices - */ - public static final int AVERAGE_MODE=2; - /** - * With SINGLE_VALUE_MODE a runtime exception is thrown if there is - * more than one record in the table that matches the indices - */ - public static final int SINGLE_VALUE_MODE=3; - - private boolean errorOnMissingValues = false; - - public boolean isErrorOnMissingValues() { - return errorOnMissingValues; - } - - public void setErrorOnMissingValues(boolean errorOnMissingValues) { - this.errorOnMissingValues = errorOnMissingValues; - } - - public TableDataSetIndexedValue( - String tableName, - String[] stringKeyNames, - String[] intKeyNames, - String[][] stringIndexValues, - int[][] intIndexValues, - String columnName) { - - stringKeyNameValues = new String[1+stringIndexValues.length][]; - stringKeyNameValues[0] = stringKeyNames; - for (int i=0;i1 && valueMode == SINGLE_VALUE_MODE) { - throw new MultipleValueException("Multiple matching index values in SINGLE_VALUE_MODE "+this); - } - if (lastRowNumbers.length<1 && isErrorOnMissingValues()) { - throw new MissingValueException("No matching index values for "+this); - } - float sum=0; - float denominator= 0; - for (int r=0;r1 && valueMode == SINGLE_VALUE_MODE) { - throw new RuntimeException("Multiple matching index values in SINGLE_VALUE_MODE "+this); - } - float value= compositeValue; - if (valueMode == SUM_MODE) { - value = compositeValue/lastRowNumbers.length; - } - for (int r=0;r0) { - intKeyValues[r-1][i] = Integer.valueOf(intKeyNameValues[r][i]).intValue(); - } else { - intKeyValues[r-1][i]=0; - } - } catch (java.lang.NumberFormatException e) { - lastExceptionFound = e;; - } - - } - } - } - - public void setMyFieldName(String myFieldName) { - this.myFieldName = myFieldName; - //myLastCollection = null; - //myLastIndex = null; - lastDataColumnNumber = -1; - } - - /** - * To manually set the data column number. You have to know what column number - * has the data you are interested in; if you don't use setMyFieldName instead. - * @param myDataColumnNumber - */ - public void setMyDataColumn(String dataColumnName, int dataColumnNumber) { - myFieldName = dataColumnName; - lastDataColumnNumber = dataColumnNumber; - } - - public String getMyFieldName() { - return myFieldName; - } - - - public String toString() { - StringBuffer myInfo = new StringBuffer(); - myInfo.append(getMyTableName()); - myInfo.append(" "); - myInfo.append(getMyFieldName()); - myInfo.append(" ("); - for (int s=0;s4) myInfo.append("..."+(stringKeyNameValues.length-4)+" more..."); - myInfo.append(" "); - } - for (int i=0;i4) myInfo.append("..."+(intKeyNameValues.length-4)+" more..."); - myInfo.append(" "); - } - myInfo.append(")"); - return myInfo.toString(); - } - /** - * @return Returns the intKeyNameValues. - */ - public String[][] getIntKeyNameValues() { - return intKeyNameValues; - } - - /** - * @param intKeyNameValues The intKeyNameValues to set. - */ - public void setIntKeyNameValues(String[][] intKeyNameValues) { - this.intKeyNameValues = intKeyNameValues; - updateIntKeys(); - myLastCollection= null; - myLastIndex = null; - } - - /** - * @return Returns the stringKeyNameValues. - */ - public String[][] getStringKeyNameValues() { - return stringKeyNameValues; - } - - /** - * @param stringKeyNameValues The stringKeyNameValues to set. - */ - public void setStringKeyNameValues(String[][] stringKeyNameValues) { - this.stringKeyNameValues = stringKeyNameValues; - myLastCollection= null; - myLastIndex = null; - } - - public void setValueMode(int valueMode) { - this.valueMode = valueMode; - } - - public int getValueMode() { - return valueMode; - } - - /* (non-Javadoc) - * @see org.sandag.cvm.common.datafile.TableDataSetIndex.ChangeListener#indexChanged() - */ - public void indexChanged(TableDataSetIndex r) { - lastRowNumbers=null; - } - - /** - * @param newFieldName - * @param newFieldValues - */ - public void addNewStringKey(String newFieldName, String[] newFieldValues) { - if (newFieldValues.length >0) { - String[][] oldKeyValues = stringKeyNameValues; - - // number of permutations and combinations - int numRows = (oldKeyValues.length-1)*newFieldValues.length; - stringKeyNameValues = new String[numRows+1][]; - stringKeyNameValues[0] = new String[oldKeyValues[0].length+1]; - System.arraycopy(oldKeyValues[0],0,stringKeyNameValues[0],0,oldKeyValues[0].length); - stringKeyNameValues[0][oldKeyValues[0].length] = newFieldName; - - //also need to make more int key rows - String[][] oldIntKeyNameValues = intKeyNameValues; - intKeyNameValues = new String[numRows+1][]; - intKeyNameValues[0] = oldIntKeyNameValues[0]; - - for (int originalRow=1;originalRow0) { - String[][] oldKeyValues = intKeyNameValues; - - // number of permutations and combinations - int numRows = (oldKeyValues.length-1)*newFieldValues.length; - intKeyNameValues = new String[numRows+1][]; - intKeyNameValues[0] = new String[oldKeyValues[0].length+1]; - System.arraycopy(oldKeyValues[0],0,intKeyNameValues[0],0,oldKeyValues[0].length); - intKeyNameValues[0][oldKeyValues[0].length] = newFieldName; - - //also need to make more string key rows - String[][] oldStringKeyNameValues = stringKeyNameValues; - stringKeyNameValues = new String[numRows+1][]; - stringKeyNameValues[0] = oldStringKeyNameValues[0]; - - for (int originalRow=1;originalRow1 && valueMode == SINGLE_VALUE_MODE) { - return false; - } - return true; - } catch (RuntimeException e) { - return false; - } - } - - public boolean hasValidLinks() { - return hasValidLinks(myLastCollection); - } - - /** - * @param collection - * @return String indicating the retrieval status - */ - public String retrieveValueStatusString(TableDataSetCollection collection) { - try { - updateLinks(collection); - } catch (RuntimeException e) { - return "Error updating links "+e; - } - if (lastRowNumbers.length==0) { - return "no matching rows"; - } - if (lastRowNumbers.length>1 && valueMode == SINGLE_VALUE_MODE) { - return "Multiple matching index values in SINGLE_VALUE_MODE"; - } - return String.valueOf(retrieveValue(collection)); - } - - /** - * @param e - */ - public void updateIntKeys(TableModelEvent e, String[][] newIntKeyNameValues) { - boolean updateAll = true; - if (e.getType()== e.UPDATE) { - if (e.getFirstRow()>0 && e.getLastRow()<=intKeyValues.length) { - updateAll=false; - for (int i = e.getFirstRow(); i<= e.getLastRow();i++ ) { - for (int c=0;c { - - //Holds end-of-line separator for current platform - public static String EOL; - - //Get end-of-line separator from operating system - static { - EOL = System.getProperty("line.separator"); - } - - String fileName; - - - /** - * Constructor to create a brand new text file. - * - */ - public TextFile() { - - } - - /** - * Basic constructor which reads file line by line. - * - * @param fileName name of file to open and read - */ - public TextFile(String fileName) { - this(fileName, EOL); - } - - /** - * Read a file and split by any regular expression. Default splitter - * is the platform dependent end-of-line character. - * - * @param fileName name of file to open and read - * @param splitter end-of-line separator, default is EOL for system - */ - public TextFile(String fileName, String splitter) { - super(Arrays.asList(readFrom(fileName).split(splitter))); - - //Remember file name for write() method - this.fileName = fileName; - - //Regular expression split() often leaves an empty - //String at the first position: - if (get(0).equals("")) - remove(0); - } - - /** - * Adds a string to the current TextFile object. Wrapper for underlying - * ArrayList.add() method. - * - * @param line string to add to file - */ - public void addLine(String line) { - add(line); - } - - /** - * Returns a line from the current TextFile object given a line number. Line - * numbers start at 0. Wrapper for underlying ArrayList.get() method. - * - * @param lineNumber string to add to file - */ - public String getLine(int lineNumber) { - return get(lineNumber); - } - - /** - * Updates a line in the current TextFile object given a line number. Line - * numbers start at 0. The old line is returned. Wrapper for underlying - * ArrayList.set() method. - * - * @param lineNumber string to add to file - * @param newLine represents the new line character - * @return the old line - */ - public String setLine(int lineNumber, String newLine) { - return set(lineNumber, newLine); - } - - /** - * Read a file as a single string. This method is static and can - * be called directly without creating a TextFile object. - * - * @param fileName name of the file to read from - */ - public static String readFrom(String fileName) { - StringBuilder sb = new StringBuilder(4096); - try { - BufferedReader in = new BufferedReader( - new FileReader( - new File(fileName).getAbsoluteFile())); - try { - String s; - while ((s = in.readLine()) != null) { - sb.append(s); - sb.append(EOL); - } - } finally { - in.close(); - } - } catch (IOException e) { - throw new RuntimeException(e); - } - return sb.toString(); - } - - synchronized public void write() { - writeTo(this.fileName, false); - } - - synchronized public void write(boolean append) { - writeTo(this.fileName, append); - } - - /** - * Write a single file in one method call. This method is static and can - * be called directly without creating a TextFile object. - * - * @param fileName name of file to create and write to - * @param text contents of file - */ - synchronized public static void writeTo(String fileName, String text) { - try { - PrintWriter out = new PrintWriter( - new File(fileName).getAbsoluteFile()); - try { - out.print(text); - } finally { - out.close(); - } - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - /** - * Writes the contents of a TextFile object to a specified file. The - * destination file can be opened in append mode otherwise it will be - * created if it does not exist. - * - * @param fileName name of file to write to - * @param append flag, true to open file for appending - */ - synchronized public void writeTo(String fileName, boolean append) { - try { - FileWriter fWriter = - new FileWriter(new File(fileName).getAbsoluteFile(), append); - BufferedWriter bWriter = new BufferedWriter(fWriter); - PrintWriter out = new PrintWriter(bWriter); - try { - for (String item : this) - out.println(item); - } finally { - out.close(); - } - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - synchronized public void writeTo(String fileName) { - writeTo(fileName, false); - } - - //Used for testing - public static void main(String[] args) { - - //Create a text file in memory and then write it to a file - TextFile newFile = new TextFile(); - newFile.addLine("line #1"); - newFile.addLine("line #2"); - newFile.addLine("line #3"); - newFile.addLine("line #4"); - newFile.setLine(2, "line #3 was replaced with this line"); - newFile.writeTo("testfile.txt"); - - //Example #1: Read previous file into one string - String fileContents = TextFile.readFrom("testfile.txt"); - - //Example #2: Write a string to a new file - TextFile.writeTo("copy of test.txt", fileContents); - - //Example #3: Read a file into an Arraylist based on separator - TextFile txtFile = new TextFile("testfile.txt"); - - System.out.println("number of lines="+ txtFile.size()); - - //Example #3 (cont'd): Change a line by directly accessing it - txtFile.setLine(2, "line #3 has been changed"); - - //Print out to console to check - for (String s : txtFile) { - System.out.println(s); - } - } -} -/*Output: -number of lines=10 -49 35 91 41 82 58 63 46 32 21 -68 33 20 17 43 58 49 89 21 37 -line #2 has been changed -17 30 58 86 83 42 43 50 41 18 -75 20 17 88 49 46 68 60 58 23 -61 31 36 58 42 74 42 72 71 44 -30 47 67 18 94 51 61 78 72 58 -35 84 15 97 98 20 49 61 70 63 -67 39 12 87 34 88 47 47 12 43 -70 15 87 95 77 55 76 55 93 36 -*/ diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java deleted file mode 100644 index fdc8bc1..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/BinaryFileTest.java +++ /dev/null @@ -1,105 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.File; -import java.io.IOException; - -import org.sandag.cvm.common.datafile.BinaryFileReader; -import org.sandag.cvm.common.datafile.BinaryFileWriter; -import org.sandag.cvm.common.datafile.CSVFileReader; -import org.sandag.cvm.common.datafile.TableDataSet; - - -/** - * Tests the BinaryFileReader and BinaryFile Writer classes. - * - * @author Tim Heier - * @version 1.0, 5/08/2004 - */ -public class BinaryFileTest { - - - public static void main(String[] args) { - - BinaryFileTest.testWrite(); - BinaryFileTest.testRead(); - } - - - public static void testWrite() { - TableDataSet table = null; - - //Read sample file from common-base/src/sql directory - long startTime = System.currentTimeMillis(); - try { - CSVFileReader reader = new CSVFileReader(); - table = reader.readFile(new File("src/sql/zonedata.csv")); - } catch (IOException e) { - e.printStackTrace(); - } - long stopTime = System.currentTimeMillis(); - System.out.println("Time used to read CSV file: " + (stopTime-startTime)); - - //Write the table out to a new file name - BinaryFileWriter writer = new BinaryFileWriter(); - try { - writer.writeFile(table, new File("src/sql/zonedata.bin")); - } catch (IOException e) { - e.printStackTrace(); - } - } - - - public static void testRead() { - TableDataSet table = null; - - //Read sample file from common-base/src/sql directory - long startTime = System.currentTimeMillis(); - try { - BinaryFileReader reader = new BinaryFileReader(); - table = reader.readFile(new File("src/sql/zonedata.bin")); - } catch (IOException e) { - e.printStackTrace(); - } - long stopTime = System.currentTimeMillis(); - System.out.println("Time used to read Binary file: " + (stopTime-startTime)); - - //Display some statistics about the file - System.out.println("Number of columns: " + table.getColumnCount()); - System.out.println("Number of rows: " + table.getRowCount()); - - //Display column titles - String[] labels = table.getColumnLabels(); - for (int i = 0; i < labels.length; i++) { - System.out.print( String.format(" %10s", labels[i]) ); - } - System.out.println(); - - //Print data - for (int i=1; i <= 10; i++) { - //Get a row from table - String row[] = table.getRowValuesAsString(i); - - for (int j=0; j < row.length; j++) { - System.out.print( String.format(" %10s", row[j]) ); - } - System.out.println(); - } - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java deleted file mode 100644 index cc52aed..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileReaderTest.java +++ /dev/null @@ -1,83 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.File; -import java.io.IOException; - -import org.sandag.cvm.common.datafile.CSVFileReader; -import org.sandag.cvm.common.datafile.TableDataSet; -import org.sandag.cvm.common.datafile.OLD_CSVFileReader; -import com.pb.common.util.PerformanceTimer; -import com.pb.common.util.PerformanceTimerType; - -/** - * Tests the CSVFileReader class. - * - * @author Tim Heier - * @version 1.0, 2/7/2004 - */ -public class CSVFileReaderTest { - - public static void main(String[] args) { - - CSVFileReaderTest.testRead(); - } - - public static void testRead() { - - CSVFileReader reader = new CSVFileReader(); - // Can set the delimiter set if needed - // reader.setDelimSet(" ,\t\n\r\f\""); - - PerformanceTimer timer = PerformanceTimer.createNewTimer("CsvFileTest", - PerformanceTimerType.CSV_READ); - timer.start(); - - // Read sample file from common-base/src/sql directory - TableDataSet table = null; - try { - table = reader.readFile(new File("test.csv")); - timer.stop(); - } catch (IOException e) { - e.printStackTrace(); - } - - // Display some statistics about the file - System.out.println("Number of columns: " + table.getColumnCount()); - System.out.println("Number of rows: " + table.getRowCount()); - - // Display column titles - String[] labels = table.getColumnLabels(); - for (int i = 0; i < labels.length; i++) { - System.out.print(String.format(" %10s", labels[i])); - } - System.out.println(); - - // Print data - for (int i = 1; i <= 10; i++) { - // Get a row from table - String row[] = table.getRowValuesAsString(i); - - for (int j = 0; j < row.length; j++) { - System.out.print(String.format(" %10s", row[j])); - } - System.out.println(); - } - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java deleted file mode 100644 index 4f20c1f..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/CSVFileWriterTest.java +++ /dev/null @@ -1,64 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.File; -import java.io.IOException; - -import org.sandag.cvm.common.datafile.CSVFileReader; -import org.sandag.cvm.common.datafile.CSVFileWriter; -import org.sandag.cvm.common.datafile.TableDataSet; - - -/** - * Tests the CSVFileWriter class. - * - * @author Tim Heier - * @version 1.0, 2/7/2004 - */ -public class CSVFileWriterTest { - - - public static void main(String[] args) { - - CSVFileWriterTest.testWrite(); - } - - - public static void testWrite() { - - //Read a CSV file first - TableDataSet table = null; - try { - CSVFileReader reader = new CSVFileReader(); - table = reader.readFile(new File("src/sql/zonedata.csv")); - } catch (IOException e) { - e.printStackTrace(); - } - - //Write the table out to a new file name - CSVFileWriter writer = new CSVFileWriter(); - try { - writer.writeFile(table, new File("src/sql/zonedata_new.csv")); - } catch (IOException e) { - e.printStackTrace(); - } - - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java deleted file mode 100644 index 722fb0e..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DataFileTest.java +++ /dev/null @@ -1,85 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.File; - -import java.util.Date; - -import org.sandag.cvm.common.datafile.DataFile; -import org.sandag.cvm.common.datafile.DataReader; -import org.sandag.cvm.common.datafile.DataWriter; - -/** - * Tests the records package. - * class. - * - * @author Tim Heier - * @version 1.0, 4/15/2000 - * - */ -public class DataFileTest { - - public static void main(String[] args) throws Exception { - int numRecords = 100; - String fileName = "sample.db"; - - //Delete file if it exists as part of the test - File file = new File(fileName); - - if (file.exists()) { - file.delete(); - } - - System.out.println("creating data file: sample.db"); - - DataFile dataFile = new DataFile("sample.db", 100); - - System.out.println("adding data: lastAccessTime"); - - DataWriter dw = new DataWriter("lastAccessTime"); - - dw.writeObject(new Date()); - dataFile.insertRecord(dw); - - DataReader dr = dataFile.readRecord("lastAccessTime"); - Date d = (Date) dr.readObject(); - - System.out.println("lastAccessTime = " + d.toString()); - - System.out.println("updating data: lastAccessTime"); - dw = new DataWriter("lastAccessTime"); - dw.writeObject(new Date()); - dataFile.updateRecord(dw); - - System.out.println("reading data: lastAccessTime"); - dr = dataFile.readRecord("lastAccessTime"); - d = (Date) dr.readObject(); - System.out.println("lastAccessTime = " + d.toString()); - - System.out.println("deleting data: lastAccessTime"); - dataFile.deleteRecord("lastAccessTime"); - - if (dataFile.recordExists("lastAccessTime")) { - throw new Exception("data not deleted"); - } else { - System.out.println("data successfully deleted."); - } - - System.out.println("test completed."); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java deleted file mode 100644 index 2d3585f..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/DiskObjectArrayTest.java +++ /dev/null @@ -1,193 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.IOException; - -import org.sandag.cvm.common.datafile.DiskObjectArray; -import com.pb.common.util.ObjectUtil; - -/** - * - * @author Tim Heier - * @version 1.0, 8/13/2003 - * - */ - -public class DiskObjectArrayTest { - - public static final int ARRAY_SIZE = 500000; - public static final int DATA_SIZE = 1000; - - - public static void main(String[] args) { - - DiskObjectArrayTest test = new DiskObjectArrayTest(); - test.testCreate(); - test.testAddElements(); - test.testFillArray(); - test.testUpdateArray(); - test.testAddLargeElement(); - } - - - public void testCreate() { - - //Create a large object array - DiskObjectArray ba = null; - try { - ba = new DiskObjectArray("test.array", ARRAY_SIZE, DATA_SIZE); - } - catch (IOException e) { - e.printStackTrace(); - } - - System.out.println("testCreate() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); - - ba.close(); - } - - - public void testAddElements() { - - //Create a large object array - DiskObjectArray ba = null; - try { - ba = new DiskObjectArray("test.array"); - } - catch (IOException e) { - e.printStackTrace(); - } - - //----- Add elements to array - - ba.add( 1, new Integer(1) ); - ba.add( 1, new Integer(2) ); //try adding to same location - - int[] intArray = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; - ba.add( 2, intArray); - - ba.add( 1000, new Integer(1000) ); - - //----- Read elements from array - - System.out.println("element 1 = " + ba.get( 1 )); - - intArray = (int[]) ba.get( 2 ); - for (int i=0; i < 10; i++) { - System.out.println("element 2["+i+"]" + " = " + intArray[i]); - } - - System.out.println("element 1000 = " + ba.get( 1000 )); - - System.out.println("testAddElements() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); - - ba.close(); - } - - - public void testFillArray() { - - long startTime=System.currentTimeMillis(), endTime; - - //Create a large object array - DiskObjectArray ba = null; - try { - ba = new DiskObjectArray("test.array"); - } - catch (IOException e) { - e.printStackTrace(); - } - - //----- Add a small object to each location in array - for (int i=0; i < ARRAY_SIZE; i++) { - //for (int i=0; i < 5000; i++) { - if ((i % 500) == 0) { - endTime = System.currentTimeMillis(); - System.out.println("adding="+ i + ", time="+(endTime-startTime)); - startTime = System.currentTimeMillis(); - } - ba.add( i, new Integer(i)); - } - - System.out.println("testFillArray() done. sizeOf DiskObjectArray = " + ObjectUtil.sizeOf(ba) + " bytes"); - - //Close it - ba.close(); - } - - - public void testUpdateArray() { - - long startTime=System.currentTimeMillis(), endTime; - - //Create a large object array - DiskObjectArray ba = null; - try { - ba = new DiskObjectArray("test.array"); - } - catch (IOException e) { - e.printStackTrace(); - } - - //Update an element in the middle of the array - int[] intArray = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19 }; - ba.add( 250000, intArray); - - intArray = (int[]) ba.get( 250000 ); - - System.out.println("element 250000[5]" + " = " + intArray[5]); - - System.out.println("testUpdateArray() done."); - - //Close it - ba.close(); - } - - - public void testAddLargeElement() { - - //Create a large object array - DiskObjectArray ba = null; - try { - ba = new DiskObjectArray("test.array"); - } - catch (IOException e) { - e.printStackTrace(); - } - - //----- Add elements to array - - ba.add( 1, new Integer(1) ); - ba.add( 2, new Integer(2) ); - ba.add( 3, new Integer(3) ); - - int[] intArray = new int[210]; - System.out.println("size of new element 2 = " + ObjectUtil.sizeOf(intArray)); - - ba.add( 2, intArray); - - System.out.println("Trying to read element 3..."); - System.out.println("element 3 = " + ba.get( 3 )); - - System.out.println("testAddLargeElement() done."); - - ba.close(); - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java deleted file mode 100644 index f342d72..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/JDBCTableReaderTest.java +++ /dev/null @@ -1,87 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import java.io.IOException; - -import org.sandag.cvm.common.datafile.JDBCTableReader; -import org.sandag.cvm.common.datafile.TableDataSet; -import com.pb.common.sql.JDBCConnection; - -/** - * Tests the JDBCTableReader class. - * - * @author Tim Heier - * @version 1.0, 2/7/2004 - */ -public class JDBCTableReaderTest { - - public static String HOST = "localhost"; - public static String DATABASE = "test"; - public static String USER = ""; - public static String PASSWD = ""; - public static String TABLE = "zonedata"; - public static String DRIVER = "com.mysql.jdbc.Driver"; - - - //Database url string - specific to vendor - public static String URL = "jdbc:mysql://" + HOST + "/" + DATABASE + "?user=" + USER + "&password=" + PASSWD; - - - public static void main(String[] args) { - - JDBCTableReaderTest.testLoad(); - } - - - public static void testLoad() { - JDBCConnection jdbcConn = new JDBCConnection(URL, "com.mysql.jdbc.Driver", USER, PASSWD); - - JDBCTableReader reader = new JDBCTableReader(jdbcConn); - - TableDataSet table = null; - try { - table = reader.readTable(TABLE); - } catch (IOException e) { - e.printStackTrace(); - } - jdbcConn.close(); - - //Display some statistics about the file - System.out.println("Number of columns: " + table.getColumnCount()); - System.out.println("Number of rows: " + table.getRowCount()); - - //Display column titles - String[] labels = table.getColumnLabels(); - for (int i = 0; i < labels.length; i++) { - System.out.print( String.format("%10s", labels[i]) ); - } - System.out.println(); - - //Print data - for (int i=1; i <= 10; i++) { - - //Get a row from table - String row[] = table.getRowValuesAsString(i); - - for (int j=0; j < row.length; j++) { - System.out.print( String.format(" %10s", row[j]) ); - } - System.out.println(); - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java b/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java deleted file mode 100644 index 97fe080..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/datafile/tests/TableDataSetTest.java +++ /dev/null @@ -1,322 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.datafile.tests; - -import org.sandag.cvm.common.datafile.CSVFileReader; -import org.sandag.cvm.common.datafile.CSVFileWriter; -import org.sandag.cvm.common.datafile.TableDataSet; - -import java.io.File; -import java.io.IOException; -import org.apache.log4j.Logger; - -/** - * Tests the TableDataSet class. - * - * @author Tim Heier - * @version 1.0, 2/8/2003 - */ -public class TableDataSetTest { - - protected static transient Logger logger = Logger.getLogger("org.sandag.cvm.common.datafile.test"); - - public static void main(String[] args) { - TableDataSet table = TableDataSetTest.readCSVFile(); - TableDataSetTest.testGetRowValues( table ); - TableDataSetTest.testGetValueAt( table ); - TableDataSetTest.testColmnNameRetrieval( table ); - TableDataSetTest.testSetValueAt( table ); - TableDataSetTest.testSaveFile( table ); - TableDataSetTest.testBuildIndex( table ); - TableDataSetTest.readCSVFileNoLabels(); - TableDataSetTest.readCSVFileFilteringColumns(); - //TableDataSetTest.testPerformance(); - } - - -/* TCH - Test method for my machine. - - public static void testPerformance() { - TableDataSetTest.printMemory(); - - TableDataSet[] tables = new TableDataSet[3]; - long startTime, stopTime = 0; - try { - for (int i=0; i < 3; i++) { - startTime = System.currentTimeMillis(); - tables[i] = new TableDataSet(new File("c:/temp/morpc/M5678.csv")); - stopTime = System.currentTimeMillis(); - logger.info("Finished reading " + i + ", " + (stopTime-startTime)); - TableDataSetTest.printMemory(); - } - startTime = System.currentTimeMillis(); - tables[2].saveFile(new File("c:/temp/morpc/M5678_new.csv"), 0, new DecimalFormat("#.000000")); - stopTime = System.currentTimeMillis(); - logger.info("Finished writing " + (stopTime-startTime)); - TableDataSetTest.printMemory(); - } - catch (IOException e) { - e.printStackTrace(); - return; - } - } - - public static void printMemory() { - logger.info("Total memory : " + Runtime.getRuntime().totalMemory()); - logger.info("Max memory : " + Runtime.getRuntime().maxMemory()); - logger.info("Free memory : " + Runtime.getRuntime().freeMemory()); - } -*/ - - /** - * Read a table data set. - * - * @return a fully populated TableDataSet - */ - public static TableDataSet readCSVFile() { - System.out.println("executing readCSVFile()"); - TableDataSet table = null; - - try { - CSVFileReader reader = new CSVFileReader(); - table = reader.readFile(new File("src/sql/bufcrl1.txt")); - } - catch (IOException e) { - e.printStackTrace(); - } - - return table; - } - - - /** - * Tests the getRowValues() method. - * - */ - public static void testGetRowValues(TableDataSet table) { - System.out.println("executing testGetRowValues()"); - - //Display some statistics about the file - System.out.println("Number of columns: " + table.getColumnCount()); - System.out.println("Number of rows: " + table.getRowCount()); - - //Display column titles - String[] titles = table.getColumnLabels(); - for (int i = 0; i < titles.length; i++) { - System.out.print( String.format("%10s", titles[i]) ); - } - System.out.println(); - - try { - //Print the first 10 rows - for (int i=1; i <= 10; i++) { - - //Get a row from table - float row[] = (float[]) table.getRowValues(i); - - for (int j=0; j < row.length; j++) { - System.out.print( String.format(" %9.2f", row[j]) ); - } - System.out.println(); - } - } - catch (Throwable e) { - System.out.println("Exception in testGetRowValues()"); - e.printStackTrace(); - } - } - - - /** - * Tests the getValueAt() and getValues() methods. - * - */ - public static void testGetValueAt(TableDataSet table) { - System.out.println("executing testGetValueAt()"); - - //Get the name of a column - float value1 = table.getValueAt( 1, 10 ); - float value2 = table.getValueAt( 2, 10 ); - String strValue = table.getStringValueAt( 2, 19 ); - - System.out.println( String.format("value at (1,10) =%7.2f", value1) ); - System.out.println( String.format("value at (2,10) =%7.2f", value2) ); - - System.out.println( String.format("value at (2,19) =%s", strValue) ); - System.exit(1); - - //Ask for data as a float[][]. This can be used to create a Matrix - float[][] values = null; - try { - values = table.getValues(); - - for (int i = 0; i < 10; i++) { - - for (int j = 0; j < table.getColumnCount(); j++) { - System.out.print( String.format(" %9.2f", values[i][j]) ); - } - System.out.println(); - } - } - catch (Exception e) { - System.out.println("Exception in testGetRowValues()"); - e.printStackTrace(); - } - } - - - /** - * Test the column look-up features of the table dataset. - * - */ - public static void testColmnNameRetrieval(TableDataSet table) { - System.out.println("executing testColmnNameRetrieval()"); - - //Get the name of a column - String name1 = table.getColumnLabel( 1 ); - String name2 = table.getColumnLabel( 2 ); - - System.out.println( "column 1 = " + name1 ); - System.out.println( "column 2 = " + name2 ); - - //Get the position of a column given it's name - case is ignored - int position1 = table.getColumnPosition("zone"); - int position2 = table.getColumnPosition("totpop"); - - System.out.println( "position of zone = " + position1 ); - System.out.println( "position of totpop = " + position2 ); - - String hhInc = table.getStringValueAt(1, "totpop"); - System.out.println("totpop on row 1 = " + hhInc); - } - - - /** - * Test the column look-up features of the table dataset. - * - */ - public static void testSetValueAt(TableDataSet table) { - System.out.println("executing testSetValueAt()"); - - float value1 = table.getValueAt( 1, 10 ); - System.out.println( String.format("value before update (1,10) =%7.2f", value1) ); - - table.setValueAt(1, 10, (float)300.1); - - System.out.println( String.format("value after update (1,10) =%7.2f", table.getValueAt( 1, 10 )) ); - - } - - - /** - * Test the indexing feature of the table dataset. - * - */ - public static void testSaveFile(TableDataSet table) { - System.out.println("executing testSaveFile()"); - - try { - CSVFileWriter writer = new CSVFileWriter(); - writer.writeFile(table, new File("testtable_new.csv")); - } - catch (IOException e) { - e.printStackTrace(); - } - } - - - /** - * Test the indexing feature of the table dataset. - * - */ - public static void testBuildIndex(TableDataSet table) { - System.out.println("executing testBuildIndex()"); - - //Build an index on column 1 - table.buildIndex(1); - float value1 = table.getIndexedValueAt( 25, 1 ); - - System.out.println( String.format("looking up indexed value=25 in column=1: %7.2f", value1) ); - } - - - /** - * Read a table data set. - */ - public static void readCSVFileNoLabels() { - System.out.println("executing readCSVFileNoLabels()"); - - TableDataSet table = null; - - try { - CSVFileReader reader = new CSVFileReader(); - table = reader.readFile(new File("src/sql/zonedata_nolabels.csv"), false); - } - catch (IOException e) { - e.printStackTrace(); - System.exit(1); - } - - //Display column titles - String[] titles = table.getColumnLabels(); - for (int i = 0; i < titles.length; i++) { - System.out.print( String.format("%10s", titles[i]) ); - } - System.out.println(); - - } - - - /** - * Read a table data set. - */ - public static void readCSVFileFilteringColumns() { - System.out.println("executing readCSVFileFilteringColumns()"); - TableDataSet table = null; - - String[] columnsToRead = { "tothh", "hhinc1" }; - - try { - CSVFileReader reader = new CSVFileReader(); - table = reader.readFile(new File("src/sql/zonedata.csv"), columnsToRead); - - //Display column titles - String[] titles = table.getColumnLabels(); - for (int i = 0; i < titles.length; i++) { - System.out.print( String.format("%10s", titles[i]) ); - } - System.out.println(); - - //Print a couple of values - float value1 = table.getValueAt( 1, 1 ); - float value2 = table.getValueAt( 1, 2 ); - float value3 = table.getValueAt( 2, 1 ); - float value4 = table.getValueAt( 2, 2 ); - - System.out.println( String.format("value at (1,1) =%7.2f", value1) ); - System.out.println( String.format("value at (1,2) =%7.2f", value2) ); - System.out.println( String.format("value at (2,1) =%7.2f", value3) ); - System.out.println( String.format("value at (2,2) =%7.2f", value4) ); - } - catch (Exception e) { - e.printStackTrace(); - } - - } - - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java deleted file mode 100644 index 2441f6c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventDispatcher.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.sandag.cvm.common.discreteEvent; - -import java.util.NoSuchElementException; - -import org.apache.log4j.Logger; - -public class EventDispatcher { - static final Logger logger = Logger.getLogger(EventDispatcher.class); - EventQueue myQueue = new EventQueue(); - - public EventDispatcher() { - - } - - public void dispatchEvents() { - try { - while(true) { - TimedEvent nextEvent = myQueue.popNextEvent(); - nextEvent.handleEvent(this); - } - } catch (NoSuchElementException e) { - logger.info("Event queue is empty"); - } - } - - public EventQueue getMyQueue() { - return myQueue; - } - - public void setMyQueue(EventQueue myQueue) { - this.myQueue = myQueue; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java deleted file mode 100644 index 0dba10e..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/EventQueue.java +++ /dev/null @@ -1,34 +0,0 @@ -package org.sandag.cvm.common.discreteEvent; - -import java.util.NoSuchElementException; -import java.util.TreeSet; - -public class EventQueue { - - //ENHANCEMENT look at using other storage for events. - // For instance we could have 3600*24 buckets, one for - // each second in the day, and then optionally sort the events - // within the buckets. - // Then we could use an array or linked list for each bucket? - TreeSet futureEvents; - - public EventQueue() { - futureEvents = new TreeSet (); - } - - public void enqueue(TimedEvent e) { - futureEvents.add(e); - } - - public TimedEvent popNextEvent() throws NoSuchElementException { - TimedEvent nextEvent = futureEvents.first(); - futureEvents.remove(nextEvent); - return nextEvent; - - } - - public int size() { - return futureEvents.size(); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java b/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java deleted file mode 100644 index 083db80..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/discreteEvent/TestEventDispatcher.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.sandag.cvm.common.discreteEvent; - -import org.junit.BeforeClass; -import org.junit.Test; - -public class TestEventDispatcher { - - class RequeueEvent extends TimedEvent { - - public RequeueEvent(double myTime) { - super(myTime); - } - - @Override - public void handleEvent(EventDispatcher dispatch) { - numOfEvents++; - if (numOfEvents + averageQueueSize < numberOfEventsToSimulate) { - double selector = (int) (Math.random()*3); - // queue up 0, 1 or 2 new events. - for (int i=0;i ((TimedEvent)arg0).myTime) { - return 2; - } - if (myTime < ((TimedEvent)arg0).myTime) { - return -2; - } - if (arg0==this) return 0; - // ok, our times are identical! - if (myRandomNumber ==0) myRandomNumber = (Math.random()+.001); - TimedEvent other = (TimedEvent) arg0; - if (other.myRandomNumber == 0) other.myRandomNumber = (Math.random()+.001); - if (myRandomNumber > other.myRandomNumber) return 1; - if (myRandomNumber < other.myRandomNumber) return -1; - //TODO maybe do something smarter? - logger.error("randomly generated event sortings are the same"); - return 0; - } - - @Override - public String toString() { - return "Event at "+myTime; - } - - public boolean isProcessEvenIfAfterSimulationTime() { - return processEvenIfAfterSimulationTime; - } - - public void setProcessEvenIfAfterSimulationTime( - boolean processEvenIfAfterSimulationTime) { - this.processEvenIfAfterSimulationTime = processEvenIfAfterSimulationTime; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java deleted file mode 100644 index 877efca..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/Emme2MatrixHashtableReader.java +++ /dev/null @@ -1,65 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - -package org.sandag.cvm.common.emme2; - -import java.io.File; -import java.util.*; - -import com.pb.common.matrix.*; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class Emme2MatrixHashtableReader extends Emme2MatrixReader { - - private Hashtable readMatrices = new Hashtable(); - - /** Reads an entire matrix from an Emme2 databank, but if it's already been - * read once before returns the previously read one hashtable - * - * @param index the short name of the matrix, eg. "mf10" - * @return a complete matrix - * @throws MatrixException - */ - public Matrix readMatrix(String index) throws MatrixException { - - if (readMatrices.containsKey(index)) { - return (Matrix) readMatrices.get(index); - } - Matrix m = super.readMatrix(index ); - readMatrices.put(index,m); - return m; - - } - - - - /** - * Constructor for Emme2MatrixHashtableReader. - * @param file - */ - public Emme2MatrixHashtableReader(File file) { - super(file); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java deleted file mode 100644 index 6d05feb..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/emme2/IndexConditionFunction.java +++ /dev/null @@ -1,124 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - -package org.sandag.cvm.common.emme2; - -import com.pb.common.matrix.*; - -import java.util.*; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class IndexConditionFunction { - - public void readMatrices(MatrixCacheReader mr) { - coefficients = new double[coefficientValues.size()]; - matrices = new Matrix[matrixNames.size()]; -// conditions = new int[conditionValues.size()]; - for (int i=0;i - * MSValue * MFValue - *
    Where MSValue is constant and MFValue is the value from the MF matrix - * @param matrix the MF matrix where the value is looked up based on the indices - * @param coeffMSMatrix the MS matrix where the coefficient value comes from - */ - public void addCoefficient(String matrix, String coeffMSMatrix) { - matrixNames.add(matrix); - coefficientValues.add(coeffMSMatrix); - coefficients = null; - matrices = null; - } - - /** - * Method addCoefficient, adds a term into the utility function
    - * coeffValue * MFValue - * @param matrix the emme/2 name of the matrix to be used (if - * blank or "none" the coefficient will be taken as part of the constant term) - * @param coeffValue value of the coefficient to be multiplied by the emme/2 matrix term - */ - public void addCoefficient(String matrix, double coeffValue) { - if (matrix.length() ==0 || matrix.equalsIgnoreCase("none")) addConstant(coeffValue); - else { - matrixNames.add(matrix); - coefficientValues.add(new Double(coeffValue)); - coefficients = null; - matrices = null; - } - } - - /** Changes the value of a specific coefficient (normally use addCoefficient instead) - * @param coeffIndex the previously added coefficient to change the value of - * @param matrix the matrix that is multiplied by the coefficient value - * @param coeffValue the coefficient value - */ - void setCoefficient(int coeffIndex,String matrix,double coeffValue) { - coefficientValues.set(coeffIndex,new Double(coeffValue)); - matrixNames.set(coeffIndex,matrix); - coefficients = null; - matrices = null; - } - - /** - * Reads in the emme2matrices into memory, also initializes some internal data. Call this method after the - * terms have been added to the utility function but before the utility function is evaluated. - * @param matrixReader the matrix reader to use to read in the matrices from the emme2 databank - */ - public void readMatrices(MatrixCacheReader matrixReader) { - coefficients = new double[coefficientValues.size()]; - matrices = new Matrix[matrixNames.size()]; - for (int i=0;i=4.712999999 || m.getValueAt(1619,1610)<=4.7130000001); - assertTrue("name should be mf22 but is "+m.getName(),m.getName().equals("mf22")); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java deleted file mode 100644 index 45d4549..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/AggregateAlternative.java +++ /dev/null @@ -1,26 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 John Abraham jabraham@ucalgary.ca and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public interface AggregateAlternative extends Alternative { - void setAggregateQuantity(double amount, double derivative); -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java deleted file mode 100644 index 3d6712b..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/Alternative.java +++ /dev/null @@ -1,26 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public interface Alternative { - double getUtility(); -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java deleted file mode 100644 index 3f36357..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/ChoiceModelOverflowException.java +++ /dev/null @@ -1,20 +0,0 @@ -package org.sandag.cvm.common.model; - -public class ChoiceModelOverflowException extends RuntimeException { - - public ChoiceModelOverflowException() { - } - - public ChoiceModelOverflowException(String message) { - super(message); - } - - public ChoiceModelOverflowException(Throwable cause) { - super(cause); - } - - public ChoiceModelOverflowException(String message, Throwable cause) { - super(message, cause); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java deleted file mode 100644 index ddd447c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModel.java +++ /dev/null @@ -1,63 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - - -package org.sandag.cvm.common.model; - -import java.util.Random; - -public abstract class DiscreteChoiceModel implements DiscreteChoiceModelInterface { - /** Picks one of the alternatives based on the logit model probabilities - * @throws ChoiceModelOverflowException */ - public abstract Alternative monteCarloChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException ; - - /** Picks one of the alternatives based on the logit model probabilities and random number given*/ - public abstract Alternative monteCarloChoice(double r) throws NoAlternativeAvailable ; - - public Alternative monteCarloElementalChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException { - Alternative a = monteCarloChoice(); - while (a instanceof DiscreteChoiceModel) { - a = ((DiscreteChoiceModel) a).monteCarloChoice(); - } - return a; - } - /** Use this method if you want to give a random number */ - public Alternative monteCarloElementalChoice(double r) throws NoAlternativeAvailable { - Alternative a = monteCarloChoice(r ); - Random newRandom = new Random(new Double(r*1000).longValue()); - while (a instanceof DiscreteChoiceModel) { - a = ((DiscreteChoiceModel) a).monteCarloChoice(newRandom.nextDouble()); - } - return a; - } - - /** @param a the alternative to add into the choice set */ - public abstract void addAlternative(Alternative a); - - public abstract Alternative alternativeAt(int i); - - public abstract double[] getChoiceProbabilities(); - - abstract public void allocateQuantity(double amount); - - -} - diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java deleted file mode 100644 index 24c59ed..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/DiscreteChoiceModelInterface.java +++ /dev/null @@ -1,47 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - -package org.sandag.cvm.common.model; - -/** - * @author jabraham - * - * To change the template for this generated type comment go to - * Window - Preferences - Java - Code Generation - Code and Comments - */ -public interface DiscreteChoiceModelInterface { - /** Picks one of the alternatives based on the logit model probabilities - * @throws ChoiceModelOverflowException */ - public abstract Alternative monteCarloChoice() - throws NoAlternativeAvailable, ChoiceModelOverflowException; - /** Picks one of the alternatives based on the logit model probabilities and random number given*/ - public abstract Alternative monteCarloChoice(double r) - throws NoAlternativeAvailable; - public abstract Alternative monteCarloElementalChoice() - throws NoAlternativeAvailable, ChoiceModelOverflowException; - /** Use this method if you want to give a random number */ - public abstract Alternative monteCarloElementalChoice(double r) - throws NoAlternativeAvailable; - /** @param a the alternative to add into the choice set */ - public abstract void addAlternative(Alternative a); - public abstract Alternative alternativeAt(int i); - public abstract double[] getChoiceProbabilities(); - public abstract void allocateQuantity(double amount); -} \ No newline at end of file diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java deleted file mode 100644 index b772896..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/FixedUtilityAlternative.java +++ /dev/null @@ -1,37 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public class FixedUtilityAlternative implements Alternative { - public FixedUtilityAlternative(double utilityValue) { - this.utilityValue=utilityValue; - } - - public double getUtility() {return utilityValue;} - - public double getUtilityValue(){ return utilityValue; } - - public void setUtilityValue(double utilityValue){ this.utilityValue = utilityValue; } - - private double utilityValue; - public String toString() {return "FixedUtility - "+utilityValue;}; -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java deleted file mode 100644 index eb38dfa..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/GumbelErrorTerm.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * - */ -package org.sandag.cvm.common.model; - -import java.util.Random; - -/** - * This class is used to sample and store Weibull distributed values - * @author John Abraham - * - */ -public class GumbelErrorTerm extends RandomVariable { - - static final double eulersConstant = 0.5772156649015328606; - - /** - * Static method to sample and return - * @param dispersionParameter - * @return a sample value from the Gumbel distribution with mean 0. - */ - public static double sample(double dispersionParameter) { - double sample = Math.random(); - return eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); - } - - /** - * Static method to sample and return - * @param dispersionParameter - * @param r The random number generator to use - * @return a sample value from the Gumbel distribution with mean 0. - */ - public static double sample(double dispersionParameter, Random r) { - double sample = r.nextDouble(); - return eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); - } - - private double dispersionParameter; - - /** - * Constructor - */ - public GumbelErrorTerm(double dispersionParameter) { - value = 0; - this.dispersionParameter = dispersionParameter; - } - - @Override - public Object clone() throws CloneNotSupportedException { - return super.clone(); - } - - public double getDispersionParameter() { - return dispersionParameter; - } - - /** (non-Javadoc) - * @see org.sandag.cvm.common.model.RandomVariable#sample() - * Changes the value to a new random term by sampling from the Probability - * Density Function z*exp(-z)/beta where z=exp(-(x)/beta)) - */ - @Override - public double sample() { - double sample = Math.random(); - value = eulersConstant-dispersionParameter*Math.log(-Math.log(sample)); - return value; - } - - public static double transformUniformToGumble(double uniform, double dispersionParameter) { - return eulersConstant - dispersionParameter*Math.log(-Math.log(uniform)); - } - - public void setDispersionParameter(double dispersionParameter) { - this.dispersionParameter = dispersionParameter; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java deleted file mode 100644 index 9091d4c..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/LinearInParametersFunction.java +++ /dev/null @@ -1,72 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - -package org.sandag.cvm.common.model; - -/** - * @author John Abraham - * - * A cool class created by John Abraham (c) 2003 - */ -public class LinearInParametersFunction { - - /** - * Constructor for LinearInParametersFunction. - */ - public LinearInParametersFunction() { - new LinearInParametersFunction(0); - } - - public LinearInParametersFunction(int size) { - coefficients = new double[size]; - } - - private double[] coefficients; - - public void addCoefficient(double coeffValue) { - double[] oldCoefficients = coefficients; - coefficients = new double[oldCoefficients.length+1]; - System.arraycopy(oldCoefficients,0,coefficients,0,oldCoefficients.length); - coefficients[coefficients.length-1]=coeffValue; - } - - public void setCoefficient(int coeffIndex,double coeffValue) { - if (coefficients.length <= coeffIndex) { - double[] oldCoefficients = coefficients; - coefficients = new double[coeffIndex+1]; - System.arraycopy(oldCoefficients,0,coefficients,0,oldCoefficients.length); - } - coefficients[coeffIndex]=coeffValue; - } - - public double getCoefficient(int coeffIndex) { - return coefficients[coeffIndex]; - } - - public double calcProduct(double[] values) { - double value = 0; - for (int i=0; i(); - dispersionParameter = 1.0; - } - //use this constructor if you know how many alternatives - public LogitModel(int numberOfAlternatives) { - alternatives = new ArrayList(numberOfAlternatives); - dispersionParameter = 1.0; - } - - - /** @return the composite utility (log sum value) of all the alternatives */ - public double getUtility() { - double sum = 0; - int i = 0; - while (i it = alternatives.listIterator(); - int i = 0; - while (it.hasNext()) { - Alternative a = (Alternative) it.next(); - double utility = a.getUtility(); - weights[i] = Math.exp(dispersionParameter * utility); - if (Double.isNaN(weights[i])) { - System.out.println("hmm, alternative "+a+" was such that LogitModel weight was NaN"); - System.out.println("dispersionParameter ="+dispersionParameter+", utility ="+utility); - throw new Error("NAN in weight for alternative "+a); - } - sum += weights[i]; - i++; - } - if (sum!=0) { - for (i = 0; i < weights.length; i++) { - weights[i] /= sum; - } - } - return weights; - } - } - - public Alternative alternativeAt(int i) { return (Alternative) alternatives.get(i);}// should throw an error if out of range - - - /** Picks one of the alternatives based on the logit model probabilities - * @throws ChoiceModelOverflowException */ - public Alternative monteCarloChoice() throws NoAlternativeAvailable, ChoiceModelOverflowException { - // synchronized(alternatives) { - double[] weights = new double[alternatives.size()]; - double sum = 0; - Iterator it = alternatives.listIterator(); - int i = 0; - while (it.hasNext()) { - Alternative a = it.next(); - double utility = a.getUtility(); - weights[i] = Math.exp(dispersionParameter * utility); - if (Double.isNaN(weights[i])) { - System.out.println("hmm, alternative "+a+" was such that LogitModel weight was NaN"); - System.out.println("dispersionParameter ="+dispersionParameter+", utility ="+utility); - } - sum += weights[i]; - i++; - } - if (Double.isInfinite(sum)) { - logger .fatal("Overflow error in choice model, list of alternatives follows"); - it = alternatives.listIterator(); - while (it.hasNext()) { - Alternative a = (Alternative) it.next(); - double utility = a.getUtility(); - System.out.println(" U:"+utility+", W:"+Math.exp(dispersionParameter * utility)+" for "+a); - } - - throw new ChoiceModelOverflowException("Infinite weight(s) in logit model choice function"); - } - if (sum==0) throw new NoAlternativeAvailable(); - double selector = myRandom.nextDouble() * sum; - sum = 0; - for (i = 0; i < weights.length; i++) { - sum += weights[i]; - if (selector <= sum) return (Alternative)alternatives.get(i); - } - //yikes! - System.out.println("Error: problem with logit model. sum is "+sum+", rand is "+selector); - System.out.println("Alternative,weight"); - for (i=0; i < weights.length; i++){ - System.out.println((Alternative)alternatives.get(i)+","+weights[i]); - } - throw new Error("Random Number Generator in Logit Model didn't return value between 0 and 1"); - // } - } - - /** Picks one of the alternatives based on the logit model probabilities; - use this if you want to give method random number */ - public Alternative monteCarloChoice(double randomNumber) throws NoAlternativeAvailable { - synchronized(alternatives) { - double[] weights = new double[alternatives.size()]; - double sum = 0; - Iterator it = alternatives.listIterator(); - int i = 0; - while (it.hasNext()) { - double utility = ((Alternative)it.next()).getUtility(); - weights[i] = Math.exp(dispersionParameter * utility); - if (Double.isNaN(weights[i])) { - System.out.println("hmm, alternative was such that LogitModel weight was NaN"); - } - sum += weights[i]; - i++; - } - if (sum==0) throw new NoAlternativeAvailable(); - double selector = randomNumber * sum; - sum = 0; - for (i = 0; i < weights.length; i++) { - sum += weights[i]; - if (selector <= sum) return (Alternative)alternatives.get(i); - } - //yikes! - System.out.println("Error: problem with logit model. sum is "+sum+", rand is "+randomNumber); - System.out.println("Alternative,weight"); - for (i=0; i < weights.length; i++){ - System.out.println((Alternative)alternatives.get(i)+","+weights[i]); - } - throw new Error("Random Number Generator in Logit Model didn't return value between 0 and 1"); - } - } - - - private double dispersionParameter; - private double constantUtility=0; - protected ArrayList alternatives; - - public String toString() { - StringBuffer altsString = new StringBuffer(); - int alternativeCounter = 0; - if (alternatives.size() > 5) { altsString.append("LogitModel with " + alternatives.size() + "alternatives {"); } - else altsString.append("LogitModel, choice between "); - Iterator it = alternatives.iterator(); - while (it.hasNext() && alternativeCounter < 5) { - altsString.append(it.next()); - altsString.append(","); - alternativeCounter ++; - } - if (it.hasNext()) altsString.append("...}"); else altsString.append("}"); - return new String(altsString); - } - - public double getConstantUtility(){ return constantUtility; } - - public void setConstantUtility(double constantUtility){ this.constantUtility = constantUtility; } - - /** - * Method arrayCoefficientSimplifiedChoice. - * @param theCoefficients - * @param theAttributes - * @return int - */ - public static int arrayCoefficientSimplifiedChoice( - double[][] theCoefficients, - double[] theAttributes) { - - double[] utilities = new double[theCoefficients.length]; - int alt; - for (alt =0; alt < theCoefficients.length; alt++){ - utilities[alt] = 0; - for (int c=0;c getAlternativesIterator() { - return alternatives.iterator(); - } - -} - - diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java deleted file mode 100644 index 3e377ed..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NoAlternativeAvailable.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public class NoAlternativeAvailable extends Exception { - - public NoAlternativeAvailable() { - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java deleted file mode 100644 index 3962bbe..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/NumericalDerivativeSingleParameterFunction.java +++ /dev/null @@ -1,34 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public abstract class NumericalDerivativeSingleParameterFunction implements SingleParameterFunction { - double delta; - public NumericalDerivativeSingleParameterFunction(double delta) { - this.delta=delta; - } - - public double derivative(double point){ - double perturbed = evaluate(point+delta); - return (perturbed-evaluate(point))/delta; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java deleted file mode 100644 index b5d0f59..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/RandomVariable.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.sandag.cvm.common.model; - -public abstract class RandomVariable implements Cloneable { - - double value; - - private boolean validValue; - - public RandomVariable() { - super(); - value = 0; - validValue = false; - } - - @Override - public Object clone() throws CloneNotSupportedException { - return super.clone(); - } - - abstract public double sample(); - - public double sampleIfNecessaryAndSave() { - if (validValue) { - return value; - } - value = sample(); - validValue = true; - return value; - } - - public void setInvalid() { - validValue = false; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java deleted file mode 100644 index 9063198..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/SingleParameterFunction.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - -package org.sandag.cvm.common.model; - -public interface SingleParameterFunction { - double evaluate(double point); - - double derivative(double point); -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java deleted file mode 100644 index 9476ff7..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/TestLogitModel.java +++ /dev/null @@ -1,63 +0,0 @@ -/* - Travel Model Microsimulation library - Copyright (C) 2005 PbConsult, JE Abraham and others - - - 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. - -*/ - - - - -package org.sandag.cvm.common.model; - -public class TestLogitModel { - public static void main(String[] args) throws ChoiceModelOverflowException { - try { - LogitModel lm = new LogitModel(); - Alternative a = new FixedUtilityAlternative(1.0); - Alternative b = new FixedUtilityAlternative(2.0); - lm.addAlternative(a); - lm.addAlternative(b); - for (double dp=.001;dp<100000 ;dp*=2 ) - { - lm.setDispersionParameter(dp); - int acount=0; - int bcount=0; - for (int i=0;i<1000 ;i++ ) - { - if (lm.monteCarloChoice()==a) acount++; else bcount++; - } - System.out.println("DispersionParameter="+dp+" acount="+acount+" bcount"+bcount); - } - lm=new LogitModel(); - lm.addAlternative(new FixedUtilityAlternative(Math.exp(50000))); - System.out.println("Composite utility of one infinite utility alternative is "+lm.getUtility()); - lm.monteCarloChoice(); - lm=new LogitModel(); - lm.addAlternative(new FixedUtilityAlternative(Math.log(0))); - lm.addAlternative(new FixedUtilityAlternative(5.0)); - - System.out.println("Composite utility of one negative infinite utility alternative and a '5' alt is "+lm.getUtility()); - lm.monteCarloChoice(); - - lm.addAlternative(new FixedUtilityAlternative(Math.log(0))); - System.out.println("Composite utility of one negative infinite utility alternative and a '5' alt and an alternative with zero size is "+lm.getUtility()); - lm.monteCarloChoice(); - } catch (NoAlternativeAvailable e) { - System.out.println("No alternative available somewhere here..."); - } - - } -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java deleted file mode 100644 index 8d312b8..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModel.java +++ /dev/null @@ -1,64 +0,0 @@ -package org.sandag.cvm.common.model; - -import java.util.ArrayList; - -import org.apache.log4j.Logger; - - -public class UtilityMaximizingChoiceModel extends - DiscreteChoiceModel { - - ArrayList alternatives = new ArrayList(); - - private static Logger logger = Logger.getLogger(UtilityMaximizingChoiceModel.class); - - public UtilityMaximizingChoiceModel(RandomVariable myRandomVariable) { - super(); - } - - @Override - public void addAlternative(Alternative a) { - alternatives.add(a); - } - - @Override - public void allocateQuantity(double amount) { - String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - @Override - public Alternative alternativeAt(int i) { - return alternatives.get(i); - } - - @Override - public double[] getChoiceProbabilities() { - String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - - public Alternative monteCarloChoice() { - int maxAlternative = 0; - double maxUtility = Double.NEGATIVE_INFINITY; - for (int i=0;imaxUtility) { - maxUtility = utility; - maxAlternative=i; - } - } - return alternatives.get(maxAlternative); - } - - @Override - public Alternative monteCarloChoice(double r) throws NoAlternativeAvailable { - String msg = this.getClass().toString()+" can't take a random number parameter"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java b/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java deleted file mode 100644 index 59f15f1..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/model/UtilityMaximizingChoiceModelWithErrorTermVector.java +++ /dev/null @@ -1,84 +0,0 @@ -package org.sandag.cvm.common.model; - -import java.util.ArrayList; - -import org.apache.log4j.Logger; - - -public class UtilityMaximizingChoiceModelWithErrorTermVector extends - DiscreteChoiceModel { - - ArrayList alternatives = new ArrayList(); - double[] errorTerms = null; - RandomVariable myRandomVariable; - - private static Logger logger = Logger.getLogger(UtilityMaximizingChoiceModelWithErrorTermVector.class); - - public UtilityMaximizingChoiceModelWithErrorTermVector(RandomVariable myRandomVariable) { - super(); - this.myRandomVariable = myRandomVariable; - } - - @Override - public void addAlternative(Alternative a) { - alternatives.add(a); - errorTerms = null; - } - - @Override - public void allocateQuantity(double amount) { - String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - @Override - public Alternative alternativeAt(int i) { - return alternatives.get(i); - } - - @Override - public double[] getChoiceProbabilities() { - String msg = this.getClass().toString()+" can't allocate quantity amongst alternatives -- it is only for simulation"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - @Override - public Alternative monteCarloChoice() throws NoAlternativeAvailable { - if (errorTerms !=null) { - if (errorTerms.length != alternatives.size()) { - errorTerms = null; - } - } - if (errorTerms ==null) { - errorTerms = new double[alternatives.size()]; - for (int i =0;imaxUtility) { - maxUtility = utility; - maxAlternative=i; - } - } - return alternatives.get(maxAlternative); - } - - @Override - public Alternative monteCarloChoice(double r) throws NoAlternativeAvailable { - String msg = this.getClass().toString()+" can't take a random number parameter"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java deleted file mode 100644 index 7edb73f..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/HDF5MatrixReader.java +++ /dev/null @@ -1,395 +0,0 @@ -package org.sandag.cvm.common.skims; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.Enumeration; -import java.util.List; -import java.util.TreeSet; - -import javax.swing.tree.DefaultMutableTreeNode; - -import ncsa.hdf.object.CompoundDS; -import ncsa.hdf.object.FileFormat; -import ncsa.hdf.object.HObject; -import ncsa.hdf.object.h5.H5File; - -import org.apache.log4j.Logger; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixException; -import com.pb.common.matrix.MatrixReader; - - -public class HDF5MatrixReader extends MatrixReader { - - public static void main(String args[]) { - String[] skimsToGet = new String[] {"Light_Off","Light_AM","Light_Mid","Light_PM","Medium_Off", - "Medium_AM","Medium_Mid","Medium_PM","Heavy_Off","Heavy_AM","Heavy_Mid","Heavy_PM"}; - String[] nodes = new String[] {"cvm"}; - HDF5MatrixReader r = new HDF5MatrixReader(new File("/ProjectWork/CSTDM2009 105073x4/Technical/Skims and OD/skims.h5"), nodes, skimsToGet); - -// r.testMatrixFile(); - r.readMatrices(); - } - - static Logger logger = Logger.getLogger(HDF5MatrixReader.class); - - File hdf5File; - - private List nodeNames; - - private String[] initialMatrixNames; - - public HDF5MatrixReader(File hdf5File, String[] nodeNames, String[] matrixNames) { - this.nodeNames = Arrays.asList(nodeNames); - this.hdf5File = hdf5File; - this.initialMatrixNames = matrixNames; - } - - public HDF5MatrixReader(File file, String node) { - throw new RuntimeException("Not implemented yet"); - //FIXME implement read all skims from node - } - - /** - * Builds the user to sequential lookup table and the sequential to user lookup table - * @param origins - * @param destinations - * @return [0] is sequentialToUser, to lookup user zone numbers, [1] is userToSequential to lookup index. - */ - private static int[][] buildCrossLookups(int[] origins, int[] destinations) { - TreeSet zoneSet = new TreeSet(); - int mod = 0; - for (int o : origins) { - zoneSet.add(o); - if (++mod % 250000 == 0) logger.info(" Processed line "+mod+" origin "+o); - } - mod = 0; - for (int d : destinations) { - zoneSet.add(d); - if (++mod % 250000 == 0) logger.info(" Processed line "+mod+" destination "+d); - } - int maxZone = Collections.max(zoneSet); - int[][] userSequentialCrossLookup= new int[2][]; - int[] sequentialToUserLookup = new int[zoneSet.size()]; - userSequentialCrossLookup[0] = sequentialToUserLookup; - int[] userToSequentialLookup = new int[maxZone+1]; - userSequentialCrossLookup[1] = userToSequentialLookup; - int z=0; - for (int z1 : userToSequentialLookup) { - userToSequentialLookup[z++] = -1; - } - z=0; - for (int z2 : zoneSet) { - sequentialToUserLookup[z] = z2; - userToSequentialLookup[z2] = z++; - } - return userSequentialCrossLookup; - } - - - /** - * Check if the dataset contains a column with the correct name - * If it does return the index number, else return -1 - * Also if it does contain the column, select the column for retrieval - * @param hdfDataset the dataset to be checked - * @param name the name of the column to be checked and marked for retrieval - * @return the index of the column, -1 if the column was not present - */ - static int selectHDFFieldByName(CompoundDS hdfDataset, String name) { - List nameList = Arrays.asList(hdfDataset.getMemberNames()); - int index = nameList.indexOf(name); - if (index != -1) { - hdfDataset.selectMember(index); - } - return index; - } - - - static class intKeyString implements Comparable { - int myInt; - String myString; - boolean found = false; - - intKeyString(int i, String s) { - myInt = i; - myString = s; - } - - @Override - public int compareTo(Object o) { - int otherInt = ((intKeyString) o).myInt; - if (otherInt > myInt) return -1; - if (otherInt < myInt) return 1; - if (otherInt == myInt) return 0; - assert false; - return 0; - } - } - - - @Override - public Matrix[] readMatrices() throws MatrixException { - return readMatrices(initialMatrixNames); - } - - public Matrix[] readMatrices(String[] matrixNames) { - ArrayList matrixList = new ArrayList(); - FileFormat f = null; - - intKeyString[] skimIndices = new intKeyString[matrixNames.length]; - int j=0; - for (String skimName : matrixNames) { - skimIndices[j] = new intKeyString(-1, skimName); - j++; - } - - try { - f = new H5File(hdf5File.getAbsolutePath(), H5File.READ); - f.open(); - DefaultMutableTreeNode theRoot = (DefaultMutableTreeNode) f.getRootNode(); - if (theRoot == null) { - String msg= "Null root in HDF5 skim file "+hdf5File; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - - Enumeration local_enum = ((DefaultMutableTreeNode) theRoot).breadthFirstEnumeration(); - while (local_enum.hasMoreElements()) { - DefaultMutableTreeNode theNode = (DefaultMutableTreeNode) local_enum.nextElement(); - HObject theObj = (HObject) theNode.getUserObject(); - String theName = theObj.getName(); - if (nodeNames.contains(theName)){ - logger.info("Found object \""+theName+"\" in HDF5File "+hdf5File+", reading skims"); - if (!(theObj instanceof CompoundDS)) { - String msg = "object \""+theName+"\" in HDF5File "+hdf5File+" is not a compound dataset, can't read skims"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - theObj.getMetadata(); - CompoundDS skims = (CompoundDS) theObj; - assert skims.getRank() ==1 : "Skim object in HDF5 file should only be of rank 1"; - skims.setMemberSelection(false); - int originFieldIndex = selectHDFFieldByName(skims, "origin"); - assert originFieldIndex == 0 : "Origin needs to be first member in HDF skim dataset"; - int destinationFieldIndex = selectHDFFieldByName(skims, "destination"); - assert destinationFieldIndex == 1 : "Destination needs to be second member in HDF skim dataset"; - - // get origin and destination zones, get all rows from file but no content yet. - long[] selected = skims.getSelectedDims(); - logger.info("getting zone numbers from "+selected[0]+" rows of origins and destinations"); - - List odNumbers = (List) skims.read(); - assert odNumbers.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; - int[] origins = (int[]) odNumbers.get(0); - assert odNumbers.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; - int[] destinations = (int[]) odNumbers.get(1); - int[][] userSequentialCrossLookup = buildCrossLookups( - origins, destinations); - - odNumbers = null; // forget it so we can collect the memory with garbage collection - - int count = 0; - for (intKeyString identifier : skimIndices) { - int index = selectHDFFieldByName(skims,identifier.myString); - if (index >=0 ) { - // found - if (identifier.found) { - String msg = "found "+identifier.myString+" in more than one node in skim file, not sure which one to use"; - logger.fatal(msg); - throw new RuntimeException(msg); - } else { - logger.info("Reading "+identifier.myString+" from node "+theName); - identifier.myInt = index; - identifier.found = true; - count ++; - } - } else { - identifier.myInt = -1; - } - } - Arrays.sort(skimIndices); // important to sort them so we get them in the correct order below, -1 should be first; - - if (count ==0) { - logger.warn("No relevant skims in node "+theName+" skipping"); - } else { - - // allocate the storage for the arrays based on the number of skims and the number of zones - float[][][] matrixArrays = new float[count][userSequentialCrossLookup[0].length][userSequentialCrossLookup[0].length]; - - // now get content 100 rows at a time - final long HOWMANY = 250000; - long[] start = skims.getStartDims(); - selected[0] = HOWMANY; - long size = skims.getDims()[0]; - boolean verbose = false; - int startingCol = 0; - while (skimIndices[startingCol].myInt<0) startingCol++; - assert skimIndices.length-startingCol == count; - - for (long beginAt = 0; beginAt <= size; beginAt += HOWMANY) { - logger.info("Processing line "+beginAt+" from skims"); - start[0] = beginAt; - if (beginAt+HOWMANY >= size) // should be >=? - { - //verbose = true; - selected[0] = size - beginAt; - } - List skimData = (List) skims.read(); - assert skimData.size() == count+2; - assert skimData.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; - origins = (int[]) skimData.get(0); - assert skimData.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; - destinations = (int[]) skimData.get(1); - for (int r = 0; r < origins.length; r++ ) { - int originArrayIndex = userSequentialCrossLookup[1][origins[r]]; - int destinationArrayIndex = userSequentialCrossLookup[1][destinations[r]]; - if (verbose) System.out.println("processing origin "+origins[r]+","+destinations[r]+" (index "+originArrayIndex+","+destinationArrayIndex); - for (int col = 0; col + 2 < skimData.size(); col ++) { - matrixArrays[col][originArrayIndex][destinationArrayIndex] = ((float[]) skimData.get(col+2))[r]; - } - } - verbose = false; - - } - - - int[] externalZoneNumbers = new int[userSequentialCrossLookup[0].length+1]; - for(int k=1;k99999)) - matrixValue=0; - utility += coefficients[i]*matrixValue; - } - return utility; - } - if (travelConditions instanceof SomeSkims) { - lastSkims = (SomeSkims) travelConditions; - matrixIndices = new int[namesList.size()]; - for (int i=0;i99999)) - matrixValue=0; - components[i] = coefficients[i]*matrixValue; - } - return components; - } - if (travelConditions instanceof SomeSkims) { - lastSkims = (SomeSkims) travelConditions; - matrixIndices = new int[namesList.size()]; - for (int i=0;i2) { - String msg = "Matrix name "+name+" has more than 1 part"; - } - OMXMatrixReader r = new OMXMatrixReader(new File(directoryOfMatrices,split[0]+".omx")); - // if (split.length == 1) return r.readMatrix(0); - return r.readMatrix(split[1]); - } - - /* (non-Javadoc) - * @see com.pb.common.matrix.MatrixReader#readMatrix() - */ - @Override - public Matrix readMatrix() throws MatrixException { - throw new RuntimeException("Can't read OMX Matrix without specifying file_name:matrix_name"); - } - - /* (non-Javadoc) - * @see com.pb.common.matrix.MatrixReader#readMatrices() - */ - @Override - public Matrix[] readMatrices() throws MatrixException { - throw new RuntimeException("Can't read OMX Matrices without specifying the file name, this java class is to be used for an entire directory of files."); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java deleted file mode 100644 index 206e589..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/SomeSkims.java +++ /dev/null @@ -1,437 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -package org.sandag.cvm.common.skims; - -import org.sandag.cvm.common.datafile.TableDataSet; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.MatrixType; -import com.pb.common.matrix.ZipMatrixReader; - -import drasys.or.util.Array; - -import ncsa.hdf.object.h5.H5CompoundDS; -import ncsa.hdf.object.FileFormat; -import ncsa.hdf.object.HObject; -import ncsa.hdf.object.h5.H5File; - -import org.apache.log4j.Logger; - -import java.io.File; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.Enumeration; -import java.util.List; -import java.util.TreeSet; - -import javax.swing.tree.DefaultMutableTreeNode; - -/** - * A class that reads in peak auto skims and facilitates zone pair disutility calculations - * @author John Abraham, Joel Freedman - * - */ -public class SomeSkims implements TravelAttributesInterface { - protected static Logger logger = Logger.getLogger("com.pb.models.pecas"); - - private ArrayList matrixList = new ArrayList(); - public Matrix[] matrices = new Matrix[0]; - private ArrayList matrixNameList = new ArrayList(); - - String my1stPath; - String my2ndPath; - - - public SomeSkims() { - my1stPath = System.getProperty("user.dir"); - } - - public SomeSkims(String firstPath, String secondPath) { - my1stPath = firstPath; - my2ndPath =secondPath; - }; - - public Matrix getMatrix(String name) { - int place = matrixNameList.indexOf(name); - if (place >=0) return matrices[place]; - return null; - } - - public void addZipMatrix(String matrixName) { - if (matrixNameList.contains(matrixName)) { - logger.info("SomeSkims already contains matrix named "+matrixName+", not reading it in again"); - } else { - File skim = new File(my1stPath+matrixName+".zip"); - if (!skim.exists()) skim = new File(my1stPath+matrixName+".zipMatrix"); - if(!skim.exists()) skim = new File(my1stPath+matrixName+".zmx"); - if(!skim.exists()){ - skim = new File(my2ndPath+matrixName+".zip"); - if(!skim.exists()) skim = new File(my2ndPath+matrixName+".zipMatrix"); - if(!skim.exists()) skim = new File(my2ndPath+matrixName+".zmx"); - if (!skim.exists()) { - logger.fatal("Could not find "+ matrixName+".zip, .zipMatrix or .zmx in either directory"); - throw new RuntimeException("Could not find "+ matrixName+".zip, .zipMatrix or .zmx in either directory"); - } - } - matrixList.add(new ZipMatrixReader(skim).readMatrix()); - matrixNameList.add(matrixName); - matrices = (Matrix[]) matrixList.toArray(matrices); - } - - if(logger.isDebugEnabled()) logger.debug("finished reading zipmatrix of skims "+matrixName+" into memory"); - } - - public void addTableDataSetSkims(TableDataSet s, String[] fieldsToAdd, int maxZoneNumber) { - addTableDataSetSkims(s, fieldsToAdd, maxZoneNumber, "origin", "destination"); - - } - - public void addMatrixFromFile(String fileName, String matrixName) { - File f = new File(fileName); - MatrixType type = MatrixReader.determineMatrixType(f); - if (type == null) { - logger.error("Can't determine matrix type for "+fileName); - } else { - MatrixReader r = MatrixReader.createReader(type,f); - Matrix m = r.readMatrix(); - matrixNameList.add(matrixName); - matrixList.add(m); - m.setName(matrixName); - matrices = (Matrix[]) matrixList.toArray(matrices); - } - - } - - public void addMatrix(Matrix m, String name) { - matrixNameList.add(name); - matrixList.add(m); - m.setName(name); - matrices = (Matrix[]) matrixList.toArray(matrices); - } - - public void addMatrixCSVSkims(TableDataSet s, String name) { - int rows = s.getRowCount(); - int columns = s.getColumnCount()-1; - if (rows!=columns) { - logger.fatal("Trying to add CSV Matrix Skims and number of columns does not equal number of rows"); - throw new RuntimeException("Trying to add CSV Matrix Skims and number of columns does not equal number of rows"); - } - float[][] tempArray = new float[rows][columns]; - int[] userToSequentialLookup = new int[rows+1]; - // check order of rows and columns - for (int check = 1;check < s.getRowCount();check++) { - if (!(s.getColumnLabel(check+1).equals(String.valueOf((int) (s.getValueAt(check,1)))))) { - logger.fatal("CSVMatrixSkims have columns out of order (needs to be the same as rows)"); - throw new RuntimeException("CSVMatrixSkims have columns out of order (needs to be the same as rows)"); - } - } - // TODO check for missing skims when using CSV format - for (int tdsRow = 1;tdsRow <= s.getRowCount();tdsRow++) { - userToSequentialLookup[tdsRow]=(int) s.getValueAt(tdsRow,1); - for (int tdsCol=2;tdsCol<=s.getColumnCount();tdsCol++) { - tempArray[tdsRow-1][tdsCol-2]=s.getValueAt(tdsRow,tdsCol); - } - } - Matrix m = new Matrix(name,"",tempArray); - matrixNameList.add(name); - m.setExternalNumbers(userToSequentialLookup); - this.matrixList.add(m); - matrices = (Matrix[]) matrixList.toArray(matrices); - } - - /** Adds a table data set of skims into the set of skims that are available - * - * @param s the table dataset of skims. There must be a column called "origin" - * and another column called "destination" - * @param fieldsToAdd the names of the fields from which to create matrices from, all other fields - * will be ignored. - */ - public void addTableDataSetSkims(TableDataSet s, String[] fieldsToAdd, int maxZoneNumber, String originFieldName, String destinationFieldName) { - int originField = s.checkColumnPosition(originFieldName); - int destinationField = s.checkColumnPosition(destinationFieldName); - int[] userToSequentialLookup = new int[maxZoneNumber]; - int[] sequentialToUserLookup = new int[maxZoneNumber]; - for (int i =0; i0) { - matrixArrays[entry][userToSequentialLookup[origin]][userToSequentialLookup[destination]] = s.getValueAt(row,fieldIds[entry]); - } - } - } - - for (int matrixToBeAdded =0; matrixToBeAdded < fieldsToAdd.length; matrixToBeAdded++) { - if (fieldIds[matrixToBeAdded]>0) { - matrixNameList.add(fieldsToAdd[matrixToBeAdded]); - Matrix m = new Matrix(fieldsToAdd[matrixToBeAdded],"",matrixArrays[matrixToBeAdded]); - m.setExternalNumbers(externalZoneNumbers); - this.matrixList.add(m); - } - } - - matrices = (Matrix[]) matrixList.toArray(matrices); - - logger.info("Finished reading TableDataSet skims "+s+" into memory"); - } - - - static int selectHDFFieldByName(H5CompoundDS hdfDataset, String name) { - List nameList = Arrays.asList(hdfDataset.getMemberNames()); - int index = nameList.indexOf(name); - if (index == -1) { - String msg = "No field of name "+name+" in HDF5 file node"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - hdfDataset.selectMember(index); - return index; - } - - - static class intKeyString implements Comparable { - int myInt; - String myString; - - intKeyString(int i, String s) { - myInt = i; - myString = s; - } - - @Override - public int compareTo(Object o) { - int otherInt = ((intKeyString) o).myInt; - if (otherInt > myInt) return -1; - if (otherInt < myInt) return 1; - if (otherInt == myInt) return 0; - assert false; - return 0; - } - } - - /** - * Adds some skims from an HDF5 file of skims into the set of skims that are available - * @param hdf5File - * @param nodeName - * @param fieldsToAdd - * @param maxZoneNumber - * @param originFieldName - * @param destinationFieldName - */ - public void addHDF5Skims(File hdf5File, String nodeName, String[] fieldsToAdd, int maxZoneNumber, String originFieldName, String destinationFieldName) { - logger.error("HDF5 Skims have not been tested yet, test SomeSkims.addHDF5Skims before using it"); - FileFormat f = null; - try { - f = new H5File(hdf5File.getAbsolutePath(), H5File.READ); - f.open(); - DefaultMutableTreeNode theRoot = (DefaultMutableTreeNode) f.getRootNode(); - if (theRoot == null) { - String msg= "Null root in HDF5 skim file "+hdf5File; - logger.fatal(msg); - throw new RuntimeException(msg); - } - - Enumeration local_enum = ((DefaultMutableTreeNode) theRoot).breadthFirstEnumeration(); - while (local_enum.hasMoreElements()) { - DefaultMutableTreeNode theNode = (DefaultMutableTreeNode) local_enum.nextElement(); - HObject theObj = (HObject) theNode.getUserObject(); - String theName = theObj.getName(); - if (theName.equals(nodeName)){ - logger.info("Found object \""+theName+"\" in HDF5File "+hdf5File+", reading skims"); - if (!(theObj instanceof H5CompoundDS)) { - String msg = "object \""+theName+"\" in HDF5File "+hdf5File+" is not a compound dataset, can't read skims"; - logger.fatal(msg); - throw new RuntimeException(msg); - } - H5CompoundDS skims = (H5CompoundDS) theObj; - assert skims.getRank() ==1 : "Skim object in HDF5 file should only be of rank 1"; - skims.setMemberSelection(false); - int originFieldIndex = selectHDFFieldByName(skims, originFieldName); - assert originFieldIndex == 0 : "Origin needs to be first member in HDF skim dataset"; - int destinationFieldIndex = selectHDFFieldByName(skims, destinationFieldName); - assert destinationFieldIndex == 1 : "Destination needs to be second member in HDF skim dataset"; - - // get origin and destination zones, get all rows from file but no content yet. - long[] selected = skims.getSelectedDims(); - logger.info("getting zone numbers from "+selected[0]+" rows of origins and destinations"); - - List odNumbers = (List) skims.read(); - assert odNumbers.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; - int[] origins = (int[]) odNumbers.get(0); - assert odNumbers.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; - int[] destinations = (int[]) odNumbers.get(1); - int[][] userSequentialCrossLookup = buildCrossLookups( - origins, destinations); - - odNumbers = null; // forget it so we can collect the memory with garbage collection - - intKeyString[] skimIndices = new intKeyString[fieldsToAdd.length]; - int i = 0; - for (String skimName : fieldsToAdd) { - skimIndices[i] = new intKeyString(selectHDFFieldByName(skims,skimName), skimName); - i++; - } - Arrays.sort(skimIndices); // important to sort them so we get them in the correct order below. - - float[][][] matrixArrays = new float[skimIndices.length][userSequentialCrossLookup[0].length][userSequentialCrossLookup[0].length]; - - // now get content 100 rows at a time - final long HOWMANY = 100; - long[] start = skims.getStartDims(); - selected[0] = HOWMANY; - long size = skims.getDims()[0]; - for (long beginAt = 0; beginAt <= size; beginAt += HOWMANY) { - if (beginAt+HOWMANY >= size) // should be >=? - { - selected[0] = size - beginAt; - } - List skimData = (List) skims.read(); - assert skimData.get(0) instanceof int[] : "Skim origins are not integer in HDF skim dataset"; - origins = (int[]) skimData.get(0); - assert skimData.get(1) instanceof int[] : "Skim destinations are not integer in HDF skim dataset"; - destinations = (int[]) skimData.get(1); - for (int r = 0; r < origins.length; r++ ) { - int originArrayIndex = userSequentialCrossLookup[0][origins[r]]; - int destinationArrayIndex = userSequentialCrossLookup[0][destinations[r]]; - for (int col = 0; col + 2 < skimData.size(); col ++) { - matrixArrays[col][originArrayIndex][destinationArrayIndex] = ((float[]) skimData.get(col+2))[r]; - } - } - - } - - - int[] externalZoneNumbers = new int[userSequentialCrossLookup[0].length+1]; - for(int k=1;k zoneSet = new TreeSet(); - for (int o : origins) { - zoneSet.add(o); - } - for (int d : destinations) { - zoneSet.add(d); - } - int maxZone = Collections.max(zoneSet); - int[][] userSequentialCrossLookup= new int[2][]; - int[] sequentialToUserLookup = new int[zoneSet.size()]; - userSequentialCrossLookup[0] = sequentialToUserLookup; - int[] userToSequentialLookup = new int[maxZone+1]; - userSequentialCrossLookup[1] = userToSequentialLookup; - int z=0; - for (int z1 : userToSequentialLookup) { - userToSequentialLookup[z++] = -1; - } - z=0; - for (int z2 : zoneSet) { - sequentialToUserLookup[z] = z2; - userToSequentialLookup[z2] = z++; - } - return userSequentialCrossLookup; - } - - - public int getMatrixId(String string) { - - return matrixNameList.indexOf(string); - } - - - /** - * @param my1stPath The my1stPath to set. - */ - public void setMy1stPath(String my1stPath) { - this.my1stPath = my1stPath; - } - - /** - * @param my2ndPath The my2ndPath to set. - */ - public void setMy2ndPath(String my2ndPath) { - this.my2ndPath = my2ndPath; - } - -}; diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java deleted file mode 100644 index 64611ab..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TranscadMatrixCollectionReader.java +++ /dev/null @@ -1,67 +0,0 @@ -/** - * - */ -package org.sandag.cvm.common.skims; - -import java.io.File; - -import org.apache.log4j.Logger; - -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixException; -import com.pb.common.matrix.MatrixReader; -import com.pb.common.matrix.TranscadMatrixReader; - -/** - * Reads a matrix by name with two parts separated by a colon. The first part is the file - * name (without the .mtx extension), the second part is the "core" name within the file. - * - * Transcad creates matrix files, but they are three dimensional matrices, with the z dimension being - * referred to as "cores". This allows treating these files as a list of two dimensional matrices - * @author johna - * - */ -public class TranscadMatrixCollectionReader extends MatrixReader { - - protected static Logger logger = Logger.getLogger(TranscadMatrixCollectionReader.class); - - protected File directoryOfMatrices; - - /** - * - */ - public TranscadMatrixCollectionReader(File directory) { - directoryOfMatrices = directory; - } - - /* (non-Javadoc) - * @see com.pb.common.matrix.MatrixReader#readMatrix(java.lang.String) - */ - @Override - public Matrix readMatrix(String name) throws MatrixException { - String[] split = name.split(":"); - if (split.length>2) { - String msg = "Matrix name "+name+" has more than 1 part"; - } - TranscadMatrixReader r = new TranscadMatrixReader(new File(directoryOfMatrices,split[0]+".mtx")); - if (split.length == 1) return r.readMatrix(0); - return r.readMatrix(split[1]); - } - - /* (non-Javadoc) - * @see com.pb.common.matrix.MatrixReader#readMatrix() - */ - @Override - public Matrix readMatrix() throws MatrixException { - throw new RuntimeException("Can't read Transcad Matrix without specifying file_name:matrix_name"); - } - - /* (non-Javadoc) - * @see com.pb.common.matrix.MatrixReader#readMatrices() - */ - @Override - public Matrix[] readMatrices() throws MatrixException { - throw new RuntimeException("Can't read Transcad Matrices without specifying the file name, this java class is to be used for an entire directory of files."); - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java deleted file mode 100644 index 208960a..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelAttributesInterface.java +++ /dev/null @@ -1,23 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ -/* Generated by Together */ - -package org.sandag.cvm.common.skims; - -public interface TravelAttributesInterface { - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java b/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java deleted file mode 100644 index 15d787b..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/common/skims/TravelUtilityCalculatorInterface.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - * Copyright 2005 PB Consult 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. - * - */ - -package org.sandag.cvm.common.skims; - - -/** A class that represents how the preferences for travel by different modes and different times of day - * - * @author J. Abraham - */ -public interface TravelUtilityCalculatorInterface { - public double getUtility(Location origin, Location destination, TravelAttributesInterface travelConditions); - -} diff --git a/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java b/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java deleted file mode 100644 index 625e869..0000000 --- a/sandag_abm/src/main/java/org/sandag/cvm/model/patternDetail/DestinationRandomTerms.java +++ /dev/null @@ -1,121 +0,0 @@ -package org.sandag.cvm.model.patternDetail; - -import java.util.Random; - -import org.apache.log4j.Logger; - -import org.sandag.cvm.common.model.GumbelErrorTerm; - -public class DestinationRandomTerms { - - protected static Logger logger = Logger.getLogger(DestinationRandomTerms.class); - - - static final Random random = new Random(); - static final double log4pi = Math.log(4*Math.PI); - - final int normalPoolSize = 100000; - /** - * This is the pool of standard normals, used for zones with size < useGumbel - */ - double[] normalPool = null; - - final int uniformPoolSize = 100000; - /** - * This is the pool of uniform distributed numbers, used with size > useGumbel - */ - double[] uniformPool = null; - - // TODO set useGumbel to something > 1 - /** - * if n > useGumbel, we'll use the Hall approximation to the Gumbel Distribution - */ - int useGumbel = 200; - - /** - * Gets a extreme normal random variable for a zone. If zone, poolOffset, poolSkip and n - * are the same it will return the same value. - * @param zone the destination zone under consideration - * @param poolOffset something stored in Preferences to distinguish decision makers - * @param poolSkip something stored in Preferences to distinguish decision makers - * @param n how large the population is in the zone - * @param stdDev the variance of the underlying normal distribution - * @return the sample from the extreme normal distribution - */ - public double getExtremeNormal(int zone, int poolOffset, int poolSkip, int n, double stdDev) { - if (poolSkip ==0) { - logger.warn("poolSkip is zero, setting it to 1"); - poolSkip = 1; - } - checkPools(); - if (stdDev == 0) return 0; - if (n 1.9999); - } - // let's check for n=20, see if we get expected within a certain tolerance - // these are the expected histograms from another experiment, with 4777 maximum random draws - // they are 0.1 apart and the first upper bound is 0.2 - int[] expectedHistograms = new int[]{ - 0, - 0, - 1, - 0, - 5, - 13, - 26, - 38, - 74, - 91, - 155, - 217, - 254, - 327, - 355, - 350, - 370, - 394, - 330, - 330, - 305, - 229, - 205, - 163, - 116, - 114, - 97, - 61, - 47, - 31, - 17, - 22, - 15, - 3, - 10, - 2, - 3, - 4, - 1, - 0, - 0, - 0, - 0, - 0, - 2, - 0, - 0, - 0, - 0}; - int offSet = (int) (Math.random()*10000); - int skip = (int) (Math.random()*10); - for (int i=0;i<4777;i++) { - int fakeZone = (int) (Math.random()*15000); - double sample1 = sampler.getExtremeNormal(fakeZone, offSet, skip, 20, 1); - int bin = (int) ((sample1-0.1)/0.1); - if (bin <0) bin =0; - if (bin >= expectedHistograms.length) bin = expectedHistograms.length-1; - expectedHistograms[bin]--; - } - // now check to see if maximum diff < 70; - int maximumDiff = 0; - for (int i=0;i= expectedHistograms.length) bin = expectedHistograms.length-1; - expectedHistograms[bin]--; - } - // now check to see if maximum diff < 100; - maximumDiff = 0; - for (int i=0;i= expectedHistograms.length) bin = expectedHistograms.length-1; - expectedHistograms[bin]--; - } - // now check to see if maximum diff < 5; - maximumDiff = 0; - for (int i=0;i cntFlows; - private Matrix[] truckFlowsSUT; - private Matrix[] truckFlowsMUT; - private Matrix emptySUT; - private Matrix emptyMUT; - - - public SandagCountyModel(ReadFAF4 faf4, disaggregateFlows df) { - // constructor - this.faf4 = faf4; - this.df = df; - - } - - - public void runSandagCountyModel () { - // run model to disaggregate flows from FAF zones to counties - - logger.info("Model to disaggregate flows from FAF zones to counties"); - - df.getUScountyEmploymentByIndustry(utilities.getRb()); - utilities.createZoneList(); - faf4.readAllData(utilities.getRb(), utilities.getYear(), "tons"); - - faf4.definePortsOfEntry(utilities.getRb()); - if (utilities.getBooleanProperty("read.in.raw.faf.data", true)) extractTruckData(); - disaggregateFromFafToCounties(); - - convertTonsToTrucks cttt = new convertTonsToTrucks(utilities.getRb()); - cttt.readData(); - convertTonsToTrucks(cttt); - addEmptyTrucks(); - writeCountyTripTables(); - } - - - private void extractTruckData() { - // extract truck data and write flows to file - logger.info("Extracting FAF truck data"); - String[] scaleTokens = ResourceUtil.getArray(utilities.getRb(), "scaling.truck.trips.tokens"); - double[] scaleValues = ResourceUtil.getDoubleArray(utilities.getRb(), "scaling.truck.trips.values"); - HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); - String truckFileNameT = ResourceUtil.getProperty(utilities.getRb(), "processed.truck.faf.data") + "_" + - utilities.getYear(); - - // create output directory if it does not exist yet - File file = new File ("output/temp"); - if (!file.exists()) { - boolean outputDirectorySuccessfullyCreated = file.mkdir(); - if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); - } - faf4.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); - } - - - private void disaggregateFromFafToCounties() { - // disaggregates freight flows from FAF zoneArray to counties - - logger.info(" Disaggregating FAF data from FAF zones to counties for year " + utilities.getYear() + "."); - - int matrixSize = utilities.countyFips.length; - cntFlows = new HashMap<>(); - - float globalScale = (float) ResourceUtil.getDoubleProperty(utilities.getRb(), "overall.scaling.factor.truck"); - - // regular method - for (String com: ReadFAF4.sctgStringCommodities) { - float[][] dummy = new float[matrixSize][matrixSize]; - cntFlows.put(com, dummy); - } - df.prepareCountyDataForFAFwithDetailedEmployment(utilities.getRb(), utilities.getYear(), false); - df.scaleSelectedCounties(utilities.getRb()); - - java.util.concurrent.ForkJoinPool pool = new java.util.concurrent.ForkJoinPool(); - DnCRecursiveAction action = new DissaggregateFafAction(globalScale); - pool.execute(action); - action.getResult(); - } - - - private class DissaggregateFafAction extends DnCRecursiveAction { - private final float globalScale; - - private DissaggregateFafAction(float globalScale) { - super(0,ReadFAF4.sctgStringCommodities.length); - this.globalScale = globalScale; - } - - private DissaggregateFafAction(float globalScale, long start, long length, DnCRecursiveAction next) { - super(start,length,next); - this.globalScale = globalScale; - } - - @Override - protected void computeAction(long start, long length) { - long end = start + length; - for (int comm = (int) start; comm < end; comm++) { - int cm = ReadFAF4.sctgCommodities[comm]; - - String fileName = ResourceUtil.getProperty(utilities.getRb(), "processed.truck.faf.data") + "_" + utilities.getYear(); - if (cm < 10) fileName = fileName + "_SCTG0" + cm + ".csv"; - else fileName = fileName + "_SCTG" + cm + ".csv"; - logger.info(" Working on " + fileName); - String sctg = ReadFAF4.getSCTGname(cm); - float[][] values = cntFlows.get(sctg); - TableDataSet tblFlows = fafUtils.importTable(fileName); - for (int row = 1; row <= tblFlows.getRowCount(); row++) { - float shortTons = tblFlows.getValueAt(row, "shortTons"); - if (shortTons == 0) continue; - String dir = tblFlows.getStringValueAt(row, "flowDirection"); - int orig = (int) tblFlows.getValueAt(row, "originFAF"); - int dest = (int) tblFlows.getValueAt(row, "destinationFAF"); - TableDataSet singleFlow; - if (dir.startsWith("import") || dir.startsWith("export")) { - TableDataSet poe = null; - // Entry through land border - switch (dir) { - case "import": - poe = ReadFAF4.getPortsOfEntry(orig); - break; - // Entry through marine port - case "import_port": - poe = ReadFAF4.getMarinePortsOfEntry(orig); - break; - // Entry through airport - case "import_airport": - poe = ReadFAF4.getAirPortsOfEntry(orig); - break; - // Exit through land border - case "export": - poe = ReadFAF4.getPortsOfEntry(dest); - break; - // Exit through marine port - case "export_port": - poe = ReadFAF4.getMarinePortsOfEntry(dest); - break; - // Exit through airport - case "export_airport": - poe = ReadFAF4.getAirPortsOfEntry(dest); - break; - } - singleFlow = df.disaggregateSingleFAFFlowThroughPOE(dir, poe, orig, dest, sctg, shortTons, 1); - } else singleFlow = df.disaggregateSingleFAFFlow(orig, dest, sctg, shortTons, 1); - for (int i = 1; i <= singleFlow.getRowCount(); i++) { - int oFips = (int) singleFlow.getValueAt(i, "oFips"); - int oZone = utilities.countyFipsIndex[oFips]; - int dFips = (int) singleFlow.getValueAt(i, "dFips"); - int dZone = utilities.countyFipsIndex[dFips]; - float thisFlow = singleFlow.getValueAt(i, "Tons") * globalScale; - values[oZone][dZone] += thisFlow; - } - } - } - } - - @Override - protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) { - return new DissaggregateFafAction(globalScale,start,length,next); - } - - @Override - protected boolean continueDividing(long length) { - return getSurplusQueuedTaskCount() < 3 && length > 1; - } - } - - - private void convertTonsToTrucks (convertTonsToTrucks cttt) { - // convert flows in tons into flows in trucks using average payload factors - - logger.info(" Converting tons into trucks"); - - int highestGroupCode = utilities.getHighestVal(utilities.getCommodityGroupOfSCTG()); - truckFlowsSUT = new Matrix[highestGroupCode + 1]; - truckFlowsMUT = new Matrix[highestGroupCode + 1]; - float aawdtFactor = (float) ResourceUtil.getDoubleProperty(utilities.getRb(), "AADT.to.AAWDT.factor"); - for (int i = 0; i <= highestGroupCode; i++) { - truckFlowsSUT[i] = createCountyMatrix(); - truckFlowsMUT[i] = createCountyMatrix(); - } - - for (String com: ReadFAF4.sctgStringCommodities) { - int comGroup = utilities.getCommodityGroupOfSCTG()[Integer.parseInt(com.substring(4))]; - float[][] flowsThisCommodity = cntFlows.get(com); - for (int oFips: utilities.countyFips) { - for (int dFips: utilities.countyFips) { - int oZone = utilities.countyFipsIndex[oFips]; - int dZone = utilities.countyFipsIndex[dFips]; - float distance = df.getCountyDistance(oFips, dFips); - float truckByType[] = cttt.convertThisFlowFromTonsToTrucks(com, distance, flowsThisCommodity[oZone][dZone]); - float oldValueSUT = truckFlowsSUT[comGroup].getValueAt(oFips, dFips); - float newValueSut = truckByType[0] / 365.25f * aawdtFactor; - truckFlowsSUT[comGroup].setValueAt(oFips, dFips, oldValueSUT + newValueSut); - float oldValueMUT = truckFlowsMUT[comGroup].getValueAt(oFips, dFips); - float newValueMUT = (truckByType[1] + truckByType[2] + truckByType[3]) / 365.25f * aawdtFactor; - truckFlowsMUT[comGroup].setValueAt(oFips, dFips, oldValueMUT + newValueMUT); - } - } - } - } - - - private Matrix createCountyMatrix() { - Matrix mat = new Matrix(utilities.countyFips.length, utilities.countyFips.length); - mat.setExternalNumbersZeroBased(utilities.countyFips); - return mat; - } - - private void addEmptyTrucks() { - // Empty truck model to ensure balanced truck volumes entering and leaving every zone - - double emptyRate = 1f - ResourceUtil.getDoubleProperty(utilities.getRb(), "empty.truck.rate"); - - int highestGroupCode = utilities.getHighestVal(utilities.getCommodityGroupOfSCTG()); - double[] balSut = new double[utilities.countyFips.length]; - double[] balMut = new double[utilities.countyFips.length]; - Matrix loadedSutTot = createCountyMatrix(); - Matrix loadedMutTot = createCountyMatrix(); - for (int orig = 0; orig < utilities.countyFips.length; orig++) { - for (int dest = 0; dest < utilities.countyFips.length; dest++) { - for (int comGroup = 0; comGroup <= highestGroupCode; comGroup++) { - float sut = truckFlowsSUT[comGroup].getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]); - float mut = truckFlowsMUT[comGroup].getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]); - balSut[orig] -= sut; - balSut[dest] += sut; - balMut[orig] -= mut; - balMut[dest] += mut; - loadedSutTot.setValueAt(utilities.countyFips[orig], utilities.countyFips[dest], - (loadedSutTot.getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]) + sut)); - loadedMutTot.setValueAt(utilities.countyFips[orig], utilities.countyFips[dest], - (loadedMutTot.getValueAt(utilities.countyFips[orig], utilities.countyFips[dest]) + mut)); - } - } - } - Matrix emptyBalancedSut = balanceEmpties(balSut); - Matrix emptyBalancedMut = balanceEmpties(balMut); - double targetSut = loadedSutTot.getSum() / emptyRate; - double targetMut = loadedMutTot.getSum() / emptyRate; - double emptySutRetTot = emptyBalancedSut.getSum(); - double emptyMutRetTot = emptyBalancedMut.getSum(); - - logger.info(" Trucks generated by commodity flows: " + Math.round(loadedSutTot.getSum()) + " SUT and " + - Math.round(loadedMutTot.getSum()) + " MUT."); - logger.info(" Empty trucks generated by balancing: " + Math.round((float) emptySutRetTot) + " SUT and " + - Math.round((float) emptyMutRetTot) + " MUT."); - double correctedEmptyTruckRate = emptyRate + (emptySutRetTot + emptyMutRetTot) / (targetSut + targetMut); - if (correctedEmptyTruckRate < 0) logger.warn("Empty truck rate for returning trucks is with " + - utilities.rounder(((emptySutRetTot + emptyMutRetTot) / (targetSut + targetMut)), 2) + - " greater than global empty-truck rate of " + utilities.rounder(emptyRate, 2)); - logger.info(" Empty trucks added by statistics: " + Math.round((float) ((1 - correctedEmptyTruckRate) * targetSut)) + - " SUT and " + Math.round((float) ((1 - correctedEmptyTruckRate) * targetMut)) + " MUT."); - - emptySUT = createCountyMatrix(); - emptyMUT = createCountyMatrix(); - for (int origin : utilities.countyFips) { - for (int destination : utilities.countyFips) { - float emptySutReturn = emptyBalancedSut.getValueAt(destination, origin); // note: orig and dest are switched to get return trip - float emptyMutReturn = emptyBalancedMut.getValueAt(destination, origin); // note: orig and dest are switched to get return trip - double emptySutStat = (loadedSutTot.getValueAt(origin, destination) + emptySutReturn) / correctedEmptyTruckRate - - (loadedSutTot.getValueAt(origin, destination) + emptySutReturn); - double emptyMutStat = (loadedMutTot.getValueAt(origin, destination) + emptyMutReturn) / correctedEmptyTruckRate - - (loadedMutTot.getValueAt(origin, destination) + emptyMutReturn); - - emptySUT.setValueAt(origin, destination, (float) (emptySutReturn + emptySutStat)); - emptyMUT.setValueAt(origin, destination, (float) (emptyMutReturn + emptyMutStat)); - } - } - } - - - private Matrix balanceEmpties(double[] trucks) { - // generate empty truck trips - - RowVector emptyTruckDest = new RowVector(utilities.countyFips.length); - emptyTruckDest.setExternalNumbersZeroBased(utilities.countyFips); - ColumnVector emptyTruckOrig = new ColumnVector(utilities.countyFips.length); - emptyTruckOrig.setExternalNumbersZeroBased(utilities.countyFips); - for (int zn = 0; zn < utilities.countyFips.length; zn++) { - if (trucks[zn] > 0) { - emptyTruckDest.setValueAt(utilities.countyFips[zn], (float) trucks[zn]); - emptyTruckOrig.setValueAt(utilities.countyFips[zn], 0f); - } - else { - emptyTruckOrig.setValueAt(utilities.countyFips[zn], (float) trucks[zn]); - emptyTruckDest.setValueAt(utilities.countyFips[zn], 0f); - } - } - Matrix seed = createCountyMatrix(); - for (int o: utilities.countyFips) { - for (int d: utilities.countyFips) { - float friction = (float) Math.exp(-0.001 * df.getCountyDistance(o, d)); - seed.setValueAt(o, d, friction); - } - } - MatrixBalancerRM mb = new MatrixBalancerRM(seed, emptyTruckOrig, emptyTruckDest, 0.001, 10, MatrixBalancerRM.ADJUST.BOTH_USING_AVERAGE); - return mb.balance(); - } - - - private void writeCountyTripTables() { - // write out county-to-county trip tables - - logger.info(" Writing county-to-county truck trip table"); - String fileName = utilities.getRb().getString("county.to.county.trip.table") + "_" + utilities.getYear() + ".csv"; - PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); - pw.println("origFips,destFips,sut1,sut2,sut3,sut4,sut5,sut6,emptySut,mut1,mut2,mut3,mut4,mut5,mut6,emptyMut"); - for (int oFips: utilities.countyFips) { - for (int dFips: utilities.countyFips) { - pw.print(oFips+","+dFips); - for (int i = 1; i < truckFlowsSUT.length; i++) pw.print("," + truckFlowsSUT[i].getValueAt(oFips,dFips)); - pw.print("," + emptySUT.getValueAt(oFips, dFips)); - for (int i = 1; i < truckFlowsMUT.length; i++) pw.print("," + truckFlowsMUT[i].getValueAt(oFips,dFips)); - pw.println("," + emptyMUT.getValueAt(oFips, dFips)); - } - } - pw.close(); - } - - - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java b/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java deleted file mode 100644 index 0cd3513..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/applications/ohio.java +++ /dev/null @@ -1,118 +0,0 @@ -package org.sandag.htm.applications; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import org.sandag.htm.processFAF.countyTruckModel; -import org.sandag.htm.processFAF.fafUtils; -import org.sandag.htm.processFAF.readFAF3; - -import org.apache.log4j.Logger; - -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; - -/** - * Application of countyTruckModel for Ohio State - * Author: Rolf Moeckel, PB Albuquerque - * Date: June 15, 2012 (Chicago IL) - - */ -public class ohio { - - private static Logger logger = Logger.getLogger(ohio.class); - private static String[] listOfRailRegions; - private static String[] railRegionReference; - private static boolean[] relevantCommodities; - - public static void summarizeFAFData (ResourceBundle appRb, int[] countyFips) { - // Summarize commodity flows of FAF data - -// if (!ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data")) { -// logger.error("Cannot summarize data for Ohio, set \"read.in.raw.faf.data\" to true."); -// return; -// } - if (ResourceUtil.getBooleanProperty(appRb, "summarize.by.ohio.rail.zones")) { - readOhioRailRegions(appRb, countyFips); - readRelevantCommodities(appRb); - } - - } - - - private static void readOhioRailRegions (ResourceBundle appRb, int[] countyFips) { - // create reference between fips code and Ohio Rail Region - - logger.info("Reading Ohio Rail Regions"); - TableDataSet railRegions = fafUtils.importTable(appRb.getString("rail.zone.definition")); - int highestFips = fafUtils.getHighestVal(countyFips); - - railRegionReference = new String[highestFips + 1]; - for (int row = 1; row <= railRegions.getRowCount(); row++) { - int fips = (int) railRegions.getValueAt(row, "fips"); - String reg = railRegions.getStringValueAt(row, "ohioRailRegion"); - railRegionReference[fips] = reg; - } - listOfRailRegions = fafUtils.getUniqueListOfValues(railRegionReference); - } - - - private static void readRelevantCommodities (ResourceBundle appRb) { - // Read how commodities are grouped by SCTG cagegory - - TableDataSet comGroups = fafUtils.importTable(appRb.getString("commodity.grouping")); - relevantCommodities = new boolean[fafUtils.getHighestVal(readFAF3.sctgCommodities) + 1]; - for (int i = 0; i < relevantCommodities.length; i++) relevantCommodities[i] = false; - for (int row = 1; row <= comGroups.getRowCount(); row++) { - int sctg = (int) comGroups.getValueAt(row, "SCTG"); - String truckType = comGroups.getStringValueAt(row, "MainTruckType"); - relevantCommodities[sctg] = truckType.equals("Van"); // set relevantCommodities to true if truckType equals Van - } - } - - - public static void sumFlowByRailZone(ResourceBundle appRb, int year, int[] countyFips, int[] countyIndex, HashMap cntFlows) { - // summarize flows by rail regions - - // Step 1: Initialize counter - HashMap railRegionIndex = new HashMap<>(); - int regionCounter = 0; - for (String txt: listOfRailRegions) { - railRegionIndex.put(txt,regionCounter); - regionCounter++; - } - double[][] summaryRailRegions = new double[listOfRailRegions.length][listOfRailRegions.length]; - - // Step 2: Summarize flows - String[] commodities = readFAF3.sctgStringCommodities; - for (String com: commodities) { - if (!relevantCommodities[Integer.parseInt(com.substring(4))]) continue; - float[][] flows = cntFlows.get(com); - for (int oFips: countyFips) { - if (railRegionReference[oFips] != null) { - int origRailRegion = railRegionIndex.get(railRegionReference[oFips]); - for (int dFips: countyFips) { - if (railRegionReference[dFips] != null) { - int destRailRegion = railRegionIndex.get(railRegionReference[dFips]); - summaryRailRegions[origRailRegion][destRailRegion] += flows[countyIndex[oFips]][countyIndex[dFips]]; - } - } - } - } - } - - PrintWriter pw = fafUtils.openFileForSequentialWriting(appRb.getString("rail.zone.output") + "_" + year + ".csv"); - - pw.print("Region"); - for (String txt: listOfRailRegions) pw.print("," + txt); - pw.println(); - for (String orig: listOfRailRegions) { - pw.print(orig); - for (String dest: listOfRailRegions) { - pw.print("," + summaryRailRegions[railRegionIndex.get(orig)][railRegionIndex.get(dest)]); - } - pw.println(); - } - pw.close(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java b/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java deleted file mode 100644 index 1e5e07d..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/applications/sandagZonalModel.java +++ /dev/null @@ -1,338 +0,0 @@ -package org.sandag.htm.applications; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import org.sandag.htm.processFAF.disaggregateFlows; -import org.sandag.htm.processFAF.fafUtils; -import com.pb.sawdust.calculator.Function1; -import com.pb.sawdust.util.array.ArrayUtil; -import com.pb.sawdust.util.concurrent.ForkJoinPoolFactory; -import com.pb.sawdust.util.concurrent.IteratorAction; -import org.apache.log4j.Logger; - -import java.io.PrintWriter; -import java.util.*; -import java.util.concurrent.ForkJoinPool; - -/** - * SANDAG external truck model - * Class to disaggregate FAF flows from counties to SANDAG zones - * Author: Rolf Moeckel, PB Albuquerque - * Date: 6 March 2013 (Santa Fe, NM) - * Version 1.0 - */ - -public class sandagZonalModel { - - private static Logger logger = Logger.getLogger(sandagZonalModel.class); - private int[] zones; - private HashMap useHshLocal; - private HashMap makeHshLocal; - private String[] industries; - private float[][] zonalEmployment; - private TableDataSet countyFlows; - private int[] sandagExtStatID; - private boolean[] sandagExtStat_border; - private int[] sanDiegoNodes; - private final HashMap disaggregatedFlows = new HashMap<>(); - private String[] todNames; - private float[][] todSutShare; - private float[][] todMutShare; - - - public sandagZonalModel() { - // constructor - } - - - public void runSandagZonalModel () { - // run model to disaggregate flows from counties to SANDAG zones - - logger.info("Model to disaggregate flows from counties to zones"); - - readInputData(); - disaggregateFlowsFromCountiesToZones(); - writeOutDisaggregatedFlows(); - } - - - private void readInputData() { - // read input data - - logger.info(" Reading input data"); - // define TAZ and MGRA system - TableDataSet zonesTbl = fafUtils.importTable(utilities.getRb().getString("local.zones")); - zones = zonesTbl.getColumnAsInt("TAZ"); - TableDataSet mgraZones = fafUtils.importTable(utilities.getRb().getString("local.mgra.zones")); - int[] mgras = mgraZones.getColumnAsInt("mgra13"); - int[] mgraTazReference = new int[fafUtils.getHighestVal(mgras) + 1]; - for (int row = 1; row <= mgraZones.getRowCount(); row++) { - mgraTazReference[(int) mgraZones.getValueAt(row, "mgra13")] = (int) mgraZones.getValueAt(row, "taz13"); - } - - // process local make/use coefficients - String useToken = "faf.use.coefficients.local"; - String makeToken = "faf.make.coefficients.local"; - useHshLocal = disaggregateFlows.createMakeUseHashMap(utilities.getRb(), useToken); - makeHshLocal = disaggregateFlows.createMakeUseHashMap(utilities.getRb(), makeToken); - TableDataSet useCoeff = fafUtils.importTable(utilities.getRb().getString(useToken)); - industries = useCoeff.getColumnAsString("Industry"); - - // read local employment data - TableDataSet employmentMGRA = fafUtils.importTable(utilities.getRb().getString("local.employment.data") + - utilities.getYear() + ".csv"); - zonalEmployment = new float[utilities.getHighestVal(zones)+1][industries.length]; - for (int row = 1; row <= employmentMGRA.getRowCount(); row++) { - int mgra = (int) employmentMGRA.getValueAt(row, "mgra"); - int taz = mgraTazReference[mgra]; - for (int ind = 0; ind < industries.length; ind++) { - zonalEmployment[taz][ind] += employmentMGRA.getValueAt(row, industries[ind]); - } - } - - // read external station IDs - TableDataSet extStations = fafUtils.importTable(utilities.getRb().getString("external.station.definition")); - sandagExtStatID = new int[utilities.getHighestVal(extStations.getColumnAsInt("natExtStat")) + 1]; - sandagExtStat_border = new boolean[utilities.getHighestVal(zones) + 1]; - for (int row = 1; row <= extStations.getRowCount(); row++) { - sandagExtStatID[(int) extStations.getValueAt(row, "natExtStat")] = - (int) extStations.getValueAt(row, "sandagExtStat"); - sandagExtStat_border[(int) extStations.getValueAt(row, "sandagExtStat")] = - extStations.getBooleanValueAt(row, "borderCrossing"); - } - - // read flows at external stations - String fileName = utilities.getRb().getString("external.station.flows"); - countyFlows = utilities.importTableFromDBF(fileName); - if (countyFlows.getColumnCount() != 16) logger.error("Excepted 16 but found " + countyFlows.getColumnCount() + - " columns in " + fileName); - String[] expectedLabels = {"SUBAREA_NO","SUBAREA_N1","DEMAND_SUT","DEMAND_SU1","DEMAND_SU2","DEMAND_SU3","DEMAND_SU4","DEMAND_SU5","DEMAND_EMP","DEMAND_MUT","DEMAND_MU1","DEMAND_MU2","DEMAND_MU3","DEMAND_MU4","DEMAND_MU5","DEMAND_EM1"}; - String[] actualLabels = countyFlows.getColumnLabels(); - boolean wrongHeader = false; - for (int col = 0; col < countyFlows.getColumnCount(); col++) { - if (!actualLabels[col].equalsIgnoreCase(expectedLabels[col])) - wrongHeader = true; - } - if (wrongHeader) { - logger.error("File " + fileName + " has unexpected headers:"); - logger.info("Column,ExpectedLabel,ActualLabel"); - for (int col = 0; col < countyFlows.getColumnCount(); col++) - logger.info(col+1 + "," + expectedLabels[col] + "," + actualLabels[col] + "," + (expectedLabels[col].equals(actualLabels[col]))); - System.exit(1); - } - countyFlows.setColumnLabels(new String[]{"orig","dest","sut1","sut2","sut3","sut4","sut5","sut6","sutEmpty", - "mut1","mut2","mut3","mut4","mut5","mut6","mutEmpty"}); - - sanDiegoNodes = ResourceUtil.getIntegerArray(utilities.getRb(), "internal.nodes.san.diego"); - - // read time-of-day shares - TableDataSet TODValuesGeneral = fafUtils.importTable(utilities.getRb().getString("time.of.day.shares.general")); - TableDataSet TODValuesBorder = fafUtils.importTable(utilities.getRb().getString("time.of.day.shares.border")); - todNames = TODValuesGeneral.getColumnAsString("DESCRIPTION"); - todSutShare = new float[2][TODValuesGeneral.getRowCount()]; - todMutShare = new float[2][TODValuesGeneral.getRowCount()]; - float[] checkSum = new float[4] ; - for (int row = 1; row <= TODValuesGeneral.getRowCount(); row++) { - todSutShare[0][row-1] = TODValuesGeneral.getValueAt(row, "ShareSUT"); - todMutShare[0][row-1] = TODValuesGeneral.getValueAt(row, "ShareMUT"); - todSutShare[1][row-1] = TODValuesBorder.getValueAt(row, "ShareSUT"); - todMutShare[1][row-1] = TODValuesBorder.getValueAt(row, "ShareMUT"); - checkSum[0] += todSutShare[0][row-1]; - checkSum[1] += todMutShare[0][row-1]; - checkSum[2] += todSutShare[1][row-1]; - checkSum[3] += todMutShare[1][row-1]; - } - if (checkSum[0] > 1.001 || checkSum[0] < 0.999) logger.warn("Time of day share for SUT (general) does not add up to 1 but " + checkSum[0]); - if (checkSum[1] > 1.001 || checkSum[1] < 0.999) logger.warn("Time of day share for MUT (general) does not add up to 1 but " + checkSum[1]); - if (checkSum[2] > 1.001 || checkSum[2] < 0.999) logger.warn("Time of day share for SUT (border) does not add up to 1 but " + checkSum[2]); - if (checkSum[3] > 1.001 || checkSum[3] < 0.999) logger.warn("Time of day share for MUT (border) does not add up to 1 but " + checkSum[3]); - } - - - private void disaggregateFlowsFromCountiesToZones() { - // calculate local weights and disaggregate flows from counties/external stations to zones/external stations - - final HashMap weights = prepareZonalWeights(); - logger.info(" Disaggregating flows to zones"); - - int[] listOfCommodityGroupsPlusEmpties = new int[utilities.getListOfCommodityGroups().length + 1]; - listOfCommodityGroupsPlusEmpties[0] = 0; // category for empty trucks - System.arraycopy(utilities.getListOfCommodityGroups(), 0, listOfCommodityGroupsPlusEmpties, 1, utilities.getListOfCommodityGroups().length); - Integer[] list = new Integer[listOfCommodityGroupsPlusEmpties.length]; - for (int i = 0; i < listOfCommodityGroupsPlusEmpties.length; i++) list[i] = listOfCommodityGroupsPlusEmpties[i]; - Function1 commodityDisaggregationFunctionFAF = new Function1() { - public Void apply(Integer com) { - processCommodityDisaggregation(com, weights); - return null; - } - }; - - Iterator commodityIterator = ArrayUtil.getIterator(list); - IteratorAction itTask = new IteratorAction<>(commodityIterator, commodityDisaggregationFunctionFAF); - ForkJoinPool pool = ForkJoinPoolFactory.getForkJoinPool(); - pool.execute(itTask); - itTask.waitForCompletion(); - } - - - - - private HashMap prepareZonalWeights() { - // prepare zonal weights based on employment by industry - - logger.info(" Calculating zonal weights"); - HashMap weights = new HashMap<>(); - - double[] makeEmpty = new double[zones.length]; - double[] useEmpty = new double[zones.length]; - - for (int comGrp: utilities.getListOfCommodityGroups()) { - double[] makeWeight = new double[zones.length]; - double[] useWeight = new double[zones.length]; - for (int com: utilities.getComGroupDefinition().get(comGrp)) { - for (int iz = 0; iz < zones.length; iz++) { - int zn = zones[iz]; - for (int ind = 0; ind < industries.length; ind++) { - String industry = industries[ind]; - String code; - if (com <= 9) code = industry + "_SCTG0" + com; - else code = industry + "_SCTG" + com; - makeWeight[iz] += zonalEmployment[zn][ind] * makeHshLocal.get(code); - useWeight[iz] += zonalEmployment[zn][ind] * useHshLocal.get(code); - makeEmpty[iz] += zonalEmployment[zn][ind] * makeHshLocal.get(code); - useEmpty[iz] += zonalEmployment[zn][ind] * useHshLocal.get(code); - } - } - } - String mCode = comGrp + "_make"; - weights.put(mCode, makeWeight); - String uCode = comGrp + "_use"; - weights.put(uCode, useWeight); - } - weights.put("0_make", makeEmpty); - weights.put("0_use", useEmpty); - return weights; - } - - - - private boolean checkIfCountyInSanDiego(int node) { - // check if county is either San Diego County of San Diego Phantom County - - boolean insideSandag = false; - for (int i: sanDiegoNodes) if (i == node) insideSandag = true; - return insideSandag; - } - - - private void processCommodityDisaggregation(Integer comGroup, Map weights) { - // Disaggregate a single commodity from county-to-county flows to zone-to-zone flows - logger.info(" Processing commodity group " + comGroup); - - ArrayList origAL = new ArrayList<>(); - ArrayList destAL = new ArrayList<>(); - ArrayList flowSutAL = new ArrayList<>(); - ArrayList flowMutAL = new ArrayList<>(); - for (int row = 1; row <= countyFlows.getRowCount(); row ++) { - int orig = (int) countyFlows.getValueAt(row, "orig"); - int dest = (int) countyFlows.getValueAt(row, "dest"); - String sutLabel; - if (comGroup == 0) sutLabel = "sutEmpty"; - else sutLabel = "sut" + comGroup; - float sutTrk = countyFlows.getValueAt(row, sutLabel); - String mutLabel; - if (comGroup == 0) mutLabel = "mutEmpty"; - else mutLabel = "mut" + comGroup; - float mutTrk = countyFlows.getValueAt(row, mutLabel); - - if (checkIfCountyInSanDiego(orig) && checkIfCountyInSanDiego(dest)) { - // flows from San Diego County to San Diego Phantom County or vice versa - // ignore as these flows are internal to SANDAG and known to be underestimated - } else if (checkIfCountyInSanDiego(orig)) { - // flows from San Diego to elsewhere - double[] makeShare = weights.get(comGroup + "_make"); - double makeShareSum = utilities.getSum(makeShare); - for (int zone = 0; zone < zones.length; zone++) { - origAL.add(zones[zone]); - destAL.add(sandagExtStatID[dest]); - flowSutAL.add((float) (sutTrk * makeShare[zone] / makeShareSum)); - flowMutAL.add((float) (mutTrk * makeShare[zone] / makeShareSum)); - } - } else if (checkIfCountyInSanDiego(dest)) { - // flows from elsewhere to San Diego - double[] useShare = weights.get(comGroup + "_use"); - double useShareSum = utilities.getSum(useShare); - for (int zone = 0; zone < zones.length; zone++) { - origAL.add(sandagExtStatID[orig]); - destAL.add(zones[zone]); - flowSutAL.add((float) (sutTrk * useShare[zone] / useShareSum)); - flowMutAL.add((float) (mutTrk * useShare[zone] / useShareSum)); - } - } else { - // through flows through San Diego - origAL.add(sandagExtStatID[orig]); - destAL.add(sandagExtStatID[dest]); - flowSutAL.add(sutTrk); - flowMutAL.add(mutTrk); - } - - TableDataSet disFlows = new TableDataSet(); - disFlows.appendColumn(utilities.convertIntArrayListToArray(origAL), "orig"); - disFlows.appendColumn(utilities.convertIntArrayListToArray(destAL), "dest"); - disFlows.appendColumn(utilities.convertFloatArrayListToArray(flowSutAL), "sut"); - disFlows.appendColumn(utilities.convertFloatArrayListToArray(flowMutAL), "mut"); - - synchronized (disaggregatedFlows) { - disaggregatedFlows.put(comGroup, disFlows); - } - } - } - - - private void writeOutDisaggregatedFlows () { - // write out disaggregated flows to csv file - - logger.info(" Writing zone-to-external station truck trip table"); - String fileName = utilities.getRb().getString("zone.to.ext.stat.trip.table") + "_" + utilities.getYear() + ".csv"; - PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); - pw.print("orig,dest"); - for (String tod: todNames) pw.print(",SUT " + tod); - for (String tod: todNames) pw.print(",MUT " + tod); - pw.println(); - - HashMap summarizedFlows = new HashMap<>(); - - for (int comGrp = 0; comGrp <= utilities.getHighestVal(utilities.getListOfCommodityGroups()); comGrp++) { - TableDataSet flows = disaggregatedFlows.get(comGrp); - - for (int row = 1; row <= flows.getRowCount(); row++) { - String key = (int) flows.getValueAt(row, "orig") + "," + (int) flows.getValueAt(row, "dest"); - float[] values; - if (summarizedFlows.containsKey(key)) { - values = summarizedFlows.get(key); - } else { - values = new float[]{0,0}; - } - values[0] += flows.getValueAt(row, "sut"); - values[1] += flows.getValueAt(row, "mut"); - summarizedFlows.put(key, values); - } - } - - for (String key: summarizedFlows.keySet()) { - float[] values = summarizedFlows.get(key); - pw.print(key); - - // check if flow crosses border with Mexico, in which case different time-of-day split is used - String[] odPair = key.split(","); - int border = 0; - if (sandagExtStat_border[Integer.parseInt(odPair[0])] || - sandagExtStat_border[Integer.parseInt(odPair[1])]) border = 1; - - for (int tod = 0; tod < todNames.length; tod++) pw.print("," + values[0] * todSutShare[border][tod]); - for (int tod = 0; tod < todNames.length; tod++) pw.print("," + values[1] * todMutShare[border][tod]); - pw.println(); - } - pw.close(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java b/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java deleted file mode 100644 index fcd4ed9..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/applications/sandag_tm.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.sandag.htm.applications; - -import org.sandag.htm.processFAF.disaggregateFlows; -import org.sandag.htm.processFAF.readFAF3; -import org.sandag.htm.processFAF.ReadFAF4; - -import org.apache.log4j.Logger; - -/** - * Program to model SANDAG external truck flows based on FAF3 data - * Author: Rolf Moeckel, PB Albuquerque - * Date: 6 March 2013 (Santa Fe, NM) - * Version 1.0 - * - * Modified 2017-12-28 to use FAF4 data by JEF, RSG - */ - -public class sandag_tm { - - private static Logger logger = Logger.getLogger(sandag_tm.class); - - - public static void main(String[] args) { - - long startTime = System.currentTimeMillis(); - - ReadFAF4 faf4 = new ReadFAF4(); - disaggregateFlows df = new disaggregateFlows(); - utilities.truckModelInitialization(args, faf4, df); - - if (utilities.getModel().equalsIgnoreCase("counties")) { - SandagCountyModel scm = new SandagCountyModel(faf4, df); - scm.runSandagCountyModel(); - } else { - sandagZonalModel szm = new sandagZonalModel(); - szm.runSandagZonalModel(); - } - - logger.info("Finished SANDAG Truck Model for year " + utilities.getYear()); - float endTime = utilities.rounder(((System.currentTimeMillis() - startTime) / 60000), 1); - int hours = (int) (endTime / 60); - int min = (int) (endTime - 60 * hours); - logger.info("Runtime: " + hours + " hours and " + min + " minutes."); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java b/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java deleted file mode 100644 index 330aca4..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/applications/utilities.java +++ /dev/null @@ -1,217 +0,0 @@ -package org.sandag.htm.applications; - -import com.pb.common.datafile.DBFFileReader; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import org.sandag.htm.processFAF.disaggregateFlows; -import org.sandag.htm.processFAF.fafUtils; -import org.sandag.htm.processFAF.readFAF3; -import org.sandag.htm.processFAF.ReadFAF4; - -import org.apache.log4j.Logger; - -import java.io.File; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ResourceBundle; - -/** - * Utility methods for SANDAG external truck model - * Author: Rolf Moeckel, PB Albuquerque - * Date: 6 March 2013 (Santa Fe, NM) - * Version 1.0 - */ - -public class utilities { - - - private static ResourceBundle rb; - private static String model; - private static int year; - public static int[] countyFips; - public static int[] countyFipsIndex; - private static int[] commodityGroupOfSCTG; - private static HashMap comGroupDefinition; - private static int[] listOfCommodityGroups; - private static Logger logger = Logger.getLogger(utilities.class); - - - public static void truckModelInitialization(String[] args, readFAF3 faf3, disaggregateFlows df) { - // read in properties file and define basic variables, such as model year - - String rbName = args[0]; - File propFile = new File(rbName); - rb = ResourceUtil.getPropertyBundle(propFile); - model = args[1]; - if (!model.equalsIgnoreCase("counties") && !model.equalsIgnoreCase("zones")) { - logger.error("Call program with parameters "); - logger.error("Geography " + args[1] + " not understood. Choose \"counties\" or \"zones\""); - System.exit(1); - } - year = Integer.parseInt(args[2]); - logger.info("Starting SANDAG Truck Model for year " + utilities.getYear()); - readCommodityGrouping(); - } - - - public static void truckModelInitialization(String[] args, ReadFAF4 faf4, disaggregateFlows df) { - // read in properties file and define basic variables, such as model year - - String rbName = args[0]; - File propFile = new File(rbName); - rb = ResourceUtil.getPropertyBundle(propFile); - model = args[1]; - if (!model.equalsIgnoreCase("counties") && !model.equalsIgnoreCase("zones")) { - logger.error("Call program with parameters "); - logger.error("Geography " + args[1] + " not understood. Choose \"counties\" or \"zones\""); - System.exit(1); - } - year = Integer.parseInt(args[2]); - logger.info("Starting SANDAG Truck Model for year " + utilities.getYear()); - readCommodityGrouping(); - } - - private static void readCommodityGrouping () { - // read in commodity grouping - - TableDataSet commodityGrouping = fafUtils.importTable(utilities.getRb().getString("commodity.grouping")); - int highestValue = utilities.getHighestVal(commodityGrouping.getColumnAsInt("SCTG")); - commodityGroupOfSCTG = new int[highestValue + 1]; - comGroupDefinition = new HashMap<>(); - for (int row = 1; row <= commodityGrouping.getRowCount(); row++) { - int sctg = (int) commodityGrouping.getValueAt(row, "SCTG"); - int grp = (int) commodityGrouping.getValueAt(row, "CommodityGroup"); - commodityGroupOfSCTG[sctg] = grp; - if (comGroupDefinition.containsKey(grp)) { - int[] commodities = comGroupDefinition.get(grp); - comGroupDefinition.put(grp, expandArrayByOneElement(commodities, sctg)); - } else { - comGroupDefinition.put(grp, new int[]{sctg}); - } - } - listOfCommodityGroups = new int[comGroupDefinition.size()]; - int count = 0; - for (int comGrp: comGroupDefinition.keySet()) { - listOfCommodityGroups[count] = comGrp; - count++; - } - } - - - public static int[] getCommodityGroupOfSCTG() { - return commodityGroupOfSCTG; - } - - - public static HashMap getComGroupDefinition() { - return comGroupDefinition; - } - - - public static int[] getListOfCommodityGroups() { - return listOfCommodityGroups; - } - - - public static float rounder(float value, int digits) { - // rounds value to digits behind the decimal point - return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); - } - - - public static int getYear() { - return year; - } - - public static ResourceBundle getRb() { - return rb; - } - - public static String getModel() { - return model; - } - - public static boolean getBooleanProperty (String token, boolean defaultIfNotAvailable) { - return ResourceUtil.getBooleanProperty(rb, token, defaultIfNotAvailable); - } - - - public static int getHighestVal(int[] array) { - // return highest number in array - - int high = Integer.MIN_VALUE; - for (int num: array) high = Math.max(high, num); - return high; - } - - - public static float rounder(double value, int digits) { - // rounds value to digits behind the decimal point - return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); - } - - - public static int[] expandArrayByOneElement (int[] existing, int addElement) { - // create new array that has length of existing.length + 1 and copy values into new array - int[] expanded = new int[existing.length + 1]; - System.arraycopy(existing, 0, expanded, 0, existing.length); - expanded[expanded.length - 1] = addElement; - return expanded; - } - - - public static void createZoneList() { - // Create array with specialRegions that serve as port of entry/exit - - int[] poeLand = fafUtils.importTable(rb.getString("ports.of.entry")).getColumnAsInt("pointOfEntry"); - int[] poeSea = fafUtils.importTable(rb.getString("marine.ports.of.entry")).getColumnAsInt("pointOfEntry"); - int[] poeAir = fafUtils.importTable(rb.getString("air.ports.of.entry")).getColumnAsInt("pointOfEntry"); - int[] list = new int[poeLand.length + poeSea.length + poeAir.length]; - System.arraycopy(poeLand, 0, list, 0, poeLand.length); - System.arraycopy(poeSea, 0, list, poeLand.length, poeSea.length); - System.arraycopy(poeAir, 0, list, poeLand.length + poeSea.length, poeAir.length); - countyFips = fafUtils.createCountyFipsArray(list); - countyFipsIndex = new int[fafUtils.getHighestVal(countyFips) + 1]; - for (int i = 0; i < countyFips.length; i++) { - countyFipsIndex[countyFips[i]] = i; - } - } - - - public static TableDataSet importTableFromDBF(String filePath) { - // read a dbf file into a TableDataSet - - TableDataSet tblData; - DBFFileReader dbfReader = new DBFFileReader(); - try { - tblData = dbfReader.readFile(new File( filePath )); - } catch (Exception e) { - throw new RuntimeException("File not found: <" + filePath + ">.", e); - } - dbfReader.close(); - return tblData; - } - - - public static double getSum (double[] array) { - // return sum of all elements in array - double sum = 0; - for (double val: array) sum += val; - return sum; - } - - - public static float[] convertIntArrayListToArray(ArrayList al) { - float[] array = new float[al.size()]; - for (int i = 0; i < al.size(); i++) array[i] = al.get(i); - return array; - } - - - public static float[] convertFloatArrayListToArray(ArrayList al) { - float[] array = new float[al.size()]; - for (int i = 0; i < al.size(); i++) array[i] = al.get(i); - return array; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java b/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java deleted file mode 100644 index 72a67eb..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/applications/yumaMPO.java +++ /dev/null @@ -1,180 +0,0 @@ -package org.sandag.htm.applications; - -import com.pb.common.datafile.TableDataSet; -import org.sandag.htm.processFAF.fafUtils; -import org.sandag.htm.processFAF.readFAF3; - -import java.io.PrintWriter; -import java.util.ResourceBundle; - -/** - * Application of countyTruckModel for Yuma, AZ MPO - * Author: Rolf Moeckel, PB Albuquerque - * Date: April 5, 2012 (Santa Fe NM) - - */ -public class yumaMPO { - - private static float[][][] tons_yuma_border; - private static float[][][] tons_yuma_fafzones; - private static float[][][] tons_yuma_counties; - private static float[][] tons_yuma_mex; - private static TableDataSet counties; - private static int[] countyIndex; - - - public static void initializeVariables(ResourceBundle appRb) { - // set up variables for summaries - int maxCom = fafUtils.getHighestVal(readFAF3.sctgCommodities); - tons_yuma_border = new float[2][560 + 1][maxCom + 1]; // by direction, FAF zone and commodity - tons_yuma_fafzones = new float[2][560 + 1][maxCom + 1]; // by direction, FAF zone and commodity - tons_yuma_counties = new float[2][15 + 1][maxCom + 1]; // by direction, county and commodity - counties = fafUtils.importTable(appRb.getString("county.ID")); - counties.buildIndex(counties.getColumnPosition("COUNTYFIPS")); - countyIndex = new int[fafUtils.getHighestVal(counties.getColumnAsInt("COUNTYFIPS")) + 1]; - for (int i = 0; i < countyIndex.length; i++) countyIndex[i] = -1; - countyIndex[4001] = 0; // index counties within Arizona - countyIndex[4003] = 1; - countyIndex[4005] = 2; - countyIndex[4007] = 3; - countyIndex[4009] = 4; - countyIndex[4011] = 5; - countyIndex[4012] = 6; - countyIndex[4013] = 7; - countyIndex[4015] = 8; - countyIndex[4017] = 9; - countyIndex[4019] = 10; - countyIndex[4021] = 11; - countyIndex[4023] = 12; - countyIndex[4025] = 13; - countyIndex[4027] = 14; - countyIndex[6025] = 15; // index Imperial County in California - tons_yuma_mex = new float[2][2]; - } - - - public static void saveForYuma(int com, int oFips, int dFips, float tons) { - // save relevant data for Yuma-specific summaries - -// try { - int oFAF = 0; - if (oFips < 60000) oFAF = (int) counties.getIndexedValueAt(oFips, "FAF3region"); - int dFAF = 0; - if (dFips < 60000) dFAF = (int) counties.getIndexedValueAt(dFips, "FAF3region"); - - // Flows through Yuma border crossing - if (oFips == 61934 && dFAF != 0) { - tons_yuma_border[0][dFAF][com] += tons; - } else if (dFips == 61934 && oFAF != 0) { - tons_yuma_border[1][oFAF][com] += tons; - } - // Flows to/from Yuma - if (oFips == 4027 && dFAF != 0) { - tons_yuma_fafzones[0][dFAF][com] += tons; - if (countyIndex[dFips] >= 0) tons_yuma_counties[0][countyIndex[dFips]][com] += tons; - } else if (dFips == 4027 && oFAF != 0) { - tons_yuma_fafzones[1][oFAF][com] += tons; - if (countyIndex[oFips] >= 0) tons_yuma_counties[1][countyIndex[oFips]][com] += tons; - } - // Flows between Yuma and Mexico - if (oFips == 4027) tons_yuma_mex[0][0] += tons; - if (oFips == 4027 && dFips > 60000) tons_yuma_mex[0][1] += tons; - if (dFips == 4027) tons_yuma_mex[1][0] += tons; - if (dFips == 4027 && oFips > 60000) tons_yuma_mex[1][1] += tons; - -// } catch (Exception e) { -// System.out.println("Error: " + com+" "+oFips+" "+dFips); -// } - } - - - public static void writeOutResults(ResourceBundle appRb, int year) { - // write results to summary file - - PrintWriter pw = fafUtils.openFileForSequentialWriting(appRb.getString("yuma.summary") + year + ".csv"); - - pw.println("Total tons leaving Yuma : " + tons_yuma_mex[0][0]); - pw.println("Tons from Yuma to Mexico: " + tons_yuma_mex[0][1]); - pw.println("Total tons entering Yuma: " + tons_yuma_mex[1][0]); - pw.println("Tons from Mexico to Yuma: " + tons_yuma_mex[1][1]); - pw.println(); - - pw.println("SUMMARY BY FAF ZONE"); - pw.print("FlowsFromYumaToFAFZone"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int faf = 1; faf <= 560; faf++) { - float sum = 0; - for (int i: readFAF3.sctgCommodities) sum += tons_yuma_fafzones[0][faf][i]; - if (sum > 0) { - pw.print(faf); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_fafzones[0][faf][i]); - pw.println(); - } - } - pw.println(); - pw.print("FlowsToYumaFromFAFZone"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int faf = 1; faf <= 560; faf++) { - float sum = 0; - for (int i: readFAF3.sctgCommodities) sum += tons_yuma_fafzones[1][faf][i]; - if (sum > 0) { - pw.print(faf); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_fafzones[1][faf][i]); - pw.println(); - } - } - pw.println(); - - pw.println("SUMMARY BY COUNTY"); - pw.print("FlowsFromYumaToCounty"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int county: counties.getColumnAsInt("COUNTYFIPS")) { - if (countyIndex[county] == -1) continue; - pw.print(county); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_counties[0][countyIndex[county]][i]); - pw.println(); - } - pw.println(); - pw.print("FlowsToYumaFromCounty"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int county: counties.getColumnAsInt("COUNTYFIPS")) { - if (countyIndex[county] == -1) continue; - pw.print(county); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_counties[1][countyIndex[county]][i]); - pw.println(); - } - pw.println(); - - pw.println("SUMMARY BY BORDER ZONE"); - pw.print("FlowsFromYumaBorderToFAFZone"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int faf = 1; faf <= 560; faf++) { - float sum = 0; - for (int i: readFAF3.sctgCommodities) sum += tons_yuma_border[0][faf][i]; - if (sum > 0) { - pw.print(faf); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_border[0][faf][i]); - pw.println(); - } - } - pw.println(); - pw.print("FlowsToYumaBorderFromFAFZone"); - for (int i: readFAF3.sctgCommodities) pw.print(",SCTG" + i); - pw.println(); - for (int faf = 1; faf <= 560; faf++) { - float sum = 0; - for (int i: readFAF3.sctgCommodities) sum += tons_yuma_border[1][faf][i]; - if (sum > 0) { - pw.print(faf); - for (int i: readFAF3.sctgCommodities) pw.print("," + tons_yuma_border[1][faf][i]); - pw.println(); - } - } - pw.close(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java deleted file mode 100644 index ef5e308..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ModesFAF.java +++ /dev/null @@ -1,19 +0,0 @@ -package org.sandag.htm.processFAF; - -/** - * Defines modes available in FAF - * Author: Rolf Moeckel (PB Albuquerque) - * Date: September 9, 2010 - * Edited jef 2017-12-28 to remove reference to FAF3 since FAF3 modes are same as FAF4 - */ - -public enum ModesFAF { - - Truck, - Rail, - Water, - Air, - MultipleModesAndMail, - Pipeline, - OtherAndUnknown -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java deleted file mode 100644 index 8984f5c..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/ReadFAF4.java +++ /dev/null @@ -1,522 +0,0 @@ -package org.sandag.htm.processFAF; - -import org.apache.log4j.Logger; - -import java.util.ResourceBundle; -import java.util.HashMap; -import java.io.PrintWriter; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -/** - * This class reads FAF4 data and stores data in a TableDataSet - * Author: Joel Freedman, RSG - based on readFAF3 by Rolf Moeckel, PB - * Date: Dec 27, 2017 - */ - -public class ReadFAF4 { - Logger logger = Logger.getLogger(ReadFAF4.class); - private int factor; - private String[] valueColumnName; - private TableDataSet faf4commodityFlows; - public static TableDataSet fafRegionList; - private String[] regionState; - private static int[] domRegionIndex; - static public int[] sctgCommodities; - static public String[] sctgStringCommodities; - static private int[] sctgStringIndex; - private static HashMap portsOfEntry; - private static HashMap marinePortsOfEntry; - private static HashMap railPortsOfEntry; - private static HashMap airPortsOfEntry; - private static int[] listOfBorderPortOfEntries; - - - public void readAllData (ResourceBundle appRb, int year, String unit) { - // read input data - - if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) - readAllFAF4DataSets(appRb, unit, year); - readCommodityList(appRb); - readFAF4ReferenceLists(appRb); - } - - - public void readCommodityList(ResourceBundle appRb) { - // read commodity names - TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf4.sctg.commodity.list")); - sctgCommodities = new int[sctgComList.getRowCount()]; - sctgStringCommodities = new String[sctgCommodities.length]; - for (int i = 1; i <= sctgComList.getRowCount(); i++) { - sctgCommodities[i-1] = (int) sctgComList.getValueAt(i, "SCTG"); - if (sctgCommodities[i-1] < 10) sctgStringCommodities[i-1] = "SCTG0" + sctgCommodities[i-1]; - else sctgStringCommodities[i-1] = "SCTG" + sctgCommodities[i-1]; - } - sctgStringIndex = new int[fafUtils.getHighestVal(sctgCommodities) + 1]; - for (int num = 0; num < sctgCommodities.length; num++) sctgStringIndex[sctgCommodities[num]] = num; - } - - - public int getIndexOfCommodity (int commodity) { - return sctgStringIndex[commodity]; - } - - - public static String getSCTGname(int sctgInt) { - // get String name from sctg number - return sctgStringCommodities[sctgStringIndex[sctgInt]]; - } - - - public static String getFAFzoneName(int fafInt) { - // get String name from int FAF zone code number - return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "FAF4 Zones -Short Description"); - } - - - public static String getFAFzoneState(int fafInt) { - // get String two-letter abbreviation of state of fafInt - return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "State"); - } - - - public void definePortsOfEntry(ResourceBundle appRb) { - // read data to translate ports of entry in network links - - // Border crossings - portsOfEntry = new HashMap<>(); - TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); - for (int row = 1; row <= poe.getRowCount(); row++) { - int fafID = (int) poe.getValueAt(row, "faf4id"); - int node = (int) poe.getValueAt(row, "pointOfEntry"); - float weight = poe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (portsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = portsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - portsOfEntry.put(fafID, newPortsOfEntry); - } - listOfBorderPortOfEntries = poe.getColumnAsInt("pointOfEntry"); - - // Marine ports - marinePortsOfEntry = new HashMap<>(); - if (appRb.containsKey("marine.ports.of.entry")) { - TableDataSet mpoe = fafUtils.importTable(appRb.getString("marine.ports.of.entry")); - for (int row = 1; row <= mpoe.getRowCount(); row++) { - int fafID = (int) mpoe.getValueAt(row, "faf4id"); - int node = (int) mpoe.getValueAt(row, "pointOfEntry"); - float weight = mpoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (marinePortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = marinePortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - marinePortsOfEntry.put(fafID, newPortsOfEntry); - } - } - // Rail ports (railyards) - railPortsOfEntry = new HashMap<>(); - if (appRb.containsKey("rail.ports.of.entry")) { - TableDataSet rpoe = fafUtils.importTable(appRb.getString("rail.ports.of.entry")); - for (int row = 1; row <= rpoe.getRowCount(); row++) { - int fafID = (int) rpoe.getValueAt(row, "faf4id"); - int node = (int) rpoe.getValueAt(row, "pointOfEntry"); - float weight = rpoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (railPortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = railPortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - railPortsOfEntry.put(fafID, newPortsOfEntry); - } - } - // Airports - airPortsOfEntry = new HashMap<>(); - if (appRb.containsKey("air.ports.of.entry")) { - TableDataSet apoe = fafUtils.importTable(appRb.getString("air.ports.of.entry")); - for (int row = 1; row <= apoe.getRowCount(); row++) { - int fafID = (int) apoe.getValueAt(row, "faf4id"); - int node = (int) apoe.getValueAt(row, "pointOfEntry"); - float weight = apoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (airPortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = airPortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - airPortsOfEntry.put(fafID, newPortsOfEntry); - } - } - } - - - public static int[] getListOfBorderPortOfEntries() { - return listOfBorderPortOfEntries; - } - - - - public static TableDataSet getPortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (portsOfEntry.containsKey(fafZone)) return portsOfEntry.get(fafZone); - else return null; - } - - - public static TableDataSet getMarinePortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (marinePortsOfEntry.containsKey(fafZone)) return marinePortsOfEntry.get(fafZone); - else return null; - } - - - public static TableDataSet getAirPortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (airPortsOfEntry.containsKey(fafZone)) return airPortsOfEntry.get(fafZone); - else return null; - } - - - public void readAllFAF4dataSets2015(ResourceBundle appRb, String unit) { - // read all FAF4 data into TableDataSets in unit (= tons or dollars) - - logger.info ("Reading domestic FAF4 data in " + unit); - if (unit.equals("tons")) { - factor = 1000; // tons are in 1,000s - valueColumnName = new String[]{"tons_2015"}; - } else if (unit.equals("dollars")) { - factor = 1000000; // dollars are in 1,000,000s - valueColumnName = new String[]{"value_2015"}; - } else { - logger.fatal ("Wrong token " + unit + " in method readAllFAF4dataSets2015. Use tons or dollars."); - } - faf4commodityFlows = readFAF4CommodityFlows(appRb, unit); - } - - - public double[] summarizeFlowByCommodity (ModesFAF fafMode) { - // sum commodity flows by commodity and return array with total tons - - double[] totalFlows = new double[sctgCommodities.length]; - int modeNum = fafUtils.getEnumOrderNumber(fafMode); - for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { - if (faf4commodityFlows.getValueAt(row, "dms_mode") != modeNum) continue; - int com = sctgStringIndex[(int) faf4commodityFlows.getValueAt(row, "sctg2")]; - if (valueColumnName.length == 1) { - // use year provided by user - totalFlows[com] += faf4commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); - totalFlows[com] += val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - } - return totalFlows; - } - - - public void readAllFAF4DataSets(ResourceBundle appRb, String unit, int year) { - // read all FAF4 data into TableDataSets in unit (= tons or dollars) - - logger.info (" Reading FAF4 data in " + unit); - switch (unit) { - case "tons": - factor = 1000; // tons are provided in 1,000s - break; - case "dollars": - factor = 1000000; // dollars are provided in 1,000,000s - break; - default: - logger.fatal("Wrong token " + unit + " in method readAllFAF4DataSets. Use tons or dollars."); - throw new RuntimeException(); - } - int[] availYears = {2012, 2013, 2014, 2015, 2020, 2025, 2030, 2035, 2040, 2045}; - boolean yearInFaf = false; - for (int y: availYears) if (year == y) yearInFaf = true; - if (!yearInFaf) { // interpolate between two years - logger.info(" Year " + year + " does not exist in FAF4 data."); - int year1 = availYears[0]; - int year2 = availYears[availYears.length-1]; - for (int availYear : availYears) if (availYear < year) year1 = availYear; - for (int i = availYears.length - 1; i >= 0; i--) if (availYears[i] > year) year2 = availYears[i]; - logger.info(" FAF4 data are interpolated between " + year1 + " and " + year2 + "."); - // first position: lower year, second position: higher year, third position: steps away from lower year - valueColumnName = new String[]{unit + "_" + year1, unit + "_" + year2, String.valueOf((1f * (year - year1)) / (1f * (year2 - year1)))}; - } else { // use year provided by user - valueColumnName = new String[]{unit + "_" + year}; - } - faf4commodityFlows = readFAF4CommodityFlows(appRb, unit); - } - - - private TableDataSet readFAF4CommodityFlows(ResourceBundle appRb, String unit) { - // read FAF4 data and return TableDataSet with flow data - String fileName = ResourceUtil.getProperty(appRb, ("faf4.data")); - return fafUtils.importTable(fileName); - } - - - public HashMap createScaler(String[] tokens, double[] values) { - // create HashMap with state O-D pairs that need to be scaled - - HashMap scaler = new HashMap(); - if (tokens.length != values.length) { - throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); - } - for (int i=0; i scaler) { - // extract truck flows for year yr and scale flows according to scaler HashMap (no special regions specified) - - PrintWriter outFile = fafUtils.openFileForSequentialWriting(outFileName); - outFile.println("originFAF,destinationFAF,flowDirection," + commodityClassType.SCTG + "_commodity,shortTons"); - int modeNum = fafUtils.getEnumOrderNumber(mode); - for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { - int type = (int) faf4commodityFlows.getValueAt(row, "trade_type"); - double val; - if (valueColumnName.length == 1) { - // use year provided by user - val = faf4commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); - val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - val *= factor * odScaler(row, type, scaler); - if (val == 0) continue; - if (type == 1) writeDomesticFlow(modeNum, val, row, outFile); - else if (type == 2) writeImportFlow(modeNum, val, row, outFile, repF); - else if (type == 3) writeExportFlow(modeNum, val, row, outFile, repF); - else if (type == 4) writeThroughFlow(modeNum, val, row, outFile, repF); - else{ - logger.info("Invalid trade_type in FAF4 dataset in row " + row + ": " + type); - } - } - outFile.close(); - } - - - public void writeFlowsByModeAndCommodity (String outFileName, ModesFAF mode, reportFormat repF, - HashMap scaler) { - // extract truck flows for year yr and scale flows according to scaler HashMap, including special regions - - PrintWriter outFile[] = new PrintWriter[sctgCommodities.length]; - for (int com: sctgCommodities) { - String fileName; - if (com < 10) fileName = outFileName + "_SCTG0" + com + ".csv"; - else fileName = outFileName + "_SCTG" + com + ".csv"; - outFile[sctgStringIndex[com]] = fafUtils.openFileForSequentialWriting(fileName); - outFile[sctgStringIndex[com]].println("originFAF,destinationFAF,flowDirection,SCTG_commodity,shortTons"); - } - int modeNum = fafUtils.getEnumOrderNumber(mode); - for (int row = 1; row <= faf4commodityFlows.getRowCount(); row++) { - int type = (int) faf4commodityFlows.getValueAt(row, "trade_type"); - double val; - if (valueColumnName.length == 1) { - // use year provided by user - val = faf4commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf4commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf4commodityFlows.getValueAt(row, valueColumnName[1]); - val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - val *= factor * odScaler(row, type, scaler); - if (val == 0) continue; - int comIndex = getIndexOfCommodity((int) faf4commodityFlows.getValueAt(row, "sctg2")); - if (type == 1) writeDomesticFlow(modeNum, val, row, outFile[comIndex]); - else if (type == 2) writeImportFlow(modeNum, val, row, outFile[comIndex], repF); - else if (type == 3) writeExportFlow(modeNum, val, row, outFile[comIndex], repF); - else if (type == 4) writeThroughFlow(modeNum, val, row, outFile[comIndex], repF); - else logger.info("Invalid trade_type in FAF4 dataset in row " + row + ": " + type); - } - for (int com: sctgCommodities) outFile[sctgStringIndex[com]].close(); - } - - - public float odScaler (int row, int type, HashMap scaler) { - // find scaler for origin destination pair in row - - int orig; - int dest; - if (type == 1) { - orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - } else if (type == 2) { - orig = (int) faf4commodityFlows.getValueAt(row, "fr_orig"); - dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - } else if (type == 3) { - orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - dest = (int) faf4commodityFlows.getValueAt(row, "fr_dest"); - } else { - orig = (int) faf4commodityFlows.getValueAt(row, "fr_orig"); - dest = (int) faf4commodityFlows.getValueAt(row, "fr_dest"); - } - String stateLevelToken = regionState[orig] + "_" + regionState[dest]; - String combo1Token = orig + "_" + regionState[dest]; - String combo2Token = regionState[orig] + "_" + dest; - String fafLevelToken = orig + "_" + dest; - float adj = 1; - if (scaler.containsKey(stateLevelToken)) adj = scaler.get(stateLevelToken); - if (scaler.containsKey(combo1Token)) adj = scaler.get(combo1Token); - if (scaler.containsKey(combo2Token)) adj = scaler.get(combo2Token); - if (scaler.containsKey(fafLevelToken)) adj = scaler.get(fafLevelToken); - return adj; - } - - - private int tryGettingThisValue(int row, String token) { - // for some flows, international zones/modes are empty -> catch this case and set zone to 0 - - int region; - try { - region = (int) faf4commodityFlows.getValueAt(row, token); - } catch (Exception e) { - region = 0; - } - return region; - } - - - public void writeDomesticFlow (int modeNum, double val, int row, PrintWriter outFile) { - // internal US flow - if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - int dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); - outFile.println(orig + "," + dest + ",domestic," + comm + "," + val); - } - } - - - public void writeImportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // from abroad to US - - int frInMode = (int) faf4commodityFlows.getValueAt(row, "fr_inmode"); - int borderZone = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); - if (frInMode == modeNum && repF != reportFormat.internat_domesticPart) { - int orig = tryGettingThisValue(row, "fr_orig"); - outFile.println(orig + "," + borderZone + ",import," + comm + "," + val); - } - if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int dest = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - String txt; - if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",import_port,"; - else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",import_rail,"; - else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",import_airport,"; - else txt = ",import,"; - outFile.println(borderZone + "," + dest + txt + comm + "," + val); - } - } - - - public void writeExportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // from US to abroad - int frOutMode = tryGettingThisValue(row, "fr_outmode"); - int borderZone = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - int comm = (int) faf4commodityFlows.getValueAt(row, "sctg2"); - if (frOutMode == modeNum && repF != reportFormat.internat_domesticPart) { - int dest = tryGettingThisValue(row, "fr_dest"); - outFile.println(borderZone + "," + dest + ",export," + comm + "," + val); - } - if (faf4commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int orig = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - String txt = ",export,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",export_port,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",export_rail,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",export_airport,"; - outFile.println(orig + "," + borderZone + txt + comm + "," + val); - } - } - - - public void writeThroughFlow(int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // flows in transit through US - if ((int) faf4commodityFlows.getValueAt(row, "dms_mode") != modeNum) return; - int borderInZone = (int) faf4commodityFlows.getValueAt(row, "dms_orig"); - int borderOutZone = (int) faf4commodityFlows.getValueAt(row, "dms_dest"); - logger.warn("Through flows not yet implemented. This flow from " + borderInZone + " to " + borderOutZone + " is lost."); - } - - - public void readFAF4ReferenceLists(ResourceBundle rb) { - // read list of regions for FAF4 - String regFileName = rb.getString("faf4.region.list"); - fafRegionList = fafUtils.importTable(regFileName); - int[] reg = fafRegionList.getColumnAsInt(fafRegionList.getColumnPosition("ZoneID")); - domRegionIndex = new int[fafUtils.getHighestVal(reg) + 1]; - for (int num = 0; num < reg.length; num++) domRegionIndex[reg[num]] = num + 1; - regionState = new String[fafUtils.getHighestVal(reg) + 1]; - for (int row = 1; row <= fafRegionList.getRowCount(); row++) { - int zone = (int) fafRegionList.getValueAt(row, "ZoneID"); - regionState[zone] = fafRegionList.getStringValueAt(row, "State"); - } - } - - - public int[] getFAFzoneIDs () { - return fafRegionList.getColumnAsInt("ZoneID"); - } - - - public TableDataSet getFAF4Flows() { - return faf4commodityFlows; - } - - - public int getFactor() { - return factor; - } - - - public TableDataSet getFAF4CommodityFlows() { - return faf4commodityFlows; - } - - - public String[] getValueColumnName() { - return valueColumnName; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java deleted file mode 100644 index 17e8314..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/commodityClassType.java +++ /dev/null @@ -1,13 +0,0 @@ -package org.sandag.htm.processFAF; - -/** - * Defines commodity classification - * STCC - * SCTG - * User: Moeckel - * Date: May 7, 2009 - */ - -public enum commodityClassType { - STCC, SCTG -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java deleted file mode 100644 index 3d578cd..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/convertTonsToTrucks.java +++ /dev/null @@ -1,113 +0,0 @@ -package org.sandag.htm.processFAF; - -import com.pb.common.datafile.TableDataSet; -import org.apache.log4j.Logger; - -import java.util.ResourceBundle; - -/** - * Class to convert flows from tons into trucks by type - * Author: Rolf Moeckel, PB Albuquerque - * Data: 11 March 2013 (Santa Fe) - */ - -public class convertTonsToTrucks { - - private static Logger logger = Logger.getLogger(convertTonsToTrucks.class); - private String[] truckTypes = new String[] {"SingleUnit","TruckTrailer","CombinationSemitrailer","CombinationDoubleTriple"}; - private int[] fromMiles; - private float[][] truckShareByDistance; - private TableDataSet[] payloadByTruckType; - private ResourceBundle appRb; - - public convertTonsToTrucks(ResourceBundle appRb) { - this.appRb = appRb; - } - - - public void readData () { - // read data to convert tons into trucks - - logger.info(" Reading FAF3 payload factors"); - // truck types by distance class - TableDataSet truckTypeByDist = fafUtils.importTable(appRb.getString("truck.type.share.by.distance")); - fromMiles = truckTypeByDist.getColumnAsInt("fromMiles"); - truckShareByDistance = new float[fromMiles.length][truckTypes.length]; - // todo: allow adjustment of SUT/MUT share - for (int row = 1; row <= truckTypeByDist.getRowCount(); row++) { - for (int col = 0; col < truckTypes.length; col++) { - truckShareByDistance[row-1][col] = truckTypeByDist.getValueAt(row, truckTypes[col]); - } - } - - // truck body type by commodity - payloadByTruckType = new TableDataSet[4]; - payloadByTruckType[0] = fafUtils.importTable(appRb.getString("truck.body.share.single.unit")); - payloadByTruckType[0].buildIndex(payloadByTruckType[0].getColumnPosition("sctg")); - payloadByTruckType[1] = fafUtils.importTable(appRb.getString("truck.body.share.tractor.trl")); - payloadByTruckType[1].buildIndex(payloadByTruckType[1].getColumnPosition("sctg")); - payloadByTruckType[2] = fafUtils.importTable(appRb.getString("truck.body.share.comb.semi.t")); - payloadByTruckType[2].buildIndex(payloadByTruckType[2].getColumnPosition("sctg")); - payloadByTruckType[3] = fafUtils.importTable(appRb.getString("truck.body.share.comb.dbl.tr")); - payloadByTruckType[3].buildIndex(payloadByTruckType[3].getColumnPosition("sctg")); - } - - - public float[] convertThisFlowFromTonsToTrucks (String commodity, float distance, float flowInTons) { - // convert commodity flow in tons into number of trucks by truck type - - // find correct distance class - int distClass = 0; - for (int i = 0; i < fromMiles.length; i++) if (distance > fromMiles[i]) distClass = i; - - // calculate trucks by truck type - float[] trucksByType = new float[truckTypes.length]; - for (int i = 0; i < truckTypes.length; i++) trucksByType[i] = loadTruckShare(i, distClass, commodity, flowInTons); - - return trucksByType; - } - - - private float loadTruckShare (int truckType, int distClass, String commodity, float flowInTons) { - // calculate payload for with for - - int com = Integer.parseInt(commodity.substring(4)); - float tonsOnThisTruckType = flowInTons * truckShareByDistance[distClass][truckType]; - - float trucks = 0; - for (int col = 2; col <= payloadByTruckType[truckType].getColumnCount(); col++) { - trucks += tonsOnThisTruckType * payloadByTruckType[truckType].getIndexedValueAt(com, col); - } - return trucks; - } - - - public double[] convertThisFlowFromTonsToTrucks (String commodity, float distance, double flowInTons) { - // convert commodity flow in tons into number of trucks by truck type - - // find correct distance class - int distClass = 0; - for (int i = 0; i < fromMiles.length; i++) if (distance > fromMiles[i]) distClass = i; - - // calculate trucks by truck type - double[] trucksByType = new double[truckTypes.length]; - for (int i = 0; i < truckTypes.length; i++) trucksByType[i] = loadTruckShare(i, distClass, commodity, flowInTons); - - return trucksByType; - } - - - private double loadTruckShare (int truckType, int distClass, String commodity, double flowInTons) { - // calculate payload for with for - - int com = Integer.parseInt(commodity.substring(4)); - double tonsOnThisTruckType = flowInTons * truckShareByDistance[distClass][truckType]; - - double trucks = 0; - for (int col = 2; col <= payloadByTruckType[truckType].getColumnCount(); col++) { - trucks += tonsOnThisTruckType * payloadByTruckType[truckType].getIndexedValueAt(com, col); - } - return trucks; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java deleted file mode 100644 index 0be8116..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/countyTruckModel.java +++ /dev/null @@ -1,597 +0,0 @@ -package org.sandag.htm.processFAF; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import org.sandag.htm.applications.ohio; -import org.sandag.htm.applications.yumaMPO; -import org.apache.log4j.Logger; -import com.pb.sawdust.util.concurrent.DnCRecursiveAction; - -import java.io.File; -import java.io.PrintWriter; -import java.util.HashMap; -import java.util.ResourceBundle; - -/** - * Reads FAF3 data, disaggregates them to county-to-county flows and converts them into truck trips - * Author: Rolf Moeckel, PB Albuquerque - * Date: August 22, 2011 (Santa Fe NM) - * Revised for Yuma County: March 29, 2012 (Albuquerque, NM) - */ - -public class countyTruckModel { - - private static Logger logger = Logger.getLogger(countyTruckModel.class); - private ResourceBundle appRb; - private int year; - private disaggregateFlows df; - private HashMap cntFlows; - private int[] countyFips; - private int[] countyFipsIndex; - private int[] commodityGrouping; - private int numCommodityGroups; - private boolean yumaSummary; - - - public countyTruckModel(ResourceBundle rb, int yr) { - this.appRb = rb; - this.year = yr; - } - - - public static void main(String[] args) { - // construct main model - - long startTime = System.currentTimeMillis(); - ResourceBundle appRb = fafUtils.getResourceBundle(args[0]); - int year = Integer.parseInt(args[1]); - countyTruckModel ctm = new countyTruckModel(appRb, year); - ctm.run(); - logger.info("County Truck Model completed."); - float endTime = fafUtils.rounder(((System.currentTimeMillis() - startTime) / 60000), 1); - logger.info("Runtime: " + endTime + " minutes."); - } - - - private void run() { - // main run method - - logger.info("Started FAF4 model to generate county-to-county truck flows for " + year); - ReadFAF4 faf4 = new ReadFAF4(); - df = new disaggregateFlows(); - df.getUScountyEmploymentByIndustry(appRb); - faf4.readAllData(appRb, year, "tons"); - faf4.definePortsOfEntry(appRb); - if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) extractTruckData(faf4); - createZoneList(); - String truckTypeDefinition = "sut_mut"; - df.defineTruckTypes(truckTypeDefinition, appRb); - if (ResourceUtil.getBooleanProperty(appRb, "save.results.for.ohio", false)) ohio.summarizeFAFData(appRb, countyFips); - yumaSummary = ResourceUtil.getBooleanProperty(appRb, "save.results.for.yuma", false); - if (yumaSummary) yumaMPO.initializeVariables(appRb); - disaggregateFromFafToCounties(); - if (yumaSummary) yumaMPO.writeOutResults(appRb, year); - if (ResourceUtil.getBooleanProperty(appRb, "report.by.employment")) { - convertFromCommoditiesToTrucksByEmpType(); - } else { - convertFromCommoditiesToTrucks(); - } - if (ResourceUtil.getBooleanProperty(appRb, "analyze.commodity.groups", false)) { - readCommodityGrouping(); - convertFromCommoditiesToTrucksByComGroup(); - } - } - - private void extractTruckData(readFAF3 faf3) { - // extract truck data and write flows to file - logger.info("Extracting FAF3 truck data"); - String[] scaleTokens = ResourceUtil.getArray(appRb, "scaling.truck.trips.tokens"); - double[] scaleValues = ResourceUtil.getDoubleArray(appRb, "scaling.truck.trips.values"); - HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); - String truckFileNameT = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; - - // create output directory if it does not exist yet - File file = new File ("output/temp"); - if (!file.exists()) { - boolean outputDirectorySuccessfullyCreated = file.mkdir(); - if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); - } - - faf3.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); - } - - private void extractTruckData(ReadFAF4 faf4) { - // extract truck data and write flows to file - logger.info("Extracting FAF4 truck data"); - String[] scaleTokens = ResourceUtil.getArray(appRb, "scaling.truck.trips.tokens"); - double[] scaleValues = ResourceUtil.getDoubleArray(appRb, "scaling.truck.trips.values"); - HashMap scaler = fafUtils.createScalerHashMap(scaleTokens, scaleValues); - String truckFileNameT = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; - - // create output directory if it does not exist yet - File file = new File ("output/temp"); - if (!file.exists()) { - boolean outputDirectorySuccessfullyCreated = file.mkdir(); - if (!outputDirectorySuccessfullyCreated) logger.warn("Could not create scenario directory output/temp/"); - } - - faf4.writeFlowsByModeAndCommodity(truckFileNameT, ModesFAF.Truck, reportFormat.internat_domesticPart, scaler); - } - - private void createZoneList() { - // Create array with specialRegions that serve as port of entry/exit - - TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); - countyFips = fafUtils.createCountyFipsArray(poe.getColumnAsInt("pointOfEntry")); - countyFipsIndex = new int[fafUtils.getHighestVal(countyFips) + 1]; - for (int i = 0; i < countyFips.length; i++) { - countyFipsIndex[countyFips[i]] = i; - } - } - - - private void disaggregateFromFafToCounties() { - // disaggregates freight flows from FAF zoneArray to counties - - logger.info("Disaggregating FAF3 data from FAF zones to counties for year " + year + "."); - - cntFlows = new HashMap<>(); - - String[] commodities; - commodities = readFAF3.sctgStringCommodities; - int matrixSize = countyFips.length; - cntFlows = new HashMap<>(); - - float globalScale = (float) ResourceUtil.getDoubleProperty(appRb, "overall.scaling.factor.truck"); - - // regular method - for (String com: commodities) { - float[][] dummy = new float[matrixSize][matrixSize]; - cntFlows.put(com, dummy); - } - boolean keepTrackOfEmplType = ResourceUtil.getBooleanProperty(appRb, "report.by.employment", false); - df.prepareCountyDataForFAFwithDetailedEmployment(appRb, year, keepTrackOfEmplType); - - java.util.concurrent.ForkJoinPool pool = new java.util.concurrent.ForkJoinPool(); - DnCRecursiveAction action = new DissaggregateFafAction(globalScale); - pool.execute(action); - action.getResult(); - - if (ResourceUtil.getBooleanProperty(appRb, "summarize.by.ohio.rail.zones", false)) - ohio.sumFlowByRailZone(appRb, year, countyFips, countyFipsIndex, cntFlows); - } - - - private class DissaggregateFafAction extends DnCRecursiveAction { - private final float globalScale; - - private DissaggregateFafAction(float globalScale) { - super(0,readFAF3.sctgStringCommodities.length); - this.globalScale = globalScale; - } - - private DissaggregateFafAction(float globalScale, long start, long length, DnCRecursiveAction next) { - super(start,length,next); - this.globalScale = globalScale; - } - - @Override - protected void computeAction(long start, long length) { - long end = start + length; - for (int comm = (int) start; comm < end; comm++) { - int cm = readFAF3.sctgCommodities[comm]; - String fileName = ResourceUtil.getProperty(appRb, "temp.truck.flows.faf.zones") + "_" + year; - if (cm < 10) fileName = fileName + "_SCTG0" + cm + ".csv"; - else fileName = fileName + "_SCTG" + cm + ".csv"; - logger.info(" Working on " + fileName); - String sctg = readFAF3.getSCTGname(cm); - float[][] values = cntFlows.get(sctg); - TableDataSet tblFlows = fafUtils.importTable(fileName); - for (int row = 1; row <= tblFlows.getRowCount(); row++) { - float shortTons = tblFlows.getValueAt(row, "shortTons"); - if (shortTons == 0) continue; - String dir = tblFlows.getStringValueAt(row, "flowDirection"); - int orig = (int) tblFlows.getValueAt(row, "originFAF"); - int dest = (int) tblFlows.getValueAt(row, "destinationFAF"); - TableDataSet singleFlow; - if (dir.equals("import") || dir.equals("export")) { - TableDataSet poe; - if (dir.equals("import")) poe = readFAF3.getPortsOfEntry(orig); - else poe = readFAF3.getPortsOfEntry(dest); - singleFlow = df.disaggregateSingleFAFFlowThroughPOE(dir, poe, orig, dest, sctg, shortTons, 1); - } else singleFlow = df.disaggregateSingleFAFFlow(orig, dest, sctg, shortTons, 1); - for (int i = 1; i <= singleFlow.getRowCount(); i++) { - int oFips = (int) singleFlow.getValueAt(i, "oFips"); - int oZone = getCountyId(oFips); - int dFips = (int) singleFlow.getValueAt(i, "dFips"); - int dZone = getCountyId(dFips); - float thisFlow = singleFlow.getValueAt(i, "Tons") * globalScale; - values[oZone][dZone] += thisFlow; - if (yumaSummary) yumaMPO.saveForYuma(cm, oFips, dFips, thisFlow); - } - } - } - } - - @Override - protected DnCRecursiveAction getNextAction(long start, long length, DnCRecursiveAction next) { - return new DissaggregateFafAction(globalScale,start,length,next); - } - - @Override - protected boolean continueDividing(long length) { - return getSurplusQueuedTaskCount() < 3 && length > 1; - } - } - - - private int getCountyId(int fips) { - // Return region code of regName - return countyFipsIndex[fips]; - } - - - private void readCommodityGrouping () { - // Read how commodities are grouped by SCTG cagegory - - TableDataSet comGroups = fafUtils.importTable(appRb.getString("commodity.grouping")); - commodityGrouping = new int[fafUtils.getHighestVal(readFAF3.sctgCommodities) + 1]; - for (int row = 1; row <= comGroups.getRowCount(); row++) { - int sctg = (int) comGroups.getValueAt(row, "SCTG"); - int group = (int) comGroups.getValueAt(row, "Group"); - commodityGrouping[sctg] = group; - } - numCommodityGroups = 0; - for (int i = 0; i < commodityGrouping.length; i++) { - if (commodityGrouping[i] == 0) continue; - boolean alreadyExists = false; - for (int j = 0; j < i; j++) { - if (j == i) continue; - if (commodityGrouping[j] == commodityGrouping[i]) alreadyExists = true; - } - if (!alreadyExists) numCommodityGroups++; - } - } - - - private void convertFromCommoditiesToTrucks() { - // generate truck flows based on commodity flows - logger.info("Converting flows in tons into truck trips"); - float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); - float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); - double[][]sluTrucks = new double[countyFips.length][countyFips.length]; - double[][]mtuTrucks = new double[countyFips.length][countyFips.length]; - for (String com: readFAF3.sctgStringCommodities) { - double avPayload = fafUtils.findAveragePayload(com, "SCTG"); - double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; - double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; - float[][] tonFlows = cntFlows.get(com); - for (int i: countyFips) { - for (int j: countyFips) { - float dist = df.getCountyDistance(i, j); - if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. - int orig = getCountyId(i); - int dest = getCountyId(j); - if (tonFlows[orig][dest] == 0) continue; - double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); - // add empty trucks - trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); - trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); - // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor - trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); - trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); - sluTrucks[orig][dest] += trucksByType[0]; - mtuTrucks[orig][dest] += trucksByType[1]; - } - } - } - if (ResourceUtil.getBooleanProperty(appRb, "write.cnty.to.cnty.truck.trps", false)) - writeOutDisaggregatedTruckTrips(sluTrucks, mtuTrucks); - if (ResourceUtil.getBooleanProperty(appRb, "write.ii.ei.ee.trips", false)) - writeOutDisaggregatedTruckTripsByDirection(sluTrucks, mtuTrucks); - } - - - private void convertFromCommoditiesToTrucksByEmpType() { - // generate truck flows based on commodity flows, keeping track of employment type generating/attracting trucks - - logger.info("Converting flows in tons into truck trips, keeping track of employment types generating trucks"); - float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); - float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); - HashMap countyWeights = getCountyWeightsWithDetEmpl(); - - String[] emplCat = df.getEmpCats(); - - double[][][]sluTrucks = new double[emplCat.length][countyFips.length][countyFips.length]; - double[][][]mtuTrucks = new double[emplCat.length][countyFips.length][countyFips.length]; - - for (String com: readFAF3.sctgStringCommodities) { - double avPayload = fafUtils.findAveragePayload(com, "SCTG"); - double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; - double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; - float[][] tonFlows = cntFlows.get(com); - for (int i: countyFips) { - float[] weights = countyWeights.get(i); - for (int j: countyFips) { - float dist = df.getCountyDistance(i, j); - if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. - int orig = getCountyId(i); - int dest = getCountyId(j); - if (tonFlows[orig][dest] == 0) continue; - double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); - // add empty trucks - trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); - trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); - // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor - trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); - trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); - for (int eCat = 0; eCat < emplCat.length; eCat++) { - if (trucksByType[0] + trucksByType[1] == 0) continue; - try { - sluTrucks[eCat][orig][dest] += trucksByType[0] * weights[eCat]; - mtuTrucks[eCat][orig][dest] += trucksByType[1] * weights[eCat]; - } catch (Exception e){ - logger.warn(eCat+" "+orig+" "+dest+" "+i+" "+j+" "+com+": "+trucksByType[0]+trucksByType[1]); - } - } - } - } - } - writeOutDisaggregatedTruckTripsDetEmpl(sluTrucks, mtuTrucks); - } - - - private HashMap getCountyWeightsWithDetEmpl() { - // look up employment weight for each county - - HashMap weights = new HashMap<>(); - TableDataSet counties = fafUtils.importTable(ResourceUtil.getProperty(appRb, "county.ID")); - String[] emplCats = df.getEmpCats(); - for (int row = 1; row <= counties.getRowCount(); row++) { - int fips = (int) counties.getValueAt(row, "COUNTYFIPS"); - int faf = (int) counties.getValueAt(row, "FAF3region"); - if (faf == -99) continue; - for (String com: readFAF3.sctgStringCommodities) { - TableDataSet countyWeightsInOrigInFafZone = df.getCountyWeights("orig", faf, com); - for (int rw = 1; rw <= countyWeightsInOrigInFafZone.getRowCount(); rw++) { - if (countyWeightsInOrigInFafZone.getValueAt(rw, "COUNTYFIPS") == fips) { - float[] wght = new float[emplCats.length]; - for (int eCat = 0; eCat < emplCats.length; eCat++) { - wght[eCat] = countyWeightsInOrigInFafZone.getValueAt(rw, emplCats[eCat]); - } - float sum = fafUtils.getSum(wght); - // normalize weights from 0 - 1 - for (int eCat = 0; eCat < emplCats.length; eCat++) { - wght[eCat] = wght[eCat] / sum; - } - weights.put(fips, wght); - } - } - } - } - return weights; - } - - - private void writeOutDisaggregatedTruckTrips(double[][]sluTrucks, double[][]mtuTrucks) { - // write out disaggregated truck trips - - double[] truckProd = new double[fafUtils.getHighestVal(countyFips) + 1]; - - String fileName = appRb.getString("cnt.to.cnt.truck.flows") + year + ".csv"; - logger.info("Writing results to file " + fileName); - PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); - pw.println("OrigFips,DestFips,sut,mut"); - for (int i: countyFips) { - for (int j: countyFips) { - int orig = getCountyId(i); - int dest = getCountyId(j); - double slu = sluTrucks[orig][dest]; - double mtu = mtuTrucks[orig][dest]; - if (slu + mtu >= 0.00001) { - pw.format ("%d,%d,%.5f,%.5f", i, j, slu, mtu); - pw.println(); - truckProd[i] += slu + mtu; - } - } - } - pw.close(); - - if (ResourceUtil.getBooleanProperty(appRb, "report.truck.prod.by.county", false)) { - String fileNameProd = appRb.getString("truck.prod.by.county") + year + ".csv"; - PrintWriter pwp = fafUtils.openFileForSequentialWriting(fileNameProd); - pwp.println("CountyFips,TrucksGenerated"); - for (int i: countyFips) { - if (truckProd[i] > 0) pwp.println(i + "," + truckProd[i]); - } - pwp.close(); - } - } - - - private void writeOutDisaggregatedTruckTripsDetEmpl(double[][][]sluTrucks, double[][][]mtuTrucks) { - // write out disaggregated truck trips - - String fileName = appRb.getString("cnt.to.cnt.truck.flows") + "ByEmployment" + year + ".csv"; - logger.info("Writing results to file " + fileName); - PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); - pw.print("OrigFips,DestFips"); - String[] emplCats = df.getEmpCats(); - for (String emplCat : emplCats) { - if (emplCat.contains("Trade") || emplCat.contains("Financial") || - emplCat.contains("Education") || emplCat.contains("Other")) continue; - pw.print("," + emplCat + "_SUT," + emplCat + "_MUT"); - } - pw.println(); - for (int i: countyFips) { - for (int j: countyFips) { - int orig = getCountyId(i); - int dest = getCountyId(j); - pw.print(i + "," + j); - for (int eCat = 0; eCat < emplCats.length; eCat++) { - if (emplCats[eCat].contains("Trade") || emplCats[eCat].contains("Financial") || - emplCats[eCat].contains("Education") || emplCats[eCat].contains("Other")) continue; - double slu = sluTrucks[eCat][orig][dest]; - double mtu = mtuTrucks[eCat][orig][dest]; - pw.print("," + slu + "," + mtu); - } - pw.println(); - } - } - pw.close(); - } - - - private void writeOutDisaggregatedTruckTripsByDirection(double[][]sluTrucks, double[][]mtuTrucks) { - // write out disaggregated truck trips distinguishing II, EI/IE and EE trips - - TableDataSet counties = disaggregateFlows.countyIDsWithEmployment; - boolean[] relevantCounty = new boolean[fafUtils.getHighestVal(countyFips) + 1]; - String state = appRb.getString("ii.state"); - - for (int row = 1; row <= counties.getRowCount(); row++) { - relevantCounty[(int) counties.getValueAt(row, "COUNTYFIPS")] = counties.getStringValueAt(row, "StateCode").equals(state); - } - - String fileName = appRb.getString("cnt.to.cnt.truck.flows.by.dir") + year + ".csv"; - logger.info("Writing ii/ei/ie/ee flows to file " + fileName); - PrintWriter pw = fafUtils.openFileForSequentialWriting(fileName); - pw.println("OrigFips,DestFips,iiSut,iiMut,eiSut,eiMut,eeSut,eeMut,totSut,totMut"); - for (int i: countyFips) { - for (int j: countyFips) { - int orig = getCountyId(i); - int dest = getCountyId(j); - double slu = sluTrucks[orig][dest]; - double mtu = mtuTrucks[orig][dest]; - if (slu + mtu >= 0.00001) { - if (relevantCounty[i] && relevantCounty[j]) { - pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, slu, mtu, 0., 0., 0., 0., slu, mtu); - pw.println(); - } else if ((relevantCounty[i] && !relevantCounty[j]) || (!relevantCounty[i] && relevantCounty[j])) { - pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, 0., 0., slu, mtu, 0., 0., slu, mtu); - pw.println(); - } else { - pw.format ("%d,%d,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f,%.5f", i, j, 0., 0., 0., 0., slu, mtu, slu, mtu); - pw.println(); - } - } - } - } - pw.close(); - } - - - private void convertFromCommoditiesToTrucksByComGroup () { - // read flows in tons and convert into flows in trucks, distinguishing commodity groups - - float emptyTruckRate = (float) ResourceUtil.getDoubleProperty(appRb, "empty.truck.rate"); - float aawdtFactor = (float) ResourceUtil.getDoubleProperty(appRb, "AADT.to.AAWDT.factor"); - double[][][] sluTrucks = new double[numCommodityGroups+1][countyFips.length][countyFips.length]; - double[][][] mtuTrucks = new double[numCommodityGroups+1][countyFips.length][countyFips.length]; - for (String com: readFAF3.sctgStringCommodities) { - int iCom = Integer.parseInt(com.substring(4, 6)); // convert "SCTG00" into "00" - double avPayload = fafUtils.findAveragePayload(com, "SCTG"); - double sutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.SUT.payload") * avPayload; - double mutPL = ResourceUtil.getDoubleProperty(appRb, "multiplier.MUT.payload") * avPayload; - float[][] tonFlows = cntFlows.get(com); - for (int i: countyFips) { - for (int j: countyFips) { - int orig = getCountyId(i); - int dest = getCountyId(j); - if (tonFlows[orig][dest] == 0) continue; - float dist = df.getCountyDistance(i, j); - if (dist < 0) continue; // skip flows to Guam, Puerto Rico, Hawaii, Alaskan Islands etc. - double[] trucksByType = df.getTrucksByType(dist, sutPL, mutPL, tonFlows[orig][dest]); - // add empty trucks - trucksByType[0] += trucksByType[0] * (emptyTruckRate/100.0); - trucksByType[1] += trucksByType[1] * (emptyTruckRate/100.0); - // Annual cntFlows divided by 365.25 days plus AAWDT-over-AADT factor - trucksByType[0] = trucksByType[0] / 365.25f * (1 + (aawdtFactor / 100)); - trucksByType[1] = trucksByType[1] / 365.25f * (1 + (aawdtFactor / 100)); - sluTrucks[commodityGrouping[iCom]][orig][dest] += trucksByType[0]; - mtuTrucks[commodityGrouping[iCom]][orig][dest] += trucksByType[1]; - } - } - } - writeOutDisaggregatedTruckTrips(sluTrucks, mtuTrucks); - } - - - private void writeOutDisaggregatedTruckTrips(double sluTrucks[][][], double[][][] mutTrucks) { - // Write out truck trips by commodity group - - TableDataSet counties = disaggregateFlows.countyIDsWithEmployment; - boolean[] relevantCounty = new boolean[fafUtils.getHighestVal(countyFips) + 1]; - String state = appRb.getString("ii.state"); - double[][] prodByCounty = new double[countyFips.length][numCommodityGroups + 1]; - double[][] attrByCounty = new double[countyFips.length][numCommodityGroups + 1]; - - for (int row = 1; row <= counties.getRowCount(); row++) { - relevantCounty[(int) counties.getValueAt(row, "COUNTYFIPS")] = - counties.getStringValueAt(row, "StateCode").equals(state); - } - - boolean seperateEEflows = ResourceUtil.getBooleanProperty(appRb, "report.ext.flows.separately"); - String cgFile = appRb.getString("trucks.by.commodity.group") + year + ".csv"; - PrintWriter pwc = fafUtils.openFileForSequentialWriting(cgFile); - pwc.print("orig,dest"); - for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print(",com" + cg); - if (seperateEEflows) { - pwc.println(",external"); - } else { - pwc.println(); - } - for (int i = 0; i < countyFips.length; i++) { - for (int j = 0; j < countyFips.length; j++) { - - double sm = 0.; - for (int cg = 1; cg <= numCommodityGroups; cg++) sm += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; - - if (sm > 0) { - pwc.print(countyFips[i] + "," + countyFips[j]); - if (seperateEEflows) { - if (!relevantCounty[countyFips[i]] && !relevantCounty[countyFips[j]]) { - // E-E flows - for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print(",0"); - pwc.println("," + sm); - } else { - // I-I, I-E and E-I flows - for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print("," + - (sluTrucks[cg][i][j] + mutTrucks[cg][i][j])); - pwc.println(",0"); - } - } else { - // I-I, I-E, E-I and E-E flows - for (int cg = 1; cg <= numCommodityGroups; cg++) pwc.print("," + - (sluTrucks[cg][i][j] + mutTrucks[cg][i][j])); - pwc.println(); - } - for (int cg = 1; cg <= numCommodityGroups; cg++) { - prodByCounty[i][cg] += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; - attrByCounty[j][cg] += sluTrucks[cg][i][j] + mutTrucks[cg][i][j]; - } - } - } - } - pwc.close(); - - String cgFileAgg = appRb.getString("trucks.by.commodity.group") + year + "_prodAttr.csv"; - PrintWriter pwca = fafUtils.openFileForSequentialWriting(cgFileAgg); - pwca.print("countyFips"); - for (int cg = 1; cg <= numCommodityGroups; cg++) pwca.print(",prod_" + cg + ",attr_" + cg); - pwca.println(); - for (int i = 0; i < countyFips.length; i++) { - float sm = 0; - for (int cg = 1; cg <= numCommodityGroups; cg++) { - sm += prodByCounty[i][cg] + attrByCounty[i][cg]; - } - if (sm > 0) { - pwca.print(countyFips[i]); - for (int cg = 1; cg <= numCommodityGroups; cg++) pwca.print("," + prodByCounty[i][cg] + "," + - attrByCounty[i][cg]); - pwca.println(); - } - } - pwca.close(); - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java deleted file mode 100644 index 004554c..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateAndAggregateFlows.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.sandag.htm.processFAF; - -import com.pb.common.datafile.TableDataSet; - -import java.util.HashMap; -import java.util.ResourceBundle; - -/** - * This class disaggregates and aggregates FAF flows from FAF zones to model zones (which may be smaller or larger than FAF zones) - * User: Rolf Moeckel, PB Albuquerque - * Date: November 17, 2011 (Santa Fe, NM) - */ - -// todo: Started this class for NCSTM, but then realized that FAF zones do not nest in RMZ of NCSTM. Some FAF zones -// todo: would have to be split in parts and aggregated in other parts. -// todo: May be useful in other projects. - -public class disaggregateAndAggregateFlows { - - private ResourceBundle appRb; - private HashMap zonesToDisaggregate; - private int[] zonesToAggregate; - - public disaggregateAndAggregateFlows(ResourceBundle appRb) { - this.appRb = appRb; - } - - - public void defineZonesToAggregate(String fileName) { - // define which FAF zones need to be aggregated to larger model zones - - TableDataSet fafZones = fafUtils.importTable(fileName); - int highestFAF = -1; - for (int row = 1; row <= fafZones.getRowCount(); row++) { - if (fafZones.getValueAt(row, "modelZone") != -1) highestFAF = (int) Math.max(highestFAF, fafZones.getValueAt(row, "modelZone")); - } - zonesToAggregate = new int[highestFAF + 1]; - for (int row = 1; row <= fafZones.getRowCount(); row++) { - if (fafZones.getValueAt(row, "modelZone") == -1) { - zonesToAggregate[(int) fafZones.getValueAt(row, "ZoneID")] = -1; - } else { - zonesToAggregate[(int) fafZones.getValueAt(row, "ZoneID")] = (int) fafZones.getValueAt(row, "modelZone"); - } - } - } - - - public void defineZonesToDisaggregate(TableDataSet zoneSystem) { - // define which FAF zones need to be disaggregated to smaller model zones - - zonesToDisaggregate = new HashMap<>(); - for (int row = 1; row <= zoneSystem.getRowCount(); row++) { - int taz = (int) zoneSystem.getValueAt(row, "TAZ"); - int faf = (int) zoneSystem.getValueAt(row, "FAFzone"); - if (!zoneSystem.getBooleanValueAt(row, "disaggregate")) continue; - if (zonesToDisaggregate.containsKey(faf)) { - int[] zones = zonesToDisaggregate.get(faf); - int[] zonesNew = new int[zones.length + 1]; - System.arraycopy(zones, 0, zonesNew, 0, zones.length); - zonesNew[zones.length] = taz; - zonesToDisaggregate.put(faf, zonesNew); - } else { - zonesToDisaggregate.put(faf, new int[]{taz}); - } - } - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java deleted file mode 100644 index 5045b90..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/disaggregateFlows.java +++ /dev/null @@ -1,1357 +0,0 @@ -package org.sandag.htm.processFAF; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; -import com.pb.common.matrix.Matrix; -import com.pb.common.matrix.MatrixReader; -import java.util.HashMap; -import java.util.ResourceBundle; -import java.io.File; -import org.apache.log4j.Logger; - - -/** - * This class disaggregates FAF flows from FAF zones to counties - * User: Rolf Moeckel - * Date: May 6, 2009 - * - * Updated 2017-12-27 JEF - */ - -public class disaggregateFlows { - - static Logger logger = Logger.getLogger(disaggregateFlows.class); - public static TableDataSet countyIDsWithEmployment; - private HashMap countyShares; - private TableDataSet truckTypeShares; - private int[] truckTypeShareDistBin; - private float[] truckTypeShareMT; - private Matrix distCounties; - private static int[] countyFips; - private int FAFVersion=4; - - - private static String[] empCats = {"Agriculture", "Construction Natural Resources and Mining", "Manufacturing", - "Trade Transportation and Utilities", "Information", "Financial Activities", - "Professional and Business Services", "Education and Health Services", "Leisure and Hospitality", - "Other Services", "coalProduction"}; //, "Government"}; (no government employment available at county-level nationwide at this point - - - public void prepareCountyData(ResourceBundle rb) { - // prepare county data to provide total employment as a weight - logger.info("Preparing county data for flow disaggregation"); - - getUScountyEmployment(rb); - if (readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); - //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share - countyShares = new HashMap<>(); - for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { - for (String com: readFAF2.sctgCommodities) { - TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, null, null); - // dir is not really needed here, but kept for consistency reason as it is required - // in prepareCountyDataForFAF(ResourceBundle rb, TableDataSet detailEmployment) - String codeo = "orig_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; - countyShares.put(codeo, CountyList); - String coded = "dest_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; - countyShares.put(coded, CountyList); - } - } - } - - - public void prepareCountyData(ResourceBundle rb, int yr, TableDataSet detailEmployment, String truckType) { - // prepare county data to provide detailed employment given in detailEmployment and total employment elsewhere - // as a weight - logger.info("Preparing county data for flow disaggregation for year " + yr + "..."); - - HashMap useC = createMakeUseHashMap(rb, "use.coefficients"); - HashMap makeC = createMakeUseHashMap(rb, "make.coefficients"); - - truckTypeShares = fafUtils.importTable(rb.getString("truck.type.by.distance")); - if (truckType.equalsIgnoreCase("weight")) createTruckShareArraysWeight(); - else createTruckShareArraysUnit(); - distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); - - getUScountyEmployment(rb); - - if (readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); - //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share - countyShares = new HashMap<>(); - String[] direction = {"orig", "dest"}; - for (String dir: direction) { - HashMap factors; - if (dir.equals("orig")) factors = makeC; - else factors = useC; - for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { - for (String com: readFAF2.sctgCommodities) { - TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, detailEmployment, factors); - String code = dir + "_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; - countyShares.put(code, CountyList); - } - } - } - } - - - private void createTruckShareArraysWeight() { - // create array with distance bins and shares or medium-heavy trucks - truckTypeShareDistBin = new int[truckTypeShares.getRowCount()]; - truckTypeShareMT = new float[truckTypeShares.getRowCount()]; - for (int row = 1; row <= truckTypeShares.getRowCount(); row++) { - truckTypeShareDistBin[row - 1] = (int) truckTypeShares.getValueAt(row, "DistanceGreaterThan"); - truckTypeShareMT[row - 1] = truckTypeShares.getValueAt(row, "MT(<26k_lbs)"); - // data quality check - float htShare = truckTypeShares.getValueAt(row, "HT(>26k_lbs)"); - if (htShare + truckTypeShareMT[row - 1] != 1) logger.warn("Shares of Medium and Heavy Trucks add up to " + - htShare + truckTypeShareMT[row - 1] + " instead of 1 for distance class " + truckTypeShareDistBin[row - 1]); - } - } - - - private void createTruckShareArraysUnit() { - // create array with distance bins and shares or Single-Unit/Multi-Unit trucks - truckTypeShareDistBin = new int[truckTypeShares.getRowCount()]; - truckTypeShareMT = new float[truckTypeShares.getRowCount()]; - for (int row = 1; row <= truckTypeShares.getRowCount(); row++) { - truckTypeShareDistBin[row - 1] = (int) truckTypeShares.getValueAt(row, "DistanceGreaterThan"); - truckTypeShareMT[row - 1] = truckTypeShares.getValueAt(row, "SUT"); - // data quality check - float htShare = truckTypeShares.getValueAt(row, "MUT"); - if (htShare + truckTypeShareMT[row - 1] != 1) logger.warn("Shares of SUT and MUT Trucks add up to " + - htShare + truckTypeShareMT[row - 1] + " instead of 1 for distance class " + truckTypeShareDistBin[row - 1]); - } - } - - - public void prepareCountyDataForFAF(ResourceBundle rb, int yr, TableDataSet detailEmployment, - TableDataSet specialRegions, int fafVersion) { - // prepare county data to provide detailed employment given in detailEmployment and total employment elsewhere - // as a weight using the "fafVersion" zone system - logger.info("Preparing county data for FAF" + fafVersion + " flow disaggregation for year " + yr + "..."); - - String useToken = "faf" + fafVersion + ".use.coefficients"; - String makeToken = "faf" + fafVersion + ".make.coefficients"; - HashMap useC = createMakeUseHashMap(rb, useToken); - HashMap makeC = createMakeUseHashMap(rb, makeToken); - - distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); - - if (fafVersion == 2 && readFAF2.domRegionList == null) readFAF2.readFAF2ReferenceLists(rb); -// if (fafVersion == 3 && readFAF3.fafRegionList == null) readFAF3.readFAF3referenceLists(rb); // has been read earlier for FAF3 - - //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share - countyShares = new HashMap<>(); - String[] direction = {"orig", "dest"}; - for (String dir: direction) { - HashMap factors; - if (dir.equals("orig")) factors = makeC; - else factors = useC; - - if (fafVersion == 2) { - for (int fafNum = 1; fafNum <= readFAF2.domRegionList.getRowCount(); fafNum++) { - for (String com: readFAF2.sctgCommodities) { - TableDataSet CountyList = getCountySpecificDataByFAF2(fafNum, com, detailEmployment, factors); - String code = dir + "_" + readFAF2.domRegionList.getStringValueAt(fafNum, "RegionName") + "_" + com; - countyShares.put(code, CountyList); - } - } - } else if (fafVersion==3){ // fafVersion == 3 - for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { - for (String com: readFAF3.sctgStringCommodities) { - int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); - TableDataSet CountyList = getCountySpecificDataByFAF(zoneNum, com, detailEmployment, factors); - String code = dir + "_" + zoneNum + "_" + com; - countyShares.put(code, CountyList); - } - } - } - else{ // fafVersion == 4 - for (int fafNum = 1; fafNum <= ReadFAF4.fafRegionList.getRowCount(); fafNum++) { - for (String com: ReadFAF4.sctgStringCommodities) { - int zoneNum = (int) ReadFAF4.fafRegionList.getValueAt(fafNum, "ZoneID"); - TableDataSet CountyList = getCountySpecificDataByFAF(zoneNum, com, detailEmployment, factors); - String code = dir + "_" + zoneNum + "_" + com; - countyShares.put(code, CountyList); - } - } - } - } - - // create fake county lists for special regions such as airports or seaports that shall be kept separate from the FAF regions - if (specialRegions == null) return; - for (int row = 1; row <= specialRegions.getRowCount(); row++) { - int[] fips = {(int) specialRegions.getValueAt(row, "modelCode")}; - String[] name; - if (fafVersion == 2) { - name = new String[]{specialRegions.getStringValueAt(row, "Region")}; - }else if(fafVersion==3){ - name = new String[]{specialRegions.getStringValueAt(row, "faf3code")}; - }else{ - name = new String[]{specialRegions.getStringValueAt(row, "faf4code")}; - } - - TableDataSet CountyList = new TableDataSet(); - float[] emplDummy = {1}; - CountyList.appendColumn(name, "Name"); - CountyList.appendColumn(fips, "COUNTYFIPS"); - CountyList.appendColumn(name, "FAFRegion"); - CountyList.appendColumn(emplDummy, "Employment"); - for (String dir: direction) { - if (fafVersion == 2) { - for (String com: readFAF2.sctgCommodities) { - String code = dir + "_" + name[0] + "_" + com; - countyShares.put(code, CountyList); - } - } else if (fafVersion == 3){ - for (String com: readFAF3.sctgStringCommodities) { - String code = dir + "_" + fips[0] + "_" + com; - countyShares.put(code, CountyList); - } - }else{ - for (String com: ReadFAF4.sctgStringCommodities) { - String code = dir + "_" + fips[0] + "_" + com; - countyShares.put(code, CountyList); - } - - } - - } - } - } - - - public static HashMap createMakeUseHashMap (ResourceBundle rb, String token) { - // create HashMap with make/use coefficients - TableDataSet coeff = fafUtils.importTable(ResourceUtil.getProperty(rb, token)); - HashMap hsm = new HashMap<>(); - for (int i = 1; i <= coeff.getRowCount(); i++) { - String industry = coeff.getStringValueAt(i, "Industry"); - for (int j = 2; j <= coeff.getColumnCount(); j++) { - String sctg = coeff.getColumnLabel(j); - String code = industry + "_" + sctg; - hsm.put(code, coeff.getValueAt(i, j)); - } - } - return hsm; - } - - - public TableDataSet disaggregateSingleFAF2flow(String orig, String dest, String sctg, float tons) { - // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF2 - - TableDataSet flows = new TableDataSet(); - // get county specific data for origin FAF and destination FAF - TableDataSet CountyDatI = getCountyTableDataSet("orig", orig, sctg); - TableDataSet CountyDatJ = getCountyTableDataSet("dest", dest, sctg); - - // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf - double EmplICplusJCtotal = 0; - int count = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - EmplICplusJCtotal += CountyDatI.getValueAt(origCounty, "Employment") + - CountyDatJ.getValueAt(destCounty, "Employment"); - count++; - } - } - int[] oFips = new int[count]; - int[] dFips = new int[count]; - String[] codes = new String[count]; - float[] tonShare = new float[count]; - int k = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); - double EmplICplusJC = CountyDatI.getValueAt(origCounty, "Employment") + - CountyDatJ.getValueAt(destCounty, "Employment"); - tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); - oFips[k] = origFips; - dFips[k] = destFips; - codes[k] = String.valueOf(origFips) + "_" + String.valueOf(destFips); - k++; - } - } - flows.appendColumn(oFips, "oFips"); - flows.appendColumn(dFips, "dFips"); - flows.appendColumn(codes, "Codes"); - flows.appendColumn(tonShare, "Tons"); - return flows; - } - - - public TableDataSet disaggregateSingleFAFFlow(int orig, int dest, String sctg, float tons, float centerDamper) { - // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF3 - - TableDataSet flows = new TableDataSet(); - // get county specific data for origin FAF and destination FAF - - TableDataSet CountyDatI = getCountyWeights("orig", orig, sctg); - TableDataSet CountyDatJ = getCountyWeights("dest", dest, sctg); - if (CountyDatI == null) { - logger.error("Could not find table for disaggregating county " + orig + " with commodity " + sctg); - return null; - } - if (CountyDatJ == null) { - logger.error("Could not find table for disaggregating county " + dest + " with commodity " + sctg); - return null; - } - // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf - double EmplICplusJCtotal = 0; - int count = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - EmplICplusJCtotal += Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * - CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); - count++; - } - } - int[] oFips = new int[count]; - int[] dFips = new int[count]; - String[] codes = new String[count]; - float[] tonShare = new float[count]; - int k = 0; - - if (EmplICplusJCtotal == 0) logger.error("Could not find weight for FAF zone " + orig + " to " + dest + " for " + sctg); - - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); - double EmplICplusJC = Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * - CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); - tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); - - oFips[k] = origFips; - dFips[k] = destFips; - codes[k] = origFips + "_" + destFips; - k++; - } - } - - flows.appendColumn(oFips, "oFips"); - flows.appendColumn(dFips, "dFips"); - flows.appendColumn(codes, "Codes"); - flows.appendColumn(tonShare, "Tons"); - return flows; - } - - - public TableDataSet disaggregateSingleFAFFlowThroughPOE(String dir, TableDataSet poe, int orig, int dest, String sctg, - float tons, float centerDamper) { - // disaggregate tons from orig FAFzone to dest FAFzone to the county level for FAF3 for import/exports through ports - // of entry/exit - - // get county specific data for origin FAF and destination FAF - TableDataSet CountyDatI = getCountyWeights("orig", orig, sctg); - if (dir.startsWith("import") && poe != null) CountyDatI = poe; - TableDataSet CountyDatJ = getCountyWeights("dest", dest, sctg); - if (dir.startsWith("export") && poe != null) CountyDatJ = poe; - if (CountyDatI == null) { - logger.error("Could not find table for " + dir + " disaggregating county " + orig + " with commodity " + sctg); - return null; - } - if (CountyDatJ == null) { - logger.error("Could not find table for " + dir + " disaggregating county " + dest + " with commodity " + sctg); - return null; - } - // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf - double EmplICplusJCtotal = 0; - int count = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - EmplICplusJCtotal += Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * - CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); - count++; - } - } - int[] oFips = new int[count]; - int[] dFips = new int[count]; - String[] codes = new String[count]; - float[] tonShare = new float[count]; - int k = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); - double EmplICplusJC = Math.pow(CountyDatI.getValueAt(origCounty, "Employment") * - CountyDatJ.getValueAt(destCounty, "Employment"), centerDamper); - tonShare[k] = (float) (tons * EmplICplusJC / EmplICplusJCtotal); - oFips[k] = origFips; - dFips[k] = destFips; - codes[k] = origFips + "_" + destFips; - k++; - } - } - TableDataSet flows = new TableDataSet(); - flows.appendColumn(oFips, "oFips"); - flows.appendColumn(dFips, "dFips"); - flows.appendColumn(codes, "Codes"); - flows.appendColumn(tonShare, "Tons"); - return flows; - } - - - public float getCountyDistance(int origFips, int destFips) { - // return distance from origFips to destFips - - try { - return distCounties.getValueAt(origFips, destFips); - } catch (Exception e){ - return -1; - } - } - - - public float[] splitToTruckTypes(int origFips, int destFips, float trucks) { - // split trucks for this OD pair into two truck types - float dist; - try { - dist = distCounties.getValueAt(origFips, destFips); - } catch (Exception e) { - dist = 999; - } - float[] splitTrucks = {0, 0}; - for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) { - if (dist >= truckTypeShareDistBin[dClass]) { - splitTrucks[0] = trucks * truckTypeShareMT[dClass]; - splitTrucks[1] = trucks * (1 - truckTypeShareMT[dClass]); - return splitTrucks; } - } - logger.warn("Could not find truck share for distance " + dist); - return null; - } - - - public double[] splitToTruckTypes(int origFips, int destFips, double trucks) { - // split trucks for this OD pair into two truck types - int dist; - try { - dist = (int) distCounties.getValueAt(origFips, destFips); - if (dist == -999) dist = 9999; // destination unreachable on network, therefore skim is negative, but there might be trips in reality, including ferry - } catch (Exception e) { - dist = 999; - } - double[] splitTrucks = {0, 0}; - for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) - if (dist >= truckTypeShareDistBin[dClass]) { - splitTrucks[0] = trucks * truckTypeShareMT[dClass]; - splitTrucks[1] = trucks * (1 - truckTypeShareMT[dClass]); - return splitTrucks; - } - logger.warn("Could not find truck share for distance " + dist); - return null; - } - - - public double[] splitToTruckTypes(int origFips, int destFips) { - // split trucks for this OD pair into two truck types - int dist; - try { - dist = (int) distCounties.getValueAt(origFips, destFips); - if (dist == -999) dist = 9999; // destination unreachable on network, therefore skim is negative, but there might be trips in reality, including ferry - } catch (Exception e) { - dist = 999; - } - double[] splitTrucks = {0, 0}; - for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) - if (dist >= truckTypeShareDistBin[dClass]) { - splitTrucks[0] = truckTypeShareMT[dClass]; - splitTrucks[1] = (1 - truckTypeShareMT[dClass]); - return splitTrucks; - } - logger.warn("Could not find truck share for distance " + dist); - return null; - } - - - public double[] getTrucksByType(float dist, double sutPL, double mutPL, double tonFlow) { - // convert tons into trucks and split trucks into single-unit and multi-unit trucks - - double[] truckShares = splitToTruckTypes(dist); - double[] trucksByType = new double[2]; - trucksByType[0] = tonFlow / (sutPL + truckShares[1]/truckShares[0] * mutPL); - trucksByType[1] = tonFlow / (truckShares[0]/truckShares[1] * sutPL + mutPL); - return trucksByType; - } - - - public double[] getTrucksByType(float dist, double sutPL, double mutPL, double tonFlow, float adjustmentSUT) { - // convert tons into trucks and split trucks into single-unit and multi-unit trucks - - double[] truckShares = splitToTruckTypes(dist); - truckShares[0] = truckShares[0] * (1. + adjustmentSUT); - truckShares[1] = 1. - truckShares[0]; - double[] trucksByType = new double[2]; - trucksByType[0] = tonFlow / (sutPL + truckShares[1]/truckShares[0] * mutPL); - trucksByType[1] = tonFlow / (truckShares[0]/truckShares[1] * sutPL + mutPL); - return trucksByType; - } - - - public double[] splitToTruckTypes(float distance) { - // split trucks for this OD pair into two truck types - double[] splitTrucks = {0, 0}; - for (int dClass = truckTypeShareDistBin.length - 1; dClass >= 0; dClass--) - if (distance >= truckTypeShareDistBin[dClass]) { - splitTrucks[0] = truckTypeShareMT[dClass]; - splitTrucks[1] = (1 - truckTypeShareMT[dClass]); - return splitTrucks; - } - logger.warn("Could not find truck share for distance " + distance); - return null; - } - - - public float[] splitToTruckTypes(String code, float trucks) { - // split trucks for this OD pair into two truck types - String[] origDestFips = code.split("_"); - int origFips = Integer.parseInt(origDestFips[0]); - int destFips = Integer.parseInt(origDestFips[1]); - float dist; - try { - dist = distCounties.getValueAt(origFips, destFips); - } catch (Exception e) { - dist = 1; - } - int distClass = 1; - for (int row = 1; row <= truckTypeShares.getRowCount(); row++) - if (dist > truckTypeShares.getValueAt(row, "DistanceGreaterThan")) distClass = row; - float[] splitTrucks = {0, 0}; - splitTrucks[0] = trucks * truckTypeShares.getValueAt(distClass, "MT(<26k_lbs)"); - splitTrucks[1] = trucks * truckTypeShares.getValueAt(distClass, "HT(>26k_lbs)"); - return splitTrucks; - } - - - private TableDataSet getCountyTableDataSet(String direction, String FAFregion, String sctgCode) { - // check if this FAFregion equals a special generator - TableDataSet tbl; - if (FAFregion.equals("OR Portl_Airport") || FAFregion.equals("OR Portl_Port") || - FAFregion.equals("OR rem_Airport") || FAFregion.equals("OR rem_Port") || FAFregion.equals("Other")) { - tbl = new TableDataSet(); - String[] a = {FAFregion}; - float[] b = new float[1]; - if (FAFregion.equals("OR Portl_Airport")) b[0] = 90001f; - else if (FAFregion.equals("OR Portl_Port")) b[0] = 90002f; - else if (FAFregion.equals("OR rem_Airport")) b[0] = 90003f; - else if (FAFregion.equals("OR rem_Port")) b[0] = 90004f; - else if (FAFregion.equals("Other")) b[0] = 90005f; - float[] c = {1f}; - tbl.appendColumn(a, "Name"); - tbl.appendColumn(b, "COUNTYFIPS"); - tbl.appendColumn(a, "FAFRegion"); - tbl.appendColumn(c, "Employment"); - tbl.setName(FAFregion); - } else { - String code = direction + "_" + FAFregion + "_" + sctgCode; - tbl = countyShares.get(code); - } - return tbl; - } - - - public TableDataSet getCountyWeights(String direction, int FAFregion, String sctgCode) { - // check if this FAFregion equals a special generator - String code = direction + "_" + FAFregion + "_" + sctgCode; - return countyShares.get(code); - } - - - public TableDataSet disaggregateSingleFlowInOregon(String orig, String dest, String sctg, float trucks) { - // disaggregate trucks from orig FAFzone to dest FAFzone to the county level - - if (!orig.contains("OR Portl") && !orig.contains("OR rem")) orig = "Other"; - if (!dest.contains("OR Portl") && !dest.contains("OR rem")) dest = "Other"; - TableDataSet flows = new TableDataSet(); - // get county specific data for origin FAF and destination FAF - TableDataSet CountyDatI = getCountyTableDataSet("orig", orig, sctg); - TableDataSet CountyDatJ = getCountyTableDataSet("dest", dest, sctg); - - // walk through every county combination ic/jc within current FAF Region combination origFaf/destFaf - double EmplICplusJCtotal = 0; - int count = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - EmplICplusJCtotal += CountyDatI.getValueAt(origCounty, "Employment") + - CountyDatJ.getValueAt(destCounty, "Employment"); - count++; - } - } - String[] codes = new String[count]; - float[] tonShare = new float[count]; - int k = 0; - for (int origCounty = 1; origCounty <= CountyDatI.getRowCount(); origCounty++) { - int origFips = (int) CountyDatI.getValueAt(origCounty, "COUNTYFIPS"); - for (int destCounty = 1; destCounty <= CountyDatJ.getRowCount(); destCounty++) { - int destFips = (int) CountyDatJ.getValueAt(destCounty, "COUNTYFIPS"); - double EmplICplusJC = CountyDatI.getValueAt(origCounty, "Employment") + - CountyDatJ.getValueAt(destCounty, "Employment"); - tonShare[k] = (float) (trucks * EmplICplusJC / EmplICplusJCtotal); - codes[k] = String.valueOf(origFips) + "_" + String.valueOf(destFips); - k++; - } - } - flows.appendColumn(codes, "Codes"); - flows.appendColumn(tonShare, "ShortTons"); - return flows; - } - - - public static void getUScountyEmployment(ResourceBundle rb) { - // read employment and county id data - - logger.info("Reading County Employment Data..."); - countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); - - TableDataSet StatesTable = fafUtils.importTable(ResourceUtil.getProperty(rb, "state.list")); - String[] StateNames = StatesTable.getColumnAsString("StateName"); - TableDataSet[] StateEmployment = new TableDataSet[StateNames.length]; - for (int st = 0; st < StateNames.length; st++) { - String fileName = ResourceUtil.getProperty(rb, "state.employment.prefix") + StateNames[st] + ".csv"; - StateEmployment[st] = fafUtils.importTable(fileName); - StateEmployment[st].setName(StateNames[st]); - } - - String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; - int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; - countyIDsWithEmployment.appendColumn(tempString, "stateName"); - countyIDsWithEmployment.appendColumn(tempInt, "Employment"); - - // assign employment to every county - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; - String CountyStateAbbreviation = countyIDsWithEmployment.getStringValueAt(i, "StateCode"); - String CountyName = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - - // find full state name - String StateNameOfCounty = " "; - for (int j = 1; j <= StatesTable.getRowCount(); j++) - if (StatesTable.getStringValueAt(j, "StateAbbreviation").equals(CountyStateAbbreviation)) - StateNameOfCounty = StatesTable.getStringValueAt(j, "stateName"); - - // find right employment table for this state - int StateNumber = 0; - for (int j = 0; j < StateEmployment.length; j++) { - if (StateEmployment[j].getName().equals(StateNameOfCounty)) StateNumber = j; - } - - // delete state code from county name - StringBuffer sb = new StringBuffer(); - sb.append(CountyName); - int n = sb.lastIndexOf(CountyStateAbbreviation); - sb.delete(n-1, n+2); - // add ", StateAbbreviation" - sb.append(", "); - sb.append(CountyStateAbbreviation); - CountyName = sb.toString(); - - // find employment of current county - boolean found = false; - for (int k = 1; k <= StateEmployment[StateNumber].getRowCount(); k++) { - if (StateEmployment[StateNumber].getStringValueAt(k, "Region").equalsIgnoreCase(CountyName)) { - countyIDsWithEmployment.setValueAt - (i, "Employment", StateEmployment[StateNumber].getValueAt(k, "Employment")); - countyIDsWithEmployment.setStringValueAt - (i, countyIDsWithEmployment.getColumnPosition("stateName"), StateNameOfCounty); - found = true; - } - } - - // County Broomfield, Colorado has no polygon in the county layer. Employment is added to Boulder County: - if (CountyName.equals("BOULDER, CO")) { - for (int k = 1; k <= StateEmployment[StateNumber].getRowCount(); k++) { - if (StateEmployment[StateNumber].getStringValueAt(k, "Region").equals("Broomfield, CO")) { - float BoulderBroomfieldEmpl = countyIDsWithEmployment.getValueAt(i, "Employment") + - StateEmployment[StateNumber].getValueAt(k, "Employment"); - countyIDsWithEmployment.setValueAt(i, "Employment", BoulderBroomfieldEmpl); - } - } - } - - // write error message (unless it is an island of UM or Guam) - if (!found && !CountyStateAbbreviation.equals("UM") && !CountyStateAbbreviation.equals("GU")) - logger.warn("Not found: " + CountyName + " (" + - countyIDsWithEmployment.getStringValueAt(i, "TYPE") + ")"); - } - } - - - public static void getUScountyEmploymentFromOneFile (ResourceBundle rb) { - // read file with county employment in the US - - logger.info("Reading County Employment Data..."); - countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); - TableDataSet countyEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "us.county.employment")); - - String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; - int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; - countyIDsWithEmployment.appendColumn(tempString, "stateName"); - countyIDsWithEmployment.appendColumn(tempInt, "Employment"); - - // assign employment to every county - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; - // skip Guam and United States Minor Outlying Islands - if (countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("GU") || - countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("UM")) continue; - float fips = countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - // find employment of current county - boolean found = false; - for (int row = 1; row <= countyEmployment.getRowCount(); row++) { - if (countyEmployment.getValueAt(row, "fips") == fips) { - countyIDsWithEmployment.setValueAt - (i, "Employment", countyEmployment.getValueAt(row, "totalAnnualAverageEmployment")); - found = true; - } - } - - // write error message if county was not found - if (!found) { - String CountyName = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - logger.warn("Not found: " + CountyName + " (" + - countyIDsWithEmployment.getStringValueAt(i, "TYPE") + ")"); - } - } - } - - - public void getUScountyEmploymentByIndustry (ResourceBundle rb) { - // read file with employment by county by industry - - logger.info(" Reading County Employment Data..."); - countyIDsWithEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "county.ID")); - TableDataSet countyEmployment = fafUtils.importTable(ResourceUtil.getProperty(rb, "us.county.employment.by.ind")); - countyEmployment.buildIndex(countyEmployment.getColumnPosition("FIPS")); - TableDataSet agEmpl = fafUtils.importTable(rb.getString("us.county.employment.agricult")); - agEmpl.buildIndex(agEmpl.getColumnPosition("FIPS")); - - String[] tempString = new String[countyIDsWithEmployment.getRowCount()]; - int[] tempInt = new int[countyIDsWithEmployment.getRowCount()]; - countyIDsWithEmployment.appendColumn(tempString, "stateName"); - for (String emp: empCats) countyIDsWithEmployment.appendColumn(tempInt, emp); - - // assign employment to every county - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (!countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("County") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Legal County Equivalent")&& - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Independent City") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Census Area") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("City and Borough") && - !countyIDsWithEmployment.getStringValueAt(i, "TYPE").equals("Parish")) continue; - // skip Guam and United States Minor Outlying Islands - if (countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("GU") || - countyIDsWithEmployment.getStringValueAt(i, "StateCode").equals("UM")) continue; - int fips = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - // find employment of current county - float consNatResMinEmployment = countyEmployment.getIndexedValueAt(fips, "Natural Resources and Mining") + - countyEmployment.getIndexedValueAt(fips, "Construction"); - countyIDsWithEmployment.setValueAt(i, "Construction Natural Resources and Mining", consNatResMinEmployment); - countyIDsWithEmployment.setValueAt(i, "Manufacturing", countyEmployment.getIndexedValueAt(fips, "Manufacturing")); - countyIDsWithEmployment.setValueAt(i, "Trade Transportation and Utilities", countyEmployment.getIndexedValueAt(fips, "Trade, Transportation, and Utilities")); - countyIDsWithEmployment.setValueAt(i, "Information", countyEmployment.getIndexedValueAt(fips, "Information")); - countyIDsWithEmployment.setValueAt(i, "Financial Activities", countyEmployment.getIndexedValueAt(fips, "Financial Activities")); - countyIDsWithEmployment.setValueAt(i, "Professional and Business Services", countyEmployment.getIndexedValueAt(fips, "Professional and Business Services")); - countyIDsWithEmployment.setValueAt(i, "Education and Health Services", countyEmployment.getIndexedValueAt(fips, "Education and Health Services")); - countyIDsWithEmployment.setValueAt(i, "Leisure and Hospitality", countyEmployment.getIndexedValueAt(fips, "Leisure and Hospitality")); - countyIDsWithEmployment.setValueAt(i, "Other Services", countyEmployment.getIndexedValueAt(fips, "Other Services")); - - // agricultural employment is missing in most of Alaska and in independent cities (deemed not to be relevant) - try { - countyIDsWithEmployment.setValueAt(i, "Agriculture", agEmpl.getIndexedValueAt(fips, "agEmployment")); - } catch (Exception e) { - // Set to minor value to ensure that all FAF ag production can be allocated. If the FAF zone of this - // county has other counties with ag production, this miniWeight will be irrelevant as it is small. - // If no county in this FAF zone has ag production, the employment "Construction Natural Resources and Mining" - // will be used to disaggregate flows (with all counties being multiplied by 0.0001, which doesn't affect the result. - float miniWeight = 0.0001f * countyIDsWithEmployment.getValueAt(i, "Construction Natural Resources and Mining"); - countyIDsWithEmployment.setValueAt(i, "Agriculture", miniWeight); - - } - } - // try to add coal production as a weight - try { - TableDataSet coalProd = fafUtils.importTable(rb.getString("mining.production")); - coalProd.buildIndex(coalProd.getColumnPosition("FIPS")); - countyIDsWithEmployment.appendColumn(tempInt, "coalProduction"); - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - int fips = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - try { - countyIDsWithEmployment.setValueAt(i, "coalProduction", coalProd.getIndexedValueAt(fips, "Production_total")); - } catch (Exception e) { - // Set to minor value to ensure that all FAF coal production can be allocated. If the FAF zone of this - // county has other counties with coal production, this miniWeight will be irrelevant as it is small. - // If no county in this FAF zone has coal production, the employment "Construction Natural Resources and Mining" - // will be used to disaggregate flows (with all counties being multiplied by 0.0001, which doesn't affect the result. - float miniWeight = 0.0001f * countyIDsWithEmployment.getValueAt(i, "Construction Natural Resources and Mining"); - countyIDsWithEmployment.setValueAt(i, "coalProduction", miniWeight); - } - } - } catch (Exception e) { - logger.warn("Coal production not defined."); - } - countyFips = countyIDsWithEmployment.getColumnAsInt( - countyIDsWithEmployment.getColumnPosition("COUNTYFIPS")); - } - - - public int[] getCountyFips () { - return countyFips; - } - - - public int getStateNumberOfCounty (int fips) { - for (int row = 1; row <= countyIDsWithEmployment.getRowCount(); row ++) { - if (countyIDsWithEmployment.getValueAt(row, "COUNTYFIPS") == fips) return (int) countyIDsWithEmployment.getValueAt(row, "State"); - } - logger.warn ("State of county " + fips + " was not found."); - return -1; - } - - - public void defineTruckTypes (String truckType, ResourceBundle rb) { - // define truck types by weight or size - - truckTypeShares = fafUtils.importTable(rb.getString("truck.type.by.distance")); - if (truckType.equalsIgnoreCase("weight")) createTruckShareArraysWeight(); - else createTruckShareArraysUnit(); - } - - - private TableDataSet getCountySpecificDataByFAF2(int FAFNumber, String com, TableDataSet detEmpl, - HashMap factors) { - // get employment as a weight for FAF zone FAFNumber in FAF2 - // where detailed employment is available in detEmpl, use make/use factors for commodity-specific weights - - TableDataSet CountySpecifics = new TableDataSet(); - String nameOfFAF = readFAF2.domRegionList.getStringValueAt(FAFNumber, "RegionName"); - CountySpecifics.setName(nameOfFAF); - - int NoCountiesInFAF = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getStringValueAt(i, "FafRegion").equals(nameOfFAF)) NoCountiesInFAF += 1; - } - String[] CountyName = new String[NoCountiesInFAF]; - int[] CountyFips = new int[NoCountiesInFAF]; - String[] CountyFAFRegion = new String[NoCountiesInFAF]; - float[] countyEmpl = new float[NoCountiesInFAF]; - boolean hasDetailedEmployment = checkIfFAFzoneHasDetailedEmployment(nameOfFAF, NoCountiesInFAF, detEmpl, 2); - - int k = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getStringValueAt(i, "FafRegion").equals(nameOfFAF)){ - CountyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - CountyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - CountyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, "FafRegion"); - if (hasDetailedEmployment) { - for (int row = 1; row <= detEmpl.getRowCount(); row++) - if (detEmpl.getValueAt(row, "CountyFips") == CountyFips[k]) - countyEmpl[k] = getWeightedEmpl(row, com, detEmpl, factors); - } else { - countyEmpl[k] = countyIDsWithEmployment.getValueAt(i, "Employment"); - } - k++; - } - } - CountySpecifics.appendColumn(CountyName, "Name"); - CountySpecifics.appendColumn(CountyFips, "COUNTYFIPS"); - CountySpecifics.appendColumn(CountyFAFRegion, "FAFRegion"); - CountySpecifics.appendColumn(countyEmpl, "Employment"); - return CountySpecifics; - } - - - private TableDataSet getCountySpecificDataByFAF(int FAFNumber, String com, TableDataSet detEmployment, - HashMap factors) { - - String fieldName = "FAF3region"; - if(FAFVersion==4) - fieldName = "FAF4region"; - - // get employment as a weight for FAF zone FAFNumber in FAF3 - // where detailed employment is available in detEmpl, use make/use factors for commodity-specific weights - - if (FAFNumber > 800) { - // foreign FAF zone - TableDataSet foreignCountry = new TableDataSet(); - String[] countyName = {String.valueOf(FAFNumber)}; - int[] countyFips = {-1}; - String[] countyFAFRegion = {String.valueOf(FAFNumber)}; - int[] countyEmpl = {1}; - foreignCountry.appendColumn(countyName, "Name"); - foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); - foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); - foreignCountry.appendColumn(countyEmpl, "Employment"); - return foreignCountry; - } - TableDataSet CountySpecifics = new TableDataSet(); - String nameOfFAF = null; - if(FAFVersion==3) - nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); - else - nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); - CountySpecifics.setName(nameOfFAF); - - int NoCountiesInFAF = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; - } - if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); - String[] countyName = new String[NoCountiesInFAF]; - int[] countyFips = new int[NoCountiesInFAF]; - String[] countyFAFRegion = new String[NoCountiesInFAF]; - float[] countyEmpl = new float[NoCountiesInFAF]; - boolean hasDetailedEmployment = checkIfFAFzoneHasDetailedEmployment(nameOfFAF, NoCountiesInFAF, detEmployment, 3); - - int k = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ - countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); - if (hasDetailedEmployment) { - for (int row = 1; row <= detEmployment.getRowCount(); row++) - if (detEmployment.getValueAt(row, "CountyFips") == countyFips[k]) - countyEmpl[k] = getWeightedEmpl(row, com, detEmployment, factors); - } else { - countyEmpl[k] = countyIDsWithEmployment.getValueAt(i, "Employment"); - } - k++; - } - } - CountySpecifics.appendColumn(countyName, "Name"); - CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); - CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); - CountySpecifics.appendColumn(countyEmpl, "Employment"); - return CountySpecifics; - } - - - private TableDataSet getCountySpecificDataByFAFwithDetailedEmployment (int FAFNumber, String com, - HashMap factors, float officeReduction) { - // get employment as a weight for FAF zone FAFNumber in FAF3, use make/use factors for commodity-specific weights - String fieldName = "FAF3region"; - if(FAFVersion==4) - fieldName = "FAF4region"; - - if (FAFNumber > 800) { - // foreign FAF zone - TableDataSet foreignCountry = new TableDataSet(); - String[] countyName = {String.valueOf(FAFNumber)}; - int[] countyFips = {-1}; - String[] countyFAFRegion = {String.valueOf(FAFNumber)}; - int[] countyEmpl = {1}; - foreignCountry.appendColumn(countyName, "Name"); - foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); - foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); - foreignCountry.appendColumn(countyEmpl, "Employment"); - return foreignCountry; - } - TableDataSet CountySpecifics = new TableDataSet(); - String nameOfFAF = null; - - if(FAFVersion==3) - nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); - else - nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); - - CountySpecifics.setName(nameOfFAF); - - int NoCountiesInFAF = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; - } - if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); - String[] countyName = new String[NoCountiesInFAF]; - int[] countyFips = new int[NoCountiesInFAF]; - String[] countyFAFRegion = new String[NoCountiesInFAF]; - float[] countyEmpl = new float[NoCountiesInFAF]; - - int k = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ - countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); - countyEmpl[k] = getWeightedEmpl(i, com, factors, officeReduction); -// if(countyFips[k]==37082 || countyFips[k]==37058 || countyFips[k]==37152 ){//|| destFips==37082 || destFips==37058 || destFips==37152){ -// logger.info("FIPS: ("+countyFips[k]+") WeightedEmp: "+countyEmpl[k]); -// } - k++; - } - } - CountySpecifics.appendColumn(countyName, "Name"); - CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); - CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); - CountySpecifics.appendColumn(countyEmpl, "Employment"); - return CountySpecifics; - } - - - private TableDataSet getCountySpecificDataByFAFwithDetailedEmploymentByType (int FAFNumber, String com, - HashMap factors, float officeReduction) { - // get employment as a weight for FAF zone FAFNumber in FAF3, use make/use factors for commodity-specific weights - // keep track of single employment types - - String fieldName = "FAF3region"; - if(FAFVersion==4) - fieldName = "FAF4region"; - - if (FAFNumber > 800) { - // foreign FAF zone - TableDataSet foreignCountry = new TableDataSet(); - String[] countyName = {String.valueOf(FAFNumber)}; - int[] countyFips = {-1}; - String[] countyFAFRegion = {String.valueOf(FAFNumber)}; - int[] countyEmpl = {1}; - foreignCountry.appendColumn(countyName, "Name"); - foreignCountry.appendColumn(countyFips, "COUNTYFIPS"); - foreignCountry.appendColumn(countyFAFRegion, "FAFRegion"); - foreignCountry.appendColumn(countyEmpl, "Employment"); - return foreignCountry; - } - TableDataSet CountySpecifics = new TableDataSet(); - - String nameOfFAF=null; - if(FAFVersion==3) - nameOfFAF = readFAF3.getFAFzoneName(FAFNumber); - else - nameOfFAF = ReadFAF4.getFAFzoneName(FAFNumber); - - CountySpecifics.setName(nameOfFAF); - - int NoCountiesInFAF = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber) NoCountiesInFAF += 1; - } - if (NoCountiesInFAF == 0) logger.warn("No counties for FAF zone " + FAFNumber + " have been found."); - String[] countyName = new String[NoCountiesInFAF]; - int[] countyFips = new int[NoCountiesInFAF]; - String[] countyFAFRegion = new String[NoCountiesInFAF]; - float[][] countyEmpl = new float[NoCountiesInFAF][empCats.length+1]; - - int k = 0; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) { - if (countyIDsWithEmployment.getValueAt(i, fieldName) == FAFNumber){ - countyName[k] = countyIDsWithEmployment.getStringValueAt(i, "NAME"); - countyFips[k] = (int) countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS"); - countyFAFRegion[k] = countyIDsWithEmployment.getStringValueAt(i, fieldName); - HashMap detEmpl = getWeightedEmplByEmpl(i, com, factors, officeReduction); - countyEmpl[k][empCats.length] = detEmpl.get("total"); - for (int emp = 0; emp < empCats.length; emp++) countyEmpl[k][emp] = detEmpl.get(empCats[emp]); - k++; - } - } - CountySpecifics.appendColumn(countyName, "Name"); - CountySpecifics.appendColumn(countyFips, "COUNTYFIPS"); - CountySpecifics.appendColumn(countyFAFRegion, "FAFRegion"); - for (int emp = 0; emp <= empCats.length; emp++) { - float[] copy = new float[NoCountiesInFAF]; - for (int i = 0; i < copy.length; i++) copy[i] = countyEmpl[i][emp]; - if (emp < empCats.length) CountySpecifics.appendColumn(copy, empCats[emp]); - else CountySpecifics.appendColumn(copy, "Employment"); - } - return CountySpecifics; - } - - - public void prepareCountyDataForFAFwithDetailedEmployment(ResourceBundle rb, int yr, TableDataSet specialRegions) { - // prepare county data to provide detailed employment as a weight using the "fafVersion" zone system - - logger.info("Preparing county data with detailed employment for FAF flow disaggregation for year " + yr + "..."); - HashMap useC = createMakeUseHashMap(rb, "faf.use.coefficients"); - HashMap makeC = createMakeUseHashMap(rb, "faf.make.coefficients"); - distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); - float reduction = (float) ResourceUtil.getDoubleProperty(rb, "reduction.local.office.weight", 1d); - - //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share - countyShares = new HashMap<>(); - String[] direction = {"orig", "dest"}; - for (String dir: direction) { - HashMap factors; - if (dir.equals("orig")) factors = makeC; - else factors = useC; - - for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { - for (String com: readFAF3.sctgStringCommodities) { - int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); - TableDataSet CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); - String code = dir + "_" + zoneNum + "_" + com; - countyShares.put(code, CountyList); - } - } - } - - // create fake county lists for special regions such as airports or seaports that shall be kept separate from the FAF regions - if (specialRegions == null) return; // OK truck flows - for (int row = 1; row <= specialRegions.getRowCount(); row++) { - int[] fips = {(int) specialRegions.getValueAt(row, "modelCode")}; - String[] name = new String[]{specialRegions.getStringValueAt(row, "faf3code")}; - TableDataSet CountyList = new TableDataSet(); - float[] emplDummy = {1}; - CountyList.appendColumn(name, "Name"); - CountyList.appendColumn(fips, "COUNTYFIPS"); - CountyList.appendColumn(name, "FAFRegion"); - CountyList.appendColumn(emplDummy, "Employment"); - for (String dir: direction) { - for (String com: readFAF3.sctgStringCommodities) { - String code = dir + "_" + fips[0] + "_" + com; - countyShares.put(code, CountyList); - } - } - } - } - - - public void prepareCountyDataForFAFwithDetailedEmployment(ResourceBundle rb, int yr, boolean keepTrackOfEmpType) { - // prepare county data to provide detailed employment as a weight - - logger.info(" Preparing county data with detailed employment for FAF flow disaggregation for year " + yr + "..."); - HashMap useC = createMakeUseHashMap(rb, "faf.use.coefficients"); - HashMap makeC = createMakeUseHashMap(rb, "faf.make.coefficients"); - distCounties = MatrixReader.readMatrix(new File(rb.getString("county.distance.in.miles")), "Distance"); - float reduction = (float) ResourceUtil.getDoubleProperty(rb, "reduction.local.office.weight", 1d); - - //create HashMap that contains for every FAF region a TableDataSet of counties with their employment share - countyShares = new HashMap<>(); - String[] direction = {"orig", "dest"}; - for (String dir: direction) { - HashMap factors; - if (dir.equals("orig")) factors = makeC; - else factors = useC; - - Object readFAF = null; - if(FAFVersion==3){ - - for (int fafNum = 1; fafNum <= readFAF3.fafRegionList.getRowCount(); fafNum++) { - for (String com: readFAF3.sctgStringCommodities) { - int zoneNum = (int) readFAF3.fafRegionList.getValueAt(fafNum, "ZoneID"); - TableDataSet CountyList; - if (!keepTrackOfEmpType) { - CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); - } else { - CountyList = getCountySpecificDataByFAFwithDetailedEmploymentByType(zoneNum, com, factors, reduction); - } - String code = dir + "_" + zoneNum + "_" + com; - countyShares.put(code, CountyList); - } - } - }else{ - for (int fafNum = 1; fafNum <= ReadFAF4.fafRegionList.getRowCount(); fafNum++) { - for (String com: ReadFAF4.sctgStringCommodities) { - int zoneNum = (int) ReadFAF4.fafRegionList.getValueAt(fafNum, "ZoneID"); - TableDataSet CountyList; - if (!keepTrackOfEmpType) { - CountyList = getCountySpecificDataByFAFwithDetailedEmployment(zoneNum, com, factors, reduction); - } else { - CountyList = getCountySpecificDataByFAFwithDetailedEmploymentByType(zoneNum, com, factors, reduction); - } - String code = dir + "_" + zoneNum + "_" + com; - countyShares.put(code, CountyList); - } - } - - } - } - } - - - public void scaleSelectedCounties (ResourceBundle rb) { - // scale weight of employment up or down for selected counties - String fieldName = "FAF3region"; - if(FAFVersion==4) - fieldName = "FAF4region"; - - TableDataSet countyScaler = fafUtils.importTable(rb.getString("county.scaler")); - for (int row = 1; row <= countyScaler.getRowCount(); row++) { - int fips= (int) countyScaler.getValueAt(row, "countyFips"); - float scaler = countyScaler.getValueAt(row, "scaler"); - // find FAF zone - int fafZone = 0; - for (int countyRow = 1; countyRow <= countyIDsWithEmployment.getRowCount(); countyRow++) { - if (countyIDsWithEmployment.getValueAt(countyRow, "COUNTYFIPS") == fips) fafZone = - (int) countyIDsWithEmployment.getValueAt(countyRow, fieldName); - } - if (fafZone == 0) logger.error("Could not find FAF zone of FIPS " + fips + " in file " + rb.getString("county.ID")); - String[] direction = {"orig", "dest"}; - for (String dir: direction) { - - if(FAFVersion==3){ - for (String com: readFAF3.sctgStringCommodities) { - String code = dir + "_" + fafZone + "_" + com; - TableDataSet scalerThisCounty = countyShares.get(code); - for (int scalerRow = 1; scalerRow <= scalerThisCounty.getRowCount(); scalerRow++) { - if (scalerThisCounty.getValueAt(scalerRow, "COUNTYFIPS") == fips) { - float value = scalerThisCounty.getValueAt(scalerRow, "Employment") * scaler; - scalerThisCounty.setValueAt(scalerRow, "Employment", value); - } - } - } - }else{ - for (String com: ReadFAF4.sctgStringCommodities) { - String code = dir + "_" + fafZone + "_" + com; - TableDataSet scalerThisCounty = countyShares.get(code); - for (int scalerRow = 1; scalerRow <= scalerThisCounty.getRowCount(); scalerRow++) { - if (scalerThisCounty.getValueAt(scalerRow, "COUNTYFIPS") == fips) { - float value = scalerThisCounty.getValueAt(scalerRow, "Employment") * scaler; - scalerThisCounty.setValueAt(scalerRow, "Employment", value); - } - } - } - - - } - } - } - } - - - private boolean checkIfFAFzoneHasDetailedEmployment(String nameOfFAF, int noCountiesInFAF, TableDataSet detEmpl, - int fafVersion) { - // Check if detEmpl has detailed employment for all counties in FAF zone FAFNumber - - if (detEmpl == null) return false; - boolean[] countyHasDet = new boolean[noCountiesInFAF]; - int k = 0; - String colLabel; - if (fafVersion == 2) colLabel = "FafRegion"; - else colLabel = "FAF3region"; - for (int i = 1; i <= countyIDsWithEmployment.getRowCount(); i++) - if (countyIDsWithEmployment.getStringValueAt(i, colLabel).equals(nameOfFAF)) - for (int row = 1; row <= detEmpl.getRowCount(); row++) { - if (countyIDsWithEmployment.getValueAt(i, "COUNTYFIPS") == detEmpl.getValueAt(row, "CountyFips")) { - countyHasDet[k] = true; - k++; - } - } - int count = 0; - for (boolean county: countyHasDet) if (county) count++; - if (count == 0) return false; - else if (count == noCountiesInFAF) return true; - else logger.fatal("Detailed employment covers FAF zone " + nameOfFAF + " only partly."); - return false; - } - - - private float getWeightedEmpl(int row, String comFAF2, TableDataSet detEmpl, HashMap factors) { - // calculate weighted employment based on make/use coefficients - - float empl = 0; - String[] emplCategoriesTemp = detEmpl.getColumnLabels(); - String[] emplCategories = new String[emplCategoriesTemp.length-2]; - System.arraycopy(emplCategoriesTemp, 2, emplCategories, 0, emplCategories.length); - for (String emplCat: emplCategories) { - String code = emplCat + "_" + comFAF2; - float coeff = factors.get(code); - for (int col = 3; col <= detEmpl.getColumnCount(); col++) { - empl += detEmpl.getValueAt(row, col) * coeff; - } - } - return empl; - } - - - private float getWeightedEmpl(int row, int comFAF3, TableDataSet detEmpl, HashMap factors) { - // calculate weighted employment based on make/use coefficients - - float empl = 0; - String[] emplCategoriesTemp = detEmpl.getColumnLabels(); - String[] emplCategories = new String[emplCategoriesTemp.length-2]; - System.arraycopy(emplCategoriesTemp, 2, emplCategories, 0, emplCategories.length); - for (String emplCat: emplCategories) { - String code = emplCat + "_" + String.valueOf(comFAF3); - float coeff = factors.get(code); - for (int col = 3; col <= detEmpl.getColumnCount(); col++) { - empl += detEmpl.getValueAt(row, col) * coeff; - } - } - return empl; - } - - - private float getWeightedEmpl(int row, String comFAF, HashMap factors, float officeReduction) { - // calculate weighted employment based on make/use coefficients - float empl = 0; - for (String emp: empCats) { - String code = emp + "_" + comFAF; - float coeff = factors.get(code); - float factor = 1; - if (fafUtils.arrayContainsElement(emp, new String[]{"Information", "Financial Activities", - "Professional and Business Services", "Education and Health Services", - "Leisure and Hospitality"})) factor = officeReduction; - empl += countyIDsWithEmployment.getValueAt(row, emp) * coeff * factor; - } - return empl; - } - - - private HashMap getWeightedEmplByEmpl(int row, String comFAF, HashMap factors, float officeReduction) { - // calculate weighted employment based on make/use coefficients, keep information by employment type - - HashMap empl = new HashMap<>(); - float total = 0; - for (String emp: empCats) { - String code = emp + "_" + comFAF; - float coeff = factors.get(code); - float factor = 1; - if (fafUtils.arrayContainsElement(emp, new String[]{"Information", "Financial Activities", - "Professional and Business Services", "Education and Health Services", - "Leisure and Hospitality"})) factor = officeReduction; - float thisVal = countyIDsWithEmployment.getValueAt(row, emp) * coeff * factor; - empl.put(emp, thisVal); - total += thisVal; - } - empl.put("total", total); - return empl; - } - - - public double[][] disaggCountyToZones(float flow, double[] weightsA, double[] weightsB) { - // disaggregate flow from orig county A to orig zones i and dest county B to dest zones j - - // sum up weight products - double sm = 0; - for (double thisWeightsA : weightsA) { - for (double thisWeightsB : weightsB) { - sm += thisWeightsA * thisWeightsB; - } - } - // disaggregate flow - double[][] disFlow = new double[weightsA.length][weightsB.length]; - for (int i = 0; i < weightsA.length; i++) { - for (int j = 0; j < weightsB.length; j++) { - disFlow[i][j] = flow * weightsA[i] * weightsB[j] / sm; - } - } - return disFlow; - } - - - public static int getCountyId(int fips) { - // Return region code of regName - for (int i = 0; i < countyFips.length; i++) { - if (countyFips[i] == fips) return i; - } - logger.error("Could not find county FIPS code " + fips); - return -1; - } - - - public String[] getEmpCats() { - return empCats; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java deleted file mode 100644 index ee20bc4..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/fafUtils.java +++ /dev/null @@ -1,331 +0,0 @@ -package org.sandag.htm.processFAF; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.datafile.CSVFileReader; -import com.pb.common.util.ResourceUtil; - -import java.io.*; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.ResourceBundle; -import org.apache.log4j.Logger; - -/** - * Utilities to process FAF2 data - * User: Rolf Moeckel - * Date: May 6, 2009 - */ -public class fafUtils { - - private static Logger logger = Logger.getLogger(fafUtils.class); - private static ResourceBundle rb; - private static TableDataSet payloadSTCC; - private static TableDataSet payloadSCTG; - - public static TableDataSet importTable(String filePath) { - // read a csv file into a TableDataSet - TableDataSet tblData; - CSVFileReader cfrReader = new CSVFileReader(); - try { - tblData = cfrReader.readFile(new File( filePath )); - } catch (Exception e) { - throw new RuntimeException("File not found: <" + filePath + ">.", e); - } - cfrReader.close(); - return tblData; - } - - - public static ResourceBundle getResourceBundle(String pathToRb) { - File propFile = new File(pathToRb); - rb = ResourceUtil.getPropertyBundle(propFile); - if (rb == null) logger.fatal ("Problem loading resource bundle: " + pathToRb); - return rb; - } - - - public static void setResourceBundle (ResourceBundle appRb) { - rb = appRb; - } - - - public static PrintWriter openFileForSequentialWriting(String fileName) { - File outputFile = new File(fileName); - FileWriter fw = null; - try { - fw = new FileWriter(outputFile); - } catch (IOException e) { - logger.error("Could not open file <" + fileName + ">."); - } - BufferedWriter bw = new BufferedWriter(fw); - return new PrintWriter(bw); - } - - - public static HashMap createScalerHashMap (String[] tokens, double[] values) { - // create HashMap with state O-D pairs that need to be scaled - - HashMap scaler = new HashMap<>(); - if (tokens.length != values.length) { - throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); - } - for (int i=0; i createScalerHashMap (TableDataSet scaleTable, String[] columnNames) { - // create HashMap with state O-D pairs that need to be scaled - - HashMap scaler = new HashMap<>(); - for (int row = 1; row < scaleTable.getRowCount(); row++) { - String token = String.valueOf((int) scaleTable.getValueAt(row, columnNames[0])) + "_" + - String.valueOf((int) scaleTable.getValueAt(row, columnNames[1])); - scaler.put(token, scaleTable.getValueAt(row, columnNames[2])); - } - return scaler; - } - - - public static int isOnPosition (String txt, String[] txtArray) { - // checks if txt is a value of txtArray - int position = -1; - for (int i = 0; i < txtArray.length; i++) if (txtArray[i].equals(txt)) position = i; - return position; - } - - - public static int[] createCountyFipsArray (int[] specRegCodes) { - // create array with county FIPS codes including special regions (such as airports) - - // Note: Used to readFAF2 employment from here, but should be done separately for each project -// disaggregateFlows.getUScountyEmploymentFromOneFile(appRb); - int[] countyFipsS = disaggregateFlows.countyIDsWithEmployment.getColumnAsInt( - disaggregateFlows.countyIDsWithEmployment.getColumnPosition("COUNTYFIPS")); - int[] countyFips = new int[countyFipsS.length + specRegCodes.length]; - System.arraycopy(countyFipsS, 0, countyFips, 0, countyFipsS.length); - System.arraycopy(specRegCodes, 0, countyFips, countyFipsS.length, specRegCodes.length); - return countyFips; - } - - - public static float findAveragePayload(String comm, String comClass) { - // returns average payload in tons per truck for commodity comm - - if (comm.equals("SCTG99")) comm = "SCTG42"; // FAF3 calls unknown SCTG99 instead of SCTG42 - TableDataSet payload = new TableDataSet(); - if (comClass.equals("STCC")) { - if (payloadSTCC == null) payloadSTCC = importTable(ResourceUtil.getProperty(rb, "truck.commodity.payload")); - payload = payloadSTCC; - } else if (comClass.equals("SCTG")) { - if (payloadSCTG == null) payloadSCTG = importTable(ResourceUtil.getProperty(rb, "truck.SCTG.commodity.payload")); - payload = payloadSCTG; - } - int n = -1; - for (int k = 1; k <= payload.getRowCount(); k++) { - if (payload.getStringValueAt(k, "Commodity").equals(comm)) n = k; - } - if (n == -1) { - logger.fatal("Commodity " + comm + " not found in payload factor file."); - System.exit(1); - } - // weight of each truck type, using an average for all commodities, derived from fhwa website - if (payload.containsColumn("Single Unit Trucks")) { - float[] weights = new float[] {0.307f, 0.155f, 0.269f, 0.269f}; // including pickups, minivans, other light vans, SUVs: 0.93642251f,0.022471435f,0.010687433f,0.030418621f - return (payload.getValueAt(n, "Single Unit Trucks") * weights[0]) + - (payload.getValueAt(n, "Semi Trailer") * weights[1]) + - (payload.getValueAt(n, "Double Trailers") * weights[2]) + - (payload.getValueAt(n, "Triples") * weights[3]); - } else { - return (payload.getValueAt(n, "Payload (lbs)") * (float) 0.0005); - } - } - - - public static void readPayloadFactors (ResourceBundle appRb) { - payloadSCTG = importTable(ResourceUtil.getProperty(appRb, "truck.SCTG.commodity.payload")); - } - - - public static float findAveragePayload(String comm) { - // returns average payload in tons per truck for commodity comm - - if (comm.equals("SCTG99")) comm = "SCTG42"; // FAF3 calls unknown SCTG99 instead of SCTG42 - int n = -1; - for (int k = 1; k <= payloadSCTG.getRowCount(); k++) { - if (payloadSCTG.getStringValueAt(k, "Commodity").equals(comm)) n = k; - } - if (n == -1) { - logger.fatal("Commodity " + comm + " not found in payload factor file."); - System.exit(1); - } - return (payloadSCTG.getValueAt(n, "Payload (lbs)") * (float) 0.0005); - } - - - public static int getEnumOrderNumber(ModesFAF mode) { - // return order number of mode - int num = 1; - for (ModesFAF thisMode: ModesFAF.values()) { - if (thisMode == mode) return num; - num++; - } - logger.warn("Could not find mode " + mode.toString()); - return 0; - } - - - public static ModesFAF getModeName (int mode) { - return ModesFAF.values()[mode - 1]; - } - - - public static int getHighestVal(int[] array) { - // return highest number in array - int high = Integer.MIN_VALUE; - for (int num: array) high = Math.max(high, num); - return high; - } - - - public static double sumArray (double[][] arr) { - // sum a two-dimensional double array - - double sum = 0; - for (double[] anArr : arr) { - for (int j = 0; j < arr[1].length; j++) { - sum += anArr[j]; - } - } - return sum; - } - - - public static TableDataSet createSpecialRegions(String[] specRegNames, String[] specRegModes, int[] specRegCodes, - int [] specRegZones, int[] specRegFAFCodes) { - // create TableDataSet with special regions - - if (specRegCodes.length != specRegNames.length || specRegCodes.length != specRegModes.length|| specRegCodes.length != specRegFAFCodes.length) { - logger.error ("Names of special regions and modes of special regions and codes of special regions and " + - "FAF codes of special regions all need to be of same length. No special regions created."); - return null; - } - TableDataSet specRegions = new TableDataSet(); - specRegions.appendColumn(specRegNames, "Name"); - specRegions.appendColumn(specRegCodes, "modelCode"); - specRegions.appendColumn(specRegZones, "modelZone"); - specRegions.appendColumn(specRegFAFCodes, "faf3code"); - specRegions.appendColumn(specRegModes, "mode"); - float[] dummyEmployment = new float[specRegCodes.length]; - for (int i = 0; i < dummyEmployment.length; i++) dummyEmployment[i] = 1; - specRegions.appendColumn(dummyEmployment, "Employment"); - return specRegions; - } - - - public static boolean countyFlowConnectsWithHawaii (int orig, int dest) { - // check if flow connects with Hawaii county - int oState = (int) (orig / 1000f); - int dState = (int) (dest / 1000f); - return oState == 15 || dState == 15; - } - - - public static boolean arrayContainsElement (String element, String[] array) { - // Check if Array contains Element - - boolean result = false; - for (String t: array) if (t.equals(element)) result = true; - return result; - } - - - public static int[] expandArrayByOneElement (int[] existing, int addElement) { - // create new array that has length of existing.length + 1 and copy values into new array - int[] expanded = new int[existing.length + 1]; - System.arraycopy(existing, 0, expanded, 0, existing.length); - expanded[expanded.length - 1] = addElement; - return expanded; - } - - - public static float[] expandArrayByOneElement (float[] existing, float addElement) { - // create new array that has length of existing.length + 1 and copy values into new array - float[] expanded = new float[existing.length + 1]; - System.arraycopy(existing, 0, expanded, 0, existing.length); - expanded[expanded.length - 1] = addElement; - return expanded; - } - - - public static String[] expandArrayByOneElement (String[] existing, String addElement) { - // create new array that has length of existing.length + 1 and copy values into new array - String[] expanded = new String[existing.length + 1]; - System.arraycopy(existing, 0, expanded, 0, existing.length); - expanded[expanded.length - 1] = addElement; - return expanded; - } - - - public static float rounder(float value, int digits) { - // rounds value to digits behind the decimal point - return Math.round(value * Math.pow(10, digits) + 0.5)/(float) Math.pow(10, digits); - } - - - public static String[] findUniqueElements (String[] list) { - // find unique elements in list[] and return string[] with these elements - - ArrayList unique = new ArrayList(); - for (String txt: list) { - if (!unique.contains(txt)) unique.add(txt); - } - String[] elements = new String[unique.size()]; - for (int i = 0; i < unique.size(); i++) elements[i] = unique.get(i); - return elements; - } - - - public static float getSum (float[] array) { - float sum = 0; - for (float val: array) sum += val; - return sum; - } - - public static double getSum (double[] array) { - double sum = 0; - for (double val: array) sum += val; - return sum; - } - - public static float getSum (float[][] array) { - float sum = 0; - for (float[] anArray : array) { - for (int j = 0; j < array[0].length; j++) sum += anArray[j]; - } - return sum; - } - - public static double getSum (double[][] array) { - double sum = 0; - for (double[] anArray : array) { - for (int j = 0; j < array[0].length; j++) sum += anArray[j]; - } - return sum; - } - - public static String[] getUniqueListOfValues (String[] list) { - // itentify unique list of value in list[] and return as shortened string list - ArrayList al = new ArrayList<>(); - for (String txt: list) { - if (!al.contains(txt) && txt != null) al.add(txt); - } - String[] shortenedList = new String[al.size()]; - for (int i = 0; i < al.size(); i++) { - shortenedList[i] = al.get(i); - } - return shortenedList; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java deleted file mode 100644 index 658cc10..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF2.java +++ /dev/null @@ -1,547 +0,0 @@ -package org.sandag.htm.processFAF; - -import org.apache.log4j.Logger; -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -import java.util.ResourceBundle; -import java.util.HashMap; - -/** - * This class reads FAF2 data and stores data in TableDataSets - * User: Rolf Moeckel - * Date: May 6, 2009 - */ - -public class readFAF2 { - - Logger logger = Logger.getLogger(readFAF2.class); - static public TableDataSet domRegionList; - static public TableDataSet rowRegionList; - static public String[] FAFzones; - static public String[] sctgCommodities; - static public String[] stccCommodities; - static private int highestDomRegion; - static public HashMap SCTGtoSTCCconversion; - static private HashMap sctgCode; - - private boolean referenceListsAreRead = false; - private TableDataSet domesticTonCommodityFlows; - private TableDataSet intBorderTonCommodityFlows; - private TableDataSet intSeaTonCommodityFlows; - private TableDataSet intAirTonCommodityFlows; - private TableDataSet domesticDollarCommodityFlows; - private TableDataSet intBorderDollarCommodityFlows; - private TableDataSet intSeaDollarCommodityFlows; - private TableDataSet intAirDollarCommodityFlows; - private int factor; - - - public void readCommodityList(ResourceBundle appRb) { - // read commodity names - TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.sctg.commodity.list")); - sctgCommodities = new String[sctgComList.getRowCount()]; - for (int i = 1; i <= sctgComList.getRowCount(); i++) sctgCommodities[i-1] = sctgComList.getStringValueAt(i, "SCTG"); - } - - - public void readAllFAF2dataSets(ResourceBundle appRb, String unit) { - // read all FAF2 data into TableDataSets in unit (= tons or dollars) - if (unit.equals("tons")) { - domesticTonCommodityFlows = readDomesticCommodityFlows(appRb, unit); - intBorderTonCommodityFlows = readInternationalCommodityFlowsThroughLandBorder(appRb, unit); - intSeaTonCommodityFlows = readInternationalCommodityFlowsBySea(appRb, unit); - intAirTonCommodityFlows = readInternationalCommodityFlowsByAir(appRb, unit); - factor = 1000; // tons are in 1,000s - } else if (unit.equals("dollars")) { - domesticDollarCommodityFlows = readDomesticCommodityFlows(appRb, unit); - intBorderDollarCommodityFlows = readInternationalCommodityFlowsThroughLandBorder(appRb, unit); - intSeaDollarCommodityFlows = readInternationalCommodityFlowsBySea(appRb, unit); - intAirDollarCommodityFlows = readInternationalCommodityFlowsByAir(appRb, unit); - factor = 1000000; // dollars are in 1,000,000s - } else { - logger.error("Wrong token " + unit + " in method readAllFAF2dataSets. Use tons or dollars."); - } - } - - - public void adjustTruckFAF2data(ResourceBundle appRb, String mode, int[] years) { - // adjust FAF2 growth to exogenous adjustment - String fileName = appRb.getString("adjustment.of.faf." + mode); - logger.info("Adjusting FAF2 " + mode + " forecast to growth rate set in " + fileName); - TableDataSet adjust = fafUtils.importTable(fileName); - adjust.buildIndex(adjust.getColumnPosition("Year")); - for (int yr: years) { - float fafForecast = adjust.getIndexedValueAt(yr, "FAF2forecast"); - float adjForecast = adjust.getIndexedValueAt(yr, "AdjustedNumber"); - if (fafForecast == adjForecast) continue; - for (int row = 1; row <= domesticTonCommodityFlows.getRowCount(); row++) { - if (!domesticTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; - float value = domesticTonCommodityFlows.getValueAt(row, Integer.toString(yr)); - value = value * adjForecast / fafForecast; - domesticTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); - } - for (int row = 1; row <= intBorderTonCommodityFlows.getRowCount(); row++) { - if (!intBorderTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; - float value = intBorderTonCommodityFlows.getValueAt(row, Integer.toString(yr)); - value = value * adjForecast / fafForecast; - intBorderTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); - } - for (int row = 1; row <= intSeaTonCommodityFlows.getRowCount(); row++) { - if (!intSeaTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase(mode)) continue; - float value = intSeaTonCommodityFlows.getValueAt(row, Integer.toString(yr)); - value = value * adjForecast / fafForecast; - intSeaTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); - } - for (int row = 1; row <= intAirTonCommodityFlows.getRowCount(); row++) { - if (!intAirTonCommodityFlows.getStringValueAt(row, "Mode").equalsIgnoreCase("Air & " + mode)) continue; - float value = intAirTonCommodityFlows.getValueAt(row, Integer.toString(yr)); - value = value * adjForecast / fafForecast; - intAirTonCommodityFlows.setValueAt(row, Integer.toString(yr), value); - } - } - } - - - public TableDataSet readDomesticCommodityFlows(ResourceBundle appRb, String unit) { - // read domestic FAF2 flows in unit (tons or dollars) - logger.info ("Reading domestic FAF2 data in " + unit); - - String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.domestic." + unit)); - TableDataSet flows = fafUtils.importTable(fileName); - if (!referenceListsAreRead) { - readFAF2ReferenceLists(appRb); - referenceListsAreRead = true; - } - int[] originCodes = new int[flows.getRowCount()]; - int[] destinationCodes = new int[flows.getRowCount()]; - for (int i = 1; i <= flows.getRowCount(); i++) { - //find Origin and Destination codes - originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); - destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); - } - flows.appendColumn(originCodes, "OriginCode"); - flows.appendColumn(destinationCodes, "DestinationCode"); - return flows; - } - - - public static void readFAF2ReferenceLists(ResourceBundle appRb) { - // read reference lists for zones and commodities - domRegionList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.region.list")); - rowRegionList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.row.region.list")); - - highestDomRegion = 0; - for (int k = 1; k <= domRegionList.getRowCount(); k++) - highestDomRegion = Math.max((int) domRegionList.getValueAt(k, "RegionCode"), highestDomRegion); - int highestRegionOverAll = highestDomRegion; - for (int k = 1; k <= rowRegionList.getRowCount(); k++) - highestRegionOverAll = Math.max((int) rowRegionList.getValueAt(k, "ROWCode"), highestRegionOverAll); - FAFzones = new String[highestRegionOverAll + 1]; - for (int k = 1; k <= domRegionList.getRowCount(); k++) - FAFzones[(int) domRegionList.getValueAt(k, "RegionCode")] = domRegionList.getStringValueAt(k, "RegionName"); - for (int k = 1; k <= rowRegionList.getRowCount(); k++) - FAFzones[(int) rowRegionList.getValueAt(k, "ROWCode")] = rowRegionList.getStringValueAt(k, "ROWRegionName"); - - domRegionList.buildIndex(1); - rowRegionList.buildIndex(1); - - TableDataSet sctgNumber = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf2.commodity.reference")); - sctgCode = new HashMap(); - for (int i = 1; i <= sctgNumber.getRowCount(); i++) - sctgCode.put(sctgNumber.getStringValueAt(i, "FlowTableCategories"), (int) sctgNumber.getValueAt(i, "SCTG")); - - } - - - private int findZoneCode(String strZone) { - // find code of zone with name strZone - int zoneCode = -1; - // Domestic FAF zones - for (int k = 1; k <= domRegionList.getRowCount(); k++) - if (domRegionList.getStringValueAt(k, "RegionName").equals(strZone)) - zoneCode = (int) domRegionList.getValueAt(k, "RegionCode"); - // International FAF zones - if (zoneCode == -1) { - for (int k = 1; k <= rowRegionList.getRowCount(); k++) - if (rowRegionList.getStringValueAt(k, "ROWRegionName").equals(strZone)) - zoneCode = (int) rowRegionList.getValueAt(k, "ROWCode"); - } - if (zoneCode == -1) { - logger.error ("Unknown Zone in FAF2 data: " + strZone); - System.exit(1); - } - return zoneCode; - } - - - public HashMap getDomesticFlows(String mode, commodityClassType comClass, int yr, String unit) { - // extract domestic FAF2 flows - TableDataSet flowTbl; - if (unit.equals("tons")) { - flowTbl = domesticTonCommodityFlows; - } else { - flowTbl = domesticDollarCommodityFlows; - } - - HashMap hshFlows = new HashMap(); - for (int i = 1; i <= flowTbl.getRowCount(); i++) { - String thisMode = flowTbl.getStringValueAt(i, "Mode"); - if (!mode.equals("all") && !thisMode.equalsIgnoreCase(mode)) continue; - int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); - int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); - float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; - String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); - int intSctgCommmodity = sctgCode.get(fafDataCommodity); - String sctgCommodity; - if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; - else sctgCommodity = "SCTG" + intSctgCommmodity; - // report flows by STCC commodity classification - if (comClass.equals(commodityClassType.STCC)) { - for (String com: stccCommodities) { - float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; - String code; - if (flowShare > 0) { - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + com + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + com; - } - - if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); - hshFlows.put(code, flowShare); - } - } - } else { - // report flows by SCTG commodity classification - String code; - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + sctgCommodity; - } - if (hshFlows.containsKey(code)) flows += hshFlows.get(code); - hshFlows.put(code, flows); - } - } - return hshFlows; - } - - - public TableDataSet readInternationalCommodityFlowsThroughLandBorder(ResourceBundle appRb, String unit) { - // read international FAF2 flows that cross the U.S. border by land in unit (tons or dollars) - logger.info ("Reading international FAF2 data crossing borders by land in " + unit); - - String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.border." + unit)); - TableDataSet flows = fafUtils.importTable(fileName); - if (!referenceListsAreRead) { - readFAF2ReferenceLists(appRb); - referenceListsAreRead = true; - } - int[] originCodes = new int[flows.getRowCount()]; - int[] portOfEntryCodes = new int[flows.getRowCount()]; - int[] destinationCodes = new int[flows.getRowCount()]; - for (int i = 1; i <= flows.getRowCount(); i++) { - //find Origin and Destination codes - originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); - portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "PortOfEntryExit")); - destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); - } - flows.appendColumn(originCodes, "OriginCode"); - flows.appendColumn(portOfEntryCodes, "PortOfEntryCode"); - flows.appendColumn(destinationCodes, "DestinationCode"); - return flows; - } - - - public HashMap getIntBorderFlows(reportFormat repform, String mode, commodityClassType comClass, - int yr, String unit) { - // extract international FAF2 flows (international mode truck or train) - - TableDataSet flowTbl; - if (unit.equals("tons")) { - flowTbl = intBorderTonCommodityFlows; - } else { - flowTbl = intBorderDollarCommodityFlows; - } - - HashMap hshFlows = new HashMap(); - for (int i = 1; i <= flowTbl.getRowCount(); i++) { - String thisMode = flowTbl.getStringValueAt(i, "Mode"); - if (!mode.equals("all") && !thisMode.equals(mode)) continue; - int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); - int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); - String strOrig = flowTbl.getStringValueAt(i, "Origin"); - String strDest = flowTbl.getStringValueAt(i, "Destination"); - int borderCode = (int) flowTbl.getValueAt(i, "PortOfEntryCode"); - // if flow goes from WA Blain to Canada with port of exit WA Blain, don't report domestic part. - Changed my mind, do report this flow -// if (repform == reportFormat.internat_domesticPart && (destCode == borderCode || origCode == borderCode)) continue; - if (repform == reportFormat.internat_domesticPart) { - boolean isCanadaOrMexico = checkIfMexicoOrCanada(strOrig); - if (isCanadaOrMexico) origCode = borderCode; - isCanadaOrMexico = checkIfMexicoOrCanada(strDest); - if (isCanadaOrMexico) destCode = borderCode; - } else if (repform == reportFormat.internat_internationalPart) { - boolean isCanadaOrMexico = checkIfMexicoOrCanada(strOrig); - if (isCanadaOrMexico) destCode = borderCode; - isCanadaOrMexico = checkIfMexicoOrCanada(strDest); - if (isCanadaOrMexico) origCode = borderCode; - } - // note that border flows are done differently than sea and air flows. Oregon has port and airports, that's - // why those flows are extracted separately. Oregon has no international border, therefore border flows are - // added to the US FAF region where the goods cross the border (unless reportFormat.internatOrigToDest has - // been chosen). - float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; - String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); - int intSctgCommmodity = sctgCode.get(fafDataCommodity); - String sctgCommodity; - if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; - else sctgCommodity = "SCTG" + intSctgCommmodity; - if (comClass.equals(commodityClassType.STCC)) { - // report by STCC commodity classification - for (String com: stccCommodities) { - float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; - if (flowShare > 0) { - String code; - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + sctgCommodity; - } - if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); - hshFlows.put(code, flowShare); - } - } - } else { - // report by SCTG commodity classification - String code; - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + sctgCommodity; - } - if (hshFlows.containsKey(code)) flows += hshFlows.get(code); - hshFlows.put(code, flows); - } - } - return hshFlows; - } - - - public TableDataSet readInternationalCommodityFlowsBySea(ResourceBundle appRb, String unit) { - // read international FAF2 flows that cross the U.S. border by sea in unit (tons or dollars) - logger.info ("Reading international FAF2 data by sea in " + unit); - - String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.sea." + unit)); - TableDataSet flows = fafUtils.importTable(fileName); - if (!referenceListsAreRead) { - readFAF2ReferenceLists(appRb); - referenceListsAreRead = true; - } - int[] originCodes = new int[flows.getRowCount()]; - int[] portOfEntryCodes = new int[flows.getRowCount()]; - int[] destinationCodes = new int[flows.getRowCount()]; - for (int i = 1; i <= flows.getRowCount(); i++) { - //find Origin and Destination codes - originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); - portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Port")); - destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); - } - flows.appendColumn(originCodes, "OriginCode"); - flows.appendColumn(portOfEntryCodes, "PortCode"); - flows.appendColumn(destinationCodes, "DestinationCode"); - return flows; - } - - - public HashMap getIntSeaFlows (reportFormat repform, String mode, commodityClassType comClass, - int yr, String unit) { - // extract international FAF2 flows (international mode sea) - // If repform is set to internat_domesticPart port entry and exit points are set as negative origin or destination codes - - TableDataSet flowTbl; - if (unit.equals("tons")) { - flowTbl = intSeaTonCommodityFlows; - } else { - flowTbl = intSeaDollarCommodityFlows; - } - - HashMap hshFlows = new HashMap(); - for (int i = 1; i <= flowTbl.getRowCount(); i++) { - String thisMode = flowTbl.getStringValueAt(i, "Mode"); - if (!mode.equals("all") && !thisMode.equalsIgnoreCase(mode) && - repform != reportFormat.internat_internationalPart) continue; - if (!mode.equals("all") && !mode.equalsIgnoreCase("Water") && - repform == reportFormat.internat_internationalPart) continue; - int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); - int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); - int portCode = -1 * (int) flowTbl.getValueAt(i, "PortCode"); - // if flow goes from Savannah GA to Europe with port of exit Savannah GA, don't report domestic part - changed my mind: do report this flow -// if (repform == reportFormat.internat_domesticPart && (destCode == -portCode || origCode == -portCode)) continue; - if (repform == reportFormat.internat_domesticPart) { - if (origCode > highestDomRegion && destCode <= highestDomRegion) - // from abroad to U.S. -> set origin to entry port - origCode = portCode; - else if (origCode <= highestDomRegion && destCode > highestDomRegion) - // from U.S. to abroad -> set destination to exit port - destCode = portCode; - else if (origCode > highestDomRegion && destCode > highestDomRegion) { - // from abroad through U.S. port to abroad - boolean origMexCan = checkIfMexicoOrCanada(flowTbl.getStringValueAt(i, "Origin")); - // from Mexico or Canada through U.S. port to Overseas -> set destination to exit port - if (origMexCan) destCode = portCode; - // from Overseas through U.S. port to Mexico or Canada -> set origin to entry port - else origCode = portCode; - // Note: if both origin and destination are MEX/CAN or both origin and destination are overseas, - // it is impossible to determine which part of the trips was made by mode mode, therefore it is - // reported as from abroad to abroad without noting the entry/exit port [seems not to exist in FAF2 data] - } - } else if (repform == reportFormat.internat_internationalPart) { - if (origCode > highestDomRegion && destCode <= highestDomRegion) - // from abroad to U.S. -> set destination to entry port - destCode = portCode; - else if (origCode <= highestDomRegion && destCode > highestDomRegion) - // from U.S. to abroad -> set origin to exit port - origCode = portCode; - else if (origCode > highestDomRegion && destCode > highestDomRegion) { - // from abroad through U.S. port to abroad - boolean origMexCan = checkIfMexicoOrCanada(flowTbl.getStringValueAt(i, "Origin")); - // from Mexico or Canada through U.S. port to Overseas -> set origin to exit port - if (origMexCan) origCode = portCode; - // from Overseas through U.S. port to Mexico or Canada -> set destination to entry port - else destCode = portCode; - // Note: if both origin and destination are MEX/CAN or both origin and destination are overseas, - // it is impossible to determine which part of the trips was made by mode mode, therefore it is - // reported as from abroad to abroad without noting the entry/exit port [seems not to exist in FAF2 data] - } - } - float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; - String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); - int intSctgCommmodity = sctgCode.get(fafDataCommodity); - String sctgCommodity; - if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; - else sctgCommodity = "SCTG" + intSctgCommmodity; - if (comClass.equals(commodityClassType.STCC)) { - for (String com: stccCommodities) { - float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; - if (flowShare > 0) { - String code; - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + sctgCommodity; - } - if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); - hshFlows.put(code, flowShare); - } - } - } else { - // report by SCTG commodity classification - String code; - if (mode.equals("all")) { - code = origCode + "_" + destCode + "_" + sctgCommodity + "_" + thisMode; - } else { - code = origCode + "_" + destCode + "_" + sctgCommodity; - } - if (hshFlows.containsKey(code)) flows += hshFlows.get(code); - hshFlows.put(code, flows); - } - } - return hshFlows; - } - - - private boolean checkIfMexicoOrCanada (String name) { - // check if name is Canada or Mexico and return true or false - boolean contains = false; - if (name.equals("Canada") || name.equals("Mexico")) contains = true; - return contains; - } - - - public TableDataSet readInternationalCommodityFlowsByAir(ResourceBundle appRb, String unit) { - // read international FAF2 flows that cross the U.S. border by air in unit (tons or dollars) - logger.info ("Reading international FAF2 data by air in " + unit); - - String fileName = ResourceUtil.getProperty(appRb, ("faf2.data.air." + unit)); - TableDataSet flows = fafUtils.importTable(fileName); - if (!referenceListsAreRead) { - readFAF2ReferenceLists(appRb); - referenceListsAreRead = true; - } - int[] originCodes = new int[flows.getRowCount()]; - int[] portOfEntryCodes = new int[flows.getRowCount()]; - int[] destinationCodes = new int[flows.getRowCount()]; - for (int i = 1; i <= flows.getRowCount(); i++) { - //find Origin and Destination codes - originCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Origin")); - portOfEntryCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Coast")); - destinationCodes[i-1] = findZoneCode(flows.getStringValueAt(i, "Destination")); - } - flows.appendColumn(originCodes, "OriginCode"); - flows.appendColumn(portOfEntryCodes, "PortCode"); - flows.appendColumn(destinationCodes, "DestinationCode"); - return flows; - } - - - public HashMap getIntAirFlows(reportFormat repform, String mode, commodityClassType comClass, - int yr, String unit) { - // extract international FAF2 flows (international mode air) - // If repform is set to internat_domesticPart port entry and exit points are set as negative origin or destination codes - - TableDataSet flowTbl; - if (unit.equals("tons")) { - flowTbl = intAirTonCommodityFlows; - } else { - flowTbl = intAirDollarCommodityFlows; - } - - HashMap hshFlows = new HashMap(); - for (int i = 1; i <= flowTbl.getRowCount(); i++) { - // Note: Only mode "Air & Truck" is available in this data set - if (repform == reportFormat.internat_domesticPart && - !flowTbl.getStringValueAt(i, "Mode").equals(mode)) continue; - if (repform == reportFormat.internat_internationalPart && - !flowTbl.getStringValueAt(i, "Mode").equals("Air & Truck")) continue; - int origCode = (int) flowTbl.getValueAt(i, "OriginCode"); - int destCode = (int) flowTbl.getValueAt(i, "DestinationCode"); - int portCode = -1 * (int) flowTbl.getValueAt(i, "PortCode"); - // if flow goes from Houston TX to Europe with port of exit Houston TX, don't report domestic part - changed my mind: do report this flow -// if (repform == reportFormat.internat_domesticPart && (destCode == -portCode || origCode == -portCode)) continue; - if (origCode <= highestDomRegion || destCode <= highestDomRegion) { - // this should be the case for every record, there should be no international to international records - if (repform == reportFormat.internat_domesticPart) { - if (origCode > highestDomRegion) origCode = portCode; - if (destCode > highestDomRegion) destCode = portCode; - } else if (repform == reportFormat.internat_internationalPart) { - if (origCode > highestDomRegion) destCode = portCode; - if (destCode > highestDomRegion) origCode = portCode; - } - } - float flows = flowTbl.getValueAt(i, Integer.toString(yr)) * factor; - String fafDataCommodity = flowTbl.getStringValueAt(i, "Commodity"); - int intSctgCommmodity = sctgCode.get(fafDataCommodity); - String sctgCommodity; - if (intSctgCommmodity <= 9) sctgCommodity = "SCTG0" + intSctgCommmodity; - else sctgCommodity = "SCTG" + intSctgCommmodity; - if (comClass.equals(commodityClassType.STCC)) { - for (String com: stccCommodities) { - float flowShare = SCTGtoSTCCconversion.get(sctgCommodity + "_" + com) * flows; - if (flowShare > 0) { - String code = origCode + "_" + destCode + "_" + com; - if (hshFlows.containsKey(code)) flowShare += hshFlows.get(code); - hshFlows.put(code, flowShare); - } - } - } else { - // report by SCTG commodity classification - String code = origCode + "_" + destCode + "_" + sctgCommodity; - if (hshFlows.containsKey(code)) flows += hshFlows.get(code); - hshFlows.put(code, flows); - } - - } - return hshFlows; - } -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java deleted file mode 100644 index 615d08b..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/readFAF3.java +++ /dev/null @@ -1,520 +0,0 @@ -package org.sandag.htm.processFAF; - -import org.apache.log4j.Logger; - -import java.util.ResourceBundle; -import java.util.HashMap; -import java.io.PrintWriter; - -import com.pb.common.datafile.TableDataSet; -import com.pb.common.util.ResourceUtil; - -/** - * This class reads FAF3 data and stores data in a TableDataSet - * Author: Rolf Moeckel, PB - * Date: September 9, 2010 - */ - -public class readFAF3 { - Logger logger = Logger.getLogger(readFAF3.class); - private int factor; - private String[] valueColumnName; - private TableDataSet faf3commodityFlows; - public static TableDataSet fafRegionList; - private String[] regionState; - private static int[] domRegionIndex; - static public int[] sctgCommodities; - static public String[] sctgStringCommodities; - static private int[] sctgStringIndex; - private static HashMap portsOfEntry; - private static HashMap marinePortsOfEntry; - private static HashMap railPortsOfEntry; - private static HashMap airPortsOfEntry; - private static int[] listOfBorderPortOfEntries; - - - public void readAllData (ResourceBundle appRb, int year, String unit) { - // read input data - - if (ResourceUtil.getBooleanProperty(appRb, "read.in.raw.faf.data", true)) - readAllFAF3dataSets(appRb, unit, year); - readCommodityList(appRb); - readFAF3referenceLists(appRb); - } - - - public void readCommodityList(ResourceBundle appRb) { - // read commodity names - TableDataSet sctgComList = fafUtils.importTable(ResourceUtil.getProperty(appRb, "faf3.sctg.commodity.list")); - sctgCommodities = new int[sctgComList.getRowCount()]; - sctgStringCommodities = new String[sctgCommodities.length]; - for (int i = 1; i <= sctgComList.getRowCount(); i++) { - sctgCommodities[i-1] = (int) sctgComList.getValueAt(i, "SCTG"); - if (sctgCommodities[i-1] < 10) sctgStringCommodities[i-1] = "SCTG0" + sctgCommodities[i-1]; - else sctgStringCommodities[i-1] = "SCTG" + sctgCommodities[i-1]; - } - sctgStringIndex = new int[fafUtils.getHighestVal(sctgCommodities) + 1]; - for (int num = 0; num < sctgCommodities.length; num++) sctgStringIndex[sctgCommodities[num]] = num; - } - - - public int getIndexOfCommodity (int commodity) { - return sctgStringIndex[commodity]; - } - - - public static String getSCTGname(int sctgInt) { - // get String name from sctg number - return sctgStringCommodities[sctgStringIndex[sctgInt]]; - } - - - public static String getFAFzoneName(int fafInt) { - // get String name from int FAF zone code number - return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "FAF3 Zones -Short Description"); - } - - - public static String getFAFzoneState(int fafInt) { - // get String two-letter abbreviation of state of fafInt - return fafRegionList.getStringValueAt(domRegionIndex[fafInt], "State"); - } - - - public void definePortsOfEntry(ResourceBundle appRb) { - // read data to translate ports of entry in network links - - // Border crossings - portsOfEntry = new HashMap<>(); - TableDataSet poe = fafUtils.importTable(appRb.getString("ports.of.entry")); - for (int row = 1; row <= poe.getRowCount(); row++) { - int fafID = (int) poe.getValueAt(row, "faf3id"); - int node = (int) poe.getValueAt(row, "pointOfEntry"); - float weight = poe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (portsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = portsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - portsOfEntry.put(fafID, newPortsOfEntry); - } - listOfBorderPortOfEntries = poe.getColumnAsInt("pointOfEntry"); - - // Marine ports - marinePortsOfEntry = new HashMap<>(); - if (appRb.containsKey("marine.ports.of.entry")) { - TableDataSet mpoe = fafUtils.importTable(appRb.getString("marine.ports.of.entry")); - for (int row = 1; row <= mpoe.getRowCount(); row++) { - int fafID = (int) mpoe.getValueAt(row, "faf3id"); - int node = (int) mpoe.getValueAt(row, "pointOfEntry"); - float weight = mpoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (marinePortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = marinePortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - marinePortsOfEntry.put(fafID, newPortsOfEntry); - } - } - // Rail ports (railyards) - railPortsOfEntry = new HashMap<>(); - if (appRb.containsKey("rail.ports.of.entry")) { - TableDataSet rpoe = fafUtils.importTable(appRb.getString("rail.ports.of.entry")); - for (int row = 1; row <= rpoe.getRowCount(); row++) { - int fafID = (int) rpoe.getValueAt(row, "faf3id"); - int node = (int) rpoe.getValueAt(row, "pointOfEntry"); - float weight = rpoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (railPortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = railPortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - railPortsOfEntry.put(fafID, newPortsOfEntry); - } - } - // Airports - airPortsOfEntry = new HashMap<>(); - if (appRb.containsKey("air.ports.of.entry")) { - TableDataSet apoe = fafUtils.importTable(appRb.getString("air.ports.of.entry")); - for (int row = 1; row <= apoe.getRowCount(); row++) { - int fafID = (int) apoe.getValueAt(row, "faf3id"); - int node = (int) apoe.getValueAt(row, "pointOfEntry"); - float weight = apoe.getValueAt(row, "weight"); - TableDataSet newPortsOfEntry = new TableDataSet(); - if (airPortsOfEntry.containsKey(fafID)) { - TableDataSet existingNodes = airPortsOfEntry.get(fafID); - int[] nodes = existingNodes.getColumnAsInt("COUNTYFIPS"); // use same column labels as for - float[] weights = existingNodes.getColumnAsFloat("Employment"); //county TableDataSets to ease disaggregation - int[] newNodes = fafUtils.expandArrayByOneElement(nodes, node); - float[] newWeights = fafUtils.expandArrayByOneElement(weights, weight); - newPortsOfEntry.appendColumn(newNodes, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(newWeights, "Employment"); - } else { - newPortsOfEntry.appendColumn(new int[]{node}, "COUNTYFIPS"); - newPortsOfEntry.appendColumn(new float[]{weight}, "Employment"); - } - airPortsOfEntry.put(fafID, newPortsOfEntry); - } - } - } - - - public static int[] getListOfBorderPortOfEntries() { - return listOfBorderPortOfEntries; - } - - - - public static TableDataSet getPortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (portsOfEntry.containsKey(fafZone)) return portsOfEntry.get(fafZone); - else return null; - } - - - public static TableDataSet getMarinePortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (marinePortsOfEntry.containsKey(fafZone)) return marinePortsOfEntry.get(fafZone); - else return null; - } - - - public static TableDataSet getAirPortsOfEntry (int fafZone) { - // return list of ports of entry if available, otherwise return fafZone - if (airPortsOfEntry.containsKey(fafZone)) return airPortsOfEntry.get(fafZone); - else return null; - } - - - public void readAllFAF3dataSets2007(ResourceBundle appRb, String unit) { - // read all FAF3 data into TableDataSets in unit (= tons or dollars) - - logger.info ("Reading domestic FAF3 data in " + unit); - if (unit.equals("tons")) { - factor = 1000; // tons are in 1,000s - valueColumnName = new String[]{"tons_2007"}; - } else if (unit.equals("dollars")) { - factor = 1000000; // dollars are in 1,000,000s - valueColumnName = new String[]{"value_2007"}; - } else { - logger.fatal ("Wrong token " + unit + " in method readAllFAF3dataSets2007. Use tons or dollars."); - } - faf3commodityFlows = readFAF3commodityFlows(appRb, unit); - } - - - public double[] summarizeFlowByCommodity (ModesFAF fafMode) { - // sum commodity flows by commodity and return array with total tons - - double[] totalFlows = new double[sctgCommodities.length]; - int modeNum = fafUtils.getEnumOrderNumber(fafMode); - for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { - if (faf3commodityFlows.getValueAt(row, "dms_mode") != modeNum) continue; - int com = sctgStringIndex[(int) faf3commodityFlows.getValueAt(row, "sctg2")]; - if (valueColumnName.length == 1) { - // use year provided by user - totalFlows[com] += faf3commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); - totalFlows[com] += val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - } - return totalFlows; - } - - - public void readAllFAF3dataSets(ResourceBundle appRb, String unit, int year) { - // read all FAF3 data into TableDataSets in unit (= tons or dollars) - - logger.info (" Reading FAF3 data in " + unit); - switch (unit) { - case "tons": - factor = 1000; // tons are provided in 1,000s - break; - case "dollars": - factor = 1000000; // dollars are provided in 1,000,000s - break; - default: - logger.fatal("Wrong token " + unit + " in method readAllFAF3dataSets. Use tons or dollars."); - System.exit(0); - } - int[] availYears = {2007, 2015, 2020, 2025, 2030, 2035, 2040}; - boolean yearInFaf = false; - for (int y: availYears) if (year == y) yearInFaf = true; - if (!yearInFaf) { // interpolate between two years - logger.info(" Year " + year + " does not exist in FAF3 data."); - int year1 = 2007; - int year2 = 2040; - for (int availYear : availYears) if (availYear < year) year1 = availYear; - for (int i = availYears.length - 1; i >= 0; i--) if (availYears[i] > year) year2 = availYears[i]; - logger.info(" FAF3 data are interpolated between " + year1 + " and " + year2 + "."); - // first position: lower year, second position: higher year, third position: steps away from lower year - valueColumnName = new String[]{unit + "_" + year1, unit + "_" + year2, String.valueOf((1f * (year - year1)) / (1f * (year2 - year1)))}; - } else { // use year provided by user - valueColumnName = new String[]{unit + "_" + year}; - } - faf3commodityFlows = readFAF3commodityFlows(appRb, unit); - } - - - private TableDataSet readFAF3commodityFlows(ResourceBundle appRb, String unit) { - // read FAF3 data and return TableDataSet with flow data - String fileName = ResourceUtil.getProperty(appRb, ("faf3.data")); - return fafUtils.importTable(fileName); - } - - - public HashMap createScaler(String[] tokens, double[] values) { - // create HashMap with state O-D pairs that need to be scaled - - HashMap scaler = new HashMap(); - if (tokens.length != values.length) { - throw new RuntimeException("Error. scaling.truck.trips.tokens must be same length as scaling.truck.trips.values"); - } - for (int i=0; i scaler) { - // extract truck flows for year yr and scale flows according to scaler HashMap (no special regions specified) - - PrintWriter outFile = fafUtils.openFileForSequentialWriting(outFileName); - outFile.println("originFAF,destinationFAF,flowDirection," + commodityClassType.SCTG + "_commodity,shortTons"); - int modeNum = fafUtils.getEnumOrderNumber(mode); - for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { - int type = (int) faf3commodityFlows.getValueAt(row, "trade_type"); - double val; - if (valueColumnName.length == 1) { - // use year provided by user - val = faf3commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); - val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - val *= factor * odScaler(row, type, scaler); - if (val == 0) continue; - if (type == 1) writeDomesticFlow(modeNum, val, row, outFile); - else if (type == 2) writeImportFlow(modeNum, val, row, outFile, repF); - else if (type == 3) writeExportFlow(modeNum, val, row, outFile, repF); - else if (type == 4) writeThroughFlow(modeNum, val, row, outFile, repF); - else logger.info("Invalid trade_type in FAF3 dataset in row " + row + ": " + type); - } - outFile.close(); - } - - - public void writeFlowsByModeAndCommodity (String outFileName, ModesFAF mode, reportFormat repF, - HashMap scaler) { - // extract truck flows for year yr and scale flows according to scaler HashMap, including special regions - - PrintWriter outFile[] = new PrintWriter[sctgCommodities.length]; - for (int com: sctgCommodities) { - String fileName; - if (com < 10) fileName = outFileName + "_SCTG0" + com + ".csv"; - else fileName = outFileName + "_SCTG" + com + ".csv"; - outFile[sctgStringIndex[com]] = fafUtils.openFileForSequentialWriting(fileName); - outFile[sctgStringIndex[com]].println("originFAF,destinationFAF,flowDirection,SCTG_commodity,shortTons"); - } - int modeNum = fafUtils.getEnumOrderNumber(mode); - for (int row = 1; row <= faf3commodityFlows.getRowCount(); row++) { - int type = (int) faf3commodityFlows.getValueAt(row, "trade_type"); - double val; - if (valueColumnName.length == 1) { - // use year provided by user - val = faf3commodityFlows.getValueAt(row, valueColumnName[0]); - } else { - // interpolate between two years - float val1 = faf3commodityFlows.getValueAt(row, valueColumnName[0]); - float val2 = faf3commodityFlows.getValueAt(row, valueColumnName[1]); - val = val1 + (val2 - val1) * Float.parseFloat(valueColumnName[2]); - } - val *= factor * odScaler(row, type, scaler); - if (val == 0) continue; - int comIndex = getIndexOfCommodity((int) faf3commodityFlows.getValueAt(row, "sctg2")); - if (type == 1) writeDomesticFlow(modeNum, val, row, outFile[comIndex]); - else if (type == 2) writeImportFlow(modeNum, val, row, outFile[comIndex], repF); - else if (type == 3) writeExportFlow(modeNum, val, row, outFile[comIndex], repF); - else if (type == 4) writeThroughFlow(modeNum, val, row, outFile[comIndex], repF); - else logger.info("Invalid trade_type in FAF3 dataset in row " + row + ": " + type); - } - for (int com: sctgCommodities) outFile[sctgStringIndex[com]].close(); - } - - - public float odScaler (int row, int type, HashMap scaler) { - // find scaler for origin destination pair in row - - int orig; - int dest; - if (type == 1) { - orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - } else if (type == 2) { - orig = (int) faf3commodityFlows.getValueAt(row, "fr_orig"); - dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - } else if (type == 3) { - orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - dest = (int) faf3commodityFlows.getValueAt(row, "fr_dest"); - } else { - orig = (int) faf3commodityFlows.getValueAt(row, "fr_orig"); - dest = (int) faf3commodityFlows.getValueAt(row, "fr_dest"); - } - String stateLevelToken = regionState[orig] + "_" + regionState[dest]; - String combo1Token = orig + "_" + regionState[dest]; - String combo2Token = regionState[orig] + "_" + dest; - String fafLevelToken = orig + "_" + dest; - float adj = 1; - if (scaler.containsKey(stateLevelToken)) adj = scaler.get(stateLevelToken); - if (scaler.containsKey(combo1Token)) adj = scaler.get(combo1Token); - if (scaler.containsKey(combo2Token)) adj = scaler.get(combo2Token); - if (scaler.containsKey(fafLevelToken)) adj = scaler.get(fafLevelToken); - return adj; - } - - - private int tryGettingThisValue(int row, String token) { - // for some flows, international zones/modes are empty -> catch this case and set zone to 0 - - int region; - try { - region = (int) faf3commodityFlows.getValueAt(row, token); - } catch (Exception e) { - region = 0; - } - return region; - } - - - public void writeDomesticFlow (int modeNum, double val, int row, PrintWriter outFile) { - // internal US flow - if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - int dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); - outFile.println(orig + "," + dest + ",domestic," + comm + "," + val); - } - } - - - public void writeImportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // from abroad to US - - int frInMode = (int) faf3commodityFlows.getValueAt(row, "fr_inmode"); - int borderZone = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); - if (frInMode == modeNum && repF != reportFormat.internat_domesticPart) { - int orig = tryGettingThisValue(row, "fr_orig"); - outFile.println(orig + "," + borderZone + ",import," + comm + "," + val); - } - if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int dest = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - String txt; - if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",import_port,"; - else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",import_rail,"; - else if (frInMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",import_airport,"; - else txt = ",import,"; - outFile.println(borderZone + "," + dest + txt + comm + "," + val); - } - } - - - public void writeExportFlow (int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // from US to abroad - int frOutMode = tryGettingThisValue(row, "fr_outmode"); - int borderZone = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - int comm = (int) faf3commodityFlows.getValueAt(row, "sctg2"); - if (frOutMode == modeNum && repF != reportFormat.internat_domesticPart) { - int dest = tryGettingThisValue(row, "fr_dest"); - outFile.println(borderZone + "," + dest + ",export," + comm + "," + val); - } - if (faf3commodityFlows.getValueAt(row, "dms_mode") == modeNum) { - int orig = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - String txt = ",export,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Water)) txt = ",export_port,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Rail)) txt = ",export_rail,"; - if (frOutMode == fafUtils.getEnumOrderNumber(ModesFAF.Air)) txt = ",export_airport,"; - outFile.println(orig + "," + borderZone + txt + comm + "," + val); - } - } - - - public void writeThroughFlow(int modeNum, double val, int row, PrintWriter outFile, reportFormat repF) { - // flows in transit through US - if ((int) faf3commodityFlows.getValueAt(row, "dms_mode") != modeNum) return; - int borderInZone = (int) faf3commodityFlows.getValueAt(row, "dms_orig"); - int borderOutZone = (int) faf3commodityFlows.getValueAt(row, "dms_dest"); - logger.warn("Through flows not yet implemented. This flow from " + borderInZone + " to " + borderOutZone + " is lost."); - } - - - public void readFAF3referenceLists(ResourceBundle rb) { - // read list of regions for FAF3 - String regFileName = rb.getString("faf3.region.list"); - fafRegionList = fafUtils.importTable(regFileName); - int[] reg = fafRegionList.getColumnAsInt(fafRegionList.getColumnPosition("ZoneID")); - domRegionIndex = new int[fafUtils.getHighestVal(reg) + 1]; - for (int num = 0; num < reg.length; num++) domRegionIndex[reg[num]] = num + 1; - regionState = new String[fafUtils.getHighestVal(reg) + 1]; - for (int row = 1; row <= fafRegionList.getRowCount(); row++) { - int zone = (int) fafRegionList.getValueAt(row, "ZoneID"); - regionState[zone] = fafRegionList.getStringValueAt(row, "State"); - } - } - - - public int[] getFAFzoneIDs () { - return fafRegionList.getColumnAsInt("ZoneID"); - } - - - public TableDataSet getFAF3flows() { - return faf3commodityFlows; - } - - - public int getFactor() { - return factor; - } - - - public TableDataSet getFaf3commodityFlows() { - return faf3commodityFlows; - } - - - public String[] getValueColumnName() { - return valueColumnName; - } - -} diff --git a/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java b/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java deleted file mode 100644 index a33927a..0000000 --- a/sandag_abm/src/main/java/org/sandag/htm/processFAF/reportFormat.java +++ /dev/null @@ -1,23 +0,0 @@ -package org.sandag.htm.processFAF; - -/** - * Defines reporting method - * internationalByEntryPoint reports international flows - * internat_domesticPart = International flow from port of entry to final domestic destination or from - * domestic origin to port of exit - * internat_internationalPart = International flow from foreign origin to port of entry or from port of exit to - * foreign destination - * internatOrigToDest = International flow from foreign origin to domestic destination or from domestic - * origin to foreign destination - * internatOrigToBorderToDest = International flow from origin to port of entry/port of exit to destination (2 flows) - * - * User: Rolf Moeckel, PB New York - * Date: May 7, 2009 - */ - -public enum reportFormat { - internat_domesticPart, - internat_internationalPart, - internatOrigToDest, - internatOrigToBorderToDest, -} diff --git a/sandag_abm/src/main/python/assignScenarioID.py b/sandag_abm/src/main/python/assignScenarioID.py deleted file mode 100644 index 0571fe4..0000000 --- a/sandag_abm/src/main/python/assignScenarioID.py +++ /dev/null @@ -1,27 +0,0 @@ -import os -import glob -import pandas as pd -import pyodbc - -toolpath = os.getcwd()[3:-7] - -conn = pyodbc.connect("DRIVER={SQL Server};" - "SERVER=DDAMWSQL16;" - "DATABASE=abm_14_2_0;" - "Trusted_Connection=yes;") -sql = ("SELECT * FROM [abm_14_2_0].[dimension].[scenario] where RIGHT(path, len(path)-23) = '%s'" % toolpath) -df_sql = pd.read_sql_query(sql, conn) -scenid = df_sql['scenario_id'].max() -list = glob.glob(os.getcwd()[:-6]+'report\\hwyload*') -list_shape = glob.glob(os.getcwd()[:-6]+'report\\hwyload*.shp') - -if len(list_shape) and len(df_sql): - for item in list: - if 'csv' not in item: - try: - os.rename(item, os.getcwd()[:-6]+'report\\hwyLoad_'+ str(scenid) + item[-4:]) - except Exception as error: - print('Caught this error: ' + repr(error)) - print ('The scenaio ID has been added to the shapefile.') -else: - print ("Cannot find the scenario in the SQ database or hwyloadshape file is not available. Please check...") \ No newline at end of file diff --git a/sandag_abm/src/main/python/calculate_micromobility.py b/sandag_abm/src/main/python/calculate_micromobility.py deleted file mode 100644 index a277359..0000000 --- a/sandag_abm/src/main/python/calculate_micromobility.py +++ /dev/null @@ -1,273 +0,0 @@ -#!/usr/bin/env python3 -# -*- coding: utf-8 -*- -# Author: RSG Inc. -"""calculate_micromobility.py - -This Python 3 script calculates Generalized Time between MGRAs for micromobility -and microtransit modes in the SANDAG activity-based model. - -Constants are read from the properties file specified at the command line. - -Currently this script uses three files to perform its calculations: - - mgra.socec.file, contains MicroAccessTime for origin MRGAs - - active.logsum.matrix.file.walk.mgra, contains the pre-calculated - walk times between MGRAs - - active.logsum.matrix.file.walk.mgratap, contains the pre-calculated - walk times from MGRAs to TAPs - - active.microtransit.tap.file, contains a list of TAPs with microtransit availability - - active.microtransit.mgra.file, contains a list of MGRAs with microtransit availability - - -The script then writes a fresh MGRA file with the newly calculated micromobility calculations: - - walkTime: the original walk time - - mmTime: the micro-mobility time, including travel time, rental time, access time - - mmCost: micro-mobility variable cost * travel time + fixed cost - - mmGenTime: mmTime + mmCost converted to time + constant - - mtTime: the micro-transit time, including travel time, wait time, access time - - mtCost: micro-transit variable cost * travel time + fixed cost - - minTime: minimum of walkTime, mmGenTime, and mtGenTime - -Run `python calculate_micromobility.py -h` for more command-line usage. -""" - -import argparse -import os -import pandas as pd - - -def process_file(config, zone): - """Performs micromobility calculations using given output_file - and attributes from the provided MGRA file and properties file - - Writes newly calculated micromobility time and intermediate calculations - """ - - filename = config.walk_mgra_output_file if zone == 'mgra' else config.walk_mgra_tap_output_file - - output_file = os.path.join(config.cli.outputs_directory, filename) - config.validate_file(output_file) - - if zone == 'mgra': - walk_time_col = 'actual' - orig_col = 'i' - dest_col = 'j' - - else: - walk_time_col = 'boardingActual' - orig_col = 'mgra' - dest_col = 'tap' - - print('Processing %s ...' % output_file) - df = pd.read_csv(output_file, usecols=[walk_time_col, orig_col, dest_col]) - df.rename(columns={walk_time_col:'walkTime'}, inplace=True) - - # OD vectors - length = df['walkTime'] / config.walk_coef - - # availability masks - if zone == 'mgra': - mt_avail = \ - (df[orig_col].isin(config.mt_mgras) & df[dest_col].isin(config.mt_mgras)) & \ - (length <= config.mt_max_dist_mgra) - - walk_avail = length <= config.walk_max_dist_mgra - mm_avail = length <= config.mm_max_dist_mgra - - else: - mt_avail = \ - df[orig_col].isin(config.mt_mgras) & df[dest_col].isin(config.mt_taps) & \ - (length <= config.mt_max_dist_tap) - walk_avail = length <= config.walk_max_dist_tap - mm_avail = length <= config.mm_max_dist_tap - - all_rows = df.shape[0] - df = df[mt_avail | walk_avail | mm_avail] - print('Filtered out %s unavailable pairs' % str(all_rows - df.shape[0])) - - # micro-mobility - mm_ivt = length * 60 / config.mm_speed # micro-mobility in-vehicle time - orig_mat = df[orig_col].map(config.mat) # micro-access time at origin - mm_time = mm_ivt + config.mm_rental_time + orig_mat # total mm time - mm_cost = config.mm_variable_cost * mm_ivt + config.mm_fixed_cost - mm_cost_as_time = mm_cost * 60 / config.vot - - # micro-transit - mt_ivt = length * 60 / config.mt_speed - mt_time = mt_ivt + 2 * config.mt_wait_time + config.mt_access_time - mt_cost = mt_time * config.mt_variable_cost + config.mt_fixed_cost - mt_cost_as_time = mt_cost * 60 / config.vot - - # save intermediate calculations - df['dist'] = length - df['mmTime'] = mm_time - df['mmCost'] = mm_cost - df['mtTime'] = mt_time - df['mtCost'] = mt_cost - - # calculate micromobility and microtransit Generalized Time - df['mmGenTime'] = mm_time + mm_cost_as_time + config.mm_constant - df['mtGenTime'] = mt_time + mt_cost_as_time + config.mt_constant - - # update zones with unavailable walk, micromobility, and microtransit - df.loc[~walk_avail, ['walkTime']] = config.mt_not_avail - df.loc[~mm_avail, ['mmTime', 'mmCost', 'mmGenTime']] = config.mt_not_avail - df.loc[~mt_avail, ['mtTime', 'mtCost', 'mtGenTime']] = config.mt_not_avail - - # calculate the minimum of walk time vs. generalized time - df['minTime'] = df[['walkTime', 'mmGenTime', 'mtGenTime']].min(axis=1) - - # write output - outfile = os.path.join( - config.cli.outputs_directory, - os.path.basename(output_file).replace('walk', 'micro') - ) - - print("Writing final table to %s" % outfile) - df.to_csv(outfile, index=False) - print("Done.") - - -class Config(): - - def __init__(self): - - self.init_cli_args() - self.init_properties() - self.init_micro_access_time() - self.init_tap_mgra_lists() - - def init_cli_args(self): - """Use argparse to set command-line args - - """ - - self.parser = argparse.ArgumentParser(formatter_class=argparse.ArgumentDefaultsHelpFormatter) - - self.parser.add_argument( - '-p', '--properties_file', - default=os.path.join('..', 'conf', 'sandag_abm.properties'), - help="Java properties file.") - - self.parser.add_argument( - '-o', '--outputs_directory', - default=os.path.join('..', 'output'), - help="Directory containing walk MGRA output files.") - - self.parser.add_argument( - '-i', '--inputs_parent_directory', - default='..', - help="Directory containing 'input' folder") - - self.cli = self.parser.parse_args() - - def validate_file(self, filename): - if not os.path.isfile(filename): - self.parser.print_help() - raise IOError("Could not locate %s" % filename) - - def init_properties(self): - """Parses attributes from a Java properties file - - """ - - filename = self.cli.properties_file - self.validate_file(filename) - print('Parsing tokens from %s ...' % filename) - - all_props = {} - with open(filename, 'r') as f: - for line in f: - if line.startswith("#"): - continue - if '=' in line: - atr, val = list(map(str.strip, line.split('='))) - all_props[atr] = val - - def parse(property_name): - if property_name not in all_props: - raise KeyError("Could not find %s in %s" % (property_name, filename)) - - return all_props.get(property_name) - - self.mgra_file = parse('mgra.socec.file') - self.walk_mgra_output_file = parse('active.logsum.matrix.file.walk.mgra') - self.walk_mgra_tap_output_file = parse('active.logsum.matrix.file.walk.mgratap') - self.mt_tap_file = parse('active.microtransit.tap.file') - self.mt_mgra_file = parse('active.microtransit.mgra.file') - - self.walk_coef = float(parse('active.walk.minutes.per.mile')) - self.walk_max_dist_mgra = float(parse('active.maxdist.walk.mgra')) - self.walk_max_dist_tap = float(parse('active.maxdist.walk.tap')) - - self.vot = float(parse('active.micromobility.vot')) - - self.mm_speed = float(parse('active.micromobility.speed')) - self.mm_rental_time = float(parse('active.micromobility.rentalTime')) - self.mm_constant = float(parse('active.micromobility.constant')) - self.mm_variable_cost = float(parse('active.micromobility.variableCost')) - self.mm_fixed_cost = float(parse('active.micromobility.fixedCost')) - self.mm_max_dist_mgra = float(parse('active.maxdist.micromobility.mgra')) - self.mm_max_dist_tap = float(parse('active.maxdist.micromobility.tap')) - - self.mt_speed = float(parse('active.microtransit.speed')) - self.mt_wait_time = float(parse('active.microtransit.waitTime')) - self.mt_access_time = float(parse('active.microtransit.accessTime')) - self.mt_constant = float(parse('active.microtransit.constant')) - self.mt_variable_cost = float(parse('active.microtransit.variableCost')) - self.mt_fixed_cost = float(parse('active.microtransit.fixedCost')) - self.mt_not_avail = float(parse('active.microtransit.notAvailable')) - self.mt_max_dist_mgra = float(parse('active.maxdist.microtransit.mgra')) - self.mt_max_dist_tap = float(parse('active.maxdist.microtransit.tap')) - - def init_micro_access_time(self): - """Reads the MicroAccessTime for each origin MGRA from - the provided MGRA file. If no MicroAccessTime is found, - a simple calculation is performed instead. - - """ - mgra_file_path = os.path.join(self.cli.inputs_parent_directory, self.mgra_file) - self.validate_file(mgra_file_path) - - with open(mgra_file_path, 'r') as f: - use_dummy = 'MicroAccessTime' not in f.readline() - - if use_dummy: - print('No MicroAccessTime column found in %s, using 2 minute ' - 'default for PARKAREA==1, 15 minutes otherwise.' % mgra_file_path) - park_area = pd.read_csv(mgra_file_path, usecols=['mgra', 'parkarea'], - index_col='mgra', dtype='Int64', squeeze=True) - mat = pd.Series(index=park_area.index, data=15.0, name='MicroAccessTime') - mat.loc[park_area == 1] = 2.0 - else: - mat = pd.read_csv(mgra_file_path, usecols=['mgra', 'MicroAccessTime'], - index_col='mgra', squeeze=True) - - self.mat = mat - - def init_tap_mgra_lists(self): - """Reads in lists of ids that identify micro-transit accessibility TAPs/MGRAs - - """ - mt_tap_file_path = os.path.join(self.cli.inputs_parent_directory, self.mt_tap_file) - mt_mgra_file_path = os.path.join(self.cli.inputs_parent_directory, self.mt_mgra_file) - self.validate_file(mt_tap_file_path) - self.validate_file(mt_mgra_file_path) - - self.mt_taps = \ - pd.read_csv(mt_tap_file_path, - usecols=lambda x: x.strip().lower() == 'tap', - squeeze=True).values - - self.mt_mgras = \ - pd.read_csv(mt_mgra_file_path, - usecols=lambda x: x.strip().lower() == 'mgra', - squeeze=True).values - - -if __name__ == '__main__': - - config = Config() - process_file(config, zone='tap') - process_file(config, zone='mgra') - - print('Finished!') diff --git a/sandag_abm/src/main/python/checkFreeSpace.py b/sandag_abm/src/main/python/checkFreeSpace.py deleted file mode 100644 index d94812f..0000000 --- a/sandag_abm/src/main/python/checkFreeSpace.py +++ /dev/null @@ -1,33 +0,0 @@ -__author__ = 'wsu' -import sys -import ctypes -""" -ctypes is a foreign function library for Python. -It provides C compatible data types, and allows calling functions in DLLs or shared libraries. - -sys provides access to operating system. -It provides access to some variables used by the interpreter. -""" -path=sys.argv[1] -minSpace=sys.argv[2] -_, total, free = ctypes.c_ulonglong(), ctypes.c_ulonglong(), ctypes.c_ulonglong() -if sys.version_info >= (3,) or isinstance(path, unicode): - fun = ctypes.windll.kernel32.GetDiskFreeSpaceExW -else: - fun = ctypes.windll.kernel32.GetDiskFreeSpaceExA -ret = fun(path, ctypes.byref(_), ctypes.byref(total), ctypes.byref(free)) -if ret == 0: - raise ctypes.WinError() -totalMB=total.value/1024.0/1024.0 -freeMB=free.value/2014.0/1024.0 -usedMB = totalMB- freeMB - -if freeMB < int(minSpace): - print "free space on C <",minSpace,"MB!" - sys.exit() -else: - print "Total MB on C:",totalMB - print "Used MB on C:",usedMB - print "Free MB on C:",freeMB - - diff --git a/sandag_abm/src/main/python/check_output.py b/sandag_abm/src/main/python/check_output.py deleted file mode 100644 index b0dc43b..0000000 --- a/sandag_abm/src/main/python/check_output.py +++ /dev/null @@ -1,139 +0,0 @@ -""" Output Checker - -Checks that ABM components successfully generate required files. - -""" - -# Import libraries -import os -import sys - - -# Define model-output dictionary -output_dict = { - "Setup": [ - "walkMgraTapEquivMinutes.csv", - "microMgraTapEquivMinutes.csv", - "microMgraEquivMinutes.csv", - "bikeTazLogsum.csv", - "bikeMgraLogsum.csv", - "walkMgraEquivMinutes.csv" - ], - "SDRM": [ - "wsLocResults_ITER.csv", - "aoResults.csv", - "householdData_ITER.csv", - "indivTourData_ITER.csv", - "indivTripData_ITER.csv", - "jointTourData_ITER.csv", - "jointTripData_ITER.csv", - "personData_ITER.csv" - ], - "IE": [ - "internalExternalTrips.csv" - ], - "SAN": [ - "airport_out.SAN.csv" - ], - "CBX": [ - "airport_out.CBX.csv" - ], - "CBM": [ - "crossBorderTrips.csv", - "crossBorderTours.csv" - ], - "Visitor": [ - "visitorTrips.csv", - "visitorTours.csv" - ], - "TNC": [ - "TNCTrips.csv" - ], - "AV": [ - "householdAVTrips.csv" - ], - "CVM": [ - "Gen and trip sum.csv" - ], - "Exporter": [ - # NOTE: This is an incomplete list of the 40+ output files. - # These shapefiles are last to be generated and their - # existence indicates a successful Data Export. - "hwyLoad.prj", - "hwyLoad.cpg", - "hwyLoad.shx", - "hwyLoad.shp", - "hwyLoad.dbf" - ] -} - - -def check_output(scenario_fp, component, iteration=None): - """ - Checks that a specific ABM component generated - required files. - - :param component: String representing ABM component - :param scenario_fp: String representing scenario file path - :param iteration: Integer representing ABM iteration - :returns: Exit code - """ - - # Get required files - files = output_dict[component] - - # Construct output file path - if component == 'Exporter': - out_dir = 'report' - else: - out_dir = 'output' - output_dir = os.path.join(scenario_fp, out_dir) - - # Check that required files were generated - missing = [] - for file in files: - - # Append iteration integer if needed - if 'ITER' in file: - file = file.replace('ITER', iteration) - - file_path = os.path.join(output_dir, file) - if os.path.exists(file_path): - continue - else: - missing.append(file+'\n') - - # Write out missing files to log file - if len(missing) > 0: - create_log(scenario_fp, component, missing) - return sys.exit(2) - - return sys.exit(0) - - -def create_log(scenario_fp, component, lst): - """ - Creates a log file containing the files an ABM component - failed to generate. - - :param scenario_fp: String representing scenario file path - :param component: ABM component - :param lst: List of missing file names - """ - - # Create log file path - log_fp = os.path.join(scenario_fp, 'logFiles', 'missing_files.log') - - # Create and write to log file - with open(log_fp, "w") as log: - header = "'{}' failed to generate the following files:\n".format(component) - log.write(header) - log.writelines(lst) - - return - - -if __name__ == '__main__': - targets = sys.argv[1:] - check_output(*targets) - diff --git a/sandag_abm/src/main/python/cvm_analysis.zip b/sandag_abm/src/main/python/cvm_analysis.zip deleted file mode 100644 index dd5fd7bcfbcda20b66279f6646d20f9efa93128a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 29456 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 BF;f5l diff --git a/sandag_abm/src/main/python/cvm_input_create.py b/sandag_abm/src/main/python/cvm_input_create.py deleted file mode 100644 index 16868e9..0000000 --- a/sandag_abm/src/main/python/cvm_input_create.py +++ /dev/null @@ -1,462 +0,0 @@ -''' -PURPOSE: -Commercial Vehicle Model (CVM) input file creation - -INSTALL (LIBRARY): -Numpy -https://pypi.python.org/pypi/numpy - -Pandas -https://pypi.python.org/pypi/pandas - -HOW TO RUN: -[script_filepath] [Project Directory] [MGRA filename] [TAZ Centroid flename] [output filename] - -example: cvm_input_create.py "C:\Projects\SANDAG_CTM_Validation\_Tasks" "input/mgra13_based_input2012.csv" "tazcentroids_cvm.csv" "Zonal Properties CVM.csv" - -Note: The script name is sufficient if the run folder is the same as the script folder. Otherwise, a full path of the script would be needed. Also, if the output file does not contain spaces, the command line does not need to have quotation marks around the arguments. - -STEPS: -STEP 0: read mgra socio-economic file -STEP 1: find min and max tazids -STEP 2: read taz centroids. -STEP 3: calculate taz level variables -STEP 4. write to output - -REFERENCES: -Following reference are used for calculations: -T:\devel\CVM\sr13\2012_calib5\input\mgra13_based_input2012_CVM.xlsx -T:\devel\CVM\sr13\2012_calib5\CVM\Zonal Properties SDCVM_SR13 KJS rcu_check.xlsx -The final example product: T:\devel\CVM\sr13\2012_calib5\CVM\Inputs\Zonal Properties CVM.csv - -CREATED BY: -nagendra.dhakar@rsginc.com - -LAST MODIFIED: -03/13/2018 - -Updates: -03/13/2018 - nagendra.dhakar@rsginc.com -Updated to remove transformation of coordinates -Instead an input file is provided with taz centroids (tazcentroids_cvm.csv -the script now reads the input centroid file that already has transformed coordinates that are reported in the output -no need for GDAL library -''' - -import sys -import os -import numpy as np -import datetime -import math -import csv -import pandas as pd - -class Constant: - """ Represents constants in the script - constants: FEET_TO_MILE, COORDS_EPSG_SOURCE,COORDS_EPSG_TARGET - """ - ACRES_TO_SQMILE = 0.0015625 - #COORDS_EPSG_SOURCE = 2230 #EPSG: 2230 - NAD83/ California zone 6 (ftUS) - #COORDS_EPSG_TARGET = 3310 #EPSG: 3310 - NAD83/ California Albers - -class Header: - """ Represents header for output files """ - temptazdatafile = ['taz','pop','hh','i1','i2','i3','i4','i5','i6','i7','i8','i9','i10','emp_total','emp_fed_mil','sqmile','land_sqmile','Emp_IN','Emp_RE','Emp_SV','Emp_TU','Emp_WH','Emp_OFF', - 'HHIncome','EmpDens','PopDens','Per/Emp','Emp_ServRet_Pct','Ret_ServRet','Emp_Office','Low Density','Residential','Commercial','Industrial','Employment Node', - 'Industrial_pct','TU_pct','Wholesale_pct','Retail_pct','Service_pct','Office_pct','E500_Industrial','E500_TU','E500_Wholesale','E500_Retail','E500_Service', - 'E500_Office','RetailZone','ZoneType'] - - temptazcentroidsfile = ["hnode","x_coord_spft","y_coord_spft","x_coord_albers","y_coord_albers"] - - outfile = ['TAZ','Pop','Income','Area_SqMi','x-meters','y-meters','EmpDens','PopDens','TotEmp','Military', - 'CVM_IN','CVM_RE','CVM_SV','CVM_TH','CVM_WH','CVM_GO','CVM_LU_Type','SqrtArea','CVM_LU_Low','CVM_LU_Res', - 'CVM_LU_Ret','CVM_LU_Ind','CVM_LU_Emp','Emp_LU_Lo','Emp_LU_Re','Emp_LU_RC','Emp_LU_In','Emp_LU_EN'] - -class TazId: - """ Represents Taz Id range - attributes: min, max - """ - -class Input: - """ Represents input settings - attributes: dir, mgrafile, nodebinfile. - """ -class Output: - """ Represents output settings - attributes: dir, outfile, temp_centroidfile, temp_tazfile - """ - -def read_node_header(node_file): - """Returns name and type of the fields in the node bin file """ - - # node header file - nodeheaderfile = os.path.join(os.path.splitext(node_file)[0]+".DCB") - - # first row is blank - # second row is total bytes in a row - # start reading fields names from third row - # header file format - field_name, type, start_byte, length, .. - - fields_info=[] - with open(nodeheaderfile) as headerfile: - reader = csv.reader(headerfile) - i=0 - for row in reader: - if i >= 2: - field_name = row[0] - field_length = row[3] - - # integer - if row[1] == 'I': - field_type = int - - # character/string - elif row[1] == 'C': - field_type = 'S' + field_length - - # first field - if i == 2: - fields_info.append([field_name]) - fields_info.append([field_type]) - - # remaining fields - else: - fields_info[0].append(field_name) - fields_info[1].append(field_type) - - i=i+1 - - return fields_info - -def get_tazid_range(data): - """Returns min and maz taz ids""" - - # max and min tazid - - id_list = np.array(data.keys()) - id_list = id_list.astype(np.float) - - id_max = int(max(id_list)) - id_min = int(min(id_list)) - - return([id_min,id_max]) - -def read_node_file(tazcentroid_file): - """ - Reads taz centroids from node bin file - Transforms them into the coordinate system expected by the CTM - Returns transformed (projected) coordinates - """ - - centroids = pd.read_csv(tazcentroid_file) - centroids = centroids[centroids['taz']>0] - centroids = centroids.reset_index() - - coords_proj = {} - for i in range(0,max(centroids['taz'])): - coords_proj[str(centroids['taz'][i])] = [float(centroids['x_coord_albers'][i]), float(centroids['y_coord_albers'][i])] - - return coords_proj - -def read_mgra_input(mgrafile): - """ - Reads MGRA socia-economic file - Calculates some variables at MGRA level - Aggregates data by TAZ - Returns TAZ level data - """ - - data={} - - with open (mgrafile) as csvfile: - reader = csv.DictReader(csvfile) # read in dictionary format - i=1 - for row in reader: - # calculate new variables at MGRA - row['sqmile'] = float(row['acres'])*Constant.ACRES_TO_SQMILE - row['land_sqmile'] = float(row['land_acres'])*Constant.ACRES_TO_SQMILE - - row['CVM_IN'] = float(row['emp_ag']) + float(row['emp_const_non_bldg_prod']) + float(row['emp_const_non_bldg_office']) + \ - float(row['emp_const_bldg_prod']) + float(row['emp_const_bldg_office']) + float(row['emp_mfg_prod']) + float(row['emp_mfg_office']) - - row['CVM_RE'] = float(row['emp_retail']) - - row['CVM_SV'] = float(row['emp_pvt_ed_k12']) + float(row['emp_pvt_ed_post_k12_oth']) + float(row['emp_health']) + \ - float(row['emp_personal_svcs_office']) + float(row['emp_amusement']) + float(row['emp_hotel']) + \ - float(row['emp_restaurant_bar']) + float(row['emp_personal_svcs_retail']) + float(row['emp_religious']) + \ - float(row['emp_pvt_hh']) + float(row['emp_public_ed']) - - row['CVM_TH'] = float(row['emp_utilities_prod']) + float(row['emp_utilities_office']) + float(row['emp_trans']) - - row['CVM_WH'] = float(row['emp_whsle_whs']) - - - row['CVM_OFF'] = float(row['emp_prof_bus_svcs']) + float(row['emp_prof_bus_svcs_bldg_maint']) + \ - float(row['emp_state_local_gov_ent']) + float(row['emp_fed_non_mil']) + float(row['emp_state_local_gov_blue']) + \ - float(row['emp_state_local_gov_white']) + float(row['emp_own_occ_dwell_mgmt']) - - # aggregate data by TAZ - if row['taz'] not in data: - data[row['taz']] = [int(row['taz']),int(row['pop']), int(row['hh']), float(row['i1']), float(row['i2']), float(row['i3']), float(row['i4']), float(row['i5']), float(row['i6']), float(row['i7']), - float(row['i8']), float(row['i9']), float(row['i10']), float(row['emp_total']), float(row['emp_fed_mil']), float(row['sqmile']), - float(row['land_sqmile']), float(row['CVM_IN']), float(row['CVM_RE']), float(row['CVM_SV']), float(row['CVM_TH']), float(row['CVM_WH']), float(row['CVM_OFF'])] - - else: - #taz_data[row['TAZ']][0] = int(row['TAZ']) - data[row['taz']][1] += int(row['pop']) - data[row['taz']][2] += int(row['hh']) - data[row['taz']][3] += float(row['i1']) - data[row['taz']][4] += float(row['i2']) - data[row['taz']][5] += float(row['i3']) - data[row['taz']][6] += float(row['i4']) - data[row['taz']][7] += float(row['i5']) - data[row['taz']][8] += float(row['i6']) - data[row['taz']][9] += float(row['i7']) - data[row['taz']][10] += float(row['i8']) - data[row['taz']][11] += float(row['i9']) - data[row['taz']][12] += float(row['i10']) - data[row['taz']][13] += float(row['emp_total']) - data[row['taz']][14] += float(row['emp_fed_mil']) - data[row['taz']][15] += float(row['sqmile']) - data[row['taz']][16] += float(row['land_sqmile']) - data[row['taz']][17] += float(row['CVM_IN']) - data[row['taz']][18] += float(row['CVM_RE']) - data[row['taz']][19] += float(row['CVM_SV']) - data[row['taz']][20] += float(row['CVM_TH']) - data[row['taz']][21] += float(row['CVM_WH']) - data[row['taz']][22] += float(row['CVM_OFF']) - - return data - -# calculate taz level variables -def calculate_taz_variables(data, coords, taz_ids, outfile): - """ - Reads TAZ data stored in read_mgra_input - Calculates variables - Returns calculated taz variables that would be in the output file - """ - - data_calc = {} - with open(outfile,"wb") as csvfile: - fieldnames = Header.temptazdatafile - - writer = csv.writer(csvfile) - writer.writerow(fieldnames) - - for taz in range(1, taz_ids.max+1): - - if taz>=taz_ids.min: - - # initialize variables to 0 - emp_dens, pop_dens, cvm_emp_dens, emp_cvm_total=(0,)*4 - per_emp, emp_servret_pct, ret_servret, emp_office, low_dens, residential, commercial, industrial=(0,)*8 - industrial_pct, transport_pct, wholesale_pct, retail_pct, service_pct, office_pct=(0,)*6 - emp_office, retail_zone=(0,)*2 - industrial_e500, transport_e500, wholesale_e500, retail_e500, service_e500, office_e500=(0,)*6 - cvm_lu_low, cvm_lu_res, cvm_lu_ret, cvm_lu_ind, cvm_lu_emp=(0,)*5 - - # get data - [taz_id,pop,hh,inc1,inc2,inc3,inc4,inc5,inc6,inc7,inc8,inc9,inc10,emp_total,emp_fed_mil,sqmile,land_sqmile,cvm_in,cvm_re,cvm_sv,cvm_th,cvm_wh,cvm_off]=data[str(taz)] - - # average hh income - if (hh>0): - hh_income = (inc1*7500+inc2*22500+inc3*37500+inc4*52500+inc5*67500+inc6*87500+inc7*112500+inc8*137500+inc9*175000+inc10*225)/hh - else: - hh_income=64678 - - # total CVM employment = industrial + retail + service + transport + wholesale + office - emp_cvm_total = cvm_in + cvm_re + cvm_sv + cvm_th + cvm_wh + cvm_off - - # calculate densities - if (land_sqmile > 0): - emp_dens = emp_total/land_sqmile - pop_dens = pop/land_sqmile - cvm_emp_dens = emp_cvm_total/land_sqmile - - # share of employment in each sector - if (emp_total > 0): - per_emp = pop/emp_total - emp_servret_pct = (cvm_re + cvm_sv + cvm_off)/emp_total - - if ((cvm_re+cvm_sv+cvm_off)/emp_total) < 0.8: - emp_office = 1 - - # additional shares/variables - not for the final output - industrial_pct = cvm_in/emp_total - transport_pct = cvm_th/emp_total - wholesale_pct = cvm_wh/emp_total - retail_pct = cvm_re/emp_total - service_pct = cvm_sv/emp_total - office_pct = cvm_off/emp_total - - if cvm_re/emp_total > 0.5: - retail_zone = 1 - - # end of additional variables - - # calculate flags - if ((cvm_re+cvm_sv) > 0): - if ((cvm_re/(cvm_re+cvm_sv+cvm_off)) > 0.25): - ret_servret = 1 - - # landuse flags - - if (emp_dens < 250 and pop_dens < 250): - # low density - low_dens = 1 - - if (low_dens == 0 and pop_dens > 250 and per_emp > 2): - # residential - residential = 1 - - if (low_dens == 0 and residential == 0 and emp_servret_pct > 0.6 and emp_dens > 1500 and ret_servret == 1): - # retail/commercial - commercial = 1 - - if (low_dens == 0 and residential == 0 and commercial == 0 and emp_dens < 15000 and emp_office == 1): - # industrial - industrial = 1 - - if (low_dens == 1 or residential == 1 or commercial == 1 or industrial == 1): - employment_node = 0 - else: - # other - employment_node = 1 - - # employment more than 500 flags - additional, not for the final output - if cvm_in > 500: - industrial_e500 = 1 - if cvm_th > 500: - transport_e500 = 1 - if cvm_wh > 500: - wholesale_e500 = 1 - if cvm_re > 500: - retail_e500 = 1 - if cvm_sv > 500: - service_e500 = 1 - if cvm_off > 500: - office_e500 = 1 - - # end of additional variables - - # zone type - zone_type = 1*low_dens + 2*residential + 3*commercial + 4*industrial + 5*employment_node - sqrt_area = math.sqrt(land_sqmile) - - # TAZ centroids - taz_x_meters = coords[str(taz)][0] - taz_y_meters = coords[str(taz)][1] - - # landuse flags - if zone_type == 1: - # low density - cvm_lu_low = 1 - elif zone_type == 2: - # residential - cvm_lu_res = 1 - elif zone_type == 3: - # retail/commercial - cvm_lu_ret = 1 - elif zone_type == 4: - # industrial - cvm_lu_ind = 1 - elif zone_type == 5: - # other - cvm_lu_emp = 1 - - # employment by land use - emp_lu_low = cvm_lu_low * emp_cvm_total - emp_lu_res = cvm_lu_res * emp_cvm_total - emp_lu_ret = cvm_lu_ret * emp_cvm_total - emp_lu_ind = cvm_lu_ind * emp_cvm_total - emp_lu_emp = cvm_lu_emp * emp_cvm_total - - # write all taz data to a temp file - data[str(taz)].extend([hh_income,emp_dens,pop_dens,per_emp,emp_servret_pct,ret_servret,emp_office, - low_dens,residential,commercial,industrial,employment_node,industrial_pct, - transport_pct,wholesale_pct,retail_pct,service_pct,office_pct,industrial_e500, - transport_e500,wholesale_e500,retail_e500,service_e500,office_e500,retail_zone,zone_type]) - writer.writerow(data[str(taz)]) - - # store calculated variables - data_calc[str(taz)]=[taz_id,pop,hh_income,land_sqmile,taz_x_meters,taz_y_meters,cvm_emp_dens,pop_dens,emp_cvm_total, - emp_fed_mil,cvm_in,cvm_re,cvm_sv,cvm_th,cvm_wh,cvm_off,zone_type,sqrt_area, - cvm_lu_low,cvm_lu_res,cvm_lu_ret,cvm_lu_ind,cvm_lu_emp, - emp_lu_low,emp_lu_res,emp_lu_ret,emp_lu_ind,emp_lu_emp] - return data_calc - -def write_output(data, taz_ids, outfile): - """ - Writes taz data to an output - """ - - with open(outfile,"wb") as csvfile: - fieldnames = Header.outfile - - writer = csv.writer(csvfile) - writer.writerow(fieldnames) - - for taz in range(1, taz_ids.max+1): - if taz None: - self.scenario_path = scenario_path - - @property - @lru_cache(maxsize=1) - def mgra_xref(self) -> pd.DataFrame: - """ Cross reference of Master Geographic Reference Area (MGRA) model - geography to Transportation Analysis Zone (TAZ) and Land Use Zone - (LUZ) model geographies. Cross reference is stored in each ABM - scenario input MGRA file (input/mgra13_based_input<>.csv). - """ - - # load the mgra based input file - fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" - - mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), - usecols=["mgra", # MGRA geography - "taz", # TAZ geography - "luz_id"], - dtype={"mgra": "int16", - "taz": "int16", - "luz_id": "int16"}) # LUZ geography - - # genericize column names - mgra.rename(columns={"mgra": "MGRA", - "taz": "TAZ", - "luz_id": "LUZ"}, - inplace=True) - - return mgra - - @property - @lru_cache(maxsize=1) - def pnr_taps(self) -> pd.DataFrame: - """ Create the transit TAP park and ride lot data-set. - - Read in and combine the transit TAP parking lot type information and - parking lot vehicles by time of day information. - - Returns: - A Pandas DataFrame of the transit TAP park and ride lot data-set """ - # load parking lot type data-set - lots = pd.read_fwf( - os.path.join(self.scenario_path, "input", "tap.ptype"), - names=["TAP", - "lotID", - "parkingType", - "lotTAZ", - "capacity", - "distance", - "mode"], - header=None, - widths=[5, 6, 6, 5, 5, 5, 3]) - - # replicate parking lot data by ABM five time of day - five_tod = pd.DataFrame( - {"key": [0]*5, - "timeFiveTod": ["EA", "AM", "MD", "PM", "EV"]}) - - lots["key"] = 0 - lots = lots.merge(five_tod) - - # load parking lot vehicles by time of day - vehicles = pd.read_csv( - os.path.join(self.scenario_path, "output", "PNRByTAP_Vehicles.csv"), - usecols=["TAP", - "EA", - "AM", - "MD", - "PM", - "EV"]) - - # restructure vehicle data from wide to long by ABM five time of day - vehicles = pd.melt( - vehicles, - id_vars=["TAP"], - value_vars=["EA", "AM", "MD", "PM", "EV"], - var_name="timeFiveTod", - value_name="vehicles") - - # merge parking lot and vehicle data - lots = lots.merge(vehicles, how="left") - - # set missing vehicle fields to 0 - lots["vehicles"] = lots["vehicles"].fillna(0) - - # convert distance field from feet to miles - lots["distance"] = lots["distance"] / 5280 - - # apply exhaustive field mappings where applicable - mappings = { - "timeFiveTod": {"EA": 1, - "AM": 2, - "MD": 3, - "PM": 4, - "EV": 5}, - "parkingType": {1: "Formal Parking", - 2: "Other Parking", - 3: "Other Light Rail Trolley Parking", - 4: "Non-formal parking area based on the on-board survey", - 5: "Non-formal parking area based on the on-board survey"} - } - - for field in mappings: - lots[field] = lots[field].map(mappings[field]) - - # rename columns to standard/generic ABM naming conventions - lots.rename(columns={"TAP": "tapID"}, inplace=True) - - return lots[["tapID", - "lotID", - "lotTAZ", - "timeFiveTod", - "parkingType", - "capacity", - "distance", - "vehicles"]] - - @property - @lru_cache(maxsize=1) - def properties(self) -> dict: - """ Get the ABM scenario properties from the ABM scenario - properties file (conf/sandag_abm.properties). - - The return dictionary contains the following ABM scenario properties: - cvmScaleLight - commercial vehicle model trip scaling factor for - light vehicles for each ABM five time of day - cvmScaleMedium - commercial vehicle model trip scaling factor for - intermediate and medium vehicles for each ABM five time of day - cvmScaleHeavy - commercial vehicle model trip scaling factor for - heavy vehicles for each ABM five time of day - cvmShareLight - commercial vehicle model trip intermediate vehicle - share factor for light vehicles - cvmShareMedium - commercial vehicle model trip intermediate vehicle - share factor for medium vehicles - cvmShareHeavy - commercial vehicle model trip intermediate vehicle - share factor for heavy vehicles - iterations - number of model iteration - nonPooledTNCPassengers - average number of passengers to assume for - Non-Pooled TNC mode in models without party size specifications - pooledTNCPassengers - average number of passengers to assume for - Pooled TNC mode in models without party size specifications - sr2Passengers - average number of passengers to assume for Shared - Ride 2 mode in models without party size specifications - sr3Passengers - average number of passengers to assume for Shared - Ride 3+ mode in models without party size specifications - taxiPassengers - average number of passengers to assume for Taxi - mode in models without party size specifications - timePeriodWidthTNC - time period width (in minutes) for custom - fixed-width time periods used in TNC routing model, note that - this is not currently restricted to nest within ABM model time - periods - sampleRate - sample rate of final iteration - valueOfTimeLow - upper limit of 'Low' value of time category - valueOfTimeMedium - upper limit of 'Medium' value of time category - year - analysis year of the ABM scenario - - Returns: - A dictionary defining the ABM scenario properties. """ - - # create dictionary holding ABM properties file information - # each property contains a dictionary {line, value} where the line - # is the string to match in the properties file to - # return the value of the property - lookup = { - "cvmScaleLight": { - "line": "cvm.scale_light=", - "type": "list", - "value": None}, - "cvmScaleMedium": { - "line": "cvm.scale_medium=", - "type": "list", - "value": None}, - "cvmScaleHeavy": { - "line": "cvm.scale_heavy=", - "type": "list", - "value": None}, - "cvmShareLight": { - "line": "cvm.share.light=", - "type": "float", - "value": None}, - "cvmShareMedium": { - "line": "cvm.share.medium=", - "type": "float", - "value": None}, - "cvmShareHeavy": { - "line": "cvm.share.heavy=", - "type": "float", - "value": None}, - "iterations": { - "line": None, - "type": "int", - "value": None}, - "nonPooledTNCPassengers": { - "line": "TNC.single.passengersPerVehicle=", - "type": "float", - "value": None}, - "pooledTNCPassengers": { - "line": "TNC.shared.passengersPerVehicle=", - "type": "float", - "value": None}, - "sr2Passengers": { - "line": None, - "type": "int", - "value": 2}, - "sr3Passengers": { - "line": None, - "type": "float", - "value": 3.34}, - "taxiPassengers": { - "line": "Taxi.passengersPerVehicle=", - "type": "float", - "value": None}, - "timePeriodWidthTNC": { - "line": "Maas.RoutingModel.minutesPerSimulationPeriod=", - "type": "int", - "value": None}, - "sampleRate": { - "line": "sample_rates=", - "type": "float", - "value": None}, - "valueOfTimeLow": { - "line": "valueOfTime.threshold.low=", - "type": "float", - "value": None}, - "valueOfTimeMedium": { - "line": "valueOfTime.threshold.med=", - "type": "float", - "value": None}, - "year": { - "line": "scenarioYear=", - "type": "int", - "value": None} - } - - # open the ABM scenario properties file - file = open(os.path.join(self.scenario_path, "conf", "sandag_abm.properties"), "r") - - # loop through each line of the properties file - for line in file: - # strip all white space from the line - line = line.replace(" ", "") - - # for each element of the properties dictionary - for name in lookup: - item = lookup[name] - - # if the properties file contains the matching line - if item["line"] is not None: - match = re.compile(item["line"]).match(line) - else: - match = False - - if match: - # if the match is for the sample rate element - # then take the portion of the line after the matching string - # and split by the comma character - if name == "sampleRate": - line = line[match.end():].split(",") - - # set number of iterations to number of sample rates - # that are specified - lookup["iterations"]["value"] = len(line) - - # if the split line contains a single element - # return that element otherwise return the final element - if len(line) == 1: - value = line[0] - else: - value = line[-1] - # if the match is for a cvm scale element then take the - # portion of the line after the matching string and split - # by the comma character into a list of floats - elif "cvmScale" in name: - value = line[match.end():].split(",") - value = list(map(float, value)) - # otherwise take the final element of the line - else: - value = line[match.end():] - - # update the dictionary value using the appropriate data type - if item["type"] == "float": - value = float(value) - elif item["type"] == "int": - value = int(value) - else: - pass - - item["value"] = value - - break - - file.close() - - # convert the property name and value to a non-nested dictionary - results = {} - for name in lookup: - results[name] = lookup[name]["value"] - - return results - - @property - def time_periods(self) -> dict: - """ Dictionary of ABM model time resolution periods with start and - end times where the start time is inclusive and the end time is - exclusive. Dictionary is of the form: - {"period": "startTime": "endTime":} - - Returns: - A Dictionary of the ABM model time resolution periods. - """ - periods = { - "abmHalfHour": [ - {"period": 1, - "startTime": time(3, 0), - "endTime": time(5, 0)}, - {"period": 2, - "startTime": time(5, 0), - "endTime": time(5, 30)}, - {"period": 3, - "startTime": time(5, 30), - "endTime": time(6, 0)}, - {"period": 4, - "startTime": time(6, 0), - "endTime": time(6, 30)}, - {"period": 5, - "startTime": time(6, 30), - "endTime": time(7, 0)}, - {"period": 6, - "startTime": time(7, 0), - "endTime": time(7, 30)}, - {"period": 7, - "startTime": time(7, 30), - "endTime": time(8, 0)}, - {"period": 8, - "startTime": time(8, 0), - "endTime": time(8, 30)}, - {"period": 9, - "startTime": time(8, 30), - "endTime": time(9, 0)}, - {"period": 10, - "startTime": time(9, 0), - "endTime": time(9, 30)}, - {"period": 11, - "startTime": time(9, 30), - "endTime": time(10, 0)}, - {"period": 12, - "startTime": time(10, 0), - "endTime": time(10, 30)}, - {"period": 13, - "startTime": time(10, 30), - "endTime": time(11, 0)}, - {"period": 14, - "startTime": time(11, 0), - "endTime": time(11, 30)}, - {"period": 15, - "startTime": time(11, 30), - "endTime": time(12, 0)}, - {"period": 16, - "startTime": time(12, 0), - "endTime": time(12, 30)}, - {"period": 17, - "startTime": time(12, 30), - "endTime": time(13, 0)}, - {"period": 18, - "startTime": time(13, 0), - "endTime": time(13, 30)}, - {"period": 19, - "startTime": time(13, 30), - "endTime": time(14, 0)}, - {"period": 20, - "startTime": time(14, 0), - "endTime": time(14, 30)}, - {"period": 21, - "startTime": time(14, 30), - "endTime": time(15, 0)}, - {"period": 22, - "startTime": time(15, 0), - "endTime": time(15, 30)}, - {"period": 23, - "startTime": time(15, 30), - "endTime": time(16, 0)}, - {"period": 24, - "startTime": time(16, 0), - "endTime": time(16, 30)}, - {"period": 25, - "startTime": time(16, 30), - "endTime": time(17, 0)}, - {"period": 26, - "startTime": time(17, 0), - "endTime": time(17, 30)}, - {"period": 27, - "startTime": time(17, 30), - "endTime": time(18, 0)}, - {"period": 28, - "startTime": time(18, 0), - "endTime": time(18, 30)}, - {"period": 29, - "startTime": time(18, 30), - "endTime": time(19, 0)}, - {"period": 30, - "startTime": time(19, 0), - "endTime": time(19, 30)}, - {"period": 31, - "startTime": time(19, 30), - "endTime": time(20, 0)}, - {"period": 32, - "startTime": time(20, 0), - "endTime": time(20, 30)}, - {"period": 33, - "startTime": time(20, 30), - "endTime": time(21, 0)}, - {"period": 34, - "startTime": time(21, 0), - "endTime": time(21, 30)}, - {"period": 35, - "startTime": time(21, 30), - "endTime": time(22, 0)}, - {"period": 36, - "startTime": time(22, 0), - "endTime": time(22, 30)}, - {"period": 37, - "startTime": time(22, 30), - "endTime": time(23, 0)}, - {"period": 38, - "startTime": time(23, 0), - "endTime": time(23, 30)}, - {"period": 39, - "startTime": time(23, 30), - "endTime": time.max}, - {"period": 40, - "startTime": time.min, - "endTime": time(3, 0)} - ], - "abm5Tod": [ - {"period": 1, - "startTime": time(3, 0), - "endTime": time(6, 0)}, - {"period": 2, - "startTime": time(6, 0), - "endTime": time(9, 0)}, - {"period": 3, - "startTime": time(9, 0, 0), - "endTime": time(15, 30)}, - {"period": 4, - "startTime": time(15, 30), - "endTime": time(19, 0)}, - {"period": 5, - "startTime": time(19, 0), - "endTime": time.max}, - {"period": 5, - "startTime": time.min, - "endTime": time(3, 0)} - ] - } - - return periods - - @staticmethod - def _map_time_periods(abm_half_hour: pd.Series) -> pd.Series: - """ Map ABM half hour time periods to ABM five time of day periods - - Returns: - A Pandas Series of ABM five time of day periods """ - - conditions = [abm_half_hour.between(1, 3), - abm_half_hour.between(4, 9), - abm_half_hour.between(10, 22), - abm_half_hour.between(23, 29), - abm_half_hour.between(30, 40)] - - choices = [1, 2, 3, 4, 5] - - abm_5_tod = np.select(conditions, choices, default=np.NaN) - - return pd.Series(abm_5_tod).astype("float") - - def _map_vot_categories(self, vot: pd.Series) -> pd.Series: - """ Map Pandas Series of continuous ABM value of time (vot) values to - vot categories ("Low", "Medium", "High") defined in the ABM scenario - properties file. - - Returns: - A Pandas Series of value of time categories. """ - - # get vot thresholds - low = self.properties["valueOfTimeLow"] - med = self.properties["valueOfTimeMedium"] - - # map continuous values of time to categories - conditions = [vot < low, - (low <= vot) & (vot < med), - vot >= med] - - choices = ["Low", "Medium", "High"] - - vot_category = np.select(conditions, choices, default=np.NaN) - - return pd.Series(vot_category).astype("category") - - -class LandUse(ScenarioData): - """ A subclass of the ScenarioData class. Holds all land use - data for a completed ABM scenario model run. As of now, this includes only - the MGRA-based input file. This is held as a class property. - - Properties: - mgra_input: MGRA-based input file - """ - @property - @lru_cache(maxsize=1) - def mgra_input(self) -> pd.DataFrame: - """ Create the MGRA-based input file data-set. """ - # load the MGRA-based input file - fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" - - mgra = pd.read_csv( - os.path.join(self.scenario_path, "input", fn), - usecols=["mgra", - "taz", - "hs", - "hs_sf", - "hs_mf", - "hs_mh", - "hh", - "hh_sf", - "hh_mf", - "hh_mh", - "gq_civ", - "gq_mil", - "i1", - "i2", - "i3", - "i4", - "i5", - "i6", - "i7", - "i8", - "i9", - "i10", - "hhs", - "pop", - "hhp", - "emp_ag", - "emp_const_non_bldg_prod", - "emp_const_non_bldg_office", - "emp_utilities_prod", - "emp_utilities_office", - "emp_const_bldg_prod", - "emp_const_bldg_office", - "emp_mfg_prod", - "emp_mfg_office", - "emp_whsle_whs", - "emp_trans", - "emp_retail", - "emp_prof_bus_svcs", - "emp_prof_bus_svcs_bldg_maint", - "emp_pvt_ed_k12", - "emp_pvt_ed_post_k12_oth", - "emp_health", - "emp_personal_svcs_office", - "emp_amusement", - "emp_hotel", - "emp_restaurant_bar", - "emp_personal_svcs_retail", - "emp_religious", - "emp_pvt_hh", - "emp_state_local_gov_ent", - "emp_fed_non_mil", - "emp_fed_mil", - "emp_state_local_gov_blue", - "emp_state_local_gov_white", - "emp_public_ed", - "emp_own_occ_dwell_mgmt", - "emp_fed_gov_accts", - "emp_st_lcl_gov_accts", - "emp_cap_accts", - "emp_total", - "enrollgradekto8", - "enrollgrade9to12", - "collegeenroll", - "othercollegeenroll", - "adultschenrl", - "ech_dist", - "hch_dist", - "pseudomsa", - "parkarea", - "hstallsoth", - "hstallssam", - "hparkcost", - "numfreehrs", - "dstallsoth", - "dstallssam", - "dparkcost", - "mstallsoth", - "mstallssam", - "mparkcost", - "zip09", - "parkactive", - "openspaceparkpreserve", - "beachactive", - "hotelroomtotal", - "truckregiontype", - "district27", - "milestocoast", - "acres", - "effective_acres", - "land_acres", - "MicroAccessTime", - "remoteAVParking", - "refueling_stations", - "totint", - "duden", - "empden", - "popden", - "retempden", - "totintbin", - "empdenbin", - "dudenbin", - "PopEmpDenPerMi"]) - - return mgra - - -class SyntheticPopulation(ScenarioData): - """ A subclass of the ScenarioData class. Holds all synthetic population - data for a completed ABM scenario model run. This includes the input - synthetic persons and households sampled in the ABM model run as well as - model results pertaining to person and household attributes (e.g. work - location, parking reimbursement, etc...). The synthetic population persons - and households are held as class properties and include: - Synthetic Households - Synthetic Persons - - Properties: - households: Synthetic households sampled - persons: Synthetic persons sampled - """ - @property - @lru_cache(maxsize=1) - def households(self) -> pd.DataFrame: - """ Create the synthetic households data-set. - - Read in the input synthetic household list and the sampled synthetic - household list, combine the lists taking only sampled households, - map field values, and genericize field names. - - Returns: - A Pandas DataFrame of the synthetic households """ - # load input synthetic household list into Pandas DataFrame - input_households = pd.read_csv( - os.path.join(self.scenario_path, "input", "households.csv"), - usecols=["hhid", - "taz", - "mgra", - "hinccat1", - "hinc", - "hworkers", - "persons", - "bldgsz", - "unittype", - "poverty"], - dtype={"hhid": "int32", - "taz": "int16", - "mgra": "int16", - "hinccat1": "int8", - "hinc": "int32", - "hworkers": "int8", - "persons": "int8", - "bldgsz": "int8", - "unittype": "int8", - "poverty": "float32"}) - - # load output sampled synthetic household list - fn = "householdData_" + str(self.properties["iterations"]) + ".csv" - output_households = pd.read_csv( - os.path.join(self.scenario_path, "output", fn), - usecols=["hh_id", - "autos", - "HVs", - "AVs", - "transponder"], - dtype={"hh_id": "int32", - "autos": "int8", - "HVs": "int8", - "AVs": "int8", - "transponder": "bool"}) - - # merge output sampled households with input sampled households - # keep only households present in the sampled households - households = output_households.merge( - input_households, - how="inner", - left_on="hh_id", - right_on="hhid" - ) - - # apply exhaustive field mappings where applicable - mappings = { - "hinccat1": {1: "Less than 30k", - 2: "30k-60k", - 3: "60k-100k", - 4: "100k-150k", - 5: "150k+"}, - "bldgsz": {1: "Mobile Home or Trailer", - 2: "Single Family Home - Detached", - 3: "Single Family Home - Attached", - 8: "Multi-Family Home", - 9: "Other (includes Group Quarters)"}, - "unittype": {0: "Non-Group Quarters", - 1: "Group Quarters"} - } - - for field in mappings: - households[field] = households[field].map(mappings[field]).astype("category") - - # rename columns to standard/generic ABM naming conventions - households.rename(columns={"hh_id": "hhId", - "HVs": "autosHumanVehicles", - "AVs": "autosAutonomousVehicles", - "transponder": "transponderAvailable", - "mgra": "homeMGRA", - "taz": "homeTAZ", - "hinccat1": "hhIncomeCategory", - "hinc": "hhIncome", - "hworkers": "hhWorkers", - "persons": "hhPersons", - "bldgsz": "buildingCategory", - "unittype": "unitType"}, - inplace=True) - - return households[["hhId", - "autos", - "autosHumanVehicles", - "autosAutonomousVehicles", - "transponderAvailable", - "homeMGRA", - "homeTAZ", - "hhIncomeCategory", - "hhIncome", - "hhWorkers", - "hhPersons", - "buildingCategory", - "unitType", - "poverty"]] - - @property - @lru_cache(maxsize=1) - def persons(self) -> pd.DataFrame: - """ Create the synthetic persons data-set. - - Read in the input synthetic person list and the sampled synthetic - person list, combine the lists taking only sampled persons, - map field values, and genericize field names. - - Returns: - A Pandas DataFrame of the synthetic persons """ - # load input synthetic person list into Pandas DataFrame - input_persons = pd.read_csv( - os.path.join(self.scenario_path, "input", "persons.csv"), - usecols=["hhid", - "perid", - "pnum", - "age", - "sex", - "miltary", - "pemploy", - "pstudent", - "ptype", - "educ", - "grade", - "weeks", - "hours", - "rac1p", - "hisp"], - dtype={"hhid": "int32", - "perid": "int32", - "pnum": "int8", - "age": "int8", - "sex": "int8", - "miltary": "int8", - "pemploy": "int8", - "pstudent": "int8", - "ptype": "int8", - "educ": "int8", - "grade": "int8", - "weeks": "int8", - "hours": "int8", - "rac1p": "int8", - "hisp": "int8"}) - - # load output sampled synthetic person list - fn_person_data = "personData_" + str(self.properties["iterations"]) + ".csv" - output_persons = pd.read_csv( - os.path.join(self.scenario_path, "output", fn_person_data), - usecols=["person_id", - "activity_pattern", - "fp_choice", - "reimb_pct", - "tele_choice"], - dtype={"person_id": "int32", - "activity_pattern": "string", - "fp_choice": "int8", - "reimb_pct": "float32", - "tele_choice": "int8"}) - - # load work-school location model results - fn_ws_loc_results = "wsLocResults_" + str(self.properties["iterations"]) + ".csv" - ws_loc_results = pd.read_csv( - os.path.join(self.scenario_path, "output", fn_ws_loc_results), - usecols=["PersonID", - "HomeMGRA", - "WorkSegment", - "SchoolSegment", - "WorkLocation", - "SchoolLocation"], - dtype={"PersonID": "int32", - "HomeMGRA": "int16", - "WorkSegment": "int32", - "SchoolSegment": "int32", - "WorkLocation": "int32", - "SchoolLocation": "int32"}) - - # merge output sampled persons with input sampled persons - # keep only persons present in the sampled persons - persons = output_persons.merge( - input_persons, - how="inner", - left_on="person_id", - right_on="perid" - ) - - # merge in work-school location model results - persons = persons.merge( - ws_loc_results, - how="inner", - left_on="person_id", - right_on="PersonID" - ) - - # if person works at home set work location to home MGRA - # if person is home-school set school location to home MGRA - persons["WorkLocation"] = np.where( - persons["WorkSegment"] == 99999, - persons["HomeMGRA"], - persons["WorkLocation"]) - - persons["SchoolLocation"] = np.where( - persons["SchoolSegment"] == 88888, - persons["HomeMGRA"], - persons["SchoolLocation"]) - - # apply exhaustive field mappings where applicable - mappings = { - "sex": {1: "Male", - 2: "Female"}, - "miltary": {0: "Not Active Military", - 1: "Active Military"}, - "pemploy": {1: "Employed Full-Time", - 2: "Employed Part-Time", - 3: "Unemployed or Not in Labor Force", - 4: "Less than 16 Years Old"}, - "pstudent": {1: "Pre K-12", - 2: "College Undergrad+Grad and Prof. School", - 3: "Not Attending School"}, - "ptype": {1: "Full-Time Worker", - 2: "Part-Time Worker", - 3: "College Student", - 4: "Non-Working Adult", - 5: "Non-Working Senior", - 6: "Driving Age Student", - 7: "Non-Driving Age Student", - 8: "Pre K or Child too Young for School"}, - "educ": {1: "Not a High School Graduate", - 9: "High School Graduate or Associates Degree", - 13: "Bachelors Degree or Higher"}, - "grade": {0: "Preschool or Not Attending School", - 2: "Kindergarten - Grade 8", - 5: "Grade 9 to Grade 12", - 6: "College Undergraduate or Higher"}, - "weeks": {1: "27 or More Weeks Worked per Year", - 5: "Less than 27 Weeks Worked per Year"}, - "hours": {0: "Less than 35 Hours Worked or Not Working", - 35: "35 or More Hours Worked"}, - "rac1p": {1: "White Alone", - 2: "Black or African American Alone", - 3: "American Indian Alone", - 4: "Alaska Native Alone", - 5: "American Indian and Alaska Native Tribes specified; or American Indian or Alaska Native not specified and no other races", - 6: "Asian Alone", - 7: "Native Hawaiian and Other Pacific Islander Alone", - 8: "Some Other Race Alone", - 9: "Two or More Major Race Groups"}, - "hisp": {1: "Non-Hispanic", - 2: "Hispanic"}, - "activity_pattern": {"H": "Home", - "M": "Mandatory", - "N": "Non-Mandatory"}, - "fp_choice": {1: "Has Free Parking", - 2: "Employer Pays for Parking", - 3: "Employer Reimburses for Parking"}, - "tele_choice": {0: "No telecommute", - 1: "One Day a Week", - 2: "Two-Three Days a Week", - 3: "Four or More Days a Week", - 9: "Telecommuter Only"}, - "WorkSegment": {0: "Management Business Science and Labor", - 1: "Services Labor", - 2: "Sales and Office Labor", - 3: "Natural Resources Construction and Maintenance Labor", - 4: "Production Transportation and Material Moving Labor", - 5: "Military Labor", - 99999: "Work from Home"}, - "SchoolSegment": {**{88888: "Home Schooled"}, - **{key: value for (key, value) in zip(list(range(0, 57)), - ["Unknown"] * 57)} - }, - "WorkLocation": {key: value for (key, value) in zip(list(range(1, 23003)), - list(range(1, 23003)))}, - "SchoolLocation": {key: value for (key, value) in zip(list(range(1, 23003)), - list(range(1, 23003)))} - } - - for field in mappings: - if field in ["WorkLocation", "SchoolLocation"]: - persons[field] = persons[field].map(mappings[field]).astype("float32") - else: - persons[field] = persons[field].map(mappings[field]).astype("category") - - # if employer does not reimburse for parking - # set parking reimbursement percentage to missing - persons.loc[persons["fp_choice"] != "Employer Reimburses for Parking", "reimb_pct"] = np.nan - persons.loc[persons["fp_choice"].isna(), "reimb_pct"] = np.nan - - # rename columns to standard/generic ABM naming conventions - persons.rename(columns={"perid": "personId", - "hhid": "hhId", - "pnum": "personNumber", - "miltary": "militaryStatus", - "pemploy": "employmentStatus", - "pstudent": "studentStatus", - "ptype": "abmPersonType", - "educ": "education", - "rac1p": "race", - "hisp": "hispanic", - "activity_pattern": "abmActivityPattern", - "fp_choice": "freeParkingChoice", - "reimb_pct": "parkingReimbursementPercentage", - "tele_choice": "telecommuteChoice", - "WorkSegment": "workSegment", - "SchoolSegment": "schoolSegment", - "WorkLocation": "workLocation", - "SchoolLocation": "schoolLocation"}, - inplace=True) - - return persons[["personId", - "hhId", - "personNumber", - "age", - "sex", - "militaryStatus", - "employmentStatus", - "studentStatus", - "abmPersonType", - "education", - "grade", - "weeks", - "hours", - "race", - "hispanic", - "abmActivityPattern", - "freeParkingChoice", - "parkingReimbursementPercentage", - "telecommuteChoice", - "workSegment", - "schoolSegment", - "workLocation", - "schoolLocation"]] - - -class TourLists(ScenarioData): - """ A subclass of the ScenarioData class. Holds all tour list data for a - completed ABM scenario model run. This includes all data from the ABM - sub-models with tours. These are held as class properties and include: - Cross Border Model - Commercial Vehicle Model - Internal-External Model - Individual Model - Joint Model - Visitor Model - - The tour list data is loaded from raw ABM output files in the scenario - output folder and transformed where applicable. - - Properties: - cross_border: Mexican Resident Cross Border model tour list - cvm: Commercial Vehicle model tour list - ie: San Diego Resident Internal-External model tour list - individual: San Diego Resident Individual travel model tour list - joint: San Diego Resident Joint travel model tour list - Visitor: Visitor model tour list - """ - @property - @lru_cache(maxsize=1) - def cross_border(self) -> pd.DataFrame: - """ Create the Cross-border Model tour list. - - Read in the Cross-border tour list, map field values, and genericize - field names. - - Returns: - A Pandas DataFrame of the Cross-border tour list """ - - # load tour list into Pandas DataFrame - tours = pd.read_csv( - os.path.join(self.scenario_path, "output", "crossBorderTours.csv"), - usecols=["id", - "purpose", - "sentri", - "poe", - "departTime", - "arriveTime", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"], - dtype={"id": "int32", - "purpose": "int8", - "sentri": "boolean", - "poe": "int8", - "departTime": "int8", - "arriveTime": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tourMode": "int8"}) - - # apply exhaustive field mappings where applicable - mappings = { - "purpose": {0: "Work", - 1: "School", - 2: "Cargo", - 3: "Shop", - 4: "Visit", - 5: "Other"}, - "poe": {0: "San Ysidro", - 1: "Otay Mesa", - 2: "Tecate", - 3: "Otay Mesa East", - 4: "Jacumba"}, - "tourMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk"} - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTime) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTime) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"id": "tourID", - "purpose": "tourPurpose", - "poe": "pointOfEntry", - "departTime": "departTimeAbmHalfHour", - "arriveTime": "arriveTimeAbmHalfHour"}, - inplace=True) - - return tours[["tourID", - "tourPurpose", - "sentri", - "pointOfEntry", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"]] - - @property - @lru_cache(maxsize=1) - def cvm(self) -> pd.DataFrame: - """ Create the Commercial Vehicle Model tour list. - - Read in the Commercial Vehicle trip lists, apply share allocation, map - field values, genericize field names, and create the tour list from the - trip list. - - Returns: - A Pandas DataFrame of the Commercial Vehicle tour list """ - # create list of all Commercial Vehicle model trip list files - # files are of the form Trip_<>_<> - files = ["Trip" + "_" + i + "_" + j + ".csv" for i, j in - itertools.product(["FA", "GO", "IN", "RE", "SV", "TH", "WH"], - ["OE", "AM", "MD", "PM", "OL"])] - - # read all trip list files into a Pandas DataFrame - trips = pd.concat(( - pd.read_csv(os.path.join(self.scenario_path, "output", file), - usecols=["SerialNo", - "Trip", - "ActorType", - "HomeZone", - "Mode", - "StartTime", - "EndTime", - "TourType", - "OriginalTimePeriod"], - dtype={"SerialNo": "int32", - "Trip": "int8", - "ActorType": "string", - "HomeZone": "int16", - "Mode": "string", - "StartTime": "float32", - "EndTime": "float32", - "TourType": "string", - "OriginalTimePeriod": "string"}) - for file in files)) - - # apply re-allocation originally implemented in - # Java by Nagendra Dhakar + Joel Freedman at RSG - - # create lookup table of mode-tod-share using scenario properties - lookup = pd.DataFrame( - {"Mode": ["L"] * 5 + ["I"] * 5 + ["M"] * 5 + ["H"] * 5, - "OriginalTimePeriod": ["OE", "AM", "MD", "PM", "OL"] * 4, - "cvmShare": [self.properties["cvmShareLight"]] * 5 + - [0] * 5 + - [self.properties["cvmShareMedium"]] * 5 + - [self.properties["cvmShareHeavy"]] * 5}) - - # merge trip list and lookup table - trips = trips.merge(lookup) - - # within each mode, the properties file designates a percentage of the - # trip weight to be removed from the original trip and given to a new - # identical trip with the "I" (light-heavy duty truck) mode - new_trips = trips.loc[trips["cvmShare"] > 0].copy() - new_trips.reset_index(drop=True, inplace=True) - new_trips["Mode"] = "I" - trips = pd.concat([trips, new_trips], ignore_index=True) - - # create tour surrogate key - # unique tour is defined by (SerialNo, Mode) - trips["tourID"] = trips.groupby(["SerialNo", "Mode"]).ngroup().astype("int32") + 1 - - # create tour list using the first and last trip within each tour - # all tour data constant across trips excepting start/end times - # first trip provides start time, last trip provides end time - tours = trips.sort_values(by=["tourID", "Trip"]).groupby(["tourID"]) - tours = tours.head(1).merge(tours.tail(1)[["tourID", "EndTime"]], - on="tourID", - suffixes=("_start", "")) - - # apply exhaustive field mappings where applicable - mappings = { - "ActorType": {"FA": "Fleet Allocator", - "GO": "Government\\Office", - "IN": "Industry", - "RE": "Retail", - "SV": "Service", - "TH": "Transport", - "WH": "Wholesale"}, - "Mode": {"L": "Drive Alone", - "I": "Light Heavy Duty Truck", - "M": "Medium Heavy Duty Truck", - "H": "Heavy Heavy Duty Truck"}, - "TourType": {"G": "Goods", - "S": "Service", - "O": "Other"} - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # map continuous start and end times to ABM half hour time periods - # times are in continuous hours of the day (0-24) and can wrap into - # the following day or even multiple following days (>24) with no - # upper limit - - # create times from continuous hour start and end times - # taking into account their wrapping into subsequent days - tours["StartTime"] = tours["StartTime"].apply( - lambda x: (datetime.combine(date.today(), time.min) + - timedelta(hours=(x % 24))).time()) - tours["EndTime"] = tours["EndTime"].apply( - lambda x: (datetime.combine(date.today(), time.min) + - timedelta(hours=(x % 24))).time()) - - # map continuous times to abm half hour periods - depart_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in tours["StartTime"]] - depart_half_hour = [val for sublist in depart_half_hour for val in sublist] - tours = tours.assign(departTimeAbmHalfHour=depart_half_hour) - tours["departTimeAbmHalfHour"] = tours["departTimeAbmHalfHour"].astype("int8") - - arrive_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in tours["EndTime"]] - arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] - tours = tours.assign(arriveTimeAbmHalfHour=arrive_half_hour) - tours["arriveTimeAbmHalfHour"] = tours["arriveTimeAbmHalfHour"].astype("int8") - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTimeAbmHalfHour) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTimeAbmHalfHour) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"ActorType": "actorType", - "TourType": "tourPurpose", - "HomeZone": "originTAZ", - "Mode": "tourMode"}, - inplace=True) - - return tours[["tourID", - "actorType", - "tourPurpose", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originTAZ", - "tourMode"]] - - @property - @lru_cache(maxsize=1) - def ie(self) -> pd.DataFrame: - """ Create the Internal-External Model tour list. - - Read in the Internal-External trip list, map field values, genericize - field names, and create the tour list from the trip list. - - Returns: - A Pandas DataFrame of the Internal-External tour list """ - - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "internalExternalTrips.csv"), - usecols=["personID", - "tourID", - "inbound", - "period", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tripMode"], - dtype={"personID": "int32", - "tourID": "int32", - "inbound": "boolean", - "period": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tripMode": "int8"}) - - # create tour list using the first and last trip within each tour - # all tour data constant across trips excepting start/end times - # first trip provides start time, last trip provides end time - # first trip also provides the tour destination - tours = trips.sort_values(by=["tourID", "inbound"]).groupby(["tourID"]) - tours = tours.head(1).merge(tours.tail(1)[["tourID", "period"]], - on="tourID", - suffixes=("Start", "End")) - - # apply exhaustive field mappings where applicable - mappings = { - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.periodStart) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.periodEnd) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"periodStart": "departTimeAbmHalfHour", - "periodEnd": "arriveTimeAbmHalfHour", - "tripMode": "tourMode"}, - inplace=True) - - return tours[["tourID", - "personID", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"]] - - @property - @lru_cache(maxsize=1) - def individual(self) -> pd.DataFrame: - """ Create the Individual Model tour list. - - Read in the Individual tour list, map field values and genericize - field names. - - Returns: - A Pandas DataFrame of the Individual tour list """ - - # load tour list into Pandas DataFrame - fn = "indivTourData_" + str(self.properties["iterations"]) + ".csv" - tours = pd.read_csv( - os.path.join(self.scenario_path, "output", fn), - usecols=["person_id", - "tour_id", - "tour_category", - "tour_purpose", - "orig_mgra", - "dest_mgra", - "start_period", - "end_period", - "tour_mode"], - dtype={"person_id": "int32", - "tour_id": "int8", - "tour_category": "string", - "tour_purpose": "string", - "orig_mgra": "int16", - "dest_mgra": "int16", - "start_period": "int8", - "end_period": "int8", - "tour_mode": "int8"}) - - # apply exhaustive field mappings where applicable - mappings = { - "tour_category": {"AT_WORK": "At-Work", - "INDIVIDUAL_NON_MANDATORY": "Individual Non-Mandatory", - "MANDATORY": "Mandatory"}, - "tour_mode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC", - 13: "School Bus"} - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # create tour surrogate key (person_id, tour_id, tour_purpose) - tour_key = ["person_id", "tour_id", "tour_purpose"] - tours["tourID"] = tours.groupby(tour_key).ngroup().astype("int32") + 1 - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - tours = tours.merge(taz_info, left_on="orig_mgra", right_on="MGRA") - tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - tours = tours.merge(taz_info, left_on="dest_mgra", right_on="MGRA") - tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.start_period) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.end_period) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"person_id": "personID", - "tour_category": "tourCategory", - "tour_purpose": "tourPurpose", - "start_period": "departTimeAbmHalfHour", - "end_period": "arriveTimeAbmHalfHour", - "orig_mgra": "originMGRA", - "dest_mgra": "destinationMGRA", - "tour_mode": "tourMode"}, - inplace=True) - - return tours[["tourID", - "personID", - "tourCategory", - "tourPurpose", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"]] - - @property - @lru_cache(maxsize=1) - def joint(self) -> pd.DataFrame: - """ Create the Joint Model tour list. - - Read in the Joint tour list, map field values and genericize field - names. - - Returns: - A Pandas DataFrame of the Joint tour list """ - - # load tour list into Pandas DataFrame - fn = "jointTourData_" + str(self.properties["iterations"]) + ".csv" - tours = pd.read_csv( - os.path.join(self.scenario_path, "output", fn), - usecols=["hh_id", - "tour_id", - "tour_category", - "tour_purpose", - "tour_participants", - "orig_mgra", - "dest_mgra", - "start_period", - "end_period", - "tour_mode"], - dtype={"hh_id": "int32", - "tour_id": "int8", - "tour_category": "string", - "tour_purpose": "string", - "tour_participants": "string", - "orig_mgra": "int16", - "dest_mgra": "int16", - "start_period": "int8", - "end_period": "int8", - "tour_mode": "int8"}) - - # apply exhaustive field mappings where applicable - mappings = { - "tour_category": {"JOINT_NON_MANDATORY": "Joint Non-Mandatory"}, - "tour_mode": {2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"} - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # create tour surrogate key (hh_id, tour_id) - tour_key = ["hh_id", "tour_id"] - tours["tourID"] = tours.groupby(tour_key).ngroup().astype("int32") + 1 - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - tours = tours.merge(taz_info, left_on="orig_mgra", right_on="MGRA") - tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - tours = tours.merge(taz_info, left_on="dest_mgra", right_on="MGRA") - tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.start_period) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.end_period) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"hh_id": "hhID", - "tour_category": "tourCategory", - "tour_purpose": "tourPurpose", - "tour_participants": "tourParticipants", - "start_period": "departTimeAbmHalfHour", - "end_period": "arriveTimeAbmHalfHour", - "orig_mgra": "originMGRA", - "dest_mgra": "destinationMGRA", - "tour_mode": "tourMode"}, - inplace=True) - - return tours[["tourID", - "hhID", - "tourParticipants", - "tourCategory", - "tourPurpose", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"]] - - @property - @lru_cache(maxsize=1) - def visitor(self) -> pd.DataFrame: - """ Create the Visitor Model tour list. - - Read in the Visitor tour list, map field values and genericize field - names. - - Returns: - A Pandas DataFrame of the Visitor tour list """ - - # load tour list into Pandas DataFrame - tours = pd.read_csv( - os.path.join(self.scenario_path, "output", "visitorTours.csv"), - usecols=["id", - "segment", - "purpose", - "partySize", - "income", - "departTime", - "arriveTime", - "originMGRA", - "destinationMGRA", - "tourMode"], - dtype={"id": "int32", - "segment": "int8", - "purpose": "int8", - "partySize": "int8", - "income": "int8", - "departTime": "int8", - "arriveTime": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "tourMode": "int8"}) - - # apply exhaustive field mappings where applicable - mappings = { - "segment": {0: "Business", - 1: "Personal"}, - "purpose": {0: "Work", - 1: "Recreation", - 2: "Dining"}, - "income": {0: "Less than 30k", - 1: "30k-60k", - 2: "60k-100k", - 3: "100k-150k", - 4: "150k+"}, - "tourMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"} - } - - for field in mappings: - tours[field] = tours[field].map(mappings[field]).astype("category") - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - tours = tours.merge(taz_info, left_on="originMGRA", right_on="MGRA") - tours.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - tours = tours.merge(taz_info, left_on="destinationMGRA", right_on="MGRA") - tours.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - tours["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.departTime) - tours["arriveTimeFiveTod"] = self._map_time_periods(abm_half_hour=tours.arriveTime) - - # rename columns to standard/generic ABM naming conventions - tours.rename(columns={"id": "tourID", - "purpose": "tourPurpose", - "departTime": "departTimeAbmHalfHour", - "arriveTime": "arriveTimeAbmHalfHour"}, - inplace=True) - - return tours[["tourID", - "segment", - "tourPurpose", - "partySize", - "income", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tourMode"]] - - -class TripLists(ScenarioData): - """ A subclass of the ScenarioData class. Holds all trip list data for a - completed ABM scenario model run. This includes all data from the ten ABM - sub-models. These are held as class properties and include: - Airport (CBX) Model - Airport (SAN) Model - Cross Border Model - Commercial Vehicle Model - External-External Model - External-Internal Model - Internal-External Model - Individual Model - Joint Model - Truck Model - Visitor Model - Zombie AV Trips - Zombie TNC Trips - - The trip list data is loaded from raw ABM output files in the scenario - output folder and given consistent fields and field values. - - Methods: - _combine_mode_set: Combines ABM mode field with transit skim set - field - _combine_mode_walk: Recodes ABM mode field using the ABM walk mode - field for walk mode trips - - Properties: - airport_cbx: Cross Border Express (CBX) model trip list - airport_san: San Diego Airport (SAN) model trip list - cross_border: Mexican Resident Cross Border model trip list - cvm: Commercial Vehicle model trip list - ee: External-External model trip list, placed here until it can be - properly incorporated into the EMME data exporter process - ei: External-Internal model trip list, placed here until it can be - properly incorporated into the EMME data exporter process - ie: San Diego Resident Internal-External model trip list - individual: San Diego Resident Individual travel model trip list - joint: San Diego Resident Joint travel model trip list - truck: Truck model trip list, placed here until it can be - properly incorporated into the EMME data exporter process - visitor: Visitor model trip list - zombie_av: 0-passenger Autonomous Vehicle trip list - zombie_tnc: 0-passenger TNC Vehicle trip list - """ - - @staticmethod - def _combine_mode_set(mode: pd.Series, transit_set: pd.Series) -> pd.Series: - """ Combine Pandas Series of ABM mode field values with Pandas Series - of ABM transit skim set field values. - - Returns: - A Pandas Series of the combined mode and transit skim set field - values - """ - - # ensure series are string data type - mode = mode.astype("string") - transit_set = transit_set.astype("string") - - # if ABM mode field value contains the string Transit - # append the transit skim set field value - mode = np.where(mode.str.contains("Transit"), - mode + " - " + transit_set, - mode) - - return pd.Series(mode).astype("category") - - @staticmethod - def _combine_mode_walk(mode: pd.Series, walk_mode: pd.Series) -> pd.Series: - """ Combine Pandas Series of ABM mode field values with Pandas Series - of ABM walk mode (micro_walkMode) field values. Update the ABM mode - field value to the indicated ABM walk mode where appropriate. - - Returns: - A Pandas Series of the recoded ABM mode field values. - """ - - # ensure series are string data type - mode = mode.astype("string") - walk_mode = walk_mode.astype("string") - - # if ABM mode field value is Walk then use the ABM walk mode field - # value as the ABM mode, otherwise use the ABM mode field value - mode = np.where(mode == "Walk", - walk_mode, - mode) - - return pd.Series(mode).astype("category") - - @property - @lru_cache(maxsize=1) - def airport_cbx(self) -> pd.DataFrame: - """ Create the Cross Border Express (CBX) Airport Model trip list. - - Read in the CBX trip list, map field values, and genericize field - names. - - Returns: - A Pandas DataFrame of the CBX trip list """ - - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "airport_out.CBX.csv"), - usecols=["id", - "direction", - "purpose", - "size", - "income", - "nights", - "departTime", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tripMode", - "arrivalMode", - "boardingTAP", - "alightingTAP", - "set", - "valueOfTime"], - dtype={"id": "int32", - "direction": "bool", - "purpose": "int8", - "size": "int8", - "income": "int8", - "nights": "int8", - "departTime": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tripMode": "int8", - "arrivalMode": "int8", - "boardingTAP": "int16", - "alightingTAP": "int16", - "set": "int8", - "valueOfTime": "float32"}) - - # apply exhaustive field mappings where applicable - mappings = { - "purpose": {0: "Resident Business", - 1: "Resident Personal", - 2: "Visitor Business", - 3: "Visitor Personal", - 4: "External"}, - "income": {0: "Less than 25k", - 1: "25k-50k", - 2: "50k-75k", - 3: "75k-100k", - 4: "100k-125k", - 5: "125k-150k", - 6: "150k-200k", - 7: "200k+"}, - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "arrivalMode": {1: "Parking lot terminal", - 2: "Parking lot off-site San Diego airport area", - 3: "Parking lot off-site private", - 4: "Pickup/Drop-off escort", - 5: "Pickup/Drop-off curbside", - 6: "Rental car", - 7: "Taxi", - 8: "Non-Pooled TNC", - 9: "Pooled TNC", - 10: "Shuttle/van/courtesy vehicle", - 11: "Transit"}, - "boardingTAP": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "alightingTAP": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {-1: "", - 0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"} - } - - for field in mappings: - if field in ["boardingTAP", "alightingTAP"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.departTime) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, transit_set=trips.set) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories(vot=trips.valueOfTime) - - # no airport trips use transponders or AVs - # no airport trips are allowed to park in another MGRA - # there is no destination purpose - trips["transponderAvailable"] = False - trips["avUsed"] = False - trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - trips["weightTrip"] = 1 / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - trips["weightPersonTrip"] = pd.Series(trips["size"] / self.properties["sampleRate"], dtype="float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"id": "tripID", - "direction": "inbound", - "purpose": "tripPurpose", - "income": "incomeCategory", - "nights": "nightsStayed", - "departTime": "departTimeAbmHalfHour"}, - inplace=True) - - return trips[["tripID", - "inbound", - "tripPurpose", - "incomeCategory", - "nightsStayed", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "arrivalMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"]] - - @property - @lru_cache(maxsize=1) - def airport_san(self) -> pd.DataFrame: - """ Create the San Diego (SAN) Airport Model trip list. - - Read in the SAN trip list, map field values, and genericize field - names. - - Returns: - A Pandas DataFrame of the SAN trip list """ - - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "airport_out.SAN.csv"), - usecols=["id", - "direction", - "purpose", - "size", - "income", - "nights", - "departTime", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tripMode", - "arrivalMode", - "boardingTAP", - "alightingTAP", - "set", - "valueOfTime"], - dtype={"id": "int32", - "direction": "bool", - "purpose": "int8", - "size": "int8", - "income": "int8", - "nights": "int8", - "departTime": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tripMode": "int8", - "arrivalMode": "int8", - "boardingTAP": "int16", - "alightingTAP": "int16", - "set": "int8", - "valueOfTime": "float32"}) - - # apply exhaustive field mappings where applicable - mappings = { - "purpose": {0: "Resident Business", - 1: "Resident Personal", - 2: "Visitor Business", - 3: "Visitor Personal", - 4: "External"}, - "income": {0: "Less than 25k", - 1: "25k-50k", - 2: "50k-75k", - 3: "75k-100k", - 4: "100k-125k", - 5: "125k-150k", - 6: "150k-200k", - 7: "200k+"}, - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "arrivalMode": {1: "Parking lot terminal", - 2: "Parking lot off-site San Diego airport area", - 3: "Parking lot off-site private", - 4: "Pickup/Drop-off escort", - 5: "Pickup/Drop-off curbside", - 6: "Rental car", - 7: "Taxi", - 8: "Non-Pooled TNC", - 9: "Pooled TNC", - 10: "Shuttle/van/courtesy vehicle", - 11: "Transit"}, - "boardingTAP": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "alightingTAP": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {-1: "", - 0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"} - } - - for field in mappings: - if field in ["boardingTAP", "alightingTAP"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.departTime) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, transit_set=trips.set) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories(vot=trips.valueOfTime) - - # no airport trips use transponders or AVs - # no airport trips are allowed to park in another MGRA - # there is no destination purpose - trips["transponderAvailable"] = False - trips["avUsed"] = False - trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - trips["weightTrip"] = 1 / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - trips["weightPersonTrip"] = pd.Series(trips["size"] / self.properties["sampleRate"], dtype="float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"id": "tripID", - "direction": "inbound", - "purpose": "tripPurpose", - "income": "incomeCategory", - "nights": "nightsStayed", - "departTime": "departTimeAbmHalfHour"}, - inplace=True) - - return trips[["tripID", - "inbound", - "tripPurpose", - "incomeCategory", - "nightsStayed", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "arrivalMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"]] - - @property - @lru_cache(maxsize=1) - def cross_border(self) -> pd.DataFrame: - """ Create the Cross-border Model trip list. - - Read in the Cross-border trip list, map field values, and genericize - field names. - - Returns: - A Pandas DataFrame of the Cross-border trip list """ - - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "crossBorderTrips.csv"), - usecols=["tourID", - "tripID", - "inbound", - "period", - "originPurp", - "destPurp", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tripMode", - "boardingTap", - "alightingTap", - "set", - "valueOfTime", - "parkingCost"], - dtype={"tourID": "int32", - "tripID": "int8", - "inbound": "boolean", - "period": "int8", - "originPurp": "int8", - "destPurp": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tripMode": "int8", - "boardingTap": "int16", - "alightingTap": "int16", - "set": "int8", - "valueOfTime": "float32", - "parkingCost": "float32"}) - - # use the tripID column from the data-set as stopID within the tour - # and create actual tripID field - trips.rename(columns={"tripID": "stopID"}, inplace=True) - trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # apply exhaustive field mappings where applicable - mappings = { - "originPurp": {-1: "Unknown", - 0: "Work", - 1: "School", - 2: "Cargo", - 3: "Shop", - 4: "Visit", - 5: "Other"}, - "destPurp": {-1: "Unknown", - 0: "Work", - 1: "School", - 2: "Cargo", - 3: "Shop", - 4: "Visit", - 5: "Other"}, - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "boardingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "alightingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {-1: "", - 0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"} - } - - for field in mappings: - if field in ["boardingTap", "alightingTap"]: - trips[field] = trips[field].map(mappings[field]).astype("float32") - else: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.period) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set(mode=trips.tripMode, - transit_set=trips.set) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories( - vot=trips.valueOfTime) - - # transform parking cost from cents to dollars - trips["parkingCost"] = round(trips.parkingCost / 100, 2) - - # no cross-border trips use transponders or AVs - # no cross-border trips are allowed to park in another MGRA - trips["transponderAvailable"] = False - trips["avUsed"] = False - trips["parkingMGRA"] = pd.Series(np.NaN, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - conditions = [(trips["tripMode"] == "Shared Ride 2"), - (trips["tripMode"] == "Shared Ride 3+"), - (trips["tripMode"] == "Taxi"), - (trips["tripMode"] == "Non-Pooled TNC"), - (trips["tripMode"] == "Pooled TNC")] - choices = [1 / self.properties["sr2Passengers"], - 1 / self.properties["sr3Passengers"], - 1 / self.properties["taxiPassengers"], - 1 / self.properties["nonPooledTNCPassengers"], - 1 / self.properties["pooledTNCPassengers"]] - - trips["weightTrip"] = pd.Series(np.select(conditions, choices, default=1) / self.properties["sampleRate"], dtype="float32") - trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] - trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"id": "tripID", - "period": "departTimeAbmHalfHour", - "originPurp": "tripPurposeOrigin", - "destPurp": "tripPurposeDestination", - "parkingCost": "costParking", - "boardingTap": "boardingTAP", - "alightingTap": "alightingTAP"}, - inplace=True) - - return trips[["tripID", - "tourID", - "stopID", - "inbound", - "tripPurposeOrigin", - "tripPurposeDestination", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "costParking"]] - - @property - @lru_cache(maxsize=1) - def cvm(self) -> pd.DataFrame: - """ Create the Commercial Vehicle Model trip list. - - Read in the Commercial Vehicle trip list, apply trip scaling and share - allocation, map field values, and genericize field names. - - Returns: - A Pandas DataFrame of the Commercial Vehicle trip list """ - - # create list of all Commercial Vehicle model trip list files - # files are of the form Trip_<>_<> - files = ["Trip" + "_" + i + "_" + j + ".csv" for i, j in - itertools.product(["FA", "GO", "IN", "RE", "SV", "TH", "WH"], - ["OE", "AM", "MD", "PM", "OL"])] - - # read all trip list files into a Pandas DataFrame - trips = pd.concat(( - pd.read_csv(os.path.join(self.scenario_path, "output", file), - usecols=["SerialNo", - "Trip", - "HomeZone", - "ActorType", - "OPurp", - "DPurp", - "I", - "J", - "Mode", - "StartTime", - "EndTime", - "StopDuration", - "TourType", - "OriginalTimePeriod"], - dtype={"SerialNo": "int32", - "Trip": "int8", - "ActorType": "string", - "HomeZone": "int16", - "OPurp": "string", - "DPurp": "string", - "I": "int16", - "J": "int16", - "Mode": "string", - "StartTime": "float32", - "EndTime": "float32", - "StopDuration": "float32", - "TourType": "string", - "OriginalTimePeriod": "string"} - ) - for file in files)) - - # apply weighting and share re-allocation originally implemented in - # Java by Nagendra Dhakar + Joel Freedman at RSG - - # create lookup table of mode-tod-scale-share using scenario properties - lookup = pd.DataFrame( - {"Mode": ["L"] * 5 + ["I"] * 5 + ["M"] * 5 + ["H"] * 5, - "OriginalTimePeriod": ["OE", "AM", "MD", "PM", "OL"] * 4, - "cvmScale": self.properties["cvmScaleLight"] + - self.properties["cvmScaleMedium"] + - self.properties["cvmScaleMedium"] + - self.properties["cvmScaleHeavy"], - "cvmShare": [self.properties["cvmShareLight"]] * 5 + - [0] * 5 + - [self.properties["cvmShareMedium"]] * 5 + - [self.properties["cvmShareHeavy"]] * 5}) - - # merge trip list and lookup table - trips = trips.merge(lookup) - - # within each mode, the properties file designates a percentage of the - # trip weight to be removed from the original trip and given to a new - # identical trip with the "I" (light-heavy duty truck) mode - new_trips = trips.loc[trips["cvmShare"] > 0].copy() - new_trips.reset_index(drop=True, inplace=True) - new_trips["Mode"] = "I" - - # within each mode and tour start abm five time of day period, the - # properties file designates a scaling factor to apply to the trip - # weight taking into account the share factor - new_trips["weightTrip"] = new_trips["cvmScale"] * new_trips["cvmShare"] - trips["weightTrip"] = trips["cvmScale"] * (1 - trips["cvmShare"]) - trips = pd.concat([trips, new_trips], ignore_index=True) - - # apply exhaustive field mappings where applicable - mappings = { - "OPurp": {"Est": "Return to Establishment", - "Gds": "Goods", - "Srv": "Service", - "Oth": "Other"}, - "DPurp": {"Est": "Return to Establishment", - "Gds": "Goods", - "Srv": "Service", - "Oth": "Other"}, - "Mode": {"L": "Drive Alone", - "I": "Light Heavy Duty Truck", - "M": "Medium Heavy Duty Truck", - "H": "Heavy Heavy Duty Truck"}, - } - - for field in mappings: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # create tour and trip surrogate keys - # unique tour is defined by (SerialNo, Mode) - # unique trip is defined by (SerialNo, Mode, Trip) - trips["tourID"] = trips.groupby(["SerialNo", "Mode"]).ngroup().astype("int32") + 1 - trips = trips.sort_values(by=["SerialNo", "Mode", "Trip"]).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # map continuous start and end times to ABM half hour time periods - # times are in continuous hours of the day (0-24) and can wrap into - # the following day or even multiple following days (>24) with no - # upper limit - - # create times from continuous hour start and end times - # taking into account their wrapping into subsequent days - trips["StartTime"] = trips["StartTime"].apply( - lambda x: (datetime.combine(date.today(), time.min) + - timedelta(hours=(x % 24))).time()) - trips["EndTime"] = trips["EndTime"].apply( - lambda x: (datetime.combine(date.today(), time.min) + - timedelta(hours=(x % 24))).time()) - - # map continuous times to abm half hour periods - depart_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in trips["StartTime"]] - depart_half_hour = [val for sublist in depart_half_hour for val in sublist] - trips = trips.assign(departTimeAbmHalfHour=depart_half_hour) - trips["departTimeAbmHalfHour"] = trips["departTimeAbmHalfHour"].astype("int8") - - arrive_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in trips["EndTime"]] - arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] - trips = trips.assign(arriveTimeAbmHalfHour=arrive_half_hour) - trips["arriveTimeAbmHalfHour"] = trips["arriveTimeAbmHalfHour"].astype("int8") - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.departTimeAbmHalfHour) - - trips["arriveTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.arriveTimeAbmHalfHour) - - # all cvm trips are high value of time - # only Drive Alone cvm trips use transponders - # no cvm trips use AVs - # no cvm trips are allowed to park in another MGRA - trips["valueOfTimeCategory"] = "High" - trips["valueOfTimeCategory"] = trips["valueOfTimeCategory"].astype("category") - trips["transponderAvailable"] = np.where(trips["Mode"] == "Drive Alone", True, False) - trips["avUsed"] = False - trips["parkingTAZ"] = pd.Series(np.NaN, dtype="float32") - - # add person-based weight and adjust weights - # by the ABM scenario final iteration sample rate - trips["weightTrip"] = pd.Series(trips["weightTrip"] / self.properties["sampleRate"], dtype="float32") - trips["weightPersonTrip"] = trips["weightTrip"] - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"Trip": "stopID", - "OPurp": "tripPurposeOrigin", - "DPurp": "tripPurposeDestination", - "I": "originTAZ", - "J": "destinationTAZ", - "Mode": "tripMode", - "StopDuration": "stopDuration"}, - inplace=True) - - return trips[["tripID", - "tourID", - "stopID", - "tripPurposeOrigin", - "tripPurposeDestination", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "stopDuration", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"]] - - @property - @lru_cache(maxsize=1) - def ee(self) -> pd.DataFrame: - """ Create the External-External Model trip list. - - Read in the External-External trip last, map field values, and - genericize field names. - - Returns: - A Pandas DataFrame of the External-External trips list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "report", "eetrip.csv"), - usecols=["OTAZ", - "DTAZ", - "TOD", - "MODE", - "TRIPS", - "TIME", - "DIST", - "AOC", - "TOLLCOST"], - dtype={"OTAZ": "int16", - "DTAZ": "int16", - "TOD": "string", - "MODE": "string", - "TRIPS": "float32", - "TIME": "float32", - "DIST": "float32", - "AOC": "float32", - "TOLLCOST": "float32"}) - - # expand trip list by 3x - # divide [TRIPS] field by 3 - # assign each copy of trip list to each value of time category - trips_low = trips.copy() - trips_low["valueOfTimeCategory"] = "Low" - - trips_med = trips.copy() - trips_med["valueOfTimeCategory"] = "Medium" - - trips_high = trips.copy() - trips_high["valueOfTimeCategory"] = "High" - - trips = pd.concat([trips_low, trips_med, trips_high], ignore_index=True) - - trips["TRIPS"] = trips["TRIPS"] / 3 - - # create trip surrogate key - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # apply exhaustive field mappings where applicable - mappings = { - "TOD": {"EA": 1, - "AM": 2, - "MD": 3, - "PM": 4, - "EV": 5}, - "MODE": {"DA": "Drive Alone", - "S2": "Shared Ride 2", - "S3": "Shared Ride 3+"} - } - - for field in mappings: - if field == "TOD": - trips[field] = trips[field].map(mappings[field]).astype("int8") - else: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # convert cents-based cost fields to dollars - trips["AOC"] = trips["AOC"] / 100 - trips["TOLLCOST"] = trips["TOLLCOST"] / 100 - - # no trips use transponders or autonomous vehicles - trips["transponderAvailable"] = False - trips["avUsed"] = False - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - conditions = [(trips["MODE"] == "Shared Ride 2"), - (trips["MODE"] == "Shared Ride 3+")] - choices = [self.properties["sr2Passengers"], - self.properties["sr3Passengers"]] - - trips["weightPersonTrip"] = pd.Series( - trips["TRIPS"] * np.select(conditions, choices, default=1) / self.properties["sampleRate"], - dtype="float32") - trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"OTAZ": "originTAZ", - "DTAZ": "destinationTAZ", - "TOD": "departTimeFiveTod", - "MODE": "tripMode", - "TIME": "timeDrive", - "DIST": "distanceDrive", - "AOC": "costOperatingDrive", - "TOLLCOST": "costTollDrive"}, - inplace=True) - - # create total time/distance/cost columns - trips["timeTotal"] = trips["timeDrive"] - trips["distanceTotal"] = trips["distanceDrive"] - trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] - - return trips[["tripID", - "departTimeFiveTod", - "originTAZ", - "destinationTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "timeDrive", - "distanceDrive", - "costTollDrive", - "costOperatingDrive", - "timeTotal", - "distanceTotal", - "costTotal"]] - - @property - @lru_cache(maxsize=1) - def ei(self) -> pd.DataFrame: - """ Create the External-Internal Model trip list. - - Read in the External-Internal trip last, map field values, and - genericize field names. - - Returns: - A Pandas DataFrame of the External-Internal trips list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "report", "eitrip.csv"), - usecols=["OTAZ", - "DTAZ", - "TOD", - "MODE", - "PURPOSE", - "TRIPS", - "TIME", - "DIST", - "AOC", - "TOLLCOST"], - dtype={"OTAZ": "int16", - "DTAZ": "int16", - "TOD": "string", - "MODE": "string", - "PURPOSE": "string", - "TRIPS": "float32", - "TIME": "float32", - "DIST": "float32", - "AOC": "float32", - "TOLLCOST": "float32"}) - - # expand trip list by 3x - # divide [TRIPS] field by 3 - # assign each copy of trip list to each value of time category - trips_low = trips.copy() - trips_low["valueOfTimeCategory"] = "Low" - - trips_med = trips.copy() - trips_med["valueOfTimeCategory"] = "Medium" - - trips_high = trips.copy() - trips_high["valueOfTimeCategory"] = "High" - - trips = pd.concat([trips_low, trips_med, trips_high], ignore_index=True) - - trips["TRIPS"] = trips["TRIPS"] / 3 - - # create trip surrogate key - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # apply exhaustive field mappings where applicable - mappings = { - "TOD": {"EA": 1, - "AM": 2, - "MD": 3, - "PM": 4, - "EV": 5}, - "MODE": {"DAN": "Drive Alone", - "DAT": "Drive Alone", - "S2N": "Shared Ride 2", - "S2T": "Shared Ride 2", - "S3N": "Shared Ride 3+", - "S3T": "Shared Ride 3+"}, - "PURPOSE": {"NONWORK": "Non-Work", - "WORK": "Work"} - } - - for field in mappings: - if field == "TOD": - trips[field] = trips[field].map(mappings[field]).astype("int8") - else: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # convert cents-based cost fields to dollars - trips["AOC"] = trips["AOC"] / 100 - trips["TOLLCOST"] = trips["TOLLCOST"] / 100 - - # no trips use transponders or autonomous vehicles - trips["transponderAvailable"] = False - trips["avUsed"] = False - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - conditions = [(trips["MODE"] == "Shared Ride 2"), - (trips["MODE"] == "Shared Ride 3+")] - choices = [self.properties["sr2Passengers"], - self.properties["sr3Passengers"]] - - trips["weightPersonTrip"] = pd.Series( - trips["TRIPS"] * np.select(conditions, choices, default=1) / self.properties["sampleRate"], - dtype="float32") - trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"OTAZ": "originTAZ", - "DTAZ": "destinationTAZ", - "TOD": "departTimeFiveTod", - "MODE": "tripMode", - "PURPOSE": "tripPurpose", - "TIME": "timeDrive", - "DIST": "distanceDrive", - "AOC": "costOperatingDrive", - "TOLLCOST": "costTollDrive"}, - inplace=True) - - # create total time/distance/cost columns - trips["timeTotal"] = trips["timeDrive"] - trips["distanceTotal"] = trips["distanceDrive"] - trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] - - return trips[["tripID", - "departTimeFiveTod", - "originTAZ", - "destinationTAZ", - "tripMode", - "tripPurpose", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "timeDrive", - "distanceDrive", - "costTollDrive", - "costOperatingDrive", - "timeTotal", - "distanceTotal", - "costTotal"]] - - @property - @lru_cache(maxsize=1) - def ie(self) -> pd.DataFrame: - """ Create the Internal-External Model trip list. - - Read in the Internal-External trip list, map field values, - and genericize field names. - - Returns: - A Pandas DataFrame of the Internal-External trip list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "internalExternalTrips.csv"), - usecols=["hhID", - "personID", - "tourID", - "inbound", - "period", - "originMGRA", - "destinationMGRA", - "originTAZ", - "destinationTAZ", - "tripMode", - "av_avail", - "boardingTap", - "alightingTap", - "set", - "valueOfTime"], - dtype={"hhID": "int32", - "personID": "int32", - "tourID": "int32", - "inbound": "boolean", - "period": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "originTAZ": "int16", - "destinationTAZ": "int16", - "tripMode": "int8", - "av_avail": "bool", - "boardingTap": "int16", - "alightingTap": "int16", - "set": "int8", - "valueOfTime": "float32"}) - - # load output household transponder ownership data - hh_fn = "householdData_" + str(self.properties["iterations"]) + ".csv" - hh = pd.read_csv( - os.path.join(self.scenario_path, "output", hh_fn), - usecols=["hh_id", - "transponder"], - dtype={"hh_id": "int32", - "transponder": "bool"}) - - # if household has a transponder then all trips can use it - trips = trips.merge(hh, left_on="hhID", right_on="hh_id") - - # apply exhaustive field mappings where applicable - mappings = { - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "boardingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "alightingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {-1: "", - 0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"} - } - - for field in mappings: - if field in ["boardingTap", "alightingTap"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # create trip surrogate key - # create stop surrogate key - # every tourID contains only two trips (outbound and inbound) - trips["stopID"] = trips.sort_values(by=["tourID", "inbound"]).groupby(["tourID"]).cumcount().astype("int8") + 1 - trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.period - ) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set( - mode=trips.tripMode, - transit_set=trips.set - ) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories( - vot=trips.valueOfTime - ) - - # no internal-external trips are allowed to park in another MGRA - trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - conditions = [(trips["tripMode"] == "Shared Ride 2"), - (trips["tripMode"] == "Shared Ride 3+"), - (trips["tripMode"] == "Taxi"), - (trips["tripMode"] == "Non-Pooled TNC"), - (trips["tripMode"] == "Pooled TNC")] - choices = [1 / self.properties["sr2Passengers"], - 1 / self.properties["sr3Passengers"], - 1 / self.properties["taxiPassengers"], - 1 / self.properties["nonPooledTNCPassengers"], - 1 / self.properties["pooledTNCPassengers"]] - - trips["weightTrip"] = pd.Series( - np.select(conditions, choices, default=1) / self.properties["sampleRate"], - dtype="float32") - trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] - trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"period": "departTimeAbmHalfHour", - "av_avail": "avUsed", - "boardingTap": "boardingTAP", - "alightingTap": "alightingTAP", - "transponder": "transponderAvailable"}, - inplace=True) - - return trips[["tripID", - "personID", - "tourID", - "stopID", - "inbound", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"]] - - @property - @lru_cache(maxsize=1) - def individual(self) -> pd.DataFrame: - """ Create the Individual Model trip list. - - Read in the Individual trip list, map field values, and genericize - field names. - - Returns: - A Pandas DataFrame of the Individual trip list """ - # load trip list into Pandas DataFrame - fn = "indivTripData_" + str(self.properties["iterations"]) + ".csv" - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", fn), - usecols=["person_id", - "tour_id", - "stop_id", - "inbound", - "tour_purpose", - "orig_purpose", - "dest_purpose", - "orig_mgra", - "dest_mgra", - "parking_mgra", - "stop_period", - "trip_mode", - "av_avail", - "trip_board_tap", - "trip_alight_tap", - "set", - "valueOfTime", - "transponder_avail", - "micro_walkMode", - "micro_trnAcc", - "micro_trnEgr", - "parkingCost"], - dtype={"person_id": "int32", - "tour_id": "int8", - "stop_id": "int8", - "inbound": "bool", - "tour_purpose": "string", - "orig_purpose": "string", - "dest_purpose": "string", - "orig_mgra": "int16", - "dest_mgra": "int16", - "parking_mgra": "int16", - "stop_period": "int8", - "trip_mode": "int8", - "av_avail": "bool", - "trip_board_tap": "int16", - "trip_alight_tap": "int16", - "set": "int8", - "valueOfTime": "float32", - "transponder_avail": "bool", - "micro_walkMode": "int8", - "micro_trnAcc": "int8", - "micro_trnEgr": "int8", - "parkingCost": "float32"}) - - # apply exhaustive field mappings where applicable - mappings = { - "parking_mgra": {key: value for (key, value) in - zip(list(range(1, 23003)), - list(range(1, 23003)))}, - "trip_mode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC", - 13: "School Bus"}, - "trip_board_tap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "trip_alight_tap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"}, - "micro_walkMode": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"}, - "micro_trnAcc": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"}, - "micro_trnEgr": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"} - } - - for field in mappings: - if field in ["parking_mgra", "trip_board_tap", "trip_alight_tap"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # create tour surrogate key (person_id, tour_id, tour_purpose) - tour_key = ["person_id", "tour_id", "tour_purpose"] - trips["tourID"] = pd.Series(trips.groupby(tour_key).ngroup() + 1, dtype="int32") - - # create tour stop surrogate key (inbound, stop_id) - stop_key = ["inbound", "stop_id"] - trips["stopID"] = pd.Series(trips.sort_values(by=stop_key).groupby(tour_key).cumcount() + 1, dtype="int8") - - # create unique trip surrogate key - trips = trips.sort_values(by=tour_key + stop_key).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - trips = trips.merge(taz_info, left_on="dest_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - trips = trips.merge(taz_info, how="left", left_on="parking_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "parkingTAZ"}, inplace=True) - trips["parkingTAZ"] = trips["parkingTAZ"].astype("float32") - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.stop_period - ) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set( - mode=trips.trip_mode, - transit_set=trips.set - ) - - # set appropriate walk mode for walk trips - trips["tripMode"] = self._combine_mode_walk( - mode=trips.tripMode, - walk_mode=trips.micro_walkMode - ) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories( - vot=trips.valueOfTime - ) - - # transform parking cost from cents to dollars - trips["parkingCost"] = round(trips.parkingCost / 100, 2) - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - conditions = [(trips["tripMode"] == "Shared Ride 2"), - (trips["tripMode"] == "Shared Ride 3+"), - (trips["tripMode"] == "Taxi"), - (trips["tripMode"] == "Non-Pooled TNC"), - (trips["tripMode"] == "Pooled TNC")] - choices = [1 / self.properties["sr2Passengers"], - 1 / self.properties["sr3Passengers"], - 1 / self.properties["taxiPassengers"], - 1 / self.properties["nonPooledTNCPassengers"], - 1 / self.properties["pooledTNCPassengers"]] - - trips["weightTrip"] = pd.Series(np.select(conditions, choices, default=1) / self.properties["sampleRate"], dtype="float32") - trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] - trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"person_id": "personID", - "orig_purpose": "tripPurposeOrigin", - "dest_purpose": "tripPurposeDestination", - "stop_period": "departTimeAbmHalfHour", - "orig_mgra": "originMGRA", - "dest_mgra": "destinationMGRA", - "parking_mgra": "parkingMGRA", - "parkingCost": "costParking", - "av_avail": "avUsed", - "trip_board_tap": "boardingTAP", - "trip_alight_tap": "alightingTAP", - "transponder_avail": "transponderAvailable", - "micro_trnAcc": "microMobilityTransitAccess", - "micro_trnEgr": "microMobilityTransitEgress"}, - inplace=True) - - return trips[["tripID", - "personID", - "tourID", - "stopID", - "inbound", - "tripPurposeOrigin", - "tripPurposeDestination", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "microMobilityTransitAccess", - "microMobilityTransitEgress", - "weightTrip", - "weightPersonTrip", - "costParking"]] - - @property - @lru_cache(maxsize=1) - def joint(self) -> pd.DataFrame: - """ Create the Joint Model trip list. - - Read in the Joint trip list, map field values, genericize field - names, append skim values, replicate data-set records for each trip - participant creating data-set format of one record per participant, - and assign trip weights accounting for replicated records. - - Returns: - A Pandas DataFrame of the Joint trip list """ - # load trip list into Pandas DataFrame - fn_trips = "jointTripData_" + str(self.properties["iterations"]) + ".csv" - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", fn_trips), - usecols=["hh_id", - "tour_id", - "stop_id", - "inbound", - "orig_purpose", - "dest_purpose", - "orig_mgra", - "dest_mgra", - "parking_mgra", - "stop_period", - "trip_mode", - "av_avail", - "num_participants", - "trip_board_tap", - "trip_alight_tap", - "set", - "valueOfTime", - "transponder_avail", - "parkingCost"], - dtype={"hh_id": "int32", - "tour_id": "int8", - "stop_id": "int8", - "inbound": "bool", - "orig_purpose": "string", - "dest_purpose": "string", - "orig_mgra": "int16", - "dest_mgra": "int16", - "parking_mgra": "int16", - "stop_period": "int8", - "trip_mode": "int8", - "av_avail": "bool", - "num_participants": "int8", - "trip_board_tap": "int16", - "trip_alight_tap": "int16", - "set": "int8", - "valueOfTime": "float32", - "transponder_avail": "bool", - "parkingCost": "float32"}) - - # apply exhaustive field mappings where applicable - mappings = { - "parking_mgra": {key: value for (key, value) in - zip(list(range(1, 23003)), - list(range(1, 23003)))}, - "trip_mode": {2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "trip_board_tap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "trip_alight_tap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"}, - } - - for field in mappings: - if field in ["parking_mgra", "trip_board_tap", "trip_alight_tap"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # create tour surrogate key (hh_id, tour_id) - tour_key = ["hh_id", "tour_id"] - trips["tourID"] = pd.Series( - trips.groupby(tour_key).ngroup() + 1, - dtype="int32") - - # create tour stop surrogate key (inbound, stop_id) - stop_key = ["inbound", "stop_id"] - trips["stopID"] = pd.Series( - trips.sort_values(by=stop_key).groupby(tour_key).cumcount() + 1, - dtype="int8") - - # create unique trip surrogate key - trips = trips.sort_values(by=tour_key + stop_key).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - trips = trips.merge(taz_info, left_on="dest_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - trips = trips.merge(taz_info, how="left", left_on="parking_mgra", - right_on="MGRA") - trips.rename(columns={"TAZ": "parkingTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.stop_period - ) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set( - mode=trips.trip_mode, - transit_set=trips.set - ) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories( - vot=trips.valueOfTime - ) - - # transform parking cost from cents to dollars - trips["parkingCost"] = round(trips.parkingCost / 100, 2) - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"orig_purpose": "tripPurposeOrigin", - "dest_purpose": "tripPurposeDestination", - "stop_period": "departTimeAbmHalfHour", - "orig_mgra": "originMGRA", - "dest_mgra": "destinationMGRA", - "parking_mgra": "parkingMGRA", - "parkingCost": "costParking", - "av_avail": "avUsed", - "trip_board_tap": "boardingTAP", - "trip_alight_tap": "alightingTAP", - "transponder_avail": "transponderAvailable"}, - inplace=True) - - # load tour list into Pandas DataFrame - fn_tours = "jointTourData_" + str( - self.properties["iterations"]) + ".csv" - tours = pd.read_csv( - os.path.join(self.scenario_path, "output", fn_tours), - usecols=["hh_id", - "tour_id", - "tour_participants"], - dtype={"hh_id": "int32", - "tour_id": "int8", - "tour_participants": "string"}) - - # split the tour participants column by " " and append in wide-format - # to each record - tours = pd.concat( - [tours[["hh_id", "tour_id"]], - tours["tour_participants"].str.split(" ", expand=True)], - axis=1 - ) - - # melt the wide-format tour participants to long-format - tours = pd.melt(tours, id_vars=["hh_id", "tour_id"], - value_name="person_num") - tours = tours[tours["person_num"].notnull()] - tours["person_num"] = tours["person_num"].astype("int8") - - # load output person data into Pandas DataFrame - fn_persons = "personData_" + str(self.properties["iterations"]) + ".csv" - persons = pd.read_csv( - os.path.join(self.scenario_path, "output", fn_persons), - usecols=["hh_id", - "person_num", - "person_id"], - dtype={"hh_id": "int32", - "person_num": "int8", - "person_id": "int32"}) - persons.rename(columns={"person_id": "personID"}, inplace=True) - - # merge persons with the long-format tour participants to get the person id - tours = tours.merge(persons, on=["hh_id", "person_num"]) - - # merge long-format tour participants with the trip list - # this many-to-one merge replicates trip records for each participant - # as well as appending the person id to each replicated record - trips = trips.merge(tours, on=["hh_id", "tour_id"]) - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - # each record is per-person on trip - # data-set has single person record with multiple trip records - trips["weightTrip"] = pd.Series( - 1 / (trips["num_participants"] * self.properties["sampleRate"]), - dtype="float32") - trips["weightPersonTrip"] = 1 / self.properties["sampleRate"] - trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("float32") - - return trips[["tripID", - "personID", - "tourID", - "stopID", - "inbound", - "tripPurposeOrigin", - "tripPurposeDestination", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "costParking"]] - - @property - @lru_cache(maxsize=1) - def truck(self) -> pd.DataFrame: - """ Create the Truck Model trip list. - - Read in the External-External trip last, map field values, and - genericize field names. - - Returns: - A Pandas DataFrame of the External-External trips list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "report", "trucktrip.csv"), - usecols=["OTAZ", - "DTAZ", - "TOD", - "MODE", - "TRIPS", - "TIME", - "DIST", - "AOC", - "TOLLCOST"], - dtype={"OTAZ": "int16", - "DTAZ": "int16", - "TOD": "string", - "MODE": "string", - "TRIPS": "float32", - "TIME": "float32", - "DIST": "float32", - "AOC": "float32", - "TOLLCOST": "float32"}) - - # create trip surrogate key - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # apply exhaustive field mappings where applicable - mappings = { - "TOD": {"EA": 1, - "AM": 2, - "MD": 3, - "PM": 4, - "EV": 5}, - "MODE": {"lhdn": "Light Heavy Duty Truck", - "lhdt": "Light Heavy Duty Truck", - "mhdn": "Medium Heavy Duty Truck", - "mhdt": "Medium Heavy Duty Truck", - "hhdn": "Heavy Heavy Duty Truck", - "hhdt": "Heavy Heavy Duty Truck"} - } - - for field in mappings: - if field == "TOD": - trips[field] = trips[field].map(mappings[field]).astype("int8") - else: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # convert cents-based cost fields to dollars - trips["AOC"] = trips["AOC"] / 100 - trips["TOLLCOST"] = trips["TOLLCOST"] / 100 - - # all trips are High value of time - # no trips use transponders or autonomous vehicles - trips["valueOfTimeCategory"] = "High" - trips["transponderAvailable"] = False - trips["avUsed"] = False - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - trips["weightTrip"] = trips["TRIPS"] / self.properties["sampleRate"] - trips["weightPersonTrip"] = trips["TRIPS"] / self.properties["sampleRate"] - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"OTAZ": "originTAZ", - "DTAZ": "destinationTAZ", - "TOD": "departTimeFiveTod", - "MODE": "tripMode", - "TIME": "timeDrive", - "DIST": "distanceDrive", - "AOC": "costOperatingDrive", - "TOLLCOST": "costTollDrive"}, - inplace=True) - - # create total time/distance/cost columns - trips["timeTotal"] = trips["timeDrive"] - trips["distanceTotal"] = trips["distanceDrive"] - trips["costTotal"] = trips["costTollDrive"] + trips["costOperatingDrive"] - - return trips[["tripID", - "departTimeFiveTod", - "originTAZ", - "destinationTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "timeDrive", - "distanceDrive", - "costTollDrive", - "costOperatingDrive", - "timeTotal", - "distanceTotal", - "costTotal"]] - - @property - @lru_cache(maxsize=1) - def visitor(self) -> pd.DataFrame: - """ Create the Visitor Model trip list. - - Read in the Visitor trip list, map field values, and genericize - field names. - - Returns: - A Pandas DataFrame of the Visitor trip list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "visitorTrips.csv"), - usecols=["tourID", - "tripID", - "originPurp", - "destPurp", - "originMGRA", - "destinationMGRA", - "inbound", - "period", - "tripMode", - "avAvailable", - "boardingTap", - "alightingTap", - "set", - "valueOfTime", - "partySize", - "micro_walkMode", - "micro_trnAcc", - "micro_trnEgr", - "parkingCost"], - dtype={"tourID": "int32", - "tripID": "int8", - "originPurp": "int8", - "destPurp": "int8", - "originMGRA": "int16", - "destinationMGRA": "int16", - "inbound": "boolean", - "period": "int8", - "tripMode": "int8", - "avAvailable": "bool", - "boardingTap": "int16", - "alightingTap": "int16", - "set": "int8", - "valueOfTime": "float32", - "partySize": "int8", - "micro_walkMode": "int8", - "micro_trnAcc": "int8", - "micro_trnEgr": "int8", - "parkingCost": "float32"}) - - # apply exhaustive field mappings where applicable - mappings = { - "originPurp": {-1: "Unknown", - 0: "Work", - 1: "Recreation", - 2: "Dining"}, - "destPurp": {-1: "Unknown", - 0: "Work", - 1: "Recreation", - 2: "Dining"}, - "tripMode": {1: "Drive Alone", - 2: "Shared Ride 2", - 3: "Shared Ride 3+", - 4: "Walk", - 5: "Bike", - 6: "Walk to Transit", - 7: "Park and Ride to Transit", - 8: "Kiss and Ride to Transit", - 9: "TNC to Transit", - 10: "Taxi", - 11: "Non-Pooled TNC", - 12: "Pooled TNC"}, - "boardingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "alightingTap": {key: value for (key, value) in - zip(list(range(1, 99999)), - list(range(1, 99999)))}, - "set": {0: "Local Bus", - 1: "Premium Transit", - 2: "Local Bus and Premium Transit"}, - "micro_walkMode": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"}, - "micro_trnAcc": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"}, - "micro_trnEgr": {1: "Walk", - 2: "Micro-Mobility", - 3: "Micro-Transit"} - } - - for field in mappings: - if field in ["boardingTap", "alightingTap"]: - data_type = "float32" - else: - data_type = "category" - trips[field] = trips[field].map(mappings[field]).astype(data_type) - - # create unique trip surrogate key - # the tripID field included in the data-set is a stopID - trips.rename(columns={"tripID": "stopID"}, inplace=True) - trips = trips.sort_values(by=["tourID", "stopID"]).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - trips = trips.merge(taz_info, left_on="originMGRA", right_on="MGRA") - trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - trips = trips.merge(taz_info, left_on="destinationMGRA", right_on="MGRA") - trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.period - ) - - # concatenate mode and transit skim set for transit trips - trips["tripMode"] = self._combine_mode_set( - mode=trips.tripMode, - transit_set=trips.set - ) - - # set appropriate walk mode for walk trips - trips["tripMode"] = self._combine_mode_walk( - mode=trips.tripMode, - walk_mode=trips.micro_walkMode - ) - - # calculate value of time category auto skim set used - trips["valueOfTimeCategory"] = self._map_vot_categories( - vot=trips.valueOfTime - ) - - # transform parking cost from cents to dollars - trips["parkingCost"] = round(trips.parkingCost / 100, 2) - - # no visitor trips use transponders - # no visitor trips are allowed to park in another MGRA - trips["transponderAvailable"] = False - trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") - - # add vehicle/trip-based weight and person-based weight - # adjust by the ABM scenario final iteration sample rate - trips["weightTrip"] = 1 / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - trips["weightPersonTrip"] = pd.Series( - trips["partySize"] / self.properties["sampleRate"], - dtype="float32") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"originPurp": "tripPurposeOrigin", - "destPurp": "tripPurposeDestination", - "period": "departTimeAbmHalfHour", - "parkingCost": "costParking", - "avAvailable": "avUsed", - "boardingTap": "boardingTAP", - "alightingTap": "alightingTAP", - "micro_trnAcc": "microMobilityTransitAccess", - "micro_trnEgr": "microMobilityTransitEgress"}, - inplace=True) - - return trips[["tripID", - "tourID", - "stopID", - "inbound", - "tripPurposeOrigin", - "tripPurposeDestination", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "boardingTAP", - "alightingTAP", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "microMobilityTransitAccess", - "microMobilityTransitEgress", - "weightTrip", - "weightPersonTrip", - "costParking"]] - - @property - @lru_cache(maxsize=1) - def zombie_av(self) -> pd.DataFrame: - """ Create the 0-Passenger Autonomous Vehicle trip list. - - Read in the Autonomous Vehicle trip list, select only 0-passenger - trips, map field values, and genericize field names. - - Returns: - A Pandas DataFrame of the 0-Passenger Autonomous Vehicle trip - list """ - fn = os.path.join(self.scenario_path, "output", "householdAVTrips.csv") - # file does not exist if AV-component of model is turned off - if os.path.isfile(fn): - # load trip list into Pandas DataFrame - trips = pd.read_csv( - fn, - usecols=["hh_id", - "veh_id", - "vehicleTrip_id", - "orig_mgra", - "dest_gra", - "period", - "occupants", - "originIsHome", - "destinationIsHome", - "originIsRemoteParking", - "destinationIsRemoteParking", - "remoteParkingCostAtDest"], - dtype={"hh_id": "int32", - "veh_id": "int32", - "vehicleTrip_id": "int32", - "orig_mgra": "int32", - "dest_gra": "int32", - "period": "int32", - "occupants": "int32", - "originIsHome": "bool", - "destinationIsHome": "bool", - "originIsRemoteParking": "bool", - "destinationIsRemoteParking": "bool", - "remoteParkingCostAtDest": "float32"} - ) - - # filter trip list to empty/zombie av trips - trips = trips.loc[trips["occupants"] == 0].copy() - - # create unique trip surrogate key - trip_key = ["hh_id", "veh_id", "vehicleTrip_id"] - trips = trips.sort_values(by=trip_key).reset_index(drop=True) - trips["tripID"] = pd.Series(trips.index + 1, dtype="int32") - - # load output household transponder ownership data - hh_fn = "householdData_" + str(self.properties["iterations"]) + ".csv" - hh = pd.read_csv( - os.path.join(self.scenario_path, "output", hh_fn), - usecols=["hh_id", "transponder"], - dtype={"hh_id": "int32", - "transponder": "bool"}) - - # if household has a transponder then all trips can use it - trips = trips.merge(hh, on="hh_id") - - # add TAZ information in addition to MGRA information - taz_info = self.mgra_xref[["MGRA", "TAZ"]] - - trips = trips.merge(taz_info, left_on="orig_mgra", right_on="MGRA") - trips.rename(columns={"TAZ": "originTAZ"}, inplace=True) - - trips = trips.merge(taz_info, left_on="dest_gra", right_on="MGRA") - trips.rename(columns={"TAZ": "destinationTAZ"}, inplace=True) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.period) - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods(abm_half_hour=trips.period) - - # all zombie AV trips are Drive Alone and High vot - trips["tripMode"] = "Drive Alone" - trips["valueOfTimeCategory"] = "High" - trips["avUsed"] = True - trips["parkingMGRA"] = np.nan - trips["parkingTAZ"] = np.nan - - # add person-based weight and adjust weights - # by the ABM scenario final iteration sample rate - # no people are in zombie AV trips - trips["weightTrip"] = 1 / self.properties["sampleRate"] - trips["weightPersonTrip"] = 0 - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"hh_id": "hhID", - "veh_id": "vehID", - "vehicleTrip_id": "vehicleTripID", - "orig_mgra": "originMGRA", - "dest_gra": "destinationMGRA", - "period": "departTimeAbmHalfHour", - "transponder": "transponderAvailable", - "remoteParkingCostAtDest": "costParking"}, - inplace=True) - - return trips[["tripID", - "hhID", - "vehID", - "vehicleTripID", - "originIsHome", - "destinationIsHome", - "originIsRemoteParking", - "destinationIsRemoteParking", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip", - "costParking"]] - else: # return empty DataFrame if file does not exist - return(pd.DataFrame( - columns=["tripID", - "hhID", - "vehID", - "vehicleTripID", - "originIsHome", - "destinationIsHome", - "originIsRemoteParking", - "destinationIsRemoteParking", - "parkingChoiceAtDestination", - "departTimeAbmHalfHour", - "departTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"])) - - @property - @lru_cache(maxsize=1) - def zombie_tnc(self) -> pd.DataFrame: - """ Create the 0-Passenger TNC trip list. - - Read in the TNC Vehicle trip list, select only 0-passenger - trips, map field values, and genericize field names. - - Returns: - A Pandas DataFrame of the 0-Passenger TNC Vehicle trip list """ - # load trip list into Pandas DataFrame - trips = pd.read_csv( - os.path.join(self.scenario_path, "output", "TNCTrips.csv"), - usecols=["trip_ID", - "originMgra", - "destinationMgra", - "originTaz", - "destinationTaz", - "totalPassengers", - "startPeriod", - "endPeriod", - " originPurpose", - " destinationPurpose"], - dtype={"trip_ID": "int32", - "originMgra": "int16", - "destinationMgra": "int16", - "originTaz": "int16", - "destinationTaz": "int16", - "totalPassengers": "int8", - "startPeriod": "int16", - "endPeriod": "int16", - " originPurpose": "int8", - " destinationPurpose": "int8"}) - - # filter trip list to empty/zombie tnc trips - trips = trips.loc[trips["totalPassengers"] == 0].copy() - trips.reset_index(drop=True, inplace=True) - - # apply exhaustive field mappings where applicable - mappings = { - " originPurpose": {0: "Home", - 1: "Pickup Only", - 2: "Drop-off Only", - 3: "Pickup and Drop-off", - 4: "Refuel"}, - " destinationPurpose": {0: "Home", - 1: "Pickup Only", - 2: "Drop-off Only", - 3: "Pickup and Drop-off", - 4: "Refuel"} - } - - for field in mappings: - trips[field] = trips[field].map(mappings[field]).astype("category") - - # only map TNC time periods to ABM time periods if they nest - period_width = self.properties["timePeriodWidthTNC"] - if 30 % period_width == 0: - # map TNC time periods to actual period start times - # take the defined width of the time periods multiplied - # by the time period number as the minutes after 3am allowing - # the time periods to wrap around 12am and set the time value - trips["StartTime"] = trips["startPeriod"].apply( - lambda x: (datetime.combine(date.today(), time(3, 0)) + - timedelta(minutes=(x-1) * period_width)).time()) - trips["EndTime"] = trips["endPeriod"].apply( - lambda x: (datetime.combine(date.today(), time(3, 0)) + - timedelta(minutes=(x-1) * period_width)).time()) - - # map continuous times to abm half hour periods - depart_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in trips["StartTime"]] - depart_half_hour = [val for sublist in depart_half_hour for val in sublist] - trips = trips.assign(departTimeAbmHalfHour=depart_half_hour) - trips["departTimeAbmHalfHour"] = trips["departTimeAbmHalfHour"].astype("int8") - - arrive_half_hour = [ - [p["period"] for p in self.time_periods["abmHalfHour"] - if p["startTime"] <= x < p["endTime"]] - for x in trips["EndTime"]] - arrive_half_hour = [val for sublist in arrive_half_hour for val in sublist] - trips = trips.assign(arriveTimeAbmHalfHour=arrive_half_hour) - trips["arriveTimeAbmHalfHour"] = trips["arriveTimeAbmHalfHour"].astype("int8") - - # map abm half hours to abm five time of day - trips["departTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.departTimeAbmHalfHour) - - trips["arriveTimeFiveTod"] = self._map_time_periods( - abm_half_hour=trips.arriveTimeAbmHalfHour) - else: - # if time periods are not able to nest within ABM model time periods - # set the ABM model time period fields to NaN - # RSG has been made aware of this issue - trips["departTimeAbmHalfHour"] = pd.Series(np.nan, dtype="float32") - trips["departTimeFiveTod"] = pd.Series(np.nan, dtype="float32") - trips["arriveTimeAbmHalfHour"] = pd.Series(np.nan, dtype="float32") - trips["arriveTimeFiveTod"] = pd.Series(np.nan, dtype="float32") - - # all zombie TNC trips are Drive Alone and High vot - # assumed Transponder ownership for Drive Alone TNC - trips["tripMode"] = "Drive Alone" - trips["tripMode"] = trips["tripMode"].astype("category") - trips["valueOfTimeCategory"] = "High" - trips["valueOfTimeCategory"] = trips["valueOfTimeCategory"].astype("category") - trips["transponderAvailable"] = True - trips["avUsed"] = False - trips["parkingMGRA"] = pd.Series(np.nan, dtype="float32") - trips["parkingTAZ"] = pd.Series(np.nan, dtype="float32") - - # add person-based weight and adjust weights - # by the ABM scenario final iteration sample rate - # no people are in zombie AV trips - trips["weightTrip"] = 1 / self.properties["sampleRate"] - trips["weightTrip"] = trips["weightTrip"].astype("float32") - trips["weightPersonTrip"] = 0 - trips["weightPersonTrip"] = trips["weightPersonTrip"].astype("int8") - - # rename columns to standard/generic ABM naming conventions - trips.rename(columns={"trip_ID": "tripID", - "originMgra": "originMGRA", - "destinationMgra": "destinationMGRA", - "originTaz": "originTAZ", - "destinationTaz": "destinationTAZ", - " originPurpose": "originPurpose", - " destinationPurpose": "destinationPurpose"}, - inplace=True) - - return trips[["tripID", - "originPurpose", - "destinationPurpose", - "departTimeAbmHalfHour", - "arriveTimeAbmHalfHour", - "departTimeFiveTod", - "arriveTimeFiveTod", - "originMGRA", - "destinationMGRA", - "parkingMGRA", - "originTAZ", - "destinationTAZ", - "parkingTAZ", - "tripMode", - "valueOfTimeCategory", - "transponderAvailable", - "avUsed", - "weightTrip", - "weightPersonTrip"]] diff --git a/sandag_abm/src/main/python/dataExporter/environment.yml b/sandag_abm/src/main/python/dataExporter/environment.yml deleted file mode 100644 index e00766c..0000000 --- a/sandag_abm/src/main/python/dataExporter/environment.yml +++ /dev/null @@ -1,92 +0,0 @@ -name: abmDataExporter -channels: - - defaults -dependencies: - - attrs=19.3.0=py_0 - - blas=1.0=mkl - - bzip2=1.0.8=he774522_0 - - ca-certificates=2020.6.24=0 - - certifi=2020.6.20=py38_0 - - cfitsio=3.470=he774522_5 - - click=7.1.2=py_0 - - click-plugins=1.1.1=py_0 - - cligj=0.5.0=py38_0 - - curl=7.67.0=h2a8f88b_0 - - expat=2.2.9=h33f27b4_2 - - fiona=1.8.13.post1=py38hd760492_0 - - freexl=1.0.5=hfa6e2cd_0 - - gdal=3.0.2=py38hdf43c64_0 - - geopandas=0.8.1=py_0 - - geos=3.8.0=h33f27b4_0 - - geotiff=1.5.1=h5770a2b_1 - - hdf4=4.2.13=h712560f_2 - - hdf5=1.10.4=h7ebc959_0 - - icc_rt=2019.0.0=h0cc432a_1 - - icu=58.2=ha925a31_3 - - intel-openmp=2020.0=166 - - jpeg=9b=hb83a4c4_2 - - kealib=1.4.7=h07cbb95_6 - - krb5=1.16.4=hc04afaa_0 - - libboost=1.67.0=hd9e427e_4 - - libcurl=7.67.0=h2a8f88b_0 - - libgdal=3.0.2=h1155b67_0 - - libiconv=1.15=h1df5818_7 - - libkml=1.3.0=he5f2a48_4 - - libnetcdf=4.6.1=h411e497_2 - - libpng=1.6.37=h2a8f88b_0 - - libpq=11.2=h3235a2c_0 - - libspatialindex=1.9.3=h33f27b4_0 - - libspatialite=4.3.0a=h7ffb84d_0 - - libssh2=1.9.0=h7a1dbc1_1 - - libtiff=4.1.0=h56a325e_0 - - libxml2=2.9.10=h464c3ec_1 - - lz4-c=1.8.1.2=h2fa13f4_0 - - m2w64-expat=2.1.1=2 - - m2w64-gcc-libgfortran=5.3.0=6 - - m2w64-gcc-libs=5.3.0=7 - - m2w64-gcc-libs-core=5.3.0=7 - - m2w64-gettext=0.19.7=2 - - m2w64-gmp=6.1.0=2 - - m2w64-libiconv=1.14=6 - - m2w64-libwinpthread-git=5.0.0.4634.697f757=2 - - m2w64-xz=5.2.2=2 - - mkl=2020.0=166 - - mkl-service=2.3.0=py38hb782905_0 - - mkl_fft=1.0.15=py38h14836fe_0 - - mkl_random=1.1.0=py38hf9181ef_0 - - msys2-conda-epoch=20160418=1 - - munch=2.5.0=py_0 - - numpy=1.18.1=py38h93ca92e_0 - - numpy-base=1.18.1=py38hc3f5095_1 - - openjpeg=2.3.0=h5ec785f_1 - - openssl=1.1.1g=he774522_1 - - pandas=1.0.1=py38h47e9c7a_0 - - pcre=8.44=ha925a31_0 - - pip=20.0.2=py38_1 - - postgresql=11.2=h3235a2c_0 - - proj=6.2.1=h9f7ef89_0 - - pyproj=2.6.1.post1=py38hcfa1391_1 - - python=3.8.1=h5fd99cc_1 - - python-dateutil=2.8.1=py_0 - - pytz=2019.3=py_0 - - rtree=0.9.4=py38h21ff451_1 - - setuptools=45.2.0=py38_0 - - shapely=1.7.0=py38h210f175_0 - - six=1.14.0=py38_0 - - sqlite=3.31.1=he774522_0 - - tbb=2018.0.5=he980bc4_0 - - tiledb=1.6.3=h7b000aa_0 - - tk=8.6.10=he774522_0 - - vc=14.1=h0510ff6_4 - - vs2015_runtime=14.16.27012=hf0eaf9b_1 - - wheel=0.34.2=py38_0 - - wincertstore=0.2=py38_0 - - xerces-c=3.2.2=ha925a31_0 - - xz=5.2.5=h62dcd97_0 - - zlib=1.2.11=h62dcd97_4 - - zstd=1.3.7=h508b16e_0 - - pip: - - numexpr==2.7.1 - - openmatrix==0.3.5.0 - - tables==3.6.1 - diff --git a/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py b/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py deleted file mode 100644 index 1fe6271..0000000 --- a/sandag_abm/src/main/python/dataExporter/hwyShapeExport.py +++ /dev/null @@ -1,523 +0,0 @@ -import geopandas -import numpy as np -import os -import pandas as pd -from shapely import wkt - - -def export_highway_shape(scenario_path: str) -> geopandas.GeoDataFrame: - """ Takes an input path to a completed ABM scenario model run, reads the - input and loaded highway networks from the report folder, and outputs a - geography shape file to the report folder of the loaded highway network. - - Args: - scenario_path: String location of the completed ABM scenario folder - - Returns: - A GeoPandas GeoDataFrame of the loaded highway network """ - # read in input highway network - hwy_tcad = pd.read_csv(os.path.join(scenario_path, "report", "hwyTcad.csv"), - usecols=["ID", # highway coverage id - "NM", # link name - "Length", # link length in miles - "COJUR", # count jurisdiction code - "COSTAT", # count station number - "COLOC", # count location code - "IFC", # initial functional class - "IHOV", # link operation type - "ITRUCK", # truck restriction code - "ISPD", # posted speed limit - "IWAY", # one or two way operations - "IMED", # median type - "AN", # A node number - "FXNM", # cross street name at from end of link - "BN", # B node number - "TXNM", # cross street name at to end of link - "ABLN_EA", # lanes - from-to - Early AM - "ABLN_AM", # lanes - from-to - AM Peak - "ABLN_MD", # lanes - from-to - Midday - "ABLN_PM", # lanes - from-to - PM Peak - "ABLN_EV", # lanes - from-to - Evening - "BALN_EA", # lanes - to-from - Early AM - "BALN_AM", # lanes - to-from - AM Peak - "BALN_MD", # lanes - to-from - Midday - "BALN_PM", # lanes - to-from - PM Peak - "BALN_EV", # lanes - to-from - Evening - "ABPRELOAD_EA", # preloaded bus flow - to-from - Early AM - "BAPRELOAD_EA", # preloaded bus flow - from-to - Early AM - "ABPRELOAD_AM", # preloaded bus flow - to-from - AM Peak - "BAPRELOAD_AM", # preloaded bus flow - from-to - AM Peak - "ABPRELOAD_MD", # preloaded bus flow - to-from - Midday - "BAPRELOAD_MD", # preloaded bus flow - from-to - Midday - "ABPRELOAD_PM", # preloaded bus flow - to-from - PM Peak - "BAPRELOAD_PM", # preloaded bus flow - from-to - PM Peak - "ABPRELOAD_EV", # preloaded bus flow - to-from - Evening - "BAPRELOAD_EV", # preloaded bus flow - from-to - Evening - "geometry"]) # WKT geometry - - # read in loaded highway network for each time period - for tod in ["EA", "AM", "MD", "PM", "EV"]: - fn = "hwyload_" + tod + ".csv" - - file = pd.read_csv(os.path.join(scenario_path, "report", fn), - usecols=["ID1", # highway coverage id - "AB_Time", # a-b loaded travel time - "BA_Time", # b-a loaded travel time - "AB_Speed", # a-b loaded speed - "BA_Speed", # b-a loaded speed - "AB_VOC", # a-b volume to capacity - "BA_VOC", # b-a volume to capacity - "AB_Flow_SOV_NTPL", - "BA_Flow_SOV_NTPL", - "AB_Flow_SOV_TPL", - "BA_Flow_SOV_TPL", - "AB_Flow_SR2L", - "BA_Flow_SR2L", - "AB_Flow_SR3L", - "BA_Flow_SR3L", - "AB_Flow_SOV_NTPM", - "BA_Flow_SOV_NTPM", - "AB_Flow_SOV_TPM", - "BA_Flow_SOV_TPM", - "AB_Flow_SR2M", - "BA_Flow_SR2M", - "AB_Flow_SR3M", - "BA_Flow_SR3M", - "AB_Flow_SOV_NTPH", - "BA_Flow_SOV_NTPH", - "AB_Flow_SOV_TPH", - "BA_Flow_SOV_TPH", - "AB_Flow_SR2H", - "BA_Flow_SR2H", - "AB_Flow_SR3H", - "BA_Flow_SR3H", - "AB_Flow_lhd", - "BA_Flow_lhd", - "AB_Flow_mhd", - "BA_Flow_mhd", - "AB_Flow_hhd", - "BA_Flow_hhd"]) - - # match input highway network to loaded highway network - # to get preload bus flows - file = file.merge(right=hwy_tcad, - how="inner", - left_on="ID1", - right_on="ID") - - # calculate aggregated flows - file["AB_Flow_SOV"] = file[["AB_Flow_SOV_NTPL", - "AB_Flow_SOV_TPL", - "AB_Flow_SOV_NTPM", - "AB_Flow_SOV_TPM", - "AB_Flow_SOV_NTPH", - "AB_Flow_SOV_TPH"]].sum(axis=1) - - file["BA_Flow_SOV"] = file[["BA_Flow_SOV_NTPL", - "BA_Flow_SOV_TPL", - "BA_Flow_SOV_NTPM", - "BA_Flow_SOV_TPM", - "BA_Flow_SOV_NTPH", - "BA_Flow_SOV_TPH"]].sum(axis=1) - - file["AB_Flow_SR2"] = file[["AB_Flow_SR2L", - "AB_Flow_SR2M", - "AB_Flow_SR2H"]].sum(axis=1) - - file["BA_Flow_SR2"] = file[["BA_Flow_SR2L", - "BA_Flow_SR2M", - "BA_Flow_SR2H"]].sum(axis=1) - - file["AB_Flow_SR3"] = file[["AB_Flow_SR3L", - "AB_Flow_SR3M", - "AB_Flow_SR3H"]].sum(axis=1) - - file["BA_Flow_SR3"] = file[["BA_Flow_SR3L", - "BA_Flow_SR3M", - "BA_Flow_SR3H"]].sum(axis=1) - - file["AB_Flow_Truck"] = file[["AB_Flow_lhd", - "AB_Flow_mhd", - "AB_Flow_hhd"]].sum(axis=1) - - file["BA_Flow_Truck"] = file[["BA_Flow_lhd", - "BA_Flow_mhd", - "BA_Flow_hhd"]].sum(axis=1) - - file["AB_Flow_Bus"] = file["ABPRELOAD_" + tod] - - file["BA_Flow_Bus"] = file["BAPRELOAD_" + tod] - - file["AB_Flow"] = file[["AB_Flow_SOV", - "AB_Flow_SR2", - "AB_Flow_SR3", - "AB_Flow_Truck", - "AB_Flow_Bus"]].sum(axis=1) - - file["BA_Flow"] = file[["BA_Flow_SOV", - "BA_Flow_SR2", - "BA_Flow_SR3", - "BA_Flow_Truck", - "BA_Flow_Bus"]].sum(axis=1) - - # fill NAs with 0s - na_vars = ["AB_Time", - "BA_Time", - "AB_Speed", - "BA_Speed", - "AB_VOC", - "BA_VOC"] - - file[na_vars] = file[na_vars].fillna(0) - - # select columns of interest - file = file[["ID1", - "AB_Time", - "BA_Time", - "AB_Speed", - "BA_Speed", - "AB_VOC", - "BA_VOC", - "AB_Flow_SOV", - "BA_Flow_SOV", - "AB_Flow_SR2", - "BA_Flow_SR2", - "AB_Flow_SR3", - "BA_Flow_SR3", - "AB_Flow_Truck", - "BA_Flow_Truck", - "AB_Flow_Bus", - "BA_Flow_Bus", - "AB_Flow", - "BA_Flow"]] - - # add time of day suffix to column names - file = file.add_suffix("_" + tod) - - # merge loaded highway network into input highway network - hwy_tcad = hwy_tcad.merge(right=file, - how="inner", - left_on="ID", - right_on="ID1_" + tod) - - # create string description of [IFC] field - conditions = [hwy_tcad["IFC"] == 1, - hwy_tcad["IFC"] == 2, - hwy_tcad["IFC"] == 3, - hwy_tcad["IFC"] == 4, - hwy_tcad["IFC"] == 5, - hwy_tcad["IFC"] == 6, - hwy_tcad["IFC"] == 7, - hwy_tcad["IFC"] == 8, - hwy_tcad["IFC"] == 9, - hwy_tcad["IFC"] == 10] - - choices = ["Freeway", - "Prime Arterial", - "Major Arterial", - "Collector", - "Local Collector", - "Rural Collector", - "Local (non-circulation element) Road", - "Freeway Connector Ramp", - "Local Ramp", - "Zone Connector"] - - hwy_tcad["IFC_Desc"] = np.select(conditions, choices, default="") - - # calculate aggregate flows - hwy_tcad["AB_Flow_SOV"] = hwy_tcad[["AB_Flow_SOV_EA", - "AB_Flow_SOV_AM", - "AB_Flow_SOV_MD", - "AB_Flow_SOV_PM", - "AB_Flow_SOV_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow_SOV"] = hwy_tcad[["BA_Flow_SOV_EA", - "BA_Flow_SOV_AM", - "BA_Flow_SOV_MD", - "BA_Flow_SOV_PM", - "BA_Flow_SOV_EV"]].sum(axis=1) - - hwy_tcad["AB_Flow_SR2"] = hwy_tcad[["AB_Flow_SR2_EA", - "AB_Flow_SR2_AM", - "AB_Flow_SR2_MD", - "AB_Flow_SR2_PM", - "AB_Flow_SR2_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow_SR2"] = hwy_tcad[["BA_Flow_SR2_EA", - "BA_Flow_SR2_AM", - "BA_Flow_SR2_MD", - "BA_Flow_SR2_PM", - "BA_Flow_SR2_EV"]].sum(axis=1) - - hwy_tcad["AB_Flow_SR3"] = hwy_tcad[["AB_Flow_SR3_EA", - "AB_Flow_SR3_AM", - "AB_Flow_SR3_MD", - "AB_Flow_SR3_PM", - "AB_Flow_SR3_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow_SR3"] = hwy_tcad[["BA_Flow_SR3_EA", - "BA_Flow_SR3_AM", - "BA_Flow_SR3_MD", - "BA_Flow_SR3_PM", - "BA_Flow_SR3_EV"]].sum(axis=1) - - hwy_tcad["AB_Flow_Truck"] = hwy_tcad[["AB_Flow_Truck_EA", - "AB_Flow_Truck_AM", - "AB_Flow_Truck_MD", - "AB_Flow_Truck_PM", - "AB_Flow_Truck_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow_Truck"] = hwy_tcad[["BA_Flow_Truck_EA", - "BA_Flow_Truck_AM", - "BA_Flow_Truck_MD", - "BA_Flow_Truck_PM", - "BA_Flow_Truck_EV"]].sum(axis=1) - - hwy_tcad["AB_Flow_Bus"] = hwy_tcad[["AB_Flow_Bus_EA", - "AB_Flow_Bus_AM", - "AB_Flow_Bus_MD", - "AB_Flow_Bus_PM", - "AB_Flow_Bus_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow_Bus"] = hwy_tcad[["BA_Flow_Bus_EA", - "BA_Flow_Bus_AM", - "BA_Flow_Bus_MD", - "BA_Flow_Bus_PM", - "BA_Flow_Bus_EV"]].sum(axis=1) - - hwy_tcad["AB_Flow_Auto"] = hwy_tcad[["AB_Flow_SOV", - "AB_Flow_SR2", - "AB_Flow_SR3"]].sum(axis=1) - - hwy_tcad["BA_Flow_Auto"] = hwy_tcad[["BA_Flow_SOV", - "BA_Flow_SR2", - "BA_Flow_SR3"]].sum(axis=1) - - hwy_tcad["AB_Flow"] = hwy_tcad[["AB_Flow_EA", - "AB_Flow_AM", - "AB_Flow_MD", - "AB_Flow_PM", - "AB_Flow_EV"]].sum(axis=1) - - hwy_tcad["BA_Flow"] = hwy_tcad[["BA_Flow_EA", - "BA_Flow_AM", - "BA_Flow_MD", - "BA_Flow_PM", - "BA_Flow_EV"]].sum(axis=1) - - hwy_tcad["Flow"] = hwy_tcad[["AB_Flow", - "BA_Flow"]].sum(axis=1) - - # calculate vehicle miles travelled (vmt) - hwy_tcad["AB_VMT"] = hwy_tcad["AB_Flow"] * hwy_tcad["Length"] - hwy_tcad["BA_VMT"] = hwy_tcad["BA_Flow"] * hwy_tcad["Length"] - hwy_tcad["VMT"] = hwy_tcad["Flow"] * hwy_tcad["Length"] - - # calculate vehicle hours travelled (vht) - hwy_tcad["AB_VHT"] = hwy_tcad["AB_Time_EA"] * hwy_tcad["AB_Flow_EA"] + \ - hwy_tcad["AB_Time_AM"] * hwy_tcad["AB_Flow_AM"] + \ - hwy_tcad["AB_Time_MD"] * hwy_tcad["AB_Flow_MD"] + \ - hwy_tcad["AB_Time_PM"] * hwy_tcad["AB_Flow_PM"] + \ - hwy_tcad["AB_Time_EV"] * hwy_tcad["AB_Flow_EV"] - - hwy_tcad["BA_VHT"] = hwy_tcad["BA_Time_EA"] * hwy_tcad["BA_Flow_EA"] + \ - hwy_tcad["BA_Time_AM"] * hwy_tcad["BA_Flow_AM"] + \ - hwy_tcad["BA_Time_MD"] * hwy_tcad["BA_Flow_MD"] + \ - hwy_tcad["BA_Time_PM"] * hwy_tcad["BA_Flow_PM"] + \ - hwy_tcad["BA_Time_EV"] * hwy_tcad["BA_Flow_EV"] - - hwy_tcad["VHT"] = hwy_tcad["AB_VHT"] + hwy_tcad["BA_VHT"] - - # select columns of interest - hwy_tcad = hwy_tcad[["ID", - "NM", - "Length", - "COJUR", - "COSTAT", - "COLOC", - "IFC", - "IFC_Desc", - "IHOV", - "ITRUCK", - "ISPD", - "IWAY", - "IMED", - "AN", - "FXNM", - "BN", - "TXNM", - "Flow", - "AB_Flow", - "BA_Flow", - "AB_VMT", - "BA_VMT", - "VMT", - "AB_VHT", - "BA_VHT", - "VHT", - "AB_Flow_EA", - "BA_Flow_EA", - "AB_Flow_AM", - "BA_Flow_AM", - "AB_Flow_MD", - "BA_Flow_MD", - "AB_Flow_PM", - "BA_Flow_PM", - "AB_Flow_EV", - "BA_Flow_EV", - "AB_Flow_Auto", - "BA_Flow_Auto", - "AB_Flow_SOV", - "BA_Flow_SOV", - "AB_Flow_SR2", - "BA_Flow_SR2", - "AB_Flow_SR3", - "BA_Flow_SR3", - "AB_Flow_Truck", - "BA_Flow_Truck", - "AB_Flow_Bus", - "BA_Flow_Bus", - "AB_Speed_EA", - "BA_Speed_EA", - "AB_Speed_AM", - "BA_Speed_AM", - "AB_Speed_MD", - "BA_Speed_MD", - "AB_Speed_PM", - "BA_Speed_PM", - "AB_Speed_EV", - "BA_Speed_EV", - "AB_Time_EA", - "BA_Time_EA", - "AB_Time_AM", - "BA_Time_AM", - "AB_Time_MD", - "BA_Time_MD", - "AB_Time_PM", - "BA_Time_PM", - "AB_Time_EV", - "BA_Time_EV", - "ABLN_EA", - "BALN_EA", - "ABLN_AM", - "BALN_AM", - "ABLN_MD", - "BALN_MD", - "ABLN_PM", - "BALN_PM", - "ABLN_EV", - "BALN_EV", - "AB_VOC_EA", - "BA_VOC_EA", - "AB_VOC_AM", - "BA_VOC_AM", - "AB_VOC_MD", - "BA_VOC_MD", - "AB_VOC_PM", - "BA_VOC_PM", - "AB_VOC_EV", - "BA_VOC_EV", - "geometry"]] - - # rename fields to match old process field names - hwy_tcad.rename(columns={"ID": "hwycovid", - "NM": "link_name", - "Length": "len_mile", - "COJUR": "count_jur", - "COSTAT": "count_stat", - "COLOC": "count_loc", - "IFC": "ifc", - "IFC_Desc": "ifc_desc", - "IHOV": "ihov", - "ITRUCK": "itruck", - "ISPD": "post_speed", - "IWAY": "iway", - "IMED": "imed", - "AN": "from_node", - "FXNM": "from_nm", - "BN": "to_node", - "TXNM": "to_nm", - "Flow": "total_flow", - "AB_Flow": "abTotFlow", - "BA_Flow": "baTotFlow", - "AB_VMT": "ab_vmt", - "BA_VMT": "ba_vmt", - "VMT": "vmt", - "AB_VHT": "ab_vht", - "BA_VHT": "ba_vht", - "VHT": "vht", - "AB_Flow_EA": "ab_ea_flow", - "BA_Flow_EA": "ba_ea_flow", - "AB_Flow_AM": "ab_am_flow", - "BA_Flow_AM": "ba_am_flow", - "AB_Flow_MD": "ab_md_flow", - "BA_Flow_MD": "ba_md_flow", - "AB_Flow_PM": "ab_pm_flow", - "BA_Flow_PM": "ba_pm_flow", - "AB_Flow_EV": "ab_ev_flow", - "BA_Flow_EV": "ba_ev_flow", - "AB_Flow_Auto": "abAutoFlow", - "BA_Flow_Auto": "baAutoFlow", - "AB_Flow_SOV": "abSovFlow", - "BA_Flow_SOV": "baSovFlow", - "AB_Flow_SR2": "abHov2Flow", - "BA_Flow_SR2": "baHov2Flow", - "AB_Flow_SR3": "abHov3Flow", - "BA_Flow_SR3": "baHov3Flow", - "AB_Flow_Truck": "abTrucFlow", - "BA_Flow_Truck": "baTrucFlow", - "AB_Flow_Bus": "abBusFlow", - "BA_Flow_Bus": "baBusFlow", - "AB_Speed_EA": "ab_ea_mph", - "BA_Speed_EA": "ba_ea_mph", - "AB_Speed_AM": "ab_am_mph", - "BA_Speed_AM": "ba_am_mph", - "AB_Speed_MD": "ab_md_mph", - "BA_Speed_MD": "ba_md_mph", - "AB_Speed_PM": "ab_pm_mph", - "BA_Speed_PM": "ba_pm_mph", - "AB_Speed_EV": "ab_ev_mph", - "BA_Speed_EV": "ba_ev_mph", - "AB_Time_EA": "ab_ea_min", - "BA_Time_EA": "ba_ea_min", - "AB_Time_AM": "ab_am_min", - "BA_Time_AM": "ba_am_min", - "AB_Time_MD": "ab_md_min", - "BA_Time_MD": "ba_md_min", - "AB_Time_PM": "ab_pm_min", - "BA_Time_PM": "ba_pm_min", - "AB_Time_EV": "ab_ev_min", - "BA_Time_EV": "ba_ev_min", - "ABLN_EA": "ab_ea_lane", - "BALN_EA": "ba_ea_lane", - "ABLN_AM": "ab_am_lane", - "BALN_AM": "ba_am_lane", - "ABLN_MD": "ab_md_lane", - "BALN_MD": "ba_md_lane", - "ABLN_PM": "ab_pm_lane", - "BALN_PM": "ba_pm_lane", - "ABLN_EV": "ab_ev_lane", - "BALN_EV": "ba_ev_lane", - "AB_VOC_EA": "ab_ea_voc", - "BA_VOC_EA": "ba_ea_voc", - "AB_VOC_AM": "ab_am_voc", - "BA_VOC_AM": "ba_am_voc", - "AB_VOC_MD": "ab_md_voc", - "BA_VOC_MD": "ba_md_voc", - "AB_VOC_PM": "ab_pm_voc", - "BA_VOC_PM": "ba_pm_voc", - "AB_VOC_EV": "ab_ev_voc", - "BA_VOC_EV": "ba_ev_voc"}, - inplace=True) - - # create geometry from WKT geometry field - hwy_tcad["geometry"] = hwy_tcad["geometry"].apply(wkt.loads) - - # create GeoPandas DataFrame - hwy_tcad = geopandas.GeoDataFrame( - hwy_tcad, - geometry="geometry", - crs=2230) - - return hwy_tcad diff --git a/sandag_abm/src/main/python/dataExporter/serialRun.py b/sandag_abm/src/main/python/dataExporter/serialRun.py deleted file mode 100644 index 9b6b7b0..0000000 --- a/sandag_abm/src/main/python/dataExporter/serialRun.py +++ /dev/null @@ -1,168 +0,0 @@ -from hwyShapeExport import export_highway_shape -from skimAppender import SkimAppender -from abmScenario import ScenarioData, LandUse, SyntheticPopulation, TourLists, TripLists -import os -import sys - - -def export_data(fp): - # set file path to completed ABM run scenario folder - # set report folder path - scenarioPath = fp - reportPath = os.path.join(scenarioPath, "report") - - - # initialize base ABM scenario data class - print("Initializing Scenario Data") - scenario_data = ScenarioData(scenarioPath) - - # write out transit TAP park and ride file - print("Writing: Transit PNR Input File") - scenario_data.pnr_taps.to_csv(os.path.join(reportPath, "transitPNR.csv"), index=False) - - - # initialize land use class - # write out MGRA-based input file - print("Initializing Land Use Output") - land_use = LandUse(scenarioPath) - print("Writing: MGRA-Based Input File") - land_use.mgra_input.to_csv(os.path.join(reportPath, "mgraBasedInput.csv"), index=False) - - - # initialize synthetic population class - # write out households and persons files - print("Initializing Synthetic Population Output") - population = SyntheticPopulation(scenarioPath) - - print("Writing: Households File") - population.households.to_csv(os.path.join(reportPath, "households.csv"), index=False) - - print("Writing: Persons File") - population.persons.to_csv(os.path.join(reportPath, "persons.csv"), index=False) - - - # initialize tour list class - # write out tour list files - print("Initializing Tour List Output") - tours = TourLists(scenarioPath) - - print("Writing: Commercial Vehicle Tours") - tours.cvm.to_csv(os.path.join(reportPath, "commercialVehicleTours.csv"), index=False) - - print("Writing: Cross Border Tours") - tours.cross_border.to_csv(os.path.join(reportPath, "crossBorderTours.csv"), index=False) - - print("Writing: Individual Tours") - tours.individual.to_csv(os.path.join(reportPath, "individualTours.csv"), index=False) - - print("Writing: Internal-External Tours") - tours.ie.to_csv(os.path.join(reportPath, "internalExternalTours.csv"), index=False) - - print("Writing: Joint Tours") - tours.joint.to_csv(os.path.join(reportPath, "jointTours.csv"), index=False) - - print("Writing: Visitor Tours") - tours.visitor.to_csv(os.path.join(reportPath, "visitorTours.csv"), index=False) - - - print("Initializing Trip List Output") - - # initialize trip list class - trips = TripLists(scenarioPath) - - # initialize skim appender class - skims = SkimAppender(scenarioPath) - - # write out trip list files - print("Writing: Airport-SAN Trips") - skims.append_skims(trips.airport_san, - auto_only=False, - terminal_skims=False).to_csv( - os.path.join(reportPath, "airportSANTrips.csv"), - index=False) - - print("Writing: Airport-CBX Trips") - skims.append_skims(trips.airport_cbx, - auto_only=False, - terminal_skims=False).to_csv( - os.path.join(reportPath, "airportCBXTrips.csv"), - index=False) - - print("Writing: Commercial Vehicle Trips") - skims.append_skims(trips.cvm, - auto_only=True, - terminal_skims=False).to_csv( - os.path.join(reportPath, "commercialVehicleTrips.csv"), - index=False) - - print("Writing: Cross-Border Trips") - skims.append_skims(trips.cross_border, - auto_only=False, - terminal_skims=False).to_csv( - os.path.join(reportPath, "crossBorderTrips.csv"), - index=False) - - print("Writing: External-External Trips") - trips.ee.to_csv( - os.path.join(reportPath, "externalExternalTrips.csv"), - index=False) - - print("Writing: External-Internal Trips") - trips.ei.to_csv( - os.path.join(reportPath, "externalInternalTrips.csv"), - index=False) - - print("Writing: Individual Trips") - skims.append_skims(trips.individual, - auto_only=False, - terminal_skims=True).to_csv( - os.path.join(reportPath, "individualTrips.csv"), - index=False) - - print("Writing: Internal-External Trips") - skims.append_skims(trips.ie, - auto_only=False, - terminal_skims=False).to_csv( - os.path.join(reportPath, "internalExternalTrips.csv"), - index=False) - - print("Writing: Joint Trips") - skims.append_skims(trips.joint, - auto_only=False, - terminal_skims=True).to_csv( - os.path.join(reportPath, "jointTrips.csv"), - index=False) - - print("Writing: Truck Trips") - trips.truck.to_csv( - os.path.join(reportPath, "truckTrips.csv"), - index=False) - - print("Writing: Visitor Trips") - skims.append_skims(trips.visitor, - auto_only=False, - terminal_skims=False).to_csv( - os.path.join(reportPath, "visitorTrips.csv"), - index=False) - - print("Writing: Zombie AV Trips") - skims.append_skims(trips.zombie_av, - auto_only=True, - terminal_skims=False).to_csv( - os.path.join(reportPath, "zombieAVTrips.csv"), - index=False) - - print("Writing: Zombie TNC Trips") - skims.append_skims(trips.zombie_tnc, - auto_only=True, - terminal_skims=False).to_csv( - os.path.join(reportPath, "zombieTNCTrips.csv"), - index=False) - - print("Writing: Highway Load Shape File") - export_highway_shape(scenarioPath).to_file( - os.path.join(reportPath, "hwyLoad.shp")) - -if __name__ == '__main__': - targets = sys.argv[1:] - export_data(targets[0]) diff --git a/sandag_abm/src/main/python/dataExporter/skimAppender.py b/sandag_abm/src/main/python/dataExporter/skimAppender.py deleted file mode 100644 index cb6c5f1..0000000 --- a/sandag_abm/src/main/python/dataExporter/skimAppender.py +++ /dev/null @@ -1,1895 +0,0 @@ -# -*- coding: utf-8 -*- -""" ABM Scenario Skim Appender Module. - -This module contains classes holding all utilities relating to appending -transportation skims to a completed SANDAG Activity-Based Model (ABM) -scenario. This module is used to append time, distance, cost, and other -related transportation skims to ABM trip lists. - -Notes: - docstring style guide - http://google.github.io/styleguide/pyguide.html -""" - -import itertools -import os -import re -from functools import lru_cache # caching decorator for modules -import numpy as np -import openmatrix as omx # https://github.com/osPlanning/omx-python -import pandas as pd - - -class SkimAppender(object): - """ This class holds all utilities relating to appending transportation - skims to a completed SANDAG Activity-Based Model (ABM) scenario - - Args: - scenario_path: String location of the completed ABM scenario folder - - Methods: - _get_omx_auto_skim_dataset: Maps ABM trip list records to OMX files - and OMX skim matrices - append_skims: Master method to append all skims to ABM trip lists - auto_fare_cost: Appends auto fare cost to ABM trip list records - auto_operating_cost: Appends auto operating cost to ABM trip list records - auto_terminal_skims: Appends auto-mode terminal walk time and distance - from the zone.term file to ABM trip list records - auto_wait_time: Appends auto wait times to ABM trip list records - bicycle_skims: Appends bicycle mode skims (time, distance) to ABM trip - list records - drive_transit_skims: Appends input file accessam.csv drive to - transit auto mode skims (time, distance, no cost) to ABM trip - list records - omx_auto_skim_appender: Appends OMX auto mode skims (time, distance, - cost) to ABM trip list records - omx_transit_skims: Appends OMX transit mode skims (time, distance, - cost) to transit mode trip list records - tnc_fare_cost: Appends TNC fare costs to ABM trip list records - tnc_wait_time: Appends TNC wait time to ABM trip list records - walk_skims: Appends walk/micro-mobility/micro-transit mode skims - (time, distance, cost) to ABM trip list records for - walk/micro-mobility/micro-transit mode trips - _walk_skims_at: Appends walk/micro-mobility/micro-transit mode skims - (time, distance, cost) to ABM trip list records for - walk/micro-mobility/micro-transit mode trips that do not use auto - mode skim sets for trip time - _walk_skims_auto: Appends walk/micro-mobility/micro-transit mode skims - (time, distance, cost) to ABM trip list records for - walk/micro-mobility/micro-transit mode trips that use auto mode - skim sets for trip time - walk_transit_skims: Appends walk/micro-mobility/micro-transit - access/egress to/from transit to ABM transit mode trip list - records - - Properties: - mgra_xref: Pandas DataFrame geography cross-reference of MGRAs to - TAZs and LUZs - properties: Dictionary of ABM properties file token values - (conf/sandag_abm.properties) """ - - def __init__(self, scenario_path: str) -> None: - self.scenario_path = scenario_path - - @property - @lru_cache(maxsize=1) - def mgra_xref(self) -> pd.DataFrame: - """ Cross reference of Master Geographic Reference Area (MGRA) model - geography to Transportation Analysis Zone (TAZ) and Land Use Zone - (LUZ) model geographies. Cross reference is stored in each ABM - scenario input MGRA file (input/mgra13_based_input<>.csv). - """ - - # load the mgra based input file - fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" - - mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), - usecols=["mgra", # MGRA geography - "taz", # TAZ geography - "luz_id", # LUZ geography - "MicroAccessTime"], # Micro-Mobility AccessTime - dtype={"mgra": "int16", - "taz": "int16", - "luz_id": "int16"}) - - # genericize column names - mgra.rename(columns={"mgra": "MGRA", - "taz": "TAZ", - "luz_id": "LUZ"}, - inplace=True) - - return mgra - - @property - @lru_cache(maxsize=1) - def properties(self) -> dict: - """ Get the ABM scenario properties from the ABM scenario - properties file (conf/sandag_abm.properties). - - The return dictionary contains the following ABM scenario properties: - accessTimeMicroTransit - Micro-Transit access time in minutes - aocFuel - auto operating fuel cost in $/mile - aocMaintenance - auto operating maintenance cost in $/mile - baseFareMicroMobility - initial Micro-Mobility fare in $ - baseFareMicroTransit - initial Micro-Mobility fare in $ - baseFareNonPooledTNC - initial Non-Pooled TNC fare in $ - baseFarePooledTNC - initial Pooled TNC fare in $ - baseFareTaxi - initial taxi fare in $ - bicycleSpeed - bicycle mode speed (miles/hour) - costMinimumNonPooledTNC - minimum Non-Pooled TNC fare cost in $ - costMinimumPooledTNC - minimum Pooled TNC fare cost in $ - costPerMileFactorAV - auto operating cost per mile factor to - apply to AV trips - costPerMileNonPooledTNC - Non-Pooled TNC fare cost per mile in $ - costPerMilePooledTNC - Pooled TNC fare cost per mile in $ - costPerMileTaxi - Taxi fare cost per mile in $ - costPerMinuteMicroMobility - Micro-Mobility fare cost per minute in $ - costPerMinuteMicroTransit - Micro-Transit fare cost per minute in $ - costPerMinuteNonPooledTNC - Non-Pooled TNC fare cost per minute in $ - costPerMinutePooledTNC - Pooled TNC fare cost per minute in $ - costPerMinuteTaxi - Taxi fare cost per minute in $ - microMobilitySpeed - Micro-Mobility mode speed (miles/hour) - microTransitSpeed - Micro-Transit mode speed (miles/hour) - terminalTimeFactorAV - terminal time factor to apply to AV trips - waitTimeMicroTransit - Micro-Mobility wait time in minutes - waitTimeNonPooledTNC - list of mean wait times in minutes for - Non-Pooled TNC by PopEmpDenPerMi categories - (see waitTimePopEmpDenPerMi) - waitTimePooledTNC - list of mean wait times in minutes for Pooled - TNC by PopEmpDenPerMi categories (see waitTimePopEmpDenPerMi) - waitTimePopEmpDenPerMi - list of MGRA-Based input file - PopEmpDenPerMi values defining the wait time categories for - Taxi/TNC modes - waitTimeTaxi - list of mean wait times in minutes for Taxi by - PopEmpDenPerMi categories (see waitTimePopEmpDenPerMi) - walkSpeed - walk mode speed (miles/hour) - year - analysis year of the ABM scenario - - Returns: - A dictionary defining the ABM scenario properties. """ - - # create dictionary holding ABM properties file information - # each property contains a dictionary {line, value} where the line - # is the string to match in the properties file to - # return the value of the property - lookup = { - "accessTimeMicroTransit": { - "line": "active.microtransit.accessTime=", - "type": "float", - "value": None - }, - "aocFuel": { - "line": "aoc.fuel=", - "type": "float", - "value": None - }, - "aocMaintenance": { - "line": "aoc.maintenance=", - "type": "float", - "value": None - }, - "baseFareMicroMobility": { - "line": "active.micromobility.fixedCost=", - "type": "float", - "value": None - }, - "baseFareMicroTransit": { - "line": "active.microtransit.fixedCost=", - "type": "float", - "value": None - }, - "baseFareNonPooledTNC": { - "line": "TNC.single.baseFare=", - "type": "float", - "value": None - }, - "baseFarePooledTNC": { - "line": "TNC.shared.baseFare=", - "type": "float", - "value": None - }, - "baseFareTaxi": { - "line": "taxi.baseFare=", - "type": "float", - "value": None - }, - "bicycleSpeed": { - "line": "active.bike.minutes.per.mile=", - "type": "float", - "value": None}, - "costMinimumNonPooledTNC": { - "line": "TNC.single.costMinimum=", - "type": "float", - "value": None - }, - "costMinimumPooledTNC": { - "line": "TNC.shared.costMinimum=", - "type": "float", - "value": None - }, - "costPerMileFactorAV": { - "line": "Mobility.AV.CostPerMileFactor=", - "type": "float", - "value": None - }, - "costPerMileNonPooledTNC": { - "line": "TNC.single.costPerMile=", - "type": "float", - "value": None - }, - "costPerMilePooledTNC": { - "line": "TNC.shared.costPerMile=", - "type": "float", - "value": None - }, - "costPerMileTaxi": { - "line": "taxi.costPerMile=", - "type": "float", - "value": None - }, - "costPerMinuteMicroMobility": { - "line": "active.micromobility.variableCost=", - "type": "float", - "value": None - }, - "costPerMinuteMicroTransit": { - "line": "active.microtransit.variableCost=", - "type": "float", - "value": None - }, - "costPerMinuteNonPooledTNC": { - "line": "TNC.single.costPerMinute=", - "type": "float", - "value": None - }, - "costPerMinutePooledTNC": { - "line": "TNC.shared.costPerMinute=", - "type": "float", - "value": None - }, - "costPerMinuteTaxi": { - "line": "taxi.costPerMinute=", - "type": "float", - "value": None - }, - "microMobilitySpeed": { - "line": "active.micromobility.speed=", - "type": "float", - "value": None}, - "microTransitSpeed": { - "line": "active.microtransit.speed=", - "type": "float", - "value": None}, - "terminalTimeFactorAV": { - "line": "Mobility.AV.TerminalTimeFactor=", - "type": "float", - "value": None - }, - "waitTimeMicroTransit": { - "line": "active.microtransit.waitTime=", - "type": "float", - "value": None - }, - "waitTimeNonPooledTNC": { - "line": "TNC.single.waitTime.mean=", - "type": "list", - "value": None - }, - "waitTimePooledTNC": { - "line": "TNC.shared.waitTime.mean=", - "type": "list", - "value": None - }, - "waitTimePopEmpDenPerMi": { - "line": "WaitTimeDistribution.EndPopEmpPerSqMi=", - "type": "list", - "value": None - }, - "waitTimeTaxi": { - "line": "Taxi.waitTime.mean=", - "type": "list", - "value": None - }, - "walkSpeed": { - "line": "active.walk.minutes.per.mile=", - "type": "float", - "value": None}, - "year": { - "line": "scenarioYear=", - "type": "int", - "value": None} - } - - # open the ABM scenario properties file - file = open(os.path.join(self.scenario_path, "conf", "sandag_abm.properties"), "r") - - # loop through each line of the properties file - for line in file: - # strip all white space from the line - line = line.replace(" ", "") - - # for each element of the properties dictionary - for name in lookup: - item = lookup[name] - - if item["line"] is not None: - match = re.compile(item["line"]).match(line) - # if the properties file contains the matching line - if match: - # for waitTime properties create list from matching line - # - if "waitTime" in name and item["type"] == "list": - value = line[match.end():].split(",") - value = list(map(float, value)) - # otherwise take the final element of the line - else: - value = line[match.end():] - - # update the dictionary value using the appropriate data type - if item["type"] == "float": - value = float(value) - elif item["type"] == "int": - value = int(value) - else: - pass - - item["value"] = value - break - - file.close() - - # convert the property name and value to a non-nested dictionary - results = {} - for name in lookup: - results[name] = lookup[name]["value"] - - # convert auto operating costs from cents per mile to dollars per mile - results["aocFuel"] = results["aocFuel"] / 100 - results["aocMaintenance"] = results["aocMaintenance"] / 100 - - # convert AT speeds from minutes per mile to miles per hour - results["bicycleSpeed"] = (results["bicycleSpeed"] * 1/60)**-1 - results["microMobilitySpeed"] = (results["microMobilitySpeed"] * 1 / 60) ** -1 - results["microTransitSpeed"] = (results["microTransitSpeed"] * 1 / 60) ** -1 - results["walkSpeed"] = (results["walkSpeed"] * 1/60)**-1 - - return results - - @staticmethod - def _get_omx_auto_skim_dataset(df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame and returns a subset of trips - with two columns indicating the proper auto-mode omx file name and - skim matrix to use to append omx auto-mode transportation skims. - - If trips in the input DataFrame are not mapped to omx files - and matrices (non-auto mode trips) they are not present in the return - DataFrame. The return DataFrame is an interim DataFrame ready for - input to the omx_auto_skim_appender - - The process uses the trip departure ABM 5 TOD period, trip mode, - transponder availability, and the trip value of time (vot) category - to select the correct auto skim-set file and matrix to use to get - the auto-mode trip times, distances, and toll costs. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [transponderAvailable] - indicator if transponder available - on the trip (False, True) - [valueOfTimeCategory] - trip value of time categories - (Low, Medium, High) - [originTAZ] - trip origin TAZ (1-4996) - [destinationTAZ] - trip destination TAZ (1-4996) - [parkingTAZ] - trip parking TAZ ("", 1-4996) - - Returns: - A Pandas DataFrame containing mapping of auto-mode trips to omx - files and matrices containing auto-mode skims: - [tripID] - unique identifier of a trip - [omxFileName] - omx skim file name (e.g. traffic_skims_EA) - [matrixName] - omx skim matrix name (e.g. EA_HOV2_M) - [originTAZ] - trip origin TAZ (1-4996) - [destinationTAZ] - trip destination TAZ (1-4996) """ - - # set destinationTAZ equal to parkingTAZ is parkingTAZ is not missing - # auto portion of trip ends at the parkingTAZ - df.destinationTAZ = np.where(pd.notna(df.parkingTAZ), - df.parkingTAZ, - df.destinationTAZ).astype("int16") - - # create mappings from trip list to skim matrices - # segmented by SOV and Non-SOV as process differs between the two - # due to Transponder/Non-Transponder skimming in SOV mode - # SOV - departTimeFiveTod + tripMode + transponderAvailable + - # valueOfTimeCategory - # nonSOV - departTimeFiveTod + tripMode + valueOfTimeCategory - skim_map = {"SOV": {"values": [[1, 2, 3, 4, 5], - ["Drive Alone"], - [False, True], - ["Low", "Medium", "High"]], - "labels": [["EA", "AM", "MD", "PM", "EV"], - ["SOV"], - ["NT", "TR"], - ["L", "M", "H"]], - "cols": ["departTimeFiveTod", - "tripMode", - "transponderAvailable", - "valueOfTimeCategory", - "matrixName"]}, - "Non-SOV": {"values": [[1, 2, 3, 4, 5], - ["Shared Ride 2", - "Shared Ride 3+", - "Light Heavy Duty Truck", - "Medium Heavy Duty Truck", - "Heavy Heavy Duty Truck", - "Taxi", - "Non-Pooled TNC", - "Pooled TNC", - "School Bus"], - ["Low", "Medium", "High"]], - "labels": [["EA", "AM", "MD", "PM", "EV"], - ["HOV2", "HOV3", "TRK", "TRK", - "TRK", "HOV3", "HOV3", "HOV3", - "HOV3"], - ["L", "M", "H"]], - "cols": ["departTimeFiveTod", - "tripMode", - "valueOfTimeCategory", - "matrixName"]}} - - # initialize empty auto trips DataFrame - trips = pd.DataFrame() - - # map possible values of trip list columns to skim matrix names - # filter input DataFrame to auto trips mapped to skim matrices - for key in skim_map: - values = skim_map[key]["values"] - labels = skim_map[key]["labels"] - cols = skim_map[key]["cols"] - - mapping = [list(i) + ["_".join(j)] for i, j in - zip(itertools.product(*values), - itertools.product(*labels))] - - # create Pandas DataFrame lookup table of column values - # to skim matrix names - lookup = pd.DataFrame(mapping, columns=cols) - lookup["omxFileName"] = "traffic_skims_" + lookup["matrixName"].str[0:2] - - # set lookup DataFrame data types - lookup = lookup.astype({ - "departTimeFiveTod": "int8", - "tripMode": "category", - "valueOfTimeCategory": "category", - "omxFileName": "category", - "matrixName": "category"}) - - # merge lookup table to trip list and append to auto trips DataFrame - trips = trips.append(df.merge(lookup, how="inner"), ignore_index=True) - - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - trips.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - return trips[["tripID", - "omxFileName", - "matrixName", - "originTAZ", - "destinationTAZ"]] - - def append_skims(self, df: pd.DataFrame, auto_only: bool, terminal_skims: bool) -> pd.DataFrame: - """ Takes an input Pandas DataFrame, runs all skimming class - methods and appends all skims to the input Pandas DataFrame. - See documentation of included class methods. Additionally creates and - appends three fields to the input Pandas DataFrame: - [timeTotal] - total trip time in minutes - [distanceTotal] - total trip distance in miles - [costTotal] - total trip cost in dollars - - Args: - df: Input Pandas DataFrame containing fields required by the - included class methods. - terminal_skims: Boolean of whether to include auto-mode terminal - skims, only apply to Resident model trip lists. - auto_only: Boolean of whether to include only basic auto-mode - skims, applies to Commercial Vehicle, Zombie AV, and Zombie - TNC trip lists. - - Returns: - A Pandas DataFrame containing all skims appended by the included - class methods. The DataFrame contains all original fields - of the input DataFrame along with fields appended by the - aforementioned class methods. """ - - # append omx auto skims - df = self.omx_auto_skim_appender(df) - - # append auto operating cost - df = self.auto_operating_cost(df) - - if auto_only: - pass - else: - # append auto terminal skims if applicable - if terminal_skims: - df = self.auto_terminal_skims(df) - - # append TNC fare costs - df = self.tnc_fare_cost(df) - - # append TNC wait times - df = self.tnc_wait_time(df) - - # append omx transit skims - df = self.omx_transit_skims(df) - - # append drive to transit skims - # auto-operating cost not included - # TNC fare costs and wait times not included - df = self.drive_transit_skims(df) - - # append (micro-mobility/micro-transit/walk) to transit skims - df = self.walk_transit_skims(df) - - # append micro-mobility/micro-transit/walk skims - df = self.walk_skims(df) - - # append bicycle skims - df = self.bicycle_skims(df) - - # append total trip time, distance, and cost skims - # transit in vehicle times by line haul mode and initial wait time - # are not included due to double-counting - transit_line_haul_cols = ["timeTier1TransitInVehicle", - "timeFreewayRapidTransitInVehicle", - "timeArterialRapidTransitInVehicle", - "timeExpressBusTransitInVehicle", - "timeLocalBusTransitInVehicle", - "timeLightRailTransitInVehicle", - "timeCommuterRailTransitInVehicle", - "timeTransitInitialWait"] - - df["timeTotal"] = df[df.columns.difference(transit_line_haul_cols)].filter(regex="^time").sum(axis=1) - - df["distanceTotal"] = df.filter(regex="^distance").sum(axis=1) - - df["costTotal"] = df.filter(regex="^cost").sum(axis=1) - - # sort return DataFrame by tripID for user-experience - # and faster database loading via ORDER hints - df.sort_values(by="tripID", inplace=True) - - return df - - def auto_operating_cost(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [avUsed]) and the fields appended by the - SkimAppender class method omx_auto_skim_appender - ([distanceDrive]) and returns the associated - auto-mode operating cost. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [avUsed] - indicator if AV used on trip (True, False) - [distanceDrive] - distance in miles for auto mode - - Returns: - A Pandas DataFrame containing all associated auto-mode - operating costs. The DataFrame contains all original fields - of the input DataFrame along with the field: - [costOperatingDrive] - auto operating cost in $ """ - - # calculate auto operating cost as fuel+maintenance cents per mile - # note this is scaled for AV trips by a factor - aoc = self.properties["aocFuel"] + self.properties["aocMaintenance"] - aoc_av = aoc * self.properties["costPerMileFactorAV"] - - # calculate auto operating cost for auto-mode trips - # exclude Taxi/TNC as operating cost is passed to drivers - # who are not considered part of the model universe - auto_modes = ["Drive Alone", - "Shared Ride 2", - "Shared Ride 3+", - "Light Heavy Duty Truck", - "Medium Heavy Duty Truck", - "Heavy Heavy Duty Truck"] - - conditions = [ - np.array(df["tripMode"].isin(auto_modes) & ~df["avUsed"], - dtype="bool"), - np.array(df["tripMode"].isin(auto_modes) & df["avUsed"], - dtype="bool") - ] - - choices = [df["distanceDrive"] * aoc, - df["distanceDrive"] * aoc_av] - - df["costOperatingDrive"] = pd.Series( - np.select(conditions, choices, default=np.NaN), - dtype="float32") - - # return input DataFrame with appended auto operating cost column - return df - - def auto_terminal_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [destinationTAZ]) and appends the auto-mode - terminal walk time from the input/zone.term file. Distance is created - from the walk speed specified in the properties file - conf/sandag_abm.properties - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [destinationTAZ] - indicator if AV used on trip (True, False) - - Returns: - A Pandas DataFrame containing auto-mode terminal walk time and - distance. The DataFrame contains all original fields of the input - DataFrame along with the fields: - [timeAutoTerminalWalk] - auto terminal walk time in minutes - [distanceAutoTerminalWalk] - auto terminal walk distance in - miles - """ - # read in auto terminal time fixed width file - skims = pd.read_fwf( - os.path.join(self.scenario_path, "input", "zone.term"), - widths=[5, 7], - names=["destinationTAZ", - "timeAutoTerminalWalk"], - dtype={"destinationTAZ": "int16", - "timeAutoTerminalWalk": "float32"} - ) - - # merge auto terminal times into input DataFrame - # add 0s for TAZs with no terminal times - df = df.merge(skims, on="destinationTAZ", how="left") - df["timeAutoTerminalWalk"] = df["timeAutoTerminalWalk"].fillna(0) - - # set auto terminal times to 0 for non-auto modes - # reduce auto terminal time if AV is used - auto_modes = ["Drive Alone", "Shared Ride 2", "Shared Ride 3+"] - av_factor = self.properties["terminalTimeFactorAV"] - - conditions = [ - np.array(df["tripMode"].isin(auto_modes) & ~df["avUsed"], - dtype="bool"), - np.array(df["tripMode"].isin(auto_modes) & df["avUsed"], - dtype="bool") - ] - - choices = [df["timeAutoTerminalWalk"], - df["timeAutoTerminalWalk"] * av_factor] - - df["timeAutoTerminalWalk"] = pd.Series( - np.select(conditions, choices, default=np.NaN), - dtype="float32") - - # create auto terminal distance from walk speed property - df["distanceAutoTerminalWalk"] = pd.Series( - df["timeAutoTerminalWalk"] * self.properties["walkSpeed"] / 60, - dtype="float32") - - return df - - def bicycle_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originMGRA], [destinationMGRA], [originTAZ], - [destinationTAZ]) and returns the associated bicycle mode skims for - time and distance. - - Bicycle mode skims are given at the MGRA-MGRA level by the - output/bikeMgraLogsum.csv file and TAZ-TAZ level by the - output/bikeTazLogsum.csv file. If a MGRA-MGRA o-d pair is not present - in the MGRA-MGRA level then the TAZ-TAZ level skim is used. Note the - skims only provide time in minutes so distance is created using the - bicycle speed property set in the conf/sandag_abm.properties file. - Non-bicycle modes have all skims set to NaN. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA (1-23002) - [destinationMGRA] - trip destination MGRA (1-23002) - [originTAZ] - trip origin TAZ (1-4996) - [destinationTAZ] - trip destination TAZ (1-4996) - - Returns: - A Pandas DataFrame containing all associated bicycle mode - skims for time and distance. The DataFrame contains all the - original fields of the input DataFrame along with the fields: - [timeBike] - time in minutes for bicycle mode - [distanceBike] - distance in miles for bicycle mode """ - - # load the MGRA-MGRA bicycle skims - mgra_skims = pd.read_csv( - os.path.join(self.scenario_path, "output", "bikeMgraLogsum.csv"), - usecols=["i", # origin MGRA geography - "j", # destination MGRA geography - "time"], # time in minutes - dtype={"i": "int16", - "j": "int16", - "time": "float32"} - ) - - # load the TAZ-TAZ bicycle skims - taz_skims = pd.read_csv( - os.path.join(self.scenario_path, "output", "bikeTazLogsum.csv"), - usecols=["i", # origin TAZ geography - "j", # destination TAZ geography - "time"], # time in minutes - dtype={"i": "int16", - "j": "int16", - "time": "float32"} - ) - - # merge the skims with the input DataFrame bicycle mode records - # use left outer joins to keep all bicycle mode records - # if MGRA-MGRA skims do not exist - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - records = df.loc[(df["tripMode"] == "Bike")].copy() - records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - records = records.merge( - right=mgra_skims, - how="left", - left_on=["originMGRA", "destinationMGRA"], - right_on=["i", "j"] - ) - records = records.merge( - right=taz_skims, - how="left", - left_on=["originTAZ", "destinationTAZ"], - right_on=["i", "j"], - suffixes=["MGRA", "TAZ"] - ) - - # if MGRA-MGRA skims do not exist use TAZ-TAZ skims - records["timeBike"] = pd.Series( - np.where(records["timeMGRA"].isna(), - records["timeTAZ"], - records["timeMGRA"]), - dtype="float32") - - # calculate distance using bicycle speed - records["distanceBike"] = pd.Series( - records["timeBike"] * self.properties["bicycleSpeed"] / 60, - dtype="float32") - - # merge result set DataFrame back into initial trip list - # keep missing skim records as missing skim means no bike trip - records = records[["tripID", "timeBike", "distanceBike"]] - df = df.merge(records, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df - - def drive_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originTAZ], [destinationTAZ], - [boardingTAP], [alightingTAP], and [inbound]) and returns the - associated drive to transit auto-mode skims for time and distance. - - The model uses an on-the-fly input file (input\accessam.csv) for - drive to transit auto-mode skims and assumes no fare or toll costs. - Non-drive to transit modes have all skims set to NaN. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originTAZ] - trip origin TAZ (1-4996) - [destinationTAZ] - trip destination TAZ (1-4996) - [boardingTAP] - trip boarding transit access point (TAP) - [alightingTAP] - trip alighting transit access point (TAP) - [inbound] - direction of trip on tour (False, True) - - Returns: - A Pandas DataFrame containing all associated drive to transit - auto-mode skims for time and distance (assumed no toll cost). - The DataFrame contains all the original fields of the input - DataFrame along with the fields: - [timeDriveTransit] - time in minutes for auto mode - [distanceDriveTransit] - distance in miles for auto mode """ - - # read in the input accessam csv file - # this file is used in place of auto skim matrices for drive to transit - skims = pd.read_csv(self.scenario_path + "/input/accessam.csv", - names=["TAZ", # TAZ geography - "TAP", # transit access point (TAP) - "timeDriveTransit", # time in minutes - "distanceDriveTransit", # distance in miles - "mode"], - usecols=["TAZ", - "TAP", - "timeDriveTransit", - "distanceDriveTransit"], - dtype={"TAZ": "int16", - "TAP": "int16", - "timeDriveTransit": "float32", - "distanceDriveTransit": "float32"}) - - # select drive to transit records - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - modes = ["Park and Ride to Transit - Local Bus", - "Park and Ride to Transit - Premium Transit", - "Park and Ride to Transit - Local Bus and Premium Transit", - "Kiss and Ride to Transit - Local Bus", - "Kiss and Ride to Transit - Premium Transit", - "Kiss and Ride to Transit - Local Bus and Premium Transit", - "TNC to Transit - Local Bus", - "TNC to Transit - Premium Transit", - "TNC to Transit - Local Bus and Premium Transit"] - records = df.loc[(df["tripMode"].isin(modes))].copy() - records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - # create MGRA-TAP origin-destinations based on inbound direction - records["MGRA"] = np.where(records["inbound"], - records["destinationMGRA"], - records["originMGRA"]) - - records["TAP"] = np.where(records["inbound"], - records["alightingTAP"], - records["boardingTAP"]) - - # use the origin/destination MGRA to derive the origin/destination - # TAZ, this accounts for issues where external TAZs (1-12) do not have - # TAP-based skims, the skims are derived from the internal TAZ of the - # internal MGRA of the trip origin/destination - records = records.merge(self.mgra_xref, on="MGRA") - - # merge with the drive to transit access skims - records = records[["tripID", "TAZ", "TAP"]] - records = records.merge(skims, how="inner", on=["TAZ", "TAP"]) - records = records[["tripID", "timeDriveTransit", "distanceDriveTransit"]] - - # merge result set DataFrame back into initial trip list - # keep missing skim records as missing skim means no transit trip - df = df.merge(records, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df - - def omx_auto_skim_appender(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame and returns the DataFrame with - associated auto-mode skims for time, distance, and toll cost appended. - - The process uses the trip departure ABM 5 TOD period, trip mode, - transponder availability, and the trip value of time (vot) category - to select the correct auto skim-set to get the trip time, distance, - and toll costs. Non-auto modes have all skims set to NaN. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [transponderAvailable] - indicator if transponder available - on the trip (False, True) - [valueOfTimeCategory] - trip value of time categories - (Low, Medium, High) - [originTAZ] - trip origin TAZ (1-4996) - [destinationTAZ] - trip destination TAZ (1-4996) - [parkingTAZ] - trip parking TAZ ("", 1-4996) - - Returns: - The input Pandas DataFrame with auto-mode skims for time, - distance, and toll cost appended. The DataFrame contains all the - original fields of the input DataFrame along with the fields: - [timeDrive] - time in minutes for auto mode - [distanceDrive] - distance in miles for auto mode - [costTollDrive] - toll cost in $ for auto mode """ - - # prepare the input DataFrame for skim appending - # selecting records that use the input omx file - df_map = self._get_omx_auto_skim_dataset(df) - - # initialize output skim list - output = [] - - for omx_fn in df_map.omxFileName.unique(): - - # open the input omx file and TAZ:element mapping - fn = os.path.join(self.scenario_path, "output", omx_fn + ".omx") - omx_file = omx.open_file(fn) - omx_map = omx_file.mapping("zone_number") - - # filter records mapped to omx file - records_omx = df_map.loc[df_map.omxFileName == omx_fn] - - # for each skim matrix in the data-set - for matrix in records_omx.matrixName.unique(): - # filter records mapped to skim matrix - records = records_omx.loc[records_omx.matrixName == matrix] - - # create set of unique origin-destination pairs - # get time, distance, cost associated with the o-d pairs - od = set(zip(records.originTAZ, records.destinationTAZ)) - o, d = zip(*od) - - # map o-ds to omx matrix indices - o_idx = [omx_map[number] for number in o] - d_idx = [omx_map[number] for number in d] - - skims = list(zip( - [omx_fn] * len(o), - [matrix] * len(o), - o, d, - omx_file[matrix + "_TIME"][o_idx, d_idx], - omx_file[matrix + "_DIST"][o_idx, d_idx], - omx_file[matrix + "_TOLLCOST"][o_idx, d_idx] / 100)) - - output.extend(skims) - - omx_file.close() - - # create DataFrame from output skim list - output = pd.DataFrame(data=output, - columns=["omxFileName", - "matrixName", - "originTAZ", - "destinationTAZ", - "timeDrive", - "distanceDrive", - "costTollDrive"]) - - # set data types of output skims list - output = output.astype({ - "omxFileName": "category", - "matrixName": "category", - "originTAZ": "int16", - "destinationTAZ": "int16", - "timeDrive": "float32", - "distanceDrive": "float32", - "costTollDrive": "float32" - }) - - # merge the output skim list back to the mapped input DataFrame - # appending auto skims to tripIDs - output = output.merge(right=df_map, how="inner") - - output = output[["tripID", - "timeDrive", - "distanceDrive", - "costTollDrive"]] - - # merge the output skim list back to the original input DataFrame - # keep missing skim records as missing skim means no auto trip - df = df.merge(right=output, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df - - def omx_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the fields - ([tripID], [departTimeFiveTod], [tripMode], [boardingTAP], - [alightingTAP]) and returns the associated transit mode skims for time, - distance, and fare cost. - - The process uses the trip departure ABM 5 TOD period and trip mode to - select the correct transit skim-set to use to get the trip time, - distance, and fare costs. Non-transit modes have all skims set to NaN. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [departTimeFiveTod] - trip departure ABM 5 TOD periods (1-5) - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [boardingTAP] - trip boarding transit access point (TAP) - [alightingTAP] - trip alighting transit access point (TAP) - - Returns: - A Pandas DataFrame containing all associated transit-mode skims for - time, distance, and fare cost. The DataFrame contains all the - original fields of the input DataFrame along with the fields: - [timeTransitInVehicle] - transit in-vehicle time in minutes - [timeTier1TransitInVehicle] - tier 1 line haul mode transit - in-vehicle time in minutes - [timeFreewayRapidTransitInVehicle] - freeway rapid line haul - mode transit in-vehicle time in minutes - [timeArterialRapidTransitInVehicle] - arterial rapid line haul - mode transit in-vehicle time in minutes - [timeExpressBusTransitInVehicle] - express bus line haul - mode transit in-vehicle time in minutes - [timeLocalBusTransitInVehicle] - local bus line haul - mode transit in-vehicle time in minutes - [timeLightRailTransitInVehicle] - light rail line haul - mode transit in-vehicle time in minutes - [timeCommuterRailTransitInVehicle] - commuter rail line haul - mode transit in-vehicle time in minutes - [timeTransitInitialWait] - initial transit wait time in minutes - [timeTransitWait] - total transit wait time in minutes - [timeTransitWalk] - total transit walk time in minutes - [distanceTransitInVehicle] - transit in-vehicle distance in miles - [distanceTransitWalk] - total transit walk distance in miles - [costFareTransit] - fare cost in $ for transit mode - [transfersTransit] - number of transfers """ - - # create mappings from trip list to skim matrices - # departTimeFiveTod + tripMode - skim_map = {"values": [[1, 2, 3, 4, 5], - ["Walk to Transit - Local Bus", - "Walk to Transit - Premium Transit", - "Walk to Transit - Local Bus and Premium Transit", - "Park and Ride to Transit - Local Bus", - "Park and Ride to Transit - Premium Transit", - "Park and Ride to Transit - Local Bus and Premium Transit", - "Kiss and Ride to Transit - Local Bus", - "Kiss and Ride to Transit - Premium Transit", - "Kiss and Ride to Transit - Local Bus and Premium Transit", - "TNC to Transit - Local Bus", - "TNC to Transit - Premium Transit", - "TNC to Transit - Local Bus and Premium Transit"]], - "labels": [["EA", "AM", "MD", "PM", "EV"], - ["BUS", "PREM", "ALLPEN", - "BUS", "PREM", "ALLPEN", - "BUS", "PREM", "ALLPEN", - "BUS", "PREM", "ALLPEN"]], - "cols": ["departTimeFiveTod", - "tripMode", - "matrixName"]} - - # initialize empty result set DataFrame - result = pd.DataFrame() - - # map possible values of trip list columns to skim matrix names - mapping = [list(i) + ["_".join(j)] for i, j in - zip(itertools.product(*skim_map["values"]), - itertools.product(*skim_map["labels"]))] - - # create Pandas DataFrame lookup table of column values - # to skim matrix names - lookup = pd.DataFrame(mapping, columns=skim_map["cols"]) - - # set data types of lookup table - lookup = lookup.astype({ - "departTimeFiveTod": "int8", - "tripMode": "category", - "matrixName": "category" - }) - - # merge lookup table to trip list make DataFrame unique by tripID - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - trips = df.merge(lookup, how="inner") - trips.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - # open omx transit skim file and get TAP:element matrix mapping - omx_file = omx.open_file(self.scenario_path + "/output/transit_skims.omx") - omx_map = omx_file.mapping("zone_number") - - # for each skim matrix in the data-set - for matrix in trips.matrixName.unique(): - # select records that use the skim matrix - records = trips.loc[(trips["matrixName"] == matrix)].copy() - - # get lists of o-ds - o = records.boardingTAP.astype("int16").tolist() - d = records.alightingTAP.astype("int16").tolist() - - # map o-ds to omx matrix indices - o_idx = [omx_map[number] for number in o] - d_idx = [omx_map[number] for number in d] - - # append skims - records["timeTransitInVehicle"] = omx_file[matrix + "_TOTALIVTT"][o_idx, d_idx] - records["timeTier1TransitInVehicle"] = omx_file[matrix + "_TIER1IVTT"][o_idx, d_idx] - records["timeFreewayRapidTransitInVehicle"] = omx_file[matrix + "_BRTYELIVTT"][o_idx, d_idx] - records["timeArterialRapidTransitInVehicle"] = omx_file[matrix + "_BRTREDIVTT"][o_idx, d_idx] - records["timeExpressBusTransitInVehicle"] = omx_file[matrix + "_EXPIVTT"][o_idx, d_idx] - records["timeLocalBusTransitInVehicle"] = omx_file[matrix + "_BUSIVTT"][o_idx, d_idx] - records["timeLightRailTransitInVehicle"] = omx_file[matrix + "_LRTIVTT"][o_idx, d_idx] - records["timeCommuterRailTransitInVehicle"] = omx_file[matrix + "_CMRIVTT"][o_idx, d_idx] - records["timeTransitInitialWait"] = omx_file[matrix + "_FIRSTWAIT"][o_idx, d_idx] - records["timeTransitWait"] = omx_file[matrix + "_TOTALWAIT"][o_idx, d_idx] - records["timeTransitWalk"] = omx_file[matrix + "_TOTALWALK"][o_idx, d_idx] - records["distanceTransitInVehicle"] = omx_file[matrix + "_TOTDIST"][o_idx, d_idx] - records["costFareTransit"] = omx_file[matrix + "_FARE"][o_idx, d_idx] - records["transfersTransit"] = omx_file[matrix + "_XFERS"][o_idx, d_idx] - records["distanceTransitWalk"] = records.timeTransitWalk * self.properties["walkSpeed"] / 60 - - # set skim data types - records = records.astype({ - "timeTransitInVehicle": "float32", - "timeTransitInitialWait": "float32", - "timeTransitWait": "float32", - "timeTransitWalk": "float32", - "distanceTransitInVehicle": "float32", - "costFareTransit": "float32", - "transfersTransit": "float32", - "distanceTransitWalk": "float32"}) - - records = records[["tripID", - "timeTransitInVehicle", - "timeTier1TransitInVehicle", - "timeFreewayRapidTransitInVehicle", - "timeArterialRapidTransitInVehicle", - "timeExpressBusTransitInVehicle", - "timeLocalBusTransitInVehicle", - "timeLightRailTransitInVehicle", - "timeCommuterRailTransitInVehicle", - "timeTransitInitialWait", - "timeTransitWait", - "timeTransitWalk", - "distanceTransitInVehicle", - "distanceTransitWalk", - "costFareTransit", - "transfersTransit"]] - - result = result.append(records, ignore_index=True) - - omx_file.close() - - skim_cols = ["timeTransitInVehicle", - "timeTier1TransitInVehicle", - "timeFreewayRapidTransitInVehicle", - "timeArterialRapidTransitInVehicle", - "timeExpressBusTransitInVehicle", - "timeLocalBusTransitInVehicle", - "timeLightRailTransitInVehicle", - "timeCommuterRailTransitInVehicle", - "timeTransitInitialWait", - "timeTransitWait", - "timeTransitWalk", - "distanceTransitInVehicle", - "distanceTransitWalk", - "costFareTransit", - "transfersTransit"] - - # if there are no transit trip skim records - if result.empty: - # append skim columns to input DataFrame - # set to missing as a missing skim means no transit trip - for col in skim_cols: - df[col] = np.NaN - else: - # merge result set DataFrame back into initial trip list - # keep missing skim records as missing skim means no transit trip - df = df.merge(result, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df - - def tnc_fare_cost(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [avUsed]) and the fields appended by the - SkimAppender class method omx_auto_skim_appender - ([timeDrive], [distanceDrive]) and returns the associated - fare costs. - - Note that drive to transit modes assume no fare costs and are not - included here. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [timeDrive] - time in minutes for auto mode - [distanceDrive] - distance in miles for auto mode - - Returns: - A Pandas DataFrame containing all associated auto-mode fare - costs. The DataFrame contains all original fields of the input - DataFrame along with the field: - [costFareDrive] - auto mode fare cost in $ """ - - # calculate fare costs for non-pooled TNC, pooled TNC, and taxi modes - conditions = [df["tripMode"] == "Non-Pooled TNC", - df["tripMode"] == "Pooled TNC", - df["tripMode"] == "Taxi"] - - # calculate non-pooled TNC fare incorporating minimum fare - non_pooled_tnc_fare = pd.Series( - self.properties["baseFareNonPooledTNC"] + - self.properties["costPerMileNonPooledTNC"] * df["distanceDrive"] + - self.properties["costPerMinuteNonPooledTNC"] * df["timeDrive"], - dtype="float32") - - non_pooled_tnc_fare = np.where( - non_pooled_tnc_fare < self.properties["costMinimumNonPooledTNC"], - self.properties["costMinimumNonPooledTNC"], - non_pooled_tnc_fare) - - # calculate pooled TNC fare incorporating minimum fare - pooled_tnc_fare = pd.Series( - self.properties["baseFarePooledTNC"] + - self.properties["costPerMilePooledTNC"] * df["distanceDrive"] + - self.properties["costPerMinutePooledTNC"] * df["timeDrive"], - dtype="float32") - - pooled_tnc_fare = np.where( - pooled_tnc_fare < self.properties["costMinimumPooledTNC"], - self.properties["costMinimumPooledTNC"], - pooled_tnc_fare) - - choices = [ - non_pooled_tnc_fare, - pooled_tnc_fare, - self.properties["baseFareTaxi"] + - self.properties["costPerMileTaxi"] * df["distanceDrive"] + - self.properties["costPerMinuteTaxi"] * df["timeDrive"] - ] - - df["costFareDrive"] = pd.Series( - np.select(conditions, choices, default=np.NaN), - dtype="float32") - - # return input DataFrame with appended auto fare cost column - return df - - def tnc_wait_time(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originMGRA]) and returns the associated wait - times for Taxi and TNC modes. - - Note that drive to transit modes assume no auto-mode wait time and - are not included here. - - Note that the actual wait times experienced by trips are drawn from a - distribution but not written out to the trip output files making it - impossible to write out the actual wait time experienced. The mean is - used here for all trips as an approximation. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA geography - - Returns: - A Pandas DataFrame containing all associated auto-mode wait times. - The DataFrame contains all original fields of the input DataFrame - along with the field: - [timeWaitDrive] - auto mode wait time in minutes """ - - # load the mgra based input file - fn = "mgra13_based_input" + str(self.properties["year"]) + ".csv" - - mgra = pd.read_csv(os.path.join(self.scenario_path, "input", fn), - usecols=["mgra", # MGRA geography - "PopEmpDenPerMi"], # density per mi - dtype={"mgra": "int16", - "PopEmpDenPerMi": "float32"}) - - # add PopEmpDenPerMi field to the input DataFrame - df = df.merge(mgra, left_on="originMGRA", right_on="mgra") - - # select mean wait time for Taxi/TNC mode trips based on - # the category the PopEmpDenPerMi value falls in - # note the first true condition encountered is chosen - conditions = [ - ((df["tripMode"] == "Non-Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), - ((df["tripMode"] == "Non-Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), - ((df["tripMode"] == "Non-Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), - ((df["tripMode"] == "Non-Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), - ((df["tripMode"] == "Non-Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])), - ((df["tripMode"] == "Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), - ((df["tripMode"] == "Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), - ((df["tripMode"] == "Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), - ((df["tripMode"] == "Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), - ((df["tripMode"] == "Pooled TNC") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])), - ((df["tripMode"] == "Taxi") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][0])), - ((df["tripMode"] == "Taxi") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][1])), - ((df["tripMode"] == "Taxi") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][2])), - ((df["tripMode"] == "Taxi") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][3])), - ((df["tripMode"] == "Taxi") & ( - df["PopEmpDenPerMi"] < self.properties["waitTimePopEmpDenPerMi"][4])) - ] - - choices = [ - *self.properties["waitTimeNonPooledTNC"], - *self.properties["waitTimePooledTNC"], - *self.properties["waitTimeTaxi"], - ] - - df["timeWaitDrive"] = pd.Series( - np.select(conditions, choices, default=np.NaN), - dtype="float32") - - # remove mgra, PopEmpDenPerMi from the input DataFrame - df.drop(columns=["mgra", "PopEmpDenPerMi"], inplace=True) - - # return input DataFrame with appended auto wait time column - return df - - def walk_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originMGRA], [destinationMGRA]) and returns - the associated micro-mobility, micro-transit, and walk mode skims for - time, distance, and fare cost. Non-micro-mobility/micro-transit/walk - modes have all skims set to NaN. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA (1-23002) - [destinationMGRA] - trip destination MGRA (1-23002) - - Returns: - A Pandas DataFrame containing all associated micro-mobility, - micro-transit, and walk mode skims for time, distance, and - fare cost. The DataFrame contains all original fields of the - input DataFrame along with the field: - [timeWalk] - time in minutes for walk mode - [distanceWalk] - distance in miles for walk mode - [timeMM] - time in minutes for micro-mobility mode - [distanceMM] - distance in miles for micro-mobility mode - [costFareMM] - fare cost in dollars for micro-mobility mode - [timeMT] - time in minutes for micro-transit mode - [distanceMT] - distance in miles for micro-transit mode - [costFareMT] - fare cost in dollars for micro-transit mode - """ - # get skims for walk/micro-mobility/micro-transit trips - # some use AT skim sets while others use auto skim set - records_at = self._walk_skims_at(df) - records_auto = self._walk_skims_auto(df) - - # if there are no mm/mt/walk trip skim records - if records_at.empty and records_auto.empty: - # append skim columns to input DataFrame - # set to missing as a missing skim means no walk trip - skim_cols = ["timeWalk", - "distanceWalk", - "timeMM", - "distanceMM", - "costFareMM", - "timeMT", - "distanceMT", - "costFareMT"] - - for col in skim_cols: - df[col] = np.NaN - else: - # merge result set DataFrame back into initial trip list - # keep missing skim records as missing skim means no walk trip - records = records_at.append(records_auto, ignore_index=True) - df = df.merge(records, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df - - def _walk_skims_at(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originMGRA], [destinationMGRA]) and returns - the associated micro-mobility, micro-transit, and walk mode skims for - time, distance, and fare cost. Non-micro-mobility/micro-transit/walk - modes have all skims set to NaN. Note that some micro-mobility, - micro-transit and walk mode trips use an auto mode skim to define - time, distance, and fare cost. These are removed from consideration - in this method and handled elsewhere. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA (1-23002) - [destinationMGRA] - trip destination MGRA (1-23002) - - Returns: - A Pandas DataFrame containing all associated micro-mobility, - micro-transit, and walk mode skims for time, distance, and - fare cost. The DataFrame contains only micro-mobility, - micro-transit, and walk mode trip records that do not use the - auto mode skim set for time: - [tripID] - unique identifier of a trip - [timeWalk] - time in minutes for walk mode - [distanceWalk] - distance in miles for walk mode - [timeMM] - time in minutes for micro-mobility mode - [distanceMM] - distance in miles for micro-mobility mode - [costFareMM] - fare cost in dollars for micro-mobility mode - [timeMT] - time in minutes for micro-transit mode - [distanceMT] - distance in miles for micro-transit mode - [costFareMT] - fare cost in dollars for micro-transit mode - """ - # load the mgra-mgra walk/micro-mobility/micro-transit skim file - skims = pd.read_csv(os.path.join(self.scenario_path, - "output", - "microMgraEquivMinutes.csv"), - usecols=["i", # origin MGRA geography - "j", # destination MGRA geography - "walkTime", # walk time in minutes - "dist", # distance in miles - "mmTime", # micro-mobility time in minutes - "mmCost", # micro-mobility cost in dollars - "mtTime", # micro-transit time in minutes - "mtCost"], # micro-transit cost in dollars - dtype={"i": "int16", - "j": "int16", - "walkTime": "float32", - "dist": "float32", - "mmTime": "float32", - "mmCost": "float32", - "mtTime": "float32", - "mtCost": "float32"}) - - # merge the skims with the input DataFrame walk/mm/mt mode records - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - records = df.loc[(df["tripMode"].isin(["Micro-Mobility", - "Micro-Transit", - "Walk"]))].copy() - - records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - records = records.merge( - right=skims, - how="inner", # some of these trips can use auto skims, use inner join to remove them - left_on=["originMGRA", "destinationMGRA"], - right_on=["i", "j"] - ) - - # set skims based on mode - records["timeWalk"] = np.where(records["tripMode"] == "Walk", - records["walkTime"], - 0) - - records["distanceWalk"] = np.where(records["tripMode"] == "Walk", - records["dist"], - 0) - - records["timeMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["mmTime"], - 0) - - records["distanceMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["dist"], - 0) - - records["costFareMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["mmCost"], - 0) - - records["timeMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["mtTime"], - 0) - - records["distanceMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["dist"], - 0) - - records["costFareMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["mtCost"], - 0) - - # return result set of walk/micro-mobility/micro-transit trips - # that use AT skim sets - return records[["tripID", - "timeWalk", - "distanceWalk", - "timeMM", - "distanceMM", - "costFareMM", - "timeMT", - "distanceMT", - "costFareMT"]] - - def _walk_skims_auto(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [tripMode], [originMGRA], [originTAZ] [destinationMGRA], - [destinationTAZ]) and returns the associated micro-mobility, - micro-transit, and walk mode skims for time, distance, and fare cost - for micro-mobility, micro-transit, and walk mode trips that use - auto mode skim set for time. Note that micro-mobility, micro-transit, - and walk mode trips that do not use auto mode skim set for time are - removed from consideration and handled elsewhere. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA (1-23002) - [originTAZ] - trip origin TAZ (1-4996) - [destinationMGRA] - trip destination MGRA (1-23002) - [destinationTAZ] - trip destination TAZ (1-4996) - - Returns: - A Pandas DataFrame containing all associated micro-mobility, - micro-transit, and walk mode skims for time, distance, and - fare cost. The DataFrame contains only micro-mobility, - micro-transit, and walk mode trip records that use the auto - mode skim set for time: - [tripID] - unique identifier of a trip - [timeWalk] - time in minutes for walk mode - [distanceWalk] - distance in miles for walk mode - [timeMM] - time in minutes for micro-mobility mode - [distanceMM] - distance in miles for micro-mobility mode - [costFareMM] - fare cost in dollars for micro-mobility mode - [timeMT] - time in minutes for micro-transit mode - [distanceMT] - distance in miles for micro-transit mode - [costFareMT] - fare cost in dollars for micro-transit mode - """ - # load the mgra-mgra walk/micro-mobility/micro-transit skim file - skims = pd.read_csv(os.path.join(self.scenario_path, - "output", - "microMgraEquivMinutes.csv"), - usecols=["i", # origin MGRA geography - "j", # destination MGRA geography - "walkTime", # walk time in minutes - "dist", # distance in miles - "mmTime", # micro-mobility time in minutes - "mmCost", # micro-mobility cost in dollars - "mtTime", # micro-transit time in minutes - "mtCost"], # micro-transit cost in dollars - dtype={"i": "int16", - "j": "int16", - "walkTime": "float32", - "dist": "float32", - "mmTime": "float32", - "mmCost": "float32", - "mtTime": "float32", - "mtCost": "float32"}) - - # merge the skims with the input DataFrame walk/mm/mt mode records - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - records = df.loc[(df["tripMode"].isin(["Micro-Mobility", - "Micro-Transit", - "Walk"]))].copy() - - records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - # select records that are NOT in the mgra-mgra - # walk/micro-mobility/micro-transit skim file - records = records.merge( - right=skims, - how="left", # use outer join to keep all trip records - left_on=["originMGRA", "destinationMGRA"], - right_on=["i", "j"], - indicator=True - ) - - records = records[records["_merge"] == "left_only"] - - # if there are no eligible records return an empty DataFrame - if records.empty: - return pd.DataFrame( - columns=["tripID", - "timeWalk", - "distanceWalk", - "timeMM", - "distanceMM", - "costFareMM", - "timeMT", - "distanceMT", - "costFareMT"] - ) - else: - # append trip time from auto skim set - # midday drive alone non-transponder low value of time - - # open omx transit skim file and get TAZ matrix mapping - omx_file = omx.open_file(self.scenario_path + "/output/traffic_skims_MD.omx") - omx_map = omx_file.mapping("zone_number") - - # get lists of o-ds - o = records.originTAZ.astype("int16").tolist() - d = records.destinationTAZ.astype("int16").tolist() - - # map o-ds to omx matrix indices - o_idx = [omx_map[number] for number in o] - d_idx = [omx_map[number] for number in d] - - # append travel time skim from auto skim set - records["sovTime"] = omx_file["MD_SOV_NT_M_TIME"][o_idx, d_idx] - - omx_file.close() - - # load the MGRA-MGRA based input file - # merge with trips to get micro-mobility access time for origin MGRAs - records = records.merge( - right=self.mgra_xref, - how="inner", - left_on="originMGRA", - right_on="MGRA" - ) - - # set skims based on mode - records["timeWalk"] = np.where(records["tripMode"] == "Walk", - records["sovTime"], - 0) - - records["distanceWalk"] = np.where(records["tripMode"] == "Walk", - records["sovTime"] * self.properties["walkSpeed"] / 60, - 0) - - records["timeMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["sovTime"] + records["MicroAccessTime"], - 0) - - records["distanceMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["sovTime"] * self.properties["microMobilitySpeed"] / 60, - 0) - - records["costFareMM"] = np.where(records["tripMode"] == "Micro-Mobility", - records["sovTime"] * self.properties["costPerMinuteMicroMobility"] + - self.properties["baseFareMicroMobility"], - 0) - - records["timeMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["sovTime"] + self.properties["accessTimeMicroTransit"] + - self.properties["waitTimeMicroTransit"], - 0) - - records["distanceMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["sovTime"] * self.properties["microTransitSpeed"] / 60, - 0) - - records["costFareMT"] = np.where(records["tripMode"] == "Micro-Transit", - records["sovTime"] * self.properties["costPerMinuteMicroTransit"] + - self.properties["baseFareMicroTransit"], - 0) - - # return result set of walk/micro-mobility/micro-transit trips - # that use auto mode skim set - return records[["tripID", - "timeWalk", - "distanceWalk", - "timeMM", - "distanceMM", - "costFareMM", - "timeMT", - "distanceMT", - "costFareMT"]] - - def walk_transit_skims(self, df: pd.DataFrame) -> pd.DataFrame: - """ Takes an input Pandas DataFrame containing the ABM fields - ([tripID], [inbound], [tripMode], [originMGRA], [destinationMGRA], - [boardingTAP], [alightingTAP]) and the optional fields - ([microMobilityTransitAccess], [microMobilityTransitEgress]) and - returns the associated micro-mobility, micro-transit, and walk mode - skims for transit access/egress for time, distance, and fare cost. - Trips without micro-mobility, micro-transit, or walk mode - access/egress to transit have all skims set to 0. - - Args: - df: Input Pandas DataFrame containing the fields - [tripID] - unique identifier of a trip - [inbound] - boolean indicator of inbound/outbound direction - [tripMode] - ABM trip modes (Drive Alone, Shared Ride 2, etc...) - [originMGRA] - trip origin MGRA (1-23002) - [destinationMGRA] - trip destination MGRA (1-23002) - [boardingTAP] - transit boarding TAP - [alightingTAP] - transit alighting TAP - [microMobilityTransitAccess] - optional field indicating if - micro-mobility, micro-transit, or walk mode was used to - access transit - [microMobilityTransitEgress] - optional field indicating if - micro-mobility, micro-transit, or walk mode was used to - egress transit - - Returns: - A Pandas DataFrame containing all associated micro-mobility, - micro-transit, and walk to transit mode access/egress skims for - time, distance, and fare cost. The DataFrame contains all the - original fields of the input DataFrame along with the fields: - [timeTransitWalkAccessEgress] - time in minutes for walk - portion of transit access/egress - [distanceTransitWalkAccessEgress] - distance in miles for - walk portion of transit access/egress - [timeTransitMMAccessEgress] - time in minutes for - micro-mobility portion of transit access/egress - [distanceTransitMMAccessEgress] - distance in miles for - micro-mobility portion of transit access/egress - [costFareTransitMMAccessEgress] - fare cost in dollars for - micro-mobility portion of transit access/egress - [timeTransitMTAccessEgress] - time in minutes for - micro-transit portion of transit access/egress - [distanceTransitMTAccessEgress] - distance in miles for - micro-transit portion of transit access/egress - [costFareTransitMTAccessEgress] - fare cost in dollars for - micro-transit portion of transit access/egress """ - - # load the micro-mobility, micro-transit, and walk from MGRA-TAP - # skim file containing time, distance, and cost - skims = pd.read_csv(os.path.join(self.scenario_path, - "output", - "microMgraTapEquivMinutes.csv"), - usecols=["mgra", # origin MGRA geography - "tap", # destination TAP - "walkTime", # walk time in minutes - "dist", # distance in miles - "mmTime", # micro-mobility time in minutes - "mmCost", # micro-mobility cost in dollars - "mtTime", # micro-transit time in minutes - "mtCost"], # micro-transit cost in dollars - dtype={"mgra": "int16", - "tap": "int16", - "walkTime": "float32", - "dist": "float32", - "mmTime": "float32", - "mmCost": "float32", - "mtTime": "float32", - "mtCost": "float32"}) - - # filter input DataFrame to records that have access/egress - # micro-mobility, micro-transit, or walk segments - records = df.loc[(df["tripMode"].isin( - ["Walk to Transit - Local Bus", - "Walk to Transit - Premium Transit", - "Walk to Transit - Local Bus and Premium Transit", - "Park and Ride to Transit - Local Bus", - "Park and Ride to Transit - Premium Transit", - "Park and Ride to Transit - Local Bus and Premium Transit", - "Kiss and Ride to Transit - Local Bus", - "Kiss and Ride to Transit - Premium Transit", - "Kiss and Ride to Transit - Local Bus and Premium Transit", - "TNC to Transit - Local Bus", - "TNC to Transit - Premium Transit", - "TNC to Transit - Local Bus and Premium Transit"]))].copy() - - # ABM Joint sub-model has multiple records per tripID - # per person on trip records are identical otherwise - records.drop_duplicates(subset="tripID", inplace=True, ignore_index=True) - - # merge micro-mobility, micro-transit, and walk access/egress skims - # for both the access and egress portions - records = records.merge( - right=skims, - how="left", - left_on=["originMGRA", "boardingTAP"], - right_on=["mgra", "tap"] - ) - - records = records.merge( - right=skims, - how="left", - left_on=["destinationMGRA", "alightingTAP"], - right_on=["mgra", "tap"], - suffixes=["Access", "Egress"] - ) - - # conditionally set access/egress skim fields to 0 based on trip mode - # and inbound direction of trip, note that walk to transit uses both - records.loc[(records["tripMode"].isin( - ["Park and Ride to Transit - Local Bus", - "Park and Ride to Transit - Premium Transit", - "Park and Ride to Transit - Local Bus and Premium Transit", - "Kiss and Ride to Transit - Local Bus", - "Kiss and Ride to Transit - Premium Transit", - "Kiss and Ride to Transit - Local Bus and Premium Transit", - "TNC to Transit - Local Bus", - "TNC to Transit - Premium Transit", - "TNC to Transit - Local Bus and Premium Transit"])) & - (records["inbound"] == False), - ["walkTimeAccess", - "distAccess", - "mmTimeAccess", - "mmCostAccess", - "mtTimeAccess", - "mtCostAccess"]] = 0 - - records.loc[(records["tripMode"].isin([ - "Park and Ride to Transit - Local Bus", - "Park and Ride to Transit - Premium Transit", - "Park and Ride to Transit - Local Bus and Premium Transit", - "Kiss and Ride to Transit - Local Bus", - "Kiss and Ride to Transit - Premium Transit", - "Kiss and Ride to Transit - Local Bus and Premium Transit", - "TNC to Transit - Local Bus", - "TNC to Transit - Premium Transit", - "TNC to Transit - Local Bus and Premium Transit"])) & - (records["inbound"] == True), - ["walkTimeEgress", - "distEgress", - "mmTimeEgress", - "mmCostEgress", - "mtTimeEgress", - "mtCostEgress"]] = 0 - - # if optional micro-mobility access/egress fields are not present - # assume Walk modes only for access/egress skims - if not {"microMobilityTransitAccess", "microMobilityTransitEgress"}.issubset(df.columns): - records["timeTransitWalkAccessEgress"] = records["walkTimeAccess"] + records["walkTimeEgress"] - records["distanceTransitWalkAccessEgress"] = records["distAccess"] + records["distEgress"] - records["timeTransitMMAccessEgress"] = 0 - records["distanceTransitMMAccessEgress"] = 0 - records["costFareTransitMMAccessEgress"] = 0 - records["timeTransitMTAccessEgress"] = 0 - records["distanceTransitMTAccessEgress"] = 0 - records["costFareTransitMTAccessEgress"] = 0 - else: # if optional micro-mobility access/egress fields are present - # set skim fields based on micro-mobility access/egress fields - records["timeTransitWalkAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Walk", - records["walkTimeAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Walk", - records["walkTimeEgress"], 0) - - records["distanceTransitWalkAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Walk", - records["distAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Walk", - records["distEgress"], 0) - - records["timeTransitMMAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", - records["mmTimeAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", - records["mmTimeEgress"], 0) - - records["distanceTransitMMAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", - records["distAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", - records["distEgress"], 0) - - records["costFareTransitMMAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Mobility", - records["mmCostAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Mobility", - records["mmCostEgress"], 0) - - records["timeTransitMTAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Transit", - records["mtTimeAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Transit", - records["mtTimeEgress"], 0) - - records["distanceTransitMTAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Transit", - records["distAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Transit", - records["distEgress"], 0) - - records["costFareTransitMTAccessEgress"] = \ - np.where(records["microMobilityTransitAccess"] == "Micro-Transit", - records["mtCostAccess"], 0) + \ - np.where(records["microMobilityTransitEgress"] == "Micro-Transit", - records["mtCostEgress"], 0) - - # merge result set DataFrame back into initial trip list - records = records[["tripID", - "timeTransitWalkAccessEgress", - "distanceTransitWalkAccessEgress", - "timeTransitMMAccessEgress", - "distanceTransitMMAccessEgress", - "costFareTransitMMAccessEgress", - "timeTransitMTAccessEgress", - "distanceTransitMTAccessEgress", - "costFareTransitMTAccessEgress"]] - - # access/egress transit trip - skim_cols = ["timeTransitWalkAccessEgress", - "distanceTransitWalkAccessEgress", - "timeTransitMMAccessEgress", - "distanceTransitMMAccessEgress", - "costFareTransitMMAccessEgress", - "timeTransitMTAccessEgress", - "distanceTransitMTAccessEgress", - "costFareTransitMTAccessEgress"] - - # if there are no mm/mt/walk access/egress skim records - if records.empty: - # append skim columns to input DataFrame - # set to missing as a missing skim means no transit trip - for col in skim_cols: - df[col] = np.NaN - else: - # merge result set DataFrame back into initial trip list - # keep missing skim records as missing skim means no transit trip - df = df.merge(records, on="tripID", how="left") - - # return input DataFrame with appended skim columns - return df diff --git a/sandag_abm/src/main/python/database_summary.py b/sandag_abm/src/main/python/database_summary.py deleted file mode 100644 index 63f441f..0000000 --- a/sandag_abm/src/main/python/database_summary.py +++ /dev/null @@ -1,190 +0,0 @@ -# __author__ = 'yma' -# This file is to get data summary from the database, 1/23/2019 - -import openpyxl -from openpyxl import load_workbook -from datetime import datetime -from pandas import DataFrame -import pandas as pd -import pyodbc -import sys -import os - -# usage = ("Correct Usage: database_summary.py ") - -# check if too few/many arguments passed raise error -if len(sys.argv) != 4: - sys.exit(-1) - -output_path = str(sys.argv[1]) -sceYear = int(sys.argv[2]) -scenario = int(sys.argv[3]) - -# create settings data frame -settings = {"Parameter": ["Date", "Scenario_id","Year"], - "Value": [datetime.now().strftime("%x %X"),scenario,sceYear]} -settings = pd.DataFrame(data=settings) - - -# set sql server connection to ws -sql_con = pyodbc.connect(driver='{SQL Server}', - server='sql2014a8', - database='ws', - trusted_connection='yes') - - -### data summary for sensitivty analysis - -# 0_scenarioInfor -scenarioInfor = pd.read_sql_query( - sql="exec [ws].[sst2].[m0_scenario] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 1_modeshare -modeShare = pd.read_sql_query( - sql="exec [ws].[sst2].[m1_mode_share] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 4_PtripLengthByPurpose -ptripLengthByPurpose = pd.read_sql_query( - sql="exec [ws].[sst2].[m4_ptrip_distance_purpose] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 5_PtripLengthByMode -ptripDistanceMode = pd.read_sql_query( - sql="exec [ws].[sst2].[m5_ptrip_distance_mode] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 23_VMT -VMT = pd.read_sql_query( - sql="exec [ws].[sst2].[m23_vmt_capita] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 6_VHT -VHT = pd.read_sql_query( - sql="exec [ws].[sst2].[m6_vht_capita] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 7_VHD -VHD = pd.read_sql_query( - sql="exec [ws].[sst2].[m7_vhd_capita] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 8_TransitBoardingByMode -transitBoardingLinehaulMode = pd.read_sql_query( - sql="exec [ws].[sst2].[m8_transit_boarding_linehaulmode] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - -# 9_TransitTripsbyMode -transitTripsbyMode = pd.read_sql_query( - sql="exec [ws].[sst2].[m9_transit_trips_by_mode] @scenario_id = ? ", - con=sql_con, - params=[scenario] -) - - -# output to excel book 'source_sensitivity' -output_file_sens = output_path + "\\analysis\\summary\\source_summary.xlsx" -book = load_workbook(output_file_sens) - -with pd.ExcelWriter(output_file_sens, engine='openpyxl') as writer: - - writer.book = book - writer.sheets = dict((ws.title, ws) for ws in book.worksheets) - - settings.to_excel(writer,'Settings',index=False) - scenarioInfor.to_excel(writer,'0_scenarioInfor', index=False) - modeShare.to_excel(writer,'1_modeshare', index=False) - ptripLengthByPurpose.to_excel(writer, '4_PtripLengthByPurpose', index=False) - ptripDistanceMode.to_excel(writer, '5_PtripLengthByMode', index=False) - VMT.to_excel(writer, '23_VMT', index=False) - VHT.to_excel(writer, '6_VHT', index=False) - VHD.to_excel(writer, '7_VHD', index=False) - transitBoardingLinehaulMode.to_excel(writer, '8_TransitBoardingByMode', index=False) - transitTripsbyMode.to_excel(writer, '9_TransitTripsbyMode', index=False) - - writer.save() - - -### data summary for validation analysis, if scenario_year = 2016 or 2018 - -if sceYear == 2016 or sceYear == 2018: - - # get hwy flow given scenario_id - hwyFlow = pd.read_sql_query( - sql="select * from [ws].[validate2].[FlowDay2016_nonmsa] (?)", - con=sql_con, - params=[scenario] - ) - - # get freeway flow by TOD given scenario_id - fwyFlow = pd.read_sql_query( - sql="select * from [ws].[validate2].[FlowFreewayTod2016] (?)", - con=sql_con, - params=[scenario] - ) - - # get truck flow given scenario_id - truckFlow = pd.read_sql_query( - sql="select * from [ws].[validate2].[TruckFlow2016] (?)", - con=sql_con, - params=[scenario] - ) - - # get hwy speed given scenario_id - hwySpeed = pd.read_sql_query( - sql="select * from [ws].[validate2].[SpeedFreewayTod2016] (?)", - con=sql_con, - params=[scenario] - ) - - # get transit general summary given scenario_id - transitGeneral = pd.read_sql_query( - sql="select * from [ws].[validate2].[transit2016] (?)", - con=sql_con, - params=[scenario] - ) - - # get transit hub summary given scenario_id - transitHub = pd.read_sql_query( - sql="select * from [ws].[validate2].[transit2016_Hub] (?)", - con=sql_con, - params=[scenario] - ) - - - # output to excel book 'source' - output_file_vald = output_path + "\\analysis\\validation\\source.xlsx" - book = load_workbook(output_file_vald) - - with pd.ExcelWriter(output_file_vald, engine='openpyxl') as writer: - - writer.book = book - writer.sheets = dict((ws.title, ws) for ws in book.worksheets) - - settings.to_excel(writer,'Settings',index=False) - hwyFlow.to_excel(writer,'flow_raw', index=False) - fwyFlow.to_excel(writer,'flow_freeway', index=False) - truckFlow.to_excel(writer,'truck_flow', index=False) - hwySpeed.to_excel(writer,'hwy_speed', index=False) - transitGeneral.to_excel(writer,'transit_general', index=False) - transitHub.to_excel(writer,'transit_hub', index=False) - - writer.save() - diff --git a/sandag_abm/src/main/python/excel_update.py b/sandag_abm/src/main/python/excel_update.py deleted file mode 100644 index ad17762..0000000 --- a/sandag_abm/src/main/python/excel_update.py +++ /dev/null @@ -1,85 +0,0 @@ -# __author__ = 'yma' -# This file is to force to update links in excel, 1/23/2019 - -import os -import sys -import win32com.client -from win32com.client import Dispatch - -# usage = ("Correct Usage: excel_update.py ") - -# check if too few/many arguments passed raise error -if len(sys.argv) != 4: - sys.exit(-1) - -output_path = str(sys.argv[1]) -sceYear = int(sys.argv[2]) -scenario = str(sys.argv[3]) - - -path_name_s = output_path + "\\analysis\\summary\\" -file_names_s = ["ModelResultSummary"] - -path_name_v = output_path + "\\analysis\\validation\\" -file_names_v = ["HighwayAssignmentValidation_2016_AllClass_EMME", - "HighwayAssignmentValidation_2016_Truck_EMME", - "HighwayAssignmentValidation_2016_Speed_FreewayCorridor_AMPM_EMME", - "HighwayAssignmentValidation_2016_FreewayCorridor_Daily_EMME", - "HighwayAssignmentValidation_2016_FreewayCorridor_AM_EMME", - "HighwayAssignmentValidation_2016_FreewayCorridor_PM_EMME", - "TransitAssignmentValidation_2016_General_EMME", - #"TransitAssignmentValidation_2016_Hub", - ] -ext = ".xlsm" - -# file list of sensitivity -list_s = [] -list_s_new = [] -for file in file_names_s: - file_s = path_name_s + file + ext - #file_s_new = path_name_s + file + "_" + scenario + ext - file_s_new = path_name_s + file + ext - list_s.append(file_s) - list_s_new.append(file_s_new) - -# file list of validation -list_v = [] -list_v_new = [] -for file in file_names_v: - file_v = path_name_v + file + ext - #file_v_new = path_name_v + file + "_" + scenario + ext # since the data was read from EMME, scenario is not needed - CL 05222020 - file_v_new = path_name_v + file + ext - list_v.append(file_v) - list_v_new.append(file_v_new) - - -file_list = list_s -file_list_new = list_s_new - -if sceYear == 2016 or sceYear == 2018: - file_list = file_list + list_v - file_list_new = file_list_new + list_v_new - - -xl = Dispatch("Excel.Application") -xl.Visible = False -xl.DisplayAlerts = False -xl.AskToUpdateLinks = False - -for file in file_list: - wb = xl.workbooks.open(file) - xl.Application.Run("UpdateLinks") - wb.Close(True) - -EMME = os.path.join(path_name_v, 'source_EMME.xlsx') #with source_EMME.xlsx opened, links in validation Excel files will work well. This is a temporary solution - CL 05222020 -wb_emme = xl.Workbooks.Open(EMME) -for file in file_list: - wb = xl.workbooks.open(file) - xl.Application.Run("UpdateLinks") - wb.Close(True) -wb_emme.Close(True) - -xl.Quit() - -#for i in range(len(file_list)): # since the data was read from EMME, scenario is not needed and the file names are not renamed with scenrio ID - CL 05222020 -# os.rename(file_list[i], file_list_new[i]) diff --git a/sandag_abm/src/main/python/parameterUpdate.py b/sandag_abm/src/main/python/parameterUpdate.py deleted file mode 100644 index cfa06ca..0000000 --- a/sandag_abm/src/main/python/parameterUpdate.py +++ /dev/null @@ -1,60 +0,0 @@ -# Author:Yun.Ma@sandag.org -# Oct 5,2016 - -#import -import os -import csv -import string -import sys - -# check if property file exists -if not os.path.isfile('..\\conf\\sandag_abm.properties'): - print "Property File Not Found" - raise sys.exit() - -# search scenarioYear -TheYear='' -propFile = open('..\\conf\\sandag_abm.properties','r') -for line in propFile: - if line.find('scenarioYear') > -1: - TheYear = line.strip('\n').split('=')[1] - break -else: - print "scenarioYear Not Found" -propFile.close() - -# read csv file -ParaDict={} -csvFileIn = open('..\\input\\parametersByYears.csv','rU') -reader = csv.DictReader(csvFileIn) -for row in reader: - if row['year'] == TheYear: - ParaDict=row.copy() - break -csvFileIn.close() -ParaDict.pop('year',ParaDict['year']) - -# read and update str in property file -OldVal='' -NewVal='' -Paralines=[] - -propInFile = open('..\\conf\\sandag_abm.properties','r') -for line in propInFile: - for key in ParaDict: - if line.find(key) > -1: - NewVal = ParaDict[key] - print NewVal - OldVal = line.strip('\n').split('=')[1] - line = string.replace(line,OldVal,NewVal) - print line - break - Paralines.append(line) -propInFile.close() - -# write into property file -propOutFile = open('..\\conf\\sandag_abm.properties','w') -for line in Paralines: - propOutFile.write(line) -propOutFile.close() - diff --git a/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py b/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py deleted file mode 100644 index a6a9eb9..0000000 --- a/sandag_abm/src/main/python/pythonGUI/createStudyAndScenario.py +++ /dev/null @@ -1,237 +0,0 @@ -__author__ = 'wsu' -#Wu.Sun@sandag.org 10-27-2016 -import Tkinter -import Tkconstants -import tkFileDialog -import os -from Tkinter import * -from PIL import Image,ImageTk -import popupMsg - -class CreateScenarioGUI(Tkinter.Frame): - def __init__(self, root): - Tkinter.Frame.__init__(self, root, border=5) - body = Tkinter.Frame(self) - body.pack(fill=Tkconstants.X, expand=1) - sticky = Tkconstants.E + Tkconstants.W - body.grid_columnconfigure(1, weight=2) - - #divider line - divider=u"_"*120 - self.releaseDir='T:\\ABM\\release\\ABM' - self.defaultScenarioDir="T:\\projects\\sr14" - self.defaultNetworkDir="T:\\projects\\sr14\\version14_2_0\\network_build" - - self.buttonVar= IntVar(root) - self.yButton=Radiobutton(body, text="Yes", variable=self.buttonVar, value=1, command=self.initStudy) - self.nButton=Radiobutton(body, text="No", variable=self.buttonVar, value=0,command=self.initStudy) - Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=0,columnspan=5) - Tkinter.Label(body, text=u"Create an ABM Work Space", font=("Helvetica", 10, 'bold')).grid(row=1,columnspan=3) - self.yButton.grid(row=2,column=0, columnspan=2) - self.nButton.grid(row=2,column=1, columnspan=2) - - Tkinter.Label(body, text=u"Study Folder", font=("Helvetica", 8, 'bold')).grid(row=3) - self.studypath = Tkinter.Entry(body, width=40) - self.studypath.grid(row=3, column=1, sticky=sticky) - self.studypath.delete(0, Tkconstants.END) - self.studypath.insert(0, self.defaultScenarioDir) - self.studybutton = Tkinter.Button(body, text=u"...",width=4,command=lambda:self.get_path("study")) - self.studybutton.grid(row=3, column=2) - - Tkinter.Label(body, text=u"Network Folder",font=("Helvetica", 8, 'bold')).grid(row=4) - self.studynetworkpath = Tkinter.Entry(body, width=40) - self.studynetworkpath.grid(row=4, column=1, sticky=sticky) - self.studynetworkpath.delete(0, Tkconstants.END) - self.studynetworkpath.insert(0, self.defaultNetworkDir) - self.studynetworkbutton = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("studynetwork")) - self.studynetworkbutton.grid(row=4, column=2) - - self.copyButton = Tkinter.Button(body, text=u"Create", font=("Helvetica", 8, 'bold'),width=10, command=lambda: self.checkPath("study")) - self.copyButton.grid(row=5,column=0,columnspan=4) - - Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=6,columnspan=5) - Tkinter.Label(body, text=u"Create an ABM scenario", font=("Helvetica", 10, 'bold')).grid(row=7,columnspan=3) - - Tkinter.Label(body, text=u"Version", font=("Helvetica", 8, 'bold')).grid(row=8) - var = StringVar(root) - self.version="version_14_2_2" - optionList=["version_14_2_2"] - option=Tkinter.OptionMenu(body,var,*optionList,command=self.setversion) - option.config(width=50) - option.grid(row=8, column=1) - - Tkinter.Label(body, text=u"Emme Version", font=("Helvetica", 8, 'bold')).grid(row=9) - var = StringVar(root) - self.emme_version = "4.4.4.1" - optionList = ["4.3.7", "4.4.4.1"] - option = Tkinter.OptionMenu(body, var, *optionList, command=self.setEmmeVersion) - option.config(width=50) - option.grid(row=9, column=1) - - Tkinter.Label(body, text=u"Year", font=("Helvetica", 8, 'bold')).grid(row=10) - - var = StringVar(root) - self.year="2016" - yearOptionList = ["2016", "2020", "2023", "2025", "2025nb", "2026", "2029", "2030", "2030nb", "2032", "2035", "2035nb", "2040", "2040nb", "2050","2050nb"] - option=Tkinter.OptionMenu(body,var,*yearOptionList,command=self.setyear) - option.config(width=50) - option.grid(row=10, column=1) - - Tkinter.Label(body, text=u"Scenario Folder", font=("Helvetica", 8, 'bold')).grid(row=11) - self.scenariopath = Tkinter.Entry(body, width=40) - self.scenariopath.grid(row=11, column=1, sticky=sticky) - button = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("scenario")) - button.grid(row=11, column=2) - - Tkinter.Label(body, text=u"Network Folder",font=("Helvetica", 8, 'bold')).grid(row=12) - self.networkpath = Tkinter.Entry(body, width=40) - self.networkpath.grid(row=12, column=1, sticky=sticky) - button = Tkinter.Button(body, text=u"...",width=4,command=lambda: self.get_path("network")) - button.grid(row=12, column=2) - - buttons = Tkinter.Frame(self) - buttons.pack() - botton = Tkinter.Button(buttons, text=u"Create", font=("Helvetica", 8, 'bold'),width=10, command=lambda: self.checkPath("scenario")) - botton.pack(side=Tkconstants.LEFT) - Tkinter.Frame(buttons, width=10).pack(side=Tkconstants.LEFT) - button = Tkinter.Button(buttons, text=u"Quit", font=("Helvetica", 8, 'bold'), width=10, command=self.quit) - button.pack(side=Tkconstants.RIGHT) - - self.initStudy() - - def initStudy(self): - #disable study setting buttons - if self.buttonVar.get()==1: - self.studypath.config(state=NORMAL) - self.studybutton.config(state=NORMAL) - self.studynetworkpath.config(state=NORMAL) - self.studynetworkbutton.config(state=NORMAL) - self.copyButton.configure(state=NORMAL) - #enable study setting buttons - elif self.buttonVar.get()==0: - self.studypath.config(state=DISABLED) - self.studybutton.config(state=DISABLED) - self.studynetworkpath.config(state=DISABLED) - self.studynetworkbutton.config(state=DISABLED) - self.copyButton.configure(state=DISABLED) - - #set default input and network paths based on selected year - def setversion(self,value): - self.version=value - return - - # set Emme version - def setEmmeVersion(self, value): - self.emme_version = value - return - - #set default input and network paths based on selected year - def setyear(self,value): - self.defaultpath=self.releaseDir+"\\"+self.version+'\\input\\'+value - self.scenariopath.delete(0, Tkconstants.END) - self.scenariopath.insert(0, self.defaultScenarioDir) - self.networkpath.delete(0, Tkconstants.END) - self.networkpath.insert(0, self.defaultpath) - self.year=value - return - - #set cluster - def setcluster(self,value): - self.cluster=value - return - - #set default options for folded browsers - def setPathOptions(self): - self.dir_opt = options = {} - options['initialdir'] = self.defaultScenarioDir - options['mustexist'] = False - options['parent'] = root - options['title'] = 'This is a title' - - #get a path after the browse button is clicked on - def get_path(self,type): - self.setPathOptions() - path = tkFileDialog.askdirectory(**self.dir_opt) - if type=="scenario": - if path: - spath = os.path.normpath(path) - self.scenariopath.delete(0, Tkconstants.END) - self.scenariopath.insert(0, spath) - elif type=="network": - if path: - npath = os.path.normpath(path) - self.networkpath.delete(0, Tkconstants.END) - self.networkpath.insert(0, npath) - elif type=="study": - if path: - studypath = os.path.normpath(path) - self.studypath.delete(0, Tkconstants.END) - self.studypath.insert(0, studypath) - elif type=="studynetwork": - if path: - studynetworkpath = os.path.normpath(path) - self.studynetworkpath.delete(0, Tkconstants.END) - self.studynetworkpath.insert(0, studynetworkpath) - return - - #check if a path already exisits or is empty - def checkPath(self,type): - self.popup=Tkinter.Tk() - if type=="scenario": - if os.path.exists(self.scenariopath.get()): - if not self.networkpath.get(): - popupMsg.popupmsg(self,"Network folder is empty!",1,type) - else: - popupMsg.popupmsg(self,"Selected scenario folder already exists! Proceeding will overwrite existing files!",2,type) - else: - if not self.scenariopath.get(): - popupMsg.popupmsg(self,"Scenario folder is empty!",1,type) - elif not self.networkpath.get(): - popupMsg.popupmsg(self,"Network folder is empty!",1,type) - else: - self.executeBatch(type) - elif type=="study": - if os.path.exists(self.studypath.get()): - if not self.studynetworkpath.get(): - popupMsg.popupmsg(self,"Network folder is empty!",1,type) - else: - popupMsg.popupmsg(self,"Selected study folder already exists! Proceeding will overwrite existing files!",2,type) - else: - if not self.studypath.get(): - popupMsg.popupmsg(self,"Study folder is empty!",1,type) - elif not self.studynetworkpath.get(): - popupMsg.popupmsg(self,"Network folder is empty!",1,type) - else: - self.executeBatch(type) - return - - #execute DOS commands - def executeBatch(self, type): - self.popup.destroy() - if type=="scenario": - commandstr = u"create_scenario.cmd %s %s %s %s" % ( - self.scenariopath.get(), - self.year, - self.networkpath.get(), - self.emme_version - ) - elif type=="study": - commandstr=u"copy_networkfiles_to_study.cmd "+self.studypath.get()+" "+self.studynetworkpath.get() - print (commandstr) - os.chdir(self.releaseDir+"\\"+self.version+'\\') - os.system(commandstr) - self.popup=Tkinter.Tk() - msg="You have successfully created the "+ type+"!" - popupMsg.popupmsg(self,msg,1,type) - return - -root = Tkinter.Tk() -root.resizable(True, False) -root.minsize(370, 0) -logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") -w=Label(root, image=logo, width=200) -w.pack(side='top', fill='both', expand='yes') -CreateScenarioGUI(root).pack(fill=Tkconstants.X, expand=1) -root.mainloop() - - diff --git a/sandag_abm/src/main/python/pythonGUI/parameterEditor.py b/sandag_abm/src/main/python/pythonGUI/parameterEditor.py deleted file mode 100644 index bdc12cb..0000000 --- a/sandag_abm/src/main/python/pythonGUI/parameterEditor.py +++ /dev/null @@ -1,253 +0,0 @@ -__author__ = 'wsu' -#Wu.Sun@sandag.org 7-20-2016 - -import Tkinter -import Tkconstants -import tkFileDialog -import os -from Tkinter import * -from PIL import Image,ImageTk -import stringFinder -import popupMsg - - -class ParametersGUI(Tkinter.Frame): - def __init__(self, root): - Tkinter.Frame.__init__(self, root, border=5) - self.status = Tkinter.Label(self, text=u"ABM Parameter Editor", font=("Helvetica", 12, 'bold')) - self.status.pack(fill=Tkconstants.X, expand=1) - body = Tkinter.Frame(self) - body.pack(fill=Tkconstants.X, expand=1) - sticky = Tkconstants.E + Tkconstants.W - body.grid_columnconfigure(1, weight=2) - - #section labels - sectionLabels=(u"Model Initial Start Options",u"Network Building Options",u"Final Assignment Options:",u"Data Loading Options:") - #radio button lables - rbLabels=(u"Copy warm start trip tables:",u"Copy bike AT access files:",u"Create bike AT access files:",u"Build highway network:",u"Build transit network:",u"Run highway assignment:", - u"Run highway skimming:",u"Run transit assignment:",u"Run transit skimming:",u"Export results to CSVs:",u"Load results to database:") - #properties - self.properties=("RunModel.skipCopyWarmupTripTables","RunModel.skipCopyBikeLogsum","RunModel.skipBikeLogsums","RunModel.skipBuildHwyNetwork","RunModel.skipBuildTransitNetwork","RunModel.skipFinalHighwayAssignment", - "RunModel.skipFinalHighwaySkimming","RunModel.skipFinalTransitAssignment","RunModel.skipFinalTransitSkimming", - "RunModel.skipDataExport","RunModel.skipDataLoadRequest") - - #divider line - divider=u"_"*120 - - #number of properties in GUI - self.pNum=self.properties.__len__() - - #initialize yes and no buttons - self.yButton = [0 for x in range(self.pNum)] - self.nButton = [0 for x in range(self.pNum)] - self.buttonVar= [0 for x in range(self.pNum)] - for i in range(self.pNum): - self.buttonVar[i] = IntVar(root) - self.yButton[i]=Radiobutton(body, text="Yes", variable=self.buttonVar[i], value=1) - self.nButton[i]=Radiobutton(body, text="No", variable=self.buttonVar[i], value=0) - - #set standard property values - self.setDefaultProperties() - - #set AT states-activate and deactivate by selections - #self.setATButtons() - - #scenario folder browser - Tkinter.Label(body, text=u"Scenario Folder", font=("Helvetica", 8, 'bold'),width=15).grid(row=0) - self.scenariopath = Tkinter.Entry(body, width=25) - self.scenariopath.grid(row=0, column=1, sticky=sticky, columnspan=3) - self.scenariopath.insert(0,sys.argv[1]) - button = Tkinter.Button(body, text=u"...",width=4,command=self.get_scenariopath) - button.grid(row=0, column=4) - - #initial start section - for i in range(1,25): - if i==1: #intial start section header - Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) - Tkinter.Label(body, text=sectionLabels[0], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) - elif i>2 and i<6: - Tkinter.Label(body, text=rbLabels[i-3], font=("Helvetica", 8, 'bold')).grid(row=i) - self.yButton[i-3].grid(row=i,column=1) - self.nButton[i-3].grid(row=i,column=3) - elif i==6: #network building section header - Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) - Tkinter.Label(body, text=sectionLabels[1], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) - elif i>7 and i<10: - Tkinter.Label(body, text=rbLabels[i-5], font=("Helvetica", 8, 'bold')).grid(row=i) - self.yButton[i-5].grid(row=i,column=1) - self.nButton[i-5].grid(row=i,column=3) - elif i==10: #final assignment section header - Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) - Tkinter.Label(body, text=sectionLabels[2], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) - elif i>11 and i<16: - Tkinter.Label(body, text=rbLabels[i-7], font=("Helvetica", 8, 'bold')).grid(row=i) - self.yButton[i-7].grid(row=i,column=1) - self.nButton[i-7].grid(row=i,column=3) - elif i==16: #data load section header - Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) - Tkinter.Label(body, text=sectionLabels[3], font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) - elif i>17 and i<20: - Tkinter.Label(body, text=rbLabels[i-9], font=("Helvetica", 8, 'bold')).grid(row=i) - self.yButton[i-9].grid(row=i,column=1) - self.nButton[i-9].grid(row=i,column=3) - elif i==20: #iteration section - Tkinter.Label(body, text=divider, font=("Helvetica", 10, 'bold'), width=40, fg='royal blue').grid(row=i,columnspan=5) - Tkinter.Label(body, text=u"Iteration Options", font=("Helvetica", 10, 'bold'), width=30, fg='royal blue').grid(row=i+1,columnspan=5) - Tkinter.Label(body, text=u"Start from iteration:", font=("Helvetica", 8, 'bold')).grid(row=i+2) - self.var = IntVar(root) - self.button1=Radiobutton(body, text="1", variable=self.var, value=1) - self.button1.grid(row=i+2,column=1) - self.button1.select() - self.button2=Radiobutton(body, text="2", variable=self.var, value=2).grid(row=i+2,column=2) - self.button3=Radiobutton(body, text="3", variable=self.var, value=3).grid(row=i+2,column=3) - self.button4=Radiobutton(body, text="Skip", variable=self.var, value=4).grid(row=i+2,column=4) - elif i==23: - Tkinter.Label(body, text=u"Sample rates:", font=("Helvetica", 8, 'bold')).grid(row=i) - sv = StringVar(root) - sv.set("0.2,0.5,1.0") - self.samplerates="0.2,0.5,1.0" - sv.trace("w", lambda name, index, mode, sv=sv: self.setsamplerates(sv)) - e = Entry(body, textvariable=sv) - e.config(width=15) - e.grid(row=i,column=1,columnspan=3) - elif i==24:#action buttons - Tkinter.Label(body, text=u"", width=30).grid(row=i,columnspan=2) - buttons = Tkinter.Frame(self) - buttons.pack() - botton = Tkinter.Button(buttons, text=u"Update", font=("Helvetica", 9, 'bold'),width=10, command=self.update_parameters) - botton.pack(side=Tkconstants.LEFT) - Tkinter.Frame(buttons, width=10).pack(side=Tkconstants.LEFT) - button = Tkinter.Button(buttons, text=u"Quit", font=("Helvetica", 9, 'bold'), width=10, command=self.quit) - button.pack(side=Tkconstants.RIGHT) - - def setsamplerates(self,value): - self.samplerates=value.get() - - def setDefaultProperties(self): - self.runtimeFile=sys.argv[1]+"\\conf\\sandag_abm.properties" - self.standardFile=sys.argv[1]+"\\conf\\sandag_abm_standard.properties" - self.populateProperties() - """ - for i in range(self.pNum): - if i<3 or i>6: - self.yButton[i].select() - self.nButton[i].deselect() - else: - self.yButton[i].deselect() - self.nButton[i].select() - """ - - def setATButtons(self): - #disable create bike and walk logsums if 'copy' is chosen - self.yButton[1].config(command=lambda: self.yButton[3].config(state=DISABLED)) - #self.yButton[2].config(command=lambda: self.yButton[4].config(state=DISABLED)) - #disable copy bike and walk logsums if 'create' is chosen - self.yButton[3].config(command=lambda: self.yButton[1].config(state=DISABLED)) - #self.yButton[4].config(command=lambda: self.yButton[2].config(state=DISABLED)) - #enable create bike and walk logsums if NOT 'copy' is chosen - self.nButton[1].config(command=lambda: self.yButton[3].config(state=ACTIVE)) - #self.nButton[2].config(command=lambda: self.yButton[4].config(state=ACTIVE)) - #enable copy bike and walk logsums if NOT 'create' is chosen - self.nButton[3].config(command=lambda: self.yButton[1].config(state=ACTIVE)) - #self.nButton[4].config(command=lambda: self.yButton[2].config(state=ACTIVE)) - - #set scenario path - def get_scenariopath(self): - # defining options for opening a directory; initialize default path from command line - self.dir_opt = options = {} - options['initialdir'] = sys.argv[1] - options['mustexist'] = False - options['parent'] = root - options['title'] = 'This is a title' - scenariopath = tkFileDialog.askdirectory(**self.dir_opt) - if scenariopath: - scenariopath = os.path.normpath(scenariopath) - self.scenariopath.delete(0, Tkconstants.END) - self.scenariopath.insert(0, scenariopath) - else: - self.scenariopath.delete(0, Tkconstants.END) - self.scenariopath.insert(0, sys.argv[1]) - - #property file settings - self.runtimeFile=self.scenariopath.get()+"\\conf\\sandag_abm.properties" - self.standardFile=self.scenariopath.get()+"\\conf\\sandag_abm_standard.properties" - - #populate properties - self.populateProperties() - - return - - #populate properties with exisiting settings in scenario folder - def populateProperties(self): - if self.checkFile(): - for i in range(self.pNum): - if stringFinder.find(self.runtimeFile, self.properties[i]+" = true"): - self.yButton[i].deselect() - self.nButton[i].select() - elif stringFinder.find(self.runtimeFile, self.properties[i]+" = false"): - self.yButton[i].select() - self.nButton[i].deselect() - else: - print "Invalid property "+self.properties[i]+" value!, Property either has to be set to true or false." - return - - # update parameters with user inputs - def update_parameters(self): - #property file settings - self.runtimeFile=self.scenariopath.get()+"\\conf\\sandag_abm.properties" - self.standardFile=self.scenariopath.get()+"\\conf\\sandag_abm_standard.properties" - - self.deleteProperty() - self.old_text = [0 for x in range(self.pNum)] - self.new_text = [0 for x in range(self.pNum)] - for i in range(self.pNum): - if self.buttonVar[i].get()==1: - self.old_text[i]=self.properties[i]+" = true" - self.new_text[i]=self.properties[i]+" = false" - elif self.buttonVar[i].get()==0: - self.old_text[i]=self.properties[i]+" = false" - self.new_text[i]=self.properties[i]+" = true" - - #create a property update dictionary - dic=[] - for i in range(self.pNum): - pair=(self.old_text[i],self.new_text[i]) - dic.append(pair) - print dic[i][0],dic[i][1] - #add iteration update to dictionary - dic.append(("RunModel.startFromIteration = 1","RunModel.startFromIteration = "+str(self.var.get()))) - #add sample rates update to dictionary - dic.append(("sample_rates=0.2,0.5,1.0","sample_rates="+self.samplerates)) - stringFinder.replace(self.standardFile,self.runtimeFile,dic) - self.quit() - - #check if property file exists - def checkFile(self): - result=True - if not os.path.exists(self.runtimeFile): - self.popup=Tkinter.Tk() - popupMsg.popupmsg(self,self.runtimeFile+" doesn't exist!",1) - result=False - return result - - #close popup window and run batch - def letsgo(self): - self.popup.destroy() - return - - #run batch - def deleteProperty(self): - commandstr=u"del "+self.scenariopath.get()+"\\conf\\sandag_abm.properties" - print commandstr - os.system(commandstr) - return - -root = Tkinter.Tk() -root.resizable(True, False) -root.minsize(370, 0) -logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") -w=Label(root, image=logo, width=200) -w.pack(side='top', fill='both', expand='yes') -ParametersGUI(root).pack(fill=Tkconstants.X, expand=1, anchor=W) - -root.mainloop() diff --git a/sandag_abm/src/main/python/pythonGUI/popupMsg.py b/sandag_abm/src/main/python/pythonGUI/popupMsg.py deleted file mode 100644 index e9ce061..0000000 --- a/sandag_abm/src/main/python/pythonGUI/popupMsg.py +++ /dev/null @@ -1,17 +0,0 @@ -__author__ = 'wsu' -import Tkinter -import Tkconstants -#popup window for path validity checking -def popupmsg(self,msg,numButtons,type): - self.popup.wm_title("!!!WARNING!!!") - label = Tkinter.Label(self.popup, text=msg) - label.pack(side="top", fill="x", pady=10) - popbuttons = Tkinter.Frame(self.popup) - popbuttons.pack() - #can't pass arguments to a callback, otherwise callback is called before widget is constructed; use lambda function instead - B1 = Tkinter.Button(popbuttons, text="Proceed", command =lambda: self.executeBatch(type)) - B2 = Tkinter.Button(popbuttons, text="Quit", command = self.popup.destroy) - if numButtons>1: - B1.pack(side=Tkconstants.LEFT) - B2.pack(side=Tkconstants.RIGHT) - Tkinter.Frame(popbuttons, width=10).pack(side=Tkconstants.LEFT) diff --git a/sandag_abm/src/main/python/pythonGUI/setup.py b/sandag_abm/src/main/python/pythonGUI/setup.py deleted file mode 100644 index 08db6ca..0000000 --- a/sandag_abm/src/main/python/pythonGUI/setup.py +++ /dev/null @@ -1,9 +0,0 @@ -__author__ = 'wsu' -from distutils.core import setup -import py2exe - -setup(windows=['./src/main/python/pythonGUI/parameterEditor.py']) -setup(windows=['./src/main/python/pythonGUI/createStudyAndScenario.py']) -setup(windows=['./src/main/python/pythonGUI/validatorGUI.py']) - - diff --git a/sandag_abm/src/main/python/pythonGUI/stringFinder.py b/sandag_abm/src/main/python/pythonGUI/stringFinder.py deleted file mode 100644 index 3679eef..0000000 --- a/sandag_abm/src/main/python/pythonGUI/stringFinder.py +++ /dev/null @@ -1,21 +0,0 @@ -__author__ = 'wsu' -def check(fname, txt): - with open(fname) as dataf: - return any(txt in line for line in dataf) - -def replace(old_fname, new_fname, replaceDic): - f1 = open(old_fname, 'r') - f2 = open(new_fname, 'w') - for line in f1: - for pair in replaceDic: - line=line.replace(pair[0], pair[1]) - f2.write(line) - f1.close() - f2.close() - return - -def find(fname, txt): - if check(fname, txt): - return True - else: - return False \ No newline at end of file diff --git a/sandag_abm/src/main/python/pythonGUI/validatorGUI.py b/sandag_abm/src/main/python/pythonGUI/validatorGUI.py deleted file mode 100644 index b3770ce..0000000 --- a/sandag_abm/src/main/python/pythonGUI/validatorGUI.py +++ /dev/null @@ -1,122 +0,0 @@ -__author__ = 'wsu' -#Wu.Sun@sandag.org 2-10-2017 -#wsu updated 5/2/2017 for release 13.3.2 -import Tkinter -import Tkconstants -import tkFileDialog -import os -from Tkinter import * -from PIL import Image,ImageTk -import popupMsg -import stringFinder - -class CreateScenarioGUI(Tkinter.Frame): - def __init__(self, root): - Tkinter.Frame.__init__(self, root, border=5) - body = Tkinter.Frame(self) - body.pack(fill=Tkconstants.X, expand=1) - sticky = Tkconstants.E + Tkconstants.W - body.grid_columnconfigure(1, weight=2) - - self.version="version_13_3_2" - - #divider line - divider=u"_"*120 - self.releaseDir='T:\\ABM\\release\\ABM' - self.defaultScenarioDir="T:\\projects\\sr13" - - #validation section - Tkinter.Label(body, text=divider, font=("Helvetica", 11, 'bold'), width=50, fg='royal blue').grid(row=13,columnspan=5) - Tkinter.Label(body, text=u"Validate a Base Year Scenario", font=("Helvetica", 10, 'bold')).grid(row=14,columnspan=3) - Tkinter.Label(body, text=u"Base Year", font=("Helvetica", 8, 'bold')).grid(row=15) - var = StringVar(root) - self.year="2012" - optionList=["2012", "2014"] - option=Tkinter.OptionMenu(body,var,*optionList,command=self.setyear) - option.config(width=50) - option.grid(row=15, column=1) - - Tkinter.Label(body, text=u"Scenario Number", font=("Helvetica", 8, 'bold')).grid(row=16) - vv = StringVar(root) - vv.set("540") - self.validationScenario="540" - vv.trace("w", lambda name, index, mode, vv=vv: self.setValidationScenario(vv)) - self.ve = Entry(body, textvariable=vv) - self.ve.config(width=15) - self.ve.grid(row=16,column=1,sticky=sticky) - - Tkinter.Label(body, text=u"Output Folder", font=("Helvetica", 8, 'bold')).grid(row=17) - self.validationpath = Tkinter.Entry(body, width=40) - self.validationpath.grid(row=17, column=1, sticky=sticky) - self.voutbutton = Tkinter.Button(body, text=u"...",width=4,command=lambda:self.get_path("validate")) - self.voutbutton.grid(row=17, column=2) - - self.validateButton = Tkinter.Button(body, text=u"Validate", font=("Helvetica", 8, 'bold'),width=10, command=lambda:self.checkPath("validate")) - self.validateButton.grid(row=18,column=0,columnspan=4) - - #set default input and network paths based on selected year - def setyear(self,value): - dic=[] - pair=("xxxx",value) - dic.append(pair) - dir=self.releaseDir+"\\"+self.version+'\\validation\\' - stringFinder.replace(dir+"sandag_validate_generic.properties",dir+"sandag_validate.properties",dic) - return - - #set default options for folded browsers - def setPathOptions(self): - self.dir_opt = options = {} - options['initialdir'] = self.defaultScenarioDir - options['mustexist'] = False - options['parent'] = root - options['title'] = 'This is a title' - - #get a path after the browse button is clicked on - def get_path(self,type): - self.setPathOptions() - path = tkFileDialog.askdirectory(**self.dir_opt) - if path: - vpath = os.path.normpath(path) - self.validationpath.delete(0, Tkconstants.END) - self.validationpath.insert(0, vpath) - return - - #check if a path already exisits or is empty - def checkPath(self,type): - self.popup=Tkinter.Tk() - if not self.validationScenario: - popupMsg.popupmsg(self,"No validation scenario was selected!",1,type) - else: - if not self.validationpath.get(): - popupMsg.popupmsg(self,"Validation output directory is empty!",1,type) - else: - self.executeBatch(type) - return - - #execute DOS commands - def executeBatch(self, type): - self.popup.destroy() - msg="You have successfully created the "+ type+"!" - commandstr=u"validate.cmd "+self.validationScenario+" "+self.validationpath.get()+"\\" - msg="Validation results are in "+self.validationpath.get() - dir=self.releaseDir+"\\"+self.version+'\\validation\\' - print commandstr - os.chdir(dir) - os.system(commandstr) - self.popup=Tkinter.Tk() - popupMsg.popupmsg(self,msg,1,type) - return - - def setValidationScenario(self,value): - self.validationScenario=value.get() - -root = Tkinter.Tk() -root.resizable(True, False) -root.minsize(370, 0) -logo = Tkinter.PhotoImage(file=r"T:\ABM\release\ABM\SANDAG_logo.gif") -w=Label(root, image=logo, width=200) -w.pack(side='top', fill='both', expand='yes') -CreateScenarioGUI(root).pack(fill=Tkconstants.X, expand=1) -root.mainloop() - - diff --git a/sandag_abm/src/main/python/remote_run_traffic.py b/sandag_abm/src/main/python/remote_run_traffic.py deleted file mode 100644 index 57634e0..0000000 --- a/sandag_abm/src/main/python/remote_run_traffic.py +++ /dev/null @@ -1,123 +0,0 @@ -#////////////////////////////////////////////////////////////////////////////// -#//// /// -#//// Copyright INRO, 2016-2017. /// -#//// Rights to use and modify are granted to the /// -#//// San Diego Association of Governments and partner agencies. /// -#//// This copyright notice must be preserved. /// -#//// /// -#//// remote_run_traffic.py /// -#//// /// -#//// Runs the traffic assignment(s) for the specified periods. /// -#//// For running assignments on a remote server using PsExec, /// -#//// via batch file which configures for Emme python, starts /// -#//// or restarts the ISM and and maps T drive. /// -#//// /// -#//// The input arguments for the traffic assignment is read from /// -#//// start_*.args file in the database directory (database_dir). /// -#//// The "*" is one of the five time period abbreviations. /// -#//// /// -#//// Usage: remote_run_traffic.py database_dir /// -#//// /// -#//// database_dir: The path to the directory with the period /// -#//// specific traffic assignment data (scenarios and /// -#//// matrices). /// -#//// /// -#//// /// -#//// /// -#////////////////////////////////////////////////////////////////////////////// - - -import inro.emme.desktop.app as _app -import inro.modeller as _m -import inro.emme.database.emmebank as _emmebank -import json as _json -import traceback as _traceback -import glob as _glob -import time as _time -import sys -import os - -_join = os.path.join -_dir = os.path.dirname - - -class LogFile(object): - def __init__(self, log_path): - self._log_path = log_path - def write(self, text): - with open(self._log_path, 'a') as f: - f.write(text) - def write_timestamp(self, text): - text = "%s - %s\n" % (_time.strftime("%Y-%m-%d %H:%M:%S"), text) - self.write(text) - def write_dict(self, value): - with open(self._log_path, 'a') as f: - _json.dump(value, f, indent=4) - f.write("\n") - - -def run_assignment(modeller, database_path, period, msa_iteration, - relative_gap, max_assign_iterations, num_processors, - period_scenario, select_link, logger): - logger.write_timestamp("start for period %s" % period) - traffic_assign = modeller.tool("sandag.assignment.traffic_assignment") - export_traffic_skims = modeller.tool("sandag.export.export_traffic_skims") - with _emmebank.Emmebank(_join(database_path, 'emmebank')) as eb: - period_scenario = eb.scenario(period_scenario) - logger.write_timestamp("start traffic assignment") - traffic_assign( - period, msa_iteration, relative_gap, max_assign_iterations, - num_processors, period_scenario, select_link) - logger.write_timestamp("traffic assignment finished, start export to OMX") - output_dir = _join(_dir(_dir(database_path)), "output") - omx_file = _join(output_dir, "traffic_skims_%s.omx" % period) - logger.write_timestamp("start export to OMX %s" % omx_file) - if msa_iteration < 4: - export_traffic_skims(period, omx_file, period_scenario) - logger.write_timestamp("export to OMX finished") - logger.write_timestamp("period %s completed successfully" % period) - - -if __name__ == "__main__": - python_file, database_dir = sys.argv - file_ref = os.path.split(database_dir)[1].lower() - log_path = _join(_dir(_dir(database_dir)), "logFiles", "traffic_assign_%s.log" % file_ref) - logger = LogFile(log_path) - try: - logger.write_timestamp("remote process started") - # Test out licence by using the API - eb = _emmebank.Emmebank(_join(database_dir, 'emmebank')) - eb.close() - logger.write_timestamp("starting Emme Desktop application") - project_path = _join(_dir(database_dir), "emme_project.emp") - desktop = _app.start_dedicated(True, "abc", project_path) - try: - logger.write_timestamp("Emme Desktop open") - proc_logbook = _join("%<$ProjectPath>%", "Logbook", "project_%s_temp.mlbk" % file_ref) - desktop.project.par("ModellerLogbook").set(proc_logbook) - modeller = _m.Modeller(desktop) - - from_file = _join(database_dir, "start_*.args") - all_files = _glob.glob(from_file) - for path in all_files: - input_args_file = _join(database_dir, path) # communication file - logger.write_timestamp("input args read from %s" % input_args_file) - with open(input_args_file, 'r') as f: - assign_args = _json.load(f) - logger.write_dict(assign_args) - assign_args["logger"] = logger - assign_args["modeller"] = modeller - run_assignment(**assign_args) - finally: - desktop.close() - except Exception as error: - with open(_join(database_dir, "finish"), 'w') as f: - f.write("FATAL ERROR\n") - logger.write_timestamp("FATAL error execution stopped:") - logger.write(unicode(error) + "\n") - logger.write(_traceback.format_exc(error)) - finally: - _time.sleep(1) - with open(_join(database_dir, "finish"), 'a') as f: - f.write("finish\n") - sys.exit(0) diff --git a/sandag_abm/src/main/python/sdcvm.py b/sandag_abm/src/main/python/sdcvm.py deleted file mode 100644 index aee0e26..0000000 --- a/sandag_abm/src/main/python/sdcvm.py +++ /dev/null @@ -1,840 +0,0 @@ -''' -Created on 2010-04-20 - -@author: Kevin -''' - - -# Python libraries -import copy -import csv -import math -import random -import time -from array import array -from optparse import OptionParser - - -# CSTDM libraries -import sdcvm_settings as settings - - -class excelOne(csv.excel): - # define CSV dialect for Excel to avoid blank lines from default \r\n - lineterminator = "\n" - - - - -def logitNestToLogsums(nestDict): - - #print "Evaluating logit tree structure" - # Logit tree evaluation bit - # - # What this does is start with a dictionary where each key corresponds to a node on a - # nested logit tree, including the special code "top" which is the top of the tree - # and the values for utility for each of the alternatives at the bottom of the tree. - # When the dictionary starts out, the nest nodes should have a list of connections; - # each connection is itself a list of [lower node, nest coefficient]. - # - # For instance, a classic mode choice nest would look like: - # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], ['walk', 1.0] ] - # 'auto': [ ['sov', 1.0], ['hov', 0.75] ] - # 'transit': [ ['bus', 0.2], ['lrt', 0.2] ] - # 'hov': [ ['hov 2', 0.6], ['hov 3', 0.6] ] - # 'walk': -4.2911 - # 'sov': 1.0109 - # 'hov 2': -1.201 - # 'hov 3': -1.415 - # 'bus': -3.504 - # 'lrt': -2.879 } - # - # The loop then iterates through the keys of the dictionary (nodes). - # - If the key contains only a float (e.g. 'walk' above), this is assumed to be the - # utility or logsum, and nothing more needs to be done. - # - If the key contains any list elements (connections), the nodes they refer to will be checked; - # if the node it refers to contains only a float (e.g. the ['bus', 0.2] connection - # for the transit nest above, where 'bus' contains the float -3.504), then the nested utility - # will be multiplied by the coefficient and replace the list element. (e.g. -3.504 * 0.2 = -.7008) - # if the node it refers to contains a list (e.g. the ['hov', 0.75] connection above), nothing is done. - # - If the key contains all values as floats (not the case right now with the example above, but after - # the first run through, the 'hov' and 'transit' nodes will be in this state), then the logsum is taken - # of all of the floats, and this replaces the node value. - # - # After the first iteration, the mode choice nest would have: - # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], -4.2911 ] - # 'auto': [ 1.0109, ['hov', 0.75] ] - # 'transit': [ -0.7008, -0.5758 ] - # 'hov': [ -0.7206, -0.849 ] - # 'walk': -4.2911 - # 'sov': 1.0109 - # 'hov 2': -1.201 - # 'hov 3': -1.415 - # 'bus': -3.504 - # 'lrt': -2.879 } - # - # After the second iteration, the mode choice nest would have: - # nestDict = {'top': [ ['auto', 0.8], ['transit', 0.5], -4.2911 ] - # 'auto': [ 1.0109, ['hov', 0.75] ] - # 'transit': 0.0568 - # 'hov': -0.0896 - # 'walk': -4.2911 - # 'sov': 1.0109 - # 'hov 2': -1.201 - # 'hov 3': -1.415 - # 'bus': -3.504 - # 'lrt': -2.879 } - # - # - # In essence, what happens is that the utilities are passed up; when all of the utilities for a node - # have been calculated, then the logsum is taken at that node. The nest coefficients are always multiplied - # so a value of 1 needs to be specified if no other value is to be used. - # - # Note that this, in the way it works, destroys the connection information, replacing it by logsums - # at the node location instead. So a copy (not a reference, which nestDict = keepDict would do) is needed - # if the connection information needs to be used again. - - x = 0 - nodeList = nestDict.keys() - - # Do this until the top node has been resolved into a float - while isinstance(nestDict["top"], list) is True: - #print 40 * "." - for node in nodeList: - #print node, nestDict[node] - if isinstance(nestDict[node], float): # if it's already only a float, it's a utility or logsum; do nothing - pass - else: - countFloat = 0 - countSub = 0 - - for sn in range(len(nestDict[node])): - if isinstance(nestDict[node][sn], float): - countFloat = countFloat + 1 - elif isinstance(nestDict[node][sn], list): - subName = nestDict[node][sn][0] - if isinstance(nestDict[subName], float): # if the value this refers to is a float - #print "... ", node, subName, nestDict[subName], nestDict[node][sn][1] - nestDict[node][sn] = nestDict[subName] * nestDict[node][sn][1] - - countSub = countSub + 1 - - #print node, countFloat, countSub - if countFloat == len(nestDict[node]): # all the values are floats; take their logsum - expsum = 0 - for element in nestDict[node]: - expsum = expsum + math.exp(element) - if expsum > 0: - nestDict[node] = math.log(expsum) - else: - nestDict[node] = -99999.9 - - if x > 250: # Prevent nesting dictionary errors - print 250 * "!" - print nestDict - raise RuntimeError("Can't resolve nesting structure!") - x = x + 1 - return nestDict - - - -#def zonalProperties(tazList=None, fileName = cvmZonalProperties): -def zonalProperties(fileName, tazList=None): - #=========================================================================== - # Returns a dictionary of zonal properties, which are anything that applies - # to a zone under all circumstances - # - # The dictionary uses the property names as keys (e.g. "PSE" or "Ag Employment") - # and each key leads to a list of properties, in the same order as the tazList - # - # Reads the zonal property file by default, but can also read in another file - # in the same format if specified. - # - # If the tazList argument is blank, reads in all TAZ and returns both the - # props dictionary and the tazList containing all zones. - # - #=========================================================================== - - print " Reading zonal property file", fileName - fin = open(fileName, "rU") - inFile = csv.reader(fin) - header = inFile.next() - - - tempTazDict = {} - tempTazList = [] - for row in inFile: - try: - taz = int(row[header.index("TAZ")]) - except ValueError: - print 120 * "#" - print "Couldn't process a zone in zonal properties file." - raise - - tempTazList.append(taz) - list = [] - for thing in row: - try: - list.append(float(thing)) - except ValueError: - list.append(thing) - tempTazDict[taz] = list - - if tazList is None: - tazList = tempTazList - returnBoth = True - else: - returnBoth = False - - propsDict = {} - for thing in header: - propsDict[thing] = [] - for zone in tazList: - for n in range(len(header)): - propsDict[header[n]].append(tempTazDict[zone][n]) - - if returnBoth == True: - return tazList, propsDict - else: - return propsDict - - - -def hdf5Skim(fromList, toList, fromZoneDict, toZoneDict, table, skimDict, skimList): - # Reads a skim matrix; the from are the rows and the to are the columns - # so skim references are skim[from][to]. - # In the application of the CVM, the from and to are the same, so the matrix is - # symmetric (in that the 15th cell in the 4th row and 4th cell in 15th row refer - # to the same two zones; one-way roads and congestion means that they won't have - # symmetrical costs) - - - # Create blank skims to hold all of the data - - for s in skimList: - skimDict[s] = [] - for c in range(len(fromList)): - for s in skimList: - row = len(toList) * [99999.9] - row = array('f', row) - skimDict[s].append(row) - - # Read in the table row by row - x = 0 - for row in table.iterrows(): - if fromZoneDict.has_key(row['origin']): - iTaz = fromZoneDict[row['origin']] - if toZoneDict.has_key(row['destination']): - jTaz = toZoneDict[row['destination']] - for s in skimList: - try: - skimDict[s][iTaz][jTaz] = row[s] - except: - print "Skim reading error!", iTaz, jTaz - - x = x + 1 - if x % 500000 == 0: - pass - print " read", x/1000000.0, "million rows." - - return skimDict - - -def csvSkim(fromList, toList, fromZoneDict, toZoneDict, skimFile, skimDict, skimName): - # Reads a skim matrix; the from are the rows and the to are the columns - # so skim references are skim[from][to]. - # In the application of the CVM, the from and to are the same, so the matrix is - # symmetric (in that the 15th cell in the 4th row and 4th cell in 15th row refer - # to the same two zones; one-way roads and congestion means that they won't have - # symmetrical costs) - # Uses TransCad GISDK output which is in "row format", i.e. each row is all destinations for a given origin. - - - # Create blank skims to hold all of the data - print skimFile - skimDict[skimName] = [] - for c in range(len(fromList)): - row = len(toList) * [99999.9] - row = array('f', row) - skimDict[skimName].append(row) - - - - # Open skim file and read header - fin = open(skimFile, "r") - inFile = csv.reader(fin) -# header = inFile.next() -# -# for i in range(len(skimList)): #Replace column names in skim list with index of same-named header -# try: -# skimList[i] = header.index(skimList[i]) -# except ValueError: -# print "Skim name", skimList[i], "nots found in skim file", skimFile -# print "Header:", header -# raise ValueError - - # Read in the table row by row - x = 0 - err = 0 - for row in inFile: - - if fromZoneDict.has_key(int(row[0])): #Orig - iTaz = fromZoneDict[int(row[0])] -# if skimName == "Time_Mid": - - while row.count("") > 0: - row[row.index("")] = "0" - err = err + 1 - floatrow = map(float, row[1:]) - skimDict[skimName][iTaz] = array("f", floatrow) - -# else: -# for jTaz in toList: -# if skimName == "Light_Mid": -# skimDict[skimName][iTaz][jTaz-1] = float(row[jTaz]) -# else: -# skimDict[skimName][iTaz][jTaz-1] = float(row[jTaz]) * -0.3 - - x = x + 1 - if x % 500 == 0: - pass - print " read", x, "rows." - print "Replaced", err, "null values." - return skimDict - - - - -def bigrun(): - ts = time.clock() - - # =============================================================================== - # Set Parser Options - # =============================================================================== - parser = OptionParser() - parser.add_option("-s", "--scale", - action="store", dest="scale", default=1.0, - help="scale factor for multiple runs") - parser.add_option("-p", "--path", - action="store", dest="path", - help="project scenario path") - (options, args) = parser.parse_args() - # =============================================================================== - # Input File Names - # =============================================================================== - cvmInputPath = options.path + "/input/" - - cvmZonalProperties = cvmInputPath + "Zonal Properties CVM.csv" - - skimPath = options.path + "/output/" - - skimFileDict = {"Light_Mid": [skimPath + "impldt_MD_DU.TXT"], - "Medium_Mid": [skimPath + "impmhdt_MD_DU.TXT"], - "Heavy_Mid": [skimPath + "imphhdt_MD_DU.TXT"], - "Time_Mid": [skimPath + "impldt_MD_Time.TXT"]} - # =============================================================================== - # Read in scale factor - # =============================================================================== -# fin = open(settings.scaleFactorSource, "r") -# for row in fin: -# if len(row) > 15: # key word has 15 chars, so don't need to look at shorter lines -# if row[0:15] == "cvm.scaleFactor": -# s = row.index("=") -# scale = float(row[s+1:]) -# scaleName = row - - scale = float(options.scale) - print 40*"-" - print "Scaling tour gen with scale factor", scale - print 40*"-" - print - - - testZones = [1578, 88, 971, 2178, 3798, 2711, 4286] - fout = open("AccessVals.csv", "w") - outFileTest = csv.writer(fout, excelOne) - outFileTest.writerow(["I", "J", "AccType", "Cost", "Attr", "AccVal"]) - - - # =============================================================================== - # Produce accessibilities and other CVM-specific derived attributes - # =============================================================================== - # Read in zonal properties file - tazList, zonals = zonalProperties(fileName=cvmZonalProperties) - tazDict = {} - #tazList = tazList[:25] - for t in range(len(tazList)): - tazDict[tazList[t]] = t - - # Calculate accessibilities - print "Calculating accessibilities", round(time.clock(), 2) - - cvmZonals = {} # This is a zonal properties style dictionary, indexed by thing and containing a list in order of properties - accDict = settings.cvmAccDict # Dictionary for creating accessibilities; [skim, property, lambda] - accList = accDict.keys() - - - skimList = [] - for accType in accList: - cvmZonals[accType] = [] - if skimList.count(accDict[accType][0]) == 0: - skimList.append(accDict[accType][0]) - cvmZonals["LnJobs30"] = [] - cvmZonals["REZone"] = [] - - # Calculate percentage employment by industry; binary over 3000 flag - sectors = ["SV", "IN", "RE", "TH", "WH", "GO"] - for sect in sectors: - cvmZonals["Pct" + sect] = [] - cvmZonals["Over3K_" + sect] = [] - for taz in tazList: - idx = tazDict[taz] - for sect in sectors: - cvmZonals["Pct" + sect].append(zonals["CVM_" + sect][idx] / (zonals["TotEmp"][idx] + 0.0001)) - if zonals["CVM_" + sect][idx] > 3000: - cvmZonals["Over3K_" + sect].append(1) - else: - cvmZonals["Over3K_" + sect].append(0) - if cvmZonals["PctRE"][idx] > 0.5: - cvmZonals["REZone"].append(1) - else: - cvmZonals["REZone"].append(0) - - # Calculate employment and population density and cap if necessary - cvmZonals["PopDensCap"] = [] - cvmZonals["EmpDensCap"] = [] - for taz in tazList: - idx = tazDict[taz] - pop = zonals["Pop"][idx] - emp = zonals["TotEmp"][idx] - area = zonals["Area_SqMi"][idx] - if area > 0: - cvmZonals["PopDensCap"].append(min(pop/area, 50000)) - cvmZonals["EmpDensCap"].append(min(emp/area, 100000)) - else: - cvmZonals["PopDensCap"].append(0) - cvmZonals["EmpDensCap"].append(0) - - - # Read in skims - - print "Reading in CVM skims. Time:", round(time.clock()-ts, 2) - skimDict = {} - - for skimName in skimFileDict.keys(): - print "...", skimName, round(time.clock()-ts, 2) - skimList.append(skimName) - skimDict = csvSkim(tazList, tazList, tazDict, tazDict, - skimFileDict[skimName][0], skimDict, skimName) - - print skimDict.keys() - print len(tazDict.keys()) - print len(tazList) - #print tazDict - - print "Skims read in. Time:", round(time.clock()-ts, 2) - idx = 0 - for iTaz in tazList: - currAcc = len(accList) * [0] - jobs30 = 0 -# maxCost = [999999, -1, -1] - for jTaz in tazList: - iIdx = tazDict[iTaz] - jIdx = tazDict[jTaz] - for accType in accList: - skimType = accDict[accType][0] - try: - cost = skimDict[skimType][iIdx][jIdx] - except: - print skimType, iTaz, jTaz, iIdx, jIdx - print len(skimDict[skimType]) - print len(skimDict[skimType][0]) - crash -# if accType == "Acc_LE" and cost < maxCost[0]: -# maxCost[0] = cost -# maxCost[1] = jTaz -# maxCost[2] = jIdx - attr = zonals[accDict[accType][1]][jIdx] - lam = accDict[accType][2] - accVal = attr * math.exp(cost * lam) - #print accType, accList - currAcc[accList.index(accType)] = currAcc[accList.index(accType)] + accVal - if testZones.count(iTaz) > 0: - outFileTest.writerow([iTaz, jTaz, accType, cost, attr, accVal]) - - - cost = skimDict["Time_Mid"][iIdx][jIdx] - if cost < 30: - jobs30 = jobs30 + zonals["TotEmp"][jIdx] - - if idx % 250000 == 0: - print "Processed", idx, "OD pairs, most recently", iTaz, jTaz, round(time.clock()-ts,2) - - idx = idx + 1 -# if idx < 2000000: -# print iTaz, iIdx, maxCost - for c in range(len(accList)): - cvmZonals[accList[c]].append(currAcc[c]) - if jobs30 > 0: - cvmZonals["LnJobs30"].append(math.log(jobs30)) - else: - cvmZonals["LnJobs30"].append(0) - - - # First set is ship/no ship, second set is tours/emp - rangeDict = {'FA': [[0.01, 0.05], [1.22, 2.09]], - 'IN': [[0.37, 0.59], [0.13, 0.52]], - 'WH': [[0.35, 0.50], [0.17, 0.47]], - 'RE': [[0.10, 0.48], [0.33, 0.65]], - 'SV': [[0.10, 0.33], [0.08, 0.33]], - 'GO': [[0.10, 0.33], [0.08, 0.33]], - 'TH': [[0.12, 0.55], [0.25, 0.55]]} - - - adjDict = {'FA': [[-3.4677, -0.9635, -0.6686, -0.3527, 0.0091],[5.0449, 1.6675, 2.3762, 2.0193, 2.0415]], - 'IN': [[-0.6544, -1.8124, -1.3965, -1.2595, 1.094], [-0.2966, 0.9763, 0.9086, 1.0257, 1.1646]], - 'RE': [[-0.456, -1.4629, -1.6433, -0.666, 1.2942], [-0.1883, 0.4861, 1.1003, 0.8668, 0.8273]], - 'SV': [[-0.6344, -0.5504, -0.2762, -0.0793, 0.0627], [0.6254, 1.8833, 2.5968, 2.3156, 2.2625]], - 'GO': [[-0.6344, -0.5504, -0.2762, -0.0793, 0.0627], [0.6254, 1.8833, 2.5968, 2.3156, 2.2625]], - 'TH': [[-1.9161, -1.623, -1.5503, -1.1891, 0.0705], [0.5195, 0.3715, 0.8332, 0.6878, 0.3449]], - 'WH': [[0.1322, -1.0171, -1.1068, -0.8744, 1.2775], [-1.3517, 0.5526, 1.1136, 0.7629, 0.794]]} - - # Scale to match proportions for SANDAG - adjSandagDict = settings.genCalibDict - - - -# cout = open("e:/sjvitm_sdcvm_calibration.csv", "w") -# calibOut = csv.writer(cout, excelOne) -# calibOut.writerow(['Sector', 'Model', 'Iteration', 'Below', 'Within', 'Above', 'Score', 'Average', 'Param', 'Tours']) - - for iter in range(1): - print 15 * "-", "Iteration", iter, 15 * "-" - - # =========================================================================== - # Big loop: iterate through each industry and create generation - # =========================================================================== - for sector in settings.cvmSectors: - print "Calculating tour generation for sector", sector, round(time.clock()-ts, 2) - # Read in control file for this sector - fin = open(cvmInputPath + sector + ".csv", "r") - inFile = csv.reader(fin) - - paramDict = {} - paramDict["ShipNoShip"] = {} - paramDict["GenPerEmployee"] = {} - paramDict["TourTOD"] ={} - paramDict["VehicleTourType"] = {} - - paramDict["TourTOD_nest"] = {} - paramDict["VehicleTourType_nest"] = {} - - for row in inFile: - model = row[0] - if paramDict.has_key(model): - alt = row[1] - type = row[2] - nest = row[3] - param = float(row[5]) - if nest == "nest": - # put into nesting dictionary - if paramDict[model+"_nest"].has_key(type): - pass - else: - paramDict[model+"_nest"][type] = [] - - if param == 0: - paramDict[model+"_nest"][type].append([alt, 1]) - else: - paramDict[model+"_nest"][type].append([alt, param]) - - else: - if paramDict[model].has_key(alt): - pass - else: - paramDict[model][alt] = [] - parSet = (type, param) - paramDict[model][alt].append(parSet) - - #print paramDict - - # add to CVM zonals file / clear out values - for timePer in settings.cvmTimes: - cvmZonals[sector + "_" + timePer] = [] - cvmZonals[sector + "_Ship"] = [] - cvmZonals[sector + "_ToursEmp"] = [] - - - # Medium loop: iterate through each TAZ - for tazNum in tazList: - taz = tazDict[tazNum] - lu = int(round(float(zonals["CVM_LU_Type"][taz])))-1 - - # =================================================================== - # Phase One: Pass logsums up nested logit structure - # =================================================================== - - # --------------------------------- Tour vehicle type / purpose nest - - model = "VehicleTourType" - #print "Processing:", model, round(time.clock(), 2) - - altList = paramDict[model].keys() - utilDict = {} - - # Begin by calculating the utilities for each alternative - for alt in altList: - util = 0 - for name, par in paramDict[model][alt]: - if name == '': - util = util + par - val = "const" - else: - if cvmZonals.has_key(name): - util = util + cvmZonals[name][taz] * par - val = cvmZonals[name][taz] - elif zonals.has_key(name): - util = util + zonals[name][taz] * par - val = zonals[name][taz] - else: - raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) - utilDict[alt] = util -# if tazNum < 20: -# print alt, util, name, par, val -# if tazNum < 20: -# print 20 * "-" - - # Add the parameters to the nested model and pass it to the logsumerizer - modNest = "VehicleTourType_nest" - - for alt in altList: - paramDict[modNest][alt] = utilDict[alt] - - nestDict = copy.deepcopy(paramDict[modNest]) - #print taz, nestDict - vehAndPurpNest = logitNestToLogsums(nestDict) - CUPurpVeh = vehAndPurpNest["top"] - #print CUPurpVeh - - - # ------------------------------------------------------ Time Of Day - model = "TourTOD" - #print "Processing:", model, round(time.clock(), 2) - - altList = paramDict[model].keys() - utilDict = {} - - # Begin by calculating the utilities for each alternative - for alt in altList: - util = 0 - for name, par in paramDict[model][alt]: - if name == '': - util = util + par - val = "const" - else: - if cvmZonals.has_key(name): - util = util + cvmZonals[name][taz] * par - val = cvmZonals[name][taz] - elif zonals.has_key(name): - util = util + zonals[name][taz] * par - val = zonals[name][taz] - elif name == "CUPurpVeh": - util = util + CUPurpVeh * par - val = CUPurpVeh - else: - raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) - utilDict[alt] = util - #print alt, util, name, par, val - #print 20 * "-" - - # Add the parameters to the nested model and pass it to the logsumerizer - modNest = "TourTOD_nest" - - for alt in altList: - paramDict[modNest][alt] = utilDict[alt] - - nestDict = copy.deepcopy(paramDict[modNest]) - #print nestDict - tourTODNest = logitNestToLogsums(nestDict) - CUTimeOD = tourTODNest["top"] - #print tourTODNest - #print CUTimeOD - - - # ----------------------------------------------- Trips per employee - model = "GenPerEmployee" - #print "Processing:", model, round(time.clock(), 2) - - altList = paramDict[model].keys() - utilDict = {} - # Begin by calculating the utilities for each alternative - for alt in altList: - util = 0 - for name, par in paramDict[model][alt]: - if name == '': - util = util + par - val = "const" - else: - if cvmZonals.has_key(name): - util = util + cvmZonals[name][taz] * par - val = cvmZonals[name][taz] - elif zonals.has_key(name): - util = util + zonals[name][taz] * par - val = zonals[name][taz] - elif name == "CUTimeOD": - util = util + CUTimeOD * par - val = CUTimeOD - else: - raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) - utilDict[alt] = util - #print alt, util, name, par, val - #print 20 * "-" - - # Calibration Adjustments; form: xn, xn-1, fn, fn-1 - utilDict["Gen"] = utilDict["Gen"] + adjDict[sector][1][lu] - - genUtil = utilDict["Gen"] - CUGen = math.log(math.exp(genUtil) + math.exp(0)) # 0 is utility for no tours - - # ---------------------------------------------------- Ship / No Ship - model = "ShipNoShip" - #print "Processing:", model, round(time.clock(), 2) - - altList = paramDict[model].keys() - utilDict = {} - - # Begin by calculating the utilities for each alternative - for alt in altList: - util = 0 - for name, par in paramDict[model][alt]: - if name == '': - util = util + par - val = "const" - else: - if cvmZonals.has_key(name): - util = util + cvmZonals[name][taz] * par - val = cvmZonals[name][taz] - elif zonals.has_key(name): - util = util + zonals[name][taz] * par - val = zonals[name][taz] - elif name == "CUGen": - util = util + CUGen * par - val = CUGen - else: - raise LookupError("Couldn't find value " + name + " in zonal properties files, for alternative " + alt) - utilDict[alt] = util - #print alt, util, name, par, val - # Calibration Adjustments; form: xn, xn-1, fn, fn-1 - utilDict["Ship"] = utilDict["Ship"] + adjDict[sector][0][lu] - - - # =================================================================== - # Phase The Second: Calculate tours by time period - # =================================================================== - - # Get base employment (total employment for fleet allocators) - if sector == "FA": - baseEmp = zonals["TotEmp"][taz] - else: - baseEmp = zonals["CVM_"+sector][taz] - - #print "Base Employment:", baseEmp - - # Ship/No Ship proportions - - uShip = utilDict["Ship"] - uNoShip = utilDict["NoShip"] - - propShip = math.exp(uShip) / (math.exp(uShip) + math.exp(uNoShip)) - - shipEmp = baseEmp * propShip - -# tempOut.writerow([sector, tazList[taz], 'ShipNoShip', 'PropShip', propShip]) - cvmZonals[sector + "_Ship"].append(propShip) - #print "Shipping employees:", shipEmp, propShip - - # Generation: tours per employee - toursPerEmp = (10 * math.exp(genUtil)) / (1 + math.exp(genUtil)) - toursAllDay = shipEmp * toursPerEmp - - # Scale for SANDAG behaviours - toursAllDay = toursAllDay * adjSandagDict[sector][lu] - - - #print "Tours per employee:", toursPerEmp, toursAllDay -# tempOut.writerow([sector, tazList[taz], 'Generation', 'ToursPerEmp', toursPerEmp]) - cvmZonals[sector + "_ToursEmp"].append(toursPerEmp) - - # Time period: calculate probabilities of time period - linkDict = paramDict["TourTOD_nest"] - utilDict = tourTODNest - - #print linkDict - #print utilDict - - #print 20 * "-" - probDict = {} - - # First calculate probabilities for each node with subnodes - for node in linkDict.keys(): - if isinstance(linkDict[node], float): - pass - else: - #print node - expSum = 0 - for subNode in linkDict[node]: - expSum = expSum + math.exp(utilDict[subNode[0]] * subNode[1]) - #print subNode[0], utilDict[subNode[0]], subNode[1], expSum - - for subNode in linkDict[node]: - probDict[subNode[0]] = math.exp(utilDict[subNode[0]] * subNode[1]) / expSum - #print probDict - - # Now go down from top node and scale probabilities of any subnests by the nest prob - # (note: this only goes down one level; this works for the existing time of day nests, but needs to be changed for reuse - for node, coeff in linkDict["top"]: - if isinstance(linkDict[node], float): - pass - else: - for subNode, subcoeff in linkDict[node]: - probDict[subNode] = probDict[subNode] * probDict[node] - #print probDict - - - # Divide into time periods and write out - for timePer in settings.cvmTimes: - - try: - tours = probDict[timePer] * toursAllDay * scale - except: - print "Error in scaling tours:", probDict[timePer], toursAllDay, scale - tours = 0 - cvmZonals[sector + "_" + timePer].append(round(tours, 2)) - - -# ========================================================================== -# Output CVM Gen and Accessibility file -# ========================================================================== - print "Writing the data out...", round(time.clock(),2) - fout = open(cvmInputPath + "CVMToursAccess.csv", "w") - outFile = csv.writer(fout, excelOne) - - header = ["Taz"] - keyList = cvmZonals.keys() - keyList.sort() - header.extend(keyList) - outFile.writerow(header) - - for c in range(len(tazList)): - rowOut = [tazList[c]] - for keyType in keyList: - rowOut.append(cvmZonals[keyType][c]) - outFile.writerow(rowOut) - fout.close() - -# tout.close() -# cout.close() - print "DonE!" - -if __name__ == '__main__': - bigrun() diff --git a/sandag_abm/src/main/python/sdcvm_settings.py b/sandag_abm/src/main/python/sdcvm_settings.py deleted file mode 100644 index e5c1192..0000000 --- a/sandag_abm/src/main/python/sdcvm_settings.py +++ /dev/null @@ -1,43 +0,0 @@ -# Created on 2012-10-16 -# -# @author: Kevin - - -# =============================================================================== -# SDCVM properties -# =============================================================================== -cvmModes = ["Light", "Medium", "Heavy"] -cvmTimes = ["OE", "AM", "MD", "PM", "OL"] -# cvmTimes = ["OL"] -cvmSectors = ["GO", "SV", "IN", "RE", "TH", "WH", "FA"] # sectors to run; actual industries are hardcoded into sdcvm.py -# cvmSectors = ["WH"] # sectors to run; actual industries are hardcoded into sdcvm.py - -opCostScale = 1.0 # Scale factor for operating cost - -maxTaz = 4996 # highest taz number; assumes skims are 1-maxTaz without gaps - - -# Costs defined as [time, distance, money]. Not used right now. -cvmCostDict = {"Light": [-0.313, -0.138 * opCostScale, -1], - "Medium": [-0.313, -0.492 * opCostScale, -1], - "Heavy": [-0.302, -0.580 * opCostScale, -1]} - -# Dictionary for creating accessibilities; [skim, property, lambda] -cvmAccDict = {"Acc_LE": ["Light_Mid", "TotEmp", 3.0], - "Acc_LP": ["Light_Mid", "Pop", 3.0], - "Acc_ME": ["Medium_Mid", "TotEmp", 2.0], - "Acc_MP": ["Medium_Mid", "Pop", 2.0], - "Acc_HE": ["Heavy_Mid", "TotEmp", 1.0], - "Acc_HP": ["Heavy_Mid", "Pop", 1.0] - } - -# Calibration adjustment scale factors for tour generation -# Factors by land use type: -# [Low dens, Residential, Retail/Comm, Industrial, Emp Node] -genCalibDict = {'FA': [0.4931, 1.8562, 2.3996, 1.7171, 2.4541], - 'GO': [11.8418, 4.5588, 4.1018, 5.1797, 14.1177], - 'IN': [0.6712, 0.7201, 0.7671, 0.7934, 0.2481], - 'RE': [0.6764, 1.2354, 1.7748, 1.2966, 0.1446], - 'SV': [4.1374, 2.2546, 2.6429, 3.2391, 4.2139], - 'TH': [0.7609, 0.4813, 0.4271, 1.1211, 0.5308], - 'WH': [1.2189, 0.5536, 0.5035, 0.4371, 0.2212]} diff --git a/sandag_abm/src/main/python/sdcvm_settings.pyc b/sandag_abm/src/main/python/sdcvm_settings.pyc deleted file mode 100644 index 83196f823130412577bf7fa69ec67a3c4feb5385..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1332 zcmYk4drVtZ9LIlc>5GTJfFd%dn**vM8Tl`ul7N>w+oO( zm$^8T;T99k7GH^p%hb%E`-iE~hN)4Ek3ZDJiIHt_*)Zxb%q)iK{M}O?ySF{R^Zoo@ z=k!<8LRqoBas5C7mL4lviN5M#0|3&;0B8hmf(Y$RfJK02KntK1xEUhkwE@}zivddj zO99IOZvbwEh!LxI8u_%{?Z`3w3s>S6)K3b+xl0k9FU z3DBuulWye3+X3$Yyc4h)@Ge1ZL|Xti0d59t1#}5oO!RKRHoz@_Zoqp4Eg`xU@Ls@e zfcF8m3u-6Y0oV!103|?=pru5+0KI_SfIh$;LCffTbhus!IsG&s%}h(5!jqb2ptj9X(z`fA26GTuM&g8Xb^j`1_ER380k>RZOI{#1rR&oL#`l%! zQCAtCKF{8m{rEEDnYKNX(%w^yzjt=-#S`Py)cNH2`%l=75`ShoJoV;Av&1h=f712N zoL%A#Ipa&V8H(}!IDh&6N2%#V!>>=j_Vq7}Kk$XM5D82|i(lh0Y^|DuDmB}wvJP7G=&#rgr;Vne=}Tf)I*>(K zT^3;_O+p_Qc|vkR%MPnr3CZk`H>+x)L_8&nL8ap(;rN(vQKatcRubBBCQVDoQp)6V z#v4y5gF0}MhSf!FA?e)~^>hVd!?}zab!~3l5{nO}UD2UYEt$gCD}HO6|A#b_z2HB=jFO;v`9rMJpbXK<{%{{wY;X`BE6 diff --git a/sandag_abm/src/main/python/sdcvm_summarize.py b/sandag_abm/src/main/python/sdcvm_summarize.py deleted file mode 100644 index 16f802a..0000000 --- a/sandag_abm/src/main/python/sdcvm_summarize.py +++ /dev/null @@ -1,99 +0,0 @@ -import csv, time -import sdcvm -# CSTDM libraries -import sdcvm_settings as settings -from optparse import OptionParser - -# External libraries -ts = time.clock() - - -class excelOne(csv.excel): - # define CSV dialect for Excel to avoid blank lines from default \r\n - lineterminator = "\n" - -# =============================================================================== -# Set Parser Options -# =============================================================================== -parser = OptionParser() -parser.add_option("-p", "--path", - action="store", dest="path", - help="project scenario path") -(options, args) = parser.parse_args() -# =============================================================================== -# Input File Names -# =============================================================================== -cvmInputPath = options.path + "/input/" -cvmZonalProperties = cvmInputPath + "Zonal Properties CVM.csv" -skimPath = options.path + "/output/" -cvmPath = options.path + "/output/" - -tazList = range(1, settings.maxTaz+1) -tazList, zonals = sdcvm.zonalProperties(fileName=cvmZonalProperties) -tazDict = {} -# tazList = tazList[:25] -for t in range(len(tazList)): - tazDict[tazList[t]] = t - -# Read in skims - -print "Reading in CVM skims. Time:", round(time.clock()-ts, 2) -skimDict = {} -skimList = [] - - -print "... Midday distance", round(time.clock()-ts, 2) -skimList.append("Dist_Mid") -skimDict = sdcvm.csvSkim(tazList, tazList, tazDict, tazDict, - skimPath + "impldt_MD_Dist.TXT", skimDict, "Dist_Mid") - - -bigDict = {} -for ind in settings.cvmSectors: - for tim in settings.cvmTimes: - print ind, tim, round(time.clock()-ts, 2) - fin = open(cvmPath + "Trip_" + ind + "_" + tim + ".csv", "r") - inFile = csv.reader(fin) - header = inFile.next() - for row in inFile: - mode = row[header.index("Mode")] - trip = int(row[header.index("Trip")]) - purp = row[header.index("TourType")] - toll = row[header.index("TripMode")] - tollAv = row[header.index("TollAvailable")] - home = int(row[header.index("HomeZone")]) - iTaz = int(row[header.index("I")]) - jTaz = int(row[header.index("J")]) - tim = row[header.index("TripTime")] - - iIdx = tazList.index(iTaz) - jIdx = tazList.index(jTaz) - newjIdx = jIdx # new index created to add 12 to include external zones 1-12 in the skim matrix - - key = (ind, mode, purp, toll, tollAv, tim, home) - if bigDict.has_key(key): - pass - else: - bigDict[key] = [0, 0, 0, 0] - - bigDict[key][1] = bigDict[key][1] + 1 - bigDict[key][2] = bigDict[key][2] + skimDict["Dist_Mid"][iIdx][newjIdx] - if trip == 1: - bigDict[key][0] = bigDict[key][0] + 1 - if iTaz == jTaz: - bigDict[key][3] = bigDict[key][3] + 1 - -fout = open(cvmPath + "Gen and trip sum.csv", "w") -outFile = csv.writer(fout, excelOne) - -keyList = bigDict.keys() -keyList.sort() -header = ["Industry", "Mode", "Purpose", "Toll", "TollAv", "Time", "TAZ", "Tours", "Trips", "Dist", "Intra"] -outFile.writerow(header) - - -for key in keyList: - outRow = list(key) - outRow.extend(bigDict[key]) - outFile.writerow(outRow) -fout.close() diff --git a/sandag_abm/src/main/python/serverswap.py b/sandag_abm/src/main/python/serverswap.py deleted file mode 100644 index 5da8c6d..0000000 --- a/sandag_abm/src/main/python/serverswap.py +++ /dev/null @@ -1,92 +0,0 @@ -# Rick.Curry@sandag.org -# July 12, 2016 - -import os -import socket -import csv -from optparse import OptionParser - - -def swap_servers(infile, outfile, searchitem, sep_str, swap_value, skip_lines): - found = 0 - print infile, outfile - print searchitem, sep_str, swap_value - lines = [] - with open(infile) as propInFile: - for line in propInFile: - if line.find(searchitem) > -1: - if found == skip_lines: - line = line.split(sep_str, 1)[0] + sep_str + swap_value + "\n" - else: - found += 1 - lines.append(line) - with open(outfile, 'w') as propOutFile: - for line in lines: - propOutFile.write(line) - - -# Set Parser Options -parser = OptionParser() -parser.add_option("-p", "--path", - action="store", dest="path", - help="project scenario path") -(options, args) = parser.parse_args() - -# Set Paths -dst_dir_bin = options.path + "/bin/" -dst_dir_conf = options.path + "/conf/" - -# Get IP Address -ip = socket.gethostbyname(socket.gethostname()) -print str(ip) - -# Read Server Info -fileServer = options.path + "/conf/server-config-local.csv" -if not os.path.exists(fileServer): - fileServer = "T:/ABM/release/ABM/config/server-config.csv" -print fileServer -logFile = options.path + "/logFiles/serverswap.log" -if os.path.exists(logFile): - os.remove(logFile) -dictServer = csv.DictReader(open(fileServer)) - -# Check for Matching IP Address and Stop on Row -match = 'false' -for row in dictServer: - print row - if row['ActualIP'] == str(ip): - match = 'true' - print match - serverName = row['ServerName'] - print serverName - modelIP = row['ModelIP'] - break - -# Write error log if IP address not found -logWriteFile = open(logFile, "w") -if match == 'false': - logWriteFile.write('Using server-config file in: ' + fileServer + "\n") - logWriteFile.write('FATAL, Head Node not found - check for ' + str(ip)) - print 'Head Node not found' -else: - # Update Files in serverswap_files.csv - logWriteFile.write('Using server-config file in: ' + fileServer + "\n") - logWriteFile.write('MATCH, Head Node found - ' + str(ip)) - skip = 0 - fileUpdate = options.path + "/conf/serverswap_files.csv" - print fileUpdate - filesToUpdate = csv.DictReader(open(fileUpdate)) - for update in filesToUpdate: - print update - # Special section for StopABM.cmd which does not have a property=value format - if update['property'] == 'pskill': - refValue = row[update['refValue']] + ' java.exe' - print row[update['refValue']] - print refValue - swap_servers(options.path + update['fileName'], options.path + update['fileName'], - update['property'], update['separator'], refValue, skip) - skip += 1 - else: - # General section for file updates - swap_servers(options.path + update['fileName'], options.path + update['fileName'], - update['property'], update['separator'], row[update['refValue']], 0) diff --git a/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R b/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R deleted file mode 100644 index 6b1b3a7..0000000 --- a/sandag_abm/src/main/r/INRIX_OutlierAnalysis_Final.R +++ /dev/null @@ -1,532 +0,0 @@ -### -# OBJECTIVE: -# outlier analysis for SANDAG INRIX travel time data -# travel time and speed plots -# std dev and mean - -# INPUTS: -# inrix data - "2012_10.csv" -# inrix to tcoved correspondence (created by Fehr and Peers) - "inrix_2012hwy.txt" (also an input to regression analysis script) - -# OUTPUTS: -# travel time SD/ mean travel time in 15 mins - "inrix2012_nooutliers_traveltime.dbf" (input to regression analysis script) - -# INRIX DATA: -# october 2012 inrix data is used. -# the inrix data are in 1-mins increment and available for freeways, major/minor arterials, collectors and ramps. - -# INRIX TO MODEL NETWORK CORRESPONDENCE: -# Doubtful if ramps are in the data, perhaps correspondence to ramps isn't correct. -# fehr and peers established an initial correspondence between the model network segments and INRIX TMC segments. -# the correspondence included one record for each model link-to-inrix segments. Some model links may corresponds to multiple TMCs, -# a TMC proportion which is proportion of the TMC feature covered by the corresponding model network feature was also provided. -# The records that may not represent true correspondence, due to high frequency of link join, were flagged in the correspondence. - -# DATA PROCESSING: -# While preparing the data, the records that were flagged in the correspondence file were removed. -# Then, a TMC segment was assigned with one model link by finding the record with the highest TMC proportion. The characteristics of that model link were attached to the TMC segment. -# The analysis was restricted to weekdays. weekend’s data points were eliminated from the dataset -# plots of speeds and travel times are created -# outlier analysis using adjusted boxplot -# removed data points in period 4:15 am - 4:30 am (unexpected variation). also, removed based on reference_speed (if "reference_speed-speed>5") -# comparison of before and after outlier analysis plots -# travel time variability (standard deviation) is calculated for every 15 minutes time interval -# travel time unit = travel time per sec per mile (=travel_time_sec/seg.length) -# seg.length:=speed*travel_time_minutes/60 -# for a tmc segment (travel time sd/travel time mean) is calculated over all weekdays in 15-mins bins - -# OTHER SCRIPTS USED: -# utilfunc.R - -# by: nagendra.dhakar@rsginc.com -# for: SANDAG SHRP C04 - Tolling and Reliability -### - -library(foreign) -library (stringr) -library(xtable) -library(reshape) -library(reshape2) -library(XLConnect) -library(descr) -library(Hmisc) -library(data.table) -library(plyr) -library(gtools) -library(vioplot) -library(lattice) -library(grid) -library(timeDate) -library(ggplot2) -library(robustbase) -library(readr) - -# -------------------- 0. SOURCE FILES AND CONFIG SETTINGS ----------------------- - -# workspace and source files -setwd("E:/Projects/Clients/SANDAG/Data_2015") -source("utilfunc.R") - -# input files -inrix_2012_file = "./INRIX/2012_10/2012_10.csv" -inrix_2014_file = "./INRIX/2014_10/october_2014.csv" -inrixhwycorresp_file = "./TMC/inrix_2012hwy.txt" - -# config settings -year = "2012" # data year -interval<-15 # time bin interval in minutes -bptype<-"adjusted" # box plot - original or adjusted -field="travel.time.sec.per.mile" -ByDOW<-FALSE # by day of week - false, analysis is performed for all weekdays -OUTLIER<-TRUE # outlier analysis or confidence score (CS) removal - -# -------------------- 1. LOAD DATA ---------------------- - -# read and load inrix data -if (year=="2012"){ - # read and load 2012 data - outputsDir="./INRIX/2012_10/" - readSaveRdata(inrix_2012_file,"inrix_2012") - inrix_2012 <- assignLoad(paste0(inrix_2012_file,".Rdata")) -} else{ - # read and load 2014 data - outputsDir="./INRIX/2014_10/" - readSaveRdata(inrix_2014_file,"inrix_2014") - inrix_2014 <- assignLoad(paste0(inrix_2014_file,".Rdata")) -} - -# read and load inrix to hwy correspondence data -readSaveRdata(inrixhwycorresp_file,"inrixhwycorresp") -inrixhwycorresp <- assignLoad(paste0(inrixhwycorresp_file,".Rdata")) - -# -------------------- 2. CLEAN/PROCESS DATA -------------- - -# add a new field tmc_code without first character ('-' or '+') -inrixhwycorresp[,tmc_code:=substr(TMC,2,nchar(TMC))] - -# remove the segments that are flagged -inrixhwycorresp<-subset(inrixhwycorresp,FLAG==0) - -# get unique tmc_code with variables corresponding to maximum of TMCProp -df.orig <-inrixhwycorresp -df.agg<-aggregate(TMCProp~tmc_code,inrixhwycorresp,max) -df.max <- merge(df.agg, df.orig) - -# facility type - keep only two columns -#tmc_ifc <-df.max[,c("tmc_code","IFC","HWYCOV_ID"),] -tmc_ifc <-df.max[,c("tmc_code","TMC_Len","IFC","HWYCOV_ID","NM","LENGTH","ISPD","ABLNO","ABLNA","ABLNP","BALNO","BALNA","BALNP","ABCNT","BACNT","IHOV","ABAU","BAAU"),] -tmc_ifc$LENGTH<-as.numeric(gsub(",","",tmc_ifc$LENGTH)) -tmc_ifc$HWYCOV_ID<-as.numeric(gsub(",","",tmc_ifc$HWYCOV_ID)) - -# for plots -tmc_ifc_temp <-df.max[,c("tmc_code","IFC"),] - -# merge INRIX travel time data with hwyinrix correspondence file -inrix_2012_ifc<-merge(inrix_2012,tmc_ifc,by="tmc_code") - -# set data -alldata<-inrix_2012_ifc - -# determine Day Of Week (DOW): weekday (1) or weekend (0) -alldata[,DOW:=ifelse(isWeekday(as.Date(measurement_tstamp)),1,0)] - -# keep only weekdays -alldata<-alldata[DOW==1] - -# calculate time stamp -alldata[,date:=substr(measurement_tstamp,0,10)] -alldata[,hour:=as.numeric(substr(measurement_tstamp, 12, 13))] -alldata[,min:=as.numeric(substr(measurement_tstamp, 15, 16))] -alldata[,totalmin:=hour*60+min] -alldata[,totalhour:=hour+(min/60)] - -#travel time -alldata[,travel_time_sec:=travel_time_minutes*60] - -# determine time of day (TOD) -alldata[totalmin>=0 & totalmin<=209,tod:='EV1'] -alldata[totalmin>=210 & totalmin<=359,tod:='EA'] #1-early AM -alldata[totalmin>=360 & totalmin<=539,tod:='AM'] #2-AM peak -alldata[totalmin>=540 & totalmin<=929,tod:='MD'] #3-Midday -alldata[totalmin>=930 & totalmin<=1139,tod:='PM'] #4-PM Peak -alldata[totalmin>=1140 & totalmin<=1440,tod:='EV2'] - -# create time intervals -alldata[,todcat:=findInterval(totalmin,seq(0,1440,by=interval))] - -# --------------- 3. ANALYSIS (OUTLIER DETECTION) ------------------- - -if (OUTLIER) { - # ---------------- outlier analysis -------------------- - # create a columns "outlier" with default set to 0 - alldata[,outlier:=0] - - # find tmc segments - segmentslist<-unique(alldata$tmc_code) - - if (length(segmentslist)==0) { - stop("no tmc segments in the dataset") - } - - # create dataframes by tmc_code - alldata_bytmc<-by(alldata,alldata$tmc_code, function(x) x) - - if (length(segmentslist)>length(alldata_bytmc)) { - warning("not all tmc segments have dataframe") - } - - # identify outliers for each dataframe - lapply(alldata_bytmc, detectoutliers) - - # combine dataframes into one dataset - alldata.outliers<-do.call(rbind,alldata_bytmc) - - # calculate length (mile) - alldata.outliers[,seg.length:=speed*travel_time_minutes/60] - alldata.outliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] - - #free-up memory - #rm(alldata) - rm(inrix_2012) - gc() - - if (sum(alldata.outliers$outlier)==0) { - stop("no outliers are detected") - } - - outliers_byIFC<-c(0,0,0,0,0,0,0,0,0) - outliers_byIFC[1]<-sum(alldata.outliers[IFC==1]$outlier) - outliers_byIFC[2]<-sum(alldata.outliers[IFC==2]$outlier) - outliers_byIFC[3]<-sum(alldata.outliers[IFC==3]$outlier) - outliers_byIFC[4]<-sum(alldata.outliers[IFC==4]$outlier) - outliers_byIFC[5]<-sum(alldata.outliers[IFC==5]$outlier) - outliers_byIFC[6]<-sum(alldata.outliers[IFC==6]$outlier) - outliers_byIFC[7]<-sum(alldata.outliers[IFC==7]$outlier) - outliers_byIFC[8]<-sum(alldata.outliers[IFC==8]$outlier) - outliers_byIFC[9]<-sum(alldata.outliers[IFC==9]$outlier) - - # also remove data points in period 4:15 am - 4:30 am. also, remove based on reference_speed - alldata.nooutliers<-alldata.outliers[todcat==18 & reference_speed-speed>5,outlier:=1] - - # remove outliers - alldata.nooutliers<-alldata.outliers[outlier==0] - -} else { - # filter out data points based on confidence score - # lose about 19% of the data points - alldata.nooutliers <- alldata[alldata$confidence_score>10] - alldata.nooutliers[,seg.length:=speed*travel_time_minutes/60] - alldata.nooutliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] -} - -# free-up memory -rm(alldata_bytmc) -rm(alldata) -rm(alldata.outliers) -rm(inrix_2012_ifc) -gc() - -alldata.nooutliers$dayofweek<-dayOfWeek(as.timeDate(alldata.nooutliers$date)) - -# ---------------- 4. OUTPUT ---------------------- -if (OUTLIER){ - # write out the new dataset - write.table(alldata.nooutliers,"inrix_2012_nooutliers_15mins.txt",sep=",",row.names=F,quote=F) -} else { - # write out the new dataset - no confidence score 10 - write.table(alldata.nooutliers,"inrix_2012_noCS10_15mins.txt",sep=",",row.names=F,quote=F) -} - - -if (FALSE){ - # for test - withnooutliers_file="./inrix_2012_nooutliers.txt" - readSaveRdata(withnooutliers_file,"inrix_2012_nooutliers") - inrix_2012_nooutliers <- assignLoad(paste0(withnooutliers_file,".Rdata")) - alldata.nooutliers <- inrix_2012_nooutliers - - alldata.nooutliers[,seg.length:=speed*travel_time_minutes/60] - alldata.nooutliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] - - # for test - withoutliers_file="./inrix_2012_outliers.txt" - readSaveRdata(withoutliers_file,"inrix_2012_outliers") - inrix_2012_outliers <- assignLoad(paste0(withoutliers_file,".Rdata")) - - inrix_2012_outliers[,seg.length:=speed*travel_time_minutes/60] - inrix_2012_outliers[,travel.time.sec.per.mile:=travel_time_sec/seg.length] - - alldata.outliers=inrix_2012_outliers - alldata.nooutliers=inrix_2012_outliers[outlier==0] - -} - -# ----------------- 5. Statistics ----------------- - -# for raw data set - outliers are included (though identified) -if (FALSE){ - # calculate SD - all data points - #temp<-cast(alldata.outliers,tmc_code~todcat,sd,value=field) - temp<-dcast(alldata.outliers,tmc_code+date+todcat~field,fun.aggregate=sd) - - alldata.outliers.sd<-temp[complete.cases(temp),] # keeps only complete values - no missing values - temp=alldata.outliers.sd - temp<-merge(temp,tmc_ifc,by="tmc_code") - - #output SD file - #outfile=paste("inrix2012_outliers_",field,"_sd.txt") - #write.table(temp,outfile,sep="\t",row.names=F,quote=F) - - outfile=paste("inrix2012_outliers_",field,"_sd.dbf") - write.dbf(temp,outfile) -} - -# save data -if (OUTLIER) { - # outlier method - save(alldata.nooutliers,file = "alldatanooutliers.Rdata") -} else { - # confidence score 10 method - save(alldata.nooutliers,file = "alldatanoCS10.Rdata") -} - -# load data -if (OUTLIER) { - # outlier method - alldata.nooutliers <- assignLoad("alldatanooutliers.Rdata") -} else { - # confidence score 10 method - alldata.nooutliers <- assignLoad("alldatanoCS10.Rdata") -} - -# std. dev of all data points -if (!ByDOW){ - # std dev for all weekdays - alldata.nooutliers.sd<-cast(alldata.nooutliers,tmc_code~todcat,sd,value=field) - - # reshape the dataset - alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id="tmc_code") - setnames(alldata.nooutliers.sd.melt,"value","sd") - - # field name got changed to "value" due to cast in the previous step - set back to field name - setnames(alldata.nooutliers,"value",field) -} - -if (FALSE) { - # avg speed for Wu - 05/09/2016 - alldata.nooutliers.mean<-aggregate(alldata.nooutliers$speed,by=list(alldata.nooutliers$tmc_code, alldata.nooutliers$todcat),FUN = mean, na.rm=TRUE, na.action = na.pass) - tmc_length<-aggregate(alldata.nooutliers$seg.length,by=list(alldata.nooutliers$tmc_code),FUN = mean, na.rm=TRUE, na.action = na.pass) - - setnames(alldata.nooutliers.mean,c("Group.1","Group.2","x"),c("tmc_code","todcat","avg_speed")) - setnames(tmc_length,c("Group.1","x"),c("tmc_code","avg_tmc_length")) - - alldata.nooutliers.mean.hwycov<-merge(alldata.nooutliers.mean,tmc_ifc,by="tmc_code") - alldata.nooutliers.mean.hwycov<-merge(alldata.nooutliers.mean.hwycov,tmc_length,by="tmc_code") - - alldata.nooutliers.mean.hwycov$TMC_Len<-as.numeric(gsub(",","",alldata.nooutliers.mean.hwycov$TMC_Len)) - write.table(alldata.nooutliers.mean.hwycov,"inrix_2012_avgspeed_wu.csv",sep=",",row.names=F,quote=F) - -} - -# std. dev of all datapoints of a weekday (ex. all monday, all tuesday,.., all friday) - not used -if (ByDOW) { - alldata.nooutliers.sd<-dcast(alldata.nooutliers,tmc_code+todcat~dayofweek,value.var=field,fun.aggregate=sd,na.rm=TRUE) - - if (FALSE){ - - if (field=="speed"){ - alldata.nooutliers.sd<-aggregate(alldata.nooutliers$speed,by=list(alldata.nooutliers$tmc_code,alldata.nooutliers$dayofweek,alldata.nooutliers$todcat),FUN = sd) - } else { - alldata.nooutliers.sd<-aggregate(alldata.nooutliers$travel.time.sec.per.mile,by=list(alldata.nooutliers$tmc_code,alldata.nooutliers$dayofweek,alldata.nooutliers$todcat),FUN = sd) - } - } - # reshape the dataset - not needed in aggregate method - alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id=c("tmc_code","todcat")) - setnames(alldata.nooutliers.sd.melt,c("variable","value"),c("dayofweek","sd")) - -} - -#alldata.nooutliers.sd<-alldata.nooutliers.sd[complete.cases(alldata.nooutliers.sd),] - -# For Shift Variables - now they are calculated in regression analysis script -# calculate mean - -# all data points -if (!ByDOW){ - # calculate mean - alldata.nooutliers.mean<-dcast(alldata.nooutliers,tmc_code~todcat,value.var=field,fun.aggregate=mean,na.rm=TRUE) - alldata.nooutliers.mean.melt<-melt(alldata.nooutliers.mean,id="tmc_code") - setnames(alldata.nooutliers.mean.melt,c("variable","value"),c("todcat","mean")) -} - -# data points by weekday -if (ByDOW){ - alldata.nooutliers.mean<-dcast(alldata.nooutliers,tmc_code+todcat~dayofweek,value.var=field,fun.aggregate=mean,na.rm=TRUE) - alldata.nooutliers.mean.melt<-melt(alldata.nooutliers.mean,id=c("tmc_code","todcat")) - - # set column names - setnames(alldata.nooutliers.mean.melt,c("variable","value"),c("dayofweek","mean")) - -} - -# five model time periods -alldata.nooutliers.mean.melt$todcat.int <- as.integer(alldata.nooutliers.mean.melt$todcat) - -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>0 & alldata.nooutliers.mean.melt$todcat.int<=14,'EV','') -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>14 & alldata.nooutliers.mean.melt$todcat.int<=24,'EA',alldata.nooutliers.mean.melt$tod) -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>24 & alldata.nooutliers.mean.melt$todcat.int<=36,'AM',alldata.nooutliers.mean.melt$tod) -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>36 & alldata.nooutliers.mean.melt$todcat.int<=62,'MD',alldata.nooutliers.mean.melt$tod) -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>62 & alldata.nooutliers.mean.melt$todcat.int<=76,'PM',alldata.nooutliers.mean.melt$tod) -alldata.nooutliers.mean.melt$tod<-ifelse(alldata.nooutliers.mean.melt$todcat.int>76 & alldata.nooutliers.mean.melt$todcat.int<=96,'EV',alldata.nooutliers.mean.melt$tod) - -if (FALSE) -{ - # max time by tmc_code and dayofweek - #MaxTime.AM<-do.call(rbind,lapply(split(MeanTime.AM,list(MeanTime.AM$tmc_code,MeanTime.AM$dayofweek)), function(x) x[which.max(x$mean),])) - #MaxTime.AM<-MaxTime.AM[,c("tmc_code","dayofweek","todcat")] - #setnames(MaxTime.AM,"todcat","MaxAMtod") - - MaxTime.AM<-do.call(rbind,lapply(split(MeanTime.AM,list(MeanTime.AM$tmc_code)), function(x) x[which.max(x$mean),])) - MaxTime.AM<-MaxTime.AM[,c("tmc_code","todcat.int")] - setnames(MaxTime.AM,"todcat.int","MaxAMtod") - - #MaxTime.PM<-do.call(rbind,lapply(split(MeanTime.PM,list(MeanTime.PM$tmc_code,MeanTime.PM$dayofweek)), function(x) x[which.max(x$mean),])) - #MaxTime.PM<-MaxTime.PM[,c("tmc_code","dayofweek","todcat")] - #setnames(MaxTime.PM,"todcat","MaxPMtod") - - MaxTime.PM<-do.call(rbind,lapply(split(MeanTime.PM,list(MeanTime.PM$tmc_code)), function(x) x[which.max(x$mean),])) - MaxTime.PM<-MaxTime.PM[,c("tmc_code","todcat.int")] - setnames(MaxTime.PM,"todcat.int","MaxPMtod") - - # merge - #temp<-merge(alldata.nooutliers.sd.melt,MaxTime.AM,by=c("tmc_code","dayofweek"), all.x=TRUE) - #temp1<-merge(temp,MaxTime.PM,by=c("tmc_code","dayofweek"), all.x=TRUE) - - temp<-merge(alldata.nooutliers.sd.melt,MaxTime.AM,by="tmc_code", all.x=TRUE) - temp1<-merge(temp,MaxTime.PM,by="tmc_code", all.x=TRUE) -} - -# merge sd and mean values -temp<-merge(alldata.nooutliers.sd.melt,alldata.nooutliers.mean.melt,by=c("tmc_code","todcat")) -temp$sdpermean<-temp$sd/temp$mean - -# estimation dataset -data.est <- temp -data.est <- data.est[,c("tmc_code","todcat","sd","mean","todcat.int","tod","sdpermean")] - -# save data -if (OUTLIER) { - save(data.est,file = "data.est.nooutliers.Rdata") -} else { - save(data.est,file = "data.est.noCS10.Rdata") -} - -# load data -if (OUTLIER) { - data.est <- assignLoad("data.est.nooutliers.Rdata") -} else { - data.est <- assignLoad("data.est.noCS10.Rdata") -} - -data.est<-na.omit(data.est) - -# don't merge attributes here -#data.est<-merge(data.est,tmc_ifc,by="tmc_code") - -if (OUTLIER) { - # output without outliers SD file - in DBF format - outfile="inrix2012_nooutliers_traveltime.dbf" -} else { - outfile="inrix2012_noCS10_traveltime.dbf" -} - -# write to dbf file -write.dbf(data.est,outfile) - -# output without outliers SD file - in text format -if (FALSE){ - outfile=paste("inrix2012_nooutliers_",field,"_sd.txt") - write.table(temp,outfile,sep="\t",row.names=F,quote=F) -} - -rm(alldata.nooutliers.mean) -gc() - -# ----------------- 6. Plots ----------------- - -# arrange data -alldata.outliers.sd.melt<-melt(alldata.outliers.sd,id="tmc_code") -alldata.nooutliers.sd.melt<-melt(alldata.nooutliers.sd,id="tmc_code") - -# determine time of day (TOD) - -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=0 & alldata.outliers.sd.melt$todcat<=7,'EV1','') -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=7 & alldata.outliers.sd.melt$todcat<=12,'EA',alldata.outliers.sd.melt$tod) -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=12 & alldata.outliers.sd.melt$todcat<=18,'AM',alldata.outliers.sd.melt$tod) -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=18 & alldata.outliers.sd.melt$todcat<=31,'MD',alldata.outliers.sd.melt$tod) -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=31 & alldata.outliers.sd.melt$todcat<=38,'PM',alldata.outliers.sd.melt$tod) -alldata.outliers.sd.melt$tod<-ifelse(alldata.outliers.sd.melt$todcat>=38 & alldata.outliers.sd.melt$todcat<=48,'EV2',alldata.outliers.sd.melt$tod) - -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=0 & alldata.nooutliers.sd.melt$todcat<7,'EV1','') -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=7 & alldata.nooutliers.sd.melt$todcat<12,'EA',alldata.nooutliers.sd.melt$tod) -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=12 & alldata.nooutliers.sd.melt$todcat<18,'AM',alldata.nooutliers.sd.melt$tod) -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=18 & alldata.nooutliers.sd.melt$todcat<31,'MD',alldata.nooutliers.sd.melt$tod) -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=31 & alldata.nooutliers.sd.melt$todcat<38,'PM',alldata.nooutliers.sd.melt$tod) -alldata.nooutliers.sd.melt$tod<-ifelse(alldata.nooutliers.sd.melt$todcat>=38 & alldata.nooutliers.sd.melt$todcat<=48,'EV2',alldata.nooutliers.sd.melt$tod) - -# arrange data -alldata.outliers.sd.melt<-merge(alldata.outliers.sd.melt,tmc_ifc_temp,by="tmc_code") -alldata.nooutliers.sd.melt<-merge(alldata.nooutliers.sd.melt,tmc_ifc_temp,by="tmc_code") - -# segment by facility type (IFC) -alldata.outliers.sd.melt.byifc<-by(alldata.outliers.sd.melt,alldata.outliers.sd.melt$IFC, function(x) x) -alldata.nooutliers.sd.melt.byifc<-by(alldata.nooutliers.sd.melt,alldata.nooutliers.sd.melt$IFC, function(x) x) - -# make plots -for (p in 1:length(alldata.outliers.sd.melt.byifc)) { - - plot1<-myplot_sd(alldata.outliers.sd.melt.byifc[[p]],"raw",outliers_byIFC[p],field) - plot2<-myplot_sd(alldata.nooutliers.sd.melt.byifc[[p]],"nooutliers",outliers_byIFC[p],field) - - # save as JPEG - print(multiplot(plot1,plot2),cols=1) - dev.copy(jpeg,filename=paste(outputsDir,field, "_SD_","IFC_",p,".jpeg", sep = ""),width=1280, height=1280) - dev.off() - -} - -#debug -#myplot_sd(alldata.nooutliers.sd.melt.byifc[["9"]]) - -myplot_sd(alldata.outliers.sd.melt,"SD_per_mile_outliers.jpeg") -myplot_sd(alldata.nooutliers.sd.melt,"SD_per_mile_nooutliers.jpeg") - -# ---------------- 5. PLTOS ---------------------- -if(FALSE) { - alldata.outliers.day<-alldata.outliers[date=="2012-10-01"] - - alldata.outliers.day[outlier==0, color.codes:="#000000"] #black - alldata.outliers.day[outlier==1, color.codes:="#FF0000"] #red - - alldata.outliers.day[outlier==0, color.names:="valid"] - alldata.outliers.day[outlier==1, color.names:="outlier"] - - # all freeways of a day data points - alldata.facility.type<-by(alldata.outliers.day,alldata.outliers.day$IFC, function(x) x) - - lapply(alldata.facility.type, myplot1) - - alldata.nooutliers.day<-alldata.nooutliers[date=="2012-10-01"] - - alldata.nooutliers.day[outlier==0, color.codes:="#000000"] #black - alldata.nooutliers.day[outlier==1, color.codes:="#FF0000"] #red - - alldata.nooutliers.day[outlier==0, color.names:="valid"] - alldata.nooutliers.day[outlier==1, color.names:="outlier"] - - alldata.facility.type<-by(alldata.nooutliers.day,alldata.nooutliers.day$IFC, function(x) x) - - lapply(alldata.facility.type, myplot1) -} - - diff --git a/sandag_abm/src/main/r/RegressionAnalysis_Final.R b/sandag_abm/src/main/r/RegressionAnalysis_Final.R deleted file mode 100644 index 830bb36..0000000 --- a/sandag_abm/src/main/r/RegressionAnalysis_Final.R +++ /dev/null @@ -1,953 +0,0 @@ -### -# OBJECTIVE: -# Regression analysis for SANDAG INRIX travel time data -# -# INPUTS: -# travel time SD/ mean travel time in 15 mins (*.dbf) - "inrix2012_nooutliers_traveltime.dbf" (output of inrix outlier analysis) -# inrix to tcoved correspondence (created by Fehr and Peers) - "inrix_2012hwy.txt" -# ABM assignment results by 5 time periods - "hwyload_EA.csv", "hwyload_AM.csv", "hwyload_MD.csv", "hwyload_PM.csv", "hwyload_EV.csv" -# Major interchange distance from SANDAG ABM output folder - "MajorInterchangeDistance.csv" -# other inputs (not used in final analysis) - "InterchangeDistance.csv", "LaneChangeDistance.csv", "FreewayRampMeters.csv" - -# DATA: -# IFC in the roadway network database is re-categorized into four facility classes -# Facility Class IFC Description -# Freeway 1 Freeways -# Arterial 2,3 Major arterials, prime arterials -# Ramp 8,9 Local ramps, freeway ramps -# Other 4,5,6,7 Collectors and local streets -# 80% for estimation (regression analysis) and 20% kept for validation -# low sample size for ramps and others - only freeway and arterial estimations are used - -# REGRESSION VARIABLES (final): -# -# Dependent Variable: -# Travel time per mile Std. Dev. per mean travel time in 15 mins time slices -# -# Independent Variables: -# Number of lanes categories (one, two, three, four, five and more) -# Level of service (LOS) -# LOSC+ = (V/C-0.69)* if (V/C>=0.70) -# LOSD+ = (V/C-0.79)* if (V/C>=0.80) -# LOSE+ = (V/C-0.89)* if (V/C>=0.90) -# LOSF_LOW = (V/C-0.99)* if (V/C>=1.00) -# LOSF_MED = (V/C-1.09)* if (V/C>=1.10) -# LOSF_HIGH = (V/C-1.19)* if (V/C>=1.20) -# Speed -# ISPD70 (1 if posted speed is 70mph, 0 otherwise) for Freeways -# Posted speed categories (ISPD<35, ISPD=35, ISPD=40, ISPD=45, ISPD=50, ISPD>50) for Arterials -# Shift variables (BeforeAM.Shift, AfterAM.Shift, BeforePM.Shift, AfterPM.Shift) -# Control type (none, signal, stop, railroad, ramp meter) -# Upstream and downstream major interchange distance - from midpoint of a freeway segment. inverse distances are used in estimation - -# IMPLEMENTATION (in SANDAG ABM): -# -# By facility type and by 5 time periods -# Estimations for freeways and arterials -# Ramp and other facility types are applied with arterial estimation -# Two reliability components -# LOS -# Static (speed, distance to/from interchanges, intersection type etc.) -# Sum of the two reliability components is multiplied by mean travel time and link length -# Variable calculations are automated including distance to/from interchanges -# Reliability fields are added to highway network -# LOS: include only coefficients -# Static: sum of remaining (un)reliability including the intercept -# Skimming: -# Standard deviation is not additive but variance is -# A link (un)reliability = (MSA Cost – MSA Time) -# Skimmed variance (square of link (un)reliability) -# Final skims are square root of the skimmed value - -# OTHER SCRIPTS USED: -# utilfunc.R - -# by: nagendra.dhakar@rsginc.com -# for: SANDAG SHRP C04 - Tolling and Reliability -#---------------------------------------------------- - -library(foreign) -library(stringr) -library(xtable) -library(reshape) -library(XLConnect) -library(descr) -library(Hmisc) -library(data.table) -library(plyr) -library(gtools) -library(vioplot) -library(lattice) -library(grid) -library(timeDate) -library(ggplot2) -library(robustbase) -library(readr) - -# -------------------- 0. SOURCE FILES AND CONFIG SETTINGS ----------------------- - -# workspace and source files -setwd("E:/Projects/Clients/SANDAG") -source("./Data_2015/utilfunc.R") - -# method of outlier removal - outlier analysis (this is used) or confidence interval 10 -OUTLIER=TRUE - -# input files -inrixhwycorresp.file = "./Data_2015/TMC/inrix_2012hwy.txt" -AssignResultsEA.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_EA.csv" -AssignResultsAM.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_AM.csv" -AssignResultsMD.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_MD.csv" -AssignResultsPM.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_PM.csv" -AssignResultsEV.file = "./SandagReliability/ModelData/OldApp_OldPopSyn/AssignmentResults/hwyload_EV.csv" -InterchangeDistance.file = "./Data_2015/InterchangeDistance.csv" -InterchangeDistanceHOV.file = "./Data_2015/InterchangeDistance_HOV.csv" -MajorInterchangeDistance.file = "./Data_2015/MajorInterchangeDistance.csv" -LaneChangeDistance.file = "./Data_2015/LaneChangeDistance.csv" -FreewayRampMeters.file = "./Data_2015/FreewayRampMeters.csv" - -# travel time SD input file -if (OUTLIER) { - StdDev.file = "./Data_2015/inrix2012_nooutliers_traveltime.dbf" -} else { - StdDev.file = "./Data_2015/inrix2012_noCS10_traveltime.dbf" -} - -# outputs directory -outputsDir="./Data_2015/Results" - -# -------------------- 1. LOAD DATA ---------------------- -# load SD file, assignment results, interchange distance etc. -StdDev<- read.dbf(StdDev.file) -AssignResultsEA <- read.csv(AssignResultsEA.file) -AssignResultsAM <- read.csv(AssignResultsAM.file) -AssignResultsMD <- read.csv(AssignResultsMD.file) -AssignResultsPM <- read.csv(AssignResultsPM.file) -AssignResultsEV <- read.csv(AssignResultsEV.file) -InterchangeDistance <- read.csv(InterchangeDistance.file) -InterchangeDistanceHOV <- read.csv(InterchangeDistanceHOV.file) -MajorInterchangeDistance <- read.csv(MajorInterchangeDistance.file) -LaneChangeDistance <- read.csv(LaneChangeDistance.file) # LinkID,Length,LaneIncrease,LaneDrop,DeadEnd,RegionEnd,DownstreamDistance,ihov,BaseThruLanes,DownThruLanes,DownLinks,QueryDown -FreewayRampMeters <- read.csv(FreewayRampMeters.file) # LinkID,Length,IsRampMeter,QueryNode - -# -------------------- 2. SET UP DATA -------------- - -# read and load inrix to hwy correspondence data -readSaveRdata(inrixhwycorresp.file,"inrixhwycorresp") -inrixhwycorresp <- assignLoad(paste0(inrixhwycorresp.file,".Rdata")) - -# add a new field tmc_code without first character ('-' or '+') -inrixhwycorresp[,tmc_code:=substr(TMC,2,nchar(TMC))] - -# remove the segments that are flagged -inrixhwycorresp<-subset(inrixhwycorresp,FLAG==0) - -# get unique tmc_code with variables corresponding to maximum of TMCProp -df.orig <-inrixhwycorresp -df.agg<-aggregate(TMCProp~tmc_code,inrixhwycorresp,max) -df.max <- merge(df.agg, df.orig) - -# Note: there could be cases where one hwcov_id is associated with multiple tmc_code - -# facility type - different capacity fields for mid-link capacity (period fields) -tmc_ifc <-df.max[,c("tmc_code","IFC","HWYCOV_ID","NM","LENGTH","ISPD","ABLNO","ABLNA","ABLNP", - "BALNO","BALNA","BALNP","ABCPO","ABCPA","ABCPP","BACPO","BACPA","BACPP", - "ABCXO","ABCXA","ABCXP","BACXO","BACXA","BACXP","ABCNT","BACNT", - "ABTL","ABRL","ABLL","BATL","BARL","BALL","ABGC","BAGC","IHOV","ABAU","BAAU"),] - -tmc_ifc$LENGTH<-as.numeric(gsub(",","",tmc_ifc$LENGTH)) -tmc_ifc$HWYCOV_ID<-as.numeric(gsub(",","",tmc_ifc$HWYCOV_ID)) - -# merge attributes to INRIX data -StdDev<-merge(StdDev,tmc_ifc,by="tmc_code") - -# ---------------------------------------------------- - -# write to file -if (FALSE) { - write.table(tmc_ifc,"./Data_2015/tmc_hwycov_atrributes.csv",sep=",",row.names=F,quote=F) -} - -# ID1 in the results is the same as HWYCOV_ID in the link file -AssignResultsEA <- AssignResultsEA[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] -AssignResultsAM <- AssignResultsAM[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] -AssignResultsMD <- AssignResultsMD[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] -AssignResultsPM <- AssignResultsPM[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] -AssignResultsEV <- AssignResultsEV[,c("ID1","AB_Time","BA_Time","AB_Flow","BA_Flow","AB_Speed","BA_Speed")] - -# use 80% for estimation and 20% for validation - sample segments by facility type - -# create new facility type - freeways, arterials, ramps, and others -StdDev$IFC_Est <- ifelse(StdDev$IFC==1,1,0) # freeways -StdDev$IFC_Est <- ifelse(StdDev$IFC==2 | StdDev$IFC==3,2,StdDev$IFC_Est) # arterials - major and prime -StdDev$IFC_Est <- ifelse(StdDev$IFC==8 | StdDev$IFC==9,3,StdDev$IFC_Est) # ramps - local ramps and freeways ramps -StdDev$IFC_Est <- ifelse(StdDev$IFC>=4 & StdDev$IFC<=7,4,StdDev$IFC_Est) # others - collectors and local streets - -Sample.Rate<-0.8 - -# Freeways -StdDev.Freeways <- subset(StdDev, IFC_Est==1) -StdDev.Freeways.Seg <- unique(StdDev.Freeways$tmc_code) -StdDev.Freeways.Seg<-as.data.frame(StdDev.Freeways.Seg) - -sample_size <- floor(Sample.Rate *nrow(StdDev.Freeways.Seg)) -set.seed(123) -Est.Ind <- sample(seq_len(nrow(StdDev.Freeways.Seg)), size = sample_size) -StdDev.Freeways.Seg.Est <- StdDev.Freeways.Seg[Est.Ind,] -StdDev.Freeways.Seg.Val <- StdDev.Freeways.Seg[-Est.Ind,] - -# Arterials -StdDev.Arterials <- subset(StdDev, IFC_Est==2) -StdDev.Arterials.Seg <- unique(StdDev.Arterials$tmc_code) -StdDev.Arterials.Seg<-as.data.frame(StdDev.Arterials.Seg) - -sample_size <- floor(Sample.Rate *nrow(StdDev.Arterials.Seg)) -set.seed(123) -Est.Ind <- sample(seq_len(nrow(StdDev.Arterials.Seg)), size = sample_size) -StdDev.Arterials.Seg.Est <- StdDev.Arterials.Seg[Est.Ind,] -StdDev.Arterials.Seg.Val <- StdDev.Arterials.Seg[-Est.Ind,] - -# Ramps -StdDev.Ramps <- subset(StdDev, IFC_Est==3) -StdDev.Ramps.Seg <- unique(StdDev.Ramps$tmc_code) -StdDev.Ramps.Seg<-as.data.frame(StdDev.Ramps.Seg) - -sample_size <- floor(Sample.Rate *nrow(StdDev.Ramps.Seg)) -set.seed(123) -Est.Ind <- sample(seq_len(nrow(StdDev.Ramps.Seg)), size = sample_size) -StdDev.Ramps.Seg.Est <- StdDev.Ramps.Seg[Est.Ind,] -StdDev.Ramps.Seg.Val <- StdDev.Ramps.Seg[-Est.Ind,] - -# Others -StdDev.Others <- subset(StdDev, IFC_Est==4) -StdDev.Others.Seg <- unique(StdDev.Others$tmc_code) -StdDev.Others.Seg<-as.data.frame(StdDev.Others.Seg) - -sample_size <- floor(Sample.Rate *nrow(StdDev.Others.Seg)) -set.seed(123) -Est.Ind <- sample(seq_len(nrow(StdDev.Others.Seg)), size = sample_size) -StdDev.Others.Seg.Est <- StdDev.Others.Seg[Est.Ind,] -StdDev.Others.Seg.Val <- StdDev.Others.Seg[-Est.Ind,] - -# ------------------ Estimation Dataset -------------------------------------- - -StdDev.Freeways.Seg.Est<-as.data.frame(StdDev.Freeways.Seg.Est) -StdDev.Arterials.Seg.Est<-as.data.frame(StdDev.Arterials.Seg.Est) -StdDev.Ramps.Seg.Est<-as.data.frame(StdDev.Ramps.Seg.Est) -StdDev.Others.Seg.Est<-as.data.frame(StdDev.Others.Seg.Est) - -setnames(StdDev.Freeways.Seg.Est,"StdDev.Freeways.Seg.Est","tmc_code") -setnames(StdDev.Arterials.Seg.Est,"StdDev.Arterials.Seg.Est","tmc_code") -setnames(StdDev.Ramps.Seg.Est,"StdDev.Ramps.Seg.Est","tmc_code") -setnames(StdDev.Others.Seg.Est,"StdDev.Others.Seg.Est","tmc_code") - -# combine dataframes into one -StdDev.Seg.Est <- do.call(rbind,list(StdDev.Freeways.Seg.Est,StdDev.Arterials.Seg.Est,StdDev.Ramps.Seg.Est,StdDev.Others.Seg.Est)) - -# merge data -StdDev.Est<-merge(StdDev,StdDev.Seg.Est,by="tmc_code") - -# ---------------------------- Validation Dataset ------------------------------ - -StdDev.Freeways.Seg.Val<-as.data.frame(StdDev.Freeways.Seg.Val) -StdDev.Arterials.Seg.Val<-as.data.frame(StdDev.Arterials.Seg.Val) -StdDev.Ramps.Seg.Val<-as.data.frame(StdDev.Ramps.Seg.Val) -StdDev.Others.Seg.Val<-as.data.frame(StdDev.Others.Seg.Val) - -setnames(StdDev.Freeways.Seg.Val,"StdDev.Freeways.Seg.Val","tmc_code") -setnames(StdDev.Arterials.Seg.Val,"StdDev.Arterials.Seg.Val","tmc_code") -setnames(StdDev.Ramps.Seg.Val,"StdDev.Ramps.Seg.Val","tmc_code") -setnames(StdDev.Others.Seg.Val,"StdDev.Others.Seg.Val","tmc_code") - -# combine dataframes into one -StdDev.Seg.Val <- do.call(rbind,list(StdDev.Freeways.Seg.Val,StdDev.Arterials.Seg.Val,StdDev.Ramps.Seg.Val,StdDev.Others.Seg.Val)) - -# merge data -StdDev.Val<-merge(StdDev,StdDev.Seg.Val,by="tmc_code") - -# ------------------- Add Interchange Distance ----------------------- -if (FALSE) { - # Interchange distance - InterchangeDistance <-InterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] - InterchangeDistanceHOV <-InterchangeDistanceHOV[,c("LinkID","Length","upstream.distance","downstream.distance")] - - temp<-merge(InterchangeDistance,InterchangeDistanceHOV,by="LinkID",all.x = TRUE) - temp$upstream.distance <- ifelse(!is.na(temp$Length),temp$upstream.distance.y,temp$upstream.distance.x) - temp$downstream.distance<-ifelse(!is.na(temp$Length),temp$downstream.distance.y,temp$downstream.distance.x) - - InterchangeDistance<-temp - InterchangeDistance <-InterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] - - temp<-merge(StdDev.Est,InterchangeDistance,by.x ="HWYCOV_ID",by.y = "LinkID",all.x=TRUE) - - StdDev.Est<-temp -} - -# Major Interchange distance -MajorInterchangeDistance <-MajorInterchangeDistance[,c("LinkID","upstream.distance","downstream.distance")] - -temp<-merge(MajorInterchangeDistance,InterchangeDistanceHOV,by="LinkID",all.x = TRUE) -temp$majorupstream.distance <- ifelse(!is.na(temp$Length),temp$upstream.distance.y,temp$upstream.distance.x) -temp$majordownstream.distance<-ifelse(!is.na(temp$Length),temp$downstream.distance.y,temp$downstream.distance.x) - -MajorInterchangeDistance<-temp -MajorInterchangeDistance <-MajorInterchangeDistance[,c("LinkID","majorupstream.distance","majordownstream.distance")] - -temp<-merge(StdDev.Est,MajorInterchangeDistance,by.x ="HWYCOV_ID",by.y = "LinkID",all.x=TRUE) - -StdDev.Est<-temp - -# FF Time (seconds) -StdDev.Est$FF_Time <- as.numeric(StdDev.Est$LENGTH)*(1/5280)*(1/StdDev.Est$ISPD)*60 -#test <- subset(StdDev.Est, is.na(StdDev.Est$FF_Time)) # just to see if there are any NA values - -# model tod -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>0 & StdDev.Est$todcat<=14,'EV','') #3.5 hours -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>14 & StdDev.Est$todcat<=24,'EA',StdDev.Est$tod.model) #2.5 hours -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>24 & StdDev.Est$todcat<=36,'AM',StdDev.Est$tod.model) #3 hours -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>36 & StdDev.Est$todcat<=62,'MD',StdDev.Est$tod.model) #6.5 hours -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>62 & StdDev.Est$todcat<=76,'PM',StdDev.Est$tod.model) #3.5 hours -StdDev.Est$tod.model<-ifelse(StdDev.Est$todcat>76 & StdDev.Est$todcat<=96,'EV',StdDev.Est$tod.model) #5 hours - -# convert to integer - mid link (*CPO, *CPA, *CPP) and intersection (*CXO, *CXA, *CXP) capacities -StdDev.Est$ABCPO<-as.integer(gsub(",","",StdDev.Est$ABCPO)) -StdDev.Est$ABCPA<-as.integer(gsub(",","",StdDev.Est$ABCPA)) -StdDev.Est$ABCPP<-as.integer(gsub(",","",StdDev.Est$ABCPP)) -StdDev.Est$BACPO<-as.integer(gsub(",","",StdDev.Est$BACPO)) -StdDev.Est$BACPA<-as.integer(gsub(",","",StdDev.Est$BACPA)) -StdDev.Est$BACPP<-as.integer(gsub(",","",StdDev.Est$BACPP)) -StdDev.Est$ABCXO<-as.integer(gsub(",","",StdDev.Est$ABCXO)) -StdDev.Est$ABCXA<-as.integer(gsub(",","",StdDev.Est$ABCXA)) -StdDev.Est$ABCXP<-as.integer(gsub(",","",StdDev.Est$ABCXP)) -StdDev.Est$BACXO<-as.integer(gsub(",","",StdDev.Est$BACXO)) -StdDev.Est$BACXA<-as.integer(gsub(",","",StdDev.Est$BACXA)) -StdDev.Est$BACXP<-as.integer(gsub(",","",StdDev.Est$BACXP)) - -# set initial values to 0 -StdDev.Est$NumLanes <-0 -StdDev.Est$ICNT <-0 # intersection control -StdDev.Est$Flow <-0 -StdDev.Est$CAP.MidLink <-0 # mid-link capacity for freeways and ramps -StdDev.Est$CAP.IntAppr <-0 # intersection-approach capacity for arterials and others -StdDev.Est$AuxLanes <-0 -StdDev.Est$ThruLanes <-0 -StdDev.Est$LeftLanes <-0 -StdDev.Est$RightLanes <-0 -StdDev.Est$GCRatio <-0 -Default.Cap<-9999999 # set a very high - -# TMC Codes -# External (between interchanges): '+' (NB or WB - positive direction), '-' (SB or EB - negative direction) -# Internal (within interchanges): 'P' (NB or WB), 'N' (SB or EB) - -# -------------------------------- ESTIMATION --------------------------------- -# This section estimates regression equations using estimation dataset (StdDev.Est) - -# the below is how SANDAG ABM (gisdk) converts capacities from three periods to 5 model periods: -# set capacity fields -# tod_fld ={{"ABCP_EA"},{"ABCP_AM"},{"ABCP_MD"},{"ABCP_PM"},{"ABCP_EV"}, //BA link capacity -# {"BACP_EA"},{"BACP_AM"},{"BACP_MD"},{"BACP_PM"},{"BACP_EV"}, //AB link capacity -# {"ABCX_EA"},{"ABCX_AM"},{"ABCX_MD"},{"ABCX_PM"},{"ABCX_EV"}, //BA intersection capacity -# {"BACX_EA"},{"BACX_AM"},{"BACX_MD"},{"BACX_PM"},{"BACX_EV"}} //AB intersection capacity - -# org_fld ={"ABCPO","ABCPA","ABCPO","ABCPP","ABCPO", -# "BACPO","BACPA","BACPO","BACPP","BACPO", -# "ABCXO","ABCXA","ABCXO","ABCXP","ABCXO", -# "BACXO","BACXA","BACXO","BACXP","BACXO"} - -# factor ={"3/12","1","6.5/12","3.5/3","8/12", -# "3/12","1","6.5/12","3.5/3","8/12", -# "3/12","1","6.5/12","3.5/3","8/12", -# "3/12","1","6.5/12","3.5/3","8/12"} -# -# the capacity calculations below are consistent with the ABM gisdk - -# MERGE assignment results to the estimation data - -setnames(StdDev.Est,"HWYCOV_ID","ID1") -nrow(StdDev.Est) - -# EA Period -temp2 <- merge(StdDev.Est,AssignResultsEA,by="ID1", all.x=TRUE) -temp2$Flow <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) -temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*3/12,temp2$ABCPO*3/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*3/12,temp2$BACPO*3/12)),temp2$CAP.MidLink)) -temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*3/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*3/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*3/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*3/12,Default.Cap))),temp2$CAP.IntAppr)) -temp2$NumLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) -temp2$ICNT <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) -temp2$AuxLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) -temp2$ThruLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) -temp2$LeftLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) -temp2$RightLanes <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) -temp2$GCRatio <- ifelse(temp2$tod.model=='EA', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) - -temp2<-temp2[,-c(61:66)] -nrow(temp2) - -# AM Period -temp2 <- merge(temp2,AssignResultsAM,by="ID1", all.x=TRUE) -temp2$Flow <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) -temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPA,temp2$ABCPA),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPA,temp2$BACPA)),temp2$CAP.MidLink)) # AM capacity is for 3 hours -temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXA,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXA,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXA,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXA,Default.Cap))),temp2$CAP.IntAppr)) -temp2$NumLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNA,temp2$ABLNA),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNA,temp2$BALNA)),temp2$NumLanes) -temp2$ICNT <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) -temp2$AuxLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) -temp2$ThruLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) -temp2$LeftLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) -temp2$RightLanes <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) -temp2$GCRatio <- ifelse(temp2$tod.model=='AM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) - -temp2<-temp2[,-c(61:66)] -nrow(temp2) - -# MD Period -temp2 <- merge(temp2,AssignResultsMD,by="ID1", all.x=TRUE) -temp2$Flow <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # the flow is for 9 am - 3:30 pm -temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*6.5/12,temp2$ABCPO*6.5/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*6.5/12,temp2$BACPO*6.5/12)),temp2$CAP.MidLink)) -temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*6.5/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*6.5/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*6.5/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*6.5/12,Default.Cap))),temp2$CAP.IntAppr)) -temp2$NumLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) -temp2$ICNT <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) -temp2$AuxLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) -temp2$ThruLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) -temp2$LeftLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) -temp2$RightLanes <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) -temp2$GCRatio <- ifelse(temp2$tod.model=='MD', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) - -temp2<-temp2[,-c(61:66)] -nrow(temp2) - -# PM Period -temp2 <- merge(temp2,AssignResultsPM,by="ID1", all.x=TRUE) -temp2$Flow <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # flow is from 3:30 pm to 7 pm -temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPP*3.5/3,temp2$ABCPP*3.5/3),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPP*3.5/3,temp2$BACPP*3.5/3)),temp2$CAP.MidLink)) # PM capacity is for 3 hours -temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXP*3.5/3,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXP*3.5/3,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXP*3.5/3,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXP*3.5/3,Default.Cap))),temp2$CAP.IntAppr)) -temp2$NumLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNP,temp2$ABLNP),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNP,temp2$BALNP)),temp2$NumLanes) -temp2$ICNT <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) -temp2$AuxLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) -temp2$ThruLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) -temp2$LeftLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) -temp2$RightLanes <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) -temp2$GCRatio <- ifelse(temp2$tod.model=='PM', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) - -temp2<-temp2[,-c(61:66)] -nrow(temp2) - -# EV Period -temp2 <- merge(temp2,AssignResultsPM,by="ID1", all.x=TRUE) -temp2$Flow <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BA_Flow,temp2$AB_Flow),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$AB_Flow,temp2$BA_Flow)),temp2$Flow) # flow is from 7 pm to 3:30 am -temp2$CAP.MidLink <- (ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code), ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACPO*8/12,temp2$ABCPO*8/12),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCPO*8/12,temp2$BACPO*8/12)),temp2$CAP.MidLink)) # OP capacity is for 18 hours -temp2$CAP.IntAppr <- (ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*8/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*8/12,Default.Cap)),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,ifelse(temp2$BACNT==1,temp2$BACXO*8/12,Default.Cap),ifelse(temp2$ABCNT==1,temp2$ABCXO*8/12,Default.Cap))),temp2$CAP.IntAppr)) -temp2$NumLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALNO,temp2$ABLNO),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLNO,temp2$BALNO)),temp2$NumLanes) -temp2$ICNT <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BACNT,temp2$ABCNT),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABCNT,temp2$BACNT)),temp2$ICNT) -temp2$AuxLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAAU,temp2$ABAU),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABAU,temp2$BAAU)),temp2$AuxLanes) -temp2$ThruLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BATL,temp2$ABTL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABTL,temp2$BATL)),temp2$ThruLanes) -temp2$LeftLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BALL,temp2$ABLL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABLL,temp2$BALL)),temp2$LeftLanes) -temp2$RightLanes <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BARL,temp2$ABRL),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABRL,temp2$BARL)),temp2$RightLanes) -temp2$GCRatio <- ifelse(temp2$tod.model=='EV', ifelse(grepl("\\+",temp2$tmc_code) | grepl("P",temp2$tmc_code),ifelse(is.na(temp2$AB_Flow) | temp2$AB_Flow==0,temp2$BAGC,temp2$ABGC),ifelse(is.na(temp2$BA_Flow) | temp2$BA_Flow==0,temp2$ABGC,temp2$BAGC)),temp2$GCRatio) - -temp2<-temp2[,-c(61:66)] - -nrow(temp2) - -# intersection approach capacity, when not available set to 0 -#temp2$CAP.IntAppr<-ifelse(temp2$CAP.IntAppr==9999999,0,temp2$CAP.IntAppr) -temp2$CAP.IntAppr<-ifelse(temp2$CAP.IntAppr==9999999,temp2$CAP.MidLink,temp2$CAP.IntAppr) - -# values of greater and equal to 7 are for non-availability for some reasons, set them to 0 -temp2$ThruLanes <-ifelse(temp2$ThruLanes>=7,0,temp2$ThruLanes) -temp2$LeftLanes <-ifelse(temp2$LeftLanes>=7,0,temp2$LeftLanes) -temp2$RightLanes <-ifelse(temp2$RightLanes>=7,0,temp2$RightLanes) - -# recalculate GC ratio as per createhwynet.rsc (line 1045) -temp2$GCRatio<-ifelse(temp2$GCRatio>10,temp2$GCRatio/100,temp2$GCRatio) -temp2$GCRatio<-ifelse(temp2$GCRatio>1,1,temp2$GCRatio) - -# ----------------------- Create Variables for Regression Models ----------------------------- -EstDataSet<-temp2 - -# NOTE: Link ID (HWYCOV_ID)=29412 29413 31194 31202,40479 are not in the model network. So remove those for now -EstDataSet<-subset(EstDataSet, !is.na(EstDataSet$Flow)) - -# Capacity by facility type (12/07/2015) -# freeways and ramps - mid link capacity -# arterials and others - intersection approach -#EstDataSet$CAP<-ifelse(EstDataSet$IFC_Est==1 | EstDataSet$IFC_Est==3, EstDataSet$CAP.MidLink, EstDataSet$CAP.IntAppr) - -EstDataSet$CAP<-EstDataSet$CAP.MidLink # mid link capacity for all -EstDataSet$VOC <- EstDataSet$Flow/EstDataSet$CAP - -# control type - 0-none, 1-signal, 2-stop, 3-railroad -EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT==1,"Signal","None") # signal -EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT==2 | EstDataSet$ICNT==3,"Stop",EstDataSet$ICNT.Est) # all-way and two way stop -EstDataSet$ICNT.Est <-ifelse(EstDataSet$ICNT>3,"RailRoad",EstDataSet$ICNT.Est) # other-primarily rail-road crossing -#set the order of factors -EstDataSet$ICNT.Est<-factor(EstDataSet$ICNT.Est, levels = c("None","Signal", "Stop", "RailRoad")) - -# for descriptives -EstDataSet$ICNT.Signal <-ifelse(EstDataSet$ICNT==1,1,0) # signal -EstDataSet$ICNT.Stop <-ifelse(EstDataSet$ICNT==2 | EstDataSet$ICNT==3,1,0) # all-way and two way stop -EstDataSet$ICNT.RailRoad <-ifelse(EstDataSet$ICNT>3,1,0) # other-primarily rail-road crossing -EstDataSet$ICNT.RampMeter <-ifelse(EstDataSet$ICNT==4 | EstDataSet$ICNT==5,1,0) # Ramp meter, ramp meter with HOV bypass (12/07/2015) - -# Major and Minor Arterial - this would be used only for arterials -EstDataSet$MajorArterial <-ifelse(EstDataSet$IFC==2,1,0) - -# I-15 (managed lanes) and SR-125 -EstDataSet$I15<-ifelse(EstDataSet$IFC==1 & EstDataSet$IHOV==2 & str_sub(EstDataSet$NM,1,4)=="I-15",1,0) # I15 -#EstDataSet$SR125<-ifelse(EstDataSet$IFC==1 & EstDataSet$ihov==4,1,0) # No SR125 facility in the dataset - -# LOS variables - additive and multiplicative (12/07/2015) -EstDataSet$LOSC.Up <- ifelse(EstDataSet$VOC>=0.70,EstDataSet$VOC-0.69,0) # LOS C+ -EstDataSet$LOSD.Up <- ifelse(EstDataSet$VOC>=0.80,EstDataSet$VOC-0.79,0) # LOS D+ -EstDataSet$LOSE.Up <- ifelse(EstDataSet$VOC>=0.90,EstDataSet$VOC-0.89,0) # LOS E+ -EstDataSet$LOSF.Low.Up <- ifelse(EstDataSet$VOC>=1.00,EstDataSet$VOC-0.99,0) # LOS F Low+ -EstDataSet$LOSF.Med.Up <- ifelse(EstDataSet$VOC>=1.10,EstDataSet$VOC-1.09,0) # LOS F Med+ -EstDataSet$LOSF.High.Up <- ifelse(EstDataSet$VOC>=1.20,EstDataSet$VOC-1.19,0) # LOS F High+ - -# LOS variables - additive and multiplicative - capping VOC -#EstDataSet$LOSC.Up <- ifelse(EstDataSet$VOC>=0.70 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.69,0) # LOS C+ -#EstDataSet$LOSD.Up <- ifelse(EstDataSet$VOC>=0.80 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.79,0) # LOS D+ -#EstDataSet$LOSE.Up <- ifelse(EstDataSet$VOC>=0.90 & EstDataSet$VOC<=1.00,EstDataSet$VOC-0.89,0) # LOS E+ -#EstDataSet$LOSF.Low.Up <- ifelse(EstDataSet$VOC>=1.00,EstDataSet$VOC-0.99,0) # LOS F Low+ -#EstDataSet$LOSF.Med.Up <- ifelse(EstDataSet$VOC>=1.10,EstDataSet$VOC-1.09,0) # LOS F Med+ -#EstDataSet$LOSF.High.Up <- ifelse(EstDataSet$VOC>=1.20,EstDataSet$VOC-1.19,0) # LOS F High+ - -# LOS categories - for descriptives only -EstDataSet$LOS.Cat <-"LOSB" -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.70 & EstDataSet$VOC<0.80,"LOSC",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.80 & EstDataSet$VOC<0.90,"LOSD",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=0.90 & EstDataSet$VOC<1.00,"LOSE",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.00 & EstDataSet$VOC<1.10,"LOSF.Low",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.10 & EstDataSet$VOC<1.20,"LOSF.Med",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat <- ifelse(EstDataSet$VOC>=1.20,"LOSF.High",EstDataSet$LOS.Cat) -EstDataSet$LOS.Cat<-factor(EstDataSet$LOS.Cat, levels = c("LOSB","LOSC","LOSD", "LOSE", "LOSF.Low","LOSF.Med","LOSF.High")) - -# NumLanes -EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==1,"OneLane","NoLane") -EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==2,"TwoLane",EstDataSet$NumLanesCat) -EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==3,"ThreeLane",EstDataSet$NumLanesCat) -EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes==4,"FourLanes",EstDataSet$NumLanesCat) -EstDataSet$NumLanesCat <- ifelse(EstDataSet$NumLanes>=5,"FiveLanes+",EstDataSet$NumLanesCat) -#set order of factors -EstDataSet$NumLanesCat<-factor(EstDataSet$NumLanesCat, levels = c("NoLane","OneLane", "TwoLane", "ThreeLane","FourLanes", "FiveLanes+")) - -# for descriptives -EstDataSet$OneLane <- ifelse(EstDataSet$NumLanes==1,1,0) -EstDataSet$TwoLane <- ifelse(EstDataSet$NumLanes==2,1,0) -EstDataSet$ThreeLane <- ifelse(EstDataSet$NumLanes==3,1,0) -EstDataSet$FourLane <- ifelse(EstDataSet$NumLanes==4,1,0) -EstDataSet$FiveMoreLane <- ifelse(EstDataSet$NumLanes>=5,1,0) - -# calculate reference bins (shift variables) - two peaks (AM and PM) and one low (MD) - generic, for all facility types -#Pre-AM: AM Peak to start of day -#Post-AM: AM Peak to MD low -#Pre-PM: PM Peak to MD low (backward) -#Post-PM: PM Peak to end of day -if (TRUE) { - EstDataSet.mean.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") - EstDataSet.mean.mean<-aggregate(EstDataSet$mean,by=list(EstDataSet$todcat),FUN=mean) - - temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=25 & EstDataSet.mean.mean$Group.1<=36) - Peak_AM <- temp[which.max(temp$x),1] #32 - - temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=37 & EstDataSet.mean.mean$Group.1<=62) - Low_MD <- temp[which.min(temp$x),1] #41 - - temp<-subset(EstDataSet.mean.mean,EstDataSet.mean.mean$Group.1>=63 & EstDataSet.mean.mean$Group.1<=76) - Peak_PM <- temp[which.max(temp$x),1] #70 - - # calculate before and after variables for AM and PM periods - EstDataSet$BeforeAM <-ifelse(EstDataSet$todcat<=Peak_AM,Peak_AM-EstDataSet$todcat,99) - EstDataSet$BeforePM <-ifelse(EstDataSet$todcat>=Low_MD & EstDataSet$todcat <= Peak_PM,Peak_PM-EstDataSet$todcat,99) - - EstDataSet$AfterAM <-ifelse(EstDataSet$todcat>=Peak_AM & EstDataSet$todcat=Peak_PM,EstDataSet$todcat-Peak_PM,99) - -} - -EstDataSet$IsBeforeAM<- ifelse(EstDataSet$BeforeAM<99,1,0) -EstDataSet$IsAfterAM<- ifelse(EstDataSet$AfterAM<99,1,0) -EstDataSet$IsBeforePM<- ifelse(EstDataSet$BeforePM<99,1,0) -EstDataSet$IsAfterPM<- ifelse(EstDataSet$AfterPM<99,1,0) - -# apply shift -EstDataSet$IsBeforeAM.Shift<- EstDataSet$IsBeforeAM*EstDataSet$BeforeAM -EstDataSet$IsAfterAM.Shift<- EstDataSet$IsAfterAM*EstDataSet$AfterAM -EstDataSet$IsBeforePM.Shift<- EstDataSet$IsBeforePM*EstDataSet$BeforePM -EstDataSet$IsAfterPM.Shift<- EstDataSet$IsAfterPM*EstDataSet$AfterPM - -# piecewise functions for shift variables -# Before AM: 32 (Peak_AM) to 29, 29 to 26, 26 to 20, 20 to 1 -# After AM: 32 to 36, 36 to 39, 39 to 41 (Low_MD) -# Before PM: 70 (Peak_PM) to 66, 66 to 62, 62 to 58, 58 to 41 (Low_MD) -# After PM: 70 (Peak_PM) to 71, 71 to 79, 79 to 96 - -EstDataSet$BeforeAM.Step1<-ifelse(EstDataSet$IsBeforeAM==1, EstDataSet$BeforeAM,0) -EstDataSet$BeforeAM.Step2<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<29,EstDataSet$BeforeAM-(Peak_AM-29),0) -EstDataSet$BeforeAM.Step3<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<26,EstDataSet$BeforeAM-(Peak_AM-26),0) -EstDataSet$BeforeAM.Step4<-ifelse(EstDataSet$IsBeforeAM==1 & EstDataSet$todcat<20,EstDataSet$BeforeAM-(Peak_AM-20),0) - -EstDataSet$AfterAM.Step1<-ifelse(EstDataSet$IsAfterAM==1, EstDataSet$AfterAM,0) -EstDataSet$AfterAM.Step2<-ifelse(EstDataSet$IsAfterAM==1 & EstDataSet$todcat>36,EstDataSet$AfterAM-(36-Peak_AM),0) -EstDataSet$AfterAM.Step3<-ifelse(EstDataSet$IsAfterAM==1 & EstDataSet$todcat>39,EstDataSet$AfterAM-(39-Peak_AM),0) - -EstDataSet$BeforePM.Step1<-ifelse(EstDataSet$IsBeforePM==1, EstDataSet$BeforePM,0) -EstDataSet$BeforePM.Step2<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<66,EstDataSet$BeforePM-(Peak_PM-66),0) -EstDataSet$BeforePM.Step3<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<62,EstDataSet$BeforePM-(Peak_PM-62),0) -EstDataSet$BeforePM.Step4<-ifelse(EstDataSet$IsBeforePM==1 & EstDataSet$todcat<58,EstDataSet$BeforePM-(Peak_PM-58),0) - -EstDataSet$AfterPM.Step1<-ifelse(EstDataSet$IsAfterPM==1, EstDataSet$AfterPM,0) -EstDataSet$AfterPM.Step2<-ifelse(EstDataSet$IsAfterPM==1 & EstDataSet$todcat>71,EstDataSet$AfterPM-(71-Peak_PM),0) -EstDataSet$AfterPM.Step3<-ifelse(EstDataSet$IsAfterPM==1 & EstDataSet$todcat>79,EstDataSet$AfterPM-(79-Peak_PM),0) - -if (TRUE){ - # calculate mean SD in time slices for the four variables - BeforeAM.sd.mean<-cast(EstDataSet,BeforeAM~IFC_Est,mean,value="sd") - BeforePM.sd.mean<-cast(EstDataSet,BeforePM~IFC_Est,mean,value="sd") - AfterAM.sd.mean<-cast(EstDataSet,AfterAM~IFC_Est,mean,value="sd") - AfterPM.sd.mean<-cast(EstDataSet,AfterPM~IFC_Est,mean,value="sd") - EstDataSet.sd.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="sd") - - setnames(BeforeAM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) - setnames(BeforePM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) - setnames(AfterAM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) - setnames(AfterPM.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) - setnames(EstDataSet.sd.mean, c("1","2","3","4"), c("Freeways.SD","Arterials.SD","Ramps.SD", "Others.SD")) - - if (OUTLIER) { - # outlier method - write.table(BeforeAM.sd.mean,"BeforeAM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(BeforePM.sd.mean,"BeforePM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(AfterAM.sd.mean,"AfterAM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(AfterPM.sd.mean,"AfterPM_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(EstDataSet.sd.mean,"EstDataSet_SDMean_nooutlier.csv",sep = ",",row.names = FALSE) - - } else { - # confidence score 10 method - write.table(BeforeAM.sd.mean,"BeforeAM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(BeforePM.sd.mean,"BeforePM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(AfterAM.sd.mean,"AfterAM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(AfterPM.sd.mean,"AfterPM_SDMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(EstDataSet.sd.mean,"EstDataSet_SDMean_noCS10.csv",sep = ",",row.names = FALSE) - - } -} - -if (TRUE){ - # calculate mean SD in time slices for the four variables - BeforeAM.mean.mean<-cast(EstDataSet,BeforeAM~IFC_Est,mean,value="mean") - BeforePM.mean.mean<-cast(EstDataSet,BeforePM~IFC_Est,mean,value="mean") - AfterAM.mean.mean<-cast(EstDataSet,AfterAM~IFC_Est,mean,value="mean") - AfterPM.mean.mean<-cast(EstDataSet,AfterPM~IFC_Est,mean,value="mean") - EstDataSet.mean.mean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") - - setnames(BeforeAM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) - setnames(BeforePM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) - setnames(AfterAM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) - setnames(AfterPM.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) - setnames(EstDataSet.mean.mean, c("1","2","3","4"), c("Freeways.Mean","Arterials.Mean","Ramps.Mean", "Others.Mean")) - - if (OUTLIER) { - # outlier method - write.table(BeforeAM.mean.mean,"BeforeAM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(BeforePM.mean.mean,"BeforePM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(AfterAM.mean.mean,"AfterAM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(AfterPM.mean.mean,"AfterPM_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) - write.table(EstDataSet.mean.mean,"EstDataSet_MeanMean_nooutlier.csv",sep = ",",row.names = FALSE) - - } else { - # confidence score 10 method - write.table(BeforeAM.mean.mean,"BeforeAM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(BeforePM.mean.mean,"BeforePM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(AfterAM.mean.mean,"AfterAM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(AfterPM.mean.mean,"AfterPM_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) - write.table(EstDataSet.mean.mean,"EstDataSet_MeanMean_noCS10.csv",sep = ",",row.names = FALSE) - - } -} - -# InterchangeDistance -#EstDataSet$Upstream <- EstDataSet$upstream.distance -#EstDataSet$Downstream <- EstDataSet$downstream.distance -EstDataSet$MajorUpstream <- EstDataSet$majorupstream.distance -EstDataSet$MajorDownstream <- EstDataSet$majordownstream.distance - -if (FALSE){ - EstDataSet$AllInt.UpDist<-ifelse(EstDataSet$upstream.distance<0.5,"Short","Long") - EstDataSet$AllInt.UpDist<-ifelse(EstDataSet$upstream.distance>=0.5 & EstDataSet$upstream.distance<=2,"Med",EstDataSet$AllInt.UpDist) - EstDataSet$AllInt.DownDist<-ifelse(EstDataSet$downstream.distance<0.5,"Short","Long") - EstDataSet$AllInt.DownDist<-ifelse(EstDataSet$downstream.distance>=0.5 & EstDataSet$downstream.distance<=2,"Med",EstDataSet$AllInt.DownDist) - - EstDataSet$AllInt.UpDist<-factor(EstDataSet$AllInt.UpDist, levels = c("Short","Med", "Long")) - EstDataSet$AllInt.DownDist<-factor(EstDataSet$AllInt.DownDist, levels = c("Short","Med", "Long")) - - EstDataSet$MajInt.UpDist<-ifelse(EstDataSet$majorupstream.distance<0.5,"Short","Long") - EstDataSet$MajInt.UpDist<-ifelse(EstDataSet$majorupstream.distance>=0.5 & EstDataSet$majorupstream.distance<=2,"Med",EstDataSet$MajInt.UpDist) - EstDataSet$MajInt.DownDist<-ifelse(EstDataSet$majordownstream.distance<0.5,"Short","Long") - EstDataSet$MajInt.DownDist<-ifelse(EstDataSet$majordownstream.distance>=0.5 & EstDataSet$majordownstream.distance<=2,"Med",EstDataSet$MajInt.DownDist) - - EstDataSet$MajInt.UpDist<-factor(EstDataSet$MajInt.UpDist, levels = c("Short","Med", "Long")) - EstDataSet$MajInt.DownDist<-factor(EstDataSet$MajInt.DownDist, levels = c("Short","Med", "Long")) -} - -EstDataSet$MajorUpstream.Inverse <- (1/EstDataSet$MajorUpstream) -EstDataSet$MajorDownstream.Inverse <- (1/EstDataSet$MajorDownstream) - -EstDataSet$AuxLanesBinary <- ifelse(EstDataSet$AuxLanes>0,1,0) -EstDataSet$ISPD70 <- ifelse(EstDataSet$ISPD==70,1,0) - -# subset by facility type - four for now (freeways, arterials, ramps, and others) -EstDataSet.freeways <- subset(EstDataSet,IFC_Est==1) -EstDataSet.arterials <- subset(EstDataSet,IFC_Est==2) -EstDataSet.ramps <- subset(EstDataSet,IFC_Est==3) -EstDataSet.others <- subset(EstDataSet,IFC_Est==4) - -# ----------------------------- 1.FREEWAYS ----------------------------- -# remove 1-lane freeways -EstDataSet.freeways <- subset(EstDataSet.freeways,NumLanes>1) - -# add lane change distance (12/7/2015) -LaneChangeDistance <-LaneChangeDistance[,c("LinkID","LaneIncrease","LaneDrop","DeadEnd","RegionEnd","DownstreamDistance")] -temp<-merge(EstDataSet.freeways,LaneChangeDistance,by.x ="ID1",by.y = "LinkID",all.x=TRUE) - -EstDataSet.freeways<-temp -EstDataSet.freeways$Down.Lane.Increase<-0 -EstDataSet.freeways$Down.Lane.Drop<-0 - -EstDataSet.freeways$Down.Lane.Increase<-ifelse(!is.na(EstDataSet.freeways$LaneIncrease) & EstDataSet.freeways$LaneIncrease>0,EstDataSet.freeways$DownstreamDistance,0) -EstDataSet.freeways$Down.Lane.Drop<-ifelse(!is.na(EstDataSet.freeways$LaneDrop) & EstDataSet.freeways$LaneDrop>0,EstDataSet.freeways$DownstreamDistance,0) - -EstDataSet.freeways$Down.Lane.Increase.Inv<-ifelse(!is.na(EstDataSet.freeways$LaneIncrease) & EstDataSet.freeways$LaneIncrease>0,1/EstDataSet.freeways$DownstreamDistance,0) -EstDataSet.freeways$Down.Lane.Drop.Inv<-ifelse(!is.na(EstDataSet.freeways$LaneDrop) & EstDataSet.freeways$LaneDrop>0,1/EstDataSet.freeways$DownstreamDistance,0) - -# add upstream/downstream node ramp meter (12/16/2015) - only within the periods of 7am-9am and 4pm-6pm -FreewayRampMeters <-FreewayRampMeters[,c("LinkID","IsRampMeterUp","IsRampMeterDown")] -temp<-merge(EstDataSet.freeways,FreewayRampMeters,by.x ="ID1",by.y = "LinkID",all.x=TRUE) -EstDataSet.freeways<-temp - -EstDataSet.freeways$DownNode.RampMeter<-0 -EstDataSet.freeways$DownNode.RampMeter<-ifelse(!is.na(EstDataSet.freeways$IsRampMeterDown) & EstDataSet.freeways$IsRampMeterDown==1 - & ((EstDataSet.freeways$todcat>=28 & EstDataSet.freeways$todcat<=36) - | (EstDataSet.freeways$todcat>=64 & EstDataSet.freeways$todcat<=72)),1,0) - -# Regression model -if (FALSE) { - model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up+ISPD70+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3 - +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) -} - -# significant - remove ISPD70 -model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.High.Up+ISPD70+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3 - +AfterPM.Step1+AfterPM.Step3 - +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) - -# no shift vars -model.freeways = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.High.Up+ISPD70 - +MajorUpstream.Inverse+MajorDownstream.Inverse, data=EstDataSet.freeways) - - -summary(model.freeways) - -# ---------------------------- 2.ARTERIALS ----------------------------- -# Arterials - segment speed (12/7/2015) -EstDataSet.arterials$ISPD.Cat<-"" -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD<35,"ISPD35Less",EstDataSet.arterials$ISPD.Cat) -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==35,"ISPD35",EstDataSet.arterials$ISPD.Cat) -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==40,"ISPD40",EstDataSet.arterials$ISPD.Cat) -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==45,"ISPD45",EstDataSet.arterials$ISPD.Cat) -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD==50,"ISPD50",EstDataSet.arterials$ISPD.Cat) -EstDataSet.arterials$ISPD.Cat <- ifelse(EstDataSet.arterials$ISPD>50,"ISPD50More",EstDataSet.arterials$ISPD.Cat) -#set order of factors -EstDataSet.arterials$ISPD.Cat<-factor(EstDataSet.arterials$ISPD.Cat, levels = c("ISPD35Less","ISPD35", "ISPD40", "ISPD45","ISPD50", "ISPD50More")) - -EstDataSet.arterials<-subset(EstDataSet.arterials,!is.na(EstDataSet.arterials$VOC)) -# Regression model -if (FALSE) { - model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up - +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) -} - -model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSF.Low.Up - +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) -# no shift vars -model.arterials = lm(sdpermean~NumLanesCat+LOSC.Up+LOSF.Low.Up - +ISPD.Cat - +ICNT.Est, data=EstDataSet.arterials) - -# different measures of capacity -model.arterials = lm(sdpermen~GCRatio+RightLanes+LeftLanes - +ISPD.Cat+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.Est, data=EstDataSet.arterials) - -summary(model.arterials) - -# significant - aggregate from lower LOS - -# ---------------------------- 3.RAMPS --------------------------------- -model.ramps = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up - +BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3+ICNT.RampMeter, data=EstDataSet.ramps) - -# only significant - don't include after LOSE.Up -model.ramps = lm(sdpermean~LOSC.Up+LOSE.Up - +BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step4 - +AfterPM.Step1+AfterPM.Step3+ICNT.RampMeter, data=EstDataSet.ramps) - -# no shift vars -model.ramps = lm(sdpermean~LOSC.Up+LOSE.Up - +ICNT.RampMeter, data=EstDataSet.ramps) - - -summary(model.ramps) - -# ---------------------------- 4.OTHERS -------------------------------- -# combine twolane and threelanes for others -EstDataSet.others$NumLanesCat <- ifelse(EstDataSet.others$NumLanes==1,"OneLane","NoLane") -EstDataSet.others$NumLanesCat <- ifelse(EstDataSet.others$NumLanes>=2,"TwoLane+",EstDataSet.others$NumLanesCat) -EstDataSet.others$NumLanesCat<-factor(EstDataSet.others$NumLanesCat, levels = c("NoLane","OneLane", "TwoLane+")) - -EstDataSet.others$OneLane <- ifelse(EstDataSet.others$NumLanes==1,1,0) -EstDataSet.others$TwoLane <- ifelse(EstDataSet.others$NumLanes>=2,1,0) -EstDataSet.others$ThreeLane <- 0 - -# Regression model -model.others = lm(sdpermean~LOSC.Up+LOSD.Up+LOSE.Up+LOSF.Low.Up+LOSF.Med.Up+LOSF.High.Up - +ISPD+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+AfterAM.Step2+AfterAM.Step3+BeforePM.Step1+BeforePM.Step2+BeforePM.Step3+BeforePM.Step4 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3, data=EstDataSet.others) - -model.others = lm(sdpermean~ISPD+BeforeAM.Step1+BeforeAM.Step2+BeforeAM.Step3+BeforeAM.Step4 - +AfterAM.Step1+BeforePM.Step1+BeforePM.Step3 - +AfterPM.Step1+AfterPM.Step2+AfterPM.Step3, data=EstDataSet.others) - -# no shift vars -model.others = lm(sdpermean~ISPD, data=EstDataSet.others) - -# view results -summary(model.others) - -# ---------------------------------- Correlation Matrix ------------------------------------- -library(Hmisc) -library(corrplot) - -# Freeways -# keep only a few variables -temp1<-EstDataSet.freeways[,c("sd","sdpermean", - "ISPD70","VOC","LOSC.Up","LOSD.Up","LOSE.Up","LOSF.Low.Up","LOSF.Med.Up","LOSF.High.Up", - "BeforeAM.Step1","BeforeAM.Step2","BeforeAM.Step3","BeforeAM.Step4","AfterAM.Step1","AfterAM.Step2","AfterAM.Step3", - "BeforePM.Step1","BeforePM.Step2","BeforePM.Step3","BeforePM.Step4","AfterPM.Step1","AfterPM.Step2","AfterPM.Step3", - "MajorUpstream.Inverse","MajorDownstream.Inverse")] -mcor<-rcorr(as.matrix(temp1)) -# make plot and write the correlations to a csv file -corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) -write.table(mcor$r,"correlation_freeways.csv",row.names = FALSE, sep = ",") - -# Arterials -temp2<-EstDataSet.arterials[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ThreeLane","FourLane", - "AuxLanesBinary","ISPD","VOC","LOSC","LOSD","LOSE", - "LOSF_Low","LOSF_Med","LOSF_High","ICNT.Signal","ICNT.Stop","ICNT.RailRoad", - "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] -mcor<-rcorr(as.matrix(temp2)) -# make plot and write the correlations to a csv file -corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) -write.table(mcor$r,"correlation_arterials.csv",row.names = TRUE, sep = ",") - -# Ramps -temp3<-EstDataSet.ramps[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ThreeLane","ISPD","VOC", - "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] -mcor<-rcorr(as.matrix(temp3)) -# make plot and write the correlations to a csv file -corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) -write.table(mcor$r,"correlation_ramps.csv",row.names = TRUE, sep = ",") - -# Others -temp4<-EstDataSet.others[,c("sd","sdpermean","NumLanes","OneLane","TwoLane","ISPD","VOC","LOSC","LOSD","LOSE", - "LOSF_Low","LOSF_Med", - "IsBeforeAM.Shift","IsAfterAM.Shift","IsBeforePM.Shift","IsAfterPM.Shift")] -mcor<-rcorr(as.matrix(temp4)) -# make plot and write the correlations to a csv file -corrplot(mcor$r,type="upper",tl.col="black",tl.srt=45) -write.table(mcor$r,"correlation_others.csv",row.names = TRUE, sep = ",") - -# ----------------------- PLOTS ------------ -p1<-plot(EstDataSet.freeways$VOC,EstDataSet.freeways$sdpermean,type="p") -p2<-plot(EstDataSet.arterials$VOC,EstDataSet.arterials$sdpermean,type="p") -p3<-plot(EstDataSet.ramps$VOC,EstDataSet.ramps$sdpermean,type="p") -p4<-plot(EstDataSet.others$VOC,EstDataSet.others$sdpermean,type="p") - -# Shift variable plot -EstDataSet.sdmean<-cast(EstDataSet,todcat~IFC_Est,mean,value="sd") -EstDataSet.meanmean<-cast(EstDataSet,todcat~IFC_Est,mean,value="mean") -setnames(EstDataSet.sdmean,c("1","2","3","4"),c("Freeways","Arterials","Ramps","Others")) -setnames(EstDataSet.meanmean,c("1","2","3","4"),c("Freeways","Arterials","Ramps","Others")) - -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>0 & EstDataSet.sdmean$todcat<=14,'EV1','') -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>14 & EstDataSet.sdmean$todcat<=24,'EA',EstDataSet.sdmean$tod.model) -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>24 & EstDataSet.sdmean$todcat<=36,'AM',EstDataSet.sdmean$tod.model) -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>36 & EstDataSet.sdmean$todcat<=62,'MD',EstDataSet.sdmean$tod.model) -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>62 & EstDataSet.sdmean$todcat<=76,'PM',EstDataSet.sdmean$tod.model) -EstDataSet.sdmean$tod.model<-ifelse(EstDataSet.sdmean$todcat>76 & EstDataSet.sdmean$todcat<=96,'EV2',EstDataSet.sdmean$tod.model) - -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>0 & EstDataSet.meanmean$todcat<=14,'EV1','') -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>14 & EstDataSet.meanmean$todcat<=24,'EA',EstDataSet.meanmean$tod.model) -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>24 & EstDataSet.meanmean$todcat<=36,'AM',EstDataSet.meanmean$tod.model) -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>36 & EstDataSet.meanmean$todcat<=62,'MD',EstDataSet.meanmean$tod.model) -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>62 & EstDataSet.meanmean$todcat<=76,'PM',EstDataSet.meanmean$tod.model) -EstDataSet.meanmean$tod.model<-ifelse(EstDataSet.meanmean$todcat>76 & EstDataSet.meanmean$todcat<=96,'EV2',EstDataSet.meanmean$tod.model) - -EstDataSet.sdmean$tod.model<-factor(EstDataSet.sdmean$tod.model, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) -EstDataSet.meanmean$tod.model<-factor(EstDataSet.meanmean$tod.model, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) - - -Peak_AM <- 32 -Low_MD <- 60 -Peak_PM <-76 - -EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat<=Peak_AM,1,0) #blue -EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Peak_AM & EstDataSet.sdmean$todcat<=Low_MD,2,EstDataSet.sdmean$shift.period) # darkgreen -EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Low_MD & EstDataSet.sdmean$todcat<=Peak_PM,3,EstDataSet.sdmean$shift.period) # darkorange4 -EstDataSet.sdmean$shift.period<-ifelse(EstDataSet.sdmean$todcat>=Peak_PM,4,EstDataSet.sdmean$shift.period) #chocolate4 - -EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat<=Peak_AM,1,0) #blue -EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Peak_AM & EstDataSet.meanmean$todcat<=Low_MD,2,EstDataSet.meanmean$shift.period) # darkgreen -EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Low_MD & EstDataSet.meanmean$todcat<=Peak_PM,3,EstDataSet.meanmean$shift.period) # darkorange4 -EstDataSet.meanmean$shift.period<-ifelse(EstDataSet.meanmean$todcat>=Peak_PM,4,EstDataSet.meanmean$shift.period) #chocolate4 - -#colors -EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat<=Peak_AM,"#0000FF",0) -EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Peak_AM & EstDataSet.sdmean$todcat<=Low_MD,"#006400",EstDataSet.sdmean$shift.period.color) -EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Low_MD & EstDataSet.sdmean$todcat<=Peak_PM,"#BB4500",EstDataSet.sdmean$shift.period.color) -EstDataSet.sdmean$shift.period.color<-ifelse(EstDataSet.sdmean$todcat>=Peak_PM,"#BB4513",EstDataSet.sdmean$shift.period.color) - -EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat<=Peak_AM,"#0000FF",0) -EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Peak_AM & EstDataSet.meanmean$todcat<=Low_MD,"#006400",EstDataSet.meanmean$shift.period.color) -EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Low_MD & EstDataSet.meanmean$todcat<=Peak_PM,"#BB4500",EstDataSet.meanmean$shift.period.color) -EstDataSet.meanmean$shift.period.color<-ifelse(EstDataSet.meanmean$todcat>=Peak_PM,"#BB4513",EstDataSet.meanmean$shift.period.color) - -#plot -p1<-ggplot(data=EstDataSet.sdmean,aes(x=todcat,y=Freeways)) + geom_point() + geom_line(aes(colour=EstDataSet.sdmean$shift.period.color)) + theme_bw() -p2<-ggplot(data=EstDataSet.meanmean,aes(x=todcat,y=Freeways)) + geom_point() + geom_line(aes(colour=EstDataSet.meanmean$shift.period.color)) + theme_bw() %+replace% theme(panel.background = element_rect(fill = NA)) - -p<-p + facet_grid( . ~ tod.model, scales="free_x", space="free") - -#p<-p + geom_line(colour=EstDataSet.sdmean$shift.period) -#p<-p + scale_colour_manual(breaks=EstDataSet.sdmean$shift.period, values = unique(as.character(EstDataSet.sdmean$shift.period.color))) - -p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), - plot.title = element_text(lineheight=.8, face="bold", vjust=2)) -p<-p+expand_limits(y = 0) - -p<-p+scale_x_continuous(breaks = seq(0,96, by = 1), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) - -p<-p + labs(x="Time of Day Bins",y="Travel Time SD", title="Travel Time Reliability") -p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - diff --git a/sandag_abm/src/main/r/summarize_SR125data.R b/sandag_abm/src/main/r/summarize_SR125data.R deleted file mode 100644 index 7629531..0000000 --- a/sandag_abm/src/main/r/summarize_SR125data.R +++ /dev/null @@ -1,141 +0,0 @@ -library(stringr) -library(xtable) -library(foreign) -library(data.table) - -setwd("C:/Projects/SANDAG_PricingAndReliability/data/toll facility data") - -tripdata <- read.csv("RequestedNumbers.csv",header=TRUE) -nrow(tripdata) -tripdata<-data.table(tripdata) - -#start time -tripdata[, starthour := as.numeric(substr(descr,1,2))] -tripdata[, startmin := as.numeric(substr(descr,4,5))] -tripdata[,starttime:=as.numeric(starthour*60+startmin)] - -# end time -tripdata[, endhour := as.numeric(substr(descr,9,10))] -tripdata[, endmin := as.numeric(substr(descr,12,13))] -tripdata[,endtime:=as.numeric(endhour*60+endmin)] - -tripdata[,midtime:=(starttime+endtime)/2] - -# time periods (mins from midnight) -#Early AM (EA) 210 - 359 -#AM Peak (AM) 360 - 539 -#Midday (MD) 540 - 929 -#PM Peak (PM) 930 - 1139 -#Evening (EV) 1140 - 1440 and 0 - 209 - -tripdata$tod=5 # 5-evening -tripdata[midtime>=210 & midtime<=359,tod:=1] #1-early AM -tripdata[midtime>=360 & midtime<=539,tod:=2] #2-AM peak -tripdata[midtime>=540 & midtime<=929,tod:=3] #3-Midday -tripdata[midtime>=930 & midtime<=1139,tod:=4] #4-PM Peak - -tripdata$ea<-ifelse(tripdata$tod==1,1,0) -tripdata$am<-ifelse(tripdata$tod==2,1,0) -tripdata$md<-ifelse(tripdata$tod==3,1,0) -tripdata$pm<-ifelse(tripdata$tod==4,1,0) -tripdata$ev<-ifelse(tripdata$tod==5,1,0) - -#ea trips -tripdata[,ea_FT2Axle:=FT2Axle*ea] -tripdata[,ea_CashCC2Axle:=CashCC2Axle*ea] -tripdata[,ea_FT3PlusAxle:=FT3PlusAxle*ea] -tripdata[,ea_CashCC3PlusAxle:=CashCC3PlusAxle*ea] - -#am trips -tripdata[,am_FT2Axle:=FT2Axle*am] -tripdata[,am_CashCC2Axle:=CashCC2Axle*am] -tripdata[,am_FT3PlusAxle:=FT3PlusAxle*am] -tripdata[,am_CashCC3PlusAxle:=CashCC3PlusAxle*am] - -#md trips -tripdata[,md_FT2Axle:=FT2Axle*md] -tripdata[,md_CashCC2Axle:=CashCC2Axle*md] -tripdata[,md_FT3PlusAxle:=FT3PlusAxle*md] -tripdata[,md_CashCC3PlusAxle:=CashCC3PlusAxle*md] - -#pm trips -tripdata[,pm_FT2Axle:=FT2Axle*pm] -tripdata[,pm_CashCC2Axle:=CashCC2Axle*pm] -tripdata[,pm_FT3PlusAxle:=FT3PlusAxle*pm] -tripdata[,pm_CashCC3PlusAxle:=CashCC3PlusAxle*pm] - -#ev trips -tripdata[,ev_FT2Axle:=FT2Axle*ev] -tripdata[,ev_CashCC2Axle:=CashCC2Axle*ev] -tripdata[,ev_FT3PlusAxle:=FT3PlusAxle*ev] -tripdata[,ev_CashCC3PlusAxle:=CashCC3PlusAxle*ev] - -#ea -trips_FT2Axle<-aggregate(x=tripdata$ea_FT2Axle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$ea_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$ea_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$ea_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) - -trips_ea<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) -trips_ea$CashCC2Axle<-trips_CashCC2Axle$x -trips_ea$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_ea$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -#am -trips_FT2Axle<-aggregate(x=tripdata$am_FT2Axle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$am_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$am_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$am_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) - -trips_am<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) -trips_am$CashCC2Axle<-trips_CashCC2Axle$x -trips_am$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_am$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -#md -trips_FT2Axle<-aggregate(x=tripdata$md_FT2Axle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$md_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$md_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$md_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) - -trips_md<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) -trips_md$CashCC2Axle<-trips_CashCC2Axle$x -trips_md$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_md$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -#pm -trips_FT2Axle<-aggregate(x=tripdata$pm_FT2Axle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$pm_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$pm_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$pm_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) - -trips_pm<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) -trips_pm$CashCC2Axle<-trips_CashCC2Axle$x -trips_pm$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_pm$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -#ev -trips_FT2Axle<-aggregate(x=tripdata$ev_FT2Axle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$ev_CashCC2Axle,by=list(key=tripdata$key),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$ev_FT3PlusAxle,by=list(key=tripdata$key),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$ev_CashCC3PlusAxle,by=list(key=tripdata$key),FUN=sum) - -trips_ev<-data.frame(key=trips_FT2Axle$key,FT2Axle=trips_FT2Axle$x) -trips_ev$CashCC2Axle<-trips_CashCC2Axle$x -trips_ev$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_ev$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -#all -trips_FT2Axle<-aggregate(x=tripdata$FT2Axle,by=list(tod=tripdata$tod),FUN=sum) -trips_CashCC2Axle<-aggregate(x=tripdata$CashCC2Axle,by=list(tod=tripdata$tod),FUN=sum) -trips_FT3PlusAxle<-aggregate(x=tripdata$FT3PlusAxle,by=list(tod=tripdata$tod),FUN=sum) -trips_CashCC3PlusAxle<-aggregate(x=tripdata$CashCC3PlusAxle,by=list(tod=tripdata$tod),FUN=sum) - -trips_all<-data.frame(tod=trips_FT2Axle$tod,FT2Axle=trips_FT2Axle$x) -trips_all$CashCC2Axle<-trips_CashCC2Axle$x -trips_all$FT3PlusAxle<-trips_FT3PlusAxle$x -trips_all$CashCC3PlusAxle<-trips_CashCC3PlusAxle$x - -write.table(trips_all,"sr125_summary.csv",row.names=F,quote=F,sep = ",") - -rm(tripdata) \ No newline at end of file diff --git a/sandag_abm/src/main/r/utilfunc.R b/sandag_abm/src/main/r/utilfunc.R deleted file mode 100644 index 1aead08..0000000 --- a/sandag_abm/src/main/r/utilfunc.R +++ /dev/null @@ -1,256 +0,0 @@ - -#read a file, save to Rdata, and remove from workspace -readSaveRdata <- function(filename,objname) { - assign(objname,fread(filename)) - save(list=objname,file=paste0(filename,".Rdata")) - rm(list=objname) -} - -#load an RData object to an object name -assignLoad <- function(filename) { - load(filename) - get(ls()[ls() != "filename"]) -} - -regressionmodel<-function(myDF) { - -model = lm(SpeedStdDev~NumLanes+VOC+CongSpeed+Ratio.FFTime.CongTime+IFC, data=myDF) -summary(model) - -} - -detectoutliers<-function(myDF) { - # Detects outliers in a dataset - # - # Args: - # myDF: dataframe in list of dataframes - # type: box plot to use - original or adjusted - # - # Returns: - # mydata with outlier=1 for outliers - type<-"adjusted" - - if (type=="original") { - values<-boxplot(travel_time_sec ~ todcat, myDF, main = "Original Boxplot") - } else { - values<-adjbox(travel_time_sec ~ todcat, myDF, main = "Adjusted Boxplot") - } - - if (length(table(myDF$todcat))<48) { - warning(paste("tmc segment ", myDF$tmc_code[1], " is missing some data")) - } - - for (cat in 1:length(unique(myDF$todcat))) { - limit_lower<-values[["stats"]][1,cat] - limit_upper<-values[["stats"]][5,cat] - myDF[todcat==cat & (travel_time_seclimit_upper), outlier:=1] - } - -} - -myplot_sd<-function(myDF,datatype,numoutliers,field) { - # Plots data points by facility type - # - # Args: - # myDF: dataframe in list of dataframes - # outputsDir: directory to save the plot - # - # Returns: - # Nothing. saves a plot in JPEG format in OutputsDir - - # to change the order of variables in plot - outputsDir="./INRIX/2012_10/" - - # include only segments that have data for the entire period - if (length(table(myDF$todcat))==48) { - - myDF$tod<-factor(myDF$tod, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) - IFC<-myDF$IFC[1] - - #p<-ggplot(data=myDF,aes(todcat,value, colour=tmc_code),na.rm=TRUE) + geom_point() - #p<-ggplot(data=myDF,aes(todcat,value),na.rm=TRUE) + geom_point() - - p<-ggplot(data=myDF,aes(todcat,value, group=tmc_code),na.rm=TRUE) + geom_line()+ geom_point() - - # add grids for TOD - p<-p + facet_grid(. ~ tod, scales="free_x", space="free") - - # set space between facets - p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), - plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - # set consistent axis and force axis to start at 0 - p<-p+expand_limits(y = 0) - p<-p+scale_x_continuous(breaks = seq(0,48, by = 1), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) - - # add title - if (datatype=="raw") { - outfile = paste(field, "_SD_rawdata_",IFC,".jpeg") - maintitle<-paste(field," SD by TOD (raw data)") - } else { - outfile = paste("SD_per_mile_nooutliers_",IFC,".jpeg") - maintitle<-paste(field, " SD by TOD (", numoutliers, " outliers removed)") - } - - p<-p + labs(x="Time of Day Category (30 mins interval)",y=paste(field, " SD"), title=maintitle) - p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - #print(p) - - return(p) - - # save as JPEG - #dev.copy(jpeg,filename=paste(outputsDir,outfile, sep = ""),width=1280, height=1280) - #dev.off() - - } - -} - -myplot1<-function(myDF) { - # Plots data points by facility type - # - # Args: - # myDF: dataframe in list of dataframes - # outputsDir: directory to save the plot - # - # Returns: - # Nothing. saves a plot in JPEG format in OutputsDir - - # to change the order of variables in plot - outputsDir="./INRIX/2012_10/" - myDF$tod<-factor(myDF$tod, levels = c("EV1","EA", "AM", "MD", "PM", "EV2")) - IFC<-myDF$IFC[1] - - p<-ggplot(data=myDF,aes(totalhour,speed, colour=color.names)) + geom_point() - p<-p + scale_colour_manual(breaks=myDF$color.names, values = unique(as.character(myDF$color.codes))) - - # add grids for TOD - p<-p + facet_grid(. ~ tod, scales="free_x", space="free") - - # set space between facets - p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), - plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - # set consistent axis and force axis to start at 0 - p<-p+expand_limits(y = 0) - p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) - - numoutliers<-sum(myDF$outlier) - # add title - maintitle<-paste("Speed by TOD (outliers=",numoutliers,")") - p<-p + labs(x="Time of Day (hour)",y="Speed (mph)", title=maintitle) - p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - print(p) - # save as JPEG - dev.copy(jpeg,filename=paste(outputsDir,"Speed_IFC_", IFC, "_outliers30mins.jpeg", sep = ""),width=1280, height=1280) - dev.off() - -} - -myplot<- function(indata,datatype,analysistype,xlbl,ylbl,plottitle,ymax) { - # Makes a scatter plot - # - # Args: - # indata: dataset with datapoints - # datatype: all weekdays or one - # analysistype: travel time or speed analysis - # xlbl: x-axis label - # ylbl: y-axis label - # plottitle: plot title - # ymax: y-axis limit - # - # Returns: - # a scatter plot of datapoints in indata - - # setup a plot - if (analysistype == "Time") { - p<-ggplot(data=indata,aes(x=totalhour,y=travel_time_sec)) + geom_point(colour='black',size=2) - #ymax<-max(indata$travel_time_sec)+100 # add 100sec to have some space on top - ybreak <-200 # 200 seconds - } else { - p<-ggplot(data=indata,aes(x=totalhour,y=speed)) + geom_point(colour='black',size=2) - #ymax<-max(indata$speed)+10 # add 10 mph to have some space on top - ybreak <-20 # 20 mph - } - - # set different colors for outliers - p<-p + scale_colour_hue(breaks=indata$outlier) - - # add grids for TOD - if (datatype=="All") { - p<-p + facet_grid(wday ~ tod, scales="free_x", space="free") - } else { - p<-p + facet_grid(day ~ tod, scales="free_x", space="free") - } - - # set space between facets - p<-p + theme_bw() + theme(panel.margin.x=unit(0,"lines"),panel.margin.y=unit(0.25,"lines"), - plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - # set consistent axis and force axis to start at 0 - p<-p+expand_limits(y = 0) - - # modify x and y axis - if (analysistype == "Time") { - p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(expand = c(0, 0)) #breaks=seq(0,1000, by = ybreak), - } else { - p<-p+scale_x_continuous(breaks = seq(0,24, by = 2), expand=c(0,0))+scale_y_continuous(breaks=seq(0,100, by = 20), expand = c(0, 0)) #breaks=seq(0,100, by = ybreak), - } - - # add title - p<-p + labs(x=xlbl,y=ylbl, title=plottitle) - p<-p+theme(plot.title = element_text(lineheight=.8, face="bold", vjust=2)) - - return(p) -} - -# function to add multiple plots in a print area -multiplot <- function(..., plotlist=NULL, file, cols=1, layout=NULL) { - # Combines multiple plots in one print area - # - # Args: - # ...: plots seperated by commas. Provide as many plots as you want. - # plotlist: - # file: - # cols: number of columns in the plot area - # layout: if NULL then 'cols' is used to determine layout - # - # Returns: - # None. Displays the multiplot - - library(grid) - - # Make a list from the ... arguments and plotlist - plots <- c(list(...), plotlist) - - numPlots = length(plots) - - # If layout is NULL, then use 'cols' to determine layout - if (is.null(layout)) { - # Make the panel - # ncol: Number of columns of plots - # nrow: Number of rows needed, calculated from # of cols - layout <- matrix(seq(1, cols * ceiling(numPlots/cols)), - ncol = cols, nrow = ceiling(numPlots/cols)) - } - - if (numPlots==1) { - print(plots[[1]]) - - } else { - # Set up the page - grid.newpage() - pushViewport(viewport(layout = grid.layout(nrow(layout), ncol(layout)))) - - # Make each plot, in the correct location - for (i in 1:numPlots) { - # Get the i,j matrix positions of the regions that contain this subplot - matchidx <- as.data.frame(which(layout == i, arr.ind = TRUE)) - - print(plots[[i]], vp = viewport(layout.pos.row = matchidx$row, - layout.pos.col = matchidx$col)) - } - } -} \ No newline at end of file diff --git a/sandag_abm/src/main/r/visualizer/Master.R b/sandag_abm/src/main/r/visualizer/Master.R deleted file mode 100644 index bfc4337..0000000 --- a/sandag_abm/src/main/r/visualizer/Master.R +++ /dev/null @@ -1,94 +0,0 @@ -############################################################################################################################# -# Master script to render final HTML file from R Markdown file -# Loads all required packages from the dependencies folder -# -# Make sure the 'plyr' is not loaded after 'dplyr' library in the same R session -# Under such case, the group_by features of dplyr library does not work. Restart RStudio and make sure -# plyr library is not loaded while generating dashboard -# For more info on this issue: -# https://stackoverflow.com/questions/26923862/why-are-my-dplyr-group-by-summarize-not-working-properly-name-collision-with -# -############################################################################################################################# - -##### LIST OF ALL INPUT FILES ##### -## 0. Path input data : parameters.csv -## 1. Base scenario summary file names : summaryFilesNames_survey.csv -## 2. Build scenario summary file names : summaryFilesNames.csv -## 3. Model area shapefile : summaryFilesNames.csv -## 4. All REF and ABM summary output files - -### Read Command Line Arguments -args <- commandArgs(trailingOnly = TRUE) -Parameters_File <- args[1] -showWarnings=FALSE - -### Read parameters from Parameters_File -parameters <- read.csv(Parameters_File, header = TRUE) -WORKING_DIR <- trimws(paste(parameters$Value[parameters$Key=="WORKING_DIR"])) -BASE_SUMMARY_DIR <- trimws(paste(parameters$Value[parameters$Key=="BASE_SUMMARY_DIR"])) -BUILD_SUMMARY_DIR <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SUMMARY_DIR"])) -BASE_SCENARIO_NAME <- trimws(paste(parameters$Value[parameters$Key=="BASE_SCENARIO_NAME"])) -BUILD_SCENARIO_NAME <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SCENARIO_NAME"])) -BASE_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BASE_SAMPLE_RATE"]))) -BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BUILD_SAMPLE_RATE"]))) -R_LIBRARY <- trimws(paste(parameters$Value[parameters$Key=="R_LIBRARY"])) -OUTPUT_HTML_NAME <- trimws(paste(parameters$Value[parameters$Key=="OUTPUT_HTML_NAME"])) -SHP_FILE_NAME <- trimws(paste(parameters$Value[parameters$Key=="SHP_FILE_NAME"])) -IS_BASE_SURVEY <- trimws(paste(parameters$Value[parameters$Key=="IS_BASE_SURVEY"])) - -### Initialization -# Load global variables -.libPaths(R_LIBRARY) -source(paste(WORKING_DIR, "scripts/_SYSTEM_VARIABLES.R", sep = "/")) - -###create directories -dir.create(BASE_DATA_PATH) -dir.create(BUILD_DATA_PATH) - -### Copy summary CSVs -base_CSV_list <- ifelse(IS_BASE_SURVEY=="Yes", "summaryFilesNames_survey.csv", "summaryFilesNames.csv") -summaryFileList_base <- read.csv(paste(SYSTEM_TEMPLATES_PATH, base_CSV_list, sep = '/'), as.is = T) -summaryFileList_base <- as.list(summaryFileList_base$summaryFile) -retVal <- copyFile(summaryFileList_base, sourceDir = BASE_SUMMARY_DIR, targetDir = BASE_DATA_PATH) -if(retVal) q(save = "no", status = 11) -summaryFileList_build <- read.csv(paste(SYSTEM_TEMPLATES_PATH, "summaryFilesNames.csv", sep = '/'), as.is = T) -summaryFileList_build <- as.list(summaryFileList_build$summaryFile) -retVal <- copyFile(summaryFileList_build, sourceDir = BUILD_SUMMARY_DIR, targetDir = BUILD_DATA_PATH) -if(retVal) q(save = "no", status = 11) - -### Load required libraries -SYSTEM_REPORT_PKGS <- c("DT", "flexdashboard", "leaflet", "geojsonio", "htmltools", "htmlwidgets", "kableExtra", - "knitr", "mapview", "plotly", "RColorBrewer", "rgdal", "rgeos", "crosstalk","treemap", "htmlTable", - "rmarkdown", "scales", "stringr", "jsonlite", "pander", "ggplot2", "reshape", "raster", "dplyr") - -lapply(SYSTEM_REPORT_PKGS, library, character.only = TRUE) - -### Read Target and Output SUmmary files -currDir <- getwd() -setwd(BASE_DATA_PATH) -base_csv = list.files(pattern="*.csv") -base_data <- lapply(base_csv, read.csv) -base_csv_names <- unlist(lapply(base_csv, function (x) {gsub(".csv", "", x)})) - -setwd(BUILD_DATA_PATH) -build_csv = list.files(pattern="*.csv") -build_data <- lapply(build_csv, read.csv) -build_csv_names <- unlist(lapply(build_csv, function (x) {gsub(".csv", "", x)})) - -## Read SHP file -setwd(SYSTEM_SHP_PATH) -zone_shp <- shapefile(SHP_FILE_NAME) -zone_shp <- spTransform(zone_shp, CRS("+proj=longlat +ellps=GRS80")) - -setwd(currDir) - -### Generate dashboard -rmarkdown::render(file.path(SYSTEM_TEMPLATES_PATH, "template.Rmd"), - output_dir = RUNTIME_PATH, - intermediates_dir = RUNTIME_PATH, quiet = TRUE) -template.html <- readLines(file.path(RUNTIME_PATH, "template.html")) -idx <- which(template.html == "window.FlexDashboardComponents = [];")[1] -template.html <- append(template.html, "L_PREFER_CANVAS = true;", after = idx) -writeLines(template.html, file.path(OUTPUT_PATH, paste(OUTPUT_HTML_NAME, ".html", sep = ""))) - -# finish \ No newline at end of file diff --git a/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R b/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R deleted file mode 100644 index fb6eb50..0000000 --- a/sandag_abm/src/main/r/visualizer/SummarizeABM2016.R +++ /dev/null @@ -1,2448 +0,0 @@ -####################################################### -### Script for summarizing SANDAG ABM Output -### Author: Binny M Paul, binny.paul@rsginc.com, Oct 2017 -### Edited: Khademul Haque, khademul.haque@rsginc.com, Mar 2019 -####################################################### - -##### LIST OF ALL INPUT FILES ##### -## 0. Path input data : summ_inputs_abm.csv -## 1. household data : householdData_3.csv -## 2. person data : personData_3.csv -## 3. Individual tour data : indivTourData_3.csv -## 4. Individual trip data : indivTripData_3.csv -## 5. Joint tour data : jointTripData_3.csv -## 6. Joint trip data : jointTourData_3.csv -## 7. Work school location data : wsLocResults_3.csv -## 8. Auto ownership data : aoResults.csv -## 9. Auto ownership data : aoResults_Pre.csv -## 10. Geographic crosswalk data : geographicXwalk_PMSA.csv -## 11. Distance skim : traffic_skims_MD.omx -> MD_SOVTOLLH_DIST - -start_time <- Sys.time() - -library(plyr) -library(weights) -library(reshape) -library(data.table) -library(omxr) -showWarnings=FALSE - -# Read Command Line Arguments -args <- commandArgs(trailingOnly = TRUE) -inputs_File <- args[1] - -inputs <- read.csv(inputs_File, header = TRUE) -WD <- trimws(paste(inputs$Value[inputs$Key=="WD"])) -ABMOutputDir <- trimws(paste(inputs$Value[inputs$Key=="ABMOutputDir"])) -geogXWalkDir <- trimws(paste(inputs$Value[inputs$Key=="geogXWalkDir"])) -SkimDir <- trimws(paste(inputs$Value[inputs$Key=="SkimDir"])) -MAX_ITER <- trimws(paste(inputs$Value[inputs$Key=="MAX_ITER"])) - -setwd(ABMOutputDir) -#full model run -hh <- fread(paste("householdData_",MAX_ITER,".csv", sep = "")) -per <- fread(paste("personData_",MAX_ITER,".csv", sep = "")) -tours <- fread(paste("indivTourData_",MAX_ITER,".csv", sep = "")) -trips <- fread(paste("indivTripData_",MAX_ITER,".csv", sep = "")) -jtrips <- fread(paste("jointTripData_",MAX_ITER,".csv", sep = "")) -unique_joint_tours <- fread(paste("jointTourData_",MAX_ITER,".csv", sep = "")) -wsLoc <- fread(paste("wsLocResults_",MAX_ITER,".csv", sep = "")) -aoResults <- fread("aoResults.csv") -aoResults_Pre <- fread("aoResults_Pre.csv") - -visitor_trips <- fread(paste("visitorTrips.csv", sep = "")) - -mazCorrespondence <- fread(paste(geogXWalkDir, "geographicXwalk_PMSA.csv", sep = "/"), stringsAsFactors = F) -districtList <- sort(unique(mazCorrespondence$pmsa)) - -SkimFile <- paste(SkimDir, "traffic_skims_MD.omx", sep = "/") -DST_SKM <- read_omx(SkimFile, "MD_SOV_TR_H_DIST") -skimLookUp <- read_lookup(SkimFile, "zone_number") - -pertypeCodes <- data.frame(code = c(1,2,3,4,5,6,7,8,"All"), - name = c("FT Worker", "PT Worker", "Univ Stud", "Non Worker", "Retiree", "Driv Stud", "NonDriv Stud", "Pre-School", "All")) - -#------------------------------------------- -# Prepare files for computing summary statistics -dir.create(WD, showWarnings = FALSE) -setwd(WD) - -aoResults$HHVEH[aoResults$AO == 0] <- 0 -aoResults$HHVEH[aoResults$AO == 1] <- 1 -aoResults$HHVEH[aoResults$AO == 2] <- 2 -aoResults$HHVEH[aoResults$AO == 3] <- 3 -aoResults$HHVEH[aoResults$AO >= 4] <- 4 - -aoResults_Pre$HHVEH[aoResults_Pre$AO == 0] <- 0 -aoResults_Pre$HHVEH[aoResults_Pre$AO == 1] <- 1 -aoResults_Pre$HHVEH[aoResults_Pre$AO == 2] <- 2 -aoResults_Pre$HHVEH[aoResults_Pre$AO == 3] <- 3 -aoResults_Pre$HHVEH[aoResults_Pre$AO >= 4] <- 4 - -hh$HHVEH[hh$autos == 0] <- 0 -hh$HHVEH[hh$autos == 1] <- 1 -hh$HHVEH[hh$autos == 2] <- 2 -hh$HHVEH[hh$autos == 3] <- 3 -hh$HHVEH[hh$autos >= 4] <- 4 - -hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 0] <- 1 -hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 0] <- 2 -hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 1] <- 3 -hh$VEH_NEWCAT[(hh$HVs == 2) & (hh$AVs) == 0] <- 4 -hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 2] <- 5 -hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 1] <- 6 -hh$VEH_NEWCAT[(hh$HVs == 3) & (hh$AVs) == 0] <- 7 -hh$VEH_NEWCAT[(hh$HVs == 0) & (hh$AVs) == 3] <- 8 -hh$VEH_NEWCAT[(hh$HVs == 2) & (hh$AVs) == 1] <- 9 -hh$VEH_NEWCAT[(hh$HVs == 1) & (hh$AVs) == 2] <- 10 -hh$VEH_NEWCAT[(hh$HVs == 4) & (hh$AVs) == 0] <- 11 - -#HH Size -hhsize <- count(per, c("hh_id"), "hh_id>0") -hh$HHSIZ <- hhsize$freq[match(hh$hh_id, hhsize$hh_id)] -hh$HHSIZE[hh$HHSIZ == 1] <- 1 -hh$HHSIZE[hh$HHSIZ == 2] <- 2 -hh$HHSIZE[hh$HHSIZ == 3] <- 3 -hh$HHSIZE[hh$HHSIZ == 4] <- 4 -hh$HHSIZE[hh$HHSIZ >= 5] <- 5 - -#Adults in the HH -adults <- count(per, c("hh_id"), "age>=18 & age<99") -hh$ADULTS <- adults$freq[match(hh$hh_id, adults$hh_id)] - -per$PERTYPE[per$type=="Full-time worker"] <- 1 -per$PERTYPE[per$type=="Part-time worker"] <- 2 -per$PERTYPE[per$type=="University student"] <- 3 -per$PERTYPE[per$type=="Non-worker"] <- 4 -per$PERTYPE[per$type=="Retired"] <- 5 -per$PERTYPE[per$type=="Student of driving age"] <- 6 -per$PERTYPE[per$type=="Student of non-driving age"] <- 7 -per$PERTYPE[per$type=="Child too young for school"] <- 8 - -# Districts are Pseudo MSA -wsLoc$HDISTRICT <- mazCorrespondence$pmsa[match(wsLoc$HomeMGRA, mazCorrespondence$mgra)] -wsLoc$WDISTRICT <- mazCorrespondence$pmsa[match(wsLoc$WorkLocation, mazCorrespondence$mgra)] - -# Get home, work and school location TAZs -wsLoc$HHTAZ <- mazCorrespondence$taz[match(wsLoc$HomeMGRA, mazCorrespondence$mgra)] -wsLoc$WTAZ <- mazCorrespondence$taz[match(wsLoc$WorkLocation, mazCorrespondence$mgra)] -wsLoc$STAZ <- mazCorrespondence$taz[match(wsLoc$SchoolLocation, mazCorrespondence$mgra)] - -wsLoc$oindex<-match(wsLoc$HHTAZ, skimLookUp$Lookup) -wsLoc$dindex<-match(wsLoc$WTAZ, skimLookUp$Lookup) -wsLoc$dindex2<-match(wsLoc$STAZ, skimLookUp$Lookup) -wsLoc$WorkLocationDistance<-DST_SKM[cbind(wsLoc$oindex, wsLoc$dindex)] -wsLoc$WorkLocationDistance[is.na(wsLoc$WorkLocationDistance)] <- 0 - -wsLoc$SchoolLocationDistance<-DST_SKM[cbind(wsLoc$oindex, wsLoc$dindex2)] -wsLoc$SchoolLocationDistance[is.na(wsLoc$SchoolLocationDistance)] <- 0 - -#--------Compute Summary Statistics------- -#***************************************** - -# Auto ownership -autoOwnership_Pre <- count(aoResults_Pre, c("HHVEH")) -write.csv(autoOwnership_Pre, "autoOwnership_Pre.csv", row.names = TRUE) - -autoOwnership <- count(aoResults, c("HHVEH")) -write.csv(autoOwnership, "autoOwnership.csv", row.names = TRUE) - -autoOwnership_AV <- count(hh, c("AVs")) -write.csv(autoOwnership_AV, "autoOwnership_AV.csv", row.names = TRUE) - -autoOwnership_new <- count(hh, c("VEH_NEWCAT")) -write.csv(autoOwnership_new, "autoOwnership_new.csv", row.names = TRUE) - -# Zero auto HHs by TAZ -hh$HHTAZ <- mazCorrespondence$taz[match(hh$home_mgra, mazCorrespondence$mgra)] -hh$ZeroAutoWgt[hh$HHVEH==0] <- 1 -hh$ZeroAutoWgt[is.na(hh$ZeroAutoWgt)] <- 0 -zeroAutoByTaz <- aggregate(hh$ZeroAutoWgt, list(TAZ = hh$HHTAZ), sum) -write.csv(zeroAutoByTaz, "zeroAutoByTaz.csv", row.names = TRUE) - -# Persons by person type -pertypeDistbn <- count(per, c("PERTYPE")) -write.csv(pertypeDistbn, "pertypeDistbn.csv", row.names = TRUE) - -# Telecommute Freuency -teleCommute <- count(per, c("tele_choice")) -write.csv(teleCommute, "teleCommute_frequency.csv", row.names = TRUE) - -# HH Transponder Ownership -transponder <- count(hh, c("transponder")) -write.csv(transponder, "transponder_ownership.csv", row.names = TRUE) - -# Micro-mobility -micro_r1 <- count(trips, c('micro_walkMode')) -micro_r2 <- count(trips, c('micro_trnAcc')) -micro_r3 <- count(trips, c('micro_trnEgr')) -colnames(micro_r1) <- c("micro_mode","trips") -colnames(micro_r2) <- c("micro_mode","trips") -colnames(micro_r3) <- c("micro_mode","trips") - -micro_v1 <- count(visitor_trips, c('micro_walkMode')) -micro_v2 <- count(visitor_trips, c('micro_trnAcc')) -micro_v3 <- count(visitor_trips, c('micro_trnEgr')) -colnames(micro_v1) <- c("micro_mode","trips") -colnames(micro_v2) <- c("micro_mode","trips") -colnames(micro_v3) <- c("micro_mode","trips") - -micromobility <- rbind(micro_r1,micro_r2,micro_r3,micro_v1,micro_v2,micro_v3) -micromobility_summary <- aggregate(trips ~ micro_mode, data=micromobility, FUN = sum) -write.csv(micromobility_summary, "micormobility.csv", row.names = TRUE) - -# Mandatory DC -workers <- wsLoc[wsLoc$WorkLocation > 0 & wsLoc$WorkLocation != 99999,] -students <- wsLoc[wsLoc$SchoolLocation > 0 & wsLoc$SchoolLocation != 88888,] - -# code distance bins -workers$distbin <- cut(workers$WorkLocationDistance, breaks = c(seq(0,50, by=1), 9999), labels = F, right = F) -students$distbin <- cut(students$SchoolLocationDistance, breaks = c(seq(0,50, by=1), 9999), labels = F, right = F) - -distBinCat <- data.frame(distbin = seq(1,51, by=1)) -districtList_df <- data.frame(id = districtList) - -# compute TLFDs by district and total -tlfd_work <- ddply(workers[,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, work = sum(HDISTRICT>0)) -tlfd_work <- cast(tlfd_work, distbin~HDISTRICT, value = "work", sum) -work_ditbins <- tlfd_work$distbin -tlfd_work <- transpose(tlfd_work[,!colnames(tlfd_work) %in% c("distbin")]) -tlfd_work$id <- row.names(tlfd_work) -tlfd_work <- merge(x = districtList_df, y = tlfd_work, by = "id", all.x = TRUE) -tlfd_work[is.na(tlfd_work)] <- 0 -tlfd_work <- transpose(tlfd_work[,!colnames(tlfd_work) %in% c("id")]) -tlfd_work <- cbind(data.frame(distbin = work_ditbins), tlfd_work) -tlfd_work$Total <- rowSums(tlfd_work[,!colnames(tlfd_work) %in% c("distbin")]) -names(tlfd_work) <- sub("V", "District_", names(tlfd_work)) -tlfd_work_df <- merge(x = distBinCat, y = tlfd_work, by = "distbin", all.x = TRUE) -tlfd_work_df[is.na(tlfd_work_df)] <- 0 - -tlfd_univ <- ddply(students[students$PersonType==3,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, univ = sum(HDISTRICT>0)) -tlfd_univ <- cast(tlfd_univ, distbin~HDISTRICT, value = "univ", sum) -univ_ditbins <- tlfd_univ$distbin -tlfd_univ <- transpose(tlfd_univ[,!colnames(tlfd_univ) %in% c("distbin")]) -tlfd_univ$id <- row.names(tlfd_univ) -tlfd_univ <- merge(x = districtList_df, y = tlfd_univ, by = "id", all.x = TRUE) -tlfd_univ[is.na(tlfd_univ)] <- 0 -tlfd_univ <- transpose(tlfd_univ[,!colnames(tlfd_univ) %in% c("id")]) -tlfd_univ <- cbind(data.frame(distbin = univ_ditbins), tlfd_univ) -tlfd_univ$Total <- rowSums(tlfd_univ[,!colnames(tlfd_univ) %in% c("distbin")]) -names(tlfd_univ) <- sub("V", "District_", names(tlfd_univ)) -tlfd_univ_df <- merge(x = distBinCat, y = tlfd_univ, by = "distbin", all.x = TRUE) -tlfd_univ_df[is.na(tlfd_univ_df)] <- 0 - -tlfd_schl <- ddply(students[students$PersonType>=6,c("HDISTRICT", "distbin")], c("HDISTRICT", "distbin"), summarise, schl = sum(HDISTRICT>0)) -tlfd_schl <- cast(tlfd_schl, distbin~HDISTRICT, value = "schl", sum) -schl_ditbins <- tlfd_schl$distbin -tlfd_schl <- transpose(tlfd_schl[,!colnames(tlfd_schl) %in% c("distbin")]) -tlfd_schl$id <- row.names(tlfd_schl) -tlfd_schl <- merge(x = districtList_df, y = tlfd_schl, by = "id", all.x = TRUE) -tlfd_schl[is.na(tlfd_schl)] <- 0 -tlfd_schl <- transpose(tlfd_schl[,!colnames(tlfd_schl) %in% c("id")]) -tlfd_schl <- cbind(data.frame(distbin = schl_ditbins), tlfd_schl) -tlfd_schl$Total <- rowSums(tlfd_schl[,!colnames(tlfd_schl) %in% c("distbin")]) -names(tlfd_schl) <- sub("V", "District_", names(tlfd_schl)) -tlfd_schl_df <- merge(x = distBinCat, y = tlfd_schl, by = "distbin", all.x = TRUE) -tlfd_schl_df[is.na(tlfd_schl_df)] <- 0 - -write.csv(tlfd_work_df, "workTLFD.csv", row.names = F) -write.csv(tlfd_univ_df, "univTLFD.csv", row.names = F) -write.csv(tlfd_schl_df, "schlTLFD.csv", row.names = F) - -cat("\n Average distance to workplace (Total): ", mean(workers$WorkLocationDistance, na.rm = TRUE)) -cat("\n Average distance to university (Total): ", mean(students$SchoolLocationDistance[students$PersonType==3], na.rm = TRUE)) -cat("\n Average distance to school (Total): ", mean(students$SchoolLocationDistance[students$PersonType>=6], na.rm = TRUE)) - -## Output avg trip lengths for visualizer -workTripLengths <- ddply(workers[,c("HDISTRICT", "WorkLocationDistance")], c("HDISTRICT"), summarise, work = mean(WorkLocationDistance)) -totalLength <- data.frame("Total", mean(workers$WorkLocationDistance)) -colnames(totalLength) <- colnames(workTripLengths) -workTripLengths <- rbind(workTripLengths, totalLength) - -univTripLengths <- ddply(students[students$PersonType==3,c("HDISTRICT", "SchoolLocationDistance")], c("HDISTRICT"), summarise, univ = mean(SchoolLocationDistance)) -totalLength <- data.frame("Total", mean(students$SchoolLocationDistance[students$PersonType==3])) -colnames(totalLength) <- colnames(univTripLengths) -univTripLengths <- rbind(univTripLengths, totalLength) - -schlTripLengths <- ddply(students[students$PersonType>=6,c("HDISTRICT", "SchoolLocationDistance")], c("HDISTRICT"), summarise, schl = mean(SchoolLocationDistance)) -totalLength <- data.frame("Total", mean(students$SchoolLocationDistance[students$PersonType>=6])) -colnames(totalLength) <- colnames(schlTripLengths) -schlTripLengths <- rbind(schlTripLengths, totalLength) - -mandTripLengths <- cbind(workTripLengths, univTripLengths$univ, schlTripLengths$schl) -colnames(mandTripLengths) <- c("District", "Work", "Univ", "Schl") -write.csv(mandTripLengths, "mandTripLengths.csv", row.names = F) - -# Work from home [for each district and total] -districtWorkers <- ddply(wsLoc[wsLoc$WorkLocation > 0,c("HDISTRICT")], c("HDISTRICT"), summarise, workers = sum(HDISTRICT>0)) -districtWfh <- ddply(wsLoc[wsLoc$WorkLocation==99999,c("HDISTRICT", "WorkLocation")], c("HDISTRICT"), summarise, wfh = sum(HDISTRICT>0)) -wfh_summary <- cbind(districtWorkers, districtWfh$wfh) -colnames(wfh_summary) <- c("District", "Workers", "WFH") -totalwfh <- data.frame("Total", sum(wsLoc$WorkLocation>0), sum(wsLoc$WorkLocation==99999)) -colnames(totalwfh) <- colnames(wfh_summary) -wfh_summary <- rbind(wfh_summary, totalwfh) -write.csv(wfh_summary, "wfh_summary.csv", row.names = F) -write.csv(totalwfh, "wfh_summary_region.csv", row.names = F) - -# County-County Flows -countyFlows <- xtabs(~HDISTRICT+WDISTRICT, data = workers) -countyFlows[is.na(countyFlows)] <- 0 -countyFlows <- addmargins(as.table(countyFlows)) -countyFlows <- as.data.frame.matrix(countyFlows) -colnames(countyFlows)[colnames(countyFlows)=="Sum"] <- "Total" -colnames(countyFlows) <- paste("District", colnames(countyFlows), sep = "_") -rownames(countyFlows)[rownames(countyFlows)=="Sum"] <- "Total" -rownames(countyFlows) <- paste("District", rownames(countyFlows), sep = "_") -write.csv(countyFlows, "countyFlows.csv", row.names = T) - -# Process Tour file -#------------------ -tours$PERTYPE <- tours$person_type -tours$DISTMILE <- tours$tour_distance -tours$HHVEH <- hh$HHVEH[match(tours$hh_id, hh$hh_id)] -tours$ADULTS <- hh$ADULTS[match(tours$hh_id, hh$hh_id)] -tours$AUTOSUFF[tours$HHVEH == 0] <- 0 -tours$AUTOSUFF[tours$HHVEH < tours$ADULTS & tours$HHVEH > 0] <- 1 -tours$AUTOSUFF[tours$HHVEH >= tours$ADULTS & tours$HHVEH > 0] <- 2 - -tours$num_tot_stops <- tours$num_ob_stops + tours$num_ib_stops - -tours$OTAZ <- mazCorrespondence$taz[match(tours$orig_mgra, mazCorrespondence$mgra)] -tours$DTAZ <- mazCorrespondence$taz[match(tours$dest_mgra, mazCorrespondence$mgra)] - -tours$oindex<-match(tours$OTAZ, skimLookUp$Lookup) -tours$dindex<-match(tours$DTAZ, skimLookUp$Lookup) -tours$SKIMDIST<-DST_SKM[cbind(tours$oindex, tours$dindex)] - - -unique_joint_tours$HHVEH <- hh$HHVEH[match(unique_joint_tours$hh_id, hh$hh_id)] -unique_joint_tours$ADULTS <- hh$ADULTS[match(unique_joint_tours$hh_id, hh$hh_id)] -unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH == 0] <- 0 -unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH < unique_joint_tours$ADULTS & unique_joint_tours$HHVEH > 0] <- 1 -unique_joint_tours$AUTOSUFF[unique_joint_tours$HHVEH >= unique_joint_tours$ADULTS] <- 2 - -#Code tour purposes -tours$TOURPURP[tours$tour_purpose=="Work"] <- 1 -tours$TOURPURP[tours$tour_purpose=="University"] <- 2 -tours$TOURPURP[tours$tour_purpose=="School"] <- 3 -tours$TOURPURP[tours$tour_purpose=="Escort"] <- 4 -tours$TOURPURP[tours$tour_purpose=="Shop"] <- 5 -tours$TOURPURP[tours$tour_purpose=="Maintenance"] <- 6 -tours$TOURPURP[tours$tour_purpose=="Eating Out"] <- 7 -tours$TOURPURP[tours$tour_purpose=="Visiting"] <- 8 -tours$TOURPURP[tours$tour_purpose=="Discretionary"] <- 9 -tours$TOURPURP[tours$tour_purpose=="Work-Based"] <- 10 - -#[0:Mandatory, 1: Indi Non Mand, 2: At Work] -tours$TOURCAT[tours$tour_purpose=="Work"] <- 0 -tours$TOURCAT[tours$tour_purpose=="University"] <- 0 -tours$TOURCAT[tours$tour_purpose=="School"] <- 0 -tours$TOURCAT[tours$tour_purpose=="Escort"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Shop"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Maintenance"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Eating Out"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Visiting"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Discretionary"] <- 1 -tours$TOURCAT[tours$tour_purpose=="Work-Based"] <- 2 - -#compute duration -tours$tourdur <- tours$end_period - tours$start_period + 1 #[to match survey] - -tours$TOURMODE <- tours$tour_mode -#tours$TOURMODE[tours$tour_mode==1] <- 1 -#tours$TOURMODE[tours$tour_mode==2] <- 2 -#tours$TOURMODE[tours$tour_mode==3] <- 3 -#tours$TOURMODE[tours$tour_mode>=7 & tours$tour_mode<=13] <- tours$tour_mode[tours$tour_mode>=7 & tours$tour_mode<=13]-3 -#tours$TOURMODE[tours$tour_mode>=14 & tours$tour_mode<=15] <- 11 -#tours$TOURMODE[tours$tour_mode==16] <- 12 - -# exclude school escorting stop from ride sharing mandatory tours - -unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Shop'] <- 5 -unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Maintenance'] <- 6 -unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Eating Out'] <- 7 -unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Visiting'] <- 8 -unique_joint_tours$JOINT_PURP[unique_joint_tours$tour_purpose=='Discretionary'] <- 9 - -unique_joint_tours$NUMBER_HH <- as.integer((nchar(as.character(unique_joint_tours$tour_participants))+1)/2) - -# get participant IDs -unique_joint_tours$PER1[unique_joint_tours$NUMBER_HH>=1] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=1]), 1, 1) -unique_joint_tours$PER2[unique_joint_tours$NUMBER_HH>=2] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=2]), 3, 3) -unique_joint_tours$PER3[unique_joint_tours$NUMBER_HH>=3] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=3]), 5, 5) -unique_joint_tours$PER4[unique_joint_tours$NUMBER_HH>=4] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=4]), 7, 7) -unique_joint_tours$PER5[unique_joint_tours$NUMBER_HH>=5] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=5]), 9, 9) -unique_joint_tours$PER6[unique_joint_tours$NUMBER_HH>=6] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=6]), 11, 11) -unique_joint_tours$PER7[unique_joint_tours$NUMBER_HH>=7] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=7]), 13, 13) -unique_joint_tours$PER8[unique_joint_tours$NUMBER_HH>=8] <- substr(as.character(unique_joint_tours$tour_participants[unique_joint_tours$NUMBER_HH>=8]), 15, 15) - -unique_joint_tours[is.na(unique_joint_tours)] <- 0 - -# get person type for each participant -unique_joint_tours$PTYPE1 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER1, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE2 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER2, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE3 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER3, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE4 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER4, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE5 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER5, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE6 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER6, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE7 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER7, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] -unique_joint_tours$PTYPE8 <- per$PERTYPE[match(paste(unique_joint_tours$hh_id,unique_joint_tours$PER8, sep = "-"), paste(per$hh_id,per$person_num, sep = "-"))] - -unique_joint_tours[is.na(unique_joint_tours)] <- 0 - -unique_joint_tours$num_tot_stops <- unique_joint_tours$num_ob_stops + unique_joint_tours$num_ib_stops - -unique_joint_tours$OTAZ <- mazCorrespondence$taz[match(unique_joint_tours$orig_mgra, mazCorrespondence$mgra)] -unique_joint_tours$DTAZ <- mazCorrespondence$taz[match(unique_joint_tours$dest_mgra, mazCorrespondence$mgra)] - -#compute duration -unique_joint_tours$tourdur <- unique_joint_tours$end_period - unique_joint_tours$start_period + 1 #[to match survye] - -unique_joint_tours$TOURMODE <- unique_joint_tours$tour_mode -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode<=2] <- 1 -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=3 & unique_joint_tours$tour_mode<=4] <- 2 -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=5 & unique_joint_tours$tour_mode<=6] <- 3 -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=7 & unique_joint_tours$tour_mode<=13] <- unique_joint_tours$tour_mode[unique_joint_tours$tour_mode>=7 & unique_joint_tours$tour_mode<=13]-3 -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode>=14 & unique_joint_tours$tour_mode<=15] <- 11 -#unique_joint_tours$TOURMODE[unique_joint_tours$tour_mode==16] <- 12 - -# ---- -# this part is added by nagendra.dhakar@rsginc.com from binny.paul@rsginc.com soabm summaries - -# create a combined temp tour file for creating stop freq model summary -temp_tour1 <- tours[,c("TOURPURP","num_ob_stops","num_ib_stops")] -temp_tour2 <- unique_joint_tours[,c("JOINT_PURP","num_ob_stops","num_ib_stops")] -colnames(temp_tour2) <- colnames(temp_tour1) -temp_tour <- rbind(temp_tour1,temp_tour2) - -# code stop frequency model alternatives -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==0] <- 1 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==1] <- 2 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops==2] <- 3 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==0 & temp_tour$num_ib_stops>=3] <- 4 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==0] <- 5 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==1] <- 6 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops==2] <- 7 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==1 & temp_tour$num_ib_stops>=3] <- 8 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==0] <- 9 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==1] <- 10 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops==2] <- 11 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops==2 & temp_tour$num_ib_stops>=3] <- 12 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==0] <- 13 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==1] <- 14 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops==2] <- 15 -temp_tour$STOP_FREQ_ALT[temp_tour$num_ob_stops>=3 & temp_tour$num_ib_stops>=3] <- 16 -temp_tour$STOP_FREQ_ALT[is.na(temp_tour$STOP_FREQ_ALT)] <- 0 - -stopFreqModel_summary <- xtabs(~STOP_FREQ_ALT+TOURPURP, data = temp_tour[temp_tour$TOURPURP<=10,]) -write.csv(stopFreqModel_summary, "stopFreqModel_summary.csv", row.names = T) - -# ------ - -# Process Trip file -#------------------ -trips$TOURMODE <- trips$tour_mode -trips$TRIPMODE <- trips$trip_mode - -#trips$TOURMODE[trips$tour_mode<=2] <- 1 -#trips$TOURMODE[trips$tour_mode>=3 & trips$tour_mode<=4] <- 2 -#trips$TOURMODE[trips$tour_mode>=5 & trips$tour_mode<=6] <- 3 -#trips$TOURMODE[trips$tour_mode>=7 & trips$tour_mode<=13] <- trips$tour_mode[trips$tour_mode>=7 & trips$tour_mode<=13]-3 -#trips$TOURMODE[trips$tour_mode>=14 & trips$tour_mode<=15] <- 11 -#trips$TOURMODE[trips$tour_mode==16] <- 12 -# -#trips$TRIPMODE[trips$trip_mode<=2] <- 1 -#trips$TRIPMODE[trips$trip_mode>=3 & trips$trip_mode<=4] <- 2 -#trips$TRIPMODE[trips$trip_mode>=5 & trips$trip_mode<=6] <- 3 -#trips$TRIPMODE[trips$trip_mode>=7 & trips$trip_mode<=13] <- trips$trip_mode[trips$trip_mode>=7 & trips$trip_mode<=13]-3 -#trips$TRIPMODE[trips$trip_mode>=14 & trips$trip_mode<=15] <- 11 -#trips$TRIPMODE[trips$trip_mode==16] <- 12 - -#Code tour purposes -trips$TOURPURP[trips$tour_purpose=="Home"] <- 0 -trips$TOURPURP[trips$tour_purpose=="Work"] <- 1 -trips$TOURPURP[trips$tour_purpose=="University"] <- 2 -trips$TOURPURP[trips$tour_purpose=="School"] <- 3 -trips$TOURPURP[trips$tour_purpose=="Escort"] <- 4 -trips$TOURPURP[trips$tour_purpose=="Shop"] <- 5 -trips$TOURPURP[trips$tour_purpose=="Maintenance"] <- 6 -trips$TOURPURP[trips$tour_purpose=="Eating Out"] <- 7 -trips$TOURPURP[trips$tour_purpose=="Visiting"] <- 8 -trips$TOURPURP[trips$tour_purpose=="Discretionary"] <- 9 -trips$TOURPURP[trips$tour_purpose=="Work-Based" | trips$tour_purpose=="work related"] <- 10 - -trips$OPURP[trips$orig_purpose=="Home"] <- 0 -trips$OPURP[trips$orig_purpose=="Work"] <- 1 -trips$OPURP[trips$orig_purpose=="University"] <- 2 -trips$OPURP[trips$orig_purpose=="School"] <- 3 -trips$OPURP[trips$orig_purpose=="Escort"] <- 4 -trips$OPURP[trips$orig_purpose=="Shop"] <- 5 -trips$OPURP[trips$orig_purpose=="Maintenance"] <- 6 -trips$OPURP[trips$orig_purpose=="Eating Out"] <- 7 -trips$OPURP[trips$orig_purpose=="Visiting"] <- 8 -trips$OPURP[trips$orig_purpose=="Discretionary"] <- 9 -trips$OPURP[trips$orig_purpose=="Work-Based" | trips$orig_purpose=="work related"] <- 10 - -trips$DPURP[trips$dest_purpose=="Home"] <- 0 -trips$DPURP[trips$dest_purpose=="Work"] <- 1 -trips$DPURP[trips$dest_purpose=="University"] <- 2 -trips$DPURP[trips$dest_purpose=="School"] <- 3 -trips$DPURP[trips$dest_purpose=="Escort"] <- 4 -trips$DPURP[trips$dest_purpose=="Shop"] <- 5 -trips$DPURP[trips$dest_purpose=="Maintenance"] <- 6 -trips$DPURP[trips$dest_purpose=="Eating Out"] <- 7 -trips$DPURP[trips$dest_purpose=="Visiting"] <- 8 -trips$DPURP[trips$dest_purpose=="Discretionary"] <- 9 -trips$DPURP[trips$dest_purpose=="Work-Based" | trips$dest_purpose=="work related"] <- 10 - -#[0:Mandatory, 1: Indi Non Mand, 3: At Work] -trips$TOURCAT[trips$tour_purpose=="Work"] <- 0 -trips$TOURCAT[trips$tour_purpose=="University"] <- 0 -trips$TOURCAT[trips$tour_purpose=="School"] <- 0 -trips$TOURCAT[trips$tour_purpose=="Escort"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Shop"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Maintenance"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Eating Out"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Visiting"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Discretionary"] <- 1 -trips$TOURCAT[trips$tour_purpose=="Work-Based"] <- 2 - -#Mark stops and get other attributes -nr <- nrow(trips) -trips$inb_next <- 0 -trips$inb_next[1:nr-1] <- trips$inbound[2:nr] -trips$stops[trips$DPURP>0 & ((trips$inbound==0 & trips$inb_next==0) | (trips$inbound==1 & trips$inb_next==1))] <- 1 -trips$stops[is.na(trips$stops)] <- 0 - -trips$OTAZ <- mazCorrespondence$taz[match(trips$orig_mgra, mazCorrespondence$mgra)] -trips$DTAZ <- mazCorrespondence$taz[match(trips$dest_mgra, mazCorrespondence$mgra)] - -trips$TOUROTAZ <- tours$OTAZ[match(trips$hh_id*1000+trips$person_num*100+trips$TOURCAT*10+trips$tour_id, - tours$hh_id*1000+tours$person_num*100+tours$TOURCAT*10+tours$tour_id)] -trips$TOURDTAZ <- tours$DTAZ[match(trips$hh_id*1000+trips$person_num*100+trips$TOURCAT*10+trips$tour_id, - tours$hh_id*1000+tours$person_num*100+tours$TOURCAT*10+tours$tour_id)] - -# trips$od_dist <- DST_SKM$dist[match(paste(trips$OTAZ, trips$DTAZ, sep = "-"), paste(DST_SKM$o, DST_SKM$d, sep = "-"))] -trips$oindex<-match(trips$OTAZ, skimLookUp$Lookup) -trips$dindex<-match(trips$DTAZ, skimLookUp$Lookup) -trips$od_dist<-DST_SKM[cbind(trips$oindex, trips$dindex)] - -#create stops table -stops <- trips[trips$stops==1,] - -stops$finaldestTAZ[stops$inbound==0] <- stops$TOURDTAZ[stops$inbound==0] -stops$finaldestTAZ[stops$inbound==1] <- stops$TOUROTAZ[stops$inbound==1] - -stops$oindex<-match(stops$OTAZ, skimLookUp$Lookup) -stops$dindex<-match(stops$finaldestTAZ, skimLookUp$Lookup) -stops$od_dist <- DST_SKM[cbind(stops$oindex, stops$dindex)] - -stops$oindex2<-match(stops$OTAZ, skimLookUp$Lookup) -stops$dindex2<-match(stops$DTAZ, skimLookUp$Lookup) -stops$os_dist <- DST_SKM[cbind(stops$oindex2, stops$dindex2)] - -stops$oindex3<-match(stops$DTAZ, skimLookUp$Lookup) -stops$dindex3<-match(stops$finaldestTAZ, skimLookUp$Lookup) -stops$sd_dist <- DST_SKM[cbind(stops$oindex3, stops$dindex3)] - -stops$out_dir_dist <- stops$os_dist + stops$sd_dist - stops$od_dist - -#joint trip - -jtrips$TOURMODE <- jtrips$tour_mode -#jtrips$TOURMODE[jtrips$tour_mode<=2] <- 1 -#jtrips$TOURMODE[jtrips$tour_mode>=3 & jtrips$tour_mode<=4] <- 2 -#jtrips$TOURMODE[jtrips$tour_mode>=5 & jtrips$tour_mode<=6] <- 3 -#jtrips$TOURMODE[jtrips$tour_mode>=7 & jtrips$tour_mode<=13] <- jtrips$tour_mode[jtrips$tour_mode>=7 & jtrips$tour_mode<=13]-3 -#jtrips$TOURMODE[jtrips$tour_mode>=14 & jtrips$tour_mode<=15] <- 11 -#jtrips$TOURMODE[jtrips$tour_mode==16] <- 12 - -jtrips$TRIPMODE <- jtrips$trip_mode -#jtrips$TRIPMODE[jtrips$trip_mode<=2] <- 1 -#jtrips$TRIPMODE[jtrips$trip_mode>=3 & jtrips$trip_mode<=4] <- 2 -#jtrips$TRIPMODE[jtrips$trip_mode>=5 & jtrips$trip_mode<=6] <- 3 -#jtrips$TRIPMODE[jtrips$trip_mode>=7 & jtrips$trip_mode<=13] <- jtrips$trip_mode[jtrips$trip_mode>=7 & jtrips$trip_mode<=13]-3 -#jtrips$TRIPMODE[jtrips$trip_mode>=14 & jtrips$trip_mode<=15] <- 11 -#jtrips$TRIPMODE[jtrips$trip_mode==16] <- 12 - -#Code joint tour purposes -jtrips$TOURPURP[jtrips$tour_purpose=="Work"] <- 1 -jtrips$TOURPURP[jtrips$tour_purpose=="University"] <- 2 -jtrips$TOURPURP[jtrips$tour_purpose=="School"] <- 3 -jtrips$TOURPURP[jtrips$tour_purpose=="Escort"] <- 4 -jtrips$TOURPURP[jtrips$tour_purpose=="Shop"] <- 5 -jtrips$TOURPURP[jtrips$tour_purpose=="Maintenance"] <- 6 -jtrips$TOURPURP[jtrips$tour_purpose=="Eating Out"] <- 7 -jtrips$TOURPURP[jtrips$tour_purpose=="Visiting"] <- 8 -jtrips$TOURPURP[jtrips$tour_purpose=="Discretionary"] <- 9 -jtrips$TOURPURP[jtrips$tour_purpose=="Work-Based" | jtrips$tour_purpose=="work related"] <- 10 - -jtrips$OPURP[jtrips$orig_purpose=="Home"] <- 0 -jtrips$OPURP[jtrips$orig_purpose=="Work"] <- 1 -jtrips$OPURP[jtrips$orig_purpose=="University"] <- 2 -jtrips$OPURP[jtrips$orig_purpose=="School"] <- 3 -jtrips$OPURP[jtrips$orig_purpose=="Escort"] <- 4 -jtrips$OPURP[jtrips$orig_purpose=="Shop"] <- 5 -jtrips$OPURP[jtrips$orig_purpose=="Maintenance"] <- 6 -jtrips$OPURP[jtrips$orig_purpose=="Eating Out"] <- 7 -jtrips$OPURP[jtrips$orig_purpose=="Visiting"] <- 8 -jtrips$OPURP[jtrips$orig_purpose=="Discretionary"] <- 9 -jtrips$OPURP[jtrips$orig_purpose=="Work-Based" | jtrips$orig_purpose=="work related"] <- 10 - -jtrips$DPURP[jtrips$dest_purpose=="Home"] <- 0 -jtrips$DPURP[jtrips$dest_purpose=="Work"] <- 1 -jtrips$DPURP[jtrips$dest_purpose=="University"] <- 2 -jtrips$DPURP[jtrips$dest_purpose=="School"] <- 3 -jtrips$DPURP[jtrips$dest_purpose=="Escort"] <- 4 -jtrips$DPURP[jtrips$dest_purpose=="Shop"] <- 5 -jtrips$DPURP[jtrips$dest_purpose=="Maintenance"] <- 6 -jtrips$DPURP[jtrips$dest_purpose=="Eating Out"] <- 7 -jtrips$DPURP[jtrips$dest_purpose=="Visiting"] <- 8 -jtrips$DPURP[jtrips$dest_purpose=="Discretionary"] <- 9 -jtrips$DPURP[jtrips$dest_purpose=="Work-Based" | jtrips$dest_purpose=="work related"] <- 10 - -#[0:Mandatory, 1: Indi Non Mand, 3: At Work] -jtrips$TOURCAT[jtrips$tour_purpose=="Work"] <- 0 -jtrips$TOURCAT[jtrips$tour_purpose=="University"] <- 0 -jtrips$TOURCAT[jtrips$tour_purpose=="School"] <- 0 -jtrips$TOURCAT[jtrips$tour_purpose=="Escort"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Shop"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Maintenance"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Eating Out"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Visiting"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Discretionary"] <- 1 -jtrips$TOURCAT[jtrips$tour_purpose=="Work-Based"] <- 2 - -#Mark stops and get other attributes -nr <- nrow(jtrips) -jtrips$inb_next <- 0 -jtrips$inb_next[1:nr-1] <- jtrips$inbound[2:nr] -jtrips$stops[jtrips$DPURP>0 & ((jtrips$inbound==0 & jtrips$inb_next==0) | (jtrips$inbound==1 & jtrips$inb_next==1))] <- 1 -jtrips$stops[is.na(jtrips$stops)] <- 0 - -jtrips$OTAZ <- mazCorrespondence$taz[match(jtrips$orig_mgra, mazCorrespondence$mgra)] -jtrips$DTAZ <- mazCorrespondence$taz[match(jtrips$dest_mgra, mazCorrespondence$mgra)] - -jtrips$TOUROTAZ <- unique_joint_tours$OTAZ[match(jtrips$hh_id*1000+jtrips$tour_id, - unique_joint_tours$hh_id*1000+unique_joint_tours$tour_id)] -jtrips$TOURDTAZ <- unique_joint_tours$DTAZ[match(jtrips$hh_id*1000+jtrips$tour_id, - unique_joint_tours$hh_id*1000+unique_joint_tours$tour_id)] - -#create stops table -jstops <- jtrips[jtrips$stops==1,] - -jstops$finaldestTAZ[jstops$inbound==0] <- jstops$TOURDTAZ[jstops$inbound==0] -jstops$finaldestTAZ[jstops$inbound==1] <- jstops$TOUROTAZ[jstops$inbound==1] - -jstops$oindex<-match(jstops$OTAZ, skimLookUp$Lookup) -jstops$dindex<-match(jstops$finaldestTAZ, skimLookUp$Lookup) -jstops$od_dist <- DST_SKM[cbind(jstops$oindex, jstops$dindex)] - -jstops$oindex2<-match(jstops$OTAZ, skimLookUp$Lookup) -jstops$dindex2<-match(jstops$DTAZ, skimLookUp$Lookup) -jstops$os_dist <- DST_SKM[cbind(jstops$oindex2, jstops$dindex2)] - -jstops$oindex3<-match(jstops$DTAZ, skimLookUp$Lookup) -jstops$dindex3<-match(jstops$finaldestTAZ, skimLookUp$Lookup) -jstops$sd_dist <- DST_SKM[cbind(jstops$oindex3, jstops$dindex3)] - -jstops$out_dir_dist <- jstops$os_dist + jstops$sd_dist - jstops$od_dist - -#--------------------------------------------------------------------------- - -# Recode workrelated tours which are not at work subtour as work tour -#tours$TOURPURP[tours$TOURPURP == 10 & tours$IS_SUBTOUR == 0] <- 1 - -workCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 1") #[excluding at work subtours] -schlCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 2 | TOURPURP == 3") -inmCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP>=4 & TOURPURP<=9") - -# ----------------------- -# added for calibration by nagendra.dhakar@rsginc.com -# for indivudal NM tour generation -workCounts_temp <- workCounts -schlCounts_temp <- schlCounts -inmCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP>4 & TOURPURP<=9") #excluding school escort tours -atWorkCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP == 10") -escortCounts_temp <- count(tours, c("hh_id", "person_num"), "TOURPURP==4") - - -colnames(workCounts_temp)[3] <- "freq_work" -colnames(schlCounts_temp)[3] <- "freq_schl" -colnames(inmCounts_temp)[3] <- "freq_inm" -colnames(atWorkCounts_temp)[3] <- "freq_atwork" -colnames(escortCounts_temp)[3] <- "freq_escort" - -temp <- merge(workCounts_temp, schlCounts_temp, by = c("hh_id", "person_num")) -temp1 <- merge(temp, inmCounts_temp, by = c("hh_id", "person_num")) -temp1$freq_m <- temp1$freq_work + temp1$freq_schl -temp1$freq_itours <- temp1$freq_m+temp1$freq_inm - -#joint tours -#identify persons that made joint tour -#temp_joint <- melt(unique_joint_tours[,c("hh_id","tour_id" ,"PER1", "PER2", "PER3", "PER4", "PER5", "PER6", "PER7", "PER8")], id = c("hh_id","tour_id")) -temp_joint <- melt(unique_joint_tours[,c("hh_id","tour_id" ,"PER1", "PER2", "PER3", "PER4", "PER5", "PER6", "PER7")], id = c("hh_id","tour_id")) -colnames(temp_joint) <- c("hh_id", "tour_id", "var", "person_num") -temp_joint <- as.data.frame(temp_joint) -temp_joint$person_num <- as.integer(temp_joint$person_num) -temp_joint$joint<- 0 -temp_joint$joint[temp_joint$person_num>0] <- 1 - -temp_joint <- temp_joint[temp_joint$joint==1,] -person_unique_joint <- aggregate(joint~hh_id+person_num, temp_joint, sum) - -temp2 <- merge(temp1, person_unique_joint, by = c("hh_id", "person_num"), all = T) -temp2 <- merge(temp2, atWorkCounts_temp, by = c("hh_id", "person_num"), all = T) -temp2 <- merge(temp2, escortCounts_temp, by = c("hh_id", "person_num"), all = T) -temp2[is.na(temp2)] <- 0 - -#add number of joint tours to non-mandatory -temp2$freq_nm <- temp2$freq_inm + temp2$joint - -#get person type -temp2$PERTYPE <- per$PERTYPE[match(temp2$hh_id*10+temp2$person_num,per$hh_id*10+per$person_num)] - -#total tours -temp2$total_tours <- temp2$freq_nm+temp2$freq_m+temp2$freq_atwork+temp2$freq_escort - -persons_mand <- temp2[temp2$freq_m>0,] #persons with atleast 1 mandatory tours -persons_nomand <- temp2[temp2$freq_m==0,] #active persons with 0 mandatory tours - -freq_nmtours_mand <- count(persons_mand, c("PERTYPE","freq_nm")) -freq_nmtours_nomand <- count(persons_nomand, c("PERTYPE","freq_nm")) -test <- count(temp2, c("PERTYPE","freq_inm","freq_m","freq_nm","freq_atwork","freq_escort")) -write.csv(test, "tour_rate_debug.csv", row.names = F) -write.csv(temp2,"temp2.csv", row.names = F) - -write.table("Non-Mandatory Tours for Persons with at-least 1 Mandatory Tour", "indivNMTourFreq.csv", sep = ",", row.names = F, append = F) -write.table(freq_nmtours_mand, "indivNMTourFreq.csv", sep = ",", row.names = F, append = T) -write.table("Non-Mandatory Tours for Active Persons with 0 Mandatory Tour", "indivNMTourFreq.csv", sep = ",", row.names = F, append = T) -write.table(freq_nmtours_nomand, "indivNMTourFreq.csv", sep = ",", row.names = F, append = TRUE) - -# end of addition for calibration -# ----------------------- - - -# ---------------------- -# added for calibration by nagendra.dhakar@rsginc.com - -i4tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 4") -i5tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 5") -i6tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 6") -i7tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 7") -i8tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 8") -i9tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP == 9") - -# end of addition for calibration -# ----------------------- - -tourCounts <- count(tours, c("hh_id", "person_num"), "TOURPURP <= 9") #number of tours per person [excluding at work subtours] -joint5 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==5") -joint6 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==6") -joint7 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==7") -joint8 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==8") -joint9 <- count(unique_joint_tours, c("hh_id"), "JOINT_PURP==9") - -hh$joint5 <- joint5$freq[match(hh$hh_id, joint5$hh_id)] -hh$joint6 <- joint6$freq[match(hh$hh_id, joint6$hh_id)] -hh$joint7 <- joint7$freq[match(hh$hh_id, joint7$hh_id)] -hh$joint8 <- joint8$freq[match(hh$hh_id, joint8$hh_id)] -hh$joint9 <- joint9$freq[match(hh$hh_id, joint9$hh_id)] -hh$jtours <- hh$joint5+hh$joint6+hh$joint7+hh$joint8+hh$joint9 - -hh$joint5[is.na(hh$joint5)] <- 0 -hh$joint6[is.na(hh$joint6)] <- 0 -hh$joint7[is.na(hh$joint7)] <- 0 -hh$joint8[is.na(hh$joint8)] <- 0 -hh$joint9[is.na(hh$joint9)] <- 0 -hh$jtours[is.na(hh$jtours)] <- 0 - -#joint tour indicator -hh$JOINT <- 0 -hh$JOINT[substr(hh$cdap_pattern, nchar(as.character(hh$cdap_pattern)), nchar(as.character(hh$cdap_pattern)))=="j"] <- 1 - -# code JTF category -hh$jtf[hh$jtours==0] <- 1 -hh$jtf[hh$joint5==1] <- 2 -hh$jtf[hh$joint6==1] <- 3 -hh$jtf[hh$joint7==1] <- 4 -hh$jtf[hh$joint8==1] <- 5 -hh$jtf[hh$joint9==1] <- 6 - -hh$jtf[hh$joint5>=2] <- 7 -hh$jtf[hh$joint6>=2] <- 8 -hh$jtf[hh$joint7>=2] <- 9 -hh$jtf[hh$joint8>=2] <- 10 -hh$jtf[hh$joint9>=2] <- 11 - -hh$jtf[hh$joint5>=1 & hh$joint6>=1] <- 12 -hh$jtf[hh$joint5>=1 & hh$joint7>=1] <- 13 -hh$jtf[hh$joint5>=1 & hh$joint8>=1] <- 14 -hh$jtf[hh$joint5>=1 & hh$joint9>=1] <- 15 - -hh$jtf[hh$joint6>=1 & hh$joint7>=1] <- 16 -hh$jtf[hh$joint6>=1 & hh$joint8>=1] <- 17 -hh$jtf[hh$joint6>=1 & hh$joint9>=1] <- 18 - -hh$jtf[hh$joint7>=1 & hh$joint8>=1] <- 19 -hh$jtf[hh$joint7>=1 & hh$joint9>=1] <- 20 - -hh$jtf[hh$joint8>=1 & hh$joint9>=1] <- 21 - -per$workTours <- workCounts$freq[match(per$hh_id*10+per$person_num, workCounts$hh_id*10+workCounts$person_num)] -per$schlTours <- schlCounts$freq[match(per$hh_id*10+per$person_num, schlCounts$hh_id*10+schlCounts$person_num)] -per$inmTours <- inmCounts$freq[match(per$hh_id*10+per$person_num, inmCounts$hh_id*10+inmCounts$person_num)] -per$inmTours[is.na(per$inmTours)] <- 0 -per$numTours <- tourCounts$freq[match(per$hh_id*10+per$person_num, tourCounts$hh_id*10+tourCounts$person_num)] -per$numTours[is.na(per$numTours)] <- 0 - -# --------------------------------------------------- -# added for calibration by nagendra.dhakar@rsginc.com - -per$i4numTours <- i4tourCounts$freq[match(per$hh_id*10+per$person_num, i4tourCounts$hh_id*10+i4tourCounts$person_num)] -per$i4numTours[is.na(per$i4numTours)] <- 0 -per$i5numTours <- i5tourCounts$freq[match(per$hh_id*10+per$person_num, i5tourCounts$hh_id*10+i5tourCounts$person_num)] -per$i5numTours[is.na(per$i5numTours)] <- 0 -per$i6numTours <- i6tourCounts$freq[match(per$hh_id*10+per$person_num, i6tourCounts$hh_id*10+i6tourCounts$person_num)] -per$i6numTours[is.na(per$i6numTours)] <- 0 -per$i7numTours <- i7tourCounts$freq[match(per$hh_id*10+per$person_num, i7tourCounts$hh_id*10+i7tourCounts$person_num)] -per$i7numTours[is.na(per$i7numTours)] <- 0 -per$i8numTours <- i8tourCounts$freq[match(per$hh_id*10+per$person_num, i8tourCounts$hh_id*10+i8tourCounts$person_num)] -per$i8numTours[is.na(per$i8numTours)] <- 0 -per$i9numTours <- i9tourCounts$freq[match(per$hh_id*10+per$person_num, i9tourCounts$hh_id*10+i9tourCounts$person_num)] -per$i9numTours[is.na(per$i9numTours)] <- 0 - -# end of addition for calibration -# --------------------------------------------------- - -# Total tours by person type -per$numTours[is.na(per$numTours)] <- 0 -toursPertypeDistbn <- count(tours[tours$PERTYPE>0 & tours$TOURPURP!=10,], c("PERTYPE")) -write.csv(toursPertypeDistbn, "toursPertypeDistbn.csv", row.names = TRUE) - -# count joint tour fr each person type -temp_joint <- melt(unique_joint_tours[, c("hh_id","tour_id","PTYPE1","PTYPE2","PTYPE3","PTYPE4","PTYPE5","PTYPE6","PTYPE7","PTYPE8")], id = c("hh_id", "tour_id")) -names(temp_joint)[names(temp_joint)=="value"] <- "PERTYPE" -jtoursPertypeDistbn <- count(temp_joint[temp_joint$PERTYPE>0,], c("PERTYPE")) - -# Total tours by person type for visualizer -totaltoursPertypeDistbn <- toursPertypeDistbn -totaltoursPertypeDistbn$freq <- totaltoursPertypeDistbn$freq + jtoursPertypeDistbn$freq -write.csv(totaltoursPertypeDistbn, "total_tours_by_pertype_vis.csv", row.names = F) - -# Total indi NM tours by person type and purpose -tours_pertype_purpose <- count(tours[tours$TOURPURP>=4 & tours$TOURPURP<=9,], c("PERTYPE", "TOURPURP")) -write.csv(tours_pertype_purpose, "tours_pertype_purpose.csv", row.names = TRUE) - -# --------------------------------------------------- -# added for calibration by nagendra.dhakar@rsginc.com - -# code indi NM tour category -per$i4numTours[per$i4numTours>=2] <- 2 -per$i5numTours[per$i5numTours>=2] <- 2 -per$i6numTours[per$i6numTours>=2] <- 2 -per$i7numTours[per$i7numTours>=1] <- 1 -per$i8numTours[per$i8numTours>=1] <- 1 -per$i9numTours[per$i9numTours>=2] <- 2 - -tours_pertype_esco <- count(per, c("PERTYPE", "i4numTours")) -tours_pertype_shop <- count(per, c("PERTYPE", "i5numTours")) -tours_pertype_main <- count(per, c("PERTYPE", "i6numTours")) -tours_pertype_eati <- count(per, c("PERTYPE", "i7numTours")) -tours_pertype_visi <- count(per, c("PERTYPE", "i8numTours")) -tours_pertype_disc <- count(per, c("PERTYPE", "i9numTours")) - - -colnames(tours_pertype_esco) <- c("PERTYPE","inumTours","freq") -colnames(tours_pertype_shop) <- c("PERTYPE","inumTours","freq") -colnames(tours_pertype_main) <- c("PERTYPE","inumTours","freq") -colnames(tours_pertype_eati) <- c("PERTYPE","inumTours","freq") -colnames(tours_pertype_visi) <- c("PERTYPE","inumTours","freq") -colnames(tours_pertype_disc) <- c("PERTYPE","inumTours","freq") - -tours_pertype_esco$purpose <- 4 -tours_pertype_shop$purpose <- 5 -tours_pertype_main$purpose <- 6 -tours_pertype_eati$purpose <- 7 -tours_pertype_visi$purpose <- 8 -tours_pertype_disc$purpose <- 9 - -indi_nm_tours_pertype <- rbind(tours_pertype_esco,tours_pertype_shop,tours_pertype_main,tours_pertype_eati,tours_pertype_visi,tours_pertype_disc) -write.csv(indi_nm_tours_pertype, "inmtours_pertype_purpose.csv", row.names = F) - -# end of addition for calibration -# --------------------------------------------------- - -tours_pertype_purpose <- xtabs(freq~PERTYPE+TOURPURP, tours_pertype_purpose) -tours_pertype_purpose[is.na(tours_pertype_purpose)] <- 0 -tours_pertype_purpose <- addmargins(as.table(tours_pertype_purpose)) -tours_pertype_purpose <- as.data.frame.matrix(tours_pertype_purpose) - -totalPersons <- sum(pertypeDistbn$freq) -totalPersons_DF <- data.frame("Total", totalPersons) -colnames(totalPersons_DF) <- colnames(pertypeDistbn) -pertypeDF <- rbind(pertypeDistbn, totalPersons_DF) -nm_tour_rates <- tours_pertype_purpose/pertypeDF$freq -nm_tour_rates$pertype <- row.names(nm_tour_rates) -nm_tour_rates <- melt(nm_tour_rates, id = c("pertype")) -colnames(nm_tour_rates) <- c("pertype", "tour_purp", "tour_rate") -nm_tour_rates$pertype <- as.character(nm_tour_rates$pertype) -nm_tour_rates$tour_purp <- as.character(nm_tour_rates$tour_purp) -nm_tour_rates$pertype[nm_tour_rates$pertype=="Sum"] <- "All" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp=="Sum"] <- "All" -nm_tour_rates$pertype <- pertypeCodes$name[match(nm_tour_rates$pertype, pertypeCodes$code)] - -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==4] <- "Escorting" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==5] <- "Shopping" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==6] <- "Maintenance" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==7] <- "EatingOut" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==8] <- "Visiting" -nm_tour_rates$tour_purp[nm_tour_rates$tour_purp==9] <- "Discretionary" - -write.csv(nm_tour_rates, "nm_tour_rates.csv", row.names = F) - -# Total tours by purpose X tourtype -t1 <- hist(tours$TOURPURP[tours$TOURPURP<10], breaks = seq(1,10, by=1), freq = NULL, right=FALSE) -t3 <- hist(unique_joint_tours$JOINT_PURP, breaks = seq(1,10, by=1), freq = NULL, right=FALSE) -tours_purpose_type <- data.frame(t1$counts, t3$counts) -colnames(tours_purpose_type) <- c("indi", "joint") -write.csv(tours_purpose_type, "tours_purpose_type.csv", row.names = TRUE) - -# DAP by pertype -# recode pattern type for at-work and home schooling persons. -# these person have DAP as M. They should be recoded to N or H. -per[per$activity_pattern == 'M' & per$imf_choice==0 & per$inmf_choice>0]$activity_pattern = 'N' -per[per$activity_pattern == 'M' & per$imf_choice==0 & per$inmf_choice==0]$activity_pattern = 'H' - -dapSummary <- count(per, c("PERTYPE", "activity_pattern")) -write.csv(dapSummary, "dapSummary.csv", row.names = TRUE) - -# Prepare DAP summary for visualizer -dapSummary_vis <- xtabs(freq~PERTYPE+activity_pattern, dapSummary) -dapSummary_vis <- addmargins(as.table(dapSummary_vis)) -dapSummary_vis <- as.data.frame.matrix(dapSummary_vis) - -dapSummary_vis$id <- row.names(dapSummary_vis) -dapSummary_vis <- melt(dapSummary_vis, id = c("id")) -colnames(dapSummary_vis) <- c("PERTYPE", "DAP", "freq") -dapSummary_vis$PERTYPE <- as.character(dapSummary_vis$PERTYPE) -dapSummary_vis$DAP <- as.character(dapSummary_vis$DAP) -dapSummary_vis <- dapSummary_vis[dapSummary_vis$DAP!="Sum",] -dapSummary_vis$PERTYPE[dapSummary_vis$PERTYPE=="Sum"] <- "Total" -write.csv(dapSummary_vis, "dapSummary_vis.csv", row.names = TRUE) - -# HHSize X Joint -hhsizeJoint <- count(hh[hh$HHSIZE>=2,], c("HHSIZE", "JOINT")) -write.csv(hhsizeJoint, "hhsizeJoint.csv", row.names = TRUE) - -#mandatory tour frequency -mtfSummary <- count(per[per$imf_choice > 0,], c("PERTYPE", "imf_choice")) -write.csv(mtfSummary, "mtfSummary.csv") -#write.csv(tours, "tours_test.csv") - -# Prepare MTF summary for visualizer -mtfSummary_vis <- xtabs(freq~PERTYPE+imf_choice, mtfSummary) -mtfSummary_vis <- addmargins(as.table(mtfSummary_vis)) -mtfSummary_vis <- as.data.frame.matrix(mtfSummary_vis) - -mtfSummary_vis$id <- row.names(mtfSummary_vis) -mtfSummary_vis <- melt(mtfSummary_vis, id = c("id")) -colnames(mtfSummary_vis) <- c("PERTYPE", "MTF", "freq") -mtfSummary_vis$PERTYPE <- as.character(mtfSummary_vis$PERTYPE) -mtfSummary_vis$MTF <- as.character(mtfSummary_vis$MTF) -mtfSummary_vis <- mtfSummary_vis[mtfSummary_vis$MTF!="Sum",] -mtfSummary_vis$PERTYPE[mtfSummary_vis$PERTYPE=="Sum"] <- "Total" -write.csv(mtfSummary_vis, "mtfSummary_vis.csv") - -# indi NM summary -inm0Summary <- count(per[per$inmTours==0,], c("PERTYPE")) -inm1Summary <- count(per[per$inmTours==1,], c("PERTYPE")) -inm2Summary <- count(per[per$inmTours==2,], c("PERTYPE")) -inm3Summary <- count(per[per$inmTours>=3,], c("PERTYPE")) - -inmSummary <- data.frame(PERTYPE = c(1,2,3,4,5,6,7,8)) -inmSummary$tour0 <- inm0Summary$freq[match(inmSummary$PERTYPE, inm0Summary$PERTYPE)] -inmSummary$tour1 <- inm1Summary$freq[match(inmSummary$PERTYPE, inm1Summary$PERTYPE)] -inmSummary$tour2 <- inm2Summary$freq[match(inmSummary$PERTYPE, inm2Summary$PERTYPE)] -inmSummary$tour3pl <- inm3Summary$freq[match(inmSummary$PERTYPE, inm3Summary$PERTYPE)] - -write.table(inmSummary, "innmSummary.csv", col.names=TRUE, sep=",") - -# prepare INM summary for visualizer -inmSummary_vis <- melt(inmSummary, id=c("PERTYPE")) -inmSummary_vis$variable <- as.character(inmSummary_vis$variable) -inmSummary_vis$variable[inmSummary_vis$variable=="tour0"] <- "0" -inmSummary_vis$variable[inmSummary_vis$variable=="tour1"] <- "1" -inmSummary_vis$variable[inmSummary_vis$variable=="tour2"] <- "2" -inmSummary_vis$variable[inmSummary_vis$variable=="tour3pl"] <- "3pl" -inmSummary_vis <- xtabs(value~PERTYPE+variable, inmSummary_vis) -inmSummary_vis <- addmargins(as.table(inmSummary_vis)) -inmSummary_vis <- as.data.frame.matrix(inmSummary_vis) - -inmSummary_vis$id <- row.names(inmSummary_vis) -inmSummary_vis <- melt(inmSummary_vis, id = c("id")) -colnames(inmSummary_vis) <- c("PERTYPE", "nmtours", "freq") -inmSummary_vis$PERTYPE <- as.character(inmSummary_vis$PERTYPE) -inmSummary_vis$nmtours <- as.character(inmSummary_vis$nmtours) -inmSummary_vis <- inmSummary_vis[inmSummary_vis$nmtours!="Sum",] -inmSummary_vis$PERTYPE[inmSummary_vis$PERTYPE=="Sum"] <- "Total" -write.csv(inmSummary_vis, "inmSummary_vis.csv") - -# Joint Tour Frequency and composition -jtfSummary <- count(hh[!is.na(hh$jtf),], c("jtf")) -jointComp <- count(unique_joint_tours, c("tour_composition")) -jointPartySize <- count(unique_joint_tours, c("NUMBER_HH")) -jointCompPartySize <- count(unique_joint_tours, c("tour_composition","NUMBER_HH")) - -hh$jointCat[hh$jtours==0] <- 0 -hh$jointCat[hh$jtours==1] <- 1 -hh$jointCat[hh$jtours>=2] <- 2 - -jointToursHHSize <- count(hh[!is.na(hh$HHSIZE) & !is.na(hh$jointCat),], c("HHSIZE", "jointCat")) - -write.table(jtfSummary, "jtfSummary.csv", col.names=TRUE, sep=",") -write.table(jointComp, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) -write.table(jointPartySize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) -write.table(jointCompPartySize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) -write.table(jointToursHHSize, "jtfSummary.csv", col.names=TRUE, sep=",", append=TRUE) - -#cap joint party size to 5+ -jointPartySize$freq[jointPartySize$NUMBER_HH==5] <- sum(jointPartySize$freq[jointPartySize$NUMBER_HH>=5]) -jointPartySize <- jointPartySize[jointPartySize$NUMBER_HH<=5, ] - -jtf <- data.frame(jtf_code = seq(from = 1, to = 21), - alt_name = c("No Joint Tours", "1 Shopping", "1 Maintenance", "1 Eating Out", "1 Visiting", "1 Other Discretionary", - "2 Shopping", "1 Shopping / 1 Maintenance", "1 Shopping / 1 Eating Out", "1 Shopping / 1 Visiting", - "1 Shopping / 1 Other Discretionary", "2 Maintenance", "1 Maintenance / 1 Eating Out", - "1 Maintenance / 1 Visiting", "1 Maintenance / 1 Other Discretionary", "2 Eating Out", "1 Eating Out / 1 Visiting", - "1 Eating Out / 1 Other Discretionary", "2 Visiting", "1 Visiting / 1 Other Discretionary", "2 Other Discretionary")) -jtf$freq <- jtfSummary$freq[match(jtf$jtf_code, jtfSummary$jtf)] -jtf[is.na(jtf)] <- 0 - -jointComp$tour_composition[jointComp$tour_composition==1] <- "All Adult" -jointComp$tour_composition[jointComp$tour_composition==2] <- "All Children" -jointComp$tour_composition[jointComp$tour_composition==3] <- "Mixed" - -jointToursHHSizeProp <- xtabs(freq~jointCat+HHSIZE, jointToursHHSize[jointToursHHSize$HHSIZE>1,]) -jointToursHHSizeProp <- addmargins(as.table(jointToursHHSizeProp)) -jointToursHHSizeProp <- jointToursHHSizeProp[-4,] #remove last row -jointToursHHSizeProp <- prop.table(jointToursHHSizeProp, margin = 2) -jointToursHHSizeProp <- as.data.frame.matrix(jointToursHHSizeProp) -jointToursHHSizeProp <- jointToursHHSizeProp*100 -jointToursHHSizeProp$jointTours <- row.names(jointToursHHSizeProp) -jointToursHHSizeProp <- melt(jointToursHHSizeProp, id = c("jointTours")) -colnames(jointToursHHSizeProp) <- c("jointTours", "hhsize", "freq") -jointToursHHSizeProp$hhsize <- as.character(jointToursHHSizeProp$hhsize) -jointToursHHSizeProp$hhsize[jointToursHHSizeProp$hhsize=="Sum"] <- "Total" - -jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==1] <- "All Adult" -jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==2] <- "All Children" -jointCompPartySize$tour_composition[jointCompPartySize$tour_composition==3] <- "Mixed" - -jointCompPartySizeProp <- xtabs(freq~tour_composition+NUMBER_HH, jointCompPartySize) -jointCompPartySizeProp <- addmargins(as.table(jointCompPartySizeProp)) -jointCompPartySizeProp <- jointCompPartySizeProp[,-6] #remove last row -jointCompPartySizeProp <- prop.table(jointCompPartySizeProp, margin = 1) -jointCompPartySizeProp <- as.data.frame.matrix(jointCompPartySizeProp) -jointCompPartySizeProp <- jointCompPartySizeProp*100 -jointCompPartySizeProp$comp <- row.names(jointCompPartySizeProp) -jointCompPartySizeProp <- melt(jointCompPartySizeProp, id = c("comp")) -colnames(jointCompPartySizeProp) <- c("comp", "partysize", "freq") -jointCompPartySizeProp$comp <- as.character(jointCompPartySizeProp$comp) -jointCompPartySizeProp$comp[jointCompPartySizeProp$comp=="Sum"] <- "Total" - -# Cap joint comp party size at 5 -jointCompPartySizeProp <- jointCompPartySizeProp[jointCompPartySizeProp$partysize!="Sum",] -jointCompPartySizeProp$partysize <- as.numeric(as.character(jointCompPartySizeProp$partysize)) -jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Adult" & jointCompPartySizeProp$partysize==5] <- - sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Adult" & jointCompPartySizeProp$partysize>=5]) -jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Children" & jointCompPartySizeProp$partysize==5] <- - sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="All Children" & jointCompPartySizeProp$partysize>=5]) -jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Mixed" & jointCompPartySizeProp$partysize==5] <- - sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Mixed" & jointCompPartySizeProp$partysize>=5]) -jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Total" & jointCompPartySizeProp$partysize==5] <- - sum(jointCompPartySizeProp$freq[jointCompPartySizeProp$comp=="Total" & jointCompPartySizeProp$partysize>=5]) - -jointCompPartySizeProp <- jointCompPartySizeProp[jointCompPartySizeProp$partysize<=5,] - - -write.csv(jtf, "jtf.csv", row.names = F) -write.csv(jointComp, "jointComp.csv", row.names = F) -write.csv(jointPartySize, "jointPartySize.csv", row.names = F) -write.csv(jointCompPartySizeProp, "jointCompPartySize.csv", row.names = F) -write.csv(jointToursHHSizeProp, "jointToursHHSize.csv", row.names = F) - -# TOD Profile -#work.dep <- table(cut(tours$ANCHOR_DEPART_BIN[!is.na(tours$ANCHOR_DEPART_BIN)], seq(1,48, by=1), right=FALSE)) - -tod1 <- hist(tours$start_period[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod1_2 <- hist(tours$start_period[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod2 <- hist(tours$start_period[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod3 <- hist(tours$start_period[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod4 <- hist(tours$start_period[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi56 <- hist(tours$start_period[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi789 <- hist(tours$start_period[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod6 <- hist(tours$start_period[tours$TOURPURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod7 <- hist(tours$start_period[tours$TOURPURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod8 <- hist(tours$start_period[tours$TOURPURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod9 <- hist(tours$start_period[tours$TOURPURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todj56 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todj789 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod11 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod12 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod13 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod14 <- hist(unique_joint_tours$start_period[unique_joint_tours$JOINT_PURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod15 <- hist(tours$start_period[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) - -todDepProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts - , todj56$counts, todj789$counts, tod15$counts) -colnames(todDepProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", - "jmain", "jdisc", "atwork") -write.csv(todDepProfile, "todDepProfile.csv") - -# prepare input for visualizer -todDepProfile_vis <- todDepProfile -todDepProfile_vis$id <- row.names(todDepProfile_vis) -todDepProfile_vis <- melt(todDepProfile_vis, id = c("id")) -colnames(todDepProfile_vis) <- c("id", "purpose", "freq_dep") - -todDepProfile_vis$purpose <- as.character(todDepProfile_vis$purpose) -todDepProfile_vis <- xtabs(freq_dep~id+purpose, todDepProfile_vis) -todDepProfile_vis <- addmargins(as.table(todDepProfile_vis)) -todDepProfile_vis <- as.data.frame.matrix(todDepProfile_vis) -todDepProfile_vis$id <- row.names(todDepProfile_vis) -todDepProfile_vis <- melt(todDepProfile_vis, id = c("id")) -colnames(todDepProfile_vis) <- c("timebin", "PURPOSE", "freq") -todDepProfile_vis$PURPOSE <- as.character(todDepProfile_vis$PURPOSE) -todDepProfile_vis$timebin <- as.character(todDepProfile_vis$timebin) -todDepProfile_vis <- todDepProfile_vis[todDepProfile_vis$timebin!="Sum",] -todDepProfile_vis$PURPOSE[todDepProfile_vis$PURPOSE=="Sum"] <- "Total" -todDepProfile_vis$timebin <- as.numeric(todDepProfile_vis$timebin) - -tod1 <- hist(tours$end_period[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod1_2 <- hist(tours$end_period[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod2 <- hist(tours$end_period[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod3 <- hist(tours$end_period[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod4 <- hist(tours$end_period[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi56 <- hist(tours$end_period[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi789 <- hist(tours$end_period[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod8 <- hist(tours$end_period[tours$TOURPURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod9 <- hist(tours$end_period[tours$TOURPURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todj56 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todj789 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod11 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod12 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==7], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod13 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==8], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod14 <- hist(unique_joint_tours$end_period[unique_joint_tours$JOINT_PURP==9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod15 <- hist(tours$end_period[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) - -todArrProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts - , todj56$counts, todj789$counts, tod15$counts) -colnames(todArrProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", - "jmain", "jdisc", "atwork") -write.csv(todArrProfile, "todArrProfile.csv") - - -##stops by direction, purpose and model tod - -tours$start_tod <- 5 # EA: 3 am - 6 am -tours$start_tod <- ifelse(tours$start_period>=4 & tours$start_period<=9, 1, tours$start_tod) # AM: 6 am - 9 am -tours$start_tod <- ifelse(tours$start_period>=10 & tours$start_period<=22, 2, tours$start_tod) # MD: 9 am - 3:30 pm -tours$start_tod <- ifelse(tours$start_period>=23 & tours$start_period<=29, 3, tours$start_tod) # PM: 3:30 pm - 7 pm -tours$start_tod <- ifelse(tours$start_period>=30 & tours$start_period<=40, 4, tours$start_tod) # EV: 7 pm - 3 am - -tours$end_tod <- 5 # EA: 3 am - 6 am -tours$end_tod <- ifelse(tours$end_period>=4 & tours$end_period<=9, 1, tours$end_tod) # AM: 6 am - 9 am -tours$end_tod <- ifelse(tours$end_period>=10 & tours$end_period<=22, 2, tours$end_tod) # MD: 9 am - 3:30 pm -tours$end_tod <- ifelse(tours$end_period>=23 & tours$end_period<=29, 3, tours$end_tod) # PM: 3:30 pm - 7 pm -tours$end_tod <- ifelse(tours$end_period>=30 & tours$end_period<=40, 4, tours$end_tod) # EV: 7 pm - 3 am - -stops_ib_tod <- aggregate(num_ib_stops~tour_purpose+start_tod+end_tod, data=tours, FUN = sum) -stops_ob_tod <- aggregate(num_ob_stops~tour_purpose+start_tod+end_tod, data=tours, FUN = sum) -write.csv(stops_ib_tod, "todStopsIB.csv", row.names = F) -write.csv(stops_ob_tod, "todStopsOB.csv", row.names = F) - -#joint tours -unique_joint_tours$start_tod <- 5 # EA: 3 am - 6 am -unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=4 & unique_joint_tours$start_period<=9, 1, unique_joint_tours$start_tod) # AM: 6 am - 9 am -unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=10 & unique_joint_tours$start_period<=22, 2, unique_joint_tours$start_tod) # MD: 9 am - 3:30 pm -unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=23 & unique_joint_tours$start_period<=29, 3, unique_joint_tours$start_tod) # PM: 3:30 pm - 7 pm -unique_joint_tours$start_tod <- ifelse(unique_joint_tours$start_period>=30 & unique_joint_tours$start_period<=40, 4, unique_joint_tours$start_tod) # EV: 7 pm - 3 am - -unique_joint_tours$end_tod <- 5 # EA: 3 am - 6 am -unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=4 & unique_joint_tours$end_period<=9, 1, unique_joint_tours$end_tod) # AM: 6 am - 9 am -unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=10 & unique_joint_tours$end_period<=22, 2, unique_joint_tours$end_tod) # MD: 9 am - 3:30 pm -unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=23 & unique_joint_tours$end_period<=29, 3, unique_joint_tours$end_tod) # PM: 3:30 pm - 7 pm -unique_joint_tours$end_tod <- ifelse(unique_joint_tours$end_period>=30 & unique_joint_tours$end_period<=40, 4, unique_joint_tours$end_tod) # EV: 7 pm - 3 am - -jstops_ib_tod <- aggregate(num_ib_stops~tour_purpose+start_tod+end_tod, data=unique_joint_tours, FUN = sum) -jstops_ob_tod <- aggregate(num_ob_stops~tour_purpose+start_tod+end_tod, data=unique_joint_tours, FUN = sum) -write.csv(jstops_ib_tod, "todStopsIB_joint.csv", row.names = F) -write.csv(jstops_ob_tod, "todStopsOB_joint.csv", row.names = F) - -# prepare input for visualizer -todArrProfile_vis <- todArrProfile -todArrProfile_vis$id <- row.names(todArrProfile_vis) -todArrProfile_vis <- melt(todArrProfile_vis, id = c("id")) -colnames(todArrProfile_vis) <- c("id", "purpose", "freq_arr") - -todArrProfile_vis$purpose <- as.character(todArrProfile_vis$purpose) -todArrProfile_vis <- xtabs(freq_arr~id+purpose, todArrProfile_vis) -todArrProfile_vis <- addmargins(as.table(todArrProfile_vis)) -todArrProfile_vis <- as.data.frame.matrix(todArrProfile_vis) -todArrProfile_vis$id <- row.names(todArrProfile_vis) -todArrProfile_vis <- melt(todArrProfile_vis, id = c("id")) -colnames(todArrProfile_vis) <- c("timebin", "PURPOSE", "freq") -todArrProfile_vis$PURPOSE <- as.character(todArrProfile_vis$PURPOSE) -todArrProfile_vis$timebin <- as.character(todArrProfile_vis$timebin) -todArrProfile_vis <- todArrProfile_vis[todArrProfile_vis$timebin!="Sum",] -todArrProfile_vis$PURPOSE[todArrProfile_vis$PURPOSE=="Sum"] <- "Total" -todArrProfile_vis$timebin <- as.numeric(todArrProfile_vis$timebin) - - -tod1 <- hist(tours$tourdur[tours$TOURPURP==1], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod1_2 <- hist(tours$tourdur[tours$TOURPURP==1 & tours$PERTYPE==2], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -tod2 <- hist(tours$tourdur[tours$TOURPURP==2], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod3 <- hist(tours$tourdur[tours$TOURPURP==3], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -tod4 <- hist(tours$tourdur[tours$TOURPURP==4], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi56 <- hist(tours$tourdur[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todi789 <- hist(tours$tourdur[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod6 <- hist(tours$tourdur[tours$TOURPURP==6], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod7 <- hist(tours$tourdur[tours$TOURPURP==7], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod8 <- hist(tours$tourdur[tours$TOURPURP==8], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod9 <- hist(tours$tourdur[tours$TOURPURP==9], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -todj56 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -todj789 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) -#tod11 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==6], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod12 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==7], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod13 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==8], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -#tod14 <- hist(unique_joint_tours$tourdur[unique_joint_tours$JOINT_PURP==9], breaks = seq(0,41, by=1), freq = NULL, right=FALSE) -tod15 <- hist(tours$tourdur[tours$TOURPURP==10], breaks = seq(1,41, by=1), freq = NULL, right=FALSE) - -todDurProfile <- data.frame(tod1$counts, tod2$counts, tod3$counts, tod4$counts, todi56$counts, todi789$counts - , todj56$counts, todj789$counts, tod15$counts) -colnames(todDurProfile) <- c("work", "univ", "sch", "esc", "imain", "idisc", - "jmain", "jdisc", "atwork") -write.csv(todDurProfile, "todDurProfile.csv") - -# prepare input for visualizer -todDurProfile_vis <- todDurProfile -todDurProfile_vis$id <- row.names(todDurProfile_vis) -todDurProfile_vis <- melt(todDurProfile_vis, id = c("id")) -colnames(todDurProfile_vis) <- c("id", "purpose", "freq_dur") - -todDurProfile_vis$purpose <- as.character(todDurProfile_vis$purpose) -todDurProfile_vis <- xtabs(freq_dur~id+purpose, todDurProfile_vis) -todDurProfile_vis <- addmargins(as.table(todDurProfile_vis)) -todDurProfile_vis <- as.data.frame.matrix(todDurProfile_vis) -todDurProfile_vis$id <- row.names(todDurProfile_vis) -todDurProfile_vis <- melt(todDurProfile_vis, id = c("id")) -colnames(todDurProfile_vis) <- c("timebin", "PURPOSE", "freq") -todDurProfile_vis$PURPOSE <- as.character(todDurProfile_vis$PURPOSE) -todDurProfile_vis$timebin <- as.character(todDurProfile_vis$timebin) -todDurProfile_vis <- todDurProfile_vis[todDurProfile_vis$timebin!="Sum",] -todDurProfile_vis$PURPOSE[todDurProfile_vis$PURPOSE=="Sum"] <- "Total" -todDurProfile_vis$timebin <- as.numeric(todDurProfile_vis$timebin) - -todDepProfile_vis <- todDepProfile_vis[order(todDepProfile_vis$timebin, todDepProfile_vis$PURPOSE), ] -todArrProfile_vis <- todArrProfile_vis[order(todArrProfile_vis$timebin, todArrProfile_vis$PURPOSE), ] -todDurProfile_vis <- todDurProfile_vis[order(todDurProfile_vis$timebin, todDurProfile_vis$PURPOSE), ] -todProfile_vis <- data.frame(todDepProfile_vis, todArrProfile_vis$freq, todDurProfile_vis$freq) -colnames(todProfile_vis) <- c("id", "purpose", "freq_dep", "freq_arr", "freq_dur") -write.csv(todProfile_vis, "todProfile_vis.csv", row.names = F) - -# Tour Mode X Auto Suff (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) -tmode1_as0 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode2_as0 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode3_as0 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode4_as0 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode5_as0 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode6_as0 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode7_as0 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode8_as0 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==0], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tmodeAS0Profile <- data.frame(tmode1_as0$counts, tmode2_as0$counts, tmode3_as0$counts, tmode4_as0$counts, - tmode5_as0$counts, tmode6_as0$counts, tmode7_as0$counts, tmode8_as0$counts) -colnames(tmodeAS0Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(tmodeAS0Profile, "tmodeAS0Profile.csv") - -# Prepeare data for visualizer (changed from 9 to 12) -tmodeAS0Profile_vis <- tmodeAS0Profile[1:13,] -tmodeAS0Profile_vis$id <- row.names(tmodeAS0Profile_vis) -tmodeAS0Profile_vis <- melt(tmodeAS0Profile_vis, id = c("id")) -colnames(tmodeAS0Profile_vis) <- c("id", "purpose", "freq_as0") - -tmodeAS0Profile_vis <- xtabs(freq_as0~id+purpose, tmodeAS0Profile_vis) -tmodeAS0Profile_vis[is.na(tmodeAS0Profile_vis)] <- 0 -tmodeAS0Profile_vis <- addmargins(as.table(tmodeAS0Profile_vis)) -tmodeAS0Profile_vis <- as.data.frame.matrix(tmodeAS0Profile_vis) - -tmodeAS0Profile_vis$id <- row.names(tmodeAS0Profile_vis) -tmodeAS0Profile_vis <- melt(tmodeAS0Profile_vis, id = c("id")) -colnames(tmodeAS0Profile_vis) <- c("id", "purpose", "freq_as0") -tmodeAS0Profile_vis$id <- as.character(tmodeAS0Profile_vis$id) -tmodeAS0Profile_vis$purpose <- as.character(tmodeAS0Profile_vis$purpose) -tmodeAS0Profile_vis <- tmodeAS0Profile_vis[tmodeAS0Profile_vis$id!="Sum",] -tmodeAS0Profile_vis$purpose[tmodeAS0Profile_vis$purpose=="Sum"] <- "Total" - -# (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) -tmode1_as1 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode2_as1 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode3_as1 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode4_as1 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode5_as1 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode6_as1 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode7_as1 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode8_as1 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tmodeAS1Profile <- data.frame(tmode1_as1$counts, tmode2_as1$counts, tmode3_as1$counts, tmode4_as1$counts, - tmode5_as1$counts, tmode6_as1$counts, tmode7_as1$counts, tmode8_as1$counts) -colnames(tmodeAS1Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(tmodeAS1Profile, "tmodeAS1Profile.csv") - -# Prepeare data for visualizer (changed from 9 to 12) -tmodeAS1Profile_vis <- tmodeAS1Profile[1:13,] -tmodeAS1Profile_vis$id <- row.names(tmodeAS1Profile_vis) -tmodeAS1Profile_vis <- melt(tmodeAS1Profile_vis, id = c("id")) -colnames(tmodeAS1Profile_vis) <- c("id", "purpose", "freq_as1") - -tmodeAS1Profile_vis <- xtabs(freq_as1~id+purpose, tmodeAS1Profile_vis) -tmodeAS1Profile_vis[is.na(tmodeAS1Profile_vis)] <- 0 -tmodeAS1Profile_vis <- addmargins(as.table(tmodeAS1Profile_vis)) -tmodeAS1Profile_vis <- as.data.frame.matrix(tmodeAS1Profile_vis) - -tmodeAS1Profile_vis$id <- row.names(tmodeAS1Profile_vis) -tmodeAS1Profile_vis <- melt(tmodeAS1Profile_vis, id = c("id")) -colnames(tmodeAS1Profile_vis) <- c("id", "purpose", "freq_as1") -tmodeAS1Profile_vis$id <- as.character(tmodeAS1Profile_vis$id) -tmodeAS1Profile_vis$purpose <- as.character(tmodeAS1Profile_vis$purpose) -tmodeAS1Profile_vis <- tmodeAS1Profile_vis[tmodeAS1Profile_vis$id!="Sum",] -tmodeAS1Profile_vis$purpose[tmodeAS1Profile_vis$purpose=="Sum"] <- "Total" - -# (seq changed from 10 to 13 due to increase in number of modes, changed by Khademul.haque@rsginc.com) -tmode1_as2 <- hist(tours$TOURMODE[tours$TOURPURP==1 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode2_as2 <- hist(tours$TOURMODE[tours$TOURPURP==2 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode3_as2 <- hist(tours$TOURMODE[tours$TOURPURP==3 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode4_as2 <- hist(tours$TOURMODE[tours$TOURPURP>=4 & tours$TOURPURP<=6 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode5_as2 <- hist(tours$TOURMODE[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode6_as2 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=4 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode7_as2 <- hist(unique_joint_tours$TOURMODE[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tmode8_as2 <- hist(tours$TOURMODE[tours$TOURPURP==10 & tours$AUTOSUFF==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tmodeAS2Profile <- data.frame(tmode1_as2$counts, tmode2_as2$counts, tmode3_as2$counts, tmode4_as2$counts, - tmode5_as2$counts, tmode6_as2$counts, tmode7_as2$counts, tmode8_as2$counts) -colnames(tmodeAS2Profile) <- c("work", "univ", "sch", "imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(tmodeAS2Profile, "tmodeAS2Profile.csv") - -# Prepeare data for visualizer (changed from 9 to 12) -tmodeAS2Profile_vis <- tmodeAS2Profile[1:13,] -tmodeAS2Profile_vis$id <- row.names(tmodeAS2Profile_vis) -tmodeAS2Profile_vis <- melt(tmodeAS2Profile_vis, id = c("id")) -colnames(tmodeAS2Profile_vis) <- c("id", "purpose", "freq_as2") - -tmodeAS2Profile_vis <- xtabs(freq_as2~id+purpose, tmodeAS2Profile_vis) -tmodeAS2Profile_vis[is.na(tmodeAS2Profile_vis)] <- 0 -tmodeAS2Profile_vis <- addmargins(as.table(tmodeAS2Profile_vis)) -tmodeAS2Profile_vis <- as.data.frame.matrix(tmodeAS2Profile_vis) - -tmodeAS2Profile_vis$id <- row.names(tmodeAS2Profile_vis) -tmodeAS2Profile_vis <- melt(tmodeAS2Profile_vis, id = c("id")) -colnames(tmodeAS2Profile_vis) <- c("id", "purpose", "freq_as2") -tmodeAS2Profile_vis$id <- as.character(tmodeAS2Profile_vis$id) -tmodeAS2Profile_vis$purpose <- as.character(tmodeAS2Profile_vis$purpose) -tmodeAS2Profile_vis <- tmodeAS2Profile_vis[tmodeAS2Profile_vis$id!="Sum",] -tmodeAS2Profile_vis$purpose[tmodeAS2Profile_vis$purpose=="Sum"] <- "Total" - - -# Combine three AS groups -tmodeProfile_vis <- data.frame(tmodeAS0Profile_vis, tmodeAS1Profile_vis$freq_as1, tmodeAS2Profile_vis$freq_as2) -colnames(tmodeProfile_vis) <- c("id", "purpose", "freq_as0", "freq_as1", "freq_as2") -tmodeProfile_vis$freq_all <- tmodeProfile_vis$freq_as0 + tmodeProfile_vis$freq_as1 + tmodeProfile_vis$freq_as2 -write.csv(tmodeProfile_vis, "tmodeProfile_vis.csv", row.names = F) - - -# Non Mand Tour lengths -tourdist4 <- hist(tours$tour_distance[tours$TOURPURP==4], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -tourdisti56 <- hist(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -tourdisti789 <- hist(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -tourdistj56 <- hist(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -tourdistj789 <- hist(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -tourdist10 <- hist(tours$tour_distance[tours$TOURPURP==10], breaks = c(seq(0,40, by=1), 9999), freq = NULL, right=FALSE) - -tourDistProfile <- data.frame(tourdist4$counts, tourdisti56$counts, tourdisti789$counts, tourdistj56$counts, tourdistj789$counts, tourdist10$counts) - -colnames(tourDistProfile) <- c("esco", "imain", "idisc", "jmain", "jdisc", "atwork") - -write.csv(tourDistProfile, "nonMandTourDistProfile.csv") - -#prepare input for visualizer -tourDistProfile_vis <- tourDistProfile -tourDistProfile_vis$id <- row.names(tourDistProfile_vis) -tourDistProfile_vis <- melt(tourDistProfile_vis, id = c("id")) -colnames(tourDistProfile_vis) <- c("id", "purpose", "freq") - -tourDistProfile_vis <- xtabs(freq~id+purpose, tourDistProfile_vis) -tourDistProfile_vis <- addmargins(as.table(tourDistProfile_vis)) -tourDistProfile_vis <- as.data.frame.matrix(tourDistProfile_vis) -tourDistProfile_vis$id <- row.names(tourDistProfile_vis) -tourDistProfile_vis <- melt(tourDistProfile_vis, id = c("id")) -colnames(tourDistProfile_vis) <- c("distbin", "PURPOSE", "freq") -tourDistProfile_vis$PURPOSE <- as.character(tourDistProfile_vis$PURPOSE) -tourDistProfile_vis$distbin <- as.character(tourDistProfile_vis$distbin) -tourDistProfile_vis <- tourDistProfile_vis[tourDistProfile_vis$distbin!="Sum",] -tourDistProfile_vis$PURPOSE[tourDistProfile_vis$PURPOSE=="Sum"] <- "Total" -tourDistProfile_vis$distbin <- as.numeric(tourDistProfile_vis$distbin) - -write.csv(tourDistProfile_vis, "tourDistProfile_vis.csv", row.names = F) - -cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4], na.rm = TRUE)) -cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], na.rm = TRUE)) -cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], na.rm = TRUE)) -cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], na.rm = TRUE)) -cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], na.rm = TRUE)) -cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10], na.rm = TRUE)) - -## Retirees -#cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4 & tours$PERTYPE==5], na.rm = TRUE)) -#cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6 & tours$PERTYPE==5], na.rm = TRUE)) -#cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$PERTYPE==5], na.rm = TRUE)) -#cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$PERTYPE==5], na.rm = TRUE)) -#cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$PERTYPE==5], na.rm = TRUE)) -#cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10 & tours$PERTYPE==5], na.rm = TRUE)) -# -## Non-reitrees -#cat("\n Average Tour Distance [esco]: ", mean(tours$tour_distance[tours$TOURPURP==4 & tours$PERTYPE!=5], na.rm = TRUE)) -#cat("\n Average Tour Distance [imain]: ", mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6 & tours$PERTYPE!=5], na.rm = TRUE)) -#cat("\n Average Tour Distance [idisc]: ", mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9 & tours$PERTYPE!=5], na.rm = TRUE)) -#cat("\n Average Tour Distance [jmain]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6 & unique_joint_tours$PERTYPE!=5], na.rm = TRUE)) -#cat("\n Average Tour Distance [jdisc]: ", mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9 & unique_joint_tours$PERTYPE!=5], na.rm = TRUE)) -#cat("\n Average Tour Distance [atwork]: ", mean(tours$tour_distance[tours$TOURPURP==10 & tours$PERTYPE!=5], na.rm = TRUE)) -# - -## Output average trips lengths for visualizer - -avgTripLengths <- c(mean(tours$tour_distance[tours$TOURPURP==4], na.rm = TRUE), - mean(tours$tour_distance[tours$TOURPURP>=5 & tours$TOURPURP<=6], na.rm = TRUE), - mean(tours$tour_distance[tours$TOURPURP>=7 & tours$TOURPURP<=9], na.rm = TRUE), - mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], na.rm = TRUE), - mean(unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], na.rm = TRUE), - mean(tours$tour_distance[tours$TOURPURP==10], na.rm = TRUE)) - -totAvgNonMand <- mean(c(tours$tour_distance[tours$TOURPURP %in% c(4,5,6,7,8,9,10)], - unique_joint_tours$tour_distance[unique_joint_tours$JOINT_PURP %in% c(5,6,7,8,9)]), - na.rm = T) -avgTripLengths <- c(avgTripLengths, totAvgNonMand) - -nonMandTourPurpose <- c("esco", "imain", "idisc", "jmain", "jdisc", "atwork", "Total") - -nonMandTripLengths <- data.frame(purpose = nonMandTourPurpose, avgTripLength = avgTripLengths) - -write.csv(nonMandTripLengths, "nonMandTripLengths.csv", row.names = F) - -# STop Frequency -#Outbound -stopfreq1 <- hist(tours$num_ob_stops[tours$TOURPURP==1], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(tours$num_ob_stops[tours$TOURPURP==2], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(tours$num_ob_stops[tours$TOURPURP==3], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(tours$num_ob_stops[tours$TOURPURP==4], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(tours$num_ob_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(tours$num_ob_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(unique_joint_tours$num_ob_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(unique_joint_tours$num_ob_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(tours$num_ob_stops[tours$TOURPURP==10], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopFreqOutProfile.csv") - -# prepare stop frequency input for visualizer -stopFreqout_vis <- stopFreq -stopFreqout_vis$id <- row.names(stopFreqout_vis) -stopFreqout_vis <- melt(stopFreqout_vis, id = c("id")) -colnames(stopFreqout_vis) <- c("id", "purpose", "freq") - -stopFreqout_vis <- xtabs(freq~purpose+id, stopFreqout_vis) -stopFreqout_vis <- addmargins(as.table(stopFreqout_vis)) -stopFreqout_vis <- as.data.frame.matrix(stopFreqout_vis) -stopFreqout_vis$id <- row.names(stopFreqout_vis) -stopFreqout_vis <- melt(stopFreqout_vis, id = c("id")) -colnames(stopFreqout_vis) <- c("purpose", "nstops", "freq") -stopFreqout_vis$purpose <- as.character(stopFreqout_vis$purpose) -stopFreqout_vis$nstops <- as.character(stopFreqout_vis$nstops) -stopFreqout_vis <- stopFreqout_vis[stopFreqout_vis$nstops!="Sum",] -stopFreqout_vis$purpose[stopFreqout_vis$purpose=="Sum"] <- "Total" - -#Inbound -stopfreq1 <- hist(tours$num_ib_stops[tours$TOURPURP==1], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(tours$num_ib_stops[tours$TOURPURP==2], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(tours$num_ib_stops[tours$TOURPURP==3], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(tours$num_ib_stops[tours$TOURPURP==4], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(tours$num_ib_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(tours$num_ib_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(unique_joint_tours$num_ib_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(unique_joint_tours$num_ib_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(tours$num_ib_stops[tours$TOURPURP==10], breaks = c(seq(0,3, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopFreqInbProfile.csv") - -# prepare stop frequency input for visualizer -stopFreqinb_vis <- stopFreq -stopFreqinb_vis$id <- row.names(stopFreqinb_vis) -stopFreqinb_vis <- melt(stopFreqinb_vis, id = c("id")) -colnames(stopFreqinb_vis) <- c("id", "purpose", "freq") - -stopFreqinb_vis <- xtabs(freq~purpose+id, stopFreqinb_vis) -stopFreqinb_vis <- addmargins(as.table(stopFreqinb_vis)) -stopFreqinb_vis <- as.data.frame.matrix(stopFreqinb_vis) -stopFreqinb_vis$id <- row.names(stopFreqinb_vis) -stopFreqinb_vis <- melt(stopFreqinb_vis, id = c("id")) -colnames(stopFreqinb_vis) <- c("purpose", "nstops", "freq") -stopFreqinb_vis$purpose <- as.character(stopFreqinb_vis$purpose) -stopFreqinb_vis$nstops <- as.character(stopFreqinb_vis$nstops) -stopFreqinb_vis <- stopFreqinb_vis[stopFreqinb_vis$nstops!="Sum",] -stopFreqinb_vis$purpose[stopFreqinb_vis$purpose=="Sum"] <- "Total" - - -stopfreqDir_vis <- data.frame(stopFreqout_vis, stopFreqinb_vis$freq) -colnames(stopfreqDir_vis) <- c("purpose", "nstops", "freq_out", "freq_inb") -write.csv(stopfreqDir_vis, "stopfreqDir_vis.csv", row.names = F) - - -#Total -stopfreq1 <- hist(tours$num_tot_stops[tours$TOURPURP==1], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(tours$num_tot_stops[tours$TOURPURP==2], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(tours$num_tot_stops[tours$TOURPURP==3], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(tours$num_tot_stops[tours$TOURPURP==4], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(tours$num_tot_stops[tours$TOURPURP>=5 & tours$TOURPURP<=6], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(tours$num_tot_stops[tours$TOURPURP>=7 & tours$TOURPURP<=9], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(unique_joint_tours$num_tot_stops[unique_joint_tours$JOINT_PURP>=5 & unique_joint_tours$JOINT_PURP<=6], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(unique_joint_tours$num_tot_stops[unique_joint_tours$JOINT_PURP>=7 & unique_joint_tours$JOINT_PURP<=9], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(tours$num_tot_stops[tours$TOURPURP==10], breaks = c(seq(0,6, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopFreqTotProfile.csv") - -# prepare stop frequency input for visualizer -stopFreq_vis <- stopFreq -stopFreq_vis$id <- row.names(stopFreq_vis) -stopFreq_vis <- melt(stopFreq_vis, id = c("id")) -colnames(stopFreq_vis) <- c("id", "purpose", "freq") - -stopFreq_vis <- xtabs(freq~purpose+id, stopFreq_vis) -stopFreq_vis <- addmargins(as.table(stopFreq_vis)) -stopFreq_vis <- as.data.frame.matrix(stopFreq_vis) -stopFreq_vis$id <- row.names(stopFreq_vis) -stopFreq_vis <- melt(stopFreq_vis, id = c("id")) -colnames(stopFreq_vis) <- c("purpose", "nstops", "freq") -stopFreq_vis$purpose <- as.character(stopFreq_vis$purpose) -stopFreq_vis$nstops <- as.character(stopFreq_vis$nstops) -stopFreq_vis <- stopFreq_vis[stopFreq_vis$nstops!="Sum",] -stopFreq_vis$purpose[stopFreq_vis$purpose=="Sum"] <- "Total" - -write.csv(stopFreq_vis, "stopfreq_total_vis.csv", row.names = F) - -#STop purpose X TourPurpose -stopfreq1 <- hist(stops$DPURP[stops$TOURPURP==1], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(stops$DPURP[stops$TOURPURP==2], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(stops$DPURP[stops$TOURPURP==3], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(stops$DPURP[stops$TOURPURP==4], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(stops$DPURP[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(stops$DPURP[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(jstops$DPURP[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(jstops$DPURP[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(stops$DPURP[stops$TOURPURP==10], breaks = c(seq(1,10, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopPurposeByTourPurpose.csv") - -# prepare stop frequency input for visualizer -stopFreq_vis <- stopFreq -stopFreq_vis$id <- row.names(stopFreq_vis) -stopFreq_vis <- melt(stopFreq_vis, id = c("id")) -colnames(stopFreq_vis) <- c("stop_purp", "purpose", "freq") - -stopFreq_vis <- xtabs(freq~purpose+stop_purp, stopFreq_vis) -stopFreq_vis <- addmargins(as.table(stopFreq_vis)) -stopFreq_vis <- as.data.frame.matrix(stopFreq_vis) -stopFreq_vis$purpose <- row.names(stopFreq_vis) -stopFreq_vis <- melt(stopFreq_vis, id = c("purpose")) -colnames(stopFreq_vis) <- c("purpose", "stop_purp", "freq") -stopFreq_vis$purpose <- as.character(stopFreq_vis$purpose) -stopFreq_vis$stop_purp <- as.character(stopFreq_vis$stop_purp) -stopFreq_vis <- stopFreq_vis[stopFreq_vis$stop_purp!="Sum",] -stopFreq_vis$purpose[stopFreq_vis$purpose=="Sum"] <- "Total" - -write.csv(stopFreq_vis, "stoppurpose_tourpurpose_vis.csv", row.names = F) - -#Out of direction - Stop Location -stopfreq1 <- hist(stops$out_dir_dist[stops$TOURPURP==1], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(stops$out_dir_dist[stops$TOURPURP==2], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(stops$out_dir_dist[stops$TOURPURP==3], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(stops$out_dir_dist[stops$TOURPURP==4], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(stops$out_dir_dist[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(stops$out_dir_dist[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(jstops$out_dir_dist[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(jstops$out_dir_dist[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(stops$out_dir_dist[stops$TOURPURP==10], breaks = c(-9999,seq(0,40, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopOutOfDirectionDC.csv") - -# prepare stop location input for visualizer -stopDC_vis <- stopFreq -stopDC_vis$id <- row.names(stopDC_vis) -stopDC_vis <- melt(stopDC_vis, id = c("id")) -colnames(stopDC_vis) <- c("id", "purpose", "freq") - -stopDC_vis <- xtabs(freq~id+purpose, stopDC_vis) -stopDC_vis <- addmargins(as.table(stopDC_vis)) -stopDC_vis <- as.data.frame.matrix(stopDC_vis) -stopDC_vis$id <- row.names(stopDC_vis) -stopDC_vis <- melt(stopDC_vis, id = c("id")) -colnames(stopDC_vis) <- c("distbin", "PURPOSE", "freq") -stopDC_vis$PURPOSE <- as.character(stopDC_vis$PURPOSE) -stopDC_vis$distbin <- as.character(stopDC_vis$distbin) -stopDC_vis <- stopDC_vis[stopDC_vis$distbin!="Sum",] -stopDC_vis$PURPOSE[stopDC_vis$PURPOSE=="Sum"] <- "Total" -stopDC_vis$distbin <- as.numeric(stopDC_vis$distbin) - -write.csv(stopDC_vis, "stopDC_vis.csv", row.names = F) - -# compute average out of dir distance for visualizer -avgDistances <- c(mean(stops$out_dir_dist[stops$TOURPURP==1], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP==2], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP==3], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP==4], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP>=5 & stops$TOURPURP<=6], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP>=7 & stops$TOURPURP<=9], na.rm = TRUE), - mean(jstops$out_dir_dist[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], na.rm = TRUE), - mean(jstops$out_dir_dist[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], na.rm = TRUE), - mean(stops$out_dir_dist[stops$TOURPURP==10], na.rm = TRUE), - mean(stops$out_dir_dist, na.rm = TRUE)) - -purp <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork", "total") - -avgStopOutofDirectionDist <- data.frame(purpose = purp, avgDist = avgDistances) - -write.csv(avgStopOutofDirectionDist, "avgStopOutofDirectionDist_vis.csv", row.names = F) - -#Stop Departure Time -stopfreq1 <- hist(stops$stop_period[stops$TOURPURP==1], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(stops$stop_period[stops$TOURPURP==2], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(stops$stop_period[stops$TOURPURP==3], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(stops$stop_period[stops$TOURPURP==4], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(stops$stop_period[stops$TOURPURP>=5 & stops$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(stops$stop_period[stops$TOURPURP>=7 & stops$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(jstops$stop_period[jstops$TOURPURP>=5 & jstops$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(jstops$stop_period[jstops$TOURPURP>=7 & jstops$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(stops$stop_period[stops$TOURPURP==10], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "stopDeparture.csv") - -# prepare stop departure input for visualizer -stopDep_vis <- stopFreq -stopDep_vis$id <- row.names(stopDep_vis) -stopDep_vis <- melt(stopDep_vis, id = c("id")) -colnames(stopDep_vis) <- c("id", "purpose", "freq_stop") - -stopDep_vis$purpose <- as.character(stopDep_vis$purpose) -stopDep_vis <- xtabs(freq_stop~id+purpose, stopDep_vis) -stopDep_vis <- addmargins(as.table(stopDep_vis)) -stopDep_vis <- as.data.frame.matrix(stopDep_vis) -stopDep_vis$id <- row.names(stopDep_vis) -stopDep_vis <- melt(stopDep_vis, id = c("id")) -colnames(stopDep_vis) <- c("timebin", "PURPOSE", "freq") -stopDep_vis$PURPOSE <- as.character(stopDep_vis$PURPOSE) -stopDep_vis$timebin <- as.character(stopDep_vis$timebin) -stopDep_vis <- stopDep_vis[stopDep_vis$timebin!="Sum",] -stopDep_vis$PURPOSE[stopDep_vis$PURPOSE=="Sum"] <- "Total" -stopDep_vis$timebin <- as.numeric(stopDep_vis$timebin) - -#Trip Departure Time -stopfreq1 <- hist(trips$stop_period[trips$TOURPURP==1], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq2 <- hist(trips$stop_period[trips$TOURPURP==2], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq3 <- hist(trips$stop_period[trips$TOURPURP==3], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq4 <- hist(trips$stop_period[trips$TOURPURP==4], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi56 <- hist(trips$stop_period[trips$TOURPURP>=5 & trips$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqi789 <- hist(trips$stop_period[trips$TOURPURP>=7 & trips$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj56 <- hist(jtrips$stop_period[jtrips$TOURPURP>=5 & jtrips$TOURPURP<=6], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreqj789 <- hist(jtrips$stop_period[jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) -stopfreq10 <- hist(trips$stop_period[trips$TOURPURP==10], breaks = c(seq(1,40, by=1), 9999), freq = NULL, right=FALSE) - -stopFreq <- data.frame(stopfreq1$counts, stopfreq2$counts, stopfreq3$counts, stopfreq4$counts, stopfreqi56$counts - , stopfreqi789$counts, stopfreqj56$counts, stopfreqj789$counts, stopfreq10$counts) -colnames(stopFreq) <- c("work", "univ", "sch", "esco","imain", "idisc", "jmain", "jdisc", "atwork") -write.csv(stopFreq, "tripDeparture.csv") - -# prepare stop departure input for visualizer -tripDep_vis <- stopFreq -tripDep_vis$id <- row.names(tripDep_vis) -tripDep_vis <- melt(tripDep_vis, id = c("id")) -colnames(tripDep_vis) <- c("id", "purpose", "freq_trip") - -tripDep_vis$purpose <- as.character(tripDep_vis$purpose) -tripDep_vis <- xtabs(freq_trip~id+purpose, tripDep_vis) -tripDep_vis <- addmargins(as.table(tripDep_vis)) -tripDep_vis <- as.data.frame.matrix(tripDep_vis) -tripDep_vis$id <- row.names(tripDep_vis) -tripDep_vis <- melt(tripDep_vis, id = c("id")) -colnames(tripDep_vis) <- c("timebin", "PURPOSE", "freq") -tripDep_vis$PURPOSE <- as.character(tripDep_vis$PURPOSE) -tripDep_vis$timebin <- as.character(tripDep_vis$timebin) -tripDep_vis <- tripDep_vis[tripDep_vis$timebin!="Sum",] -tripDep_vis$PURPOSE[tripDep_vis$PURPOSE=="Sum"] <- "Total" -tripDep_vis$timebin <- as.numeric(tripDep_vis$timebin) - -stopTripDep_vis <- data.frame(stopDep_vis, tripDep_vis$freq) -colnames(stopTripDep_vis) <- c("id", "purpose", "freq_stop", "freq_trip") -write.csv(stopTripDep_vis, "stopTripDep_vis.csv", row.names = F) - -#Trip Mode Summary (added 3 lines due to change in mode codes, changed seq 9 to 13, Khademul Haque) -#Work -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==1 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_Work.csv") - -# Prepare data for visualizer (changed from 9 to 12) -tripModeProfile1_vis <- tripModeProfile[1:13,] -tripModeProfile1_vis$id <- row.names(tripModeProfile1_vis) -tripModeProfile1_vis <- melt(tripModeProfile1_vis, id = c("id")) -colnames(tripModeProfile1_vis) <- c("id", "purpose", "freq1") - -tripModeProfile1_vis <- xtabs(freq1~id+purpose, tripModeProfile1_vis) -tripModeProfile1_vis[is.na(tripModeProfile1_vis)] <- 0 -tripModeProfile1_vis <- addmargins(as.table(tripModeProfile1_vis)) -tripModeProfile1_vis <- as.data.frame.matrix(tripModeProfile1_vis) - -tripModeProfile1_vis$id <- row.names(tripModeProfile1_vis) -tripModeProfile1_vis <- melt(tripModeProfile1_vis, id = c("id")) -colnames(tripModeProfile1_vis) <- c("id", "purpose", "freq1") -tripModeProfile1_vis$id <- as.character(tripModeProfile1_vis$id) -tripModeProfile1_vis$purpose <- as.character(tripModeProfile1_vis$purpose) -tripModeProfile1_vis <- tripModeProfile1_vis[tripModeProfile1_vis$id!="Sum",] -tripModeProfile1_vis$purpose[tripModeProfile1_vis$purpose=="Sum"] <- "Total" - - -#University (added 3 lines due to change in mode codes, changed seq 9 to 13, Khademul Haque) -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==2 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_Univ.csv") - -tripModeProfile2_vis <- tripModeProfile[1:13,] -tripModeProfile2_vis$id <- row.names(tripModeProfile2_vis) -tripModeProfile2_vis <- melt(tripModeProfile2_vis, id = c("id")) -colnames(tripModeProfile2_vis) <- c("id", "purpose", "freq2") - -tripModeProfile2_vis <- xtabs(freq2~id+purpose, tripModeProfile2_vis) -tripModeProfile2_vis[is.na(tripModeProfile2_vis)] <- 0 -tripModeProfile2_vis <- addmargins(as.table(tripModeProfile2_vis)) -tripModeProfile2_vis <- as.data.frame.matrix(tripModeProfile2_vis) - -tripModeProfile2_vis$id <- row.names(tripModeProfile2_vis) -tripModeProfile2_vis <- melt(tripModeProfile2_vis, id = c("id")) -colnames(tripModeProfile2_vis) <- c("id", "purpose", "freq2") -tripModeProfile2_vis$id <- as.character(tripModeProfile2_vis$id) -tripModeProfile2_vis$purpose <- as.character(tripModeProfile2_vis$purpose) -tripModeProfile2_vis <- tripModeProfile2_vis[tripModeProfile2_vis$id!="Sum",] -tripModeProfile2_vis$purpose[tripModeProfile2_vis$purpose=="Sum"] <- "Total" - -#School -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==3 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_Schl.csv") - -tripModeProfile3_vis <- tripModeProfile[1:13,] -tripModeProfile3_vis$id <- row.names(tripModeProfile3_vis) -tripModeProfile3_vis <- melt(tripModeProfile3_vis, id = c("id")) -colnames(tripModeProfile3_vis) <- c("id", "purpose", "freq3") - -tripModeProfile3_vis <- xtabs(freq3~id+purpose, tripModeProfile3_vis) -tripModeProfile3_vis[is.na(tripModeProfile3_vis)] <- 0 -tripModeProfile3_vis <- addmargins(as.table(tripModeProfile3_vis)) -tripModeProfile3_vis <- as.data.frame.matrix(tripModeProfile3_vis) - -tripModeProfile3_vis$id <- row.names(tripModeProfile3_vis) -tripModeProfile3_vis <- melt(tripModeProfile3_vis, id = c("id")) -colnames(tripModeProfile3_vis) <- c("id", "purpose", "freq3") -tripModeProfile3_vis$id <- as.character(tripModeProfile3_vis$id) -tripModeProfile3_vis$purpose <- as.character(tripModeProfile3_vis$purpose) -tripModeProfile3_vis <- tripModeProfile3_vis[tripModeProfile3_vis$id!="Sum",] -tripModeProfile3_vis$purpose[tripModeProfile3_vis$purpose=="Sum"] <- "Total" - -#iMain -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=4 & trips$TOURPURP<=6 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_iMain.csv") - -tripModeProfile4_vis <- tripModeProfile[1:13,] -tripModeProfile4_vis$id <- row.names(tripModeProfile4_vis) -tripModeProfile4_vis <- melt(tripModeProfile4_vis, id = c("id")) -colnames(tripModeProfile4_vis) <- c("id", "purpose", "freq4") - -tripModeProfile4_vis <- xtabs(freq4~id+purpose, tripModeProfile4_vis) -tripModeProfile4_vis[is.na(tripModeProfile4_vis)] <- 0 -tripModeProfile4_vis <- addmargins(as.table(tripModeProfile4_vis)) -tripModeProfile4_vis <- as.data.frame.matrix(tripModeProfile4_vis) - -tripModeProfile4_vis$id <- row.names(tripModeProfile4_vis) -tripModeProfile4_vis <- melt(tripModeProfile4_vis, id = c("id")) -colnames(tripModeProfile4_vis) <- c("id", "purpose", "freq4") -tripModeProfile4_vis$id <- as.character(tripModeProfile4_vis$id) -tripModeProfile4_vis$purpose <- as.character(tripModeProfile4_vis$purpose) -tripModeProfile4_vis <- tripModeProfile4_vis[tripModeProfile4_vis$id!="Sum",] -tripModeProfile4_vis$purpose[tripModeProfile4_vis$purpose=="Sum"] <- "Total" - -#iDisc -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP>=7 & trips$TOURPURP<=9 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_iDisc.csv") - -tripModeProfile5_vis <- tripModeProfile[1:13,] -tripModeProfile5_vis$id <- row.names(tripModeProfile5_vis) -tripModeProfile5_vis <- melt(tripModeProfile5_vis, id = c("id")) -colnames(tripModeProfile5_vis) <- c("id", "purpose", "freq5") - -tripModeProfile5_vis <- xtabs(freq5~id+purpose, tripModeProfile5_vis) -tripModeProfile5_vis[is.na(tripModeProfile5_vis)] <- 0 -tripModeProfile5_vis <- addmargins(as.table(tripModeProfile5_vis)) -tripModeProfile5_vis <- as.data.frame.matrix(tripModeProfile5_vis) - -tripModeProfile5_vis$id <- row.names(tripModeProfile5_vis) -tripModeProfile5_vis <- melt(tripModeProfile5_vis, id = c("id")) -colnames(tripModeProfile5_vis) <- c("id", "purpose", "freq5") -tripModeProfile5_vis$id <- as.character(tripModeProfile5_vis$id) -tripModeProfile5_vis$purpose <- as.character(tripModeProfile5_vis$purpose) -tripModeProfile5_vis <- tripModeProfile5_vis[tripModeProfile5_vis$id!="Sum",] -tripModeProfile5_vis$purpose[tripModeProfile5_vis$purpose=="Sum"] <- "Total" - -#jMain -tripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=4 & jtrips$TOURPURP<=6 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_jMain.csv") - -tripModeProfile6_vis <- tripModeProfile[1:13,] -tripModeProfile6_vis$id <- row.names(tripModeProfile6_vis) -tripModeProfile6_vis <- melt(tripModeProfile6_vis, id = c("id")) -colnames(tripModeProfile6_vis) <- c("id", "purpose", "freq6") - -tripModeProfile6_vis <- xtabs(freq6~id+purpose, tripModeProfile6_vis) -tripModeProfile6_vis[is.na(tripModeProfile6_vis)] <- 0 -tripModeProfile6_vis <- addmargins(as.table(tripModeProfile6_vis)) -tripModeProfile6_vis <- as.data.frame.matrix(tripModeProfile6_vis) - -tripModeProfile6_vis$id <- row.names(tripModeProfile6_vis) -tripModeProfile6_vis <- melt(tripModeProfile6_vis, id = c("id")) -colnames(tripModeProfile6_vis) <- c("id", "purpose", "freq6") -tripModeProfile6_vis$id <- as.character(tripModeProfile6_vis$id) -tripModeProfile6_vis$purpose <- as.character(tripModeProfile6_vis$purpose) -tripModeProfile6_vis <- tripModeProfile6_vis[tripModeProfile6_vis$id!="Sum",] -tripModeProfile6_vis$purpose[tripModeProfile6_vis$purpose=="Sum"] <- "Total" - -#jDisc -tripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURPURP>=7 & jtrips$TOURPURP<=9 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_jDisc.csv") - -tripModeProfile7_vis <- tripModeProfile[1:13,] -tripModeProfile7_vis$id <- row.names(tripModeProfile7_vis) -tripModeProfile7_vis <- melt(tripModeProfile7_vis, id = c("id")) -colnames(tripModeProfile7_vis) <- c("id", "purpose", "freq7") - -tripModeProfile7_vis <- xtabs(freq7~id+purpose, tripModeProfile7_vis) -tripModeProfile7_vis[is.na(tripModeProfile7_vis)] <- 0 -tripModeProfile7_vis <- addmargins(as.table(tripModeProfile7_vis)) -tripModeProfile7_vis <- as.data.frame.matrix(tripModeProfile7_vis) - -tripModeProfile7_vis$id <- row.names(tripModeProfile7_vis) -tripModeProfile7_vis <- melt(tripModeProfile7_vis, id = c("id")) -colnames(tripModeProfile7_vis) <- c("id", "purpose", "freq7") -tripModeProfile7_vis$id <- as.character(tripModeProfile7_vis$id) -tripModeProfile7_vis$purpose <- as.character(tripModeProfile7_vis$purpose) -tripModeProfile7_vis <- tripModeProfile7_vis[tripModeProfile7_vis$id!="Sum",] -tripModeProfile7_vis$purpose[tripModeProfile7_vis$purpose=="Sum"] <- "Total" - -#At work -tripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -tripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURPURP==10 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(tripmode1$counts, tripmode2$counts, tripmode3$counts, tripmode4$counts, - tripmode5$counts, tripmode6$counts, tripmode7$counts, tripmode8$counts, tripmode9$counts, - tripmode10$counts, tripmode11$counts, tripmode12$counts, tripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_AtWork.csv") - -tripModeProfile8_vis <- tripModeProfile[1:13,] -tripModeProfile8_vis$id <- row.names(tripModeProfile8_vis) -tripModeProfile8_vis <- melt(tripModeProfile8_vis, id = c("id")) -colnames(tripModeProfile8_vis) <- c("id", "purpose", "freq8") - -tripModeProfile8_vis <- xtabs(freq8~id+purpose, tripModeProfile8_vis) -tripModeProfile8_vis[is.na(tripModeProfile8_vis)] <- 0 -tripModeProfile8_vis <- addmargins(as.table(tripModeProfile8_vis)) -tripModeProfile8_vis <- as.data.frame.matrix(tripModeProfile8_vis) - -tripModeProfile8_vis$id <- row.names(tripModeProfile8_vis) -tripModeProfile8_vis <- melt(tripModeProfile8_vis, id = c("id")) -colnames(tripModeProfile8_vis) <- c("id", "purpose", "freq8") -tripModeProfile8_vis$id <- as.character(tripModeProfile8_vis$id) -tripModeProfile8_vis$purpose <- as.character(tripModeProfile8_vis$purpose) -tripModeProfile8_vis <- tripModeProfile8_vis[tripModeProfile8_vis$id!="Sum",] -tripModeProfile8_vis$purpose[tripModeProfile8_vis$purpose=="Sum"] <- "Total" - -#iTotal -itripmode1 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode2 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode3 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode4 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode5 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode6 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode7 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode8 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode9 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode10 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode11 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode12 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -itripmode13 <- hist(trips$TRIPMODE[trips$TRIPMODE>0 & trips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -#jTotal -jtripmode1 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==1], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode2 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==2], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode3 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==3], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode4 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==4], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode5 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==5], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode6 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==6], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode7 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==7], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode8 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==8], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode9 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==9], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode10 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==10], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode11 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==11], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode12 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==12], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) -jtripmode13 <- hist(jtrips$TRIPMODE[jtrips$TRIPMODE>0 & jtrips$TOURMODE==13], breaks = seq(1,14, by=1), freq = NULL, right=FALSE) - -tripModeProfile <- data.frame(itripmode1$counts+jtripmode1$counts, itripmode2$counts+jtripmode2$counts, itripmode3$counts+jtripmode3$counts, itripmode4$counts+jtripmode4$counts, - itripmode5$counts+jtripmode5$counts, itripmode6$counts+jtripmode6$counts, itripmode7$counts+jtripmode7$counts, itripmode8$counts+jtripmode8$counts, - itripmode9$counts+jtripmode9$counts, itripmode10$counts+jtripmode10$counts, itripmode11$counts+jtripmode11$counts, itripmode12$counts+jtripmode12$counts, itripmode13$counts+jtripmode13$counts) -colnames(tripModeProfile) <- c("tourmode1", "tourmode2", "tourmode3", "tourmode4", "tourmode5", "tourmode6", "tourmode7", "tourmode8", "tourmode9", "tourmode10", "tourmode11", "tourmode12", "tourmode13") -write.csv(tripModeProfile, "tripModeProfile_Total.csv") - -tripModeProfile9_vis <- tripModeProfile[1:13,] -tripModeProfile9_vis$id <- row.names(tripModeProfile9_vis) -tripModeProfile9_vis <- melt(tripModeProfile9_vis, id = c("id")) -colnames(tripModeProfile9_vis) <- c("id", "purpose", "freq9") - -tripModeProfile9_vis <- xtabs(freq9~id+purpose, tripModeProfile9_vis) -tripModeProfile9_vis[is.na(tripModeProfile9_vis)] <- 0 -tripModeProfile9_vis <- addmargins(as.table(tripModeProfile9_vis)) -tripModeProfile9_vis <- as.data.frame.matrix(tripModeProfile9_vis) - -tripModeProfile9_vis$id <- row.names(tripModeProfile9_vis) -tripModeProfile9_vis <- melt(tripModeProfile9_vis, id = c("id")) -colnames(tripModeProfile9_vis) <- c("id", "purpose", "freq9") -tripModeProfile9_vis$id <- as.character(tripModeProfile9_vis$id) -tripModeProfile9_vis$purpose <- as.character(tripModeProfile9_vis$purpose) -tripModeProfile9_vis <- tripModeProfile9_vis[tripModeProfile9_vis$id!="Sum",] -tripModeProfile9_vis$purpose[tripModeProfile9_vis$purpose=="Sum"] <- "Total" - - -# combine all tripmode profile for visualizer -tripModeProfile_vis <- data.frame(tripModeProfile1_vis, tripModeProfile2_vis$freq2, tripModeProfile3_vis$freq3 - , tripModeProfile4_vis$freq4, tripModeProfile5_vis$freq5, tripModeProfile6_vis$freq6 - , tripModeProfile7_vis$freq7, tripModeProfile8_vis$freq8, tripModeProfile9_vis$freq9) -colnames(tripModeProfile_vis) <- c("tripmode", "tourmode", "work", "univ", "schl", "imain", "idisc", "jmain", "jdisc", "atwork", "total") - -temp <- melt(tripModeProfile_vis, id = c("tripmode", "tourmode")) -#tripModeProfile_vis <- cast(temp, tripmode+variable~tourmode) -#write.csv(tripModeProfile_vis, "tripModeProfile_vis.csv", row.names = F) -temp$grp_var <- paste(temp$variable, temp$tourmode, sep = "") - -# rename tour mode to standard names -temp$tourmode[temp$tourmode=="tourmode1"] <- 'Auto SOV' -temp$tourmode[temp$tourmode=="tourmode2"] <- 'Auto 2 Person' -temp$tourmode[temp$tourmode=="tourmode3"] <- 'Auto 3+ Person' -temp$tourmode[temp$tourmode=="tourmode4"] <- 'Walk' -temp$tourmode[temp$tourmode=="tourmode5"] <- 'Bike/Moped' -temp$tourmode[temp$tourmode=="tourmode6"] <- 'Walk-Transit' -temp$tourmode[temp$tourmode=="tourmode7"] <- 'PNR-Transit' -temp$tourmode[temp$tourmode=="tourmode8"] <- 'KNR-Transit' -temp$tourmode[temp$tourmode=="tourmode9"] <- 'TNC-Transit' -temp$tourmode[temp$tourmode=="tourmode10"] <- 'Taxi' -temp$tourmode[temp$tourmode=="tourmode11"] <- 'TNC-Single' -temp$tourmode[temp$tourmode=="tourmode12"] <- 'TNC-Shared' -temp$tourmode[temp$tourmode=="tourmode13"] <- 'School Bus' - -colnames(temp) <- c("tripmode","tourmode","purpose","value","grp_var") - -write.csv(temp, "tripModeProfile_vis.csv", row.names = F) - - -### -#trip mode by time period -#calculate time of day -trips$tod <- 5 # EA: 3 am - 6 am -trips$tod <- ifelse(trips$stop_period>=4 & trips$stop_period<=9, 1, trips$tod) # AM: 6 am - 9 am -trips$tod <- ifelse(trips$stop_period>=10 & trips$stop_period<=22, 2, trips$tod) # MD: 9 am - 3:30 pm -trips$tod <- ifelse(trips$stop_period>=23 & trips$stop_period<=29, 3, trips$tod) # PM: 3:30 pm - 7 pm -trips$tod <- ifelse(trips$stop_period>=30 & trips$stop_period<=40, 4, trips$tod) # EV: 7 pm - 3 am -trips$num_trips <- 1 - -jtrips$tod <- 5 # EA: 3 am - 6 am -jtrips$tod <- ifelse(jtrips$stop_period>=4 & jtrips$stop_period<=9, 1, jtrips$tod) # AM: 6 am - 9 am -jtrips$tod <- ifelse(jtrips$stop_period>=10 & jtrips$stop_period<=22, 2, jtrips$tod) # MD: 9 am - 3:30 pm -jtrips$tod <- ifelse(jtrips$stop_period>=23 & jtrips$stop_period<=29, 3, jtrips$tod) # PM: 3:30 pm - 7 pm -jtrips$tod <- ifelse(jtrips$stop_period>=30 & jtrips$stop_period<=40, 4, jtrips$tod) # EV: 7 pm - 3 am -jtrips$num_trips <- 1 -#jtrips$num_trips <- jtrips$num_participants - -itrips_summary <- aggregate(num_trips~tod+TOURPURP+TOURMODE+TRIPMODE, data=trips, FUN=sum) -jtrips_summary <- aggregate(num_trips~tod+TOURPURP+TOURMODE+TRIPMODE, data=jtrips, FUN=sum) - -write.csv(itrips_summary, "itrips_tripmode_summary.csv", row.names = F) -write.csv(jtrips_summary, "jtrips_tripmode_summary.csv", row.names = F) - -### - - - - -# Total number of stops, trips & tours -cat("Total number of stops : ", nrow(stops) + nrow(jstops)) -cat("Total number of trips : ", nrow(trips) + nrow(jtrips)) -cat("Total number of tours : ", nrow(tours) + sum(unique_joint_tours$NUMBER_HH)) - - -# output total numbers in a file -total_population <- sum(pertypeDistbn$freq) -total_households <- nrow(hh) -total_tours <- nrow(tours) + sum(unique_joint_tours$NUMBER_HH) -total_trips <- nrow(trips) + nrow(jtrips) -total_stops <- nrow(stops) + nrow(jstops) - -trips$num_travel[trips$TRIPMODE==1] <- 1 #sov -trips$num_travel[trips$TRIPMODE==2] <- 2 #hov2 -trips$num_travel[trips$TRIPMODE==3] <- 3.5 #hov3 -trips$num_travel[trips$TRIPMODE==10] <- 1.1 #taxi -trips$num_travel[trips$TRIPMODE==11] <- 1.2 #tnc single -trips$num_travel[trips$TRIPMODE==12] <- 2.0 #tnc shared -trips$num_travel[is.na(trips$num_travel)] <- 0 - -total_vmt <- sum((trips$od_dist[trips$TRIPMODE<=3])/trips$num_travel[trips$TRIPMODE<=3]) + sum((trips$od_dist[trips$TRIPMODE>=10 & trips$TRIPMODE<=12])/trips$num_travel[trips$TRIPMODE>=10 & trips$TRIPMODE<=12]) - -totals_var <- c("total_population", "total_households", "total_tours", "total_trips", "total_stops", "total_vmt") -totals_val <- c(total_population,total_households, total_tours, total_trips, total_stops, total_vmt) - -totals_df <- data.frame(name = totals_var, value = totals_val) - -write.csv(totals_df, "totals.csv", row.names = F) - -# HH Size distribution -hhSizeDist <- count(hh, c("HHSIZE")) -write.csv(hhSizeDist, "hhSizeDist.csv", row.names = F) - -# Persons by person type -actpertypeDistbn <- count(per[per$activity_pattern!="H"], c("PERTYPE")) -write.csv(actpertypeDistbn, "activePertypeDistbn.csv", row.names = TRUE) - - -### Generate school escorting summaries - -# detach plyr and load dplyr -#detach("package:plyr", unload=TRUE) -if (!"dplyr" %in% installed.packages()) install.packages("dplyr", repos='http://cran.us.r-project.org') -library(dplyr) - -# get driver person type -tours$out_chauffuer_ptype <- per$PERTYPE[match(tours$hh_id*100+tours$driver_num_out, - per$hh_id*100+per$person_num)] -tours$inb_chauffuer_ptype <- per$PERTYPE[match(tours$hh_id*100+tours$driver_num_in, - per$hh_id*100+per$person_num)] - -#tours$out_chauffuer_dap <- per$activity_pattern[match(tours$hh_id*100+tours$driver_num_out, per$hh_id*100+per$person_num)] -#tours$inb_chauffuer_dap <- per$activity_pattern[match(tours$hh_id*100+tours$driver_num_in, per$hh_id*100+per$person_num)] - - -tours[is.na(tours)] <- 0 - -tours_sample <- select(tours, hh_id, person_id, person_num, tour_id, tour_purpose, escort_type_out, escort_type_in, - driver_num_out, driver_num_in, person_type) - -tours_sample <- tours[tours$tour_purpose=="School" & tours$person_type>=6, ] - -# Code no escort as "3" to be same as OHAS data -tours_sample$escort_type_out[tours_sample$escort_type_out==0] <- 3 -tours_sample$escort_type_in[tours_sample$escort_type_in==0] <- 3 - - -# School tours by Escort Type X Child Type -out_table1 <- table(tours_sample$escort_type_out, tours_sample$person_type) -inb_table1 <- table(tours_sample$escort_type_in, tours_sample$person_type) - -# School tours by Escort Type X Chauffuer Type -out_sample2 <- filter(tours_sample, out_chauffuer_ptype>0) -inb_sample2 <- filter(tours_sample, inb_chauffuer_ptype>0) -out_table2 <- table(out_sample2$escort_type_out, out_sample2$out_chauffuer_ptype) -inb_table2 <- table(inb_sample2$escort_type_in, inb_sample2$inb_chauffuer_ptype) - -## Workers summary -# summary of worker with a child which went to school -# by escort type, can be separated by outbound and inbound direction - -#get list of active workers with at least one work tour -active_workers <- tours %>% - filter(tour_purpose %in% c("Work","Work-Based")) %>% #work and work-related - filter(person_type %in% c(1,2)) %>% #full and part-time worker - group_by(hh_id, person_num) %>% - summarise(person_type=max(person_type)) %>% - ungroup() - -workers <- per[per$PERTYPE %in% c(1,2), ] - -#get list of students with at least one school tour -active_students <- tours %>% - filter(tour_purpose %in% c("School")) %>% #school tour - filter(person_type %in% c(6,7,8)) %>% #all school students - group_by(hh_id, person_num) %>% - summarise(person_type=max(person_type)) %>% - ungroup() - -students <- per[per$PERTYPE %in% c(6,7,8), ] - -hh_active_student <- active_students %>% - group_by(hh_id) %>% - mutate(active_student=1) %>% - summarise(active_student = max(active_student)) %>% - ungroup() - -#tag active workers with active students in household -active_workers <- active_workers %>% - left_join(hh_active_student, by = c("hh_id")) %>% - mutate(active_student=ifelse(is.na(active_student), 0, active_student)) - - -#list of workers who did ride share or pure escort for school student -out_rs_workers <- tours %>% - select(hh_id, person_num, tour_id, tour_purpose, - escort_type_out, driver_num_out, out_chauffuer_ptype) %>% - filter(tour_purpose=="School" & escort_type_out==1) %>% - group_by(hh_id, driver_num_out) %>% - mutate(num_escort = 1) %>% - summarise(out_rs_escort = sum(num_escort)) - -out_pe_workers <- tours %>% - select(hh_id, person_num, tour_id, tour_purpose, - escort_type_out, driver_num_out, out_chauffuer_ptype) %>% - filter(tour_purpose=="School" & escort_type_out==2) %>% - group_by(hh_id, driver_num_out) %>% - mutate(num_escort = 1) %>% - summarise(out_pe_escort = sum(num_escort)) - -inb_rs_workers <- tours %>% - select(hh_id, person_num, tour_id, tour_purpose, - escort_type_in, driver_num_in, inb_chauffuer_ptype) %>% - filter(tour_purpose=="School" & escort_type_in==1) %>% - group_by(hh_id, driver_num_in) %>% - mutate(num_escort = 1) %>% - summarise(inb_rs_escort = sum(num_escort)) - -inb_pe_workers <- tours %>% - select(hh_id, person_num, tour_id, tour_purpose, - escort_type_in, driver_num_in, inb_chauffuer_ptype) %>% - filter(tour_purpose=="School" & escort_type_in==2) %>% - group_by(hh_id, driver_num_in) %>% - mutate(num_escort = 1) %>% - summarise(inb_pe_escort = sum(num_escort)) - -active_workers <- active_workers %>% - left_join(out_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% - left_join(out_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% - left_join(inb_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) %>% - left_join(inb_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) - -active_workers[is.na(active_workers)] <- 0 - -#workers <- workers %>% -# left_join(out_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% -# left_join(out_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_out")) %>% -# left_join(inb_rs_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) %>% -# left_join(inb_pe_workers, by = c("hh_id"="hh_id", "person_num"="driver_num_in")) -# -#workers[is.na(workers)] <- 0 - -active_workers <- active_workers %>% - mutate(out_escort_type = 3) %>% - mutate(out_escort_type = ifelse(out_rs_escort>0, 1, out_escort_type)) %>% - mutate(out_escort_type = ifelse(out_pe_escort>0, 2, out_escort_type)) %>% - mutate(inb_escort_type = 3) %>% - mutate(inb_escort_type = ifelse(inb_rs_escort>0, 1, inb_escort_type)) %>% - mutate(inb_escort_type = ifelse(inb_pe_escort>0, 2, inb_escort_type)) - -temp <- filter(active_workers, active_student==1) -worker_table <- table(temp$out_escort_type, temp$inb_escort_type) - -## add marginal totals to all final tables -out_table1 <- addmargins(as.table(out_table1)) -inb_table1 <- addmargins(as.table(inb_table1)) -out_table2 <- addmargins(as.table(out_table2)) -inb_table2 <- addmargins(as.table(inb_table2)) -worker_table <- addmargins(as.table(worker_table)) - -## reshape data in required form for visualizer -out_table1 <- as.data.frame.matrix(out_table1) -out_table1$id <- row.names(out_table1) -out_table1 <- melt(out_table1, id = c("id")) -colnames(out_table1) <- c("esc_type", "child_type", "freq_out") -out_table1$esc_type <- as.character(out_table1$esc_type) -out_table1$child_type <- as.character(out_table1$child_type) -out_table1 <- out_table1[out_table1$esc_type!="Sum",] -out_table1$child_type[out_table1$child_type=="Sum"] <- "Total" - -inb_table1 <- as.data.frame.matrix(inb_table1) -inb_table1$id <- row.names(inb_table1) -inb_table1 <- melt(inb_table1, id = c("id")) -colnames(inb_table1) <- c("esc_type", "child_type", "freq_inb") -inb_table1$esc_type <- as.character(inb_table1$esc_type) -inb_table1$child_type <- as.character(inb_table1$child_type) -inb_table1 <- inb_table1[inb_table1$esc_type!="Sum",] -inb_table1$child_type[inb_table1$child_type=="Sum"] <- "Total" - -table1 <- out_table1 -table1$freq_inb <- inb_table1$freq_inb -table1$esc_type[table1$esc_type=='1'] <- "Ride Share" -table1$esc_type[table1$esc_type=='2'] <- "Pure Escort" -table1$esc_type[table1$esc_type=='3'] <- "No Escort" -table1$child_type[table1$child_type=='6'] <- 'Driv Student' -table1$child_type[table1$child_type=='7'] <- 'Non-DrivStudent' -table1$child_type[table1$child_type=='8'] <- 'Pre-Schooler' - - -out_table2 <- as.data.frame.matrix(out_table2) -out_table2$id <- row.names(out_table2) -out_table2 <- melt(out_table2, id = c("id")) -colnames(out_table2) <- c("esc_type", "chauffeur", "freq_out") -out_table2$esc_type <- as.character(out_table2$esc_type) -out_table2$chauffeur <- as.character(out_table2$chauffeur) -out_table2 <- out_table2[out_table2$esc_type!="Sum",] -out_table2$chauffeur[out_table2$chauffeur=="Sum"] <- "Total" - -inb_table2 <- as.data.frame.matrix(inb_table2) -inb_table2$id <- row.names(inb_table2) -inb_table2 <- melt(inb_table2, id = c("id")) -colnames(inb_table2) <- c("esc_type", "chauffeur", "freq_inb") -inb_table2$esc_type <- as.character(inb_table2$esc_type) -inb_table2$chauffeur <- as.character(inb_table2$chauffeur) -inb_table2 <- inb_table2[inb_table2$esc_type!="Sum",] -inb_table2$chauffeur[inb_table2$chauffeur=="Sum"] <- "Total" - -table2 <- out_table2 -table2$freq_inb <- inb_table2$freq_inb -table2$esc_type[table2$esc_type=="1"] <- "Ride Share" -table2$esc_type[table2$esc_type=="2"] <- "Pure Escort" -table2$esc_type[table2$esc_type=="3"] <- "No Escort" -table2$chauffeur[table2$chauffeur=='1'] <- "FT Worker" -table2$chauffeur[table2$chauffeur=='2'] <- "PT Worker" -table2$chauffeur[table2$chauffeur=='3'] <- "Univ Stud" -table2$chauffeur[table2$chauffeur=='4'] <- "Non-Worker" -table2$chauffeur[table2$chauffeur=='5'] <- "Retiree" -table2$chauffeur[table2$chauffeur=='6'] <- "Driv Student" - -worker_table <- as.data.frame.matrix(worker_table) -colnames(worker_table) <- c("Ride Share", "Pure Escort", "No Escort", "Total") -worker_table$DropOff <- row.names(worker_table) -worker_table$DropOff[worker_table$DropOff=="1"] <- "Ride Share" -worker_table$DropOff[worker_table$DropOff=="2"] <- "Pure Escort" -worker_table$DropOff[worker_table$DropOff=="3"] <- "No Escort" -worker_table$DropOff[worker_table$DropOff=="Sum"] <- "Total" - -worker_table <- worker_table[, c("DropOff", "Ride Share","Pure Escort","No Escort","Total")] - -## write outputs -write.csv(table1, "esctype_by_childtype.csv", row.names = F) -write.csv(table2, "esctype_by_chauffeurtype.csv", row.names = F) -write.csv(worker_table, "worker_school_escorting.csv", row.names = F) - -detach("package:dplyr", unload=TRUE) - - -#District level summary of transit tours and trips -#segment by Walk, PNR, and KNR -# tour mode/trip mode -# 9-Walk to Transit -# 10-PNR -# 11-KNR - -#tours -tours$ODISTRICT <- mazCorrespondence$pmsa[match(tours$orig_mgra, mazCorrespondence$mgra)] -tours$DDISTRICT <- mazCorrespondence$pmsa[match(tours$dest_mgra, mazCorrespondence$mgra)] -tours_transit <- tours[tours$tour_mode>=9 & tours$tour_mode<=12,] -tours_transit <- tours_transit[,c("ODISTRICT","DDISTRICT","tour_mode")] -tours_transit$NUMBER_HH <- 1 - -unique_joint_tours$ODISTRICT <- mazCorrespondence$pmsa[match(unique_joint_tours$orig_mgra, mazCorrespondence$mgra)] -unique_joint_tours$DDISTRICT <- mazCorrespondence$pmsa[match(unique_joint_tours$dest_mgra, mazCorrespondence$mgra)] -unique_joint_tours_transit <- unique_joint_tours[unique_joint_tours$tour_mode>=9 & unique_joint_tours$tour_mode<=12,] -unique_joint_tours_transit <- unique_joint_tours_transit[,c("ODISTRICT","DDISTRICT","tour_mode", "NUMBER_HH")] - -tours_transit_all <- rbind(tours_transit, unique_joint_tours_transit) - -district_flow_tours <- xtabs(NUMBER_HH~tour_mode+ODISTRICT+DDISTRICT, data=tours_transit_all) -write.csv(district_flow_tours, "district_flow_transit_tours.csv") - -#trips -trips$ODISTRICT <- mazCorrespondence$pmsa[match(trips$orig_mgra, mazCorrespondence$mgra)] -trips$DDISTRICT <- mazCorrespondence$pmsa[match(trips$dest_mgra, mazCorrespondence$mgra)] -trips_transit <- trips[trips$trip_mode>=9 & trips$trip_mode<=12,] -trips_transit <- trips_transit[,c("ODISTRICT","DDISTRICT","trip_mode")] -trips_transit$num_participants <- 1 - -jtrips$ODISTRICT <- mazCorrespondence$pmsa[match(jtrips$orig_mgra, mazCorrespondence$mgra)] -jtrips$DDISTRICT <- mazCorrespondence$pmsa[match(jtrips$dest_mgra, mazCorrespondence$mgra)] -jtrips_transit <- jtrips[jtrips$trip_mode>=9 & jtrips$trip_mode<=12,] -jtrips_transit <- jtrips_transit[,c("ODISTRICT","DDISTRICT","trip_mode","num_participants")] - -trips_transit_all <- rbind(trips_transit, jtrips_transit) - -district_flow_trips <- xtabs(num_participants~trip_mode+ODISTRICT+DDISTRICT, data=trips_transit_all) -write.csv(district_flow_trips, "district_flow_transit_trips.csv") - -# finish - -end_time <- Sys.time() -end_time - start_time diff --git a/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R b/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R deleted file mode 100644 index bcbbaad..0000000 --- a/sandag_abm/src/main/r/visualizer/_SYSTEM_VARIABLES.R +++ /dev/null @@ -1,100 +0,0 @@ -### Paths -SYSTEM_APP_PATH <- WORKING_DIR -SYSTEM_DATA_PATH <- file.path(SYSTEM_APP_PATH, "data") -SYSTEM_SHP_PATH <- file.path(SYSTEM_DATA_PATH, "SHP") -SYSTEM_TEMPLATES_PATH <- file.path(SYSTEM_APP_PATH, "templates") -SYSTEM_SCRIPTS_PATH <- file.path(SYSTEM_APP_PATH, "scripts") -OUTPUT_PATH <- file.path(SYSTEM_APP_PATH, "outputs") -RUNTIME_PATH <- file.path(SYSTEM_APP_PATH, "runtime") -BASE_DATA_PATH <- file.path(SYSTEM_DATA_PATH, "base") -BUILD_DATA_PATH <- file.path(SYSTEM_DATA_PATH, "build") - -### Names -if(IS_BASE_SURVEY=="Yes"){ - # Surey Base - BASE_SCENARIO_ALT <- "HTS" - DISTRICT_FLOW_CENSUS <- "HTS" - AO_CENSUS_SHORT <- "HTS" - AO_CENSUS_LONG <- "HTS" -}else{ - # Non-Survey Base - BASE_SCENARIO_ALT <- BASE_SCENARIO_NAME - DISTRICT_FLOW_CENSUS <- BASE_SCENARIO_NAME - AO_CENSUS_SHORT <- BASE_SCENARIO_NAME - AO_CENSUS_LONG <- BASE_SCENARIO_NAME -} - -### Other Codes -person_type_codes <- c(1, 2, 3, 4, 5, 6, 7, 8, "Total") -person_type_names <- c("1.FT Worker", "2.PT Worker", "3.Univ Stud", "4.Non-Worker", "5.Retiree", "6.Driv Student", "7.Non-DrivStudent", "8.Pre-Schooler", "Total") -person_type_char <- c("FT Worker", "PT Worker", "Univ Stud", "Non-Worker", "Retiree", "Driv Student", "Non-DrivStudent", "Pre-Schooler", "Total") -person_type_df <- data.frame(code = person_type_codes, name = person_type_names, name_char = person_type_char) - -purpose_type_codes <- c("atwork", "esc", "esco", "idisc", "imain", "jdisc", "jmain", "sch", "schl", "univ", "work", "total", "Total") -purpose_type_names <- c("At-Work", "Escorting", "Escorting", "Indi-Discretionary", "Indi-Maintenance", "Joint-Discretionary", "Joint-Maintenance", "School", "School", "University", "Work", "Total", "Total") -purpose_type_df <- data.frame(code = purpose_type_codes, name = purpose_type_names) - -mtf_codes <- c(1, 2, 3, 4, 5) -mtf_names <- c("1 Work", "2 Work", "1 School", "2 School", "1 Work & 1 School") -mtf_df <- data.frame(code = mtf_codes, name = mtf_names) -dap_types <- c("M", "N", "H") -jtf_alternatives <- c("No Joint Tours", "1 Shopping", "1 Maintenance", "1 Eating Out", "1 Visiting", "1 Other Discretionary", - "2 Shopping", "1 Shopping / 1 Maintenance", "1 Shopping / 1 Eating Out", "1 Shopping / 1 Visiting", - "1 Shopping / 1 Other Discretionary", "2 Maintenance", "1 Maintenance / 1 Eating Out", - "1 Maintenance / 1 Visiting", "1 Maintenance / 1 Other Discretionary", "2 Eating Out", "1 Eating Out / 1 Visiting", - "1 Eating Out / 1 Other Discretionary", "2 Visiting", "1 Visiting / 1 Other Discretionary", "2 Other Discretionary") -todBins <- c("03:00 AM to 05:00 AM","05:00 AM to 05:30 AM","05:30 AM to 06:00 AM","06:00 AM to 06:30 AM","06:30 AM to 07:00 AM", - "07:00 AM to 07:30 AM","07:30 AM to 08:00 AM","08:00 AM to 08:30 AM", "08:30 AM to 09:00 AM","09:00 AM to 09:30 AM", - "09:30 AM to 10:00 AM","10:00 AM to 10:30 AM", "10:30 AM to 11:00 AM","11:00 AM to 11:30 AM", "11:30 AM to 12:00 PM", - "12:00 PM to 12:30 PM", "12:30 PM to 01:00 PM","01:00 PM to 01:30 PM", "01:30 PM to 02:00 PM","02:00 PM to 02:30 PM", - "02:30 PM to 03:00 PM","03:00 PM to 03:30 PM", "03:30 PM to 04:00 PM","04:00 PM to 04:30 PM", "04:30 PM to 05:00 PM", - "05:00 PM to 05:30 PM", "05:30 PM to 06:00 PM","06:00 PM to 06:30 PM", "06:30 PM to 07:00 PM","07:00 PM to 07:30 PM", - "07:30 PM to 08:00 PM","08:00 PM to 08:30 PM", "08:30 PM to 09:00 PM","09:00 PM to 09:30 PM", "09:30 PM to 10:00 PM", - "10:00 PM to 10:30 PM", "10:30 PM to 11:00 PM","11:00 PM to 11:30 PM", "11:30 PM to 12:00 AM","12:00 PM to 03:00 AM") -tod_df <- data.frame(id = seq(from=1, to=40), bin = todBins) -durBins <- c("(0.5 hours)","(1 hours)","(1.5 hours)","(2 hours)","(2.5 hours)","(3 hours)","(3.5 hours)","(4 hours)","(4.5 hours)", - "(5 hours)","(5.5 hours)","(6 hours)","(6.5 hours)","(7 hours)","(7.5 hours)","(8 hours)","(8.5 hours)","(9 hours)", - "(9.5 hours)","(10 hours)","(10.5 hours)","(11 hours)","(11.5 hours)","(12 hours)","(12.5 hours)","(13 hours)", - "(13.5 hours)","(14 hours)","(14.5 hours)","(15 hours)","(15.5 hours)","(16 hours)","(16.5 hours)","(17 hours)", - "(17.5 hours)","(18 hours)","(18.5 hours)","(19 hours)","(19.5 hours)","(20 hours)") -dur_df <- data.frame(id = seq(from=1, to=40), bin = durBins) -stopPurposes <- c("Work","Univ","Schl","Esco","Shop","Main","Eati","Visi","Disc","Work-related") -outDirDist <- c("< 0", "0-1", "1-2", "2-3", "3-4", "4-5", "5-6", "6-7", "7-8", "8-9", "9-10", "10-11", "11-12", "12-13", "13-14", "14-15", "15-16", "16-17", "17-18", "18-19", "19-20", "20-21", - "21-22", "22-23", "23-24", "24-25", "25-26", "26-27", "27-28", "28-29", "29-30", "30-31", "31-32", "32-33", "33-34", "34-35", "35-36", "36-37", "37-38", "38-39", "39-40", "40p") -tourMode <- c('Auto SOV','Auto 2 Person','Auto 3+ Person','Walk','Bike/Moped','Walk-Transit','PNR-Transit','KNR-Transit','TNC-Transit','Taxi','TNC-Single','TNC-Shared','School Bus') -tripMode <- c('Auto SOV','Auto 2 Person','Auto 3+ Person','Walk','Bike/Moped','Walk-Transit','PNR-Transit','KNR-Transit','TNC-Transit','Taxi','TNC-Single','TNC-Shared','School Bus') -sch_esc_types <- c('Ride Share', 'Pure Escort', 'No Escort') -sch_esc_codes <- c(1, 2, 3) -sch_esc_df <- data.frame(code = sch_esc_codes, type = sch_esc_types) -facility_types <- c('Interstate', 'Principal Arterial', 'Minor Arterial', 'Major Collector', 'Minor Collector', 'Local Road', 'Ramp') -facility_codes <- c(1, 3, 4, 5, 6, 7, 30) -facility_df <- data.frame(code = facility_codes, type = facility_types) -timePeriods <- c("EV1","EA","AM","MD","PM","EV") -timePeriodBreaks <- c(0,1,4,10,23,30, 41) -occp_type_codes <- c("occ1", "occ2", "occ3", "occ4", "occ5", "occ6", "Total") -occp_type_names <- c("Management", "Service", "Sales & Office", "Natural Resources", "Production", "Military", "Total") -occp_type_df <- data.frame(code = occp_type_codes, name = occp_type_names) - -### Functions -copyFile <- function(fileList, sourceDir, targetDir){ - error <- F - setwd(sourceDir) - for(file in fileList){ - full_file <- paste(sourceDir, file, sep = "/") - ## check if file exists - copy if exists else error out - if(file.exists(file)){ - file.copy(full_file, targetDir, overwrite = T, copy.date = T) - }else{ - #winDialog("ok", paste(file, "does not exist in", sourceDir)) - write.table(paste(file, "does not exist in", sourceDir), paste(OUTPUT_PATH, "error.txt", sep = "/")) - error <- T - } - if(error) break - } - return(error) -} - - - - - diff --git a/sandag_abm/src/main/r/visualizer/workersByMAZ.R b/sandag_abm/src/main/r/visualizer/workersByMAZ.R deleted file mode 100644 index 29f9615..0000000 --- a/sandag_abm/src/main/r/visualizer/workersByMAZ.R +++ /dev/null @@ -1,114 +0,0 @@ -########################################################## -### Script to summarize workers by MAZ and Occupation Type - -##### LIST OF ALL INPUT FILES ##### -## 0. Path input data : parameters.csv -## 1. person data : personData_3.csv -## 2. Work school location data : wsLocResults_3.csv -## 3. MAZ data : mgra13_based_input2016.csv -## 4. Occupation factors data : occFactors.csv -## 5. Geographic crosswalk data : geographicXwalk_PMSA.csv - -### Read Command Line Arguments -args <- commandArgs(trailingOnly = TRUE) -Parameters_File <- args[1] -REF <- args[2] - -SYSTEM_REPORT_PKGS <- c("reshape", "dplyr", "data.table") -lib_sink <- suppressWarnings(suppressMessages(lapply(SYSTEM_REPORT_PKGS, library, character.only = TRUE))) - -### Read parameters file -parameters <- read.csv(Parameters_File, header = TRUE) - -### Read parameters from Parameters_File (REF) -PROJECT_DIR <- trimws(paste(parameters$Value[parameters$Key=="PROJECT_DIR"])) -if(REF){ - WD <- trimws(paste(parameters$Value[parameters$Key=="BASE_SUMMARY_DIR"])) - ABMOutputDir <- trimws(paste(parameters$Value[parameters$Key=="REF_DIR"])) - ABMInputDir <- trimws(paste(parameters$Value[parameters$Key=="REF_DIR_INP"])) - BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BASE_SAMPLE_RATE"]))) -} else { - WD <- trimws(paste(parameters$Value[parameters$Key=="BUILD_SUMMARY_DIR"])) - - ABMOutputDir <- file.path(PROJECT_DIR, "output") - ABMInputDir <- file.path(PROJECT_DIR, "input") - BUILD_SAMPLE_RATE <- as.numeric(trimws(paste(parameters$Value[parameters$Key=="BUILD_SAMPLE_RATE"]))) -} - -MAX_ITER <- trimws(paste(parameters$Value[parameters$Key=="MAX_ITER"])) -WORKING_DIR <- trimws(paste(parameters$Value[parameters$Key=="WORKING_DIR"])) -geogXWalkDir <- trimws(paste(parameters$Value[parameters$Key=="geogXWalkDir"])) -mazFile <- trimws(paste(parameters$Value[parameters$Key=="mgraInputFile"])) -factorDir <- file.path(WORKING_DIR, "data") - -# read data -per <- read.csv(paste(ABMOutputDir, paste("personData_",MAX_ITER, ".csv", sep = ""), sep = "/"), as.is = T) -wsLoc <- read.csv(paste(ABMOutputDir, paste("wsLocResults_",MAX_ITER, ".csv", sep = ""), sep = "/"), as.is = T) -mazData <- read.csv(paste(ABMInputDir, basename(mazFile), sep = "/"), as.is = T) -occFac <- read.csv(paste(factorDir, "occFactors.csv", sep = "/"), as.is = T) -mazCorrespondence <- fread(paste(geogXWalkDir, "geographicXwalk_PMSA.csv", sep = "/"), stringsAsFactors = F) - -# workers by occupation type -workersbyMAZ <- wsLoc[wsLoc$PersonType<=3 & wsLoc$WorkLocation>0 & wsLoc$WorkSegment %in% c(0,1,2,3,4,5),] %>% - mutate(weight = 1/BUILD_SAMPLE_RATE) %>% - group_by(WorkLocation, WorkSegment) %>% - mutate(num_workers = sum(weight)) %>% - select(WorkLocation, WorkSegment, num_workers) - -ABM_Summary <- cast(workersbyMAZ, WorkLocation~WorkSegment, value = "num_workers", fun.aggregate = max) -ABM_Summary$`0`[is.infinite(ABM_Summary$`0`)] <- 0 -ABM_Summary$`1`[is.infinite(ABM_Summary$`1`)] <- 0 -ABM_Summary$`2`[is.infinite(ABM_Summary$`2`)] <- 0 -ABM_Summary$`3`[is.infinite(ABM_Summary$`3`)] <- 0 -ABM_Summary$`4`[is.infinite(ABM_Summary$`4`)] <- 0 -ABM_Summary$`5`[is.infinite(ABM_Summary$`5`)] <- 0 - -colnames(ABM_Summary) <- c("mgra", "occ1", "occ2", "occ3", "occ4", "occ5", "occ6") - - -# compute jobs by occupation type -empCat <- colnames(occFac)[colnames(occFac)!="emp_code"] - -mazData$occ1 <- 0 -mazData$occ2 <- 0 -mazData$occ3 <- 0 -mazData$occ4 <- 0 -mazData$occ5 <- 0 -mazData$occ6 <- 0 - -for(cat in empCat){ - mazData$occ1 <- mazData$occ1 + mazData[,c(cat)]*occFac[1,c(cat)] - mazData$occ2 <- mazData$occ2 + mazData[,c(cat)]*occFac[2,c(cat)] - mazData$occ3 <- mazData$occ3 + mazData[,c(cat)]*occFac[3,c(cat)] - mazData$occ4 <- mazData$occ4 + mazData[,c(cat)]*occFac[4,c(cat)] - mazData$occ5 <- mazData$occ5 + mazData[,c(cat)]*occFac[5,c(cat)] - mazData$occ6 <- mazData$occ6 + mazData[,c(cat)]*occFac[6,c(cat)] -} - -### get df in right format before outputting -df1 <- mazData[,c("mgra", "hhs")] %>% - left_join(ABM_Summary, by = c("mgra"="mgra")) %>% - select(-hhs) - -df1[is.na(df1)] <- 0 -df1$Total <- rowSums(df1[,!colnames(df1) %in% c("mgra")]) -df1[is.na(df1)] <- 0 -df1 <- melt(df1, id = c("mgra")) -colnames(df1) <- c("mgra", "occp", "value") - -df2 <- mazData[,c("mgra","occ1", "occ2", "occ3", "occ4", "occ5", "occ6")] -df2[is.na(df2)] <- 0 -df2$Total <- rowSums(df2[,!colnames(df2) %in% c("mgra")]) -df2[is.na(df2)] <- 0 -df2 <- melt(df2, id = c("mgra")) -colnames(df2) <- c("mgra", "occp", "value") - -df <- cbind(df1, df2$value) -colnames(df) <- c("mgra", "occp", "workers", "jobs") - -df$DISTRICT <- mazCorrespondence$pmsa[match(df$mgra, mazCorrespondence$mgra)] - -### Write outputs -write.csv(df, paste(WD, "job_worker_summary.csv", sep = "/"), row.names = F) - -# finish \ No newline at end of file diff --git a/sandag_abm/src/main/resources/BatchSubstitute.bat b/sandag_abm/src/main/resources/BatchSubstitute.bat deleted file mode 100644 index 99bffdc..0000000 --- a/sandag_abm/src/main/resources/BatchSubstitute.bat +++ /dev/null @@ -1,20 +0,0 @@ -@echo off -REM -- Prepare the Command Processor -- -SETLOCAL ENABLEEXTENSIONS -SETLOCAL DISABLEDELAYEDEXPANSION - -::BatchSubstitude - parses a File line by line and replaces a substring" -::syntax: BatchSubstitude.bat OldStr NewStr File -:: OldStr [in] - string to be replaced -:: NewStr [in] - string to replace with -:: File [in] - file to be parsed -:$changed 20100115 -:$source http://www.dostips.com -if "%~1"=="" findstr "^::" "%~f0"&GOTO:EOF -for /f "tokens=1,* delims=]" %%A in ('"type %3|find /n /v """') do ( - set "line=%%B" - if defined line ( - call set "line=echo.%%line:%~1=%~2%%" - for /f "delims=" %%X in ('"echo."%%line%%""') do %%~X - ) ELSE echo. -) diff --git a/sandag_abm/src/main/resources/CTRampEnv.bat b/sandag_abm/src/main/resources/CTRampEnv.bat deleted file mode 100644 index 5f3e826..0000000 --- a/sandag_abm/src/main/resources/CTRampEnv.bat +++ /dev/null @@ -1,61 +0,0 @@ -rem this file has environment variables for CT-RAMP batch files - -rem set ports -set MATRIX_MANAGER_PORT=${matrix.server.port} -set HH_MANAGER_PORT=${household.server.port} - -rem set single node index -set SNODE=${snode} - -rem set machine names -set MAIN=${master.node.name} -set NODE1=${node.1.name} -set NODE2=${node.2.name} -set NODE3=${node.3.name} - - -rem set IP addresses -set MAIN_IP=${master.node.ip} -set HHMGR_IP=${household.server.host} - -rem JVM memory allocations -set MEMORY_MTXMGR_MIN=${mtxmgr.memory.min} -set MEMORY_MTXMGR_MAX=${mtxmgr.memory.max} -set MEMORY_HHMGR_MIN=${hhmgr.memory.min} -set MEMORY_HHMGR_MAX=${hhmgr.memory.max} -set MEMORY_CLIENT_MIN=${client.memory.min} -set MEMORY_CLIENT_MAX=${client.memory.max} -set MEMORY_SPMARKET_MIN=${spmarket.memory.min} -set MEMORY_SPMARKET_MAX=${spmarket.memory.max} -set MEMORY_BIKELOGSUM_MIN=${bikelogsum.memory.min} -set MEMORY_BIKELOGSUM_MAX=${bikelogsum.memory.max} -set MEMORY_WALKLOGSUM_MIN=${walklogsum.memory.min} -set MEMORY_WALKLOGSUM_MAX=${walklogsum.memory.max} -set MEMORY_BIKEROUTE_MIN=${bikeroute.memory.min} -set MEMORY_BIKEROUTE_MAX=${bikeroute.memory.max} -rem set MEMORY_DATAEXPORT_MIN=${dataexport.memory.min} -rem set MEMORY_DATAEXPORT_MAX=${dataexport.memory.max} -set MEMORY_EMFAC_MIN=${emfac.memory.min} -set MEMORY_EMFAC_MAX=${emfac.memory.max} -set MEMORY_VALIDATE_MIN=${validate.memory.min} -set MEMORY_VALIDATE_MAX=${validate.memory.max} - -rem set main property file name -set PROPERTIES_NAME=sandag_abm - -rem all nodes need to map the scenario drive, currently mapped as x: -set MAPDRIVE=${MAPDRIVE} -rem set MAPDRIVEFOLDER=\\${master.node.name}\${map.folder} -rem uncomment next line if use T drive as data folder. -rem !!!Note: much slower than a local data folder!!! -set MAPDRIVEFOLDER=${MAPDRIVEFOLDER} - -rem account settings for remote access using psexec -set USERNAME=${USERNAME} -set PASSWORD=${PASSWORD} - -rem location of mapAndRun.bat on remote machines -set MAPANDRUN=${MAPANDRUN} - -rem set location of java -set JAVA_64_PATH=${JAVA_64_PATH} \ No newline at end of file diff --git a/sandag_abm/src/main/resources/CheckOutput.bat b/sandag_abm/src/main/resources/CheckOutput.bat deleted file mode 100644 index 18b2c00..0000000 --- a/sandag_abm/src/main/resources/CheckOutput.bat +++ /dev/null @@ -1,8 +0,0 @@ -rem ### Declaring required environment variables -set PROJECT_DIRECTORY=%1 -set CHECK=%2 -set ITERATION=%3 - -rem ### Checking that files were generated -python %PROJECT_DIRECTORY%\python\check_output.py %PROJECT_DIRECTORY% %CHECK% %ITERATION% -if ERRORLEVEL 1 exit 2 diff --git a/sandag_abm/src/main/resources/CreateD2TAccessFile.bat b/sandag_abm/src/main/resources/CreateD2TAccessFile.bat deleted file mode 100644 index fffc318..0000000 --- a/sandag_abm/src/main/resources/CreateD2TAccessFile.bat +++ /dev/null @@ -1,9 +0,0 @@ -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call bin\CTRampEnv.bat -set JAR_LOCATION=%PROJECT_DIRECTORY%/application - -%JAVA_64_PATH%\bin\java -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -Djxl.nowarnings=true -Dlog4j.configuration=log4j_d2t.xml -cp application/*;conf/ -Dproject.folder=%PROJECT_DIRECTORY% -Djava.library.path=%JAR_LOCATION% org.sandag.abm.application.SandagMGRAtoPNR %PROPERTIES_NAME% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataExporter.bat b/sandag_abm/src/main/resources/DataExporter.bat deleted file mode 100644 index 7bf4bc1..0000000 --- a/sandag_abm/src/main/resources/DataExporter.bat +++ /dev/null @@ -1,33 +0,0 @@ -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call bin\CTRampEnv.bat -set JAR_LOCATION=%PROJECT_DIRECTORY%/application - -rem ### Connecting to Anaconda3 Environment -set ENV=C:\ProgramData\Anaconda3 -call %ENV%\Scripts\activate.bat %ENV% - -rem ### Checking if Data Exporter environment exists -rem ### Otherwise creates environment -set EXPORT_ENV=%PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\dataExporter\environment.yml -call conda env list | find /i "abmDataExporter" -if not errorlevel 1 ( - call conda env update --name abmDataExporter --file %EXPORT_ENV% - call activate abmDataExporter -) else ( - call conda env create -f %EXPORT_ENV% - call activate abmDataExporter -) - -rem ### Running Data Exporter on scenario -python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\dataExporter\serialRun.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem ### Check for the Data Exporter output files -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% Exporter %ITERATION% - -rem ### Exiting all Anaconda3 environments -call conda deactivate -call conda deactivate \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataLoadRequest.bat b/sandag_abm/src/main/resources/DataLoadRequest.bat deleted file mode 100644 index 74be0d6..0000000 --- a/sandag_abm/src/main/resources/DataLoadRequest.bat +++ /dev/null @@ -1,7 +0,0 @@ -set PROJECT_DIRECTORY="%1" -set ITERATION=%2 -set YEAR=%3 -set SAMPLE_RATE=%4 -set ABM_VERSION=${version} - -sqlcmd -d ${database_name} -E -S ${database_server} -Q "EXEC [data_load].[SP_REQUEST] $(year),'$(path)',$(iteration),$(sample_rate),'$(abm_version)'" -v year=%YEAR% path=%PROJECT_DIRECTORY% iteration=%ITERATION% sample_rate=%SAMPLE_RATE% abm_version=%ABM_VERSION% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/DataSummary.bat b/sandag_abm/src/main/resources/DataSummary.bat deleted file mode 100644 index 46f0427..0000000 --- a/sandag_abm/src/main/resources/DataSummary.bat +++ /dev/null @@ -1,14 +0,0 @@ -rem forced to update excel links for auto reporting, YMA, 1/23/2019 - -@echo on - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SCENARIOYEAR=%3 -set SCENARIOID=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\database_summary.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% %SCENARIOYEAR% %SCENARIOID% - - diff --git a/sandag_abm/src/main/resources/ExcelUpdate.bat b/sandag_abm/src/main/resources/ExcelUpdate.bat deleted file mode 100644 index 4c893e9..0000000 --- a/sandag_abm/src/main/resources/ExcelUpdate.bat +++ /dev/null @@ -1,16 +0,0 @@ -rem forced to update excel links for auto reporting, YMA, 1/23/2019 - -@echo on - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SCENARIOYEAR=%3 -set SCENARIOID=%4 - -@echo path - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\excel_update.py %PROJECT_DRIVE%%PROJECT_DIRECTORY% %SCENARIOYEAR% %SCENARIOID% - - diff --git a/sandag_abm/src/main/resources/FHWADataExporter.bat b/sandag_abm/src/main/resources/FHWADataExporter.bat deleted file mode 100644 index 7b38dc0..0000000 --- a/sandag_abm/src/main/resources/FHWADataExporter.bat +++ /dev/null @@ -1,142 +0,0 @@ -set PROJECT_DRIVE_BASE=%1 -set PROJECT_DIRECTORY_BASE=%2 -set PROJECT_DRIVE_BUILD=%3 -set PROJECT_DIRECTORY_BUILD=%4 - -# ********************************************************************************************************************************** -# STEP 1: -# Create base trips with base skims -# ********************************************************************************************************************************** - -%PROJECT_DRIVE_BASE% -cd %PROJECT_DIRECTORY_BASE% -call bin\CTRampEnv.bat - -set PATH=%TRANSCAD_PATH%;C:\Windows\System32;application - -%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter - -# -# copy reports directory -# -echo d|xcopy report report_basetripbaseskim /S /Y - -# ********************************************************************************************************************************** -# STEP 2: -# Create build trips with build skims -# ********************************************************************************************************************************** - -%PROJECT_DRIVE_BUILD% -cd %PROJECT_DIRECTORY_BUILD% - -%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter - -# -# copy reports directory -# -echo d|xcopy report report_buildtripbuildskim /S /Y - -# ********************************************************************************************************************************** -# STEP 3: -# Create base trips with build skims -# ********************************************************************************************************************************** - -# currently in build directory: rename the disaggregate data -# -rename output\airport_out.csv airport_out.csv.build -rename output\crossBorderTours.csv crossBorderTours.csv.build -rename output\crossBorderTrips.csv crossBorderTrips.csv.build -rename output\householdData_3.csv householdData_3.csv.build -rename output\indivTourData_3.csv indivTourData_3.csv.build -rename output\indivTripData_3.csv indivTripData_3.csv.build -rename output\internalExternalTrips.csv internalExternalTrips.csv.build -rename output\jointTourData_3.csv jointTourData_3.csv.build -rename output\jointTripData_3.csv jointTripData_3.csv.build -rename output\luLogsums_logit.csv luLogsums_logit.csv.build -rename output\luLogsums_simple.csv luLogsums_simple.csv.build -rename output\personData_3.csv personData_3.csv.build -rename output\visitorTours.csv visitorTours.csv.build -rename output\visitorTrips.csv visitorTrips.csv.build -# -# copy base data to build directory -# -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\airport_out.csv output\airport_out.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\crossBorderTours.csv output\crossBorderTours.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\crossBorderTrips.csv output\crossBorderTrips.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\householdData_3.csv output\householdData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\indivTourData_3.csv output\indivTourData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\indivTripData_3.csv output\indivTripData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\internalExternalTrips.csv output\internalExternalTrips.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\jointTourData_3.csv output\jointTourData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\jointTripData_3.csv output\jointTripData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\luLogsums_logit.csv output\luLogsums_logit.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\luLogsums_simple.csv output\luLogsums_simple.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\personData_3.csv output\personData_3.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\visitorTours.csv output\visitorTours.csv -copy %PROJECT_DRIVE_BASE%\%PROJECT_DIRECTORY_BASE%\output\visitorTrips.csv output\visitorTrips.csv -# -# run -# - -cd %PROJECT_DIRECTORY_BUILD% - -%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter -# -# rename report directory -# -echo d|xcopy report report_basetripbuildskim /S /Y -# -# ********************************************************************************************************************************** -# STEP 4: -# Create build trips with base skims -# ********************************************************************************************************************************** - -%PROJECT_DRIVE_BASE% -cd %PROJECT_DIRECTORY_BASE% - -# currently in base directory: rename the disaggregate data -# -rename output\airport_out.csv airport_out.csv.base -rename output\crossBorderTours.csv crossBorderTours.csv.base -rename output\crossBorderTrips.csv crossBorderTrips.csv.base -rename output\householdData_3.csv householdData_3.csv.base -rename output\indivTourData_3.csv indivTourData_3.csv.base -rename output\indivTripData_3.csv indivTripData_3.csv.base -rename output\internalExternalTrips.csv internalExternalTrips.csv.base -rename output\jointTourData_3.csv jointTourData_3.csv.base -rename output\jointTripData_3.csv jointTripData_3.csv.base -rename output\luLogsums_logit.csv luLogsums_logit.csv.base -rename output\luLogsums_simple.csv luLogsums_simple.csv.base -rename output\personData_3.csv personData_3.csv.base -rename output\visitorTours.csv visitorTours.csv.base -rename output\visitorTrips.csv visitorTrips.csv.base - -# -# copy build data to base directory -# - -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\airport_out.csv.build output\airport_out.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\crossBorderTours.csv.build output\crossBorderTours.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\crossBorderTrips.csv.build output\crossBorderTrips.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\householdData_3.csv.build output\householdData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\indivTourData_3.csv.build output\indivTourData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\indivTripData_3.csv.build output\indivTripData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\internalExternalTrips.csv.build output\internalExternalTrips.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\jointTourData_3.csv.build output\jointTourData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\jointTripData_3.csv.build output\jointTripData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\luLogsums_logit.csv.build output\luLogsums_logit.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\luLogsums_simple.csv.build output\luLogsums_simple.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\personData_3.csv.build output\personData_3.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\visitorTours.csv.build output\visitorTours.csv -copy %PROJECT_DRIVE_BUILD%\%PROJECT_DIRECTORY_BUILD%\output\visitorTrips.csv.build output\visitorTrips.csv - -# -# run -# - -%JAVA_64_PATH%\bin\java -Xms%MEMORY_DATAEXPORT_MIN% -Xmx%MEMORY_DATAEXPORT_MAX% -Djava.library.path=%TRANSCAD_PATH%;application -cp %TRANSCAD_PATH%/GISDK/Matrices/*;application/*;conf/ org.sandag.abm.reporting.DataExporter - -# -# copy report directory -# -echo d|xcopy report report_buildtripbaseskim /S /Y diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll b/sandag_abm/src/main/resources/GnuWin32/bin/libiconv2.dll deleted file mode 100644 index 747073f1b192ee74ba8612cb7ea58d908e6e7734..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 898048 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%0DI0m1qIKnk`}jJOuz?W;@^j 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

    4hSI1e4&>{2)L-9JBpSQCe{?5+- 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

    Nw&OK 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 z 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 P40!s_t4cr`a#qgovK_^ox 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%0D^wd+N0L+pJ%NBCx6r0yVgFm_N}!at^KOKr1rAf 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 z�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 z�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 diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll b/sandag_abm/src/main/resources/GnuWin32/bin/libintl3.dll deleted file mode 100644 index 4f309be7f6ad138ec936f463bfad8c1fe4fc65d0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 101888 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 YWq3vyJ7TX0)SmMR$^;*8|Ed=F9}hIWod5s; diff --git a/sandag_abm/src/main/resources/GnuWin32/bin/tee.exe b/sandag_abm/src/main/resources/GnuWin32/bin/tee.exe deleted file mode 100644 index aacad3799eb4909ac000c4f0d4fa0ec18773c471..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 24576 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 Km2mvue*JIUyc?MS diff --git a/sandag_abm/src/main/resources/HPPowerOff.bat b/sandag_abm/src/main/resources/HPPowerOff.bat deleted file mode 100644 index 94bbe66..0000000 --- a/sandag_abm/src/main/resources/HPPowerOff.bat +++ /dev/null @@ -1 +0,0 @@ -powercfg -SETACTIVE SCHEME_BALANCED \ No newline at end of file diff --git a/sandag_abm/src/main/resources/HPPowerOn.bat b/sandag_abm/src/main/resources/HPPowerOn.bat deleted file mode 100644 index fd86012..0000000 --- a/sandag_abm/src/main/resources/HPPowerOn.bat +++ /dev/null @@ -1 +0,0 @@ -powercfg -SETACTIVE SCHEME_MIN \ No newline at end of file diff --git a/sandag_abm/src/main/resources/RunEMFAC2011.cmd b/sandag_abm/src/main/resources/RunEMFAC2011.cmd deleted file mode 100644 index 37cec7d..0000000 --- a/sandag_abm/src/main/resources/RunEMFAC2011.cmd +++ /dev/null @@ -1,12 +0,0 @@ -rem Run EMFAC2011 after scenario is loaded into database -rem Takes 3 arguments: 1-prject drive 2-porject directory 3-scenario_id -rem EMFAC2011 results are written to a default location-%PROJECT_DRIVE%\%PROJECT_DIRECTORY%\output - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SCENARIO=%3 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat -python.exe %PROJECT_DIRECTORY%\bin\emfac2011_abm.py %SCENARO% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output \ No newline at end of file diff --git a/sandag_abm/src/main/resources/RunEMFAC2014.cmd b/sandag_abm/src/main/resources/RunEMFAC2014.cmd deleted file mode 100644 index 7f242d4..0000000 --- a/sandag_abm/src/main/resources/RunEMFAC2014.cmd +++ /dev/null @@ -1,16 +0,0 @@ -rem Run EMFAC2014 after scenario is loaded into database -rem Takes 3 arguments: 1-prject drive 2-porject directory 3-scenario_id 4-SB375 switch -rem EMFAC2014 results are written to a default location-%PROJECT_DRIVE%\%PROJECT_DIRECTORY%\output - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SCENARIO=%3 -set SB375=%4 - -call %PROJECT_DRIVE%%PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python - -python.exe emfac2014_abm.py %SCENARIO% Annual %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output -python.exe emfac2014_abm.py %SCENARIO% Summer %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output -python.exe emfac2014_abm.py %SCENARIO% Winter %SB375% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output diff --git a/sandag_abm/src/main/resources/RunViz.bat b/sandag_abm/src/main/resources/RunViz.bat deleted file mode 100644 index b13f18f..0000000 --- a/sandag_abm/src/main/resources/RunViz.bat +++ /dev/null @@ -1,126 +0,0 @@ -:: ############################################################################ -:: # Batch file to summarize CT-RAMP outputs and generate HTML Visualizer -:: # khademul.haque@rsginc.com, March 2019 -:: # -:: # This script summarizes the CTRAMP outputs of both ABM and Reference scenarios and generates a visualizer comparing the ABM to the reference scenario -:: # ------------------------------------------------------------------------------ -:: # To-Do -:: # 1. create log files from scripts (for later) -:: # 2. identify the input files names used by the summary R scripts and visualizer. maybe we can put them in the batch file -:: # script arguments (for testing): -:: SET PROJECT_DRIVE=C: -:: SET PROJECT_DIRECTORY=\ABM_runs\maint_2019_RSG\Model\ABM2_14_2_0 -:: SET REFER_DIR=T:\projects\sr14\abm2_test\abm_runs\14_1_0\2016_local_mask_2\ -:: SET OUTPUT_HTML_NAME=SANDAG_Dashboard_2016_calib_3_19_19_final_test -:: SET IS_BASE_SURVEY=No -:: SET BASE_SCENARIO_NAME=REFERENCE -:: SET BUILD_SCENARIO_NAME=SDABM16 -:: SET MGRA_INPUT_FILE=input/mgra13_based_input2016.csv - -:: ############################################################################ - -@ECHO off -:: Inputs from arguments -SET PROJECT_DRIVE=%1 -SET PROJECT_DIRECTORY=%2 -SET REFER_DIR=%3 -SET OUTPUT_HTML_NAME=%4 -SET IS_BASE_SURVEY=%5 -SET BASE_SCENARIO_NAME=%6 -SET BUILD_SCENARIO_NAME=%7 -SET MGRA_INPUT_FILE=%8 - -:: Default inputs -SET MAX_ITER=3 -SET BASE_SAMPLE_RATE=1.0 -SET BUILD_SAMPLE_RATE=1.0 -SET SHP_FILE_NAME=pseudomsa.shp - -:: Set Directories - -SET PROJECT_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%\ -SET REFER_DIR=%REFER_DIR%\ -SET CURRENT_DIR=%PROJECT_DIR%visualizer\ -SET WORKING_DIR=%CURRENT_DIR% -SET SUMM_DIR=%WORKING_DIR%outputs\summaries\ -SET REF_DIR=%REFER_DIR%output -SET REF_DIR_INP=%REFER_DIR%input -SET BASE_SUMMARY_DIR=%SUMM_DIR%REF -SET BUILD_SUMMARY_DIR=%SUMM_DIR%BUILD -SET R_SCRIPT=%WORKING_DIR%dependencies\R-3.4.1\bin\Rscript -SET R_LIBRARY=%WORKING_DIR%dependencies\R-3.4.1\library -SET RSTUDIO_PANDOC=%WORKING_DIR%dependencies\Pandoc - -:: Extract Dependencies.zip -IF NOT EXIST %WORKING_DIR%dependencies unzip %WORKING_DIR%dependencies.zip -d %WORKING_DIR% - -:: Summarize BUILD -SET WD=%BUILD_SUMMARY_DIR% -SET ABMOutputDir=%PROJECT_DIR%output -SET INPUT_FILE_ABM=%SUMM_DIR%summ_inputs_abm.csv - -ECHO Key,Value > %INPUT_FILE_ABM% -ECHO WD,%WD% >> %INPUT_FILE_ABM% -ECHO ABMOutputDir,%ABMOutputDir% >> %INPUT_FILE_ABM% -ECHO geogXWalkDir,%WORKING_DIR%data >> %INPUT_FILE_ABM% -ECHO SkimDir,%ABMOutputDir% >> %INPUT_FILE_ABM% -ECHO MAX_ITER,%MAX_ITER% >> %INPUT_FILE_ABM% -:: Call R script to summarize BUILD outputs -ECHO %startTime%%Time%: Running R script to summarize BUILD outputs... -%R_SCRIPT% %WORKING_DIR%scripts\SummarizeABM2016.R %INPUT_FILE_ABM% - -:: Summarize REF -SET WD=%BASE_SUMMARY_DIR% -SET INPUT_FILE_REF=%SUMM_DIR%summ_inputs_ref.csv - -ECHO Key,Value > %INPUT_FILE_REF% -ECHO WD,%WD% >> %INPUT_FILE_REF% -ECHO ABMOutputDir,%REF_DIR% >> %INPUT_FILE_REF% -ECHO geogXWalkDir,%WORKING_DIR%data >> %INPUT_FILE_REF% -ECHO SkimDir,%REF_DIR% >> %INPUT_FILE_REF% -ECHO MAX_ITER,%MAX_ITER% >> %INPUT_FILE_REF% - -:: Call R script to summarize REF outputs -ECHO %startTime%%Time%: Running R script to summarize REF outputs... -%R_SCRIPT% %WORKING_DIR%scripts\SummarizeABM2016.R %INPUT_FILE_REF% - -:: Create Visualizer -:: Parameters file -SET PARAMETERS_FILE=%WORKING_DIR%runtime\parameters.csv - -ECHO Key,Value > %PARAMETERS_FILE% -ECHO PROJECT_DIR,%PROJECT_DIR% >> %PARAMETERS_FILE% -ECHO WORKING_DIR,%WORKING_DIR% >> %PARAMETERS_FILE% -ECHO REF_DIR,%REF_DIR% >> %PARAMETERS_FILE% -ECHO REF_DIR_INP,%REF_DIR_INP% >> %PARAMETERS_FILE% -ECHO BASE_SUMMARY_DIR,%BASE_SUMMARY_DIR% >> %PARAMETERS_FILE% -ECHO BUILD_SUMMARY_DIR,%BUILD_SUMMARY_DIR% >> %PARAMETERS_FILE% -ECHO BASE_SCENARIO_NAME,%BASE_SCENARIO_NAME% >> %PARAMETERS_FILE% -ECHO BUILD_SCENARIO_NAME,%BUILD_SCENARIO_NAME% >> %PARAMETERS_FILE% -ECHO BASE_SAMPLE_RATE,%BASE_SAMPLE_RATE% >> %PARAMETERS_FILE% -ECHO BUILD_SAMPLE_RATE,%BUILD_SAMPLE_RATE% >> %PARAMETERS_FILE% -ECHO R_LIBRARY,%R_LIBRARY% >> %PARAMETERS_FILE% -ECHO OUTPUT_HTML_NAME,%OUTPUT_HTML_NAME% >> %PARAMETERS_FILE% -ECHO SHP_FILE_NAME,%SHP_FILE_NAME% >> %PARAMETERS_FILE% -ECHO IS_BASE_SURVEY,%IS_BASE_SURVEY% >> %PARAMETERS_FILE% -ECHO MAX_ITER,%MAX_ITER% >> %PARAMETERS_FILE% -ECHO geogXWalkDir,%WORKING_DIR%data >> %PARAMETERS_FILE% -ECHO mgraInputFile,%MGRA_INPUT_FILE% >> %PARAMETERS_FILE% - -:: Call the R Script to process REF and BUILD output -:: ####################################### -ECHO %startTime%%Time%: Running R script to process REF output... -%R_SCRIPT% %WORKING_DIR%scripts\workersByMAZ.R %PARAMETERS_FILE% TRUE - -ECHO %startTime%%Time%: Running R script to process BUILD output... -%R_SCRIPT% %WORKING_DIR%scripts\workersByMAZ.R %PARAMETERS_FILE% FALSE - -:: Call the master R script -:: ######################## -ECHO %startTime%%Time%: Running R script to generate visualizer... -%R_SCRIPT% %WORKING_DIR%scripts\Master.R %PARAMETERS_FILE% -IF %ERRORLEVEL% EQU 11 ( - ECHO File missing error. Check error file in outputs. - EXIT /b %errorlevel% -) -ECHO %startTime%%Time%: Dashboard creation complete... \ No newline at end of file diff --git a/sandag_abm/src/main/resources/StartHHAndNodes.cmd b/sandag_abm/src/main/resources/StartHHAndNodes.cmd deleted file mode 100644 index ba4ffc8..0000000 --- a/sandag_abm/src/main/resources/StartHHAndNodes.cmd +++ /dev/null @@ -1,44 +0,0 @@ -set PROJECT_DRIVE=%1 -set PATH_NO_DRIVE=%2 - -%PROJECT_DRIVE% -cd %PATH_NO_DRIVE% - -rem remove active connections so that limit is not exceeded -net session /delete /Y - -call %PATH_NO_DRIVE%\bin\CTRampEnv.bat - -If %SNODE%==yes goto :snode - -%PATH_NO_DRIVE%\bin\pskill \\%NODE1% java -%PATH_NO_DRIVE%\bin\pskill \\%NODE2% java -%PATH_NO_DRIVE%\bin\pskill \\%NODE3% java - -rem Start HH Manager on master node -call %PATH_NO_DRIVE%\bin\runHhMgr.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% - -rem Start remote worker nodes: SANDAG02 -set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag02.cmd %MAPDRIVE% %PATH_NO_DRIVE% -start %PATH_NO_DRIVE%\bin\psExec \\%NODE1% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% - -rem start remote worker nodes: SANDAG03 -set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag03.cmd %MAPDRIVE% %PATH_NO_DRIVE% -start %PATH_NO_DRIVE%\bin\psExec \\%NODE2% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% - -rem start remote worker nodes: SANDAG04 -set PROGRAMSTRING=%PATH_NO_DRIVE%\bin\runSandag04.cmd %MAPDRIVE% %PATH_NO_DRIVE% -start %PATH_NO_DRIVE%\bin\psExec \\%NODE3% -s -c -f %PATH_NO_DRIVE%\bin\%MAPANDRUN% %MAPDRIVE% %MAPDRIVEFOLDER% %PASSWORD% %USERNAME% %PATH_NO_DRIVE% %PROGRAMSTRING% -goto :end - -:snode -rem Start HH Manager on master node -call %PATH_NO_DRIVE%\bin\runHhMgr.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% -call %PATH_NO_DRIVE%\bin\runSandag01.cmd %PROJECT_DRIVE% %PATH_NO_DRIVE% - -:end - - - - - diff --git a/sandag_abm/src/main/resources/assignScenarioID.cmd b/sandag_abm/src/main/resources/assignScenarioID.cmd deleted file mode 100644 index 9c8e0f7..0000000 --- a/sandag_abm/src/main/resources/assignScenarioID.cmd +++ /dev/null @@ -1,6 +0,0 @@ -set root=C:\ProgramData\Anaconda3 -call %root%\Scripts\activate.bat %root% -cd.. -cd python -python assignScenarioID.py -pause \ No newline at end of file diff --git a/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd b/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd deleted file mode 100644 index c4707c8..0000000 --- a/sandag_abm/src/main/resources/checkAtTransitNetworkConsistency.cmd +++ /dev/null @@ -1,44 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem AT and Transit Networks Consistency Checking Java -%JAVA_64_PATH%\bin\java -showversion -server -Xms12000m -Xmx15000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_AtTransitCheck.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.utilities.TapAtConsistencyCheck %PROPERTIES_NAME% %PROJECT_DRIVE%%PROJECT_DIRECTORY%\input - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat b/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat deleted file mode 100644 index 487231c..0000000 --- a/sandag_abm/src/main/resources/checkFreeSpaceOnC.bat +++ /dev/null @@ -1,2 +0,0 @@ -set minSpace=%1 -python.exe ..\python\checkFreeSpace.py "c:\\" %minSpace% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd b/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd deleted file mode 100644 index 909a941..0000000 --- a/sandag_abm/src/main/resources/copy_networkfiles_to_study.cmd +++ /dev/null @@ -1,18 +0,0 @@ -@echo off - -if "%1"=="" goto usage -if "%2"=="" goto usage - -set STUDY_FOLDER=%1 -set NETWORKDIR=%2 - -@echo creating study folder -md %STUDY_FOLDER%\network_build -cacls %STUDY_FOLDER% /t /e /g Everyone:f - -@echo /Y/E %NETWORKDIR%\"*.*" %STUDY_FOLDER%\network_build -xcopy /Y/E %NETWORKDIR%\"*.*" %STUDY_FOLDER%\network_build - -:usage - -@echo Usage: %0 ^ ^ diff --git a/sandag_abm/src/main/resources/copy_networks.cmd b/sandag_abm/src/main/resources/copy_networks.cmd deleted file mode 100644 index 6219ef8..0000000 --- a/sandag_abm/src/main/resources/copy_networks.cmd +++ /dev/null @@ -1,17 +0,0 @@ -@echo off - -if "%1"=="" goto usage -if "%2"=="" goto usage - -set FILE_LIST=hwycov.e00 trcov.e00 turns.csv trlink.csv trrt.csv trstop.csv tap.elev tap.ptype timexfer_EA.csv timexfer_AM.csv timexfer_MD.csv timexfer_PM.csv timexfer_EV.csv special_fares.txt linktypeturns.dbf tapcov.dbf tapcov.shp tapcov.shx tapcov.shp.xml mobilityhubtaps.csv mobilityHubMGRAs.csv SANDAG_Bike_Net.sbn SANDAG_Bike_Net.sbx SANDAG_Bike_Net.dbf SANDAG_Bike_Net.shp SANDAG_Bike_Net.shx SANDAG_Bike_Net.prj SANDAG_Bike_Node.sbn SANDAG_Bike_Node.sbx SANDAG_Bike_Node.dbf SANDAG_Bike_Node.shp SANDAG_Bike_Node.shx SANDAG_Bike_Node.prj rtcov.shp rtcov.shx rtcov.dbf rtcov.shp.xml - -@echo %FILE_LIST% - -for %%i in (%FILE_LIST%) do ( -@echo Copying and overwriting %1\%%i -xcopy /Y %1\%%i %2) - -goto :eof - -:usage -@echo Usage: %0 ^ ^ \ No newline at end of file diff --git a/sandag_abm/src/main/resources/create_scenario.cmd b/sandag_abm/src/main/resources/create_scenario.cmd deleted file mode 100644 index 843d69b..0000000 --- a/sandag_abm/src/main/resources/create_scenario.cmd +++ /dev/null @@ -1,107 +0,0 @@ -rem create_scenario.cmd T:\projects\sr14\version_14_1_x\abm_runs\2016 2016 T:\projects\sr14\version_14_1_x\network_build\2016 4.4.0 - -@echo off - -if "%1"=="" goto usage -if "%2"=="" goto usage -if "%3"=="" goto usage -if "%4"=="" goto usage - -set SCENARIO_FOLDER=%1 -set YEAR=%2 -set NETWORKDIR=%3 -set EMME_VERSION=%4 - -@echo creating scenario folders -set FOLDERS=input application bin conf input_truck logFiles output python report sql uec analysis visualizer visualizer\outputs\summaries input_checker -for %%i in (%FOLDERS%) do ( -md %SCENARIO_FOLDER%\%%i) - -rem grant full permissions to scenario folder -cacls %SCENARIO_FOLDER% /t /e /g Everyone:f - -rem copy master server-config.csv to a scenario folder -rem to make local copy of server configuration file effective, user needs to rename it to server-config-local.csv -xcopy /Y T:\ABM\release\ABM\config\server-config.csv %SCENARIO_FOLDER%\conf - -rem setup model folders -xcopy /Y .\common\application\"*.*" %SCENARIO_FOLDER%\application -xcopy /E/Y/i .\common\application\GnuWin32\"*.*" %SCENARIO_FOLDER%\application\GnuWin32 -xcopy /Y/E .\common\python\"*.*" %SCENARIO_FOLDER%\python -xcopy /Y/E .\common\sql\"*.*" %SCENARIO_FOLDER%\sql -xcopy /Y .\common\uec\"*.*" %SCENARIO_FOLDER%\uec -xcopy /Y .\common\bin\"*.*" %SCENARIO_FOLDER%\bin -rem xcopy /Y .\conf\%YEAR%\"*.*" %SCENARIO_FOLDER%\conf -xcopy /Y .\common\conf\"*.*" %SCENARIO_FOLDER%\conf -xcopy /Y .\common\output\"*.*" %SCENARIO_FOLDER%\output -xcopy /s/Y .\common\visualizer %SCENARIO_FOLDER%\visualizer -xcopy /s/Y .\dependencies.* %SCENARIO_FOLDER%\visualizer -xcopy /Y/s/E .\common\input\input_checker\"*.*" %SCENARIO_FOLDER%\input_checker - -@echo assemble inputs -del %SCENARIO_FOLDER%\input /q -rem copy pop, hh, landuse, and other input files -xcopy /Y .\input\%YEAR%\"*.*" %SCENARIO_FOLDER%\input -rem copy common geography files to input folder -xcopy /Y .\common\input\geography\"*.*" %SCENARIO_FOLDER%\input -rem copy ctm paramter tables to input folder -xcopy /Y .\common\input\ctm\"*.*" %SCENARIO_FOLDER%\input -rem copy common model files to input folder -xcopy /Y .\common\input\model\"*.*" %SCENARIO_FOLDER%\input -rem copy common truck files to input_truck folder -xcopy /Y .\common\input\truck\"*.*" %SCENARIO_FOLDER%\input_truck -rem copy airport input files -xcopy /Y .\common\input\airports\"*.*" %SCENARIO_FOLDER%\input -rem copy ei input files -xcopy /Y .\common\input\ei\"*.*" %SCENARIO_FOLDER%\input -rem copy ie input files -xcopy /Y .\common\input\ie\"*.*" %SCENARIO_FOLDER%\input -rem copy ee input files -xcopy /Y .\common\input\ee\"*.*" %SCENARIO_FOLDER%\input -rem copy emfact input files -xcopy /Y .\common\input\emfact\"*.*" %SCENARIO_FOLDER%\input -rem copy special event input files -xcopy /Y .\common\input\specialevent\"*.*" %SCENARIO_FOLDER%\input -rem copy xborder input files -xcopy /Y .\common\input\xborder\"*.*" %SCENARIO_FOLDER%\input -rem copy visitor input files -xcopy /Y .\common\input\visitor\"*.*" %SCENARIO_FOLDER%\input -rem copy input checker config files -xcopy /Y .\common\input\input_checker\"*.*" %SCENARIO_FOLDER% -rem copy network inputs -call copy_networks.cmd %NETWORKDIR% %SCENARIO_FOLDER%\input - - -rem copy analysis templates -@echo copy analysis templates -if %YEAR%==2016 (xcopy /Y/S .\common\input\template\validation\2016\"*.*" %SCENARIO_FOLDER%\analysis\validation\) -if %YEAR%==2018 (xcopy /Y/S .\common\input\template\validation\2018\"*.*" %SCENARIO_FOLDER%\\analysis\validation\) -xcopy /Y/S .\common\input\template\summary\"*.*" %SCENARIO_FOLDER%\analysis\summary\ - -rem populate scenario year into sandag_abm.properties -set PROP_FILE=%SCENARIO_FOLDER%\conf\sandag_abm.properties -set TEMP_FILE=%SCENARIO_FOLDER%\conf\temp.properties -set RAW_YEAR=%YEAR:nb=% -type nul>%TEMP_FILE% -for /f "USEBACKQ delims=" %%A in (`type "%PROP_FILE%" ^| find /V /N ""`) do ( - set ln=%%A - setlocal enableDelayedExpansion - set ln=!ln:${year}=%RAW_YEAR%! - set ln=!ln:${year_build}=%YEAR%! - set ln=!ln:*]=! - echo(!ln!>>%TEMP_FILE% - endlocal -) -del %PROP_FILE% -move %TEMP_FILE% %PROP_FILE% - -@echo init emme folder -call init_emme.cmd %SCENARIO_FOLDER% %EMME_VERSION% - -:usage - -@echo Usage: %0 ^ ^ ^ ^ -@echo If 3rd parameter is empty, default network inputs in standard release are used - - - diff --git a/sandag_abm/src/main/resources/cvm.bat b/sandag_abm/src/main/resources/cvm.bat deleted file mode 100644 index 7ded2f6..0000000 --- a/sandag_abm/src/main/resources/cvm.bat +++ /dev/null @@ -1,30 +0,0 @@ -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set PROJECT_DIRECTORY_FWD=%3 -set CVM_ScaleFactor=%4 -set MGRA_DATA=%5 -set TAZ_CENTROIDS=%6 - -set "SCEN_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%" -set "SCEN_DIR_FWD=%PROJECT_DRIVE%%PROJECT_DIRECTORY_FWD%" -%PROJECT_DRIVE% -cd %SCEN_DIR% - -call %SCEN_DIR%\bin\CTRampEnv.bat - -set CLASSPATH=%SCEN_DIR%/application/* - -REM create the land-use data -python %SCEN_DIR%\python\cvm_input_create.py %SCEN_DIR% %MGRA_DATA% %TAZ_CENTROIDS% "Zonal Properties CVM.csv" - -REM create the commercial vehicle tours -python %SCEN_DIR%\python\sdcvm.py -s %CVM_ScaleFactor% -p %SCEN_DIR% - -REM run the java code -%JAVA_64_PATH%\bin\java.exe -Xmx24000m -Xmn16000M -Dlog4j.configuration=file:./conf/log4j.xml -Djava.library.path=%SCEN_DIR%/application -DSCENDIR=%SCEN_DIR_FWD% -cp %CLASSPATH% org.sandag.cvm.activityTravel.cvm.GenerateCommercialTours "conf/cvm.properties" - -REM summarize model outputs -python %SCEN_DIR%\python\sdcvm_summarize.py -p %SCEN_DIR% - -REM checking for CVM outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CVM diff --git a/sandag_abm/src/main/resources/cvm.properties b/sandag_abm/src/main/resources/cvm.properties deleted file mode 100644 index 6eade7a..0000000 --- a/sandag_abm/src/main/resources/cvm.properties +++ /dev/null @@ -1,39 +0,0 @@ -# location of input files -CSVFileLocation=%SCENDIR%/input/ -# input files names -ZonalPropertiesFileName=Zonal Properties CVM -ZonalPropertiesFileName2=CVMToursAccess - -UseTripModes = true - -# output file location -TripLogPath=%SCENDIR%/output/ - -# Note with TRANSCAD SKIMS you need -# you need to put the Transcad path's into your system path (see the end of the line below) -#C:\Program Files\Microsoft Visual Studio\Common\Tools;C:\Program Files\Microsoft Visual Studio\Common\Msdev98\BIN;C:\Program Files\Microsoft Visual Studio\DF98\BIN;C:\Program Files\Microsoft Visual Studio\VC98\BIN;C:\WINDOWS\system32;C:\WINDOWS;C:\WINDOWS\System32\Wbem;C:\Program Files (x86)\Dell\SysMgt\RAC5;C:\Program Files (x86)\Dell\SysMgt\oma\bin;C:\WINDOWS\system32\WindowsPowerShell\v1.0;C:\Program Files (x86)\Common Files\Acronis\SnapAPI\;C:\WINDOWS\system32\WindowsPowerShell\v1.0;C:\mpj-v0_38\bin;C:\Program Files\Java\jre6\bin;C:\arcgis\arcexe10x\bin;C:\Program Files\SlikSvn\bin;C:\Program Files\TortoiseSVN\bin;C:\Program Files (x86)\TransCAD;;C:\Program Files\TransCAD\GISDK\Matrices - - -OMXSkimLocation=%SCENDIR%/output/ -#These next two are to write new Transcad Matrix files -#CSVOutputFileLocation=%SCENDIR%/output/ -#ReadOutputMatrices=FALSE - -# To write CSV files instead of TRANSCAD matrices -# First need to tell program not to read the output matrices, to instead create them -ReadOutputMatrices=False -# Then need to explain where to write them when they are populated with trips. -CSVOutputFileLocation=%SCENDIR%/output/TripMatrices.csv - -StartZone=13 -EndZone=4996 -#RunZones=101, 102, 103 - -nThreads=22 - -# Tour counts are FirstPart_SecondPart in zonal properties files -# Coefficient files are FirstPart.csv and SecondPart.csv -FirstPart=FA,GO,IN,SV,TH,RE,WH -SecondPart=MD,AM,PM,OE,OL -#FirstPart=GO -#SecondPart=OE,OL diff --git a/sandag_abm/src/main/resources/emme_python.bat b/sandag_abm/src/main/resources/emme_python.bat deleted file mode 100644 index 4059839..0000000 --- a/sandag_abm/src/main/resources/emme_python.bat +++ /dev/null @@ -1,31 +0,0 @@ -rem ////////////////////////////////////////////////////////////////////////////// -rem //// -rem //// emme_python.bat -rem //// -rem //// Configure environment and start Python script to run Emme-related task. -rem //// Passes the input script name and one argument for the python script. -rem //// 1 : drive, e.g. "T:" -rem //// 2 : full path for working directory, including drive -rem //// 3 : full path to Emme python script -rem //// 4 : single argument for python script -rem //// -rem //// -rem ////////////////////////////////////////////////////////////////////////////// -rem -rem if necessary can set the EMMEPATH to point to a specific version of Emme -rem set EMMEPATH=C:\Program Files\INRO\Emme\Emme 4\Emme-4.3.5 -rem -rem -set MODELLER_PYTHON=%EMMEPATH%\Python27\ -set path=%EMMEPATH%\programs;%MODELLER_PYTHON%;%PATH% -rem map T drive for file access -net use t: \\sandag.org\transdata /persistent:yes -%1 rem set the drive -cd %2 rem change to the correct directory -rem restart the ISM as script user, must be configured to already be connected to mustang -taskkill /F /IM INROSoftwareManager.exe /T -PING localhost -n 5 >NUL -start /d "C:\Program Files (x86)\INRO\INRO Software Manager\INRO Software Manager 1.1.0" INROSoftwareManager.exe -PING localhost -n 5 >NUL -rem start the python script with one input -python %3 %4 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/init_emme.cmd b/sandag_abm/src/main/resources/init_emme.cmd deleted file mode 100644 index 97ecddc..0000000 --- a/sandag_abm/src/main/resources/init_emme.cmd +++ /dev/null @@ -1,37 +0,0 @@ -rem @echo off - -if "%1"=="" goto usage -if "%2"=="" goto usage -set SCENARIO_FOLDER=%1 -set EMME_VERSION=%2 - -rem add EMME to PATH -set E_PATH=C:\\Program Files\\INRO\\Emme\\Emme 4\\Emme-%EMME_VERSION% -set PATH=%E_PATH%\\programs;%E_PATH%\\python27;%PATH% -set EMMEPATH=%E_PATH% - -rem delete existing emme_project folder -:removedir -if exist %SCENARIO_FOLDER%\emme_project ( - rd /s /q %SCENARIO_FOLDER%\\emme_project - goto removedir -) - -rem create EMME project folder -python .\\common\\python\\emme\\init_emme_project.py -r %SCENARIO_FOLDER% -t emmebank -v %EMME_VERSION% - -rem create toolbox -python .\\common\\python\\emme\\toolbox\\build_toolbox.py -s .\\common\\python\\emme\\toolbox -p %SCENARIO_FOLDER%\emme_project\Scripts\sandag_toolbox.mtbx -copy .\\common\\python\\emme\\solutions.mtbx %SCENARIO_FOLDER%\emme_project\Scripts\solutions.mtbx - -rem create a batch script at startup -( -echo set python_virtualenv=C:\python_virtualenv\abm14_2_0 -echo start "TITLE" "%E_PATH%\\programs\\EmmeDesktop.exe" ./emme_project.emp -)>%SCENARIO_FOLDER%\emme_project\start_emme_with_virtualenv.bat - -rem mkdir %SCENARIO_FOLDER%\emme_project\Scripts\yaml -rem copy .\\common\\python\\emme\\yaml\\*.* %SCENARIO_FOLDER%\emme_project\Scripts\yaml - -:usage -@echo Usage: %0 ^ ^ \ No newline at end of file diff --git a/sandag_abm/src/main/resources/jhdf.dll b/sandag_abm/src/main/resources/jhdf.dll deleted file mode 100644 index fd03758eae7789c3a9582d1773b6b758329fbc99..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 843264 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 z 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%0AUghYR?2?Y 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 &FYdX;dY>GC#T1w{l$c5hH5Bu6o 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

    o~a@RWsJD_N| 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%0A|W4d-jl&hHy%+Jy1}ilyjp~}Q*UNqUC=yD? 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≪NlR{6ZgUG zq!|}augfrDT)c>caIuM$;$o{)9)leX{G0Lwe7@1PCknqHkvDk0!~Irs|Hs8E$y{9A zLyB>+ON9B2i#ena7e^B}E*6t?T%1V~95)E}eVVr2$?pup3bKJZ;xl9o9)%}AlVNgO z`2Gw0iKOG=fzR?hqmFneN#NoUPPrTY(!}*eee+y~dEj}*6c54GWG*gV^a9U%T=Wq? zE*{uy`@{|JCGC`pZ;}>VJmN+A3KxgJME~RBXcEE$@XME(Kg_EH{Nxq-o$@$b@G4{c zCZF}eYhL4h02fP04ld3j1{d#i%7btv(VP_5lK^$Z7#WI-Uy~8IxWg$=z#gx2?v%Tr zkL<+7(zRTBxOf+7!NvKc9v2sr3Oo#Zy}>){Til1?UBruv{abi02EA?0{^3)f@D4^B!tmfv=__2kmc(#z z0;$Kv-d}J{;o|W` z_TCf$}Uk2BZ1a-uL?(4Zu;rpbM`r=((tczjCMOS8*WE;d&$XvD+3rR69enxz_xQi6vVl1l* z*D?JJ>$AI<5FUZgcI{$n@F=Y4X7?Y4KM>8g1ax=rV)jr+e5XejvlSONlQvxJAZzdh ze1ipabsvbq5?2>9djt1&m>_;!y!gN_W+*QDNdYcSB7Jf3@4j3*iRE}2(9e{M_Y;+e zp#MbrnsV{e{>(8v4!aEKVz%I-o3!BKxdUkjE{-BgaB&hT#{=*IlJ_3p_lM1=b1cfm zFUU?j4$nS=`qUA7p2-}-#R4({7thhZaUcAVENL7pIb8xcCAoz{M|}aMA%k5^%BMr7l%0>gv*I*6Q4W5?@1?h#8WTj{-FBs7Se`` z`IpfrxEH2eZja@HqsTP&BW@yoTx=Xp+i-C^8HkISS9LLca2NCu{d}f4g|y*fIaz~? z^RDLl!o`K80#6bz!^3b=QJ3V``vJIy=$R6Q7meT^N*x~@c)e2&y*F4FhuvuJ=YDvy zpK{)Nz3?icYf~IHl6%?*JX_(lq$e&uK(cVL!YLOwIOQ>T$0){^I$|~P;bMY#aq-U4 z_IeLP_qZ;mlyY%6(a)QSv&j-%e1nwZF?hgDww-Qx0og;jxRJzhaT965<8Vc>bDx1f z6U`qpp80$;bAZppT<}h!a}UCaCAK^OmlORwsyJl=ZNxlY4276Dk^TY$^ z5FIN7UnaWN#IJ~-a`ET@$HhHxrd(}+XG~_F8W)%`#kSuCPbE5*7giG0k0i^Olk6ut zmFFN)c>*3a-L}UMb7nB#s3R7TC@u~sWwFd%>iFS3BWJNEF_&=hNYWGcz-pq;tRnE-+w3{|;G0D4kHOU0+;gcfeojho z@n^go{^_Anu0?iT81?FSvb7>yyhrop-XEVozD@i(@KlV#v9B5_9MJND_wR?Zw%YO_ zTtU=V5ty}&zU5dhcp*`_51#Z5_ioB#@WJhN+c5lrXboYp%eNdC7u&w$o`Z|ukR`bI z9hr+K;E)}R<;Q#<1U^F2anbz)&u;37*Ao4Hiuf=op^o_0kIZ>oY$ro-aSO@As2V;6-7{%PmGSW3dwNfHm?VfZ;o|CIYA?7D~ZqFg*^FMW-R zhmjUs>`$6-FPur#9&s<}%YMZD|Kh%lyW!1$(4n4OutUOcdjXj}0a{hR%Z?yi}p6c>La zs*`}XceC3D;ezg&W(jpd@Twk}_I?6=``G$H_$ATx<=!{byh+-qFW%4zvR*gw;_w(vZKkpx?a6f#LsE*k6uuQWC7dMi6 zJO;NA{p_L`@5#APE*^Avrs=@la01bNEC9Fm$~3v3(dY1#BW!Ph59Wi?1DlSN-r7qPZl#N4oR1kGSyoOtXvnVk6m#i|3udoWXtY+`e{O zAH1iZ^$^_8W8Dp>65W%7aE((Qh5Mgu`@{{e?r-O$AC{34+8{0_MYxzcfNKpGGl>Tm zGX`>=xEmgOirtR~))1Xn1P(vdx*xv59}7}>3?AZjt_7HP4*k!2o%kCmW1nKmxm z?P0&Gna8*|m}KETcyWO*la>G>tk5t!!ZI>KErpQx|I9b^RM z;*VqqF76=%aWgX0d`)z{Bw)%Y<{RbW7!tw7VWTt6B3xWZ=Hg-_nTAK;>qI|;C;G-P zKXCE+v6-fmI%3W^#s(J$kapY)^KY{EWG}3_nQ>v;2%KNSvGEYxcLM(n7kf=)d~vZi z3E&<$ds3$T{v&)bz?kl4PQpHuIWBd?r$`JJSCc3nfxl0qow%9K@yQUzLOgf|*C8&t zNiHt-x|QSM;vJ>TTU=a1e0Uh1Hq(xm7k)Y0j(Y-@-fo|j;&*ql4fVy{q#S3#S@S5- zd=T^RqMf*S6N%sf_#WxGmHQ`r=5E?gx%dj{z@zXZGMhSL?Of&qE{>g_X$o<14#~&G zRU{9O!2R!G9`L*neWZeN@g`D+2jIPA4=z4jmTBU+*nBT@8yDXrYjE)oQjePjndUw+ zWLvi+w>-eP;Nmx=JDz~^7t(LIxFJM8<6_2xvEN9K zK3u~!kBiGmIv#-s)zTi^4c8I9AI9K8b=*%V7Z;HcxVVB0)wb}cRm^`}ENq}2E?!CI z;(oZ6Y{kX*8yQnvjFTuX?j*i%dEbLKuI75g#g`*o27NhjNyrx|ymdw2|{ zJwsnpNBoVH;9|$iD!6b@{5mJwfEo2EEgWa1r)^~iK1221y_ERoiL)vgZ{GRA{ z5E8KbdFvtA^+nqr7d-GKJ14~wQb2w2d6I{VnJ-%xPjy_JZ3d5DW8Z!#a$emIYe!Nsh%*bnZ4 ztBHQjIRZaWIoG3j%G-=Pb;R8yiknvYnW&DKwvKCg2V(~}lXP6%Ms`tOOnry3(6M0K zdfU%&xZei5A2(c1G#?_cexvjL1GC;`4zN#gDe>ZB9r551xQ&Ey@yr<44=$cZN^l>X zPhz-u_WRsRaq&u0kNaUc$=b>F4iDW#`zaTXAzN_|oJEFFM{FVmxY$DSaPec3i^t)> z57;j*Ued;M3l~R{VmtsJC;GYZF#MXRZ^d38(oV|7Gf55Zh1YM^cr(_p{$tLI@(4U; z3(qXv1K%Khzh|ug*zhT-C~WC$LGp*Y9F#b?MG zT)cNX*B~ygBIUTaj?Bhm@Ujj&zCKt-@_*p_aqtz=2Ny5>mj1`ZJ4q+oipPA%dEp*- z{`Z^%b$sx#ADI92Ll{PWWUMI{uT9wbBd8Uqz!lwzDMG?xRk$8+l-6LNeGX?zez!Y z&n@BVj4YFbi_ems)QQ3ex@4IW%EeKcS*8dV-|v=X3h_AX-Xkk{?YQ8PM8Bsdo=%#n zBObR;mZ`+WTgW0@+)C!+33$f7S!Ne5c9Jc)*vpk=HsEeJmK6NNI}bd6f7*Hz$A?}Cfz^vpP7Oy62C>Jvi&N3CacmOHG-7rAR&x{3p_s}fUPI;1e zD;|e;_R31WuLj{=N7z0I!Iz1bZN;;W$ud20@dDBv_rVa++HvBS#80{S6Y=5ZSo)9X zZ}UaqV|iI-8s%Yl^>JCr@2C6Wb;nb0HRlMciPi-Y_dOxYgsCqcK|;6}UPfHpd&PUn zPRhlvNE_8$g@!B=HtJ4!aM`GJSXPynypb z4lXVx>3A4^NYrm}c-<*ke0E45!gt9u>WD|3##rFuiNuS0;O2bWo;ciex^;5~~l;b4;UEAJuDoX?!0TzJv+qnhttp07jGkT@ethaYL4|V{RU4fq93?ddtt^k%oFN}2VKiO z3Kx$jHMj>ZCa&N490@jE#~h8Z0^L0A>jks}heU`~5b8)en z1aPsI`0)rFeFJj<55Q;07F^tXBjb*XUy=x(fJ6LTYn^=N5APY7W%f`m-cNSoA(0eO zNBnLy{fUcz5)W?1WSJR6b0Y|!C;H4m>>y3l5mU!<-@wHJQjUv5$!uJFl&FsQGAY4h zaQ-+uULm+(Joib;LvU6J^8^pVvdQd+ukl0h)lz%^jKXVX^9-PlAI`hOj$;V+ywe#| zIL7e+Y@Ne+{lRhJq4&@~xEqcu=iKoC9P}W4f_vc&ix~^t54%4}J8%~~YX#?`I^L{Y7~Ij(pvg2VkPbmW#f1_PoS*-m!figMHWAbMe55 zF?*~4{Hu+*&Hl}Y^i?~3jR)cKPdR5i0{eVn`^^LQ|Jpvw+^}q`-De1X?|1@E|CTwX z{lK-qGQM~W-m{yzf`?$&zvv6@doI{FJ=^5r9=M}hHfyUe_uzT1Y~#m$aQ*(-X3Jmf z2QEDzJ9*EH!}ku(PJaFqgOht^o4M2v!rb23W(n?w!;j52O}HPvePXuRfX86!z-+S> zcfotk$ToZM5Zr!Fwi&U9Ho&w(yMGt_X=t{|q1;@OZH~S&+mz!Tu_)X0#Y6D<5!q%4 z9);J8WdFDyUObxp;}Q64vE9EJpKTtTkZq<>9)|hTZ2P_NnOS!KQTQ=wqmJmkopZs( z4R>anN<0S5Tzf1rWxhRD0N%HNzSv9K;MEH`1|Em0q3q;6O)MlK%EjqqHZCq8CE6Cg zM|91Kr#;BADHlhQCS06MDpenTLiAaXc;`cGOS!m+Y{kQH2k|qe;+$Ig85fr{WScBp zjF7%~l8v_91LqUnpF{9vqTgi`eYyAsCO+|IB&u#unxV z<$id|Tbw)Yg+Zd9p%T;I&NgL~i#enecf$&z@3V+4>$6P<<>Hu)%ne-JLt1drAEQ1l zmXalS5PnL6rhAeL-p@A0xcKa*Y%>fO*O4K(_#Mf^#h-`^7gIlA&fzY2E=jKO0zGnpcf5EX*x+i&aoMYhu_$n#H z#UH<74&!3^*V(2I?%A4c4%^1L;BM$8O}My>RN&&UZ@AWQ@n$j&7Ynx2r>YN+?XdS5 z4;)Prl#BcSK%e8{A*2O&!+R67lXrj+Tug?gc2DwUyZ{$x?#ebjaq$V#9Tx}u%J{J# z-*4IGs^6Xa8=Tk4SW+hhKi$ps#d}2@-uS1T&t^}y>GhYLpKkcL>6-jnBMkedbTy0E zR{V_waj|=9SN^^kV+RwY2KT0QHBWTun!HBB@XYM4$>+5fen<3uL($u{tBI*TypBY1 zKiqm6!4t4H-|okKdRMdVOpdG1zF^fL>k&BO9P55~`Vi}0xbuAL3ApY;>oIuMWwbku zW5H&k_C(?5mv=RND38MfudwA};gwy@P|C$iNCEDHlZJIoj!^)9MD*?*hh2u-ZN(SJ zZ0d^#Ue(o%!Noz2d*L{uHi!pb-PP1nN1RR;;o?%q!*D}U*W@)4gM&wO<>$P+Cm9^o z)wJN=G3xKGrUQ?_QFnDUu5{+}oUY~$Qt=VjFI++NcU2;Ao#WzHjwj%LciTE#p#T2w z|EC^cr3(JX-{j%{Nl^dN|NbdwOFnR#51OjE*?8!u_i6Gl!P zKW^;QX;+P%GIjiAV<%7NC}YR?ri`C4zGUo8W2gStfBc{S`Mgp#okl~3j-4^K#9gBQKdbl1 zsTWS1G3l1EQ+m6nk3VPh=&@5dw{as&rjG4BsNkgk+V_8M{9pV3A2t4OM=v<(KiXPw z(tn+_?SnC617jzS89QA^qaX|mx z?ujEO@Q+LW8HIjV4L;q|Z}1hvdb?-*)Ay(M^YrU~vZsIle)+xKfhm)8qLZeaGi7w~ z_-SKDPn$l49gLhX=G1}w*Xg55#*e;u?95^Gi-v97==@OwMx8$Lq3yO*y`S;)d-}=LZcZ|)HF0Vv^KOibTo7}WHshC<~8Ow7B&_&7B`kQmNiy1)-*OZwl=mmb~JW2 zX06U$owquFb>Zrw)y1nzSC_4>SY5Nad3EdR_SGG$J6H3|G`tJ(H-p2u;k!n&e5e_f!iw63f!R2QzRscWu_*2U`D z>pJQZb!Ju8s@zrXRi0J(s|r{7R{2*IuPR*?ToqbXv8rZOWL0!k>#Fuu@l}acovX6y zUG?tzy!!llZ@sU+sJ^&9P#>%>tFNdJ*EiQk>tprp^&Rzzdee~AklWyH@HFH%6gK!8 z{K?r6YzQUiL!=>^oDuPcL~>5J8r{iR;cfIK=S83~n4B5mMicL9deWW#m4TJPm7$g4 zm64Uvm9dral@uC2L*)O1%rfKVdqVVgU}Y)w%BWdE-5P2)Q@@oh+S#UKWnyLLN>i0p z<*Lf9a#!V5d8+cOyj6u&zN(@se^qf+psKVgSXEXPs;a08SJhNSs+y~!RjpOAs`jdQ zRYz5#s>qM z$uajPN4+RH?t$dUmnFwOoE-h;(2GLou$QI9E}kva`~x%v$MLnZMGzvT&tuWzkCi|1rA%Ij-*i7}5V6%jkcM z;{Wr=X}mhPGF>%!tLV&s|MzbX{M!Tn_Q1bA@NWGYleHOP!tnD+r?BXl4 z%dfa>(iIn6l0D&qOE0~wBKtQNW?xZxY4*jJW*43{Ec=qnCSG`K=g#}(n5OT1zw3SP zRb9K&`TvAEd*{1&uGz5N{5I@-N1nfu-#w?b+4;7Vzi#K#^8VYMN6YWFo&S;F9rAn5 z#S_ZdW;z8WMUlwFKeUZ>>tmnkz;;Fs%kJ2+LnLxl&q(Aq?d0{c8+t?{UFFZJ7X0Dg zEhCXO_z!;Mm#-r=eT8|+tNP-7tVPwI?vU4}F89ku6&Fsv3=8h-cZrmI<<8{b;LTgeL;1sb$-p#HQvAY%x$?v`8D|)@>kn=m$Z!J*TiQo z%k@gct0(7lABj@g#a3h7svkPLrm|-$`tJ2TBl)Z2$&%8zH904SI@#mvYu;@UiCZnu zT-@3iuO5?=7q70!8IYP6Zxe~clhMe`o`A24TbokR+q41e)ws3Oo`Z@BYb^f1lr^oj z1*uz0SjS=;v%jq95VzK)qEk^PZlOi5&!xmakxb;o63I4QQ>oN*P&Z&^AAfPnh*c<< z`KVd5hG=zliOK`IaqAP{wnW-~D^-G5jd2j!m7T}gQwgZc8xPD{C6b+|qa3=rM2Z8| zT9Oz`&1;W-K?-VCoP&~a>kYtC^9FzlqL-bpR>Z3dbFu{{^;Q<<N`YGtUQ4nrkeoN3B=zA1D|8x3Qp_5efZ`b`4^nD`;H&5R?=neF(#n1no zz5znt%h&!C^i}L5eczxr(6=2w|8x2t2NP4#53cz>^u>nJR$6b_ga6nvGO*WJt52k| zb3ECmVipjFl^ng)K2jhZ9Y_xkk|mJ$Z!mRM(OznuFK%LMiyX*R9%QpXK6W5E9^@r~ zTp`v*NU>ArBaSz1?L2|>bs)ERkW&QmD+dzyAcqU2 zw*&c-(%05H3S>{U;lW)VWLJ_rsB<8vdXS9*Njeal%CD_GFOaJp$e%pOeFCX=ASZc{ zYJpe|@(ENPLy8dFCM|1Y%1{6^-rLR6M{EjDb#xdkH%&Mn{^Js^K`+}ahl8tlK1 zY>8o#$eD_Pe48h;xr`UQEbrRJRbMkZwOjuhm1!m298dF+P{%Y|o z3aqv9>Locl7RP;(Xn>yuIW*4q6ccGPbe0uQFV5MnAT#9yUwDy(%5QFJEk zOY6v>jtOhI8bfPmLF#6DPP4z9)*k+Z{VoIx&qI7P^=Djcl%OCzaPrdVxeX}Ua}*!a z1K*2I2*v|FBB3X4=z$8P(en#K&m?Fd={W}4Wv#RC_@EiA`?*-d8%ygG+S^|I(Ku0E zUtq1WANUBmya}S*56WSG@(oA^Zgd?5&A)?1a?5W^PO z0{d3QIVi(|F(f)}4bLg1B(seeYeX+yF;>x*fcfQLD%k^|&eod+)*5@5t-x$3_|FMS5ME$7NW8#qmlw^kJ1DOTC}TxWGzvg>Sv*E zBfpY3Mltje@=GIRhs?6I_QEfekf9`GBMB*b7%XoD%lm>HA0;pcwT2XOVBFWKR6`U# zFKmHON_c7>DYaHoEB*4dro-Qc!~!HzUOsuRvaf!`FYIr>N)vXI@*N-ZmtUKyysih% z<-}cV?JB@{)bV^+VqhYtq%vEq7>t;zqFL^Yt3PgI%Sh4e)z?vFiohv52HwWaxcAm5 z{(3?@n+#?r~?1k=ZdrjLdkfwl%TVfCaYmL#lovSx$iZY>v8 zhB*PgX#|VnHR12*5B>bT1ePx__`LCE7LjT8B?0v8exHkenf~_OzU-K;{A&l=eD=l0 z$l9w_TnT%oqsyVu?K9@smLg11xv(xIAkS{|lAlW2!vK}CYKBGB^=0=&H729#;W=GL zoIA2$xS5`3Y>UYnu|=WvK~a7RMze^TR$pxWCpB*r*j|tv*)=g?XR)=e*!nINJ?Bai z#}#E|H^}i;p>?g@^HZU<{xLKr)8rc*k54azE&>Zop0NVVtJSk|7}iQ~aca+fuT^B= zUGk#P+JcEA?;4mS${fp0DB1%pTEhaavFAP83O!<>z4Z!Forvl3fnD0=AhcLC`^#&4 zqfl0FS`e`z@j*-DSwl9(XTH=mK66Kl%57|O*{!V@ARr_7pWJxzASyRB5^wA`=W)S5 z1~c~T&6NWcmjdf`R_tkzY2_euz9KUi?gket0#kf4f~oQtfT)~EKQTUvnxOd|{=B&PPBsk|W@<2A{YcI{ zG?%}61LWUcDwdTUgvL4KnA1GTA|gDus(z@aczFwDvF$L9kdmNO@vFv z&hlc3-Fd7nPAw})El&*k@AS3o?>O|A0OQyi+v%CrJBAKB{U<(qbw%4$^m3w+S{fhp z-t>CGs}wIVc1TMC0wK*DLuLG(v5c&R`lgK^Ix^kG=yEz2A8}QYRgX#CF1WuXsMrFP z?t$j*m(`Fu5x(q#v1n^4*D4h`ar<={1t{B{!}+SGgjM!U#`vxVYw(_PL(F~_mde7y zX_H-d6FIvkp7qa7_Fz_-P@9V0atYbWB~N8faN+8?c%O_;pTf6w_T&a(c0(nl%UQqS zUJSo=K=&LJoV|ToM<{SAIs%O}4557-+H|fEH)g27^+(JPycEf>cAe^ zNLQ3ai|yI(GL+|F00HzCVPQ1TY1rWUm}dSqL110)*v3ENdS@H<73@jd)S#d>w;_p` z`qn#`h@AC-%xPTj{1KIbnLWfXf$JTtNS*Z#?FUyt;ttV}+1I{oW_Gy7w0)NCtapU0 zT6_CWC5!WU8aDg3-WfxB<|{tm+aF082W`qHD%4+c*;b;9p3nZ&j8pPUqAo6f6Q{(L zMtb}@ChIl7Jl+8c_D1cW_>ZDsOJSvA?s~0E)@xWHG?B*?ga4v>#s7SaKH6W31ag5W zAcZ`5(k*c6t$ArA{@g)KZ6PS*~$f+=DGLYT-*Z+-h=*F1*_T z?cW_2)<*va~^SA&kY$SPHPzK3@l8!|&P|06Q8u$yW0pC79oD0bMl#=a2?ji3r{F(%kJTwostx2Jy} zX&`DiOlpC(j1HBLePlsX{o@i$$A6=be+Q6%-zr8jJjauNn5?1C>a8)*(F;6TOisRr zk~A`9&S;k6Fc-945h+*7at~)Lw>O)?!SDqaC9oOEk92ISpN9=XF%gV%&`UL2p9zid zWwsJmo0bNfmevD1S=QgC&aP{8Dj1+)fX&2a2R0QQL?GPPD2#5QCZqMT6sWb||4^lQ zi7_cdt#wiw!=g^iOL-I7Vr3!W#0(7^i$iV><+SBVJf#x<4eW4%wK4%2Ae;cQY}E%;)00aIjcT{2$6&+sR7@_{j@8J_BJgi&Y2J(AJM z!cO=@A26bg5?*y0T^iOxLqEdoboC0MJ+#RQglJ+3tSJ_-3?KtIpIpl+7aeXhboB3ag=82)l}<^*bvxInoX znn*^k7jf0u=e#d1-HpoWmDW&&ovZIQ&NZZH237`o|J3n4!TbMfW-f2Y@=i)Wawz_fk5$IU}Fzg>2(DP5@ywT zUwjZ*<5_TP0fHL|p4!C4de zj+Nq%IQcpX4=~2yA?FJ`l#3`{bJ?ksCz5yNQ~}#!YjpQw>q^X&A2M>L|6+)Hz#fzb zxJrBQHITBu`w#JkNXi+qpz}w5M1+4&vcS8+0yN_;=M&ZUiz^kd8GX=nT~gF)t#y zUzuV@t5Ht{LjVND)%ikx++=M-gf z%1^;0Wt?)@AUJPepgPtr2hIB1>^G8&-hLzJyLW(^y%l)J{)iu7f5kDDhX99( z#J`^iOM(Yn70%RjD_`&ctO?jlV&77*ue0|f_8B4vajsY6*fKRn@zj5W8%0o5XFmbM zt5Sg7hdf+VTw&l3N1qu939e61CnaNnjOufqyFe<}#VPT%PfmxcKI@Vk5tsm#^zZMQ z%pqHA=e(`Rl>}N>eJ-clStdT`#GFx%McRu#cQcLFy`}nm9_&#S#<8sPhgwioE|tU} zEqNxW4{1wI?A1$o&-*ihn29jNMzvu;G`6HH@pDu4qdkLm*o1SGTv%Vgkhe|80A4-i zeE^WQzcm;S0ZY#QJ>qV)_UZV+Ud;j6MJPn`f zcN6&f(E7WX%-QDdQ20iHuMUMT68I&d@c9BC5elCo@Szum@pq)aXN1E03%qkU{Q}<= zT7P#L=}(5%-ze~pL+dXR_}@e8&)51B!|0nL@cl#Uj}-XW(Ej@i{Nhk}H|;+ZzIzhs zeX1wK0z-e2I)hqm8M;9rE+-+dA3e=8Ke zQQ$udg)b8L{-OQP7x=)?`cnixBeecVfft0r`wRSnP%{NH^c>AyX+{zidc z5?X(ez^g;+&lmU?7lidcMc_|_!bb|cIJE!%0^eR5)_ynbKNP-uBI*A)6uwd5mxtoF zNZ_kO>(3YX*iihZ2>jX5`XdGYvru?{f&VHLziwJT6ux_cjQ{ar{M#t-wW07u09o{-30ztD17$?q`yNbe51hEj}60rk-)zQh0hoGv!U=Q0v{U+ zA1Ux~{`D95#i8}P34GU>F#LCylK!_s;Tr|MIuyQ0;2(#==L@_t6u&6~|5a%Hkpk}% z3hyuQvqRhOCh%iI>+c><`u`qUf1|*wL*a`Aeqm_)^A$cijQ>*v{%9Edx%RSldO3Oe z9{Wb5BQNfXXR{LE`oIsLOgPJjz!QY?oL&h0Xuu^lWtJJ~?RR2*1RcZbZ#x%oCZ2}0 zzlLxg6gBY0y;0V;;*q+(=lZKZY~==Xu^y^_cu(+o^C7JR@XaR%pTE95_&jJv;CXTP zwEkmh&sAy9HLZg9@?fD4-^CFyKky7;1)j6ho?~gx=rB83OvUb-;z@- z|IY0nT<08ZWWmp=*&^gYXZb>SF}r(U7RQ20WDpBxIGBk(?<@JRx16AB+9@I9gS zsGq=Bhr-(l{LxVO*7HgK&7ts>0uQ(63k4oNpUn|?L1_Dv1b%oZe2Bn1gu?p?e0!)p zZYS{dq42Hek^ZMc;VT9H=TP`UfnO5}pCj-KLgAAHUK9!+BJd+a;r#^OITYSb;Jd=+ z8++?0(tlni_~Pue@egy@bo@*GqLE?q-5h}*6AGUs@WVslLj-<#CivoxL3|c>2|kN` zYA5x7m8rhKw_=jB+RQjN4F8n^zc>`WP~ew@!siHlL@0ccz3Q3U4Ry*`e^QI2UKN zxi}QQQs5mz;R^*mHWWTb;3GoelLS6A6h1`Y9}f%Te?Ni05ejc7@U@}vt;0zF@=*9n zfsYM^FBJIDQ1~2y7lpzn3H;S}1&>z%LJl z&k^{=q3}rpZxIR~BJf=$Vf^hU@cK}AJAuC+3g22n`lp4$R|>pWD14#7zc?!lzc~V5 zo(UejJ!H+A#aQk%@h^r*{i00uW&Z9b@FPRv?F8OA6u$K=(!cA>F#J~v{EblfLV^D| z6h24b*M!0+3A|4ze2Bn1gu?p?eAm!0{M!kg*W~{|uK7s|c|CzBA|8oSv6W??2d_+w0Gq7vzo(I2cD0Z!f{9CgE($d|{`P=Y( z{QjuS&HsMS={|kJzfUju_&8#RG<;+cF%1mA*~oAn_%QdpaVQ*FnHdjBta>p#1E2I{ z%Nam#DCUR2OJyH;3PuRpa8LB3$?J(3^l28&RKCAf)dCXt% z*T(@%q*~e~$MguxJJEE~1A(-gZ4xJK%+PGQUl07d^8f)-Whe(8&?WxMfQS885C0ge zj+Ol3yy6MFcs_T4WHo*`kO<2`EFMvrZ17*UNJTJGQYBZ}|2UfX;)oeJYz~-7Zq&|H z^Z_`h#ts6B7$JSGdY4C^9w+aFEt{N|HBOW zqe<6RZTx#B0Fk)5GgLS$4RLuqLMmxMm5&=8LN2%HQ2NCJv!36V(<&yvL-Yx$98B-r zjB`=mVf~ZY3{pFpg(0#ey@B)IwTt}aJ2zXt=X3t@?=DGe-@TvZ))B2DOb0n8e*-gw zaNOCRx2ko-T4gob8SIXpTA7;pr7qnDy6f9u~EX`-C0iSxb_k-I9#RzT%* zG)V>4>LTlXWC+k`A6C93@(S}Z{TUO_KL?Q`z+#t z%K_}ufJeiAgpEPR*3OyVp}p~J@5`*YBk{{~adXivvV1F#fn=ATPRZ8UYoBo~PYG2F zxi~nnp{GFPh|AKdC6W{wsrVU^L77MdEqo1CaV-^0B{G$%vO$2}5`YwDy;oqM>1)Q* zs^Ygau~K?CmbB(dQc<_9t&wZQlX&O5-RGhg=OO_NZbhL{P^hyHRs&jpJQ$Lx94{0g zx&NmFyuK64xh?lX4Y_|H&@2uk)PMY)G#3Q2Edo~&n2h!;R!&Y(PS!u6uh+!&bpo$F zo@#OyjwZP?gg1VvDXJvka7E5kxu1b1%Hd-%da@5uCe|;9FhM~Tevn9x{I`mO&2%vp z7RV8f-EdIgt$TmNcDeCkqaaypZ~d1_=2E7_dFShRl><&~O*T@w*>3H2X*E{#%y_c{ zCDPCxM$C~F33JD{{wGq+ohvIOpPu=02}APdPf{)@p(1f=<46(BnQ)W|^hmhl$?5n6JO0x4R`CgI z<1fA6DiwV%pSL4yG6y$~0=Y%Wp)K7RRd29BZ;uU81NsN_M(bV*X3XAxZEu`2&g!kQ zvsxCD6Ox+F7 z`!{rABlLV8beQJc9Id?wWq1OkgNk?jSe&D!Q7pB$p?e+burGD?Q1nmhW6szyBamg) z3~qUXwheg5ncJ4rXnX7I9y&SI*JC6&X45XZ7DeUE?dW0NncKRlsKPV1hyIdsk#h`( zTI=j~L-9TtU2vL?*~Wj0wCZn>u?5GeDE}h}f#;)f0=JOH+^+Zla{-Ux)+T(%aMvA& zW4Ob4jO=&^R}4#Z_%r}S_7b%c4JVrRc)tQglB=yA`>|=Y60J(b>8R+O$<0G$ig=7l zca$*7_y6;hB0#O5EH?(Rb!8R32e6^p~HO^z_^*IdELbd*V<)fBIpHkgUeHJ z^UI3rwy2V@yj;Zr+i7X1b^s+V#gm! zoJq=>^@pA=(DC`aa>g5D-rS9W#);c?8}{8Fdes?4>JJ@h`nvoPl|u9WP`mfOr$5wI z=bKP}sH^dZ-g`nv|Bvy9*p|;9>KIc_4pUA#{?L+B^z})2?eR2|Kh&JR{K=SR^$EGDlKYHUHA<>V6hdBCiGz{%N=*R5_{ktC)hW)hq;aXYMj}E4aHnv>#sE`t}Ao|DY$3vr<)sOi_0sXiG30IixH`)K$ zVf5q8Qyu--236Vz`!T|xKX&2&KtJrDeyl!GIk^8n_EkR?Zv3A5@eeb2!}KH9=*P9D z+>g-@w&l~0hb8`2XMd*DIr=fiynY9-f1G~YaBj2uF=cQ-Kc+ro^y9_tMn9g3Ir^~% zsG#v-uRyS@dq<_!}KH1=*Kuy?#Ji{+w$p$ z)nBQ5?S7%o(T|hN>m_)-H~okW^KQ#^j;|JRHPI+R`|XuIu^^a(#_U^^eQ*oS4Y)XP zY~c+58(i2T(9O63Zd}8vw=b$3Eds8!$DkJ1Z4&49RmBBZTzr(;4`b>v@$s(o)d#7S zuOGlDo?PNJ9VgqPWBQUz2}AR~iMvInwRW$+Db;Nx<|pgsGFhBqNpabd{a1N=P?e;07kWA1}^t#t%aZ1wR^JXC^!XJpNN*#gH}s#tAHbA2aRFIIS(kIwRVrNfODk=?)olE2 z^{*iHy9}F_ql51sf76u!)?*rf8;KSv<4Fs2Bk8q zs9UYw?(Tiz&lBr^6o2TaIQ;no%nyF;PPi>+apf>$q(J-pI6THG(VG_IB>uEw(zn zXYTX7w+FN@wX}H9JJWH=Fg8L?5FL_B6L`WJC^VVbdB4o_qyhRzG zI55{JV_!c8$y;wbccEbaL4w}qO4a9h`oVXHau?}QM9+&}ddqlNYj3>6(WMwaZGfF6 ze%*ldFF<|A8zL_tTWnrhwcNH$;3Ed)ggxpNy338;6}iVbHyF#d&Up{e!vnnweg=Az z(du82?6vj<+G~9t6z<>nzg%D37bOGt6+^EFHMviDja_k~bDuKqeBu>G2+HT-qS2+! z#me^e{q>IBJl?Sz#AxM0gHad7XyvhrQQdqU0`~x!{LQ>g8M}jeLt5PWE~|GTn8AK{ zd91apz+9mGvs1hal=DECd%f-mkZCT;u~y+O27$>f$|-E0uk2bd=smqSQZEQRlU3(1ssIy0CqYdZge;K#_Zz%Q*V;3pEdE3xIFgE5k zF5Jw6Zse;Lb)I`1w>mYXqA|9aTAo@_JZSrLgypdhbqro%v1HnuAG%tW9_Y&4WTytv%y*Z&cxv zA-u7TLy9)UxpK`JS_|;bTWE0iuiS3A602SmO?%QpePnPJ1xp4Pw%4zH~LBN=**p$0J6rNAE^#}^C*4}ul)NdGN)_i-r z-xWowAT4tp%)7Ii$t|1wXV`P>EM>mk)!n~zu6k*b&oqYgb@S*Oz@{V}%B3%EZgLGK zn{gvAW$4>|i_+)6fA>DCnT)0)^ zvas{#ATZ>9lw<@Ohe6cTF5HdmG^zXV%mPUc&xtcZ>du>apN%HH9m;Bjnrx>^Lj4Hj zNB#US{(iQ@q6hZ(o0RYPS6U?8*58<_#NE<$_H(F$Yy60t$-#1PPj}FgENnj67IIb;`mfsnk7EjCWjz~*}GV>^pIs(aRSVb$v=9- z*`gI2!!)<2k5tX=bu*F0M~r0fFnJ(9Me|YWzXO!2FxgaCg9`QUL4mQ2$=1s?U4^Cs zcla`2#@yk{O5IGQMe|VUermnOM4WgDt~x4^3x>qv&%mw>1a*qL6)ra+dC4&G*{O=| zE)cF|T!I`Xc?2wrVG?MFRUv@Uq7qj#@*^o3n&^!Jc`b0iv6pibXo)5TVG!YgpD`-M z7+fcl4hf5nbR)T7GYVo~&c#T36qJIMymU-_Y8hHkQpfB=fv@A0s+A~AWx4y0%6}P= zrjjIw!483PT#E-sE;WzqSi@Id-PFz09*^1$({btjyWlPSEclSA=oMYrG}bJeugwwi zUrqeH29x0|mWjZh?8@Ys_{;`4{HHfo{vQ$)nSgQY->JuI`7Ox-XaV=*`Pj$X@pKOZ zUEqqm_QZWXtcr0oO@^fqpnP)vEBAZJ$FxAvp19TdFT46+`uYUQ$rvF9X$*6Fv4t=3 zWkYF^0NQyPN&uWH$yN0q1*5LF%_W!%5Ix(#=QWaOT&Re8tMpR6War7Scet5v1b(Rh z)y@@#yt`64Sg{`!sSPvxpd_6;`g^O#+=7V|aX@niCq4}@7rod;dE^fIPrzG#73?iY z&8w0HfjGl#=;fV0wCgtuZm3X*4=FGU6t2LV^!ZD~r9y2i@1#Y5p5p=X)Vu{EWw*i0 z0}al08q`EFZIgZs4FKwT&FN|*VeJoI9nLm5d9!P{Z@8g$Ss~VW$e)>v?77q}{M3h_ zp?FHIH(|~sxh{1}Oq%Wd-yNiF|I|6Tyq{F&B4rFe+)xF~d^Z+4;!+JDdfmC)p!-NJ zB%^;uW60&yeCX7?P11$>mHZ;e)I@c!OW7aNA+u5ZL^BDUlXNF(&d+&VWh|v{&O+M{ zS9^9$rb3Xn6nelEYM-%?+OrZ(;^0`q&2O}4Cb&#{HpDcr)zbNJ0*yFsi|c&wx4fMV zUpkTO>umeWKXVrY#;h+~9A9Q1^FvHR8g2X^VN4qifjug54Hn|Nih=p6lVpcr=!_Ak zM2U9B7Jps-I!-5Dbb6fmnoHNy4dRObna}X7F_HbvNeH)4*RG*bB=|Jl-TOKRg$U2NK zK01Q#0V*~efP9&&(B89LqRnjVkX5*6_V$Xx*_*G6K`tl3o6N_wumC_#3(nG567ca} zz1CC;Ja`!;DkkG z%{0os0pWl9F$f-)G{V4RN~Z0igYmQ7EjVv=r9_{HMRp3@fEKi<(>Q+leL)TdT-@kR z)e@BXcMjUW^aj;&sJDI}P-gJbxNeeQBGB;07MMKwI^Mac#BR^6Hm2A(3{n{~=T!fx zY-H5h2cT^+*Zwl9^yq-oECuaecZdl@Ka_O6ssHI*RhyT<6n^Cm*D!Kf*VBOi1r*S7 z{oe@o-J^)M(J*2i4)S8($kAubj>n(1;f9NFCo1+WAZNW9`nY$vX6Pt$77&5wp8$m-?{`Js zm;-C=wJ;?C88@jv$gO|5ssG@Ay!y%X`W(-)^fsTKgQ3(5N{J6G-DtFDAEC^qSr|Z9 z#U!pGDgav&5o6bpt*FNAQul#Av zmv`hjh;)fO-=CxMJo3MuJif&M5A^Ex2ba3CQ!09VJ5*1`^BP(eW9a{Sleny+3NNQu z6@)2IU0)Q)mMGES^~>+{;mCEC2=ya?3vX)gl8SCUz7f#5J{5W{&_t8|8?v4LFEIT-_=zXoYXX&Je4MMk z6DABzM1?)zyw54u>svIqOX5ZCkUFJQg7bD z;*pz~=7a#Neb8I+`2F?L;!IRaB+r5aiF@tgB{AXuELdWy&aqgyq@pugaoN&S-7Wuj z8#Ftw*3O1Cgif8rwX`g5mKQkrxVp+@K{Bc>Th7VOje*~o?>bO3>)}DjW?}44BBHNk7jj^9^vY|9PErZ@NtO&1fJ=e!7=R@OGfiQI!Yh8CvXCH>X)we8nxY zjLI0iMg3$LEjGcktZdHLm=3End9Jd5w+mr%gk|(&$KPV0287w9{9hn5ul(Bd^6vO| z;*VHI#9_K3_LKX!;W9y{nLn{wAP(sadbd8z{I#8EDSqEs>WN=84)r+DIe+Pr#T{SW z%^_poefG2u%dlB%cLR6*_CYktl}>p4cRVW7KJA{~WPItd)_eLjGrk^$WlA>^rgu*s z;_rswUr78%Uh~87{|`2NiGP2CzxiJFrrc(*-i-kmf^`#pXz0o@yosYs!oUAU-SF_Q zjGsWf`D%#Q)2D0LGlrhy7>a+FEnlQpKN%@nvPZj< zi9vFv#grxAq(&gp%^9kZZ@djQJUZ}dtzZA8&d)kE(^HI_Amk_pLna_**XemGbSm2Z z8=l$T0@5pg!Eh_KXK_2?L@l9u9 z8e8&3OuVMBE=)!5{zMOh^f0?~wl5~SBvCiaX>8N@=d8zk=fkpDWt4knis2=$XJ>#=`w>tHyy{6L z1CZc7qBqX`>*GCZ^fRoPKTXHi@Wcd6U0{nIlq&zcHQM0x5E4gn9GfAqyMfPwt5GMo zsTz}9(GNKnmZr}#zdi6N`xTXzFTmR{9|5Iwj1_(})i3kv%aC|LjJsc}y#`oc?-z)Z zcAXzFA97Baip^Jh5EvRy}_!_oQ|WsR)y z?OA=AwLgm5U9-FfCZd5)v${5ELHN%do36C~B@CsT89>6m{0|oRsD}R|lVP>E_`g6B zfZcMcYg9`)VU|HDN@P;NqDJ>CW;V9EhL=v%8IJxlqzaTgPd;y>@dSA211zNH!p*jq zfCdNyD%{;(pk|mOEZC#-owtp;)4_Q301Vc6H6}}mx!1(2>M+c%3TTY zRVI3WqmG8T^5InaSJ-M`f|hFS1I1pt^Qcpv{PgA{ewD=gJ`Cs61a)%dRW9p*5X0eE zfVuOj?>r&Hb-wuVR@vyUwHH?Ecps$eHJMAHHrcQl5&J9H*$`hxF+UHMT8DK^-5U=B zpT8Y4pRNH4b~_J$2_NmNKX94Be`|%}@7-@vG`bn2@5eB1YKt&U1y&19?8z#7(CPm9 zaV(qhTN1G!%w`hznufVl?upR(F(u!y8%3HARGyP#6c?vQe4dyAv$BYkI)J6ND`n}W z)R05J#E@eW^$C`1@yC#gK&4uDr7))cPO<*2SXFx9#1EB7;} zr<%5|*R~oGm=91&?tEBhKXR!sT3VJhF2bU|FGjk}xe-wFy~rF)F-T-_^UOqbe2j4P zKm2<;}9;1|#L4`TmG=i0NAG zNd{2b_f~rl{%AVfxInTG4(}>3awz;|2deOID_7zFB;Q*-%QSil8l~iY-&=jek+yu8 zmci?N8$Tbxd<4f2ZAjDatxD?cuygULdTzoYXoqQ?(=Pp)?i?PL;)D;bvjAdtS4Wz09#iyMCAR)Z6lKbo?CxR9m^=4K(vxOSH)XFd&3-}Z*eru^PP479VY539P-W?7 zntg>>f-zP61!H)3o|EK!pDEkZH2Wu%p}kGc%Go4;a%I7N{u(?5)RLZgk*WRdC8|Ng z0#)fHCQNF8lbj)u?r|*Ut$-wDI2nC)e?|LoZHzm#a|H$MoS0YPnQ3myAmu+ry#Vk1 z;&kzMxX#QF+Ob@&!cIl^PWThsVCu}$L&D<6Ma#?#AhN5$tGnV=KOx02=<7lyAd42d zkL+TsP>BAt9|sN7Nm}RVD!ljKN}fusedlCVC%rt#*Sn(c5IBInPc^;gIlU`}CT`0` zWdJX)|L=p?gIOxB76!4zcc~O(irLtnewA{2y)=M)0&A%yR@2hwj)k>A2Bt)^?yM&0`LHi3GQ?PqvGne_$h?DSzLWud;E!* z4_uiR_$v{wE`UQU%T%OS7FW}qE)mf1>VnpID3t=3hA|plae^4O#+(O6%FTPcl8+aY za_=osfUAhod#3@#vg_ab21=NWe$yFt?G%-OHvqVQj$nv{Laa zin1`6K(kZc}|0^@TpKvoCh^*;pu#p+&PlYDS}<-zt5KGksjEeeh$dAsT%n zDj1EP!MZ?fAjZED4OANJU~o@EUgV&&($Qb$a~gkF_g5vKlkDw5_P(nK82xs`sm>A~ z>p7X9bRU%d>=MDe&R%w*>aVk{CrL)QI8TD3bzBw1m~I0;({GAT*yLn!rlL>$m(faL zJRJIVdQ+P65{DRQY0=Lirk^qB2P$1ZN6ZrVzs9Zx|G{&ZMC^;YcK`+&klWDCMnBBs z?4TyGVF*)3`-qWVi%_a_enJAO?KNMtK=MWP_P?eLM$jM_&t(jFE?cBX7j=0j3^(f` zxo@i?$0WHhh60wF`?h9(c^&g2 zFM|gorK?3J*YP9EdePQ}V;F`aPqBAS^EX^tsI@!&R;x6eCn}~>_5iV$;?W@i0i};% z>16a5rk|~*pN1TPwg*(UjhRtn>0P9p{?zP(Pr;2e5NntMU3BA9CX?Wi3%%W*a?dch zYD@mLH}Sq1+iE~3_Jc7Ea{O{rJ8r#gullXDQ5m0bCeD7GhS~k=m96K(xr29iTG#l5 zbB|uxdY(MW0XcIz?!=EkBVci>OT4=Edac|Rhi6v#&v#@!(+Y>>a13qksi|2tI4`Gt zbT=I|T*J9un>n+6b?cj{@n=e(zyi;DCeL)a^25+BoqXk{{Sm`6v*|0aIYIc$?s&|_ zW9qb{`89O1g^|$FUGY8i?t$;2d-}&U8Vkm)PbH@SImpFGZ0uPk%sVwc;S6*%xPATx zhj;p-Yv8d}y!xP3DAt|D4J3JLi~8AUSUE8at(-wl;HRrq1Qz2B)-3cSaVRXEB~N`o zmaq~+=cbfMgV?|Z9<(a!nYJQ<6|Mw^9c>kLNAumufx$;xXB?fsVbB@5S$Dr|z3i|< zKMw!9Y%s%1cEpH8J|Y9hd9qUc1UlYvKaQn6_|&;>0aRk>Be;n(HDbMZk0{j-hD(=G zGDGB#mQ47#v%tkVGV8XIA%w+5T@_K;FB`ZTogS@%8uJjo;g^c$~!&?!(+-~FYqmDW0m ziCuMSGZCaWCOQ$MSUMJl5Fb5G{#%V?W@sJ-qVWZX`y;zE*&=kz-TBl8g8YE`rp4pG zN%cVdL+N4V?-f7$D=O1t=r&II;XmJWWnnWc*nmkOB#7o?f!J%=nislP*$-pO+}rO> zvJpSkG9Q$hy`XGvTD<*pF-TpB=r|OYl&w!a45e>8cW<%B43Q?yC+#Zi+^7)tg< zzmdAX+BceEpd6Zeea%TvlD>JCCxE2UtJ?9s~jQ>wvfhI0kvw_a?oP7Yp$_hayP z4c4VrZE^M0i|t;IiE`t_sD=o*Vg<>$&L!g6_ScU}i7t$5VtkjY;=yo@@jZG9o#6ck z!_4`1j=QP(NYXTiZxhyzgtZm%fUH>>uMt7k5JPTx>b`_bXc$=079zk}d(U}lZD%f< z%9BX+QYEXtc zry2Sfx(xO4Y6plQ&M*T*p@*6JpN_&IzALqI9OB=C${YvU2xxWUgfZ5nx8e)Bd2u?l z_&`f_*B7wGK3ClCc1uM+!V4_F;b~;ex)w#HP9^@p*k_nJf!JJ(sA2-gvH@C~hijtfG3Dx*p9yP? z8xdHRm)xJoHmr@jQvgd6MC-EjKI{T3aXSvJzg#>B`8|nO8>izuUmZ(fnFBL0ufS@3 zV?6az@gVNUztT8;2VsTcG??@82ghe`sOaaglS?0SESnxG!cB!dXvwX1E@x6J<1<&Z zFBtS4&c|cxkJcCERlAMYhJEz!r@WVb=qFloWVQi{_?R{qtAWi5h`~llT=G8AnJ!DELmfJCzrempU+5}PO^jQi2-qUe z`mMpy8Q4GFghHHDccHkku!UUUR@8R+{K#^;k&$!Nu<3SD0X%pAW(xVu%@ihpT*~6> zNALiV?p4qld)4CD3t1o?JS8$@Vj4|H7M-&@QH?A3uP~@ED{&leL0QHbungk#;N+w= zQuD=nFz~v;IA^~fuJxak`eL`VKG*8r298)CH)_(VQ0|vs>FgqL_@m8pO`AuUHt}6N zQ3r#QlewmLMXH(BT2n(8Ogt@BYbPpEb*%?5-(c>S!>G^cXPj1&1>jmB}Ah~#B z&|P4`5UB0u|1^!Sl?>%cGPq=KnfTg%P*nTfbGRnJ=u8qVoaz z0kbCSsm)pB54sojoQz%r>}gW|-(CFWJ2zYYFP;76--R*Td;5cMZ=i?Iqs^A@*wJ79 zn)LE+d@v5LUhcDa4jc!$?J1BmS)Arzrjy7Ppx5q#QzjlV+3Af#cY@`=!;wp`{7zWO z0I^NV-}9!wd`(FCY}o)QhL&&w((hlr{uQ+Ey0c{L(2*As-1vHomKb(xfHyUu9%rHo z)Fb5+&*Z`dB{&m#^%KchdmNo>EjxP1|F=)t0t{{NtFP}QHXItd=P)E@b6VCK7KI^Ra?eB0v$>Zk&ObMa<H+C%VvYO$1rzVo-rba5cyWlR_-SoA=rq+E!(`zvlJRHaSF-w(UxYO4So;L2A zvB3?Elz5$e1<+_12d{1}Oci-vjvI!AQL}QYXuS}<(yL-Ny&ke$MnF$ckkB0ibWG52 zuCr(fQz*xR1GGB(Lot|7(O~Qk)!^cYWb{rl^m+1TChi^L8FIr>cxjSrk0WiqOgG;8 z(ckc9;Tg)CGPf!wL*~$D;OAyT{xBm2uMUfYm*rwgg`F5K?9pr<9g(rU(Wbq=z@h#q zI3e`o$1h6XG!LQEPjs%=9V++f=Hl%rTxqL_}7ANPoz1xBV8G=a}Xc2`_ye0hW=>!LS zDUEEMT@Ne8cx@iF1D0U2q)MyEnU9cpy!8a1?R!9hJ8GI6Z!a0LtBcb|&fdn`1XC-Y zT?Izz4;*iwe5t~12{iU?ysdiUhm5ydV2d)0H^a{Y?#5ukkl{xA^*hjMpGnp@R!I3- zYDOD2Bz&Wd_G^E`nSt?i&VZk|y&$vw`j;X9!zQ(qpSbQrcFOnJew}CI+B=LX%S-Rm_!l0&$iMf+ zqlxe5UjW}B2H*I;ZhjEoTTqk3`6%K$gtM(&F-F8`IC8ndg6YO2qKl=L8=E0xy|Lmk zaSnW__}DN0l}8Ww@7urNWq%V%&lI4f2Oqze1lkKlwf3kYP&B}mz&d;7J4AgSj!Sj{ z3GA3&O`7}-5kdXIT;r1mgPMZ>A|QY*t|Y>Er z7v3h4ZT`EQHMqx_nx{8L)nH{)$$FbVQ}t;WETQ%SXiK)q1bhwZ6U%z0Fo)>?b7gf- zeZt!kl;TY`qs;PMEJr(RdJ2x_>ZB%Y3)7z&+jPG3iGY|L3Eb`Dm3Koqp=*$3G5RoE zUw}WGaQziSP!A!fz9VYFIB^}${fSlbtI=H$AOZSV+_yi$sDm?9CN(FTf54bRGz(Ej zMf35e8AS7dzO=EijncT-j1Erc>+u$%Ib&ISLN=+NjJ7fee|CyAb03<48a;rYbbN^|8;4N`ZtP1?&bG@`eO%m- zcwqMbc-LH1?ghF3B^`caU@D;-rvMjO#F=YdO|AkWQwYXSW!YqO;4ZB)FeX*7kqFor zX$RX{%uHfPm2p*Y4V-rP?2|6?Bh1~3r4X=p-yf5X_5ON*ry9Z2jB1ZmZX!aO-GZ~^o-jY2qIxH@eJOLKN`etebUQP@m)lEH` z_feHcW@rKS*I?5X*Z_d((QcOb)-v-5o-(fBtLL$QjG3(45cgnE8M@NwEZUlJWK@4chw|5k?Uk zPoD{BV&FDCrP~QmQrr^{Uz$Np&Ig46t1U9-2T3mW>4?n3`XHj4NqKLCg`~{x28W$( z1fCS33^_<~Yz$WDp6AI@u|k>kZ|(mn&o3FI&HWn9r6W)5)BuUO^r1MU8HFtIDP$R= z>@vN?{R;W|K|Y1ty+pD3CyaQSLPF3P6q2(p_>);7VW)neLS9BZ;0JA2bn=urs*vn5 zlR^$c9Zro5%IiRK0@n~dm%)&>Y7j!$RcIt|-E#PPxMMw(^Cn;bjo!ET(-V@oTxU-^ zF~j;TH2yTq)auKwev0+*+mcpXXMZ{nUiLo4pX?VyJXK?1xbRb%`r}X6%Upn%SP*Nh z-@5aBD-S&&SF0X3jgGpj547bg%r(V0rH{BUrivQGr}PR{47AQCND1c(RmbAZTvjgW z3z(&_`Hb5^+nHkin^a#CApEXRaSj50SH}%h9+?X`tEpdX#v-QX&b5CbGN!9~?2h0` zTF3bk%_|{P&Mqh*8O1l;F}86s4as>-Gh}p#JJ8$_Pj;aJ3VBh-`v>; zIPFiTOf6UuzPOC64)`nHHL0E7Ln+*Yid%Bd!A+=6;tXv!dE-tkE~8C+-Q?@KH=xc( zGx%J&o-9X*o12pua~N4;a>{W1>1FK31iU{p=QM^5P`i#IF*+^`m?PfH3s4ysf5a3! zI&(2u<$}47ev1(FMrRg1>yOAqYlLye-bt<90*0S^{lN|b-jLGc+~*2BTA=IIf9?Z! z)7(zxP-8wXP<`@!fO0=L8zo;fNY3f+(!zWmCwl5Xgh0jz1E`#%&NU5Obb=aI^%B%A z##b_A(|JEr<}mG~;Y#Ozj@7V0q_cwooDV;B0mR06f>W;(UW9%?1#omKo@{dsbLJ$4 zT>XxB!7*oL%`F7?N=oOZ$t5s5WFoL9Cq42EA7y{QuMbqlWOVi&q7ja6rfUWaFWQ1; zj6gX@GoCf{3{+;M3(nDu-jD+5_VJywb85WJY|vLQ+@JPDCH>2?+xDP|0+ybOyjbblUZN7m^My0UiF0^ z=7>Yw6uYA@FUy2cYk#U`8m@8lWtsuJWdIYQFZ7%;=nGb12`Ejn&Csu)FUm|Il%KQU zbGChDbu*fCJ?59aX^tvTFdmZ~h$r!Wirav+-A^&5M!=ihw~!uVtiLc6x62oHnfu@v zepzQU`qie^YfWnX!1(%proq1*x3B)h;$O6d(aF4{5x$@GOPIN)p<(PN&)AktXPci$ z&?^s^Wt`Jryz~QUARA#Ru-2NfjODmd*b2JQpt`}Fofuxb6SvGhe=>@*E|09$62TTQ z_gsKNnAop>l4e-*0ellXC$TKk%$&^4VEol!{Lw&VqXiBtyjsSwEIWVP5n9D3tXl^F zM-<4zt2L}seSc0|UtvVda&u;4e4s1LK?;{abCdW0KVp`iw8sN{vRRthgojc-G@3fF z@~za9;7Lt#Z+}2QwsL0_7a!*KmR*y`9CeLk?3qH#gV842-kj4*sDQR^$QUucGJ)TkG zB<&g=I)cjGu2pYERS;2yA4keAWt^gg98iWb^H>JnS{W#%l*JW*(@>6Q5%FS{=X#Yb z#K;?+WQ?2_2-wJ(W}U@E&a&#kL#ZoTI(lRw9X%=*S3uMzN7db-BIlDwnh_R=mGl0G zM)1kDgi=$(-G`r*GYajQaIK6l|85pJb*nJB&i>so!sJ6*=|ogYbdt(4jg!A}JPmaQ zYk^pr#L0gKqltju+gRUY&55SFYSUhvoc%2o$<*0jA1$RH(bCY@J5{LvevFc95-Mla zxT#vBRWC$)Cm*sV$eR%IFXt&Vafme;pAUd*c7u?IYAoU# zpHrpAdRkMCxhC5wgN?*%j}-O{!}z4hK~eD+s+wS9mHp-WP^kFJ0r6b6sZ^7LHq^)5cii4g^GKosrq%U&@)p9@fI_PW4g5x z1BoBu6;u#t)5h9n5SJ;TC3v+BuK?Za9I8Wuw3o9L8BKVY$_GwYad&@amYJ?J4+sX# znSH7E|M-vSXw#uQRsr@gO3kBeQg)XH002g8e~=Y34&_hYQ@#JRpD^MIsX9yU>ETF> zB`^qqDmXGLCs%+n`@)$B4rNXtIOxcsjIuZCM8XF621oTnhVo$TzUff@>RrX_e8o%8 z@pCAj2LMRwAtYrS%I!?m-dgpq5D<`GK+y?8FaQokGBfhXr;3+zBW*NCI9^e_PvK7| z)Pq+dX7#8{R0Jbu-1HIYy#3N+@Yiknj^X+nI?#9iq!5ZsJN5e;w zem{w4mC3$B-Y3<0)Lg8Ro5ug@HU5X=A)QZ<5+*1IUa>CkgB?$AUi2^8(G>B={t%U< zLMJ;P3?iXISuaq<5c|{f1I{cbR}(PyS2u@Z`VI|3z&rr?!gP)bRimK17l72-bppC6#RWg&98J8c*3x&mZrItvn z-a*O|gke@t+`5CsF|1J^$MTd_`~c|wtf(ifoo+zM>u)I~ekb3fVkB_F9)G|<49@mRD%U5e zJWo=vRUx9Xq6Y#&3G`wknoZ^w;R{E&00&D6U!eDkStP>=^m2deaiR+U-^MTfe-*zu zSQ8jwzR?>N|3&M2jsLQD6aPIBs>K!3u*2LiSS&;QcMUX*#g5Kg%#Hsdz~{yG{&^@A8%{z3Jf@9RpYcp!J= zRDQwsO`y3U*SLT<5Ag;ZtS0lv+2~3mWr#Kp$QHJ%p9{wxIe){{ zVO_@A8x-%_5`D*Y3@ml=ALMe|5kMq zP2;c%BpXo{&>&S1mBl&~NJS54zu0P{EC!k`-e(t(#Zt-w=^rfww>tY-*cZxzS8Btv zD`kdJo-oH{jVBYa2hMpB24a7X3!c359hac3U_9I@%kM+r)X6F~kY`sO@2}NAy_QHt zu3Gl`({wev0c#t+jKYa2cz} zCL+eZo({#{hjWBTl!lh=IMYO*9@0d^*!awvPKo51?J+c4)ApK+fm@sm%53b89*zO( zI}@m(Yv<%pElea3Y3PPR&~G^y=ar!6&;6G)%0lE{JYhXmU=urbkF@<39Rx(+RmNYG z2lyL$UrcKdYj;yh4HkR3FGdf(ibW@<{`!V8dv|xGaU_!svaNOYKYk4i`F%@2Ye1?J zE?1O6Mvud!k~P`f+|Cx}ERRfdmXXdC*{r4-b^mZE>3zpgB~|7Sv5(<&MC&v%@dT)o*eac?(e&qU5v z*nK5y*V?0Fd(%3HwIg7s$=Y0xwa8Uy%O|`-j>Bo~h08e2D=ZFG!vKHjC1S~dAu9YW z@bmXW47g_b3pXNde2)RK_I#z3%z~d{rc60v;2Yn{-`0jXg{pE*s|bH#C8%oDdz>*s zf*Ewm9pCqwleq;4E00QWC=lbCUx4_IBsDvv8`T``bx7x>V`fIB$4})W$FqH^pOJT> z^$d*k)%WaWQ+^HyarO|UEFA^tD^*F4Y-Hlh4$0-IwfjYtGYRF&s=@ohl@E-$+?%0m z%9Tv~m^<3fj~meL-udC4rzA5wj%8){Jal5Owbef2`#J9S0V)FD2l!HQt`~qU_GbqP zPsmf}^EqSWNFUt+Bss$09YsCE2XJ26(W+q3H*zEO$%do1D@W_=WtAv8S}J~oEMzbT z)rgLYWu>$A?s_#NcONLd)<42}6XZ-$N&KbvTcx58-^PXbCUchWD9$c&r2$4V+!KuJ z2(yNg(FvxdVcJpyx?PJ;;v8aa%<5gpanYi9=F;}~CJz4pNekS*(^%O6FW7ueVC|T; z=miI|y)WZPNPvw?=H<~q!^v&gf-%X+!DJcP}lAr8ZXP-O{+R2HL4XDqM)`DCf@a8Sr#!Y4lNFe9r+9Tkmg6GA!dh;Up zd+l-kpb><(193RszbOsejs|dJ0p#X3j|GyL`W19FK@dnZqH+a9mO(!5eQd$`^n#oH z!}2rOm<+>`^*#I4)SsIU?$`^JH54uop}Nduu3Dz~^bo_%;VPb{RhpkqAFNYmp50Xh z(h!_aza~|oCZ8OGaP2l`uE_l<=2OofQ~OQjcRs!I6p*#&D5fMfWZh1W*_@u$YfB}y z?JPs-q%5Vhe_{ecdS=#UlFqMT9G&n4C&(sAI=x`0_L`*Q>4)*}H;?yg#k&}W0j*$p z+~Qg=LT)*R#@(M&g#l5-(-KPeN7J}a97DP<^XTSjix@-F1!JI-Tq-e0y; zeyREVb1$AnDY}>X;cJK<{;3MwW-Ja$3A{IFMij4&$q}OshE^xlZG1`rO93pW z^a9`EWo8&31ydPdD#Bw*JLfEp<$-ES8V+H}C-uj>Z%7 z=u>;QH_~Kuu3iHbztO17A1FwVT;r*#7vOrTlTd>g&+n)-pT&8EDTDTi%l2lq97@kp zJDb5#WeHt-f=Jiv&OoG$cOyUFrkURIq(N#M;;YQ&g=IMUfnYr1$7|!xcS-Ec&jZuO zhKoQf&$(2AZrMFkx7twk9;NwnrP1F>=Nw~|l|^ph)Y#}yd? zOQNxAD3j#Odz*MolsczvcsUmtdpS!TWJ=wEQg9-utW#6bi*6v#B@#(rsJ9WpkoIcVE<#~2 zm}YSkyA{up{!n>9df8$d4`W-&=)Yc3F@9*;=#}1vx?-^Cjw>eWegelK?$HuqsspPL z6xf?p)zb2DrsZnW@(Z5PCU(y4RJz(j+>2{fgmuqC%Z15ew2UF{d1~;E9RCY3Wt}Ya zjxHMd5!&USRx2?Fn=a2lm&I0F=rMOpxg!;4KXDp)+yEjLA3IXh5zoT}ohxZM){U|^ zWN%na)kTfs>Mrn#>?y&fyHQoRQBmx})WE zN9FWTDd*7zg0>kd-Z0?6ZQ_44+~gmxV=+4f#;aRCPri;=USz#WmX{)gLHkk$FGjwm zm|bAKV%Od3Z)<M@F=mHU4A+hqp(GHbTvqFCQ@d%Fq%o>UVH z0S^Xqm7bKsMU+{y4}&HptnKlEkH{ux)@-JLC9IEWYOz2s;?-_+YBb>jIyRu#YMn#N zIUs+7Of^)zn7!>b)v5vT^NXv8iiX8z*A!QG%Hf4ZOECH(1zGWJa-kr;WyKSgdAQh! z9tOlpUc_loA3!-k#b=Urf-^bdB{!4JqB4*32GrZJv)f^;x#Tls8t;yBrB6R<{5ZMp zTaGo8QD_Ex(C|Uj=w!`)?K{}KWb}oVsyvUi7Ul6(730IIN`PZCxKPfT{Q_%a5Cq)|z|41pOWmCK`f9S$+iwouhVsDLwqo0<{w$lrh>_5XUKZ3O!1-Un7twWmd8yXT-)k%o3fvsPCg zmA?Un#W>kVY~-X269i|m*3Lz9=z_MI0`Y8?C?;usnSZho|PT@LnU~$}m^{M1l9gfzUW~AYvZ?Tp;nT zMDzsohxwymL~A!fm-`BS9x9CAO_4w zG8Z20DWZEOD_fq2|2;YUsB#E>u(FF+=4 zt!JV{SeAIPHC;;345I%XOSQG#sqYwVukYn+4Z#{|4ZE05YvSk>N^86@lwwAa)I7EZ zTdeL6-Wq4bbaI2~${gjo>k8@uDfco~&=TzS9z7u&jxcglf{bg_C zT;>DtvqsI!kJ|5QGQRCkz#YgQ2VZgjKYHM$nmU#ltCA!O$9RKB0sg`X;;_!W(9ho< zXyZrl_hs;R<+sA0=J~rRXh#>x!G0AY$pbH z6Irks8=8qfjqKf>;=Y2t`QiK-4iX*uzBhk%)TsW=`8th1Z7<#z{>0Q$HvCxrWSUQ% z@wEmh+A9^m?@ix$L*K%$(&)SDf_A51^Y`qdy-2gib5NOPZ%U);i))XMcoP_g`dyAa&NF5_+Y9P{@B9PT z9+NR=Jw{V}ZTrVtr~$4J_j)y~q2p}%@-dwJb2pS`PRPqi_>;+&DkY1>~A>ZBRd4bLlc&fl%fDfGqDuonePU?uIl{vT~`0%!H~|Bp}0)JT~uqs7vV zi6P4{p%mt-sXLk$3aMy_WKU)&T1`Xac9ZOc5S8o^Qti%eT$XB^0<-a9CkOJq>d+ssE$Xd z{WVIrtx)(LGDrD-cc;oO6{X7!*F}|Nl+DD?Se)(>XETe{IpM)b8*9J#lpz)hUjo+V zQVoi9&n-!1lmzZk{{LQdG>G`3w_?$ut3T9qRbV;-Q3dsmTZ?{;AMCjE`a@52^gbIc-o`Se%t~}I#^=Wsg|ePn8~a#Wqmd1pwTs1Z)}er&80l_EI59g zh+1OZ@wZeF9W&sNYG+I+Q-d5owr8@ANG2l;XZ)D(Kx0e4-V}&~QRLyfSEJlkS z`=Hd!XwgUQn^LWS4;X5nN}WI>!~D=nYZqj#V?qQIaX?n_U|A2~861*WY$HVjim8IQ z|JQUMm|2aEIN2Pg+;X*j|6(Y?`UBvlDW2V(_`zZZh-?`40FgncUa_M@2>?KWtk(Ye z#giM+9sL8x1l@1`VcrUM1uS=b1(W?!ywDIU(d4{7CqD5#1Zp)3i*&E@NUB7hal+0dg@By(4HPU9`8S7WTsBFnLtpJ;$GO@xP7>Cw*5sZp8 zycE08#ItbJ)TikmYYiA`$Z^MQC|pOzOqh$n8fxw0OCT0bJYZvtnRvL^F@`ZzJ$4*3 z04x!z$X*$w;rt)%JSt@E3;hIV%9I7NO`AWT1o@qjJ|*UeR*jUaPgNt`sJ*_0CXb+3 zh}WZS;^OQRXC6a{@A_%Ct(~2JVeF23#2GP zPaxW}`IL7K?Ea7f$9;$uN}%FArVp^#W!Ai%X2Xz%0OL0d1yF1m3emJ>Q`=r}6?$BI zvz~)}ug{$qbc9dG6D0t8>>_t_iMUY7iaZEj9{r)g{!Pi@oo=GHgA{@7l2 zegOR~MmbG;@k%g$9>JqN4qIMAW~y&$xwe;99_x%MsQ5geR~J58I-|%<-uQz?7^$qf`uMQrc+|XuwmwYI&O8mudK-13{PbugM7TmTMqPm2~t*d>X3j1_W_9EUsw_KEcuwY}}}Q|9zmE>FAwD+D!Qz=?7bFu+{7tfx5vhZR`e4Vz{~0A2b@*3+z* zhUCqWJi&SzGipH0V2X(u=w#Ah=Gm%}D4&DlIT9|WNoT6XCe;L_;`JyNJTHwI?0OHc zr2E+C#=3SKRI{w~P2 z!Z_pcLp_JLAvY>hnQG?BKT1#pjM2RK_K?jFF2BYh9AS&-K+W+@eE!7*p zsPRmPMk$buC|%qFq7RXW>f=yxpu1$qUPQ#P{~){V+Y9Mr1X+!}!F_{bZ@RpJcD(Ud zDTyb4W%gz_&C^#XxQ)r)-!)URWG>d;JjXmQD6{3yqxC1WH+f8TpGn2rs-CU59a8$8z^qa96bTM28COB8`X{^k=7xYqjZvs zq_r?)eluj=i!~ZLDSjyC=*#Pc^7F>fg$?B^VkuW?j+_eV-1MvM!!c7P6e6azmm>ab z%uy4AxX}&=9fCPJSlsH&(ZYNY!Xi_ohc>8|y8OSGqj^&jn4?jsS&k7ugwosML;%Te zBg~jrW#$+*uc0)U?+XWKt|Vyipp<78U81G z;*4(?cG3G67rJ4G6=-_?u!EBxVa<^ovK>cDbFb=?!|Fcsb(T#=Le@nK;5~6^M+zI~ zNRaanf2xdMa6D3*ze9EO51e$m2HTbzK9 zb~RAy_u$ zj<04jkJknEaO+vW=it>F>*9?#D7?@^UcAc=4$q8*^L}*E8X~7wNw# z%Btj%I!gqyd4sor)0wF$fmr!2BAGs9H*&2h8k8>X{R>p~?c#lQTL(%0z-mhV(5iiX8 zI=tN4JQbE4Z4pN{m07(3CMjfLt9$Cm{k_?krNd_vYHN7uvsRN=jFjjjcR@cxaxy<@ z?(FR(S<9>_S`wR1oe-iZ7Ab?xlUUmMcX5n{nH_gm89#eQ2!c9 zMgP?(=qhjVBVJiq__26>!G+%6n&G$6|3#y?)1JFFLc-xn>*IA!bpivpL`1`*;cpNt z%v)@#It=(q>oXWqNl{IGcjwO5<+NQgocekkU{+UyBqlF7`X3f$1mB*?x0(L8d!4r{ zU&Gs@d~aPtco|o#I9<%~C}BcFXiR{KpCpWzv%k5K1zEht5eVWZj3uJLi9+Tv{mmYf zlT~;smP)gldHS0VxGbjH{^q(LG-D&FYThY|gY9ps<8Kp1xmnBSsjLUb-+hL^#jE4^ zYc=v<_#2^y>;IuYt&YFj$zR?7k-t51?pv9)`P(@DcGf=_{$4bO^?%o&+6Szkm$gp8 zNU%n*?OeDG(uPKoxq5~eEWy0M!UV>=)KO{BJgId@V(_~s4vw|Yf;bJ zh}NK~Rnmy!?bmDs$ZY%dA~X8>GffZ22KHl}H1(z|W=>o|%*y_DwY&wPz3>-=`cHA} zVdiK=Z%=Q``BxJMy_b#nYC`;?N@OL^rp=-;<|b|Uxd8_AYkJE4J8XPgEM6Z>v$)RG zf|nB$aBtP*L336a$hhX`1$Gt=B8o1UuKR)Yu#`a3-U@92unc* zABO^EfE}8ZhNeKAPwp!VjxgEGZf6kYwq@DX3RCnt->?{X%W=GuoWo(6s^!>(GISPQ zIUe)1hQM!MX$$C)7YWsNc5Pn5xru6x*t2Rqt8C=mrF2fwTmgxG?K7avdOd3qV4eAm z3qiip?#JCf;`pEcq6f3ZdH`+5Kc4a7v0BW@@PaKz1|OKI&(2!JbvUOASNP%n0EB<6 zJHHmOfiH}M=+ZYAm!6J&O>yjBV5*QemB*!Tc)GVLT|sEyToI*Y@U`pI5rwp>Z%KHY zYE9wQfQ)u~(VWVXMaB+$iu5t46ZYDaQ#a+WM)Obd1aja zow5w&tG4Q$sAi?gs^|65q*!Fj`4>|k#ND@Md#l83v_fB%aPsm)O#UTS*tc0|cx>kC zc)f)&pq7bjUF7bjQk_9$m?^GfepQWfeiIWlDm>1Pt3b4gMG*yu*Zm zWCj0dpVbc&c60${tk*ICGi`v%h6!IVy)%lE;(B@}dt*NL*I8FP2EyG~XKGno{c1Vd zyJ+YHm=qVKW;>lN%L$2*%76HhkDy_~sdjfzxwj=Ry4OHEM+G(sn#9^7kUApiK1kmdg&Fv+Ml zHoZNY%++w)8588}&19sIV>-xe2KzVcT7{lEqi`*aWq{UnMsd z%7pJT;4W)7%$!vDKj~KCu>0Noo5=^P(gD`_psDkm-0;6F|jZ^nzA_0ZU+k zMlT?3Jwt}a%<_b3)_hiq?f6RxFp$BTuT)j49F%f-HV9{u@Yz(}u}Ek(=-E;OvRiMvlsU%cFtQ;)3%Yze=g>`0iAV8#fbtOn+q_P9)&NL=rSXA#z z6J>X*{)kD+_nj=m%a|GAg802C16J1np%)+dd<^7c5FbN$0`@2t(;Q}wwezWC*5QZn zpLr21U&Ikpaz0qC4qX4twSdtH10foA4#e~}&)$=SXO zE}_?oV8BwW`q}YzXwO&Cv`w9wCgtK_LcGQ0%sqKedc9C|ZYcW2*iaO^meKVg_FIuG ze<(z@PMdglg|ol1rAHU$vUcvTn>sEid%cYgB5>wYXX(umoS#JY;^gbCE~KZ$2!&IlX|*~wO^PlG z6)jFFI#74mGi)4eBy5aqlyR-q^w*W%@Af#Du&!H3PXohJt_9K4bUb^1JI()v3t7f%B2iCrZ+CX}=Ju;!PbW-0i|I4sznw!%@< zQd=WwHyDCn%*VRW9^oX^3via8b55ka&|*6jE{7X;kkXe~wOM)`t%B2Vt+DN- zrJU;TAU%$;SW49O&taA;ntJ-OxHQ}ifa1gGJ5hO0;r|cHKjs;=lpVfs%RjBnL6$!X z^ZNf-ekcB8_BMZVjXS!Y4Dvemba{5tJLB=^dP@%3?@N0xjy>FYwg6w~NQgN3O)Ov- z&epU?^O*;Q*SoKacVN*$t8q_Q`hoa5*1lS0O!^=ptQQg?@eoZs4aD$m zf_|HeZ_GMPv#uo)-o9(HZp3$d{a6#z`(xryG%=k+yj-F$*|p)@GW{kuPlfB?NCzb1 zUaYpTXS{Tc4+9{tB&)Wc9XdFP0LU0*lK zFFev5tHO0}3Xc5%3=Z-^C3|FmRI*oPyo^eQD;W2}CFg=a1Q=Bk?CR01Vrm5LHtVIs&5>FHr`~M^%*jD z^`q$uqOuODv(^Hw1L7L#9?0WO&QCr;&SM`T9rf2eNp?2RTR&_>LAbXKz21bzng=cXE~6~@H6a1io8w8qTQ(0!4u|`c&x=f1(G0y2PJdKgJ33&>dj3t!+ zkzSr~Sg}@3r?N~+Z$;kRD!S}>)xo&g$oy?R5eQIo6!j^2n-H~!$#^tTpVy3bgNR3M z?MMAdJ!R`yqF(2qR$#Fk^!Kj3O^AAj$@l_=)WVONp}ew-$`QlNfA5A+k`yUHW|OAaQY@hgdX$b04Is7P-gw`5%Q%wfLzB1J>DsA69Q(L zjQt5%fsjxF13o7G#yG&TCsj1J5pbsF1BdUT`~ZAX-X;XR!erb*(cJF`jHx&Sf$C|K z8=p|fy|a)r+kw1DA>Ws`2_f$@8Apo#tArfnhiqVS787#0451OfX>f5r;d0I6EKowo zC*)O`RiBVJGtVp5cr!4>kJ`*+P9^Hq4(g`s1of|pm7=~auga{asFV}^sGj;dDo&d@ zCgTeqBI9pj%E{+LirA=f#BU!{oum`-P0eVVfq?1%;rPV?-e~fjLBIkB@NQHNRG;(Z zZ9>8RJwX9yQZ{YpQ~WDj|p5VE1km`%tf%8Jcn4CFI@ z$QR|r=rXG#A;)V@i1qAi1#*zQO$d3O$+(>2xzZ2migmKS@H7$o{N~j&k?#Ztu&MI+ zG!I`+Twi<2t1_!E0XNz9Tm85g9;bn<2lAJQD&z)2zM(n6<5bjPi1h<`n~=w$CSy5; z!@VmW%M%0nt{*bh&8{ey4A;2We;FilkchbknblMB+?9YQ^3aZHlZ$8mnh%~1bj_1+D#;0u*>{{%{Q3? zMAchWiFy-^0BX>=@-`vrwy}yjmGU{;j~Y{hR{0@|P0qL8Mb2F^v?AnF*9c@cd7BWj zw#nFvke_Hqn^9M?YvWY;#KT(Db_5*b0Jc!T>*Q@5Ah&fNZ8FY>a4Ncbnuv?`gAa|F zUVKQ=R=k5uH~~l2tt_`b&vt_&ucDzdC@Q`^Ce5}B5yYqY7vstv=@?gzUa%KmarK%L z)YJG0#@l_Zj;ob3 zY)2s{Bczziksx~x1|B`RMM|y=&b0EdHh_DgGPo1NoL`$+JU4`o81{aOrjpblHT0Vv zt%?I!Ie7ISUyCTo6M~PoGF0&$UmJgBE7p<+l7^=ssdY9C0K<)g7&5ibuJ7vvlkJE@ z%$c>j^~~Cdri=dfEfFj)D==A6b2Y1dAS{Cb8WT<}#>{;+XV%s*gkJ{X@@w4#YZ)fD zDWmlE@e&p#%Q>Xf;<2C|8R(o>8-H}KeO@iz$j%oN7ocMjNxR&*0H+ZboBpD#AMx?I zdKwRGNCEzdR(I}PtyVlD$UR4tdHGNC+h@0~GiSFB&={sD;SJDmD`q@iZ!7?@EAa@V z^@jnS^d{c85B14T-Q7poF3;6^6Gcw=uUyPz@qE&`a=y&JVBcy0-ZkI|eaiWya>v1& z>7yWg`v{PCnex{+weIBmljDFVw4kv_jq^-15J}Gsud*+pNGZTU;b|8dF+ZqK<)uhl zbf89Rah@1VZ-)Aq*^Q(vHSW&kCIgAX?yyx2Qtv?;oZgsVjt$|E{}z+)ni;wrg~rNc z;v5m%F}B(>PGIIMa*%+VLS<4%jbD;hjRVI0IYLRXxnS-Ht(#s2N!6h%wdV9Hw!t7zk z{DhwW1^HR~-q6eo!4ejviAqsGSEZK%<`N_6cT_y(G@x9#R@ed+z>~VZMBA1cVU9L> zxx?rMcAJdT$U-j?J=G(HsLTSBY20*GW|p~9Xqk07wg`EyXd(%2*q%t~*@w4vM8D%K zosn})IK6_I)wmV*vi^uY0oHhKN9&OFcf8S{Go`UYrJ?JH(3#)@EL}JR`B7|Kd{UDM z7owP?-xQ+BPU-Yi2pnmKUOE6 zQ{VQ8$5~lq__NPPsrbF0;L8Cq7O0g^^HAL%%Uap#`LV1G3tPI;Onyb9e^} zbxjSP2#rj`*u$#j^3#jvS=J9h-CPJPjUM)m6|;!nSw2`6hGiaK8M-Us!lx~O>7-B^rhP5h?k>0su-L|9- zZW}ry9fk+Xrs9uBCLO=^VF{f5iDgzjn2(AxVycjly2t)_a+yPq(4M*V0m6dB;GVfs z)UHULqK~W0^+=g{{k5l&^E9Xu9%#r$^%h20R(D9*UdA^N)H10PZGye1$w5>g*V6W5 z>PsOJrY;{2rOJl*=eWlhSXxaVv^U#0+dKv1CxBAot(2+V!y_u zj}krFCJbw%J=L6f*O_6((kZ538;}8~lTq9G(X{Vdz}MH?)y}52(FVMSWE*{-ah7`0 zQ(iSho&*t9xZJSDUM*lP`jTEHC#{E)mUbJETH-Bt@49|)@=7s>4f zKCPu|l!iA15E@{I<_F2|?0FoLrsolwO@yXgvzE^)fO>$%C#(V_p*_*sO?PVDX*w;ScCe-BNSTa_1}{_?{`4rCgpcd zW>r~B$2Y)?EK5$LGaMJJ33S)LfZT^>0DYw)R|Imo;gvYb4$~UA0TIeiaQ`8#Tt1jA zte3SLYthPo0ONMy?aIM&YZWVm7lgAXcz2B-9>bbIG_4*Ai;~nbB`JGZ8QTe(!ptux z2{YzEP%No1>&$$#XlQ_d^%9DY`WB#TrVvv>0uXXf4l46N1M!#?dqqz1))6Pb8diSa zBmKDmHcfN&(`svJ1T_!}!7XF3dUo4!faU5CE*v~eirbO61$;7w>xI2BZ-yU*GvYEr0i>{}FB8 zm)q26|{ytQRWQB7tubOqglqZUOf~? z+U-3jqTOP7qBj+$e%XE7sjwTM3sb?QB!dt75`#fIFMJro5FCcW7-0bCF@^M1MwT7NZKv?1Svd0 zQj2ep2#*5k8B6g%yN(yDX{*O4coblKv5Ni0=T+?gLDuM9CseQ2NmUcxHAV^N7{X${ z`XeBkc9(=Y)tRJD zm4&0rLvmehmG{-6p8a;5OV0?y&_lQKYtq>a!d6FdtT##2XBi}IHBxCH8#V?q>ON`IIQKYrc5qTSpnInNk*^^@FrO8vRA)V-d}*8+Y%FhL z*pPPH1q3uOB^Y(zPyT;BY%t^oa2vE$GjzCkyXDu zT+|-AafunL5y$zpAW7?pr&Ui2RN)Z2Hj+^P?V)*p)(YWtDY4~8fKjQ18=1YOs85D- zj#F!fbA_g5eE5_QuDBE3LGkVpEyCZWHX8*HRI3E_uOc}MtDHBp_rdbxrT3x7MMaDH z#BY$yKwUhfQH>yebDcpen0D41*+PaZ_vk6|l+^%l9JUW5hwNkeciGRFGRoV zxdc9qdHf&AyAMVw-UWIcG#r)r^2gvWv7RsVg5MDf?m6F#dLF_lrFTV+a=-eq&}1$( znXUo{$3xI9J43v=NXmf^P2NAyjkWJdJxP6_nk*)UN}zc9x`R~vxNcdk3mb)A&c3(c zWKg)zX!mJ=Nq|DUeJ>^04EqK?jY&J_W}ER4$Al?{%!DVj_GO{}rE`>TtY3G>`aPBA zSU-3}nK34irMA5rkR3W&jZYgx_G`|x;Xp}7ui*{AMl_aqL4pb>#!d@v+f~~M0&X^O z&v%WU7gvnao@2x~ts12T`1o<A*M zkckSCWk7lukWDf@Pwtw4y9o)~7uU0X3@K#o4k3bogG&$Og?}$UU%j9ZZq9U+W@e#G zchi@0L5H246X{md77UG~SZZ@ag~w8J_}V&}W*tpAU^!wm?MHr$q+K;qC3Bro&Dkil zNajPlcMMBv@hBA1#^f~MTzr?nk)vlphUsxcXGDbGnSoqN1{u6E?8oAoa&>phm80c) zXoQxl&P3JN({P71oPfdiuOUY{^1cHf-63Q@zKEm^8LnIg9#byg6W;*6@&$M= z(pN`!s=tWZL&NS?QbfM@714C*z9JgEP+EK~-Kfo#QA_EOPVwKyN4NNs`2ypQVOsp_ zCus5YcJ?aUa~}Eu#3NubAns|vhQcP}560j$G#Pn}Dp(DN)&=`VBkt$aCdUnk)dv!} zyqe0LWD1TdzpEh&9)*>3RV;K}@ky7p9O>v-*y%(r=Inj%IPEp0v&a3{z+}_2P)kPV zBA?DhcRClnTr+G;zpuB6>+hCW6<2xtoBe2zzRoVbf2vs*^>!`HmX6oUkmHcX>!13^ z>*4D2Bi_?fCUpZ6@4>nS`%Za{_hg_xTUn1vZSV#``d&95UX;Xn=tOt7wp~kC4{h$d zu=O!2Js2s&uE?%dUjO}hsqbK}#J&%w|B~FcD~Gdl3Y*$CQ|Xt>_73%hC5nX62xy0F zhgke6<~!Nsn-kGmx(0Bm??6|zw(_1KaibO=!I~^Cu=>!xD{jS7;he1pw7L&AgvNuA z1dww3f5%t zzNv#F4$e=thUl#R%C1YSU+~&%RiAbc&T7`z=yNr=w-uUGys1NK*NwUT_l06q8Zs(< z#z=Idk!W+Z)+R8>{)+)(Ol;YGKDubCG}Gj>!m83u&;)EselYc&1)=^+RHuK}r=-L$ zrZUVlhp%4;sn_mTrM_TDJxisM7Bq@51l`_WFrTjXh@Y+TKOyXT1R?L32xYc{YR)UV zb98>1bk8e7Ex#Ke|Ejvr1;ND}B`*5)kG3$`#a5RsN;aFn?a3}tiC@3i|01f*mOUG$}4^{n6f`o5^j*XlbRgf47D@_V;DAT zT&3cP{!m2=ja+jcQ56Nyfzt$|iJI0*d*7O)aLmD*7;avqmE7|qA)>6qsXwPpY9|E| zc$!iq?Gyv^Q;EXp-Lx`5M%i@VzmyE#Vg^0D)}D6hBPL$Y;(DkJn3o-@m13-++Ee;v zL}OmUd(Zp8Ju1Cz&iQXcok3ypja(0riI7=B6`>^hzzz|o!4?h{MyYRy*nE_qN z-W{dkGQ3gZEQ4eVAt`2HzK2bFy@5s?_DXA?TgbV_VjS!*4`+t&dPxnkjmjYAj7*oqg7{Rxmg*o3`p2W}hzB zVq)_@DiP1#Ij@ASXcbpV^n`ijXi>`VhpnRgycNrTAI1;&qWpUlUu6JkFi^sy;MCeO zVkAyfT=_>LGh6wZF%?H*DMJH@cq$_}aqHQ|=R$Bw&R zk@2x^qv|FIwzL5)C)nciU%?2o4LRQH^424?RBAKD=mYfSXXfWm@Md0)hVRQ98xK08 zd7^j_?S)gss`1LQym;5f;>DHk@ROLnXm7Nz9xcZiu^D`53C06`TG;b(N}vLZPJJBu z3B=(a=jNCBycx)Uu;T-xpNvzyEZ? zuV&zBuH820Gq(O!GXsqJ`sDh}LQ^k1gCWDNUkWTrjpG)nD} z=X}{s&!LPKblSq>+rWP69HM=I#Q|f88`(aFFNJ!-=r6?HI>n0PPv_6wGk#%mxmv%* zB(Inr2b52;O2#K&+M5}ZnagRhstay}(VYq~O<&_7|47zq=dySv=Cw z4=lBP5DEJW)ChH}i}ofDXMbKB>8X+eRf#d~o-E=t!}*{h%&;p)tIrNn5_gYKtqur9 zy4J^{iiUZSt|_|hfM+k{MLO0F!;$>X8bXm(tLvjor&X1%?as(4uk8<*`kU`PSVb1{ zbwFXQsVFHq9m44RC^(C#zxmRi#>r>XUCLZfh%{jqiJ*@q4M#)>%X>zqtoZjDDqV+T zX+LiY4TZ6lATK?eE@ErS2(`9jN+{A3AMg!cO2>u}7CwT-j#w+rlSRTE1NGB@NfvB%R)+Uy_B`O2T zf})mL&45YQ#E_q|5BU$#evf_C+g@+2vH)@T z>!{9CJ}5Qjf!x)1-1Dsv~s zx8CsLZ$ZmE1pdQs`taYZGIwJ9kKgj*-&!SwIR0ng&_36BIWsmhO{^Z3@7^amb`Gn& z48c%kIT$kCZCwd_U?&%r-`STouE6jR_y7Kj-D<__3pq%=`!<@emDX27)P!Yn5aXdD zqpQjsY|?I6#rSF%h+9_=b^04vE{;Ou{eeW|t5hjjnRNk5hVfYif~*`|h2E`ZG_nU~ zw35M9jJ&0PomS#KGITaGSH>=u{4q;;a&fIBF(HZQjeW>V@75~mRVTl3w?cl6lz*Dk zLz3Uw$V|Jr=U$cH1&lj)x1$A+9)~Q?X2?u=PA&A+$7#!}sgE~5P^lDS+B5vc zm51DzL%ZF^{K-IV%(YQO@^?s$87hs_&zw`e`g!60>hw8dy;nbX#HMF_*(`7|XDD$I zJhvabf|6Tjs$^*&uysP#GDwsRog1p>U1UuAaD8m;(d9&TC-r7UagrJI3*#ayA7q1d$e{8Kx{U|*z9s^ zH-uuhGsZkIFJLy-Vez+TNYjOrG5#S?fuVHllWN${n+Xm@Y|hoMMs5)q_w}kdsT#XUsQI z(3K4AV;XL*h*{$ICnqFW&k#Hb#i;10GNwQJJw=>qE7-j~*__9iFT#WNS0doe=wb1C zI+ZJ}FYnVM8BtL@rBmtAUp4hL1k;JLD@~;>o;#9t_HC5Kx<>Ww)>$x;9pc2L=)^+& ze+L?Dl8jP-ipZ>m>+a zv0ED@@XAeEvHvwFPoZMxVK)z_o>#F_zV`zqrqku9cC44ImJ>OcMy)(ni2d;wi(Y0; zFucDdc!5%x^p<^F9iVWkCf$QzDDqf{8 zGm&=Vrw(Y38ehpH*@Lxufimg^939^E||dLk1U#S*Xj(njhlK9qQ|iZhhTI zA%IVW+GlJ2lTH2y?vnhYMfy0V86zc;ewLVoMs|}X#qe6Q8V*$xBP-ClleADrPk~v? znp<%R8icxZwl~Ll{6eg4LE1|d_ksS}N^1#jMSXd9iqM%)L8F?>G(ca862neHPTUCge_Iali2k`@l`hT*|j~fCiRS*?Vby zqjByYXPo27e2N3RScc-;VVtj($;odaqf^v4JLy@|A7clMnvN(#U5zrx=5d%Ilu9Rn zR$6D$$fGUPwXL`NnzMkg8eq-GtfwyXBkc$LLBxA#j{W<|VMjJ1)Ii^W5-&yBtv69{ znp^HuHS;F`k;yI$p<*Ii{e;^x>*?FHM&*HFcKFMz4_CtrVhGEvJJR&ow|nYNS6D4I z!Le`SW0j6+?CWi>2g%@f8^sNMA z5(b7;Wo6d=FqjDfL;6~#Yn||G0+u}MV>w}m8#~uwpo{IXBWbw5BMap!ZM2I}q7xvv zGz-%~df>>eF zi(O#W4{iZUD80X|SJ_eC#PN|{haC!|`;fg( zVB8grtp#>WN{>bg0!AZNkC`i~Qj1U310ROtfHVr|rhy_Lu0xfY?-x(SlTIs+8;{I@ z*W$)U8c4xppd)s2es&Lg1wHm`IVyKLh{^2u%^xHCL&LRKgrf68ZI_pHcy!+|8~~*8SAS+KNIXpX{j&rY zPOt277q=7)P*Pk*&oHWIzH^C{nv!U;b2?0-Gui1$FR_y|{t=6SxD)(YwVfpIL}+8v zAKy3;L`N@#qkE72u5|vs=e1Dko&l=VLyeE@>8n<5%}r{*&y_o+G{|(%E$)zwR-0#b zXqRPnC`^PIswBwd;lS8uQ+~|H{N;A@?6jAm2-XECKkKmgW(PBQPZK>@3cu27io#dq zFre<<+{V~H1L`5?xcXMtR5CoWhe7j=wM5(Y7sSI1K1%!I#>mVZwy*8zKWUuVzS{Hw zD|tp0h|+UYA=($31e^%Y2*CN^^1#1(BV7lSn!h?XxM^aw&oaPKk3%O)WH`v7Qv`Wg z)mq#r8Y^Fl<|U&Gj$X9k1=}tt2;eu{2s#7)fU&(ik$}!X26i{nuYsFdU)GOB({^v7 zp#@rWM23tcSLL;w>sx=98?N6lRP+V*+;M%=WqS+%gxw8tqst0wGCF<7RkR19-knHT z!Z}G00BWxW?u@R2Lo$}f_Gow%g%MvB_UuT?abN$c>=|0TJ=Y{6q&<888WrI$*azwN z$kiM=_KY?IVuD3fd$!k=(~4ic_AFzqZO^i}7FtcuOeFDY!MoOt8tkElxI(ryVy!DM*{w zm)NxNUV2ls1im)oYOeM~OUvhCxq+xzSc>U-kl6B2fcZXmQsLhaj+MhrL+4OZnOW=k z8NNdOpO~^vG;RIQ&}(#lsO^Tq<Zv_s@Lx1E1u53ui=2ny9yH|4p$m0fUP&J`NAx+&#Pq{O z$!38v_<@K2b-{I^HkYRqKoYnSlxq*t>Pfk`huS`naxWBgG}5BwxI@X(_G?O}}hx5afl;eT->PbhK=MubbF{N6r=KSY;? zYL$oDT7#R&{+7rwC}5ejlbf8GqOEmLg{+weG2kLAj#b<7cGlOrUNd!8{bPBVPhO(~ zxXlOtgutOBxd>iLVSLR;#;f$O=6E?q$if_w>t~V zkuYr?fD_Yyuk}_L{9la2k_wt>L%^$ytff{9Y@9%JDC=u{E`?D?xD#*SQlm9kAHP>C zN!~=?uLU=1nYHGtI{eF}_vq>Jt*aw-lqzjEe(6;JZAvp!>Nk^ODw_~0DJjkZA+*?) zSBfd*rm|HiI4__Yg}+uO2rnjpu`ugvUBS=b z9&lS*5!{@fwVMrB^8G0LkI+R#rNg-_(r!;^-fy*yuLy5?0!opirLex*AC2bVtN|R| zIQ~o&qf2;mUgV;a@fJOhz5aUm27A(ukiz#3tLLt+fwd%231;XBdX$3>Uxk}A7THa^Tu{aJuqf& z;a@@C7qtM#Z~Tb^_>wS&Fe`{5I<&li*m~NvkS(}$>QMx@r3VBz!7HyN_(kRQ1wMJ? zwfKksYk9pziia+*j*<=XD&&X%Kjno=^Whch2wa03)1!;>YV8l}8#z8tMDPbqCA1w5S?VV19zr+h!GLHRCX`QDejNt!oD@)jOz zO_E>VwnR8*`zfwPUZwa`SC`{QNU;K2qpf_&J6rSq{kL%3th9-gW)f0C54dIt{qlg+ zJJA)V_Wru!T&tyID$z`zX{Pf{rkgcW2wWgr22N2%e?A~Lb;K%gb(NH%ESh#03veq6 zP!vrmY?Bwk#Y3!-rAhb+;|kP}#q$`XNF93*e~COH?=jr4`t6SLxZ3XJ{Uvv;!hT3x z548`~DG&;uFE#ICW>SYNd_ZnR3`K@UE9xQXHYAmKJzU?{NN9Jf5lwsVJNgC~(qTxW zmEtb^QoaN}L{YSkhWU{Bi>ENrkNt^VD9YvGz2wkZzAK7}9=Z}3f)0&zi=uU8$_|mo zZ@{{Wp12RS42agQ0rCxtChK2}+#SNqlWmRPCk$n_0fFH-IcPXrtayv5A0?jvqyZ(J zqN%fzkHN>Wooc`r%C`R9j@E|s(Yde;8H64-9_6aYlX6*YQ6iDiB~t!+sl|Uop-8aE zBxuS6n18K{bplI}J=0pd8x&>vUWT=A3Au#GkqN)S1^jnuUAlU6pd)k^OTg#x9{&a| z)}&=jEK#nN!?y82rq*y~3a*3SI$#1LG?|bC?A=Oi`Ghl(6aaXe4cOXXX8pord@J@Z*Z2hou(3E^Jt@ z*CDf_vuvvzh5O>35C0+u=u#R6}n4d$=9Kq zYuh(HV>qILK84+mVwht0$43?$W}9m#Gcf?G1s0GkM7YdhQxR0nMCnCUPtbUM{TpqA zwMkm?O;mCj+Q1l|!ET+8BBO`}sS`rcC8OHaNx}p!CK7jm_g^nl-rN6ec<%&a5KiE8 zDh$I*9JzA!9XRj|?q)vHzk~F0_F{Ogb#{;1Fc4V4NiE%umk*IxME7b()AGMz2cva? zjBnfl2(J7Uq7w(zk%q7e)%z)eue4H|Ld-Z{xGVm!nM&F3hfy;AHrBV+FGjnA)o(}I z*!WZwtx&~&CV4pYMeZtk%t-ooz#hZTX zRWa$Ecr1Nx_Pgj)Ly#c)gMZzHwzHTGFJxHl>(VhljKMf|0B*-%mPd2*Jkw^7j5YXj z&QT*rhP#*qGC&vuqbPfym_*gNNA%l}$l2=KU8CwDPgcY;7|xSX5<@KkI|5}%tfMS6 zZ);7SZ1W6cXIA83WuU!PR}HTTfdHwuK{~3NOvJwcUXhri&+c|aX{``i%O(J=f>3A7 zO2pFH(Me1EhR~^?>x-y7Dlew|u<_cQhl6o^(Y)HBqAfLoB^^@s6>cLvF58)JgXu(! zvk%z7n21g;7XTq@$IwlHgK?^W0=3<$gz~Qo7Ofppu0-+&JzM?~1*H5xG2Hyn)kFak z$sh1O`R@fYRLs0>HF?DLw1o0^;;j;)ilkj?kS2B!`Tvzb{+26$IY?OKA4})^jw=7t zQDQ8}m<~D+Y*Dm!h!b*4hGFs*dRP_dv~QkOGddStVh30Uq~h?|8Lx(GLO+b)Hq0%u zYLo=G2g4oe?JNoI3}3nz+f~Dt?kfrID_qwBTg{sUTmAzxEjggkFKTO!eT7@&beaLb z5<$9{kh52(6s77E+VJQ!p*~*%tJX=vXCi$jYEM0lbm~LOQg0IJlNBpbd-`fCo704Z z>Wpm$>vL2M^E-61q~6@9cho$NNJF;rbr>PKu)P}-In z1K6StB#?eXzw{5OMJ1H~!zi2eiWKZ%<=@V*GY(=oWcknAT8j!`*HFDZ`=Wy?;L^)P z0eT*#UjZ8rMghGIo$rNC#Wkt`tXJ^aGX^Hj@q*aJor3v}xXK^+e-J;$hY9+V`%pCN zB&lGDXz z(>qt_aNViTp7l5=dlnM&6o7dO&^(1of*Y~rz8=D$LG+fFBHj{2PkkxElfk`(U(>7| zr)EtvB5?i%qHB^72iwN!Hq)P`St}f>R`-u9gGl8TKcXFJA2nrdF&G9Cl}#oSj3m$U0@)y z3O;6T%hkCpzt#Ewy11epy-|6Mk`5Q{D}-+8Uf4K3-$KaKk#f|vsJs)GVc2t{P6CI+ z6N6fWMZ#hE$w+NG9tW>auwmUKJjBREcGFj`&m#?Zu7D@h=h33iyQojS=(jR`YF}2J zK8?Wlb~sdhY7bY9KAW5BGBn50=gD#UO#O%oMgLY_V*oUIz_zcpMtyyoqcD#~bE?+p z?S{t>RHMyfH7fH{PW!}SDO_!k+kdT0(ruOY`Nl=i=aa$%^~vQ4G3|48;`aHTK(Dlh zig+t-m&L1G=wsUF`qZ%(cYb;wm()+*&`_E~a>`h;!7{6UoqOl#KB( zH@0J~#rU_qM~|jSdWx%ycjeSu6v5+j!bt&T9^V&BdBw#_d6rPFn1Z-P(Sh3ZHrquy zs-GEN)ge;5SX`B69WYX^rT*o7_?TPhUxwR@ciCRN%l6`3uJ6!x(eTE>mfI1WVsbR+ z`7&gn>j~#l7dv9Wpj@hRX%>??Wpt{*!t#1j1)N2cu}rLk0x6#NL=WXZBk1ck>gw(D z{gAHpqzK^9+H-ReUYRxYQmOxm?6bD#ZPmADb&aZN&(qqf^~wl(>cK(TXLXpWY0u|~ z9LucfLSF21e0ydkFItRlm1o6wxX|&J?Dp()mi~wKJWT3(nYBrTTK+$@=cbU7wNQ9C zwDvsDG_&d1qR&L_xx2LIzp2lhgzZ^ftZLeG=t8Zq`ImU~SY3Pe_*|8@=bxm8th9a? z%HH7`ddNM zyry3)`oD)fN}$H|w`z<2&!+zWfl0)=gUB&r8{vhhQIE9y1ZVV(5(n9?25`5`rDpV2 z%mT+gOP-bxl7RHHxj zQXKs?H(Zyu6aC$lK!0?SBlSgpJknnDhrYwn-LPtPM}Gwpe$88Zbb_*0pl@y;MK-odgvbx|Ds)i9i&CNix^)czW}`b);gxc1j8_h8x| z9S-PlwBd7I8(W7s3H?UKwSA?Q_BY5DZq@D2P_Anz_q$MPqR}Bb)72CHPHpIGfq$nS z*&#iESag7G@n_Uo8SZFfr~j9*EfR;CtZcYr%F*ot)#TG6(ee|icXzy+jrrm)hFaBz6nj$a6L@J;lpwpLz?E5m4(8RK{$rjH1@S z!Ku~2tU6!CIaXjQo>sGZ7S!s-UonSE{i0W-leDGA4(d09b6Ob^qS$bd8YUz)e0C*m z)fB0LDbj}`>GC?hwn{J9qi?j=R(1$VZR!k)7xOF8uHYHiwN=x#U2)I1er(^23Ar}_u z`G07dvpHd?i}2ME!PtmPdEn)6b$aJ>q2J$IQFd$*tyv4QWz0JtyqMLyxN(?zGW+%V-&DqZT#ww1V_NY6cgN+U4Dg zF4;z1&+?w{lsA-`f#jlDlvnhcjs0tOc@gz;$~&4le@K`URu@b2(I|VN(7_Tg3Re$A zs47Y&6bdi)0J+0Sd%vAm-k3X23Z{Ov7BK~z0k6RCr`QU!5{a!Efp=%l;~WOOsv2;x z5_ja?opOe#l56gAOn~Sh1;roWmp_+aQ2q^3U;bkuW}VC8SL7Lq2NFn}X*V$Kt-@+W z0G3ZK6M^Mpmli8z%wibh{LfU=UMX=_#qQqq+rIiu6L#rNr~EKsHDx30t>c@Wxb7qf4Cb|;7YHex|rhXaYk~vpSb{Ab#yt^5u zqQ=|^(?JF@OAkkxb-5zf)LFlOS39Y+K7Pa6h41;f0Y7iIR^W5w_{>@xuo!pzZ(Kx- z$X08t4QY_|HDXDT%Nkgn*VIlbS{}tRm{THM>dP8v>jM0SFV>^T9!{J0c{8qF>ae;b zZ5BUbdGJV(G8F1that6qTm6upo9*)Py5!6nbDQD02U$wa+$zss<+)CtE9JRRo^#~+ zo;;_@^96Z6F3%Eqj+AGiJa3cd_42%0o?YbGUY>2_dA2-T$n!XP9x2bd^4$L!OZB%r z%jLO2o@w*=`{DAeARXUcQBJYSOMlkyD9bF@6~ljj}s>@Ux2o8|eVJin3WQh9zZ&kyA}15Zve+V%n_z&+$B zW4^S0>!=^lENLfjKcwA253H|^jMC4*ig2DdusFHJ4j32w(*y9a++)7ZU8?)N0Q6WfM0}jb?}Yjm)7Cr z!%^;F=~pngTr(C zq7}COSONJRdrb7j8ot0rrIl7)ebEpv^bVL{={meb(azMOb=9KPb&EF0E((g3q(#XH zCoh_93YCUJVc$(w)M|Txymp36YFTINXNabKiggmdL>jb57Ou=HY2VPgq?~CNSHO9J zGsL+ID6JP2z)^QLExfxPDtwr=f#r@g$goM&vi4hbWeKITqm|}Au|9T`iawx@X@3z` zmJ{WI=Qd}GI(D=absUXvMjgX|6D&Q6##+>og73CEmVuh6ZkfIX{b1>Ar%c^FWm@~O;rUdQ zDJN?a7ef^-2ytb(+#8$Eb}$fyyZ{N=`wrsTvTUod$XlX`U4Xq5ppeIxr;EH_Wf4>! z^#%Jg2m2Y2vE|X%#-8z#V3$YXw!)jbeSCmL35IuuN^-EH@4Psbl)E(S$oUv}WnA(xApLG8b-NyQEWXylZSdr|%j|jwkUhx(H$C z3=FU{!iz#}li!G-zz_D8F;wzv69X_BI5@O8MaC~_lkh{R#JkvdrVtwZT{52Z|-(nwPJPO3ukt4gV-QaaP6G(yVsl|$(|52fEf zG~As>N^<#OB>5VpR8J}W%0J3oE1|T)p>&Lg(j=Qw#Uxna95jPpkP$+ff2s&+l1pYQ zvIR@a95Tz1)fUo~Hkm0PgDq?Ysc3FXKA=rlZEgRxE-KH0@}oU7h}pQhQx0d!hXuGK z96;BR7yj8c1D|Ob?$Rw}p2;#+r_|xh1tv#IFT?u*EZ2Qo9+l9dAw07svC(iL?R>VFl)^ z6_`>&E6^2IpnL(TMv`ASr72W!wG3&byVyEYFK6f>ZIwzG*=-_ zZ%K27o8~nq%>_(zZm_f)URE3tEY0KJI>Az2q=%f@$Z4wfHL7!9U!u9u`7eFH87uI) zEu;dk#~-Ps*dO$q)C=A56gJvs&(6((*)rnv))`N9!qeYdC$WY{8dSU|9iDac&oMo{ zW|RJ&@tEwQiyp=%73k&P@Vdea7bo9~Z_sO!^(;CAob+=lIO^zfv;n0LmM(;S)EZt3 z-^HUUy66JZDLiFya!Z>8wiA%V(TkHC^PMVa9XPId4~Mj#0?L~n4xl~GEu_Q;;sqKo z4Y3-Ak)T~M09$9fKA)5{TgIMc)+=k_Ya&NFa@;F9zIJom?dYfs{8zNXxDz8`1Z=I_ z(Ep(**CJOya;`3Rstr7DPJlI=fINhF!S*#V`Gpq7^t@4F7RR%=- z25lAUrvQZXTtI|W1X-cxe1PKtp{?>>1$2Mr0_hiQehEgpXGbnuW*uvA&IQitg0sQJ z*`+jr)8D~qU~np)Qewr}n54vTAM6)e=<`7AN5mk5rJG!0(;Q;QI>hF_tqLlqn2~Ah zU~B`ITX(=l7!*P!$(vx`@Mk&ypdA7Z-`41lbM?nR8fkUN3`Q;=8N&9JZ0vo=ND=H$ z(l^TA6NZfmB;n@(e_otC7&%dA@acAZVrvhToKS*~(X{Yn?DE1Uf^4XtF+eDGCnHsz zxgA8qgVf3b{DxAV*-V6eY;!FFob%7YRIqd#NNSDjgzt8ZT=|x%ktf2RqgG}Q7Oi0Q zSTGVg+&%~Y^`AQmh=!C_$xzS}I+ClDJ| z9|N)HZVwCftp$t8!j?CM1)|J3TTp5%%4&QwC>II~w2CUI6Fn%eBfp^RdR!TK3@9k; zRmiT46LH<|qHGY9vmBK9$YYDEvyJlHV=Auml@X4wm65-;xr{^kAbl=lHp-wfkF&d2#M36fx%v-3M1z`5Kns`n%NL{8i;y+M%JMFC`mpN zsSG14bco(aQ7m_e-bGMaJ1FOPP)es6aitiPDR&4X|6nAkjP(1{)yRJe@j(-jRj~`v z5hR19Z5@aeAZyF&DjVXpM^#oqKzORw5yD6wQW-{)6{V}9G;>kL2+Bnc%0Lgw_Nj)E zOAN~T+l7(682Bk8Pi=J>IaeVn8Ve(zxWne*AQ>#pb|8{H5D^>Vhw;kD5I}es$w41n zlKe1I8AjUa;JjE-u5wZ42+E}n%1g*&i|bSyv`q z0AzxtISxcS55$7kjjWC~5bxaPV`SJoGV%pd8Ak5W(eo$l>~UV0i?TydIyorcA&_4zMx$0pmgw{ zESqA)b(}%@=oVpQ97b5m$lxt5BS$DBi_=6_&$tlRfMl?=rvvdb$l8qDVnci!R$27~ zgr|)(7e?+uDx;Cc%EpUhXoIqYwx5 zZuph%aB>331WT`RAR2ohCckQAwe1mQBn$`-BfUPOM&3p$!$>#Kg+`uHlv`Ys6@t>o zL3tl}Y;j#^qm&qw7Rty^9P)|NxY2SM8Kw~DDMTF?Vll`BORsYvx_KbhykZzR#Xx*< zvyYKSKP4mONM#s#2touSD~}evPsTS>wUULAd2XlRaQ3x!qY}t2_w%Tm0_f%GE$%@ zIWEc@f&ypIpp5sR9BZT8WKe1-Bllw5r;L2^o6ATqg*Z|neqqo;4dbgI6D-A;%s{mC zK+Jr}$f~A+c=1LbBe!9FQ%Uk_q%w@$0J@OnjD{kvVJ^zgf`akBL0O1Awz#_3D3cyi zakT~tsJwuor84sOCYO;%6(Xb%OMhLJV~Vl~>5M= zh$DrOVGhKz9*E{P#2^Dv-_OXpkI6`9q%w@Gf)G)JCW^8}M(g5ZcN3Hm4oa2>WzGvm zT=fjf>o*7^6&NrpBiH}xYGl7c%sN73^{@*O0?APw5-0DHuG089zG$=n_FN_@JF!IbVE+efJVrv6o;^9P-=`2oEEjKB7kEA(dg|V$g*~9#)iIF3P8Z0;^%19fG{8b|>e|8zUO(D`1Vi&_4%E&B`36{b+Fc24eAQnAuWR+$h z-pltfGGZ1P*@skykwVY~BcIh5aXspy>=YEZo(5$t^4N^z+bHimsN(7j6j14fVYxC= z$HT}g3eiU)&Tt{Nf=sX!V>1I$=z-Y(oM9x-Kv?~JjC}YZ8My#C3?pwsh$zCY!-bJW z45p~KS_&f=`5Tl}56aUv%I~P))cb=#0hRkOELTP@+UROzokC1hi2g1_x-i12R6ruE z`6kHPvT9{R6d8zSen$TIfQ$@AD#OUH5F(0jhNA3eq(d3GK~NBKHYgoEC@Y>d;yS^g z&|w0V@fa{GBLy2=Mj9$3OY4cOo^v7kfMl=~vn2-NSCF+Cxy^={JxXPD6(BrqNh~J4d?MrQovGSW#Qk`&@A22)g4 z6G0|e`jP|D)C2L#Ge%Z_jZ{V=fbcNVdnOMhT3eF*8xn$-P`JTnWJiE21ON#2vBMmk zt4IsWE<0eA2aX6z>xvY?(wBiNVhdOigK?U&@e{_f%EqQ2T{cE3-8Q6)_1G>OWgs6c zo#L?31K(|Ztbf|Dak_z6)Yr$x-QkjZ14|M@WsHZ6F8A?$`vIS*^gOct+dGjeFuK$KBBToYbRMH(ruK2;#$e#)^ zwXVo&hzrpMB!i`II}r0g)|OSS4e^YDXys?5hA{FxQW-|}qBfuiK}9*rMY&H<-gQv= zcu;s?+>QeM{A5nkS8G)2XA2dH9cjf1ialx<#y z+bGN6!%*)7f#OlhdGAsw-yoIY7EJ)jahA1I*mI7YM%FMqCcd3i)p+AGR2F3M_93YIQ# zP_Ff${PwuvrIkTh)yv1r3vZK`_Q+v)c>>h}rTDS7@bWfeDk`y~g_lJR${tX*c^PS= zthir!xfLj&@)3rc4lgHpc==3G?o*VIi*lIoveZF&%!AUzMj2>OYWsQl>Mio}08$xV zmY`ao6h|t`FAS?FFP#LX%t2}CL76$xNUWwodGTuD*{5PqRglz5*y~C zTm-7Y(iINMLQu9P*2PAdgkc5SOKYHb>R7Vy!m=1%_MuvUmmEbo)#=Npu-5kmr%%@`3nygahT<>eIRWkXHjRAP67YOwSh2jw47ws{$CqpTXD zyxa~H4=-)rq+TkJ%J7n@ybM*8%UqOq1ZA~@^0)`(I2+|wgHlI%`5S{mhnLU3b9w2l zC`T*GCPqG#m)Af^C$jM!?$?oqvax!ulUTy?kNc625BC!!J z%CCa*gTu>WP_`x3%|>|ZZ{|?eqL5jCohj9mEmO>syIrK zrYOHN1fslj7L?x|l(Ri3AB{H>t7A}J=^?yy!T{9arT4e4UUnM!Zlg>tREf0%ibpSp2`^Y|JklnMu>zXzovVtBdGpnP|Q z@X{XxM2DC0-?+S-t|-4A5MDlDtVJbO2&%!-KOL0)pltI}Y@>XOI1g*`AfR}7xo8^o zQp@F~73e}Q!xg2gi!xJCwmT?Kdr+F$D0do^ddka4j1e7PzWCbZ+2x>=fwC>J z9yZD=_o&1!1`4P=j{%&+%kHmSUM4C^XGJ;QMfnkwf~9*Ml$$&#Tf>Hzb_Qi_cONgW zy-r^4K@P*qBvf&f;{^e@Xn*_POq zHp-N{Rbtsd@zk*f!ppBnWq7HrC|wlgL>Hw=NO>s$iielr6zb(TmzUO{3yF?a zlq+16j|Al~2jzKCwk3A5jWXDv9HG3_^7681h0DtgigKKy{K2@0^70la1xt@`P|ow9 zd^*-htiC~+*2Tw5;j84O2XYv_+zz_n<->nOVh_0}6@tQp5>ziMLD}ZzY8&Nsg!gGJ zF9iyyG{cC_(aXQfU0$A2lx~U=a8Z5+rC=$KH&T>aJScxXY*Q1x|uaK88 zau{AFqq(9K<$Hyf*^G{;#F_~&X%0#)4@%fZ`TkDj!SPwNmi(lK^f;kxsau}Bq@|rfkIqXyi8nPM-9bg8Ja7!IQu7a*+_{Ju~;4E(!`*Y z^q@SZQBLB1D@F2yK!KF^u|6?entHhG6O;vl(#J))nYqLplvg2HTdclDnXXVO73T8Y zOT^_kN+~W!&|INKZ9(~i5+%Z=H={H+D6KpwTOU>ytDsO;c3>`Bv2rk6M(lL`QdGEn za-J==$VEwmR4dfNpd5l^&1HZ_d2h5>EC3Wwdx>T)6=dZ462&56Tl7<lbp#5e(E7yqrKpF?%YxEdP~u#ad`QCcEe2(h2j%h{#U-FneoQLF z<%1W9%O|LzxGYE411)}+54&%nM2T4JHs;dFpp@~TJf%^-A0b>G0t%$WV|`+{+_l~H z%a?*;3rc?%r3R#0A$p~sSnLf*))s51QD!NWs)f1yw3xV@M=8bS8+1L;qOPF)MTrvO z(vMNP8I(32l)Q(O#VRY5)$N$eeONgdE)Q&Txs(ttpZ&&d|8WHj7Ir!m~fd36c3lQ=gBWaQA7D933B0=pr8zN zQMNKle}nQCBx{Q`)+i4tlxo6d4(9ZR%a>bRF5?B|HbJ>Wi4x%wf+Q<6*q~S*lwGrw z#i}ZlkE}vm<~>JTmY{~>G97ZkW%F6K*pn{GMMfEFP>w^g<}y^HtR5;B>kJe~S%Nve z;ZnlGwy**&M=o9 zlqeC4HDE5I42s`_@|;FFJw&)X3=~ND46|*+<(|#1U%nQU#{?zaMXAkP#u${hAz54O zPK`2Gp;Rx-CI1=XQpV+S3SAGhs4pmgQ=&w;3}TdV2Bn<`W%q;1V$~GN+BVGPIA+_1 z%lJ($m(s#zYXMvAX&0p@q*|eg2IT}KYc9hz%1496VqJjZX)iUH3+bi&Qc1WB5R^tP z%2SLo*`Q4GpoC{AE}a$1FZVK+i$wU7seu%tQ^vB>=hb%M*e!#6`(tlxYU#T}ajzi`OU*E0mhTrJ|S1 z*Bf0flLe)Lp!`FL65+A|lC02!1|`{pvUj?&SapT6E};;YoTrG(hp3^r%z|8S*^ZCa z!(z|6C|4L|mO=Rzk~No+8fEPuu~;{tKuTlG>5X4Xd$_zQD1!v0iHq_(Bw3+324#i^ zrHDr9qEODXF2p7DBylM~DaB0Cx86R&4m(M9tA{J}RT;>^+iXN0D)0D-287N%l z0|in#VYY3!+`GZ`%P~QDN>GNmD0Ly#3S}FV6_Bhg)?A}xE0mbRT+Tj0Tsv9Gk(g~8E|WiXxs($wc|Wtoo_A6DLaG(Y zH7MUfvgR^cqpTYs7V7~NPkX7&T>3$R@=I0WGFVWWx+u>vifvG4dQggKlx_;;*L#>t z7Usc*%bTCLT-plCmD9{+4<$;(VnIm4{wjkK?Lm1dQ(5dxf8nwaC>}07A1A*&g&K-W zD&)d1&j`v07iBl2JZVtghh%NByEV#ug>tKKc?|Pl!{yj|m&;T^X)GuaF3M&|vO>=p zloSuj7a596j6(UeWg#xPj}e#MsG+#b#gG6lyMAJey+G+0u~?W<78{fwAX#%6qfyrP z6N~i%3Z#698LHt@&co#$K^ZD2&0LfVkYt5kG$^w@C_atSL!q2&QHaZiM~O=@m&-@! zMxn)3d?p@ke;*}EgiAbgdD);;@t`c7qAd37{leuDpg;=EgAJFqA=fV_1?4$G8ReqX zXD+WAln)?TTdbuBBNL*S$g7S-lZWLPFDJVr~d)@DB|7t=vIr<1E`1k~U*1Ix#ia}6!Y_*jwi=#qI|Rr1-sDDtNf86qJ#I z(!xc#3`th#V}ml!gHlSP^ie2(HY>zs^CQIN7MII93<=Pph@gB)i4x&*H(QLJc*i+% z7q@s&UZ0>W_D3(_@+eRsr2%HB#xL#HxPJLwP?pdK?1Rx`U6h8*3ic39(vL&9mw87lbaCz_} zmrG^gvj1D=@(QJ6#A1UX)e3DkDEW}AxlGb1n|g}H`UAz&Ug|R!8xj~w zS#!zMDBHS<#nOS|X)g_#%j=M!xWo#Vae{KMi$Z0r&?$qmz=Pt~C<7JB-;J2dMa+Z@ zm(}mPTsjL%NkKV6i4x(m7?P~e4+f>S2j%V2%3_zg2$!dU;^8tloBTp{Rb2W&F8uQ9 zF}B!b7v(Uc{A5r*fn;s5b{geLh0;j4RP=H=z0&3Ku%NUOl(H_$UP!V+jzQ_^K{+u> zak)dG>}*(w%QIQTrJ~Da5r#K#Idqh{yh-U8u~-@A@{2+F4U#pNX&Pn6ePXd8K!KFT znAI7-RP%6ID<~5LCDBDG!d%W8lw1$WO&TR#q5N}4AuhWfCN7;ILHT79hBs(YN>Gkb zqC~i~VHAATS&fx-JSgvuR2KWYvv7G9D3H<#^DV=r^9t85zX;0fN7!N+E=qGqwL*Uw zlns!qEtafNo>nML3Uj$SkGNz(g5vTghBs)DASmTrlqrmYZ#OG0y*()3j!;}0DU{C} zFc+F{87>dM=W?kjTn-;*F3Tw$BNiJCsaEKsLHQk$HJ2G0WoMdLY#2~H?d4A9@(Cm; zE^)$TlAyG8QQl&d%Lc{vpj6Z-Llnxj+nGxiWWQB?q)!y&xLHTZ&;?hK+?5SUf%i=l2rGd-k zaSYYqa`X^$d56+5Vlh8+@fno!kgU1P)F_{K5{r!h3Z&SWqZlqVJzUlc$`nCKa#4KD zCDNb-Jt)x{Wtc*VD9mN=L&Rk$Bq+aZ!%z(^q6Fn6B}#-#J4PvOP;T>}ygyV~>{>_R z@c%3BB7V$)remXKA0NlqWnWH*1tp3Z=Mk31L(>Tt0ov<{BGEbu%=pbCi0tHg?G2R+3u^uj)1Z9SxbaYY5Fqb+81-2w4j`(G>LF|3zDo*Q-jjjgYxkJWw8loUa!=As;jBr9~cK^f{n zDd?}b+@nywu3d=B>(hwK0O+aw@&YCu(Bg-^%;h6WlZeG`W-cua%2i0#TyiwZ!FIxB zB2XYD8RMd(iMZP;nM$A*Dsd@<%2zJv4>rhc93d?EQ7Ke zlC{OUYn0d8ipA~)il@DlVJ>?iL2)T6D4hhQx{ET8QQ8@l;U1K;_bV=~70QuX%w;&n zLc`_pS6nW42+B{pnaf&AlZeHpL8=vMZ&1RJthp@EC|_H`1vz0amo}N?7oW@JF35#n z<_Jm`7iA5j1Psbz4@zx~GEt$F6)rO{78)+wmbzTh1?6Tz`GwLX!sT5^vO=8l0jS&pr`W71CR?YAAZgj%XU%n870l2?1f})v7Q=bS)y1h5h#!n z#3*IBTzc8%^0c6&2}(^DtM$AuexCAugHFQ*n72Qz&Th z^DgGHj?yGzu^P;!n?Wh!LCMu9N7@LN44^>Ds~DvWmpeRMb_&WoLFwkARAeqa4ayQo z))uRyQ6?*ta)r4ZpG;g(v?wq8qU112e}gjGgYtVX#U)Xpd~*wP31R#&T%LZ><dPk+}4Pp2{!dAs1Xe z-oX}I=%Soulo1By3rN;n`f8Nrt;J$VK!KE9=-CaIe_n98JTE9+1?5&3z8*Y5SJkIR9s%e6bf1tY-cW?Qkq08R-3tuF(^I{N>HPG(@MBZ z2MVOn_+hv-@o?EAC^>@C%SEZeTqYWnrI4&GcAG|-s!%Ev=JM@$;<6nQ6qm0tg@P8f z1mzs1NrX#JMwx6-T6$16cUKlGuTb8r%3Lm^&ox|zEq479AzVV+*kX^kD4ie`yT=U5 zSCFi^^wTKs+#?oi4-`*(smNTeLW1H_TDWu}j$t|oE$Rr$pOh#OE`1p#XiyS7DBJH-7OSLCR#jpy zndoy3mr>8ST#5^q4V&0vkGd%LK`M5S8I;42tho%*C@b$0i**Ewr@d5RF3~QRa>Avz zpv1W-K}LDfpiJ_hT<)y61Qg1T6`9Ku=nV~*7oT>yv=EfvKVvRiC{ZF7%YsxZSY?H>x~R<60;9ZaP>w>f<}z5LtZF6}O9cv~Y(_t2xD@wrc|}nA3d-#+N&zI{by)^w ziU;LNs^Zd7p`5;{5SI@}5|<6oQ*n6*Q(I_pegkvaMu`%!SbgU5xbbQfhiqpUV4Z$q-S*qs_>u0p9UTxz2?G=4ewn9F6Npwt(XzbR27Tt0y$E40R- zwDX|sPEi)Crcl8z_*{75$Xq()v-?FGmFB2|*d+qSRq7 zn+?jlkgP2huTdUWC^ZXnDHuXrqFgTDV`>X68VJfilqeA{gBfM3K}q(Y?Cqc|R$Za2 zE6ZHQp$9QsCI($DWrfT3b!@R`U6kICirr%dN`jE(+yL8Rqf?dJw~9*&>%qqM%&BMCX^i*6P zf?ROPTgw)E-bJ~}C|??s?;u%o8Ld&)-60n10Tf920&UxHDeK{~Tu=rJN>dl*JS176 z!v84PAEmeriCxeK~S?H;_tj07QT3lJfT=r0+L@ahEb2(~IqCF@tB`J%Y zX&_t{0tHf{z4NieM_j*rBPh=Z$_N+bHsk1*MsbvY1hRFetMqlHDMiiH| zSaLv%@CVH003}L7wE!Ik-JfcwQ73OmOe&UhP=1DF&1Hf{*;r332Hm~wM$ z2|X2;n!;tIptNvNUSgDg49YwYN-2%fN1^;#jJa$^doWzy%XYa01f__ed`XEC;qnwD zSt0r&GtTMx2M@~Y_bQA1QCGM;3KUP<@7I_7vI2T4F5Ms(7G3f_TWqY0@&%(5L3Ke{ z1IgNA2^wXQLZPo}L&^!X2gBu5mdj<9pfneh5-v&}Bw3;22IW2v%HagXrJh3B0?HmP zkM$ugKS58$Wj>Z1&|?2e=JE=qW5i-U=2F6-gp_D6m%ziWUw#ynmsYUF#=9tunM-+t@-ZZ9i?z`xk1CYg z3v;>Hi@5ZJp2{!3V#xt5?iQ3t7iAoyR4^!AJt#+8DK2peWm^PuxfAWd_+{2S;nESC z?WziwFWzG=uTeTiEH)HUtx&W*tg z=kn^KyA>wkZhDG+1mE2a*uT?0No0`1`pvJoy8;f z;KRE|fRdH45jFY4yQ{T`Af|*6fe-I4rEB;HVd)T7@Bu!G8ipeRIe%r$4CI6}rUr73 zXG}^Z68KQ=K4-{7;tBe`ZhII;9?w%s3>FgregfbN`djOu4_@U>pIR*KMbDUu zn%P*;S)uoFRn_cdYBr~ux1!7uF7uXK<`YvU3S~eOJ1Up7hJK{8L@q?^Zi}^zAe6Wykr8*As;#p;*hj=Rv~Mz>^*=YTkQQ8fbp73 zln{GY!+HKwD4&(knS{`Xj@xPxH85=zdpD+Q_y>OHpe)<_F~p0#+v$TS%(_$tU+xb2 zS~K^nF87axF)Vu%fC?I-d>odk5ZVXDPhd@n-nFPPXK_>fbPH{&i8w;*{O)Zw)x+-^ zQ@szB@VYEARR&xSQ{e-}{kGyO!2FkjRK@4mSbWkqBetpv(sAQaGsSiS_8<1;%&~dC zzW9M{HVb)s=++my)u%=zHroOXyc=*%->r2!h$HBhDY~78HFol~E(mYHzB& zJ=NcKGnz;^dTM=8Zgq9CZ%0z2Fi^rXdf3sw-^E{Fco0iwFtwK(tE>YAKA#74foz6z z>9%rv#Kz!Z#lx_QQyM!`=uyZB=Kxs13bz3SQtX$ zc>@l){wZYDVJ0FTq-G<&smEHZG+K0ox+hJO$i17RG` zpAqJ6XJL1W=dv8nW3jllLLa&o{uyvW*8&j7a~~{;A(Y}d$i*plr?exRNb$-jjw3Yv zSDJIoOnM6(g3+sCP!%)ZV*7>)nPoBm8P9Hi``Q>Nonl|Ik4bO)7|z*RtC6=?)*3|7 zDb~6KY`lEfSgShD^BbT-Rzeq&MAk~uB5GmcD%QG_u8Fn&fWB<4C*W9gv3F!$DUv}x z`5o(BybIXnzD*dT=~pI{?NmP}%z?Xx%hJqU>GixR`%5#FU5v64@cW!xtU4&*se6g$ zdY9dLeN4U8s8=v7)y}iOqVJ*?dr-RjYN&M^*ShM~nhbpk8hJbUGS2s%*^Av!{#&f7 z@WE;fmP)%3C@)^@0u=G$v3U04CvUJ9&%9~8c=IjBizi{C|L(xA-#^i#S8(d(6#HDj z{zaO>TKvY2J&KgPa_oExA#rRiD26wgnE}2X)XRSzX3R>MM$*W!W3-6&m@SH9`_MJ~ z1M8b1j2*iUVYnWdMQRIWv9!>}?}8+N73%Do<+vCF4sH&#f)=Uv{#1KMz&=C{Xj_ro zy=g`di2e&pZY$KytvMUz#h~G)+Mu~OLI?B6GB)TE*i|(PB(6+Xx(K_q(1+Zp``DxV z4|^1G#_VGb4%lq3Auq4Yb|0Lj%;o^p3XL&lD~t2|-@rX9A(2JEY|XR?KPIMPw%T+J z|G@2gD9UD=2fyRn-W~6(72!000H2Omope43;eAFf`DQ{HX4`?X`SVy#)DW}=6On=3 zg)u&e%UxKW{sg09=OCPO7skntRq=;D0-ZNE5=V&Zh`hOFaYRG|d2^fKkp4O4QcX<1 z%P~F8)#6E`#VlwcFqRTI;t5gmG@-DB*1%SdH!KMvU5O>yr) z+51~^sf6itAXuR%jlExN0t?vCH7nu32oeE%Z^HqLSUOkR`vba$f3Wvt2rKx&iJ#`S z@BPRe;O!(HZa^vAzRNmAI~iLn71^7F+4@W@2yBewXvG27yn*VqIFlHST7h|qy#Kye#3LEpY zSB){xL>psb322OIrf}MxpA0khT!BLlj|0fID|^l+FNr-X0~F28*z+Dh=P!XVvJ%FV z5Q?xNT0{aC1!B+p=ojP;9?zd#z+)+ z>YuikJ=EB1HV)ZduOOqZ?3D)fmAy^_)e5aO_9~6@{ExsmE8$)eLiTE^MMTZk_NqnK z@DKJn1x*V+$T^;Q3r-4(#&EoK!tn%Lz&%EYF6fF`Y&!NR>9G6=b1WT}3%M9Ro~~8O zf(7AB*z&|3u;uLmo67P?(gO(`E40fk8!zTV*)=FD(bfh>Fyw)kIoi4|HHM_UC3;fc z7S5%S#4Huqf;=9pRYHd$pom$sG~>H)#%An`G`})q323g&xD=Sk-x)J*Z2&XUVmT|} z;$UUQ0vxc29W%8VkI*$S<6`K^W~_()ep!(iaNdQcbp;Sv_e&NfH_sR4ky2@TN--Q?EX zgYx2zZgPY<>^HZwiD$gTCjJ$FRIJ{D%_u5XPeCRO%rkvhc@Qnk+!G1dR0ysbz&eF^UH=e&Vm z?5G6j0V&6z161ims*vjR{W?49)A~?173xMkj3X=ymf$*mh>722g?fXPv!#4QL{czw z9*mn$FWrax{Rn4q|9)p^m*P;#VZ33q>1Czu z8;)a?ig7;jlQpIlfi-vmnQF5G;1A0EMAAptHjgP8k&H!dX<2=CgR!`|PqYSE1FZpr z`fUX_YN5n&6|;Ccb~*jcodvg0|C1fN6X$HG?AT2>Ot$x@*n3%nEf`=C9`isYjO+?r zI{gKv@*Vv_5m0#n>RO>0hDrqwl^1SPRPwQf!=e}LP`k-*1mEe-9Dj9x{>cFJ6X=NU z&hxFnA(eIxAWshsXk9ehHD6`MeBNN*1Dt}9SqYO!0(IuYb^Sz4p->--;cj#d{~)O% zBo%yw&inyttvj7}5q9L}r{!`pP_;tK-I)4VXrTXXE=Oo_pVi~eVftdW#~1jcI)`hp znd%(2qVeh87onU47~PXSpC)6&T&Ob^#GO_HNg62Ou1aS!2aN>!Le91{-80#E~#EWOXg#Vx~4^DY_<> z`W&=%ckVlM1&tc(fnGrAiwjohvP)?q5t#-fv_Vj7BRqwvJJlfXj^r8v~yFeB1{|7?B6y2LfW2&dEpU>Bux@P-Pj0I)xg)*2m z2YZn9vGCpLR4>IO{E#~dX(cOk7sT~O#a6CR;w22v*3rJ0 zh9meQ{aN-!F8-)?v=%!*lrLtY_prC72K&cOOtJfwPqC*|O|gIAu@=c0bcCtMv*nig zp9Q3o`5*r2)>r#S&3o>}5A2`8w;KPHqv((>?k(tOg+_viXutbb_=i5El9dp~!#eDr z^EhA;`zCAuoT6*upI5<)8%q5^u*Q3r$T7z=d%{LdaKQ?Vb#2rESoteZOE;47DKzIf zqCM4yD@=4N++&RTG}J1X!=B>%C%c`=*bRV&!Rs8PwD6pTTkj;5?tt)aH4Eb*x!3>n zCid`dw{^{m?E=b4&I}PmUVWJbWxIkFLy*{|iX5SLe6==v^@XR|s~_Tz^6IHbIPIoby&3%j?h+mJjGTWfIrHrw6{dHmHUuF zFEk;gTSVpJcV(@kd2$i|bDML^NT1e4C!{eO4P+%W??o1PAkSAH2W-UR6WId2q271E z0`f19Gx85l8j~)9N>+##kHT#=}+Qy^|h4Whcc(<{~GLmwixAj)g%+Y$B)VXT=7fO^nsw?b83HPj9P-nYDM}yMDhU z*_Ts;4L?PSEj8HuCH%4I6@?MdCbMifdKf7XoS0rvEX8hOr%9gCc{90qgq>ELla81o zI?v)9H{URo^JhPV+GybZY<^5~RwrKE0X30Q!6Bbmz=6snkB83-}8N~y+?IO7bOhqT?J`+hWpP(O2ET3iv zBGXVh82t_ZKN-6c#_YbsN=)sv=c7|7Ofue@qQVoxu^x{6WbxfAVH)9S`cFL1Ms8j+7oa zxhh2$&T)qeEg^{_BpkiSPxg;mE;3$*3~L?Mros}_`aKuXM)5@QRO$eGG-#__3serASH%MiIl=6I7I5A#Q#%>PXoSZ=ZZ8NGIB+n4H>y2&W4O!5ofoJ zNzLWz_=TiGL?Y4-y{J2-lmmLu%anA%oFJjjIU2dbndLBc@H!>N8i*xh0KMgOTK6YH zom8y)yHIoLlNzks6y{0uC!YetK(JvYoWUi(Q0xU8{)%)PcUD<#V*FX(P&=f_Io+9y$_(7$jsx6qRKvU{-X+}w=)kti_=*Xyb4Jc`=x^|CbS#`O|>efm^%k11HF z2P>iS;qp%7ez*%^V=h`L7mhy+W0oYrftLOESItPly4;B>Tq4F(A|~UuB&bjc>s{yc z{Yp@3t}l>Vd_N0K!g0&$d6_;~sGvk3yg88TCrSZYi+7=+1e0Re#$!Di&r)L}yZ7{t z8|j=uNw)LT=x!J8NB_+=%CS`fjH@bg=UZp7T!V{?(tl<9{NJ zAod>t`zST3sT({U2%%#oA+ zSR`DjionnCIg8lSsaIm;bhbK;lhg4ywZhv1;q8H}UyEcuX(8C^n<*1Hn`1g)&%%Nb zDL>2{&|h8RHT-%HcP;LCUc!Z==4mmiiNhlYw8+Jij-mTZ!)Y}7P@+rrMIR)c- zOhEHhCr^TlGNgN;F%%z1*~Vb>o20+6ya*w91_1Jf>olA*Czv^0iFwC3o&RAY1I05X z^`N=-uF?sX6myrtA)9;CpT^u97P1<@lCy)+wOlnuu^PV{H5LI{t1%tNdA{`>J#*v@ zSF}PfH`+}?R>HjB0GEluJbEsU$dqvbJL+j1hriSY3LtO57dK^;LBVM_j^7nNpf7tE zQG1K&MYCF1aw6JTnw`&g;g3Ra$*jMNSpMXLR#uB5f!z0EA4li}azBlI9)DAFKjI$1 zpK}gpX)V(u{n;=W)4NsTZsu2lnj=d4u)m!5LP?g^xZfeM(9Oes-Ro(ZE_UjOuZZqTUmR;skS(cc07_31GaLi z{UT?AK?bou(Hda&A1G@e?vME%g++TDF{5U+vPW!}_)V!n zBwsi;hkSIf3t|SIqyM}%U4sevy zOA7UpR}G`c8YE<3h;V{bX2U9~_5?#beUmmsO*x8j#$A&31io{LKgh~$ z8t=KMdF)BGzv+(k=e#oktR~Q`vfH7}#cvDbX2&+cFPKHe|Jbrdxem1={MkE@3rooz z6Pc1brF=^6%&HWAG048e?@G%Z<2c_HgfY%H{9^_-iS>NS7h#{n1P}XUJ4FWOU5

    pPS|hl<&1oOEP2;}0fV+hNym^tz&Cd|G13+Bdu_Q0!>IT{)W`Y8nW( z^92%E81AY;7_4)YIw7hz_j>i5YGorMZ0Clm^fIcvF&J`!(T_r$QKjMaMBZ4H&RUfV zMXp1k*afHI!D9cV%&zIE#&(I2p0$WqnOCkS^v23;rshL-8FjsSQ074Ve=5`3C^La1 z1*0?Y$5^J~^@QG7nPys<;}O@lOdYzNb7MEXKTTWaBsGR$^w0QXl$n1$p*L3MEfPVN z>3qFkp3oaxCP*U4GR>}6uh<2DP88Nz&isQ#rFo~IK}M4 zp1-c&y01`bnlyUtdi8St^5@J%*Ww(+gKga5Tvww>nc7B=lel2?+xTN_6u6#b*ysit zjn*2Kzg|6O{Y}v1#({h~L#y(3T~?*0SCtJ{uWy_ibbTk0AZO#`T(2HtC-}*KD$~~} zlR=V#(NExyu}hQd3B7Ubbk)jS`sX?%piC3o#`+J-?3to1^8=-^gVDvk%DjF(p*L1$ z8;PK>8FRgQ6v#=KayT!ggu^!un|q8#X`<0&{4wUa>3WiH95zk0Mn|t)$2=6s^syYr zu8!0gjhxpFu|lXM&-0dYpz$1GWO@v?|JsiF5 za@B}%bR+ys^GHlz0_nKE64%4g%W*AWCyYLUwJ9d9i993wC>Af7!O!o-rqf^KIh!C` z6OO*Zg`-=eF!T#>vdq^AXP7^UoZ55=8LtC08>anpcn%~%)?Sp}py?ew2UP8B6?zVc z=6_ay&yzqhOUz1$raBqLu{PhD=kwtNF`<%mD3VCC$E-7>FmJGHp4}%C*)F{n@npv~>f2b{73aWtwoAjPZKDij_J1 zi63G4Q%~F+VWUiSoTl2RklxE^jdY#8Gbz}yXi~6kVmP`KE`@N19G2}xmFV6@0gRBE z8-s=EC#+`x%RApHI>c?4n0)&1Q6RiEpLVK1!_g04BqmrvWQAj{v4DOj_e-AJxZp*eM~@AOe*815jYs5nX<#m%h{Msg}%%d)rU zoK4NGy9uRmUz986`qopOO9Z3pyhL?y^Q};wMTP1-LUp!z>I|nkh%Bnp7^%#hvnlp= zD79DT_d4dLHmQo0R{1~zbvBcKWAyyFsFGq|NVD_k)=qF1lx}dzUn>#Tyw3d}e;~Xg zg=A!1t1x+A!7YS>O!5u_3T!@AT!G+Xy>RsEzohNK4ACneL%XD99)&=CTwwG9AOE4S z%~^z%SwSg?L*5bBsU1>8QX&+!^eR)rI+YTNsmPUK!JUfbHyYQ5d^E1CZKF)fO6=aV z7gk&8{8oC0)9NoUg1dtZ{h6&KiyNa)SDi2G*i`PVM#;ppjq zP+$1~|NqaQRgv_KdZGpOjlO4#lK}2IZ$WWqEekk;GS|^+hJv|{Ru6H#S5OaHeNlzA zYOA#xjEmTZ2*fP02qdkgb$>gAQ8Dvf@ zLK;=^aMBYa5o8i_&=q3_1u2E~>C-%BP!tbD(jIq=zEepBj0JJdTfeDXW*oNTo(TYP z_#vb%zO>f0Kn%k}k@ zAL~hxesWex&IZU?Q8^nVXXnVHxxT@2c2dq1-X1w4{eaG~m+KoTXUpZ_Uoah6#bhMDziIP5I>r3zx~?dB@a+UHx&vr@YE zCxi=<*%Td;|A~yCX6d_tEHr*V5~xAACNz&;hJyGZ&h}Nn4=R~G=(NtR2DUm|W)?;G z&Y+!49at)5_X?Ifx1--xznA3S->2Unceb>Qq~9x`d>&f975V_b=VNDF*vb2|q`owZ zFMUXt?x#!dT_~n6wd6}{=u#V80@+)r2l*V6BpesRaX9*=U#TD4>8!zzoV}$z65fA$ zzogv^(L%c==iq=^qVt;M%qS;+ALgKV2!(=Etn~f0n#w$7aa7P0d+|Q}8QMJ*&x+Fx zgDb2ZeF0b1dBoI-kR#Y~$JLr-?YEHahK&xAmj{)OrHziwpjkn0@H!43Ct=bt z9XB#3(idKMrz;)ACE2@^@c7)XbT0tIJJ#zax~4k)SbFI2@bevNTP{tyn!CgFzAk8ky1 zyMw;pM;q67Ij_Tm^vG<9yb=UX6YCOrC8d&$guHrcx z-aqs^4p{d}qC4#$+B|`E|6Dd=lyP-Wg??7(n$dmY6<7EEIA-17{a))n$(~HPgkj>mO_2PDI~HTrqt*dCKM?GQI}690hdXlrKy8Wo6Eu@oh-xspHqMl{ zhJck7`q?nv@wdzP0UR^q(09g1xZOzeDvlj&rGT_faq1a5tP;9718weIngjE74qh zI>k93&SX`P-}Xgvb~5b#45g>Sk~q@#WhpLwz+o_?_G zZyev^4y>ou>+XyI}>GSfPF9cAgI`baQ+Al5*gsZ^eOhXN9-o zV-FGC%BO|WDNO!T=xKtFV%LG3Q+*^k;=Xj~%D1HYIHmcQ-q3tk!0uKxV4uU?^_?h- zFh0R)KOV1gaRj4-nm8_hL%Re0rA)4K?>nmTBC5 zILAF4KsX=IWHda1iEVQSoaJaZN%keEzwiV+Lh7pbf234(?80Sdp)Br|tb)E&eg&7O zyF$UJ6lfD@wF9fPa{>SR7x1`+Gw{xmn7)$PGcmeHEyX+xW?zND9PK8btPl6Lu%k|2 zB(wiSX6L6#9>qD$)a&lx|8G*gYp0tpp=gFmhas zhZ*B%gK@jTIPe#v5k?J#(Ht157-0~RPseToS?nT7^}oN}Y4%N(2<5LN{3Sck;wK$` zN~D#23))W8B-P$)*@tO+O&t9f<0Sie(m_+47oZZ1@UD!pKjJ>V^5$T84v)5**vo}l zI`$fpiqzH5Iw3}QnhoC}Jz#ffLHCR+!6G{$z*z#5bHL?VEpXE{U3RCX=8kHSUxe!5 zId?!pf!jrap{@dnY@t&|fk=-6?_E#|T=_;6pe@oQKhUyI3x7@y+u>^{(GE(${fC?( zjSDM61vAr8V(};n<1k>9_kSeW?Dwg11i!bIqZlYUPX@*H@F+YqlkjOXi!?C0<5#E= za$6IhNff5wvF;tttYcLW<5-ozpVZg_!TZo@n;9e1Pc3%hpfU6*IohX1{a@>Q= zhA!SGT+jXKI_7B{vtxcZ&UV4FB?UE;bHdZpEHt%mklsY6W7)o6AcddI0me#uyhTcg%G&N)Hp|?SE80w`Q?kYBt zPsABz*co_1xH8~S5Yw%sB#zk`y^cwcc!Go0Z4mCyz6(_Yc3L_m7pRM-p4rJ9%uSpy z`Z%L-f@DL>0f`ySxn#?}EixM*ny$rvnS%YN2%GpL=ms#?P$K?~1M#L)M=zgJmHYNX zG)ZOs-b8yLEbiLV?5NJ{F5gnn1nK3pUlCW;qjmHC;O2owaT|^hEUU=P7#8W#I($tm z!Wns#>E@5cXh&Ushq2P-H>5TOUFg<5vHUEvP`|$lPf`P>3w7l5!DsIDX}jEWh086Z za&DjhF1LIbD%JiC0n&|hZ##1<3Aj>VEAR4n#lF)iWgEQ9K#~ zFeL0twzuN}Ckg`kFUDz7iU$RnQsqDk>K7j7aNn;3!1krWcvdI~#o#IQ7a7o6hkI2V z<5$f84|CsBFp6R{#XhKlbm5^Y5e2mT2}2&FcM<86Teo0*nBwh@<@J+>xMi?sQC0SqSf-Y~iF<;g}^MEm=rkSI96UWZzi`iDx05 zAY?{dz{X#@tC_0QfLon&aWq4q0#*w=3RY}z*XUyT>l z2h)DYAyf+g&@Z}bZHF>q-rvq>^Pa#F%zJt+x3@tEEM?y62gSUwY`?8pGm+A~lb1h# zF6QMAWhq{TGpx{D7htRb_!?phULuQSO;0T{Ers?(9+SKcS*!!sobT3bYHGfSn)&H$ zvF_;z&V#U}^IJ6-D7PIRHVFq(X({JiZN<|S61D}p{FRU~1-sG$%P zz$=CF+NG0aoF$OS%UA{aoyVs$c~Yu5dLL}TrEXc2sctOUO`y?+F8v}(K>H1|So^7Scn*2DD3zaw zjEC&9@@?Z_&UOV35;3r55hL&|C9q&c0G(9rn$#3{Wj2s&FLZ0?8eeRcI#6e{s;!3U z{mwbOPYqVaBqQkiheKnLE1`{%FwaQ1O-s04e7gW;XejC8M%1PGb9`j3pf4cm9DR`0d6EKEg5oeB@vyg1=WbB} zzBu71W$qwl216O=HB62*!IOw`SZ;wxOmro_%>-#*tPs2kIhtUumU(19WJW<|Dmnrf zXgakSdnetob6)ICS!sW+pFQYnE)2GI;X0Xg6LSOBIIX^{6?<)-7KH(c8mP4Jo2Y$`8)PP{uQ2$_wwfCY9@1!Mh1Kx zK?ZaSgVf7K8Do+syQveTv4ES%)0z{*O@ucE0`Q*76#u(;`H+dEw!!=T z2iF|W;8^_sIr(4k5O`myBT@lSRsb>uppgquhL!)*0PKfQ4KPgu?AS#&Uyf&Vr3qpE3w+xr*^au1>yZ7t zSpH^`O%{=H`Vp$k<8%`)79B#M;)fVo_Fo2`GZ9kVn?2@El;`0k&-x@C&%t9G0^&iv z7oi@`ASJ>{!TkvI0W_!ltEX|wV*s^6HzUBI5@takeup_F(ZqWZuyH`%i+}?bu~sPI zO)RJAnwTsX!uWamIK(1&|9*`ol_l_~OVHO4ZO;nTq$Vpd&;@?Um#0hc=mAQ9Re?JP zDtr!sV})Ye!gZ+dKp^8TmOmSVK*ZcO5&rCEIKaH|1ImbDZh=+!VW)K_wM3p6-DgLw z2ML;IBKUpXbPp%yz6F=W+?!AF+TaD;WF>QB0B?hkwL$wI+}bNm?dhluIheZK zD(YRqy;7*Q3_-);7UK+tViaVV6lB}*PzH1w4V_xDRQmS1(rK>9f>Y~}PV~xWAM1Uq zTV|(0u4a&zf6IoMC&;(E$UpDU_wZJrC8?04m#L%|dbXMtb$I?FCAS|QsThMtD)6e0 z+?n(`fO%);!U8G5b~|xfCneZvT}p1dbvQj`+4;#xB-zLC!zO1=rBV^d`))$nB<#+h z-atK_f>);Q$08!esa~1G2CF5X#hhxQiXURw!1ZN^or?6mHAI zy|LAqZz2i=VoqE7%QCVz<5@F2_MxG1fd z6f6ta-?9|v+lp)q?5y?u2ob=%ihH%-I3Acv)cdGrg_=ONglR>b>;CKG6Ji{vged1j zcxASyTgMj7%hS)XhRyA$MyLdk{V<&sYOeJZ?psI*a~~x~G5L>SsfkzDr3U3G9eEqb z7`(4wN=%A9Gmf6k*RO1F@5j<2F{ZZ)m8^sZ@n8_P6#S00B3>~|nSs$*|KmqEEW)Y# zJF7;oD=ONRnO79%85Fz{4lAIJywp{gfANjWd;^a2d=&`;|8PCVDY}gV;vdLY8-yt;1(Jj@fj1TUah`HmbmE5A!I=ozS;Ja)wAgL|zifm09v6Bl$s(KL>p|Tf(Q(|qv;y{L0AO{<6-v$cym2gq4d&q zUlG~`!Ce&r&WAg#(DT|Jw(kouC}OSw0WL^;uajiUp>aKzg|N4sA2H_O&Ee?t5PjAS-MM8cry$QL+0QA@K*&q#Od`{xSaCW zt++3s{Ix&T_-ho@K=+BIqbRuOi1F7E)MS6{(IOr@qx}`4YxoDYLm-U(RRqDN{nZl* zIFG-+<_%xXX@M$suT2rl7MV$aUVl}isu+^U4UclYyWM*E#**z&&-GUt7i{Sk+-V9{ zqk`nGu1E!W{Dn1mGt5!BT#rUr% zG{Q%}jQ{FEuY6hqWhG1|A>^46T15SP?V0X$OnV4{z?LoEz+}R)TXOK>S)yj$-oZMJwvD4dlx5IIA{K9e$V1puJ zb11Z1(Vk~5y#%-xfK|ub2O3$Sr=hkOeb$%md}tJo^L!g4B`8_X6(P)o)q<&yXZIWL ztidDUG?TzOwo4@KlB3se;Vm$-Wj@dfpFF`9xZeZ*wiwrsV+RnXKYrY@Btr&2t`f0> z9vZ;dr{vQ69Z4A9GyM1X_djpK;OV3vF}=pK8TI#1mx`LK!Fn#d-z|I=isgSuh3WY$+C81r zqK-eCl4My|i})8zqJwKi{R?Q0O^*Q}!s%tnkKwPK@B_L>+t(B6Bwkx#K2Ht~V4KHW zT*s;qi6dkdaBH99FBo5c8{fPM`BDUdBWiRQuUp}B)coL4#&F&rM zPR&n(f|LC*wmk=j771WXDwtzOS0#1!I}HPdfo=eHW`gBn1rH_b02jGMGLCqbTFTOMW<0X?Z~HJxPlTuK?FSpK{&#& zT?0ojWXiMx$V>YeY`9x6#z0nK%U}1UR{Askfh|EcNFCq?D#7knJ zf#CHZL_p4M|d{JxiHJEkqe>DtTkbLFRF->L>&ijNWgtu zw3cm%4?V0^#WM*s;h?R`FJjonzAsntpN--cWDCmvy==`@>1@reP)%909MIyA6W+mf z9z2Pq#6;>#{`CZ`qSOi7c&u$xZl~g&i>B|SgTm_vysG)j zqEL&+5BfA0!f7LZVA|gy4XVQF(L_V^Hju&8#t`ka2jNr;6|xeZBT+Q9v9*ZKvPi;$ zZ#rEg9fQ8w5XK9LP4E`B_{;Y*xRaxYQP4|F%a=a10!n|#?erq>@|SUa={Sd7~=@KE2nDjz@Bh6BDWSZrX4;r5y{T zuAwT&U>E|GbEko$xb=<CQYPqm@*nV5thpH2!s=@$U*- zowDZZxo|zV@L8ypPY?29Qk}XH{mFdbJm|tH_BTKKq7r9|AIhTK)=-V zXV1Z3jEDzd6&ev=#9uq%3v}@OPT}pT4Yup&Tdx}*{_K?`5$UmXNpC{?L^_>LJOa*J z*CH{3*0_X(oaA$n1I2znysG{nJ=hcDw2$Cn{eI9I!?aqvv~HDPKu_Ff7YJ4&RiiJX zmxDRV-KB=d9x^=M%b|@OQrdt=$u)$nu6D-{y;C2(%CN0dJbLR;j8#u z7|#}|kXImN5HHJ>eX=4YD=-3^a*>C>`FEOY8!AefHyAxUpM! zD;a(aO{zQ3Z55G`f}2V`km%y*z%cvcepRRY1942O|w^+ZmsPHV9m)bZ;$q${h6h|$d+V(#~oU| ze`<1vRtfkFL&{>$hwH?N+#U0w8S0Di$nuEVA;ay2e7BjH*bXox}~d8 zY2GwN{nr>&k-wOQ zI6H%UkczXfoyX*nGwOndVQPekIMkaXVr3uWn_J)uy!#!}R67E;o+w@1$^nJ`^WFc^ z1#P`-Vt6MyG%NI=tI!h25NSRf6||r(sBZyfjZUD)utGynl>3}sOeNi|`2@6)nm-6O zf_V~-pn5e?{XG7tK4;ZCo<3*F%KufL^PlmZY=4(b>s8FWXt)ksmgsJczX*T`XAts> zD!R+wZ7lc(^us4Qj0JaXfjcaY?#d(q(Op_g*emHpME8|T9No+AQqkS+J?@0c;_(?h zTnE@#%U$b?p}06f!KsGpU(cJ^F2jYfFGPpay;OJ=6{bjj3oZXDz-FLIDSjO13K(H0L=Nsu3ru91TLDh3KGw9(b-BXKK&{Flwp zuNpx|+vvg&cXpkn@a)Fyn9gp7NtG@amB$Gt4eK7_c5amq@_ zQ~q`m8W`f{?;afTpSW8S;@}`Fv=y=?)Y74e4*B9ZWouDQVp~!;;5j#8FPPRx2!@*UaPL*)0K2eV{R{I`JOB138<#xe-j^3{No5bnhKBe z_@m?hEpPmziT*FfT)kc$^B2ZJS+5SjA>0t*%>B%Scf)2TymtWJ3Vjb9MX_H#gCi)n zk@bIgdlR^*s`n2Vgi%m%sKi~HQj<~>B^5-Bt0)74OYUo$W~60kpjhq$>U5pdGPTS- zOS8qav~kHbLCw9=GEG}AQfZ5(X7YZY=bUrz9rEAud*9E?=c9Ax&Uv=)dCvBnbI(a? zNld7ty+^V5@wn<}tI==xgOxktWJmkp8f%id;X5(OBsg0!oZnk8Of%v90@6z8(&S$a z4n5PHbTx`n^OsoJ4-2e-{tL(sHGx_pQb4UrcP{;H_>ibQ^{Uwg-j4>voCEK;$!xUj z{46={GQWrYSwf4%OQ_nIl{UpEqxTCo1*37GS?S^Ch`zL_fnPL#C2NEA1fd+%eDwpR z<~)&dLDlgOu&TGRs>bnEv~(z@jhKAMyRR{($W6kw+JmszIC8l>{GTJhgyPg@1Pd01 zRSWQu;0<|$kSclh1JaXExbBSzX(zXvZaqj1lOcXw1%PkZOWhu`TsmyEeRX`g@aZx( z)~0sG>E-jcQC=tjC|uNy6&^x*19Z$M9)cXqT%dyRmGbzmYS!!9*k+e;^%VexCaEs3 zRU9z-Vhb%XugqiY$o;Ti^EYKP_~p)9$P9kHZxgY|K4ZgoJ1dxj8{!b z2b}FdCbCYawo)+RZeoW&q1|>oL&}9&SOknkF>;^-XaC^<*}#y8(x>DXOP?>0W_^}2 zGUG{~39+ot3tbuVpXfvA(~$K!YUH4`C<#a`?T7US`YKMO^{ z?8T=g9%yGzEGKj|G z8jZ)>V4U<3CV1v}e4LwU$AWeMMn*Vuz@j}4JA^YQAjO-Vs-p?URfkp{@&S>jiO7xi zXUk@8e}1E-d0%u=7qk7jw%lxgwv(ZOL+~==6ZKMqh$wJK{WIRNf>SUM4}bNd7@@p` zPxhE?2-gl@&OSqm(lBo+v%KcZONBchfRQtyptgeN$0 zNPQ`s2QMyU;nZw3MtOy6>z!aN{h$!Cis?oRF-AFPq@$sRN;Lj{gfrK-GQUKbX=uV@ z6dIAF#NXx2f0&|%*wq9ouTE72#Y_j+#9-I7xP>+3Va=;v$w7JOI z(AZ=MYtz!wrn)FRw3nUz(2HJo)}t?=(MIZ{g9!305)~TvcO)B#?1Zw_oOl6L5gbyr+EuN&?i5s!6?0m2&hr+#00-u1WQ0r^04k` zjW82OwXpOETVv#k7%hj9-`5h*QQK^=lmBk9Xau!|D=dCbw#^gJ-fI2}R-XGIM&y}` zJS@O*J5uP}cVH0~9=06&1f)9;df9B7>rtV|w%JNW0WrinxXIM5CZBQ2)lJLZ5pP}1yi#zRcS=Xem+DXtVr7S+8()rr3UpK;^| z4emYGAu2@sgK0u+^&)UcF}=N7%mC5&r!E83(Gv0r>=k8JX#>Q42M&lUgP-(pTJ$9k zr_LgS)_lcnC@rz|+Uh5uIlQxZ3VUZrs_7POz*6&*2|~o`^|tm8X*_iqi(dRoKT#cK zp+u>lsDsv2^YgKE$PFwn3Oe+8{gATv6U{qO)&xl7eqt?GEF6QQO5rtC(*!u6$fXX2dFSjni185aw>_iw> z)}>JmOwcMoIm8~}vqITMP_Z})Up-0Vh!wTdBNdB#Mfei!bRYT+f50pl%s5WG>?M@a z0pn2|@1xr3Rp|uM0ICrJ&mxE`evMG#9R;|59u-7Zr!3Aq$jZEkGCxI`X+A{T2Bx${ zv&+RpZzj;`fBa@bQzYaS1sDYl)a$%}R!w(QK(ve<3n!QH#*I&*gv$uSBwWr5hv<;pa%oQmYe0~&`bI~F@;rf_69hpz- zk?4@}i5oO(D&lW>Q0SEo#5vdETq3x@k%?ak-nw>VV+4U*oHhGGpe z!V$TYe!KBb({H;#5zW5{+6ug$d<%e6JMBr@PTa_EHz*ck1u*d2W%L`t<#VKg8T+lV zfc^HJ1D4=vKUhGLx51M1s9E2MG=> z4xj|>grQ+O{Fc^%;J2gnI;YV-;{9?&xuoCzbr;MDtk}t2FiuQGEJt1r1YlM>F2?mr z8{x={-&Bs=UZmKO<4D>8!(CH2@^ny7KxeP{-%`}NqZ-+A31DwaVh)HuZZ+XoZRvgA{bDA#{5k zE-rohk>{0fH-W2IzHRP#v4%a;zpef0cYXzj6Xz4^KhwHhWjsK!2N}%}k(7RY=62Js z$3Y2PYK!eT0=`9Sf&U@o_auFPPWtsJB$&k##);PL=zSvkgRP^%jQu+51r<&_WDGlI zg%b^K8%A3@fZ{|E0Zt#Jl?+9rl6yC8ccU(`IYUR+t;;4n(#-uJ0=P78WF_yG;SR61RXaH$wgLQ^2<_qRN5MY18 zO-T!jBnKIVf}gKZ^=JxG@YC^a7`1Y6SLvtQ=a|Fud{ilg8A58}(jIgFjT$|e@!P=t z8HjGhoyyeD@x@l><>rijUy8ALXM8fQqp<=Yajjd&#B~#t)BN2`Tu-eAuJmqULY)}m zfYqMbio=&%nFCgPBJl|3Z>$YSX>&%;=b-*!qwZd-N$$suKraiLiBw`Ll9G(vL7$sV zavjdx$I9H5GK)wGRx&A)!Yi2~l7hhnMN(+}rvpB*eSS^6JP1sRX`vHKK-^H5 zVt|o!ADTtPsS((c4fSE(sv#CxUc;PB&IK{R2nQnImd@Tl7h!Wf678}B=!gNv+SfSc z!cAS#i_kd1o^@5M0me7nzz$W^x??i97JmRw#3l}^dtZxnVR|ud4%hFIR^Fy0g|`a6 zKvG2fU!;N&yNs`Wtd5Ux(IM0UN&{t0;q%7psW#j$A=zNPy-@$Mty8SLO+C)64Lg>dl4@rmLN zGUg!sC>=bhrs?3HLPO0z3uXvNudPBsWU_;&5(hYVqT+Bv;06b8j7LE4lN!>&b7r%H zC+)Tzd_0;zoH}gb^DunZeaWE zcuxI^QO96nneZ}iz-33D%q8!iJZ3vgKql*hGW8Ije zQDx{ryPiL>_T^5*1i|urC!m7SUD3G@CU(aX_7!t*!YVvR`7 zvllJ5EG+<69+9~9dGc1aBI6sea7^w$m+Sr-iy)RnvHQ;_QSAP0M2g(MFD+I3qBqC0 z`#-Hq_aADx{|OP{BKOZgUBURsV*b2f1ouxsN^HXw+lj(i@t2U5MvDbz{R3EGpczJw6 z{pDa!(u~izzm)O3vG|6<4>Ddvm{PiZ$7-hAe+gAI|2||8P=?@p@j(!HlDZQg>Mv6j ziGJ-wk-OGSM@i4F_->02-^*%Oz zKbbT0>bA)IBxUBQH_e8sf{Md)kx1x{a+nI@?J5tq!qG=65tIndVAR4${;|jO*bJAVdS2;I*H2Am)9mwu<4hd=b4a zN)jBJk_2CLwi1MW*NpP#U?fmJx7-S0c#&q6PtJtOXqxdN`e7qcM9Anx^mTA#&41Pc z@pnN1-&~9ougzp|^G)M<3H1GtC}hnQxiKue9Vw_c2QCWrj^aZOFa|-gXo2ga4CO#L z;6-%Pzpy^m@C3?U9fwd2Ur-)FMr#BPrF(r}MIx*_ra+|Tr)|MPnOByfNSEQOC+W9O zrF$7jFpCq0a<5P5H!=qMzrl=$V>M7WFo*bRn{Y3VLvgJQ1F^JPqr&7SVG#*p-;n`{ z%}3eMi0lMxm+Zbl@!rD{;*Ve+jAo2DoGEVfjxk5AkC5Fi$Vulkgj-_O__KM z+=VhP69;bfI*Jsvdb=SND*uKL_-w&+6sNdqm|qn77MtHtNc5smKW1FrV!T{1fmfxP zjDK9h9y3ZXUCK-$(q%LjjD^!&K+T}n1L%bTIb|~aCE1ho`^V%j9JDG%5)y-q2rN=a zZ}H!1ddmY4tofNgCws^%s^^Ujm z2X}nf8u0z%;VzD5fSag{GuHe?><`c*tDuOC`!Sl5)rWl3YdC1Tfl5q4S zFQl9j2Gq8sdRJLGqe^Hib zsor}@`CEIDa+LrVD$Xas2*o*&ijy`^wddM51f?dOT zca461g z;;2YA0gm4q-^H-bo3!^JYSYw7P%s6arq)UUi_==3;PJ!Y* zkRW9~%_7DK5vX+)ln97BFkiNKH6bPr?^xoNs90lLL)2h8cs1uc^lMo~{&laUv;z7$;o> z-yMvP5&+<$g(a>rqK`{-lR)X`_-QO2<0Wu(;bF1 zd*agjxyY!Q`3NQyI?^c@{%}|`be1UQ3Ppk6(ic>#XfRdkCG9WC|m zp7Rc8J{|SpIUk(U*>kO&vw+Saiz`hS-+1huahK`7KpZWQ?A9Zi8J-j57M)^BQDOFm zC5~@A<{SD-zvH}c@f*Ix8~yO5q)Xi8gwc)1#?m3yY4Ko?V6?;Y481>I`{{%S841-5 z3aBv4y>3sTD{*w=F*r+rS;iYxh)Xz#V~lX-PTe+-Nw~9dyXbW4wnipcNPK_c(bp*p zi~IIybMJ-(wqD4(IMALm> zE~t0IUAevTMShQ(Vl>}jHdSxqJ2h3e<58JHr2Yl36Gy53Bn2F?cew{03Y}3u3^z$o z2a^}~Bd!j^iydgOC|K%FXtJRYu}t(gui=vm`>WPwf8%s;`x_L;rFSAX%6yICc)^^nocRc(iO2|MlCrV0;2u2S!~ekw7Zt?26OTHl6+sv0^jR3u zmMu;QA)y$pMkjV{Job*i{O)c{3}z668HXTW-|yaet{m=@o{U2(PK-Dw-ip+Ym+o|Xc7-Oo8;`j_U%|q2HP|_Q zB|OtNx>Vln#qYyo#Rq7meHt4-tSBA(}xATPd}-0M{n)v?Ytn{PPGA(-ps^Z6E?^@kK%qTJtTFY zx3SS90p1BP3cs&r$*}sMC$DkP*gLRAX;J=C~Twm#+E@J{WKk>?0uOJ z0Zr32+!2>S5fC>#Ohey+SIsqDYmCoia)$S@sg=u@Ha4@(@ic6d&rGLCn2&_6E;)qMSeGoTtQfPPxy zsWGoo8@UM{iv2ef9MSY$dX=ee+(|ZAcjAtcBluV34(S!27D6{%?Jx%AR>RQBInx6g z=WIIb5cg<%3PUAEiC8pnde-5wlmp8p7y?=D{!{eOc-=_`k^W?qv_Z|6|IQZY%nB^~ z8u}p1m=yhZ@%<=4&Hp<3-D0|RP}Xo%E2_vJT%n5WxJHF6Yx0?=?KPRV?bvJbcKX>~ zljngcRn!cs!lF7i!iKsu{a9wBT><6W&Kps#Y#lGL*KrCN`m|EqcSNYqAw=o5aolsG zU&-s^|z*VW^Ck6$)_+w*#$?BSR0xD4@zMnT?K! z{J<})Jz`l=jf!$Bpk8B4_XBHPuy0c3_TJjhwD(SUCY0awg0o~M9+O*~<-~(oVqcGcdGku||6Lr^z z3OvClbakUzQqv=Ke(~ly4`ZEQmd#_Q82wMr_{Krb>GT$HIiN-mUkIPh)WV;PWd-C%sCIG(EJYyU~$3fVXu-RBk1A>5lw;xj)-#THfQ6 zho$K$HN!gT88yPt1&QcW>dXE}7-!r3Tb>q=XJQllmas$$1 z8T8;e6^?|V#zqtl=2y;X-C$lxH}0-V($T+`binQc_&kQ*tFH2u+oNK5M7rQ^b3|%P zmm^RN+VhY`gM?{iTaj`ZzQ?Inh%Svz+AA{F?xVSz=)!Wk(vuhQ#5P_+Ps=fb`>1JV z9Wk(M=x(&1%hS2BTAJ?B@0n>;mpn^^C_AI^k~5&g|l;c<90~Rx=~|8&`ty?% z03gQy-Ff7XeGqu5BLfYCel?-a2YADWVrT2|37-ive!SP5;nfrnZ)wO7-$zJ;7b5>3 z7SN!7{cUQ{Uz~3Z`aeP1Tjvq5Ltfmg@fb!u2Pl?L%Rhtqy8(3G+Far1|&24r$mAq&zR@rHd3iCsd?d z#dE&Ee_E<)rm9Ol%E|kRl&knVfG1V$Q6B?pY}q($ zJfBWCIB+D!K#^+x&?=_kzY0oJnQkVxN{ZV;!R^^@_{35s+bByjp|9InK$Ek?(HYMC zSS0x8El31;N7Ft*nUM}jJZwz4m#SK7j9!w0#rT)&&qTI_K8>Bz`BfeLSJ9p_y76g4 zcesO7nuSTU(IaOxt4?E*G+9CfSR8voCXqUI$zo0Hoh0* z_L)BXLRZiwlUvddw#&&hw1*jq?x>vmNiyl}j;Cidr^W~&rCC(k??1y|@QJ6yo~*z+ zi`HmgO32O>w6gS|BuxE8Y)a88>~$q(p-rpJO?K~?w4p`r;PfcQ%fbmQTc^9IaK5#Q zy4N)2mG4efMLkNSC_GrJKG)BA_&|8@%`9r)s5;qeoopNEBLnDlpsBCND4%mH&nzh& zoD+@tWpHkUUQSEYs*erTyjpcME;JIP8i%o8^#nZsK|R2CnZ?E&$>g5A+3H6G6$U3X z8MLV`%89N4OlWc^(iD5k)-O4q-OgnaS(l#;z{m;lIJ66YV*FbhbNG?5IoR3L1IxyU zne@HyPsF<0XXC~StuP364Quexh=5@23$-oQ^nnWZSbbM%_gHv)1weA` zQG%osM+zK9Z5(CIAqO<#4x=)C^F_Xm+C=e`!`SEJ)+1+btQ%Ci*X?cW#CR_(u|*cX zJ7+z_CC6$);gSl-_#l2TM(k#ykw|-+B;kLz*U^zbnzW+d)A765F5vpUv=r^U=1U7f zp4LoIU*v2+IWOXfW)Tpdzcd8VwZ~BM5L$|}Sk2U6Gu6*b4KP#WD}th{nMyZPDQ2pz znTj`4&COI}Gj*4la+#@`W~z#rs%)mh%v3oubsgR+w7P7jE}E(HX6hR=b;eAcG*idS z)JJA&ubJ9mrZ$@?sxLzBTV`stnOb3{UN%$n&D0z-^^BRyH&c_$RGyg{W2PQ6Q^U>F zU^CUvO!Y8Roz0ZnOtmvpZOl|lGZky5?lDvK%~V}8RozV8YNjI0RCzNMWTvi7QK+Dx4=Q%B9z0WIE}3+e~@Q zlt-q#O-`?`YhF*^b>MCup5fDKv&_h-2#jWXlGcDEh$5rKxg!f53-OJ*Gvjyzb7T~B zvAS4(n_b5o6s?@5HOVl=snR@7HRLt z12nAbRRd7^UV5ABWi;IAXPo(2l)w?I1vH}aObo)ROIxHs>B8N*Va8Jj5uWFZi=74Z zCLx;eR+#9xeVMVwr0aCs8{alH)WtFB2%+)|R!Ka=DnDYXJPB&5HNloj<$@u&bmmAH zb^o-mB1EH=HrhbClyDSfi4OMlS`12@C80LTTJw)ICDgV__;|bor5;Hb1qq6~F$D4? z>t2F&7s2`@*?^+L*!mV${t1)yPOwwTk5{aV27~o&%o@rUvVYQ)8*63gX@ltHxq^Lo z5d$Hfh4@|^0~F;-!s3Z2ed9GM{U>AGrI-my)O%nT97aI7SO)>lRsS$?*5M35;gy<< zLK92XJ7G1h?KD-bYE$)4oCR||+rJGkb$%cy9qB}UFIe_gM<9R9;Q@ZD&i9V)rihd$PKTJ(SZ`OsPP5r=^ zc++%T4-F#-x^lS57M~BvXtcwMzuxqG%$%B9=`Ms$(mcCeDT$vtJ(tLMn)a7$&%{NS zGr#5Uc-<)jyEBqWwURJSW>U1N9cCNtV<;k$u4jZ*&qPoMaSgJI=IRurXcy&9(4uk% z5h!AgK-~L?U`ldSy7TRn@F)<6SI@v)FYWhiT+Woz9HadXFj5qJx}Fjaj$N@HOo(U1 z;tQfq?x;rTQNc1MJ4;5RpiJp*ilOgCSQZL<^|T3oHus@87;>{~jo`_|DpA-ioz2YihAONBOr`q+G_y{?Gx- zu83EWZ&MG$*M%oX8JGH2L%~9c3+&a$xmkz2bj)|W=yMF z)~tcfn*S%$nqYianEIcTn1+!(-MP<8k1<|)PgXRq7E=;^)9?XrrI;4V3(s_w;U*L! z#77D7FA}Dru^7SPD_{=*zGFNv4I-8J)=Ip$AH-9ng*R1{?5d!8CAVPfr(N2<=yn#{ zR?IfsWZTHb_Ukdys^-LYqO8g|4Bc}r1TbF1nkDZ(N87>s9kB4O$;j5OSmPiAg3uP#U|+bXlZ6A5;UI$&0>r6V2oyR;yhPEa_(hU6sF$^({t*WTby!m(X`w^ zl!d5>81hkU>VT1OTQp&C4h1_Mm;-}WLbV+Pg}|T&RMGs^P1V*w9fiRVrP{ha=&QyeWB&B zdthuOY`@KDw2SjmmYNjUr2Hl5RX$bu`H81Em_i`g&itoQ7M@~sxoaW9b2Y?SunbSq z5}%54de(4FJ%Dx3DfrQ?|70w$$`L}^+42@Dh35BK@Ev`E%Lt{_6sh7rK&;=u;HI8_ zk=|)NVBB{Bx9TK*I2OyH21Xmiv$Qbr#+Y+)mr?WSE9DyWGu^#IF_e%G-HBJ6o_k0O zpQAF%n-6*7&Pr^Ob{4DyksetAo^SMe0ow*KI;De5_M=v4{xz0v52F~1Y5P1Ka|`xG zM#Eq1gEkuEMPP1%pD0;dMbl`)clkSpR#3Nv<9+94&yom6eW^QjDJb=HIM{#tG@LUVa zspYvAo>SFxtzynCo@P)O%6wH2-nS)Nr%> z`#-K~Yw=^er|a{|TvKn0GWT@FU@He|YZk+&k3A1kTil#=}lCtA2W0@yY zznZC^&Ct(h&}H>Es3Hu8Kli}Du1F6^Xi;G$nj2gT@y80;+_bF9%?(`(G4w1CGzJK* zCOXX~v@#UODVb5KZ)^_rRTEklX>W^lU5&S>l_KRT>5Ins zF1WHAb}g?PGzEf;8!Ktgpo8A)O7kMcZxYd2uF=f;s>S-EU_*Vr$=WShqdCotB7Nv_~iA(6#bG0R~tSjv1H;{ zUuYwQWnXNx>*$q?p%>HjD!8TKBicD;d~g!C#OMaMJbGj{xFI8D!21x$5MwMB86dDb z{aEnhcvMoTq3LHzYT)anO5&?;RB%any50s$2l$oFf1XbI?0`O~d_B-eI?9p8FY_Tw zK(r6N@OMup{tLz740m`Y#|Pzx%da`NX6VD}4^GpsXW;V7tm3sm4<%I$2RA;#i?ebm zKa2d3&xgH*M-^%ODrEEs?1`8DxaCo*67jdFDv^rss!CKsnkz%Ut~i~DYh zcL)X>M(1fELFivo>i0mM>IVOZ3TO>_qm=_nSs2(dJmQtY0@5i1%iodrBL(=fVG!!K1v8bdpgI1HGd1JNG6HucI zEdLSh8Wb2s3izUDD*$k(4RpYG(=4n!Xrf#@7HRg6$(c%x`EEr%Dn92-dT;qAc`eDZ1 zJC2)lS80!!7yw^0ZU?=umWqd|d2a8FDd1;3k0LGUgwE47+3gDj1ZP7L zI|#*o5`@R_(iuWJYX1|qa=A-)&P=(b6x%0$L6{{b#ybu@Nop*F8$Zp0&IgR|@qq$R z;JZCP#%i7)f}P%#9QTMk@e+n#s|YopW2Kmy55WF(c*MjC(ld3(`iIFQ9ydK=_%O>O z>LSe^@i@YgTuF;(bNM-H3wDu$-MtpOq0DZi$*!7>-Ql5jov@>Usp+ z2@l!#Q9y+HK8-g*b|G!h?70iva@5uZIvxjmWJLFe zUhpD!fiH;+s3%?NM|6P>P5-H+3iW>FJNSlk#}`U#=Az^CQID7vcUiWD--uT*(h{#@ zS0VX1mVF1H|1}<}`e&&cWcX6*XM;YOofAV9`iw<8K`Xv$s3F&ewbCB-)zWV%;K!aL zyt2=$y{^OJE4OhD;}(2H549ex0lF^Z*c|FP9s0t$L779(qvbNb zkHp>1X(xc>`%?(skW=o&lejSOq}x~JG#&&Q6IKzIB6KFsg5{J2=L&qxBhvjLT*@V( zW+}IMh)Q{bS;|F2*liHneO9Dg#%Z@~p{RwSApv%nU(aczdI^V-afk+0qSYq_M8%xgxrFmggmsWVqc5MNY7~dAt)YI*)GiL#^dmVG}L4_8gmF~s4z4(?k z^b8F9Ol!0^1KTlt>0t;eBLW!TjUs@KA;?kc=)}ZrPU?}PkwP?kc%fq`QYgwGV+jVB zXfSBRdO)C1JD5@UmcS6cJ{GpxU&VWAu;uI1k@mKz4tB`vG7hJrje(=lduFs#j9i*x zt~{NIg=`!^@%?-$o6caymj8UUrHoos&v?vxDio$BlXe{a5F`4sBYhR^!25>!9=Ep- zcAmIhZg1D9GV>M9^AQ~EvrKxI;u@A;z!e)yf6(;b5SJ*RcAP8A@igE+24w@U$e)R) z(EThv)TIs9Nv|jI6ww*vXM+#%M3zVWG%f!U-JIsfwl9TUYr=;o6bm0j6IdR-nC|eu zKySECst2GxK$3zKi8XRCE)6bB&SS$pJHy=AW|jLr)#R6X6j~Mm_s~c$6W4hLAKP4{ zSK_E+d*5+ zPm^+?{h>itnLmd#8-EUOk!kv#Vg%eu^~gWwk(1WB&L56H@ljGCL$C82eIL3r11r0- z!$C)r7IL68dUIFu(&*#JkTJB*R{FqDljd(-e4+id(%qRMJ4Ab!;;pk7Kc^(Ln1qjZ zdsBAZjurv?l-v>RycxS(*z-bZawMAa(6jpK-jrQ2&S$rv1CoPwZ5tIo>O~rur0EkJ zK{RN07Tn4$e;05`MerY-x<+UTT_c;R2oY4lxPFg0jTQF%3(nQ)dy-|cC@@64x0gBmko{1EXI4a}YSV#P)jKw6 z=B~Frf;Pc!-#pP8c-v=V9VHBp^LPgyjbJ=eM8T|k4h7>K;K2AZiu)F-ZLl;sZU>D! z$#9H36gqZbWZzQS8?w5%j1xto(KQZH!Wa1jsv)(vv!B7v(St^OK10yD`+g>EDjDU6QeuXFtyeFNJTXv>t#nBES85Gfvmr8?O;V%-uQ36_w! ztGramhRsJkrC{+!Z~35-4p1!;s)6Hoe=x_}!BMK%11jPab+bl?L9`mN?C2+Jv=d_G zD7%UWy%FB^s0EYZRt|iny8bwpTdl?yDSqFZI{7I1Vsyp1{9@uS?x9z6pDc7-nIoI~ zI2yt47(W7bmx?TaQm9Gqy0smK^}n&ffYiTkSuLmimXlgZyxXC1N5UChFa=1vj zc<_ffF!AYUa6jWc#8t#mf}U3M`xOV>Q68L&=X1SvEa4wNuzrj-wM!-K=y+V?(VeDZ z)Ct}55qaKjizb(pU`#6z4V|b*ow7F%XI+%fZAx*+xKeK(_D@p)GA~(tW=Fa8Kg?IE ztz#ZNrHQ;7U3?3rZzt$m6uyO_8HQSV(kFOgEZkulLR}ONY3(El1#fYmvT31Z}weqOw?WiX^fAE;ief=R%iojixO=$60iUn*V!? zI?fcluQ#gzzx0Td3$-o^JF?+*Ypzi}Hg>g@ueCe#i@ANntL?tXtz=+@>441_ITLBr zURwAuek&j(|KiQiqB(1E*3#lzArT(h9;uQM$a5x$H2uNK^P{Qh?Ubh@@;t+NE?9Z? znR))`g*$;z|J%(De~;6<=_E>f`e&MR!~78osCts_ZnF%N~=VPKDk4llsE z)#8pJ0slIsh6Ig~|F)!s1ckdpf}X_Z0{pW#Fv-1eJ1O-$n2^BfcA;Pj^En8IXFQ-X$ya$RdDi z)S3Wyg)W1sMWUDH$6(T_`Tw-oG%(A1vdJb=vMI)rA_ia6h||%h1*blWQw(z|wK$zJ z6>4E}+TDZAb`@7(Z}WUU&VD525MzuvGIqAW-O=2;QpWE9@?8H>xI5J0K*8bxc*jdSpwM{zmi!g zZYXQ^@_?V-0+Z6l3tWNAalsq743Sd!A_6_~JPbA(5FaR%tvF3!PT>}(rY0w^$*G#+ zL@{zb@)hEAYP{f-r#LldPL(W9-wHIL)BPr=gAd43f08*3B2IH9CwiTfJ<&ik1q=HY zr$r{GOM;%-KgH>8=2V?Hr7|b6(_S%K#LS!)v(6^7)h4r+irE(cTaP>p42)|SEmLXp zE;*@LXJ&Sb#Vpul*571S+)dgzo0&}~W^Xbx3@J%1s&qy2a$QRC1!Z zZF^!fTw52e8;f> z$L=TC#|iesEcV;L0#g^$pDfsqw%G3n3tXsU zn%vLE{#M0)h-5zu?EkaLZf|zHvGftL*9(u8_P&L&ChUFt2e$Vtd7rR1n#;gaXkUh9 z?^LAO-Xn3s-s3%$y=`$Jpv~TKf_+!PeyYX(bFjcUC8oV2Z0zq;>_5Jrg!rlg40G2|Zb+YW8i!|GNwz;tP>mI`114d^GSzB%O z48eY|U_aMle;zDw61{2fQ#SU^75k}@eLmR#Z}x6~nCyKAWhd=Dgh(IkJ&euFus6t3 zs09~*rBHFYW$$vN+1`uIguTCZSN0C#=(gEvEfDOp1^eaX)&kCJV4?Z%GwprG#=f0m zKUcD!2ljt!@6B2*_UFUH$lec8cGBK62!O!eFN@jUC`+LZTmY6rSJNze-$9z~4U!6l znD&l37*M{i3HDP2`*+B#g}oKo-d3i)FWJ~Xpx7^u6>z@t!2fFRmqW?k&ro*K-k%UY zfxQbW;NOVVYlyxwetB#H3^WBNk`>^y)bfqelXIsiK#yQEQZ3dzBR-Pd88-R^a% zmfcxMBbUqAbg!^`b62uk09)@s0I&mufEGf)a7(~;Fu|?{6WA^`0Tq;hUQ$3`7VtO2 zx|>xUbBEpkPvG7?guso1!$@%B5e5Xf3EwfeBd9hBaF>>`y?aLs+)79@xc!<4a0hm= zz}>qq0NheUWFX+=w=5vs67Ue1X#Nf+xW9C;1iX$k3%J@?2)N$)AH3G)Xy(x3AHDW| z<&p5(`>b>y#9-hnD3ZeD!KF};(>AYFd2L>M# zS8!AoA`Qi_WsvOx+$aj_Tcd|GA>d9Spt~jD4KTr_cqX`QYyy63Ck3>Z0y?sQ|K22+ zv!VZ2?d??M#Wv6VYkNz+jYB+P5Y87JNH{lu|45u`B9;T3YkkEyXHXp$I8QADLkpO$ zx||<12}7H^1L|Hg@f(Xjt>3t`83Wq-nXCoq&auScj8IB&rITr?Knkk<(qYLtQDV6p{6 z0n+TxA2$>*oJdn+!*#5`6#HDsJ`e2wc5HaF zxW(VL!G+0gqA{NDM|k*QtHfgkLMy;y<(G`d$K-wjk8l*dUEyu_TX^Im&3Np=X@XS0 z3*8Ej=DP!G_ejA$MX=Ac*dGE5&5zbb;xWj^zN%vXuwIqC{cTkvI z+aBN}ae{q>V4q~MpAQzA9}x}dBX`-@pGlOMG?DCM!TxVCx!FMZU-?L#{#!lQ{&c*30mkI5+tkMkb{;89Pozy3M1Z(^~Z4i=i9 z&U}P848f|}*dJ;w@u(x&yTJZGd`v|cATTK4* zcwe=v4SNtC52O4g9xV}x0UoV{!dc|o0*}RI6z&Ek6&~SA;XW>bNB?#TkG)$1@F+## z1ME+pWIVzx_78ytE{`+u_$A(A|2oo8!ev~&U9i8d{SSEjYah7{<1DWu6Xh>4xl<@z zPbl4ks*k|LTSn<5o6?tBNu^Vy(y36I_HP_C)^7>Gqy)hcV6x`~V{(Z+QLyg}cAEc5 z6O%JwsUV-P*neABU}7Zy11A4@JmSrE8xgA|YC>7JrR>Bq$~LnpTijA& z5igZZgtCkUtu3viLC=!S0hpXXpc0s@`;;*`N8TtfX$`iTf1HWQey~=UOjBZx))AO| zi~;N4)|;DQ@{fK(xLm-T$LnZ@@|U>WK%4{$mwv(u$65+MTt?vcjS9jLAjvZ=q}h6qH@r&zf79%te|p zIa@F?qL|z+_W`4U;*9NzHcyFzGJDG!|k8T4FYWmFAyq zV$#tj=0;N~CR2*(3Nil~lYiwaBe9tjYfNxXiOW+67y*~34>K-rkdF&o&XrO41t_T+ z!c3*`%T)y~OXEpl@v0?!1|4(}f%E730B~9gSxzA<*^>1l_-X!yCOCK7WSzTLf)guc z#j&jaBP6JHJb2Q$PM~a6`6FaQ<$p+&Kl!XI|1z@Qggj;E!%FtM(X#wo2|KBq6?XDK zBs$Q}&O%m0A*+ui>mBgZ{4bk!YBpI{n@Bs|QdTFH^$&I~LXg_)_!DI%?R*`f7}&Y? zAlvx``Ju4$7Pixp{UPKjJ0~jHA5{@{9&Rb@+`)DlE#3(z$5BE?s*sUq$v6r=xO~X8 zZ-`As4J9K>${5Kq{=vR2sbt^XmVNsXR)Kv7K4SZ>lPd}PnlZMP>~9-e_AN#lxm?B% zw+Z_$v=H`D7l>uTp6deadsfJJM95fZ$@m$3H2(_IzNt1D2};IHDMM!&|6t!KY%KLU zI-qQ1`JG3Y1oo92VEbxYvb!<1mTZSj_O?dSz6dGXDea@~4EE)|9bn&ELdFas<9%{d z=_734D$~B@;H!M3uadF)R)OJKtl|8hjg+c~e_(YAds0v~(!Pp9b|oRZjU{_@8QBeN zvQOV5?Q1M$H)Ywg>QlD_`(Ah}z`jEWs{q5-_A`bj$xWq?fT!kPZQ8dJe3gCUm5icW zgnb9%{)K(D*ghZ1M%s6ekljeg9$?9yTSj&Vo9sXDmiA>x*g z0{aU0v3*y_O@)2kz*F$?aT>yCs+ax7>mQ!FmyP0TTWc|CeHo(4n zg^UV9M!Y3sF8FBvji!Bf*kpXxP}&zGWi(eIS3#b#Z=jO>MrC2&+p)qvS^uH~?dvRLG!!!WSTf!LAG|GM+NarMT)j)$ z=aw=$v5bGPZ;?j!HGuEQ@_QYT7TC9TFWdJ8xv8-47Pill{UPKj`z9*cA4LlL4mTC{ z&5`}z>jC9AO2|kRGV&}LN5Mz)Z#C^3Vv|ur$;gs2MzV~5uy0E;+1CxeC+*vh=n3pQ z@FClGom@!R*Nm~XWPjVhvTrfc$c6qdLfCiVUSS`N(op|;t_iU3Ss~*QA!DH><7e>E z{M$|YrrKmAC>b-Q44q~CgMFv&C;J{p*+~1&BMJihO7^gQwJq7*7+XuW!zO!MeQ95W zlL;fN}d4uz+WTfGy;s0>CIXuE;d*6);r}GC&DfTS)-;Rucjc2eB#OK^d>7-j(N*64y*L`i7^hgB{VjmBqbVO zApQX~PVZ(kDqG67FQe@5^(|%H>kiQBe>X$L%iCzuM8Gq~ONv>~YOfR{+9o< zFQB?w#*jDfu%LPvX@=_EFag!3djQcUZvaGj^O$pj;5@?OydBh<|C|X?cN^zQigQ27 zc_29dEkrjB15lm9X+i(algz`}$eY-yfEGn_jh=~w)(LpGL~9s=JV5K=9gNm|sw4uf z17&QQ?6S~mg*2lzy@Ei?d$(|`9VBDw%mCEdJBRH?)}76GyHME_BSqCzr_XB6`h~k9N5D9LpIkkg90OSp;6@ zJGNLN++Np`vleM@izKk)68X)6qa6kqe@>?|D5CdqXyk&IQk(_U#bblV`nfEnl92KW zIkT|l5;%z$Qt=&9rV?j{7?2?ri#r*RpUfg13ql1*ImM(K zn9!Ds9t}m2!n1L*6ADj*x8m(q+%UtNM)4_on1mJsdjPgkg^q6cWY}gEGHmopnO_*x zS`^Q5vP&2=3=-|I&8%Z-lY_LkMR~A8UYD^gl<+#_#)-f*k}alNDsk^B%mIQNDqUNE z9QeX0;#H!ako?FT#$kuRp|2(Rb#NE2!U`x`+9Y4AEoG*ak@*9IfLCON4uFMP0%HXR z?(|a80}xmh{USWEix6m##lqAA7DzA93V}x$r0#Akn7+EsOe80BsGWtc#!nn8A zV$e;yxGWN~UCsh}H4E~`qW6US!`ryvvn=^LAw#@^E97^x$q!fZ`vl6Tb08#|Iqm2K zXAChq0C8guuGn##=iEky;~6ghi81o{xj+Z6Bdg(uEHk`tpW6vKHs8oPKEH)^+(S^= zF}i3v=V!IJAn>-=BO>K8HkK2v`DlIe#{t^5=$#zR-z3*f*AicnkOdbH~)dkvP)h0InJGQ`I8&E+Ydgj&#NX zz0H?_f}(dg^^>~e8w5Crg`|C9ekAQes|s#5Vkd~DoGXZXR07=_pyM2pZ~`t=XG!D* zLP!za7Wgq%^hhqrBf0bw(LWX__wmUrcRsk=7@Ko z#7r`DIuL`rsr@ry__h;Hu@m78Tj5;8`kPdE0e&IM1)#JeykM?PDv6K?ZwDzk12x1- zYJbo|ND5d<$#*Xa!!epgHjy(V-D;b*^#on0MK^k7S-LKgZW-uE&kKkukX$Kf-8!i0 zi}oX&5h$;kG?8ewfCianA(JHAmtD3vcYwb%@e8cP(d}8ucBdZ~C^s*t60vLsmUM0$ zRnt{Wrm894mnG53fn-A}GTASPPAIZbl57eqVAZnnft1x8Ou5=dd0zmf6nGd>2^&7K zRhVpujqFb>sM(tn5=^ST5J+XRVNNMyI)$z(?mpHLRgk^*NU z7ajiFm83Nclve$hiiD@vM%I(ba2^g71JM%MMFb^m=5@QmWbJHZbtM_@VwR$|2GZ0y z&NMeL%U2qn$9ftWlV{S*45aaY!8Ds~G#e!io(}84-UuCj^HnKI645BQnQUb{e-e58sH!0l5Fq!IMWN}3XlAT%4WQQ@_ zQDlishU^NA&k?+UA&ZblVv@JDB7E(jq6iNVWU06ugf>9#+{3ck+hiTaDw{G&i5n@z zMOfnc?hg?65))6yYTK+PoqG{A{Nz=!q2@O;(dIBpq7UasT=9V1G zrh^povkG}s0h2Ygk^O;{HF9@|vdHEKk}c4g%)rcAN!ugIP$6*&1J%Zd0%?v9WSUoO zG;^3{7S~^stWhA@fQOiDppC4rT~r>4`W;aPn^k#POjE-~(*QK8g!WNxbb{2*sb(h+ zJ>t2?nCuv4zRIefuncF`eYR>=ps0aendVs=&1Ol%Z49f|Cy?gRUQCm2qnR#gP_ra@ z^#f^ERbiU)Hkt<{4XTx-xkxcVt8jpDKIm+U2lrTLWn#?_!!sHk$KTsv`_o z4%5sGr0Lt3XcNw~Ffn+P1gN#}xxcyIelG#?IPnwB=26`)b>){%0Had`is;amVL-VG1> z1ye_*+bAiA>mKVC5h!PIGnVs~ji$Av;kJZzJ4xXY3xj>Fm}Z2HraaSF^_4{}4<0fO#h3Rz#-oMRN|vrt$W#Zsz(1Gfoieu5+o*H@-l5=gVX2Gblt zNI_}#5fMWC@ zPBB@qjjRciseaC){4kJm{+CR-3Be0x*soZDuDkAAg8V~8v6Cr; zEJZX>5Vf_4YMl=tngt@2ITo2oBx;q=RLbeIhqWvATiVh3AZ10Zv5dNiz>W=$okdKx z!bVnCl9?UMHju!mQOF~O9XJ%A-w&&$Af==(C>mQ7S3eG**eoef|18yd2C7zdA**%* z@mYnQN0OncTV!Zr(~lr*K^+aQ7EJCH|W zciJ1E--kmnq5>M67?m87cB^4%P|DYZ8VW(L^0XwKr(-MChKP-`*fASf%`wU zDkqTU(m7VIs*Praq(POHCM5^bO#O~&j=gJ{G|DcAb&Ck3dGiORdDceL+AfD_PSyy3 zA>n7HNw?9I2aPJy*O8m*0QaARr606-pJ~e5XpX%iFhGTs_GAajS-zENc5SfiSt4ne zH|y3WkmmF*rkP};8DW>hG+}`>X@yMF+D6loX{>IRi|A%%U42Q0Nj?75A zNDo}*5?znUjA1N@r==3@a)DCfUS}!ayknUYD`^HZFA{X9S^xz1ivZy=8_o3<1c4Q( zW8@YDlEv5LFjik1*@sNVVL_?h&_J4D4VWg{Ml)B6q5zrcLyZH;-i&3kL+dQ7`jyd; zA-_~Lz_81SOg6(tR?n^>)9ehSxigh%+S_O@EEixX4Vmm2NrtkifV{HYhCNWpY>3{M zWFj7o;UPiaJy6=a16kU8Z(D{QAkCR)m}aSsW-Zgqs;@9)vW|gd4d*ji z9~)T?$W$$?PPt{kLSQO-IrQ3(FK5cCHpLV4}Lx%S&tp%{21@X`+7u4d`7k zvKbQ^yO_o*XeJvNNLKkOi@LDZ5~WKrRav@HWw|SmtU_o6p}E>d*2AtL)BJL4fK~OJ zOf$?z<0_*elWhqkYh07b>e$GN7Yi`ZjOLr`eEET7EgCS{r>|Rvt+&YJ@{7$G zz`#b+v>nszSYuiB#mfQ=7D28^a>oUdo$SeE<85TCBpEt>3&Z3L>XR;nPvWqW~o|ZsW+%p3yAKlB86>XFonNp~z+NuXY1t5;UYPMBV zUL%mq8PilIf_93+B`D@v6mM+_py(pSkYb9cq%zfkgR&eVkFaiS!9vY0YJ!Nbs$eb0lztyC6h4f3ou%Isfn;5)GFdkp*hX^t%saWC04~;4-m_e>6hvb3p~I-%XCQ%WPM;zJQuiqo6-Nc2A^(hT4H^ z2KGNMfD54>FA|;vS+kUS@O@YgZ`0@sCBHIoZ)gzz1 zQ76b4Usr6DqMN-xT*1tr70kE4WHO&}m6$g)nSTt1iuqv0{OvQ$yyRg|(i7lZnmd4A z4=plUr3GVW;6OaWjy=RO94&>IhC)mqON@_gXl9B@u!*^PS=!(hV$^;Lqi87AAhFE> z+kbMxNahClkoGW`50}V?%j^%mcvx_Gd>gvns+DIe?+Fj0`-%=5qwrG+E;8zkHZ=0w zHGtK)0bgzKCen1_5AJO=YS1HGaApuwT^yvY4qL}1uIi+9aq2(cCnB#B8HX(up!i(B zn%lb2F&y8}MFtt(OE+j|Bi(h;a)em`hhaUA7{!i1lp{*_*GqMRiuWSFCyDvw-c{%r zsaU*8G>sZ0LKMts<6L`uAbP9j$895StR#D!IFEm_sIs_`Gj5T<&3&fd@})U z!Sl?a{F;@2g(yyl7yuE}KH>})q~O~PUtr%(UP$$a-L~f!@T>!+zE0(gnrCm6>bu3Y zC?2|f!G4{>N6FoBk8iRksTOu*1nHj)s#s)V$R>a-#hwTWY&MuNuueE8cE6p(!R@$?-I@P9GI*{ya;PY4F>j`W!YLPAqzZON zqxq{dpIpRf`8j)ZPB+?o=zoyQh^`Cj%v@ixxXu;4fZ+Fn7o2W6QsB7jIp%l(z9*gT z>}NMhc~_M2>S?%NQ7PT_(f{WDwiX*9q`xgj5-0~5JAO9(?eXEJzm_11b>3+Bw+ zdv-tj?6Xh%Vkc643eiM!1_+pvODY3}E{HqvV@S0X+`KnJNIMM3H3FpZnY?Ti6a!Sv z0IaMXR_2Pk^aR9hnnG$g{yikroM4`R56&P%tBjv;H$el^|$w=bsuouF0?si z-5qdiU2 zX_JLtc@jAWb$Iq8afX^u^_Be+ukgwTUNgPwz0c+J43Ug)tnFFNyFzbCab=)`D~b}r zGg#>lea~I3A<%IK^~Yktz@Sg;2~F&s^#pVCZ@&HpQ;s#?AuB3GZ!dkRR4R3m!2{{7 zm_x~zd`B1Jlhb5Y*D8q82vjr_+UlC+XXGGaHH&{B}eN;<=%tnzIw%> z=)!%`eRYF(O6L->3PRSG_6fCINV5_46&E*r=URVdUD0a7jWR#WIncZP`))za@zRCu znh6bD1(^M1ai{2kdd0!$fz;xoq6Zoj2cid#Dro}LMTbNtB_}K05r`jF%Y8@7LUNI1 z_M}$h-m7U~ZLA46cd08P!7MFW=rG-~YkwB2CLH^_jiTDmGrxHD*Yhe*%s>AsV&3D4 z`6#}N{oN-J#Qw_M`U@X+?QfhsGxqnxFvtFG$5LGT`-%%tT>I;Or`X?LzWu!@y3?4d z_7~Uw$|+%gXP2w}eSj@E_P27qXMgw6IF(!a1m|+k1|`QtC!x`XCJH*-wW&Z{dB`#i zr@y#?qRL}djpx~N;DQ76Q@uvA){us0J5YmXdkz#COAKtoi8xwKuOjMATKsegSF9ZE1$3 z6?Qrn?n-y!vA%SFDHFmk_6hHX`V6jTfo-s~rG#|9x0JLiAkC5PjbO|8M(|yjh)=pJ z{&ADqKlkHuBD|44r6JP0H)n=voBm(tg*)s`xTmgK|;81OGO;ReQ{)Zdc0o{)&-%{87FC& z_#=bEaQqs-!UkS56ka72J{R1??yWU;Zw-peo|YQd`Zc~aa>i8JN{y$b27!J_d+rd8 z?oTdm8{OZaq@H*ljs3eT#_Rn0PkQxtRMYgX)@rHZt6C*Z2!hd8jH9Vh(=NKdeu?}x zM}NCPemg}kkgc+NPgJ1|=IM?4^kyGN9gnM_;4oHkA|~LovPe({MovEw1?4mGB3Ykg#ne zycMZ965g{)`Kz7*6cP!~@KIQv$85S>gPbe51=&@>{gwh-zRLb9GH-%FzRu(;ceDz5 z@*B>|6b90O_9L@uIeq&!TLdi31I+@Ka+QsfVE zUHgu*2t)0=7yMIJ$;`c1CDRd9o39YbTq$n}ZSlB|w9Lo@q;r3ACG(^yg=%_nFn+XsiYc4>Vl%#>b(W8> zlhu4Z3nfZ!h+S00r(Mr5Cf#(#0BxeB253)_l^B^7_zsNWc!T5ICR&&8YOZwCzx@DR z&8N%S3u&J1p}}jO{#B-Om2q@ZYTaP|cVbBc4s)C)JoEfz2C!8AqbmdHlvVyi%T@jo zwcp)e$^4V<9bLX8)EY9|E1BncWDbNhm&|UID+6D$JVO4I)Vn`3q^dE=)udMM)aZef z;>OVf$;ILx^g%-S7#01o`QZwJKa*)W9CGOHh?GD=(^Vezh+%ztsDmEg;r6 zy@`|Jn=W`96S$eOihRyp9t)fLiVN)TP!o>d4b2V{w;O>XBFLVlbQY8oF^H7(&o z*$Ua4Dic^JESEJe<$()HGRYT0UuQ_4^;^WY94S++|h}v zk9S#G;WG0AWo2ed%wwjJG9zm}P->`CXgW4V4dLRy%9@q7s**yIv!%$>|N z^OKJVIV1-V&FLbrB&f)8X)K!+&hKAPdtSg15M;`j_hID5{)hk!5rug&b&bB=Z%J=?A>=@PTig zVF2Sl0U`sHaW05A?o>8N{1269{K3%L-p)YfCY0|E+A2~Qw2?ER5lTA$!il2xqy16w zy)audT*f9=5gt0Z_!4o`9ZYOH;ZH>)_EM!4kMk>@4ZTJ_rBGUPo@+0>dkMMKaJ-5C z?gb$6-=##~b}Y3p-GR5M_&ZMbOt&}p?mBY-$KzGp!DxbWqnK!qC$MUR_}EOZ)aQ#z z`8?8&j+8ePc~Tw=)^PkbPs$H&M#}Fazi3V~0V7gw`pr8X%-cwE`8O)btSKtVUD&hZ1nO6Kl3WR~{}vaxQcaX+C2JiV zn~hUcX>KM7=SVZw%ai6A&=8K#V!UH$>Ix-#F?%%UGXWyfT;PJ(U+PNpb$P}gLO2Cr zD$S43x@)t+x<;B3OGisYN)P%&s_#50WeTx5#te+yK3c!-O45s`0{H+ zX!?Is`p29_uRrVRoW&c4MmcwcLrGmIoO=Ou!*EO>s5$4EQV5f7GFcUI9hOaGaXb07 z=o*=bEAA^kxNbKUgG`U!q=29F0e@|P;1}C@fVXc_?S6Et0&S{5K^d~oHGs{`#V?4@ zs!SNTxr%i@op{w7e}(&oFAw(%>o}}SMr}{4x~Gi6@~y4;Svi#~fDn)M5ud94@5tsm z+P8Xo+E0hZaQxMyT{hfi@% zNj|yJ7O_DH>5tK37V7>);w7Yc!|@Lx+3~u=pxLN$!A7WmN+7580vwu}*`*HCdUc9l zq+(O8_#PE&7_OP2BM5SPS{#Z(9A6k4lin`hS{k;#C=}%KU&yY9Q6e|-pR>s=GW+6V z{uz7*b3Ub-bHY|-jZC1Pq%Y3(U;Nc0v74zvlmBvq8pZLG)F^7P4X4RB{nTsnn-Hs2 zQCQ9Zy}%mSuD}|dSzwLNEwFwF4-F3q4-OCVhnp+uFIz-&enK-z#l7Qt5SPmD!r$t0 zt_6et6EoH!-tz7#_9FR7b2A}gxE2-qw%uc|=m(Kk2A&Wo{JGJ63)3yONSsk~`^WL} zt`ppipseP`3Q$==AmfpP-A#ONCQOFNY@mqS3cQR2!tv(3Z-iL&mETE!K)EuI=8(bj zkbUqIpAM-dCUb)mUCG54h*uJSdh2Tg-!q+r=H$ z|72iao3e6VXN^qBn99J@d@2+3(I8A*0Tb;F6R)6IMh1`3>K!JwL1;LBiN{2}1SWQU ziGAesm8CbrK(@m`VkDGTghQz#;^8ge`{Nkr7;_{F>b(09XXCP2=SZ0 ze4*^nMBXtaamZ_oL_$=V|mF7}leIXs$UGd|SYR>IqEmhc| z0_)4*X}Q!)!C-qt!nF${v6Q1C(Z#9x*?F1c{u8m_AZ#to%FCS4EK=TM*vQ~%`J}aC z=kjKBQL4POi)2p`j17opCYR*N9DkMl`kg6ao);m`u{5r5JekEfjyR_o(IL*q*1O^y zZc3r*IT#1{M2UF9F`Br_6>l;mv3)!GlvJOqPckGKD|%XX#M+hdja_x7Y-spi;!;uR z6b+gd`!rYvA2g(SG;CRCY!#LEH3r!e&7#s%=Npx_B4*dfm5d&;%(oVcNe#2#e6&6i zv|VIvVOaWU1EYah2TcH{vH-@z@nzn!q{l(%Hk7L!16n`7YfNKQ@`!HDDe$ zo7uIVy=QiPF#=f;wjMGe`kDaahoL0|}?_qyw`$jyEx#7)ETeuWH1uLo)r-^sG8K_hK>7 zWRx0?x0iW-BbR<2_+k%yCkH%&TyG}5=IHc>ecK0q_AodvT&Hwl@dW_A1G$9b**?(y z1^_gl<^iook- z3w|2lk!_lgZ>_SE_G(Wia03SJO9t*VAMQbg+sA|3Gy(3$RgPHBF>pUIJ=G*U2*ler z9}MU^t~GY$4E6B^Aew?ikWY?5bi9w~CPg&BL-aDhUHRPPB3kgCFi_LKpu8!werxPg zEd?x7UdaaVk7lmr6M=6ebEyh`kO%nE1i;6;fUi-&(u`Tqi?QMGib+PSw#PJ270=Y) zYP^hh!ILpo8JHq?e7oMi(zWXfN?2A$2ypFs1i0;FFlbg>J4!c#bADmKMDJ-bk%teM zqLVU{N|=bA)Gn!{0dxIzi<2`JR%k@4-5dc*s?`~<|20Q@g|*Mg3b!b3*d=3Km|iEk zB()-+B^deHoij^nBC%7_Wdf}YF*G0N6j(h7!>`DVwA>UttyKDc6jvvbmC~?ecrBF| zrG~TDj{8O4?bp*PAWT?MoIxd`(iO$$86Y`@Emu&fAS~iL^Vc|0aJnKNrFZ%NsW#3sl~HNq%|f#rB0;WN572 z+$7d=U{bf6)NzGBqtOUegyWY{8AajB`d0ZSbqZyE4cVjL(JmWFC1lNf9r#l#(VSk% zj1jEg<$TL9MOsmkJ%}kZ=#cX*H+3QgsD+Xye)c*B8myrWL>U_9d?qx#U}&Nvc#!XE z=Al^5&bmz1eDk*02?inVRU=bkHnCyxXhrtHI^u40NhGsmf$|*X{V9_~uJor#)+!zu znHs!0+$RM_E!JFiy5_C$QV6T*QyLYxG*+3QzE>{5&q@I@>?=(M@#HjlNzbaYbP==aoi8LX;Ty*_$b^Wh6aYH3w)w1C2EOBRK*HkAr@sN>g%^e z0DEgj5T>gGjKn?~1s?nK2P8-c#-u)2I?w{a^1MmyuGAS94W2GnEL9$s!3kK-amvwdZH#p%D+gVH`(0n23c z;Yws>whoyR^s2=pl3}9dRA;?*<)P+Y{)?j8+h$U#*qdaeVjVpncLrpI+sX2bKY*PAFq8knCZd0_-fJTd5X@h^?XSVjGQv58 zPQ&p7QWc#!efyu4u53Ovr6clL$@a8uCZDB?O3DKhuNEkWTqvAp0ole}R|o{nXN@FHBEV@z^Gszyf>`IB z<3+XVCq{Y5_bk)&MlXYSls`vvkwI=c=zWMxlr2h697~E9aq|m*9rSehD@9dA6LO)B zE9QZ#!*Q{AL(Q$wP%X;=qB*O@AVimoU32*MfUC>7@=UlX4-^4Rb-80M39S@< zL_B%Fk{2P0kQ~rZmCp^ne2(?xGX`wB0ynxt9jxR&c9KVA*3f@_s1Xn@wGOWvJyk7L zQ4N}=I?AEh7`J90^@fJzvXp)eZ3BBpMgvVM=E&8}PCi|NK7V~iZpN>y%V3=q1GlW! zRev{6Dj8WB_>@nT;@A6C1QOo&&!RE5ew8QT_zyn0DTYYMZ3B^p+*1s>Uz$mmdzf@_ z$Q{VIk_1h)40~9bz=jJeuT`vDD~7I2?{sTtdYWipX09a*%xPI0&B{yH-Ib5kz= zA~-DxOfDbURDIQ2qmies6A2zgOzRk1mLUX^K3#%(N70K!vPS0?=URO-@%MLNi}vYt z>m`xN18J^|?$0Hr8aWFxa+zmme7`VarPP)2Hz1EGesyyJrekjW+ZcK%e3-^9?s(-0 z)kSko62o`a)$A`BUm2+9n*ZthRB^*l+*9qHuJdx4g+W7%fiFccwD#Q?lxYYzSBPF^ zXl+z@1>fDS^ZmbI?U~n{uJdrhkmu3Zx9LAWJ`iyGPl6? zYmlL)Q(R>9pWiQ5D+<|dK}bCP9&*bU|Gq);CLK(ENP^wdXa)GK=?40_KJ*R>J;Q@O zABf5d&~JC4fBvclyH|*|1ay_%*7UI=u{l^QpwBX(SBn`N!naVJ@u3HLXaeYST+kB~ zv~+$<6gA#S`;eU2!kr?U${GglnpXWc6}$c{?*M9*d?~5M3n^<;?6pTVDyhbm7b$%Y znJI`iK7F?+eM3F^W-sztzmqca`TLD0gP)MS@)dOIB}}yus{3#wG?$*)Y#?&h`<7Ew za^6~~{#88QV90Rqtj=nL`qgc3l3GA`EqaelDh|3fu19QrPR0S!~L1i3MqZIDA- zyAx~T2Ph5jLpVOi2nnP7^;5sC6$ro!g-&aWZz*25YF&zJ43X(^OOS=5Z zh((fMe{J#UtRD}~-4r1Qntfng^}o<6lKz*UtxGfU?#BW-7XZx!Km=D`vdbB+Ij!T! zzFrjK+GE2%J$wB7567ND3p{&Vb%7^U_3yeK(-Q(EfmlV_A;W!KAT?@FzP=VFm z`w6U{&#Xw@TN=>M8VsgHH{>i z=x#M#)IHX;qM&wR&+NrhK9zy{5hVKGMGmB=jP?J+|D1=l8+o!W=XWuOG}r#ynFw9X z;f=Op4u9=7W*m+`>$BI-NKthA5$s)9D4h}A-)cM8pAKrQLXms(9HF%GYTZaJV-cB@ z(85+G-!bafP=;d>w>(%)#fsasvz+r~K20npbJ*x!P^pR~aV@;Z~kRgWJ*;f&^ z_YX>MQHf_1Ct%y8I2lJsKi8a=8;*bGqfPbD_5)eXog|)&Vh@jzgmwMBMS5F@F3^wTs-*q)|7cGWKRJX zA?rK@m~X8i&8#UyXgtGqiy><4C&fvPIw0pO42I> z7g5mqMK#(Gj>%XU3#9WS94|eO-gzD~gCnqLw|D;VTfcYy0+Fy`2j~vE-G*KCw-iLi zVs4^@+(zB3>e!0-oObf&le7~PERM%z_`P!rSufIvsOW!N;{5OJy(hbdaN^a{FAm>j z7PMW=Z>GJI?RU}^^|7^pk%|2nS*@|}-ds0vf76TD4EGiL;#y*agt#6SgfeO;O?d*HUCFV)EYa!Q zr+B~G221o%D0t^OL5CHku~B5t)LRHZ zp7Il?snESyUllO8yM2qwBm<;Dh4zk*v@@tM1-P8`9@Zz5>!5|R0K9qE@^^hYCsD2pY+!eo{3(qK9>)dH zgB?zjlO+~k*%O>@wi3KAsYFWrx!6hIGlJ{}bV@jwV@xSjlh15@p|y;)5e(yZf2-7s zH;gEhTYM<*ni{C1jt8ac9fcP{vds*eSqPYxf0WYvKMtzh&X-T8J^G!{Vq2S{%2$`~ zzR3^Th(0In=(NMBXL}Mo0gA%$hdqgALznK!i{{+B-D!tW3JT(>pWSviR-W+(yT<@b z(}! zRlV)%qKpy}{B4XH%HA71Pq*Yz&pKmIBv_|6?DRVTGq>g>q+EYL`9JU{;}ez>j-jmV z-%CEykzmKOJPCdY4dFP`x{jsne+vmNKsM2wp4E;7!xR+6pr2d`wvlH>f@{(p3EqhA z#gmBCj#d5}0@ktp7mdYWMSHjlai!$E4%%z>fASl!|ZEH$My+ z?)o_LuV&QdHxQSe*MFjt>Az(!nd)L8NqMLLz5#L^0_rINuXqHU2U5xR%n-EFE0wtB zFzW|bmsdL^!B{eobbS3B7U;HyJzx4g__4J1aQqEl>`RTX5&LPNGGc$@O%?lu>s9Qh zAtgubzdYoL{q4tmvC9Tz23TzLmhq}HAM^oRpPcp3uToT=SCUum|=j& z^TnXAegYXsbME@tVS6eC1@YK6m+jH=j6c|J1Tb|1uVWW3+qGZ7wpr^S7Tt{I8IG^< zId51Uwb6ql>ZnRe@BEDpYDlp4RkQ2 z=816k97I@$3elWyVtZma*{APmQ+_%tpslp(l{uc6xhI@e-mwe8D_b=WF1xliLR!|~~U zz5Bg-&q%$~sCSr>e1Di|-;aEcnB4Pg?&OP__iLo>9Qyj7>Y4XW;L?YK(pW-M!6TRD zMRP`NbIf}X1qE@7>BR}x3*;Gp5WT&*W8N_o?s|xC*NTUbv1)O?Yw|1PH-5EA|BR=%ehN~M20c#8IKIR?QH`2aKho6s4KH%)Z~S@nwF zzkNWN{Fi{{kM~*seXlV@Y;a&jWuDAtC7FQ5oM0mQG*h3tR5ILGRv?pNu(m;*wKprL z3cc$5r>ozuH*0Ir`AR_qlnlOOjb|qic`W5vPBGw=z=4K9J(?ol z+K@G#Il>u8qfjOubupf`rNFWatR)34f06mfCXuYBEykZJi-IG#tK)iFjT#n6`n^G6 z_GiJ{XTxCavT*IinFK0|#QJC(9B10#$bs4hN1&*da2<~KgH?z34#J;u-S=hJ zTuwEmP|bL1T&0tw9s5ftx^OX>`T@`MY;lRLT?FA#Lh*W8ps z=wkj5)rEDH>SEB{=ptd4)Dx>!7_tT7RkTH>+6Vle0RrFyFSvl0Py*;R2I$2KRNGau z0c=cX8A6mX(mz*>B>MBpuSLJke!>s<0&4DqX9~g5@SV*(4Tqp59RGwjjE3`}qxx&! zkLEloW+EEC+clDxH@X_0B+vMRy^{c@5nmj0bT!xh?T4z2H?k^gNZ?n|+B+;PRL8xzhQ6uN9a36_bsa5qOM>Mw5Lg zA@GBPREtZ8dY=FOyFAa||DGc}|9|OB|84x;c@q*M?=SfcO&7M|2S`e?dozoGB80_5 z$9sx+1bR5>gcP(<#7ED;BEyYizj3rskAl`eeS@oo-J)otg$IG8!;K8=#nr-{D~vQ{ z82vmZ6^{4tQ{4SLp}YkuBa}{*5XxL5l(#TOr#rkY<_V=UYIeIre^Ht@boa;8nozAi zj#R)rXDxZxqe>Qx-o`1-GKk##QtRUc3|)!nGStX3Dwg|thMLtY)% z)Y4cF$ZwMx>!CJU8=5L^>%o?#sN(^jb+v4sVtBOZELM|NS-I`=`tM zvN?le@@l8%6r~iREH9eyi|amvWl`!`jh_J-uON zt)S>1W2PiNdBK(1n)eP=pS=7I&7zpoHXE)`i;y)1&9Lim{8At4nZllS_16JQO&+-C zQwHYs2Ilj?&rZ%hzHnS zqRo)S+(R?-t*R~@aa3y7UlvQya7%6>lIWn3DO6JZr@ChNxtuexMcjaw!#pA&^i26T2L~KyyrS*->&e#_@PUVlS$No$UjkX^ep;`jiKJlmooxS*xPv zeHDsvV9r8mQJD6h2}x>a(-&tvOj~eZ4wwN}bl!ar@z( z71Dp(l2R0J^^{h<&qsHnS9QEp-32FXXqMBaFov?n`Lvp!g}3CmX>l24wZXE*@?(yT zqp;pfe@1@X6D->c3l`Quy4`b4gC_wxtg#dSoVul zZ9EM05csXy!E5}a@-Y6ZcsAyx{%fQug)|Yv{%FBhWsgeL{n5cvInN3p6BUTMI3u78 zV>lR7IRL5~372W&!?5y}Zsj(n@^M;O53w_q_dppXDi=xRH9~IM+fsS8{abwpb3Ie} zO~Q|G{3VxsD-ic8A19TMbd)LfnD-BJlsy%|93Q|h!m(n%@o~424KpQaWUB^iBRix$ zy#4fy1V_2+O|?gHl%J}jeA|cml7RqE%{-__Q3B#*196TYGZX;TILe^ghzGIT5}!0N zf(c9)u$&oYHgX@){`^5~k*wmC1J?4bZR}HSU9$%NIjl0Ug->libxD8Heq@i^+{RuL zj6fBOdBI_%Fy=8=3RhBs-fv8da}ctf80WF+8sk)liIK8eKm9p)&ZrDLz^CFl=?j%h zFxI-1pTKrH*lK3sqZZ!ox6ubb^V;ZRDm`1+G#2rrC%sP4P<;csmp0map5I1$Y8`16 zPMx>BI!8&JEb5pxy4rsAQ)#2ReNNiw0XPZArBygpOT4OU9;NCbSp`EooX#6o&M^EW zEmporik$yk+!;#T_L~ zWWj`$Hd-oI*{`^jBc>IN(#re%%HzGttH54;@NWdJTcFIj?OO@2aYP^8j zU|Itv01OZSVj9L>Eq&OP)d!SFVh@v$!`UO!0J7}8w{ghnjdjRcTWQ&3vt;>2#lIKX8bm<0Hml(KIoWy(bZB;VH9 zTV7Ms7b}|sa8}{@LVCc|M?nEf)sM<1kzb3CG7bNDaV#`=XOAkGiHWA?m8hhM1ck;* z|JMITX8a<(Ni^p`6`i7WQK94=WwqS_W{a7N`<|bLSUZ`C5T^%ztAW@23!sJ7%pS)# zJOpi@)4@e^macMo;5Y>Zk-OIIfnSnm{K04*z;sx<4aG_iT*S`DdCRnQOA>y48xcfD zUl8>Wa`gyMsQ&XPOur;4SlWRCGkK$|=D*{T*#}KTCY_upK3Bd@lv2b#8OKq`SAHu@ zFCtW9@ftl~$X;BfIQxO~1XN1k>f?OM6aLHx!M>9xR$5Q+&^qp7(m_^p+)c$xiK)VI zIY7ZN?JGoAAISIe2h%@Pjtk}F=VkWp?rKj{sWnS=-N&z03ld{NCsM+KU+S+xF#%i( zKVY(N4Dj4(_sJqRIJMWV5DtQ|V#OEuytNQ>tq=AE5A5OxAnQDyaJ_|4Ri{`)J+VM4 zT<%wxE!rHfjUM<=&3c}{R!c+E)lQuGU!dd!<|*G-xHU~X8w?WTlX{ub5t*SBJNTyQ zT%BDQMG9rG3x)Nj9>G@#1a%C_rX-GGRzK}rpG7$I4xMjB4iX~oz;{U=`@Wae+^&PO zA-uvW9v^ZS1BKS~!ToL?TMey~|IUjZ_PxAgu$NH+^YI4rIf|JlA$!Rrw-dgL_RVOe zu-wFH9I4WTBUPGlq)Lb*Ros0TZi95^OD8Xv=4a0Tq6ig9_N@E7CYMvsYjR&f0>?)2 z28dzczU&+1iQ|aFCzrMxD3*xTebtAbRl#tvneN|__GI#ET19wd{ z4xRl+y{2f&Lv!zmNq1LRd?x1Kx>b5I%c7_Z4`M#SaZ?Ro(`km!fjk_4i#LqSmd!$D zMSSH>j}?y0j-{X=9;$SfN(b7gk#2S#S%u(P|0%-#orgL8-{gbxMV zHj%v*X(u_>f@Ydz9lO%a63F!#hj&^ZpNV|-zWl=94!#EDSddamZT|N(?`|P5AD#D_D(=02}#!WaUD8Kk~RNQA+N#* z{Fea&;2TT{CpDB3K${q#brdKlL-tb!u<5j}LX!VTvL@>H+NFqJdLnHW{rCo2?k6H! z=&1PYWKYGPf-@Z7!W%}#Kiq@RU*P>{PPW*Ij0M}eRx)petKw7S8Go=Auj{CI02b-0 zcngdUOo6C9?J-N7stje$^(P z*e<53(MJnPh^@M}>f*D@ye4vOiPuD0;b+{eAnoKxcXC+!IA;mG8R4BsQH9r$tfC{l z>N=kAZU-|QM97GHQby(f{g z_7&1Zn&pZ78iBHQW`wTWMKtyJi&6?(KI znI$!;@<|h~NnK<$ew~-7JWMM$bt{`D_37PilX}~fLe=j8#h(JXtiECw#rH}V?9E&N zR^T!K8%e!I<(eDKEZ10W^+mQ)W9d0k;}pL}$g9y+YIJnt484sengnszgvNQpP-J6C6QQZ z`a(n<&ADTdXP^@)s)5dyys+6o9jQIRwUn*UMAja{jo1TxuWmut70nr3;TULd3JPN5 zQrAE`$us_7{hLE47z4ev*f-GCZ;OHMbJH)kDRm3tM{?gn?^lnB6Yd__*S8(buud_@~!V-Rkq|0rm=H8h!5c)^$e@W zQ$iho^>jnPI{ZKg_!^$r=|a2R%&{cDF`Q;S&qr?wOI}84$yoxpj}N%g=oyWlE~MQf zwbF(#(9D}}1l&%*wGAWPiK0^G!4>QTAM8qi8L;OGSnZUNojStOJu)IR z1mlA3V)>q3F@bCB%#WN8;l4XS-<=|PY7!9}?T9j6d?{N`d*Y#G6@M8gZaKs2_F+cD zKh3SnFmaXrEuU5(8T5ta&7V1N0_TYhVw|mkY3xpEvs0nBx&`f$_RoOH>=ohEGTzB` z4@i_v9hst)c41kTx+dbPObQ{f4Ex8YC5f{dk#w?0Imn0cDO;&zkL(NdV617irwF0t zfwWUZ1$X+gN)-!vL?E3a(iiXX zUL3@WF8*`}|G`B8uXPNdg-Yuf!19bJM?ONs;nd1LyG zm$b)9O~=L0QG?p(SDWTl+f+)mu~hS1{0H9vgTpX&cC)jp`)j{u8fc7S`b*7i(2n0u zRx-V=sH5UzbA?N0(8G6~)RGi*q1ClrIQrpEigkl!ZJ+?9Runc%Dwwe_q+Qi=dvlF? zW|~M~22aBAt$?f&wBIY*hk! zRS$vAzE+puPLOHXERDMM*Oe0X*NyQd*lj)>t1i19i}+?@5d+cJ=4ktry!e(A1m+GO z<{ci)BoF4U+ZE<+#$UjE(1&?GFpW*`#k&7x0#WxM9T7~s;Ncl=At2B63d~sZYXE8&UT5xOYAS94Ud-5eh5g`YzXjN@7(14+`h)F7 zq+HPCV*9`&{1{*xEKko+k$%`kMH+@QM@`?2VtKj%ATxnMM>d>-@`Sih5L)#+-ZLs=V;DAu~m^wFV|AP;wl0onU3>8tVCZA8zeYvdYU`2*bmJ3+U1FXHi4oH?R=QP(rB#?RquPNOb0 z45O~|0H!1A6_^}4xPP|lAXp~3KGYX1l_}=(qB=|*)O%5cuAPSn{WP3lIR36L&UH{z zGg*XQ__hjN`h(AD*JP}CtLAbt=gq9Qs;psArvvIxi z5j)Bf^hJ~s8^DPHqZoCLh>$*5(Eq&uS-o47DGaY=0Aa$aFTF>zyT z;`e;mSvj&tOd8;y?>HK5MjF!5=+d2@Mu$LhI9?53hMC)^pwZ>@O<6hLU<)#bxyH4L zejlhpgRz{2{De>j14s9n9=ryos_dnYi82GhvTFb!A>4cVLvpaJuk?0hzXnTX7%JLb zZMR@_;Y2Vp`U&HWn7i`@{0Ed(UnuoZ@aVa!U}@VsHD4vWt*aaI)T$yjkJ$6s+X)AS z4u^TFt^+=_PVi+oTrwHBKV!3a)0B6$R*_TdPo?3IUn|wCb&u4NR4Y;WY7%4`71LMtGU z1HzNoFFkAgzld+wSt7nvzuNnv$UntiWcjQ8_)69AD2LMu6=Ea*^{+fEb8e;LGAdFl zKLNuitk0%)vpa}l0M_&JeT1_=?y z^OLk?rgYMl*@X|MWo{dGs8$cSmTA$#bhf%*)IPIWQrU{Bz3i{x9(i@;Ty3iBaE{i! zFh^4sIPuHm*YdQz<_ECk@IyF$23$JOOAm)$q+q9cU>`~Vdz}aNuZarw84v8w{1A?( z`(RrqfQ?vv#r)P4AA@CH$au}}<(L$5>PtSudGtHUHv2-?+F*4Q#PqRuJ)=BIwx(@o zomc+8qsr)X(7U=)w$1~;(b$#HqYT1wcgp-4vL73&E{?h1T6g}o-|#~^VKiNgsp9Y{ zqsc8~nB`94Cg#Vxy-7#n?U#~0WLSKXeFYg<$9U`g;<>MV5rlrm5~ao@x45yR8ySZ^(RsMKh-l2}C~VhQ(D zdv}ID30W}fz+Vu2s%nw#b?E86;+54HT~#^91R9}Rw+m-;v)D=hc+I27_E$z+zQ1 zKQoCRR0vOyymuOJf$a%#8?=-D_JmlE@6`*DP&DW2IgT9rQBV*C5NYI?C(rmpM4SKP zG~UTrnUxm9-y_FETSbln;7`d1I34UF;u7eq~ok z?k%uWs})jB0`E0;vm4E|JQCS|tj`jbQuOjb-T+9Wnzoc^u#rntH3d|08tjc%daB6~ z{Z|lSvg)P2oFGQ8Cdsz|ms7^D;{OuYC;H3wV!CpO6bA&iaIN-a_&{26TFi$(pTNKy zLN-X}2Whn7QILkYybo!{psaDC3OQ`k&_15;hT2&34QSDxe}x$p>NrDP%lszjpuGF< zwQ1jtK&6a%E(hDVpVV!PHXUtW;oYEoUoprc4udi5V!((4l=(Ge_XUeAc6LsIP&_1b z2k=`gHEZaI;FA>+7W+gvc6p>d>iSzVLla$98_hY+Smq`ZHjA*X;T)(}9n<>F9A_#s znik}%uonB>UpCYsteLQ6NFQ>u26#!>v$KeVbIlVrPy{#$$aE z-Bzd2vXj?^_bjPfoD!VBy+PReBl>d_EvNEpik6OwPE1WINtIQz73hukg5eM6325$b zNIN*}gd{J~{~txW$Va;X{tV}1#zH_nA%KNHGGeoq*QR|yWo+%H+O!u1OfxQJNqc}2 zCO5^z5Se_W=bLZ{Qk)HC5W3pQSC2Q zn>OTX?9ZQ&tpr;*zTBt315_LOR|2J49nPo`sHB)<$ky1?-j)zNkKAvUqtV|m9?ri?Y*I3Vl~&W$Z!}R!nb^uLEa4^`LTgxEHj@I%Nr$P zW_cl0eVqOlPt^`~OF^VP?<#SDY;1J0uzQdacftDv#|*Ow;AA2AAu3uuM&!%HB!j%k zMYU;Xf?AUAx?;FI@PtD6+lSE5V>x~!E>4zz4byjY2!k+i0s_0{0-ju~Pu}!v-$rd6 z^`_{PIsTJt4}%LFT5mTnpzB$rplK#o`GLtDmrHAQ)9bWqLckCic0#}LT*&l;e+kp! z{c*0i0QP79p4EKdON>FslLO)tDw6EZ=JtSLkiBKA+E}K+TP+`kvAZCM6WB3Rz9Ymp zZ}20O$0^5}rvQ#;A^V5E>WW>5<8Q;r8Zu(ag?OAPZLO|jtX^ZX7meF+OCzi6D67{f z%2g9?(ajA1tjO_=nWEUUB3VN-IZ-2f@3^Oek5v52i)>CV<$`J6URV51q~+S+ywnbj znv|T#nLwp0N`C)mjg!BLenV=G!$;g_>LBrfZa%W*i`p0H`KYH*R*?n$^HJw~uOds! zQdP~9472BLXk}q@*%+S+&A;s(^7MGmZKRqH9 z^{B)BaxPyPUzMzh=A6?uXfh45Dg&b(42K;Xxd7xwi}q-Y@)Ere{seKEw;U6R4V6o& z{N4aiopmB)x40<7*+m|J`lf`%J_IN%_G|`vSnOOroSylM%cPehmFDo~E<|zVTlKsz z-;XIPBXi9ur*i&m;TDzePo$#(+)v;eD)xerc#$r_Fmp2B6(`ixr_BAB0-8UbhwMV3 zBSF4Bn8f6SzrM8ZtRszhYz7w(k z@ti??SoXLZG#N^7zFtevIg}Do3mN7n@!?4Asoo~UM{tZey`!1kaZwO63&*phJ2b*L z!&6ByAZwn@x~LI8KUgZBQZj(LoP4!iIm=5BNgA?+=J;iKy#r>wN8{G(@REAWL3!GJ zB6PaPD}YkAaOzMAY>QCFB>jE0{rN_nIueL&_$eGO^dSy5yy7zM6^H>wLs2G>lZsD_ zl=m8esl^(KbsWZ|@>||5Hxgr#n`gY`wn|lvr=MnetyeVX&SzCMo8&-8?vE|utD4a$ zb8M-M2Ls=VVg|~K{DtGU`Xs&yVMZ|}VNqD| zBqrm>6-TnWL?)zi{0`7-#)w&Bz9mJG?C-{(7qPl<@yG5+c~ceF#7Op`NU-x>R<@Gx zvG4AAd{U(4p-A#~HC)s*NVa-NWl{=_*&=GzBi+b5(&~jEUU3-XY-Qnv^=?drAc*@f zR6(ql4`X$$U`7oV_5O0LE4#Z)iTjT)m8S0kb=uxwXV@t1D~^@UuY6aR1(gh zZH#bSOX#R8t5pRB;Ez~y(x+1|68#&c$kKI`(pHobEIn7gWmg7o4{0GUGC_`Vn}v;V z+m}eq2&kbE1>#vX(iP8WQwo{ed(h-4xfN+_Bs;Gtc!zA@-~fgJ=w?so2iNH=<7%dsiz3O(?VKLW@yCYKLdRo=Le_)nGaeCAeh0fz*t@0#MZHZ)8r{XhNO{_|e9-7-6^bE= znuH`{NDQ}qXFw^ppP4b6{3P0zlRi1$Y7Au|!X9c)Hr4Y#Hk$tAd__>t7ud^S561&u z-m7YaZ%7|dt_(C6(EO<(9qeIbh7TB05uG{~XL}O3tV>650PxHp#pu*w{1~7lwvL>J z6p@xev)b^jskyaj|LLp4ykc=5AE7B`aQ%$c0+1RJ%`zoK)W(!Tnj$k3JHgYY3sE38 z(=oMaF@ZXKIRl8{wP_>x63t90mYL*CMIncNI`%o-L($qpQ8irAiWU~OQz32T11%f{ z$D#l@p-nr@c}PVhc|X?<$snw3qU-?h-l*(4RtPiRdx&#Sgtz`9SK9utXin8^)z^N} z*TgT-&6t!;GL8a+ubt^ncelVc;(74U=^E@up;D6!0d|g$H=zv z@#fD-w zh)}Hop1kM_;7o=5jFWut#{tf=b$OX;|ulehTef>u8c{nKEEn$tXL8avd$;?-oC zVew2j#d-^&&av5n6=qRw^fRPgWiS3qjg6J{17HxwZ$^pX_~|YpvkZ92Fc{X2bFZ0F zsAe`J$g&j^>(!>U?M;ibdNA%-Q5>+3c|mLfuc+UW`G%cYRV?M7XeW22I|k{m&-3fT zgt*h>kC#3QpOQb0Q&3SPMw{VMxdEb>QITPN&;pC5L zRpj2B^<5!mA2f}*AC2^e^6-ZZAO}g*rhSI+mgULKgB)p&jkKFtPi~u>*|m=OsB1ow z%}0v)2$&D&P4av!3{-iBoPL2X^qWmh20LrUbg&iItE@YePt9EGcq(Ckxke&uNNk<- z5TeqZZQJR8Syz}#l_bgPmP9H8`yX`latdiiJgjv4E7nQ!K@$L1BW+f$qdB9AJBGN; zq}Y@~)oXzk&Ds30D4MMHS_-P@zr1YhC-RIxKwk>zn&&a!U(uPB_7`57j8<+HAedj5 zZev7CG1g-N;gAp%^VC85bsqA)gY~yiO+O}9O04EP6pnD8i<%<#+)^1A;Qd&^D(5^| zX-laA^m+>YbRYWLFleARbfCxjaz7*IXBdZaIc*1Wzsq-*RJ}mEYX!Izt z8`0Y2R?XuT1N5vow9~>GX@ekQ)n=hu7oQ5PMi*bA7FXMm7+$upVN@Nf$!=txx6nAp!TVd zD<`=V-3_hKw7kY}nw#(3Q=Gj@Rt^IGd$bgATrQoC!FoJc!|_Q52}qxFkszm$ zlzA&;zaJ){S6eesE_qT$oFj1MopHY}?{vywWIo(n)g|(t z1N$oPs`EU~2SEU`D1cl;-Lxx&+7~=|U(0t_-s|r}-f!k6$UELdOkZN=+O%(bc*2%5 z^GB{!VGk8ttP2&f7-7Gv7-UVZ!FZ920b%bS;0QYc7S7CHqAjbwrWd-=N9Cnjy;8Fq ze~sTMxiVuN5}6j2Dg0n;vAoHBb>P9Gs%b-D9p_^`8AikLxgOTn!Q^(lQ5WmFOH{?Z z?Sg5(V&7nK2_PRfAW!o_7AZ)Xp)icRkN~os3v!x*^c4FHM>zkRVh1ps{D)%Q_SDCW zm-@_7{hITDDT6r4zR9G7CH`7jCGppJ@)my#AbG)&d5Iwh(|)?Y>j26r!Pi&pq(W5P z(38=#y|R-Vbdb~(-%^D%aIoUg9_6)ge6tU^(t~^j@M>NV>;35Q!y+cvJ|DQ(x6e|_ zO8;DwpB4LT1pR8C1BE1GEFS|D>co#4A#dp?dOS#7zGE?!e0S|L1EBWNQ1AJJI3clX zpE(7G+2^tqYM(o3wjhXzb`8n~pbW=r4F)9kkc$CqS5W4yko`)I+NT`WWYxzR1hX6W z#>qNHxqZ4Z%0+^m`)dJ?q)QE|Z+-6WP*efJSHs-#38;R%+(DJ2sA_uK?@jTIve}sm z@_YmGE+6Da3bKI*GME5zwhMB5N1@&`%7%~qyHQrKB3%-DiOvR_bo0R#<-R?>M_Ki` z#bj8;9y=o8%E06D7JuP*TQ^L%0zVl_kZ#bFknP`n)gC9DtqS=7&K!H}2({jLz-nlB zWD?LZ#|AJSj$hyd&NK=F@MnM{b%!~&dh{@JTxsI>fY2pZsqMepAwAtsWzSe{;&($! zOOdiMj#iKxju(02Edq<{Mmkci47>(7`BO8Qk$tH4LxLyr*@Wbg4^#CKMj>C_ut)r7 zlm7mmx)<)i<`ehST`3^-JiJXZ;E1>ckKe7sAz84pwo+u>#=*^ziM~iCL6$ME=li&A z|2|4sWCvUu#@_=r9a*pKpt9D52*G*X>ILWZuOFP(i}-$9YNR2vG~Ed2-68?LTO{On ziwpv^Xg)p<&I>vv23iwkpmh{;ixK|i(wFM3#N43Rhx~q@NO4yZ{rT2skk(B_&k+Ul zr#r%|P0Q}W&^WLaehSM6an=!XbM_^DewnC+?e0m{CyTUWeR3s6@wd8c%#^|06X`a< zA~TjqWbb9AME;@6&ECcj$A)mK-7msbwERB8Sv_T*E{5@hgLE>!Dd0?<=#M*eI-P^b zsQss z%Sr|?=W8{-k1rYP><)MqCxX5+U4Z=U1KH*QnJ7SJ2{n~@xe$=cXTFA_t$+a@* zX*dx_Wilzy4tgkvpzSDs(;9Ny)u$P#k#c_T6>?eq)nY*3$VdgP(X8*x8&#Xuqci#~zdL;xzq%JP znRm!z@Jw34Fj8MV65~bAjBs3g$w$DzUdK$nHyT;SQwcn6`$rzN^vk7zxO)tkZhA=1Lg|*hn9zG zW#*j&V{{d)z3sR$it{G-;Y%fs>1X>O3&1v;SOvSi8F@}oj zkw_io%+bAwfBH>tH&ApX!mauwVP5xJ4}M$VyB*L+J=LpE!4J@)NxnADz@Cg>v&I)qxKS=7hqh?PU=pjptVTh5IhSVO5$e`1tz8z0R;Ff; zFBK5^?;xCS!;@@0Hj;vkQR(g`A_o_mG0l zw|>sLXmKeXWE^|J_!QS-Z#CCZ4TX_T3pIqOZy2%V!!S|}vFuIt7-^iq$o?KMa?xnO zDLb7rOyt90dzj?S+_R}A9aLofhVFuQX!60#Qyk)m(|ni_`DR_1Rm{r0Mp^dX4OEqZ zaP1P=AdJDsF05a>RJA&~Xx!&IN5H<*ViA7}1djN7aB7LG3#N()I%H0}#^m4W$ykUx`g&J>FE+%dcCe1hzS2}AgZ)ra;k z(;3I&2K)!rJzDsi>k{JY;R0DES_xA7x`a-v)|gM$TWtPB30;MAKv#DNy+)*WLIfvk z5J2#qQ26x%UssP%uGBq;O0;Ngb31&K7hq!{!z7vFuV=~a(?5K9xZj5}>1FJ;t=!P) z@5z2>#FJ>wyECA-r<^uOR3!HnO0YVcAGK*LGgnhduzIE;ypE+hXn2>04^kZtWl)y6KXZ zovj^WuG1mXM(CXA>N=h2cZloK-42n*9BpQQPQF?@#AZj+fAtw#$lQhWiHa}!SrdEh zdGsyP{$nZU(h&k(J3@jD1AV$c)Q*s!1mYTwwoBxqlnj$_kz0_xZ^9XcE%QRyYS{YH zp0}qFJ;KDRA}uH9N}sSawfbE-m8<1s=MB~<*!P*2`eoEQugv0=!r1s>84IhQ0Wiw1 zmnJ=Hu0*#+>5p~OMvk4fq@=6$F4O*Qfd9_>@juH`6)p40?*{tczw(rJJ~+GH@@S+7{-U8?0pO>Iw6;5oa2T z7RiSZ%VJ0k$4lYIF!6I2zb&kz9J?qVa6AjyCquk8V4ZlU?K0IXIXRuawr(qm?x+(P zzGdRMPJ9@#dg6l`ZL;4*TTYzbSd#&@Ya-E_I+MQBbt;nKmBS-sRY-6yiwU}fBFQ^> zG|Wu!XYXV^7KR|`JYOpX9nVUC>8{6m{5Bdbb4s$Tag$v4dj~(LAl^2&SxNr%=(vnv z$H=Svh2tOjn*2c6RZq4x-__(AN@%j#3988$A5NHYbSu@QxL3IJ)?-;|*Wtd&wtq(o zo9Ij9wkt7!aQssrQ?-ZbBjDFO=&qkSofl=Wa^Jh&$m3^J6^<|R>o)i5UQ6BT%O#Mc zdCKiI8j2#r!t8|;Bn-bx>n`!@-t5)Qle#BTH)CCawUCt@y1Jyt0Bb|x@a5dpvBDnQ zRGRqV-2I#U#Rz37c(`9!&(Y>Q{3!@mOj;H%=Uc6xrk^l7mh-Ixg)GFtU)SK(RW-+( zo4LeakN-ezbO|k&j;8tXo{gRgv5acr!>Ffoo^PXVC|3ra0G#}(osK8g7k|BrAFL|f z-DYpOPU3`pq7m88mfJ+bgewfHcjSYrgnvIMsy!a6_6eveIy(*FSOSo?Arr?vAwlI^SAbXnlqnK9{v*VQWJ{u-oQH_Ft^x+M;O$`_o1eZB1+)N!}ev zu8Fk#tK=;S0;b5i1zpnRu2kZZk)h_K4O%PvG>|IxJglk8I4&QC@sTh`XU@4uM#gRU z?gn0;g&oFEJogaPEU9@oq5b`f{%<`(|9HrM=zo%{e{;W<>i>P!|K{VAw&p% zH8}cbAjkfeJTAHN^p8pW|Iq)m|404T$JEjP#AA*A<-;(3Jkkp{?yf)qQUG)|2Oj@2EUi;o5MY-^quebPLpk zyME?sc=p!M{Iny$`kB9q8P%BeGdtblVZE!YpIOHf`Ab+oqh0L$X8g#EcX{{f+O(x@ zyzJ~{zP+AiSbE0yY)?YAx#P&N6i8sHI@{Ghqk5L$K?-sMTCnskN)eftlfUVOJS_V^#JzcZR7Li{8z2o4 zh#kd-s35IIje`pgF2RVVAt4>;NJdd{B^pHB(R5e@5$r^0FRkdH;)0@sitD&b5HWxd z_Dw|)L~y|yBZ$n%D)jrFQ}=dv&^I&h_s`1*Zr5Gv)TvWvtL@bB@`!$+Y_qj?Mz%I< zv1`{n-7mKNuwCnV0Jq`Y8IrF$XDpCh2j#g|DkejX+F74e zJ3W<>N6-~D&G`8FNlKnTnx}sIuH%K03mU>m6OKtR`bA|A&$Jw#mvY24S;+&s65F?#aqrMp`Q~-OqsD z5Ii!A6$##1n_E}1oh6hr0<@pJza`d{)r5ttQ662`C7d&~u*;yF@W_sVai13S^ri*k zD;l#3>1`E)bRkC?2=We`Hp%M$4LH_BeJSJIbu>iWS9q$s*Ch(-EmUO z+X&1amP%qgsAzcPG2<4RDlTvIQZrM-*PTjaK8^;nqeDdi)aw;nb|g4~{jmo7YWab} zc)j<61>5fvOm!Idu6Qm#&uNE_g5hSqaVQlH=1CMjwt6sAQ(Q)0f(fzWVD!!bao>(p z5Z>r`X*F3MUnQmIKrKq2itB}aYMCs3zLqW#EDaajN{Mdi{oK-@J&w|E5OUH$GC2ib z$9)BqgYFUVQvnG03k~>}<;PTRkb;-sC#P{*nnEUD6N5>E1Chp6IFA667ld}rJ3pS08-z-U)q&OOmV@M9Y z`NSfg&BzEJ>CCIOAroJY;OQ>?K8rmtmVV`Y(U7ZOj7@yuKlF=w^Th-y&8W`y5Z5=a z14u13?^f}y@rbzZQ@M@Ql~bfVvJ8qUQ0E4%hvnJefcK8i<+2bUXG6k&dAF|Nf89>+ zb-s?=@AM8>HMqn(W|e($6(W-W(Wdeo7likYwE|%Q5Y}Wih42;kNeFEPLUm3V^y%)o z6Sd~8mQ(Fy`gE-P#(ekly^N~wtQ3voeMsheLb35frLr;KAm{C%>H0Q0RWo@y@fbch zx2>|-bMj;O;QMog$sbGbK^FHeo7rbOd~jYzmZG^jZbXLg*r1c38{VMAiY!235#Y!k zS|NhPflQ8$ME*oa3htt4Wn4$B0OK}dSLlIVGmw2jta@Ayr*>&Bsv%u?*lU%B^j*DxdSiz};iVs%}37bj&?p{Ttrx5g=_%VPOio4rP*ifTSMn%3=`7x$f% zqiVD=?rS{JRio=kQLz_VYsL1<50r4#=zK?Z3&8F^#ZF=+Ikb8T%(T_0*MowaF3-s| zDZf4Z7HL>yWPS5CgiRU@A))xg5%vO1KNw zrb}VY=#i~FqH*Jk2Yc_B8?-DH8SfpdV8-2WU8=DBLYs{tR%DWJ zu*px}zmwNKc$fRhSp8&5lTU*3$p$`2Rw<)WAWduX*H z6_kMX-4evPjxX&kL$%Gu0s}vqJ-k8dU@*V3pt5IP+r04E-Vf$lD?_n<@j#bF!Vl0? z=r0o_4wdcak`Nd`ez0v`ki%#5M-|*X=T9=fb4cbJ$ZT-TI265`_|)!cfQuskAJN-$ zSX1=Uk`VqodLJEYeAWpr9SY{YaA_0o0Gw7kc12jgeg0senZ4X&K;5Q*Zd`SuhK6Iy~T?H*4!rGmrwk_;?+Od7;;mtgh&Ouh?nW*1v^lB(p%Bp-{ z5v;_BgI~q%hMbw$N7SFF(}MHPe$F@fI_^j#$iQ7RjMQyVH_p`nxW!nI30SetXrCxZ4tC z@(<_<=|S(4urO4@zs#s?6^xIcMrCKDd#fcHcF1B2obg&=&K-1@ONoUz^A_nUJts_D z;+^%~;gxsNSrL%ZH?>YUHHJH(zF7ub8p4gfw3K{GnL1l?-=aO zU2CjdDMj(*!#c{^={-V!8_{G61qdZTuygq+S&Oz!T3}-(Rg|sxx(5SKat|n_oF5yk z$b1BDw>AayfAtoxpG0fve$Fr0&v}gQF>S(r2jhOR(0L-}gXtrdTJ<4WhhpD+Gbq9{lACpak@}3u-$wSXvm03! zyS;NbdMv>?4$U358iFI{u+@wLfWc@6r_`-Z*iIfcj!ClJ z8j!Z%CP$Mm3EOQ{)opxE!%TBznZEzXTqQ7M?@Nl7Iy=r-j(XK=PSNJMng`k(emmPs z=fmmjg9ccnVGyO;vyP-U*QiYTCX={zC!PkHrB45&zAdrVyZZLmkabY?Z9l~xyo1yZ z6rblr4u^%#H!SoaPYmwN2W!@OYhDqeh;kMM zlT^-+%H#a`-ic93SS47hxx=m$w#pR?C!R=+h|7pmFeIK|>B!Hne}vTp>8#3q0kz#p7-)y=-KdVKsE1TT zHiJJ&nR6_SBXj@0BO!Cr&-!nw{3Hqt(dpkdsEL66-&OgibRkm(tUpcARKV`1sQ;wO z#{j{;mXDHE%xt2{kB};gDnElZki{LM%F*AeVNPqi>hA`lzZa;r-FuXyzqj#S+_y$< zjsC7D*=@0v%lJ2`KT+Qcd2BeP`6252LQ&tRK)q6Hp^Uux?N=*6E1|xp3&7>7y{|N| z6WUwiO&INs8Xw>&?+~?il=oTtrQr`z-g!By9eVICP|My2haY}BF{!*CAq|&s{!eR| z{1J!~oDyuYPIR{@?ypeXe;}#DHFIuDa7}gHNYT_&VE!5K`L{ybSmwQGD>Sv%dL%lt ziKbpCY>K89DonqD?vdI9zbsXJSlr%{7AwYfPpfK!PM1YT_hqD-jLiSD*Kj48GL_XD z_8rP<9Y97{Z6-*F`j+2Rub~|f?48%QlpYB?v-%KWecA)l%B>3S5C>F1T zRO55WF_pu_&5XwBr8W8zC9VKHTE`iWxC80k%o&siSnvF&o^?AW@I z2+_&01#5XG7Tey-5IRn_>~MQ&a-!uf0t*gW;tNyh3O`s8=XA{gK*|2$g(Nipsb;YZ z`-k6^cS2|OKLJVOoNk~wEnfrr81;GGLsE%#GM^;DH2ZB6B8cDMu*<##5&Bb{NQ>q9 zwzJasGme*l#)EbzXncGtXl(ZCZbg#Wt9znUyujhb(Jb(Lh(#}ECxym0J@Y7;X(GIX z^a@9~J$)ILOYVmmH+LPBZpfih(jQCC-?57zLn-a2MOk_;UEi)eu=ekKGq`(o+_#?A zl};u-;Z@S3l})R`E=#>1v5n>+b45DP9%EPERFvuxE0a*jCL zID*}XD_@=C=ZUQ6o5ruTUc9evi%YGBQaPi3rybt~7EA4QMS?+VZ%rF0VZX`*sO8H& zNgME5_Exc_A0MA?lEmhGJx3;JWOOo*-%?ItVD$yFY;wUNVWk%T%R@0dlhT^+iH&FN zw+9Hs%(ZB0yVGP=z%b|)iMNUt&K@-SyHHG(K67FUZd(<-@%_QrdpU2xC4t!Ot*Flk z6gaG~SJVvA{k?P2TM-vAOIY{4-wTABA`7z=>b7Q848gsw(@&+Wd}TG-`~YoNGVAX- zC6CHK`u05X7R@b**9GEpiAPgH3*EZ-U~u=AV3#Gq)eXTGOM-J7nwLcOwCWR`_~}sK zk6s*#&CVG?%GzIIZ6|spW40664-IWvd1LGVk_UzL!HOM$?j7F?cd~C}?$h#p2jT&% z?R$26=&xjyz1<~N;cwE-^;RF`E-%|#Ede2gC=1e|%jjtVtECuB@m{0N0#@(zXds=R zjA$T(pSICJTYj>lfh>NGiUyA2Cp#L*4p`~P^u!!CGHSlU9Fwsy$K))`FQi@NelkjGY%TQr2` zE_QOuRK4Ubc5)Xxxr?3L#hROr@eGIv!{z}XGW!{+Lt0e_yA#u{R_4>FV*6XRBSowA z4f4g3v{s|nB1!rG7D+m%KVM|^GLpnToXi}?ZtP#SizFS*yT(5D8q}V>TrP{aSeXxW z5r%u5DGJ3*1W zeT=Ik=a`hg@gjRabB6XVx%|q7_3sA2Z3^IrrT|_}0Jzry5VAZcpm~w5W61zHm;G`; z*E0PDU)p|7z&<6H#}xLZ0vjAvLw8z@&y}(*@wxpC53vlMvQ!D~R|W1ZQ4-@{A)^r}IkeJF3V2 zi)72pVN;qNr%gn)G8>uM4ll^w^fP;K)U0oI37jk1wMfff7Y^8OcA~DY z$YnYg%X0&*Pfb=5SUFI8hBPEDgtHCvj-%by=xS2_#*2Z~#wYBc=NH~$UniihluLhw zccg)LlfvuT6y9+Oc$>2ovL8~%j#C82fIN+JCh@zqr>kqqWc?O?0JD16%@-${T|n`0 zu9X&TWfi4Ii_?+dXmN&B)HYh&7AcMvXIVu@MT?I@lB31hMcuYZ=kC-JmYKw^?P#@N zA!l3GfjrGYa?$iTjV-vSPMx1F+IsXR)OomniS>0bR>hS2g;LnrViy?!F!pe9>kerp z(e#Z$tNma$=TwUbr+GhUO_bAMeBOj$oLHwj8q-Uor)^XLUdU)DLj0}>$yh!OVWf2@ zQ*YK?4@HacGXm|A5wvRLy(kirI&h9AA}Jb@r+$)COfr|`G?TO=2NCr_35l(Q7{^+%0+fbm1!98Iv}6Qe^Zq1!+}B*hc}n(CV}v*oDz+ zo+)rmz&D6R9%5l-{*^s*c4iMF@wX%K$KyfTKigsg+Vie)puO^F>bpcP3%GDL6Rio+ zx+0ZOxfWM5g}x8@T>6%gf~b3e7pM&<8-3bOO5Vn!EYJI;0)hbwYZVQ8Q`>~!spLr| zzpMD1UvXa94=>7Zui=kdw!t4K34XqRf*8xU*AXdP;ws%>A1=5bCYN-rz%1H>3Jgc# zQh^;3CYg3|6x8yg+zLEQiVEau1vs12Rmk5oIQ&7NVd2_=^NkWMvwwh}t;}!Ou4d25 zPb$$B|AwSb{%DlwT;7Tjt>qUik38H_qBm!Q-ypdRS9!sE2A)H<&>Q0PWH?ZAUP1CVX1-<>ns z;Fh0k5mmfZ)uf4!?6?rtGVM!uChXt{jyQF zaI&AiNb~soczsD^p1O$Z=YE$?sK3ZO>7j@M(vb$abmf0u%J_Z2YF1On7UH^K#U3j( z+W?#`0FR~bau5rW(BMo`wF*YI>sd5p21DJ+>}}SNqRVec42YLG%l_1vVRtO6v!ngmookjm}Fg+ zRlZ6LUAwZSv3_uZScjX=)AiBU*cVHd61fygKF?oa zQW*3TBn0A=>&Rxp!SEg7%J0f7eLrHSNYJU$6VoFU?ao_Oej8o@>CzV4y4CfK_5Vf7 zJfzx5dn}yDy@V9u;)-)bUD4J3B}UE{RM8> zHMdpR=kB+P&s`{8ym>sV+8Li;68SV;to-t#{)|xy3o}Y0 z^gUl-XR){X5sEE|-IrHFv8Fta{UWx|S+C5h;+F-oYLj|=ATqLnl%)Zacx-oR2X~w- zm*cruo^NP-Smbr1!f*doSm0#(>7uC=`WUveSPT7H82|nKXxzltm?1Bp)@^ATb(8+tLX@Rc zkH(i7vAFMT-pKfg3>4rh4G;xdZp@B-4pXsqSGjcIVtJ1CYOUvpe8f6EL2XhD+p0{r zwjYq9wgZ0=ssxoPTI-h78ui}zsLrn=5l!@ysng!tj>=0%3R)W^B*kXyU-r*GVE8%^ zGJ{3tf#^{=))ncc*3SV*ogSb^NJy@>^2>#8v{ihR{2oQ`Hq_-8NoL1xZDo0t#V%od zfJZ%`4?8;~umA+1tv6$6Dj5}c#-1c6fH+;*?IqT-xbLBO`!xI1BOpVLTsm{HJYUnK zaKNl1l@_;BSE_v5cB%4cs+>ZY`nE1%rjl}$#J(*q2f*e|$6<)j-t-=>rT=iKK39JP zX`|IQelM+VwS7*TdlRQV)7+h1tbbH3zV4uj(=83FUW7fEX;S-oxm?4Av%_d7st`w` z(VUCGJq7D#39Nr?qs=)`gkO{EQ-o_#5ZxAYNQ&K_S_G`$ zgR!Zi)I;4*3=f-?LF*S<^J0DW+PfK9+*LD)R~g}B;yyq1fyCoLQ%>TD1?H-?w&~cor#&so;pTKy7|g&c+l3t!lO|W7o6@DJBmE&RV0HJPmA9JNWZ@S=5ypk0A zT4bG;Q&h99&|A2zq~bR#^qzPq(Zcs6m~xGtD-)T$TdKwy#bsP&A-eQ^?TR01_Su?F z3!mhUPs!_PycYe9PGq5t?x-J(mTNptUTPPbfjjxg9=Mz@B?Qe&D};sb=~< zsGI@LChG9fAx7*ptCfI%q`+iXn*)>1x5B(yKt&F67J*&X79Q^_m)>09?P5;k3)FrO z?>5{`l_H;P28*=t$qc9Mwn+u^|HY!LP4NZlMI1nAzNYUkQDpApb308y?Qg?n7Z(Qc z1kee*v99?rkGL;|n}QYp3MT2bQ9wHMqLOEc_w=7AHVuw-DRwp*MF2=*6&B!@Q~XLpvm9U~VX7YNq=Cn4AY>e*N*o7j&Fws>Z@ zx~$BG+~LwMlcf-bemosuuV=Ev{#-6Axe%O+Nhn~=YhWhUGd4q&&Ap;?l3m9!EmgO) zvfV`(r95!#$*UVs#@cs!CP#~AR3RBlM&u!z^=!t*;{Qu1w~DXq$(yjNw>_K3$*_fu zk^Rl8;FJf-X)BpIIV_mJ$2)$XHc$VYA+*h7keiSWL!5vS(DYr?Ur+G{S zC0?^Q+MY(VT5XIU2}B&MSkB4GU1qN^?~DL1yykAWyy{0=2&2KD!n4Ka%Ef zAhS!zNQ&k|lQ_UX#H1NXF)ti|X#yuACuHc6e+)=w1Y6ABt_=?2iwtIcXjKZ3JBOPJ zgqtKz*C#Ng%T1R`8N}A;PnQvH3fh;DL%Hcm@J6&1T}1o*MT)S(V~oP^P=WA}fNU#Z zl!tgEV6K+jh;x{s%?;rb5LiX9F*`D6v!LXy{)sCqg16iH`%OvyfsnWOmy-N%_}NeA zqy$$w5cwVLcM;mJH2*-UytWpY&Aa119ZNp9A9dZe%qiF{Vbt&BvWbi3c_v-ibUi2D zaa4;)KP=%45>0OIGV3iSC36ZdOWqgcGF1UwZ2)Xm0PI{y1#oEsz^M)Ze3uK+DlmTi zcRJ!S;!C438v4kjLLY%!B$omOez1jNH%fs&+7$S%W-fNiOv>N*d{g@^`i3bcH~xn8 z6$V%uvCLY^G=lwcwjNA;oc+0n0$0jq2^WO+QZ4U2%4_T&tRO>VU(4ZVSfMBKa8eM5 z!X*r_R4`&Yais`^UHKgY0r<~Nx-DY_;7GZI4b-o{7aDWyII8s|shH(CUvx#1;3rZM zNb~x)u4>(xmQbzz)*3(8l<`Mrm#V9OWwdL>JeE8ZTK&i671gZ4VOUdAu`giJ*P6Ij zF+O6R9LTaxx4u=_pQVH43b`!iVtK~XT3|OF#SS%Y2k#Ul$AA;P8HUmNlLFlXI@tCV z^>S3$^eT$xJ&*@z%alYiq!wGq%b87Ou6Ywm2B?e$EvTDI7w&SMbcrTy$P1 z_A{KE92WiX8#?ayuVGkpPt@}&Zy6Tt@dCfZhupuXd0PISs@5cT>}w9mH;eDALB>mC zMHduV8AUaVX}u*S70{qsD_4?VKfH|`u+$O-6R;ZM^GYIh>EiRQl3AEm!HNY%*4fpv z@o@qK4|Um(wJWgo@?OkjLRjyMNLZ}f!C|7w1!T#^` zdI#pUZ*MO2It9C@^m>+OR_2s*p;zTeNqUXn1ida! z{E?w);SCRvZ7<%s=E6u70QV+=b!5Eg{z)JE~|9UZmNi43KR<0eT*XB1uuhyZ5>DTuusl4%;a7{5eU_POJV0dNTgy^v4l= znA2|cnG6X9>mxSIMoY7{3H4wU2PvIH;O!3gZGB`3Hzwz89HALL^p`lpC1t#8Qfbjw8Lv0UM9AG|Yfco#V2Z;ZE9h9RU`^790p-(n+ zus2hNb8t@gGZZv2XDn~xKKrNkX*zzz*cT&U_8ykd?24G-uM^_F;~P0e&G{If+X>If zI2Ot!?t6@^u6FtrX+qEn9Iu+K7RCj>8SZkW>;U3wylP&HJN)}>N5p!~k>R((kNR7G zmy%uVG|27lue6I+z)VcFR>wHms8&c{WPoyFSR~pzI|^CmB9MKqq>Sbdkg2r=UyD0o<0HUM5~XgWX8}Rk%F^+3?}C zaUD;Xch!;a5sHCu@#na$%ua zA0UfpBMaDnJ5fVh1n2&k9<;uY;jeb<;H?(F*zT|eQtPAPL?Db2(b(BD|vGikw zsmnkcQO;E>X1~PN8fHBT?U>}6AeV?__1tAR;ib=o6Yhf(QuyVvoi4w4P0HVJ5+j!w zP8QP&{344r%73V72s<$-L9c|pHx5Hejj`=4SE$GNu0C6sfUr<|BrQzf*mlUld;x11 zYK6A{7WaBHZ?%NLtUba_e=J{SyOZsu<$DVk{z>t=a)KyLJKrpc5;Bn}iE~?(oVX;| zqCxD$T%#Eewi-PfJK?Q94oOhkg0dB(8)2DGVAlhaz?}X3n?Q?s;qP?jsBEpVxN+af zJYZhfJU%t?*vkXlhLYHId9p-0VEq<~oqq9ED5*(^_)#Qkc9nRG4#a(z@ns<)Sz^9# z5LPjFs##ZjD>G7N3Ysa&C2`+rri>Szuj0Pf_=GZovD1IMlE$^ZF0p=-c*3zB?{QzV zA*k?w=nO5Q1CvL!DVFXQ63dL%6G|akbZ807 z6q5YMRLE(fIO))zmL*&+M=LOG$S{W+qwGp4!ePdP&RWbKTC8kKsPdFpUo^7Qp1lbx z#NM`~Stb-7ZgwX`+o_t+e#(Ko(T|kS17B=TS}C-rv69w)S}u>vnp#i)9jf+AJ`-V| z-&EK`+g)Kl!KC~RqB#X#gE?xZ&@K~DedW?y;iVgRTR2QGP+Q*=-tXI7cxz0`-(W+A z|H=RXt3%Bi#Czn?LK8;W8_urK+^N142 ztPe_})dvUgl@7m32jKTsqo-riSl2BkO^kJ&0Kr(_@)-)1Q&!|_LL)l{+phSPdR`}& zE4gsC8`VHaHm;V+E^k^{Z=$k4{H)dXV}##9X_6tbLdQ_go@Y$!Jt(b&Gobz+BWQlp zt?Rl#OgzVZM%ZxD@Xkj0QU<&TX)Ys%1kh$ipPj8NGs9Sl)QBOp_kMaH&SnX3bs-K- zVC+SVPn=ImPfl1Ph%$QZtzM)k6F5@*mU3g(u+FolQE%e^Z4x%$A^di=J!A*9u8_-h zTo_}dQdHx~^&D+4 z4}f)jPxl_P02^-4qQOUgXfAt%TEK!MYpj^c=mVR%;=GIcjH*4dcQ0WZl?=CYQTp8Lnp_1fa1yBHJ|f*U%w_PVIkk>GXSpHuj4ggBGZ5 zOwuXz33L)`$W9;oqF=NzF1OnY{3GPjRts5T3iDj5L^42! z#ayu#?UFNSQlgxCpyW|bTNkTxdg5apPjbt!ZxYBi$mLoE zf2;vN?R^1X-xU1S3GfX+YC-gidCU?kY`@54uUk{p~ zqrMpgy6xgn2g97|y9451k6~&hl1|A)#?Udyz14KlKEynSwcgCbr8)B)%ki zYlihxX|y6M@x!Ts`MD)>1N0=2SY#Z<7I zdzY_rTlhWPc0rRY9G{FUD&Dey9BrNFKwsNZ|2-hz+F zh`BoF1l^+fkRJGv2;TUzzFlg+e2Kn|nrRtNIge@hCsFEvXr3(l6sV9Iz4+3Uvb(rt z-$U7~10KOaBCP0q{WDsZp)iTtST4qbxQ#Q95I$QdT3VEugldqn4jJCtthAFQ(q?E+ z&e}j{*Eg<#CL`Vw2G?~Nh8Dd{nXtFdq)=@C_}0*X3amprQ-J{WV9@_t24_V3PXW>y z3h62eb*Nfp+80PmO@1I<=|F0H&&s@#_vG93Pr*POW6j6FB}W52Dc|Xs<|!uSZ@fRD z1x|@cPC2!Z}GpfW&RLzz_c{xV63tLQN9`yQNF$u zbpVMA0<}v;Ul}YGP~(ON<-VyUAm%sKd_AWQK*1W@TT{XV0?M3fqJ6QQ$3*+mlSB^* zD07>{OQs;G{TJ#Rs9hz%OnF~yR=QH(mrV*DIme{@km&_Z<%HWl;|FdF*P|znA#-Vd zX<1dPK;+&5&BAx$y_#TOY@fe>l}JY2$g9x%Bn>kZkqXqRU7%2x3I2)7a+%;?ZzP)G z>;ytB69`TJ#x#DzEi$?z126d^T8P*p$iess@lenpY-7xCka$*t5uJ4nh%$2}^8kC| z-kP<18u!hWTLPkNl$e}iqT;?cc|>58u5sV@q$IBiY&DIyCa~4(JeN^7d4f8N``+b& zB}YBReM6i)GDkFmCLfv-@r1oZ(49#V&sVe9Mj5+I!d3O-I87N)V;GQm7N;3{QbXu` zEAMbQp|Ih>7Cd%&@VQNn1ZF!Qxm7btpI4Q0^>) z7l?lIwaBq_ETJLXd=vK$g748OZjh~V!a8-Jbcm4U$*Y>p-WoaRw{I-{cq3fLbEl#7 z%qT%>v?gingm9gMi3!wpqhtrS5fp72vX4oD&bp1J8L7o(pHd6vfB8YUItX$e!9awo zC9>5#d`F6Cow7V3TE8!IMe8S0f!aK$NLRG_)0VDiO?fpzs`aL5X!M#%fli5oPQz`8 z7VA*F)p9g}L?0H>a`*pKWZs2d3RoLM`CAY$SG*)%90;g*$yWNf?*_S*ctc6?>c=A^ zUNxltM7l2KaTDpfOujSH)!JlH;ga4x(k1(eA!<2&ZCAQ}pDBb?>1sA#s(XlZ*&CEq zGkM2^oZ$?p@FI_0=?ahrs^6Z(n2al3jUyegvO`_qir)~cAd&-X(@ZR6zBf25od?8j zX}rD(CmbsX0gnDk7{Ijk&BV<%tZ`KR%6N|tE5NxYoiBvBy}FXuJz4%Es~h7R`k7_vaDu3$+8z_&dqYV`74A-ZoS zKd$^UrsJ${6C3Qr%lT!odZ|nTdIr5B;HM|S7aO2}KblnIbN@qZ9C1QSeIq?Ku%UZ*4;fRB2=B(*yoC=T&{P1LS0#TnwQU#G;| zM?VYVjjBr!FPpI>ClJop^C?7E+Oh6)6@wY{Pzl>cK$-{xYhRPT?Ju(DFrx8izIT&A;5-oS${=0yCxGDv1OzUq-~yU--+ImXlk%6z(l!9q`K z?wyy0U~lla3Bg32WTfmLa~w1saAnD&wUgBwGyjPY$~m*O$a+Cu7jN}jB=GvZ)oe3_ zin2s`q@)s#TCir1x27kLBw8dh){iZ6mt>SSV|GY-;`a5dqbS(5Fe}_uCmmw?#_nsL zHe*sd<~u_KOY+}{@9NH$i?z(g^-Nztb`!!V26+?J1o_!(0AQcYt*b%66?cJIXiDGwT;QtX>4PYxv|GQ7sr z?n_KYq9$1qxCMS9*fG(9;hfHIo$WE+P3td3)@=uku-R zl!2nO5Ja87RCufSFl1c~qxLEY<*zI87K`OcfYuxd_<+;;?6IGeY_Y5JLeb8pZ0)uYnre1IvVh~F)063>*vbbV^nu4}s9hh+$su5rh!coE4 zY3Bta2e2GF80lc1ESugQm|o_i|2Tg z@;7W0NX#Pp;IZJ{Ob)Z#NkbRg``-5<4-FUc0jFn6)Dcf)Pm8LS!S18V!ncZas)zAq z+}C`r%IKwTU_vKOp?wPo**D6iTv2<7LP2f566fTms9l^u?dO$>q+eAt+{nVdUi6lj zx{&#c5y5H zsIn3#oXwSiz+O=|dj|+AZENJhz(V%Idj7~dRySoYZU)ESkK22N@J{Oz2h+~qf9_cW?vEwmqe&qo>QXF^3tjveU5B8SG0 z?Hoa>vs{i*%mxT%u}%x7N&^#dEVY3%5*WR-!cdVqzSqeUJF%F2n)G@(Sek^wAnw}Y_@&M=E%%@a&qSw`o@W#YAQUynJ;#OEH$#4ie{ zX>xg*i{)9oTx)z#Ycy7X6WNm1s$`0k33nqu3zP-pzBGWsu;FfpLhUC2`^cq8p^h?8 zXDZYfP#g2ff+_$q5`b(4$;$M=?}z<_w;X_~_FT#VgIX^~%+ zX*&#HT2D0+O;gPyl$PKA?s-RRDw_)3lg*OCA&2O5mJZm9h08}>!_a*MOo_61pE24; ze5|y*!5*>-BrD`{9T&?pnKD7HR@8i~(^p`Vf@5EU5U4dQRcv3?S;9}Koufs&yO<_a zx+}j9egA{HUsmAg`n?E2mLPtgAg()sEnKed#RcB#g+g$m97|p4fecHTN9~uiPVxrYIY*^W+F8$krJC^;EheB?oOlzB~q6=DY|ks zE5V8F{14Uh8|!x%xN}0pRHfuCcTq0GONZpu7U6DwYBvScKO3V|=ixy)H0r3}?Hg~=@oLynlD6&x4 zzLHe(%UG}G`YGcmYz>q@&7_={>-e+62_~qS?#9d#Ny}&DqX?Ks zMjv1-x}}VREfTw$9DGaT-ql~9);s#E9i47h@YTvXa>2?7;sD8jvYq>eFD-Jgc@^C( zuN^szNjKKUoEW(;yIEDc$bB7&FDfr?abC2p@6}OEaI`Wz7`ZPC_Lfm|R9i;ex4jne zmm^1oImgN4nCI^E75!ca66lA~;m%mj`=A)89fJZxq8YCWaPOk;anbMprugQ%zQbb= zZ!HsoeF48X?6Gf_yI5vVF`~>?J@{L?^iXIX$DPoM(m}{TA!W(rX2I(|zEn>`o6+%t zRg0_sPL5xs#77BPTPS5C_CvQ`+7g@a;DJO75kAE8aCfcZP?=Two#UmqOE@U-hLCd!)WsmLkl0zX^&Zop4Zj^ergPYAjg?D2t-t4Zkd_ z8ew{46U|!G8bUPn%9XkQvGR?#XtjJlP14a`>=w8*NGJ<((5~9JvdV&HEEkE?4g=9u zlJ+;!W8eX(7t}4zRYU7NrQ8a`+B367sg9JXnfLQLpg6n9hx`RzO=XBs4qCy*E&|V z&mS|1H(lk;=gymc=1tX)f!eq8k=<0m19BXA)I0ee$gnF|wMCPL@Vq~(2b15QFI|4)uU-kjGD;KTZg03_L$ugtV zc;i-yEB`Io{WJ>o1TVI+ujUR`JsX28tU8)SW?&g{yWOUL9*E@ecD)>^6(|!*auAIK zOZ^)l4vVi!GJa+g*l~}-Uboo5_8$UU+Y5Vdf5~@RXzu*PctB*IXE>W~GCw+ge>_+* z&zd|g{Tq}=i`DUOC&XG~MY2jA&y>f*wE)@q)>c1&H?MnV%$Nvic58?kk{BOdq{uF+_ zLJr=<$IAyN<-dm@}%^`qcoEH0xhhYlf0$>uL1FHZT8 zW1s!y4W@kbkBY^Ksq4j!e7xs;^8d9wZQp_t*3n+_yG)?hRmo(}=&%wlg~;X0Cj2tX1o78$w+c6?#8N>wSKDMpawl zAQuraaXUeIJjd2W)-m@K&XD*FBFo4tvf48vlwVibhRGcagFYXYsk><1TZ;2bqqGTj zzsg_rrHDF1GgtY;>x20{{Z;g>1G(<|+^P@V_qpL(-se`m#CwFudGg*1&^&KVB|;%Cois&6ia8^ZwtIN#VOy?sVXwx^5P6qA*+P* z02V9Uptt5&9&xL1IjiGn`69m$FPcj(qr>#uGH=ZBrY`t9QIvEZBKt)A=LwhdaBWCh}@@!pEoxui_WSGF_$`c*gJt+sld);(~Vw|YK$ z2~;w~DPYhkv41e5=lN|pw1w7`c-Abgl)o`nh|TyHBm2Iac2s z?q1~X1FL5)nuxn6xSK)A+K4h}mseFS#M5N>pA3aJfd=^E~)&#geV)khj?IRyVkR_FUgQ=8iAY8|_|Q~-{vo=htTtx~_e zqrrJ1ONcd(ec)!PPN|>06ZWr~-Xm4GLj7VL-(r#aOvxt!PS@FQNV3a(?+0zpZ`(qc za6U{(lr5NWab$g5wt@eo#jUX4QE=qiVZUr3H%=Fm*q%x@EO2qU#^6--li-tUm%1p9 zF(2gG`Q}M@p!pzV5XS25unUq12LE3&kj!Wo>oUmJHpec8A-XLVX^f{D0RlH+&2$H?h;VCJN(LB zydS{ndqn%47M~lXG#mx-5~%hLMqSBw7PIhomPYe{vKkXMk=%-7?Gy za_DnrzE~EQAwH-2IX}ppc$)o#Pht^mmI_=#1umw>_~>80tad#Qf;hzLPDKK>o2i-c zCk}%`t^>XJL&vUXKj)fc4Lo->tM^KCPCB6<|O06nD+3r3?PRI%0Q9{srzzz;ZMr5T88yGOk>M97~qv{Z11?xdzgfoR6 zRGiLAh7w|Df)Hs`-X%oc2M!^;&r0OuWw;A=HMkHLaVnans2jGGW8l7l@RLd+V;IVK zKnAT!tc$za6K`~D?yY`@hf@0iKbYEsDz#>sc9~r@z;N3b(a%RH1*qRZ=P8}jQDtf2 zUN9VkBdce~*r-moRQIBM(~G_oP>bvrpZ4)i2^k`O#HD4T!zb*OQHIxZ#!`L}@_uxZ z!l%P0$n^w0jg%d#VE5CY1jaQNBaWrkxl+BF6G)d>*QU|E(B8xK+IzTKdk>chX3}x! zq(zyILuc9lD`i6I1v}|Gm^&V%+2;sZVwo?^HX_pg`$M}A>1>J=!gS^Aq{@pG$+Ce^7At$6 zh9%&P_ogdn_*f$2aK`2Di->eM!WEHk-%E1FOQhBH{q5-lXWRlBxSS6F^JQ6R6Z4cC zm~-XFaNARGRG>B|0rSxmn3uRPpHISUuP__VqzN27;r4HPnac3S(1|NWxN*Hxtu zueBjwJC;L$UG+Z_#9Q#L5KnBn(OG}v-W63JX>E`}PpQsoC{O|E4RcYxDp|b%rLo^$1g{%c&p#!sngG0VUPMryka#+ zf3-5tA{Wp;ps543$GXs1Y0SF)H{Jm^jJ}n5r%Cx6`x)RtQ2WAectTM7tOSW2woSLm z$zh|t6qH0+br8GG@CWwo8Uds0{*E&#Kg{+WQ_|3@KPyd`#JKiJ$RO_7a{GS9HEZG{ zvG^OpU6Q8jRvsruKd$ADc0U(wDD6Imm>b5SD;G;YL}X|v?OdSyZpCI1|AbZaCTKYs zPa>n8@sqHZl0;6StIbg1zJv^M0^QzT@{Hi^He z&jQi0lI5@lwm zl)3ILr_56*vw@k0G-FJs7~p;vc$NX)@T^k$Csrvq6>8K99hU(95dhr^jdFo6c>>@O z7r0EhkFd|B1jk(Wu`;IvhB`b6*3{vCDMkopxacNlE_MiU|h`k3j4W4302Pqt=oaO z&08ZosxcMu;_Slicjkr%aLul|glosD5S|yZcAmX|q@$goLsjZ!2C>YKj!<|;MfvYo|z4dslL1$BLRAY4|TBCI|}%2^t3*iXSkuB7v7CeyaoH)AT4N{J!aVu;yC zfX96Kz;ML1Tr2fVqEc;BDz)r&r&*45ikLzXL8~(-e2D@*$&C+#C!R-L!VGy_TELFr zRqWS3-too1t1r$L3ak4e*d?feY@E0R6g}N3lEJd$1VviP2@)GIg)=->L<+(@^pxU$ zAMFyTeHDaNtN86Pf7kj;Cl0w;Y0trHGNnQAL}{re8BQ326BUYO&f87AwOCm}tj}Wm z13YAhT1(HAGDH9zDm`wH!QP}N(0cPVLmg0yJfWz44|TCO+a5IyANh31{8?kEI|ZEz zQabbepnb6^=dZL%g_a{rA)Rv0NR;!8fll>%JLNP!E_^&x{0OFeob$#}Y~`Xp+1t;W z3L>(UPmvwN-#3XTL*V9kAFJb;v{P0jk7?!awLzG7qR|dx7~lJlY>y~nAt?{qSsC`L ztDB)AR&ndC-m4$?l|?7HIL&ejcmX;iXrkuHEj|^4IwZQ1$dd#g;#raKmsm^e+Zfq2 zO4eKbkW|*X6n(NL%pB>W_!?1|(p7*woc|q&1rM{kERfmXzFE;`2IVyG#2n4nD_bx; zfl5?$l2yzrY(I*aPqX#YY^lq{*9BqXw{yfwq&_1!E-rmX@BRC9U=HOgqj};KODHV# zV!3Rm7%O?O?jwX{hDrzmnSvT3PVuh(!VJ-5Bh{J)ff?Gpsw_YwJ^zYfhQ=Gn(N-5s zJ(|q1-+V`oH_UhBxYNyn-|2Z(nfjSzj>U-_qY^oKI5}XrBOgw(jZOeNBLQs33hS(wDU)r+blr`5 zy`1t~$Dt?bde16$+{vG+*YOW0q%0sY24b0SU}0qB4}oAX&3*vC&2XJ4*~vhl^s|oD z(sv5|q^iQSR%Sb?n)(y#Usff13}^}Wia${UM)uKC8cO!$R???!h4&?{7l$^5qIckZ z-H~k%_lWn^TfHyQxSdf~wA2LCQj0}H7}_qOFo>O~=Tj8v*7GXTTDw;sGUZ8BXXAi% zFz^u^j0W%@rtN{BYvA9Ugnx^|e>4IA{1o{6UUJ}q&d7k7xEhXkF|Nu?Cqz+Dd35>f>XIU&edm zC@Pwj2^^}%idHWxwN4SQ%~xWj6K(|y8|pqJYgo&GZcfncaL)+-M8uQBCmKCR=P7g2!%$GbI1_ti~C-U+=xS^`d*!4kzh&B44% z>cjbD26?;oQkQyFkCa+D9C`e`w`P!fxyLaf`~N;zvV(y^H5T}nI#{CPX))iFEjm~t zck*XvNR48dBhkG=)&^qrh{aFNwSGy4LY6(4B@=6n5A^LE9E^2PI?CaGBLC}QQh0gi zBk(foe_WR4yuKGjD%A8g%28Fx^E}u>3{?SY=8mwWD?lj3hb2xuv=1q{lx_~(g9XsFzeG^ zfweKfLP=mTG6ZU0OaMDN1=!awIMqDO0oGoCv88I+Hw>hEuHsYNv~Q%+NjGhy$%K7) zmJ}6uVv>?}-J@bsJ_Ul~rhOST)s>wQ&~6eq@%7E->sR^OQI8P_Z?=FN+KmFNOS^m* z!v`H;vaLuk{EK23Br6!cfl5rl@caaZzdx@S{`Rn9c&=h7d)plh&vAfu2B_t^Rjbo! zhrx3$c~W3*1rq2{>lvirqy{J*sGXApW=}BH6~~I|UP)n8_a0O9N763j$I_-#OrZ7` zGMeg61fE;nkSWD)KkWb;Yk>7Pz)m;7?nnYF*4F(k0qmj_VDV{A>t1YtHJ-}>bWw(N zkfI!2yNP~jS85U2?}`j-X(_|f>d50GUBow48|!=EQT*{0_8GS5O)<+d$E7o&L&Rd!Y$u9%db)$9`2RxbBx6MkzL20mEYf2#dp#H`-$I;pM%$Q5sTxx68O=j= zQGUcj$Nj&|%6x_LXtGW1aISWxcnJ%hgMw=D-X#S_8z&lzC+$yL1ZrQx(t*`ET>xrt zeLz;*Eo>;HInZ36h?S&z<@UBOU4zEVRLt{|;jo7R2B2xusD?Fupp-!ETfj7|aToV4 zYn&yx6z2Htv6KRq*1-m64lWqYv)gQ;gsA%WT0}cj#A&989?2r^*CO6e6mfV;5i6c` zis+OmLPk$4&)?XY(rs6LFTc<&+UvA39p~@^)F-9vpMm=5mAbM*~v@+MeUZ#O$ z;TbyE9Gx8VTx;0(SxYPYPiafs4(SL1QW*GG1}0vS z8at!8J{c@2fR_C?#N+DLEG<>zaRHiCjK}NCMs2y>oEnJ8&?zZT8GJIKBR;@IQqW~Sag4EAns6GZ7Cg!<4CNgqI61udnVeQ9( z**Hz;S_cXtYeirtQlnTXXQ~a{t>%0Dx=Yt?1;`TfIDgPqRf4abvbBzWfrQ(1Ut;!`dG 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�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
    2dzK*)zj;p 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 z%0A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Ê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%0Dc1*lYae0b(GV18*s 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%0D%`=P9i@mh+Ymme4+c) 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%0DE<&4#{#&5GUug*O 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
  • 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%0AHCD!}Ym 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 diff --git a/sandag_abm/src/main/resources/runDriver.cmd b/sandag_abm/src/main/resources/runDriver.cmd deleted file mode 100644 index 7ace356..0000000 --- a/sandag_abm/src/main/resources/runDriver.cmd +++ /dev/null @@ -1,17 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ############ PARAMETERS ############ - -rem ############ JPPF DRIVER ############ -set JPPF_LIB=%PROJECT_DIRECTORY%\application\* -set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% - -start %JAVA_64_PATH%\bin\java -server -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-driver.properties -Djppf.config=jppf-driver.properties org.jppf.server.DriverLauncher diff --git a/sandag_abm/src/main/resources/runHhMgr.cmd b/sandag_abm/src/main/resources/runHhMgr.cmd deleted file mode 100644 index 2fd44c6..0000000 --- a/sandag_abm/src/main/resources/runHhMgr.cmd +++ /dev/null @@ -1,54 +0,0 @@ -rem @echo off - -rem %1 is the project directory -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem JVM allocations -set MEMORY_HHMGR_MIN=%MEMORY_HHMGR_MIN% -set MEMORY_HHMGR_MAX=%MEMORY_HHMGR_MAX% - -rem Running on SAG02 -set HOST_IP_ADDRESS=%HHMGR_IP% - -rem 1129 used to calibrate the 20% sample -set HOST_PORT=%HH_MANAGER_PORT% - -rem (X:) is mapped to \\w-ampdx-d-sag01\C -rem set DRIVE=%MAPDRIVE% -set DRIVE=%PROJECT_DRIVE% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -rem set RUNTIME=%DRIVE%%PROJECT_DIRECTORY% -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - -set JAR_LOCATION=%RUNTIME%/application - -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%;%JAR_LOCATION%\* - - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_64_PATH%\bin;%OLDPATH% - -rem ### Change current directory to RUNTIME, and issue the java command to run the model. -rem Note: Running java script in separate window to properly redirect console output -start %PROJECT_DIRECTORY%\bin\runHhMgr_log.bat %JAVA_64_PATH% %MEMORY_HHMGR_MIN% %MEMORY_HHMGR_MAX% %CLASSPATH% %HOST_IP_ADDRESS% %HOST_PORT% %RUNTIME% -rem start %JAVA_64_PATH%/bin/java -server -Xms%MEMORY_HHMGR_MIN% -Xmx%MEMORY_HHMGR_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% -port %HOST_PORT% -rem java -Xdebug -Xrunjdwp:transport=dt_socket,address=1044,server=y,suspend=y -server -Xmx12000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% - -rem ### restore saved environment variable values, and change back to original current directory -set PATH=%OLDPATH% - diff --git a/sandag_abm/src/main/resources/runHhMgr_log.bat b/sandag_abm/src/main/resources/runHhMgr_log.bat deleted file mode 100644 index ae594fa..0000000 --- a/sandag_abm/src/main/resources/runHhMgr_log.bat +++ /dev/null @@ -1,18 +0,0 @@ -rem @echo off - -rem ### Declaring required environment variables -set JAVA_64_PATH = %1 -set MEMORY_HHMGR_MIN = %2 -set MEMORY_HHMGR_MAX = %3 -set CLASSPATH = %4;%5;%6;%7 -set HOST_IP_ADDRESS = %8 -set HOST_PORT = %9 -shift -set RUNTIME = %9 - - -rem ### Running household manager and redirecting output to {PROJECT_DIRECTORY}\logFiles\hhMgrConsole.log -2>&1 (%JAVA_64_PATH%/bin/java -server -Xms%MEMORY_HHMGR_MIN% -Xmx%MEMORY_HHMGR_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_hh.xml org.sandag.abm.application.SandagHouseholdDataManager2 -hostname %HOST_IP_ADDRESS% -port %HOST_PORT%) | %RUNTIME%\application\GnuWin32\bin\tee.exe %RUNTIME%\logFiles\hhMgrConsole.log - -rem ### Exit window -exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runMtxMgr.cmd b/sandag_abm/src/main/resources/runMtxMgr.cmd deleted file mode 100644 index 6503f3a..0000000 --- a/sandag_abm/src/main/resources/runMtxMgr.cmd +++ /dev/null @@ -1,43 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem call ctramp properties -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem JVM allocations -set MEMORY_MTXMGR_MIN=%MEMORY_MTXMGR_MIN% -set MEMORY_MTXMGR_MAX=%MEMORY_MTXMGR_MAX% - -rem Run the matrix manager on SAG1 -set HOST_IP_ADDRESS=%MAIN_IP% - -rem kill java tasks -taskkill /F /IM java.exe - -rem run ping to add a pause so that taskkill has time to fully kill java processes -ping -n 10 %MAIN% > nul - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set CONFIG=%PROJECT_DIRECTORY%/conf - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set OLDPATH=%PATH% -set CLASSPATH=%CONFIG%;%JAR_LOCATION%\* - -rem java -Dname=p%2 -Xdebug -Xrunjdwp:transport=dt_socket,address=1049,server=y,suspend=y -server -Xms8000m -Xmx8000m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j_mtx.xml org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %HOST_MATRIX_PORT% -label "SANDAG Matrix Sever" -rem Note: Running Java script in separate window to properly redirect console output -start %PROJECT_DIRECTORY%\bin\runMtxMgr_log.bat %JAVA_64_PATH% %MEMORY_MTXMGR_MIN% %MEMORY_MTXMGR_MAX% %JAR_LOCATION% %HOST_IP_ADDRESS% %MATRIX_MANAGER_PORT% %PROJECT_DIRECTORY% -rem start %JAVA_64_PATH%\bin\java -server -Xms%MEMORY_MTXMGR_MIN% -Xmx%MEMORY_MTXMGR_MAX% -Dlog4j.configuration=log4j_mtx.xml -Djava.library.path=%JAR_LOCATION% org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %MATRIX_MANAGER_PORT% -ram 1500 -label "SANDAG Matrix Server" - -set CLASSPATH=%OLDCLASSPATH% -set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runMtxMgr_log.bat b/sandag_abm/src/main/resources/runMtxMgr_log.bat deleted file mode 100644 index 7b5f6b0..0000000 --- a/sandag_abm/src/main/resources/runMtxMgr_log.bat +++ /dev/null @@ -1,17 +0,0 @@ -rem @echo off - -rem ### Declaring required environment variables -set JAVA_64_PATH = %1 -set MEMORY_MTXMGR_MIN = %2 -set MEMORY_MTXMGR_MAX = %3 -set JAR_LOCATION = %4 -set HOST_IP_ADDRESS = %5 -set MATRIX_MANAGER_PORT = %6 -set PROJECT_DIRECTORY = %7 - - -rem ### Running matrix manager and redirecting output to {PROJECT_DIRECTORY}\logFiles\mtxMgrConsole.log -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_MTXMGR_MIN% -Xmx%MEMORY_MTXMGR_MAX% -Dlog4j.configuration=log4j_mtx.xml -Djava.library.path=%PROJECT_DIRECTORY%\application org.sandag.abm.ctramp.MatrixDataServer -hostname %HOST_IP_ADDRESS% -port %MATRIX_MANAGER_PORT% -ram 1500 -label "SANDAG Matrix Server" 2>&1 | %PROJECT_DIRECTORY%\application\GnuWin32\bin\tee.exe %PROJECT_DIRECTORY%\logFiles\mtxMgrConsole.log - -rem ### Exit window -exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandag01.cmd b/sandag_abm/src/main/resources/runSandag01.cmd deleted file mode 100644 index 9bcf0be..0000000 --- a/sandag_abm/src/main/resources/runSandag01.cmd +++ /dev/null @@ -1,16 +0,0 @@ -@echo off - -rem ############ PARAMETERS ############ -set DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -rem ############ JPPF DRIVER ############ -set JPPF_LIB=%PROJECT_DIRECTORY%\application\* -set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -%DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -start %PROJECT_DIRECTORY%\bin\runSandag01_log.bat %JAVA_64_PATH% %CLASSPATH% %PROJECT_DIRECTORY% -rem start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag01.properties -Djppf.config=jppf-sandag01.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag01_log.bat b/sandag_abm/src/main/resources/runSandag01_log.bat deleted file mode 100644 index 93ae263..0000000 --- a/sandag_abm/src/main/resources/runSandag01_log.bat +++ /dev/null @@ -1,13 +0,0 @@ -rem @echo off - -rem ### Declaring required environment variables -set JAVA_64_PATH = %1 -set CLASSPATH = %2;%3 -set PROJECT_DIRECTORY = %4 - - -rem ### Running master node and redirecting output to {PROJECT_DIRECTORY}\logFiles\sandag01Console.log -2>&1 (%JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag01.properties -Djppf.config=jppf-sandag01.properties org.jppf.node.NodeLauncher) | %PROJECT_DIRECTORY%\application\GnuWin32\bin\tee.exe %PROJECT_DIRECTORY%\logFiles\sandag01Console.log - -rem ### Exit window -exit 0 \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandag02.cmd b/sandag_abm/src/main/resources/runSandag02.cmd deleted file mode 100644 index 4171880..0000000 --- a/sandag_abm/src/main/resources/runSandag02.cmd +++ /dev/null @@ -1,15 +0,0 @@ -@echo off - -rem ############ PARAMETERS ############ -set DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -rem ############ JPPF DRIVER ############ -set JPPF_LIB=%PROJECT_DIRECTORY%\application\* -set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -%DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag02.properties -Djppf.config=jppf-sandag02.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag03.cmd b/sandag_abm/src/main/resources/runSandag03.cmd deleted file mode 100644 index 97591ae..0000000 --- a/sandag_abm/src/main/resources/runSandag03.cmd +++ /dev/null @@ -1,15 +0,0 @@ -@echo off - -rem ############ PARAMETERS ############ -set DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -rem ############ JPPF DRIVER ############ -set JPPF_LIB=%PROJECT_DIRECTORY%\application\* -set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -%DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag03.properties -Djppf.config=jppf-sandag03.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandag04.cmd b/sandag_abm/src/main/resources/runSandag04.cmd deleted file mode 100644 index 42aa4fb..0000000 --- a/sandag_abm/src/main/resources/runSandag04.cmd +++ /dev/null @@ -1,15 +0,0 @@ -@echo off - -rem ############ PARAMETERS ############ -set DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -rem ############ JPPF DRIVER ############ -set JPPF_LIB=%PROJECT_DIRECTORY%\application\* -set CLASSPATH=%PROJECT_DIRECTORY%\conf;%JPPF_LIB% - -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -%DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -start %JAVA_64_PATH%\bin\java -server -Xms16m -Xmx16m -cp "%CLASSPATH%" -Dlog4j.configuration=log4j-sandag04.properties -Djppf.config=jppf-sandag04.properties org.jppf.node.NodeLauncher diff --git a/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd b/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd deleted file mode 100644 index 05e804b..0000000 --- a/sandag_abm/src/main/resources/runSandagAbm_MAAS.cmd +++ /dev/null @@ -1,59 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start -ping -n 10 %MAIN% > nul - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem ### TNC Fleet Model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.maas.TNCFleetModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for TNC outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% TNC %ITERATION% - -rem ### Household AV Allocation Model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.maas.HouseholdAVAllocationModelRunner %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for AV outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% AV %ITERATION% - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd b/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd deleted file mode 100644 index dd526a2..0000000 --- a/sandag_abm/src/main/resources/runSandagAbm_MCDiagnostic.cmd +++ /dev/null @@ -1,55 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 -set SEED=2354345 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat -set PROPERTIES_NAME=sandag_abm_mcd - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -rem run ping to add a pause so that hhMgr and mtxMgr have time to fully start -ping -n 10 %MAIN% > nul - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem ## works for both single node and distributed settings; modified jppf-clientDistrubuted.properties to handle both single and distributed settings## -%JAVA_64_PATH%\bin\java -showversion -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.utilities.RunModeChoice %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -sampleSeed %SEED% - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% - -rem kill java tasks -taskkill /F /IM java.exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd deleted file mode 100644 index 95810a6..0000000 --- a/sandag_abm/src/main/resources/runSandagAbm_SDRM.cmd +++ /dev/null @@ -1,68 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%JAR_LOCATION%\GnuWin32\bin;%OLDPATH% - -rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start -ping -n 10 %MAIN% > nul - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem ## works for both single node and distributed settings; modified jppf-clientDistrubuted.properties to handle both single and distributed settings## -%JAVA_64_PATH%\bin\java -showversion -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% -Djppf.config=jppf-client.properties org.sandag.abm.application.SandagTourBasedModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -sampleSeed %SEED% -luAcc false 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\sdrmConsole_%ITERATION%.log - -rem ### Build trip tables -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.application.SandagTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for Resident Model outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% SDRM %ITERATION% - -rem ### Internal-external model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.internalexternal.InternalExternalModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Build internal-external model trip tables -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.internalexternal.InternalExternalTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for IE outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% IE %ITERATION% - -rem kill java tasks -rem taskkill /F /IM java.exe - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd deleted file mode 100644 index c3c1283..0000000 --- a/sandag_abm/src/main/resources/runSandagAbm_SEM.cmd +++ /dev/null @@ -1,57 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -rem run ping to add a pause so that hhMgr and mtxMgr have time to fully start -ping -n 10 %MAIN% > nul - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem Special Event model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.specialevent.SpecialEventModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem Build Special Event model trip tables -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -Djava.library.path=%TRANSCAD_PATH% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.specialevent.SpecialEventTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - - -rem kill java tasks -taskkill /F /IM java.exe - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd b/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd deleted file mode 100644 index d243f44..0000000 --- a/sandag_abm/src/main/resources/runSandagAbm_SMM.cmd +++ /dev/null @@ -1,86 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%JAR_LOCATION%\GnuWin32\bin;%OLDPATH% - -rem ### Run ping to add a pause so that hhMgr and mtxMgr have time to fully start -ping -n 10 %MAIN% > nul - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem ### Airport model - SAN -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport SAN 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\airportSANModelConsole_%ITERATION%.log - -rem ### Build airport model trip tables - SAN -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport SAN - -rem ### Checking for SAN outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% SAN %ITERATION% - -rem ### Airport model - CBX -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport CBX 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\airportCBXModelConsole_%ITERATION%.log - -rem ### Build airport model trip tables - CBX -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.airport.AirportTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% -airport CBX - -rem ### Checking for CBX outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CBX %ITERATION% - -rem ### Cross-border model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.crossborder.CrossBorderModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\crossBorderModelConsole_%ITERATION%.log - -rem ### Build cross-border model trip tables -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.crossborder.CrossBorderTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for CBM outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% CBM %ITERATION% - -rem ### Visitor model -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_CLIENT_MIN% -Xmx%MEMORY_CLIENT_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.visitor.VisitorModel %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% 2>&1 | tee.exe %PROJECT_DIRECTORY%\logFiles\visitorModelConsole_%ITERATION%.log - -rem ### Build visitor model trip tables -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.visitor.VisitorTripTables %PROPERTIES_NAME% -iteration %ITERATION% -sampleRate %SAMPLERATE% - -rem ### Checking for Visitor outputs -call %PROJECT_DIRECTORY%\bin\CheckOutput.bat %PROJECT_DIRECTORY% Visitor %ITERATION% - -rem kill java tasks -rem taskkill /F /IM java.exe - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd b/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd deleted file mode 100644 index 79a20c9..0000000 --- a/sandag_abm/src/main/resources/runSandagBikeLogsums.cmd +++ /dev/null @@ -1,45 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem rem build bike logsums -%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_BIKELOGSUM_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagBikePathChoiceLogsumMatrixApplication %PROPERTIES_NAME% -if ERRORLEVEL 1 goto DONE - -:done -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% - - diff --git a/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd b/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd deleted file mode 100644 index aa243b7..0000000 --- a/sandag_abm/src/main/resources/runSandagBikeRouteChoice.cmd +++ /dev/null @@ -1,38 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set SAMPLERATE=%3 -set ITERATION=%4 -set PROPERTIES_NAME=sandag_abm - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - -rem ### Set the name of the properties file the application uses by giving just the base part of the name (with ".xxx" extension) -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem run bike assignment -%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_BIKEROUTE_MAX% -XX:-UseGCOverheadLimit -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagBikePathChoiceEdgeAssignmentApplication %PROPERTIES_NAME% %SAMPLERATE% %ITERATION% -if ERRORLEVEL 1 goto DONE - -:done -rem kill java tasks -rem taskkill /F /IM java.exe - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd b/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd deleted file mode 100644 index 248541c..0000000 --- a/sandag_abm/src/main/resources/runSandagWalkLogsums.cmd +++ /dev/null @@ -1,45 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem rem build walk skims -%JAVA_64_PATH%\bin\java -showversion -server -Xmx%MEMORY_WALKLOGSUM_MAX% -cp "%CLASSPATH%" -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.active.sandag.SandagWalkPathChoiceLogsumMatrixApplication %PROPERTIES_NAME% -if ERRORLEVEL 1 goto DONE - -python %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python\calculate_micromobility.py --properties_file %PROJECT_DRIVE%%PROJECT_DIRECTORY%\conf\sandag_abm.properties --outputs_directory %PROJECT_DRIVE%%PROJECT_DIRECTORY%\output --inputs_parent_directory %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -:done -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% diff --git a/sandag_abm/src/main/resources/runTransitReporter.cmd b/sandag_abm/src/main/resources/runTransitReporter.cmd deleted file mode 100644 index 3162c79..0000000 --- a/sandag_abm/src/main/resources/runTransitReporter.cmd +++ /dev/null @@ -1,48 +0,0 @@ -rem @echo off - -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set THRESHOLD=%3 -set TOD=%4 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% -call %PROJECT_DIRECTORY%\bin\CTRampEnv.bat - -rem ### First save the JAVA_PATH environment variable so it s value can be restored at the end. -set OLDJAVAPATH=%JAVA_PATH% - -rem ### Set the directory of the jdk version desired for this model run -rem ### Note that a jdk is required; a jre is not sufficient, as the UEC class generates -rem ### and compiles code during the model run, and uses javac in the jdk to do this. -set JAVA_PATH=%JAVA_64_PATH% - -rem ### Name the project directory. This directory will hava data and runtime subdirectories -set RUNTIME=%PROJECT_DIRECTORY% -set CONFIG=%RUNTIME%/conf - - -set JAR_LOCATION=%PROJECT_DIRECTORY%/application -set LIB_JAR_PATH=%JAR_LOCATION%\* - -rem ### Define the CLASSPATH environment variable for the classpath needed in this model run. -set OLDCLASSPATH=%CLASSPATH% -set CLASSPATH=%CONFIG%;%RUNTIME%;%LIB_JAR_PATH%; - -rem ### Save the name of the PATH environment variable, so it can be restored at the end of the model run. -set OLDPATH=%PATH% - -rem ### Change the PATH environment variable so that JAVA_HOME is listed first in the PATH. -rem ### Doing this ensures that the JAVA_HOME path we defined above is the on that gets used in case other java paths are in PATH. -set PATH=%JAVA_PATH%\bin;%OLDPATH% - -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY% - -rem TransitTimeReporter -rem %JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.reporting.TransitTimeReporter %PROPERTIES_NAME% -threshold %THRESHOLD% -period %TOD% -outWalkFileName walkMgrasWithin%THRESHOLD%Min.csv -outDriveFileName driveMgrasWithin%THRESHOLD%Min.csv -%JAVA_64_PATH%\bin\java -server -Xms%MEMORY_SPMARKET_MIN% -Xmx%MEMORY_SPMARKET_MAX% -cp "%CLASSPATH%" -Djxl.nowarnings=true -Dlog4j.configuration=log4j.xml -Dproject.folder=%PROJECT_DIRECTORY% org.sandag.abm.reporting.TransitTimeReporter %PROPERTIES_NAME% -threshold %THRESHOLD% -period %TOD% -outWalkFileName walkMgrasWithin%THRESHOLD%Min.csv - -rem ### restore saved environment variable values, and change back to original current directory -set JAVA_PATH=%OLDJAVAPATH% -set PATH=%OLDPATH% -set CLASSPATH=%OLDCLASSPATH% diff --git a/sandag_abm/src/main/resources/sandag_abm.properties b/sandag_abm/src/main/resources/sandag_abm.properties deleted file mode 100644 index a93d575..0000000 --- a/sandag_abm/src/main/resources/sandag_abm.properties +++ /dev/null @@ -1,1357 +0,0 @@ -#SANDAG ABM Properties -#Software Version -version=${version} -############################################################################################################################################################################# -# -# CLUSTER PROPERTIES: MODIFY WHEN CHANGING CLUSTER CONFIGURATION OR MOVING TO NEW CLUSTER. -# -############################################################################################################################################################################# -RunModel.MatrixServerAddress=${matrix.server.host} -RunModel.MatrixServerPort=${matrix.server.port} -RunModel.HouseholdServerAddress=${household.server.host} -RunModel.HouseholdServerPort=${household.server.port} - -############################################################################################################################################################################# -# -# RUN PROPERTIES: MODEL COMPONENT SWITCHES -# -############################################################################################################################################################################# -#set sample rates -sample_rates=0.2,0.5,1.0 - -#highway assignment convergence criteria -convergence=0.0005 - -#set warm up inputs -RunModel.useLocalDrive = true -RunModel.skipInitialization = false -RunModel.deleteAllMatrices = true -RunModel.skip4Ds = false -RunModel.skipInputChecker = false -RunModel.skipCopyWarmupTripTables = false -RunModel.skipCopyBikeLogsum = false -RunModel.skipBikeLogsums = true -#always create walk logsum (walk to TAP file); modification from GUI disallowed -RunModel.skipCopyWalkImpedance = true -RunModel.skipWalkLogsums = false - -#build networks -RunModel.skipBuildNetwork = false - -# start looping -# set startFromIteration to 4 if only want -# to run final hwy and transit steps -RunModel.startFromIteration = 1 -RunModel.skipHighwayAssignment = false,false,false -RunModel.skipTransitSkimming = false,false,false -RunModel.skipTransponderExport = false,false,false -RunModel.skipCoreABM = false,false,false -RunModel.skipOtherSimulateModel = false,false,false -RunModel.skipMAASModel = false,false,false -RunModel.skipSpecialEventModel = true,true,true -RunModel.skipCTM = false,false,false -RunModel.skipEI = false,false,false -RunModel.skipTruck = false,false,false -RunModel.skipTripTableCreation = false,false,false -# end looping - -RunModel.skipFinalHighwayAssignment = false -RunModel.skipFinalHighwayAssignmentStochastic = true -RunModel.skipFinalTransitAssignment = false -RunModel.collapseOnOffByRoute=false -RunModel.skipLUZSkimCreation = true -RunModel.skipVisualizer = true -RunModel.skipDataExport = false -RunModel.skipDataLoadRequest = false -RunModel.skipDeleteIntermediateFiles = false -RunModel.MatrixPrecision = 0.0005 -# minimual space (MB) on C drive -RunModel.minSpaceOnC =250 - -TNC.totalThreads=10 - -############################################################################################################################################################################# -# -# LOGGING PROPERTIES: USE FOR TRACING HOUSEHOLDS OR AGENTS THROUGH SIMULATION. -# -# Note that the way that logging works right now, the trace zones also have to be valid transit stops or the code will crash. Check the skims to make sure they exist. -# Turn off trace debugging in routine model runs to speed things up (comment out Debug.Trace.HouseholdIdList) -# -############################################################################################################################################################################# -# Resident models -Trace = false -#Trace.otaz = 1638 -#Trace.dtaz = 2447 -Trace.otaz = -Trace.dtaz = -Seek = false -Process.Debug.HHs.Only = false -Debug.Trace.HouseholdIdList= - -# Internal-External models -internalExternal.seek = false -internalExternal.trace = 50 - -# Cross-Border models -crossBorder.seek = false -# trace by tourId -crossBorder.trace = 12 - -# Visitor models -visitor.seek = false -#trace by tourId -#visitor.trace = 742 -visitor.trace = 742 - -# Special event models -specialEvent.seek = false -specialEvent.trace = 5855 - -# Trace TransCAD trip table creation by TAZ (to/from); only applies to SD resident model -tripTable.trace=4384 - -RunModel.LogResults = true - -############################################################################################################################################################################# -# PATH PROPERTIES: MODIFY AS NEEDED WHEN COPY RELEASE TO A LOCAL RUN FOLDER -############################################################################################################################################################################# -Project.Directory = %project.folder%/ -generic.path = %project.folder%/input/ -scenario.path = %project.folder%/ -skims.path = %project.folder%/output/ -uec.path = %project.folder%/uec/ -report.path = %project.folder%/report/ - -# Visitor model is run using Java 7 Fork\Join Framework. Parallelism controls number of simultaneous threads. Can increase if more processors. -# 5 threads provided optimum runtimes on a 6 core, 24 thread machine with 128GB of RAM. -visitor.run.concurrent = true -visitor.concurrent.parallelism = 5 - -############################################################################################################################################################################# -# -# SCENARIO PROPERTIES: MODIFY WHEN RUNNING NEW SCENARIO, IF NECESSARY -# -############################################################################################################################################################################# -# MGRA data file: this token is referred to in many UECs -mgra.socec.file = input/mgra13_based_input${year}.csv - -# scenario year -scenarioYear=${year} - -# scenario build -scenarioBuild=${year_build} - -# Auto operating costs: these tokens are referred to in many UECs -aoc.fuel =${aoc.fuel} -aoc.maintenance =${aoc.maintenance} - -# Cross border model is run using Java 7 Fork\Join Framework. Parallelism controls number of simultaneous threads. Can increase if more processors. -crossBorder.run.concurrent = true -crossBorder.concurrent.parallelism = 8 - -# Cross border model settings: Number of tours, share of tours that are SENTRI. -crossBorder.tours =${crossBorder.tours} -crossBorder.sentriShare = ${crossBorder.sentriShare} - -# Visitor model settings: occupancy rates for hotels, households and share of each that are business visitors -visitor.hotel.occupancyRate = 0.7 -visitor.household.occupancyRate = 0.018 -visitor.hotel.businessPercent = 0.3 -visitor.household.businessPercent = 0.04 - -# Airport model settings: enplanements, connecting passengers, average party size, MGRA that the airport is in -airport.SAN.enplanements =${airport.SAN.enplanements} -airport.SAN.connecting =${airport.SAN.connecting} -airport.SAN.annualizationFactor = 365 -airport.SAN.averageSize = 1.7 -airport.SAN.airportMgra =${airport.SAN.airportMgra} - -airport.CBX.enplanements =${airport.CBX.enplanements} -airport.CBX.connecting =${airport.CBX.connecting} -airport.CBX.annualizationFactor = 365 -airport.CBX.averageSize = 2.2 -airport.CBX.airportMgra =${airport.CBX.airportMgra} - -# Truck model settings: -truck.FFyear =${year} - -# Destination zones for the transponder accessibility calculator -transponder.destinations = 4027,2563,2258 - -#taz crosswalk file -taz.to.cluster.crosswalk.file = input/taz_crosswalk.csv -cluster.zone.centroid.file = input/cluster_zones.csv - -############################################################################################ -# EMERGING MOBILITY SECTION: MODIFY WHEN CHANGE AV, TNC, and MICROMOBILITY ASSUMPTIONS -#------------------------------------------------------------------------------------------- -# AV Mobility Scenario Parameters -#------------------------------------------------------------------------------------------- -# AV.Share: the share of vehicles assumed to be AVs in the vehicle fleet; Auto ownership ASCs will be calibrated for different levels of AV penetration -# AV.ProbabilityBoost: the increased probability (multiplicative) for using AVs for tours, based on autos to drivers. The highest this should go is 1.2 -# AV.IVTFactor: the auto in-vehicle time factor to apply to AVs -# AV.ParkingCostFactor: The auto parking cost factor to apply to AVs, assuming some AVs are sent to remote locations or home -# AV.CostPerMileFactor: The auto cost per mile factor to apply to AVs, assuming AVs are more efficient in terms of fuel consumption than human-driven vehicles -# AV.TerminalTimeFactor: The factor to apply to terminal time for AVs, assuming AVs offer curbside passenger pickup/dropoff -# TNC.shared.IVTFactor: The factor to apply to in-vehicle time for shared TNC mode, reflecting out-direction travel for pickup/dropoff of other passengers - -Mobility.AV.Share = ${Mobility.AV.Share} -Mobility.AV.ProbabilityBoost.AutosLTDrivers = 1.2 -Mobility.AV.ProbabilityBoost.AutosGEDrivers = 1.1 -Mobility.AV.IVTFactor = 0.75 -Mobility.AV.ParkingCostFactor = 0.5 -Mobility.AV.CostPerMileFactor = 0.7 -Mobility.AV.TerminalTimeFactor = 0.65 -Mobility.AV.MinimumAgeDriveAlone = 13 -Mobility.TNC.shared.IVTFactor = 1.25 -crossBorder.avShare = 0.0 - -#------------------------------------------------------------------------------------------- -# Taxi and TNC cost and wait time parameters -#------------------------------------------------------------------------------------------- -# 3 modes: taxi, TNC - single, and TNC - shared -# baseFare: Initial fare -# costPerMile: The cost per mile -# costPerMinute: The cost per minute -# costMinimum: The minimum cost (for TNC modes only) -# -# Wait times are drawn from a distribution by area type (emp+hh)/sq. miles -# The mean and standard deviation is given for each area type range -# The ranges are configurable, set by WaitTimeDistribution.EndPopEmpPerSqMi - -taxi.baseFare = ${taxi.baseFare} -taxi.costPerMile = ${taxi.costPerMile} -taxi.costPerMinute = ${taxi.costPerMinute} - -TNC.single.baseFare = ${TNC.single.baseFare} -TNC.single.costPerMile = ${TNC.single.costPerMile} -TNC.single.costPerMinute = ${TNC.single.costPerMinute} -TNC.single.costMinimum = ${TNC.single.costMinimum} - -# use lower costs - these are synthesized, need real prices -TNC.shared.baseFare = ${TNC.shared.baseFare} -TNC.shared.costPerMile = ${TNC.shared.costPerMile} -TNC.shared.costPerMinute = ${TNC.shared.costPerMinute} -TNC.shared.costMinimum = ${TNC.shared.costMinimum} - -#Note: the following comma-separated value properties cannot have spaces between them, or else the RuntimeConfiguration.py code won't work -TNC.single.waitTime.mean = 10.3,8.5,8.4,6.3,4.7 -TNC.single.waitTime.sd = 4.1,4.1,4.1,4.1,4.1 - -TNC.shared.waitTime.mean = 15.0,15.0,11.0,8.0,7.0 -TNC.shared.waitTime.sd = 4.1,4.1,4.1,4.1,4.1 - -Taxi.waitTime.mean = 26.5,17.3,13.3,9.5,5.5 -Taxi.waitTime.sd = 6.4,6.4,6.4,6.4,6.4 - -WaitTimeDistribution.EndPopEmpPerSqMi = 500,2000,5000,15000,9999999999 - -#------------------------------------------------------------------------------------------- -# Taxi and TNC vehcicle trip conversion factors -#------------------------------------------------------------------------------------------- -# The following properties are used to split out the taxi, TNC-single, and TNC-shared trips into vehicle trips to be added to the rest of the vehicle trips by occupancy prior to assignment. - -Taxi.da.share = 0.0 -Taxi.s2.share = 0.9 -Taxi.s3.share = 0.1 -Taxi.passengersPerVehicle = 1.1 - -TNC.single.da.share = 0.0 -TNC.single.s2.share = 0.8 -TNC.single.s3.share = 0.2 -TNC.single.passengersPerVehicle = 1.2 - -TNC.shared.da.share = 0.0 -TNC.shared.s2.share = 0.3 -TNC.shared.s3.share = 0.7 -TNC.shared.passengersPerVehicle = 2.0 - -#------------------------------------------------------------------------------------------- -# Maas Routing Model Properties -#------------------------------------------------------------------------------------------- -Maas.RoutingModel.maxDistanceForPickup = 5 -Maas.RoutingModel.maxDiversionTimeForPickup = 5 -Maas.RoutingModel.minutesPerSimulationPeriod = 5 -Maas.RoutingModel.maxPassengers=6 -Maas.RoutingModel.maxWalkDistance = 0.15 -Maas.RoutingModel.vehicletrip.output.file=output/TNCTrips.csv -Maas.RoutingModel.vehicletrip.output.matrix=output/TNCVehicleTrips - -Maas.RoutingModel.routeIntrazonal=false -#NULL,DRIVEALONE,SHARED2,SHARED3,WALK,BIKE,WALK_SET,PNR_SET,KNR_SET,TNC_SET,TAXI,TNC_SINGLE,TNC_SHARED,SCHBUS -Maas.RoutingModel.Modes =0,0,0,0,0,0,0,0,0,0,1,1,1,0 -Maas.RoutingModel.SharedEligible=0,0,0,0,0,0,0,0,0,0,0,0,1,0 -Maas.RoutingModel.maxDistanceBeforeRefuel = 300 -Maas.RoutingModel.timeRequiredForRefuel = 15 - -Maas.AVAllocationModel.vehicletrip.output.file = output/householdAVTrips.csv -Maas.AVAllocationModel.vehicletrip.output.matrix = output/emptyAVTrips - -Maas.AVAllocation.uec.file = AutonomousVehicleAllocationChoice.xls -Maas.AVAllocation.data.page = 0 -Maas.AVAllocation.vehiclechoice.model.page = 1 -Maas.AVAllocation.parkingchoice.model.page = 2 -Maas.AVAllocation.triputility.model.page = 3 -Mobility.AV.RemoteParkingCostPerHour = ${Mobility.AV.RemoteParkingCostPerHour} - -# END--EMERGING MOBILITY SECTION -############################################################################################ - -############################################################################################################################################################################# -# -# CORE MODEL RUN PROPERTIES: CONTROL STEPS RUN IN CORE MODEL -# -############################################################################################################################################################################# -Model.Random.Seed = 1 - -RunModel.Clear.MatrixMgr.At.Start=false - -# Set to true if read the accessibilities from an input file instead of calculating them prior to running CTRAMP -acc.read.input.file = false - -# Setting shadow price files to null will reset prices to 0. If running new land-use scenario, set files to null and set maximum iterations to 20. -# Then copy shadow price output files to input directory, set maximum iterations to 1 for any subsequent runs with the same land-use file. -UsualWorkLocationChoice.ShadowPrice.Input.File = input/${workShadowPricing.iteration} -UsualSchoolLocationChoice.ShadowPrice.Input.File = input/${schoolShadowPricing.iteration} -uwsl.ShadowPricing.Work.MaximumIterations = 1 -uwsl.ShadowPricing.School.MaximumIterations = 1 -uwsl.ShadowPricing.OutputFile = output/ShadowPricingOutput.csv - -uwsl.run.workLocChoice = true -uwsl.run.schoolLocChoice = true -uwsl.write.results = true - -uwsl.use.new.soa = false -nmdc.use.new.soa = false -slc.use.new.soa = false - -# properties for distributed time coefficient -distributedTimeCoefficients = true - -timeDistribution.mean.work = 1.0 -timeDistribution.standardDeviation.work = 0.7 -timeDistribution.mean.nonWork = 1.0 -timeDistribution.standardDeviation.nonWork = 0.6 - -timeDistribution.randomSeed = 2301832 - -# value of time thresholds for skimming, assignment, mode choice UECs and trip tables ($/hr). -valueOfTime.threshold.low = 8.81 -valueOfTime.threshold.med = 18.00 - - -# save tour mode choice utilities and probabilities (for debugging purpose) -TourModeChoice.Save.UtilsAndProbs = true - -# packet size for distributing households, DO NOT change -distributed.task.packet.size = 200 - -#RunModel.RestartWithHhServer = uwsl -RunModel.RestartWithHhServer = none -#RunModel.RestartWithHhServer = ao -#RunModel.RestartWithHhServer = stf - -# Model Component run flags; Wu's note: not functional yet -RunModel.PreAutoOwnership = true -RunModel.UsualWorkAndSchoolLocationChoice = true -RunModel.AutoOwnership = true -RunModel.TransponderChoice = true -RunModel.FreeParking = true -RunModel.CoordinatedDailyActivityPattern = true -RunModel.IndividualMandatoryTourFrequency = true -RunModel.MandatoryTourModeChoice = true -RunModel.MandatoryTourDepartureTimeAndDuration = true -RunModel.SchoolEscortModel = true -RunModel.JointTourFrequency = true -RunModel.JointTourLocationChoice = true -RunModel.JointTourDepartureTimeAndDuration = true -RunModel.JointTourModeChoice = true -RunModel.IndividualNonMandatoryTourFrequency = true -RunModel.IndividualNonMandatoryTourLocationChoice = true -RunModel.IndividualNonMandatoryTourDepartureTimeAndDuration = true -RunModel.IndividualNonMandatoryTourModeChoice = true -RunModel.AtWorkSubTourFrequency = true -RunModel.AtWorkSubTourLocationChoice = true -RunModel.AtWorkSubTourDepartureTimeAndDuration = true -RunModel.AtWorkSubTourModeChoice = true -RunModel.StopFrequency =true -RunModel.StopLocation = true - -############################################################################################################################################################################# -# -# INPUT PROPERTIES -# -############################################################################################################################################################################# -#PopSyn Inputs -PopulationSynthesizer.InputToCTRAMP.HouseholdFile = input/households.csv -PopulationSynthesizer.InputToCTRAMP.PersonFile = input/persons.csv -PopulationSynthesizer.OccupCodes = input/pecas_occ_occsoc_acs.csv -PopulationSynthesizer.IndustryCodes = input/activity_code_indcen_acs.csv -# -# The military industry ranges are used to recode military occupation. This is -# necessary because military workers identify themselves as non-military occupations. -# The models need to be consistent with PECAS, where all military workers are in -# the military occupation category 56. -PopulationSynthesizer.MilitaryIndustryRange=9670,9870 - -# auxiliary inputs, these are scenario-specific -taz.driveaccess.taps.file = input/accessam.csv -tap.ptype.file = input/tap.ptype -taz.parkingtype.file = input/zone.park -taz.terminal.time.file = input/zone.term -maz.tap.tapLines = output/tapLines.csv - -# transit stop attribute file -transit.stop.file = input/trstop.csv - -############################################################################################################################################################################# -# -# OUTPUT PROPERTIES -# -############################################################################################################################################################################# -Results.WriteDataToFiles= true -Results.HouseholdDataFile = output/householdData.csv -Results.PersonDataFile = output/personData.csv -Results.IndivTourDataFile = output/indivTourData.csv -Results.JointTourDataFile = output/jointTourData.csv -Results.IndivTripDataFile = output/indivTripData.csv -Results.JointTripDataFile = output/jointTripData.csv -Results.WriteDataToDatabase = false -Results.HouseholdTable = household_data -Results.PersonTable = person_data -Results.IndivTourTable = indiv_tour_data -Results.JointTourTable = joint_tour_data -Results.IndivTripTable = indiv_trip_data -Results.JointTripTable = joint_trip_data -Results.AutoTripMatrix = output/autoTrips -Results.TranTripMatrix = output/tranTrips -Results.NMotTripMatrix = output/nmotTrips -Results.OthrTripMatrix = output/othrTrips -Results.PNRFile = output/PNRByTAP_Vehicles.csv -Results.CBDFile = output/CBDByMGRA_Vehicles.csv -Results.MatrixType = OMX -Results.segmentByTransponderOwnership = true - - -Results.AutoOwnership=output/aoResults.csv -read.pre.ao.results=false -read.pre.ao.filename=output/aoResults_pre.csv - -Results.UsualWorkAndSchoolLocationChoice=output/wsLocResults.csv -read.uwsl.results=false -read.uwsl.filename=output/wsLocResults_1.csv - -############################################################################################################################################################################# -# -# CORE MODEL UECS -# -############################################################################################################################################################################# -# UECs for calculating accessibilities -acc.uec.file = %project.folder%/uec/Accessibilities.xls -acc.data.page = 0 -acc.sov.offpeak.page = 1 -acc.sov.peak.page = 2 -acc.hov.offpeak.page = 3 -acc.hov.peak.page = 4 -acc.maas.offpeak.page = 5 -acc.maas.peak.page = 6 -acc.nonmotorized.page = 7 -acc.constants.page = 8 -acc.sizeTerm.page = 9 -acc.schoolSizeTerm.page = 10 -acc.workerSizeTerm.page = 11 -acc.dcUtility.uec.file = %project.folder%/uec/Accessibilities_DC.xls -acc.dcUtility.data.page = 0 -acc.dcUtility.page = 1 - -# accessibility file location -acc.output.file = input/accessibilities.csv - -#UECs for calculating destination choice based land use accessibilities -lu.acc.dcUtility.uec.file = %project.folder%/uec/Accessibilities_LU_DC.xls -lu.acc.dcUtility.data.page = 0 -lu.acc.dcUtility.page = 1 -lu.accessibility.alts.file = Acc_LU_alts.csv - -# land use accessibililty file locations -lu.acc.output.file = output/luAccessibilities.csv -lu.acc.mc.logsums.output.file = output/luLogsums.csv - -# set either or both averaging methods to be used to write LU accessibilities files -# also requires command line parameter "-luAcc true" and acc.read.input.file = false -lu.acc.simple.averaging.method = true -lu.acc.logit.averaging.method = true - -accessibility.alts.file = Acc_alts.csv - -#UEC for Mandatory accessibilities -acc.mandatory.uec.file = %project.folder%/uec/MandatoryAccess.xls -acc.mandatory.data.page = 0 -acc.mandatory.auto.page = 1 -acc.mandatory.autoLogsum.page = 2 -acc.mandatory.bestWalkTransit.page = 3 -acc.mandatory.bestDriveTransit.page = 4 -acc.mandatory.transitLogsum.page = 5 - -# UECs for auto ownership model -ao.uec.file = AutoOwnership.xls -ao.data.page = 0 -ao.model.page = 1 - -# UECs for Mandatory tour destination choice model -uwsl.dc.uec.file = ${uwsl.dc.uec.file} -uwsl.dc2.uec.file = TourDestinationChoice2.xls -uwsl.soa.uec.file = DestinationChoiceAlternativeSample.xls -uwsl.soa.alts.file = DestinationChoiceAlternatives.csv -uwsl.work.soa.SampleSize = 30 -uwsl.school.soa.SampleSize = 30 - -# The UEC file for work purposes includes TAZ Size in the expressions -work.soa.uec.file = TourDcSoaDistance.xls -work.soa.uec.data = 0 -work.soa.uec.model = 1 - -# The UEC file for school purposes does not include TAZ Size in the expressions -# so that the utilities can be stored as exponentiated distance utility matrices -# for univ, hs, gs, and ps, and then multiplied by the various school segment -# size terms for each of these 4 groups of school segments. -univ.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls -univ.soa.uec.data = 0 -univ.soa.uec.model = 1 - -hs.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls -hs.soa.uec.data = 0 -hs.soa.uec.model = 2 - -gs.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls -gs.soa.uec.data = 0 -gs.soa.uec.model = 3 - -ps.soa.uec.file = TourDcSoaDistanceNoSchoolSize.xls -ps.soa.uec.data = 0 -ps.soa.uec.model = 4 - -#UECs for transponder ownership model -tc.choice.avgtts.file = output/transponderModelAccessibilities.csv -tc.uec.file = TransponderOwnership.xls -tc.data.page = 0 -tc.model.page = 1 -tc.everyone.owns = ${tc.everyone.owns} - - -#UECs for parking provision model -fp.uec.file = ParkingProvision.xls -fp.data.page = 0 -fp.model.page = 1 - -#UEC for telecommute model -te.uec.file = Telecommute.xls -te.data.page = 0 -te.model.page = 1 - - -#UECs for CDAP model -cdap.uec.file = CoordinatedDailyActivityPattern.xls -cdap.data.page = 0 -cdap.one.person.page = 1 -cdap.two.person.page = 2 -cdap.three.person.page = 3 -cdap.all.person.page = 4 -cdap.joint.page = 5 - -#UECs for individual mandatory tour frequency model -imtf.uec.file = MandatoryTourFrequency.xls -imtf.data.page = 0 -imtf.model.page = 1 - -#UECs for Non-Mandatory tour destination sampling -nonSchool.soa.uec.file = TourDcSoaDistance.xls -escort.soa.uec.data = 0 -escort.soa.uec.model = 2 -other.nonman.soa.uec.data = 0 -other.nonman.soa.uec.model = 3 -atwork.soa.uec.data = 0 -atwork.soa.uec.model = 4 - -soa.taz.dist.alts.file = SoaTazDistAlts.csv - -nmdc.dist.alts.file = NonMandatoryTlcAlternatives.csv -nmdc.soa.alts.file = DestinationChoiceAlternatives.csv -nmdc.soa.SampleSize = 30 - -#UECs for Non-Mandatory tour destination choice model -nmdc.uec.file2 = TourDestinationChoice2.xls -nmdc.uec.file = ${nmdc.uec.file} -nmdc.data.page = 0 -nmdc.escort.model.page = 7 -nmdc.shop.model.page = 8 -nmdc.maint.model.page = 9 -nmdc.eat.model.page = 10 -nmdc.visit.model.page = 11 -nmdc.discr.model.page = 12 -nmdc.atwork.model.page = 13 - -# following properties use tod sampling instead of logsums -nmdc.SampleTODPeriod = true -nmdc.SampleTODPeriod.file = input/Non_Mand_Tours_ArrDep_Distbn.csv - -#UECs for Non-Mandatory tour destination sampling -nmdc.soa.uec.file = DestinationChoiceAlternativeSample.xls -nmdc.soa.data.page = 0 -nmdc.soa.escort.model.page = 6 -nmdc.soa.shop.model.page = 7 -nmdc.soa.maint.model.page = 7 -nmdc.soa.eat.model.page = 7 -nmdc.soa.visit.model.page = 7 -nmdc.soa.discr.model.page = 7 -nmdc.soa.atwork.model.page = 8 - -#UECs for School Escorting Model -school.escort.uec.filename = SchoolEscorting.xls -school.escort.alts.file = SchoolEscortingAlts.csv -school.escort.data.sheet = 0 -school.escort.outbound.model.sheet = 1 -school.escort.inbound.conditonal.model.sheet = 2 -school.escort.outbound.conditonal.model.sheet = 3 -school.escort.RNG.offset = 384571483 - -#UECs for tour mode choice model -tourModeChoice.uec.file =TourModeChoice.xls -tourModeChoice.maint.model.page = 4 -tourModeChoice.discr.model.page = 5 -tourModeChoice.atwork.model.page = 6 - -# utility coefficients by tour purpose (work, univ, school, maintenance, discretionary, work-based). These are at tour level. -tour.utility.ivt.coeffs = -0.016,-0.016,-0.01,-0.017,-0.015,-0.032 -tour.utility.income.coeffs = -0.625,-0.262,-0.262,-0.262,-0.262,-0.262 -tour.utility.income.exponents = 0.6,0.5,0.5,0.5,0.5,0.5 - -#UECs for tour TOD choice model -departTime.uec.file = TourDepartureAndDuration.xls -departTime.data.page = 0 -departTime.work.page = 1 -departTime.univ.page = 2 -departTime.school.page = 3 -departTime.escort.page = 4 -departTime.shop.page = 5 -departTime.maint.page = 6 -departTime.eat.page = 7 -departTime.visit.page = 8 -departTime.discr.page = 9 -departTime.atwork.page = 10 -departTime.alts.file = DepartureTimeAndDurationAlternatives.csv - -#UECs for joint tour frequency choice model -jtfcp.uec.file = JointTourFrequency.xls -jtfcp.alternatives.file = JointAlternatives.csv -jtfcp.data.page = 0 -jtfcp.freq.comp.page = 1 -jtfcp.participate.page = 2 - -#UECs for individual non-mandatory tour frequency model -inmtf.uec.file = NonMandatoryIndividualTourFrequency.xls -inmtf.FrequencyExtension.ProbabilityFile = IndividualNonMandatoryTourFrequencyExtensionProbabilities_p1.csv -IndividualNonMandatoryTourFrequency.AlternativesList.InputFile = IndividualNonMandatoryTourFrequencyAlternatives.csv -inmtf.data.page = 0 -inmtf.perstype1.page = 1 -inmtf.perstype2.page = 2 -inmtf.perstype3.page = 3 -inmtf.perstype4.page = 4 -inmtf.perstype5.page = 5 -inmtf.perstype6.page = 6 -inmtf.perstype7.page = 7 -inmtf.perstype8.page = 8 - -#UECs for at work subtour frequency model -awtf.uec.file = AtWorkSubtourFrequency.xls -awtf.data.page = 0 -awtf.model.page = 1 - -#UECs for stop frequency model -stf.uec.file = StopFrequency.xls -stf.purposeLookup.proportions = StopPurposeLookupProportions.csv -stf.data.page = 0 -stf.work.page = 1 -stf.univ.page = 2 -stf.school.page = 3 -stf.escort.page = 4 -stf.shop.page = 5 -stf.maint.page = 6 -stf.eat.page = 7 -stf.visit.page = 8 -stf.discr.page = 9 -stf.subtour.page = 10 - -#UECs for stop location choice model -slc.uec.file = StopLocationChoice.xls -slc.uec.data.page = 0 -slc.mandatory.uec.model.page = 1 -slc.maintenance.uec.model.page = 2 -slc.discretionary.uec.model.page = 3 -slc.alts.file = SlcAlternatives.csv - -slc.soa.uec.file = SlcSoaSize.xls -slc.soa.alts.file = DestinationChoiceAlternatives.csv - -auto.slc.soa.distance.uec.file = SlcSoaDistanceUtility.xls -auto.slc.soa.distance.data.page = 0 -auto.slc.soa.distance.model.page = 1 - -slc.soa.size.uec.file = SlcSoaSize.xls -slc.soa.size.uec.data.page = 0 -slc.soa.size.uec.model.page = 1 - -stop.depart.arrive.proportions = StopDepartArriveProportions.csv - -#UECs for trip mode choice model -tripModeChoice.uec.file =TripModeChoice.xls - -# utility coefficients by tour purpose (work, univ, school, maintenance, discretionary, work-based). These are at trip level. -trip.utility.ivt.coeffs = -0.032,-0.032,-0.02,-0.034,-0.03,-0.064 -trip.utility.income.coeffs = -1.25,-0.524,-0.524,-0.524,-0.524,-0.524 -trip.utility.income.exponents = 0.6,0.5,0.5,0.5,0.5,0.5 - - -#UECs for parking location choice model -plc.uec.file = ParkLocationChoice.xls -plc.uec.data.page = 0 -plc.uec.model.page = 1 - -plc.alts.corresp.file = ParkLocationAlts.csv -plc.alts.file = ParkLocationSampleAlts.csv - -mgra.avg.cost.output.file = output/mgraParkingCost.csv - -mgra.avg.cost.dist.coeff.work = -8.6 -mgra.avg.cost.dist.coeff.other = -4.9 - -park.cost.reimb.mean = -0.05 -park.cost.reimb.std.dev = 0.54 - -#UECs for best transit path finding -utility.bestTransitPath.uec.file =BestTransitPathUtility.xls -utility.bestTransitPath.data.page = 0 -utility.bestTransitPath.tapToTap.page = 1 -utility.bestTransitPath.walkAccess.page = 2 -utility.bestTransitPath.driveAccess.page = 3 -utility.bestTransitPath.walkEgress.page = 4 -utility.bestTransitPath.driveEgress.page = 5 -utility.bestTransitPath.driveAccDisutility.page = 6 -utility.bestTransitPath.driveEgrDisutility.page = 7 -utility.bestTransitPath.skim.sets = 3 -utility.bestTransitPath.alts = 4 -utility.bestTransitPath.maxPathsPerSkimSetForLogsum = 1,1,1 -utility.bestTransitPath.nesting.coeff = 0.24 - -#UECs for auto skimming -skims.auto.uec.file = AutoSkims.xls -skims.auto.data.page = 0 -skims.auto.ea.page = 1 -skims.auto.am.page = 2 -skims.auto.md.page = 3 -skims.auto.pm.page = 4 -skims.auto.ev.page = 5 - -#UECs for TAZ distances -taz.distance.uec.file = tazDistance.xls -taz.distance.data.page = 0 -taz.od.distance.ea.page = 1 -taz.od.distance.am.page = 2 -taz.od.distance.md.page = 3 -taz.od.distance.pm.page = 4 -taz.od.distance.ev.page = 5 - -#UECs for TAZ times -taz.od.time.ea.page = 6 -taz.od.time.am.page = 7 -taz.od.time.md.page = 8 -taz.od.time.pm.page = 9 -taz.od.time.ev.page = 10 - - -#UECs for walk-transit-walk skimming -skim.walk.transit.walk.uec.file = WalkTransitWalkSkims.xls -skim.walk.transit.walk.data.page = 0 -skim.walk.transit.walk.skim.page = 1 -skim.walk.transit.walk.skims = 13 - -#UECs for walk-transit-drive skimming -skim.walk.transit.drive.uec.file = WalkTransitDriveSkims.xls -skim.walk.transit.drive.data.page = 0 -skim.walk.transit.drive.skim.page = 1 -skim.walk.transit.drive.skims = 13 - -#UECs for drive-transit-walk skimming -skim.drive.transit.walk.uec.file = DriveTransitWalkSkims.xls -skim.drive.transit.walk.data.page = 0 -skim.drive.transit.walk.skim.page = 1 -skim.drive.transit.walk.skims = 13 - - -##################################################################################### -# IE Model Settings (run as part of CT-RAMP) -##################################################################################### - -RunModel.InternalExternal = true - -ie.uec.file = InternalExternalTripChoice.xls -ie.data.page = 0 -ie.model.page = 1 -ie.logsum.distance.coeff = -0.05 -external.tazs = 1,2,3,4,5,6,7,8,9,10,11,12 - - -internalExternal.dc.uec.file = InternalExternalDestinationChoice.xls -internalExternal.dc.uec.data.page = 0 -internalExternal.dc.uec.model.page = 1 -internalExternal.dc.uec.alts.file = InternalExternalDestinationChoiceAlternatives.csv - -internalExternal.tour.tod.file = input/internalExternal_tourTOD.csv - -internalExternal.trip.mc.uec.file =InternalExternalTripModeChoice.xls -internalExternal.trip.mc.data.page = 0 -internalExternal.trip.mc.model.page = 1 - -internalExternal.trip.output.file = output/internalExternalTrips.csv - -internalExternal.results.autoTripMatrix = output/autoInternalExternalTrips -internalExternal.results.nMotTripMatrix = output/nmotInternalExternalTrips -internalExternal.results.tranTripMatrix = output/tranInternalExternalTrips -internalExternal.results.othrTripMatrix = output/othrInternalExternalTrips - -##################################################################################### -# Cross-Border Model Settings -##################################################################################### -crossBorder.purpose.nonsentri.file = input/crossBorder_tourPurpose_nonSENTRI.csv -crossBorder.purpose.sentri.file = input/crossBorder_tourPurpose_SENTRI.csv - -crossBorder.tour.tod.file = input/crossBorder_tourEntryAndReturn.csv - -crossBorder.dc.soa.uec.file = CrossBorderDestinationChoiceSample.xls -crossBorder.dc.soa.data.page = 0 -crossBorder.dc.soa.model.page = 1 -crossBorder.dc.soa.size.page = 2 -crossborder.dc.soa.alts.file =${crossborder.dc.soa.alts.file} - -crossBorder.dc.uec.file =${crossBorder.dc.uec.file} -crossBorder.dc.data.page = 0 -crossBorder.dc.model.page = 1 -crossborder.dc.alts.file = CrossBorderDestinationChoiceAlternatives.csv - -crossBorder.dc.colonia.file = input/crossBorder_supercolonia.csv -crossBorder.dc.colonia.distance.parameter = -0.19 -crossBorder.dc.soa.sampleRate = 30 - -#crossBorder.tour.mc.uec.file = CrossBorderTourModeChoice.xls -crossBorder.tour.mc.uec.file =${crossBorder.tour.mc.uec.file} -crossBorder.tour.mc.data.page = 0 -crossBorder.tour.mc.mandatory.model.page = 1 -crossBorder.tour.mc.nonmandatory.model.page = 2 -crossBorder.poe.waittime.file = input/crossBorder_pointOfEntryWaitTime.csv - -crossBorder.trip.mc.uec.file =CrossBorderTripModeChoice.xls -crossBorder.trip.mc.data.page = 0 -crossBorder.trip.mc.model.page = 1 - -crossBorder.stop.frequency.file = input/crossBorder_stopFrequency.csv -crossBorder.stop.purpose.file = input/crossBorder_stopPurpose.csv - -crossBorder.slc.soa.uec.file = CrossBorderStopLocationChoiceSample.xls -crossBorder.slc.soa.data.page = 0 -crossBorder.slc.soa.model.page = 1 -crossBorder.slc.soa.alts.file = SoaTazDistAlts.csv - -crossBorder.slc.uec.file = CrossBorderStopLocationChoice.xls -crossBorder.slc.data.page = 0 -crossBorder.slc.model.page = 1 - -crossBorder.stop.outbound.duration.file = input/crossBorder_outboundStopDuration.csv -crossBorder.stop.inbound.duration.file = input/crossBorder_inboundStopDuration.csv - -crossBorder.tour.output.file = output/crossBorderTours.csv -crossBorder.trip.output.file = output/crossBorderTrips.csv - -crossBorder.results.autoTripMatrix = output/autoCrossBorderTrips -crossBorder.results.nMotTripMatrix = output/nmotCrossBorderTrips -crossBorder.results.tranTripMatrix = output/tranCrossBorderTrips -crossBorder.results.othrTripMatrix = output/othrCrossBorderTrips - -##################################################################################### -# Visitor Model Settings -##################################################################################### -visitor.business.tour.file = input/visitor_businessFrequency.csv -visitor.personal.tour.file = input/visitor_personalFrequency.csv - -visitor.partySize.file = input/visitor_partySize.csv -visitor.autoAvailable.file = input/visitor_autoAvailable.csv -visitor.income.file = input/visitor_income.csv - -visitor.dc.soa.uec.file = VisitorDestinationChoiceSample.xls -visitor.dc.soa.data.page = 0 -visitor.dc.soa.work.page = 1 -visitor.dc.soa.recreate.page = 2 -visitor.dc.soa.dining.page = 3 -visitor.dc.soa.size.page = 4 -visitor.dc.soa.alts.file = SoaTazDistAlts.csv - -visitor.dc.uec.file = VisitorDestinationChoice.xls -visitor.dc.data.page = 0 -visitor.dc.work.page = 1 -visitor.dc.recreate.page = 2 -visitor.dc.dining.page = 3 - -visitor.tour.tod.file = input/visitor_tourTOD.csv - -visitor.mc.uec.file =VisitorTourModeChoice.xls -visitor.mc.data.page = 0 -visitor.mc.model.page = 1 - -visitor.stop.frequency.file = input/visitor_stopFrequency.csv -visitor.stop.purpose.file = input/visitor_stopPurpose.csv -visitor.stop.outbound.duration.file = input/visitor_outboundStopDuration.csv -visitor.stop.inbound.duration.file = input/visitor_inboundStopDuration.csv - -visitor.slc.soa.uec.file = VisitorStopLocationChoiceSample.xls -visitor.slc.soa.data.page = 0 -visitor.slc.soa.model.page = 1 - -visitor.slc.uec.file = VisitorStopLocationChoice.xls -visitor.slc.data.page = 0 -visitor.slc.model.page = 1 - -visitor.trip.mc.uec.file =VisitorTripModeChoice.xls -visitor.trip.mc.data.page = 0 -visitor.trip.mc.model.page = 1 - -visitor.micromobility.uec.file = VisitorMicromobilityChoice.xls -visitor.micromobility.data.page = 0 -visitor.micromobility.model.page = 1 - - - - -visitor.tour.output.file = output/visitorTours.csv -visitor.trip.output.file = output/visitorTrips.csv - -visitor.results.autoTripMatrix = output/autoVisitorTrips -visitor.results.nMotTripMatrix = output/nmotVisitorTrips -visitor.results.tranTripMatrix = output/tranVisitorTrips -visitor.results.othrTripMatrix = output/othrVisitorTrips - - -# These settings are for building an estimation file, not used for main visitor model code -visitor.uec.file = VisitorSize.xls -visitor.uec.data.page = 0 -visitor.uec.sizeTerms.page = 1 - -##################################################################################### -# SAN Airport Model Settings -##################################################################################### -airport.SAN.purpose.file = input/airport_purpose.SAN.csv -airport.SAN.size.file = input/airport_party.SAN.csv -airport.SAN.duration.file = input/airport_nights.SAN.csv -airport.SAN.income.file = input/airport_income.SAN.csv -airport.SAN.departureTime.file = input/airport_departure.SAN.csv -airport.SAN.arrivalTime.file = input/airport_arrival.SAN.csv -airport.SAN.output.file = output/airport_out.SAN.csv - -airport.SAN.dc.uec.file = AirportDestinationChoice.SAN.xls -airport.SAN.dc.data.page = 0 -airport.SAN.dc.size.page = 5 -airport.SAN.dc.segment1.page = 1 -airport.SAN.dc.segment2.page = 2 -airport.SAN.dc.segment3.page = 3 -airport.SAN.dc.segment4.page = 4 - -airport.SAN.mc.uec.file =AirportModeChoice.SAN.xls -airport.SAN.mc.data.page = 0 -airport.SAN.mc.da.page = 1 -airport.SAN.mc.s2.page = 2 -airport.SAN.mc.s3.page = 3 -airport.SAN.mc.transit.page = 4 -airport.SAN.mc.accessMode.page = 5 - -airport.SAN.externalStationFile = uec/InternalExternalDestinationChoiceAlternatives.csv - -airport.SAN.results.autoTripMatrix = output/autoAirportTrips.SAN -airport.SAN.results.nMotTripMatrix = output/nmotAirportTrips.SAN -airport.SAN.results.tranTripMatrix = output/tranAirportTrips.SAN -airport.SAN.results.othrTripMatrix = output/othrAirportTrips.SAN - -##################################################################################### -# CBX Airport Model Settings -##################################################################################### -airport.CBX.purpose.file = input/airport_purpose.CBX.csv -airport.CBX.size.file = input/airport_party.CBX.csv -airport.CBX.duration.file = input/airport_nights.CBX.csv -airport.CBX.income.file = input/airport_income.CBX.csv -airport.CBX.departureTime.file = input/airport_departure.CBX.csv -airport.CBX.arrivalTime.file = input/airport_arrival.CBX.csv -airport.CBX.output.file = output/airport_out.CBX.csv - -airport.CBX.dc.uec.file = AirportDestinationChoice.CBX.xls -airport.CBX.dc.data.page = 0 -airport.CBX.dc.size.page = 5 -airport.CBX.dc.segment1.page = 1 -airport.CBX.dc.segment2.page = 2 -airport.CBX.dc.segment3.page = 3 -airport.CBX.dc.segment4.page = 4 - -airport.CBX.mc.uec.file =AirportModeChoice.CBX.xls -airport.CBX.mc.data.page = 0 -airport.CBX.mc.da.page = 1 -airport.CBX.mc.s2.page = 2 -airport.CBX.mc.s3.page = 3 -airport.CBX.mc.transit.page = 4 -airport.CBX.mc.accessMode.page = 5 - -airport.CBX.externalStationFile = uec/InternalExternalDestinationChoiceAlternatives.csv - -airport.CBX.results.autoTripMatrix = output/autoAirportTrips.CBX -airport.CBX.results.nMotTripMatrix = output/nmotAirportTrips.CBX -airport.CBX.results.tranTripMatrix = output/tranAirportTrips.CBX -airport.CBX.results.othrTripMatrix = output/othrAirportTrips.CBX - -##################################################################################### -# Truck Model Settings -##################################################################################### -truck.DFyear = ${model_years} -truck.luOverRide = "False" - -##################################################################################### -# Commercial Vehicle Model Settings -##################################################################################### -#scale factor to use in cvm trip generation. Also, used during demand import to factor-in demand accordingly -cvm.scale_factor = 1 -#scale factors by vehicle (light, medium, and heavy) and time of day (ea,am,md,pm,ev) - used to boost cvm demand -#light vehicles -cvm.scale_light = 1,2,3.5,2,1 -#medium vehicles -cvm.scale_medium = 1,1,1,1,1 -#heavy vehicles -cvm.scale_heavy = 1,1,1,1,1 -#cvm vehicle shares representing portions of the cvm vehicle trips that go to light-heavy trucks. -#share value should be between 0 and 1. 0 representing none will go to light-heavy truck and 1 means all will go. -cvm.share.light = 0.04 -cvm.share.medium = 0.64 -cvm.share.heavy = 0 - -################################################################# -# Report Section -################################################################# -Report.exportData=True -Report.iteration=3 -Report.tables = taztotap,indivtrips,jointtrips,airporttripsSAN,airporttripsCBX,cbtrips,visitortours,visitortrips,ietrip,commtrip -#aggregate trips eetrip, eitrip, and trucktrip are exported in Python, always -#Report.writeTransitIVT = True -##################################################################################### -# Trip Table Settings -##################################################################################### -# occupancies needed for trip table creation -occ3plus.purpose.Work = 3.34 -occ3plus.purpose.University = 3.34 -occ3plus.purpose.School = 3.34 -occ3plus.purpose.Escort = 3.34 -occ3plus.purpose.Shop = 3.34 -occ3plus.purpose.Maintenance = 3.34 -occ3plus.purpose.EatingOut = 3.34 -occ3plus.purpose.Visiting = 3.34 -occ3plus.purpose.Discretionary = 3.34 -occ3plus.purpose.WorkBased = 3.34 - -################################################################# -# Active Transportation Model Settings -# updated 4/2/2014 wsu -################################################################# -active.node.file = %project.folder%/input/SANDAG_Bike_NODE.dbf -active.node.id = NodeLev_ID -active.node.fieldnames = mgra,taz,x,y,tap,signalized -active.node.columns = MGRA,TAZ,XCOORD,YCOORD,TAP,Signal -active.edge.file = %project.folder%/input/SANDAG_Bike_NET.dbf -active.edge.anode = A -active.edge.bnode = B -active.edge.directional = false -active.edge.fieldnames = functionalClass,distance,gain,bikeClass,lanes,cycleTrack,bikeBlvd,roadsegid -active.edge.columns.ab = Func_Class,Distance,AB_Gain,ABBikeClas,AB_Lanes,Bike2Sep,Bike3Blvd,ROADSEGID -active.edge.columns.ba = Func_Class,Distance,BA_Gain,BABikeClas,BA_Lanes,Bike2Sep,Bike3Blvd,ROADSEGID -active.edge.centroid.field = functionalClass -active.edge.centroid.value = 10 -active.edge.autospermitted.field = functionalClass -active.edge.autospermitted.values = 1, 2, 3, 4, 5, 6, 7 -# distance bins for control of path sampling -active.sample.distance.breaks = 99 -# minimum path sizes of alternative lists for each distance bin -active.sample.pathsizes = 2 -# minimum count of samples for each distance bin -active.sample.count.min = 10 -# maximum count of samples for each distance bin -active.sample.count.max = 100 -# scale of random cost for each sampling iteration where random cost = cost + scale * unif(0,1) * distance -active.sample.random.scale.coef = 0.5 -active.sample.random.scale.link = 0.7 -active.sample.random.seeded = true -active.sample.maxcost = 998 -active.maxdist.walk.mgra = 3.0 -active.maxdist.walk.tap = 1.0 -active.maxdist.bike.taz = ${active.maxdist.bike.taz} -active.maxdist.bike.mgra = ${active.maxdist.bike.mgra} -active.maxdist.micromobility.mgra = 3.0 -active.maxdist.micromobility.tap = 1.0 -active.maxdist.microtransit.mgra = 3.0 -active.maxdist.microtransit.tap = 3.0 -active.output.bike = %project.folder%/output/ -active.output.walk = %project.folder%/output/ -active.coef.distcla0 = ${active.coef.distcla0} -active.coef.distcla1 = ${active.coef.distcla1} -active.coef.distcla2 = ${active.coef.distcla2} -active.coef.distcla3 = ${active.coef.distcla3} -active.coef.dartne2 = ${active.coef.dartne2} -active.coef.dwrongwy = ${active.coef.dwrongwy} -active.coef.dcyctrac = ${active.coef.dcyctrac} -active.coef.dbikblvd = ${active.coef.dbikblvd} -active.coef.nonscenic = 0.300 -active.coef.gain = 0.015 -active.coef.turn = 0.083 -active.coef.signals = 0.040 -active.coef.unlfrma = 0.360 -active.coef.unlfrmi = 0.150 -active.coef.untoma = 0.480 -active.coef.untomi = 0.100 -active.coef.gain.walk = 0.034 - -active.walk.minutes.per.mile = 20 -active.bike.minutes.per.mile = ${active.bike.minutes.per.mile} -active.ebike.ownership = ${active.ebike.ownership} -active.ebike.max.benefit = 10 -active.micromobility.speed = 15 -active.micromobility.variableCost = ${active.micromobility.variableCost} -active.micromobility.fixedCost = ${active.micromobility.fixedCost} -active.micromobility.rentalTime = 1 -active.micromobility.constant = 60 -# 2020 VOT $15 converted to 2010 $ at $12.17 -active.micromobility.vot = 12.17 - -micromobility.uec.file = MicromobilityChoice.xls -micromobility.data.page = 0 -micromobility.model.page = 1 - -active.microtransit.speed = 17 -active.microtransit.variableCost = 0.0 -active.microtransit.fixedCost = ${active.microtransit.fixedCost} -active.microtransit.waitTime = 4.0 -active.microtransit.accessTime = 0.0 -active.microtransit.constant = 120 -active.microtransit.notAvailable = 999 - -active.microtransit.tap.file = input/mobilityHubTaps.csv -active.microtransit.mgra.file = input/mobilityHubMGRAs.csv - -#active.trace.origins.taz = 500, 1000, 1500, 2000, 2500, 3000, 3500, 4000, 4500 -#active.trace.origins.mgra = 1000, 2000, 3000, 4000, 5000, 6000, 7000, 8000, 9000, 10000, 11000, 12000, 13000, 14000, 15000, 16000, 17000, 18000, 19000, 20000 -#active.trace.origins.tap = 1, 3, 5, 7, 8, 9, 15 -#active.trace.exclusive = false -#active.debug.origin = 200003500 -#active.debug.destination = 200003601 - -active.trace.outputassignmentpaths = false - -path.choice.uec.spreadsheet = %project.folder%/uec/BikeTripPathChoice.xls -path.choice.uec.model.sheet = 1 -path.choice.uec.data.sheet = 0 -path.choice.max.path.count = 200 -btpc.alts.file = bike_path_alts.csv -active.logsum.matrix.file.bike.taz = bikeTazLogsum.csv -active.logsum.matrix.file.bike.mgra = bikeMgraLogsum.csv -active.logsum.matrix.file.walk.mgra = walkMgraEquivMinutes.csv -active.logsum.matrix.file.walk.mgratap = walkMgraTapEquivMinutes.csv - -active.bike.write.derived.network = true -active.bike.derived.network.edges = derivedBikeEdges.csv -active.bike.derived.network.nodes = derivedBikeNodes.csv -active.bike.derived.network.traversals = derivedBikeTraversals.csv - -active.assignment.file.bike = bikeAssignmentResults.csv -active.micromobility.file.walk.mgra = microMgraEquivMinutes.csv -active.micromobility.file.walk.mgratap = microMgraTapEquivMinutes.csv - -AtTransitConsistency.xThreshold=1.0 -AtTransitConsistency.yThreshold=1.0 - -##################################################################################### -# SUMMIT Settings -##################################################################################### -summit.output.directory = output/ -# Purposes (which correspond to SUMMIT files) are as follows: -summit.purpose.Work = 1 -summit.purpose.University = 2 -summit.purpose.School = 3 -summit.purpose.Escort = 4 -summit.purpose.Shop = 4 -summit.purpose.Maintenance = 4 -summit.purpose.EatingOut = 5 -summit.purpose.Visiting = 5 -summit.purpose.Discretionary = 5 -summit.purpose.WorkBased = 6 - -summit.filename.1 = Work -summit.filename.2 = University -summit.filename.3 = School -summit.filename.4 = Maintenance -summit.filename.5 = Discretionary -summit.filename.6 = Workbased - -summit.ivt.file.1 = -0.016 -summit.ivt.file.2 = -0.016 -summit.ivt.file.3 = -0.010 -summit.ivt.file.4 = -0.017 -summit.ivt.file.5 = -0.015 -summit.ivt.file.6 = -0.032 - -summit.modes = 26 -# 1=wt,2=dt 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 -summit.mode.array = 0,0,0,0,0,0,0,0,0, 0, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0 - -summit.upperEA = 3 -summit.upperAM = 9 -summit.upperMD = 22 -summit.upperPM = 29 - -################################################################# -# TAPS Creation Settings -# updated 4/28/2015 ymm -################################################################# -taps.formal.premium.maxDist = 10.0 -taps.formal.express.maxDist = 4.0 -taps.formal.local.maxDist = 4.0 -taps.informal.premium.maxDist = 4.0 -taps.informal.express.maxDist = 2.0 -taps.informal.local.maxDist = 2.0 - -taps.premium.modes = 4,5,6,7 -taps.express.modes = 8,9 -taps.local.modes = 10 - -taps.skim = traffic_skims_AM.omx -taps.skim.dist = AM_SOV_NT_M_DIST -taps.skim.time = AM_SOV_NT_M_TIME - -################################################################# -# Military Adjustment Section -# updated 4/26/2016 Wu Sun -################################################################# -RunModel.militaryCtmAdjustment=true - -##################################################################################### -# Special Event Model Settings -# Wu Sun 5/15/2017 -##################################################################################### -specialEvent.event.file = input/specialEvent_eventData.csv - -specialEvent.partySize.file = input/specialEvent_partySize.csv -specialEvent.income.file = input/specialEvent_income.csv - -specialEvent.dc.uec.file = SpecialEventOriginChoice.xls -specialEvent.dc.data.page = 0 -specialEvent.dc.model.page = 1 -specialEvent.dc.size.page = 2 - -specialEvent.saveUtilsAndProbs= false - -specialEvent.trip.mc.uec.file =SpecialEventTripModeChoice.xls -specialEvent.trip.mc.data.page = 0 -specialEvent.trip.mc.model.page = 1 - -specialEvent.tour.output.file = output/specialEventTours.csv -specialEvent.trip.output.file = output/specialEventTrips.csv - -specialEvent.results.autoTripMatrix = output/autoSpecialEventTrips -specialEvent.results.nMotTripMatrix = output/nmotSpecialEventTrips -specialEvent.results.tranTripMatrix = output/tranSpecialEventTrips -specialEvent.results.othrTripMatrix = output/othrSpecialEventTrips - -##################################################################################### -# Transit Shed Properties wsu 8/7/18 -##################################################################################### -RunModel.skipTransitShed= true -#transit access threshold (in minutes, must be integer) -transitShed.threshold=30 -#TOD to use in Transit Shed analysis-EA, AM, MD, PM, and EV -transitShed.TOD=AM -#Transit Shed time components (walk to transit). Options: walkAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime -transitShed.walkTransitTimeComponents=walkAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime -#Transit Shed time components (drive to transit). Options: driveAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime -transitShed.driveTransitTimeComponents=drvAccTime,walkEgrTime,walkAuxTime,1stWaitTime,xferWaitTime,IVTime - -##################################################################################### -# Smart Signal Properties wsu 8/22/18 -##################################################################################### -smartSignal.factor.LC=${smartSignal.factor.LC} -smartSignal.factor.MA=${smartSignal.factor.MA} -smartSignal.factor.PA=${smartSignal.factor.PA} - -##################################################################################### -##################################################################################### -# Transit Tier 1 EMME Link Name zou 5/7/20 -##################################################################################### -transit.newMode = TIER 1 RAIL -transit.newMode.route = 581,582,583 - -##################################################################################### -# ATDM Properties wsu 8/22/18 -##################################################################################### -atdm.factor = ${atdm.factor} -##################################################################################### -# Transit PCE VEH Conversion cliu 8/19/20 -##################################################################################### -transit.bus.pceveh = 3.0 -##################################################################################### -# Local Drive Run Settings wsu 1/19/19 -##################################################################################### -RunModel.FileMask.Download = output,report,sql,logFiles -RunModel.FileMask.Upload = application,bin,input_truck,uec,output\iter*,output\*_1.csv,output\*_2.csv -##################################################################################### -# Visualizer Settings (run once after feedback loops) -##################################################################################### -visualizer.reference.path = ${visualizer.reference.path} -visualizer.output = SANDAG_Dashboard -visualizer.reference.label = REFERENCE -visualizer.build.label = SDABM - -##################################################################################### -# add year specific vehicle class toll factor wsu 6/18/20 -##################################################################################### -vehicle.class.toll.factor=vehicle_class_toll_factors.csv -vehicle.class.toll.factor.path = input/vehicle_class_toll_factors.csv - -##################################################################################### -# Stochastic traffic assignment settings ag 10/07/20 -##################################################################################### -stochasticHighwayAssignment.distributionType = GUMBEL -stochasticHighwayAssignment.replications = 10 -stochasticHighwayAssignment.aParameter = 1.0 -stochasticHighwayAssignment.bParameter = 0.05 -stochasticHighwayAssignment.seed = 1 - -############################################################################### -# DTA post-processing properties -############################################################################### - -dta.postprocessing.RandomSeed = 1004831 - -dta.postprocessing.outputs.path = %project.folder%/output/ - -dta.postprocessing.disaggregateTOD.path = %project.folder%//input/ -dta.postprocessing.disaggregateZone.path = %project.folder%//input/ -dta.postprocessing.disaggregateNode.path = %project.folder%//input/ - -dta.postprocessing.DetailedTODFile = DetailedTODFactors.csv -dta.postprocessing.NodeFile = NodeFactors.csv -dta.postprocessing.ZoneFile = MGRAFactors.csv -dta.postprocessing.BroadTODFile = BroadTODFactors.csv - -skims.path = %project.folder%/output/ -skims.extension = .omx -da.no.toll.skims.prefix = traffic_skims_ -skims.mat.name.suffix = SOVGPM_TIME - -dta.postprocessing.outputs.TripFile = dtaTripsOut.csv \ No newline at end of file diff --git a/sandag_abm/src/main/resources/serverswap.bat b/sandag_abm/src/main/resources/serverswap.bat deleted file mode 100644 index 303c7da..0000000 --- a/sandag_abm/src/main/resources/serverswap.bat +++ /dev/null @@ -1,9 +0,0 @@ -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 -set PROJECT_DIRECTORY_FWD=%3 - -%PROJECT_DRIVE% -set "SCEN_DIR=%PROJECT_DRIVE%%PROJECT_DIRECTORY%" -set "SCEN_DIR_FWD=%PROJECT_DRIVE%%PROJECT_DIRECTORY_FWD%" - -python.exe %SCEN_DIR%\python\serverswap.py -p %SCEN_DIR_FWD% \ No newline at end of file diff --git a/sandag_abm/src/main/resources/serverswap_files.csv b/sandag_abm/src/main/resources/serverswap_files.csv deleted file mode 100644 index 4f155bd..0000000 --- a/sandag_abm/src/main/resources/serverswap_files.csv +++ /dev/null @@ -1,25 +0,0 @@ -fileName,property,separator,refValue -/bin/CTRampEnv.bat,MAIN,=,MAIN -/bin/CTRampEnv.bat,NODE1,=,NODE1 -/bin/CTRampEnv.bat,NODE2,=,NODE2 -/bin/CTRampEnv.bat,NODE3,=,NODE3 -/bin/CTRampEnv.bat,MAIN_IP,=,ModelIP -/bin/CTRampEnv.bat,HHMGR_IP,=,ModelIP -/bin/CTRampEnv.bat,SNODE,=,SNODE -/bin/CTRampEnv.bat,TRANSCAD_PATH,=,TRANSCAD_PATH -/bin/stopABM.cmd,pskill,\\,MAIN -/bin/stopABM.cmd,pskill,\\,NODE1 -/bin/stopABM.cmd,pskill,\\,NODE2 -/conf/jppf-client.properties,driver1.jppf.server.host, = ,MAIN -/conf/jppf-client.properties,jppf.processing.threads, = ,THREADN1 -/conf/jppf-clientDistributed.properties,driver1.jppf.server.host, = ,MAIN -/conf/jppf-sandag01.properties,jppf.server.host, = ,MAIN -/conf/jppf-sandag01.properties,jppf.processing.threads, = ,THREADM -/conf/jppf-sandag02.properties,jppf.server.host, = ,MAIN -/conf/jppf-sandag02.properties,jppf.processing.threads, = ,THREADN1 -/conf/jppf-sandag03.properties,jppf.server.host, = ,MAIN -/conf/jppf-sandag03.properties,jppf.processing.threads, = ,THREADN2 -/conf/jppf-sandag04.properties,jppf.server.host, = ,MAIN -/conf/jppf-sandag04.properties,jppf.processing.threads, = ,THREADN3 -/conf/sandag_abm.properties,RunModel.MatrixServerAddress,=,ModelIP -/conf/sandag_abm.properties,RunModel.HouseholdServerAddress,=,ModelIP diff --git a/sandag_abm/src/main/resources/setup.bat b/sandag_abm/src/main/resources/setup.bat deleted file mode 100644 index 88bf3fe..0000000 --- a/sandag_abm/src/main/resources/setup.bat +++ /dev/null @@ -1 +0,0 @@ -python ./src/main/python/pythonGUI/setup.py py2exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/stopABM.cmd b/sandag_abm/src/main/resources/stopABM.cmd deleted file mode 100644 index 509cfd6..0000000 --- a/sandag_abm/src/main/resources/stopABM.cmd +++ /dev/null @@ -1,5 +0,0 @@ -rem stopping all java processes on cluster - -%CD%\pskill \\${master.node.name} java.exe -%CD%\pskill \\${node.1.name} java.exe -%CD%\pskill \\${node.2.name} java.exe diff --git a/sandag_abm/src/main/resources/taskkill.bat b/sandag_abm/src/main/resources/taskkill.bat deleted file mode 100644 index 29cd2a6..0000000 --- a/sandag_abm/src/main/resources/taskkill.bat +++ /dev/null @@ -1,2 +0,0 @@ -rem kill java tasks -taskkill /F /IM java.exe \ No newline at end of file diff --git a/sandag_abm/src/main/resources/updateYearSpecificProps.bat b/sandag_abm/src/main/resources/updateYearSpecificProps.bat deleted file mode 100644 index fe51ecd..0000000 --- a/sandag_abm/src/main/resources/updateYearSpecificProps.bat +++ /dev/null @@ -1,6 +0,0 @@ -set PROJECT_DRIVE=%1 -set PROJECT_DIRECTORY=%2 - -%PROJECT_DRIVE% -cd %PROJECT_DRIVE%%PROJECT_DIRECTORY%\python -python.exe parameterUpdate.py diff --git a/sandag_abm/src/main/resources/w9xpopen.exe b/sandag_abm/src/main/resources/w9xpopen.exe deleted file mode 100644 index f6033fdeaaaf14dc80579adeb1a8827f08d1de4f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 49664 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 Tr^i`Tghv5p&5Ap|&Gr8P1XY4u
  • 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 x;a)38D;#^NsXjIQ&a2< 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%0A_aAnYzk!umR7wy^?%RNcuq-&B?F06%1I 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 =C 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 z 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%0Agfw(T_P^Pq5;SLLRP-%+K-7X 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%0Ap|)s-@(E zK!L%ydbqJ3NqX!UF`mh3&r&8J9etnDRcyZ{?8WqcbLD=lhSPph${$Ow)fUin^Ua0( zo#i*ZefR9_+wYz$=~WI$&1$*<%KKZl?wLpjLIX&kULz3$1p) zvCwS}FxnyV1{|1~?3@4x!8TlK*B81SG*#UUL3OO{o2#qc4hAKWd9^F>SVox^vogux z@$h2`%4XNf{B7E(KrSjW- zTy75rFl?tSF%d2ytN>Rkkm*7XMEk=1RTQyyLiZ{p_pqw(SURU3;ihOd7zWk)$xI3E zKrd+pDNV{N!J$8!wabiMN>>rCR7azaa$sT9$1<7o?3mhuAFI99iF{b-wwwKR_#ll} zm)2uTH>l7wx)@$*8zy`k51+JU3BH2Cpr8Cud|<3~9gd$EU%|*Uy^4XBzA{>_=FPsH zd)Lt`Jv|?o$XB6<4E02~1f{_(w-!uyb~7<%rDg0Xah_}}a8s%N#`^1acz;RY z3s4De?O%6Q^wUBom3|oEDw+l5-nGs+@$@`5@^w2a5YS>BleDbxnC>F33}f{u;>l_{ z22X&obcnxdLXz}}gd}h@Yvz?zK9WwBli_hfkbkt|H7T3Z$mcxWGm;7TLHcs<1Z735 zER?hKL4TnrjIwm*kH{ll=F8$KpU~fFQbVzeJ*-qtVkkmi1(?ZP(q(Yi)&5=fx%IXN~U0Fn9jFKv8 zr<`vIe7=5mzFE0i(H!S(75EI!ah2#ZMidc^g)>B@^g<7}bk+u%I2ttXG<8)x6m7PTT)+)+4}K*Zv^{E3+uhkVuu|h^1r|*VQAK*MLN*Wz|Zyhq;Tu= z$?&aPFT+;>H%qx>{jxZkI8M{_;wWM|v%_{`RvuJNr?`v@4i|x9j;z^$Z_ez}(1%o= z%P>`>r_G+^hq^a0e@HLohv5q*IhqRTmx>xylVbf~&g?8z@DVlXF5wL%WK|rGH^8rh zmvW$@?%HoF=!ozEain*t+2q?M%t$HQEno@bBgacZnWGZxO#p&Z}W9Y;r=?ir=kFb5u(RrAxX z`d#fIyegM^t=VZa=EGt9rde|!sDwW#9uj-RG^%qsea!h&vWv(0b1t7h=O|!I885Q{ z^(&@lHAvDYa&kOehK~GZc_bv_o1k-7!e?T(F8rv3D_6I=e1WT^bKNz8m8)+1o7+W zL8WwOwGa^VhY2d1$STsa{A1)kvYB*_Q-G9r)7^K^&fd+Ra)*zj;vg*JeM3+&G9qB3 zvvi4IayqFvU#0U;HR%)PA+M>_Z!A9|rGX>hg;6t^o{FrK=EVpOipJ8DvN%x0c^W^` zK1y4jzsRT9N(l<#!$f|?Z0y%QFDC=9JfJ4PW&N=Lm#km2Ze1)lfy>Yl&h+v1^xJW8 zIlfDsAG~u_i~^TtoF4#IoF5LZGe3m%3MnV@rYNVqyuv)Ac#_83D86zdqvxP1#d0%& zhxBUuUEaLJlfWg^xxH+o@Rd{woz?g=Y6Xt{JKEXGI$*(0DB~f*XG%*EUsA72!kgsV zDE{$$Na~m1E%D)y=zr?t0d2#U_#(Y?laa`eeLE+?=hpEbl76#M_B@OfEekmEbO+^D3v zD*cW1yo_;ps>*?%V7s~`Vu|zwxQ?s-oPc2nau>}bYQ#6`&9*gFR^HclXgvvTGfKW+Or53s%CU1=_b&W zIM%DMKPCK3;APlG=&+w(377H=!y?o>!N+<;_aCDV7O*ZCu_G>+osSjpp~~Jq z?3aMU{0*EbVW|L#FdptT6MG6t{+x7ckS-w-}<5j(rXqH|Ue z-$F~-T5wY?)dO%wVu_C&hk=huUxX=^$JPa1;UJ@>i^m3{GuLEYD0fGzCdqWJ>k!JO2cVKQ*h_|82Gf8Mjj{S%<~v_LoO{~G0UA+i)S8C ziF{lZ;JWR_g2$S*O98ImIlh!jFQ=j!-pU_Ow%4~%j?hz?%gPy0eQHoEXo&jY&OjoJ zoTh{?qU$y2V*#$jK*||y0K*?aa`=GS;6%uDfGvz0 z)Dd!*>TQSIh$!Ru%`}!T zF2A6pkA;i%b=FklGwGZN@oKoDbe=?CDy1V_bGg&yX=!+R4~A3agE0Gc9#MT^`#{7d zK!aA6wN!q*Ch*Deb0AlsFTvq_$$UeIDd|TtFF8JeD@li>UaMXD7f?L#m_Vs$=Vvp* zbygNiV3EEEM|@EH8dMM2JM$T7eEvJX9PN(Sk+@dmrd9H$)3yR_?7m>v$ zs1Z;5Ehch$ftPjXiu5G6^cysicovp3D}S?e&kVfr{6Iy)Rq39W3_*IXZ>fO9&|2&q z-+-QJ0?Xhd<_vF;x8tCN4)(LxsxJ<@qyEw8nr3JC)QzAq4_E0H6%w7(^Zv&A6yXy0 z1I?R`1>*zDFXnTXcDP)IlPP|_MG!GBe?Z@){7X=DCL*x>;%+>|VWa^rw}8uh`xG@p zIZgxK=JjNGMfgGnA8nzXOYnK;X93yr%=2@wYshZ8XzGp(L0Jn20m5LQ|`+wIrRHdx#&*tTC3>C?EiAOI8!Ncr)ai>H|9JTfg=nS#a@yjF0kjo?1AZ- zx6K@J?3MfxTpXz=rGX-dblB@D+i!<(st-mBOPXFqpXKLtwa`1hf*HJ3HwAKq=Z);N zlSDB;`*F%Y%U|8WCcq17I+UbiWjy?))eoB~?cU-d+B(Xx0V9LIg!Y3JPSbHw9Trb@ z7h6B7uF}?T+Hi&w8KI6a2<}j&0$MaLG4fnN*iFh8dIRh?{SHpxmvRR=Qz*C0YvWrH zF3_6t*yI<-fy#T#Ws4p7L+Ur3&b+@=!l9pqiOkDa0c!XrIKfxL7iX_n{|v9uSyBEX z_2%W#umc@LmIBRIE4#eG$}Pb$BCYs1xaH<*3YOf%c|=aG`$$E8@>j|kVBLal*l(VM z`7_HM#&ZQ+TDwX(9&joltSHyvGK5yY)~U!Z`5R}56KzKP%I>+R>~eF@J$FxI7khSk z8g6IP)B3$hL-og^@1#3ga zhTHFnj$G&%iyoK1UY$`ZJ}zwn9o#NzwV`gy7EWAuay)1C&H#hPs+Tj=YE^OTluHsD zFgms)7c2Q7^%6j`nV2@Dyez+&8l^4Mvac|+qY0BW`5oKPA!p`19*LWao9ihRj93mKAe z7UM`tMSdR#0Wg+U2@hp1j*6;+!JquS z9)6I;yDL^2i+!wF;+;!l@<~1N3|gOJ+%ynO;2UqI?=-u4-74VGuOlF7;!iQIBA@e* z^KA;A4x=ji>ctE!uPTo56UueXW6;3GpSYkl=y+rD)k_%wSQR%WztXS6Zr5I~Ke)@E;I~YyKkYT* z@1)Bv+VI!&zL0v9^Qjlcpxdi-b&+x_%bVbhv2W*LOnz9Id5MFUd9oQPvM`n&T4$s_$-@T39$M&k3 zvRF<<#)G@>&hxO~G+%jFI8PUX!o35BaTb~`C3rJJd6d-+crBfeN(jKES4!#2@!+_` zIL=g6u!4^4#4ys?pOobl^^nF2bFw?Av4drn?g|Lk|yI_rdq10xkf- zKm+kod3`4MYJLF`f5;b*3w5lTCzhWY{=lcP<%I^F&=Dq;7xE_^0_9Ov%jip3Gq!3dXT5w$Db`-)aso-?sx+pY zkk1PlQxAblI|Sf#dm@e#4NLbbP{H9F?){#W >HFW{#*(Nz`UB?$e}@5Lo-fEPxx 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 leIu}M1on- -# with optional space-separated arguments -#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl-server.properties - -# enable secure communications with other servers; defaults to false (disabled)# -#jppf.peer.ssl.enabled = true - -#------------------------------------------------------------------------------# -# Enabling and configuring JMX features # -#------------------------------------------------------------------------------# - -# non-secure JMX connections; default is true (enabled) -#jppf.management.enabled = true - -# secure JMX connections via SSL/TLS; default is false (disabled) -#jppf.management.ssl.enabled = true - -# JMX management host IP address. If not specified (recommended), the first non-local -# IP address (i.e. neither 127.0.0.1 nor localhost) on this machine will be used. -# If no non-local IP is found, localhost will be used -#jppf.management.host = localhost - -# JMX management port. Defaults to 11198. If the port is already bound, the driver -# will scan for the first available port instead. -#jppf.management.port = 11198 - -#------------------------------------------------------------------------------# -# Configuration of the driver discovery broadcast service # -#------------------------------------------------------------------------------# - -# Enable/Disable automatic discovery of this JPPF drivers; default to true -#jppf.discovery.enabled = false - -# UDP multicast group to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default value is 230.0.0.1 -#jppf.discovery.group = 230.0.0.1 - -# UDP multicast port to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default value is 11111 -#jppf.discovery.port = 11111 - -# Time between 2 broadcasts, in milliseconds. Default value is 1000 -#jppf.discovery.broadcast.interval = 1000 - -# IPv4 inclusion patterns: broadcast these ipv4 addresses -#jppf.discovery.broadcast.include.ipv4 = 192.168.1.; 192.168.1.0/24 - -# IPv4 exclusion patterns: do not broadcast these ipv4 addresses -#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 - -# IPv6 inclusion patterns: broadcast these ipv6 addresses -#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 - -# IPv6 exclusion patterns: do not broadcast these ipv6 addresses -#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/64 - -#------------------------------------------------------------------------------# -# Connection with other servers, enabling P2P communication # -#------------------------------------------------------------------------------# - -# Enable/disable auto-discovery of remote peer drivers. Default value is false -#jppf.peer.discovery.enabled = true - -# manual configuration of peer servers, as a space-separated list of peers names to connect to -#jppf.peers = server_1 server_2 - -# enable both automatic and manual discovery -#jppf.peers = jppf_discovery server_1 server_2 - -# connection to server_1 -#jppf.peer.server_1.server.host = host_1 -#jppf.peer.server_1.server.port = 11111 -# connection to server_2 -#jppf.peer.server_2.server.host = host_2 -#jppf.peer.server_2.server.port = 11112 - -#------------------------------------------------------------------------------# -# Load-balancing configuration # -#------------------------------------------------------------------------------# - -# name of the load-balancing algorithm to use; pre-defined possible values are: -# manual | autotuned | proportional | rl | nodethreads -# it can also be the name of a user-defined algorithm. Default value is "manual" -jppf.load.balancing.algorithm = proportional - -# name of the set of parameter values (aka profile) to use for the algorithm -jppf.load.balancing.profile = proportional_profile - -# "manual" profile -jppf.load.balancing.profile.manual_profile.size = 1 - -# "autotuned" profile -jppf.load.balancing.profile.autotuned_profile.size = 5 -jppf.load.balancing.profile.autotuned_profile.minSamplesToAnalyse = 100 -jppf.load.balancing.profile.autotuned_profile.minSamplesToCheckConvergence = 50 -jppf.load.balancing.profile.autotuned_profile.maxDeviation = 0.2 -jppf.load.balancing.profile.autotuned_profile.maxGuessToStable = 50 -jppf.load.balancing.profile.autotuned_profile.sizeRatioDeviation = 1.5 -jppf.load.balancing.profile.autotuned_profile.decreaseRatio = 0.2 - -# "proportional" profile -jppf.load.balancing.profile.proportional_profile.size = 5 -jppf.load.balancing.profile.proportional_profile.initialMeanTime = 1e10 -jppf.load.balancing.profile.proportional_profile.performanceCacheSize = 300 -jppf.load.balancing.profile.proportional_profile.proportionalityFactor = 1 - -# "rl" profile -jppf.load.balancing.profile.rl_profile.performanceCacheSize = 1000 -jppf.load.balancing.profile.rl_profile.performanceVariationThreshold = 0.0001 -jppf.load.balancing.profile.rl_profile.maxActionRange = 10 - -# "nodethreads" profile -jppf.load.balancing.profile.nodethreads_profile.multiplicator = 1 - -#------------------------------------------------------------------------------# -# Other JVM options added to the java command line when the driver is started # -# as a subprocess. Multiple options are separated by spaces. # -#------------------------------------------------------------------------------# - -jppf.jvm.options = -server -XX:+HeapDumpOnOutOfMemoryError -Xms1000m -Xmx9000m - -# example with remote debugging options -#jppf.jvm.options = -server -Xmx256m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n - -#------------------------------------------------------------------------------# -# Specify alternate serialization schemes. # -# Defaults to org.jppf.serialization.DefaultJavaSerialization. # -#------------------------------------------------------------------------------# - -# default -#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization - -# built-in object serialization schemes -#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization -#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization - -# defined in the "Kryo Serialization" sample -#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization - -#------------------------------------------------------------------------------# -# Specify a data transformation class. If unspecified, no transformation occurs# -#------------------------------------------------------------------------------# - -# Defined in the "Network Data Encryption" sample -#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform - - -#------------------------------------------------------------------------------# -# whether to resolve the nodes' ip addresses into host names # -# defaults to true (resolve the addresses) # -#------------------------------------------------------------------------------# - -org.jppf.resolve.addresses = true - -#------------------------------------------------------------------------------# -# Local (in-JVM) node. When enabled, any node-specific properties will apply # -#------------------------------------------------------------------------------# - -# Enable/disable the local node. Default is false (disabled) -#jppf.local.node.enabled = false - -#------------------------------------------------------------------------------# -# In idle mode configuration. In this mode the server or node starts when no # -# mouse or keyboard activity has occurred since the specified timeout, and is # -# stopped when any new activity occurs. # -#------------------------------------------------------------------------------# - -# Idle mode enabled/disabled. Default is false (disabled) -#jppf.idle.mode.enabled = false - -# Fully qualified class name of the factory object that instantiates a platform-specific idle state detector -#jppf.idle.detector.factory = org.jppf.example.idlesystem.IdleTimeDetectorFactoryImpl - -# Time of keyboard and mouse inactivity to consider the system idle, in milliseconds -# Default value is 300000 (5 minutes) -#jppf.idle.timeout = 6000 - -# Interval between 2 successive calls to the native APIs to determine idle state changes -# Default value is 1000 -#jppf.idle.poll.interval = 1000 - -#------------------------------------------------------------------------------# -# Automatic recovery from hard failure of the nodes connections. These # -# parameters configure how the driver reacts when a node fails to respond to # -# its heartbeat messages. # -#------------------------------------------------------------------------------# - -# Enable recovery from failures on the nodes. Default to false (disabled) -#jppf.recovery.enabled = false - -# Max number of attempts to get a response from the node before the connection -# is considered broken. Default value is 3 -#jppf.recovery.max.retries = 3 - -# Max time in milliseconds allowed for each attempt to get a response from the node. -# Default value is 6000 (6 seconds) -#jppf.recovery.read.timeout = 6000 - -# Dedicated port number for the detection of node failure. Defaults to 22222. -# If server discovery is enabled on the nodes, this value will override the port number specified in the nodes -#jppf.recovery.server.port = 22222 - -# Interval in milliseconds between two runs of the connection reaper -# Default value is 60000 (1 minute) -#jppf.recovery.reaper.run.interval = 60000 - -# Number of threads allocated to the reaper. Default to the number of available CPUs -#jppf.recovery.reaper.pool.size = 8 - -#------------------------------------------------------------------------------# -# Redirecting System.out and System.err to files. # -#------------------------------------------------------------------------------# - -# file path on the file system where System.out is redirected. -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.out = System.out.log -# whether to append to an existing file or to create a new one -jppf.redirect.out.append = false - -# file path on the file system where System.err is redirected -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.err = System.err.log -# whether to append to an existing file or to create a new one -jppf.redirect.err.append = false - -#------------------------------------------------------------------------------# -# Global performance tuning parameters. These affect the performance and # -# throughput of I/O operations in JPPF. The values provided in the vanilla # -# JPPF distribution are known to offer a good performance in most situations # -# and environments. # -#------------------------------------------------------------------------------# - -# Size of send and receive buffer for socket connections. -# Defaults to 32768 and must be in range [1024, 1024*1024] -# 128 * 1024 = 131072 -jppf.socket.buffer.size = 131072 -# Size of temporary buffers (including direct buffers) used in I/O transfers. -# Defaults to 32768 and must be in range [1024, 1024*1024] -jppf.temp.buffer.size = 12288 -# Maximum size of temporary buffers pool (excluding direct buffers). When this size -# is reached, new buffers are still created, but not released into the pool, so they -# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} -# Defaults to 10 and must be in range [1, 2048] -jppf.temp.buffer.pool.size = 200 -# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). -# Defaults to 100 and must be in range [1, 2048] -jppf.length.buffer.pool.size = 100 - -#------------------------------------------------------------------------------# -# Enabling or disabling the lookup of classpath resources in the file system # -# Defaults to true (enabled) # -#------------------------------------------------------------------------------# - -#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag01.properties b/sandag_abm/src/main/resources/jppf-sandag01.properties deleted file mode 100644 index 1d5dce8..0000000 --- a/sandag_abm/src/main/resources/jppf-sandag01.properties +++ /dev/null @@ -1,298 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2015 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -#------------------------------------------------------------------------------# -# Manual configuration of the server connection # -#------------------------------------------------------------------------------# - -# Host name, or ip address, of the host the JPPF driver is running on -# defaults to 'localhost' -jppf.server.host = ${master.node.name} - -# port number the server is listening to for connections, defaults to 11111 -#jppf.server.port = 11111 - -#------------------------------------------------------------------------------# -# Configuration of JMX management server # -#------------------------------------------------------------------------------# - -# enable/disable management, defaults to true -#jppf.management.enabled = true - -# JMX management host IP address. If unspecified (recommended), the first -# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine -# will be used. If no non-local IP is found, 'localhost' will be used. -#jppf.management.host = localhost - -# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port -# is already bound, the node will automatically scan for the next available port. -#jppf.node.management.port = 12001 - -#------------------------------------------------------------------------------# -# SSL Settings # -#------------------------------------------------------------------------------# - -# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established -#jppf.ssl.enabled = true - -# location of the SSL configuration on the file system -#jppf.ssl.configuration.file = config/ssl/ssl.properties - -# SSL configuration as an arbitrary source. Value is the fully qualified name of -# an implementation of java.util.concurrent.Callable with optional space-separated arguments -#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties - -#------------------------------------------------------------------------------# -# path to the JPPF security policy file # -# comment out this entry to disable security on the node # -#------------------------------------------------------------------------------# - -#jppf.policy.file = config/jppf.policy - -#------------------------------------------------------------------------------# -# Server discovery. # -#------------------------------------------------------------------------------# - -# Enable/disable automatic discovery of JPPF drivers, defaults to true. -jppf.discovery.enabled = false - -# UDP multicast group to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default to 230.0.0.1 -#jppf.discovery.group = 230.0.0.1 - -# UDP multicast port to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Defaults to 11111 -#jppf.discovery.port = 11111 - -# How long the node will attempt to automatically discover a driver before falling -# back to the parameters specified in this configuration file. Defaults to 5000 ms -#jppf.discovery.timeout = 5000 - -# IPv4 address patterns included in the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be included -# in the list of discovered drivers. -#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 - -# IPv4 address patterns excluded from the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be excluded -# from the list of discovered drivers. -#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 - -# IPv6 address patterns included in the server dscovery mechanism -#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 - -# IPv6 address patterns excluded from the server dscovery mechanism -#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 - -#------------------------------------------------------------------------------# -# Automatic recovery from disconnection from the server # -#------------------------------------------------------------------------------# - -# number of seconds before the first reconnection attempt, default to 1 -#jppf.reconnect.initial.delay = 1 - -# time after which the node stops trying to reconnect, in seconds. Defaults to 60 -jppf.reconnect.max.time = 5 - -# time between two connection attempts, in seconds. Defaults to 1 -#jppf.reconnect.interval = 1 - -#------------------------------------------------------------------------------# -# Processing Threads: number of threads running tasks in this node. # -# default value is the number of available CPUs; uncomment to specify a # -# different value. Blocking tasks might benefit from a number larger than CPUs # -#------------------------------------------------------------------------------# - -jppf.processing.threads = ${node.1.execution.threads} - -#------------------------------------------------------------------------------# -# Thread Manager: manager that wraps the executor service for running tasks. # -# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # -#------------------------------------------------------------------------------# - -# built-in thread manager -#jppf.thread.manager.class = default - -# fork/join thread manager -#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin - -#------------------------------------------------------------------------------# -# Specify alternate serialization schemes. # -# Defaults to org.jppf.serialization.DefaultJavaSerialization. # -#------------------------------------------------------------------------------# - -# default -#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization - -# built-in object serialization schemes -#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization -#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization - -# defined in the "Kryo Serialization" sample -#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization - -#------------------------------------------------------------------------------# -# Specify a data transformation class. If unspecified, none is used. # -#------------------------------------------------------------------------------# - -# Defined in the "Network Data Encryption" sample -#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform - -#------------------------------------------------------------------------------# -# Other JVM options added to the java command line when the node is started as # -# a subprocess. Multiple options are separated by spaces. # -#------------------------------------------------------------------------------# - -jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag01.xml -Dnode.name=sandag01 - -# example with remote debugging options -#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n - -#------------------------------------------------------------------------------# -# Idle mode configuration. In idle mode, the server ot node starts when no # -# mouse or keyboard activity has occurred since the specified tiemout, and is # -# stopped when any new activity occurs. See "jppf.idle.timeout" below. # -#------------------------------------------------------------------------------# - -# enable/disable idle mode, defaults to false (disabled) -#jppf.idle.mode.enabled = false - -# Time of keyboard and mouse inactivity after which the system is considered -# idle, in ms. Defaults to 300000 (5 minutes) -#jppf.idle.timeout = 6000 - -# Interval between 2 successive calls to the native APIs to determine whether -# the system idle state has changed. Defaults to 1000 ms. -#jppf.idle.poll.interval = 1000 - -# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, -# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). -#jppf.idle.interruptIfRunning = true - -#------------------------------------------------------------------------------# -# Automatic recovery from hard failure of the server connection. These # -# parameters configure how the node reacts to the heartbeats sent by the # -# server, or lack thereof. # -#------------------------------------------------------------------------------# - -# Enable recovery from hardware failures, defaults to false (disabled) -#jppf.recovery.enabled = true - -# Dedicated port number for the detection of connection failure, must be the -# same as the value specified in the server configuration. Defaults to 22222. -# If server discovery is enabled, this value will be overridden by the port -# number specified in the driver configuration. -#jppf.recovery.server.port = 22222 - -# Maximum number of attempts to get a message from the server before the -# connection is considered broken. Default value is 2 -#jppf.recovery.max.retries = 2 - -# Maximum time in milliseconds allowed for each attempt to get a response from -# the node. Default value is 60000 ms (1 minute). - -#jppf.recovery.read.timeout = 60000 - -#------------------------------------------------------------------------------# -# JPPF class loader-related properties # -#------------------------------------------------------------------------------# - -# JPPF class loader delegation model. values: parent | url, defaults to parent -#jppf.classloader.delegation = parent - -# size of the class loader cache in the node, defaults to 50 -#jppf.classloader.cache.size = 50 - -# class loader resource cache enabled? defaults to true. -#jppf.resource.cache.enabled = true -# resource cache's type of storage: either "file" (the default) or "memory" -#jppf.resource.cache.storage = file -# root location of the file-persisted caches -#jppf.resource.cache.dir = some_directory - -#------------------------------------------------------------------------------# -# Screen saver settings # -#------------------------------------------------------------------------------# - -# include the settings from a separate file to avoid cluttering this config file -#!include file config/screensaver.properties - -#------------------------------------------------------------------------------# -# Redirecting System.out and System.err to files. # -#------------------------------------------------------------------------------# - -# file path on the file system where System.out is redirected. -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.out = System.out.log -# whether to append to an existing file or to create a new one -#jppf.redirect.out.append = false - -# file path on the file system where System.err is redirected -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.err = System.err.log -# whether to append to an existing file or to create a new one -#jppf.redirect.err.append = false - -#------------------------------------------------------------------------------# -# Node provisioning configuration # -#------------------------------------------------------------------------------# - -# Define a node as master. Defaults to true -#jppf.node.provisioning.master = true -# Define a node as a slave. Defaults to false -#jppf.node.provisioning.slave = false -# Specify the path prefix used for the root directory of each slave node -# defaults to "slave_nodes/node_", relative to the master root directory -#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ -# Specify the directory where slave-specific configuration files are located -# Defaults to the "config" folder, relative to the master root directory -#jppf.node.provisioning.slave.config.path = config -# A set of space-separated JVM options always added to the slave startup command -#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties -# Specify the number of slaves to launch upon master node startup. Defaults to 0 -#jppf.node.provisioning.startup.slaves = 0 - -#------------------------------------------------------------------------------# -# Global performance tuning parameters. These affect the performance and # -# throughput of I/O operations in JPPF. The values provided in the vanilla # -# JPPF distribution are known to offer a good performance in most situations # -# and environments. # -#------------------------------------------------------------------------------# - -# Size of send and receive buffer for socket connections. -# Defaults to 32768 and must be in range [1024, 1024*1024] -# 128 * 1024 = 131072 -#jppf.socket.buffer.size = 131072 -# Size of temporary buffers (including direct buffers) used in I/O transfers. -# Defaults to 32768 and must be in range [1024, 1024*1024] -#jppf.temp.buffer.size = 12288 -# Maximum size of temporary buffers pool (excluding direct buffers). When this size -# is reached, new buffers are still created, but not released into the pool, so they -# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} -# Defaults to 10 and must be in range [1, 2048] -#jppf.temp.buffer.pool.size = 200 -# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). -# Defaults to 100 and must be in range [1, 2048] -#jppf.length.buffer.pool.size = 100 - -#------------------------------------------------------------------------------# -# Enabling or disabling the lookup of classpath resources in the file system # -# Defaults to true (enabled) # -#------------------------------------------------------------------------------# - -#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag02.properties b/sandag_abm/src/main/resources/jppf-sandag02.properties deleted file mode 100644 index 290be6d..0000000 --- a/sandag_abm/src/main/resources/jppf-sandag02.properties +++ /dev/null @@ -1,298 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2015 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -#------------------------------------------------------------------------------# -# Manual configuration of the server connection # -#------------------------------------------------------------------------------# - -# Host name, or ip address, of the host the JPPF driver is running on -# defaults to 'localhost' -jppf.server.host = ${master.node.name} - -# port number the server is listening to for connections, defaults to 11111 -#jppf.server.port = 11111 - -#------------------------------------------------------------------------------# -# Configuration of JMX management server # -#------------------------------------------------------------------------------# - -# enable/disable management, defaults to true -#jppf.management.enabled = true - -# JMX management host IP address. If unspecified (recommended), the first -# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine -# will be used. If no non-local IP is found, 'localhost' will be used. -#jppf.management.host = localhost - -# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port -# is already bound, the node will automatically scan for the next available port. -#jppf.node.management.port = 12001 - -#------------------------------------------------------------------------------# -# SSL Settings # -#------------------------------------------------------------------------------# - -# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established -#jppf.ssl.enabled = true - -# location of the SSL configuration on the file system -#jppf.ssl.configuration.file = config/ssl/ssl.properties - -# SSL configuration as an arbitrary source. Value is the fully qualified name of -# an implementation of java.util.concurrent.Callable with optional space-separated arguments -#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties - -#------------------------------------------------------------------------------# -# path to the JPPF security policy file # -# comment out this entry to disable security on the node # -#------------------------------------------------------------------------------# - -#jppf.policy.file = config/jppf.policy - -#------------------------------------------------------------------------------# -# Server discovery. # -#------------------------------------------------------------------------------# - -# Enable/disable automatic discovery of JPPF drivers, defaults to true. -jppf.discovery.enabled = false - -# UDP multicast group to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default to 230.0.0.1 -#jppf.discovery.group = 230.0.0.1 - -# UDP multicast port to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Defaults to 11111 -#jppf.discovery.port = 11111 - -# How long the node will attempt to automatically discover a driver before falling -# back to the parameters specified in this configuration file. Defaults to 5000 ms -#jppf.discovery.timeout = 5000 - -# IPv4 address patterns included in the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be included -# in the list of discovered drivers. -#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 - -# IPv4 address patterns excluded from the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be excluded -# from the list of discovered drivers. -#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 - -# IPv6 address patterns included in the server dscovery mechanism -#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 - -# IPv6 address patterns excluded from the server dscovery mechanism -#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 - -#------------------------------------------------------------------------------# -# Automatic recovery from disconnection from the server # -#------------------------------------------------------------------------------# - -# number of seconds before the first reconnection attempt, default to 1 -#jppf.reconnect.initial.delay = 1 - -# time after which the node stops trying to reconnect, in seconds. Defaults to 60 -jppf.reconnect.max.time = 5 - -# time between two connection attempts, in seconds. Defaults to 1 -#jppf.reconnect.interval = 1 - -#------------------------------------------------------------------------------# -# Processing Threads: number of threads running tasks in this node. # -# default value is the number of available CPUs; uncomment to specify a # -# different value. Blocking tasks might benefit from a number larger than CPUs # -#------------------------------------------------------------------------------# - -jppf.processing.threads = ${node.1.execution.threads} - -#------------------------------------------------------------------------------# -# Thread Manager: manager that wraps the executor service for running tasks. # -# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # -#------------------------------------------------------------------------------# - -# built-in thread manager -#jppf.thread.manager.class = default - -# fork/join thread manager -#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin - -#------------------------------------------------------------------------------# -# Specify alternate serialization schemes. # -# Defaults to org.jppf.serialization.DefaultJavaSerialization. # -#------------------------------------------------------------------------------# - -# default -#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization - -# built-in object serialization schemes -#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization -#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization - -# defined in the "Kryo Serialization" sample -#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization - -#------------------------------------------------------------------------------# -# Specify a data transformation class. If unspecified, none is used. # -#------------------------------------------------------------------------------# - -# Defined in the "Network Data Encryption" sample -#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform - -#------------------------------------------------------------------------------# -# Other JVM options added to the java command line when the node is started as # -# a subprocess. Multiple options are separated by spaces. # -#------------------------------------------------------------------------------# - -jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag02.xml -Dnode.name=sandag02 - -# example with remote debugging options -#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n - -#------------------------------------------------------------------------------# -# Idle mode configuration. In idle mode, the server ot node starts when no # -# mouse or keyboard activity has occurred since the specified tiemout, and is # -# stopped when any new activity occurs. See "jppf.idle.timeout" below. # -#------------------------------------------------------------------------------# - -# enable/disable idle mode, defaults to false (disabled) -#jppf.idle.mode.enabled = false - -# Time of keyboard and mouse inactivity after which the system is considered -# idle, in ms. Defaults to 300000 (5 minutes) -#jppf.idle.timeout = 6000 - -# Interval between 2 successive calls to the native APIs to determine whether -# the system idle state has changed. Defaults to 1000 ms. -#jppf.idle.poll.interval = 1000 - -# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, -# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). -#jppf.idle.interruptIfRunning = true - -#------------------------------------------------------------------------------# -# Automatic recovery from hard failure of the server connection. These # -# parameters configure how the node reacts to the heartbeats sent by the # -# server, or lack thereof. # -#------------------------------------------------------------------------------# - -# Enable recovery from hardware failures, defaults to false (disabled) -#jppf.recovery.enabled = true - -# Dedicated port number for the detection of connection failure, must be the -# same as the value specified in the server configuration. Defaults to 22222. -# If server discovery is enabled, this value will be overridden by the port -# number specified in the driver configuration. -#jppf.recovery.server.port = 22222 - -# Maximum number of attempts to get a message from the server before the -# connection is considered broken. Default value is 2 -#jppf.recovery.max.retries = 2 - -# Maximum time in milliseconds allowed for each attempt to get a response from -# the node. Default value is 60000 ms (1 minute). - -#jppf.recovery.read.timeout = 60000 - -#------------------------------------------------------------------------------# -# JPPF class loader-related properties # -#------------------------------------------------------------------------------# - -# JPPF class loader delegation model. values: parent | url, defaults to parent -#jppf.classloader.delegation = parent - -# size of the class loader cache in the node, defaults to 50 -#jppf.classloader.cache.size = 50 - -# class loader resource cache enabled? defaults to true. -#jppf.resource.cache.enabled = true -# resource cache's type of storage: either "file" (the default) or "memory" -#jppf.resource.cache.storage = file -# root location of the file-persisted caches -#jppf.resource.cache.dir = some_directory - -#------------------------------------------------------------------------------# -# Screen saver settings # -#------------------------------------------------------------------------------# - -# include the settings from a separate file to avoid cluttering this config file -#!include file config/screensaver.properties - -#------------------------------------------------------------------------------# -# Redirecting System.out and System.err to files. # -#------------------------------------------------------------------------------# - -# file path on the file system where System.out is redirected. -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.out = System.out.log -# whether to append to an existing file or to create a new one -#jppf.redirect.out.append = false - -# file path on the file system where System.err is redirected -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.err = System.err.log -# whether to append to an existing file or to create a new one -#jppf.redirect.err.append = false - -#------------------------------------------------------------------------------# -# Node provisioning configuration # -#------------------------------------------------------------------------------# - -# Define a node as master. Defaults to true -#jppf.node.provisioning.master = true -# Define a node as a slave. Defaults to false -#jppf.node.provisioning.slave = false -# Specify the path prefix used for the root directory of each slave node -# defaults to "slave_nodes/node_", relative to the master root directory -#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ -# Specify the directory where slave-specific configuration files are located -# Defaults to the "config" folder, relative to the master root directory -#jppf.node.provisioning.slave.config.path = config -# A set of space-separated JVM options always added to the slave startup command -#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties -# Specify the number of slaves to launch upon master node startup. Defaults to 0 -#jppf.node.provisioning.startup.slaves = 0 - -#------------------------------------------------------------------------------# -# Global performance tuning parameters. These affect the performance and # -# throughput of I/O operations in JPPF. The values provided in the vanilla # -# JPPF distribution are known to offer a good performance in most situations # -# and environments. # -#------------------------------------------------------------------------------# - -# Size of send and receive buffer for socket connections. -# Defaults to 32768 and must be in range [1024, 1024*1024] -# 128 * 1024 = 131072 -#jppf.socket.buffer.size = 131072 -# Size of temporary buffers (including direct buffers) used in I/O transfers. -# Defaults to 32768 and must be in range [1024, 1024*1024] -#jppf.temp.buffer.size = 12288 -# Maximum size of temporary buffers pool (excluding direct buffers). When this size -# is reached, new buffers are still created, but not released into the pool, so they -# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} -# Defaults to 10 and must be in range [1, 2048] -#jppf.temp.buffer.pool.size = 200 -# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). -# Defaults to 100 and must be in range [1, 2048] -#jppf.length.buffer.pool.size = 100 - -#------------------------------------------------------------------------------# -# Enabling or disabling the lookup of classpath resources in the file system # -# Defaults to true (enabled) # -#------------------------------------------------------------------------------# - -#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag03.properties b/sandag_abm/src/main/resources/jppf-sandag03.properties deleted file mode 100644 index ab3a6a2..0000000 --- a/sandag_abm/src/main/resources/jppf-sandag03.properties +++ /dev/null @@ -1,298 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2015 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -#------------------------------------------------------------------------------# -# Manual configuration of the server connection # -#------------------------------------------------------------------------------# - -# Host name, or ip address, of the host the JPPF driver is running on -# defaults to 'localhost' -jppf.server.host = ${master.node.name} - -# port number the server is listening to for connections, defaults to 11111 -#jppf.server.port = 11111 - -#------------------------------------------------------------------------------# -# Configuration of JMX management server # -#------------------------------------------------------------------------------# - -# enable/disable management, defaults to true -#jppf.management.enabled = true - -# JMX management host IP address. If unspecified (recommended), the first -# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine -# will be used. If no non-local IP is found, 'localhost' will be used. -#jppf.management.host = localhost - -# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port -# is already bound, the node will automatically scan for the next available port. -#jppf.node.management.port = 12001 - -#------------------------------------------------------------------------------# -# SSL Settings # -#------------------------------------------------------------------------------# - -# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established -#jppf.ssl.enabled = true - -# location of the SSL configuration on the file system -#jppf.ssl.configuration.file = config/ssl/ssl.properties - -# SSL configuration as an arbitrary source. Value is the fully qualified name of -# an implementation of java.util.concurrent.Callable with optional space-separated arguments -#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties - -#------------------------------------------------------------------------------# -# path to the JPPF security policy file # -# comment out this entry to disable security on the node # -#------------------------------------------------------------------------------# - -#jppf.policy.file = config/jppf.policy - -#------------------------------------------------------------------------------# -# Server discovery. # -#------------------------------------------------------------------------------# - -# Enable/disable automatic discovery of JPPF drivers, defaults to true. -jppf.discovery.enabled = false - -# UDP multicast group to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default to 230.0.0.1 -#jppf.discovery.group = 230.0.0.1 - -# UDP multicast port to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Defaults to 11111 -#jppf.discovery.port = 11111 - -# How long the node will attempt to automatically discover a driver before falling -# back to the parameters specified in this configuration file. Defaults to 5000 ms -#jppf.discovery.timeout = 5000 - -# IPv4 address patterns included in the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be included -# in the list of discovered drivers. -#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 - -# IPv4 address patterns excluded from the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be excluded -# from the list of discovered drivers. -#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 - -# IPv6 address patterns included in the server dscovery mechanism -#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 - -# IPv6 address patterns excluded from the server dscovery mechanism -#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 - -#------------------------------------------------------------------------------# -# Automatic recovery from disconnection from the server # -#------------------------------------------------------------------------------# - -# number of seconds before the first reconnection attempt, default to 1 -#jppf.reconnect.initial.delay = 1 - -# time after which the node stops trying to reconnect, in seconds. Defaults to 60 -jppf.reconnect.max.time = 5 - -# time between two connection attempts, in seconds. Defaults to 1 -#jppf.reconnect.interval = 1 - -#------------------------------------------------------------------------------# -# Processing Threads: number of threads running tasks in this node. # -# default value is the number of available CPUs; uncomment to specify a # -# different value. Blocking tasks might benefit from a number larger than CPUs # -#------------------------------------------------------------------------------# - -jppf.processing.threads = ${node.2.execution.threads} - -#------------------------------------------------------------------------------# -# Thread Manager: manager that wraps the executor service for running tasks. # -# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # -#------------------------------------------------------------------------------# - -# built-in thread manager -#jppf.thread.manager.class = default - -# fork/join thread manager -#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin - -#------------------------------------------------------------------------------# -# Specify alternate serialization schemes. # -# Defaults to org.jppf.serialization.DefaultJavaSerialization. # -#------------------------------------------------------------------------------# - -# default -#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization - -# built-in object serialization schemes -#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization -#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization - -# defined in the "Kryo Serialization" sample -#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization - -#------------------------------------------------------------------------------# -# Specify a data transformation class. If unspecified, none is used. # -#------------------------------------------------------------------------------# - -# Defined in the "Network Data Encryption" sample -#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform - -#------------------------------------------------------------------------------# -# Other JVM options added to the java command line when the node is started as # -# a subprocess. Multiple options are separated by spaces. # -#------------------------------------------------------------------------------# - -jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag03.xml -Dnode.name=sandag03 - -# example with remote debugging options -#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n - -#------------------------------------------------------------------------------# -# Idle mode configuration. In idle mode, the server ot node starts when no # -# mouse or keyboard activity has occurred since the specified tiemout, and is # -# stopped when any new activity occurs. See "jppf.idle.timeout" below. # -#------------------------------------------------------------------------------# - -# enable/disable idle mode, defaults to false (disabled) -#jppf.idle.mode.enabled = false - -# Time of keyboard and mouse inactivity after which the system is considered -# idle, in ms. Defaults to 300000 (5 minutes) -#jppf.idle.timeout = 6000 - -# Interval between 2 successive calls to the native APIs to determine whether -# the system idle state has changed. Defaults to 1000 ms. -#jppf.idle.poll.interval = 1000 - -# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, -# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). -#jppf.idle.interruptIfRunning = true - -#------------------------------------------------------------------------------# -# Automatic recovery from hard failure of the server connection. These # -# parameters configure how the node reacts to the heartbeats sent by the # -# server, or lack thereof. # -#------------------------------------------------------------------------------# - -# Enable recovery from hardware failures, defaults to false (disabled) -#jppf.recovery.enabled = true - -# Dedicated port number for the detection of connection failure, must be the -# same as the value specified in the server configuration. Defaults to 22222. -# If server discovery is enabled, this value will be overridden by the port -# number specified in the driver configuration. -#jppf.recovery.server.port = 22222 - -# Maximum number of attempts to get a message from the server before the -# connection is considered broken. Default value is 2 -#jppf.recovery.max.retries = 2 - -# Maximum time in milliseconds allowed for each attempt to get a response from -# the node. Default value is 60000 ms (1 minute). - -#jppf.recovery.read.timeout = 60000 - -#------------------------------------------------------------------------------# -# JPPF class loader-related properties # -#------------------------------------------------------------------------------# - -# JPPF class loader delegation model. values: parent | url, defaults to parent -#jppf.classloader.delegation = parent - -# size of the class loader cache in the node, defaults to 50 -#jppf.classloader.cache.size = 50 - -# class loader resource cache enabled? defaults to true. -#jppf.resource.cache.enabled = true -# resource cache's type of storage: either "file" (the default) or "memory" -#jppf.resource.cache.storage = file -# root location of the file-persisted caches -#jppf.resource.cache.dir = some_directory - -#------------------------------------------------------------------------------# -# Screen saver settings # -#------------------------------------------------------------------------------# - -# include the settings from a separate file to avoid cluttering this config file -#!include file config/screensaver.properties - -#------------------------------------------------------------------------------# -# Redirecting System.out and System.err to files. # -#------------------------------------------------------------------------------# - -# file path on the file system where System.out is redirected. -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.out = System.out.log -# whether to append to an existing file or to create a new one -#jppf.redirect.out.append = false - -# file path on the file system where System.err is redirected -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.err = System.err.log -# whether to append to an existing file or to create a new one -#jppf.redirect.err.append = false - -#------------------------------------------------------------------------------# -# Node provisioning configuration # -#------------------------------------------------------------------------------# - -# Define a node as master. Defaults to true -#jppf.node.provisioning.master = true -# Define a node as a slave. Defaults to false -#jppf.node.provisioning.slave = false -# Specify the path prefix used for the root directory of each slave node -# defaults to "slave_nodes/node_", relative to the master root directory -#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ -# Specify the directory where slave-specific configuration files are located -# Defaults to the "config" folder, relative to the master root directory -#jppf.node.provisioning.slave.config.path = config -# A set of space-separated JVM options always added to the slave startup command -#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties -# Specify the number of slaves to launch upon master node startup. Defaults to 0 -#jppf.node.provisioning.startup.slaves = 0 - -#------------------------------------------------------------------------------# -# Global performance tuning parameters. These affect the performance and # -# throughput of I/O operations in JPPF. The values provided in the vanilla # -# JPPF distribution are known to offer a good performance in most situations # -# and environments. # -#------------------------------------------------------------------------------# - -# Size of send and receive buffer for socket connections. -# Defaults to 32768 and must be in range [1024, 1024*1024] -# 128 * 1024 = 131072 -#jppf.socket.buffer.size = 131072 -# Size of temporary buffers (including direct buffers) used in I/O transfers. -# Defaults to 32768 and must be in range [1024, 1024*1024] -#jppf.temp.buffer.size = 12288 -# Maximum size of temporary buffers pool (excluding direct buffers). When this size -# is reached, new buffers are still created, but not released into the pool, so they -# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} -# Defaults to 10 and must be in range [1, 2048] -#jppf.temp.buffer.pool.size = 200 -# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). -# Defaults to 100 and must be in range [1, 2048] -#jppf.length.buffer.pool.size = 100 - -#------------------------------------------------------------------------------# -# Enabling or disabling the lookup of classpath resources in the file system # -# Defaults to true (enabled) # -#------------------------------------------------------------------------------# - -#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag04.properties b/sandag_abm/src/main/resources/jppf-sandag04.properties deleted file mode 100644 index 61f887c..0000000 --- a/sandag_abm/src/main/resources/jppf-sandag04.properties +++ /dev/null @@ -1,298 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2015 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -#------------------------------------------------------------------------------# -# Manual configuration of the server connection # -#------------------------------------------------------------------------------# - -# Host name, or ip address, of the host the JPPF driver is running on -# defaults to 'localhost' -jppf.server.host = ${master.node.name} - -# port number the server is listening to for connections, defaults to 11111 -#jppf.server.port = 11111 - -#------------------------------------------------------------------------------# -# Configuration of JMX management server # -#------------------------------------------------------------------------------# - -# enable/disable management, defaults to true -#jppf.management.enabled = true - -# JMX management host IP address. If unspecified (recommended), the first -# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine -# will be used. If no non-local IP is found, 'localhost' will be used. -#jppf.management.host = localhost - -# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port -# is already bound, the node will automatically scan for the next available port. -#jppf.node.management.port = 12001 - -#------------------------------------------------------------------------------# -# SSL Settings # -#------------------------------------------------------------------------------# - -# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established -#jppf.ssl.enabled = true - -# location of the SSL configuration on the file system -#jppf.ssl.configuration.file = config/ssl/ssl.properties - -# SSL configuration as an arbitrary source. Value is the fully qualified name of -# an implementation of java.util.concurrent.Callable with optional space-separated arguments -#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties - -#------------------------------------------------------------------------------# -# path to the JPPF security policy file # -# comment out this entry to disable security on the node # -#------------------------------------------------------------------------------# - -#jppf.policy.file = config/jppf.policy - -#------------------------------------------------------------------------------# -# Server discovery. # -#------------------------------------------------------------------------------# - -# Enable/disable automatic discovery of JPPF drivers, defaults to true. -jppf.discovery.enabled = false - -# UDP multicast group to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Default to 230.0.0.1 -#jppf.discovery.group = 230.0.0.1 - -# UDP multicast port to which drivers broadcast their connection parameters -# and to which clients and nodes listen. Defaults to 11111 -#jppf.discovery.port = 11111 - -# How long the node will attempt to automatically discover a driver before falling -# back to the parameters specified in this configuration file. Defaults to 5000 ms -#jppf.discovery.timeout = 5000 - -# IPv4 address patterns included in the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be included -# in the list of discovered drivers. -#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 - -# IPv4 address patterns excluded from the server dscovery mechanism -# Drivers whose IPv4 address matches the pattern will be excluded -# from the list of discovered drivers. -#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 - -# IPv6 address patterns included in the server dscovery mechanism -#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 - -# IPv6 address patterns excluded from the server dscovery mechanism -#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 - -#------------------------------------------------------------------------------# -# Automatic recovery from disconnection from the server # -#------------------------------------------------------------------------------# - -# number of seconds before the first reconnection attempt, default to 1 -#jppf.reconnect.initial.delay = 1 - -# time after which the node stops trying to reconnect, in seconds. Defaults to 60 -jppf.reconnect.max.time = 5 - -# time between two connection attempts, in seconds. Defaults to 1 -#jppf.reconnect.interval = 1 - -#------------------------------------------------------------------------------# -# Processing Threads: number of threads running tasks in this node. # -# default value is the number of available CPUs; uncomment to specify a # -# different value. Blocking tasks might benefit from a number larger than CPUs # -#------------------------------------------------------------------------------# - -jppf.processing.threads = ${node.3.execution.threads} - -#------------------------------------------------------------------------------# -# Thread Manager: manager that wraps the executor service for running tasks. # -# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # -#------------------------------------------------------------------------------# - -# built-in thread manager -#jppf.thread.manager.class = default - -# fork/join thread manager -#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin - -#------------------------------------------------------------------------------# -# Specify alternate serialization schemes. # -# Defaults to org.jppf.serialization.DefaultJavaSerialization. # -#------------------------------------------------------------------------------# - -# default -#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization - -# built-in object serialization schemes -#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization -#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization - -# defined in the "Kryo Serialization" sample -#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization - -#------------------------------------------------------------------------------# -# Specify a data transformation class. If unspecified, none is used. # -#------------------------------------------------------------------------------# - -# Defined in the "Network Data Encryption" sample -#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform - -#------------------------------------------------------------------------------# -# Other JVM options added to the java command line when the node is started as # -# a subprocess. Multiple options are separated by spaces. # -#------------------------------------------------------------------------------# - -jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag04.xml -Dnode.name=sandag04 - -# example with remote debugging options -#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n - -#------------------------------------------------------------------------------# -# Idle mode configuration. In idle mode, the server ot node starts when no # -# mouse or keyboard activity has occurred since the specified tiemout, and is # -# stopped when any new activity occurs. See "jppf.idle.timeout" below. # -#------------------------------------------------------------------------------# - -# enable/disable idle mode, defaults to false (disabled) -#jppf.idle.mode.enabled = false - -# Time of keyboard and mouse inactivity after which the system is considered -# idle, in ms. Defaults to 300000 (5 minutes) -#jppf.idle.timeout = 6000 - -# Interval between 2 successive calls to the native APIs to determine whether -# the system idle state has changed. Defaults to 1000 ms. -#jppf.idle.poll.interval = 1000 - -# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, -# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). -#jppf.idle.interruptIfRunning = true - -#------------------------------------------------------------------------------# -# Automatic recovery from hard failure of the server connection. These # -# parameters configure how the node reacts to the heartbeats sent by the # -# server, or lack thereof. # -#------------------------------------------------------------------------------# - -# Enable recovery from hardware failures, defaults to false (disabled) -#jppf.recovery.enabled = true - -# Dedicated port number for the detection of connection failure, must be the -# same as the value specified in the server configuration. Defaults to 22222. -# If server discovery is enabled, this value will be overridden by the port -# number specified in the driver configuration. -#jppf.recovery.server.port = 22222 - -# Maximum number of attempts to get a message from the server before the -# connection is considered broken. Default value is 2 -#jppf.recovery.max.retries = 2 - -# Maximum time in milliseconds allowed for each attempt to get a response from -# the node. Default value is 60000 ms (1 minute). - -#jppf.recovery.read.timeout = 60000 - -#------------------------------------------------------------------------------# -# JPPF class loader-related properties # -#------------------------------------------------------------------------------# - -# JPPF class loader delegation model. values: parent | url, defaults to parent -#jppf.classloader.delegation = parent - -# size of the class loader cache in the node, defaults to 50 -#jppf.classloader.cache.size = 50 - -# class loader resource cache enabled? defaults to true. -#jppf.resource.cache.enabled = true -# resource cache's type of storage: either "file" (the default) or "memory" -#jppf.resource.cache.storage = file -# root location of the file-persisted caches -#jppf.resource.cache.dir = some_directory - -#------------------------------------------------------------------------------# -# Screen saver settings # -#------------------------------------------------------------------------------# - -# include the settings from a separate file to avoid cluttering this config file -#!include file config/screensaver.properties - -#------------------------------------------------------------------------------# -# Redirecting System.out and System.err to files. # -#------------------------------------------------------------------------------# - -# file path on the file system where System.out is redirected. -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.out = System.out.log -# whether to append to an existing file or to create a new one -#jppf.redirect.out.append = false - -# file path on the file system where System.err is redirected -# if unspecified or invalid, then no redirection occurs -#jppf.redirect.err = System.err.log -# whether to append to an existing file or to create a new one -#jppf.redirect.err.append = false - -#------------------------------------------------------------------------------# -# Node provisioning configuration # -#------------------------------------------------------------------------------# - -# Define a node as master. Defaults to true -#jppf.node.provisioning.master = true -# Define a node as a slave. Defaults to false -#jppf.node.provisioning.slave = false -# Specify the path prefix used for the root directory of each slave node -# defaults to "slave_nodes/node_", relative to the master root directory -#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ -# Specify the directory where slave-specific configuration files are located -# Defaults to the "config" folder, relative to the master root directory -#jppf.node.provisioning.slave.config.path = config -# A set of space-separated JVM options always added to the slave startup command -#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties -# Specify the number of slaves to launch upon master node startup. Defaults to 0 -#jppf.node.provisioning.startup.slaves = 0 - -#------------------------------------------------------------------------------# -# Global performance tuning parameters. These affect the performance and # -# throughput of I/O operations in JPPF. The values provided in the vanilla # -# JPPF distribution are known to offer a good performance in most situations # -# and environments. # -#------------------------------------------------------------------------------# - -# Size of send and receive buffer for socket connections. -# Defaults to 32768 and must be in range [1024, 1024*1024] -# 128 * 1024 = 131072 -#jppf.socket.buffer.size = 131072 -# Size of temporary buffers (including direct buffers) used in I/O transfers. -# Defaults to 32768 and must be in range [1024, 1024*1024] -#jppf.temp.buffer.size = 12288 -# Maximum size of temporary buffers pool (excluding direct buffers). When this size -# is reached, new buffers are still created, but not released into the pool, so they -# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} -# Defaults to 10 and must be in range [1, 2048] -#jppf.temp.buffer.pool.size = 200 -# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). -# Defaults to 100 and must be in range [1, 2048] -#jppf.length.buffer.pool.size = 100 - -#------------------------------------------------------------------------------# -# Enabling or disabling the lookup of classpath resources in the file system # -# Defaults to true (enabled) # -#------------------------------------------------------------------------------# - -#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/log4j-client.properties b/sandag_abm/src/main/resources/log4j-client.properties deleted file mode 100644 index 4f4307a..0000000 --- a/sandag_abm/src/main/resources/log4j-client.properties +++ /dev/null @@ -1,39 +0,0 @@ -#------------------------------------------------------------------------------# -# Java Parallel Processing Framework. # -# Copyright (C) 2005-2008 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-client.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=logfiles/jppf-client.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -log4j.rootLogger=INFO, JPPF -#log4j.rootLogger=DEBUG, JPPF - -#log4j.logger.org.jppf.client=DEBUG -#log4j.logger.org.jppf.common.socket=DEBUG diff --git a/sandag_abm/src/main/resources/log4j-driver.properties b/sandag_abm/src/main/resources/log4j-driver.properties deleted file mode 100644 index dbf6fa8..0000000 --- a/sandag_abm/src/main/resources/log4j-driver.properties +++ /dev/null @@ -1,42 +0,0 @@ -#------------------------------------------------------------------------------# -# Java Parallel Processing Framework. # -# Copyright (C) 2005-2008 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-driver.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=./logFiles/jppf-driver.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -#log4j.logger.org.jppf.server.nio.StateTransitionTask=DEBUG - -# log information about interactions between the client and server -#log4j.logger.org.jppf.server.app=DEBUG -#log4j.logger.org.jppf.io.IOHelper=DEBUG - -log4j.rootLogger=INFO, JPPF -#log4j.rootLogger=DEBUG, JPPF diff --git a/sandag_abm/src/main/resources/log4j-sandag01.properties b/sandag_abm/src/main/resources/log4j-sandag01.properties deleted file mode 100644 index 01af856..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag01.properties +++ /dev/null @@ -1,46 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2010 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-node.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=logFiles/event-sandag01.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### direct messages to the JMX Logger ### -log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender -log4j.appender.JMX.layout=org.apache.log4j.PatternLayout -log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -# will produce messages like "writing object size = " when sending to the server -log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE -# will produce messages like "i = , read data size = " when receiving from the server -log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE - -#log4j.rootLogger=TRACE, DEBUG, JPPF -log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag01.xml b/sandag_abm/src/main/resources/log4j-sandag01.xml deleted file mode 100644 index 4f5ed1d..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag01.xml +++ /dev/null @@ -1,439 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j-sandag02.properties b/sandag_abm/src/main/resources/log4j-sandag02.properties deleted file mode 100644 index 9aae59a..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag02.properties +++ /dev/null @@ -1,46 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2010 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-node.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=logFiles/event-sandag02.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### direct messages to the JMX Logger ### -log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender -log4j.appender.JMX.layout=org.apache.log4j.PatternLayout -log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -# will produce messages like "writing object size = " when sending to the server -log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE -# will produce messages like "i = , read data size = " when receiving from the server -log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE - -#log4j.rootLogger=TRACE, DEBUG, JPPF -log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag02.xml b/sandag_abm/src/main/resources/log4j-sandag02.xml deleted file mode 100644 index 5629089..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag02.xml +++ /dev/null @@ -1,439 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j-sandag03.properties b/sandag_abm/src/main/resources/log4j-sandag03.properties deleted file mode 100644 index 771824a..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag03.properties +++ /dev/null @@ -1,46 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2010 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-node.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=logFiles/event-sandag03.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### direct messages to the JMX Logger ### -log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender -log4j.appender.JMX.layout=org.apache.log4j.PatternLayout -log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -# will produce messages like "writing object size = " when sending to the server -log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE -# will produce messages like "i = , read data size = " when receiving from the server -log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE - -#log4j.rootLogger=TRACE, DEBUG, JPPF -log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag03.xml b/sandag_abm/src/main/resources/log4j-sandag03.xml deleted file mode 100644 index 2377353..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag03.xml +++ /dev/null @@ -1,439 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j-sandag04.properties b/sandag_abm/src/main/resources/log4j-sandag04.properties deleted file mode 100644 index fea28c2..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag04.properties +++ /dev/null @@ -1,46 +0,0 @@ -#------------------------------------------------------------------------------# -# JPPF # -# Copyright (C) 2005-2010 JPPF Team. # -# http://www.jppf.org # -# # -# 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. # -#------------------------------------------------------------------------------# - -### direct log messages to stdout ### -#log4j.appender.stdout=org.apache.log4j.ConsoleAppender -#log4j.appender.stdout.Target=System.out -#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n - -### direct messages to file jppf-node.log ### -log4j.appender.JPPF=org.apache.log4j.FileAppender -log4j.appender.JPPF.File=logFiles/event-sandag04.log -log4j.appender.JPPF.Append=false -log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout -#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n -log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### direct messages to the JMX Logger ### -log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender -log4j.appender.JMX.layout=org.apache.log4j.PatternLayout -log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n - -### set log levels - for more verbose logging change 'info' to 'debug' ### - -# will produce messages like "writing object size = " when sending to the server -log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE -# will produce messages like "i = , read data size = " when receiving from the server -log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE - -#log4j.rootLogger=TRACE, DEBUG, JPPF -log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag04.xml b/sandag_abm/src/main/resources/log4j-sandag04.xml deleted file mode 100644 index e7937af..0000000 --- a/sandag_abm/src/main/resources/log4j-sandag04.xml +++ /dev/null @@ -1,439 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j.xml b/sandag_abm/src/main/resources/log4j.xml deleted file mode 100644 index 97a3996..0000000 --- a/sandag_abm/src/main/resources/log4j.xml +++ /dev/null @@ -1,443 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml b/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml deleted file mode 100644 index ad2ebf9..0000000 --- a/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j_d2t.xml b/sandag_abm/src/main/resources/log4j_d2t.xml deleted file mode 100644 index ec13a3d..0000000 --- a/sandag_abm/src/main/resources/log4j_d2t.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j_hh.xml b/sandag_abm/src/main/resources/log4j_hh.xml deleted file mode 100644 index 834f12e..0000000 --- a/sandag_abm/src/main/resources/log4j_hh.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j_mtx.xml b/sandag_abm/src/main/resources/log4j_mtx.xml deleted file mode 100644 index d8c6e40..0000000 --- a/sandag_abm/src/main/resources/log4j_mtx.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/log4j_test.xml b/sandag_abm/src/main/resources/log4j_test.xml deleted file mode 100644 index 7f52d65..0000000 --- a/sandag_abm/src/main/resources/log4j_test.xml +++ /dev/null @@ -1,455 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/sandag_abm/src/main/resources/mapAndRun.bat b/sandag_abm/src/main/resources/mapAndRun.bat deleted file mode 100644 index b1318cf..0000000 --- a/sandag_abm/src/main/resources/mapAndRun.bat +++ /dev/null @@ -1,33 +0,0 @@ -:: script to map a drive and then call a batch file remotely using psexec -:: 1 is the drive letter to map (e.g. “M:”) -:: 2 is the share to map (e.g. “\\w-ampdx-d-sag01\mtc”) -:: 3 is the password -:: 4 is the user -:: 5 is the working directory for calling the batch file (starting from the mapped drive letter) -:: 6 is the name of the batch file to call -:: 7-10 are extra arguments (note DOS only does 9 arguments unless you use SHIFT) - -SET ONE=%1 -SET TWO=%2 -SET THREE=%3 -SET FOUR=%4 -SET FIVE=%5 -SET SIX=%6 -SET SEVEN=%7 -SET EIGHT=%8 -SET NINE=%9 -SHIFT -SHIFT -SHIFT -SHIFT -SHIFT -SHIFT -SHIFT -SHIFT -SHIFT -SET TEN=%1 - -net use %ONE% %TWO% /persistent:yes -%ONE% -cd %FIVE% -call %SIX% %SEVEN% %EIGHT% %NINE% %TEN% diff --git a/sandag_abm/src/main/resources/pskill.exe b/sandag_abm/src/main/resources/pskill.exe deleted file mode 100644 index 2e9d0d49731b66e7885344e9cd2d398c274c56b6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 621944 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

  • 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 x;a)38D;#^NsXjIQ&a2< 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%0A_aAnYzk!umR7wy^?%RNcuq-&B?F06%1I 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 =C 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 z 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%0Agfw(T_P^Pq5;SLLRP-%+K-7X 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%0Ap|)s-@(E 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 >HFW{#*(Nz`UB?$e}@5Lo-fEPxx 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 leIu}M1on- +# with optional space-separated arguments +#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl-server.properties + +# enable secure communications with other servers; defaults to false (disabled)# +#jppf.peer.ssl.enabled = true + +#------------------------------------------------------------------------------# +# Enabling and configuring JMX features # +#------------------------------------------------------------------------------# + +# non-secure JMX connections; default is true (enabled) +#jppf.management.enabled = true + +# secure JMX connections via SSL/TLS; default is false (disabled) +#jppf.management.ssl.enabled = true + +# JMX management host IP address. If not specified (recommended), the first non-local +# IP address (i.e. neither 127.0.0.1 nor localhost) on this machine will be used. +# If no non-local IP is found, localhost will be used +#jppf.management.host = localhost + +# JMX management port. Defaults to 11198. If the port is already bound, the driver +# will scan for the first available port instead. +#jppf.management.port = 11198 + +#------------------------------------------------------------------------------# +# Configuration of the driver discovery broadcast service # +#------------------------------------------------------------------------------# + +# Enable/Disable automatic discovery of this JPPF drivers; default to true +#jppf.discovery.enabled = false + +# UDP multicast group to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default value is 230.0.0.1 +#jppf.discovery.group = 230.0.0.1 + +# UDP multicast port to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default value is 11111 +#jppf.discovery.port = 11111 + +# Time between 2 broadcasts, in milliseconds. Default value is 1000 +#jppf.discovery.broadcast.interval = 1000 + +# IPv4 inclusion patterns: broadcast these ipv4 addresses +#jppf.discovery.broadcast.include.ipv4 = 192.168.1.; 192.168.1.0/24 + +# IPv4 exclusion patterns: do not broadcast these ipv4 addresses +#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 + +# IPv6 inclusion patterns: broadcast these ipv6 addresses +#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 + +# IPv6 exclusion patterns: do not broadcast these ipv6 addresses +#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/64 + +#------------------------------------------------------------------------------# +# Connection with other servers, enabling P2P communication # +#------------------------------------------------------------------------------# + +# Enable/disable auto-discovery of remote peer drivers. Default value is false +#jppf.peer.discovery.enabled = true + +# manual configuration of peer servers, as a space-separated list of peers names to connect to +#jppf.peers = server_1 server_2 + +# enable both automatic and manual discovery +#jppf.peers = jppf_discovery server_1 server_2 + +# connection to server_1 +#jppf.peer.server_1.server.host = host_1 +#jppf.peer.server_1.server.port = 11111 +# connection to server_2 +#jppf.peer.server_2.server.host = host_2 +#jppf.peer.server_2.server.port = 11112 + +#------------------------------------------------------------------------------# +# Load-balancing configuration # +#------------------------------------------------------------------------------# + +# name of the load-balancing algorithm to use; pre-defined possible values are: +# manual | autotuned | proportional | rl | nodethreads +# it can also be the name of a user-defined algorithm. Default value is "manual" +jppf.load.balancing.algorithm = proportional + +# name of the set of parameter values (aka profile) to use for the algorithm +jppf.load.balancing.profile = proportional_profile + +# "manual" profile +jppf.load.balancing.profile.manual_profile.size = 1 + +# "autotuned" profile +jppf.load.balancing.profile.autotuned_profile.size = 5 +jppf.load.balancing.profile.autotuned_profile.minSamplesToAnalyse = 100 +jppf.load.balancing.profile.autotuned_profile.minSamplesToCheckConvergence = 50 +jppf.load.balancing.profile.autotuned_profile.maxDeviation = 0.2 +jppf.load.balancing.profile.autotuned_profile.maxGuessToStable = 50 +jppf.load.balancing.profile.autotuned_profile.sizeRatioDeviation = 1.5 +jppf.load.balancing.profile.autotuned_profile.decreaseRatio = 0.2 + +# "proportional" profile +jppf.load.balancing.profile.proportional_profile.size = 5 +jppf.load.balancing.profile.proportional_profile.initialMeanTime = 1e10 +jppf.load.balancing.profile.proportional_profile.performanceCacheSize = 300 +jppf.load.balancing.profile.proportional_profile.proportionalityFactor = 1 + +# "rl" profile +jppf.load.balancing.profile.rl_profile.performanceCacheSize = 1000 +jppf.load.balancing.profile.rl_profile.performanceVariationThreshold = 0.0001 +jppf.load.balancing.profile.rl_profile.maxActionRange = 10 + +# "nodethreads" profile +jppf.load.balancing.profile.nodethreads_profile.multiplicator = 1 + +#------------------------------------------------------------------------------# +# Other JVM options added to the java command line when the driver is started # +# as a subprocess. Multiple options are separated by spaces. # +#------------------------------------------------------------------------------# + +jppf.jvm.options = -server -XX:+HeapDumpOnOutOfMemoryError -Xms1000m -Xmx9000m + +# example with remote debugging options +#jppf.jvm.options = -server -Xmx256m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n + +#------------------------------------------------------------------------------# +# Specify alternate serialization schemes. # +# Defaults to org.jppf.serialization.DefaultJavaSerialization. # +#------------------------------------------------------------------------------# + +# default +#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization + +# built-in object serialization schemes +#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization +#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization + +# defined in the "Kryo Serialization" sample +#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization + +#------------------------------------------------------------------------------# +# Specify a data transformation class. If unspecified, no transformation occurs# +#------------------------------------------------------------------------------# + +# Defined in the "Network Data Encryption" sample +#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform + + +#------------------------------------------------------------------------------# +# whether to resolve the nodes' ip addresses into host names # +# defaults to true (resolve the addresses) # +#------------------------------------------------------------------------------# + +org.jppf.resolve.addresses = true + +#------------------------------------------------------------------------------# +# Local (in-JVM) node. When enabled, any node-specific properties will apply # +#------------------------------------------------------------------------------# + +# Enable/disable the local node. Default is false (disabled) +#jppf.local.node.enabled = false + +#------------------------------------------------------------------------------# +# In idle mode configuration. In this mode the server or node starts when no # +# mouse or keyboard activity has occurred since the specified timeout, and is # +# stopped when any new activity occurs. # +#------------------------------------------------------------------------------# + +# Idle mode enabled/disabled. Default is false (disabled) +#jppf.idle.mode.enabled = false + +# Fully qualified class name of the factory object that instantiates a platform-specific idle state detector +#jppf.idle.detector.factory = org.jppf.example.idlesystem.IdleTimeDetectorFactoryImpl + +# Time of keyboard and mouse inactivity to consider the system idle, in milliseconds +# Default value is 300000 (5 minutes) +#jppf.idle.timeout = 6000 + +# Interval between 2 successive calls to the native APIs to determine idle state changes +# Default value is 1000 +#jppf.idle.poll.interval = 1000 + +#------------------------------------------------------------------------------# +# Automatic recovery from hard failure of the nodes connections. These # +# parameters configure how the driver reacts when a node fails to respond to # +# its heartbeat messages. # +#------------------------------------------------------------------------------# + +# Enable recovery from failures on the nodes. Default to false (disabled) +#jppf.recovery.enabled = false + +# Max number of attempts to get a response from the node before the connection +# is considered broken. Default value is 3 +#jppf.recovery.max.retries = 3 + +# Max time in milliseconds allowed for each attempt to get a response from the node. +# Default value is 6000 (6 seconds) +#jppf.recovery.read.timeout = 6000 + +# Dedicated port number for the detection of node failure. Defaults to 22222. +# If server discovery is enabled on the nodes, this value will override the port number specified in the nodes +#jppf.recovery.server.port = 22222 + +# Interval in milliseconds between two runs of the connection reaper +# Default value is 60000 (1 minute) +#jppf.recovery.reaper.run.interval = 60000 + +# Number of threads allocated to the reaper. Default to the number of available CPUs +#jppf.recovery.reaper.pool.size = 8 + +#------------------------------------------------------------------------------# +# Redirecting System.out and System.err to files. # +#------------------------------------------------------------------------------# + +# file path on the file system where System.out is redirected. +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.out = System.out.log +# whether to append to an existing file or to create a new one +jppf.redirect.out.append = false + +# file path on the file system where System.err is redirected +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.err = System.err.log +# whether to append to an existing file or to create a new one +jppf.redirect.err.append = false + +#------------------------------------------------------------------------------# +# Global performance tuning parameters. These affect the performance and # +# throughput of I/O operations in JPPF. The values provided in the vanilla # +# JPPF distribution are known to offer a good performance in most situations # +# and environments. # +#------------------------------------------------------------------------------# + +# Size of send and receive buffer for socket connections. +# Defaults to 32768 and must be in range [1024, 1024*1024] +# 128 * 1024 = 131072 +jppf.socket.buffer.size = 131072 +# Size of temporary buffers (including direct buffers) used in I/O transfers. +# Defaults to 32768 and must be in range [1024, 1024*1024] +jppf.temp.buffer.size = 12288 +# Maximum size of temporary buffers pool (excluding direct buffers). When this size +# is reached, new buffers are still created, but not released into the pool, so they +# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} +# Defaults to 10 and must be in range [1, 2048] +jppf.temp.buffer.pool.size = 200 +# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). +# Defaults to 100 and must be in range [1, 2048] +jppf.length.buffer.pool.size = 100 + +#------------------------------------------------------------------------------# +# Enabling or disabling the lookup of classpath resources in the file system # +# Defaults to true (enabled) # +#------------------------------------------------------------------------------# + +#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag01.properties b/sandag_abm/src/main/resources/jppf-sandag01.properties new file mode 100644 index 0000000..1d5dce8 --- /dev/null +++ b/sandag_abm/src/main/resources/jppf-sandag01.properties @@ -0,0 +1,298 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2015 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +#------------------------------------------------------------------------------# +# Manual configuration of the server connection # +#------------------------------------------------------------------------------# + +# Host name, or ip address, of the host the JPPF driver is running on +# defaults to 'localhost' +jppf.server.host = ${master.node.name} + +# port number the server is listening to for connections, defaults to 11111 +#jppf.server.port = 11111 + +#------------------------------------------------------------------------------# +# Configuration of JMX management server # +#------------------------------------------------------------------------------# + +# enable/disable management, defaults to true +#jppf.management.enabled = true + +# JMX management host IP address. If unspecified (recommended), the first +# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine +# will be used. If no non-local IP is found, 'localhost' will be used. +#jppf.management.host = localhost + +# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port +# is already bound, the node will automatically scan for the next available port. +#jppf.node.management.port = 12001 + +#------------------------------------------------------------------------------# +# SSL Settings # +#------------------------------------------------------------------------------# + +# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established +#jppf.ssl.enabled = true + +# location of the SSL configuration on the file system +#jppf.ssl.configuration.file = config/ssl/ssl.properties + +# SSL configuration as an arbitrary source. Value is the fully qualified name of +# an implementation of java.util.concurrent.Callable with optional space-separated arguments +#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties + +#------------------------------------------------------------------------------# +# path to the JPPF security policy file # +# comment out this entry to disable security on the node # +#------------------------------------------------------------------------------# + +#jppf.policy.file = config/jppf.policy + +#------------------------------------------------------------------------------# +# Server discovery. # +#------------------------------------------------------------------------------# + +# Enable/disable automatic discovery of JPPF drivers, defaults to true. +jppf.discovery.enabled = false + +# UDP multicast group to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default to 230.0.0.1 +#jppf.discovery.group = 230.0.0.1 + +# UDP multicast port to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Defaults to 11111 +#jppf.discovery.port = 11111 + +# How long the node will attempt to automatically discover a driver before falling +# back to the parameters specified in this configuration file. Defaults to 5000 ms +#jppf.discovery.timeout = 5000 + +# IPv4 address patterns included in the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be included +# in the list of discovered drivers. +#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 + +# IPv4 address patterns excluded from the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be excluded +# from the list of discovered drivers. +#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 + +# IPv6 address patterns included in the server dscovery mechanism +#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 + +# IPv6 address patterns excluded from the server dscovery mechanism +#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 + +#------------------------------------------------------------------------------# +# Automatic recovery from disconnection from the server # +#------------------------------------------------------------------------------# + +# number of seconds before the first reconnection attempt, default to 1 +#jppf.reconnect.initial.delay = 1 + +# time after which the node stops trying to reconnect, in seconds. Defaults to 60 +jppf.reconnect.max.time = 5 + +# time between two connection attempts, in seconds. Defaults to 1 +#jppf.reconnect.interval = 1 + +#------------------------------------------------------------------------------# +# Processing Threads: number of threads running tasks in this node. # +# default value is the number of available CPUs; uncomment to specify a # +# different value. Blocking tasks might benefit from a number larger than CPUs # +#------------------------------------------------------------------------------# + +jppf.processing.threads = ${node.1.execution.threads} + +#------------------------------------------------------------------------------# +# Thread Manager: manager that wraps the executor service for running tasks. # +# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # +#------------------------------------------------------------------------------# + +# built-in thread manager +#jppf.thread.manager.class = default + +# fork/join thread manager +#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin + +#------------------------------------------------------------------------------# +# Specify alternate serialization schemes. # +# Defaults to org.jppf.serialization.DefaultJavaSerialization. # +#------------------------------------------------------------------------------# + +# default +#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization + +# built-in object serialization schemes +#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization +#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization + +# defined in the "Kryo Serialization" sample +#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization + +#------------------------------------------------------------------------------# +# Specify a data transformation class. If unspecified, none is used. # +#------------------------------------------------------------------------------# + +# Defined in the "Network Data Encryption" sample +#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform + +#------------------------------------------------------------------------------# +# Other JVM options added to the java command line when the node is started as # +# a subprocess. Multiple options are separated by spaces. # +#------------------------------------------------------------------------------# + +jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag01.xml -Dnode.name=sandag01 + +# example with remote debugging options +#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n + +#------------------------------------------------------------------------------# +# Idle mode configuration. In idle mode, the server ot node starts when no # +# mouse or keyboard activity has occurred since the specified tiemout, and is # +# stopped when any new activity occurs. See "jppf.idle.timeout" below. # +#------------------------------------------------------------------------------# + +# enable/disable idle mode, defaults to false (disabled) +#jppf.idle.mode.enabled = false + +# Time of keyboard and mouse inactivity after which the system is considered +# idle, in ms. Defaults to 300000 (5 minutes) +#jppf.idle.timeout = 6000 + +# Interval between 2 successive calls to the native APIs to determine whether +# the system idle state has changed. Defaults to 1000 ms. +#jppf.idle.poll.interval = 1000 + +# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, +# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). +#jppf.idle.interruptIfRunning = true + +#------------------------------------------------------------------------------# +# Automatic recovery from hard failure of the server connection. These # +# parameters configure how the node reacts to the heartbeats sent by the # +# server, or lack thereof. # +#------------------------------------------------------------------------------# + +# Enable recovery from hardware failures, defaults to false (disabled) +#jppf.recovery.enabled = true + +# Dedicated port number for the detection of connection failure, must be the +# same as the value specified in the server configuration. Defaults to 22222. +# If server discovery is enabled, this value will be overridden by the port +# number specified in the driver configuration. +#jppf.recovery.server.port = 22222 + +# Maximum number of attempts to get a message from the server before the +# connection is considered broken. Default value is 2 +#jppf.recovery.max.retries = 2 + +# Maximum time in milliseconds allowed for each attempt to get a response from +# the node. Default value is 60000 ms (1 minute). + +#jppf.recovery.read.timeout = 60000 + +#------------------------------------------------------------------------------# +# JPPF class loader-related properties # +#------------------------------------------------------------------------------# + +# JPPF class loader delegation model. values: parent | url, defaults to parent +#jppf.classloader.delegation = parent + +# size of the class loader cache in the node, defaults to 50 +#jppf.classloader.cache.size = 50 + +# class loader resource cache enabled? defaults to true. +#jppf.resource.cache.enabled = true +# resource cache's type of storage: either "file" (the default) or "memory" +#jppf.resource.cache.storage = file +# root location of the file-persisted caches +#jppf.resource.cache.dir = some_directory + +#------------------------------------------------------------------------------# +# Screen saver settings # +#------------------------------------------------------------------------------# + +# include the settings from a separate file to avoid cluttering this config file +#!include file config/screensaver.properties + +#------------------------------------------------------------------------------# +# Redirecting System.out and System.err to files. # +#------------------------------------------------------------------------------# + +# file path on the file system where System.out is redirected. +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.out = System.out.log +# whether to append to an existing file or to create a new one +#jppf.redirect.out.append = false + +# file path on the file system where System.err is redirected +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.err = System.err.log +# whether to append to an existing file or to create a new one +#jppf.redirect.err.append = false + +#------------------------------------------------------------------------------# +# Node provisioning configuration # +#------------------------------------------------------------------------------# + +# Define a node as master. Defaults to true +#jppf.node.provisioning.master = true +# Define a node as a slave. Defaults to false +#jppf.node.provisioning.slave = false +# Specify the path prefix used for the root directory of each slave node +# defaults to "slave_nodes/node_", relative to the master root directory +#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ +# Specify the directory where slave-specific configuration files are located +# Defaults to the "config" folder, relative to the master root directory +#jppf.node.provisioning.slave.config.path = config +# A set of space-separated JVM options always added to the slave startup command +#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties +# Specify the number of slaves to launch upon master node startup. Defaults to 0 +#jppf.node.provisioning.startup.slaves = 0 + +#------------------------------------------------------------------------------# +# Global performance tuning parameters. These affect the performance and # +# throughput of I/O operations in JPPF. The values provided in the vanilla # +# JPPF distribution are known to offer a good performance in most situations # +# and environments. # +#------------------------------------------------------------------------------# + +# Size of send and receive buffer for socket connections. +# Defaults to 32768 and must be in range [1024, 1024*1024] +# 128 * 1024 = 131072 +#jppf.socket.buffer.size = 131072 +# Size of temporary buffers (including direct buffers) used in I/O transfers. +# Defaults to 32768 and must be in range [1024, 1024*1024] +#jppf.temp.buffer.size = 12288 +# Maximum size of temporary buffers pool (excluding direct buffers). When this size +# is reached, new buffers are still created, but not released into the pool, so they +# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} +# Defaults to 10 and must be in range [1, 2048] +#jppf.temp.buffer.pool.size = 200 +# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). +# Defaults to 100 and must be in range [1, 2048] +#jppf.length.buffer.pool.size = 100 + +#------------------------------------------------------------------------------# +# Enabling or disabling the lookup of classpath resources in the file system # +# Defaults to true (enabled) # +#------------------------------------------------------------------------------# + +#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag02.properties b/sandag_abm/src/main/resources/jppf-sandag02.properties new file mode 100644 index 0000000..290be6d --- /dev/null +++ b/sandag_abm/src/main/resources/jppf-sandag02.properties @@ -0,0 +1,298 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2015 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +#------------------------------------------------------------------------------# +# Manual configuration of the server connection # +#------------------------------------------------------------------------------# + +# Host name, or ip address, of the host the JPPF driver is running on +# defaults to 'localhost' +jppf.server.host = ${master.node.name} + +# port number the server is listening to for connections, defaults to 11111 +#jppf.server.port = 11111 + +#------------------------------------------------------------------------------# +# Configuration of JMX management server # +#------------------------------------------------------------------------------# + +# enable/disable management, defaults to true +#jppf.management.enabled = true + +# JMX management host IP address. If unspecified (recommended), the first +# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine +# will be used. If no non-local IP is found, 'localhost' will be used. +#jppf.management.host = localhost + +# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port +# is already bound, the node will automatically scan for the next available port. +#jppf.node.management.port = 12001 + +#------------------------------------------------------------------------------# +# SSL Settings # +#------------------------------------------------------------------------------# + +# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established +#jppf.ssl.enabled = true + +# location of the SSL configuration on the file system +#jppf.ssl.configuration.file = config/ssl/ssl.properties + +# SSL configuration as an arbitrary source. Value is the fully qualified name of +# an implementation of java.util.concurrent.Callable with optional space-separated arguments +#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties + +#------------------------------------------------------------------------------# +# path to the JPPF security policy file # +# comment out this entry to disable security on the node # +#------------------------------------------------------------------------------# + +#jppf.policy.file = config/jppf.policy + +#------------------------------------------------------------------------------# +# Server discovery. # +#------------------------------------------------------------------------------# + +# Enable/disable automatic discovery of JPPF drivers, defaults to true. +jppf.discovery.enabled = false + +# UDP multicast group to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default to 230.0.0.1 +#jppf.discovery.group = 230.0.0.1 + +# UDP multicast port to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Defaults to 11111 +#jppf.discovery.port = 11111 + +# How long the node will attempt to automatically discover a driver before falling +# back to the parameters specified in this configuration file. Defaults to 5000 ms +#jppf.discovery.timeout = 5000 + +# IPv4 address patterns included in the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be included +# in the list of discovered drivers. +#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 + +# IPv4 address patterns excluded from the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be excluded +# from the list of discovered drivers. +#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 + +# IPv6 address patterns included in the server dscovery mechanism +#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 + +# IPv6 address patterns excluded from the server dscovery mechanism +#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 + +#------------------------------------------------------------------------------# +# Automatic recovery from disconnection from the server # +#------------------------------------------------------------------------------# + +# number of seconds before the first reconnection attempt, default to 1 +#jppf.reconnect.initial.delay = 1 + +# time after which the node stops trying to reconnect, in seconds. Defaults to 60 +jppf.reconnect.max.time = 5 + +# time between two connection attempts, in seconds. Defaults to 1 +#jppf.reconnect.interval = 1 + +#------------------------------------------------------------------------------# +# Processing Threads: number of threads running tasks in this node. # +# default value is the number of available CPUs; uncomment to specify a # +# different value. Blocking tasks might benefit from a number larger than CPUs # +#------------------------------------------------------------------------------# + +jppf.processing.threads = ${node.1.execution.threads} + +#------------------------------------------------------------------------------# +# Thread Manager: manager that wraps the executor service for running tasks. # +# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # +#------------------------------------------------------------------------------# + +# built-in thread manager +#jppf.thread.manager.class = default + +# fork/join thread manager +#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin + +#------------------------------------------------------------------------------# +# Specify alternate serialization schemes. # +# Defaults to org.jppf.serialization.DefaultJavaSerialization. # +#------------------------------------------------------------------------------# + +# default +#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization + +# built-in object serialization schemes +#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization +#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization + +# defined in the "Kryo Serialization" sample +#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization + +#------------------------------------------------------------------------------# +# Specify a data transformation class. If unspecified, none is used. # +#------------------------------------------------------------------------------# + +# Defined in the "Network Data Encryption" sample +#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform + +#------------------------------------------------------------------------------# +# Other JVM options added to the java command line when the node is started as # +# a subprocess. Multiple options are separated by spaces. # +#------------------------------------------------------------------------------# + +jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag02.xml -Dnode.name=sandag02 + +# example with remote debugging options +#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n + +#------------------------------------------------------------------------------# +# Idle mode configuration. In idle mode, the server ot node starts when no # +# mouse or keyboard activity has occurred since the specified tiemout, and is # +# stopped when any new activity occurs. See "jppf.idle.timeout" below. # +#------------------------------------------------------------------------------# + +# enable/disable idle mode, defaults to false (disabled) +#jppf.idle.mode.enabled = false + +# Time of keyboard and mouse inactivity after which the system is considered +# idle, in ms. Defaults to 300000 (5 minutes) +#jppf.idle.timeout = 6000 + +# Interval between 2 successive calls to the native APIs to determine whether +# the system idle state has changed. Defaults to 1000 ms. +#jppf.idle.poll.interval = 1000 + +# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, +# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). +#jppf.idle.interruptIfRunning = true + +#------------------------------------------------------------------------------# +# Automatic recovery from hard failure of the server connection. These # +# parameters configure how the node reacts to the heartbeats sent by the # +# server, or lack thereof. # +#------------------------------------------------------------------------------# + +# Enable recovery from hardware failures, defaults to false (disabled) +#jppf.recovery.enabled = true + +# Dedicated port number for the detection of connection failure, must be the +# same as the value specified in the server configuration. Defaults to 22222. +# If server discovery is enabled, this value will be overridden by the port +# number specified in the driver configuration. +#jppf.recovery.server.port = 22222 + +# Maximum number of attempts to get a message from the server before the +# connection is considered broken. Default value is 2 +#jppf.recovery.max.retries = 2 + +# Maximum time in milliseconds allowed for each attempt to get a response from +# the node. Default value is 60000 ms (1 minute). + +#jppf.recovery.read.timeout = 60000 + +#------------------------------------------------------------------------------# +# JPPF class loader-related properties # +#------------------------------------------------------------------------------# + +# JPPF class loader delegation model. values: parent | url, defaults to parent +#jppf.classloader.delegation = parent + +# size of the class loader cache in the node, defaults to 50 +#jppf.classloader.cache.size = 50 + +# class loader resource cache enabled? defaults to true. +#jppf.resource.cache.enabled = true +# resource cache's type of storage: either "file" (the default) or "memory" +#jppf.resource.cache.storage = file +# root location of the file-persisted caches +#jppf.resource.cache.dir = some_directory + +#------------------------------------------------------------------------------# +# Screen saver settings # +#------------------------------------------------------------------------------# + +# include the settings from a separate file to avoid cluttering this config file +#!include file config/screensaver.properties + +#------------------------------------------------------------------------------# +# Redirecting System.out and System.err to files. # +#------------------------------------------------------------------------------# + +# file path on the file system where System.out is redirected. +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.out = System.out.log +# whether to append to an existing file or to create a new one +#jppf.redirect.out.append = false + +# file path on the file system where System.err is redirected +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.err = System.err.log +# whether to append to an existing file or to create a new one +#jppf.redirect.err.append = false + +#------------------------------------------------------------------------------# +# Node provisioning configuration # +#------------------------------------------------------------------------------# + +# Define a node as master. Defaults to true +#jppf.node.provisioning.master = true +# Define a node as a slave. Defaults to false +#jppf.node.provisioning.slave = false +# Specify the path prefix used for the root directory of each slave node +# defaults to "slave_nodes/node_", relative to the master root directory +#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ +# Specify the directory where slave-specific configuration files are located +# Defaults to the "config" folder, relative to the master root directory +#jppf.node.provisioning.slave.config.path = config +# A set of space-separated JVM options always added to the slave startup command +#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties +# Specify the number of slaves to launch upon master node startup. Defaults to 0 +#jppf.node.provisioning.startup.slaves = 0 + +#------------------------------------------------------------------------------# +# Global performance tuning parameters. These affect the performance and # +# throughput of I/O operations in JPPF. The values provided in the vanilla # +# JPPF distribution are known to offer a good performance in most situations # +# and environments. # +#------------------------------------------------------------------------------# + +# Size of send and receive buffer for socket connections. +# Defaults to 32768 and must be in range [1024, 1024*1024] +# 128 * 1024 = 131072 +#jppf.socket.buffer.size = 131072 +# Size of temporary buffers (including direct buffers) used in I/O transfers. +# Defaults to 32768 and must be in range [1024, 1024*1024] +#jppf.temp.buffer.size = 12288 +# Maximum size of temporary buffers pool (excluding direct buffers). When this size +# is reached, new buffers are still created, but not released into the pool, so they +# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} +# Defaults to 10 and must be in range [1, 2048] +#jppf.temp.buffer.pool.size = 200 +# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). +# Defaults to 100 and must be in range [1, 2048] +#jppf.length.buffer.pool.size = 100 + +#------------------------------------------------------------------------------# +# Enabling or disabling the lookup of classpath resources in the file system # +# Defaults to true (enabled) # +#------------------------------------------------------------------------------# + +#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag03.properties b/sandag_abm/src/main/resources/jppf-sandag03.properties new file mode 100644 index 0000000..ab3a6a2 --- /dev/null +++ b/sandag_abm/src/main/resources/jppf-sandag03.properties @@ -0,0 +1,298 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2015 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +#------------------------------------------------------------------------------# +# Manual configuration of the server connection # +#------------------------------------------------------------------------------# + +# Host name, or ip address, of the host the JPPF driver is running on +# defaults to 'localhost' +jppf.server.host = ${master.node.name} + +# port number the server is listening to for connections, defaults to 11111 +#jppf.server.port = 11111 + +#------------------------------------------------------------------------------# +# Configuration of JMX management server # +#------------------------------------------------------------------------------# + +# enable/disable management, defaults to true +#jppf.management.enabled = true + +# JMX management host IP address. If unspecified (recommended), the first +# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine +# will be used. If no non-local IP is found, 'localhost' will be used. +#jppf.management.host = localhost + +# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port +# is already bound, the node will automatically scan for the next available port. +#jppf.node.management.port = 12001 + +#------------------------------------------------------------------------------# +# SSL Settings # +#------------------------------------------------------------------------------# + +# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established +#jppf.ssl.enabled = true + +# location of the SSL configuration on the file system +#jppf.ssl.configuration.file = config/ssl/ssl.properties + +# SSL configuration as an arbitrary source. Value is the fully qualified name of +# an implementation of java.util.concurrent.Callable with optional space-separated arguments +#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties + +#------------------------------------------------------------------------------# +# path to the JPPF security policy file # +# comment out this entry to disable security on the node # +#------------------------------------------------------------------------------# + +#jppf.policy.file = config/jppf.policy + +#------------------------------------------------------------------------------# +# Server discovery. # +#------------------------------------------------------------------------------# + +# Enable/disable automatic discovery of JPPF drivers, defaults to true. +jppf.discovery.enabled = false + +# UDP multicast group to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default to 230.0.0.1 +#jppf.discovery.group = 230.0.0.1 + +# UDP multicast port to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Defaults to 11111 +#jppf.discovery.port = 11111 + +# How long the node will attempt to automatically discover a driver before falling +# back to the parameters specified in this configuration file. Defaults to 5000 ms +#jppf.discovery.timeout = 5000 + +# IPv4 address patterns included in the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be included +# in the list of discovered drivers. +#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 + +# IPv4 address patterns excluded from the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be excluded +# from the list of discovered drivers. +#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 + +# IPv6 address patterns included in the server dscovery mechanism +#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 + +# IPv6 address patterns excluded from the server dscovery mechanism +#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 + +#------------------------------------------------------------------------------# +# Automatic recovery from disconnection from the server # +#------------------------------------------------------------------------------# + +# number of seconds before the first reconnection attempt, default to 1 +#jppf.reconnect.initial.delay = 1 + +# time after which the node stops trying to reconnect, in seconds. Defaults to 60 +jppf.reconnect.max.time = 5 + +# time between two connection attempts, in seconds. Defaults to 1 +#jppf.reconnect.interval = 1 + +#------------------------------------------------------------------------------# +# Processing Threads: number of threads running tasks in this node. # +# default value is the number of available CPUs; uncomment to specify a # +# different value. Blocking tasks might benefit from a number larger than CPUs # +#------------------------------------------------------------------------------# + +jppf.processing.threads = ${node.2.execution.threads} + +#------------------------------------------------------------------------------# +# Thread Manager: manager that wraps the executor service for running tasks. # +# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # +#------------------------------------------------------------------------------# + +# built-in thread manager +#jppf.thread.manager.class = default + +# fork/join thread manager +#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin + +#------------------------------------------------------------------------------# +# Specify alternate serialization schemes. # +# Defaults to org.jppf.serialization.DefaultJavaSerialization. # +#------------------------------------------------------------------------------# + +# default +#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization + +# built-in object serialization schemes +#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization +#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization + +# defined in the "Kryo Serialization" sample +#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization + +#------------------------------------------------------------------------------# +# Specify a data transformation class. If unspecified, none is used. # +#------------------------------------------------------------------------------# + +# Defined in the "Network Data Encryption" sample +#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform + +#------------------------------------------------------------------------------# +# Other JVM options added to the java command line when the node is started as # +# a subprocess. Multiple options are separated by spaces. # +#------------------------------------------------------------------------------# + +jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag03.xml -Dnode.name=sandag03 + +# example with remote debugging options +#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n + +#------------------------------------------------------------------------------# +# Idle mode configuration. In idle mode, the server ot node starts when no # +# mouse or keyboard activity has occurred since the specified tiemout, and is # +# stopped when any new activity occurs. See "jppf.idle.timeout" below. # +#------------------------------------------------------------------------------# + +# enable/disable idle mode, defaults to false (disabled) +#jppf.idle.mode.enabled = false + +# Time of keyboard and mouse inactivity after which the system is considered +# idle, in ms. Defaults to 300000 (5 minutes) +#jppf.idle.timeout = 6000 + +# Interval between 2 successive calls to the native APIs to determine whether +# the system idle state has changed. Defaults to 1000 ms. +#jppf.idle.poll.interval = 1000 + +# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, +# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). +#jppf.idle.interruptIfRunning = true + +#------------------------------------------------------------------------------# +# Automatic recovery from hard failure of the server connection. These # +# parameters configure how the node reacts to the heartbeats sent by the # +# server, or lack thereof. # +#------------------------------------------------------------------------------# + +# Enable recovery from hardware failures, defaults to false (disabled) +#jppf.recovery.enabled = true + +# Dedicated port number for the detection of connection failure, must be the +# same as the value specified in the server configuration. Defaults to 22222. +# If server discovery is enabled, this value will be overridden by the port +# number specified in the driver configuration. +#jppf.recovery.server.port = 22222 + +# Maximum number of attempts to get a message from the server before the +# connection is considered broken. Default value is 2 +#jppf.recovery.max.retries = 2 + +# Maximum time in milliseconds allowed for each attempt to get a response from +# the node. Default value is 60000 ms (1 minute). + +#jppf.recovery.read.timeout = 60000 + +#------------------------------------------------------------------------------# +# JPPF class loader-related properties # +#------------------------------------------------------------------------------# + +# JPPF class loader delegation model. values: parent | url, defaults to parent +#jppf.classloader.delegation = parent + +# size of the class loader cache in the node, defaults to 50 +#jppf.classloader.cache.size = 50 + +# class loader resource cache enabled? defaults to true. +#jppf.resource.cache.enabled = true +# resource cache's type of storage: either "file" (the default) or "memory" +#jppf.resource.cache.storage = file +# root location of the file-persisted caches +#jppf.resource.cache.dir = some_directory + +#------------------------------------------------------------------------------# +# Screen saver settings # +#------------------------------------------------------------------------------# + +# include the settings from a separate file to avoid cluttering this config file +#!include file config/screensaver.properties + +#------------------------------------------------------------------------------# +# Redirecting System.out and System.err to files. # +#------------------------------------------------------------------------------# + +# file path on the file system where System.out is redirected. +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.out = System.out.log +# whether to append to an existing file or to create a new one +#jppf.redirect.out.append = false + +# file path on the file system where System.err is redirected +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.err = System.err.log +# whether to append to an existing file or to create a new one +#jppf.redirect.err.append = false + +#------------------------------------------------------------------------------# +# Node provisioning configuration # +#------------------------------------------------------------------------------# + +# Define a node as master. Defaults to true +#jppf.node.provisioning.master = true +# Define a node as a slave. Defaults to false +#jppf.node.provisioning.slave = false +# Specify the path prefix used for the root directory of each slave node +# defaults to "slave_nodes/node_", relative to the master root directory +#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ +# Specify the directory where slave-specific configuration files are located +# Defaults to the "config" folder, relative to the master root directory +#jppf.node.provisioning.slave.config.path = config +# A set of space-separated JVM options always added to the slave startup command +#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties +# Specify the number of slaves to launch upon master node startup. Defaults to 0 +#jppf.node.provisioning.startup.slaves = 0 + +#------------------------------------------------------------------------------# +# Global performance tuning parameters. These affect the performance and # +# throughput of I/O operations in JPPF. The values provided in the vanilla # +# JPPF distribution are known to offer a good performance in most situations # +# and environments. # +#------------------------------------------------------------------------------# + +# Size of send and receive buffer for socket connections. +# Defaults to 32768 and must be in range [1024, 1024*1024] +# 128 * 1024 = 131072 +#jppf.socket.buffer.size = 131072 +# Size of temporary buffers (including direct buffers) used in I/O transfers. +# Defaults to 32768 and must be in range [1024, 1024*1024] +#jppf.temp.buffer.size = 12288 +# Maximum size of temporary buffers pool (excluding direct buffers). When this size +# is reached, new buffers are still created, but not released into the pool, so they +# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} +# Defaults to 10 and must be in range [1, 2048] +#jppf.temp.buffer.pool.size = 200 +# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). +# Defaults to 100 and must be in range [1, 2048] +#jppf.length.buffer.pool.size = 100 + +#------------------------------------------------------------------------------# +# Enabling or disabling the lookup of classpath resources in the file system # +# Defaults to true (enabled) # +#------------------------------------------------------------------------------# + +#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/jppf-sandag04.properties b/sandag_abm/src/main/resources/jppf-sandag04.properties new file mode 100644 index 0000000..61f887c --- /dev/null +++ b/sandag_abm/src/main/resources/jppf-sandag04.properties @@ -0,0 +1,298 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2015 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +#------------------------------------------------------------------------------# +# Manual configuration of the server connection # +#------------------------------------------------------------------------------# + +# Host name, or ip address, of the host the JPPF driver is running on +# defaults to 'localhost' +jppf.server.host = ${master.node.name} + +# port number the server is listening to for connections, defaults to 11111 +#jppf.server.port = 11111 + +#------------------------------------------------------------------------------# +# Configuration of JMX management server # +#------------------------------------------------------------------------------# + +# enable/disable management, defaults to true +#jppf.management.enabled = true + +# JMX management host IP address. If unspecified (recommended), the first +# non-local IP address (i.e. neither 127.0.0.1 nor localhost) on this machine +# will be used. If no non-local IP is found, 'localhost' will be used. +#jppf.management.host = localhost + +# JMX management port, defaults to 11198 (no SSL) or 11193 with SSL. If the port +# is already bound, the node will automatically scan for the next available port. +#jppf.node.management.port = 12001 + +#------------------------------------------------------------------------------# +# SSL Settings # +#------------------------------------------------------------------------------# + +# Enable SSL, defaults to false (disabled). If enabled, only SSL connections are established +#jppf.ssl.enabled = true + +# location of the SSL configuration on the file system +#jppf.ssl.configuration.file = config/ssl/ssl.properties + +# SSL configuration as an arbitrary source. Value is the fully qualified name of +# an implementation of java.util.concurrent.Callable with optional space-separated arguments +#jppf.ssl.configuration.source = org.jppf.ssl.FileStoreSource config/ssl/ssl.properties + +#------------------------------------------------------------------------------# +# path to the JPPF security policy file # +# comment out this entry to disable security on the node # +#------------------------------------------------------------------------------# + +#jppf.policy.file = config/jppf.policy + +#------------------------------------------------------------------------------# +# Server discovery. # +#------------------------------------------------------------------------------# + +# Enable/disable automatic discovery of JPPF drivers, defaults to true. +jppf.discovery.enabled = false + +# UDP multicast group to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Default to 230.0.0.1 +#jppf.discovery.group = 230.0.0.1 + +# UDP multicast port to which drivers broadcast their connection parameters +# and to which clients and nodes listen. Defaults to 11111 +#jppf.discovery.port = 11111 + +# How long the node will attempt to automatically discover a driver before falling +# back to the parameters specified in this configuration file. Defaults to 5000 ms +#jppf.discovery.timeout = 5000 + +# IPv4 address patterns included in the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be included +# in the list of discovered drivers. +#jppf.discovery.include.ipv4 = 192.168.1.; 192.168.1.0/24 + +# IPv4 address patterns excluded from the server dscovery mechanism +# Drivers whose IPv4 address matches the pattern will be excluded +# from the list of discovered drivers. +#jppf.discovery.exclude.ipv4 = 192.168.1.128-; 192.168.1.0/25 + +# IPv6 address patterns included in the server dscovery mechanism +#jppf.discovery.include.ipv6 = 1080:0:0:0:8:800:200C-20FF:-; ::1/80 + +# IPv6 address patterns excluded from the server dscovery mechanism +#jppf.discovery.exclude.ipv6 = 1080:0:0:0:8:800:200C-20FF:0C00-0EFF; ::1/96 + +#------------------------------------------------------------------------------# +# Automatic recovery from disconnection from the server # +#------------------------------------------------------------------------------# + +# number of seconds before the first reconnection attempt, default to 1 +#jppf.reconnect.initial.delay = 1 + +# time after which the node stops trying to reconnect, in seconds. Defaults to 60 +jppf.reconnect.max.time = 5 + +# time between two connection attempts, in seconds. Defaults to 1 +#jppf.reconnect.interval = 1 + +#------------------------------------------------------------------------------# +# Processing Threads: number of threads running tasks in this node. # +# default value is the number of available CPUs; uncomment to specify a # +# different value. Blocking tasks might benefit from a number larger than CPUs # +#------------------------------------------------------------------------------# + +jppf.processing.threads = ${node.3.execution.threads} + +#------------------------------------------------------------------------------# +# Thread Manager: manager that wraps the executor service for running tasks. # +# default value is ThreadManagerThreadPool - that wraps ThreadPoolExecutor # +#------------------------------------------------------------------------------# + +# built-in thread manager +#jppf.thread.manager.class = default + +# fork/join thread manager +#jppf.thread.manager.class = org.jppf.server.node.fj.ThreadManagerForkJoin + +#------------------------------------------------------------------------------# +# Specify alternate serialization schemes. # +# Defaults to org.jppf.serialization.DefaultJavaSerialization. # +#------------------------------------------------------------------------------# + +# default +#jppf.object.serialization.class = org.jppf.serialization.DefaultJavaSerialization + +# built-in object serialization schemes +#jppf.object.serialization.class = org.jppf.serialization.DefaultJPPFSerialization +#jppf.object.serialization.class = org.jppf.serialization.XstreamSerialization + +# defined in the "Kryo Serialization" sample +#jppf.object.serialization.class = org.jppf.serialization.kryo.KryoSerialization + +#------------------------------------------------------------------------------# +# Specify a data transformation class. If unspecified, none is used. # +#------------------------------------------------------------------------------# + +# Defined in the "Network Data Encryption" sample +#jppf.data.transform.class = org.jppf.example.dataencryption.SecureKeyCipherTransform + +#------------------------------------------------------------------------------# +# Other JVM options added to the java command line when the node is started as # +# a subprocess. Multiple options are separated by spaces. # +#------------------------------------------------------------------------------# + +jppf.jvm.options = -Xms45000m -Xmx120000m -Dlog4j.configuration=log4j-sandag04.xml -Dnode.name=sandag04 + +# example with remote debugging options +#jppf.jvm.options = -server -Xmx128m -Xrunjdwp:transport=dt_socket,address=localhost:8000,server=y,suspend=n + +#------------------------------------------------------------------------------# +# Idle mode configuration. In idle mode, the server ot node starts when no # +# mouse or keyboard activity has occurred since the specified tiemout, and is # +# stopped when any new activity occurs. See "jppf.idle.timeout" below. # +#------------------------------------------------------------------------------# + +# enable/disable idle mode, defaults to false (disabled) +#jppf.idle.mode.enabled = false + +# Time of keyboard and mouse inactivity after which the system is considered +# idle, in ms. Defaults to 300000 (5 minutes) +#jppf.idle.timeout = 6000 + +# Interval between 2 successive calls to the native APIs to determine whether +# the system idle state has changed. Defaults to 1000 ms. +#jppf.idle.poll.interval = 1000 + +# Whether to shutdown the node immediately when a mouse/keyboard activity is detected, +# or wait until the node is no longer executing tasks. Defaults to true (immediate shutdown). +#jppf.idle.interruptIfRunning = true + +#------------------------------------------------------------------------------# +# Automatic recovery from hard failure of the server connection. These # +# parameters configure how the node reacts to the heartbeats sent by the # +# server, or lack thereof. # +#------------------------------------------------------------------------------# + +# Enable recovery from hardware failures, defaults to false (disabled) +#jppf.recovery.enabled = true + +# Dedicated port number for the detection of connection failure, must be the +# same as the value specified in the server configuration. Defaults to 22222. +# If server discovery is enabled, this value will be overridden by the port +# number specified in the driver configuration. +#jppf.recovery.server.port = 22222 + +# Maximum number of attempts to get a message from the server before the +# connection is considered broken. Default value is 2 +#jppf.recovery.max.retries = 2 + +# Maximum time in milliseconds allowed for each attempt to get a response from +# the node. Default value is 60000 ms (1 minute). + +#jppf.recovery.read.timeout = 60000 + +#------------------------------------------------------------------------------# +# JPPF class loader-related properties # +#------------------------------------------------------------------------------# + +# JPPF class loader delegation model. values: parent | url, defaults to parent +#jppf.classloader.delegation = parent + +# size of the class loader cache in the node, defaults to 50 +#jppf.classloader.cache.size = 50 + +# class loader resource cache enabled? defaults to true. +#jppf.resource.cache.enabled = true +# resource cache's type of storage: either "file" (the default) or "memory" +#jppf.resource.cache.storage = file +# root location of the file-persisted caches +#jppf.resource.cache.dir = some_directory + +#------------------------------------------------------------------------------# +# Screen saver settings # +#------------------------------------------------------------------------------# + +# include the settings from a separate file to avoid cluttering this config file +#!include file config/screensaver.properties + +#------------------------------------------------------------------------------# +# Redirecting System.out and System.err to files. # +#------------------------------------------------------------------------------# + +# file path on the file system where System.out is redirected. +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.out = System.out.log +# whether to append to an existing file or to create a new one +#jppf.redirect.out.append = false + +# file path on the file system where System.err is redirected +# if unspecified or invalid, then no redirection occurs +#jppf.redirect.err = System.err.log +# whether to append to an existing file or to create a new one +#jppf.redirect.err.append = false + +#------------------------------------------------------------------------------# +# Node provisioning configuration # +#------------------------------------------------------------------------------# + +# Define a node as master. Defaults to true +#jppf.node.provisioning.master = true +# Define a node as a slave. Defaults to false +#jppf.node.provisioning.slave = false +# Specify the path prefix used for the root directory of each slave node +# defaults to "slave_nodes/node_", relative to the master root directory +#jppf.node.provisioning.slave.path.prefix = slave_nodes/node_ +# Specify the directory where slave-specific configuration files are located +# Defaults to the "config" folder, relative to the master root directory +#jppf.node.provisioning.slave.config.path = config +# A set of space-separated JVM options always added to the slave startup command +#jppf.node.provisioning.slave.jvm.options = -Dlog4j.configuration=config/log4j-node.properties +# Specify the number of slaves to launch upon master node startup. Defaults to 0 +#jppf.node.provisioning.startup.slaves = 0 + +#------------------------------------------------------------------------------# +# Global performance tuning parameters. These affect the performance and # +# throughput of I/O operations in JPPF. The values provided in the vanilla # +# JPPF distribution are known to offer a good performance in most situations # +# and environments. # +#------------------------------------------------------------------------------# + +# Size of send and receive buffer for socket connections. +# Defaults to 32768 and must be in range [1024, 1024*1024] +# 128 * 1024 = 131072 +#jppf.socket.buffer.size = 131072 +# Size of temporary buffers (including direct buffers) used in I/O transfers. +# Defaults to 32768 and must be in range [1024, 1024*1024] +#jppf.temp.buffer.size = 12288 +# Maximum size of temporary buffers pool (excluding direct buffers). When this size +# is reached, new buffers are still created, but not released into the pool, so they +# can be quickly garbage-collected. The size of each buffer is defined with ${jppf.temp.buffer.size} +# Defaults to 10 and must be in range [1, 2048] +#jppf.temp.buffer.pool.size = 200 +# Size of temporary buffer pool for reading lengths as ints (size of each buffer is 4). +# Defaults to 100 and must be in range [1, 2048] +#jppf.length.buffer.pool.size = 100 + +#------------------------------------------------------------------------------# +# Enabling or disabling the lookup of classpath resources in the file system # +# Defaults to true (enabled) # +#------------------------------------------------------------------------------# + +#jppf.classloader.file.lookup = true diff --git a/sandag_abm/src/main/resources/log4j-client.properties b/sandag_abm/src/main/resources/log4j-client.properties new file mode 100644 index 0000000..4f4307a --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-client.properties @@ -0,0 +1,39 @@ +#------------------------------------------------------------------------------# +# Java Parallel Processing Framework. # +# Copyright (C) 2005-2008 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-client.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=logfiles/jppf-client.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +log4j.rootLogger=INFO, JPPF +#log4j.rootLogger=DEBUG, JPPF + +#log4j.logger.org.jppf.client=DEBUG +#log4j.logger.org.jppf.common.socket=DEBUG diff --git a/sandag_abm/src/main/resources/log4j-driver.properties b/sandag_abm/src/main/resources/log4j-driver.properties new file mode 100644 index 0000000..dbf6fa8 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-driver.properties @@ -0,0 +1,42 @@ +#------------------------------------------------------------------------------# +# Java Parallel Processing Framework. # +# Copyright (C) 2005-2008 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-driver.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=./logFiles/jppf-driver.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +#log4j.logger.org.jppf.server.nio.StateTransitionTask=DEBUG + +# log information about interactions between the client and server +#log4j.logger.org.jppf.server.app=DEBUG +#log4j.logger.org.jppf.io.IOHelper=DEBUG + +log4j.rootLogger=INFO, JPPF +#log4j.rootLogger=DEBUG, JPPF diff --git a/sandag_abm/src/main/resources/log4j-sandag01.properties b/sandag_abm/src/main/resources/log4j-sandag01.properties new file mode 100644 index 0000000..01af856 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag01.properties @@ -0,0 +1,46 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2010 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-node.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=logFiles/event-sandag01.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### direct messages to the JMX Logger ### +log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender +log4j.appender.JMX.layout=org.apache.log4j.PatternLayout +log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +# will produce messages like "writing object size = " when sending to the server +log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE +# will produce messages like "i = , read data size = " when receiving from the server +log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE + +#log4j.rootLogger=TRACE, DEBUG, JPPF +log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag01.xml b/sandag_abm/src/main/resources/log4j-sandag01.xml new file mode 100644 index 0000000..4f5ed1d --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag01.xml @@ -0,0 +1,439 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j-sandag02.properties b/sandag_abm/src/main/resources/log4j-sandag02.properties new file mode 100644 index 0000000..9aae59a --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag02.properties @@ -0,0 +1,46 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2010 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-node.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=logFiles/event-sandag02.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### direct messages to the JMX Logger ### +log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender +log4j.appender.JMX.layout=org.apache.log4j.PatternLayout +log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +# will produce messages like "writing object size = " when sending to the server +log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE +# will produce messages like "i = , read data size = " when receiving from the server +log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE + +#log4j.rootLogger=TRACE, DEBUG, JPPF +log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag02.xml b/sandag_abm/src/main/resources/log4j-sandag02.xml new file mode 100644 index 0000000..5629089 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag02.xml @@ -0,0 +1,439 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j-sandag03.properties b/sandag_abm/src/main/resources/log4j-sandag03.properties new file mode 100644 index 0000000..771824a --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag03.properties @@ -0,0 +1,46 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2010 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-node.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=logFiles/event-sandag03.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### direct messages to the JMX Logger ### +log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender +log4j.appender.JMX.layout=org.apache.log4j.PatternLayout +log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +# will produce messages like "writing object size = " when sending to the server +log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE +# will produce messages like "i = , read data size = " when receiving from the server +log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE + +#log4j.rootLogger=TRACE, DEBUG, JPPF +log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag03.xml b/sandag_abm/src/main/resources/log4j-sandag03.xml new file mode 100644 index 0000000..2377353 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag03.xml @@ -0,0 +1,439 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j-sandag04.properties b/sandag_abm/src/main/resources/log4j-sandag04.properties new file mode 100644 index 0000000..fea28c2 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag04.properties @@ -0,0 +1,46 @@ +#------------------------------------------------------------------------------# +# JPPF # +# Copyright (C) 2005-2010 JPPF Team. # +# http://www.jppf.org # +# # +# 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. # +#------------------------------------------------------------------------------# + +### direct log messages to stdout ### +#log4j.appender.stdout=org.apache.log4j.ConsoleAppender +#log4j.appender.stdout.Target=System.out +#log4j.appender.stdout.layout=org.apache.log4j.PatternLayout +#log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n + +### direct messages to file jppf-node.log ### +log4j.appender.JPPF=org.apache.log4j.FileAppender +log4j.appender.JPPF.File=logFiles/event-sandag04.log +log4j.appender.JPPF.Append=false +log4j.appender.JPPF.layout=org.apache.log4j.PatternLayout +#log4j.appender.JPPF.layout.ConversionPattern=%d{ABSOLUTE} [%-5p][%c.%M(%L)]: %m\n +log4j.appender.JPPF.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### direct messages to the JMX Logger ### +log4j.appender.JMX=org.jppf.logging.log4j.JmxAppender +log4j.appender.JMX.layout=org.apache.log4j.PatternLayout +log4j.appender.JMX.layout.ConversionPattern=%d [%-5p][%c.%M(%L)]: %m\n + +### set log levels - for more verbose logging change 'info' to 'debug' ### + +# will produce messages like "writing object size = " when sending to the server +log4j.logger.org.jppf.server.node.remote.RemoteNodeIO=TRACE +# will produce messages like "i = , read data size = " when receiving from the server +log4j.logger.org.jppf.server.node.remote.JPPFRemoteContainer=TRACE + +#log4j.rootLogger=TRACE, DEBUG, JPPF +log4j.rootLogger=INFO, JPPF, JMX diff --git a/sandag_abm/src/main/resources/log4j-sandag04.xml b/sandag_abm/src/main/resources/log4j-sandag04.xml new file mode 100644 index 0000000..e7937af --- /dev/null +++ b/sandag_abm/src/main/resources/log4j-sandag04.xml @@ -0,0 +1,439 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j.xml b/sandag_abm/src/main/resources/log4j.xml new file mode 100644 index 0000000..97a3996 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j.xml @@ -0,0 +1,443 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml b/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml new file mode 100644 index 0000000..ad2ebf9 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j_AtTransitCheck.xml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j_d2t.xml b/sandag_abm/src/main/resources/log4j_d2t.xml new file mode 100644 index 0000000..ec13a3d --- /dev/null +++ b/sandag_abm/src/main/resources/log4j_d2t.xml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j_hh.xml b/sandag_abm/src/main/resources/log4j_hh.xml new file mode 100644 index 0000000..834f12e --- /dev/null +++ b/sandag_abm/src/main/resources/log4j_hh.xml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j_mtx.xml b/sandag_abm/src/main/resources/log4j_mtx.xml new file mode 100644 index 0000000..d8c6e40 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j_mtx.xml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/log4j_test.xml b/sandag_abm/src/main/resources/log4j_test.xml new file mode 100644 index 0000000..7f52d65 --- /dev/null +++ b/sandag_abm/src/main/resources/log4j_test.xml @@ -0,0 +1,455 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/sandag_abm/src/main/resources/mapAndRun.bat b/sandag_abm/src/main/resources/mapAndRun.bat new file mode 100644 index 0000000..b1318cf --- /dev/null +++ b/sandag_abm/src/main/resources/mapAndRun.bat @@ -0,0 +1,33 @@ +:: script to map a drive and then call a batch file remotely using psexec +:: 1 is the drive letter to map (e.g. “M:”) +:: 2 is the share to map (e.g. “\\w-ampdx-d-sag01\mtc”) +:: 3 is the password +:: 4 is the user +:: 5 is the working directory for calling the batch file (starting from the mapped drive letter) +:: 6 is the name of the batch file to call +:: 7-10 are extra arguments (note DOS only does 9 arguments unless you use SHIFT) + +SET ONE=%1 +SET TWO=%2 +SET THREE=%3 +SET FOUR=%4 +SET FIVE=%5 +SET SIX=%6 +SET SEVEN=%7 +SET EIGHT=%8 +SET NINE=%9 +SHIFT +SHIFT +SHIFT +SHIFT +SHIFT +SHIFT +SHIFT +SHIFT +SHIFT +SET TEN=%1 + +net use %ONE% %TWO% /persistent:yes +%ONE% +cd %FIVE% +call %SIX% %SEVEN% %EIGHT% %NINE% %TEN% diff --git a/sandag_abm/src/main/resources/pskill.exe b/sandag_abm/src/main/resources/pskill.exe new file mode 100644 index 0000000000000000000000000000000000000000..2e9d0d49731b66e7885344e9cd2d398c274c56b6 GIT binary patch literal 621944 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